ModAPI.meta.title("LibCustomRender"); ModAPI.meta.credits("By ZXMushroom63"); ModAPI.meta.icon(""); ModAPI.meta.description("Library to make retexturing LCI items easier. Requires AsyncSink."); (async function LibRender() { var BreakingFour = ModAPI.reflect.getClassByName("BreakingFour").constructors[0]; var BakedQuad = ModAPI.reflect.getClassByName("BakedQuad").constructors[0]; var EnumFacing = ModAPI.reflect.getClassByName("EnumFacing"); function createBreakingFour(sprite$) { var sprite = ModAPI.util.wrap(sprite$); var vertexData = ModAPI.array.int(28); // 7 integers per vertex, 4 vertices var vertexDataInternal = vertexData.data; var vertexDataWithNormals = ModAPI.array.int(32); // 8 integers per vertex, 4 vertices var normalDataInternal = vertexDataWithNormals.data; var vertices = [ [0, 0, 0], [1, 0, 0], [1, 1, 0], [0, 1, 0] ]; var uvs = [ [0, 0], [1, 0], [1, 1], [0, 1] ]; for (let i = 0; i < 4; i++) { let j = i * 7; vertexDataInternal[j] = ModAPI.hooks.methods.jl_Float_floatToIntBits(vertices[i][0]); vertexDataInternal[j + 1] = ModAPI.hooks.methods.jl_Float_floatToIntBits(vertices[i][1]); vertexDataInternal[j + 2] = ModAPI.hooks.methods.jl_Float_floatToIntBits(vertices[i][2]); vertexDataInternal[j + 3] = rgbToInt(255, 0, 0); // Color (red) vertexDataInternal[j + 4] = ModAPI.hooks.methods.jl_Float_floatToIntBits(sprite.getInterpolatedU(uvs[i][0] * 16)); vertexDataInternal[j + 5] = ModAPI.hooks.methods.jl_Float_floatToIntBits(sprite.getInterpolatedV(uvs[i][1] * 16)); vertexDataInternal[j + 6] = 0; // Normal let k = i * 8; normalDataInternal[k] = vertexDataInternal[j]; normalDataInternal[k + 1] = vertexDataInternal[j + 1]; normalDataInternal[k + 2] = vertexDataInternal[j + 2]; normalDataInternal[k + 3] = vertexDataInternal[j + 3]; normalDataInternal[k + 4] = vertexDataInternal[j + 4]; normalDataInternal[k + 5] = vertexDataInternal[j + 5]; normalDataInternal[k + 6] = vertexDataInternal[j + 6]; normalDataInternal[k + 7] = 0; } var baseQuad = BakedQuad(vertexData, vertexDataWithNormals, -1, EnumFacing.staticVariables.NORTH); return BreakingFour(baseQuad, sprite$); } function waitUntilPropertyExists(obj, prop) { return new Promise((res, rej) => { var timer = setInterval(() => { if (obj[prop]) { clearInterval(timer); res(); } }, 50); }); } function rgbToInt(red, green, blue) { return (red << 16) | (green << 8) | blue; } function rgbaToInt(red, green, blue, alpha) { return (alpha << 24) | (red << 16) | (green << 8) | blue; } function getLore(item) { if (item.$stackTagCompound && item.$stackTagCompound.$hasKey(ModAPI.util.str("display"), 10)) { var displayTag = item.$stackTagCompound.$getCompoundTag(ModAPI.util.str("display")); if (displayTag.$hasKey(ModAPI.util.str("Lore"), 9)) { var loreTag = displayTag.$getTag(ModAPI.util.str("Lore")); if (loreTag.$tagCount() > 0) { return ModAPI.util.ustr(loreTag.$getStringTagAt(0)); } } } } function recursiveAssign(target, patch) { var keys = Object.keys(patch); keys.forEach(k => { if (typeof patch[k] === "object" && patch[k]) { recursiveAssign(target[k], patch[k]); } else if (typeof patch[k] === "number") { target[k] = patch[k]; } }); } function cloneBaseModel(baseModel, newTexture, texName) { var sprite = eaglerTextureAtlasSprite(imageDataToLaxImgData(newTexture), ModAPI.util.str(texName)); var newBaseModelBuilder = ModAPI.reflect.getClassByName("SimpleBakedModel$Builder").constructors[0](0, 0, ModAPI.reflect.getClassByName("ItemCameraTransforms").constructors.find(x => x.length === 0)()); newBaseModelBuilder.$builderTexture = sprite; ModAPI.hooks.methods.nmcrm_SimpleBakedModel$Builder_addGeneralQuad(newBaseModelBuilder, createBreakingFour(sprite)); var newBaseModel = ModAPI.hooks.methods.nmcrm_SimpleBakedModel$Builder_makeBakedModel(newBaseModelBuilder); //newBaseModel.$generalQuads = baseModel.$generalQuads.$clone(); //newBaseModel.$faceQuads = baseModel.$faceQuads.$clone(); //var cameraTransformsId = ModAPI.util.getNearestProperty(newBaseModel, "$cameraTransforms"); //recursiveAssign(newBaseModel[cameraTransformsId], baseModel[cameraTransformsId]); return newBaseModel; } ModAPI.events.newEvent("lib:libcustomrender:loaded"); await waitUntilPropertyExists(ModAPI.minecraft, "renderItem"); var ItemRenderer = ModAPI.minecraft.renderItem; var ItemModelMesher = ItemRenderer.itemModelMesher; var laxImgDataClass = ModAPI.reflect.getClassByName("ImageData").class; var makeLax1dudeImageData = ModAPI.reflect.getClassByName("ImageData").constructors.find(x => x.length === 4); var eaglerTextureAtlasSprite = (imageData, name) => { var atlas = ModAPI.reflect.getClassByName("EaglerTextureAtlasSprite").constructors[0](ModAPI.util.str(name)); var alias = ModAPI.util.wrap(atlas); alias.loadSprite(ModAPI.util.makeArray(laxImgDataClass, [imageData]), null); alias.initSprite(1, 1, 0, 0, 0); return atlas; }; /** * @type {ImageData} */ function imageDataToLaxImgData(imageData) { const { data, width, height } = imageData; const intArray = []; for (let i = 0; i < data.length; i += 4) { const red = data[i]; const green = data[i + 1]; const blue = data[i + 2]; const alpha = data[i + 3]; intArray.push(rgbaToInt(red, green, blue, alpha)); } return makeLax1dudeImageData(width, height, ModAPI.array.int(intArray), 1); } const LibCustomRender = {}; LibCustomRender.map = {}; LibCustomRender.addRetextureRule = (loreString, textureBuffer, baseItem) => { baseItem ||= "paper"; var actualLoreStr = loreString; loreString = loreString.replaceAll(":", "_").toLowerCase().replace(/[^a-z_]/g, ''); if (!(textureBuffer instanceof ImageData)) { return console.error("Texture for retexture rule is not an ImageData."); } if (!(typeof loreString === "string")) { return console.error("loreString for retexture rule is not a string."); } var baseModel = ItemModelMesher.simpleShapesCache.get(ModAPI.hooks.methods.jl_Integer_valueOf(ItemModelMesher.getIndex(ModAPI.items[baseItem].getRef(), 0))); LibCustomRender.map[loreString] = { lore: actualLoreStr, identifier: loreString, model: cloneBaseModel(baseModel.getRef(), textureBuffer, loreString) } return LibCustomRender.map[loreString].model; } // override // public IBakedModel getItemModel(ItemStack stack) { // Item item = stack.getItem(); // In ItemModelMesher.java var methods = Object.keys(ModAPI.hooks.methods); var prefix = ModAPI.util.getMethodFromPackage("net.minecraft.client.renderer.ItemModelMesher", "getItemModel"); var methodName = methods.find(x => x.startsWith(prefix) && ModAPI.hooks.methods[x].length === 2); var original = ModAPI.hooks.methods[methodName]; ModAPI.hooks.methods[methodName] = function (...args) { var item = args[1]; var lore = item ? getLore(item) : ""; if (!item) { return original.apply(this, args); } var overrides = Object.values(LibCustomRender.map); for (let i = 0; i < overrides.length; i++) { const override = overrides[i]; if (lore === override.lore) { return override.model; } } return original.apply(this, args); } ModAPI.events.callEvent("lib:libcustomrender:loaded", {}); globalThis.LibCustomRender = LibCustomRender; })(); //LibCustomRender.addRetextureRule("mymod:test_item_1", new ImageData(1, 1)); // const imageData = new ImageData(16, 16); // for (let i = 0; i < imageData.data.length; i += 4) { // imageData.data[i] = Math.floor(Math.random() * 256); // imageData.data[i + 1] = Math.floor(Math.random() * 256); // imageData.data[i + 2] = Math.floor(Math.random() * 256); // imageData.data[i + 3] = 255; // } // LibCustomRender.addRetextureRule("mymod:test_item_1", imageData);