diff options
| author | Alex Yatskov <alex@foosoft.net> | 2020-05-22 17:46:16 -0700 | 
|---|---|---|
| committer | Alex Yatskov <alex@foosoft.net> | 2020-05-22 17:46:16 -0700 | 
| commit | 1480288561cb8b9fb87ad711d970c548329fea98 (patch) | |
| tree | 87c2247f6d144407afcc6de316bbacc264582248 /ext/mixed/js/media-loader.js | |
| parent | f2186c51e4ef219d158735d30a32bbf3e49c4e1a (diff) | |
| parent | d0dcff765f740bf6f0f6523b09cb8b21eb85cd93 (diff) | |
Merge branch 'master' into testing
Diffstat (limited to 'ext/mixed/js/media-loader.js')
| -rw-r--r-- | ext/mixed/js/media-loader.js | 107 | 
1 files changed, 107 insertions, 0 deletions
| diff --git a/ext/mixed/js/media-loader.js b/ext/mixed/js/media-loader.js new file mode 100644 index 00000000..64ccd715 --- /dev/null +++ b/ext/mixed/js/media-loader.js @@ -0,0 +1,107 @@ +/* + * Copyright (C) 2020  Yomichan Authors + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program.  If not, see <https://www.gnu.org/licenses/>. + */ + +/* global + * apiGetMedia + */ + +class MediaLoader { +    constructor() { +        this._token = {}; +        this._mediaCache = new Map(); +        this._loadMediaData = []; +    } + +    async loadMedia(path, dictionaryName, onLoad, onUnload) { +        const token = this.token; +        const data = {onUnload, loaded: false}; + +        this._loadMediaData.push(data); + +        const media = await this.getMedia(path, dictionaryName); +        if (token !== this.token) { return; } + +        onLoad(media.url); +        data.loaded = true; +    } + +    unloadAll() { +        for (const {onUnload, loaded} of this._loadMediaData) { +            if (typeof onUnload === 'function') { +                onUnload(loaded); +            } +        } +        this._loadMediaData = []; + +        for (const map of this._mediaCache.values()) { +            for (const {url} of map.values()) { +                if (url !== null) { +                    URL.revokeObjectURL(url); +                } +            } +        } +        this._mediaCache.clear(); + +        this._token = {}; +    } + +    async getMedia(path, dictionaryName) { +        let cachedData; +        let dictionaryCache = this._mediaCache.get(dictionaryName); +        if (typeof dictionaryCache !== 'undefined') { +            cachedData = dictionaryCache.get(path); +        } else { +            dictionaryCache = new Map(); +            this._mediaCache.set(dictionaryName, dictionaryCache); +        } + +        if (typeof cachedData === 'undefined') { +            cachedData = { +                promise: null, +                data: null, +                url: null +            }; +            dictionaryCache.set(path, cachedData); +            cachedData.promise = this._getMediaData(path, dictionaryName, cachedData); +        } + +        return cachedData.promise; +    } + +    async _getMediaData(path, dictionaryName, cachedData) { +        const token = this._token; +        const data = (await apiGetMedia([{path, dictionaryName}]))[0]; +        if (token === this._token && data !== null) { +            const contentArrayBuffer = this._base64ToArrayBuffer(data.content); +            const blob = new Blob([contentArrayBuffer], {type: data.mediaType}); +            const url = URL.createObjectURL(blob); +            cachedData.data = data; +            cachedData.url = url; +        } +        return cachedData; +    } + +    _base64ToArrayBuffer(content) { +        const binaryContent = window.atob(content); +        const length = binaryContent.length; +        const array = new Uint8Array(length); +        for (let i = 0; i < length; ++i) { +            array[i] = binaryContent.charCodeAt(i); +        } +        return array.buffer; +    } +} |