diff --git a/development/builder.js b/development/builder.js
index 9a2f025994..3d755fd631 100644
--- a/development/builder.js
+++ b/development/builder.js
@@ -105,6 +105,39 @@ class SVGFile extends ImageFile {
}
}
+class SitemapFile extends DiskFile {
+ constructor (build) {
+ super(null);
+ this.getDiskPath = null;
+ this.build = build;
+ }
+
+ getType () {
+ return '.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', ''))
+ .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');
+
+ xml += '\n';
+ return xml;
+ }
+}
+
const IMAGE_FORMATS = new Map();
IMAGE_FORMATS.set('.png', ImageFile);
IMAGE_FORMATS.set('.jpg', ImageFile);
@@ -116,7 +149,7 @@ class Build {
}
getFile (path) {
- return this.files[path] || this.files[`${path}index.html`] || null;
+ return this.files[path] || this.files[`${path}.html`] || this.files[`${path}index.html`] || null;
}
export (root) {
@@ -192,6 +225,10 @@ class Builder {
build.files[oldPath] = build.files[newPath];
}
+ if (this.mode !== 'desktop') {
+ build.files['/sitemap.xml'] = new SitemapFile(build);
+ }
+
const mostRecentExtensions = extensionFiles
.sort((a, b) => b.getLastModified() - a.getLastModified())
.slice(0, 5)
diff --git a/extensions/Alestore/nfcwarp.js b/extensions/Alestore/nfcwarp.js
new file mode 100644
index 0000000000..c731f04095
--- /dev/null
+++ b/extensions/Alestore/nfcwarp.js
@@ -0,0 +1,73 @@
+(function(Scratch) {
+ 'use strict';
+
+ /* globals NDEFReader */
+
+ const extIcon = '';
+ const blocksIcon = '';
+
+ class NFCWarp {
+ getInfo() {
+ return {
+ 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'
+ },
+ {
+ opcode: 'supported',
+ blockType: Scratch.BlockType.BOOLEAN,
+ text: 'NFC supported?'
+ },
+ {
+ opcode: 'nfcRead',
+ blockType: Scratch.BlockType.REPORTER,
+ text: 'read NFC tag',
+ disableMonitor: true
+ }
+ ]
+ };
+ }
+
+ supported () {
+ return typeof NDEFReader !== 'undefined';
+ }
+
+ nfcRead() {
+ if (!this.supported()) {
+ return 'NFC not supported';
+ }
+ return new Promise((resolve, reject) => {
+ const ndef = new NDEFReader();
+ ndef.scan()
+ .then(() => {
+ ndef.onreadingerror = event => {
+ console.log('Reading error', event);
+ resolve('Tag not supported');
+ };
+ 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));
+ resolve(decoder.decode(record.data));
+ };
+ })
+ .catch(error => {
+ console.log('Scan error', error);
+ resolve(`Error: ${error}`);
+ });
+ });
+ }
+ }
+
+ Scratch.extensions.register(new NFCWarp());
+})(Scratch);
diff --git a/extensions/Lily/Cast.js b/extensions/Lily/Cast.js
new file mode 100644
index 0000000000..5035acae5a
--- /dev/null
+++ b/extensions/Lily/Cast.js
@@ -0,0 +1,73 @@
+(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/TempVariables2.js b/extensions/Lily/TempVariables2.js
index ad131fbc18..13912281f8 100644
--- a/extensions/Lily/TempVariables2.js
+++ b/extensions/Lily/TempVariables2.js
@@ -14,7 +14,6 @@
});
function resetRuntimeVariables() {
- console.log('runtime variables cleared');
runtimeVariables = Object.create(null);
}
diff --git a/extensions/clipboard.js b/extensions/clipboard.js
new file mode 100644
index 0000000000..7064a0194f
--- /dev/null
+++ b/extensions/clipboard.js
@@ -0,0 +1,118 @@
+/*!
+ * Copyright 2023 tomyo-code + AdamMady
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+(function(Scratch) {
+ 'use strict';
+
+ if (!Scratch.extensions.unsandboxed) {
+ throw new Error('Clipboard must run unsandboxed');
+ }
+
+ const extensionicon = "";
+
+ let lastPastedText = '';
+
+ window.addEventListener('copy', (event) => {
+ Scratch.vm.runtime.startHats('clipboard_whenCopied') ;
+ });
+ window.addEventListener('paste', (event) => {
+ Scratch.vm.runtime.startHats('clipboard_whenPasted');
+ const clipboardData = event.clipboardData || window.clipboardData;
+ const pastedText = clipboardData.getData('Text');
+ lastPastedText = pastedText;
+ });
+
+ class Clipboard {
+ getInfo() {
+ return {
+ id: 'clipboard',
+ name: 'Clipboard',
+ blockIconURI: extensionicon,
+ color1: '#008080',
+ color2: '#006666',
+ blocks: [
+ {
+ opcode: 'whenCopied',
+ blockType: Scratch.BlockType.HAT,
+ text: 'when something is copied',
+ isEdgeActivated: false
+ },
+ {
+ opcode: 'whenPasted',
+ blockType: Scratch.BlockType.HAT,
+ text: 'when something is pasted',
+ isEdgeActivated: false
+ },
+ '---',
+ {
+ opcode: 'setClipboard',
+ blockType: Scratch.BlockType.COMMAND,
+ text: 'copy to clipboard: [TEXT]',
+ arguments: {
+ TEXT: {
+ type: Scratch.ArgumentType.STRING
+ }
+ }
+ },
+ {
+ opcode: 'resetClipboard',
+ blockType: Scratch.BlockType.COMMAND,
+ text: 'reset clipboard'
+ },
+ '---',
+ {
+ opcode: 'clipboard',
+ blockType: Scratch.BlockType.REPORTER,
+ text: 'clipboard',
+ disableMonitor: true
+ },
+ {
+ opcode: 'getLastPastedText',
+ blockType: Scratch.BlockType.REPORTER,
+ text: 'last pasted text',
+ disableMonitor: true
+ }
+ ],
+ };
+ }
+
+ setClipboard(args) {
+ navigator.clipboard.writeText(args.TEXT);
+ }
+
+ resetClipboard() {
+ navigator.clipboard.writeText('');
+ }
+
+ clipboard() {
+ if (navigator.clipboard && navigator.clipboard.readText) {
+ return Scratch.canReadClipboard().then(allowed => {
+ if (allowed) {
+ return navigator.clipboard.readText();
+ }
+ return '';
+ });
+ }
+ return '';
+ }
+
+ getLastPastedText() {
+ return lastPastedText;
+ }
+ }
+
+ Scratch.extensions.register(new Clipboard());
+})(Scratch);
diff --git a/extensions/files.js b/extensions/files.js
index 25842fd6b2..2d1cea7860 100644
--- a/extensions/files.js
+++ b/extensions/files.js
@@ -165,21 +165,58 @@
});
/**
- * @param {string} text Text to download
- * @param {string} file Name of the file
+ * @param {string} url a data:, blob:, or same-origin URL
+ * @param {string} file
*/
- const download = (text, file) => {
- const blob = new Blob([text]);
- const url = URL.createObjectURL(blob);
+ const downloadURL = (url, file) => {
const link = document.createElement('a');
link.href = url;
link.download = file;
document.body.appendChild(link);
link.click();
link.remove();
+ };
+
+ /**
+ * @param {Blob} blob Data to download
+ * @param {string} file Name of the file
+ */
+ const downloadBlob = (blob, file) => {
+ const url = URL.createObjectURL(blob);
+ downloadURL(url, file);
URL.revokeObjectURL(url);
};
+ /**
+ * @param {string} url
+ * @returns {boolean}
+ */
+ const isDataURL = (url) => {
+ try {
+ const parsed = new URL(url);
+ return parsed.protocol === 'data:';
+ } catch (e) {
+ return false;
+ }
+ };
+
+ /**
+ * @param {string} url
+ * @param {string} file
+ */
+ const downloadUntrustedURL = (url, file) => {
+ // Don't want to return a Promise here when not actually needed
+ if (isDataURL(url)) {
+ downloadURL(url, file);
+ } else {
+ return Scratch.fetch(url)
+ .then(res => res.blob())
+ .then(blob => {
+ downloadBlob(blob, file);
+ });
+ }
+ };
+
class Files {
getInfo () {
return {
@@ -253,6 +290,24 @@
}
}
},
+ {
+ opcode: 'downloadURL',
+ blockType: Scratch.BlockType.COMMAND,
+ text: 'download URL [url] as [file]',
+ arguments: {
+ url: {
+ type: Scratch.ArgumentType.STRING,
+ defaultValue: 'data:text/plain;base64,SGVsbG8sIHdvcmxkIQ=='
+ },
+ file: {
+ type: Scratch.ArgumentType.STRING,
+ defaultValue: 'save.txt'
+ }
+ }
+ },
+
+ '---',
+
{
opcode: 'setOpenMode',
blockType: Scratch.BlockType.COMMAND,
@@ -314,7 +369,11 @@
}
download (args) {
- download(args.text, args.file);
+ downloadBlob(new Blob([Scratch.Cast.toString(args.text)]), Scratch.Cast.toString(args.file));
+ }
+
+ downloadURL (args) {
+ return downloadUntrustedURL(Scratch.Cast.toString(args.url), Scratch.Cast.toString(args.file));
}
setOpenMode (args) {
diff --git a/images/Lily/Cast.svg b/images/Lily/Cast.svg
new file mode 100644
index 0000000000..c133d27df6
--- /dev/null
+++ b/images/Lily/Cast.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/images/README.md b/images/README.md
index 9b629d4eb8..76bd6f0da2 100644
--- a/images/README.md
+++ b/images/README.md
@@ -233,3 +233,6 @@ All images in this folder are licensed under the [GNU General Public License ver
## Lily/LooksPlus.svg
- Created by [@LilyMakesThings](https://github.com/LilyMakesThings) in https://github.com/TurboWarp/extensions/pull/656
+
+## clipboard.svg
+ - Created by [@AdamMady](https://github.com/AdamMady/)
diff --git a/images/clipboard.svg b/images/clipboard.svg
new file mode 100644
index 0000000000..f02be5a0ee
--- /dev/null
+++ b/images/clipboard.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/images/cursor.png b/images/cursor.png
new file mode 100644
index 0000000000..b53b880cd6
Binary files /dev/null and b/images/cursor.png differ
diff --git a/images/cursor.svg b/images/cursor.svg
deleted file mode 100644
index 5a0ab87dbb..0000000000
--- a/images/cursor.svg
+++ /dev/null
@@ -1 +0,0 @@
-
\ No newline at end of file
diff --git a/images/gamejolt.png b/images/gamejolt.png
new file mode 100644
index 0000000000..d72c154054
Binary files /dev/null and b/images/gamejolt.png differ
diff --git a/images/gamejolt.svg b/images/gamejolt.svg
deleted file mode 100644
index 103ef7b390..0000000000
--- a/images/gamejolt.svg
+++ /dev/null
@@ -1 +0,0 @@
-
\ No newline at end of file
diff --git a/images/pointerlock.png b/images/pointerlock.png
new file mode 100644
index 0000000000..e653ce69e4
Binary files /dev/null and b/images/pointerlock.png differ
diff --git a/images/pointerlock.svg b/images/pointerlock.svg
deleted file mode 100644
index 871de74ca4..0000000000
--- a/images/pointerlock.svg
+++ /dev/null
@@ -1 +0,0 @@
-
\ No newline at end of file
diff --git a/website/index.ejs b/website/index.ejs
index a6a750270a..9f95fc9a41 100644
--- a/website/index.ejs
+++ b/website/index.ejs
@@ -476,6 +476,12 @@
Clipping outside of a specified rectangular area and additive color blending. Created by Vadik1.
+
+ <%- banner('clipboard') %>
+
Clipboard
+
Read and write from the system clipboard.
+
+
<%- banner('penplus') %>
Pen Plus
@@ -572,6 +578,12 @@
A few adapter blocks. Created by TrueFantom.
+
+ <%- banner('Lily/Cast') %>
+
Cast
+
Convert values between types. Created by LilyMakesThings
+
+
<%- banner('-SIPC-/time') %>
Time
@@ -710,6 +722,12 @@
Allows you to use webhooks. Created by CubesterYT.
+
+ <%- banner('Alestore/nfcwarp') %>
+
NFCWarp
+
Allows reading data from NFC (NDEF) devices. Only works in Chrome on Android. Created by Alestore Games.
+
+
<%- banner('itchio') %>
itch.io