LoginSignup
7
1

More than 1 year has passed since last update.

WebAssembly 版 ImageMagick 〜 magick-wasm の紹介

Last updated at Posted at 2023-01-15

はじめに

サーバでなくブラウザ側で画像処理が動く WebAssembly 版 ImageMagick 〜 magick-wasm の使い方と 2023年初頭での状況報告です。

magick-wasm 本体にまだブラックボックス要素があるので、本体自体の解説はせず、今回は、実際に magick-wasm を使う事ができた。でも機能的に物足りなく感じた。それだけの解説エントリです。

4月20日に開発者のアナウンスにて、次リリース(恐らく 0.0.20)から wasm を js ファイルに含めないとの宣言がありました。
https://twitter.com/MagickNET/status/1648982963491618817
この記事で説明した手順では動かせなくなるかもしれません。少なくとも index.mjs を import するだけでは駄目になりそうです。

書いたきっかけ

magick-wasm 使いたいけど分からないと呟いてたら作者からリプライを貰ったので、自分で解説書くしかないなと。。٩( 'ω' )و

まとめ

  • 一言でいうとアルファ版相当。絶賛開発中。バージョン 0.0.18 だし
  • 入出力できる画像形式は充実している一方、画像処理の対応は控えめ
  • 回転や輝度/色フィルター、合成も出来る。文字やベクター描画系は無し
  • マニュアルの代わりに TypeScript + Vue.js サンプルが用意されている
  • API は見た感じ Magick++ (ImageMagick の C++ API)相当。 .NET 版と同じ?
  • 試したら、直接 JavaScript からも使えた
  • npm install @imagemagick/magick-wasm でインストール可能
  • @imagemagick/magick-wasm パッケージのビルド方法は不明。(権限の問題?)
  • 読み込む magick-wasm の MJS ファイルは今のところ 2MB 程度。思ったより小さい

以下のサイトにデモがあって気軽に試せます。ソースコードもあります。

magick-wasm 関連レポジトリ

主に、以下の4つです。

本体をいじりたい場合は別ですが、TS/JS ライブラリとして使うだけであれば magick-wasm-docs を見ると良いです。

Magick.Native

ImageMagick を色んな環境でビルドする為のレポジトリで、主に .NET ライブラリの開発に使っているようで、ここの移植対象として wasm が追加された形です。ちなみに Docker ゲスト環境でビルドします。いまどきですね。

magick-wasm

こちらが Magick.Native の wasm コードを取り込む、TS/JS ラッパーです。

なお、ビルドの仕方は分かりません。(@dlemstra さんから権限をもらう必要ある?)

npm install で @dlemstra/magick-native パッケージの D/L が 401 Unauthorized エラーで止まります。(2023/1/7現在)
ちなみに、@dlemstra/magick-native は npm サーバでなく github にホストしてるので、何も設定しないと E404 、設定すると E401 エラーになります。

出来れば改造してみたいのですが、とりあえず npm install で得られるファイルを使いましょう。

% npm install @imagemagick/magick-wasm
% ls -lR node_modules/\@imagemagick/magick-wasm
total 400
-rw-r--r--  1 yoya  staff   11358  1  9 18:23 LICENSE
-rw-r--r--  1 yoya  staff  183236  1  9 18:23 NOTICE
-rw-r--r--  1 yoya  staff    1979  1  9 18:23 README.md
drwxr-xr-x  5 yoya  staff     160  1  9 18:23 dist
-rw-r--r--  1 yoya  staff    1929  1  9 18:23 package.json

node_modules/@imagemagick/magick-wasm/dist:
total 87056
-rw-r--r--  1 yoya  staff     47280  1  9 18:23 index.d.ts
-rw-r--r--  1 yoya  staff  22306698  1  9 18:23 index.mjs
-rw-r--r--  1 yoya  staff  22213834  1  9 18:23 index.umd.js

magick-wasm-docs

URL がマニュアルっぽいですが、今のところデモサンプルが動作します。

こちらは普通にビルドして試せます。README の通りです。

% npm install
% npm run serve

とすれば、ローカルでサーバが立ち上がって試せますし。

% npm install
% npm run build

こちらで build すれば dist に結果ファイルが生成されて、適当なサーバに upload すれば Web サイトとして動きます。

何もいじらずに npm run build した場合は、上記のデモサイトと同じページが作れます。実際に試した結果が以下のサイトです。

magick-wasm-docs は https://<host名>/magick-wasm-docs/ 決め打ちの設定なので、上記のように /magick-wasm/magick-wasm-docs/ のパスにしたい場合は、vue.config.js の publicPath を変更します。

