[Scratch3 拡張機能] svg画像からRPG風のマップを生成する拡張機能を作る〜準備編①


2021/02/01

RPG風のマップをペン機能で作るという試みは、ウェブ検索するといくつか引っかかります。Scratch2時代のものですが、
この解説記事がとても詳しく解説してあります。

基本的にはタイル(マップを構成する画像の単位要素)となるコスチュームを用意しておいて、一定の座標をずらして並べているようなプログラムになるのですが、今回はもっと上級者向けの試みとして、svg画像から生成したマップの画像を直接内部でレンダリングさせる拡張機能を作ってみたいと思います。

SVGマップを拡張機能化する意義

先にRPG風のSVGマップを作るモチベーションに関して挙げると、半分は思いつきですが、以下のマップをScratchのレンダラー(scratch-render)から直接操作することで、色んな利点が考えられます。

            1. 背景とは別にマップを呼び出すだけで使いたい
2. タイルの座標値からイベントを発生させたい
3. マップを効率的に切り替えたい
4. テキストだけで複数のマップを管理したい
        
などなどです。

現時点では抽象的なアイデアのお話をしているだけですが、以降の記事では順を追いながらこのポイントを念頭に具体的な実装を説明していきます。

SVGからタイルマップを作る方法

まず今回は、SVGタイルマップを作るための基本的なアイデアから説明していきます。

defsとuseを使ってsvg画像を作成する

簡単のために全体40x40のサイズ、タイルのサイズ10x10のSVG画像をマップと見立てて考えます。

            <svg xmlns="http://www.w3.org/2000/svg" width="40" height="40" version="1.1">
    <rect width="9" height="9" stroke="black" stroke-width="1" fill="yellow" transform="translate(0,0)"/>
    <rect width="9" height="9" stroke="black" stroke-width="1" fill="yellow" transform="translate(10,0)"/>
    <rect width="9" height="9" stroke="black" stroke-width="1" fill="yellow" transform="translate(20,0)"/>
    <rect width="9" height="9" stroke="black" stroke-width="1" fill="yellow" transform="translate(30,0)"/>
    <rect width="9" height="9" stroke="black" stroke-width="1" fill="yellow" transform="translate(0,10)"/>
    <rect width="9" height="9" stroke="black" stroke-width="1" fill="yellow" transform="translate(10,10)"/>
    <rect width="9" height="9" stroke="black" stroke-width="1" fill="yellow" transform="translate(20,10)"/>
    <rect width="9" height="9" stroke="black" stroke-width="1" fill="yellow" transform="translate(30,10)"/>
    <rect width="9" height="9" stroke="black" stroke-width="1" fill="yellow" transform="translate(0,20)"/>
    <rect width="9" height="9" stroke="black" stroke-width="1" fill="yellow" transform="translate(10,20)"/>
    <rect width="9" height="9" stroke="black" stroke-width="1" fill="yellow" transform="translate(20,20)"/>
    <rect width="9" height="9" stroke="black" stroke-width="1" fill="yellow" transform="translate(30,20)"/>
    <rect width="9" height="9" stroke="black" stroke-width="1" fill="yellow" transform="translate(0,30)"/>
    <rect width="9" height="9" stroke="black" stroke-width="1" fill="yellow" transform="translate(10,30)"/>
    <rect width="9" height="9" stroke="black" stroke-width="1" fill="yellow" transform="translate(20,30)"/>
    <rect width="9" height="9" stroke="black" stroke-width="1" fill="yellow" transform="translate(30,30)"/>
</svg>
        
これをSVG画像のテストサイトなどで試すと、

といった感じに4x4のタイルが作成されます。

ただこの画像は同じ
rect要素を繰り返し使っているだけですので助長な書き方になっています。そこでdefs要素とuse要素を使うことで、もう少しコードの再利用性を考慮した書き方にします。

            <svg xmlns="http://www.w3.org/2000/svg" width="40" height="40" version="1.1" viewBox="0 0 40 40">
    <defs>
        <rect id="tile1" width="10" height="10" stroke="black" fill="green" viewBox="0 0 10 10"/>
    </defs>
    <use xlink:href="#tile1" x="0" y="0"/>
    <use xlink:href="#tile1" x="10" y="0"/>
    <use xlink:href="#tile1" x="20" y="0"/>
    <use xlink:href="#tile1" x="30" y="0"/>
    <use xlink:href="#tile1" x="0" y="10"/>
    <use xlink:href="#tile1" x="10" y="10"/>
    <use xlink:href="#tile1" x="20" y="10"/>
    <use xlink:href="#tile1" x="30" y="10"/>
    <use xlink:href="#tile1" x="0" y="20"/>
    <use xlink:href="#tile1" x="10" y="20"/>
    <use xlink:href="#tile1" x="20" y="20"/>
    <use xlink:href="#tile1" x="30" y="20"/>
    <use xlink:href="#tile1" x="0" y="30"/>
    <use xlink:href="#tile1" x="10" y="30"/>
    <use xlink:href="#tile1" x="20" y="30"/>
    <use xlink:href="#tile1" x="30" y="30"/>
</svg>
        
ということで、defs要素内で定義したタイル要素画像はid属性を通じてuseで呼び出すことができます。

