briskforwarding.com Open in urlscan Pro
103.147.226.26  Public Scan

Submitted URL: http://briskforwarding.com/js/filters/jquery.isotope.js
Effective URL: https://briskforwarding.com/js/filters/jquery.isotope.js
Submission: On November 03 via api from US — Scanned from DE

Form analysis 0 forms found in the DOM

Text Content

/**

 * Isotope v1.5.25

 * An exquisite jQuery plugin for magical layouts

 * http://isotope.metafizzy.co

 *

 * Commercial use requires one-time license fee

 * http://metafizzy.co/#licenses

 *

 * Copyright 2012 David DeSandro / Metafizzy

 */



/*jshint asi: true, browser: true, curly: true, eqeqeq: true, forin: false, immed: false, newcap: true, noempty: true, strict: true, undef: true */

/*global jQuery: false */



(function( window, $, undefined ){



  'use strict';



  // get global vars

  var document = window.document;

  var Modernizr = window.Modernizr;



  // 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 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;

    }

  };



  var testName;



  if ( Modernizr ) {

    // if there's a previous Modernzir, check if there are necessary tests

    for ( 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

    Modernizr = window.Modernizr = {

      _version : '1.6ish: miniModernizr for Isotope'

    };



    var classes = ' ';

    var result;



    // Run through tests

    for ( testName in tests) {

      result = tests[ testName ]();

      Modernizr[ testName ] = result;

      classes += ' ' + ( result ?  '' : 'no-' ) + testName;

    }



    // Add the new classes to the <html> element.

    $('html').addClass( classes );

  }





  // ========================= 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 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,

      dispatchMethod = $.event.handle ? 'handle' : 'dispatch',

      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() {

        $event[ dispatchMethod ].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 = [ 'width', 'height' ];



  var $window = $(window);



  $.Isotope.settings = {

    resizable: true,

    layoutMode : 'masonry',

    containerClass : 'isotope',

    itemClass : 'isotope-item',

    hiddenClass : 'isotope-hidden',

    hiddenStyle: { opacity: 0, scale: 0.001 },

    visibleStyle: { opacity: 1, scale: 1 },

    containerStyle: {

      position: 'relative',

      overflow: 'hidden'

    },

    animationEngine: '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 = {};

      // keep track of container styles

      var containerStyles = isoContainerStyles.slice(0);

      for ( var prop in this.options.containerStyle ) {

        containerStyles.push( prop );

      }

      for ( var i=0, len = containerStyles.length; i < len; i++ ) {

        prop = containerStyles[i];

        this.originalStyle[ prop ] = elemStyle[ prop ] || '';

      }

      // apply container style from options

      this.element.css( this.options.containerStyle );



      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

      this.offset = {

        left: parseInt( ( this.element.css('padding-left') || 0 ), 10 ),

        top: parseInt( ( this.element.css('padding-top') || 0 ), 10 )

      };



      // 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' };



      // filter out text nodes

      $atoms = $atoms.filter( function( i, atom ) {

        return atom.nodeType === 1;

      });



      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, '');

      var isUsingJQueryAnimation;

      // set applyStyleFnName

      switch ( animationEngine ) {

        case 'css' :

        case 'none' :

          isUsingJQueryAnimation = false;

          break;

        case 'jquery' :

          isUsingJQueryAnimation = true;

          break;

        default : // best available

          isUsingJQueryAnimation = !Modernizr.csstransitions;

      }

      this.isUsingJQueryAnimation = isUsingJQueryAnimation;

      this._updateUsingTransforms();

    },



    _updateTransformsEnabled : function() {

      this._updateUsingTransforms();

    },



    _updateUsingTransforms : function() {

      var usingTransforms = this.usingTransforms = this.options.transformsEnabled &&

        Modernizr.csstransforms && Modernizr.csstransitions && !this.isUsingJQueryAnimation;



      // prevent scales when transforms are disabled

      if ( !usingTransforms ) {

        delete this.options.hiddenStyle.scale;

        delete this.options.visibleStyle.scale;

      }



      this.getPositionStyles = 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 = Math.round( x + this.offset.left );

      y = Math.round( 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,

          onLayout = this.options.onLayout,

          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 || onLayout || animOpts.complete ) {

        // has callback

        var isCallbackTriggered = false,

            // array of possible callbacks to trigger

            callbacks = [ callback, onLayout, animOpts.complete ],

            instance = this;

        triggerCallbackNow = true;

        // trigger callback only once

        callbackFn = function() {

          if ( isCallbackTriggered ) {

            return;

          }

          var hollaback;

          for (var i=0, len = callbacks.length; i < len; i++) {

            hollaback = callbacks[i];

            if ( typeof hollaback === 'function' ) {

              hollaback.call( instance.element, $elems, instance );

            }

          }

          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,

              firstItem = this.styleQueue[0],

              testElem = firstItem && firstItem.$el,

              styleObj;

          // get first non-empty jQ object

          while ( !testElem || !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 );

        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, callback ) {

      // remove elements immediately from Isotope instance

      this.$allAtoms = this.$allAtoms.not( $content );

      this.$filteredAtoms = this.$filteredAtoms.not( $content );

      // remove() as a callback, for after transition / animation

      var instance = this;

      var removeContent = function() {

        $content.remove();

        if ( callback ) {

          callback.call( instance.element );

        }

      };



      if ( $content.filter( ':not(.' + this.options.hiddenClass + ')' ).length ) {

        // if any non-hidden content needs to be removed

        this.styleQueue.push({ $el: $content, style: this.options.hiddenStyle });

        this._sort();

        this.reLayout( removeContent );

      } else {

        // remove it now

        removeContent();

      }



    },



    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;

      var options = this.options;



      this.$allAtoms

        .removeClass( options.hiddenClass + ' ' + options.itemClass )

        .each(function(){

          var style = this.style;

          style.position = '';

          style.top = '';

          style.left = '';

          style.opacity = '';

          if ( usingTransforms ) {

            style[ transformProp ] = '';

          }

        });



      // re-apply saved container styles

      var elemStyle = this.element[0].style;

      for ( var prop in this.originalStyle ) {

        elemStyle[ prop ] = this.originalStyle[ prop ];

      }



      this.element

        .unbind('.isotope')

        .undelegate( '.' + options.hiddenClass, 'click' )

        .removeClass( 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 = ( col + 0.5 ) * props.columnWidth - $this.outerWidth(true) / 2,

            y = ( 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 = ( col + 0.5 ) * props.columnWidth - $this.outerWidth(true) / 2,

            y = ( 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.1.0

   * 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 = '',

        loaded = [];



    function triggerCallback() {

      callback.call( $this, $images );

    }



    function imgLoaded( event ) {

      var img = event.target;

      if ( img.src !== blank && $.inArray( img, loaded ) === -1 ){

        loaded.push( img );

        if ( --len <= 0 ){

          setTimeout( triggerCallback );

          $images.unbind( '.imagesLoaded', imgLoaded );

        }

      }

    }



    // if no images, trigger immediately

    if ( !len ) {

      triggerCallback();

    }



    $images.bind( 'load.imagesLoaded error.imagesLoaded',  imgLoaded ).each( function() {

      // cached images don't fire load sometimes, so we reset src.

      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 );