diff --git a/.eslintrc.js b/.eslintrc.js index 7e9fd14fee..db32ceab95 100644 --- a/.eslintrc.js +++ b/.eslintrc.js @@ -17,24 +17,6 @@ module.exports = { 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 9af7a91309..16779c139d 100644 --- a/CONTRIBUTING.md +++ b/CONTRIBUTING.md @@ -29,6 +29,8 @@ Every merged extension is more code that we will be expected to maintain indefin 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. +Every extension is also covered under [our bug bounty](https://github.com/TurboWarp/extensions/security/policy), so mindlessly merging things will have a direct impact on my wallet. + ## Writing extensions 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. @@ -45,12 +47,13 @@ The header comments look like this: ```js // Name: Example Extension +// ID: extensionid // Description: Does a very cool thing. This must have punctuation at the end! // By: GarboMuffin // Original: TestMuffin ``` -Remember, this has to be the *very first* thing in the JS file. `Name` and `Description` are required. 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. +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. This metadata is parsed by a script to generate the website and extension library. It tries to be pretty loose, but don't deviate too far. You must use `//`, not `/* */`. 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. @@ -112,41 +115,40 @@ You **MUST** avoid using any code or images under these licenses as we believe t We take licenses very seriously. License violations are one of the few things that can force us to break project compatibility. -## Code style - -Our preferred code style is: - - - Indent with 2 spaces - - Use semicolons - - Use whitespace liberally - - Comment liberally, but don't just re-explain what the code literally says (`var x = 3; // set x to 3` is a bad comment) - - Pick one type of quotes and be consistent (if unsure, we like single quotes) - ## Type checking If you use our development server, 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. 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. -## Linting +## Linting, validation, and formatting -We use ESLint to automatically common problems in pull requests. To run our linting rules on your computer, run: +All pull requests are automatically checked by a combination of custom validation scripts, [ESLint](https://eslint.org/), and [Prettier](https://prettier.io/). Don't worry about passing these checks on the first attempt -- most don't. That's why we have these checks. + +Our custom validation scripts do things like making sure you have the correct headers at the start of your extension and that the images are the right size. **Your extension must pass validation.** You can run them locally with: ```bash -npm run lint +npm run validate ``` -ESLint can automatically fix certain issues. You can run this with: +ESLint detects common JavaScript errors such as referencing non-existant variables. **Your extension must pass linting.** You can run it locally with: ```bash -npm run fix +npm run lint ``` -Our ESLint configuration separates between *warnings* and *errors*. They both cause big red error messages to appear on pull requests, but they are different: +You are allowed to [disable ESLint warnings and errors](https://eslint.org/docs/latest/use/configure/rules#disabling-rules) as needed, but please only do so if actually required. - - Warnings are typically minor style issues. If you ignore these we will fix it ourselves. This usually takes 15 seconds to address. - - Errors are actually problems. Please read and address all of them. Extensions with errors may take longer to review as your extension is unfinished. +When including third-party code, especially minified code, you may use `/* eslint-disable*/` and `/* eslint-enable */` markers to disable linting for that entire section. -You are allowed to [disable warnings or errors](https://eslint.org/docs/latest/use/configure/rules#disabling-rules) as needed, but only if actually required. +We use Prettier to ensure consistent code formatting. **Your extension does not need to pass format; we will fix it for you if linting and validation pass.** You can format your code automatically with: -When including third-party code, especially minified code, you may use `/* eslint-disable*/` and `/* eslint-enable */` markers to disable linting for that entire section. +```bash +npm run format +``` + +To just check formatting, use: + +```bash +npm run check-format +``` diff --git a/development/build-production.js b/development/build-production.js index c4664ac017..39f50abffe 100644 --- a/development/build-production.js +++ b/development/build-production.js @@ -1,9 +1,9 @@ -const pathUtil = require('path'); -const Builder = require('./builder'); +const pathUtil = require("path"); +const Builder = require("./builder"); -const outputDirectory = pathUtil.join(__dirname, '..', 'build'); +const outputDirectory = pathUtil.join(__dirname, "..", "build"); -const builder = new Builder('production'); +const builder = new Builder("production"); const build = builder.build(); build.export(outputDirectory); diff --git a/development/builder.js b/development/builder.js index a9c735b62c..292758e484 100644 --- a/development/builder.js +++ b/development/builder.js @@ -1,11 +1,9 @@ -const fs = require('fs'); -const pathUtil = require('path'); -const sizeOfImage = require('image-size'); -const renderTemplate = require('./render-template'); -const renderDocs = require('./render-docs'); -const compatibilityAliases = require('./compatibility-aliases'); -const parseMetadata = require('./parse-extension-metadata'); -const featuredExtensionsIDs = require('../extensions/extensions.json'); +const fs = require("fs"); +const AdmZip = require("adm-zip"); +const pathUtil = require("path"); +const compatibilityAliases = require("./compatibility-aliases"); +const parseMetadata = require("./parse-extension-metadata"); +const featuredExtensionSlugs = require("../extensions/extensions.json"); /** * @typedef {'development'|'production'|'desktop'} Mode @@ -20,14 +18,17 @@ const featuredExtensionsIDs = require('../extensions/extensions.json'); const recursiveReadDirectory = (directory) => { const result = []; for (const name of fs.readdirSync(directory)) { - if (name.startsWith('.')) { + if (name.startsWith(".")) { // Ignore .eslintrc.js, .DS_Store, etc. continue; } const absolutePath = pathUtil.join(directory, name); const stat = fs.statSync(absolutePath); if (stat.isDirectory()) { - for (const [relativeToChildName, childAbsolutePath] of recursiveReadDirectory(absolutePath)) { + for (const [ + relativeToChildName, + childAbsolutePath, + ] of recursiveReadDirectory(absolutePath)) { // This always needs to use / on all systems result.push([`${name}/${relativeToChildName}`, childAbsolutePath]); } @@ -39,72 +40,87 @@ const recursiveReadDirectory = (directory) => { }; class BuildFile { - constructor (source) { + constructor(source) { this.sourcePath = source; } - getType () { + getType() { return pathUtil.extname(this.sourcePath); } - getLastModified () { + getLastModified() { return fs.statSync(this.sourcePath).mtimeMs; } - read () { + read() { return fs.readFileSync(this.sourcePath); } - validate () { + validate() { // no-op by default } } class ExtensionFile extends BuildFile { - constructor (absolutePath, featured) { + constructor(absolutePath, featured) { super(absolutePath); this.featured = featured; } - getMetadata () { - const data = fs.readFileSync(this.sourcePath, 'utf-8'); + getMetadata() { + const data = fs.readFileSync(this.sourcePath, "utf-8"); return parseMetadata(data); } - validate () { + validate() { if (!this.featured) { return; } const metadata = this.getMetadata(); + if (!metadata.id) { + throw new Error("Missing // ID:"); + } + if (!metadata.name) { - throw new Error('Missing // Name:'); + throw new Error("Missing // Name:"); } if (!metadata.description) { - throw new Error('Missing // Description:'); + throw new Error("Missing // Description:"); } - const PUNCTUATION = ['.', '!', '?']; - if (!PUNCTUATION.some((punctuation) => metadata.description.endsWith(punctuation))) { - throw new Error(`Description is missing punctuation: ${metadata.description}`); + const PUNCTUATION = [".", "!", "?"]; + if ( + !PUNCTUATION.some((punctuation) => + metadata.description.endsWith(punctuation) + ) + ) { + throw new Error( + `Description is missing punctuation: ${metadata.description}` + ); } for (const person of [...metadata.by, ...metadata.original]) { if (!person.name) { - throw new Error('Person is missing name'); + throw new Error("Person is missing name"); } - if (person.link && !person.link.startsWith('https://scratch.mit.edu/users/')) { - throw new Error(`Link for ${person.name} does not point to a Scratch user`); + if ( + person.link && + !person.link.startsWith("https://scratch.mit.edu/users/") + ) { + throw new Error( + `Link for ${person.name} does not point to a Scratch user` + ); } } } } class HomepageFile extends BuildFile { - constructor (extensionFiles, extensionImages, mode) { - super(pathUtil.join(__dirname, 'homepage-template.ejs')); + constructor(extensionFiles, extensionImages, withDocs, samples, mode) { + super(pathUtil.join(__dirname, "homepage-template.ejs")); /** @type {Record} */ this.extensionFiles = extensionFiles; @@ -112,34 +128,66 @@ class HomepageFile extends BuildFile { /** @type {Record} */ this.extensionImages = extensionImages; + /** @type {Map} */ + this.withDocs = withDocs; + + /** @type {SampleFile[]} */ + this.samples = samples; + /** @type {Mode} */ this.mode = mode; - this.host = mode === 'development' ? 'http://localhost:8000/' : 'https://extensions.turbowarp.org/'; + this.host = + mode === "development" + ? "http://localhost:8000/" + : "https://extensions.turbowarp.org/"; } - getType () { - return '.html'; + getType() { + return ".html"; } - getFullExtensionURL (extensionID) { - return `${this.host}${extensionID}.js`; + getFullExtensionURL(extensionSlug) { + return `${this.host}${extensionSlug}.js`; } - getRunExtensionURL (extensionID) { - return `https://turbowarp.org/editor?extension=${this.getFullExtensionURL(extensionID)}`; + getDocumentationURL(extensionSlug) { + return `${this.host}${extensionSlug}`; } - read () { + getRunExtensionURL(extensionSlug) { + return `https://turbowarp.org/editor?extension=${this.getFullExtensionURL( + extensionSlug + )}`; + } + + /** + * @param {SampleFile} sampleFile + * @returns {string} + */ + getRunSampleURL(sampleFile) { + const path = encodeURIComponent(`samples/${sampleFile.getSlug()}`); + return `https://turbowarp.org/editor?project_url=${this.host}${path}`; + } + + read() { + const renderTemplate = require("./render-template"); + const mostRecentExtensions = Object.entries(this.extensionFiles) .sort((a, b) => b[1].getLastModified() - a[1].getLastModified()) .slice(0, 5) .map((i) => i[0]); - const extensionMetadata = Object.fromEntries(featuredExtensionsIDs.map((id) => [ - id, - this.extensionFiles[id].getMetadata() - ])); + const extensionMetadata = Object.fromEntries( + featuredExtensionSlugs.map((slug) => [ + slug, + { + ...this.extensionFiles[slug].getMetadata(), + hasDocumentation: this.withDocs.has(slug), + samples: this.samples.get(slug) || [], + }, + ]) + ); return renderTemplate(this.sourcePath, { mode: this.mode, @@ -147,13 +195,15 @@ class HomepageFile extends BuildFile { extensionImages: this.extensionImages, extensionMetadata, getFullExtensionURL: this.getFullExtensionURL.bind(this), - getRunExtensionURL: this.getRunExtensionURL.bind(this) + getRunExtensionURL: this.getRunExtensionURL.bind(this), + getDocumentationURL: this.getDocumentationURL.bind(this), + getRunSampleURL: this.getRunSampleURL.bind(this), }); } } class JSONMetadataFile extends BuildFile { - constructor (extensionFiles, extensionImages) { + constructor(extensionFiles, extensionImages, withDocs, samples) { super(null); /** @type {Record} */ @@ -161,142 +211,207 @@ class JSONMetadataFile extends BuildFile { /** @type {Record} */ this.extensionImages = extensionImages; + + /** @type {Set} */ + this.withDocs = withDocs; + + /** @type {Map} */ + this.samples = samples; } - getType () { - return '.json'; + getType() { + return ".json"; } - read () { + read() { const extensions = []; - for (const extensionID of featuredExtensionsIDs) { + for (const extensionSlug of featuredExtensionSlugs) { const extension = {}; - const file = this.extensionFiles[extensionID]; + const file = this.extensionFiles[extensionSlug]; const metadata = file.getMetadata(); - const image = this.extensionImages[extensionID]; + const image = this.extensionImages[extensionSlug]; - extension.id = extensionID; + extension.slug = extensionSlug; + extension.id = metadata.id; extension.name = metadata.name; extension.description = metadata.description; if (image) { extension.image = image; } + if (metadata.by.length) { + extension.by = metadata.by; + } + if (metadata.original.length) { + extension.original = metadata.original; + } + if (this.withDocs.has(extensionSlug)) { + extension.docs = true; + } + const samples = this.samples.get(extensionSlug); + if (samples) { + extension.samples = samples.map((i) => i.getTitle()); + } + extensions.push(extension); } const data = { - extensions + extensions, }; return JSON.stringify(data); } } class ImageFile extends BuildFile { - validate () { + validate() { + const sizeOfImage = require("image-size"); const contents = this.read(); - const {width, height} = sizeOfImage(contents); + const { width, height } = sizeOfImage(contents); const aspectRatio = width / height; if (aspectRatio !== 2) { - throw new Error(`Aspect ratio must be exactly 2, but found ${aspectRatio.toFixed(4)} (${width}x${height})`); + throw new Error( + `Aspect ratio must be exactly 2, but found ${aspectRatio.toFixed( + 4 + )} (${width}x${height})` + ); } } } class SVGFile extends ImageFile { - validate () { + validate() { const contents = this.read(); - if (contents.includes(' elements -- please convert the text to a path. This ensures it will display correctly on all devices.'); + if (contents.includes(" elements -- please convert the text to a path. This ensures it will display correctly on all devices." + ); } super.validate(); } } +const IMAGE_FORMATS = new Map(); +IMAGE_FORMATS.set(".png", ImageFile); +IMAGE_FORMATS.set(".jpg", ImageFile); +IMAGE_FORMATS.set(".svg", SVGFile); + class SitemapFile extends BuildFile { - constructor (build) { + constructor(build) { super(null); this.build = build; } - getType () { - return '.xml'; + getType() { + return ".xml"; } - read () { - let xml = ''; + read() { + let xml = ""; xml += '\n'; xml += '\n'; xml += Object.keys(this.build.files) - .filter(file => file.endsWith('.html')) - .map(file => file.replace('index.html', '').replace('.html', '')) + .filter((file) => file.endsWith(".html")) + .map((file) => file.replace("index.html", "").replace(".html", "")) .sort((a, b) => { if (a.length < b.length) return -1; if (a.length > b.length) return 1; return a - b; }) - .map(path => `https://extensions.turbowarp.org${path}`) - .map(absoluteURL => `${absoluteURL}`) - .join('\n'); + .map((path) => `https://extensions.turbowarp.org${path}`) + .map((absoluteURL) => `${absoluteURL}`) + .join("\n"); - xml += '\n'; + xml += "\n"; return xml; } } -const IMAGE_FORMATS = new Map(); -IMAGE_FORMATS.set('.png', ImageFile); -IMAGE_FORMATS.set('.jpg', ImageFile); -IMAGE_FORMATS.set('.svg', SVGFile); - class DocsFile extends BuildFile { - constructor (absolutePath, extensionId) { + constructor(absolutePath, extensionSlug) { super(absolutePath); - this.extensionId = extensionId; + this.extensionSlug = extensionSlug; } - read () { - const markdown = super.read().toString('utf-8'); - return renderDocs(markdown, this.extensionId); + read() { + const renderDocs = require("./render-docs"); + const markdown = super.read().toString("utf-8"); + return renderDocs(markdown, this.extensionSlug); } - getType () { - return '.html'; + getType() { + return ".html"; } } -class Build { - constructor (mode) { - /** @type {Mode} */ - this.mode = mode; +class SampleFile extends BuildFile { + getSlug() { + return pathUtil.basename(this.sourcePath); + } + + getTitle() { + return this.getSlug().replace(".sb3", ""); + } + + /** @returns {string[]} list of full URLs */ + getExtensionURLs() { + const zip = new AdmZip(this.sourcePath); + const entry = zip.getEntry("project.json"); + if (!entry) { + throw new Error("package.json missing"); + } + const data = JSON.parse(entry.getData().toString("utf-8")); + return data.extensionURLs ? Object.values(data.extensionURLs) : []; + } + + validate() { + const urls = this.getExtensionURLs(); + + if (urls.length === 0) { + throw new Error("Has no extensions"); + } + for (const url of urls) { + if ( + !url.startsWith("https://extensions.turbowarp.org/") || + !url.endsWith(".js") + ) { + throw new Error(`Invalid extension URL for sample: ${url}`); + } + } + } +} + +class Build { + constructor() { this.files = {}; } - getFile (path) { - return this.files[path] || this.files[`${path}.html`] || this.files[`${path}index.html`] || null; + getFile(path) { + return ( + this.files[path] || + this.files[`${path}.html`] || + this.files[`${path}index.html`] || + null + ); } - export (root) { + export(root) { try { fs.rmSync(root, { - recursive: true + recursive: true, }); } catch (e) { - if (e.code !== 'ENOENT') { + if (e.code !== "ENOENT") { throw e; } } for (const [relativePath, file] of Object.entries(this.files)) { - if (this.mode === 'desktop' && relativePath.endsWith('.html')) { - continue; - } - const directoryName = pathUtil.dirname(relativePath); fs.mkdirSync(pathUtil.join(root, directoryName), { - recursive: true + recursive: true, }); fs.writeFileSync(pathUtil.join(root, relativePath), file.read()); } @@ -307,82 +422,137 @@ class Builder { /** * @param {Mode} mode */ - constructor (mode) { - if (process.argv.includes('--production')) { - this.mode = 'production'; - } else if (process.argv.includes('--development')) { - this.mode = 'development'; - } else if (process.argv.includes('--desktop')) { - this.mode = 'desktop'; + constructor(mode) { + if (process.argv.includes("--production")) { + this.mode = "production"; + } else if (process.argv.includes("--development")) { + this.mode = "development"; + } else if (process.argv.includes("--desktop")) { + this.mode = "desktop"; } else { /** @type {Mode} */ this.mode = mode; } - this.extensionsRoot = pathUtil.join(__dirname, '../extensions'); - this.websiteRoot = pathUtil.join(__dirname, '../website'); - this.imagesRoot = pathUtil.join(__dirname, '../images'); - this.docsRoot = pathUtil.join(__dirname, '../docs'); + this.extensionsRoot = pathUtil.join(__dirname, "../extensions"); + this.websiteRoot = pathUtil.join(__dirname, "../website"); + this.imagesRoot = pathUtil.join(__dirname, "../images"); + this.docsRoot = pathUtil.join(__dirname, "../docs"); + this.samplesRoot = pathUtil.join(__dirname, "../samples"); } - build () { + build() { const build = new Build(this.mode); + /** @type {Record} */ + const extensionFiles = {}; + for (const [filename, absolutePath] of recursiveReadDirectory( + this.extensionsRoot + )) { + if (!filename.endsWith(".js")) { + continue; + } + const extensionSlug = filename.split(".")[0]; + const featured = featuredExtensionSlugs.includes(extensionSlug); + const file = new ExtensionFile(absolutePath, featured); + extensionFiles[extensionSlug] = file; + build.files[`/${filename}`] = file; + } + /** @type {Record} */ const extensionImages = {}; - for (const [filename, absolutePath] of recursiveReadDirectory(this.imagesRoot)) { + for (const [filename, absolutePath] of recursiveReadDirectory( + this.imagesRoot + )) { const extension = pathUtil.extname(filename); const ImageFileClass = IMAGE_FORMATS.get(extension); if (!ImageFileClass) { continue; } - const extensionId = filename.split('.')[0]; - if (extensionId !== 'unknown') { - extensionImages[extensionId] = `images/${filename}`; + const extensionSlug = filename.split(".")[0]; + if (extensionSlug !== "unknown") { + extensionImages[extensionSlug] = `images/${filename}`; } build.files[`/images/${filename}`] = new ImageFileClass(absolutePath); } - for (const [filename, absolutePath] of recursiveReadDirectory(this.docsRoot)) { - if (!filename.endsWith('.md')) { + /** @type {Set} */ + const extensionsWithDocs = new Set(); + + /** @type {Map} */ + const samples = new Map(); + for (const [filename, absolutePath] of recursiveReadDirectory( + this.samplesRoot + )) { + if (!filename.endsWith(".sb3")) { continue; } - const extensionId = filename.split('.')[0]; - build.files[`/${extensionId}.html`] = new DocsFile(absolutePath, extensionId); + + const file = new SampleFile(absolutePath); + for (const url of file.getExtensionURLs()) { + const slug = new URL(url).pathname.substring(1).replace(".js", ""); + if (samples.has(slug)) { + samples.get(slug).push(file); + } else { + samples.set(slug, [file]); + } + } + build.files[`/samples/${filename}`] = file; } - const scratchblocksPath = pathUtil.join(__dirname, '../node_modules/scratchblocks/build/scratchblocks.min.js'); - build.files['/docs-internal/scratchblocks.js'] = new BuildFile(scratchblocksPath); + if (this.mode !== "desktop") { + for (const [filename, absolutePath] of recursiveReadDirectory( + this.websiteRoot + )) { + build.files[`/${filename}`] = new BuildFile(absolutePath); + } - /** @type {Record} */ - const extensionFiles = {}; - for (const [filename, absolutePath] of recursiveReadDirectory(this.extensionsRoot)) { - if (!filename.endsWith('.js')) { - continue; + for (const [filename, absolutePath] of recursiveReadDirectory( + this.docsRoot + )) { + if (!filename.endsWith(".md")) { + continue; + } + const extensionSlug = filename.split(".")[0]; + const file = new DocsFile(absolutePath, extensionSlug); + extensionsWithDocs.add(extensionSlug); + build.files[`/${extensionSlug}.html`] = file; } - const extensionId = filename.split('.')[0]; - const featured = featuredExtensionsIDs.includes(extensionId); - const file = new ExtensionFile(absolutePath, featured); - extensionFiles[extensionId] = file; - build.files[`/${filename}`] = file; + + const scratchblocksPath = pathUtil.join( + __dirname, + "../node_modules/scratchblocks/build/scratchblocks.min.js" + ); + build.files["/docs-internal/scratchblocks.js"] = new BuildFile( + scratchblocksPath + ); + + build.files["/index.html"] = new HomepageFile( + extensionFiles, + extensionImages, + extensionsWithDocs, + samples, + this.mode + ); + build.files["/sitemap.xml"] = new SitemapFile(build); } + build.files["/generated-metadata/extensions-v0.json"] = + new JSONMetadataFile( + extensionFiles, + extensionImages, + extensionsWithDocs, + samples + ); + for (const [oldPath, newPath] of Object.entries(compatibilityAliases)) { build.files[oldPath] = build.files[newPath]; } - for (const [filename, absolutePath] of recursiveReadDirectory(this.websiteRoot)) { - build.files[`/${filename}`] = new BuildFile(absolutePath); - } - - build.files['/index.html'] = new HomepageFile(extensionFiles, extensionImages, this.mode); - build.files['/generated-metadata/extensions-v0.json'] = new JSONMetadataFile(extensionFiles, extensionImages); - build.files['/sitemap.xml'] = new SitemapFile(build); - return build; } - tryBuild (...args) { + tryBuild(...args) { const start = new Date(); process.stdout.write(`[${start.toLocaleTimeString()}] Building... `); @@ -392,30 +562,36 @@ class Builder { console.log(`done in ${time}ms`); return build; } catch (error) { - console.log('error'); + console.log("error"); console.error(error); } return null; } - startWatcher (callback) { + startWatcher(callback) { // Load chokidar lazily. - const chokidar = require('chokidar'); + const chokidar = require("chokidar"); callback(this.tryBuild()); - chokidar.watch([ - `${this.extensionsRoot}/**/*`, - `${this.imagesRoot}/**/*`, - `${this.websiteRoot}/**/*`, - `${this.docsRoot}/**/*`, - ], { - ignoreInitial: true - }).on('all', () => { - callback(this.tryBuild()); - }); + chokidar + .watch( + [ + `${this.extensionsRoot}/**/*`, + `${this.imagesRoot}/**/*`, + `${this.websiteRoot}/**/*`, + `${this.docsRoot}/**/*`, + `${this.samplesRoot}/**/*`, + ], + { + ignoreInitial: true, + } + ) + .on("all", () => { + callback(this.tryBuild()); + }); } - validate () { + validate() { const errors = []; const build = this.build(); for (const [fileName, file] of Object.entries(build.files)) { @@ -424,7 +600,7 @@ class Builder { } catch (e) { errors.push({ fileName, - error: e + error: e, }); } } diff --git a/development/colors.js b/development/colors.js index 6dc21194d1..7e6fa1eaa8 100644 --- a/development/colors.js +++ b/development/colors.js @@ -1,9 +1,9 @@ const enableColor = !process.env.NO_COLOR; -const color = (i) => enableColor ? i : ''; +const color = (i) => (enableColor ? i : ""); module.exports = { - RESET: color('\x1b[0m'), - BOLD: color('\x1b[1m'), - RED: color('\x1b[31m'), - GREEN: color('\x1b[32m') + RESET: color("\x1b[0m"), + BOLD: color("\x1b[1m"), + RED: color("\x1b[31m"), + GREEN: color("\x1b[32m"), }; diff --git a/development/compatibility-aliases.js b/development/compatibility-aliases.js index a3302ea292..e1722787fa 100644 --- a/development/compatibility-aliases.js +++ b/development/compatibility-aliases.js @@ -1,13 +1,13 @@ const extensions = { // maps old path to new path - '/LukeManiaStudios/ClonesPlus.js': '/Lily/ClonesPlus.js', - '/LukeManiaStudios/CommentBlocks.js': '/Lily/CommentBlocks.js', - '/LukeManiaStudios/lmsutils.js': '/Lily/lmsutils.js', - '/LukeManiaStudios/LooksPlus.js': '/Lily/LooksPlus.js', - '/LukeManiaStudios/McUtils.js': '/Lily/McUtils.js', - '/LukeManiaStudios/MoreTimers.js': '/Lily/MoreTimers.js', - '/LukeManiaStudios/TempVariables.js': '/Lily/TempVariables.js', - '/LukeManiaStudios/TempVariables2.js': '/Lily/TempVariables2.js' + "/LukeManiaStudios/ClonesPlus.js": "/Lily/ClonesPlus.js", + "/LukeManiaStudios/CommentBlocks.js": "/Lily/CommentBlocks.js", + "/LukeManiaStudios/lmsutils.js": "/Lily/lmsutils.js", + "/LukeManiaStudios/LooksPlus.js": "/Lily/LooksPlus.js", + "/LukeManiaStudios/McUtils.js": "/Lily/McUtils.js", + "/LukeManiaStudios/MoreTimers.js": "/Lily/MoreTimers.js", + "/LukeManiaStudios/TempVariables.js": "/Lily/TempVariables.js", + "/LukeManiaStudios/TempVariables2.js": "/Lily/TempVariables2.js", }; module.exports = extensions; diff --git a/development/homepage-template.ejs b/development/homepage-template.ejs index ad693af7a2..95b3c5fbf3 100644 --- a/development/homepage-template.ejs +++ b/development/homepage-template.ejs @@ -107,6 +107,7 @@ } .extension { + position: relative; border: 2px solid #ccc; border-radius: 8px; margin: 4px; @@ -121,6 +122,9 @@ .extension h3 { font-size: 1.5em; } + .extension > :last-child { + margin-bottom: 0; + } .extension-banner { position: relative; margin: -8px -8px 0 -8px; @@ -137,10 +141,13 @@ top: 0; left: 0; display: flex; + flex-wrap: wrap; width: 100%; height: 100%; align-items: center; + align-content: center; justify-content: center; + gap: 0.5rem; opacity: 0; transition: opacity .15s; background: rgba(0, 0, 0, 0.5); @@ -148,14 +155,12 @@ } .extension:hover .extension-buttons, .extension:focus-within .extension-buttons { opacity: 1; - } - .extension:hover .extension-buttons { backdrop-filter: blur(0.6px); } .extension-buttons > * { - padding: 8px; + padding: 0.5rem; background-color: #4c97ff; - border-radius: 8px; + border-radius: 0.5rem; border: none; font: inherit; cursor: pointer; @@ -175,17 +180,52 @@ .extension-buttons *:disabled { opacity: 0.5; } - .extension-buttons .copy { - margin: 0 8px 0 0; - } .extension-buttons .open { background-color: #ff4c4c; color: white; } + .extension-buttons .docs { + background-color: #FFAB19; + color: white; + } + .extension-buttons .sample { + background-color: #40BF4A; + color: white; + } .extension-buttons :disabled { opacity: 0.5; } + .sample-list { + display: none; + position: absolute; + left: 0.5rem; + right: 0.5rem; + width: calc(100% - 1rem); + margin-top: -1.5rem; + padding: 0.5rem; + box-sizing: border-box; + background-color: white; + border-radius: 0.5rem; + box-shadow: 0px 0px 8px 1px rgba(0, 0, 0, .3); + border: 1px solid rgba(0, 0, 0, 0.15); + flex-direction: column; + gap: 0.5rem; + } + .sample-list h3 { + font-size: 1rem; + margin: 0; + } + .extension:hover[data-samples-open="true"] .sample-list { + display: flex; + } + @media (prefers-color-scheme: dark) { + .sample-list { + border: 1px solid rgba(255, 255, 255, 0.15); + background-color: #333; + } + } + footer { opacity: 0.8; width: 100%; @@ -230,6 +270,7 @@
+
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.
@@ -241,8 +282,8 @@

Development Server Tools

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

@@ -273,6 +314,11 @@ e.target.focus(); } } + + if (e.target.className.includes('sample-list-button')) { + var extension = e.target.closest('.extension'); + extension.dataset.samplesOpen = extension.dataset.samplesOpen !== 'true'; + } }); @@ -297,17 +343,36 @@ return result; }; %> - <% for (const [extensionID, metadata] of Object.entries(extensionMetadata)) { %> + <% for (const [extensionSlug, metadata] of Object.entries(extensionMetadata)) { %>
- <% const image = extensionImages[extensionID] || 'images/unknown.svg'; %> + <% const image = extensionImages[extensionSlug] || 'images/unknown.svg'; %>
- - Open Extension + + Open Extension + + <% if (metadata.hasDocumentation) { %> + Documentation + <% } %> + + <% if (metadata.samples.length === 1) { %> + Sample Project + <% } else if (metadata.samples.length > 1) { %> + + <% } %>
+ + <% if (metadata.samples.length > 1) { %> +
+

<%= metadata.name %> Sample Projects:

+ <% for (const sample of metadata.samples) { %> + <%= sample.getTitle() %> + <% } %> +
+ <% } %>

<%= metadata.name %>

diff --git a/development/parse-extension-metadata.js b/development/parse-extension-metadata.js index a640b2ef33..632f01216c 100644 --- a/development/parse-extension-metadata.js +++ b/development/parse-extension-metadata.js @@ -1,12 +1,12 @@ class Person { - constructor (name, link) { + constructor(name, link) { /** @type {string} */ this.name = name; /** @type {string|null} */ this.link = link; } - toHTML () { + toHTML() { // Don't need to bother escaping here. There's no vulnerability. if (this.link) { return `${this.name}`; @@ -16,10 +16,10 @@ class Person { } class Extension { - constructor () { - this.id = ''; - this.name = ''; - this.description = ''; + constructor() { + this.id = ""; + this.name = ""; + this.description = ""; /** @type {Person[]} */ this.by = []; /** @type {Person[]} */ @@ -45,13 +45,13 @@ const splitFirst = (string, split) => { * @returns {Person} */ const parsePerson = (person) => { - const parts = splitFirst(person, '<'); + const parts = splitFirst(person, "<"); if (parts.length === 1) { return new Person(person, null); } const name = parts[0].trim(); - const link = parts[1].replace('>', ''); + const link = parts[1].replace(">", ""); return new Person(name, link); }; @@ -62,14 +62,14 @@ const parsePerson = (person) => { const parseMetadata = (extensionCode) => { const metadata = new Extension(); - for (const line of extensionCode.split('\n')) { - if (!line.startsWith('//')) { + for (const line of extensionCode.split("\n")) { + if (!line.startsWith("//")) { // End of header. break; } const withoutComment = line.substring(2).trim(); - const parts = splitFirst(withoutComment, ':'); + const parts = splitFirst(withoutComment, ":"); if (parts.length === 1) { // Invalid. continue; @@ -79,16 +79,19 @@ const parseMetadata = (extensionCode) => { const value = parts[1].trim(); switch (key) { - case 'name': + case "id": + metadata.id = value; + break; + case "name": metadata.name = value; break; - case 'description': + case "description": metadata.description = value; break; - case 'by': + case "by": metadata.by.push(parsePerson(value)); break; - case 'original': + case "original": metadata.original.push(parsePerson(value)); break; default: diff --git a/development/render-docs.js b/development/render-docs.js index b882596354..3bdcfab88e 100644 --- a/development/render-docs.js +++ b/development/render-docs.js @@ -1,24 +1,28 @@ -const path = require('path'); -const MarkdownIt = require('markdown-it'); -const renderTemplate = require('./render-template'); +const path = require("path"); +const MarkdownIt = require("markdown-it"); +const renderTemplate = require("./render-template"); const md = new MarkdownIt({ html: true, linkify: true, - breaks: true + breaks: true, }); md.renderer.rules.fence = function (tokens, idx, options, env, self) { const token = tokens[idx]; - if (token.info === 'scratch') { + if (token.info === "scratch") { env.usesScratchBlocks = true; - return `

${md.utils.escapeHtml(token.content)}
`; + 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)}
`; + return `
${md.utils.escapeHtml(token.content)}
`; }; /** @@ -31,12 +35,19 @@ const renderDocs = (markdownSource, slug) => { const tokens = md.parse(markdownSource, env); // Extract the header - let headerHTML = '## file did not contain 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'); + 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); + const headerTokens = tokens.splice( + headerStart, + headerEnd - headerStart + 1 + ); // Discard the header tokens themselves, but render the HTML title with any formatting headerTokens.shift(); @@ -44,18 +55,20 @@ const renderDocs = (markdownSource, slug) => { 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'); + 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'), { + return renderTemplate(path.join(__dirname, "docs-template.ejs"), { slug, headerHTML, headerText, bodyHTML, - usesScratchBlocks: !!env.usesScratchBlocks + usesScratchBlocks: !!env.usesScratchBlocks, }); }; 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 e0bdfee855..517ccdc68a 100644 --- a/development/server.js +++ b/development/server.js @@ -1,50 +1,53 @@ -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; } - const fileInBuild = mostRecentBuild.getFile(req.path); + const fileInBuild = mostRecentBuild.getFile(decodeURIComponent(req.path)); if (!fileInBuild) { return next(); } @@ -54,9 +57,9 @@ app.get('/*', (req, res, next) => { }); 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/CubesterYT/WindowControls.md b/docs/CubesterYT/WindowControls.md new file mode 100644 index 0000000000..b0efe57279 --- /dev/null +++ b/docs/CubesterYT/WindowControls.md @@ -0,0 +1,463 @@ +# Window Controls + +This extension provides a set of blocks that gives you greater control over the Program Window. + +Note: Most of these blocks only work in Electron, Pop Ups/Web Apps containing HTML packaged projects, and normal Web Apps. + +Examples include, but are not limited to: TurboWarp Desktop App, TurboWarp Web App, Pop Up/Web App windows that contain the HTML packaged project, and plain Electron. + +Blocks that still work outside of these will be specified. + +
+ +

Move Window Block (#)

+ +```scratch +move window to x: (0) y: (0) :: #359ed4 +``` + +Moves the Program Window to the defined "x" and "y" coordinate on the screen. + +
+ +

Move Window to Preset Block (#)

+ +Moves the Program Window to a preset. + +The menu area has ten options, ("center", "right", "left", "top", "bottom", "top right", "top left", "bottom right", "bottom left", "random position") + +#### Center + +```scratch +move window to the (center v) :: #359ed4 +``` + +When choosing "center", it will move the Program Window to the center of the screen. + +#### Right + +```scratch +move window to the (right v) :: #359ed4 +``` + +When choosing "right", it will move the Program Window to the right of the screen. + +#### Left + +```scratch +move window to the (left v) :: #359ed4 +``` + +When choosing "left", it will move the Program Window to the left of the screen. + +#### Top + +```scratch +move window to the (top v) :: #359ed4 +``` + +When choosing "top", it will move the Program Window to the top of the screen. + +#### Bottom + +```scratch +move window to the (bottom v) :: #359ed4 +``` + +When choosing "bottom", it will move the Program Window to the bottom of the screen. + +#### Top Right + +```scratch +move window to the (top right v) :: #359ed4 +``` + +When choosing "top right", it will move the Program Window to the top right of the screen. + +#### Top Left + +```scratch +move window to the (top left v) :: #359ed4 +``` + +When choosing "top left", it will move the Program Window to the top left of the screen. + +#### Bottom Right + +```scratch +move window to the (bottom right v) :: #359ed4 +``` + +When choosing "bottom right", it will move the Program Window to the bottom right of the screen. + +#### Bottom Left + +```scratch +move window to the (bottom left v) :: #359ed4 +``` + +When choosing "bottom left", it will move the Program Window to the bottom left of the screen. + +#### Random Position + +```scratch +move window to the (random position v) :: #359ed4 +``` + +When choosing "random position", it will move the Program Window to a random position on the screen. + +
+ +

Change "x" Block (#)

+ +```scratch +change window x by (50) :: #359ed4 +``` + +Dynamically changes the "x" position of the Program Window on the screen. + +
+ +

Set "x" Block (#)

+ +```scratch +set window x to (100) :: #359ed4 +``` + +Statically changes the "x" position of the Program Window on the screen. + +
+ +

Change "y" Block (#)

+ +```scratch +change window y by (50) :: #359ed4 +``` + +Dynamically changes the "y" position of the Program Window on the screen. + +
+ +

Set "y" Block (#)

+ +```scratch +set window y to (100) :: #359ed4 +``` + +Statically changes the "y" position of the Program Window on the screen. + +
+ +

Window "x" Reporter (#)

+ +```scratch +(window x :: #359ed4) +``` + +This reporter returns the "x" position of the Program Window. + +This is supported outside of Electron, Pop Ups, and Web Apps. + +
+ +

Window "y" Reporter (#)

+ +```scratch +(window y :: #359ed4) +``` + +This reporter returns the "y" position of the Program Window. + +This is supported outside of Electron, Pop Ups, and Web Apps. + +
+ +

Resize Window Block (#)

+ +```scratch +resize window to width: (1000) height: (1000) :: #359ed4 +``` + +Resizes the Program Window to the defined width and height values. + +
+ +

Resize Window Preset Block (#)

+ +Resizes the Program Window to a preset. + +The menu area has eight options, ("480x360", "640x480", "1280x720", "1920x1080", "2560x1440", "2048x1080", "3840x2160", "7680x4320") + +#### 480x360 + +```scratch +resize window to (480x360 v) :: #359ed4 +``` + +When choosing "480x360", it will resize the Program Window to 480x360 (360p). The aspect ratio for this size is 4:3. + +#### 640x480 + +```scratch +resize window to (640x480 v) :: #359ed4 +``` + +When choosing "640x480", it will resize the Program Window to 640x480 (480p). The aspect ratio for this size is 4:3. + +#### 1280x720 + +```scratch +resize window to (1280x720 v) :: #359ed4 +``` + +When choosing "1280x720", it will resize the Program Window to 1280x720 (720p). The aspect ratio for this size is 16:9. + +#### 1920x1080 + +```scratch +resize window to (1920x1080 v) :: #359ed4 +``` + +When choosing "1920x1080", it will resize the Program Window to 1920x1080 (1080p). The aspect ratio for this size is 16:9. + +#### 2560x1440 + +```scratch +resize window to (2560x1440 v) :: #359ed4 +``` + +When choosing "2560x1440", it will resize the Program Window to 2560x1440 (1440p). The aspect ratio for this size is 16:9. + +#### 2048x1080 + +```scratch +resize window to (2048x1080 v) :: #359ed4 +``` + +When choosing "2048x1080", it will resize the Program Window to 2048x1080 (2K/1080p[Higher Pixel Rate]). The aspect ratio for this size is 1:1.77. + +#### 3840x2160 + +```scratch +resize window to (3840x2160 v) :: #359ed4 +``` + +When choosing "3840x2160", it will resize the Program Window to 3840x2160 (4K). The aspect ratio for this size is 1:1.9. + +#### 7680x4320 + +```scratch +resize window to (7680x4320 v) :: #359ed4 +``` + +When choosing "7680x4320", it will resize the Program Window to 7680x4320 (8K). The aspect ratio for this size is 16:9. + +
+ +

Change Width Block (#)

+ +```scratch +change window width by (50) :: #359ed4 +``` + +Dynamically changes the width of the Program Window. + +
+ +

Set Width Block (#)

+ +```scratch +set window width to (1000) :: #359ed4 +``` + +Statically changes the width of the Program Window. + +
+ +

Change Height Block (#)

+ +```scratch +change window height by (50) :: #359ed4 +``` + +Dynamically changes the height of the Program Window. + +
+ +

Set Height Block (#)

+ +```scratch +set window height to (1000) :: #359ed4 +``` + +Statically changes the height of the Program Window. + +
+ +

Match Stage Size Block (#)

+ +```scratch +match stage size :: #359ed4 +``` + +Resizes the Program Window to match the aspect ratio of the stage. Works best when the stage is dynamically changed. + +Example: When using runtime options to change the stage size, using this block can help you adapt to the new stage size. + +Try this example script in a packaged project: + +```scratch +when green flag clicked +wait (1) seconds +set stage size width: (360) height: (480) :: #8c9abf +match stage size :: #359ed4 +move window to the (center v) :: #359ed4 +wait (1) seconds +set stage size width: (480) height: (360) :: #8c9abf +match stage size :: #359ed4 +move window to the (center v) :: #359ed4 +``` + +
+ +

Window Width Reporter (#)

+ +```scratch +(window width :: #359ed4) +``` + +This reporter returns the width of the Program Window. + +This is supported outside of Electron, Pop Ups, and Web Apps. + +
+ +

Window Height Reporter (#)

+ +```scratch +(window height :: #359ed4) +``` + +This reporter returns the height of the Program Window. + +This is supported outside of Electron, Pop Ups, and Web Apps. + +
+ +

Is Window Touching Screen Edge Boolean (#)

+ +```scratch + +``` + +This boolean returns true or false for whether or not the Program Window is touching the screen's edge. + +This is supported outside of Electron, Pop Ups, and Web Apps. + +
+ +

Screen Width Reporter (#)

+ +```scratch +(screen width :: #359ed4) +``` + +This reporter returns the width of the Screen. + +This is supported outside of Electron, Pop Ups, and Web Apps. + +
+ +

Screen Height Reporter (#)

+ +```scratch +(screen height :: #359ed4) +``` + +This reporter returns the height of the Screen. + +This is supported outside of Electron, Pop Ups, and Web Apps. + +
+ +

Is Window Focused Boolean (#)

+ +```scratch + +``` + +This boolean returns true or false for whether or not the Program Window is in focus. + +This is supported outside of Electron, Pop Ups, and Web Apps. + +
+ +

Set Window Title Block (#)

+ +```scratch +set window title to ["Hello World!] :: #359ed4 +``` + +Changes the title of the Program Window. + +This is supported outside of Electron, Pop Ups, and Web Apps. + +
+ +

Window Title Reporter (#)

+ +```scratch +(window title :: #359ed4) +``` + +This reporter returns the title of the Program Window. + +This is supported outside of Electron, Pop Ups, and Web Apps. + +
+ +

Enter Fullscreen Block (#)

+ +```scratch +enter fullscreen :: #359ed4 +``` + +Makes the Program Window enter Fullscreen. + +This is supported outside of Electron, Pop Ups, and Web Apps. + +
+ +

Exit Fullscreen Block (#)

+ +```scratch +exit fullscreen :: #359ed4 +``` + +Makes the Program Window exit Fullscreen. + +This is supported outside of Electron, Pop Ups, and Web Apps. + +
+ +

Is Window Fullscreen Boolean (#)

+ +```scratch + +``` + +This boolean returns true or false for whether or not the Program Window is in fullscreen. + +This is supported outside of Electron, Pop Ups, and Web Apps. + +
+ +

Close Window Block (#)

+ +```scratch +close window :: cap :: #359ed4 +``` + +Closes the Program Window. + +This is supported outside of Electron, Pop Ups, and Web Apps. \ No newline at end of file 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/TheShovel/ShovelUtils.md b/docs/TheShovel/ShovelUtils.md new file mode 100644 index 0000000000..bf852c9523 --- /dev/null +++ b/docs/TheShovel/ShovelUtils.md @@ -0,0 +1,105 @@ +# ShovelUtils + +Shovel Utils is an extension focused mostly on injecting and modifying sprites and assets inside the project, as well as several other functions. + +**Disclaimer: Modifying and importing assets can be dangerous, and has the potential to corrupt your project. Be careful!** + +## Importing Assets + +Shovel Utils offers an easy way to import several types of assets, including sprites, costumes, sounds, extensions, and even full projects. + +--- + +**This goes for all blocks that fetch from a link: If you're experiences errors and are not able to import an asset from a link, check your console! You may be running into a CORS error. To resolve this, use a proxy like [corsproxy.io](https://corsproxy.io).** + +```scratch +Import sprite from [Link or data uri here] +``` + +Imports a sprite into the project using a DataURI or link. Be mindful of sprite names; having two or more sprites with the same name can cause issues. + +```scratch +Import image from [https://extensions.turbowarp.org/dango.png] name [Dango] +``` + +Imports a costume from a PNG, Bitmap, or JPEG. **Does not work with SVGS**. The costume imports into the current sprite/backdrop the user has selected. + +```scratch +Import sound from [https://extensions.turbowarp.org/meow.mp3] name [Meow] +``` + +Imports a sound from any Scratch-compatible sound file. The sound imports into the current sprite/backdrop the user has selected. + +```scratch +Import project from [https://theshovel.github.io/Bullet-Hell/Bullet%20Hell] +``` + +Imports a full project from a link. This project will completely replace the contents of the current one. If the project is unsandboxed, it will ask permission before swapping contents. + +```scratch +Load extension from [https://extensions.turbowarp.org/utilities.js] +``` + +Imports any extension from a link. Extensions from the [Extension Gallery](https://extensions.turbowarp.org) can run unsandboxed, and don't require permission to import. + +## Other Ways to Modify The Project + +Aside from importing assets, Shovel Utils provides multiple miscellaneous features to modify and straight up delete parts of your projects. + +```scratch +Set editing target to [Sprite1] +``` + +Sets the selected sprite in the editor. You can also set your input to "Stage" to set the selected target to the backdrop. This does work packaged, however will not have a visual effect. + +```scratch +(get all sprites ::) +``` + +Gets the names of all the sprites (and the stage) as a JSON array. This can then be parsed using the JSON Extension. + +```scratch +Restart project +``` + +Emulates a green flag click on a project, even if the green flag isn't present. + +```scratch +Delete costume [costume1] in [Sprite1] +``` + +Deletes a costume from the specified sprite. If the costume doesn't exist, the block simply doesn't do anything. + +```scratch +Delete sprite [Sprite1] +``` + +Deletes a the specified sprite. If the user has the "Sprite Deletion Confirmation" addon enabled and the project is unpackaged, it will ask permission before deleting sprites. + +## Miscellaneous Features + +Aside from project modification, there's several utility blocks present in Shovel Utils. + +```scratch +(fps::) +``` + +Get the accurate FPS, or frames per second, of the current project. This is *not* the same as the "framerate limit" block from Runtime Options, as the block in Shovel Utils accounts for lag. + +```scratch +(Get list [MyList]) +``` + +Get the values of a list, exported as a JSON array. If the specified list has not been created yet, or is empty, the block will return empty. + +```scratch +Set list [MyList] to [⟦1,2⟧] +``` + +Sets the values of lists. Accepts JSON arrays as inputs. If the specified list has not been created yet, the block simply doesn't do anything. + +```scratch +(Get brightness of [ #ffffff] ::) +``` + +Gets the brightness of a hex value. Reports a whole number between 0 and 255. To transfer this to a value between 0 and 100 (what TurboWarp uses), divide the output of the block by 2.55 and round. diff --git a/extensions/-SIPC-/consoles.js b/extensions/-SIPC-/consoles.js index 4591507ace..3454a518e1 100644 --- a/extensions/-SIPC-/consoles.js +++ b/extensions/-SIPC-/consoles.js @@ -1,189 +1,190 @@ // Name: Consoles +// ID: sipcconsole // Description: Blocks that interact the JavaScript console built in to your browser's developer tools. // By: -SIPC- (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); - } + "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 9343f52677..6a8413c821 100644 --- a/extensions/-SIPC-/time.js +++ b/extensions/-SIPC-/time.js @@ -1,122 +1,143 @@ // Name: Time +// ID: sipctime // Description: Blocks for interacting with unix timestamps and other date strings. // By: -SIPC- (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: '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' - } - } - } - ], - 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(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; - } + "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: "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", + }, + }, + }, + ], + menus: { + Time: { + acceptReporters: true, + items: ["year", "month", "day", "hour", "minute", "second"], + }, + }, + }; } - Scratch.extensions.register(new Time()); + 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; + } + } + 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 5f1a514945..871ef0b1ca 100644 --- a/extensions/0832/rxFS2.js +++ b/extensions/0832/rxFS2.js @@ -1,4 +1,5 @@ // Name: rxFS +// ID: 0832rxfs2 // Description: Blocks for interacting with a virtual in-memory filesystem. // By: 0832 @@ -6,290 +7,321 @@ * 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'; + "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] を検索する' - } - }); - - 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 1976fe7349..6e1face62c 100644 --- a/extensions/Alestore/nfcwarp.js +++ b/extensions/Alestore/nfcwarp.js @@ -1,72 +1,76 @@ // 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'; +(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 d573367947..a362fd17d7 100644 --- a/extensions/CST1229/zip.js +++ b/extensions/CST1229/zip.js @@ -1,43 +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 }); @@ -479,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; @@ -501,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() { @@ -528,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; } @@ -556,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 ""; } } @@ -591,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, { @@ -600,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 }) { @@ -652,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) { @@ -678,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) { @@ -700,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) { @@ -740,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); @@ -779,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); @@ -844,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 e7995a2f81..a87d1192a4 100644 --- a/extensions/CubesterYT/TurboHook.js +++ b/extensions/CubesterYT/TurboHook.js @@ -1,11 +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 { @@ -34,9 +36,9 @@ blockType: Scratch.BlockType.COMMAND, arguments: { hookURL: { - type: Scratch.ArgumentType.STRING - } - } + type: Scratch.ArgumentType.STRING, + }, + }, }, { opcode: "params", @@ -45,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 @@ -76,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 }); @@ -92,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/CubesterYT/WindowControls.js b/extensions/CubesterYT/WindowControls.js new file mode 100644 index 0000000000..67fe6832fd --- /dev/null +++ b/extensions/CubesterYT/WindowControls.js @@ -0,0 +1,510 @@ +// Name: Window Controls +// ID: cubesterWindowControls +// Description: Move, resize, rename the window, enter fullscreen, get screen size, and more. +// By: CubesterYT + +// Version V.1.0.0 + +(function (Scratch) { + "use strict"; + + const icon = + ""; + + function getRandomInt(min, max) { + min = Math.ceil(min); + max = Math.floor(max); + return Math.floor(Math.random() * (max - min + 1)) + min; + } + + class WindowControls { + getInfo() { + return { + id: "cubesterWindowControls", + name: "Window Controls", + color1: "#359ed4", + color2: "#298ec2", + color3: "#2081b3", + menuIconURI: icon, + docsURI: "https://extensions.turbowarp.org/CubesterYT/WindowControls", + + blocks: [ + { + blockType: "label", + text: "May not work in normal browser tabs", + }, + { + blockType: "label", + text: "Refer to Documentation for details", + }, + { + opcode: "moveTo", + blockType: Scratch.BlockType.COMMAND, + text: "move window to x: [X] y: [Y]", + arguments: { + X: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + Y: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + }, + }, + { + opcode: "moveToPresets", + blockType: Scratch.BlockType.COMMAND, + text: "move window to the [PRESETS]", + arguments: { + PRESETS: { + type: Scratch.ArgumentType.STRING, + menu: "MOVE", + }, + }, + }, + { + opcode: "changeX", + blockType: Scratch.BlockType.COMMAND, + text: "change window x by [X]", + arguments: { + X: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "50", + }, + }, + }, + { + opcode: "setX", + blockType: Scratch.BlockType.COMMAND, + text: "set window x to [X]", + arguments: { + X: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "100", + }, + }, + }, + { + opcode: "changeY", + blockType: Scratch.BlockType.COMMAND, + text: "change window y by [Y]", + arguments: { + Y: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "50", + }, + }, + }, + { + opcode: "setY", + blockType: Scratch.BlockType.COMMAND, + text: "set window y to [Y]", + arguments: { + Y: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "100", + }, + }, + }, + { + opcode: "windowX", + blockType: Scratch.BlockType.REPORTER, + text: "window x", + }, + { + opcode: "windowY", + blockType: Scratch.BlockType.REPORTER, + text: "window y", + }, + + "---", + + { + opcode: "resizeTo", + blockType: Scratch.BlockType.COMMAND, + text: "resize window to width: [W] height: [H]", + arguments: { + W: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "480", + }, + H: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "360", + }, + }, + }, + { + opcode: "resizeToPresets", + blockType: Scratch.BlockType.COMMAND, + text: "resize window to [PRESETS]", + arguments: { + PRESETS: { + type: Scratch.ArgumentType.STRING, + menu: "RESIZE", + }, + }, + }, + { + opcode: "changeW", + blockType: Scratch.BlockType.COMMAND, + text: "change window width by [W]", + arguments: { + W: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "50", + }, + }, + }, + { + opcode: "setW", + blockType: Scratch.BlockType.COMMAND, + text: "set window width to [W]", + arguments: { + W: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "1000", + }, + }, + }, + { + opcode: "changeH", + blockType: Scratch.BlockType.COMMAND, + text: "change window height by [H]", + arguments: { + H: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "50", + }, + }, + }, + { + opcode: "setH", + blockType: Scratch.BlockType.COMMAND, + text: "set window height to [H]", + arguments: { + H: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "1000", + }, + }, + }, + { + opcode: "matchStageSize", + blockType: Scratch.BlockType.COMMAND, + text: "match stage size", + }, + { + opcode: "windowW", + blockType: Scratch.BlockType.REPORTER, + text: "window width", + }, + { + opcode: "windowH", + blockType: Scratch.BlockType.REPORTER, + text: "window height", + }, + + "---", + + { + opcode: "isTouchingEdge", + blockType: Scratch.BlockType.BOOLEAN, + text: "is window touching screen edge?", + }, + { + opcode: "screenW", + blockType: Scratch.BlockType.REPORTER, + text: "screen width", + }, + { + opcode: "screenH", + blockType: Scratch.BlockType.REPORTER, + text: "screen height", + }, + + "---", + + { + opcode: "isFocused", + blockType: Scratch.BlockType.BOOLEAN, + text: "is window focused?", + }, + + "---", + + { + opcode: "changeTitleTo", + blockType: Scratch.BlockType.COMMAND, + text: "set window title to [TITLE]", + arguments: { + TITLE: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Hello World!", + }, + }, + }, + { + opcode: "windowTitle", + blockType: Scratch.BlockType.REPORTER, + text: "window title", + }, + + "---", + + { + opcode: "enterFullscreen", + blockType: Scratch.BlockType.COMMAND, + text: "enter fullscreen", + }, + { + opcode: "exitFullscreen", + blockType: Scratch.BlockType.COMMAND, + text: "exit fullscreen", + }, + { + opcode: "isFullscreen", + blockType: Scratch.BlockType.BOOLEAN, + text: "is window fullscreen?", + }, + + "---", + + { + opcode: "closeWindow", + blockType: Scratch.BlockType.COMMAND, + isTerminal: true, + text: "close window", + }, + ], + menus: { + MOVE: { + acceptReporters: true, + items: [ + "center", + "right", + "left", + "top", + "bottom", + "top right", + "top left", + "bottom right", + "bottom left", + "random position", + ], + }, + RESIZE: { + acceptReporters: true, + items: [ + "480x360", + "640x480", + "1280x720", + "1920x1080", + "2560x1440", + "2048x1080", + "3840x2160", + "7680x4320", + ], + }, + }, + }; + } + + moveTo(args) { + window.moveTo(args.X, args.Y); + Scratch.vm.runtime.requestRedraw(); + } + moveToPresets(args) { + if (args.PRESETS == "center") { + const left = (screen.width - window.outerWidth) / 2; + const top = (screen.height - window.outerHeight) / 2; + window.moveTo(left, top); + } else if (args.PRESETS == "right") { + const right = screen.width - window.outerWidth; + const top = (screen.height - window.outerHeight) / 2; + window.moveTo(right, top); + } else if (args.PRESETS == "left") { + const top = (screen.height - window.outerHeight) / 2; + window.moveTo(0, top); + } else if (args.PRESETS == "top") { + const left = (screen.width - window.outerWidth) / 2; + window.moveTo(left, 0); + } else if (args.PRESETS == "bottom") { + const left = (screen.width - window.outerWidth) / 2; + const bottom = screen.height - window.outerHeight; + window.moveTo(left, bottom); + } else if (args.PRESETS == "top right") { + const right = screen.width - window.outerWidth; + window.moveTo(right, 0); + } else if (args.PRESETS == "top left") { + window.moveTo(0, 0); + } else if (args.PRESETS == "bottom right") { + const right = screen.width - window.outerWidth; + const bottom = screen.height - window.outerHeight; + window.moveTo(right, bottom); + } else if (args.PRESETS == "bottom left") { + const bottom = screen.height - window.outerHeight; + window.moveTo(0, bottom); + } else if (args.PRESETS == "random position") { + const randomX = getRandomInt(0, screen.width); + const randomY = getRandomInt(0, screen.height); + window.moveTo(randomX, randomY); + } + Scratch.vm.runtime.requestRedraw(); + } + changeX(args) { + window.moveBy(args.X, 0); + Scratch.vm.runtime.requestRedraw(); + } + setX(args) { + const currentY = window.screenY; + window.moveTo(args.X, currentY); + Scratch.vm.runtime.requestRedraw(); + } + changeY(args) { + window.moveBy(0, args.Y); + Scratch.vm.runtime.requestRedraw(); + } + setY(args) { + const currentX = window.screenX; + window.moveTo(currentX, args.Y); + Scratch.vm.runtime.requestRedraw(); + } + windowX() { + return window.screenLeft; + } + windowY() { + return window.screenTop; + } + resizeTo(args) { + window.resizeTo(args.W, args.H); + Scratch.vm.runtime.requestRedraw(); + } + resizeToPresets(args) { + if (args.PRESETS == "480x360") { + window.resizeTo( + 480 + (window.outerWidth - window.innerWidth), + 360 + (window.outerHeight - window.innerHeight) + ); + } else if (args.PRESETS == "640x480") { + window.resizeTo( + 640 + (window.outerWidth - window.innerWidth), + 480 + (window.outerHeight - window.innerHeight) + ); + } else if (args.PRESETS == "1280x720") { + window.resizeTo( + 1280 + (window.outerWidth - window.innerWidth), + 720 + (window.outerHeight - window.innerHeight) + ); + } else if (args.PRESETS == "1920x1080") { + window.resizeTo( + 1920 + (window.outerWidth - window.innerWidth), + 1080 + (window.outerHeight - window.innerHeight) + ); + } else if (args.PRESETS == "2560x1440") { + window.resizeTo( + 2560 + (window.outerWidth - window.innerWidth), + 1440 + (window.outerHeight - window.innerHeight) + ); + } else if (args.PRESETS == "2048x1080") { + window.resizeTo( + 2048 + (window.outerWidth - window.innerWidth), + 1080 + (window.outerHeight - window.innerHeight) + ); + } else if (args.PRESETS == "3840x2160") { + window.resizeTo( + 3840 + (window.outerWidth - window.innerWidth), + 2160 + (window.outerHeight - window.innerHeight) + ); + } else if (args.PRESETS == "7680x4320") { + window.resizeTo( + 7680 + (window.outerWidth - window.innerWidth), + 4320 + (window.outerHeight - window.innerHeight) + ); + } + Scratch.vm.runtime.requestRedraw(); + } + changeW(args) { + window.resizeBy(args.W, 0); + Scratch.vm.runtime.requestRedraw(); + } + setW(args) { + const currentH = window.outerHeight; + window.resizeTo(args.W, currentH); + Scratch.vm.runtime.requestRedraw(); + } + changeH(args) { + window.resizeBy(0, args.H); + Scratch.vm.runtime.requestRedraw(); + } + setH(args) { + const currentW = window.outerWidth; + window.resizeTo(currentW, args.H); + Scratch.vm.runtime.requestRedraw(); + } + matchStageSize() { + window.resizeTo( + Scratch.vm.runtime.stageWidth + (window.outerWidth - window.innerWidth), + Scratch.vm.runtime.stageHeight + + (window.outerHeight - window.innerHeight) + ); + Scratch.vm.runtime.requestRedraw(); + } + windowW() { + return window.outerWidth; + } + windowH() { + return window.outerHeight; + } + isTouchingEdge() { + const edgeX = screen.width - window.outerWidth; + const edgeY = screen.height - window.outerHeight; + return ( + window.screenLeft <= 0 || + window.screenTop <= 0 || + window.screenLeft >= edgeX || + window.screenTop >= edgeY + ); + } + screenW() { + return screen.width; + } + screenH() { + return screen.height; + } + isFocused() { + return document.hasFocus(); + } + changeTitleTo(args) { + document.title = args.TITLE; + } + windowTitle() { + return document.title; + } + enterFullscreen() { + if (document.fullscreenElement == null) { + document.documentElement.requestFullscreen(); + } + } + exitFullscreen() { + if (document.fullscreenElement !== null) { + document.exitFullscreen(); + } + } + isFullscreen() { + return document.fullscreenElement !== null; + } + closeWindow() { + const editorConfirmation = [ + "Are you sure you want to close this window?", + "", + "(This message will not appear when the project is packaged)", + ].join("\n"); + if (typeof ScratchBlocks === "undefined" || confirm(editorConfirmation)) { + window.close(); + } + } + } + Scratch.extensions.register(new WindowControls()); +})(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 d8bbc94788..fb7944affc 100644 --- a/extensions/DT/cameracontrols.js +++ b/extensions/DT/cameracontrols.js @@ -1,17 +1,21 @@ -// Name: Camera Controls +// Name: Camera Controls (Very Buggy) +// ID: DTcameracontrols // Description: Move the visible part of the stage. // By: DT -(Scratch => { - 'use strict'; +((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; @@ -19,269 +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 (Very Buggy)", - 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; } @@ -351,14 +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(); @@ -386,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 0850f41f98..e19e1b5d94 100644 --- a/extensions/JeremyGamer13/tween.js +++ b/extensions/JeremyGamer13/tween.js @@ -1,308 +1,313 @@ -// Name: Tween -// 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); +// 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 index 31755ccf58..827b03aed3 100644 --- a/extensions/Lily/AllMenus.js +++ b/extensions/Lily/AllMenus.js @@ -1,56 +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'; + "use strict"; var blockXML; - const blacklist = ['looks_costumenumbername', 'extension_wedo_tilt_menu']; + const blacklist = ["looks_costumenumbername", "extension_wedo_tilt_menu"]; - Scratch.vm.addListener('BLOCKSINFO_UPDATE', refreshMenus); + Scratch.vm.addListener("BLOCKSINFO_UPDATE", refreshMenus); function refreshMenus() { if (!window.ScratchBlocks) return; - Scratch.vm.removeListener('BLOCKSINFO_UPDATE', refreshMenus); + Scratch.vm.removeListener("BLOCKSINFO_UPDATE", refreshMenus); let allBlocks = Object.keys(ScratchBlocks.Blocks); allBlocks = allBlocks.filter( - item => item.includes('menu') && - !blacklist.includes(item) + (item) => item.includes("menu") && !blacklist.includes(item) ); const menuBlocks = allBlocks.map( - item => '' + (item) => '' ); - blockXML = menuBlocks.join(''); + blockXML = menuBlocks.join(""); Scratch.vm.runtime.extensionManager.refreshBlocks(); } class AllMenus { - constructor () { - Scratch.vm.runtime.on('EXTENSION_ADDED', () => { + constructor() { + Scratch.vm.runtime.on("EXTENSION_ADDED", () => { refreshMenus(); }); } getInfo() { return { - id: 'lmsAllMenus', - name: 'All Menus', + id: "lmsAllMenus", + name: "All Menus", blocks: [ { blockType: Scratch.BlockType.XML, - xml: blockXML - } - ] + xml: blockXML, + }, + ], }; } } refreshMenus(); -Scratch.extensions.register(new AllMenus()); + Scratch.extensions.register(new AllMenus()); })(Scratch); diff --git a/extensions/Lily/Cast.js b/extensions/Lily/Cast.js index fda432c8b7..3baf507a99 100644 --- a/extensions/Lily/Cast.js +++ b/extensions/Lily/Cast.js @@ -1,77 +1,86 @@ -// Name: Cast -// 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); +// 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 655fa8e9cc..009e997692 100644 --- a/extensions/Lily/ClonesPlus.js +++ b/extensions/Lily/ClonesPlus.js @@ -1,615 +1,656 @@ -// Name: Clones Plus -// 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); +// 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 f94fbf668a..bba99b6571 100644 --- a/extensions/Lily/CommentBlocks.js +++ b/extensions/Lily/CommentBlocks.js @@ -1,88 +1,104 @@ // 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: "commentC", + blockType: Scratch.BlockType.CONDITIONAL, + text: "// [COMMENT]", + arguments: { + COMMENT: { + type: Scratch.ArgumentType.STRING, + defaultValue: "comment", + }, + }, + }, + { + 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) { + commentC(args, util) { + return true; + } + + 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 65d3a94e7d..49e17834f9 100644 --- a/extensions/Lily/LooksPlus.js +++ b/extensions/Lily/LooksPlus.js @@ -1,15 +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; @@ -31,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, @@ -201,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: { @@ -245,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", + }, + }, }; } @@ -358,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) { @@ -388,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 ""; } } @@ -420,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); }); }); @@ -431,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(); @@ -458,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(); @@ -510,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; @@ -522,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; @@ -535,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() { @@ -549,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, }); } } @@ -563,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 a51f7000b5..9e4d80758e 100644 --- a/extensions/Lily/McUtils.js +++ b/extensions/Lily/McUtils.js @@ -1,4 +1,5 @@ // Name: McUtils +// ID: lmsmcutils // Description: Helpful utilities for any fast food employee. // By: LilyMakesThings @@ -6,80 +7,80 @@ * 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", + }, + ], + }, + }, }; } @@ -92,7 +93,7 @@ } icecreammachine(args, util) { - if (args.INPUT === 'working') { + if (args.INPUT === "working") { return false; } else { return true; @@ -104,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 fad8572c82..c6417906b9 100644 --- a/extensions/Lily/MoreTimers.js +++ b/extensions/Lily/MoreTimers.js @@ -1,283 +1,287 @@ -// Name: More Timers -// 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); +// 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 index 7449ed2995..2c28f6cc94 100644 --- a/extensions/Lily/Skins.js +++ b/extensions/Lily/Skins.js @@ -1,444 +1,459 @@ -// Name: Skins -// 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', - 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); +// 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/SoundExpanded.js b/extensions/Lily/SoundExpanded.js new file mode 100644 index 0000000000..aefe95072b --- /dev/null +++ b/extensions/Lily/SoundExpanded.js @@ -0,0 +1,359 @@ +// Name: Sound Expanded +// Description: Adds more sound-related blocks. +// ID: lmsSoundExpanded +// By: LilyMakesThings + +(function (Scratch) { + "use strict"; + + const vm = Scratch.vm; + const runtime = vm.runtime; + const soundCategory = runtime.ext_scratch3_sound; + + class SoundExpanded { + getInfo() { + return { + id: "lmsSoundExpanded", + color1: "#CF63CF", + color2: "#C94FC9", + color3: "#BD42BD", + name: "Sound Expanded", + blocks: [ + { + opcode: "startLooping", + blockType: Scratch.BlockType.COMMAND, + text: "start looping [SOUND]", + arguments: { + SOUND: { + type: Scratch.ArgumentType.SOUND, + }, + START: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: 0, + }, + }, + }, + { + opcode: "stopLooping", + blockType: Scratch.BlockType.COMMAND, + text: "end looping [SOUND]", + arguments: { + SOUND: { + type: Scratch.ArgumentType.SOUND, + }, + }, + }, + { + opcode: "isLooping", + blockType: Scratch.BlockType.BOOLEAN, + text: "[SOUND] is looping?", + arguments: { + SOUND: { + type: Scratch.ArgumentType.SOUND, + }, + }, + }, + + "---", + + { + opcode: "stopSound", + blockType: Scratch.BlockType.COMMAND, + text: "stop sound [SOUND]", + arguments: { + SOUND: { + type: Scratch.ArgumentType.SOUND, + }, + }, + }, + { + opcode: "pauseSounds", + blockType: Scratch.BlockType.COMMAND, + text: "pause all sounds", + arguments: { + SOUND: { + type: Scratch.ArgumentType.SOUND, + }, + }, + }, + { + opcode: "resumeSounds", + blockType: Scratch.BlockType.COMMAND, + text: "resume all sounds", + arguments: { + SOUND: { + type: Scratch.ArgumentType.SOUND, + }, + }, + }, + + "---", + + { + opcode: "isSoundPlaying", + blockType: Scratch.BlockType.BOOLEAN, + text: "sound [SOUND] is playing?", + arguments: { + SOUND: { + type: Scratch.ArgumentType.SOUND, + }, + }, + }, + { + opcode: "attributeOfSound", + blockType: Scratch.BlockType.REPORTER, + text: "[ATTRIBUTE] of [SOUND]", + arguments: { + ATTRIBUTE: { + type: Scratch.ArgumentType.STRING, + menu: "attribute", + }, + SOUND: { + type: Scratch.ArgumentType.SOUND, + }, + }, + }, + { + opcode: "getSoundEffect", + blockType: Scratch.BlockType.REPORTER, + text: "[EFFECT] of [TARGET]", + arguments: { + EFFECT: { + type: Scratch.ArgumentType.STRING, + menu: "effect", + }, + TARGET: { + type: Scratch.ArgumentType.STRING, + menu: "targets", + }, + }, + }, + "---", + { + opcode: "setProjectVolume", + blockType: Scratch.BlockType.COMMAND, + text: "set project volume to [VALUE]%", + arguments: { + VALUE: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: 100, + }, + }, + }, + { + opcode: "changeProjectVolume", + blockType: Scratch.BlockType.COMMAND, + text: "change project volume by [VALUE]", + arguments: { + VALUE: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: -10, + }, + }, + }, + { + opcode: "getProjectVolume", + blockType: Scratch.BlockType.REPORTER, + text: "project volume", + }, + ], + menus: { + attribute: { + acceptReporters: false, + items: ["length", "channels", "sample rate", "dataURI"], + }, + effect: { + acceptReporters: false, + items: ["pitch", "pan"], + }, + targets: { + acceptReporters: true, + items: "_getTargets", + }, + }, + }; + } + + startLooping(args, util) { + const index = this._getSoundIndex(args.SOUND, util); + if (index < 0) return 0; + const target = util.target; + const sprite = util.target.sprite; + + const soundId = sprite.sounds[index].soundId; + const soundPlayer = sprite.soundBank.soundPlayers[soundId]; + + if (!soundPlayer.isPlaying) { + soundCategory._addWaitingSound(target.id, soundId); + sprite.soundBank.playSound(util.target, soundId); + } + + if (!soundPlayer.outputNode) return; + soundPlayer.outputNode.loop = true; + } + + stopLooping(args, util) { + const index = this._getSoundIndex(args.SOUND, util); + if (index < 0) return false; + const sprite = util.target.sprite; + + const soundId = sprite.sounds[index].soundId; + const soundPlayer = sprite.soundBank.soundPlayers[soundId]; + + if (!soundPlayer.outputNode) return; + soundPlayer.outputNode.loop = false; + } + + isLooping(args, util) { + const index = this._getSoundIndex(args.SOUND, util); + if (index < 0) return false; + const sprite = util.target.sprite; + + const soundId = sprite.sounds[index].soundId; + const soundPlayer = sprite.soundBank.soundPlayers[soundId]; + + if (!soundPlayer.outputNode) return false; + return soundPlayer.outputNode.loop; + } + + stopSound(args, util) { + const index = this._getSoundIndex(args.SOUND, util); + if (index < 0) return 0; + const target = util.target; + const sprite = target.sprite; + + const soundId = sprite.sounds[index].soundId; + const soundBank = sprite.soundBank; + soundBank.stop(target, soundId); + } + + pauseSounds(args, util) { + this._toggleSoundState(args, util, true); + } + + resumeSounds(args, util) { + this._toggleSoundState(args, util, false); + } + + _toggleSoundState(args, util, state) { + const sprite = util.target.sprite; + const audioContext = sprite.soundBank.audioEngine.audioContext; + + if (state) { + audioContext.suspend(); + return; + } else { + audioContext.resume(); + return; + } + } + + isSoundPlaying(args, util) { + const index = this._getSoundIndex(args.SOUND, util); + if (index < 0) return false; + const sprite = util.target.sprite; + + const soundId = sprite.sounds[index].soundId; + const soundPlayers = sprite.soundBank.soundPlayers; + return soundPlayers[soundId].isPlaying; + } + + attributeOfSound(args, util) { + const index = this._getSoundIndex(args.SOUND, util); + if (index < 0) return 0; + const sprite = util.target.sprite; + + const sound = sprite.sounds[index]; + const soundId = sound.soundId; + const soundPlayer = sprite.soundBank.soundPlayers[soundId]; + const soundBuffer = soundPlayer.buffer; + + switch (args.ATTRIBUTE) { + case "length": + return Math.round(soundBuffer.duration * 100) / 100; + case "channels": + return soundBuffer.numberOfChannels; + case "sample rate": + return soundBuffer.sampleRate; + case "dataURI": + return sound.asset.encodeDataURI(); + } + } + + getSoundEffect(args, util) { + let target = Scratch.vm.runtime.getSpriteTargetByName(args.TARGET); + if (args.TARGET === "_myself_") target = util.target; + if (args.TARGET === "_stage_") target = runtime.getTargetForStage(); + const effects = target.soundEffects; + if (!effects) return 0; + return effects[args.EFFECT]; + } + + setProjectVolume(args) { + const value = Scratch.Cast.toNumber(args.VALUE); + const newVolume = this._wrapClamp(value / 100, 0, 1); + runtime.audioEngine.inputNode.gain.value = newVolume; + } + + changeProjectVolume(args) { + const value = Scratch.Cast.toNumber(args.VALUE) / 100; + const volume = runtime.audioEngine.inputNode.gain.value; + const newVolume = Scratch.Cast.toNumber( + Math.min(Math.max(volume + value, 1), 0) + ); + runtime.audioEngine.inputNode.gain.value = newVolume; + } + + getProjectVolume() { + const volume = runtime.audioEngine.inputNode.gain.value; + return Math.round(volume * 10000) / 100; + } + + /* Utility Functions */ + + _getSoundIndex(soundName, util) { + const len = util.target.sprite.sounds.length; + if (len === 0) { + return -1; + } + const index = this._getSoundIndexByName(soundName, util); + if (index !== -1) { + return index; + } + const oneIndexedIndex = parseInt(soundName, 10); + if (!isNaN(oneIndexedIndex)) { + return this._wrapClamp(oneIndexedIndex - 1, 0, len - 1); + } + return -1; + } + + _getSoundIndexByName(soundName, util) { + const sounds = util.target.sprite.sounds; + for (let i = 0; i < sounds.length; i++) { + if (sounds[i].name === soundName) { + return i; + } + } + return -1; + } + + _wrapClamp(n, min, max) { + const range = max - min + 1; + return n - Math.floor((n - min) / range) * range; + } + + _getTargets() { + let spriteNames = [ + { text: "myself", value: "_myself_" }, + { text: "Stage", value: "_stage_" }, + ]; + const targets = Scratch.vm.runtime.targets + .filter((target) => target.isOriginal && !target.isStage) + .map((target) => target.getName()); + spriteNames = spriteNames.concat(targets); + return spriteNames; + } + } + + Scratch.extensions.register(new SoundExpanded()); +})(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 959fb6c89b..a7dfb04db3 100644 --- a/extensions/Lily/TempVariables2.js +++ b/extensions/Lily/TempVariables2.js @@ -1,306 +1,304 @@ -// Name: Temporary Variables -// 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); +// 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", + }, + }, + }, + + "---", + + { + 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; + return 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..9f3be6ba3f 100644 --- a/extensions/Lily/lmsutils.js +++ b/extensions/Lily/lmsutils.js @@ -1,1354 +1,1382 @@ -(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); +// Name: Lily's Toolbox +// ID: lmsutilsblocks +// Description: Previously called LMS Utilities. +// By: LilyMakesThings + +(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: "Lily's Toolbox", + 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 0a3bd925cc..0d0c6e2523 100644 --- a/extensions/Longboost/color_channels.js +++ b/extensions/Longboost/color_channels.js @@ -1,94 +1,97 @@ // Name: RGB Channels +// ID: lbdrawtest // Description: Only render or stamp certain RGB channels. -(function(Scratch) { - 'use strict'; +(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"], + }, + }, }; } @@ -100,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 929c86c41f..44bf8d5f3a 100644 --- a/extensions/NOname-awa/graphics2d.js +++ b/extensions/NOname-awa/graphics2d.js @@ -1,364 +1,462 @@ -// Name: Graphics 2D -// 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); +// 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])的距离", + ray_direction: "([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: "ray_direction", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate({ + id: "ray_direction", + default: "direction of ([x1],[y1]) to ([x2],[y2])", + }), + arguments: { + x1: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + y1: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + x2: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "100", + }, + 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) + ); + } + ray_direction(args) { + // Added by NexusKitten + // 由 NexusKitten 添加 + const dx = + Scratch.Cast.toNumber(args.x2) - Scratch.Cast.toNumber(args.x1); + const dy = + Scratch.Cast.toNumber(args.y2) - Scratch.Cast.toNumber(args.y1); + if (dx === 0 && dy === 0) { + return 0; + } else if (dy < 0) { + return (180 / Math.PI) * Math.atan(dx / dy) + 180; + } else { + return (180 / Math.PI) * Math.atan(dx / dy); + } + } + 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 342a3cfc33..19a08b5e6b 100644 --- a/extensions/NOname-awa/more-comparisons.js +++ b/extensions/NOname-awa/more-comparisons.js @@ -1,538 +1,568 @@ -// Name: More Comparisons -// 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); +// 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 index 0d6e1fe507..d3aee71d53 100644 --- a/extensions/NexusKitten/moremotion.js +++ b/extensions/NexusKitten/moremotion.js @@ -1,364 +1,392 @@ -// Name: More Motion -// Description: More motion-related blocks. -// By: NamelessCat - -(function(Scratch) { - 'use strict'; - - if (!Scratch.extensions.unsandboxed) { - throw new Error('More Motion must run unsandboxed'); - } - - 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(); - - // This is bad, need to rewrite this when renderer exports Rectangle - const Rectangle = Object.getPrototypeOf(drawableBounds).constructor; - - /** @type {RenderWebGL.Rectangle} */ - const containsBounds = new Rectangle(); - containsBounds.initFromBounds(left, right, bottom, top); - containsBounds.snapToInt(); - - if (!containsBounds.intersects(drawableBounds)) { - return false; - } - - drawable.updateCPURenderAttributes(); - - /** @type {RenderWebGL.Rectangle} */ - 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); +// 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 594401326c..6066bea0d2 100644 --- a/extensions/NexusKitten/sgrab.js +++ b/extensions/NexusKitten/sgrab.js @@ -1,253 +1,269 @@ -// Name: S-Grab -// 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); +// 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 30679902ab..acea113026 100644 --- a/extensions/Skyhigh173/bigint.js +++ b/extensions/Skyhigh173/bigint.js @@ -1,18 +1,19 @@ // Name: BigInt +// ID: skyhigh173BigInt // Description: Math blocks that work on infinitely large integers (no decimals). // By: Skyhigh173 -(function(Scratch){ - 'use strict'; +(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) { @@ -20,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) { @@ -39,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 }) { @@ -386,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(); } @@ -418,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 9ea47b5cd0..8fe25e8f4c 100644 --- a/extensions/Skyhigh173/json.js +++ b/extensions/Skyhigh173/json.js @@ -1,529 +1,557 @@ // Name: JSON +// ID: skyhigh173JSON // Description: Handle JSON strings and arrays. // By: Skyhigh173 -(function(Scratch) { - 'use strict'; +(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'), + "---", { - opcode: 'json_vm_getlist', + opcode: "json_array_sort", blockType: Scratch.BlockType.REPORTER, - text: 'get list [list] as array', + text: "sort array [list] in [order] order", + disableMonitor: true, arguments: { list: { type: Scratch.ArgumentType.STRING, - menu: 'get_list' + defaultValue: + "[5.23, 214, 522, 61, 5.24, 62.2, 1, 51212, 0, 0]", }, - } + order: { + type: Scratch.ArgumentType.STRING, + menu: "sort_order", + }, + }, }, + makeLabel("Lists"), { - opcode: 'json_vm_setlist', + opcode: "json_vm_getlist", + blockType: Scratch.BlockType.REPORTER, + text: "get list [list] as array", + arguments: { + list: { + type: Scratch.ArgumentType.STRING, + menu: "get_list", + }, + }, + }, + { + 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: ["=", "≠"], + }, + sort_order: { + items: ["ascending", "descending"], + acceptReporters: true, + }, + }, }; } - 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; } @@ -531,9 +559,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 { @@ -546,10 +577,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 { @@ -561,13 +595,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; @@ -579,21 +616,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; } @@ -616,30 +659,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 ""; } } @@ -648,7 +695,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; @@ -657,14 +704,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; } @@ -676,7 +723,7 @@ json[item] = value; return JSON.stringify(json); } catch { - return ''; + return ""; } } @@ -686,7 +733,7 @@ delete json[item]; return JSON.stringify(json); } catch { - return ''; + return ""; } } @@ -699,7 +746,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--; } @@ -710,13 +757,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 ""; } } @@ -727,7 +774,7 @@ let result = JSON.stringify(json.indexOf(item) + 1); return result; } catch { - return ''; + return ""; } } @@ -735,7 +782,7 @@ try { return JSON.stringify(Array.from(String(json))); } catch { - return ''; + return ""; } } @@ -745,7 +792,7 @@ json2 = JSON.parse(json2); return JSON.stringify(json.concat(json2)); } catch { - return ''; + return ""; } } @@ -756,7 +803,7 @@ json.push(item); return JSON.stringify(json); } catch { - return ''; + return ""; } } @@ -767,17 +814,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 ""; } } @@ -787,7 +836,7 @@ json.splice(item - 1, 1); return JSON.stringify(json); } catch { - return ''; + return ""; } } @@ -805,15 +854,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 ""; } } @@ -821,7 +870,7 @@ try { return JSON.stringify(JSON.parse(json).reverse()); } catch { - return ''; + return ""; } } @@ -829,7 +878,7 @@ try { return JSON.stringify(JSON.parse(json).flat(depth)); } catch { - return ''; + return ""; } } @@ -841,21 +890,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 ""; } } @@ -865,7 +916,7 @@ json.length = len; return JSON.stringify(json); } catch { - return ''; + return ""; } } @@ -878,7 +929,7 @@ } catch (e) { // ignore } - return ''; + return ""; } json_vm_setlist({ list, json }, util) { try { @@ -886,8 +937,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; @@ -896,7 +947,22 @@ } catch (e) { // ignore } - return ''; + return ""; + } + + json_array_sort(args) { + let list; + try { + list = JSON.parse(args.list); + } catch { + return ""; + } + if (!Array.isArray(list)) { + return ""; + } + list.sort(Scratch.Cast.compare); + if (args.order === "descending") list.reverse(); + return JSON.stringify(list); } } Scratch.extensions.register(new JSONS()); diff --git a/extensions/TheShovel/CanvasEffects.js b/extensions/TheShovel/CanvasEffects.js index 572483c3e5..e14c905bf5 100644 --- a/extensions/TheShovel/CanvasEffects.js +++ b/extensions/TheShovel/CanvasEffects.js @@ -1,235 +1,275 @@ // 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'); - } +(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 = () => { - // 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 - }); + 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 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'; + 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', '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); - } + 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 index 979f8c6d7a..fda63171b0 100644 --- a/extensions/TheShovel/CustomStyles.js +++ b/extensions/TheShovel/CustomStyles.js @@ -1,739 +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; }`; - } +(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; - }; + 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 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); + 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. + // 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 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; - } + // 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); - } - }); - } + // 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' - ] - } - } - }; - } + 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(); - }); - } + 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 + ')'; - } + 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(); - } + 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(); - } + 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(); - } + 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) - ); - } + 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) - ); - } + 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'."); - } + 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'; - } + transparentinput() { + return "transparent"; + } - pictureinput(args) { - return `url("${encodeURI(args.URL)}")`; - } + 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(); - } + 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 ''; - } + 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(); - } - }); + setAskURI(args) { + return Scratch.canFetch(args.URL).then((allowed) => { + if (allowed) { + askButtonImage = args.URL; + applyCSS(); } + }); } + } - Scratch.extensions.register(new MonitorStyles()); + Scratch.extensions.register(new MonitorStyles()); })(Scratch); diff --git a/extensions/TheShovel/LZ-String.js b/extensions/TheShovel/LZ-String.js index 73095e6709..635ed972ef 100644 --- a/extensions/TheShovel/LZ-String.js +++ b/extensions/TheShovel/LZ-String.js @@ -1,305 +1,435 @@ // Name: LZ Compress +// ID: shovellzcompress // Description: Compress and decompress text using lz-string. -(function(Scratch) { - 'use strict'; +(function (Scratch) { + "use strict"; - /* eslint-disable */ - // Code from https://github.com/pieroxy/lz-string/ - // MIT License + /* eslint-disable */ + // Code from https://github.com/pieroxy/lz-string/ + // MIT License - // Copyright (c) 2013 pieroxy + // 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: + // 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 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 = {}; + // 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] + 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 + "="; } - 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++) - } + }, + 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); } - }; - 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'] - } - } - }; + 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++); } - 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 ''; + 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 */ - 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 ''; + 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.extensions.register(new lzcompress()); })(Scratch); diff --git a/extensions/TheShovel/ShovelUtils.js b/extensions/TheShovel/ShovelUtils.js index f667a302c9..fff74dc116 100644 --- a/extensions/TheShovel/ShovelUtils.js +++ b/extensions/TheShovel/ShovelUtils.js @@ -1,11 +1,12 @@ // Name: ShovelUtils +// ID: ShovelUtils // Description: A bunch of miscellaneous blocks. // By: TheShovel (function (Scratch) { - 'use strict'; + "use strict"; if (!Scratch.extensions.unsandboxed) { - throw new Error('ShovelUtils must be run unsandboxed'); + throw new Error("ShovelUtils must be run unsandboxed"); } console.log("ShovelUtils v1.4"); const vm = Scratch.vm; @@ -27,174 +28,176 @@ class ShovelUtils { getInfo() { return { - id: 'ShovelUtils', - name: 'ShovelUtils', - color1: '#f54242', - color2: '#f54242', - color3: '#f54242', + id: "ShovelUtils", + name: "ShovelUtils", + color1: "#f54242", + color2: "#f54242", + color3: "#f54242", + docsURI: "https://extensions.turbowarp.org/TheShovel/ShovelUtils", 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', + opcode: "deleteImage", blockType: Scratch.BlockType.COMMAND, - text: 'Delete costume [COSNAME] in [SPRITE]', + text: "Delete costume [COSNAME] in [SPRITE]", arguments: { COSNAME: { type: Scratch.ArgumentType.STRING, - defaultValue: 'costume1' + defaultValue: "costume1", }, SPRITE: { type: Scratch.ArgumentType.STRING, - defaultValue: 'Sprite1' - } - } + 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: 'getAllSprites', + opcode: "getAllSprites", blockType: Scratch.BlockType.REPORTER, - text: 'get all sprites' + text: "get all sprites", }, { - opcode: 'getfps', + opcode: "getfps", blockType: Scratch.BlockType.REPORTER, - text: 'Fps' + text: "Fps", }, - ] + ], }; } @@ -203,23 +206,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"); }) @@ -234,8 +237,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; } } @@ -255,24 +262,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(); @@ -293,7 +304,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 { @@ -319,7 +332,9 @@ } } - const list = vm.runtime.getTargetForStage().lookupVariableByNameAndType(NAME, 'list'); + const list = vm.runtime + .getTargetForStage() + .lookupVariableByNameAndType(NAME, "list"); if (!list) { return; // List was not found } @@ -348,15 +363,15 @@ */ 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 }){ + deleteImage({ SPRITE, COSNAME }) { // 0znzw, since shovel did not add it yet. const target = vm.runtime.getSpriteTargetByName(SPRITE); if (!target) { @@ -365,7 +380,7 @@ target.deleteCostume(target.getCostumeIndexByName(COSNAME)); } - getAllSprites(){ + getAllSprites() { // 0znzw, since shovel did not add it yet. let sprites = []; for (const target of vm.runtime.targets) { @@ -375,5 +390,5 @@ } } 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 e307bbc77a..3b79908c01 100644 --- a/extensions/Xeltalliv/clippingblending.js +++ b/extensions/Xeltalliv/clippingblending.js @@ -1,17 +1,19 @@ // Name: Clipping & Blending +// ID: xeltallivclipblend // Description: Clipping outside of a specified rectangular area and additive color blending. // By: Vadik1 -(function(Scratch) { - 'use strict'; +(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(` @@ -22,7 +24,6 @@ `); - let toCorrectThing = null; let active = false; let flipY = false; @@ -38,7 +39,6 @@ let scratchUnitHeight = 360; let penDirty = false; - renderer._drawThese = function (drawables, drawMode, projection, opts) { active = true; [scratchUnitWidth, scratchUnitHeight] = renderer.getNativeSize(); @@ -74,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; @@ -132,7 +132,6 @@ this.blendMode = blendMode; }; - // Expanding renderer renderer.updateDrawableClipBox = function (drawableID, clipbox) { const drawable = this._allDrawables[drawableID]; @@ -145,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; @@ -163,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 = {}; @@ -188,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(); } @@ -205,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; @@ -217,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); }; @@ -230,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); }; @@ -248,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; @@ -259,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(); @@ -389,7 +412,7 @@ } } - clearClipbox (args, {target}) { + clearClipbox(args, { target }) { if (target.isStage) return; target.clipbox = null; penDirty = true; @@ -402,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: @@ -432,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(); @@ -441,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 index 3e01ec6473..0d6f3a7de0 100644 --- a/extensions/XeroName/Deltatime.js +++ b/extensions/XeroName/Deltatime.js @@ -1,14 +1,16 @@ // Name: Deltatime +// ID: dtbyxeroname // Description: Precise delta timing blocks. // By: XeroName (function (Scratch) { - 'use strict'; + "use strict"; - const icon = ''; + const icon = + ""; if (!Scratch.extensions.unsandboxed) { - throw new Error('DeltaTime must be run unsandboxed'); + throw new Error("DeltaTime must be run unsandboxed"); } const vm = Scratch.vm; @@ -16,7 +18,7 @@ let deltaTime = 0; let previousTime = 0; - vm.runtime.on('BEFORE_EXECUTE', () => { + vm.runtime.on("BEFORE_EXECUTE", () => { const now = performance.now(); deltaTime = previousTime === 0 ? 0 : (now - previousTime) / 1000; previousTime = now; @@ -25,24 +27,24 @@ class Dt { getInfo() { return { - id: 'dtbyxeroname', - name: 'Deltatime', - color1: '#333333', - color2: '#444444', - color3: '#ffffff', + id: "dtbyxeroname", + name: "Deltatime", + color1: "#333333", + color2: "#444444", + color3: "#ffffff", menuIconURI: icon, blocks: [ { - opcode: 'dt', + opcode: "dt", blockType: Scratch.BlockType.REPORTER, - text: 'ΔT' + text: "ΔT", }, { - opcode: 'fps', + opcode: "fps", blockType: Scratch.BlockType.REPORTER, - text: 'fps' - } - ] + text: "fps", + }, + ], }; } diff --git a/extensions/ZXMushroom63/searchApi.js b/extensions/ZXMushroom63/searchApi.js index cd4274710f..6d25fe08b5 100644 --- a/extensions/ZXMushroom63/searchApi.js +++ b/extensions/ZXMushroom63/searchApi.js @@ -1,214 +1,203 @@ -// Name: Search Params -// 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."); - } - - // 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 597df3757c..0d9ad60a71 100644 --- a/extensions/ar.js +++ b/extensions/ar.js @@ -1,809 +1,845 @@ // 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"); +(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", - 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 7e491ea656..fcb432b803 100644 --- a/extensions/battery.js +++ b/extensions/battery.js @@ -1,8 +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; @@ -21,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); }); }; @@ -57,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', + opcode: "chargingChanged", blockType: Scratch.BlockType.EVENT, - text: 'when charging changed', - isEdgeActivated: false + text: "when charging changed", + isEdgeActivated: false, }, { - opcode: 'levelChanged', + opcode: "levelChanged", blockType: Scratch.BlockType.EVENT, - text: 'when battery level changed', - isEdgeActivated: false + text: "when battery level changed", + isEdgeActivated: false, }, { - opcode: 'chargeTimeChanged', + opcode: "chargeTimeChanged", blockType: Scratch.BlockType.EVENT, - text: 'when time until charged changed', - isEdgeActivated: false + text: "when time until charged changed", + isEdgeActivated: false, }, { - opcode: 'dischargeTimeChanged', + opcode: "dischargeTimeChanged", blockType: Scratch.BlockType.EVENT, - text: 'when time until empty changed', - isEdgeActivated: false + 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 b9208ed3c0..d78d514037 100644 --- a/extensions/bitwise.js +++ b/extensions/bitwise.js @@ -1,20 +1,22 @@ // Name: Bitwise +// ID: Bitwise // Description: Blocks that operate on the binary representation of numbers in computers. // By: TrueFantom -(Scratch => { - 'use strict'; +((Scratch) => { + "use strict"; - const icon = ''; + const icon = + ""; - const isNumberBits = bits => { + 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) => { @@ -25,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", 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 d554534600..42df1ab034 100644 --- a/extensions/box2d.js +++ b/extensions/box2d.js @@ -1,4 +1,5 @@ // Name: Box2D Physics +// ID: griffpatch // Description: Two dimensional physics. // Original: griffpatch @@ -8,11 +9,11 @@ /* 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. @@ -20,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) { @@ -82,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 = {}; @@ -173,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; @@ -519,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; @@ -546,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; @@ -606,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; @@ -961,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; @@ -1007,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) { @@ -1053,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; @@ -1160,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; @@ -1207,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; @@ -1493,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) { @@ -1752,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; @@ -2279,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(); @@ -2306,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(); @@ -2322,7 +2370,10 @@ } } }; - b2SeparationFunction.prototype.Evaluate = function (transformA, transformB) { + b2SeparationFunction.prototype.Evaluate = function ( + transformA, + transformB + ) { var axisA; var axisB; var localPointA; @@ -3137,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 * @@ -3399,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( @@ -3417,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; @@ -3525,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; @@ -3650,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) { @@ -3707,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); } @@ -3929,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); } @@ -4408,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; }; @@ -4784,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, @@ -4795,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, @@ -4988,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; @@ -5130,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 * @@ -5317,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) { @@ -5351,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; @@ -5565,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; @@ -5733,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 ( @@ -5983,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) @@ -6067,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; @@ -6149,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( @@ -6874,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; @@ -7117,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, @@ -7128,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, @@ -7241,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) { @@ -7262,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; @@ -7554,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]; @@ -7633,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; @@ -7708,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; @@ -7992,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, @@ -8055,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); }; @@ -8161,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(); @@ -8328,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; @@ -8422,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) { @@ -8445,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) { @@ -8505,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; }; @@ -8568,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; }; @@ -9220,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) { @@ -9286,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); }; @@ -9379,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: @@ -9677,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; @@ -9920,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; @@ -9961,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(); @@ -10150,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(); @@ -10351,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; @@ -10599,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; @@ -11374,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; @@ -11733,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(); @@ -11848,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; @@ -12007,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; @@ -12212,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; @@ -12295,8 +12446,10 @@ } }; - const blockIconURI = ""; - const menuIconURI = ""; + const blockIconURI = + ""; + const menuIconURI = + ""; const vm = Scratch.vm; class Scratch3Griffpatch { @@ -12613,7 +12766,7 @@ filter: [Scratch.TargetType.SPRITE], }, - '---', + "---", { opcode: "setAngVelocity", @@ -12641,7 +12794,7 @@ }), filter: [Scratch.TargetType.SPRITE], }, - + "---", { @@ -12666,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], }, "---", @@ -12680,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: { @@ -12698,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], }, @@ -12820,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: { @@ -12942,7 +13096,7 @@ description: "get the y scroll", }), blockType: BlockType.REPORTER, - } + }, ], menus: { @@ -13108,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( @@ -13354,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) { @@ -13389,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(); } @@ -13410,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(); } @@ -13699,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 7169483623..07f1752857 100644 --- a/extensions/clipboard.js +++ b/extensions/clipboard.js @@ -1,121 +1,123 @@ -// Name: 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); +// 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 9defe28429..93a130de1e 100644 --- a/extensions/clouddata-ping.js +++ b/extensions/clouddata-ping.js @@ -1,4 +1,5 @@ // Name: Ping Cloud Data +// ID: clouddataping // Description: Determine whether a cloud variable server is probably up. // Original: TheShovel @@ -21,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, }; } @@ -37,7 +38,7 @@ } catch (e) { return { expires: 0, - value: false + value: false, }; } @@ -64,7 +65,7 @@ return { expires: Date.now() + 60000, - value: isUp + value: isUp, }; }; diff --git a/extensions/cloudlink.js b/extensions/cloudlink.js index 5db7711f5e..1583cc1e23 100644 --- a/extensions/cloudlink.js +++ b/extensions/cloudlink.js @@ -1,1717 +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, - "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)); - +(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 c3204b2bb8..5212283722 100644 --- a/extensions/cs2627883/numericalencoding.js +++ b/extensions/cs2627883/numericalencoding.js @@ -1,52 +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) { @@ -56,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; @@ -69,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 9c011998b3..3fdf7c47ca 100644 --- a/extensions/cursor.js +++ b/extensions/cursor.js @@ -1,289 +1,323 @@ -// Name: Mouse Cursor -// 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); +// 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 0b9df60f93..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: { diff --git a/extensions/encoding.js b/extensions/encoding.js index d2ad7bc1a1..f506508052 100644 --- a/extensions/encoding.js +++ b/extensions/encoding.js @@ -1,622 +1,634 @@ -// Name: 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); +// 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 index 7ec0ffa91d..12abcf6abc 100644 --- a/extensions/extensions.json +++ b/extensions/extensions.json @@ -22,14 +22,18 @@ "obviousAlexC/SensingPlus", "Lily/ClonesPlus", "Lily/LooksPlus", + "Lily/MoreEvents", "NexusKitten/moremotion", + "CubesterYT/WindowControls", "navigator", "battery", "TheShovel/CustomStyles", + "NexusKitten/controlcontrols", "mdwalters/notifications", "XeroName/Deltatime", "ar", "encoding", + "Lily/SoundExpanded", "Lily/TempVariables2", "Lily/MoreTimers", "clouddata-ping", @@ -44,6 +48,7 @@ "-SIPC-/consoles", "ZXMushroom63/searchApi", "TheShovel/ShovelUtils", + "DNin/wake-lock", "Skyhigh173/json", "cs2627883/numericalencoding", "DT/cameracontrols", @@ -57,6 +62,7 @@ "NOname-awa/more-comparisons", "JeremyGamer13/tween", "rixxyx", + "Lily/lmsutils", "qxsck/data-analysis", "qxsck/var-and-list", "vercte/dictionaries", @@ -69,4 +75,4 @@ "gamejolt", "obviousAlexC/newgroundsIO", "Lily/McUtils" -] \ No newline at end of file +] diff --git a/extensions/fetch.js b/extensions/fetch.js index 04646a942e..10ff8e13e4 100644 --- a/extensions/fetch.js +++ b/extensions/fetch.js @@ -1,34 +1,35 @@ // Name: Fetch +// ID: fetch // Description: Make requests to the broader internet. -(function(Scratch) { - 'use strict'; +(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 e381f6637e..c4212b34c8 100644 --- a/extensions/files.js +++ b/extensions/files.js @@ -1,178 +1,187 @@ // Name: Files +// ID: files // Description: Read and download files. -(function(Scratch) { - 'use strict'; +(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