あとはタイルごとに異なる画像のバリエーションを
defs要素に追加していくことで基本的なマップが構成できるようになります。

            <svg xmlns="http://www.w3.org/2000/svg" width="40" height="40" version="1.1" viewBox="0 0 40 40">
    <defs>
        <rect id="tile1" width="10" height="10" stroke="black" fill="green" viewBox="0 0 10 10"/>
        <rect id="tile2" width="10" height="10" stroke="black" fill="blue" viewBox="0 0 10 10"/>
        <rect id="tile3" width="10" height="10" stroke="black" fill="red" viewBox="0 0 10 10"/>
        <rect id="tile4" width="10" height="10" stroke="black" fill="yellow" viewBox="0 0 10 10"/>
    </defs>
    <use xlink:href="#tile1" x="0" y="0"/>
    <use xlink:href="#tile2" x="10" y="0"/>
    <use xlink:href="#tile3" x="20" y="0"/>
    <use xlink:href="#tile4" x="30" y="0"/>
    <use xlink:href="#tile3" x="0" y="10"/>
    <use xlink:href="#tile2" x="10" y="10"/>
    <use xlink:href="#tile4" x="20" y="10"/>
    <use xlink:href="#tile1" x="30" y="10"/>
    <use xlink:href="#tile4" x="0" y="20"/>
    <use xlink:href="#tile2" x="10" y="20"/>
    <use xlink:href="#tile3" x="20" y="20"/>
    <use xlink:href="#tile1" x="30" y="20"/>
    <use xlink:href="#tile3" x="0" y="30"/>
    <use xlink:href="#tile4" x="10" y="30"/>
    <use xlink:href="#tile2" x="20" y="30"/>
    <use xlink:href="#tile1" x="30" y="30"/>
</svg>
        

あとはRPG風の画像を持ったタイルを用意できれば
defsにガンガン登録するのですが、肝心の画像は、自分でベクター画像を書いたり、何かしらのペインターソフトでフリーの素材のpng画像を変換したりすることになります。

一から全てのタイルの絵を描くとなるとそれはそれで大変ですので、個人的にやっているタイルの用意のやり方を次の節で紹介しておきます。

タイルの作成方法

本番用のスクラッチのステージ画面は480x360のフレームサイズがデフォルトになっています。ということでタイルサイズ選びにはステージの縦横の値が割り切れるような正方形を選ぶほうが都合が良いですので、ここでは24x24で行います。

以下の例はまだ画像は定義してませんが、
image要素にxlink:href="data:image/png;base64,"という定義途中の属性を先に仕込んでいます。

            <svg xmlns="http://www.w3.org/2000/svg" width="96" height="96" version="1.1" viewBox="0 0 96 96">
    <defs>
    <image id="tile1" x="0" y="0" width="24" height="24" preserveAspectRatio="none" xlink:href="data:image/png;base64,"/>
    </defs>
    <use xlink:href="#tile1" x="0" y="0"/>
    <use xlink:href="#tile1" x="24" y="0"/>
    <use xlink:href="#tile1" x="48" y="0"/>
    <use xlink:href="#tile1" x="72" y="0"/>
    <use xlink:href="#tile1" x="0" y="24"/>
    <use xlink:href="#tile1" x="24" y="24"/>
    <use xlink:href="#tile1" x="48" y="24"/>
    <use xlink:href="#tile1" x="72" y="24"/>
    <use xlink:href="#tile1" x="0" y="48"/>
    <use xlink:href="#tile1" x="24" y="48"/>
    <use xlink:href="#tile1" x="48" y="48"/>
    <use xlink:href="#tile1" x="72" y="48"/>
    <use xlink:href="#tile1" x="0" y="72"/>
    <use xlink:href="#tile1" x="24" y="72"/>
    <use xlink:href="#tile1" x="48" y="72"/>
    <use xlink:href="#tile1" x="72" y="72"/>
</svg>
        
そしてタイルにしたい24px四方のpng画像をbase64で持ってくるわけですが、ちょうど利用させていただけるような例えばこちらのサイトで公開されている画像を利用させていただきます。

例えば草原のタイルを
grassland.pngでローカルに保存しておきます。

次にこちらの
png > base64に画像変換して貰えるサイトでこのpng画像をbase64変換します。

ということで、得られたbase64画像の文字列を先程の
xlink:hrefの部分に割り当ててみます。

            ...
    <image x="0" y="0" width="24" height="24" preserveAspectRatio="none" xlink:href=""/>
...
        
こうすると、以下のように草原のタイルがマップに割り当てられることが分かると思います。

いくつかタイルにバリエーションをもたせた例では、

            <svg xmlns="http://www.w3.org/2000/svg" width="96" height="96" version="1.1" viewBox="0 0 96 96">
    <defs>
    <image id="tile1" x="0" y="0" width="24" height="24" preserveAspectRatio="none" xlink:href=""/>
    <image id="tile2" x="0" y="0" width="24" height="24" preserveAspectRatio="none" xlink:href=""/>
    <image id="tile3" x="0" y="0" width="24" height="24" preserveAspectRatio="none" xlink:href=""/>
    <image id="tile4" x="0" y="0" width="24" height="24" preserveAspectRatio="none" xlink:href=""/>
    </defs>
    <use xlink:href="#tile1" x="0" y="0"/>
    <use xlink:href="#tile2" x="24" y="0"/>
    <use xlink:href="#tile3" x="48" y="0"/>
    <use xlink:href="#tile4" x="72" y="0"/>
    <use xlink:href="#tile3" x="0" y="24"/>
    <use xlink:href="#tile2" x="24" y="24"/>
    <use xlink:href="#tile4" x="48" y="24"/>
    <use xlink:href="#tile1" x="72" y="24"/>
    <use xlink:href="#tile4" x="0" y="48"/>
    <use xlink:href="#tile2" x="24" y="48"/>
    <use xlink:href="#tile3" x="48" y="48"/>
    <use xlink:href="#tile1" x="72" y="48"/>
    <use xlink:href="#tile3" x="0" y="72"/>
    <use xlink:href="#tile4" x="24" y="72"/>
    <use xlink:href="#tile2" x="48" y="72"/>
    <use xlink:href="#tile1" x="72" y="72"/>
</svg>