diff --git a/extensions/SharkPool/Font-Manager.js b/extensions/SharkPool/Font-Manager.js new file mode 100644 index 0000000000..ee3627d1cd --- /dev/null +++ b/extensions/SharkPool/Font-Manager.js @@ -0,0 +1,363 @@ +// Name: Font Manager +// ID: SPASfontManager +// Description: Add, delete, and manage fonts. +// By: SharkPool +// By: Ashimee +// License: MIT + +// Version V.1.1.0 + +(function (Scratch) { + "use strict"; + + if (!Scratch.extensions.unsandboxed) { + throw new Error("Font Manager must be run unsandboxed"); + } + + const extensionId = "SPASfontManager"; + const vm = Scratch.vm; + const runtime = vm.runtime; + const storage = runtime.storage; + const fontManager = runtime.fontManager; + + const FONT_EXTENSIONS = [ + storage.DataFormat.TTF, + storage.DataFormat.OTF, + storage.DataFormat.WOFF, + storage.DataFormat.WOFF2, + ]; + + const menuIconURI = + ""; + + class SPASfontManager { + constructor() { + /** @type {string[]} */ + this.oldFonts = []; + + /** @type {string[]} */ + this.addedFonts = []; + + /** @type {string[]} */ + this.removedFonts = []; + + fontManager.on("change", () => { + this._onchange(); + }); + } + + getInfo() { + return { + id: extensionId, + name: Scratch.translate("Font Manager"), + color1: "#2b7d6e", + color2: "#24675b", + color3: "#1c4e45", + menuIconURI, + blocks: [ + { + opcode: "fontNames", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("added font names"), + disableMonitor: true, + }, + { + opcode: "fontAdded", + blockType: Scratch.BlockType.BOOLEAN, + text: Scratch.translate("font [NAME] added?"), + arguments: { + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Comic Sans MS", + }, + }, + }, + { + opcode: "fontDetail", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("[DATA] of font [NAME]"), + arguments: { + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Comic Sans MS", + }, + DATA: { + type: Scratch.ArgumentType.STRING, + menu: "DATA", + }, + }, + }, + "---", + { + opcode: "addSystemFont", + blockType: Scratch.BlockType.COMMAND, + text: Scratch.translate( + "add system font named [NAME] with fallback [BACKUP]" + ), + arguments: { + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Comic Sans MS", + }, + BACKUP: { + type: Scratch.ArgumentType.STRING, + menu: "FALLBACKS", + }, + }, + }, + { + opcode: "addCustomFont", + blockType: Scratch.BlockType.COMMAND, + text: Scratch.translate( + "add custom font named [NAME] with fallback [BACKUP] from URL [URL]" + ), + arguments: { + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Pusab", + }, + URL: { + type: Scratch.ArgumentType.STRING, + defaultValue: "", + }, + BACKUP: { + type: Scratch.ArgumentType.STRING, + menu: "FALLBACKS", + }, + }, + }, + { + opcode: "removeFont", + blockType: Scratch.BlockType.COMMAND, + text: "remove font [NAME]", + arguments: { + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Comic Sans MS", + }, + }, + }, + { + opcode: "removeAllFonts", + blockType: Scratch.BlockType.COMMAND, + text: "remove all fonts", + }, + "---", + { + opcode: "whenFont", + blockType: Scratch.BlockType.EVENT, + text: "when font is [ADDED]", + isEdgeActivated: false, + arguments: { + ADDED: { + type: Scratch.ArgumentType.STRING, + menu: "ADDED_FIELD", + }, + }, + }, + { + disableMonitor: true, + opcode: "fontsChanged", + blockType: Scratch.BlockType.REPORTER, + text: "[ADDED] fonts", + arguments: { + ADDED: { + type: Scratch.ArgumentType.STRING, + menu: "ADDED_INPUT", + }, + }, + }, + ], + menus: { + DATA: { + acceptReporters: true, + items: [ + { + text: Scratch.translate("fallback"), + value: "fallback", + }, + { + text: Scratch.translate("is system"), + value: "is system", + }, + { + text: Scratch.translate("data: uri"), + value: "data: uri", + }, + { + text: Scratch.translate("format"), + value: "format", + }, + ], + }, + ADDED_FIELD: { + acceptReporters: false, + items: [ + { + text: Scratch.translate("added"), + value: "added", + }, + { + text: Scratch.translate("removed"), + value: "removed", + }, + ], + }, + ADDED_INPUT: { + acceptReporters: true, + items: [ + { + text: Scratch.translate("added"), + value: "added", + }, + { + text: Scratch.translate("removed"), + value: "removed", + }, + ], + }, + FALLBACKS: [ + "Sans Serif", + "Serif", + "Handwriting", + "Marker", + "Curly", + "Pixel", + "Scratch", + ], + FILES: { + acceptReporters: true, + items: FONT_EXTENSIONS.map((i) => `.${i}`), + }, + }, + }; + } + + fontNames() { + return JSON.stringify(fontManager.fonts.map((i) => i.family)); + } + + fontAdded(args) { + return fontManager.hasFont(Scratch.Cast.toString(args.NAME)); + } + + fontDetail(args) { + const name = Scratch.Cast.toString(args.NAME); + const font = fontManager.fonts.find((item) => item.family === name); + if (!font) return ""; + switch (Scratch.Cast.toString(args.DATA)) { + case "is system": + return font.system; + case "data: uri": + return font.asset ? font.asset.encodeDataURI() : ""; + case "format": + return font.asset ? font.asset.dataFormat : ""; + case "fallback": + return font.fallback; + } + return ""; + } + + addSystemFont(args) { + const name = Scratch.Cast.toString(args.NAME); + if (fontManager.isValidFamily(name)) { + fontManager.addSystemFont(name, Scratch.Cast.toString(args.BACKUP)); + } + } + + async addCustomFont(args) { + const name = Scratch.Cast.toString(args.NAME); + if (!fontManager.isValidFamily(name)) { + return; + } + + try { + const response = await Scratch.fetch(Scratch.Cast.toString(args.URL)); + const arrayBuffer = await response.arrayBuffer(); + const uint8array = new Uint8Array(arrayBuffer); + + // font files should have a content-type of font/ttf, font/otf, etc. + // if we can't detect it, we'll just assume it's ttf, browser can figure it out anyways + const contentType = ( + response.headers.get("content-type") || "" + ).toLowerCase(); + let dataFormat = vm.runtime.storage.DataFormat.TTF; + for (const extension of FONT_EXTENSIONS) { + if (contentType === `font/${extension}`) { + dataFormat = extension; + break; + } + } + + const asset = vm.runtime.storage.createAsset( + vm.runtime.storage.AssetType.Font, + dataFormat, + uint8array, + null, + true + ); + fontManager.addCustomFont( + name, + Scratch.Cast.toString(args.BACKUP), + asset + ); + } catch (e) { + console.warn(e); + } + } + + removeFont(args) { + const name = args.NAME; + const index = fontManager.fonts.findIndex( + (i) => i.family === Scratch.Cast.toString(name) + ); + if (index !== -1) { + fontManager.deleteFont(index); + } + } + + removeAllFonts() { + fontManager.clear(); + } + + fontsChanged(args, util) { + const added = Scratch.Cast.toString(args.ADDED); + if (added === "added") { + return JSON.stringify(this.addedFonts); + } else if (added === "removed") { + return JSON.stringify(this.removedFonts); + } else { + return ""; + } + } + + _onchange() { + this.removedFonts = []; + this.addedFonts = []; + for (const family of this.oldFonts) { + if (!fontManager.hasFont(family)) { + this.removedFonts.push(family); + } + } + for (const { family } of fontManager.fonts) { + if (!this.oldFonts.includes(family)) { + this.addedFonts.push(family); + } + } + this.oldFonts = fontManager.fonts.map((i) => i.family); + + if (this.addedFonts.length) { + Scratch.vm.runtime.startHats(`${extensionId}_whenFont`, { + ADDED: "added", + }); + } + if (this.removedFonts.length) { + Scratch.vm.runtime.startHats(`${extensionId}_whenFont`, { + ADDED: "removed", + }); + } + } + } + + Scratch.extensions.register(new SPASfontManager()); +})(Scratch); diff --git a/extensions/extensions.json b/extensions/extensions.json index c3d5a4c9e5..0d6b5f5078 100644 --- a/extensions/extensions.json +++ b/extensions/extensions.json @@ -61,6 +61,7 @@ "ZXMushroom63/searchApi", "TheShovel/ShovelUtils", "Lily/Assets", + "SharkPool/Font-Manager", "DNin/wake-lock", "Skyhigh173/json", "mbw/xml", diff --git a/images/SharkPool/Font-Manager.svg b/images/SharkPool/Font-Manager.svg new file mode 100644 index 0000000000..30d441454b --- /dev/null +++ b/images/SharkPool/Font-Manager.svg @@ -0,0 +1,31 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +