😾

Scratchを拡張しよう!(1)天気予報を取得

2022/10/08に公開

はじめに

CoderDojoというボランティア団体で、子供達のプログラミングスキル習得の支援をしております。プログラミングの開発環境は、主にScratchを使っていますが、標準機能だけでは飽き足らず、このシリーズではScratchの拡張機能開発に挑戦します!

完成したサイトはこちらになります。

天気予報拡張機能付きScratch

天気予報拡張機能付きScratch
天気予報拡張機能付きScratch

拡張機能から天気予報を追加すると、

  • 現在地の天気予報、もしくは、都道府県別の天気予報を取得するブロックが使えるようになります。
  • 1週間分の天気予報をリストに格納する機能が追加されます。

Scratchの拡張機能開発

Scratchの拡張機能の開発はScratch Japan WikiタコキンのPスクール・ブログに丁寧に解説されていますが、色々つまづきポイントもありましたので備忘も含めまとめます。今回は、気軽に天気予報が取得できるOpen Meteoから情報を取得して、Scratchで表示する拡張機能を作成します。

https://open-meteo.com

下準備

まずは、scratch本家のリポジトリを自分のリポジトリにフォークしてコードを変更できるようにします。以下、2つのリポジトリをフォークします。フォークはGithub上でリポジトリを自分用にコピーする機能で、詳細はこちらを参照ください。

リポジトリ 内容
https://github.com/LLK/scratch-vm スクラッチの実行エンジン
https://github.com/LLK/scratch-gui スクラッチのユーザーインタフェースのリポジトリ

次に、フォークしたリポジトリをローカルにクローンして開発環境を整えます。

git clone --depth 1 https://github.com/[GitHubのアカウント]/scratch-vm.git # 1
git clone --depth 1 https://github.com/[GitHubのアカウント]/scratch-gui.git # 2
cd scratch-vm # 3
npm i # 4
npm run build # 5
npm link # 6
cd ../scratch-gui # 7
npm i # 8
npm link scratch-vm # 9
npm start # 10

以下に各々のコマンドの意味を説明します;

  1. scratch-vmをクローンします。--depth 1はデフォルトブランチの最後のコミットのみをクローンするオプションで、過去の履歴が必要ない場合に指定します(ダウンロードする容量が削減されるため高速化に寄与します)。[GitHubのアカウント]には個人のアカウントを指定してください。httpsでクローンする場合はhttps://github.com/[GitHubのアカウント]/scratch-vm.gitですが、SSHの場合はgit@github.com:[GitHubのアカウント]/scratch-vm.gitでクローンして下さい。
  2. scratch-guiをクローンします。詳細は1と同様です。
  3. scratch-vmディレクトリに移動します。
  4. npm inpm installの省略形で、scratch-vmに必要なパッケージをインストールします。
  5. scratch-vmの実行環境を構築します。※インターネット上の記事ではこのコマンドが省略されていますが、今回、このコマンドを実行しないとブラウザ上でエラーが発生しました。
  6. npm linkでscratch-vmの環境をローカルPC内の他のプロジェクトから参照できるように設定します。後ほど、scratch-guiのプロジェクトからこのローカル環境を参照するように設定します。
  7. scratch-guiディレクトリに移動します。
  8. scratch-guiに必要なパッケージをインストールします。
  9. link scratch-vmで今回ダウンロードしたscratch-vmを参照するようにします。この設定をしないと、ローカルのscratch-vmのソースコードの変更がscratch-gui側に反映されません。
  10. ローカルでWeb Serverを立ち上げて、scratchをテストできるようにします。

上記後、http://localhost:8601にブラウザでアクセスするとお馴染みのScratchの編集画面が立ち上がります。

scratch-gui側の開発

scratch-gui/src/lib/libraries/extensionsopenmeteoディレクトリを作りましょう。この中にScratchの拡張機能のアイコンを格納します。ラージサイズのアイコンは600px x 372px, スモールサイズのアイコンは80px x 80pxになります。

ラージサイズ
ラージサイズ 600px x 372px

スモールサイズ
スモールサイズ 80px x 80px

この後は、ソースコードの修正になります。scratch-gui/src/lib/libraries/extensions/index.jsxに以下のコードを追加します。

...中略

import gdxforInsetIconURL from './gdxfor/gdxfor-small.svg';
import gdxforConnectionIconURL from './gdxfor/gdxfor-illustration.svg';
import gdxforConnectionSmallIconURL from './gdxfor/gdxfor-small.svg';

import openMeteoIconURL from './openmeteo/open-meteo.png'; // この行を追加
import openMeteoInsetIconURL from './openmeteo/open-meteo-small.png'; // この行を追加

