Source: texture/LazyTexture.js

import Class from '../core/Class';
import EventMixin from '../core/EventMixin';
import Texture from './Texture';
import Loader from '../loader/Loader';
import log from '../utils/log';

const placeHolder = new Image();
placeHolder.src = '';

/**
 * 懒加载纹理
 * @class
 * @extends Texture
 * @mixes EventMixin
 * @fires load 加载成功事件
 * @fires error 加载失败事件
 * @example
 * var material = new Hilo3d.BasicMaterial({
 *     diffuse: new Hilo3d.LazyTexture({
 *         crossOrigin: true,
 *         src: '//img.alicdn.com/tfs/TB1aNxtQpXXXXX1XVXXXXXXXXXX-1024-1024.jpg'
 *     });
 * });
 */
const LazyTexture = Class.create(/** @lends LazyTexture.prototype */ {
    Extends: Texture,
    Mixes: EventMixin,

    /**
     * @default true
     * @type {boolean}
     */
    isLazyTexture: true,

    /**
     * @default LazyTexture
     * @type {string}
     */
    className: 'LazyTexture',

    _src: '',

    /**
     * 图片是否跨域
     * @default false
     * @type {boolean}
     */
    crossOrigin: false,

    /**
     * 是否在设置src后立即加载图片
     * @default true
     * @type {boolean}
     */
    autoLoad: true,
    /**
     * 资源类型,用于加载时判断
     * @type {string}
     */
    resType: '',

    /**
     * 图片地址
     * @type {string}
     */
    src: {
        get() {
            return this._src;
        },
        set(src) {
            if (this._src !== src) {
                this._src = src;
                if (this.autoLoad) {
                    this.load();
                }
            }
        }
    },

    /**
     * @constructs
     * @param {object} [params] 初始化参数,所有params都会复制到实例上
     * @param {boolean} [params.crossOrigin=false] 是否跨域
     * @param {HTMLImageElement} [params.placeHolder] 占位图片,默认为1像素的透明图片
     * @param {boolean} [params.autoLoad=true] 是否自动加载
     * @param {string} [params.src] 图片地址
     * @param {any} [params.[value:string]] 其它属性
     */
    constructor(params) {
        if (params) {
            // 必须在src设置前赋值
            if ('crossOrigin' in params) {
                this.crossOrigin = params.crossOrigin;
            }
            if ('autoLoad' in params) {
                this.autoLoad = params.autoLoad;
            }
        }
        LazyTexture.superclass.constructor.call(this, params);
        this.image = this.placeHolder || placeHolder;
    },

    /**
     * 加载图片
     * @param {boolean} [throwError=false] 是否 throw error
     * @return {Promise<void>} 返回加载的Promise
     */
    load(throwError) {
        LazyTexture.loader = LazyTexture.loader || new Loader();
        return LazyTexture.loader.load({
            src: this.src,
            crossOrigin: this.crossOrigin,
            type: this.resType,
            defaultType: 'img'
        }).then((img) => {
            if (img.isTexture) {
                Object.assign(this, img);
                this.needUpdate = true;
                this.needDestroy = true;
                this.fire('load');
            } else {
                this.image = img;
                this.needUpdate = true;
                this.fire('load');
            }
        }, (err) => {
            this.fire('error');
            if (throwError) {
                throw new Error(`LazyTexture Failed ${err}`);
            } else {
                log.warn(`LazyTexture Failed ${err}`);
            }
        });
    },
    _releaseImage() {
        if (this._src && typeof this._src !== 'string') {
            this._src = '';
        }
        LazyTexture.superclass._releaseImage.call(this);
    }
});

export default LazyTexture;