vue.config.js
module.exports = {
-    publicPath: '/magick-wasm-docs/',
+    publicPath: '/magick-wasm/magick-wasm-docs/',

利用する

magick-wasm-docs の改造

ビルドして試せるという事は、気軽にコードを改造できるという事です。

magick-wasm-docs を改造していくと理解が早いと思います。

今回は折角なので例のレトロ変換を目指しましたが、magick-wasm に機能が足りなくて諦めました。 -ordered-dither と -map 相当の機能が不足してるんですよね。以下のが結果です。

VueJS の作法に従って、以下のように改造して、デモページを増やします。

  • src/routes.ts で retro の URL パスを追加
  • views/demos/Retro/*.vue に retro ページ本体と処理。
src/views/demos/Retro/Retro.vue
import { ImageMagick } from '@imagemagick/magick-wasm/image-magick'

const canvas = document.getElementById('canvasId');

ImageMagick.readFromCanvas(canvas, (image) => {
    image.resize(320, 240);
    image.sigmoidalContrast(0, 8, 0.5);  // 0: _sharpen flag
    image.modulate(new Percentage(100), new Percentage(200), 
                   new Percentage(100));
    // -ordered-dither 4x4,8,8,8 
    // +dither -map palette.png
    image.filterType = FilterType.Point;
    image.resize(new MagickGeometry('200%'));
    image.writeToCanvas(canvas);
})

差分はこちらにまとめました。

JavaScript での利用

ちなみに、JavaScript から直接使えます。npm install で生成される node_modules/@imagemagick/magick-wasm/dist/index.mjs ファイルを読み込むだけです。

mjs 拡張子を import する際にブラウザが content-type チェックを行う為、HTTP サーバに mjs の設定を入れる必要があります。それができない場合は、拡張子を .mjs から .js に変えても import できますが、(少なくともファイル管理的に)お行儀が悪そうです。

例えば、回転させるだけならこれにて出来ます。

% npm install @imagemagick/magick-wasm
% cp node_modules/\@imagemagick/magick-wasm/dist/index.mjs \
     lib/magick-wasm/0.0.17/index.mjs
rotate.html
<html>
  <body>
    <canvas id="canvasId"> </canvas>
    <script type="module">
      import { initializeImageMagick,
               ImageMagick } from './lib/magick-wasm/0.0.17/index.mjs';
      const canvas = document.getElementById('canvasId');
      initializeImageMagick().then(() => {
          ImageMagick.read("logo:", (image) => image.writeToCanvas(canvas));
          ImageMagick.readFromCanvas(canvas, (image) => {
              const degrees = 90;
              image.rotate(degrees);
              image.writeToCanvas(canvas);
          })
      }).catch(err => {
          console.error(err);
      });
    </script>
  </body>
</html>

以下のところに実際に動作します。

rotate.html の表示
image.png

magick-wasm の開発状況

ImageMagick の基礎的な部分はだいぶ実装が進んでいるので、API を追加すれば色々動きそうに見えます。デモでも試せるように、リサイズやフィルタ周りはだいぶ対応しているようです。

あと、画像合成も動きます。 (2つ画像を読み込む良い方法が分からないので、readFromCanvas は二度入れ子に使ってます)

composite.html
      initializeImageMagick().then(() => {
          ImageMagick.read("logo:", (image) => image.writeToCanvas(canvas1));
          ImageMagick.read("wizard:", (image) => image.writeToCanvas(canvas2));
          ImageMagick.readFromCanvas(canvas1, (image1) => {
              ImageMagick.readFromCanvas(canvas2, (image2) => {
                  image1.composite(image2, CompositeOperator.Darken);
                  image1.writeToCanvas(canvas3);
              })
          })
      }).catch(err => {
          console.error(err);
      });
composite.html の表示
スクリーンショット 2023-01-16 3.31.23.png

描画系 API は文字周りは用意されているのですが、試した感じ今のところ動く気配がないです。おそらく開発中かも?

備考

WebAssembly 版 ImageMagick

実は WebAssembly 版 ImageMagick はずっと前から色んな人が公開しているのですが、今回紹介したパッケージ以外は、いずれもメンテナンスが止まってます。

ちなみに今回紹介したレポジトリの所有者 dlemstra さんは ImageMagick の .NET 版開発者で、ImageMagick のコア開発者としても、原作者 cristy さんと並んで活躍してる方です。

7
1
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
7
1