UNPKG

@minecraft/creator-tools

Version:

Minecraft Creator Tools command line and libraries.

215 lines (214 loc) 8.07 kB
"use strict"; // Copyright (c) Microsoft Corporation. // Licensed under the MIT License. var __importDefault = (this && this.__importDefault) || function (mod) { return (mod && mod.__esModule) ? mod : { "default": mod }; }; Object.defineProperty(exports, "__esModule", { value: true }); const ste_events_1 = require("ste-events"); const StorageUtilities_1 = __importDefault(require("../storage/StorageUtilities")); const Database_1 = __importDefault(require("./Database")); const MinecraftUtilities_1 = __importDefault(require("./MinecraftUtilities")); const IProjectItemData_1 = require("../app/IProjectItemData"); const ItemTypeDefinition_1 = __importDefault(require("./ItemTypeDefinition")); class RecipeBehaviorDefinition { _file; _id; _isLoaded = false; _loadedWithComments = false; _data; _interior; _onLoaded = new ste_events_1.EventDispatcher(); get data() { return this._data; } set data(newData) { this._data = newData; if (this._data && this._data["minecraft:recipe_shaped"]) { this._interior = this._data["minecraft:recipe_shaped"]; } else if (this._data && this._data["minecraft:recipe_shapeless"]) { this._interior = this._data["minecraft:recipe_shapeless"]; } else { this._interior = undefined; } } get isLoaded() { return this._isLoaded; } get file() { return this._file; } get onLoaded() { return this._onLoaded.asEvent(); } set file(newFile) { this._file = newFile; } get id() { if (this._interior && this._interior.description && this._interior.description.identifier) { return this._interior.description.identifier; } return this._id; } set id(newId) { if (this._interior && this._interior.description) { this._interior.description.identifier = newId; } this._id = newId; } get shortId() { if (this._id !== undefined) { if (this._id.startsWith("minecraft:")) { return this._id.substring(10, this._id.length); } return this._id; } return undefined; } async getFormatVersionIsCurrent() { const fv = this.getFormatVersion(); if (fv === undefined || fv.length !== 3) { return false; } return await Database_1.default.isRecentVersionFromVersionArray(fv); } getFormatVersion() { if (!this._data || !this._data.format_version) { return undefined; } return MinecraftUtilities_1.default.getVersionArrayFrom(this._data.format_version); } setBehaviorPackFormatVersion(versionStr) { this._ensureDataInitialized(); if (this._data) { this._data.format_version = versionStr; } } _ensureDataInitialized() { if (this._data === undefined) { this._data = {}; } } async addChildItems(project, recipeItem, index) { const dependentItemIds = []; // Collect all item IDs that this recipe depends on if (this._interior) { const ingredientsArray = this._interior.ingredients; // For shapeless recipes, collect ingredients if (ingredientsArray && Array.isArray(ingredientsArray)) { for (const ingredient of ingredientsArray) { if (ingredient && ingredient.item) { dependentItemIds.push(ingredient.item); } } } // For shaped recipes, collect key items if ("key" in this._interior && this._interior.key) { for (const keyItem of Object.values(this._interior.key)) { if (keyItem && keyItem.item) { dependentItemIds.push(keyItem.item); } } } } // Check each dependent item ID against project items for (const dependentItemId of dependentItemIds) { let foundMatch = false; // Use index for O(1) lookup when available if (index) { const matchingItems = index.getItemsById(index.itemTypesById, dependentItemId); if (matchingItems.length > 0) { for (const matchItem of matchingItems) { recipeItem.addChildItem(matchItem); } foundMatch = true; } } else { const itemTypeItems = project.getItemsByType(IProjectItemData_1.ProjectItemType.itemTypeBehavior); for (const candItem of itemTypeItems) { if (!candItem.isContentLoaded) { await candItem.loadContent(); } if (candItem.primaryFile) { const itd = await ItemTypeDefinition_1.default.ensureOnFile(candItem.primaryFile); if (itd && itd.id === dependentItemId) { recipeItem.addChildItem(candItem); foundMatch = true; break; } } } } // If no matching item type was found, add as unfulfilled relationship if (!foundMatch) { recipeItem.addUnfulfilledRelationship(dependentItemId, IProjectItemData_1.ProjectItemType.itemTypeBehavior, false); } } } static async ensureOnFile(file, loadHandler) { let rbd; if (file.manager === undefined) { rbd = new RecipeBehaviorDefinition(); rbd.file = file; file.manager = rbd; } if (file.manager !== undefined && file.manager instanceof RecipeBehaviorDefinition) { rbd = file.manager; if (!rbd.isLoaded) { if (loadHandler) { rbd.onLoaded.subscribe(loadHandler); } await rbd.load(); } } return rbd; } persist() { if (this._file === undefined) { return false; } if (!this._data) { return false; } return this._file.setObjectContentIfSemanticallyDifferent(this._data); } /** * Loads the definition from the file. * @param preserveComments If true, uses comment-preserving JSON parsing for edit/save cycles. * If false (default), uses efficient standard JSON parsing. * Can be called again with true to "upgrade" a read-only load to read/write. */ async load(preserveComments = false) { // If already loaded with comments, we have the "best" version - nothing more to do if (this._isLoaded && this._loadedWithComments) { return; } // If already loaded without comments and caller doesn't need comments, we're done if (this._isLoaded && !preserveComments) { return; } if (this._file === undefined) { return; } if (!this._file.isContentLoaded) { await this._file.loadContent(); } if (this._file.content === null || this._file.content instanceof Uint8Array) { this._isLoaded = true; this._loadedWithComments = preserveComments; this._onLoaded.dispatch(this, this); return; } // Use comment-preserving parser only when needed for editing this.data = preserveComments ? StorageUtilities_1.default.getJsonObjectWithComments(this._file) : StorageUtilities_1.default.getJsonObject(this._file); this._isLoaded = true; this._loadedWithComments = preserveComments; this._onLoaded.dispatch(this, this); } } exports.default = RecipeBehaviorDefinition;