| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356 | /** * Isotope v1.5.03 * An exquisite jQuery plugin for magical layouts * http://isotope.metafizzy.co * * Commercial use requires one-time license fee * http://metafizzy.co/#licenses * * Copyright 2011 David DeSandro / Metafizzy *//*jshint curly: true, eqeqeq: true, forin: false, immed: false, newcap: true, noempty: true, undef: true *//*global Modernizr: true, jQuery: true */(function( window, $, undefined ){  'use strict';  // helper function  var capitalize = function( str ) {    return str.charAt(0).toUpperCase() + str.slice(1);  };  // ========================= getStyleProperty by kangax ===============================  // http://perfectionkills.com/feature-testing-css-properties/  var prefixes = 'Moz Webkit Khtml O Ms'.split(' ');  var getStyleProperty = function( propName ) {    var style = document.documentElement.style,        prefixed;    // test standard property first    if ( typeof style[propName] === 'string' ) {      return propName;    }    // capitalize    propName = capitalize( propName );    // test vendor specific properties    for ( var i=0, len = prefixes.length; i < len; i++ ) {      prefixed = prefixes[i] + propName;      if ( typeof style[ prefixed ] === 'string' ) {        return prefixed;      }    }  };  var transformProp = getStyleProperty('transform'),      transitionProp = getStyleProperty('transitionProperty');        // ========================= miniModernizr ===============================  // <3<3<3 and thanks to Faruk and Paul for doing the heavy lifting  /*!   * Modernizr v1.6ish: miniModernizr for Isotope   * http://www.modernizr.com   *   * Developed by:    * - Faruk Ates  http://farukat.es/   * - Paul Irish  http://paulirish.com/   *   * Copyright (c) 2009-2010   * Dual-licensed under the BSD or MIT licenses.   * http://www.modernizr.com/license/   */  /*   * This version whittles down the script just to check support for   * CSS transitions, transforms, and 3D transforms.  */    var tests = {    csstransforms: function() {      return !!transformProp;    },    csstransforms3d: function() {      var test = !!getStyleProperty('perspective');      // double check for Chrome's false positive      if ( test ) {        var vendorCSSPrefixes = ' -o- -moz- -ms- -webkit- -khtml- '.split(' '),            mediaQuery = '@media (' + vendorCSSPrefixes.join('transform-3d),(') + 'modernizr)',            $style = $('<style>' + mediaQuery + '{#modernizr{height:3px}}' + '</style>')                        .appendTo('head'),            $div = $('<div id="modernizr" />').appendTo('html');        test = $div.height() === 3;        $div.remove();        $style.remove();      }      return test;    },    csstransitions: function() {      return !!transitionProp;    }  };  if ( window.Modernizr ) {    // if there's a previous Modernzir, check if there are necessary tests    for ( var testName in tests) {      if ( !Modernizr.hasOwnProperty( testName ) ) {        // if test hasn't been run, use addTest to run it        Modernizr.addTest( testName, tests[ testName ] );      }    }  } else {    // or create new mini Modernizr that just has the 3 tests    window.Modernizr = (function(){      var miniModernizr = {            _version : '1.6ish: miniModernizr for Isotope'          },          classes = ' ',          result, testName;      // Run through tests      for ( testName in tests) {        result = tests[ testName ]();        miniModernizr[ testName ] = result;        classes += ' ' + ( result ?  '' : 'no-' ) + testName;      }      // Add the new classes to the <html> element.      $('html').addClass( classes );      return miniModernizr;    })();  }  // ========================= isoTransform ===============================  /**   *  provides hooks for .css({ scale: value, translate: [x, y] })   *  Progressively enhanced CSS transforms   *  Uses hardware accelerated 3D transforms for Safari   *  or falls back to 2D transforms.   */    if ( Modernizr.csstransforms ) {            // i.e. transformFnNotations.scale(0.5) >> 'scale3d( 0.5, 0.5, 1)'    var transformFnNotations = Modernizr.csstransforms3d ?       { // 3D transform functions        translate : function ( position ) {          return 'translate3d(' + position[0] + 'px, ' + position[1] + 'px, 0) ';        },        scale : function ( scale ) {          return 'scale3d(' + scale + ', ' + scale + ', 1) ';        }      } :      { // 2D transform functions        translate : function ( position ) {          return 'translate(' + position[0] + 'px, ' + position[1] + 'px) ';        },        scale : function ( scale ) {          return 'scale(' + scale + ') ';        }      }    ;    var setIsoTransform = function ( elem, name, value ) {          // unpack current transform data      var data =  $.data( elem, 'isoTransform' ) || {},          newData = {},          fnName,          transformObj = {},          transformValue;      // i.e. newData.scale = 0.5      newData[ name ] = value;      // extend new value over current data      $.extend( data, newData );      for ( fnName in data ) {        transformValue = data[ fnName ];        transformObj[ fnName ] = transformFnNotations[ fnName ]( transformValue );      }      // get proper order      // ideally, we could loop through this give an array, but since we only have      // a couple transforms we're keeping track of, we'll do it like so      var translateFn = transformObj.translate || '',          scaleFn = transformObj.scale || '',          // sorting so translate always comes first          valueFns = translateFn + scaleFn;      // set data back in elem      $.data( elem, 'isoTransform', data );      // set name to vendor specific property      elem.style[ transformProp ] = valueFns;    };       // ==================== scale ===================      $.cssNumber.scale = true;      $.cssHooks.scale = {      set: function( elem, value ) {        // uncomment this bit if you want to properly parse strings        // if ( typeof value === 'string' ) {        //   value = parseFloat( value );        // }        setIsoTransform( elem, 'scale', value );      },      get: function( elem, computed ) {        var transform = $.data( elem, 'isoTransform' );        return transform && transform.scale ? transform.scale : 1;      }    };    $.fx.step.scale = function( fx ) {      $.cssHooks.scale.set( fx.elem, fx.now+fx.unit );    };        // ==================== translate ===================        $.cssNumber.translate = true;      $.cssHooks.translate = {      set: function( elem, value ) {        // uncomment this bit if you want to properly parse strings        // if ( typeof value === 'string' ) {        //   value = value.split(' ');        // }        //         // var i, val;        // for ( i = 0; i < 2; i++ ) {        //   val = value[i];        //   if ( typeof val === 'string' ) {        //     val = parseInt( val );        //   }        // }        setIsoTransform( elem, 'translate', value );      },          get: function( elem, computed ) {        var transform = $.data( elem, 'isoTransform' );        return transform && transform.translate ? transform.translate : [ 0, 0 ];      }    };  }    // ========================= get transition-end event ===============================  var transitionEndEvent, transitionDurProp;    if ( Modernizr.csstransitions ) {    transitionEndEvent = {      WebkitTransitionProperty: 'webkitTransitionEnd',  // webkit      MozTransitionProperty: 'transitionend',      OTransitionProperty: 'oTransitionEnd',      transitionProperty: 'transitionEnd'    }[ transitionProp ];    transitionDurProp = getStyleProperty('transitionDuration');  }  // ========================= smartresize ===============================  /*   * smartresize: debounced resize event for jQuery   *   * latest version and complete README available on Github:   * https://github.com/louisremi/jquery.smartresize.js   *   * Copyright 2011 @louis_remi   * Licensed under the MIT license.   */  var $event = $.event,      resizeTimeout;  $event.special.smartresize = {    setup: function() {      $(this).bind( "resize", $event.special.smartresize.handler );    },    teardown: function() {      $(this).unbind( "resize", $event.special.smartresize.handler );    },    handler: function( event, execAsap ) {      // Save the context      var context = this,          args = arguments;      // set correct event type      event.type = "smartresize";      if ( resizeTimeout ) { clearTimeout( resizeTimeout ); }      resizeTimeout = setTimeout(function() {        jQuery.event.handle.apply( context, args );      }, execAsap === "execAsap"? 0 : 100 );    }  };  $.fn.smartresize = function( fn ) {    return fn ? this.bind( "smartresize", fn ) : this.trigger( "smartresize", ["execAsap"] );  };// ========================= Isotope ===============================  // our "Widget" object constructor  $.Isotope = function( options, element, callback ){    this.element = $( element );    this._create( options );    this._init( callback );  };    // styles of container element we want to keep track of  var isoContainerStyles = [ 'overflow', 'position', 'width', 'height' ];    $.Isotope.settings = {    resizable: true,    layoutMode : 'masonry',    containerClass : 'isotope',    itemClass : 'isotope-item',    hiddenClass : 'isotope-hidden',    hiddenStyle : Modernizr.csstransforms && !$.browser.opera ?       { opacity : 0, scale : 0.001 } : // browsers support CSS transforms, not Opera      { opacity : 0 }, // other browsers, including Opera    visibleStyle : Modernizr.csstransforms && !$.browser.opera ?       { opacity : 1, scale : 1 } : // browsers support CSS transforms, not Opera      { opacity : 1 },  // other browsers, including Opera    animationEngine : $.browser.opera ? 'jquery' : 'best-available',    animationOptions: {      queue: false,      duration: 800    },    sortBy : 'original-order',    sortAscending : true,    resizesContainer : true,    transformsEnabled : true,    itemPositionDataEnabled: false  };  $.Isotope.prototype = {    // sets up widget    _create : function( options ) {            this.options = $.extend( {}, $.Isotope.settings, options );            this.styleQueue = [];      this.elemCount = 0;      // get original styles in case we re-apply them in .destroy()      var elemStyle = this.element[0].style;      this.originalStyle = {};      for ( var i=0, len = isoContainerStyles.length; i < len; i++ ) {        var prop = isoContainerStyles[i];        this.originalStyle[ prop ] = elemStyle[ prop ] || '';      }            this.element.css({        overflow : 'hidden',        position : 'relative'      });            this._updateAnimationEngine();      this._updateUsingTransforms();            // sorting      var originalOrderSorter = {        'original-order' : function( $elem, instance ) {          instance.elemCount ++;          return instance.elemCount;        },        random : function() {          return Math.random();        }      };      this.options.getSortData = $.extend( this.options.getSortData, originalOrderSorter );      // need to get atoms      this.reloadItems();            // get top left position of where the bricks should be      var $cursor = $( document.createElement('div') ).prependTo( this.element );      this.offset = $cursor.position();      $cursor.remove();      // add isotope class first time around      var instance = this;      setTimeout( function() {        instance.element.addClass( instance.options.containerClass );      }, 0 );            // bind resize method      if ( this.options.resizable ) {        $(window).bind( 'smartresize.isotope', function() {           instance.resize();        });      }            // dismiss all click events from hidden events      this.element.delegate( '.' + this.options.hiddenClass, 'click', function(){        return false;      });          },        _getAtoms : function( $elems ) {      var selector = this.options.itemSelector,          // filter & find           $atoms = selector ? $elems.filter( selector ).add( $elems.find( selector ) ) : $elems,          // base style for atoms          atomStyle = { position: 'absolute' };                if ( this.usingTransforms ) {        atomStyle.left = 0;        atomStyle.top = 0;      }      $atoms.css( atomStyle ).addClass( this.options.itemClass );      this.updateSortData( $atoms, true );            return $atoms;    },      // _init fires when your instance is first created    // (from the constructor above), and when you    // attempt to initialize the widget again (by the bridge)    // after it has already been initialized.    _init : function( callback ) {            this.$filteredAtoms = this._filter( this.$allAtoms );      this._sort();      this.reLayout( callback );    },    option : function( opts ){      // change options AFTER initialization:      // signature: $('#foo').bar({ cool:false });      if ( $.isPlainObject( opts ) ){        this.options = $.extend( true, this.options, opts );        // trigger _updateOptionName if it exists        var updateOptionFn;        for ( var optionName in opts ) {          updateOptionFn = '_update' + capitalize( optionName );          if ( this[ updateOptionFn ] ) {            this[ updateOptionFn ]();          }        }      }    },        // ====================== updaters ====================== //    // kind of like setters        _updateAnimationEngine : function() {      var animationEngine = this.options.animationEngine.toLowerCase().replace( /[ _\-]/g, '');      // set applyStyleFnName      switch ( animationEngine ) {        case 'css' :        case 'none' :          this.isUsingJQueryAnimation = false;          break;        case 'jquery' :          this.isUsingJQueryAnimation = true;          break;        default : // best available          this.isUsingJQueryAnimation = !Modernizr.csstransitions;      }            this._updateUsingTransforms();    },        _updateTransformsEnabled : function() {      this._updateUsingTransforms();    },        _updateUsingTransforms : function() {      this.usingTransforms = this.options.transformsEnabled && Modernizr.csstransforms && Modernizr.csstransitions && !this.isUsingJQueryAnimation;      this.getPositionStyles = this.usingTransforms ? this._translate : this._positionAbs;    },        // ====================== Filtering ======================    _filter : function( $atoms ) {      var filter = this.options.filter === '' ? '*' : this.options.filter;      if ( !filter ) {        return $atoms;      }      var hiddenClass    = this.options.hiddenClass,          hiddenSelector = '.' + hiddenClass,          $hiddenAtoms   = $atoms.filter( hiddenSelector ),          $atomsToShow   = $hiddenAtoms;      if ( filter !== '*' ) {        $atomsToShow = $hiddenAtoms.filter( filter );        var $atomsToHide = $atoms.not( hiddenSelector ).not( filter ).addClass( hiddenClass );        this.styleQueue.push({ $el: $atomsToHide, style: this.options.hiddenStyle });      }      this.styleQueue.push({ $el: $atomsToShow, style: this.options.visibleStyle });      $atomsToShow.removeClass( hiddenClass );      return $atoms.filter( filter );    },        // ====================== Sorting ======================        updateSortData : function( $atoms, isIncrementingElemCount ) {      var instance = this,          getSortData = this.options.getSortData,          $this, sortData;      $atoms.each(function(){        $this = $(this);        sortData = {};        // get value for sort data based on fn( $elem ) passed in        for ( var key in getSortData ) {          if ( !isIncrementingElemCount && key === 'original-order' ) {            // keep original order original            sortData[ key ] = $.data( this, 'isotope-sort-data' )[ key ];          } else {            sortData[ key ] = getSortData[ key ]( $this, instance );          }        }        // apply sort data to element        $.data( this, 'isotope-sort-data', sortData );      });    },        // used on all the filtered atoms    _sort : function() {            var sortBy = this.options.sortBy,          getSorter = this._getSorter,          sortDir = this.options.sortAscending ? 1 : -1,          sortFn = function( alpha, beta ) {            var a = getSorter( alpha, sortBy ),                b = getSorter( beta, sortBy );            // fall back to original order if data matches            if ( a === b && sortBy !== 'original-order') {              a = getSorter( alpha, 'original-order' );              b = getSorter( beta, 'original-order' );            }            return ( ( a > b ) ? 1 : ( a < b ) ? -1 : 0 ) * sortDir;          };            this.$filteredAtoms.sort( sortFn );    },    _getSorter : function( elem, sortBy ) {      return $.data( elem, 'isotope-sort-data' )[ sortBy ];    },    // ====================== Layout Helpers ======================    _translate : function( x, y ) {      return { translate : [ x, y ] };    },        _positionAbs : function( x, y ) {      return { left: x, top: y };    },    _pushPosition : function( $elem, x, y ) {      x += this.offset.left;      y += this.offset.top;      var position = this.getPositionStyles( x, y );      this.styleQueue.push({ $el: $elem, style: position });      if ( this.options.itemPositionDataEnabled ) {        $elem.data('isotope-item-position', {x: x, y: y} );      }    },    // ====================== General Layout ======================    // used on collection of atoms (should be filtered, and sorted before )    // accepts atoms-to-be-laid-out to start with    layout : function( $elems, callback ) {      var layoutMode = this.options.layoutMode;      // layout logic      this[ '_' +  layoutMode + 'Layout' ]( $elems );            // set the size of the container      if ( this.options.resizesContainer ) {        var containerStyle = this[ '_' +  layoutMode + 'GetContainerSize' ]();        this.styleQueue.push({ $el: this.element, style: containerStyle });      }      this._processStyleQueue( $elems, callback );            this.isLaidOut = true;    },        _processStyleQueue : function( $elems, callback ) {      // are we animating the layout arrangement?      // use plugin-ish syntax for css or animate      var styleFn = !this.isLaidOut ? 'css' : (            this.isUsingJQueryAnimation ? 'animate' : 'css'          ),          animOpts = this.options.animationOptions,          objStyleFn, processor,          triggerCallbackNow, callbackFn;      // default styleQueue processor, may be overwritten down below      processor = function( i, obj ) {        obj.$el[ styleFn ]( obj.style, animOpts );      };      if ( this._isInserting && this.isUsingJQueryAnimation ) {        // if using styleQueue to insert items        processor = function( i, obj ) {          // only animate if it not being inserted          objStyleFn = obj.$el.hasClass('no-transition') ? 'css' : styleFn;          obj.$el[ objStyleFn ]( obj.style, animOpts );        };              } else if ( callback ) {        // has callback        var isCallbackTriggered = false,            instance = this;        triggerCallbackNow = true;        // trigger callback only once        callbackFn = function() {          if ( isCallbackTriggered ) {            return;          }          callback.call( instance.element, $elems );          isCallbackTriggered = true;        };                if ( this.isUsingJQueryAnimation && styleFn === 'animate' ) {          // add callback to animation options          animOpts.complete = callbackFn;          triggerCallbackNow = false;        } else if ( Modernizr.csstransitions ) {          // detect if first item has transition          var i = 0,              testElem = this.styleQueue[0].$el,              styleObj;          // get first non-empty jQ object          while ( !testElem.length ) {            styleObj = this.styleQueue[ i++ ];            // HACK: sometimes styleQueue[i] is undefined            if ( !styleObj ) {              return;            }            testElem = styleObj.$el;          }          // get transition duration of the first element in that object          // yeah, this is inexact          var duration = parseFloat( getComputedStyle( testElem[0] )[ transitionDurProp ] );          if ( duration > 0 ) {            processor = function( i, obj ) {              obj.$el[ styleFn ]( obj.style, animOpts )                // trigger callback at transition end                .one( transitionEndEvent, callbackFn );            };            triggerCallbackNow = false;          }        }      }      // process styleQueue      $.each( this.styleQueue, processor );            if ( triggerCallbackNow ) {        callbackFn();      }      // clear out queue for next time      this.styleQueue = [];    },            resize : function() {      if ( this[ '_' + this.options.layoutMode + 'ResizeChanged' ]() ) {        this.reLayout();      }    },            reLayout : function( callback ) {            this[ '_' +  this.options.layoutMode + 'Reset' ]();      this.layout( this.$filteredAtoms, callback );          },        // ====================== Convenience methods ======================        // ====================== Adding items ======================        // adds a jQuery object of items to a isotope container    addItems : function( $content, callback ) {      var $newAtoms = this._getAtoms( $content );      // add new atoms to atoms pools      this.$allAtoms = this.$allAtoms.add( $newAtoms );      if ( callback ) {        callback( $newAtoms );      }    },        // convienence method for adding elements properly to any layout    // positions items, hides them, then animates them back in <--- very sezzy    insert : function( $content, callback ) {      // position items      this.element.append( $content );            var instance = this;      this.addItems( $content, function( $newAtoms ) {        var $newFilteredAtoms = instance._filter( $newAtoms, true );        instance._addHideAppended( $newFilteredAtoms );        instance._sort();        instance.reLayout();        instance._revealAppended( $newFilteredAtoms, callback );      });          },        // convienence method for working with Infinite Scroll    appended : function( $content, callback ) {      var instance = this;      this.addItems( $content, function( $newAtoms ) {        instance._addHideAppended( $newAtoms );        instance.layout( $newAtoms );        instance._revealAppended( $newAtoms, callback );      });    },        // adds new atoms, then hides them before positioning    _addHideAppended : function( $newAtoms ) {      this.$filteredAtoms = this.$filteredAtoms.add( $newAtoms );      $newAtoms.addClass('no-transition');            this._isInserting = true;            // apply hidden styles      this.styleQueue.push({ $el: $newAtoms, style: this.options.hiddenStyle });    },        // sets visible style on new atoms    _revealAppended : function( $newAtoms, callback ) {      var instance = this;      // apply visible style after a sec      setTimeout( function() {        // enable animation        $newAtoms.removeClass('no-transition');        // reveal newly inserted filtered elements        instance.styleQueue.push({ $el: $newAtoms, style: instance.options.visibleStyle });        instance._isInserting = false;        instance._processStyleQueue( $newAtoms, callback );      }, 10 );    },        // gathers all atoms    reloadItems : function() {      this.$allAtoms = this._getAtoms( this.element.children() );    },        // removes elements from Isotope widget    remove : function( $content ) {      this.$allAtoms = this.$allAtoms.not( $content );      this.$filteredAtoms = this.$filteredAtoms.not( $content );      $content.remove();          },        shuffle : function( callback ) {      this.updateSortData( this.$allAtoms );      this.options.sortBy = 'random';      this._sort();      this.reLayout( callback );    },        // destroys widget, returns elements and container back (close) to original style    destroy : function() {      var usingTransforms = this.usingTransforms;      this.$allAtoms        .removeClass( this.options.hiddenClass + ' ' + this.options.itemClass )        .each(function(){          this.style.position = '';          this.style.top = '';          this.style.left = '';          this.style.opacity = '';          if ( usingTransforms ) {            this.style[ transformProp ] = '';          }        });            // re-apply saved container styles      var elemStyle = this.element[0].style;      for ( var i=0, len = isoContainerStyles.length; i < len; i++ ) {        var prop = isoContainerStyles[i];        elemStyle[ prop ] = this.originalStyle[ prop ];      }            this.element        .unbind('.isotope')        .undelegate( '.' + this.options.hiddenClass, 'click' )        .removeClass( this.options.containerClass )        .removeData('isotope');            $(window).unbind('.isotope');    },            // ====================== LAYOUTS ======================        // calculates number of rows or columns    // requires columnWidth or rowHeight to be set on namespaced object    // i.e. this.masonry.columnWidth = 200    _getSegments : function( isRows ) {      var namespace = this.options.layoutMode,          measure  = isRows ? 'rowHeight' : 'columnWidth',          size     = isRows ? 'height' : 'width',          segmentsName = isRows ? 'rows' : 'cols',          containerSize = this.element[ size ](),          segments,                    // i.e. options.masonry && options.masonry.columnWidth          segmentSize = this.options[ namespace ] && this.options[ namespace ][ measure ] ||                    // or use the size of the first item, i.e. outerWidth                    this.$filteredAtoms[ 'outer' + capitalize(size) ](true) ||                    // if there's no items, use size of container                    containerSize;            segments = Math.floor( containerSize / segmentSize );      segments = Math.max( segments, 1 );      // i.e. this.masonry.cols = ....      this[ namespace ][ segmentsName ] = segments;      // i.e. this.masonry.columnWidth = ...      this[ namespace ][ measure ] = segmentSize;          },        _checkIfSegmentsChanged : function( isRows ) {      var namespace = this.options.layoutMode,          segmentsName = isRows ? 'rows' : 'cols',          prevSegments = this[ namespace ][ segmentsName ];      // update cols/rows      this._getSegments( isRows );      // return if updated cols/rows is not equal to previous      return ( this[ namespace ][ segmentsName ] !== prevSegments );    },    // ====================== Masonry ======================      _masonryReset : function() {      // layout-specific props      this.masonry = {};      // FIXME shouldn't have to call this again      this._getSegments();      var i = this.masonry.cols;      this.masonry.colYs = [];      while (i--) {        this.masonry.colYs.push( 0 );      }    },      _masonryLayout : function( $elems ) {      var instance = this,          props = instance.masonry;      $elems.each(function(){        var $this  = $(this),            //how many columns does this brick span            colSpan = Math.ceil( $this.outerWidth(true) / props.columnWidth );        colSpan = Math.min( colSpan, props.cols );        if ( colSpan === 1 ) {          // if brick spans only one column, just like singleMode          instance._masonryPlaceBrick( $this, props.colYs );        } else {          // brick spans more than one column          // how many different places could this brick fit horizontally          var groupCount = props.cols + 1 - colSpan,              groupY = [],              groupColY,              i;          // for each group potential horizontal position          for ( i=0; i < groupCount; i++ ) {            // make an array of colY values for that one group            groupColY = props.colYs.slice( i, i+colSpan );            // and get the max value of the array            groupY[i] = Math.max.apply( Math, groupColY );          }                  instance._masonryPlaceBrick( $this, groupY );        }      });    },        // worker method that places brick in the columnSet    //   with the the minY    _masonryPlaceBrick : function( $brick, setY ) {      // get the minimum Y value from the columns      var minimumY = Math.min.apply( Math, setY ),          shortCol = 0;      // Find index of short column, the first from the left      for (var i=0, len = setY.length; i < len; i++) {        if ( setY[i] === minimumY ) {          shortCol = i;          break;        }      }          // position the brick      var x = this.masonry.columnWidth * shortCol,          y = minimumY;      this._pushPosition( $brick, x, y );      // apply setHeight to necessary columns      var setHeight = minimumY + $brick.outerHeight(true),          setSpan = this.masonry.cols + 1 - len;      for ( i=0; i < setSpan; i++ ) {        this.masonry.colYs[ shortCol + i ] = setHeight;      }    },        _masonryGetContainerSize : function() {      var containerHeight = Math.max.apply( Math, this.masonry.colYs );      return { height: containerHeight };    },      _masonryResizeChanged : function() {      return this._checkIfSegmentsChanged();    },      // ====================== fitRows ======================        _fitRowsReset : function() {      this.fitRows = {        x : 0,        y : 0,        height : 0      };    },      _fitRowsLayout : function( $elems ) {      var instance = this,          containerWidth = this.element.width(),          props = this.fitRows;            $elems.each( function() {        var $this = $(this),            atomW = $this.outerWidth(true),            atomH = $this.outerHeight(true);              if ( props.x !== 0 && atomW + props.x > containerWidth ) {          // if this element cannot fit in the current row          props.x = 0;          props.y = props.height;        }               // position the atom        instance._pushPosition( $this, props.x, props.y );          props.height = Math.max( props.y + atomH, props.height );        props.x += atomW;        });    },      _fitRowsGetContainerSize : function () {      return { height : this.fitRows.height };    },      _fitRowsResizeChanged : function() {      return true;    },      // ====================== cellsByRow ======================      _cellsByRowReset : function() {      this.cellsByRow = {        index : 0      };      // get this.cellsByRow.columnWidth      this._getSegments();      // get this.cellsByRow.rowHeight      this._getSegments(true);    },    _cellsByRowLayout : function( $elems ) {      var instance = this,          props = this.cellsByRow;      $elems.each( function(){        var $this = $(this),            col = props.index % props.cols,            row = Math.floor( props.index / props.cols ),            x = Math.round( ( col + 0.5 ) * props.columnWidth - $this.outerWidth(true) / 2 ),            y = Math.round( ( row + 0.5 ) * props.rowHeight - $this.outerHeight(true) / 2 );        instance._pushPosition( $this, x, y );        props.index ++;      });    },    _cellsByRowGetContainerSize : function() {      return { height : Math.ceil( this.$filteredAtoms.length / this.cellsByRow.cols ) * this.cellsByRow.rowHeight + this.offset.top };    },    _cellsByRowResizeChanged : function() {      return this._checkIfSegmentsChanged();    },        // ====================== straightDown ======================      _straightDownReset : function() {      this.straightDown = {        y : 0      };    },    _straightDownLayout : function( $elems ) {      var instance = this;      $elems.each( function( i ){        var $this = $(this);        instance._pushPosition( $this, 0, instance.straightDown.y );        instance.straightDown.y += $this.outerHeight(true);      });    },    _straightDownGetContainerSize : function() {      return { height : this.straightDown.y };    },    _straightDownResizeChanged : function() {      return true;    },    // ====================== masonryHorizontal ======================        _masonryHorizontalReset : function() {      // layout-specific props      this.masonryHorizontal = {};      // FIXME shouldn't have to call this again      this._getSegments( true );      var i = this.masonryHorizontal.rows;      this.masonryHorizontal.rowXs = [];      while (i--) {        this.masonryHorizontal.rowXs.push( 0 );      }    },      _masonryHorizontalLayout : function( $elems ) {      var instance = this,          props = instance.masonryHorizontal;      $elems.each(function(){        var $this  = $(this),            //how many rows does this brick span            rowSpan = Math.ceil( $this.outerHeight(true) / props.rowHeight );        rowSpan = Math.min( rowSpan, props.rows );        if ( rowSpan === 1 ) {          // if brick spans only one column, just like singleMode          instance._masonryHorizontalPlaceBrick( $this, props.rowXs );        } else {          // brick spans more than one row          // how many different places could this brick fit horizontally          var groupCount = props.rows + 1 - rowSpan,              groupX = [],              groupRowX, i;          // for each group potential horizontal position          for ( i=0; i < groupCount; i++ ) {            // make an array of colY values for that one group            groupRowX = props.rowXs.slice( i, i+rowSpan );            // and get the max value of the array            groupX[i] = Math.max.apply( Math, groupRowX );          }          instance._masonryHorizontalPlaceBrick( $this, groupX );        }      });    },        _masonryHorizontalPlaceBrick : function( $brick, setX ) {      // get the minimum Y value from the columns      var minimumX  = Math.min.apply( Math, setX ),          smallRow  = 0;      // Find index of smallest row, the first from the top      for (var i=0, len = setX.length; i < len; i++) {        if ( setX[i] === minimumX ) {          smallRow = i;          break;        }      }      // position the brick      var x = minimumX,          y = this.masonryHorizontal.rowHeight * smallRow;      this._pushPosition( $brick, x, y );      // apply setHeight to necessary columns      var setWidth = minimumX + $brick.outerWidth(true),          setSpan = this.masonryHorizontal.rows + 1 - len;      for ( i=0; i < setSpan; i++ ) {        this.masonryHorizontal.rowXs[ smallRow + i ] = setWidth;      }    },    _masonryHorizontalGetContainerSize : function() {      var containerWidth = Math.max.apply( Math, this.masonryHorizontal.rowXs );      return { width: containerWidth };    },        _masonryHorizontalResizeChanged : function() {      return this._checkIfSegmentsChanged(true);    },    // ====================== fitColumns ======================      _fitColumnsReset : function() {      this.fitColumns = {        x : 0,        y : 0,        width : 0      };    },        _fitColumnsLayout : function( $elems ) {      var instance = this,          containerHeight = this.element.height(),          props = this.fitColumns;      $elems.each( function() {        var $this = $(this),            atomW = $this.outerWidth(true),            atomH = $this.outerHeight(true);        if ( props.y !== 0 && atomH + props.y > containerHeight ) {          // if this element cannot fit in the current column          props.x = props.width;          props.y = 0;        }         // position the atom        instance._pushPosition( $this, props.x, props.y );        props.width = Math.max( props.x + atomW, props.width );        props.y += atomH;      });    },        _fitColumnsGetContainerSize : function () {      return { width : this.fitColumns.width };    },        _fitColumnsResizeChanged : function() {      return true;    },          // ====================== cellsByColumn ======================      _cellsByColumnReset : function() {      this.cellsByColumn = {        index : 0      };      // get this.cellsByColumn.columnWidth      this._getSegments();      // get this.cellsByColumn.rowHeight      this._getSegments(true);    },    _cellsByColumnLayout : function( $elems ) {      var instance = this,          props = this.cellsByColumn;      $elems.each( function(){        var $this = $(this),            col = Math.floor( props.index / props.rows ),            row = props.index % props.rows,            x = Math.round( ( col + 0.5 ) * props.columnWidth - $this.outerWidth(true) / 2 ),            y = Math.round( ( row + 0.5 ) * props.rowHeight - $this.outerHeight(true) / 2 );        instance._pushPosition( $this, x, y );        props.index ++;      });    },    _cellsByColumnGetContainerSize : function() {      return { width : Math.ceil( this.$filteredAtoms.length / this.cellsByColumn.rows ) * this.cellsByColumn.columnWidth };    },    _cellsByColumnResizeChanged : function() {      return this._checkIfSegmentsChanged(true);    },        // ====================== straightAcross ======================    _straightAcrossReset : function() {      this.straightAcross = {        x : 0      };    },    _straightAcrossLayout : function( $elems ) {      var instance = this;      $elems.each( function( i ){        var $this = $(this);        instance._pushPosition( $this, instance.straightAcross.x, 0 );        instance.straightAcross.x += $this.outerWidth(true);      });    },    _straightAcrossGetContainerSize : function() {      return { width : this.straightAcross.x };    },    _straightAcrossResizeChanged : function() {      return true;    }  };      // ======================= imagesLoaded Plugin ===============================  /*!   * jQuery imagesLoaded plugin v1.0.3   * http://github.com/desandro/imagesloaded   *   * MIT License. by Paul Irish et al.   */  // $('#my-container').imagesLoaded(myFunction)  // or  // $('img').imagesLoaded(myFunction)  // execute a callback when all images have loaded.  // needed because .load() doesn't work on cached images  // callback function gets image collection as argument  //  `this` is the container  $.fn.imagesLoaded = function( callback ) {    var $this = this,        $images = $this.find('img').add( $this.filter('img') ),        len = $images.length,        blank = '';    function triggerCallback() {      callback.call( $this, $images );    }    function imgLoaded( event ) {      if ( --len <= 0 && event.target.src !== blank ){        setTimeout( triggerCallback );        $images.unbind( 'load error', imgLoaded );      }    }    if ( !len ) {      triggerCallback();    }    $images.bind( 'load error',  imgLoaded ).each( function() {      // cached images don't fire load sometimes, so we reset src.      if (this.complete || this.complete === undefined){        var src = this.src;        // webkit hack from http://groups.google.com/group/jquery-dev/browse_thread/thread/eee6ab7b2da50e1f        // data uri bypasses webkit log warning (thx doug jones)        this.src = blank;        this.src = src;      }    });    return $this;  };  // helper function for logging errors  // $.error breaks jQuery chaining  var logError = function( message ) {    if ( window.console ) {      window.console.error( message );    }  };  // =======================  Plugin bridge  ===============================  // leverages data method to either create or return $.Isotope constructor  // A bit from jQuery UI  //   https://github.com/jquery/jquery-ui/blob/master/ui/jquery.ui.widget.js  // A bit from jcarousel   //   https://github.com/jsor/jcarousel/blob/master/lib/jquery.jcarousel.js  $.fn.isotope = function( options, callback ) {    if ( typeof options === 'string' ) {      // call method      var args = Array.prototype.slice.call( arguments, 1 );      this.each(function(){        var instance = $.data( this, 'isotope' );        if ( !instance ) {          logError( "cannot call methods on isotope prior to initialization; " +              "attempted to call method '" + options + "'" );          return;        }        if ( !$.isFunction( instance[options] ) || options.charAt(0) === "_" ) {          logError( "no such method '" + options + "' for isotope instance" );          return;        }        // apply method        instance[ options ].apply( instance, args );      });    } else {      this.each(function() {        var instance = $.data( this, 'isotope' );        if ( instance ) {          // apply options & init          instance.option( options );          instance._init( callback );        } else {          // initialize new instance          $.data( this, 'isotope', new $.Isotope( options, this, callback ) );        }      });    }    // return jQuery object    // so plugin methods do not have to    return this;  };})( window, jQuery );
 |