JavaScriptでjsgifを使ってアニメーションGIFを動的生成する

先日の記事、「JavaScriptでcanvasを録画して動画ファイルに保存する方法」では複数の画像ファイルからwebm形式の動画を生成しましたが、Twitter等のSNSではwebm形式に対応していないようなので、今回は汎用性の高いアニメーションGIFを作ってみようと思います。

必要なライブラリ

canvasに描いた画像をアニメーションGIFにしてくれるjsgifというオープンソースのライブラリがあるので、ありがたく使わせて頂きましょう。

https://github.com/antimatter15/jsgif

リンク先のZIP内にある下記4つのJavaScriptファイルを読み込んで使う感じです。

  • LZWEncoder.js
  • NeuQuant.js
  • GIFEncoder.js
  • b64.js

HTMLの例

<script src="/js/b64.js"></script>
<script src="/js/LZWEncoder.js"></script>
<script src="/js/NeuQuant.js"></script>
<script src="/js/GIFEncoder.js"></script>

<div id="anime">
<img src="[画像ファイル01]" width="80" />
<img src="[画像ファイル02]" width="80" />
<img src="[画像ファイル03]" width="80" />
<img src="[画像ファイル04]" width="80" />
<img src="[画像ファイル05]" width="80" />
<img src="[画像ファイル06]" width="80" />
<img src="[画像ファイル07]" width="80" />
<img src="[画像ファイル08]" width="80" />
<img src="[画像ファイル09]" width="80" />
<img src="[画像ファイル10]" width="80" />
<img src="[画像ファイル11]" width="80" />
<img src="[画像ファイル12]" width="80" />
<img src="[画像ファイル13]" width="80" />
</div>

<p>速度:<input type="range" id="anime_speed" value="500" min="100" max="500"></p>
<p><button onclick="createGIF();">アニメGIFを作成する</button></p>

<canvas id="canvas" style="display:none;"></canvas>

<p><img id="anime_gif" src=""></p>

<p><button id="download" onclick="downloadGIF();" style="display:none;">ダウンロード</button></p>

解説

最初の4つのスクリプト読み込みはjsgif用です。それぞれのjsファイルを置いたパスに適宜書き換えてください。

次にGIFアニメーションにする画像ファイルを並べます。13枚という数に特に意味はありません。以前撮った写真を使いまわしているだけですw

任意の画像を選択してアニメーションGIFを作りたい場合はこの部分を「JavaScriptでFileReaderを使った複数画像のプレビュー例」あたりを参考に書き換えると良いでしょう。

input type=”range”の速度調整は「JavaScriptで画像を連続表示してコマ撮りムービー風の表示をしてみる」で使ったスライダーですが、そのときと違ってCSSによる左右反転はしていません。というのも、Chromeだと良い感じでしたが、Edgeだと反転させただけでは見た目がイマイチだったんですよね。

それなら変に小細工せずにそのまま使っちゃおうかな、と。

canvasはアニメーションGIFの作成時の一時的な保管場所として使うだけなのでdisplay:none;で非表示にしてあります。

また、imgタグのID=”anime_gif”には完成したアニメーションGIFが入る予定なので、空の画像 ()を入れています。このエンコード、わざわざファイルを用意しなくても良いので地味に便利。

ボタン類はJavaScriptのほうで解説しましょう。

JavaScriptの例

<script>
var encoder;

function createGIF()
{
	//canvasの取得
	var canvas = document.getElementById('canvas');
	var ctx = canvas.getContext('2d');
	//GIFEncoderの初期処理
	encoder = new GIFEncoder();
	encoder.setRepeat(0); //繰り返し回数 0=無限ループ
	encoder.setDelay(document.getElementById('anime_speed').value); //1コマあたりの待機秒数(ミリ秒)
	encoder.start();
	//画像ファイル一覧を取得
	frames = document.getElementById('anime').getElementsByTagName('img');
	//canvasのサイズを1枚目のコマに合わせる
	canvas.width = frames[0].naturalWidth;
	canvas.height = frames[0].naturalHeight;
	//全ての画像をcanvasへ描画
	for (var frame_no = 0; frame_no < frames.length; frame_no++) {
		ctx.drawImage(frames[frame_no], 0, 0);
		encoder.addFrame(ctx); //コマ追加
	}
	//アニメGIFの生成
	encoder.finish();
	document.getElementById('anime_gif').src = 'data:image/gif;base64,' + encode64(encoder.stream().getData());
	//ダウンロードボタンを表示
	document.getElementById('download').style.display = 'block';
}
function downloadGIF()
{
	encoder.download("download.gif");
}
</script>

解説

[アニメGIFを作成する]というボタンから呼ばれるcreateGIF関数と、[ダウンロード]ボタンから呼ばれるdownloadGIF関数を用意しました。

createGIFでは、

  • canvasの取得
  • GIFEncoderの初期処理(速度とか繰り返し回数とか)
  • 画像ファイルの一覧取得
  • canvasの大きさ調整
  • 全画像をcanvasへ書きつつ、GIFEncoderへコマ追加
  • 最後に生成されたデータをimgタグのsrcへセット
  • [ダウンロード]ボタンの表示

というようなことを実行しています。

downloadGIFでは、GIFEncoderのdownload関数を呼び出しているだけですね。そのために変数encoderをグローバル変数として宣言しています。

自前でファイルを作りたい人は下記のように encoder.stream().bin からblobデータを生成して、アンカータグのURLに埋め込むなんて方法もアリです。

	var bin = new Uint8Array(encoder.stream().bin);
	var blob = new Blob([bin.buffer], {type: 'image/gif'});
	var dataUrl = window.URL.createObjectURL(blob);
	var anchor = document.createElement('a');
	anchor.download = 'download.gif';
	anchor.href = dataUrl;

実行結果

速度:

[アニメGIFを作成する]ボタンをクリックするといつものクマちゃんのアニメーションGIFが生成されて表示されるハズです。
(PCスペック次第ですが、生成まで少し時間がかかるかも)

一応、ChromeとEdgeで動作確認済み。

アニメーションGIFって256色しか使えないからかなり画質が劣化するだろうと思いましたが、意外と綺麗ですね。

adsbygoogle

フォロー