/*============================================================================= ApngPicture.js ---------------------------------------------------------------------------- (C)2019 Triacontane This software is released under the MIT License. http://opensource.org/licenses/mit-license.php ---------------------------------------------------------------------------- Version 2.4.2 2023/10/27 シーンのAPNGに表示スイッチを指定したとき、スイッチがOFFでも一瞬だけ表示されてしまう問題を修正 2.4.1 2023/03/21 GIFファイルを使用する場合のヘルプ文言を分かりやすく修正 2.4.0 2022/12/25 APNG画像のぼかしを除去できる設定を追加 2.3.4 2022/12/15 GIFファイルを選択したとき内部で保持する画像サイズが0になってしまう問題を修正 2.3.3 2022/12/14 サイドビュー用の敵キャラファイルの設定パラメータにgifファイルかどうかのフラグがなかった問題を修正 2.3.2 2022/10/27 1セルのフレーム数を変更したとき、ループ回数の設定が機能しなくなる問題を修正 2.3.1 2022/06/06 apngのフレーム数を指定したとき停止スイッチが機能しない問題を修正 2.3.0 2022/02/06 1セルごとのフレーム数をゲーム側で設定できるパラメータを追加 2.2.1 2021/10/09 AltMenuScreen2MZとの並び順を指定するアノテーションを追加 2.2.0 2021/04/26 2.1.6の修正でGIFファイルが使えなくなっていた問題を修正 GIFファイルの指定方法を変更 2.1.6 2021/03/10 システム画像や敵キャラ画像としてapngを使用するとき、ピクチャにも同名の画像がないとエラーになる問題を修正 2.1.5 2021/02/01 数字のみのファイルをapng指定して起動するとエラーになる問題を修正 2.1.4 2021/01/18 2.1.3の修正でapngでないピクチャや敵キャラを表示しようとするとエラーになる問題を修正 2.1.3 2021/01/17 キャッシュ方針を「あり」にした画像を再表示するとき、フレームを初期化するよう修正 2.1.2 2021/01/13 2.1.1の修正で一度消去したピクチャを再表示しようとするとエラーになる問題を修正 2.1.1 2020/12/11 キャッシュ方針を「キャッシュしない」以外に設定すると消去にエラーが起きる場合がある問題を修正 2.1.0 2020/11/11 APNGのアニメーションを停止、全停止できるスイッチを追加 2.0.1 2020/11/03 プラグイン上でapng画像の高さを正しく取得できるよう修正 2.0.0 2020/10/29 MZで動作するよう全面的に修正 1.6.0 2020/10/24 再生回数を指定したときに最初ではなく最後のフレームでアニメーションが止まる設定を追加 1.5.0 2020/10/17 サイドビューの敵キャラをapng化できるよう修正。機能が不完全であることに変わりはありません。 1.4.3 2020/03/17 ライブラリがpixi5.0対応によるバージョンアップで使用できなくなったのでヘルプの取得元を旧版に変更 1.4.2 2020/03/07 キャッシュしない設定のapngを繰り返し表示、削除し続けるとメモリリークが発生する問題を修正 1.4.1 2020/02/23 英語版のプラグインパラメータの記述が不足していたので修正 1.4.0 2020/02/01 アニメーションのループ回数を指定できる機能を追加 1.3.1 2019/12/31 画像を登録せずゲーム開始するとローディングが完了しない問題を修正 1.3.0 2019/12/31 APNGのキャッシュ機能を追加 1.2.1 2019/12/31 シーン追加画像でgifが表示されない問題を修正 1.2.0 2019/12/31 APNGのみツクール本体の暗号化機能に対応 1.1.0 2019/12/29 シーン追加画像の表示優先度を設定できる機能を追加 1.0.0 2019/12/27 初版 ---------------------------------------------------------------------------- [Blog] : https://triacontane.blogspot.jp/ [Twitter]: https://twitter.com/triacontane/ [GitHub] : https://github.com/triacontane/ =============================================================================*/ /*: * @plugindesc ApngSupportPlugin * @target MZ * @url https://github.com/triacontane/RPGMakerMV/tree/mz_master/ApngPicture.js * @base PluginCommonBase * @base PixiApngAndGif * @orderAfter PluginCommonBase * @orderAfter PixiApngAndGif * @orderBefore AltMenuScreen2MZ * @author triacontane * * @param PictureList * @desc List of picture images to be handled as APNG. * @default [] * @type struct[] * * @param EnemyList * @desc List of enemy images to be handled as APNG. * @default [] * @type struct[] * * @param SideEnemyList * @desc List of enemy images to be handled as APNG. * @default [] * @type struct[] * * @param SceneApngList * @desc This is a list of APNG to be displayed for each scene. * @default [] * @type struct[] * * @param DefaultLoopTimes * @desc The number of animation loops. It stops after looping the specified number of times. * @default 0 * @type number * * @param StopLastFrame * @desc The animation stops at the last frame, not at the beginning. * @default false * @type boolean * * @param AllStopSwitch * @desc All animations stop when the specified number switch is ON. * @default 0 * @type switch * * @help ApngPicture.js * * Enables handling of APNGs and GIF animations. * You can add APNGs to pictures, enemy characters, and any scene. * * After registering the file as an APNG from the parameters, * you can add You just need to designate them as pictures and enemy characters, * just like normal images. * * The following libraries are required for use. * https://github.com/sbfkcel/pixi-apngAndGif * * Download * https://github.com/sbfkcel/pixi-apngAndGif/blob/master/dist/PixiApngAndGif.js * * Attention! * It is not recommended to register a large number of images to * load the registered images at the start of the game. * In addition, loading large APNG images may slow down the display. * If the display is slow, please try GIF animation. * * If you want to use GIFs, please note that the editor will not recognize * files without a GIF extension. * Please enter the file name with the extension directly in the parameter. * You can also display the picture from a script or by using * Please use a dummy png file of the same name to specify it. * Also, GIFs are not subject to the editor's encryption feature. * * This plugin is released under the MIT License. */ /*:ja * @plugindesc APNGピクチャプラグイン * @target MZ * @url https://github.com/triacontane/RPGMakerMV/tree/mz_master/ApngPicture.js * @base PluginCommonBase * @base PixiApngAndGif * @orderAfter PluginCommonBase * @orderAfter PixiApngAndGif * @orderBefore AltMenuScreen2MZ * @author トリアコンタン * * @param PictureList * @text APNGのピクチャリスト * @desc APNGとして扱うピクチャ画像のリストです。GIFを指定したい場合は直接入力してください。 * @default [] * @type struct[] * * @param EnemyList * @text APNGの敵キャラリスト * @desc APNGとして扱う敵キャラ画像のリストです。GIFを指定したい場合は直接入力してください。 * @default [] * @type struct[] * * @param SideEnemyList * @text APNGのSV敵キャラリスト * @desc APNGとして扱うSV敵キャラ画像のリストです。サイドビューの画像を使用したい場合はこちらから登録してください。 * @default [] * @type struct[] * * @param SceneApngList * @text シーンAPNGのリスト * @desc シーンごとに表示するAPNGのリストです。GIFを指定したい場合は直接入力してください。 * @default [] * @type struct[] * * @param DefaultLoopTimes * @text デフォルトループ回数 * @desc アニメーションのループ回数です。指定した回数分ループ再生すると止まります。0を指定すると無限にアニメーションします。 * @default 0 * @type number * * @param StopLastFrame * @text 最終フレームで止める * @desc ループ回数が決まっているアニメーションを再生したとき最初ではなく最後のフレームでアニメーションが止まります。 * @default false * @type boolean * * @param AllStopSwitch * @text 全停止スイッチ * @desc 指定した番号スイッチがONのとき全てのアニメーションが停止します。 * @default 0 * @type switch * * @param FrameCount * @text 1セルのフレーム数 * @desc 設定すると1セルごとのフレーム数をゲーム側で固定にできます。 * @default 0 * @type number * * @param NoSmooth * @text ぼかし除去 * @desc 画像拡大時にぼかしが入らなくなります。 * @default false * @type boolean * * @help ApngPicture.js * * APNG、もしくはGIFアニメの取り扱いを可能にします。 * ピクチャ、敵キャラのほか任意のシーンにAPNGを追加表示できます。 * * パラメータからAPNGとしてファイルを登録したら後は * 通常の画像と同じようにピクチャや敵キャラとして指定するだけです。 * * 使用には以下のライブラリが必要です。 * https://github.com/sbfkcel/pixi-apngAndGif * * ライセンスを確認のうえ以下からダウンロードしてください。 * https://github.com/sbfkcel/pixi-apngAndGif/blob/master/dist/PixiApngAndGif.js * * 注意! * ゲーム開始時に登録した画像を読み込むため大量の画像を登録することは推奨しません。 * また、画像サイズの大きいAPNGを読み込むと、表示が遅くなる場合があります。 * 表示が遅い場合はGIFアニメもお試しください。 * * GIFを使用したい場合、拡張子がgifのファイルはエディタで認識されないので * ファイル名が同一のダミーpngファイルを用意して選択してください。 * ただし、GIFはエディタの暗号化機能の対象外となります。 * * 利用規約: * 作者に無断で改変、再配布が可能で、利用形態(商用、18禁利用等) * についても制限はありません。 * このプラグインはもうあなたのものです。 */ /*~struct~SceneApngRecord:ja * * @param SceneName * @text 対象シーン * @desc 画像を追加するシーンです。 * @type select * @default Scene_Title * @option タイトル * @value Scene_Title * @option ゲームオーバー * @value Scene_Gameover * @option バトル * @value Scene_Battle * @option メインメニュー * @value Scene_Menu * @option アイテム * @value Scene_Item * @option スキル * @value Scene_Skill * @option 装備 * @value Scene_Equip * @option ステータス * @value Scene_Status * @option オプション * @value Scene_Options * @option セーブ * @value Scene_Save * @option ロード * @value Scene_Load * @option ゲーム終了 * @value Scene_End * @option ショップ * @value Scene_Shop * @option 名前入力 * @value Scene_Name * @option デバッグ * @value Scene_Debug * @option サウンドテスト * @value Scene_SoundTest * @option 用語辞典 * @value Scene_Glossary * * @param FileName * @text ファイル名 * @desc 追加するAPNGのファイル名です。 * @default * @require 1 * @dir img/system/ * @type file * * @param Gif * @text GIFファイル * @desc 対象がGIFファイルの場合はONにしてください。ファイル名は拡張子なしのファイル名を指定してください。 * @default false * @type boolean * * @param CachePolicy * @text キャッシュ方針 * @desc 画像のキャッシュ方針です。大量にキャッシュするとメモリ使用量に影響が出る場合があります。 * @default 0 * @type select * @option キャッシュしない * @value 0 * @option 初回表示時にキャッシュ * @value 1 * @option ゲーム起動時にキャッシュ * @value 2 * * @param X * @text X座標 * @desc 追加するAPNGのX座標です。 * @default 0 * @type number * * @param Y * @text Y座標 * @desc 追加するAPNGのY座標です。 * @default 0 * @type number * * @param Origin * @text 原点 * @desc 追加するAPNGの原点です。 * @default 0 * @type select * @option 左上 * @value 0 * @option 中央 * @value 1 * * @param Priority * @text 優先度 * @desc 画像の表示優先度です。最背面は画面上に表示されないことが多いので通常の使用では推奨しません。 * @default 0 * @type select * @option 最前面 * @value 0 * @option ウィンドウの下 * @value 1 * @option 最背面 * @value 2 * * @param Switch * @text 出現条件スイッチ * @desc 指定したスイッチがONのときのみ表示されます。指定しない場合、常に表示されます。 * @default 0 * @type switch * * @param LoopTimes * @text ループ回数 * @desc アニメーションのループ回数です。0を指定するとデフォルト設定に従います。 * @default 0 * @type number * * @param StopSwitch * @text 停止スイッチ * @desc 指定した番号スイッチがONのときアニメーションが停止します。 * @default 0 * @type switch */ /*~struct~PictureApngRecord:ja * * @param FileName * @text ファイル名 * @desc 追加するAPNGのファイル名です。 * @default * @require 1 * @dir img/pictures/ * @type file * * @param Gif * @text GIFファイル * @desc 対象がGIFファイルの場合はONにしてください。ファイル名は拡張子なしのファイル名を指定してください。 * @default false * @type boolean * * @param CachePolicy * @text キャッシュ方針 * @desc 画像のキャッシュ方針です。大量にキャッシュするとメモリ使用量に影響が出る場合があります。 * @default 0 * @type select * @option キャッシュしない * @value 0 * @option 初回表示時にキャッシュ * @value 1 * @option ゲーム起動時にキャッシュ * @value 2 * * @param LoopTimes * @text ループ回数 * @desc アニメーションのループ回数です。0を指定するとデフォルト設定に従います。 * @default 0 * @type number * * @param StopSwitch * @text 停止スイッチ * @desc 指定した番号スイッチがONのときアニメーションが停止します。 * @default 0 * @type switch */ /*~struct~EnemyApngRecord:ja * * @param FileName * @text ファイル名 * @desc 追加するAPNGのファイル名です。 * @default * @require 1 * @dir img/enemies/ * @type file * * @param Gif * @text GIFファイル * @desc 対象がGIFファイルの場合はONにしてください。ファイル名は拡張子なしのファイル名を指定してください。 * @default false * @type boolean * * @param CachePolicy * @text キャッシュ方針 * @desc 画像のキャッシュ方針です。大量にキャッシュするとメモリ使用量に影響が出る場合があります。 * @default 0 * @type select * @option キャッシュしない * @value 0 * @option 初回表示時にキャッシュ * @value 1 * @option ゲーム起動時にキャッシュ * @value 2 * * @param LoopTimes * @text ループ回数 * @desc アニメーションのループ回数です。0を指定するとデフォルト設定に従います。 * @default 0 * @type number * * @param StopSwitch * @text 停止スイッチ * @desc 指定した番号スイッチがONのときアニメーションが停止します。 * @default 0 * @type switch */ /*~struct~SideEnemyApngRecord:ja * * @param FileName * @text ファイル名 * @desc 追加するAPNGのファイル名です。 * @default * @require 1 * @dir img/sv_enemies/ * @type file * * @param Gif * @text GIFファイル * @desc 対象がGIFファイルの場合はONにしてください。ファイル名は拡張子なしのファイル名を指定してください。 * @default false * @type boolean * * @param CachePolicy * @text キャッシュ方針 * @desc 画像のキャッシュ方針です。大量にキャッシュするとメモリ使用量に影響が出る場合があります。 * @default 0 * @type select * @option キャッシュしない * @value 0 * @option 初回表示時にキャッシュ * @value 1 * @option ゲーム起動時にキャッシュ * @value 2 * * @param LoopTimes * @text ループ回数 * @desc アニメーションのループ回数です。0を指定するとデフォルト設定に従います。 * @default 0 * @type number * * @param StopSwitch * @text 停止スイッチ * @desc 指定した番号スイッチがONのときアニメーションが停止します。 * @default 0 * @type switch */ /*~struct~SceneApngRecord: * * @param SceneName * @desc Target Scene * @type select * @default Scene_Title * @option Scene_Title * @option Scene_Gameover * @option Scene_Battle * @option Scene_Menu * @option Scene_Item * @option Scene_Skill * @option Scene_Equip * @option Scene_Status * @option Scene_Options * @option Scene_Save * @option Scene_Load * @option Scene_End * @option Scene_Shop * @option Scene_Name * @option Scene_Debug * @option Scene_SoundTest * @option Scene_Glossary * * @param FileName * @desc File name of apng * @default * @require 1 * @dir img/system/ * @type file * * @param Gif * @text * @desc * @default false * @type boolean * * @param CachePolicy * @desc Cache policy * @default 0 * @type select * @option None * @value 0 * @option Cache on first display * @value 1 * @option Cache at game start * @value 2 * * @param X * @desc X of apng * @default 0 * @type number * * @param Y * @desc Y of apng * @default 0 * @type number * * @param Origin * @desc Origin of apng * @default 0 * @type select * @option Upper left * @value 0 * @option Center * @value 1 * * @param Priority * @desc Priority of apng * @default 0 * @type select * @option Front * @value 0 * @option Under window * @value 1 * @option Back * @value 2 * * @param Switch * @desc Displayed only when the specified switch is ON * @default 0 * @type switch * * @param LoopTimes * @desc The number of animation loops. Specifying 0 follows the default setting. * @default 0 * @type number * * @param StopSwitch * @desc The animation stops when the specified number switch is turned on. * @default 0 * @type switch */ /*~struct~PictureApngRecord: * * @param FileName * @desc File name of apng * @default * @require 1 * @dir img/pictures/ * @type file * * @param Gif * @text * @desc * @default false * @type boolean * * @param CachePolicy * @desc Cache policy * @default 0 * @type select * @option None * @value 0 * @option Cache on first display * @value 1 * @option Cache at game start * @value 2 * * @param LoopTimes * @desc The number of animation loops. Specifying 0 follows the default setting. * @default 0 * @type number * * @param StopSwitch * @desc The animation stops when the specified number switch is turned on. * @default 0 * @type switch */ /*~struct~EnemyApngRecord: * * @param FileName * @desc File name of apng * @default * @require 1 * @dir img/enemies/ * @type file * * @param Gif * @text * @desc * @default false * @type boolean * * @param CachePolicy * @desc Cache policy * @default 0 * @type select * @option None * @value 0 * @option Cache on first display * @value 1 * @option Cache at game start * @value 2 * * @param LoopTimes * @desc The number of animation loops. Specifying 0 follows the default setting. * @default 0 * @type number * * @param StopSwitch * @desc The animation stops when the specified number switch is turned on. * @default 0 * @type switch */ /*~struct~SideEnemyApngRecord: * * @param FileName * @desc File name of apng * @default * @require 1 * @dir img/sv_enemies/ * @type file * * @param Gif * @text * @desc * @default false * @type boolean * * @param CachePolicy * @desc Cache policy * @default 0 * @type select * @option None * @value 0 * @option Cache on first display * @value 1 * @option Cache at game start * @value 2 * * @param LoopTimes * @desc The number of animation loops. Specifying 0 follows the default setting. * @default 0 * @type number * * @param StopSwitch * @desc The animation stops when the specified number switch is turned on. * @default 0 * @type switch */ (function() { 'use strict'; const script = document.currentScript; const param = PluginManagerEx.createParameter(script); /** * ApngLoader * APNGもしくはGIF画像を読み込んで保持します。 */ class ApngLoader { constructor(folder, paramList) { this._folder = folder; this._fileHash = {}; this._cachePolicy = {}; this._options = {}; this._paramList = paramList; if (this._paramList && this._paramList.length > 0) { this.addAllImage(); } this._spriteCache = {}; } addAllImage() { const option = this.getLoadOption(); this._paramList.forEach(function(item) { this.addImage(item, option); }, this); PIXI.Loader.shared.onComplete.add(this.cacheStartup.bind(this)); ApngLoader.startLoading(); } addImage(item, option) { const name = String(item.FileName) || ''; const ext = this.findExt(item); const path = name.match(/http:/) ? name : `img/${this._folder}/${name}.${ext}`; if (!this._fileHash.hasOwnProperty(name)) { this._fileHash[name] = ApngLoader.convertDecryptExt(path); this._cachePolicy[name] = item.CachePolicy; this._options[name] = item; PIXI.Loader.shared.add(path, option); } } findExt(item) { if (item.Gif) { return 'gif' } else { return Utils.hasEncryptedImages() ? 'png_' : 'png'; } }; getLoadOption() { return { loadType : PIXI.LoaderResource.LOAD_TYPE.XHR, xhrType : PIXI.LoaderResource.XHR_RESPONSE_TYPE.BUFFER, crossOrigin: '' } } createSprite(name) { if (!this.isApng(name)) { return null; } if (this._isNeedCache(name)) { if (this._spriteCache[name]) { return this._spriteCache[name]; } const sprite = this._createPixiApngAndGif(name); this._spriteCache[name] = sprite; return sprite; } else { return this._createPixiApngAndGif(name); } } _createPixiApngAndGif(name) { const pixiApng = new PixiApngAndGif(this._fileHash[name], ApngLoader._resource); const loopCount = this._options[name].LoopTimes || param.DefaultLoopTimes; if (loopCount > 0) { pixiApng.play(loopCount); } const sprite = pixiApng.sprite; sprite.pixiApng = pixiApng; sprite.pixiApngOption = this._options[name]; if (param.NoSmooth) { pixiApng.textures.forEach(texture => texture.baseTexture.scaleMode = PIXI.SCALE_MODES.NEAREST); } return sprite; } _isNeedCache(name) { return this._cachePolicy[name] > 0; } isApng(name) { return !!this._fileHash[name]; } cacheStartup() { Object.keys(this._cachePolicy).forEach(function(name) { if (this._cachePolicy[name] === 2) { this.createSprite(name) } }, this); } static startLoading() { this._loading = true; }; static onLoadResource(progress, resource) { this._resource = resource; Object.keys(this._resource).forEach(function(key) { if (this._resource[key].extension === 'png_') { ApngLoader.decryptResource(key); } }, this); } static decryptResource(key) { const resource = this._resource[key]; resource.data = Utils.decryptArrayBuffer(resource.data); const newKey = ApngLoader.convertDecryptExt(key); resource.name = newKey; resource.url = newKey; resource.extension = 'png'; this._resource[newKey] = resource; delete this._resource[key]; }; static isReady() { return !!this._resource || !this._loading; } static convertDecryptExt(key) { return key.replace(/\.png_$/, '.png'); } } ApngLoader._resource = null; const _Scene_Boot_isReady = Scene_Boot.prototype.isReady; Scene_Boot.prototype.isReady = function() { const result = _Scene_Boot_isReady.apply(this, arguments); if (result) { SceneManager.setupApngLoaderIfNeed(); } return result && ApngLoader.isReady(); }; const _Scene_Base_create = Scene_Base.prototype.create; Scene_Base.prototype.create = function() { _Scene_Base_create.apply(this, arguments); this.createSceneApng(); }; Scene_Base.prototype.createSceneApng = function() { this._apngList = this.findSceneApngList().map(function(item) { return new SpriteSceneApng(item); }, this); }; const _Scene_Base_terminate = Scene_Base.prototype.terminate; Scene_Base.prototype.terminate = function() { _Scene_Base_terminate.apply(this, arguments); this.destroySceneApng(); if (this._spriteset) { this._spriteset.destroyApngPicture(); } }; Scene_Base.prototype.destroySceneApng = function() { this._apngList.forEach(function(sprite) { sprite.destroyApngIfNeed(); }) }; const _Scene_Base_start = Scene_Base.prototype.start; Scene_Base.prototype.start = function() { _Scene_Base_start.apply(this, arguments); this.setApngPriority(); }; Scene_Base.prototype.setApngPriority = function() { let windowLayerIndex = this._windowLayer ? this.getChildIndex(this._windowLayer) : 0; this._apngList.forEach(function(sprite) { switch (sprite.getPriority()) { case 0: this.addChild(sprite); break; case 1: this.addChildAt(sprite, windowLayerIndex); windowLayerIndex++; break; default: this.addChildAt(sprite, 0); } }, this); }; Scene_Base.prototype.findSceneApngList = function() { const currentSceneName = PluginManagerEx.findClassName(this); return (param.SceneApngList || []).filter(function(data) { return data.SceneName === currentSceneName; }, this); }; Spriteset_Base.prototype.destroyApngPicture = function() { this.destroyApngPictureContainer(this._pictureContainer); // for PicturePriorityCustomize.js this.destroyApngPictureContainer(this._pictureContainerLower); this.destroyApngPictureContainer(this._pictureContainerMiddle); this.destroyApngPictureContainer(this._pictureContainerUpper); }; Spriteset_Base.prototype.destroyApngPictureContainer = function(container) { if (!container) { return; } container.children.forEach(function(sprite) { if (sprite.destroyApngIfNeed) { sprite.destroyApngIfNeed(); } }); }; Spriteset_Battle.prototype.destroyApngPicture = function() { Spriteset_Base.prototype.destroyApngPicture.call(this); this._enemySprites.forEach(function(sprite) { sprite.destroyApngIfNeed(); }); }; /** * SceneManager * APNGのローダを管理します。 */ SceneManager.setupApngLoaderIfNeed = function() { if (this._apngLoaderPicture) { return; } PIXI.Loader.shared.onComplete.add(ApngLoader.onLoadResource.bind(ApngLoader)); this._apngLoaderPicture = new ApngLoader('pictures', param.PictureList); this._apngLoaderEnemy = new ApngLoader('enemies', param.EnemyList); this._apngLoaderSideEnemy = new ApngLoader('sv_enemies', param.SideEnemyList); this._apngLoaderSystem = new ApngLoader('system', param.SceneApngList); PIXI.Loader.shared.load(); }; SceneManager.tryLoadApngPicture = function(name) { return this._apngLoaderPicture.createSprite(name); }; SceneManager.tryLoadApngEnemy = function(name) { return this._apngLoaderEnemy.createSprite(name); }; SceneManager.tryLoadApngSideEnemy = function(name) { return this._apngLoaderSideEnemy.createSprite(name); }; SceneManager.tryLoadApngSystem = function(name) { return this._apngLoaderSystem.createSprite(name); }; /** * Sprite * APNGの読み込み処理を追加します。 */ Sprite.prototype.addApngChild = function(name) { if (this._apngSprite) { this.destroyApngIfNeed(); } this._apngSprite = this.loadApngSprite(name); if (this._apngSprite) { if (this.isApngCache()) { this._apngSprite.pixiApng.jumpToFrame(0); this._apngSprite.pixiApng.play(); } this.addChild(this._apngSprite); const frame = this._apngSprite.pixiApng.textures[0]._frame; this.bitmap = new Bitmap(frame.width, frame.height); this.updateApngAnchor(); this.updateApngBlendMode(); } this._apngLoopCount = 1; this._apngLoopFrame = 0; }; Sprite.prototype.destroyApngIfNeed = function() { if (this._apngSprite) { if (!this.isApngCache()) { this.destroyApng(); } else { this.removeApng(); } } }; Sprite.prototype.destroyApng = function() { const pixiApng = this._apngSprite.pixiApng; if (pixiApng) { pixiApng.textures.forEach(function(texture) { texture.baseTexture.destroy(); texture.destroy(); }); pixiApng.stop(); } this.removeApng(); }; Sprite.prototype.removeApng = function() { this.removeChild(this._apngSprite); this._apngSprite = null; }; Sprite.prototype.isApngCache = function() { return this._apngSprite.pixiApngOption.CachePolicy !== 0; }; Sprite.prototype.loadApngSprite = function() { return null; }; Sprite.prototype.updateApngAnchor = function() { if (this._apngSprite) { this._apngSprite.anchor.x = this.anchor.x; this._apngSprite.anchor.y = this.anchor.y; } }; Sprite.prototype.updateApngBlendMode = function() { if (this._apngSprite) { this._apngSprite.blendMode = this.blendMode; } }; const _Sprite_update = Sprite.prototype.update; Sprite.prototype.update = function() { _Sprite_update.apply(this, arguments); if (this._apngSprite) { if (param.FrameCount > 0 && !this._apngSpritePause) { this.updateApngFrame(); } this.updateApngSwitchStop(); this.updateApngFrameStop(); } }; Sprite.prototype.updateApngFrame = function() { if (this._apngStop) { return; } const frameLength = this._apngSprite.pixiApng.getFramesLength(); const frame = Math.floor(Graphics.frameCount / param.FrameCount) % frameLength; this._apngSprite.pixiApng.jumpToFrame(frame); }; Sprite.prototype.updateApngFrameStop = function() { if (!param.StopLastFrame && !param.FrameCount) { return; } const frame = this._apngSprite.pixiApng.__status.frame; if (frame < this._apngLoopFrame) { this._apngLoopCount++; } this._apngLoopFrame = frame; const loopLimit = this.getLoopTimes(); if (loopLimit <= 0) { return; } const frameLength = this._apngSprite.pixiApng.getFramesLength(); if (loopLimit <= this._apngLoopCount && frameLength <= frame + 1) { this._apngSprite.pixiApng.stop(); this._apngStop = true; } }; Sprite.prototype.updateApngSwitchStop = function() { if ($gameSwitches.value(this.getStopSwitch()) || $gameSwitches.value(param.AllStopSwitch)) { this._apngSprite.pixiApng.stop(); this._apngSpritePause = true; } else if (this._apngSpritePause) { this._apngSprite.pixiApng.play(); this._apngSpritePause = false; } }; Sprite.prototype.getLoopTimes = function() { return this._apngSprite.pixiApngOption.LoopTimes || param.DefaultLoopTimes; }; Sprite.prototype.getStopSwitch = function() { return this._apngSprite.pixiApngOption.StopSwitch; }; Sprite.prototype.isGif = function() { return this._apngSprite && this._apngSprite.pixiApngOption.Gif; }; /** * Sprite_Picture * APNGとして登録されているピクチャの読み込みを追加します。 */ const _Sprite_Picture_loadBitmap = Sprite_Picture.prototype.loadBitmap; Sprite_Picture.prototype.loadBitmap = function() { _Sprite_Picture_loadBitmap.apply(this, arguments); this.addApngChild(this._pictureName); }; Sprite_Picture.prototype.loadApngSprite = function(name) { return SceneManager.tryLoadApngPicture(name); }; const _Sprite_Picture_updateOrigin = Sprite_Picture.prototype.updateOrigin; Sprite_Picture.prototype.updateOrigin = function() { _Sprite_Picture_updateOrigin.apply(this, arguments); this.updateApngAnchor(); }; const _Sprite_Picture_updateOther = Sprite_Picture.prototype.updateOther; Sprite_Picture.prototype.updateOther = function() { _Sprite_Picture_updateOther.apply(this, arguments); this.updateApngBlendMode(); }; const _Sprite_Picture_updateBitmap =Sprite_Picture.prototype.updateBitmap; Sprite_Picture.prototype.updateBitmap = function() { _Sprite_Picture_updateBitmap.apply(this, arguments); const picture = this.picture(); if (!picture && this._apngSprite) { this.destroyApngIfNeed(); } }; /** * Sprite_Enemy * APNGとして登録されている敵キャラの読み込みを追加します。 */ const _Sprite_Enemy_loadBitmap = Sprite_Enemy.prototype.loadBitmap; Sprite_Enemy.prototype.loadBitmap = function(name, hue) { _Sprite_Enemy_loadBitmap.apply(this, arguments); this.addApngChild(name); }; Sprite_Enemy.prototype.loadApngSprite = function(name) { if ($gameSystem.isSideView()) { return SceneManager.tryLoadApngSideEnemy(name); } else { return SceneManager.tryLoadApngEnemy(name); } }; /** * SpriteSceneApng * シーンに追加表示するAPNGスプライトです。 */ class SpriteSceneApng extends Sprite { constructor(item) { super(); this.setup(item) } setup(item) { this.addApngChild(item.FileName); this.x = item.X; this.y = item.Y; if (item.Origin === 1) { this.anchor.x = 0.5; this.anchor.y = 0.5; } this._switch = item.Switch; this._priority = item.Priority; this.updateVisibly(); } loadApngSprite(name) { return SceneManager.tryLoadApngSystem(name.replace(/\..*/, '')); } update() { super.update(); this.updateVisibly(); } updateVisibly() { this.visible = this.isValid(); } isValid() { return !this._switch || $gameSwitches.value(this._switch); } getPriority() { return this._priority; } } })();