mirror of
https://github.com/eaglerforge/EaglerForgeInjector
synced 2025-07-24 22:51:18 -09:00
188 lines
9.1 KiB
JavaScript
188 lines
9.1 KiB
JavaScript
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);
|