【Scratch拡張機能をJavascriptで自作】svg画像からRPG風のマップを生成する拡張機能を作る〜準備編①


※ 当ページには【広告/PR】を含む場合があります。
2021/02/01
【Scratch拡張機能をJavascriptで自作】svg画像からRPG風のマップを生成する拡張機能を作る〜マップ作成編②

RPG風のマップをペン機能で作るという試みは、ウェブ検索するといくつか引っかかります。

Scratch2時代のものですが、
この解説記事がとても詳しく解説してあります。

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

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

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


JavaScriptとHTMLで「レトロ風RPG」を作ろう 全コード解説

【Svelte.js入門】ReactやVueに挫折した人でも大丈夫!Svelteとfirebaseでシンプルアプリ開発

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

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

先にRPG風のSVGマップを作るモチベーションに関して挙げると、半分は思いつきです。

以下のマップをScratchのレンダラー(scratch-render)から直接操作することで、色んな利点が考えられます。

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

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


JavaScriptとHTMLで「レトロ風RPG」を作ろう 全コード解説

【Svelte.js入門】ReactやVueに挫折した人でも大丈夫!Svelteとfirebaseでシンプルアプリ開発

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

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画像のテストサイトなどで試すと、

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

といった感じに
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で呼び出すことができます。

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

あとはタイルごとに異なる画像のバリエーションを
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>
        

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

あとは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で持ってくるわけですが、ちょうど利用させていただけるような例えばこちらのサイトで公開されている画像を利用させていただきます。

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

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

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

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

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

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

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

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

            
            <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>
        

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

上の図のようにできます。

まだ4x4マス程度ですが、これでも十分マップに見えてきます。


JavaScriptとHTMLで「レトロ風RPG」を作ろう 全コード解説

【Svelte.js入門】ReactやVueに挫折した人でも大丈夫!Svelteとfirebaseでシンプルアプリ開発

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

まとめ

今回はRPG風のマップを操作するための拡張機能を作成する前段階の知識として、タイル画像を張り合わせ一つのsvg画像に仕立てるテクニックを紹介しました。

次回はcsv形式のマップの構成ファイルから、svg画像を作成する方法を考えてみたいと思います。