diff --git a/.eslintrc.js b/.eslintrc.js index 1ca8091ca7..db32ceab95 100644 --- a/.eslintrc.js +++ b/.eslintrc.js @@ -10,29 +10,13 @@ module.exports = { ecmaVersion: 'latest' }, globals: { + Blockly: 'readonly', Scratch: 'readonly', + ScratchBlocks: 'readonly', ScratchExtensions: 'readonly', scaffolding: 'readonly' }, rules: { - // Here is where we enforce rules to have somewhat consistent code style without being overbearing - 'semi': [ - 'warn', - 'always' - ], - 'brace-style': 'warn', - 'key-spacing': 'warn', - 'keyword-spacing': 'warn', - 'new-parens': 'warn', - 'no-trailing-spaces': [ - 'warn', - { - ignoreComments: true - } - ], - 'space-infix-ops': 'warn', - 'no-tabs': 'warn', - 'no-unused-vars': 'off', // Allow while (true) { } 'no-constant-condition': [ diff --git a/.github/workflows/validate.yml b/.github/workflows/validate.yml index bd20acc560..2db07a1930 100644 --- a/.github/workflows/validate.yml +++ b/.github/workflows/validate.yml @@ -4,7 +4,7 @@ on: pull_request: jobs: - build: + validate: runs-on: ubuntu-latest steps: - name: Checkout @@ -12,10 +12,39 @@ jobs: - name: Install Node.js uses: actions/setup-node@v3 with: - node-version: 16 + node-version: '14.x' + cache: 'npm' - name: Install dependencies run: npm ci - name: Validate run: npm run validate - - name: Lint + + lint: + runs-on: ubuntu-latest + steps: + - name: Checkout + uses: actions/checkout@v3 + - name: Install Node.js + uses: actions/setup-node@v3 + with: + node-version: '14.x' + cache: 'npm' + - name: Install dependencies + run: npm ci + - name: Validate run: npm run lint + + format: + runs-on: ubuntu-latest + steps: + - name: Checkout + uses: actions/checkout@v3 + - name: Install Node.js + uses: actions/setup-node@v3 + with: + node-version: '14.x' + cache: 'npm' + - name: Install dependencies + run: npm ci + - name: Validate + run: npm run check-format diff --git a/.prettierignore b/.prettierignore new file mode 100644 index 0000000000..0a39a3b587 --- /dev/null +++ b/.prettierignore @@ -0,0 +1,9 @@ +build +images +licenses +website +.eslintrc.js +*.json +*.yml +*.md +extensions/docs-examples diff --git a/.prettierrc.json b/.prettierrc.json new file mode 100644 index 0000000000..37cdf9afd4 --- /dev/null +++ b/.prettierrc.json @@ -0,0 +1,4 @@ +{ + "trailingComma": "es5", + "endOfLine": "auto" +} diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md index 96b706319c..e522ab1276 100644 --- a/CONTRIBUTING.md +++ b/CONTRIBUTING.md @@ -1,6 +1,6 @@ # Contributing extensions -Before you submit extensions, please read the NEW custom extension tutorial in full: +Before you submit extensions, please read the custom extension tutorial **in full**: - https://docs.turbowarp.org/development/extensions/introduction @@ -10,76 +10,74 @@ Please pay special attention to: - Maintaining backward compatibility: https://docs.turbowarp.org/development/extensions/compatibility - A better development server: https://docs.turbowarp.org/development/extensions/better-development-server -Pull requests that don't follow the guidelines outlined in these documents tend to take much longer to be reviewed and merged. +Read this document **in full** too. Pull requests that don't follow the guidelines will take *much* longer to be reviewed. -## Local development server +## Acceptance criteria -We recommend using our local development server: +Strictly, nothing is banned, but the following are *highly* discouraged: -```bash -# Clone the repository -git clone https://github.com/TurboWarp/extensions.git -cd extensions - -# Install dependencies -npm ci + - Broad "Utilities" extensions (break them up into multiple extensions, see https://github.com/TurboWarp/extensions/issues/674) + - Extensions that are very similar to existing ones (consider modifying the existing one instead) + - One-use personal extensions (load the extension as a local file instead) + - Joke extensions (they aren't funny when they cause us to get bug reports) -# Start development server -npm run dev -``` +Some extensions were added before these rules existed. That doesn't mean you will be exempted too. -This starts an HTTP server on [http://localhost:8000/](http://localhost:8000/) in development mode which adds a couple of extra tools to the homepage. +## Important context -After installing npm dependencies, TypeScript aware editors such as Visual Studio Code will give you smart autocomplete suggestions for most Scratch and extension APIs based on [@turbowarp/types](https://github.com/TurboWarp/types) and [@turbowarp/types-tw](https://github.com/TurboWarp/types-tw). Note that these types are not perfect; some methods are missing or incorrect. Please report any issues you find. +Every merged extension is more code that we will be expected to maintain indefinitely, even if you disappear. Remember: broken extensions mean that real projects by real people no longer work. If the renderer is rewritten one day, we will have to ensure that extensions like Clipping & Blending, RGB Channels, and Augmented Reality still work. That's not a small commitment. -If you encounter a TypeScript error, as long as you understand the error, feel free to add `// @ts-ignore`, `// @ts-expect-error`, or just ignore the error entirely. We currently do not require extensions to pass type checking. +We're all volunteers who all have lives outside of Scratch extensions. Many have full time jobs or are full time students. We'll get to you as soon as we can, so please be patient. -## Alternative development server - -If for some reason you can't set up our local development server, you can start any other HTTP server in the `extensions` folder. You won't get some of the nice things our server has, but it may be good enough. If you have Python 3 installed this is very simple: +## Writing extensions -```bash -cd extensions -python3 -m http.server -``` +Extension source code goes in the [`extensions`](extensions) folder. For example, an extension placed at `extensions/hello-world.js` would be accessible at [http://localhost:8000/hello-world.js](http://localhost:8000/hello-world.js) using our development server. -Note that browsers tend to aggressively cache JavaScript files that don't opt out of caching as our development server does, so you may have to do hard reloads to ensure that changes to your scripts are applied. +New extensions should be added in a user folder. You can name your folder anything you want; common choices are your GitHub username or your Scratch username. If your username is `TestMuffin123`, then `TestMuffin123`, `TestMuffin`, or even just `Muffin` would all be accepted -- we are very lenient about this. Do note that user folders are just for organization; other people are still allowed to edit your extension. Renaming your folder later is only allowed in very rare circumstances, so please get it right the first time. -## Types of extensions we accept +Extensions must be self-contained. All libraries and hardcoded resources should be embedded into the extension's JavaScript file. If you include minified code, please link where to find the unminified code and include a copy of the original license. -We strive to be tolerant of accepting almost any extension, including one-use novelty extensions or extensions that are similar to ones that already exist. +## Website stuff -Extensions end up in one of these categories depending on various qualities: +To add an extension to the homepage, you need to add metadata comments at the very start of the extension's JavaScript, and add the extension's path (without .js) to `extensions/extensions.json`. The order of that list determines the order of the library. Don't worry about putting it in the right spot, we'll move it if we disagree. - - Extensions that are in the repository, but not listed on the website - - Extensions that are listed on the website - - Extensions that are listed in the editor's builtin extension library +The header comments look like this: -## Writing extensions +```js +// Name: Example Extension +// ID: extensionid +// Description: Does a very cool thing. This must have punctuation at the end! +// By: GarboMuffin +// Original: TestMuffin +``` -Extension source code goes in the `extensions` folder. For example, an extension placed at `extensions/hello-world.js` would be accessible at [http://localhost:8000/hello-world.js](http://localhost:8000/hello-world.js). +Remember, this has to be the *very first* thing in the JS file. `Name`, `Description`, and `ID` are required. Make sure that `ID` exactly matches what you return in `getInfo()`. You can have zero or more `By` and `Original`. Put credit links in `` if you have one. It must point to a Scratch user profile. The parser is pretty loose, but try not to deviate too far from this format. -New extensions should be added in a user folder. You can name your folder your GitHub username, your Scratch username, or something else. For example, if your GitHub username is "TestMuffin", you could make a `TestMuffin` folder inside of the `extensions` folder and put your extensions inside there. You could then access a file placed at `extensions/TestMuffin/hello-world.js` at [http://localhost:8000/TestMuffin/hello-world.js](http://localhost:8000/TestMuffin/hello-world.js). +New extensions do not *need* images, but they are highly encouraged. Save the image in the `images` folder with the same folder name and file name (but different file extension) as the extension's source code. For example, if your extension is located in `extensions/TestMuffin/fetch.js`, save the image as `images/TestMuffin/fetch.svg` or `images/TestMuffin/fetch.png`. The homepage generator will detect it automatically. Images are displayed in a 2:1 aspect ratio. SVG (preferred), PNG, or JPG are accepted. PNG or JPG should be 600x300 in resolution. Please add proper attribution to `images/README.md` for *any* resources that were not made by you. -Static resources go in the `website` folder. This is where some example resources used by extensions such as fetch are placed. It works similarly to the `extensions` folder. +Most extensions shouldn't need external documentation -- it should be obvious what to do just by looking at the blocks. That said, some do need more explanation. Documentation is written in markdown and placed in the `docs` folder with a similar layout to images. For example, documentation for `extensions/TestMuffin/fetch.js` would be saved as `docs/TestMuffin/fetch.md`. Our version of markdown is slightly extended to allow rendering [scratchblocks](https://scratchblocks.github.io/). Just look at the existing documentation for syntax examples. It's not a perfect experience: block colors have to be manually copied, and icons aren't supported, but it's better than what we had before. Once you put your markdown there, you can set a `docsURI` like `https://extensions.turbowarp.org/TestMuffin/fetch`. -Extensions must not use `eval()`, `new Function()`, untrusted ` + + <% } %> + + diff --git a/development/homepage-template.ejs b/development/homepage-template.ejs new file mode 100644 index 0000000000..18fd2a84fa --- /dev/null +++ b/development/homepage-template.ejs @@ -0,0 +1,334 @@ + + + + + + TurboWarp Extension Gallery + + + + + + +
+

+ +
TurboWarp Extension Gallery
+

+ +

Unlike custom extensions on other websites, these aren't limited by the extension sandbox, so they are a lot more powerful. All extensions are reviewed for safety.

+

+ To use multiple of these extensions in TurboWarp, hover over the extension and press the button to copy its URL. Then go to the editor, open the extension chooser, then choose the "Custom Extension" option at the bottom, and enter the URL. +

+ +
+
These extensions are not compatible with Scratch.
+ Projects that use these extensions can't be uploaded to the Scratch website. + They can, however, be used in the packager. +
+ +
+ For compatibility, security, and offline support, each TurboWarp Desktop update contains an offline copy of these extensions from its release date, so some extensions may be outdated or missing. + Use the latest update for best results. +
+
+ + <% if (mode === "development") { %> +
+
+

Development Server Tools

+

+ Most recently modified extensions: + <% for (const extensionSlug of mostRecentExtensions) { %> + <%= extensionSlug %>.js + <% } %> +

+
+
+ <% } %> + + + +
+
+ <% + const peopleToHTML = (people) => { + let result = ''; + for (let i = 0; i < people.length; i++) { + result += people[i].toHTML(); + if (i === people.length - 1) { + // do nothing + } else if (i === people.length - 2) { + if (people.length > 2) { + result += ','; + } + result += ' and '; + } else { + result += ', '; + } + } + return result; + }; + %> + <% for (const [extensionSlug, metadata] of Object.entries(extensionMetadata)) { %> +
+
+ <% const image = extensionImages[extensionSlug] || 'images/unknown.svg'; %> + + +
+ + Open Extension +
+
+ +

<%= metadata.name %>

+

+ <%= metadata.description %> + <%- metadata.by.length ? `Created by ${peopleToHTML(metadata.by)}.` : '' %> + <%- metadata.original.length ? `Originally created by ${peopleToHTML(metadata.original)}.` : '' %> +

+
+ <% } %> +
+
+ + + + diff --git a/development/parse-extension-metadata.js b/development/parse-extension-metadata.js new file mode 100644 index 0000000000..632f01216c --- /dev/null +++ b/development/parse-extension-metadata.js @@ -0,0 +1,106 @@ +class Person { + constructor(name, link) { + /** @type {string} */ + this.name = name; + /** @type {string|null} */ + this.link = link; + } + + toHTML() { + // Don't need to bother escaping here. There's no vulnerability. + if (this.link) { + return `${this.name}`; + } + return this.name; + } +} + +class Extension { + constructor() { + this.id = ""; + this.name = ""; + this.description = ""; + /** @type {Person[]} */ + this.by = []; + /** @type {Person[]} */ + this.original = []; + } +} + +/** + * @param {string} string + * @param {string} split + * @returns {string[]} + */ +const splitFirst = (string, split) => { + const idx = string.indexOf(split); + if (idx === -1) { + return [string]; + } + return [string.substring(0, idx), string.substring(idx + split.length)]; +}; + +/** + * @param {string} person + * @returns {Person} + */ +const parsePerson = (person) => { + const parts = splitFirst(person, "<"); + if (parts.length === 1) { + return new Person(person, null); + } + + const name = parts[0].trim(); + const link = parts[1].replace(">", ""); + return new Person(name, link); +}; + +/** + * @param {string} extensionCode + * @return {Extension} + */ +const parseMetadata = (extensionCode) => { + const metadata = new Extension(); + + for (const line of extensionCode.split("\n")) { + if (!line.startsWith("//")) { + // End of header. + break; + } + + const withoutComment = line.substring(2).trim(); + const parts = splitFirst(withoutComment, ":"); + if (parts.length === 1) { + // Invalid. + continue; + } + + const key = parts[0].toLowerCase().trim(); + const value = parts[1].trim(); + + switch (key) { + case "id": + metadata.id = value; + break; + case "name": + metadata.name = value; + break; + case "description": + metadata.description = value; + break; + case "by": + metadata.by.push(parsePerson(value)); + break; + case "original": + metadata.original.push(parsePerson(value)); + break; + default: + // TODO + break; + } + } + + return metadata; +}; + +module.exports = parseMetadata; diff --git a/development/render-docs.js b/development/render-docs.js new file mode 100644 index 0000000000..3bdcfab88e --- /dev/null +++ b/development/render-docs.js @@ -0,0 +1,75 @@ +const path = require("path"); +const MarkdownIt = require("markdown-it"); +const renderTemplate = require("./render-template"); + +const md = new MarkdownIt({ + html: true, + linkify: true, + breaks: true, +}); + +md.renderer.rules.fence = function (tokens, idx, options, env, self) { + const token = tokens[idx]; + + if (token.info === "scratch") { + env.usesScratchBlocks = true; + return `
${md.utils.escapeHtml( + token.content + )}
`; + } + + // By default markdown-it will use a strange combination of and
; we'd rather it
+  // just use 
+  return `
${md.utils.escapeHtml(token.content)}
`; +}; + +/** + * @param {string} markdownSource Markdown code + * @param {string} slug Path slug like 'TestMuffin/fetch' + * @returns {string} HTML source code + */ +const renderDocs = (markdownSource, slug) => { + const env = {}; + const tokens = md.parse(markdownSource, env); + + // Extract the header + let headerHTML = "## file did not contain header ##"; + let headerText = headerHTML; + const headerStart = tokens.findIndex( + (token) => token.type === "heading_open" && token.tag === "h1" + ); + const headerEnd = tokens.findIndex( + (token) => token.type === "heading_close" && token.tag === "h1" + ); + if (headerStart !== -1 && headerEnd !== -1) { + const headerTokens = tokens.splice( + headerStart, + headerEnd - headerStart + 1 + ); + + // Discard the header tokens themselves, but render the HTML title with any formatting + headerTokens.shift(); + headerTokens.pop(); + headerHTML = md.renderer.render(headerTokens, md.options, env); + + // We also need a no-formatting version for the title + const justTextTokens = headerTokens.filter( + (token) => token.type === "inline" + ); + headerText = md.renderer.render(justTextTokens, md.options, env); + } + + const bodyHTML = md.renderer.render(tokens, md.options, env); + + return renderTemplate(path.join(__dirname, "docs-template.ejs"), { + slug, + headerHTML, + headerText, + bodyHTML, + usesScratchBlocks: !!env.usesScratchBlocks, + }); +}; + +module.exports = renderDocs; diff --git a/development/render-template.js b/development/render-template.js index fe8afd689d..41f036b4d1 100644 --- a/development/render-template.js +++ b/development/render-template.js @@ -1,10 +1,10 @@ -const fs = require('fs'); -const ejs = require('ejs'); +const fs = require("fs"); +const ejs = require("ejs"); // TODO: Investigate the value of removing dependency on `ejs` and possibly writing our own DSL. const renderTemplate = (path, data) => { - const inputEJS = fs.readFileSync(path, 'utf-8'); + const inputEJS = fs.readFileSync(path, "utf-8"); const outputHTML = ejs.render(inputEJS, data); return outputHTML; }; diff --git a/development/server.js b/development/server.js index 4f80d73dd3..875d6f6b8f 100644 --- a/development/server.js +++ b/development/server.js @@ -1,46 +1,49 @@ -const express = require('express'); -const Builder = require('./builder'); +const express = require("express"); +const Builder = require("./builder"); let mostRecentBuild = null; -const builder = new Builder('development'); +const builder = new Builder("development"); builder.startWatcher((newBuild) => { mostRecentBuild = newBuild; }); const app = express(); -app.set('strict routing', true); -app.set('x-powered-by', false); +app.set("strict routing", true); +app.set("x-powered-by", false); app.use((req, res, next) => { // If we don't tell the browser not to cache files, it does by default, and people will get confused when // script changes aren't being applied if they don't do a reload without cache. - res.setHeader('Cache-Control', 'no-store'); - res.setHeader('Pragma', 'no-cache'); + res.setHeader("Cache-Control", "no-store"); + res.setHeader("Pragma", "no-cache"); // Prevent browser from trying to guess file types. - res.setHeader('X-Content-Type-Options', 'nosniff'); + res.setHeader("X-Content-Type-Options", "nosniff"); // We don't want this site to be embedded in frames. - res.setHeader('X-Frame-Options', 'DENY'); + res.setHeader("X-Frame-Options", "DENY"); // No need to leak Referer headers. - res.setHeader('Referrer-Policy', 'no-referrer'); + res.setHeader("Referrer-Policy", "no-referrer"); // We want all resources used by the website to be local. // This CSP does *not* apply to the extensions, just the website. - res.setHeader('Content-Security-Policy', "default-src 'self' 'unsafe-inline' data: blob:"); + res.setHeader( + "Content-Security-Policy", + "default-src 'self' 'unsafe-inline' data: blob:" + ); // Allows loading cross-origin example images and matches GitHub pages. - res.setHeader('Access-Control-Allow-Origin', '*'); + res.setHeader("Access-Control-Allow-Origin", "*"); next(); }); -app.get('/*', (req, res, next) => { +app.get("/*", (req, res, next) => { if (!mostRecentBuild) { - res.contentType('text/plain'); + res.contentType("text/plain"); res.status(500); - res.send('Build Failed; See Console'); + res.send("Build Failed; See Console"); return; } @@ -49,18 +52,14 @@ app.get('/*', (req, res, next) => { return next(); } - if (fileInBuild.getDiskPath) { - res.sendFile(fileInBuild.getDiskPath()); - } else { - res.contentType(fileInBuild.getType()); - res.send(fileInBuild.read()); - } + res.contentType(fileInBuild.getType()); + res.send(fileInBuild.read()); }); app.use((req, res) => { - res.contentType('text/plain'); + res.contentType("text/plain"); res.status(404); - res.send('404 Not Found'); + res.send("404 Not Found"); }); // The port the server runs on matters. The editor only treats port 8000 as unsandboxed. diff --git a/development/validate.js b/development/validate.js index e9386baebe..64bf9496ab 100644 --- a/development/validate.js +++ b/development/validate.js @@ -1,16 +1,22 @@ -const Builder = require('./builder'); -const Colors = require('./colors'); +const Builder = require("./builder"); +const Colors = require("./colors"); -const builder = new Builder('production'); +const builder = new Builder("production"); const errors = builder.validate(); if (errors.length === 0) { - console.log(`${Colors.GREEN}${Colors.BOLD}Validation checks passed.${Colors.RESET}`); + console.log( + `${Colors.GREEN}${Colors.BOLD}Validation checks passed.${Colors.RESET}` + ); process.exit(0); } else { - console.error(`${Colors.RED}${Colors.BOLD}${errors.length} ${errors.length === 1 ? 'file' : 'files'} failed validation.${Colors.RESET}`); - console.error(''); - for (const {fileName, error} of errors) { + console.error( + `${Colors.RED}${Colors.BOLD}${errors.length} ${ + errors.length === 1 ? "file" : "files" + } failed validation.${Colors.RESET}` + ); + console.error(""); + for (const { fileName, error } of errors) { console.error(`${Colors.BOLD}${fileName}${Colors.RESET}: ${error}`); } console.error(``); diff --git a/docs/CST1229/zip.md b/docs/CST1229/zip.md new file mode 100644 index 0000000000..ef32c21961 --- /dev/null +++ b/docs/CST1229/zip.md @@ -0,0 +1,227 @@ +# Zip + +The Zip extension allows you to read, create and edit .zip format files, including Scratch project and sprite files (.sb3, .sprite3). + +## Paths + +Most blocks in this extension work with a path format: + + - The path is relative to the current directory by default + - Directories (folders) and the filename are separated by slashes, like `folder1/folder2/file.txt` + - `..` goes to the parent directory, like `../file.txt` + - `.` goes to the current directory, like `./file.txt` + - A `/` at the very start goes to the root directory, like `/file.txt` + - A `/` at the end denotes a directory, like `folder/` + - Multiple slashes in a row or trying to go above the root directory will result in an error (usually the block doing nothing or returning the empty value) + +## Archive management blocks + +Blocks for creating and saving the current archive. Only one archive can be open at a time. + +--- + +```scratch +create empty archive :: #a49a3a +``` +Creates and opens an empty archive with nothing in it. + +--- + +```scratch +open zip from (URL v) [https://extensions.turbowarp.org] :: #a49a3a +``` + Opens a .zip (or .sb3 or .sprite3...) file. + +The type can be one of the following: + + - **URL**: A URL, which can be either a web URL or data: URL. Recommended. + - **base64**: A base64 string without the data URL header. + - **hex**: A sequence of hexadecimal bytes (like `101A1B1C`), without a separator. + - **binary**: A sequence of binary bytes (like `000000010010101001101011`), without a separator. + - **string**: Plain text. **Not recommended!** Text encoding behavior will likely break it, as it's a binary file. + +If the file is not of zip format (e.g RAR or 7z) or is password-protected, it won't be opened. Make sure to check if it loaded successfully with the archive `is open?` block. + +--- + +```scratch +(output zip type (data: URI v) compression level (6 v) :: #a49a3a) +``` +Save the zip data into a string, which can be saved with e.g the Files extension. + +The type can be one of the following: + + - **data**: URL: A base64-encoded data URL. Recommended. + - **base64**: A base64 string without the data URL header. + - **hex**: A sequence of hexadecimal bytes (like `101A1B1C`), without a separator. + - **binary**: A sequence of binary bytes (like `000000010010101001101011`), without a separator. + - **string**: Plain text. **Not recommended!** Text encoding behavior will likely break it, as it's a binary file. + +The compression level decides how much the zip is compressed. +Lower compression levels will result in a bigger file, but are faster to create. Usually, high compression levels provide diminishing returns (lesser gains the higher you go) with much slower speeds. + +A compression level of 0 (no compression) is the fastest, but will often result in a very big file. + +--- + +```scratch +close archive :: #a49a3a +``` +Closes the archive. Use after you're done working with it. + +--- + +```scratch + +``` +Returns true if an archive is open. + +## File blocks + +Blocks for working with files (and blocks that are general to both files and folders/directories.) + +--- + +```scratch + +``` +Returns if a file or directory exists or not. The slash at the end matters! If a directory named `folder` exists, `[folder] exists?` will return false, but `[folder/] exists?` will return true. + +--- + +```scratch +write file [new file.txt] content [Hello, world?] type (text v) :: #a49a3a +``` +Writes content to a file, creating the file if it doesn't exist and replacing its existing data if it does. + +The type can be one of the following: + + - **text**: Plain text. + - **URL**: A URL, which can be either a web URL or data: URL. Best for binary data (like other zip files). + - **base64**: A base64 string without the data URL header. + - **hex**: A sequence of hexadecimal bytes (like `101A1B1C`), without a separator. + - **binary**: A sequence of binary bytes (like `000000010010101001101011`), without a separator. + +--- + +```scratch +rename [hello.txt] to [hello renamed.txt] :: #a49a3a +``` +Renames a file or directory to another name. If the target file already exists, it will be overwritten. The current directory will also be updated. This block can also be used to move files to a different directory. + +--- + +```scratch +delete [hello.txt] :: #a49a3a +``` + +Deletes a file or directory (including everything in it). + +If the current directory is in that directory, it will be set to the closest existing parent directory. + +--- + +```scratch +(file [hello.txt] as (text v) :: #a49a3a) +``` +Get the contents of a file. + +The type can be one of the following: + + - **text**: Plain text. + - **data**: URL: A base64-encoded data URL. Best for binary data (like other zip files). + - **base64**: A base64 string without the data URL header. + - **hex**: A sequence of hexadecimal bytes (like `101A1B1C`), without a separator. + - **binary**: A sequence of binary bytes (like `000000010010101001101011`), without a separator. + +## File Info Blocks + +Blocks for getting and setting additional information on a file. + +--- + +```scratch +set (modified days since 2000 v) of [folder/dango.png] to [0] :: #a49a3a +``` +Set additional info on a file or directory. + +Available options: + + - **modified days since 2000**: The modification date of the file, as days since 2000. + - **unix modified timestamp**: The modification date of the file, as a Unix timestamp (milliseconds since 1970). Useful when combined with e.g the Time extension. + - **comment**: A comment on the file. Can be any text. Some programs may show this as metadata. + +--- + +```scratch +((name v) of [folder/dango.png] :: #a49a3a) +``` + Get additional info on a file or directory. + +Available options: + + - **name**: Just the name of this file (without the directories it's in). For example, the name of `/folder1/folder2/dango.png` would be `dango.png`. + - **path**: The full absolute path of this file (its name and any directories it's in). + - **folder**: Just the folders this file is in (without its filename). For example, the folder of `/folder1/folder2/dango.png` would be `/folder1/folder2/`. + - **modification date**: A human-readable version of the file's modification date. The output of this depends on the browser's language and possibly other factors. + - **long modification date**: A longer human-readable version of the file's modification date. The output of this depends on the browser's language and possibly other factors. + - **modified days since 2000**: The modification date of the file, as days since 2000. + - **unix modified timestamp**: The modification date of the file, as a Unix timestamp (milliseconds since 1970). Useful when combined with e.g the Time extension. + - **comment**: A comment on the file. Can be any text. Some programs may show this as metadata. + +## Directory blocks + +Blocks that deal with directories and the current directory. + +--- + +```scratch +create directory [new folder] :: #a49a3a +``` +Creates a directory with a name. This can create multiple directories at once (by having multiple directores in the path, like `/new folder1/new folder2/new folder3/`). + +--- + +```scratch +go to directory [folder] :: #a49a3a +``` +Moves the current directory (the default origin of most file operations) to the specified directory. If it doesn't exist, this block will do nothing. Use `..` to go to the previous (parent) directory, and `/` to go to the root. + +--- + +```scratch +(contents of directory [.] :: #a49a3a) +``` +Returns a list of files in a directory, as JSON (which you can parse with the JSON extension). + +--- + +```scratch +current directory path :: #a49a3a +``` +Returns the absolute path to the current directory. + +## Other blocks + +Miscellaneous stuff. + +--- + +```scratch +set archive comment to [any text] :: #a49a3a +``` +Sets the archive's comment to some text. Just like file comments, this is saved and may be displayed as metadata by some programs. + +--- + +```scratch +archive comment :: #a49a3a +``` +Returns the archive's comment. + +--- + +```scratch +(path [../folder3/] from [/folder/folder2] :: #a49a3a) +``` +Returns an absolute path from an origin path and a target path. Does not depend on the archive, so it works without one open. This is mostly a utility used internally, but it might be useful for users too. diff --git a/docs/CubesterYT/TurboHook.md b/docs/CubesterYT/TurboHook.md new file mode 100644 index 0000000000..4f93b66743 --- /dev/null +++ b/docs/CubesterYT/TurboHook.md @@ -0,0 +1,44 @@ +# TurboHook + +This extension allows you to post to webhooks from some common third-party websites or programs. + +## WebHook block + +Posts a message to a webhook. + +```scratch +webhook data: [] webhook url: [] :: #3c48c2 +``` +The empty area is where you insert the data reporter and/or connector reporter. The string area is where you put your webhook's URL. + +## Data Reporter + +The Menu area of the Data Reporter currently has three options, ("content","name","icon"). + +### Content + +```scratch +((content v) [] :: #3c48c2) +``` +When choosing "content", you can type in the text you want to send as the body of the webhook. + +### Name + +```scratch +((name v) [] :: #3c48c2) +``` +When choosing "name", you can type in the name of the name you want the webhook to display as on the third-party website. + +### Icon + +```scratch +((icon v) [] :: #3c48c2) +``` +When choosing "icon", you can type in a URL to the image you want to set as the profile picture of the webhook. + +## Connector Reporter + +```scratch +([], [] :: #3c48c2) +``` +This reporter connects two different data reporters together and/or another connector reporter so you can send multiple pieces of data to the webhook. diff --git a/docs/DNin/wake-lock.md b/docs/DNin/wake-lock.md new file mode 100644 index 0000000000..000f7f8ea2 --- /dev/null +++ b/docs/DNin/wake-lock.md @@ -0,0 +1,26 @@ +# Wake Lock + +The Wake Lock feature allows you to keep your computer's screen on while a project is running. This can be helpful when playing media or performing an important but time-consuming task. + +## Activating and Releasing Wake Lock + +```scratch +set wake lock to [on v] :: #0FBD8C +``` +This block will activate wake lock. + +If you ever need to check that wake lock has properly been activated, use the `is wake lock active?` boolean reporter. + +To release wake lock, simply change "on" to "off". + +You can also insert boolean reporters into the menu input. + +Wake lock will also be released automatically when the project stops or is restarted to ensure it isn't accidentally left on forever. + +## Browser support + +Not all browsers support wake lock (notably, Firefox does not). In these browsers requesting wake lock will not do anything. + +## Note + +The wake lock block takes a moment to finish running as it activates wake lock, so if you put it in a script with other blocks, it will yield briefly, so try keeping it separate from your other scripts. The `is wake lock active?` boolean reporter, however, does not have a delay. diff --git a/docs/Lily/Skins.md b/docs/Lily/Skins.md new file mode 100644 index 0000000000..2bd8771766 --- /dev/null +++ b/docs/Lily/Skins.md @@ -0,0 +1,109 @@ +# Skins + +This extension allows you to load and display images onto sprites, as Skins. + +In this extension, a "Skin" is an image that can replace what a sprite looks like. + +Unlike costumes, Skins are not loaded when the project is opened. Instead, Skins are loaded with blocks as the project is running. + +## Loading Skins + +Skins can be created in 3 different ways. Each way requires you to give the skin a name, which will be used by other blocks to reference the skin later. + +Loading a skin with the same name as another skin will overwrite the data of that skin. Any sprite that was using this skin will now show the new skin. + +--- + +```scratch +create SVG skin [] as [my skin] :: #6b56ff +``` +The first way is by creating a new skin with SVG markup data. The advantage to this is that it loads much quicker than the other loading blocks. The obvious disadvantage is that, unlike the other 2 blocks, it can only work with SVGs. + +--- + +```scratch +load skin from (costume 1 v) as [my skin] :: #6b56ff +``` +The second way is by loading a skin from a costume. + +It's important to note that this block will require the Advanced Option "Remove raw asset data after loading to save RAM" to be disabled in the packager in order for this block to work correctly in a packaged environment. **You do not need to do this within the editor.** + +If you intend to package your project, we don't encourage using this block for that reason. **None of the other blocks in this extension require this option to be disabled.** + +--- + +```scratch +load skin from URL [https://...] as [my skin] :: #6b56ff +``` +The final way is loading a skin through a URL. This block allows you to load any bitmap image as well as SVGs. + +```scratch +load skin from URL (snapshot stage :: #9966ff) as [my skin] :: #6b56ff +``` +While this block can work with a website URL, it's primarily designed to work with data URIs. Try using this with the "snapshot stage" block from the "Looks Plus" extension. + +For the final 2 blocks, the block will pause the script for a moment in order to load the skin. Treat them like "wait" blocks in your scripts, don't expect them to finish instantaneously. + +## Using Skins + +```scratch +set skin of (myself v) to [my skin] :: #6b56ff +``` +Skins can be applied to a sprite with this block, so long as you loaded the skin beforehand. Skins can be applied to multiple sprites/clones. + +Using the "myself" option will apply the skin to the sprite the block is running in: if the block is running in a clone, it will apply the skin to the clone. **Do not confuse "myself" with the sprite's name.** + +Skins will automatically be removed from every sprite when the project has stopped. + +--- + +```scratch +restore skin of (myself v) :: #6b56ff +``` +You can remove the skin of a sprite with the "restore skin" block. This will remove the skin from that specific sprite. + +--- + +```scratch +restore targets with skin [my skin] :: #6b56ff +``` +You can remove a skin from every sprite that has it applied with the "restore targets with skin" block. "Target" refers to "sprite" in this context. + +## Deleting Skins + +Skins that have been loaded will still exist after the project has stopped. In order to truly delete a skin, you have 2 methods. + +--- + +```scratch +delete skin [my skin] :: #6b56ff +``` +Delete a specified skin, and reset any sprite that had it applied. + +--- + +```scratch +delete all skins :: #6b56ff +``` +Delete every skin that has been loaded and reset all sprites that had any skin applied. + +## Other Blocks + +```scratch + +``` +Check whether a skin is actually loaded. This becomes true **after** the block has finished loading the skin. + +--- + +```scratch +((width v) of [my skin] :: #6b56ff) +``` +Get the width/height of a skin. The values are rounded. + +--- + +```scratch +(current skin of (myself v) :: #6b56ff) +``` +The name of the skin that is applied to the specified sprite. \ No newline at end of file diff --git a/docs/ar.md b/docs/ar.md new file mode 100644 index 0000000000..2946f5c96d --- /dev/null +++ b/docs/ar.md @@ -0,0 +1,344 @@ +# Augmented Reality + +## Requirements + + - [ARCore](https://play.google.com/store/apps/details?id=com.google.ar.core) + - browser with [WebXR API and immersive-ar](https://immersive-web.github.io/webxr-samples/report/) session type support + +At the moment of writing, only Chromium-based browsers support immersive-ar session type. + +## Other information + + - It is generally recommended to use this extension with fps set to 0, which in turbowarp means running project at the screen refresh rate. + - While exiting AR mode, there is a small chance TurboWarp will lose WebGL context. This problem has also been observed on other websites with AR and likely can't be fixed. If that happens, you can save the project, refresh the tab, load the project and continue. + - It isn't possible to be in AR mode and have video sensing enabled at the same time. Entering AR disables video sensing and makes it not toggleable until AR session ends. + +## Blocks + +```scratch +enter ar mode :: #d10000 +``` +If AR is not supported and the error message haven't been shown yet, shows a popup with an error message. + +If AR is supported and project is currently not in AR, attempts to enter AR mode. While doing so, it will pause the script it is in, in the same way ask block does. It will first try to enter AR mode directly. That may fail because entering AR can only be triggred by user interaction. If it fails, it will make it so after user clicks/taps the project will attempt to enter AR. If that fails as well or either of those 2 attempts succeed it will resume the script execution. After it resumes, project may or may not be in AR mode. Projects should handle both cases. + +The origin of coordinate system is placed at or close to position of the device at the time this block was called. + +--- + +```scratch +exit ar mode :: #d10000 +``` + +If the project is in AR mode, exits it. + +--- + + +```scratch + +``` +Tells if the project is currently in AR mode. + +--- + +```scratch + +``` +Tells if AR is supported on this device. + +--- + +```scratch + +``` +Tells if AR engine knows what the current camera position and orientation is. + +After entering AR mode, is is not immediately availible as the map of the environment needs to be built first. After enough information about environment has been gathered and processed, it becomes availible. It can temporarily become unavailible due to lack of detailed features in the view of camera that are used for motion tracking, fast motion causing camera image to become too blurry or camera getting covered. + +--- + +```scratch + +``` +Tells if AR engine knows where the point of ray intersection is. + +Can become unavailible for the same reasons as [is [pose] availible?] + +--- + +```scratch +(stage width :: #d10000) +``` + +Tells stage width in scratch units. Default is `480`. + +This value may change when entering AR. + +--- + +```scratch +(stage height :: #d10000) +``` + +Tells stage height in scratch units. Default is `360`. + +This value will not change when entering AR. + +--- + +```scratch +(item [1] of [view v] matrix :: #d10000) +``` + +view matrix - is a matrix that can be used to transform points from the world space (relatively to the world origin) to the view space (with origin is at the camera). It includes rotation and translation of the camera. + +Also: + +``` +view[13] = position[x] +view[14] = position[y] +view[15] = position[z] +``` + +--- + +```scratch +(item [1] of [inverse view v] matrix :: #d10000) +``` + +inverse view matrix = view matrix-1 + +Describes the opposite transformation to the view matrix. + +--- + +```scratch +(item [1] of [projection v] matrix :: #d10000) +``` + +For perspective projection in scratch you are likely used to doing something like this: + +``` +screenX = x * dist / z +screenY = y * dist / z +``` + +With this extension it's more complicated: projection is done by first doing a 4x4 projection matrix multiplication by 4D vector `x,y,z,1`, with result being `X,Y,Z,W`, then performing division of `X,Y,Z` by `W` to get screen coordinates in range from -1 to 1, and then multiplying them by half of stage width and height to get scratch coordinates. + +The matrix is a perspective projection matrix with the assumption camera faces negative z and that screen coordinates range from -1 to 1. + +It's calculated as: + +``` + /\ + /||\ + || OUT + IN || [X,Y,Z,W] +[x,y,z,1] || + \ [ 2 * dist / width, 0, 0, 0 ] [ 1, 2, 3, 4 ] +_______\ [ 0, 2 * dist / height, 0, 0 ] [ 5, 6, 7, 8 ] +^^^^^^^/ [ 0, 0, (near + far) / (near - far), -1 ] [ 9,10,11,12 ] + / [ 0, 0, near * far / (near - far) * 2, 0 ] [ 13,14,15,16 ] + +near = depth of near plane +far = depth of far plane +``` + +Point is only visible if `-W` < `X,Y,Z` < `W`. + +``` +screenX = X / W +screenY = Y / W +screenZ = Z / W (for depth buffer) +``` + +If the first codition was true, then `-1` < `screenX,screenY,screenZ` < `1` is also always true. Note that the opposite is not always true (think of what happens when W is negative). + +To get a better understanding of this topic, you may read: + + - https://webglfundamentals.org/webgl/lessons/webgl-3d-perspective.html + - https://stackoverflow.com/questions/41085117/why-does-gl-divide-gl-position-by-w-for-you-rather-than-letting-you-do-it-your + +Projection matrix contains a lot of 0s, and can be simplified. As the result of simplifications it is possible to make it look closer to how it is usually done on scratch: + +``` +screenX = x * (2 * dist / width) / -z * (width / 2) +screenY = y * (2 * dist / height) / -z * (height / 2) + +screenX = x * (2 * dist / width) * -1 * (width / 2) / z +screenY = y * (2 * dist / height) * -1 * (height / 2) / z + +negative_dist = (2 * dist / width) * (width / -2) = (2 * dist / height) * (height / -2) + +(2 * dist / width) = projection[1] +(2 * dist / height) = projection[6] + +negative_dist = projection[1] * (width / -2) = projection[6] * (height / -2) +screenX = x * negative_dist / z +screenY = y * negative_dist / z +``` + +--- + +```scratch +(item [1] of [combined v] matrix :: #d10000) +``` +combined matrix = projection matrix * inverse view matrix + +--- + +```scratch +(position [x v] :: #d10000) +``` +```scratch +(position [y v] :: #d10000) +``` +```scratch +(position [z v] :: #d10000) +``` +Camera position relatively to the world origin. + +--- + +```scratch +(orientation [r v] :: #d10000) +``` +```scratch +(orientation [i v] :: #d10000) +``` +```scratch +(orientation [j v] :: #d10000) +``` +```scratch +(orientation [k v] :: #d10000) +``` +Camera orientation represented as quaternion. + +--- + +```scratch +(hit position [x v] :: #d10000) +``` +```scratch +(hit position [y v] :: #d10000) +``` +```scratch +(hit position [z v] :: #d10000) +``` +A ray that originates from camera in the direction the camera is facing (center of the screen) intersects the first detected real world surface. This block returns the coordinates of that intersection point. + +--- + +```scratch +move everything by x: [0] y: [0] z: [0] :: #d10000 +``` +Moves the coordinate system by specified amount. + +It can also be understood as switching to a new coordinate system with origin at a given location in the current coordinate system. + +**Usage example:** + +After starting the project, it may be a good idea to give user a way to pick location for AR content and then perform this before starting the main AR game/animation/etc. + +```scratch +move everything by x: (hit position [x v] :: #d10000) y: (hit position [y v] :: #d10000) z: (hit position [z v] :: #d10000) :: #d10000 +``` + +--- + +```scratch +turn everything by r: [0] i: [0] j: [0] k: [0] :: #d10000 +``` + +Internally it also does quaternion normalization, so you don't have to worry about doing it yourself. + +**Usage example:** + +This script can be used to rotate `XZ` around `Y`-axis: + +```scratch +turn everything by r: ([cos v] of ((angle) / [2])) i: [0] j: ([sin v] of ((angle) / [2])) k: [0] :: #d10000 +``` + +--- + +```scratch +set resolution [1] :: #d10000 +``` + +accepts values from `0.1` to `1`. changes the resolution at which the project is rendered + + - `1` - is native screen resolution + - `0.5` - half the screen resolution + - `0.1` - one tenth of screen resolution + +Reducing resolution can improve performance and reduce memory usage. + +## Examples + +### Example 1 + +```scratch +when flag clicked +enter AR mode :: #d10000 +repeat until > + clear + point at [0.1] [0.5] [-0.3] +end + +define point at (x) (y) (z) +set [X v] to ((((x) * (item [1] of [combined v] matrix :: #d10000)) + ((y) * (item [5] of [combined v] matrix :: #d10000))) + (((z) * (item [9] of [combined v] matrix :: #d10000)) + (item [13] of [combined v] matrix :: #d10000))) +set [Y v] to ((((x) * (item [2] of [combined v] matrix :: #d10000)) + ((y) * (item [6] of [combined v] matrix :: #d10000))) + (((z) * (item [10] of [combined v] matrix :: #d10000)) + (item [14] of [combined v] matrix :: #d10000))) +set [Z v] to ((((x) * (item [3] of [combined v] matrix :: #d10000)) + ((y) * (item [7] of [combined v] matrix :: #d10000))) + (((z) * (item [11] of [combined v] matrix :: #d10000)) + (item [15] of [combined v] matrix :: #d10000))) +set [W v] to ((((x) * (item [4] of [combined v] matrix :: #d10000)) + ((y) * (item [8] of [combined v] matrix :: #d10000))) + (((z) * (item [12] of [combined v] matrix :: #d10000)) + (item [16] of [combined v] matrix :: #d10000))) +if <([abs v] of (X)) < (W)> then + if <([abs v] of (Y)) < (W)> then + if <([abs v] of (Z)) < (W)> then + go to x: (((X) / (W)) * ((stage width :: #d10000) / [2])) y: (((Y) / (W)) * ((stage height :: #d10000) / [2])) + pen down + pen up + end + end +end +``` + +### Example 2 + +```scratch +when flag clicked +enter AR mode :: #d10000 +repeat until > + clear + View matrix and camera coords + point at [0.1] [0.5] [-0.3] +end + +define View matrix and camera coords +set [MXX v] to (item [1] of [view v] matrix :: #d10000) +set [MXY v] to (item [5] of [view v] matrix :: #d10000) +set [MXZ v] to ([0] - (item [9] of [view v] matrix :: #d10000)) +set [MYX v] to (item [2] of [view v] matrix :: #d10000) +set [MYY v] to (item [6] of [view v] matrix :: #d10000) +set [MYZ v] to ([0] - (item [10] of [view v] matrix :: #d10000)) +set [MZX v] to (item [3] of [view v] matrix :: #d10000) +set [MZY v] to (item [7] of [view v] matrix :: #d10000) +set [MZZ v] to ([0] - (item [11] of [view v] matrix :: #d10000)) +set [camX v] to (item [13] of [view v] matrix :: #d10000) +set [camY v] to (item [14] of [view v] matrix :: #d10000) +set [camZ v] to (item [15] of [view v] matrix :: #d10000) +set [dist v] to ((item [6] of [projection v] matrix :: #d10000) * ((stage height :: #d10000) / [2])) + +define point at (x) (y) (z) +set [x2 v] to ((x) - (camX)) +set [y2 v] to ((y) - (camY)) +set [z2 v] to ((z) - (camZ)) +set [rotX v] to ((((x2) * (MXX)) + ((y2) * (MYX))) + ((z2) * (MZX))) +set [rotY v] to ((((x2) * (MXY)) + ((y2) * (MYY))) + ((z2) * (MZY))) +set [rotZ v] to ((((x2) * (MXZ)) + ((y2) * (MYZ))) + ((z2) * (MZZ))) +if <(rotZ) > [0]> then + go to x: (((rotX) * (dist)) / (rotZ)) y: (((rotY) * (dist)) / (rotZ)) + pen down + pen up +end +``` diff --git a/docs/bitwise.md b/docs/bitwise.md new file mode 100644 index 0000000000..979da2dcf1 --- /dev/null +++ b/docs/bitwise.md @@ -0,0 +1,149 @@ +# Bitwise + +This extension allows you to perform bit shifts and logic operations on integers as if they were encoded in binary. These operations can be used to create programmer calculators and more. + +The extension uses signed 32-bit integers, which can range from -2,147,483,648 to 2,147,483,647. + +## Blocks + +```scratch + +``` +Returns true if the input is a binary number. Binary numbers can only contain 0s and 1s, so this essentially tells you if the input only contains 0s and 1s. + +--- + +```scratch +([32] to binary :: #17cde6) +``` + +Converts the input from decimal to binary and returns the result. + +Note: All other blocks except for `is () binary?` and `() to number` will treat this result as a decimal number, so make sure to convert it back to decimal before you perform operations on it. + +--- + +```scratch +([0000000000100000] to number :: #17cde6) +``` + +Converts the input from binary to decimal and returns the result. + +--- + +```scratch +([x] >> [y] :: #17cde6) +``` +Arithmatically shifts each bit of the binary representation of _x_ to the right _y_ times and returns the result. + +Example: `32` → `100000` → `010000` → `16` + +The sign will be preserved, so negative numbers will stay negative. + +This is essentially the same thing as dividing by 2^*y* and rounding down to the nearest integer. + +--- + +```scratch +([x] \<\< [y] :: #17cde6) +``` +Arithmatically shifts each bit of the binary representation of _x_ to the left _y_ times and returns the result. + +Example: `32` → `0100000` → `1000000` → `64` + +The sign will be preserved, so negative numbers will stay negative. + +This is essentially the same thing as multiplying by 2^*y*. + +--- + +```scratch +([x] >>> [y] :: #17cde6) +``` +Logically shifts each bit of the binary representation of _x_ to the right _y_ times and returns the result. + +This also moves the sign bit, so it can result in negative numbers becoming positive. + +--- + +```scratch +([x] ↻ [y] :: #17cde6) +``` +Shifts each bit of the binary representation of _x_ to the right _y_ times. Numbers shifted past one end will reappear on the other (circle around). + +Example: `3` → `00000000000000000000000000000011` → `10000000000000000000000000000001` → `-2147483647` + +--- + +```scratch +([x] ↺ [y] :: #17cde6) +``` +Shifts each bit of the binary representation of _x_ to the left _y_ times. Numbers shifted past one end will reappear on the other (circle around). + +Example: `-2147483647` → `10000000000000000000000000000001` → `00000000000000000000000000000011` → `3` + +--- + +```scratch +([x] and [y] :: #17cde6) +``` +Logically ANDs the binary representation of the inputs together and returns the result. + +Example: + +``` + 14 1110 +AND 7 AND 0111 +—————————————————— + 6 0110 +``` + +--- + +```scratch +([x] or [y] :: #17cde6) +``` + +Logically ORs the binary representation of the inputs together and returns the result. + +Example: + +``` + 14 1110 +OR 7 OR 0111 +—————————————————— + 15 1111 +``` + +--- + +```scratch +([x] xor [y] :: #17cde6) +``` + +Logically XORs the binary representation of the inputs together and returns the result. + +Example: + +``` + 14 1110 +XOR 3 XOR 0011 +—————————————————— + 13 1101 +``` + +--- + +```scratch +(not [x] :: #17cde6) +``` + +Flips all bits of the binary representation of _x_ and returns the result. + +Example: + +``` +NOT 42 NOT ...00101010 +————————————————————————— + -43 ...11010101 +``` diff --git a/docs/box2d.md b/docs/box2d.md new file mode 100644 index 0000000000..c4ac9f5abf --- /dev/null +++ b/docs/box2d.md @@ -0,0 +1,175 @@ +# Box2D + +This extension allows you to easily implement proper physics using a physics library called Box2D. + +## World + +Adjust the physics of all sprites. + +```scratch +setup stage [boxed stage v] :: #0FBD8C +``` +Choose a type of containment to keep sprites within the stage. + + - Boxed stage: Keeps sprites from going off the bottom and sides. + - Open (with floor): Keeps sprites from going off the bottom. + - Open (no floor): Removes all boundaries; sprites can go wherever they want. + +--- + +```scratch +set gravity to x: [0] y: [-10] :: #0FBD8C +``` +Change the direction and strength of gravity. + +--- + +```scratch +step simulation :: #0FBD8C +``` + +Move forward in time by one step. Run this in a loop to keep the physics going. + +## Sprites + +Manipulate individual sprites. + +```scratch +enable for [this costume v] mode [normal v] :: #0FBD8C +``` + +Make physics apply to this sprite. It can also collide with other sprites that have physics enabled. + + - Enable for this costume: Enable physics only for the current sprite or clone. + - Enable for this circle: Enable physics for the current sprite or clone as if it were shaped like a circle. + - Enable for all sprites: Enable physics for all sprites. + +Precision mode will make the sprite work extra hard to make sure it doesn't overlap with anything. Note that this can decrease performance and even cause the project to get stuck, so use with care. + +--- + +```scratch +go to x: [0] y: [0] [in world v] :: #0FBD8C +``` + +Make the sprite go to the specified location. + + - In world: Relative to the center of the world. + - On stage: Relative to the center of the screen (if you've scrolled it). + - Relative: Relative to itself. + +--- + +```scratch +set velocity to sx: [0] sy: [0] :: #0FBD8C +``` + +Set the velocity (speed) of the sprite to the specified value. + +You can get the velocity of the current sprite with the (x velocity) and (y velocity) reporters. + +--- + +```scratch +push with force [25] in direction [0] :: #0FBD8C +``` + +Directly send the sprite flying in a certain direction, adding on to its current velocity. + +--- + +```scratch +set angular velocity to [30] :: #0FBD8C +``` +Set the angular velocity (rotational speed) of the sprite to the specified value. + +You can get the angular velocity of the current sprite with the (angular velocity) reporter. + +--- + +```scratch +spin with force [500] :: #0FBD8C +``` +Directly send the sprite spinning, adding on to its current angular (rotational) velocity. + +--- + +```scratch +set fixed [fixed in place v] :: #0FBD8C +``` +Choose whether the sprite is fixed in place or can move around. + +You can tell if the sprite is currently fixed in place with the (fixed?) reporter. + +--- + +```scratch +set density [normal v] :: #0FBD8C +``` +Set the sprite's density, which affects how heavy it is. + +You can get the sprite's current density with the (density) reporter. + +--- + +```scratch +set friction [normal v] :: #0FBD8C +``` +Set the sprite's roughness. Smoother settings make the sprite slipperier. + +You can get the sprite's current friction with the (friction) reporter. + +--- + +```scratch +set bounce [normal v] :: #0FBD8C +``` +Set the sprite's bounciness. + +You can get the sprite's current bounciness with the (bounce) reporter. + +--- + +```scratch +(touching [any v] :: #0FBD8C) +``` +Returns what other sprites the sprite is touching. Also includes the edges of the stage. + +If there are multiple sprites touching, it will return a comma-separated list of them. + +## Screen + +Move the camera around the world. + +```scratch +set scroll x: [0] y: [0] :: #0FBD8C +``` + +Scroll to the desired location. + +You can get the current screen position with the (x scroll) and (y scroll) reporters. + +This will not affect the world boundaries set by the [setup stage] block. + +## Example + +```scratch +when flag clicked +setup stage [boxed stage v] :: #0FBD8C +set gravity to x: [0] y: [-10] :: #0FBD8C +create clone of (myself v) +repeat [20] + step simulation :: #0FBD8C +end +create clone of (myself v) +forever + step simulation :: #0FBD8C +end + +when I start as a clone +show +enable for [this costume v] mode [normal v] :: #0FBD8C +go to x: [-150] y: [240] [in world v] :: #0FBD8C +set bounce [quite bouncy v] :: #0FBD8C +set angular velocity to [-4.1] :: #0FBD8C +``` diff --git a/docs/local-storage.md b/docs/local-storage.md new file mode 100644 index 0000000000..7969e6a33d --- /dev/null +++ b/docs/local-storage.md @@ -0,0 +1,125 @@ +# Local Storage + +This extension allows you to automatically save plain text in storage. Forget save codes! With this extension, we can make a game that doesn't require any user interaction to save progress. + +## Namespaces + +The namespace is basically like the file you want to read and write to. Each project should set this to something unique, such as developer/project title so each project gets its own storage space. If two different projects use the same namespace, they will overwrite eachother's data and cause very bad things! + +You can set which namespace to use with this block: + +```scratch +set storage namespace ID to [project title] :: #0FBD8C +``` + +Some example namespaces: + + - `griffpatch/Paper Minecraft` + - `Untitled-37 by TestMuffin` + +The format isn't important -- it just needs to be unique. Don't include a version number like "v1.4" in the namespace unless you want to discard data from old versions of your project. + +## Reading and writing + +After setting the namespace, you're ready to read from and write to storage. + +You can store data in keys, which are kind of like names of variables in any Scratch project, except they persist between sessions. + +You can store data in storage keys with this block: + +```scratch +set key [score] to [1000] :: #0FBD8C +``` + +And read it with: + +```scratch +(get key [score] :: #0FBD8C) +``` + +If you want to delete a key and its value, use this block: + +```scratch +delete key [score] :: #0FBD8C +``` + +Or wipe everything stored in the namespace: + +```scratch +delete all keys :: #0FBD8C +``` + +## Loading data into memory + +Relying on the disk to read information that gets saved such as a player's progress or stats can be pretty slow. That's why it's very useful to store this data in variables while it's in use. Variables are like your project's random-access memory. One way of doing this is by getting all keys from storage and putting their values in variables as part of your project's initialization process. + +If you're unfamiliar with computer memory, think of reading local storage as opening up your dictionary to look for a definition for someone and reading memory as remembering the definition you just read and saying it to someone. Of course, you wouldn't want to be constantly running to get the dictionary every time the same definition was requested. + +While browsers actually hold this data in memory automatically for quicker access, it's still more efficient and a better practice to take some work off the browser by not constantly getting the same storage key over and over again for no reason, and is much more important to know and think about if you ever use other programming languages where you don't want the disk to have to spend time reading the same data over and over again when it could be kept in memory (that's what memory is for). + +For example, in a game, you can speed up code that needs to know how many coins the player has collected by getting the storage key for coins and then loading it into a variable, just once on startup. + +```scratch +forever + if <(get key [coins] :: #0FBD8C) > [99]> then // Don't do this + broadcast (1-UP v) + end +end + +set [coins v] to (get key [coins] :: #0FBD8C) // Do this instead +forever + if <(coins) > [99]> then + broadcast (1-UP v) + end +end +``` + +So in general, we don't really need to load data from storage again once we have it in memory. If we already know what's in storage and what we're writing to storage because that data is already present in the project's variables, we really only need to read from storage once to initialize and then we're good. + +...Right? + +## Handling interference from other windows + +Sometimes a user may open the same project in multiple tabs or windows, each of which could be trying to read and write data to the same space. If this causes a desync, it can result in unexpected behavior. + +Here's an example scenario. Suppose someone opens the same game twice by accident. They play in Window A for a while and save the game. Then they close that window and do something else. Later, they come back to the other window they had opened before and start playing in Window B, but all the progress is "gone" because that window had already been running the game and had already loaded the save data before Window A had saved the progress that was made, and it's too late because they had saved the game in Window B before they realized the problem. This is unfortunate for the player, but more importantly for you, the developer, what is the project supposed to do now? Mix the data? + +So you can see how it's a good idea to consider that people may have multiple windows of the same project open intentionally or accidentally. + +You don't have to do things like auto-refresh content if you didn't intend for your project to support multi-window usage, but it's nice to at least make sure no glitches happen if someone accidentally had multiple windows open. Here are a few ways you can deal with this problem: + +## Initialize by loading from storage + +For games and other projects that are intended for use only in one window, we recommend you load all storage keys you need into variables only when the project starts so that nothing changes if a second instance of the project writes to storage while you're still using the first one. Then, when you need to save data, rewrite everything in the same group of data (like everything in the same save file in a game) to storage at the same time so nothing from other instances of the project gets mixed in. + +The worst that could happen with this implementation is that "data A" might be overwritten by "data B", if the user made a mistake. + +## Reload data from storage as needed + +Sometimes you might want to respond to changes in local storage. There is a block to help with this: + +```scratch +when another window changes storage :: hat #0FBD8C +``` + +The code under this block will run whenever a different instance of the project writes to storage or if a different project that's using the same namespace writes to storage. This allows your project to properly respond to and handle these events as they happen. + +This may or may not be important. You probably wouldn't want to use this technique in a game - games don't need to respond to other instances of themselves writing to storage. Plus, if game save data from one window is mixed in with data from another, it can cause glitches like sequence breaks. + +This kind of thing is more useful if you made something like a file system simulator that you want to have auto-refresh if the user makes edits in other windows, you may want to use this so that the content being displayed stays up to date even if another window modifies it. + +You could do this by constantly getting the storage key, but it's better to only grab keys from storage when necessary. The block above is how you do that. + +## Merge the data + +This one is more advanced and the way you would code it depends on the project, but you could make it so that when you're about to save data and another window wrote data that was not loaded into the first window, the two are merged - the data being written is merged with the data that was already present, so that if you collected 100 coins in one session and 100 XP in another, both of those changes in the save data would stay. + +Make sure to do this correctly because if data is merged incorrectly, it can cause glitches like sequence breaks in a game. + +I said that this is advanced, because sometimes these algorithms can get confused when merging changes to the same piece of data. (Like, what are we supposed to do if we're trying to merge two changes, one of which changes "A" to "B" and the other changes the same "A" to "C"?) This is known as a merge conflict. If you don't have any way to prioritize one change over another, you'll just be stuck with two branches of data. + +## Local storage limits + +This extension uses the browser's local storage API, which limits each website to around 5 MiB or 5,242,800 bytes of local storage data, so if we want local storage to be able to hold data for many projects, each one should stay well below this limit. We recommend only storing small files such as game save data or settings in local storage. + +In rare instances, such as when a system is running out of disk space, the browser may delete our data to make room for something else. We, unfortunately, cannot influence when this happens. diff --git a/extensions/-SIPC-/consoles.js b/extensions/-SIPC-/consoles.js index ccfa67059f..3454a518e1 100644 --- a/extensions/-SIPC-/consoles.js +++ b/extensions/-SIPC-/consoles.js @@ -1,185 +1,190 @@ -(function (Scratch) { - 'use strict'; - const icon = ''; - const icon2 = ''; - class Consoles { - constructor () {} - getInfo() { - return { - id: 'sipcconsole', - name: 'Consoles', - color1: '#808080', - color2: '#8c8c8c', - color3: '#999999', - menuIconURI: icon, - blockIconURI: icon2, - blocks: [ - { - opcode: 'Emptying', - blockType: Scratch.BlockType.COMMAND, - text: 'Clear Console', - arguments: {} - }, - { - opcode: 'Information', - blockType: Scratch.BlockType.COMMAND, - text: 'Information [string]', - arguments: { - string: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'Information' - } - } - }, - { - opcode: 'Journal', - blockType: Scratch.BlockType.COMMAND, - text: 'Journal [string]', - arguments: { - string: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'Journal' - } - } - }, - { - opcode: 'Warning', - blockType: Scratch.BlockType.COMMAND, - text: 'Warning [string]', - arguments: { - string: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'Warning' - } - } - }, - { - opcode: 'Error', - blockType: Scratch.BlockType.COMMAND, - text: 'Error [string]', - arguments: { - string: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'Error' - } - } - }, - { - opcode: 'debug', - blockType: Scratch.BlockType.COMMAND, - text: 'Debug [string]', - arguments: { - string: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'Debug' - } - } - }, - +// Name: Consoles +// ID: sipcconsole +// Description: Blocks that interact the JavaScript console built in to your browser's developer tools. +// By: -SIPC- - '---', - { - opcode: 'group', - blockType: Scratch.BlockType.COMMAND, - text: 'Create a group named [string]', - arguments: { - string: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'group' - } - } - }, - { - opcode: 'groupCollapsed', - blockType: Scratch.BlockType.COMMAND, - text: 'Create a collapsed group named [string]', - arguments: { - string: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'group' - } - } - }, - { - opcode: 'groupEnd', - blockType: Scratch.BlockType.COMMAND, - text: 'Exit the current group', - arguments: {} - }, - '---', - { - opcode: 'Timeron', - blockType: Scratch.BlockType.COMMAND, - text: 'Start a timer named [string]', - arguments: { - string: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'Time' - } - } - }, - { - opcode: 'Timerlog', - blockType: Scratch.BlockType.COMMAND, - text: 'Print the time run by the timer named [string]', - arguments: { - string: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'Time' - } - } - }, - { - opcode: 'Timeroff', - blockType: Scratch.BlockType.COMMAND, - text: 'End the timer named [string] and print the time elapsed from start to end', - arguments: { - string: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'Time' - } - } - }, - ] - }; - } - Emptying () { - console.clear(); - } - Information ({string}) { - console.info(string); - } - Journal ({string}) { - console.log(string); - } - Warning ({string}) { - console.warn(string); - } - Error ({string}) { - console.error(string); - } - debug ({string}) { - console.debug(string); - } - group({string}) { - console.group(string); - } - groupCollapsed({string}) { - console.groupCollapsed(string); - } - groupEnd() { - console.groupEnd(); - } - Timeron ({string}) { - console.time(string); - } - Timerlog ({string}) { - console.timeLog(string); - } - Timeroff ({string}) { - console.timeEnd(string); - } +(function (Scratch) { + "use strict"; + const icon = + ""; + const icon2 = + ""; + class Consoles { + constructor() {} + getInfo() { + return { + id: "sipcconsole", + name: "Consoles", + color1: "#808080", + color2: "#8c8c8c", + color3: "#999999", + menuIconURI: icon, + blockIconURI: icon2, + blocks: [ + { + opcode: "Emptying", + blockType: Scratch.BlockType.COMMAND, + text: "Clear Console", + arguments: {}, + }, + { + opcode: "Information", + blockType: Scratch.BlockType.COMMAND, + text: "Information [string]", + arguments: { + string: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Information", + }, + }, + }, + { + opcode: "Journal", + blockType: Scratch.BlockType.COMMAND, + text: "Journal [string]", + arguments: { + string: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Journal", + }, + }, + }, + { + opcode: "Warning", + blockType: Scratch.BlockType.COMMAND, + text: "Warning [string]", + arguments: { + string: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Warning", + }, + }, + }, + { + opcode: "Error", + blockType: Scratch.BlockType.COMMAND, + text: "Error [string]", + arguments: { + string: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Error", + }, + }, + }, + { + opcode: "debug", + blockType: Scratch.BlockType.COMMAND, + text: "Debug [string]", + arguments: { + string: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Debug", + }, + }, + }, + "---", + { + opcode: "group", + blockType: Scratch.BlockType.COMMAND, + text: "Create a group named [string]", + arguments: { + string: { + type: Scratch.ArgumentType.STRING, + defaultValue: "group", + }, + }, + }, + { + opcode: "groupCollapsed", + blockType: Scratch.BlockType.COMMAND, + text: "Create a collapsed group named [string]", + arguments: { + string: { + type: Scratch.ArgumentType.STRING, + defaultValue: "group", + }, + }, + }, + { + opcode: "groupEnd", + blockType: Scratch.BlockType.COMMAND, + text: "Exit the current group", + arguments: {}, + }, + "---", + { + opcode: "Timeron", + blockType: Scratch.BlockType.COMMAND, + text: "Start a timer named [string]", + arguments: { + string: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Time", + }, + }, + }, + { + opcode: "Timerlog", + blockType: Scratch.BlockType.COMMAND, + text: "Print the time run by the timer named [string]", + arguments: { + string: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Time", + }, + }, + }, + { + opcode: "Timeroff", + blockType: Scratch.BlockType.COMMAND, + text: "End the timer named [string] and print the time elapsed from start to end", + arguments: { + string: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Time", + }, + }, + }, + ], + }; + } + Emptying() { + console.clear(); + } + Information({ string }) { + console.info(string); + } + Journal({ string }) { + console.log(string); + } + Warning({ string }) { + console.warn(string); + } + Error({ string }) { + console.error(string); + } + debug({ string }) { + console.debug(string); + } + group({ string }) { + console.group(string); + } + groupCollapsed({ string }) { + console.groupCollapsed(string); + } + groupEnd() { + console.groupEnd(); + } + Timeron({ string }) { + console.time(string); + } + Timerlog({ string }) { + console.timeLog(string); + } + Timeroff({ string }) { + console.timeEnd(string); } - Scratch.extensions.register(new Consoles()); + } + Scratch.extensions.register(new Consoles()); })(Scratch); diff --git a/extensions/-SIPC-/recording.js b/extensions/-SIPC-/recording.js index 40b2fd8770..2265654c94 100644 --- a/extensions/-SIPC-/recording.js +++ b/extensions/-SIPC-/recording.js @@ -1,111 +1,114 @@ -(function (Scratch) { - 'use strict'; - if (!Scratch.extensions.unsandboxed) { - throw new Error('Recording must be run unsandboxed'); - } - /** @type {MediaRecorder|null} */ - let mediaRecorder = null; - let recordedChunks = []; - const icon = ''; - class Recording { - getInfo() { - return { - id: 'sipcrecording', - name: 'Recording', - color1: '#696969', - blocks: [ - { - opcode: 'startRecording', - blockType: Scratch.BlockType.COMMAND, - text: 'Start recording', - blockIconURI: icon, - arguments: {} - }, - { - opcode: 'stopRecording', - blockType: Scratch.BlockType.COMMAND, - text: 'Stop recording', - blockIconURI: icon, - arguments: {} - }, - { - opcode: 'stopRecordingAndDownload', - blockType: Scratch.BlockType.COMMAND, - text: 'Stop recording and download with [name] as filename', - blockIconURI: icon, - arguments: { - name: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'recording.wav' - } - } - }, - { - opcode: 'isRecording', - blockType: Scratch.BlockType.BOOLEAN, - text: 'Recording?', - blockIconURI: icon, - arguments: {} - }, - ] - }; - } - async startRecording() { - recordedChunks = []; - if (!navigator.mediaDevices || !navigator.mediaDevices.getUserMedia) { - console.error('The recording function is not supported by the browser'); - return; - } - try { - if (!await Scratch.canRecordAudio()) { - throw new Error('VM denied permission'); - } - const stream = await navigator.mediaDevices.getUserMedia({ audio: true }); - mediaRecorder = new MediaRecorder(stream); - mediaRecorder.addEventListener('dataavailable', function (e) { - recordedChunks.push(e.data); - }); - mediaRecorder.start(); - console.log('Start recording'); - } catch (e) { - console.error('Could not start recording', e); - } - } - stopRecording() { - if (!mediaRecorder) { - console.error('Recording not started'); - return; - } - console.log('Stop recording'); - mediaRecorder.stop(); - mediaRecorder = null; - recordedChunks = []; - } - stopRecordingAndDownload({name}) { - if (!mediaRecorder) { - console.error('Recording not started'); - return; - } - console.log('Stop recording'); - mediaRecorder.addEventListener('stop', function () { - const blob = new Blob(recordedChunks, { type: 'audio/wav' }); - const url = URL.createObjectURL(blob); - const downloadLink = document.createElement('a'); - downloadLink.href = url; - downloadLink.download = name; - document.body.appendChild(downloadLink); - downloadLink.click(); - document.body.removeChild(downloadLink); - URL.revokeObjectURL(url); - recordedChunks = []; - }); - mediaRecorder.stop(); - mediaRecorder = null; - } - isRecording() { - return !!mediaRecorder; - } - } - Scratch.extensions.register(new Recording()); -})(Scratch); -//BY -SIPC- 502415953 +(function (Scratch) { + "use strict"; + if (!Scratch.extensions.unsandboxed) { + throw new Error("Recording must be run unsandboxed"); + } + /** @type {MediaRecorder|null} */ + let mediaRecorder = null; + let recordedChunks = []; + const icon = + ""; + class Recording { + getInfo() { + return { + id: "sipcrecording", + name: "Recording", + color1: "#696969", + blocks: [ + { + opcode: "startRecording", + blockType: Scratch.BlockType.COMMAND, + text: "Start recording", + blockIconURI: icon, + arguments: {}, + }, + { + opcode: "stopRecording", + blockType: Scratch.BlockType.COMMAND, + text: "Stop recording", + blockIconURI: icon, + arguments: {}, + }, + { + opcode: "stopRecordingAndDownload", + blockType: Scratch.BlockType.COMMAND, + text: "Stop recording and download with [name] as filename", + blockIconURI: icon, + arguments: { + name: { + type: Scratch.ArgumentType.STRING, + defaultValue: "recording.wav", + }, + }, + }, + { + opcode: "isRecording", + blockType: Scratch.BlockType.BOOLEAN, + text: "Recording?", + blockIconURI: icon, + arguments: {}, + }, + ], + }; + } + async startRecording() { + recordedChunks = []; + if (!navigator.mediaDevices || !navigator.mediaDevices.getUserMedia) { + console.error("The recording function is not supported by the browser"); + return; + } + try { + if (!(await Scratch.canRecordAudio())) { + throw new Error("VM denied permission"); + } + const stream = await navigator.mediaDevices.getUserMedia({ + audio: true, + }); + mediaRecorder = new MediaRecorder(stream); + mediaRecorder.addEventListener("dataavailable", function (e) { + recordedChunks.push(e.data); + }); + mediaRecorder.start(); + console.log("Start recording"); + } catch (e) { + console.error("Could not start recording", e); + } + } + stopRecording() { + if (!mediaRecorder) { + console.error("Recording not started"); + return; + } + console.log("Stop recording"); + mediaRecorder.stop(); + mediaRecorder = null; + recordedChunks = []; + } + stopRecordingAndDownload({ name }) { + if (!mediaRecorder) { + console.error("Recording not started"); + return; + } + console.log("Stop recording"); + mediaRecorder.addEventListener("stop", function () { + const blob = new Blob(recordedChunks, { type: "audio/wav" }); + const url = URL.createObjectURL(blob); + const downloadLink = document.createElement("a"); + downloadLink.href = url; + downloadLink.download = name; + document.body.appendChild(downloadLink); + downloadLink.click(); + document.body.removeChild(downloadLink); + URL.revokeObjectURL(url); + recordedChunks = []; + }); + mediaRecorder.stop(); + mediaRecorder = null; + } + isRecording() { + return !!mediaRecorder; + } + } + Scratch.extensions.register(new Recording()); +})(Scratch); +//BY -SIPC- 502415953 diff --git a/extensions/-SIPC-/time.js b/extensions/-SIPC-/time.js index 778f94c0c5..cd689f4e40 100644 --- a/extensions/-SIPC-/time.js +++ b/extensions/-SIPC-/time.js @@ -1,118 +1,345 @@ +/* Name: Time +* ID: sipctime +* Old Description: Blocks for interacting with unix timestamps and other date strings. +* Updated Description: Blocks for Time, Date, and Time Zone calculations and interactions. +* Main Extension By: -SIPC- +* Calculation Blocks By: SharkPool +*/ + (function (Scratch) { - 'use strict'; - const icon = ''; - const icon2 = ''; - class Time { - getInfo() { - return { - id: 'sipctime', - name: 'Time', - color1: '#ff8000', - color2: '#804000', - color3: '#804000', - menuIconURI: icon, - blockIconURI: icon2, - blocks: [ - { - opcode: 'Timestamp', - blockType: Scratch.BlockType.REPORTER, - text: 'The current timestamp', - arguments: {} - }, - { - opcode: 'timezone', - blockType: Scratch.BlockType.REPORTER, - text: 'The current time zone', - arguments: {} - }, - { - opcode: 'Timedata', - blockType: Scratch.BlockType.REPORTER, - text: 'Extract [Timedata] from [timestamp]', - arguments: { - timestamp: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '1145141980000' - }, - Timedata: { - type: Scratch.ArgumentType.STRING, - menu: "Time", - defaultValue: 'year' - } - } - }, - { - opcode: 'TimestampToTime', - blockType: Scratch.BlockType.REPORTER, - text: 'Converts [timestamp] to a datetime', - arguments: { - timestamp: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '1145141980000' - } - } - }, - { - opcode: 'TimeToTimestamp', - blockType: Scratch.BlockType.REPORTER, - text: 'Converts [time] to a timestamp', - arguments: { - time: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '2006-04-16 06:59:40' - } - } - } - ], - menus: { - Time: { - acceptReporters: true, - items: ['year', 'month', 'day', 'hour', 'minute', 'second'] - }, - } - }; - } - Timestamp() { - return Date.now(); - } - timezone() { - return 'UTC+' + new Date().getTimezoneOffset() / -60; - } - Timedata(args) { - args.timestamp = args.timestamp ? args.timestamp : null; - let date1 = new Date(args.timestamp); - switch (args.Timedata) { - case 'year': - return date1.getFullYear(); - case 'month': - return date1.getMonth() + 1 < 10 ? '0' + (date1.getMonth() + 1) : date1.getMonth() + 1; - case 'day': - return date1.getDate() < 10 ? '0' + date1.getDate() : date1.getDate(); - case 'hour': - return date1.getHours() < 10 ? '0' + date1.getHours() : date1.getHours(); - case 'minute': - return date1.getMinutes() < 10 ? '0' + date1.getMinutes() : date1.getMinutes(); - case 'second': - return date1.getSeconds() < 10 ? '0' + date1.getSeconds() : date1.getSeconds(); + "use strict"; + + const menuIconURI = + ""; + + const blockIconURI = + ""; + + class Time { + getInfo() { + return { + id: "sipctime", + name: "Time", + color1: "#ff8000", + color2: "#804000", + color3: "#804000", + menuIconURI, + blockIconURI, + blocks: [ + { + blockType: Scratch.BlockType.LABEL, + text: 'Time Conversions', + }, + { + opcode: "Timestamp", + blockType: Scratch.BlockType.REPORTER, + text: "current timestamp", + arguments: {}, + }, + { + opcode: "timezone", + blockType: Scratch.BlockType.REPORTER, + text: "current time zone", + arguments: {}, + }, + { + opcode: "Timedata", + blockType: Scratch.BlockType.REPORTER, + text: "get [Timedata] from [timestamp]", + arguments: { + timestamp: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "1145141980000", + }, + Timedata: { + type: Scratch.ArgumentType.STRING, + menu: "Time", + defaultValue: "year", + }, + }, + }, + { + opcode: "TimestampToTime", + blockType: Scratch.BlockType.REPORTER, + text: "convert [timestamp] to datetime", + arguments: { + timestamp: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "1145141980000", + }, + }, + }, + { + opcode: "TimeToTimestamp", + blockType: Scratch.BlockType.REPORTER, + text: "convert [time] to timestamp", + arguments: { + time: { + type: Scratch.ArgumentType.STRING, + defaultValue: "2006-04-16 06:59:40", + }, + }, + }, + { + blockType: Scratch.BlockType.LABEL, + text: 'Time Calculations', + }, + { + opcode: 'calculatetimedurationfromdate', + blockType: Scratch.BlockType.REPORTER, + text: 'time duration from [DATE] to current date in [TIME_MENU]', + arguments: { + DATE: { + type: Scratch.ArgumentType.STRING, + defaultValue: '2006-04-16' + }, + TIME_MENU: { + type: Scratch.ArgumentType.STRING, + menu: 'Time', + defaultValue: 'day' + }, } - return 0; - } - TimestampToTime({ timestamp }) { - timestamp = timestamp ? timestamp : null; - let date2 = new Date(timestamp); - let Y = date2.getFullYear() + '-'; - let M = (date2.getMonth() + 1 < 10 ? '0' + (date2.getMonth() + 1) : date2.getMonth() + 1) + '-'; - let D = (date2.getDate() < 10 ? '0' + date2.getDate() : date2.getDate()) + ' '; - let h = (date2.getHours() < 10 ? '0' + date2.getHours() : date2.getHours()) + ':'; - let m = (date2.getMinutes() < 10 ? '0' + date2.getMinutes() : date2.getMinutes()) + ':'; - let s = date2.getSeconds() < 10 ? '0' + date2.getSeconds() : date2.getSeconds(); - return Y + M + D + h + m + s; - } - TimeToTimestamp({ time }) { - let data3 = time; - let timestamp = Date.parse(data3); - return timestamp; - } + }, + { + opcode: 'calculatetimedurationfromtime', + blockType: Scratch.BlockType.REPORTER, + text: 'time duration from [START_HOUR]:[START_MINUTE] to current time in [TIME_MENU]', + arguments: { + START_HOUR: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: 0 + }, + START_MINUTE: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: 0 + }, + TIME_MENU: { + type: Scratch.ArgumentType.STRING, + menu: 'Time', + defaultValue: 'hour' + }, + } + }, + { + opcode: 'calculatetimedifference', + blockType: Scratch.BlockType.REPORTER, + text: 'difference between [START_TIME] and [END_TIME] in [TIME_MENU]', + arguments: { + START_TIME: { + type: Scratch.ArgumentType.STRING, + defaultValue: '00:00' + }, + END_TIME: { + type: Scratch.ArgumentType.STRING, + defaultValue: '00:00' + }, + TIME_MENU: { + type: Scratch.ArgumentType.STRING, + menu: 'Time', + defaultValue: 'hour' + }, + } + }, + { + opcode: 'converttotime', + blockType: Scratch.BlockType.REPORTER, + text: 'convert [VALUE] to time (day:hour:minute)', + arguments: { + VALUE: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: 0 + }, + } + }, + { + opcode: 'daysinmonth', + blockType: Scratch.BlockType.REPORTER, + text: 'number of days in [MONTH] [YEAR]', + arguments: { + MONTH: { + type: Scratch.ArgumentType.STRING, + menu: 'Months', + defaultValue: 'January' + }, + YEAR: { + type: Scratch.ArgumentType.STRING, + defaultValue: '2000' + }, + } + } + ], + menus: { + Time: { + acceptReporters: true, + items: ["year", "month", "day", "hour", "minute", "second"], + }, + Months: { + acceptReporters: true, + items: [ + 'January', + 'February', + 'March', + 'April', + 'May', + 'June', + 'July', + 'August', + 'September', + 'October', + 'November', + 'December' + ] + }, + }, + }; + } + Timestamp() { + return Date.now(); + } + timezone() { + return "UTC+" + new Date().getTimezoneOffset() / -60; + } + Timedata(args) { + args.timestamp = args.timestamp ? args.timestamp : null; + let date1 = new Date(Scratch.Cast.toNumber(args.timestamp)); + switch (args.Timedata) { + case "year": + return date1.getFullYear(); + case "month": + return date1.getMonth() + 1 < 10 + ? "0" + (date1.getMonth() + 1) + : date1.getMonth() + 1; + case "day": + return date1.getDate() < 10 ? "0" + date1.getDate() : date1.getDate(); + case "hour": + return date1.getHours() < 10 + ? "0" + date1.getHours() + : date1.getHours(); + case "minute": + return date1.getMinutes() < 10 + ? "0" + date1.getMinutes() + : date1.getMinutes(); + case "second": + return date1.getSeconds() < 10 + ? "0" + date1.getSeconds() + : date1.getSeconds(); + } + return 0; + } + TimestampToTime({ timestamp }) { + timestamp = timestamp ? timestamp : null; + let date2 = new Date(timestamp); + let Y = date2.getFullYear() + "-"; + let M = + (date2.getMonth() + 1 < 10 + ? "0" + (date2.getMonth() + 1) + : date2.getMonth() + 1) + "-"; + let D = + (date2.getDate() < 10 ? "0" + date2.getDate() : date2.getDate()) + " "; + let h = + (date2.getHours() < 10 ? "0" + date2.getHours() : date2.getHours()) + + ":"; + let m = + (date2.getMinutes() < 10 + ? "0" + date2.getMinutes() + : date2.getMinutes()) + ":"; + let s = + date2.getSeconds() < 10 ? "0" + date2.getSeconds() : date2.getSeconds(); + return Y + M + D + h + m + s; + } + TimeToTimestamp({ time }) { + let data3 = time; + let timestamp = Date.parse(data3); + return timestamp; + } + + calculateTimeDifference(startDate, endDate, timeMenu) { + const timeDiff = Math.abs(endDate.getTime() - startDate.getTime()); + + switch (timeMenu) { + case 'year': + return Math.floor(timeDiff / (1000 * 60 * 60 * 24 * 365)); + case 'month': + return Math.floor(timeDiff / (1000 * 60 * 60 * 24 * 30)); + case 'day': + return Math.floor(timeDiff / (1000 * 60 * 60 * 24)); + case 'hour': + return Math.floor(timeDiff / (1000 * 60 * 60)); + case 'minute': + return Math.floor(timeDiff / (1000 * 60)); + case 'second': + return Math.floor(timeDiff / 1000); + } + } + + calculatetimedurationfromdate(args) { + const dateString = args.DATE ? args.DATE : null; + const timeMenu = args.TIME_MENU; + const startDate = new Date(dateString); + const endDate = new Date(); + return this.calculateTimeDifference(startDate, endDate, timeMenu); + } + + calculatetimedurationfromtime(args) { + const startHour = args.START_HOUR ? args.START_HOUR : null; + const startMinute = args.START_MINUTE ? args.START_MINUTE : null; + const timeMenu = args.TIME_MENU; + const startDate = new Date(); + startDate.setHours(startHour, startMinute, 0, 0); + const endDate = new Date(); + return this.calculateTimeDifference(startDate, endDate, timeMenu); + } + + calculatetimedifference(args) { + const startTime = args.START_TIME ? args.START_TIME : null; + const endTime = args.END_TIME ? args.END_TIME : null; + const timeMenu = args.TIME_MENU; + const startDate = new Date(); + const endDate = new Date(); + const startHour = parseInt(startTime.split(':')[0]); + const startMinute = parseInt(startTime.split(':')[1]); + const endHour = parseInt(endTime.split(':')[0]); + const endMinute = parseInt(endTime.split(':')[1]); + startDate.setHours(startHour, startMinute, 0, 0); + endDate.setHours(endHour, endMinute, 0, 0); + return this.calculateTimeDifference(startDate, endDate, timeMenu); + } + + converttotime(args) { + const value = args.VALUE ? args.VALUE : 0; + const seconds = Math.floor(value); + const minutes = Math.floor(seconds / 60); + const hours = Math.floor(minutes / 60); + const days = Math.floor(hours / 24); + + const timeString = `${hours % 24}:${minutes % 60}:${seconds % 60}`; + return timeString; + } + + daysinmonth(args) { + const month = args.MONTH; + const year = args.YEAR ? args.YEAR : null; + const date = new Date(year, this._getMonthIndex(month) + 1, 0); + return date.getDate(); + } + + _getMonthIndex(month) { + const months = [ + 'January', + 'February', + 'March', + 'April', + 'May', + 'June', + 'July', + 'August', + 'September', + 'October', + 'November', + 'December' + ]; + return months.indexOf(month); } - Scratch.extensions.register(new Time()); + } + Scratch.extensions.register(new Time()); })(Scratch); diff --git a/extensions/0832/rxFS.js b/extensions/0832/rxFS.js index 85d204539d..b903ed8828 100644 --- a/extensions/0832/rxFS.js +++ b/extensions/0832/rxFS.js @@ -2,243 +2,255 @@ * Made by 0832 * This file was originally under the rxLI Version 2.1 license: * https://0832k12.github.io/rxLi/2.1/ - * + * * However they have since claimed it to be "directly compatible with MIT license", * which is the license we use this file under. */ (function (Scratch) { - 'use strict'; - - var rxFSfi = new Array(); - var rxFSsy = new Array(); - var Search, i, str, str2; - - const file = ''; - const wenj = ''; - - - class rxFS { - getInfo() { - return { - id: '0832rxfs', - name: 'rxFS', - color1: '#2bdab7', - blocks: [ - { - blockIconURI: wenj, - opcode: 'start', - blockType: Scratch.BlockType.COMMAND, - text: 'New [STR] ', - arguments: { - STR: { - type: Scratch.ArgumentType.STRING, - defaultValue: '📃' - } - } - }, - { - blockIconURI: wenj, - opcode: 'file', - blockType: Scratch.BlockType.COMMAND, - text: 'Set [STR] to [STR2] ', - arguments: { - STR: { - type: Scratch.ArgumentType.STRING, - defaultValue: '📃' - }, - STR2: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'rxFS is good!' - } - } - }, - { - blockIconURI: wenj, - opcode: 'sync', - blockType: Scratch.BlockType.COMMAND, - text: 'Change the location of [STR] to [STR2] ', - arguments: { - STR: { - type: Scratch.ArgumentType.STRING, - defaultValue: '📃' - }, - STR2: { - type: Scratch.ArgumentType.STRING, - defaultValue: '📃' - } - } - }, - { - blockIconURI: wenj, - opcode: 'del', - blockType: Scratch.BlockType.COMMAND, - text: 'Delete [STR] ', - arguments: { - STR: { - type: Scratch.ArgumentType.STRING, - defaultValue: '📃' - } - } - }, - { - blockIconURI: wenj, - opcode: 'webin', - blockType: Scratch.BlockType.REPORTER, - text: 'Load [STR] from the network', - arguments: { - STR: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'https://0832k12.github.io/rxFS/hello.txt' - } - } - }, - { - blockIconURI: wenj, - opcode: 'open', - blockType: Scratch.BlockType.REPORTER, - text: 'Open [STR]', - arguments: { - STR: { - type: Scratch.ArgumentType.STRING, - defaultValue: '📃' - } - } - }, - { - blockIconURI: file, - opcode: 'clean', - blockType: Scratch.BlockType.COMMAND, - text: 'Clear file system', - arguments: {} - }, - { - blockIconURI: file, - opcode: 'in', - blockType: Scratch.BlockType.COMMAND, - text: 'Import file system from [STR]', - arguments: { - STR: { - type: Scratch.ArgumentType.STRING, - defaultValue: '📁' - } - } - }, - { - blockIconURI: file, - opcode: 'out', - blockType: Scratch.BlockType.REPORTER, - text: 'Export file system', - arguments: {} - }, - { - blockIconURI: file, - opcode: 'list', - blockType: Scratch.BlockType.REPORTER, - text: 'List the contents under the same folder [STR]', - arguments: { - STR: { - type: Scratch.ArgumentType.STRING, - defaultValue: '📁' - } - } - }, - { - blockIconURI: file, - opcode: 'search', - blockType: Scratch.BlockType.REPORTER, - text: 'Search [STR]', - arguments: { - STR: { - type: Scratch.ArgumentType.STRING, - defaultValue: '📃' - } - } - } - ] - }; - } + "use strict"; + var rxFSfi = new Array(); + var rxFSsy = new Array(); + var Search, i, str, str2; - clean() { - rxFSfi = []; - rxFSsy = []; - } + const file = + ""; + const wenj = + ""; - sync({ STR, STR2 }) { - str = btoa(unescape(encodeURIComponent(STR))); - str2 = btoa(unescape(encodeURIComponent(STR2))); - if (rxFSsy.indexOf(str) + 1 == 0) { - rxFSsy[((rxFSsy.indexOf(str) + 1) - 1)] = str2; - } - } + class rxFS { + getInfo() { + return { + id: "0832rxfs", + name: "rxFS", + color1: "#2bdab7", + blocks: [ + { + blockIconURI: wenj, + opcode: "start", + blockType: Scratch.BlockType.COMMAND, + text: "New [STR] ", + arguments: { + STR: { + type: Scratch.ArgumentType.STRING, + defaultValue: "📃", + }, + }, + }, + { + blockIconURI: wenj, + opcode: "file", + blockType: Scratch.BlockType.COMMAND, + text: "Set [STR] to [STR2] ", + arguments: { + STR: { + type: Scratch.ArgumentType.STRING, + defaultValue: "📃", + }, + STR2: { + type: Scratch.ArgumentType.STRING, + defaultValue: "rxFS is good!", + }, + }, + }, + { + blockIconURI: wenj, + opcode: "sync", + blockType: Scratch.BlockType.COMMAND, + text: "Change the location of [STR] to [STR2] ", + arguments: { + STR: { + type: Scratch.ArgumentType.STRING, + defaultValue: "📃", + }, + STR2: { + type: Scratch.ArgumentType.STRING, + defaultValue: "📃", + }, + }, + }, + { + blockIconURI: wenj, + opcode: "del", + blockType: Scratch.BlockType.COMMAND, + text: "Delete [STR] ", + arguments: { + STR: { + type: Scratch.ArgumentType.STRING, + defaultValue: "📃", + }, + }, + }, + { + blockIconURI: wenj, + opcode: "webin", + blockType: Scratch.BlockType.REPORTER, + text: "Load [STR] from the network", + arguments: { + STR: { + type: Scratch.ArgumentType.STRING, + defaultValue: "https://0832k12.github.io/rxFS/hello.txt", + }, + }, + }, + { + blockIconURI: wenj, + opcode: "open", + blockType: Scratch.BlockType.REPORTER, + text: "Open [STR]", + arguments: { + STR: { + type: Scratch.ArgumentType.STRING, + defaultValue: "📃", + }, + }, + }, + { + blockIconURI: file, + opcode: "clean", + blockType: Scratch.BlockType.COMMAND, + text: "Clear file system", + arguments: {}, + }, + { + blockIconURI: file, + opcode: "in", + blockType: Scratch.BlockType.COMMAND, + text: "Import file system from [STR]", + arguments: { + STR: { + type: Scratch.ArgumentType.STRING, + defaultValue: "📁", + }, + }, + }, + { + blockIconURI: file, + opcode: "out", + blockType: Scratch.BlockType.REPORTER, + text: "Export file system", + arguments: {}, + }, + { + blockIconURI: file, + opcode: "list", + blockType: Scratch.BlockType.REPORTER, + text: "List the contents under the same folder [STR]", + arguments: { + STR: { + type: Scratch.ArgumentType.STRING, + defaultValue: "📁", + }, + }, + }, + { + blockIconURI: file, + opcode: "search", + blockType: Scratch.BlockType.REPORTER, + text: "Search [STR]", + arguments: { + STR: { + type: Scratch.ArgumentType.STRING, + defaultValue: "📃", + }, + }, + }, + ], + }; + } - start({ STR }) { - str = btoa(unescape(encodeURIComponent(STR))); - if (!(str.charAt((str.length - 1)) == '/') && rxFSsy.indexOf(str) + 1 == 0) { - rxFSfi.splice(((rxFSfi.length + 1) - 1), 0, null); - rxFSsy.splice(((rxFSsy.length + 1) - 1), 0, str); - } - } + clean() { + rxFSfi = []; + rxFSsy = []; + } - open({ STR }) { - return decodeURIComponent(escape(atob(rxFSfi[((rxFSsy.indexOf(btoa(unescape(encodeURIComponent(STR)))) + 1) - 1)]))); - } + sync({ STR, STR2 }) { + str = btoa(unescape(encodeURIComponent(STR))); + str2 = btoa(unescape(encodeURIComponent(STR2))); + if (rxFSsy.indexOf(str) + 1 == 0) { + rxFSsy[rxFSsy.indexOf(str) + 1 - 1] = str2; + } + } - del({ STR }) { - str = btoa(unescape(encodeURIComponent(STR))); - rxFSfi[((rxFSsy.indexOf(str) + 1) - 1)] = undefined; - rxFSsy[((rxFSsy.indexOf(str) + 1) - 1)] = undefined; - } + start({ STR }) { + str = btoa(unescape(encodeURIComponent(STR))); + if ( + !(str.charAt(str.length - 1) == "/") && + rxFSsy.indexOf(str) + 1 == 0 + ) { + rxFSfi.splice(rxFSfi.length + 1 - 1, 0, null); + rxFSsy.splice(rxFSsy.length + 1 - 1, 0, str); + } + } - file({ STR, STR2 }) { - rxFSfi[((rxFSsy.indexOf(btoa(unescape(encodeURIComponent(STR)))) + 1) - 1)] = btoa(unescape(encodeURIComponent(STR2))); - } + open({ STR }) { + return decodeURIComponent( + escape( + atob( + rxFSfi[ + rxFSsy.indexOf(btoa(unescape(encodeURIComponent(STR)))) + 1 - 1 + ] + ) + ) + ); + } - search({ STR }) { - Search = ''; - i = 0; - str = btoa(unescape(encodeURIComponent(STR))); - for (var i in rxFSsy) { - if (!(rxFSsy[(i)].indexOf(str) == undefined)) { - Search = [Search, 'LA==', rxFSsy[(i)]].join(''); - } - } - return decodeURIComponent(escape(atob(Search))); - } + del({ STR }) { + str = btoa(unescape(encodeURIComponent(STR))); + rxFSfi[rxFSsy.indexOf(str) + 1 - 1] = undefined; + rxFSsy[rxFSsy.indexOf(str) + 1 - 1] = undefined; + } - list({ STR }) { - Search = ''; - i = 0; - str = btoa(unescape(encodeURIComponent(STR))); - for (var i in rxFSsy) { - if (rxFSsy[(i)].slice(0, str.length) == str) { - Search = [Search, 'LA==', rxFSsy[(i)]].join(''); - } - } - return decodeURIComponent(escape(atob(Search))); - } + file({ STR, STR2 }) { + rxFSfi[rxFSsy.indexOf(btoa(unescape(encodeURIComponent(STR)))) + 1 - 1] = + btoa(unescape(encodeURIComponent(STR2))); + } - webin({ STR }) { - return Scratch.fetch(STR) - .then((response) => { - return response.text(); - }) - .catch((error) => { - console.error(error); - return 'undefined'; - }); + search({ STR }) { + Search = ""; + i = 0; + str = btoa(unescape(encodeURIComponent(STR))); + for (var i in rxFSsy) { + if (!(rxFSsy[i].indexOf(str) == undefined)) { + Search = [Search, "LA==", rxFSsy[i]].join(""); } + } + return decodeURIComponent(escape(atob(Search))); + } - in({ STR }) { - rxFSfi = STR.slice(0, STR.indexOf('|')).split(','); - rxFSsy = STR.slice(((STR.indexOf('|') + 1)), STR.length).split(','); + list({ STR }) { + Search = ""; + i = 0; + str = btoa(unescape(encodeURIComponent(STR))); + for (var i in rxFSsy) { + if (rxFSsy[i].slice(0, str.length) == str) { + Search = [Search, "LA==", rxFSsy[i]].join(""); } + } + return decodeURIComponent(escape(atob(Search))); + } - out() { - return [rxFSfi.join(','), '|', rxFSsy.join(',')].join(''); - } + webin({ STR }) { + return Scratch.fetch(STR) + .then((response) => { + return response.text(); + }) + .catch((error) => { + console.error(error); + return "undefined"; + }); + } + + in({ STR }) { + rxFSfi = STR.slice(0, STR.indexOf("|")).split(","); + rxFSsy = STR.slice(STR.indexOf("|") + 1, STR.length).split(","); + } + + out() { + return [rxFSfi.join(","), "|", rxFSsy.join(",")].join(""); } + } - Scratch.extensions.register(new rxFS()); + Scratch.extensions.register(new rxFS()); })(Scratch); diff --git a/extensions/0832/rxFS2.js b/extensions/0832/rxFS2.js index 0651762905..871ef0b1ca 100644 --- a/extensions/0832/rxFS2.js +++ b/extensions/0832/rxFS2.js @@ -1,291 +1,327 @@ +// Name: rxFS +// ID: 0832rxfs2 +// Description: Blocks for interacting with a virtual in-memory filesystem. +// By: 0832 + /*! * Made by 0832 * This file was originally under the rxLI Version 2.1 license: * https://0832k12.github.io/rxLi/2.1/ - * + * * However they have since claimed it to be "directly compatible with MIT license", * which is the license we use this file under. */ (function (Scratch) { - 'use strict'; - - Scratch.translate.setup({ - zh: { - start: '新建 [STR] ', - folder: '设置 [STR] 为 [STR2] ', - folder_default: '大主教大祭司主宰世界!', - sync: '将 [STR] 的位置更改为 [STR2] ', - del: '删除 [STR] ', - webin: '从网络加载 [STR]', - open: '打开 [STR]', - clean: '清空文件系统', - in: '从 [STR] 导入文件系统', - out: '导出文件系统', - list: '列出 [STR] 下的所有文件', - search: '搜索 [STR]' - }, - ru: { - start: 'Создать [STR]', - folder: 'Установить [STR] в [STR2]', - folder_default: 'Архиепископ Верховный жрец Правитель мира!', - sync: 'Изменить расположение [STR] на [STR2]', - del: 'Удалить [STR]', - webin: 'Загрузить [STR] из Интернета', - open: 'Открыть [STR]', - clean: 'Очистить файловую систему', - in: 'Импортировать файловую систему из [STR]', - out: 'Экспортировать файловую систему', - list: 'Список всех файлов в [STR]', - search: 'Поиск [STR]' - }, - jp: { - start: '新規作成 [STR]', - folder: '[STR] を [STR2] に設定する', - folder_default: '大主教大祭司世界の支配者!', - sync: '[STR] の位置を [STR2] に変更する', - del: '[STR] を削除する', - webin: '[STR] をウェブから読み込む', - open: '[STR] を開く', - clean: 'ファイルシステムをクリアする', - in: '[STR] からファイルシステムをインポートする', - out: 'ファイルシステムをエクスポートする', - list: '[STR] にあるすべてのファイルをリストする', - search: '[STR] を検索する' - } - }); + "use strict"; - var rxFSfi = new Array(); - var rxFSsy = new Array(); - var Search, i, str, str2; + Scratch.translate.setup({ + zh: { + start: "新建 [STR] ", + folder: "设置 [STR] 为 [STR2] ", + folder_default: "大主教大祭司主宰世界!", + sync: "将 [STR] 的位置更改为 [STR2] ", + del: "删除 [STR] ", + webin: "从网络加载 [STR]", + open: "打开 [STR]", + clean: "清空文件系统", + in: "从 [STR] 导入文件系统", + out: "导出文件系统", + list: "列出 [STR] 下的所有文件", + search: "搜索 [STR]", + }, + ru: { + start: "Создать [STR]", + folder: "Установить [STR] в [STR2]", + folder_default: "Архиепископ Верховный жрец Правитель мира!", + sync: "Изменить расположение [STR] на [STR2]", + del: "Удалить [STR]", + webin: "Загрузить [STR] из Интернета", + open: "Открыть [STR]", + clean: "Очистить файловую систему", + in: "Импортировать файловую систему из [STR]", + out: "Экспортировать файловую систему", + list: "Список всех файлов в [STR]", + search: "Поиск [STR]", + }, + jp: { + start: "新規作成 [STR]", + folder: "[STR] を [STR2] に設定する", + folder_default: "大主教大祭司世界の支配者!", + sync: "[STR] の位置を [STR2] に変更する", + del: "[STR] を削除する", + webin: "[STR] をウェブから読み込む", + open: "[STR] を開く", + clean: "ファイルシステムをクリアする", + in: "[STR] からファイルシステムをインポートする", + out: "ファイルシステムをエクスポートする", + list: "[STR] にあるすべてのファイルをリストする", + search: "[STR] を検索する", + }, + }); - const folder = ''; - const file = ''; + var rxFSfi = new Array(); + var rxFSsy = new Array(); + var Search, i, str, str2; - class rxFS { - getInfo() { - return { - id: '0832rxfs2', - name: 'rxFS', - color1: '#192d50', - color2: '#192d50', - color3: '#192d50', - blocks: [ - { - blockIconURI: file, - opcode: 'start', - blockType: Scratch.BlockType.COMMAND, - text: Scratch.translate({ id: 'start', default: 'Create [STR]' }), - arguments: { - STR: { - type: Scratch.ArgumentType.STRING, - defaultValue: '/rxFS/example' - } - } - }, - { - blockIconURI: file, - opcode: 'folder', - blockType: Scratch.BlockType.COMMAND, - text: Scratch.translate({ id: 'folder', default: 'Set [STR] to [STR2]' }), - arguments: { - STR: { - type: Scratch.ArgumentType.STRING, - defaultValue: '/rxFS/example' - }, - STR2: { - type: Scratch.ArgumentType.STRING, - defaultValue: Scratch.translate({ id: 'folder_default', default: 'rxFS is good!' }), - } - } - }, - { - blockIconURI: file, - opcode: 'sync', - blockType: Scratch.BlockType.COMMAND, - text: Scratch.translate({ id: 'sync', default: 'Change the location of [STR] to [STR2]' }), - arguments: { - STR: { - type: Scratch.ArgumentType.STRING, - defaultValue: '/rxFS/example' - }, - STR2: { - type: Scratch.ArgumentType.STRING, - defaultValue: '/rxFS/example' - } - } - }, - { - blockIconURI: file, - opcode: 'del', - blockType: Scratch.BlockType.COMMAND, - text: Scratch.translate({ id: 'del', default: 'Delete [STR]' }), - arguments: { - STR: { - type: Scratch.ArgumentType.STRING, - defaultValue: '/rxFS/example' - } - } - }, - { - blockIconURI: file, - opcode: 'webin', - blockType: Scratch.BlockType.REPORTER, - text: Scratch.translate({ id: 'webin', default: 'Load [STR] from the web' }), - arguments: { - STR: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'https://0832k12.github.io/rxFS/hello.txt' - } - } - }, - { - blockIconURI: file, - opcode: 'open', - blockType: Scratch.BlockType.REPORTER, - text: Scratch.translate({ id: 'open', default: 'Open [STR]' }), - arguments: { - STR: { - type: Scratch.ArgumentType.STRING, - defaultValue: '/rxFS/example' - } - } - }, - '---', - { - blockIconURI: folder, - opcode: 'clean', - blockType: Scratch.BlockType.COMMAND, - text: Scratch.translate({ id: 'clean', default: 'Clear the file system' }), - arguments: {} - }, - { - blockIconURI: folder, - opcode: 'in', - blockType: Scratch.BlockType.COMMAND, - text: Scratch.translate({ id: 'in', default: 'Import file system from [STR]' }), - arguments: { - STR: { - type: Scratch.ArgumentType.STRING, - defaultValue: '/rxFS/' - } - } - }, - { - blockIconURI: folder, - opcode: 'out', - blockType: Scratch.BlockType.REPORTER, - text: Scratch.translate({ id: 'out', default: 'Export file system' }), - arguments: {} - }, - { - blockIconURI: folder, - opcode: 'list', - blockType: Scratch.BlockType.REPORTER, - text: Scratch.translate({ id: 'list', default: 'List all files under [STR]' }), - arguments: { - STR: { - type: Scratch.ArgumentType.STRING, - defaultValue: '/rxFS/' - } - } - }, - { - blockIconURI: folder, - opcode: 'search', - blockType: Scratch.BlockType.REPORTER, - text: Scratch.translate({ id: 'search', default: 'Search [STR]' }), - arguments: { - STR: { - type: Scratch.ArgumentType.STRING, - defaultValue: '/rxFS/example' - } - } - } - ] - }; - } + const folder = + ""; + const file = + ""; + class rxFS { + getInfo() { + return { + id: "0832rxfs2", + name: "rxFS", + color1: "#192d50", + color2: "#192d50", + color3: "#192d50", + blocks: [ + { + blockIconURI: file, + opcode: "start", + blockType: Scratch.BlockType.COMMAND, + text: Scratch.translate({ id: "start", default: "Create [STR]" }), + arguments: { + STR: { + type: Scratch.ArgumentType.STRING, + defaultValue: "/rxFS/example", + }, + }, + }, + { + blockIconURI: file, + opcode: "folder", + blockType: Scratch.BlockType.COMMAND, + text: Scratch.translate({ + id: "folder", + default: "Set [STR] to [STR2]", + }), + arguments: { + STR: { + type: Scratch.ArgumentType.STRING, + defaultValue: "/rxFS/example", + }, + STR2: { + type: Scratch.ArgumentType.STRING, + defaultValue: Scratch.translate({ + id: "folder_default", + default: "rxFS is good!", + }), + }, + }, + }, + { + blockIconURI: file, + opcode: "sync", + blockType: Scratch.BlockType.COMMAND, + text: Scratch.translate({ + id: "sync", + default: "Change the location of [STR] to [STR2]", + }), + arguments: { + STR: { + type: Scratch.ArgumentType.STRING, + defaultValue: "/rxFS/example", + }, + STR2: { + type: Scratch.ArgumentType.STRING, + defaultValue: "/rxFS/example", + }, + }, + }, + { + blockIconURI: file, + opcode: "del", + blockType: Scratch.BlockType.COMMAND, + text: Scratch.translate({ id: "del", default: "Delete [STR]" }), + arguments: { + STR: { + type: Scratch.ArgumentType.STRING, + defaultValue: "/rxFS/example", + }, + }, + }, + { + blockIconURI: file, + opcode: "webin", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate({ + id: "webin", + default: "Load [STR] from the web", + }), + arguments: { + STR: { + type: Scratch.ArgumentType.STRING, + defaultValue: "https://0832k12.github.io/rxFS/hello.txt", + }, + }, + }, + { + blockIconURI: file, + opcode: "open", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate({ id: "open", default: "Open [STR]" }), + arguments: { + STR: { + type: Scratch.ArgumentType.STRING, + defaultValue: "/rxFS/example", + }, + }, + }, + "---", + { + blockIconURI: folder, + opcode: "clean", + blockType: Scratch.BlockType.COMMAND, + text: Scratch.translate({ + id: "clean", + default: "Clear the file system", + }), + arguments: {}, + }, + { + blockIconURI: folder, + opcode: "in", + blockType: Scratch.BlockType.COMMAND, + text: Scratch.translate({ + id: "in", + default: "Import file system from [STR]", + }), + arguments: { + STR: { + type: Scratch.ArgumentType.STRING, + defaultValue: "/rxFS/", + }, + }, + }, + { + blockIconURI: folder, + opcode: "out", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate({ + id: "out", + default: "Export file system", + }), + arguments: {}, + }, + { + blockIconURI: folder, + opcode: "list", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate({ + id: "list", + default: "List all files under [STR]", + }), + arguments: { + STR: { + type: Scratch.ArgumentType.STRING, + defaultValue: "/rxFS/", + }, + }, + }, + { + blockIconURI: folder, + opcode: "search", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate({ id: "search", default: "Search [STR]" }), + arguments: { + STR: { + type: Scratch.ArgumentType.STRING, + defaultValue: "/rxFS/example", + }, + }, + }, + ], + }; + } - clean() { - rxFSfi = []; - rxFSsy = []; - } + clean() { + rxFSfi = []; + rxFSsy = []; + } - sync({ STR, STR2 }) { - str = encodeURIComponent(STR); - str2 = encodeURIComponent(STR2); - if (rxFSsy.indexOf(str) + 1 == 0) { - rxFSsy[((rxFSsy.indexOf(str) + 1) - 1)] = str2; - } - } + sync({ STR, STR2 }) { + str = encodeURIComponent(STR); + str2 = encodeURIComponent(STR2); + if (rxFSsy.indexOf(str) + 1 == 0) { + rxFSsy[rxFSsy.indexOf(str) + 1 - 1] = str2; + } + } - start({ STR }) { - str = encodeURIComponent(STR); - if (!(str.charAt((str.length - 1)) == '/') && rxFSsy.indexOf(str) + 1 == 0) { - rxFSfi.splice(((rxFSfi.length + 1) - 1), 0, null); - rxFSsy.splice(((rxFSsy.length + 1) - 1), 0, str); - } - } + start({ STR }) { + str = encodeURIComponent(STR); + if ( + !(str.charAt(str.length - 1) == "/") && + rxFSsy.indexOf(str) + 1 == 0 + ) { + rxFSfi.splice(rxFSfi.length + 1 - 1, 0, null); + rxFSsy.splice(rxFSsy.length + 1 - 1, 0, str); + } + } - open({ STR }) { - return decodeURIComponent(rxFSfi[((rxFSsy.indexOf(encodeURIComponent(STR)) + 1) - 1)]); - } + open({ STR }) { + return decodeURIComponent( + rxFSfi[rxFSsy.indexOf(encodeURIComponent(STR)) + 1 - 1] + ); + } - del({ STR }) { - str = encodeURIComponent(STR); - rxFSfi[((rxFSsy.indexOf(str) + 1) - 1)] = undefined; - rxFSsy[((rxFSsy.indexOf(str) + 1) - 1)] = undefined; - } + del({ STR }) { + str = encodeURIComponent(STR); + rxFSfi[rxFSsy.indexOf(str) + 1 - 1] = undefined; + rxFSsy[rxFSsy.indexOf(str) + 1 - 1] = undefined; + } - folder({ STR, STR2 }) { - rxFSfi[((rxFSsy.indexOf(encodeURIComponent(STR)) + 1) - 1)] = encodeURIComponent(STR2); - } + folder({ STR, STR2 }) { + rxFSfi[rxFSsy.indexOf(encodeURIComponent(STR)) + 1 - 1] = + encodeURIComponent(STR2); + } - search({ STR }) { - Search = ''; - i = 0; - str = encodeURIComponent(STR); - for (var i in rxFSsy) { - if (!(rxFSsy[(i)].indexOf(str) == undefined)) { - Search = [Search, ',"', rxFSsy[(i)], '"'].join(''); - } - } - return decodeURIComponent(Search); + search({ STR }) { + Search = ""; + i = 0; + str = encodeURIComponent(STR); + for (var i in rxFSsy) { + if (!(rxFSsy[i].indexOf(str) == undefined)) { + Search = [Search, ',"', rxFSsy[i], '"'].join(""); } + } + return decodeURIComponent(Search); + } - list({ STR }) { - Search = ''; - i = 0; - str = encodeURIComponent(STR); - for (var i in rxFSsy) { - if (rxFSsy[(i)].slice(0, str.length) == str) { - Search = [Search, ',"', rxFSsy[(i)], '"'].join(''); - } - } - return decodeURIComponent(Search); + list({ STR }) { + Search = ""; + i = 0; + str = encodeURIComponent(STR); + for (var i in rxFSsy) { + if (rxFSsy[i].slice(0, str.length) == str) { + Search = [Search, ',"', rxFSsy[i], '"'].join(""); } + } + return decodeURIComponent(Search); + } - webin({ STR }) { - return Scratch.fetch(STR) - .then((response) => { - return response.text(); - }) - .catch((error) => { - console.error(error); - return 'undefined'; - }); - } + webin({ STR }) { + return Scratch.fetch(STR) + .then((response) => { + return response.text(); + }) + .catch((error) => { + console.error(error); + return "undefined"; + }); + } - in({ STR }) { - rxFSfi = STR.slice(0, STR.indexOf('|')).split(','); - rxFSsy = STR.slice(((STR.indexOf('|') + 1)), STR.length).split(','); - } + in({ STR }) { + rxFSfi = STR.slice(0, STR.indexOf("|")).split(","); + rxFSsy = STR.slice(STR.indexOf("|") + 1, STR.length).split(","); + } - out() { - return [rxFSfi.join(','), '|', rxFSsy.join(',')].join(''); - } + out() { + return [rxFSfi.join(","), "|", rxFSsy.join(",")].join(""); } + } - Scratch.extensions.register(new rxFS()); + Scratch.extensions.register(new rxFS()); })(Scratch); diff --git a/extensions/Alestore/nfcwarp.js b/extensions/Alestore/nfcwarp.js index c731f04095..6e1face62c 100644 --- a/extensions/Alestore/nfcwarp.js +++ b/extensions/Alestore/nfcwarp.js @@ -1,68 +1,76 @@ -(function(Scratch) { - 'use strict'; +// Name: NFCWarp +// ID: alestorenfc +// Description: Allows reading data from NFC (NDEF) devices. Only works in Chrome on Android. +// By: Alestore Games + +(function (Scratch) { + "use strict"; /* globals NDEFReader */ - const extIcon = ''; - const blocksIcon = ''; + const extIcon = + ""; + const blocksIcon = + ""; class NFCWarp { getInfo() { return { - id: 'alestorenfc', - name: 'NFCWarp', - color1: '#FF4646', - color2: '#FF0000', - color3: '#990033', + id: "alestorenfc", + name: "NFCWarp", + color1: "#FF4646", + color2: "#FF0000", + color3: "#990033", menuIconURI: extIcon, blockIconURI: blocksIcon, blocks: [ { blockType: Scratch.BlockType.LABEL, - text: 'Only works in Chrome on Android' + text: "Only works in Chrome on Android", }, { - opcode: 'supported', + opcode: "supported", blockType: Scratch.BlockType.BOOLEAN, - text: 'NFC supported?' + text: "NFC supported?", }, { - opcode: 'nfcRead', + opcode: "nfcRead", blockType: Scratch.BlockType.REPORTER, - text: 'read NFC tag', - disableMonitor: true - } - ] + text: "read NFC tag", + disableMonitor: true, + }, + ], }; } - supported () { - return typeof NDEFReader !== 'undefined'; + supported() { + return typeof NDEFReader !== "undefined"; } nfcRead() { if (!this.supported()) { - return 'NFC not supported'; + return "NFC not supported"; } return new Promise((resolve, reject) => { const ndef = new NDEFReader(); - ndef.scan() + ndef + .scan() .then(() => { - ndef.onreadingerror = event => { - console.log('Reading error', event); - resolve('Tag not supported'); + ndef.onreadingerror = (event) => { + console.log("Reading error", event); + resolve("Tag not supported"); }; - ndef.onreading = evt => { + ndef.onreading = (evt) => { const decoder = new TextDecoder(); const record = evt.message.records[0]; - console.log('Record type: ' + record.recordType); - console.log('Record encoding: ' + record.encoding); - console.log('Record data: ' + decoder.decode(record.data)); + console.log("Record type: " + record.recordType); + console.log("Record encoding: " + record.encoding); + console.log("Record data: " + decoder.decode(record.data)); resolve(decoder.decode(record.data)); }; }) - .catch(error => { - console.log('Scan error', error); + .catch((error) => { + console.log("Scan error", error); resolve(`Error: ${error}`); }); }); diff --git a/extensions/CST1229/images.js b/extensions/CST1229/images.js index a7a296b461..7524b64dea 100644 --- a/extensions/CST1229/images.js +++ b/extensions/CST1229/images.js @@ -202,7 +202,7 @@ case "image/bmp": case "image/jpeg": { - if (!await Scratch.canFetch(IMAGEURL)) return; + if (!(await Scratch.canFetch(IMAGEURL))) return; // eslint-disable-next-line no-restricted-syntax const image = new Image(); image.crossOrigin = "anonymous"; @@ -377,6 +377,7 @@ } } - if (!Scratch.extensions.unsandboxed) throw new Error("This extension cannot run in sandboxed mode."); + if (!Scratch.extensions.unsandboxed) + throw new Error("This extension cannot run in sandboxed mode."); Scratch.extensions.register(new ImagesExt(Scratch.vm)); })(globalThis.Scratch); diff --git a/extensions/CST1229/zip.js b/extensions/CST1229/zip.js index 9bbb0392ce..a362fd17d7 100644 --- a/extensions/CST1229/zip.js +++ b/extensions/CST1229/zip.js @@ -1,39 +1,16 @@ +// Name: Zip +// ID: cst1229zip +// Description: Create and edit .zip format files, including .sb3 files. +// By: CST1229 + (function (Scratch) { "use strict"; - // Tricking JSZip into thinking it's running as a CommonJS module - // is probably better than letting it overwrite globals - const exports = {}; - const module = { exports: null }; - - // jszip source code: - // https://github.com/Stuk/jszip - // using it under the MIT license - - // minified code from: https://cdn.jsdelivr.net/npm/jszip@3/dist/jszip.min.js - // in a function for code folding - function jsZip() { - /*! - - JSZip v3.10.1 - A JavaScript class for generating and reading zip files - - - (c) 2009-2016 Stuart Knightley - Dual licenced under the MIT license or GPLv3. See https://raw.github.com/Stuk/jszip/main/LICENSE.markdown. - - JSZip uses the library pako released under the MIT license : - https://github.com/nodeca/pako/blob/main/LICENSE - */ - /* eslint-disable */ - // @ts-ignore - !function(e){if("object"==typeof exports&&"undefined"!=typeof module)module.exports=e();else if("function"==typeof define&&define.amd)define([],e);else{("undefined"!=typeof window?window:"undefined"!=typeof global?global:"undefined"!=typeof self?self:this).JSZip=e()}}(function(){return function s(a,o,h){function u(r,e){if(!o[r]){if(!a[r]){var t="function"==typeof require&&require;if(!e&&t)return t(r,!0);if(l)return l(r,!0);var n=new Error("Cannot find module '"+r+"'");throw n.code="MODULE_NOT_FOUND",n}var i=o[r]={exports:{}};a[r][0].call(i.exports,function(e){var t=a[r][1][e];return u(t||e)},i,i.exports,s,a,o,h)}return o[r].exports}for(var l="function"==typeof require&&require,e=0;e>2,s=(3&t)<<4|r>>4,a=1>6:64,o=2>4,r=(15&i)<<4|(s=p.indexOf(e.charAt(o++)))>>2,n=(3&s)<<6|(a=p.indexOf(e.charAt(o++))),l[h++]=t,64!==s&&(l[h++]=r),64!==a&&(l[h++]=n);return l}},{"./support":30,"./utils":32}],2:[function(e,t,r){"use strict";var n=e("./external"),i=e("./stream/DataWorker"),s=e("./stream/Crc32Probe"),a=e("./stream/DataLengthProbe");function o(e,t,r,n,i){this.compressedSize=e,this.uncompressedSize=t,this.crc32=r,this.compression=n,this.compressedContent=i}o.prototype={getContentWorker:function(){var e=new i(n.Promise.resolve(this.compressedContent)).pipe(this.compression.uncompressWorker()).pipe(new a("data_length")),t=this;return e.on("end",function(){if(this.streamInfo.data_length!==t.uncompressedSize)throw new Error("Bug : uncompressed data size mismatch")}),e},getCompressedWorker:function(){return new i(n.Promise.resolve(this.compressedContent)).withStreamInfo("compressedSize",this.compressedSize).withStreamInfo("uncompressedSize",this.uncompressedSize).withStreamInfo("crc32",this.crc32).withStreamInfo("compression",this.compression)}},o.createWorkerFrom=function(e,t,r){return e.pipe(new s).pipe(new a("uncompressedSize")).pipe(t.compressWorker(r)).pipe(new a("compressedSize")).withStreamInfo("compression",t)},t.exports=o},{"./external":6,"./stream/Crc32Probe":25,"./stream/DataLengthProbe":26,"./stream/DataWorker":27}],3:[function(e,t,r){"use strict";var n=e("./stream/GenericWorker");r.STORE={magic:"\0\0",compressWorker:function(){return new n("STORE compression")},uncompressWorker:function(){return new n("STORE decompression")}},r.DEFLATE=e("./flate")},{"./flate":7,"./stream/GenericWorker":28}],4:[function(e,t,r){"use strict";var n=e("./utils");var o=function(){for(var e,t=[],r=0;r<256;r++){e=r;for(var n=0;n<8;n++)e=1&e?3988292384^e>>>1:e>>>1;t[r]=e}return t}();t.exports=function(e,t){return void 0!==e&&e.length?"string"!==n.getTypeOf(e)?function(e,t,r,n){var i=o,s=n+r;e^=-1;for(var a=n;a>>8^i[255&(e^t[a])];return-1^e}(0|t,e,e.length,0):function(e,t,r,n){var i=o,s=n+r;e^=-1;for(var a=n;a>>8^i[255&(e^t.charCodeAt(a))];return-1^e}(0|t,e,e.length,0):0}},{"./utils":32}],5:[function(e,t,r){"use strict";r.base64=!1,r.binary=!1,r.dir=!1,r.createFolders=!0,r.date=null,r.compression=null,r.compressionOptions=null,r.comment=null,r.unixPermissions=null,r.dosPermissions=null},{}],6:[function(e,t,r){"use strict";var n=null;n="undefined"!=typeof Promise?Promise:e("lie"),t.exports={Promise:n}},{lie:37}],7:[function(e,t,r){"use strict";var n="undefined"!=typeof Uint8Array&&"undefined"!=typeof Uint16Array&&"undefined"!=typeof Uint32Array,i=e("pako"),s=e("./utils"),a=e("./stream/GenericWorker"),o=n?"uint8array":"array";function h(e,t){a.call(this,"FlateWorker/"+e),this._pako=null,this._pakoAction=e,this._pakoOptions=t,this.meta={}}r.magic="\b\0",s.inherits(h,a),h.prototype.processChunk=function(e){this.meta=e.meta,null===this._pako&&this._createPako(),this._pako.push(s.transformTo(o,e.data),!1)},h.prototype.flush=function(){a.prototype.flush.call(this),null===this._pako&&this._createPako(),this._pako.push([],!0)},h.prototype.cleanUp=function(){a.prototype.cleanUp.call(this),this._pako=null},h.prototype._createPako=function(){this._pako=new i[this._pakoAction]({raw:!0,level:this._pakoOptions.level||-1});var t=this;this._pako.onData=function(e){t.push({data:e,meta:t.meta})}},r.compressWorker=function(e){return new h("Deflate",e)},r.uncompressWorker=function(){return new h("Inflate",{})}},{"./stream/GenericWorker":28,"./utils":32,pako:38}],8:[function(e,t,r){"use strict";function A(e,t){var r,n="";for(r=0;r>>=8;return n}function n(e,t,r,n,i,s){var a,o,h=e.file,u=e.compression,l=s!==O.utf8encode,f=I.transformTo("string",s(h.name)),c=I.transformTo("string",O.utf8encode(h.name)),d=h.comment,p=I.transformTo("string",s(d)),m=I.transformTo("string",O.utf8encode(d)),_=c.length!==h.name.length,g=m.length!==d.length,b="",v="",y="",w=h.dir,k=h.date,x={crc32:0,compressedSize:0,uncompressedSize:0};t&&!r||(x.crc32=e.crc32,x.compressedSize=e.compressedSize,x.uncompressedSize=e.uncompressedSize);var S=0;t&&(S|=8),l||!_&&!g||(S|=2048);var z=0,C=0;w&&(z|=16),"UNIX"===i?(C=798,z|=function(e,t){var r=e;return e||(r=t?16893:33204),(65535&r)<<16}(h.unixPermissions,w)):(C=20,z|=function(e){return 63&(e||0)}(h.dosPermissions)),a=k.getUTCHours(),a<<=6,a|=k.getUTCMinutes(),a<<=5,a|=k.getUTCSeconds()/2,o=k.getUTCFullYear()-1980,o<<=4,o|=k.getUTCMonth()+1,o<<=5,o|=k.getUTCDate(),_&&(v=A(1,1)+A(B(f),4)+c,b+="up"+A(v.length,2)+v),g&&(y=A(1,1)+A(B(p),4)+m,b+="uc"+A(y.length,2)+y);var E="";return E+="\n\0",E+=A(S,2),E+=u.magic,E+=A(a,2),E+=A(o,2),E+=A(x.crc32,4),E+=A(x.compressedSize,4),E+=A(x.uncompressedSize,4),E+=A(f.length,2),E+=A(b.length,2),{fileRecord:R.LOCAL_FILE_HEADER+E+f+b,dirRecord:R.CENTRAL_FILE_HEADER+A(C,2)+E+A(p.length,2)+"\0\0\0\0"+A(z,4)+A(n,4)+f+b+p}}var I=e("../utils"),i=e("../stream/GenericWorker"),O=e("../utf8"),B=e("../crc32"),R=e("../signature");function s(e,t,r,n){i.call(this,"ZipFileWorker"),this.bytesWritten=0,this.zipComment=t,this.zipPlatform=r,this.encodeFileName=n,this.streamFiles=e,this.accumulate=!1,this.contentBuffer=[],this.dirRecords=[],this.currentSourceOffset=0,this.entriesCount=0,this.currentFile=null,this._sources=[]}I.inherits(s,i),s.prototype.push=function(e){var t=e.meta.percent||0,r=this.entriesCount,n=this._sources.length;this.accumulate?this.contentBuffer.push(e):(this.bytesWritten+=e.data.length,i.prototype.push.call(this,{data:e.data,meta:{currentFile:this.currentFile,percent:r?(t+100*(r-n-1))/r:100}}))},s.prototype.openedSource=function(e){this.currentSourceOffset=this.bytesWritten,this.currentFile=e.file.name;var t=this.streamFiles&&!e.file.dir;if(t){var r=n(e,t,!1,this.currentSourceOffset,this.zipPlatform,this.encodeFileName);this.push({data:r.fileRecord,meta:{percent:0}})}else this.accumulate=!0},s.prototype.closedSource=function(e){this.accumulate=!1;var t=this.streamFiles&&!e.file.dir,r=n(e,t,!0,this.currentSourceOffset,this.zipPlatform,this.encodeFileName);if(this.dirRecords.push(r.dirRecord),t)this.push({data:function(e){return R.DATA_DESCRIPTOR+A(e.crc32,4)+A(e.compressedSize,4)+A(e.uncompressedSize,4)}(e),meta:{percent:100}});else for(this.push({data:r.fileRecord,meta:{percent:0}});this.contentBuffer.length;)this.push(this.contentBuffer.shift());this.currentFile=null},s.prototype.flush=function(){for(var e=this.bytesWritten,t=0;t=this.index;t--)r=(r<<8)+this.byteAt(t);return this.index+=e,r},readString:function(e){return n.transformTo("string",this.readData(e))},readData:function(){},lastIndexOfSignature:function(){},readAndCheckSignature:function(){},readDate:function(){var e=this.readInt(4);return new Date(Date.UTC(1980+(e>>25&127),(e>>21&15)-1,e>>16&31,e>>11&31,e>>5&63,(31&e)<<1))}},t.exports=i},{"../utils":32}],19:[function(e,t,r){"use strict";var n=e("./Uint8ArrayReader");function i(e){n.call(this,e)}e("../utils").inherits(i,n),i.prototype.readData=function(e){this.checkOffset(e);var t=this.data.slice(this.zero+this.index,this.zero+this.index+e);return this.index+=e,t},t.exports=i},{"../utils":32,"./Uint8ArrayReader":21}],20:[function(e,t,r){"use strict";var n=e("./DataReader");function i(e){n.call(this,e)}e("../utils").inherits(i,n),i.prototype.byteAt=function(e){return this.data.charCodeAt(this.zero+e)},i.prototype.lastIndexOfSignature=function(e){return this.data.lastIndexOf(e)-this.zero},i.prototype.readAndCheckSignature=function(e){return e===this.readData(4)},i.prototype.readData=function(e){this.checkOffset(e);var t=this.data.slice(this.zero+this.index,this.zero+this.index+e);return this.index+=e,t},t.exports=i},{"../utils":32,"./DataReader":18}],21:[function(e,t,r){"use strict";var n=e("./ArrayReader");function i(e){n.call(this,e)}e("../utils").inherits(i,n),i.prototype.readData=function(e){if(this.checkOffset(e),0===e)return new Uint8Array(0);var t=this.data.subarray(this.zero+this.index,this.zero+this.index+e);return this.index+=e,t},t.exports=i},{"../utils":32,"./ArrayReader":17}],22:[function(e,t,r){"use strict";var n=e("../utils"),i=e("../support"),s=e("./ArrayReader"),a=e("./StringReader"),o=e("./NodeBufferReader"),h=e("./Uint8ArrayReader");t.exports=function(e){var t=n.getTypeOf(e);return n.checkSupport(t),"string"!==t||i.uint8array?"nodebuffer"===t?new o(e):i.uint8array?new h(n.transformTo("uint8array",e)):new s(n.transformTo("array",e)):new a(e)}},{"../support":30,"../utils":32,"./ArrayReader":17,"./NodeBufferReader":19,"./StringReader":20,"./Uint8ArrayReader":21}],23:[function(e,t,r){"use strict";r.LOCAL_FILE_HEADER="PK",r.CENTRAL_FILE_HEADER="PK",r.CENTRAL_DIRECTORY_END="PK",r.ZIP64_CENTRAL_DIRECTORY_LOCATOR="PK",r.ZIP64_CENTRAL_DIRECTORY_END="PK",r.DATA_DESCRIPTOR="PK\b"},{}],24:[function(e,t,r){"use strict";var n=e("./GenericWorker"),i=e("../utils");function s(e){n.call(this,"ConvertWorker to "+e),this.destType=e}i.inherits(s,n),s.prototype.processChunk=function(e){this.push({data:i.transformTo(this.destType,e.data),meta:e.meta})},t.exports=s},{"../utils":32,"./GenericWorker":28}],25:[function(e,t,r){"use strict";var n=e("./GenericWorker"),i=e("../crc32");function s(){n.call(this,"Crc32Probe"),this.withStreamInfo("crc32",0)}e("../utils").inherits(s,n),s.prototype.processChunk=function(e){this.streamInfo.crc32=i(e.data,this.streamInfo.crc32||0),this.push(e)},t.exports=s},{"../crc32":4,"../utils":32,"./GenericWorker":28}],26:[function(e,t,r){"use strict";var n=e("../utils"),i=e("./GenericWorker");function s(e){i.call(this,"DataLengthProbe for "+e),this.propName=e,this.withStreamInfo(e,0)}n.inherits(s,i),s.prototype.processChunk=function(e){if(e){var t=this.streamInfo[this.propName]||0;this.streamInfo[this.propName]=t+e.data.length}i.prototype.processChunk.call(this,e)},t.exports=s},{"../utils":32,"./GenericWorker":28}],27:[function(e,t,r){"use strict";var n=e("../utils"),i=e("./GenericWorker");function s(e){i.call(this,"DataWorker");var t=this;this.dataIsReady=!1,this.index=0,this.max=0,this.data=null,this.type="",this._tickScheduled=!1,e.then(function(e){t.dataIsReady=!0,t.data=e,t.max=e&&e.length||0,t.type=n.getTypeOf(e),t.isPaused||t._tickAndRepeat()},function(e){t.error(e)})}n.inherits(s,i),s.prototype.cleanUp=function(){i.prototype.cleanUp.call(this),this.data=null},s.prototype.resume=function(){return!!i.prototype.resume.call(this)&&(!this._tickScheduled&&this.dataIsReady&&(this._tickScheduled=!0,n.delay(this._tickAndRepeat,[],this)),!0)},s.prototype._tickAndRepeat=function(){this._tickScheduled=!1,this.isPaused||this.isFinished||(this._tick(),this.isFinished||(n.delay(this._tickAndRepeat,[],this),this._tickScheduled=!0))},s.prototype._tick=function(){if(this.isPaused||this.isFinished)return!1;var e=null,t=Math.min(this.max,this.index+16384);if(this.index>=this.max)return this.end();switch(this.type){case"string":e=this.data.substring(this.index,t);break;case"uint8array":e=this.data.subarray(this.index,t);break;case"array":case"nodebuffer":e=this.data.slice(this.index,t)}return this.index=t,this.push({data:e,meta:{percent:this.max?this.index/this.max*100:0}})},t.exports=s},{"../utils":32,"./GenericWorker":28}],28:[function(e,t,r){"use strict";function n(e){this.name=e||"default",this.streamInfo={},this.generatedError=null,this.extraStreamInfo={},this.isPaused=!0,this.isFinished=!1,this.isLocked=!1,this._listeners={data:[],end:[],error:[]},this.previous=null}n.prototype={push:function(e){this.emit("data",e)},end:function(){if(this.isFinished)return!1;this.flush();try{this.emit("end"),this.cleanUp(),this.isFinished=!0}catch(e){this.emit("error",e)}return!0},error:function(e){return!this.isFinished&&(this.isPaused?this.generatedError=e:(this.isFinished=!0,this.emit("error",e),this.previous&&this.previous.error(e),this.cleanUp()),!0)},on:function(e,t){return this._listeners[e].push(t),this},cleanUp:function(){this.streamInfo=this.generatedError=this.extraStreamInfo=null,this._listeners=[]},emit:function(e,t){if(this._listeners[e])for(var r=0;r "+e:e}},t.exports=n},{}],29:[function(e,t,r){"use strict";var h=e("../utils"),i=e("./ConvertWorker"),s=e("./GenericWorker"),u=e("../base64"),n=e("../support"),a=e("../external"),o=null;if(n.nodestream)try{o=e("../nodejs/NodejsStreamOutputAdapter")}catch(e){}function l(e,o){return new a.Promise(function(t,r){var n=[],i=e._internalType,s=e._outputType,a=e._mimeType;e.on("data",function(e,t){n.push(e),o&&o(t)}).on("error",function(e){n=[],r(e)}).on("end",function(){try{var e=function(e,t,r){switch(e){case"blob":return h.newBlob(h.transformTo("arraybuffer",t),r);case"base64":return u.encode(t);default:return h.transformTo(e,t)}}(s,function(e,t){var r,n=0,i=null,s=0;for(r=0;r>>6:(r<65536?t[s++]=224|r>>>12:(t[s++]=240|r>>>18,t[s++]=128|r>>>12&63),t[s++]=128|r>>>6&63),t[s++]=128|63&r);return t}(e)},s.utf8decode=function(e){return h.nodebuffer?o.transformTo("nodebuffer",e).toString("utf-8"):function(e){var t,r,n,i,s=e.length,a=new Array(2*s);for(t=r=0;t>10&1023,a[r++]=56320|1023&n)}return a.length!==r&&(a.subarray?a=a.subarray(0,r):a.length=r),o.applyFromCharCode(a)}(e=o.transformTo(h.uint8array?"uint8array":"array",e))},o.inherits(a,n),a.prototype.processChunk=function(e){var t=o.transformTo(h.uint8array?"uint8array":"array",e.data);if(this.leftOver&&this.leftOver.length){if(h.uint8array){var r=t;(t=new Uint8Array(r.length+this.leftOver.length)).set(this.leftOver,0),t.set(r,this.leftOver.length)}else t=this.leftOver.concat(t);this.leftOver=null}var n=function(e,t){var r;for((t=t||e.length)>e.length&&(t=e.length),r=t-1;0<=r&&128==(192&e[r]);)r--;return r<0?t:0===r?t:r+u[e[r]]>t?r:t}(t),i=t;n!==t.length&&(h.uint8array?(i=t.subarray(0,n),this.leftOver=t.subarray(n,t.length)):(i=t.slice(0,n),this.leftOver=t.slice(n,t.length))),this.push({data:s.utf8decode(i),meta:e.meta})},a.prototype.flush=function(){this.leftOver&&this.leftOver.length&&(this.push({data:s.utf8decode(this.leftOver),meta:{}}),this.leftOver=null)},s.Utf8DecodeWorker=a,o.inherits(l,n),l.prototype.processChunk=function(e){this.push({data:s.utf8encode(e.data),meta:e.meta})},s.Utf8EncodeWorker=l},{"./nodejsUtils":14,"./stream/GenericWorker":28,"./support":30,"./utils":32}],32:[function(e,t,a){"use strict";var o=e("./support"),h=e("./base64"),r=e("./nodejsUtils"),u=e("./external");function n(e){return e}function l(e,t){for(var r=0;r>8;this.dir=!!(16&this.externalFileAttributes),0==e&&(this.dosPermissions=63&this.externalFileAttributes),3==e&&(this.unixPermissions=this.externalFileAttributes>>16&65535),this.dir||"/"!==this.fileNameStr.slice(-1)||(this.dir=!0)},parseZIP64ExtraField:function(){if(this.extraFields[1]){var e=n(this.extraFields[1].value);this.uncompressedSize===s.MAX_VALUE_32BITS&&(this.uncompressedSize=e.readInt(8)),this.compressedSize===s.MAX_VALUE_32BITS&&(this.compressedSize=e.readInt(8)),this.localHeaderOffset===s.MAX_VALUE_32BITS&&(this.localHeaderOffset=e.readInt(8)),this.diskNumberStart===s.MAX_VALUE_32BITS&&(this.diskNumberStart=e.readInt(4))}},readExtraFields:function(e){var t,r,n,i=e.index+this.extraFieldsLength;for(this.extraFields||(this.extraFields={});e.index+4>>6:(r<65536?t[s++]=224|r>>>12:(t[s++]=240|r>>>18,t[s++]=128|r>>>12&63),t[s++]=128|r>>>6&63),t[s++]=128|63&r);return t},r.buf2binstring=function(e){return l(e,e.length)},r.binstring2buf=function(e){for(var t=new h.Buf8(e.length),r=0,n=t.length;r>10&1023,o[n++]=56320|1023&i)}return l(o,n)},r.utf8border=function(e,t){var r;for((t=t||e.length)>e.length&&(t=e.length),r=t-1;0<=r&&128==(192&e[r]);)r--;return r<0?t:0===r?t:r+u[e[r]]>t?r:t}},{"./common":41}],43:[function(e,t,r){"use strict";t.exports=function(e,t,r,n){for(var i=65535&e|0,s=e>>>16&65535|0,a=0;0!==r;){for(r-=a=2e3>>1:e>>>1;t[r]=e}return t}();t.exports=function(e,t,r,n){var i=o,s=n+r;e^=-1;for(var a=n;a>>8^i[255&(e^t[a])];return-1^e}},{}],46:[function(e,t,r){"use strict";var h,c=e("../utils/common"),u=e("./trees"),d=e("./adler32"),p=e("./crc32"),n=e("./messages"),l=0,f=4,m=0,_=-2,g=-1,b=4,i=2,v=8,y=9,s=286,a=30,o=19,w=2*s+1,k=15,x=3,S=258,z=S+x+1,C=42,E=113,A=1,I=2,O=3,B=4;function R(e,t){return e.msg=n[t],t}function T(e){return(e<<1)-(4e.avail_out&&(r=e.avail_out),0!==r&&(c.arraySet(e.output,t.pending_buf,t.pending_out,r,e.next_out),e.next_out+=r,t.pending_out+=r,e.total_out+=r,e.avail_out-=r,t.pending-=r,0===t.pending&&(t.pending_out=0))}function N(e,t){u._tr_flush_block(e,0<=e.block_start?e.block_start:-1,e.strstart-e.block_start,t),e.block_start=e.strstart,F(e.strm)}function U(e,t){e.pending_buf[e.pending++]=t}function P(e,t){e.pending_buf[e.pending++]=t>>>8&255,e.pending_buf[e.pending++]=255&t}function L(e,t){var r,n,i=e.max_chain_length,s=e.strstart,a=e.prev_length,o=e.nice_match,h=e.strstart>e.w_size-z?e.strstart-(e.w_size-z):0,u=e.window,l=e.w_mask,f=e.prev,c=e.strstart+S,d=u[s+a-1],p=u[s+a];e.prev_length>=e.good_match&&(i>>=2),o>e.lookahead&&(o=e.lookahead);do{if(u[(r=t)+a]===p&&u[r+a-1]===d&&u[r]===u[s]&&u[++r]===u[s+1]){s+=2,r++;do{}while(u[++s]===u[++r]&&u[++s]===u[++r]&&u[++s]===u[++r]&&u[++s]===u[++r]&&u[++s]===u[++r]&&u[++s]===u[++r]&&u[++s]===u[++r]&&u[++s]===u[++r]&&sh&&0!=--i);return a<=e.lookahead?a:e.lookahead}function j(e){var t,r,n,i,s,a,o,h,u,l,f=e.w_size;do{if(i=e.window_size-e.lookahead-e.strstart,e.strstart>=f+(f-z)){for(c.arraySet(e.window,e.window,f,f,0),e.match_start-=f,e.strstart-=f,e.block_start-=f,t=r=e.hash_size;n=e.head[--t],e.head[t]=f<=n?n-f:0,--r;);for(t=r=f;n=e.prev[--t],e.prev[t]=f<=n?n-f:0,--r;);i+=f}if(0===e.strm.avail_in)break;if(a=e.strm,o=e.window,h=e.strstart+e.lookahead,u=i,l=void 0,l=a.avail_in,u=x)for(s=e.strstart-e.insert,e.ins_h=e.window[s],e.ins_h=(e.ins_h<=x&&(e.ins_h=(e.ins_h<=x)if(n=u._tr_tally(e,e.strstart-e.match_start,e.match_length-x),e.lookahead-=e.match_length,e.match_length<=e.max_lazy_match&&e.lookahead>=x){for(e.match_length--;e.strstart++,e.ins_h=(e.ins_h<=x&&(e.ins_h=(e.ins_h<=x&&e.match_length<=e.prev_length){for(i=e.strstart+e.lookahead-x,n=u._tr_tally(e,e.strstart-1-e.prev_match,e.prev_length-x),e.lookahead-=e.prev_length-1,e.prev_length-=2;++e.strstart<=i&&(e.ins_h=(e.ins_h<e.pending_buf_size-5&&(r=e.pending_buf_size-5);;){if(e.lookahead<=1){if(j(e),0===e.lookahead&&t===l)return A;if(0===e.lookahead)break}e.strstart+=e.lookahead,e.lookahead=0;var n=e.block_start+r;if((0===e.strstart||e.strstart>=n)&&(e.lookahead=e.strstart-n,e.strstart=n,N(e,!1),0===e.strm.avail_out))return A;if(e.strstart-e.block_start>=e.w_size-z&&(N(e,!1),0===e.strm.avail_out))return A}return e.insert=0,t===f?(N(e,!0),0===e.strm.avail_out?O:B):(e.strstart>e.block_start&&(N(e,!1),e.strm.avail_out),A)}),new M(4,4,8,4,Z),new M(4,5,16,8,Z),new M(4,6,32,32,Z),new M(4,4,16,16,W),new M(8,16,32,32,W),new M(8,16,128,128,W),new M(8,32,128,256,W),new M(32,128,258,1024,W),new M(32,258,258,4096,W)],r.deflateInit=function(e,t){return Y(e,t,v,15,8,0)},r.deflateInit2=Y,r.deflateReset=K,r.deflateResetKeep=G,r.deflateSetHeader=function(e,t){return e&&e.state?2!==e.state.wrap?_:(e.state.gzhead=t,m):_},r.deflate=function(e,t){var r,n,i,s;if(!e||!e.state||5>8&255),U(n,n.gzhead.time>>16&255),U(n,n.gzhead.time>>24&255),U(n,9===n.level?2:2<=n.strategy||n.level<2?4:0),U(n,255&n.gzhead.os),n.gzhead.extra&&n.gzhead.extra.length&&(U(n,255&n.gzhead.extra.length),U(n,n.gzhead.extra.length>>8&255)),n.gzhead.hcrc&&(e.adler=p(e.adler,n.pending_buf,n.pending,0)),n.gzindex=0,n.status=69):(U(n,0),U(n,0),U(n,0),U(n,0),U(n,0),U(n,9===n.level?2:2<=n.strategy||n.level<2?4:0),U(n,3),n.status=E);else{var a=v+(n.w_bits-8<<4)<<8;a|=(2<=n.strategy||n.level<2?0:n.level<6?1:6===n.level?2:3)<<6,0!==n.strstart&&(a|=32),a+=31-a%31,n.status=E,P(n,a),0!==n.strstart&&(P(n,e.adler>>>16),P(n,65535&e.adler)),e.adler=1}if(69===n.status)if(n.gzhead.extra){for(i=n.pending;n.gzindex<(65535&n.gzhead.extra.length)&&(n.pending!==n.pending_buf_size||(n.gzhead.hcrc&&n.pending>i&&(e.adler=p(e.adler,n.pending_buf,n.pending-i,i)),F(e),i=n.pending,n.pending!==n.pending_buf_size));)U(n,255&n.gzhead.extra[n.gzindex]),n.gzindex++;n.gzhead.hcrc&&n.pending>i&&(e.adler=p(e.adler,n.pending_buf,n.pending-i,i)),n.gzindex===n.gzhead.extra.length&&(n.gzindex=0,n.status=73)}else n.status=73;if(73===n.status)if(n.gzhead.name){i=n.pending;do{if(n.pending===n.pending_buf_size&&(n.gzhead.hcrc&&n.pending>i&&(e.adler=p(e.adler,n.pending_buf,n.pending-i,i)),F(e),i=n.pending,n.pending===n.pending_buf_size)){s=1;break}s=n.gzindexi&&(e.adler=p(e.adler,n.pending_buf,n.pending-i,i)),0===s&&(n.gzindex=0,n.status=91)}else n.status=91;if(91===n.status)if(n.gzhead.comment){i=n.pending;do{if(n.pending===n.pending_buf_size&&(n.gzhead.hcrc&&n.pending>i&&(e.adler=p(e.adler,n.pending_buf,n.pending-i,i)),F(e),i=n.pending,n.pending===n.pending_buf_size)){s=1;break}s=n.gzindexi&&(e.adler=p(e.adler,n.pending_buf,n.pending-i,i)),0===s&&(n.status=103)}else n.status=103;if(103===n.status&&(n.gzhead.hcrc?(n.pending+2>n.pending_buf_size&&F(e),n.pending+2<=n.pending_buf_size&&(U(n,255&e.adler),U(n,e.adler>>8&255),e.adler=0,n.status=E)):n.status=E),0!==n.pending){if(F(e),0===e.avail_out)return n.last_flush=-1,m}else if(0===e.avail_in&&T(t)<=T(r)&&t!==f)return R(e,-5);if(666===n.status&&0!==e.avail_in)return R(e,-5);if(0!==e.avail_in||0!==n.lookahead||t!==l&&666!==n.status){var o=2===n.strategy?function(e,t){for(var r;;){if(0===e.lookahead&&(j(e),0===e.lookahead)){if(t===l)return A;break}if(e.match_length=0,r=u._tr_tally(e,0,e.window[e.strstart]),e.lookahead--,e.strstart++,r&&(N(e,!1),0===e.strm.avail_out))return A}return e.insert=0,t===f?(N(e,!0),0===e.strm.avail_out?O:B):e.last_lit&&(N(e,!1),0===e.strm.avail_out)?A:I}(n,t):3===n.strategy?function(e,t){for(var r,n,i,s,a=e.window;;){if(e.lookahead<=S){if(j(e),e.lookahead<=S&&t===l)return A;if(0===e.lookahead)break}if(e.match_length=0,e.lookahead>=x&&0e.lookahead&&(e.match_length=e.lookahead)}if(e.match_length>=x?(r=u._tr_tally(e,1,e.match_length-x),e.lookahead-=e.match_length,e.strstart+=e.match_length,e.match_length=0):(r=u._tr_tally(e,0,e.window[e.strstart]),e.lookahead--,e.strstart++),r&&(N(e,!1),0===e.strm.avail_out))return A}return e.insert=0,t===f?(N(e,!0),0===e.strm.avail_out?O:B):e.last_lit&&(N(e,!1),0===e.strm.avail_out)?A:I}(n,t):h[n.level].func(n,t);if(o!==O&&o!==B||(n.status=666),o===A||o===O)return 0===e.avail_out&&(n.last_flush=-1),m;if(o===I&&(1===t?u._tr_align(n):5!==t&&(u._tr_stored_block(n,0,0,!1),3===t&&(D(n.head),0===n.lookahead&&(n.strstart=0,n.block_start=0,n.insert=0))),F(e),0===e.avail_out))return n.last_flush=-1,m}return t!==f?m:n.wrap<=0?1:(2===n.wrap?(U(n,255&e.adler),U(n,e.adler>>8&255),U(n,e.adler>>16&255),U(n,e.adler>>24&255),U(n,255&e.total_in),U(n,e.total_in>>8&255),U(n,e.total_in>>16&255),U(n,e.total_in>>24&255)):(P(n,e.adler>>>16),P(n,65535&e.adler)),F(e),0=r.w_size&&(0===s&&(D(r.head),r.strstart=0,r.block_start=0,r.insert=0),u=new c.Buf8(r.w_size),c.arraySet(u,t,l-r.w_size,r.w_size,0),t=u,l=r.w_size),a=e.avail_in,o=e.next_in,h=e.input,e.avail_in=l,e.next_in=0,e.input=t,j(r);r.lookahead>=x;){for(n=r.strstart,i=r.lookahead-(x-1);r.ins_h=(r.ins_h<>>=y=v>>>24,p-=y,0===(y=v>>>16&255))C[s++]=65535&v;else{if(!(16&y)){if(0==(64&y)){v=m[(65535&v)+(d&(1<>>=y,p-=y),p<15&&(d+=z[n++]<>>=y=v>>>24,p-=y,!(16&(y=v>>>16&255))){if(0==(64&y)){v=_[(65535&v)+(d&(1<>>=y,p-=y,(y=s-a)>3,d&=(1<<(p-=w<<3))-1,e.next_in=n,e.next_out=s,e.avail_in=n>>24&255)+(e>>>8&65280)+((65280&e)<<8)+((255&e)<<24)}function s(){this.mode=0,this.last=!1,this.wrap=0,this.havedict=!1,this.flags=0,this.dmax=0,this.check=0,this.total=0,this.head=null,this.wbits=0,this.wsize=0,this.whave=0,this.wnext=0,this.window=null,this.hold=0,this.bits=0,this.length=0,this.offset=0,this.extra=0,this.lencode=null,this.distcode=null,this.lenbits=0,this.distbits=0,this.ncode=0,this.nlen=0,this.ndist=0,this.have=0,this.next=null,this.lens=new I.Buf16(320),this.work=new I.Buf16(288),this.lendyn=null,this.distdyn=null,this.sane=0,this.back=0,this.was=0}function a(e){var t;return e&&e.state?(t=e.state,e.total_in=e.total_out=t.total=0,e.msg="",t.wrap&&(e.adler=1&t.wrap),t.mode=P,t.last=0,t.havedict=0,t.dmax=32768,t.head=null,t.hold=0,t.bits=0,t.lencode=t.lendyn=new I.Buf32(n),t.distcode=t.distdyn=new I.Buf32(i),t.sane=1,t.back=-1,N):U}function o(e){var t;return e&&e.state?((t=e.state).wsize=0,t.whave=0,t.wnext=0,a(e)):U}function h(e,t){var r,n;return e&&e.state?(n=e.state,t<0?(r=0,t=-t):(r=1+(t>>4),t<48&&(t&=15)),t&&(t<8||15=s.wsize?(I.arraySet(s.window,t,r-s.wsize,s.wsize,0),s.wnext=0,s.whave=s.wsize):(n<(i=s.wsize-s.wnext)&&(i=n),I.arraySet(s.window,t,r-n,i,s.wnext),(n-=i)?(I.arraySet(s.window,t,r-n,n,0),s.wnext=n,s.whave=s.wsize):(s.wnext+=i,s.wnext===s.wsize&&(s.wnext=0),s.whave>>8&255,r.check=B(r.check,E,2,0),l=u=0,r.mode=2;break}if(r.flags=0,r.head&&(r.head.done=!1),!(1&r.wrap)||(((255&u)<<8)+(u>>8))%31){e.msg="incorrect header check",r.mode=30;break}if(8!=(15&u)){e.msg="unknown compression method",r.mode=30;break}if(l-=4,k=8+(15&(u>>>=4)),0===r.wbits)r.wbits=k;else if(k>r.wbits){e.msg="invalid window size",r.mode=30;break}r.dmax=1<>8&1),512&r.flags&&(E[0]=255&u,E[1]=u>>>8&255,r.check=B(r.check,E,2,0)),l=u=0,r.mode=3;case 3:for(;l<32;){if(0===o)break e;o--,u+=n[s++]<>>8&255,E[2]=u>>>16&255,E[3]=u>>>24&255,r.check=B(r.check,E,4,0)),l=u=0,r.mode=4;case 4:for(;l<16;){if(0===o)break e;o--,u+=n[s++]<>8),512&r.flags&&(E[0]=255&u,E[1]=u>>>8&255,r.check=B(r.check,E,2,0)),l=u=0,r.mode=5;case 5:if(1024&r.flags){for(;l<16;){if(0===o)break e;o--,u+=n[s++]<>>8&255,r.check=B(r.check,E,2,0)),l=u=0}else r.head&&(r.head.extra=null);r.mode=6;case 6:if(1024&r.flags&&(o<(d=r.length)&&(d=o),d&&(r.head&&(k=r.head.extra_len-r.length,r.head.extra||(r.head.extra=new Array(r.head.extra_len)),I.arraySet(r.head.extra,n,s,d,k)),512&r.flags&&(r.check=B(r.check,n,d,s)),o-=d,s+=d,r.length-=d),r.length))break e;r.length=0,r.mode=7;case 7:if(2048&r.flags){if(0===o)break e;for(d=0;k=n[s+d++],r.head&&k&&r.length<65536&&(r.head.name+=String.fromCharCode(k)),k&&d>9&1,r.head.done=!0),e.adler=r.check=0,r.mode=12;break;case 10:for(;l<32;){if(0===o)break e;o--,u+=n[s++]<>>=7&l,l-=7&l,r.mode=27;break}for(;l<3;){if(0===o)break e;o--,u+=n[s++]<>>=1)){case 0:r.mode=14;break;case 1:if(j(r),r.mode=20,6!==t)break;u>>>=2,l-=2;break e;case 2:r.mode=17;break;case 3:e.msg="invalid block type",r.mode=30}u>>>=2,l-=2;break;case 14:for(u>>>=7&l,l-=7&l;l<32;){if(0===o)break e;o--,u+=n[s++]<>>16^65535)){e.msg="invalid stored block lengths",r.mode=30;break}if(r.length=65535&u,l=u=0,r.mode=15,6===t)break e;case 15:r.mode=16;case 16:if(d=r.length){if(o>>=5,l-=5,r.ndist=1+(31&u),u>>>=5,l-=5,r.ncode=4+(15&u),u>>>=4,l-=4,286>>=3,l-=3}for(;r.have<19;)r.lens[A[r.have++]]=0;if(r.lencode=r.lendyn,r.lenbits=7,S={bits:r.lenbits},x=T(0,r.lens,0,19,r.lencode,0,r.work,S),r.lenbits=S.bits,x){e.msg="invalid code lengths set",r.mode=30;break}r.have=0,r.mode=19;case 19:for(;r.have>>16&255,b=65535&C,!((_=C>>>24)<=l);){if(0===o)break e;o--,u+=n[s++]<>>=_,l-=_,r.lens[r.have++]=b;else{if(16===b){for(z=_+2;l>>=_,l-=_,0===r.have){e.msg="invalid bit length repeat",r.mode=30;break}k=r.lens[r.have-1],d=3+(3&u),u>>>=2,l-=2}else if(17===b){for(z=_+3;l>>=_)),u>>>=3,l-=3}else{for(z=_+7;l>>=_)),u>>>=7,l-=7}if(r.have+d>r.nlen+r.ndist){e.msg="invalid bit length repeat",r.mode=30;break}for(;d--;)r.lens[r.have++]=k}}if(30===r.mode)break;if(0===r.lens[256]){e.msg="invalid code -- missing end-of-block",r.mode=30;break}if(r.lenbits=9,S={bits:r.lenbits},x=T(D,r.lens,0,r.nlen,r.lencode,0,r.work,S),r.lenbits=S.bits,x){e.msg="invalid literal/lengths set",r.mode=30;break}if(r.distbits=6,r.distcode=r.distdyn,S={bits:r.distbits},x=T(F,r.lens,r.nlen,r.ndist,r.distcode,0,r.work,S),r.distbits=S.bits,x){e.msg="invalid distances set",r.mode=30;break}if(r.mode=20,6===t)break e;case 20:r.mode=21;case 21:if(6<=o&&258<=h){e.next_out=a,e.avail_out=h,e.next_in=s,e.avail_in=o,r.hold=u,r.bits=l,R(e,c),a=e.next_out,i=e.output,h=e.avail_out,s=e.next_in,n=e.input,o=e.avail_in,u=r.hold,l=r.bits,12===r.mode&&(r.back=-1);break}for(r.back=0;g=(C=r.lencode[u&(1<>>16&255,b=65535&C,!((_=C>>>24)<=l);){if(0===o)break e;o--,u+=n[s++]<>v)])>>>16&255,b=65535&C,!(v+(_=C>>>24)<=l);){if(0===o)break e;o--,u+=n[s++]<>>=v,l-=v,r.back+=v}if(u>>>=_,l-=_,r.back+=_,r.length=b,0===g){r.mode=26;break}if(32&g){r.back=-1,r.mode=12;break}if(64&g){e.msg="invalid literal/length code",r.mode=30;break}r.extra=15&g,r.mode=22;case 22:if(r.extra){for(z=r.extra;l>>=r.extra,l-=r.extra,r.back+=r.extra}r.was=r.length,r.mode=23;case 23:for(;g=(C=r.distcode[u&(1<>>16&255,b=65535&C,!((_=C>>>24)<=l);){if(0===o)break e;o--,u+=n[s++]<>v)])>>>16&255,b=65535&C,!(v+(_=C>>>24)<=l);){if(0===o)break e;o--,u+=n[s++]<>>=v,l-=v,r.back+=v}if(u>>>=_,l-=_,r.back+=_,64&g){e.msg="invalid distance code",r.mode=30;break}r.offset=b,r.extra=15&g,r.mode=24;case 24:if(r.extra){for(z=r.extra;l>>=r.extra,l-=r.extra,r.back+=r.extra}if(r.offset>r.dmax){e.msg="invalid distance too far back",r.mode=30;break}r.mode=25;case 25:if(0===h)break e;if(d=c-h,r.offset>d){if((d=r.offset-d)>r.whave&&r.sane){e.msg="invalid distance too far back",r.mode=30;break}p=d>r.wnext?(d-=r.wnext,r.wsize-d):r.wnext-d,d>r.length&&(d=r.length),m=r.window}else m=i,p=a-r.offset,d=r.length;for(hd?(m=R[T+a[v]],A[I+a[v]]):(m=96,0),h=1<>S)+(u-=h)]=p<<24|m<<16|_|0,0!==u;);for(h=1<>=1;if(0!==h?(E&=h-1,E+=h):E=0,v++,0==--O[b]){if(b===w)break;b=t[r+a[v]]}if(k>>7)]}function U(e,t){e.pending_buf[e.pending++]=255&t,e.pending_buf[e.pending++]=t>>>8&255}function P(e,t,r){e.bi_valid>d-r?(e.bi_buf|=t<>d-e.bi_valid,e.bi_valid+=r-d):(e.bi_buf|=t<>>=1,r<<=1,0<--t;);return r>>>1}function Z(e,t,r){var n,i,s=new Array(g+1),a=0;for(n=1;n<=g;n++)s[n]=a=a+r[n-1]<<1;for(i=0;i<=t;i++){var o=e[2*i+1];0!==o&&(e[2*i]=j(s[o]++,o))}}function W(e){var t;for(t=0;t>1;1<=r;r--)G(e,s,r);for(i=h;r=e.heap[1],e.heap[1]=e.heap[e.heap_len--],G(e,s,1),n=e.heap[1],e.heap[--e.heap_max]=r,e.heap[--e.heap_max]=n,s[2*i]=s[2*r]+s[2*n],e.depth[i]=(e.depth[r]>=e.depth[n]?e.depth[r]:e.depth[n])+1,s[2*r+1]=s[2*n+1]=i,e.heap[1]=i++,G(e,s,1),2<=e.heap_len;);e.heap[--e.heap_max]=e.heap[1],function(e,t){var r,n,i,s,a,o,h=t.dyn_tree,u=t.max_code,l=t.stat_desc.static_tree,f=t.stat_desc.has_stree,c=t.stat_desc.extra_bits,d=t.stat_desc.extra_base,p=t.stat_desc.max_length,m=0;for(s=0;s<=g;s++)e.bl_count[s]=0;for(h[2*e.heap[e.heap_max]+1]=0,r=e.heap_max+1;r<_;r++)p<(s=h[2*h[2*(n=e.heap[r])+1]+1]+1)&&(s=p,m++),h[2*n+1]=s,u>=7;n>>=1)if(1&r&&0!==e.dyn_ltree[2*t])return o;if(0!==e.dyn_ltree[18]||0!==e.dyn_ltree[20]||0!==e.dyn_ltree[26])return h;for(t=32;t>>3,(s=e.static_len+3+7>>>3)<=i&&(i=s)):i=s=r+5,r+4<=i&&-1!==t?J(e,t,r,n):4===e.strategy||s===i?(P(e,2+(n?1:0),3),K(e,z,C)):(P(e,4+(n?1:0),3),function(e,t,r,n){var i;for(P(e,t-257,5),P(e,r-1,5),P(e,n-4,4),i=0;i>>8&255,e.pending_buf[e.d_buf+2*e.last_lit+1]=255&t,e.pending_buf[e.l_buf+e.last_lit]=255&r,e.last_lit++,0===t?e.dyn_ltree[2*r]++:(e.matches++,t--,e.dyn_ltree[2*(A[r]+u+1)]++,e.dyn_dtree[2*N(t)]++),e.last_lit===e.lit_bufsize-1},r._tr_align=function(e){P(e,2,3),L(e,m,z),function(e){16===e.bi_valid?(U(e,e.bi_buf),e.bi_buf=0,e.bi_valid=0):8<=e.bi_valid&&(e.pending_buf[e.pending++]=255&e.bi_buf,e.bi_buf>>=8,e.bi_valid-=8)}(e)}},{"../utils/common":41}],53:[function(e,t,r){"use strict";t.exports=function(){this.input=null,this.next_in=0,this.avail_in=0,this.total_in=0,this.output=null,this.next_out=0,this.avail_out=0,this.total_out=0,this.msg="",this.state=null,this.data_type=2,this.adler=0}},{}],54:[function(e,t,r){(function(e){!function(r,n){"use strict";if(!r.setImmediate){var i,s,t,a,o=1,h={},u=!1,l=r.document,e=Object.getPrototypeOf&&Object.getPrototypeOf(r);e=e&&e.setTimeout?e:r,i="[object process]"==={}.toString.call(r.process)?function(e){process.nextTick(function(){c(e)})}:function(){if(r.postMessage&&!r.importScripts){var e=!0,t=r.onmessage;return r.onmessage=function(){e=!1},r.postMessage("","*"),r.onmessage=t,e}}()?(a="setImmediate$"+Math.random()+"$",r.addEventListener?r.addEventListener("message",d,!1):r.attachEvent("onmessage",d),function(e){r.postMessage(a+e,"*")}):r.MessageChannel?((t=new MessageChannel).port1.onmessage=function(e){c(e.data)},function(e){t.port2.postMessage(e)}):l&&"onreadystatechange"in l.createElement("script")?(s=l.documentElement,function(e){var t=l.createElement("script");t.onreadystatechange=function(){c(e),t.onreadystatechange=null,s.removeChild(t),t=null},s.appendChild(t)}):function(e){setTimeout(c,0,e)},e.setImmediate=function(e){"function"!=typeof e&&(e=new Function(""+e));for(var t=new Array(arguments.length-1),r=0;r parseInt(o, 16))); - } break; - case "binary": { - if (!/^(?:[01]{8})*$/i.test(DATA)) return; - const dataArr = this.splitIntoParts(DATA, 8); - DATA = Uint8Array.from(dataArr.map(o => parseInt(o, 2))); - } break; + case "URL": + { + if (TYPE === "base64") + DATA = "data:application/zip;base64," + DATA; + const resp = await Scratch.fetch(DATA); + DATA = await resp.blob(); + } + break; + case "hex": + { + if (!/^(?:[0-9A-F]{2})*$/i.test(DATA)) return; + const dataArr = this.splitIntoParts(DATA, 2); + DATA = Uint8Array.from(dataArr.map((o) => parseInt(o, 16))); + } + break; + case "binary": + { + if (!/^(?:[01]{8})*$/i.test(DATA)) return; + const dataArr = this.splitIntoParts(DATA, 8); + DATA = Uint8Array.from(dataArr.map((o) => parseInt(o, 2))); + } + break; } this.zip = await JSZip.loadAsync(DATA, { createFolders: true }); @@ -476,20 +460,23 @@ COMPRESSION = Math.max(Math.min(Math.round(COMPRESSION), 9), 0); const compType = COMPRESSION === 0 ? "STORE" : "DEFLATE"; - const options = { compression: compType, compressionOptions: { level: COMPRESSION } }; + const options = { + compression: compType, + compressionOptions: { level: COMPRESSION }, + }; switch (TYPE) { case "text": case "string": return await this.zip.generateAsync({ type: "binarystring", - ...options + ...options, }); case "base64": case "data: URL": { let data = await this.zip.generateAsync({ type: "base64", - ...options + ...options, }); if (TYPE === "data: URL") data = "data:application/zip;base64," + data; @@ -498,21 +485,29 @@ case "hex": { const data = await this.zip.generateAsync({ type: "array", - ...options + ...options, }); - return data.map(data => data.toString(16).padStart(2, "0")).join(""); + return data + .map((data) => data.toString(16).padStart(2, "0")) + .join(""); } case "binary": { const data = await this.zip.generateAsync({ type: "array", - ...options + ...options, }); - return data.map(data => data.toString(2).padStart(8, "0")).join(""); + return data + .map((data) => data.toString(2).padStart(8, "0")) + .join(""); } - default: return ""; + default: + return ""; } } catch (e) { - console.error(`Zip extension: Error creating zip with type ${TYPE} compression ${COMPRESSION}:`, e); + console.error( + `Zip extension: Error creating zip with type ${TYPE} compression ${COMPRESSION}:`, + e + ); } } close() { @@ -525,7 +520,9 @@ exists({ OBJECT }) { try { - return !!this.getObj(this.normalize(this.zipPath, Scratch.Cast.toString(OBJECT))); + return !!this.getObj( + this.normalize(this.zipPath, Scratch.Cast.toString(OBJECT)) + ); } catch (e) { return false; } @@ -553,16 +550,24 @@ } case "hex": { const data = await obj.async("array"); - return data.map(data => data.toString(16).padStart(2, "0")).join(""); + return data + .map((data) => data.toString(16).padStart(2, "0")) + .join(""); } case "binary": { const data = await obj.async("array"); - return data.map(data => data.toString(2).padStart(8, "0")).join(""); + return data + .map((data) => data.toString(2).padStart(8, "0")) + .join(""); } - default: return ""; + default: + return ""; } } catch (e) { - console.error(`Zip extension: Error getting file ${FILE} with type ${TYPE}:`, e); + console.error( + `Zip extension: Error getting file ${FILE} with type ${TYPE}:`, + e + ); return ""; } } @@ -588,7 +593,8 @@ }); break; case "base64": - case "data: URL": { // compatibility + case "data: URL": { + // compatibility if (TYPE === "data: URL") CONTENT = CONTENT.substring(CONTENT.indexOf(",")); this.zip.file(path, CONTENT, { @@ -597,33 +603,43 @@ }); break; } - case "URL": { - const resp = await Scratch.fetch(CONTENT); - this.zip.file(path, await resp.blob(), { - base64: true, - createFolders: true, - }); - } break; - case "hex": { - if (!/^(?:[0-9A-F]{2})*$/i.test(CONTENT)) return ""; - const dataArr = this.splitIntoParts(CONTENT, 2); - const data = Uint8Array.from(dataArr.map(o => parseInt(o, 16))); - this.zip.file(path, data, { - createFolders: true, - }); - } break; - case "binary": { - if (!/^(?:[01]{8})*$/i.test(CONTENT)) return ""; - const dataArr = this.splitIntoParts(CONTENT, 8); - const data = Uint8Array.from(dataArr.map(o => parseInt(o, 2))); - this.zip.file(path, data, { - createFolders: true, - }); - } break; - default: return ""; + case "URL": + { + const resp = await Scratch.fetch(CONTENT); + this.zip.file(path, await resp.blob(), { + base64: true, + createFolders: true, + }); + } + break; + case "hex": + { + if (!/^(?:[0-9A-F]{2})*$/i.test(CONTENT)) return ""; + const dataArr = this.splitIntoParts(CONTENT, 2); + const data = Uint8Array.from(dataArr.map((o) => parseInt(o, 16))); + this.zip.file(path, data, { + createFolders: true, + }); + } + break; + case "binary": + { + if (!/^(?:[01]{8})*$/i.test(CONTENT)) return ""; + const dataArr = this.splitIntoParts(CONTENT, 8); + const data = Uint8Array.from(dataArr.map((o) => parseInt(o, 2))); + this.zip.file(path, data, { + createFolders: true, + }); + } + break; + default: + return ""; } } catch (e) { - console.error(`Zip extension: Error writing to file ${FILE} type ${TYPE}:`, e); + console.error( + `Zip extension: Error writing to file ${FILE} type ${TYPE}:`, + e + ); } } renameFile({ FROM, TO }) { @@ -649,7 +665,10 @@ let toPath = this.normalize(this.zipPath, TO); const replacedTo = TO.replaceAll(/\\/g, "/"); const slashes = replacedTo.split("/").length - 1; - if (slashes <= +fromObj.dir && (slashes === 0 || replacedTo.endsWith("/"))) { + if ( + slashes <= +fromObj.dir && + (slashes === 0 || replacedTo.endsWith("/")) + ) { // this is a name-only change toPath = this.normalize(fromPath, "../" + replacedTo); if (fromObj.dir) { @@ -675,7 +694,7 @@ // Move current directory if (this.zipPath.substring(1).startsWith(fromPath)) { this.zipPath = - "/" + toPath + (this.zipPath.substring(1).substring(fromPath.length)); + "/" + toPath + this.zipPath.substring(1).substring(fromPath.length); } for (const path in this.zip.files) { @@ -697,10 +716,10 @@ if (!this.getObj(path)) return; if (path === "/") return; - const shouldGoBack = this.getObj(path).dir && this.zipPath.startsWith(path); + const shouldGoBack = + this.getObj(path).dir && this.zipPath.startsWith(path); if (path.startsWith("/")) path = path.substring(1); - this.zip.remove(path); if (shouldGoBack) { @@ -737,18 +756,23 @@ const obj = this.getObj(normalized); if (!obj) return ""; switch (META) { - case "modified days since 2000": { - const msPerDay = 24 * 60 * 60 * 1000; - const start = +(new Date(2000, 0, 1)); - obj.date = new Date(start + (Scratch.Cast.toNumber(VALUE) * msPerDay)); - } break; + case "modified days since 2000": + { + const msPerDay = 24 * 60 * 60 * 1000; + const start = +new Date(2000, 0, 1); + obj.date = new Date( + start + Scratch.Cast.toNumber(VALUE) * msPerDay + ); + } + break; case "unix modified timestamp": obj.date = new Date(Scratch.Cast.toNumber(VALUE)); break; case "comment": obj.comment = VALUE; break; - default: return; + default: + return; } } catch (e) { console.error(`Zip extension: Error getting ${META} of ${FILE}:`, e); @@ -776,28 +800,29 @@ case "folder": { /** @type {Array} */ const splitPath = obj.name.split("/"); - const folders = (splitPath.slice( - 0, splitPath.length - 1 - +obj.dir - ).join("/")); + const folders = splitPath + .slice(0, splitPath.length - 1 - +obj.dir) + .join("/"); return "/" + folders + (folders === "" ? "" : "/"); } case "modification date": return obj.date.toLocaleString(navigator.language); case "long modification date": - return new Date().toLocaleString( - navigator.language, - { dateStyle: "full", timeStyle: "medium" } - ); + return new Date().toLocaleString(navigator.language, { + dateStyle: "full", + timeStyle: "medium", + }); case "modified days since 2000": { const msPerDay = 24 * 60 * 60 * 1000; - const start = +(new Date(2000, 0, 1)); + const start = +new Date(2000, 0, 1); return (+obj.date - start) / msPerDay; } case "unix modified timestamp": return +obj.date; case "comment": return obj.comment || ""; - default: return ""; + default: + return ""; } } catch (e) { console.error(`Zip extension: Error getting ${META} of ${FILE}:`, e); @@ -824,7 +849,7 @@ try { let newPath = this.normalize(this.zipPath, DIR); if (!newPath.endsWith("/")) newPath += "/"; - if (!this.getObj(newPath)) return; + if (!this.getObj(newPath) && newPath !== "/") return; this.zipPath = newPath; } catch (e) { console.error(`Error going to directory ${DIR}:`, e); @@ -841,15 +866,20 @@ const dir = normalized.substring(1); const length = dir.length; - return JSON.stringify(Object.values(this.zip.files).filter((obj) => { - // Above the current directory - if (!obj.name.startsWith(dir)) return false; - // Below the current directory - if (obj.name.substring(length).split("/").length > (obj.dir + 1)) return false; - // Is the current directory - if (obj.name === dir) return false; - return true; - }).map(obj => obj.name.substring(length))); + return JSON.stringify( + Object.values(this.zip.files) + .filter((obj) => { + // Above the current directory + if (!obj.name.startsWith(dir)) return false; + // Below the current directory + if (obj.name.substring(length).split("/").length > obj.dir + 1) + return false; + // Is the current directory + if (obj.name === dir) return false; + return true; + }) + .map((obj) => obj.name.substring(length)) + ); } catch (e) { console.error(`Zip extension: Could not get directory ${DIR}:`, e); return ""; diff --git a/extensions/CubesterYT/TurboHook.js b/extensions/CubesterYT/TurboHook.js index 864b3c474c..a87d1192a4 100644 --- a/extensions/CubesterYT/TurboHook.js +++ b/extensions/CubesterYT/TurboHook.js @@ -1,7 +1,13 @@ +// Name: TurboHook +// ID: cubesterTurboHook +// Description: Allows you to use webhooks. +// By: CubesterYT + (function (Scratch) { "use strict"; - const icon = ""; + const icon = + ""; const parseOrEmptyObject = (str) => { try { @@ -21,7 +27,7 @@ color2: "#2f39a1", color3: "#28318f", menuIconURI: icon, - docsURI: "https://extensions.turbowarp.org/CubesterYT/TurboHook.html", + docsURI: "https://extensions.turbowarp.org/CubesterYT/TurboHook", blocks: [ { @@ -30,9 +36,9 @@ blockType: Scratch.BlockType.COMMAND, arguments: { hookURL: { - type: Scratch.ArgumentType.STRING - } - } + type: Scratch.ArgumentType.STRING, + }, + }, }, { opcode: "params", @@ -41,29 +47,29 @@ arguments: { MENU: { type: Scratch.ArgumentType.STRING, - menu: "PARAMS" + menu: "PARAMS", }, DATA: { - type: Scratch.ArgumentType.STRING - } - } + type: Scratch.ArgumentType.STRING, + }, + }, }, { opcode: "connector", blockType: Scratch.BlockType.REPORTER, - text: "[STRING1] , [STRING2]" - } + text: "[STRING1] , [STRING2]", + }, ], menus: { PARAMS: { acceptReporters: true, - items: ["content", "name", "icon"] - } - } + items: ["content", "name", "icon"], + }, + }, }; } - webhook ({hookDATA, hookURL}) { + webhook({ hookDATA, hookURL }) { const data = parseOrEmptyObject(hookDATA); if (!data.content) { // Typically this can't be empty, so put something there if they forgot to @@ -72,12 +78,12 @@ Scratch.fetch(hookURL, { method: "POST", headers: { - "Content-Type": "application/json" + "Content-Type": "application/json", }, - body: JSON.stringify(data) + body: JSON.stringify(data), }); } - params ({MENU, DATA}) { + params({ MENU, DATA }) { DATA = Scratch.Cast.toString(DATA); if (MENU == "content") { return JSON.stringify({ content: DATA }); @@ -88,13 +94,13 @@ } return "{}"; } - connector ({STRING1, STRING2}) { + connector({ STRING1, STRING2 }) { return JSON.stringify({ ...parseOrEmptyObject(STRING1), - ...parseOrEmptyObject(STRING2) + ...parseOrEmptyObject(STRING2), }); } } Scratch.extensions.register(new TurboHook()); -})(Scratch); \ No newline at end of file +})(Scratch); diff --git a/extensions/DNin/wake-lock.js b/extensions/DNin/wake-lock.js new file mode 100644 index 0000000000..41b98c8aa2 --- /dev/null +++ b/extensions/DNin/wake-lock.js @@ -0,0 +1,118 @@ +// Name: Wake Lock +// ID: dninwakelock +// Description: Prevent the computer from falling asleep. +// By: D-ScratchNinja + +(function (Scratch) { + "use strict"; + + if (!Scratch.extensions.unsandboxed) { + throw new Error("Wake Lock extension must run unsandboxed"); + } + + /** @type {WakeLockSentinel} */ + let wakeLock = null; + let latestEnabled = false; + let promise = Promise.resolve(); + + class WakeLock { + constructor(runtime) { + this.runtime = runtime; + this.runtime.on("PROJECT_STOP_ALL", this.stopAll.bind(this)); + } + + getInfo() { + return { + id: "dninwakelock", + name: "Wake Lock", + docsURI: "https://extensions.turbowarp.org/DNin/wake-lock", + blocks: [ + { + opcode: "setWakeLock", + blockType: Scratch.BlockType.COMMAND, + text: "turn wake lock [enabled]", + arguments: { + enabled: { + type: Scratch.ArgumentType.STRING, + menu: "state", + defaultValue: "true", + }, + }, + }, + { + opcode: "isLocked", + blockType: Scratch.BlockType.BOOLEAN, + text: "is wake lock active?", + }, + ], + menus: { + state: { + acceptReporters: true, + items: [ + { + text: "on", + value: "true", + }, + { + text: "off", + value: "false", + }, + ], + }, + }, + }; + } + + stopAll() { + this.setWakeLock({ + enabled: false, + }); + } + + setWakeLock(args) { + if (!navigator.wakeLock) { + // Not supported in this browser. + return; + } + + const previousEnabled = latestEnabled; + latestEnabled = Scratch.Cast.toBoolean(args.enabled); + if (latestEnabled && !previousEnabled) { + promise = promise + .then(() => navigator.wakeLock.request("screen")) + .then((sentinel) => { + wakeLock = sentinel; + }) + .catch((error) => { + console.error(error); + // Allow to retry + latestEnabled = false; + }); + return promise; + } else if (!latestEnabled && previousEnabled) { + promise = promise + .then(() => { + if (wakeLock) { + return wakeLock.release(); + } else { + // Attempt to enable in the first place didn't work + } + }) + .then(() => { + wakeLock = null; + }) + .catch((error) => { + console.error(error); + wakeLock = null; + }); + return promise; + } + } + + isLocked() { + return !!wakeLock; + } + } + + Scratch.extensions.register(new WakeLock(Scratch.vm.runtime)); +})(Scratch); diff --git a/extensions/DT/cameracontrols.js b/extensions/DT/cameracontrols.js index 0c5f0cc4ab..4dc52493d6 100644 --- a/extensions/DT/cameracontrols.js +++ b/extensions/DT/cameracontrols.js @@ -1,13 +1,21 @@ -(Scratch => { - 'use strict'; +// Name: Camera Controls +// ID: DTcameracontrols +// Description: Move the visible part of the stage. +// By: DT + +((Scratch) => { + "use strict"; if (!Scratch.extensions.unsandboxed) { - throw new Error('Camera extension must be run unsandboxed'); + throw new Error("Camera extension must be run unsandboxed"); } - const icon = ''; - const CW = ''; - const CCW = ''; + const icon = + ""; + const CW = + ""; + const CCW = + ""; const vm = Scratch.vm; @@ -15,268 +23,284 @@ let cameraY = 0; let cameraZoom = 100; let cameraDirection = 90; - let cameraBG = '#ffffff'; + let cameraBG = "#ffffff"; vm.runtime.runtimeOptions.fencing = false; vm.renderer.offscreenTouching = true; + vm.setInterpolation(false); - function updateCamera(x = cameraX, y = cameraY, scale = cameraZoom / 100, rot = -cameraDirection + 90) { - rot = rot / 180 * Math.PI; + function updateCamera( + x = cameraX, + y = cameraY, + scale = cameraZoom / 100, + rot = -cameraDirection + 90 + ) { + rot = (rot / 180) * Math.PI; let s = Math.sin(rot) * scale; let c = Math.cos(rot) * scale; let w = vm.runtime.stageWidth / 2; let h = vm.runtime.stageHeight / 2; vm.renderer._projection = [ - c / w, -s / h, 0, 0, - s / w, c / h, 0, 0, - 0, 0, -1, 0, - (c * -x + s * -y) / w, (c * -y - s * -x) / h, 0, 1 + c / w, + -s / h, + 0, + 0, + s / w, + c / h, + 0, + 0, + 0, + 0, + -1, + 0, + (c * -x + s * -y) / w, + (c * -y - s * -x) / h, + 0, + 1, ]; vm.renderer.dirty = true; } // tell resize to update camera as well - vm.runtime.on('STAGE_SIZE_CHANGED', _ => updateCamera()); + vm.runtime.on("STAGE_SIZE_CHANGED", (_) => updateCamera()); // fix mouse positions let oldSX = vm.runtime.ioDevices.mouse.getScratchX; let oldSY = vm.runtime.ioDevices.mouse.getScratchY; - vm.runtime.ioDevices.mouse.getScratchX = function(...a) { - return (oldSX.apply(this, a) + cameraX) / cameraZoom * 100; + vm.runtime.ioDevices.mouse.getScratchX = function (...a) { + return ((oldSX.apply(this, a) + cameraX) / cameraZoom) * 100; }; - vm.runtime.ioDevices.mouse.getScratchY = function(...a) { - return (oldSY.apply(this, a) + cameraY) / cameraZoom * 100; + vm.runtime.ioDevices.mouse.getScratchY = function (...a) { + return ((oldSY.apply(this, a) + cameraY) / cameraZoom) * 100; }; class Camera { - getInfo() { return { + id: "DTcameracontrols", + name: "Camera", - id: 'DTcameracontrols', - name: 'Camera', - - color1: '#ff4da7', - color2: '#de4391', - color3: '#c83c82', + color1: "#ff4da7", + color2: "#de4391", + color3: "#c83c82", menuIconURI: icon, blocks: [ { - opcode: 'moveSteps', + opcode: "moveSteps", blockType: Scratch.BlockType.COMMAND, - text: 'move camera [val] steps', + text: "move camera [val] steps", arguments: { val: { type: Scratch.ArgumentType.NUMBER, - defaultValue: 10 + defaultValue: 10, }, - } + }, }, { - opcode: 'rotateCW', + opcode: "rotateCW", blockType: Scratch.BlockType.COMMAND, - text: 'turn camera [image] [val] degrees', + text: "turn camera [image] [val] degrees", arguments: { image: { type: Scratch.ArgumentType.IMAGE, - dataURI: CW + dataURI: CW, }, val: { type: Scratch.ArgumentType.ANGLE, - defaultValue: 15 - } - } + defaultValue: 15, + }, + }, }, { - opcode: 'rotateCCW', + opcode: "rotateCCW", blockType: Scratch.BlockType.COMMAND, - text: 'turn camera [image] [val] degrees', + text: "turn camera [image] [val] degrees", arguments: { image: { type: Scratch.ArgumentType.IMAGE, - dataURI: CCW + dataURI: CCW, }, val: { type: Scratch.ArgumentType.ANGLE, - defaultValue: 15 - } - } + defaultValue: 15, + }, + }, }, - '---', + "---", { - opcode: 'goTo', + opcode: "goTo", blockType: Scratch.BlockType.COMMAND, - text: 'move camera to [sprite]', + text: "move camera to [sprite]", arguments: { sprite: { type: Scratch.ArgumentType.STRING, menu: "sprites", }, - } + }, }, { - opcode: 'setBoth', + opcode: "setBoth", blockType: Scratch.BlockType.COMMAND, - text: 'set camera to x: [x] y: [y]', + text: "set camera to x: [x] y: [y]", arguments: { x: { type: Scratch.ArgumentType.NUMBER, - defaultValue: 0 + defaultValue: 0, }, y: { type: Scratch.ArgumentType.NUMBER, - defaultValue: 0 + defaultValue: 0, }, - } + }, }, - '---', + "---", { - opcode: 'setDirection', + opcode: "setDirection", blockType: Scratch.BlockType.COMMAND, - text: 'set camera direction to [val]', + text: "set camera direction to [val]", arguments: { val: { type: Scratch.ArgumentType.ANGLE, - defaultValue: 90 - } - } + defaultValue: 90, + }, + }, }, { - opcode: 'pointTowards', + opcode: "pointTowards", blockType: Scratch.BlockType.COMMAND, - text: 'point camera towards [sprite]', + text: "point camera towards [sprite]", arguments: { sprite: { type: Scratch.ArgumentType.STRING, menu: "sprites", }, - } + }, }, - '---', + "---", { - opcode: 'changeX', + opcode: "changeX", blockType: Scratch.BlockType.COMMAND, - text: 'change camera x by [val]', + text: "change camera x by [val]", arguments: { val: { type: Scratch.ArgumentType.NUMBER, - defaultValue: 10 - } - } + defaultValue: 10, + }, + }, }, { - opcode: 'setX', + opcode: "setX", blockType: Scratch.BlockType.COMMAND, - text: 'set camera x to [val]', + text: "set camera x to [val]", arguments: { val: { type: Scratch.ArgumentType.NUMBER, - defaultValue: 0 - } - } + defaultValue: 0, + }, + }, }, { - opcode: 'changeY', + opcode: "changeY", blockType: Scratch.BlockType.COMMAND, - text: 'change camera y by [val]', + text: "change camera y by [val]", arguments: { val: { type: Scratch.ArgumentType.NUMBER, - defaultValue: 10 - } - } + defaultValue: 10, + }, + }, }, { - opcode: 'setY', + opcode: "setY", blockType: Scratch.BlockType.COMMAND, - text: 'set camera y to [val]', + text: "set camera y to [val]", arguments: { val: { type: Scratch.ArgumentType.NUMBER, - defaultValue: 0 - } - } + defaultValue: 0, + }, + }, }, - '---', + "---", { - opcode: 'getX', + opcode: "getX", blockType: Scratch.BlockType.REPORTER, - text: 'camera x', + text: "camera x", }, { - opcode: 'getY', + opcode: "getY", blockType: Scratch.BlockType.REPORTER, - text: 'camera y', + text: "camera y", }, { - opcode: 'getDirection', + opcode: "getDirection", blockType: Scratch.BlockType.REPORTER, - text: 'camera direction', + text: "camera direction", }, - '---', + "---", { - opcode: 'changeZoom', + opcode: "changeZoom", blockType: Scratch.BlockType.COMMAND, - text: 'change camera zoom by [val]', + text: "change camera zoom by [val]", arguments: { val: { type: Scratch.ArgumentType.NUMBER, - defaultValue: 10 - } - } + defaultValue: 10, + }, + }, }, { - opcode: 'setZoom', + opcode: "setZoom", blockType: Scratch.BlockType.COMMAND, - text: 'set camera zoom to [val] %', + text: "set camera zoom to [val] %", arguments: { val: { type: Scratch.ArgumentType.NUMBER, - defaultValue: 100 - } - } + defaultValue: 100, + }, + }, }, { - opcode: 'getZoom', + opcode: "getZoom", blockType: Scratch.BlockType.REPORTER, - text: 'camera zoom', + text: "camera zoom", }, - '---', + "---", { - opcode: 'setCol', + opcode: "setCol", blockType: Scratch.BlockType.COMMAND, - text: 'set background color to [val]', + text: "set background color to [val]", arguments: { val: { - type: Scratch.ArgumentType.COLOR - } - } + type: Scratch.ArgumentType.COLOR, + }, + }, }, { - opcode: 'getCol', + opcode: "getCol", blockType: Scratch.BlockType.REPORTER, - text: 'background color', + text: "background color", }, ], menus: { sprites: { - items: 'getSprites', + items: "getSprites", acceptReporters: true, - } + }, }, }; } - getSprites(){ + getSprites() { let sprites = []; - Scratch.vm.runtime.targets.forEach(e=>{ + Scratch.vm.runtime.targets.forEach((e) => { if (e.isOriginal && !e.isStage) sprites.push(e.sprite.name); }); if (sprites.length === 0) { - sprites.push('no sprites exist'); + sprites.push("no sprites exist"); } return sprites; } @@ -346,13 +370,18 @@ } setCol(args, util) { const rgb = Scratch.Cast.toRgbColorList(args.val); - Scratch.vm.renderer.setBackgroundColor(rgb[0] / 255, rgb[1] / 255, rgb[2] / 255); + Scratch.vm.renderer.setBackgroundColor( + rgb[0] / 255, + rgb[1] / 255, + rgb[2] / 255 + ); + cameraBG = args.val; } getCol() { return cameraBG; } moveSteps(args) { - let dir = (-cameraDirection + 90) * Math.PI / 180; + let dir = ((-cameraDirection + 90) * Math.PI) / 180; cameraX += args.val * Math.cos(dir); cameraY += args.val * Math.sin(dir); updateCamera(); @@ -380,7 +409,7 @@ vm.runtime.requestRedraw(); } radToDeg(rad) { - return rad * 180 / Math.PI; + return (rad * 180) / Math.PI; } } diff --git a/extensions/JeremyGamer13/tween.js b/extensions/JeremyGamer13/tween.js index 5b6fe513e4..e19e1b5d94 100644 --- a/extensions/JeremyGamer13/tween.js +++ b/extensions/JeremyGamer13/tween.js @@ -1,304 +1,313 @@ -(function (Scratch) { - 'use strict'; - - const EasingMethods = [ - "linear", - "sine", - "quad", - "cubic", - "quart", - "quint", - "expo", - "circ", - "back", - "elastic", - "bounce" - ]; - - const BlockType = Scratch.BlockType; - const ArgumentType = Scratch.ArgumentType; - const Cast = Scratch.Cast; - - class Tween { - getInfo() { - return { - id: 'jeremygamerTweening', - name: 'Tweening', - blocks: [ - { - opcode: 'tweenValue', - text: '[MODE] ease [DIRECTION] [START] to [END] by [AMOUNT]%', - disableMonitor: true, - blockType: BlockType.REPORTER, - arguments: { - MODE: { type: ArgumentType.STRING, menu: 'modes' }, - DIRECTION: { type: ArgumentType.STRING, menu: 'direction' }, - START: { type: ArgumentType.NUMBER, defaultValue: 0 }, - END: { type: ArgumentType.NUMBER, defaultValue: 100 }, - AMOUNT: { type: ArgumentType.NUMBER, defaultValue: 50 }, - } - } - ], - menus: { - modes: { - acceptReporters: true, - items: EasingMethods.map(item => ({ text: item, value: item })) - }, - direction: { - acceptReporters: true, - items: [ - "in", - "out", - "in out" - ].map(item => ({ text: item, value: item })) - } - } - }; - } - - // utilities - multiplierToNormalNumber(mul, start, end) { - const multiplier = end - start; - const result = (mul * multiplier) + start; - return result; - } - - // blocks - tweenValue(args) { - const easeMethod = Cast.toString(args.MODE); - const easeDirection = Cast.toString(args.DIRECTION); - - const start = Cast.toNumber(args.START); - const end = Cast.toNumber(args.END); - - // easing method does not exist, return starting number - if (!EasingMethods.includes(easeMethod)) return start; - // easing method is not implemented, return starting number - if (!this[easeMethod]) return start; - - const progress = Cast.toNumber(args.AMOUNT) / 100; - - const tweened = this[easeMethod](progress, easeDirection); - return this.multiplierToNormalNumber(tweened, start, end); - } - - // easing functions (placed below blocks for organization) - linear(x) { - // lol - return x; - } - - sine(x, dir) { - switch (dir) { - case "in": { - return 1 - Math.cos((x * Math.PI) / 2); - } - case "out": { - return Math.sin((x * Math.PI) / 2); - } - case "in out": { - return -(Math.cos(Math.PI * x) - 1) / 2; - } - default: - return 0; - } - } - - quad(x, dir) { - switch (dir) { - case "in": { - return x * x; - } - case "out": { - return 1 - (1 - x) * (1 - x); - } - case "in out": { - return x < 0.5 ? 2 * x * x : 1 - Math.pow(-2 * x + 2, 2) / 2; - } - default: - return 0; - } - } - - cubic(x, dir) { - switch (dir) { - case "in": { - return x * x * x; - } - case "out": { - return 1 - Math.pow(1 - x, 3); - } - case "in out": { - return x < 0.5 ? 4 * x * x * x : 1 - Math.pow(-2 * x + 2, 3) / 2; - } - default: - return 0; - } - } - - quart(x, dir) { - switch (dir) { - case "in": { - return x * x * x * x; - } - case "out": { - return 1 - Math.pow(1 - x, 4); - } - case "in out": { - return x < 0.5 ? 8 * x * x * x * x : 1 - Math.pow(-2 * x + 2, 4) / 2; - } - default: - return 0; - } - } - - quint(x, dir) { - switch (dir) { - case "in": { - return x * x * x * x * x; - } - case "out": { - return 1 - Math.pow(1 - x, 5); - } - case "in out": { - return x < 0.5 ? 16 * x * x * x * x * x : 1 - Math.pow(-2 * x + 2, 5) / 2; - } - default: - return 0; - } - } - - expo(x, dir) { - switch (dir) { - case "in": { - return x === 0 ? 0 : Math.pow(2, 10 * x - 10); - } - case "out": { - return x === 1 ? 1 : 1 - Math.pow(2, -10 * x); - } - case "in out": { - return x === 0 - ? 0 - : x === 1 - ? 1 - : x < 0.5 ? Math.pow(2, 20 * x - 10) / 2 - : (2 - Math.pow(2, -20 * x + 10)) / 2; - } - default: - return 0; - } - } - - circ(x, dir) { - switch (dir) { - case "in": { - return 1 - Math.sqrt(1 - Math.pow(x, 2)); - } - case "out": { - return Math.sqrt(1 - Math.pow(x - 1, 2)); - } - case "in out": { - return x < 0.5 - ? (1 - Math.sqrt(1 - Math.pow(2 * x, 2))) / 2 - : (Math.sqrt(1 - Math.pow(-2 * x + 2, 2)) + 1) / 2; - } - default: - return 0; - } - } - - back(x, dir) { - switch (dir) { - case "in": { - const c1 = 1.70158; - const c3 = c1 + 1; - - return c3 * x * x * x - c1 * x * x; - } - case "out": { - const c1 = 1.70158; - const c3 = c1 + 1; - - return 1 + c3 * Math.pow(x - 1, 3) + c1 * Math.pow(x - 1, 2); - } - case "in out": { - const c1 = 1.70158; - const c2 = c1 * 1.525; - - return x < 0.5 - ? (Math.pow(2 * x, 2) * ((c2 + 1) * 2 * x - c2)) / 2 - : (Math.pow(2 * x - 2, 2) * ((c2 + 1) * (x * 2 - 2) + c2) + 2) / 2; - } - default: - return 0; - } - } - - elastic(x, dir) { - switch (dir) { - case "in": { - const c4 = (2 * Math.PI) / 3; - - return x === 0 - ? 0 - : x === 1 - ? 1 - : -Math.pow(2, 10 * x - 10) * Math.sin((x * 10 - 10.75) * c4); - } - case "out": { - const c4 = (2 * Math.PI) / 3; - - return x === 0 - ? 0 - : x === 1 - ? 1 - : Math.pow(2, -10 * x) * Math.sin((x * 10 - 0.75) * c4) + 1; - } - case "in out": { - const c5 = (2 * Math.PI) / 4.5; - - return x === 0 - ? 0 - : x === 1 - ? 1 - : x < 0.5 - ? -(Math.pow(2, 20 * x - 10) * Math.sin((20 * x - 11.125) * c5)) / 2 - : (Math.pow(2, -20 * x + 10) * Math.sin((20 * x - 11.125) * c5)) / 2 + 1; - } - default: - return 0; - } - } - - bounce(x, dir) { - switch (dir) { - case "in": { - return 1 - this.bounce(1 - x, "out"); - } - case "out": { - const n1 = 7.5625; - const d1 = 2.75; - - if (x < 1 / d1) { - return n1 * x * x; - } else if (x < 2 / d1) { - return n1 * (x -= 1.5 / d1) * x + 0.75; - } else if (x < 2.5 / d1) { - return n1 * (x -= 2.25 / d1) * x + 0.9375; - } else { - return n1 * (x -= 2.625 / d1) * x + 0.984375; - } - } - case "in out": { - return x < 0.5 - ? (1 - this.bounce(1 - 2 * x, "out")) / 2 - : (1 + this.bounce(2 * x - 1, "out")) / 2; - } - default: - return 0; - } - } - } - - Scratch.extensions.register(new Tween()); -})(Scratch); +// Name: Tween +// ID: jeremygamerTweening +// Description: Easing methods for smooth animations. +// By: JeremyGamer13 + +(function (Scratch) { + "use strict"; + + const EasingMethods = [ + "linear", + "sine", + "quad", + "cubic", + "quart", + "quint", + "expo", + "circ", + "back", + "elastic", + "bounce", + ]; + + const BlockType = Scratch.BlockType; + const ArgumentType = Scratch.ArgumentType; + const Cast = Scratch.Cast; + + class Tween { + getInfo() { + return { + id: "jeremygamerTweening", + name: "Tweening", + blocks: [ + { + opcode: "tweenValue", + text: "[MODE] ease [DIRECTION] [START] to [END] by [AMOUNT]%", + disableMonitor: true, + blockType: BlockType.REPORTER, + arguments: { + MODE: { type: ArgumentType.STRING, menu: "modes" }, + DIRECTION: { type: ArgumentType.STRING, menu: "direction" }, + START: { type: ArgumentType.NUMBER, defaultValue: 0 }, + END: { type: ArgumentType.NUMBER, defaultValue: 100 }, + AMOUNT: { type: ArgumentType.NUMBER, defaultValue: 50 }, + }, + }, + ], + menus: { + modes: { + acceptReporters: true, + items: EasingMethods.map((item) => ({ text: item, value: item })), + }, + direction: { + acceptReporters: true, + items: ["in", "out", "in out"].map((item) => ({ + text: item, + value: item, + })), + }, + }, + }; + } + + // utilities + multiplierToNormalNumber(mul, start, end) { + const multiplier = end - start; + const result = mul * multiplier + start; + return result; + } + + // blocks + tweenValue(args) { + const easeMethod = Cast.toString(args.MODE); + const easeDirection = Cast.toString(args.DIRECTION); + + const start = Cast.toNumber(args.START); + const end = Cast.toNumber(args.END); + + // easing method does not exist, return starting number + if (!EasingMethods.includes(easeMethod)) return start; + // easing method is not implemented, return starting number + if (!this[easeMethod]) return start; + + const progress = Cast.toNumber(args.AMOUNT) / 100; + + const tweened = this[easeMethod](progress, easeDirection); + return this.multiplierToNormalNumber(tweened, start, end); + } + + // easing functions (placed below blocks for organization) + linear(x) { + // lol + return x; + } + + sine(x, dir) { + switch (dir) { + case "in": { + return 1 - Math.cos((x * Math.PI) / 2); + } + case "out": { + return Math.sin((x * Math.PI) / 2); + } + case "in out": { + return -(Math.cos(Math.PI * x) - 1) / 2; + } + default: + return 0; + } + } + + quad(x, dir) { + switch (dir) { + case "in": { + return x * x; + } + case "out": { + return 1 - (1 - x) * (1 - x); + } + case "in out": { + return x < 0.5 ? 2 * x * x : 1 - Math.pow(-2 * x + 2, 2) / 2; + } + default: + return 0; + } + } + + cubic(x, dir) { + switch (dir) { + case "in": { + return x * x * x; + } + case "out": { + return 1 - Math.pow(1 - x, 3); + } + case "in out": { + return x < 0.5 ? 4 * x * x * x : 1 - Math.pow(-2 * x + 2, 3) / 2; + } + default: + return 0; + } + } + + quart(x, dir) { + switch (dir) { + case "in": { + return x * x * x * x; + } + case "out": { + return 1 - Math.pow(1 - x, 4); + } + case "in out": { + return x < 0.5 ? 8 * x * x * x * x : 1 - Math.pow(-2 * x + 2, 4) / 2; + } + default: + return 0; + } + } + + quint(x, dir) { + switch (dir) { + case "in": { + return x * x * x * x * x; + } + case "out": { + return 1 - Math.pow(1 - x, 5); + } + case "in out": { + return x < 0.5 + ? 16 * x * x * x * x * x + : 1 - Math.pow(-2 * x + 2, 5) / 2; + } + default: + return 0; + } + } + + expo(x, dir) { + switch (dir) { + case "in": { + return x === 0 ? 0 : Math.pow(2, 10 * x - 10); + } + case "out": { + return x === 1 ? 1 : 1 - Math.pow(2, -10 * x); + } + case "in out": { + return x === 0 + ? 0 + : x === 1 + ? 1 + : x < 0.5 + ? Math.pow(2, 20 * x - 10) / 2 + : (2 - Math.pow(2, -20 * x + 10)) / 2; + } + default: + return 0; + } + } + + circ(x, dir) { + switch (dir) { + case "in": { + return 1 - Math.sqrt(1 - Math.pow(x, 2)); + } + case "out": { + return Math.sqrt(1 - Math.pow(x - 1, 2)); + } + case "in out": { + return x < 0.5 + ? (1 - Math.sqrt(1 - Math.pow(2 * x, 2))) / 2 + : (Math.sqrt(1 - Math.pow(-2 * x + 2, 2)) + 1) / 2; + } + default: + return 0; + } + } + + back(x, dir) { + switch (dir) { + case "in": { + const c1 = 1.70158; + const c3 = c1 + 1; + + return c3 * x * x * x - c1 * x * x; + } + case "out": { + const c1 = 1.70158; + const c3 = c1 + 1; + + return 1 + c3 * Math.pow(x - 1, 3) + c1 * Math.pow(x - 1, 2); + } + case "in out": { + const c1 = 1.70158; + const c2 = c1 * 1.525; + + return x < 0.5 + ? (Math.pow(2 * x, 2) * ((c2 + 1) * 2 * x - c2)) / 2 + : (Math.pow(2 * x - 2, 2) * ((c2 + 1) * (x * 2 - 2) + c2) + 2) / 2; + } + default: + return 0; + } + } + + elastic(x, dir) { + switch (dir) { + case "in": { + const c4 = (2 * Math.PI) / 3; + + return x === 0 + ? 0 + : x === 1 + ? 1 + : -Math.pow(2, 10 * x - 10) * Math.sin((x * 10 - 10.75) * c4); + } + case "out": { + const c4 = (2 * Math.PI) / 3; + + return x === 0 + ? 0 + : x === 1 + ? 1 + : Math.pow(2, -10 * x) * Math.sin((x * 10 - 0.75) * c4) + 1; + } + case "in out": { + const c5 = (2 * Math.PI) / 4.5; + + return x === 0 + ? 0 + : x === 1 + ? 1 + : x < 0.5 + ? -(Math.pow(2, 20 * x - 10) * Math.sin((20 * x - 11.125) * c5)) / 2 + : (Math.pow(2, -20 * x + 10) * Math.sin((20 * x - 11.125) * c5)) / + 2 + + 1; + } + default: + return 0; + } + } + + bounce(x, dir) { + switch (dir) { + case "in": { + return 1 - this.bounce(1 - x, "out"); + } + case "out": { + const n1 = 7.5625; + const d1 = 2.75; + + if (x < 1 / d1) { + return n1 * x * x; + } else if (x < 2 / d1) { + return n1 * (x -= 1.5 / d1) * x + 0.75; + } else if (x < 2.5 / d1) { + return n1 * (x -= 2.25 / d1) * x + 0.9375; + } else { + return n1 * (x -= 2.625 / d1) * x + 0.984375; + } + } + case "in out": { + return x < 0.5 + ? (1 - this.bounce(1 - 2 * x, "out")) / 2 + : (1 + this.bounce(2 * x - 1, "out")) / 2; + } + default: + return 0; + } + } + } + + Scratch.extensions.register(new Tween()); +})(Scratch); diff --git a/extensions/Lily/AllMenus.js b/extensions/Lily/AllMenus.js new file mode 100644 index 0000000000..827b03aed3 --- /dev/null +++ b/extensions/Lily/AllMenus.js @@ -0,0 +1,56 @@ +// Name: All Menus +// ID: lmsAllMenus +// Description: Special category with every menu from every Scratch category and extensions. +// By: LilyMakesThings + +(function (Scratch) { + "use strict"; + + var blockXML; + const blacklist = ["looks_costumenumbername", "extension_wedo_tilt_menu"]; + + Scratch.vm.addListener("BLOCKSINFO_UPDATE", refreshMenus); + + function refreshMenus() { + if (!window.ScratchBlocks) return; + Scratch.vm.removeListener("BLOCKSINFO_UPDATE", refreshMenus); + + let allBlocks = Object.keys(ScratchBlocks.Blocks); + + allBlocks = allBlocks.filter( + (item) => item.includes("menu") && !blacklist.includes(item) + ); + + const menuBlocks = allBlocks.map( + (item) => '' + ); + + blockXML = menuBlocks.join(""); + Scratch.vm.runtime.extensionManager.refreshBlocks(); + } + + class AllMenus { + constructor() { + Scratch.vm.runtime.on("EXTENSION_ADDED", () => { + refreshMenus(); + }); + } + + getInfo() { + return { + id: "lmsAllMenus", + name: "All Menus", + blocks: [ + { + blockType: Scratch.BlockType.XML, + xml: blockXML, + }, + ], + }; + } + } + + refreshMenus(); + + Scratch.extensions.register(new AllMenus()); +})(Scratch); diff --git a/extensions/Lily/Cast.js b/extensions/Lily/Cast.js index 5035acae5a..3baf507a99 100644 --- a/extensions/Lily/Cast.js +++ b/extensions/Lily/Cast.js @@ -1,73 +1,86 @@ -(function (Scratch) { - 'use strict'; - - const Cast = Scratch.Cast; - - class CastUtil { - getInfo() { - return { - id: 'lmsCast', - name: 'Cast', - blocks: [ - { - opcode: 'toType', - blockType: Scratch.BlockType.REPORTER, - text: 'cast [INPUT] to [TYPE]', - allowDropAnywhere: true, - disableMonitor: true, - arguments: { - INPUT: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'apple' - }, - TYPE: { - type: Scratch.ArgumentType.STRING, - menu: 'type' - } - } - }, - { - opcode: 'typeOf', - blockType: Scratch.BlockType.REPORTER, - text: 'type of [INPUT]', - disableMonitor: true, - arguments: { - INPUT: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'apple' - } - } - } - ], - menus: { - type: { - acceptReporters: true, - items: ['number', 'string', 'boolean', 'default'] - } - } - }; - } - - toType(args) { - const input = args.INPUT; - switch (args.TYPE) { - case ('number'): return Cast.toNumber(input); - case ('string'): return Cast.toString(input); - case ('boolean'): return Cast.toBoolean(input); - default: return input; - } - } - - typeOf(args) { - const input = args.INPUT; - switch (typeof input) { - case ('number'): return 'number'; - case ('string'): return 'string'; - case ('boolean'): return 'boolean'; - default: return ''; - } - } - } - - Scratch.extensions.register(new CastUtil()); -})(Scratch); +// Name: Cast +// ID: lmsCast +// Description: Convert values between types. +// By: LilyMakesThings + +(function (Scratch) { + "use strict"; + + const Cast = Scratch.Cast; + + class CastUtil { + getInfo() { + return { + id: "lmsCast", + name: "Cast", + blocks: [ + { + opcode: "toType", + blockType: Scratch.BlockType.REPORTER, + text: "cast [INPUT] to [TYPE]", + allowDropAnywhere: true, + disableMonitor: true, + arguments: { + INPUT: { + type: Scratch.ArgumentType.STRING, + defaultValue: "apple", + }, + TYPE: { + type: Scratch.ArgumentType.STRING, + menu: "type", + }, + }, + }, + { + opcode: "typeOf", + blockType: Scratch.BlockType.REPORTER, + text: "type of [INPUT]", + disableMonitor: true, + arguments: { + INPUT: { + type: Scratch.ArgumentType.STRING, + defaultValue: "apple", + }, + }, + }, + ], + menus: { + type: { + acceptReporters: true, + items: ["number", "string", "boolean", "default"], + }, + }, + }; + } + + toType(args) { + const input = args.INPUT; + switch (args.TYPE) { + case "number": + return Cast.toNumber(input); + case "string": + return Cast.toString(input); + case "boolean": + return Cast.toBoolean(input); + default: + return input; + } + } + + typeOf(args) { + const input = args.INPUT; + switch (typeof input) { + case "number": + return "number"; + case "string": + return "string"; + case "boolean": + return "boolean"; + default: + return ""; + } + } + } + + Scratch.extensions.register(new CastUtil()); +})(Scratch); diff --git a/extensions/Lily/ClonesPlus.js b/extensions/Lily/ClonesPlus.js index f98fb470bf..009e997692 100644 --- a/extensions/Lily/ClonesPlus.js +++ b/extensions/Lily/ClonesPlus.js @@ -1,611 +1,656 @@ -(function (Scratch) { - 'use strict'; - - const menuIconURI = ""; - - /** - * @param {VM.Target|null} target - * @param {string|unknown} thing - * @returns {string|number|boolean} - */ - const getThingOfTarget = (target, thing) => { - if (!target) { - return ''; - } - if (thing === 'x position') { - return target.x; - } - if (thing === 'y position') { - return target.y; - } - if (thing === 'direction') { - return target.direction; - } - if (thing === 'costume num') { - return (target.currentCostume + 1); - } - if (thing === 'costume name') { - return target.getCostumes()[target.currentCostume].name; - } - if (thing === 'size') { - return target.size; - } - if (thing === 'volume') { - return target.volume; - } - // this should never happen - return ''; - }; - - class ClonesPlus { - getInfo() { - return { - id: 'lmsclonesplus', - name: 'Clones+', - color1: '#FFAB19', - color2: '#EC9C13', - color3: '#CF8B17', - menuIconURI: menuIconURI, - blocks: [ - { - opcode: 'whenCloneStartsWithVar', - blockType: Scratch.BlockType.HAT, - text: 'when I start as a clone with [INPUTA] set to [INPUTB]', - filter: [Scratch.TargetType.SPRITE], - arguments: { - INPUTA: { - type: Scratch.ArgumentType.STRING, - menu: 'variablesMenu' - }, - INPUTB: { - type: Scratch.ArgumentType.STRING, - defaultValue: '1' - } - } - }, - { - opcode: 'createCloneWithVar', - blockType: Scratch.BlockType.COMMAND, - text: 'create clone with [INPUTA] set to [INPUTB]', - filter: [Scratch.TargetType.SPRITE], - arguments: { - INPUTA: { - type: Scratch.ArgumentType.STRING, - menu: 'variablesMenu' - }, - INPUTB: { - type: Scratch.ArgumentType.STRING, - defaultValue: '1' - } - } - }, - - '---', - - { - opcode: 'touchingCloneWithVar', - blockType: Scratch.BlockType.BOOLEAN, - text: 'touching clone with [INPUTA] set to [INPUTB]?', - filter: [Scratch.TargetType.SPRITE], - arguments: { - INPUTA: { - type: Scratch.ArgumentType.STRING, - menu: 'variablesMenu' - }, - INPUTB: { - type: Scratch.ArgumentType.STRING, - defaultValue: '1' - } - } - }, - { - opcode: 'touchingMainSprite', - blockType: Scratch.BlockType.BOOLEAN, - text: 'touching main sprite?', - filter: [Scratch.TargetType.SPRITE] - }, - - '---', - - { - opcode: 'setVariableOfClone', - blockType: Scratch.BlockType.COMMAND, - text: 'set variable [INPUTA] to [INPUTB] for clones with [INPUTC] set to [INPUTD]', - filter: [Scratch.TargetType.SPRITE], - arguments: { - INPUTA: { - type: Scratch.ArgumentType.STRING, - menu: 'variablesMenu' - }, - INPUTB: { - type: Scratch.ArgumentType.STRING, - defaultValue: '0' - }, - INPUTC: { - type: Scratch.ArgumentType.STRING, - menu: 'variablesMenu' - }, - INPUTD: { - type: Scratch.ArgumentType.STRING, - defaultValue: '1' - } - } - }, - { - opcode: 'getVariableOfClone', - blockType: Scratch.BlockType.REPORTER, - text: 'variable [INPUTA] of clone with [INPUTB] set to [INPUTC]', - filter: [Scratch.TargetType.SPRITE], - disableMonitor: true, - arguments: { - INPUTA: { - type: Scratch.ArgumentType.STRING, - menu: 'variablesMenu' - }, - INPUTB: { - type: Scratch.ArgumentType.STRING, - menu: 'variablesMenu' - }, - INPUTC: { - type: Scratch.ArgumentType.STRING, - defaultValue: '1' - } - } - }, - { - opcode: 'setVariableOfMainSprite', - blockType: Scratch.BlockType.COMMAND, - text: 'set variable [INPUTA] to [INPUTB] for main sprite', - filter: [Scratch.TargetType.SPRITE], - arguments: { - INPUTA: { - type: Scratch.ArgumentType.STRING, - menu: 'variablesMenu' - }, - INPUTB: { - type: Scratch.ArgumentType.STRING, - defaultValue: '1' - } - } - }, - { - opcode: 'getVariableOfMainSprite', - blockType: Scratch.BlockType.REPORTER, - text: 'variable [INPUT] of main sprite', - filter: [Scratch.TargetType.SPRITE], - disableMonitor: true, - arguments: { - INPUT: { - type: Scratch.ArgumentType.STRING, - menu: 'variablesMenu' - } - } - }, - - '---', - - { - opcode: 'cloneExists', - blockType: Scratch.BlockType.BOOLEAN, - text: 'clone with [INPUTA] set to [INPUTB] exists?', - filter: [Scratch.TargetType.SPRITE], - arguments: { - INPUTA: { - type: Scratch.ArgumentType.STRING, - menu: 'variablesMenu' - }, - INPUTB: { - type: Scratch.ArgumentType.STRING, - defaultValue: '1' - } - } - }, - { - opcode: 'getThingOfClone', - blockType: Scratch.BlockType.REPORTER, - text: '[INPUTA] of clone with [INPUTB] set to [INPUTC]', - filter: [Scratch.TargetType.SPRITE], - disableMonitor: true, - arguments: { - INPUTA: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'x position', - menu: 'thingOfMenu' - }, - INPUTB: { - type: Scratch.ArgumentType.STRING, - menu: 'variablesMenu' - }, - INPUTC: { - type: Scratch.ArgumentType.STRING, - defaultValue: '1' - } - } - }, - { - opcode: 'getThingOfMainSprite', - blockType: Scratch.BlockType.REPORTER, - text: '[INPUT] of main sprite', - filter: [Scratch.TargetType.SPRITE], - disableMonitor: true, - arguments: { - INPUT: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'x position', - menu: 'thingOfMenu' - } - } - }, - - '---', - - { - opcode: 'stopScriptsInSprite', - blockType: Scratch.BlockType.COMMAND, - text: 'stop scripts in [INPUT]', - arguments: { - INPUT: { - type: Scratch.ArgumentType.STRING, - menu: 'spriteMenu' - } - } - }, - { - opcode: 'stopScriptsInClone', - blockType: Scratch.BlockType.COMMAND, - text: 'stop scripts in clones with [INPUTA] set to [INPUTB]', - filter: [Scratch.TargetType.SPRITE], - arguments: { - INPUTA: { - type: Scratch.ArgumentType.STRING, - menu: 'variablesMenu' - }, - INPUTB: { - type: Scratch.ArgumentType.STRING, - defaultValue: '1' - } - } - }, - { - opcode: 'stopScriptsInMainSprite', - blockType: Scratch.BlockType.COMMAND, - text: 'stop scripts in main sprite', - filter: [Scratch.TargetType.SPRITE] - }, - - '---', - - { - opcode: 'deleteClonesInSprite', - blockType: Scratch.BlockType.COMMAND, - text: 'delete clones in [INPUT]', - arguments: { - INPUT: { - type: Scratch.ArgumentType.STRING, - menu: 'spriteMenu' - } - } - }, - { - opcode: 'deleteCloneWithVar', - blockType: Scratch.BlockType.COMMAND, - text: 'delete clones with [INPUTA] set to [INPUTB]', - filter: [Scratch.TargetType.SPRITE], - arguments: { - INPUTA: { - type: Scratch.ArgumentType.STRING, - menu: 'variablesMenu' - }, - INPUTB: { - type: Scratch.ArgumentType.STRING, - defaultValue: '1' - } - } - }, - - '---', - - { - opcode: 'isClone', - blockType: Scratch.BlockType.BOOLEAN, - text: 'is clone?', - filter: [Scratch.TargetType.SPRITE] - }, - - '---', - - { - opcode: 'cloneCount', - blockType: Scratch.BlockType.REPORTER, - text: 'clone count' - }, - { - opcode: 'spriteCloneCount', - blockType: Scratch.BlockType.REPORTER, - text: 'clone count of [INPUT]', - disableMonitor: true, - arguments: { - INPUT: { - type: Scratch.ArgumentType.STRING, - menu: 'spriteMenu' - } - } - } - ], - menus: { - spriteMenu: { - acceptReporters: true, - items: 'getSprites' - }, - // menus use acceptReporters: false for Scratch parity - variablesMenu: { - acceptReporters: false, - items: 'getVariables' - }, - thingOfMenu: { - acceptReporters: false, - items: [ - { - text: 'x position', - value: 'x position' - }, - { - text: 'y position', - value: 'y position' - }, - { - text: 'direction', - value: 'direction' - }, - { - text: 'costume #', - value: 'costume num' - }, - { - text: 'costume name', - value: 'costume name' - }, - { - text: 'size', - value: 'size' - }, - { - text: 'volume', - value: 'volume' - }, - ] - } - } - }; - } - - whenCloneStartsWithVar(args, util) { - // TODO: this is really not ideal. this should be an event-based hat ideally, but we don't have a good - // way to do that right now... - if (util.target.isOriginal) { - return false; - } - const variable = util.target.lookupVariableById(args.INPUTA); - const expectedValue = args.INPUTB; - if (variable) { - return Scratch.Cast.compare(variable.value, expectedValue) === 0; - } - return false; - } - - createCloneWithVar(args, util) { - // @ts-expect-error - not typed yet - Scratch.vm.runtime.ext_scratch3_control._createClone(util.target.sprite.name, util.target); - const clones = util.target.sprite.clones; - const cloneNum = clones.length - 1; - const cloneVariable = clones[cloneNum].lookupVariableById(args.INPUTA); - if (cloneVariable) { - cloneVariable.value = args.INPUTB; - } - } - - touchingCloneWithVar(args, util) { - const drawableCandidates = util.target.sprite.clones - .filter(clone => { - const variable = clone.lookupVariableById(args.INPUTA); - return variable && Scratch.Cast.compare(variable.value, args.INPUTB) === 0; - }) - .map(clone => clone.drawableID); - if (drawableCandidates.length === 0) { - return false; - } - return Scratch.vm.renderer.isTouchingDrawables(util.target.drawableID, drawableCandidates); - } - - touchingMainSprite(args, util) { - if (util.target.isOriginal) { - return false; - } - const main = util.target.sprite.clones[0]; - const drawableCandidates = [main.drawableID]; - return Scratch.vm.renderer.isTouchingDrawables(util.target.drawableID, drawableCandidates); - } - - setVariableOfClone(args, util) { - const newVariableValue = args.INPUTB; - const expectedVarValue = args.INPUTD; - const clones = util.target.sprite.clones; - for (let index = 1; index < clones.length; index++) { - const checkVar = clones[index].lookupVariableById(args.INPUTC); - if (checkVar && Scratch.Cast.compare(checkVar.value, expectedVarValue) === 0) { - const editVar = clones[index].lookupVariableById(args.INPUTA); - if (editVar) { - editVar.value = newVariableValue; - } - } - } - } - - getVariableOfClone(args, util) { - const clone = this.getCloneFromVariable(args.INPUTB, args.INPUTC, util.target.sprite.clones); - if (!clone) { - return ''; - } - // guaranteed to exist by getCloneFromVariable - const cloneVar = clone.lookupVariableById(args.INPUTA); - return cloneVar.value; - } - - setVariableOfMainSprite(args, util) { - const main = util.target.sprite.clones[0]; - const variableObj = main.lookupVariableById(args.INPUTA); - if (variableObj) { - variableObj.value = args.INPUTB; - } - } - - getVariableOfMainSprite(args, util) { - const main = util.target.sprite.clones[0]; - const variableObj = main.lookupVariableById(args.INPUT); - if (variableObj) { - return variableObj.value; - } - return ''; - } - - cloneExists(args, util) { - const clone = this.getCloneFromVariable(args.INPUTA, args.INPUTB, util.target.sprite.clones); - return !!clone; - } - - getThingOfClone(args, util) { - const clone = this.getCloneFromVariable(args.INPUTB, args.INPUTC, util.target.sprite.clones); - return getThingOfTarget(clone, args.INPUTA); - } - - getThingOfMainSprite(args, util) { - const main = util.target.sprite.clones[0]; - return getThingOfTarget(main, args.INPUT); - } - - stopScriptsInSprite(args) { - const targetObj = Scratch.vm.runtime.getSpriteTargetByName(args.INPUT); - if (targetObj) { - Scratch.vm.runtime.stopForTarget(targetObj); - } - } - - stopScriptsInMainSprite(args, util) { - Scratch.vm.runtime.stopForTarget(util.target.sprite.clones[0]); - } - - stopScriptsInClone(args, util) { - const clones = util.target.sprite.clones; - let expectedValue = args.INPUTB; - for (let index = 1; index < clones.length; index++) { - const cloneVariable = clones[index].lookupVariableById(args.INPUTA); - if (cloneVariable && Scratch.Cast.compare(cloneVariable.value, expectedValue) === 0) { - Scratch.vm.runtime.stopForTarget(clones[index]); - } - } - } - - deleteClonesInSprite(args, util) { - const target = Scratch.vm.runtime.getSpriteTargetByName(args.INPUT); - if (!target) { - return; - } - const clones = target.sprite.clones; - for (let index = clones.length - 1; index > 0; index--) { - Scratch.vm.runtime.disposeTarget(clones[index]); - } - } - - deleteCloneWithVar(args, util) { - const clones = util.target.sprite.clones; - const expectedValue = args.INPUTB; - for (let index = clones.length - 1; index > 0; index--) { - const cloneVar = clones[index].lookupVariableById(args.INPUTA); - if (cloneVar && Scratch.Cast.compare(cloneVar.value, expectedValue) === 0) { - Scratch.vm.runtime.disposeTarget(clones[index]); - } - } - } - - isClone(args, util) { - return !util.target.isOriginal; - } - - cloneCount(args, util) { - return Scratch.vm.runtime._cloneCounter; - } - - spriteCloneCount(args, util) { - const target = Scratch.vm.runtime.getSpriteTargetByName(args.INPUT); - if (target) { - return (target.sprite.clones.length - 1); - } - return 0; - } - - /** - * @param {string} variableId - * @param {unknown} expectedValue - * @param {VM.Target[]} clones - * @returns {VM.Target|null} - */ - getCloneFromVariable(variableId, expectedValue, clones) { - for (let index = 1; index < clones.length; index++) { - const cloneVar = clones[index].lookupVariableById(variableId); - if (cloneVar && Scratch.Cast.compare(cloneVar.value, expectedValue) === 0) { - return clones[index]; - } - } - return null; - } - - getSprites() { - let spriteNames = []; - const targets = Scratch.vm.runtime.targets; - const myself = Scratch.vm.runtime.getEditingTarget().sprite.name; - for (let index = 1; index < targets.length; index++) { - const curTarget = targets[index].sprite; - let display = curTarget.name; - if (myself === curTarget.name) { - display = 'myself'; - } - if (targets[index].isOriginal) { - const jsonOBJ = { - text: display, - value: curTarget.name - }; - spriteNames.push(jsonOBJ); - } - } - if (spriteNames.length > 0) { - return spriteNames; - } else { - return [{ text: "", value: 0 }]; //this should never happen but it's a failsafe - } - } - - getSpriteObj(name) { //This is unused but I'm leaving it in for potential future blocks - const spriteObj = Scratch.vm.runtime.getSpriteTargetByName(name); - return JSON.stringify(spriteObj); - } - - getVariables() { - // @ts-expect-error - Blockly not typed yet - // eslint-disable-next-line no-undef - const variables = typeof Blockly === 'undefined' ? [] : Blockly.getMainWorkspace() - .getVariableMap() - .getVariablesOfType('') - .filter(model => model.isLocal) - .map(model => ({ - text: model.name, - value: model.getId() - })); - if (variables.length > 0) { - return variables; - } else { - return [{ text: "", value: "" }]; - } - } - } - Scratch.extensions.register(new ClonesPlus()); -})(Scratch); +// Name: Clones Plus +// ID: lmsclonesplus +// Description: Expansion of Scratch's clone features. +// By: LilyMakesThings + +(function (Scratch) { + "use strict"; + + const menuIconURI = + ""; + + /** + * @param {VM.Target|null} target + * @param {string|unknown} thing + * @returns {string|number|boolean} + */ + const getThingOfTarget = (target, thing) => { + if (!target) { + return ""; + } + if (thing === "x position") { + return target.x; + } + if (thing === "y position") { + return target.y; + } + if (thing === "direction") { + return target.direction; + } + if (thing === "costume num") { + return target.currentCostume + 1; + } + if (thing === "costume name") { + return target.getCostumes()[target.currentCostume].name; + } + if (thing === "size") { + return target.size; + } + if (thing === "volume") { + return target.volume; + } + // this should never happen + return ""; + }; + + class ClonesPlus { + getInfo() { + return { + id: "lmsclonesplus", + name: "Clones+", + color1: "#FFAB19", + color2: "#EC9C13", + color3: "#CF8B17", + menuIconURI: menuIconURI, + blocks: [ + { + opcode: "whenCloneStartsWithVar", + blockType: Scratch.BlockType.HAT, + text: "when I start as a clone with [INPUTA] set to [INPUTB]", + filter: [Scratch.TargetType.SPRITE], + arguments: { + INPUTA: { + type: Scratch.ArgumentType.STRING, + menu: "variablesMenu", + }, + INPUTB: { + type: Scratch.ArgumentType.STRING, + defaultValue: "1", + }, + }, + }, + { + opcode: "createCloneWithVar", + blockType: Scratch.BlockType.COMMAND, + text: "create clone with [INPUTA] set to [INPUTB]", + filter: [Scratch.TargetType.SPRITE], + arguments: { + INPUTA: { + type: Scratch.ArgumentType.STRING, + menu: "variablesMenu", + }, + INPUTB: { + type: Scratch.ArgumentType.STRING, + defaultValue: "1", + }, + }, + }, + + "---", + + { + opcode: "touchingCloneWithVar", + blockType: Scratch.BlockType.BOOLEAN, + text: "touching clone with [INPUTA] set to [INPUTB]?", + filter: [Scratch.TargetType.SPRITE], + arguments: { + INPUTA: { + type: Scratch.ArgumentType.STRING, + menu: "variablesMenu", + }, + INPUTB: { + type: Scratch.ArgumentType.STRING, + defaultValue: "1", + }, + }, + }, + { + opcode: "touchingMainSprite", + blockType: Scratch.BlockType.BOOLEAN, + text: "touching main sprite?", + filter: [Scratch.TargetType.SPRITE], + }, + + "---", + + { + opcode: "setVariableOfClone", + blockType: Scratch.BlockType.COMMAND, + text: "set variable [INPUTA] to [INPUTB] for clones with [INPUTC] set to [INPUTD]", + filter: [Scratch.TargetType.SPRITE], + arguments: { + INPUTA: { + type: Scratch.ArgumentType.STRING, + menu: "variablesMenu", + }, + INPUTB: { + type: Scratch.ArgumentType.STRING, + defaultValue: "0", + }, + INPUTC: { + type: Scratch.ArgumentType.STRING, + menu: "variablesMenu", + }, + INPUTD: { + type: Scratch.ArgumentType.STRING, + defaultValue: "1", + }, + }, + }, + { + opcode: "getVariableOfClone", + blockType: Scratch.BlockType.REPORTER, + text: "variable [INPUTA] of clone with [INPUTB] set to [INPUTC]", + filter: [Scratch.TargetType.SPRITE], + disableMonitor: true, + arguments: { + INPUTA: { + type: Scratch.ArgumentType.STRING, + menu: "variablesMenu", + }, + INPUTB: { + type: Scratch.ArgumentType.STRING, + menu: "variablesMenu", + }, + INPUTC: { + type: Scratch.ArgumentType.STRING, + defaultValue: "1", + }, + }, + }, + { + opcode: "setVariableOfMainSprite", + blockType: Scratch.BlockType.COMMAND, + text: "set variable [INPUTA] to [INPUTB] for main sprite", + filter: [Scratch.TargetType.SPRITE], + arguments: { + INPUTA: { + type: Scratch.ArgumentType.STRING, + menu: "variablesMenu", + }, + INPUTB: { + type: Scratch.ArgumentType.STRING, + defaultValue: "1", + }, + }, + }, + { + opcode: "getVariableOfMainSprite", + blockType: Scratch.BlockType.REPORTER, + text: "variable [INPUT] of main sprite", + filter: [Scratch.TargetType.SPRITE], + disableMonitor: true, + arguments: { + INPUT: { + type: Scratch.ArgumentType.STRING, + menu: "variablesMenu", + }, + }, + }, + + "---", + + { + opcode: "cloneExists", + blockType: Scratch.BlockType.BOOLEAN, + text: "clone with [INPUTA] set to [INPUTB] exists?", + filter: [Scratch.TargetType.SPRITE], + arguments: { + INPUTA: { + type: Scratch.ArgumentType.STRING, + menu: "variablesMenu", + }, + INPUTB: { + type: Scratch.ArgumentType.STRING, + defaultValue: "1", + }, + }, + }, + { + opcode: "getThingOfClone", + blockType: Scratch.BlockType.REPORTER, + text: "[INPUTA] of clone with [INPUTB] set to [INPUTC]", + filter: [Scratch.TargetType.SPRITE], + disableMonitor: true, + arguments: { + INPUTA: { + type: Scratch.ArgumentType.STRING, + defaultValue: "x position", + menu: "thingOfMenu", + }, + INPUTB: { + type: Scratch.ArgumentType.STRING, + menu: "variablesMenu", + }, + INPUTC: { + type: Scratch.ArgumentType.STRING, + defaultValue: "1", + }, + }, + }, + { + opcode: "getThingOfMainSprite", + blockType: Scratch.BlockType.REPORTER, + text: "[INPUT] of main sprite", + filter: [Scratch.TargetType.SPRITE], + disableMonitor: true, + arguments: { + INPUT: { + type: Scratch.ArgumentType.STRING, + defaultValue: "x position", + menu: "thingOfMenu", + }, + }, + }, + + "---", + + { + opcode: "stopScriptsInSprite", + blockType: Scratch.BlockType.COMMAND, + text: "stop scripts in [INPUT]", + arguments: { + INPUT: { + type: Scratch.ArgumentType.STRING, + menu: "spriteMenu", + }, + }, + }, + { + opcode: "stopScriptsInClone", + blockType: Scratch.BlockType.COMMAND, + text: "stop scripts in clones with [INPUTA] set to [INPUTB]", + filter: [Scratch.TargetType.SPRITE], + arguments: { + INPUTA: { + type: Scratch.ArgumentType.STRING, + menu: "variablesMenu", + }, + INPUTB: { + type: Scratch.ArgumentType.STRING, + defaultValue: "1", + }, + }, + }, + { + opcode: "stopScriptsInMainSprite", + blockType: Scratch.BlockType.COMMAND, + text: "stop scripts in main sprite", + filter: [Scratch.TargetType.SPRITE], + }, + + "---", + + { + opcode: "deleteClonesInSprite", + blockType: Scratch.BlockType.COMMAND, + text: "delete clones in [INPUT]", + arguments: { + INPUT: { + type: Scratch.ArgumentType.STRING, + menu: "spriteMenu", + }, + }, + }, + { + opcode: "deleteCloneWithVar", + blockType: Scratch.BlockType.COMMAND, + text: "delete clones with [INPUTA] set to [INPUTB]", + filter: [Scratch.TargetType.SPRITE], + arguments: { + INPUTA: { + type: Scratch.ArgumentType.STRING, + menu: "variablesMenu", + }, + INPUTB: { + type: Scratch.ArgumentType.STRING, + defaultValue: "1", + }, + }, + }, + + "---", + + { + opcode: "isClone", + blockType: Scratch.BlockType.BOOLEAN, + text: "is clone?", + filter: [Scratch.TargetType.SPRITE], + }, + + "---", + + { + opcode: "cloneCount", + blockType: Scratch.BlockType.REPORTER, + text: "clone count", + }, + { + opcode: "spriteCloneCount", + blockType: Scratch.BlockType.REPORTER, + text: "clone count of [INPUT]", + disableMonitor: true, + arguments: { + INPUT: { + type: Scratch.ArgumentType.STRING, + menu: "spriteMenu", + }, + }, + }, + ], + menus: { + spriteMenu: { + acceptReporters: true, + items: "getSprites", + }, + // menus use acceptReporters: false for Scratch parity + variablesMenu: { + acceptReporters: false, + items: "getVariables", + }, + thingOfMenu: { + acceptReporters: false, + items: [ + { + text: "x position", + value: "x position", + }, + { + text: "y position", + value: "y position", + }, + { + text: "direction", + value: "direction", + }, + { + text: "costume #", + value: "costume num", + }, + { + text: "costume name", + value: "costume name", + }, + { + text: "size", + value: "size", + }, + { + text: "volume", + value: "volume", + }, + ], + }, + }, + }; + } + + whenCloneStartsWithVar(args, util) { + // TODO: this is really not ideal. this should be an event-based hat ideally, but we don't have a good + // way to do that right now... + if (util.target.isOriginal) { + return false; + } + const variable = util.target.lookupVariableById(args.INPUTA); + const expectedValue = args.INPUTB; + if (variable) { + return Scratch.Cast.compare(variable.value, expectedValue) === 0; + } + return false; + } + + createCloneWithVar(args, util) { + // @ts-expect-error - not typed yet + Scratch.vm.runtime.ext_scratch3_control._createClone( + util.target.sprite.name, + util.target + ); + const clones = util.target.sprite.clones; + const cloneNum = clones.length - 1; + const cloneVariable = clones[cloneNum].lookupVariableById(args.INPUTA); + if (cloneVariable) { + cloneVariable.value = args.INPUTB; + } + } + + touchingCloneWithVar(args, util) { + const drawableCandidates = util.target.sprite.clones + .filter((clone) => { + const variable = clone.lookupVariableById(args.INPUTA); + return ( + variable && Scratch.Cast.compare(variable.value, args.INPUTB) === 0 + ); + }) + .map((clone) => clone.drawableID); + if (drawableCandidates.length === 0) { + return false; + } + return Scratch.vm.renderer.isTouchingDrawables( + util.target.drawableID, + drawableCandidates + ); + } + + touchingMainSprite(args, util) { + if (util.target.isOriginal) { + return false; + } + const main = util.target.sprite.clones[0]; + const drawableCandidates = [main.drawableID]; + return Scratch.vm.renderer.isTouchingDrawables( + util.target.drawableID, + drawableCandidates + ); + } + + setVariableOfClone(args, util) { + const newVariableValue = args.INPUTB; + const expectedVarValue = args.INPUTD; + const clones = util.target.sprite.clones; + for (let index = 1; index < clones.length; index++) { + const checkVar = clones[index].lookupVariableById(args.INPUTC); + if ( + checkVar && + Scratch.Cast.compare(checkVar.value, expectedVarValue) === 0 + ) { + const editVar = clones[index].lookupVariableById(args.INPUTA); + if (editVar) { + editVar.value = newVariableValue; + } + } + } + } + + getVariableOfClone(args, util) { + const clone = this.getCloneFromVariable( + args.INPUTB, + args.INPUTC, + util.target.sprite.clones + ); + if (!clone) { + return ""; + } + // guaranteed to exist by getCloneFromVariable + const cloneVar = clone.lookupVariableById(args.INPUTA); + return cloneVar.value; + } + + setVariableOfMainSprite(args, util) { + const main = util.target.sprite.clones[0]; + const variableObj = main.lookupVariableById(args.INPUTA); + if (variableObj) { + variableObj.value = args.INPUTB; + } + } + + getVariableOfMainSprite(args, util) { + const main = util.target.sprite.clones[0]; + const variableObj = main.lookupVariableById(args.INPUT); + if (variableObj) { + return variableObj.value; + } + return ""; + } + + cloneExists(args, util) { + const clone = this.getCloneFromVariable( + args.INPUTA, + args.INPUTB, + util.target.sprite.clones + ); + return !!clone; + } + + getThingOfClone(args, util) { + const clone = this.getCloneFromVariable( + args.INPUTB, + args.INPUTC, + util.target.sprite.clones + ); + return getThingOfTarget(clone, args.INPUTA); + } + + getThingOfMainSprite(args, util) { + const main = util.target.sprite.clones[0]; + return getThingOfTarget(main, args.INPUT); + } + + stopScriptsInSprite(args) { + const targetObj = Scratch.vm.runtime.getSpriteTargetByName(args.INPUT); + if (targetObj) { + Scratch.vm.runtime.stopForTarget(targetObj); + } + } + + stopScriptsInMainSprite(args, util) { + Scratch.vm.runtime.stopForTarget(util.target.sprite.clones[0]); + } + + stopScriptsInClone(args, util) { + const clones = util.target.sprite.clones; + let expectedValue = args.INPUTB; + for (let index = 1; index < clones.length; index++) { + const cloneVariable = clones[index].lookupVariableById(args.INPUTA); + if ( + cloneVariable && + Scratch.Cast.compare(cloneVariable.value, expectedValue) === 0 + ) { + Scratch.vm.runtime.stopForTarget(clones[index]); + } + } + } + + deleteClonesInSprite(args, util) { + const target = Scratch.vm.runtime.getSpriteTargetByName(args.INPUT); + if (!target) { + return; + } + const clones = target.sprite.clones; + for (let index = clones.length - 1; index > 0; index--) { + Scratch.vm.runtime.disposeTarget(clones[index]); + } + } + + deleteCloneWithVar(args, util) { + const clones = util.target.sprite.clones; + const expectedValue = args.INPUTB; + for (let index = clones.length - 1; index > 0; index--) { + const cloneVar = clones[index].lookupVariableById(args.INPUTA); + if ( + cloneVar && + Scratch.Cast.compare(cloneVar.value, expectedValue) === 0 + ) { + Scratch.vm.runtime.disposeTarget(clones[index]); + } + } + } + + isClone(args, util) { + return !util.target.isOriginal; + } + + cloneCount(args, util) { + return Scratch.vm.runtime._cloneCounter; + } + + spriteCloneCount(args, util) { + const target = Scratch.vm.runtime.getSpriteTargetByName(args.INPUT); + if (target) { + return target.sprite.clones.length - 1; + } + return 0; + } + + /** + * @param {string} variableId + * @param {unknown} expectedValue + * @param {VM.Target[]} clones + * @returns {VM.Target|null} + */ + getCloneFromVariable(variableId, expectedValue, clones) { + for (let index = 1; index < clones.length; index++) { + const cloneVar = clones[index].lookupVariableById(variableId); + if ( + cloneVar && + Scratch.Cast.compare(cloneVar.value, expectedValue) === 0 + ) { + return clones[index]; + } + } + return null; + } + + getSprites() { + let spriteNames = []; + const targets = Scratch.vm.runtime.targets; + const myself = Scratch.vm.runtime.getEditingTarget().sprite.name; + for (let index = 1; index < targets.length; index++) { + const curTarget = targets[index].sprite; + let display = curTarget.name; + if (myself === curTarget.name) { + display = "myself"; + } + if (targets[index].isOriginal) { + const jsonOBJ = { + text: display, + value: curTarget.name, + }; + spriteNames.push(jsonOBJ); + } + } + if (spriteNames.length > 0) { + return spriteNames; + } else { + return [{ text: "", value: 0 }]; //this should never happen but it's a failsafe + } + } + + getSpriteObj(name) { + //This is unused but I'm leaving it in for potential future blocks + const spriteObj = Scratch.vm.runtime.getSpriteTargetByName(name); + return JSON.stringify(spriteObj); + } + + getVariables() { + // @ts-expect-error - Blockly not typed yet + // eslint-disable-next-line no-undef + const variables = + typeof Blockly === "undefined" + ? [] + : Blockly.getMainWorkspace() + .getVariableMap() + .getVariablesOfType("") + .filter((model) => model.isLocal) + .map((model) => ({ + text: model.name, + value: model.getId(), + })); + if (variables.length > 0) { + return variables; + } else { + return [{ text: "", value: "" }]; + } + } + } + Scratch.extensions.register(new ClonesPlus()); +})(Scratch); diff --git a/extensions/Lily/CommentBlocks.js b/extensions/Lily/CommentBlocks.js index bd831f0b4f..d2a25c4851 100644 --- a/extensions/Lily/CommentBlocks.js +++ b/extensions/Lily/CommentBlocks.js @@ -1,84 +1,89 @@ +// Name: Comment Blocks +// ID: lmscomments +// Description: Annotate your scripts. +// By: LilyMakesThings + (function (Scratch) { - 'use strict'; + "use strict"; class CommentBlocks { getInfo() { return { - id: 'lmscomments', - name: 'Comment Blocks', - color1: '#e4db8c', - color2: '#c6be79', - color3: '#a8a167', + id: "lmscomments", + name: "Comment Blocks", + color1: "#e4db8c", + color2: "#c6be79", + color3: "#a8a167", blocks: [ { - opcode: 'commentHat', + opcode: "commentHat", blockType: Scratch.BlockType.HAT, - text: '// [COMMENT]', + text: "// [COMMENT]", isEdgeActivated: false, arguments: { COMMENT: { type: Scratch.ArgumentType.STRING, - defaultValue: 'comment' - } - } + defaultValue: "comment", + }, + }, }, { - opcode: 'commentCommand', + opcode: "commentCommand", blockType: Scratch.BlockType.COMMAND, - text: '// [COMMENT]', + text: "// [COMMENT]", arguments: { COMMENT: { type: Scratch.ArgumentType.STRING, - defaultValue: 'comment' - } - } + defaultValue: "comment", + }, + }, }, { - opcode: 'commentReporter', + opcode: "commentReporter", blockType: Scratch.BlockType.REPORTER, - text: '[INPUT] // [COMMENT]', + text: "[INPUT] // [COMMENT]", arguments: { COMMENT: { type: Scratch.ArgumentType.STRING, - defaultValue: 'comment' + defaultValue: "comment", }, INPUT: { type: Scratch.ArgumentType.STRING, - defaultValue: '' - } - } + defaultValue: "", + }, + }, }, { - opcode: 'commentBoolean', + opcode: "commentBoolean", blockType: Scratch.BlockType.BOOLEAN, - text: '[INPUT] // [COMMENT]', + text: "[INPUT] // [COMMENT]", arguments: { COMMENT: { type: Scratch.ArgumentType.STRING, - defaultValue: 'comment' + defaultValue: "comment", }, INPUT: { - type: Scratch.ArgumentType.BOOLEAN - } - } - } - ] + type: Scratch.ArgumentType.BOOLEAN, + }, + }, + }, + ], }; } - commentHat () { + commentHat() { // no-op } - commentCommand () { + commentCommand() { // no-op } - commentReporter (args) { + commentReporter(args) { return args.INPUT; } - commentBoolean (args) { + commentBoolean(args) { return args.INPUT || false; } } diff --git a/extensions/Lily/LooksPlus.js b/extensions/Lily/LooksPlus.js index 6c4c193c99..49e17834f9 100644 --- a/extensions/Lily/LooksPlus.js +++ b/extensions/Lily/LooksPlus.js @@ -1,11 +1,19 @@ +// Name: Looks Plus +// ID: lmsLooksPlus +// Description: Expands upon the looks category, allowing you to show/hide, get costume data and edit SVG skins on sprites. +// By: LilyMakesThings + (function (Scratch) { - 'use strict'; + "use strict"; - const menuIconURI = ''; + const menuIconURI = + ""; const requireNonPackagedRuntime = (blockName) => { if (Scratch.vm.runtime.isPackaged) { - alert(`To use the Looks+ ${blockName} block, the creator of the packaged project must uncheck "Remove raw asset data after loading to save RAM" under advanced settings in the packager.`); + alert( + `To use the Looks+ ${blockName} block, the creator of the packaged project must uncheck "Remove raw asset data after loading to save RAM" under advanced settings in the packager.` + ); return false; } return true; @@ -27,169 +35,169 @@ class LooksPlus { getInfo() { return { - id: 'lmsLooksPlus', - name: 'Looks+', - color1: '#9966FF', - color2: '#855CD6', - color3: '#774DCB', + id: "lmsLooksPlus", + name: "Looks+", + color1: "#9966FF", + color2: "#855CD6", + color3: "#774DCB", menuIconURI: menuIconURI, blocks: [ { - opcode: 'showSprite', + opcode: "showSprite", blockType: Scratch.BlockType.COMMAND, - text: 'show [TARGET]', + text: "show [TARGET]", arguments: { TARGET: { type: Scratch.ArgumentType.STRING, - menu: 'spriteMenu' - } - } + menu: "spriteMenu", + }, + }, }, { - opcode: 'hideSprite', + opcode: "hideSprite", blockType: Scratch.BlockType.COMMAND, - text: 'hide [TARGET]', + text: "hide [TARGET]", arguments: { TARGET: { type: Scratch.ArgumentType.STRING, - menu: 'spriteMenu' - } - } + menu: "spriteMenu", + }, + }, }, { - opcode: 'spriteVisible', + opcode: "spriteVisible", blockType: Scratch.BlockType.BOOLEAN, - text: '[TARGET] visible?', + text: "[TARGET] visible?", arguments: { TARGET: { type: Scratch.ArgumentType.STRING, - menu: 'spriteMenu' - } - } + menu: "spriteMenu", + }, + }, }, - '---', + "---", { - opcode: 'setLayerTo', + opcode: "setLayerTo", blockType: Scratch.BlockType.COMMAND, - text: 'set layer # of [TARGET] to [LAYER]', + text: "set layer # of [TARGET] to [LAYER]", arguments: { TARGET: { type: Scratch.ArgumentType.STRING, - menu: 'spriteMenu' + menu: "spriteMenu", }, LAYER: { type: Scratch.ArgumentType.NUMBER, - defaultValue: '1' - } - } + defaultValue: "1", + }, + }, }, { - opcode: 'spriteLayerNumber', + opcode: "spriteLayerNumber", blockType: Scratch.BlockType.REPORTER, - text: 'layer # of [TARGET]', + text: "layer # of [TARGET]", arguments: { TARGET: { type: Scratch.ArgumentType.STRING, - menu: 'spriteMenu' - } - } + menu: "spriteMenu", + }, + }, }, { - opcode: 'effectValue', + opcode: "effectValue", blockType: Scratch.BlockType.REPORTER, - text: '[EFFECT] effect of [TARGET]', + text: "[EFFECT] effect of [TARGET]", arguments: { EFFECT: { type: Scratch.ArgumentType.STRING, - defaultValue: 'color', - menu: 'effectMenu' + defaultValue: "color", + menu: "effectMenu", }, TARGET: { type: Scratch.ArgumentType.STRING, - menu: 'spriteMenu' - } - } + menu: "spriteMenu", + }, + }, }, - '---', + "---", { - opcode: 'targetCostumeNumber', + opcode: "targetCostumeNumber", blockType: Scratch.BlockType.REPORTER, - text: '# of costumes in [TARGET]', + text: "# of costumes in [TARGET]", arguments: { TARGET: { type: Scratch.ArgumentType.STRING, - menu: 'spriteMenu' - } - } + menu: "spriteMenu", + }, + }, }, { - opcode: 'costumeAttribute', + opcode: "costumeAttribute", blockType: Scratch.BlockType.REPORTER, - text: '[ATTRIBUTE] of [COSTUME]', + text: "[ATTRIBUTE] of [COSTUME]", arguments: { ATTRIBUTE: { type: Scratch.ArgumentType.STRING, - menu: 'costumeAttribute' + menu: "costumeAttribute", }, COSTUME: { - type: Scratch.ArgumentType.COSTUME - } - } + type: Scratch.ArgumentType.COSTUME, + }, + }, }, - '---', + "---", { - opcode: 'snapshotStage', + opcode: "snapshotStage", blockType: Scratch.BlockType.REPORTER, - text: 'snapshot stage', - disableMonitor: true + text: "snapshot stage", + disableMonitor: true, }, - '---', + "---", { - opcode: 'replaceCostumeContent', + opcode: "replaceCostumeContent", blockType: Scratch.BlockType.COMMAND, - text: 'set [TYPE] for [COSTUME] to [CONTENT]', + text: "set [TYPE] for [COSTUME] to [CONTENT]", arguments: { TYPE: { type: Scratch.ArgumentType.STRING, - menu: 'SVGPNG', - defaultValue: 'SVG' + menu: "SVGPNG", + defaultValue: "SVG", }, COSTUME: { - type: Scratch.ArgumentType.COSTUME + type: Scratch.ArgumentType.COSTUME, }, CONTENT: { type: Scratch.ArgumentType.STRING, - defaultValue: '' - } - } + defaultValue: "", + }, + }, }, { - opcode: 'restoreCostumeContent', + opcode: "restoreCostumeContent", blockType: Scratch.BlockType.COMMAND, - text: 'restore content for [COSTUME]', + text: "restore content for [COSTUME]", arguments: { COSTUME: { - type: Scratch.ArgumentType.COSTUME - } - } + type: Scratch.ArgumentType.COSTUME, + }, + }, }, { - opcode: 'costumeContent', + opcode: "costumeContent", blockType: Scratch.BlockType.REPORTER, - text: '[CONTENT] of costume # [COSTUME] of [TARGET]', + text: "[CONTENT] of costume # [COSTUME] of [TARGET]", arguments: { CONTENT: { type: Scratch.ArgumentType.STRING, - menu: 'contentType', - defaultValue: 'content' + menu: "contentType", + defaultValue: "content", }, COSTUME: { type: Scratch.ArgumentType.NUMBER, @@ -197,43 +205,43 @@ }, TARGET: { type: Scratch.ArgumentType.STRING, - menu: 'spriteMenu' - } - } + menu: "spriteMenu", + }, + }, }, - '---', + "---", { - opcode: 'replaceColors', + opcode: "replaceColors", blockType: Scratch.BlockType.REPORTER, - text: 'replace [COLOR1] with [COLOR2] in [SVG]', + text: "replace [COLOR1] with [COLOR2] in [SVG]", arguments: { COLOR1: { type: Scratch.ArgumentType.COLOR, - defaultValue: '#FCB1E3' + defaultValue: "#FCB1E3", }, COLOR2: { type: Scratch.ArgumentType.COLOR, - defaultValue: '#8ECAFF' + defaultValue: "#8ECAFF", }, SVG: { type: Scratch.ArgumentType.STRING, - defaultValue: '' - } - } + defaultValue: "", + }, + }, }, { - opcode: 'colorHex', + opcode: "colorHex", blockType: Scratch.BlockType.REPORTER, - text: 'hex of [COLOR]', + text: "hex of [COLOR]", arguments: { COLOR: { type: Scratch.ArgumentType.COLOR, - defaultValue: '#FFD983' - } - } - } + defaultValue: "#FFD983", + }, + }, + }, ], menus: { effectMenu: { @@ -241,87 +249,87 @@ acceptReporters: false, items: [ { - text: 'color', - value: 'color' + text: "color", + value: "color", }, { - text: 'fisheye', - value: 'fisheye' + text: "fisheye", + value: "fisheye", }, { - text: 'whirl', - value: 'whirl' + text: "whirl", + value: "whirl", }, { - text: 'pixelate', - value: 'pixelate' + text: "pixelate", + value: "pixelate", }, { - text: 'mosaic', - value: 'mosaic' + text: "mosaic", + value: "mosaic", }, { - text: 'brightness', - value: 'brightness' + text: "brightness", + value: "brightness", }, { - text: 'ghost', - value: 'ghost' - } - ] + text: "ghost", + value: "ghost", + }, + ], }, costumeAttribute: { acceptReporters: false, items: [ { - text: 'width', - value: 'width' + text: "width", + value: "width", }, { - text: 'height', - value: 'height' + text: "height", + value: "height", }, { - text: 'format', - value: 'format' + text: "format", + value: "format", }, { - text: 'rotation center x', - value: 'rotationCenterX' + text: "rotation center x", + value: "rotationCenterX", }, { - text: 'rotation center y', - value: 'rotationCenterY' - } - ] + text: "rotation center y", + value: "rotationCenterY", + }, + ], }, contentType: { acceptReporters: false, items: [ { - text: 'content', - value: 'content' + text: "content", + value: "content", }, { - text: 'dataURI', - value: 'dataURI' - } - ] + text: "dataURI", + value: "dataURI", + }, + ], }, SVGPNG: { acceptReporters: false, items: [ { - text: 'SVG', - value: 'SVG' - } - ] + text: "SVG", + value: "SVG", + }, + ], }, spriteMenu: { acceptReporters: true, - items: 'getSprites' - } - } + items: "getSprites", + }, + }, }; } @@ -354,8 +362,8 @@ } const drawableID = target.drawableID; const layerOrder = target.getLayerOrder(); - const newLayer = (args.LAYER - layerOrder); - target.renderer.setDrawableOrder(drawableID, newLayer, 'sprite', true); + const newLayer = args.LAYER - layerOrder; + target.renderer.setDrawableOrder(drawableID, newLayer, "sprite", true); } spriteLayerNumber(args, util) { @@ -384,26 +392,26 @@ const costumeIndex = this.getCostumeInput(args.COSTUME, util.target); const costume = util.target.sprite.costumes[costumeIndex]; if (!costume) { - console.error('Costume doesn\'t exist'); + console.error("Costume doesn't exist"); return 0; } const attribute = args.ATTRIBUTE; - if (attribute === 'width') { + if (attribute === "width") { return Math.ceil(Scratch.Cast.toNumber(costume.size[0])); - } else if (attribute === 'height') { + } else if (attribute === "height") { return Math.ceil(Scratch.Cast.toNumber(costume.size[1])); - } else if (attribute === 'format') { - if (!requireNonPackagedRuntime('costume format')) { - return 'unknown'; + } else if (attribute === "format") { + if (!requireNonPackagedRuntime("costume format")) { + return "unknown"; } return costume.asset.assetType.runtimeFormat; - } else if (attribute === 'rotationCenterX') { + } else if (attribute === "rotationCenterX") { return costume.rotationCenterX; - } else if (attribute === 'rotationCenterY') { + } else if (attribute === "rotationCenterY") { return costume.rotationCenterY; } else { - return ''; + return ""; } } @@ -416,8 +424,8 @@ } snapshotStage(args, util) { - return new Promise(resolve => { - Scratch.vm.runtime.renderer.requestSnapshot(uri => { + return new Promise((resolve) => { + Scratch.vm.runtime.renderer.requestSnapshot((uri) => { resolve(uri); }); }); @@ -427,24 +435,27 @@ const costumeIndex = this.getCostumeInput(args.COSTUME, util.target); const costume = util.target.sprite.costumes[costumeIndex]; if (!costume) { - console.error('Costume doesn\'t exist'); + console.error("Costume doesn't exist"); return; } //This is here to ensure no changes are made to bitmap costumes, as changes are irreversible //Check will be removed when it's possible to edit bitmap skins const format = costume.asset.assetType.runtimeFormat; - if (format !== 'svg') { - console.error('Costume is not vector'); + if (format !== "svg") { + console.error("Costume is not vector"); return; } const contentType = args.TYPE; const content = args.CONTENT; - if (contentType === 'SVG') { - Scratch.vm.runtime.renderer.updateSVGSkin(costume.skinId, Scratch.Cast.toString(content)); + if (contentType === "SVG") { + Scratch.vm.runtime.renderer.updateSVGSkin( + costume.skinId, + Scratch.Cast.toString(content) + ); } else { - console.error('Options other than SVG are currently unavailable'); + console.error("Options other than SVG are currently unavailable"); return; } Scratch.vm.emitTargetsUpdate(); @@ -454,47 +465,50 @@ const costumeIndex = this.getCostumeInput(args.COSTUME, util.target); const costume = util.target.sprite.costumes[costumeIndex]; if (!costume) { - console.error('Costume doesn\'t exist'); + console.error("Costume doesn't exist"); return; } - if (!requireNonPackagedRuntime('restore costume content')) { + if (!requireNonPackagedRuntime("restore costume content")) { return; } //This is here to ensure no changes are made to bitmap costumes, as changes are irreversible //Check will be removed when it's possible to edit bitmap skins const format = costume.asset.assetType.runtimeFormat; - if (format !== 'svg') { - console.error('Costume is not vector'); + if (format !== "svg") { + console.error("Costume is not vector"); return; } const content = costume.asset.decodeText(); const rotationCenterX = costume.rotationCenterX; const rotationCenterY = costume.rotationCenterY; - util.target.renderer.updateSVGSkin(costume.skinId, content, [rotationCenterX, rotationCenterY]); + util.target.renderer.updateSVGSkin(costume.skinId, content, [ + rotationCenterX, + rotationCenterY, + ]); } costumeContent(args, util) { const target = getSpriteTargetByName(util, args.TARGET); if (!target) { - console.error('Target does not exist'); - return ''; + console.error("Target does not exist"); + return ""; } - if (!requireNonPackagedRuntime('costume content')) { - return ''; + if (!requireNonPackagedRuntime("costume content")) { + return ""; } - const costume = target.sprite.costumes[(args.COSTUME - 1)]; + const costume = target.sprite.costumes[args.COSTUME - 1]; if (!costume) { - console.error('Target costume does not exist'); - return ''; + console.error("Target costume does not exist"); + return ""; } const format = args.CONTENT; - if (format === 'content') { + if (format === "content") { return costume.asset.decodeText(); } else { return costume.asset.encodeDataURI(); @@ -506,7 +520,7 @@ const color1 = args.COLOR1; const color2 = args.COLOR2; try { - return svg.replace(new RegExp(color1, 'gi'), color2); + return svg.replace(new RegExp(color1, "gi"), color2); } catch (e) { // regex was invalid, don't replace anything return svg; @@ -518,7 +532,7 @@ } getCostumeInput(costume, target) { - if (typeof costume === 'number') { + if (typeof costume === "number") { costume = Math.round(costume - 1); if (costume === Infinity || costume === -Infinity || !costume) { costume = 0; @@ -531,8 +545,8 @@ } wrapClamp(n, min, max) { - const range = (max - min) + 1; - return n - (Math.floor((n - min) / range) * range); + const range = max - min + 1; + return n - Math.floor((n - min) / range) * range; } getSprites() { @@ -545,13 +559,13 @@ const targetName = target.getName(); if (targetName === myself) { spriteNames.unshift({ - text: 'this sprite', - value: targetName + text: "this sprite", + value: targetName, }); } else { spriteNames.push({ text: targetName, - value: targetName + value: targetName, }); } } @@ -559,7 +573,7 @@ if (spriteNames.length > 0) { return spriteNames; } else { - return [{text: "", value: 0}]; //this should never happen but it's a failsafe + return [{ text: "", value: 0 }]; //this should never happen but it's a failsafe } } } diff --git a/extensions/Lily/McUtils.js b/extensions/Lily/McUtils.js index 5303bc39e7..9e4d80758e 100644 --- a/extensions/Lily/McUtils.js +++ b/extensions/Lily/McUtils.js @@ -1,81 +1,86 @@ +// Name: McUtils +// ID: lmsmcutils +// Description: Helpful utilities for any fast food employee. +// By: LilyMakesThings + /*! * Credit to NexusKitten (NamelessCat) for the idea */ -(function(Scratch) { - 'use strict'; +(function (Scratch) { + "use strict"; class lmsmcutils { getInfo() { return { - id: 'lmsmcutils', - name: 'McUtils', - color1: '#ec2020', - color3: '#ffe427', + id: "lmsmcutils", + name: "McUtils", + color1: "#ec2020", + color3: "#ffe427", blocks: [ { - opcode: 'managerReporter', + opcode: "managerReporter", blockType: Scratch.BlockType.REPORTER, - text: 'if [INPUTA] is manager then [INPUTB] else [INPUTC]', + text: "if [INPUTA] is manager then [INPUTB] else [INPUTC]", arguments: { INPUTA: { - type: Scratch.ArgumentType.BOOLEAN + type: Scratch.ArgumentType.BOOLEAN, }, INPUTB: { - type: Scratch.ArgumentType.STRING + type: Scratch.ArgumentType.STRING, }, INPUTC: { - type: Scratch.ArgumentType.STRING - } - } + type: Scratch.ArgumentType.STRING, + }, + }, }, { - opcode: 'icecreammachine', + opcode: "icecreammachine", blockType: Scratch.BlockType.BOOLEAN, - text: 'is ice cream machine [INPUT]', + text: "is ice cream machine [INPUT]", arguments: { INPUT: { type: Scratch.ArgumentType.STRING, - menu: 'iceCreamMenu' - } - } + menu: "iceCreamMenu", + }, + }, }, { - opcode: 'talkToManager', + opcode: "talkToManager", blockType: Scratch.BlockType.BOOLEAN, - text: 'talk to manager [INPUT]', + text: "talk to manager [INPUT]", arguments: { INPUT: { type: Scratch.ArgumentType.STRING, - } - } + }, + }, }, { - opcode: 'placeOrder', + opcode: "placeOrder", blockType: Scratch.BlockType.REPORTER, - text: 'place order [INPUT]', + text: "place order [INPUT]", arguments: { INPUT: { type: Scratch.ArgumentType.STRING, - } - } - } + }, + }, + }, ], menus: { iceCreamMenu: { acceptReporters: true, items: [ { - text: 'working', - value: 'working' + text: "working", + value: "working", }, { - text: 'broken', - value: 'broken' - } - ] - } - } + text: "broken", + value: "broken", + }, + ], + }, + }, }; } @@ -88,7 +93,7 @@ } icecreammachine(args, util) { - if (args.INPUT === 'working') { + if (args.INPUT === "working") { return false; } else { return true; @@ -100,7 +105,7 @@ } placeOrder(args, util) { - if (args.INPUT.includes('ice cream')) { + if (args.INPUT.includes("ice cream")) { return false; } else { return args.INPUT; diff --git a/extensions/Lily/MoreEvents.js b/extensions/Lily/MoreEvents.js new file mode 100644 index 0000000000..cd1f329b65 --- /dev/null +++ b/extensions/Lily/MoreEvents.js @@ -0,0 +1,595 @@ +// Name: More Events +// ID: lmsMoreEvents +// Description: Start your scripts in new ways. +// By: LilyMakesThings + +(function (Scratch) { + "use strict"; + + const vm = Scratch.vm; + const runtime = vm.runtime; + + const stopIcon = + ""; + + // Source: + // https://github.com/TurboWarp/scratch-vm/blob/develop/src/io/keyboard.js + // https://github.com/TurboWarp/scratch-blocks/blob/develop/blocks_vertical/event.js + const validKeyboardInputs = [ + // Special Inputs + { text: "space", value: "space" }, + { text: "up arrow", value: "up arrow" }, + { text: "down arrow", value: "down arrow" }, + { text: "right arrow", value: "right arrow" }, + { text: "left arrow", value: "left arrow" }, + { text: "enter", value: "enter" }, + // TW: Extra Special Inputs + { text: "backspace", value: "backspace" }, + { text: "delete", value: "delete" }, + { text: "shift", value: "shift" }, + { text: "caps lock", value: "caps lock" }, + { text: "scroll lock", value: "scroll lock" }, + { text: "control", value: "control" }, + { text: "escape", value: "escape" }, + { text: "insert", value: "insert" }, + { text: "home", value: "home" }, + { text: "end", value: "end" }, + { text: "page up", value: "page up" }, + { text: "page down", value: "page down" }, + // Letter Keyboard Inputs + { text: "a", value: "a" }, + { text: "b", value: "b" }, + { text: "c", value: "c" }, + { text: "d", value: "d" }, + { text: "e", value: "e" }, + { text: "f", value: "f" }, + { text: "g", value: "g" }, + { text: "h", value: "h" }, + { text: "i", value: "i" }, + { text: "j", value: "j" }, + { text: "k", value: "k" }, + { text: "l", value: "l" }, + { text: "m", value: "m" }, + { text: "n", value: "n" }, + { text: "o", value: "o" }, + { text: "p", value: "p" }, + { text: "q", value: "q" }, + { text: "r", value: "r" }, + { text: "s", value: "s" }, + { text: "t", value: "t" }, + { text: "u", value: "u" }, + { text: "v", value: "v" }, + { text: "w", value: "w" }, + { text: "x", value: "x" }, + { text: "y", value: "y" }, + { text: "z", value: "z" }, + // Number Keyboard Inputs + { text: "0", value: "0" }, + { text: "1", value: "1" }, + { text: "2", value: "2" }, + { text: "3", value: "3" }, + { text: "4", value: "4" }, + { text: "5", value: "5" }, + { text: "6", value: "6" }, + { text: "7", value: "7" }, + { text: "8", value: "8" }, + { text: "9", value: "9" }, + ]; + + var lastValues = {}; + var runTimer = 0; + + class MoreEvents { + constructor() { + // Stop Sign Clicked contributed by @CST1229 + runtime.shouldExecuteStopClicked = true; + runtime.on("BEFORE_EXECUTE", () => { + runTimer++; + runtime.shouldExecuteStopClicked = false; + + runtime.startHats("lmsMoreEvents_forever"); + runtime.startHats("lmsMoreEvents_whileTrueFalse"); + runtime.startHats("lmsMoreEvents_whenValueChanged"); + runtime.startHats("lmsMoreEvents_everyDuration"); + runtime.startHats("lmsMoreEvents_whileKeyPressed"); + }); + runtime.on("PROJECT_START", () => { + runTimer = 0; + }); + runtime.on("PROJECT_STOP_ALL", () => { + runTimer = 0; + if (runtime.shouldExecuteStopClicked) + queueMicrotask(() => + runtime.startHats("lmsMoreEvents_whenStopClicked") + ); + }); + runtime.on("AFTER_EXECUTE", () => { + runtime.shouldExecuteStopClicked = true; + }); + const originalGreenFlag = vm.greenFlag; + vm.greenFlag = function () { + runtime.shouldExecuteStopClicked = false; + originalGreenFlag.call(this); + }; + } + + getInfo() { + return { + id: "lmsMoreEvents", + name: "More Events", + color1: "#FFBF00", + color2: "#E6AC00", + color3: "#CC9900", + blocks: [ + { + opcode: "whenStopClicked", + blockType: Scratch.BlockType.EVENT, + text: "when [STOP] clicked", + isEdgeActivated: false, + arguments: { + STOP: { + type: Scratch.ArgumentType.IMAGE, + dataURI: stopIcon, + }, + }, + }, + { + opcode: "forever", + blockType: Scratch.BlockType.EVENT, + text: "forever", + isEdgeActivated: false, + }, + + "---", + + { + opcode: "whenTrueFalse", + blockType: Scratch.BlockType.HAT, + text: "when [CONDITION] becomes [STATE]", + isEdgeActivated: true, + arguments: { + CONDITION: { + type: Scratch.ArgumentType.BOOLEAN, + }, + STATE: { + type: Scratch.ArgumentType.STRING, + menu: "boolean", + }, + }, + }, + { + opcode: "whileTrueFalse", + blockType: Scratch.BlockType.HAT, + text: "while [CONDITION] is [STATE]", + isEdgeActivated: false, + arguments: { + CONDITION: { + type: Scratch.ArgumentType.BOOLEAN, + }, + STATE: { + type: Scratch.ArgumentType.STRING, + menu: "boolean", + }, + }, + }, + + "---", + + { + opcode: "whenValueChanged", + blockType: Scratch.BlockType.HAT, + text: "when [INPUT] is changed", + isEdgeActivated: false, + arguments: { + INPUT: { + // Intentional: + // Encourages people to place a block + // (as opposed to typing a value) + type: null, + }, + }, + }, + { + opcode: "everyDuration", + blockType: Scratch.BlockType.HAT, + text: "every [DURATION] frames", + isEdgeActivated: false, + arguments: { + DURATION: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: 3, + }, + }, + }, + + "---", + + { + opcode: "whenKeyAction", + blockType: Scratch.BlockType.HAT, + text: "when [KEY_OPTION] key [ACTION]", + isEdgeActivated: true, + arguments: { + KEY_OPTION: { + type: Scratch.ArgumentType.STRING, + defaultValue: "space", + menu: "keyboardButtons", + }, + ACTION: { + type: Scratch.ArgumentType.STRING, + menu: "action", + }, + }, + }, + { + opcode: "whileKeyPressed", + blockType: Scratch.BlockType.HAT, + text: "while [KEY_OPTION] key pressed", + isEdgeActivated: false, + arguments: { + KEY_OPTION: { + type: Scratch.ArgumentType.STRING, + defaultValue: "space", + menu: "keyboardButtons", + }, + }, + }, + + "---", + + { + opcode: "broadcastToTarget", + blockType: Scratch.BlockType.COMMAND, + text: "broadcast [BROADCAST_OPTION] to [TARGET]", + arguments: { + BROADCAST_OPTION: { + type: null, + }, + TARGET: { + type: Scratch.ArgumentType.STRING, + menu: "targetMenu", + }, + }, + hideFromPalette: true, + }, + { + opcode: "broadcastToTargetAndWait", + blockType: Scratch.BlockType.COMMAND, + text: "broadcast [BROADCAST_OPTION] to [TARGET] and wait", + arguments: { + BROADCAST_OPTION: { + type: null, + }, + TARGET: { + type: Scratch.ArgumentType.STRING, + menu: "targetMenu", + }, + }, + hideFromPalette: true, + }, + + "---", + + { + opcode: "broadcastData", + blockType: Scratch.BlockType.COMMAND, + text: "broadcast [BROADCAST_OPTION] with data [DATA]", + arguments: { + BROADCAST_OPTION: { + type: null, + }, + DATA: { + type: Scratch.ArgumentType.STRING, + }, + }, + hideFromPalette: true, + }, + { + opcode: "broadcastDataAndWait", + blockType: Scratch.BlockType.COMMAND, + text: "broadcast [BROADCAST_OPTION] with data [DATA] and wait", + arguments: { + BROADCAST_OPTION: { + type: null, + }, + DATA: { + type: Scratch.ArgumentType.STRING, + }, + }, + hideFromPalette: true, + }, + { + blockType: Scratch.BlockType.XML, + xml: '', + }, + { + opcode: "receivedData", + blockType: Scratch.BlockType.REPORTER, + text: "received data", + disableMonitor: true, + allowDropAnywhere: true, + }, + + "---", + + { + opcode: "broadcastDataToTarget", + blockType: Scratch.BlockType.COMMAND, + text: "broadcast [BROADCAST_OPTION] to [TARGET] with data [DATA]", + func: "broadcastToTarget", + arguments: { + BROADCAST_OPTION: { + type: null, + }, + TARGET: { + type: Scratch.ArgumentType.STRING, + menu: "targetMenu", + }, + DATA: { + type: Scratch.ArgumentType.STRING, + }, + }, + hideFromPalette: true, + }, + { + opcode: "broadcastDataToTargetAndWait", + blockType: Scratch.BlockType.COMMAND, + text: "broadcast [BROADCAST_OPTION] to [TARGET] with data [DATA] and wait", + func: "broadcastToTargetAndWait", + arguments: { + BROADCAST_OPTION: { + type: null, + }, + TARGET: { + type: Scratch.ArgumentType.STRING, + menu: "targetMenu", + }, + DATA: { + type: Scratch.ArgumentType.STRING, + }, + }, + hideFromPalette: true, + }, + { + blockType: Scratch.BlockType.XML, + xml: '', + }, + ], + menus: { + // Targets have acceptReporters: true + targetMenu: { + acceptReporters: true, + items: "_getTargets", + }, + keyboardButtons: { + acceptReporters: true, + items: validKeyboardInputs, + }, + // Attributes have acceptReporters: false + action: { + acceptReporters: false, + items: ["hit", "released"], + }, + boolean: { + acceptReporters: false, + items: ["true", "false"], + }, + state: { + acceptReporters: false, + items: ["enabled", "disabled"], + }, + }, + }; + } + + whenTrueFalse(args) { + return args.STATE === "true" ? args.CONDITION : !args.CONDITION; + } + + whileTrueFalse(args) { + return args.STATE === "true" ? args.CONDITION : !args.CONDITION; + } + + whenValueChanged(args, util) { + const blockId = util.thread.peekStack(); + if (!lastValues[blockId]) + lastValues[blockId] = Scratch.Cast.toString(args.INPUT); + if (lastValues[blockId] !== Scratch.Cast.toString(args.INPUT)) { + lastValues[blockId] = Scratch.Cast.toString(args.INPUT); + return true; + } + return false; + } + + everyDuration(args, util) { + const duration = Math.max( + Math.round(Scratch.Cast.toNumber(args.DURATION)), + 0 + ); + return !!(runTimer % duration === 0); + } + + whenKeyAction(args, util) { + const key = Scratch.Cast.toString(args.KEY_OPTION).toLowerCase(); + const pressed = util.ioQuery("keyboard", "getKeyIsDown", [key]); + return args.ACTION === "released" ? !pressed : pressed; + } + + whileKeyPressed(args, util) { + const key = Scratch.Cast.toString(args.KEY_OPTION).toLowerCase(); + return util.ioQuery("keyboard", "getKeyIsDown", [key]); + } + + broadcastToTarget(args, util) { + const broadcastOption = Scratch.Cast.toString(args.BROADCAST_OPTION); + if (!broadcastOption) return; + + const data = Scratch.Cast.toString(args.DATA); + console.log(data); + + const cloneTargets = this._getTargetFromMenu(args.TARGET).sprite.clones; + let startedThreads = []; + + for (const clone of cloneTargets) { + startedThreads = [ + ...startedThreads, + ...util.startHats( + "event_whenbroadcastreceived", + { + BROADCAST_OPTION: broadcastOption, + }, + clone + ), + ]; + if (data) { + startedThreads.forEach((thread) => (thread.receivedData = args.DATA)); + } + } + } + + broadcastToTargetAndWait(args, util) { + if (!util.stackFrame.broadcastVar) { + util.stackFrame.broadcastVar = Scratch.Cast.toString( + args.BROADCAST_OPTION + ); + } + + const spriteTarget = this._getTargetFromMenu(args.TARGET); + if (!spriteTarget) return; + const cloneTargets = spriteTarget.sprite.clones; + + const data = Scratch.Cast.toString(args.DATA); + + if (util.stackFrame.broadcastVar) { + const broadcastOption = util.stackFrame.broadcastVar; + if (!util.stackFrame.startedThreads) { + util.stackFrame.startedThreads = []; + for (const clone of cloneTargets) { + util.stackFrame.startedThreads = [ + ...util.stackFrame.startedThreads, + ...util.startHats( + "event_whenbroadcastreceived", + { + BROADCAST_OPTION: broadcastOption, + }, + clone + ), + ]; + if (data) { + util.stackFrame.startedThreads.forEach( + (thread) => (thread.receivedData = args.DATA) + ); + } + } + if (util.stackFrame.startedThreads.length === 0) { + return; + } + } + + const waiting = util.stackFrame.startedThreads.some( + (thread) => runtime.threads.indexOf(thread) !== -1 + ); + if (waiting) { + if ( + util.stackFrame.startedThreads.every((thread) => + runtime.isWaitingThread(thread) + ) + ) { + util.yieldTick(); + } else { + util.yield(); + } + } + } + } + + broadcastData(args, util) { + const broadcast = Scratch.Cast.toString(args.BROADCAST_OPTION); + if (!broadcast) return; + + const data = Scratch.Cast.toString(args.DATA); + + let threads = util.startHats("event_whenbroadcastreceived", { + BROADCAST_OPTION: broadcast, + }); + threads.forEach((thread) => (thread.receivedData = data)); + } + + broadcastDataAndWait(args, util) { + const data = Scratch.Cast.toString(args.DATA); + + if (!util.stackFrame.broadcastVar) { + util.stackFrame.broadcastVar = Scratch.Cast.toString( + args.BROADCAST_OPTION + ); + } + + if (util.stackFrame.broadcastVar) { + const broadcastOption = util.stackFrame.broadcastVar; + if (!util.stackFrame.startedThreads) { + util.stackFrame.startedThreads = util.startHats( + "event_whenbroadcastreceived", + { + BROADCAST_OPTION: broadcastOption, + } + ); + if (util.stackFrame.startedThreads.length === 0) { + return; + } else { + util.stackFrame.startedThreads.forEach( + (thread) => (thread.receivedData = data) + ); + } + } + + const waiting = util.stackFrame.startedThreads.some( + (thread) => runtime.threads.indexOf(thread) !== -1 + ); + if (waiting) { + if ( + util.stackFrame.startedThreads.every((thread) => + runtime.isWaitingThread(thread) + ) + ) { + util.yieldTick(); + } else { + util.yield(); + } + } + } + } + + receivedData(args, util) { + const received = util.thread.receivedData; + return received ? received : ""; + } + + _getTargetFromMenu(targetName) { + let target = Scratch.vm.runtime.getSpriteTargetByName(targetName); + if (targetName === "_stage_") target = runtime.getTargetForStage(); + return target; + } + + _getTargets() { + const spriteNames = [{ text: "Stage", value: "_stage_" }]; + const targets = Scratch.vm.runtime.targets; + for (let index = 1; index < targets.length; index++) { + const target = targets[index]; + if (target.isOriginal) { + const targetName = target.getName(); + spriteNames.push({ + text: targetName, + value: targetName, + }); + } + } + if (spriteNames.length > 0) { + return spriteNames; + } else { + return [{ text: "", value: 0 }]; //this should never happen but it's a failsafe + } + } + } + + Scratch.extensions.register(new MoreEvents()); +})(Scratch); diff --git a/extensions/Lily/MoreTimers.js b/extensions/Lily/MoreTimers.js index 6878a4bcde..c6417906b9 100644 --- a/extensions/Lily/MoreTimers.js +++ b/extensions/Lily/MoreTimers.js @@ -1,279 +1,287 @@ -(function (Scratch) { - 'use strict'; - - const vm = Scratch.vm; - - /** - * @typedef Timer - * @property {number} startTime - * @property {number} pauseTime - * @property {boolean} paused - */ - - /** @type {Record} */ - let timers = Object.create(null); - - /** - * @param {Timer} timer - * @return {number} - */ - const timerValue = timer => { - return ((timer.paused ? 0 : (Date.now() - timer.startTime)) + timer.pauseTime) / 1000; - }; - - class Timers { - constructor() { - vm.runtime.on('PROJECT_START', () => { - timers = Object.create(null); - }); - } - - getInfo() { - return { - id: 'lmsTimers', - name: 'More Timers', - color1: '#5cb1d6', - color2: '#428faf', - color3: '#3281a3', - blocks: [ - { - opcode: 'whenTimerOp', - blockType: Scratch.BlockType.HAT, - extensions: ['colours_sensing'], - text: 'when timer [TIMER] [OP] [NUM]', - arguments: { - TIMER: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'timer' - }, - OP: { - type: Scratch.ArgumentType.STRING, - menu: 'operation' - }, - NUM: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '5' - } - } - }, - - '---', - - { - opcode: 'startResetTimer', - blockType: Scratch.BlockType.COMMAND, - extensions: ['colours_sensing'], - text: 'start/reset timer [TIMER]', - arguments: { - TIMER: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'timer' - } - } - }, - { - opcode: 'valueOfTimer', - blockType: Scratch.BlockType.REPORTER, - extensions: ['colours_sensing'], - text: 'timer [TIMER]', - arguments: { - TIMER: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'timer' - } - } - }, - - '---', - - { - opcode: 'pauseTimer', - blockType: Scratch.BlockType.COMMAND, - extensions: ['colours_sensing'], - text: 'pause timer [TIMER]', - arguments: { - TIMER: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'timer' - } - } - }, - { - opcode: 'resumeTimer', - blockType: Scratch.BlockType.COMMAND, - extensions: ['colours_sensing'], - text: 'resume timer [TIMER]', - arguments: { - TIMER: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'timer' - } - } - }, - - '---', - - { - opcode: 'setTimer', - blockType: Scratch.BlockType.COMMAND, - extensions: ['colours_sensing'], - text: 'set timer [TIMER] to [NUM]', - arguments: { - TIMER: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'timer' - }, - NUM: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '10' - } - } - }, - { - opcode: 'changeTimer', - blockType: Scratch.BlockType.COMMAND, - extensions: ['colours_sensing'], - text: 'change timer [TIMER] by [NUM]', - arguments: { - TIMER: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'timer' - }, - NUM: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '10' - } - } - }, - - '---', - - { - opcode: 'removeTimer', - blockType: Scratch.BlockType.COMMAND, - extensions: ['colours_sensing'], - text: 'remove timer [TIMER]', - arguments: { - TIMER: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'timer' - }, - } - }, - { - opcode: 'removeTimers', - blockType: Scratch.BlockType.COMMAND, - extensions: ['colours_sensing'], - text: 'remove all timers' - }, - { - opcode: 'timerExists', - blockType: Scratch.BlockType.BOOLEAN, - extensions: ['colours_sensing'], - text: 'timer [TIMER] exists?', - arguments: { - TIMER: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'timer' - }, - } - }, - { - opcode: 'listExistingTimers', - blockType: Scratch.BlockType.REPORTER, - extensions: ['colours_sensing'], - text: 'list existing timers', - disableMonitor: false - } - ], - menus: { - operation: { - // false for Scratch parity - acceptReporters: false, - items: ['>','<'] - } - } - }; - } - - whenTimerOp(args) { - if (!timers[args.TIMER]) return false; - const value = timerValue(timers[args.TIMER]); - if (args.OP === '>') return value > args.NUM; - if (args.OP === '<') return value < args.NUM; - return false; - } - - startResetTimer(args) { - timers[args.TIMER] = { - startTime: Date.now(), - pauseTime: 0, - paused: false - }; - } - - pauseTimer(args) { - const timer = timers[args.TIMER]; - if (!timer) return; - timer.pauseTime = timerValue(timer) * 1000; - timer.paused = true; - } - - resumeTimer(args) { - const timer = timers[args.TIMER]; - if (!timer) return; - if (timer.paused === false) return; - timer.paused = false; - timer.startTime = Date.now(); - } - - valueOfTimer(args) { - if (!timers[args.TIMER]) return ''; - return timerValue(timers[args.TIMER]); - } - - setTimer(args) { - timers[args.TIMER] = { - paused: false, - startTime: Date.now(), - pauseTime: Scratch.Cast.toNumber(args.NUM) * 1000 - }; - } - - changeTimer(args) { - if (!timers[args.TIMER]) this.startResetTimer(args); - timers[args.TIMER].pauseTime += Scratch.Cast.toNumber(args.NUM) * 1000; - } - - removeTimers(args) { - timers = Object.create(null); - } - - removeTimer(args) { - Reflect.deleteProperty(timers, args.TIMER); - } - - timerExists(args) { - return !!timers[args.TIMER]; - } - - listExistingTimers(args) { - return Object.keys(timers).join(','); - } - } - - // "Extension" option reimplementation by Xeltalliv - // https://github.com/Xeltalliv/extensions/blob/examples/examples/extension-colors.js - - // const cbfsb = Scratch.vm.runtime._convertBlockForScratchBlocks.bind(Scratch.vm.runtime); - // Scratch.vm.runtime._convertBlockForScratchBlocks = function(blockInfo, categoryInfo) { - // const res = cbfsb(blockInfo, categoryInfo); - // if (blockInfo.extensions) { - // if (!res.json.extensions) res.json.extensions = []; - // res.json.extensions.push(...blockInfo.extensions); - // } - // return res; - // }; - - Scratch.extensions.register(new Timers()); -})(Scratch); +// Name: More Timers +// ID: lmsTimers +// Description: Control several timers at once. +// By: LilyMakesThings + +(function (Scratch) { + "use strict"; + + const vm = Scratch.vm; + + /** + * @typedef Timer + * @property {number} startTime + * @property {number} pauseTime + * @property {boolean} paused + */ + + /** @type {Record} */ + let timers = Object.create(null); + + /** + * @param {Timer} timer + * @return {number} + */ + const timerValue = (timer) => { + return ( + ((timer.paused ? 0 : Date.now() - timer.startTime) + timer.pauseTime) / + 1000 + ); + }; + + class Timers { + constructor() { + vm.runtime.on("PROJECT_START", () => { + timers = Object.create(null); + }); + } + + getInfo() { + return { + id: "lmsTimers", + name: "More Timers", + color1: "#5cb1d6", + color2: "#428faf", + color3: "#3281a3", + blocks: [ + { + opcode: "whenTimerOp", + blockType: Scratch.BlockType.HAT, + extensions: ["colours_sensing"], + text: "when timer [TIMER] [OP] [NUM]", + arguments: { + TIMER: { + type: Scratch.ArgumentType.STRING, + defaultValue: "timer", + }, + OP: { + type: Scratch.ArgumentType.STRING, + menu: "operation", + }, + NUM: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "5", + }, + }, + }, + + "---", + + { + opcode: "startResetTimer", + blockType: Scratch.BlockType.COMMAND, + extensions: ["colours_sensing"], + text: "start/reset timer [TIMER]", + arguments: { + TIMER: { + type: Scratch.ArgumentType.STRING, + defaultValue: "timer", + }, + }, + }, + { + opcode: "valueOfTimer", + blockType: Scratch.BlockType.REPORTER, + extensions: ["colours_sensing"], + text: "timer [TIMER]", + arguments: { + TIMER: { + type: Scratch.ArgumentType.STRING, + defaultValue: "timer", + }, + }, + }, + + "---", + + { + opcode: "pauseTimer", + blockType: Scratch.BlockType.COMMAND, + extensions: ["colours_sensing"], + text: "pause timer [TIMER]", + arguments: { + TIMER: { + type: Scratch.ArgumentType.STRING, + defaultValue: "timer", + }, + }, + }, + { + opcode: "resumeTimer", + blockType: Scratch.BlockType.COMMAND, + extensions: ["colours_sensing"], + text: "resume timer [TIMER]", + arguments: { + TIMER: { + type: Scratch.ArgumentType.STRING, + defaultValue: "timer", + }, + }, + }, + + "---", + + { + opcode: "setTimer", + blockType: Scratch.BlockType.COMMAND, + extensions: ["colours_sensing"], + text: "set timer [TIMER] to [NUM]", + arguments: { + TIMER: { + type: Scratch.ArgumentType.STRING, + defaultValue: "timer", + }, + NUM: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "10", + }, + }, + }, + { + opcode: "changeTimer", + blockType: Scratch.BlockType.COMMAND, + extensions: ["colours_sensing"], + text: "change timer [TIMER] by [NUM]", + arguments: { + TIMER: { + type: Scratch.ArgumentType.STRING, + defaultValue: "timer", + }, + NUM: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "10", + }, + }, + }, + + "---", + + { + opcode: "removeTimer", + blockType: Scratch.BlockType.COMMAND, + extensions: ["colours_sensing"], + text: "remove timer [TIMER]", + arguments: { + TIMER: { + type: Scratch.ArgumentType.STRING, + defaultValue: "timer", + }, + }, + }, + { + opcode: "removeTimers", + blockType: Scratch.BlockType.COMMAND, + extensions: ["colours_sensing"], + text: "remove all timers", + }, + { + opcode: "timerExists", + blockType: Scratch.BlockType.BOOLEAN, + extensions: ["colours_sensing"], + text: "timer [TIMER] exists?", + arguments: { + TIMER: { + type: Scratch.ArgumentType.STRING, + defaultValue: "timer", + }, + }, + }, + { + opcode: "listExistingTimers", + blockType: Scratch.BlockType.REPORTER, + extensions: ["colours_sensing"], + text: "list existing timers", + disableMonitor: false, + }, + ], + menus: { + operation: { + // false for Scratch parity + acceptReporters: false, + items: [">", "<"], + }, + }, + }; + } + + whenTimerOp(args) { + if (!timers[args.TIMER]) return false; + const value = timerValue(timers[args.TIMER]); + if (args.OP === ">") return value > args.NUM; + if (args.OP === "<") return value < args.NUM; + return false; + } + + startResetTimer(args) { + timers[args.TIMER] = { + startTime: Date.now(), + pauseTime: 0, + paused: false, + }; + } + + pauseTimer(args) { + const timer = timers[args.TIMER]; + if (!timer) return; + timer.pauseTime = timerValue(timer) * 1000; + timer.paused = true; + } + + resumeTimer(args) { + const timer = timers[args.TIMER]; + if (!timer) return; + if (timer.paused === false) return; + timer.paused = false; + timer.startTime = Date.now(); + } + + valueOfTimer(args) { + if (!timers[args.TIMER]) return ""; + return timerValue(timers[args.TIMER]); + } + + setTimer(args) { + timers[args.TIMER] = { + paused: false, + startTime: Date.now(), + pauseTime: Scratch.Cast.toNumber(args.NUM) * 1000, + }; + } + + changeTimer(args) { + if (!timers[args.TIMER]) this.startResetTimer(args); + timers[args.TIMER].pauseTime += Scratch.Cast.toNumber(args.NUM) * 1000; + } + + removeTimers(args) { + timers = Object.create(null); + } + + removeTimer(args) { + Reflect.deleteProperty(timers, args.TIMER); + } + + timerExists(args) { + return !!timers[args.TIMER]; + } + + listExistingTimers(args) { + return Object.keys(timers).join(","); + } + } + + // "Extension" option reimplementation by Xeltalliv + // https://github.com/Xeltalliv/extensions/blob/examples/examples/extension-colors.js + + // const cbfsb = Scratch.vm.runtime._convertBlockForScratchBlocks.bind(Scratch.vm.runtime); + // Scratch.vm.runtime._convertBlockForScratchBlocks = function(blockInfo, categoryInfo) { + // const res = cbfsb(blockInfo, categoryInfo); + // if (blockInfo.extensions) { + // if (!res.json.extensions) res.json.extensions = []; + // res.json.extensions.push(...blockInfo.extensions); + // } + // return res; + // }; + + Scratch.extensions.register(new Timers()); +})(Scratch); diff --git a/extensions/Lily/Skins.js b/extensions/Lily/Skins.js new file mode 100644 index 0000000000..2c28f6cc94 --- /dev/null +++ b/extensions/Lily/Skins.js @@ -0,0 +1,459 @@ +// Name: Skins +// ID: lmsSkins +// Description: Have your sprites render as other images or costumes. +// By: LilyMakesThings + +(function (Scratch) { + "use strict"; + + const requireNonPackagedRuntime = (blockName) => { + if (Scratch.vm.runtime.isPackaged) { + alert( + `To use the Skins ${blockName} block, the creator of the packaged project must uncheck "Remove raw asset data after loading to save RAM" under advanced settings in the packager.` + ); + return false; + } + return true; + }; + + /** + * @param {RenderWebGL.SVGSkin} svgSkin + * @returns {Promise} + */ + const svgSkinFinishedLoading = (svgSkin) => + new Promise((resolve) => { + if (svgSkin._svgImageLoaded) { + resolve(); + } else { + svgSkin._svgImage.addEventListener("load", () => { + resolve(); + }); + svgSkin._svgImage.addEventListener("error", () => { + resolve(); + }); + } + }); + + const vm = Scratch.vm; + const runtime = vm.runtime; + const renderer = runtime.renderer; + const Cast = Scratch.Cast; + + var createdSkins = []; + + class Skins { + constructor() { + runtime.on("PROJECT_START", () => { + this._refreshTargets(); + }); + + runtime.on("PROJECT_STOP_ALL", () => { + this._refreshTargets(); + }); + } + + getInfo() { + return { + id: "lmsSkins", + name: "Skins", + color1: "#6b56ff", + color2: "#604de6", + color3: "#5645cc", + docsURI: "https://extensions.turbowarp.org/Lily/Skins", + blocks: [ + { + opcode: "registerSVGSkin", + blockType: Scratch.BlockType.COMMAND, + text: "create SVG skin [SVG] as [NAME]", + arguments: { + SVG: { + type: Scratch.ArgumentType.STRING, + defaultValue: "", + }, + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "my skin", + }, + }, + }, + + "---", + + { + opcode: "registerCostumeSkin", + blockType: Scratch.BlockType.COMMAND, + text: "load skin from [COSTUME] as [NAME]", + arguments: { + COSTUME: { + type: Scratch.ArgumentType.COSTUME, + }, + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "my skin", + }, + }, + }, + { + opcode: "registerURLSkin", + blockType: Scratch.BlockType.COMMAND, + text: "load skin from URL [URL] as [NAME]", + arguments: { + URL: { + type: Scratch.ArgumentType.STRING, + defaultValue: "https://extensions.turbowarp.org/dango.png", + }, + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "my skin", + }, + }, + }, + { + opcode: "getSkinLoaded", + blockType: Scratch.BlockType.BOOLEAN, + text: "skin [NAME] is loaded?", + arguments: { + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "my skin", + }, + }, + }, + + "---", + + { + opcode: "setSkin", + blockType: Scratch.BlockType.COMMAND, + text: "set skin of [TARGET] to [NAME]", + arguments: { + TARGET: { + type: Scratch.ArgumentType.STRING, + menu: "targetMenu", + }, + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "my skin", + }, + }, + }, + { + opcode: "restoreSkin", + blockType: Scratch.BlockType.COMMAND, + text: "restore skin of [TARGET]", + arguments: { + TARGET: { + type: Scratch.ArgumentType.STRING, + menu: "targetMenu", + }, + }, + }, + { + opcode: "restoreTargets", + blockType: Scratch.BlockType.COMMAND, + text: "restore targets with skin [NAME]", + arguments: { + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "my skin", + }, + }, + }, + + "---", + + { + opcode: "getCurrentSkin", + blockType: Scratch.BlockType.REPORTER, + text: "current skin of [TARGET]", + arguments: { + TARGET: { + type: Scratch.ArgumentType.STRING, + menu: "targetMenu", + }, + }, + }, + { + opcode: "getSkinAttribute", + blockType: Scratch.BlockType.REPORTER, + text: "[ATTRIBUTE] of skin [NAME]", + arguments: { + ATTRIBUTE: { + type: Scratch.ArgumentType.STRING, + menu: "skinAttributes", + }, + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "my skin", + }, + }, + }, + + "---", + + { + opcode: "deleteSkin", + blockType: Scratch.BlockType.COMMAND, + text: "delete skin [NAME]", + arguments: { + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "my skin", + }, + }, + }, + { + opcode: "deleteAllSkins", + blockType: Scratch.BlockType.COMMAND, + text: "delete all skins", + }, + ], + menus: { + targetMenu: { + acceptReporters: true, + items: "_getTargets", + }, + skinAttributes: { + acceptReporters: true, + items: ["width", "height"], + }, + }, + }; + } + + async registerSVGSkin(args) { + const skinName = Cast.toString(args.NAME); + const svgData = Cast.toString(args.SVG); + + let oldSkinId = null; + if (createdSkins[skinName]) { + oldSkinId = createdSkins[skinName]; + } + + // This generally takes a few frames, so yield the block + const skinId = renderer.createSVGSkin(svgData); + createdSkins[skinName] = skinId; + + await svgSkinFinishedLoading(renderer._allSkins[skinId]); + + if (oldSkinId) { + this._refreshTargetsFromID(oldSkinId, false, skinId); + renderer.destroySkin(oldSkinId); + } + } + + async registerCostumeSkin(args, util) { + if (!requireNonPackagedRuntime("add costume skin")) { + return; + } + + const skinName = Cast.toString(args.NAME); + const costumeIndex = util.target.getCostumeIndexByName(args.COSTUME); + if (costumeIndex === -1) return; + const costume = util.target.sprite.costumes[costumeIndex]; + + const url = costume.asset.encodeDataURI(); + const rotationCenterX = costume.rotationCenterX; + const rotationCenterY = costume.rotationCenterY; + + let rotationCenter = [rotationCenterX, rotationCenterY]; + if (!rotationCenterX || !rotationCenterY) rotationCenter = null; + + let oldSkinId = null; + if (createdSkins[skinName]) { + oldSkinId = createdSkins[skinName]; + } + + const skinId = await this._createURLSkin(url, rotationCenter); + createdSkins[skinName] = skinId; + + if (oldSkinId) { + this._refreshTargetsFromID(oldSkinId, false, skinId); + renderer.destroySkin(oldSkinId); + } + } + + async registerURLSkin(args) { + const skinName = Cast.toString(args.NAME); + const url = Cast.toString(args.URL); + + let oldSkinId = null; + if (createdSkins[skinName]) { + oldSkinId = createdSkins[skinName]; + } + + const skinId = await this._createURLSkin(url); + if (!skinId) return; + createdSkins[skinName] = skinId; + + if (oldSkinId) { + this._refreshTargetsFromID(oldSkinId, false, skinId); + renderer.destroySkin(oldSkinId); + } + } + + getSkinLoaded(args) { + const skinName = Cast.toString(args.NAME); + return !!createdSkins[skinName]; + } + + setSkin(args, util) { + const skinName = Cast.toString(args.NAME); + if (!createdSkins[skinName]) return; + + const targetName = Cast.toString(args.TARGET); + const target = this._getTargetFromMenu(targetName, util); + if (!target) return; + const drawableID = target.drawableID; + + const skinId = createdSkins[skinName]; + renderer._allDrawables[drawableID].skin = renderer._allSkins[skinId]; + } + + restoreSkin(args, util) { + const targetName = Cast.toString(args.TARGET); + const target = this._getTargetFromMenu(targetName, util); + if (!target) return; + target.updateAllDrawableProperties(); + } + + getCurrentSkin(args, util) { + const targetName = Cast.toString(args.TARGET); + const target = this._getTargetFromMenu(targetName, util); + if (!target) return; + const drawableID = target.drawableID; + + const skinId = renderer._allDrawables[drawableID].skin._id; + const skinName = this._getSkinNameFromID(skinId); + return skinName ? skinName : ""; + } + + getSkinAttribute(args) { + const skins = renderer._allSkins; + const skinName = Cast.toString(args.NAME); + + if (!createdSkins[skinName]) return 0; + const skinId = createdSkins[skinName]; + if (!skins[skinId]) return 0; + + const size = skins[skinId].size; + const attribute = Cast.toString(args.ATTRIBUTE).toLowerCase(); + + switch (attribute) { + case "width": + return Math.ceil(size[0]); + case "height": + return Math.ceil(size[1]); + default: + return 0; + } + } + + deleteSkin(args) { + const skinName = Cast.toString(args.NAME); + if (!createdSkins[skinName]) return; + const skinId = createdSkins[skinName]; + + this._refreshTargetsFromID(skinId, true); + renderer.destroySkin(skinId); + Reflect.deleteProperty(createdSkins, skinName); + } + + deleteAllSkins() { + this._refreshTargets(); + for (let i = 0; i < createdSkins.length; i++) + renderer.destroySkin(createdSkins[i]); + createdSkins = []; + } + + restoreTargets(args) { + const skinName = Cast.toString(args.NAME); + if (!createdSkins[skinName]) return; + const skinId = createdSkins[skinName]; + + this._refreshTargetsFromID(skinId, true); + } + + // Utility Functions + + _refreshTargetsFromID(skinId, reset, newId) { + const drawables = renderer._allDrawables; + const skins = renderer._allSkins; + + for (const target of runtime.targets) { + const drawableID = target.drawableID; + const targetSkin = drawables[drawableID].skin.id; + + if (targetSkin === skinId) { + target.updateAllDrawableProperties(); + if (!reset) + drawables[drawableID].skin = newId ? skins[newId] : skins[skinId]; + } + } + } + + _refreshTargets() { + for (const target of runtime.targets) { + target.updateAllDrawableProperties(); + } + } + + _getSkinNameFromID(skinId) { + for (const skinName in createdSkins) { + if (createdSkins[skinName] === skinId) return skinName; + } + } + + _getTargetFromMenu(targetName, util) { + let target = Scratch.vm.runtime.getSpriteTargetByName(targetName); + if (targetName === "_myself_") target = util.target; + if (targetName === "_stage_") target = runtime.getTargetForStage(); + return target; + } + + async _createURLSkin(URL, rotationCenter) { + let imageData; + if (await Scratch.canFetch(URL)) { + imageData = await Scratch.fetch(URL); + } else { + return; + } + + const contentType = imageData.headers.get("Content-Type"); + if (contentType === "image/svg+xml") { + return renderer.createSVGSkin(await imageData.text(), rotationCenter); + } else if ( + contentType === "image/png" || + contentType === "image/jpeg" || + contentType === "image/bmp" + ) { + // eslint-disable-next-line no-restricted-syntax + const output = new Image(); + output.src = URL; + output.crossOrigin = "anonymous"; + await output.decode(); + return renderer.createBitmapSkin(output); + } + } + + _getTargets() { + const spriteNames = [ + { text: "myself", value: "_myself_" }, + { text: "Stage", value: "_stage_" }, + ]; + const targets = Scratch.vm.runtime.targets; + for (let index = 1; index < targets.length; index++) { + const target = targets[index]; + if (target.isOriginal) { + const targetName = target.getName(); + spriteNames.push({ + text: targetName, + value: targetName, + }); + } + } + return spriteNames; + } + } + Scratch.extensions.register(new Skins()); +})(Scratch); diff --git a/extensions/Lily/TempVariables.js b/extensions/Lily/TempVariables.js index 0e40763ee1..a64d63d8d4 100644 --- a/extensions/Lily/TempVariables.js +++ b/extensions/Lily/TempVariables.js @@ -1,7 +1,7 @@ -(function(Scratch) { - 'use strict'; +(function (Scratch) { + "use strict"; - const menuIconURI = ''; + const menuIconURI = ""; // Object.create(null) prevents "variable [toString]" from returning a function let variables = Object.create(null); @@ -9,114 +9,114 @@ class TempVars { getInfo() { return { - id: 'lmstempvars', - name: 'Temporary Variables', - color1: '#FF791A', - color2: '#E15D00', + id: "lmstempvars", + name: "Temporary Variables", + color1: "#FF791A", + color2: "#E15D00", menuIconURI: menuIconURI, blocks: [ { - opcode: 'setVariableTo', + opcode: "setVariableTo", blockType: Scratch.BlockType.COMMAND, - text: 'set variable [INPUTA] to [INPUTB]', + text: "set variable [INPUTA] to [INPUTB]", arguments: { INPUTA: { type: Scratch.ArgumentType.STRING, - defaultValue: 'my variable' + defaultValue: "my variable", }, INPUTB: { type: Scratch.ArgumentType.STRING, - defaultValue: '0' - } - } + defaultValue: "0", + }, + }, }, { - opcode: 'changeVariableBy', + opcode: "changeVariableBy", blockType: Scratch.BlockType.COMMAND, - text: 'change variable [INPUTA] by [INPUTB]', + text: "change variable [INPUTA] by [INPUTB]", arguments: { INPUTA: { type: Scratch.ArgumentType.STRING, - defaultValue: 'my variable' + defaultValue: "my variable", }, INPUTB: { type: Scratch.ArgumentType.STRING, - defaultValue: '1' - } - } + defaultValue: "1", + }, + }, }, { - opcode: 'getVariable', + opcode: "getVariable", blockType: Scratch.BlockType.REPORTER, - text: 'variable [INPUT]', + text: "variable [INPUT]", disableMonitor: true, arguments: { INPUT: { type: Scratch.ArgumentType.STRING, - defaultValue: 'my variable' - } - } + defaultValue: "my variable", + }, + }, }, - '---', + "---", { - opcode: 'deleteVariable', + opcode: "deleteVariable", blockType: Scratch.BlockType.COMMAND, - text: 'delete variable [INPUT]', + text: "delete variable [INPUT]", arguments: { INPUT: { type: Scratch.ArgumentType.STRING, - defaultValue: 'my variable' - } - } + defaultValue: "my variable", + }, + }, }, { - opcode: 'deleteAllVariables', + opcode: "deleteAllVariables", blockType: Scratch.BlockType.COMMAND, - text: 'delete all variables', + text: "delete all variables", }, { - opcode: 'listVariables', + opcode: "listVariables", blockType: Scratch.BlockType.REPORTER, - text: 'list active variables', + text: "list active variables", disableMonitor: true, - } - ] + }, + ], }; } - getVariable (args) { + getVariable(args) { if (args.INPUT in variables) { - return (variables[args.INPUT]); + return variables[args.INPUT]; } else { - return ''; + return ""; } } - setVariableTo (args) { + setVariableTo(args) { variables[args.INPUTA] = args.INPUTB; } - changeVariableBy (args) { + changeVariableBy(args) { if (args.INPUTA in variables) { const prev = Scratch.Cast.toNumber(variables[args.INPUTA]); const next = Scratch.Cast.toNumber(args.INPUTB); - variables[args.INPUTA] = (prev + next); + variables[args.INPUTA] = prev + next; } else { variables[args.INPUTA] = args.INPUTB; } } - listVariables (args, util) { - return Object.keys(variables).join(','); + listVariables(args, util) { + return Object.keys(variables).join(","); } - deleteVariable (args) { + deleteVariable(args) { Reflect.deleteProperty(variables, args.INPUT); } - deleteAllVariables () { + deleteAllVariables() { variables = Object.create(null); } } diff --git a/extensions/Lily/TempVariables2.js b/extensions/Lily/TempVariables2.js index 13912281f8..9b28309e35 100644 --- a/extensions/Lily/TempVariables2.js +++ b/extensions/Lily/TempVariables2.js @@ -1,302 +1,306 @@ -(function(Scratch) { - 'use strict'; - - const menuIconURI = ''; - - // Object.create(null) prevents "variable [toString]" from returning a function - let runtimeVariables = Object.create(null); - - // Credit to skyhigh173 for the idea of this - const label = (name, hidden) => ({ - blockType: Scratch.BlockType.LABEL, - text: name, - hideFromPalette: hidden - }); - - function resetRuntimeVariables() { - runtimeVariables = Object.create(null); - } - - class TempVars { - constructor() { - Scratch.vm.runtime.on('PROJECT_START', () => { - resetRuntimeVariables(); - }); - - Scratch.vm.runtime.on('PROJECT_STOP_ALL', () => { - resetRuntimeVariables(); - }); - } - - getInfo() { - return { - id: 'lmsTempVars2', - name: 'Temporary Variables', - color1: '#FF791A', - color2: '#E15D00', - menuIconURI: menuIconURI, // I intend on making one later - blocks: [ - - label('Thread Variables', false), - - { - opcode: 'setThreadVariable', - blockType: Scratch.BlockType.COMMAND, - text: 'set thread var [VAR] to [STRING]', - arguments: { - VAR: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'variable' - }, - STRING: { - type: Scratch.ArgumentType.STRING, - defaultValue: '0' - } - }, - }, - { - opcode: 'changeThreadVariable', - blockType: Scratch.BlockType.COMMAND, - text: 'change thread var [VAR] by [NUM]', - arguments: { - VAR: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'variable' - }, - NUM: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '1' - } - }, - }, - - '---', - - { - opcode: 'getThreadVariable', - blockType: Scratch.BlockType.REPORTER, - text: 'thread var [VAR]', - arguments: { - VAR: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'variable' - } - } - }, - { - opcode: 'threadVariableExists', - blockType: Scratch.BlockType.BOOLEAN, - text: 'thread var [VAR] exists?', - arguments: { - VAR: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'variable' - } - } - }, - - '---', - - /* Add this when the compiler supports it - { - opcode: 'forEachThreadVariable', - blockType: Scratch.BlockType.LOOP, - text: 'for each [VAR] in [NUM]', - arguments: { - VAR: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'thread variable' - }, - NUM: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '10' - } - } - }, - */ - { - opcode: 'listThreadVariables', - blockType: Scratch.BlockType.REPORTER, - text: 'list active thread variables', - disableMonitor: true - }, - - '---', - - label('Runtime Variables', false), - - { - opcode: 'setRuntimeVariable', - blockType: Scratch.BlockType.COMMAND, - text: 'set runtime var [VAR] to [STRING]', - arguments: { - VAR: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'variable' - }, - STRING: { - type: Scratch.ArgumentType.STRING, - defaultValue: '0' - } - } - }, - { - opcode: 'changeRuntimeVariable', - blockType: Scratch.BlockType.COMMAND, - text: 'change runtime var [VAR] by [NUM]', - arguments: { - VAR: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'variable' - }, - NUM: { - type: Scratch.ArgumentType.STRING, - defaultValue: '1' - } - } - }, - - '---', - - { - opcode: 'getRuntimeVariable', - blockType: Scratch.BlockType.REPORTER, - text: 'runtime var [VAR]', - disableMonitor: true, - arguments: { - VAR: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'variable' - } - } - }, - { - opcode: 'runtimeVariableExists', - blockType: Scratch.BlockType.BOOLEAN, - text: 'runtime var [VAR] exists?', - arguments: { - VAR: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'variable' - } - } - }, - - '---', - - { - opcode: 'deleteRuntimeVariable', - blockType: Scratch.BlockType.COMMAND, - text: 'delete runtime var [VAR]', - arguments: { - VAR: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'variable' - } - } - }, - { - opcode: 'deleteAllRuntimeVariables', - blockType: Scratch.BlockType.COMMAND, - text: 'delete all runtime variables', - }, - { - opcode: 'listRuntimeVariables', - blockType: Scratch.BlockType.REPORTER, - text: 'list active runtime variables' - } - ] - }; - } - - /* THREAD VARIABLES */ - - setThreadVariable(args, util) { - const thread = util.thread; - if (!thread.variables) thread.variables = Object.create(null); - const vars = thread.variables; - vars[args.VAR] = args.STRING; - } - - changeThreadVariable(args, util) { - const thread = util.thread; - if (!thread.variables) thread.variables = Object.create(null); - const vars = thread.variables; - const prev = Scratch.Cast.toNumber(vars[args.VAR]); - const next = Scratch.Cast.toNumber(args.NUM); - vars[args.VAR] = prev + next; - } - - getThreadVariable (args, util) { - const thread = util.thread; - if (!thread.variables) thread.variables = Object.create(null); - const vars = thread.variables; - const varValue = vars[args.VAR]; - if (typeof varValue === 'undefined') return ''; - return varValue; - } - - threadVariableExists (args, util) { - const thread = util.thread; - if (!thread.variables) thread.variables = Object.create(null); - const vars = thread.variables; - const varValue = vars[args.VAR]; - return !(typeof varValue === 'undefined'); - } - - forEachThreadVariable(args, util) { - const thread = util.thread; - if (!thread.variables) thread.variables = Object.create(null); - const vars = thread.variables; - if (typeof util.stackFrame.index === 'undefined') { - util.stackFrame.index = 0; - } - if (util.stackFrame.index < Number(args.NUM)) { - util.stackFrame.index++; - vars[args.VAR] = util.stackFrame.index; - util.startBranch(1, true); - } - } - - listThreadVariables(args, util) { - const thread = util.thread; - if (!thread.variables) thread.variables = Object.create(null); - const vars = thread.variables; - return Object.keys(vars).join(','); - } - - /* RUNTIME VARIABLES */ - - setRuntimeVariable (args) { - runtimeVariables[args.VAR] = args.STRING; - } - - changeRuntimeVariable (args) { - const prev = Scratch.Cast.toNumber(runtimeVariables[args.VAR]); - const next = Scratch.Cast.toNumber(args.NUM); - runtimeVariables[args.VAR] = prev + next; - } - - getRuntimeVariable (args) { - if (!(args.VAR in runtimeVariables)) return ''; - return runtimeVariables[args.VAR]; - } - - runtimeVariableExists (args) { - return (args.VAR in runtimeVariables); - } - - listRuntimeVariables (args, util) { - return Object.keys(runtimeVariables).join(','); - } - - deleteRuntimeVariable (args) { - Reflect.deleteProperty(runtimeVariables, args.VAR); - } - - deleteAllRuntimeVariables () { - runtimeVariables = Object.create(null); - } - } - Scratch.extensions.register(new TempVars()); -})(Scratch); +// Name: Temporary Variables +// ID: lmsTempVars2 +// Description: Create disposable runtime or thread variables. +// By: LilyMakesThings + +(function (Scratch) { + "use strict"; + + const menuIconURI = ""; + + // Object.create(null) prevents "variable [toString]" from returning a function + let runtimeVariables = Object.create(null); + + // Credit to skyhigh173 for the idea of this + const label = (name, hidden) => ({ + blockType: Scratch.BlockType.LABEL, + text: name, + hideFromPalette: hidden, + }); + + function resetRuntimeVariables() { + runtimeVariables = Object.create(null); + } + + class TempVars { + constructor() { + Scratch.vm.runtime.on("PROJECT_START", () => { + resetRuntimeVariables(); + }); + + Scratch.vm.runtime.on("PROJECT_STOP_ALL", () => { + resetRuntimeVariables(); + }); + } + + getInfo() { + return { + id: "lmsTempVars2", + name: "Temporary Variables", + color1: "#FF791A", + color2: "#E15D00", + menuIconURI: menuIconURI, // I intend on making one later + blocks: [ + label("Thread Variables", false), + + { + opcode: "setThreadVariable", + blockType: Scratch.BlockType.COMMAND, + text: "set thread var [VAR] to [STRING]", + arguments: { + VAR: { + type: Scratch.ArgumentType.STRING, + defaultValue: "variable", + }, + STRING: { + type: Scratch.ArgumentType.STRING, + defaultValue: "0", + }, + }, + }, + { + opcode: "changeThreadVariable", + blockType: Scratch.BlockType.COMMAND, + text: "change thread var [VAR] by [NUM]", + arguments: { + VAR: { + type: Scratch.ArgumentType.STRING, + defaultValue: "variable", + }, + NUM: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "1", + }, + }, + }, + + "---", + + { + opcode: "getThreadVariable", + blockType: Scratch.BlockType.REPORTER, + text: "thread var [VAR]", + arguments: { + VAR: { + type: Scratch.ArgumentType.STRING, + defaultValue: "variable", + }, + }, + }, + { + opcode: "threadVariableExists", + blockType: Scratch.BlockType.BOOLEAN, + text: "thread var [VAR] exists?", + arguments: { + VAR: { + type: Scratch.ArgumentType.STRING, + defaultValue: "variable", + }, + }, + }, + + "---", + + /* Add this when the compiler supports it + { + opcode: 'forEachThreadVariable', + blockType: Scratch.BlockType.LOOP, + text: 'for each [VAR] in [NUM]', + arguments: { + VAR: { + type: Scratch.ArgumentType.STRING, + defaultValue: 'thread variable' + }, + NUM: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: '10' + } + } + }, + */ + { + opcode: "listThreadVariables", + blockType: Scratch.BlockType.REPORTER, + text: "list active thread variables", + disableMonitor: true, + }, + + "---", + + label("Runtime Variables", false), + + { + opcode: "setRuntimeVariable", + blockType: Scratch.BlockType.COMMAND, + text: "set runtime var [VAR] to [STRING]", + arguments: { + VAR: { + type: Scratch.ArgumentType.STRING, + defaultValue: "variable", + }, + STRING: { + type: Scratch.ArgumentType.STRING, + defaultValue: "0", + }, + }, + }, + { + opcode: "changeRuntimeVariable", + blockType: Scratch.BlockType.COMMAND, + text: "change runtime var [VAR] by [NUM]", + arguments: { + VAR: { + type: Scratch.ArgumentType.STRING, + defaultValue: "variable", + }, + NUM: { + type: Scratch.ArgumentType.STRING, + defaultValue: "1", + }, + }, + }, + + "---", + + { + opcode: "getRuntimeVariable", + blockType: Scratch.BlockType.REPORTER, + text: "runtime var [VAR]", + disableMonitor: true, + arguments: { + VAR: { + type: Scratch.ArgumentType.STRING, + defaultValue: "variable", + }, + }, + }, + { + opcode: "runtimeVariableExists", + blockType: Scratch.BlockType.BOOLEAN, + text: "runtime var [VAR] exists?", + arguments: { + VAR: { + type: Scratch.ArgumentType.STRING, + defaultValue: "variable", + }, + }, + }, + + "---", + + { + opcode: "deleteRuntimeVariable", + blockType: Scratch.BlockType.COMMAND, + text: "delete runtime var [VAR]", + arguments: { + VAR: { + type: Scratch.ArgumentType.STRING, + defaultValue: "variable", + }, + }, + }, + { + opcode: "deleteAllRuntimeVariables", + blockType: Scratch.BlockType.COMMAND, + text: "delete all runtime variables", + }, + { + opcode: "listRuntimeVariables", + blockType: Scratch.BlockType.REPORTER, + text: "list active runtime variables", + }, + ], + }; + } + + /* THREAD VARIABLES */ + + setThreadVariable(args, util) { + const thread = util.thread; + if (!thread.variables) thread.variables = Object.create(null); + const vars = thread.variables; + vars[args.VAR] = args.STRING; + } + + changeThreadVariable(args, util) { + const thread = util.thread; + if (!thread.variables) thread.variables = Object.create(null); + const vars = thread.variables; + const prev = Scratch.Cast.toNumber(vars[args.VAR]); + const next = Scratch.Cast.toNumber(args.NUM); + vars[args.VAR] = prev + next; + } + + getThreadVariable(args, util) { + const thread = util.thread; + if (!thread.variables) thread.variables = Object.create(null); + const vars = thread.variables; + const varValue = vars[args.VAR]; + if (typeof varValue === "undefined") return ""; + return varValue; + } + + threadVariableExists(args, util) { + const thread = util.thread; + if (!thread.variables) thread.variables = Object.create(null); + const vars = thread.variables; + const varValue = vars[args.VAR]; + return !(typeof varValue === "undefined"); + } + + forEachThreadVariable(args, util) { + const thread = util.thread; + if (!thread.variables) thread.variables = Object.create(null); + const vars = thread.variables; + if (typeof util.stackFrame.index === "undefined") { + util.stackFrame.index = 0; + } + if (util.stackFrame.index < Number(args.NUM)) { + util.stackFrame.index++; + vars[args.VAR] = util.stackFrame.index; + util.startBranch(1, true); + } + } + + listThreadVariables(args, util) { + const thread = util.thread; + if (!thread.variables) thread.variables = Object.create(null); + const vars = thread.variables; + return Object.keys(vars).join(","); + } + + /* RUNTIME VARIABLES */ + + setRuntimeVariable(args) { + runtimeVariables[args.VAR] = args.STRING; + } + + changeRuntimeVariable(args) { + const prev = Scratch.Cast.toNumber(runtimeVariables[args.VAR]); + const next = Scratch.Cast.toNumber(args.NUM); + runtimeVariables[args.VAR] = prev + next; + } + + getRuntimeVariable(args) { + if (!(args.VAR in runtimeVariables)) return ""; + return runtimeVariables[args.VAR]; + } + + runtimeVariableExists(args) { + return args.VAR in runtimeVariables; + } + + listRuntimeVariables(args, util) { + return Object.keys(runtimeVariables).join(","); + } + + deleteRuntimeVariable(args) { + Reflect.deleteProperty(runtimeVariables, args.VAR); + } + + deleteAllRuntimeVariables() { + runtimeVariables = Object.create(null); + } + } + Scratch.extensions.register(new TempVars()); +})(Scratch); diff --git a/extensions/Lily/lmsutils.js b/extensions/Lily/lmsutils.js index f72f40e4ae..91c42fdc38 100644 --- a/extensions/Lily/lmsutils.js +++ b/extensions/Lily/lmsutils.js @@ -1,1354 +1,1377 @@ -(function(Scratch) { - 'use strict'; - const menuIconURI = ''; - - let hideLegacyBlocks = true; - - var vars = {}; - vars['variables'] = Object.create(null); - - - if (!Scratch.extensions.unsandboxed) { - throw new Error('This extension must run unsandboxed'); - } - - class LMSUtils { - constructor(runtime) { - this.runtime = runtime; - } - getInfo() { - return { - id: 'lmsutilsblocks', - name: 'LMS Utilities', - color1: '#3bb2ed', - color2: '#37a1de', - color3: '#3693d9', - menuIconURI: menuIconURI, - blocks: [ - { - opcode: 'whenBooleanHat', - blockType: Scratch.BlockType.HAT, - text: 'when [INPUT] is true', - isEdgeActivated: true, - arguments: { - INPUT: { - type: Scratch.ArgumentType.BOOLEAN, - defaultValue: '' - } - } - }, - { - opcode: 'whenKeyString', - blockType: Scratch.BlockType.HAT, - text: 'when key [KEY_OPTION] pressed', - isEdgeActivated: true, - arguments: { - KEY_OPTION: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'enter' - } - } - }, - - '---', - - { - opcode: 'keyStringPressed', - blockType: Scratch.BlockType.BOOLEAN, - text: 'key [KEY_OPTION] pressed?', - arguments: { - KEY_OPTION: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'enter' - } - } - }, - { - opcode: 'trueFalseBoolean', - blockType: Scratch.BlockType.BOOLEAN, - text: '[TRUEFALSE]', - arguments: { - TRUEFALSE: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'true', - menu: 'trueFalseMenu' - } - } - }, - { - opcode: 'stringIf', - blockType: Scratch.BlockType.REPORTER, - text: 'if [BOOLEAN] then [INPUTA]', - disableMonitor: true, - arguments: { - BOOLEAN: { - type: Scratch.ArgumentType.BOOLEAN, - defaultValue: '' - }, - INPUTA: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'apple' - } - } - }, - { - opcode: 'stringIfElse', - blockType: Scratch.BlockType.REPORTER, - text: 'if [BOOLEAN] then [INPUTA] else [INPUTB]', - disableMonitor: true, - arguments: { - BOOLEAN: { - type: Scratch.ArgumentType.BOOLEAN, - defaultValue: '' - }, - INPUTA: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'apple' - }, - INPUTB: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'banana' - } - } - }, - - '---', - - { - opcode: 'getEffectValue', - blockType: Scratch.BlockType.REPORTER, - text: 'effect [INPUT]', - arguments: { - INPUT: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'color', - menu: 'colorMenu' - } - } - }, - { - opcode: 'clonesBeingUsed', - blockType: Scratch.BlockType.REPORTER, - text: 'clone count', - }, - { - opcode: 'isClone', - blockType: Scratch.BlockType.BOOLEAN, - text: 'is clone?', - filter: [Scratch.TargetType.SPRITE] - }, - { - opcode: 'spriteClicked', - blockType: Scratch.BlockType.BOOLEAN, - text: 'sprite clicked?', - filter: [Scratch.TargetType.SPRITE] - }, - - '---', - - { - opcode: 'lettersToOf', - blockType: Scratch.BlockType.REPORTER, - text: 'letters [INPUTA] to [INPUTB] of [STRING]', - disableMonitor: true, - arguments: { - INPUTA: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '1' - }, - INPUTB: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '3' - }, - STRING: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'suspicious' - } - } - }, - { - opcode: 'replaceWords', - blockType: Scratch.BlockType.REPORTER, - text: 'replace first [INPUTA] with [INPUTB] in [STRING]', - disableMonitor: true, - arguments: { - INPUTA: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'Scratch' - }, - INPUTB: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'Turbowarp' - }, - STRING: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'Scratch is brilliant!' - } - } - }, - { - opcode: 'findIndexOfString', - blockType: Scratch.BlockType.REPORTER, - text: 'index of [INPUTA] in [INPUTB]', - arguments: { - INPUTA: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'brilliant' - }, - INPUTB: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'Turbowarp is brilliant!' - } - } - }, - { - opcode: 'itemOfFromString', - blockType: Scratch.BlockType.REPORTER, - text: 'item [INPUTA] of [INPUTB] split by [INPUTC]', - arguments: { - INPUTA: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '2' - }, - INPUTB: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'apple|banana' - }, - INPUTC: { - type: Scratch.ArgumentType.STRING, - defaultValue: '|' - } - } - }, - { - opcode: 'stringToUpperCase', - blockType: Scratch.BlockType.REPORTER, - text: '[STRING] to uppercase', - disableMonitor: true, - arguments: { - STRING: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'apple' - } - } - }, - { - opcode: 'stringToLowerCase', - blockType: Scratch.BlockType.REPORTER, - text: '[STRING] to lowercase', - disableMonitor: true, - arguments: { - STRING: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'APPLE' - } - } - }, - { - opcode: 'reverseString', - blockType: Scratch.BlockType.REPORTER, - text: 'reverse [STRING]', - disableMonitor: true, - arguments: { - STRING: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'prawobrut' - } - } - }, - - '---', - - { - opcode: 'norBoolean', - blockType: Scratch.BlockType.BOOLEAN, - text: '[INPUTA] nor [INPUTB]', - arguments: { - INPUTA: { - type: Scratch.ArgumentType.BOOLEAN, - defaultValue: '' - }, - INPUTB: { - type: Scratch.ArgumentType.BOOLEAN, - defaultValue: '' - } - } - }, - { - opcode: 'xorBoolean', - blockType: Scratch.BlockType.BOOLEAN, - text: '[INPUTA] xor [INPUTB]', - arguments: { - INPUTA: { - type: Scratch.ArgumentType.BOOLEAN, - defaultValue: '' - }, - INPUTB: { - type: Scratch.ArgumentType.BOOLEAN, - defaultValue: '' - } - } - }, - { - opcode: 'xnorBoolean', - blockType: Scratch.BlockType.BOOLEAN, - text: '[INPUTA] xnor [INPUTB]', - arguments: { - INPUTA: { - type: Scratch.ArgumentType.BOOLEAN, - defaultValue: '' - }, - INPUTB: { - type: Scratch.ArgumentType.BOOLEAN, - defaultValue: '' - } - } - }, - { - opcode: 'nandBoolean', - blockType: Scratch.BlockType.BOOLEAN, - text: '[INPUTA] nand [INPUTB]', - arguments: { - INPUTA: { - type: Scratch.ArgumentType.BOOLEAN, - defaultValue: '' - }, - INPUTB: { - type: Scratch.ArgumentType.BOOLEAN, - defaultValue: '' - } - } - }, - - '---', - - { - opcode: 'stringReporter', - blockType: Scratch.BlockType.REPORTER, - text: '[STRING]', - disableMonitor: true, - hideFromPalette: hideLegacyBlocks, - arguments: { - STRING: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'apple' - } - } - }, - { - opcode: 'colourHex', - blockType: Scratch.BlockType.REPORTER, - text: 'color [COLOUR]', - hideFromPalette: hideLegacyBlocks, - arguments: { - COLOUR: { - type: Scratch.ArgumentType.COLOR, - defaultValue: '#0088ff' - } - } - }, - { - opcode: 'angleReporter', - blockType: Scratch.BlockType.REPORTER, - text: 'angle [ANGLE]', - hideFromPalette: hideLegacyBlocks, - arguments: { - ANGLE: { - type: Scratch.ArgumentType.ANGLE, - defaultValue: '90' - } - } - }, - { - opcode: 'matrixReporter', - blockType: Scratch.BlockType.REPORTER, - text: 'matrix [MATRIX]', - hideFromPalette: hideLegacyBlocks, - arguments: { - MATRIX: { - type: Scratch.ArgumentType.MATRIX, - defaultValue: '0101001010000001000101110' - } - } - }, - { - opcode: 'noteReporter', - blockType: Scratch.BlockType.REPORTER, - text: 'note [NOTE]', - hideFromPalette: hideLegacyBlocks, - arguments: { - NOTE: { - type: Scratch.ArgumentType.NOTE, - defaultValue: '' - } - } - }, - { - opcode: 'newlineCharacter', - blockType: Scratch.BlockType.REPORTER, - text: 'newline character', - hideFromPalette: hideLegacyBlocks, - disableMonitor: true - }, - - '---', - - { - opcode: 'equalsExactly', - blockType: Scratch.BlockType.BOOLEAN, - text: '[ONE] === [TWO]', - arguments: { - ONE: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'apple' - }, - TWO: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'banana' - } - } - }, - { - opcode: 'notEqualTo', - blockType: Scratch.BlockType.BOOLEAN, - text: '[INPUTA] ≠ [INPUTB]', - arguments: { - INPUTA: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'apple' - }, - INPUTB: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'banana' - } - } - }, - { - opcode: 'moreThanEqual', - blockType: Scratch.BlockType.BOOLEAN, - text: '[INPUTA] ≥ [INPUTB]', - arguments: { - INPUTA: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '16' - }, - INPUTB: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '25' - } - } - }, - { - opcode: 'lessThanEqual', - blockType: Scratch.BlockType.BOOLEAN, - text: '[INPUTA] ≤ [INPUTB]', - arguments: { - INPUTA: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '16' - }, - INPUTB: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '25' - } - } - }, - { - opcode: 'stringCheckBoolean', - blockType: Scratch.BlockType.BOOLEAN, - text: '[INPUT] is [DROPDOWN]', - arguments: { - INPUT: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'apple' - }, - DROPDOWN: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'text', - menu: 'stringCheckMenu' - } - } - }, - - '---', - - { - opcode: 'encodeToBlock', - blockType: Scratch.BlockType.REPORTER, - text: 'encode [STRING] to [DROPDOWN]', - disableMonitor: true, - hideFromPalette: hideLegacyBlocks, - arguments: { - STRING: { - type: Scratch.ArgumentType.STRING, - defaultValue: '' - }, - DROPDOWN: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'base64', - menu: 'conversionMenu' - } - } - }, - { - opcode: 'decodeFromBlock', - blockType: Scratch.BlockType.REPORTER, - text: 'decode [STRING] from [DROPDOWN]', - disableMonitor: true, - hideFromPalette: hideLegacyBlocks, - arguments: { - STRING: { - type: Scratch.ArgumentType.STRING, - defaultValue: '' - }, - DROPDOWN: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'base64', - menu: 'conversionMenu' - } - } - }, - - '---', - - { - opcode: 'negativeReporter', - blockType: Scratch.BlockType.REPORTER, - text: '- [INPUT]', - disableMonitor: true, - arguments: { - INPUT: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '' - } - } - }, - { - opcode: 'exponentBlock', - blockType: Scratch.BlockType.REPORTER, - text: '[INPUTA] ^ [INPUTB]', - disableMonitor: true, - arguments: { - INPUTA: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '' - }, - INPUTB: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '' - } - } - }, - { - opcode: 'rootBlock', - blockType: Scratch.BlockType.REPORTER, - text: '[INPUTA] √ [INPUTB]', - arguments: { - INPUTA: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '' - }, - INPUTB: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '' - } - } - }, - { - opcode: 'normaliseValue', - blockType: Scratch.BlockType.REPORTER, - text: 'normalise [INPUT]', - disableMonitor: true, - arguments: { - INPUT: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '100' - } - } - }, - { - opcode: 'clampNumber', - blockType: Scratch.BlockType.REPORTER, - text: 'clamp [INPUTA] between [INPUTB] and [INPUTC]', - arguments: { - INPUTA: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '100' - }, - INPUTB: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '25' - }, - INPUTC: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '50' - } - } - }, - - '---', - - { - opcode: 'setVariableTo', - blockType: Scratch.BlockType.COMMAND, - text: 'set variable [INPUTA] to [INPUTB]', - hideFromPalette: hideLegacyBlocks, - arguments: { - INPUTA: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'my variable' - }, - INPUTB: { - type: Scratch.ArgumentType.STRING, - defaultValue: '0' - } - } - }, - { - opcode: 'changeVariableBy', - blockType: Scratch.BlockType.COMMAND, - text: 'change variable [INPUTA] by [INPUTB]', - hideFromPalette: hideLegacyBlocks, - arguments: { - INPUTA: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'my variable' - }, - INPUTB: { - type: Scratch.ArgumentType.STRING, - defaultValue: '1' - } - } - }, - { - opcode: 'getVariable', - blockType: Scratch.BlockType.REPORTER, - text: 'variable [INPUT]', - disableMonitor: true, - hideFromPalette: hideLegacyBlocks, - arguments: { - INPUT: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'my variable' - } - } - }, - { - opcode: 'deleteVariable', - blockType: Scratch.BlockType.COMMAND, - text: 'delete variable [INPUT]', - hideFromPalette: hideLegacyBlocks, - arguments: { - INPUT: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'my variable' - } - } - }, - { - opcode: 'deleteAllVariables', - blockType: Scratch.BlockType.COMMAND, - text: 'delete all variables', - hideFromPalette: hideLegacyBlocks - }, - { - opcode: 'listVariables', - blockType: Scratch.BlockType.REPORTER, - text: 'list active variables', - disableMonitor: true, - hideFromPalette: hideLegacyBlocks - }, - - '---', - - { - opcode: 'greenFlag', - blockType: Scratch.BlockType.COMMAND, - text: 'green flag', - hideFromPalette: hideLegacyBlocks - }, - { - opcode: 'setUsername', - blockType: Scratch.BlockType.COMMAND, - text: 'set username to [INPUT]', - hideFromPalette: hideLegacyBlocks, - arguments: { - INPUT: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'LilyMakesThings' - } - } - }, - - '---', - - { - opcode: 'setSpriteSVG', - blockType: Scratch.BlockType.COMMAND, - text: 'replace SVG data for costume [INPUTA] with [INPUTB]', - hideFromPalette: hideLegacyBlocks, - arguments: { - INPUTA: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '1' - }, - INPUTB: { - type: Scratch.ArgumentType.STRING, - defaultValue: '' - } - } - }, - - '---', - - { - opcode: 'alertBlock', - blockType: Scratch.BlockType.COMMAND, - text: 'alert [STRING]', - hideFromPalette: hideLegacyBlocks, - arguments: { - STRING: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'A red spy is in the base!' - } - } - }, - { - opcode: 'inputPromptBlock', - blockType: Scratch.BlockType.REPORTER, - text: 'prompt [STRING]', - hideFromPalette: hideLegacyBlocks, - disableMonitor: true, - arguments: { - STRING: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'The code is 1, 1, 1.. err... 1!' - } - } - }, - { - opcode: 'confirmationBlock', - blockType: Scratch.BlockType.BOOLEAN, - text: 'confirm [STRING]', - hideFromPalette: hideLegacyBlocks, - arguments: { - STRING: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'Are you the red spy?' - } - } - }, - - '---', - - { - opcode: 'goToLink', - blockType: Scratch.BlockType.COMMAND, - text: 'open link [INPUT] in new tab', - hideFromPalette: hideLegacyBlocks, - arguments: { - INPUT: { - type: Scratch.ArgumentType.STRING, - defaultValue: '' - } - } - }, - { - opcode: 'redirectToLink', - blockType: Scratch.BlockType.COMMAND, - text: 'redirect to link [INPUT]', - hideFromPalette: hideLegacyBlocks, - arguments: { - INPUT: { - type: Scratch.ArgumentType.STRING, - defaultValue: '' - } - } - }, - - '---', - - { - opcode: 'setClipboard', - blockType: Scratch.BlockType.COMMAND, - text: 'set [STRING] to clipboard', - hideFromPalette: hideLegacyBlocks, - arguments: { - STRING: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'apple', - } - } - }, - { - opcode: 'readClipboard', - blockType: Scratch.BlockType.REPORTER, - text: 'clipboard', - hideFromPalette: hideLegacyBlocks - }, - - '---', - - { - opcode: 'isUserMobile', - blockType: Scratch.BlockType.BOOLEAN, - text: 'is mobile?' - }, - { - opcode: 'screenReporter', - blockType: Scratch.BlockType.REPORTER, - text: 'screen [DROPDOWN]', - disableMonitor: true, - arguments: { - DROPDOWN: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'width', - menu: 'screenReporterMenu' - } - } - }, - { - opcode: 'windowReporter', - blockType: Scratch.BlockType.REPORTER, - text: 'window [DROPDOWN]', - disableMonitor: true, - arguments: { - DROPDOWN: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'width', - menu: 'screenReporterMenu' - } - } - }, - { - opcode: 'osBrowserDetails', - blockType: Scratch.BlockType.REPORTER, - text: 'get [DROPDOWN] of user', - disableMonitor: true, - arguments: { - DROPDOWN: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'operating system', - menu: 'osBrowserMenu' - } - } - }, - { - opcode: 'projectURL', - blockType: Scratch.BlockType.REPORTER, - text: 'project URL', - disableMonitor: true, - }, - - '---', - - { - opcode: 'consoleLog', - blockType: Scratch.BlockType.COMMAND, - text: 'console [DROPDOWN] [INPUT]', - disableMonitor: true, - hideFromPalette: hideLegacyBlocks, - arguments: { - DROPDOWN: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'log', - menu: 'consoleLogMenu' - }, - INPUT: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'Apple' - } - } - }, - { - opcode: 'clearConsole', - blockType: Scratch.BlockType.COMMAND, - text: 'clear console', - hideFromPalette: hideLegacyBlocks - }, - - '---', - - { - opcode: 'commentHat', - blockType: Scratch.BlockType.HAT, - text: '// [STRING]', - hideFromPalette: hideLegacyBlocks, - arguments: { - STRING: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'comment', - } - } - }, - { - opcode: 'commentCommand', - blockType: Scratch.BlockType.COMMAND, - text: '// [STRING]', - hideFromPalette: hideLegacyBlocks, - arguments: { - STRING: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'comment', - } - } - }, - { - opcode: 'commentString', - blockType: Scratch.BlockType.REPORTER, - text: '// [INPUTA] [INPUTB]', - disableMonitor: true, - hideFromPalette: hideLegacyBlocks, - arguments: { - INPUTA: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'comment' - }, - INPUTB: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'input' - } - } - }, - { - opcode: 'commentBool', - blockType: Scratch.BlockType.BOOLEAN, - text: '// [INPUTA] [INPUTB]', - hideFromPalette: hideLegacyBlocks, - arguments: { - INPUTA: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'comment' - }, - INPUTB: { - type: Scratch.ArgumentType.BOOLEAN, - } - } - }, - - '---', - - { - func: 'showLegacyBlocks', - blockType: Scratch.BlockType.BUTTON, - text: 'Show Legacy Blocks', - hideFromPalette: !hideLegacyBlocks - }, - { - func: 'hideLegacyBlocks', - blockType: Scratch.BlockType.BUTTON, - text: 'Hide Legacy Blocks', - hideFromPalette: hideLegacyBlocks - }, - ], - menus: { - conversionMenu: { - acceptReporters: true, - items: ['base64', 'binary'] - }, - trueFalseMenu: { - acceptReporters: true, - items: ['true', 'false', 'random'] - }, - screenReporterMenu: { - acceptReporters: true, - items: ['width', 'height'] - }, - windowReporterMenu: { - acceptReporters: true, - items: ['width', 'height'] - }, - stringCheckMenu: { - acceptReporters: true, - items: ['text', 'number', 'uppercase', 'lowercase'] - }, - osBrowserMenu: { - acceptReporters: true, - items: ['operating system', 'browser'] - }, - consoleLogMenu: { - acceptReporters: false, - items: ['log', 'error', 'warn'] - }, - colorMenu: { - acceptReporters: true, - items: ['color', 'fisheye', 'whirl', 'pixelate', 'mosaic', 'brightness', 'ghost'] - }, - } - }; - } - - showLegacyBlocks() { - if (confirm('Are you sure you want to show legacy blocks? \n \n These blocks were removed because they were buggy or implemented better in other extensions.')) { - hideLegacyBlocks = false; - Scratch.vm.extensionManager.refreshBlocks(); - } else { - // - } - } - - hideLegacyBlocks() { - hideLegacyBlocks = true; - Scratch.vm.extensionManager.refreshBlocks(); - } - - whenBooleanHat(args) { - return args.INPUT; - } - - whenKeyString(args, util) { - return util.ioQuery('keyboard', 'getKeyIsDown', [args.KEY_OPTION]); - } - - keyStringPressed(args, util) { - return util.ioQuery('keyboard', 'getKeyIsDown', [args.KEY_OPTION]); - } - - trueFalseBoolean(args) { - if (args.TRUEFALSE === 'random') return Math.random() > 0.5; - if (args.TRUEFALSE === 'true') return true; - return false; - } - - stringIf(args) { - if (args.BOOLEAN) return args.INPUTA; - return ''; - } - - stringIfElse(args) { - if (args.BOOLEAN) return args.INPUTA; - return args.INPUTB; - } - - getEffectValue (args, util) { - return util.target.effects[args.INPUT]; - } - - clonesBeingUsed(args, util) { - return Scratch.vm.runtime._cloneCounter; - } - - isClone(args, util) { - return util.target.isOriginal ? false : true; - } - - spriteClicked(args, util) { - return (util.ioQuery('mouse', 'getIsDown') && util.target.isTouchingObject('_mouse_')); - } - - lettersToOf(args) { - var string = args.STRING.toString(); - var input1 = args.INPUTA - 1; - var input2 = args.INPUTB; - return string.slice(input1, input2); - } - - replaceWords(args) { - var input1 = args.INPUTA; - var input2 = args.INPUTB; - var string = args.STRING; - return string.replace(input1, input2); - } - - findIndexOfString (args) { - var input1 = args.INPUTA; - var input2 = args.INPUTB; - if (input2.includes(input1)) return (input2.indexOf(input1) + 1); - return ''; - } - - itemOfFromString (args, util) { - var input1 = (args.INPUTA - 1); - var input2 = String(args.INPUTB); - var input3 = args.INPUTC; - var output = input2.split(input3)[input1] || ''; - return output; - } - - stringToUpperCase(args) { - return args.STRING.toUpperCase(); - } - - stringToLowerCase(args) { - return args.STRING.toLowerCase(); - } - - reverseString(args) { - var input = args.STRING; - var splitInput = input.split(''); - var reversedInput = splitInput.reverse(); - var joinedArray = reversedInput.join(''); - return joinedArray; - } - - norBoolean(args) { - return !(args.INPUTA || args.INPUTB); - } - - xorBoolean(args) { - return (args.INPUTA !== args.INPUTB); - } - - xnorBoolean(args) { - return (args.INPUTA === args.INPUTB); - } - - nandBoolean(args) { - return !(args.INPUTA && args.INPUTB); - } - - stringReporter(args) { - return args.STRING; - } - - colourHex(args) { - return args.COLOUR; - } - - angleReporter(args) { - return args.ANGLE; - } - - matrixReporter(args) { - return args.MATRIX; - } - - noteReporter(args) { - return args.NOTE; - } - - newlineCharacter() { - return '\n'; - } - - equalsExactly(args) { - return args.ONE === args.TWO; - } - - notEqualTo(args) { - return (args.INPUTA != args.INPUTB); - } - - moreThanEqual(args) { - return (args.INPUTA >= args.INPUTB); - } - - lessThanEqual(args) { - return (args.INPUTA <= args.INPUTB); - } - - stringCheckBoolean(args) { - const input = args.INPUT; - const dropdown = args.DROPDOWN; - if (dropdown === 'text') return isNaN(input); - if (dropdown === 'number') return !isNaN(input); - if (dropdown === 'uppercase') return (input == input.toUpperCase()); - if (dropdown === 'lowercase') return (input == input.toLowerCase()); - return false; - } - - encodeToBlock(args) { - if (args.STRING === '') return ''; - if (args.DROPDOWN === 'base64') return btoa(args.STRING); - if (args.DROPDOWN === 'binary') { - return args.STRING.split('').map(function (char) { - return char.charCodeAt(0).toString(2); - }).join(' '); - } - return ''; - } - - decodeFromBlock(args) { - if (args.STRING === '') return ''; - if (args.DROPDOWN === 'base64') return atob(args.STRING); - if (args.DROPDOWN === 'binary') { - var output = args.STRING.toString(); - return output.split(' ').map((x) => x = String.fromCharCode(parseInt(x, 2))).join(''); - } - return ''; - } - - negativeReporter (args) { - return (args.INPUT * -1); - } - - exponentBlock(args) { - return Math.pow(args.INPUTA, args.INPUTB); - } - - rootBlock(args) { - return Math.pow(args.INPUTB, 1 / args.INPUTA); - } - - normaliseValue(args) { - var input1 = args.INPUT; - var input2 = Math.abs(input1); - var output = (input1 / input2); - if (isNaN(output)) return '0'; - return output; - } - - clampNumber (args) { - var input1 = args.INPUTA; - var input2 = args.INPUTB; - var input3 = args.INPUTC; - return Math.min(Math.max(input1, input2), input3); - } - - setVariableTo (args) { - vars['variables'][args.INPUTA] = args.INPUTB; - } - - changeVariableBy (args) { - if (args.INPUTA in vars['variables']) { - var prev = vars['variables'][args.INPUTA]; - var next = args.INPUTB; - vars['variables'][args.INPUTA] = (prev + next); - } else { - vars['variables'][args.INPUTA] = args.INPUTB; - } - } - - getVariable (args) { - if (args.INPUT in vars['variables']) return (vars['variables'][args.INPUT]); - return ''; - } - - deleteVariable (args) { - Reflect.deleteProperty(vars['variables'], args.INPUT); - } - - deleteAllVariables () { - Reflect.deleteProperty(vars, 'variables'); - vars['variables'] = {}; - } - - greenFlag(args, util) { - util.runtime.greenFlag(); - } - - setUsername(args, util) { - util.runtime.ioDevices.userData._username = args.INPUT; - } - - setSpriteSVG (args, util) { - try { - Scratch.vm.runtime.renderer.updateSVGSkin(util.target.sprite.costumes[args.INPUTA - 1].skinId,args.INPUTB); - } catch (error){ - return; - } - Scratch.vm.emitTargetsUpdate(); - } - - alertBlock(args) { - alert(args.STRING); - } - - inputPromptBlock(args) { - return prompt(args.STRING); - } - - confirmationBlock(args) { - return confirm(args.STRING); - } - - goToLink(args) { - Scratch.openWindow(args.INPUT); - } - - redirectToLink(args) { - Scratch.redirect(args.INPUT); - } - - setClipboard(args) { - if (navigator.clipboard && navigator.clipboard.writeText) { - navigator.clipboard.writeText(args.STRING); - } - } - - readClipboard(args) { - if (navigator.clipboard && navigator.clipboard.readText) { - return Scratch.canReadClipboard().then(allowed => { - if (allowed) { - return navigator.clipboard.readText(); - } - return ''; - }); - } - return ''; - } - - isUserMobile (args, util) { - return navigator.userAgent.includes('Mobile'); - } - - screenReporter(args) { - if (args.DROPDOWN === 'width') return screen.width; - if (args.DROPDOWN === 'height') return screen.height; - return ''; - } - - windowReporter(args) { - if (args.DROPDOWN === 'width') return window.innerWidth; - if (args.DROPDOWN === 'height') return window.innerHeight; - return ''; - } - - osBrowserDetails(args) { - var user = navigator.userAgent; - if (args.DROPDOWN === 'operating system') { - if (user.includes('Mac OS')) return 'macOS'; - if (user.includes('CrOS')) return 'ChromeOS'; - if (user.includes('Linux')) return 'Linux'; - if (user.includes('Windows')) return 'Windows'; - if (user.includes('iPad')) return 'iOS'; - if (user.includes('iPod')) return 'iOS'; - if (user.includes('iPhone')) return 'iOS'; - if (user.includes('Android')) return 'Android'; - return 'Other'; - } - if (args.DROPDOWN === 'browser') { - if (user.includes('Chrome')) return 'Chrome'; - if (user.includes('MSIE')) return 'Internet Explorer'; - if (user.includes('Firefox')) return 'Firefox'; - if (user.includes('Safari')) return 'Safari'; - return 'Other'; - } - } - - projectURL() { - return window.location.href; - } - - consoleLog(args) { - if (args.DROPDOWN === 'log') { - console.log(args.INPUT); - } else if (args.DROPDOWN === 'error') { - console.error(args.INPUT); - } else if (args.DROPDOWN === 'warn') { - console.warn(args.INPUT); - } - } - - clearConsole() { - console.clear(); - } - - commentHat () { - // no-op - } - - commentCommand () { - // no-op - } - - commentString (args) { - return args.INPUT; - } - - commentBool (args) { - return args.INPUT || false; - } - } - Scratch.extensions.register(new LMSUtils()); -})(Scratch); +(function (Scratch) { + "use strict"; + const menuIconURI = + ""; + + let hideLegacyBlocks = true; + + var vars = {}; + vars["variables"] = Object.create(null); + + if (!Scratch.extensions.unsandboxed) { + throw new Error("This extension must run unsandboxed"); + } + + class LMSUtils { + constructor(runtime) { + this.runtime = runtime; + } + getInfo() { + return { + id: "lmsutilsblocks", + name: "LMS Utilities", + color1: "#3bb2ed", + color2: "#37a1de", + color3: "#3693d9", + menuIconURI: menuIconURI, + blocks: [ + { + opcode: "whenBooleanHat", + blockType: Scratch.BlockType.HAT, + text: "when [INPUT] is true", + isEdgeActivated: true, + arguments: { + INPUT: { + type: Scratch.ArgumentType.BOOLEAN, + defaultValue: "", + }, + }, + }, + { + opcode: "whenKeyString", + blockType: Scratch.BlockType.HAT, + text: "when key [KEY_OPTION] pressed", + isEdgeActivated: true, + arguments: { + KEY_OPTION: { + type: Scratch.ArgumentType.STRING, + defaultValue: "enter", + }, + }, + }, + + "---", + + { + opcode: "keyStringPressed", + blockType: Scratch.BlockType.BOOLEAN, + text: "key [KEY_OPTION] pressed?", + arguments: { + KEY_OPTION: { + type: Scratch.ArgumentType.STRING, + defaultValue: "enter", + }, + }, + }, + { + opcode: "trueFalseBoolean", + blockType: Scratch.BlockType.BOOLEAN, + text: "[TRUEFALSE]", + arguments: { + TRUEFALSE: { + type: Scratch.ArgumentType.STRING, + defaultValue: "true", + menu: "trueFalseMenu", + }, + }, + }, + { + opcode: "stringIf", + blockType: Scratch.BlockType.REPORTER, + text: "if [BOOLEAN] then [INPUTA]", + disableMonitor: true, + arguments: { + BOOLEAN: { + type: Scratch.ArgumentType.BOOLEAN, + defaultValue: "", + }, + INPUTA: { + type: Scratch.ArgumentType.STRING, + defaultValue: "apple", + }, + }, + }, + { + opcode: "stringIfElse", + blockType: Scratch.BlockType.REPORTER, + text: "if [BOOLEAN] then [INPUTA] else [INPUTB]", + disableMonitor: true, + arguments: { + BOOLEAN: { + type: Scratch.ArgumentType.BOOLEAN, + defaultValue: "", + }, + INPUTA: { + type: Scratch.ArgumentType.STRING, + defaultValue: "apple", + }, + INPUTB: { + type: Scratch.ArgumentType.STRING, + defaultValue: "banana", + }, + }, + }, + + "---", + + { + opcode: "getEffectValue", + blockType: Scratch.BlockType.REPORTER, + text: "effect [INPUT]", + arguments: { + INPUT: { + type: Scratch.ArgumentType.STRING, + defaultValue: "color", + menu: "colorMenu", + }, + }, + }, + { + opcode: "clonesBeingUsed", + blockType: Scratch.BlockType.REPORTER, + text: "clone count", + }, + { + opcode: "isClone", + blockType: Scratch.BlockType.BOOLEAN, + text: "is clone?", + filter: [Scratch.TargetType.SPRITE], + }, + { + opcode: "spriteClicked", + blockType: Scratch.BlockType.BOOLEAN, + text: "sprite clicked?", + filter: [Scratch.TargetType.SPRITE], + }, + + "---", + + { + opcode: "lettersToOf", + blockType: Scratch.BlockType.REPORTER, + text: "letters [INPUTA] to [INPUTB] of [STRING]", + disableMonitor: true, + arguments: { + INPUTA: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "1", + }, + INPUTB: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "3", + }, + STRING: { + type: Scratch.ArgumentType.STRING, + defaultValue: "suspicious", + }, + }, + }, + { + opcode: "replaceWords", + blockType: Scratch.BlockType.REPORTER, + text: "replace first [INPUTA] with [INPUTB] in [STRING]", + disableMonitor: true, + arguments: { + INPUTA: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Scratch", + }, + INPUTB: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Turbowarp", + }, + STRING: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Scratch is brilliant!", + }, + }, + }, + { + opcode: "findIndexOfString", + blockType: Scratch.BlockType.REPORTER, + text: "index of [INPUTA] in [INPUTB]", + arguments: { + INPUTA: { + type: Scratch.ArgumentType.STRING, + defaultValue: "brilliant", + }, + INPUTB: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Turbowarp is brilliant!", + }, + }, + }, + { + opcode: "itemOfFromString", + blockType: Scratch.BlockType.REPORTER, + text: "item [INPUTA] of [INPUTB] split by [INPUTC]", + arguments: { + INPUTA: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "2", + }, + INPUTB: { + type: Scratch.ArgumentType.STRING, + defaultValue: "apple|banana", + }, + INPUTC: { + type: Scratch.ArgumentType.STRING, + defaultValue: "|", + }, + }, + }, + { + opcode: "stringToUpperCase", + blockType: Scratch.BlockType.REPORTER, + text: "[STRING] to uppercase", + disableMonitor: true, + arguments: { + STRING: { + type: Scratch.ArgumentType.STRING, + defaultValue: "apple", + }, + }, + }, + { + opcode: "stringToLowerCase", + blockType: Scratch.BlockType.REPORTER, + text: "[STRING] to lowercase", + disableMonitor: true, + arguments: { + STRING: { + type: Scratch.ArgumentType.STRING, + defaultValue: "APPLE", + }, + }, + }, + { + opcode: "reverseString", + blockType: Scratch.BlockType.REPORTER, + text: "reverse [STRING]", + disableMonitor: true, + arguments: { + STRING: { + type: Scratch.ArgumentType.STRING, + defaultValue: "prawobrut", + }, + }, + }, + + "---", + + { + opcode: "norBoolean", + blockType: Scratch.BlockType.BOOLEAN, + text: "[INPUTA] nor [INPUTB]", + arguments: { + INPUTA: { + type: Scratch.ArgumentType.BOOLEAN, + defaultValue: "", + }, + INPUTB: { + type: Scratch.ArgumentType.BOOLEAN, + defaultValue: "", + }, + }, + }, + { + opcode: "xorBoolean", + blockType: Scratch.BlockType.BOOLEAN, + text: "[INPUTA] xor [INPUTB]", + arguments: { + INPUTA: { + type: Scratch.ArgumentType.BOOLEAN, + defaultValue: "", + }, + INPUTB: { + type: Scratch.ArgumentType.BOOLEAN, + defaultValue: "", + }, + }, + }, + { + opcode: "xnorBoolean", + blockType: Scratch.BlockType.BOOLEAN, + text: "[INPUTA] xnor [INPUTB]", + arguments: { + INPUTA: { + type: Scratch.ArgumentType.BOOLEAN, + defaultValue: "", + }, + INPUTB: { + type: Scratch.ArgumentType.BOOLEAN, + defaultValue: "", + }, + }, + }, + { + opcode: "nandBoolean", + blockType: Scratch.BlockType.BOOLEAN, + text: "[INPUTA] nand [INPUTB]", + arguments: { + INPUTA: { + type: Scratch.ArgumentType.BOOLEAN, + defaultValue: "", + }, + INPUTB: { + type: Scratch.ArgumentType.BOOLEAN, + defaultValue: "", + }, + }, + }, + + "---", + + { + opcode: "stringReporter", + blockType: Scratch.BlockType.REPORTER, + text: "[STRING]", + disableMonitor: true, + hideFromPalette: hideLegacyBlocks, + arguments: { + STRING: { + type: Scratch.ArgumentType.STRING, + defaultValue: "apple", + }, + }, + }, + { + opcode: "colourHex", + blockType: Scratch.BlockType.REPORTER, + text: "color [COLOUR]", + hideFromPalette: hideLegacyBlocks, + arguments: { + COLOUR: { + type: Scratch.ArgumentType.COLOR, + defaultValue: "#0088ff", + }, + }, + }, + { + opcode: "angleReporter", + blockType: Scratch.BlockType.REPORTER, + text: "angle [ANGLE]", + hideFromPalette: hideLegacyBlocks, + arguments: { + ANGLE: { + type: Scratch.ArgumentType.ANGLE, + defaultValue: "90", + }, + }, + }, + { + opcode: "matrixReporter", + blockType: Scratch.BlockType.REPORTER, + text: "matrix [MATRIX]", + hideFromPalette: hideLegacyBlocks, + arguments: { + MATRIX: { + type: Scratch.ArgumentType.MATRIX, + defaultValue: "0101001010000001000101110", + }, + }, + }, + { + opcode: "noteReporter", + blockType: Scratch.BlockType.REPORTER, + text: "note [NOTE]", + hideFromPalette: hideLegacyBlocks, + arguments: { + NOTE: { + type: Scratch.ArgumentType.NOTE, + defaultValue: "", + }, + }, + }, + { + opcode: "newlineCharacter", + blockType: Scratch.BlockType.REPORTER, + text: "newline character", + hideFromPalette: hideLegacyBlocks, + disableMonitor: true, + }, + + "---", + + { + opcode: "equalsExactly", + blockType: Scratch.BlockType.BOOLEAN, + text: "[ONE] === [TWO]", + arguments: { + ONE: { + type: Scratch.ArgumentType.STRING, + defaultValue: "apple", + }, + TWO: { + type: Scratch.ArgumentType.STRING, + defaultValue: "banana", + }, + }, + }, + { + opcode: "notEqualTo", + blockType: Scratch.BlockType.BOOLEAN, + text: "[INPUTA] ≠ [INPUTB]", + arguments: { + INPUTA: { + type: Scratch.ArgumentType.STRING, + defaultValue: "apple", + }, + INPUTB: { + type: Scratch.ArgumentType.STRING, + defaultValue: "banana", + }, + }, + }, + { + opcode: "moreThanEqual", + blockType: Scratch.BlockType.BOOLEAN, + text: "[INPUTA] ≥ [INPUTB]", + arguments: { + INPUTA: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "16", + }, + INPUTB: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "25", + }, + }, + }, + { + opcode: "lessThanEqual", + blockType: Scratch.BlockType.BOOLEAN, + text: "[INPUTA] ≤ [INPUTB]", + arguments: { + INPUTA: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "16", + }, + INPUTB: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "25", + }, + }, + }, + { + opcode: "stringCheckBoolean", + blockType: Scratch.BlockType.BOOLEAN, + text: "[INPUT] is [DROPDOWN]", + arguments: { + INPUT: { + type: Scratch.ArgumentType.STRING, + defaultValue: "apple", + }, + DROPDOWN: { + type: Scratch.ArgumentType.STRING, + defaultValue: "text", + menu: "stringCheckMenu", + }, + }, + }, + + "---", + + { + opcode: "encodeToBlock", + blockType: Scratch.BlockType.REPORTER, + text: "encode [STRING] to [DROPDOWN]", + disableMonitor: true, + hideFromPalette: hideLegacyBlocks, + arguments: { + STRING: { + type: Scratch.ArgumentType.STRING, + defaultValue: "", + }, + DROPDOWN: { + type: Scratch.ArgumentType.STRING, + defaultValue: "base64", + menu: "conversionMenu", + }, + }, + }, + { + opcode: "decodeFromBlock", + blockType: Scratch.BlockType.REPORTER, + text: "decode [STRING] from [DROPDOWN]", + disableMonitor: true, + hideFromPalette: hideLegacyBlocks, + arguments: { + STRING: { + type: Scratch.ArgumentType.STRING, + defaultValue: "", + }, + DROPDOWN: { + type: Scratch.ArgumentType.STRING, + defaultValue: "base64", + menu: "conversionMenu", + }, + }, + }, + + "---", + + { + opcode: "negativeReporter", + blockType: Scratch.BlockType.REPORTER, + text: "- [INPUT]", + disableMonitor: true, + arguments: { + INPUT: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "", + }, + }, + }, + { + opcode: "exponentBlock", + blockType: Scratch.BlockType.REPORTER, + text: "[INPUTA] ^ [INPUTB]", + disableMonitor: true, + arguments: { + INPUTA: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "", + }, + INPUTB: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "", + }, + }, + }, + { + opcode: "rootBlock", + blockType: Scratch.BlockType.REPORTER, + text: "[INPUTA] √ [INPUTB]", + arguments: { + INPUTA: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "", + }, + INPUTB: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "", + }, + }, + }, + { + opcode: "normaliseValue", + blockType: Scratch.BlockType.REPORTER, + text: "normalise [INPUT]", + disableMonitor: true, + arguments: { + INPUT: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "100", + }, + }, + }, + { + opcode: "clampNumber", + blockType: Scratch.BlockType.REPORTER, + text: "clamp [INPUTA] between [INPUTB] and [INPUTC]", + arguments: { + INPUTA: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "100", + }, + INPUTB: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "25", + }, + INPUTC: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "50", + }, + }, + }, + + "---", + + { + opcode: "setVariableTo", + blockType: Scratch.BlockType.COMMAND, + text: "set variable [INPUTA] to [INPUTB]", + hideFromPalette: hideLegacyBlocks, + arguments: { + INPUTA: { + type: Scratch.ArgumentType.STRING, + defaultValue: "my variable", + }, + INPUTB: { + type: Scratch.ArgumentType.STRING, + defaultValue: "0", + }, + }, + }, + { + opcode: "changeVariableBy", + blockType: Scratch.BlockType.COMMAND, + text: "change variable [INPUTA] by [INPUTB]", + hideFromPalette: hideLegacyBlocks, + arguments: { + INPUTA: { + type: Scratch.ArgumentType.STRING, + defaultValue: "my variable", + }, + INPUTB: { + type: Scratch.ArgumentType.STRING, + defaultValue: "1", + }, + }, + }, + { + opcode: "getVariable", + blockType: Scratch.BlockType.REPORTER, + text: "variable [INPUT]", + disableMonitor: true, + hideFromPalette: hideLegacyBlocks, + arguments: { + INPUT: { + type: Scratch.ArgumentType.STRING, + defaultValue: "my variable", + }, + }, + }, + { + opcode: "deleteVariable", + blockType: Scratch.BlockType.COMMAND, + text: "delete variable [INPUT]", + hideFromPalette: hideLegacyBlocks, + arguments: { + INPUT: { + type: Scratch.ArgumentType.STRING, + defaultValue: "my variable", + }, + }, + }, + { + opcode: "deleteAllVariables", + blockType: Scratch.BlockType.COMMAND, + text: "delete all variables", + hideFromPalette: hideLegacyBlocks, + }, + { + opcode: "listVariables", + blockType: Scratch.BlockType.REPORTER, + text: "list active variables", + disableMonitor: true, + hideFromPalette: hideLegacyBlocks, + }, + + "---", + + { + opcode: "greenFlag", + blockType: Scratch.BlockType.COMMAND, + text: "green flag", + hideFromPalette: hideLegacyBlocks, + }, + { + opcode: "setUsername", + blockType: Scratch.BlockType.COMMAND, + text: "set username to [INPUT]", + hideFromPalette: hideLegacyBlocks, + arguments: { + INPUT: { + type: Scratch.ArgumentType.STRING, + defaultValue: "LilyMakesThings", + }, + }, + }, + + "---", + + { + opcode: "setSpriteSVG", + blockType: Scratch.BlockType.COMMAND, + text: "replace SVG data for costume [INPUTA] with [INPUTB]", + hideFromPalette: hideLegacyBlocks, + arguments: { + INPUTA: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "1", + }, + INPUTB: { + type: Scratch.ArgumentType.STRING, + defaultValue: "", + }, + }, + }, + + "---", + + { + opcode: "alertBlock", + blockType: Scratch.BlockType.COMMAND, + text: "alert [STRING]", + hideFromPalette: hideLegacyBlocks, + arguments: { + STRING: { + type: Scratch.ArgumentType.STRING, + defaultValue: "A red spy is in the base!", + }, + }, + }, + { + opcode: "inputPromptBlock", + blockType: Scratch.BlockType.REPORTER, + text: "prompt [STRING]", + hideFromPalette: hideLegacyBlocks, + disableMonitor: true, + arguments: { + STRING: { + type: Scratch.ArgumentType.STRING, + defaultValue: "The code is 1, 1, 1.. err... 1!", + }, + }, + }, + { + opcode: "confirmationBlock", + blockType: Scratch.BlockType.BOOLEAN, + text: "confirm [STRING]", + hideFromPalette: hideLegacyBlocks, + arguments: { + STRING: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Are you the red spy?", + }, + }, + }, + + "---", + + { + opcode: "goToLink", + blockType: Scratch.BlockType.COMMAND, + text: "open link [INPUT] in new tab", + hideFromPalette: hideLegacyBlocks, + arguments: { + INPUT: { + type: Scratch.ArgumentType.STRING, + defaultValue: "", + }, + }, + }, + { + opcode: "redirectToLink", + blockType: Scratch.BlockType.COMMAND, + text: "redirect to link [INPUT]", + hideFromPalette: hideLegacyBlocks, + arguments: { + INPUT: { + type: Scratch.ArgumentType.STRING, + defaultValue: "", + }, + }, + }, + + "---", + + { + opcode: "setClipboard", + blockType: Scratch.BlockType.COMMAND, + text: "set [STRING] to clipboard", + hideFromPalette: hideLegacyBlocks, + arguments: { + STRING: { + type: Scratch.ArgumentType.STRING, + defaultValue: "apple", + }, + }, + }, + { + opcode: "readClipboard", + blockType: Scratch.BlockType.REPORTER, + text: "clipboard", + hideFromPalette: hideLegacyBlocks, + }, + + "---", + + { + opcode: "isUserMobile", + blockType: Scratch.BlockType.BOOLEAN, + text: "is mobile?", + }, + { + opcode: "screenReporter", + blockType: Scratch.BlockType.REPORTER, + text: "screen [DROPDOWN]", + disableMonitor: true, + arguments: { + DROPDOWN: { + type: Scratch.ArgumentType.STRING, + defaultValue: "width", + menu: "screenReporterMenu", + }, + }, + }, + { + opcode: "windowReporter", + blockType: Scratch.BlockType.REPORTER, + text: "window [DROPDOWN]", + disableMonitor: true, + arguments: { + DROPDOWN: { + type: Scratch.ArgumentType.STRING, + defaultValue: "width", + menu: "screenReporterMenu", + }, + }, + }, + { + opcode: "osBrowserDetails", + blockType: Scratch.BlockType.REPORTER, + text: "get [DROPDOWN] of user", + disableMonitor: true, + arguments: { + DROPDOWN: { + type: Scratch.ArgumentType.STRING, + defaultValue: "operating system", + menu: "osBrowserMenu", + }, + }, + }, + { + opcode: "projectURL", + blockType: Scratch.BlockType.REPORTER, + text: "project URL", + disableMonitor: true, + }, + + "---", + + { + opcode: "consoleLog", + blockType: Scratch.BlockType.COMMAND, + text: "console [DROPDOWN] [INPUT]", + disableMonitor: true, + hideFromPalette: hideLegacyBlocks, + arguments: { + DROPDOWN: { + type: Scratch.ArgumentType.STRING, + defaultValue: "log", + menu: "consoleLogMenu", + }, + INPUT: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Apple", + }, + }, + }, + { + opcode: "clearConsole", + blockType: Scratch.BlockType.COMMAND, + text: "clear console", + hideFromPalette: hideLegacyBlocks, + }, + + "---", + + { + opcode: "commentHat", + blockType: Scratch.BlockType.HAT, + text: "// [STRING]", + hideFromPalette: hideLegacyBlocks, + arguments: { + STRING: { + type: Scratch.ArgumentType.STRING, + defaultValue: "comment", + }, + }, + }, + { + opcode: "commentCommand", + blockType: Scratch.BlockType.COMMAND, + text: "// [STRING]", + hideFromPalette: hideLegacyBlocks, + arguments: { + STRING: { + type: Scratch.ArgumentType.STRING, + defaultValue: "comment", + }, + }, + }, + { + opcode: "commentString", + blockType: Scratch.BlockType.REPORTER, + text: "// [INPUTA] [INPUTB]", + disableMonitor: true, + hideFromPalette: hideLegacyBlocks, + arguments: { + INPUTA: { + type: Scratch.ArgumentType.STRING, + defaultValue: "comment", + }, + INPUTB: { + type: Scratch.ArgumentType.STRING, + defaultValue: "input", + }, + }, + }, + { + opcode: "commentBool", + blockType: Scratch.BlockType.BOOLEAN, + text: "// [INPUTA] [INPUTB]", + hideFromPalette: hideLegacyBlocks, + arguments: { + INPUTA: { + type: Scratch.ArgumentType.STRING, + defaultValue: "comment", + }, + INPUTB: { + type: Scratch.ArgumentType.BOOLEAN, + }, + }, + }, + + "---", + + { + func: "showLegacyBlocks", + blockType: Scratch.BlockType.BUTTON, + text: "Show Legacy Blocks", + hideFromPalette: !hideLegacyBlocks, + }, + { + func: "hideLegacyBlocks", + blockType: Scratch.BlockType.BUTTON, + text: "Hide Legacy Blocks", + hideFromPalette: hideLegacyBlocks, + }, + ], + menus: { + conversionMenu: { + acceptReporters: true, + items: ["base64", "binary"], + }, + trueFalseMenu: { + acceptReporters: true, + items: ["true", "false", "random"], + }, + screenReporterMenu: { + acceptReporters: true, + items: ["width", "height"], + }, + windowReporterMenu: { + acceptReporters: true, + items: ["width", "height"], + }, + stringCheckMenu: { + acceptReporters: true, + items: ["text", "number", "uppercase", "lowercase"], + }, + osBrowserMenu: { + acceptReporters: true, + items: ["operating system", "browser"], + }, + consoleLogMenu: { + acceptReporters: false, + items: ["log", "error", "warn"], + }, + colorMenu: { + acceptReporters: true, + items: [ + "color", + "fisheye", + "whirl", + "pixelate", + "mosaic", + "brightness", + "ghost", + ], + }, + }, + }; + } + + showLegacyBlocks() { + if ( + confirm( + "Are you sure you want to show legacy blocks? \n \n These blocks were removed because they were buggy or implemented better in other extensions." + ) + ) { + hideLegacyBlocks = false; + Scratch.vm.extensionManager.refreshBlocks(); + } else { + // + } + } + + hideLegacyBlocks() { + hideLegacyBlocks = true; + Scratch.vm.extensionManager.refreshBlocks(); + } + + whenBooleanHat(args) { + return args.INPUT; + } + + whenKeyString(args, util) { + return util.ioQuery("keyboard", "getKeyIsDown", [args.KEY_OPTION]); + } + + keyStringPressed(args, util) { + return util.ioQuery("keyboard", "getKeyIsDown", [args.KEY_OPTION]); + } + + trueFalseBoolean(args) { + if (args.TRUEFALSE === "random") return Math.random() > 0.5; + if (args.TRUEFALSE === "true") return true; + return false; + } + + stringIf(args) { + if (args.BOOLEAN) return args.INPUTA; + return ""; + } + + stringIfElse(args) { + if (args.BOOLEAN) return args.INPUTA; + return args.INPUTB; + } + + getEffectValue(args, util) { + return util.target.effects[args.INPUT]; + } + + clonesBeingUsed(args, util) { + return Scratch.vm.runtime._cloneCounter; + } + + isClone(args, util) { + return util.target.isOriginal ? false : true; + } + + spriteClicked(args, util) { + return ( + util.ioQuery("mouse", "getIsDown") && + util.target.isTouchingObject("_mouse_") + ); + } + + lettersToOf(args) { + var string = args.STRING.toString(); + var input1 = args.INPUTA - 1; + var input2 = args.INPUTB; + return string.slice(input1, input2); + } + + replaceWords(args) { + var input1 = args.INPUTA; + var input2 = args.INPUTB; + var string = args.STRING; + return string.replace(input1, input2); + } + + findIndexOfString(args) { + var input1 = args.INPUTA; + var input2 = args.INPUTB; + if (input2.includes(input1)) return input2.indexOf(input1) + 1; + return ""; + } + + itemOfFromString(args, util) { + var input1 = args.INPUTA - 1; + var input2 = String(args.INPUTB); + var input3 = args.INPUTC; + var output = input2.split(input3)[input1] || ""; + return output; + } + + stringToUpperCase(args) { + return args.STRING.toUpperCase(); + } + + stringToLowerCase(args) { + return args.STRING.toLowerCase(); + } + + reverseString(args) { + var input = args.STRING; + var splitInput = input.split(""); + var reversedInput = splitInput.reverse(); + var joinedArray = reversedInput.join(""); + return joinedArray; + } + + norBoolean(args) { + return !(args.INPUTA || args.INPUTB); + } + + xorBoolean(args) { + return args.INPUTA !== args.INPUTB; + } + + xnorBoolean(args) { + return args.INPUTA === args.INPUTB; + } + + nandBoolean(args) { + return !(args.INPUTA && args.INPUTB); + } + + stringReporter(args) { + return args.STRING; + } + + colourHex(args) { + return args.COLOUR; + } + + angleReporter(args) { + return args.ANGLE; + } + + matrixReporter(args) { + return args.MATRIX; + } + + noteReporter(args) { + return args.NOTE; + } + + newlineCharacter() { + return "\n"; + } + + equalsExactly(args) { + return args.ONE === args.TWO; + } + + notEqualTo(args) { + return args.INPUTA != args.INPUTB; + } + + moreThanEqual(args) { + return args.INPUTA >= args.INPUTB; + } + + lessThanEqual(args) { + return args.INPUTA <= args.INPUTB; + } + + stringCheckBoolean(args) { + const input = args.INPUT; + const dropdown = args.DROPDOWN; + if (dropdown === "text") return isNaN(input); + if (dropdown === "number") return !isNaN(input); + if (dropdown === "uppercase") return input == input.toUpperCase(); + if (dropdown === "lowercase") return input == input.toLowerCase(); + return false; + } + + encodeToBlock(args) { + if (args.STRING === "") return ""; + if (args.DROPDOWN === "base64") return btoa(args.STRING); + if (args.DROPDOWN === "binary") { + return args.STRING.split("") + .map(function (char) { + return char.charCodeAt(0).toString(2); + }) + .join(" "); + } + return ""; + } + + decodeFromBlock(args) { + if (args.STRING === "") return ""; + if (args.DROPDOWN === "base64") return atob(args.STRING); + if (args.DROPDOWN === "binary") { + var output = args.STRING.toString(); + return output + .split(" ") + .map((x) => (x = String.fromCharCode(parseInt(x, 2)))) + .join(""); + } + return ""; + } + + negativeReporter(args) { + return args.INPUT * -1; + } + + exponentBlock(args) { + return Math.pow(args.INPUTA, args.INPUTB); + } + + rootBlock(args) { + return Math.pow(args.INPUTB, 1 / args.INPUTA); + } + + normaliseValue(args) { + var input1 = args.INPUT; + var input2 = Math.abs(input1); + var output = input1 / input2; + if (isNaN(output)) return "0"; + return output; + } + + clampNumber(args) { + var input1 = args.INPUTA; + var input2 = args.INPUTB; + var input3 = args.INPUTC; + return Math.min(Math.max(input1, input2), input3); + } + + setVariableTo(args) { + vars["variables"][args.INPUTA] = args.INPUTB; + } + + changeVariableBy(args) { + if (args.INPUTA in vars["variables"]) { + var prev = vars["variables"][args.INPUTA]; + var next = args.INPUTB; + vars["variables"][args.INPUTA] = prev + next; + } else { + vars["variables"][args.INPUTA] = args.INPUTB; + } + } + + getVariable(args) { + if (args.INPUT in vars["variables"]) return vars["variables"][args.INPUT]; + return ""; + } + + deleteVariable(args) { + Reflect.deleteProperty(vars["variables"], args.INPUT); + } + + deleteAllVariables() { + Reflect.deleteProperty(vars, "variables"); + vars["variables"] = {}; + } + + greenFlag(args, util) { + util.runtime.greenFlag(); + } + + setUsername(args, util) { + util.runtime.ioDevices.userData._username = args.INPUT; + } + + setSpriteSVG(args, util) { + try { + Scratch.vm.runtime.renderer.updateSVGSkin( + util.target.sprite.costumes[args.INPUTA - 1].skinId, + args.INPUTB + ); + } catch (error) { + return; + } + Scratch.vm.emitTargetsUpdate(); + } + + alertBlock(args) { + alert(args.STRING); + } + + inputPromptBlock(args) { + return prompt(args.STRING); + } + + confirmationBlock(args) { + return confirm(args.STRING); + } + + goToLink(args) { + Scratch.openWindow(args.INPUT); + } + + redirectToLink(args) { + Scratch.redirect(args.INPUT); + } + + setClipboard(args) { + if (navigator.clipboard && navigator.clipboard.writeText) { + navigator.clipboard.writeText(args.STRING); + } + } + + readClipboard(args) { + if (navigator.clipboard && navigator.clipboard.readText) { + return Scratch.canReadClipboard().then((allowed) => { + if (allowed) { + return navigator.clipboard.readText(); + } + return ""; + }); + } + return ""; + } + + isUserMobile(args, util) { + return navigator.userAgent.includes("Mobile"); + } + + screenReporter(args) { + if (args.DROPDOWN === "width") return screen.width; + if (args.DROPDOWN === "height") return screen.height; + return ""; + } + + windowReporter(args) { + if (args.DROPDOWN === "width") return window.innerWidth; + if (args.DROPDOWN === "height") return window.innerHeight; + return ""; + } + + osBrowserDetails(args) { + var user = navigator.userAgent; + if (args.DROPDOWN === "operating system") { + if (user.includes("Mac OS")) return "macOS"; + if (user.includes("CrOS")) return "ChromeOS"; + if (user.includes("Linux")) return "Linux"; + if (user.includes("Windows")) return "Windows"; + if (user.includes("iPad")) return "iOS"; + if (user.includes("iPod")) return "iOS"; + if (user.includes("iPhone")) return "iOS"; + if (user.includes("Android")) return "Android"; + return "Other"; + } + if (args.DROPDOWN === "browser") { + if (user.includes("Chrome")) return "Chrome"; + if (user.includes("MSIE")) return "Internet Explorer"; + if (user.includes("Firefox")) return "Firefox"; + if (user.includes("Safari")) return "Safari"; + return "Other"; + } + } + + projectURL() { + return window.location.href; + } + + consoleLog(args) { + if (args.DROPDOWN === "log") { + console.log(args.INPUT); + } else if (args.DROPDOWN === "error") { + console.error(args.INPUT); + } else if (args.DROPDOWN === "warn") { + console.warn(args.INPUT); + } + } + + clearConsole() { + console.clear(); + } + + commentHat() { + // no-op + } + + commentCommand() { + // no-op + } + + commentString(args) { + return args.INPUT; + } + + commentBool(args) { + return args.INPUT || false; + } + } + Scratch.extensions.register(new LMSUtils()); +})(Scratch); diff --git a/extensions/Longboost/color_channels.js b/extensions/Longboost/color_channels.js index 71c8aea841..0d0c6e2523 100644 --- a/extensions/Longboost/color_channels.js +++ b/extensions/Longboost/color_channels.js @@ -1,91 +1,97 @@ -(function(Scratch) { - 'use strict'; +// Name: RGB Channels +// ID: lbdrawtest +// Description: Only render or stamp certain RGB channels. + +(function (Scratch) { + "use strict"; const renderer = Scratch.vm.renderer; const gl = renderer._gl; let channel_array = [true, true, true, true]; class LBdrawtest { getInfo() { return { - id: 'lbdrawtest', - name: 'RGB Channels', - menuIconURI: '', - blockIconURI: '', - color1: '#aaaaaa', - color2: '#888888', - color3: '#888888', + id: "lbdrawtest", + name: "RGB Channels", + menuIconURI: + "", + blockIconURI: + "", + color1: "#aaaaaa", + color2: "#888888", + color3: "#888888", blocks: [ { - opcode: 'true', + opcode: "true", blockType: Scratch.BlockType.BOOLEAN, - text: 'true' + text: "true", }, { - opcode: 'false', + opcode: "false", blockType: Scratch.BlockType.BOOLEAN, - text: 'false', - hideFromPalette: true + text: "false", + hideFromPalette: true, }, { - opcode: 'enabledCheck', + opcode: "enabledCheck", blockType: Scratch.BlockType.BOOLEAN, - text: '[COLOR] channel enabled?', + text: "[COLOR] channel enabled?", arguments: { COLOR: { type: Scratch.ArgumentType.STRING, - menu: 'COLOR_MENU' - } - } + menu: "COLOR_MENU", + }, + }, }, { - opcode: 'draw', + opcode: "draw", blockType: Scratch.BlockType.COMMAND, - text: 'only draw colors:[R]green:[G]blue:[B]', + text: "only draw colors:[R]green:[G]blue:[B]", arguments: { R: { - type: Scratch.ArgumentType.BOOLEAN + type: Scratch.ArgumentType.BOOLEAN, }, G: { - type: Scratch.ArgumentType.BOOLEAN + type: Scratch.ArgumentType.BOOLEAN, }, B: { - type: Scratch.ArgumentType.BOOLEAN - } - } + type: Scratch.ArgumentType.BOOLEAN, + }, + }, }, { - opcode: 'drawOneColor', + opcode: "drawOneColor", blockType: Scratch.BlockType.COMMAND, - text: 'only draw [COLOR]', + text: "only draw [COLOR]", arguments: { COLOR: { type: Scratch.ArgumentType.STRING, - menu: 'COLOR_MENU' - } - } + menu: "COLOR_MENU", + }, + }, }, { - opcode: 'drawDepth', + opcode: "drawDepth", blockType: Scratch.BlockType.COMMAND, - text: 'enable depth mask?[DRAW]', + text: "enable depth mask?[DRAW]", hideFromPalette: true, arguments: { DRAW: { - type: Scratch.ArgumentType.BOOLEAN - } - } + type: Scratch.ArgumentType.BOOLEAN, + }, + }, }, { - opcode: 'clearEffects', + opcode: "clearEffects", blockType: Scratch.BlockType.COMMAND, - text: 'clear color draw effects', - } + text: "clear color draw effects", + }, ], menus: { COLOR_MENU: { acceptReporters: true, - items: ['red', 'green', 'blue'] - } - } + items: ["red", "green", "blue"], + }, + }, }; } @@ -97,44 +103,63 @@ return false; } - enabledCheck({COLOR}) { - if ((COLOR == 'red' && channel_array[0]) || (COLOR == 'green' && channel_array[1]) || (COLOR == 'blue' && channel_array[2])) { + enabledCheck({ COLOR }) { + if ( + (COLOR == "red" && channel_array[0]) || + (COLOR == "green" && channel_array[1]) || + (COLOR == "blue" && channel_array[2]) + ) { return true; } else { return false; } } - draw({R, G, B}) { + draw({ R, G, B }) { channel_array = [R, G, B, true]; - gl.colorMask(channel_array[0], channel_array[1], channel_array[2], channel_array[3]); + gl.colorMask( + channel_array[0], + channel_array[1], + channel_array[2], + channel_array[3] + ); Scratch.vm.renderer.dirty = true; } - drawOneColor({COLOR}) { - if (COLOR == 'red') { + drawOneColor({ COLOR }) { + if (COLOR == "red") { channel_array = [true, false, false, true]; - } else if (COLOR == 'green') { + } else if (COLOR == "green") { channel_array = [false, true, false, true]; } else { channel_array = [false, false, true, true]; } - gl.colorMask(channel_array[0], channel_array[1], channel_array[2], channel_array[3]); + gl.colorMask( + channel_array[0], + channel_array[1], + channel_array[2], + channel_array[3] + ); Scratch.vm.renderer.dirty = true; } - drawDepth({DRAW}) { + drawDepth({ DRAW }) { gl.depthMask(DRAW); Scratch.vm.renderer.dirty = true; } clearEffects() { channel_array = [true, true, true, true]; - gl.colorMask(channel_array[0], channel_array[1], channel_array[2], channel_array[3]); + gl.colorMask( + channel_array[0], + channel_array[1], + channel_array[2], + channel_array[3] + ); gl.depthMask(true); Scratch.vm.renderer.dirty = true; } } Scratch.extensions.register(new LBdrawtest()); -})(Scratch); \ No newline at end of file +})(Scratch); diff --git a/extensions/Medericoder/textcase.js b/extensions/Medericoder/textcase.js index 2745ad484c..c08c3577be 100644 --- a/extensions/Medericoder/textcase.js +++ b/extensions/Medericoder/textcase.js @@ -15,332 +15,384 @@ */ (function (Scratch) { - 'use strict'; - function funchangecase (text, format) { - if (format === 'uppercase') { - return text.toString().toUpperCase(); - } else if (format === 'lowercase') { - return text.toString().toLowerCase(); - } else if (format === 'invert') { - let x = ""; - for (let i = 0; i < text.length; i++) { - if (text.toString()[i] === text.toString()[i].toUpperCase()) { - x += text.toString()[i].toLowerCase(); - } else { - x += text.toString()[i].toUpperCase(); - } - } - return x; - } else if (format === 'begin') { - let x = ""; - for (let i = 0; i < text.length; i++) { - if (i == 0) { - x += text.toString()[i].toUpperCase(); - } else { - x += text.toString()[i]; - } - } - return x; - } else if (format === 'begin words') { - let x = ""; - for (let i = 0; i < text.length; i++) { - if (i == 0) { - x += text.toString()[i].toUpperCase(); - } else if (text.toString()[i - 1].toLowerCase() === text.toString()[i - 1].toUpperCase()) { - x += text.toString()[i].toUpperCase(); - } else { - x += text.toString()[i]; - } - } - return x; - } else if (format === 'begin sentences') { - let x = ""; - let mybool = true; - for (let i = 0; i < text.length; i++) { - if (mybool){ - x += text.toString()[i].toUpperCase(); - if (!(text.toString()[i].toLowerCase() === text.toString()[i].toUpperCase())){ - mybool = false; - } - } else if (text.toString()[i] == '.' || text.toString()[i] == '!' || text.toString()[i] == '?') { - x += text.toString()[i]; - mybool = true; - } else { - x += text.toString()[i]; - } - } - return x; - } else if (format === 'begin only') { - let x = ""; - for (let i = 0; i < text.length; i++) { - if (i == 0) { - x += text.toString()[i].toUpperCase(); - } else { - x += text.toString()[i].toLowerCase(); - } - } - return x; - } else if (format === 'begin words only') { - let x = ""; - for (let i = 0; i < text.length; i++) { - if (i == 0) { - x += text.toString()[i].toUpperCase(); - } else if (text.toString()[i - 1].toLowerCase() === text.toString()[i - 1].toUpperCase()) { - x += text.toString()[i].toUpperCase(); - } else { - x += text.toString()[i].toLowerCase(); - } - } - return x; - } else if (format === 'begin sentences only') { - let x = ""; - let mybool = true; - for (let i = 0; i < text.length; i++) { - if (mybool){ - x += text.toString()[i].toUpperCase(); - if (!(text.toString()[i].toLowerCase() === text.toString()[i].toUpperCase())){ - mybool = false; - } - } else if (text.toString()[i] == '.' || text.toString()[i] == '!' || text.toString()[i] == '?') { - x += text.toString()[i]; - mybool = true; - } else { - x += text.toString()[i].toLowerCase(); - } - } - return x; - } else if (format === '1/2 up') { - let x = ""; - for (let i = 0; i < text.length; i++) { - if (i % 2 == 0){ - x += text.toString()[i].toUpperCase(); - } else { - x += text.toString()[i].toLowerCase(); - } - } - return x; - } else if (format === '1/2 low') { - let x = ""; - for (let i = 0; i < text.length; i++) { - if (i % 2 == 1){ - x += text.toString()[i].toUpperCase(); - } else { - x += text.toString()[i].toLowerCase(); - } - } - return x; - } else if (format === '1/2 up letters only') { - let x = ""; - let noletters = 0; - for (let i = 0; i < text.length; i++) { - if (text.toString()[i].toUpperCase() === text.toString()[i].toLowerCase()){ - noletters += 1; - x += text.toString()[i]; - } else if ((i - noletters) % 2 == 0){ - x += text.toString()[i].toUpperCase(); - } else { - x += text.toString()[i].toLowerCase(); - } - } - return x; - } else if (format === '1/2 low letters only') { - let x = ""; - let noletters = 0; - for (let i = 0; i < text.length; i++) { - if (text.toString()[i].toUpperCase() === text.toString()[i].toLowerCase()){ - noletters += 1; - x += text.toString()[i]; - } else if ((i - noletters) % 2 == 1){ - x += text.toString()[i].toUpperCase(); - } else { - x += text.toString()[i].toLowerCase(); - } - } - return x; - } else if (format === 'random') { - let x = ""; - for (let i = 0; i < text.length; i++) { - if (Math.random() < 0.5){ - x += text.toString()[i].toUpperCase(); - } else { - x += text.toString()[i].toLowerCase(); - } - } - return x; + "use strict"; + function funchangecase(text, format) { + if (format === "uppercase") { + return text.toString().toUpperCase(); + } else if (format === "lowercase") { + return text.toString().toLowerCase(); + } else if (format === "invert") { + let x = ""; + for (let i = 0; i < text.length; i++) { + if (text.toString()[i] === text.toString()[i].toUpperCase()) { + x += text.toString()[i].toLowerCase(); } else { - return text.toString(); + x += text.toString()[i].toUpperCase(); } - } - - function fungetcase (args) { - var low = 0; - var up = 0; - for (let i = 0; i < args.TEXT.length; i++) { - if (args.TEXT.toString()[i].toLowerCase() === args.TEXT.toString()[i].toUpperCase()) { - low += 0; - up += 0; - } else if (args.TEXT.toString()[i].toLowerCase() === args.TEXT.toString()[i]) { - low += 1; - } else { - up += 1; - } + } + return x; + } else if (format === "begin") { + let x = ""; + for (let i = 0; i < text.length; i++) { + if (i == 0) { + x += text.toString()[i].toUpperCase(); + } else { + x += text.toString()[i]; + } + } + return x; + } else if (format === "begin words") { + let x = ""; + for (let i = 0; i < text.length; i++) { + if (i == 0) { + x += text.toString()[i].toUpperCase(); + } else if ( + text.toString()[i - 1].toLowerCase() === + text.toString()[i - 1].toUpperCase() + ) { + x += text.toString()[i].toUpperCase(); + } else { + x += text.toString()[i]; + } + } + return x; + } else if (format === "begin sentences") { + let x = ""; + let mybool = true; + for (let i = 0; i < text.length; i++) { + if (mybool) { + x += text.toString()[i].toUpperCase(); + if ( + !( + text.toString()[i].toLowerCase() === + text.toString()[i].toUpperCase() + ) + ) { + mybool = false; + } + } else if ( + text.toString()[i] == "." || + text.toString()[i] == "!" || + text.toString()[i] == "?" + ) { + x += text.toString()[i]; + mybool = true; + } else { + x += text.toString()[i]; + } + } + return x; + } else if (format === "begin only") { + let x = ""; + for (let i = 0; i < text.length; i++) { + if (i == 0) { + x += text.toString()[i].toUpperCase(); + } else { + x += text.toString()[i].toLowerCase(); + } + } + return x; + } else if (format === "begin words only") { + let x = ""; + for (let i = 0; i < text.length; i++) { + if (i == 0) { + x += text.toString()[i].toUpperCase(); + } else if ( + text.toString()[i - 1].toLowerCase() === + text.toString()[i - 1].toUpperCase() + ) { + x += text.toString()[i].toUpperCase(); + } else { + x += text.toString()[i].toLowerCase(); + } + } + return x; + } else if (format === "begin sentences only") { + let x = ""; + let mybool = true; + for (let i = 0; i < text.length; i++) { + if (mybool) { + x += text.toString()[i].toUpperCase(); + if ( + !( + text.toString()[i].toLowerCase() === + text.toString()[i].toUpperCase() + ) + ) { + mybool = false; + } + } else if ( + text.toString()[i] == "." || + text.toString()[i] == "!" || + text.toString()[i] == "?" + ) { + x += text.toString()[i]; + mybool = true; + } else { + x += text.toString()[i].toLowerCase(); + } + } + return x; + } else if (format === "1/2 up") { + let x = ""; + for (let i = 0; i < text.length; i++) { + if (i % 2 == 0) { + x += text.toString()[i].toUpperCase(); + } else { + x += text.toString()[i].toLowerCase(); + } + } + return x; + } else if (format === "1/2 low") { + let x = ""; + for (let i = 0; i < text.length; i++) { + if (i % 2 == 1) { + x += text.toString()[i].toUpperCase(); + } else { + x += text.toString()[i].toLowerCase(); + } + } + return x; + } else if (format === "1/2 up letters only") { + let x = ""; + let noletters = 0; + for (let i = 0; i < text.length; i++) { + if ( + text.toString()[i].toUpperCase() === text.toString()[i].toLowerCase() + ) { + noletters += 1; + x += text.toString()[i]; + } else if ((i - noletters) % 2 == 0) { + x += text.toString()[i].toUpperCase(); + } else { + x += text.toString()[i].toLowerCase(); } - if (up == 0 && low == 0) { - return ''; - } else if (up > 0 && low > 0) { - if (up == low) { - return 'mixed'; - } else if (up > low) { - return 'mixed (upper)'; - } else { - return 'mixed (lower)'; - } - } else if (up > low) { - return 'upper'; + } + return x; + } else if (format === "1/2 low letters only") { + let x = ""; + let noletters = 0; + for (let i = 0; i < text.length; i++) { + if ( + text.toString()[i].toUpperCase() === text.toString()[i].toLowerCase() + ) { + noletters += 1; + x += text.toString()[i]; + } else if ((i - noletters) % 2 == 1) { + x += text.toString()[i].toUpperCase(); } else { - return 'lower'; + x += text.toString()[i].toLowerCase(); } + } + return x; + } else if (format === "random") { + let x = ""; + for (let i = 0; i < text.length; i++) { + if (Math.random() < 0.5) { + x += text.toString()[i].toUpperCase(); + } else { + x += text.toString()[i].toLowerCase(); + } + } + return x; + } else { + return text.toString(); } + } - class TextCase { - getInfo() { - return { - id: 'medericodertextcase', - name: 'Text Case', - blocks: [ - { - opcode: 'strictlyequal', - blockType: Scratch.BlockType.BOOLEAN, - text: '[TEXT1] ≡ [TEXT2]', - arguments: { - TEXT1: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'Hello' - }, - TEXT2: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'hello' - } - } - }, - { - opcode: 'quasiequal', - blockType: Scratch.BlockType.BOOLEAN, - text: '[TEXT1] ≈ [TEXT2]', - arguments: { - TEXT1: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'Hello' - }, - TEXT2: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'hello' - } - } - }, - '---', - { - opcode: 'changecase', - blockType: Scratch.BlockType.REPORTER, - text: 'convert [TEXT] to case [FORMAT]', - arguments: { - TEXT: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'Where is the apple? It is here!' - }, - FORMAT: { - type: Scratch.ArgumentType.STRING, - menu: 'FORMAT_MENU' - } - } - }, - { - opcode: 'getcase', - blockType: Scratch.BlockType.REPORTER, - text: 'get case from [TEXT]', - arguments: { - TEXT: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'hello world' - } - } - }, - { - opcode: 'iscase', - blockType: Scratch.BlockType.BOOLEAN, - text: 'is [TEXT] [FORMAT]', - arguments: { - TEXT: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'hello world' - }, - FORMAT: { - type: Scratch.ArgumentType.STRING, - menu: 'FORMAT_MENU' - } - } - }, - '---', - { - opcode: 'glitch', - blockType: Scratch.BlockType.REPORTER, - text: 'glitch [TEXT] level [PROBA]%', - arguments: { - TEXT: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'Hello' - }, - PROBA: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: 10 - } - } - } - ], - menus: { - FORMAT_MENU: { - acceptReporters: true, - items: ['uppercase', 'lowercase', 'invert', 'begin', 'begin words', 'begin sentences', 'begin only', 'begin words only', 'begin sentences only', '1/2 up', '1/2 low', '1/2 up letters only', '1/2 low letters only', 'random', 'identity'] - } - } - }; - } + function fungetcase(args) { + var low = 0; + var up = 0; + for (let i = 0; i < args.TEXT.length; i++) { + if ( + args.TEXT.toString()[i].toLowerCase() === + args.TEXT.toString()[i].toUpperCase() + ) { + low += 0; + up += 0; + } else if ( + args.TEXT.toString()[i].toLowerCase() === args.TEXT.toString()[i] + ) { + low += 1; + } else { + up += 1; + } + } + if (up == 0 && low == 0) { + return ""; + } else if (up > 0 && low > 0) { + if (up == low) { + return "mixed"; + } else if (up > low) { + return "mixed (upper)"; + } else { + return "mixed (lower)"; + } + } else if (up > low) { + return "upper"; + } else { + return "lower"; + } + } - changecase (args) { - return funchangecase(args.TEXT, args.FORMAT); - } + class TextCase { + getInfo() { + return { + id: "medericodertextcase", + name: "Text Case", + blocks: [ + { + opcode: "strictlyequal", + blockType: Scratch.BlockType.BOOLEAN, + text: "[TEXT1] ≡ [TEXT2]", + arguments: { + TEXT1: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Hello", + }, + TEXT2: { + type: Scratch.ArgumentType.STRING, + defaultValue: "hello", + }, + }, + }, + { + opcode: "quasiequal", + blockType: Scratch.BlockType.BOOLEAN, + text: "[TEXT1] ≈ [TEXT2]", + arguments: { + TEXT1: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Hello", + }, + TEXT2: { + type: Scratch.ArgumentType.STRING, + defaultValue: "hello", + }, + }, + }, + "---", + { + opcode: "changecase", + blockType: Scratch.BlockType.REPORTER, + text: "convert [TEXT] to case [FORMAT]", + arguments: { + TEXT: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Where is the apple? It is here!", + }, + FORMAT: { + type: Scratch.ArgumentType.STRING, + menu: "FORMAT_MENU", + }, + }, + }, + { + opcode: "getcase", + blockType: Scratch.BlockType.REPORTER, + text: "get case from [TEXT]", + arguments: { + TEXT: { + type: Scratch.ArgumentType.STRING, + defaultValue: "hello world", + }, + }, + }, + { + opcode: "iscase", + blockType: Scratch.BlockType.BOOLEAN, + text: "is [TEXT] [FORMAT]", + arguments: { + TEXT: { + type: Scratch.ArgumentType.STRING, + defaultValue: "hello world", + }, + FORMAT: { + type: Scratch.ArgumentType.STRING, + menu: "FORMAT_MENU", + }, + }, + }, + "---", + { + opcode: "glitch", + blockType: Scratch.BlockType.REPORTER, + text: "glitch [TEXT] level [PROBA]%", + arguments: { + TEXT: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Hello", + }, + PROBA: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: 10, + }, + }, + }, + ], + menus: { + FORMAT_MENU: { + acceptReporters: true, + items: [ + "uppercase", + "lowercase", + "invert", + "begin", + "begin words", + "begin sentences", + "begin only", + "begin words only", + "begin sentences only", + "1/2 up", + "1/2 low", + "1/2 up letters only", + "1/2 low letters only", + "random", + "identity", + ], + }, + }, + }; + } - getcase (args) { - return fungetcase(args); - } + changecase(args) { + return funchangecase(args.TEXT, args.FORMAT); + } - iscase (args) { - if (args.FORMAT == 'random'){ - return fungetcase(args).includes('mixed'); - } else { - return args.TEXT.toString() === funchangecase(args.TEXT, args.FORMAT); - } - } + getcase(args) { + return fungetcase(args); + } - strictlyequal (args) { - return args.TEXT1 === args.TEXT2; - } + iscase(args) { + if (args.FORMAT == "random") { + return fungetcase(args).includes("mixed"); + } else { + return args.TEXT.toString() === funchangecase(args.TEXT, args.FORMAT); + } + } - quasiequal (args) { - return args.TEXT1.toString().toLowerCase() === args.TEXT2.toString().toLowerCase(); - } + strictlyequal(args) { + return args.TEXT1 === args.TEXT2; + } + + quasiequal(args) { + return ( + args.TEXT1.toString().toLowerCase() === + args.TEXT2.toString().toLowerCase() + ); + } - glitch (args) { - var x = ''; - for (let i = 0; i < args.TEXT.toString().length; i++){ - if (Math.random() * 100 < args.PROBA){ - x += funchangecase(args.TEXT.toString()[i],'invert'); - } else { - x += args.TEXT.toString()[i]; - } - } - return x; + glitch(args) { + var x = ""; + for (let i = 0; i < args.TEXT.toString().length; i++) { + if (Math.random() * 100 < args.PROBA) { + x += funchangecase(args.TEXT.toString()[i], "invert"); + } else { + x += args.TEXT.toString()[i]; } + } + return x; } - Scratch.extensions.register(new TextCase()); + } + Scratch.extensions.register(new TextCase()); })(Scratch); diff --git a/extensions/NOname-awa/cn-number.js b/extensions/NOname-awa/cn-number.js index de029ed77d..a3db7e1208 100644 --- a/extensions/NOname-awa/cn-number.js +++ b/extensions/NOname-awa/cn-number.js @@ -1,32 +1,46 @@ (function (Scratch) { - 'use strict'; + "use strict"; - let Number2, null2, units, Number_in, uppercase, i, N_Z, o, j, After_decimal_point, unit, k, C_Number, m, n; + let Number2, + null2, + units, + Number_in, + uppercase, + i, + N_Z, + o, + j, + After_decimal_point, + unit, + k, + C_Number, + m, + n; class CNNUMBER { getInfo() { return { - id: 'nonameawacnnumber', - name: '中文数字', - color1: '#cb0000', - color2: '#a20000', - color3: '#a20000', + id: "nonameawacnnumber", + name: "中文数字", + color1: "#cb0000", + color2: "#a20000", + color3: "#a20000", blocks: [ { - opcode: 'CN_number', + opcode: "CN_number", blockType: Scratch.BlockType.REPORTER, disableMonitor: true, - text: '中文数字 [a] 大写 [u]', + text: "中文数字 [a] 大写 [u]", arguments: { a: { type: Scratch.ArgumentType.NUMBER, - defaultValue: '2020' + defaultValue: "2020", }, u: { type: Scratch.ArgumentType.STRING, - menu: 'uppercase' + menu: "uppercase", }, - } + }, }, ], menus: { @@ -34,16 +48,16 @@ acceptReporters: true, items: [ { - text: '关闭', - value: '0' + text: "关闭", + value: "0", }, { - text: '打开', - value: '1' - } - ] - } - } + text: "打开", + value: "1", + }, + ], + }, + }, }; } CN_number(args) { @@ -54,12 +68,11 @@ } } - const main = (Number2, null2, units) => { if (Number2 == 0) { - return null2 ? '零' : ''; + return null2 ? "零" : ""; } - i = ''; + i = ""; let j_start = String(Number2).length; let j_inc = 1; if (j_start > 1) { @@ -67,18 +80,51 @@ } for (j = j_start; j_inc >= 0 ? j <= 1 : j >= 1; j += j_inc) { if (units) { - k = String(Number2).charAt(((String(Number2).length - (j - 1)) - 1)); + k = String(Number2).charAt(String(Number2).length - (j - 1) - 1); if (Number2 >= 10000) { - return ''; + return ""; } - i = String(i) + String(String(Number2).slice((-j)).charAt(0) == '2' && j >= 3 ? '两' : (k == '0' ? (j == 1 ? '' : (i.slice(-1) == '零' ? '' : (String(Number2).slice(-1) == '0' && (j != 3 || String(Number2).charAt(2) == '0') ? ' ' : '零'))) : C_Number[(k - 1)])); - i = String(i) + String(i.slice(-1) == '零' || i.slice(-1) == ' ' ? '' : (unit[(j - 1)] == '个' ? '' : unit[(j - 1)])); + i = + String(i) + + String( + String(Number2).slice(-j).charAt(0) == "2" && j >= 3 + ? "两" + : k == "0" + ? j == 1 + ? "" + : i.slice(-1) == "零" + ? "" + : String(Number2).slice(-1) == "0" && + (j != 3 || String(Number2).charAt(2) == "0") + ? " " + : "零" + : C_Number[k - 1] + ); + i = + String(i) + + String( + i.slice(-1) == "零" || i.slice(-1) == " " + ? "" + : unit[j - 1] == "个" + ? "" + : unit[j - 1] + ); } else { if (j != 1) { - k = String(Number2).slice((-(j - 1))).charAt(0); - i = String(i) + String(k == 0 ? '零' : C_Number[([k].reduce(function (x, y) { - return x + y; - }) - 1)]); + k = String(Number2) + .slice(-(j - 1)) + .charAt(0); + i = + String(i) + + String( + k == 0 + ? "零" + : C_Number[ + [k].reduce(function (x, y) { + return x + y; + }) - 1 + ] + ); } } } @@ -89,33 +135,57 @@ }; const CN_NUMBER = (Number_in, uppercase) => { - if (String(Math.abs(Number_in >= 0 ? Math.floor(Number_in) : Math.ceil(Number_in))).length > 20) { - return ''; + if ( + String( + Math.abs(Number_in >= 0 ? Math.floor(Number_in) : Math.ceil(Number_in)) + ).length > 20 + ) { + return ""; } - N_Z = Math.abs(Number_in >= 0 ? Math.floor(Number_in) : Math.ceil(Number_in)); - After_decimal_point = '.' + String(String(Number_in).split('.')[1]); - unit = (uppercase ? '个、拾、佰、仟、万、亿、兆、京' : '个、十、百、千、万、亿、兆、京').split('、'); - C_Number = (uppercase ? '壹、贰、叁、肆、伍、陆、柒、捌、玖、拾' : '一、二、三、四、五、六、七、八、九、十').split('、'); - m = ''; - o = ''; + N_Z = Math.abs( + Number_in >= 0 ? Math.floor(Number_in) : Math.ceil(Number_in) + ); + After_decimal_point = "." + String(String(Number_in).split(".")[1]); + unit = ( + uppercase + ? "个、拾、佰、仟、万、亿、兆、京" + : "个、十、百、千、万、亿、兆、京" + ).split("、"); + C_Number = ( + uppercase + ? "壹、贰、叁、肆、伍、陆、柒、捌、玖、拾" + : "一、二、三、四、五、六、七、八、九、十" + ).split("、"); + m = ""; + o = ""; let n_end = String(N_Z).length; let n_inc = 1; if (1 > n_end) { n_inc = -n_inc; } for (n = 1; n_inc >= 0 ? n <= n_end : n >= n_end; n += n_inc) { - m = [n % 4 == 0 ? ' ' : '', String(N_Z).slice((-n)).charAt(0), m].join(''); + m = [n % 4 == 0 ? " " : "", String(N_Z).slice(-n).charAt(0), m].join(""); } - m = m.trim().split(' '); + m = m.trim().split(" "); let n_end2 = m.length; let n_inc2 = 1; if (1 > n_end2) { n_inc2 = -n_inc2; } for (n = 1; n_inc2 >= 0 ? n <= n_end2 : n >= n_end2; n += n_inc2) { - o = [m.length == n ? '' : unit[((n + 4) - 1)], main(m.slice((-n))[0], m.length == n, true), o].join(''); + o = [ + m.length == n ? "" : unit[n + 4 - 1], + main(m.slice(-n)[0], m.length == n, true), + o, + ].join(""); } - return [Number_in < 0 ? '负' : '', o, Number_in % 1 != 0 ? '点' + String(main(After_decimal_point, false, false)) : ''].join(''); + return [ + Number_in < 0 ? "负" : "", + o, + Number_in % 1 != 0 + ? "点" + String(main(After_decimal_point, false, false)) + : "", + ].join(""); }; Scratch.extensions.register(new CNNUMBER()); diff --git a/extensions/NOname-awa/global-coordinate.js b/extensions/NOname-awa/global-coordinate.js index d546845192..3613073ce8 100644 --- a/extensions/NOname-awa/global-coordinate.js +++ b/extensions/NOname-awa/global-coordinate.js @@ -1,474 +1,511 @@ (function (Scratch) { - 'use strict'; - const icon = ''; - const tr_icon = ''; - const tl_icon = ''; - let gx = [0]; - let gy = [0]; - let gr = [90]; - let gs = [100]; - let rm = [0]; - class Global_Coordinate { - getInfo() { - return { - id: 'globalCoordinate', - color1: '#2ea4a4', - menuIconURI: icon, - name: 'Global Coordinate', - blocks: [ - { - opcode: 'SET', - filter: [Scratch.TargetType.SPRITE], - blockType: Scratch.BlockType.COMMAND, - blockIconURI: icon, - text: 'go to x: [x] y: [y] direction [r] size [s] - use screens [screen]', - arguments: { - x: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - y: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - r: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '90' - }, - s: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '100' - }, - screen: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '1' - }, - } - }, - { - opcode: 'rotation_mode', - filter: [Scratch.TargetType.SPRITE], - blockType: Scratch.BlockType.COMMAND, - text: 'set screens [screen] \'s rotation mode to [m]', - arguments: { - screen: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '1' - }, - m: { - type: Scratch.ArgumentType.NUMBER, - menu: 'rotation_mode' - } - } - }, - { - opcode: 'set', - blockType: Scratch.BlockType.COMMAND, - text: 'set screens [screen] \'s x: [x] y: [y] direction: [r] size: [s]', - arguments: { - x: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - y: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - r: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '90' - }, - s: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '100' - }, - screen: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '1' - }, - } - }, - '---', - { - opcode: 'Set_Co', - blockType: Scratch.BlockType.COMMAND, - text: 'set screens [screen] \'s x [x] y: [y]', - arguments: { - x: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - y: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - screen: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '1' - }, - } - }, - { - opcode: 'Set_GX', - blockType: Scratch.BlockType.COMMAND, - text: 'set screens [screen] \'s x to [x]', - arguments: { - x: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - screen: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '1' - }, - } - }, - { - opcode: 'Set_GY', - blockType: Scratch.BlockType.COMMAND, - text: 'set screens [screen] \'s y to: [y]', - arguments: { - y: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - screen: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '1' - }, - } - }, - '---', - { - opcode: 'CX', - blockType: Scratch.BlockType.COMMAND, - text: 'change screens [screen] \'s x by [x]', - arguments: { - x: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '10' - }, - screen: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '1' - }, - } - }, - { - opcode: 'CY', - blockType: Scratch.BlockType.COMMAND, - text: 'change screens [screen] \'s y by [y]', - arguments: { - y: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '10' - }, - screen: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '1' - }, - } - }, - '---', - { - opcode: 'Set_GR', - blockType: Scratch.BlockType.COMMAND, - text: 'set screens [screen] \'s direction [r]', - arguments: { - r: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '90' - }, - screen: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '1' - }, - } - }, - { - opcode: 'TR', - blockType: Scratch.BlockType.COMMAND, - text: 'turn [tr_icon] [r] degrees - screens [screen]', - arguments: { - r: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '15' - }, - tr_icon: { - type: Scratch.ArgumentType.IMAGE, - dataURI: tr_icon - }, - screen: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '1' - }, - } - }, - { - opcode: 'TL', - blockType: Scratch.BlockType.COMMAND, - text: 'turn [tr_icon] [r] degrees - screens [screen]', - arguments: { - r: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '15' - }, - tr_icon: { - type: Scratch.ArgumentType.IMAGE, - dataURI: tl_icon - }, - screen: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '1' - }, - } - }, - '---', - { - opcode: 'Set_si', - blockType: Scratch.BlockType.COMMAND, - text: 'set screens [screen] \'s size [s]', - arguments: { - s: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '100' - }, - screen: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '1' - }, - } - }, - { - opcode: 'CS', - blockType: Scratch.BlockType.COMMAND, - text: 'change screens [screen] \'s size by [s]', - arguments: { - s: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '10' - }, - screen: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '1' - }, - } - }, - '---', - { - opcode: 'x', - blockType: Scratch.BlockType.REPORTER, - text: 'screens [screen] x', - arguments: { - screen: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '1' - }, - } - }, - { - opcode: 'y', - blockType: Scratch.BlockType.REPORTER, - text: 'screens [screen] y', - arguments: { - screen: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '1' - }, - } - }, - { - opcode: 'r', - blockType: Scratch.BlockType.REPORTER, - text: 'screens [screen] direction', - arguments: { - screen: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '1' - }, - } - }, - { - opcode: 's', - blockType: Scratch.BlockType.REPORTER, - text: 'screens [screen] size', - arguments: { - screen: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '1' - }, - } - }, - { - opcode: 'rm', - blockType: Scratch.BlockType.BOOLEAN, - text: 'screens [screen] rotation mode is screen?', - arguments: { - screen: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '1' - }, - } - }, - ], - menus: { - rotation_mode: { - acceptReporters: true, - items: [ - { - text: 'center of stage', - value: '0' - }, - { - text: 'center of screen', - value: '1' - } - ] - } - } - }; - } - SET(args, { target }) { - if (isNaN(args.x)) { - args.x = 0; - } - if (isNaN(args.y)) { - args.y = 0; - } - if (isNaN(args.r)) { - args.r = 0; - } - if (isNaN(args.s)) { - args.s = 0; - } - target.setSize((args.s / 100) * gs[args.screen - 1]); - target.setDirection((- ((args.r - (gr[args.screen - 1] - 90) - 90))) + 90); - if (rm[args.screen - 1] == 1) { - target.setXY( - (gx[args.screen - 1] / 100 * gs[args.screen - 1]) + (gs[args.screen - 1] / 100) * ((- (args.x)) * Math.cos((((- ((gr[args.screen - 1] - 90 - 90))) + 90)) / 180 * Math.PI) - (- (args.y)) * Math.sin((((- ((gr[args.screen - 1] - 90 - 90))) + 90)) / 180 * Math.PI)), - (gy[args.screen - 1] / 100 * gs[args.screen - 1]) + (gs[args.screen - 1] / 100) * ((- (args.x)) * Math.sin((((- ((gr[args.screen - 1] - 90 - 90))) + 90)) / 180 * Math.PI) + (- (args.y)) * Math.cos((((- ((gr[args.screen - 1] - 90 - 90))) + 90)) / 180 * Math.PI)), - true - ); - } else { - target.setXY( - (gs[args.screen - 1] / 100) * ((- (args.x + gx[args.screen - 1])) * Math.cos((((- ((gr[args.screen - 1] - 90 - 90))) + 90)) / 180 * Math.PI) - (- (args.y + gy[args.screen - 1])) * Math.sin((((- ((gr[args.screen - 1] - 90 - 90))) + 90)) / 180 * Math.PI)), - (gs[args.screen - 1] / 100) * ((- (args.x + gx[args.screen - 1])) * Math.sin((((- ((gr[args.screen - 1] - 90 - 90))) + 90)) / 180 * Math.PI) + (- (args.y + gy[args.screen - 1])) * Math.cos((((- ((gr[args.screen - 1] - 90 - 90))) + 90)) / 180 * Math.PI)), - true - ); - } - } - rotation_mode(args) { - rm[args.screen - 1] = args.m; - } - set({ x, y, r, s, screen }) { - if (!isNaN(x)) { - gx[screen - 1] = x; - } else { - gx[screen - 1] = 0; - } - if (!isNaN(y)) { - gy[screen - 1] = y; - } else { - gy[screen - 1] = 0; - } - if (!isNaN(r)) { - gr[screen - 1] = r; - } else { - gr[screen - 1] = 0; - } - if (!isNaN(s)) { - gs[screen - 1] = s; - } else { - gs[screen - 1] = 0; - } - } - Set_Co({ x, y, screen }) { - if (!isNaN(x + y)) { - gx[screen - 1] = x; - gy[screen - 1] = y; - } else { - if (isNaN(x)) { - x = 0; - } - if (isNaN(y)) { - y = 0; - } - } - } - Set_GX({ x, screen }) { - if (!isNaN(x)) { - gx[screen - 1] = x; - } else { - gx[screen - 1] = 0; - } - } - Set_GY({ y, screen }) { - if (!isNaN(y)) { - gy[screen - 1] = y; - } else { - gy[screen - 1] = 0; - } - } - CX({ x, screen }) { - if (!isNaN(x)) { - gx[screen - 1] += x; - } - } - CY({ y, screen }) { - if (!isNaN(y)) { - gy[screen - 1] += y; - } - } - Set_GR({ r, screen }) { - if (!isNaN(r)) { - gr[screen - 1] = r; - } else { - gr[screen - 1] = 0; - } - } - TR({ r, screen }) { - if (!isNaN(r)) { - gr[screen - 1] += r; - } - } - TL({ r, screen }) { - if (!isNaN(r)) { - gr[screen - 1] -= r; - } - } - Set_si({ s, screen }) { - if (!isNaN(s)) { - gs[screen - 1] = s; - } else { - gs[screen - 1] = 0; - } - if (gs[screen - 1] < 0) { - gs[screen - 1] = 0; - } - } - CS({ s, screen }) { - if (!isNaN(s)) { - gs[screen - 1] += s; - } - if (gs[screen - 1] < 0) { - gs[screen - 1] = 0; - } - } - x({ screen }) { - return gx[screen - 1]; - } - y({ screen }) { - return gy[screen - 1]; - } - r({ screen }) { - return gr[screen - 1]; - } - s({ screen }) { - return gs[screen - 1]; + "use strict"; + const icon = + ""; + const tr_icon = + ""; + const tl_icon = + ""; + let gx = [0]; + let gy = [0]; + let gr = [90]; + let gs = [100]; + let rm = [0]; + class Global_Coordinate { + getInfo() { + return { + id: "globalCoordinate", + color1: "#2ea4a4", + menuIconURI: icon, + name: "Global Coordinate", + blocks: [ + { + opcode: "SET", + filter: [Scratch.TargetType.SPRITE], + blockType: Scratch.BlockType.COMMAND, + blockIconURI: icon, + text: "go to x: [x] y: [y] direction [r] size [s] - use screens [screen]", + arguments: { + x: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + y: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + r: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "90", + }, + s: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "100", + }, + screen: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "1", + }, + }, + }, + { + opcode: "rotation_mode", + filter: [Scratch.TargetType.SPRITE], + blockType: Scratch.BlockType.COMMAND, + text: "set screens [screen] 's rotation mode to [m]", + arguments: { + screen: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "1", + }, + m: { + type: Scratch.ArgumentType.NUMBER, + menu: "rotation_mode", + }, + }, + }, + { + opcode: "set", + blockType: Scratch.BlockType.COMMAND, + text: "set screens [screen] 's x: [x] y: [y] direction: [r] size: [s]", + arguments: { + x: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + y: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + r: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "90", + }, + s: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "100", + }, + screen: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "1", + }, + }, + }, + "---", + { + opcode: "Set_Co", + blockType: Scratch.BlockType.COMMAND, + text: "set screens [screen] 's x [x] y: [y]", + arguments: { + x: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + y: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + screen: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "1", + }, + }, + }, + { + opcode: "Set_GX", + blockType: Scratch.BlockType.COMMAND, + text: "set screens [screen] 's x to [x]", + arguments: { + x: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + screen: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "1", + }, + }, + }, + { + opcode: "Set_GY", + blockType: Scratch.BlockType.COMMAND, + text: "set screens [screen] 's y to: [y]", + arguments: { + y: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + screen: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "1", + }, + }, + }, + "---", + { + opcode: "CX", + blockType: Scratch.BlockType.COMMAND, + text: "change screens [screen] 's x by [x]", + arguments: { + x: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "10", + }, + screen: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "1", + }, + }, + }, + { + opcode: "CY", + blockType: Scratch.BlockType.COMMAND, + text: "change screens [screen] 's y by [y]", + arguments: { + y: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "10", + }, + screen: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "1", + }, + }, + }, + "---", + { + opcode: "Set_GR", + blockType: Scratch.BlockType.COMMAND, + text: "set screens [screen] 's direction [r]", + arguments: { + r: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "90", + }, + screen: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "1", + }, + }, + }, + { + opcode: "TR", + blockType: Scratch.BlockType.COMMAND, + text: "turn [tr_icon] [r] degrees - screens [screen]", + arguments: { + r: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "15", + }, + tr_icon: { + type: Scratch.ArgumentType.IMAGE, + dataURI: tr_icon, + }, + screen: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "1", + }, + }, + }, + { + opcode: "TL", + blockType: Scratch.BlockType.COMMAND, + text: "turn [tr_icon] [r] degrees - screens [screen]", + arguments: { + r: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "15", + }, + tr_icon: { + type: Scratch.ArgumentType.IMAGE, + dataURI: tl_icon, + }, + screen: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "1", + }, + }, + }, + "---", + { + opcode: "Set_si", + blockType: Scratch.BlockType.COMMAND, + text: "set screens [screen] 's size [s]", + arguments: { + s: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "100", + }, + screen: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "1", + }, + }, + }, + { + opcode: "CS", + blockType: Scratch.BlockType.COMMAND, + text: "change screens [screen] 's size by [s]", + arguments: { + s: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "10", + }, + screen: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "1", + }, + }, + }, + "---", + { + opcode: "x", + blockType: Scratch.BlockType.REPORTER, + text: "screens [screen] x", + arguments: { + screen: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "1", + }, + }, + }, + { + opcode: "y", + blockType: Scratch.BlockType.REPORTER, + text: "screens [screen] y", + arguments: { + screen: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "1", + }, + }, + }, + { + opcode: "r", + blockType: Scratch.BlockType.REPORTER, + text: "screens [screen] direction", + arguments: { + screen: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "1", + }, + }, + }, + { + opcode: "s", + blockType: Scratch.BlockType.REPORTER, + text: "screens [screen] size", + arguments: { + screen: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "1", + }, + }, + }, + { + opcode: "rm", + blockType: Scratch.BlockType.BOOLEAN, + text: "screens [screen] rotation mode is screen?", + arguments: { + screen: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "1", + }, + }, + }, + ], + menus: { + rotation_mode: { + acceptReporters: true, + items: [ + { + text: "center of stage", + value: "0", + }, + { + text: "center of screen", + value: "1", + }, + ], + }, + }, + }; + } + SET(args, { target }) { + if (isNaN(args.x)) { + args.x = 0; + } + if (isNaN(args.y)) { + args.y = 0; + } + if (isNaN(args.r)) { + args.r = 0; + } + if (isNaN(args.s)) { + args.s = 0; + } + target.setSize((args.s / 100) * gs[args.screen - 1]); + target.setDirection(-(args.r - (gr[args.screen - 1] - 90) - 90) + 90); + if (rm[args.screen - 1] == 1) { + target.setXY( + (gx[args.screen - 1] / 100) * gs[args.screen - 1] + + (gs[args.screen - 1] / 100) * + (-args.x * + Math.cos( + ((-(gr[args.screen - 1] - 90 - 90) + 90) / 180) * Math.PI + ) - + -args.y * + Math.sin( + ((-(gr[args.screen - 1] - 90 - 90) + 90) / 180) * Math.PI + )), + (gy[args.screen - 1] / 100) * gs[args.screen - 1] + + (gs[args.screen - 1] / 100) * + (-args.x * + Math.sin( + ((-(gr[args.screen - 1] - 90 - 90) + 90) / 180) * Math.PI + ) + + -args.y * + Math.cos( + ((-(gr[args.screen - 1] - 90 - 90) + 90) / 180) * Math.PI + )), + true + ); + } else { + target.setXY( + (gs[args.screen - 1] / 100) * + (-(args.x + gx[args.screen - 1]) * + Math.cos( + ((-(gr[args.screen - 1] - 90 - 90) + 90) / 180) * Math.PI + ) - + -(args.y + gy[args.screen - 1]) * + Math.sin( + ((-(gr[args.screen - 1] - 90 - 90) + 90) / 180) * Math.PI + )), + (gs[args.screen - 1] / 100) * + (-(args.x + gx[args.screen - 1]) * + Math.sin( + ((-(gr[args.screen - 1] - 90 - 90) + 90) / 180) * Math.PI + ) + + -(args.y + gy[args.screen - 1]) * + Math.cos( + ((-(gr[args.screen - 1] - 90 - 90) + 90) / 180) * Math.PI + )), + true + ); + } + } + rotation_mode(args) { + rm[args.screen - 1] = args.m; + } + set({ x, y, r, s, screen }) { + if (!isNaN(x)) { + gx[screen - 1] = x; + } else { + gx[screen - 1] = 0; + } + if (!isNaN(y)) { + gy[screen - 1] = y; + } else { + gy[screen - 1] = 0; + } + if (!isNaN(r)) { + gr[screen - 1] = r; + } else { + gr[screen - 1] = 0; + } + if (!isNaN(s)) { + gs[screen - 1] = s; + } else { + gs[screen - 1] = 0; + } + } + Set_Co({ x, y, screen }) { + if (!isNaN(x + y)) { + gx[screen - 1] = x; + gy[screen - 1] = y; + } else { + if (isNaN(x)) { + x = 0; } - rm({ screen }) { - return rm[screen - 1] == 1; + if (isNaN(y)) { + y = 0; } + } + } + Set_GX({ x, screen }) { + if (!isNaN(x)) { + gx[screen - 1] = x; + } else { + gx[screen - 1] = 0; + } + } + Set_GY({ y, screen }) { + if (!isNaN(y)) { + gy[screen - 1] = y; + } else { + gy[screen - 1] = 0; + } + } + CX({ x, screen }) { + if (!isNaN(x)) { + gx[screen - 1] += x; + } + } + CY({ y, screen }) { + if (!isNaN(y)) { + gy[screen - 1] += y; + } + } + Set_GR({ r, screen }) { + if (!isNaN(r)) { + gr[screen - 1] = r; + } else { + gr[screen - 1] = 0; + } + } + TR({ r, screen }) { + if (!isNaN(r)) { + gr[screen - 1] += r; + } + } + TL({ r, screen }) { + if (!isNaN(r)) { + gr[screen - 1] -= r; + } + } + Set_si({ s, screen }) { + if (!isNaN(s)) { + gs[screen - 1] = s; + } else { + gs[screen - 1] = 0; + } + if (gs[screen - 1] < 0) { + gs[screen - 1] = 0; + } + } + CS({ s, screen }) { + if (!isNaN(s)) { + gs[screen - 1] += s; + } + if (gs[screen - 1] < 0) { + gs[screen - 1] = 0; + } + } + x({ screen }) { + return gx[screen - 1]; + } + y({ screen }) { + return gy[screen - 1]; + } + r({ screen }) { + return gr[screen - 1]; + } + s({ screen }) { + return gs[screen - 1]; + } + rm({ screen }) { + return rm[screen - 1] == 1; } - Scratch.extensions.register(new Global_Coordinate()); + } + Scratch.extensions.register(new Global_Coordinate()); })(Scratch); diff --git a/extensions/NOname-awa/graphics2d.js b/extensions/NOname-awa/graphics2d.js index 3eec30f6ed..cd6efa33e6 100644 --- a/extensions/NOname-awa/graphics2d.js +++ b/extensions/NOname-awa/graphics2d.js @@ -1,360 +1,420 @@ -(function (Scratch) { - 'use strict'; - Scratch.translate.setup({ - zh: { - name: '图形 2D', - line_section: '线段([x1],[y1])到([x2],[y2])', - triangle: '三角形([x1],[y1])([x2],[y2])([x3],[y3])的 [CS]', - triangle_s: '三角形 [s1] [s2] [s3] 的面积', - quadrilateral: '四边形([x1],[y1])([x2],[y2])([x3],[y3])([x4],[y4])的 [CS]', - graph: '图形 [graph] 的 [CS]', - round: '[rd] 为 [a] 的圆的 [CS]', - pi: '派', - radius: '半径', - diameter: '直径', - area: '面积', - circumference: '周长', - } - }); - class graph { - getInfo() { - return { - id: 'nonameawagraph', - name: Scratch.translate({ id: 'name', default: 'Graphics 2D' }), - color1: '#ff976c', - color2: '#cc7956', - color3: '#e58861', - blocks: [ - { - opcode: 'line_section', - blockType: Scratch.BlockType.REPORTER, - text: Scratch.translate({ id: 'line_section', default: 'length from ([x1],[y1]) to ([x2],[y2])' }), - arguments: { - x1: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '-100' - }, - y1: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - x2: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - y2: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - } - }, - { - opcode: 'vertical', - blockType: Scratch.BlockType.BOOLEAN, - text: '[a] ⊥ [b]', - arguments: { - a: { - type: Scratch.ArgumentType.ANGLE, - defaultValue: '0' - }, - b: { - type: Scratch.ArgumentType.ANGLE, - defaultValue: '90' - }, - } - }, - '---', - { - opcode: 'triangle', - blockType: Scratch.BlockType.REPORTER, - text: Scratch.translate({ id: 'triangle', default: 'triangle ([x1],[y1]) ([x2],[y2]) ([x3],[y3]) \'s [CS]' }), - arguments: { - x1: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - y1: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '10' - }, - x2: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '10' - }, - y2: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '10' - }, - x3: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '10' - }, - y3: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - CS: { - type: Scratch.ArgumentType.STRING, - menu: 'cs' - }, - }, - }, - { - opcode: 'triangle_s', - blockType: Scratch.BlockType.REPORTER, - text: Scratch.translate({ id: 'triangle_s', default: 'triangle [s1] [s2] [s3] \'s area' }), - arguments: { - s1: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '3' - }, - s2: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '4' - }, - s3: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '5' - }, - }, - }, - { - opcode: 'quadrilateral', - blockType: Scratch.BlockType.REPORTER, - text: Scratch.translate({ id: 'quadrilateral', default: 'quadrangle ([x1],[y1]) ([x2],[y2]) ([x3],[y3]) ([x4],[y4]) \'s [CS]' }), - arguments: { - x1: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - y1: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '10' - }, - x2: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '10' - }, - y2: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '10' - }, - x3: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '10' - }, - y3: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - x4: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - y4: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - CS: { - type: Scratch.ArgumentType.STRING, - menu: 'cs' - }, - }, - }, - { - opcode: 'graph', - blockType: Scratch.BlockType.REPORTER, - text: Scratch.translate({ id: 'graph', default: 'graph [graph] \'s [CS]' }), - arguments: { - graph: { - type: Scratch.ArgumentType.STRING, - defaultValue: '[[0,0], [0,2], [2,4], [4,2], [4,0]]' - }, - CS: { - type: Scratch.ArgumentType.STRING, - menu: 'cs' - }, - }, - }, - '---', - { - opcode: 'round', - blockType: Scratch.BlockType.REPORTER, - text: Scratch.translate({ id: 'round', default: 'circle of [rd][a]\'s [CS]' }), - arguments: { - rd: { - type: Scratch.ArgumentType.STRING, - menu: 'rd' - }, - a: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '10' - }, - CS: { - type: Scratch.ArgumentType.STRING, - menu: 'cs' - }, - }, - }, - '---', - { - opcode: 'pi', - disableMonitor: true, - blockType: Scratch.BlockType.REPORTER, - text: Scratch.translate({ id: 'pi', default: 'pi' }), - }, - ], - menus: { - rd: { - acceptReporters: true, - items: [ - { - text: Scratch.translate({ id: 'radius', default: 'radius' }), - value: 'r' - }, - { - text: Scratch.translate({ id: 'diameter', default: 'diameter' }), - value: 'd' - } - ] - }, - cs: { - acceptReporters: true, - items: [ - { - text: Scratch.translate({ id: 'area', default: 'area' }), - value: 's' - }, - { - text: Scratch.translate({ id: 'circumference', default: 'circumference' }), - value: 'c' - } - ] - } - } - }; - } - line_section(args) { - return Math.sqrt(Math.pow(args.x1 - args.x2, 2) + Math.pow(args.y1 - args.y2, 2)); - } - vertical(args) { - if (isNaN(args.a) || isNaN(args.b)) { - return false; - } else { - return ((args.a - (args.b - 90)) % 180) == 0; - } - } - triangle(args) { - if (args.CS == 's') { - let points = [[args.x1, args.y1], [args.x2, args.y2], [args.x3, args.y3]]; - let area = 0; - let n = points.length; - for (let i = 0; i < n; i++) { - let x1 = points[i][0]; - let y1 = points[i][1]; - let x2 = points[(i + 1) % n][0]; - let y2 = points[(i + 1) % n][1]; - area += x1 * y2; - area -= x2 * y1; - } - area = Math.abs(area) / 2; - return (area); - } - if (args.CS == 'c') { - let i = 0; - i += Math.sqrt(Math.pow(args.x1 - args.x2, 2) + Math.pow(args.y1 - args.y2, 2)); - i += Math.sqrt(Math.pow(args.x2 - args.x3, 2) + Math.pow(args.y2 - args.y3, 2)); - i += Math.sqrt(Math.pow(args.x3 - args.x1, 2) + Math.pow(args.y3 - args.y1, 2)); - return i; - } - return 0; - } - triangle_s(args) { - const s = (args.s1 + args.s2 + args.s3) / 2; - const area = Math.sqrt(s * (s - args.s1) * (s - args.s2) * (s - args.s3)); - return area; - } - quadrilateral(args) { - if (args.CS == 's') { - let points = [[args.x1, args.y1], [args.x2, args.y2], [args.x3, args.y3], [args.x4, args.y4]]; - let area = 0; - let n = points.length; - for (let i = 0; i < n; i++) { - let x1 = points[i][0]; - let y1 = points[i][1]; - let x2 = points[(i + 1) % n][0]; - let y2 = points[(i + 1) % n][1]; - area += x1 * y2; - area -= x2 * y1; - } - area = Math.abs(area) / 2; - return (area); - } - if (args.CS == 'c') { - let i = 0; - i += Math.sqrt(Math.pow(args.x1 - args.x2, 2) + Math.pow(args.y1 - args.y2, 2)); - i += Math.sqrt(Math.pow(args.x2 - args.x3, 2) + Math.pow(args.y2 - args.y3, 2)); - i += Math.sqrt(Math.pow(args.x3 - args.x4, 2) + Math.pow(args.y3 - args.y4, 2)); - i += Math.sqrt(Math.pow(args.x4 - args.x1, 2) + Math.pow(args.y4 - args.y1, 2)); - return i; - } - return 0; - } - graph(args) { - let points; - try { - points = JSON.parse(args.graph); - } catch (error) { - return 0; - } - if (!Array.isArray(points)) { - return 0; - } - let n = points.length; - if (args.CS == 's') { - let area = 0; - for (let i = 0; i < n; i++) { - let x1 = points[i][0]; - let y1 = points[i][1]; - let x2 = points[(i + 1) % n][0]; - let y2 = points[(i + 1) % n][1]; - area += x1 * y2; - area -= x2 * y1; - } - area = Math.abs(area) / 2; - return (area); - } - if (args.CS == 'c') { - let x1, x2, y1, y2; - let j = 0; - j = 0; - var i_end = n - 1; - var i_inc = 1; - if (0 > i_end) { - i_inc = -i_inc; - } - for (let i = 0; i_inc >= 0 ? i <= i_end : i >= i_end; i += i_inc) { - x1 = points[((i + 1) - 1)][0]; - x2 = i == n - 1 ? points[0][0] : points[((i + 2) - 1)][0]; - y1 = points[((i + 1) - 1)][1]; - y2 = i == n - 1 ? points[0][1] : points[((i + 2) - 1)][1]; - j = (typeof j == 'number' ? j : 0) + Math.sqrt(Math.pow(Math.abs(x1 - x2), 2) + Math.pow(Math.abs(y1 - y2), 2)); - } - return j; - } - return 0; - } - round(args) { - if (args.CS == 'c') { - return 2 * Math.PI * (args.rd == 'r' ? args.a : args.a / 2); - } - if (args.CS == 's') { - return Math.PI * ((args.rd == 'r' ? args.a : args.a / 2) ** 2); - } - } - pi() { - return Math.PI; - } - } - Scratch.extensions.register(new graph()); -})(Scratch); +// Name: Graphics 2D +// ID: nonameawagraph +// Description: Blocks to compute lengths, angles, and areas in two dimensions. +// By: NOname-awa + +(function (Scratch) { + "use strict"; + Scratch.translate.setup({ + zh: { + name: "图形 2D", + line_section: "线段([x1],[y1])到([x2],[y2])", + triangle: "三角形([x1],[y1])([x2],[y2])([x3],[y3])的 [CS]", + triangle_s: "三角形 [s1] [s2] [s3] 的面积", + quadrilateral: + "四边形([x1],[y1])([x2],[y2])([x3],[y3])([x4],[y4])的 [CS]", + graph: "图形 [graph] 的 [CS]", + round: "[rd] 为 [a] 的圆的 [CS]", + pi: "派", + radius: "半径", + diameter: "直径", + area: "面积", + circumference: "周长", + }, + }); + class graph { + getInfo() { + return { + id: "nonameawagraph", + name: Scratch.translate({ id: "name", default: "Graphics 2D" }), + color1: "#ff976c", + color2: "#cc7956", + color3: "#e58861", + blocks: [ + { + opcode: "line_section", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate({ + id: "line_section", + default: "length from ([x1],[y1]) to ([x2],[y2])", + }), + arguments: { + x1: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "-100", + }, + y1: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + x2: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + y2: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + }, + }, + { + opcode: "vertical", + blockType: Scratch.BlockType.BOOLEAN, + text: "[a] ⊥ [b]", + arguments: { + a: { + type: Scratch.ArgumentType.ANGLE, + defaultValue: "0", + }, + b: { + type: Scratch.ArgumentType.ANGLE, + defaultValue: "90", + }, + }, + }, + "---", + { + opcode: "triangle", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate({ + id: "triangle", + default: "triangle ([x1],[y1]) ([x2],[y2]) ([x3],[y3]) 's [CS]", + }), + arguments: { + x1: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + y1: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "10", + }, + x2: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "10", + }, + y2: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "10", + }, + x3: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "10", + }, + y3: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + CS: { + type: Scratch.ArgumentType.STRING, + menu: "cs", + }, + }, + }, + { + opcode: "triangle_s", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate({ + id: "triangle_s", + default: "triangle [s1] [s2] [s3] 's area", + }), + arguments: { + s1: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "3", + }, + s2: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "4", + }, + s3: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "5", + }, + }, + }, + { + opcode: "quadrilateral", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate({ + id: "quadrilateral", + default: + "quadrangle ([x1],[y1]) ([x2],[y2]) ([x3],[y3]) ([x4],[y4]) 's [CS]", + }), + arguments: { + x1: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + y1: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "10", + }, + x2: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "10", + }, + y2: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "10", + }, + x3: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "10", + }, + y3: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + x4: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + y4: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + CS: { + type: Scratch.ArgumentType.STRING, + menu: "cs", + }, + }, + }, + { + opcode: "graph", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate({ + id: "graph", + default: "graph [graph] 's [CS]", + }), + arguments: { + graph: { + type: Scratch.ArgumentType.STRING, + defaultValue: "[[0,0], [0,2], [2,4], [4,2], [4,0]]", + }, + CS: { + type: Scratch.ArgumentType.STRING, + menu: "cs", + }, + }, + }, + "---", + { + opcode: "round", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate({ + id: "round", + default: "circle of [rd][a]'s [CS]", + }), + arguments: { + rd: { + type: Scratch.ArgumentType.STRING, + menu: "rd", + }, + a: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "10", + }, + CS: { + type: Scratch.ArgumentType.STRING, + menu: "cs", + }, + }, + }, + "---", + { + opcode: "pi", + disableMonitor: true, + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate({ id: "pi", default: "pi" }), + }, + ], + menus: { + rd: { + acceptReporters: true, + items: [ + { + text: Scratch.translate({ id: "radius", default: "radius" }), + value: "r", + }, + { + text: Scratch.translate({ + id: "diameter", + default: "diameter", + }), + value: "d", + }, + ], + }, + cs: { + acceptReporters: true, + items: [ + { + text: Scratch.translate({ id: "area", default: "area" }), + value: "s", + }, + { + text: Scratch.translate({ + id: "circumference", + default: "circumference", + }), + value: "c", + }, + ], + }, + }, + }; + } + line_section(args) { + return Math.sqrt( + Math.pow(args.x1 - args.x2, 2) + Math.pow(args.y1 - args.y2, 2) + ); + } + vertical(args) { + if (isNaN(args.a) || isNaN(args.b)) { + return false; + } else { + return (args.a - (args.b - 90)) % 180 == 0; + } + } + triangle(args) { + if (args.CS == "s") { + let points = [ + [args.x1, args.y1], + [args.x2, args.y2], + [args.x3, args.y3], + ]; + let area = 0; + let n = points.length; + for (let i = 0; i < n; i++) { + let x1 = points[i][0]; + let y1 = points[i][1]; + let x2 = points[(i + 1) % n][0]; + let y2 = points[(i + 1) % n][1]; + area += x1 * y2; + area -= x2 * y1; + } + area = Math.abs(area) / 2; + return area; + } + if (args.CS == "c") { + let i = 0; + i += Math.sqrt( + Math.pow(args.x1 - args.x2, 2) + Math.pow(args.y1 - args.y2, 2) + ); + i += Math.sqrt( + Math.pow(args.x2 - args.x3, 2) + Math.pow(args.y2 - args.y3, 2) + ); + i += Math.sqrt( + Math.pow(args.x3 - args.x1, 2) + Math.pow(args.y3 - args.y1, 2) + ); + return i; + } + return 0; + } + triangle_s(args) { + const s = (args.s1 + args.s2 + args.s3) / 2; + const area = Math.sqrt(s * (s - args.s1) * (s - args.s2) * (s - args.s3)); + return area; + } + quadrilateral(args) { + if (args.CS == "s") { + let points = [ + [args.x1, args.y1], + [args.x2, args.y2], + [args.x3, args.y3], + [args.x4, args.y4], + ]; + let area = 0; + let n = points.length; + for (let i = 0; i < n; i++) { + let x1 = points[i][0]; + let y1 = points[i][1]; + let x2 = points[(i + 1) % n][0]; + let y2 = points[(i + 1) % n][1]; + area += x1 * y2; + area -= x2 * y1; + } + area = Math.abs(area) / 2; + return area; + } + if (args.CS == "c") { + let i = 0; + i += Math.sqrt( + Math.pow(args.x1 - args.x2, 2) + Math.pow(args.y1 - args.y2, 2) + ); + i += Math.sqrt( + Math.pow(args.x2 - args.x3, 2) + Math.pow(args.y2 - args.y3, 2) + ); + i += Math.sqrt( + Math.pow(args.x3 - args.x4, 2) + Math.pow(args.y3 - args.y4, 2) + ); + i += Math.sqrt( + Math.pow(args.x4 - args.x1, 2) + Math.pow(args.y4 - args.y1, 2) + ); + return i; + } + return 0; + } + graph(args) { + let points; + try { + points = JSON.parse(args.graph); + } catch (error) { + return 0; + } + if (!Array.isArray(points)) { + return 0; + } + let n = points.length; + if (args.CS == "s") { + let area = 0; + for (let i = 0; i < n; i++) { + let x1 = points[i][0]; + let y1 = points[i][1]; + let x2 = points[(i + 1) % n][0]; + let y2 = points[(i + 1) % n][1]; + area += x1 * y2; + area -= x2 * y1; + } + area = Math.abs(area) / 2; + return area; + } + if (args.CS == "c") { + let x1, x2, y1, y2; + let j = 0; + j = 0; + var i_end = n - 1; + var i_inc = 1; + if (0 > i_end) { + i_inc = -i_inc; + } + for (let i = 0; i_inc >= 0 ? i <= i_end : i >= i_end; i += i_inc) { + x1 = points[i + 1 - 1][0]; + x2 = i == n - 1 ? points[0][0] : points[i + 2 - 1][0]; + y1 = points[i + 1 - 1][1]; + y2 = i == n - 1 ? points[0][1] : points[i + 2 - 1][1]; + j = + (typeof j == "number" ? j : 0) + + Math.sqrt( + Math.pow(Math.abs(x1 - x2), 2) + Math.pow(Math.abs(y1 - y2), 2) + ); + } + return j; + } + return 0; + } + round(args) { + if (args.CS == "c") { + return 2 * Math.PI * (args.rd == "r" ? args.a : args.a / 2); + } + if (args.CS == "s") { + return Math.PI * (args.rd == "r" ? args.a : args.a / 2) ** 2; + } + } + pi() { + return Math.PI; + } + } + Scratch.extensions.register(new graph()); +})(Scratch); diff --git a/extensions/NOname-awa/math-and-string.js b/extensions/NOname-awa/math-and-string.js index 7ea26b9b7a..2793a40fc8 100644 --- a/extensions/NOname-awa/math-and-string.js +++ b/extensions/NOname-awa/math-and-string.js @@ -1,1123 +1,1202 @@ (function (Scratch) { - 'use strict'; - class MathAndString { - getInfo() { - return { - color1: '#5ac900', - color2: '#48a100', - color3: '#48a100', - id: 'nonameawamathandstring', - name: 'Math And String', - blocks: [ - { - opcode: 'exponent', - blockType: Scratch.BlockType.REPORTER, - text: '[A] ^ [B]', - arguments: { - A: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '2' - }, - B: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '10' - } - } - }, - { - opcode: 'negative', - blockType: Scratch.BlockType.REPORTER, - text: '- [A]', - arguments: { - A: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '2' - }, - } - }, - { - opcode: 'n_th_Root', - blockType: Scratch.BlockType.REPORTER, - text: '[A] √ [B]', - arguments: { - A: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '3' - }, - B: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '8' - } - } - }, '---', - { - opcode: 'astrict', - blockType: Scratch.BlockType.REPORTER, - text: 'constrain [A] low [B] high [C]', - arguments: { - A: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '50' - }, - B: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '1' - }, - C: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '100' - } - } - }, - { - opcode: 'round', - blockType: Scratch.BlockType.REPORTER, - text: 'Round [A] to [B] decimal places', - arguments: { - A: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '3.14' - }, - B: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '1' - }, - C: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '100' - } - } - }, "---", - { - opcode: 'boolean', - blockType: Scratch.BlockType.BOOLEAN, - text: '[a]', - arguments: { - a: { - type: Scratch.ArgumentType.STRING, - defaultValue: '' - }, - } - }, - { - opcode: 'booleanToInt', - blockType: Scratch.BlockType.REPORTER, - text: '[a]', - arguments: { - a: { - type: Scratch.ArgumentType.BOOLEAN, - }, - } - }, - '---', - { - opcode: 'equal', - blockType: Scratch.BlockType.BOOLEAN, - text: '[a] ⩵ [b]', - arguments: { - a: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'A' - }, - b: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'a' - }, - } - }, - { - opcode: 'equalNegative', - blockType: Scratch.BlockType.BOOLEAN, - text: '[a] =- [b]', - arguments: { - a: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '5' - }, - b: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '-5' - }, - } - }, - { - opcode: 'equalPlusMinus', - blockType: Scratch.BlockType.BOOLEAN, - text: '[a] =± [b]', - arguments: { - a: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '5' - }, - b: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '-5' - }, - } - }, - { - opcode: 'notEqual', - blockType: Scratch.BlockType.BOOLEAN, - text: '[a] ≠ [b]', - arguments: { - a: { - type: Scratch.ArgumentType.STRING, - defaultValue: '' - }, - b: { - type: Scratch.ArgumentType.STRING, - defaultValue: '' - }, - } - }, - { - opcode: 'almostEqual2n', - blockType: Scratch.BlockType.BOOLEAN, - text: '[a] ≈ [b]', - arguments: { - a: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '5.5' - }, - b: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '6' - }, - } - }, - { - opcode: 'almostEqual3n', - blockType: Scratch.BlockType.BOOLEAN, - text: '[a] ≈ [b] ± [c]', - arguments: { - a: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '5' - }, - b: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '6' - }, - c: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '1' - }, - } - }, - { - opcode: 'xor', - blockType: Scratch.BlockType.BOOLEAN, - text: '[a] ^ [b]', - arguments: { - a: { - type: Scratch.ArgumentType.BOOLEAN, - }, - b: { - type: Scratch.ArgumentType.BOOLEAN, - }, - } - }, - '---', - { - opcode: 'equalOrGreater', - blockType: Scratch.BlockType.BOOLEAN, - text: '[a] ≥ [b]', - arguments: { - a: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '' - }, - b: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '50' - }, - } - }, - { - opcode: 'equalOrLess', - blockType: Scratch.BlockType.BOOLEAN, - text: '[a] ≤ [b]', - arguments: { - a: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '' - }, - b: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '50' - }, - } - }, - { - opcode: 'between', - blockType: Scratch.BlockType.BOOLEAN, - text: '[a] < [b] < [c]', - arguments: { - a: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '' - }, - b: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '' - }, - c: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '' - }, - } - }, - { - opcode: 'betweenEqual', - blockType: Scratch.BlockType.BOOLEAN, - text: '[a] ≤ [b] ≤ [c]', - arguments: { - a: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '' - }, - b: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '' - }, - c: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '' - }, - } - }, '---', - { - opcode: 'vertical', - blockType: Scratch.BlockType.BOOLEAN, - text: '[a] ⊥ [b]', - arguments: { - a: { - type: Scratch.ArgumentType.ANGLE, - defaultValue: '0' - }, - b: { - type: Scratch.ArgumentType.ANGLE, - defaultValue: '90' - }, - } - }, '---', { - opcode: 'text', - blockType: Scratch.BlockType.REPORTER, - text: '[a]', - arguments: { - a: { - type: Scratch.ArgumentType.STRING, - defaultValue: '' - }, - } - }, '---', { - opcode: 'repeat', - blockType: Scratch.BlockType.REPORTER, - text: 'repeat [text] [n] times', - arguments: { - text: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'Text ' - }, - n: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '3' - }, - } - }, { - opcode: 'trim', - blockType: Scratch.BlockType.REPORTER, - text: 'trim spaces from both sides of [text]', - arguments: { - text: { - type: Scratch.ArgumentType.STRING, - defaultValue: ' Text ' - }, - } - }, { - opcode: 'intercept', - blockType: Scratch.BlockType.REPORTER, - text: 'in text [text] get substring from [h] to [e]', - arguments: { - text: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'this is text test' - }, - h: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '6' - }, - e: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '7' - }, - } - }, { - opcode: "replace", - blockType: Scratch.BlockType.REPORTER, - text: "replace [o] of [text] with [n]", - arguments: { - o: { - type: Scratch.ArgumentType.STRING, - defaultValue: "world" - }, - text: { - type: Scratch.ArgumentType.STRING, - defaultValue: "Hello world!" - }, - n: { - type: Scratch.ArgumentType.STRING, - defaultValue: "Scratch" - } - } - }, { - opcode: 'Split', - blockType: Scratch.BlockType.REPORTER, - text: 'divide [text] according to [symbol] to take the [n] th item', - arguments: { - text: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'a_b_c' - }, - symbol: { - type: Scratch.ArgumentType.STRING, - defaultValue: '_' - }, - n: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '2' - }, - } - }, '---', { - opcode: 'toUpperCase', - blockType: Scratch.BlockType.REPORTER, - text: 'UPPER CASE [text]', - arguments: { - text: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'Text' - }, - } - }, { - opcode: 'toLowerCase', - blockType: Scratch.BlockType.REPORTER, - text: 'lower case [text]', - arguments: { - text: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'Text' - }, - } - }, { - opcode: 'textToTitleCase', - blockType: Scratch.BlockType.REPORTER, - text: 'Title Case [text]', - arguments: { - text: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'the text test' - }, - } - }, '---', { - opcode: 'indexOf', - blockType: Scratch.BlockType.REPORTER, - text: '[a] the first appearance in [text]', - arguments: { - text: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'The text test' - }, - a: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'te' - }, - } - }, { - opcode: 'lastIndexOf', - blockType: Scratch.BlockType.REPORTER, - text: '[a] the position of last occurrence in [text]', - arguments: { - text: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'The text test' - }, - a: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'te' - }, - } - }, { - opcode: 'countKeyword', - blockType: Scratch.BlockType.REPORTER, - text: '[a] the number of occurrences in [text]', - arguments: { - text: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'The text test' - }, - a: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'te' - }, - } - }, '---', { - opcode: 'startsWith', - blockType: Scratch.BlockType.BOOLEAN, - text: 'does [a] begin with a text?', - arguments: { - a: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'Abc123' - }, - } - }, - { - opcode: 'matchTextWithPattern', - blockType: Scratch.BlockType.BOOLEAN, - text: 'match [text] as [pattern] - [flags]', - arguments: { - text: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'abc' - }, - pattern: { - type: Scratch.ArgumentType.STRING, - defaultValue: '[ -~]' - }, - flags: { - type: Scratch.ArgumentType.STRING, - menu: 'flags' - }, - } - }, '---', { - opcode: 'ascii', - blockType: Scratch.BlockType.REPORTER, - text: '[a]\'s ascii', - arguments: { - a: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'a' - }, - } - }, { - opcode: 'ascii_', - blockType: Scratch.BlockType.REPORTER, - text: 'ascii is [a] \'s text', - arguments: { - a: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '97' - }, - } - }, '---', { - opcode: 'line_segment', - blockType: Scratch.BlockType.REPORTER, - text: 'line segment ([x1],[y1]) to ([x2],[y2])', - arguments: { - x1: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '-100' - }, - y1: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - x2: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - y2: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - } - }, - '---', - { - opcode: 'triangle', - blockType: Scratch.BlockType.REPORTER, - text: 'triangle ([x1],[y1]) ([x2],[y2]) ([x3],[y3]) \'s [CS]', - arguments: { - x1: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - y1: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '10' - }, - x2: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '10' - }, - y2: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '10' - }, - x3: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '10' - }, - y3: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - CS: { - type: Scratch.ArgumentType.STRING, - menu: 'cs' - }, - }, - }, - { - opcode: 'triangle_s', - blockType: Scratch.BlockType.REPORTER, - text: 'triangle [s1] [s2] [s3] \'s square', - arguments: { - s1: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '3' - }, - s2: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '4' - }, - s3: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '5' - }, - }, - }, - { - opcode: 'rectangle', - blockType: Scratch.BlockType.REPORTER, - text: 'rectangle ([x1],[y1]) ([x2],[y2]) ([x3],[y3]) ([x4],[y4]) \'s [CS]', - arguments: { - x1: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - y1: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '10' - }, - x2: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '10' - }, - y2: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '10' - }, - x3: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '10' - }, - y3: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - x4: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - y4: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - CS: { - type: Scratch.ArgumentType.STRING, - menu: 'cs' - }, - }, - }, - { - opcode: 'graph', - blockType: Scratch.BlockType.REPORTER, - text: 'graph [graph] \'s [CS]', - arguments: { - graph: { - type: Scratch.ArgumentType.STRING, - defaultValue: '[[0,0], [0,2], [2,4], [4,2], [4,0]]' - }, - CS: { - type: Scratch.ArgumentType.STRING, - menu: 'cs' - }, - }, - }, - '---', - { - opcode: 'circle', - blockType: Scratch.BlockType.REPORTER, - text: 'circle: [rd][a] \'s [CS]', - arguments: { - rd: { - type: Scratch.ArgumentType.STRING, - menu: 'rd' - }, - a: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '10' - }, - CS: { - type: Scratch.ArgumentType.STRING, - menu: 'cs' - }, - }, - }, - '---', - { - opcode: 'words', - blockType: Scratch.BlockType.REPORTER, - disableMonitor: true, - text: 'sort unique words in [text] as [language]', - arguments: { - text: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'movie dog restaurant book school' - }, - language: { - type: Scratch.ArgumentType.STRING, - menu: 'language' - } - } - }, '---', - { - opcode: 'true', - blockType: Scratch.BlockType.BOOLEAN, - text: 'true', - }, - { - opcode: 'false', - blockType: Scratch.BlockType.BOOLEAN, - text: 'false', - }, - { - opcode: 'new_line', - disableMonitor: true, - blockType: Scratch.BlockType.REPORTER, - text: '\\n', - }, - { - opcode: 'pi', - disableMonitor: true, - blockType: Scratch.BlockType.REPORTER, - text: 'π', - }, - { - opcode: 'phi', - disableMonitor: true, - blockType: Scratch.BlockType.REPORTER, - text: 'φ', - }, - { - opcode: 'e', - disableMonitor: true, - blockType: Scratch.BlockType.REPORTER, - text: 'e', - }, - { - opcode: 'infinity', - disableMonitor: true, - blockType: Scratch.BlockType.REPORTER, - text: '∞', - }, - ], - menus: { - rd: { - acceptReporters: true, - items: [ - { - text: 'radius (r)', - value: 'r' - }, - { - text: 'diameter (d)', - value: 'd' - } - ] - }, - cs: { - acceptReporters: true, - items: [ - { - text: 'square (s)', - value: 's' - }, - { - text: 'circumference (c)', - value: 'c' - } - ] - }, - language: { - acceptReporters: true, - items: [ - { - text: 'English (en)', - value: 'en' - }, - { - text: 'Chinese (zh)', - value: 'zh' - } - ] - }, - flags: { - acceptReporters: false, - items: [ - { - text: 'global (g)', - value: 'g' - }, - { - text: 'ignoring case (i)', - value: 'i' - } - ] - } - } - }; - } - exponent({ A, B }) { - return A ** B; - } - negative({ A }) { - return 0 - A; - } - n_th_Root({ A, B }) { - return Math.pow(B, 1 / A); - } - astrict({ A, B, C }) { - return Math.min(Math.max(A, B), C); - } - round(args) { - return args.A.toFixed(args.B); - } + "use strict"; + class MathAndString { + getInfo() { + return { + color1: "#5ac900", + color2: "#48a100", + color3: "#48a100", + id: "nonameawamathandstring", + name: "Math And String", + blocks: [ + { + opcode: "exponent", + blockType: Scratch.BlockType.REPORTER, + text: "[A] ^ [B]", + arguments: { + A: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "2", + }, + B: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "10", + }, + }, + }, + { + opcode: "negative", + blockType: Scratch.BlockType.REPORTER, + text: "- [A]", + arguments: { + A: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "2", + }, + }, + }, + { + opcode: "n_th_Root", + blockType: Scratch.BlockType.REPORTER, + text: "[A] √ [B]", + arguments: { + A: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "3", + }, + B: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "8", + }, + }, + }, + "---", + { + opcode: "astrict", + blockType: Scratch.BlockType.REPORTER, + text: "constrain [A] low [B] high [C]", + arguments: { + A: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "50", + }, + B: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "1", + }, + C: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "100", + }, + }, + }, + { + opcode: "round", + blockType: Scratch.BlockType.REPORTER, + text: "Round [A] to [B] decimal places", + arguments: { + A: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "3.14", + }, + B: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "1", + }, + C: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "100", + }, + }, + }, + "---", + { + opcode: "boolean", + blockType: Scratch.BlockType.BOOLEAN, + text: "[a]", + arguments: { + a: { + type: Scratch.ArgumentType.STRING, + defaultValue: "", + }, + }, + }, + { + opcode: "booleanToInt", + blockType: Scratch.BlockType.REPORTER, + text: "[a]", + arguments: { + a: { + type: Scratch.ArgumentType.BOOLEAN, + }, + }, + }, + "---", + { + opcode: "equal", + blockType: Scratch.BlockType.BOOLEAN, + text: "[a] ⩵ [b]", + arguments: { + a: { + type: Scratch.ArgumentType.STRING, + defaultValue: "A", + }, + b: { + type: Scratch.ArgumentType.STRING, + defaultValue: "a", + }, + }, + }, + { + opcode: "equalNegative", + blockType: Scratch.BlockType.BOOLEAN, + text: "[a] =- [b]", + arguments: { + a: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "5", + }, + b: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "-5", + }, + }, + }, + { + opcode: "equalPlusMinus", + blockType: Scratch.BlockType.BOOLEAN, + text: "[a] =± [b]", + arguments: { + a: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "5", + }, + b: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "-5", + }, + }, + }, + { + opcode: "notEqual", + blockType: Scratch.BlockType.BOOLEAN, + text: "[a] ≠ [b]", + arguments: { + a: { + type: Scratch.ArgumentType.STRING, + defaultValue: "", + }, + b: { + type: Scratch.ArgumentType.STRING, + defaultValue: "", + }, + }, + }, + { + opcode: "almostEqual2n", + blockType: Scratch.BlockType.BOOLEAN, + text: "[a] ≈ [b]", + arguments: { + a: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "5.5", + }, + b: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "6", + }, + }, + }, + { + opcode: "almostEqual3n", + blockType: Scratch.BlockType.BOOLEAN, + text: "[a] ≈ [b] ± [c]", + arguments: { + a: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "5", + }, + b: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "6", + }, + c: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "1", + }, + }, + }, + { + opcode: "xor", + blockType: Scratch.BlockType.BOOLEAN, + text: "[a] ^ [b]", + arguments: { + a: { + type: Scratch.ArgumentType.BOOLEAN, + }, + b: { + type: Scratch.ArgumentType.BOOLEAN, + }, + }, + }, + "---", + { + opcode: "equalOrGreater", + blockType: Scratch.BlockType.BOOLEAN, + text: "[a] ≥ [b]", + arguments: { + a: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "", + }, + b: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "50", + }, + }, + }, + { + opcode: "equalOrLess", + blockType: Scratch.BlockType.BOOLEAN, + text: "[a] ≤ [b]", + arguments: { + a: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "", + }, + b: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "50", + }, + }, + }, + { + opcode: "between", + blockType: Scratch.BlockType.BOOLEAN, + text: "[a] < [b] < [c]", + arguments: { + a: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "", + }, + b: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "", + }, + c: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "", + }, + }, + }, + { + opcode: "betweenEqual", + blockType: Scratch.BlockType.BOOLEAN, + text: "[a] ≤ [b] ≤ [c]", + arguments: { + a: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "", + }, + b: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "", + }, + c: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "", + }, + }, + }, + "---", + { + opcode: "vertical", + blockType: Scratch.BlockType.BOOLEAN, + text: "[a] ⊥ [b]", + arguments: { + a: { + type: Scratch.ArgumentType.ANGLE, + defaultValue: "0", + }, + b: { + type: Scratch.ArgumentType.ANGLE, + defaultValue: "90", + }, + }, + }, + "---", + { + opcode: "text", + blockType: Scratch.BlockType.REPORTER, + text: "[a]", + arguments: { + a: { + type: Scratch.ArgumentType.STRING, + defaultValue: "", + }, + }, + }, + "---", + { + opcode: "repeat", + blockType: Scratch.BlockType.REPORTER, + text: "repeat [text] [n] times", + arguments: { + text: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Text ", + }, + n: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "3", + }, + }, + }, + { + opcode: "trim", + blockType: Scratch.BlockType.REPORTER, + text: "trim spaces from both sides of [text]", + arguments: { + text: { + type: Scratch.ArgumentType.STRING, + defaultValue: " Text ", + }, + }, + }, + { + opcode: "intercept", + blockType: Scratch.BlockType.REPORTER, + text: "in text [text] get substring from [h] to [e]", + arguments: { + text: { + type: Scratch.ArgumentType.STRING, + defaultValue: "this is text test", + }, + h: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "6", + }, + e: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "7", + }, + }, + }, + { + opcode: "replace", + blockType: Scratch.BlockType.REPORTER, + text: "replace [o] of [text] with [n]", + arguments: { + o: { + type: Scratch.ArgumentType.STRING, + defaultValue: "world", + }, + text: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Hello world!", + }, + n: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Scratch", + }, + }, + }, + { + opcode: "Split", + blockType: Scratch.BlockType.REPORTER, + text: "divide [text] according to [symbol] to take the [n] th item", + arguments: { + text: { + type: Scratch.ArgumentType.STRING, + defaultValue: "a_b_c", + }, + symbol: { + type: Scratch.ArgumentType.STRING, + defaultValue: "_", + }, + n: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "2", + }, + }, + }, + "---", + { + opcode: "toUpperCase", + blockType: Scratch.BlockType.REPORTER, + text: "UPPER CASE [text]", + arguments: { + text: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Text", + }, + }, + }, + { + opcode: "toLowerCase", + blockType: Scratch.BlockType.REPORTER, + text: "lower case [text]", + arguments: { + text: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Text", + }, + }, + }, + { + opcode: "textToTitleCase", + blockType: Scratch.BlockType.REPORTER, + text: "Title Case [text]", + arguments: { + text: { + type: Scratch.ArgumentType.STRING, + defaultValue: "the text test", + }, + }, + }, + "---", + { + opcode: "indexOf", + blockType: Scratch.BlockType.REPORTER, + text: "[a] the first appearance in [text]", + arguments: { + text: { + type: Scratch.ArgumentType.STRING, + defaultValue: "The text test", + }, + a: { + type: Scratch.ArgumentType.STRING, + defaultValue: "te", + }, + }, + }, + { + opcode: "lastIndexOf", + blockType: Scratch.BlockType.REPORTER, + text: "[a] the position of last occurrence in [text]", + arguments: { + text: { + type: Scratch.ArgumentType.STRING, + defaultValue: "The text test", + }, + a: { + type: Scratch.ArgumentType.STRING, + defaultValue: "te", + }, + }, + }, + { + opcode: "countKeyword", + blockType: Scratch.BlockType.REPORTER, + text: "[a] the number of occurrences in [text]", + arguments: { + text: { + type: Scratch.ArgumentType.STRING, + defaultValue: "The text test", + }, + a: { + type: Scratch.ArgumentType.STRING, + defaultValue: "te", + }, + }, + }, + "---", + { + opcode: "startsWith", + blockType: Scratch.BlockType.BOOLEAN, + text: "does [a] begin with a text?", + arguments: { + a: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Abc123", + }, + }, + }, + { + opcode: "matchTextWithPattern", + blockType: Scratch.BlockType.BOOLEAN, + text: "match [text] as [pattern] - [flags]", + arguments: { + text: { + type: Scratch.ArgumentType.STRING, + defaultValue: "abc", + }, + pattern: { + type: Scratch.ArgumentType.STRING, + defaultValue: "[ -~]", + }, + flags: { + type: Scratch.ArgumentType.STRING, + menu: "flags", + }, + }, + }, + "---", + { + opcode: "ascii", + blockType: Scratch.BlockType.REPORTER, + text: "[a]'s ascii", + arguments: { + a: { + type: Scratch.ArgumentType.STRING, + defaultValue: "a", + }, + }, + }, + { + opcode: "ascii_", + blockType: Scratch.BlockType.REPORTER, + text: "ascii is [a] 's text", + arguments: { + a: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "97", + }, + }, + }, + "---", + { + opcode: "line_segment", + blockType: Scratch.BlockType.REPORTER, + text: "line segment ([x1],[y1]) to ([x2],[y2])", + arguments: { + x1: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "-100", + }, + y1: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + x2: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + y2: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + }, + }, + "---", + { + opcode: "triangle", + blockType: Scratch.BlockType.REPORTER, + text: "triangle ([x1],[y1]) ([x2],[y2]) ([x3],[y3]) 's [CS]", + arguments: { + x1: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + y1: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "10", + }, + x2: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "10", + }, + y2: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "10", + }, + x3: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "10", + }, + y3: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + CS: { + type: Scratch.ArgumentType.STRING, + menu: "cs", + }, + }, + }, + { + opcode: "triangle_s", + blockType: Scratch.BlockType.REPORTER, + text: "triangle [s1] [s2] [s3] 's square", + arguments: { + s1: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "3", + }, + s2: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "4", + }, + s3: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "5", + }, + }, + }, + { + opcode: "rectangle", + blockType: Scratch.BlockType.REPORTER, + text: "rectangle ([x1],[y1]) ([x2],[y2]) ([x3],[y3]) ([x4],[y4]) 's [CS]", + arguments: { + x1: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + y1: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "10", + }, + x2: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "10", + }, + y2: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "10", + }, + x3: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "10", + }, + y3: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + x4: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + y4: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + CS: { + type: Scratch.ArgumentType.STRING, + menu: "cs", + }, + }, + }, + { + opcode: "graph", + blockType: Scratch.BlockType.REPORTER, + text: "graph [graph] 's [CS]", + arguments: { + graph: { + type: Scratch.ArgumentType.STRING, + defaultValue: "[[0,0], [0,2], [2,4], [4,2], [4,0]]", + }, + CS: { + type: Scratch.ArgumentType.STRING, + menu: "cs", + }, + }, + }, + "---", + { + opcode: "circle", + blockType: Scratch.BlockType.REPORTER, + text: "circle: [rd][a] 's [CS]", + arguments: { + rd: { + type: Scratch.ArgumentType.STRING, + menu: "rd", + }, + a: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "10", + }, + CS: { + type: Scratch.ArgumentType.STRING, + menu: "cs", + }, + }, + }, + "---", + { + opcode: "words", + blockType: Scratch.BlockType.REPORTER, + disableMonitor: true, + text: "sort unique words in [text] as [language]", + arguments: { + text: { + type: Scratch.ArgumentType.STRING, + defaultValue: "movie dog restaurant book school", + }, + language: { + type: Scratch.ArgumentType.STRING, + menu: "language", + }, + }, + }, + "---", + { + opcode: "true", + blockType: Scratch.BlockType.BOOLEAN, + text: "true", + }, + { + opcode: "false", + blockType: Scratch.BlockType.BOOLEAN, + text: "false", + }, + { + opcode: "new_line", + disableMonitor: true, + blockType: Scratch.BlockType.REPORTER, + text: "\\n", + }, + { + opcode: "pi", + disableMonitor: true, + blockType: Scratch.BlockType.REPORTER, + text: "π", + }, + { + opcode: "phi", + disableMonitor: true, + blockType: Scratch.BlockType.REPORTER, + text: "φ", + }, + { + opcode: "e", + disableMonitor: true, + blockType: Scratch.BlockType.REPORTER, + text: "e", + }, + { + opcode: "infinity", + disableMonitor: true, + blockType: Scratch.BlockType.REPORTER, + text: "∞", + }, + ], + menus: { + rd: { + acceptReporters: true, + items: [ + { + text: "radius (r)", + value: "r", + }, + { + text: "diameter (d)", + value: "d", + }, + ], + }, + cs: { + acceptReporters: true, + items: [ + { + text: "square (s)", + value: "s", + }, + { + text: "circumference (c)", + value: "c", + }, + ], + }, + language: { + acceptReporters: true, + items: [ + { + text: "English (en)", + value: "en", + }, + { + text: "Chinese (zh)", + value: "zh", + }, + ], + }, + flags: { + acceptReporters: false, + items: [ + { + text: "global (g)", + value: "g", + }, + { + text: "ignoring case (i)", + value: "i", + }, + ], + }, + }, + }; + } + exponent({ A, B }) { + return A ** B; + } + negative({ A }) { + return 0 - A; + } + n_th_Root({ A, B }) { + return Math.pow(B, 1 / A); + } + astrict({ A, B, C }) { + return Math.min(Math.max(A, B), C); + } + round(args) { + return args.A.toFixed(args.B); + } - true() { - return true; - } - false() { - return false; - } - boolean(args) { - return Scratch.Cast.toBoolean(args.a); - } - booleanToInt(args) { - if (Scratch.Cast.toBoolean(args.a)) { - return 1; - } - return 0; - } - equal(args) { - return (args.a == args.b); - } - equalNegative(args) { - if (isNaN(args.a) || isNaN(args.b)) { - return false; - } else { - return (args.a == (0 - args.b)); - } - } - equalPlusMinus(args) { - if (isNaN(args.a) || isNaN(args.b)) { - return false; - } else { - return (args.a == (0 - args.b)) || (args.a == args.b); - } - } - almostEqual2n(args) { - return (Math.round(args.a) == Math.round(args.b)); - } - almostEqual3n(args) { - return (Math.abs(args.a - args.b) <= args.c); - } - between(args) { - return (args.a < args.b) && (args.b < args.c); - } - betweenEqual(args) { - return (args.a <= args.b) && (args.b <= args.c); - } - notEqual(args) { - return (args.a != args.b); - } - xor(args) { - return Scratch.Cast.toBoolean(args.a) !== Scratch.Cast.toBoolean(args.b); - } - equalOrGreater(args) { - return (args.a >= args.b); - } - equalOrLess(args) { - return (args.a <= args.b); - } - vertical(args) { - if (isNaN(args.a) || isNaN(args.b)) { - return false; - } else { - return ((args.a - (args.b - 90)) % 180) == 0; - } - } - segment_one(args) { - return Math.round(Math.sqrt(Math.pow(args.x1 - args.x2, 2) + Math.pow(args.y1 - args.y2, 2))) == args.n; - } - segment_two(args) { - return Math.round(Math.sqrt(Math.pow(args.x11 - args.x12, 2) + Math.pow(args.y11 - args.y12, 2))) - == Math.round(Math.sqrt(Math.pow(args.x21 - args.x22, 2) + Math.pow(args.y21 - args.y22, 2))); - } + true() { + return true; + } + false() { + return false; + } + boolean(args) { + return Scratch.Cast.toBoolean(args.a); + } + booleanToInt(args) { + if (Scratch.Cast.toBoolean(args.a)) { + return 1; + } + return 0; + } + equal(args) { + return args.a == args.b; + } + equalNegative(args) { + if (isNaN(args.a) || isNaN(args.b)) { + return false; + } else { + return args.a == 0 - args.b; + } + } + equalPlusMinus(args) { + if (isNaN(args.a) || isNaN(args.b)) { + return false; + } else { + return args.a == 0 - args.b || args.a == args.b; + } + } + almostEqual2n(args) { + return Math.round(args.a) == Math.round(args.b); + } + almostEqual3n(args) { + return Math.abs(args.a - args.b) <= args.c; + } + between(args) { + return args.a < args.b && args.b < args.c; + } + betweenEqual(args) { + return args.a <= args.b && args.b <= args.c; + } + notEqual(args) { + return args.a != args.b; + } + xor(args) { + return Scratch.Cast.toBoolean(args.a) !== Scratch.Cast.toBoolean(args.b); + } + equalOrGreater(args) { + return args.a >= args.b; + } + equalOrLess(args) { + return args.a <= args.b; + } + vertical(args) { + if (isNaN(args.a) || isNaN(args.b)) { + return false; + } else { + return (args.a - (args.b - 90)) % 180 == 0; + } + } + segment_one(args) { + return ( + Math.round( + Math.sqrt( + Math.pow(args.x1 - args.x2, 2) + Math.pow(args.y1 - args.y2, 2) + ) + ) == args.n + ); + } + segment_two(args) { + return ( + Math.round( + Math.sqrt( + Math.pow(args.x11 - args.x12, 2) + Math.pow(args.y11 - args.y12, 2) + ) + ) == + Math.round( + Math.sqrt( + Math.pow(args.x21 - args.x22, 2) + Math.pow(args.y21 - args.y22, 2) + ) + ) + ); + } - ascii(args) { - return args.a.charCodeAt(); - } - ascii_(args) { - return String.fromCharCode(args.a); - } - text(args) { - return args.a; - } - repeat(args) { - if (args.n > 0) { - let repeat_i; - let repeat_j = ''; - let repeat_i_inc = 1; - if (1 > args.n) { - repeat_i_inc = -repeat_i_inc; - } - for (repeat_i = 1; repeat_i_inc >= 0 ? repeat_i <= args.n : repeat_i >= args.n; repeat_i += repeat_i_inc) { - repeat_j = String(repeat_j) + String(args.text); - } - return repeat_j; - } - return ''; - } - intercept(args) { - return args.text.slice((args.h - 1), args.e); - } - toUpperCase(args) { - return args.text.toUpperCase(); - } - toLowerCase(args) { - return args.text.toLowerCase(); - } - textToTitleCase(args) { - return textToTitleCase(args.text); - } - 'trim'(args) { - return args.text.trim(); - } - new_line() { - return '\n'; - } - 'Split'(args) { - const symbol = args.symbol === '.' ? '\\.' : args.symbol; - if (args.text && typeof args.text.split === 'function' && args.text.split(symbol)[(args.n - 1)] != undefined) { - return args.text.split(symbol)[(args.n - 1)]; - } - return ''; - } - indexOf(args) { - return args.text.indexOf(args.a) + 1; - } - lastIndexOf(args) { - return args.text.lastIndexOf(args.a) + 1; - } - replace(args) { - return replaceText(args.text, args.o, args.n); - } - startsWith(args) { - if (typeof args.a === 'string' && args.a.startsWith(args.a)) { - return true; - } else { - return false; - } + ascii(args) { + return args.a.charCodeAt(); + } + ascii_(args) { + return String.fromCharCode(args.a); + } + text(args) { + return args.a; + } + repeat(args) { + if (args.n > 0) { + let repeat_i; + let repeat_j = ""; + let repeat_i_inc = 1; + if (1 > args.n) { + repeat_i_inc = -repeat_i_inc; } - 'countKeyword'(args) { - return countKeyword(args.text, args.a); + for ( + repeat_i = 1; + repeat_i_inc >= 0 ? repeat_i <= args.n : repeat_i >= args.n; + repeat_i += repeat_i_inc + ) { + repeat_j = String(repeat_j) + String(args.text); } + return repeat_j; + } + return ""; + } + intercept(args) { + return args.text.slice(args.h - 1, args.e); + } + toUpperCase(args) { + return args.text.toUpperCase(); + } + toLowerCase(args) { + return args.text.toLowerCase(); + } + textToTitleCase(args) { + return textToTitleCase(args.text); + } + trim(args) { + return args.text.trim(); + } + new_line() { + return "\n"; + } + Split(args) { + const symbol = args.symbol === "." ? "\\." : args.symbol; + if ( + args.text && + typeof args.text.split === "function" && + args.text.split(symbol)[args.n - 1] != undefined + ) { + return args.text.split(symbol)[args.n - 1]; + } + return ""; + } + indexOf(args) { + return args.text.indexOf(args.a) + 1; + } + lastIndexOf(args) { + return args.text.lastIndexOf(args.a) + 1; + } + replace(args) { + return replaceText(args.text, args.o, args.n); + } + startsWith(args) { + if (typeof args.a === "string" && args.a.startsWith(args.a)) { + return true; + } else { + return false; + } + } + countKeyword(args) { + return countKeyword(args.text, args.a); + } - line_segment(args) { - return Math.sqrt(Math.pow(args.x1 - args.x2, 2) + Math.pow(args.y1 - args.y2, 2)); - } - triangle(args) { - if (args.CS == 's') { - let points = [[args.x1, args.y1], [args.x2, args.y2], [args.x3, args.y3]]; - let area = 0; - let n = points.length; - for (let i = 0; i < n; i++) { - let x1 = points[i][0]; - let y1 = points[i][1]; - let x2 = points[(i + 1) % n][0]; - let y2 = points[(i + 1) % n][1]; - area += x1 * y2; - area -= x2 * y1; - } - area = Math.abs(area) / 2; - return (area); - } - if (args.CS == 'c') { - let i = 0; - i += Math.sqrt(Math.pow(args.x1 - args.x2, 2) + Math.pow(args.y1 - args.y2, 2)); - i += Math.sqrt(Math.pow(args.x2 - args.x3, 2) + Math.pow(args.y2 - args.y3, 2)); - i += Math.sqrt(Math.pow(args.x3 - args.x1, 2) + Math.pow(args.y3 - args.y1, 2)); - return i; - } - return 0; - } - triangle_s(args) { - const s = (args.s1 + args.s2 + args.s3) / 2; - const area = Math.sqrt(s * (s - args.s1) * (s - args.s2) * (s - args.s3)); - return area; + line_segment(args) { + return Math.sqrt( + Math.pow(args.x1 - args.x2, 2) + Math.pow(args.y1 - args.y2, 2) + ); + } + triangle(args) { + if (args.CS == "s") { + let points = [ + [args.x1, args.y1], + [args.x2, args.y2], + [args.x3, args.y3], + ]; + let area = 0; + let n = points.length; + for (let i = 0; i < n; i++) { + let x1 = points[i][0]; + let y1 = points[i][1]; + let x2 = points[(i + 1) % n][0]; + let y2 = points[(i + 1) % n][1]; + area += x1 * y2; + area -= x2 * y1; } - rectangle(args) { - if (args.CS == 's') { - let points = [[args.x1, args.y1], [args.x2, args.y2], [args.x3, args.y3], [args.x4, args.y4]]; - let area = 0; - let n = points.length; - for (let i = 0; i < n; i++) { - let x1 = points[i][0]; - let y1 = points[i][1]; - let x2 = points[(i + 1) % n][0]; - let y2 = points[(i + 1) % n][1]; - area += x1 * y2; - area -= x2 * y1; - } - area = Math.abs(area) / 2; - return (area); - } - if (args.CS == 'c') { - let i = 0; - i += Math.sqrt(Math.pow(args.x1 - args.x2, 2) + Math.pow(args.y1 - args.y2, 2)); - i += Math.sqrt(Math.pow(args.x2 - args.x3, 2) + Math.pow(args.y2 - args.y3, 2)); - i += Math.sqrt(Math.pow(args.x3 - args.x4, 2) + Math.pow(args.y3 - args.y4, 2)); - i += Math.sqrt(Math.pow(args.x4 - args.x1, 2) + Math.pow(args.y4 - args.y1, 2)); - return i; - } - return 0; + area = Math.abs(area) / 2; + return area; + } + if (args.CS == "c") { + let i = 0; + i += Math.sqrt( + Math.pow(args.x1 - args.x2, 2) + Math.pow(args.y1 - args.y2, 2) + ); + i += Math.sqrt( + Math.pow(args.x2 - args.x3, 2) + Math.pow(args.y2 - args.y3, 2) + ); + i += Math.sqrt( + Math.pow(args.x3 - args.x1, 2) + Math.pow(args.y3 - args.y1, 2) + ); + return i; + } + return 0; + } + triangle_s(args) { + const s = (args.s1 + args.s2 + args.s3) / 2; + const area = Math.sqrt(s * (s - args.s1) * (s - args.s2) * (s - args.s3)); + return area; + } + rectangle(args) { + if (args.CS == "s") { + let points = [ + [args.x1, args.y1], + [args.x2, args.y2], + [args.x3, args.y3], + [args.x4, args.y4], + ]; + let area = 0; + let n = points.length; + for (let i = 0; i < n; i++) { + let x1 = points[i][0]; + let y1 = points[i][1]; + let x2 = points[(i + 1) % n][0]; + let y2 = points[(i + 1) % n][1]; + area += x1 * y2; + area -= x2 * y1; } - graph(args) { - let points = JSON.parse(args.graph); - let n = points.length; - if (args.CS == 's') { - let area = 0; - for (let i = 0; i < n; i++) { - let x1 = points[i][0]; - let y1 = points[i][1]; - let x2 = points[(i + 1) % n][0]; - let y2 = points[(i + 1) % n][1]; - area += x1 * y2; - area -= x2 * y1; - } - area = Math.abs(area) / 2; - return (area); - } - if (args.CS == 'c') { - let x1, x2, y1, y2; - let j = 0; - j = 0; - var i_end = n - 1; - var i_inc = 1; - if (0 > i_end) { - i_inc = -i_inc; - } - for (let i = 0; i_inc >= 0 ? i <= i_end : i >= i_end; i += i_inc) { - x1 = points[((i + 1) - 1)][0]; - x2 = i == n - 1 ? points[0][0] : points[((i + 2) - 1)][0]; - y1 = points[((i + 1) - 1)][1]; - y2 = i == n - 1 ? points[0][1] : points[((i + 2) - 1)][1]; - j = (typeof j == 'number' ? j : 0) + Math.sqrt(Math.pow(Math.abs(x1 - x2), 2) + Math.pow(Math.abs(y1 - y2), 2)); - } - return j; - } - return 0; + area = Math.abs(area) / 2; + return area; + } + if (args.CS == "c") { + let i = 0; + i += Math.sqrt( + Math.pow(args.x1 - args.x2, 2) + Math.pow(args.y1 - args.y2, 2) + ); + i += Math.sqrt( + Math.pow(args.x2 - args.x3, 2) + Math.pow(args.y2 - args.y3, 2) + ); + i += Math.sqrt( + Math.pow(args.x3 - args.x4, 2) + Math.pow(args.y3 - args.y4, 2) + ); + i += Math.sqrt( + Math.pow(args.x4 - args.x1, 2) + Math.pow(args.y4 - args.y1, 2) + ); + return i; + } + return 0; + } + graph(args) { + let points = JSON.parse(args.graph); + let n = points.length; + if (args.CS == "s") { + let area = 0; + for (let i = 0; i < n; i++) { + let x1 = points[i][0]; + let y1 = points[i][1]; + let x2 = points[(i + 1) % n][0]; + let y2 = points[(i + 1) % n][1]; + area += x1 * y2; + area -= x2 * y1; } - circle(args) { - if (args.CS == 'c') { - return 2 * Math.PI * (args.rd == 'r' ? args.a : args.a / 2); - } - if (args.CS == 's') { - return Math.PI * ((args.rd == 'r' ? args.a : args.a / 2) ** 2); - } + area = Math.abs(area) / 2; + return area; + } + if (args.CS == "c") { + let x1, x2, y1, y2; + let j = 0; + j = 0; + var i_end = n - 1; + var i_inc = 1; + if (0 > i_end) { + i_inc = -i_inc; } - pi() { - return Math.PI; + for (let i = 0; i_inc >= 0 ? i <= i_end : i >= i_end; i += i_inc) { + x1 = points[i + 1 - 1][0]; + x2 = i == n - 1 ? points[0][0] : points[i + 2 - 1][0]; + y1 = points[i + 1 - 1][1]; + y2 = i == n - 1 ? points[0][1] : points[i + 2 - 1][1]; + j = + (typeof j == "number" ? j : 0) + + Math.sqrt( + Math.pow(Math.abs(x1 - x2), 2) + Math.pow(Math.abs(y1 - y2), 2) + ); } + return j; + } + return 0; + } + circle(args) { + if (args.CS == "c") { + return 2 * Math.PI * (args.rd == "r" ? args.a : args.a / 2); + } + if (args.CS == "s") { + return Math.PI * (args.rd == "r" ? args.a : args.a / 2) ** 2; + } + } + pi() { + return Math.PI; + } - words(args) { - const text = Scratch.Cast.toString(args.text); - const words = parse(text, args.language); - return words.join(' '); - } + words(args) { + const text = Scratch.Cast.toString(args.text); + const words = parse(text, args.language); + return words.join(" "); + } - phi() { - return (1 + Math.sqrt(5)) / 2; - } - e() { - return Math.E; - } - infinity() { - return 'Infinity'; - } + phi() { + return (1 + Math.sqrt(5)) / 2; + } + e() { + return Math.E; + } + infinity() { + return "Infinity"; + } - matchTextWithPattern({ text, pattern, flags }) { - const regex = new RegExp(pattern, flags); - return regex.test(text); - } + matchTextWithPattern({ text, pattern, flags }) { + const regex = new RegExp(pattern, flags); + return regex.test(text); } + } - const textToTitleCase = (str) => { - return str.replace(/\S+/g, - function (txt) { - return txt[0].toUpperCase() + txt.substring(1).toLowerCase(); -}); - }; + const textToTitleCase = (str) => { + return str.replace(/\S+/g, function (txt) { + return txt[0].toUpperCase() + txt.substring(1).toLowerCase(); + }); + }; - const replaceText = (text, oldStr, newStr) => { - return text.replace(new RegExp(oldStr, 'g'), newStr); - }; + const replaceText = (text, oldStr, newStr) => { + return text.replace(new RegExp(oldStr, "g"), newStr); + }; - const sortAndUniqueWords_en = (text) => { - let words = text.toLowerCase().match(/\b\w+\b/g); - words = Array.from(new Set(words)); - words.sort(); - return words.join(' '); - }; + const sortAndUniqueWords_en = (text) => { + let words = text.toLowerCase().match(/\b\w+\b/g); + words = Array.from(new Set(words)); + words.sort(); + return words.join(" "); + }; - const sortAndUniqueWords_cn = (text) => { - let words = text.match(/[^\u4e00-\u9fa5]+|[\u4e00-\u9fa5]+/g); - words = Array.from(new Set(words)); - words.sort(function (a, b) { - return a.localeCompare(b, 'zh-Hans-CN', { sensitivity: 'accent' }); - }); - return words.join(' '); - }; + const sortAndUniqueWords_cn = (text) => { + let words = text.match(/[^\u4e00-\u9fa5]+|[\u4e00-\u9fa5]+/g); + words = Array.from(new Set(words)); + words.sort(function (a, b) { + return a.localeCompare(b, "zh-Hans-CN", { sensitivity: "accent" }); + }); + return words.join(" "); + }; - const countKeyword = (sentence, keyword) => { - const count = (sentence.match(new RegExp(keyword, 'gi')) || []).length; - return count; - }; + const countKeyword = (sentence, keyword) => { + const count = (sentence.match(new RegExp(keyword, "gi")) || []).length; + return count; + }; - const parseEnglish = (text) => { - const words = text.toLowerCase().match(/\b\w+\b/g); - const uniques = Array.from(new Set(words)); - uniques.sort(); - return uniques; - }; + const parseEnglish = (text) => { + const words = text.toLowerCase().match(/\b\w+\b/g); + const uniques = Array.from(new Set(words)); + uniques.sort(); + return uniques; + }; - const parseChinese = (text) => { - const words = text.match(/[^\u4e00-\u9fa5]+|[\u4e00-\u9fa5]+/g); - const uniques = Array.from(new Set(words)); - uniques.sort(function (a, b) { - return a.localeCompare(b, 'zh-Hans-CN', { sensitivity: 'accent' }); - }); - return uniques; - }; + const parseChinese = (text) => { + const words = text.match(/[^\u4e00-\u9fa5]+|[\u4e00-\u9fa5]+/g); + const uniques = Array.from(new Set(words)); + uniques.sort(function (a, b) { + return a.localeCompare(b, "zh-Hans-CN", { sensitivity: "accent" }); + }); + return uniques; + }; - const parse = (text, language) => { - if (language === 'zh') { - return parseChinese(text); - } - return parseEnglish(text); - }; + const parse = (text, language) => { + if (language === "zh") { + return parseChinese(text); + } + return parseEnglish(text); + }; - Scratch.extensions.register(new MathAndString()); + Scratch.extensions.register(new MathAndString()); })(Scratch); diff --git a/extensions/NOname-awa/more-comparisons.js b/extensions/NOname-awa/more-comparisons.js index 38dc0e5aa7..19a08b5e6b 100644 --- a/extensions/NOname-awa/more-comparisons.js +++ b/extensions/NOname-awa/more-comparisons.js @@ -1,534 +1,568 @@ -(function(Scratch) { - 'use strict'; - const quadrilateral = ""; - class MoreComparisons { - getInfo() { - return { - id: 'nonameawacomparisons', - name: 'More Comparisons', - color1: '#00a889', - color2: '#1e8c76', - color3: '#1e8c76', - blocks: [ - { - opcode: 'true', - blockType: Scratch.BlockType.BOOLEAN, - text: 'true', - arguments: {} - }, - { - opcode: 'false', - blockType: Scratch.BlockType.BOOLEAN, - text: 'false', - arguments: {}, - }, - { - opcode: 'boolean', - blockType: Scratch.BlockType.BOOLEAN, - text: '[a]', - arguments: { - a: { - type: Scratch.ArgumentType.STRING, - defaultValue: '' - }, - } - }, - { - opcode: 'booleanToInt', - blockType: Scratch.BlockType.REPORTER, - text: '[a]', - arguments: { - a: { - type: Scratch.ArgumentType.BOOLEAN, - }, - } - }, - '---', - { - opcode: 'equal', - blockType: Scratch.BlockType.BOOLEAN, - text: '[a] ⩵ [b]', - arguments: { - a: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'A' - }, - b: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'a' - }, - } - }, - { - opcode: 'equalNegative', - blockType: Scratch.BlockType.BOOLEAN, - text: '[a] =- [b]', - arguments: { - a: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '5' - }, - b: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '-5' - }, - } - }, - { - opcode: 'equalPlusMinus', - blockType: Scratch.BlockType.BOOLEAN, - text: '[a] =± [b]', - arguments: { - a: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '5' - }, - b: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '-5' - }, - } - }, - { - opcode: 'notEqual', - blockType: Scratch.BlockType.BOOLEAN, - text: '[a] ≠ [b]', - arguments: { - a: { - type: Scratch.ArgumentType.STRING, - defaultValue: '\n' - }, - b: { - type: Scratch.ArgumentType.STRING, - defaultValue: '\n' - }, - } - }, - { - opcode: 'almostEqual2n', - blockType: Scratch.BlockType.BOOLEAN, - text: '[a] ≈ [b]', - arguments: { - a: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '5.5' - }, - b: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '6' - }, - } - }, - { - opcode: 'almostEqual3n', - blockType: Scratch.BlockType.BOOLEAN, - text: '[a] ≈ [b] ± [c]', - arguments: { - a: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '5' - }, - b: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '6' - }, - c: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '1' - }, - } - }, - { - opcode: 'xor', - blockType: Scratch.BlockType.BOOLEAN, - text: '[a] ^ [b]', - arguments: { - a: { - type: Scratch.ArgumentType.BOOLEAN, - }, - b: { - type: Scratch.ArgumentType.BOOLEAN, - }, - } - }, - '---', - { - opcode: 'equalOrGreater', - blockType: Scratch.BlockType.BOOLEAN, - text: '[a] ≥ [b]', - arguments: { - a: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '\n' - }, - b: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '50' - }, - } - }, - { - opcode: 'equalOrLess', - blockType: Scratch.BlockType.BOOLEAN, - text: '[a] ≤ [b]', - arguments: { - a: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '\n' - }, - b: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '50' - }, - } - }, - { - opcode: 'between', - blockType: Scratch.BlockType.BOOLEAN, - text: '[a] < [b] < [c]', - arguments: { - a: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '\n' - }, - b: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '\n' - }, - c: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '\n' - }, - } - }, - { - opcode: 'betweenEqual', - blockType: Scratch.BlockType.BOOLEAN, - text: '[a] ≤ [b] ≤ [c]', - arguments: { - a: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '\n' - }, - b: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '\n' - }, - c: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '\n' - }, - } - }, - '---', - { - opcode: 'vertical', - blockType: Scratch.BlockType.BOOLEAN, - text: '[a] ⊥ [b]', - arguments: { - a: { - type: Scratch.ArgumentType.ANGLE, - defaultValue: '0' - }, - b: { - type: Scratch.ArgumentType.ANGLE, - defaultValue: '90' - }, - } - }, - { - opcode: 'segment_one', - blockType: Scratch.BlockType.BOOLEAN, - text: '⎵ ([x1],[y1]) ([x2],[y2]) = [n]', - arguments: { - x1: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '-100' - }, - y1: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - x2: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - y2: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - - n: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '100' - }, - } - }, - { - opcode: 'segment_two', - blockType: Scratch.BlockType.BOOLEAN, - text: '⎵ ([x11],[y11]) ([x12],[y12]) = ⎵ ([x21],[y21]) ([x22],[y22])', - arguments: { - x11: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '-100' - }, - y11: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - x12: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - y12: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '-100' - }, - - x21: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '100' - }, - y21: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - x22: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - y22: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '100' - }, - } - }, - { - opcode: 'segment', - blockType: Scratch.BlockType.REPORTER, - text: '⎵ ([x1],[y1]) ([x2],[y2])', - arguments: { - x1: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '-100' - }, - y1: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - x2: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - y2: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - } - }, - '---', - { - opcode: 'Squadrilateral_one', - blockType: Scratch.BlockType.BOOLEAN, - text: '[IMAGE] ([x1],[y1]) ([x2],[y2]) ([x3],[y3]) ([x4],[y4]) = [n]', - arguments: { - x1: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - y1: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '10' - }, - x2: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '10' - }, - y2: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '10' - }, - x3: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '10' - }, - y3: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - x4: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - y4: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - - n: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '100' - }, - - IMAGE: { - type: Scratch.ArgumentType.IMAGE, - dataURI: quadrilateral, - flipRTL: true - } - }, - }, - { - opcode: 'Squadrilateral', - blockType: Scratch.BlockType.REPORTER, - text: '[IMAGE] ([x1],[y1]) ([x2],[y2]) ([x3],[y3]) ([x4],[y4])', - arguments: { - x1: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - y1: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '10' - }, - x2: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '10' - }, - y2: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '10' - }, - x3: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '10' - }, - y3: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - x4: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - y4: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' - }, - - IMAGE: { - type: Scratch.ArgumentType.IMAGE, - dataURI: quadrilateral, - flipRTL: true - } - }, - } - ] - }; - } - true(){ - return true; - } - false(){ - return false; - } - boolean(args){ - return Scratch.Cast.toBoolean(args.a); - } - booleanToInt(args){ - if (Scratch.Cast.toBoolean(args.a)) { - return 1; - } - return 0; - } - equal(args) { - return (args.a == args.b); - } - equalNegative(args) { - if (isNaN(args.a) || isNaN(args.b)){ - return false; - } else { - return (args.a == (0 - args.b)); - } - } - equalPlusMinus(args) { - if (isNaN(args.a) || isNaN(args.b)){ - return false; - } else { - return (args.a == (0 - args.b)) || (args.a == args.b); - } - } - almostEqual2n(args) { - return (Math.round(args.a) == Math.round(args.b)); - } - almostEqual3n(args) { - return (Math.abs(args.a - args.b) <= args.c); - } - between(args) { - return (args.a < args.b) && (args.b < args.c); - } - betweenEqual(args) { - return (args.a <= args.b) && (args.b <= args.c); - } - notEqual(args){ - return (args.a != args.b); - } - xor(args){ - return Scratch.Cast.toBoolean(args.a) !== Scratch.Cast.toBoolean(args.b); - } - equalOrGreater(args) { - return (args.a >= args.b); - } - equalOrLess(args) { - return (args.a <= args.b); - } - vertical(args) { - if (isNaN(args.a) || isNaN(args.b)){ - return false; - } else { - return ((args.a - (args.b - 90)) % 180) == 0; - } - } - segment_one(args) { - return Math.round(Math.sqrt(Math.pow(args.x1 - args.x2, 2) + Math.pow(args.y1 - args.y2, 2))) == args.n; - } - segment_two(args) { - return Math.round(Math.sqrt(Math.pow(args.x11 - args.x12, 2) + Math.pow(args.y11 - args.y12, 2))) - == Math.round(Math.sqrt(Math.pow(args.x21 - args.x22, 2) + Math.pow(args.y21 - args.y22, 2))); - } - segment(args) { - return Math.sqrt(Math.pow(args.x1 - args.x2, 2) + Math.pow(args.y1 - args.y2, 2)); - } - Squadrilateral_one(args) { - let points = [[args.x1,args.y1], [args.x2,args.y2], [args.x3,args.y3], [args.x4,args.y4]]; - let area = 0; - let n = points.length; - for (let i = 0; i < n; i++) { - let x1 = points[i][0]; - let y1 = points[i][1]; - let x2 = points[(i + 1) % n][0]; - let y2 = points[(i + 1) % n][1]; - area += x1 * y2; - area -= x2 * y1; - } - area = Math.abs(area) / 2; - return Math.round(area) == args.n; - } - Squadrilateral(args) { - let points = [[args.x1,args.y1], [args.x2,args.y2], [args.x3,args.y3], [args.x4,args.y4]]; - let area = 0; - let n = points.length; - for (let i = 0; i < n; i++) { - let x1 = points[i][0]; - let y1 = points[i][1]; - let x2 = points[(i + 1) % n][0]; - let y2 = points[(i + 1) % n][1]; - area += x1 * y2; - area -= x2 * y1; - } - area = Math.abs(area) / 2; - return (area); - } - } - Scratch.extensions.register(new MoreComparisons()); -})(Scratch); +// Name: More Comparisons +// ID: nonameawacomparisons +// Description: More comparison blocks. +// By: NOname-awa + +(function (Scratch) { + "use strict"; + const quadrilateral = + ""; + class MoreComparisons { + getInfo() { + return { + id: "nonameawacomparisons", + name: "More Comparisons", + color1: "#00a889", + color2: "#1e8c76", + color3: "#1e8c76", + blocks: [ + { + opcode: "true", + blockType: Scratch.BlockType.BOOLEAN, + text: "true", + arguments: {}, + }, + { + opcode: "false", + blockType: Scratch.BlockType.BOOLEAN, + text: "false", + arguments: {}, + }, + { + opcode: "boolean", + blockType: Scratch.BlockType.BOOLEAN, + text: "[a]", + arguments: { + a: { + type: Scratch.ArgumentType.STRING, + defaultValue: "", + }, + }, + }, + { + opcode: "booleanToInt", + blockType: Scratch.BlockType.REPORTER, + text: "[a]", + arguments: { + a: { + type: Scratch.ArgumentType.BOOLEAN, + }, + }, + }, + "---", + { + opcode: "equal", + blockType: Scratch.BlockType.BOOLEAN, + text: "[a] ⩵ [b]", + arguments: { + a: { + type: Scratch.ArgumentType.STRING, + defaultValue: "A", + }, + b: { + type: Scratch.ArgumentType.STRING, + defaultValue: "a", + }, + }, + }, + { + opcode: "equalNegative", + blockType: Scratch.BlockType.BOOLEAN, + text: "[a] =- [b]", + arguments: { + a: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "5", + }, + b: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "-5", + }, + }, + }, + { + opcode: "equalPlusMinus", + blockType: Scratch.BlockType.BOOLEAN, + text: "[a] =± [b]", + arguments: { + a: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "5", + }, + b: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "-5", + }, + }, + }, + { + opcode: "notEqual", + blockType: Scratch.BlockType.BOOLEAN, + text: "[a] ≠ [b]", + arguments: { + a: { + type: Scratch.ArgumentType.STRING, + defaultValue: "\n", + }, + b: { + type: Scratch.ArgumentType.STRING, + defaultValue: "\n", + }, + }, + }, + { + opcode: "almostEqual2n", + blockType: Scratch.BlockType.BOOLEAN, + text: "[a] ≈ [b]", + arguments: { + a: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "5.5", + }, + b: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "6", + }, + }, + }, + { + opcode: "almostEqual3n", + blockType: Scratch.BlockType.BOOLEAN, + text: "[a] ≈ [b] ± [c]", + arguments: { + a: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "5", + }, + b: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "6", + }, + c: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "1", + }, + }, + }, + { + opcode: "xor", + blockType: Scratch.BlockType.BOOLEAN, + text: "[a] ^ [b]", + arguments: { + a: { + type: Scratch.ArgumentType.BOOLEAN, + }, + b: { + type: Scratch.ArgumentType.BOOLEAN, + }, + }, + }, + "---", + { + opcode: "equalOrGreater", + blockType: Scratch.BlockType.BOOLEAN, + text: "[a] ≥ [b]", + arguments: { + a: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "\n", + }, + b: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "50", + }, + }, + }, + { + opcode: "equalOrLess", + blockType: Scratch.BlockType.BOOLEAN, + text: "[a] ≤ [b]", + arguments: { + a: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "\n", + }, + b: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "50", + }, + }, + }, + { + opcode: "between", + blockType: Scratch.BlockType.BOOLEAN, + text: "[a] < [b] < [c]", + arguments: { + a: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "\n", + }, + b: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "\n", + }, + c: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "\n", + }, + }, + }, + { + opcode: "betweenEqual", + blockType: Scratch.BlockType.BOOLEAN, + text: "[a] ≤ [b] ≤ [c]", + arguments: { + a: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "\n", + }, + b: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "\n", + }, + c: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "\n", + }, + }, + }, + "---", + { + opcode: "vertical", + blockType: Scratch.BlockType.BOOLEAN, + text: "[a] ⊥ [b]", + arguments: { + a: { + type: Scratch.ArgumentType.ANGLE, + defaultValue: "0", + }, + b: { + type: Scratch.ArgumentType.ANGLE, + defaultValue: "90", + }, + }, + }, + { + opcode: "segment_one", + blockType: Scratch.BlockType.BOOLEAN, + text: "⎵ ([x1],[y1]) ([x2],[y2]) = [n]", + arguments: { + x1: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "-100", + }, + y1: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + x2: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + y2: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + + n: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "100", + }, + }, + }, + { + opcode: "segment_two", + blockType: Scratch.BlockType.BOOLEAN, + text: "⎵ ([x11],[y11]) ([x12],[y12]) = ⎵ ([x21],[y21]) ([x22],[y22])", + arguments: { + x11: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "-100", + }, + y11: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + x12: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + y12: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "-100", + }, + + x21: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "100", + }, + y21: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + x22: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + y22: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "100", + }, + }, + }, + { + opcode: "segment", + blockType: Scratch.BlockType.REPORTER, + text: "⎵ ([x1],[y1]) ([x2],[y2])", + arguments: { + x1: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "-100", + }, + y1: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + x2: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + y2: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + }, + }, + "---", + { + opcode: "Squadrilateral_one", + blockType: Scratch.BlockType.BOOLEAN, + text: "[IMAGE] ([x1],[y1]) ([x2],[y2]) ([x3],[y3]) ([x4],[y4]) = [n]", + arguments: { + x1: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + y1: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "10", + }, + x2: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "10", + }, + y2: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "10", + }, + x3: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "10", + }, + y3: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + x4: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + y4: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + + n: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "100", + }, + + IMAGE: { + type: Scratch.ArgumentType.IMAGE, + dataURI: quadrilateral, + flipRTL: true, + }, + }, + }, + { + opcode: "Squadrilateral", + blockType: Scratch.BlockType.REPORTER, + text: "[IMAGE] ([x1],[y1]) ([x2],[y2]) ([x3],[y3]) ([x4],[y4])", + arguments: { + x1: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + y1: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "10", + }, + x2: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "10", + }, + y2: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "10", + }, + x3: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "10", + }, + y3: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + x4: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + y4: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + + IMAGE: { + type: Scratch.ArgumentType.IMAGE, + dataURI: quadrilateral, + flipRTL: true, + }, + }, + }, + ], + }; + } + true() { + return true; + } + false() { + return false; + } + boolean(args) { + return Scratch.Cast.toBoolean(args.a); + } + booleanToInt(args) { + if (Scratch.Cast.toBoolean(args.a)) { + return 1; + } + return 0; + } + equal(args) { + return args.a == args.b; + } + equalNegative(args) { + if (isNaN(args.a) || isNaN(args.b)) { + return false; + } else { + return args.a == 0 - args.b; + } + } + equalPlusMinus(args) { + if (isNaN(args.a) || isNaN(args.b)) { + return false; + } else { + return args.a == 0 - args.b || args.a == args.b; + } + } + almostEqual2n(args) { + return Math.round(args.a) == Math.round(args.b); + } + almostEqual3n(args) { + return Math.abs(args.a - args.b) <= args.c; + } + between(args) { + return args.a < args.b && args.b < args.c; + } + betweenEqual(args) { + return args.a <= args.b && args.b <= args.c; + } + notEqual(args) { + return args.a != args.b; + } + xor(args) { + return Scratch.Cast.toBoolean(args.a) !== Scratch.Cast.toBoolean(args.b); + } + equalOrGreater(args) { + return args.a >= args.b; + } + equalOrLess(args) { + return args.a <= args.b; + } + vertical(args) { + if (isNaN(args.a) || isNaN(args.b)) { + return false; + } else { + return (args.a - (args.b - 90)) % 180 == 0; + } + } + segment_one(args) { + return ( + Math.round( + Math.sqrt( + Math.pow(args.x1 - args.x2, 2) + Math.pow(args.y1 - args.y2, 2) + ) + ) == args.n + ); + } + segment_two(args) { + return ( + Math.round( + Math.sqrt( + Math.pow(args.x11 - args.x12, 2) + Math.pow(args.y11 - args.y12, 2) + ) + ) == + Math.round( + Math.sqrt( + Math.pow(args.x21 - args.x22, 2) + Math.pow(args.y21 - args.y22, 2) + ) + ) + ); + } + segment(args) { + return Math.sqrt( + Math.pow(args.x1 - args.x2, 2) + Math.pow(args.y1 - args.y2, 2) + ); + } + Squadrilateral_one(args) { + let points = [ + [args.x1, args.y1], + [args.x2, args.y2], + [args.x3, args.y3], + [args.x4, args.y4], + ]; + let area = 0; + let n = points.length; + for (let i = 0; i < n; i++) { + let x1 = points[i][0]; + let y1 = points[i][1]; + let x2 = points[(i + 1) % n][0]; + let y2 = points[(i + 1) % n][1]; + area += x1 * y2; + area -= x2 * y1; + } + area = Math.abs(area) / 2; + return Math.round(area) == args.n; + } + Squadrilateral(args) { + let points = [ + [args.x1, args.y1], + [args.x2, args.y2], + [args.x3, args.y3], + [args.x4, args.y4], + ]; + let area = 0; + let n = points.length; + for (let i = 0; i < n; i++) { + let x1 = points[i][0]; + let y1 = points[i][1]; + let x2 = points[(i + 1) % n][0]; + let y2 = points[(i + 1) % n][1]; + area += x1 * y2; + area -= x2 * y1; + } + area = Math.abs(area) / 2; + return area; + } + } + Scratch.extensions.register(new MoreComparisons()); +})(Scratch); diff --git a/extensions/NOname-awa/regular-expression.js b/extensions/NOname-awa/regular-expression.js index 9e8aadcdcc..0d79427d1c 100644 --- a/extensions/NOname-awa/regular-expression.js +++ b/extensions/NOname-awa/regular-expression.js @@ -1,241 +1,246 @@ (function (Scratch) { - 'use strict'; - let args = ["", "", "",""]; - class RegularExpression { - getInfo() { - return { - color1: '#0081d3', - color2: '#0067a9', - color3: '#0067a9', - id: 'nonameawaregex', - name: 'Regular Expression', - blocks: [ - { - opcode: 'set', - blockType: Scratch.BlockType.COMMAND, - text: 'set a: [one] b: [two] c: [three]', - arguments: { - one: { - type: Scratch.ArgumentType.STRING, - defaultValue: '' - }, - two: { - type: Scratch.ArgumentType.STRING, - defaultValue: '' - }, - three: { - type: Scratch.ArgumentType.STRING, - defaultValue: '' - } - } - }, - { - opcode: 'matchText', - blockType: Scratch.BlockType.BOOLEAN, - text: 'matchText: b in a [flags]', - arguments: { - flags: { - type: Scratch.ArgumentType.STRING, - menu: 'flags' - } - } - }, - { - opcode: 'searchText', - blockType: Scratch.BlockType.REPORTER, - text: 'searchText: b in a', - }, - { - opcode: 'replaceText', - blockType: Scratch.BlockType.REPORTER, - text: 'replaceText: b in a with c', - }, '---', - { - opcode: 'matchTextWithPattern', - blockType: Scratch.BlockType.BOOLEAN, - text: 'match [pattern] in [text] - [flags]', - arguments: { - text: { - type: Scratch.ArgumentType.STRING, - defaultValue: '' - }, - pattern: { - type: Scratch.ArgumentType.STRING, - defaultValue: '' - }, - flags: { - type: Scratch.ArgumentType.STRING, - menu: 'flags' - } - } - }, - { - opcode: 'searchTextWithPattern', - blockType: Scratch.BlockType.REPORTER, - text: 'search [pattern] in [text] ', - arguments: { - text: { - type: Scratch.ArgumentType.STRING, - defaultValue: '' - }, - pattern: { - type: Scratch.ArgumentType.STRING, - defaultValue: '' - } - } - }, - { - opcode: 'replaceTextWithPattern', - blockType: Scratch.BlockType.REPORTER, - text: 'replace [pattern] in [text] with [replacement]', - arguments: { - text: { - type: Scratch.ArgumentType.STRING, - defaultValue: '' - }, - pattern: { - type: Scratch.ArgumentType.STRING, - defaultValue: '' - }, - replacement: { - type: Scratch.ArgumentType.STRING, - defaultValue: '' - } - } - }, '---', - { - opcode: 'constant', - blockType: Scratch.BlockType.REPORTER, - text: '[constant]', - arguments: { - constant: { - type: Scratch.ArgumentType.STRING, - menu: 'constant' - } - } - }, '---', - { - opcode: 'text', - blockType: Scratch.BlockType.REPORTER, - text: 'a', - }, - { - opcode: 'pattern', - blockType: Scratch.BlockType.REPORTER, - text: 'b', - }, - { - opcode: 'attach', - blockType: Scratch.BlockType.REPORTER, - text: 'c', - } - ], - menus: { - flags: { - acceptReporters: false, - items: [ - { - text: 'global (g)', - value: 'g' - }, - { - text: 'ignore case (i)', - value: 'i' - } - ] - }, - constant: { - acceptReporters: true, - items: [ - { - text: 'english', - value: '[a-zA-Z]' - }, - { - text: 'english uppercase', - value: '[A-Z]' - }, - { - text: 'english lowercase', - value: '[a-z]' - }, - { - text: 'number', - value: '[0-9]' - }, - { - text: 'numeric integer', - value: '^-?[1-9]\\d*$' - }, - { - text: 'positive integer', - value: '^[1-9]\\d*$' - }, - { - text: 'negative integer', - value: '^-[1-9]\\d*$' - }, - { - text: 'non-negative integers', - value: '^[1-9]\\d*|0$' - }, - { - text: 'non-positive integer', - value: '^-[1-9]\\d*|0$' - }, - { - text: 'chinese', - value: '[\u4e00-\u9fa5]' - }, - { - text: 'double-byte', - value: '[^\x00-\xff]' - } - ] - } - } - }; - } - set({ one, two, three }) { - args[1] = one; - args[2] = two; - args[3] = three; - } - matchText({ flags }) { - const regex = new RegExp(args[2], flags); - return regex.test(args[1]); - } - searchText() { - return args[1].toString().search(args[2].toString()); - } - replaceText() { - return args[1].toString().replace(args[2].toString(), args[3].toString()); - } + "use strict"; + let args = ["", "", "", ""]; + class RegularExpression { + getInfo() { + return { + color1: "#0081d3", + color2: "#0067a9", + color3: "#0067a9", + id: "nonameawaregex", + name: "Regular Expression", + blocks: [ + { + opcode: "set", + blockType: Scratch.BlockType.COMMAND, + text: "set a: [one] b: [two] c: [three]", + arguments: { + one: { + type: Scratch.ArgumentType.STRING, + defaultValue: "", + }, + two: { + type: Scratch.ArgumentType.STRING, + defaultValue: "", + }, + three: { + type: Scratch.ArgumentType.STRING, + defaultValue: "", + }, + }, + }, + { + opcode: "matchText", + blockType: Scratch.BlockType.BOOLEAN, + text: "matchText: b in a [flags]", + arguments: { + flags: { + type: Scratch.ArgumentType.STRING, + menu: "flags", + }, + }, + }, + { + opcode: "searchText", + blockType: Scratch.BlockType.REPORTER, + text: "searchText: b in a", + }, + { + opcode: "replaceText", + blockType: Scratch.BlockType.REPORTER, + text: "replaceText: b in a with c", + }, + "---", + { + opcode: "matchTextWithPattern", + blockType: Scratch.BlockType.BOOLEAN, + text: "match [pattern] in [text] - [flags]", + arguments: { + text: { + type: Scratch.ArgumentType.STRING, + defaultValue: "", + }, + pattern: { + type: Scratch.ArgumentType.STRING, + defaultValue: "", + }, + flags: { + type: Scratch.ArgumentType.STRING, + menu: "flags", + }, + }, + }, + { + opcode: "searchTextWithPattern", + blockType: Scratch.BlockType.REPORTER, + text: "search [pattern] in [text] ", + arguments: { + text: { + type: Scratch.ArgumentType.STRING, + defaultValue: "", + }, + pattern: { + type: Scratch.ArgumentType.STRING, + defaultValue: "", + }, + }, + }, + { + opcode: "replaceTextWithPattern", + blockType: Scratch.BlockType.REPORTER, + text: "replace [pattern] in [text] with [replacement]", + arguments: { + text: { + type: Scratch.ArgumentType.STRING, + defaultValue: "", + }, + pattern: { + type: Scratch.ArgumentType.STRING, + defaultValue: "", + }, + replacement: { + type: Scratch.ArgumentType.STRING, + defaultValue: "", + }, + }, + }, + "---", + { + opcode: "constant", + blockType: Scratch.BlockType.REPORTER, + text: "[constant]", + arguments: { + constant: { + type: Scratch.ArgumentType.STRING, + menu: "constant", + }, + }, + }, + "---", + { + opcode: "text", + blockType: Scratch.BlockType.REPORTER, + text: "a", + }, + { + opcode: "pattern", + blockType: Scratch.BlockType.REPORTER, + text: "b", + }, + { + opcode: "attach", + blockType: Scratch.BlockType.REPORTER, + text: "c", + }, + ], + menus: { + flags: { + acceptReporters: false, + items: [ + { + text: "global (g)", + value: "g", + }, + { + text: "ignore case (i)", + value: "i", + }, + ], + }, + constant: { + acceptReporters: true, + items: [ + { + text: "english", + value: "[a-zA-Z]", + }, + { + text: "english uppercase", + value: "[A-Z]", + }, + { + text: "english lowercase", + value: "[a-z]", + }, + { + text: "number", + value: "[0-9]", + }, + { + text: "numeric integer", + value: "^-?[1-9]\\d*$", + }, + { + text: "positive integer", + value: "^[1-9]\\d*$", + }, + { + text: "negative integer", + value: "^-[1-9]\\d*$", + }, + { + text: "non-negative integers", + value: "^[1-9]\\d*|0$", + }, + { + text: "non-positive integer", + value: "^-[1-9]\\d*|0$", + }, + { + text: "chinese", + value: "[\u4e00-\u9fa5]", + }, + { + text: "double-byte", + value: "[^\x00-\xff]", + }, + ], + }, + }, + }; + } + set({ one, two, three }) { + args[1] = one; + args[2] = two; + args[3] = three; + } + matchText({ flags }) { + const regex = new RegExp(args[2], flags); + return regex.test(args[1]); + } + searchText() { + return args[1].toString().search(args[2].toString()); + } + replaceText() { + return args[1].toString().replace(args[2].toString(), args[3].toString()); + } - matchTextWithPattern({ text, pattern, flags }) { - const regex = new RegExp(pattern, flags); - return regex.test(text); - } - searchTextWithPattern({ text, pattern }) { - return text.toString().search(pattern.toString()); - } - replaceTextWithPattern({ text, pattern, replacement }) { - return text.toString().replace(pattern.toString(), replacement.toString()); - } + matchTextWithPattern({ text, pattern, flags }) { + const regex = new RegExp(pattern, flags); + return regex.test(text); + } + searchTextWithPattern({ text, pattern }) { + return text.toString().search(pattern.toString()); + } + replaceTextWithPattern({ text, pattern, replacement }) { + return text + .toString() + .replace(pattern.toString(), replacement.toString()); + } - constant({ constant }) { - return constant; - } + constant({ constant }) { + return constant; + } - text() { - return args[1]; - } - pattern() { - return args[2]; - } - attach() { - return args[3]; - } + text() { + return args[1]; + } + pattern() { + return args[2]; + } + attach() { + return args[3]; } - Scratch.extensions.register(new RegularExpression()); + } + Scratch.extensions.register(new RegularExpression()); })(Scratch); diff --git a/extensions/NOname-awa/sort-unique-words.js b/extensions/NOname-awa/sort-unique-words.js index f900d9ee26..d4bcc388b7 100644 --- a/extensions/NOname-awa/sort-unique-words.js +++ b/extensions/NOname-awa/sort-unique-words.js @@ -1,78 +1,78 @@ -(function(Scratch) { - 'use strict'; - - const parseEnglish = (text) => { - const words = text.toLowerCase().match(/\b\w+\b/g); - const uniques = Array.from(new Set(words)); - uniques.sort(); - return uniques; - }; - - const parseChinese = (text) => { - const words = text.match(/[^\u4e00-\u9fa5]+|[\u4e00-\u9fa5]+/g); - const uniques = Array.from(new Set(words)); - uniques.sort(function(a, b) { - return a.localeCompare(b, 'zh-Hans-CN', {sensitivity: 'accent'}); - }); - return uniques; - }; - - const parse = (text, language) => { - if (language === 'zh') { - return parseChinese(text); - } - return parseEnglish(text); - }; - - class SortUniqueWords { - getInfo() { - return { - id: 'nonameawasortuniquewords', - name: 'Sort Unique Words', - color1: '#5a8b9e', - color2: '#427081', - color3: '#427081', - blocks: [ - { - opcode: 'words', - blockType: Scratch.BlockType.REPORTER, - disableMonitor: true, - text: 'sort unique words in [text] as [language]', - arguments: { - text: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'movie dog restaurant book school' - }, - language: { - type: Scratch.ArgumentType.STRING, - menu: 'language' - } - } - } - ], - menus: { - language: { - acceptReporters: true, - items: [ - { - text: 'English (en)', - value: 'en' - }, - { - text: 'Chinese (zh)', - value: 'zh' - } - ] - } - } - }; - } - words(args) { - const text = Scratch.Cast.toString(args.text); - const words = parse(text, args.language); - return words.join(' '); - } - } - - Scratch.extensions.register(new SortUniqueWords()); -})(Scratch); +(function (Scratch) { + "use strict"; + + const parseEnglish = (text) => { + const words = text.toLowerCase().match(/\b\w+\b/g); + const uniques = Array.from(new Set(words)); + uniques.sort(); + return uniques; + }; + + const parseChinese = (text) => { + const words = text.match(/[^\u4e00-\u9fa5]+|[\u4e00-\u9fa5]+/g); + const uniques = Array.from(new Set(words)); + uniques.sort(function (a, b) { + return a.localeCompare(b, "zh-Hans-CN", { sensitivity: "accent" }); + }); + return uniques; + }; + + const parse = (text, language) => { + if (language === "zh") { + return parseChinese(text); + } + return parseEnglish(text); + }; + + class SortUniqueWords { + getInfo() { + return { + id: "nonameawasortuniquewords", + name: "Sort Unique Words", + color1: "#5a8b9e", + color2: "#427081", + color3: "#427081", + blocks: [ + { + opcode: "words", + blockType: Scratch.BlockType.REPORTER, + disableMonitor: true, + text: "sort unique words in [text] as [language]", + arguments: { + text: { + type: Scratch.ArgumentType.STRING, + defaultValue: "movie dog restaurant book school", + }, + language: { + type: Scratch.ArgumentType.STRING, + menu: "language", + }, + }, + }, + ], + menus: { + language: { + acceptReporters: true, + items: [ + { + text: "English (en)", + value: "en", + }, + { + text: "Chinese (zh)", + value: "zh", + }, + ], + }, + }, + }; + } + words(args) { + const text = Scratch.Cast.toString(args.text); + const words = parse(text, args.language); + return words.join(" "); + } + } + + Scratch.extensions.register(new SortUniqueWords()); +})(Scratch); diff --git a/extensions/NexusKitten/controlcontrols.js b/extensions/NexusKitten/controlcontrols.js new file mode 100644 index 0000000000..3a4910012e --- /dev/null +++ b/extensions/NexusKitten/controlcontrols.js @@ -0,0 +1,165 @@ +// Name: Control Controls +// ID: nkcontrols +// Description: Show and hide the project's controls. +// By: NamelessCat + +(function (Scratch) { + "use strict"; + + if (!Scratch.extensions.unsandboxed) { + throw new Error("Control Controls must run unsandboxed"); + } + + var fullScreen; + var greenFlag; + var pauseButton; + var stopButton; + + const getButtons = () => { + fullScreen = undefined; + greenFlag = undefined; + pauseButton = undefined; + stopButton = undefined; + + const rightButtons = document.querySelectorAll( + '[class*="stage-header_stage-button_"]' + ); + fullScreen = rightButtons[rightButtons.length - 1]; + if (!fullScreen) { + fullScreen = + document.querySelector(".fullscreen-button") || + document.querySelector(".standalone-fullscreen-button"); + } + + greenFlag = + document.querySelector('[class*="green-flag_green-flag_"]') || + document.querySelector(".green-flag-button"); + pauseButton = + document.querySelector(".pause-btn") || + document.querySelector(".pause-button"); + stopButton = + document.querySelector('[class*="stop-all_stop-all_"]') || + document.querySelector(".stop-all-button"); + }; + + class controlcontrols { + getInfo() { + return { + id: "nkcontrols", + name: "Control Controls", + color1: "#ffab19", + color2: "#ec9c13", + color3: "#b87d17", + blocks: [ + { + opcode: "showOption", + blockType: Scratch.BlockType.COMMAND, + text: "show [OPTION]", + arguments: { + OPTION: { + type: Scratch.ArgumentType.STRING, + menu: "OPTION", + }, + }, + }, + { + opcode: "hideOption", + blockType: Scratch.BlockType.COMMAND, + text: "hide [OPTION]", + arguments: { + OPTION: { + type: Scratch.ArgumentType.STRING, + menu: "OPTION", + }, + }, + }, + "---", + { + opcode: "optionShown", + blockType: Scratch.BlockType.BOOLEAN, + text: "[OPTION] shown?", + arguments: { + OPTION: { + type: Scratch.ArgumentType.STRING, + menu: "OPTION", + }, + }, + }, + "---", + { + opcode: "optionExists", + blockType: Scratch.BlockType.BOOLEAN, + text: "[OPTION] exists?", + arguments: { + OPTION: { + type: Scratch.ArgumentType.STRING, + menu: "OPTION", + }, + }, + }, + ], + menus: { + OPTION: { + acceptReporters: true, + items: ["green flag", "pause", "stop", "fullscreen"], + }, + }, + }; + } + + showOption(args) { + getButtons(); + if (args.OPTION === "green flag" && greenFlag) { + greenFlag.style.display = "block"; + } else if (args.OPTION === "pause" && pauseButton) { + pauseButton.style.display = "block"; + } else if (args.OPTION === "stop" && stopButton) { + stopButton.style.display = "block"; + } else if (args.OPTION === "fullscreen" && fullScreen) { + fullScreen.style.display = "block"; + } + } + + hideOption(args) { + getButtons(); + if (args.OPTION === "green flag" && greenFlag) { + greenFlag.style.display = "none"; + } else if (args.OPTION === "pause" && pauseButton) { + pauseButton.style.display = "none"; + } else if (args.OPTION === "stop" && stopButton) { + stopButton.style.display = "none"; + } else if (args.OPTION === "fullscreen" && fullScreen) { + fullScreen.style.display = "none"; + } + } + + optionShown(args) { + getButtons(); + if (args.OPTION === "green flag" && greenFlag) { + return greenFlag.style.display !== "none"; + } else if (args.OPTION === "pause" && pauseButton) { + return pauseButton.style.display !== "none"; + } else if (args.OPTION === "stop" && stopButton) { + return stopButton.style.display !== "none"; + } else if (args.OPTION === "fullscreen" && fullScreen) { + return fullScreen.style.display !== "none"; + } + return false; + } + + optionExists(args) { + getButtons(); + if (args.OPTION === "green flag" && greenFlag) { + return true; + } else if (args.OPTION === "pause" && pauseButton) { + return true; + } else if (args.OPTION === "stop" && stopButton) { + return true; + } else if (args.OPTION === "fullscreen" && fullScreen) { + return true; + } + return false; + } + } + Scratch.extensions.register(new controlcontrols()); +})(Scratch); diff --git a/extensions/NexusKitten/moremotion.js b/extensions/NexusKitten/moremotion.js new file mode 100644 index 0000000000..d3aee71d53 --- /dev/null +++ b/extensions/NexusKitten/moremotion.js @@ -0,0 +1,392 @@ +// Name: More Motion +// ID: nkmoremotion +// Description: More motion-related blocks. +// By: NamelessCat + +(function (Scratch) { + "use strict"; + + if (!Scratch.extensions.unsandboxed) { + throw new Error("More Motion must run unsandboxed"); + } + + // @ts-expect-error - not typed yet + const Rectangle = Scratch.vm.renderer.exports.Rectangle; + + class nkmoremotion { + getInfo() { + return { + id: "nkmoremotion", + name: "More Motion", + color1: "#4c97ff", + color2: "#3373cc", + blocks: [ + { + filter: [Scratch.TargetType.STAGE], + blockType: Scratch.BlockType.LABEL, + text: "Stage selected: no motion blocks", + }, + { + filter: [Scratch.TargetType.SPRITE], + opcode: "changexy", + blockType: Scratch.BlockType.COMMAND, + text: "change x: [X] y: [Y]", + arguments: { + X: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + Y: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + }, + }, + { + filter: [Scratch.TargetType.SPRITE], + opcode: "pointto", + blockType: Scratch.BlockType.COMMAND, + text: "point towards x: [X] y: [Y]", + arguments: { + X: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + Y: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + }, + }, + { + filter: [Scratch.TargetType.SPRITE], + opcode: "rotationStyle", + blockType: Scratch.BlockType.REPORTER, + text: "rotation style", + disableMonitor: true, + }, + "---", + { + filter: [Scratch.TargetType.SPRITE], + opcode: "fence", + blockType: Scratch.BlockType.COMMAND, + text: "manually fence", + }, + "---", + { + filter: [Scratch.TargetType.SPRITE], + opcode: "steptowards", + blockType: Scratch.BlockType.COMMAND, + text: "move [STEPS] steps towards x: [X] y: [Y]", + arguments: { + STEPS: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "10", + }, + X: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + Y: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + }, + }, + { + filter: [Scratch.TargetType.SPRITE], + opcode: "tweentowards", + blockType: Scratch.BlockType.COMMAND, + text: "move [PERCENT]% of the way to x: [X] y: [Y]", + arguments: { + PERCENT: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "10", + }, + X: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + Y: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + }, + }, + "---", + { + filter: [Scratch.TargetType.SPRITE], + opcode: "directionto", + blockType: Scratch.BlockType.REPORTER, + text: "direction to x: [X] y: [Y]", + arguments: { + X: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + Y: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + }, + }, + { + filter: [Scratch.TargetType.SPRITE], + opcode: "distanceto", + blockType: Scratch.BlockType.REPORTER, + text: "distance from x: [X] y: [Y]", + arguments: { + X: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + Y: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + }, + }, + { + filter: [Scratch.TargetType.SPRITE], + opcode: "spritewh", + blockType: Scratch.BlockType.REPORTER, + text: "sprite [WHAT]", + disableMonitor: true, + arguments: { + WHAT: { + type: Scratch.ArgumentType.STRING, + menu: "WHAT", + }, + }, + }, + "---", + { + filter: [Scratch.TargetType.SPRITE], + opcode: "touchingxy", + blockType: Scratch.BlockType.BOOLEAN, + text: "touching x: [X] y: [Y]?", + arguments: { + X: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + Y: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + }, + }, + { + filter: [Scratch.TargetType.SPRITE], + opcode: "touchingrect", + blockType: Scratch.BlockType.BOOLEAN, + text: "touching rectangle x1: [X1] y1: [Y1] x2: [X2] y2: [Y2]?", + arguments: { + X1: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "-100", + }, + Y1: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "-100", + }, + X2: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "100", + }, + Y2: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "100", + }, + }, + }, + ], + menus: { + WHAT: { + acceptreporters: true, + items: ["width", "height", "costume width", "costume height"], + }, + }, + }; + } + + changexy(args, util) { + const x = Scratch.Cast.toNumber(args.X); + const y = Scratch.Cast.toNumber(args.Y); + util.target.setXY(util.target.x + x, util.target.y + y); + } + + // LORAX APPROVED + pointto(args, util) { + const x = Scratch.Cast.toNumber(args.X); + const y = Scratch.Cast.toNumber(args.Y); + if (util.target.y > y) { + util.target.setDirection( + (180 / Math.PI) * + Math.atan((x - util.target.x) / (y - util.target.y)) + + 180 + ); + } else { + util.target.setDirection( + (180 / Math.PI) * Math.atan((x - util.target.x) / (y - util.target.y)) + ); + } + } + + rotationStyle(args, util) { + return util.target.rotationStyle; + } + + fence(args, util) { + const newpos = Scratch.vm.renderer.getFencedPositionOfDrawable( + util.target.drawableID, + [util.target.x, util.target.y] + ); + util.target.setXY(newpos[0], newpos[1]); + } + + directionto(args, util) { + const x = Scratch.Cast.toNumber(args.X); + const y = Scratch.Cast.toNumber(args.Y); + if (util.target.y > y) { + return ( + (180 / Math.PI) * + Math.atan((x - util.target.x) / (y - util.target.y)) + + 180 + ); + } else { + return ( + (180 / Math.PI) * Math.atan((x - util.target.x) / (y - util.target.y)) + ); + } + } + + distanceto(args, util) { + const x = Scratch.Cast.toNumber(args.X); + const y = Scratch.Cast.toNumber(args.Y); + // Shoutout to Pythagoras! + return Math.sqrt((x - util.target.x) ** 2 + (y - util.target.y) ** 2); + } + + steptowards(args, util) { + const x = Scratch.Cast.toNumber(args.X); + const y = Scratch.Cast.toNumber(args.Y); + const steps = Scratch.Cast.toNumber(args.STEPS); + const val = + steps / Math.sqrt((x - util.target.x) ** 2 + (y - util.target.y) ** 2); + if (val >= 1) { + util.target.setXY(x, y); + } else { + util.target.setXY( + (x - util.target.x) * val + util.target.x, + (y - util.target.y) * val + util.target.y + ); + } + } + + tweentowards(args, util) { + const x = Scratch.Cast.toNumber(args.X); + const y = Scratch.Cast.toNumber(args.Y); + const val = Scratch.Cast.toNumber(args.PERCENT); + // Essentially a smooth glide script. + util.target.setXY( + (x - util.target.x) * (val / 100) + util.target.x, + (y - util.target.y) * (val / 100) + util.target.y + ); + } + + touchingrect(args, util) { + let left = Scratch.Cast.toNumber(args.X1); + let right = Scratch.Cast.toNumber(args.X2); + let bottom = Scratch.Cast.toNumber(args.Y1); + let top = Scratch.Cast.toNumber(args.Y2); + + // Fix argument order if they got it backwards + if (left > right) { + let temp = left; + left = right; + right = temp; + } + if (bottom > top) { + let temp = bottom; + bottom = top; + bottom = temp; + } + + const drawable = + Scratch.vm.renderer._allDrawables[util.target.drawableID]; + if (!drawable) { + return false; + } + + // See renderer.isTouchingDrawables + + const drawableBounds = drawable.getFastBounds(); + drawableBounds.snapToInt(); + + const containsBounds = new Rectangle(); + containsBounds.initFromBounds(left, right, bottom, top); + containsBounds.snapToInt(); + + if (!containsBounds.intersects(drawableBounds)) { + return false; + } + + drawable.updateCPURenderAttributes(); + + const intersectingBounds = Rectangle.intersect( + drawableBounds, + containsBounds + ); + for (let x = intersectingBounds.left; x < intersectingBounds.right; x++) { + for ( + let y = intersectingBounds.bottom; + y < intersectingBounds.top; + y++ + ) { + // technically should be a twgl vec3, but does not actually need to be + if (drawable.isTouching([x, y])) { + return true; + } + } + } + return false; + } + + touchingxy(args, util) { + const x = Scratch.Cast.toNumber(args.X); + const y = Scratch.Cast.toNumber(args.Y); + const drawable = + Scratch.vm.renderer._allDrawables[util.target.drawableID]; + if (!drawable) { + return false; + } + // Position should technically be a twgl vec3, but it doesn't actually need to be + drawable.updateCPURenderAttributes(); + return drawable.isTouching([x, y]); + } + + spritewh(args, util) { + if (args.WHAT === "width" || args.WHAT === "height") { + const bounds = Scratch.vm.renderer.getBounds(util.target.drawableID); + if (args.WHAT === "width") { + return Math.ceil(bounds.width); + } else { + return Math.ceil(bounds.height); + } + } else if ( + args.WHAT === "costume width" || + args.WHAT === "costume height" + ) { + const costume = util.target.sprite.costumes[util.target.currentCostume]; + if (args.WHAT === "costume width") { + return Math.ceil(costume.size[0]); + } else { + return Math.ceil(costume.size[1]); + } + } + } + } + + Scratch.extensions.register(new nkmoremotion()); +})(Scratch); diff --git a/extensions/NexusKitten/sgrab.js b/extensions/NexusKitten/sgrab.js index 6443180b21..6066bea0d2 100644 --- a/extensions/NexusKitten/sgrab.js +++ b/extensions/NexusKitten/sgrab.js @@ -1,249 +1,269 @@ -(function(Scratch) { - 'use strict'; - - if (!Scratch.extensions.unsandboxed) { - throw new Error('This Extension must run unsandboxed'); - } - - const icon = ""; - - class nexuskittensgrab { - getInfo() { - return { - id: 'nexuskittensgrab', - name: 'S-Grab', - menuIconURI: icon, - color1: '#ECA90B', - color2: '#EBAF00', - blocks: [ - { - opcode: 'usergrab', - blockType: Scratch.BlockType.REPORTER, - text: 'grab [WHAT] count of user [WHO]', - arguments: { - WHAT: { - type: Scratch.ArgumentType.STRING, - menu: 'WHAT' - }, - WHO: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'john' - } - } - }, - { - opcode: 'rankusergrab', - blockType: Scratch.BlockType.REPORTER, - text: 'global [WHAT] ranking for [WHO]', - arguments: { - WHAT: { - type: Scratch.ArgumentType.STRING, - menu: 'WHAT2' - }, - WHO: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'john' - } - } - }, - { - opcode: 'usergrab2', - blockType: Scratch.BlockType.REPORTER, - text: '[WHAT] of user [WHO]', - arguments: { - WHAT: { - type: Scratch.ArgumentType.STRING, - menu: 'WHAT5' - }, - WHO: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'john' - } - } - }, - '---', - { - opcode: 'projectgrab', - blockType: Scratch.BlockType.REPORTER, - text: 'grab [WHAT] count of project id [WHO]', - arguments: { - WHAT: { - type: Scratch.ArgumentType.STRING, - menu: 'WHAT3' - }, - WHO: { - type: Scratch.ArgumentType.STRING, - defaultValue: '717954208' - } - } - }, - { - opcode: 'rankprojectgrab', - blockType: Scratch.BlockType.REPORTER, - text: 'global [WHAT] ranking for project id [WHO]', - arguments: { - WHAT: { - type: Scratch.ArgumentType.STRING, - menu: 'WHAT4' - }, - WHO: { - type: Scratch.ArgumentType.STRING, - defaultValue: '717954208' - } - } - }, - { - opcode: 'idtoname', - blockType: Scratch.BlockType.REPORTER, - text: 'name of project id [WHO]', - arguments: { - WHO: { - type: Scratch.ArgumentType.STRING, - defaultValue: '717954208' - } - } - }, - { - opcode: 'idtoowner', - blockType: Scratch.BlockType.REPORTER, - text: 'creator of project id [WHO]', - arguments: { - WHO: { - type: Scratch.ArgumentType.STRING, - defaultValue: '717954208' - } - } - }, - ], - menus: { - WHAT: { - acceptReporters: true, - items: ['follower', 'following'] - }, - WHAT2: { - acceptReporters: true, - items: ['follower', 'love', 'favorite', 'view'] - }, - WHAT3: { - acceptReporters: true, - items: ['love', 'favorite', 'view'] - }, - WHAT4: { - acceptReporters: true, - items: ['love', 'favorite', 'view'] - }, - WHAT5: { - acceptReporters: true, - items: ['about me', 'wiwo', 'location', 'status'] - } - } - }; - } - async usergrab(args) { - try { - const response = await Scratch.fetch('https://scratchdb.lefty.one/v3/user/info/' + args.WHO); - const jsonData = await response.json(); - if (args.WHAT === 'follower') { - return jsonData.statistics.followers; - } else if (args.WHAT === 'following') { - return jsonData.statistics.following; - } else { - return ''; - } - } catch (error){ - return ''; - } - } - async rankusergrab(args) { - try { - const response = await Scratch.fetch('https://scratchdb.lefty.one/v3/user/info/' + args.WHO); - const jsonData = await response.json(); - if (args.WHAT === 'follower') { - return jsonData.statistics.ranks.followers; - } else if (args.WHAT === 'love') { - return jsonData.statistics.ranks.loves; - } else if (args.WHAT === 'favorite') { - return jsonData.statistics.ranks.favorites; - } else if (args.WHAT === 'view') { - return jsonData.statistics.ranks.views; - } else { - return ''; - } - } catch (error){ - return ''; - } - } - async usergrab2(args) { - try { - const response = await Scratch.fetch('https://scratchdb.lefty.one/v3/user/info/' + args.WHO); - const jsonData = await response.json(); - if (args.WHAT === 'about me') { - return jsonData.bio; - } else if (args.WHAT === 'wiwo') { - return jsonData.work; - } else if (args.WHAT === 'location') { - return jsonData.country; - } else if (args.WHAT === 'status') { - return jsonData.status; - } else { - return ''; - } - } catch (error){ - return ''; - } - } - async projectgrab(args) { - try { - const response = await Scratch.fetch('https://scratchdb.lefty.one/v3/project/info/' + args.WHO); - const jsonData = await response.json(); - if (args.WHAT === 'love') { - return jsonData.statistics.loves; - } else if (args.WHAT === 'favorite') { - return jsonData.statistics.favorites; - } else if (args.WHAT === 'view') { - return jsonData.statistics.views; - } else { - return ''; - } - } catch (error){ - return ''; - } - } - async rankprojectgrab(args) { - try { - const response = await Scratch.fetch('https://scratchdb.lefty.one/v3/project/info/' + args.WHO); - const jsonData = await response.json(); - if (args.WHAT === 'love') { - return jsonData.statistics.ranks.loves; - } else if (args.WHAT === 'favorite') { - return jsonData.statistics.ranks.favorites; - } else if (args.WHAT === 'view') { - return jsonData.statistics.ranks.views; - } else { - return ''; - } - } catch (error){ - return ''; - } - } - async idtoname(args) { - try { - const response = await Scratch.fetch('https://scratchdb.lefty.one/v3/project/info/' + args.WHO); - const jsonData = await response.json(); - return jsonData.title; - } catch (error){ - return ''; - } - } - async idtoowner(args) { - try { - const response = await Scratch.fetch('https://scratchdb.lefty.one/v3/project/info/' + args.WHO); - const jsonData = await response.json(); - return jsonData.username; - } catch (error){ - return ''; - } - } - } - Scratch.extensions.register(new nexuskittensgrab()); -})(Scratch); +// Name: S-Grab +// ID: nexuskittensgrab +// Description: Get information about Scratch projects and Scratch users. +// By: NamelessCat + +(function (Scratch) { + "use strict"; + + if (!Scratch.extensions.unsandboxed) { + throw new Error("This Extension must run unsandboxed"); + } + + const icon = + ""; + + class nexuskittensgrab { + getInfo() { + return { + id: "nexuskittensgrab", + name: "S-Grab", + menuIconURI: icon, + color1: "#ECA90B", + color2: "#EBAF00", + blocks: [ + { + opcode: "usergrab", + blockType: Scratch.BlockType.REPORTER, + text: "grab [WHAT] count of user [WHO]", + arguments: { + WHAT: { + type: Scratch.ArgumentType.STRING, + menu: "WHAT", + }, + WHO: { + type: Scratch.ArgumentType.STRING, + defaultValue: "john", + }, + }, + }, + { + opcode: "rankusergrab", + blockType: Scratch.BlockType.REPORTER, + text: "global [WHAT] ranking for [WHO]", + arguments: { + WHAT: { + type: Scratch.ArgumentType.STRING, + menu: "WHAT2", + }, + WHO: { + type: Scratch.ArgumentType.STRING, + defaultValue: "john", + }, + }, + }, + { + opcode: "usergrab2", + blockType: Scratch.BlockType.REPORTER, + text: "[WHAT] of user [WHO]", + arguments: { + WHAT: { + type: Scratch.ArgumentType.STRING, + menu: "WHAT5", + }, + WHO: { + type: Scratch.ArgumentType.STRING, + defaultValue: "john", + }, + }, + }, + "---", + { + opcode: "projectgrab", + blockType: Scratch.BlockType.REPORTER, + text: "grab [WHAT] count of project id [WHO]", + arguments: { + WHAT: { + type: Scratch.ArgumentType.STRING, + menu: "WHAT3", + }, + WHO: { + type: Scratch.ArgumentType.STRING, + defaultValue: "717954208", + }, + }, + }, + { + opcode: "rankprojectgrab", + blockType: Scratch.BlockType.REPORTER, + text: "global [WHAT] ranking for project id [WHO]", + arguments: { + WHAT: { + type: Scratch.ArgumentType.STRING, + menu: "WHAT4", + }, + WHO: { + type: Scratch.ArgumentType.STRING, + defaultValue: "717954208", + }, + }, + }, + { + opcode: "idtoname", + blockType: Scratch.BlockType.REPORTER, + text: "name of project id [WHO]", + arguments: { + WHO: { + type: Scratch.ArgumentType.STRING, + defaultValue: "717954208", + }, + }, + }, + { + opcode: "idtoowner", + blockType: Scratch.BlockType.REPORTER, + text: "creator of project id [WHO]", + arguments: { + WHO: { + type: Scratch.ArgumentType.STRING, + defaultValue: "717954208", + }, + }, + }, + ], + menus: { + WHAT: { + acceptReporters: true, + items: ["follower", "following"], + }, + WHAT2: { + acceptReporters: true, + items: ["follower", "love", "favorite", "view"], + }, + WHAT3: { + acceptReporters: true, + items: ["love", "favorite", "view"], + }, + WHAT4: { + acceptReporters: true, + items: ["love", "favorite", "view"], + }, + WHAT5: { + acceptReporters: true, + items: ["about me", "wiwo", "location", "status"], + }, + }, + }; + } + async usergrab(args) { + try { + const response = await Scratch.fetch( + "https://scratchdb.lefty.one/v3/user/info/" + args.WHO + ); + const jsonData = await response.json(); + if (args.WHAT === "follower") { + return jsonData.statistics.followers; + } else if (args.WHAT === "following") { + return jsonData.statistics.following; + } else { + return ""; + } + } catch (error) { + return ""; + } + } + async rankusergrab(args) { + try { + const response = await Scratch.fetch( + "https://scratchdb.lefty.one/v3/user/info/" + args.WHO + ); + const jsonData = await response.json(); + if (args.WHAT === "follower") { + return jsonData.statistics.ranks.followers; + } else if (args.WHAT === "love") { + return jsonData.statistics.ranks.loves; + } else if (args.WHAT === "favorite") { + return jsonData.statistics.ranks.favorites; + } else if (args.WHAT === "view") { + return jsonData.statistics.ranks.views; + } else { + return ""; + } + } catch (error) { + return ""; + } + } + async usergrab2(args) { + try { + const response = await Scratch.fetch( + "https://scratchdb.lefty.one/v3/user/info/" + args.WHO + ); + const jsonData = await response.json(); + if (args.WHAT === "about me") { + return jsonData.bio; + } else if (args.WHAT === "wiwo") { + return jsonData.work; + } else if (args.WHAT === "location") { + return jsonData.country; + } else if (args.WHAT === "status") { + return jsonData.status; + } else { + return ""; + } + } catch (error) { + return ""; + } + } + async projectgrab(args) { + try { + const response = await Scratch.fetch( + "https://scratchdb.lefty.one/v3/project/info/" + args.WHO + ); + const jsonData = await response.json(); + if (args.WHAT === "love") { + return jsonData.statistics.loves; + } else if (args.WHAT === "favorite") { + return jsonData.statistics.favorites; + } else if (args.WHAT === "view") { + return jsonData.statistics.views; + } else { + return ""; + } + } catch (error) { + return ""; + } + } + async rankprojectgrab(args) { + try { + const response = await Scratch.fetch( + "https://scratchdb.lefty.one/v3/project/info/" + args.WHO + ); + const jsonData = await response.json(); + if (args.WHAT === "love") { + return jsonData.statistics.ranks.loves; + } else if (args.WHAT === "favorite") { + return jsonData.statistics.ranks.favorites; + } else if (args.WHAT === "view") { + return jsonData.statistics.ranks.views; + } else { + return ""; + } + } catch (error) { + return ""; + } + } + async idtoname(args) { + try { + const response = await Scratch.fetch( + "https://scratchdb.lefty.one/v3/project/info/" + args.WHO + ); + const jsonData = await response.json(); + return jsonData.title; + } catch (error) { + return ""; + } + } + async idtoowner(args) { + try { + const response = await Scratch.fetch( + "https://scratchdb.lefty.one/v3/project/info/" + args.WHO + ); + const jsonData = await response.json(); + return jsonData.username; + } catch (error) { + return ""; + } + } + } + Scratch.extensions.register(new nexuskittensgrab()); +})(Scratch); diff --git a/extensions/Skyhigh173/bigint.js b/extensions/Skyhigh173/bigint.js index 14f137d6ca..acea113026 100644 --- a/extensions/Skyhigh173/bigint.js +++ b/extensions/Skyhigh173/bigint.js @@ -1,14 +1,19 @@ -(function(Scratch){ - 'use strict'; +// Name: BigInt +// ID: skyhigh173BigInt +// Description: Math blocks that work on infinitely large integers (no decimals). +// By: Skyhigh173 + +(function (Scratch) { + "use strict"; /** * @param {unknown} x * @returns {bigint} */ - const bi = x => { - if (typeof x === 'string') { + const bi = (x) => { + if (typeof x === "string") { // Try to parse things like '8n' - if (x.charAt(x.length - 1) === 'n') { + if (x.charAt(x.length - 1) === "n") { try { return BigInt(x.slice(0, -1)); } catch (e) { @@ -16,9 +21,10 @@ } } // Must remove decimal using string operations. Math.trunc will convert to float - // which ruins the point of using bigints. - const decimalIndex = x.indexOf('.'); - const withoutDecimal = decimalIndex === -1 ? x : x.substring(0, decimalIndex); + // which ruins the point of using bigints. + const decimalIndex = x.indexOf("."); + const withoutDecimal = + decimalIndex === -1 ? x : x.substring(0, decimalIndex); try { return BigInt(withoutDecimal); } catch (e) { @@ -35,335 +41,335 @@ }; const makeLabel = (text) => ({ - blockType: 'label', - text: text + blockType: "label", + text: text, }); class BigIntExtension { getInfo() { return { - id: 'skyhigh173BigInt', - name: 'BigInt', - color1: '#59C093', + id: "skyhigh173BigInt", + name: "BigInt", + color1: "#59C093", blocks: [ { - opcode: 'from', + opcode: "from", blockType: Scratch.BlockType.REPORTER, - text: 'To BigInt [text]', + text: "To BigInt [text]", arguments: { text: { type: Scratch.ArgumentType.STRING, - defaultValue: '' - } - } + defaultValue: "", + }, + }, }, { - opcode: 'to', + opcode: "to", blockType: Scratch.BlockType.REPORTER, - text: 'To Number [text]', + text: "To Number [text]", arguments: { text: { type: Scratch.ArgumentType.STRING, - defaultValue: '' - } - } + defaultValue: "", + }, + }, }, - makeLabel('Arithmetic'), + makeLabel("Arithmetic"), { - opcode: 'add', + opcode: "add", blockType: Scratch.BlockType.REPORTER, - text: '[a] + [b]', + text: "[a] + [b]", arguments: { a: { type: Scratch.ArgumentType.STRING, - defaultValue: '' + defaultValue: "", }, b: { type: Scratch.ArgumentType.STRING, - defaultValue: '' - } - } + defaultValue: "", + }, + }, }, { - opcode: 'sub', + opcode: "sub", blockType: Scratch.BlockType.REPORTER, - text: '[a] - [b]', + text: "[a] - [b]", arguments: { a: { type: Scratch.ArgumentType.STRING, - defaultValue: '' + defaultValue: "", }, b: { type: Scratch.ArgumentType.STRING, - defaultValue: '' - } - } + defaultValue: "", + }, + }, }, { - opcode: 'mul', + opcode: "mul", blockType: Scratch.BlockType.REPORTER, - text: '[a] * [b]', + text: "[a] * [b]", arguments: { a: { type: Scratch.ArgumentType.STRING, - defaultValue: '' + defaultValue: "", }, b: { type: Scratch.ArgumentType.STRING, - defaultValue: '' - } - } + defaultValue: "", + }, + }, }, { - opcode: 'div', + opcode: "div", blockType: Scratch.BlockType.REPORTER, - text: '[a] / [b]', + text: "[a] / [b]", arguments: { a: { type: Scratch.ArgumentType.STRING, - defaultValue: '' + defaultValue: "", }, b: { type: Scratch.ArgumentType.STRING, - defaultValue: '' - } - } + defaultValue: "", + }, + }, }, { - opcode: 'pow', + opcode: "pow", blockType: Scratch.BlockType.REPORTER, - text: '[a] ** [b]', + text: "[a] ** [b]", arguments: { a: { type: Scratch.ArgumentType.STRING, - defaultValue: '' + defaultValue: "", }, b: { type: Scratch.ArgumentType.STRING, - defaultValue: '' - } - } + defaultValue: "", + }, + }, }, { - opcode: 'mod', + opcode: "mod", blockType: Scratch.BlockType.REPORTER, - text: '[a] mod [b]', + text: "[a] mod [b]", arguments: { a: { type: Scratch.ArgumentType.STRING, - defaultValue: '' + defaultValue: "", }, b: { type: Scratch.ArgumentType.STRING, - defaultValue: '' - } - } + defaultValue: "", + }, + }, }, { - opcode: 'select', + opcode: "select", blockType: Scratch.BlockType.REPORTER, - text: '[a] [sel] [b]', + text: "[a] [sel] [b]", arguments: { a: { type: Scratch.ArgumentType.STRING, - defaultValue: '' + defaultValue: "", }, b: { type: Scratch.ArgumentType.STRING, - defaultValue: '' + defaultValue: "", }, sel: { type: Scratch.ArgumentType.STRING, - defaultValue: '+', - menu: 'op' - } - } + defaultValue: "+", + menu: "op", + }, + }, }, - makeLabel('Logic'), + makeLabel("Logic"), { - opcode: 'lt', + opcode: "lt", blockType: Scratch.BlockType.BOOLEAN, - text: '[a] < [b]', + text: "[a] < [b]", arguments: { a: { type: Scratch.ArgumentType.STRING, - defaultValue: '' + defaultValue: "", }, b: { type: Scratch.ArgumentType.STRING, - defaultValue: '' - } - } + defaultValue: "", + }, + }, }, { - opcode: 'le', + opcode: "le", blockType: Scratch.BlockType.BOOLEAN, - text: '[a] ≤ [b]', + text: "[a] ≤ [b]", arguments: { a: { type: Scratch.ArgumentType.STRING, - defaultValue: '' + defaultValue: "", }, b: { type: Scratch.ArgumentType.STRING, - defaultValue: '' - } - } + defaultValue: "", + }, + }, }, { - opcode: 'eq', + opcode: "eq", blockType: Scratch.BlockType.BOOLEAN, - text: '[a] = [b]', + text: "[a] = [b]", arguments: { a: { type: Scratch.ArgumentType.STRING, - defaultValue: '' + defaultValue: "", }, b: { type: Scratch.ArgumentType.STRING, - defaultValue: '' - } - } + defaultValue: "", + }, + }, }, { - opcode: 'neq', + opcode: "neq", blockType: Scratch.BlockType.BOOLEAN, - text: '[a] ≠ [b]', + text: "[a] ≠ [b]", arguments: { a: { type: Scratch.ArgumentType.STRING, - defaultValue: '' + defaultValue: "", }, b: { type: Scratch.ArgumentType.STRING, - defaultValue: '' - } - } + defaultValue: "", + }, + }, }, { - opcode: 'ge', + opcode: "ge", blockType: Scratch.BlockType.BOOLEAN, - text: '[a] ≥ [b]', + text: "[a] ≥ [b]", arguments: { a: { type: Scratch.ArgumentType.STRING, - defaultValue: '' + defaultValue: "", }, b: { type: Scratch.ArgumentType.STRING, - defaultValue: '' - } - } + defaultValue: "", + }, + }, }, { - opcode: 'gt', + opcode: "gt", blockType: Scratch.BlockType.BOOLEAN, - text: '[a] > [b]', + text: "[a] > [b]", arguments: { a: { type: Scratch.ArgumentType.STRING, - defaultValue: '' + defaultValue: "", }, b: { type: Scratch.ArgumentType.STRING, - defaultValue: '' - } - } + defaultValue: "", + }, + }, }, - makeLabel('Bitwise'), + makeLabel("Bitwise"), { - opcode: 'and', + opcode: "and", blockType: Scratch.BlockType.REPORTER, - text: '[a] & [b]', + text: "[a] & [b]", arguments: { a: { type: Scratch.ArgumentType.STRING, - defaultValue: '' + defaultValue: "", }, b: { type: Scratch.ArgumentType.STRING, - defaultValue: '' - } - } + defaultValue: "", + }, + }, }, { - opcode: 'or', + opcode: "or", blockType: Scratch.BlockType.REPORTER, - text: '[a] | [b]', + text: "[a] | [b]", arguments: { a: { type: Scratch.ArgumentType.STRING, - defaultValue: '' + defaultValue: "", }, b: { type: Scratch.ArgumentType.STRING, - defaultValue: '' - } - } + defaultValue: "", + }, + }, }, { - opcode: 'xor', + opcode: "xor", blockType: Scratch.BlockType.REPORTER, - text: '[a] ^ [b]', + text: "[a] ^ [b]", arguments: { a: { type: Scratch.ArgumentType.STRING, - defaultValue: '' + defaultValue: "", }, b: { type: Scratch.ArgumentType.STRING, - defaultValue: '' - } - } + defaultValue: "", + }, + }, }, { - opcode: 'ls', + opcode: "ls", blockType: Scratch.BlockType.REPORTER, - text: '[a] << [b]', + text: "[a] << [b]", arguments: { a: { type: Scratch.ArgumentType.STRING, - defaultValue: '' + defaultValue: "", }, b: { type: Scratch.ArgumentType.STRING, - defaultValue: '' - } - } + defaultValue: "", + }, + }, }, { - opcode: 'rs', + opcode: "rs", blockType: Scratch.BlockType.REPORTER, - text: '[a] >> [b]', + text: "[a] >> [b]", arguments: { a: { type: Scratch.ArgumentType.STRING, - defaultValue: '' + defaultValue: "", }, b: { type: Scratch.ArgumentType.STRING, - defaultValue: '' - } - } + defaultValue: "", + }, + }, }, { - opcode: 'not', + opcode: "not", blockType: Scratch.BlockType.REPORTER, - text: '~ [a]', + text: "~ [a]", arguments: { a: { type: Scratch.ArgumentType.STRING, - defaultValue: '' + defaultValue: "", }, - } + }, }, ], menus: { op: { - items: ['+', '-', '*', '/', '%', '^'], - acceptReporters: true - } - } + items: ["+", "-", "*", "/", "%", "^"], + acceptReporters: true, + }, + }, }; } from({ text }) { @@ -382,14 +388,14 @@ return (bi(a) * bi(b)).toString(); } div({ a, b }) { - if (Number(b) == 0) return 'NaN'; + if (Number(b) == 0) return "NaN"; return (bi(a) / bi(b)).toString(); } pow({ a, b }) { return (bi(a) ** bi(b)).toString(); } mod({ a, b }) { - if (Number(b) == 0) return 'NaN'; + if (Number(b) == 0) return "NaN"; return (bi(a) % bi(b)).toString(); } @@ -414,19 +420,25 @@ select({ a, sel, b }) { switch (sel) { - case '+': return (bi(a) + bi(b)).toString(); - case '-': return (bi(a) - bi(b)).toString(); - case '*': return (bi(a) * bi(b)).toString(); - case '/': { - if (Number(b) == 0) return 'NaN'; - return (bi(a) / bi(b)).toString(); - } - case '%': { - if (Number(b) == 0) return 'NaN'; - return (bi(a) % bi(b)).toString(); - } - case '^': case '**': return (bi(a) ** bi(b)).toString(); - default: return '0'; + case "+": + return (bi(a) + bi(b)).toString(); + case "-": + return (bi(a) - bi(b)).toString(); + case "*": + return (bi(a) * bi(b)).toString(); + case "/": { + if (Number(b) == 0) return "NaN"; + return (bi(a) / bi(b)).toString(); + } + case "%": { + if (Number(b) == 0) return "NaN"; + return (bi(a) % bi(b)).toString(); + } + case "^": + case "**": + return (bi(a) ** bi(b)).toString(); + default: + return "0"; } } diff --git a/extensions/Skyhigh173/json.js b/extensions/Skyhigh173/json.js index 65496e698d..5bceb7485a 100644 --- a/extensions/Skyhigh173/json.js +++ b/extensions/Skyhigh173/json.js @@ -1,525 +1,535 @@ -(function(Scratch) { - 'use strict'; +// Name: JSON +// ID: skyhigh173JSON +// Description: Handle JSON strings and arrays. +// By: Skyhigh173 + +(function (Scratch) { + "use strict"; /* - * JSON extension v2.5 by skyhigh173 (English Version) - * Do not remove this comment - */ + * JSON extension v2.5 by skyhigh173 (English Version) + * Do not remove this comment + */ const vm = Scratch.vm; - const hasOwn = (obj, property) => Object.prototype.hasOwnProperty.call(obj, property); + const hasOwn = (obj, property) => + Object.prototype.hasOwnProperty.call(obj, property); const makeLabel = (text) => ({ - blockType: 'label', - text: text + blockType: "label", + text: text, }); class JSONS { getInfo() { return { - id: 'skyhigh173JSON', - name: 'JSON', - color1: '#3271D0', + id: "skyhigh173JSON", + name: "JSON", + color1: "#3271D0", blocks: [ - makeLabel('General Utils'), + makeLabel("General Utils"), { - opcode: 'json_is_valid', + opcode: "json_is_valid", blockType: Scratch.BlockType.BOOLEAN, - text: 'is JSON [json] valid', + text: "is JSON [json] valid", arguments: { json: { type: Scratch.ArgumentType.STRING, - defaultValue: '{"key":"value"}' - } - } + defaultValue: '{"key":"value"}', + }, + }, }, { - opcode: 'json_is', + opcode: "json_is", blockType: Scratch.BlockType.BOOLEAN, - text: 'is [json] [types]', + text: "is [json] [types]", arguments: { json: { type: Scratch.ArgumentType.STRING, - defaultValue: '{"key":"value"}' + defaultValue: '{"key":"value"}', }, types: { type: Scratch.ArgumentType.STRING, - defaultValue: 'Object', - menu: 'types' + defaultValue: "Object", + menu: "types", }, - } + }, }, - '---', + "---", { - opcode: 'json_get_all', + opcode: "json_get_all", blockType: Scratch.BlockType.REPORTER, - text: 'get all [Stype] of [json]', + text: "get all [Stype] of [json]", arguments: { Stype: { type: Scratch.ArgumentType.STRING, - menu: 'get_all' + menu: "get_all", }, json: { type: Scratch.ArgumentType.STRING, - defaultValue: '{"key":"value","key2":"value2"}' - } - } + defaultValue: '{"key":"value","key2":"value2"}', + }, + }, }, { - opcode: 'json_new', + opcode: "json_new", blockType: Scratch.BlockType.REPORTER, - text: 'new [json]', + text: "new [json]", arguments: { json: { type: Scratch.ArgumentType.STRING, - defaultValue: 'Object', - menu: 'types' - } - } + defaultValue: "Object", + menu: "types", + }, + }, }, - '---', + "---", { - opcode: 'json_has_key', + opcode: "json_has_key", blockType: Scratch.BlockType.BOOLEAN, - text: '[json] contains key [key]?', + text: "[json] contains key [key]?", arguments: { key: { type: Scratch.ArgumentType.STRING, - defaultValue: 'key2' + defaultValue: "key2", }, json: { type: Scratch.ArgumentType.STRING, - defaultValue: '{"key":"value"}' - } - } + defaultValue: '{"key":"value"}', + }, + }, }, { - opcode: 'json_has_value', + opcode: "json_has_value", blockType: Scratch.BlockType.BOOLEAN, - text: '[json] contains value [value]?', + text: "[json] contains value [value]?", arguments: { value: { type: Scratch.ArgumentType.STRING, - defaultValue: 'scratch' + defaultValue: "scratch", }, json: { type: Scratch.ArgumentType.STRING, - defaultValue: '["TurboWarp","scratch"]' - } - } + defaultValue: '["TurboWarp","scratch"]', + }, + }, }, { - opcode: 'json_equal', + opcode: "json_equal", blockType: Scratch.BlockType.BOOLEAN, - text: '[json1] [equal] [json2]', + text: "[json1] [equal] [json2]", arguments: { json1: { type: Scratch.ArgumentType.STRING, - defaultValue: '{"a":0,"b":1}' + defaultValue: '{"a":0,"b":1}', }, json2: { type: Scratch.ArgumentType.STRING, - defaultValue: '{"b":1,"a":0}' + defaultValue: '{"b":1,"a":0}', }, equal: { type: Scratch.ArgumentType.STRING, - defaultValue: '=', - menu: 'equal' - } - } + defaultValue: "=", + menu: "equal", + }, + }, }, - makeLabel('JSON Strings'), + makeLabel("JSON Strings"), { - opcode: 'json_jlength', + opcode: "json_jlength", blockType: Scratch.BlockType.REPORTER, - text: 'length of json [json]', + text: "length of json [json]", arguments: { json: { type: Scratch.ArgumentType.STRING, - defaultValue: '{"key":"value","key2":"value2"}' - } - } + defaultValue: '{"key":"value","key2":"value2"}', + }, + }, }, { - opcode: 'json_get', + opcode: "json_get", blockType: Scratch.BlockType.REPORTER, - text: 'get [item] in [json]', + text: "get [item] in [json]", arguments: { item: { type: Scratch.ArgumentType.STRING, - defaultValue: 'key' + defaultValue: "key", }, json: { type: Scratch.ArgumentType.STRING, - defaultValue: '{"key":"value"}' - } - } + defaultValue: '{"key":"value"}', + }, + }, }, { - opcode: 'json_set', + opcode: "json_set", blockType: Scratch.BlockType.REPORTER, - text: 'set [item] to [value] in [json]', + text: "set [item] to [value] in [json]", arguments: { item: { type: Scratch.ArgumentType.STRING, - defaultValue: 'key' + defaultValue: "key", }, value: { type: Scratch.ArgumentType.STRING, - defaultValue: 'new value' + defaultValue: "new value", }, json: { type: Scratch.ArgumentType.STRING, - defaultValue: '{"key":"value"}' - } - } + defaultValue: '{"key":"value"}', + }, + }, }, { - opcode: 'json_delete', + opcode: "json_delete", blockType: Scratch.BlockType.REPORTER, - text: 'delete [item] in [json]', + text: "delete [item] in [json]", arguments: { item: { type: Scratch.ArgumentType.STRING, - defaultValue: 'key2' + defaultValue: "key2", }, json: { type: Scratch.ArgumentType.STRING, - defaultValue: '{"key":"value","key2":"value2"}' - } - } + defaultValue: '{"key":"value","key2":"value2"}', + }, + }, }, - makeLabel('Array'), + makeLabel("Array"), { - opcode: 'json_length', + opcode: "json_length", blockType: Scratch.BlockType.REPORTER, - text: 'length of array [json]', + text: "length of array [json]", arguments: { json: { type: Scratch.ArgumentType.STRING, - defaultValue: '[1,2,3]' - } - } + defaultValue: "[1,2,3]", + }, + }, }, { - opcode: 'json_array_get', + opcode: "json_array_get", blockType: Scratch.BlockType.REPORTER, - text: 'item [item] of array [json]', + text: "item [item] of array [json]", arguments: { item: { type: Scratch.ArgumentType.NUMBER, - defaultValue: 1 + defaultValue: 1, }, json: { type: Scratch.ArgumentType.STRING, - defaultValue: '["scratch","TurboWarp"]' - } - } + defaultValue: '["scratch","TurboWarp"]', + }, + }, }, { - opcode: 'json_array_push', + opcode: "json_array_push", blockType: Scratch.BlockType.REPORTER, - text: 'add [item] to array [json]', + text: "add [item] to array [json]", arguments: { item: { type: Scratch.ArgumentType.STRING, - defaultValue: 'TurboWarp' + defaultValue: "TurboWarp", }, json: { type: Scratch.ArgumentType.STRING, - defaultValue: '["scratch"]' - } - } + defaultValue: '["scratch"]', + }, + }, }, { - opcode: 'json_array_set', + opcode: "json_array_set", blockType: Scratch.BlockType.REPORTER, - text: 'replace item [pos] of [json] to [item]', + text: "replace item [pos] of [json] to [item]", arguments: { item: { type: Scratch.ArgumentType.STRING, - defaultValue: 'fav' + defaultValue: "fav", }, pos: { type: Scratch.ArgumentType.NUMBER, - defaultValue: 2 + defaultValue: 2, }, json: { type: Scratch.ArgumentType.STRING, - defaultValue: '["love","heart","follow"]' - } - } + defaultValue: '["love","heart","follow"]', + }, + }, }, { - opcode: 'json_array_insert', + opcode: "json_array_insert", blockType: Scratch.BlockType.REPORTER, - text: 'insert [item] at [pos] of array [json]', + text: "insert [item] at [pos] of array [json]", arguments: { item: { type: Scratch.ArgumentType.STRING, - defaultValue: 'fav' + defaultValue: "fav", }, pos: { type: Scratch.ArgumentType.NUMBER, - defaultValue: 2 + defaultValue: 2, }, json: { type: Scratch.ArgumentType.STRING, - defaultValue: '["love","follow"]' - } - } + defaultValue: '["love","follow"]', + }, + }, }, - '---', + "---", { - opcode: 'json_array_delete', + opcode: "json_array_delete", blockType: Scratch.BlockType.REPORTER, - text: 'delete item [item] of array [json]', + text: "delete item [item] of array [json]", arguments: { item: { type: Scratch.ArgumentType.NUMBER, - defaultValue: 2 + defaultValue: 2, }, json: { type: Scratch.ArgumentType.STRING, - defaultValue: '["scratch","a","TurboWarp"]' - } - } + defaultValue: '["scratch","a","TurboWarp"]', + }, + }, }, { - opcode: 'json_array_remove_all', + opcode: "json_array_remove_all", blockType: Scratch.BlockType.REPORTER, - text: 'delete all [item] in array [json]', + text: "delete all [item] in array [json]", arguments: { item: { type: Scratch.ArgumentType.STRING, - defaultValue: 'a' + defaultValue: "a", }, json: { type: Scratch.ArgumentType.STRING, - defaultValue: '["scratch","a","TurboWarp","a","a"]' - } - } + defaultValue: '["scratch","a","TurboWarp","a","a"]', + }, + }, }, - '---', + "---", { - opcode: 'json_array_itemH', + opcode: "json_array_itemH", blockType: Scratch.BlockType.REPORTER, - text: 'item # of [item] in array [json]', + text: "item # of [item] in array [json]", arguments: { item: { type: Scratch.ArgumentType.STRING, - defaultValue: 'scratch' + defaultValue: "scratch", }, json: { type: Scratch.ArgumentType.STRING, - defaultValue: '["scratch","TurboWarp"]' - } - } + defaultValue: '["scratch","TurboWarp"]', + }, + }, }, - makeLabel('Advanced'), + makeLabel("Advanced"), { - opcode: 'json_array_from', + opcode: "json_array_from", blockType: Scratch.BlockType.REPORTER, - text: 'array from text [json]', + text: "array from text [json]", arguments: { json: { type: Scratch.ArgumentType.STRING, - defaultValue: 'abcd' - } - } + defaultValue: "abcd", + }, + }, }, { - opcode: 'json_array_fromto', + opcode: "json_array_fromto", blockType: Scratch.BlockType.REPORTER, - text: 'array [json] from item [item] to [item2]', + text: "array [json] from item [item] to [item2]", arguments: { json: { type: Scratch.ArgumentType.STRING, - defaultValue: '[1,2,3,4]' + defaultValue: "[1,2,3,4]", }, item: { type: Scratch.ArgumentType.NUMBER, - defaultValue: 2 + defaultValue: 2, }, item2: { type: Scratch.ArgumentType.NUMBER, - defaultValue: 3 - } - } + defaultValue: 3, + }, + }, }, { - opcode: 'json_array_reverse', + opcode: "json_array_reverse", blockType: Scratch.BlockType.REPORTER, - text: 'reverse array [json]', + text: "reverse array [json]", arguments: { json: { type: Scratch.ArgumentType.STRING, - defaultValue: '["a","b","c","d","e","f"]' - } - } + defaultValue: '["a","b","c","d","e","f"]', + }, + }, }, { - opcode: 'json_array_flat', + opcode: "json_array_flat", blockType: Scratch.BlockType.REPORTER, - text: 'flat array [json] by depth [depth]', + text: "flat array [json] by depth [depth]", arguments: { json: { type: Scratch.ArgumentType.STRING, - defaultValue: '[[1],2,[3,4],[5,[6]]]' + defaultValue: "[[1],2,[3,4],[5,[6]]]", }, depth: { type: Scratch.ArgumentType.NUMBER, - defaultValue: 2 - } - } + defaultValue: 2, + }, + }, }, { - opcode: 'json_array_concat', + opcode: "json_array_concat", blockType: Scratch.BlockType.REPORTER, - text: 'array concat [json] [json2]', + text: "array concat [json] [json2]", arguments: { json: { type: Scratch.ArgumentType.STRING, - defaultValue: '["a","b"]' + defaultValue: '["a","b"]', }, json2: { type: Scratch.ArgumentType.STRING, - defaultValue: '["c","d"]' - } - } + defaultValue: '["c","d"]', + }, + }, }, { - opcode: 'json_array_filter', + opcode: "json_array_filter", blockType: Scratch.BlockType.REPORTER, - text: 'get all value with key [key] in array [json]', + text: "get all value with key [key] in array [json]", arguments: { key: { type: Scratch.ArgumentType.STRING, - defaultValue: 'id' + defaultValue: "id", }, json: { type: Scratch.ArgumentType.STRING, - defaultValue: '[{"id":12},{"id":24}]' - } - } + defaultValue: '[{"id":12},{"id":24}]', + }, + }, }, { - opcode: 'json_array_setlen', + opcode: "json_array_setlen", blockType: Scratch.BlockType.REPORTER, - text: 'set length of array [json] to [len]', + text: "set length of array [json] to [len]", arguments: { json: { type: Scratch.ArgumentType.STRING, - defaultValue: '["a","b","c"]' + defaultValue: '["a","b","c"]', }, len: { type: Scratch.ArgumentType.NUMBER, - defaultValue: 2 - } - } + defaultValue: 2, + }, + }, }, - '---', + "---", { - opcode: 'json_array_create', + opcode: "json_array_create", blockType: Scratch.BlockType.REPORTER, - text: 'create array by [text] with delimiter [d]', + text: "create array by [text] with delimiter [d]", arguments: { text: { type: Scratch.ArgumentType.STRING, - defaultValue: 'a,b,c' + defaultValue: "a,b,c", }, d: { type: Scratch.ArgumentType.STRING, - defaultValue: ',' - } - } + defaultValue: ",", + }, + }, }, { - opcode: 'json_array_join', + opcode: "json_array_join", blockType: Scratch.BlockType.REPORTER, - text: 'join string by array [json] with delimiter [d]', + text: "join string by array [json] with delimiter [d]", arguments: { json: { type: Scratch.ArgumentType.STRING, - defaultValue: '["a","b","c"]' + defaultValue: '["a","b","c"]', }, d: { type: Scratch.ArgumentType.STRING, - defaultValue: ',' - } - } + defaultValue: ",", + }, + }, }, - makeLabel('Lists'), + makeLabel("Lists"), { - opcode: 'json_vm_getlist', + opcode: "json_vm_getlist", blockType: Scratch.BlockType.REPORTER, - text: 'get list [list] as array', + text: "get list [list] as array", arguments: { list: { type: Scratch.ArgumentType.STRING, - menu: 'get_list' + menu: "get_list", }, - } + }, }, { - opcode: 'json_vm_setlist', + opcode: "json_vm_setlist", blockType: Scratch.BlockType.COMMAND, - text: 'set list [list] to content [json]', + text: "set list [list] to content [json]", arguments: { list: { type: Scratch.ArgumentType.STRING, - menu: 'get_list' + menu: "get_list", }, json: { type: Scratch.ArgumentType.STRING, - defaultValue: '["apple","banana"]' + defaultValue: '["apple","banana"]', }, - } + }, }, ], menus: { get_all: { - items: ['keys','values','datas'] + items: ["keys", "values", "datas"], }, get_list: { acceptReporters: true, - items: 'getLists' + items: "getLists", }, types: { acceptReporters: true, - items: ['Object', 'Array'] + items: ["Object", "Array"], }, equal: { acceptReporters: true, - items: ['=','≠'] - } - } + items: ["=", "≠"], + }, + }, }; } - getLists () { - const globalLists = Object.values(vm.runtime.getTargetForStage().variables).filter(x => x.type == 'list'); - const localLists = Object.values(vm.editingTarget.variables).filter(x => x.type == 'list'); + getLists() { + const globalLists = Object.values( + vm.runtime.getTargetForStage().variables + ).filter((x) => x.type == "list"); + const localLists = Object.values(vm.editingTarget.variables).filter( + (x) => x.type == "list" + ); const uniqueLists = [...new Set([...globalLists, ...localLists])]; if (uniqueLists.length === 0) { return [ { - text: 'select a list', - value: 'select a list' - } + text: "select a list", + value: "select a list", + }, ]; } - return uniqueLists.map(i => ({ + return uniqueLists.map((i) => ({ text: i.name, - value: i.id + value: i.id, })); } lookupList(list, util) { const byId = util.target.lookupVariableById(list); - if (byId && byId.type === 'list') { + if (byId && byId.type === "list") { return byId; } - const byName = util.target.lookupVariableByNameAndType(list, 'list'); + const byName = util.target.lookupVariableByNameAndType(list, "list"); if (byName) { return byName; } @@ -527,9 +537,12 @@ } json_is_valid({ json }) { - if (typeof json != 'string') { + if (typeof json != "string") { return false; - } else if ((json.slice(0,1) != '[' || json.slice(-1) != ']') && (json.slice(0,1) != '{' || json.slice(-1) != '}')) { + } else if ( + (json.slice(0, 1) != "[" || json.slice(-1) != "]") && + (json.slice(0, 1) != "{" || json.slice(-1) != "}") + ) { return false; } else { try { @@ -542,10 +555,13 @@ } // return object if its json else string - json_valid_return(json){ - if (typeof json != 'string') { + json_valid_return(json) { + if (typeof json != "string") { return json; - } else if ((json.slice(0,1) != '[' || json.slice(-1) != ']') && (json.slice(0,1) != '{' || json.slice(-1) != '}')) { + } else if ( + (json.slice(0, 1) != "[" || json.slice(-1) != "]") && + (json.slice(0, 1) != "{" || json.slice(-1) != "}") + ) { return json; } else { try { @@ -557,13 +573,16 @@ } json_is({ json, types }) { - if (!this.json_is_valid({json: json})) return false; + if (!this.json_is_valid({ json: json })) return false; try { json = JSON.parse(json); switch (types) { - case 'Object': return !Array.isArray(json); - case 'Array': return Array.isArray(json); - default: return false; + case "Object": + return !Array.isArray(json); + case "Array": + return Array.isArray(json); + default: + return false; } } catch { return false; @@ -575,21 +594,27 @@ json = JSON.parse(json); return Object.keys(json).length; } catch { - return ' '; + return " "; } } json_new({ json }) { switch (json) { - case 'Object': return '{}'; - case 'Array': return '[]'; - default: return ''; + case "Object": + return "{}"; + case "Array": + return "[]"; + default: + return ""; } } json_has_key({ json, key }) { try { - return this._fixInvalidJSONValues(this.json_valid_return(key)) in JSON.parse(json); + return ( + this._fixInvalidJSONValues(this.json_valid_return(key)) in + JSON.parse(json) + ); } catch { return false; } @@ -612,30 +637,34 @@ const keys1 = Object.keys(json1); const keys2 = Object.keys(json2); - const result = keys1.length === keys2.length && Object.keys(json1).every(key=>json1[key] === json2[key]); - if (equal === '=') return result; - if (equal === '≠') return !result; + const result = + keys1.length === keys2.length && + Object.keys(json1).every((key) => json1[key] === json2[key]); + if (equal === "=") return result; + if (equal === "≠") return !result; } catch { // ignore } return false; } - - json_get_all({ Stype,json }) { + json_get_all({ Stype, json }) { try { json = JSON.parse(json); switch (Stype) { - case 'keys': - return JSON.stringify(Object.keys(json).map(key => key)); - case 'values': - return JSON.stringify(Object.keys(json).map(key => json[key])); - case 'datas': - return JSON.stringify(Object.keys(json).map(key => [key, json[key]])); - default: return ''; + case "keys": + return JSON.stringify(Object.keys(json).map((key) => key)); + case "values": + return JSON.stringify(Object.keys(json).map((key) => json[key])); + case "datas": + return JSON.stringify( + Object.keys(json).map((key) => [key, json[key]]) + ); + default: + return ""; } } catch { - return ''; + return ""; } } @@ -644,7 +673,7 @@ json = JSON.parse(json); if (hasOwn(json, item)) { const result = json[item]; - if (typeof result === 'object') { + if (typeof result === "object") { return JSON.stringify(result); } else { return result; @@ -653,14 +682,14 @@ } catch { // ignore } - return ''; + return ""; } _fixInvalidJSONValues(value) { // JSON does not support these values, so convert to string. - if (Number.isNaN(value)) return 'NaN'; - if (value === Infinity) return 'Infinity'; - if (value === -Infinity) return '-Infinity'; + if (Number.isNaN(value)) return "NaN"; + if (value === Infinity) return "Infinity"; + if (value === -Infinity) return "-Infinity"; return value; } @@ -672,7 +701,7 @@ json[item] = value; return JSON.stringify(json); } catch { - return ''; + return ""; } } @@ -682,7 +711,7 @@ delete json[item]; return JSON.stringify(json); } catch { - return ''; + return ""; } } @@ -695,7 +724,7 @@ // 1...length : array content, -1...-length : reverse array content, 0 : ERROR try { item = Scratch.Cast.toNumber(item); - if (item == 0) return ''; + if (item == 0) return ""; if (item > 0) { item--; } @@ -706,13 +735,13 @@ } else { result = json[json.length + item]; } - if (typeof result == 'object') { + if (typeof result == "object") { return JSON.stringify(result); } else { return result; } } catch { - return ''; + return ""; } } @@ -723,7 +752,7 @@ let result = JSON.stringify(json.indexOf(item) + 1); return result; } catch { - return ''; + return ""; } } @@ -731,7 +760,7 @@ try { return JSON.stringify(Array.from(String(json))); } catch { - return ''; + return ""; } } @@ -741,7 +770,7 @@ json2 = JSON.parse(json2); return JSON.stringify(json.concat(json2)); } catch { - return ''; + return ""; } } @@ -752,7 +781,7 @@ json.push(item); return JSON.stringify(json); } catch { - return ''; + return ""; } } @@ -763,17 +792,19 @@ json.splice(pos - 1, 0, item); return JSON.stringify(json); } catch { - return ''; + return ""; } } json_array_set({ item, pos, json }) { try { json = JSON.parse(json); - json[pos - 1] = this._fixInvalidJSONValues(this.json_valid_return(item)); + json[pos - 1] = this._fixInvalidJSONValues( + this.json_valid_return(item) + ); return JSON.stringify(json); } catch { - return ''; + return ""; } } @@ -783,7 +814,7 @@ json.splice(item - 1, 1); return JSON.stringify(json); } catch { - return ''; + return ""; } } @@ -801,15 +832,15 @@ } return JSON.stringify(json); } catch { - return ''; + return ""; } } json_array_fromto({ json, item, item2 }) { try { - return JSON.stringify(JSON.parse(json).slice(item - 1,item2)); + return JSON.stringify(JSON.parse(json).slice(item - 1, item2)); } catch { - return ''; + return ""; } } @@ -817,7 +848,7 @@ try { return JSON.stringify(JSON.parse(json).reverse()); } catch { - return ''; + return ""; } } @@ -825,7 +856,7 @@ try { return JSON.stringify(JSON.parse(json).flat(depth)); } catch { - return ''; + return ""; } } @@ -837,21 +868,23 @@ try { return JSON.parse(json).join(d); } catch { - return ''; + return ""; } } json_array_filter({ key, json }) { try { json = JSON.parse(json); - return JSON.stringify(json.map(x => { - if (hasOwn(x, key)) { - return x[key]; - } - return null; - })); + return JSON.stringify( + json.map((x) => { + if (hasOwn(x, key)) { + return x[key]; + } + return null; + }) + ); } catch (e) { - return ''; + return ""; } } @@ -861,7 +894,7 @@ json.length = len; return JSON.stringify(json); } catch { - return ''; + return ""; } } @@ -874,7 +907,7 @@ } catch (e) { // ignore } - return ''; + return ""; } json_vm_setlist({ list, json }, util) { try { @@ -882,8 +915,8 @@ if (listVariable) { const array = JSON.parse(json); if (Array.isArray(array)) { - const safeArray = array.map(i => { - if (typeof i === 'object') return JSON.stringify(i); + const safeArray = array.map((i) => { + if (typeof i === "object") return JSON.stringify(i); return i; }); listVariable.value = safeArray; @@ -892,7 +925,7 @@ } catch (e) { // ignore } - return ''; + return ""; } } Scratch.extensions.register(new JSONS()); diff --git a/extensions/TheShovel/CanvasEffects.js b/extensions/TheShovel/CanvasEffects.js index 4d53b09cc2..e14c905bf5 100644 --- a/extensions/TheShovel/CanvasEffects.js +++ b/extensions/TheShovel/CanvasEffects.js @@ -1,165 +1,275 @@ -(function(Scratch) { - 'use strict'; - if (!Scratch.extensions.unsandboxed) { - throw new Error('This extension must run unsandboxed'); - } +// Name: Canvas Effects +// ID: theshovelcanvaseffects +// Description: Apply visual effects to the entire stage. +// By: TheShovel + +(function (Scratch) { + "use strict"; + if (!Scratch.extensions.unsandboxed) { + throw new Error("This extension must run unsandboxed"); + } - const canvas = Scratch.renderer.canvas; + const canvas = Scratch.renderer.canvas; - const updateStyle = () => { - const filter = `blur(${blur}px) contrast(${contrast / 100}) saturate(${saturation}%) hue-rotate(${color}deg) brightness(${brightness}%) invert(${invert}%)`; - if (canvas.style.filter !== filter) { - canvas.style.filter = filter; - } - const imageRendering = resizeMode === 'pixelated' ? 'pixelated' : ''; - if (canvas.style.imageRendering !== imageRendering) { - canvas.style.imageRendering = imageRendering; - } - }; - // scratch-gui will sometimes reset the cursor when resizing the window or going in/out of fullscreen - new MutationObserver(updateStyle).observe(canvas, { - attributeFilter: ['style'], - attributes: true - }); + const updateStyle = () => { + // Gotta keep the translation to % because of the stage size, window size and so on + const transform = `rotate(${rotation}deg) scale(${scale}%) skew(${skewX}deg, ${skewY}deg) translate(${offsetX}%, ${ + 0 - offsetY + }%)`; + if (canvas.style.transform !== transform) { + canvas.style.transform = transform; + } + const filter = `blur(${blur}px) contrast(${ + contrast / 100 + }) saturate(${saturation}%) hue-rotate(${color}deg) brightness(${brightness}%) invert(${invert}%) sepia(${sepia}%) opacity(${ + 100 - transparency + }%)`; + if (canvas.style.filter !== filter) { + canvas.style.filter = filter; + } + const cssBorderRadius = borderRadius === 0 ? "" : `${borderRadius}%`; + if (canvas.style.borderRadius !== cssBorderRadius) { + canvas.style.borderRadius = cssBorderRadius; + } + const imageRendering = resizeMode === "pixelated" ? "pixelated" : ""; + if (canvas.style.imageRendering !== imageRendering) { + canvas.style.imageRendering = imageRendering; + } + }; + // scratch-gui may reset canvas styles when resizing the window or going in/out of fullscreen + new MutationObserver(updateStyle).observe(canvas, { + attributeFilter: ["style"], + attributes: true, + }); - let blur = 0; - let contrast = 100; - let saturation = 100; - let color = 0; - let brightness = 100; - let invert = 0; - let resizeMode = 'default'; + let borderRadius = 0; + let rotation = 0; + let offsetY = 0; + let offsetX = 0; + let skewY = 0; + let skewX = 0; + let scale = 100; + // Thanks SharkPool for telling me about these + let transparency = 0; + let sepia = 0; + let blur = 0; + let contrast = 100; + let saturation = 100; + let color = 0; + let brightness = 100; + let invert = 0; + let resizeMode = "default"; - class CanvasEffects { - getInfo() { - return { - id: 'theshovelcanvaseffects', - name: 'Canvas Effects', - blocks: [ - { - opcode: 'seteffect', - blockType: Scratch.BlockType.COMMAND, - text: 'set canvas [EFFECT] to [NUMBER]', - arguments: { - EFFECT: { - type: Scratch.ArgumentType.STRING, - menu: 'EFFECTMENU' - }, - NUMBER: { - type: Scratch.ArgumentType.NUMBER - } - }, - }, - { - opcode: 'geteffect', - blockType: Scratch.BlockType.REPORTER, - text: 'get canvas [EFFECT]', - arguments: { - EFFECT: { - type: Scratch.ArgumentType.STRING, - menu: 'EFFECTGETMENU' - } - } - }, - { - opcode: 'cleareffects', - blockType: Scratch.BlockType.COMMAND, - text: 'clear canvas effects' - }, - { - opcode: 'renderscale', - blockType: Scratch.BlockType.COMMAND, - text: 'set canvas render size to width:[X] height:[Y]', - arguments: { - X: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: 100 - }, - Y: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: 100 - } - } - }, - { - opcode: 'setrendermode', - blockType: Scratch.BlockType.COMMAND, - text: 'set canvas resize rendering mode [EFFECT]', - arguments: { - EFFECT: { - type: Scratch.ArgumentType.STRING, - menu: 'RENDERMODE' - } - }, - }, - ], - menus: { - EFFECTMENU: { - acceptReporters: true, - items: ['blur', 'contrast', 'saturation', 'color shift', 'brightness', 'invert'] - }, - RENDERMODE: { - acceptReporters: true, - items: ['pixelated', 'default'] - }, - EFFECTGETMENU: { - acceptReporters: true, - items: ['blur', 'contrast', 'saturation', 'color shift', 'brightness', 'invert', 'resize rendering mode'] - } - } - }; - } - geteffect({EFFECT}) { - if (EFFECT === 'blur') { - return blur; - } else if (EFFECT === 'contrast') { - return contrast; - } else if (EFFECT === 'saturation') { - return saturation; - } else if (EFFECT === 'color shift') { - return color; - } else if (EFFECT === 'brightness') { - return brightness; - } else if (EFFECT === 'invert') { - return invert; - } else if (EFFECT === 'resize rendering mode') { - return resizeMode; - } - return ''; - } - seteffect({EFFECT, NUMBER}) { - if (EFFECT === 'blur') { - blur = NUMBER; - } else if (EFFECT === 'contrast') { - contrast = NUMBER; - } else if (EFFECT === 'saturation') { - saturation = NUMBER; - } else if (EFFECT === 'color shift') { - color = NUMBER; - } else if (EFFECT === 'brightness') { - brightness = NUMBER; - } else if (EFFECT === 'invert') { - invert = NUMBER; - } - updateStyle(); - } - cleareffects() { - blur = 0; - contrast = 100; - saturation = 100; - color = 0; - brightness = 100; - invert = 0; - resizeMode = 'default'; - updateStyle(); - } - setrendermode({EFFECT}) { - resizeMode = EFFECT; - updateStyle(); - } - renderscale({X, Y}) { - Scratch.vm.renderer.resize(X, Y); - } + class CanvasEffects { + getInfo() { + return { + id: "theshovelcanvaseffects", + name: "Canvas Effects", + blocks: [ + { + opcode: "seteffect", + blockType: Scratch.BlockType.COMMAND, + text: "set canvas [EFFECT] to [NUMBER]", + arguments: { + EFFECT: { + type: Scratch.ArgumentType.STRING, + menu: "EFFECTMENU", + }, + NUMBER: { + type: Scratch.ArgumentType.NUMBER, + }, + }, + }, + { + opcode: "geteffect", + blockType: Scratch.BlockType.REPORTER, + text: "get canvas [EFFECT]", + arguments: { + EFFECT: { + type: Scratch.ArgumentType.STRING, + menu: "EFFECTGETMENU", + }, + }, + }, + { + opcode: "cleareffects", + blockType: Scratch.BlockType.COMMAND, + text: "clear canvas effects", + }, + { + opcode: "renderscale", + blockType: Scratch.BlockType.COMMAND, + text: "set canvas render size to width:[X] height:[Y]", + arguments: { + X: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: 100, + }, + Y: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: 100, + }, + }, + }, + { + opcode: "setrendermode", + blockType: Scratch.BlockType.COMMAND, + text: "set canvas resize rendering mode [EFFECT]", + arguments: { + EFFECT: { + type: Scratch.ArgumentType.STRING, + menu: "RENDERMODE", + }, + }, + }, + ], + menus: { + EFFECTMENU: { + acceptReporters: true, + items: [ + "blur", + "contrast", + "saturation", + "color shift", + "brightness", + "invert", + "sepia", + "transparency", + "scale", + "skew X", + "skew Y", + "offset X", + "offset Y", + "rotation", + "border radius", + ], + }, + RENDERMODE: { + acceptReporters: true, + items: ["pixelated", "default"], + }, + EFFECTGETMENU: { + acceptReporters: true, + // this contains 'resize rendering mode', EFFECTMENU does not + items: [ + "blur", + "contrast", + "saturation", + "color shift", + "brightness", + "invert", + "resize rendering mode", + "sepia", + "transparency", + "scale", + "skew X", + "skew Y", + "offset X", + "offset Y", + "rotation", + "border radius", + ], + }, + }, + }; + } + geteffect({ EFFECT }) { + if (EFFECT === "blur") { + return blur; + } else if (EFFECT === "contrast") { + return contrast; + } else if (EFFECT === "saturation") { + return saturation; + } else if (EFFECT === "color shift") { + return color; + } else if (EFFECT === "brightness") { + return brightness; + } else if (EFFECT === "invert") { + return invert; + } else if (EFFECT === "resize rendering mode") { + return resizeMode; + } else if (EFFECT === "sepia") { + return sepia; + } else if (EFFECT === "transparency") { + return transparency; + } else if (EFFECT === "scale") { + return scale; + } else if (EFFECT === "skew X") { + return skewX; + } else if (EFFECT === "skew Y") { + return skewY; + } else if (EFFECT === "offset X") { + return offsetX; + } else if (EFFECT === "offset Y") { + return offsetY; + } else if (EFFECT === "rotation") { + return rotation; + } else if (EFFECT === "border radius") { + return borderRadius; + } + return ""; + } + seteffect({ EFFECT, NUMBER }) { + NUMBER = Scratch.Cast.toNumber(NUMBER); + if (EFFECT === "blur") { + blur = NUMBER; + } else if (EFFECT === "contrast") { + contrast = NUMBER; + } else if (EFFECT === "saturation") { + saturation = NUMBER; + } else if (EFFECT === "color shift") { + color = NUMBER; + } else if (EFFECT === "brightness") { + brightness = NUMBER; + } else if (EFFECT === "invert") { + invert = NUMBER; + } else if (EFFECT === "sepia") { + sepia = NUMBER; + } else if (EFFECT === "transparency") { + transparency = NUMBER; + } else if (EFFECT === "scale") { + scale = NUMBER; + } else if (EFFECT === "skew X") { + skewX = NUMBER; + } else if (EFFECT === "skew Y") { + skewY = NUMBER; + } else if (EFFECT === "offset X") { + offsetX = NUMBER; + } else if (EFFECT === "offset Y") { + offsetY = NUMBER; + } else if (EFFECT === "rotation") { + rotation = NUMBER; + } else if (EFFECT === "border radius") { + borderRadius = NUMBER; + } + updateStyle(); + } + cleareffects() { + borderRadius = 0; + rotation = 0; + offsetY = 0; + offsetX = 0; + skewY = 0; + skewX = 0; + scale = 100; + transparency = 0; + sepia = 0; + blur = 0; + contrast = 100; + saturation = 100; + color = 0; + brightness = 100; + invert = 0; + resizeMode = "default"; + updateStyle(); + } + setrendermode({ EFFECT }) { + resizeMode = EFFECT; + updateStyle(); + } + renderscale({ X, Y }) { + Scratch.vm.renderer.resize(X, Y); } - Scratch.extensions.register(new CanvasEffects()); + } + Scratch.extensions.register(new CanvasEffects()); })(Scratch); diff --git a/extensions/TheShovel/CustomStyles.js b/extensions/TheShovel/CustomStyles.js new file mode 100644 index 0000000000..fda63171b0 --- /dev/null +++ b/extensions/TheShovel/CustomStyles.js @@ -0,0 +1,760 @@ +// Name: Custom Styles +// ID: shovelcss +// Description: Customize the appearance of variable monitors and prompts in your project. +// By: TheShovel + +// Thanks LilyMakesThings for the awesome banner! +(function (Scratch) { + "use strict"; + + // Styles + let monitorText = ""; + let monitorBorder = ""; + let monitorBackgroundColor = ""; + let variableValueBackground = ""; + let variableValueTextColor = ""; + let listFooterBackground = ""; + let listHeaderBackground = ""; + let listValueText = ""; + let listValueBackground = ""; + let variableValueRoundness = -1; + let listValueRoundness = -1; + let monitorBackgroundRoundness = -1; + let monitorBackgroundBorderWidth = -1; + let allowScrolling = ""; + let askBackground = ""; + let askBackgroundRoundness = -1; + let askBackgroundBorderWidth = -1; + let askButtonBackground = ""; + let askButtonRoundness = -1; + let askInputBackground = ""; + let askInputRoundness = -1; + let askInputBorderWidth = -1; + let askBoxIcon = ""; + let askInputText = ""; + let askButtonImage = ""; + let askInputBorder = ""; + + // CSS selectors + let monitorRoot; + let monitorValue; + let monitorListHeader; + let monitorListFooter; + let monitorRowValueOuter; + let monitorRowsInner; + let monitorRowsScroller; + let monitorRowIndex; + let monitorValueLarge; + let askBoxBG; + let askBoxButton; + let askBoxInner; + let askBoxBorderMain; + let askBoxBorderOuter; + if (typeof scaffolding !== "undefined") { + monitorRoot = ".sc-monitor-root"; + monitorValue = ".sc-monitor-value"; + monitorListHeader = ".sc-monitor-list-label"; + monitorListFooter = ".sc-monitor-list-footer"; + monitorRowValueOuter = ".sc-monitor-row-value-outer"; + monitorRowsInner = ".sc-monitor-rows-inner"; + monitorRowsScroller = monitorRowsInner; + monitorRowIndex = ".sc-monitor-row-index"; + monitorValueLarge = ".sc-monitor-large-value"; + askBoxBG = ".sc-question-inner"; + askBoxButton = ".sc-question-submit-button"; + askBoxInner = ".sc-question-input"; + askBoxBorderMain = ".sc-question-input:hover"; + askBoxBorderOuter = ".sc-question-input:focus"; + } else { + monitorRoot = 'div[class^="monitor_monitor-container_"]'; + monitorValue = 'div[class^="monitor_value_"]'; + monitorListHeader = 'div[class^="monitor_list-header_"]'; + monitorListFooter = 'div[class^="monitor_list-footer_"]'; + monitorRowValueOuter = 'div[class^="monitor_list-value_"]'; + monitorRowsInner = 'div[class^="monitor_list-body_"]'; + monitorRowsScroller = + 'div[class^="monitor_list-body_"] > .ReactVirtualized__List'; + monitorRowIndex = 'div[class^="monitor_list-index_"]'; + monitorValueLarge = 'div[class^="monitor_large-value_"]'; + askBoxBG = 'div[class^="question_question-container_"]'; + askBoxButton = 'button[class^="question_question-submit-button_"]'; + askBoxInner = + '[class^="question_question-container_"] input[class^="input_input-form_"]'; + askBoxIcon = 'img[class^="question_question-submit-button-icon_"]'; + askBoxBorderMain = + '[class^="question_question-input_"] input:focus, [class^="question_question-input_"] input:hover'; + askBoxBorderOuter = '[class^="question_question-input_"] > input:focus'; + } + + const ColorIcon = + ""; + const BorderIcon = + ""; + const extensionIcon = + ""; + const miscIcon = + ""; + const TransparentIcon = + ""; + const GradientIcon = + ""; + const PictureIcon = + ""; + const ResetIcon = + ""; + + const stylesheet = document.createElement("style"); + stylesheet.className = "shovelcss-style"; + // end of for higher precedence than other sheets + document.body.appendChild(stylesheet); + + const applyCSS = () => { + let css = ""; + + // We assume all values are sanitized when they are set, so then we can just use them as-is here. + + if (monitorText) { + css += `${monitorRoot}, ${monitorListFooter}, ${monitorListHeader}, ${monitorRowIndex} { color: ${monitorText}; }`; + } + if (monitorBackgroundColor) { + css += `${monitorRoot}, ${monitorRowsInner} { background: ${monitorBackgroundColor}; }`; + } + if (monitorBorder) { + css += `${monitorRoot} { border-color: ${monitorBorder}; }`; + } + if (monitorBackgroundRoundness >= 0) { + css += `${monitorRoot} { border-radius: ${monitorBackgroundRoundness}px; }`; + } + if (monitorBackgroundBorderWidth >= 0) { + css += `${monitorRoot} { border-width: ${monitorBackgroundBorderWidth}px; }`; + } + if (variableValueBackground) { + css += `${monitorValue}, ${monitorValueLarge} { background: ${variableValueBackground} !important; }`; + } + if (variableValueTextColor) { + css += `${monitorValue}, ${monitorValueLarge} { color: ${variableValueTextColor}; }`; + } + if (variableValueRoundness >= 0) { + css += `${monitorValue} { border-radius: ${variableValueRoundness}px; }`; + } + if (listHeaderBackground) { + css += `${monitorListHeader} { background: ${listHeaderBackground}; }`; + } + if (listFooterBackground) { + css += `${monitorListFooter} { background: ${listHeaderBackground}; }`; + } + if (listValueBackground) { + css += `${monitorRowValueOuter} { background: ${listValueBackground} !important; }`; + } + if (listValueText) { + css += `${monitorRowValueOuter} { color: ${listValueText}; }`; + } + if (listValueRoundness >= 0) { + css += `${monitorRowValueOuter} { border-radius: ${listValueRoundness}px; }`; + } + if (allowScrolling) { + css += `${monitorRowsScroller} { overflow: ${allowScrolling} !important; }`; + } + if (askBackground) { + css += `${askBoxBG} { background: ${askBackground} !important; border: none !important; }`; + } + if (askBackgroundRoundness >= 0) { + css += `${askBoxBG} { border-radius: ${askBackgroundRoundness}px !important; }`; + } + if (askBackgroundBorderWidth >= 0) { + css += `${askBoxBG} { border-width: ${askBackgroundBorderWidth}px !important; }`; + } + if (askButtonBackground) { + css += `${askBoxButton} { background-color: ${askButtonBackground}; }`; + } + if (askButtonRoundness >= 0) { + css += `${askBoxButton} { border-radius: ${askButtonRoundness}px !important; }`; + } + if (askInputBackground) { + css += `${askBoxInner} { background: ${askInputBackground} !important; }`; + css += `${askBoxInner} { border: none !important; }`; + } + if (askInputText) { + css += `${askBoxInner} { color: ${askInputText} !important; }`; + } + if (askInputRoundness >= 0) { + css += `${askBoxInner} { border-radius: ${askInputRoundness}px !important; }`; + } + if (askInputBorderWidth >= 0) { + css += `${askBoxInner} { border-width: ${askInputBorderWidth}px !important; }`; + } + if (askButtonImage) { + css += `${askBoxButton} { background-image: url("${encodeURI( + askButtonImage + )}") !important; background-repeat: no-repeat; background-size: contain; }`; + css += `${askBoxIcon} { visibility: hidden; }`; + } + if (askInputBorder) { + css += `${askBoxBorderMain}, ${askBoxBorderOuter} { border-color: ${askInputBorder} !important; }`; + css += `${askBoxBorderOuter} { box-shadow: none !important; }`; + } + + stylesheet.textContent = css; + }; + + const getMonitorRoot = (id) => { + const allMonitors = document.querySelectorAll(monitorRoot); + for (const monitor of allMonitors) { + if (monitor.dataset.id === id) { + return monitor; + } + } + return null; + }; + + /** + * @param {string} id + * @param {number} x + * @param {number} y + */ + const setMonitorPosition = (id, x, y) => { + const root = getMonitorRoot(id); + if (root) { + root.style.transform = `translate(${x}px, ${y}px)`; + root.style.left = "0px"; + root.style.top = "0px"; + } + }; + + /** + * @param {VM.Target} target + * @param {string} name + * @param {VM.VariableType} type + * @param {number} x + * @param {number} y + */ + const setVariableMonitorPosition = (target, name, type, x, y) => { + // @ts-expect-error + const variable = target.lookupVariableByNameAndType(name, type); + if (variable) { + // @ts-expect-error + setMonitorPosition(variable.id, x, y); + } + }; + + const parseColor = (color, callback) => { + color = Scratch.Cast.toString(color); + + // These might have some exponential backtracking/ReDoS, but that's not really a concern here. + // If a project wanted to get stuck in an infinite loop, there are so many other ways to do that. + + // Simple color code or name + if (/^#?[a-z0-9]+$/.test(color)) { + callback(color); + return; + } + + // Simple linear gradient + if (/^linear-gradient\(\d+deg,#?[a-z0-9]+,#?[a-z0-9]+\)$/.test(color)) { + callback(color); + return; + } + + // URL + // see list of non-escaped characters: + // https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/encodeURI#description + const match = color.match(/^url\("([A-Za-z0-9\-_.!~*'();/?:@&=+$,#]+)"\)$/); + if (match) { + const url = match[1]; + return Scratch.canFetch(url).then((allowed) => { + if (allowed) { + callback(color); + } + }); + } + + console.error("Invalid color", color); + }; + + class MonitorStyles { + getInfo() { + return { + id: "shovelcss", + name: "Custom Styles", + menuIconURI: extensionIcon, + color1: "#0072d6", + color2: "#0064bc", + color3: "#01539b", + blocks: [ + { + blockIconURI: ColorIcon, + opcode: "changecss", + blockType: Scratch.BlockType.COMMAND, + text: "set [COLORABLE] to [COLOR]", + arguments: { + COLORABLE: { + type: Scratch.ArgumentType.STRING, + menu: "COLORABLE_MENU", + }, + COLOR: { + type: Scratch.ArgumentType.COLOR, + defaultValue: "#ff0000", + }, + }, + }, + { + blockIconURI: GradientIcon, + opcode: "gradientAngle", + blockType: Scratch.BlockType.REPORTER, + text: "make a gradient with [COLOR1] and [COLOR2] at angle [ANGLE]", + arguments: { + COLOR1: { + type: Scratch.ArgumentType.COLOR, + defaultValue: "#ff0000", + }, + COLOR2: { + type: Scratch.ArgumentType.COLOR, + defaultValue: "#6ed02d", + }, + ANGLE: { + type: Scratch.ArgumentType.ANGLE, + defaultValue: "90", + }, + }, + }, + { + blockIconURI: TransparentIcon, + disableMonitor: true, + opcode: "transparentinput", + blockType: Scratch.BlockType.REPORTER, + text: "transparent", + }, + { + blockIconURI: PictureIcon, + disableMonitor: true, + opcode: "pictureinput", + blockType: Scratch.BlockType.REPORTER, + text: "image [URL]", + arguments: { + URL: { + type: Scratch.ArgumentType.STRING, + defaultValue: "https://extensions.turbowarp.org/dango.png", + }, + }, + }, + "---", + { + blockIconURI: PictureIcon, + disableMonitor: true, + opcode: "setAskURI", + blockType: Scratch.BlockType.COMMAND, + text: "set ask prompt button image to [URL]", + arguments: { + URL: { + type: Scratch.ArgumentType.STRING, + defaultValue: "https://extensions.turbowarp.org/dango.png", + }, + }, + }, + "---", + { + blockIconURI: BorderIcon, + opcode: "setbordersize", + blockType: Scratch.BlockType.COMMAND, + text: "set border width of [BORDER] to [SIZE]", + arguments: { + BORDER: { + type: Scratch.ArgumentType.STRING, + menu: "BORDER_WIDTH_MENU", + }, + SIZE: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "2", + }, + }, + }, + { + blockIconURI: BorderIcon, + opcode: "setborderradius", + blockType: Scratch.BlockType.COMMAND, + text: "set roundness of [CORNER] to [SIZE]", + arguments: { + SIZE: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "4", + }, + CORNER: { + type: Scratch.ArgumentType.STRING, + menu: "BORDER_ROUNDNESS_MENU", + }, + }, + }, + "---", + { + blockIconURI: ResetIcon, + opcode: "clearCSS", + blockType: Scratch.BlockType.COMMAND, + text: "reset styles", + }, + "---", + { + blockIconURI: miscIcon, + opcode: "allowscrollrule", + blockType: Scratch.BlockType.COMMAND, + text: "set list scrolling to [SCROLLRULE]", + arguments: { + SCROLLRULE: { + type: Scratch.ArgumentType.STRING, + menu: "SCROLL_MENU", + }, + }, + }, + { + blockIconURI: miscIcon, + opcode: "getValue", + blockType: Scratch.BlockType.REPORTER, + text: "get [ITEM]", + arguments: { + ITEM: { + type: Scratch.ArgumentType.STRING, + menu: "VALUEGET_LIST", + }, + }, + }, + "---", + { + blockIconURI: miscIcon, + opcode: "setvarpos", + blockType: Scratch.BlockType.COMMAND, + text: "set position of variable [NAME] to x: [X] y: [Y]", + arguments: { + X: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + Y: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "my variable", + }, + }, + }, + { + blockIconURI: miscIcon, + opcode: "setlistpos", + blockType: Scratch.BlockType.COMMAND, + text: "set position of list [NAME] to x: [X] y: [Y]", + arguments: { + X: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + Y: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "my variable", + }, + }, + }, + ], + // Accepting reporters because there can't be errors in case the value is not correct + menus: { + COLORABLE_MENU: { + acceptReporters: true, + items: [ + "monitor text", + "monitor background", + "monitor border", + "variable value background", + "variable value text", + "list header background", + "list footer background", + "list value background", + "list value text", + "ask prompt background", + "ask prompt button background", + "ask prompt input background", + "ask prompt input text", + "ask prompt input border", + ], + }, + BORDER_WIDTH_MENU: { + acceptReporters: true, + items: [ + "monitor background", + "ask prompt background", + "ask prompt input", + ], + }, + BORDER_ROUNDNESS_MENU: { + acceptReporters: true, + items: [ + "monitor background", + "variable value", + "list value", + "ask prompt background", + "ask prompt button", + "ask prompt input", + ], + }, + SCROLL_MENU: { + acceptReporters: true, + items: ["enabled", "disabled"], + }, + VALUEGET_LIST: { + acceptReporters: true, + items: [ + "monitor text", + "monitor background", + "monitor border color", + "variable value background", + "variable value text", + "list header background", + "list footer background", + "list value background", + "list value text", + "ask prompt background", + "ask prompt button background", + "ask prompt input background", + "ask prompt input text", + "ask prompt input border", + "monitor background border width", + "ask prompt background border width", + "ask prompt input border width", + "monitor background roundness", + "variable value roundness", + "list value roundness", + "ask prompt background roundness", + "ask prompt button roundness", + "ask prompt input roundness", + "ask prompt button image", + "list scroll rule", + ], + }, + }, + }; + } + + changecss(args) { + return parseColor(args.COLOR, (color) => { + if (args.COLORABLE === "monitor text") { + monitorText = color; + } else if (args.COLORABLE === "monitor background") { + monitorBackgroundColor = color; + } else if (args.COLORABLE === "monitor border") { + monitorBorder = color; + } else if (args.COLORABLE === "variable value background") { + variableValueBackground = color; + } else if (args.COLORABLE === "variable value text") { + variableValueTextColor = color; + } else if (args.COLORABLE === "list header background") { + listHeaderBackground = color; + } else if (args.COLORABLE === "list footer background") { + listFooterBackground = color; + } else if (args.COLORABLE === "list value background") { + listValueBackground = color; + } else if (args.COLORABLE === "list value text") { + listValueText = color; + } else if (args.COLORABLE === "ask prompt background") { + askBackground = color; + } else if (args.COLORABLE === "ask prompt button background") { + askButtonBackground = color; + } else if (args.COLORABLE === "ask prompt input background") { + askInputBackground = color; + } else if (args.COLORABLE === "ask prompt input text") { + askInputText = color; + } else if (args.COLORABLE === "ask prompt input border") { + askInputBorder = color; + } + + applyCSS(); + }); + } + + gradientAngle(args) { + return ( + "linear-gradient(" + + args.ANGLE + + "deg," + + args.COLOR1 + + "," + + args.COLOR2 + + ")" + ); + } + + setbordersize(args) { + const size = Scratch.Cast.toNumber(args.SIZE); + if (args.BORDER === "monitor background") { + monitorBackgroundBorderWidth = size; + } else if (args.BORDER === "ask prompt background") { + askBackgroundBorderWidth = size; + } else if (args.BORDER === "ask prompt input") { + askInputBorderWidth = size; + } + applyCSS(); + } + + setborderradius(args) { + const size = Scratch.Cast.toNumber(args.SIZE); + if (args.CORNER === "monitor background") { + monitorBackgroundRoundness = size; + } else if (args.CORNER === "variable value") { + variableValueRoundness = size; + } else if (args.CORNER === "list value") { + listValueRoundness = size; + } else if (args.CORNER === "ask prompt background") { + askBackgroundRoundness = size; + } else if (args.CORNER === "ask prompt button") { + askButtonRoundness = size; + } else if (args.CORNER === "ask prompt input") { + askInputRoundness = size; + } + applyCSS(); + } + + allowscrollrule(args) { + if (args.SCROLLRULE === "enabled") { + allowScrolling = "auto"; + } else { + allowScrolling = "hidden"; + } + applyCSS(); + } + + setvarpos(args, util) { + setVariableMonitorPosition( + util.target, + args.NAME, + "", + Scratch.Cast.toNumber(args.X) + Scratch.vm.runtime.stageWidth / 2, + Scratch.vm.runtime.stageHeight / 2 - Scratch.Cast.toNumber(args.Y) + ); + } + + setlistpos(args, util) { + setVariableMonitorPosition( + util.target, + args.NAME, + "list", + Scratch.Cast.toNumber(args.X) + Scratch.vm.runtime.stageWidth / 2, + Scratch.vm.runtime.stageHeight / 2 - Scratch.Cast.toNumber(args.Y) + ); + } + + help() { + alert( + "\nThis is a short introduction to how to use the Monitor Styles extension!\n\n𝗟𝗼𝗼𝗸𝘀 𝗯𝗹𝗼𝗰𝗸𝘀\nThese blocks change the appearance of the variable and list didsplays. You can use the drop-down menu to select what component you want to modify. 𝙏𝙝𝙚 𝙘𝙤𝙡𝙤𝙧 𝙗𝙡𝙤𝙘𝙠 modifieas the color of a component. You can use the 𝙜𝙧𝙖𝙙𝙞𝙚𝙣𝙩 block inside the color input, to create gradients or the 𝙄𝙢𝙖𝙜𝙚 block to use a image instead of solid colors. 𝙏𝙝𝙚𝙨𝙚 𝙩𝙬𝙤 𝙤𝙣𝙡𝙮 𝙬𝙤𝙧𝙠 𝙤𝙣 𝙘𝙚𝙧𝙩𝙖𝙞𝙣 𝙘𝙤𝙢𝙥𝙤𝙣𝙚𝙣𝙩𝙨! You can also use the 𝙩𝙧𝙖𝙣𝙨𝙥𝙖𝙧𝙚𝙣𝙩 𝙗𝙡𝙤𝙘𝙠 as a color input, to make components invisible. The 𝙗𝙤𝙧𝙙𝙚𝙧 𝙗𝙡𝙤𝙘𝙠𝙨 modify the borders of components.\n\n𝗦𝗲𝗻𝘀𝗶𝗻𝗴 𝗯𝗹𝗼𝗰𝗸𝘀\nThese blocks can change the behaviour of certain components. The 𝙨𝙘𝙧𝙤𝙡𝙡 𝙧𝙪𝙡𝙚 block change the behaviour for lists. On 'auto' they will show the scroll bar, and allow you to school, but on 'hidden', they won't let you do that, and the scroll bar will be hidden.\n\n𝗠𝗼𝘁𝗶𝗼𝗻 𝗯𝗹𝗼𝗰𝗸𝘀\nThese blocks allow you to move variable and list displays around. You need to use their 𝙡𝙖𝙗𝙚𝙡 𝙣𝙖𝙢𝙚. The label name is the text that displays on the monitor. For example, a 'for this sprite only' variable will be like 'Sprite1: my variable'." + ); + } + + transparentinput() { + return "transparent"; + } + + pictureinput(args) { + return `url("${encodeURI(args.URL)}")`; + } + + clearCSS() { + monitorText = ""; + monitorBorder = ""; + monitorBackgroundColor = ""; + variableValueBackground = ""; + variableValueTextColor = ""; + listFooterBackground = ""; + listHeaderBackground = ""; + listValueText = ""; + listValueBackground = ""; + variableValueRoundness = -1; + listValueRoundness = -1; + monitorBackgroundRoundness = -1; + monitorBackgroundBorderWidth = -1; + allowScrolling = ""; + askBackground = ""; + askBackgroundRoundness = -1; + askBackgroundBorderWidth = -1; + askButtonBackground = ""; + askButtonRoundness = -1; + askInputBackground = ""; + askInputRoundness = -1; + askInputBorderWidth = -1; + askBoxIcon = ""; + askInputText = ""; + askButtonImage = ""; + askInputBorder = ""; + applyCSS(); + } + + getValue(args) { + if (args.ITEM === "monitor text") { + return monitorText; + } else if (args.ITEM === "monitor background") { + return monitorBackgroundColor; + } else if (args.ITEM === "monitor border color") { + return monitorBorder; + } else if (args.ITEM === "variable value background") { + return variableValueBackground; + } else if (args.ITEM === "variable value text") { + return variableValueTextColor; + } else if (args.ITEM === "list header background") { + return listHeaderBackground; + } else if (args.ITEM === "list footer background") { + return listFooterBackground; + } else if (args.ITEM === "list value background") { + return listValueBackground; + } else if (args.ITEM === "list value text") { + return listValueText; + } else if (args.ITEM === "ask prompt background") { + return askBackground; + } else if (args.ITEM === "ask prompt button background") { + return askButtonBackground; + } else if (args.ITEM === "ask prompt input background") { + return askInputBackground; + } else if (args.ITEM === "ask prompt input text") { + return askInputText; + } else if (args.ITEM === "ask prompt input border") { + return askInputBorder; + } else if (args.ITEM === "monitor background border width") { + return monitorBackgroundBorderWidth; + } else if (args.ITEM === "ask prompt background border width") { + return askBackgroundBorderWidth; + } else if (args.ITEM === "ask prompt input border width") { + return askInputBorderWidth; + } else if (args.ITEM === "monitor background roundness") { + return monitorBackgroundRoundness; + } else if (args.ITEM === "variable value roundness") { + return variableValueRoundness; + } else if (args.ITEM === "list value roundness") { + return listValueRoundness; + } else if (args.ITEM === "ask prompt background roundness") { + return askBackgroundRoundness; + } else if (args.ITEM === "ask prompt button roundness") { + return askButtonRoundness; + } else if (args.ITEM === "ask prompt input roundness") { + return askInputRoundness; + } else if (args.ITEM === "ask prompt button image") { + return askButtonImage; + } else if (args.ITEM === "list scrolling") { + if (allowScrolling === "auto") { + return "enabled"; + } else { + return "disabled"; + } + } + return ""; + } + + setAskURI(args) { + return Scratch.canFetch(args.URL).then((allowed) => { + if (allowed) { + askButtonImage = args.URL; + applyCSS(); + } + }); + } + } + + Scratch.extensions.register(new MonitorStyles()); +})(Scratch); diff --git a/extensions/TheShovel/LZ-String.js b/extensions/TheShovel/LZ-String.js new file mode 100644 index 0000000000..635ed972ef --- /dev/null +++ b/extensions/TheShovel/LZ-String.js @@ -0,0 +1,435 @@ +// Name: LZ Compress +// ID: shovellzcompress +// Description: Compress and decompress text using lz-string. + +(function (Scratch) { + "use strict"; + + /* eslint-disable */ + // Code from https://github.com/pieroxy/lz-string/ + // MIT License + + // Copyright (c) 2013 pieroxy + + // Permission is hereby granted, free of charge, to any person obtaining a copy + // of this software and associated documentation files (the "Software"), to deal + // in the Software without restriction, including without limitation the rights + // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + // copies of the Software, and to permit persons to whom the Software is + // furnished to do so, subject to the following conditions: + + // The above copyright notice and this permission notice shall be included in all + // copies or substantial portions of the Software. + + // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE + // SOFTWARE. + var LZString = (function () { + var r = String.fromCharCode, + o = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=", + n = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+-$", + e = {}; + + function t(r, o) { + if (!e[r]) { + e[r] = {}; + for (var n = 0; n < r.length; n++) e[r][r.charAt(n)] = n; + } + return e[r][o]; + } + var i = { + compressToBase64: function (r) { + if (null == r) return ""; + var n = i._compress(r, 6, function (r) { + return o.charAt(r); + }); + switch (n.length % 4) { + default: + case 0: + return n; + case 1: + return n + "==="; + case 2: + return n + "=="; + case 3: + return n + "="; + } + }, + decompressFromBase64: function (r) { + return null == r + ? "" + : "" == r + ? null + : i._decompress(r.length, 32, function (n) { + return t(o, r.charAt(n)); + }); + }, + compressToUTF16: function (o) { + return null == o + ? "" + : i._compress(o, 15, function (o) { + return r(o + 32); + }) + " "; + }, + decompressFromUTF16: function (r) { + return null == r + ? "" + : "" == r + ? null + : i._decompress(r.length, 16384, function (o) { + return r.charCodeAt(o) - 32; + }); + }, + compressToUint8Array: function (r) { + for ( + var o = i.compress(r), + n = new Uint8Array(2 * o.length), + e = 0, + t = o.length; + e < t; + e++ + ) { + var s = o.charCodeAt(e); + (n[2 * e] = s >>> 8), (n[2 * e + 1] = s % 256); + } + return n; + }, + decompressFromUint8Array: function (o) { + if (null == o) return i.decompress(o); + for (var n = new Array(o.length / 2), e = 0, t = n.length; e < t; e++) + n[e] = 256 * o[2 * e] + o[2 * e + 1]; + var s = []; + return ( + n.forEach(function (o) { + s.push(r(o)); + }), + i.decompress(s.join("")) + ); + }, + compressToEncodedURIComponent: function (r) { + return null == r + ? "" + : i._compress(r, 6, function (r) { + return n.charAt(r); + }); + }, + decompressFromEncodedURIComponent: function (r) { + return null == r + ? "" + : "" == r + ? null + : ((r = r.replace(/ /g, "+")), + i._decompress(r.length, 32, function (o) { + return t(n, r.charAt(o)); + })); + }, + compress: function (o) { + return i._compress(o, 16, function (o) { + return r(o); + }); + }, + _compress: function (r, o, n) { + if (null == r) return ""; + var e, + t, + i, + s = {}, + u = {}, + a = "", + p = "", + c = "", + l = 2, + f = 3, + h = 2, + d = [], + m = 0, + v = 0; + for (i = 0; i < r.length; i += 1) + if ( + ((a = r.charAt(i)), + Object.prototype.hasOwnProperty.call(s, a) || + ((s[a] = f++), (u[a] = !0)), + (p = c + a), + Object.prototype.hasOwnProperty.call(s, p)) + ) + c = p; + else { + if (Object.prototype.hasOwnProperty.call(u, c)) { + if (c.charCodeAt(0) < 256) { + for (e = 0; e < h; e++) + (m <<= 1), + v == o - 1 ? ((v = 0), d.push(n(m)), (m = 0)) : v++; + for (t = c.charCodeAt(0), e = 0; e < 8; e++) + (m = (m << 1) | (1 & t)), + v == o - 1 ? ((v = 0), d.push(n(m)), (m = 0)) : v++, + (t >>= 1); + } else { + for (t = 1, e = 0; e < h; e++) + (m = (m << 1) | t), + v == o - 1 ? ((v = 0), d.push(n(m)), (m = 0)) : v++, + (t = 0); + for (t = c.charCodeAt(0), e = 0; e < 16; e++) + (m = (m << 1) | (1 & t)), + v == o - 1 ? ((v = 0), d.push(n(m)), (m = 0)) : v++, + (t >>= 1); + } + 0 == --l && ((l = Math.pow(2, h)), h++), delete u[c]; + } else + for (t = s[c], e = 0; e < h; e++) + (m = (m << 1) | (1 & t)), + v == o - 1 ? ((v = 0), d.push(n(m)), (m = 0)) : v++, + (t >>= 1); + 0 == --l && ((l = Math.pow(2, h)), h++), + (s[p] = f++), + (c = String(a)); + } + if ("" !== c) { + if (Object.prototype.hasOwnProperty.call(u, c)) { + if (c.charCodeAt(0) < 256) { + for (e = 0; e < h; e++) + (m <<= 1), v == o - 1 ? ((v = 0), d.push(n(m)), (m = 0)) : v++; + for (t = c.charCodeAt(0), e = 0; e < 8; e++) + (m = (m << 1) | (1 & t)), + v == o - 1 ? ((v = 0), d.push(n(m)), (m = 0)) : v++, + (t >>= 1); + } else { + for (t = 1, e = 0; e < h; e++) + (m = (m << 1) | t), + v == o - 1 ? ((v = 0), d.push(n(m)), (m = 0)) : v++, + (t = 0); + for (t = c.charCodeAt(0), e = 0; e < 16; e++) + (m = (m << 1) | (1 & t)), + v == o - 1 ? ((v = 0), d.push(n(m)), (m = 0)) : v++, + (t >>= 1); + } + 0 == --l && ((l = Math.pow(2, h)), h++), delete u[c]; + } else + for (t = s[c], e = 0; e < h; e++) + (m = (m << 1) | (1 & t)), + v == o - 1 ? ((v = 0), d.push(n(m)), (m = 0)) : v++, + (t >>= 1); + 0 == --l && ((l = Math.pow(2, h)), h++); + } + for (t = 2, e = 0; e < h; e++) + (m = (m << 1) | (1 & t)), + v == o - 1 ? ((v = 0), d.push(n(m)), (m = 0)) : v++, + (t >>= 1); + for (;;) { + if (((m <<= 1), v == o - 1)) { + d.push(n(m)); + break; + } + v++; + } + return d.join(""); + }, + decompress: function (r) { + return null == r + ? "" + : "" == r + ? null + : i._decompress(r.length, 32768, function (o) { + return r.charCodeAt(o); + }); + }, + _decompress: function (o, n, e) { + var t, + i, + s, + u, + a, + p, + c, + l = [], + f = 4, + h = 4, + d = 3, + m = "", + v = [], + g = { + val: e(0), + position: n, + index: 1, + }; + for (t = 0; t < 3; t += 1) l[t] = t; + for (s = 0, a = Math.pow(2, 2), p = 1; p != a; ) + (u = g.val & g.position), + (g.position >>= 1), + 0 == g.position && ((g.position = n), (g.val = e(g.index++))), + (s |= (u > 0 ? 1 : 0) * p), + (p <<= 1); + switch (s) { + case 0: + for (s = 0, a = Math.pow(2, 8), p = 1; p != a; ) + (u = g.val & g.position), + (g.position >>= 1), + 0 == g.position && ((g.position = n), (g.val = e(g.index++))), + (s |= (u > 0 ? 1 : 0) * p), + (p <<= 1); + c = r(s); + break; + case 1: + for (s = 0, a = Math.pow(2, 16), p = 1; p != a; ) + (u = g.val & g.position), + (g.position >>= 1), + 0 == g.position && ((g.position = n), (g.val = e(g.index++))), + (s |= (u > 0 ? 1 : 0) * p), + (p <<= 1); + c = r(s); + break; + case 2: + return ""; + } + for (l[3] = c, i = c, v.push(c); ; ) { + if (g.index > o) return ""; + for (s = 0, a = Math.pow(2, d), p = 1; p != a; ) + (u = g.val & g.position), + (g.position >>= 1), + 0 == g.position && ((g.position = n), (g.val = e(g.index++))), + (s |= (u > 0 ? 1 : 0) * p), + (p <<= 1); + switch ((c = s)) { + case 0: + for (s = 0, a = Math.pow(2, 8), p = 1; p != a; ) + (u = g.val & g.position), + (g.position >>= 1), + 0 == g.position && ((g.position = n), (g.val = e(g.index++))), + (s |= (u > 0 ? 1 : 0) * p), + (p <<= 1); + (l[h++] = r(s)), (c = h - 1), f--; + break; + case 1: + for (s = 0, a = Math.pow(2, 16), p = 1; p != a; ) + (u = g.val & g.position), + (g.position >>= 1), + 0 == g.position && ((g.position = n), (g.val = e(g.index++))), + (s |= (u > 0 ? 1 : 0) * p), + (p <<= 1); + (l[h++] = r(s)), (c = h - 1), f--; + break; + case 2: + return v.join(""); + } + // @ts-ignore + if ((0 == f && ((f = Math.pow(2, d)), d++), l[c])) m = l[c]; + else { + if (c !== h) return null; + m = i + i.charAt(0); + } + v.push(m), + (l[h++] = i + m.charAt(0)), + (i = m), + 0 == --f && ((f = Math.pow(2, d)), d++); + } + }, + }; + return i; + })(); + // @ts-ignore + "function" == typeof define && define.amd + ? define(function () { + return LZString; + // @ts-ignore + }) + : "undefined" != typeof module && null != module + ? (module.exports = LZString) + : "undefined" != typeof angular && + null != angular && + angular.module("LZString", []).factory("LZString", function () { + return LZString; + }); + /* eslint-enable */ + + class lzcompress { + getInfo() { + return { + id: "shovellzcompress", + name: "LZ Compress", + blocks: [ + { + opcode: "compress", + blockType: Scratch.BlockType.REPORTER, + text: "compress [TEXT] to [TYPE]", + arguments: { + TEXT: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Hello world!", + }, + TYPE: { + type: Scratch.ArgumentType.STRING, + menu: "COMPRESSIONTYPES", + }, + }, + }, + { + opcode: "decompress", + blockType: Scratch.BlockType.REPORTER, + text: "decompress [TEXT] from [TYPE]", + arguments: { + TEXT: { + type: Scratch.ArgumentType.STRING, + defaultValue: "҅〶惶@✰Ӏ葀", + }, + TYPE: { + type: Scratch.ArgumentType.STRING, + menu: "COMPRESSIONTYPES", + }, + }, + }, + ], + menus: { + COMPRESSIONTYPES: { + acceptReporters: true, + items: [ + "Raw", + "Base64", + "EncodedURIComponent", + "Uint8Array", + "UTF16", + ], + }, + }, + }; + } + compress(args) { + const text = Scratch.Cast.toString(args.TEXT); + if (args.TYPE == "Raw") { + return LZString.compress(text); + } else if (args.TYPE == "Base64") { + return LZString.compressToBase64(text); + } else if (args.TYPE == "EncodedURIComponent") { + return LZString.compressToEncodedURIComponent(text); + } else if (args.TYPE == "Uint8Array") { + return LZString.compressToUint8Array(text); + } else if (args.TYPE == "UTF16") { + return LZString.compressToUTF16(text); + } + return ""; + } + + decompress(args) { + try { + const text = Scratch.Cast.toString(args.TEXT); + if (args.TYPE == "Raw") { + return LZString.decompress(text) || ""; + } else if (args.TYPE == "Base64") { + return LZString.decompressFromBase64(text) || ""; + } else if (args.TYPE == "EncodedURIComponent") { + return LZString.decompressFromEncodedURIComponent(text) || ""; + } else if (args.TYPE == "Uint8Array") { + return LZString.decompressFromUint8Array(text) || ""; + } else if (args.TYPE == "UTF16") { + return LZString.decompressFromUTF16(text) || ""; + } + } catch (e) { + console.error("decompress error", e); + } + return ""; + } + } + Scratch.extensions.register(new lzcompress()); +})(Scratch); diff --git a/extensions/TheShovel/ShovelUtils.js b/extensions/TheShovel/ShovelUtils.js index bbbba655e0..0c8211915e 100644 --- a/extensions/TheShovel/ShovelUtils.js +++ b/extensions/TheShovel/ShovelUtils.js @@ -1,6 +1,14 @@ +// Name: ShovelUtils +// ID: ShovelUtils +// Description: A bunch of miscellaneous blocks. +// By: TheShovel + (function (Scratch) { - 'use strict'; - console.log("ShovelUtils v1.3"); + "use strict"; + if (!Scratch.extensions.unsandboxed) { + throw new Error("ShovelUtils must be run unsandboxed"); + } + console.log("ShovelUtils v1.4"); const vm = Scratch.vm; // Based on from https://www.growingwiththeweb.com/2017/12/fast-simple-js-fps-counter.html @@ -20,154 +28,175 @@ class ShovelUtils { getInfo() { return { - id: 'ShovelUtils', - name: 'ShovelUtils', - color1: '#f54242', - color2: '#f54242', - color3: '#f54242', + id: "ShovelUtils", + name: "ShovelUtils", + color1: "#f54242", + color2: "#f54242", + color3: "#f54242", blocks: [ { - opcode: 'importImage', + opcode: "importImage", blockType: Scratch.BlockType.COMMAND, - text: 'Import image from [TEXT] name [NAME]', + text: "Import image from [TEXT] name [NAME]", arguments: { TEXT: { type: Scratch.ArgumentType.STRING, - defaultValue: 'https://extensions.turbowarp.org/dango.png', + defaultValue: "https://extensions.turbowarp.org/dango.png", }, NAME: { type: Scratch.ArgumentType.STRING, - defaultValue: 'Dango', - } - } + defaultValue: "Dango", + }, + }, }, { - opcode: 'getlist', + opcode: "getlist", blockType: Scratch.BlockType.REPORTER, - text: 'Get list [TEXT]', + text: "Get list [TEXT]", arguments: { TEXT: { type: Scratch.ArgumentType.STRING, - defaultValue: 'MyList', - } - } + defaultValue: "MyList", + }, + }, }, { - opcode: 'setlist', + opcode: "setlist", blockType: Scratch.BlockType.COMMAND, - text: 'Set list [NAME] to [TEXT]', + text: "Set list [NAME] to [TEXT]", arguments: { TEXT: { type: Scratch.ArgumentType.STRING, - defaultValue: '[1,2]', + defaultValue: "[1,2]", }, NAME: { type: Scratch.ArgumentType.STRING, - defaultValue: 'MyList', - } - } + defaultValue: "MyList", + }, + }, }, { - opcode: 'importSprite', + opcode: "importSprite", blockType: Scratch.BlockType.COMMAND, - text: 'Import sprite from [TEXT]', + text: "Import sprite from [TEXT]", arguments: { TEXT: { type: Scratch.ArgumentType.STRING, - defaultValue: 'Link or data uri here', - } - } + defaultValue: "Link or data uri here", + }, + }, }, { - opcode: 'importSound', + opcode: "importSound", blockType: Scratch.BlockType.COMMAND, - text: 'Import sound from [TEXT] name [NAME]', + text: "Import sound from [TEXT] name [NAME]", arguments: { TEXT: { type: Scratch.ArgumentType.STRING, - defaultValue: 'https://extensions.turbowarp.org/meow.mp3', + defaultValue: "https://extensions.turbowarp.org/meow.mp3", }, NAME: { type: Scratch.ArgumentType.STRING, - defaultValue: 'Meow', - } - } + defaultValue: "Meow", + }, + }, }, { - opcode: 'importProject', + opcode: "importProject", blockType: Scratch.BlockType.COMMAND, - text: 'Import project from [TEXT]', + text: "Import project from [TEXT]", arguments: { TEXT: { type: Scratch.ArgumentType.STRING, - defaultValue: 'https://theshovel.github.io/Bullet-Hell/Bullet%20Hell', - } - } + defaultValue: + "https://theshovel.github.io/Bullet-Hell/Bullet%20Hell", + }, + }, }, { - opcode: 'loadExtension', + opcode: "loadExtension", blockType: Scratch.BlockType.COMMAND, - text: 'Load extension from [TEXT]', + text: "Load extension from [TEXT]", arguments: { TEXT: { type: Scratch.ArgumentType.STRING, - defaultValue: 'https://extensions.turbowarp.org/utilities.js', - } - } + defaultValue: "https://extensions.turbowarp.org/utilities.js", + }, + }, }, { - opcode: 'restartProject', + opcode: "restartProject", blockType: Scratch.BlockType.COMMAND, - text: 'Restart project', + text: "Restart project", arguments: { TEXT: { type: Scratch.ArgumentType.STRING, - defaultValue: '0', - } - } + defaultValue: "0", + }, + }, }, { - opcode: 'deleteSprite', + opcode: "deleteSprite", blockType: Scratch.BlockType.COMMAND, - text: 'Delete sprite [SPRITE]', + text: "Delete sprite [SPRITE]", arguments: { SPRITE: { type: Scratch.ArgumentType.STRING, - defaultValue: 'Sprite1', - } - } + defaultValue: "Sprite1", + }, + }, + }, + { + opcode: "deleteImage", + blockType: Scratch.BlockType.COMMAND, + text: "Delete costume [COSNAME] in [SPRITE]", + arguments: { + COSNAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "costume1", + }, + SPRITE: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Sprite1", + }, + }, }, { - opcode: 'setedtarget', + opcode: "setedtarget", blockType: Scratch.BlockType.COMMAND, - text: 'Set editing target to [NAME]', + text: "Set editing target to [NAME]", arguments: { NAME: { type: Scratch.ArgumentType.STRING, - defaultValue: 'Sprite1', - } - } + defaultValue: "Sprite1", + }, + }, }, { - opcode: 'brightnessByColor', + opcode: "brightnessByColor", blockType: Scratch.BlockType.REPORTER, - text: 'Get brightness of [color]', + text: "Get brightness of [color]", arguments: { color: { type: Scratch.ArgumentType.STRING, - defaultValue: '#ffffff', - } - } + defaultValue: "#ffffff", + }, + }, }, { - opcode: 'getfps', + opcode: "getAllSprites", blockType: Scratch.BlockType.REPORTER, - text: 'Fps' + text: "get all sprites", }, - ] + { + opcode: "getfps", + blockType: Scratch.BlockType.REPORTER, + text: "Fps", + }, + ], }; } @@ -176,23 +205,23 @@ .then((r) => r.arrayBuffer()) .then((arrayBuffer) => { const storage = vm.runtime.storage; - vm.addCostume(NAME + '.PNG', { - name: NAME + '', + vm.addCostume(NAME + ".PNG", { + name: NAME + "", asset: new storage.Asset( storage.AssetType.ImageBitmap, null, // asset id, doesn't need to be set here because of `true` at the end will make Scratch generate it for you storage.DataFormat.PNG, new Uint8Array(arrayBuffer), true - ) + ), }); }); } importSprite({ TEXT }) { Scratch.fetch(TEXT) - .then(r => r.arrayBuffer()) - .then(buffer => vm.addSprite(buffer)) + .then((r) => r.arrayBuffer()) + .then((buffer) => vm.addSprite(buffer)) .then(() => { console.log("Done"); }) @@ -207,8 +236,12 @@ return; } // @ts-expect-error - if (typeof ScratchBlocks !== 'undefined') { - if (!confirm(`Do you want to delete the sprite "${SPRITE}"? This cannot be undone.`)) { + if (typeof ScratchBlocks !== "undefined") { + if ( + !confirm( + `Do you want to delete the sprite "${SPRITE}"? This cannot be undone.` + ) + ) { return; } } @@ -228,24 +261,28 @@ true ); vm.addSound({ - md5: asset.assetId + '.' + asset.dataFormat, + md5: asset.assetId + "." + asset.dataFormat, asset: asset, - name: NAME + '' + name: NAME + "", }); }); } importProject({ TEXT }) { // @ts-ignore - if (typeof ScratchBlocks !== 'undefined') { + if (typeof ScratchBlocks !== "undefined") { // We are in the editor. Ask before loading a new project to avoid unrecoverable data loss. - if (!confirm(`Do you want to import a project from "${TEXT}"? Everything in the current project will be permanently deleted.`)) { + if ( + !confirm( + `Do you want to import a project from "${TEXT}"? Everything in the current project will be permanently deleted.` + ) + ) { return; } } Scratch.fetch(TEXT) - .then(r => r.arrayBuffer()) - .then(buffer => vm.loadProject(buffer)) + .then((r) => r.arrayBuffer()) + .then((buffer) => vm.loadProject(buffer)) .then(() => { console.log("Done"); vm.greenFlag(); @@ -266,7 +303,9 @@ } getlist({ TEXT }) { - const list = vm.runtime.getTargetForStage().lookupVariableByNameAndType(TEXT, 'list'); + const list = vm.runtime + .getTargetForStage() + .lookupVariableByNameAndType(TEXT, "list"); if (list) { return JSON.stringify(list.value); } else { @@ -292,7 +331,9 @@ } } - const list = vm.runtime.getTargetForStage().lookupVariableByNameAndType(NAME, 'list'); + const list = vm.runtime + .getTargetForStage() + .lookupVariableByNameAndType(NAME, "list"); if (!list) { return; // List was not found } @@ -321,15 +362,32 @@ */ brightnessByColor({ color }) { // https://www.w3.org/TR/AERT/#color-contrast - const {r, g, b} = Scratch.Cast.toRgbColorObject(color); - return ((r * 299) + (g * 587) + (b * 114)) / 1000; + const { r, g, b } = Scratch.Cast.toRgbColorObject(color); + return (r * 299 + g * 587 + b * 114) / 1000; } - getfps(){ + getfps() { return fps; } - } + deleteImage({ SPRITE, COSNAME }) { + // 0znzw, since shovel did not add it yet. + const target = vm.runtime.getSpriteTargetByName(SPRITE); + if (!target) { + return; + } + target.deleteCostume(target.getCostumeIndexByName(COSNAME)); + } + + getAllSprites() { + // 0znzw, since shovel did not add it yet. + let sprites = []; + for (const target of vm.runtime.targets) { + if (target.isOriginal) sprites.push(target.sprite.name); + } + return JSON.stringify(sprites); + } + } Scratch.extensions.register(new ShovelUtils()); -// @ts-ignore + // @ts-ignore })(Scratch); diff --git a/extensions/TheShovel/profanity.js b/extensions/TheShovel/profanity.js index 028023e4db..337dc397dc 100644 --- a/extensions/TheShovel/profanity.js +++ b/extensions/TheShovel/profanity.js @@ -1,8 +1,18 @@ -(function(Scratch) { - 'use strict'; +(function (Scratch) { + "use strict"; - const encode = (str) => btoa(str).split('').map((i) => String.fromCharCode(i.charCodeAt(0) + 1)).join(''); - const decode = (str) => atob(str.split('').map((i) => String.fromCharCode(i.charCodeAt(0) - 1)).join('')); + const encode = (str) => + btoa(str) + .split("") + .map((i) => String.fromCharCode(i.charCodeAt(0) + 1)) + .join(""); + const decode = (str) => + atob( + str + .split("") + .map((i) => String.fromCharCode(i.charCodeAt(0) - 1)) + .join("") + ); // A forewarning for the reader: // This list contains some very bad naughty words, so we've encoded it in a way that @@ -214,44 +224,44 @@ "e3iwdnV>", "e3m{[XG{dx>>", "e3m{[XG{d3W{", - "e3:x" + "e3:x", ].map(decode); // Put the longest words first so that if "test" and "tests" are in the word list in // that order, redacting "tests" will give "***" instead of "***s" NAUGHTY_WORDS.sort((a, b) => b.length - a.length); - const regex = new RegExp(NAUGHTY_WORDS.join('|'), 'gi'); + const regex = new RegExp(NAUGHTY_WORDS.join("|"), "gi"); class Profanity { - getInfo () { + getInfo() { return { - id: 'theshovelprofanity', - name: 'Bad Word Remover', - color1: '#cf6a3c', - color2: '#cf6a3c', - color3: '#cf6a3c', + id: "theshovelprofanity", + name: "Bad Word Remover", + color1: "#cf6a3c", + color2: "#cf6a3c", + color3: "#cf6a3c", blocks: [ { - opcode: 'checkProfanity', + opcode: "checkProfanity", blockType: Scratch.BlockType.REPORTER, - text: 'replace bad words in [TEXT] with [REPLACEMENT]', + text: "replace bad words in [TEXT] with [REPLACEMENT]", arguments: { REPLACEMENT: { type: Scratch.ArgumentType.STRING, - defaultValue: '***', + defaultValue: "***", }, TEXT: { type: Scratch.ArgumentType.STRING, - defaultValue: 'Hello!', - } - } + defaultValue: "Hello!", + }, + }, }, - ] + ], }; } - checkProfanity({TEXT, REPLACEMENT}) { + checkProfanity({ TEXT, REPLACEMENT }) { // Use a function as the second argument so that replacing with "$&" does not allow // bypass. return String(TEXT).replace(regex, () => REPLACEMENT); diff --git a/extensions/TheShovel/profanityAPI.js b/extensions/TheShovel/profanityAPI.js index ce2cef49fc..773158a22b 100644 --- a/extensions/TheShovel/profanityAPI.js +++ b/extensions/TheShovel/profanityAPI.js @@ -1,33 +1,35 @@ -(function(Scratch) { - 'use strict'; +(function (Scratch) { + "use strict"; class profanityAPI { - getInfo () { + getInfo() { return { - id: 'profanityAPI', - name: 'profanityAPI', - color1: '#cf6a3c', - color2: '#cf6a3c', - color3: '#cf6a3c', + id: "profanityAPI", + name: "profanityAPI", + color1: "#cf6a3c", + color2: "#cf6a3c", + color3: "#cf6a3c", blocks: [ { - opcode: 'checkProfanity', + opcode: "checkProfanity", blockType: Scratch.BlockType.REPORTER, text: "Remove profanity from [TEXT]", arguments: { TEXT: { type: Scratch.ArgumentType.STRING, - defaultValue: 'Hello, I love pizza!', - } - } + defaultValue: "Hello, I love pizza!", + }, + }, }, - ] + ], }; } - checkProfanity({TEXT}) { - return Scratch.fetch("https://www.purgomalum.com/service/plain?text=" + TEXT) - .then(r => r.text()) - .catch(() => ''); + checkProfanity({ TEXT }) { + return Scratch.fetch( + "https://www.purgomalum.com/service/plain?text=" + TEXT + ) + .then((r) => r.text()) + .catch(() => ""); } } diff --git a/extensions/WP-Studio01/text2speech.js b/extensions/WP-Studio01/text2speech.js index 74b72c3c86..a51c7cb015 100644 --- a/extensions/WP-Studio01/text2speech.js +++ b/extensions/WP-Studio01/text2speech.js @@ -1,40 +1,40 @@ -(function(Scratch) { - 'use strict'; - - class Tools { - getInfo () { - return { - id: 'wpstudio01tts', - name: 'System Text To Speech', - blocks: [ - { - opcode: 'speak', - blockType: Scratch.BlockType.COMMAND, - text: 'speak [TEXT]', - arguments: { - TEXT: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'Hello' - } - } - } - ] - }; - } - - speak (args) { - return new Promise((resolve, reject) => { - const utterance = new SpeechSynthesisUtterance(args.TEXT); - utterance.onend = () => { - resolve(); - }; - utterance.onerror = () => { - reject(new Error('Utterance error')); - }; - speechSynthesis.speak(utterance); - }); - } - } - - Scratch.extensions.register(new Tools()); -})(Scratch); \ No newline at end of file +(function (Scratch) { + "use strict"; + + class Tools { + getInfo() { + return { + id: "wpstudio01tts", + name: "System Text To Speech", + blocks: [ + { + opcode: "speak", + blockType: Scratch.BlockType.COMMAND, + text: "speak [TEXT]", + arguments: { + TEXT: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Hello", + }, + }, + }, + ], + }; + } + + speak(args) { + return new Promise((resolve, reject) => { + const utterance = new SpeechSynthesisUtterance(args.TEXT); + utterance.onend = () => { + resolve(); + }; + utterance.onerror = () => { + reject(new Error("Utterance error")); + }; + speechSynthesis.speak(utterance); + }); + } + } + + Scratch.extensions.register(new Tools()); +})(Scratch); diff --git a/extensions/Xeltalliv/clippingblending.js b/extensions/Xeltalliv/clippingblending.js index 9eb79bb8e5..3b79908c01 100644 --- a/extensions/Xeltalliv/clippingblending.js +++ b/extensions/Xeltalliv/clippingblending.js @@ -1,13 +1,19 @@ -(function(Scratch) { - 'use strict'; +// Name: Clipping & Blending +// ID: xeltallivclipblend +// Description: Clipping outside of a specified rectangular area and additive color blending. +// By: Vadik1 + +(function (Scratch) { + "use strict"; if (!Scratch.extensions.unsandboxed) { throw new Error("Effects extension must be run unsandboxed"); } - // Simplified remake of an icon by True-Fantom - const icon = 'data:image/svg+xml,' + encodeURIComponent(` + const icon = + "data:image/svg+xml," + + encodeURIComponent(` @@ -18,7 +24,6 @@ `); - let toCorrectThing = null; let active = false; let flipY = false; @@ -34,7 +39,6 @@ let scratchUnitHeight = 360; let penDirty = false; - renderer._drawThese = function (drawables, drawMode, projection, opts) { active = true; [scratchUnitWidth, scratchUnitHeight] = renderer.getNativeSize(); @@ -70,40 +74,40 @@ }; // Getting Drawable - const dr = renderer.createDrawable('background'); + const dr = renderer.createDrawable("background"); const DrawableProto = renderer._allDrawables[dr].__proto__; - renderer.destroyDrawable(dr, 'background'); + renderer.destroyDrawable(dr, "background"); function setupModes(clipbox, blendMode, flipY) { if (clipbox) { gl.enable(gl.SCISSOR_TEST); - let x = (clipbox.x_min / scratchUnitWidth + 0.5) * width | 0; - let y = (clipbox.y_min / scratchUnitHeight + 0.5) * height | 0; - let x2 = (clipbox.x_max / scratchUnitWidth + 0.5) * width | 0; - let y2 = (clipbox.y_max / scratchUnitHeight + 0.5) * height | 0; + let x = ((clipbox.x_min / scratchUnitWidth + 0.5) * width) | 0; + let y = ((clipbox.y_min / scratchUnitHeight + 0.5) * height) | 0; + let x2 = ((clipbox.x_max / scratchUnitWidth + 0.5) * width) | 0; + let y2 = ((clipbox.y_max / scratchUnitHeight + 0.5) * height) | 0; let w = x2 - x; let h = y2 - y; if (flipY) { - y = (-(clipbox.y_max) / scratchUnitHeight + 0.5) * height | 0; + y = ((-clipbox.y_max / scratchUnitHeight + 0.5) * height) | 0; } gl.scissor(x, y, w, h); } else { gl.disable(gl.SCISSOR_TEST); } switch (blendMode) { - case 'additive': + case "additive": gl.blendEquation(gl.FUNC_ADD); gl.blendFunc(gl.ONE, gl.ONE); break; - case 'subtract': + case "subtract": gl.blendEquation(gl.FUNC_REVERSE_SUBTRACT); gl.blendFunc(gl.ONE, gl.ONE); break; - case 'multiply': + case "multiply": gl.blendEquation(gl.FUNC_ADD); gl.blendFunc(gl.DST_COLOR, gl.ONE_MINUS_SRC_ALPHA); break; - case 'invert': + case "invert": gl.blendEquation(gl.FUNC_ADD); gl.blendFunc(gl.ONE_MINUS_DST_COLOR, gl.ONE_MINUS_SRC_COLOR); break; @@ -128,7 +132,6 @@ this.blendMode = blendMode; }; - // Expanding renderer renderer.updateDrawableClipBox = function (drawableID, clipbox) { const drawable = this._allDrawables[drawableID]; @@ -141,16 +144,23 @@ drawable.updateBlendMode(blendMode); }; - // Reset on stop & clones inherit effects const regTargetStuff = function (args) { if (args.editingTarget) { - vm.removeListener('targetsUpdate', regTargetStuff); + vm.removeListener("targetsUpdate", regTargetStuff); const proto = vm.runtime.targets[0].__proto__; const osa = proto.onStopAll; proto.onStopAll = function () { - this.renderer.updateDrawableClipBox.call(renderer, this.drawableID, null); - this.renderer.updateDrawableBlendMode.call(renderer, this.drawableID, null); + this.renderer.updateDrawableClipBox.call( + renderer, + this.drawableID, + null + ); + this.renderer.updateDrawableBlendMode.call( + renderer, + this.drawableID, + null + ); osa.call(this); }; const mc = proto.makeClone; @@ -159,15 +169,22 @@ if (this.clipbox || this.blendMode) { newTarget.clipbox = Object.assign({}, this.clipbox); newTarget.blendMode = this.blendMode; - renderer.updateDrawableClipBox.call(renderer, newTarget.drawableID, this.clipbox); - renderer.updateDrawableBlendMode.call(renderer, newTarget.drawableID, this.blendMode); + renderer.updateDrawableClipBox.call( + renderer, + newTarget.drawableID, + this.clipbox + ); + renderer.updateDrawableBlendMode.call( + renderer, + newTarget.drawableID, + this.blendMode + ); } return newTarget; }; } }; - vm.on('targetsUpdate', regTargetStuff); - + vm.on("targetsUpdate", regTargetStuff); // Pen lines support let emptyObject = {}; @@ -184,14 +201,20 @@ lastClipbox = null; lastBlendMode = "default"; }; - const willDrawPenWithTarget = function(target) { + const willDrawPenWithTarget = function (target) { if (!penDirty && target == lastTarget) return; penDirty = false; const clipbox = target.clipbox; - if ((!lastClipbox ^ !clipbox) || - (lastBlendMode != target.blendMode) || - (clipbox && (clipbox.x_min != lastClipbox.x_min || clipbox.y_min != lastClipbox.y_min || clipbox.x_max != lastClipbox.x_max || clipbox.y_max != lastClipbox.y_max))) { + if ( + !lastClipbox ^ !clipbox || + lastBlendMode != target.blendMode || + (clipbox && + (clipbox.x_min != lastClipbox.x_min || + clipbox.y_min != lastClipbox.y_min || + clipbox.x_max != lastClipbox.x_max || + clipbox.y_max != lastClipbox.y_max)) + ) { if (skin.a_lineColorIndex) { skin._flushLines(); } @@ -201,7 +224,7 @@ x_min: clipbox.x_min, y_min: clipbox.y_min, x_max: clipbox.x_max, - y_max: clipbox.y_max + y_max: clipbox.y_max, }; } else { lastClipbox = null; @@ -213,7 +236,7 @@ // When drawing a line it is important to know the target. // This saves target. const onTargetMoved = ext_pen._onTargetMoved; - ext_pen._onTargetMoved = function(target, oldX, oldY, isForce) { + ext_pen._onTargetMoved = function (target, oldX, oldY, isForce) { willDrawPenWithTarget(target); onTargetMoved.call(this, target, oldX, oldY, isForce); }; @@ -226,13 +249,13 @@ // When drawing a dot it is important to know the target. // This saves target. const penDown = ext_pen._penDown; - ext_pen._penDown = function(target) { + ext_pen._penDown = function (target) { willDrawPenWithTarget(target); penDown.call(this, target); }; // Set up correct clipping/blending before drawing const flushLines = skin.__proto__._flushLines; - skin.__proto__._flushLines = function() { + skin.__proto__._flushLines = function () { setupModes(lastClipbox, lastBlendMode, true); flushLines.call(this); }; @@ -244,7 +267,7 @@ // If pen skin does not exist, wait until it will, // trigger code once, and return everything as it was const createPenSkin = renderer.createPenSkin; - renderer.createPenSkin = function() { + renderer.createPenSkin = function () { let skinId = createPenSkin.call(this); patchPen(renderer._allSkins[skinId]); renderer.createPenSkin = createPenSkin; @@ -255,128 +278,132 @@ class Extension { getInfo() { return { - id: 'xeltallivclipblend', - name: 'Clipping & Blending', - color1: '#9966FF', - color2: '#855CD6', - color3: '#774DCB', + id: "xeltallivclipblend", + name: "Clipping & Blending", + color1: "#9966FF", + color2: "#855CD6", + color3: "#774DCB", menuIconURI: icon, blocks: [ { - opcode: 'setClipbox', + opcode: "setClipbox", blockType: Scratch.BlockType.COMMAND, - text: 'set clipping box x1:[X1] y1:[Y1] x2:[X2] y2:[Y2]', + text: "set clipping box x1:[X1] y1:[Y1] x2:[X2] y2:[Y2]", arguments: { X1: { type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' + defaultValue: "0", }, Y1: { type: Scratch.ArgumentType.NUMBER, - defaultValue: '0' + defaultValue: "0", }, X2: { type: Scratch.ArgumentType.NUMBER, - defaultValue: '100' + defaultValue: "100", }, Y2: { type: Scratch.ArgumentType.NUMBER, - defaultValue: '100' - } + defaultValue: "100", + }, }, - filter: [Scratch.TargetType.SPRITE] + filter: [Scratch.TargetType.SPRITE], }, { - opcode: 'clearClipbox', + opcode: "clearClipbox", blockType: Scratch.BlockType.COMMAND, - text: 'clear clipping box', - filter: [Scratch.TargetType.SPRITE] + text: "clear clipping box", + filter: [Scratch.TargetType.SPRITE], }, { - opcode: 'getClipbox', + opcode: "getClipbox", blockType: Scratch.BlockType.REPORTER, - text: 'clipping box [PROP]', + text: "clipping box [PROP]", arguments: { PROP: { type: Scratch.ArgumentType.STRING, - defaultValue: 'width', - menu: 'props' - } + defaultValue: "width", + menu: "props", + }, }, - filter: [Scratch.TargetType.SPRITE] + filter: [Scratch.TargetType.SPRITE], }, - '---', + "---", { - opcode: 'setBlend', + opcode: "setBlend", blockType: Scratch.BlockType.COMMAND, - text: 'use [BLENDMODE] blending ', + text: "use [BLENDMODE] blending ", arguments: { BLENDMODE: { type: Scratch.ArgumentType.STRING, - defaultValue: 'default', - menu: 'blends' - } + defaultValue: "default", + menu: "blends", + }, }, - filter: [Scratch.TargetType.SPRITE] + filter: [Scratch.TargetType.SPRITE], }, { - opcode: 'getBlend', + opcode: "getBlend", blockType: Scratch.BlockType.REPORTER, - text: 'blending', + text: "blending", filter: [Scratch.TargetType.SPRITE], - disableMonitor: true + disableMonitor: true, }, - '---', + "---", { - opcode: 'setAdditiveBlend', + opcode: "setAdditiveBlend", blockType: Scratch.BlockType.COMMAND, - text: 'turn additive blending [STATE]', + text: "turn additive blending [STATE]", arguments: { STATE: { type: Scratch.ArgumentType.STRING, - defaultValue: 'on', - menu: 'states' - } + defaultValue: "on", + menu: "states", + }, }, filter: [Scratch.TargetType.SPRITE], - hideFromPalette: true + hideFromPalette: true, }, { - opcode: 'getAdditiveBlend', + opcode: "getAdditiveBlend", blockType: Scratch.BlockType.BOOLEAN, - text: 'is additive blending on?', + text: "is additive blending on?", filter: [Scratch.TargetType.SPRITE], - hideFromPalette: true + hideFromPalette: true, }, ], menus: { states: { acceptReporters: true, - items: ['on', 'off'] + items: ["on", "off"], }, blends: { acceptReporters: true, - items: ['default', 'additive', 'subtract', 'multiply', 'invert'] + items: ["default", "additive", "subtract", "multiply", "invert"], }, props: { acceptReporters: true, - items: ['width', 'height', 'min x', 'min y', 'max x', 'max y'] + items: ["width", "height", "min x", "min y", "max x", "max y"], }, - } + }, }; } - setClipbox ({X1, Y1, X2, Y2}, {target}) { + setClipbox({ X1, Y1, X2, Y2 }, { target }) { if (target.isStage) return; const newClipbox = { x_min: Math.min(X1, X2), y_min: Math.min(Y1, Y2), x_max: Math.max(X1, X2), - y_max: Math.max(Y1, Y2) + y_max: Math.max(Y1, Y2), }; penDirty = true; target.clipbox = newClipbox; - renderer.updateDrawableClipBox.call(renderer, target.drawableID, newClipbox); + renderer.updateDrawableClipBox.call( + renderer, + target.drawableID, + newClipbox + ); if (target.visible) { renderer.dirty = true; target.emitVisualChange(); @@ -385,7 +412,7 @@ } } - clearClipbox (args, {target}) { + clearClipbox(args, { target }) { if (target.isStage) return; target.clipbox = null; penDirty = true; @@ -398,28 +425,35 @@ } } - getClipbox ({PROP}, {target}) { + getClipbox({ PROP }, { target }) { const clipbox = target.clipbox; - if (!clipbox) return ''; + if (!clipbox) return ""; switch (PROP) { - case 'width': return clipbox.x_max - clipbox.x_min; - case 'height': return clipbox.y_max - clipbox.y_min; - case 'min x': return clipbox.x_min; - case 'min y': return clipbox.y_min; - case 'max x': return clipbox.x_max; - case 'max y': return clipbox.y_max; - default: return ''; + case "width": + return clipbox.x_max - clipbox.x_min; + case "height": + return clipbox.y_max - clipbox.y_min; + case "min x": + return clipbox.x_min; + case "min y": + return clipbox.y_min; + case "max x": + return clipbox.x_max; + case "max y": + return clipbox.y_max; + default: + return ""; } } - setBlend ({BLENDMODE}, {target}) { + setBlend({ BLENDMODE }, { target }) { let newValue = null; switch (BLENDMODE) { - case 'default': - case 'additive': - case 'subtract': - case 'multiply': - case 'invert': + case "default": + case "additive": + case "subtract": + case "multiply": + case "invert": newValue = BLENDMODE; break; default: @@ -428,7 +462,11 @@ if (target.isStage) return; penDirty = true; target.blendMode = newValue; - renderer.updateDrawableBlendMode.call(renderer, target.drawableID, newValue); + renderer.updateDrawableBlendMode.call( + renderer, + target.drawableID, + newValue + ); if (target.visible) { renderer.dirty = true; target.emitVisualChange(); @@ -437,19 +475,19 @@ } } - getBlend (args, {target}) { - return target.blendMode ?? 'default'; + getBlend(args, { target }) { + return target.blendMode ?? "default"; } - setAdditiveBlend ({STATE}, util) { - if (STATE === 'on') this.setBlend ({BLENDMODE: 'additive'}, util); - if (STATE === 'off') this.setBlend ({BLENDMODE: 'default'}, util); + setAdditiveBlend({ STATE }, util) { + if (STATE === "on") this.setBlend({ BLENDMODE: "additive" }, util); + if (STATE === "off") this.setBlend({ BLENDMODE: "default" }, util); } - getAdditiveBlend (args, {target}) { - return target.blendMode === 'additive'; + getAdditiveBlend(args, { target }) { + return target.blendMode === "additive"; } } Scratch.extensions.register(new Extension()); -})(Scratch); \ No newline at end of file +})(Scratch); diff --git a/extensions/XeroName/Deltatime.js b/extensions/XeroName/Deltatime.js new file mode 100644 index 0000000000..0d6f3a7de0 --- /dev/null +++ b/extensions/XeroName/Deltatime.js @@ -0,0 +1,61 @@ +// Name: Deltatime +// ID: dtbyxeroname +// Description: Precise delta timing blocks. +// By: XeroName + +(function (Scratch) { + "use strict"; + + const icon = + ""; + + if (!Scratch.extensions.unsandboxed) { + throw new Error("DeltaTime must be run unsandboxed"); + } + + const vm = Scratch.vm; + + let deltaTime = 0; + let previousTime = 0; + + vm.runtime.on("BEFORE_EXECUTE", () => { + const now = performance.now(); + deltaTime = previousTime === 0 ? 0 : (now - previousTime) / 1000; + previousTime = now; + }); + + class Dt { + getInfo() { + return { + id: "dtbyxeroname", + name: "Deltatime", + color1: "#333333", + color2: "#444444", + color3: "#ffffff", + menuIconURI: icon, + blocks: [ + { + opcode: "dt", + blockType: Scratch.BlockType.REPORTER, + text: "ΔT", + }, + { + opcode: "fps", + blockType: Scratch.BlockType.REPORTER, + text: "fps", + }, + ], + }; + } + + dt() { + return deltaTime; + } + + fps() { + return +(1 / deltaTime).toFixed(2); + } + } + + Scratch.extensions.register(new Dt()); +})(Scratch); diff --git a/extensions/ZXMushroom63/searchApi.js b/extensions/ZXMushroom63/searchApi.js index 060096f0a5..6d25fe08b5 100644 --- a/extensions/ZXMushroom63/searchApi.js +++ b/extensions/ZXMushroom63/searchApi.js @@ -1,210 +1,203 @@ -(function (Scratch) { - "use strict"; - if (!Scratch.extensions.unsandboxed) { - throw new Error("SearchParams must be run unsandboxed."); - } - - // Disable in desktop app editor for security reasons. - // @ts-ignore - const disabled = typeof TWD !== 'undefined'; - - class SearchApi { - getInfo() { - return { - id: "zxmushroom63searchparams", - name: "Search Params", - color1: "#b4b4b4", - color2: "#9c9c9c", - color3: "#646464", - blocks: [ - { - blockType: Scratch.BlockType.LABEL, - text: 'Disabled in desktop editor for security', - hideFromPalette: !disabled - }, - { - opcode: "searchparam", - blockType: Scratch.BlockType.REPORTER, - text: "value of search parameter [ID]", - arguments: { - ID: { - type: Scratch.ArgumentType.STRING, - defaultValue: "x", - }, - }, - }, - { - opcode: "occurencesofsearchparam", - blockType: Scratch.BlockType.REPORTER, - text: "occurences of search parameter [ID]", - arguments: { - ID: { - type: Scratch.ArgumentType.STRING, - defaultValue: "x", - }, - }, - }, - { - opcode: "indexedsearchparam", - blockType: Scratch.BlockType.REPORTER, - text: "index [I] of search parameters [ID]", - arguments: { - ID: { - type: Scratch.ArgumentType.STRING, - defaultValue: "x", - }, - I: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: 1, - }, - }, - }, - { - opcode: "setsearchparam", - blockType: Scratch.BlockType.COMMAND, - text: "set search parameter [ID] to [VAL]", - arguments: { - ID: { - type: Scratch.ArgumentType.STRING, - defaultValue: "x", - }, - VAL: { - type: Scratch.ArgumentType.STRING, - defaultValue: "15", - }, - }, - }, - { - opcode: "deletesearchparam", - blockType: Scratch.BlockType.COMMAND, - text: "delete search parameter [ID]", - arguments: { - ID: { - type: Scratch.ArgumentType.STRING, - defaultValue: "x", - }, - }, - }, - { - opcode: "appendsearchparam", - blockType: Scratch.BlockType.COMMAND, - text: "append search parameter [ID] with value [VAL]", - arguments: { - ID: { - type: Scratch.ArgumentType.STRING, - defaultValue: "x", - }, - VAL: { - type: Scratch.ArgumentType.STRING, - defaultValue: "15", - }, - }, - }, - { - opcode: "hassearchparam", - blockType: Scratch.BlockType.BOOLEAN, - text: "has search parameter [ID]", - arguments: { - ID: { - type: Scratch.ArgumentType.STRING, - defaultValue: "x", - }, - }, - }, - { - opcode: "searchparamslength", - blockType: Scratch.BlockType.REPORTER, - text: "length of search parameters", - }, - { - opcode: "searchparamatindex", - blockType: Scratch.BlockType.REPORTER, - text: "search parameter [PARAM] at index [I]", - arguments: { - PARAM: { - type: Scratch.ArgumentType.STRING, - menu: "PARAM", - }, - I: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: 1, - }, - }, - }, - ], - menus: { - PARAM: { - acceptReporters: true, - items: ["value", "name"], - }, - }, - }; - } - - searchparam({ ID }) { - if (disabled) return ""; - return new URLSearchParams(location.search).get(ID.toString()) || ""; - } - - occurencesofsearchparam({ ID }) { - if (disabled) return 0; - return new URLSearchParams(location.search).getAll(ID.toString()).length || 0; - } - - indexedsearchparam({ ID, I }) { - if (disabled) return ""; - return new URLSearchParams(location.search).getAll(ID.toString())[parseInt(I) - 1] || ""; - } - - setsearchparam({ ID, VAL }) { - if (disabled) return; - var s = new URLSearchParams(location.search); - s.set(ID.toString(), VAL.toString()); - history.replaceState("", "", "?" + s.toString()); - } - - searchparamslength() { - if (disabled) return 0; - var s = new URLSearchParams(location.search); - // @ts-ignore - return typeof s.size !== "object" ? s.size : 0; - } - - deletesearchparam({ ID }) { - if (disabled) return; - var s = new URLSearchParams(location.search); - s.delete(ID.toString()); - history.replaceState("", "", "?" + s.toString()); - } - - appendsearchparam({ ID, VAL }) { - if (disabled) return; - var s = new URLSearchParams(location.search); - s.append(ID.toString(), VAL.toString()); - history.replaceState("", "", "?" + s.toString()); - } - - hassearchparam({ ID }) { - if (disabled) return false; - var s = new URLSearchParams(location.search); - return s.has(ID.toString()) || false; - } - - searchparamatindex({ PARAM, I }) { - if (disabled) return ""; - var index = parseInt(I) - 1 || 0; - index = Math.max(0, index); - var s = new URLSearchParams(location.search); - var values = PARAM.toString() === "value" ? s.values() : s.keys(); - var i = 0; - for (const value of values) { - if (i === index) { - return value; - } - i++; - } - return ""; - } - } - Scratch.extensions.register(new SearchApi()); -})(Scratch); +// Name: Search Params +// ID: zxmushroom63searchparams +// Description: Interact with URL search parameters: the part of the URL after a question mark. +// By: ZXMushroom63 + +(function (Scratch) { + "use strict"; + if (!Scratch.extensions.unsandboxed) { + throw new Error("SearchParams must be run unsandboxed."); + } + + class SearchApi { + getInfo() { + return { + id: "zxmushroom63searchparams", + name: "Search Params", + color1: "#b4b4b4", + color2: "#9c9c9c", + color3: "#646464", + blocks: [ + { + opcode: "searchparam", + blockType: Scratch.BlockType.REPORTER, + text: "value of search parameter [ID]", + arguments: { + ID: { + type: Scratch.ArgumentType.STRING, + defaultValue: "x", + }, + }, + }, + { + opcode: "occurencesofsearchparam", + blockType: Scratch.BlockType.REPORTER, + text: "occurences of search parameter [ID]", + arguments: { + ID: { + type: Scratch.ArgumentType.STRING, + defaultValue: "x", + }, + }, + }, + { + opcode: "indexedsearchparam", + blockType: Scratch.BlockType.REPORTER, + text: "index [I] of search parameters [ID]", + arguments: { + ID: { + type: Scratch.ArgumentType.STRING, + defaultValue: "x", + }, + I: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: 1, + }, + }, + }, + { + opcode: "setsearchparam", + blockType: Scratch.BlockType.COMMAND, + text: "set search parameter [ID] to [VAL]", + arguments: { + ID: { + type: Scratch.ArgumentType.STRING, + defaultValue: "x", + }, + VAL: { + type: Scratch.ArgumentType.STRING, + defaultValue: "15", + }, + }, + }, + { + opcode: "deletesearchparam", + blockType: Scratch.BlockType.COMMAND, + text: "delete search parameter [ID]", + arguments: { + ID: { + type: Scratch.ArgumentType.STRING, + defaultValue: "x", + }, + }, + }, + { + opcode: "appendsearchparam", + blockType: Scratch.BlockType.COMMAND, + text: "append search parameter [ID] with value [VAL]", + arguments: { + ID: { + type: Scratch.ArgumentType.STRING, + defaultValue: "x", + }, + VAL: { + type: Scratch.ArgumentType.STRING, + defaultValue: "15", + }, + }, + }, + { + opcode: "hassearchparam", + blockType: Scratch.BlockType.BOOLEAN, + text: "has search parameter [ID]", + arguments: { + ID: { + type: Scratch.ArgumentType.STRING, + defaultValue: "x", + }, + }, + }, + { + opcode: "searchparamslength", + blockType: Scratch.BlockType.REPORTER, + text: "length of search parameters", + }, + { + opcode: "searchparamatindex", + blockType: Scratch.BlockType.REPORTER, + text: "search parameter [PARAM] at index [I]", + arguments: { + PARAM: { + type: Scratch.ArgumentType.STRING, + menu: "PARAM", + }, + I: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: 1, + }, + }, + }, + ], + menus: { + PARAM: { + acceptReporters: true, + items: ["value", "name"], + }, + }, + }; + } + + searchparam({ ID }) { + return new URLSearchParams(location.search).get(ID.toString()) || ""; + } + + occurencesofsearchparam({ ID }) { + return ( + new URLSearchParams(location.search).getAll(ID.toString()).length || 0 + ); + } + + indexedsearchparam({ ID, I }) { + return ( + new URLSearchParams(location.search).getAll(ID.toString())[ + parseInt(I) - 1 + ] || "" + ); + } + + setsearchparam({ ID, VAL }) { + var s = new URLSearchParams(location.search); + s.set(ID.toString(), VAL.toString()); + history.replaceState("", "", "?" + s.toString()); + } + + searchparamslength() { + var s = new URLSearchParams(location.search); + // @ts-ignore + return typeof s.size !== "object" ? s.size : 0; + } + + deletesearchparam({ ID }) { + var s = new URLSearchParams(location.search); + s.delete(ID.toString()); + history.replaceState("", "", "?" + s.toString()); + } + + appendsearchparam({ ID, VAL }) { + var s = new URLSearchParams(location.search); + s.append(ID.toString(), VAL.toString()); + history.replaceState("", "", "?" + s.toString()); + } + + hassearchparam({ ID }) { + var s = new URLSearchParams(location.search); + return s.has(ID.toString()) || false; + } + + searchparamatindex({ PARAM, I }) { + var index = parseInt(I) - 1 || 0; + index = Math.max(0, index); + var s = new URLSearchParams(location.search); + var values = PARAM.toString() === "value" ? s.values() : s.keys(); + var i = 0; + for (const value of values) { + if (i === index) { + return value; + } + i++; + } + return ""; + } + } + Scratch.extensions.register(new SearchApi()); +})(Scratch); diff --git a/extensions/ar.js b/extensions/ar.js index ed507084e9..0d9ad60a71 100644 --- a/extensions/ar.js +++ b/extensions/ar.js @@ -1,805 +1,845 @@ -(function(Scratch) { - "use strict"; - - /* globals XRWebGLLayer, XRRigidTransform, XRWebGLLayer */ - - if (!Scratch.extensions.unsandboxed) { - throw new Error("AR extension must be run unsandboxed"); +// Name: Augmented Reality +// ID: AR +// Description: Shows image from camera and performs motion tracking, allowing 3D projects to correctly overlay virtual objects on real world. +// By: Vadik1 + +(function (Scratch) { + "use strict"; + + /* globals XRWebGLLayer, XRRigidTransform, XRWebGLLayer */ + + if (!Scratch.extensions.unsandboxed) { + throw new Error("AR extension must be run unsandboxed"); + } + + const ArgumentType = Scratch.ArgumentType; + const BlockType = Scratch.BlockType; + + const vm = Scratch.vm; + const renderer = vm.renderer; + const runtime = vm.runtime; + const frameLoop = runtime.frameLoop; + const mouse = runtime.ioDevices.mouse; + const video = runtime.ioDevices.video; + + let arResolution = 1; + let isPackaged = false; + + let arFail = "uninitialized"; + let xrSession = null; + let xrState = false; + let xrRefSpace; + let xrViewSpace; + let xrProjectionMatrix; + let xrTransform; + let xrCombinedMatrix; + let xrHitTestSource; + let hitPosition; + let hitPositionAvailible; + let oldWidth = 0; + let oldHeight = 0; + let xrNeedsResize = false; + let poseAvailible = false; + let enterARDone = []; + + let stageWrapper; + let stageWrapperParent; + let scLayers; + let scControlsBar; + const div = document.createElement("div"); + document.body.append(div); + const canvas = Scratch.vm.renderer.canvas; + const gl = Scratch.vm.renderer.gl; + const enableVideoOriginal = video.enableVideo; + + // Checking whether AR is supported. + // If not, extension should still load, to let people + // develop AR projects on non-AR-capable devices and then + // test them on AR-capable mobile devices + if (!window.isSecureContext) { + console.error( + (arFail = + "Window is not secure context. WebXR only works in secure context") + ); + } else if (!navigator.xr) { + console.error( + (arFail = "navigator.xr is not defined in the browser you are using") + ); + } else { + gl.makeXRCompatible().catch((error) => { + console.error((arFail = "gl.makeXRCompatible rejected with: " + error)); + }); + navigator.xr.isSessionSupported("immersive-ar").then((supported) => { + if (!supported) { + console.error( + (arFail = + "WebXR exists in the browser you are using, but 'immersive-ar' session type is not supported") + ); + } else { + arFail = null; + } + }); + } + + const onSuccess = function (session) { + xrSession = session; + xrRefSpace = null; + xrViewSpace = null; + xrHitTestSource = null; + hitPosition = null; + hitPositionAvailible = false; + poseAvailible = false; + + session.updateRenderState({ + baseLayer: new XRWebGLLayer(session, gl, { + framebufferScaleFactor: arResolution, + }), + }); + session.addEventListener("end", () => { + xrSession = null; + updateState(); + }); + session.requestReferenceSpace("local").then((refSpace) => { + xrRefSpace = refSpace; + }); + session + .requestReferenceSpace("viewer") + .then((viewSpace) => { + xrViewSpace = viewSpace; + return session.requestHitTestSource({ space: viewSpace }); + }) + .then((hts) => { + xrHitTestSource = hts; + }); + updateState(); + + // [enter AR] blocks should continue after success + enterARDone.forEach((fn) => fn()); + enterARDone = []; + }; + const onError = function (error) { + // This shouldn't set arFail, because arFail is for cases when it permanently failed. + // This might fail once, but work on the next attempt. + console.error( + "Even though 'immersive-ar' is supported in your browser, requesting it failed" + ); + console.error(error); + + // [enter AR] blocks should continue after failure + enterARDone.forEach((fn) => fn()); + enterARDone = []; + }; + const onErrorTryTap = function (error) { + canvas.removeEventListener("pointerup", enterAR); + canvas.addEventListener("pointerup", enterAR, { once: true }); + }; + + const updateState = function () { + const state = !!xrSession; + if (state === xrState) return; + + xrState = state; + renderer.draw = state ? drawXR : drawOrig; + renderer.xr = xrSession; + frameLoop.inXR = state; + if (frameLoop.running) { + frameLoop.stop(); + frameLoop.start(); } - - - const ArgumentType = Scratch.ArgumentType; - const BlockType = Scratch.BlockType; - - const vm = Scratch.vm; - const renderer = vm.renderer; - const runtime = vm.runtime; - const frameLoop = runtime.frameLoop; - const mouse = runtime.ioDevices.mouse; - const video = runtime.ioDevices.video; - - let arResolution = 1; - let isPackaged = false; - - let arFail = "uninitialized"; - let xrSession = null; - let xrState = false; - let xrRefSpace; - let xrViewSpace; - let xrProjectionMatrix; - let xrTransform; - let xrCombinedMatrix; - let xrHitTestSource; - let hitPosition; - let hitPositionAvailible; - let oldWidth = 0; - let oldHeight = 0; - let xrNeedsResize = false; - let poseAvailible = false; - let enterARDone = []; - - let stageWrapper; - let stageWrapperParent; - let scLayers; - let scControlsBar; - const div = document.createElement("div"); - document.body.append(div); - const canvas = Scratch.vm.renderer.canvas; - const gl = Scratch.vm.renderer.gl; - const enableVideoOriginal = video.enableVideo; - - // Checking whether AR is supported. - // If not, extension should still load, to let people - // develop AR projects on non-AR-capable devices and then - // test them on AR-capable mobile devices - if (!window.isSecureContext) { - console.error(arFail = "Window is not secure context. WebXR only works in secure context"); - } else if (!navigator.xr) { - console.error(arFail = "navigator.xr is not defined in the browser you are using"); + canvas.removeEventListener("pointerup", enterAR); + if (state) { + video.disableVideo(); // Hiding it, since it freezes anyways + video.enableVideo = () => null; + + // css "transform" doesn't work directly on domOverlay element, + // but works on it's children. stageWrapper needs to have "transform: scale" + // on it, so that is why it is placed into another div + div.append(stageWrapper); + + xrNeedsResize = true; + oldWidth = runtime.stageWidth; + oldHeight = runtime.stageHeight; } else { - gl.makeXRCompatible().catch( - (error) => { - console.error(arFail = "gl.makeXRCompatible rejected with: " + error); - } - ); - navigator.xr.isSessionSupported("immersive-ar").then( - (supported) => { - if (!supported) { - console.error(arFail = "WebXR exists in the browser you are using, but 'immersive-ar' session type is not supported"); - } else { - arFail = null; - } - } - ); + video.enableVideo = enableVideoOriginal; // After exiting AR, video sensing can be used again + + if (!isPackaged) { + const borderThing = stageWrapper.children[0].children[0].style; + borderThing["border"] = ""; + borderThing["border-radius"] = ""; + } else { + scControlsBar.style["display"] = null; + scLayers.style["transform"] = null; + stageWrapper.style["align-items"] = null; + stageWrapper.style["justify-content"] = null; + runtime.setStageSize(oldWidth, oldHeight); + } + stageWrapper.style = ""; + stageWrapperParent.append(stageWrapper); + + canvas.style.opacity = ""; + gl.bindFramebuffer(gl.FRAMEBUFFER, null); } - - - const onSuccess = function(session) { - xrSession = session; - xrRefSpace = null; - xrViewSpace = null; - xrHitTestSource = null; - hitPosition = null; - hitPositionAvailible = false; - poseAvailible = false; - - session.updateRenderState({ - baseLayer: new XRWebGLLayer(session, gl, { framebufferScaleFactor: arResolution }) - }); - session.addEventListener("end", () => { - xrSession = null; - updateState(); - }); - session.requestReferenceSpace("local").then((refSpace) => { - xrRefSpace = refSpace; - }); - session.requestReferenceSpace("viewer").then((viewSpace) => { - xrViewSpace = viewSpace; - return session.requestHitTestSource({ space: viewSpace }); - }).then((hts) => { - xrHitTestSource = hts; - }); - updateState(); - - // [enter AR] blocks should continue after success - enterARDone.forEach(fn => fn()); - enterARDone = []; + }; + + // Code copied from tw-frame-loop.js because existing code can't be accesed + const _requestAnimationFrame = + typeof requestAnimationFrame === "function" + ? requestAnimationFrame + : (f) => setTimeout(f, 1000 / 60); + const _cancelAnimationFrame = + typeof requestAnimationFrame === "function" + ? cancelAnimationFrame + : clearTimeout; + + const animationFrameWrapper = (callback) => { + let id; + const handle = () => { + id = _requestAnimationFrame(handle); + callback(); }; - const onError = function(error) { - // This shouldn't set arFail, because arFail is for cases when it permanently failed. - // This might fail once, but work on the next attempt. - console.error("Even though 'immersive-ar' is supported in your browser, requesting it failed"); - console.error(error); - - // [enter AR] blocks should continue after failure - enterARDone.forEach(fn => fn()); - enterARDone = []; - }; - const onErrorTryTap = function(error) { - canvas.removeEventListener("pointerup", enterAR); - canvas.addEventListener("pointerup", enterAR, {once: true}); - }; - - const updateState = function() { - const state = !!xrSession; - if (state === xrState) return; - - xrState = state; - renderer.draw = state ? drawXR : drawOrig; - renderer.xr = xrSession; - frameLoop.inXR = state; - if (frameLoop.running) { - frameLoop.stop(); - frameLoop.start(); - } - canvas.removeEventListener("pointerup", enterAR); - if (state) { - video.disableVideo(); // Hiding it, since it freezes anyways - video.enableVideo = () => null; - - // css "transform" doesn't work directly on domOverlay element, - // but works on it's children. stageWrapper needs to have "transform: scale" - // on it, so that is why it is placed into another div - div.append(stageWrapper); - - xrNeedsResize = true; - oldWidth = runtime.stageWidth; - oldHeight = runtime.stageHeight; - } else { - video.enableVideo = enableVideoOriginal; // After exiting AR, video sensing can be used again - - if (!isPackaged) { - const borderThing = stageWrapper.children[0].children[0].style; - borderThing["border"] = ""; - borderThing["border-radius"] = ""; - } else { - scControlsBar.style["display"] = null; - scLayers.style["transform"] = null; - stageWrapper.style["align-items"] = null; - stageWrapper.style["justify-content"] = null; - runtime.setStageSize(oldWidth, oldHeight); - } - stageWrapper.style = ""; - stageWrapperParent.append(stageWrapper); - - canvas.style.opacity = ""; - gl.bindFramebuffer(gl.FRAMEBUFFER, null); - } + const cancel = () => _cancelAnimationFrame(id); + id = _requestAnimationFrame(handle); + return { + cancel, }; - - - - // Code copied from tw-frame-loop.js because existing code can't be accesed - const _requestAnimationFrame = typeof requestAnimationFrame === 'function' ? - requestAnimationFrame : - (f => setTimeout(f, 1000 / 60)); - const _cancelAnimationFrame = typeof requestAnimationFrame === 'function' ? - cancelAnimationFrame : - clearTimeout; - - const animationFrameWrapper = callback => { - let id; - const handle = () => { - id = _requestAnimationFrame(handle); - callback(); - }; - const cancel = () => _cancelAnimationFrame(id); - id = _requestAnimationFrame(handle); - return { - cancel - }; - }; - - - - // Patching frameLoop to use xrSession.requestAnimationFrame when in AR mode - const xrAnimationFrameWrapper = (callback, fps = 30) => { - const xrSessionBackup = xrSession; - let shouldTriggerAgain = false; - let id; - let idIsXR; - let interval; - const handle = (t, frame) => { - // If fps = 0, then run at screen's refresh rate - // and always use xr animation frame. - // In other cases keep using normal animation frame - // and waiting until shouldTriggerAgain gets set - // to true, and only then use xr animation frame - // once and resume waiting. shouldTriggerAgain is - // set to true by the interval located below. - if (fps === 0 || shouldTriggerAgain) { - shouldTriggerAgain = false; - id = xrSession.requestAnimationFrame(handle); - idIsXR = true; - } else { - id = window.requestAnimationFrame(handle); - idIsXR = false; - } - // Normal animation frames are just for waiting and - // shouldn't trigger callback() - if (!frame) return; - - if (xrNeedsResize) { - xrNeedsResize = false; - - // This needs to run before setStageSize - if (isPackaged) { - scControlsBar.style["display"] = "none"; - scLayers.style["transform"] = "translate(0px, 0px)"; - stageWrapper.style["align-items"] = "normal"; - stageWrapper.style["justify-content"] = "flex-start"; - } - - const bl = xrSession.renderState.baseLayer; - const newWidth = Math.round(bl.framebufferWidth / bl.framebufferHeight * oldHeight); - if (runtime.stageWidth !== newWidth) { - runtime.setStageSize(newWidth, oldHeight); - } - - const scale = div.clientHeight / canvas.clientHeight; - stageWrapper.style = "transform-origin: top left; transform: scale(" + scale + "," + scale + ")"; - canvas.style.opacity = "0"; - - if (!isPackaged) { - const borderThing = stageWrapper.children[0].children[0].style; - borderThing["border"] = "none"; - borderThing["border-radius"] = "0"; - borderThing["transform"] = ""; // Removes translateX - } - } - poseAvailible = false; - if (xrRefSpace) { - const pose = frame.getViewerPose(xrRefSpace); - if (pose) { - poseAvailible = true; - xrProjectionMatrix = pose.views[0].projectionMatrix; - xrTransform = pose.views[0].transform; - const inverseTransformMatrix = xrTransform.inverse.matrix; - const a00 = xrProjectionMatrix[0]; - const a01 = xrProjectionMatrix[1]; - const a02 = xrProjectionMatrix[2]; - const a03 = xrProjectionMatrix[3]; - const a10 = xrProjectionMatrix[4]; - const a11 = xrProjectionMatrix[5]; - const a12 = xrProjectionMatrix[6]; - const a13 = xrProjectionMatrix[7]; - const a20 = xrProjectionMatrix[8]; - const a21 = xrProjectionMatrix[9]; - const a22 = xrProjectionMatrix[10]; - const a23 = xrProjectionMatrix[11]; - const a30 = xrProjectionMatrix[12]; - const a31 = xrProjectionMatrix[13]; - const a32 = xrProjectionMatrix[14]; - const a33 = xrProjectionMatrix[15]; - const b00 = inverseTransformMatrix[0]; - const b01 = inverseTransformMatrix[1]; - const b02 = inverseTransformMatrix[2]; - const b03 = inverseTransformMatrix[3]; - const b10 = inverseTransformMatrix[4]; - const b11 = inverseTransformMatrix[5]; - const b12 = inverseTransformMatrix[6]; - const b13 = inverseTransformMatrix[7]; - const b20 = inverseTransformMatrix[8]; - const b21 = inverseTransformMatrix[9]; - const b22 = inverseTransformMatrix[10]; - const b23 = inverseTransformMatrix[11]; - const b30 = inverseTransformMatrix[12]; - const b31 = inverseTransformMatrix[13]; - const b32 = inverseTransformMatrix[14]; - const b33 = inverseTransformMatrix[15]; - xrCombinedMatrix = [ - b00 * a00 + b01 * a10 + b02 * a20 + b03 * a30, - b00 * a01 + b01 * a11 + b02 * a21 + b03 * a31, - b00 * a02 + b01 * a12 + b02 * a22 + b03 * a32, - b00 * a03 + b01 * a13 + b02 * a23 + b03 * a33, - b10 * a00 + b11 * a10 + b12 * a20 + b13 * a30, - b10 * a01 + b11 * a11 + b12 * a21 + b13 * a31, - b10 * a02 + b11 * a12 + b12 * a22 + b13 * a32, - b10 * a03 + b11 * a13 + b12 * a23 + b13 * a33, - b20 * a00 + b21 * a10 + b22 * a20 + b23 * a30, - b20 * a01 + b21 * a11 + b22 * a21 + b23 * a31, - b20 * a02 + b21 * a12 + b22 * a22 + b23 * a32, - b20 * a03 + b21 * a13 + b22 * a23 + b23 * a33, - b30 * a00 + b31 * a10 + b32 * a20 + b33 * a30, - b30 * a01 + b31 * a11 + b32 * a21 + b33 * a31, - b30 * a02 + b31 * a12 + b32 * a22 + b33 * a32, - b30 * a03 + b31 * a13 + b32 * a23 + b33 * a33 - ]; - } - } - hitPositionAvailible = false; - if (xrHitTestSource) { - const hitTestResults = frame.getHitTestResults(xrHitTestSource); - if (hitTestResults.length > 0) { - hitPositionAvailible = true; - hitPosition = hitTestResults[0].getPose(xrRefSpace).transform.position; - } - } - callback(); - }; - const cancel = () => { - if (idIsXR) { - xrSessionBackup.cancelAnimationFrame(id); - } else { - cancelAnimationFrame(id); - } - if (interval) { - clearInterval(interval); - } - }; + }; + + // Patching frameLoop to use xrSession.requestAnimationFrame when in AR mode + const xrAnimationFrameWrapper = (callback, fps = 30) => { + const xrSessionBackup = xrSession; + let shouldTriggerAgain = false; + let id; + let idIsXR; + let interval; + const handle = (t, frame) => { + // If fps = 0, then run at screen's refresh rate + // and always use xr animation frame. + // In other cases keep using normal animation frame + // and waiting until shouldTriggerAgain gets set + // to true, and only then use xr animation frame + // once and resume waiting. shouldTriggerAgain is + // set to true by the interval located below. + if (fps === 0 || shouldTriggerAgain) { + shouldTriggerAgain = false; id = xrSession.requestAnimationFrame(handle); - if (fps > 0) { - interval = setInterval(() => { - shouldTriggerAgain = true; - }, 1000 / fps); + idIsXR = true; + } else { + id = window.requestAnimationFrame(handle); + idIsXR = false; + } + // Normal animation frames are just for waiting and + // shouldn't trigger callback() + if (!frame) return; + + if (xrNeedsResize) { + xrNeedsResize = false; + + // This needs to run before setStageSize + if (isPackaged) { + scControlsBar.style["display"] = "none"; + scLayers.style["transform"] = "translate(0px, 0px)"; + stageWrapper.style["align-items"] = "normal"; + stageWrapper.style["justify-content"] = "flex-start"; } - return { - cancel - }; - }; - const start = function() { - this.running = true; - if (this.inXR) { - if (this.framerate === 0) { - this._stepAnimation = this.xrAnimationFrameWrapper(this.stepCallback, 0); - this.runtime.currentStepTime = 1000 / 60; - } else { - // Interpolation should never be enabled when framerate === 0 as that's just redundant - if (this.interpolation) { - this._interpolationAnimation = animationFrameWrapper(this.interpolationCallback); - } - this._stepAnimation = this.xrAnimationFrameWrapper(this.stepCallback, this.framerate); - this.runtime.currentStepTime = 1000 / this.framerate; - } - } else { - if (this.framerate === 0) { - this._stepAnimation = animationFrameWrapper(this.stepCallback); - this.runtime.currentStepTime = 1000 / 60; - } else { - // Interpolation should never be enabled when framerate === 0 as that's just redundant - if (this.interpolation) { - this._interpolationAnimation = animationFrameWrapper(this.interpolationCallback); - } - this._stepInterval = setInterval(this.stepCallback, 1000 / this.framerate); - this.runtime.currentStepTime = 1000 / this.framerate; - } - } - }; - frameLoop.xrAnimationFrameWrapper = xrAnimationFrameWrapper.bind(frameLoop); - frameLoop.start = start.bind(frameLoop); - frameLoop.inXR = false; - - - // Patching renderer.draw() to draw to xr framebuffer instead of canvas - const drawOrig = renderer.draw.bind(renderer); - const drawXR = (function() { - const bl = this.xr.renderState.baseLayer; // ADDED - if (!bl) return; // Should fix very rare crash during exiting // ADDED - - this._doExitDrawRegion(); - - const gl = this._gl; - - gl.bindFramebuffer(gl.FRAMEBUFFER, bl.framebuffer); // CHANGED - gl.viewport(0, 0, bl.framebufferWidth, bl.framebufferHeight); // CHANGED - gl.clearColor(0, 0, 0, 0); - gl.clear(gl.COLOR_BUFFER_BIT); - - this._drawThese(this._drawList, "default" /*ShaderManager.DRAW_MODE.default*/, this._projection, { - framebufferWidth: bl.framebufferWidth, // CHANGED - framebufferHeight: bl.framebufferHeight // CHANGED - }); - if (this._snapshotCallbacks.length > 0) { - const snapshot = gl.canvas.toDataURL(); - this._snapshotCallbacks.forEach(cb => cb(snapshot)); - this._snapshotCallbacks = []; - } - }).bind(renderer); - renderer.draw = drawOrig; - - - - // Patching _pickTarget incorrect position bug: - // When the canvas is scaled using transform:scale, - // canvas.getBoundingClientRect is affected by it, but - // canvas.clientWidth and canvas.clientHeight are not. - // - // postData receives data.x and data.y, which are mouse position in - // screen units. To be able to rescale it to usable scratch units - // it also receives data.canvasWidth and data.canvasHeight - // which are based on getBoundingClientRect. Based of that it - // calculates this._scratchX and this._scratchY. - // Even when canvas is scaled, those are calculated correctly and - // as a result, blocks (mouse x) and (mouse y) report correct values. - // - // Later, postData calls _pickTarget, while only passing data.x and data.y - // without data.canvasWidth and data.canvasHeight. That method calls - // runtime renderer.pick, which calls clientSpaceToScratchBounds, which - // uses canvas.clientWidth and canvas.clientHeight to rescale mouse - // position from screen units to scratch units. This ignores - // transform:scale and as a result, sprites can't be clicked or dragged. - // - // WARNING: Makes _pickTarget only work correctly when called from postData. - // If something else calls it directly, it may cause problems. - const postDataOriginal = mouse.postData.bind(mouse); - mouse.postData = function(data) { - this._canvasWidth = data.canvasWidth; - this._canvasHeight = data.canvasHeight; - postDataOriginal(data); - }.bind(mouse); - - const _pickTargetOriginal = mouse._pickTarget.bind(mouse); - mouse._pickTarget = function (x, y) { - return _pickTargetOriginal( - x / this._canvasWidth * canvas.clientWidth, - y / this._canvasHeight * canvas.clientHeight + const bl = xrSession.renderState.baseLayer; + const newWidth = Math.round( + (bl.framebufferWidth / bl.framebufferHeight) * oldHeight ); - }.bind(mouse); - - // This is used by . - // It was also broken in a similar way. - mouse.getClientX = function() { - return this._clientX / this._canvasWidth * canvas.clientWidth; - }.bind(mouse); - - mouse.getClientY = function() { - return this._clientY / this._canvasHeight * canvas.clientHeight; - }.bind(mouse); - // END OF WARNING - - - const enterAR = function(event) { - if (!xrSession) { - // Entering and exiting editor recreates this element - stageWrapper = document.querySelector("[class*='stage-wrapper_stage-canvas-wrapper']"); - if (!stageWrapper) { - stageWrapper = document.querySelector("[class='sc-root']"); - scControlsBar = document.querySelector("[class='sc-controls-bar']"); - scLayers = document.querySelector("[class='sc-layers']"); - if (!stageWrapper) { - console.error(arFail = "Failed to get the div element of the stage"); - return; - } - isPackaged = true; - } - stageWrapperParent = stageWrapper.parentElement; - - const noop = () => {}; - navigator.xr.requestSession("immersive-ar", { - requiredFeatures: ["hit-test", "dom-overlay"], - domOverlay: {root: div} - }).then(onSuccess, event ? onError : onErrorTryTap); - // If (event) is defined, it was from click, so something went wrong. - // If (event) is null, it was called directly, and might've been rejected due to lack of user interaction. + if (runtime.stageWidth !== newWidth) { + runtime.setStageSize(newWidth, oldHeight); } - }; - - - class ARExtension { - getInfo() { - return { - id: "AR", - color1: "#d10000", - color2: "#bd0000", - color3: "#af0100", - docsURI: "https://extensions.turbowarp.org/ar.html", - blocks: [ - { - opcode: "enterAR", - blockType: BlockType.COMMAND, - text: "enter AR mode", - arguments: {} - }, - { - opcode: "exitAR", - blockType: BlockType.COMMAND, - text: "exit AR mode", - arguments: {} - }, - { - opcode: "isInAR", - blockType: BlockType.BOOLEAN, - text: "is in AR?", - arguments: {} - }, - { - opcode: "isFeatureAvailible", - blockType: BlockType.BOOLEAN, - text: "is [FEATURE] availible?", - arguments: { - FEATURE: { - type: ArgumentType.STRING, - menu: "xrFeature", - defaultValue: "ar" - } - } - }, - "---", - { - opcode: "getStageWidth", - blockType: BlockType.REPORTER, - text: "stage width", - arguments: {} - }, - { - opcode: "getStageHeight", - blockType: BlockType.REPORTER, - text: "stage height", - arguments: {} - }, - "---", - { - opcode: "getMatrixItem", - blockType: BlockType.REPORTER, - text: "item [ITEM] of [MATRIX] matrix", - arguments: { - MATRIX: { - type: ArgumentType.STRING, - menu: "xrMatrix", - defaultValue: "combined" - }, - ITEM: { - type: ArgumentType.NUMBER, - defaultValue: 1 - } - } - }, - { - opcode: "getPosition", - blockType: BlockType.REPORTER, - text: "position [POSITION_COMPONENT]", - arguments: { - POSITION_COMPONENT: { - type: ArgumentType.STRING, - menu: "positionComponent", - defaultValue: "x" - } - } - }, - { - opcode: "getOrientation", - blockType: BlockType.REPORTER, - text: "orientation [ORIENTATION_COMPONENT]", - arguments: { - ORIENTATION_COMPONENT: { - type: ArgumentType.STRING, - menu: "orientationComponent", - defaultValue: "w" - } - } - }, - "---", - { - opcode: "getHitPosition", - blockType: BlockType.REPORTER, - text: "hit position [POSITION_COMPONENT]", - arguments: { - POSITION_COMPONENT: { - type: ArgumentType.STRING, - menu: "positionComponent", - defaultValue: "x" - } - } - }, - "---", - { - opcode: "moveSpaceBy", - blockType: BlockType.COMMAND, - text: "move everything by x:[X] y:[Y] z:[Z]", - arguments: { - X: { - type: ArgumentType.NUMBER, - defaultValue: 0 - }, - Y: { - type: ArgumentType.NUMBER, - defaultValue: 0 - }, - Z: { - type: ArgumentType.NUMBER, - defaultValue: 0 - } - } - }, - { - opcode: "turnSpaceBy", - blockType: BlockType.COMMAND, - text: "turn everything by r:[R] i:[I] j:[J] k:[K]", - arguments: { - R: { - type: ArgumentType.NUMBER, - defaultValue: 1 - }, - I: { - type: ArgumentType.NUMBER, - defaultValue: 0 - }, - J: { - type: ArgumentType.NUMBER, - defaultValue: 0 - }, - K: { - type: ArgumentType.NUMBER, - defaultValue: 0 - } - } - }, - "---", - { - opcode: "setResolution", - blockType: BlockType.COMMAND, - text: "set resolution [RESOLUTION]", - arguments: { - RESOLUTION: { - type: ArgumentType.NUMBER, - defaultValue: 1 - } - } - }, - ], - menus: { - positionComponent: { - acceptReporters: false, - items: [ - { - text: "x", - value: "x" - }, - { - text: "y", - value: "y" - }, - { - text: "z", - value: "z" - } - ] - }, - orientationComponent: { - acceptReporters: false, - items: [ - { - text: "r", - value: "w" - }, - { - text: "i", - value: "x" - }, - { - text: "j", - value: "y" - }, - { - text: "k", - value: "z" - } - ] - }, - xrMatrix: { - acceptReporters: false, - items: [ - "combined", - "projection", - "view", - "inverse view" - ] - }, - xrFeature: { - acceptReporters: false, - items: [ - "ar", - "pose", - "hit position" - ] - } - } - }; - } - enterAR() { - if (arFail) { - if (arFail !== "shown") { - // AR is used on mobile, where accessing browser console to see what's wrong can be an issue - alert("Project attempted to start AR even though it's not avalible. The reason: " + arFail + ". This message will only be shown once."); - arFail = "shown"; - } - } else { - if (!xrSession) { - if (enterARDone.length === 0) enterAR(null); - return new Promise(resolve => enterARDone.push(resolve)); - } - } - } - exitAR() { - if (xrSession) { - xrSession.end(); - } - } - isInAR() { - return !!xrSession; + const scale = div.clientHeight / canvas.clientHeight; + stageWrapper.style = + "transform-origin: top left; transform: scale(" + + scale + + "," + + scale + + ")"; + canvas.style.opacity = "0"; + + if (!isPackaged) { + const borderThing = stageWrapper.children[0].children[0].style; + borderThing["border"] = "none"; + borderThing["border-radius"] = "0"; + borderThing["transform"] = ""; // Removes translateX } - getStageWidth() { - return runtime.stageWidth; + } + poseAvailible = false; + if (xrRefSpace) { + const pose = frame.getViewerPose(xrRefSpace); + if (pose) { + poseAvailible = true; + xrProjectionMatrix = pose.views[0].projectionMatrix; + xrTransform = pose.views[0].transform; + const inverseTransformMatrix = xrTransform.inverse.matrix; + const a00 = xrProjectionMatrix[0]; + const a01 = xrProjectionMatrix[1]; + const a02 = xrProjectionMatrix[2]; + const a03 = xrProjectionMatrix[3]; + const a10 = xrProjectionMatrix[4]; + const a11 = xrProjectionMatrix[5]; + const a12 = xrProjectionMatrix[6]; + const a13 = xrProjectionMatrix[7]; + const a20 = xrProjectionMatrix[8]; + const a21 = xrProjectionMatrix[9]; + const a22 = xrProjectionMatrix[10]; + const a23 = xrProjectionMatrix[11]; + const a30 = xrProjectionMatrix[12]; + const a31 = xrProjectionMatrix[13]; + const a32 = xrProjectionMatrix[14]; + const a33 = xrProjectionMatrix[15]; + const b00 = inverseTransformMatrix[0]; + const b01 = inverseTransformMatrix[1]; + const b02 = inverseTransformMatrix[2]; + const b03 = inverseTransformMatrix[3]; + const b10 = inverseTransformMatrix[4]; + const b11 = inverseTransformMatrix[5]; + const b12 = inverseTransformMatrix[6]; + const b13 = inverseTransformMatrix[7]; + const b20 = inverseTransformMatrix[8]; + const b21 = inverseTransformMatrix[9]; + const b22 = inverseTransformMatrix[10]; + const b23 = inverseTransformMatrix[11]; + const b30 = inverseTransformMatrix[12]; + const b31 = inverseTransformMatrix[13]; + const b32 = inverseTransformMatrix[14]; + const b33 = inverseTransformMatrix[15]; + xrCombinedMatrix = [ + b00 * a00 + b01 * a10 + b02 * a20 + b03 * a30, + b00 * a01 + b01 * a11 + b02 * a21 + b03 * a31, + b00 * a02 + b01 * a12 + b02 * a22 + b03 * a32, + b00 * a03 + b01 * a13 + b02 * a23 + b03 * a33, + b10 * a00 + b11 * a10 + b12 * a20 + b13 * a30, + b10 * a01 + b11 * a11 + b12 * a21 + b13 * a31, + b10 * a02 + b11 * a12 + b12 * a22 + b13 * a32, + b10 * a03 + b11 * a13 + b12 * a23 + b13 * a33, + b20 * a00 + b21 * a10 + b22 * a20 + b23 * a30, + b20 * a01 + b21 * a11 + b22 * a21 + b23 * a31, + b20 * a02 + b21 * a12 + b22 * a22 + b23 * a32, + b20 * a03 + b21 * a13 + b22 * a23 + b23 * a33, + b30 * a00 + b31 * a10 + b32 * a20 + b33 * a30, + b30 * a01 + b31 * a11 + b32 * a21 + b33 * a31, + b30 * a02 + b31 * a12 + b32 * a22 + b33 * a32, + b30 * a03 + b31 * a13 + b32 * a23 + b33 * a33, + ]; } - getStageHeight() { - return runtime.stageHeight; + } + hitPositionAvailible = false; + if (xrHitTestSource) { + const hitTestResults = frame.getHitTestResults(xrHitTestSource); + if (hitTestResults.length > 0) { + hitPositionAvailible = true; + hitPosition = + hitTestResults[0].getPose(xrRefSpace).transform.position; } - getMatrixItem(args) { - let item = args.ITEM | 0; - if (item < 1 && item > 16) return ""; - let matrix = null; - switch (args.MATRIX) { - case "combined": - matrix = xrCombinedMatrix; - break; - case "projection": - matrix = xrProjectionMatrix; - break; - case "view": - matrix = xrTransform?.matrix; - break; - case "inverse view": - matrix = xrTransform?.inverse?.matrix; - break; - } - if (!matrix) return 0; - return matrix[item - 1] || 0; - } - moveSpaceBy(args) { - if (!xrRefSpace) return; - const x = +args.X || 0; - const y = +args.Y || 0; - const z = +args.Z || 0; - if (!isFinite(x + y + z)) return; - const offsetTransform = new XRRigidTransform({x: x, y: y, z: z}, {x: 0, y: 0, z: 0, w: 1}); - xrRefSpace = xrRefSpace.getOffsetReferenceSpace(offsetTransform); - } - turnSpaceBy(args) { - if (!xrRefSpace) return; - const r = +args.R || 0; - const i = +args.I || 0; - const j = +args.J || 0; - const k = +args.K || 0; - const len = Math.sqrt(r * r + i * i + j * j + k * k); - if (!isFinite(len) || len === 0) return; - const offsetTransform = new XRRigidTransform({x: 0, y: 0, z: 0}, {x: i / len, y: j / len, z: k / len, w: r / len}); - xrRefSpace = xrRefSpace.getOffsetReferenceSpace(offsetTransform); - } - getPosition(args) { - if (!xrTransform) return 0; - return xrTransform.position[args.POSITION_COMPONENT] || 0; + } + callback(); + }; + const cancel = () => { + if (idIsXR) { + xrSessionBackup.cancelAnimationFrame(id); + } else { + cancelAnimationFrame(id); + } + if (interval) { + clearInterval(interval); + } + }; + id = xrSession.requestAnimationFrame(handle); + if (fps > 0) { + interval = setInterval(() => { + shouldTriggerAgain = true; + }, 1000 / fps); + } + return { + cancel, + }; + }; + const start = function () { + this.running = true; + if (this.inXR) { + if (this.framerate === 0) { + this._stepAnimation = this.xrAnimationFrameWrapper( + this.stepCallback, + 0 + ); + this.runtime.currentStepTime = 1000 / 60; + } else { + // Interpolation should never be enabled when framerate === 0 as that's just redundant + if (this.interpolation) { + this._interpolationAnimation = animationFrameWrapper( + this.interpolationCallback + ); } - getOrientation(args) { - if (!xrTransform) return 0; - return xrTransform.orientation[args.ORIENTATION_COMPONENT] || 0; + this._stepAnimation = this.xrAnimationFrameWrapper( + this.stepCallback, + this.framerate + ); + this.runtime.currentStepTime = 1000 / this.framerate; + } + } else { + if (this.framerate === 0) { + this._stepAnimation = animationFrameWrapper(this.stepCallback); + this.runtime.currentStepTime = 1000 / 60; + } else { + // Interpolation should never be enabled when framerate === 0 as that's just redundant + if (this.interpolation) { + this._interpolationAnimation = animationFrameWrapper( + this.interpolationCallback + ); } - getHitPosition(args) { - if (!hitPosition) return 0; - return hitPosition[args.POSITION_COMPONENT] || 0; + this._stepInterval = setInterval( + this.stepCallback, + 1000 / this.framerate + ); + this.runtime.currentStepTime = 1000 / this.framerate; + } + } + }; + frameLoop.xrAnimationFrameWrapper = xrAnimationFrameWrapper.bind(frameLoop); + frameLoop.start = start.bind(frameLoop); + frameLoop.inXR = false; + + // Patching renderer.draw() to draw to xr framebuffer instead of canvas + const drawOrig = renderer.draw.bind(renderer); + const drawXR = function () { + const bl = this.xr.renderState.baseLayer; // ADDED + if (!bl) return; // Should fix very rare crash during exiting // ADDED + + this._doExitDrawRegion(); + + const gl = this._gl; + + gl.bindFramebuffer(gl.FRAMEBUFFER, bl.framebuffer); // CHANGED + gl.viewport(0, 0, bl.framebufferWidth, bl.framebufferHeight); // CHANGED + gl.clearColor(0, 0, 0, 0); + gl.clear(gl.COLOR_BUFFER_BIT); + + this._drawThese( + this._drawList, + "default" /*ShaderManager.DRAW_MODE.default*/, + this._projection, + { + framebufferWidth: bl.framebufferWidth, // CHANGED + framebufferHeight: bl.framebufferHeight, // CHANGED + } + ); + if (this._snapshotCallbacks.length > 0) { + const snapshot = gl.canvas.toDataURL(); + this._snapshotCallbacks.forEach((cb) => cb(snapshot)); + this._snapshotCallbacks = []; + } + }.bind(renderer); + renderer.draw = drawOrig; + + // Patching _pickTarget incorrect position bug: + // When the canvas is scaled using transform:scale, + // canvas.getBoundingClientRect is affected by it, but + // canvas.clientWidth and canvas.clientHeight are not. + // + // postData receives data.x and data.y, which are mouse position in + // screen units. To be able to rescale it to usable scratch units + // it also receives data.canvasWidth and data.canvasHeight + // which are based on getBoundingClientRect. Based of that it + // calculates this._scratchX and this._scratchY. + // Even when canvas is scaled, those are calculated correctly and + // as a result, blocks (mouse x) and (mouse y) report correct values. + // + // Later, postData calls _pickTarget, while only passing data.x and data.y + // without data.canvasWidth and data.canvasHeight. That method calls + // runtime renderer.pick, which calls clientSpaceToScratchBounds, which + // uses canvas.clientWidth and canvas.clientHeight to rescale mouse + // position from screen units to scratch units. This ignores + // transform:scale and as a result, sprites can't be clicked or dragged. + // + // WARNING: Makes _pickTarget only work correctly when called from postData. + // If something else calls it directly, it may cause problems. + const postDataOriginal = mouse.postData.bind(mouse); + mouse.postData = function (data) { + this._canvasWidth = data.canvasWidth; + this._canvasHeight = data.canvasHeight; + postDataOriginal(data); + }.bind(mouse); + + const _pickTargetOriginal = mouse._pickTarget.bind(mouse); + mouse._pickTarget = function (x, y) { + return _pickTargetOriginal( + (x / this._canvasWidth) * canvas.clientWidth, + (y / this._canvasHeight) * canvas.clientHeight + ); + }.bind(mouse); + + // This is used by . + // It was also broken in a similar way. + mouse.getClientX = function () { + return (this._clientX / this._canvasWidth) * canvas.clientWidth; + }.bind(mouse); + + mouse.getClientY = function () { + return (this._clientY / this._canvasHeight) * canvas.clientHeight; + }.bind(mouse); + // END OF WARNING + + const enterAR = function (event) { + if (!xrSession) { + // Entering and exiting editor recreates this element + stageWrapper = document.querySelector( + "[class*='stage-wrapper_stage-canvas-wrapper']" + ); + if (!stageWrapper) { + stageWrapper = document.querySelector("[class='sc-root']"); + scControlsBar = document.querySelector("[class='sc-controls-bar']"); + scLayers = document.querySelector("[class='sc-layers']"); + if (!stageWrapper) { + console.error( + (arFail = "Failed to get the div element of the stage") + ); + return; } - isFeatureAvailible(args) { - switch (args.FEATURE) { - case "ar": - return !arFail; - case "pose": - return poseAvailible; - case "hit position": - return hitPositionAvailible; - default: - return false; - } + isPackaged = true; + } + stageWrapperParent = stageWrapper.parentElement; + + const noop = () => {}; + navigator.xr + .requestSession("immersive-ar", { + requiredFeatures: ["hit-test", "dom-overlay"], + domOverlay: { root: div }, + }) + .then(onSuccess, event ? onError : onErrorTryTap); + // If (event) is defined, it was from click, so something went wrong. + // If (event) is null, it was called directly, and might've been rejected due to lack of user interaction. + } + }; + + class ARExtension { + getInfo() { + return { + id: "AR", + color1: "#d10000", + color2: "#bd0000", + color3: "#af0100", + docsURI: "https://extensions.turbowarp.org/ar", + blocks: [ + { + opcode: "enterAR", + blockType: BlockType.COMMAND, + text: "enter AR mode", + arguments: {}, + }, + { + opcode: "exitAR", + blockType: BlockType.COMMAND, + text: "exit AR mode", + arguments: {}, + }, + { + opcode: "isInAR", + blockType: BlockType.BOOLEAN, + text: "is in AR?", + arguments: {}, + }, + { + opcode: "isFeatureAvailible", + blockType: BlockType.BOOLEAN, + text: "is [FEATURE] availible?", + arguments: { + FEATURE: { + type: ArgumentType.STRING, + menu: "xrFeature", + defaultValue: "ar", + }, + }, + }, + "---", + { + opcode: "getStageWidth", + blockType: BlockType.REPORTER, + text: "stage width", + arguments: {}, + }, + { + opcode: "getStageHeight", + blockType: BlockType.REPORTER, + text: "stage height", + arguments: {}, + }, + "---", + { + opcode: "getMatrixItem", + blockType: BlockType.REPORTER, + text: "item [ITEM] of [MATRIX] matrix", + arguments: { + MATRIX: { + type: ArgumentType.STRING, + menu: "xrMatrix", + defaultValue: "combined", + }, + ITEM: { + type: ArgumentType.NUMBER, + defaultValue: 1, + }, + }, + }, + { + opcode: "getPosition", + blockType: BlockType.REPORTER, + text: "position [POSITION_COMPONENT]", + arguments: { + POSITION_COMPONENT: { + type: ArgumentType.STRING, + menu: "positionComponent", + defaultValue: "x", + }, + }, + }, + { + opcode: "getOrientation", + blockType: BlockType.REPORTER, + text: "orientation [ORIENTATION_COMPONENT]", + arguments: { + ORIENTATION_COMPONENT: { + type: ArgumentType.STRING, + menu: "orientationComponent", + defaultValue: "w", + }, + }, + }, + "---", + { + opcode: "getHitPosition", + blockType: BlockType.REPORTER, + text: "hit position [POSITION_COMPONENT]", + arguments: { + POSITION_COMPONENT: { + type: ArgumentType.STRING, + menu: "positionComponent", + defaultValue: "x", + }, + }, + }, + "---", + { + opcode: "moveSpaceBy", + blockType: BlockType.COMMAND, + text: "move everything by x:[X] y:[Y] z:[Z]", + arguments: { + X: { + type: ArgumentType.NUMBER, + defaultValue: 0, + }, + Y: { + type: ArgumentType.NUMBER, + defaultValue: 0, + }, + Z: { + type: ArgumentType.NUMBER, + defaultValue: 0, + }, + }, + }, + { + opcode: "turnSpaceBy", + blockType: BlockType.COMMAND, + text: "turn everything by r:[R] i:[I] j:[J] k:[K]", + arguments: { + R: { + type: ArgumentType.NUMBER, + defaultValue: 1, + }, + I: { + type: ArgumentType.NUMBER, + defaultValue: 0, + }, + J: { + type: ArgumentType.NUMBER, + defaultValue: 0, + }, + K: { + type: ArgumentType.NUMBER, + defaultValue: 0, + }, + }, + }, + "---", + { + opcode: "setResolution", + blockType: BlockType.COMMAND, + text: "set resolution [RESOLUTION]", + arguments: { + RESOLUTION: { + type: ArgumentType.NUMBER, + defaultValue: 1, + }, + }, + }, + ], + menus: { + positionComponent: { + acceptReporters: false, + items: [ + { + text: "x", + value: "x", + }, + { + text: "y", + value: "y", + }, + { + text: "z", + value: "z", + }, + ], + }, + orientationComponent: { + acceptReporters: false, + items: [ + { + text: "r", + value: "w", + }, + { + text: "i", + value: "x", + }, + { + text: "j", + value: "y", + }, + { + text: "k", + value: "z", + }, + ], + }, + xrMatrix: { + acceptReporters: false, + items: ["combined", "projection", "view", "inverse view"], + }, + xrFeature: { + acceptReporters: false, + items: ["ar", "pose", "hit position"], + }, + }, + }; + } + enterAR() { + if (arFail) { + if (arFail !== "shown") { + // AR is used on mobile, where accessing browser console to see what's wrong can be an issue + alert( + "Project attempted to start AR even though it's not avalible. The reason: " + + arFail + + ". This message will only be shown once." + ); + arFail = "shown"; } - setResolution(args) { - arResolution = Math.max(0.1, Math.min(1, +args.RESOLUTION || 0)); - if (xrSession) { - xrSession.updateRenderState({ - baseLayer: new XRWebGLLayer(xrSession, gl, { framebufferScaleFactor: arResolution }) - }); - } + } else { + if (!xrSession) { + if (enterARDone.length === 0) enterAR(null); + return new Promise((resolve) => enterARDone.push(resolve)); } + } + } + exitAR() { + if (xrSession) { + xrSession.end(); + } + } + isInAR() { + return !!xrSession; + } + getStageWidth() { + return runtime.stageWidth; + } + getStageHeight() { + return runtime.stageHeight; + } + getMatrixItem(args) { + let item = args.ITEM | 0; + if (item < 1 && item > 16) return ""; + let matrix = null; + switch (args.MATRIX) { + case "combined": + matrix = xrCombinedMatrix; + break; + case "projection": + matrix = xrProjectionMatrix; + break; + case "view": + matrix = xrTransform?.matrix; + break; + case "inverse view": + matrix = xrTransform?.inverse?.matrix; + break; + } + if (!matrix) return 0; + return matrix[item - 1] || 0; + } + moveSpaceBy(args) { + if (!xrRefSpace) return; + const x = +args.X || 0; + const y = +args.Y || 0; + const z = +args.Z || 0; + if (!isFinite(x + y + z)) return; + const offsetTransform = new XRRigidTransform( + { x: x, y: y, z: z }, + { x: 0, y: 0, z: 0, w: 1 } + ); + xrRefSpace = xrRefSpace.getOffsetReferenceSpace(offsetTransform); + } + turnSpaceBy(args) { + if (!xrRefSpace) return; + const r = +args.R || 0; + const i = +args.I || 0; + const j = +args.J || 0; + const k = +args.K || 0; + const len = Math.sqrt(r * r + i * i + j * j + k * k); + if (!isFinite(len) || len === 0) return; + const offsetTransform = new XRRigidTransform( + { x: 0, y: 0, z: 0 }, + { x: i / len, y: j / len, z: k / len, w: r / len } + ); + xrRefSpace = xrRefSpace.getOffsetReferenceSpace(offsetTransform); + } + getPosition(args) { + if (!xrTransform) return 0; + return xrTransform.position[args.POSITION_COMPONENT] || 0; + } + getOrientation(args) { + if (!xrTransform) return 0; + return xrTransform.orientation[args.ORIENTATION_COMPONENT] || 0; + } + getHitPosition(args) { + if (!hitPosition) return 0; + return hitPosition[args.POSITION_COMPONENT] || 0; + } + isFeatureAvailible(args) { + switch (args.FEATURE) { + case "ar": + return !arFail; + case "pose": + return poseAvailible; + case "hit position": + return hitPositionAvailible; + default: + return false; + } + } + setResolution(args) { + arResolution = Math.max(0.1, Math.min(1, +args.RESOLUTION || 0)); + if (xrSession) { + xrSession.updateRenderState({ + baseLayer: new XRWebGLLayer(xrSession, gl, { + framebufferScaleFactor: arResolution, + }), + }); + } } + } - Scratch.extensions.register(new ARExtension()); -})(Scratch); \ No newline at end of file + Scratch.extensions.register(new ARExtension()); +})(Scratch); diff --git a/extensions/battery.js b/extensions/battery.js index bf67e4ead6..fcb432b803 100644 --- a/extensions/battery.js +++ b/extensions/battery.js @@ -1,5 +1,9 @@ +// Name: Battery +// ID: battery +// Description: Access information about the battery of phones or laptops. May not work on all devices and browsers. + (function (Scratch) { - 'use strict'; + "use strict"; /** @type {Promise|null} */ let getBatteryPromise = null; @@ -18,34 +22,35 @@ return callback(cachedBattery); } if (!getBatteryPromise) { - getBatteryPromise = navigator.getBattery() - .then(battery => { + getBatteryPromise = navigator + .getBattery() + .then((battery) => { getBatteryPromise = null; cachedBattery = battery; - cachedBattery.addEventListener('chargingchange', () => { - Scratch.vm.runtime.startHats('battery_chargingChanged'); + cachedBattery.addEventListener("chargingchange", () => { + Scratch.vm.runtime.startHats("battery_chargingChanged"); }); - cachedBattery.addEventListener('levelchange', () => { - Scratch.vm.runtime.startHats('battery_levelChanged'); + cachedBattery.addEventListener("levelchange", () => { + Scratch.vm.runtime.startHats("battery_levelChanged"); }); - cachedBattery.addEventListener('chargingtimechange', () => { - Scratch.vm.runtime.startHats('battery_chargeTimeChanged'); + cachedBattery.addEventListener("chargingtimechange", () => { + Scratch.vm.runtime.startHats("battery_chargeTimeChanged"); }); - cachedBattery.addEventListener('dischargingtimechange', () => { - Scratch.vm.runtime.startHats('battery_dischargeTimeChanged'); + cachedBattery.addEventListener("dischargingtimechange", () => { + Scratch.vm.runtime.startHats("battery_dischargeTimeChanged"); }); return cachedBattery; }) - .catch(error => { + .catch((error) => { getBatteryPromise = null; - console.error('Could not get battery', error); + console.error("Could not get battery", error); batteryError = true; return null; }); } - return getBatteryPromise.then(battery => { + return getBatteryPromise.then((battery) => { return callback(battery); }); }; @@ -54,78 +59,78 @@ withBattery(() => {}); class BatteryExtension { - getInfo () { + getInfo() { return { - name: 'Battery', - id: 'battery', + name: "Battery", + id: "battery", blocks: [ { - opcode: 'charging', + opcode: "charging", blockType: Scratch.BlockType.BOOLEAN, - text: 'charging?' + text: "charging?", }, { - opcode: 'level', + opcode: "level", blockType: Scratch.BlockType.REPORTER, - text: 'battery level' + text: "battery level", }, { - opcode: 'chargeTime', + opcode: "chargeTime", blockType: Scratch.BlockType.REPORTER, - text: 'seconds until charged' + text: "seconds until charged", }, { - opcode: 'dischargeTime', + opcode: "dischargeTime", blockType: Scratch.BlockType.REPORTER, - text: 'seconds until empty' + text: "seconds until empty", }, { - opcode: 'chargingChanged', - blockType: Scratch.BlockType.HAT, - text: 'when charging changed', - isEdgeActivated: false + opcode: "chargingChanged", + blockType: Scratch.BlockType.EVENT, + text: "when charging changed", + isEdgeActivated: false, }, { - opcode: 'levelChanged', - blockType: Scratch.BlockType.HAT, - text: 'when battery level changed', - isEdgeActivated: false + opcode: "levelChanged", + blockType: Scratch.BlockType.EVENT, + text: "when battery level changed", + isEdgeActivated: false, }, { - opcode: 'chargeTimeChanged', - blockType: Scratch.BlockType.HAT, - text: 'when time until charged changed', - isEdgeActivated: false + opcode: "chargeTimeChanged", + blockType: Scratch.BlockType.EVENT, + text: "when time until charged changed", + isEdgeActivated: false, }, { - opcode: 'dischargeTimeChanged', - blockType: Scratch.BlockType.HAT, - text: 'when time until empty changed', - isEdgeActivated: false + opcode: "dischargeTimeChanged", + blockType: Scratch.BlockType.EVENT, + text: "when time until empty changed", + isEdgeActivated: false, }, - ] + ], }; } - charging () { - return withBattery(battery => { + charging() { + return withBattery((battery) => { if (!battery) return true; return battery.charging; }); } - level () { - return withBattery(battery => { + level() { + return withBattery((battery) => { if (!battery) return 100; return battery.level * 100; }); } - chargeTime () { - return withBattery(battery => { + chargeTime() { + return withBattery((battery) => { if (!battery) return 0; return battery.chargingTime; }); } - dischargeTime () { - return withBattery(battery => { + dischargeTime() { + return withBattery((battery) => { if (!battery) return Infinity; return battery.dischargingTime; }); diff --git a/extensions/bitwise.js b/extensions/bitwise.js index 938e2180c1..d78d514037 100644 --- a/extensions/bitwise.js +++ b/extensions/bitwise.js @@ -1,16 +1,22 @@ -(Scratch => { - 'use strict'; +// Name: Bitwise +// ID: Bitwise +// Description: Blocks that operate on the binary representation of numbers in computers. +// By: TrueFantom - const icon = ''; +((Scratch) => { + "use strict"; - const isNumberBits = bits => { + const icon = + ""; + + const isNumberBits = (bits) => { return /^-?[01]+$/.test(bits); }; - const number2bits = number => { + const number2bits = (number) => { return Scratch.Cast.toNumber(number).toString(2); }; - const bits2number = bits => { - return /^-?[01]+$/.test(bits) ? (parseInt(bits, 2) || 0) : 0; + const bits2number = (bits) => { + return /^-?[01]+$/.test(bits) ? parseInt(bits, 2) || 0 : 0; }; const circularRightShift = (number, k) => { @@ -21,224 +27,222 @@ }; class Bitwise { - getInfo() { return { + id: "Bitwise", + name: "Bitwise", - id: 'Bitwise', - name: 'Bitwise', - - color1: '#17cde6', + color1: "#17cde6", - docsURI: "https://extensions.turbowarp.org/bitwise.html", + docsURI: "https://extensions.turbowarp.org/bitwise", menuIconURI: icon, blocks: [ { - opcode: 'isNumberBits', + opcode: "isNumberBits", blockType: Scratch.BlockType.BOOLEAN, - text: 'is [CENTRAL] binary?', + text: "is [CENTRAL] binary?", arguments: { CENTRAL: { type: Scratch.ArgumentType.NUMBER, - defaultValue: '0000000000100000' - } - } + defaultValue: "0000000000100000", + }, + }, }, - '---', + "---", { - opcode: 'toNumberBits', + opcode: "toNumberBits", blockType: Scratch.BlockType.REPORTER, - text: '[CENTRAL] to binary', + text: "[CENTRAL] to binary", arguments: { CENTRAL: { type: Scratch.ArgumentType.NUMBER, - defaultValue: '32' - } - } + defaultValue: "32", + }, + }, }, { - opcode: 'ofNumberBits', + opcode: "ofNumberBits", blockType: Scratch.BlockType.REPORTER, - text: '[CENTRAL] to number', + text: "[CENTRAL] to number", arguments: { CENTRAL: { type: Scratch.ArgumentType.NUMBER, - defaultValue: '0000000000100000' - } - } + defaultValue: "0000000000100000", + }, + }, }, - '---', + "---", { - opcode: 'bitwiseRightShift', + opcode: "bitwiseRightShift", blockType: Scratch.BlockType.REPORTER, - text: '[LEFT] >> [RIGHT]', + text: "[LEFT] >> [RIGHT]", arguments: { LEFT: { type: Scratch.ArgumentType.NUMBER, - defaultValue: '' + defaultValue: "", }, RIGHT: { type: Scratch.ArgumentType.NUMBER, - defaultValue: '' - } - } + defaultValue: "", + }, + }, }, { - opcode: 'bitwiseLeftShift', + opcode: "bitwiseLeftShift", blockType: Scratch.BlockType.REPORTER, - text: '[LEFT] << [RIGHT]', + text: "[LEFT] << [RIGHT]", arguments: { LEFT: { type: Scratch.ArgumentType.NUMBER, - defaultValue: '' + defaultValue: "", }, RIGHT: { type: Scratch.ArgumentType.NUMBER, - defaultValue: '' - } - } + defaultValue: "", + }, + }, }, { - opcode: 'bitwiseLogicalRightShift', + opcode: "bitwiseLogicalRightShift", blockType: Scratch.BlockType.REPORTER, - text: '[LEFT] >>> [RIGHT]', + text: "[LEFT] >>> [RIGHT]", arguments: { LEFT: { type: Scratch.ArgumentType.NUMBER, - defaultValue: '' + defaultValue: "", }, RIGHT: { type: Scratch.ArgumentType.NUMBER, - defaultValue: '' - } - } + defaultValue: "", + }, + }, }, - { - opcode: 'bitwiseCircularRightShift', + { + opcode: "bitwiseCircularRightShift", blockType: Scratch.BlockType.REPORTER, - text: '[LEFT] ↻ [RIGHT]', + text: "[LEFT] ↻ [RIGHT]", arguments: { LEFT: { type: Scratch.ArgumentType.NUMBER, - defaultValue: '' + defaultValue: "", }, RIGHT: { type: Scratch.ArgumentType.NUMBER, - defaultValue: '' - } - } + defaultValue: "", + }, + }, }, { - opcode: 'bitwiseCircularLeftShift', + opcode: "bitwiseCircularLeftShift", blockType: Scratch.BlockType.REPORTER, - text: '[LEFT] ↺ [RIGHT]', + text: "[LEFT] ↺ [RIGHT]", arguments: { LEFT: { type: Scratch.ArgumentType.NUMBER, - defaultValue: '' + defaultValue: "", }, RIGHT: { type: Scratch.ArgumentType.NUMBER, - defaultValue: '' - } - } + defaultValue: "", + }, + }, }, - '---', + "---", { - opcode: 'bitwiseAnd', + opcode: "bitwiseAnd", blockType: Scratch.BlockType.REPORTER, - text: '[LEFT] and [RIGHT]', + text: "[LEFT] and [RIGHT]", arguments: { LEFT: { type: Scratch.ArgumentType.NUMBER, - defaultValue: '' + defaultValue: "", }, RIGHT: { type: Scratch.ArgumentType.NUMBER, - defaultValue: '' - } - } + defaultValue: "", + }, + }, }, { - opcode: 'bitwiseOr', + opcode: "bitwiseOr", blockType: Scratch.BlockType.REPORTER, - text: '[LEFT] or [RIGHT]', + text: "[LEFT] or [RIGHT]", arguments: { LEFT: { type: Scratch.ArgumentType.NUMBER, - defaultValue: '' + defaultValue: "", }, RIGHT: { type: Scratch.ArgumentType.NUMBER, - defaultValue: '' - } - } + defaultValue: "", + }, + }, }, { - opcode: 'bitwiseXor', + opcode: "bitwiseXor", blockType: Scratch.BlockType.REPORTER, - text: '[LEFT] xor [RIGHT]', + text: "[LEFT] xor [RIGHT]", arguments: { LEFT: { type: Scratch.ArgumentType.NUMBER, - defaultValue: '' + defaultValue: "", }, RIGHT: { type: Scratch.ArgumentType.NUMBER, - defaultValue: '' - } - } + defaultValue: "", + }, + }, }, { - opcode: 'bitwiseNot', + opcode: "bitwiseNot", blockType: Scratch.BlockType.REPORTER, - text: 'not [CENTRAL]', + text: "not [CENTRAL]", arguments: { CENTRAL: { type: Scratch.ArgumentType.NUMBER, - defaultValue: '' - } - } - } - ] + defaultValue: "", + }, + }, + }, + ], }; } - isNumberBits({CENTRAL}) { + isNumberBits({ CENTRAL }) { return isNumberBits(CENTRAL); } - toNumberBits({CENTRAL}) { + toNumberBits({ CENTRAL }) { return number2bits(CENTRAL); } - ofNumberBits({CENTRAL}) { + ofNumberBits({ CENTRAL }) { return bits2number(CENTRAL); } - bitwiseRightShift({LEFT, RIGHT}) { + bitwiseRightShift({ LEFT, RIGHT }) { return LEFT >> RIGHT; } - bitwiseLeftShift({LEFT, RIGHT}) { + bitwiseLeftShift({ LEFT, RIGHT }) { return LEFT << RIGHT; } - bitwiseLogicalRightShift({LEFT, RIGHT}) { + bitwiseLogicalRightShift({ LEFT, RIGHT }) { return LEFT >>> RIGHT; } - bitwiseCircularRightShift({LEFT, RIGHT}) { + bitwiseCircularRightShift({ LEFT, RIGHT }) { return circularRightShift(LEFT, RIGHT); } - bitwiseCircularLeftShift({LEFT, RIGHT}) { + bitwiseCircularLeftShift({ LEFT, RIGHT }) { return circularLeftShift(LEFT, RIGHT); } - bitwiseAnd({LEFT, RIGHT}) { + bitwiseAnd({ LEFT, RIGHT }) { return LEFT & RIGHT; } - bitwiseOr({LEFT, RIGHT}) { + bitwiseOr({ LEFT, RIGHT }) { return LEFT | RIGHT; } - bitwiseXor({LEFT, RIGHT}) { + bitwiseXor({ LEFT, RIGHT }) { return LEFT ^ RIGHT; } - bitwiseNot({CENTRAL}) { + bitwiseNot({ CENTRAL }) { return ~CENTRAL; } } diff --git a/extensions/box2d.js b/extensions/box2d.js index 8c5f411dc2..42df1ab034 100644 --- a/extensions/box2d.js +++ b/extensions/box2d.js @@ -1,14 +1,19 @@ +// Name: Box2D Physics +// ID: griffpatch +// Description: Two dimensional physics. +// Original: griffpatch + /*! * This is based on https://github.com/griffpatch/scratch-vm/tree/box2d/src/extensions/scratch3_griffpatch */ /* eslint-disable */ -(function(Scratch) { - 'use strict'; +(function (Scratch) { + "use strict"; if (!Scratch.extensions.unsandboxed) { - throw new Error('Box2D must be run unsandboxed'); + throw new Error("Box2D must be run unsandboxed"); } // First we need to load the Box2D physics library that this extension uses. @@ -16,22 +21,22 @@ // the source code below. Yes, this is really ugly. /*! - * Copyright (c) 2006-2007 Erin Catto http://www.gphysics.com - * - * This software is provided 'as-is', without any express or implied - * warranty. In no event will the authors be held liable for any damages - * arising from the use of this software. - * Permission is granted to anyone to use this software for any purpose, - * including commercial applications, and to alter it and redistribute it - * freely, subject to the following restrictions: - * 1. The origin of this software must not be misrepresented; you must not - * claim that you wrote the original software. If you use this software - * in a product, an acknowledgment in the product documentation would be - * appreciated but is not required. - * 2. Altered source versions must be plainly marked as such, and must not be - * misrepresented as being the original software. - * 3. This notice may not be removed or altered from any source distribution. - */ + * Copyright (c) 2006-2007 Erin Catto http://www.gphysics.com + * + * This software is provided 'as-is', without any express or implied + * warranty. In no event will the authors be held liable for any damages + * arising from the use of this software. + * Permission is granted to anyone to use this software for any purpose, + * including commercial applications, and to alter it and redistribute it + * freely, subject to the following restrictions: + * 1. The origin of this software must not be misrepresented; you must not + * claim that you wrote the original software. If you use this software + * in a product, an acknowledgment in the product documentation would be + * appreciated but is not required. + * 2. Altered source versions must be plainly marked as such, and must not be + * misrepresented as being the original software. + * 3. This notice may not be removed or altered from any source distribution. + */ var Box2D = {}; (function (a2j, undefined) { @@ -78,7 +83,8 @@ //package structure if (typeof Box2D === "undefined") Box2D = {}; if (typeof Box2D.Collision === "undefined") Box2D.Collision = {}; - if (typeof Box2D.Collision.Shapes === "undefined") Box2D.Collision.Shapes = {}; + if (typeof Box2D.Collision.Shapes === "undefined") + Box2D.Collision.Shapes = {}; if (typeof Box2D.Common === "undefined") Box2D.Common = {}; if (typeof Box2D.Common.Math === "undefined") Box2D.Common.Math = {}; if (typeof Box2D.Dynamics === "undefined") Box2D.Dynamics = {}; @@ -169,7 +175,8 @@ function b2Manifold() { b2Manifold.b2Manifold.apply(this, arguments); - if (this.constructor === b2Manifold) this.b2Manifold.apply(this, arguments); + if (this.constructor === b2Manifold) + this.b2Manifold.apply(this, arguments); } Box2D.Collision.b2Manifold = b2Manifold; @@ -515,13 +522,19 @@ Box2D.Dynamics.Controllers.b2BuoyancyController = b2BuoyancyController; function b2ConstantAccelController() { - b2ConstantAccelController.b2ConstantAccelController.apply(this, arguments); + b2ConstantAccelController.b2ConstantAccelController.apply( + this, + arguments + ); } Box2D.Dynamics.Controllers.b2ConstantAccelController = b2ConstantAccelController; function b2ConstantForceController() { - b2ConstantForceController.b2ConstantForceController.apply(this, arguments); + b2ConstantForceController.b2ConstantForceController.apply( + this, + arguments + ); } Box2D.Dynamics.Controllers.b2ConstantForceController = b2ConstantForceController; @@ -542,7 +555,10 @@ Box2D.Dynamics.Controllers.b2GravityController = b2GravityController; function b2TensorDampingController() { - b2TensorDampingController.b2TensorDampingController.apply(this, arguments); + b2TensorDampingController.b2TensorDampingController.apply( + this, + arguments + ); } Box2D.Dynamics.Controllers.b2TensorDampingController = b2TensorDampingController; @@ -602,7 +618,8 @@ function b2JointDef() { b2JointDef.b2JointDef.apply(this, arguments); - if (this.constructor === b2JointDef) this.b2JointDef.apply(this, arguments); + if (this.constructor === b2JointDef) + this.b2JointDef.apply(this, arguments); } Box2D.Dynamics.Joints.b2JointDef = b2JointDef; @@ -957,7 +974,13 @@ var separation = v2X * normal1WorldX + v2Y * normal1WorldY; return separation; }; - b2Collision.FindMaxSeparation = function (edgeIndex, poly1, xf1, poly2, xf2) { + b2Collision.FindMaxSeparation = function ( + edgeIndex, + poly1, + xf1, + poly2, + xf2 + ) { var count1 = parseInt(poly1.m_vertexCount); var normals1 = poly1.m_normals; var tVec; @@ -1003,7 +1026,8 @@ return s; } while (true) { - if (increment == -1) edge = bestEdge - 1 >= 0 ? bestEdge - 1 : count1 - 1; + if (increment == -1) + edge = bestEdge - 1 >= 0 ? bestEdge - 1 : count1 - 1; else edge = bestEdge + 1 < count1 ? bestEdge + 1 : 0; s = b2Collision.EdgeSeparation(poly1, xf1, edge, poly2, xf2); if (s > bestSeparation) { @@ -1049,16 +1073,20 @@ tClip = c[0]; tVec = vertices2[i1]; tMat = xf2.R; - tClip.v.x = xf2.position.x + (tMat.col1.x * tVec.x + tMat.col2.x * tVec.y); - tClip.v.y = xf2.position.y + (tMat.col1.y * tVec.x + tMat.col2.y * tVec.y); + tClip.v.x = + xf2.position.x + (tMat.col1.x * tVec.x + tMat.col2.x * tVec.y); + tClip.v.y = + xf2.position.y + (tMat.col1.y * tVec.x + tMat.col2.y * tVec.y); tClip.id.features.referenceEdge = edge1; tClip.id.features.incidentEdge = i1; tClip.id.features.incidentVertex = 0; tClip = c[1]; tVec = vertices2[i2]; tMat = xf2.R; - tClip.v.x = xf2.position.x + (tMat.col1.x * tVec.x + tMat.col2.x * tVec.y); - tClip.v.y = xf2.position.y + (tMat.col1.y * tVec.x + tMat.col2.y * tVec.y); + tClip.v.x = + xf2.position.x + (tMat.col1.x * tVec.x + tMat.col2.x * tVec.y); + tClip.v.y = + xf2.position.y + (tMat.col1.y * tVec.x + tMat.col2.y * tVec.y); tClip.id.features.referenceEdge = edge1; tClip.id.features.incidentEdge = i2; tClip.id.features.incidentVertex = 1; @@ -1156,13 +1184,17 @@ var v11 = b2Collision.s_v11; var v12 = b2Collision.s_v12; v11.x = - xf1.position.x + (tMat.col1.x * local_v11.x + tMat.col2.x * local_v11.y); + xf1.position.x + + (tMat.col1.x * local_v11.x + tMat.col2.x * local_v11.y); v11.y = - xf1.position.y + (tMat.col1.y * local_v11.x + tMat.col2.y * local_v11.y); + xf1.position.y + + (tMat.col1.y * local_v11.x + tMat.col2.y * local_v11.y); v12.x = - xf1.position.x + (tMat.col1.x * local_v12.x + tMat.col2.x * local_v12.y); + xf1.position.x + + (tMat.col1.x * local_v12.x + tMat.col2.x * local_v12.y); v12.y = - xf1.position.y + (tMat.col1.y * local_v12.x + tMat.col2.y * local_v12.y); + xf1.position.y + + (tMat.col1.y * local_v12.x + tMat.col2.y * local_v12.y); var frontOffset = normal.x * v11.x + normal.y * v11.y; var sideOffset1 = -tangent.x * v11.x - tangent.y * v11.y + totalRadius; var sideOffset2 = tangent.x * v12.x + tangent.y * v12.y + totalRadius; @@ -1203,7 +1235,13 @@ } manifold.m_pointCount = pointCount; }; - b2Collision.CollideCircles = function (manifold, circle1, xf1, circle2, xf2) { + b2Collision.CollideCircles = function ( + manifold, + circle1, + xf1, + circle2, + xf2 + ) { manifold.m_pointCount = 0; var tMat; var tVec; @@ -1489,7 +1527,9 @@ } b2Distance.b2_gjkMaxIters = b2Math.Max(b2Distance.b2_gjkMaxIters, iter); simplex.GetWitnessPoints(output.pointA, output.pointB); - output.distance = b2Math.SubtractVV(output.pointA, output.pointB).Length(); + output.distance = b2Math + .SubtractVV(output.pointA, output.pointB) + .Length(); output.iterations = iter; simplex.WriteCache(cache); if (input.useRadii) { @@ -1748,17 +1788,21 @@ var child2 = sibling.child2; var norm1 = Math.abs( - (child1.aabb.lowerBound.x + child1.aabb.upperBound.x) / 2 - center.x + (child1.aabb.lowerBound.x + child1.aabb.upperBound.x) / 2 - + center.x ) + Math.abs( - (child1.aabb.lowerBound.y + child1.aabb.upperBound.y) / 2 - center.y + (child1.aabb.lowerBound.y + child1.aabb.upperBound.y) / 2 - + center.y ); var norm2 = Math.abs( - (child2.aabb.lowerBound.x + child2.aabb.upperBound.x) / 2 - center.x + (child2.aabb.lowerBound.x + child2.aabb.upperBound.x) / 2 - + center.x ) + Math.abs( - (child2.aabb.lowerBound.y + child2.aabb.upperBound.y) / 2 - center.y + (child2.aabb.lowerBound.y + child2.aabb.upperBound.y) / 2 - + center.y ); if (norm1 < norm2) { sibling = child1; @@ -2275,15 +2319,19 @@ tVec = this.m_localPoint; tMat = transformB.R; pointBX = - transformB.position.x + (tMat.col1.x * tVec.x + tMat.col2.x * tVec.y); + transformB.position.x + + (tMat.col1.x * tVec.x + tMat.col2.x * tVec.y); pointBY = - transformB.position.y + (tMat.col1.y * tVec.x + tMat.col2.y * tVec.y); + transformB.position.y + + (tMat.col1.y * tVec.x + tMat.col2.y * tVec.y); tVec = localPointA; tMat = transformA.R; pointAX = - transformA.position.x + (tMat.col1.x * tVec.x + tMat.col2.x * tVec.y); + transformA.position.x + + (tMat.col1.x * tVec.x + tMat.col2.x * tVec.y); pointAY = - transformA.position.y + (tMat.col1.y * tVec.x + tMat.col2.y * tVec.y); + transformA.position.y + + (tMat.col1.y * tVec.x + tMat.col2.y * tVec.y); sgn = (pointAX - pointBX) * normalX + (pointAY - pointBY) * normalY; if (s < 0.0) { this.m_axis.NegativeSelf(); @@ -2302,15 +2350,19 @@ tVec = this.m_localPoint; tMat = transformA.R; pointAX = - transformA.position.x + (tMat.col1.x * tVec.x + tMat.col2.x * tVec.y); + transformA.position.x + + (tMat.col1.x * tVec.x + tMat.col2.x * tVec.y); pointAY = - transformA.position.y + (tMat.col1.y * tVec.x + tMat.col2.y * tVec.y); + transformA.position.y + + (tMat.col1.y * tVec.x + tMat.col2.y * tVec.y); tVec = localPointB; tMat = transformB.R; pointBX = - transformB.position.x + (tMat.col1.x * tVec.x + tMat.col2.x * tVec.y); + transformB.position.x + + (tMat.col1.x * tVec.x + tMat.col2.x * tVec.y); pointBY = - transformB.position.y + (tMat.col1.y * tVec.x + tMat.col2.y * tVec.y); + transformB.position.y + + (tMat.col1.y * tVec.x + tMat.col2.y * tVec.y); sgn = (pointBX - pointAX) * normalX + (pointBY - pointAY) * normalY; if (s < 0.0) { this.m_axis.NegativeSelf(); @@ -2318,7 +2370,10 @@ } } }; - b2SeparationFunction.prototype.Evaluate = function (transformA, transformB) { + b2SeparationFunction.prototype.Evaluate = function ( + transformA, + transformB + ) { var axisA; var axisB; var localPointA; @@ -3133,7 +3188,8 @@ }; b2CircleShape.prototype.ComputeMass = function (massData, density) { if (density === undefined) density = 0; - massData.mass = density * b2Settings.b2_pi * this.m_radius * this.m_radius; + massData.mass = + density * b2Settings.b2_pi * this.m_radius * this.m_radius; massData.center.SetV(this.m_p); massData.I = massData.mass * @@ -3395,7 +3451,10 @@ this.m_nextEdge = null; this.m_v1 = v1; this.m_v2 = v2; - this.m_direction.Set(this.m_v2.x - this.m_v1.x, this.m_v2.y - this.m_v1.y); + this.m_direction.Set( + this.m_v2.x - this.m_v1.x, + this.m_v2.y - this.m_v1.y + ); this.m_length = this.m_direction.Normalize(); this.m_normal.Set(this.m_direction.y, -this.m_direction.x); this.m_coreV1.Set( @@ -3413,13 +3472,23 @@ this.m_cornerDir1 = this.m_normal; this.m_cornerDir2.Set(-this.m_normal.x, -this.m_normal.y); }; - b2EdgeShape.prototype.SetPrevEdge = function (edge, core, cornerDir, convex) { + b2EdgeShape.prototype.SetPrevEdge = function ( + edge, + core, + cornerDir, + convex + ) { this.m_prevEdge = edge; this.m_coreV1 = core; this.m_cornerDir1 = cornerDir; this.m_cornerConvex1 = convex; }; - b2EdgeShape.prototype.SetNextEdge = function (edge, core, cornerDir, convex) { + b2EdgeShape.prototype.SetNextEdge = function ( + edge, + core, + cornerDir, + convex + ) { this.m_nextEdge = edge; this.m_coreV2 = core; this.m_cornerDir2 = cornerDir; @@ -3521,7 +3590,12 @@ polygonShape.SetAsBox(hx, hy); return polygonShape; }; - b2PolygonShape.prototype.SetAsOrientedBox = function (hx, hy, center, angle) { + b2PolygonShape.prototype.SetAsOrientedBox = function ( + hx, + hy, + center, + angle + ) { if (hx === undefined) hx = 0; if (hy === undefined) hy = 0; if (center === undefined) center = null; @@ -3646,8 +3720,10 @@ b2PolygonShape.prototype.ComputeAABB = function (aabb, xf) { var tMat = xf.R; var tVec = this.m_vertices[0]; - var lowerX = xf.position.x + (tMat.col1.x * tVec.x + tMat.col2.x * tVec.y); - var lowerY = xf.position.y + (tMat.col1.y * tVec.x + tMat.col2.y * tVec.y); + var lowerX = + xf.position.x + (tMat.col1.x * tVec.x + tMat.col2.x * tVec.y); + var lowerY = + xf.position.y + (tMat.col1.y * tVec.x + tMat.col2.y * tVec.y); var upperX = lowerX; var upperY = lowerY; for (var i = 1; i < this.m_vertexCount; ++i) { @@ -3703,11 +3779,13 @@ var ey2 = e2Y; var intx2 = k_inv3 * - (0.25 * (ex1 * ex1 + ex2 * ex1 + ex2 * ex2) + (px * ex1 + px * ex2)) + + (0.25 * (ex1 * ex1 + ex2 * ex1 + ex2 * ex2) + + (px * ex1 + px * ex2)) + 0.5 * px * px; var inty2 = k_inv3 * - (0.25 * (ey1 * ey1 + ey2 * ey1 + ey2 * ey2) + (py * ey1 + py * ey2)) + + (0.25 * (ey1 * ey1 + ey2 * ey1 + ey2 * ey2) + + (py * ey1 + py * ey2)) + 0.5 * py * py; I += D * (intx2 + inty2); } @@ -3925,8 +4003,10 @@ var centerX = 0.5 * (lowerX + upperX); var centerY = 0.5 * (lowerY + upperY); var tMat = obb.R; - obb.center.x = root.x + (tMat.col1.x * centerX + tMat.col2.x * centerY); - obb.center.y = root.y + (tMat.col1.y * centerX + tMat.col2.y * centerY); + obb.center.x = + root.x + (tMat.col1.x * centerX + tMat.col2.x * centerY); + obb.center.y = + root.y + (tMat.col1.y * centerX + tMat.col2.y * centerY); obb.extents.x = 0.5 * (upperX - lowerX); obb.extents.y = 0.5 * (upperY - lowerY); } @@ -4404,8 +4484,14 @@ return C; }; b2Math.MulTMM = function (A, B) { - var c1 = new b2Vec2(b2Math.Dot(A.col1, B.col1), b2Math.Dot(A.col2, B.col1)); - var c2 = new b2Vec2(b2Math.Dot(A.col1, B.col2), b2Math.Dot(A.col2, B.col2)); + var c1 = new b2Vec2( + b2Math.Dot(A.col1, B.col1), + b2Math.Dot(A.col2, B.col1) + ); + var c2 = new b2Vec2( + b2Math.Dot(A.col1, B.col2), + b2Math.Dot(A.col2, B.col2) + ); var C = b2Mat22.FromVV(c1, c2); return C; }; @@ -4780,7 +4866,8 @@ b2CircleContact = Box2D.Dynamics.Contacts.b2CircleContact, b2Contact = Box2D.Dynamics.Contacts.b2Contact, b2ContactConstraint = Box2D.Dynamics.Contacts.b2ContactConstraint, - b2ContactConstraintPoint = Box2D.Dynamics.Contacts.b2ContactConstraintPoint, + b2ContactConstraintPoint = + Box2D.Dynamics.Contacts.b2ContactConstraintPoint, b2ContactEdge = Box2D.Dynamics.Contacts.b2ContactEdge, b2ContactFactory = Box2D.Dynamics.Contacts.b2ContactFactory, b2ContactRegister = Box2D.Dynamics.Contacts.b2ContactRegister, @@ -4791,7 +4878,8 @@ b2PolyAndCircleContact = Box2D.Dynamics.Contacts.b2PolyAndCircleContact, b2PolyAndEdgeContact = Box2D.Dynamics.Contacts.b2PolyAndEdgeContact, b2PolygonContact = Box2D.Dynamics.Contacts.b2PolygonContact, - b2PositionSolverManifold = Box2D.Dynamics.Contacts.b2PositionSolverManifold, + b2PositionSolverManifold = + Box2D.Dynamics.Contacts.b2PositionSolverManifold, b2Controller = Box2D.Dynamics.Controllers.b2Controller, b2DistanceJoint = Box2D.Dynamics.Joints.b2DistanceJoint, b2DistanceJointDef = Box2D.Dynamics.Joints.b2DistanceJointDef, @@ -4984,7 +5072,8 @@ bd.angularDamping = this.m_angularDamping; bd.angularVelocity = this.m_angularVelocity; bd.fixedRotation = - (this.m_flags & b2Body.e_fixedRotationFlag) == b2Body.e_fixedRotationFlag; + (this.m_flags & b2Body.e_fixedRotationFlag) == + b2Body.e_fixedRotationFlag; bd.bullet = (this.m_flags & b2Body.e_bulletFlag) == b2Body.e_bulletFlag; bd.awake = (this.m_flags & b2Body.e_awakeFlag) == b2Body.e_awakeFlag; bd.linearDamping = this.m_linearDamping; @@ -5126,7 +5215,10 @@ this.m_mass = 1.0; } this.m_invMass = 1.0 / this.m_mass; - if (massData.I > 0.0 && (this.m_flags & b2Body.e_fixedRotationFlag) == 0) { + if ( + massData.I > 0.0 && + (this.m_flags & b2Body.e_fixedRotationFlag) == 0 + ) { this.m_I = massData.I - this.m_mass * @@ -5313,7 +5405,8 @@ }; b2Body.prototype.IsFixedRotation = function () { return ( - (this.m_flags & b2Body.e_fixedRotationFlag) == b2Body.e_fixedRotationFlag + (this.m_flags & b2Body.e_fixedRotationFlag) == + b2Body.e_fixedRotationFlag ); }; b2Body.prototype.SetActive = function (flag) { @@ -5347,7 +5440,9 @@ return (this.m_flags & b2Body.e_activeFlag) == b2Body.e_activeFlag; }; b2Body.prototype.IsSleepingAllowed = function () { - return (this.m_flags & b2Body.e_allowSleepFlag) == b2Body.e_allowSleepFlag; + return ( + (this.m_flags & b2Body.e_allowSleepFlag) == b2Body.e_allowSleepFlag + ); }; b2Body.prototype.GetFixtureList = function () { return this.m_fixtureList; @@ -5561,8 +5656,10 @@ proxyUserDataA, proxyUserDataB ) { - var fixtureA = proxyUserDataA instanceof b2Fixture ? proxyUserDataA : null; - var fixtureB = proxyUserDataB instanceof b2Fixture ? proxyUserDataB : null; + var fixtureA = + proxyUserDataA instanceof b2Fixture ? proxyUserDataA : null; + var fixtureB = + proxyUserDataB instanceof b2Fixture ? proxyUserDataB : null; var bodyA = fixtureA.GetBody(); var bodyB = fixtureB.GetBody(); if (bodyA == bodyB) return; @@ -5729,7 +5826,11 @@ if (xformScale === undefined) xformScale = 0; }; b2DebugDraw.prototype.GetXFormScale = function () {}; - b2DebugDraw.prototype.DrawPolygon = function (vertices, vertexCount, color) { + b2DebugDraw.prototype.DrawPolygon = function ( + vertices, + vertexCount, + color + ) { if (vertexCount === undefined) vertexCount = 0; }; b2DebugDraw.prototype.DrawSolidPolygon = function ( @@ -5979,8 +6080,10 @@ for (i = 0; i < this.m_bodyCount; ++i) { b = this.m_bodies[i]; if (b.GetType() != b2Body.b2_dynamicBody) continue; - b.m_linearVelocity.x += step.dt * (gravity.x + b.m_invMass * b.m_force.x); - b.m_linearVelocity.y += step.dt * (gravity.y + b.m_invMass * b.m_force.y); + b.m_linearVelocity.x += + step.dt * (gravity.x + b.m_invMass * b.m_force.x); + b.m_linearVelocity.y += + step.dt * (gravity.y + b.m_invMass * b.m_force.y); b.m_angularVelocity += step.dt * b.m_invI * b.m_torque; b.m_linearVelocity.Multiply( b2Math.Clamp(1.0 - step.dt * b.m_linearDamping, 0.0, 1.0) @@ -6063,7 +6166,8 @@ if (allowSleep) { var minSleepTime = Number.MAX_VALUE; var linTolSqr = - b2Settings.b2_linearSleepTolerance * b2Settings.b2_linearSleepTolerance; + b2Settings.b2_linearSleepTolerance * + b2Settings.b2_linearSleepTolerance; var angTolSqr = b2Settings.b2_angularSleepTolerance * b2Settings.b2_angularSleepTolerance; @@ -6145,7 +6249,8 @@ } var k_toiBaumgarte = 0.75; for (i = 0; i < subStep.positionIterations; ++i) { - var contactsOkay = contactSolver.SolvePositionConstraints(k_toiBaumgarte); + var contactsOkay = + contactSolver.SolvePositionConstraints(k_toiBaumgarte); var jointsOkay = true; for (j = 0; j < this.m_jointCount; ++j) { var jointOkay = this.m_joints[j].SolvePositionConstraints( @@ -6870,7 +6975,8 @@ bA = fA.m_body; bB = fB.m_body; if ( - (bA.GetType() != b2Body.b2_dynamicBody || bA.IsAwake() == false) && + (bA.GetType() != b2Body.b2_dynamicBody || + bA.IsAwake() == false) && (bB.GetType() != b2Body.b2_dynamicBody || bB.IsAwake() == false) ) { continue; @@ -7113,7 +7219,8 @@ b2CircleContact = Box2D.Dynamics.Contacts.b2CircleContact, b2Contact = Box2D.Dynamics.Contacts.b2Contact, b2ContactConstraint = Box2D.Dynamics.Contacts.b2ContactConstraint, - b2ContactConstraintPoint = Box2D.Dynamics.Contacts.b2ContactConstraintPoint, + b2ContactConstraintPoint = + Box2D.Dynamics.Contacts.b2ContactConstraintPoint, b2ContactEdge = Box2D.Dynamics.Contacts.b2ContactEdge, b2ContactFactory = Box2D.Dynamics.Contacts.b2ContactFactory, b2ContactRegister = Box2D.Dynamics.Contacts.b2ContactRegister, @@ -7124,7 +7231,8 @@ b2PolyAndCircleContact = Box2D.Dynamics.Contacts.b2PolyAndCircleContact, b2PolyAndEdgeContact = Box2D.Dynamics.Contacts.b2PolyAndEdgeContact, b2PolygonContact = Box2D.Dynamics.Contacts.b2PolygonContact, - b2PositionSolverManifold = Box2D.Dynamics.Contacts.b2PositionSolverManifold, + b2PositionSolverManifold = + Box2D.Dynamics.Contacts.b2PositionSolverManifold, b2Body = Box2D.Dynamics.b2Body, b2BodyDef = Box2D.Dynamics.b2BodyDef, b2ContactFilter = Box2D.Dynamics.b2ContactFilter, @@ -7237,7 +7345,8 @@ }; b2Contact.prototype.IsContinuous = function () { return ( - (this.m_flags & b2Contact.e_continuousFlag) == b2Contact.e_continuousFlag + (this.m_flags & b2Contact.e_continuousFlag) == + b2Contact.e_continuousFlag ); }; b2Contact.prototype.SetSensor = function (sensor) { @@ -7258,7 +7367,9 @@ } }; b2Contact.prototype.IsEnabled = function () { - return (this.m_flags & b2Contact.e_enabledFlag) == b2Contact.e_enabledFlag; + return ( + (this.m_flags & b2Contact.e_enabledFlag) == b2Contact.e_enabledFlag + ); }; b2Contact.prototype.GetNext = function () { return this.m_next; @@ -7550,7 +7661,8 @@ var tMat; this.m_constraintCount = contactCount; while (this.m_constraints.length < this.m_constraintCount) { - this.m_constraints[this.m_constraints.length] = new b2ContactConstraint(); + this.m_constraints[this.m_constraints.length] = + new b2ContactConstraint(); } for (i = 0; i < contactCount; ++i) { contact = contacts[i]; @@ -7629,7 +7741,8 @@ var kEqualized = bodyA.m_mass * bodyA.m_invMass + bodyB.m_mass * bodyB.m_invMass; kEqualized += - bodyA.m_mass * bodyA.m_invI * rnA + bodyB.m_mass * bodyB.m_invI * rnB; + bodyA.m_mass * bodyA.m_invI * rnA + + bodyB.m_mass * bodyB.m_invI * rnB; ccp.equalizedMass = 1.0 / kEqualized; var tangentX = normalY; var tangentY = -normalX; @@ -7704,8 +7817,10 @@ var ccp = c.points[j]; ccp.normalImpulse *= step.dtRatio; ccp.tangentImpulse *= step.dtRatio; - var PX = ccp.normalImpulse * normalX + ccp.tangentImpulse * tangentX; - var PY = ccp.normalImpulse * normalY + ccp.tangentImpulse * tangentY; + var PX = + ccp.normalImpulse * normalX + ccp.tangentImpulse * tangentX; + var PY = + ccp.normalImpulse * normalY + ccp.tangentImpulse * tangentY; bodyA.m_angularVelocity -= invIA * (ccp.rA.x * PY - ccp.rA.y * PX); bodyA.m_linearVelocity.x -= invMassA * PX; bodyA.m_linearVelocity.y -= invMassA * PY; @@ -7988,7 +8103,8 @@ var rAY = point.y - bodyA.m_sweep.c.y; var rBX = point.x - bodyB.m_sweep.c.x; var rBY = point.y - bodyB.m_sweep.c.y; - minSeparation = minSeparation < separation ? minSeparation : separation; + minSeparation = + minSeparation < separation ? minSeparation : separation; var C = b2Math.Clamp( baumgarte * (separation + b2Settings.b2_linearSlop), -b2Settings.b2_maxLinearCorrection, @@ -8051,7 +8167,8 @@ xf2 ) {}; Box2D.inherit(b2NullContact, Box2D.Dynamics.Contacts.b2Contact); - b2NullContact.prototype.__super = Box2D.Dynamics.Contacts.b2Contact.prototype; + b2NullContact.prototype.__super = + Box2D.Dynamics.Contacts.b2Contact.prototype; b2NullContact.b2NullContact = function () { Box2D.Dynamics.Contacts.b2Contact.b2Contact.apply(this, arguments); }; @@ -8157,7 +8274,9 @@ b2PositionSolverManifold.b2PositionSolverManifold = function () {}; b2PositionSolverManifold.prototype.b2PositionSolverManifold = function () { this.m_normal = new b2Vec2(); - this.m_separations = new Vector_a2j_Number(b2Settings.b2_maxManifoldPoints); + this.m_separations = new Vector_a2j_Number( + b2Settings.b2_maxManifoldPoints + ); this.m_points = new Vector(b2Settings.b2_maxManifoldPoints); for (var i = 0; i < b2Settings.b2_maxManifoldPoints; i++) { this.m_points[i] = new b2Vec2(); @@ -8324,11 +8443,17 @@ b2TensorDampingController = Box2D.Dynamics.Controllers.b2TensorDampingController; - Box2D.inherit(b2BuoyancyController, Box2D.Dynamics.Controllers.b2Controller); + Box2D.inherit( + b2BuoyancyController, + Box2D.Dynamics.Controllers.b2Controller + ); b2BuoyancyController.prototype.__super = Box2D.Dynamics.Controllers.b2Controller.prototype; b2BuoyancyController.b2BuoyancyController = function () { - Box2D.Dynamics.Controllers.b2Controller.b2Controller.apply(this, arguments); + Box2D.Dynamics.Controllers.b2Controller.b2Controller.apply( + this, + arguments + ); this.normal = new b2Vec2(0, -1); this.offset = 0; this.density = 0; @@ -8418,7 +8543,10 @@ b2ConstantAccelController.prototype.__super = Box2D.Dynamics.Controllers.b2Controller.prototype; b2ConstantAccelController.b2ConstantAccelController = function () { - Box2D.Dynamics.Controllers.b2Controller.b2Controller.apply(this, arguments); + Box2D.Dynamics.Controllers.b2Controller.b2Controller.apply( + this, + arguments + ); this.A = new b2Vec2(0, 0); }; b2ConstantAccelController.prototype.Step = function (step) { @@ -8441,7 +8569,10 @@ b2ConstantForceController.prototype.__super = Box2D.Dynamics.Controllers.b2Controller.prototype; b2ConstantForceController.b2ConstantForceController = function () { - Box2D.Dynamics.Controllers.b2Controller.b2Controller.apply(this, arguments); + Box2D.Dynamics.Controllers.b2Controller.b2Controller.apply( + this, + arguments + ); this.F = new b2Vec2(0, 0); }; b2ConstantForceController.prototype.Step = function (step) { @@ -8501,7 +8632,10 @@ b2GravityController.prototype.__super = Box2D.Dynamics.Controllers.b2Controller.prototype; b2GravityController.b2GravityController = function () { - Box2D.Dynamics.Controllers.b2Controller.b2Controller.apply(this, arguments); + Box2D.Dynamics.Controllers.b2Controller.b2Controller.apply( + this, + arguments + ); this.G = 1; this.invSqr = true; }; @@ -8564,7 +8698,10 @@ b2TensorDampingController.prototype.__super = Box2D.Dynamics.Controllers.b2Controller.prototype; b2TensorDampingController.b2TensorDampingController = function () { - Box2D.Dynamics.Controllers.b2Controller.b2Controller.apply(this, arguments); + Box2D.Dynamics.Controllers.b2Controller.b2Controller.apply( + this, + arguments + ); this.T = new b2Mat22(); this.maxTimestep = 0; }; @@ -9216,7 +9353,9 @@ this.m_J.linearB.Set(-this.m_ratio * ugX, -this.m_ratio * ugY); this.m_J.angularB = -this.m_ratio * crug; K += - this.m_ratio * this.m_ratio * (bB.m_invMass + bB.m_invI * crug * crug); + this.m_ratio * + this.m_ratio * + (bB.m_invMass + bB.m_invI * crug * crug); } this.m_mass = K > 0.0 ? 1.0 / K : 0.0; if (step.warmStarting) { @@ -9282,7 +9421,8 @@ return linearError < b2Settings.b2_linearSlop; }; Box2D.inherit(b2GearJointDef, Box2D.Dynamics.Joints.b2JointDef); - b2GearJointDef.prototype.__super = Box2D.Dynamics.Joints.b2JointDef.prototype; + b2GearJointDef.prototype.__super = + Box2D.Dynamics.Joints.b2JointDef.prototype; b2GearJointDef.b2GearJointDef = function () { Box2D.Dynamics.Joints.b2JointDef.b2JointDef.apply(this, arguments); }; @@ -9375,7 +9515,9 @@ break; case b2Joint.e_mouseJoint: { - joint = new b2MouseJoint(def instanceof b2MouseJointDef ? def : null); + joint = new b2MouseJoint( + def instanceof b2MouseJointDef ? def : null + ); } break; case b2Joint.e_prismaticJoint: @@ -9673,7 +9815,8 @@ var i2 = this.m_invIB; this.m_K.col1.x = m1 + m2 + i1 * this.m_s1 * this.m_s1 + i2 * this.m_s2 * this.m_s2; - this.m_K.col1.y = i1 * this.m_s1 * this.m_a1 + i2 * this.m_s2 * this.m_a2; + this.m_K.col1.y = + i1 * this.m_s1 * this.m_a1 + i2 * this.m_s2 * this.m_a2; this.m_K.col2.x = this.m_K.col1.y; this.m_K.col2.y = m1 + m2 + i1 * this.m_a1 * this.m_a1 + i2 * this.m_a2 * this.m_a2; @@ -9916,7 +10059,8 @@ i2 = this.m_invIB; this.m_K.col1.x = m1 + m2 + i1 * this.m_s1 * this.m_s1 + i2 * this.m_s2 * this.m_s2; - this.m_K.col1.y = i1 * this.m_s1 * this.m_a1 + i2 * this.m_s2 * this.m_a2; + this.m_K.col1.y = + i1 * this.m_s1 * this.m_a1 + i2 * this.m_s2 * this.m_a2; this.m_K.col2.x = this.m_K.col1.y; this.m_K.col2.y = m1 + m2 + i1 * this.m_a1 * this.m_a1 + i2 * this.m_a2 * this.m_a2; @@ -9957,7 +10101,8 @@ ); }; Box2D.inherit(b2LineJointDef, Box2D.Dynamics.Joints.b2JointDef); - b2LineJointDef.prototype.__super = Box2D.Dynamics.Joints.b2JointDef.prototype; + b2LineJointDef.prototype.__super = + Box2D.Dynamics.Joints.b2JointDef.prototype; b2LineJointDef.b2LineJointDef = function () { Box2D.Dynamics.Joints.b2JointDef.b2JointDef.apply(this, arguments); this.localAnchorA = new b2Vec2(); @@ -10146,7 +10291,8 @@ this.dampingRatio = 0.7; }; Box2D.inherit(b2PrismaticJoint, Box2D.Dynamics.Joints.b2Joint); - b2PrismaticJoint.prototype.__super = Box2D.Dynamics.Joints.b2Joint.prototype; + b2PrismaticJoint.prototype.__super = + Box2D.Dynamics.Joints.b2Joint.prototype; b2PrismaticJoint.b2PrismaticJoint = function () { Box2D.Dynamics.Joints.b2Joint.b2Joint.apply(this, arguments); this.m_localAnchor1 = new b2Vec2(); @@ -10347,7 +10493,8 @@ this.m_K.col1.x = m1 + m2 + i1 * this.m_s1 * this.m_s1 + i2 * this.m_s2 * this.m_s2; this.m_K.col1.y = i1 * this.m_s1 + i2 * this.m_s2; - this.m_K.col1.z = i1 * this.m_s1 * this.m_a1 + i2 * this.m_s2 * this.m_a2; + this.m_K.col1.z = + i1 * this.m_s1 * this.m_a1 + i2 * this.m_s2 * this.m_a2; this.m_K.col2.x = this.m_K.col1.y; this.m_K.col2.y = i1 + i2; this.m_K.col2.z = i1 * this.m_a1 + i2 * this.m_a2; @@ -10595,7 +10742,8 @@ this.m_K.col1.x = m1 + m2 + i1 * this.m_s1 * this.m_s1 + i2 * this.m_s2 * this.m_s2; this.m_K.col1.y = i1 * this.m_s1 + i2 * this.m_s2; - this.m_K.col1.z = i1 * this.m_s1 * this.m_a1 + i2 * this.m_s2 * this.m_a2; + this.m_K.col1.z = + i1 * this.m_s1 * this.m_a1 + i2 * this.m_s2 * this.m_a2; this.m_K.col2.x = this.m_K.col1.y; this.m_K.col2.y = i1 + i2; this.m_K.col2.z = i1 * this.m_a1 + i2 * this.m_a2; @@ -11370,11 +11518,13 @@ v1.x -= m1 * this.impulse3.x; v1.y -= m1 * this.impulse3.y; w1 -= - i1 * (r1X * this.impulse3.y - r1Y * this.impulse3.x + this.impulse3.z); + i1 * + (r1X * this.impulse3.y - r1Y * this.impulse3.x + this.impulse3.z); v2.x += m2 * this.impulse3.x; v2.y += m2 * this.impulse3.y; w2 += - i2 * (r2X * this.impulse3.y - r2Y * this.impulse3.x + this.impulse3.z); + i2 * + (r2X * this.impulse3.y - r2Y * this.impulse3.x + this.impulse3.z); } else { tMat = bA.m_xf.R; r1X = this.m_localAnchor1.x - bA.m_sweep.localCenter.x; @@ -11729,7 +11879,8 @@ ); }; Box2D.inherit(b2WeldJointDef, Box2D.Dynamics.Joints.b2JointDef); - b2WeldJointDef.prototype.__super = Box2D.Dynamics.Joints.b2JointDef.prototype; + b2WeldJointDef.prototype.__super = + Box2D.Dynamics.Joints.b2JointDef.prototype; b2WeldJointDef.b2WeldJointDef = function () { Box2D.Dynamics.Joints.b2JointDef.b2JointDef.apply(this, arguments); this.localAnchorA = new b2Vec2(); @@ -11844,7 +11995,11 @@ b2DebugDraw.prototype.GetXFormScale = function () { return this.m_xformScale; }; - b2DebugDraw.prototype.DrawPolygon = function (vertices, vertexCount, color) { + b2DebugDraw.prototype.DrawPolygon = function ( + vertices, + vertexCount, + color + ) { if (!vertexCount) return; var s = this.m_ctx; var drawScale = this.m_drawScale; @@ -12003,16 +12158,16 @@ const prevPos = {}; /** - * Active b2Body/s in the world. - * @type {Object.} - */ + * Active b2Body/s in the world. + * @type {Object.} + */ const bodies = {}; // const joints = {}; const pinned = {}; // Map of IDs to pinned joints /** - * The runtime instantiating this block package. - * @type {Array} - */ + * The runtime instantiating this block package. + * @type {Array} + */ const stageBodies = []; // const categorySeq = 1; @@ -12208,12 +12363,12 @@ }; /** - * Set the X and Y coordinates (No Fencing) - * @param {!RenderedTarget} rt the renderedTarget. - * @param {!number} x New X coordinate, in Scratch coordinates. - * @param {!number} y New Y coordinate, in Scratch coordinates. - * @param {?boolean} force Force setting X/Y, in case of dragging - */ + * Set the X and Y coordinates (No Fencing) + * @param {!RenderedTarget} rt the renderedTarget. + * @param {!number} x New X coordinate, in Scratch coordinates. + * @param {!number} y New Y coordinate, in Scratch coordinates. + * @param {?boolean} force Force setting X/Y, in case of dragging + */ const _setXY = function (rt, x, y, force) { if (rt.isStage) return; if (rt.dragging && !force) return; @@ -12291,8 +12446,10 @@ } }; - const blockIconURI = ""; - const menuIconURI = ""; + const blockIconURI = + ""; + const menuIconURI = + ""; const vm = Scratch.vm; class Scratch3Griffpatch { @@ -12354,7 +12511,7 @@ default: "Physics", description: "Label for the Griffpatch extension category", }), - docsURI: "https://extensions.turbowarp.org/box2d.html", + docsURI: "https://extensions.turbowarp.org/box2d", menuIconURI: menuIconURI, blockIconURI: blockIconURI, blocks: [ @@ -12609,7 +12766,7 @@ filter: [Scratch.TargetType.SPRITE], }, - '---', + "---", { opcode: "setAngVelocity", @@ -12637,7 +12794,7 @@ }), filter: [Scratch.TargetType.SPRITE], }, - + "---", { @@ -12662,10 +12819,10 @@ text: formatMessage({ id: "griffpatch.getStatic", default: "fixed?", - description: "get whether this sprite is fixed" + description: "get whether this sprite is fixed", }), blockType: BlockType.BOOLEAN, - filter: [Scratch.TargetType.SPRITE] + filter: [Scratch.TargetType.SPRITE], }, "---", @@ -12676,7 +12833,7 @@ text: formatMessage({ id: "griffpatch.setDensity", default: "set density [density]", - description: "Set the density of the object" + description: "Set the density of the object", }), arguments: { density: { @@ -12694,13 +12851,13 @@ text: formatMessage({ id: "griffpatch.setDensityValue", default: "set density to [density]", - description: "Set the density of the object" + description: "Set the density of the object", }), arguments: { density: { type: ArgumentType.NUMBER, - defaultValue: 100 - } + defaultValue: 100, + }, }, filter: [Scratch.TargetType.SPRITE], }, @@ -12816,7 +12973,8 @@ blockType: BlockType.COMMAND, text: formatMessage({ id: "griffpatch.setProperties", - default: "set density [density] roughness [friction] bounce [restitution]", + default: + "set density [density] roughness [friction] bounce [restitution]", description: "Set the density of the object", }), arguments: { @@ -12938,7 +13096,7 @@ description: "get the y scroll", }), blockType: BlockType.REPORTER, - } + }, ], menus: { @@ -13104,7 +13262,8 @@ } const prev = prevPos[targetID]; - const fixedRotation = target.rotationStyle !== ROTATION_STYLE_ALL_AROUND; + const fixedRotation = + target.rotationStyle !== ROTATION_STYLE_ALL_AROUND; if (prev && (prev.x !== target.x || prev.y !== target.y)) { const pos = new b2Vec2( @@ -13350,7 +13509,7 @@ body.GetFixtureList().SetDensity(Cast.toNumber(args.density) / 100.0); body.ResetMassData(); } - + getDensity(args, util) { let body = bodies[util.target.id]; if (!body) { @@ -13385,7 +13544,9 @@ body = this.setPhysicsFor(util.target); } - body.GetFixtureList().SetRestitution(Cast.toNumber(args.restitution) / 100.0); + body + .GetFixtureList() + .SetRestitution(Cast.toNumber(args.restitution) / 100.0); body.ResetMassData(); } @@ -13406,7 +13567,9 @@ body.GetFixtureList().SetDensity(Cast.toNumber(args.density) / 100.0); body.GetFixtureList().SetFriction(Cast.toNumber(args.friction) / 100.0); - body.GetFixtureList().SetRestitution(Cast.toNumber(args.restitution) / 100.0); + body + .GetFixtureList() + .SetRestitution(Cast.toNumber(args.restitution) / 100.0); body.ResetMassData(); } @@ -13695,7 +13858,11 @@ position.x * zoom - _scroll.x, position.y * zoom - _scroll.y ); - prevPos[targetID] = { x: target.x, y: target.y, dir: target.direction }; + prevPos[targetID] = { + x: target.x, + y: target.y, + dir: target.direction, + }; } } } diff --git a/extensions/clipboard.js b/extensions/clipboard.js index 7064a0194f..07f1752857 100644 --- a/extensions/clipboard.js +++ b/extensions/clipboard.js @@ -1,118 +1,123 @@ -/*! - * Copyright 2023 tomyo-code + AdamMady - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -(function(Scratch) { - 'use strict'; - - if (!Scratch.extensions.unsandboxed) { - throw new Error('Clipboard must run unsandboxed'); - } - - const extensionicon = ""; - - let lastPastedText = ''; - - window.addEventListener('copy', (event) => { - Scratch.vm.runtime.startHats('clipboard_whenCopied') ; - }); - window.addEventListener('paste', (event) => { - Scratch.vm.runtime.startHats('clipboard_whenPasted'); - const clipboardData = event.clipboardData || window.clipboardData; - const pastedText = clipboardData.getData('Text'); - lastPastedText = pastedText; - }); - - class Clipboard { - getInfo() { - return { - id: 'clipboard', - name: 'Clipboard', - blockIconURI: extensionicon, - color1: '#008080', - color2: '#006666', - blocks: [ - { - opcode: 'whenCopied', - blockType: Scratch.BlockType.HAT, - text: 'when something is copied', - isEdgeActivated: false - }, - { - opcode: 'whenPasted', - blockType: Scratch.BlockType.HAT, - text: 'when something is pasted', - isEdgeActivated: false - }, - '---', - { - opcode: 'setClipboard', - blockType: Scratch.BlockType.COMMAND, - text: 'copy to clipboard: [TEXT]', - arguments: { - TEXT: { - type: Scratch.ArgumentType.STRING - } - } - }, - { - opcode: 'resetClipboard', - blockType: Scratch.BlockType.COMMAND, - text: 'reset clipboard' - }, - '---', - { - opcode: 'clipboard', - blockType: Scratch.BlockType.REPORTER, - text: 'clipboard', - disableMonitor: true - }, - { - opcode: 'getLastPastedText', - blockType: Scratch.BlockType.REPORTER, - text: 'last pasted text', - disableMonitor: true - } - ], - }; - } - - setClipboard(args) { - navigator.clipboard.writeText(args.TEXT); - } - - resetClipboard() { - navigator.clipboard.writeText(''); - } - - clipboard() { - if (navigator.clipboard && navigator.clipboard.readText) { - return Scratch.canReadClipboard().then(allowed => { - if (allowed) { - return navigator.clipboard.readText(); - } - return ''; - }); - } - return ''; - } - - getLastPastedText() { - return lastPastedText; - } - } - - Scratch.extensions.register(new Clipboard()); -})(Scratch); +// Name: Clipboard +// ID: clipboard +// Description: Read and write from the system clipboard. + +/*! + * Copyright 2023 tomyo-code + AdamMady + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +(function (Scratch) { + "use strict"; + + if (!Scratch.extensions.unsandboxed) { + throw new Error("Clipboard must run unsandboxed"); + } + + const extensionicon = + ""; + + let lastPastedText = ""; + + window.addEventListener("copy", (event) => { + Scratch.vm.runtime.startHats("clipboard_whenCopied"); + }); + window.addEventListener("paste", (event) => { + Scratch.vm.runtime.startHats("clipboard_whenPasted"); + const clipboardData = event.clipboardData || window.clipboardData; + const pastedText = clipboardData.getData("Text"); + lastPastedText = pastedText; + }); + + class Clipboard { + getInfo() { + return { + id: "clipboard", + name: "Clipboard", + blockIconURI: extensionicon, + color1: "#008080", + color2: "#006666", + blocks: [ + { + opcode: "whenCopied", + blockType: Scratch.BlockType.EVENT, + text: "when something is copied", + isEdgeActivated: false, + }, + { + opcode: "whenPasted", + blockType: Scratch.BlockType.EVENT, + text: "when something is pasted", + isEdgeActivated: false, + }, + "---", + { + opcode: "setClipboard", + blockType: Scratch.BlockType.COMMAND, + text: "copy to clipboard: [TEXT]", + arguments: { + TEXT: { + type: Scratch.ArgumentType.STRING, + }, + }, + }, + { + opcode: "resetClipboard", + blockType: Scratch.BlockType.COMMAND, + text: "reset clipboard", + }, + "---", + { + opcode: "clipboard", + blockType: Scratch.BlockType.REPORTER, + text: "clipboard", + disableMonitor: true, + }, + { + opcode: "getLastPastedText", + blockType: Scratch.BlockType.REPORTER, + text: "last pasted text", + disableMonitor: true, + }, + ], + }; + } + + setClipboard(args) { + navigator.clipboard.writeText(args.TEXT); + } + + resetClipboard() { + navigator.clipboard.writeText(""); + } + + clipboard() { + if (navigator.clipboard && navigator.clipboard.readText) { + return Scratch.canReadClipboard().then((allowed) => { + if (allowed) { + return navigator.clipboard.readText(); + } + return ""; + }); + } + return ""; + } + + getLastPastedText() { + return lastPastedText; + } + } + + Scratch.extensions.register(new Clipboard()); +})(Scratch); diff --git a/extensions/clouddata-ping.js b/extensions/clouddata-ping.js index 0947f40315..93a130de1e 100644 --- a/extensions/clouddata-ping.js +++ b/extensions/clouddata-ping.js @@ -1,3 +1,8 @@ +// Name: Ping Cloud Data +// ID: clouddataping +// Description: Determine whether a cloud variable server is probably up. +// Original: TheShovel + (function (Scratch) { "use strict"; @@ -17,10 +22,10 @@ * @returns {Promise} */ const pingWebSocket = async (uri) => { - if (!await Scratch.canFetch(uri)) { + if (!(await Scratch.canFetch(uri))) { return { expires: 0, - value: false + value: false, }; } @@ -33,7 +38,7 @@ } catch (e) { return { expires: 0, - value: false + value: false, }; } @@ -60,7 +65,7 @@ return { expires: Date.now() + 60000, - value: isUp + value: isUp, }; }; diff --git a/extensions/cloudlink.js b/extensions/cloudlink.js index 03ac77a836..1583cc1e23 100644 --- a/extensions/cloudlink.js +++ b/extensions/cloudlink.js @@ -1,1712 +1,1838 @@ +// Name: Cloudlink +// ID: cloudlink +// Description: Powerful WebSocket extension for Scratch 3. +// By: MikeDEV + // Copy of S4-0_nosuite.js as of 10/31/2022 /* eslint-disable */ -(function(Scratch) { - -var servers = {}; ; // Server list -let mWS = null; - -// Get the server URL list -try { - Scratch.fetch('https://mikedev101.github.io/cloudlink/serverlist.json').then(response => { - return response.text(); - }).then(data => { - servers = JSON.parse(data); - }).catch(err => { - console.log(err); - servers = {}; - }); -} catch(err) { - console.log(err); - servers = {}; -}; - -function find_id(ID, ulist) { - // Thanks StackOverflow! - if (jsonCheck(ID) && (!intCheck(ID))) { - return ulist.some(o => ((o.username === JSON.parse(ID).username) && (o.id == JSON.parse(ID).id))); - } else { - return ulist.some(o => ((o.username === String(ID)) || (o.id == ID))); - }; -} - -function jsonCheck(JSON_STRING) { - try { - JSON.parse(JSON_STRING); - return true; - } catch (err) { - return false; - } -} - -function intCheck(value) { - return !isNaN(value); -} - -function autoConvert(value) { - // Check if the value is JSON / Dict first - try { - JSON.parse(value); - return JSON.parse(value); - } catch (err) {}; - - // Check if the value is an array - try { - tmp = value; - tmp = tmp.replace(/'/g, '"'); - JSON.parse(tmp); - return JSON.parse(tmp); - } catch (err) {}; - - // Check if an int/float - if (!isNaN(value)) { - return Number(value); - }; - - // Leave as the original value if none of the above work - return value; -} - -class CloudLink { - constructor (runtime, extensionId) { - // Extension stuff - this.runtime = runtime; - this.cl_icon = ''; - this.cl_block = ''; - - // Socket data - this.socketData = { - "gmsg": [], - "pmsg": [], - "direct": [], - "statuscode": [], - "gvar": [], - "pvar": [], - "motd": "", - "client_ip": "", - "ulist": [], - "server_version": "" - }; - this.varData = { - "gvar": {}, - "pvar": {} - }; - - this.queueableCmds = ["gmsg", "pmsg", "gvar", "pvar", "direct", "statuscode"]; - this.varCmds = ["gvar", "pvar"]; - - // Listeners - this.socketListeners = {}; - this.socketListenersData = {}; - this.newSocketData = { - "gmsg": false, - "pmsg": false, - "direct": false, - "statuscode": false, - "gvar": false, - "pvar": false - }; - - // Edge-triggered hat blocks - this.connect_hat = 0; - this.packet_hat = 0; - this.close_hat = 0; - - // Status stuff - this.isRunning = false; - this.isLinked = false; - this.version = "S4.0"; - this.link_status = 0; - this.username = ""; - this.tmp_username = ""; - this.isUsernameSyncing = false; - this.isUsernameSet = false; - this.disconnectWasClean = false; - this.wasConnectionDropped = false; - this.didConnectionFail = false; - this.protocolOk = false; - - // Listeners stuff - this.enableListener = false; - this.setListener = ""; - - // Rooms stuff - this.enableRoom = false; - this.isRoomSetting = false; - this.selectRoom = ""; - - // Remapping stuff - this.menuRemap = { - "Global data": "gmsg", - "Private data": "pmsg", - "Global variables": "gvar", - "Private variables": "pvar", - "Direct data": "direct", - "Status code": "statuscode", - "All data": "all" - }; - } - - getInfo () { - return { - "id": 'cloudlink', - "name": 'CloudLink', - "blockIconURI": this.cl_block, - "menuIconURI": this.cl_icon, - "blocks": [ - { - "opcode": 'returnGlobalData', - "blockType": "reporter", - "text": "Global data" - }, - { - "opcode": 'returnPrivateData', - "blockType": "reporter", - "text": "Private data" - }, - { - "opcode": 'returnDirectData', - "blockType": "reporter", - "text": "Direct Data" - }, - { - "opcode": 'returnLinkData', - "blockType": "reporter", - "text": "Link status" - }, - { - "opcode": 'returnStatusCode', - "blockType": "reporter", - "text": "Status code" - }, - { - "opcode": 'returnUserListData', - "blockType": "reporter", - "text": "Usernames" - }, - { - "opcode": "returnUsernameData", - "blockType": "reporter", - "text": "My username" - }, - { - "opcode": "returnVersionData", - "blockType": "reporter", - "text": "Extension version" - }, - { - "opcode": "returnServerVersion", - "blockType": "reporter", - "text": "Server version" - }, - { - "opcode": "returnServerList", - "blockType": "reporter", - "text": "Server list" - }, - { - "opcode": "returnMOTD", - "blockType": "reporter", - "text": "Server MOTD" - }, - { - "opcode": "returnClientIP", - "blockType": "reporter", - "text": "My IP address" - }, - { - "opcode": 'returnListenerData', - "blockType": "reporter", - "text": "Response for listener [ID]", - "arguments": { - "ID": { - "type": "string", - "defaultValue": "example-listener", - }, - }, - }, - { - "opcode": "readQueueSize", - "blockType": "reporter", - "text": "Size of queue for [TYPE]", - "arguments": { - "TYPE": { - "type": "string", - "menu": "allmenu", - "defaultValue": "All data", - }, - }, - }, - { - "opcode": "readQueueData", - "blockType": "reporter", - "text": "Packet queue for [TYPE]", - "arguments": { - "TYPE": { - "type": "string", - "menu": "allmenu", - "defaultValue": "All data", - }, - }, - }, - { - "opcode": 'returnVarData', - "blockType": "reporter", - "text": "[TYPE] [VAR] data", - "arguments": { - "VAR": { - "type": "string", - "defaultValue": "Apple", - }, - "TYPE": { - "type": "string", - "menu": "varmenu", - "defaultValue": "Global variables", - }, - }, - }, - { - "opcode": 'parseJSON', - "blockType": "reporter", - "text": '[PATH] of [JSON_STRING]', - "arguments": { - "PATH": { - "type": "string", - "defaultValue": 'fruit/apples', - }, - "JSON_STRING": { - "type": "string", - "defaultValue": '{"fruit": {"apples": 2, "bananas": 3}, "total_fruit": 5}', - }, - }, - }, - { - "opcode": 'getFromJSONArray', - "blockType": "reporter", - "text": 'Get [NUM] from JSON array [ARRAY]', - "arguments": { - "NUM": { - "type": "number", - "defaultValue": 0, - }, - "ARRAY": { - "type": "string", - "defaultValue": '["foo","bar"]', - } - } - }, - { - "opcode": 'fetchURL', - "blockType": "reporter", - "blockAllThreads": "true", - "text": "Fetch data from URL [url]", - "arguments": { - "url": { - "type": "string", - "defaultValue": "https://mikedev101.github.io/cloudlink/fetch_test", - }, - }, - }, - { - "opcode": 'requestURL', - "blockType": "reporter", - "blockAllThreads": "true", - "text": 'Send request with method [method] for URL [url] with data [data] and headers [headers]', - "arguments": { - "method": { - "type": "string", - "defaultValue": 'GET', - }, - "url": { - "type": "string", - "defaultValue": 'https://mikedev101.github.io/cloudlink/fetch_test', - }, - "data": { - "type": "string", - "defaultValue": '{}' - }, - "headers": { - "type": "string", - "defaultValue": '{}' - }, - } - }, - { - "opcode": 'makeJSON', - "blockType": "reporter", - "text": 'Convert [toBeJSONified] to JSON', - "arguments": { - "toBeJSONified": { - "type": "string", - "defaultValue": '{"test": true}', - }, - } - }, - { - "opcode": 'onConnect', - "blockType": "hat", - "text": 'When connected', - "blockAllThreads": "true" - }, - { - "opcode": 'onClose', - "blockType": "hat", - "text": 'When disconnected', - "blockAllThreads": "true" - }, - { - "opcode": 'onListener', - "blockType": "hat", - "text": 'When I receive new packet with listener [ID]', - "blockAllThreads": "true", - "arguments": { - "ID": { - "type": "string", - "defaultValue": "example-listener", - }, - }, - }, - { - "opcode": 'onNewPacket', - "blockType": "hat", - "text": 'When I receive new [TYPE] packet', - "blockAllThreads": "true", - "arguments": { - "TYPE": { - "type": "string", - "menu": "almostallmenu", - "defaultValue": 'Global data' - }, - }, - }, - { - "opcode": 'onNewVar', - "blockType": "hat", - "text": 'When I receive new [TYPE] data for [VAR]', - "blockAllThreads": "true", - "arguments": { - "TYPE": { - "type": "string", - "menu": "varmenu", - "defaultValue": 'Global variables', - }, - "VAR": { - "type": "string", - "defaultValue": 'Apple', - }, - }, - }, - { - "opcode": 'getComState', - "blockType": "Boolean", - "text": 'Connected?', - }, - { - "opcode": 'getRoomState', - "blockType": "Boolean", - "text": 'Linked to rooms?', - }, - { - "opcode": 'getComLostConnectionState', - "blockType": "Boolean", - "text": 'Lost connection?', - }, - { - "opcode": 'getComFailedConnectionState', - "blockType": "Boolean", - "text": 'Failed to connnect?', - }, - { - "opcode": 'getUsernameState', - "blockType": "Boolean", - "text": 'Username synced?', - }, - { - "opcode": 'returnIsNewData', - "blockType": "Boolean", - "text": 'Got New [TYPE]?', - "arguments": { - "TYPE": { - "type": "string", - "menu": "datamenu", - "defaultValue": 'Global data', - }, - }, - }, - { - "opcode": 'returnIsNewVarData', - "blockType": "Boolean", - "text": 'Got New [TYPE] data for variable [VAR]?', - "arguments": { - "TYPE": { - "type": "string", - "menu": "varmenu", - "defaultValue": 'Global variables', - }, - "VAR": { - "type": "string", - "defaultValue": 'Apple', - }, - }, - }, - { - "opcode": 'returnIsNewListener', - "blockType": "Boolean", - "text": 'Got new packet with listener [ID]?', - "blockAllThreads": "true", - "arguments": { - "ID": { - "type": "string", - "defaultValue": "example-listener", - }, - }, - }, - { - "opcode": 'checkForID', - "blockType": "Boolean", - "text": 'ID [ID] connected?', - "arguments": { - "ID": { - "type": "string", - "defaultValue": 'Another name', - }, - }, - }, - { - "opcode": 'isValidJSON', - "blockType": "Boolean", - "text": 'Is [JSON_STRING] valid JSON?', - "arguments": { - "JSON_STRING": { - "type": "string", - "defaultValue": '{"fruit": {"apples": 2, "bananas": 3}, "total_fruit": 5}', - }, - }, - }, - { - "opcode": 'openSocket', - "blockType": "command", - "text": 'Connect to [IP]', - "blockAllThreads": "true", - "arguments": { - "IP": { - "type": "string", - "defaultValue": 'ws://127.0.0.1:3000/', - }, - }, - }, - { - "opcode": 'openSocketPublicServers', - "blockType": "command", - "text": 'Connect to server [ID]', - "blockAllThreads": "true", - "arguments": { - "ID": { - "type": "number", - "defaultValue": '', - }, - }, - }, - { - "opcode": 'closeSocket', - "blockType": "command", - "blockAllThreads": "true", - "text": 'Disconnect', - }, - { - "opcode": 'setMyName', - "blockType": "command", - "text": 'Set [NAME] as username', - "blockAllThreads": "true", - "arguments": { - "NAME": { - "type": "string", - "defaultValue": "A name", - }, - }, - }, - { - "opcode": 'createListener', - "blockType": "command", - "text": 'Attach listener [ID] to next packet', - "blockAllThreads": "true", - "arguments": { - "ID": { - "type": "string", - "defaultValue": "example-listener", - }, - }, - }, - { - "opcode": 'linkToRooms', - "blockType": "command", - "text": 'Link to room(s) [ROOMS]', - "blockAllThreads": "true", - "arguments": { - "ROOMS": { - "type": "string", - "defaultValue": '["test"]', - }, - } - }, - { - "opcode": 'selectRoomsInNextPacket', - "blockType": "command", - "text": 'Select room(s) [ROOMS] for next packet', - "blockAllThreads": "true", - "arguments": { - "ROOMS": { - "type": "string", - "defaultValue": '["test"]', - }, - }, - }, - { - "opcode": 'unlinkFromRooms', - "blockType": "command", - "text": 'Unlink from all rooms', - "blockAllThreads": "true" - }, - { - "opcode": 'sendGData', - "blockType": "command", - "text": 'Send [DATA]', - "blockAllThreads": "true", - "arguments": { - "DATA": { - "type": "string", - "defaultValue": 'Apple' - } - } - }, - { - "opcode": 'sendPData', - "blockType": "command", - "text": 'Send [DATA] to [ID]', - "blockAllThreads": "true", - "arguments": { - "DATA": { - "type": "string", - "defaultValue": 'Apple' - }, - "ID": { - "type": "string", - "defaultValue": 'Another name' - } - } - }, - { - "opcode": 'sendGDataAsVar', - "blockType": "command", - "text": 'Send variable [VAR] with data [DATA]', - "blockAllThreads": "true", - "arguments": { - "DATA": { - "type": "string", - "defaultValue": 'Banana' - }, - "VAR": { - "type": "string", - "defaultValue": 'Apple' - } - } - }, - { - "opcode": 'sendPDataAsVar', - "blockType": "command", - "text": 'Send variable [VAR] to [ID] with data [DATA]', - "blockAllThreads": "true", - "arguments": { - "DATA": { - "type": "string", - "defaultValue": 'Banana' - }, - "ID": { - "type": "string", - "defaultValue": 'Another name' - }, - "VAR": { - "type": "string", - "defaultValue": 'Apple' - } - } - }, - { - "opcode": 'runCMDnoID', - "blockType": "command", - "text": 'Send command without ID [CMD] [DATA]', - "blockAllThreads": "true", - "arguments": { - "CMD": { - "type": "string", - "defaultValue": 'direct' - }, - "DATA": { - "type": "string", - "defaultValue": 'val' - } - } - }, - { - "opcode": 'runCMD', - "blockType": "command", - "text": 'Send command [CMD] [ID] [DATA]', - "blockAllThreads": "true", - "arguments": { - "CMD": { - "type": "string", - "defaultValue": 'direct' - }, - "ID": { - "type": "string", - "defaultValue": 'id' - }, - "DATA": { - "type": "string", - "defaultValue": 'val' - } - } - }, - { - "opcode": 'resetNewData', - "blockType": "command", - "text": 'Reset got new [TYPE] status', - "blockAllThreads": "true", - "arguments": { - "TYPE": { - "type": "string", - "menu": "datamenu", - "defaultValue": 'Global data' - } - } - }, - { - "opcode": 'resetNewVarData', - "blockType": "command", - "text": 'Reset got new [TYPE] [VAR] status', - "blockAllThreads": "true", - "arguments": { - "TYPE": { - "type": "string", - "menu": "varmenu", - "defaultValue": 'Global variables' - }, - "VAR": { - "type": "string", - "defaultValue": 'Apple' - } - } - }, - { - "opcode": 'resetNewListener', - "blockType": "command", - "text": 'Reset got new [ID] listener status', - "blockAllThreads": "true", - "arguments": { - "ID": { - "type": "string", - "defaultValue": 'example-listener' - } - } - }, - { - "opcode": 'clearAllPackets', - "blockType": "command", - "text": "Clear all packets for [TYPE]", - "arguments": { - "TYPE": { - "type": "string", - "menu": "allmenu", - "defaultValue": "All data" - }, - }, - } - ], - "menus": { - "coms": { - "items": ["Connected", "Username synced"] - }, - "datamenu": { - "items": ['Global data', 'Private data', 'Direct data', 'Status code'] - }, - "varmenu": { - "items": ['Global variables', 'Private variables'] - }, - "allmenu": { - "items": ['Global data', 'Private data', 'Direct data', 'Status code', "Global variables", "Private variables", "All data"] - }, - "almostallmenu": { - "items": ['Global data', 'Private data', 'Direct data', 'Status code', "Global variables", "Private variables"] - }, - }, - }; - }; - - // Code for blocks go here - - returnGlobalData() { - if (this.socketData.gmsg.length != 0) { - - let data = (this.socketData.gmsg[this.socketData.gmsg.length - 1].val); - - if (typeof(data) == "object") { - data = JSON.stringify(data); // Make the JSON safe for Scratch - } - - return data; - } else { - return ""; - }; - }; - - returnPrivateData() { - if (this.socketData.pmsg.length != 0) { - let data = (this.socketData.pmsg[this.socketData.pmsg.length - 1].val); - - if (typeof (data) == "object") { - data = JSON.stringify(data); // Make the JSON safe for Scratch - } - - return data; - } else { - return ""; - }; - }; - - returnDirectData() { - if (this.socketData.direct.length != 0) { - let data = (this.socketData.direct[this.socketData.direct.length - 1].val); - - if (typeof (data) == "object") { - data = JSON.stringify(data); // Make the JSON safe for Scratch - } - - return data; - } else { - return ""; - }; - }; - - returnLinkData() { - return String(this.link_status); - }; - - returnStatusCode() { - if (this.socketData.statuscode.length != 0) { - let data = (this.socketData.statuscode[this.socketData.statuscode.length - 1].code); - - if (typeof (data) == "object") { - data = JSON.stringify(data); // Make the JSON safe for Scratch - } - - return data; - } else { - return ""; - }; - }; - - returnUserListData() { - return JSON.stringify(this.socketData.ulist); - }; - - returnUsernameData() { - let data = this.username; - - if (typeof (data) == "object") { - data = JSON.stringify(data); // Make the JSON safe for Scratch - } - - return data; - }; - - returnVersionData() { - return String(this.version); - }; - - returnServerVersion() { - return String(this.socketData.server_version); - }; - - returnServerList() { - return JSON.stringify(servers); - }; - - returnMOTD() { - return String(this.socketData.motd); - }; - - returnClientIP() { - return String(this.socketData.client_ip); - }; - - returnListenerData({ID}) { - const self = this; - if ((this.isRunning) && (this.socketListeners.hasOwnProperty(String(ID)))) { - return JSON.stringify(this.socketListenersData[ID]); - } else { - return "{}"; - }; - }; - - readQueueSize({TYPE}) { - if (this.menuRemap[String(TYPE)] == "all") { - let tmp_size = 0; - tmp_size = tmp_size + this.socketData.gmsg.length; - tmp_size = tmp_size + this.socketData.pmsg.length; - tmp_size = tmp_size + this.socketData.direct.length; - tmp_size = tmp_size + this.socketData.statuscode.length; - tmp_size = tmp_size + this.socketData.gvar.length; - tmp_size = tmp_size + this.socketData.pvar.length; - return tmp_size; - } else { - return this.socketData[this.menuRemap[String(TYPE)]].length; - }; - }; - - readQueueData({TYPE}) { - if (this.menuRemap[String(TYPE)] == "all") { - let tmp_socketData = JSON.parse(JSON.stringify(this.socketData)); // Deep copy - - delete tmp_socketData.motd; - delete tmp_socketData.client_ip; - delete tmp_socketData.ulist; - delete tmp_socketData.server_version; - - return JSON.stringify(tmp_socketData); - } else { - return JSON.stringify(this.socketData[this.menuRemap[String(TYPE)]]); - }; - }; - - returnVarData({ TYPE, VAR }) { - if (this.isRunning) { - if (this.varData.hasOwnProperty(this.menuRemap[TYPE])) { - if (this.varData[this.menuRemap[TYPE]].hasOwnProperty(VAR)) { - return this.varData[this.menuRemap[TYPE]][VAR].value; - } else { - return ""; - }; - } else { - return ""; - }; - } else { - return ""; - }; - }; - - parseJSON({PATH, JSON_STRING}) { - try { - const path = PATH.toString().split('/').map(prop => decodeURIComponent(prop)); - if (path[0] === '') path.splice(0, 1); - if (path[path.length - 1] === '') path.splice(-1, 1); - let json; - try { - json = JSON.parse(' ' + JSON_STRING); - } catch (e) { - return e.message; - }; - path.forEach(prop => json = json[prop]); - if (json === null) return 'null'; - else if (json === undefined) return ''; - else if (typeof json === 'object') return JSON.stringify(json); - else return json.toString(); - } catch (err) { - return ''; - }; - }; - - getFromJSONArray({NUM, ARRAY}) { - var json_array = JSON.parse(ARRAY); - if (json_array[NUM] == "undefined") { - return ""; - } else { - let data = json_array[NUM]; - - if (typeof (data) == "object") { - data = JSON.stringify(data); // Make the JSON safe for Scratch - } - - return data; - } - }; - - fetchURL(args) { - return Scratch.fetch(args.url, { - method: "GET" - }).then(response => response.text()); - }; - - requestURL(args) { - if (args.method == "GET" || args.method == "HEAD") { - return Scratch.fetch(args.url, { - method: args.method, - headers: JSON.parse(args.headers) - }).then(response => response.text()); - } else { - return Scratch.fetch(args.url, { - method: args.method, - headers: JSON.parse(args.headers), - body: JSON.parse(args.data) - }).then(response => response.text()); - } - }; - - isValidJSON({JSON_STRING}) { - return jsonCheck(JSON_STRING); - }; - - makeJSON({toBeJSONified}) { - if (typeof(toBeJSONified) == "string") { - try { - JSON.parse(toBeJSONified); - return String(toBeJSONified); - } catch(err) { - return "Not JSON!"; - } - } else if (typeof(toBeJSONified) == "object") { - return JSON.stringify(toBeJSONified); - } else { - return "Not JSON!"; - }; - }; - - onConnect() { - const self = this; - if (self.connect_hat == 0 && self.isRunning && self.protocolOk) { - self.connect_hat = 1; - return true; - } else { - return false; - }; - }; - - onClose() { - const self = this; - if (self.close_hat == 0 && !self.isRunning) { - self.close_hat = 1; - return true; - } else { - return false; - }; - }; - - onListener({ ID }) { - const self = this; - if ((this.isRunning) && (this.socketListeners.hasOwnProperty(String(ID)))) { - if (self.socketListeners[String(ID)]) { - self.socketListeners[String(ID)] = false; - return true; - } else { - return false; - }; - } else { - return false; - }; - }; - - onNewPacket({ TYPE }) { - const self = this; - if ((this.isRunning) && (this.newSocketData[this.menuRemap[String(TYPE)]])) { - self.newSocketData[this.menuRemap[String(TYPE)]] = false; - return true; - } else { - return false; - }; - }; - - onNewVar({ TYPE, VAR }) { - const self = this; - if (this.isRunning) { - if (this.varData.hasOwnProperty(this.menuRemap[TYPE])) { - if (this.varData[this.menuRemap[TYPE]].hasOwnProperty(VAR)) { - if (this.varData[this.menuRemap[TYPE]][VAR].isNew) { - self.varData[this.menuRemap[TYPE]][VAR].isNew = false; - return true; - } else { - return false; - } - } else { - return false; - }; - } else { - return false; - }; - } else { - return false; - }; - }; - - getComState(){ - return String((this.link_status == 2) || this.protocolOk); - }; - - getRoomState() { - return this.isLinked; - }; - - getComLostConnectionState() { - return this.wasConnectionDropped; - }; - - getComFailedConnectionState() { - return this.didConnectionFail; - }; - - getUsernameState(){ - return this.isUsernameSet; - }; - - returnIsNewData({TYPE}){ - if (this.isRunning) { - return this.newSocketData[this.menuRemap[String(TYPE)]]; - } else { - return false; - }; - }; - - returnIsNewVarData({ TYPE, VAR }) { - if (this.isRunning) { - if (this.varData.hasOwnProperty(this.menuRemap[TYPE])) { - if (this.varData[this.menuRemap[TYPE]].hasOwnProperty(VAR)) { - return this.varData[this.menuRemap[TYPE]][VAR].isNew; - } else { - return false; - }; - } else { - return false; - }; - } else { - return false; - }; - }; - - returnIsNewListener({ ID }) { - if (this.isRunning) { - if (this.socketListeners.hasOwnProperty(String(ID))) { - return this.socketListeners[ID]; - } else { - return false; - }; - } else { - return false; - }; - }; - - checkForID({ ID }) { - return find_id(ID, this.socketData.ulist); - }; - - async openSocket({IP}) { - const self = this; - if (!self.isRunning) { - if (!await Scratch.canFetch(IP)) { - return; - } - - console.log("Starting socket."); - self.link_status = 1; - - self.disconnectWasClean = false; - self.wasConnectionDropped = false; - self.didConnectionFail = false; - - mWS = new WebSocket(String(IP)); - - mWS.onerror = function(){ - self.isRunning = false; - }; - - mWS.onopen = function(){ - self.isRunning = true; - self.packet_queue = {}; - self.link_status = 2; - - // Send the handshake request to get server to detect client protocol - mWS.send(JSON.stringify({"cmd": "handshake", "listener": "setprotocol"})) - - console.log("Successfully opened socket."); - }; - - mWS.onmessage = function(event){ - let tmp_socketData = JSON.parse(event.data); - console.log("RX:", tmp_socketData); - - if (self.queueableCmds.includes(tmp_socketData.cmd)) { - self.socketData[tmp_socketData.cmd].push(tmp_socketData); - } else { - if (tmp_socketData.cmd == "ulist") { - // ulist functionality has been changed in server 0.1.9 - if (tmp_socketData.hasOwnProperty("mode")) { - if (tmp_socketData.mode == "set") { - self.socketData["ulist"] = tmp_socketData.val; - } else if (tmp_socketData.mode == "add") { - if (!self.socketData.ulist.some(o => ((o.username === tmp_socketData.val.username) && (o.id == tmp_socketData.val.id)))) { - self.socketData["ulist"].push(tmp_socketData.val); - } else { - console.log("Could not perform ulist method add, client", tmp_socketData.val, "already exists"); - }; - } else if (tmp_socketData.mode == "remove") { - if (self.socketData.ulist.some(o => ((o.username === tmp_socketData.val.username) && (o.id == tmp_socketData.val.id)))) { - // This is by far the fugliest thing I have ever written in JS, or in any programming language... thanks I hate it - self.socketData["ulist"] = self.socketData["ulist"].filter(user => ((!(user.username === tmp_socketData.val.username)) && (!(user.id == tmp_socketData.val.id)))); - } else { - console.log("Could not perform ulist method remove, client", tmp_socketData.val, "was not found"); - }; - } else { - console.log("Could not understand ulist method:", tmp_socketData.mode); - }; - } else { - // Retain compatibility wtih existing servers - self.socketData["ulist"] = tmp_socketData.val; - }; - } else { - self.socketData[tmp_socketData.cmd] = tmp_socketData.val; - }; - }; - - if (self.newSocketData.hasOwnProperty(tmp_socketData.cmd)) { - self.newSocketData[tmp_socketData.cmd] = true; - }; - - if (self.varCmds.includes(tmp_socketData.cmd)) { - self.varData[tmp_socketData.cmd][tmp_socketData.name] = { - "value": tmp_socketData.val, - "isNew": true - }; - }; - if (tmp_socketData.hasOwnProperty("listener")) { - if (tmp_socketData.listener == "setusername") { - self.socketListeners["setusername"] = true; - if (tmp_socketData.code == "I:100 | OK") { - self.username = tmp_socketData.val; - self.isUsernameSyncing = false; - self.isUsernameSet = true; - console.log("Username was accepted by the server, and has been set to:", self.username); - } else { - console.warn("Username was rejected by the server. Error code:", String(tmp_socketData.code)); - self.isUsernameSyncing = false; - }; - } else if (tmp_socketData.listener == "roomLink") { - self.isRoomSetting = false; - self.socketListeners["roomLink"] = true; - if (tmp_socketData.code == "I:100 | OK") { - console.log("Linking to room(s) was accepted by the server!"); - self.isLinked = true; - } else { - console.warn("Linking to room(s) was rejected by the server. Error code:", String(tmp_socketData.code)); - self.enableRoom = false; - self.isLinked = false; - self.selectRoom = ""; - }; - } else if ((tmp_socketData.listener == "setprotocol") && (!this.protocolOk)) { - console.log("Server successfully set client protocol to cloudlink!"); - self.socketData.statuscode = []; - self.protocolOk = true; - self.socketListeners["setprotocol"] = true; - } else { - if (self.socketListeners.hasOwnProperty(tmp_socketData.listener)) { - self.socketListeners[tmp_socketData.listener] = true; - }; - }; - self.socketListenersData[tmp_socketData.listener] = tmp_socketData; - }; - self.packet_hat = 0; - }; - - mWS.onclose = function() { - self.isRunning = false; - self.connect_hat = 0; - self.packet_hat = 0; - self.protocolOk = false; - if (self.close_hat == 1) { - self.close_hat = 0; - }; - self.socketData = { - "gmsg": [], - "pmsg": [], - "direct": [], - "statuscode": [], - "gvar": [], - "pvar": [], - "motd": "", - "client_ip": "", - "ulist": [], - "server_version": "" - }; - self.newSocketData = { - "gmsg": false, - "pmsg": false, - "direct": false, - "statuscode": false, - "gvar": false, - "pvar": false - }; - self.socketListeners = {}; - self.username = ""; - self.tmp_username = ""; - self.isUsernameSyncing = false; - self.isUsernameSet = false; - self.enableListener = false; - self.setListener = ""; - self.enableRoom = false; - self.selectRoom = ""; - self.isLinked = false; - self.isRoomSetting = false; - - if (self.link_status != 1) { - if (self.disconnectWasClean) { - self.link_status = 3; - console.log("Socket closed."); - self.wasConnectionDropped = false; - self.didConnectionFail = false; - } else { - self.link_status = 4; - console.error("Lost connection to the server."); - self.wasConnectionDropped = true; - self.didConnectionFail = false; - }; - } else { - self.link_status = 4; - console.error("Failed to connect to server."); - self.wasConnectionDropped = false; - self.didConnectionFail = true; - }; - }; - } else { - console.warn("Socket is already open."); - }; - } - - openSocketPublicServers({ ID }){ - if (servers.hasOwnProperty(ID)) { - console.log("Connecting to:", servers[ID].url) - this.openSocket({"IP": servers[ID].url}); - }; - }; - - closeSocket(){ - const self = this; - if (this.isRunning) { - console.log("Closing socket..."); - mWS.close(1000,'script closure'); - self.disconnectWasClean = true; - } else { - console.warn("Socket is not open."); - }; - } - - setMyName({NAME}) { - const self = this; - if (this.isRunning) { - if (!this.isUsernameSyncing) { - if (!this.isUsernameSet){ - if (String(NAME) != "") { - if ((!(String(NAME).length > 20))) { - if (!(String(NAME) == "%CA%" || String(NAME) == "%CC%" || String(NAME) == "%CD%" || String(NAME) == "%MS%")){ - let tmp_msg = { - cmd: "setid", - val: String(NAME), - listener: "setusername" - }; - - console.log("TX:", tmp_msg); - mWS.send(JSON.stringify(tmp_msg)); - - self.tmp_username = String(NAME); - self.isUsernameSyncing = true; - - } else { - console.log("Blocking attempt to use reserved usernames"); - }; - } else { - console.log("Blocking attempt to use username larger than 20 characters, username is " + String(NAME).length + " characters long"); - }; - } else { - console.log("Blocking attempt to use blank username"); - }; - } else { - console.warn("Username already has been set!"); - }; - } else { - console.warn("Username is still syncing!"); - }; - }; - }; - - createListener({ ID }) { - self = this; - if (this.isRunning) { - if (!this.enableListener) { - self.enableListener = true; - self.setListener = String(ID); - } else { - console.warn("Listeners were already created!"); - }; - } else { - console.log("Cannot assign a listener to a packet while disconnected"); - }; - }; - - linkToRooms({ ROOMS }) { - const self = this; - - if (this.isRunning) { - if (!this.isRoomSetting) { - if (!(String(ROOMS).length > 1000)) { - let tmp_msg = { - cmd: "link", - val: autoConvert(ROOMS), - listener: "roomLink" - }; - - console.log("TX:", tmp_msg); - mWS.send(JSON.stringify(tmp_msg)); - - self.isRoomSetting = true; - - } else { - console.warn("Blocking attempt to send a room ID / room list larger than 1000 bytes (1 KB), room ID / room list is " + String(ROOMS).length + " bytes"); - }; - } else { - console.warn("Still linking to rooms!"); - }; - } else { - console.warn("Socket is not open."); - }; - }; - - selectRoomsInNextPacket({ROOMS}) { - const self = this; - if (this.isRunning) { - if (this.isLinked) { - if (!this.enableRoom) { - if (!(String(ROOMS).length > 1000)) { - self.enableRoom = true; - self.selectRoom = ROOMS; - } else { - console.warn("Blocking attempt to select a room ID / room list larger than 1000 bytes (1 KB), room ID / room list is " + String(ROOMS).length + " bytes"); - }; - } else { - console.warn("Rooms were already selected!"); - }; - } else { - console.warn("Not linked to any room(s)!"); - }; - } else { - console.warn("Socket is not open."); - }; - }; - - unlinkFromRooms() { - const self = this; - if (this.isRunning) { - if (this.isLinked) { - let tmp_msg = { - cmd: "unlink", - val: "" - }; - - if (this.enableListener) { - tmp_msg["listener"] = autoConvert(this.setListener); - }; - - console.log("TX:", tmp_msg); - mWS.send(JSON.stringify(tmp_msg)); - - if (this.enableListener) { - if (!self.socketListeners.hasOwnProperty(this.setListener)) { - self.socketListeners[this.setListener] = false; - }; - self.enableListener = false; - }; - - self.isLinked = false; - } else { - console.warn("Not linked to any rooms!"); - }; - } else { - console.warn("Socket is not open."); - }; - }; - - sendGData({DATA}){ - const self = this; - if (this.isRunning) { - if (!(String(DATA).length > 1000)) { - let tmp_msg = { - cmd: "gmsg", - val: autoConvert(DATA) - }; - - if (this.enableListener) { - tmp_msg["listener"] = String(this.setListener); - }; - - if (this.enableRoom) { - tmp_msg["rooms"] = autoConvert(this.selectRoom); - }; - - console.log("TX:", tmp_msg); - mWS.send(JSON.stringify(tmp_msg)); - - if (this.enableListener) { - if (!self.socketListeners.hasOwnProperty(this.setListener)) { - self.socketListeners[this.setListener] = false; - }; - self.enableListener = false; - }; - if (this.enableRoom) { - self.enableRoom = false; - self.selectRoom = ""; - }; - - } else { - console.warn("Blocking attempt to send packet larger than 1000 bytes (1 KB), packet is " + String(DATA).length + " bytes"); - }; - } else { - console.warn("Socket is not open."); - }; - }; - - sendPData({DATA, ID}) { - const self = this; - if (this.isRunning) { - if (!(String(DATA).length > 1000)) { - let tmp_msg = { - cmd: "pmsg", - val: autoConvert(DATA), - id: autoConvert(ID) - } - - if (this.enableListener) { - tmp_msg["listener"] = String(this.setListener); - }; - if (this.enableRoom) { - tmp_msg["rooms"] = autoConvert(this.selectRoom); - }; - - console.log("TX:", tmp_msg); - mWS.send(JSON.stringify(tmp_msg)); - - if (this.enableListener) { - if (!self.socketListeners.hasOwnProperty(this.setListener)) { - self.socketListeners[this.setListener] = false; - }; - self.enableListener = false; - }; - if (this.enableRoom) { - self.enableRoom = false; - self.selectRoom = ""; - }; - - } else { - console.warn("Blocking attempt to send packet larger than 1000 bytes (1 KB), packet is " + String(DATA).length + " bytes"); - }; - } else { - console.warn("Socket is not open."); - }; - }; - - sendGDataAsVar({VAR, DATA }) { - const self = this; - if (this.isRunning) { - if (!(String(DATA).length > 1000)) { - let tmp_msg = { - cmd: "gvar", - name: VAR, - val: autoConvert(DATA) - } - - if (this.enableListener) { - tmp_msg["listener"] = String(this.setListener); - }; - if (this.enableRoom) { - tmp_msg["rooms"] = autoConvert(this.selectRoom); - }; - - console.log("TX:", tmp_msg); - mWS.send(JSON.stringify(tmp_msg)); - - if (this.enableListener) { - if (!self.socketListeners.hasOwnProperty(this.setListener)) { - self.socketListeners[this.setListener] = false; - }; - self.enableListener = false; - }; - if (this.enableRoom) { - self.enableRoom = false; - self.selectRoom = ""; - }; - - } else { - console.warn("Blocking attempt to send packet larger than 1000 bytes (1 KB), packet is " + String(DATA).length + " bytes"); - }; - } else { - console.warn("Socket is not open."); - }; - }; - - sendPDataAsVar({VAR, ID, DATA}) { - const self = this; - if (this.isRunning) { - if (!(String(DATA).length > 1000)) { - let tmp_msg = { - cmd: "pvar", - name: VAR, - val: autoConvert(DATA), - id: autoConvert(ID) - } - - if (this.enableListener) { - tmp_msg["listener"] = String(this.setListener); - }; - if (this.enableRoom) { - tmp_msg["rooms"] = autoConvert(this.selectRoom); - }; - - console.log("TX:", tmp_msg); - mWS.send(JSON.stringify(tmp_msg)); - - if (this.enableListener) { - if (!self.socketListeners.hasOwnProperty(this.setListener)) { - self.socketListeners[this.setListener] = false; - }; - self.enableListener = false; - }; - if (this.enableRoom) { - self.enableRoom = false; - self.selectRoom = ""; - }; - - } else { - console.warn("Blocking attempt to send packet larger than 1000 bytes (1 KB), packet is " + String(DATA).length + " bytes"); - }; - } else { - console.warn("Socket is not open."); - }; - }; - - runCMDnoID({CMD, DATA}) { - const self = this; - if (this.isRunning) { - if (!(String(CMD).length > 100) || !(String(DATA).length > 1000)) { - let tmp_msg = { - cmd: String(CMD), - val: autoConvert(DATA) - } - - if (this.enableListener) { - tmp_msg["listener"] = String(this.setListener); - }; - if (this.enableRoom) { - tmp_msg["rooms"] = String(this.selectRoom); - }; - - console.log("TX:", tmp_msg); - mWS.send(JSON.stringify(tmp_msg)); - - if (this.enableListener) { - if (!self.socketListeners.hasOwnProperty(this.setListener)) { - self.socketListeners[this.setListener] = false; - }; - self.enableListener = false; - }; - if (this.enableRoom) { - self.enableRoom = false; - self.selectRoom = ""; - }; - - } else { - console.warn("Blocking attempt to send packet with questionably long arguments"); - }; - } else { - console.warn("Socket is not open."); - }; - }; - - runCMD({CMD, ID, DATA}) { - const self = this; - if (this.isRunning) { - if (!(String(CMD).length > 100) || !(String(ID).length > 20) || !(String(DATA).length > 1000)) { - let tmp_msg = { - cmd: String(CMD), - id: autoConvert(ID), - val: autoConvert(DATA) - } - - if (this.enableListener) { - tmp_msg["listener"] = String(this.setListener); - }; - if (this.enableRoom) { - tmp_msg["rooms"] = String(this.selectRoom); - }; - - console.log("TX:", tmp_msg); - mWS.send(JSON.stringify(tmp_msg)); - - if (this.enableListener) { - if (!self.socketListeners.hasOwnProperty(this.setListener)) { - self.socketListeners[this.setListener] = false; - }; - self.enableListener = false; - }; - if (this.enableRoom) { - self.enableRoom = false; - self.selectRoom = ""; - }; - - } else { - console.warn("Blocking attempt to send packet with questionably long arguments"); - }; - } else { - console.warn("Socket is not open."); - }; - }; - - resetNewData({TYPE}){ - const self = this; - if (this.isRunning) { - self.newSocketData[this.menuRemap[String(TYPE)]] = false; - }; - }; - - resetNewVarData({ TYPE, VAR }) { - const self = this; - if (this.isRunning) { - if (this.varData.hasOwnProperty(this.menuRemap[TYPE])) { - if (this.varData[this.menuRemap[TYPE]].hasOwnProperty(VAR)) { - self.varData[this.menuRemap[TYPE]][VAR].isNew = false; - }; - }; - }; - }; - - resetNewListener({ ID }) { - const self = this; - if (this.isRunning) { - if (this.socketListeners.hasOwnProperty(String(ID))) { - self.socketListeners[String(ID)] = false; - }; - }; - }; - - clearAllPackets({TYPE}){ - const self = this; - if (this.menuRemap[String(TYPE)] == "all") { - self.socketData.gmsg = []; - self.socketData.pmsg = []; - self.socketData.direct = []; - self.socketData.statuscode = []; - self.socketData.gvar = []; - self.socketData.pvar = []; - } else { - self.socketData[this.menuRemap[String(TYPE)]] = []; - }; - }; -}; - -console.log("CloudLink 4.0 loaded. Detecting unsandboxed mode."); -Scratch.extensions.register(new CloudLink(Scratch.vm.runtime)); - +(function (Scratch) { + var servers = {}; // Server list + let mWS = null; + + // Get the server URL list + try { + Scratch.fetch("https://mikedev101.github.io/cloudlink/serverlist.json") + .then((response) => { + return response.text(); + }) + .then((data) => { + servers = JSON.parse(data); + }) + .catch((err) => { + console.log(err); + servers = {}; + }); + } catch (err) { + console.log(err); + servers = {}; + } + + function find_id(ID, ulist) { + // Thanks StackOverflow! + if (jsonCheck(ID) && !intCheck(ID)) { + return ulist.some( + (o) => + o.username === JSON.parse(ID).username && o.id == JSON.parse(ID).id + ); + } else { + return ulist.some((o) => o.username === String(ID) || o.id == ID); + } + } + + function jsonCheck(JSON_STRING) { + try { + JSON.parse(JSON_STRING); + return true; + } catch (err) { + return false; + } + } + + function intCheck(value) { + return !isNaN(value); + } + + function autoConvert(value) { + // Check if the value is JSON / Dict first + try { + JSON.parse(value); + return JSON.parse(value); + } catch (err) {} + + // Check if the value is an array + try { + tmp = value; + tmp = tmp.replace(/'/g, '"'); + JSON.parse(tmp); + return JSON.parse(tmp); + } catch (err) {} + + // Check if an int/float + if (!isNaN(value)) { + return Number(value); + } + + // Leave as the original value if none of the above work + return value; + } + + class CloudLink { + constructor(runtime, extensionId) { + // Extension stuff + this.runtime = runtime; + this.cl_icon = + ""; + this.cl_block = + ""; + + // Socket data + this.socketData = { + gmsg: [], + pmsg: [], + direct: [], + statuscode: [], + gvar: [], + pvar: [], + motd: "", + client_ip: "", + ulist: [], + server_version: "", + }; + this.varData = { + gvar: {}, + pvar: {}, + }; + + this.queueableCmds = [ + "gmsg", + "pmsg", + "gvar", + "pvar", + "direct", + "statuscode", + ]; + this.varCmds = ["gvar", "pvar"]; + + // Listeners + this.socketListeners = {}; + this.socketListenersData = {}; + this.newSocketData = { + gmsg: false, + pmsg: false, + direct: false, + statuscode: false, + gvar: false, + pvar: false, + }; + + // Edge-triggered hat blocks + this.connect_hat = 0; + this.packet_hat = 0; + this.close_hat = 0; + + // Status stuff + this.isRunning = false; + this.isLinked = false; + this.version = "S4.0"; + this.link_status = 0; + this.username = ""; + this.tmp_username = ""; + this.isUsernameSyncing = false; + this.isUsernameSet = false; + this.disconnectWasClean = false; + this.wasConnectionDropped = false; + this.didConnectionFail = false; + this.protocolOk = false; + + // Listeners stuff + this.enableListener = false; + this.setListener = ""; + + // Rooms stuff + this.enableRoom = false; + this.isRoomSetting = false; + this.selectRoom = ""; + + // Remapping stuff + this.menuRemap = { + "Global data": "gmsg", + "Private data": "pmsg", + "Global variables": "gvar", + "Private variables": "pvar", + "Direct data": "direct", + "Status code": "statuscode", + "All data": "all", + }; + } + + getInfo() { + return { + id: "cloudlink", + name: "CloudLink", + blockIconURI: this.cl_block, + menuIconURI: this.cl_icon, + docsURI: "https://hackmd.io/@MikeDEV/HJiNYwOfo", + blocks: [ + { + opcode: "returnGlobalData", + blockType: "reporter", + text: "Global data", + }, + { + opcode: "returnPrivateData", + blockType: "reporter", + text: "Private data", + }, + { + opcode: "returnDirectData", + blockType: "reporter", + text: "Direct Data", + }, + { + opcode: "returnLinkData", + blockType: "reporter", + text: "Link status", + }, + { + opcode: "returnStatusCode", + blockType: "reporter", + text: "Status code", + }, + { + opcode: "returnUserListData", + blockType: "reporter", + text: "Usernames", + }, + { + opcode: "returnUsernameData", + blockType: "reporter", + text: "My username", + }, + { + opcode: "returnVersionData", + blockType: "reporter", + text: "Extension version", + }, + { + opcode: "returnServerVersion", + blockType: "reporter", + text: "Server version", + }, + { + opcode: "returnServerList", + blockType: "reporter", + text: "Server list", + }, + { + opcode: "returnMOTD", + blockType: "reporter", + text: "Server MOTD", + }, + { + opcode: "returnClientIP", + blockType: "reporter", + text: "My IP address", + }, + { + opcode: "returnListenerData", + blockType: "reporter", + text: "Response for listener [ID]", + arguments: { + ID: { + type: "string", + defaultValue: "example-listener", + }, + }, + }, + { + opcode: "readQueueSize", + blockType: "reporter", + text: "Size of queue for [TYPE]", + arguments: { + TYPE: { + type: "string", + menu: "allmenu", + defaultValue: "All data", + }, + }, + }, + { + opcode: "readQueueData", + blockType: "reporter", + text: "Packet queue for [TYPE]", + arguments: { + TYPE: { + type: "string", + menu: "allmenu", + defaultValue: "All data", + }, + }, + }, + { + opcode: "returnVarData", + blockType: "reporter", + text: "[TYPE] [VAR] data", + arguments: { + VAR: { + type: "string", + defaultValue: "Apple", + }, + TYPE: { + type: "string", + menu: "varmenu", + defaultValue: "Global variables", + }, + }, + }, + { + opcode: "parseJSON", + blockType: "reporter", + text: "[PATH] of [JSON_STRING]", + arguments: { + PATH: { + type: "string", + defaultValue: "fruit/apples", + }, + JSON_STRING: { + type: "string", + defaultValue: + '{"fruit": {"apples": 2, "bananas": 3}, "total_fruit": 5}', + }, + }, + }, + { + opcode: "getFromJSONArray", + blockType: "reporter", + text: "Get [NUM] from JSON array [ARRAY]", + arguments: { + NUM: { + type: "number", + defaultValue: 0, + }, + ARRAY: { + type: "string", + defaultValue: '["foo","bar"]', + }, + }, + }, + { + opcode: "fetchURL", + blockType: "reporter", + blockAllThreads: "true", + text: "Fetch data from URL [url]", + arguments: { + url: { + type: "string", + defaultValue: + "https://mikedev101.github.io/cloudlink/fetch_test", + }, + }, + }, + { + opcode: "requestURL", + blockType: "reporter", + blockAllThreads: "true", + text: "Send request with method [method] for URL [url] with data [data] and headers [headers]", + arguments: { + method: { + type: "string", + defaultValue: "GET", + }, + url: { + type: "string", + defaultValue: + "https://mikedev101.github.io/cloudlink/fetch_test", + }, + data: { + type: "string", + defaultValue: "{}", + }, + headers: { + type: "string", + defaultValue: "{}", + }, + }, + }, + { + opcode: "makeJSON", + blockType: "reporter", + text: "Convert [toBeJSONified] to JSON", + arguments: { + toBeJSONified: { + type: "string", + defaultValue: '{"test": true}', + }, + }, + }, + { + opcode: "onConnect", + blockType: "hat", + text: "When connected", + blockAllThreads: "true", + }, + { + opcode: "onClose", + blockType: "hat", + text: "When disconnected", + blockAllThreads: "true", + }, + { + opcode: "onListener", + blockType: "hat", + text: "When I receive new packet with listener [ID]", + blockAllThreads: "true", + arguments: { + ID: { + type: "string", + defaultValue: "example-listener", + }, + }, + }, + { + opcode: "onNewPacket", + blockType: "hat", + text: "When I receive new [TYPE] packet", + blockAllThreads: "true", + arguments: { + TYPE: { + type: "string", + menu: "almostallmenu", + defaultValue: "Global data", + }, + }, + }, + { + opcode: "onNewVar", + blockType: "hat", + text: "When I receive new [TYPE] data for [VAR]", + blockAllThreads: "true", + arguments: { + TYPE: { + type: "string", + menu: "varmenu", + defaultValue: "Global variables", + }, + VAR: { + type: "string", + defaultValue: "Apple", + }, + }, + }, + { + opcode: "getComState", + blockType: "Boolean", + text: "Connected?", + }, + { + opcode: "getRoomState", + blockType: "Boolean", + text: "Linked to rooms?", + }, + { + opcode: "getComLostConnectionState", + blockType: "Boolean", + text: "Lost connection?", + }, + { + opcode: "getComFailedConnectionState", + blockType: "Boolean", + text: "Failed to connnect?", + }, + { + opcode: "getUsernameState", + blockType: "Boolean", + text: "Username synced?", + }, + { + opcode: "returnIsNewData", + blockType: "Boolean", + text: "Got New [TYPE]?", + arguments: { + TYPE: { + type: "string", + menu: "datamenu", + defaultValue: "Global data", + }, + }, + }, + { + opcode: "returnIsNewVarData", + blockType: "Boolean", + text: "Got New [TYPE] data for variable [VAR]?", + arguments: { + TYPE: { + type: "string", + menu: "varmenu", + defaultValue: "Global variables", + }, + VAR: { + type: "string", + defaultValue: "Apple", + }, + }, + }, + { + opcode: "returnIsNewListener", + blockType: "Boolean", + text: "Got new packet with listener [ID]?", + blockAllThreads: "true", + arguments: { + ID: { + type: "string", + defaultValue: "example-listener", + }, + }, + }, + { + opcode: "checkForID", + blockType: "Boolean", + text: "ID [ID] connected?", + arguments: { + ID: { + type: "string", + defaultValue: "Another name", + }, + }, + }, + { + opcode: "isValidJSON", + blockType: "Boolean", + text: "Is [JSON_STRING] valid JSON?", + arguments: { + JSON_STRING: { + type: "string", + defaultValue: + '{"fruit": {"apples": 2, "bananas": 3}, "total_fruit": 5}', + }, + }, + }, + { + opcode: "openSocket", + blockType: "command", + text: "Connect to [IP]", + blockAllThreads: "true", + arguments: { + IP: { + type: "string", + defaultValue: "ws://127.0.0.1:3000/", + }, + }, + }, + { + opcode: "openSocketPublicServers", + blockType: "command", + text: "Connect to server [ID]", + blockAllThreads: "true", + arguments: { + ID: { + type: "number", + defaultValue: "", + }, + }, + }, + { + opcode: "closeSocket", + blockType: "command", + blockAllThreads: "true", + text: "Disconnect", + }, + { + opcode: "setMyName", + blockType: "command", + text: "Set [NAME] as username", + blockAllThreads: "true", + arguments: { + NAME: { + type: "string", + defaultValue: "A name", + }, + }, + }, + { + opcode: "createListener", + blockType: "command", + text: "Attach listener [ID] to next packet", + blockAllThreads: "true", + arguments: { + ID: { + type: "string", + defaultValue: "example-listener", + }, + }, + }, + { + opcode: "linkToRooms", + blockType: "command", + text: "Link to room(s) [ROOMS]", + blockAllThreads: "true", + arguments: { + ROOMS: { + type: "string", + defaultValue: '["test"]', + }, + }, + }, + { + opcode: "selectRoomsInNextPacket", + blockType: "command", + text: "Select room(s) [ROOMS] for next packet", + blockAllThreads: "true", + arguments: { + ROOMS: { + type: "string", + defaultValue: '["test"]', + }, + }, + }, + { + opcode: "unlinkFromRooms", + blockType: "command", + text: "Unlink from all rooms", + blockAllThreads: "true", + }, + { + opcode: "sendGData", + blockType: "command", + text: "Send [DATA]", + blockAllThreads: "true", + arguments: { + DATA: { + type: "string", + defaultValue: "Apple", + }, + }, + }, + { + opcode: "sendPData", + blockType: "command", + text: "Send [DATA] to [ID]", + blockAllThreads: "true", + arguments: { + DATA: { + type: "string", + defaultValue: "Apple", + }, + ID: { + type: "string", + defaultValue: "Another name", + }, + }, + }, + { + opcode: "sendGDataAsVar", + blockType: "command", + text: "Send variable [VAR] with data [DATA]", + blockAllThreads: "true", + arguments: { + DATA: { + type: "string", + defaultValue: "Banana", + }, + VAR: { + type: "string", + defaultValue: "Apple", + }, + }, + }, + { + opcode: "sendPDataAsVar", + blockType: "command", + text: "Send variable [VAR] to [ID] with data [DATA]", + blockAllThreads: "true", + arguments: { + DATA: { + type: "string", + defaultValue: "Banana", + }, + ID: { + type: "string", + defaultValue: "Another name", + }, + VAR: { + type: "string", + defaultValue: "Apple", + }, + }, + }, + { + opcode: "runCMDnoID", + blockType: "command", + text: "Send command without ID [CMD] [DATA]", + blockAllThreads: "true", + arguments: { + CMD: { + type: "string", + defaultValue: "direct", + }, + DATA: { + type: "string", + defaultValue: "val", + }, + }, + }, + { + opcode: "runCMD", + blockType: "command", + text: "Send command [CMD] [ID] [DATA]", + blockAllThreads: "true", + arguments: { + CMD: { + type: "string", + defaultValue: "direct", + }, + ID: { + type: "string", + defaultValue: "id", + }, + DATA: { + type: "string", + defaultValue: "val", + }, + }, + }, + { + opcode: "resetNewData", + blockType: "command", + text: "Reset got new [TYPE] status", + blockAllThreads: "true", + arguments: { + TYPE: { + type: "string", + menu: "datamenu", + defaultValue: "Global data", + }, + }, + }, + { + opcode: "resetNewVarData", + blockType: "command", + text: "Reset got new [TYPE] [VAR] status", + blockAllThreads: "true", + arguments: { + TYPE: { + type: "string", + menu: "varmenu", + defaultValue: "Global variables", + }, + VAR: { + type: "string", + defaultValue: "Apple", + }, + }, + }, + { + opcode: "resetNewListener", + blockType: "command", + text: "Reset got new [ID] listener status", + blockAllThreads: "true", + arguments: { + ID: { + type: "string", + defaultValue: "example-listener", + }, + }, + }, + { + opcode: "clearAllPackets", + blockType: "command", + text: "Clear all packets for [TYPE]", + arguments: { + TYPE: { + type: "string", + menu: "allmenu", + defaultValue: "All data", + }, + }, + }, + ], + menus: { + coms: { + items: ["Connected", "Username synced"], + }, + datamenu: { + items: [ + "Global data", + "Private data", + "Direct data", + "Status code", + ], + }, + varmenu: { + items: ["Global variables", "Private variables"], + }, + allmenu: { + items: [ + "Global data", + "Private data", + "Direct data", + "Status code", + "Global variables", + "Private variables", + "All data", + ], + }, + almostallmenu: { + items: [ + "Global data", + "Private data", + "Direct data", + "Status code", + "Global variables", + "Private variables", + ], + }, + }, + }; + } + + // Code for blocks go here + + returnGlobalData() { + if (this.socketData.gmsg.length != 0) { + let data = this.socketData.gmsg[this.socketData.gmsg.length - 1].val; + + if (typeof data == "object") { + data = JSON.stringify(data); // Make the JSON safe for Scratch + } + + return data; + } else { + return ""; + } + } + + returnPrivateData() { + if (this.socketData.pmsg.length != 0) { + let data = this.socketData.pmsg[this.socketData.pmsg.length - 1].val; + + if (typeof data == "object") { + data = JSON.stringify(data); // Make the JSON safe for Scratch + } + + return data; + } else { + return ""; + } + } + + returnDirectData() { + if (this.socketData.direct.length != 0) { + let data = + this.socketData.direct[this.socketData.direct.length - 1].val; + + if (typeof data == "object") { + data = JSON.stringify(data); // Make the JSON safe for Scratch + } + + return data; + } else { + return ""; + } + } + + returnLinkData() { + return String(this.link_status); + } + + returnStatusCode() { + if (this.socketData.statuscode.length != 0) { + let data = + this.socketData.statuscode[this.socketData.statuscode.length - 1] + .code; + + if (typeof data == "object") { + data = JSON.stringify(data); // Make the JSON safe for Scratch + } + + return data; + } else { + return ""; + } + } + + returnUserListData() { + return JSON.stringify(this.socketData.ulist); + } + + returnUsernameData() { + let data = this.username; + + if (typeof data == "object") { + data = JSON.stringify(data); // Make the JSON safe for Scratch + } + + return data; + } + + returnVersionData() { + return String(this.version); + } + + returnServerVersion() { + return String(this.socketData.server_version); + } + + returnServerList() { + return JSON.stringify(servers); + } + + returnMOTD() { + return String(this.socketData.motd); + } + + returnClientIP() { + return String(this.socketData.client_ip); + } + + returnListenerData({ ID }) { + const self = this; + if (this.isRunning && this.socketListeners.hasOwnProperty(String(ID))) { + return JSON.stringify(this.socketListenersData[ID]); + } else { + return "{}"; + } + } + + readQueueSize({ TYPE }) { + if (this.menuRemap[String(TYPE)] == "all") { + let tmp_size = 0; + tmp_size = tmp_size + this.socketData.gmsg.length; + tmp_size = tmp_size + this.socketData.pmsg.length; + tmp_size = tmp_size + this.socketData.direct.length; + tmp_size = tmp_size + this.socketData.statuscode.length; + tmp_size = tmp_size + this.socketData.gvar.length; + tmp_size = tmp_size + this.socketData.pvar.length; + return tmp_size; + } else { + return this.socketData[this.menuRemap[String(TYPE)]].length; + } + } + + readQueueData({ TYPE }) { + if (this.menuRemap[String(TYPE)] == "all") { + let tmp_socketData = JSON.parse(JSON.stringify(this.socketData)); // Deep copy + + delete tmp_socketData.motd; + delete tmp_socketData.client_ip; + delete tmp_socketData.ulist; + delete tmp_socketData.server_version; + + return JSON.stringify(tmp_socketData); + } else { + return JSON.stringify(this.socketData[this.menuRemap[String(TYPE)]]); + } + } + + returnVarData({ TYPE, VAR }) { + if (this.isRunning) { + if (this.varData.hasOwnProperty(this.menuRemap[TYPE])) { + if (this.varData[this.menuRemap[TYPE]].hasOwnProperty(VAR)) { + return this.varData[this.menuRemap[TYPE]][VAR].value; + } else { + return ""; + } + } else { + return ""; + } + } else { + return ""; + } + } + + parseJSON({ PATH, JSON_STRING }) { + try { + const path = PATH.toString() + .split("/") + .map((prop) => decodeURIComponent(prop)); + if (path[0] === "") path.splice(0, 1); + if (path[path.length - 1] === "") path.splice(-1, 1); + let json; + try { + json = JSON.parse(" " + JSON_STRING); + } catch (e) { + return e.message; + } + path.forEach((prop) => (json = json[prop])); + if (json === null) return "null"; + else if (json === undefined) return ""; + else if (typeof json === "object") return JSON.stringify(json); + else return json.toString(); + } catch (err) { + return ""; + } + } + + getFromJSONArray({ NUM, ARRAY }) { + var json_array = JSON.parse(ARRAY); + if (json_array[NUM] == "undefined") { + return ""; + } else { + let data = json_array[NUM]; + + if (typeof data == "object") { + data = JSON.stringify(data); // Make the JSON safe for Scratch + } + + return data; + } + } + + fetchURL(args) { + return Scratch.fetch(args.url, { + method: "GET", + }).then((response) => response.text()); + } + + requestURL(args) { + if (args.method == "GET" || args.method == "HEAD") { + return Scratch.fetch(args.url, { + method: args.method, + headers: JSON.parse(args.headers), + }).then((response) => response.text()); + } else { + return Scratch.fetch(args.url, { + method: args.method, + headers: JSON.parse(args.headers), + body: JSON.parse(args.data), + }).then((response) => response.text()); + } + } + + isValidJSON({ JSON_STRING }) { + return jsonCheck(JSON_STRING); + } + + makeJSON({ toBeJSONified }) { + if (typeof toBeJSONified == "string") { + try { + JSON.parse(toBeJSONified); + return String(toBeJSONified); + } catch (err) { + return "Not JSON!"; + } + } else if (typeof toBeJSONified == "object") { + return JSON.stringify(toBeJSONified); + } else { + return "Not JSON!"; + } + } + + onConnect() { + const self = this; + if (self.connect_hat == 0 && self.isRunning && self.protocolOk) { + self.connect_hat = 1; + return true; + } else { + return false; + } + } + + onClose() { + const self = this; + if (self.close_hat == 0 && !self.isRunning) { + self.close_hat = 1; + return true; + } else { + return false; + } + } + + onListener({ ID }) { + const self = this; + if (this.isRunning && this.socketListeners.hasOwnProperty(String(ID))) { + if (self.socketListeners[String(ID)]) { + self.socketListeners[String(ID)] = false; + return true; + } else { + return false; + } + } else { + return false; + } + } + + onNewPacket({ TYPE }) { + const self = this; + if (this.isRunning && this.newSocketData[this.menuRemap[String(TYPE)]]) { + self.newSocketData[this.menuRemap[String(TYPE)]] = false; + return true; + } else { + return false; + } + } + + onNewVar({ TYPE, VAR }) { + const self = this; + if (this.isRunning) { + if (this.varData.hasOwnProperty(this.menuRemap[TYPE])) { + if (this.varData[this.menuRemap[TYPE]].hasOwnProperty(VAR)) { + if (this.varData[this.menuRemap[TYPE]][VAR].isNew) { + self.varData[this.menuRemap[TYPE]][VAR].isNew = false; + return true; + } else { + return false; + } + } else { + return false; + } + } else { + return false; + } + } else { + return false; + } + } + + getComState() { + return String(this.link_status == 2 || this.protocolOk); + } + + getRoomState() { + return this.isLinked; + } + + getComLostConnectionState() { + return this.wasConnectionDropped; + } + + getComFailedConnectionState() { + return this.didConnectionFail; + } + + getUsernameState() { + return this.isUsernameSet; + } + + returnIsNewData({ TYPE }) { + if (this.isRunning) { + return this.newSocketData[this.menuRemap[String(TYPE)]]; + } else { + return false; + } + } + + returnIsNewVarData({ TYPE, VAR }) { + if (this.isRunning) { + if (this.varData.hasOwnProperty(this.menuRemap[TYPE])) { + if (this.varData[this.menuRemap[TYPE]].hasOwnProperty(VAR)) { + return this.varData[this.menuRemap[TYPE]][VAR].isNew; + } else { + return false; + } + } else { + return false; + } + } else { + return false; + } + } + + returnIsNewListener({ ID }) { + if (this.isRunning) { + if (this.socketListeners.hasOwnProperty(String(ID))) { + return this.socketListeners[ID]; + } else { + return false; + } + } else { + return false; + } + } + + checkForID({ ID }) { + return find_id(ID, this.socketData.ulist); + } + + async openSocket({ IP }) { + const self = this; + if (!self.isRunning) { + if (!(await Scratch.canFetch(IP))) { + return; + } + + console.log("Starting socket."); + self.link_status = 1; + + self.disconnectWasClean = false; + self.wasConnectionDropped = false; + self.didConnectionFail = false; + + mWS = new WebSocket(String(IP)); + + mWS.onerror = function () { + self.isRunning = false; + }; + + mWS.onopen = function () { + self.isRunning = true; + self.packet_queue = {}; + self.link_status = 2; + + // Send the handshake request to get server to detect client protocol + mWS.send( + JSON.stringify({ cmd: "handshake", listener: "setprotocol" }) + ); + + console.log("Successfully opened socket."); + }; + + mWS.onmessage = function (event) { + let tmp_socketData = JSON.parse(event.data); + console.log("RX:", tmp_socketData); + + if (self.queueableCmds.includes(tmp_socketData.cmd)) { + self.socketData[tmp_socketData.cmd].push(tmp_socketData); + } else { + if (tmp_socketData.cmd == "ulist") { + // ulist functionality has been changed in server 0.1.9 + if (tmp_socketData.hasOwnProperty("mode")) { + if (tmp_socketData.mode == "set") { + self.socketData["ulist"] = tmp_socketData.val; + } else if (tmp_socketData.mode == "add") { + if ( + !self.socketData.ulist.some( + (o) => + o.username === tmp_socketData.val.username && + o.id == tmp_socketData.val.id + ) + ) { + self.socketData["ulist"].push(tmp_socketData.val); + } else { + console.log( + "Could not perform ulist method add, client", + tmp_socketData.val, + "already exists" + ); + } + } else if (tmp_socketData.mode == "remove") { + if ( + self.socketData.ulist.some( + (o) => + o.username === tmp_socketData.val.username && + o.id == tmp_socketData.val.id + ) + ) { + // This is by far the fugliest thing I have ever written in JS, or in any programming language... thanks I hate it + self.socketData["ulist"] = self.socketData["ulist"].filter( + (user) => + !(user.username === tmp_socketData.val.username) && + !(user.id == tmp_socketData.val.id) + ); + } else { + console.log( + "Could not perform ulist method remove, client", + tmp_socketData.val, + "was not found" + ); + } + } else { + console.log( + "Could not understand ulist method:", + tmp_socketData.mode + ); + } + } else { + // Retain compatibility wtih existing servers + self.socketData["ulist"] = tmp_socketData.val; + } + } else { + self.socketData[tmp_socketData.cmd] = tmp_socketData.val; + } + } + + if (self.newSocketData.hasOwnProperty(tmp_socketData.cmd)) { + self.newSocketData[tmp_socketData.cmd] = true; + } + + if (self.varCmds.includes(tmp_socketData.cmd)) { + self.varData[tmp_socketData.cmd][tmp_socketData.name] = { + value: tmp_socketData.val, + isNew: true, + }; + } + if (tmp_socketData.hasOwnProperty("listener")) { + if (tmp_socketData.listener == "setusername") { + self.socketListeners["setusername"] = true; + if (tmp_socketData.code == "I:100 | OK") { + self.username = tmp_socketData.val; + self.isUsernameSyncing = false; + self.isUsernameSet = true; + console.log( + "Username was accepted by the server, and has been set to:", + self.username + ); + } else { + console.warn( + "Username was rejected by the server. Error code:", + String(tmp_socketData.code) + ); + self.isUsernameSyncing = false; + } + } else if (tmp_socketData.listener == "roomLink") { + self.isRoomSetting = false; + self.socketListeners["roomLink"] = true; + if (tmp_socketData.code == "I:100 | OK") { + console.log("Linking to room(s) was accepted by the server!"); + self.isLinked = true; + } else { + console.warn( + "Linking to room(s) was rejected by the server. Error code:", + String(tmp_socketData.code) + ); + self.enableRoom = false; + self.isLinked = false; + self.selectRoom = ""; + } + } else if ( + tmp_socketData.listener == "setprotocol" && + !this.protocolOk + ) { + console.log( + "Server successfully set client protocol to cloudlink!" + ); + self.socketData.statuscode = []; + self.protocolOk = true; + self.socketListeners["setprotocol"] = true; + } else { + if ( + self.socketListeners.hasOwnProperty(tmp_socketData.listener) + ) { + self.socketListeners[tmp_socketData.listener] = true; + } + } + self.socketListenersData[tmp_socketData.listener] = tmp_socketData; + } + self.packet_hat = 0; + }; + + mWS.onclose = function () { + self.isRunning = false; + self.connect_hat = 0; + self.packet_hat = 0; + self.protocolOk = false; + if (self.close_hat == 1) { + self.close_hat = 0; + } + self.socketData = { + gmsg: [], + pmsg: [], + direct: [], + statuscode: [], + gvar: [], + pvar: [], + motd: "", + client_ip: "", + ulist: [], + server_version: "", + }; + self.newSocketData = { + gmsg: false, + pmsg: false, + direct: false, + statuscode: false, + gvar: false, + pvar: false, + }; + self.socketListeners = {}; + self.username = ""; + self.tmp_username = ""; + self.isUsernameSyncing = false; + self.isUsernameSet = false; + self.enableListener = false; + self.setListener = ""; + self.enableRoom = false; + self.selectRoom = ""; + self.isLinked = false; + self.isRoomSetting = false; + + if (self.link_status != 1) { + if (self.disconnectWasClean) { + self.link_status = 3; + console.log("Socket closed."); + self.wasConnectionDropped = false; + self.didConnectionFail = false; + } else { + self.link_status = 4; + console.error("Lost connection to the server."); + self.wasConnectionDropped = true; + self.didConnectionFail = false; + } + } else { + self.link_status = 4; + console.error("Failed to connect to server."); + self.wasConnectionDropped = false; + self.didConnectionFail = true; + } + }; + } else { + console.warn("Socket is already open."); + } + } + + openSocketPublicServers({ ID }) { + if (servers.hasOwnProperty(ID)) { + console.log("Connecting to:", servers[ID].url); + this.openSocket({ IP: servers[ID].url }); + } + } + + closeSocket() { + const self = this; + if (this.isRunning) { + console.log("Closing socket..."); + mWS.close(1000, "script closure"); + self.disconnectWasClean = true; + } else { + console.warn("Socket is not open."); + } + } + + setMyName({ NAME }) { + const self = this; + if (this.isRunning) { + if (!this.isUsernameSyncing) { + if (!this.isUsernameSet) { + if (String(NAME) != "") { + if (!(String(NAME).length > 20)) { + if ( + !( + String(NAME) == "%CA%" || + String(NAME) == "%CC%" || + String(NAME) == "%CD%" || + String(NAME) == "%MS%" + ) + ) { + let tmp_msg = { + cmd: "setid", + val: String(NAME), + listener: "setusername", + }; + + console.log("TX:", tmp_msg); + mWS.send(JSON.stringify(tmp_msg)); + + self.tmp_username = String(NAME); + self.isUsernameSyncing = true; + } else { + console.log("Blocking attempt to use reserved usernames"); + } + } else { + console.log( + "Blocking attempt to use username larger than 20 characters, username is " + + String(NAME).length + + " characters long" + ); + } + } else { + console.log("Blocking attempt to use blank username"); + } + } else { + console.warn("Username already has been set!"); + } + } else { + console.warn("Username is still syncing!"); + } + } + } + + createListener({ ID }) { + self = this; + if (this.isRunning) { + if (!this.enableListener) { + self.enableListener = true; + self.setListener = String(ID); + } else { + console.warn("Listeners were already created!"); + } + } else { + console.log("Cannot assign a listener to a packet while disconnected"); + } + } + + linkToRooms({ ROOMS }) { + const self = this; + + if (this.isRunning) { + if (!this.isRoomSetting) { + if (!(String(ROOMS).length > 1000)) { + let tmp_msg = { + cmd: "link", + val: autoConvert(ROOMS), + listener: "roomLink", + }; + + console.log("TX:", tmp_msg); + mWS.send(JSON.stringify(tmp_msg)); + + self.isRoomSetting = true; + } else { + console.warn( + "Blocking attempt to send a room ID / room list larger than 1000 bytes (1 KB), room ID / room list is " + + String(ROOMS).length + + " bytes" + ); + } + } else { + console.warn("Still linking to rooms!"); + } + } else { + console.warn("Socket is not open."); + } + } + + selectRoomsInNextPacket({ ROOMS }) { + const self = this; + if (this.isRunning) { + if (this.isLinked) { + if (!this.enableRoom) { + if (!(String(ROOMS).length > 1000)) { + self.enableRoom = true; + self.selectRoom = ROOMS; + } else { + console.warn( + "Blocking attempt to select a room ID / room list larger than 1000 bytes (1 KB), room ID / room list is " + + String(ROOMS).length + + " bytes" + ); + } + } else { + console.warn("Rooms were already selected!"); + } + } else { + console.warn("Not linked to any room(s)!"); + } + } else { + console.warn("Socket is not open."); + } + } + + unlinkFromRooms() { + const self = this; + if (this.isRunning) { + if (this.isLinked) { + let tmp_msg = { + cmd: "unlink", + val: "", + }; + + if (this.enableListener) { + tmp_msg["listener"] = autoConvert(this.setListener); + } + + console.log("TX:", tmp_msg); + mWS.send(JSON.stringify(tmp_msg)); + + if (this.enableListener) { + if (!self.socketListeners.hasOwnProperty(this.setListener)) { + self.socketListeners[this.setListener] = false; + } + self.enableListener = false; + } + + self.isLinked = false; + } else { + console.warn("Not linked to any rooms!"); + } + } else { + console.warn("Socket is not open."); + } + } + + sendGData({ DATA }) { + const self = this; + if (this.isRunning) { + if (!(String(DATA).length > 1000)) { + let tmp_msg = { + cmd: "gmsg", + val: autoConvert(DATA), + }; + + if (this.enableListener) { + tmp_msg["listener"] = String(this.setListener); + } + + if (this.enableRoom) { + tmp_msg["rooms"] = autoConvert(this.selectRoom); + } + + console.log("TX:", tmp_msg); + mWS.send(JSON.stringify(tmp_msg)); + + if (this.enableListener) { + if (!self.socketListeners.hasOwnProperty(this.setListener)) { + self.socketListeners[this.setListener] = false; + } + self.enableListener = false; + } + if (this.enableRoom) { + self.enableRoom = false; + self.selectRoom = ""; + } + } else { + console.warn( + "Blocking attempt to send packet larger than 1000 bytes (1 KB), packet is " + + String(DATA).length + + " bytes" + ); + } + } else { + console.warn("Socket is not open."); + } + } + + sendPData({ DATA, ID }) { + const self = this; + if (this.isRunning) { + if (!(String(DATA).length > 1000)) { + let tmp_msg = { + cmd: "pmsg", + val: autoConvert(DATA), + id: autoConvert(ID), + }; + + if (this.enableListener) { + tmp_msg["listener"] = String(this.setListener); + } + if (this.enableRoom) { + tmp_msg["rooms"] = autoConvert(this.selectRoom); + } + + console.log("TX:", tmp_msg); + mWS.send(JSON.stringify(tmp_msg)); + + if (this.enableListener) { + if (!self.socketListeners.hasOwnProperty(this.setListener)) { + self.socketListeners[this.setListener] = false; + } + self.enableListener = false; + } + if (this.enableRoom) { + self.enableRoom = false; + self.selectRoom = ""; + } + } else { + console.warn( + "Blocking attempt to send packet larger than 1000 bytes (1 KB), packet is " + + String(DATA).length + + " bytes" + ); + } + } else { + console.warn("Socket is not open."); + } + } + + sendGDataAsVar({ VAR, DATA }) { + const self = this; + if (this.isRunning) { + if (!(String(DATA).length > 1000)) { + let tmp_msg = { + cmd: "gvar", + name: VAR, + val: autoConvert(DATA), + }; + + if (this.enableListener) { + tmp_msg["listener"] = String(this.setListener); + } + if (this.enableRoom) { + tmp_msg["rooms"] = autoConvert(this.selectRoom); + } + + console.log("TX:", tmp_msg); + mWS.send(JSON.stringify(tmp_msg)); + + if (this.enableListener) { + if (!self.socketListeners.hasOwnProperty(this.setListener)) { + self.socketListeners[this.setListener] = false; + } + self.enableListener = false; + } + if (this.enableRoom) { + self.enableRoom = false; + self.selectRoom = ""; + } + } else { + console.warn( + "Blocking attempt to send packet larger than 1000 bytes (1 KB), packet is " + + String(DATA).length + + " bytes" + ); + } + } else { + console.warn("Socket is not open."); + } + } + + sendPDataAsVar({ VAR, ID, DATA }) { + const self = this; + if (this.isRunning) { + if (!(String(DATA).length > 1000)) { + let tmp_msg = { + cmd: "pvar", + name: VAR, + val: autoConvert(DATA), + id: autoConvert(ID), + }; + + if (this.enableListener) { + tmp_msg["listener"] = String(this.setListener); + } + if (this.enableRoom) { + tmp_msg["rooms"] = autoConvert(this.selectRoom); + } + + console.log("TX:", tmp_msg); + mWS.send(JSON.stringify(tmp_msg)); + + if (this.enableListener) { + if (!self.socketListeners.hasOwnProperty(this.setListener)) { + self.socketListeners[this.setListener] = false; + } + self.enableListener = false; + } + if (this.enableRoom) { + self.enableRoom = false; + self.selectRoom = ""; + } + } else { + console.warn( + "Blocking attempt to send packet larger than 1000 bytes (1 KB), packet is " + + String(DATA).length + + " bytes" + ); + } + } else { + console.warn("Socket is not open."); + } + } + + runCMDnoID({ CMD, DATA }) { + const self = this; + if (this.isRunning) { + if (!(String(CMD).length > 100) || !(String(DATA).length > 1000)) { + let tmp_msg = { + cmd: String(CMD), + val: autoConvert(DATA), + }; + + if (this.enableListener) { + tmp_msg["listener"] = String(this.setListener); + } + if (this.enableRoom) { + tmp_msg["rooms"] = String(this.selectRoom); + } + + console.log("TX:", tmp_msg); + mWS.send(JSON.stringify(tmp_msg)); + + if (this.enableListener) { + if (!self.socketListeners.hasOwnProperty(this.setListener)) { + self.socketListeners[this.setListener] = false; + } + self.enableListener = false; + } + if (this.enableRoom) { + self.enableRoom = false; + self.selectRoom = ""; + } + } else { + console.warn( + "Blocking attempt to send packet with questionably long arguments" + ); + } + } else { + console.warn("Socket is not open."); + } + } + + runCMD({ CMD, ID, DATA }) { + const self = this; + if (this.isRunning) { + if ( + !(String(CMD).length > 100) || + !(String(ID).length > 20) || + !(String(DATA).length > 1000) + ) { + let tmp_msg = { + cmd: String(CMD), + id: autoConvert(ID), + val: autoConvert(DATA), + }; + + if (this.enableListener) { + tmp_msg["listener"] = String(this.setListener); + } + if (this.enableRoom) { + tmp_msg["rooms"] = String(this.selectRoom); + } + + console.log("TX:", tmp_msg); + mWS.send(JSON.stringify(tmp_msg)); + + if (this.enableListener) { + if (!self.socketListeners.hasOwnProperty(this.setListener)) { + self.socketListeners[this.setListener] = false; + } + self.enableListener = false; + } + if (this.enableRoom) { + self.enableRoom = false; + self.selectRoom = ""; + } + } else { + console.warn( + "Blocking attempt to send packet with questionably long arguments" + ); + } + } else { + console.warn("Socket is not open."); + } + } + + resetNewData({ TYPE }) { + const self = this; + if (this.isRunning) { + self.newSocketData[this.menuRemap[String(TYPE)]] = false; + } + } + + resetNewVarData({ TYPE, VAR }) { + const self = this; + if (this.isRunning) { + if (this.varData.hasOwnProperty(this.menuRemap[TYPE])) { + if (this.varData[this.menuRemap[TYPE]].hasOwnProperty(VAR)) { + self.varData[this.menuRemap[TYPE]][VAR].isNew = false; + } + } + } + } + + resetNewListener({ ID }) { + const self = this; + if (this.isRunning) { + if (this.socketListeners.hasOwnProperty(String(ID))) { + self.socketListeners[String(ID)] = false; + } + } + } + + clearAllPackets({ TYPE }) { + const self = this; + if (this.menuRemap[String(TYPE)] == "all") { + self.socketData.gmsg = []; + self.socketData.pmsg = []; + self.socketData.direct = []; + self.socketData.statuscode = []; + self.socketData.gvar = []; + self.socketData.pvar = []; + } else { + self.socketData[this.menuRemap[String(TYPE)]] = []; + } + } + } + + console.log("CloudLink 4.0 loaded. Detecting unsandboxed mode."); + Scratch.extensions.register(new CloudLink(Scratch.vm.runtime)); })(Scratch); - diff --git a/extensions/codeGIO/ExtraUtilities.js b/extensions/codeGIO/ExtraUtilities.js index 3c7414dde4..a88fee3591 100644 --- a/extensions/codeGIO/ExtraUtilities.js +++ b/extensions/codeGIO/ExtraUtilities.js @@ -1,7 +1,7 @@ -(function(Scratch) { +(function (Scratch) { "use strict"; class codegioExtension { - getInfo () { + getInfo() { return { id: "utilitiesCodegio", name: "Utilities", @@ -11,7 +11,7 @@ { opcode: "newline", blockType: Scratch.BlockType.REPORTER, - text: "New Line" + text: "New Line", }, { @@ -21,25 +21,25 @@ arguments: { one: { type: Scratch.ArgumentType.STRING, - defaultValue: "" + defaultValue: "", }, two: { type: Scratch.ArgumentType.STRING, - defaultValue: "" - } - } + defaultValue: "", + }, + }, }, { opcode: "returntrue", blockType: Scratch.BlockType.BOOLEAN, - text: "true" + text: "true", }, { opcode: "returnfalse", blockType: Scratch.BlockType.BOOLEAN, - text: "false" + text: "false", }, { @@ -49,13 +49,13 @@ arguments: { one: { type: Scratch.ArgumentType.NUMBER, - defaultValue: "" + defaultValue: "", }, two: { type: Scratch.ArgumentType.NUMBER, - defaultValue: "" - } - } + defaultValue: "", + }, + }, }, { @@ -65,33 +65,33 @@ arguments: { color: { type: Scratch.ArgumentType.COLOR, - defaultValue: "#96ccff" - } - } + defaultValue: "#96ccff", + }, + }, }, { opcode: "monitor_width", blockType: Scratch.BlockType.REPORTER, - text: "Screen | Width" + text: "Screen | Width", }, { opcode: "monitor_height", blockType: Scratch.BlockType.REPORTER, - text: "Screen | Height" + text: "Screen | Height", }, { opcode: "window_width", blockType: Scratch.BlockType.REPORTER, - text: "Window | Width" + text: "Window | Width", }, { opcode: "window_height", blockType: Scratch.BlockType.REPORTER, - text: "Window | Height" + text: "Window | Height", }, { @@ -101,9 +101,9 @@ arguments: { one: { type: Scratch.ArgumentType.STRING, - defaultValue: "Alert..." - } - } + defaultValue: "Alert...", + }, + }, }, { @@ -113,9 +113,9 @@ arguments: { one: { type: Scratch.ArgumentType.STRING, - defaultValue: "Confirm..." - } - } + defaultValue: "Confirm...", + }, + }, }, { @@ -125,13 +125,13 @@ arguments: { one: { type: Scratch.ArgumentType.STRING, - defaultValue: "Enter Username:" + defaultValue: "Enter Username:", }, two: { type: Scratch.ArgumentType.STRING, - defaultValue: "griffpatch" + defaultValue: "griffpatch", }, - } + }, }, { @@ -141,9 +141,9 @@ arguments: { one: { type: Scratch.ArgumentType.STRING, - defaultValue: "https://turbowarp.org/" - } - } + defaultValue: "https://turbowarp.org/", + }, + }, }, { @@ -153,21 +153,21 @@ arguments: { one: { type: Scratch.ArgumentType.STRING, - defaultValue: "https://turbowarp.org/" - } - } + defaultValue: "https://turbowarp.org/", + }, + }, }, { opcode: "get_current_url", blockType: Scratch.BlockType.REPORTER, - text: "Current URL" + text: "Current URL", }, { opcode: "get_current_url_hash", blockType: Scratch.BlockType.REPORTER, - text: "Current URL hash (#)" + text: "Current URL hash (#)", }, { @@ -177,27 +177,27 @@ arguments: { one: { type: Scratch.ArgumentType.STRING, - defaultValue: "" - } - } + defaultValue: "", + }, + }, }, { opcode: "get_clipboard", blockType: Scratch.BlockType.REPORTER, - text: "Clipboard" + text: "Clipboard", }, { opcode: "get_browser", blockType: Scratch.BlockType.REPORTER, - text: "Browser" + text: "Browser", }, { opcode: "get_os", blockType: Scratch.BlockType.REPORTER, - text: "Operating System" + text: "Operating System", }, { @@ -212,7 +212,7 @@ font: { type: Scratch.ArgumentType.STRING, defaultValue: "Monospace", - menu: "consoleFonts" + menu: "consoleFonts", }, size: { type: Scratch.ArgumentType.NUMBER, @@ -222,25 +222,25 @@ type: Scratch.ArgumentType.COLOR, defaultValue: "#000000", }, - } + }, }, { opcode: "consoleClear", blockType: Scratch.BlockType.COMMAND, - text: "Console | Clear" + text: "Console | Clear", }, ], menus: { consoleFonts: { acceptReporters: true, items: [ - {text: "Serif (default)", value: "serif"}, - {text: "Monospace", value: "monospace"}, - {text: "Sans-serif", value: "sans-serif"} - ] - } - } + { text: "Serif (default)", value: "serif" }, + { text: "Monospace", value: "monospace" }, + { text: "Sans-serif", value: "sans-serif" }, + ], + }, + }, }; } @@ -257,7 +257,7 @@ } strict_equality(args) { - return (args.one == args.two); + return args.one == args.two; } exponent(args) { @@ -333,28 +333,28 @@ get_clipboard() { if (navigator.clipboard && navigator.clipboard.readText) { - return Scratch.canReadClipboard().then(allowed => { + return Scratch.canReadClipboard().then((allowed) => { if (allowed) { return navigator.clipboard.readText(); } - return ''; + return ""; }); } - return ''; + return ""; } get_browser() { let userAgent = navigator.userAgent; - if (userAgent.match(/chrome|chromium|crios/i)){ + if (userAgent.match(/chrome|chromium|crios/i)) { return "Chrome"; - } else if (userAgent.match(/firefox|fxios/i)){ + } else if (userAgent.match(/firefox|fxios/i)) { return "Firefox"; - } else if (userAgent.match(/safari/i)){ + } else if (userAgent.match(/safari/i)) { return "Safari"; - } else if (userAgent.match(/opr\//i)){ + } else if (userAgent.match(/opr\//i)) { return "Opera"; - } else if (userAgent.match(/edg/i)){ + } else if (userAgent.match(/edg/i)) { return "Edge"; } else { return "No browser detection"; @@ -366,7 +366,10 @@ } consoleLog(args) { - console.log(`%c${args.input}`, `color:${args.color}; font-family:${args.font}; font-size: ${args.size}px;`); + console.log( + `%c${args.input}`, + `color:${args.color}; font-family:${args.font}; font-size: ${args.size}px;` + ); } consoleClear() { diff --git a/extensions/cs2627883/numericalencoding.js b/extensions/cs2627883/numericalencoding.js index 0f23332264..5212283722 100644 --- a/extensions/cs2627883/numericalencoding.js +++ b/extensions/cs2627883/numericalencoding.js @@ -1,48 +1,54 @@ +// Name: Numerical Encoding +// ID: cs2627883NumericalEncoding +// Description: Encode strings as numbers for cloud variables. +// By: cs2627883 + // https://github.com/CS2627883/Turbowarp-Encoding-Extension/blob/main/Encoding.js -(function(Scratch) { - 'use strict'; +(function (Scratch) { + "use strict"; class NumericalEncodingExtension { maxcharlength = 6; // There are 149,186 unicode characters, so the maximum character code length is 6 encoded = 0; decoded = 0; getInfo() { return { - id: 'cs2627883NumericalEncoding', - name: 'Numerical Encoding', - blocks: [{ - opcode: 'NumericalEncode', - blockType: Scratch.BlockType.COMMAND, - text: 'Encode [DATA] to numbers', - arguments: { - DATA: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'Hello!' - } - } - }, + id: "cs2627883NumericalEncoding", + name: "Numerical Encoding", + blocks: [ { - opcode: 'NumericalDecode', + opcode: "NumericalEncode", blockType: Scratch.BlockType.COMMAND, - text: 'Decode [ENCODED] back to text', + text: "Encode [DATA] to numbers", + arguments: { + DATA: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Hello!", + }, + }, + }, + { + opcode: "NumericalDecode", + blockType: Scratch.BlockType.COMMAND, + text: "Decode [ENCODED] back to text", arguments: { ENCODED: { type: Scratch.ArgumentType.STRING, - defaultValue: '000072000101000108000108000111000033' //Encoded "Hello!" - } - } + defaultValue: "000072000101000108000108000111000033", //Encoded "Hello!" + }, + }, }, { - opcode: 'GetNumericalEncoded', + opcode: "GetNumericalEncoded", blockType: Scratch.BlockType.REPORTER, - text: 'encoded', + text: "encoded", }, { - opcode: 'GetNumericalDecoded', + opcode: "GetNumericalDecoded", blockType: Scratch.BlockType.REPORTER, - text: 'decoded', - } - ] + text: "decoded", + }, + ], }; } NumericalEncode(args) { @@ -52,7 +58,8 @@ // Get char code of character var encodedchar = String(toencode.charCodeAt(i)); // Pad encodedchar with 0s to ensure all encodedchars are the same length - encodedchar = "0".repeat(this.maxcharlength - encodedchar.length) + encodedchar; + encodedchar = + "0".repeat(this.maxcharlength - encodedchar.length) + encodedchar; encoded += encodedchar; } this.encoded = encoded; @@ -65,7 +72,7 @@ } var decoded = ""; // Create regex to split by char length - const regex = new RegExp('.{1,' + this.maxcharlength + '}', 'g'); + const regex = new RegExp(".{1," + this.maxcharlength + "}", "g"); // Split into array of characters var encodedchars = todecode.match(regex); for (let i = 0; i < encodedchars.length; i++) { diff --git a/extensions/cursor.js b/extensions/cursor.js index 4b9420e9f8..3fdf7c47ca 100644 --- a/extensions/cursor.js +++ b/extensions/cursor.js @@ -1,286 +1,323 @@ -(function (Scratch) { - 'use strict'; - - if (!Scratch.extensions.unsandboxed) { - throw new Error('MouseCursor extension must be run unsandboxed'); - } - - const lazilyCreatedCanvas = () => { - /** @type {HTMLCanvasElement} */ - let canvas = null; - /** @type {CanvasRenderingContext2D} */ - let ctx = null; - /** - * @param {number} width - * @param {number} height - * @returns {[HTMLCanvasElement, CanvasRenderingContext2D]} - */ - return (width, height) => { - if (!canvas) { - canvas = document.createElement('canvas'); - ctx = canvas.getContext('2d'); - if (!ctx) { - throw new Error('Could not get 2d rendering context'); - } - } - // Setting canvas size also clears it - canvas.width = width; - canvas.height = height; - return [canvas, ctx]; - }; - }; - const getRawSkinCanvas = lazilyCreatedCanvas(); - - /** - * @param {RenderWebGL.Skin} skin - * @returns {string} A data: URI for the skin. - */ - const encodeSkinToURL = (skin) => { - const svgSkin = /** @type {RenderWebGL.SVGSkin} */ (skin); - if (svgSkin._svgImage) { - // This is an SVG skin - return svgSkin._svgImage.src; - } - - // It's probably a bitmap skin. - // The most reliable way to get the bitmap in every runtime is through the silhouette. - // This is very slow and could involve reading the texture from the GPU. - const silhouette = skin._silhouette; - // unlazy() only exists in TW - if (silhouette.unlazy) { - silhouette.unlazy(); - } - const colorData = silhouette._colorData; - const width = silhouette._width; - const height = silhouette._height; - const imageData = new ImageData(colorData, silhouette._width, silhouette._height); - const [canvas, ctx] = getRawSkinCanvas(width, height); - ctx.putImageData(imageData, 0, 0); - return canvas.toDataURL(); - }; - - /** - * @param {VM.Costume} costume - * @param {number} maxWidth - * @param {number} maxHeight - * @returns {{uri: string, width: number, height: number}} - */ - const costumeToCursor = (costume, maxWidth, maxHeight) => { - const skin = Scratch.vm.renderer._allSkins[costume.skinId]; - const imageURI = encodeSkinToURL(skin); - - let width = skin.size[0]; - let height = skin.size[1]; - if (width > maxWidth) { - height = height * (maxWidth / width); - width = maxWidth; - } - if (height > maxHeight) { - width = width * (maxHeight / height); - height = maxHeight; - } - width = Math.round(width); - height = Math.round(height); - - // We wrap the encoded image in an . This lets us do some clever things: - // - We can resize the image without a canvas. - // - We can give the browser an image with more raw pixels than its DPI independent size. - // The latter is important so that cursors won't look horrible on high DPI displays. For - // example, if the cursor will display at 32x32 in DPI independent units on a 2x high DPI - // display, we actually need to send a 64x64 image for it to look good. This lets us do - // that automatically. - let svg = ``; - svg += ``; - svg += ''; - // URI encoding usually results in smaller string than base 64 for the types of data we get here. - const svgURI = `data:image/svg+xml;,${encodeURIComponent(svg)}`; - - return { - uri: svgURI, - width, - height - }; - }; - - /** @type {string} */ - let nativeCursor = 'default'; - /** @type {null|string} */ - let customCursorImageName = null; - - const canvas = Scratch.renderer.canvas; - /** @type {string} */ - let currentCanvasCursor = nativeCursor; - const updateCanvasCursor = () => { - if (canvas.style.cursor !== currentCanvasCursor) { - canvas.style.cursor = currentCanvasCursor; - } - }; - - // scratch-gui will sometimes reset the cursor when resizing the window or going in/out of fullscreen - new MutationObserver(updateCanvasCursor).observe(canvas, { - attributeFilter: ['style'], - attributes: true - }); - - /** - * Parse strings like "60x12" or "77,1" - * @param {string} string - * @returns {[number, number]} - */ - const parseTuple = (string) => { - const [a, b] = ('' + string).split(/[ ,x]/); - return [ - +a || 0, - +b || 0 - ]; - }; - - const cursors = [ - 'default', 'pointer', 'move', 'grab', 'grabbing', 'text', - 'vertical-text', 'wait', 'progress', 'help', 'context-menu', - 'zoom-in', 'zoom-out', 'crosshair', 'cell', 'not-allowed', - 'copy', 'alias', 'no-drop', 'all-scroll', 'col-resize', - 'row-resize', 'n-resize', 'e-resize', 's-resize', 'w-resize', - 'ne-resize', 'nw-resize', 'se-resize', 'sw-resize', - 'ew-resize', 'ns-resize', 'nesw-resize', 'nwse-resize' - ]; - - class MouseCursor { - getInfo() { - return { - id: 'MouseCursor', - name: 'Mouse Cursor', - blocks: [ - { - opcode: 'setCur', - blockType: Scratch.BlockType.COMMAND, - text: 'set cursor to [cur]', - arguments: { - cur: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'pointer', - menu: 'cursors', - }, - }, - }, - { - opcode: 'setCursorImage', - blockType: Scratch.BlockType.COMMAND, - text: "set cursor to current costume center: [position] max size: [size]", - arguments: { - position: { - type: Scratch.ArgumentType.STRING, - defaultValue: '0,0', - menu: 'imagePositions' - }, - size: { - type: Scratch.ArgumentType.STRING, - defaultValue: '32x32', - menu: 'imageSizes' - } - } - }, - { - opcode: 'hideCur', - blockType: Scratch.BlockType.COMMAND, - text: 'hide cursor', - }, - { - opcode: 'getCur', - blockType: Scratch.BlockType.REPORTER, - text: 'cursor', - }, - ], - menus: { - cursors: { - acceptReporters: true, - items: cursors, - }, - imagePositions: { - acceptReporters: true, - items: [ - // [x, y] where x is [0=left, 100=right] and y is [0=top, 100=bottom] - { text: 'top left', value: '0,0' }, - { text: 'top right', value: '100,0' }, - { text: 'bottom left', value: '0,100' }, - { text: 'bottom right', value: '100,100' }, - { text: 'center', value: '50,50' }, - ] - }, - imageSizes: { - acceptReporters: true, - items: [ - // Some important numbers to keep in mind: - // Browsers ignore cursor images >128 in any dimension (https://searchfox.org/mozilla-central/rev/43ee5e789b079e94837a21336e9ce2420658fd19/widget/gtk/nsWindow.cpp#3393-3402) - // Browsers may refuse to display a cursor near window borders for images >32 in any dimension - { text: '4x4', value: '4x4' }, - { text: '8x8', value: '8x4' }, - { text: '12x12', value: '12x12' }, - { text: '16x16', value: '16x16' }, - { text: '32x32', value: '32x32' }, - { text: '48x48 (unreliable)', value: '48x48' }, - { text: '64x64 (unreliable)', value: '64x64' }, - { text: '128x128 (unreliable)', value: '128x128' }, - ] - } - }, - }; - } - - setCur(args) { - const newCursor = Scratch.Cast.toString(args.cur); - // Prevent setting cursor to "url(...), default" from causing fetch. - if (cursors.includes(newCursor) || newCursor === 'none') { - nativeCursor = newCursor; - customCursorImageName = null; - currentCanvasCursor = newCursor; - updateCanvasCursor(); - } - } - - setCursorImage(args, util) { - const [maxWidth, maxHeight] = parseTuple(args.size).map(i => Math.max(0, i)); - - const currentCostume = util.target.getCostumes()[util.target.currentCostume]; - const costumeName = currentCostume.name; - - let encodedCostume; - try { - encodedCostume = costumeToCursor(currentCostume, maxWidth, maxHeight); - } catch (e) { - // This could happen for a variety of reasons. - console.error(e); - } - - if (encodedCostume) { - const [percentX, percentY] = parseTuple(args.position).map(i => Math.max(0, Math.min(100, i)) / 100); - const x = percentX * encodedCostume.width; - const y = percentY * encodedCostume.height; - - currentCanvasCursor = `url("${encodedCostume.uri}") ${x} ${y}, ${nativeCursor}`; - updateCanvasCursor(); - } else { - // If for some reason the costume couldn't be encoded, we'll leave the cursor unchanged. - // This is the same behavior that would happen if we successfully encode a cursor but the browser - // is unable to parse it for some reason. - } - - customCursorImageName = costumeName; - } - - hideCur() { - this.setCur({ - cur: 'none' - }); - } - - getCur() { - if (customCursorImageName !== null) { - return customCursorImageName; - } - return nativeCursor; - } - } - - Scratch.extensions.register(new MouseCursor()); -})(Scratch); +// Name: Mouse Cursor +// ID: MouseCursor +// Description: Use custom cursors or hide the cursor. Also allows replacing the cursor with any costume image. + +(function (Scratch) { + "use strict"; + + if (!Scratch.extensions.unsandboxed) { + throw new Error("MouseCursor extension must be run unsandboxed"); + } + + const lazilyCreatedCanvas = () => { + /** @type {HTMLCanvasElement} */ + let canvas = null; + /** @type {CanvasRenderingContext2D} */ + let ctx = null; + /** + * @param {number} width + * @param {number} height + * @returns {[HTMLCanvasElement, CanvasRenderingContext2D]} + */ + return (width, height) => { + if (!canvas) { + canvas = document.createElement("canvas"); + ctx = canvas.getContext("2d"); + if (!ctx) { + throw new Error("Could not get 2d rendering context"); + } + } + // Setting canvas size also clears it + canvas.width = width; + canvas.height = height; + return [canvas, ctx]; + }; + }; + const getRawSkinCanvas = lazilyCreatedCanvas(); + + /** + * @param {RenderWebGL.Skin} skin + * @returns {string} A data: URI for the skin. + */ + const encodeSkinToURL = (skin) => { + const svgSkin = /** @type {RenderWebGL.SVGSkin} */ (skin); + if (svgSkin._svgImage) { + // This is an SVG skin + return svgSkin._svgImage.src; + } + + // It's probably a bitmap skin. + // The most reliable way to get the bitmap in every runtime is through the silhouette. + // This is very slow and could involve reading the texture from the GPU. + const silhouette = skin._silhouette; + // unlazy() only exists in TW + if (silhouette.unlazy) { + silhouette.unlazy(); + } + const colorData = silhouette._colorData; + const width = silhouette._width; + const height = silhouette._height; + const imageData = new ImageData( + colorData, + silhouette._width, + silhouette._height + ); + const [canvas, ctx] = getRawSkinCanvas(width, height); + ctx.putImageData(imageData, 0, 0); + return canvas.toDataURL(); + }; + + /** + * @param {VM.Costume} costume + * @param {number} maxWidth + * @param {number} maxHeight + * @returns {{uri: string, width: number, height: number}} + */ + const costumeToCursor = (costume, maxWidth, maxHeight) => { + const skin = Scratch.vm.renderer._allSkins[costume.skinId]; + const imageURI = encodeSkinToURL(skin); + + let width = skin.size[0]; + let height = skin.size[1]; + if (width > maxWidth) { + height = height * (maxWidth / width); + width = maxWidth; + } + if (height > maxHeight) { + width = width * (maxHeight / height); + height = maxHeight; + } + width = Math.round(width); + height = Math.round(height); + + // We wrap the encoded image in an . This lets us do some clever things: + // - We can resize the image without a canvas. + // - We can give the browser an image with more raw pixels than its DPI independent size. + // The latter is important so that cursors won't look horrible on high DPI displays. For + // example, if the cursor will display at 32x32 in DPI independent units on a 2x high DPI + // display, we actually need to send a 64x64 image for it to look good. This lets us do + // that automatically. + let svg = ``; + svg += ``; + svg += ""; + // URI encoding usually results in smaller string than base 64 for the types of data we get here. + const svgURI = `data:image/svg+xml;,${encodeURIComponent(svg)}`; + + return { + uri: svgURI, + width, + height, + }; + }; + + /** @type {string} */ + let nativeCursor = "default"; + /** @type {null|string} */ + let customCursorImageName = null; + + const canvas = Scratch.renderer.canvas; + /** @type {string} */ + let currentCanvasCursor = nativeCursor; + const updateCanvasCursor = () => { + if (canvas.style.cursor !== currentCanvasCursor) { + canvas.style.cursor = currentCanvasCursor; + } + }; + + // scratch-gui will sometimes reset the cursor when resizing the window or going in/out of fullscreen + new MutationObserver(updateCanvasCursor).observe(canvas, { + attributeFilter: ["style"], + attributes: true, + }); + + /** + * Parse strings like "60x12" or "77,1" + * @param {string} string + * @returns {[number, number]} + */ + const parseTuple = (string) => { + const [a, b] = ("" + string).split(/[ ,x]/); + return [+a || 0, +b || 0]; + }; + + const cursors = [ + "default", + "pointer", + "move", + "grab", + "grabbing", + "text", + "vertical-text", + "wait", + "progress", + "help", + "context-menu", + "zoom-in", + "zoom-out", + "crosshair", + "cell", + "not-allowed", + "copy", + "alias", + "no-drop", + "all-scroll", + "col-resize", + "row-resize", + "n-resize", + "e-resize", + "s-resize", + "w-resize", + "ne-resize", + "nw-resize", + "se-resize", + "sw-resize", + "ew-resize", + "ns-resize", + "nesw-resize", + "nwse-resize", + ]; + + class MouseCursor { + getInfo() { + return { + id: "MouseCursor", + name: "Mouse Cursor", + blocks: [ + { + opcode: "setCur", + blockType: Scratch.BlockType.COMMAND, + text: "set cursor to [cur]", + arguments: { + cur: { + type: Scratch.ArgumentType.STRING, + defaultValue: "pointer", + menu: "cursors", + }, + }, + }, + { + opcode: "setCursorImage", + blockType: Scratch.BlockType.COMMAND, + text: "set cursor to current costume center: [position] max size: [size]", + arguments: { + position: { + type: Scratch.ArgumentType.STRING, + defaultValue: "0,0", + menu: "imagePositions", + }, + size: { + type: Scratch.ArgumentType.STRING, + defaultValue: "32x32", + menu: "imageSizes", + }, + }, + }, + { + opcode: "hideCur", + blockType: Scratch.BlockType.COMMAND, + text: "hide cursor", + }, + { + opcode: "getCur", + blockType: Scratch.BlockType.REPORTER, + text: "cursor", + }, + ], + menus: { + cursors: { + acceptReporters: true, + items: cursors, + }, + imagePositions: { + acceptReporters: true, + items: [ + // [x, y] where x is [0=left, 100=right] and y is [0=top, 100=bottom] + { text: "top left", value: "0,0" }, + { text: "top right", value: "100,0" }, + { text: "bottom left", value: "0,100" }, + { text: "bottom right", value: "100,100" }, + { text: "center", value: "50,50" }, + ], + }, + imageSizes: { + acceptReporters: true, + items: [ + // Some important numbers to keep in mind: + // Browsers ignore cursor images >128 in any dimension (https://searchfox.org/mozilla-central/rev/43ee5e789b079e94837a21336e9ce2420658fd19/widget/gtk/nsWindow.cpp#3393-3402) + // Browsers may refuse to display a cursor near window borders for images >32 in any dimension + { text: "4x4", value: "4x4" }, + { text: "8x8", value: "8x4" }, + { text: "12x12", value: "12x12" }, + { text: "16x16", value: "16x16" }, + { text: "32x32", value: "32x32" }, + { text: "48x48 (unreliable)", value: "48x48" }, + { text: "64x64 (unreliable)", value: "64x64" }, + { text: "128x128 (unreliable)", value: "128x128" }, + ], + }, + }, + }; + } + + setCur(args) { + const newCursor = Scratch.Cast.toString(args.cur); + // Prevent setting cursor to "url(...), default" from causing fetch. + if (cursors.includes(newCursor) || newCursor === "none") { + nativeCursor = newCursor; + customCursorImageName = null; + currentCanvasCursor = newCursor; + updateCanvasCursor(); + } + } + + setCursorImage(args, util) { + const [maxWidth, maxHeight] = parseTuple(args.size).map((i) => + Math.max(0, i) + ); + + const currentCostume = + util.target.getCostumes()[util.target.currentCostume]; + const costumeName = currentCostume.name; + + let encodedCostume; + try { + encodedCostume = costumeToCursor(currentCostume, maxWidth, maxHeight); + } catch (e) { + // This could happen for a variety of reasons. + console.error(e); + } + + if (encodedCostume) { + const [percentX, percentY] = parseTuple(args.position).map( + (i) => Math.max(0, Math.min(100, i)) / 100 + ); + const x = percentX * encodedCostume.width; + const y = percentY * encodedCostume.height; + + currentCanvasCursor = `url("${encodedCostume.uri}") ${x} ${y}, ${nativeCursor}`; + updateCanvasCursor(); + } else { + // If for some reason the costume couldn't be encoded, we'll leave the cursor unchanged. + // This is the same behavior that would happen if we successfully encode a cursor but the browser + // is unable to parse it for some reason. + } + + customCursorImageName = costumeName; + } + + hideCur() { + this.setCur({ + cur: "none", + }); + } + + getCur() { + if (customCursorImageName !== null) { + return customCursorImageName; + } + return nativeCursor; + } + } + + Scratch.extensions.register(new MouseCursor()); +})(Scratch); diff --git a/extensions/docs-examples/unsandboxed/broadcast-5.js b/extensions/docs-examples/unsandboxed/broadcast-5.js index 40c08fd40a..1e08bff12b 100644 --- a/extensions/docs-examples/unsandboxed/broadcast-5.js +++ b/extensions/docs-examples/unsandboxed/broadcast-5.js @@ -8,7 +8,7 @@ blocks: [ { opcode: 'whenReceived', - blockType: Scratch.BlockType.HAT, + blockType: Scratch.BlockType.EVENT, text: 'when I receive [EVENT_OPTION]', isEdgeActivated: false, arguments: { @@ -22,7 +22,6 @@ opcode: 'broadcast', blockType: Scratch.BlockType.REPORTER, text: 'broadcast [EVENT]', - disableMonitor: true, arguments: { EVENT: { type: Scratch.ArgumentType.STRING, diff --git a/extensions/docs-examples/unsandboxed/every-second.js b/extensions/docs-examples/unsandboxed/every-second.js index 647ff595e2..56356bcbd0 100644 --- a/extensions/docs-examples/unsandboxed/every-second.js +++ b/extensions/docs-examples/unsandboxed/every-second.js @@ -10,13 +10,7 @@ opcode: 'everySecond', blockType: Scratch.BlockType.HAT, text: 'every second', - isEdgeActivated: false, - arguments: { - EVENT_OPTION: { - type: Scratch.ArgumentType.STRING, - menu: 'EVENT_FIELD' - } - } + isEdgeActivated: false } ] }; diff --git a/extensions/docs-examples/unsandboxed/when-key-pressed-restart.js b/extensions/docs-examples/unsandboxed/when-key-pressed-restart.js new file mode 100644 index 0000000000..6f909400e6 --- /dev/null +++ b/extensions/docs-examples/unsandboxed/when-key-pressed-restart.js @@ -0,0 +1,55 @@ +(function(Scratch) { + 'use strict'; + + if (!Scratch.extensions.unsandboxed) { + throw new Error('This example must run unsandboxed'); + } + + class WhenKeyPressed { + getInfo() { + return { + id: 'restartexampleunsandboxed', + name: 'Restart Threads Example', + blocks: [ + { + blockType: Scratch.BlockType.EVENT, + opcode: 'whenPressed', + text: 'when [KEY] key pressed', + isEdgeActivated: false, + // highlight-next-line + shouldRestartExistingThreads: true, + arguments: { + KEY: { + type: Scratch.ArgumentType.STRING, + menu: 'key' + } + } + } + ], + menus: { + key: { + acceptReporters: false, + items: [ + { + text: 'space', + value: ' ' + }, + 'a', + 'b', + 'c', + // ... + ] + } + } + }; + } + } + + document.addEventListener('keydown', (e) => { + Scratch.vm.runtime.startHats('restartexampleunsandboxed_whenPressed', { + KEY: e.key + }); + }); + + Scratch.extensions.register(new WhenKeyPressed()); +})(Scratch); diff --git a/extensions/docs-examples/unsandboxed/when-key-pressed-stage.js b/extensions/docs-examples/unsandboxed/when-key-pressed-stage.js new file mode 100644 index 0000000000..ee22bbe9f4 --- /dev/null +++ b/extensions/docs-examples/unsandboxed/when-key-pressed-stage.js @@ -0,0 +1,54 @@ +(function(Scratch) { + 'use strict'; + + if (!Scratch.extensions.unsandboxed) { + throw new Error('This example must run unsandboxed'); + } + + class WhenKeyPressedInStage { + getInfo() { + return { + id: 'eventexample3unsandboxed', + name: 'Event Block Example 3', + blocks: [ + { + blockType: Scratch.BlockType.EVENT, + opcode: 'whenPressed', + text: 'when [KEY] key pressed', + isEdgeActivated: false, + arguments: { + KEY: { + type: Scratch.ArgumentType.STRING, + menu: 'key' + } + } + } + ], + menus: { + key: { + acceptReporters: false, + items: [ + { + text: 'space', + value: ' ' + }, + 'a', + 'b', + 'c', + // ... + ] + } + } + }; + } + } + + document.addEventListener('keydown', (e) => { + Scratch.vm.runtime.startHats('eventexample3unsandboxed_whenPressed', { + KEY: e.key + // highlight-next-line + }, Scratch.vm.runtime.getTargetForStage()); + }); + + Scratch.extensions.register(new WhenKeyPressedInStage()); +})(Scratch); diff --git a/extensions/docs-examples/unsandboxed/when-key-pressed.js b/extensions/docs-examples/unsandboxed/when-key-pressed.js new file mode 100644 index 0000000000..e91b5103b3 --- /dev/null +++ b/extensions/docs-examples/unsandboxed/when-key-pressed.js @@ -0,0 +1,58 @@ +(function(Scratch) { + 'use strict'; + + if (!Scratch.extensions.unsandboxed) { + throw new Error('This example must run unsandboxed'); + } + + class WhenKeyPressed { + getInfo() { + return { + id: 'eventexample2unsandboxed', + name: 'Event Block Example 2', + blocks: [ + { + blockType: Scratch.BlockType.EVENT, + opcode: 'whenPressed', + text: 'when [KEY] key pressed', + isEdgeActivated: false, // required boilerplate + // highlight-start + arguments: { + KEY: { + type: Scratch.ArgumentType.STRING, + menu: 'key' + } + } + // highlight-end + } + ], + menus: { + key: { + acceptReporters: false, + items: [ + { + // startHats filters by *value*, not by text + text: 'space', + value: ' ' + }, + 'a', + 'b', + 'c', + // ... + ] + } + } + }; + } + } + + document.addEventListener('keydown', (e) => { + // highlight-start + Scratch.vm.runtime.startHats('eventexample2unsandboxed_whenPressed', { + KEY: e.key + }); + // highlight-end + }); + + Scratch.extensions.register(new WhenKeyPressed()); +})(Scratch); diff --git a/extensions/docs-examples/unsandboxed/when-space-key-pressed.js b/extensions/docs-examples/unsandboxed/when-space-key-pressed.js new file mode 100644 index 0000000000..da2d5f17ea --- /dev/null +++ b/extensions/docs-examples/unsandboxed/when-space-key-pressed.js @@ -0,0 +1,37 @@ +(function(Scratch) { + 'use strict'; + + if (!Scratch.extensions.unsandboxed) { + throw new Error('This example must run unsandboxed'); + } + + class WhenSpaceKeyPressed { + getInfo() { + return { + id: 'eventexampleunsandboxed', + name: 'Event Block Example', + blocks: [ + // highlight-start + { + blockType: Scratch.BlockType.EVENT, + opcode: 'whenSpacePressed', + text: 'when space key pressed', + isEdgeActivated: false // required boilerplate + } + // highlight-end + ] + }; + } + // Notice: whenSpacePressed does not have a function defined! + } + + // highlight-start + document.addEventListener('keydown', (e) => { + if (e.key === ' ') { + Scratch.vm.runtime.startHats('eventexampleunsandboxed_whenSpacePressed'); + } + }); + // highlight-end + + Scratch.extensions.register(new WhenSpaceKeyPressed()); +})(Scratch); diff --git a/extensions/docs-examples/unsandboxed/when.js b/extensions/docs-examples/unsandboxed/when.js new file mode 100644 index 0000000000..ea6fa2f2c3 --- /dev/null +++ b/extensions/docs-examples/unsandboxed/when.js @@ -0,0 +1,45 @@ +(function(Scratch) { + 'use strict'; + + if (!Scratch.extensions.unsandboxed) { + throw new Error('This example must run unsandboxed'); + } + + class When { + getInfo() { + return { + id: 'whenunsandboxed', + name: 'When', + blocks: [ + { + // highlight-start + blockType: Scratch.BlockType.HAT, + opcode: 'when', + text: 'when [CONDITION]', + isEdgeActivated: false, // required boilerplate + arguments: { + CONDITION: { + type: Scratch.BlockType.BOOLEAN + } + } + // highlight-end + } + ] + }; + } + // highlight-start + when(args) { + return Scratch.Cast.toBoolean(args.CONDITION); + } + // highlight-end + } + + // highlight-start + Scratch.vm.runtime.on('BEFORE_EXECUTE', () => { + // startHats is the same as before! + Scratch.vm.runtime.startHats('whenunsandboxed_when'); + }); + // highlight-end + + Scratch.extensions.register(new When()); +})(Scratch); diff --git a/extensions/encoding.js b/extensions/encoding.js index 9613e965c7..f506508052 100644 --- a/extensions/encoding.js +++ b/extensions/encoding.js @@ -1,618 +1,634 @@ -(function (Scratch) { - 'use strict'; - const icon = ''; - const icon2 = ''; - - /*! - This md5 function is based on https://github.com/blueimp/JavaScript-MD5/blob/master/js/md5.js - which is licensed under: - - MIT License - - Copyright © 2011 Sebastian Tschan, https://blueimp.net - - Permission is hereby granted, free of charge, to any person obtaining a copy of - this software and associated documentation files (the "Software"), to deal in - the Software without restriction, including without limitation the rights to - use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of - the Software, and to permit persons to whom the Software is furnished to do so, - subject to the following conditions: - - The above copyright notice and this permission notice shall be included in all - copies or substantial portions of the Software. - - THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR - IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS - FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR - COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER - IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN - CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. - */ - /* eslint-disable */ - const md5 = (function () { - /** - * Add integers, wrapping at 2^32. - * This uses 16-bit operations internally to work around bugs in interpreters. - * - * @param {number} x First integer - * @param {number} y Second integer - * @returns {number} Sum - */ - function safeAdd(x, y) { - var lsw = (x & 0xffff) + (y & 0xffff) - var msw = (x >> 16) + (y >> 16) + (lsw >> 16) - return (msw << 16) | (lsw & 0xffff) - } - - /** - * Bitwise rotate a 32-bit number to the left. - * - * @param {number} num 32-bit number - * @param {number} cnt Rotation count - * @returns {number} Rotated number - */ - function bitRotateLeft(num, cnt) { - return (num << cnt) | (num >>> (32 - cnt)) - } - - /** - * Basic operation the algorithm uses. - * - * @param {number} q q - * @param {number} a a - * @param {number} b b - * @param {number} x x - * @param {number} s s - * @param {number} t t - * @returns {number} Result - */ - function md5cmn(q, a, b, x, s, t) { - return safeAdd(bitRotateLeft(safeAdd(safeAdd(a, q), safeAdd(x, t)), s), b) - } - /** - * Basic operation the algorithm uses. - * - * @param {number} a a - * @param {number} b b - * @param {number} c c - * @param {number} d d - * @param {number} x x - * @param {number} s s - * @param {number} t t - * @returns {number} Result - */ - function md5ff(a, b, c, d, x, s, t) { - return md5cmn((b & c) | (~b & d), a, b, x, s, t) - } - /** - * Basic operation the algorithm uses. - * - * @param {number} a a - * @param {number} b b - * @param {number} c c - * @param {number} d d - * @param {number} x x - * @param {number} s s - * @param {number} t t - * @returns {number} Result - */ - function md5gg(a, b, c, d, x, s, t) { - return md5cmn((b & d) | (c & ~d), a, b, x, s, t) - } - /** - * Basic operation the algorithm uses. - * - * @param {number} a a - * @param {number} b b - * @param {number} c c - * @param {number} d d - * @param {number} x x - * @param {number} s s - * @param {number} t t - * @returns {number} Result - */ - function md5hh(a, b, c, d, x, s, t) { - return md5cmn(b ^ c ^ d, a, b, x, s, t) - } - /** - * Basic operation the algorithm uses. - * - * @param {number} a a - * @param {number} b b - * @param {number} c c - * @param {number} d d - * @param {number} x x - * @param {number} s s - * @param {number} t t - * @returns {number} Result - */ - function md5ii(a, b, c, d, x, s, t) { - return md5cmn(c ^ (b | ~d), a, b, x, s, t) - } - - /** - * Calculate the MD5 of an array of little-endian words, and a bit length. - * - * @param {Array} x Array of little-endian words - * @param {number} len Bit length - * @returns {Array} MD5 Array - */ - function binlMD5(x, len) { - /* append padding */ - x[len >> 5] |= 0x80 << len % 32 - x[(((len + 64) >>> 9) << 4) + 14] = len - - var i - var olda - var oldb - var oldc - var oldd - var a = 1732584193 - var b = -271733879 - var c = -1732584194 - var d = 271733878 - - for (i = 0; i < x.length; i += 16) { - olda = a - oldb = b - oldc = c - oldd = d - - a = md5ff(a, b, c, d, x[i], 7, -680876936) - d = md5ff(d, a, b, c, x[i + 1], 12, -389564586) - c = md5ff(c, d, a, b, x[i + 2], 17, 606105819) - b = md5ff(b, c, d, a, x[i + 3], 22, -1044525330) - a = md5ff(a, b, c, d, x[i + 4], 7, -176418897) - d = md5ff(d, a, b, c, x[i + 5], 12, 1200080426) - c = md5ff(c, d, a, b, x[i + 6], 17, -1473231341) - b = md5ff(b, c, d, a, x[i + 7], 22, -45705983) - a = md5ff(a, b, c, d, x[i + 8], 7, 1770035416) - d = md5ff(d, a, b, c, x[i + 9], 12, -1958414417) - c = md5ff(c, d, a, b, x[i + 10], 17, -42063) - b = md5ff(b, c, d, a, x[i + 11], 22, -1990404162) - a = md5ff(a, b, c, d, x[i + 12], 7, 1804603682) - d = md5ff(d, a, b, c, x[i + 13], 12, -40341101) - c = md5ff(c, d, a, b, x[i + 14], 17, -1502002290) - b = md5ff(b, c, d, a, x[i + 15], 22, 1236535329) - - a = md5gg(a, b, c, d, x[i + 1], 5, -165796510) - d = md5gg(d, a, b, c, x[i + 6], 9, -1069501632) - c = md5gg(c, d, a, b, x[i + 11], 14, 643717713) - b = md5gg(b, c, d, a, x[i], 20, -373897302) - a = md5gg(a, b, c, d, x[i + 5], 5, -701558691) - d = md5gg(d, a, b, c, x[i + 10], 9, 38016083) - c = md5gg(c, d, a, b, x[i + 15], 14, -660478335) - b = md5gg(b, c, d, a, x[i + 4], 20, -405537848) - a = md5gg(a, b, c, d, x[i + 9], 5, 568446438) - d = md5gg(d, a, b, c, x[i + 14], 9, -1019803690) - c = md5gg(c, d, a, b, x[i + 3], 14, -187363961) - b = md5gg(b, c, d, a, x[i + 8], 20, 1163531501) - a = md5gg(a, b, c, d, x[i + 13], 5, -1444681467) - d = md5gg(d, a, b, c, x[i + 2], 9, -51403784) - c = md5gg(c, d, a, b, x[i + 7], 14, 1735328473) - b = md5gg(b, c, d, a, x[i + 12], 20, -1926607734) - - a = md5hh(a, b, c, d, x[i + 5], 4, -378558) - d = md5hh(d, a, b, c, x[i + 8], 11, -2022574463) - c = md5hh(c, d, a, b, x[i + 11], 16, 1839030562) - b = md5hh(b, c, d, a, x[i + 14], 23, -35309556) - a = md5hh(a, b, c, d, x[i + 1], 4, -1530992060) - d = md5hh(d, a, b, c, x[i + 4], 11, 1272893353) - c = md5hh(c, d, a, b, x[i + 7], 16, -155497632) - b = md5hh(b, c, d, a, x[i + 10], 23, -1094730640) - a = md5hh(a, b, c, d, x[i + 13], 4, 681279174) - d = md5hh(d, a, b, c, x[i], 11, -358537222) - c = md5hh(c, d, a, b, x[i + 3], 16, -722521979) - b = md5hh(b, c, d, a, x[i + 6], 23, 76029189) - a = md5hh(a, b, c, d, x[i + 9], 4, -640364487) - d = md5hh(d, a, b, c, x[i + 12], 11, -421815835) - c = md5hh(c, d, a, b, x[i + 15], 16, 530742520) - b = md5hh(b, c, d, a, x[i + 2], 23, -995338651) - - a = md5ii(a, b, c, d, x[i], 6, -198630844) - d = md5ii(d, a, b, c, x[i + 7], 10, 1126891415) - c = md5ii(c, d, a, b, x[i + 14], 15, -1416354905) - b = md5ii(b, c, d, a, x[i + 5], 21, -57434055) - a = md5ii(a, b, c, d, x[i + 12], 6, 1700485571) - d = md5ii(d, a, b, c, x[i + 3], 10, -1894986606) - c = md5ii(c, d, a, b, x[i + 10], 15, -1051523) - b = md5ii(b, c, d, a, x[i + 1], 21, -2054922799) - a = md5ii(a, b, c, d, x[i + 8], 6, 1873313359) - d = md5ii(d, a, b, c, x[i + 15], 10, -30611744) - c = md5ii(c, d, a, b, x[i + 6], 15, -1560198380) - b = md5ii(b, c, d, a, x[i + 13], 21, 1309151649) - a = md5ii(a, b, c, d, x[i + 4], 6, -145523070) - d = md5ii(d, a, b, c, x[i + 11], 10, -1120210379) - c = md5ii(c, d, a, b, x[i + 2], 15, 718787259) - b = md5ii(b, c, d, a, x[i + 9], 21, -343485551) - - a = safeAdd(a, olda) - b = safeAdd(b, oldb) - c = safeAdd(c, oldc) - d = safeAdd(d, oldd) - } - return [a, b, c, d] - } - - /** - * Convert an array of little-endian words to a string - * - * @param {Array} input MD5 Array - * @returns {string} MD5 string - */ - function binl2rstr(input) { - var i - var output = '' - var length32 = input.length * 32 - for (i = 0; i < length32; i += 8) { - output += String.fromCharCode((input[i >> 5] >>> i % 32) & 0xff) - } - return output - } - - /** - * Convert a raw string to an array of little-endian words - * Characters >255 have their high-byte silently ignored. - * - * @param {string} input Raw input string - * @returns {Array} Array of little-endian words - */ - function rstr2binl(input) { - var i - var output = [] - output[(input.length >> 2) - 1] = undefined - for (i = 0; i < output.length; i += 1) { - output[i] = 0 - } - var length8 = input.length * 8 - for (i = 0; i < length8; i += 8) { - output[i >> 5] |= (input.charCodeAt(i / 8) & 0xff) << i % 32 - } - return output - } - - /** - * Calculate the MD5 of a raw string - * - * @param {string} s Input string - * @returns {string} Raw MD5 string - */ - function rstrMD5(s) { - return binl2rstr(binlMD5(rstr2binl(s), s.length * 8)) - } - - /** - * Calculates the HMAC-MD5 of a key and some data (raw strings) - * - * @param {string} key HMAC key - * @param {string} data Raw input string - * @returns {string} Raw MD5 string - */ - function rstrHMACMD5(key, data) { - var i - var bkey = rstr2binl(key) - var ipad = [] - var opad = [] - var hash - ipad[15] = opad[15] = undefined - if (bkey.length > 16) { - bkey = binlMD5(bkey, key.length * 8) - } - for (i = 0; i < 16; i += 1) { - ipad[i] = bkey[i] ^ 0x36363636 - opad[i] = bkey[i] ^ 0x5c5c5c5c - } - hash = binlMD5(ipad.concat(rstr2binl(data)), 512 + data.length * 8) - return binl2rstr(binlMD5(opad.concat(hash), 512 + 128)) - } - - /** - * Convert a raw string to a hex string - * - * @param {string} input Raw input string - * @returns {string} Hex encoded string - */ - function rstr2hex(input) { - var hexTab = '0123456789abcdef' - var output = '' - var x - var i - for (i = 0; i < input.length; i += 1) { - x = input.charCodeAt(i) - output += hexTab.charAt((x >>> 4) & 0x0f) + hexTab.charAt(x & 0x0f) - } - return output - } - - /** - * Encode a string as UTF-8 - * - * @param {string} input Input string - * @returns {string} UTF8 string - */ - function str2rstrUTF8(input) { - return unescape(encodeURIComponent(input)) - } - - /** - * Encodes input string as raw MD5 string - * - * @param {string} s Input string - * @returns {string} Raw MD5 string - */ - function rawMD5(s) { - return rstrMD5(str2rstrUTF8(s)) - } - /** - * Encodes input string as Hex encoded string - * - * @param {string} s Input string - * @returns {string} Hex encoded string - */ - function hexMD5(s) { - return rstr2hex(rawMD5(s)) - } - /** - * Calculates the raw HMAC-MD5 for the given key and data - * - * @param {string} k HMAC key - * @param {string} d Input string - * @returns {string} Raw MD5 string - */ - function rawHMACMD5(k, d) { - return rstrHMACMD5(str2rstrUTF8(k), str2rstrUTF8(d)) - } - /** - * Calculates the Hex encoded HMAC-MD5 for the given key and data - * - * @param {string} k HMAC key - * @param {string} d Input string - * @returns {string} Raw MD5 string - */ - function hexHMACMD5(k, d) { - return rstr2hex(rawHMACMD5(k, d)) - } - - /** - * Calculates MD5 value for a given string. - * If a key is provided, calculates the HMAC-MD5 value. - * Returns a Hex encoded string unless the raw argument is given. - * - * @param {string} string Input string - * @param {string} [key] HMAC key - * @param {boolean} [raw] Raw output switch - * @returns {string} MD5 output - */ - function md5(string, key, raw) { - if (!key) { - if (!raw) { - return hexMD5(string) - } - return rawMD5(string) - } - if (!raw) { - return hexHMACMD5(key, string) - } - return rawHMACMD5(key, string) - } - - return md5; - })(); - /* eslint-enable */ - - class Encoding { - getInfo() { - return { - id: 'Encoding', - name: 'Encoding', - color1: '#6495ed', - color2: '#739fee', - color3: '#83aaf0', - menuIconURI: icon2, - blockIconURI: icon, - blocks: [ - { - opcode: 'encode', - blockType: Scratch.BlockType.REPORTER, - text: 'Encode [string] in [code]', - arguments: { - string: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'apple' - }, - code: { - type: Scratch.ArgumentType.STRING, - menu: 'encode', - defaultValue: 'Base64' - } - } - }, - { - opcode: 'decode', - blockType: Scratch.BlockType.REPORTER, - text: 'Decode [string] with [code]', - arguments: { - string: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'VHVyYm9XYXJw' - }, - code: { - type: Scratch.ArgumentType.STRING, - menu: 'encode', - defaultValue: 'Base64' - } - } - }, - { - opcode: 'hash', - blockType: Scratch.BlockType.REPORTER, - text: 'Hash [string] with [hash]', - arguments: { - string: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'apple' - }, - hash: { - type: Scratch.ArgumentType.STRING, - menu: 'hash', - defaultValue: 'MD5' - } - } - }, - - '---', - - { - opcode: 'Conversioncodes', - blockType: Scratch.BlockType.REPORTER, - text: 'Convert the character [string] to [CodeList]', - arguments: { - string: { - type: Scratch.ArgumentType.STRING, - defaultValue: 'A' - }, - CodeList: { - type: Scratch.ArgumentType.STRING, - menu: 'Code', - defaultValue: 'UNICODE' - } - } - }, - { - opcode: 'Restorecode', - blockType: Scratch.BlockType.REPORTER, - text: '[string] corresponding to the [CodeList] character', - arguments: { - string: { - type: Scratch.ArgumentType.STRING, - defaultValue: '65' - }, - CodeList: { - type: Scratch.ArgumentType.STRING, - menu: 'Code', - defaultValue: 'UNICODE' - } - } - }, - - '---', - - { - opcode: 'Randomstrings', - blockType: Scratch.BlockType.REPORTER, - text: 'Randomly generated [position] character string', - arguments: { - position: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '8' - } - } - }, - { - opcode: 'Fontgenerationstring', - blockType: Scratch.BlockType.REPORTER, - text: 'Use [wordbank] to generate a random [position] character string', - arguments: { - wordbank: { - type: Scratch.ArgumentType.STRING, - defaultValue: '1234567890' - }, - position: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: '8' - } - } - }, - ], - menus: { - Code: { - acceptReporters: true, - items: [ - { - text: 'Unicode', - value: 'UNICODE' - } - ] - }, - encode: { - acceptReporters: true, - items: [ - { - text: 'Base 64', - value: 'Base64' - }, - 'URL' - ] - }, - hash: { - acceptReporters: true, - items: ['MD5'] - } - } - }; - } - encode({ string, code }) { - string = Scratch.Cast.toString(string); - switch (code) { - case 'Base64': return btoa(string); - case 'URL': return encodeURIComponent(string); - } - return ''; - } - decode({ string, code }) { - string = Scratch.Cast.toString(string); - switch (code) { - case 'Base64': - try { - return atob(string); - } catch (error) { - console.error('invalid base 64', error); - return ''; - } - case 'URL': return decodeURIComponent(string); - } - return ''; - } - hash({ string, hash }) { - string = Scratch.Cast.toString(string); - switch (hash) { - case 'MD5': return md5(string); - } - return ''; - } - Conversioncodes({ string, CodeList }) { - string = Scratch.Cast.toString(string); - switch (CodeList) { - case 'UNICODE': return string.charCodeAt(0); - } - return 0; - } - Restorecode({ string, CodeList}) { - switch (CodeList) { - case 'UNICODE': return String.fromCharCode(string); - } - return ''; - } - Randomstrings({ position }) { - position = Scratch.Cast.toNumber(position) || 32; - let t = 'ABCDEFGHJKMNPQRSTWXYZabcdefhijkmnprstwxyz2345678'; - let a = t.length; - let string = ''; - for (let i = 0; i < position; i++) { - string += t.charAt(Math.floor(Math.random() * a)); - } - return string; - } - Fontgenerationstring({ wordbank, position }) { - position = Scratch.Cast.toNumber(position) || 32; - let t = String(wordbank); - let a = t.length; - let string = ''; - for (let i = 0; i < position; i++) { - string += t.charAt(Math.floor(Math.random() * a)); - } - return string; - } - } - Scratch.extensions.register(new Encoding()); -})(Scratch); +// Name: Encoding +// ID: Encoding +// Description: Encode and decode strings into their unicode numbers, base 64, or URLs. +// By: -SIPC- + +(function (Scratch) { + "use strict"; + const icon = + ""; + const icon2 = + ""; + + /*! + This md5 function is based on https://github.com/blueimp/JavaScript-MD5/blob/master/js/md5.js + which is licensed under: + + MIT License + + Copyright © 2011 Sebastian Tschan, https://blueimp.net + + Permission is hereby granted, free of charge, to any person obtaining a copy of + this software and associated documentation files (the "Software"), to deal in + the Software without restriction, including without limitation the rights to + use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of + the Software, and to permit persons to whom the Software is furnished to do so, + subject to the following conditions: + + The above copyright notice and this permission notice shall be included in all + copies or substantial portions of the Software. + + THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS + FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR + COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER + IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN + CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. + */ + /* eslint-disable */ + const md5 = (function () { + /** + * Add integers, wrapping at 2^32. + * This uses 16-bit operations internally to work around bugs in interpreters. + * + * @param {number} x First integer + * @param {number} y Second integer + * @returns {number} Sum + */ + function safeAdd(x, y) { + var lsw = (x & 0xffff) + (y & 0xffff); + var msw = (x >> 16) + (y >> 16) + (lsw >> 16); + return (msw << 16) | (lsw & 0xffff); + } + + /** + * Bitwise rotate a 32-bit number to the left. + * + * @param {number} num 32-bit number + * @param {number} cnt Rotation count + * @returns {number} Rotated number + */ + function bitRotateLeft(num, cnt) { + return (num << cnt) | (num >>> (32 - cnt)); + } + + /** + * Basic operation the algorithm uses. + * + * @param {number} q q + * @param {number} a a + * @param {number} b b + * @param {number} x x + * @param {number} s s + * @param {number} t t + * @returns {number} Result + */ + function md5cmn(q, a, b, x, s, t) { + return safeAdd( + bitRotateLeft(safeAdd(safeAdd(a, q), safeAdd(x, t)), s), + b + ); + } + /** + * Basic operation the algorithm uses. + * + * @param {number} a a + * @param {number} b b + * @param {number} c c + * @param {number} d d + * @param {number} x x + * @param {number} s s + * @param {number} t t + * @returns {number} Result + */ + function md5ff(a, b, c, d, x, s, t) { + return md5cmn((b & c) | (~b & d), a, b, x, s, t); + } + /** + * Basic operation the algorithm uses. + * + * @param {number} a a + * @param {number} b b + * @param {number} c c + * @param {number} d d + * @param {number} x x + * @param {number} s s + * @param {number} t t + * @returns {number} Result + */ + function md5gg(a, b, c, d, x, s, t) { + return md5cmn((b & d) | (c & ~d), a, b, x, s, t); + } + /** + * Basic operation the algorithm uses. + * + * @param {number} a a + * @param {number} b b + * @param {number} c c + * @param {number} d d + * @param {number} x x + * @param {number} s s + * @param {number} t t + * @returns {number} Result + */ + function md5hh(a, b, c, d, x, s, t) { + return md5cmn(b ^ c ^ d, a, b, x, s, t); + } + /** + * Basic operation the algorithm uses. + * + * @param {number} a a + * @param {number} b b + * @param {number} c c + * @param {number} d d + * @param {number} x x + * @param {number} s s + * @param {number} t t + * @returns {number} Result + */ + function md5ii(a, b, c, d, x, s, t) { + return md5cmn(c ^ (b | ~d), a, b, x, s, t); + } + + /** + * Calculate the MD5 of an array of little-endian words, and a bit length. + * + * @param {Array} x Array of little-endian words + * @param {number} len Bit length + * @returns {Array} MD5 Array + */ + function binlMD5(x, len) { + /* append padding */ + x[len >> 5] |= 0x80 << len % 32; + x[(((len + 64) >>> 9) << 4) + 14] = len; + + var i; + var olda; + var oldb; + var oldc; + var oldd; + var a = 1732584193; + var b = -271733879; + var c = -1732584194; + var d = 271733878; + + for (i = 0; i < x.length; i += 16) { + olda = a; + oldb = b; + oldc = c; + oldd = d; + + a = md5ff(a, b, c, d, x[i], 7, -680876936); + d = md5ff(d, a, b, c, x[i + 1], 12, -389564586); + c = md5ff(c, d, a, b, x[i + 2], 17, 606105819); + b = md5ff(b, c, d, a, x[i + 3], 22, -1044525330); + a = md5ff(a, b, c, d, x[i + 4], 7, -176418897); + d = md5ff(d, a, b, c, x[i + 5], 12, 1200080426); + c = md5ff(c, d, a, b, x[i + 6], 17, -1473231341); + b = md5ff(b, c, d, a, x[i + 7], 22, -45705983); + a = md5ff(a, b, c, d, x[i + 8], 7, 1770035416); + d = md5ff(d, a, b, c, x[i + 9], 12, -1958414417); + c = md5ff(c, d, a, b, x[i + 10], 17, -42063); + b = md5ff(b, c, d, a, x[i + 11], 22, -1990404162); + a = md5ff(a, b, c, d, x[i + 12], 7, 1804603682); + d = md5ff(d, a, b, c, x[i + 13], 12, -40341101); + c = md5ff(c, d, a, b, x[i + 14], 17, -1502002290); + b = md5ff(b, c, d, a, x[i + 15], 22, 1236535329); + + a = md5gg(a, b, c, d, x[i + 1], 5, -165796510); + d = md5gg(d, a, b, c, x[i + 6], 9, -1069501632); + c = md5gg(c, d, a, b, x[i + 11], 14, 643717713); + b = md5gg(b, c, d, a, x[i], 20, -373897302); + a = md5gg(a, b, c, d, x[i + 5], 5, -701558691); + d = md5gg(d, a, b, c, x[i + 10], 9, 38016083); + c = md5gg(c, d, a, b, x[i + 15], 14, -660478335); + b = md5gg(b, c, d, a, x[i + 4], 20, -405537848); + a = md5gg(a, b, c, d, x[i + 9], 5, 568446438); + d = md5gg(d, a, b, c, x[i + 14], 9, -1019803690); + c = md5gg(c, d, a, b, x[i + 3], 14, -187363961); + b = md5gg(b, c, d, a, x[i + 8], 20, 1163531501); + a = md5gg(a, b, c, d, x[i + 13], 5, -1444681467); + d = md5gg(d, a, b, c, x[i + 2], 9, -51403784); + c = md5gg(c, d, a, b, x[i + 7], 14, 1735328473); + b = md5gg(b, c, d, a, x[i + 12], 20, -1926607734); + + a = md5hh(a, b, c, d, x[i + 5], 4, -378558); + d = md5hh(d, a, b, c, x[i + 8], 11, -2022574463); + c = md5hh(c, d, a, b, x[i + 11], 16, 1839030562); + b = md5hh(b, c, d, a, x[i + 14], 23, -35309556); + a = md5hh(a, b, c, d, x[i + 1], 4, -1530992060); + d = md5hh(d, a, b, c, x[i + 4], 11, 1272893353); + c = md5hh(c, d, a, b, x[i + 7], 16, -155497632); + b = md5hh(b, c, d, a, x[i + 10], 23, -1094730640); + a = md5hh(a, b, c, d, x[i + 13], 4, 681279174); + d = md5hh(d, a, b, c, x[i], 11, -358537222); + c = md5hh(c, d, a, b, x[i + 3], 16, -722521979); + b = md5hh(b, c, d, a, x[i + 6], 23, 76029189); + a = md5hh(a, b, c, d, x[i + 9], 4, -640364487); + d = md5hh(d, a, b, c, x[i + 12], 11, -421815835); + c = md5hh(c, d, a, b, x[i + 15], 16, 530742520); + b = md5hh(b, c, d, a, x[i + 2], 23, -995338651); + + a = md5ii(a, b, c, d, x[i], 6, -198630844); + d = md5ii(d, a, b, c, x[i + 7], 10, 1126891415); + c = md5ii(c, d, a, b, x[i + 14], 15, -1416354905); + b = md5ii(b, c, d, a, x[i + 5], 21, -57434055); + a = md5ii(a, b, c, d, x[i + 12], 6, 1700485571); + d = md5ii(d, a, b, c, x[i + 3], 10, -1894986606); + c = md5ii(c, d, a, b, x[i + 10], 15, -1051523); + b = md5ii(b, c, d, a, x[i + 1], 21, -2054922799); + a = md5ii(a, b, c, d, x[i + 8], 6, 1873313359); + d = md5ii(d, a, b, c, x[i + 15], 10, -30611744); + c = md5ii(c, d, a, b, x[i + 6], 15, -1560198380); + b = md5ii(b, c, d, a, x[i + 13], 21, 1309151649); + a = md5ii(a, b, c, d, x[i + 4], 6, -145523070); + d = md5ii(d, a, b, c, x[i + 11], 10, -1120210379); + c = md5ii(c, d, a, b, x[i + 2], 15, 718787259); + b = md5ii(b, c, d, a, x[i + 9], 21, -343485551); + + a = safeAdd(a, olda); + b = safeAdd(b, oldb); + c = safeAdd(c, oldc); + d = safeAdd(d, oldd); + } + return [a, b, c, d]; + } + + /** + * Convert an array of little-endian words to a string + * + * @param {Array} input MD5 Array + * @returns {string} MD5 string + */ + function binl2rstr(input) { + var i; + var output = ""; + var length32 = input.length * 32; + for (i = 0; i < length32; i += 8) { + output += String.fromCharCode((input[i >> 5] >>> i % 32) & 0xff); + } + return output; + } + + /** + * Convert a raw string to an array of little-endian words + * Characters >255 have their high-byte silently ignored. + * + * @param {string} input Raw input string + * @returns {Array} Array of little-endian words + */ + function rstr2binl(input) { + var i; + var output = []; + output[(input.length >> 2) - 1] = undefined; + for (i = 0; i < output.length; i += 1) { + output[i] = 0; + } + var length8 = input.length * 8; + for (i = 0; i < length8; i += 8) { + output[i >> 5] |= (input.charCodeAt(i / 8) & 0xff) << i % 32; + } + return output; + } + + /** + * Calculate the MD5 of a raw string + * + * @param {string} s Input string + * @returns {string} Raw MD5 string + */ + function rstrMD5(s) { + return binl2rstr(binlMD5(rstr2binl(s), s.length * 8)); + } + + /** + * Calculates the HMAC-MD5 of a key and some data (raw strings) + * + * @param {string} key HMAC key + * @param {string} data Raw input string + * @returns {string} Raw MD5 string + */ + function rstrHMACMD5(key, data) { + var i; + var bkey = rstr2binl(key); + var ipad = []; + var opad = []; + var hash; + ipad[15] = opad[15] = undefined; + if (bkey.length > 16) { + bkey = binlMD5(bkey, key.length * 8); + } + for (i = 0; i < 16; i += 1) { + ipad[i] = bkey[i] ^ 0x36363636; + opad[i] = bkey[i] ^ 0x5c5c5c5c; + } + hash = binlMD5(ipad.concat(rstr2binl(data)), 512 + data.length * 8); + return binl2rstr(binlMD5(opad.concat(hash), 512 + 128)); + } + + /** + * Convert a raw string to a hex string + * + * @param {string} input Raw input string + * @returns {string} Hex encoded string + */ + function rstr2hex(input) { + var hexTab = "0123456789abcdef"; + var output = ""; + var x; + var i; + for (i = 0; i < input.length; i += 1) { + x = input.charCodeAt(i); + output += hexTab.charAt((x >>> 4) & 0x0f) + hexTab.charAt(x & 0x0f); + } + return output; + } + + /** + * Encode a string as UTF-8 + * + * @param {string} input Input string + * @returns {string} UTF8 string + */ + function str2rstrUTF8(input) { + return unescape(encodeURIComponent(input)); + } + + /** + * Encodes input string as raw MD5 string + * + * @param {string} s Input string + * @returns {string} Raw MD5 string + */ + function rawMD5(s) { + return rstrMD5(str2rstrUTF8(s)); + } + /** + * Encodes input string as Hex encoded string + * + * @param {string} s Input string + * @returns {string} Hex encoded string + */ + function hexMD5(s) { + return rstr2hex(rawMD5(s)); + } + /** + * Calculates the raw HMAC-MD5 for the given key and data + * + * @param {string} k HMAC key + * @param {string} d Input string + * @returns {string} Raw MD5 string + */ + function rawHMACMD5(k, d) { + return rstrHMACMD5(str2rstrUTF8(k), str2rstrUTF8(d)); + } + /** + * Calculates the Hex encoded HMAC-MD5 for the given key and data + * + * @param {string} k HMAC key + * @param {string} d Input string + * @returns {string} Raw MD5 string + */ + function hexHMACMD5(k, d) { + return rstr2hex(rawHMACMD5(k, d)); + } + + /** + * Calculates MD5 value for a given string. + * If a key is provided, calculates the HMAC-MD5 value. + * Returns a Hex encoded string unless the raw argument is given. + * + * @param {string} string Input string + * @param {string} [key] HMAC key + * @param {boolean} [raw] Raw output switch + * @returns {string} MD5 output + */ + function md5(string, key, raw) { + if (!key) { + if (!raw) { + return hexMD5(string); + } + return rawMD5(string); + } + if (!raw) { + return hexHMACMD5(key, string); + } + return rawHMACMD5(key, string); + } + + return md5; + })(); + /* eslint-enable */ + + class Encoding { + getInfo() { + return { + id: "Encoding", + name: "Encoding", + color1: "#6495ed", + color2: "#739fee", + color3: "#83aaf0", + menuIconURI: icon2, + blockIconURI: icon, + blocks: [ + { + opcode: "encode", + blockType: Scratch.BlockType.REPORTER, + text: "Encode [string] in [code]", + arguments: { + string: { + type: Scratch.ArgumentType.STRING, + defaultValue: "apple", + }, + code: { + type: Scratch.ArgumentType.STRING, + menu: "encode", + defaultValue: "Base64", + }, + }, + }, + { + opcode: "decode", + blockType: Scratch.BlockType.REPORTER, + text: "Decode [string] with [code]", + arguments: { + string: { + type: Scratch.ArgumentType.STRING, + defaultValue: "VHVyYm9XYXJw", + }, + code: { + type: Scratch.ArgumentType.STRING, + menu: "encode", + defaultValue: "Base64", + }, + }, + }, + { + opcode: "hash", + blockType: Scratch.BlockType.REPORTER, + text: "Hash [string] with [hash]", + arguments: { + string: { + type: Scratch.ArgumentType.STRING, + defaultValue: "apple", + }, + hash: { + type: Scratch.ArgumentType.STRING, + menu: "hash", + defaultValue: "MD5", + }, + }, + }, + + "---", + + { + opcode: "Conversioncodes", + blockType: Scratch.BlockType.REPORTER, + text: "Convert the character [string] to [CodeList]", + arguments: { + string: { + type: Scratch.ArgumentType.STRING, + defaultValue: "A", + }, + CodeList: { + type: Scratch.ArgumentType.STRING, + menu: "Code", + defaultValue: "UNICODE", + }, + }, + }, + { + opcode: "Restorecode", + blockType: Scratch.BlockType.REPORTER, + text: "[string] corresponding to the [CodeList] character", + arguments: { + string: { + type: Scratch.ArgumentType.STRING, + defaultValue: "65", + }, + CodeList: { + type: Scratch.ArgumentType.STRING, + menu: "Code", + defaultValue: "UNICODE", + }, + }, + }, + + "---", + + { + opcode: "Randomstrings", + blockType: Scratch.BlockType.REPORTER, + text: "Randomly generated [position] character string", + arguments: { + position: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "8", + }, + }, + }, + { + opcode: "Fontgenerationstring", + blockType: Scratch.BlockType.REPORTER, + text: "Use [wordbank] to generate a random [position] character string", + arguments: { + wordbank: { + type: Scratch.ArgumentType.STRING, + defaultValue: "1234567890", + }, + position: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "8", + }, + }, + }, + ], + menus: { + Code: { + acceptReporters: true, + items: [ + { + text: "Unicode", + value: "UNICODE", + }, + ], + }, + encode: { + acceptReporters: true, + items: [ + { + text: "Base 64", + value: "Base64", + }, + "URL", + ], + }, + hash: { + acceptReporters: true, + items: ["MD5"], + }, + }, + }; + } + encode({ string, code }) { + string = Scratch.Cast.toString(string); + switch (code) { + case "Base64": + return btoa(string); + case "URL": + return encodeURIComponent(string); + } + return ""; + } + decode({ string, code }) { + string = Scratch.Cast.toString(string); + switch (code) { + case "Base64": + try { + return atob(string); + } catch (error) { + console.error("invalid base 64", error); + return ""; + } + case "URL": + return decodeURIComponent(string); + } + return ""; + } + hash({ string, hash }) { + string = Scratch.Cast.toString(string); + switch (hash) { + case "MD5": + return md5(string); + } + return ""; + } + Conversioncodes({ string, CodeList }) { + string = Scratch.Cast.toString(string); + switch (CodeList) { + case "UNICODE": + return string.charCodeAt(0); + } + return 0; + } + Restorecode({ string, CodeList }) { + switch (CodeList) { + case "UNICODE": + return String.fromCharCode(string); + } + return ""; + } + Randomstrings({ position }) { + position = Scratch.Cast.toNumber(position) || 32; + let t = "ABCDEFGHJKMNPQRSTWXYZabcdefhijkmnprstwxyz2345678"; + let a = t.length; + let string = ""; + for (let i = 0; i < position; i++) { + string += t.charAt(Math.floor(Math.random() * a)); + } + return string; + } + Fontgenerationstring({ wordbank, position }) { + position = Scratch.Cast.toNumber(position) || 32; + let t = String(wordbank); + let a = t.length; + let string = ""; + for (let i = 0; i < position; i++) { + string += t.charAt(Math.floor(Math.random() * a)); + } + return string; + } + } + Scratch.extensions.register(new Encoding()); +})(Scratch); diff --git a/extensions/extensions.json b/extensions/extensions.json new file mode 100644 index 0000000000..6418f9577d --- /dev/null +++ b/extensions/extensions.json @@ -0,0 +1,75 @@ +[ + "lab/text", + "stretch", + "gamepad", + "box2d", + "files", + "pointerlock", + "cursor", + "runtime-options", + "fetch", + "text", + "local-storage", + "true-fantom/base", + "bitwise", + "Skyhigh173/bigint", + "utilities", + "sound", + "Xeltalliv/clippingblending", + "clipboard", + "penplus", + "Lily/Skins", + "obviousAlexC/SensingPlus", + "Lily/ClonesPlus", + "Lily/LooksPlus", + "Lily/MoreEvents", + "NexusKitten/moremotion", + "navigator", + "battery", + "TheShovel/CustomStyles", + "NexusKitten/controlcontrols", + "mdwalters/notifications", + "XeroName/Deltatime", + "ar", + "encoding", + "Lily/TempVariables2", + "Lily/MoreTimers", + "clouddata-ping", + "cloudlink", + "true-fantom/network", + "true-fantom/math", + "true-fantom/regexp", + "true-fantom/couplers", + "Lily/AllMenus", + "Lily/Cast", + "-SIPC-/time", + "-SIPC-/consoles", + "ZXMushroom63/searchApi", + "TheShovel/ShovelUtils", + "DNin/wake-lock", + "Skyhigh173/json", + "cs2627883/numericalencoding", + "DT/cameracontrols", + "TheShovel/CanvasEffects", + "Longboost/color_channels", + "CST1229/zip", + "TheShovel/LZ-String", + "0832/rxFS2", + "NexusKitten/sgrab", + "NOname-awa/graphics2d", + "NOname-awa/more-comparisons", + "JeremyGamer13/tween", + "rixxyx", + "qxsck/data-analysis", + "qxsck/var-and-list", + "vercte/dictionaries", + "godslayerakp/http", + "Lily/CommentBlocks", + "veggiecan/LongmanDictionary", + "CubesterYT/TurboHook", + "Alestore/nfcwarp", + "itchio", + "gamejolt", + "obviousAlexC/newgroundsIO", + "Lily/McUtils" +] diff --git a/extensions/fetch.js b/extensions/fetch.js index 0340ad3aec..10ff8e13e4 100644 --- a/extensions/fetch.js +++ b/extensions/fetch.js @@ -1,31 +1,35 @@ -(function(Scratch) { - 'use strict'; +// Name: Fetch +// ID: fetch +// Description: Make requests to the broader internet. + +(function (Scratch) { + "use strict"; class Fetch { - getInfo () { + getInfo() { return { - id: 'fetch', - name: 'Fetch', + id: "fetch", + name: "Fetch", blocks: [ { - opcode: 'get', + opcode: "get", blockType: Scratch.BlockType.REPORTER, - text: 'GET [URL]', + text: "GET [URL]", arguments: { URL: { type: Scratch.ArgumentType.STRING, - defaultValue: 'https://extensions.turbowarp.org/hello.txt' - } - } - } - ] + defaultValue: "https://extensions.turbowarp.org/hello.txt", + }, + }, + }, + ], }; } - get (args) { + get(args) { return Scratch.fetch(args.URL) - .then(r => r.text()) - .catch(() => ''); + .then((r) => r.text()) + .catch(() => ""); } } diff --git a/extensions/files.js b/extensions/files.js index 2d1cea7860..c4212b34c8 100644 --- a/extensions/files.js +++ b/extensions/files.js @@ -1,175 +1,187 @@ -(function(Scratch) { - 'use strict'; +// Name: Files +// ID: files +// Description: Read and download files. + +(function (Scratch) { + "use strict"; if (!Scratch.extensions.unsandboxed) { - throw new Error('files extension must be run unsandboxed'); + throw new Error("files extension must be run unsandboxed"); } - const MODE_MODAL = 'modal'; - const MODE_IMMEDIATELY_SHOW_SELECTOR = 'selector'; - const MODE_ONLY_SELECTOR = 'only-selector'; - const ALL_MODES = [MODE_MODAL, MODE_IMMEDIATELY_SHOW_SELECTOR, MODE_ONLY_SELECTOR]; + const MODE_MODAL = "modal"; + const MODE_IMMEDIATELY_SHOW_SELECTOR = "selector"; + const MODE_ONLY_SELECTOR = "only-selector"; + const ALL_MODES = [ + MODE_MODAL, + MODE_IMMEDIATELY_SHOW_SELECTOR, + MODE_ONLY_SELECTOR, + ]; let openFileSelectorMode = MODE_MODAL; - const AS_TEXT = 'text'; - const AS_DATA_URL = 'url'; + const AS_TEXT = "text"; + const AS_DATA_URL = "url"; /** * @param {string} accept See MODE_ constants above * @param {string} as See AS_ constants above * @returns {Promise} format given by as parameter */ - const showFilePrompt = (accept, as) => new Promise((_resolve) => { - // We can't reliably show an - - - Zip docs - - - - -
-

Zip extension documentation

-
-
-

- The Zip extension allows you to read, create and edit .zip format files, - including Scratch project and sprite files - (.sb3, .sprite3). -

- -

- Paths - (link) -

-

Most blocks in this extension work with a path format:

-
    -
  • - The path is relative to the current directory by default -
  • -
  • - Directories (folders) and the filename are separated by slashes, like - folder1/folder2/file.txt -
  • -
  • - .. goes to the parent directory, like - ../file.txt -
  • -
  • - . goes to the current directory, like - ./file.txt -
  • -
  • - A / at the very start goes to the root directory, like - /file.txt -
  • -
  • - A / at the end denotes a directory, like - folder/ -
  • -
  • - - Multiple slashes in a row or trying to go above the root directory will - result in an error (usually the block doing nothing or returning the - empty value) -
  • -
- -

- - Archive management blocks - (link) -

-

- Blocks for creating and saving the current archive. - Only one archive can be open at a time. -

- -

- create empty archive - (link) -

-

- Creates and opens an empty archive with nothing in it. -

- -

- open zip from [URL] [https://extensions.turbowarp.org/hello.zip] - (link) -

-

- Opens a .zip (or .sb3 or .sprite3...) file. -

-

- The type can be one of the following: -

-
    -
  • - URL: A URL, which can be either a web URL or data: URL. Recommended. -
  • -
  • - base64: A base64 string without the data URL header. -
  • -
  • - hex: A sequence of hexadecimal bytes - (like 101A1B1C), - without a separator. -
  • -
  • - binary: A sequence of binary bytes - (like 000000010010101001101011), - without a separator. -
  • -
  • - string: Plain text. - Not recommended! - Text encoding behavior will likely break it, as it's a binary file. -
  • -
-

- If the file is not of zip format (e.g RAR or 7z) or is password-protected, - it won't be opened. - Make sure to check if it loaded successfully with the - archive is open? block. -

- -

- output zip type [data: URL] compression level [6] - (link) -

-

- Save the zip data into a string, which can be saved with e.g the Files extension. -

-

- The type can be one of the following: -

-
    -
  • - data: URL: A base64-encoded data URL. Recommended. -
  • -
  • - base64: A base64 string without the data URL header. -
  • -
  • - hex: A sequence of hexadecimal bytes - (like 101A1B1C), - without a separator. -
  • -
  • - binary: A sequence of binary bytes - (like 000000010010101001101011), - without a separator. -
  • -
  • - string: Plain text. - Not recommended! - Text encoding behavior will likely break it, as it's a binary file. -
  • -
-

- The compression level decides how much the zip is compressed. -
- Lower compression levels will result in a bigger file, - but are faster to create. Usually, high compression levels - provide diminishing returns (lesser gains the higher you go) - with much slower speeds. -
- A compression level of 0 (no compression) is the fastest, - but will often result in a very big file. -

- -

- close archive - (link) -

-

Closes the archive. Use after you're done working with it.

- -

- archive is open? - (link) -

-

Returns true if an archive is open.

- - -

- File blocks - (link) -

-

- Blocks for working with files - (and blocks that are general to both files and folders/directories.) -

- -

- [folder/] exists? - (link) -

-

- Returns if a file or directory exists or not. - The slash at the end matters! - If a directory named folder exists, - [folder] exists? will return false, - but [folder/] exists? will return true. -

- -

- write file [new file.txt] content [Hello, world?] type [text] - (link) -

-

- Writes content to a file, creating the file if it doesn't exist - and replacing its existing data if it does. -

-

- The type can be one of the following: -

-
    -
  • - text: Plain text. -
  • -
  • - URL: A URL, which can be either a web URL or data: URL. - Best for binary data (like other zip files). -
  • -
  • - base64: A base64 string without the data URL header. -
  • -
  • - hex: A sequence of hexadecimal bytes - (like 101A1B1C), - without a separator. -
  • -
  • - binary: A sequence of binary bytes - (like 000000010010101001101011), - without a separator. -
  • -
- -

- rename [hello.txt] to [hello renamed.txt] - (link) -

-

- Renames a file or directory to another name. - If the target file already exists, it will be overwritten. - The current directory will also be updated. - This block can also be used to move files to a different directory. -

- -

- delete [hello.txt] - (link) -

-

- Deletes a file or directory (including everything in it). -
- If the current directory is in that directory, it will - be set to the closest existing parent directory. -

- -

- file [hello.txt] as [text] - (link) -

-

- Get the contents of a file. -

-

- The type can be one of the following: -

-
    -
  • - text: Plain text. -
  • -
  • - data: URL: A base64-encoded data URL. - Best for binary data (like other zip files). -
  • -
  • - base64: A base64 string without the data URL header. -
  • -
  • - hex: A sequence of hexadecimal bytes - (like 101A1B1C), - without a separator. -
  • -
  • - binary: A sequence of binary bytes - (like 000000010010101001101011), - without a separator. -
  • -
- -

- File info blocks - (link) -

-

Blocks for getting and setting additional information on a file.

- -

- set [unix modified timestamp] of [folder/dango.png] to [0] - (link) -

-

- Set additional info on a file or directory. -

-

- Available options: -

-
    -
  • - modified days since 2000: - The modification date of the file, as days since 2000. -
  • -
  • - unix modified timestamp: - The modification date of the file, as a Unix timestamp (milliseconds since 1970). - Useful when combined with e.g the Time extension. -
  • -
  • - comment: A comment on the file. Can be any text. - Some programs may show this as metadata. -
  • -
- -

- [name] of [folder/dango.png] - (link) -

-

- Get additional info on a file or directory. -

-

- Available options: -

-
    -
  • - name: - Just the name of this file (without the directories it's in). - For example, the name of /folder1/folder2/dango.png - would be dango.png. -
  • -
  • - path: The full absolute path of this file - (its name and any directories it's in). -
  • -
  • - folder: - Just the folders this file is in (without its filename). - For example, the folder of /folder1/folder2/dango.png - would be /folder1/folder2/. -
  • -
  • - modification date: - A human-readable version of the file's modification date. - The output of this depends on the browser's language and possibly other factors. -
  • -
  • - long modification date: - A longer human-readable version of the file's modification date. - The output of this depends on the browser's language and possibly other factors. -
  • -
  • - modified days since 2000: - The modification date of the file, as days since 2000. -
  • -
  • - unix modified timestamp: - The modification date of the file, as a Unix timestamp (milliseconds since 1970). - Useful when combined with e.g the Time extension. -
  • -
  • - comment: A comment on the file. Can be any text. - Some programs may show this as metadata. -
  • -
- -

- Directory blocks - (link) -

-

Blocks that deal with directories and the current directory.

- - -

- create directory [new folder] - (link) -

-

- Creates a directory with a name. This can create multiple directories - at once (by having multiple directores in the path, like - /new folder1/new folder2/new folder3/). -

- -

- go to [folder] - (link) -

-

- Moves the current directory (the default origin of most file operations) - to the speficied directory. If it doesn't exist, this block will do nothing. -

- -

- contents of directory [.] - (link) -

-

- Returns a list of files in a directory, as JSON - (which you can parse with the JSON extension). -

- -

- current directory path - (link) -

-

- Returns the absolute path to the current directory. -

- -

- Other blocks - (link) -

-

- Miscellaneous stuff. -

- -

- set archive comment to [any text] - (link) -

-

- Sets the archive's comment to some text. Just like file comments, - this is saved and may be displayed as metadata by some programs. -

- -

- archive comment - (link) -

-

- Returns the archive's comment. -

- -

- path [../folder3/] from [/folder/folder2] - (link) -

-

- Returns an absolute path from an origin path and a target path. - Does not depend on the archive, so it works without one open. - This is mostly a utility used internally, but it might - be useful for users too. -

-
- - - \ No newline at end of file diff --git a/website/CubesterYT/TurboHook.html b/website/CubesterYT/TurboHook.html deleted file mode 100644 index 1222e12df7..0000000000 --- a/website/CubesterYT/TurboHook.html +++ /dev/null @@ -1,105 +0,0 @@ - - - - - - TurboHook docs - - - -
- TurboHook extension documentation -
-
-

About this extension

-

This extension allows you to post to webhooks from some common third-party websites or programs.

- -
- -

Blocks

-

Webhook Block

-

Posts a message to a webhook.

- -

The empty area is where you insert the data reporter and/or connector reporter. The string area is where you put your webhook's URL.

- -
- -

Data Reporter

-

The Menu area of the Data Reporter currently has three options, ("content","name","icon").

-

Content

- -

When choosing "content", you can type in the text you want to send as the body of the webhook.

-

Name

- -

When choosing "name", you can type in the name of the name you want the webhook to display as on the third-party website.

-

Icon

- -

When choosing "icon", you can type in a URL to the image you want to set as the profile picture of the webhook.

- -
- -

Connector Reporter

- -

This reporter connects two different data reporters together and/or another connector reporter so you can send multiple pieces of data to the webhook.

-
- - diff --git a/website/CubesterYT/connector.png b/website/CubesterYT/connector.png deleted file mode 100644 index b97a6f3e53..0000000000 Binary files a/website/CubesterYT/connector.png and /dev/null differ diff --git a/website/CubesterYT/content.png b/website/CubesterYT/content.png deleted file mode 100644 index 0afb7e96f5..0000000000 Binary files a/website/CubesterYT/content.png and /dev/null differ diff --git a/website/CubesterYT/icon.png b/website/CubesterYT/icon.png deleted file mode 100644 index 1fba21595f..0000000000 Binary files a/website/CubesterYT/icon.png and /dev/null differ diff --git a/website/CubesterYT/name.png b/website/CubesterYT/name.png deleted file mode 100644 index b729b1b79e..0000000000 Binary files a/website/CubesterYT/name.png and /dev/null differ diff --git a/website/CubesterYT/webhook.png b/website/CubesterYT/webhook.png deleted file mode 100644 index 121c8415c2..0000000000 Binary files a/website/CubesterYT/webhook.png and /dev/null differ diff --git a/website/ar.html b/website/ar.html deleted file mode 100644 index 5c40f7c9ce..0000000000 --- a/website/ar.html +++ /dev/null @@ -1,267 +0,0 @@ - - - - - - AR docs - - - - - - - -
- Augmented Reality extension documentation -
-
-

Requirements

- -

At the moment of writing, only Chromium-based browsers support immersive-ar session type.

- -

Other information

-
    -
  • It is generally recommended to use this extension with fps set to 0, which in turbowarp means running project at the screen refresh rate.
  • -
  • While exiting AR mode, there is a small chance TurboWarp will lose WebGL context. This problem has also been observed on other websites with AR and likely can't be fixed. If that happens, you can save the project, refresh the tab, load the project and continue.
  • -
  • It isn't possible to be in AR mode and have video sensing enabled at the same time. Entering AR disables video sensing and makes it not toggleable until AR session ends.
  • -
- -

Blocks

-
enterARmode
-

If AR is not supported and the error message haven't been shown yet, shows a popup with an error message.

-

If AR is supported and project is currently not in AR, attempts to enter AR mode. While doing so, it will pause the script it is in, in the same way ask block does. It will first try to enter AR mode directly. That may fail because entering AR can only be triggred by user interaction. If it fails, it will make it so after user clicks/taps the project will attempt to enter AR. If that fails as well or either of those 2 attempts succeed it will resume the script execution. After it resumes, project may or may not be in AR mode. Projects should handle both cases.

-

The origin of coordinate system is placed at or close to position of the device at the time this block was called.

- -
-
exitARmode
-

If the project is in AR mode, exits it.

- -
-
isinAR?
-

Tells if the project is currently in AR mode.

- -
-
isaravailible?
-

Tells if AR is supported on this device.

- -
-
isposeavailible?
-

Tells if AR engine knows what the current camera position and orientation is.

-

After entering AR mode, is is not immediately availible as the map of the environment needs to be built first. - After enough information about environment has been gathered and processed, it becomes availible. - It can temporarily become unavailible due to lack of detailed features in the view of camera that are used for motion tracking, - fast motion causing camera image to become too blurry or camera getting covered.

- -
-
ishit positionavailible?
-

Tells if AR engine knows where the point of ray intersection is.

-

Can become unavailible for the same reasons as [is [pose] availible?]

- -
-
stagewidth
-

Tells stage width in scratch units. Default is 480.
- This value may change when entering AR.

- -
-
stageheight
-

Tells stage height in scratch units. Default is 360.
- This value will not change when entering AR.

- -
-
item1ofviewmatrix
-

view matrix - is a matrix that can be used to transform points from the world space - (relatively to the world origin) to the view space (with origin is at the camera). It - includes rotation and translation of the camera.

- Also: - - -
-
item1ofinverse viewmatrix
-

inverse view matrix = view matrix-1
- Describes the opposite transformation to the view matrix.

- -
-
item1ofprojectionmatrix
-

For perspective projection in scratch you are likely used to doing something like this:

- - -

With this extension it's more complicated: projection is done by - first doing a 4x4 projection matrix multiplication by 4D vector x,y,z,1, - with result being X,Y,Z,W, then performing division of X,Y,Z by W to - get screen coordinates in range from -1 to 1, and then multiplying them - by half of stage width and height to get scratch coordinates.

- -

The matrix is a perspective projection matrix with the assumption camera faces negative z and that screen coordinates range from -1 to 1.

- -

It's calculated as:

- - -

Point is only visible if -W < X,Y,Z < W.

- - - - If the first codition was true, then -1 < screenX,screenY,screenZ < 1 is also always true. Note that the opposite is not always true (think of what happens when W is negative). - -

To get a better understanding of this topic, you may read:

- - -

Projection matrix contains a lot of 0s, and can be simplified. As the result of simplifications it is possible to make it look closer to how it is usually done on scratch:

- - -
-
item1ofcombinedmatrix
-

combined matrix = projection matrix * inverse view matrix

- -
-
positionx
-
positiony
-
positionz
-

Camera position relatively to the world origin.

- -
-
orientationr
-
orientationi
-
orientationj
-
orientationk
-

Camera orientation represented as quaternion.

- -
-
hitpositionx
-
hitpositiony
-
hitpositionz
-

A ray that originates from camera in the direction the camera is facing (center of the screen) intersects the first detected real world surface. - This block returns the coordinates of that intersection point.

- -
-
moveeverythingbyx:0y:0z:0
-

Moves the coordinate system by specified amount.
- It can also be understood as switching to a new coordinate system with origin at a given location in the current coordinate system.

- - Usage example: -

After starting the project, it may be a good idea to give user a way to pick location for AR content and then perform this before starting the main AR game/animation/etc.

-
moveeverythingbyx:hitpositionxy:hitpositionyz:hitpositionz
- -
-
turneverythingbyr:0i:0j:0k:0
-

Turns coodinate system around the origin by specified quaternion.
- Internally it also does quaternion normalization, so you don't have to worry about doing it yourself.

- - Usage example: -

This script can be used to rotate XZ around Y-axis:

-
turneverythingbyr:cosofangle/2i:0j:sinofangle/2k:0
- -
-
setresolution1
-

accepts values from 0.1 to 1

- -

changes the resolution at which the project is rendered
- 1 - is native screen resolution
- 0.5 - half the screen resolution
- 0.1 - one tenth of screen resolution
- Reduing resolution can improve performance and reduce memory usage.

- -

Examples

- Example 1: -
whenclickedenterARmodewhileisinAR?clearpointat0.10.5-0.3definepointatxyzsetXtox*item1ofcombinedmatrix+y*item5ofcombinedmatrix+z*item9ofcombinedmatrix+item13ofcombinedmatrixsetYtox*item2ofcombinedmatrix+y*item6ofcombinedmatrix+z*item10ofcombinedmatrix+item14ofcombinedmatrixsetZtox*item3ofcombinedmatrix+y*item7ofcombinedmatrix+z*item11ofcombinedmatrix+item15ofcombinedmatrixsetWtox*item4ofcombinedmatrix+y*item8ofcombinedmatrix+z*item12ofcombinedmatrix+item16ofcombinedmatrixifabsofX<WthenifabsofY<WthenifabsofZ<Wthengotox:X/W*stagewidth/2y:Y/W*stageheight/2pendownpenup
- Example 2: -
whenclickedenterARmodewhileisinAR?clearViewmatrixandcameracoordspointat0.10.5-0.3defineViewmatrixandcameracoordssetMXXtoitem1ofviewmatrixsetMXYtoitem5ofviewmatrixsetMXZto0-item9ofviewmatrixsetMYXtoitem2ofviewmatrixsetMYYtoitem6ofviewmatrixsetMYZto0-item10ofviewmatrixsetMZXtoitem3ofviewmatrixsetMZYtoitem7ofviewmatrixsetMZZto0-item11ofviewmatrixsetcamXtoitem13ofviewmatrixsetcamYtoitem14ofviewmatrixsetcamZtoitem15ofviewmatrixsetdisttoitem6ofprojectionmatrix*stageheight/2definepointatxyzsetx2tox-camXsety2toy-camYsetz2toz-camZsetrotXtox2*MXX+y2*MYX+z2*MZXsetrotYtox2*MXY+y2*MYY+z2*MZYsetrotZtox2*MXZ+y2*MYZ+z2*MZZifrotZ>0thengotox:rotX*dist/rotZy:rotY*dist/rotZpendownpenup
-

Credits

-

The block svg images have been generated with scratchblocks.

-
- - \ No newline at end of file diff --git a/website/bitwise.html b/website/bitwise.html deleted file mode 100644 index 04a35b6d65..0000000000 --- a/website/bitwise.html +++ /dev/null @@ -1,194 +0,0 @@ - - - - - - Bitwise docs - - - - - - - -
- Bitwise extension documentation -
-
-

About this extension

-
    -
  • This extension allows you to perform bit shifts and logic operations on integers as if they were encoded in binary. These operations can be used to create programmer calculators and more.
  • -
  • The extension uses signed 32-bit integers, which can range from -2,147,483,648 to 2,147,483,647.
  • -
- -

Blocks

- -
is0000000000100000binary?
-

Returns true if the input is a binary number. Binary numbers can only contain 0s and 1s, so this essentially tells you if the input only contains 0s and 1s.

- -
- -
32tobinary
-

Converts the input from decimal to binary and returns the result.

- -

Note: All other blocks except for [is () binary?] and [() to number] will treat this result as a decimal number, so make sure to convert it back to decimal before you perform operations on it.

- -
- -
0000000000100000tonumber
-

Converts the input from binary to decimal and returns the result.

- -
- -
x>>y
-

Arithmatically shifts each bit of the binary representation of x to the right y times and returns the result.

- -

Example: 3210000001000016

- -

The sign will be preserved, so negative numbers will stay negative.

- -

This is essentially the same thing as dividing by 2^y and rounding down to the nearest integer.

- -
- -
x<<y
-

Arithmatically shifts each bit of the binary representation of x to the left y times and returns the result.

- -

Example: 320100000100000064

- -

The sign will be preserved, so negative numbers will stay negative.

- -

This is essentially the same thing as multiplying by 2^y.

- -
- -
x>>>y
-

Logically shifts each bit of the binary representation of x to the right y times and returns the result.

- -

This also moves the sign bit, so it can result in negative numbers becoming positive.

- -
- -
xy
-

Shifts each bit of the binary representation of x to the right y times. Numbers shifted past one end will reappear on the other (circle around).

- -

Example: 30000000000000000000000000000001110000000000000000000000000000001-2147483647

- -
- -
xy
-

Shifts each bit of the binary representation of x to the left y times. Numbers shifted past one end will reappear on the other (circle around).

- -

Example: -214748364710000000000000000000000000000001000000000000000000000000000000113

- -
- -
xandy
-

Logically ANDs the binary representation of the inputs together and returns the result.

- -

Example:

- - -
- -
xory
-

Logically ORs the binary representation of the inputs together and returns the result.

- -

Example:

- - -
- -
xxory
-

Logically XORs the binary representation of the inputs together and returns the result.

- -

Example:

- - -
- -
notx
-

Flips all bits of the binary representation of x and returns the result.

- -

Example:

- - -

Credits

-

The block SVGs were generated with scratchblocks.

-
- - diff --git a/website/box2d.html b/website/box2d.html deleted file mode 100644 index c470d09bbc..0000000000 --- a/website/box2d.html +++ /dev/null @@ -1,206 +0,0 @@ - - - - - - Box2D Physics docs - - - - - - - -
- Box2D Physics extension documentation -
-
-

About this extension

-
    -
  • This extension allows you to easily implement proper physics using a physics library called Box2D.
  • -
- -

Blocks

- -

World

-

Adjust the physics of all sprites.

- -
setupstageboxed stage
-

Choose a type of containment to keep sprites within the stage.

-
    -
  • Boxed stage: Keeps sprites from going off the bottom and sides.
  • -
  • Open (with floor): Keeps sprites from going off the bottom.
  • -
  • Open (no floor): Removes all boundaries; sprites can go wherever they want.
  • -
- -
- -
setgravitytox:0y:-10
-

Change the direction and strength of gravity.

- -
- -
stepsimulation
-

Move forward in time by one step. Run this in a loop to keep the physics going.

- -

Sprites

-

Manipulate individual sprites.

- -
enableforthis costumemodenormal
-

Make physics apply to this sprite. It can also collide with other sprites that have physics enabled.

- -
    -
  • Enable for this costume: Enable physics only for the current sprite or clone.
  • -
  • Enable for this circle: Enable physics for the current sprite or clone as if it were shaped like a circle.
  • -
  • Enable for all sprites: Enable physics for all sprites.
  • -
-

Precision mode will make the sprite work extra hard to make sure it doesn't overlap with anything. Note that this can decrease performance and even cause the project to get stuck, so use with care.

- -
- -
gotox:0y:0in world
-

Make the sprite go to the specified location.

- -
    -
  • In world: Relative to the center of the world.
  • -
  • On stage: Relative to the center of the screen (if you've scrolled it).
  • -
  • Relative: Relative to itself.
  • -
- -
- -
setvelocitytosx:0sy:0
-

Set the velocity (speed) of the sprite to the specified value.

- -

You can get the velocity of the current sprite with the (x velocity) and (y velocity) reporters.

- -
- -
pushwithforce25indirection0
-

Directly send the sprite flying in a certain direction, adding on to its current velocity.

- -
- -
setangularvelocityto30
-

Set the angular velocity (rotational speed) of the sprite to the specified value.

- -

You can get the angular velocity of the current sprite with the (angular velocity) reporter.

- -
- -
spinwithforce500
-

Directly send the sprite spinning, adding on to its current angular (rotational) velocity.

- -
- -
setfixedfixed in place
-

Choose whether the sprite is fixed in place or can move around.

- -

You can tell if the sprite is currently fixed in place with the (fixed?) reporter.

- -
- -
setdensitynormal
-

Set the sprite's density, which affects how heavy it is.

- -

You can get the sprite's current density with the (density) reporter.

- -
- -
setfrictionnormal
-

Set the sprite's roughness. Smoother settings make the sprite slipperier.

- -

You can get the sprite's current friction with the (friction) reporter.

- -
- -
setbouncenormal
-

Set the sprite's bounciness.

- -

You can get the sprite's current bounciness with the (bounce) reporter.

- -
- -
touchingany
-

Returns what other sprites the sprite is touching. Also includes the edges of the stage.

- -

If there are multiple sprites touching, it will return a comma-separated list of them.

- -

Screen

-

Move the camera around the world.

- -
setscrollx:0y:0
-

Scroll to the desired location.

- -

You can get the current screen position with the (x scroll) and (y scroll) reporters.

- -

This will not affect the world boundaries set by the [setup stage] block.

- -

Example

- -
whenclickedsetupstageboxed stagesetgravitytox:0y:-10createcloneofmyselfrepeat20stepsimulationcreatecloneofmyselfforeverstepsimulationwhenIstartasacloneshowenableforthis costumemodenormalgotox:-150y:240in worldsetbouncequite bouncysetangularvelocityto-4.1
- -

Credits

-

The block SVGs were generated with scratchblocks.

-
- - diff --git a/website/index.ejs b/website/index.ejs deleted file mode 100644 index c785b75872..0000000000 --- a/website/index.ejs +++ /dev/null @@ -1,768 +0,0 @@ - - - - - - TurboWarp Extension Gallery - - - - - - -
-

- -
TurboWarp Extension Gallery
-

- -

Unlike custom extensions on other websites, these aren't limited by the extension sandbox, so they are a lot more powerful. All extensions are reviewed for safety.

-

- <% if (mode === 'desktop') { %> - To use these extensions in TurboWarp Desktop, hover over the extension and press the add button. - <% } else { %> - To use multiple of these extensions in TurboWarp, hover over the extension and press the button to copy its URL. Then go to the editor, open the extension chooser, then choose the "Custom Extension" option at the bottom, and enter the URL. - <% } %> -

- -
-
These extensions are not compatible with Scratch.
- Projects that use these extensions can't be uploaded to the Scratch website. - They can, however, be used in the packager. -
- -
- <% if (mode === 'desktop') { %> -
Some extensions may be outdated or missing.
- <% - const now = new Date(); - const date = `${now.getFullYear()}-${(now.getMonth() + 1).toString().padStart(2, '0')}-${now.getDate().toString().padStart(2, '0')}`; - %> - For compatibility, security, and offline support, TurboWarp Desktop includes an offline copy of extensions.turbowarp.org from this update's release date (<%= date %>). - Compared to the live website, some extensions may be missing or outdated. - <% } else { %> -
Some extensions may not work in TurboWarp Desktop.
- For compatibility, security, and offline support, each TurboWarp Desktop update contains an offline copy of these extensions from its release date, so some extensions may be outdated or missing. - Use the latest update for best results. - <% } %> -
-
- - <% - const getLinkToRun = (extensionPath) => `https://turbowarp.org/editor?extension=${host}${extensionPath}`; - %> - - <% if (mode === "development") { %> -
-
-

Development Server Tools

-

- Most recently modified extensions: - <% for (const extension of mostRecentExtensions) { %> - <%= extension %> - <% } %> -

-
-
- <% } %> - - - - <% if (mode === 'desktop') { %> - - <% } %> - - <% - const banner = (extensionFile, options = {}) => { - if (extensionFile.endsWith('.js')) { - return `Do not add .js when calling banner(): ${extensionFile}`; - } - const imageSource = extensionImages[extensionFile] ? `images/${extensionImages[extensionFile]}` : 'images/unknown.svg'; - const imageClasses = ["extension-image"]; - if (options.invertDark) imageClasses.push("invert-dark"); - return ` -
- -
- - ${mode === "desktop" ? (` - - `) : (` - Open Extension - `)} -
-
` - }; - - const img = (extensionFile) => { - if (extensionImages[extensionFile]) return `images/${extensionImages[extensionFile]}`; - return 'images/unknown.svg'; - }; - %> - -
-
-
- <%- banner('lab/text') %> -

Animated Text

-

An easy way to display and animate text. Compatible with Scratch Lab's Animated Text experiment.

-
- -
- <%- banner('stretch') %> -

Stretch

-

Stretch sprites horizontally or vertically.

-
- -
- <%- banner('gamepad') %> -

Gamepad

-

Directly access gamepads instead of just mapping buttons to keys.

-
- -
- <%- banner('box2d') %> -

Box2D Physics

-

Two dimensional physics. Originally created by griffpatch.

-
- -
- <%- banner('files') %> -

Files

-

Read and download files.

-
- -
- <%- banner('pointerlock') %> -

Pointerlock

-

Adds blocks for mouse locking. Mouse x & y blocks will report the change since the previous frame while the pointer is locked. Replaces the pointerlock experiment.

-
- -
- <%- banner('cursor') %> -

Mouse Cursor

-

Use custom cursors or hide the cursor. Also allows replacing the cursor with any costume image.

-
- -
- <%- banner('runtime-options') %> -

Runtime Options

-

Get and modify turbo mode, framerate, interpolation, clone limit, stage size, and more.

-
- -
- <%- banner('fetch') %> -

Fetch

-

Make requests to the broader internet.

-
- -
- <%- banner('text') %> -

Text

-

Manipulate characters and text. Originally created by CST1229.

-
- -
- <%- banner('local-storage') %> -

Local Storage

-

Store data persistently. Like cookies, but better.

-
- -
- <%- banner('true-fantom/base') %> -

Base

-

Convert numbers between bases. Created by TrueFantom.

-
- -
- <%- banner('bitwise') %> -

Bitwise

-

Blocks that operate on the binary representation of numbers in computers. Modified by TrueFantom.

-
- -
- <%- banner('Skyhigh173/bigint') %> -

BigInt

-

Math blocks that work on infinitely large integers (no decimals). Created by Skyhigh173.

-
- -
- <%- banner('utilities') %> -

Utilities

-

A bunch of interesting blocks. Originally created by Sheep_maker.

-
- -
- <%- banner('sound') %> -

Sound

-

Play sounds from URLs.

-
- -
- <%- banner('Xeltalliv/clippingblending') %> -

Clipping & Blending

-

Clipping outside of a specified rectangular area and additive color blending. Created by Vadik1.

-
- -
- <%- banner('clipboard') %> -

Clipboard

-

Read and write from the system clipboard.

-
- -
- <%- banner('penplus') %> -

Pen Plus

-

Advanced rendering capabilities. Created by ObviousAlexC.

-
- -
- <%- banner('obviousAlexC/SensingPlus') %> -

Sensing Plus

-

An extension to the sensing category. Created by ObviousAlexC.

-
- -
- <%- banner('Lily/ClonesPlus') %> -

Clones Plus

-

Expansion of Scratch's clone features. Created by LilyMakesThings.

-
- -
- <%- banner('Lily/LooksPlus') %> -

Looks Plus

-

Expands upon the looks category, allowing you to show/hide, get costume data and edit SVG skins on sprites. Created by LilyMakesThings.

-
- -
- <%- banner('navigator') %> -

Navigator

-

Details about the user's browser and operating system.

-
- -
- <%- banner('battery') %> -

Battery

-

Access information about the battery of phones or laptops. May not work on all devices and browsers.

-
- -
- <%- banner('mdwalters/notifications') %> -

Notifications

-

Display notifications.

-
- -
- <%- banner('ar') %> -

Augmented Reality

-

Shows image from camera and performs motion tracking, allowing 3D projects to correctly overlay virtual objects on real world. Created by Vadik1.

-
- -
- <%- banner('encoding') %> -

Encoding

-

Encode and decode strings into their unicode numbers, base 64, or URLs. Created by -SIPC-.

-
- -
- <%- banner('Lily/MoreTimers') %> -

More Timers

-

Control several timers at once. Created by LilyMakesThings.

-
- -
- <%- banner('clouddata-ping') %> -

Ping Cloud Data

-

Determine whether a cloud variable server is probably up. Originally created by TheShovel.

-
- -
- <%- banner('cloudlink') %> -

Cloudlink

-

Powerful WebSocket extension for Scratch 3. Created by MikeDEV.

-
- -
- <%- banner('true-fantom/network') %> -

Network

-

Various blocks for interacting with the network. Created by TrueFantom.

-
- -
- <%- banner('true-fantom/math') %> -

Math

-

A lot of operators blocks, from exponentiation to trigonometric functions. Created by TrueFantom.

-
- -
- <%- banner('true-fantom/regexp') %> -

RegExp

-

Full interface for working with Regular Expressions. Created by TrueFantom.

-
- -
- <%- banner('true-fantom/couplers') %> -

Couplers

-

A few adapter blocks. Created by TrueFantom.

-
- -
- <%- banner('Lily/Cast') %> -

Cast

-

Convert values between types. Created by LilyMakesThings

-
- -
- <%- banner('-SIPC-/time') %> -

Time

-

Blocks for interacting with unix timestamps and other date strings. Created by -SIPC-.

-
- -
- <%- banner('-SIPC-/consoles') %> -

Consoles

-

Blocks that interact the JavaScript console built in to your browser's developer tools. Created by -SIPC-.

-
- -
- <%- banner('ZXMushroom63/searchApi') %> -

Search Params

-

Interact with URL search parameters: the part of the URL after a question mark. Created by ZXMushroom63.

-
- -
- <%- banner('TheShovel/ShovelUtils') %> -

ShovelUtils

-

A bunch of miscellaneous blocks. Created by TheShovel.

-
- -
- <%- banner('Skyhigh173/json') %> -

JSON

-

Handle JSON strings and arrays. Created by Skyhigh173.

-
- -
- <%- banner('cs2627883/numericalencoding') %> -

Numerical Encoding

-

Encode strings as numbers for cloud variables. Created by cs2627883.

-
- -
- <%- banner('DT/cameracontrols') %> -

Camera Controls

-

Move the visible part of the stage. Created by DT.

-
- -
- <%- banner('TheShovel/CanvasEffects') %> -

Canvas Effects

-

Apply visual effects to the entire stage. Created by TheShovel.

-
- -
- <%- banner('Longboost/color_channels') %> -

RGB Channels

-

Only render or stamp certain RGB channels.

-
- -
- <%- banner('Lily/TempVariables2') %> -

Temporary Variables

-

Create disposable runtime or thread variables. Created by LilyMakesThings.

-
- -
- <%- banner('CST1229/zip') %> -

Zip

-

Create and edit .zip format files, including .sb3 files. Created by CST1229.

-
- -
- <%- banner('0832/rxFS2') %> -

rxFS

-

Blocks for interacting with a virtual in-memory filesystem. Created by 0832.

-
- -
- <%- banner('NexusKitten/sgrab') %> -

S-Grab

-

Get information about Scratch projects and Scratch users. Created by NamelessCat.

-
- -
- <%- banner('NOname-awa/graphics2d') %> -

Graphics 2D

-

Blocks to compute lengths, angles, and areas in two dimensions. Created by NOname-awa.

-
- -
- <%- banner('NOname-awa/more-comparisons') %> -

More Comparisons

-

More comparison blocks. Created by NOname-awa.

-
- -
- <%- banner('JeremyGamer13/tween') %> -

Tween

-

Easing methods for smooth animations. Created by JeremyGamer13

-
- -
- <%- banner('rixxyx') %> -

RixxyX

-

Various utility blocks. Created by RixTheTyrunt.

-
- -
- <%- banner('qxsck/data-analysis') %> -

Data Analysis

-

Blocks to compute means, medians, maximums, minimums, variances, and modes. Created by qxsck.

-
- -
- <%- banner('qxsck/var-and-list') %> -

Variable and list

-

More blocks related to variables and lists. Created by qxsck.

-
- -
- <%- banner('vercte/dictionaries') %> -

Dictionaries

-

Use the power of dictionaries in your project. Created by Vercte.

-
- -
- <%- banner('godslayerakp/http') %> -

HTTP

-

Comprehensive extension for interacting with external websites. Created by RedMan13.

-
- -
- <%- banner('Lily/CommentBlocks') %> -

Comment Blocks

-

Annotate your scripts. Created by LilyMakesThings.

-
- -
- <%- banner('CubesterYT/TurboHook') %> -

TurboHook

-

Allows you to use webhooks. Created by CubesterYT.

-
- -
- <%- banner('Alestore/nfcwarp') %> -

NFCWarp

-

Allows reading data from NFC (NDEF) devices. Only works in Chrome on Android. Created by Alestore Games.

-
- -
- <%- banner('itchio') %> -

itch.io

-

Blocks that interact with the itch.io website. Unofficial. Created by softed.

-
- -
- <%- banner('gamejolt') %> -

GameJolt

-

Blocks that allow games to interact with the GameJolt API. Unofficial. Created by softed.

-
- -
- <%- banner('obviousAlexC/newgroundsIO') %> -

Newgrounds

-

Blocks that allow games to interact with the Newgrounds API. Unofficial. Created by ObviousAlexC.

-
- -
- <%- banner('Lily/McUtils') %> -

McUtils

-

Helpful utilities for any fast food employee. Created by LilyMakesThings.

-
-
-
- - - - diff --git a/website/local-storage.html b/website/local-storage.html deleted file mode 100644 index 3f1262abf6..0000000000 --- a/website/local-storage.html +++ /dev/null @@ -1,188 +0,0 @@ - - - - - - Local Storage docs - - - - - - - -
- How to use Local Storage -
-
-

About this extension

- -

This extension allows you to automatically save plain text in storage. Forget save codes! With this extension, we can make a game that doesn't require any user interaction to save progress.

- -

Features

- -

Namespaces

- -

The namespace is basically like the file you want to read and write to. Each project should set this to something unique, such as developer/project title so each project gets its own storage space. If two different projects use the same namespace, they will overwrite eachother's data and cause very bad things!

- -

You can set which namespace to use with this block:

- -
setstoragenamespaceIDtoproject title
- -

Some example namespaces:

-
    -
  • griffpatch/Paper Minecraft
  • -
  • Untitled-37 by TestMuffin
  • -
- -

The format isn't important -- it just needs to be unique. Don't include a version number like "v1.4" in the namespace unless you want to discard data from old versions of your project.

- -

Reading and writing

- -

After setting the namespace, you're ready to read from and write to storage.

- -

You can store data in keys, which are kind of like names of variables in any Scratch project, except they persist between sessions.

- -

You can store data in storage keys with this block:

- -
setkeyscoreto1000
- -

And read it with:

- -
getkeyscore
- -

If you want to delete a key and its value, use this block:

- -
deletekeyscore
- -

Or wipe everything stored in the namespace:

- -
deleteallkeys
- -

Loading data into memory

- -

Relying on the disk to read information that gets saved such as a player's progress or stats can be pretty slow. That's why it's very useful to store this data in variables while it's in use. Variables are like your project's random-access memory. One way of doing this is by getting all keys from storage and putting their values in variables as part of your project's initialization process.

- -

If you're unfamiliar with computer memory, think of reading local storage as opening up your dictionary to look for a definition for someone and reading memory as remembering the definition you just read and saying it to someone. Of course, you wouldn't want to be constantly running to get the dictionary every time the same definition was requested.

- -

While browsers actually hold this data in memory automatically for quicker access, it's still more efficient and a better practice to take some work off the browser by not constantly getting the same storage key over and over again for no reason, and is much more important to know and think about if you ever use other programming languages where you don't want the disk to have to spend time reading the same data over and over again when it could be kept in memory (that's what memory is for).

- -

For example, in a game, you can speed up code that needs to know how many coins the player has collected by getting the storage key for coins and then loading it into a variable, just once on startup.

- -
foreverifgetkeycoins>99thenbroadcast1-UPDon't do thissetcoinstogetkeycoinsDo this insteadforeverifcoins>99thenbroadcast1-UP
- -

So in general, we don't really need to load data from storage again once we have it in memory. If we already know what's in storage and what we're writing to storage because that data is already present in the project's variables, we really only need to read from storage once to initialize and then we're good.

- -

...Right?

- -

Handling interference from other windows

- -

Sometimes a user may open the same project in multiple tabs or windows, each of which could be trying to read and write data to the same space. If this causes a desync, it can result in unexpected behavior.

- -

Here's an example scenario. Suppose someone opens the same game twice by accident. They play in Window A for a while and save the game. Then they close that window and do something else. Later, they come back to the other window they had opened before and start playing in Window B, but all the progress is "gone" because that window had already been running the game and had already loaded the save data before Window A had saved the progress that was made, and it's too late because they had saved the game in Window B before they realized the problem. This is unfortunate for the player, but more importantly for you, the developer, what is the project supposed to do now? Mix the data?

- -

So you can see how it's a good idea to consider that people may have multiple windows of the same project open intentionally or accidentally.

- -

You don't have to do things like auto-refresh content if you didn't intend for your project to support multi-window usage, but it's nice to at least make sure no glitches happen if someone accidentally had multiple windows open. Here are a few ways you can deal with this problem:

- -

Initialize by loading from storage

- -

For games and other projects that are intended for use only in one window, we recommend you load all storage keys you need into variables only when the project starts so that nothing changes if a second instance of the project writes to storage while you're still using the first one. Then, when you need to save data, rewrite everything in the same group of data (like everything in the same save file in a game) to storage at the same time so nothing from other instances of the project gets mixed in.

- -

The worst that could happen with this implementation is that "data A" might be overwritten by "data B", if the user made a mistake.

- -

Reload data from storage as needed

- -

Sometimes you might want to respond to changes in local storage. There is a block to help with this:

- -
whenanotherwindowchangesstorage
- -

The code under this block will run whenever a different instance of the project writes to storage or if a different project that's using the same namespace writes to storage. This allows your project to properly respond to and handle these events as they happen.

- -

This may or may not be important. You probably wouldn't want to use this technique in a game - games don't need to respond to other instances of themselves writing to storage. Plus, if game save data from one window is mixed in with data from another, it can cause glitches like sequence breaks.

- -

This kind of thing is more useful if you made something like a file system simulator that you want to have auto-refresh if the user makes edits in other windows, you may want to use this so that the content being displayed stays up to date even if another window modifies it.

- -

You could do this by constantly getting the storage key, but it's better to only grab keys from storage when necessary. The block above is how you do that.

- -

Merge the data

- -

This one is more advanced and the way you would code it depends on the project, but you could make it so that when you're about to save data and another window wrote data that was not loaded into the first window, the two are merged - the data being written is merged with the data that was already present, so that if you collected 100 coins in one session and 100 XP in another, both of those changes in the save data would stay.

- -

Make sure to do this correctly because if data is merged incorrectly, it can cause glitches like sequence breaks in a game.

- -

I said that this is advanced, because sometimes these algorithms can get confused when merging changes to the same piece of data. (Like, what are we supposed to do if we're trying to merge two changes, one of which changes "A" to "B" and the other changes the same "A" to "C"?) This is known as a merge conflict. If you don't have any way to prioritize one change over another, you'll just be stuck with two branches of data.

- -

Local storage limits

- -

This extension uses the browser's local storage API, which limits each website to around 5 MiB or 5,242,800 bytes of local storage data, so if we want local storage to be able to hold data for many projects, each one should stay well below this limit. We recommend only storing small files such as game save data or settings in local storage.

- -

In rare instances, such as when a system is running out of disk space, the browser may delete our data to make room for something else. We, unfortunately, cannot influence when this happens.

- -

Credits

- -

This page was written by DNin01.

- -

The block SVGs were generated with scratchblocks.

-
- - diff --git a/website/test/loops.js b/website/test/loops.js new file mode 100644 index 0000000000..ae1bf2f6c3 --- /dev/null +++ b/website/test/loops.js @@ -0,0 +1,71 @@ +// From https://github.com/TurboWarp/scratch-vm/pull/141 +(function (Scratch) { + "use strict"; + + class LoopsAndThings { + getInfo() { + return { + id: "loopsAndThings", + name: "Loops and things test", + blocks: [ + { + opcode: "conditional", + blockType: Scratch.BlockType.CONDITIONAL, + text: "run branch [BRANCH] of", + arguments: { + BRANCH: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: 1 + } + }, + branchCount: 3 + }, + { + opcode: "loop", + blockType: Scratch.BlockType.LOOP, + text: "my repeat [TIMES]", + arguments: { + TIMES: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: 10 + } + }, + }, + '---', + { + opcode: "testPromise", + blockType: Scratch.BlockType.REPORTER, + text: "return [VALUE] in a Promise", + arguments: { + VALUE: { + type: Scratch.ArgumentType.STRING, + defaultValue: '' + } + } + } + ] + }; + } + + conditional({BRANCH}, util) { + return Scratch.Cast.toNumber(BRANCH); + } + + loop({TIMES}, util) { + const times = Math.round(Scratch.Cast.toNumber(TIMES)); + if (typeof util.stackFrame.loopCounter === "undefined") { + util.stackFrame.loopCounter = times; + } + util.stackFrame.loopCounter--; + if (util.stackFrame.loopCounter >= 0) { + return true; + } + } + + testPromise({VALUE}) { + return Promise.resolve(VALUE); + } + } + + Scratch.extensions.register(new LoopsAndThings()); +})(Scratch); \ No newline at end of file diff --git a/website/turbowarp.svg b/website/turbowarp.svg new file mode 100644 index 0000000000..6a1ec7f87d --- /dev/null +++ b/website/turbowarp.svg @@ -0,0 +1,68 @@ + +