mirror of
https://github.com/tobspr/shapez.io.git
synced 2025-12-13 18:21:51 +00:00
Refactor mods to use signals
This commit is contained in:
parent
3e5716504a
commit
6bac7cec57
@ -233,7 +233,7 @@ export class BackgroundResourcesLoader {
|
||||
this.numAssetsToLoadTotal = 0;
|
||||
this.numAssetsLoaded = 0;
|
||||
})
|
||||
.then(MODS.hook_injectSprites.bind(MODS))
|
||||
.then(MODS.modInterface.injectSprites.bind(MODS))
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
@ -512,6 +512,8 @@ export function generateLevelDefinitions(limitedVersion = false) {
|
||||
]),
|
||||
];
|
||||
|
||||
MODS.signals.modifyLevelDefinitions.dispatch(levelDefinitions);
|
||||
|
||||
if (G_IS_DEV) {
|
||||
levelDefinitions.forEach(({ shape }) => {
|
||||
try {
|
||||
@ -522,8 +524,6 @@ export function generateLevelDefinitions(limitedVersion = false) {
|
||||
});
|
||||
}
|
||||
|
||||
MODS.callHook("modifyLevelDefinitions", levelDefinitions);
|
||||
|
||||
return levelDefinitions;
|
||||
}
|
||||
|
||||
|
||||
@ -9,5 +9,5 @@ export let THEME = THEMES.light;
|
||||
|
||||
export function applyGameTheme(id) {
|
||||
THEME = THEMES[id];
|
||||
MODS.callHook("preprocessTheme", { id, theme: THEME });
|
||||
MODS.signals.preprocessTheme.dispatch({ id, theme: THEME });
|
||||
}
|
||||
|
||||
@ -2,6 +2,8 @@ import "./core/polyfills";
|
||||
import "./core/assert";
|
||||
import "./core/error_handler";
|
||||
|
||||
import "./mods/modloader";
|
||||
|
||||
import { createLogger, logSection } from "./core/logging";
|
||||
import { Application } from "./application";
|
||||
import { IS_DEBUG } from "./core/config";
|
||||
@ -11,7 +13,6 @@ import { initItemRegistry } from "./game/item_registry";
|
||||
import { initMetaBuildingRegistry } from "./game/meta_building_registry";
|
||||
import { initGameModeRegistry } from "./game/game_mode_registry";
|
||||
import { initGameSpeedRegistry } from "./game/game_speed_registry";
|
||||
import { MODS } from "./mods/modloader";
|
||||
|
||||
const logger = createLogger("main");
|
||||
|
||||
@ -20,8 +21,6 @@ if (window.coreThreadLoadedCb) {
|
||||
window.coreThreadLoadedCb();
|
||||
}
|
||||
|
||||
MODS.hook_init();
|
||||
|
||||
// Logrocket
|
||||
// if (!G_IS_DEV && !G_IS_STANDALONE) {
|
||||
// const monthlyUsers = 300; // thousand
|
||||
|
||||
@ -1,32 +1,35 @@
|
||||
import { Mod } from "./mod";
|
||||
|
||||
export class DemoMod extends Mod {
|
||||
constructor() {
|
||||
super({
|
||||
authorContact: "tobias@tobspr.io",
|
||||
authorName: "tobspr",
|
||||
name: "Demo Mod",
|
||||
version: "1",
|
||||
id: "demo-mod",
|
||||
});
|
||||
constructor(modLoader) {
|
||||
super(
|
||||
{
|
||||
authorContact: "tobias@tobspr.io",
|
||||
authorName: "tobspr",
|
||||
name: "Demo Mod",
|
||||
version: "1",
|
||||
id: "demo-mod",
|
||||
},
|
||||
modLoader
|
||||
);
|
||||
}
|
||||
|
||||
hook_init() {
|
||||
init() {
|
||||
// Add some custom css
|
||||
this.interface.registerCss(`
|
||||
* {
|
||||
color: red !important;
|
||||
}
|
||||
this.modLoader.modInterface.registerCss(`
|
||||
* {
|
||||
color: red !important;
|
||||
}
|
||||
`);
|
||||
|
||||
// Replace a builtin sprite
|
||||
this.interface.registerSprite(
|
||||
this.modLoader.modInterface.registerSprite(
|
||||
"sprites/colors/red.png",
|
||||
""
|
||||
);
|
||||
|
||||
// Add a new type of sub shape ("Line", short code "L")
|
||||
this.interface.registerSubShapeType({
|
||||
this.modLoader.modInterface.registerSubShapeType({
|
||||
id: "line",
|
||||
shortCode: "L",
|
||||
weightComputation: distanceToOriginInChunks =>
|
||||
@ -46,16 +49,16 @@ export class DemoMod extends Mod {
|
||||
context.closePath();
|
||||
},
|
||||
});
|
||||
}
|
||||
|
||||
hook_preprocessTheme({ id, theme }) {
|
||||
// Modify the theme colors
|
||||
theme.map.background = "#eee";
|
||||
theme.items.outline = "#000";
|
||||
}
|
||||
this.modLoader.signals.preprocessTheme.add(({ theme }) => {
|
||||
theme.map.background = "#eee";
|
||||
theme.items.outline = "#000";
|
||||
});
|
||||
|
||||
hook_modifyLevelDefinitions(definitions) {
|
||||
// Modify the goal of the first level
|
||||
definitions[0].shape = "LuCuLuCu";
|
||||
this.modLoader.signals.modifyLevelDefinitions.add(definitions => {
|
||||
definitions[0].shape = "LuCuLuCu";
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
@ -1,5 +1,5 @@
|
||||
/* typehints:start */
|
||||
import { ModInterface } from "./mod_interface";
|
||||
import { ModLoader } from "./modloader";
|
||||
/* typehints:end */
|
||||
|
||||
export class Mod {
|
||||
@ -11,40 +11,13 @@ export class Mod {
|
||||
* @param {string} metadata.authorName
|
||||
* @param {string} metadata.authorContact
|
||||
* @param {string} metadata.id
|
||||
*
|
||||
* @param {ModLoader} modLoader
|
||||
*/
|
||||
constructor(metadata) {
|
||||
constructor(metadata, modLoader) {
|
||||
this.metadata = metadata;
|
||||
|
||||
/**
|
||||
* @type {ModInterface}
|
||||
*/
|
||||
this.interface = undefined;
|
||||
this.modLoader = modLoader;
|
||||
}
|
||||
|
||||
hook_init() {}
|
||||
|
||||
executeGuarded(taskName, task) {
|
||||
try {
|
||||
return task();
|
||||
} catch (ex) {
|
||||
console.error(ex);
|
||||
alert(
|
||||
"Mod " +
|
||||
this.metadata.name +
|
||||
" (version " +
|
||||
this.metadata.version +
|
||||
")" +
|
||||
" failed to execute '" +
|
||||
taskName +
|
||||
"':\n\n" +
|
||||
ex +
|
||||
"\n\nPlease forward this to the mod author:\n\n" +
|
||||
this.metadata.authorName +
|
||||
" (" +
|
||||
this.metadata.authorContact +
|
||||
")"
|
||||
);
|
||||
throw ex;
|
||||
}
|
||||
}
|
||||
init() {}
|
||||
}
|
||||
|
||||
@ -7,6 +7,7 @@ import { createLogger } from "../core/logging";
|
||||
import { AtlasSprite, SpriteAtlasLink } from "../core/sprites";
|
||||
import { Mod } from "./mod";
|
||||
import { enumShortcodeToSubShape, enumSubShape, enumSubShapeToShortcode } from "../game/shape_definition";
|
||||
import { Loader } from "../core/loader";
|
||||
|
||||
const LOG = createLogger("mod-interface");
|
||||
|
||||
@ -32,23 +33,22 @@ export class ModInterface {
|
||||
/**
|
||||
*
|
||||
* @param {ModLoader} modLoader
|
||||
* @param {Mod} mod
|
||||
*/
|
||||
constructor(modLoader, mod) {
|
||||
constructor(modLoader) {
|
||||
/**
|
||||
* @param {Application} app
|
||||
*/
|
||||
this.app = undefined;
|
||||
|
||||
this.modLoader = modLoader;
|
||||
this.mod = mod;
|
||||
|
||||
/** @type {Map<string, AtlasSprite>} */
|
||||
this.lazySprites = new Map();
|
||||
}
|
||||
|
||||
registerCss(cssString) {
|
||||
const element = document.createElement("style");
|
||||
element.textContent = cssString;
|
||||
element.setAttribute("data-mod-id", this.mod.metadata.id);
|
||||
element.setAttribute("data-mod-name", this.mod.metadata.name);
|
||||
document.head.appendChild(element);
|
||||
}
|
||||
|
||||
@ -75,23 +75,15 @@ export class ModInterface {
|
||||
sprite.linksByResolution["0.5"] = link;
|
||||
sprite.linksByResolution["0.75"] = link;
|
||||
|
||||
// @ts-ignore
|
||||
sprite.modSource = this.mod;
|
||||
this.lazySprites.set(spriteId, sprite);
|
||||
}
|
||||
|
||||
const oldSprite = this.modLoader.lazySprites.get(spriteId);
|
||||
if (oldSprite) {
|
||||
LOG.error(
|
||||
"Sprite '" +
|
||||
spriteId +
|
||||
"' is provided twice, once by mod '" +
|
||||
// @ts-ignore
|
||||
oldSprite.modSource.metadata.name +
|
||||
"' and once by mod '" +
|
||||
this.mod.metadata.name +
|
||||
"'. This could cause artifacts."
|
||||
);
|
||||
}
|
||||
this.modLoader.lazySprites.set(spriteId, sprite);
|
||||
injectSprites() {
|
||||
LOG.log("inject sprites");
|
||||
this.lazySprites.forEach((sprite, key) => {
|
||||
Loader.sprites.set(key, sprite);
|
||||
console.log("override", key);
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
|
||||
@ -1,6 +1,5 @@
|
||||
import { Loader } from "../core/loader";
|
||||
import { createLogger } from "../core/logging";
|
||||
import { AtlasSprite } from "../core/sprites";
|
||||
import { Signal } from "../core/signal";
|
||||
import { DemoMod } from "./demo_mod";
|
||||
import { Mod } from "./mod";
|
||||
import { ModInterface } from "./mod_interface";
|
||||
@ -14,64 +13,50 @@ export class ModLoader {
|
||||
/** @type {Mod[]} */
|
||||
this.mods = [];
|
||||
|
||||
/** @type {Map<string, AtlasSprite>} */
|
||||
this.lazySprites = new Map();
|
||||
this.modInterface = new ModInterface(this);
|
||||
|
||||
/** @type {(new (ModLoader) => Mod)[]} */
|
||||
this.modLoadQueue = [];
|
||||
|
||||
this.initialized = false;
|
||||
|
||||
this.signals = {
|
||||
postInit: new Signal(),
|
||||
injectSprites: new Signal(),
|
||||
preprocessTheme: /** @type {TypedSignal<[Object]>} */ (new Signal()),
|
||||
modifyLevelDefinitions: /** @type {TypedSignal<[Array[Object]]>} */ (new Signal()),
|
||||
};
|
||||
|
||||
this.registerMod(DemoMod);
|
||||
this.initMods();
|
||||
}
|
||||
|
||||
linkApp(app) {
|
||||
this.app = app;
|
||||
this.mods.forEach(mod => (mod.interface.app = app));
|
||||
}
|
||||
|
||||
hook_init() {
|
||||
initMods() {
|
||||
LOG.log("hook:init");
|
||||
this.initialized = true;
|
||||
this.mods.forEach(mod => {
|
||||
LOG.log("Loading mod", mod.metadata.name);
|
||||
mod.interface = new ModInterface(this, mod);
|
||||
mod.executeGuarded("hook_init", mod.hook_init.bind(mod));
|
||||
this.modLoadQueue.forEach(modClass => {
|
||||
const mod = new modClass(this);
|
||||
mod.init();
|
||||
this.mods.push(mod);
|
||||
});
|
||||
this.modLoadQueue = [];
|
||||
this.signals.postInit.dispatch();
|
||||
}
|
||||
|
||||
hook_injectSprites() {
|
||||
LOG.log("hook:injectSprites");
|
||||
this.lazySprites.forEach((sprite, key) => {
|
||||
Loader.sprites.set(key, sprite);
|
||||
console.log("override", key);
|
||||
});
|
||||
}
|
||||
|
||||
callHook(id, structuredArgs) {
|
||||
LOG.log("hook:" + id);
|
||||
this.mods.forEach(mod => {
|
||||
const handler = mod["hook_" + id];
|
||||
if (handler) {
|
||||
mod.executeGuarded("hook:" + id, handler.bind(mod, structuredArgs));
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
registerSprite() {}
|
||||
|
||||
registerGameState() {}
|
||||
|
||||
registerBuilding() {}
|
||||
|
||||
/**
|
||||
*
|
||||
* @param {Mod} mod
|
||||
* @param {new (ModLoader) => Mod} mod
|
||||
*/
|
||||
registerMod(mod) {
|
||||
LOG.log("Registering mod", mod.metadata.name);
|
||||
if (this.initialized) {
|
||||
throw new Error("Mods are already initialized, can not add mod afterwards.");
|
||||
}
|
||||
this.mods.push(mod);
|
||||
this.modLoadQueue.push(mod);
|
||||
}
|
||||
}
|
||||
|
||||
export const MODS = new ModLoader();
|
||||
|
||||
MODS.registerMod(new DemoMod());
|
||||
|
||||
Loading…
Reference in New Issue
Block a user