You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
tobspr_shapez.io/mod_examples/new_item_type.js

150 lines
17 KiB

// @ts-nocheck
const METADATA = {
website: "https://tobspr.io",
author: "tobspr",
name: "Mod Example: New Item Type (Fluids)",
version: "1",
id: "new-item-type",
description: "Shows how to add a new item type (fluid)",
minimumGameVersion: ">=1.5.0",
};
// Define which fluid types there are
const enumFluidType = {
water: "water",
oil: "oil",
};
// Define which color they should have on the map
const fluidColors = {
[enumFluidType.water]: "#477be7",
[enumFluidType.oil]: "#bc483a",
};
// The fluid item class (also see ColorItem and ShapeItem)
class FluidItem extends shapez.BaseItem {
static getId() {
return "fluid";
}
static getSchema() {
return shapez.types.enum(enumFluidType);
}
serialize() {
return this.fluidType;
}
deserialize(data) {
this.fluidType = data;
}
getItemType() {
return "fluid";
}
/**
* @returns {string}
*/
getAsCopyableKey() {
return this.fluidType;
}
/**
* @param {BaseItem} other
*/
equalsImpl(other) {
return this.fluidType === /** @type {FluidItem} */ (other).fluidType;
}
/**
* @param {enumFluidType} fluidType
*/
constructor(fluidType) {
super();
this.fluidType = fluidType;
}
getBackgroundColorAsResource() {
return fluidColors[this.fluidType];
}
/**
* Draws the item to a canvas
* @param {CanvasRenderingContext2D} context
* @param {number} size
*/
drawFullSizeOnCanvas(context, size) {
if (!this.cachedSprite) {
this.cachedSprite = shapez.Loader.getSprite(`sprites/fluids/${this.fluidType}.png`);
}
this.cachedSprite.drawCentered(context, size / 2, size / 2, size);
}
/**
* @param {number} x
* @param {number} y
* @param {number} diameter
* @param {DrawParameters} parameters
*/
drawItemCenteredClipped(x, y, parameters, diameter = shapez.globalConfig.defaultItemDiameter) {
const realDiameter = diameter * 0.6;
if (!this.cachedSprite) {
this.cachedSprite = shapez.Loader.getSprite(`sprites/fluids/${this.fluidType}.png`);
}
this.cachedSprite.drawCachedCentered(parameters, x, y, realDiameter);
}
}
/**
* Singleton instances.
*
* NOTICE: The game tries to instantiate as few instances as possible.
* Which means that if you have two types of fluids in this case, there should
* ONLY be 2 instances of FluidItem at *any* time.
*
* This works by having a map from fluid type to the FluidItem singleton.
* Additionally, all items are and should be immutable.
* @type {Object<enumFluidType, FluidItem>}
*/
const FLUID_ITEM_SINGLETONS = {};
for (const fluidType in enumFluidType) {
FLUID_ITEM_SINGLETONS[fluidType] = new FluidItem(fluidType);
}
class Mod extends shapez.Mod {
init() {
// Register the sprites
this.modInterface.registerSprite("sprites/fluids/oil.png", RESOURCES["oil.png"]);
this.modInterface.registerSprite("sprites/fluids/water.png", RESOURCES["water.png"]);
// Make the item spawn on the map
this.modInterface.runAfterMethod(shapez.MapChunk, "generatePatches", function ({
rng,
chunkCenter,
distanceToOriginInChunks,
}) {
// Generate a simple patch
// ALWAYS use rng and NEVER use Math.random() otherwise the map will look different
// every time you resume the game
if (rng.next() > 0.8) {
const fluidType = rng.choice(Array.from(Object.keys(enumFluidType)));
this.internalGeneratePatch(rng, 4, FLUID_ITEM_SINGLETONS[fluidType]);
}
});
this.modInterface.registerItem(FluidItem, itemData => FLUID_ITEM_SINGLETONS[itemData]);
}
}
///////////////////////////////////////
const RESOURCES = {
"oil.png":
"",
"water.png":
"",
};