Commit 68b3ed21 by lmf

增加列表瀑布流加载数据

parent fc472e36
<?php
namespace Magetop\InfiniteScroll\Block;
/**
* Class InfiniteScroll
*
*/
class InfiniteScroll extends \Magento\Framework\View\Element\Template
{
/**
* InfiniteScroll constructor.
* @param \Magento\Framework\View\Element\Template\Context $context
* @param \Magento\Store\Model\StoreManagerInterface $storeManager
* @param array $data
*/
public function __construct(
\Magento\Framework\View\Element\Template\Context $context,
\Magento\Store\Model\StoreManagerInterface $storeManager,
array $data = []
) {
parent::__construct($context, $data);
$this->storeManager = $storeManager;
}
/**
* getLoadingImage
*
* @return mixed
*/
public function getMedia($img=null)
{
$urlMedia = $this->storeManager->getStore()->getBaseUrl(\Magento\Framework\UrlInterface::URL_TYPE_MEDIA);
if($img) return $urlMedia . $img;
return $urlMedia;
}
}
\ No newline at end of file
<?php
/**
* @Author Magetop Developers
* @package Magetop_InfiniteScroll
* @copyright Copyright (c) 2018 MAGETOP (https://www.magetop.com)
* @terms https://www.magetop.com/terms
* @license https://opensource.org/licenses/afl-3.0.php Academic Free License (AFL 3.0)
**/
namespace Magetop\InfiniteScroll\Block\System\Config\Form\Field;
use Magento\Framework\Data\Form\Element\AbstractElement;
/**
* Backend system config datetime field renderer
*/
class Info extends \Magento\Config\Block\System\Config\Form\Field
{
public function __construct(
\Magento\Backend\Block\Template\Context $context,
array $data = []
) {
parent::__construct($context, $data);
}
/**
* @param AbstractElement $element
* @return string
* @codeCoverageIgnore
*/
protected function _getElementHtml(AbstractElement $element)
{
$html = '<div style="background: #EF432E;padding: 10px;border-radius: 5px;text-align: center">
<a target="_blank" href="https://www.magetop.com/magento-extensions.html" style="color: #fff">Magetop - Marketplace Extensions</a>
</div>';
return $html;
}
}
\ No newline at end of file
<?php
namespace Magetop\InfiniteScroll\Helper;
class Data extends \Magento\Framework\App\Helper\AbstractHelper
{
/**
* @var array
*/
protected $configModule;
public function __construct(
\Magento\Framework\App\Helper\Context $context
)
{
parent::__construct($context);
$this->configModule = $this->getConfig(strtolower($this->_getModuleName()));
}
public function getConfig($cfg='')
{
if($cfg) return $this->scopeConfig->getValue( $cfg, \Magento\Store\Model\ScopeInterface::SCOPE_STORE );
return $this->scopeConfig;
}
public function getConfigModule($cfg='', $value=null)
{
$values = $this->configModule;
if( !$cfg ) return $values;
$config = explode('/', $cfg);
$end = count($config) - 1;
foreach ($config as $key => $vl) {
if( isset($values[$vl]) ){
if( $key == $end ) {
$value = $values[$vl];
}else {
$values = $values[$vl];
}
}
}
return $value;
}
}
\ No newline at end of file
{
"name": "magetop/infinitescroll",
"description": "Infinite scroll for magento 2 automatically loads product catalog without reloading the page.",
"type": "magento2-module",
"license": [
"OSL-3.0",
"AFL-3.0"
],
"authors": [
{
"name": "Magetop",
"email": "support@magetop.com",
"homepage": "https://www.magetop.com",
"role": "Technical Support"
}
],
"autoload": {
"files": [
"registration.php"
],
"psr-4": {
"Magetop\\InfiniteScroll\\": ""
}
}
}
\ No newline at end of file
<?xml version="1.0"?>
<config xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="urn:magento:framework:Acl/etc/acl.xsd">
<acl>
<resources>
<resource id="Magento_Backend::admin">
<resource id="Magento_Backend::stores">
<resource id="Magento_Backend::stores_settings">
<resource id="Magento_Config::config">
<resource id="Magetop_InfiniteScroll::config" title="Infinite Scroll" />
</resource>
</resource>
</resource>
</resource>
</resources>
</acl>
</config>
<?xml version="1.0"?>
<config xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="urn:magento:module:Magento_Config:etc/system_file.xsd">
<system>
<tab id="magetop" translate="label" sortOrder="1">
<label>Magetop.com</label>
</tab>
<section id="magetop_infinitescroll" translate="label" type="text" sortOrder="1" showInDefault="1" showInWebsite="1" showInStore="1">
<class>separator-top</class>
<label>Infinite Scroll Setting</label>
<tab>magetop</tab>
<resource>Magetop_InfiniteScroll::config</resource>
<group id="info" translate="label" type="text" sortOrder="1" showInDefault="1" showInWebsite="1" showInStore="1">
<frontend_model>Magetop\InfiniteScroll\Block\System\Config\Form\Field\Info</frontend_model>
</group>
<group id="general" translate="label" type="text" sortOrder="1" showInDefault="1" showInWebsite="1" showInStore="1" canRestore="1">
<label>General Settings</label>
<field id="enabled" translate="label comment" type="select" sortOrder="1" showInDefault="1" showInWebsite="1" showInStore="1" canRestore="1">
<label>Enabled</label>
<source_model>Magento\Config\Model\Config\Source\Yesno</source_model>
</field>
<field id="delay" translate="label comment" type="text" sortOrder="2" showInDefault="1" showInWebsite="1" showInStore="1" canRestore="1">
<label>Delay (ms)</label>
<comment><![CDATA[Delay value for scroll down. default: 600]]></comment>
</field>
<field id="content" translate="label comment" type="text" sortOrder="3" showInDefault="1" showInWebsite="1" showInStore="1" canRestore="1">
<label>Content</label>
<comment><![CDATA[Selector for the element that surrounds the items you will be loading more of. (For Ex.= .classname/#id)]]></comment>
</field>
<field id="pagination" translate="label comment" type="text" sortOrder="4" showInDefault="1" showInWebsite="1" showInStore="1" canRestore="1">
<label>Pagination</label>
<comment><![CDATA[Selector for pagination. (For Ex.= .classname/#id)]]></comment>
</field>
<field id="next" translate="label comment" type="text" sortOrder="5" showInDefault="1" showInWebsite="1" showInStore="1" canRestore="1">
<label>Next</label>
<comment><![CDATA[Selector for the link to the next page. (For Ex.= .classname/#id)]]></comment>
</field>
<field id="item" translate="label comment" type="text" sortOrder="6" showInDefault="1" showInWebsite="1" showInStore="1" canRestore="1">
<label>Item</label>
<comment><![CDATA[Selector for all items you will receive. (For Ex.= .classname/#id)]]></comment>
</field>
<field id="loading_image" translate="label comment" type="image" sortOrder="25" showInDefault="1" showInWebsite="1" showInStore="1" canRestore="1">
<label>Loading Image placeholder :</label>
<comment>Allowed file types: jpg, jpeg, gif, png.</comment>
<backend_model>Magento\Config\Model\Config\Backend\Image</backend_model>
<upload_dir config="system/filesystem/media" scope_info="1">magetop/infinitescroll</upload_dir>
<base_url type="media" scope_info="1">magetop/infinitescroll</base_url>
</field>
<field id="loading_text" translate="label comment" type="text" sortOrder="8" showInDefault="1" showInWebsite="1" showInStore="1" canRestore="1">
<label>Loading Text</label>
</field>
<field id="done_text" translate="label comment" type="text" sortOrder="9" showInDefault="1" showInWebsite="1" showInStore="1" canRestore="1">
<label>Done Text</label>
</field>
<field id="load_more" translate="label comment" type="text" sortOrder="70" showInDefault="1" showInWebsite="1" showInStore="1" canRestore="1">
<label>Load More threshold</label>
<comment><![CDATA[When this page number is reached, a button to load more products will be shown instead of continue loading products automatically with the scroll (this could be useful to help the user to reach the footer). TIP: use a high number to disable this feature.]]></comment>
</field>
<field id="load_more_text" translate="label comment" type="text" sortOrder="80" showInDefault="1" showInWebsite="1" showInStore="1" canRestore="1">
<label>Load More button text</label>
<comment><![CDATA[Set an offset before page end from which the content will start to load. A high number will load the next page a long time before the user reaches the bottom of the screen.]]></comment>
</field>
</group>
</section>
</system>
</config>
<?xml version="1.0"?>
<config xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="urn:magento:module:Magento_Store:etc/config.xsd">
<default>
<magetop_infinitescroll>
<general>
<enabled>1</enabled>
<delay>600</delay>
<content>.main</content>
<pagination>.pages-items, .pager__list</pagination>
<next>.next</next>
<item>.product-item</item>
<loading_image/>
<loading_text><![CDATA[<em>Loading - please wait...</em>]]></loading_text>
<done_text><![CDATA[<em>You've reached the end of the item.</em>]]></done_text>
<load_more>3</load_more>
<load_more_text>Load more items</load_more_text>
</general>
</magetop_infinitescroll>
</default>
</config>
<?xml version="1.0"?>
<config xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="urn:magento:framework:Module/etc/module.xsd">
<module name="Magetop_InfiniteScroll" setup_version="1.0.0">
<sequence>
<module name="Magento_Catalog"/>
<module name="Magento_CatalogSearch"/>
<module name="Magento_LayeredNavigation" />
</sequence>
</module>
</config>
\ No newline at end of file
<?xml version="1.0" encoding="UTF-8"?>
<config xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="urn:magento:framework:App/etc/routes.xsd">
<router id="admin">
<route id="ajaxscroll" frontName="ajaxscroll">
<module name="Magetop_InfiniteScroll" />
</route>
</router>
</config>
\ No newline at end of file
<?php
/**
* Copyright © Magento, Inc. All rights reserved.
* See COPYING.txt for license details.
*/
\Magento\Framework\Component\ComponentRegistrar::register(
\Magento\Framework\Component\ComponentRegistrar::MODULE,
'Magetop_InfiniteScroll',
__DIR__
);
<?xml version="1.0"?>
<page xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="urn:magento:framework:View/Layout/etc/page_configuration.xsd">
<body>
<referenceContainer name="before.body.end">
<block class="Magetop\InfiniteScroll\Block\InfiniteScroll" name="infinite_scroll.block" template="Magetop_InfiniteScroll::infinitescroll.phtml" ifconfig="magetop_infinitescroll/general/enabled"/>
</referenceContainer>
</body>
</page>
<?xml version="1.0"?>
<page xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="urn:magento:framework:View/Layout/etc/page_configuration.xsd">
<update handle="ajaxscroll"/>
</page>
\ No newline at end of file
<?xml version="1.0"?>
<page xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="urn:magento:framework:View/Layout/etc/page_configuration.xsd">
<update handle="ajaxscroll"/>
</page>
\ No newline at end of file
<?xml version="1.0"?>
<page xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" layout="2columns-left" xsi:noNamespaceSchemaLocation="urn:magento:framework:View/Layout/etc/page_configuration.xsd">
<update handle="ajaxscroll"/>
</page>
\ No newline at end of file
var config = {
map: {
'*': {
infinitescroll: 'Magetop_InfiniteScroll/js/plugin/infinitescroll',
}
},
paths: {
'magetop/infinitescroll': 'Magetop_InfiniteScroll/js/plugin/infinitescroll',
},
shim: {
'magetop/infinitescroll': {
deps: ['jquery']
}
}
};
\ No newline at end of file
<?php
$helper = $this->helper('Magetop\InfiniteScroll\Helper\Data');
$isEnabled = $helper->getConfigModule('general/enabled');
$delay = $helper->getConfigModule('general/delay');
$content = $helper->getConfigModule('general/content');
$pagination = $helper->getConfigModule('general/pagination');
$next = $helper->getConfigModule('general/next');
$item = $helper->getConfigModule('general/item');
$loadingText = $helper->getConfigModule('general/loading_text');
$doneText = $helper->getConfigModule('general/done_text');
$loadMore = $helper->getConfigModule('general/load_more');
$loadMoreText = $helper->getConfigModule('general/load_more_text');
$loadingImage = $helper->getConfigModule('general/loading_image');
$imgPath = 'magetop/infinitescroll/'.$loadingImage;
if ($loadingImage) {
$loadingImage = $block->getMedia($imgPath);
}else{
$loadingImage = $this->getViewFileUrl('images/loader-1.gif');
}
?>
<script type="text/javascript">
require(['jquery', 'magetop/infinitescroll'], function ($) {
"use strict";
$(document).ready(function () {
$('body').addClass('infinitescroll');
window.ias = $.ias({
container: "<?php echo $content;?>",
item: "<?php echo $item;?>",
pagination: "<?php echo $pagination;?>",
next: "<?php echo $next;?>",
delay: "<?php echo $delay;?>",
});
window.ias.extension(new IASSpinnerExtension({
src: "<?php echo $loadingImage; ?>",
html: '<div class="iass-spinner"><img src="{src}"/><span><?php echo $loadingText; ?></span></div>'
}));
window.ias.extension(new IASNoneLeftExtension({
text: "<?php echo $doneText;?>",
html: '<div class="ias-noneleft">{text}</div>'
}));
window.ias.extension(new IASTriggerExtension({
text:"<?php echo $loadMoreText ?>",
html:'<div class="ias-trigger ias-trigger-next"><button class="load-more"><?php echo $loadMoreText ?></button></div>',
textPrev:"<?php echo $loadMoreText ?>",
htmlPrev:'<div class="ias-trigger ias-trigger-prev"><button class="load-more"><?php echo $loadMoreText ?></button></div>',
offset:"<?php echo $loadMore ?>",
}));
window.ias.on('rendered', function(items){
$('body').trigger('contentUpdated');
$( "form[data-role='tocart-form']" ).catalogAddToCart();
});
});
});
</script>
\ No newline at end of file
& when (@media-common = true) {
.iass-spinner{
text-align: center;
font-size: 16px;
color: #333;
display: block;
}
.ias-noneleft{
text-align: center;
color: #333;
letter-spacing: 0px;
font-size: 14px;
font-weight: 400;
}
.iass-spinner img,
.ias-noneleft img{
display:inline-block;
margin-left:auto;
margin-right:auto;
}
.iass-spinner,
.ias-noneleft{
display: inline-block;
width: 100%;
}
.ias-trigger-next{
text-align: center;
cursor: pointer;
display: inline-block;
width: 100%;
}
.load-more{
font-size:15px;
border: none;
}
.load-more::hover{
background:#0491ff;
}
.ias-trigger-prev{
text-align: center;
cursor: pointer;
}
.iass-spinner img {
height: 40px;
width: 40px;
margin-bottom: 7px;
}
}
\ No newline at end of file
if(typeof(IASCallbacks) == "undefined"){
/**
* IASCallbacks
* http://infiniteajaxscroll.com
*
* This file is part of the Infinite AJAX Scroll package
*
*/
var IASCallbacks = function () {
this.list = [];
this.fireStack = [];
this.isFiring = false;
this.isDisabled = false;
/**
* Calls all added callbacks
*
* @private
* @param args
*/
this.fire = function (args) {
var context = args[0],
deferred = args[1],
callbackArguments = args[2];
this.isFiring = true;
for (var i = 0, l = this.list.length; i < l; i++) {
if (false === this.list[i].fn.apply(context, callbackArguments)) {
deferred.reject();
break;
}
}
this.isFiring = false;
deferred.resolve();
if (this.fireStack.length) {
this.fire(this.fireStack.shift());
}
};
/**
* Returns index of the callback in the list in a similar way as
* the indexOf function.
*
* @param callback
* @param {number} index index to start the search from
* @returns {number}
*/
this.inList = function (callback, index) {
index = index || 0;
for (var i = index, length = this.list.length; i < length; i++) {
if (this.list[i].fn === callback || (callback.guid && this.list[i].fn.guid && callback.guid === this.list[i].fn.guid)) {
return i;
}
}
return -1;
};
return this;
};
IASCallbacks.prototype = {
/**
* Adds a callback
*
* @param callback
* @returns {IASCallbacks}
* @param priority
*/
add: function (callback, priority) {
var callbackObject = {fn: callback, priority: priority};
priority = priority || 0;
for (var i = 0, length = this.list.length; i < length; i++) {
if (priority > this.list[i].priority) {
this.list.splice(i, 0, callbackObject);
return this;
}
}
this.list.push(callbackObject);
return this;
},
/**
* Removes a callback
*
* @param callback
* @returns {IASCallbacks}
*/
remove: function (callback) {
var index = 0;
while (( index = this.inList(callback, index) ) > -1) {
this.list.splice(index, 1);
}
return this;
},
/**
* Checks if callback is added
*
* @param callback
* @returns {*}
*/
has: function (callback) {
return (this.inList(callback) > -1);
},
/**
* Calls callbacks with a context
*
* @param context
* @param args
* @returns {object|void}
*/
fireWith: function (context, args) {
var deferred = jQuery.Deferred();
if (this.isDisabled) {
return deferred.reject();
}
args = args || [];
args = [ context, deferred, args.slice ? args.slice() : args ];
if (this.isFiring) {
this.fireStack.push(args);
} else {
this.fire(args);
}
return deferred;
},
/**
* Disable firing of new events
*/
disable: function () {
this.isDisabled = true;
},
/**
* Enable firing of new events
*/
enable: function () {
this.isDisabled = false;
}
};
}
/**
* Infinite Ajax Scroll v2.1.3
* A jQuery plugin for infinite scrolling
* http://infiniteajaxscroll.com
*
* Commercial use requires one-time purchase of a commercial license
* http://infiniteajaxscroll.com/docs/license.html
*
* Non-commercial use is licensed under the MIT License
*
* Copyright 2014 Webcreate (Jeroen Fiege)
*/
define([
'jquery'
], function () {
(function($) {
'use strict';
var UNDETERMINED_SCROLLOFFSET = -1;
var IAS = function($element, options) {
this.itemsContainerSelector = options.container;
this.itemSelector = options.item;
this.nextSelector = options.next;
this.paginationSelector = options.pagination;
this.$scrollContainer = $element;
this.$itemsContainer = $(this.itemsContainerSelector);
this.$container = (window === $element.get(0) ? $(document) : $element);
this.defaultDelay = options.delay;
this.negativeMargin = options.negativeMargin;
this.nextUrl = null;
this.isBound = false;
this.listeners = {
next: new IASCallbacks(),
load: new IASCallbacks(),
loaded: new IASCallbacks(),
render: new IASCallbacks(),
rendered: new IASCallbacks(),
scroll: new IASCallbacks(),
noneLeft: new IASCallbacks(),
ready: new IASCallbacks()
};
this.extensions = [];
/**
* Scroll event handler
*
* Note: calls to this functions should be throttled
*
* @private
*/
this.scrollHandler = function() {
var currentScrollOffset = this.getCurrentScrollOffset(this.$scrollContainer),
scrollThreshold = this.getScrollThreshold()
;
// the throttle method can call the scrollHandler even thought we have called unbind()
if (!this.isBound) {
return;
}
// invalid scrollThreshold. The DOM might not have loaded yet...
if (UNDETERMINED_SCROLLOFFSET == scrollThreshold) {
return;
}
this.fire('scroll', [currentScrollOffset, scrollThreshold]);
if (currentScrollOffset >= scrollThreshold) {
this.next();
}
};
/**
* Returns the last item currently in the DOM
*
* @private
* @returns {object}
*/
this.getLastItem = function() {
return $(this.itemSelector, this.$itemsContainer.get(0)).last();
};
/**
* Returns the first item currently in the DOM
*
* @private
* @returns {object}
*/
this.getFirstItem = function() {
return $(this.itemSelector, this.$itemsContainer.get(0)).first();
};
/**
* Returns scroll threshold. This threshold marks the line from where
* IAS should start loading the next page.
*
* @private
* @param negativeMargin defaults to {this.negativeMargin}
* @return {number}
*/
this.getScrollThreshold = function(negativeMargin) {
var $lastElement;
negativeMargin = negativeMargin || this.negativeMargin;
negativeMargin = (negativeMargin >= 0 ? negativeMargin * -1 : negativeMargin);
$lastElement = this.getLastItem();
// if the don't have a last element, the DOM might not have been loaded,
// or the selector is invalid
if (0 === $lastElement.size()) {
return UNDETERMINED_SCROLLOFFSET;
}
return ($lastElement.offset().top + $lastElement.height() + negativeMargin);
};
/**
* Returns current scroll offset for the given scroll container
*
* @private
* @param $container
* @returns {number}
*/
this.getCurrentScrollOffset = function($container) {
var scrollTop = 0,
containerHeight = $container.height();
if (window === $container.get(0)) {
scrollTop = $container.scrollTop();
} else {
scrollTop = $container.offset().top;
}
// compensate for iPhone
if (navigator.platform.indexOf("iPhone") != -1 || navigator.platform.indexOf("iPod") != -1) {
containerHeight += 80;
}
return (scrollTop + containerHeight);
};
/**
* Returns the url for the next page
*
* @private
*/
this.getNextUrl = function(container) {
if (!container) {
container = this.$container;
}
// always take the last matching item
var next_url = $(this.nextSelector, container).last().attr('href');
if(typeof(next_url) != 'undefined') {
next_url += '&ajaxscroll=1';
} else {
next_url = '';
}
return next_url;
};
/**
* Loads a page url
*
* @param url
* @param callback
* @param delay
* @returns {object} jsXhr object
*/
this.load = function(url, callback, delay) {
var self = this,
$itemContainer,
items = [],
timeStart = +new Date(),
timeDiff;
delay = delay || this.defaultDelay;
var loadEvent = {
url: url
};
self.fire('load', [loadEvent]);
return $.get(loadEvent.url, null, $.proxy(function(data) {
$itemContainer = $(this.itemsContainerSelector, data).eq(0);
if (0 === $itemContainer.length) {
$itemContainer = $(data).filter(this.itemsContainerSelector).eq(0);
}
if ($itemContainer) {
$itemContainer.find(this.itemSelector).each(function() {
items.push(this);
});
}
self.fire('loaded', [data, items]);
if (callback) {
timeDiff = +new Date() - timeStart;
if (timeDiff < delay) {
setTimeout(function() {
callback.call(self, data, items);
}, delay - timeDiff);
} else {
callback.call(self, data, items);
}
}
}, self), 'html');
};
/**
* Renders items
*
* @param callback
* @param items
*/
this.render = function(items, callback) {
var self = this,
$lastItem = this.getLastItem(),
count = 0;
var promise = this.fire('render', [items]);
promise.done(function() {
$(items).hide(); // at first, hide it so we can fade it in later
$lastItem.after(items);
$(items).fadeIn(400, function() {
// complete callback get fired for each item,
// only act on the last item
if (++count < items.length) {
return;
}
self.fire('rendered', [items]);
if (callback) {
callback();
}
});
});
};
/**
* Hides the pagination
*/
this.hidePagination = function() {
if (this.paginationSelector) {
$(this.paginationSelector, this.$container).hide();
}
};
/**
* Restores the pagination
*/
this.restorePagination = function() {
if (this.paginationSelector) {
$(this.paginationSelector, this.$container).show();
}
};
/**
* Throttles a method
*
* Adopted from Ben Alman's jQuery throttle / debounce plugin
*
* @param callback
* @param delay
* @return {object}
*/
this.throttle = function(callback, delay) {
var lastExecutionTime = 0,
wrapper,
timerId
;
wrapper = function() {
var that = this,
args = arguments,
diff = +new Date() - lastExecutionTime;
function execute() {
lastExecutionTime = +new Date();
callback.apply(that, args);
}
if (!timerId) {
execute();
} else {
clearTimeout(timerId);
}
if (diff > delay) {
execute();
} else {
timerId = setTimeout(execute, delay);
}
};
if ($.guid) {
wrapper.guid = callback.guid = callback.guid || $.guid++;
}
return wrapper;
};
/**
* Fires an event with the ability to cancel further processing. This
* can be achieved by returning false in a listener.
*
* @param event
* @param args
* @returns {*}
*/
this.fire = function(event, args) {
return this.listeners[event].fireWith(this, args);
};
return this;
};
/**
* Initialize IAS
*
* Note: Should be called when the document is ready
*
* @public
*/
IAS.prototype.initialize = function() {
var currentScrollOffset = this.getCurrentScrollOffset(this.$scrollContainer),
scrollThreshold = this.getScrollThreshold();
this.hidePagination();
this.bind();
for (var i = 0, l = this.extensions.length; i < l; i++) {
this.extensions[i].bind(this);
}
this.fire('ready');
this.nextUrl = this.getNextUrl();
// start loading next page if content is shorter than page fold
if (currentScrollOffset >= scrollThreshold && this.nextUrl) {
this.next();
}
return this;
};
/**
* Binds IAS to DOM events
*
* @public
*/
IAS.prototype.bind = function() {
if (this.isBound) {
return;
}
this.$scrollContainer.on('scroll', $.proxy(this.throttle(this.scrollHandler, 150), this));
this.isBound = true;
};
/**
* Unbinds IAS to events
*
* @public
*/
IAS.prototype.unbind = function() {
if (!this.isBound) {
return;
}
this.$scrollContainer.off('scroll', this.scrollHandler);
this.isBound = false;
};
/**
* Destroys IAS instance
*
* @public
*/
IAS.prototype.destroy = function() {
this.unbind();
};
/**
* Registers an eventListener
*
* Note: chainable
*
* @public
* @returns IAS
*/
IAS.prototype.on = function(event, callback, priority) {
if (typeof this.listeners[event] == 'undefined') {
throw new Error('There is no event called "' + event + '"');
}
priority = priority || 0;
this.listeners[event].add($.proxy(callback, this), priority);
return this;
};
/**
* Registers an eventListener which only gets
* fired once.
*
* Note: chainable
*
* @public
* @returns IAS
*/
IAS.prototype.one = function(event, callback) {
var self = this;
var remover = function() {
self.off(event, callback);
self.off(event, remover);
};
this.on(event, callback);
this.on(event, remover);
return this;
};
/**
* Removes an eventListener
*
* Note: chainable
*
* @public
* @returns IAS
*/
IAS.prototype.off = function(event, callback) {
if (typeof this.listeners[event] == 'undefined') {
throw new Error('There is no event called "' + event + '"');
}
this.listeners[event].remove(callback);
return this;
};
/**
* Load the next page
*
* @public
*/
IAS.prototype.next = function() {
var url = this.nextUrl,
self = this;
this.unbind();
if (!url) {
this.fire('noneLeft', [this.getLastItem()]);
this.listeners['noneLeft'].disable(); // disable it so it only fires once
self.bind();
return false;
}
var promise = this.fire('next', [url]);
promise.done(function() {
self.load(url, function(data, items) {
self.render(items, function() {
self.nextUrl = self.getNextUrl(data);
self.bind();
});
});
});
promise.fail(function() {
self.bind();
});
return true;
};
/**
* Adds an extension
*
* @public
*/
IAS.prototype.extension = function(extension) {
if (typeof extension['bind'] == 'undefined') {
throw new Error('Extension doesn\'t have required method "bind"');
}
if (typeof extension['initialize'] != 'undefined') {
extension.initialize(this);
}
this.extensions.push(extension);
return this;
};
/**
* Shortcut. Sets the window as scroll container.
*
* @public
* @param option
* @returns {*}
*/
$.ias = function(option) {
var $window = $(window);
return $window.ias.apply($window, arguments);
};
/**
* jQuery plugin initialization
*
* @public
* @param option
* @returns {*} the last IAS instance will be returned
*/
$.fn.ias = function(option) {
var args = Array.prototype.slice.call(arguments);
var retval = this;
this.each(function() {
var $this = $(this),
data = $this.data('ias'),
options = $.extend({}, $.fn.ias.defaults, $this.data(), typeof option == 'object' && option)
;
// set a new instance as data
if (!data) {
$this.data('ias', (data = new IAS($this, options)));
$(document).ready($.proxy(data.initialize, data));
}
// when the plugin is called with a method
if (typeof option === 'string') {
if (typeof data[option] !== 'function') {
throw new Error('There is no method called "' + option + '"');
}
args.shift(); // remove first argument ('option')
data[option].apply(data, args);
if (option === 'destroy') {
$this.data('ias', null);
}
}
retval = $this.data('ias');
});
return retval;
};
/**
* Plugin defaults
*
* @public
* @type {object}
*/
$.fn.ias.defaults = {
item: '.item',
container: '.listing',
next: '.next',
pagination: false,
delay: 600,
negativeMargin: 10
};
})(jQuery);
});
if(typeof(IASHistoryExtension) == "undefined"){
/**
* IAS History Extension
* An IAS extension to enable browser history
* http://infiniteajaxscroll.com
*
* This file is part of the Infinite AJAX Scroll package
*
* Copyright 2014 Webcreate (Jeroen Fiege)
*/
var IASHistoryExtension = function (options) {
options = jQuery.extend({}, this.defaults, options);
this.ias = null;
this.prevSelector = options.prev;
this.prevUrl = null;
this.listeners = {
prev: new IASCallbacks()
};
/**
* @private
* @param pageNum
* @param scrollOffset
* @param url
*/
this.onPageChange = function (pageNum, scrollOffset, url) {
var state = {};
if (!window.history || !window.history.replaceState) {
return;
}
history.replaceState(state, document.title, url);
};
/**
* @private
* @param currentScrollOffset
* @param scrollThreshold
*/
this.onScroll = function (currentScrollOffset, scrollThreshold) {
var firstItemScrollThreshold = this.getScrollThresholdFirstItem();
if (!this.prevUrl) {
return;
}
currentScrollOffset -= this.ias.$scrollContainer.height();
if (currentScrollOffset <= firstItemScrollThreshold) {
this.prev();
}
};
/**
* Returns the url for the next page
*
* @private
*/
this.getPrevUrl = function (container) {
if (!container) {
container = this.ias.$container;
}
// always take the last matching item
var prev_url = jQuery(this.prevSelector, container).last().attr('href');
if(typeof(prev_url) != 'undefined') {
prev_url += '&ajaxscroll=1';
} else {
prev_url = '';
}
return prev_url;
};
/**
* Returns scroll threshold. This threshold marks the line from where
* IAS should start loading the next page.
*
* @private
* @return {number}
*/
this.getScrollThresholdFirstItem = function () {
var $firstElement;
$firstElement = this.ias.getFirstItem();
// if the don't have a first element, the DOM might not have been loaded,
// or the selector is invalid
if (0 === $firstElement.size()) {
return -1;
}
return ($firstElement.offset().top);
};
/**
* Renders items
*
* @private
* @param items
* @param callback
*/
this.renderBefore = function (items, callback) {
var ias = this.ias,
$firstItem = ias.getFirstItem(),
count = 0;
ias.fire('render', [items]);
jQuery(items).hide(); // at first, hide it so we can fade it in later
$firstItem.before(items);
jQuery(items).fadeIn(400, function () {
if (++count < items.length) {
return;
}
ias.fire('rendered', [items]);
if (callback) {
callback();
}
});
};
return this;
};
/**
* @public
*/
IASHistoryExtension.prototype.initialize = function (ias) {
var self = this;
this.ias = ias;
// expose the extensions listeners
jQuery.extend(ias.listeners, this.listeners);
// expose prev method
ias.prev = function() {
return self.prev();
};
this.prevUrl = this.getPrevUrl();
};
/**
* Bind to events
*
* @public
* @param ias
*/
IASHistoryExtension.prototype.bind = function (ias) {
var self = this;
ias.on('pageChange', jQuery.proxy(this.onPageChange, this));
ias.on('scroll', jQuery.proxy(this.onScroll, this));
ias.on('ready', function () {
var currentScrollOffset = ias.getCurrentScrollOffset(ias.$scrollContainer),
firstItemScrollThreshold = self.getScrollThresholdFirstItem();
currentScrollOffset -= ias.$scrollContainer.height();
if (currentScrollOffset <= firstItemScrollThreshold) {
self.prev();
}
});
};
/**
* Load the prev page
*
* @public
*/
IASHistoryExtension.prototype.prev = function () {
var url = this.prevUrl,
self = this,
ias = this.ias;
if (!url) {
return false;
}
ias.unbind();
var promise = ias.fire('prev', [url]);
promise.done(function () {
ias.load(url, function (data, items) {
self.renderBefore(items, function () {
self.prevUrl = self.getPrevUrl(data);
ias.bind();
if (self.prevUrl) {
self.prev();
}
});
});
});
promise.fail(function () {
ias.bind();
});
return true;
};
/**
* @public
*/
IASHistoryExtension.prototype.defaults = {
prev: ".prev"
};
}
if(typeof(IASNoneLeftExtension) == "undefined"){
/**
* IAS None Left Extension
* An IAS extension to show a message when there are no more pages te load
* http://infiniteajaxscroll.com
*
* This file is part of the Infinite AJAX Scroll package
*
* Copyright 2014 Webcreate (Jeroen Fiege)
*/
var IASNoneLeftExtension = function(options) {
options = jQuery.extend({}, this.defaults, options);
this.ias = null;
this.uid = (new Date()).getTime();
this.html = (options.html).replace('{text}', options.text);
/**
* Shows none left message
*/
this.showNoneLeft = function() {
var $element = jQuery(this.html).attr('id', 'ias_noneleft_' + this.uid),
$lastItem = this.ias.getLastItem();
$lastItem.after($element);
$element.fadeIn();
};
return this;
};
/**
* @public
*/
IASNoneLeftExtension.prototype.bind = function(ias) {
this.ias = ias;
ias.on('noneLeft', jQuery.proxy(this.showNoneLeft, this));
};
/**
* @public
*/
IASNoneLeftExtension.prototype.defaults = {
text: 'You reached the end.',
html: '<div class="ias-noneleft" style="text-align: center;">{text}</div>'
};
}
if(typeof(IASPagingExtension) == "undefined"){
/**
* IAS Paging Extension
* An IAS extension providing additional events
* http://infiniteajaxscroll.com
*
* This file is part of the Infinite AJAX Scroll package
*
* Copyright 2014 Webcreate (Jeroen Fiege)
*/
var IASPagingExtension = function() {
this.ias = null;
this.pagebreaks = [[0, document.location.toString()]];
this.lastPageNum = 1;
this.enabled = true;
this.listeners = {
pageChange: new IASCallbacks()
};
/**
* Fires pageChange event
*
* @param currentScrollOffset
* @param scrollThreshold
*/
this.onScroll = function(currentScrollOffset, scrollThreshold) {
if (!this.enabled) {
return;
}
var ias = this.ias,
currentPageNum = this.getCurrentPageNum(currentScrollOffset),
currentPagebreak = this.getCurrentPagebreak(currentScrollOffset),
urlPage;
if (this.lastPageNum !== currentPageNum) {
urlPage = currentPagebreak[1];
ias.fire('pageChange', [currentPageNum, currentScrollOffset, urlPage]);
}
this.lastPageNum = currentPageNum;
};
/**
* Keeps track of pagebreaks
*
* @param url
*/
this.onNext = function(url) {
var currentScrollOffset = this.ias.getCurrentScrollOffset(this.ias.$scrollContainer);
this.pagebreaks.push([currentScrollOffset, url]);
// trigger pageChange and update lastPageNum
var currentPageNum = this.getCurrentPageNum(currentScrollOffset) + 1;
this.ias.fire('pageChange', [currentPageNum, currentScrollOffset, url]);
this.lastPageNum = currentPageNum;
};
/**
* Keeps track of pagebreaks
*
* @param url
*/
this.onPrev = function(url) {
var self = this,
ias = self.ias,
currentScrollOffset = ias.getCurrentScrollOffset(ias.$scrollContainer),
prevCurrentScrollOffset = currentScrollOffset - ias.$scrollContainer.height(),
$firstItem = ias.getFirstItem();
this.enabled = false;
this.pagebreaks.unshift([0, url]);
ias.one('rendered', function() {
// update pagebreaks
for (var i = 1, l = self.pagebreaks.length; i < l; i++) {
self.pagebreaks[i][0] = self.pagebreaks[i][0] + $firstItem.offset().top;
}
// trigger pageChange and update lastPageNum
var currentPageNum = self.getCurrentPageNum(prevCurrentScrollOffset) + 1;
ias.fire('pageChange', [currentPageNum, prevCurrentScrollOffset, url]);
self.lastPageNum = currentPageNum;
self.enabled = true;
});
};
return this;
};
/**
* @public
*/
IASPagingExtension.prototype.initialize = function(ias) {
this.ias = ias;
// expose the extensions listeners
jQuery.extend(ias.listeners, this.listeners);
};
/**
* @public
*/
IASPagingExtension.prototype.bind = function(ias) {
try {
ias.on('prev', jQuery.proxy(this.onPrev, this), this.priority);
} catch (exception) {}
ias.on('next', jQuery.proxy(this.onNext, this), this.priority);
ias.on('scroll', jQuery.proxy(this.onScroll, this), this.priority);
};
/**
* Returns current page number based on scroll offset
*
* @param {number} scrollOffset
* @returns {number}
*/
IASPagingExtension.prototype.getCurrentPageNum = function(scrollOffset) {
for (var i = (this.pagebreaks.length - 1); i > 0; i--) {
if (scrollOffset > this.pagebreaks[i][0]) {
return i + 1;
}
}
return 1;
};
/**
* Returns current pagebreak information based on scroll offset
*
* @param {number} scrollOffset
* @returns {number}|null
*/
IASPagingExtension.prototype.getCurrentPagebreak = function(scrollOffset) {
for (var i = (this.pagebreaks.length - 1); i >= 0; i--) {
if (scrollOffset > this.pagebreaks[i][0]) {
return this.pagebreaks[i];
}
}
return null;
};
/**
* @public
* @type {number}
*/
IASPagingExtension.prototype.priority = 500;
}
if(typeof(IASSpinnerExtension) == "undefined"){
/**
* IAS Spinner Extension
* An IAS extension to show a spinner when loading
* http://infiniteajaxscroll.com
*
* This file is part of the Infinite AJAX Scroll package
*
* Copyright 2014 Webcreate (Jeroen Fiege)
*/
var IASSpinnerExtension = function(options) {
options = jQuery.extend({}, this.defaults, options);
this.ias = null;
this.uid = new Date().getTime();
this.src = options.src;
this.html = (options.html).replace('{src}', this.src);
/**
* Shows spinner
*/
this.showSpinner = function() {
var $spinner = this.getSpinner() || this.createSpinner(),
$lastItem = this.ias.getLastItem();
$lastItem.after($spinner);
$spinner.fadeIn();
};
/**
* Shows spinner
*/
this.showSpinnerBefore = function() {
var $spinner = this.getSpinner() || this.createSpinner(),
$firstItem = this.ias.getFirstItem();
$firstItem.before($spinner);
$spinner.fadeIn();
};
/**
* Removes spinner
*/
this.removeSpinner = function() {
if (this.hasSpinner()) {
this.getSpinner().remove();
}
};
/**
* @returns {jQuery|boolean}
*/
this.getSpinner = function() {
var $spinner = jQuery('#ias_spinner_' + this.uid);
if ($spinner.size() > 0) {
return $spinner;
}
return false;
};
/**
* @returns {boolean}
*/
this.hasSpinner = function() {
var $spinner = jQuery('#ias_spinner_' + this.uid);
return ($spinner.size() > 0);
};
/**
* @returns {jQuery}
*/
this.createSpinner = function() {
var $spinner = jQuery(this.html).attr('id', 'ias_spinner_' + this.uid);
$spinner.hide();
return $spinner;
};
return this;
};
/**
* @public
*/
IASSpinnerExtension.prototype.bind = function(ias) {
this.ias = ias;
ias.on('next', jQuery.proxy(this.showSpinner, this));
try {
ias.on('prev', jQuery.proxy(this.showSpinnerBefore, this));
} catch (exception) {}
ias.on('render', jQuery.proxy(this.removeSpinner, this));
};
/**
* @public
*/
IASSpinnerExtension.prototype.defaults = {
src: '',
html: '<div class="ias-spinner" style="text-align: center;"><img src="{src}"/></div>'
};
}
if(typeof(IASTriggerExtension) == "undefined"){
/**
* IAS Trigger Extension
* An IAS extension to show a trigger link to load the next page
* http://infiniteajaxscroll.com
*
* This file is part of the Infinite AJAX Scroll package
*
* Copyright 2014 Webcreate (Jeroen Fiege)
*/
var IASTriggerExtension = function(options) {
options = jQuery.extend({}, this.defaults, options);
this.ias = null;
this.html = (options.html).replace('{text}', options.text);
this.htmlPrev = (options.htmlPrev).replace('{text}', options.textPrev);
this.enabled = true;
this.count = 0;
this.offset = options.offset;
this.$triggerNext = null;
this.$triggerPrev = null;
/**
* Shows trigger for next page
*/
this.showTriggerNext = function() {
if (!this.enabled) {
return true;
}
if (false === this.offset || ++this.count < this.offset) {
return true;
}
var $trigger = this.$triggerNext || (this.$triggerNext = this.createTrigger(this.next, this.html));
var $lastItem = this.ias.getLastItem();
$lastItem.after($trigger);
$trigger.fadeIn();
return false;
};
/**
* Shows trigger for previous page
*/
this.showTriggerPrev = function() {
if (!this.enabled) {
return true;
}
var $trigger = this.$triggerPrev || (this.$triggerPrev = this.createTrigger(this.prev, this.htmlPrev));
var $firstItem = this.ias.getFirstItem();
$firstItem.before($trigger);
$trigger.fadeIn();
return false;
};
/**
* @param clickCallback
* @returns {*|jQuery}
* @param {string} html
*/
this.createTrigger = function(clickCallback, html) {
var uid = (new Date()).getTime(),
$trigger;
html = html || this.html;
$trigger = jQuery(html).attr('id', 'ias_trigger_' + uid);
$trigger.hide();
$trigger.on('click', jQuery.proxy(clickCallback, this));
return $trigger;
};
return this;
};
/**
* @public
* @param {object} ias
*/
IASTriggerExtension.prototype.bind = function(ias) {
var self = this;
this.ias = ias;
try {
ias.on('prev', jQuery.proxy(this.showTriggerPrev, this), this.priority);
} catch (exception) {}
ias.on('next', jQuery.proxy(this.showTriggerNext, this), this.priority);
ias.on('rendered', function () { self.enabled = true; }, this.priority);
};
/**
* @public
*/
IASTriggerExtension.prototype.next = function() {
this.enabled = false;
this.ias.unbind();
if (this.$triggerNext) {
this.$triggerNext.remove();
this.$triggerNext = null;
}
this.ias.next();
};
/**
* @public
*/
IASTriggerExtension.prototype.prev = function() {
this.enabled = false;
this.ias.unbind();
if (this.$triggerPrev) {
this.$triggerPrev.remove();
this.$triggerPrev = null;
}
this.ias.prev();
};
/**
* @public
*/
IASTriggerExtension.prototype.defaults = {
text: 'Load more items',
html: '<div class="ias-trigger ias-trigger-next" style="text-align: center; cursor: pointer;"><a>{text}</a></div>',
textPrev: 'Load previous items',
htmlPrev: '<div class="ias-trigger ias-trigger-prev" style="text-align: center; cursor: pointer;"><a>{text}</a></div>',
offset: 0
};
/**
* @public
* @type {number}
*/
IASTriggerExtension.prototype.priority = 1000;
}
window.IASCallbacks=IASCallbacks;window.IASHistoryExtension=IASHistoryExtension;window.IASTriggerExtension=IASTriggerExtension;window.IASSpinnerExtension=IASSpinnerExtension;window.IASPagingExtension=IASPagingExtension;window.IASNoneLeftExtension=IASNoneLeftExtension;
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment