【Scratch拡張機能をJavascriptで自作】SVG画像からスプライトをアニメーション的に動かす拡張機能の作成方法


※ 当ページには【広告/PR】を含む場合があります。
2021/02/17
【Scratch拡張機能をJavascriptで自作】RPG風の吹き出しを表示する拡張機能
【SvelteでRPGゲーム開発】マップ背景を表示する〜基本編

Scratchプログラミングにおいてスプライトを歩かせるように見せるには、コスチュームを一定時間ごとに切り替えてたり、位置が変わる度に進行方向に合わせたりするようにします。

このような基本的なスプライトのアニメーションの付け方は以前の記事でも取り上げていました。

合同会社タコスキングダム|タコキンのPスクール
【Scratch入門】はじめてのScratchアプリ作成〜スプライトを自由に歩かせてみよう

Scratchで画面内を自由に動き回るスプライトを作成します。

合同会社タコスキングダム|タコキンのPスクール
【Scratch入門】コスチュームを変えてスプライトのアニメーション効果を付ける

Scratchでスプライトを移動させる時にコスチューム番号をずらしてアニメーションで動かす方法を解説します。

今回は開発者向けの話題になってしまいますが、より実用性の高いSVGの使い方として
clipPathuseを用いたアニメーションをScratchアプリに組み込める拡張機能の自作方法を紹介します。


合同会社タコスキングダム|タコキンのPスクール【Html&Cssアプリ】Html/JavascriptとSassを使った四択クイズゲームの作り方

すぐに作れる ずっと使える Inkscapeのすべてが身に付く本

無償で試せる素材を提供されているサイト

RPGで使える動きをひとまとめにしたアニメーション画像セットを自作するのは中々骨の折れる作業です。

今回のように動きをプログラミングしたいだけであれば、わざわざアニメーション画像を用意しなくても無償で公開されているサイトがあります。

以降ではまず参考までに優良な無料で利用できるサイトを紹介させていただきます。

ぴぽや倉庫

ぴぽや倉庫さんは、Scratchのビギナーにはとても有り難い良質な素材を公開されている国内では有名な老舗サイトです。

なお特定のライセンスのない無償の素材とはいえ個人で使う分には問題はありませんが、もしそれらを組み込んだスクラッチアプリを公開または商用目的で利用したい場合などは、利用規約に従ってください。

Superpowers Asset Packs

Superpowers Asset Packs

Github上におかれたフリーの素材集で、もうあまりメンテナンスや更新などはされていないようですが、Scratchゲームを試したいだけであれば十分使える画像がたくさん集められています。

またライセンスも縛りの最も緩い(というか無いに等しい)CC0ですので、画像にどのような操作を加えても著作権利などを深くは考えなくてもよいのが使う側としては有り難いことでもあります。

もちろんどのような権利も持たない(=公共の財産)ので、期限切れで消滅しても一切の責任なども生じないですので誰にも不満は言えないのがこのCC0の唯一のライセンス条件と言えます。

利用される場合には、
公共の財産ですのできちんとルールを守って大切に使っていきましょう。


合同会社タコスキングダム|タコキンのPスクール【Html&Cssアプリ】Html/JavascriptとSassを使った四択クイズゲームの作り方

すぐに作れる ずっと使える Inkscapeのすべてが身に付く本

テクニックのキモ ~ SVG画像を分割する

こちらのサイトが参考になります。

use要素とclipPath要素とを組み合わせることで単一の画像を複数のパーツに分割することができます。

これを使い、一つの画像から効率的にパラパラアニメーションのようなエフェクトをSVGのみで作り出すことが可能になります。

テスト画像としては
Superpowers Asset PacksのNinja Adventure2の17.pngを特に理由はありませんが利用させていただくとします。

では
svgお試しサイトで以下のsvgインラインコードを表示してみてください。

            
            <svg xmlns="http://www.w3.org/2000/svg" width="200" height="300" version="1.1" viewBox="0 0 100 150">
    <defs>
        <g id="baseImage">
            <image width="64" height="112" xlink:href=""/>
        </g>
        <clipPath id="anim1">
            <rect width="16" height="16"/>
        </clipPath>
        <clipPath id="anim2">
            <rect y="17" width="16" height="16"/>
        </clipPath>
        <clipPath id="anim3">
            <rect y="33" width="16" height="16"/>
        </clipPath>
        <clipPath id="anim4">
            <rect y="49" width="16" height="16"/>
        </clipPath>
    </defs>
    <use xlink:href="#baseImage" clip-path="url(#anim1)" x="0" y="0"></use>
    <use xlink:href="#baseImage" clip-path="url(#anim2)" x="24" y="24"></use>
    <use xlink:href="#baseImage" clip-path="url(#anim3)" x="48" y="48"></use>
    <use xlink:href="#baseImage" clip-path="url(#anim4)" x="72" y="72"></use>
</svg>
        
このsvgコードは以下のようになります。

合同会社タコスキングダム|タコキンのPスクール

今回は意図的に一つの画像から4つの画像を意図的に抽出し、少しずらして表示しているだけですが、
clipPathで画像の一部を新しい画像として登録して、useでどの画像を表示させるかを一定のタイミングで決定させるだけでアニメーションとして仕上げるように出来ます。

なお、分割後の画像のサイズ調整などはSVGの別のテクニックが必要になります。

前回以下の記事にてSVG基本的な使いこなしをまとめていたので、画像の調整方法が気になる方はそちらの内容でチェックしてください。

合同会社タコスキングダム|タコキンのPスクール
【Scratch入門】SVG画像のviewportの意味とviewBox属性の使い方

ScratchにおけるSVG画像で重要なビューポート(Viewport)の概念と使い方の基本を説明します。

ではこの仕組みを踏まえて早速今回の拡張機能を実装してみます。

拡張機能の実装

今回も拡張機能のJavascriptの実装部分だけを紹介させていただきます。

もし詳しくScratch3の拡張機能の作成手順を知りたい方は、何度か具体的な実装を当ブログで解説していますので例えば以下のリンクの内容で確認してください。

合同会社タコスキングダム|タコキンのPスクール
【Scratch拡張機能・自作】文字からユニコード番号に変換するちょっとしたエクステンションを作る

Scratch-guiから、文字もしくは文字列からユニコード(16進数)に変換するための拡張機能を自作します。

以下に拡張用の
index.jsの実装を示します。

            
            const ArgumentType = require('../../extension-support/argument-type');
const BlockType = require('../../extension-support/block-type');
const Cast = require('../../util/cast');
const formatMessage = require('format-message');
const log = require('../../util/log');
const StageLayering = require('../../engine/stage-layering');

class Scratch3PartialImageWriter {
    constructor (runtime) {
        this._extVersion = '0.0.1';
        this.runtime = runtime;
        this._x = 0;
        this._y = 0;
        this._xOrigin = 24;
        this._yOrigin = -48;
        this._imageDrawableId = -1;
        this._imageSkinId = -1;
        this.emptyImg = '<svg version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="0" height="0" viewBox="0 0 0 0"></svg>';
    }

    _spawnSprite(partId_) {
        const xoffsets_ = [0, 0, 0, 0, -16, -16, -16, -16, -32, -32, -32, -32, -48, -48, -48, -48];
        const yoffsets_ = [0, -17, -33, -49, 0, -17, -33, -49, 0, -17, -33, -49, 0, -17, -33, -49];
        if (partId_ < 0 || partId_ > 16) { return this.emptyImg; }

        const characterSeed = `<svg version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="96" height="168" viewBox="0 0 64 112">
            <defs>
                <g id="baseImage">
                    <image width="64" height="112" xlink:href=""/>
                </g>
                <clipPath id="anim1"><rect width="16" height="16"/></clipPath>
                <clipPath id="anim2"><rect y="17" width="16" height="16"/></clipPath>
                <clipPath id="anim3"><rect y="33" width="16" height="16"/></clipPath>
                <clipPath id="anim4"><rect y="49" width="16" height="16"/></clipPath>
                <clipPath id="anim5"><rect x="17" y="0" width="16" height="16"/></clipPath>
                <clipPath id="anim6"><rect x="17" y="17" width="16" height="16"/></clipPath>
                <clipPath id="anim7"><rect x="17" y="33" width="16" height="16"/></clipPath>
                <clipPath id="anim8"><rect x="17" y="49" width="16" height="16"/></clipPath>
                <clipPath id="anim9"><rect x="33" y="0" width="16" height="16"/></clipPath>
                <clipPath id="anim10"><rect x="33" y="17" width="16" height="16"/></clipPath>
                <clipPath id="anim11"><rect x="33" y="33" width="16" height="16"/></clipPath>
                <clipPath id="anim12"><rect x="33" y="49" width="16" height="16"/></clipPath>
                <clipPath id="anim13"><rect x="49" y="0" width="16" height="16"/></clipPath>
                <clipPath id="anim14"><rect x="49" y="17" width="16" height="16"/></clipPath>
                <clipPath id="anim15"><rect x="49" y="33" width="16" height="16"/></clipPath>
                <clipPath id="anim16"><rect x="49" y="49" width="16" height="16"/></clipPath>
            </defs>
            <use xlink:href="#baseImage" clip-path="url(#anim${partId_})" x="${xoffsets_[partId_ - 1]}" y="${yoffsets_[partId_ - 1]}"></use>
        </svg>`;
        return characterSeed;
    }

    getInfo () {
        return {
            id: 'partialimagewriter',
            name: formatMessage({
                id: 'partialimagewriter.categoryName',
                default: 'PartialImageWriter',
                description: 'Label for the partialimagewriter extension category'
            }),
            blocks: [
                {
                    opcode: 'initSprite',
                    blockType: BlockType.COMMAND
                },
                {
                    opcode: 'stamp',
                    blockType: BlockType.COMMAND,
                    text: formatMessage({
                        id: 'partialimagewriter.stamp',
                        default: 'stamp [PART_ID]',
                        description: 'Render a current skin onto a drawble'
                    }),
                    arguments: {
                        PART_ID: {
                            type: ArgumentType.NUMBER,
                            defaultValue: 1
                        }
                    }
                },
                {
                    opcode: 'moveTo',
                    blockType: BlockType.COMMAND,
                    text: formatMessage({
                        id: 'partialimagewriter.moveto',
                        default: 'move to ( [X_POS], [Y_POS] )',
                        description: 'move current sprite to'
                    }),
                    arguments: {
                        X_POS: {
                            type: ArgumentType.NUMBER,
                            defaultValue: 0
                        },
                        Y_POS: {
                            type: ArgumentType.NUMBER,
                            defaultValue: 0
                        }
                    }
                },
                {
                    opcode: 'setXYOffset',
                    blockType: BlockType.COMMAND,
                    text: formatMessage({
                        id: 'partialimagewriter.setxyoffset',
                        default: 'Origin set to ( [X_ORIGIN], [Y_ORIGIN] )',
                        description: 'set new origin'
                    }),
                    arguments: {
                        X_ORIGIN: {
                            type: ArgumentType.NUMBER,
                            defaultValue: 0
                        },
                        Y_ORIGIN: {
                            type: ArgumentType.NUMBER,
                            defaultValue: 0
                        }
                    }
                },
                {
                    opcode: 'clearImage',
                    blockType: BlockType.COMMAND
                },
                {
                    opcode: 'goImageToBack',
                    blockType: BlockType.COMMAND
                },
                {
                    opcode: 'goImageToFront',
                    blockType: BlockType.COMMAND
                },
                {
                    opcode: 'getVersion',
                    text: 'Version Info',
                    blockType: BlockType.REPORTER
                }
            ],
            menus: {}
        };
    }

    stamp (args, util) {
        if (this._imageSkinId >= 0) {
            this._pi = Cast.toNumber(args.PART_ID);
            const chara1 = this._spawnSprite(this._pi);
            this.runtime.renderer.updateSVGSkin(this._imageSkinId, chara1);
        }
    }

    moveTo (args, util) {
        this._x = Cast.toNumber(args.X_POS);
        this._y = Cast.toNumber(args.Y_POS);
        if (this._imageDrawableId >= 0 && this._imageSkinId >= 0) {
            this.runtime.renderer.updateDrawableProperties(this._imageDrawableId, {
                skinId: this._imageSkinId,
                position: [this._x + this._xOrigin, this._y + this._yOrigin],
                scale: [100, 100],
                direction: 90
            });
            this.runtime.requestRedraw();
        }
    }

    setXYOffset(args, util) {
        this._xOrigin = Cast.toNumber(args.X_ORIGIN);
        this._yOrigin = Cast.toNumber(args.Y_ORIGIN);
    }

    initSprite() {
        if (this._imageSkinId < 0 && this.runtime.renderer) {
            const skinId = this.runtime.renderer.createSVGSkin(this.emptyImg);
            this._imageSkinId = skinId;
            this._imageDrawableId = this.runtime.renderer.createDrawable(StageLayering.SPRITE_LAYER);
            this.runtime.renderer.updateDrawableProperties(this._imageDrawableId, {skinId: this._imageSkinId});
        }
    }

    clearImage () {
        if (this._imageSkinId >= 0) {
            this.runtime.renderer.updateSVGSkin(this._imageSkinId, this.emptyImg);
        }
    }

    goImageToBack() {
        if (this._imageDrawableId >= 0 && this._imageSkinId >= 0) {
            this.runtime.renderer.setDrawableOrder(this._imageDrawableId, 1, StageLayering.SPRITE_LAYER);
        }
    }

    goImageToFront() {
        if (this._imageDrawableId >= 0 && this._imageSkinId >= 0) {
            this.runtime.renderer.setDrawableOrder(this._imageDrawableId, Infinity, StageLayering.SPRITE_LAYER);
        }
    }

    getVersion(args) {
        return this._extVersion;
    }
}

module.exports = Scratch3PartialImageWriter;
        
現状動かれせる画像が1つのスプライトのみしかありませんが、この拡張機能をインポートし、適当なプロジェクトを作って以下のように簡単なブロックプログラミングをしてみます。

合同会社タコスキングダム|タコキンのPスクール

テストとして動きを確認してみると、こんな感じに動いていればOKです。

合同会社タコスキングダム|タコキンのPスクール

ゲームとして遊ぶレベルになるにはまだまだ実装しなけれならない動きや操作が山積みですが、そこら辺はScratchでのプログラミングで作り込みしていくことになるので、1枚の画像から動くSVGアニメーションを作り出す拡張機能としてはだいたいこんな感じではなかろうかと思います。


合同会社タコスキングダム|タコキンのPスクール【Html&Cssアプリ】Html/JavascriptとSassを使った四択クイズゲームの作り方

すぐに作れる ずっと使える Inkscapeのすべてが身に付く本

まとめ

今回はRPGを作る上でマップを動き回る際のキャラクターチップのアニメーションを行うためだけの拡張機能を作成して動作させてみました。

この程度であれば拡張機能を使うまでも無いかも知れませんが、ゲームの作り込みが進んで、沢山のキャラクターを登場させてその一つ一つにアニメーションを付けて回るようなところまで来ると今回のような拡張機能が役に立つものと思います。

参考サイト

ぴぽや倉庫

Superpowers Asset Packs