angular实现图片懒加载

1.package.json中引入

"imgcache.js": "^1.0.0"

2.yarn install 保证插件的加入
3.verdor.js 加入引用

"imgcache.js/js/imgcache.js"

4.开始实现功能,一共自定义两个文件
5.新建image-loader.js

/**
 * 图片加载,已加载图片会缓存在应用本地,非应用环境使用浏览器缓存
 */
angular.module('app.services').factory('imageLoader', function($ionicPlatform, $cmApi, $cmLocalStorage) {
    // 本地存储预加载列表特征值
    const PRELOAD_HASH = 'preloadHash';
    // 本地存储预加载列表已加载计数
    const PRELOAD_COUNT = 'preloadCount';
    // 加载并行请求限制
    const LOAD_REQUEST_LIMIT = 5;
    // 预加载并行请求限制
    const PRELOAD_REQUEST_LIMIT = 3;
    // 图片缓存区容量
    const CACHE_QUOTA = 100 * 1024 * 1024;
    // 正在进行中的图片请求计数
    let requestCount = 0;
    // 请求队列
    const requestQueue = [];

    let imageCache = null;

    // 等待执行的预加载任务
    let preloadTask = null;

    let startPreloadWithDelay = null;

    // 初始化图片缓存
    $ionicPlatform.ready(function() {
        if (window.cordova && window.ImgCache) {
            ImgCache.options.chromeQuota = CACHE_QUOTA;
            // if (ionic.Platform.isIOS()) {
            //     ImgCache.options.useDataURI = true;
            // }
            window.ImgCache.init(function() {
                imageCache = window.ImgCache;
                // 如果有等待执行的预加载任务,则开始预加载
                if (preloadTask) {
                    preloadTask();
                    preloadTask = null;
                }
            }, angular.noop);
        }
    });

    // 处理请求队列中第一个请求
    function processRequestInQueue() {
        if (requestQueue.length) {
            requestQueue.shift()();
        }
    }

    // 生成对象特征值
    function generateHashCode(obj) {
        const str = JSON.stringify(obj);
        let hash = 0,
            i,
            chr,
            len;
        if (str.length === 0) return hash;
        for (i = 0, len = str.length; i < len; i++) {
            chr = str.charCodeAt(i);
            hash = (hash << 5) - hash + chr;
            hash |= 0;
        }
        return hash;
    }

    return {
        // 加载图片
        load: function($image, src, successCallback) {
            const deferred = $cmApi.defer();

            // 使用 web 环境直接加载图片
            const useOnlineImage = function() {
                const img = document.createElement('img');
                requestCount++;
                img.onload = function() {
                    $image[0].src = src;
                    if (successCallback) {
                        successCallback();
                    }
                    deferred.resolve();
                    requestCount--;
                };
                img.onerror = function() {
                    deferred.reject();
                    requestCount--;
                };
                img.src = src;
            };

            const loadHandler = function() {
                if (imageCache) {
                    // 加载图片,图片没有本地缓存时先缓存至本地
                    ImgCache.useCachedFileWithSource(
                        $image,
                        src,
                        function() {
                            if (successCallback) {
                                successCallback();
                            }
                            deferred.resolve();
                        },
                        function() {
                            useOnlineImage();

                            // 使用 wifi 时缓存图片
                            const connection = navigator.connection;
                            if (connection && connection.type == Connection.WIFI) {
                                ImgCache.cacheFile(src);
                            }
                        }
                    );
                } else {
                    useOnlineImage();
                }
            };

            if (requestCount <= LOAD_REQUEST_LIMIT) {
                loadHandler();
            } else {
                requestQueue.push(loadHandler);
            }

            deferred.promise.finally(processRequestInQueue);

            return deferred.promise;
        },
        // 加载本地图片
        loadLocalImage: function($image, src, successCallback) {
            const deferred = $cmApi.defer();
            if (imageCache) {
                ImgCache.useCachedFileWithSource(
                    $image,
                    src,
                    function() {
                        if (successCallback) {
                            successCallback();
                        }
                        deferred.resolve();
                    },
                    function() {
                        deferred.reject();
                    }
                );
            } else {
                deferred.reject();
            }
            return deferred.promise;
        },
        // 获取当前已缓存图片容量
        getCurrentSize: function() {
            if (imageCache) {
                return (imageCache.getCurrentSize() / 1024 / 1024).toFixed(1) + 'MB';
            }
        },
        // 清除本地缓存
        clearCache: function() {
            const deferred = $cmApi.defer();
            if (window.ImgCache) {
                ImgCache.clearCache(
                    function() {
                        // 清除本地预加载信息
                        $cmLocalStorage.remove(PRELOAD_HASH);
                        $cmLocalStorage.remove(PRELOAD_COUNT);

                        deferred.resolve({
                            status: 200,
                            data: {},
                        });
                    },
                    function() {
                        deferred.reject();
                    }
                );
            } else {
                deferred.reject();
            }
            return deferred.promise;
        },
        // 预加载图片
        preload: function(imageList) {
            const hash = generateHashCode(imageList);
            let i = 0;
            let loaded = 0; // 已完成计数
            let interval = null;

            // 检查上一次是否有部分预加载已完成
            if (hash == $cmLocalStorage.get(PRELOAD_HASH, 0)) {
                // 加载未完成内容
                loaded = +$cmLocalStorage.get(PRELOAD_COUNT, 0);
            } else {
                // 设置新预加载信息
                $cmLocalStorage.set(PRELOAD_HASH, hash);
                $cmLocalStorage.set(PRELOAD_COUNT, 0);
            }

            // 记录已加载数量
            const recordLoadedCount = function(index) {
                if (index > loaded) {
                    loaded = index;
                    // 每加载10张图或加载完成时记录一次加载进度
                    if (index % 10 === 0 || index === imageList.length - 1) {
                        $cmLocalStorage.set(PRELOAD_COUNT, index);
                    }
                }
            };

            // 预加载图片
            const preloadImage = function(index) {
                const src = imageList[index];
                ImgCache.isCached(src, function(src, cached) {
                    if (cached) {
                        recordLoadedCount(index);
                    } else {
                        requestCount++;
                        ImgCache.cacheFile(
                            src,
                            function() {
                                requestCount--;
                                recordLoadedCount(index);
                                processRequestInQueue();
                            },
                            function() {
                                requestCount--;
                                processRequestInQueue();
                            }
                        );
                    }
                });
            };

            // 停止预加载
            const stopPreload = function() {
                if (interval) {
                    clearInterval(interval);
                    interval = null;
                }
            };

            // 开始预加载
            const startPreload = function() {
                const connection = navigator.connection;
                if (!connection || connection.type != Connection.WIFI || interval) return;
                i = loaded;
                if (i < imageList.length) {
                    interval = setInterval(function() {
                        if (connection.type != Connection.WIFI) {
                            stopPreload();
                            return;
                        }
                        // 限制并发请求数量
                        if (requestCount > PRELOAD_REQUEST_LIMIT) return;

                        if (i < imageList.length) {
                            preloadImage(i);
                            i++;
                        }
                        if (i >= imageList.length) {
                            stopPreload();
                            document.removeEventListener('online', startPreloadWithDelay);
                            document.removeEventListener('offline', stopPreload);
                        }
                    }, 1000);
                }
            };

            // 网络恢复后延时1秒开始预加载
            startPreloadWithDelay = function() {
                setTimeout(function() {
                    startPreload();
                }, 1000);
            };

            document.addEventListener('online', startPreloadWithDelay, false);
            document.addEventListener('offline', stopPreload, false);
            // 检查图片缓存是否已初始化,预加载延迟至图片缓存初始化后开始
            if (imageCache) {
                startPreload();
            } else {
                preloadTask = startPreload;
            }
        },
    };
});

