From abf2fd3d9424b8eef84fd3262a3f90159cf55ee7 Mon Sep 17 00:00:00 2001 From: tobspr Date: Thu, 13 Jan 2022 21:20:42 +0100 Subject: [PATCH 001/129] initial modloader draft --- src/js/core/background_resources_loader.js | 2 + src/js/core/globals.js | 3 + src/js/core/loader.js | 4 ++ src/js/game/theme.js | 3 + src/js/main.js | 3 + src/js/mods/demo_mod.js | 31 +++++++++ src/js/mods/mod.js | 50 ++++++++++++++ src/js/mods/mod_interface.js | 77 ++++++++++++++++++++++ src/js/mods/modloader.js | 77 ++++++++++++++++++++++ 9 files changed, 250 insertions(+) create mode 100644 src/js/mods/demo_mod.js create mode 100644 src/js/mods/mod.js create mode 100644 src/js/mods/mod_interface.js create mode 100644 src/js/mods/modloader.js diff --git a/src/js/core/background_resources_loader.js b/src/js/core/background_resources_loader.js index 316619c4..e1a585b6 100644 --- a/src/js/core/background_resources_loader.js +++ b/src/js/core/background_resources_loader.js @@ -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)) ); } } diff --git a/src/js/core/globals.js b/src/js/core/globals.js index 15197880..fcb9a3e8 100644 --- a/src/js/core/globals.js +++ b/src/js/core/globals.js @@ -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); } diff --git a/src/js/core/loader.js b/src/js/core/loader.js index cadbc048..b955beda 100644 --- a/src/js/core/loader.js +++ b/src/js/core/loader.js @@ -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; } } diff --git a/src/js/game/theme.js b/src/js/game/theme.js index 251f4433..4f68997e 100644 --- a/src/js/game/theme.js +++ b/src/js/game/theme.js @@ -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 }); } diff --git a/src/js/main.js b/src/js/main.js index 94f3d37a..3046820c 100644 --- a/src/js/main.js +++ b/src/js/main.js @@ -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 diff --git a/src/js/mods/demo_mod.js b/src/js/mods/demo_mod.js new file mode 100644 index 00000000..c5003da0 --- /dev/null +++ b/src/js/mods/demo_mod.js @@ -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"; + } +} diff --git a/src/js/mods/mod.js b/src/js/mods/mod.js new file mode 100644 index 00000000..7a0b3a7f --- /dev/null +++ b/src/js/mods/mod.js @@ -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; + } + } +} diff --git a/src/js/mods/mod_interface.js b/src/js/mods/mod_interface.js new file mode 100644 index 00000000..47bff484 --- /dev/null +++ b/src/js/mods/mod_interface.js @@ -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); + } +} diff --git a/src/js/mods/modloader.js b/src/js/mods/modloader.js new file mode 100644 index 00000000..2b6cbbaf --- /dev/null +++ b/src/js/mods/modloader.js @@ -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} */ + 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()); From 3e5716504ad86229add6bcba11ec137344a9bf6e Mon Sep 17 00:00:00 2001 From: tobspr Date: Thu, 13 Jan 2022 21:45:09 +0100 Subject: [PATCH 002/129] modloader features --- src/js/game/map_chunk.js | 5 +++++ src/js/game/modes/regular.js | 3 +++ src/js/game/shape_definition.js | 34 ++++++++++++++-------------- src/js/mods/demo_mod.js | 36 +++++++++++++++++++++++++++--- src/js/mods/mod_interface.js | 39 +++++++++++++++++++++++++++++++++ 5 files changed, 98 insertions(+), 19 deletions(-) diff --git a/src/js/game/map_chunk.js b/src/js/game/map_chunk.js index 54af1125..0c5390e4 100644 --- a/src/js/game/map_chunk.js +++ b/src/js/game/map_chunk.js @@ -10,6 +10,7 @@ import { COLOR_ITEM_SINGLETONS } from "./items/color_item"; import { GameRoot } from "./root"; import { enumSubShape } from "./shape_definition"; import { Rectangle } from "../core/rectangle"; +import { MODS_ADDITIONAL_SHAPE_MAP_WEIGHTS } from "../mods/mod_interface"; const logger = createLogger("map_chunk"); @@ -192,6 +193,10 @@ export class MapChunk { [enumSubShape.windmill]: Math.round(6 + clamp(distanceToOriginInChunks / 2, 0, 20)), }; + for (const key in MODS_ADDITIONAL_SHAPE_MAP_WEIGHTS) { + weights[key] = MODS_ADDITIONAL_SHAPE_MAP_WEIGHTS[key](distanceToOriginInChunks); + } + if (distanceToOriginInChunks < 7) { // Initial chunks can not spawn the good stuff weights[enumSubShape.star] = 0; diff --git a/src/js/game/modes/regular.js b/src/js/game/modes/regular.js index 429c1515..c8d6c9e5 100644 --- a/src/js/game/modes/regular.js +++ b/src/js/game/modes/regular.js @@ -38,6 +38,7 @@ import { HUDSandboxController } from "../hud/parts/sandbox_controller"; import { queryParamOptions } from "../../core/query_parameters"; import { MetaBlockBuilding } from "../buildings/block"; import { MetaItemProducerBuilding } from "../buildings/item_producer"; +import { MODS } from "../../mods/modloader"; /** @typedef {{ * shape: string, @@ -521,6 +522,8 @@ export function generateLevelDefinitions(limitedVersion = false) { }); } + MODS.callHook("modifyLevelDefinitions", levelDefinitions); + return levelDefinitions; } diff --git a/src/js/game/shape_definition.js b/src/js/game/shape_definition.js index b09d73c5..acc0bd1c 100644 --- a/src/js/game/shape_definition.js +++ b/src/js/game/shape_definition.js @@ -3,6 +3,7 @@ import { globalConfig } from "../core/config"; import { smoothenDpi } from "../core/dpi_manager"; import { DrawParameters } from "../core/draw_parameters"; import { Vector } from "../core/vector"; +import { MODS_ADDITIONAL_SUB_SHAPE_DRAWERS } from "../mods/mod_interface"; import { BasicSerializableObject, types } from "../savegame/serialization"; import { enumColors, enumColorsToHexCode, enumColorToShortcode, enumShortcodeToColor } from "./colors"; import { THEME } from "./theme"; @@ -366,18 +367,11 @@ export class ShapeDefinition extends BasicSerializableObject { context.strokeStyle = THEME.items.outline; context.lineWidth = THEME.items.outlineWidth; - const insetPadding = 0.0; - switch (subShape) { case enumSubShape.rect: { context.beginPath(); const dims = quadrantSize * layerScale; - context.rect( - insetPadding + -quadrantHalfSize, - -insetPadding + quadrantHalfSize - dims, - dims, - dims - ); + context.rect(-quadrantHalfSize, quadrantHalfSize - dims, dims, dims); break; } @@ -385,8 +379,8 @@ export class ShapeDefinition extends BasicSerializableObject { context.beginPath(); const dims = quadrantSize * layerScale; - let originX = insetPadding - quadrantHalfSize; - let originY = -insetPadding + quadrantHalfSize - dims; + let originX = -quadrantHalfSize; + let originY = quadrantHalfSize - dims; const moveInwards = dims * 0.4; context.moveTo(originX, originY + moveInwards); @@ -401,8 +395,8 @@ export class ShapeDefinition extends BasicSerializableObject { context.beginPath(); const dims = quadrantSize * layerScale; - let originX = insetPadding - quadrantHalfSize; - let originY = -insetPadding + quadrantHalfSize - dims; + let originX = -quadrantHalfSize; + let originY = quadrantHalfSize - dims; const moveInwards = dims * 0.4; context.moveTo(originX, originY + moveInwards); context.lineTo(originX + dims, originY); @@ -414,10 +408,10 @@ export class ShapeDefinition extends BasicSerializableObject { case enumSubShape.circle: { context.beginPath(); - context.moveTo(insetPadding + -quadrantHalfSize, -insetPadding + quadrantHalfSize); + context.moveTo(-quadrantHalfSize, quadrantHalfSize); context.arc( - insetPadding + -quadrantHalfSize, - -insetPadding + quadrantHalfSize, + -quadrantHalfSize, + quadrantHalfSize, quadrantSize * layerScale, -Math.PI * 0.5, 0 @@ -427,7 +421,15 @@ export class ShapeDefinition extends BasicSerializableObject { } default: { - assertAlways(false, "Unkown sub shape: " + subShape); + if (MODS_ADDITIONAL_SUB_SHAPE_DRAWERS[subShape]) { + MODS_ADDITIONAL_SUB_SHAPE_DRAWERS[subShape]({ + context, + layerScale, + quadrantSize, + }); + } else { + throw new Error("Unkown sub shape: " + subShape); + } } } diff --git a/src/js/mods/demo_mod.js b/src/js/mods/demo_mod.js index c5003da0..7dca4bad 100644 --- a/src/js/mods/demo_mod.js +++ b/src/js/mods/demo_mod.js @@ -12,20 +12,50 @@ export class DemoMod extends Mod { } hook_init() { + // Add some custom css this.interface.registerCss(` - * { - color: red !important; - } + * { + color: red !important; + } `); + // Replace a builtin sprite this.interface.registerSprite( "sprites/colors/red.png", "" ); + + // Add a new type of sub shape ("Line", short code "L") + this.interface.registerSubShapeType({ + id: "line", + shortCode: "L", + weightComputation: distanceToOriginInChunks => + Math.round(20 + Math.max(Math.min(distanceToOriginInChunks, 30), 0)), + + shapeDrawer: ({ context, quadrantSize, layerScale }) => { + const quadrantHalfSize = quadrantSize / 2; + context.beginPath(); + context.moveTo(-quadrantHalfSize, quadrantHalfSize); + context.arc( + -quadrantHalfSize, + quadrantHalfSize, + quadrantSize * layerScale, + -Math.PI * 0.25, + 0 + ); + context.closePath(); + }, + }); } hook_preprocessTheme({ id, theme }) { + // Modify the theme colors theme.map.background = "#eee"; theme.items.outline = "#000"; } + + hook_modifyLevelDefinitions(definitions) { + // Modify the goal of the first level + definitions[0].shape = "LuCuLuCu"; + } } diff --git a/src/js/mods/mod_interface.js b/src/js/mods/mod_interface.js index 47bff484..16de4d67 100644 --- a/src/js/mods/mod_interface.js +++ b/src/js/mods/mod_interface.js @@ -6,9 +6,28 @@ import { ModLoader } from "./modloader"; import { createLogger } from "../core/logging"; import { AtlasSprite, SpriteAtlasLink } from "../core/sprites"; import { Mod } from "./mod"; +import { enumShortcodeToSubShape, enumSubShape, enumSubShapeToShortcode } from "../game/shape_definition"; const LOG = createLogger("mod-interface"); +/** + * @type {Object number>} + */ +export const MODS_ADDITIONAL_SHAPE_MAP_WEIGHTS = {}; + +/** + * @typedef {{ + * context: CanvasRenderingContext2D, + * quadrantSize: number, + * layerScale: number, + * }} SubShapeDrawOptions + */ + +/** + * @type {Object void>} + */ +export const MODS_ADDITIONAL_SUB_SHAPE_DRAWERS = {}; + export class ModInterface { /** * @@ -74,4 +93,24 @@ export class ModInterface { } this.modLoader.lazySprites.set(spriteId, sprite); } + + /** + * + * @param {object} param0 + * @param {string} param0.id + * @param {string} param0.shortCode + * @param {(distanceToOriginInChunks: number) => number} param0.weightComputation + * @param {(options: SubShapeDrawOptions) => void} param0.shapeDrawer + */ + registerSubShapeType({ id, shortCode, weightComputation, shapeDrawer }) { + if (shortCode.length !== 1) { + throw new Error("Bad short code: " + shortCode); + } + enumSubShape[id] = id; + enumSubShapeToShortcode[id] = shortCode; + enumShortcodeToSubShape[shortCode] = id; + + MODS_ADDITIONAL_SHAPE_MAP_WEIGHTS[id] = weightComputation; + MODS_ADDITIONAL_SUB_SHAPE_DRAWERS[id] = shapeDrawer; + } } From 6bac7cec57f38ab97a2ace548ad9132b0d6f3bca Mon Sep 17 00:00:00 2001 From: tobspr Date: Thu, 13 Jan 2022 22:14:49 +0100 Subject: [PATCH 003/129] Refactor mods to use signals --- src/js/core/background_resources_loader.js | 2 +- src/js/game/modes/regular.js | 4 +- src/js/game/theme.js | 2 +- src/js/main.js | 5 +- src/js/mods/demo_mod.js | 47 ++++++++-------- src/js/mods/mod.js | 39 +++----------- src/js/mods/mod_interface.js | 34 +++++------- src/js/mods/modloader.js | 63 +++++++++------------- 8 files changed, 74 insertions(+), 122 deletions(-) diff --git a/src/js/core/background_resources_loader.js b/src/js/core/background_resources_loader.js index e1a585b6..927071a9 100644 --- a/src/js/core/background_resources_loader.js +++ b/src/js/core/background_resources_loader.js @@ -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)) ); } } diff --git a/src/js/game/modes/regular.js b/src/js/game/modes/regular.js index c8d6c9e5..0aad3669 100644 --- a/src/js/game/modes/regular.js +++ b/src/js/game/modes/regular.js @@ -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; } diff --git a/src/js/game/theme.js b/src/js/game/theme.js index 4f68997e..023710dd 100644 --- a/src/js/game/theme.js +++ b/src/js/game/theme.js @@ -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 }); } diff --git a/src/js/main.js b/src/js/main.js index 3046820c..71ec59a9 100644 --- a/src/js/main.js +++ b/src/js/main.js @@ -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 diff --git a/src/js/mods/demo_mod.js b/src/js/mods/demo_mod.js index 7dca4bad..f3bcbaf8 100644 --- a/src/js/mods/demo_mod.js +++ b/src/js/mods/demo_mod.js @@ -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"; + }); } } diff --git a/src/js/mods/mod.js b/src/js/mods/mod.js index 7a0b3a7f..65cdd53a 100644 --- a/src/js/mods/mod.js +++ b/src/js/mods/mod.js @@ -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() {} } diff --git a/src/js/mods/mod_interface.js b/src/js/mods/mod_interface.js index 16de4d67..9ce15d01 100644 --- a/src/js/mods/mod_interface.js +++ b/src/js/mods/mod_interface.js @@ -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} */ + 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); + }); } /** diff --git a/src/js/mods/modloader.js b/src/js/mods/modloader.js index 2b6cbbaf..1a903ae2 100644 --- a/src/js/mods/modloader.js +++ b/src/js/mods/modloader.js @@ -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} */ - 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()); From ebb2f3a1c6c6542fdb70137ff36c11fad4d1a776 Mon Sep 17 00:00:00 2001 From: tobspr Date: Thu, 13 Jan 2022 22:30:52 +0100 Subject: [PATCH 004/129] Add support for modifying and registering new transltions --- src/js/core/background_resources_loader.js | 2 +- src/js/core/globals.js | 2 +- src/js/mods/demo_mod.js | 11 ++++++++++ src/js/mods/mod_interface.js | 19 ++++++++++++----- src/js/translations.js | 24 ++++++++++++---------- 5 files changed, 40 insertions(+), 18 deletions(-) diff --git a/src/js/core/background_resources_loader.js b/src/js/core/background_resources_loader.js index 927071a9..10f8d6f0 100644 --- a/src/js/core/background_resources_loader.js +++ b/src/js/core/background_resources_loader.js @@ -233,7 +233,7 @@ export class BackgroundResourcesLoader { this.numAssetsToLoadTotal = 0; this.numAssetsLoaded = 0; }) - .then(MODS.modInterface.injectSprites.bind(MODS)) + .then(MODS.modInterface.injectSprites.bind(MODS.modInterface)) ); } } diff --git a/src/js/core/globals.js b/src/js/core/globals.js index fcb9a3e8..4e72dddf 100644 --- a/src/js/core/globals.js +++ b/src/js/core/globals.js @@ -16,5 +16,5 @@ export let GLOBAL_APP = null; export function setGlobalApp(app) { assert(!GLOBAL_APP, "Create application twice!"); GLOBAL_APP = app; - MODS.linkApp(app); + MODS.app = app; } diff --git a/src/js/mods/demo_mod.js b/src/js/mods/demo_mod.js index f3bcbaf8..5df293ce 100644 --- a/src/js/mods/demo_mod.js +++ b/src/js/mods/demo_mod.js @@ -60,5 +60,16 @@ export class DemoMod extends Mod { this.modLoader.signals.modifyLevelDefinitions.add(definitions => { definitions[0].shape = "LuCuLuCu"; }); + + this.modLoader.modInterface.registerTranslations("en", { + ingame: { + interactiveTutorial: { + title: "Hello", + hints: { + "1_1_extractor": "World!", + }, + }, + }, + }); } } diff --git a/src/js/mods/mod_interface.js b/src/js/mods/mod_interface.js index 9ce15d01..6cfc86be 100644 --- a/src/js/mods/mod_interface.js +++ b/src/js/mods/mod_interface.js @@ -8,6 +8,8 @@ import { AtlasSprite, SpriteAtlasLink } from "../core/sprites"; import { Mod } from "./mod"; import { enumShortcodeToSubShape, enumSubShape, enumSubShapeToShortcode } from "../game/shape_definition"; import { Loader } from "../core/loader"; +import { LANGUAGES } from "../languages"; +import { matchDataRecursive, T } from "../translations"; const LOG = createLogger("mod-interface"); @@ -35,11 +37,6 @@ export class ModInterface { * @param {ModLoader} modLoader */ constructor(modLoader) { - /** - * @param {Application} app - */ - this.app = undefined; - this.modLoader = modLoader; /** @type {Map} */ @@ -105,4 +102,16 @@ export class ModInterface { MODS_ADDITIONAL_SHAPE_MAP_WEIGHTS[id] = weightComputation; MODS_ADDITIONAL_SUB_SHAPE_DRAWERS[id] = shapeDrawer; } + + registerTranslations(language, translations) { + const data = LANGUAGES[language]; + if (!data) { + throw new Error("Unknown language: " + language); + } + + matchDataRecursive(data.data, translations, true); + if (language === "en") { + matchDataRecursive(T, translations, true); + } + } } diff --git a/src/js/translations.js b/src/js/translations.js index 6ad926bc..9d976a41 100644 --- a/src/js/translations.js +++ b/src/js/translations.js @@ -24,15 +24,6 @@ if (G_IS_DEV && globalConfig.debug.testTranslations) { mapTranslations(T); } -export function applyLanguage(languageCode) { - logger.log("Applying language:", languageCode); - const data = LANGUAGES[languageCode]; - if (!data) { - logger.error("Language not found:", languageCode); - return false; - } -} - // Language key is something like de-DE or en or en-US function mapLanguageCodeToId(languageKey) { const key = languageKey.toLowerCase(); @@ -97,17 +88,20 @@ export function autoDetectLanguageId() { return "en"; } -function matchDataRecursive(dest, src) { +export function matchDataRecursive(dest, src, addNewKeys = false) { if (typeof dest !== "object" || typeof src !== "object") { return; } + if (dest === null || src === null) { + return; + } for (const key in dest) { if (src[key]) { // console.log("copy", key); const data = dest[key]; if (typeof data === "object") { - matchDataRecursive(dest[key], src[key]); + matchDataRecursive(dest[key], src[key], addNewKeys); } else if (typeof data === "string" || typeof data === "number") { // console.log("match string", key); dest[key] = src[key]; @@ -116,6 +110,14 @@ function matchDataRecursive(dest, src) { } } } + + if (addNewKeys) { + for (const key in src) { + if (!dest[key]) { + dest[key] = JSON.parse(JSON.stringify(src[key])); + } + } + } } export function updateApplicationLanguage(id) { From 7e2501ea6ef74c66340d9a96c69c981ce631d9c1 Mon Sep 17 00:00:00 2001 From: tobspr Date: Thu, 13 Jan 2022 23:16:24 +0100 Subject: [PATCH 005/129] Minor adjustments --- src/js/mods/demo_mod.js | 2 +- src/js/mods/mod_interface.js | 7 +++---- 2 files changed, 4 insertions(+), 5 deletions(-) diff --git a/src/js/mods/demo_mod.js b/src/js/mods/demo_mod.js index 5df293ce..be3a7c1b 100644 --- a/src/js/mods/demo_mod.js +++ b/src/js/mods/demo_mod.js @@ -35,7 +35,7 @@ export class DemoMod extends Mod { weightComputation: distanceToOriginInChunks => Math.round(20 + Math.max(Math.min(distanceToOriginInChunks, 30), 0)), - shapeDrawer: ({ context, quadrantSize, layerScale }) => { + draw: ({ context, quadrantSize, layerScale }) => { const quadrantHalfSize = quadrantSize / 2; context.beginPath(); context.moveTo(-quadrantHalfSize, quadrantHalfSize); diff --git a/src/js/mods/mod_interface.js b/src/js/mods/mod_interface.js index 6cfc86be..eaf1d314 100644 --- a/src/js/mods/mod_interface.js +++ b/src/js/mods/mod_interface.js @@ -5,7 +5,6 @@ import { ModLoader } from "./modloader"; 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"; import { LANGUAGES } from "../languages"; @@ -89,9 +88,9 @@ export class ModInterface { * @param {string} param0.id * @param {string} param0.shortCode * @param {(distanceToOriginInChunks: number) => number} param0.weightComputation - * @param {(options: SubShapeDrawOptions) => void} param0.shapeDrawer + * @param {(options: SubShapeDrawOptions) => void} param0.draw */ - registerSubShapeType({ id, shortCode, weightComputation, shapeDrawer }) { + registerSubShapeType({ id, shortCode, weightComputation, draw }) { if (shortCode.length !== 1) { throw new Error("Bad short code: " + shortCode); } @@ -100,7 +99,7 @@ export class ModInterface { enumShortcodeToSubShape[shortCode] = id; MODS_ADDITIONAL_SHAPE_MAP_WEIGHTS[id] = weightComputation; - MODS_ADDITIONAL_SUB_SHAPE_DRAWERS[id] = shapeDrawer; + MODS_ADDITIONAL_SUB_SHAPE_DRAWERS[id] = draw; } registerTranslations(language, translations) { From 8777e4c6eab77e363218a9ebc57fcbb5ccc3a339 Mon Sep 17 00:00:00 2001 From: tobspr Date: Fri, 14 Jan 2022 06:14:37 +0100 Subject: [PATCH 006/129] Support for string building ids for mods --- src/js/game/building_codes.js | 27 ++++++++---- src/js/game/components/static_map_entity.js | 4 +- src/js/savegame/serialization.js | 2 + src/js/savegame/serialization_data_types.js | 47 +++++++++++++++++++++ 4 files changed, 70 insertions(+), 10 deletions(-) diff --git a/src/js/game/building_codes.js b/src/js/game/building_codes.js index 78240c31..5e2b717b 100644 --- a/src/js/game/building_codes.js +++ b/src/js/game/building_codes.js @@ -19,7 +19,7 @@ import { Vector } from "../core/vector"; /** * Stores a lookup table for all building variants (for better performance) - * @type {Object} + * @type {Object} */ export const gBuildingVariants = { // Set later @@ -27,13 +27,13 @@ export const gBuildingVariants = { /** * Mapping from 'metaBuildingId/variant/rotationVariant' to building code - * @type {Map} + * @type {Map} */ const variantsCache = new Map(); /** * Registers a new variant - * @param {number} code + * @param {number|string} code * @param {typeof MetaBuilding} meta * @param {string} variant * @param {number} rotationVariant @@ -54,9 +54,20 @@ export function registerBuildingVariant( }; } +/** + * Hashes the combination of buildng, variant and rotation variant + * @param {string} buildingId + * @param {string} variant + * @param {number} rotationVariant + * @returns + */ +function generateBuildingHash(buildingId, variant, rotationVariant) { + return buildingId + "/" + variant + "/" + rotationVariant; +} + /** * - * @param {number} code + * @param {string|number} code * @returns {BuildingVariantIdentifier} */ export function getBuildingDataFromCode(code) { @@ -70,8 +81,8 @@ export function getBuildingDataFromCode(code) { export function buildBuildingCodeCache() { for (const code in gBuildingVariants) { const data = gBuildingVariants[code]; - const hash = data.metaInstance.getId() + "/" + data.variant + "/" + data.rotationVariant; - variantsCache.set(hash, +code); + const hash = generateBuildingHash(data.metaInstance.getId(), data.variant, data.rotationVariant); + variantsCache.set(hash, isNaN(+code) ? code : +code); } } @@ -80,10 +91,10 @@ export function buildBuildingCodeCache() { * @param {MetaBuilding} metaBuilding * @param {string} variant * @param {number} rotationVariant - * @returns {number} + * @returns {number|string} */ export function getCodeFromBuildingData(metaBuilding, variant, rotationVariant) { - const hash = metaBuilding.getId() + "/" + variant + "/" + rotationVariant; + const hash = generateBuildingHash(metaBuilding.getId(), variant, rotationVariant); const result = variantsCache.get(hash); if (G_IS_DEV) { if (!result) { diff --git a/src/js/game/components/static_map_entity.js b/src/js/game/components/static_map_entity.js index c76a298e..a3d6a8ca 100644 --- a/src/js/game/components/static_map_entity.js +++ b/src/js/game/components/static_map_entity.js @@ -19,7 +19,7 @@ export class StaticMapEntityComponent extends Component { originalRotation: types.float, // See building_codes.js - code: types.uint, + code: types.uintOrString, }; } @@ -99,7 +99,7 @@ export class StaticMapEntityComponent extends Component { * @param {Vector=} param0.tileSize Size of the entity in tiles * @param {number=} param0.rotation Rotation in degrees. Must be multiple of 90 * @param {number=} param0.originalRotation Original Rotation in degrees. Must be multiple of 90 - * @param {number=} param0.code Building code + * @param {number|string=} param0.code Building code */ constructor({ origin = new Vector(), diff --git a/src/js/savegame/serialization.js b/src/js/savegame/serialization.js index 78642ceb..9a0ce3a5 100644 --- a/src/js/savegame/serialization.js +++ b/src/js/savegame/serialization.js @@ -22,6 +22,7 @@ import { TypeString, TypeStructuredObject, TypeVector, + TypePositiveIntegerOrString, } from "./serialization_data_types"; const logger = createLogger("serialization"); @@ -38,6 +39,7 @@ export const types = { vector: new TypeVector(), tileVector: new TypeVector(), bool: new TypeBoolean(), + uintOrString: new TypePositiveIntegerOrString(), /** * @param {BaseDataType} wrapped diff --git a/src/js/savegame/serialization_data_types.js b/src/js/savegame/serialization_data_types.js index 9d3b689f..df352e78 100644 --- a/src/js/savegame/serialization_data_types.js +++ b/src/js/savegame/serialization_data_types.js @@ -213,6 +213,53 @@ export class TypePositiveInteger extends BaseDataType { } } +export class TypePositiveIntegerOrString extends BaseDataType { + serialize(value) { + if (Number.isInteger(value)) { + assert(value >= 0, "type integer got negative value: " + value); + } else if (typeof value === "string") { + // all good + } else { + assertAlways(false, "Type integer|string got non integer or string for serialize: " + value); + } + return value; + } + + /** + * @see BaseDataType.deserialize + * @param {any} value + * @param {GameRoot} root + * @param {object} targetObject + * @param {string|number} targetKey + * @returns {string|void} String error code or null on success + */ + deserialize(value, targetObject, targetKey, root) { + targetObject[targetKey] = value; + } + + getAsJsonSchemaUncached() { + return { + oneOf: [{ type: "integer", minimum: 0 }, { type: "string" }], + }; + } + + verifySerializedValue(value) { + if (Number.isInteger(value)) { + if (value < 0) { + return "Negative value for positive integer"; + } + } else if (typeof value === "string") { + // all good + } else { + return "Not a valid number or string: " + value; + } + } + + getCacheKey() { + return "uint_str"; + } +} + export class TypeBoolean extends BaseDataType { serialize(value) { assert(value === true || value === false, "Type bool got non bool for serialize: " + value); From 01b9bf561c5f782b411e60193e1ca071f1cb3b7c Mon Sep 17 00:00:00 2001 From: tobspr Date: Fri, 14 Jan 2022 07:05:46 +0100 Subject: [PATCH 007/129] Initial support for adding new buildings --- src/js/game/hud/hud.js | 3 + src/js/game/hud/parts/base_toolbar.js | 9 ++- src/js/game/hud/parts/building_placer.js | 15 +++-- src/js/game/meta_building_registry.js | 11 ++-- src/js/mods/demo_mod.js | 65 ++++++++++++++++++-- src/js/mods/mod_interface.js | 78 +++++++++++++++++++++++- src/js/mods/modloader.js | 4 ++ 7 files changed, 165 insertions(+), 20 deletions(-) diff --git a/src/js/game/hud/hud.js b/src/js/game/hud/hud.js index 3d22787c..cffc290d 100644 --- a/src/js/game/hud/hud.js +++ b/src/js/game/hud/hud.js @@ -1,6 +1,7 @@ import { globalConfig } from "../../core/config"; import { DrawParameters } from "../../core/draw_parameters"; import { Signal } from "../../core/signal"; +import { MODS } from "../../mods/modloader"; import { KEYMAPPINGS } from "../key_action_mapper"; import { MetaBuilding } from "../meta_building"; import { GameRoot } from "../root"; @@ -91,6 +92,7 @@ export class GameHUD { const frag = document.createDocumentFragment(); for (const key in this.parts) { + MODS.signals.hudElementInitialized.dispatch(this.parts[key]); this.parts[key].createElements(frag); } @@ -98,6 +100,7 @@ export class GameHUD { for (const key in this.parts) { this.parts[key].initialize(); + MODS.signals.hudElementFinalized.dispatch(this.parts[key]); } this.root.keyMapper.getBinding(KEYMAPPINGS.ingame.toggleHud).add(this.toggleUi, this); diff --git a/src/js/game/hud/parts/base_toolbar.js b/src/js/game/hud/parts/base_toolbar.js index 15faad66..bee88574 100644 --- a/src/js/game/hud/parts/base_toolbar.js +++ b/src/js/game/hud/parts/base_toolbar.js @@ -1,4 +1,5 @@ import { gMetaBuildingRegistry } from "../../../core/global_registries"; +import { globalWarn } from "../../../core/logging"; import { STOP_PROPAGATION } from "../../../core/signal"; import { makeDiv, safeModulo } from "../../../core/utils"; import { MetaBlockBuilding } from "../../buildings/block"; @@ -101,7 +102,12 @@ export class HUDBaseToolbar extends BaseHUDPart { rawBinding = KEYMAPPINGS.buildings[metaBuilding.getId()]; } - const binding = actionMapper.getBinding(rawBinding); + if (rawBinding) { + const binding = actionMapper.getBinding(rawBinding); + binding.add(() => this.selectBuildingForPlacement(metaBuilding)); + } else { + globalWarn("Building has no keybinding:", metaBuilding.getId()); + } const itemContainer = makeDiv( this.primaryBuildings.includes(allBuildings[i]) ? rowPrimary : rowSecondary, @@ -110,7 +116,6 @@ export class HUDBaseToolbar extends BaseHUDPart { ); itemContainer.setAttribute("data-icon", "building_icons/" + metaBuilding.getId() + ".png"); itemContainer.setAttribute("data-id", metaBuilding.getId()); - binding.add(() => this.selectBuildingForPlacement(metaBuilding)); const icon = makeDiv(itemContainer, null, ["icon"]); diff --git a/src/js/game/hud/parts/building_placer.js b/src/js/game/hud/parts/building_placer.js index 33e6ebc2..ee7bc804 100644 --- a/src/js/game/hud/parts/building_placer.js +++ b/src/js/game/hud/parts/building_placer.js @@ -126,12 +126,15 @@ export class HUDBuildingPlacer extends HUDBuildingPlacerLogic { rawBinding = KEYMAPPINGS.buildings[metaBuilding.getId()]; } - const binding = this.root.keyMapper.getBinding(rawBinding); - - this.buildingInfoElements.hotkey.innerHTML = T.ingame.buildingPlacement.hotkeyLabel.replace( - "", - "" + binding.getKeyCodeString() + "" - ); + if (rawBinding) { + const binding = this.root.keyMapper.getBinding(rawBinding); + this.buildingInfoElements.hotkey.innerHTML = T.ingame.buildingPlacement.hotkeyLabel.replace( + "", + "" + binding.getKeyCodeString() + "" + ); + } else { + this.buildingInfoElements.hotkey.innerHTML = ""; + } this.buildingInfoElements.tutorialImage.setAttribute( "data-icon", diff --git a/src/js/game/meta_building_registry.js b/src/js/game/meta_building_registry.js index 0c93153d..193d60e5 100644 --- a/src/js/game/meta_building_registry.js +++ b/src/js/game/meta_building_registry.js @@ -205,18 +205,15 @@ export function initMetaBuildingRegistry() { const id = metaBuilding.getId(); if (!["hub"].includes(id)) { if (!KEYMAPPINGS.buildings[id]) { - assertAlways( - false, + console.error( "Building " + id + " has no keybinding assigned! Add it to key_action_mapper.js" ); } if (!T.buildings[id]) { - assertAlways(false, "Translation for building " + id + " missing!"); - } - - if (!T.buildings[id].default) { - assertAlways(false, "Translation for building " + id + " missing (default variant)!"); + console.error("Translation for building " + id + " missing!"); + } else if (!T.buildings[id].default) { + console.error("Translation for building " + id + " missing (default variant)!"); } } }); diff --git a/src/js/mods/demo_mod.js b/src/js/mods/demo_mod.js index be3a7c1b..42f48fa3 100644 --- a/src/js/mods/demo_mod.js +++ b/src/js/mods/demo_mod.js @@ -1,4 +1,25 @@ +/* typehints:start */ +import { Entity } from "../game/entity"; +/* typehints:end */ + import { Mod } from "./mod"; +import { MetaBuilding } from "../game/meta_building"; + +export class MetaDemoModBuilding extends MetaBuilding { + constructor() { + super("demoModBuilding"); + } + + getSilhouetteColor() { + return "red"; + } + + /** + * Creates the entity at the given location + * @param {Entity} entity + */ + setupEntityComponents(entity) {} +} export class DemoMod extends Mod { constructor(modLoader) { @@ -23,10 +44,7 @@ export class DemoMod extends Mod { `); // Replace a builtin sprite - this.modLoader.modInterface.registerSprite( - "sprites/colors/red.png", - "" - ); + this.modLoader.modInterface.registerSprite("sprites/colors/red.png", RESOURCES["red.png"]); // Add a new type of sub shape ("Line", short code "L") this.modLoader.modInterface.registerSubShapeType({ @@ -71,5 +89,44 @@ export class DemoMod extends Mod { }, }, }); + + // Register the new building + this.modLoader.modInterface.registerNewBuilding({ + metaClass: MetaDemoModBuilding, + buildingIconBase64: RESOURCES["demoBuilding.png"], + + variantsAndRotations: [ + { + description: "A test building", + name: "A test name", + + regularImageBase64: RESOURCES["demoBuilding.png"], + blueprintImageBase64: RESOURCES["demoBuildingBlueprint.png"], + tutorialImageBase64: RESOURCES["demoBuildingBlueprint.png"], + }, + ], + }); + + // Add it to the regular toolbar + this.modLoader.signals.hudElementInitialized.add(element => { + if (element.constructor.name === "HUDBuildingsToolbar") { + // @ts-ignore + element.primaryBuildings.push(MetaDemoModBuilding); + } + }); } } + +//////////////////////////////////////////////////////////////////////// +// @notice: Later this part will be autogenerated + +const RESOURCES = { + "red.png": + "", + + "demoBuilding.png": + "", + + "demoBuildingBlueprint.png": + "", +}; diff --git a/src/js/mods/mod_interface.js b/src/js/mods/mod_interface.js index eaf1d314..fae5f2ba 100644 --- a/src/js/mods/mod_interface.js +++ b/src/js/mods/mod_interface.js @@ -1,14 +1,17 @@ /* typehints:start */ -import { Application } from "../application"; import { ModLoader } from "./modloader"; +import { MetaBuilding } from "../game/meta_building"; /* typehints:end */ +import { defaultBuildingVariant } from "../game/meta_building"; import { createLogger } from "../core/logging"; import { AtlasSprite, SpriteAtlasLink } from "../core/sprites"; import { enumShortcodeToSubShape, enumSubShape, enumSubShapeToShortcode } from "../game/shape_definition"; import { Loader } from "../core/loader"; import { LANGUAGES } from "../languages"; import { matchDataRecursive, T } from "../translations"; +import { registerBuildingVariant } from "../game/building_codes"; +import { gMetaBuildingRegistry } from "../core/global_registries"; const LOG = createLogger("mod-interface"); @@ -113,4 +116,77 @@ export class ModInterface { matchDataRecursive(T, translations, true); } } + + /** + * + * @param {object} param0 + * @param {typeof MetaBuilding} param0.metaClass + * @param {string=} param0.buildingIconBase64 + * @param {({ + * variant?: string; + * rotationVariant?: number; + * name: string; + * description: string; + * blueprintImageBase64?: string; + * regularImageBase64?: string; + * tutorialImageBase64?: string; + * }[])} param0.variantsAndRotations + */ + registerNewBuilding({ metaClass, variantsAndRotations, buildingIconBase64 }) { + const id = new /** @type {new () => MetaBuilding} */ (metaClass)().getId(); + if (gMetaBuildingRegistry.hasId(id)) { + throw new Error("Tried to register building twice: " + id); + } + gMetaBuildingRegistry.register(metaClass); + + T.buildings[id] = {}; + variantsAndRotations.forEach(payload => { + const actualVariant = payload.variant || defaultBuildingVariant; + registerBuildingVariant(id, metaClass, actualVariant, payload.rotationVariant || 0); + T.buildings[id][actualVariant] = { + name: payload.name, + description: payload.description, + }; + + const buildingIdentifier = + id + (actualVariant === defaultBuildingVariant ? "" : "-" + actualVariant); + if (payload.regularImageBase64) { + this.registerSprite( + "sprites/buildings/" + buildingIdentifier + ".png", + payload.regularImageBase64 + ); + } + if (payload.blueprintImageBase64) { + this.registerSprite( + "sprites/blueprints/" + buildingIdentifier + ".png", + payload.blueprintImageBase64 + ); + } + if (payload.tutorialImageBase64) { + this.setBuildingTutorialImage(id, actualVariant, payload.tutorialImageBase64); + } + }); + + if (buildingIconBase64) { + this.setBuildingToolbarIcon(id, buildingIconBase64); + } + } + + setBuildingToolbarIcon(buildingId, iconBase64) { + this.registerCss(` + [data-icon="building_icons/${buildingId}.png"] .icon { + background-image: url('${iconBase64}') !important; + } + `); + } + + setBuildingTutorialImage(buildingId, variant, imageBase64) { + const buildingIdentifier = buildingId + (variant === defaultBuildingVariant ? "" : "-" + variant); + + this.registerCss(` + [data-icon="building_tutorials/${buildingIdentifier}.png"] { + background-image: url('${imageBase64}') !important; + } + `); + } } diff --git a/src/js/mods/modloader.js b/src/js/mods/modloader.js index 1a903ae2..75115dd7 100644 --- a/src/js/mods/modloader.js +++ b/src/js/mods/modloader.js @@ -3,6 +3,7 @@ import { Signal } from "../core/signal"; import { DemoMod } from "./demo_mod"; import { Mod } from "./mod"; import { ModInterface } from "./mod_interface"; +import { BaseHUDPart } from "../game/hud/base_hud_part"; const LOG = createLogger("mods"); @@ -25,6 +26,9 @@ export class ModLoader { injectSprites: new Signal(), preprocessTheme: /** @type {TypedSignal<[Object]>} */ (new Signal()), modifyLevelDefinitions: /** @type {TypedSignal<[Array[Object]]>} */ (new Signal()), + + hudElementInitialized: /** @type {TypedSignal<[BaseHUDPart]>} */ (new Signal()), + hudElementFinalized: /** @type {TypedSignal<[BaseHUDPart]>} */ (new Signal()), }; this.registerMod(DemoMod); From 16ecbf9c6d45a0b1d7466f4ac8d67e805bbcc449 Mon Sep 17 00:00:00 2001 From: tobspr Date: Fri, 14 Jan 2022 07:18:25 +0100 Subject: [PATCH 008/129] Refactor how mods are loaded to resolve circular dependencies and prepare for future mod loading --- src/js/game/hud/hud.js | 6 +- src/js/game/map_chunk.js | 6 +- src/js/game/modes/regular.js | 4 +- src/js/game/shape_definition.js | 14 ++- src/js/game/theme.js | 4 +- src/js/mods/demo_mod.js | 193 ++++++++++++++++---------------- src/js/mods/mod.js | 5 + src/js/mods/mod_interface.js | 28 ++--- src/js/mods/mod_signals.js | 16 +++ src/js/mods/modloader.js | 22 ++-- 10 files changed, 159 insertions(+), 139 deletions(-) create mode 100644 src/js/mods/mod_signals.js diff --git a/src/js/game/hud/hud.js b/src/js/game/hud/hud.js index cffc290d..9fe1ba2f 100644 --- a/src/js/game/hud/hud.js +++ b/src/js/game/hud/hud.js @@ -1,7 +1,7 @@ import { globalConfig } from "../../core/config"; import { DrawParameters } from "../../core/draw_parameters"; import { Signal } from "../../core/signal"; -import { MODS } from "../../mods/modloader"; +import { MOD_SIGNALS } from "../../mods/mod_signals"; import { KEYMAPPINGS } from "../key_action_mapper"; import { MetaBuilding } from "../meta_building"; import { GameRoot } from "../root"; @@ -92,7 +92,7 @@ export class GameHUD { const frag = document.createDocumentFragment(); for (const key in this.parts) { - MODS.signals.hudElementInitialized.dispatch(this.parts[key]); + MOD_SIGNALS.hudElementInitialized.dispatch(this.parts[key]); this.parts[key].createElements(frag); } @@ -100,7 +100,7 @@ export class GameHUD { for (const key in this.parts) { this.parts[key].initialize(); - MODS.signals.hudElementFinalized.dispatch(this.parts[key]); + MOD_SIGNALS.hudElementFinalized.dispatch(this.parts[key]); } this.root.keyMapper.getBinding(KEYMAPPINGS.ingame.toggleHud).add(this.toggleUi, this); diff --git a/src/js/game/map_chunk.js b/src/js/game/map_chunk.js index 0c5390e4..ca2db36f 100644 --- a/src/js/game/map_chunk.js +++ b/src/js/game/map_chunk.js @@ -10,10 +10,14 @@ import { COLOR_ITEM_SINGLETONS } from "./items/color_item"; import { GameRoot } from "./root"; import { enumSubShape } from "./shape_definition"; import { Rectangle } from "../core/rectangle"; -import { MODS_ADDITIONAL_SHAPE_MAP_WEIGHTS } from "../mods/mod_interface"; const logger = createLogger("map_chunk"); +/** + * @type {Object number>} + */ +export const MODS_ADDITIONAL_SHAPE_MAP_WEIGHTS = {}; + export class MapChunk { /** * diff --git a/src/js/game/modes/regular.js b/src/js/game/modes/regular.js index 0aad3669..699874b5 100644 --- a/src/js/game/modes/regular.js +++ b/src/js/game/modes/regular.js @@ -38,7 +38,7 @@ import { HUDSandboxController } from "../hud/parts/sandbox_controller"; import { queryParamOptions } from "../../core/query_parameters"; import { MetaBlockBuilding } from "../buildings/block"; import { MetaItemProducerBuilding } from "../buildings/item_producer"; -import { MODS } from "../../mods/modloader"; +import { MOD_SIGNALS } from "../../mods/mod_signals"; /** @typedef {{ * shape: string, @@ -512,7 +512,7 @@ export function generateLevelDefinitions(limitedVersion = false) { ]), ]; - MODS.signals.modifyLevelDefinitions.dispatch(levelDefinitions); + MOD_SIGNALS.modifyLevelDefinitions.dispatch(levelDefinitions); if (G_IS_DEV) { levelDefinitions.forEach(({ shape }) => { diff --git a/src/js/game/shape_definition.js b/src/js/game/shape_definition.js index acc0bd1c..9cf2d094 100644 --- a/src/js/game/shape_definition.js +++ b/src/js/game/shape_definition.js @@ -3,11 +3,23 @@ import { globalConfig } from "../core/config"; import { smoothenDpi } from "../core/dpi_manager"; import { DrawParameters } from "../core/draw_parameters"; import { Vector } from "../core/vector"; -import { MODS_ADDITIONAL_SUB_SHAPE_DRAWERS } from "../mods/mod_interface"; import { BasicSerializableObject, types } from "../savegame/serialization"; import { enumColors, enumColorsToHexCode, enumColorToShortcode, enumShortcodeToColor } from "./colors"; import { THEME } from "./theme"; +/** + * @typedef {{ + * context: CanvasRenderingContext2D, + * quadrantSize: number, + * layerScale: number, + * }} SubShapeDrawOptions + */ + +/** + * @type {Object void>} + */ +export const MODS_ADDITIONAL_SUB_SHAPE_DRAWERS = {}; + /** * @typedef {{ * subShape: enumSubShape, diff --git a/src/js/game/theme.js b/src/js/game/theme.js index 023710dd..5e371740 100644 --- a/src/js/game/theme.js +++ b/src/js/game/theme.js @@ -1,4 +1,4 @@ -import { MODS } from "../mods/modloader"; +import { MOD_SIGNALS } from "../mods/mod_signals"; export const THEMES = { dark: require("./themes/dark.json"), @@ -9,5 +9,5 @@ export let THEME = THEMES.light; export function applyGameTheme(id) { THEME = THEMES[id]; - MODS.signals.preprocessTheme.dispatch({ id, theme: THEME }); + MOD_SIGNALS.preprocessTheme.dispatch({ id, theme: THEME }); } diff --git a/src/js/mods/demo_mod.js b/src/js/mods/demo_mod.js index 42f48fa3..944cae38 100644 --- a/src/js/mods/demo_mod.js +++ b/src/js/mods/demo_mod.js @@ -2,119 +2,118 @@ import { Entity } from "../game/entity"; /* typehints:end */ -import { Mod } from "./mod"; -import { MetaBuilding } from "../game/meta_building"; +export default function ({ Mod, MetaBuilding }) { + class MetaDemoModBuilding extends MetaBuilding { + constructor() { + super("demoModBuilding"); + } -export class MetaDemoModBuilding extends MetaBuilding { - constructor() { - super("demoModBuilding"); + getSilhouetteColor() { + return "red"; + } + + /** + * Creates the entity at the given location + * @param {Entity} entity + */ + setupEntityComponents(entity) {} } - getSilhouetteColor() { - return "red"; - } + return class ModImpl extends Mod { + constructor(modLoader) { + super( + { + authorContact: "tobias@tobspr.io", + authorName: "tobspr", + name: "Demo Mod", + version: "1", + id: "demo-mod", + }, + modLoader + ); + } - /** - * Creates the entity at the given location - * @param {Entity} entity - */ - setupEntityComponents(entity) {} -} + init() { + // Add some custom css + this.modInterface.registerCss(` + * { + color: red !important; + } + `); -export class DemoMod extends Mod { - constructor(modLoader) { - super( - { - authorContact: "tobias@tobspr.io", - authorName: "tobspr", - name: "Demo Mod", - version: "1", - id: "demo-mod", - }, - modLoader - ); - } + // Replace a builtin sprite + this.modInterface.registerSprite("sprites/colors/red.png", RESOURCES["red.png"]); - init() { - // Add some custom css - this.modLoader.modInterface.registerCss(` - * { - color: red !important; - } - `); + // Add a new type of sub shape ("Line", short code "L") + this.modInterface.registerSubShapeType({ + id: "line", + shortCode: "L", + weightComputation: distanceToOriginInChunks => + Math.round(20 + Math.max(Math.min(distanceToOriginInChunks, 30), 0)), - // Replace a builtin sprite - this.modLoader.modInterface.registerSprite("sprites/colors/red.png", RESOURCES["red.png"]); + draw: ({ context, quadrantSize, layerScale }) => { + const quadrantHalfSize = quadrantSize / 2; + context.beginPath(); + context.moveTo(-quadrantHalfSize, quadrantHalfSize); + context.arc( + -quadrantHalfSize, + quadrantHalfSize, + quadrantSize * layerScale, + -Math.PI * 0.25, + 0 + ); + context.closePath(); + }, + }); - // Add a new type of sub shape ("Line", short code "L") - this.modLoader.modInterface.registerSubShapeType({ - id: "line", - shortCode: "L", - weightComputation: distanceToOriginInChunks => - Math.round(20 + Math.max(Math.min(distanceToOriginInChunks, 30), 0)), + // Modify the theme colors + this.signals.preprocessTheme.add(({ theme }) => { + theme.map.background = "#eee"; + theme.items.outline = "#000"; + }); - draw: ({ context, quadrantSize, layerScale }) => { - const quadrantHalfSize = quadrantSize / 2; - context.beginPath(); - context.moveTo(-quadrantHalfSize, quadrantHalfSize); - context.arc( - -quadrantHalfSize, - quadrantHalfSize, - quadrantSize * layerScale, - -Math.PI * 0.25, - 0 - ); - context.closePath(); - }, - }); + // Modify the goal of the first level + this.signals.modifyLevelDefinitions.add(definitions => { + definitions[0].shape = "LuCuLuCu"; + }); - // Modify the theme colors - this.modLoader.signals.preprocessTheme.add(({ theme }) => { - theme.map.background = "#eee"; - theme.items.outline = "#000"; - }); - - // Modify the goal of the first level - this.modLoader.signals.modifyLevelDefinitions.add(definitions => { - definitions[0].shape = "LuCuLuCu"; - }); - - this.modLoader.modInterface.registerTranslations("en", { - ingame: { - interactiveTutorial: { - title: "Hello", - hints: { - "1_1_extractor": "World!", + this.modInterface.registerTranslations("en", { + ingame: { + interactiveTutorial: { + title: "Hello", + hints: { + "1_1_extractor": "World!", + }, }, }, - }, - }); + }); - // Register the new building - this.modLoader.modInterface.registerNewBuilding({ - metaClass: MetaDemoModBuilding, - buildingIconBase64: RESOURCES["demoBuilding.png"], + // Register the new building + this.modInterface.registerNewBuilding({ + metaClass: MetaDemoModBuilding, + buildingIconBase64: RESOURCES["demoBuilding.png"], - variantsAndRotations: [ - { - description: "A test building", - name: "A test name", + variantsAndRotations: [ + { + description: "A test building", + name: "A test name", - regularImageBase64: RESOURCES["demoBuilding.png"], - blueprintImageBase64: RESOURCES["demoBuildingBlueprint.png"], - tutorialImageBase64: RESOURCES["demoBuildingBlueprint.png"], - }, - ], - }); + regularImageBase64: RESOURCES["demoBuilding.png"], + blueprintImageBase64: RESOURCES["demoBuildingBlueprint.png"], + tutorialImageBase64: RESOURCES["demoBuildingBlueprint.png"], + }, + ], + }); - // Add it to the regular toolbar - this.modLoader.signals.hudElementInitialized.add(element => { - if (element.constructor.name === "HUDBuildingsToolbar") { - // @ts-ignore - element.primaryBuildings.push(MetaDemoModBuilding); - } - }); - } + // Add it to the regular toolbar + this.signals.hudElementInitialized.add(element => { + if (element.constructor.name === "HUDBuildingsToolbar") { + // @ts-ignore + element.primaryBuildings.push(MetaDemoModBuilding); + } + }); + } + }; } //////////////////////////////////////////////////////////////////////// diff --git a/src/js/mods/mod.js b/src/js/mods/mod.js index 65cdd53a..12992999 100644 --- a/src/js/mods/mod.js +++ b/src/js/mods/mod.js @@ -2,6 +2,8 @@ import { ModLoader } from "./modloader"; /* typehints:end */ +import { MOD_SIGNALS } from "./mod_signals"; + export class Mod { /** * @@ -17,6 +19,9 @@ export class Mod { constructor(metadata, modLoader) { this.metadata = metadata; this.modLoader = modLoader; + + this.signals = MOD_SIGNALS; + this.modInterface = modLoader.modInterface; } init() {} diff --git a/src/js/mods/mod_interface.js b/src/js/mods/mod_interface.js index fae5f2ba..5d395819 100644 --- a/src/js/mods/mod_interface.js +++ b/src/js/mods/mod_interface.js @@ -6,33 +6,21 @@ import { MetaBuilding } from "../game/meta_building"; import { defaultBuildingVariant } from "../game/meta_building"; import { createLogger } from "../core/logging"; import { AtlasSprite, SpriteAtlasLink } from "../core/sprites"; -import { enumShortcodeToSubShape, enumSubShape, enumSubShapeToShortcode } from "../game/shape_definition"; +import { + enumShortcodeToSubShape, + enumSubShape, + enumSubShapeToShortcode, + MODS_ADDITIONAL_SUB_SHAPE_DRAWERS, +} from "../game/shape_definition"; import { Loader } from "../core/loader"; import { LANGUAGES } from "../languages"; import { matchDataRecursive, T } from "../translations"; import { registerBuildingVariant } from "../game/building_codes"; import { gMetaBuildingRegistry } from "../core/global_registries"; +import { MODS_ADDITIONAL_SHAPE_MAP_WEIGHTS } from "../game/map_chunk"; const LOG = createLogger("mod-interface"); -/** - * @type {Object number>} - */ -export const MODS_ADDITIONAL_SHAPE_MAP_WEIGHTS = {}; - -/** - * @typedef {{ - * context: CanvasRenderingContext2D, - * quadrantSize: number, - * layerScale: number, - * }} SubShapeDrawOptions - */ - -/** - * @type {Object void>} - */ -export const MODS_ADDITIONAL_SUB_SHAPE_DRAWERS = {}; - export class ModInterface { /** * @@ -91,7 +79,7 @@ export class ModInterface { * @param {string} param0.id * @param {string} param0.shortCode * @param {(distanceToOriginInChunks: number) => number} param0.weightComputation - * @param {(options: SubShapeDrawOptions) => void} param0.draw + * @param {(options: import("../game/shape_definition").SubShapeDrawOptions) => void} param0.draw */ registerSubShapeType({ id, shortCode, weightComputation, draw }) { if (shortCode.length !== 1) { diff --git a/src/js/mods/mod_signals.js b/src/js/mods/mod_signals.js new file mode 100644 index 00000000..a2bc0380 --- /dev/null +++ b/src/js/mods/mod_signals.js @@ -0,0 +1,16 @@ +import { Signal } from "../core/signal"; +/* typehints:start */ +import { BaseHUDPart } from "../game/hud/base_hud_part"; +/* typehints:end */ + +// Single file to avoid circular deps + +export const MOD_SIGNALS = { + postInit: new Signal(), + injectSprites: new Signal(), + preprocessTheme: /** @type {TypedSignal<[Object]>} */ (new Signal()), + modifyLevelDefinitions: /** @type {TypedSignal<[Array[Object]]>} */ (new Signal()), + + hudElementInitialized: /** @type {TypedSignal<[BaseHUDPart]>} */ (new Signal()), + hudElementFinalized: /** @type {TypedSignal<[BaseHUDPart]>} */ (new Signal()), +}; diff --git a/src/js/mods/modloader.js b/src/js/mods/modloader.js index 75115dd7..49e30662 100644 --- a/src/js/mods/modloader.js +++ b/src/js/mods/modloader.js @@ -1,9 +1,8 @@ import { createLogger } from "../core/logging"; -import { Signal } from "../core/signal"; -import { DemoMod } from "./demo_mod"; import { Mod } from "./mod"; import { ModInterface } from "./mod_interface"; -import { BaseHUDPart } from "../game/hud/base_hud_part"; +import { MetaBuilding } from "../game/meta_building"; +import { MOD_SIGNALS } from "./mod_signals"; const LOG = createLogger("mods"); @@ -21,17 +20,14 @@ export class ModLoader { 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.signals = MOD_SIGNALS; - hudElementInitialized: /** @type {TypedSignal<[BaseHUDPart]>} */ (new Signal()), - hudElementFinalized: /** @type {TypedSignal<[BaseHUDPart]>} */ (new Signal()), - }; - - this.registerMod(DemoMod); + this.registerMod( + /** @type {any} */ (require("./demo_mod").default({ + Mod, + MetaBuilding, + })) + ); this.initMods(); } From bc5eff84f6f4266f9032898dee897699e81fd6be Mon Sep 17 00:00:00 2001 From: tobspr Date: Fri, 14 Jan 2022 07:37:26 +0100 Subject: [PATCH 009/129] Lazy Load mods to make sure all dependencies are loaded --- src/js/application.js | 6 ++++++ src/js/game/modes/regular.js | 30 ++++++++++++++++++------------ src/js/mods/demo_mod.js | 5 +++-- src/js/mods/mod.js | 5 ++++- src/js/mods/mod_interface.js | 6 +++++- src/js/mods/mod_signals.js | 3 +++ src/js/mods/modloader.js | 17 +++++++---------- 7 files changed, 46 insertions(+), 26 deletions(-) diff --git a/src/js/application.js b/src/js/application.js index c49b7027..66e9eb8c 100644 --- a/src/js/application.js +++ b/src/js/application.js @@ -35,6 +35,8 @@ import { PuzzleMenuState } from "./states/puzzle_menu"; import { ClientAPI } from "./platform/api"; import { LoginState } from "./states/login"; import { WegameSplashState } from "./states/wegame_splash"; +import { MODS } from "./mods/modloader"; +import { MOD_SIGNALS } from "./mods/mod_signals"; /** * @typedef {import("./platform/achievement_provider").AchievementProviderInterface} AchievementProviderInterface @@ -128,6 +130,8 @@ export class Application { // Store the mouse position, or null if not available /** @type {Vector|null} */ this.mousePosition = null; + + MODS.initMods(); } /** @@ -148,6 +152,8 @@ export class Application { this.analytics = new GoogleAnalyticsImpl(this); this.gameAnalytics = new ShapezGameAnalytics(this); this.achievementProvider = new NoAchievementProvider(this); + + MOD_SIGNALS.platformInstancesInitialized.dispatch(); } /** diff --git a/src/js/game/modes/regular.js b/src/js/game/modes/regular.js index 699874b5..b52cbc89 100644 --- a/src/js/game/modes/regular.js +++ b/src/js/game/modes/regular.js @@ -69,10 +69,16 @@ const tierGrowth = 2.5; const chinaShapes = G_WEGAME_VERSION || G_CHINA_VERSION; +const upgradesCache = {}; + /** * Generates all upgrades * @returns {Object} */ function generateUpgrades(limitedVersion = false) { + if (upgradesCache[limitedVersion]) { + return upgradesCache[limitedVersion]; + } + const fixedImprovements = [0.5, 0.5, 1, 1, 2, 1, 1]; const numEndgameUpgrades = limitedVersion ? 0 : 1000 - fixedImprovements.length - 1; @@ -265,6 +271,8 @@ function generateUpgrades(limitedVersion = false) { } } + MOD_SIGNALS.modifyUpgrades.dispatch(upgrades); + // VALIDATE if (G_IS_DEV) { for (const upgradeId in upgrades) { @@ -280,14 +288,20 @@ function generateUpgrades(limitedVersion = false) { } } + upgradesCache[limitedVersion] = upgrades; return upgrades; } +const levelDefinitionsCache = {}; + /** * Generates the level definitions * @param {boolean} limitedVersion */ export function generateLevelDefinitions(limitedVersion = false) { + if (levelDefinitionsCache[limitedVersion]) { + return levelDefinitionsCache[limitedVersion]; + } const levelDefinitions = [ // 1 // Circle @@ -524,15 +538,11 @@ export function generateLevelDefinitions(limitedVersion = false) { }); } + levelDefinitionsCache[limitedVersion] = levelDefinitions; + return levelDefinitions; } -const fullVersionUpgrades = generateUpgrades(false); -const demoVersionUpgrades = generateUpgrades(true); - -const fullVersionLevels = generateLevelDefinitions(false); -const demoVersionLevels = generateLevelDefinitions(true); - export class RegularGameMode extends GameMode { static getId() { return enumGameModeIds.regular; @@ -603,9 +613,7 @@ export class RegularGameMode extends GameMode { * @returns {Object} */ getUpgrades() { - return this.root.app.restrictionMgr.getHasExtendedUpgrades() - ? fullVersionUpgrades - : demoVersionUpgrades; + return generateUpgrades(!this.root.app.restrictionMgr.getHasExtendedUpgrades()); } /** @@ -613,9 +621,7 @@ export class RegularGameMode extends GameMode { * @returns {Array} */ getLevelDefinitions() { - return this.root.app.restrictionMgr.getHasExtendedLevelsAndFreeplay() - ? fullVersionLevels - : demoVersionLevels; + return generateLevelDefinitions(!this.root.app.restrictionMgr.getHasExtendedLevelsAndFreeplay()); } /** diff --git a/src/js/mods/demo_mod.js b/src/js/mods/demo_mod.js index 944cae38..e11c3667 100644 --- a/src/js/mods/demo_mod.js +++ b/src/js/mods/demo_mod.js @@ -20,8 +20,9 @@ export default function ({ Mod, MetaBuilding }) { } return class ModImpl extends Mod { - constructor(modLoader) { + constructor(app, modLoader) { super( + app, { authorContact: "tobias@tobspr.io", authorName: "tobspr", @@ -37,7 +38,7 @@ export default function ({ Mod, MetaBuilding }) { // Add some custom css this.modInterface.registerCss(` * { - color: red !important; + font-family: "Comic Sans", "Comic Sans MS", Tahoma !important; } `); diff --git a/src/js/mods/mod.js b/src/js/mods/mod.js index 12992999..b9018b68 100644 --- a/src/js/mods/mod.js +++ b/src/js/mods/mod.js @@ -1,4 +1,5 @@ /* typehints:start */ +import { Application } from "../application"; import { ModLoader } from "./modloader"; /* typehints:end */ @@ -7,6 +8,7 @@ import { MOD_SIGNALS } from "./mod_signals"; export class Mod { /** * + * @param {Application} app * @param {object} metadata * @param {string} metadata.name * @param {string} metadata.version @@ -16,7 +18,8 @@ export class Mod { * * @param {ModLoader} modLoader */ - constructor(metadata, modLoader) { + constructor(app, metadata, modLoader) { + this.app = app; this.metadata = metadata; this.modLoader = modLoader; diff --git a/src/js/mods/mod_interface.js b/src/js/mods/mod_interface.js index 5d395819..64c4971e 100644 --- a/src/js/mods/mod_interface.js +++ b/src/js/mods/mod_interface.js @@ -15,7 +15,7 @@ import { import { Loader } from "../core/loader"; import { LANGUAGES } from "../languages"; import { matchDataRecursive, T } from "../translations"; -import { registerBuildingVariant } from "../game/building_codes"; +import { gBuildingVariants, registerBuildingVariant } from "../game/building_codes"; import { gMetaBuildingRegistry } from "../core/global_registries"; import { MODS_ADDITIONAL_SHAPE_MAP_WEIGHTS } from "../game/map_chunk"; @@ -126,11 +126,15 @@ export class ModInterface { throw new Error("Tried to register building twice: " + id); } gMetaBuildingRegistry.register(metaClass); + const metaInstance = gMetaBuildingRegistry.findByClass(metaClass); T.buildings[id] = {}; variantsAndRotations.forEach(payload => { const actualVariant = payload.variant || defaultBuildingVariant; registerBuildingVariant(id, metaClass, actualVariant, payload.rotationVariant || 0); + + gBuildingVariants[id].metaInstance = metaInstance; + T.buildings[id][actualVariant] = { name: payload.name, description: payload.description, diff --git a/src/js/mods/mod_signals.js b/src/js/mods/mod_signals.js index a2bc0380..23670d8b 100644 --- a/src/js/mods/mod_signals.js +++ b/src/js/mods/mod_signals.js @@ -7,9 +7,12 @@ import { BaseHUDPart } from "../game/hud/base_hud_part"; export const MOD_SIGNALS = { postInit: new Signal(), + platformInstancesInitialized: new Signal(), + injectSprites: new Signal(), preprocessTheme: /** @type {TypedSignal<[Object]>} */ (new Signal()), modifyLevelDefinitions: /** @type {TypedSignal<[Array[Object]]>} */ (new Signal()), + modifyUpgrades: /** @type {TypedSignal<[Object]>} */ (new Signal()), hudElementInitialized: /** @type {TypedSignal<[BaseHUDPart]>} */ (new Signal()), hudElementFinalized: /** @type {TypedSignal<[BaseHUDPart]>} */ (new Signal()), diff --git a/src/js/mods/modloader.js b/src/js/mods/modloader.js index 49e30662..8edc2d3d 100644 --- a/src/js/mods/modloader.js +++ b/src/js/mods/modloader.js @@ -15,20 +15,14 @@ export class ModLoader { this.modInterface = new ModInterface(this); - /** @type {(new (ModLoader) => Mod)[]} */ + /** @type {((Object) => (new (Application, ModLoader) => Mod))[]} */ this.modLoadQueue = []; this.initialized = false; this.signals = MOD_SIGNALS; - this.registerMod( - /** @type {any} */ (require("./demo_mod").default({ - Mod, - MetaBuilding, - })) - ); - this.initMods(); + this.registerMod(/** @type {any} */ (require("./demo_mod").default)); } linkApp(app) { @@ -39,7 +33,10 @@ export class ModLoader { LOG.log("hook:init"); this.initialized = true; this.modLoadQueue.forEach(modClass => { - const mod = new modClass(this); + const mod = new (modClass({ + Mod, + MetaBuilding, + }))(this.app, this); mod.init(); this.mods.push(mod); }); @@ -49,7 +46,7 @@ export class ModLoader { /** * - * @param {new (ModLoader) => Mod} mod + * @param {(Object) => (new (Application, ModLoader) => Mod)} mod */ registerMod(mod) { if (this.initialized) { From a6b024be256c4e6dede72010089e2115ac0c8e8b Mon Sep 17 00:00:00 2001 From: tobspr Date: Fri, 14 Jan 2022 07:51:48 +0100 Subject: [PATCH 010/129] Expose all exported members automatically to mods --- src/js/mods/demo_mod.js | 6 +++--- src/js/mods/modloader.js | 33 +++++++++++++++++++++++++++++---- 2 files changed, 32 insertions(+), 7 deletions(-) diff --git a/src/js/mods/demo_mod.js b/src/js/mods/demo_mod.js index e11c3667..43ceffd4 100644 --- a/src/js/mods/demo_mod.js +++ b/src/js/mods/demo_mod.js @@ -2,8 +2,8 @@ import { Entity } from "../game/entity"; /* typehints:end */ -export default function ({ Mod, MetaBuilding }) { - class MetaDemoModBuilding extends MetaBuilding { +export default function (shapez) { + class MetaDemoModBuilding extends shapez.MetaBuilding { constructor() { super("demoModBuilding"); } @@ -19,7 +19,7 @@ export default function ({ Mod, MetaBuilding }) { setupEntityComponents(entity) {} } - return class ModImpl extends Mod { + return class ModImpl extends shapez.Mod { constructor(app, modLoader) { super( app, diff --git a/src/js/mods/modloader.js b/src/js/mods/modloader.js index 8edc2d3d..5732b083 100644 --- a/src/js/mods/modloader.js +++ b/src/js/mods/modloader.js @@ -31,12 +31,37 @@ export class ModLoader { initMods() { LOG.log("hook:init"); + + let exports = {}; + + if (G_IS_DEV || G_IS_STANDALONE) { + const modules = require.context("../", true, /\.js$/); + + Array.from(modules.keys()).forEach(key => { + // @ts-ignore + const module = modules(key); + for (const member in module) { + if (member === "default") { + continue; + } + if (exports[member]) { + continue; + } + Object.defineProperty(exports, member, { + get() { + return module[member]; + }, + set(v) { + module[member] = v; + }, + }); + } + }); + } + this.initialized = true; this.modLoadQueue.forEach(modClass => { - const mod = new (modClass({ - Mod, - MetaBuilding, - }))(this.app, this); + const mod = new (modClass(exports))(this.app, this); mod.init(); this.mods.push(mod); }); From 258dbbd98f84bea5ca4815ab80d861f82cd54415 Mon Sep 17 00:00:00 2001 From: tobspr Date: Fri, 14 Jan 2022 07:53:32 +0100 Subject: [PATCH 011/129] Fix duplicate exports --- src/js/game/buildings/logic_gate.js | 2 +- src/js/game/buildings/virtual_processor.js | 2 +- src/js/game/map_chunk_aggregate.js | 3 +-- src/js/mods/modloader.js | 3 +-- 4 files changed, 4 insertions(+), 6 deletions(-) diff --git a/src/js/game/buildings/logic_gate.js b/src/js/game/buildings/logic_gate.js index b61d4373..a0e63c06 100644 --- a/src/js/game/buildings/logic_gate.js +++ b/src/js/game/buildings/logic_gate.js @@ -15,7 +15,7 @@ export const enumLogicGateVariants = { }; /** @enum {string} */ -export const enumVariantToGate = { +const enumVariantToGate = { [defaultBuildingVariant]: enumLogicGateType.and, [enumLogicGateVariants.not]: enumLogicGateType.not, [enumLogicGateVariants.xor]: enumLogicGateType.xor, diff --git a/src/js/game/buildings/virtual_processor.js b/src/js/game/buildings/virtual_processor.js index b4f91762..ab32a0d3 100644 --- a/src/js/game/buildings/virtual_processor.js +++ b/src/js/game/buildings/virtual_processor.js @@ -19,7 +19,7 @@ export const enumVirtualProcessorVariants = { }; /** @enum {string} */ -export const enumVariantToGate = { +const enumVariantToGate = { [defaultBuildingVariant]: enumLogicGateType.cutter, [enumVirtualProcessorVariants.rotater]: enumLogicGateType.rotater, [enumVirtualProcessorVariants.unstacker]: enumLogicGateType.unstacker, diff --git a/src/js/game/map_chunk_aggregate.js b/src/js/game/map_chunk_aggregate.js index de15362d..f47ed676 100644 --- a/src/js/game/map_chunk_aggregate.js +++ b/src/js/game/map_chunk_aggregate.js @@ -2,10 +2,9 @@ import { globalConfig } from "../core/config"; import { DrawParameters } from "../core/draw_parameters"; import { drawSpriteClipped } from "../core/draw_utils"; import { safeModulo } from "../core/utils"; +import { CHUNK_OVERLAY_RES } from "./map_chunk_view"; import { GameRoot } from "./root"; -export const CHUNK_OVERLAY_RES = 3; - export class MapChunkAggregate { /** * diff --git a/src/js/mods/modloader.js b/src/js/mods/modloader.js index 5732b083..beb9e5e1 100644 --- a/src/js/mods/modloader.js +++ b/src/js/mods/modloader.js @@ -1,7 +1,6 @@ import { createLogger } from "../core/logging"; import { Mod } from "./mod"; import { ModInterface } from "./mod_interface"; -import { MetaBuilding } from "../game/meta_building"; import { MOD_SIGNALS } from "./mod_signals"; const LOG = createLogger("mods"); @@ -45,7 +44,7 @@ export class ModLoader { continue; } if (exports[member]) { - continue; + throw new Error("Duplicate export of " + member); } Object.defineProperty(exports, member, { get() { From 575abd255d39fc78ddcff1e37d53d9a0e48c12b5 Mon Sep 17 00:00:00 2001 From: tobspr Date: Fri, 14 Jan 2022 08:55:18 +0100 Subject: [PATCH 012/129] Allow loading mods from standalone --- electron/.gitignore | 1 + electron/index.js | 27 ++++++++++ electron/mods/README.txt | 6 +++ src/css/states/main_menu.scss | 69 ++++++++++++++++++++++++- src/js/application.js | 7 +-- src/js/core/config.local.template.js | 3 ++ src/js/mods/{demo_mod.js => dev_mod.js} | 18 ++----- src/js/mods/modloader.js | 41 ++++++++++----- src/js/states/main_menu.js | 61 ++++++++++++++++++++-- translations/base-en.yaml | 10 ++++ 10 files changed, 209 insertions(+), 34 deletions(-) create mode 100644 electron/.gitignore create mode 100644 electron/mods/README.txt rename src/js/mods/{demo_mod.js => dev_mod.js} (98%) diff --git a/electron/.gitignore b/electron/.gitignore new file mode 100644 index 00000000..0cdb30f4 --- /dev/null +++ b/electron/.gitignore @@ -0,0 +1 @@ +mods/*.js \ No newline at end of file diff --git a/electron/index.js b/electron/index.js index e7994050..e6f9ce1e 100644 --- a/electron/index.js +++ b/electron/index.js @@ -9,6 +9,7 @@ const asyncLock = require("async-lock"); const isDev = process.argv.indexOf("--dev") >= 0; const isLocal = process.argv.indexOf("--local") >= 0; +const safeMode = process.argv.indexOf("--safe-mode") >= 0; const roamingFolder = process.env.APPDATA || @@ -16,6 +17,7 @@ const roamingFolder = ? process.env.HOME + "/Library/Preferences" : process.env.HOME + "/.local/share"); let storePath = path.join(roamingFolder, "shapez.io", "saves"); +let modsPath = path.join(app.getAppPath(), "mods"); if (!fs.existsSync(storePath)) { // No try-catch by design @@ -79,6 +81,8 @@ function createWindow() { if (isDev) { menu = new Menu(); + win.toggleDevTools(); + const mainItem = new MenuItem({ label: "Toggle Dev Tools", click: () => win.toggleDevTools(), @@ -279,5 +283,28 @@ ipcMain.on("fs-job", async (event, arg) => { event.reply("fs-response", { id: arg.id, result }); }); +ipcMain.on("open-mods-folder", async () => { + shell.openPath(modsPath); +}); + +ipcMain.handle("get-mods", async (event, arg) => { + if (safeMode) { + console.warn("Not loading mods due to safe mode"); + return []; + } + if (!fs.existsSync(modsPath)) { + console.warn("Mods folder not found:", modsPath); + return []; + } + try { + console.log("Loading mods from", modsPath); + let entries = fs.readdirSync(modsPath); + entries = entries.filter(entry => entry.endsWith(".js")); + return entries.map(filename => fs.readFileSync(path.join(modsPath, filename), { encoding: "utf8" })); + } catch (ex) { + throw new Error(ex); + } +}); + steam.init(isDev); steam.listen(); diff --git a/electron/mods/README.txt b/electron/mods/README.txt new file mode 100644 index 00000000..666cc18f --- /dev/null +++ b/electron/mods/README.txt @@ -0,0 +1,6 @@ +Here you can place mods. Every mod should be a single file ending with ".js". + +--- WARNING --- +Mods can potentially access to your filesystem. +Please only install mods from trusted sources and developers. +--- WARNING --- diff --git a/src/css/states/main_menu.scss b/src/css/states/main_menu.scss index 15cdbe1c..2c70f5d8 100644 --- a/src/css/states/main_menu.scss +++ b/src/css/states/main_menu.scss @@ -185,7 +185,7 @@ .updateLabel { position: absolute; transform: translateX(50%) rotate(-5deg); - color: #ff590b; + color: #300bff; @include Heading; font-weight: bold; @include S(right, 40px); @@ -290,6 +290,73 @@ } } + .modsList { + display: flex; + align-items: center; + justify-content: center; + flex-direction: column; + background: #fff; + grid-row: 1 / 2; + grid-column: 2 / 3; + position: relative; + text-align: left; + align-items: flex-start; + @include S(padding, 15px); + @include S(padding-bottom, 10px); + @include S(border-radius, $globalBorderRadius); + + h3 { + @include S(margin-bottom, 10px); + @include Heading; + } + + .dlcHint { + @include SuperSmallText; + @include S(margin-top, 10px); + + display: grid; + grid-template-columns: 1fr auto; + grid-gap: 20px; + align-items: center; + } + + .mod { + background: #eee; + width: 100%; + @include S(border-radius, $globalBorderRadius); + @include S(padding, 5px); + box-sizing: border-box; + @include S(margin-bottom, 5px); + display: grid; + grid-template-columns: 1fr auto auto; + @include S(grid-gap, 5px); + + .author, + .version { + @include SuperSmallText; + } + .name { + overflow: hidden; + } + } + + .modsList { + box-sizing: border-box; + @include S(height, 100px); + @include S(padding, 5px); + border: D(1px) solid #eee; + overflow-y: scroll; + width: 100%; + display: flex; + flex-direction: column; + pointer-events: all; + + :last-child { + margin-bottom: auto; + } + } + } + .mainContainer { display: flex; align-items: center; diff --git a/src/js/application.js b/src/js/application.js index 66e9eb8c..67e9c353 100644 --- a/src/js/application.js +++ b/src/js/application.js @@ -130,8 +130,6 @@ export class Application { // Store the mouse position, or null if not available /** @type {Vector|null} */ this.mousePosition = null; - - MODS.initMods(); } /** @@ -331,8 +329,11 @@ export class Application { /** * Boots the application */ - boot() { + async boot() { console.log("Booting ..."); + + await MODS.initMods(); + this.registerStates(); this.registerEventListeners(); diff --git a/src/js/core/config.local.template.js b/src/js/core/config.local.template.js index 41677997..ed230ae4 100644 --- a/src/js/core/config.local.template.js +++ b/src/js/core/config.local.template.js @@ -116,5 +116,8 @@ export default { // Disables slow asserts, useful for debugging performance // disableSlowAsserts: true, // ----------------------------------------------------------------------------------- + // Loads the dev_mod.js for developing new mods + // loadDevMod: true, + // ----------------------------------------------------------------------------------- /* dev:end */ }; diff --git a/src/js/mods/demo_mod.js b/src/js/mods/dev_mod.js similarity index 98% rename from src/js/mods/demo_mod.js rename to src/js/mods/dev_mod.js index 43ceffd4..aa74f59a 100644 --- a/src/js/mods/demo_mod.js +++ b/src/js/mods/dev_mod.js @@ -1,7 +1,3 @@ -/* typehints:start */ -import { Entity } from "../game/entity"; -/* typehints:end */ - export default function (shapez) { class MetaDemoModBuilding extends shapez.MetaBuilding { constructor() { @@ -12,10 +8,6 @@ export default function (shapez) { return "red"; } - /** - * Creates the entity at the given location - * @param {Entity} entity - */ setupEntityComponents(entity) {} } @@ -36,11 +28,11 @@ export default function (shapez) { init() { // Add some custom css - this.modInterface.registerCss(` - * { - font-family: "Comic Sans", "Comic Sans MS", Tahoma !important; - } - `); + // this.modInterface.registerCss(` + // button { + // font-family: "Comic Sans", "Comic Sans MS", Tahoma !important; + // } + // `); // Replace a builtin sprite this.modInterface.registerSprite("sprites/colors/red.png", RESOURCES["red.png"]); diff --git a/src/js/mods/modloader.js b/src/js/mods/modloader.js index beb9e5e1..5c308d8a 100644 --- a/src/js/mods/modloader.js +++ b/src/js/mods/modloader.js @@ -1,4 +1,6 @@ +import { globalConfig } from "../core/config"; import { createLogger } from "../core/logging"; +import { getIPCRenderer } from "../core/utils"; import { Mod } from "./mod"; import { ModInterface } from "./mod_interface"; import { MOD_SIGNALS } from "./mod_signals"; @@ -20,17 +22,39 @@ export class ModLoader { this.initialized = false; this.signals = MOD_SIGNALS; - - this.registerMod(/** @type {any} */ (require("./demo_mod").default)); } linkApp(app) { this.app = app; } - initMods() { + anyModsActive() { + return this.mods.length > 0; + } + + async initMods() { LOG.log("hook:init"); + if (G_IS_STANDALONE) { + try { + const mods = await getIPCRenderer().invoke("get-mods"); + + mods.forEach(modCode => { + const registerMod = mod => { + this.modLoadQueue.push(mod); + }; + // ugh + eval(modCode); + }); + } catch (ex) { + alert("Failed to load mods: " + ex); + } + } else if (G_IS_DEV) { + if (globalConfig.debug.loadDevMod) { + this.modLoadQueue.push(/** @type {any} */ (require("./dev_mod").default)); + } + } + let exports = {}; if (G_IS_DEV || G_IS_STANDALONE) { @@ -67,17 +91,6 @@ export class ModLoader { this.modLoadQueue = []; this.signals.postInit.dispatch(); } - - /** - * - * @param {(Object) => (new (Application, ModLoader) => Mod)} mod - */ - registerMod(mod) { - if (this.initialized) { - throw new Error("Mods are already initialized, can not add mod afterwards."); - } - this.modLoadQueue.push(mod); - } } export const MODS = new ModLoader(); diff --git a/src/js/states/main_menu.js b/src/js/states/main_menu.js index 60495a9c..a6bb940a 100644 --- a/src/js/states/main_menu.js +++ b/src/js/states/main_menu.js @@ -8,6 +8,7 @@ import { ReadWriteProxy } from "../core/read_write_proxy"; import { formatSecondsToTimeAgo, generateFileDownload, + getIPCRenderer, isSupportedBrowser, makeButton, makeButtonElement, @@ -17,6 +18,7 @@ import { waitNextFrame, } from "../core/utils"; import { HUDModalDialogs } from "../game/hud/parts/modal_dialogs"; +import { MODS } from "../mods/modloader"; import { PlatformWrapperImplBrowser } from "../platform/browser/wrapper"; import { PlatformWrapperImplElectron } from "../platform/electron/wrapper"; import { getApplicationSettingById } from "../profile/application_settings"; @@ -41,6 +43,7 @@ export class MainMenuState extends GameState { const showBrowserWarning = !G_IS_STANDALONE && !isSupportedBrowser(); const showPuzzleDLC = !G_WEGAME_VERSION && (G_IS_STANDALONE || G_IS_DEV); const showWegameFooter = G_WEGAME_VERSION; + const hasMods = MODS.anyModsActive(); let showExternalLinks = true; @@ -94,7 +97,7 @@ export class MainMenuState extends GameState {
@@ -112,7 +115,7 @@ export class MainMenuState extends GameState {
${ - showPuzzleDLC && ownsPuzzleDLC + showPuzzleDLC && ownsPuzzleDLC && !hasMods ? `
@@ -147,6 +150,49 @@ export class MainMenuState extends GameState {
` : "" } + ${ + hasMods + ? ` + +
+

${T.mainMenu.mods.title} + +

+ +
+ ${MODS.mods + .map(mod => { + return ` +
+ ${mod.metadata.name} + ${T.mainMenu.mods.version.replace( + "", + mod.metadata.version + )} + ${T.mainMenu.mods.author.replace( + "", + mod.metadata.authorName + )} +
+ `; + }) + .join("")} +
+ +
+ ${T.mainMenu.modsWarningPuzzleDLC} + + +
+ + +
+ ` + : "" + } + ${ @@ -335,6 +381,7 @@ export class MainMenuState extends GameState { ".puzzleDlcPlayButton": this.onPuzzleModeButtonClicked, ".puzzleDlcGetButton": this.onPuzzleWishlistButtonClicked, ".wegameDisclaimer > .rating": this.onWegameRatingClicked, + ".modsOpenFolder": this.openModsFolder, }; for (const key in clickHandling) { @@ -716,6 +763,14 @@ export class MainMenuState extends GameState { }); } + openModsFolder() { + if (!G_IS_STANDALONE) { + this.dialogs.showWarning(T.global.error, T.mainMenu.mods.folderOnlyStandalone); + return; + } + getIPCRenderer().send("open-mods-folder"); + } + onLeave() { this.dialogs.cleanup(); } diff --git a/translations/base-en.yaml b/translations/base-en.yaml index 3f3b1412..a7a27bfe 100644 --- a/translations/base-en.yaml +++ b/translations/base-en.yaml @@ -126,6 +126,16 @@ mainMenu: puzzleDlcWishlist: Wishlist now! puzzleDlcViewNow: View Dlc + modsWarningPuzzleDLC: >- + Playing the Puzzle DLC is not possible with mods. Please disable all mods to play the DLC. + + mods: + title: Active Mods + author: by + version: v + openFolder: Open Folder + folderOnlyStandalone: Opening the mod folder is only possible when running the standalone. + puzzleMenu: play: Play edit: Edit From ed837703487484fca1b887f8c7e904c5ace9aa96 Mon Sep 17 00:00:00 2001 From: tobspr Date: Fri, 14 Jan 2022 09:05:03 +0100 Subject: [PATCH 013/129] update changelog --- src/js/changelog.js | 7 +++++++ version | 2 +- 2 files changed, 8 insertions(+), 1 deletion(-) diff --git a/src/js/changelog.js b/src/js/changelog.js index ec9a317c..a68c8ce4 100644 --- a/src/js/changelog.js +++ b/src/js/changelog.js @@ -1,4 +1,11 @@ export const CHANGELOG = [ + { + version: "1.5.0", + date: "unreleased", + entries: [ + "This version adds an official modloader! You can now load mods by placing it in the mods/ folder of the game.", + ], + }, { version: "1.4.4", date: "29.08.2021", diff --git a/version b/version index e1df5de7..3e1ad720 100644 --- a/version +++ b/version @@ -1 +1 @@ -1.4.4 \ No newline at end of file +1.5.0 \ No newline at end of file From e0e5fb8d2a7670fc7ff848787924f28bb3d36ad8 Mon Sep 17 00:00:00 2001 From: tobspr Date: Fri, 14 Jan 2022 09:55:03 +0100 Subject: [PATCH 014/129] Fix mods folder incorrect path --- electron/index.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/electron/index.js b/electron/index.js index e6f9ce1e..b3f75e72 100644 --- a/electron/index.js +++ b/electron/index.js @@ -17,7 +17,7 @@ const roamingFolder = ? process.env.HOME + "/Library/Preferences" : process.env.HOME + "/.local/share"); let storePath = path.join(roamingFolder, "shapez.io", "saves"); -let modsPath = path.join(app.getAppPath(), "mods"); +let modsPath = path.join(path.dirname(app.getPath("exe")), "mods"); if (!fs.existsSync(storePath)) { // No try-catch by design From 0e86cd3535f6c3612de3f758d9ddaf62f4c1be4a Mon Sep 17 00:00:00 2001 From: tobspr Date: Fri, 14 Jan 2022 10:14:51 +0100 Subject: [PATCH 015/129] Fix modloading in standalone --- src/js/mods/modloader.js | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/src/js/mods/modloader.js b/src/js/mods/modloader.js index 5c308d8a..3db01878 100644 --- a/src/js/mods/modloader.js +++ b/src/js/mods/modloader.js @@ -40,11 +40,12 @@ export class ModLoader { const mods = await getIPCRenderer().invoke("get-mods"); mods.forEach(modCode => { - const registerMod = mod => { + window.registerMod = mod => { this.modLoadQueue.push(mod); }; // ugh eval(modCode); + delete window.registerMod; }); } catch (ex) { alert("Failed to load mods: " + ex); From 38b4e93180cbed10c77520fdb425d9b8cf5f9ad4 Mon Sep 17 00:00:00 2001 From: tobspr Date: Fri, 14 Jan 2022 10:39:26 +0100 Subject: [PATCH 016/129] Fix sprites not getting replaced, update demo mod --- src/js/mods/dev_mod.js | 39 +++++++++++++++++++++++++++++------- src/js/mods/mod_interface.js | 1 + 2 files changed, 33 insertions(+), 7 deletions(-) diff --git a/src/js/mods/dev_mod.js b/src/js/mods/dev_mod.js index aa74f59a..dbe4aa0f 100644 --- a/src/js/mods/dev_mod.js +++ b/src/js/mods/dev_mod.js @@ -27,15 +27,23 @@ export default function (shapez) { } init() { + // Increase belt speed + shapez.globalConfig.beltSpeedItemsPerSecond = 25; + // Add some custom css - // this.modInterface.registerCss(` - // button { - // font-family: "Comic Sans", "Comic Sans MS", Tahoma !important; - // } - // `); + this.modInterface.registerCss(` + * { + font-family: "Comic Sans", "Comic Sans MS", Tahoma !important; + } + `); // Replace a builtin sprite - this.modInterface.registerSprite("sprites/colors/red.png", RESOURCES["red.png"]); + ["red", "green", "blue", "yellow", "purple", "cyan", "white"].forEach(color => { + this.modInterface.registerSprite( + "sprites/colors/" + color + ".png", + RESOURCES[color + ".png"] + ); + }); // Add a new type of sub shape ("Line", short code "L") this.modInterface.registerSubShapeType({ @@ -114,8 +122,25 @@ export default function (shapez) { const RESOURCES = { "red.png": - "", + "", + "green.png": + "", + + "purple.png": + "", + + "blue.png": + "", + + "yellow.png": + "", + + "cyan.png": + "", + + "white.png": + "", "demoBuilding.png": "", diff --git a/src/js/mods/mod_interface.js b/src/js/mods/mod_interface.js index 64c4971e..b0a8fe0c 100644 --- a/src/js/mods/mod_interface.js +++ b/src/js/mods/mod_interface.js @@ -63,6 +63,7 @@ export class ModInterface { sprite.linksByResolution["0.75"] = link; this.lazySprites.set(spriteId, sprite); + Loader.sprites.set(spriteId, sprite); } injectSprites() { From f3b8d329fcc90cf9131824c02f3629d378f3f2b5 Mon Sep 17 00:00:00 2001 From: tobspr Date: Fri, 14 Jan 2022 10:56:33 +0100 Subject: [PATCH 017/129] Load dev mod via raw loader --- gulp/package.json | 1 + gulp/yarn.lock | 46 ++++++++++++++++++++++++++++++++++++++++ src/js/mods/dev_mod.js | 7 ++---- src/js/mods/modloader.js | 14 ++++++------ 4 files changed, 57 insertions(+), 11 deletions(-) diff --git a/gulp/package.json b/gulp/package.json index 2a17b4fd..adc4389f 100644 --- a/gulp/package.json +++ b/gulp/package.json @@ -46,6 +46,7 @@ "postcss": ">=5.0.0", "promise-polyfill": "^8.1.0", "query-string": "^6.8.1", + "raw-loader": "^4.0.2", "rusha": "^0.8.13", "serialize-error": "^3.0.0", "stream-browserify": "^3.0.0", diff --git a/gulp/yarn.lock b/gulp/yarn.lock index f4f3ba7f..93ef55b1 100644 --- a/gulp/yarn.lock +++ b/gulp/yarn.lock @@ -989,6 +989,11 @@ "@types/minimatch" "*" "@types/node" "*" +"@types/json-schema@^7.0.8": + version "7.0.9" + resolved "https://registry.yarnpkg.com/@types/json-schema/-/json-schema-7.0.9.tgz#97edc9037ea0c38585320b28964dde3b39e4660d" + integrity sha512-qcUXuemtEu+E5wZSJHNxUXeCZhAfXKQ41D+duX+VYPde7xyEVZci+/oXKJL13tnRs9lR2pr4fod59GT6/X1/yQ== + "@types/minimatch@*": version "3.0.3" resolved "https://registry.npmjs.org/@types/minimatch/-/minimatch-3.0.3.tgz" @@ -1237,6 +1242,11 @@ ajv-keywords@^3.1.0, ajv-keywords@^3.4.1: resolved "https://registry.npmjs.org/ajv-keywords/-/ajv-keywords-3.4.1.tgz" integrity sha512-RO1ibKvd27e6FEShVFfPALuHI3WjSVNeK5FIsmme/LYRNxjKuNj+Dt7bucLa6NdSv3JcVTyMlm9kGR84z1XpaQ== +ajv-keywords@^3.5.2: + version "3.5.2" + resolved "https://registry.yarnpkg.com/ajv-keywords/-/ajv-keywords-3.5.2.tgz#31f29da5ab6e00d1c2d329acf7b5929614d5014d" + integrity sha512-5p6WTN0DdTGVQk6VjcEju19IgaHudalcfabD7yhDGeA6bcQnmL+CpveLJq/3hvfwd1aof6L386Ougkx6RfyMIQ== + ajv@^4.7.0: version "4.11.8" resolved "https://registry.npmjs.org/ajv/-/ajv-4.11.8.tgz" @@ -1255,6 +1265,16 @@ ajv@^6.1.0, ajv@^6.10.2, ajv@^6.12.0, ajv@^6.5.5, ajv@^6.9.1: json-schema-traverse "^0.4.1" uri-js "^4.2.2" +ajv@^6.12.5: + version "6.12.6" + resolved "https://registry.yarnpkg.com/ajv/-/ajv-6.12.6.tgz#baf5a62e802b07d977034586f8c3baf5adf26df4" + integrity sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g== + dependencies: + fast-deep-equal "^3.1.1" + fast-json-stable-stringify "^2.0.0" + json-schema-traverse "^0.4.1" + uri-js "^4.2.2" + alphanum-sort@^1.0.0: version "1.0.2" resolved "https://registry.npmjs.org/alphanum-sort/-/alphanum-sort-1.0.2.tgz" @@ -7407,6 +7427,15 @@ loader-utils@^1.0.0, loader-utils@^1.1.0, loader-utils@^1.2.3, loader-utils@^1.4 emojis-list "^3.0.0" json5 "^1.0.1" +loader-utils@^2.0.0: + version "2.0.2" + resolved "https://registry.yarnpkg.com/loader-utils/-/loader-utils-2.0.2.tgz#d6e3b4fb81870721ae4e0868ab11dd638368c129" + integrity sha512-TM57VeHptv569d/GKh6TAYdzKblwDNiumOdkFnejjD0XwTH87K90w3O7AiJRqdQoXygvi1VQTJTLGhJl7WqA7A== + dependencies: + big.js "^5.2.2" + emojis-list "^3.0.0" + json5 "^2.1.2" + localtunnel@^2.0.0: version "2.0.0" resolved "https://registry.npmjs.org/localtunnel/-/localtunnel-2.0.0.tgz" @@ -10227,6 +10256,14 @@ raw-body@^2.3.2: iconv-lite "0.4.24" unpipe "1.0.0" +raw-loader@^4.0.2: + version "4.0.2" + resolved "https://registry.yarnpkg.com/raw-loader/-/raw-loader-4.0.2.tgz#1aac6b7d1ad1501e66efdac1522c73e59a584eb6" + integrity sha512-ZnScIV3ag9A4wPX/ZayxL/jZH+euYb6FcUinPcgiQW0+UBtEv0O6Q3lGd3cqJ+GHH+rksEv3Pj99oxJ3u3VIKA== + dependencies: + loader-utils "^2.0.0" + schema-utils "^3.0.0" + rcedit@^2.0.0: version "2.0.0" resolved "https://registry.npmjs.org/rcedit/-/rcedit-2.0.0.tgz" @@ -10869,6 +10906,15 @@ schema-utils@^2.6.5: ajv "^6.12.0" ajv-keywords "^3.4.1" +schema-utils@^3.0.0: + version "3.1.1" + resolved "https://registry.yarnpkg.com/schema-utils/-/schema-utils-3.1.1.tgz#bc74c4b6b6995c1d88f76a8b77bea7219e0c8281" + integrity sha512-Y5PQxS4ITlC+EahLuXaY86TXfR7Dc5lw294alXOq86JAHCihAIZfqv8nNCWvaEJvaC51uN9hbLGeV0cFBdH+Fw== + dependencies: + "@types/json-schema" "^7.0.8" + ajv "^6.12.5" + ajv-keywords "^3.5.2" + seek-bzip@^1.0.5: version "1.0.5" resolved "https://registry.npmjs.org/seek-bzip/-/seek-bzip-1.0.5.tgz" diff --git a/src/js/mods/dev_mod.js b/src/js/mods/dev_mod.js index dbe4aa0f..b636dace 100644 --- a/src/js/mods/dev_mod.js +++ b/src/js/mods/dev_mod.js @@ -1,4 +1,4 @@ -export default function (shapez) { +registerMod(shapez => { class MetaDemoModBuilding extends shapez.MetaBuilding { constructor() { super("demoModBuilding"); @@ -27,9 +27,6 @@ export default function (shapez) { } init() { - // Increase belt speed - shapez.globalConfig.beltSpeedItemsPerSecond = 25; - // Add some custom css this.modInterface.registerCss(` * { @@ -115,7 +112,7 @@ export default function (shapez) { }); } }; -} +}); //////////////////////////////////////////////////////////////////////// // @notice: Later this part will be autogenerated diff --git a/src/js/mods/modloader.js b/src/js/mods/modloader.js index 3db01878..45c70caa 100644 --- a/src/js/mods/modloader.js +++ b/src/js/mods/modloader.js @@ -35,9 +35,15 @@ export class ModLoader { async initMods() { LOG.log("hook:init"); - if (G_IS_STANDALONE) { + if (G_IS_STANDALONE || G_IS_DEV) { try { - const mods = await getIPCRenderer().invoke("get-mods"); + let mods = []; + if (G_IS_STANDALONE) { + mods = await getIPCRenderer().invoke("get-mods"); + } else if (G_IS_DEV && globalConfig.debug.loadDevMod) { + // @ts-expect-error + mods = [require("!!raw-loader!./dev_mod")]; + } mods.forEach(modCode => { window.registerMod = mod => { @@ -50,10 +56,6 @@ export class ModLoader { } catch (ex) { alert("Failed to load mods: " + ex); } - } else if (G_IS_DEV) { - if (globalConfig.debug.loadDevMod) { - this.modLoadQueue.push(/** @type {any} */ (require("./dev_mod").default)); - } } let exports = {}; From 6fa2515d858b3797fa021c72a55f2dded9492e00 Mon Sep 17 00:00:00 2001 From: tobspr Date: Fri, 14 Jan 2022 11:08:16 +0100 Subject: [PATCH 018/129] Improve mod developing so mods are directly ready to be deployed, load mods from local file server --- gulp/gulpfile.js | 2 +- gulp/webpack.config.js | 1 + gulp/webpack.production.config.js | 1 + src/js/mods/{dev_mod.js => demo_mod.nobuild/index.js} | 0 src/js/mods/mod_interface.js | 11 ++++++++++- src/js/mods/modloader.js | 8 ++++++-- 6 files changed, 19 insertions(+), 4 deletions(-) rename src/js/mods/{dev_mod.js => demo_mod.nobuild/index.js} (100%) diff --git a/gulp/gulpfile.js b/gulp/gulpfile.js index 0f4f4185..87901377 100644 --- a/gulp/gulpfile.js +++ b/gulp/gulpfile.js @@ -146,7 +146,7 @@ gulp.task("main.webserver", () => { */ function serve({ version = "web" }) { browserSync.init({ - server: buildFolder, + server: [buildFolder, path.join(baseDir, "src", "js")], port: 3005, ghostMode: { clicks: false, diff --git a/gulp/webpack.config.js b/gulp/webpack.config.js index 14987cfa..7db0bf0b 100644 --- a/gulp/webpack.config.js +++ b/gulp/webpack.config.js @@ -71,6 +71,7 @@ module.exports = ({ watch = false, standalone = false, chineseVersion = false, w type: "javascript/auto", }, { test: /\.(png|jpe?g|svg)$/, loader: "ignore-loader" }, + { test: /\.nobuild/, loader: "ignore-loader" }, { test: /\.md$/, use: [ diff --git a/gulp/webpack.production.config.js b/gulp/webpack.production.config.js index fd7551e0..72bae9f4 100644 --- a/gulp/webpack.production.config.js +++ b/gulp/webpack.production.config.js @@ -177,6 +177,7 @@ module.exports = ({ type: "javascript/auto", }, { test: /\.(png|jpe?g|svg)$/, loader: "ignore-loader" }, + { test: /\.nobuild/, loader: "ignore-loader" }, { test: /\.js$/, enforce: "pre", diff --git a/src/js/mods/dev_mod.js b/src/js/mods/demo_mod.nobuild/index.js similarity index 100% rename from src/js/mods/dev_mod.js rename to src/js/mods/demo_mod.nobuild/index.js diff --git a/src/js/mods/mod_interface.js b/src/js/mods/mod_interface.js index b0a8fe0c..89330d3d 100644 --- a/src/js/mods/mod_interface.js +++ b/src/js/mods/mod_interface.js @@ -1,5 +1,6 @@ /* typehints:start */ import { ModLoader } from "./modloader"; +import { Component } from "../game/component"; import { MetaBuilding } from "../game/meta_building"; /* typehints:end */ @@ -16,7 +17,7 @@ import { Loader } from "../core/loader"; import { LANGUAGES } from "../languages"; import { matchDataRecursive, T } from "../translations"; import { gBuildingVariants, registerBuildingVariant } from "../game/building_codes"; -import { gMetaBuildingRegistry } from "../core/global_registries"; +import { gComponentRegistry, gMetaBuildingRegistry } from "../core/global_registries"; import { MODS_ADDITIONAL_SHAPE_MAP_WEIGHTS } from "../game/map_chunk"; const LOG = createLogger("mod-interface"); @@ -106,6 +107,14 @@ export class ModInterface { } } + /** + * + * @param {typeof Component} component + */ + registerComponent(component) { + gComponentRegistry.register(component); + } + /** * * @param {object} param0 diff --git a/src/js/mods/modloader.js b/src/js/mods/modloader.js index 45c70caa..b9a7d4d2 100644 --- a/src/js/mods/modloader.js +++ b/src/js/mods/modloader.js @@ -41,8 +41,12 @@ export class ModLoader { if (G_IS_STANDALONE) { mods = await getIPCRenderer().invoke("get-mods"); } else if (G_IS_DEV && globalConfig.debug.loadDevMod) { - // @ts-expect-error - mods = [require("!!raw-loader!./dev_mod")]; + const mod = await ( + await fetch("http://localhost:3005/mods/demo_mod.nobuild/index.js", { + method: "GET", + }) + ).text(); + mods.push(mod); } mods.forEach(modCode => { From 4176621b057dbd65d56c074544a556f7ab633157 Mon Sep 17 00:00:00 2001 From: tobspr Date: Fri, 14 Jan 2022 12:34:02 +0100 Subject: [PATCH 019/129] Proper mods ui --- res/ui/icons/mods.png | Bin 0 -> 4450 bytes res/ui/icons/mods_white.png | Bin 0 -> 4455 bytes src/css/main.scss | 1 + src/css/states/main_menu.scss | 57 ++++++++--- src/css/states/mods.scss | 131 +++++++++++++++++++++++++ src/css/states/settings.scss | 48 +++++++-- src/css/variables.scss | 1 + src/js/application.js | 2 + src/js/mods/demo_mod.nobuild/index.js | 15 +-- src/js/mods/mod.js | 5 +- src/js/states/main_menu.js | 50 ++++------ src/js/states/mods.js | 135 ++++++++++++++++++++++++++ src/js/states/settings.js | 23 +++++ translations/base-en.yaml | 26 +++-- 14 files changed, 431 insertions(+), 63 deletions(-) create mode 100644 res/ui/icons/mods.png create mode 100644 res/ui/icons/mods_white.png create mode 100644 src/css/states/mods.scss create mode 100644 src/js/states/mods.js diff --git a/res/ui/icons/mods.png b/res/ui/icons/mods.png new file mode 100644 index 0000000000000000000000000000000000000000..1b3cb477aff032e4770ee2c8010e2d919c9fc5c7 GIT binary patch literal 4450 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 literal 0 HcmV?d00001 diff --git a/res/ui/icons/mods_white.png b/res/ui/icons/mods_white.png new file mode 100644 index 0000000000000000000000000000000000000000..1cb34f27dc90aaec243ce98e1b387f56cd7a2c46 GIT binary patch literal 4455 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 .headerBar { + display: grid; + grid-template-columns: 1fr auto; + align-items: center; + + > h1 { + justify-self: start; + } + + .openModsFolder { + background-color: $modsColor; + } + } + + .noModSupport { + display: flex; + align-items: center; + justify-content: center; + height: 100%; + flex-direction: column; + + .steamLink { + @include S(height, 50px); + @include S(width, 220px); + background: #171a23 center center / contain no-repeat; + overflow: hidden; + display: block; + text-indent: -999em; + cursor: pointer; + @include S(margin-top, 30px); + pointer-events: all; + transition: all 0.12s ease-in; + transition-property: opacity, transform; + + @include S(border-radius, $globalBorderRadius); + + &:hover { + opacity: 0.9; + } + } + } + + .modsStats { + @include PlainText; + color: $accentColorDark; + + &.noMods { + @include S(width, 400px); + align-self: center; + justify-self: center; + text-align: center; + display: flex; + flex-direction: column; + align-items: center; + @include Text; + @include S(margin-top, 100px); + color: lighten($accentColorDark, 15); + + &::before { + @include S(margin-bottom, 15px); + content: ""; + @include S(width, 50px); + @include S(height, 50px); + background-position: center center; + background-size: contain; + opacity: 0.2; + } + &::before { + /* @load-async */ + background-image: uiResource("res/ui/icons/mods.png") !important; + } + } + } + + .modsList { + @include S(margin-top, 10px); + overflow-y: scroll; + pointer-events: all; + @include S(padding-right, 5px); + flex-grow: 1; + + .mod { + @include S(border-radius, $globalBorderRadius); + background: $accentColorBright; + @include S(margin-bottom, 4px); + @include S(padding, 7px); + @include S(grid-gap, 5px); + display: grid; + grid-template-columns: 1fr D(100px) D(100px) D(100px); + + .checkbox { + align-self: center; + justify-self: center; + } + + .mainInfo { + display: flex; + flex-direction: column; + + .description { + @include SuperSmallText; + @include S(margin-top, 5px); + color: $accentColorDark; + } + .website { + text-transform: uppercase; + align-self: start; + @include SuperSmallText; + @include S(margin-top, 5px); + } + } + + .version, + .author { + display: flex; + flex-direction: column; + strong { + text-transform: uppercase; + color: $accentColorDark; + @include SuperSmallText; + } + } + } + } +} diff --git a/src/css/states/settings.scss b/src/css/states/settings.scss index 5b36c677..75ef6b85 100644 --- a/src/css/states/settings.scss +++ b/src/css/states/settings.scss @@ -15,20 +15,21 @@ } .sidebar { - display: grid; + display: flex; @include S(min-width, 210px); @include S(max-width, 320px); - @include S(grid-gap, 3px); - grid-template-rows: auto auto auto auto auto 1fr; + flex-direction: column; @include StyleBelowWidth($layoutBreak) { - grid-template-rows: 1fr 1fr; - grid-template-columns: auto auto; + display: grid; + grid-template-columns: 1fr 1fr 1fr; + @include S(grid-gap, 5px); max-width: unset !important; } button { text-align: left; + @include S(margin-bottom, 3px); &::after { content: unset; } @@ -37,15 +38,26 @@ @include StyleBelowWidth($layoutBreak) { text-align: center; + height: D(30px) !important; + padding: D(5px) !important; } } .other { - @include S(margin-top, 10px); align-self: end; + margin-top: auto; @include StyleBelowWidth($layoutBreak) { margin-top: 0; + display: grid; + grid-template-columns: 1fr 1fr; + @include S(grid-gap, 5px); + max-width: unset !important; + grid-column: 1 / 3; + + button { + margin: 0 !important; + } } } @@ -69,6 +81,30 @@ } } + button.manageMods { + background-color: lighten($modsColor, 38); + color: $modsColor; + display: flex; + @include S(padding-right, 5px); + .newBadge { + color: #fff; + @include S(border-radius, $globalBorderRadius); + background: $modsColor; + margin-left: auto; + @include S(padding, 0, 3px, 0, 3px); + + @include InlineAnimation(1.3s ease-in-out infinite) { + 50% { + transform: rotate(0deg) scale(1.1); + } + } + } + + &.active { + background-color: $colorGreenBright; + } + } + button.privacy { @include S(margin-top, 4px); } diff --git a/src/css/variables.scss b/src/css/variables.scss index c7b7c17c..fe1fa864 100644 --- a/src/css/variables.scss +++ b/src/css/variables.scss @@ -38,6 +38,7 @@ $colorRedBright: #ef5072; $colorOrangeBright: #ef9d50; $themeColor: #393747; $ingameHudBg: rgba(#333438, 0.9); +$modsColor: rgb(214, 60, 228); $text3dColor: #f4ffff; diff --git a/src/js/application.js b/src/js/application.js index 67e9c353..44207e4d 100644 --- a/src/js/application.js +++ b/src/js/application.js @@ -37,6 +37,7 @@ import { LoginState } from "./states/login"; import { WegameSplashState } from "./states/wegame_splash"; import { MODS } from "./mods/modloader"; import { MOD_SIGNALS } from "./mods/mod_signals"; +import { ModsState } from "./states/mods"; /** * @typedef {import("./platform/achievement_provider").AchievementProviderInterface} AchievementProviderInterface @@ -171,6 +172,7 @@ export class Application { ChangelogState, PuzzleMenuState, LoginState, + ModsState, ]; for (let i = 0; i < states.length; ++i) { diff --git a/src/js/mods/demo_mod.nobuild/index.js b/src/js/mods/demo_mod.nobuild/index.js index b636dace..2188defc 100644 --- a/src/js/mods/demo_mod.nobuild/index.js +++ b/src/js/mods/demo_mod.nobuild/index.js @@ -16,11 +16,12 @@ registerMod(shapez => { super( app, { - authorContact: "tobias@tobspr.io", - authorName: "tobspr", + website: "https://tobspr.io", + author: "tobspr", name: "Demo Mod", version: "1", id: "demo-mod", + description: "A simple mod to demonstrate the capatibilities of the mod loader.", }, modLoader ); @@ -28,11 +29,11 @@ registerMod(shapez => { init() { // Add some custom css - this.modInterface.registerCss(` - * { - font-family: "Comic Sans", "Comic Sans MS", Tahoma !important; - } - `); + // this.modInterface.registerCss(` + // * { + // font-family: "Comic Sans", "Comic Sans MS", Tahoma !important; + // } + // `); // Replace a builtin sprite ["red", "green", "blue", "yellow", "purple", "cyan", "white"].forEach(color => { diff --git a/src/js/mods/mod.js b/src/js/mods/mod.js index b9018b68..4e14bfe2 100644 --- a/src/js/mods/mod.js +++ b/src/js/mods/mod.js @@ -12,8 +12,9 @@ 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.author + * @param {string} metadata.website + * @param {string} metadata.description * @param {string} metadata.id * * @param {ModLoader} modLoader diff --git a/src/js/states/main_menu.js b/src/js/states/main_menu.js index a6bb940a..91142c3b 100644 --- a/src/js/states/main_menu.js +++ b/src/js/states/main_menu.js @@ -155,36 +155,22 @@ export class MainMenuState extends GameState { ? `
-

${T.mainMenu.mods.title} - -

- +
+

${T.mods.title}

+ +
${MODS.mods .map(mod => { return ` -
- ${mod.metadata.name} - ${T.mainMenu.mods.version.replace( - "", - mod.metadata.version - )} - ${T.mainMenu.mods.author.replace( - "", - mod.metadata.authorName - )} -
+
${mod.metadata.name} @ v${mod.metadata.version}
`; }) .join("")}
- ${T.mainMenu.modsWarningPuzzleDLC} - - + ${T.mainMenu.mods.warningPuzzleDLC}
@@ -381,7 +367,7 @@ export class MainMenuState extends GameState { ".puzzleDlcPlayButton": this.onPuzzleModeButtonClicked, ".puzzleDlcGetButton": this.onPuzzleWishlistButtonClicked, ".wegameDisclaimer > .rating": this.onWegameRatingClicked, - ".modsOpenFolder": this.openModsFolder, + ".editMods": this.onModsClicked, }; for (const key in clickHandling) { @@ -430,6 +416,14 @@ export class MainMenuState extends GameState { this.trackClicks(playBtn, this.onPlayButtonClicked); buttonContainer.appendChild(importButtonElement); } + + // Mods + const modsBtn = makeButton( + this.htmlElement.querySelector(".mainContainer .outer"), + ["modsButton", "styledButton"], + " " + ); + this.trackClicks(modsBtn, this.onModsClicked); } onPuzzleModeButtonClicked(force = false) { @@ -742,6 +736,12 @@ export class MainMenuState extends GameState { ); } + onModsClicked() { + this.moveToState("ModsState", { + backToStateId: "MainMenuState", + }); + } + onContinueButtonClicked() { let latestLastUpdate = 0; let latestInternalId; @@ -763,14 +763,6 @@ export class MainMenuState extends GameState { }); } - openModsFolder() { - if (!G_IS_STANDALONE) { - this.dialogs.showWarning(T.global.error, T.mainMenu.mods.folderOnlyStandalone); - return; - } - getIPCRenderer().send("open-mods-folder"); - } - onLeave() { this.dialogs.cleanup(); } diff --git a/src/js/states/mods.js b/src/js/states/mods.js new file mode 100644 index 00000000..d746434d --- /dev/null +++ b/src/js/states/mods.js @@ -0,0 +1,135 @@ +import { THIRDPARTY_URLS } from "../core/config"; +import { TextualGameState } from "../core/textual_game_state"; +import { getIPCRenderer } from "../core/utils"; +import { MODS } from "../mods/modloader"; +import { T } from "../translations"; + +export class ModsState extends TextualGameState { + constructor() { + super("ModsState"); + } + + getStateHeaderTitle() { + return T.mods.title; + } + + internalGetFullHtml() { + let headerHtml = ` +
+

${this.getStateHeaderTitle()}

+ +
+ ${ + G_IS_STANDALONE || G_IS_DEV + ? `` + : "" + } +
+ +
`; + + return ` + ${headerHtml} +
+ ${this.getInnerHTML()} +
+ `; + } + + getMainContentHTML() { + if (!G_IS_STANDALONE && !G_IS_DEV) { + return ` +
+ +

${T.mods.noModSupport}

+ + Get the shapez.io standalone! + + +
+ `; + } + + if (MODS.mods.length === 0) { + return ` + +
+ + ${T.mods.modsInfo} +
+ + `; + } + + let modsHtml = ``; + + MODS.mods.forEach(mod => { + modsHtml += ` +
+
+ ${mod.metadata.name} + ${mod.metadata.description} + Website +
+ ${T.mods.version}${mod.metadata.version} + ${T.mods.author}${mod.metadata.author} +
+ +
+ +
+ `; + }); + return ` + +
+ ${T.mods.modsInfo} +
+ +
+ ${modsHtml} +
+ `; + } + + onEnter() { + const steamLink = this.htmlElement.querySelector(".steamLink"); + if (steamLink) { + this.trackClicks(steamLink, this.onSteamLinkClicked); + } + const openModsFolder = this.htmlElement.querySelector(".openModsFolder"); + if (openModsFolder) { + this.trackClicks(openModsFolder, this.openModsFolder); + } + + const checkboxes = this.htmlElement.querySelectorAll(".checkbox"); + Array.from(checkboxes).forEach(checkbox => { + this.trackClicks(checkbox, this.showModTogglingComingSoon); + }); + } + + showModTogglingComingSoon() { + this.dialogs.showWarning(T.mods.togglingComingSoon.title, T.mods.togglingComingSoon.description); + } + + openModsFolder() { + if (!G_IS_STANDALONE) { + this.dialogs.showWarning(T.global.error, T.mods.folderOnlyStandalone); + return; + } + getIPCRenderer().send("open-mods-folder"); + } + + onSteamLinkClicked() { + this.app.analytics.trackUiClick("mods_steam_link"); + this.app.platformWrapper.openExternalLink( + THIRDPARTY_URLS.stanaloneCampaignLink + "/shapez_modsettings" + ); + + return false; + } + + getDefaultPreviousState() { + return "SettingsState"; + } +} diff --git a/src/js/states/settings.js b/src/js/states/settings.js index 352e0153..b28a4136 100644 --- a/src/js/states/settings.js +++ b/src/js/states/settings.js @@ -19,6 +19,8 @@ export class SettingsState extends TextualGameState {