...中略

       connectingMessage: (
            <FormattedMessage
                defaultMessage="Connecting"
                description="Message to help people connect to their force and acceleration sensor."
                id="gui.extension.gdxfor.connectingMessage"
            />
        ),
        helpLink: 'https://scratch.mit.edu/vernier'
    },
    // 最後に以下の行を追加
    {
        name: '天気予報(てんきよほう)',
        extensionId: 'openMeteo',
        iconURL: openMeteoIconURL,
        insetIconURL: openMeteoInsetIconURL,
        description: '天気予報を取得します(てんきよほうをしゅとくします)',
        internetConnectionRequired: true,
        featured: true
    }
    // ここまで
];

index.jsxには拡張機能を選ぶの画面で表示する内容を設定します。openmeteoディレクトリに格納した画像のURLとその他各種表示情報を設定します。本当は、多言語対応、ひらがな表示対応まで設定したかったのですが、色々大変そうなため今回は日本語表示のみにしています。

scratch-vm側の開発

次に拡張機能のロジック本体を開発します。こちらはscratch-vm側を改造することになります。

先ずはscratchの開発でasync/awaitが使えるように以下のパッケージを追加します(こちらを参考にしました)。この設定は必須ではありませんが、コードの見通しが良くなるのでオススメです。以下、async/awaitが使える前提でコーディングしています。

cd ../scratch-vm
npm i @babel/runtime @babel/plugin-transform-runtime -D

src/virtual-machine.jsにの先頭に以下を追加するとasync/awaitが使えるようになります。

require('core-js');
require('regenerator-runtime/runtime');

次に拡張機能のアイコンを準備します。サイズは40px x 40pxで、メニュー用とブロック用の2つを準備します。

メニューアイコン
メニューアイコン、ブロックアイコン

上記のアイコンはソースコードに埋め込みますので、こちらのサイト等でbase64にエンコードして下さい(このあたりもタコキンのPスクール・ブログに詳しく記載されています)。

いよいよ拡張機能の開発です。src/extensionsディレクトリにscratch3_open_meteoディレクトリを作成し、以下のファイルを追加してコードを記述下さい。

index.js

拡張機能のメイン処理が書かれています。以下の機能をブロックとして公開しています。blockIconURI, menuIconURIにはbase64にエンコードしたアイコンがコピペされています。

  • getPrefectureWeatherで都道府県の天気を取得
  • getWeatherで現在地の天気を取得
  • listPrefectureWeatherで都道府県の一週間分の天気を取得しリストに格納
  • listWetherで現在地の一週間分の天気を取得しリストに格納

全体のソースコードの構造は、Scratch 3.0の拡張機能を作ってみよう/基本の書式を、リストからのデータ取得はScratch Extensionをつくってわかったことを参考に作成しています。

const ArgumentType = require('../../extension-support/argument-type');
const BlockType = require('../../extension-support/block-type');
const Cast = require('../../util/cast');
const Variable = require('../../engine/variable');
const prefectureLocations = require('./prefecture-locations');

/**
 * Icon svg to be displayed at the left edge of each extension block, encoded as a data URI.
 * @type {string}
 */
// eslint-disable-next-line max-len
const blockIconURI = '';

/**
 * Icon svg to be displayed in the category menu, encoded as a data URI.
 * @type {string}
 */
// eslint-disable-next-line max-len
const menuIconURI = '';

/**
 * Class for the new blocks in Scratch 3.0
 * @param {Runtime} runtime - the runtime instantiating this block package.
 * @constructor
 */
class Scratch3OpenMeteoBlocks {
    constructor (runtime) {
        /**
         * The runtime instantiating this block package.
         * @type {Runtime}
         */
        this.runtime = runtime;
    }

    getWeatherDescription (code = -1) {
        switch (code) {
        case 0: return '快晴';
        case 1: return '晴れ';
        case 2: return '晴れ時々曇り';
        case 3: return '曇り';
        case 45: return '霧';
        case 48: return '霧';
        case 51: return '弱い霧雨';
        case 53: return '霧雨';
        case 55: return '強い霧雨';
        case 56: return '寒い霧雨';
        case 57: return '凍える霧雨';
        case 61: return '小雨';
        case 63: return '雨';
        case 65: return '豪雨';
        case 66: return '寒い雨';
        case 67: return '凍える雨';
        case 71: return '弱い雪';
        case 73: return '雪';
        case 75: return '強い雪';
        case 77: return '霧雪';
        case 80: return '弱いにわか雨';
        case 81: return 'にわか雨';
        case 82: return '強いにわか雨';
        case 85: return '弱いにわか雪';
        case 86: return '強いにわか雪';
        case 95: return '雷雨';
        case 96: return '雷雨';
        case 99: return '強い雷雨';
        default: return '不明';
        }
    }

    /**
     * @returns {object} metadata for this extension and its blocks.
     */
    getInfo () {
        return {
            id: 'openMeteo',
            name: '天気',
            menuIconURI: menuIconURI,
            blockIconURI: blockIconURI,
            blocks: [
                {
                    opcode: 'listWeather',
                    text: '天気予報の日付を[DATE_LIST]に、天気を[WEATHER_LIST]に格納する',
                    blockType: BlockType.COMMAND,
                    arguments: {
                        DATE_LIST: {
                            type: ArgumentType.STRING,
                            defaultValue: 'リスト名'
                        },
                        WEATHER_LIST: {
                            type: ArgumentType.STRING,
                            defaultValue: 'リスト名'
                        }
                    }
                },
                {
                    opcode: 'listPrefectureWeather',
                    text: '[PREFECTURE]の天気予報の日付を[DATE_LIST]に、天気を[WEATHER_LIST]に格納する',
                    blockType: BlockType.COMMAND,
                    arguments: {
                        PREFECTURE: {
                            type: ArgumentType.NUMBER,
                            menu: 'prefectureMenu',
                            defaultValue: 0
                        },
                        DATE_LIST: {
                            type: ArgumentType.STRING,
                            defaultValue: 'リスト名'
                        },
                        WEATHER_LIST: {
                            type: ArgumentType.STRING,
                            defaultValue: 'リスト名'
                        }
                    }
                },
                {
                    opcode: 'getWeather',
                    text: '[OFFSET]日後の天気を取得',
                    blockType: BlockType.REPORTER,
                    arguments: {
                        OFFSET: {
                            type: ArgumentType.NUMBER,
                            defaultValue: 1
                        }
                    }
                },
                {
                    opcode: 'getPrefectureWeather',
                    text: '[PREFECTURE]の[OFFSET]日後の天気を取得',
                    blockType: BlockType.REPORTER,
                    arguments: {
                        PREFECTURE: {
                            type: ArgumentType.NUMBER,
                            menu: 'prefectureMenu',
                            defaultValue: 0
                        },
                        OFFSET: {
                            type: ArgumentType.NUMBER,
                            defaultValue: 1
                        }
                    }
                }
            ],
            menus: {
                prefectureMenu: {
                    acceptReporters: true,
                    items: prefectureLocations.map((loc, index) => ({
                        text: loc.prefecture,
                        value: index
                    }))
                }
            }
        };
    }

    async fetchWeather (latitude, longitude) {
        const prefix = 'https://api.open-meteo.com/v1/forecast?';
        const suffix = '&daily=weathercode&timezone=Asia%2FTokyo';

        const url = `${prefix}latitude=${latitude}&longitude=${longitude}${suffix}`;
        const res = await fetch(url);
        const result = await res.json();

        return result.daily;
    }

    async getPointWeather (latitude, longitude, offset) {
        const res = await this.fetchWeather(latitude, longitude);
        const weathercodes = res.weathercode;

        if (offset >= 0 && offset < weathercodes.length) {
            return this.getWeatherDescription(weathercodes[offset]);
        }
        return this.getWeatherDescription();
    }

    async getCurrentLocationWether (offset) {
        const position = await new Promise((resolve, reject) =>
            navigator.geolocation.getCurrentPosition(resolve, reject));
        return await this.getPointWeather(position.coords.latitude, position.coords.longitude, offset);
    }

    setList (list, array) {
        list.value = array;
        list._monitorUpToDate = false;
    }

    // 指定した県の日付指定の天気を取得
    getPrefectureWeather (args) {
        const index = Cast.toNumber(args.PREFECTURE);
        const loc = prefectureLocations[index];
        return this.getPointWeather(loc.latitude, loc.longitude, Cast.toNumber(args.OFFSET));
    }


    // 現在地の日付指定の天気を取得
    getWeather (args) {
        return this.getCurrentLocationWether(Cast.toNumber(args.OFFSET));
    }

    // 指定した県の一週間分の天気を取得
    async listPrefectureWeather (args, util) {
        const dateListName = Cast.toString(args.DATE_LIST);
        const weatherListName = Cast.toString(args.WEATHER_LIST);

        const dateList = util.target.lookupVariableByNameAndType(dateListName, Variable.LIST_TYPE);
        const weatherList = util.target.lookupVariableByNameAndType(weatherListName, Variable.LIST_TYPE);

        let res;
        if (dateList || weatherList) {
            const index = Cast.toNumber(args.PREFECTURE);
            const loc = prefectureLocations[index];
            res = await this.fetchWeather(loc.latitude, loc.longitude);
        }

        if (dateList) this.setList(dateList, res.time);
        if (weatherList) this.setList(weatherList, res.weathercode.map(wc => this.getWeatherDescription(wc)));
    }


    // 現在地の一週間分の天気を取得
    async listWeather (args, util) {
        const dateListName = Cast.toString(args.DATE_LIST);
        const weatherListName = Cast.toString(args.WEATHER_LIST);

        const dateList = util.target.lookupVariableByNameAndType(dateListName, Variable.LIST_TYPE);
        const weatherList = util.target.lookupVariableByNameAndType(weatherListName, Variable.LIST_TYPE);

        let res;
        if (dateList || weatherList) {
            const position = await new Promise((resolve, reject) =>
                navigator.geolocation.getCurrentPosition(resolve, reject));
            res = await this.fetchWeather(position.coords.latitude, position.coords.longitude);
        }

        if (dateList) this.setList(dateList, res.time);
        if (weatherList) this.setList(weatherList, res.weathercode.map(wc => this.getWeatherDescription(wc)));
    }
}

module.exports = Scratch3OpenMeteoBlocks;

prefecture-locations.js

都道府県の緯度、経度データになります。やじろべえさんのサイトのcsvデータをjsonに変換して取り込みました。

const prefectureLocations = [
    {
        prefecture: '北海道',
        latitude: 43.064359,
        longitude: 141.347449
    },
    {
        prefecture: '青森県',
        latitude: 40.824294,
        longitude: 140.74005
    },
    {
        prefecture: '岩手県',
        latitude: 39.70353,
        longitude: 141.152667
    },
    {
        prefecture: '宮城県',
        latitude: '38.268737',
        longitude: '140.872183'
    },
    {
        prefecture: '秋田県',
        latitude: 39.718175,
        longitude: 140.103356
    },
    {
        prefecture: '山形県',
        latitude: 38.240127,
        longitude: 140.362533
    },
    {
        prefecture: '福島県',
        latitude: 37.750146,
        longitude: 140.466754
    },
    {
        prefecture: '茨城県',
        latitude: 36.341817,
        longitude: 140.446796
    },
    {
        prefecture: '栃木県',
        latitude: 36.56575,
        longitude: 139.883526
    },
    {
        prefecture: '群馬県',
        latitude: 36.391205,
        longitude: 139.060917
    },
    {
        prefecture: '埼玉県',
        latitude: 35.857771,
        longitude: 139.647804
    },
    {
        prefecture: '千葉県',
        latitude: 35.604563,
        longitude: 140.123179
    },
    {
        prefecture: '東京都',
        latitude: 35.689185,
        longitude: 139.691648
    },
    {
        prefecture: '神奈川県',
        latitude: 35.447505,
        longitude: 139.642347
    },
    {
        prefecture: '新潟県',
        latitude: 37.901699,
        longitude: 139.022728
    },
    {
        prefecture: '富山県',
        latitude: 36.695274,
        longitude: 137.211302
    },
    {
        prefecture: '石川県',
        latitude: 36.594729,
        longitude: 136.62555
    },
    {
        prefecture: '福井県',
        latitude: 36.06522,
        longitude: 136.221641
    },
    {
        prefecture: '山梨県',
        latitude: 35.665102,
        longitude: 138.568985
    },
    {
        prefecture: '長野県',
        latitude: 36.651282,
        longitude: 138.180972
    },
    {
        prefecture: '岐阜県',
        latitude: 35.39116,
        longitude: 136.722204
    },
    {
        prefecture: '静岡県',
        latitude: 34.976987,
        longitude: 138.383057
    },
    {
        prefecture: '愛知県',
        latitude: 35.180247,
        longitude: 136.906698
    },
    {
        prefecture: '三重県',
        latitude: 34.730547,
        longitude: 136.50861
    },
    {
        prefecture: '滋賀県',
        latitude: 35.004532,
        longitude: 135.868588
    },
    {
        prefecture: '京都県',
        latitude: 35.0209962,
        longitude: 135.7531135
    },
    {
        prefecture: '大阪府',
        latitude: 34.686492,
        longitude: 135.518992
    },
    {
        prefecture: '兵庫県',
        latitude: 34.69128,
        longitude: 135.183087
    },
    {
        prefecture: '奈良県',
        latitude: 34.685296,
        longitude: 135.832745
    },
    {
        prefecture: '和歌山県',
        latitude: 34.224806,
        longitude: 135.16795
    },
    {
        prefecture: '鳥取県',
        latitude: 35.503463,
        longitude: 134.238258
    },
    {
        prefecture: '島根県',
        latitude: 35.472248,
        longitude: 133.05083
    },
    {
        prefecture: '岡山県',
        latitude: 34.66132,
        longitude: 133.934414
    },
    {
        prefecture: '広島県',
        latitude: 34.396033,
        longitude: 132.459595
    },
    {
        prefecture: '山口県',
        latitude: 34.185648,
        longitude: 131.470755
    },
    {
        prefecture: '徳島県',
        latitude: '34.065732',
        longitude: '134.559293'
    },
    {
        prefecture: '香川県',
        latitude: 34.34014,
        longitude: 134.04297
    },
    {
        prefecture: '愛媛県',
        latitude: 33.841649,
        longitude: 132.76585
    },
    {
        prefecture: '高知県',
        latitude: 33.55969,
        longitude: 133.530887
    },
    {
        prefecture: '福岡県',
        latitude: 33.606767,
        longitude: 130.418228
    },
    {
        prefecture: '佐賀県',
        latitude: 33.249367,
        longitude: 130.298822
    },
    {
        prefecture: '長崎県',
        latitude: 32.744542,
        longitude: 129.873037
    },
    {
        prefecture: '熊本県',
        latitude: 32.790385,
        longitude: 130.742345
    },
    {
        prefecture: '大分県',
        latitude: 33.2382,
        longitude: 131.612674
    },
    {
        prefecture: '宮崎県',
        latitude: 31.91109,
        longitude: 131.423855
    },
    {
        prefecture: '鹿児島県',
        latitude: 31.560219,
        longitude: 130.557906
    },
    {
        prefecture: '沖縄県',
        latitude: 26.211538,
        longitude: 127.681115
    }
];

module.exports = prefectureLocations;

extension-manager.jsへの登録

上記のコードをScratch-vmに登録します。src/extension-support/extension-manager.jsを以下のように修正します。

...中略
const builtinExtensions = {
    // This is an example that isn't loaded with the other core blocks,
    // but serves as a reference for loading core blocks as extensions.
    coreExample: () => require('../blocks/scratch3_core_example'),
    // These are the non-core built-in extensions.
    pen: () => require('../extensions/scratch3_pen'),
    wedo2: () => require('../extensions/scratch3_wedo2'),
    music: () => require('../extensions/scratch3_music'),
    microbit: () => require('../extensions/scratch3_microbit'),
    text2speech: () => require('../extensions/scratch3_text2speech'),
    translate: () => require('../extensions/scratch3_translate'),
    videoSensing: () => require('../extensions/scratch3_video_sensing'),
    ev3: () => require('../extensions/scratch3_ev3'),
    makeymakey: () => require('../extensions/scratch3_makeymakey'),
    boost: () => require('../extensions/scratch3_boost'),
    gdxfor: () => require('../extensions/scratch3_gdx_for'),
    openMeteo: () => require('../extensions/scratch3_open_meteo') // <- この行を追加
};
...中略

テスト

以下のコマンドを実行後、http://localhost:8601にブラウザでアクセスするとお馴染みのScratchのサイトから拡張機能を確認できます。ソースコードの変更がリアルタイムに反映されますので効率的にデバッグできます。

cd ../scratch-gui
npm start

GitHub Pagesでの公開

以上のscratch-vm, scratch-guiの変更をGitHubにpushします。GitHubのデフォルトブランチはdevelopになっていますが、新しいブランチを作成して、そちらをデフォルトブランチに設定して開発を進めるのがオススメです(developブランチは本家のscratch-vm, scratch-guiの変更が反映されていきますので)。

ここまでできれば、以下のコマンドで作成した拡張機能をGitHub Pagesに公開しましょう。

cd ../scratch-vm
npm i
npm run build # 念のためscratch-vm側もbuild
cd ../scratch-gui
npm run build
npm run deploy

上記で、https://[GitHubのアカウント].github.io/scratch-gui/に拡張機能付きのScratchが公開されます。

おわりに

既に多くの方々がScratchの拡張に挑戦していますのでインターネット上に多くの有益な情報があり大変助かりました!今回は直接Scratchのコードをフォークして拡張しましたが、Xcratchのように拡張機能を追加したScratchも公開されています。ある程度拡張のやり方がわかってきたらXcratchにも挑戦したいと思います!

GitHubで編集を提案

Discussion