2.新建lazyload.js

/**
 * 延迟图片加载,图片进入可视区时才进行加载
 * preLoadVertical  <Integer>  控制预加载高度,默认 50
 * preLoadHorizontal  <Integer>  控制预加载宽度,默认 30
 * loadImmediate  <Boolean>  不检测是否在可视范围内直接开始加载,默认 false
 * lazyloadWatch  <Boolean>  是否监视图片url变更更新图片,默认 false
 */
angular.module('app.directives').directive('cmLazyload', function($document, imageLoader) {
    // 延迟加载样式类
    const CLASS_LAZY_LOAD = 'lazyload';
    const CLASS_LOADED = 'lazyload-loaded';
    const CLASS_FINISHED = 'lazyload-finished';
    let ANIMATIONEND_EVENT = '';

    if (window.onanimationend === undefined && window.onwebkitanimationend !== undefined) {
        ANIMATIONEND_EVENT = 'webkitAnimationEnd animationend';
    } else {
        ANIMATIONEND_EVENT = 'animationend';
    }
    // 占位图片
    const SPACER =
        '';
    // 加载失败后可重试次数上限
    const RETRY_LIMIT = 3;
    // 指令自增索引
    let count = 0;

    // 所有需要监听的滚动区
    const scrollMap = {};

    let clientHeight = $document[0].documentElement.clientHeight;
    // var clientWidth = $document[0].documentElement.clientWidth;
    window.addEventListener('resize', function() {
        clientHeight = $document[0].documentElement.clientHeight;
        // clientWidth = $document[0].documentElement.clientWidth;
    });

    // 添加滚动监听
    function addScrollHandler(scrollPanel, image) {
        const key = scrollPanel[0].$$hashKey;
        let scrollDelegate = scrollMap[key];
        if (!scrollDelegate) {
            const removeImageFromDelegate = function(id) {
                delete scrollDelegate.images[id];
                scrollDelegate.length--;

                // 滚动区内所有图片已加载时,不再监听滚动
                if (scrollDelegate.length === 0) {
                    scrollPanel.off('scroll', scrollDelegate.handler);
                    delete scrollMap[key];
                }
            };

            const imageLoadSuccessHandler = function(image) {
                return function() {
                    removeImageFromDelegate(image.id);
                    image.loading = false;
                };
            };

            const imageLoadErrorHandler = function(image) {
                return function() {
                    image.retryCount++;
                    if (image.retryCount >= RETRY_LIMIT) {
                        removeImageFromDelegate(image.id);
                    }
                    image.loading = false;
                };
            };

            // 滚动处理
            const scrollHandler = function(e) {
                let scrollTop = 0;
                const event = e.originalEvent;
                if (event.detail && event.detail.scrollTop) {
                    scrollTop = event.detail.scrollTop;
                } else {
                    scrollTop = event.target.scrollTop;
                }

                for (const id in scrollDelegate.images) {
                    const image = scrollDelegate.images[id];
                    if (!image.loading && isInView(image, scrollTop)) {
                        image.loading = true;
                        loadImage(image)
                            .success(imageLoadSuccessHandler(image))
                            .error(imageLoadErrorHandler(image));
                    }
                }
            };

            scrollDelegate = {
                scrollPanel: scrollPanel,
                images: {},
                length: 0,
                handler: ionic.throttle(scrollHandler, 200),
            };
            scrollPanel.on('scroll', scrollDelegate.handler);
            scrollMap[key] = scrollDelegate;
        }
        scrollDelegate.images[image.id] = image;
        scrollDelegate.length++;
    }

    // 检查元素是否在可视区垂直高度内
    function isInView(element, scrollTop) {
        const preLoadVertical = element.preLoadVertical;
        if (scrollTop !== undefined) {
            const top = element.offsetTop - scrollTop,
                bottom = top + element.height;

            return top <= clientHeight + preLoadVertical && bottom >= -preLoadVertical;
        } else {
            const $element = element.$element;
            const imageRect = $element[0].getBoundingClientRect();
            return (
                imageRect.top <= clientHeight + preLoadVertical &&
                imageRect.bottom >= -preLoadVertical
            );
        }
    }

    // 加载图片
    function loadImage(image) {
        const $element = image.$element;
        // 加载并缓存图片
        return imageLoader.load($element, image.src, function functionName() {
            $element.removeClass(CLASS_LAZY_LOAD).addClass(CLASS_LOADED);
        });
    }

    // 淡入动画结束后修改 class
    function animationEnd() {
        $(this)
            .removeClass(CLASS_LOADED)
            .addClass(CLASS_FINISHED);
    }

    // 获取元素坐标和尺寸
    function getElementRect($element) {
        let element = $element,
            offsetLeft = 0,
            offsetTop = 0,
            rect = $element[0].getBoundingClientRect();

        if ($element.is(':visible')) {
            do {
                const offset = ionic.DomUtil.getPositionInParent(element[0]);
                offsetLeft += offset.left;
                offsetTop += offset.top;
                element = element.offsetParent();
                if (element.is('html')) {
                    break;
                }
            } while (!element.hasClass('scroll-content'));
        }

        return {
            width: rect.width,
            height: rect.height,
            offsetLeft: offsetLeft,
            offsetTop: offsetTop,
        };
    }

    return {
        restrict: 'A',
        scope: false,
        link: function($scope, $element, $attr) {
            const scrollPanel = $element.parents('.scroll-content').eq(0);
            const preLoadVertical = parseInt($attr.preLoadVertical) || 50;
            const preLoadHorizontal = parseInt($attr.preLoadHorizontal) || 30;
            const loadImmediate = $attr.loadImmediate;
            let image = null;

            // 初始化延迟加载
            function init() {
                const src = $scope.$eval($attr.cmLazyload);
                if (!src) {
                    return;
                }
                // 处理 watch 导致图片重加载
                if (image) {
                    $element.off(ANIMATIONEND_EVENT, animationEnd);
                    $element.removeClass(CLASS_LOADED).removeClass(CLASS_FINISHED);
                }
                $element[0].src = SPACER;
                $element.addClass(CLASS_LAZY_LOAD);
                $element.on(ANIMATIONEND_EVENT, animationEnd);
                image = {
                    id: count++,
                    $element: $element,
                    loading: false,
                    retryCount: 0,
                    src: src,
                    preLoadVertical: preLoadVertical,
                    preLoadHorizontal: preLoadHorizontal,
                };

                // 释放时间片,等元素渲染完毕后获取元素坐标和尺寸
                setTimeout(function() {
                    angular.extend(image, getElementRect($element));
                }, 0);

                // 检查图片是否已缓存
                imageLoader
                    .loadLocalImage($element, src, function() {
                        $element.removeClass(CLASS_LAZY_LOAD).addClass(CLASS_LOADED);
                    })
                    .error(function() {
                        // 释放时间片,等元素渲染完毕后检查元素状态
                        setTimeout(function() {
                            if (loadImmediate || isInView(image)) {
                                loadImage(image);
                            } else {
                                angular.extend(image, getElementRect($element));
                                addScrollHandler(scrollPanel, image);
                            }
                        }, 0);
                    });
            }

            if ($attr.lazyloadWatch) {
                const listener = $scope.$watch($attr.cmLazyload, init);
                $scope.$on('$destroy', listener);
            } else {
                init();
            }
        },
    };
});

6.在对应的图片加载中可以添加使用,实现懒加载
将之前的

<img ng-src="{{item.picUrl}}" />

改为

<img cm-lazyload="item.picUrl" />

如果对您有帮助,请点个❤️吧😄

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值