diff --git a/package-lock.json b/package-lock.json index 5986fd591bc..0bdd2b38372 100644 --- a/package-lock.json +++ b/package-lock.json @@ -36,7 +36,9 @@ "devDependencies": { "@babel/core": "7.26.8", "@babel/eslint-parser": "7.26.8", + "@babel/plugin-transform-runtime": "^7.26.8", "@babel/preset-env": "7.26.8", + "@babel/runtime": "^7.26.7", "@commitlint/cli": "17.8.1", "@commitlint/config-conventional": "17.8.1", "adm-zip": "0.4.11", @@ -1292,6 +1294,41 @@ "@babel/core": "^7.0.0-0" } }, + "node_modules/@babel/plugin-transform-runtime": { + "version": "7.26.8", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-runtime/-/plugin-transform-runtime-7.26.8.tgz", + "integrity": "sha512-H0jlQxFMI0Q8SyGPsj9pO3ygVQRxPkIGytsL3m1Zqca8KrCPpMlvh+e2dxknqdfS8LFwBw+PpiYPD9qy/FPQpA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-module-imports": "^7.25.9", + "@babel/helper-plugin-utils": "^7.26.5", + "babel-plugin-polyfill-corejs2": "^0.4.10", + "babel-plugin-polyfill-corejs3": "^0.10.6", + "babel-plugin-polyfill-regenerator": "^0.6.1", + "semver": "^6.3.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-runtime/node_modules/babel-plugin-polyfill-corejs3": { + "version": "0.10.6", + "resolved": "https://registry.npmjs.org/babel-plugin-polyfill-corejs3/-/babel-plugin-polyfill-corejs3-0.10.6.tgz", + "integrity": "sha512-b37+KR2i/khY5sKmWNVQAnitvquQbNdWy6lJdsr0kmquCKEEUgMKK4SboVM3HtfnZilfjr4MMQ7vY58FVWDtIA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@babel/helper-define-polyfill-provider": "^0.6.2", + "core-js-compat": "^3.38.0" + }, + "peerDependencies": { + "@babel/core": "^7.4.0 || ^8.0.0-0 <8.0.0" + } + }, "node_modules/@babel/plugin-transform-shorthand-properties": { "version": "7.25.9", "resolved": "https://registry.npmjs.org/@babel/plugin-transform-shorthand-properties/-/plugin-transform-shorthand-properties-7.25.9.tgz", @@ -1540,9 +1577,9 @@ } }, "node_modules/@babel/runtime": { - "version": "7.25.0", - "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.25.0.tgz", - "integrity": "sha512-7dRy4DwXwtzBrPbZflqxnvfxLF8kdZXPkhymtDeFoFqE6ldzjQFgYTtYIFARcLEYDrqfBfYcZt1WqFxRoyC9Rw==", + "version": "7.26.7", + "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.26.7.tgz", + "integrity": "sha512-AOPI3D+a8dXnja+iwsUqGRjr1BbZIe771sXdapOtYI531gSqpi92vXivKcq2asu/DFpdl1ceFAKZyRzK2PCVcQ==", "license": "MIT", "dependencies": { "regenerator-runtime": "^0.14.0" @@ -4496,9 +4533,9 @@ "license": "MIT" }, "node_modules/body-parser": { - "version": "1.20.2", - "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.20.2.tgz", - "integrity": "sha512-ml9pReCu3M61kGlqoTm2umSXTlRTuGTx0bfYj+uIUKKYycG5NtSbeetV3faSU6R7ajOPw0g/J1PvK4qNy7s5bA==", + "version": "1.20.3", + "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.20.3.tgz", + "integrity": "sha512-7rAxByjUMqQ3/bHJy7D6OGXvx/MMc4IqBn/X0fcM1QUcAItpZrBEYhWGem+tzXH90c+G01ypMcYJBO9Y30203g==", "dev": true, "license": "MIT", "dependencies": { @@ -4510,7 +4547,7 @@ "http-errors": "2.0.0", "iconv-lite": "0.4.24", "on-finished": "2.4.1", - "qs": "6.11.0", + "qs": "6.13.0", "raw-body": "2.5.2", "type-is": "~1.6.18", "unpipe": "1.0.0" @@ -5677,9 +5714,9 @@ "license": "MIT" }, "node_modules/cookie": { - "version": "0.6.0", - "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.6.0.tgz", - "integrity": "sha512-U71cyTamuh1CRNCfpGY6to28lxvNwPG4Guz/EVjgf3Jmzv0vlDp1atT9eS5dDjMYHucpHbWns6Lwf3BKz6svdw==", + "version": "0.7.1", + "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.7.1.tgz", + "integrity": "sha512-6DnInpx7SJ2AK3+CTUE/ZM0vWTUboZCegxhC2xiIydHR9jNuTAASBrfEpHhiGOZw/nX51bHt6YQl8jsGo4y/0w==", "dev": true, "license": "MIT", "engines": { @@ -5934,9 +5971,9 @@ } }, "node_modules/cross-spawn": { - "version": "7.0.3", - "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz", - "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==", + "version": "7.0.6", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.6.tgz", + "integrity": "sha512-uV2QOWP2nWzsy2aMp8aRibhi9dlzF5Hgh5SHaB9OiTGEyDTiJJyx0uy51QXdyWbtAHNua4XJzUKca3OzKUd3vA==", "dev": true, "license": "MIT", "dependencies": { @@ -6615,9 +6652,9 @@ } }, "node_modules/default-gateway/node_modules/cross-spawn": { - "version": "6.0.5", - "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-6.0.5.tgz", - "integrity": "sha512-eTVLrBSt7fjbDygz805pMnstIs2VTBNkRm0qxZd+M7A5XDdxVRWO5MxGBXZhjY4cqLYLdtrGqRf8mBPmzwSpWQ==", + "version": "6.0.6", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-6.0.6.tgz", + "integrity": "sha512-VqCUuhcd1iB+dsv8gxPttb5iZh/D0iubSP21g36KXdEuf6I5JiioesUVjpCdHV9MZRUfVFlvwtIUyPfxo5trtw==", "dev": true, "license": "MIT", "dependencies": { @@ -7376,9 +7413,9 @@ } }, "node_modules/encodeurl": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", - "integrity": "sha512-TPJXq8JqFaVYm2CWmPvnP2Iyo4ZSM7/QKcSmuMLDObfpH5fi7RUGmd/rTDf+rut/saiDiQEeVTNgAmJEdAOx0w==", + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-2.0.0.tgz", + "integrity": "sha512-Q0n9HRi4m6JuGIV1eFlmvJB7ZEVxu93IrMyiMsGC0lrMJMWzRgx6WGquyfQgZVb31vhGgXnfmPNNXmxnOkRBrg==", "dev": true, "license": "MIT", "engines": { @@ -8522,38 +8559,38 @@ } }, "node_modules/express": { - "version": "4.19.2", - "resolved": "https://registry.npmjs.org/express/-/express-4.19.2.tgz", - "integrity": "sha512-5T6nhjsT+EOMzuck8JjBHARTHfMht0POzlA60WV2pMD3gyXw2LZnZ+ueGdNxG+0calOJcWKbpFcuzLZ91YWq9Q==", + "version": "4.21.2", + "resolved": "https://registry.npmjs.org/express/-/express-4.21.2.tgz", + "integrity": "sha512-28HqgMZAmih1Czt9ny7qr6ek2qddF4FclbMzwhCREB6OFfH+rXAnuNCwo1/wFvrtbgsQDb4kSbX9de9lFbrXnA==", "dev": true, "license": "MIT", "dependencies": { "accepts": "~1.3.8", "array-flatten": "1.1.1", - "body-parser": "1.20.2", + "body-parser": "1.20.3", "content-disposition": "0.5.4", "content-type": "~1.0.4", - "cookie": "0.6.0", + "cookie": "0.7.1", "cookie-signature": "1.0.6", "debug": "2.6.9", "depd": "2.0.0", - "encodeurl": "~1.0.2", + "encodeurl": "~2.0.0", "escape-html": "~1.0.3", "etag": "~1.8.1", - "finalhandler": "1.2.0", + "finalhandler": "1.3.1", "fresh": "0.5.2", "http-errors": "2.0.0", - "merge-descriptors": "1.0.1", + "merge-descriptors": "1.0.3", "methods": "~1.1.2", "on-finished": "2.4.1", "parseurl": "~1.3.3", - "path-to-regexp": "0.1.7", + "path-to-regexp": "0.1.12", "proxy-addr": "~2.0.7", - "qs": "6.11.0", + "qs": "6.13.0", "range-parser": "~1.2.1", "safe-buffer": "5.2.1", - "send": "0.18.0", - "serve-static": "1.15.0", + "send": "0.19.0", + "serve-static": "1.16.2", "setprototypeof": "1.2.0", "statuses": "2.0.1", "type-is": "~1.6.18", @@ -8562,6 +8599,10 @@ }, "engines": { "node": ">= 0.10.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/express" } }, "node_modules/express/node_modules/array-flatten": { @@ -9021,14 +9062,14 @@ } }, "node_modules/finalhandler": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.2.0.tgz", - "integrity": "sha512-5uXcUVftlQMFnWC9qu/svkWv3GTd2PfUhK/3PLkYNAe7FbqJMt3515HaxE6eRL74GdsriiwujiawdaB1BpEISg==", + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.3.1.tgz", + "integrity": "sha512-6BN9trH7bp3qvnrRyzsBz+g3lZxTNZTbVO2EV1CS0WIcDbawYVdYvGflME/9QP0h0pYlCDBCTjYa9nZzMDpyxQ==", "dev": true, "license": "MIT", "dependencies": { "debug": "2.6.9", - "encodeurl": "~1.0.2", + "encodeurl": "~2.0.0", "escape-html": "~1.0.3", "on-finished": "2.4.1", "parseurl": "~1.3.3", @@ -9402,9 +9443,9 @@ } }, "node_modules/format-message-cli/node_modules/cross-spawn": { - "version": "6.0.5", - "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-6.0.5.tgz", - "integrity": "sha512-eTVLrBSt7fjbDygz805pMnstIs2VTBNkRm0qxZd+M7A5XDdxVRWO5MxGBXZhjY4cqLYLdtrGqRf8mBPmzwSpWQ==", + "version": "6.0.6", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-6.0.6.tgz", + "integrity": "sha512-VqCUuhcd1iB+dsv8gxPttb5iZh/D0iubSP21g36KXdEuf6I5JiioesUVjpCdHV9MZRUfVFlvwtIUyPfxo5trtw==", "dev": true, "license": "MIT", "dependencies": { @@ -13837,11 +13878,14 @@ } }, "node_modules/merge-descriptors": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz", - "integrity": "sha512-cCi6g3/Zr1iqQi6ySbseM1Xvooa98N0w31jzUYrXPX2xqObmFGHJ0tQ5u74H3mVh7wLouTseZyYIq39g8cNp1w==", + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.3.tgz", + "integrity": "sha512-gaNvAS7TZ897/rVaZ0nMtAyxNyi/pdbjbAwUpFQpN70GqnVfOiXpeUUMKRBmzXaSQ8DdTX4/0ms62r2K+hE6mQ==", "dev": true, - "license": "MIT" + "license": "MIT", + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } }, "node_modules/merge-source-map": { "version": "1.0.4", @@ -13886,9 +13930,9 @@ "license": "BSD" }, "node_modules/micromatch": { - "version": "4.0.7", - "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.7.tgz", - "integrity": "sha512-LPP/3KorzCwBxfeUuZmaR6bG2kdeHSbe0P2tY3FLRU4vYrjYz5hI4QZwV0njUx3jeuKe67YukQ1LSPZBKDqO/Q==", + "version": "4.0.8", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.8.tgz", + "integrity": "sha512-PXwfBhYu0hBCPw8Dn0E+WDYb7af3dSLVWKi3HGv84IdF4TyFoC0ysxFd0Goxw7nSv4T/PzEJQxsYsEiFCKo2BA==", "dev": true, "license": "MIT", "dependencies": { @@ -14343,9 +14387,9 @@ "optional": true }, "node_modules/nanoid": { - "version": "3.3.7", - "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.7.tgz", - "integrity": "sha512-eSRppjcPIatRIMC1U6UngP8XFcz8MQWGQdt1MTBQ7NaAmvXDfvNxbvWV3x2y6CdEUciCSsDHDQZbhYaB8QEo2g==", + "version": "3.3.8", + "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.8.tgz", + "integrity": "sha512-WNLf5Sd8oZxOm+TzppcYk8gVOgP+l58xNy58D0nbUnOxOWRWvlcCV4kUF7ltmI6PsrLl/BgKEyS4mqsGChFN0w==", "dev": true, "funding": [ { @@ -18136,9 +18180,9 @@ "license": "MIT" }, "node_modules/path-to-regexp": { - "version": "0.1.7", - "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz", - "integrity": "sha512-5DFkuoqlv1uYQKxy8omFBeJPQcdoE07Kv2sferDCrAq1ohOU+MSDswDIbnx3YAM60qIOnYa53wBhXW0EbMonrQ==", + "version": "0.1.12", + "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.12.tgz", + "integrity": "sha512-RA1GjUVMnvYFxuqovrEqZoxxW5NUZqbwKtYz/Tt7nXerk0LbLblQmrsgdeOxV5SFHf0UDggjS/bSeOZwt1pmEQ==", "dev": true, "license": "MIT" }, @@ -18961,13 +19005,13 @@ } }, "node_modules/qs": { - "version": "6.11.0", - "resolved": "https://registry.npmjs.org/qs/-/qs-6.11.0.tgz", - "integrity": "sha512-MvjoMCJwEarSbUYk5O+nmoSzSutSsTwF85zcHPQ9OrlFoZOYIjaqBAJIqIXjptyD5vThxGq52Xu/MaJzRkIk4Q==", + "version": "6.13.0", + "resolved": "https://registry.npmjs.org/qs/-/qs-6.13.0.tgz", + "integrity": "sha512-+38qI9SOr8tfZ4QmJNplMUxqjbe7LKvvZgWdExBOmd+egZTtjLB67Gu0HRX3u/XOq7UU2Nx6nsjvS16Z9uwfpg==", "dev": true, "license": "BSD-3-Clause", "dependencies": { - "side-channel": "^1.0.4" + "side-channel": "^1.0.6" }, "engines": { "node": ">=0.6" @@ -20170,9 +20214,9 @@ } }, "node_modules/scratch-render": { - "version": "2.0.21", - "resolved": "https://registry.npmjs.org/scratch-render/-/scratch-render-2.0.21.tgz", - "integrity": "sha512-t0ZQh2Mb2aHb+9zd2uAd6bgvQefxpvjlD+uy0QDtyN0vn7Pb62zYt+f95x0WAWdW4mpAdnKrf99OmuysvDTmSA==", + "version": "2.0.163", + "resolved": "https://registry.npmjs.org/scratch-render/-/scratch-render-2.0.163.tgz", + "integrity": "sha512-rTBcApnb2ec0e+gMwgbsEfmriO5Thx35Fz7xBxU6OZEjfRpghG1z/nktPWG6KU1JgfNagWIypjX4QUgtWBXbog==", "license": "AGPL-3.0-only", "dependencies": { "grapheme-breaker": "^0.3.2", @@ -20181,7 +20225,7 @@ "linebreak": "^0.3.0", "minilog": "^3.1.0", "raw-loader": "^0.5.1", - "scratch-svg-renderer": "^2.3.102", + "scratch-svg-renderer": "^3.0.0", "twgl.js": "^4.4.0" }, "peerDependencies": { @@ -20196,44 +20240,6 @@ "base64-loader": "^1.0.0" } }, - "node_modules/scratch-render/node_modules/base64-js": { - "version": "1.5.1", - "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.5.1.tgz", - "integrity": "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==", - "funding": [ - { - "type": "github", - "url": "https://github.com/sponsors/feross" - }, - { - "type": "patreon", - "url": "https://www.patreon.com/feross" - }, - { - "type": "consulting", - "url": "https://feross.org/support" - } - ], - "license": "MIT" - }, - "node_modules/scratch-render/node_modules/scratch-svg-renderer": { - "version": "2.5.46", - "resolved": "https://registry.npmjs.org/scratch-svg-renderer/-/scratch-svg-renderer-2.5.46.tgz", - "integrity": "sha512-SrQgHZdqnu3DX6UuJNJPg/kpJTQT/mIVbQZvm5uTE/B47U5mSaVfgoGlNFhoNUWZlbHucqySyG4KMe+ThDUR/A==", - "license": "BSD-3-Clause", - "dependencies": { - "base64-js": "^1.2.1", - "base64-loader": "^1.0.0", - "css-tree": "^1.1.3", - "fastestsmallesttextencoderdecoder": "^1.0.22", - "isomorphic-dompurify": "^2.4.0", - "minilog": "^3.1.0", - "transformation-matrix": "^1.15.0" - }, - "peerDependencies": { - "scratch-render-fonts": "^1.0.0" - } - }, "node_modules/scratch-sb1-converter": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/scratch-sb1-converter/-/scratch-sb1-converter-2.0.0.tgz", @@ -20548,9 +20554,9 @@ } }, "node_modules/send": { - "version": "0.18.0", - "resolved": "https://registry.npmjs.org/send/-/send-0.18.0.tgz", - "integrity": "sha512-qqWzuOjSFOuqPjFe4NOsMLafToQQwBSOEpS+FwEt3A2V3vKubTquT3vmLTQpFgMXp8AlFWFuP1qKaJZOtPpVXg==", + "version": "0.19.0", + "resolved": "https://registry.npmjs.org/send/-/send-0.19.0.tgz", + "integrity": "sha512-dW41u5VfLXu8SJh5bwRmyYUbAoSB3c9uQh6L8h/KtsFREPWpbX1lrljJo186Jc4nmci/sGUZ9a0a0J2zgfq2hw==", "dev": true, "license": "MIT", "dependencies": { @@ -20589,6 +20595,16 @@ "dev": true, "license": "MIT" }, + "node_modules/send/node_modules/encodeurl": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", + "integrity": "sha512-TPJXq8JqFaVYm2CWmPvnP2Iyo4ZSM7/QKcSmuMLDObfpH5fi7RUGmd/rTDf+rut/saiDiQEeVTNgAmJEdAOx0w==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 0.8" + } + }, "node_modules/send/node_modules/mime": { "version": "1.6.0", "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", @@ -20703,16 +20719,16 @@ } }, "node_modules/serve-static": { - "version": "1.15.0", - "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.15.0.tgz", - "integrity": "sha512-XGuRDNjXUijsUL0vl6nSD7cwURuzEgglbOaFuZM9g3kwDXOWVTck0jLzjPzGD+TazWbboZYu52/9/XPdUgne9g==", + "version": "1.16.2", + "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.16.2.tgz", + "integrity": "sha512-VqpjJZKadQB/PEbEwvFdO43Ax5dFBZ2UECszz8bQ7pi7wt//PWe1P6MN7eCnjsatYtBT6EuiClbjSWP2WrIoTw==", "dev": true, "license": "MIT", "dependencies": { - "encodeurl": "~1.0.2", + "encodeurl": "~2.0.0", "escape-html": "~1.0.3", "parseurl": "~1.3.3", - "send": "0.18.0" + "send": "0.19.0" }, "engines": { "node": ">= 0.8.0" @@ -25607,22 +25623,6 @@ "dev": true, "license": "MIT" }, - "node_modules/url/node_modules/qs": { - "version": "6.13.0", - "resolved": "https://registry.npmjs.org/qs/-/qs-6.13.0.tgz", - "integrity": "sha512-+38qI9SOr8tfZ4QmJNplMUxqjbe7LKvvZgWdExBOmd+egZTtjLB67Gu0HRX3u/XOq7UU2Nx6nsjvS16Z9uwfpg==", - "dev": true, - "license": "BSD-3-Clause", - "dependencies": { - "side-channel": "^1.0.6" - }, - "engines": { - "node": ">=0.6" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/use": { "version": "3.1.1", "resolved": "https://registry.npmjs.org/use/-/use-3.1.1.tgz", diff --git a/package.json b/package.json index 80318b7e8a5..3060d211e9a 100644 --- a/package.json +++ b/package.json @@ -80,7 +80,9 @@ "devDependencies": { "@babel/core": "7.26.8", "@babel/eslint-parser": "7.26.8", + "@babel/plugin-transform-runtime": "^7.26.8", "@babel/preset-env": "7.26.8", + "@babel/runtime": "^7.26.7", "@commitlint/cli": "17.8.1", "@commitlint/config-conventional": "17.8.1", "adm-zip": "0.4.11", diff --git a/src/extension-support/extension-manager.js b/src/extension-support/extension-manager.js index 94ce3b1a085..a40b058361a 100644 --- a/src/extension-support/extension-manager.js +++ b/src/extension-support/extension-manager.js @@ -23,7 +23,8 @@ const builtinExtensions = { ev3: () => require('../extensions/scratch3_ev3'), makeymakey: () => require('../extensions/scratch3_makeymakey'), boost: () => require('../extensions/scratch3_boost'), - gdxfor: () => require('../extensions/scratch3_gdx_for') + gdxfor: () => require('../extensions/scratch3_gdx_for'), + openMeteo: () => require('../extensions/scratch3_openmeteo') }; /** diff --git a/src/extensions/scratch3_openmeteo/index.js b/src/extensions/scratch3_openmeteo/index.js new file mode 100644 index 00000000000..433a42a5892 --- /dev/null +++ b/src/extensions/scratch3_openmeteo/index.js @@ -0,0 +1,239 @@ +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 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; diff --git a/src/extensions/scratch3_openmeteo/prefecture-locations.js b/src/extensions/scratch3_openmeteo/prefecture-locations.js new file mode 100644 index 00000000000..c7e3fd2060d --- /dev/null +++ b/src/extensions/scratch3_openmeteo/prefecture-locations.js @@ -0,0 +1,239 @@ +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; diff --git a/src/virtual-machine.js b/src/virtual-machine.js index 64ad7934fed..b31f3c2b5d4 100644 --- a/src/virtual-machine.js +++ b/src/virtual-machine.js @@ -1,3 +1,6 @@ +require('core-js'); +require('regenerator-runtime/runtime'); + let _TextEncoder; if (typeof TextEncoder === 'undefined') { _TextEncoder = require('text-encoding').TextEncoder;