Skip to content

Commit

Permalink
New extension: Font Manager (#1221)
Browse files Browse the repository at this point in the history
  • Loading branch information
SharkPool-SP authored May 10, 2024
1 parent cd7e529 commit 507e37b
Show file tree
Hide file tree
Showing 3 changed files with 395 additions and 0 deletions.
363 changes: 363 additions & 0 deletions extensions/SharkPool/Font-Manager.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,363 @@
// Name: Font Manager
// ID: SPASfontManager
// Description: Add, delete, and manage fonts.
// By: SharkPool
// By: Ashimee <https://scratch.mit.edu/users/0znzw/>
// 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);
1 change: 1 addition & 0 deletions extensions/extensions.json
Original file line number Diff line number Diff line change
Expand Up @@ -61,6 +61,7 @@
"ZXMushroom63/searchApi",
"TheShovel/ShovelUtils",
"Lily/Assets",
"SharkPool/Font-Manager",
"DNin/wake-lock",
"Skyhigh173/json",
"mbw/xml",
Expand Down
Loading

0 comments on commit 507e37b

Please sign in to comment.