1
0
mirror of https://github.com/tobspr/shapez.io.git synced 2025-12-13 18:21:51 +00:00

initial modloader draft

This commit is contained in:
tobspr 2022-01-13 21:20:42 +01:00
parent a7a2aad2b6
commit abf2fd3d94
9 changed files with 250 additions and 0 deletions

View File

@ -9,6 +9,7 @@ import { SOUNDS, MUSIC } from "../platform/sound";
import { AtlasDefinition, atlasFiles } from "./atlas_definitions";
import { initBuildingCodesAfterResourcesLoaded } from "../game/meta_building_registry";
import { cachebust } from "./cachebust";
import { MODS } from "../mods/modloader";
const logger = createLogger("background_loader");
@ -232,6 +233,7 @@ export class BackgroundResourcesLoader {
this.numAssetsToLoadTotal = 0;
this.numAssetsLoaded = 0;
})
.then(MODS.hook_injectSprites.bind(MODS))
);
}
}

View File

@ -2,6 +2,8 @@
import { Application } from "../application";
/* typehints:end */
import { MODS } from "../mods/modloader";
/**
* Used for the bug reporter, and the click detector which both have no handles to this.
* It would be nicer to have no globals, but this is the only one. I promise!
@ -14,4 +16,5 @@ export let GLOBAL_APP = null;
export function setGlobalApp(app) {
assert(!GLOBAL_APP, "Create application twice!");
GLOBAL_APP = app;
MODS.linkApp(app);
}

View File

@ -181,6 +181,10 @@ class LoaderImpl {
w: sourceSize.w,
h: sourceSize.h,
});
if (sprite.linksByResolution[scale]) {
// Seems data is already present, might have been replaced by a mod
continue;
}
sprite.linksByResolution[scale] = link;
}
}

View File

@ -1,3 +1,5 @@
import { MODS } from "../mods/modloader";
export const THEMES = {
dark: require("./themes/dark.json"),
light: require("./themes/light.json"),
@ -7,4 +9,5 @@ export let THEME = THEMES.light;
export function applyGameTheme(id) {
THEME = THEMES[id];
MODS.callHook("preprocessTheme", { id, theme: THEME });
}

View File

@ -11,6 +11,7 @@ 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");
@ -19,6 +20,8 @@ if (window.coreThreadLoadedCb) {
window.coreThreadLoadedCb();
}
MODS.hook_init();
// Logrocket
// if (!G_IS_DEV && !G_IS_STANDALONE) {
// const monthlyUsers = 300; // thousand

31
src/js/mods/demo_mod.js Normal file
View File

@ -0,0 +1,31 @@
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",
});
}
hook_init() {
this.interface.registerCss(`
* {
color: red !important;
}
`);
this.interface.registerSprite(
"sprites/colors/red.png",
""
);
}
hook_preprocessTheme({ id, theme }) {
theme.map.background = "#eee";
theme.items.outline = "#000";
}
}

50
src/js/mods/mod.js Normal file
View File

@ -0,0 +1,50 @@
/* typehints:start */
import { ModInterface } from "./mod_interface";
/* typehints:end */
export class Mod {
/**
*
* @param {object} metadata
* @param {string} metadata.name
* @param {string} metadata.version
* @param {string} metadata.authorName
* @param {string} metadata.authorContact
* @param {string} metadata.id
*/
constructor(metadata) {
this.metadata = metadata;
/**
* @type {ModInterface}
*/
this.interface = undefined;
}
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;
}
}
}

View File

@ -0,0 +1,77 @@
/* typehints:start */
import { Application } from "../application";
import { ModLoader } from "./modloader";
/* typehints:end */
import { createLogger } from "../core/logging";
import { AtlasSprite, SpriteAtlasLink } from "../core/sprites";
import { Mod } from "./mod";
const LOG = createLogger("mod-interface");
export class ModInterface {
/**
*
* @param {ModLoader} modLoader
* @param {Mod} mod
*/
constructor(modLoader, mod) {
/**
* @param {Application} app
*/
this.app = undefined;
this.modLoader = modLoader;
this.mod = mod;
}
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);
}
registerSprite(spriteId, base64string) {
assert(base64string.startsWith("data:image"));
const img = new Image();
img.src = base64string;
const sprite = new AtlasSprite(spriteId);
const link = new SpriteAtlasLink({
w: img.width,
h: img.height,
atlas: img,
packOffsetX: 0,
packOffsetY: 0,
packedW: img.width,
packedH: img.height,
packedX: 0,
packedY: 0,
});
sprite.linksByResolution["0.25"] = link;
sprite.linksByResolution["0.5"] = link;
sprite.linksByResolution["0.75"] = link;
// @ts-ignore
sprite.modSource = this.mod;
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);
}
}

77
src/js/mods/modloader.js Normal file
View File

@ -0,0 +1,77 @@
import { Loader } from "../core/loader";
import { createLogger } from "../core/logging";
import { AtlasSprite } from "../core/sprites";
import { DemoMod } from "./demo_mod";
import { Mod } from "./mod";
import { ModInterface } from "./mod_interface";
const LOG = createLogger("mods");
export class ModLoader {
constructor() {
LOG.log("modloader created");
/** @type {Mod[]} */
this.mods = [];
/** @type {Map<string, AtlasSprite>} */
this.lazySprites = new Map();
this.initialized = false;
}
linkApp(app) {
this.app = app;
this.mods.forEach(mod => (mod.interface.app = app));
}
hook_init() {
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));
});
}
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
*/
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);
}
}
export const MODS = new ModLoader();
MODS.registerMod(new DemoMod());