@minecraft/creator-tools
Version:
Minecraft Creator Tools command line and libraries.
215 lines (214 loc) • 8.07 kB
JavaScript
"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;