diff --git a/.vscode/launch.json b/.vscode/launch.json index 3673590..7169957 100644 --- a/.vscode/launch.json +++ b/.vscode/launch.json @@ -1,13 +1,15 @@ { - "version": "0.3.0", - "configurations": [ - { - "type": "minecraft-js", - "request": "attach", - "name": "Wait for Minecraft Debug Connections", - "mode": "listen", - "localRoot": "${workspaceFolder}/", - "port": 19144 - } - ] - } \ No newline at end of file + "version": "0.3.0", + "configurations": [ + { + "type": "minecraft-js", + "request": "attach", + "name": "Debug with Minecraft", + "mode": "listen", + "targetModuleUuid": "9717780a-c69f-4679-bbe6-dbf6dfcdd7de", + "localRoot": "${workspaceFolder}/BP/scripts/", + "generatedSourceRoot": "${workspaceFolder}/BP/scripts/", + "port": 19144 + } + ] +} diff --git a/BP/entities/block_outline.json b/BP/entities/block_outline.json new file mode 100644 index 0000000..042dc6c --- /dev/null +++ b/BP/entities/block_outline.json @@ -0,0 +1,150 @@ +{ + "format_version": "1.18.0", + + "minecraft:entity": { + "description": { + "identifier": "yn:block_outline", + "is_spawnable": true, + "is_summonable": true, + "properties": { + "yn:trunk_size": { + "type": "int", + "client_sync": true, + "default": 0, + "range": [0, 31] + }, + "yn:is_in_air": { + "type": "bool", + "default": false + } + } + }, + + "components": { + "minecraft:collision_box": { + "width": 0, + "height": 0 + }, + "minecraft:type_family": { + "family": [ + "inanimate" + ] + }, + "minecraft:health": { + "value": 1, + "max": 1 + }, + "minecraft:despawn": { + "despawn_from_distance": { + "min_distance": 32 + }, + "despawn_from_inactivity": true, + "despawn_from_simulation_edge": true, + "min_range_inactivity_timer": 15, + "remove_child_entities": false + }, + "minecraft:on_death": "despawn", + "minecraft:breathable": { + "total_supply": 1, + "suffocate_time": 0, + "breathes_solids": true, + "breathe_blocks": [], + "non_breathe_blocks": ["minecraft:air"] + }, + "minecraft:physics": { + "has_collision": false, + "has_gravity": false + }, + "minecraft:damage_sensor": { + "triggers": [ + { + "cause": "suffocation", + "deals_damage": false, + "on_damage": { + "target": "self", + "event": "despawn" + } + }, + { + "cause": "all", + "deals_damage": false + } + ] + }, + "minecraft:knockback_resistance": { + "value": 100 + }, + "minecraft:pushable": { + "is_pushable": false, + "is_pushable_by_piston": false + } + }, + "component_groups": { + "active_outline": { + "minecraft:variant": { + "value": 1 + } + }, + "inactive_outline": { + "minecraft:variant": { + "value": 0 + } + }, + "despawn": { + "minecraft:instant_despawn": {} + }, + "not_persistent": { + "minecraft:timer": { + "looping": false, + "time": 7, + "time_down_event": { + "event": "despawn" + } + } + }, + "check_for_collision": { + "minecraft:environment_sensor": { + "triggers": [ + { + "event": "on_air_detected", + "target": "self", + "filters": { + "test": "in_block", + "value": "minecraft:air" + } + } + ] + } + } + }, + + "events": { + "minecraft:entity_spawned": { + "remove": { "component_groups": [ "active_outline" ] }, + "add": { "component_groups": [ "inactive_outline", "check_for_collision" ] }, + "set_property": { + "yn:trunk_size": 0 + } + }, + "on_air_detected": { + "set_property": { + "yn:is_in_air": true + } + }, + "despawn": { + "add": { "component_groups": [ "despawn" ] } + }, + "active_outline": { + "remove": { "component_groups": [ "inactive_outline" ] }, + "add": { "component_groups": [ "active_outline" ] } + }, + "inactive_outline": { + "remove": { "component_groups": [ "active_outline" ] }, + "add": { "component_groups": [ "inactive_outline" ] } + }, + "not_persistent": { + "add": { "component_groups": [ "not_persistent" ]} + } + } + } +} \ No newline at end of file diff --git a/BP/functions/yn_axe.mcfunction b/BP/functions/LumberAxe/get_axes.mcfunction similarity index 100% rename from BP/functions/yn_axe.mcfunction rename to BP/functions/LumberAxe/get_axes.mcfunction diff --git a/BP/functions/LumberAxe/open_configuration.mcfunction b/BP/functions/LumberAxe/open_configuration.mcfunction new file mode 100644 index 0000000..35c90b7 --- /dev/null +++ b/BP/functions/LumberAxe/open_configuration.mcfunction @@ -0,0 +1 @@ +scriptevent yn:lumber config show \ No newline at end of file diff --git a/BP/functions/LumberAxe/reload.mcfunction b/BP/functions/LumberAxe/reload.mcfunction new file mode 100644 index 0000000..1baefa8 --- /dev/null +++ b/BP/functions/LumberAxe/reload.mcfunction @@ -0,0 +1 @@ +scriptevent yn:lumber database reset \ No newline at end of file diff --git a/BP/items/diamond_lumber_axe.json b/BP/items/diamond_lumber_axe.json index 3d925e6..0c358a2 100644 --- a/BP/items/diamond_lumber_axe.json +++ b/BP/items/diamond_lumber_axe.json @@ -1,13 +1,14 @@ { - "format_version": "1.16.100", + "format_version": "1.21.10", "minecraft:item": { "description": { "identifier": "yn:diamond_lumber_axe", - "category": "equipment" + "menu_category": { + "category": "equipment", + "group": "itemGroup.name.axe" + } }, "components": { - "minecraft:max_stack_size": 1, - "minecraft:hand_equipped": true, "minecraft:durability": { "max_durability": 1800, "damage_chance": { @@ -15,12 +16,10 @@ "max": 100 } }, - "tag:minecraft:transformable_items": {}, "minecraft:icon": { - "texture": "yn:diamond_lumber_axe" - }, - "minecraft:display_name": { - "value": "item.yn:diamond_lumber_axe.name" + "textures": { + "default": "yn:diamond_lumber_axe" + } }, "minecraft:repairable": { "repair_items": [ @@ -44,797 +43,648 @@ } ] }, - "minecraft:mining_speed": 1.5, - "minecraft:damage": 7, + "minecraft:display_name": { + "value": "item.yn:diamond_lumber_axe.name" + }, + "minecraft:damage": { + "value": 7 + }, + "minecraft:cooldown": { + "category": "lumberaxe_inspect_cooldown", + "duration": 3.0 + }, "minecraft:enchantable": { "value": 8, "slot": "axe" }, - "minecraft:weapon": { - "on_hurt_entity": { - "event": "defaultDamage" - } - }, - "minecraft:creative_category": { - "parent": "itemGroup.name.axe" + "minecraft:custom_components": [ + "yn:tool_durability" + ], + "minecraft:tags": { + "tags": [ + "minecraft:is_tool", + "minecraft:diamond_tier", + "minecraft:transformable_items" + ] }, + "minecraft:max_stack_size": 1, + "minecraft:hand_equipped": true, "minecraft:digger": { "use_efficiency": true, "destroy_speeds": [ + { + "block": "cherry_planks", + "speed": 9 + }, + { + "block": "cherry_slab", + "speed": 9 + }, + { + "block": "cherry_stairs", + "speed": 9 + }, + { + "block": "nether_wart_block", + "speed": 9 + }, { "block": { - "tags": "q.any_tag('wood', 'log', 'pumpkin', 'plant', 'iron_axe_diggable')" + "tags": "q.any_tag('wood', 'log', 'pumpkin', 'plant', 'iron_axe_diggable', 'k_correct_tool:axe')" }, - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "speed": 9 }, { - "block": "minecraft:chest", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "crimson_stem", + "speed": 9 }, { - "block": "minecraft:trapdoor", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "warped_stem", + "speed": 9 }, { - "block": "minecraft:wood", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "stripped_crimson_stem", + "speed": 9 }, { - "block": "minecraft:spruce_trapdoor", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "stripped_warped_stem", + "speed": 9 }, { - "block": "minecraft:birch_fence_gate", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "crimson_hyphae", + "speed": 9 }, { - "block": "minecraft:jungle_fence_gate", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "warped_hyphae", + "speed": 9 }, { - "block": "minecraft:acacia_fence_gate", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "stripped_crimson_hyphae", + "speed": 9 }, { - "block": "minecraft:dark_oak_fence_gate", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "stripped_warped_hyphae", + "speed": 9 }, { - "block": "minecraft:warped_fence_gate", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "soul_torch", + "speed": 9 }, { - "block": "minecraft:crimson_fence_gate", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "warped_standing_sign", + "speed": 9 }, { - "block": "minecraft:jukebox", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "crimson_standing_sign", + "speed": 9 }, { - "block": "minecraft:log", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "warped_wall_sign", + "speed": 9 }, { - "block": "minecraft:log2", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "crimson_wall_sign", + "speed": 9 }, { - "block": "minecraft:mangrove_log", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "crimson_door", + "speed": 9 }, { - "block": "minecraft:crimson_stem", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "warped_door", + "speed": 9 }, { - "block": "minecraft:warped_stem", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "crimson_planks", + "speed": 9 }, { - "block": "minecraft:stripped_spruce_log", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "warped_planks", + "speed": 9 }, { - "block": "minecraft:stripped_birch_log", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "crimson_trapdoor", + "speed": 9 }, { - "block": "minecraft:stripped_jungle_log", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "warped_trapdoor", + "speed": 9 }, { - "block": "minecraft:stripped_acacia_log", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "crimson_fence", + "speed": 9 }, { - "block": "minecraft:stripped_dark_oak_log", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "warped_fence", + "speed": 9 }, { - "block": "minecraft:stripped_oak_log", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "crimson_fence_gate", + "speed": 9 }, { - "block": "minecraft:stripped_crimson_stem", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "warped_fence_gate", + "speed": 9 }, { - "block": "minecraft:stripped_warped_stem", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "crimson_double_slab", + "speed": 9 }, { - "block": "minecraft:planks", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "crimson_slab", + "speed": 9 }, { - "block": "minecraft:crimson_planks", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "warped_double_slab", + "speed": 9 }, { - "block": "minecraft:warped_planks", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "warped_slab", + "speed": 9 }, { - "block": "minecraft:wooden_slab", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "crimson_stairs", + "speed": 9 }, { - "block": "minecraft:double_wooden_slab", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "warped_stairs", + "speed": 9 }, { - "block": "minecraft:oak_stairs", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "crimson_button", + "speed": 9 }, { - "block": "minecraft:spruce_stairs", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "warped_button", + "speed": 9 }, { - "block": "minecraft:jungle_stairs", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "crimson_pressure_plate", + "speed": 9 }, { - "block": "minecraft:acacia_stairs", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "warped_pressure_plate", + "speed": 9 }, { - "block": "minecraft:dark_oak_stairs", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "soul_campfire", + "speed": 9 }, { - "block": "minecraft:crimson_stairs", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "mangrove_log", + "speed": 9 }, { - "block": "minecraft:warped_stairs", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "stripped_mangrove_log", + "speed": 9 }, { - "block": "minecraft:mangrove_stairs", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "mangrove_planks", + "speed": 9 }, { - "block": "minecraft:campfire", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "mangrove_button", + "speed": 9 }, { - "block": "minecraft:soul_campfire", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "mangrove_stairs", + "speed": 9 }, { - "block": "minecraft:bookshelf", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "mangrove_slab", + "speed": 9 }, { - "block": "minecraft:standing_banner", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "mangrove_pressure_plate", + "speed": 9 }, { - "block": "minecraft:wall_banner", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "mangrove_fence", + "speed": 9 }, { - "block": "minecraft:lit_pumpkin", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "mangrove_fence_gate", + "speed": 9 }, { - "block": "minecraft:pumpkin", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "mangrove_door", + "speed": 9 }, { - "block": "minecraft:carved_pumpkin", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "mangrove_standing_sign", + "speed": 9 }, { - "block": "minecraft:standing_sign", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "mangrove_wall_sign", + "speed": 9 }, { - "block": "minecraft:crimson_standing_sign", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "mangrove_trapdoor", + "speed": 9 }, { - "block": "minecraft:warped_standing_sign", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "mangrove_wood", + "speed": 9 }, { - "block": "minecraft:wall_sign", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "stripped_mangrove_wood", + "speed": 9 }, { - "block": "minecraft:noteblock", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "mangrove_double_slab", + "speed": 9 }, { - "block": "minecraft:wooden_pressure_plate", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "planks", + "speed": 9 }, { - "block": "minecraft:beehive", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "log", + "speed": 9 }, { - "block": "minecraft:bee_nest", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "stripped_oak_log", + "speed": 9 }, { - "block": "minecraft:ladder", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "stripped_birch_log", + "speed": 9 }, { - "block": "minecraft:composter", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "stripped_dark_oak_log", + "speed": 9 }, { - "block": "minecraft:bamboo", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "stripped_acacia_log", + "speed": 9 }, { - "block": "minecraft:bamboo_sapling", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "stripped_jungle_log", + "speed": 9 }, { - "block": "minecraft:daylight_detector", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "stripped_spruce_log", + "speed": 9 }, { - "block": "minecraft:daylight_detector_inverted", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "noteblock", + "speed": 9 }, { - "block": "minecraft:brown_mushroom_block", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "jukebox", + "speed": 9 }, { - "block": "minecraft:red_mushroom_block", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "bed", + "speed": 9 }, { - "block": "minecraft:vine", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "bookshelf", + "speed": 9 }, { - "block": "minecraft:cocoa", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "torch", + "speed": 9 }, { - "block": "minecraft:birch_trapdoor", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "oak_stairs", + "speed": 9 }, { - "block": "minecraft:jungle_trapdoor", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "chest", + "speed": 9 }, { - "block": "minecraft:acacia_trapdoor", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "crafting_table", + "speed": 9 }, { - "block": "minecraft:dark_oak_trapdoor", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "standing_sign", + "speed": 9 }, { - "block": "minecraft:spruce_door", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "wooden_door", + "speed": 9 }, { - "block": "minecraft:wooden_door", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "ladder", + "speed": 9 }, { - "block": "minecraft:crimson_planks", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "wall_sign", + "speed": 9 }, { - "block": "minecraft:birch_door", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "wooden_pressure_plate", + "speed": 9 }, { - "block": "minecraft:jungle_door", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "acacia_pressure_plate", + "speed": 9 }, { - "block": "minecraft:acacia_door", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "birch_pressure_plate", + "speed": 9 }, { - "block": "minecraft:trapped_chest", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "dark_oak_pressure_plate", + "speed": 9 }, { - "block": "minecraft:lectern", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "jungle_pressure_plate", + "speed": 9 }, { - "block": "minecraft:smithing_table", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "spruce_pressure_plate", + "speed": 9 }, { - "block": "minecraft:loom", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "unlit_redstone_torch", + "speed": 9 }, { - "block": "minecraft:cartography_table", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "redstone_torch", + "speed": 9 }, { - "block": "minecraft:fletching_table", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "fence", + "speed": 9 }, { - "block": "minecraft:barrel", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "pumpkin", + "speed": 9 }, { - "block": "minecraft:fence", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "carved_pumpkin", + "speed": 9 }, { - "block": "minecraft:warped_fence", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "lit_pumpkin", + "speed": 9 }, { - "block": "minecraft:crimson_fence", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "trapdoor", + "speed": 9 }, { - "block": "minecraft:fence_gate", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "acacia_trapdoor", + "speed": 9 }, { - "block": "minecraft:spruce_fence_gate", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "birch_trapdoor", + "speed": 9 }, { - "block": "minecraft:dark_oak_door", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "dark_oak_trapdoor", + "speed": 9 }, { - "block": "minecraft:bookshelf", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "jungle_trapdoor", + "speed": 9 }, { - "block": "minecraft:melon_block", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "spruce_trapdoor", + "speed": 9 }, { - "block": "minecraft:warped_stem", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "brown_mushroom_block", + "speed": 9 }, { - "block": "minecraft:crimson_stem", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "red_mushroom_block", + "speed": 9 }, { - "block": "minecraft:warped_stem", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "melon_block", + "speed": 9 }, { - "block": "minecraft:crimson_stem", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "pumpkin_stem", + "speed": 9 }, { - "block": "minecraft:crafting_table", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "melon_stem", + "speed": 9 }, { - "block": "minecraft:crimson_planks", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "fence_gate", + "speed": 9 }, { - "block": "minecraft:warped_planks", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "cocoa", + "speed": 9 }, { - "block": "minecraft:warped_stairs", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "spruce_stairs", + "speed": 9 }, { - "block": "minecraft:warped_trapdoor", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "birch_stairs", + "speed": 9 }, { - "block": "minecraft:crimson_stairs", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "jungle_stairs", + "speed": 9 }, { - "block": "minecraft:crimson_trapdoor", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "wooden_button", + "speed": 9 }, { - "block": "minecraft:crimson_door", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "acacia_button", + "speed": 9 }, { - "block": "minecraft:crimson_double_slab", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "birch_button", + "speed": 9 }, { - "block": "minecraft:warped_door", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "dark_oak_button", + "speed": 9 }, { - "block": "minecraft:warped_double_slab", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } + "block": "jungle_button", + "speed": 9 }, { - "block": "minecraft:crafting_table", - "speed": 9, - "on_dig": { - "event": "yn:chopDamage" - } - } - ], - "on_dig": { - "event": "defaultDamage" - } - } - }, - "events": { - "defaultDamage": { - "sequence": [ - { - "damage": { - "type": "durability", - "amount": 1, - "target": "self" - } - } - ] - }, - "digDamage": { - "sequence": [ - { - "damage": { - "type": "durability", - "amount": 1, - "target": "self" - } - } - ] - }, - "yn:chopDamage": { - "sequence": [ - { - "damage": { - "type": "durability", - "amount": 2, - "target": "self" - } + "block": "spruce_button", + "speed": 9 + }, + { + "block": "trapped_chest", + "speed": 9 + }, + { + "block": "daylight_detector", + "speed": 9 + }, + { + "block": "double_wooden_slab", + "speed": 9 + }, + { + "block": "wooden_slab", + "speed": 9 + }, + { + "block": "log2", + "speed": 9 + }, + { + "block": "acacia_stairs", + "speed": 9 + }, + { + "block": "dark_oak_stairs", + "speed": 9 + }, + { + "block": "daylight_detector_inverted", + "speed": 9 + }, + { + "block": "spruce_fence_gate", + "speed": 9 + }, + { + "block": "birch_fence_gate", + "speed": 9 + }, + { + "block": "jungle_fence_gate", + "speed": 9 + }, + { + "block": "dark_oak_fence_gate", + "speed": 9 + }, + { + "block": "acacia_fence_gate", + "speed": 9 + }, + { + "block": "spruce_door", + "speed": 9 + }, + { + "block": "birch_door", + "speed": 9 + }, + { + "block": "jungle_door", + "speed": 9 + }, + { + "block": "acacia_door", + "speed": 9 + }, + { + "block": "dark_oak_door", + "speed": 9 + }, + { + "block": "standing_banner", + "speed": 9 + }, + { + "block": "wall_banner", + "speed": 9 + }, + { + "block": "bamboo", + "speed": 9 + }, + { + "block": "bamboo_sapling", + "speed": 9 + }, + { + "block": "scaffolding", + "speed": 9 + }, + { + "block": "spruce_wall_sign", + "speed": 9 + }, + { + "block": "spruce_standing_sign", + "speed": 9 + }, + { + "block": "birch_wall_sign", + "speed": 9 + }, + { + "block": "birch_standing_sign", + "speed": 9 + }, + { + "block": "jungle_wall_sign", + "speed": 9 + }, + { + "block": "jungle_standing_sign", + "speed": 9 + }, + { + "block": "acacia_wall_sign", + "speed": 9 + }, + { + "block": "acacia_standing_sign", + "speed": 9 + }, + { + "block": "darkoak_wall_sign", + "speed": 9 + }, + { + "block": "darkoak_standing_sign", + "speed": 9 + }, + { + "block": "barrel", + "speed": 9 + }, + { + "block": "smithing_table", + "speed": 9 + }, + { + "block": "cartography_table", + "speed": 9 + }, + { + "block": "fletching_table", + "speed": 9 + }, + { + "block": "campfire", + "speed": 9 + }, + { + "block": "loom", + "speed": 9 + }, + { + "block": "lectern", + "speed": 9 + }, + { + "block": "wood", + "speed": 9 + }, + { + "block": "composter", + "speed": 9 + }, + { + "block": "chorus_flower", + "speed": 9 + }, + { + "block": "chorus_plant", + "speed": 9 } ] } diff --git a/BP/items/golden_lumber_axe.json b/BP/items/golden_lumber_axe.json index da27109..dd374ee 100644 --- a/BP/items/golden_lumber_axe.json +++ b/BP/items/golden_lumber_axe.json @@ -1,842 +1,691 @@ { - "format_version": "1.16.100", - "minecraft:item": { - "description": { - "identifier": "yn:golden_lumber_axe", - "category": "equipment" - }, - "components": { - "minecraft:max_stack_size": 1, - "minecraft:hand_equipped": true, - "minecraft:durability": { - "max_durability": 68, - "damage_chance": { - "min": 60, - "max": 100 - } - }, - "minecraft:icon": { - "texture": "yn:golden_lumber_axe" - }, - "minecraft:display_name": { - "value": "item.yn:golden_lumber_axe.name" - }, - "minecraft:repairable": { - "repair_items": [ - { - "items": [ - "minecraft:gold_ingot" - ], - "repair_amount": "query.max_durability / 4" - }, - { - "items": [ - "minecraft:gold_block" - ], - "repair_amount": "query.max_durability" - }, - { - "items": [ - "yn:golden_lumber_axe" - ], - "repair_amount": "context.other->query.remaining_durability + 0.05 * context.other->query.max_durability" - } - ] - }, - "minecraft:mining_speed": 1.5, - "minecraft:damage": 4, - "minecraft:enchantable": { - "value": 15, - "slot": "axe" - }, - "minecraft:weapon": { - "on_hurt_entity": { - "event": "defaultDamage" - } - }, - "minecraft:creative_category": { - "parent": "itemGroup.name.axe" - }, - "minecraft:digger": { - "use_efficiency": true, - "destroy_speeds": [ - { - "block": { - "tags": "q.any_tag('wood', 'log', 'pumpkin', 'plant', 'iron_axe_diggable')" - }, - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:chest", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:trapdoor", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:wood", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:spruce_trapdoor", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:birch_fence_gate", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:jungle_fence_gate", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:acacia_fence_gate", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:dark_oak_fence_gate", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_fence_gate", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_fence_gate", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:jukebox", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:log", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:log2", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:mangrove_log", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_stem", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_stem", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:stripped_spruce_log", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:stripped_birch_log", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:stripped_jungle_log", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:stripped_acacia_log", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:stripped_dark_oak_log", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:stripped_oak_log", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:stripped_crimson_stem", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:stripped_warped_stem", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:planks", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_planks", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_planks", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:wooden_slab", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:double_wooden_slab", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:oak_stairs", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:spruce_stairs", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:jungle_stairs", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:acacia_stairs", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:dark_oak_stairs", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_stairs", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_stairs", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:mangrove_stairs", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:campfire", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:soul_campfire", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:bookshelf", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:standing_banner", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:wall_banner", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:lit_pumpkin", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:pumpkin", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:carved_pumpkin", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:standing_sign", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_standing_sign", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_standing_sign", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:wall_sign", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:noteblock", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:wooden_pressure_plate", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:beehive", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:bee_nest", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:ladder", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:composter", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:bamboo", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:bamboo_sapling", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:daylight_detector", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:daylight_detector_inverted", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:brown_mushroom_block", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:red_mushroom_block", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:vine", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:cocoa", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:birch_trapdoor", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:jungle_trapdoor", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:acacia_trapdoor", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:dark_oak_trapdoor", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:spruce_door", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:wooden_door", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_planks", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:birch_door", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:jungle_door", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:acacia_door", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:trapped_chest", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:lectern", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:smithing_table", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:loom", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:cartography_table", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:fletching_table", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:barrel", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:fence", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_fence", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_fence", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:fence_gate", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:spruce_fence_gate", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:dark_oak_door", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:bookshelf", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:melon_block", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_stem", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_stem", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_stem", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_stem", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crafting_table", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_planks", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_planks", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_stairs", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_trapdoor", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_stairs", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_trapdoor", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_door", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_double_slab", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_door", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_double_slab", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crafting_table", - "speed": 13, - "on_dig": { - "event": "yn:chopDamage" - } - } - ], - "on_dig": { - "event": "defaultDamage" - } - } - }, - "events": { - "defaultDamage": { - "sequence": [ - { - "damage": { - "type": "durability", - "amount": 1, - "target": "self" - } - } - ] - }, - "digDamage": { - "sequence": [ - { - "damage": { - "type": "durability", - "amount": 1, - "target": "self" - } - } - ] - }, - "yn:chopDamage": { - "sequence": [ - { - "damage": { - "type": "durability", - "amount": 2, - "target": "self" - } - } - ] - } - } - } + "format_version": "1.21.10", + "minecraft:item": { + "description": { + "identifier": "yn:golden_lumber_axe", + "menu_category": { + "category": "equipment", + "group": "itemGroup.name.axe" + } + }, + "components": { + "minecraft:durability": { + "max_durability": 68, + "damage_chance": { + "min": 60, + "max": 100 + } + }, + "minecraft:icon": { + "textures": { + "default": "yn:golden_lumber_axe" + } + }, + "minecraft:display_name": { + "value": "item.yn:golden_lumber_axe.name" + }, + "minecraft:repairable": { + "repair_items": [ + { + "items": [ + "minecraft:gold_ingot" + ], + "repair_amount": "query.max_durability / 4" + }, + { + "items": [ + "minecraft:gold_block" + ], + "repair_amount": "query.max_durability" + }, + { + "items": [ + "yn:golden_lumber_axe" + ], + "repair_amount": "context.other->query.remaining_durability + 0.05 * context.other->query.max_durability" + } + ] + }, + "minecraft:cooldown": { + "category": "lumberaxe_inspect_cooldown", + "duration": 3.0 + }, + "minecraft:enchantable": { + "value": 15, + "slot": "axe" + }, + "minecraft:damage": { + "value": 4 + }, + "minecraft:custom_components": [ + "yn:tool_durability" + ], + "minecraft:tags": { + "tags": [ + "minecraft:is_tool" + ] + }, + "minecraft:max_stack_size": 1, + "minecraft:hand_equipped": true, + "minecraft:digger": { + "use_efficiency": true, + "destroy_speeds": [ + { + "block": "cherry_planks", + "speed": 13 + }, + { + "block": "cherry_slab", + "speed": 13 + }, + { + "block": "cherry_stairs", + "speed": 13 + }, + { + "block": "nether_wart_block", + "speed": 13 + }, + { + "block": { + "tags": "q.any_tag('wood', 'log', 'pumpkin', 'plant', 'iron_axe_diggable', 'k_correct_tool:axe')" + }, + "speed": 13 + }, + { + "block": "crimson_stem", + "speed": 13 + }, + { + "block": "warped_stem", + "speed": 13 + }, + { + "block": "stripped_crimson_stem", + "speed": 13 + }, + { + "block": "stripped_warped_stem", + "speed": 13 + }, + { + "block": "crimson_hyphae", + "speed": 13 + }, + { + "block": "warped_hyphae", + "speed": 13 + }, + { + "block": "stripped_crimson_hyphae", + "speed": 13 + }, + { + "block": "stripped_warped_hyphae", + "speed": 13 + }, + { + "block": "soul_torch", + "speed": 13 + }, + { + "block": "warped_standing_sign", + "speed": 13 + }, + { + "block": "crimson_standing_sign", + "speed": 13 + }, + { + "block": "warped_wall_sign", + "speed": 13 + }, + { + "block": "crimson_wall_sign", + "speed": 13 + }, + { + "block": "crimson_door", + "speed": 13 + }, + { + "block": "warped_door", + "speed": 13 + }, + { + "block": "crimson_planks", + "speed": 13 + }, + { + "block": "warped_planks", + "speed": 13 + }, + { + "block": "crimson_trapdoor", + "speed": 13 + }, + { + "block": "warped_trapdoor", + "speed": 13 + }, + { + "block": "crimson_fence", + "speed": 13 + }, + { + "block": "warped_fence", + "speed": 13 + }, + { + "block": "crimson_fence_gate", + "speed": 13 + }, + { + "block": "warped_fence_gate", + "speed": 13 + }, + { + "block": "crimson_double_slab", + "speed": 13 + }, + { + "block": "crimson_slab", + "speed": 13 + }, + { + "block": "warped_double_slab", + "speed": 13 + }, + { + "block": "warped_slab", + "speed": 13 + }, + { + "block": "crimson_stairs", + "speed": 13 + }, + { + "block": "warped_stairs", + "speed": 13 + }, + { + "block": "crimson_button", + "speed": 13 + }, + { + "block": "warped_button", + "speed": 13 + }, + { + "block": "crimson_pressure_plate", + "speed": 13 + }, + { + "block": "warped_pressure_plate", + "speed": 13 + }, + { + "block": "soul_campfire", + "speed": 13 + }, + { + "block": "mangrove_log", + "speed": 13 + }, + { + "block": "stripped_mangrove_log", + "speed": 13 + }, + { + "block": "mangrove_planks", + "speed": 13 + }, + { + "block": "mangrove_button", + "speed": 13 + }, + { + "block": "mangrove_stairs", + "speed": 13 + }, + { + "block": "mangrove_slab", + "speed": 13 + }, + { + "block": "mangrove_pressure_plate", + "speed": 13 + }, + { + "block": "mangrove_fence", + "speed": 13 + }, + { + "block": "mangrove_fence_gate", + "speed": 13 + }, + { + "block": "mangrove_door", + "speed": 13 + }, + { + "block": "mangrove_standing_sign", + "speed": 13 + }, + { + "block": "mangrove_wall_sign", + "speed": 13 + }, + { + "block": "mangrove_trapdoor", + "speed": 13 + }, + { + "block": "mangrove_wood", + "speed": 13 + }, + { + "block": "stripped_mangrove_wood", + "speed": 13 + }, + { + "block": "mangrove_double_slab", + "speed": 13 + }, + { + "block": "planks", + "speed": 13 + }, + { + "block": "log", + "speed": 13 + }, + { + "block": "stripped_oak_log", + "speed": 13 + }, + { + "block": "stripped_birch_log", + "speed": 13 + }, + { + "block": "stripped_dark_oak_log", + "speed": 13 + }, + { + "block": "stripped_acacia_log", + "speed": 13 + }, + { + "block": "stripped_jungle_log", + "speed": 13 + }, + { + "block": "stripped_spruce_log", + "speed": 13 + }, + { + "block": "noteblock", + "speed": 13 + }, + { + "block": "jukebox", + "speed": 13 + }, + { + "block": "bed", + "speed": 13 + }, + { + "block": "bookshelf", + "speed": 13 + }, + { + "block": "torch", + "speed": 13 + }, + { + "block": "oak_stairs", + "speed": 13 + }, + { + "block": "chest", + "speed": 13 + }, + { + "block": "crafting_table", + "speed": 13 + }, + { + "block": "standing_sign", + "speed": 13 + }, + { + "block": "wooden_door", + "speed": 13 + }, + { + "block": "ladder", + "speed": 13 + }, + { + "block": "wall_sign", + "speed": 13 + }, + { + "block": "wooden_pressure_plate", + "speed": 13 + }, + { + "block": "acacia_pressure_plate", + "speed": 13 + }, + { + "block": "birch_pressure_plate", + "speed": 13 + }, + { + "block": "dark_oak_pressure_plate", + "speed": 13 + }, + { + "block": "jungle_pressure_plate", + "speed": 13 + }, + { + "block": "spruce_pressure_plate", + "speed": 13 + }, + { + "block": "unlit_redstone_torch", + "speed": 13 + }, + { + "block": "redstone_torch", + "speed": 13 + }, + { + "block": "fence", + "speed": 13 + }, + { + "block": "pumpkin", + "speed": 13 + }, + { + "block": "carved_pumpkin", + "speed": 13 + }, + { + "block": "lit_pumpkin", + "speed": 13 + }, + { + "block": "trapdoor", + "speed": 13 + }, + { + "block": "acacia_trapdoor", + "speed": 13 + }, + { + "block": "birch_trapdoor", + "speed": 13 + }, + { + "block": "dark_oak_trapdoor", + "speed": 13 + }, + { + "block": "jungle_trapdoor", + "speed": 13 + }, + { + "block": "spruce_trapdoor", + "speed": 13 + }, + { + "block": "brown_mushroom_block", + "speed": 13 + }, + { + "block": "red_mushroom_block", + "speed": 13 + }, + { + "block": "melon_block", + "speed": 13 + }, + { + "block": "pumpkin_stem", + "speed": 13 + }, + { + "block": "melon_stem", + "speed": 13 + }, + { + "block": "fence_gate", + "speed": 13 + }, + { + "block": "cocoa", + "speed": 13 + }, + { + "block": "spruce_stairs", + "speed": 13 + }, + { + "block": "birch_stairs", + "speed": 13 + }, + { + "block": "jungle_stairs", + "speed": 13 + }, + { + "block": "wooden_button", + "speed": 13 + }, + { + "block": "acacia_button", + "speed": 13 + }, + { + "block": "birch_button", + "speed": 13 + }, + { + "block": "dark_oak_button", + "speed": 13 + }, + { + "block": "jungle_button", + "speed": 13 + }, + { + "block": "spruce_button", + "speed": 13 + }, + { + "block": "trapped_chest", + "speed": 13 + }, + { + "block": "daylight_detector", + "speed": 13 + }, + { + "block": "double_wooden_slab", + "speed": 13 + }, + { + "block": "wooden_slab", + "speed": 13 + }, + { + "block": "log2", + "speed": 13 + }, + { + "block": "acacia_stairs", + "speed": 13 + }, + { + "block": "dark_oak_stairs", + "speed": 13 + }, + { + "block": "daylight_detector_inverted", + "speed": 13 + }, + { + "block": "spruce_fence_gate", + "speed": 13 + }, + { + "block": "birch_fence_gate", + "speed": 13 + }, + { + "block": "jungle_fence_gate", + "speed": 13 + }, + { + "block": "dark_oak_fence_gate", + "speed": 13 + }, + { + "block": "acacia_fence_gate", + "speed": 13 + }, + { + "block": "spruce_door", + "speed": 13 + }, + { + "block": "birch_door", + "speed": 13 + }, + { + "block": "jungle_door", + "speed": 13 + }, + { + "block": "acacia_door", + "speed": 13 + }, + { + "block": "dark_oak_door", + "speed": 13 + }, + { + "block": "standing_banner", + "speed": 13 + }, + { + "block": "wall_banner", + "speed": 13 + }, + { + "block": "bamboo", + "speed": 13 + }, + { + "block": "bamboo_sapling", + "speed": 13 + }, + { + "block": "scaffolding", + "speed": 13 + }, + { + "block": "spruce_wall_sign", + "speed": 13 + }, + { + "block": "spruce_standing_sign", + "speed": 13 + }, + { + "block": "birch_wall_sign", + "speed": 13 + }, + { + "block": "birch_standing_sign", + "speed": 13 + }, + { + "block": "jungle_wall_sign", + "speed": 13 + }, + { + "block": "jungle_standing_sign", + "speed": 13 + }, + { + "block": "acacia_wall_sign", + "speed": 13 + }, + { + "block": "acacia_standing_sign", + "speed": 13 + }, + { + "block": "darkoak_wall_sign", + "speed": 13 + }, + { + "block": "darkoak_standing_sign", + "speed": 13 + }, + { + "block": "barrel", + "speed": 13 + }, + { + "block": "smithing_table", + "speed": 13 + }, + { + "block": "cartography_table", + "speed": 13 + }, + { + "block": "fletching_table", + "speed": 13 + }, + { + "block": "campfire", + "speed": 13 + }, + { + "block": "loom", + "speed": 13 + }, + { + "block": "lectern", + "speed": 13 + }, + { + "block": "wood", + "speed": 13 + }, + { + "block": "composter", + "speed": 13 + }, + { + "block": "chorus_flower", + "speed": 13 + }, + { + "block": "chorus_plant", + "speed": 13 + } + ] + } + } + } } \ No newline at end of file diff --git a/BP/items/iron_lumber_axe.json b/BP/items/iron_lumber_axe.json index eb5e62d..b092cb5 100644 --- a/BP/items/iron_lumber_axe.json +++ b/BP/items/iron_lumber_axe.json @@ -1,842 +1,691 @@ { - "format_version": "1.16.100", - "minecraft:item": { - "description": { - "identifier": "yn:iron_lumber_axe", - "category": "equipment" - }, - "components": { - "minecraft:max_stack_size": 1, - "minecraft:hand_equipped": true, - "minecraft:durability": { - "max_durability": 345, - "damage_chance": { - "min": 60, - "max": 100 - } - }, - "minecraft:icon": { - "texture": "yn:iron_lumber_axe" - }, - "minecraft:display_name": { - "value": "item.yn:iron_lumber_axe.name" - }, - "minecraft:repairable": { - "repair_items": [ - { - "items": [ - "minecraft:iron_ingot" - ], - "repair_amount": "query.max_durability / 4" - }, - { - "items": [ - "minecraft:iron_block" - ], - "repair_amount": "query.max_durability" - }, - { - "items": [ - "yn:iron_lumber_axe" - ], - "repair_amount": "context.other->query.remaining_durability + 0.05 * context.other->query.max_durability" - } - ] - }, - "minecraft:mining_speed": 1.5, - "minecraft:damage": 6, - "minecraft:enchantable": { - "value": 10, - "slot": "axe" - }, - "minecraft:weapon": { - "on_hurt_entity": { - "event": "defaultDamage" - } - }, - "minecraft:creative_category": { - "parent": "itemGroup.name.axe" - }, - "minecraft:digger": { - "use_efficiency": true, - "destroy_speeds": [ - { - "block": { - "tags": "q.any_tag('wood', 'log', 'pumpkin', 'plant', 'iron_axe_diggable')" - }, - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:chest", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:trapdoor", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:wood", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:spruce_trapdoor", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:birch_fence_gate", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:jungle_fence_gate", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:acacia_fence_gate", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:dark_oak_fence_gate", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_fence_gate", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_fence_gate", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:jukebox", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:log", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:log2", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:mangrove_log", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_stem", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_stem", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:stripped_spruce_log", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:stripped_birch_log", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:stripped_jungle_log", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:stripped_acacia_log", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:stripped_dark_oak_log", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:stripped_oak_log", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:stripped_crimson_stem", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:stripped_warped_stem", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:planks", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_planks", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_planks", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:wooden_slab", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:double_wooden_slab", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:oak_stairs", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:spruce_stairs", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:jungle_stairs", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:acacia_stairs", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:dark_oak_stairs", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_stairs", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_stairs", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:mangrove_stairs", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:campfire", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:soul_campfire", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:bookshelf", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:standing_banner", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:wall_banner", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:lit_pumpkin", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:pumpkin", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:carved_pumpkin", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:standing_sign", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_standing_sign", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_standing_sign", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:wall_sign", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:noteblock", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:wooden_pressure_plate", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:beehive", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:bee_nest", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:ladder", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:composter", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:bamboo", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:bamboo_sapling", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:daylight_detector", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:daylight_detector_inverted", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:brown_mushroom_block", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:red_mushroom_block", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:vine", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:cocoa", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:birch_trapdoor", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:jungle_trapdoor", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:acacia_trapdoor", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:dark_oak_trapdoor", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:spruce_door", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:wooden_door", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_planks", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:birch_door", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:jungle_door", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:acacia_door", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:trapped_chest", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:lectern", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:smithing_table", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:loom", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:cartography_table", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:fletching_table", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:barrel", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:fence", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_fence", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_fence", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:fence_gate", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:spruce_fence_gate", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:dark_oak_door", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:bookshelf", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:melon_block", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_stem", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_stem", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_stem", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_stem", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crafting_table", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_planks", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_planks", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_stairs", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_trapdoor", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_stairs", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_trapdoor", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_door", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_double_slab", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_door", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_double_slab", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crafting_table", - "speed": 7, - "on_dig": { - "event": "yn:chopDamage" - } - } - ], - "on_dig": { - "event": "defaultDamage" - } - } - }, - "events": { - "defaultDamage": { - "sequence": [ - { - "damage": { - "type": "durability", - "amount": 1, - "target": "self" - } - } - ] - }, - "digDamage": { - "sequence": [ - { - "damage": { - "type": "durability", - "amount": 1, - "target": "self" - } - } - ] - }, - "yn:chopDamage": { - "sequence": [ - { - "damage": { - "type": "durability", - "amount": 2, - "target": "self" - } - } - ] - } - } - } + "format_version": "1.21.10", + "minecraft:item": { + "description": { + "identifier": "yn:iron_lumber_axe", + "menu_category": { + "category": "equipment", + "group": "itemGroup.name.axe" + } + }, + "components": { + "minecraft:durability": { + "max_durability": 345, + "damage_chance": { + "min": 60, + "max": 100 + } + }, + "minecraft:icon": { + "textures": { + "default": "yn:iron_lumber_axe" + } + }, + "minecraft:display_name": { + "value": "item.yn:iron_lumber_axe.name" + }, + "minecraft:repairable": { + "repair_items": [ + { + "items": [ + "minecraft:iron_ingot" + ], + "repair_amount": "query.max_durability / 4" + }, + { + "items": [ + "minecraft:iron_block" + ], + "repair_amount": "query.max_durability" + }, + { + "items": [ + "yn:iron_lumber_axe" + ], + "repair_amount": "context.other->query.remaining_durability + 0.05 * context.other->query.max_durability" + } + ] + }, + "minecraft:damage": { + "value": 6 + }, + "minecraft:cooldown": { + "category": "lumberaxe_inspect_cooldown", + "duration": 3.0 + }, + "minecraft:enchantable": { + "value": 10, + "slot": "axe" + }, + "minecraft:custom_components": [ + "yn:tool_durability" + ], + "minecraft:tags": { + "tags": [ + "minecraft:is_tool" + ] + }, + "minecraft:max_stack_size": 1, + "minecraft:hand_equipped": true, + "minecraft:digger": { + "use_efficiency": true, + "destroy_speeds": [ + { + "block": "cherry_planks", + "speed": 7 + }, + { + "block": "cherry_slab", + "speed": 7 + }, + { + "block": "cherry_stairs", + "speed": 7 + }, + { + "block": "nether_wart_block", + "speed": 7 + }, + { + "block": { + "tags": "q.any_tag('wood', 'log', 'pumpkin', 'plant', 'iron_axe_diggable', 'k_correct_tool:axe')" + }, + "speed": 7 + }, + { + "block": "crimson_stem", + "speed": 7 + }, + { + "block": "warped_stem", + "speed": 7 + }, + { + "block": "stripped_crimson_stem", + "speed": 7 + }, + { + "block": "stripped_warped_stem", + "speed": 7 + }, + { + "block": "crimson_hyphae", + "speed": 7 + }, + { + "block": "warped_hyphae", + "speed": 7 + }, + { + "block": "stripped_crimson_hyphae", + "speed": 7 + }, + { + "block": "stripped_warped_hyphae", + "speed": 7 + }, + { + "block": "soul_torch", + "speed": 7 + }, + { + "block": "warped_standing_sign", + "speed": 7 + }, + { + "block": "crimson_standing_sign", + "speed": 7 + }, + { + "block": "warped_wall_sign", + "speed": 7 + }, + { + "block": "crimson_wall_sign", + "speed": 7 + }, + { + "block": "crimson_door", + "speed": 7 + }, + { + "block": "warped_door", + "speed": 7 + }, + { + "block": "crimson_planks", + "speed": 7 + }, + { + "block": "warped_planks", + "speed": 7 + }, + { + "block": "crimson_trapdoor", + "speed": 7 + }, + { + "block": "warped_trapdoor", + "speed": 7 + }, + { + "block": "crimson_fence", + "speed": 7 + }, + { + "block": "warped_fence", + "speed": 7 + }, + { + "block": "crimson_fence_gate", + "speed": 7 + }, + { + "block": "warped_fence_gate", + "speed": 7 + }, + { + "block": "crimson_double_slab", + "speed": 7 + }, + { + "block": "crimson_slab", + "speed": 7 + }, + { + "block": "warped_double_slab", + "speed": 7 + }, + { + "block": "warped_slab", + "speed": 7 + }, + { + "block": "crimson_stairs", + "speed": 7 + }, + { + "block": "warped_stairs", + "speed": 7 + }, + { + "block": "crimson_button", + "speed": 7 + }, + { + "block": "warped_button", + "speed": 7 + }, + { + "block": "crimson_pressure_plate", + "speed": 7 + }, + { + "block": "warped_pressure_plate", + "speed": 7 + }, + { + "block": "soul_campfire", + "speed": 7 + }, + { + "block": "mangrove_log", + "speed": 7 + }, + { + "block": "stripped_mangrove_log", + "speed": 7 + }, + { + "block": "mangrove_planks", + "speed": 7 + }, + { + "block": "mangrove_button", + "speed": 7 + }, + { + "block": "mangrove_stairs", + "speed": 7 + }, + { + "block": "mangrove_slab", + "speed": 7 + }, + { + "block": "mangrove_pressure_plate", + "speed": 7 + }, + { + "block": "mangrove_fence", + "speed": 7 + }, + { + "block": "mangrove_fence_gate", + "speed": 7 + }, + { + "block": "mangrove_door", + "speed": 7 + }, + { + "block": "mangrove_standing_sign", + "speed": 7 + }, + { + "block": "mangrove_wall_sign", + "speed": 7 + }, + { + "block": "mangrove_trapdoor", + "speed": 7 + }, + { + "block": "mangrove_wood", + "speed": 7 + }, + { + "block": "stripped_mangrove_wood", + "speed": 7 + }, + { + "block": "mangrove_double_slab", + "speed": 7 + }, + { + "block": "planks", + "speed": 7 + }, + { + "block": "log", + "speed": 7 + }, + { + "block": "stripped_oak_log", + "speed": 7 + }, + { + "block": "stripped_birch_log", + "speed": 7 + }, + { + "block": "stripped_dark_oak_log", + "speed": 7 + }, + { + "block": "stripped_acacia_log", + "speed": 7 + }, + { + "block": "stripped_jungle_log", + "speed": 7 + }, + { + "block": "stripped_spruce_log", + "speed": 7 + }, + { + "block": "noteblock", + "speed": 7 + }, + { + "block": "jukebox", + "speed": 7 + }, + { + "block": "bed", + "speed": 7 + }, + { + "block": "bookshelf", + "speed": 7 + }, + { + "block": "torch", + "speed": 7 + }, + { + "block": "oak_stairs", + "speed": 7 + }, + { + "block": "chest", + "speed": 7 + }, + { + "block": "crafting_table", + "speed": 7 + }, + { + "block": "standing_sign", + "speed": 7 + }, + { + "block": "wooden_door", + "speed": 7 + }, + { + "block": "ladder", + "speed": 7 + }, + { + "block": "wall_sign", + "speed": 7 + }, + { + "block": "wooden_pressure_plate", + "speed": 7 + }, + { + "block": "acacia_pressure_plate", + "speed": 7 + }, + { + "block": "birch_pressure_plate", + "speed": 7 + }, + { + "block": "dark_oak_pressure_plate", + "speed": 7 + }, + { + "block": "jungle_pressure_plate", + "speed": 7 + }, + { + "block": "spruce_pressure_plate", + "speed": 7 + }, + { + "block": "unlit_redstone_torch", + "speed": 7 + }, + { + "block": "redstone_torch", + "speed": 7 + }, + { + "block": "fence", + "speed": 7 + }, + { + "block": "pumpkin", + "speed": 7 + }, + { + "block": "carved_pumpkin", + "speed": 7 + }, + { + "block": "lit_pumpkin", + "speed": 7 + }, + { + "block": "trapdoor", + "speed": 7 + }, + { + "block": "acacia_trapdoor", + "speed": 7 + }, + { + "block": "birch_trapdoor", + "speed": 7 + }, + { + "block": "dark_oak_trapdoor", + "speed": 7 + }, + { + "block": "jungle_trapdoor", + "speed": 7 + }, + { + "block": "spruce_trapdoor", + "speed": 7 + }, + { + "block": "brown_mushroom_block", + "speed": 7 + }, + { + "block": "red_mushroom_block", + "speed": 7 + }, + { + "block": "melon_block", + "speed": 7 + }, + { + "block": "pumpkin_stem", + "speed": 7 + }, + { + "block": "melon_stem", + "speed": 7 + }, + { + "block": "fence_gate", + "speed": 7 + }, + { + "block": "cocoa", + "speed": 7 + }, + { + "block": "spruce_stairs", + "speed": 7 + }, + { + "block": "birch_stairs", + "speed": 7 + }, + { + "block": "jungle_stairs", + "speed": 7 + }, + { + "block": "wooden_button", + "speed": 7 + }, + { + "block": "acacia_button", + "speed": 7 + }, + { + "block": "birch_button", + "speed": 7 + }, + { + "block": "dark_oak_button", + "speed": 7 + }, + { + "block": "jungle_button", + "speed": 7 + }, + { + "block": "spruce_button", + "speed": 7 + }, + { + "block": "trapped_chest", + "speed": 7 + }, + { + "block": "daylight_detector", + "speed": 7 + }, + { + "block": "double_wooden_slab", + "speed": 7 + }, + { + "block": "wooden_slab", + "speed": 7 + }, + { + "block": "log2", + "speed": 7 + }, + { + "block": "acacia_stairs", + "speed": 7 + }, + { + "block": "dark_oak_stairs", + "speed": 7 + }, + { + "block": "daylight_detector_inverted", + "speed": 7 + }, + { + "block": "spruce_fence_gate", + "speed": 7 + }, + { + "block": "birch_fence_gate", + "speed": 7 + }, + { + "block": "jungle_fence_gate", + "speed": 7 + }, + { + "block": "dark_oak_fence_gate", + "speed": 7 + }, + { + "block": "acacia_fence_gate", + "speed": 7 + }, + { + "block": "spruce_door", + "speed": 7 + }, + { + "block": "birch_door", + "speed": 7 + }, + { + "block": "jungle_door", + "speed": 7 + }, + { + "block": "acacia_door", + "speed": 7 + }, + { + "block": "dark_oak_door", + "speed": 7 + }, + { + "block": "standing_banner", + "speed": 7 + }, + { + "block": "wall_banner", + "speed": 7 + }, + { + "block": "bamboo", + "speed": 7 + }, + { + "block": "bamboo_sapling", + "speed": 7 + }, + { + "block": "scaffolding", + "speed": 7 + }, + { + "block": "spruce_wall_sign", + "speed": 7 + }, + { + "block": "spruce_standing_sign", + "speed": 7 + }, + { + "block": "birch_wall_sign", + "speed": 7 + }, + { + "block": "birch_standing_sign", + "speed": 7 + }, + { + "block": "jungle_wall_sign", + "speed": 7 + }, + { + "block": "jungle_standing_sign", + "speed": 7 + }, + { + "block": "acacia_wall_sign", + "speed": 7 + }, + { + "block": "acacia_standing_sign", + "speed": 7 + }, + { + "block": "darkoak_wall_sign", + "speed": 7 + }, + { + "block": "darkoak_standing_sign", + "speed": 7 + }, + { + "block": "barrel", + "speed": 7 + }, + { + "block": "smithing_table", + "speed": 7 + }, + { + "block": "cartography_table", + "speed": 7 + }, + { + "block": "fletching_table", + "speed": 7 + }, + { + "block": "campfire", + "speed": 7 + }, + { + "block": "loom", + "speed": 7 + }, + { + "block": "lectern", + "speed": 7 + }, + { + "block": "wood", + "speed": 7 + }, + { + "block": "composter", + "speed": 7 + }, + { + "block": "chorus_flower", + "speed": 7 + }, + { + "block": "chorus_plant", + "speed": 7 + } + ] + } + } + } } \ No newline at end of file diff --git a/BP/items/netherite_lumber_axe.json b/BP/items/netherite_lumber_axe.json index 882532f..212531a 100644 --- a/BP/items/netherite_lumber_axe.json +++ b/BP/items/netherite_lumber_axe.json @@ -1,836 +1,685 @@ { - "format_version": "1.16.100", - "minecraft:item": { - "description": { - "identifier": "yn:netherite_lumber_axe", - "category": "equipment" - }, - "components": { - "minecraft:max_stack_size": 1, - "minecraft:hand_equipped": true, - "minecraft:durability": { - "max_durability": 3000, - "damage_chance": { - "min": 60, - "max": 100 - } - }, - "minecraft:icon": { - "texture": "yn:netherite_lumber_axe" - }, - "minecraft:display_name": { - "value": "item.yn:netherite_lumber_axe.name" - }, - "minecraft:repairable": { - "repair_items": [ - { - "items": [ - "minecraft:netherite_ingot" - ], - "repair_amount": "query.max_durability / 2" - }, - { - "items": [ - "yn:netherite_lumber_axe" - ], - "repair_amount": "context.other->query.remaining_durability + 0.05 * context.other->query.max_durability" - } - ] - }, - "minecraft:mining_speed": 1.5, - "minecraft:damage": 8, - "minecraft:enchantable": { - "value": 5, - "slot": "axe" - }, - "minecraft:weapon": { - "on_hurt_entity": { - "event": "defaultDamage" - } - }, - "minecraft:creative_category": { - "parent": "itemGroup.name.axe" - }, - "minecraft:digger": { - "use_efficiency": true, - "destroy_speeds": [ - { - "block": { - "tags": "q.any_tag('wood', 'log', 'pumpkin', 'plant', 'iron_axe_diggable')" - }, - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:chest", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:trapdoor", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:wood", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:spruce_trapdoor", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:birch_fence_gate", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:jungle_fence_gate", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:acacia_fence_gate", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:dark_oak_fence_gate", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_fence_gate", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_fence_gate", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:jukebox", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:log", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:log2", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:mangrove_log", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_stem", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_stem", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:stripped_spruce_log", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:stripped_birch_log", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:stripped_jungle_log", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:stripped_acacia_log", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:stripped_dark_oak_log", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:stripped_oak_log", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:stripped_crimson_stem", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:stripped_warped_stem", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:planks", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_planks", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_planks", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:wooden_slab", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:double_wooden_slab", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:oak_stairs", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:spruce_stairs", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:jungle_stairs", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:acacia_stairs", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:dark_oak_stairs", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_stairs", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_stairs", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:mangrove_stairs", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:campfire", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:soul_campfire", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:bookshelf", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:standing_banner", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:wall_banner", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:lit_pumpkin", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:pumpkin", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:carved_pumpkin", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:standing_sign", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_standing_sign", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_standing_sign", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:wall_sign", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:noteblock", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:wooden_pressure_plate", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:beehive", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:bee_nest", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:ladder", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:composter", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:bamboo", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:bamboo_sapling", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:daylight_detector", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:daylight_detector_inverted", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:brown_mushroom_block", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:red_mushroom_block", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:vine", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:cocoa", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:birch_trapdoor", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:jungle_trapdoor", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:acacia_trapdoor", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:dark_oak_trapdoor", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:spruce_door", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:wooden_door", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_planks", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:birch_door", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:jungle_door", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:acacia_door", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:trapped_chest", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:lectern", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:smithing_table", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:loom", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:cartography_table", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:fletching_table", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:barrel", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:fence", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_fence", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_fence", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:fence_gate", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:spruce_fence_gate", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:dark_oak_door", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:bookshelf", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:melon_block", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_stem", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_stem", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_stem", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_stem", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crafting_table", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_planks", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_planks", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_stairs", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_trapdoor", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_stairs", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_trapdoor", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_door", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_double_slab", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_door", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_double_slab", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crafting_table", - "speed": 10, - "on_dig": { - "event": "yn:chopDamage" - } - } - ], - "on_dig": { - "event": "defaultDamage" - } - } - }, - "events": { - "defaultDamage": { - "sequence": [ - { - "damage": { - "type": "durability", - "amount": 1, - "target": "self" - } - } - ] - }, - "digDamage": { - "sequence": [ - { - "damage": { - "type": "durability", - "amount": 1, - "target": "self" - } - } - ] - }, - "yn:chopDamage": { - "sequence": [ - { - "damage": { - "type": "durability", - "amount": 2, - "target": "self" - } - } - ] - } - } - } + "format_version": "1.21.10", + "minecraft:item": { + "description": { + "identifier": "yn:netherite_lumber_axe", + "menu_category": { + "category": "equipment", + "group": "itemGroup.name.axe" + } + }, + "components": { + "minecraft:durability": { + "max_durability": 3000, + "damage_chance": { + "min": 60, + "max": 100 + } + }, + "minecraft:icon": { + "textures": { + "default": "yn:netherite_lumber_axe" + } + }, + "minecraft:display_name": { + "value": "item.yn:netherite_lumber_axe.name" + }, + "minecraft:repairable": { + "repair_items": [ + { + "items": [ + "minecraft:netherite_ingot" + ], + "repair_amount": "query.max_durability / 2" + }, + { + "items": [ + "yn:netherite_lumber_axe" + ], + "repair_amount": "context.other->query.remaining_durability + 0.05 * context.other->query.max_durability" + } + ] + }, + "minecraft:damage": { + "value": 8 + }, + "minecraft:cooldown": { + "category": "lumberaxe_inspect_cooldown", + "duration": 3.0 + }, + "minecraft:enchantable": { + "value": 5, + "slot": "axe" + }, + "minecraft:custom_components": [ + "yn:tool_durability" + ], + "minecraft:tags": { + "tags": [ + "minecraft:is_tool" + ] + }, + "minecraft:max_stack_size": 1, + "minecraft:hand_equipped": true, + "minecraft:digger": { + "use_efficiency": true, + "destroy_speeds": [ + { + "block": "cherry_planks", + "speed": 10 + }, + { + "block": "cherry_slab", + "speed": 10 + }, + { + "block": "cherry_stairs", + "speed": 10 + }, + { + "block": "nether_wart_block", + "speed": 10 + }, + { + "block": { + "tags": "q.any_tag('wood', 'log', 'pumpkin', 'plant', 'iron_axe_diggable', 'k_correct_tool:axe')" + }, + "speed": 10 + }, + { + "block": "crimson_stem", + "speed": 10 + }, + { + "block": "warped_stem", + "speed": 10 + }, + { + "block": "stripped_crimson_stem", + "speed": 10 + }, + { + "block": "stripped_warped_stem", + "speed": 10 + }, + { + "block": "crimson_hyphae", + "speed": 10 + }, + { + "block": "warped_hyphae", + "speed": 10 + }, + { + "block": "stripped_crimson_hyphae", + "speed": 10 + }, + { + "block": "stripped_warped_hyphae", + "speed": 10 + }, + { + "block": "soul_torch", + "speed": 10 + }, + { + "block": "warped_standing_sign", + "speed": 10 + }, + { + "block": "crimson_standing_sign", + "speed": 10 + }, + { + "block": "warped_wall_sign", + "speed": 10 + }, + { + "block": "crimson_wall_sign", + "speed": 10 + }, + { + "block": "crimson_door", + "speed": 10 + }, + { + "block": "warped_door", + "speed": 10 + }, + { + "block": "crimson_planks", + "speed": 10 + }, + { + "block": "warped_planks", + "speed": 10 + }, + { + "block": "crimson_trapdoor", + "speed": 10 + }, + { + "block": "warped_trapdoor", + "speed": 10 + }, + { + "block": "crimson_fence", + "speed": 10 + }, + { + "block": "warped_fence", + "speed": 10 + }, + { + "block": "crimson_fence_gate", + "speed": 10 + }, + { + "block": "warped_fence_gate", + "speed": 10 + }, + { + "block": "crimson_double_slab", + "speed": 10 + }, + { + "block": "crimson_slab", + "speed": 10 + }, + { + "block": "warped_double_slab", + "speed": 10 + }, + { + "block": "warped_slab", + "speed": 10 + }, + { + "block": "crimson_stairs", + "speed": 10 + }, + { + "block": "warped_stairs", + "speed": 10 + }, + { + "block": "crimson_button", + "speed": 10 + }, + { + "block": "warped_button", + "speed": 10 + }, + { + "block": "crimson_pressure_plate", + "speed": 10 + }, + { + "block": "warped_pressure_plate", + "speed": 10 + }, + { + "block": "soul_campfire", + "speed": 10 + }, + { + "block": "mangrove_log", + "speed": 10 + }, + { + "block": "stripped_mangrove_log", + "speed": 10 + }, + { + "block": "mangrove_planks", + "speed": 10 + }, + { + "block": "mangrove_button", + "speed": 10 + }, + { + "block": "mangrove_stairs", + "speed": 10 + }, + { + "block": "mangrove_slab", + "speed": 10 + }, + { + "block": "mangrove_pressure_plate", + "speed": 10 + }, + { + "block": "mangrove_fence", + "speed": 10 + }, + { + "block": "mangrove_fence_gate", + "speed": 10 + }, + { + "block": "mangrove_door", + "speed": 10 + }, + { + "block": "mangrove_standing_sign", + "speed": 10 + }, + { + "block": "mangrove_wall_sign", + "speed": 10 + }, + { + "block": "mangrove_trapdoor", + "speed": 10 + }, + { + "block": "mangrove_wood", + "speed": 10 + }, + { + "block": "stripped_mangrove_wood", + "speed": 10 + }, + { + "block": "mangrove_double_slab", + "speed": 10 + }, + { + "block": "planks", + "speed": 10 + }, + { + "block": "log", + "speed": 10 + }, + { + "block": "stripped_oak_log", + "speed": 10 + }, + { + "block": "stripped_birch_log", + "speed": 10 + }, + { + "block": "stripped_dark_oak_log", + "speed": 10 + }, + { + "block": "stripped_acacia_log", + "speed": 10 + }, + { + "block": "stripped_jungle_log", + "speed": 10 + }, + { + "block": "stripped_spruce_log", + "speed": 10 + }, + { + "block": "noteblock", + "speed": 10 + }, + { + "block": "jukebox", + "speed": 10 + }, + { + "block": "bed", + "speed": 10 + }, + { + "block": "bookshelf", + "speed": 10 + }, + { + "block": "torch", + "speed": 10 + }, + { + "block": "oak_stairs", + "speed": 10 + }, + { + "block": "chest", + "speed": 10 + }, + { + "block": "crafting_table", + "speed": 10 + }, + { + "block": "standing_sign", + "speed": 10 + }, + { + "block": "wooden_door", + "speed": 10 + }, + { + "block": "ladder", + "speed": 10 + }, + { + "block": "wall_sign", + "speed": 10 + }, + { + "block": "wooden_pressure_plate", + "speed": 10 + }, + { + "block": "acacia_pressure_plate", + "speed": 10 + }, + { + "block": "birch_pressure_plate", + "speed": 10 + }, + { + "block": "dark_oak_pressure_plate", + "speed": 10 + }, + { + "block": "jungle_pressure_plate", + "speed": 10 + }, + { + "block": "spruce_pressure_plate", + "speed": 10 + }, + { + "block": "unlit_redstone_torch", + "speed": 10 + }, + { + "block": "redstone_torch", + "speed": 10 + }, + { + "block": "fence", + "speed": 10 + }, + { + "block": "pumpkin", + "speed": 10 + }, + { + "block": "carved_pumpkin", + "speed": 10 + }, + { + "block": "lit_pumpkin", + "speed": 10 + }, + { + "block": "trapdoor", + "speed": 10 + }, + { + "block": "acacia_trapdoor", + "speed": 10 + }, + { + "block": "birch_trapdoor", + "speed": 10 + }, + { + "block": "dark_oak_trapdoor", + "speed": 10 + }, + { + "block": "jungle_trapdoor", + "speed": 10 + }, + { + "block": "spruce_trapdoor", + "speed": 10 + }, + { + "block": "brown_mushroom_block", + "speed": 10 + }, + { + "block": "red_mushroom_block", + "speed": 10 + }, + { + "block": "melon_block", + "speed": 10 + }, + { + "block": "pumpkin_stem", + "speed": 10 + }, + { + "block": "melon_stem", + "speed": 10 + }, + { + "block": "fence_gate", + "speed": 10 + }, + { + "block": "cocoa", + "speed": 10 + }, + { + "block": "spruce_stairs", + "speed": 10 + }, + { + "block": "birch_stairs", + "speed": 10 + }, + { + "block": "jungle_stairs", + "speed": 10 + }, + { + "block": "wooden_button", + "speed": 10 + }, + { + "block": "acacia_button", + "speed": 10 + }, + { + "block": "birch_button", + "speed": 10 + }, + { + "block": "dark_oak_button", + "speed": 10 + }, + { + "block": "jungle_button", + "speed": 10 + }, + { + "block": "spruce_button", + "speed": 10 + }, + { + "block": "trapped_chest", + "speed": 10 + }, + { + "block": "daylight_detector", + "speed": 10 + }, + { + "block": "double_wooden_slab", + "speed": 10 + }, + { + "block": "wooden_slab", + "speed": 10 + }, + { + "block": "log2", + "speed": 10 + }, + { + "block": "acacia_stairs", + "speed": 10 + }, + { + "block": "dark_oak_stairs", + "speed": 10 + }, + { + "block": "daylight_detector_inverted", + "speed": 10 + }, + { + "block": "spruce_fence_gate", + "speed": 10 + }, + { + "block": "birch_fence_gate", + "speed": 10 + }, + { + "block": "jungle_fence_gate", + "speed": 10 + }, + { + "block": "dark_oak_fence_gate", + "speed": 10 + }, + { + "block": "acacia_fence_gate", + "speed": 10 + }, + { + "block": "spruce_door", + "speed": 10 + }, + { + "block": "birch_door", + "speed": 10 + }, + { + "block": "jungle_door", + "speed": 10 + }, + { + "block": "acacia_door", + "speed": 10 + }, + { + "block": "dark_oak_door", + "speed": 10 + }, + { + "block": "standing_banner", + "speed": 10 + }, + { + "block": "wall_banner", + "speed": 10 + }, + { + "block": "bamboo", + "speed": 10 + }, + { + "block": "bamboo_sapling", + "speed": 10 + }, + { + "block": "scaffolding", + "speed": 10 + }, + { + "block": "spruce_wall_sign", + "speed": 10 + }, + { + "block": "spruce_standing_sign", + "speed": 10 + }, + { + "block": "birch_wall_sign", + "speed": 10 + }, + { + "block": "birch_standing_sign", + "speed": 10 + }, + { + "block": "jungle_wall_sign", + "speed": 10 + }, + { + "block": "jungle_standing_sign", + "speed": 10 + }, + { + "block": "acacia_wall_sign", + "speed": 10 + }, + { + "block": "acacia_standing_sign", + "speed": 10 + }, + { + "block": "darkoak_wall_sign", + "speed": 10 + }, + { + "block": "darkoak_standing_sign", + "speed": 10 + }, + { + "block": "barrel", + "speed": 10 + }, + { + "block": "smithing_table", + "speed": 10 + }, + { + "block": "cartography_table", + "speed": 10 + }, + { + "block": "fletching_table", + "speed": 10 + }, + { + "block": "campfire", + "speed": 10 + }, + { + "block": "loom", + "speed": 10 + }, + { + "block": "lectern", + "speed": 10 + }, + { + "block": "wood", + "speed": 10 + }, + { + "block": "composter", + "speed": 10 + }, + { + "block": "chorus_flower", + "speed": 10 + }, + { + "block": "chorus_plant", + "speed": 10 + } + ] + } + } + } } \ No newline at end of file diff --git a/BP/items/stone_lumber_axe.json b/BP/items/stone_lumber_axe.json index 60bc968..ecaa712 100644 --- a/BP/items/stone_lumber_axe.json +++ b/BP/items/stone_lumber_axe.json @@ -1,830 +1,679 @@ { - "format_version": "1.16.100", - "minecraft:item": { - "description": { - "identifier": "yn:stone_lumber_axe", - "category": "equipment" - }, - "components": { - "minecraft:max_stack_size": 1, - "minecraft:hand_equipped": true, - "minecraft:durability": { - "max_durability": 165, - "damage_chance": { - "min": 60, - "max": 100 - } - }, - "minecraft:icon": { - "texture": "yn:stone_lumber_axe" - }, - "minecraft:display_name": { - "value": "item.yn:stone_lumber_axe.name" - }, - "minecraft:repairable": { - "repair_items": [ - { - "items": [ - "yn:stone_lumber_axe" - ], - "repair_amount": "context.other->query.remaining_durability + 0.05 * context.other->query.max_durability" - } - ] - }, - "minecraft:mining_speed": 1.5, - "minecraft:damage": 5, - "minecraft:enchantable": { - "value": 10, - "slot": "axe" - }, - "minecraft:weapon": { - "on_hurt_entity": { - "event": "defaultDamage" - } - }, - "minecraft:creative_category": { - "parent": "itemGroup.name.axe" - }, - "minecraft:digger": { - "use_efficiency": true, - "destroy_speeds": [ - { - "block": { - "tags": "q.any_tag('wood', 'log', 'pumpkin', 'plant', 'iron_axe_diggable')" - }, - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:chest", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:trapdoor", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:wood", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:spruce_trapdoor", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:birch_fence_gate", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:jungle_fence_gate", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:acacia_fence_gate", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:dark_oak_fence_gate", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_fence_gate", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_fence_gate", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:jukebox", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:log", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:log2", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:mangrove_log", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_stem", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_stem", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:stripped_spruce_log", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:stripped_birch_log", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:stripped_jungle_log", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:stripped_acacia_log", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:stripped_dark_oak_log", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:stripped_oak_log", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:stripped_crimson_stem", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:stripped_warped_stem", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:planks", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_planks", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_planks", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:wooden_slab", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:double_wooden_slab", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:oak_stairs", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:spruce_stairs", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:jungle_stairs", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:acacia_stairs", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:dark_oak_stairs", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_stairs", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_stairs", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:mangrove_stairs", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:campfire", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:soul_campfire", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:bookshelf", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:standing_banner", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:wall_banner", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:lit_pumpkin", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:pumpkin", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:carved_pumpkin", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:standing_sign", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_standing_sign", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_standing_sign", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:wall_sign", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:noteblock", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:wooden_pressure_plate", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:beehive", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:bee_nest", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:ladder", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:composter", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:bamboo", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:bamboo_sapling", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:daylight_detector", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:daylight_detector_inverted", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:brown_mushroom_block", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:red_mushroom_block", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:vine", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:cocoa", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:birch_trapdoor", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:jungle_trapdoor", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:acacia_trapdoor", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:dark_oak_trapdoor", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:spruce_door", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:wooden_door", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_planks", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:birch_door", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:jungle_door", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:acacia_door", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:trapped_chest", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:lectern", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:smithing_table", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:loom", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:cartography_table", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:fletching_table", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:barrel", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:fence", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_fence", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_fence", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:fence_gate", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:spruce_fence_gate", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:dark_oak_door", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:bookshelf", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:melon_block", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_stem", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_stem", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_stem", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_stem", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crafting_table", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_planks", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_planks", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_stairs", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_trapdoor", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_stairs", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_trapdoor", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_door", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_double_slab", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_door", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_double_slab", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crafting_table", - "speed": 5, - "on_dig": { - "event": "yn:chopDamage" - } - } - ], - "on_dig": { - "event": "defaultDamage" - } - } - }, - "events": { - "defaultDamage": { - "sequence": [ - { - "damage": { - "type": "durability", - "amount": 1, - "target": "self" - } - } - ] - }, - "digDamage": { - "sequence": [ - { - "damage": { - "type": "durability", - "amount": 1, - "target": "self" - } - } - ] - }, - "yn:chopDamage": { - "sequence": [ - { - "damage": { - "type": "durability", - "amount": 2, - "target": "self" - } - } - ] - } - } - } + "format_version": "1.21.10", + "minecraft:item": { + "description": { + "identifier": "yn:stone_lumber_axe", + "menu_category": { + "category": "equipment", + "group": "itemGroup.name.axe" + } + }, + "components": { + "minecraft:durability": { + "max_durability": 165, + "damage_chance": { + "min": 60, + "max": 100 + } + }, + "minecraft:icon": { + "textures": { + "default": "yn:stone_lumber_axe" + } + }, + "minecraft:display_name": { + "value": "item.yn:stone_lumber_axe.name" + }, + "minecraft:repairable": { + "repair_items": [ + { + "items": [ + "yn:stone_lumber_axe" + ], + "repair_amount": "context.other->query.remaining_durability + 0.05 * context.other->query.max_durability" + } + ] + }, + "minecraft:damage": { + "value": 5 + }, + "minecraft:enchantable": { + "value": 10, + "slot": "axe" + }, + "minecraft:cooldown": { + "category": "lumberaxe_inspect_cooldown", + "duration": 3.0 + }, + "minecraft:custom_components": [ + "yn:tool_durability" + ], + "minecraft:tags": { + "tags": [ + "minecraft:is_tool" + ] + }, + "minecraft:max_stack_size": 1, + "minecraft:hand_equipped": true, + "minecraft:digger": { + "use_efficiency": true, + "destroy_speeds": [ + { + "block": "cherry_planks", + "speed": 5 + }, + { + "block": "cherry_slab", + "speed": 5 + }, + { + "block": "cherry_stairs", + "speed": 5 + }, + { + "block": "nether_wart_block", + "speed": 5 + }, + { + "block": { + "tags": "q.any_tag('wood', 'log', 'pumpkin', 'plant', 'iron_axe_diggable', 'k_correct_tool:axe')" + }, + "speed": 5 + }, + { + "block": "crimson_stem", + "speed": 5 + }, + { + "block": "warped_stem", + "speed": 5 + }, + { + "block": "stripped_crimson_stem", + "speed": 5 + }, + { + "block": "stripped_warped_stem", + "speed": 5 + }, + { + "block": "crimson_hyphae", + "speed": 5 + }, + { + "block": "warped_hyphae", + "speed": 5 + }, + { + "block": "stripped_crimson_hyphae", + "speed": 5 + }, + { + "block": "stripped_warped_hyphae", + "speed": 5 + }, + { + "block": "soul_torch", + "speed": 5 + }, + { + "block": "warped_standing_sign", + "speed": 5 + }, + { + "block": "crimson_standing_sign", + "speed": 5 + }, + { + "block": "warped_wall_sign", + "speed": 5 + }, + { + "block": "crimson_wall_sign", + "speed": 5 + }, + { + "block": "crimson_door", + "speed": 5 + }, + { + "block": "warped_door", + "speed": 5 + }, + { + "block": "crimson_planks", + "speed": 5 + }, + { + "block": "warped_planks", + "speed": 5 + }, + { + "block": "crimson_trapdoor", + "speed": 5 + }, + { + "block": "warped_trapdoor", + "speed": 5 + }, + { + "block": "crimson_fence", + "speed": 5 + }, + { + "block": "warped_fence", + "speed": 5 + }, + { + "block": "crimson_fence_gate", + "speed": 5 + }, + { + "block": "warped_fence_gate", + "speed": 5 + }, + { + "block": "crimson_double_slab", + "speed": 5 + }, + { + "block": "crimson_slab", + "speed": 5 + }, + { + "block": "warped_double_slab", + "speed": 5 + }, + { + "block": "warped_slab", + "speed": 5 + }, + { + "block": "crimson_stairs", + "speed": 5 + }, + { + "block": "warped_stairs", + "speed": 5 + }, + { + "block": "crimson_button", + "speed": 5 + }, + { + "block": "warped_button", + "speed": 5 + }, + { + "block": "crimson_pressure_plate", + "speed": 5 + }, + { + "block": "warped_pressure_plate", + "speed": 5 + }, + { + "block": "soul_campfire", + "speed": 5 + }, + { + "block": "mangrove_log", + "speed": 5 + }, + { + "block": "stripped_mangrove_log", + "speed": 5 + }, + { + "block": "mangrove_planks", + "speed": 5 + }, + { + "block": "mangrove_button", + "speed": 5 + }, + { + "block": "mangrove_stairs", + "speed": 5 + }, + { + "block": "mangrove_slab", + "speed": 5 + }, + { + "block": "mangrove_pressure_plate", + "speed": 5 + }, + { + "block": "mangrove_fence", + "speed": 5 + }, + { + "block": "mangrove_fence_gate", + "speed": 5 + }, + { + "block": "mangrove_door", + "speed": 5 + }, + { + "block": "mangrove_standing_sign", + "speed": 5 + }, + { + "block": "mangrove_wall_sign", + "speed": 5 + }, + { + "block": "mangrove_trapdoor", + "speed": 5 + }, + { + "block": "mangrove_wood", + "speed": 5 + }, + { + "block": "stripped_mangrove_wood", + "speed": 5 + }, + { + "block": "mangrove_double_slab", + "speed": 5 + }, + { + "block": "planks", + "speed": 5 + }, + { + "block": "log", + "speed": 5 + }, + { + "block": "stripped_oak_log", + "speed": 5 + }, + { + "block": "stripped_birch_log", + "speed": 5 + }, + { + "block": "stripped_dark_oak_log", + "speed": 5 + }, + { + "block": "stripped_acacia_log", + "speed": 5 + }, + { + "block": "stripped_jungle_log", + "speed": 5 + }, + { + "block": "stripped_spruce_log", + "speed": 5 + }, + { + "block": "noteblock", + "speed": 5 + }, + { + "block": "jukebox", + "speed": 5 + }, + { + "block": "bed", + "speed": 5 + }, + { + "block": "bookshelf", + "speed": 5 + }, + { + "block": "torch", + "speed": 5 + }, + { + "block": "oak_stairs", + "speed": 5 + }, + { + "block": "chest", + "speed": 5 + }, + { + "block": "crafting_table", + "speed": 5 + }, + { + "block": "standing_sign", + "speed": 5 + }, + { + "block": "wooden_door", + "speed": 5 + }, + { + "block": "ladder", + "speed": 5 + }, + { + "block": "wall_sign", + "speed": 5 + }, + { + "block": "wooden_pressure_plate", + "speed": 5 + }, + { + "block": "acacia_pressure_plate", + "speed": 5 + }, + { + "block": "birch_pressure_plate", + "speed": 5 + }, + { + "block": "dark_oak_pressure_plate", + "speed": 5 + }, + { + "block": "jungle_pressure_plate", + "speed": 5 + }, + { + "block": "spruce_pressure_plate", + "speed": 5 + }, + { + "block": "unlit_redstone_torch", + "speed": 5 + }, + { + "block": "redstone_torch", + "speed": 5 + }, + { + "block": "fence", + "speed": 5 + }, + { + "block": "pumpkin", + "speed": 5 + }, + { + "block": "carved_pumpkin", + "speed": 5 + }, + { + "block": "lit_pumpkin", + "speed": 5 + }, + { + "block": "trapdoor", + "speed": 5 + }, + { + "block": "acacia_trapdoor", + "speed": 5 + }, + { + "block": "birch_trapdoor", + "speed": 5 + }, + { + "block": "dark_oak_trapdoor", + "speed": 5 + }, + { + "block": "jungle_trapdoor", + "speed": 5 + }, + { + "block": "spruce_trapdoor", + "speed": 5 + }, + { + "block": "brown_mushroom_block", + "speed": 5 + }, + { + "block": "red_mushroom_block", + "speed": 5 + }, + { + "block": "melon_block", + "speed": 5 + }, + { + "block": "pumpkin_stem", + "speed": 5 + }, + { + "block": "melon_stem", + "speed": 5 + }, + { + "block": "fence_gate", + "speed": 5 + }, + { + "block": "cocoa", + "speed": 5 + }, + { + "block": "spruce_stairs", + "speed": 5 + }, + { + "block": "birch_stairs", + "speed": 5 + }, + { + "block": "jungle_stairs", + "speed": 5 + }, + { + "block": "wooden_button", + "speed": 5 + }, + { + "block": "acacia_button", + "speed": 5 + }, + { + "block": "birch_button", + "speed": 5 + }, + { + "block": "dark_oak_button", + "speed": 5 + }, + { + "block": "jungle_button", + "speed": 5 + }, + { + "block": "spruce_button", + "speed": 5 + }, + { + "block": "trapped_chest", + "speed": 5 + }, + { + "block": "daylight_detector", + "speed": 5 + }, + { + "block": "double_wooden_slab", + "speed": 5 + }, + { + "block": "wooden_slab", + "speed": 5 + }, + { + "block": "log2", + "speed": 5 + }, + { + "block": "acacia_stairs", + "speed": 5 + }, + { + "block": "dark_oak_stairs", + "speed": 5 + }, + { + "block": "daylight_detector_inverted", + "speed": 5 + }, + { + "block": "spruce_fence_gate", + "speed": 5 + }, + { + "block": "birch_fence_gate", + "speed": 5 + }, + { + "block": "jungle_fence_gate", + "speed": 5 + }, + { + "block": "dark_oak_fence_gate", + "speed": 5 + }, + { + "block": "acacia_fence_gate", + "speed": 5 + }, + { + "block": "spruce_door", + "speed": 5 + }, + { + "block": "birch_door", + "speed": 5 + }, + { + "block": "jungle_door", + "speed": 5 + }, + { + "block": "acacia_door", + "speed": 5 + }, + { + "block": "dark_oak_door", + "speed": 5 + }, + { + "block": "standing_banner", + "speed": 5 + }, + { + "block": "wall_banner", + "speed": 5 + }, + { + "block": "bamboo", + "speed": 5 + }, + { + "block": "bamboo_sapling", + "speed": 5 + }, + { + "block": "scaffolding", + "speed": 5 + }, + { + "block": "spruce_wall_sign", + "speed": 5 + }, + { + "block": "spruce_standing_sign", + "speed": 5 + }, + { + "block": "birch_wall_sign", + "speed": 5 + }, + { + "block": "birch_standing_sign", + "speed": 5 + }, + { + "block": "jungle_wall_sign", + "speed": 5 + }, + { + "block": "jungle_standing_sign", + "speed": 5 + }, + { + "block": "acacia_wall_sign", + "speed": 5 + }, + { + "block": "acacia_standing_sign", + "speed": 5 + }, + { + "block": "darkoak_wall_sign", + "speed": 5 + }, + { + "block": "darkoak_standing_sign", + "speed": 5 + }, + { + "block": "barrel", + "speed": 5 + }, + { + "block": "smithing_table", + "speed": 5 + }, + { + "block": "cartography_table", + "speed": 5 + }, + { + "block": "fletching_table", + "speed": 5 + }, + { + "block": "campfire", + "speed": 5 + }, + { + "block": "loom", + "speed": 5 + }, + { + "block": "lectern", + "speed": 5 + }, + { + "block": "wood", + "speed": 5 + }, + { + "block": "composter", + "speed": 5 + }, + { + "block": "chorus_flower", + "speed": 5 + }, + { + "block": "chorus_plant", + "speed": 5 + } + ] + } + } + } } \ No newline at end of file diff --git a/BP/items/wooden_lumber_axe.json b/BP/items/wooden_lumber_axe.json index 7ea9cb1..8b5dcef 100644 --- a/BP/items/wooden_lumber_axe.json +++ b/BP/items/wooden_lumber_axe.json @@ -1,830 +1,679 @@ { - "format_version": "1.16.100", - "minecraft:item": { - "description": { - "identifier": "yn:wooden_lumber_axe", - "category": "equipment" - }, - "components": { - "minecraft:max_stack_size": 1, - "minecraft:hand_equipped": true, - "minecraft:durability": { - "max_durability": 78, - "damage_chance": { - "min": 60, - "max": 100 - } - }, - "minecraft:icon": { - "texture": "yn:wooden_lumber_axe" - }, - "minecraft:display_name": { - "value": "item.yn:wooden_lumber_axe.name" - }, - "minecraft:repairable": { - "repair_items": [ - { - "items": [ - "yn:wooden_lumber_axe" - ], - "repair_amount": "context.other->query.remaining_durability + 0.05 * context.other->query.max_durability" - } - ] - }, - "minecraft:mining_speed": 1.5, - "minecraft:damage": 4, - "minecraft:enchantable": { - "value": 10, - "slot": "axe" - }, - "minecraft:weapon": { - "on_hurt_entity": { - "event": "defaultDamage" - } - }, - "minecraft:creative_category": { - "parent": "itemGroup.name.axe" - }, - "minecraft:digger": { - "use_efficiency": true, - "destroy_speeds": [ - { - "block": { - "tags": "q.any_tag('wood', 'log', 'pumpkin', 'plant', 'iron_axe_diggable')" - }, - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:chest", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:trapdoor", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:wood", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:spruce_trapdoor", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:birch_fence_gate", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:jungle_fence_gate", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:acacia_fence_gate", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:dark_oak_fence_gate", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_fence_gate", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_fence_gate", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:jukebox", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:log", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:log2", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:mangrove_log", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_stem", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_stem", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:stripped_spruce_log", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:stripped_birch_log", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:stripped_jungle_log", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:stripped_acacia_log", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:stripped_dark_oak_log", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:stripped_oak_log", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:stripped_crimson_stem", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:stripped_warped_stem", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:planks", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_planks", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_planks", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:wooden_slab", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:double_wooden_slab", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:oak_stairs", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:spruce_stairs", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:jungle_stairs", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:acacia_stairs", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:dark_oak_stairs", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_stairs", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_stairs", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:mangrove_stairs", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:campfire", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:soul_campfire", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:bookshelf", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:standing_banner", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:wall_banner", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:lit_pumpkin", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:pumpkin", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:carved_pumpkin", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:standing_sign", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_standing_sign", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_standing_sign", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:wall_sign", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:noteblock", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:wooden_pressure_plate", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:beehive", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:bee_nest", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:ladder", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:composter", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:bamboo", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:bamboo_sapling", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:daylight_detector", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:daylight_detector_inverted", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:brown_mushroom_block", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:red_mushroom_block", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:vine", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:cocoa", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:birch_trapdoor", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:jungle_trapdoor", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:acacia_trapdoor", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:dark_oak_trapdoor", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:spruce_door", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:wooden_door", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_planks", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:birch_door", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:jungle_door", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:acacia_door", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:trapped_chest", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:lectern", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:smithing_table", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:loom", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:cartography_table", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:fletching_table", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:barrel", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:fence", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_fence", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_fence", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:fence_gate", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:spruce_fence_gate", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:dark_oak_door", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:bookshelf", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:melon_block", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_stem", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_stem", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_stem", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_stem", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crafting_table", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_planks", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_planks", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_stairs", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_trapdoor", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_stairs", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_trapdoor", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_door", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crimson_double_slab", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_door", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:warped_double_slab", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - }, - { - "block": "minecraft:crafting_table", - "speed": 3, - "on_dig": { - "event": "yn:chopDamage" - } - } - ], - "on_dig": { - "event": "defaultDamage" - } - } - }, - "events": { - "defaultDamage": { - "sequence": [ - { - "damage": { - "type": "durability", - "amount": 1, - "target": "self" - } - } - ] - }, - "digDamage": { - "sequence": [ - { - "damage": { - "type": "durability", - "amount": 1, - "target": "self" - } - } - ] - }, - "yn:chopDamage": { - "sequence": [ - { - "damage": { - "type": "durability", - "amount": 2, - "target": "self" - } - } - ] - } - } - } + "format_version": "1.21.10", + "minecraft:item": { + "description": { + "identifier": "yn:wooden_lumber_axe", + "menu_category": { + "category": "equipment", + "group": "itemGroup.name.axe" + } + }, + "components": { + "minecraft:durability": { + "max_durability": 78, + "damage_chance": { + "min": 60, + "max": 100 + } + }, + "minecraft:icon": { + "textures": { + "default": "yn:wooden_lumber_axe" + } + }, + "minecraft:display_name": { + "value": "item.yn:wooden_lumber_axe.name" + }, + "minecraft:repairable": { + "repair_items": [ + { + "items": [ + "yn:wooden_lumber_axe" + ], + "repair_amount": "context.other->query.remaining_durability + 0.05 * context.other->query.max_durability" + } + ] + }, + "minecraft:cooldown": { + "category": "lumberaxe_inspect_cooldown", + "duration": 3.0 + }, + "minecraft:damage": { + "value": 4 + }, + "minecraft:enchantable": { + "value": 10, + "slot": "axe" + }, + "minecraft:custom_components": [ + "yn:tool_durability" + ], + "minecraft:tags": { + "tags": [ + "minecraft:is_tool" + ] + }, + "minecraft:max_stack_size": 1, + "minecraft:hand_equipped": true, + "minecraft:digger": { + "use_efficiency": true, + "destroy_speeds": [ + { + "block": "cherry_planks", + "speed": 3 + }, + { + "block": "cherry_slab", + "speed": 3 + }, + { + "block": "cherry_stairs", + "speed": 3 + }, + { + "block": "nether_wart_block", + "speed": 3 + }, + { + "block": { + "tags": "q.any_tag('wood', 'log', 'pumpkin', 'plant', 'iron_axe_diggable', 'k_correct_tool:axe')" + }, + "speed": 3 + }, + { + "block": "crimson_stem", + "speed": 3 + }, + { + "block": "warped_stem", + "speed": 3 + }, + { + "block": "stripped_crimson_stem", + "speed": 3 + }, + { + "block": "stripped_warped_stem", + "speed": 3 + }, + { + "block": "crimson_hyphae", + "speed": 3 + }, + { + "block": "warped_hyphae", + "speed": 3 + }, + { + "block": "stripped_crimson_hyphae", + "speed": 3 + }, + { + "block": "stripped_warped_hyphae", + "speed": 3 + }, + { + "block": "soul_torch", + "speed": 3 + }, + { + "block": "warped_standing_sign", + "speed": 3 + }, + { + "block": "crimson_standing_sign", + "speed": 3 + }, + { + "block": "warped_wall_sign", + "speed": 3 + }, + { + "block": "crimson_wall_sign", + "speed": 3 + }, + { + "block": "crimson_door", + "speed": 3 + }, + { + "block": "warped_door", + "speed": 3 + }, + { + "block": "crimson_planks", + "speed": 3 + }, + { + "block": "warped_planks", + "speed": 3 + }, + { + "block": "crimson_trapdoor", + "speed": 3 + }, + { + "block": "warped_trapdoor", + "speed": 3 + }, + { + "block": "crimson_fence", + "speed": 3 + }, + { + "block": "warped_fence", + "speed": 3 + }, + { + "block": "crimson_fence_gate", + "speed": 3 + }, + { + "block": "warped_fence_gate", + "speed": 3 + }, + { + "block": "crimson_double_slab", + "speed": 3 + }, + { + "block": "crimson_slab", + "speed": 3 + }, + { + "block": "warped_double_slab", + "speed": 3 + }, + { + "block": "warped_slab", + "speed": 3 + }, + { + "block": "crimson_stairs", + "speed": 3 + }, + { + "block": "warped_stairs", + "speed": 3 + }, + { + "block": "crimson_button", + "speed": 3 + }, + { + "block": "warped_button", + "speed": 3 + }, + { + "block": "crimson_pressure_plate", + "speed": 3 + }, + { + "block": "warped_pressure_plate", + "speed": 3 + }, + { + "block": "soul_campfire", + "speed": 3 + }, + { + "block": "mangrove_log", + "speed": 3 + }, + { + "block": "stripped_mangrove_log", + "speed": 3 + }, + { + "block": "mangrove_planks", + "speed": 3 + }, + { + "block": "mangrove_button", + "speed": 3 + }, + { + "block": "mangrove_stairs", + "speed": 3 + }, + { + "block": "mangrove_slab", + "speed": 3 + }, + { + "block": "mangrove_pressure_plate", + "speed": 3 + }, + { + "block": "mangrove_fence", + "speed": 3 + }, + { + "block": "mangrove_fence_gate", + "speed": 3 + }, + { + "block": "mangrove_door", + "speed": 3 + }, + { + "block": "mangrove_standing_sign", + "speed": 3 + }, + { + "block": "mangrove_wall_sign", + "speed": 3 + }, + { + "block": "mangrove_trapdoor", + "speed": 3 + }, + { + "block": "mangrove_wood", + "speed": 3 + }, + { + "block": "stripped_mangrove_wood", + "speed": 3 + }, + { + "block": "mangrove_double_slab", + "speed": 3 + }, + { + "block": "planks", + "speed": 3 + }, + { + "block": "log", + "speed": 3 + }, + { + "block": "stripped_oak_log", + "speed": 3 + }, + { + "block": "stripped_birch_log", + "speed": 3 + }, + { + "block": "stripped_dark_oak_log", + "speed": 3 + }, + { + "block": "stripped_acacia_log", + "speed": 3 + }, + { + "block": "stripped_jungle_log", + "speed": 3 + }, + { + "block": "stripped_spruce_log", + "speed": 3 + }, + { + "block": "noteblock", + "speed": 3 + }, + { + "block": "jukebox", + "speed": 3 + }, + { + "block": "bed", + "speed": 3 + }, + { + "block": "bookshelf", + "speed": 3 + }, + { + "block": "torch", + "speed": 3 + }, + { + "block": "oak_stairs", + "speed": 3 + }, + { + "block": "chest", + "speed": 3 + }, + { + "block": "crafting_table", + "speed": 3 + }, + { + "block": "standing_sign", + "speed": 3 + }, + { + "block": "wooden_door", + "speed": 3 + }, + { + "block": "ladder", + "speed": 3 + }, + { + "block": "wall_sign", + "speed": 3 + }, + { + "block": "wooden_pressure_plate", + "speed": 3 + }, + { + "block": "acacia_pressure_plate", + "speed": 3 + }, + { + "block": "birch_pressure_plate", + "speed": 3 + }, + { + "block": "dark_oak_pressure_plate", + "speed": 3 + }, + { + "block": "jungle_pressure_plate", + "speed": 3 + }, + { + "block": "spruce_pressure_plate", + "speed": 3 + }, + { + "block": "unlit_redstone_torch", + "speed": 3 + }, + { + "block": "redstone_torch", + "speed": 3 + }, + { + "block": "fence", + "speed": 3 + }, + { + "block": "pumpkin", + "speed": 3 + }, + { + "block": "carved_pumpkin", + "speed": 3 + }, + { + "block": "lit_pumpkin", + "speed": 3 + }, + { + "block": "trapdoor", + "speed": 3 + }, + { + "block": "acacia_trapdoor", + "speed": 3 + }, + { + "block": "birch_trapdoor", + "speed": 3 + }, + { + "block": "dark_oak_trapdoor", + "speed": 3 + }, + { + "block": "jungle_trapdoor", + "speed": 3 + }, + { + "block": "spruce_trapdoor", + "speed": 3 + }, + { + "block": "brown_mushroom_block", + "speed": 3 + }, + { + "block": "red_mushroom_block", + "speed": 3 + }, + { + "block": "melon_block", + "speed": 3 + }, + { + "block": "pumpkin_stem", + "speed": 3 + }, + { + "block": "melon_stem", + "speed": 3 + }, + { + "block": "fence_gate", + "speed": 3 + }, + { + "block": "cocoa", + "speed": 3 + }, + { + "block": "spruce_stairs", + "speed": 3 + }, + { + "block": "birch_stairs", + "speed": 3 + }, + { + "block": "jungle_stairs", + "speed": 3 + }, + { + "block": "wooden_button", + "speed": 3 + }, + { + "block": "acacia_button", + "speed": 3 + }, + { + "block": "birch_button", + "speed": 3 + }, + { + "block": "dark_oak_button", + "speed": 3 + }, + { + "block": "jungle_button", + "speed": 3 + }, + { + "block": "spruce_button", + "speed": 3 + }, + { + "block": "trapped_chest", + "speed": 3 + }, + { + "block": "daylight_detector", + "speed": 3 + }, + { + "block": "double_wooden_slab", + "speed": 3 + }, + { + "block": "wooden_slab", + "speed": 3 + }, + { + "block": "log2", + "speed": 3 + }, + { + "block": "acacia_stairs", + "speed": 3 + }, + { + "block": "dark_oak_stairs", + "speed": 3 + }, + { + "block": "daylight_detector_inverted", + "speed": 3 + }, + { + "block": "spruce_fence_gate", + "speed": 3 + }, + { + "block": "birch_fence_gate", + "speed": 3 + }, + { + "block": "jungle_fence_gate", + "speed": 3 + }, + { + "block": "dark_oak_fence_gate", + "speed": 3 + }, + { + "block": "acacia_fence_gate", + "speed": 3 + }, + { + "block": "spruce_door", + "speed": 3 + }, + { + "block": "birch_door", + "speed": 3 + }, + { + "block": "jungle_door", + "speed": 3 + }, + { + "block": "acacia_door", + "speed": 3 + }, + { + "block": "dark_oak_door", + "speed": 3 + }, + { + "block": "standing_banner", + "speed": 3 + }, + { + "block": "wall_banner", + "speed": 3 + }, + { + "block": "bamboo", + "speed": 3 + }, + { + "block": "bamboo_sapling", + "speed": 3 + }, + { + "block": "scaffolding", + "speed": 3 + }, + { + "block": "spruce_wall_sign", + "speed": 3 + }, + { + "block": "spruce_standing_sign", + "speed": 3 + }, + { + "block": "birch_wall_sign", + "speed": 3 + }, + { + "block": "birch_standing_sign", + "speed": 3 + }, + { + "block": "jungle_wall_sign", + "speed": 3 + }, + { + "block": "jungle_standing_sign", + "speed": 3 + }, + { + "block": "acacia_wall_sign", + "speed": 3 + }, + { + "block": "acacia_standing_sign", + "speed": 3 + }, + { + "block": "darkoak_wall_sign", + "speed": 3 + }, + { + "block": "darkoak_standing_sign", + "speed": 3 + }, + { + "block": "barrel", + "speed": 3 + }, + { + "block": "smithing_table", + "speed": 3 + }, + { + "block": "cartography_table", + "speed": 3 + }, + { + "block": "fletching_table", + "speed": 3 + }, + { + "block": "campfire", + "speed": 3 + }, + { + "block": "loom", + "speed": 3 + }, + { + "block": "lectern", + "speed": 3 + }, + { + "block": "wood", + "speed": 3 + }, + { + "block": "composter", + "speed": 3 + }, + { + "block": "chorus_flower", + "speed": 3 + }, + { + "block": "chorus_plant", + "speed": 3 + } + ] + } + } + } } \ No newline at end of file diff --git a/BP/manifest.json b/BP/manifest.json index 701e12c..2d2e441 100644 --- a/BP/manifest.json +++ b/BP/manifest.json @@ -1,18 +1,18 @@ { "format_version": 2, "header": { - "name": "Lumber Axe BP: Chopping made easy! 1.0.5 [DEBUG]", - "description": "Cut the whole tree with easy as a lumberjack's pie! We create addons/datapack that will ease your survival plays. \n @Created By: @h_YanG_0A & @Brilliant", + "name": "Lumber Axe BP §8(2.0.0)", + "description": "An addon that gives you a lumberjack axes, which can preview the tree's status, and chopdown trees. \n @Created By: @Adr-hyng", "uuid": "4538bab6-a78b-4710-89e5-a2e103e97a6a", "version": [ - 1, + 2, 0, - 5 + 0 ], "min_engine_version": [ 1, - 20, - 10 + 21, + 0 ] }, "modules": [ @@ -32,9 +32,9 @@ "language": "javascript", "uuid": "9717780a-c69f-4679-bbe6-dbf6dfcdd7de", "version": [ - 1, + 2, 0, - 5 + 0 ], "entry": "scripts/main.js" } @@ -45,29 +45,26 @@ "dependencies": [ { "module_name": "@minecraft/server", - "version": "1.9.0-beta" + "version": "1.13.0" }, { "module_name": "@minecraft/server-ui", - "version": "1.2.0-beta" - }, - { - "module_name": "@minecraft/server-admin", - "version": "1.0.0-beta" + "version": "1.2.0" }, { "uuid": "62b4cab4-fa42-405b-bff6-1f47b6958dd6", "version": [ - 1, + 2, 0, - 5 + 0 ] } ], "metadata": { "authors": [ "@h_YanG_0A", - "@Brilliant" + "Contributor-@Brilliant", + "Contributor-@Dal4y" ], "license": "GPL-3.0-or-later", "url": "https://twitter.com/h_YanG_0A" diff --git a/BP/pack_icon.png b/BP/pack_icon.png index a7674c9..d29dae1 100644 Binary files a/BP/pack_icon.png and b/BP/pack_icon.png differ diff --git a/BP/recipes/smithing_netherite_lumber_axe.json b/BP/recipes/smithing_netherite_lumber_axe.json index c74d2e3..c952f07 100644 --- a/BP/recipes/smithing_netherite_lumber_axe.json +++ b/BP/recipes/smithing_netherite_lumber_axe.json @@ -1,5 +1,5 @@ { - "format_version": "1.12", + "format_version": "1.20.10", "minecraft:recipe_smithing_transform": { "description": { "identifier": "yn:smithing_netherite_lumber_axe" diff --git a/BP/scripts/classes/entity_override.js b/BP/scripts/classes/entity_override.js new file mode 100644 index 0000000..c0134fe --- /dev/null +++ b/BP/scripts/classes/entity_override.js @@ -0,0 +1,3 @@ +import { Entity } from "@minecraft/server"; +import { OverTakes } from "./partial_overtakes"; +OverTakes(Entity.prototype, {}); diff --git a/BP/scripts/classes/item_equippable.js b/BP/scripts/classes/item_equippable.js new file mode 100644 index 0000000..297bdbd --- /dev/null +++ b/BP/scripts/classes/item_equippable.js @@ -0,0 +1,41 @@ +import { ItemComponentTypes } from "@minecraft/server"; +import { EntityEquippableComponent, EquipmentSlot } from "@minecraft/server"; +import { MinecraftEnchantmentTypes, MinecraftItemTypes } from "modules/vanilla-types/index"; +import { OverTakes } from "./partial_overtakes"; +OverTakes(EntityEquippableComponent.prototype, { + get equipment() { + return this.getEquipment(EquipmentSlot.Mainhand); + }, + get isEquipped() { + return (this.equipment?.typeId === MinecraftItemTypes.FishingRod); + }, + damageDurability(damageApplied) { + const equipmentToDamage = this.getEquipment(EquipmentSlot.Mainhand); + if (!equipmentToDamage) + return false; + const player = this.entity; + if (!player.isSurvival()) + return false; + if (!equipmentToDamage?.hasComponent(ItemComponentTypes.Durability)) + throw "Item doesn't have durability to damage with"; + let level = 0; + const itemDurability = equipmentToDamage.getComponent(ItemComponentTypes.Durability); + if (equipmentToDamage.hasComponent(ItemComponentTypes.Enchantable)) { + const enchantment = equipmentToDamage.getComponent(ItemComponentTypes.Enchantable); + if (enchantment.hasEnchantment(MinecraftEnchantmentTypes.Unbreaking)) + level = enchantment.getEnchantment(MinecraftEnchantmentTypes.Unbreaking).level; + } + const unbreakingMultiplier = (100 / (level + 1)) / 100; + const unbreakingDamage = damageApplied * unbreakingMultiplier; + if (itemDurability.damage + unbreakingDamage >= itemDurability.maxDurability) { + this.setEquipment(EquipmentSlot.Mainhand, undefined); + player.playSound("random.break"); + return true; + } + else if (itemDurability.damage + unbreakingDamage < itemDurability.maxDurability) { + equipmentToDamage.getComponent(ItemComponentTypes.Durability).damage += unbreakingDamage; + this.setEquipment(EquipmentSlot.Mainhand, equipmentToDamage); + return false; + } + } +}); diff --git a/BP/scripts/classes/itemstack_override.js b/BP/scripts/classes/itemstack_override.js new file mode 100644 index 0000000..e69de29 diff --git a/BP/scripts/classes/partial_overtakes.js b/BP/scripts/classes/partial_overtakes.js new file mode 100644 index 0000000..c2fdf5e --- /dev/null +++ b/BP/scripts/classes/partial_overtakes.js @@ -0,0 +1,6 @@ +export function OverTakes(prototype, object) { + const prototypeOrigin = Object.setPrototypeOf(Object.defineProperties({}, Object.getOwnPropertyDescriptors(prototype)), Object.getPrototypeOf(prototype)); + Object.setPrototypeOf(object, prototypeOrigin); + Object.defineProperties(prototype, Object.getOwnPropertyDescriptors(object)); + return prototypeOrigin; +} diff --git a/BP/scripts/classes/player.js b/BP/scripts/classes/player.js index a1ef473..a762991 100644 --- a/BP/scripts/classes/player.js +++ b/BP/scripts/classes/player.js @@ -1,4 +1,15 @@ import { GameMode, Player } from "@minecraft/server"; -Player.prototype.isSurvival = function () { - return this.dimension.getPlayers({ gameMode: GameMode.survival, name: this.name, location: this.location, maxDistance: 1, closest: 1 }).length > 0; -}; +import { OverTakes } from "./partial_overtakes"; +import { Configuration } from "configuration/configuration_screen"; +const screenConfigs = new WeakMap(); +OverTakes(Player.prototype, { + isSurvival() { + return this.getGameMode() === GameMode.survival; + }, + get configuration() { + let sc = screenConfigs.get(this); + if (!sc) + screenConfigs.set(this, sc = new Configuration(this)); + return sc; + } +}); diff --git a/BP/scripts/commands/ICommandBase.js b/BP/scripts/commands/ICommandBase.js new file mode 100644 index 0000000..cb0ff5c --- /dev/null +++ b/BP/scripts/commands/ICommandBase.js @@ -0,0 +1 @@ +export {}; diff --git a/BP/scripts/commands/command_handler.js b/BP/scripts/commands/command_handler.js new file mode 100644 index 0000000..cc6eb99 --- /dev/null +++ b/BP/scripts/commands/command_handler.js @@ -0,0 +1,10 @@ +import { ADDON_IDENTIFIER } from "constant"; +export const CommandHandler = { + prefix: `/scriptevent ${ADDON_IDENTIFIER} `, + commands: [ + 'help', + 'config', + 'database', + 'dev_helper' + ] +}; diff --git a/BP/scripts/commands/config.js b/BP/scripts/commands/config.js new file mode 100644 index 0000000..1e2c596 --- /dev/null +++ b/BP/scripts/commands/config.js @@ -0,0 +1,58 @@ +import { system } from "@minecraft/server"; +import { CommandHandler } from "commands/command_handler"; +import { SendMessageTo } from "utils/utilities"; +import "classes/player"; +var REQUIRED_PARAMETER; +(function (REQUIRED_PARAMETER) { + REQUIRED_PARAMETER["SHOW"] = "show"; + REQUIRED_PARAMETER["RESET"] = "reset"; +})(REQUIRED_PARAMETER || (REQUIRED_PARAMETER = {})); +var OPTIONAL_PARAMETER; +(function (OPTIONAL_PARAMETER) { + OPTIONAL_PARAMETER["CLIENT"] = "client"; + OPTIONAL_PARAMETER["SERVER"] = "server"; +})(OPTIONAL_PARAMETER || (OPTIONAL_PARAMETER = {})); +const command = { + name: 'config', + description: 'Show or reset configuration settings.', + format: `[${Object.values(REQUIRED_PARAMETER).join('|')}] [<${Object.values(OPTIONAL_PARAMETER).join('|')}>?]`, + usage() { + return (` + Format: + > ${CommandHandler.prefix}${this.name} ${this.format} + Usage: + > ${CommandHandler.prefix}${this.name} ${REQUIRED_PARAMETER.SHOW} = Shows config + > ${CommandHandler.prefix}${this.name} ${REQUIRED_PARAMETER.RESET} ${OPTIONAL_PARAMETER.CLIENT} = Reset caller client config + > ${CommandHandler.prefix}${this.name} ${REQUIRED_PARAMETER.RESET} ${OPTIONAL_PARAMETER.SERVER} = Reset world server config (Admin) + `).replaceAll(" ", ""); + }, + async execute(player, args) { + if (args && args.length) { + const requiredParams = (`[${Object.values(REQUIRED_PARAMETER).join('|')}]`).slice(1, -1).split('|').map(command => command.trim()); + const selectedReqParam = args[0].toLowerCase(); + const isShow = REQUIRED_PARAMETER.SHOW === selectedReqParam; + if (!requiredParams.includes(selectedReqParam)) + return SendMessageTo(player, { + rawtext: [ + { + translate: "LumberAxe.on_caught_invalid_command", + with: [command.usage()] + }, + ] + }); + if (isShow) { + system.run(() => player.configuration.showServerScreen()); + } + else { + const optionalParams = (`[${Object.values(OPTIONAL_PARAMETER).join('|')}]`).slice(1, -1).split('|').map(command => command.trim()); + const selectedOptParam = args[1]?.toLowerCase(); + let shouldResetClient = OPTIONAL_PARAMETER.CLIENT === selectedOptParam; + if (!optionalParams.includes(selectedOptParam)) + shouldResetClient = true; + if (shouldResetClient) + player.configuration.reset("CLIENT"); + } + } + } +}; +export default command; diff --git a/BP/scripts/commands/database.js b/BP/scripts/commands/database.js new file mode 100644 index 0000000..9598afc --- /dev/null +++ b/BP/scripts/commands/database.js @@ -0,0 +1,79 @@ +import { world } from "@minecraft/server"; +import { CommandHandler } from "commands/command_handler"; +import { originalDatabase, ADDON_NAME } from "constant"; +import { SendMessageTo } from "utils/utilities"; +var REQUIRED_PARAMETER; +(function (REQUIRED_PARAMETER) { + REQUIRED_PARAMETER["SHOW"] = "show"; + REQUIRED_PARAMETER["RESET"] = "reset"; +})(REQUIRED_PARAMETER || (REQUIRED_PARAMETER = {})); +const command = { + name: 'database', + description: 'Inspect or reset a database.', + format: `[${Object.values(REQUIRED_PARAMETER).join('|')}]`, + usage() { + return (` + Format: + > ${CommandHandler.prefix}${this.name} ${this.format} + Usage: + > ${CommandHandler.prefix}${this.name} ${REQUIRED_PARAMETER.SHOW} = Display database content. + > ${CommandHandler.prefix}${this.name} ${REQUIRED_PARAMETER.RESET} = Reset database content. + `).replaceAll(" ", ""); + }, + execute(player, args) { + if (args && args.length) { + const requiredParams = (`[${Object.values(REQUIRED_PARAMETER).join('|')}]`).slice(1, -1).split('|').map(command => command.trim()); + const selectedReqParam = args[0].toLowerCase(); + const isShow = REQUIRED_PARAMETER.SHOW === selectedReqParam; + if (!requiredParams.includes(selectedReqParam)) + return SendMessageTo(player, { + rawtext: [ + { + translate: "LumberAxe.on_caught_invalid_command", + with: [command.usage()] + }, + ] + }); + if (isShow) { + if (originalDatabase.size === 0) + return SendMessageTo(player, { + rawtext: [ + { + translate: "LumberAxe.on_database_empty" + }, + ] + }); + let collections = ""; + let i = 1; + for (const key of originalDatabase.keys()) { + const t = key.split("|"); + const player = world.getEntity(t[1]); + collections += `${i++}. ${player.nameTag}: ${JSON.stringify(t)}\n`; + } + SendMessageTo(player, { + rawtext: [ + { + translate: "LumberAxe.show_database", + with: [ADDON_NAME, "\n", collections] + }, + ] + }); + } + else { + SendMessageTo(player, { + rawtext: [ + { + translate: "LumberAxe.on_database_reset" + }, + ] + }); + player.configuration.reset("CLIENT"); + player.configuration.reset("SERVER"); + originalDatabase.clear(); + if (!originalDatabase.isDisposed) + originalDatabase.dispose(); + } + } + } +}; +export default command; diff --git a/BP/scripts/commands/dev_helper.js b/BP/scripts/commands/dev_helper.js new file mode 100644 index 0000000..b2ca262 --- /dev/null +++ b/BP/scripts/commands/dev_helper.js @@ -0,0 +1,58 @@ +import { EntityComponentTypes, ItemStack } from "@minecraft/server"; +import { CommandHandler } from "commands/command_handler"; +import { SendMessageTo } from "utils/utilities"; +import { axeEquipments, originalDatabase, resetOriginalDatabase, visitedLogs } from "constant"; +var REQUIRED_PARAMETER; +(function (REQUIRED_PARAMETER) { + REQUIRED_PARAMETER["GET"] = "get"; + REQUIRED_PARAMETER["TEST"] = "test"; + REQUIRED_PARAMETER["RELOAD"] = "reload"; +})(REQUIRED_PARAMETER || (REQUIRED_PARAMETER = {})); +const command = { + name: 'dev_helper', + description: 'Developer Utility Command', + format: `[${Object.values(REQUIRED_PARAMETER).join('|')}]`, + usage() { + return (` + Format: + > ${CommandHandler.prefix}${this.name} ${this.format} + Usage: + > ${CommandHandler.prefix}${this.name} ${REQUIRED_PARAMETER.GET} = GETS an enchanted fishing rod for development. + > ${CommandHandler.prefix}${this.name} ${REQUIRED_PARAMETER.TEST} = TEST a Working-in-progress features. + > ${CommandHandler.prefix}${this.name} ${REQUIRED_PARAMETER.RELOAD} = Reloads the addon. + `).replaceAll(" ", ""); + }, + execute(player, args) { + if (!(args && args.length)) + return; + const requiredParams = (`[${Object.values(REQUIRED_PARAMETER).join('|')}]`).slice(1, -1).split('|').map(command => command.trim()); + const selectedReqParam = args[0].toLowerCase(); + if (!requiredParams.includes(selectedReqParam)) + return SendMessageTo(player, { + rawtext: [ + { + translate: "LumberAxe.on_caught_invalid_command", + with: [command.usage()] + }, + ] + }); + switch (selectedReqParam) { + case REQUIRED_PARAMETER.GET: + for (const axe of axeEquipments) { + player.getComponent(EntityComponentTypes.Inventory).container.addItem(new ItemStack(axe, 1)); + } + break; + case REQUIRED_PARAMETER.TEST: + console.warn(originalDatabase.size, visitedLogs.length); + break; + case REQUIRED_PARAMETER.RELOAD: + originalDatabase.clear(); + resetOriginalDatabase(); + console.warn(originalDatabase.isValid(), originalDatabase.size); + break; + default: + break; + } + } +}; +export default command; diff --git a/BP/scripts/commands/help.js b/BP/scripts/commands/help.js new file mode 100644 index 0000000..6b30778 --- /dev/null +++ b/BP/scripts/commands/help.js @@ -0,0 +1,81 @@ +import { CommandHandler } from './command_handler'; +import { ADDON_NAME } from 'constant'; +import { SendMessageTo } from 'utils/utilities'; +const importCommand = async (player, commandName) => { + try { + const importedCommandModule = await import(`./${commandName}.js`); + return importedCommandModule.default; + } + catch (error) { + SendMessageTo(player, { + rawtext: [ + { + translate: "LumberAxe.on_caught_command_404", + with: [commandName, error.message] + } + ] + }); + return null; + } +}; +const details = { + __addonName__: ADDON_NAME, + __name__: 'help', + __description__: 'Displays the help message.', + __format__: '[?]', +}; +const command = { + name: details.__name__, + description: details.__description__, + format: details.__format__, + usage() { + return (`Format: + > ${CommandHandler.prefix}${this.name} ${this.format} + Usage: + > ${CommandHandler.prefix}${this.name} + > ${CommandHandler.prefix}${this.name} config + `).replaceAll(" ", ""); + }, + async execute(player, args) { + if (!args || args.length === 0) { + let helpMessage = `\n§aCommands available @ ${details.__addonName__}: \n`; + for (const commandName of CommandHandler.commands) { + const importedCommand = await importCommand(player, commandName); + if (importedCommand) + helpMessage += `§e${CommandHandler.prefix}${commandName}§r${importedCommand.format.length ? " " + importedCommand.format : ""} - ${importedCommand.description}\n`; + } + SendMessageTo(player, { + rawtext: [ + { + text: helpMessage + } + ] + }); + } + else { + const specifiedCommand = args[0].toLowerCase(); + if (!CommandHandler.commands.includes(specifiedCommand)) + return SendMessageTo(player, { + rawtext: [ + { + translate: "LumberAxe.on_caught_invalid_command", + with: [command.usage()] + }, + ] + }); + if (CommandHandler.commands.includes(specifiedCommand)) { + const importedCommand = await importCommand(player, specifiedCommand); + if (importedCommand) { + SendMessageTo(player, { + rawtext: [ + { + text: `\n§e${CommandHandler.prefix}${specifiedCommand}: \n${importedCommand.description}§r ${importedCommand.usage()}` + } + ] + }); + } + } + } + } +}; +export default command; diff --git a/BP/scripts/config.js b/BP/scripts/config.js deleted file mode 100644 index a46374d..0000000 --- a/BP/scripts/config.js +++ /dev/null @@ -1,9 +0,0 @@ -export default { - debug: true, - durabilityDamagePerBlock: 3, - chopLimit: 300, - includedLog: [], - excludedLog: [], - disableWatchDogTerminateLog: true, -}; -export const VERSION = "1.0.5"; diff --git a/BP/scripts/configuration/configuration_handler.js b/BP/scripts/configuration/configuration_handler.js new file mode 100644 index 0000000..e499462 --- /dev/null +++ b/BP/scripts/configuration/configuration_handler.js @@ -0,0 +1,22 @@ +import { ADDON_NAME } from "constant"; +import { FormBuilder } from "utils/form_builder"; +export const ConfigurationCollections_DB = (player, configType = "CLIENT") => `${ADDON_NAME}|${player.id}|${configType}`; +export function cloneConfiguration(config) { + let clonedConfig = {}; + for (const [key, _formBuilder] of Object.entries(config)) { + const formBuilder = _formBuilder; + const isArrayEmpty = formBuilder.values.length > 0; + const newFormBuilder = new FormBuilder(formBuilder.name); + if (typeof formBuilder.defaultValue === "string" && isArrayEmpty) { + newFormBuilder.createDropdown(formBuilder.values, formBuilder.defaultValue); + } + else if (typeof formBuilder.defaultValue === "string" && !isArrayEmpty) { + newFormBuilder.createTextField(formBuilder.defaultValue); + } + else if (typeof formBuilder.defaultValue === "boolean") { + newFormBuilder.createToggle(formBuilder.defaultValue); + } + clonedConfig[key] = newFormBuilder; + } + return clonedConfig; +} diff --git a/BP/scripts/configuration/configuration_screen.js b/BP/scripts/configuration/configuration_screen.js new file mode 100644 index 0000000..eedde74 --- /dev/null +++ b/BP/scripts/configuration/configuration_screen.js @@ -0,0 +1,258 @@ +import { ActionFormData, FormCancelationReason, ModalFormData } from "@minecraft/server-ui"; +import { ConfigurationCollections_DB } from "./configuration_handler"; +import { ADDON_NAME, originalDatabase, resetOriginalDatabase } from "constant"; +import { resetServerConfiguration, serverConfigurationCopy, setServerConfiguration } from "./server_configuration"; +import { SendMessageTo } from "utils/utilities"; +export class Configuration { + constructor(player) { + this.player = player; + this.isConfigurationSettingsOpen = false; + this.CLIENT_CONFIGURATION_DB = ConfigurationCollections_DB(this.player, "CLIENT"); + this.SERVER_CONFIGURATION_DB = ConfigurationCollections_DB(this.player, "SERVER"); + } + reset(configurationType) { + if (originalDatabase.isValid()) { + if (configurationType === "SERVER") { + resetServerConfiguration(); + originalDatabase.set(this.SERVER_CONFIGURATION_DB, serverConfigurationCopy); + } + } + else + throw new Error("Database not found"); + } + saveServer() { + setServerConfiguration(serverConfigurationCopy); + if (originalDatabase.isValid()) + originalDatabase.set(this.SERVER_CONFIGURATION_DB, serverConfigurationCopy); + else { + resetOriginalDatabase(); + } + } + loadServer() { + if (originalDatabase?.isValid()) { + if (originalDatabase.has(this.SERVER_CONFIGURATION_DB)) { + setServerConfiguration(originalDatabase.get(this.SERVER_CONFIGURATION_DB)); + } + else { + originalDatabase.set(this.SERVER_CONFIGURATION_DB, serverConfigurationCopy); + } + } + } + showServerScreen() { + const parsedAddonTitle = ADDON_NAME.toLowerCase().replace(/_/g, ' ').replace(/\b\w/g, char => char.toUpperCase()); + const form = new ActionFormData() + .title({ rawtext: [ + { translate: "LumberAxe.configuration.title", with: [parsedAddonTitle] } + ] }) + .button({ rawtext: [ + { translate: "LumberAxe.configuration.general" } + ] }) + .button({ rawtext: [ + { translate: "LumberAxe.configuration.log_include_manager" } + ] }) + .button({ rawtext: [ + { translate: "LumberAxe.configuration.log_exclude_manager" } + ] }); + form.show(this.player).then((response) => { + if (response.canceled || response.cancelationReason === FormCancelationReason.UserClosed || response.cancelationReason === FormCancelationReason.UserBusy) + return; + switch (response.selection) { + case 0: return this.showGeneralOptions(); + case 1: return this.showIncludeManager(); + case 2: return this.showExcludeManager(); + default: + break; + } + return; + }); + } + showGeneralOptions() { + const form = new ModalFormData().title({ rawtext: [ + { translate: "LumberAxe.configuration.general" } + ] }); + this.loadServer(); + const cachedConfigurationValues = []; + Object.values(serverConfigurationCopy).forEach((builder, index) => { + const isNotDropdown = (builder.values.length === 0); + if (typeof builder.defaultValue === "boolean" && isNotDropdown) { + cachedConfigurationValues.push({ result: builder.defaultValue, index }); + form.toggle({ rawtext: [{ translate: builder.name }] }, builder.defaultValue); + } + else if (typeof builder.defaultValue === "string" && isNotDropdown) { + cachedConfigurationValues.push({ result: builder.defaultValue, index }); + form.textField({ rawtext: [{ translate: builder.name }] }, builder.defaultValue, builder.defaultValue); + } + }); + form.show(this.player).then((result) => { + if (!result.formValues) + return; + const hadChanges = !cachedConfigurationValues.every(({ result: element }, i) => element === result.formValues[i]); + if (result.canceled || result.cancelationReason === FormCancelationReason.UserClosed || result.cancelationReason === FormCancelationReason.UserBusy) { + return; + } + if (hadChanges) { + result.formValues.forEach((newValue, formIndex) => { + const index = cachedConfigurationValues[formIndex].index; + const key = Object.keys(serverConfigurationCopy)[index]; + const builder = serverConfigurationCopy[key]; + switch (typeof newValue) { + case "boolean": + builder.defaultValue = newValue; + break; + case "number": + builder.defaultValue = builder.values[newValue]; + break; + case "string": + builder.defaultValue = newValue; + break; + default: + break; + } + serverConfigurationCopy[key] = builder; + }); + this.saveServer(); + } + return this.showServerScreen(); + }); + } + showIncludeManager() { + this.loadServer(); + const preResultFlags = []; + let index = 0; + preResultFlags[index] = 0; + index++; + preResultFlags[index] = ""; + index++; + preResultFlags[index] = false; + index++; + const form = new ModalFormData() + .title({ rawtext: [ + { translate: "LumberAxe.configuration.log_include_manager" } + ] }) + .dropdown({ rawtext: [ + { translate: "LumberAxe.log_include_manager.drop_down" } + ] }, [...serverConfigurationCopy.includedLog.values], 0) + .textField({ rawtext: [ + { translate: "LumberAxe.log_include_manager.text_field" } + ] }, "myaddon:custom_log", preResultFlags[1]) + .toggle({ rawtext: [ + { translate: "LumberAxe.log_include_manager.toggle" } + ] }, preResultFlags[2]); + form.show(this.player).then((response) => { + if (!response.formValues) + return; + if (response.canceled || response.cancelationReason === FormCancelationReason.UserClosed || response.cancelationReason === FormCancelationReason.UserBusy) + return; + const hadChanges = !preResultFlags.every((element, index) => element === response.formValues[index]); + const selectedIndex = response.formValues[0]; + let canUpdate = response.formValues[2]; + const dropDownContent = response.formValues[1]; + const dropDownSelected = selectedIndex; + const isEmpty = !dropDownContent.length; + if (!hadChanges) + return this.showServerScreen(); + if (isEmpty) + canUpdate = false; + if (canUpdate) { + if (selectedIndex === 0) { + serverConfigurationCopy.includedLog.values.push(dropDownContent); + SendMessageTo(this.player, { rawtext: [ + { text: `§aLumber Axe: ` }, + { translate: "LumberAxe.log_include_manager.add_success_log", with: [dropDownContent] } + ] }); + } + else { + if (serverConfigurationCopy.includedLog.values.length) { + const prevValue = serverConfigurationCopy.includedLog.values[dropDownSelected]; + serverConfigurationCopy.includedLog.values[dropDownSelected] = dropDownContent; + SendMessageTo(this.player, { rawtext: [ + { text: `§eLumber Axe: ` }, + { translate: "LumberAxe.log_include_manager.update_success_log", with: [prevValue, dropDownContent] } + ] }); + } + } + } + else { + if (serverConfigurationCopy.includedLog.values.length && selectedIndex !== 0) { + const itemDeleted = serverConfigurationCopy.includedLog.values.splice(dropDownSelected, 1)[0]; + SendMessageTo(this.player, { rawtext: [ + { text: `§cLumber Axe: ` }, + { translate: "LumberAxe.log_include_manager.remove_success_log", with: [itemDeleted] } + ] }); + } + } + this.saveServer(); + return this.showServerScreen(); + }); + } + showExcludeManager() { + this.loadServer(); + const preResultFlags = []; + let index = 0; + preResultFlags[index] = 0; + index++; + preResultFlags[index] = ""; + index++; + preResultFlags[index] = false; + index++; + const form = new ModalFormData() + .title({ rawtext: [ + { translate: "LumberAxe.configuration.log_exclude_manager" } + ] }) + .dropdown({ rawtext: [ + { translate: "LumberAxe.log_exclude_manager.drop_down" } + ] }, [...serverConfigurationCopy.excludedLog.values], 0) + .textField({ rawtext: [ + { translate: "LumberAxe.log_exclude_manager.text_field" } + ] }, "myaddon:custom_log", preResultFlags[1]) + .toggle({ rawtext: [ + { translate: "LumberAxe.log_exclude_manager.toggle" } + ] }, preResultFlags[2]); + form.show(this.player).then((response) => { + if (!response.formValues) + return; + if (response.canceled || response.cancelationReason === FormCancelationReason.UserClosed || response.cancelationReason === FormCancelationReason.UserBusy) + return; + const hadChanges = !preResultFlags.every((element, index) => element === response.formValues[index]); + const selectedIndex = response.formValues[0]; + let canUpdate = response.formValues[2]; + const dropDownContent = response.formValues[1]; + const dropDownSelected = selectedIndex; + const isEmpty = !dropDownContent.length; + if (!hadChanges) + return this.showServerScreen(); + if (isEmpty) + canUpdate = false; + if (canUpdate) { + if (selectedIndex === 0) { + serverConfigurationCopy.excludedLog.values.push(dropDownContent); + SendMessageTo(this.player, { rawtext: [ + { text: `§aLumber Axe: ` }, + { translate: "LumberAxe.log_exclude_manager.add_success_log", with: [dropDownContent] } + ] }); + } + else { + if (serverConfigurationCopy.excludedLog.values.length) { + const prevValue = serverConfigurationCopy.includedLog.values[dropDownSelected]; + serverConfigurationCopy.excludedLog.values[dropDownSelected] = dropDownContent; + SendMessageTo(this.player, { rawtext: [ + { text: `§eLumber Axe: ` }, + { translate: "LumberAxe.log_exclude_manager.update_success_log", with: [prevValue, dropDownContent] } + ] }); + } + } + } + else { + if (serverConfigurationCopy.excludedLog.values.length && selectedIndex !== 0) { + const itemDeleted = serverConfigurationCopy.excludedLog.values.splice(dropDownSelected, 1)[0]; + SendMessageTo(this.player, { rawtext: [ + { text: `§cLumber Axe: ` }, + { translate: "LumberAxe.log_exclude_manager.remove_success_log", with: [itemDeleted] } + ] }); + } + } + this.saveServer(); + return this.showServerScreen(); + }); + } +} diff --git a/BP/scripts/configuration/server_configuration.js b/BP/scripts/configuration/server_configuration.js new file mode 100644 index 0000000..e67cd24 --- /dev/null +++ b/BP/scripts/configuration/server_configuration.js @@ -0,0 +1,45 @@ +import { FormBuilder } from "utils/form_builder"; +import { cloneConfiguration } from "./configuration_handler"; + +export const serverConfiguration = { + /** + * Lumber Axe durability damage per log destroyed. + */ + durabilityDamagePerBlock: new FormBuilder("LumberAxe.server.durability_damage_per_block").createTextField("3"), + /** + * Delay for immersive mode. + */ + immersiveModeDelay: new FormBuilder("LumberAxe.server.immersive_delay").createTextField("5"), + /** + * Tree chop limitation for control purposes. + */ + chopLimit: new FormBuilder("LumberAxe.server.chop_limit").createTextField("1000"), + /** + * Included blocks for custom logs, but any custom or vanilla logs also work as long as the block identifier ends with "*_log". + * Check: https://github.com/mcbe-mods/Cut-tree-one-click by Lete114. + */ + includedLog: new FormBuilder("LumberAxe.configuration.log_include_manager").createDropdown(['Empty'], "Empty"), + /** + * Excluded blocks for block logs you don't want to be included in being chopped. + * + * Tip: + * - excludedLog is prioritized over includedLog. + * - It's unnecessary to include log blocks that have "*_log" in their block id. + */ + excludedLog: new FormBuilder("LumberAxe.configuration.log_exclude_manager").createDropdown(['Empty'], "Empty"), + /** + * Enable/Disable Progressive Chopping, which makes you chop trees slightly longer, but nice to see. + */ + immersiveMode: new FormBuilder("LumberAxe.server.immersive_chopping").createToggle(false), + /** + * Enables debug messages to content logs. + */ + debug: new FormBuilder("Debug Mode").createToggle(false), +}; + +export let serverConfigurationCopy = cloneConfiguration(serverConfiguration); +export let setServerConfiguration = (newServerConfig) => serverConfigurationCopy = newServerConfig; +export let resetServerConfiguration = () => serverConfigurationCopy = cloneConfiguration(serverConfiguration); + +// version (do not change) +export const VERSION = "2.0.0"; \ No newline at end of file diff --git a/BP/scripts/constant.js b/BP/scripts/constant.js new file mode 100644 index 0000000..758a0cd --- /dev/null +++ b/BP/scripts/constant.js @@ -0,0 +1,29 @@ +import { JsonDatabase } from "./utils/Database/con-database"; +import { MyCustomItemTypes } from 'items/CustomItemTypes'; +import { system } from "@minecraft/server"; +export const ADDON_NAMESPACE = "yn"; +export const ADDON_NAME = "Lumber_Axe"; +export const ADDON_IDENTIFIER = `${ADDON_NAMESPACE}:lumber`; +export let originalDatabase = new JsonDatabase(ADDON_NAME); +export const resetOriginalDatabase = () => { + originalDatabase = new JsonDatabase(ADDON_NAME); +}; +export const playerInteractedTimeLogMap = new Map(); +export const axeEquipments = Object.values(MyCustomItemTypes); +export const visitedLogs = []; +export function resetOutlinedTrees(result, instantDespawn = false) { + if (result.isDone) + return; + result.isDone = true; + if (!instantDespawn) + visitedLogs?.shift(); + const t = system.runJob((function* () { + for (const blockOutline of result.visitedLogs.blockOutlines) { + if (blockOutline?.isValid()) { + blockOutline.triggerEvent('despawn'); + } + yield; + } + system.clearJob(t); + })()); +} diff --git a/BP/scripts/functions/tree_utils.js b/BP/scripts/functions/tree_utils.js index 5f557a1..23287cf 100644 --- a/BP/scripts/functions/tree_utils.js +++ b/BP/scripts/functions/tree_utils.js @@ -1,134 +1,182 @@ -import { EntityEquippableComponent, EquipmentSlot, ItemDurabilityComponent, ItemEnchantableComponent, ItemLockMode, ItemStack, system } from "@minecraft/server"; -import { MinecraftBlockTypes, MinecraftEnchantmentTypes } from "../modules/vanilla-types/index"; -import { validLogBlocks, axeEquipments, stackDistribution, durabilityDamagePerBlock, excludedLog, includedLog, chopLimit } from "../index"; -async function treeCut(player, dimension, location, blockTypeId) { - const equipment = player.getComponent(EntityEquippableComponent.componentId); - const currentHeldAxe = equipment.getEquipment(EquipmentSlot.Mainhand); - if (!axeEquipments.includes(currentHeldAxe?.typeId)) - return; - if (!isLogIncluded(blockTypeId)) - return; - if (!player.isSurvival()) - return; - if (player.isSurvival()) - currentHeldAxe.lockMode = ItemLockMode.slot; - const itemDurability = currentHeldAxe.getComponent(ItemDurabilityComponent.componentId); - const enchantments = (currentHeldAxe.getComponent(ItemEnchantableComponent.componentId)); - const level = enchantments.getEnchantment(MinecraftEnchantmentTypes.Unbreaking)?.level | 0; - const unbreakingMultiplier = (100 / (level + 1)) / 100; - const unbreakingDamage = durabilityDamagePerBlock * unbreakingMultiplier; - const visited = await getTreeLogs(dimension, location, blockTypeId, (itemDurability.maxDurability - itemDurability.damage) / unbreakingDamage); - const totalDamage = visited.size * unbreakingDamage; - const postDamagedDurability = itemDurability.damage + totalDamage; - if (postDamagedDurability + 1 === itemDurability.maxDurability) { - equipment.setEquipment(EquipmentSlot.Mainhand, undefined); +import { system } from "@minecraft/server"; +import { serverConfigurationCopy, originalDatabase, hashBlock } from "../index"; +import { Graph } from "utils/graph"; +import { Vec3 } from "utils/VectorUtils"; +export function isLogIncluded(rootBlockTypeId, blockTypeId) { + const validLogBlocks = /(_log|_wood|crimson_stem|warped_stem|(?:brown|red_)?mushroom_block)$/; + function extractLogFamily(blockTypeId) { + const [, cleanedBlockTypeId] = blockTypeId.split(':'); + const parts = cleanedBlockTypeId.split('_'); + return parts.slice(0, -1).join('_'); } - else if (postDamagedDurability > itemDurability.maxDurability) { - currentHeldAxe.lockMode = ItemLockMode.none; - return; - } - else if (postDamagedDurability < itemDurability.maxDurability) { - itemDurability.damage = itemDurability.damage + totalDamage; - currentHeldAxe.lockMode = ItemLockMode.none; - equipment.setEquipment(EquipmentSlot.Mainhand, currentHeldAxe.clone()); - } - for (const group of groupAdjacentBlocks(visited)) { - const firstElement = JSON.parse(group[0]); - const lastElement = JSON.parse(group[group.length - 1]); - if (firstElement === lastElement) { - await new Promise((resolve) => { - dimension.getBlock(firstElement).setType(MinecraftBlockTypes.Air); - resolve(); - }); - continue; - } - else { - await new Promise((resolve) => { - dimension.fillBlocks(firstElement, lastElement, MinecraftBlockTypes.Air); - resolve(); - }); - } - } - system.runTimeout(async () => { - for (const group of stackDistribution(visited.size)) { - await new Promise((resolve) => { - dimension.spawnItem(new ItemStack(blockTypeId, group), location); - resolve(); - }); - } - }, 5); -} -function isLogIncluded(blockTypeId) { - if (excludedLog.includes(blockTypeId) || blockTypeId.includes('stripped_')) + if (serverConfigurationCopy.excludedLog.values.includes(blockTypeId) || blockTypeId.includes('stripped_')) return false; - if (includedLog.includes(blockTypeId) || validLogBlocks.test(blockTypeId)) + const extractedLogFamily = extractLogFamily(rootBlockTypeId); + const blockFamily = extractLogFamily(blockTypeId); + const isSameFamily = blockFamily === extractedLogFamily; + if ((serverConfigurationCopy.includedLog.values.includes(blockTypeId) || + validLogBlocks.test(blockTypeId)) && isSameFamily) return true; return false; } -function getTreeLogs(dimension, location, blockTypeId, maxNeeded) { - return new Promise((resolve) => { - const traversingTreeInterval = system.runInterval(() => { - const visited = new Set(); - let queue = getBlockNear(dimension, location); +export async function getTreeLogs(dimension, location, blockTypeId, maxNeeded, isInspectingTree = true) { + const firstBlock = dimension.getBlock(location); + const visitedTree = await new Promise((resolve) => { + const graph = new Graph(); + const visitedTypeIDs = new Map(); + const queue = [firstBlock]; + const yOffsets = new Map(); + const visited = new Set([JSON.stringify(firstBlock.location)]); + visitedTypeIDs.set(blockTypeId, 0); + const traversingTreeInterval = system.runJob(function* () { + graph.addNode(firstBlock); + originalDatabase.set(`visited_${hashBlock(firstBlock)}`, isInspectingTree); while (queue.length > 0) { - if (visited.size >= chopLimit || visited.size >= maxNeeded) { - system.clearRun(traversingTreeInterval); - resolve(visited); + const size = graph.getSize(); + if (size >= parseInt(serverConfigurationCopy.chopLimit.defaultValue + "") || size >= maxNeeded) { + break; } - const _block = queue.shift(); - if (!_block || !isLogIncluded(_block?.typeId)) - continue; - if (_block.typeId !== blockTypeId) - continue; - const pos = JSON.stringify(_block.location); - if (visited.has(pos)) + const block = queue.shift(); + const mainNode = graph.getNode(block); + if (!mainNode) continue; - visited.add(pos); - queue.push(...getBlockNear(dimension, _block.location)); + yOffsets.set(block.y, false); + for (const neighborBlock of getBlockNear(blockTypeId, block)) { + const serializedLocation = JSON.stringify(neighborBlock.location); + let neighborNode = graph.getNode(neighborBlock) ?? graph.addNode(neighborBlock); + originalDatabase.set(`visited_${hashBlock(neighborBlock)}`, isInspectingTree); + if (mainNode.neighbors.has(neighborNode)) + continue; + mainNode.addNeighbor(neighborNode); + neighborNode.addNeighbor(mainNode); + if (visited.has(serializedLocation)) + continue; + visited.add(serializedLocation); + queue.push(neighborBlock); + let currentAmount = visitedTypeIDs.get(neighborBlock.typeId) ?? 0; + currentAmount += 1; + visitedTypeIDs.set(neighborBlock.typeId, currentAmount); + yield; + } + yield; + } + system.clearJob(traversingTreeInterval); + resolve({ + source: graph, + blockOutlines: [], + yOffsets, + typeIds: visitedTypeIDs, + trunk: { + size: 0, + center: { x: 0, z: 0 } + } + }); + }()); + }); + const blockOutlines = []; + const trunk = await getTreeTrunkSize(firstBlock, blockTypeId); + return new Promise((resolve) => { + const t = system.runJob((function* () { + for (const yOffset of visitedTree.yOffsets.keys()) { + const outline = dimension.spawnEntity('yn:block_outline', { + x: trunk.center.x, + y: yOffset, + z: trunk.center.z + }); + outline.lastLocation = JSON.parse(JSON.stringify(outline.location)); + blockOutlines.push(outline); + yield; } - queue = []; - system.clearRun(traversingTreeInterval); - resolve(visited); - }, 1); + for (const blockOutline of blockOutlines) { + if (blockOutline?.isValid()) { + blockOutline.triggerEvent('not_persistent'); + } + yield; + } + system.clearJob(t); + resolve({ + typeIds: visitedTree.typeIds, + source: visitedTree.source, + blockOutlines: blockOutlines, + trunk: trunk, + yOffsets: visitedTree.yOffsets + }); + return; + })()); }); } -function getBlockNear(dimension, location, radius = 1) { - const centerX = location.x; - const centerY = location.y; - const centerZ = location.z; - const positions = []; +function* getBlockNear(initialBlockTypeID, initialBlock, radius = 1) { + const centerX = 0; + const centerY = 0; + const centerZ = 0; let _block; for (let x = centerX - radius; x <= centerX + radius; x++) { for (let y = centerY - radius; y <= centerY + radius; y++) { for (let z = centerZ - radius; z <= centerZ + radius; z++) { if (centerX === x && centerY === y && centerZ === z) continue; - _block = dimension.getBlock({ x, y, z }); - if (_block.isAir) + _block = initialBlock.offset({ x, y, z }); + if (!_block?.isValid() || !isLogIncluded(initialBlockTypeID, _block?.typeId)) continue; - positions.push(_block); + yield _block; } } } - return positions; } -function groupAdjacentBlocks(visited) { - const array = Array.from(visited).map(item => JSON.parse(item)); - array.sort((a, b) => a.x - b.x || a.z - b.z || a.y - b.y); - const groups = []; - let currentGroup = []; - for (let i = 0; i < array.length; i++) { - if (i === 0 || (array[i].x === array[i - 1].x && array[i].z === array[i - 1].z && Math.abs(array[i].y - JSON.parse(currentGroup[currentGroup.length - 1]).y) <= 2)) { - currentGroup.push(JSON.stringify(array[i])); - } - else { - groups.push(currentGroup); - currentGroup = [JSON.stringify(array[i])]; - } - } - if (currentGroup.length > 0) { - groups.push(currentGroup); - } - return groups; +export function getTreeTrunkSize(blockInteracted, blockTypeId) { + return new Promise((fetchedTrunkSizeResolved) => { + let i = 0; + let centroidLog = { + x: 0, + z: 0 + }; + const visited = new Set(); + const queue = [blockInteracted]; + const originalY = blockInteracted.y; + const t = system.runJob((function* () { + while (queue.length > 0) { + const currentBlock = queue.shift(); + if ((!currentBlock || !currentBlock.isValid()) && !Vec3.equals(blockInteracted, currentBlock)) + continue; + const blockKey = JSON.stringify({ x: currentBlock.x, z: currentBlock.z }); + if (visited.has(blockKey)) + continue; + visited.add(blockKey); + centroidLog.x += currentBlock.x; + centroidLog.z += currentBlock.z; + i++; + for (let y = -1; y <= 1; y++) { + const newY = currentBlock.y + y; + if (newY < originalY - 2 || newY > originalY + 2) + continue; + for (let x = -1; x <= 1; x++) { + for (let z = -1; z <= 1; z++) { + if (x === 0 && z === 0 && y === 0) + continue; + const neighborBlock = currentBlock.offset({ x: x, y: y, z: z }); + const neighborLoc = JSON.stringify({ x: neighborBlock.x, z: neighborBlock.z }); + if (!neighborBlock?.isValid() || visited.has(neighborLoc) || !isLogIncluded(blockTypeId, neighborBlock.typeId)) + continue; + queue.push(neighborBlock); + yield; + } + yield; + } + yield; + } + yield; + } + if (i <= 1) { + i = 1; + centroidLog = blockInteracted.center(); + } + else { + centroidLog.x = (centroidLog.x / i) + 0.5; + centroidLog.z = (centroidLog.z / i) + 0.5; + } + system.clearJob(t); + fetchedTrunkSizeResolved({ center: centroidLog, size: i }); + return; + })()); + }); } -export { treeCut, isLogIncluded, getTreeLogs }; diff --git a/BP/scripts/functions/utils.js b/BP/scripts/functions/utils.js deleted file mode 100644 index f15acd3..0000000 --- a/BP/scripts/functions/utils.js +++ /dev/null @@ -1,23 +0,0 @@ -import { system } from "@minecraft/server"; -import { FormCancelationReason } from "@minecraft/server-ui"; -function stackDistribution(number, groupSize = 64) { - const fullGroupsCount = Math.floor(number / groupSize); - const remainder = number % groupSize; - const groups = new Array(fullGroupsCount).fill(groupSize); - if (remainder > 0) { - groups.push(remainder); - } - return groups; -} -async function forceShow(player, form, timeout = Infinity) { - const startTick = system.currentTick; - while ((system.currentTick - startTick) < timeout) { - const response = await (form.show(player)).catch(er => console.error(er, er.stack)); - if (response.cancelationReason !== FormCancelationReason.UserBusy) { - return response; - } - } - ; - throw new Error(`Timed out after ${timeout} ticks`); -} -export { stackDistribution, forceShow }; diff --git a/BP/scripts/index.js b/BP/scripts/index.js index 1619ba1..654cf11 100644 --- a/BP/scripts/index.js +++ b/BP/scripts/index.js @@ -1,8 +1,8 @@ -export * from './functions/utils'; +export * from './utils/utilities'; export * from './functions/tree_utils'; export * from './classes/player'; -import Configuration from "./config"; -const { durabilityDamagePerBlock, chopLimit, includedLog, excludedLog, disableWatchDogTerminateLog } = Configuration; -export { durabilityDamagePerBlock, chopLimit, includedLog, excludedLog, disableWatchDogTerminateLog }; -export const validLogBlocks = /(_log|crimson_stem|warped_stem)$/; -export const axeEquipments = ["yn:wooden_lumber_axe", "yn:stone_lumber_axe", "yn:iron_lumber_axe", "yn:diamond_lumber_axe", "yn:golden_lumber_axe", "yn:netherite_lumber_axe"]; +export * from './classes/entity_override'; +export * from './classes/item_equippable'; +export * from "configuration/server_configuration"; +export * from "constant"; +export * from "items/axes"; diff --git a/BP/scripts/items/CustomItemTypes.js b/BP/scripts/items/CustomItemTypes.js new file mode 100644 index 0000000..8a1443d --- /dev/null +++ b/BP/scripts/items/CustomItemTypes.js @@ -0,0 +1,9 @@ +export var MyCustomItemTypes; +(function (MyCustomItemTypes) { + MyCustomItemTypes["WoodenLumberAxe"] = "yn:wooden_lumber_axe"; + MyCustomItemTypes["StoneLumberAxe"] = "yn:stone_lumber_axe"; + MyCustomItemTypes["IronLumberAxe"] = "yn:iron_lumber_axe"; + MyCustomItemTypes["DiamondLumberAxe"] = "yn:diamond_lumber_axe"; + MyCustomItemTypes["GoldenLumberAxe"] = "yn:golden_lumber_axe"; + MyCustomItemTypes["NetheriteLumberAxe"] = "yn:netherite_lumber_axe"; +})(MyCustomItemTypes || (MyCustomItemTypes = {})); diff --git a/BP/scripts/items/axes.js b/BP/scripts/items/axes.js new file mode 100644 index 0000000..2ecf83f --- /dev/null +++ b/BP/scripts/items/axes.js @@ -0,0 +1,22 @@ +import { EntityEquippableComponent, Player, world } from "@minecraft/server"; +import "classes/player"; +world.beforeEvents.worldInitialize.subscribe((registry) => { + registry.itemComponentRegistry.registerCustomComponent('yn:tool_durability', { + onHitEntity(arg) { + if (!(arg.attackingEntity instanceof Player)) + return; + const player = arg.attackingEntity; + if (!player.isSurvival()) + return; + const axe = player.getComponent(EntityEquippableComponent.componentId); + axe.damageDurability(1); + }, + onUseOn(arg) { + }, + onMineBlock(arg) { + const player = arg.source; + const axe = player.getComponent(EntityEquippableComponent.componentId); + axe.damageDurability(2); + }, + }); +}); diff --git a/BP/scripts/main.js b/BP/scripts/main.js index 49ee3ac..18119f5 100644 --- a/BP/scripts/main.js +++ b/BP/scripts/main.js @@ -1,122 +1,604 @@ -import { world, system, ItemDurabilityComponent, ItemEnchantableComponent, WatchdogTerminateReason } from '@minecraft/server'; -import { FormCancelationReason, ActionFormData } from "@minecraft/server-ui"; -import { disableWatchDogTerminateLog, durabilityDamagePerBlock, axeEquipments, forceShow, getTreeLogs, isLogIncluded, treeCut } from "./index"; -import { MinecraftEnchantmentTypes } from './modules/vanilla-types/index'; -const logMap = new Map(); -const playerInteractionMap = new Map(); -system.beforeEvents.watchdogTerminate.subscribe((e) => { - e.cancel = true; - if (e.terminateReason === WatchdogTerminateReason.Hang) { - for (const key of playerInteractionMap.keys()) { - playerInteractionMap.set(key, false); - } - if (!disableWatchDogTerminateLog) - world.sendMessage({ - rawtext: [ - { - translate: "LumberAxe.watchdogError.hang.text" - } - ] - }); - if (disableWatchDogTerminateLog) - console.warn(`Scripting Error: Try chopping or inspecting smaller trees or different angle.`); +import { world, system, ScriptEventSource, Player, EntityInventoryComponent, ItemDurabilityComponent, ItemEnchantableComponent, ItemLockMode, ItemStack, MolangVariableMap, TicksPerSecond, ItemCooldownComponent } from '@minecraft/server'; +import { ADDON_IDENTIFIER, axeEquipments, originalDatabase, forceShow, getTreeLogs, getTreeTrunkSize, hashBlock, isLogIncluded, playerInteractedTimeLogMap, resetOutlinedTrees, SendMessageTo, serverConfigurationCopy, stackDistribution, visitedLogs } from "./index"; +import { Logger } from 'utils/logger'; +import './items/axes'; +import { MinecraftEnchantmentTypes, MinecraftBlockTypes } from 'modules/vanilla-types/index'; +import { Graph } from 'utils/graph'; +import { Vec3 } from 'utils/VectorUtils'; +import { ActionFormData, FormCancelationReason } from '@minecraft/server-ui'; +const BLOCK_OUTLINES_DESPAWN_TIMER = 5; +world.afterEvents.playerSpawn.subscribe((e) => { + if (!e.initialSpawn) + return; + e.player.configuration.loadServer(); + if (!originalDatabase.has(`playerFirstJoined-${e.player.id}`)) { + originalDatabase.set(`playerFirstJoined-${e.player.id}`, false); + } + if (!originalDatabase.get(`playerFirstJoined-${e.player.id}`)) { + originalDatabase.set(`playerFirstJoined-${e.player.id}`, true); + SendMessageTo(e.player, { + rawtext: [ + { + translate: "LumberAxe.on_load_message" + } + ] + }); } - console.warn(`Watchdog Error: ${e.terminateReason}`); -}); -world.afterEvents.playerLeave.subscribe((e) => { - playerInteractionMap.set(e.playerId, false); }); -world.afterEvents.playerBreakBlock.subscribe(async (e) => { - const { dimension, player, block } = e; - const currentBreakBlock = e.brokenBlockPermutation; +world.beforeEvents.playerBreakBlock.subscribe((arg) => { + const player = arg.player; + const dimension = player.dimension; + const blockInteracted = arg.block; + const location = blockInteracted.location; + const currentHeldAxe = arg.itemStack; + const currentHeldAxeSlot = player.selectedSlotIndex; + const currentBreakBlock = arg.block.permutation; const blockTypeId = currentBreakBlock.type.id; - treeCut(player, dimension, block.location, blockTypeId); -}); -world.beforeEvents.itemUseOn.subscribe((e) => { - const currentHeldAxe = e.itemStack; - const blockInteracted = e.block; - const player = e.source; - const oldLog = logMap.get(player.name); - logMap.set(player.name, Date.now()); - if ((oldLog + 1000) >= Date.now()) + if (!player.isSurvival()) + return; + if (!isLogIncluded(blockTypeId, blockTypeId)) return; - if (!axeEquipments.includes(currentHeldAxe.typeId) || !isLogIncluded(blockInteracted.typeId)) + if (originalDatabase.has(`visited_${hashBlock(blockInteracted)}`) && !originalDatabase.get(`visited_${hashBlock(blockInteracted)}`)) { + arg.cancel = true; + return; + } + if (!axeEquipments.includes(currentHeldAxe.typeId)) + return; + const possibleVisitedLogs = []; + for (let i = 0; i < visitedLogs.length; i++) { + const currentInspectedTree = visitedLogs[i]; + const interactedTreeNode = currentInspectedTree.visitedLogs.source.getNode(blockInteracted); + if (interactedTreeNode) { + possibleVisitedLogs.push({ result: currentInspectedTree, index: i }); + } + } + let initialTreeInspection; + if (possibleVisitedLogs.length) { + const latestPossibleInspectedTree = possibleVisitedLogs[possibleVisitedLogs.length - 1]; + const index = latestPossibleInspectedTree.index; + initialTreeInspection = latestPossibleInspectedTree.result; + if (initialTreeInspection.isBeingChopped) { + arg.cancel = true; + return; + } + visitedLogs.splice(index, 1); + initialTreeInspection.isDone = true; + } + player.configuration.loadServer(); + system.run(async () => { + currentHeldAxe.lockMode = ItemLockMode.slot; + const inventory = player.getComponent(EntityInventoryComponent.componentId).container; + inventory.setItem(currentHeldAxeSlot, currentHeldAxe); + const itemDurability = currentHeldAxe.getComponent(ItemDurabilityComponent.componentId); + const enchantments = currentHeldAxe.getComponent(ItemEnchantableComponent.componentId); + const level = enchantments.getEnchantment(MinecraftEnchantmentTypes.Unbreaking)?.level | 0; + const unbreakingMultiplier = (100 / (level + 1)) / 100; + const unbreakingDamage = +serverConfigurationCopy.durabilityDamagePerBlock.defaultValue * unbreakingMultiplier; + let visited; + let destroyedTree = { + isBeingChopped: true, + initialSize: 0, + isDone: false, + visitedLogs: { + typeIds: new Map(), + blockOutlines: [], + source: new Graph(), + yOffsets: new Map(), + trunk: { + center: { + x: 0, + z: 0 + }, + size: 0 + } + } + }; + const molang = new MolangVariableMap(); + let isTreeDoneTraversing = false; + const brokenTreeTrunk = await getTreeTrunkSize(blockInteracted, blockTypeId); + const topMostBlock = blockInteracted.dimension.getTopmostBlock(brokenTreeTrunk.center); + const bottomMostBlock = await new Promise((getBottomMostBlockResolved) => { + let _bottom = blockInteracted.below(); + const _t = system.runInterval(() => { + if (!isLogIncluded(blockInteracted.typeId, _bottom.typeId)) { + system.clearRun(_t); + getBottomMostBlockResolved(_bottom); + return; + } + _bottom = _bottom.below(); + }); + }); + const mainTreeTrunkHeight = (topMostBlock.y - bottomMostBlock.y); + const isValidVerticalTree = mainTreeTrunkHeight > 2; + if (isValidVerticalTree) { + let dustRadius = 1; + molang.setFloat('trunk_size', dustRadius); + player.playSound('hit.stem'); + dimension.spawnParticle('yn:tree_dust', { x: brokenTreeTrunk.center.x, y: blockInteracted.y, z: brokenTreeTrunk.center.z }, molang); + const t = system.runInterval(() => { + molang.setFloat('trunk_size', dustRadius += 0.25); + if (isTreeDoneTraversing) { + system.clearRun(t); + return; + } + ; + player.playSound('hit.stem'); + dimension.spawnParticle('yn:tree_dust', { x: brokenTreeTrunk.center.x, y: blockInteracted.y, z: brokenTreeTrunk.center.z }, molang); + }, 12); + } + const choppedTree = initialTreeInspection === undefined ? await getTreeLogs(dimension, location, blockTypeId, (itemDurability.maxDurability - itemDurability.damage) / unbreakingDamage, false) : initialTreeInspection.visitedLogs; + isTreeDoneTraversing = true; + destroyedTree.visitedLogs = choppedTree; + visited = choppedTree.source; + const initialSize = visited.getSize() - 1; + visitedLogs.push(destroyedTree); + if (!visited) + return; + if (initialSize >= +serverConfigurationCopy.chopLimit.defaultValue) { + currentHeldAxe.lockMode = ItemLockMode.none; + inventory.setItem(currentHeldAxeSlot, currentHeldAxe); + SendMessageTo(player, { rawtext: [{ translate: `LumberAxe.server.invalid_log_amount_limitation`, with: [serverConfigurationCopy.chopLimit.defaultValue] }] }); + return await new Promise((resolve) => { + system.runJob((function* () { + for (const node of destroyedTree.visitedLogs.source.traverseIterative(blockInteracted, "BFS")) { + yield; + if (!node) + continue; + originalDatabase.delete(`visited_${hashBlock(node.block)}`); + yield; + } + resetOutlinedTrees(destroyedTree); + resolve(); + return; + })()); + }); + } + const totalDamage = initialSize * unbreakingDamage; + const postDamagedDurability = itemDurability.damage + totalDamage; + if (postDamagedDurability + 1 === itemDurability.maxDurability) { + player.playSound("random.break"); + inventory.setItem(currentHeldAxeSlot, undefined); + } + else if (postDamagedDurability > itemDurability.maxDurability) { + currentHeldAxe.lockMode = ItemLockMode.none; + inventory.setItem(currentHeldAxeSlot, currentHeldAxe); + return; + } + else if (postDamagedDurability < itemDurability.maxDurability) { + itemDurability.damage = itemDurability.damage + totalDamage; + const heldTemp = currentHeldAxe.clone(); + heldTemp.lockMode = ItemLockMode.none; + inventory.setItem(currentHeldAxeSlot, heldTemp); + } + const treeDustParseMap = { + 0: 1, + 1: 3.25, + 2: 4, + 3: 4, + 4: 4, + 5: 7, + 6: 7, + 7: 7, + 8: 7, + 9: 7 + }; + const trunkYCoordinates = Array.from(destroyedTree.visitedLogs.yOffsets.keys()).sort((a, b) => a - b); + const getTreeDustValue = (key) => key > 9 ? 7 : treeDustParseMap[key]; + molang.setFloat('trunk_size', getTreeDustValue(brokenTreeTrunk.size)); + let currentBlockOffset = 0; + if (serverConfigurationCopy.immersiveMode.defaultValue && isValidVerticalTree) { + for (const yOffset of trunkYCoordinates) { + if (currentBlockOffset % 2 === 0) { + await system.waitTicks(+serverConfigurationCopy.immersiveModeDelay.defaultValue); + const loc = { x: destroyedTree.visitedLogs.trunk.center.x, y: yOffset, z: destroyedTree.visitedLogs.trunk.center.z }; + player.playSound('mob.irongolem.crack', { location: loc }); + const molang = new MolangVariableMap(); + molang.setFloat('trunk_size', getTreeDustValue(destroyedTree.visitedLogs.trunk.size)); + dimension.spawnParticle('yn:tree_dust', loc, molang); + } + destroyedTree.visitedLogs.yOffsets.set(yOffset, true); + currentBlockOffset++; + } + } + let size = 0; + system.runJob((function* () { + if (!(serverConfigurationCopy.immersiveMode.defaultValue) && isValidVerticalTree) { + for (const yOffset of trunkYCoordinates) { + if (currentBlockOffset % 2 === 0) { + const molang = new MolangVariableMap(); + molang.setFloat('trunk_size', getTreeDustValue(destroyedTree.visitedLogs.trunk.size)); + dimension.spawnParticle('yn:tree_dust', { x: destroyedTree.visitedLogs.trunk.center.x, y: yOffset, z: destroyedTree.visitedLogs.trunk.center.z }, molang); + } + currentBlockOffset++; + yield; + } + } + const blockOutlineIterator = destroyedTree.visitedLogs.blockOutlines[Symbol.iterator](); + let blockOutlineIterResult = blockOutlineIterator.next(); + while (!blockOutlineIterResult.done) { + const blockOutline = blockOutlineIterResult.value; + if (blockOutline?.isValid()) { + blockOutline.setProperty('yn:trunk_size', destroyedTree.visitedLogs.trunk.size); + } + blockOutlineIterResult = blockOutlineIterator.next(); + yield; + } + for (const node of destroyedTree.visitedLogs.source.traverseIterative(blockInteracted, "BFS")) { + yield; + if (Vec3.equals(node.block, blockInteracted.location)) + continue; + if (!node) + continue; + if (isLogIncluded(blockTypeId, node.block.typeId)) { + size++; + system.waitTicks(5).then(() => { + dimension.setBlockType(node.block.location, MinecraftBlockTypes.Air); + }); + } + else { + destroyedTree.visitedLogs.source.removeNode(node.block); + break; + } + yield; + } + player.playSound('dig.cave_vines'); + for (const [typeIDs, typeIDSize] of choppedTree.typeIds.entries()) { + for (const stackedAmount of stackDistribution(typeIDSize)) { + dimension.spawnItem(new ItemStack(typeIDs, stackedAmount), location); + yield; + } + yield; + } + return; + })()); + await system.waitTicks(3); + system.runJob((function* () { + for (const node of destroyedTree.visitedLogs.source.traverseIterative(blockInteracted, "BFS")) { + if (node) { + originalDatabase.delete(`visited_${hashBlock(node.block)}`); + } + yield; + } + if (!destroyedTree?.isDone) + resetOutlinedTrees(destroyedTree); + return; + })()); + }); +}); +world.beforeEvents.itemUseOn.subscribe(async (arg) => { + const currentHeldAxe = arg.itemStack; + const blockInteracted = arg.block; + const player = arg.source; + if (!axeEquipments.includes(currentHeldAxe.typeId) || !isLogIncluded(blockInteracted.typeId, blockInteracted.typeId)) return; - if (playerInteractionMap.get(player.id)) + const oldLog = playerInteractedTimeLogMap.get(player.id); + playerInteractedTimeLogMap.set(player.id, system.currentTick); + if ((oldLog + 10) >= system.currentTick) return; - playerInteractionMap.set(player.id, true); + player.configuration.loadServer(); const itemDurability = currentHeldAxe.getComponent(ItemDurabilityComponent.componentId); const enchantments = currentHeldAxe.getComponent(ItemEnchantableComponent.componentId); const level = enchantments.getEnchantment(MinecraftEnchantmentTypes.Unbreaking)?.level | 0; const currentDurability = itemDurability.damage; const maxDurability = itemDurability.maxDurability; const unbreakingMultiplier = (100 / (level + 1)) / 100; - const unbreakingDamage = durabilityDamagePerBlock * unbreakingMultiplier; + const unbreakingDamage = +serverConfigurationCopy.durabilityDamagePerBlock.defaultValue * unbreakingMultiplier; const reachableLogs = (maxDurability - currentDurability) / unbreakingDamage; - getTreeLogs(player.dimension, blockInteracted.location, blockInteracted.typeId, reachableLogs + 1).then((treeCollected) => { - const totalDamage = (treeCollected.size) * unbreakingDamage; - const totalDurabilityConsumed = currentDurability + totalDamage; - const canBeChopped = (totalDurabilityConsumed === maxDurability) || (totalDurabilityConsumed < maxDurability); - const inspectionForm = new ActionFormData() - .title({ - rawtext: [ - { - translate: "LumberAxe.form.title.text" + const cooldown = currentHeldAxe.getComponent(ItemCooldownComponent.componentId); + let BLOCK_OUTLINES_DESPAWN_CD = BLOCK_OUTLINES_DESPAWN_TIMER * TicksPerSecond; + try { + if (!visitedLogs) + return; + const tempResult = await new Promise((inspectTreePromiseResolve) => { + const tMain = system.runJob((function* (inspectTreePromiseResolve) { + const possibleVisitedLogs = []; + for (let i = 0; i < visitedLogs.length; i++) { + const currentInspectedTree = visitedLogs[i]; + const interactedTreeNode = currentInspectedTree.visitedLogs.source.getNode(blockInteracted); + if (interactedTreeNode) { + possibleVisitedLogs.push({ result: currentInspectedTree, index: i }); + } } - ] - }) - .button({ - rawtext: [ - { - translate: `LumberAxe.form.treeSizeAbrev.text` - }, - { - text: ` ${treeCollected.size !== 0 ? treeCollected.size : 1}${canBeChopped ? "" : "+"} ` - }, - { - translate: `LumberAxe.form.treeSizeAbrevLogs.text` + if (!possibleVisitedLogs.length) { + if (originalDatabase.has(`visited_${hashBlock(blockInteracted)}`)) { + inspectTreePromiseResolve({ result: null, index: -100 }); + return system.clearJob(tMain); + } + inspectTreePromiseResolve({ result: null, index: -1 }); + return system.clearJob(tMain); } - ] - }, "textures/InfoUI/blocks.png") - .button({ - rawtext: [ - { - translate: `LumberAxe.form.durabilityAbrev.text` - }, - { - text: ` ${currentDurability}` + const latestPossibleInspectedTree = possibleVisitedLogs[possibleVisitedLogs.length - 1]; + const index = latestPossibleInspectedTree.index; + const initialTreeInspection = latestPossibleInspectedTree.result; + if (initialTreeInspection.isBeingChopped) { + inspectTreePromiseResolve({ result: null, index: -100 }); + return system.clearJob(tMain); } - ] - }, "textures/InfoUI/axe_durability.png") - .button({ - rawtext: [ - { - translate: `LumberAxe.form.maxDurabilityAbrev.text` - }, - { - text: ` ${maxDurability}` + for (const node of initialTreeInspection.visitedLogs.source.traverseIterative(blockInteracted, "BFS")) { + if (!node.block?.isValid() || !isLogIncluded(blockInteracted.typeId, node.block.typeId)) { + initialTreeInspection.visitedLogs.source.removeNode(node.block); + originalDatabase.delete(`visited_${hashBlock(node.block)}`); + } + yield; } - ] - }, "textures/InfoUI/required_durability.png") - .button({ - rawtext: [ - { - text: "§l" - }, - { - translate: `${canBeChopped ? "LumberAxe.form.canBeChopped.text" : "LumberAxe.form.cannotBeChopped.text"}` + if (initialTreeInspection.initialSize === initialTreeInspection.visitedLogs.source.getSize()) { + system.clearJob(tMain); + inspectTreePromiseResolve({ result: initialTreeInspection.visitedLogs, index: index }); } - ] - }, "textures/InfoUI/canBeCut.png"); - forceShow(player, inspectionForm).then((response) => { - playerInteractionMap.set(player.id, false); - if (response.canceled || response.selection === undefined || response.cancelationReason === FormCancelationReason.UserClosed) - return; - }).catch((error) => { - console.warn("Form Error: ", error, error.stack); + const finalizedTreeInspection = { + blockOutlines: [], + typeIds: new Map(), + source: new Graph(), + yOffsets: new Map(), + trunk: { + center: { + x: 0, + z: 0 + }, + size: 0 + } + }; + for (const node of initialTreeInspection.visitedLogs.source.traverseIterative(blockInteracted, "BFS")) { + if (node.block?.isValid()) { + finalizedTreeInspection.blockOutlines.push(initialTreeInspection.visitedLogs.blockOutlines[node.index]); + finalizedTreeInspection.source.addNode(node); + finalizedTreeInspection.yOffsets.set(node.block.location.y, false); + } + yield; + } + const newInspectedSubTree = { + isBeingChopped: false, + initialSize: finalizedTreeInspection.source.getSize(), + isDone: false, + visitedLogs: finalizedTreeInspection + }; + const currentChangedIndex = visitedLogs.findIndex((result) => newInspectedSubTree.visitedLogs.source.isEqual(initialTreeInspection.visitedLogs.source) && !result.isDone); + if (currentChangedIndex === -1) { + if (newInspectedSubTree.initialSize > 0) + visitedLogs.push(newInspectedSubTree); + system.waitTicks(BLOCK_OUTLINES_DESPAWN_CD).then(async (_) => { + system.runJob((function* () { + for (const node of newInspectedSubTree.visitedLogs.source.traverseIterative(blockInteracted, "BFS")) { + originalDatabase.delete(`visited_${hashBlock(node.block)}`); + yield; + } + if (!visitedLogs[index]) + return; + if (!visitedLogs[index].isDone) + resetOutlinedTrees(newInspectedSubTree); + return; + })()); + }); + } + else { + visitedLogs[index] = newInspectedSubTree; + } + system.clearJob(tMain); + inspectTreePromiseResolve({ result: finalizedTreeInspection, index: index }); + })(inspectTreePromiseResolve)); }); - }).catch((error) => { - console.warn("Tree Error: ", error, error.stack); - playerInteractionMap.set(player.id, false); + if (tempResult.index === -1) { + if (cooldown.getCooldownTicksRemaining(player) !== 0) + return; + cooldown.startCooldown(player); + const molangVariable = new MolangVariableMap(); + let isTreeDoneTraversing = false; + let treeOffsets = []; + let result = { + isBeingChopped: false, + visitedLogs: { + typeIds: new Map(), + blockOutlines: [], + source: new Graph(), + trunk: { + center: { x: 0, z: 0 }, + size: 0 + }, + yOffsets: new Map() + }, + isDone: false, + initialSize: 0, + }; + let interactedTreeTrunk = await getTreeTrunkSize(blockInteracted, blockInteracted.typeId); + const topMostBlock = blockInteracted.dimension.getTopmostBlock(interactedTreeTrunk.center); + const bottomMostBlock = await new Promise((getBottomMostBlockResolved) => { + let _bottom = blockInteracted.below(); + const _t = system.runInterval(() => { + if (!isLogIncluded(blockInteracted.typeId, _bottom.typeId)) { + system.clearRun(_t); + getBottomMostBlockResolved(_bottom); + return; + } + _bottom = _bottom.below(); + }); + }); + const trunkSizeToParticleRadiusParser = { + 1: 1.5, + 2: 2.5, + 3: 2.5, + 4: 2.5, + 5: 3.5, + 6: 3.5, + 7: 3.5, + 8: 3.5, + 9: 3.5 + }; + let treeCollectedResult = null; + const trunkHeight = (topMostBlock.y - (bottomMostBlock.y + 1)); + const isValidVerticalTree = trunkHeight > 2; + if (isValidVerticalTree) { + const { x: centerX, z: centerZ } = interactedTreeTrunk.center; + const centerBlockErrorCatch = blockInteracted.dimension.getBlock({ x: centerX, y: blockInteracted.y, z: centerZ }); + if (!isLogIncluded(blockInteracted.typeId, centerBlockErrorCatch.typeId)) { + interactedTreeTrunk.size++; + } + const it = system.runInterval(() => { + if (result.isDone) { + system.clearRun(it); + return; + } + if (isTreeDoneTraversing) { + molangVariable.setFloat('radius', trunkSizeToParticleRadiusParser[treeCollectedResult.trunk.size]); + molangVariable.setFloat('height', treeOffsets.length); + molangVariable.setFloat('max_age', 1); + molangVariable.setColorRGB('color', { red: 0.0, green: 1.0, blue: 0.0 }); + } + else { + molangVariable.setFloat('radius', trunkSizeToParticleRadiusParser[interactedTreeTrunk.size]); + molangVariable.setFloat('height', trunkHeight); + molangVariable.setFloat('max_age', 1); + molangVariable.setColorRGB('color', { red: 1.0, green: 1.0, blue: 1.0 }); + } + player.dimension.spawnParticle('yn:inspecting_indicator', { + x: interactedTreeTrunk.center.x, + y: bottomMostBlock.y + 1, + z: interactedTreeTrunk.center.z + }, molangVariable); + }, 5); + } + treeCollectedResult = await getTreeLogs(player.dimension, blockInteracted.location, blockInteracted.typeId, +serverConfigurationCopy.chopLimit.defaultValue); + isTreeDoneTraversing = true; + if (isValidVerticalTree) { + treeOffsets = Array.from(treeCollectedResult.yOffsets.keys()).sort((a, b) => a - b); + const { x: centerX, z: centerZ } = treeCollectedResult.trunk.center; + const centerBlockErrorCatch = blockInteracted.dimension.getBlock({ x: centerX, y: blockInteracted.y, z: centerZ }); + if (!isLogIncluded(blockInteracted.typeId, centerBlockErrorCatch.typeId)) { + treeCollectedResult.trunk.size++; + } + } + else { + const t = system.runJob((function* () { + for (const node of treeCollectedResult.source.traverseIterative(blockInteracted, "BFS")) { + molangVariable.setFloat('radius', 1.1); + molangVariable.setFloat('height', 0.97); + molangVariable.setFloat('max_age', BLOCK_OUTLINES_DESPAWN_CD / TicksPerSecond); + molangVariable.setColorRGB('color', { red: 0.0, green: 1.0, blue: 0.0 }); + player.dimension.spawnParticle('yn:inspecting_indicator', { x: node.block.bottomCenter().x, y: node.block.y, z: node.block.bottomCenter().z }, molangVariable); + yield; + } + system.clearJob(t); + })()); + } + result = { + isBeingChopped: false, + visitedLogs: treeCollectedResult, + isDone: false, + initialSize: treeCollectedResult.source.getSize(), + }; + if (result.initialSize > 0) + visitedLogs.push(result); + system.runTimeout(() => { + system.runJob((function* () { + for (const node of treeCollectedResult.source.traverseIterative(blockInteracted, "BFS")) { + originalDatabase.delete(`visited_${hashBlock(node.block)}`); + yield; + } + if (!result?.isDone) + resetOutlinedTrees(result); + return; + })()); + }, BLOCK_OUTLINES_DESPAWN_CD); + } + else if (tempResult.index >= 0) { + const size = tempResult.result.source.getSize(); + const totalDamage = size * unbreakingDamage; + const totalDurabilityConsumed = currentDurability + totalDamage; + const canBeChopped = ((totalDurabilityConsumed === maxDurability) || (totalDurabilityConsumed < maxDurability)) && (size <= +serverConfigurationCopy.chopLimit.defaultValue); + const inspectionForm = new ActionFormData() + .title({ + rawtext: [ + { + translate: "LumberAxe.form.title.text" + } + ] + }) + .button({ + rawtext: [ + { + translate: `LumberAxe.form.treeSizeAbrev.text` + }, + { + text: ` ${size !== 0 ? Math.round(canBeChopped ? size : reachableLogs + 1) : 1}${canBeChopped ? "" : "+"} ` + }, + { + translate: `LumberAxe.form.treeSizeAbrevLogs.text` + } + ] + }, "textures/InfoUI/total_lumber.png") + .button({ + rawtext: [ + { + text: `${tempResult.result.yOffsets.size} ` + }, + { + translate: `LumberAxe.form.trunkHeightAbrev.text` + } + ] + }, "textures/InfoUI/tree_height.png") + .button({ + rawtext: [ + { + text: `${(maxDurability - totalDurabilityConsumed) > 0 ? '+' : ''}${Math.round(maxDurability - totalDurabilityConsumed)} ` + }, + { + translate: "LumberAxe.form.treeSizeAbrevLogs.text" + } + ] + }, (maxDurability - totalDurabilityConsumed) > 0 ? "textures/InfoUI/lumber_surplus.png" : "textures/InfoUI/lumber_deficit.png") + .button({ + rawtext: [ + { + translate: `${canBeChopped ? "LumberAxe.form.canBeChopped.text" : "LumberAxe.form.cannotBeChopped.text"}` + } + ] + }, canBeChopped ? "textures/InfoUI/can_be_cut.png" : "textures/InfoUI/cannot_be_cut.png"); + forceShow(player, inspectionForm).then((response) => { + if (response.canceled || response.selection === undefined || response.cancelationReason === FormCancelationReason.UserClosed) { + return; + } + }).catch((error) => { + Logger.error("Form Error: ", error, error.stack); + }); + } + } + catch (e) { + console.warn(e, e.stack); + } +}); +system.afterEvents.scriptEventReceive.subscribe((event) => { + if (event.sourceType !== ScriptEventSource.Entity) + return; + if (!(event.sourceEntity instanceof Player)) + return; + if (event.id !== ADDON_IDENTIFIER) + return; + const player = event.sourceEntity; + const message = event.message; + const args = message.trim().split(/ +/g); + const cmd = args.shift().toLowerCase(); + system.run(async () => { + try { + const { default: CommandObject } = await import(`./commands/${cmd}.js`); + CommandObject.execute(player, args); + } + catch (err) { + if (err instanceof ReferenceError) { + SendMessageTo(player, { + rawtext: [ + { + translate: "yn:fishing_got_reel.on_caught_main_command_not_found", + with: [ + cmd, + "\n", + ADDON_IDENTIFIER + ] + } + ] + }); + } + else { + Logger.error(err, err.stack); + } + } }); }); diff --git a/BP/scripts/utils/Database/con-database.js b/BP/scripts/utils/Database/con-database.js new file mode 100644 index 0000000..910ba84 --- /dev/null +++ b/BP/scripts/utils/Database/con-database.js @@ -0,0 +1,813 @@ +import { world, World, Entity, system } from "@minecraft/server"; +import * as mc from "@minecraft/server"; +const mc_world = world; +const { setDynamicProperty: wSDP, getDynamicProperty: wGDP, getDynamicPropertyIds: wGDPI } = World.prototype; +let { isValid: isValidEntity, setDynamicProperty: eSDP, getDynamicProperty: eGDP, getDynamicPropertyIds: eGDPI } = Entity.prototype; +const DYNAMIC_DB_PREFIX = "\u1221\u2112"; +const ROOT_CONTENT_TABLE_UUID = "c0211201-0001-4001-8001-4f90af596647"; +const STRING_LIMIT = 32e3; +const TABLE_STRING_LENGTH = 31e3; +const GENERATOR_DESERIALIZER_SYMBOL = Symbol("DESERIALIZER"); +const eP = { + gDP: eGDP, + sDP: eSDP, + gDPI: eGDPI +}; +const wP = { + gDP: wGDP, + sDP: wSDP, + gDPI: wGDPI +}; +class DynamicSource { + constructor(source) { + this.source = source; + if (SOURCE_INSTANCES.has(source)) + return SOURCE_INSTANCES.get(source); + if (source === mc_world) + Object.assign(this, wP); + else if (isValidEntity.call(source)) + Object.assign(this, eP); + else + throw new ReferenceError("Invald source type: " + source); + SOURCE_INSTANCES.set(source, this); + } + getIds() { return this.gDPI.call(this.source); } + get(key) { return this.gDP.call(this.source, key); } + set(key, value) { this.sDP.call(this.source, key, value); } + delete(key) { this.sDP.call(this.source, key, undefined); return true; } + isValid() { return this.source === world || isValidEntity.call(this.source); } +} +const SOURCE_INSTANCES = new WeakMap(); +const DDB_SUBINSTANCES = new WeakMap(); +class DynamicDatabase extends Map { + constructor(source, id, kind, parser) { + super(); + this._source = new DynamicSource(source); + const PRE = `${kind}${DYNAMIC_DB_PREFIX}${id}${DYNAMIC_DB_PREFIX}`, LENGTH = PRE.length, SOURCE = this._source, PARSE = parser.parse; + const MAP_INSTANCES = DDB_SUBINSTANCES.get(SOURCE) ?? new Map; + if (MAP_INSTANCES.has(PRE)) + return MAP_INSTANCES.get(PRE); + MAP_INSTANCES.set(PRE, this); + DDB_SUBINSTANCES.set(SOURCE, MAP_INSTANCES); + if (!SOURCE.isValid()) + throw new ReferenceError("Source is no longer valid: " + SOURCE.source); + this._prefix = PRE; + this._prefixLength = LENGTH; + this._STRINGIFY = parser.stringify; + this._notDisposed = true; + for (const K of SOURCE.getIds()) + if (K.startsWith(PRE)) { + const key = K.substring(LENGTH); + const value = SOURCE.get(K); + if (typeof value === "string") + super.set(key, PARSE(value)); + } + } + set(key, value) { + if (!this.isValid()) + throw new ReferenceError("This database instance is no longer valid"); + if (key.length + this._prefixLength > STRING_LIMIT) + throw new TypeError("Key is too long: " + key.length); + if (value === undefined) { + this.delete(key); + return this; + } + const data = this._STRINGIFY(value); + if (data.length > STRING_LIMIT) + throw new TypeError("Size of data in string is too long: " + data.length); + this._source.set(this._prefix + key, data); + return super.set(key, value); + } + delete(key) { + if (!this.isValid()) + throw new ReferenceError("This database instance is no longer valid"); + if (!this.has(key)) + return false; + this._source.delete(this._prefix + key); + return super.delete(key); + } + clear() { + if (!this.isValid()) + throw new ReferenceError("This database instance is no longer valid"); + const P = this._prefix; + const s = this._source; + for (const key of this.keys()) + s.delete(P + key); + return super.clear(); + } + isValid() { return this._source.isValid() && this._notDisposed; } + dispose() { + this._notDisposed = false; + DDB_SUBINSTANCES.get(this._source)?.delete?.(this._prefix); + super.clear(); + } + get isDisposed() { return !this._notDisposed; } +} +class DynamicWrapper { + constructor(source, id, kind, parser) { + this._source = source; + const PRE = `${kind}${DYNAMIC_DB_PREFIX}${id}${DYNAMIC_DB_PREFIX}`, LENGTH = PRE.length; + this._prefix = PRE; + this._prefixLength = LENGTH; + this._STRINGIFY = parser.stringify; + this._PARSE = parser.parse; + } + clear() { for (const k of this.__getKeys()) + this._source.set(k, undefined); } + ; + delete(key) { + const has = this.has(key); + this._source.set(this._prefix + key, undefined); + return has; + } + ; + forEach(callbackfn, thisArg = null) { + for (const k of this.keys()) { + try { + callbackfn.call(thisArg ?? null, k, this.get(k), this); + } + catch (error) { + } + } + } + get(key) { const a = this._source.get(this._prefix + key); typeof a === "string" ? this._PARSE(a) : a; } + ; + has(key) { return this._source.get(this._prefix + key) !== undefined; } + set(key, value) { + this._source.set(this._prefix + key, this._STRINGIFY(value)); + return this; + } + ; + get size() { return [...this.__getKeys()].length; } + ; + [Symbol.iterator]() { return this.entries(); } + *entries() { for (const k of this.__getKeys()) + yield [k.substring(this._prefixLength), this._PARSE(this._source.get(k))]; } + ; + *keys() { for (const k of this.__getKeys()) + yield k.substring(this._prefixLength); } + ; + *values() { for (const k of this.__getKeys()) + yield this._PARSE(this._source.get(k)); } + *__getKeys() { for (const K of this._source.getIds()) + if (K.startsWith(this._prefix)) + yield K; } +} +class JsonDatabase extends DynamicDatabase { + constructor(id, source = world) { super(source, id, "JSON", JSON); } +} +class JSONDynamicWrapper extends DynamicWrapper { + constructor(id, source = world) { super(source, id, "JSON", JSON); } +} +class DynamicProxy extends JsonDatabase { + constructor(id, source = world) { + super(id, source); + return new Proxy(this, { + defineProperty(t, p, att) { + if (att.value && typeof p === "string") { + t.set(p, att.value); + return true; + } + return false; + }, + deleteProperty(t, p) { + if (typeof p === "string") + return t.delete(p); + return false; + }, + set(t, p, newValue) { + if (typeof p === "string") { + t.set(p, newValue); + return true; + } + return false; + }, + get(t, p) { + if (typeof p === "string") { + return t.get(p) ?? Object.prototype[p]; + } + return false; + }, + getPrototypeOf(t) { return Object.prototype; }, + isExtensible(t) { return true; }, + setPrototypeOf(t) { return false; }, + has(t, k) { return t.has(k); }, + preventExtensions(t) { return false; }, + ownKeys(t) { return [...t.keys()]; }, + getOwnPropertyDescriptor(t, k) { + if (t.has(k)) { + return { value: t.get(k), enumerable: true, configurable: true, writable: true }; + } + } + }); + } +} +const PARSER_SYMBOL = Symbol("SERIALIZEABLE"); +const SERIALIZERS = new Map(); +const DESERIALIZER_INFO = new WeakMap(); +const ROOT_KEY = "root::" + ROOT_CONTENT_TABLE_UUID; +const TABLE_SOURCES = new WeakMap(); +const TABLE_ID = new WeakMap(); +const ID_TABLE = new WeakMap(); +const TABLE_VALIDS = new WeakSet(); +let isNativeCall = false; +let RootTable; +function getRootTable() { + if (RootTable) + return RootTable; + return RootTable = world.getDynamicProperty(ROOT_KEY) ? DATABASE_MANAGER.deserialize(ROOT_KEY, new DynamicSource(world)) : (() => { + const source = new DynamicSource(world); + isNativeCall = true; + const value = new DynamicTable(); + isNativeCall = false; + TABLE_SOURCES.set(value, source); + TABLE_ID.set(value, ROOT_KEY); + SetTable(source, ROOT_KEY, value); + TABLE_VALIDS.add(value); + DATABASE_MANAGER.serialize(ROOT_KEY, source, value); + return value; + })(); +} +const SerializableKinds = { + Boolean: "c0211201-0001-4002-8001-4f90af596647", + Number: "c0211201-0001-4002-8002-4f90af596647", + String: "c0211201-0001-4002-8003-4f90af596647", + Object: "c0211201-0001-4002-8004-4f90af596647", + DynamicTable: "c0211201-0001-4002-8101-4f90af596647" +}; +SerializableKinds[SerializableKinds.Boolean] = "Boolean"; +SerializableKinds[SerializableKinds.Number] = "Number"; +SerializableKinds[SerializableKinds.String] = "String"; +SerializableKinds[SerializableKinds.Object] = "Object"; +SerializableKinds[SerializableKinds.DynamicTable] = "DynamicTable"; +const Serializer = { + isSerializable(object) { return object[PARSER_SYMBOL] != undefined; }, + getSerializerKind(object) { return object[PARSER_SYMBOL]; }, + isRegistredKind(kind) { return SERIALIZERS.has(kind); }, + setSerializableKind(object, kind) { + if (SERIALIZERS.has(kind)) { + object[PARSER_SYMBOL] = kind; + return true; + } + return false; + }, + registrySerializer(kind, serializer, deserializer) { + if (SERIALIZERS.has(kind)) + throw new ReferenceError("Duplicate serialization kind: " + kind); + if (typeof kind != "string") + throw new TypeError("Kind must be type of string."); + if (typeof serializer != "function" || typeof deserializer != "function") + throw new TypeError("serializer or deserializer is not a function"); + SERIALIZERS.set(kind, { serializer, deserializer }); + return kind; + }, + getSerializer(kind) { + return SERIALIZERS.get(kind)?.serializer ?? null; + }, + getDeserializer(kind) { + return SERIALIZERS.get(kind)?.deserializer ?? null; + }, + getSerializers(kind) { + const data = SERIALIZERS.get(kind); + if (!data) + return null; + return { ...data }; + }, + setSerializableClass(construct, kind, serializer, deserializer) { + if (typeof serializer !== "function" || typeof deserializer !== "function") + throw new TypeError("Serializer or deserializer is not a function"); + Serializer.registrySerializer(kind, function (obj) { + if (obj == null) + throw new TypeError("Null or Undefined is not possible to serialize."); + return serializer(obj); + }, function (obj) { + if (obj[GENERATOR_DESERIALIZER_SYMBOL] !== true) + throw new TypeError("Null or Undefined is not possible to serialize."); + return deserializer(obj); + }); + Serializer.setSerializableKind(construct.prototype, kind); + }, + getKindFromClass(construct) { + return construct?.prototype?.[PARSER_SYMBOL] ?? null; + }, + getSerializerKinds() { return SERIALIZERS.keys(); }, + overrideSerializers(kind, serializer, deserializer) { + if (typeof kind != "string") + throw new TypeError("Kind must be type of string."); + if (typeof serializer != "function" || typeof deserializer != "function") + throw new TypeError("serializer or deserializer is not a function"); + SERIALIZERS.set(kind, { serializer, deserializer }); + return kind; + } +}; +const DATABASE_MANAGER = { + getHeader(rootRef, source) { + const data = source.get(rootRef); + if (typeof data != "string") + return null; + return JSONReadable(data); + }, + serialize(rootRef, source, object) { + if (!Serializer.isRegistredKind(Serializer.getSerializerKind(object))) + throw new TypeError("object is not serializeable."); + const kind = Serializer.getSerializerKind(object); + const serializer = Serializer.getSerializer(kind); + if (!serializer) + throw new ReferenceError("No serializer for " + kind); + return this.serializationResolver(serializer(object, { kind, source, rootRef }), rootRef, source, kind); + }, + serializationResolver(gen, rootRef, source, kind) { + const oldHeader = this.getHeader(rootRef, source); + const prefix = rootRef + "::"; + let oldLength = 0, newLength = 0; + if (oldHeader) { + const [data] = oldHeader; + oldLength = parseInt(data["length"], 36); + } + try { + let genNext = gen.next(); + if (!genNext.done) { + const headerData = genNext.value + ""; + if (headerData.length > TABLE_STRING_LENGTH) + gen.throw(new RangeError("Yielded stirng is too big: " + headerData.length)); + genNext = gen.next(); + while (!genNext.done) { + const key = prefix + newLength; + try { + source.set(key, genNext.value + ""); + newLength++; + } + catch (error) { + gen.throw(error); + } + genNext = gen.next(); + } + source.set(rootRef, JSONWritable({ length: newLength.toString(36), kind }, headerData)); + } + return newLength; + } + catch (er) { + Object.setPrototypeOf(er, DataCoruptionError.prototype); + er.source = source; + er.rootKey = rootRef; + throw er; + } + finally { + for (let i = newLength; i < oldLength; i++) + source.delete(prefix + i); + } + }, + deserialize(rootRef, source, header = undefined) { + try { + const oldHeader = header ?? this.getHeader(rootRef, source); + if (!oldHeader) + return null; + const prefix = rootRef + "::"; + const [{ length: le, kind }, data] = oldHeader; + let length = parseInt(le, 36); + if (!Serializer.isRegistredKind(kind)) + throw new ReferenceError("Unknown parser kind: " + kind); + const deserializeResolver = Serializer.getDeserializer(kind); + if (!deserializeResolver) + throw new ReferenceError("No deserializer for: " + kind); + const deserializer = this.deserializer(source, rootRef, prefix, length, data); + DESERIALIZER_INFO.set(deserializer, { + source, + rootRef, + kind, + deserializeResolver, + oldHeader, + length, + }); + return deserializeResolver(deserializer); + } + catch (error) { + error.rootKey = rootRef; + error.source = source; + throw Object.setPrototypeOf(error, DataCoruptionError); + } + }, + *deserializer(source, root, prefix, length, initial) { + yield initial; + let i = 0; + while (i < length) { + const data = source.get(prefix + i); + if (!data) + throw new DataCoruptionError(source, root, "No continual data at index of " + i); + yield data; + i++; + } + }, + removeTree(rootRef, source) { + const oldHeader = this.getHeader(rootRef, source); + if (!oldHeader) + return false; + const prefix = rootRef + "::"; + const [{ length: le }] = oldHeader; + let length = parseInt(le, 36); + if (!isFinite(length)) + return false; + for (let i = 0; i < length; i++) + source.delete(prefix + i); + source.delete(rootRef); + return true; + } +}; +Object.defineProperties(DATABASE_MANAGER.deserializer.prototype, Object.getOwnPropertyDescriptors({ + [GENERATOR_DESERIALIZER_SYMBOL]: true, + return() { + return { done: true }; + }, + continue() { + return this.next(...arguments).value; + }, + get source() { + if (!DESERIALIZER_INFO.has(this)) + throw new ReferenceError("Object bound to prototype does not exist."); + return DESERIALIZER_INFO.get(this).source; + }, + get rootKey() { + if (!DESERIALIZER_INFO.has(this)) + throw new ReferenceError("Object bound to prototype does not exist."); + return DESERIALIZER_INFO.get(this).rootRef; + }, + get length() { + if (!DESERIALIZER_INFO.has(this)) + throw new ReferenceError("Object bound to prototype does not exist."); + return DESERIALIZER_INFO.get(this).length; + }, + get kind() { + if (!DESERIALIZER_INFO.has(this)) + throw new ReferenceError("Object bound to prototype does not exist."); + return DESERIALIZER_INFO.get(this).kind; + } +})); +class DynamicTable extends Map { + static get KIND() { return "c0211201-0001-4002-8101-4f90af596647"; } + get tableId() { return TABLE_ID.get(this); } + constructor() { + if (!isNativeCall) + throw new ReferenceError("No constructor for " + DynamicTable.name); + super(); + } + get(key) { + if (!this.isValid()) + throw new ReferenceError("Object bound to prototype doesn't not exist at [DynamicTable::get()]."); + if (!this.has(key)) + return; + const source = TABLE_SOURCES.get(this); + const dataId = super.get(key); + return DATABASE_MANAGER.deserialize(dataId, source); + } + set(key, value) { + if (!this.isValid()) + throw new ReferenceError("Object bound to prototype doesn't not exist at [DynamicTable::get()]."); + if (value == null) + throw new ReferenceError("You can not assign property to null or undefined"); + if (!Serializer.isRegistredKind(Serializer.getSerializerKind(value))) + throw new TypeError("value is not serializeable."); + if (value instanceof DynamicTable) + throw new TypeError("You can't set value as DynamicTable please use AddTable"); + const has = this.has(key); + const source = TABLE_SOURCES.get(this); + let newKey; + if (has) { + newKey = super.get(key); + const header = DATABASE_MANAGER.getHeader(newKey, source); + if (header?.[0]?.kind === DynamicTable.KIND) { + const a = DATABASE_MANAGER.deserialize(newKey, source, header); + a.clear(); + TABLE_VALIDS.delete(a); + } + } + else { + newKey = "k:" + v4uuid(); + super.set(key, newKey); + SaveState(this); + } + DATABASE_MANAGER.serialize(newKey, source, value); + return this; + } + clear() { + if (!this.isValid()) + throw new ReferenceError("Object bound to prototype doesn't not exist at [DynamicTable::clear()]."); + const source = TABLE_SOURCES.get(this); + const KIND = DynamicTable.KIND; + for (const k of super.keys()) { + const dataId = super.get(k); + const header = DATABASE_MANAGER.getHeader(dataId, source); + if (header?.[0]?.kind === KIND) { + const a = DATABASE_MANAGER.deserialize(dataId, source, header); + a.clear(); + TABLE_VALIDS.delete(a); + } + DATABASE_MANAGER.removeTree(dataId, source); + } + SaveState(this); + super.clear(); + } + delete(key) { + if (!this.isValid()) + throw new ReferenceError("Object bound to prototype doesn't not exist at [DynamicTable::delete()]."); + const source = TABLE_SOURCES.get(this); + if (!this.has(key)) + return false; + const dataId = super.get(key); + const header = DATABASE_MANAGER.getHeader(dataId, source); + if (header?.[0]?.kind === DynamicTable.KIND) { + const a = DATABASE_MANAGER.deserialize(dataId, source, header); + a.clear(); + TABLE_VALIDS.delete(a); + } + DATABASE_MANAGER.removeTree(dataId, source); + SaveState(this); + return super.delete(); + } + *entries() { + if (!this.isValid()) + throw new ReferenceError("Object bound to prototype doesn't not exist at [DynamicTable::entries()]."); + for (const [k, v] of super.entries()) + yield [k, this.get(k)]; + } + [Symbol.iterator]() { return this.entries(); } + *values() { + if (!this.isValid()) + throw new ReferenceError("Object bound to prototype doesn't not exist at [DynamicTable::values()]."); + for (const k of super.keys()) + yield this.get(k); + } + isValid() { + return !!(TABLE_VALIDS.has(this) && TABLE_SOURCES.get(this)?.isValid?.()); + } + static OpenCreate(id) { + let fromTable = getRootTable(); + let a = fromTable.get(id); + if (a === undefined) { + if (!fromTable.isValid()) + throw new ReferenceError("Object bound to prototype doesn't not exist at [DynamicTable::get()]."); + if (Map.prototype.has.call(fromTable, id)) + throw new ReferenceError("Value of this key already exists"); + const source = TABLE_SOURCES.get(fromTable); + let newKey = "t" + v4uuid(); + isNativeCall = true; + const value = new DynamicTable(); + isNativeCall = false; + Map.prototype.set.call(fromTable, id, newKey); + SaveState(fromTable); + DATABASE_MANAGER.serialize(newKey, source, value); + TABLE_SOURCES.set(value, source); + TABLE_ID.set(value, newKey); + SetTable(source, newKey, value); + TABLE_VALIDS.add(value); + a = value; + } + else if (!(a instanceof DynamicTable)) + throw new TypeError(`Value saved in ${id} is not a dynamic table.`); + return a; + } + static ClearAll() { getRootTable().clear(); } + static getTableIds() { return getRootTable().keys(); } + static DeleteTable(key) { return getRootTable().delete(key); } +} +function SaveState(table) { + if (table._task === undefined) { + table._task = system.run(() => { + table._task = undefined; + if (table.isValid()) { + DATABASE_MANAGER.serialize(table.tableId, TABLE_SOURCES.get(table), table); + } + }); + } +} +function GetTable(source, rootRef) { return ID_TABLE.get(source)?.get(rootRef); } +function SetTable(source, rootRef, table) { + if (!ID_TABLE.has(source)) + ID_TABLE.set(source, new Map()); + ID_TABLE.get(source).set(rootRef, table); +} +class DataCoruptionError extends ReferenceError { + constructor(source, rootKey, message) { + super(message); + this.rootKey = rootKey; + this.source = source; + } + remove() { + if (!this.source.isValid()) + throw new ReferenceError("Source is no longer valid"); + DATABASE_MANAGER.removeTree(this.rootKey, this.source); + } +} +Serializer.setSerializableClass(DynamicTable, DynamicTable.KIND, function* (table) { + let obj = {}, i = 0; + const get = Map.prototype.get, maxSize = 300; + yield Math.ceil(table.size / maxSize); + for (const key of table.keys()) { + if (++i >= maxSize) { + yield JSON.stringify(obj); + i = 0, obj = {}; + } + obj[key] = get.call(table, key); + } + if (i) + yield JSON.stringify(obj); +}, function (n) { + if (GetTable(n.source, n.rootKey)) + return GetTable(n.source, n.rootKey); + isNativeCall = true; + const table = new DynamicTable(); + isNativeCall = false; + TABLE_SOURCES.set(table, n.source); + TABLE_ID.set(table, n.rootKey); + SetTable(n.source, n.rootKey, table); + TABLE_VALIDS.add(table); + const set = Map.prototype.set; + const length = Number(n.continue()); + for (let i = 0; i < length; i++) { + const data = n.continue(); + if (!data) + throw new DataCoruptionError(n.source, n.rootKey, "Data for this dynamic table are corupted."); + const obj = JSON.parse(data); + for (const k of Object.getOwnPropertyNames(obj)) + set.call(table, k, obj[k]); + } + return table; +}); +Serializer.setSerializableClass(Boolean, SerializableKinds.Boolean, function* (n) { yield n; }, function (n) { for (const a of n) + return a === "true"; }); +Serializer.setSerializableClass(Number, SerializableKinds.Number, function* (n) { yield n; }, function (n) { for (const a of n) + return Number(a); }); +Serializer.setSerializableClass(String, SerializableKinds.String, function* (n) { + let length = n.length; + let cursor = 0; + let i = 0; + yield Math.ceil(length / TABLE_STRING_LENGTH); + while (length > 0) { + const s = n.substring(cursor, cursor + TABLE_STRING_LENGTH); + const l = s.length; + if (l <= 0) + return; + length -= l, cursor += l; + yield s; + i++; + } +}, function (n) { + const count = Number(n.continue()); + const l = new Array(count); + for (let i = 0; i < count; i++) { + l[i] = n.continue(); + } + return l.join(""); +}); +Serializer.setSerializableClass(Object, SerializableKinds.Object, function (n) { return Serializer.getSerializer(SerializableKinds.String)(JSON.stringify(n)); }, function (n) { return JSON.parse(Serializer.getDeserializer(SerializableKinds.String)(n)); }); +function v4uuid(timestamp = Date.now()) { + const { random, floor } = Math; + const uuid = 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, (c) => { + let r = (timestamp + random() * 16) % 16 | 0; + timestamp = floor(timestamp / 16); + return (c == 'x' ? r : (r & 0x3 | 0x8)).toString(16); + }); + return uuid; +} +function Readable(text) { + const size = text.charCodeAt(0); + const info = text.substring(1, 1 + size); + const data = text.substring(1 + size); + return [info, data, size]; +} +function JSONReadable(text) { + const [info, data, size] = Readable(text); + return [JSON.parse(info), data, size]; +} +function Writable(json, text) { + return `${String.fromCharCode(json.length)}${json}${text}`; +} +function JSONWritable(json, text) { return Writable(JSON.stringify(json), text); } +; +export { JsonDatabase, DynamicProxy, DynamicTable, Serializer, DataCoruptionError, SerializableKinds }; +export const APISerializableKinds = { + BlockType: "c0211201-0001-4002-8201-4f90af596647", + EntityType: "c0211201-0001-4002-8202-4f90af596647", + ItemType: "c0211201-0001-4002-8203-4f90af596647", + BlockPermutation: "c0211201-0001-4002-8204-4f90af596647", + ItemStack: "c0211201-0001-4002-8205-4f90af596647", + Vector: "c0211201-0001-4002-8206-4f90af596647", + "c0211201-0001-4002-8201-4f90af596647": "BlockType", + "c0211201-0001-4002-8202-4f90af596647": "EntityType", + "c0211201-0001-4002-8203-4f90af596647": "ItemType", + "c0211201-0001-4002-8204-4f90af596647": "BlockPermutation", + "c0211201-0001-4002-8205-4f90af596647": "ItemStack", + "c0211201-0001-4002-8206-4f90af596647": "Vector", +}; +export const registryAPISerializers = () => { + const { BlockType, BlockTypes, BlockPermutation, EntityType, EntityTypes, ItemType, ItemTypes, ItemStack, Vector } = mc; + const ItemStackSupportLevel = { + dynamicProperties: ItemStack.prototype.getDynamicProperty, + canPlaceOn: ItemStack.prototype.getCanPlaceOn, + canDestory: ItemStack.prototype.getCanDestroy, + lore: ItemStack.prototype.getLore, + lockMode: mc.ItemLockMode, + keepOnDeath: "keepOnDeath" in ItemStack.prototype, + components: ItemStack.prototype.getComponents, + enchantable: mc.ItemEnchantableComponent, + durability: mc.ItemDurabilityComponent, + }; + for (const key in ItemStackSupportLevel) { + if (Object.hasOwnProperty.call(ItemStackSupportLevel, key)) { + const element = ItemStackSupportLevel[key]; + } + } + const ItemStackComponentManager = { + serializers: {}, + deserializers: {} + }; + const { serializers: ItemComponentSerializers, deserializers: ItemComponentDeserializers } = ItemStackComponentManager; + if (ItemStackSupportLevel.durability) { + ItemComponentSerializers[ItemStackSupportLevel.durability.componentId] = function (component) { return component.damage; }; + ItemComponentDeserializers[ItemStackSupportLevel.durability.componentId] = function (component, v) { component.damage = v; }; + } + if (ItemStackSupportLevel.enchantable) { + ItemComponentSerializers[ItemStackSupportLevel.enchantable.componentId] = function (component) { return component.getEnchantments().map(e => ({ t: e.type.id, l: e.level })); }; + ItemComponentDeserializers[ItemStackSupportLevel.enchantable.componentId] = function (component, v) { component.addEnchantments(v.map(e => ({ type: e.t, level: e.l }))); }; + } + if (BlockTypes) + Serializer.setSerializableClass(BlockType, APISerializableKinds.BlockType, function* (n) { yield n.id; }, function (n) { for (const a of n) + return BlockTypes.get(a); }); + if (EntityTypes) + Serializer.setSerializableClass(EntityType, APISerializableKinds.EntityType, function* (n) { yield n.id; }, function (n) { for (const a of n) + return EntityTypes.get(a); }); + if (ItemTypes) + Serializer.setSerializableClass(ItemType, APISerializableKinds.ItemType, function* (n) { yield n.id; }, function (n) { for (const a of n) + return ItemTypes.get(a); }); + if ("type" in BlockPermutation.prototype) + Serializer.setSerializableClass(BlockPermutation, APISerializableKinds.BlockPermutation, function* (n) { + yield n.type.id; + yield JSON.stringify(n.getAllStates()); + }, function (n) { + const [typeId, states] = n; + return BlockPermutation.resolve(typeId, JSON.parse(states)); + }); + Serializer.setSerializableClass(ItemStack, APISerializableKinds.ItemStack, function* (n) { + const components = ItemStackSupportLevel.components ? [...n.getComponents()].filter(e => e && (e.typeId in ItemComponentSerializers)) : []; + const canPlaceOn = ItemStackSupportLevel.canPlaceOn ? n.getCanPlaceOn() : []; + const canDestroy = ItemStackSupportLevel.canDestory ? n.getCanDestroy() : []; + const dynamicProperties = ItemStackSupportLevel.dynamicProperties ? n.getDynamicPropertyIds() : []; + yield JSON.stringify([ + n.typeId, + n.amount, + ItemStackSupportLevel.keepOnDeath ? n.keepOnDeath : false, + ItemStackSupportLevel.lockMode ? n.lockMode : "", + typeof n.nameTag === "string", + components.length, + canPlaceOn.length, + canDestroy.length, + dynamicProperties.length + ]); + n.nameTag ? yield n.nameTag : null; + yield JSON.stringify(ItemStackSupportLevel.lore ? n.getLore() : []); + for (const com of components) + yield JSON.stringify([com.typeId, ItemComponentSerializers[com.typeId](com)]); + yield* canPlaceOn; + yield* canDestroy; + for (const k of dynamicProperties) { + const data = JSON.stringify(n.getDynamicProperty(k)); + if ((data.length + k.length) > TABLE_STRING_LENGTH) + throw new TypeError(`Dynamic property '${k}' of this item is too large'${data.length}'`); + yield Writable(k, data); + } + }, function (n) { + const [typeId, amount, keepOnDeath, lockMode, hasNameTag, componentsCount, canPlaceOnCount, canDestroyCount, dynamicPropertiesCount] = JSON.parse(n.continue()); + const item = new ItemStack(typeId, amount); + if (ItemStackSupportLevel.keepOnDeath) + item.keepOnDeath = keepOnDeath; + if (ItemStackSupportLevel.lockMode) + item.lockMode = lockMode; + if (hasNameTag) + item.nameTag = n.continue(); + const lore = JSON.parse(n.continue()); + if (ItemStackSupportLevel.lore) + item.setLore(lore); + let i = componentsCount; + while (i--) { + const [id, data] = JSON.parse(n.continue()); + ItemComponentDeserializers[id](item.getComponent(id), data); + } + i = canPlaceOnCount; + const canPlaceOn = []; + while (i--) + canPlaceOn.push(n.continue()); + i = canDestroyCount; + const canDestroy = []; + while (i--) + canDestroy.push(n.continue()); + item.setCanPlaceOn(canPlaceOn); + item.setCanDestroy(canDestroy); + i = dynamicPropertiesCount; + while (i--) { + const [k, sata] = Readable(n.continue()); + item.setDynamicProperty(k, JSON.parse(sata)); + } + return item; + }); + if (Vector) + Serializer.setSerializableClass(Vector, APISerializableKinds.Vector, function (s) { const { x, y, z } = s; return Object.Serialize({ x, y, z }); }, function (n) { const { x, y, z } = Object.Deserialize(n); return new Vector(x, y, z); }); +}; diff --git a/BP/scripts/utils/VectorUtils.js b/BP/scripts/utils/VectorUtils.js new file mode 100644 index 0000000..21ca01f --- /dev/null +++ b/BP/scripts/utils/VectorUtils.js @@ -0,0 +1,508 @@ +import { Block } from "@minecraft/server"; +export class Vec3 { + constructor(xOrOther, y, z) { + if (typeof xOrOther === "number" && y !== undefined && z !== undefined) { + this.x = xOrOther; + this.y = y; + this.z = z; + } + else if (xOrOther instanceof Vec3 || xOrOther instanceof Block || typeof xOrOther === 'object') { + this.x = xOrOther.x; + this.y = xOrOther.y; + this.z = xOrOther.z; + } + else { + throw new Error("Invalid constructor arguments"); + } + } + static fromArray(array) { + return new Vec3(array[0], array[1], array[2]); + } + static fromString(str) { + const parts = str.trim().split(',').map(part => part.trim()); + if (parts.length !== 3) { + throw new Error('Invalid vector format. Expected "x, y, z".'); + } + const [x, y, z] = parts.map(part => parseFloat(part)); + if (isNaN(x) || isNaN(y) || isNaN(z)) { + throw new Error('Invalid vector components. Ensure all components are numbers.'); + } + return new Vec3(x, y, z); + } + length() { + return Math.hypot(this.x, this.y, this.z); + } + lengthSquared() { + return Math.pow(this.x, 2) + Math.pow(this.y, 2) + Math.pow(this.z, 2); + } + normalize() { + const length = this.length(); + return length === 0 ? this : this.scale(1 / length); + } + add(other) { + return new Vec3(this.x + other.x, this.y + other.y, this.z + other.z); + } + static add(a, b) { + return new Vec3(b.x + a.x, b.y + a.y, b.z + a.z); + } + sub(other) { + return new Vec3(this.x - other.x, this.y - other.y, this.z - other.z); + } + static sub(a, b) { + return new Vec3(a.x - b.x, a.y - b.y, a.z - b.z); + } + mul(other) { + return new Vec3(this.x * other.x, this.y * other.y, this.z * other.z); + } + static mul(a, b) { + return new Vec3(a.x * b.x, a.y * b.y, a.z * b.z); + } + div(other) { + return new Vec3(this.x / other.x, this.y / other.y, this.z / other.z); + } + static div(a, b) { + return new Vec3(a.x / b.x, a.y / b.y, a.z / b.z); + } + scale(scalar) { + return new Vec3(this.x * scalar, this.y * scalar, this.z * scalar); + } + dot(other) { + return this.x * other.x + this.y * other.y + this.z * other.z; + } + cross(other) { + const x = this.y * other.z - this.z * other.y; + const y = this.z * other.x - this.x * other.z; + const z = this.x * other.y - this.y * other.x; + return new Vec3(x, y, z); + } + equals(other) { + return this.x === other.x && this.y === other.y && this.z === other.z; + } + static equals(a, b) { + return b.x === a.x && b.y === a.y && b.z === a.z; + } + equalsEpsilon(other, tolerance = Vec3.EPSILON) { + return (Math.abs(this.x - other.x) <= tolerance && + Math.abs(this.y - other.y) <= tolerance && + Math.abs(this.z - other.z) <= tolerance); + } + equalsApprox(other) { + return this.equalsEpsilon(other, Vec3.EPSILON); + } + tripleScalar(b, c) { + return this.dot(b.cross(c)); + } + barycentricCoordinates(v1, v2, v3) { + const v2MinusV1 = v2.sub(v1); + const v3MinusV1 = v3.sub(v1); + const pointMinusV1 = this.sub(v1); + const dot00 = v2MinusV1.dot(v2MinusV1); + const dot01 = v2MinusV1.dot(v3MinusV1); + const dot02 = v2MinusV1.dot(pointMinusV1); + const dot11 = v3MinusV1.dot(v3MinusV1); + const dot12 = v3MinusV1.dot(pointMinusV1); + const invDenom = 1 / (dot00 * dot11 - dot01 * dot01); + const u = (dot11 * dot02 - dot01 * dot12) * invDenom; + const v = (dot00 * dot12 - dot01 * dot02) * invDenom; + return new Vec3(1 - u - v, u, v); + } + rotate(angle, axis) { + const cosAngle = Math.cos(angle); + const sinAngle = Math.sin(angle); + const oneMinusCos = 1 - cosAngle; + const rotationMatrix = [ + [ + oneMinusCos * axis.x * axis.x + cosAngle, + oneMinusCos * axis.x * axis.y - sinAngle * axis.z, + oneMinusCos * axis.x * axis.z + sinAngle * axis.y + ], + [ + oneMinusCos * axis.y * axis.x + sinAngle * axis.z, + oneMinusCos * axis.y * axis.y + cosAngle, + oneMinusCos * axis.y * axis.z - sinAngle * axis.x + ], + [ + oneMinusCos * axis.z * axis.x - sinAngle * axis.y, + oneMinusCos * axis.z * axis.y + sinAngle * axis.x, + oneMinusCos * axis.z * axis.z + cosAngle + ] + ]; + return this.matrixProduct(rotationMatrix); + } + matrixProduct(matrix) { + if (matrix.length !== 3 || + matrix[0].length !== 3 || + matrix[1].length !== 3 || + matrix[2].length !== 3) { + throw new Error('Invalid matrix dimensions'); + } + const x = this.x * matrix[0][0] + this.y * matrix[0][1] + this.z * matrix[0][2]; + const y = this.x * matrix[1][0] + this.y * matrix[1][1] + this.z * matrix[1][2]; + const z = this.x * matrix[2][0] + this.y * matrix[2][1] + this.z * matrix[2][2]; + return new Vec3(x, y, z); + } + abs() { + return new Vec3(Math.abs(this.x), Math.abs(this.y), Math.abs(this.z)); + } + distance(other) { + return this.sub(other).length(); + } + static distance(a, b) { + const dx = b.x - a.x; + const dy = b.y - a.y; + const dz = b.z - a.z; + const distance = Math.hypot(dx, dy, dz); + return distance; + } + distanceSquared(other) { + return this.sub(other).lengthSquared(); + } + angle(other) { + return Math.acos(this.dot(other) / (this.length() * other.length())); + } + projectOnto(other) { + const lengthSquared = other.lengthSquared(); + if (lengthSquared === 0) { + throw new Error('Cannot project onto a zero vector'); + } + return other.scale(this.dot(other) / lengthSquared); + } + rejectFrom(other) { + return this.sub(this.projectOnto(other)); + } + reflect(other) { + return this.sub(this.projectOnto(other).scale(2)); + } + refract(normal, eta) { + const dot = this.dot(normal); + const k = 1 - eta * eta * (1 - dot * dot); + return k < 0 ? new Vec3(0, 0, 0) : this.scale(eta).sub(normal.scale(eta * dot + Math.sqrt(k))); + } + lerp(other, t) { + return this.add(other.sub(this).scale(t)); + } + static lerp(a, b, t) { + const dest = { x: a.x, y: a.y, z: a.z }; + dest.x += (b.x - a.x) * t; + dest.y += (b.y - a.y) * t; + dest.z += (b.z - a.z) * t; + return new Vec3(dest); + } + slerp(other, t) { + const dot = this.dot(other); + const theta = Math.acos(dot); + const sinTheta = Math.sin(theta); + const scale1 = Math.sin((1 - t) * theta) / sinTheta; + const scale2 = Math.sin(t * theta) / sinTheta; + return this.scale(scale1).add(other.scale(scale2)); + } + hermite(other, t, tangent1, tangent2) { + const t2 = t * t; + const t3 = t2 * t; + const h1 = 2 * t3 - 3 * t2 + 1; + const h2 = -2 * t3 + 3 * t2; + const h3 = t3 - 2 * t2 + t; + const h4 = t3 - t2; + return this.scale(h1).add(other.scale(h2)).add(tangent1.scale(h3)).add(tangent2.scale(h4)); + } + static quadracticBezier(start, control, end, t) { + return { + x: (1 - t) * (1 - t) * start.x + 2 * (1 - t) * t * control.x + t * t * end.x, + y: (1 - t) * (1 - t) * start.y + 2 * (1 - t) * t * control.y + t * t * end.y, + z: (1 - t) * (1 - t) * start.z + 2 * (1 - t) * t * control.z + t * t * end.z + }; + } + bezier(controlPoints, t) { + const n = controlPoints.length; + let result = new Vec3(0, 0, 0); + for (let i = 0; i < n; i++) { + const coefficient = this.binomialCoefficient(n - 1, i) * Math.pow(1 - t, n - 1 - i) * Math.pow(t, i); + result = result.add(controlPoints[i].scale(coefficient)); + } + return result; + } + binomialCoefficient(n, k) { + if (k < 0 || k > n) { + return 0; + } + if (k === 0 || k === n) { + return 1; + } + return this.binomialCoefficient(n - 1, k - 1) + this.binomialCoefficient(n - 1, k); + } + catmullRom(controlPoints, t, alpha = 0.5) { + const p0 = controlPoints[0]; + const p1 = controlPoints[1]; + const p2 = controlPoints[2]; + const p3 = controlPoints[3]; + const t2 = t * t; + const t3 = t2 * t; + const h1 = -alpha * t3 + 2 * alpha * t2 - alpha * t; + const h2 = (2 - alpha) * t3 + (alpha - 3) * t2 + 1; + const h3 = (alpha - 2) * t3 + (3 - 2 * alpha) * t2 + alpha * t; + const h4 = alpha * t3 - alpha * t2; + return p0.scale(h1).add(p1.scale(h2)).add(p2.scale(h3)).add(p3.scale(h4)); + } + min(other) { + return new Vec3(Math.min(this.x, other.x), Math.min(this.y, other.y), Math.min(this.z, other.z)); + } + max(other) { + return new Vec3(Math.max(this.x, other.x), Math.max(this.y, other.y), Math.max(this.z, other.z)); + } + clamp(min, max) { + return this.max(min).min(max); + } + floor() { + return new Vec3(Math.floor(this.x), Math.floor(this.y), Math.floor(this.z)); + } + ceil() { + return new Vec3(Math.ceil(this.x), Math.ceil(this.y), Math.ceil(this.z)); + } + round() { + return new Vec3(Math.round(this.x), Math.round(this.y), Math.round(this.z)); + } + sqrt() { + return new Vec3(Math.sqrt(this.x), Math.sqrt(this.y), Math.sqrt(this.z)); + } + pow(exponent) { + return new Vec3(Math.pow(this.x, exponent), Math.pow(this.y, exponent), Math.pow(this.z, exponent)); + } + exp() { + return new Vec3(Math.exp(this.x), Math.exp(this.y), Math.exp(this.z)); + } + log() { + return new Vec3(Math.log(this.x), Math.log(this.y), Math.log(this.z)); + } + sin() { + return new Vec3(Math.sin(this.x), Math.sin(this.y), Math.sin(this.z)); + } + cos() { + return new Vec3(Math.cos(this.x), Math.cos(this.y), Math.cos(this.z)); + } + tan() { + return new Vec3(Math.tan(this.x), Math.tan(this.y), Math.tan(this.z)); + } + asin() { + return new Vec3(Math.asin(this.x), Math.asin(this.y), Math.asin(this.z)); + } + acos() { + return new Vec3(Math.acos(this.x), Math.acos(this.y), Math.acos(this.z)); + } + atan() { + return new Vec3(Math.atan(this.x), Math.atan(this.y), Math.atan(this.z)); + } + sinh() { + return new Vec3(Math.sinh(this.x), Math.sinh(this.y), Math.sinh(this.z)); + } + cosh() { + return new Vec3(Math.cosh(this.x), Math.cosh(this.y), Math.cosh(this.z)); + } + tanh() { + return new Vec3(Math.tanh(this.x), Math.tanh(this.y), Math.tanh(this.z)); + } + asinh() { + return new Vec3(Math.asinh(this.x), Math.asinh(this.y), Math.asinh(this.z)); + } + acosh() { + return new Vec3(Math.acosh(this.x), Math.acosh(this.y), Math.acosh(this.z)); + } + atanh() { + return new Vec3(Math.atanh(this.x), Math.atanh(this.y), Math.atanh(this.z)); + } + sign() { + return new Vec3(Math.sign(this.x), Math.sign(this.y), Math.sign(this.z)); + } + fract() { + return new Vec3(this.x - Math.floor(this.x), this.y - Math.floor(this.y), this.z - Math.floor(this.z)); + } + mod(other) { + return new Vec3(this.x % other.x, this.y % other.y, this.z % other.z); + } + step(edge) { + return new Vec3(this.x < edge.x ? 0 : 1, this.y < edge.y ? 0 : 1, this.z < edge.z ? 0 : 1); + } + smoothstep(edge0, edge1) { + const t = this.sub(edge0).div(edge1.sub(edge0)).clamp(Vec3.zero, Vec3.one); + return t.mul(t).mul(new Vec3(3, 3, 3).sub(t.scale(2))); + } + toTangentSpace(normal, tangent) { + const binormal = this.cross(normal); + const tangentMatrix = [ + [tangent.x, binormal.x, normal.x], + [tangent.y, binormal.y, normal.y], + [tangent.z, binormal.z, normal.z] + ]; + return this.matrixProduct(tangentMatrix); + } + perlinNoise(seed = 0) { + const permutation = new Array(256); + for (let i = 0; i < 256; i++) { + permutation[i] = (seed + i) % 256; + } + const gradients = [ + new Vec3(1, 1, 0), + new Vec3(-1, 1, 0), + new Vec3(1, -1, 0), + new Vec3(-1, -1, 0), + new Vec3(1, 0, 1), + new Vec3(-1, 0, 1), + new Vec3(1, 0, -1), + new Vec3(-1, 0, -1), + new Vec3(0, 1, 1), + new Vec3(0, -1, 1), + new Vec3(0, 1, -1), + new Vec3(0, -1, -1), + new Vec3(1, 1, 0), + new Vec3(0, -1, 1), + new Vec3(-1, 1, 0), + new Vec3(0, -1, -1) + ]; + const fade = (t) => t * t * t * (t * (t * 6 - 15) + 10); + const dotProduct = (grad, x, y, z) => grad.x * x + grad.y * y + grad.z * z; + const unitX = Math.floor(this.x) & 255; + const unitY = Math.floor(this.y) & 255; + const unitZ = Math.floor(this.z) & 255; + const relX = this.x - Math.floor(this.x); + const relY = this.y - Math.floor(this.y); + const relZ = this.z - Math.floor(this.z); + const u = fade(relX); + const v = fade(relY); + const w = fade(relZ); + const A = permutation[unitX] + unitY; + const AA = permutation[A] + unitZ; + const AB = permutation[A + 1] + unitZ; + const B = permutation[unitX + 1] + unitY; + const BA = permutation[B] + unitZ; + const BB = permutation[B + 1] + unitZ; + const gradAA = gradients[permutation[AA] % 16]; + const gradAB = gradients[permutation[AB] % 16]; + const gradBA = gradients[permutation[BA] % 16]; + const gradBB = gradients[permutation[BB] % 16]; + const lerpX1 = dotProduct(gradAA, relX, relY, relZ); + const lerpX2 = dotProduct(gradBA, relX - 1, relY, relZ); + const lerpX3 = dotProduct(gradAB, relX, relY - 1, relZ); + const lerpX4 = dotProduct(gradBB, relX - 1, relY - 1, relZ); + const lerpX5 = dotProduct(gradAA, relX, relY, relZ - 1); + const lerpX6 = dotProduct(gradBA, relX - 1, relY, relZ - 1); + const lerpX7 = dotProduct(gradAB, relX, relY - 1, relZ - 1); + const lerpX8 = dotProduct(gradBB, relX - 1, relY - 1, relZ - 1); + const lerpY1 = this.lerp(new Vec3(lerpX1, lerpX2, lerpX3), v); + const lerpY2 = this.lerp(new Vec3(lerpX4, lerpX5, lerpX6), v); + const lerpY3 = this.lerp(new Vec3(lerpX7, lerpX8, lerpX1), v); + const lerpY4 = this.lerp(new Vec3(lerpX2, lerpX3, lerpX4), v); + const lerpZ1 = this.customLerp(lerpY1, lerpY2, w); + const lerpZ2 = this.customLerp(lerpY3, lerpY4, w); + const finalNoiseValue = this.customLerp(lerpZ1, lerpZ2, w); + return finalNoiseValue; + } + customLerp(a, b, t) { + const x = a.x + t * (b.x - a.x); + const y = a.y + t * (b.y - a.y); + const z = a.z + t * (b.z - a.z); + return new Vec3(x, y, z); + } + geodesicDistance(other) { + const radius = 1; + const angle = this.angle(other); + const distance = radius * angle; + return distance; + } + catmullRomSpline(p0, p1, p2, p3, t) { + const t2 = t * t; + const t3 = t2 * t; + const h1 = -0.5 * t3 + t2 - 0.5 * t; + const h2 = 1.5 * t3 - 2.5 * t2 + 1.0; + const h3 = -1.5 * t3 + 2.0 * t2 + 0.5 * t; + const h4 = 0.5 * t3 - 0.5 * t2; + return p0.scale(h1).add(p1.scale(h2)).add(p2.scale(h3)).add(p3.scale(h4)); + } + sphericalAngle(other) { + const dotProduct = this.dot(other); + const angle = Math.acos(dotProduct / (this.length() * other.length())); + return angle; + } + complexConjugate() { + return new Vec3(this.x, -this.y, -this.z); + } + nonUniformScale(scalingFactors) { + return new Vec3(this.x * scalingFactors.x, this.y * scalingFactors.y, this.z * scalingFactors.z); + } + surfaceNormal(u, v) { + const tangentU = this.partialDerivativeU(u, v); + const tangentV = this.partialDerivativeV(u, v); + const normal = tangentU.cross(tangentV).normalize(); + return normal; + } + partialDerivativeU(u, v) { + const deltaU = 0.0001; + const point1 = this.evaluateParametricSurface(u - deltaU, v); + const point2 = this.evaluateParametricSurface(u + deltaU, v); + const tangentU = point2.sub(point1).scale(1 / (2 * deltaU)); + return tangentU; + } + partialDerivativeV(u, v) { + const deltaV = 0.0001; + const point1 = this.evaluateParametricSurface(u, v - deltaV); + const point2 = this.evaluateParametricSurface(u, v + deltaV); + const tangentV = point2.sub(point1).scale(1 / (2 * deltaV)); + return tangentV; + } + evaluateParametricSurface(u, v) { + const radius = 1.0; + const x = radius * Math.cos(u) * Math.sin(v); + const y = radius * Math.sin(u) * Math.sin(v); + const z = radius * Math.cos(v); + return new Vec3(x, y, z); + } + divergence() { + const dx = this.x - 0; + const dy = this.y - 0; + const dz = this.z - 0; + return dx + dy + dz; + } + curl() { + const i = new Vec3(1, 0, 0); + const j = new Vec3(0, 1, 0); + const k = new Vec3(0, 0, 1); + const f_z = this.add(k).z; + const b_z = this.sub(k).z; + const e_y = this.add(j).y; + const c_y = this.sub(j).y; + const d_x = this.add(i).x; + const f_x = this.sub(i).x; + const curl_x = f_z - b_z - (e_y - c_y); + const curl_y = d_x - f_x - (f_z - b_z); + const curl_z = e_y - this.y - (this.x - d_x); + return new Vec3(curl_x, curl_y, curl_z); + } + gradient(scalarField, epsilon = 1e-6) { + const dx = (scalarField(this.add(new Vec3(epsilon, 0, 0))) - + scalarField(this.sub(new Vec3(epsilon, 0, 0)))) / + (2 * epsilon); + const dy = (scalarField(this.add(new Vec3(0, epsilon, 0))) - + scalarField(this.sub(new Vec3(0, epsilon, 0)))) / + (2 * epsilon); + const dz = (scalarField(this.add(new Vec3(0, 0, epsilon))) - + scalarField(this.sub(new Vec3(0, 0, epsilon)))) / + (2 * epsilon); + return new Vec3(dx, dy, dz); + } + toArray() { + return [this.x, this.y, this.z]; + } + toString() { + return `${this.x}, ${this.y}, ${this.z}`; + } + static toString(other) { + return `${other.x}, ${other.y}, ${other.z}`; + } +} +Vec3.EPSILON = 1e-8; +Vec3.back = new Vec3(0, 0, -1); +Vec3.down = new Vec3(0, -1, 0); +Vec3.forward = new Vec3(0, 0, 1); +Vec3.left = new Vec3(-1, 0, 0); +Vec3.one = new Vec3(1, 1, 1); +Vec3.right = new Vec3(1, 0, 0); +Vec3.up = new Vec3(0, 1, 0); +Vec3.zero = new Vec3(0, 0, 0); diff --git a/BP/scripts/utils/form_builder.js b/BP/scripts/utils/form_builder.js new file mode 100644 index 0000000..f4078c2 --- /dev/null +++ b/BP/scripts/utils/form_builder.js @@ -0,0 +1,19 @@ +export class FormBuilder { + constructor(name) { + this.name = name; + this.values = []; + } + createToggle(defaultValue) { + this.defaultValue = defaultValue; + return this; + } + createTextField(defaultValue) { + this.defaultValue = defaultValue; + return this; + } + createDropdown(dropDownOptions, defaultValue) { + this.defaultValue = defaultValue; + this.values = dropDownOptions; + return this; + } +} diff --git a/BP/scripts/utils/graph.js b/BP/scripts/utils/graph.js new file mode 100644 index 0000000..b37f690 --- /dev/null +++ b/BP/scripts/utils/graph.js @@ -0,0 +1,125 @@ +import { Vec3 } from "./VectorUtils"; +function hashBlock(block) { + const prime = 31; + let hash = 1; + hash = prime * hash + Math.imul(block.x | 0, prime); + hash = prime * hash + Math.imul(block.y | 0, prime); + hash = prime * hash + Math.imul(block.z | 0, prime); + hash ^= (hash << 13); + hash ^= (hash >> 7); + hash ^= (hash << 17); + return hash >>> 0; +} +export class GraphNode { + constructor(block) { + this.index = 0; + this.block = block; + this.neighbors = new Set(); + } + addNeighbor(node) { + this.neighbors.add(node); + } + removeNeighbor(node) { + this.neighbors.delete(node); + } +} +export class Graph { + constructor() { + this.nodes = new Map(); + this.hashes = []; + } + getNode(param) { + let node; + if (param instanceof Vec3) { + node = this.nodes.get(Vec3.toString(param)); + } + else { + node = this.nodes.get(this.serializeLocation(param.location)); + } + return node; + } + addNode(param) { + if (param instanceof GraphNode) { + this.hashes.push(hashBlock(param.block)); + const key = this.serializeLocation(param.block.location); + param.index = this.nodes.size; + this.nodes.set(key, param); + return; + } + else { + const key = this.serializeLocation(param.location); + let node = this.nodes.get(key); + if (!node) { + node = new GraphNode(param); + this.nodes.set(key, node); + } + this.hashes.push(hashBlock(node.block)); + node.index = this.nodes.size - 1; + return node; + } + } + removeNode(block) { + const key = this.serializeLocation(block.location); + const node = this.nodes.get(key); + if (!node) + return; + node.neighbors.forEach(neighbor => { + neighbor.removeNeighbor(node); + node.removeNeighbor(neighbor); + }); + this.hashes.splice(this.hashes.lastIndexOf(hashBlock(block)), 1); + this.nodes.delete(key); + } + serializeLocation(location) { + return JSON.stringify(location); + } + getSize() { + return this.nodes.size; + } + traverse(startBlock, traversalType = "DFS", visit) { + const startNode = this.getNode(startBlock); + if (!startNode) { + return; + } + const visited = new Set(); + const toVisit = [startNode]; + while (toVisit.length > 0) { + const node = traversalType === "DFS" ? toVisit.pop() : toVisit.shift(); + if (!visited.has(node)) { + visit(node); + visited.add(node); + node.neighbors.forEach(neighbor => { + if (!visited.has(neighbor)) { + toVisit.push(neighbor); + } + }); + } + } + } + hash() { + return this.hashes.reduce((accumulator, currentValue) => accumulator + currentValue, 0); + } + *traverseIterative(startBlock, traversalType = "DFS") { + const startNode = this.getNode(startBlock); + if (!startNode) { + return; + } + const visited = new Set(); + const toVisit = [startNode]; + while (toVisit.length > 0) { + const node = traversalType === "DFS" ? toVisit.pop() : toVisit.shift(); + if (!visited.has(node)) { + yield node; + visited.add(node); + node.neighbors.forEach(neighbor => { + if (!visited.has(neighbor)) { + toVisit.push(neighbor); + } + }); + } + } + } + isEqual(otherGraph) { + return this.hash() === otherGraph.hash(); + } +} diff --git a/BP/scripts/utils/logger.js b/BP/scripts/utils/logger.js new file mode 100644 index 0000000..e9a3101 --- /dev/null +++ b/BP/scripts/utils/logger.js @@ -0,0 +1,46 @@ +import { system } from "@minecraft/server"; +import { serverConfigurationCopy } from "index"; +export var LogLevel; +(function (LogLevel) { + LogLevel["DEBUG"] = "DEBUG"; + LogLevel["INFO"] = "INFO"; + LogLevel["ERROR"] = "ERROR"; +})(LogLevel || (LogLevel = {})); +export class Logger { + static setLogLevel(level) { + Logger.level = level; + } + static log(level, ...message) { + const levels = [LogLevel.DEBUG, LogLevel.INFO, LogLevel.ERROR]; + const currentLevelIndex = levels.indexOf(Logger.level); + const logLevelIndex = levels.indexOf(level); + if (logLevelIndex >= currentLevelIndex && serverConfigurationCopy.debug.defaultValue) { + const timestamp = system.currentTick; + const formattedMessage = `[${timestamp}] [${level}] - ${message}`; + switch (level) { + case LogLevel.DEBUG: + console.warn(formattedMessage); + break; + case LogLevel.INFO: + console.log(formattedMessage); + break; + case LogLevel.ERROR: + console.error(formattedMessage); + break; + } + } + } + static debug(...message) { + if (serverConfigurationCopy.debug.defaultValue) + Logger.log(LogLevel.DEBUG, message); + } + static info(...message) { + if (serverConfigurationCopy.debug.defaultValue) + Logger.log(LogLevel.INFO, message); + } + static error(...message) { + if (serverConfigurationCopy.debug.defaultValue) + Logger.log(LogLevel.ERROR, message); + } +} +Logger.level = serverConfigurationCopy.debug.defaultValue ? LogLevel.DEBUG : LogLevel.INFO; diff --git a/BP/scripts/utils/utilities.js b/BP/scripts/utils/utilities.js new file mode 100644 index 0000000..bac0872 --- /dev/null +++ b/BP/scripts/utils/utilities.js @@ -0,0 +1,53 @@ +import { system } from "@minecraft/server"; +import { FormCancelationReason } from "@minecraft/server-ui"; +export function sleep(ticks) { + return new Promise((resolve) => { + system.runTimeout(resolve, ticks); + }); +} +; +export function generateUUID16() { + const characters = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789'; + let uuid = ''; + for (let i = 0; i < 16; i++) { + const randomIndex = Math.floor(Math.random() * characters.length); + uuid += characters[randomIndex]; + } + return uuid; +} +export function ExecuteAtGivenTick(tick) { + return (system.currentTick % tick) === 0; +} +export function SendMessageTo(executor, rawMessage = { rawtext: [{ text: "Not Implemented Yet" }] }) { + const formattedRawMessage = JSON.stringify(rawMessage); + executor.runCommandAsync(`tellraw ${executor.name} ` + formattedRawMessage); +} +function stackDistribution(number, groupSize = 64) { + const fullGroupsCount = Math.floor(number / groupSize); + const remainder = number % groupSize; + const groups = new Array(fullGroupsCount).fill(groupSize); + if (remainder > 0) { + groups.push(remainder); + } + return groups; +} +export function hashBlock(block) { + const inputString = `${block.dimension.id}_${block.x}-${block.y}-${block.z}`; + let hash = 5381; + for (let i = 0; i < inputString.length; i++) { + hash = (hash * 33) ^ inputString.charCodeAt(i); + } + return (hash >>> 0).toString(16); +} +async function forceShow(player, form, timeout = Infinity) { + const startTick = system.currentTick; + while ((system.currentTick - startTick) < timeout) { + const response = await (form.show(player)).catch(er => console.error(er, er.stack)); + if (response.cancelationReason !== FormCancelationReason.UserBusy) { + return response; + } + } + ; + throw new Error(`Timed out after ${timeout} ticks`); +} +export { stackDistribution, forceShow }; diff --git a/README.md b/README.md index fe89d1d..159492d 100644 --- a/README.md +++ b/README.md @@ -1,4 +1,4 @@ -

Lumber-Axe

+

Lumber Axe

@@ -7,8 +7,8 @@
- ![Development - is_in_progress](https://img.shields.io/badge/Development-finished-2ea44f?logo=visualstudiocode&logoColor=blue) - ![Version - v1.0.5](https://img.shields.io/badge/Version-v1.0.9-2ea44f?logo=git&logoColor=red) + ![Development - is_in_progress](https://img.shields.io/badge/Development-release🚀-2ea44f?logo=visualstudiocode&logoColor=blue) + ![Version Tag](https://img.shields.io/badge/Version-v2.0.0-2ea44f?logo=git&logoColor=red) [![Twitter - h_YanG_0A](https://img.shields.io/badge/Twitter-Follow_@h__YanG__0A-2ea44f?logo=twitter&logoColor=blue)](https://twitter.com/h_YanG_0A) [![Support - Donate](https://img.shields.io/badge/Support-Donate-2ea44f?logo=paypal)](https://www.paypal.com/paypalme/DennisAbaigar?country.x=PH&locale.x=en_US) ![Minecraft BE Addon Latest Release](https://img.shields.io/github/v/release/Adr-hyng-OSS/Lumber-Axe?logo=xbox&logoColor=green&label=Minecraft) @@ -19,7 +19,7 @@ ----- -

UPLOAD HD AS LOGO (WIP)

+

UPLOAD HD AS LOGO (WIP)

🪓 Lumber axe is an addon that chops a whole tree of connected blocks of logs down with just a single break at any side of the connected block of log. This accurately chops the whole tree down with the power of Minecraft Bedrock's Experimental Features. @@ -44,15 +44,6 @@
> It will automatically install the `Behavior Pack & Resource Pack` if you opened the `.mcaddon` file.
> It will automatically install `Behavior Pack` if you installed the **v1.20.0-Lumber_Axe_BP.mcpack**, and **v1.20.0-Lumber_Axe_BP.mcpack** for `Resource Pack`. - After installing both packs, you can add it to your own world with few setup to work with In-game. -## 🕹️ *In-Game:* - -Make sure you **TURN ON** the following features or capabilities in the game: - -![Pocket Edition Experimental Toggle](./guidelines/readme/experimental.png) - -To fully activate the addon. Make sure to activate the Behavior Pack and Resource Pack to your selected world. - -

📃 License 📃

diff --git a/RP/animation_controllers/block_outline.animation_controllers.json b/RP/animation_controllers/block_outline.animation_controllers.json new file mode 100644 index 0000000..f022f7d --- /dev/null +++ b/RP/animation_controllers/block_outline.animation_controllers.json @@ -0,0 +1,40 @@ +{ + "format_version": "1.10.0", + "animation_controllers": { + "controller.animation.block_outline.controller": { + "states": { + "default": { + "on_entry": [ + "variable.trunk_size = q.has_property('yn:trunk_size') ? q.property('yn:trunk_size') : 0;", + "variable.dig_particle_texture_coordinate = query.surface_particle_texture_coordinate;", + "variable.dig_particle_texture_size = query.surface_particle_texture_size;", + "variable.dig_particle_color = query.surface_particle_color;" + ], + "on_exit": [ + "variable.trunk_size = q.has_property('yn:trunk_size') ? q.property('yn:trunk_size') : 0;", + "variable.dig_particle_texture_coordinate = query.surface_particle_texture_coordinate;", + "variable.dig_particle_texture_size = query.surface_particle_texture_size;", + "variable.dig_particle_color = query.surface_particle_color;" + ], + "transitions": [ + {"show_destroy_block": "q.has_property('yn:trunk_size') && q.property('yn:trunk_size') != 0"} + ] + }, + "show_destroy_block": { + "on_entry": [ + "variable.trunk_size = q.has_property('yn:trunk_size') ? q.property('yn:trunk_size') : 0;", + "variable.dig_particle_texture_coordinate = query.surface_particle_texture_coordinate;", + "variable.dig_particle_texture_size = query.surface_particle_texture_size;", + "variable.dig_particle_color = query.surface_particle_color;" + ], + "on_exit": [ + "variable.trunk_size = q.has_property('yn:trunk_size') ? q.property('yn:trunk_size') : 0;" + ], + "animations": [ + "show_destroy_block_anim" + ] + } + } + } + } +} \ No newline at end of file diff --git a/RP/animations/animation.block_outline.json b/RP/animations/animation.block_outline.json new file mode 100644 index 0000000..9d981ed --- /dev/null +++ b/RP/animations/animation.block_outline.json @@ -0,0 +1,26 @@ +{ + "format_version": "1.8.0", + "animations": { + "animation.block_outline.render_scale": { + "loop": true, + "bones": { + "root": { + "rotation": [0, 0, 0] + }, + "outline": { + "scale": "1.04 + math.sqrt(query.distance_from_camera) / 64" + } + } + }, + "animation.block_outline.spawn_particle": { + "loop": "hold_on_last_frame", + "animation_length": 1.0, + "particle_effects": { + "0.1": { + "pre_effect_script": "variable.trunk_size = q.has_property('yn:trunk_size') ? q.property('yn:trunk_size') : 0;variable.dig_particle_texture_coordinate = query.surface_particle_texture_coordinate; variable.dig_particle_texture_size = query.surface_particle_texture_size; variable.dig_particle_color = query.surface_particle_color;", + "effect": "destroy_block" + } + } + } + } +} \ No newline at end of file diff --git a/RP/entity/block_outline.json b/RP/entity/block_outline.json new file mode 100644 index 0000000..e042da9 --- /dev/null +++ b/RP/entity/block_outline.json @@ -0,0 +1,57 @@ +{ + "format_version": "1.10.0", + "minecraft:client_entity": { + "description": { + "identifier": "yn:block_outline", + "materials": { + "base": "block_outline.base", + "outline": "block_outline.outline" + }, + "textures": { + "default": "textures/entity/example" + }, + "geometry": { + "default": "geometry.block_outline" + }, + "render_controllers": [ + "controller.render.block_outline.base", + {"controller.render.block_outline.outline": "query.variant"} + ], + + "scripts": { + "initialize": [ + "temp.red = 1;", + "temp.green = 1;", + "temp.blue = 1;", + "variable.dig_particle_texture_coordinate = query.surface_particle_texture_coordinate;", + "variable.dig_particle_texture_size = query.surface_particle_texture_size;", + "variable.dig_particle_color = query.surface_particle_color;", + "variable.trunk_size = q.has_property('yn:trunk_size') ? q.property('yn:trunk_size') : 0;" + ], + "pre_animation": [ + "temp.red = 1;", + "temp.green = 1;", + "temp.blue = 1;", + "variable.dig_particle_texture_coordinate = query.surface_particle_texture_coordinate;", + "variable.dig_particle_texture_size = query.surface_particle_texture_size;", + "variable.dig_particle_color = query.surface_particle_color;", + "variable.trunk_size = q.has_property('yn:trunk_size') ? q.property('yn:trunk_size') : 0;" + ], + "scale": "query.variant == 1 ? (1.0) : (0.0)", + "animate": [ + "render", + "particle_controller" + ], + "should_update_effects_offscreen": "1.0" + }, + "animations": { + "render": "animation.block_outline.render_scale", + "show_destroy_block_anim": "animation.block_outline.spawn_particle", + "particle_controller": "controller.animation.block_outline.controller" + }, + "particle_effects": { + "destroy_block": "yn:custom_destroy" + } + } + } +} \ No newline at end of file diff --git a/RP/manifest.json b/RP/manifest.json index eafe645..4081702 100644 --- a/RP/manifest.json +++ b/RP/manifest.json @@ -1,18 +1,18 @@ { "format_version": 2, "header": { - "name": "Lumber Axe RP: Chopping made easy! 1.0.5 [DEBUG]", - "description": "Cut the whole tree with easy as a lumberjack's pie! We create addons/datapack that will ease your survival plays. \n @Created By: @h_YanG_0A & @Brilliant", + "name": "Lumber Axe RP §8(2.0.0)", + "description": "An addon that gives you a lumberjack axes, which can preview the tree's status, and chopdown trees. \n @Created By: @Adr-hyng", "uuid": "62b4cab4-fa42-405b-bff6-1f47b6958dd6", "version": [ - 1, + 2, 0, - 5 + 0 ], "min_engine_version": [ 1, - 20, - 10 + 21, + 0 ] }, "modules": [ @@ -29,7 +29,8 @@ "metadata": { "authors": [ "@h_YanG_0A", - "@Brilliant" + "Contributor-@Brilliant", + "Contributor-@Dal4y" ], "license": "GPL-3.0-or-later", "url": "https://twitter.com/h_YanG_0A" diff --git a/RP/materials/entity.material b/RP/materials/entity.material new file mode 100644 index 0000000..0952c17 --- /dev/null +++ b/RP/materials/entity.material @@ -0,0 +1,45 @@ +{ + "materials": { + "version": "1.0.0", + + "block_outline.base:entity_alphatest": { + "+states": [ + "EnableStencilTest", + "StencilWrite" + ], + + "frontFace": { + "stencilFunc": "Always", + "stencilFailOp": "Replace", + "stencilDepthFailOp": "Replace", + "stencilPassOp": "Replace" + }, + "backFace": { + "stencilFunc": "Always", + "stencilFailOp": "Replace", + "stencilDepthFailOp": "Replace", + "stencilPassOp": "Replace" + }, + + "stencilRef": 3 + }, + + "block_outline.outline:entity_alphatest": { + "+states": [ + "EnableStencilTest", + "InvertCulling" + ], + "-states": ["DisableCulling"], + + "-defines": ["FANCY"], + + "depthFunc": "Always", + + "frontFace": { + "stencilFunc": "NotEqual" + }, + + "stencilRef": 3 + } + } +} \ No newline at end of file diff --git a/RP/models/entity/example.json b/RP/models/entity/example.json new file mode 100644 index 0000000..3c0be7a --- /dev/null +++ b/RP/models/entity/example.json @@ -0,0 +1,61 @@ +{ + "format_version": "1.16.0", + "minecraft:geometry": [ + { + "description": { + "identifier": "geometry.block_outline", + "texture_width": 1, + "texture_height": 1, + "visible_bounds_width": 3, + "visible_bounds_height": 4, + "visible_bounds_offset": [0, 1, 0] + }, + "bones": [ + { + "name": "root", + "pivot": [0, 18, 0] + }, + { + "name": "base", + "parent": "root", + "pivot": [0, 18, 0], + "cubes": [ + { + "origin": [-8, 0, -8], + "size": [16, 16, 16], + "inflate": -0.25, + "uv": { + "north": {"uv": [0, 0], "uv_size": [1, 1]}, + "east": {"uv": [0, 0], "uv_size": [1, 1]}, + "south": {"uv": [0, 0], "uv_size": [1, 1]}, + "west": {"uv": [0, 0], "uv_size": [1, 1]}, + "up": {"uv": [0, 0], "uv_size": [1, 1]}, + "down": {"uv": [0, 0], "uv_size": [1, 1]} + } + } + ] + }, + { + "name": "outline", + "parent": "root", + "pivot": [0, 18, 0], + "cubes": [ + { + "origin": [-8, 0, -8], + "size": [16, 16, 16], + "inflate": -0.25, + "uv": { + "north": {"uv": [0, 0], "uv_size": [1, 1]}, + "east": {"uv": [0, 0], "uv_size": [1, 1]}, + "south": {"uv": [0, 0], "uv_size": [1, 1]}, + "west": {"uv": [0, 0], "uv_size": [1, 1]}, + "up": {"uv": [0, 0], "uv_size": [1, 1]}, + "down": {"uv": [0, 0], "uv_size": [1, 1]} + } + } + ] + } + ] + } + ] +} \ No newline at end of file diff --git a/RP/pack_icon.png b/RP/pack_icon.png index a7674c9..d29dae1 100644 Binary files a/RP/pack_icon.png and b/RP/pack_icon.png differ diff --git a/RP/particles/custom_destroy.particle.json b/RP/particles/custom_destroy.particle.json new file mode 100644 index 0000000..08d36a7 --- /dev/null +++ b/RP/particles/custom_destroy.particle.json @@ -0,0 +1,73 @@ +{ + "format_version": "1.10.0", + "particle_effect": { + "description": { + "identifier": "yn:custom_destroy", + "basic_render_parameters": { + "material": "particles_opaque", + "texture": "atlas.terrain" + } + }, + "curves": { + "variable.kill": { + "type": "linear", + "input": "v.particle_age", + "horizontal_range": "v.particle_lifetime", + "nodes": [0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1] + }, + "variable.collide_things": { + "type": "linear", + "input": "v.particle_age", + "horizontal_range": "v.particle_lifetime", + "nodes": [0, 0, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1] + } + }, + "components": { + "minecraft:emitter_initialization": { + "creation_expression": "temp.start_texture_color_r = 0.11;temp.start_texture_color_g = 0.11;temp.start_texture_color_b = 0.11;temp.start_texture_color_a = 0;temp.trunk_size = 1;" + }, + "minecraft:emitter_rate_instant": { + "num_particles": "Math.random(5, 15) * (t.trunk_size * (v.trunk_size > 1 ? 1.2:1))" + }, + "minecraft:emitter_lifetime_once": { + "active_time": 0.5 + }, + "minecraft:emitter_shape_sphere": { + "offset": [0, 0.5, 0], + "radius": 1, + "direction": "outwards" + }, + "minecraft:particle_initialization": { + "per_update_expression": "temp.start_texture_color_r = Math.clamp(variable.dig_particle_color.r, 0, 1);temp.start_texture_color_g = Math.clamp(variable.dig_particle_color.g, 0, 1);temp.start_texture_color_b = Math.clamp(variable.dig_particle_color.b, 0, 1);temp.start_texture_color_a = Math.clamp(variable.dig_particle_color.a, 0, 1);temp.trunk_size = v.trunk_size;" + }, + "minecraft:particle_lifetime_expression": { + "max_lifetime": "Math.random(1.0, 2.5)", + "expiration_expression": "((t.start_texture_color_r <= 0.01 && t.start_texture_color_g <= 0.01 && t.start_texture_color_b <= 0.01) && variable.kill == 1) && !query.is_in_ui" + }, + "minecraft:particle_initial_speed": 2, + "minecraft:particle_motion_dynamic": { + "linear_acceleration": [0, -9.8, 0] + }, + "minecraft:particle_appearance_billboard": { + "size": ["variable.particle_random_1 * 0.0375 + 0.0375", "variable.particle_random_1 * 0.0375 + 0.0375"], + "facing_camera_mode": "lookat_xyz", + "uv": { + "texture_width": 1, + "texture_height": 1, + "uv": ["variable.dig_particle_texture_coordinate.u + (variable.dig_particle_texture_size.u/4) * (variable.particle_random_1*3)", "variable.dig_particle_texture_coordinate.v + (variable.dig_particle_texture_size.v/4) * (variable.particle_random_2*3)"], + "uv_size": ["variable.dig_particle_texture_size.u/4", "variable.dig_particle_texture_size.v/4"] + } + }, + "minecraft:particle_motion_collision": { + "enabled": "v.collide_things", + "collision_drag": 10, + "coefficient_of_restitution": 0.1, + "collision_radius": 0.01 + }, + "minecraft:particle_appearance_lighting": {}, + "minecraft:particle_appearance_tinting": { + "color": ["t.start_texture_color_r", "t.start_texture_color_g", "t.start_texture_color_b", "t.start_texture_color_a"] + } + } + } +} \ No newline at end of file diff --git a/RP/particles/inspecting_indicator.particle.json b/RP/particles/inspecting_indicator.particle.json new file mode 100644 index 0000000..bbfd4be --- /dev/null +++ b/RP/particles/inspecting_indicator.particle.json @@ -0,0 +1,48 @@ +{ + "format_version": "1.10.0", + "particle_effect": { + "description": { + "identifier": "yn:inspecting_indicator", + "basic_render_parameters": { + "material": "particles_blend", + "texture": "textures/particles/indicator" + } + }, + "components": { + "minecraft:emitter_initialization": { + "creation_expression": "v.color;v.size = 0.5;" + }, + "minecraft:emitter_rate_instant": { + "num_particles": 1 + }, + "minecraft:emitter_lifetime_once": { + "active_time": 1 + }, + "minecraft:emitter_shape_point": { + "direction": [0, 1, 0] + }, + "minecraft:particle_lifetime_expression": { + "max_lifetime": "v.max_age" + }, + "minecraft:particle_initial_speed": "v.height / v.max_age", + "minecraft:particle_motion_dynamic": {}, + "minecraft:particle_appearance_billboard": { + "size": ["v.size * v.radius", "v.size * v.radius"], + "facing_camera_mode": "direction_z", + "direction": { + "mode": "custom", + "custom_direction": [0, 1, 0] + }, + "uv": { + "texture_width": 16, + "texture_height": 16, + "uv": [0, 0], + "uv_size": [16, 16] + } + }, + "minecraft:particle_appearance_tinting": { + "color": ["v.color.r", "v.color.g", "v.color.b", 1] + } + } + } +} \ No newline at end of file diff --git a/RP/particles/tree_dust.particle.json b/RP/particles/tree_dust.particle.json new file mode 100644 index 0000000..0bf1339 --- /dev/null +++ b/RP/particles/tree_dust.particle.json @@ -0,0 +1,54 @@ +{ + "format_version": "1.10.0", + "particle_effect": { + "description": { + "identifier": "yn:tree_dust", + "basic_render_parameters": { + "material": "particles_alpha", + "texture": "textures/particles/dust" + } + }, + "components": { + "minecraft:emitter_initialization": { + "creation_expression": "v.splash_range = 2;" + }, + "minecraft:emitter_rate_instant": { + "num_particles": "7 * v.trunk_size" + }, + "minecraft:emitter_lifetime_once": { + "active_time": 1 + }, + "minecraft:emitter_shape_point": { + "offset": ["variable.is_vertical = Math.Random(0.0, 1.0) > 0.7;variable.splash_power = Math.random(0.0, 1.0) * variable.splash_range;return 0.0;", 0.1, 0], + "direction": ["(variable.is_vertical > 0 ? 0.15 : 1.0) * Math.random(-1,1)", 0, "(variable.is_vertical > 0 ? 0.15 : 1.0) * Math.random(-1,1)"] + }, + "minecraft:particle_lifetime_expression": { + "max_lifetime": 1 + }, + "minecraft:particle_initial_speed": "v.trunk_size", + "minecraft:particle_motion_dynamic": { + "linear_acceleration": ["(variable.particle_random_1 * 2 - 1) * variable.splash_power * (variable.is_vertical > 0 ? 0.2 : 0.5) * 1.0", 0, "(variable.particle_random_2 * 2 - 1) * variable.splash_power * (variable.is_vertical > 0 ? 0.2 : 0.5) * 1.0"], + "linear_drag_coefficient": 2 + }, + "minecraft:particle_appearance_billboard": { + "size": [0.25, 0.25], + "facing_camera_mode": "lookat_xyz", + "uv": { + "texture_width": 16, + "texture_height": 192, + "flipbook": { + "base_UV": [0, 0], + "size_UV": [16, 16], + "step_UV": [0, 16], + "frames_per_second": 12, + "max_frame": 12, + "stretch_to_lifetime": true + } + } + }, + "minecraft:particle_appearance_tinting": { + "color": [0.70588, 0.44314, 0.20392, 1] + } + } + } +} \ No newline at end of file diff --git a/RP/render_controllers/block_outline.json b/RP/render_controllers/block_outline.json new file mode 100644 index 0000000..14fd303 --- /dev/null +++ b/RP/render_controllers/block_outline.json @@ -0,0 +1,38 @@ +{ + "format_version": "1.10.0", + + "render_controllers": { + "controller.render.block_outline.base": { + "geometry": "geometry.default", + "materials": [{"base": "material.base"}], + "textures": ["texture.default"], + + "part_visibility": [ + { + "*": false, + "base": true + } + ] + }, + "controller.render.block_outline.outline": { + "geometry": "geometry.default", + "materials": [{"outline": "material.outline"}], + "textures": ["texture.default"], + + "part_visibility": [ + { + "*": false, + "outline": true + } + ], + + "ignore_lighting": true, + "overlay_color": { + "r": "t.red", + "g": "t.green", + "b": "t.blue", + "a": 1 + } + } + } +} \ No newline at end of file diff --git a/RP/texts/en_GB.lang b/RP/texts/en_GB.lang new file mode 100644 index 0000000..d6447d9 --- /dev/null +++ b/RP/texts/en_GB.lang @@ -0,0 +1,60 @@ +## Items +item.yn:wooden_lumber_axe.name=Wooden Lumber Axe +item.yn:stone_lumber_axe.name=Stone Lumber Axe +item.yn:golden_lumber_axe.name=Golden Lumber Axe +item.yn:iron_lumber_axe.name=Iron Lumber Axe +item.yn:diamond_lumber_axe.name=Diamond Lumber Axe +item.yn:netherite_lumber_axe.name=Netherite Lumber Axe + +## Don't Change +LumberAxe.form.canBeChopped.text=Choppable +LumberAxe.form.cannotBeChopped.text=Cannot be Chopped + +## Form-UI +LumberAxe.form.title.text=Tree Status Overview +LumberAxe.form.treeSizeAbrev.text=Has +LumberAxe.form.treeSizeAbrevLogs.text=Log/s +LumberAxe.form.trunkHeightAbrev.text=Block/s High + +## Log Messages +LumberAxe.on_load_message=§a§lLumber Axe Addon§r §ahas been loaded successfully. Created by Adr-hyng. +LumberAxe.on_caught_invalid_command=§cInvalid Usage Format. %1 +LumberAxe.on_database_empty=§4No configuration record found in database.§r +LumberAxe.on_database_reset=§aThe database has been reset.§r +LumberAxe.show_database=Database ID: §e%1§r%2%3 +### LumberAxe.show_help_message= -> Not Implemented +LumberAxe.on_caught_command_404=§cError while fetching %1 command: %2 +LumberAxe.on_caught_main_command_not_found=§cInvalid Command: §l%1§r%2§cCheck If The Command Actually Exists. Use /scriptevent %3 help +LumberAxe.server.invalid_log_amount_limitation=Cannot chop the whole tree due to server configuration limitation of %1. Ask server operator to increase / adjust. + +## Server Options Details +LumberAxe.server.durability_damage_per_block=Durability loss per block +LumberAxe.server.chop_limit=Tree Logs Limitation +LumberAxe.server.immersive_chopping=Enable Immersive Mode +LumberAxe.server.immersive_delay=Immersive Mode Delay Time + +## Configuration Options +LumberAxe.configuration.title=%1 Settings +LumberAxe.configuration.general=General Options +LumberAxe.configuration.log_include_manager=Included Logs Manager +LumberAxe.configuration.log_exclude_manager=Excluded Logs Manager + +## Include Manager +LumberAxe.log_include_manager.drop_down=Included Choppable Logs +LumberAxe.log_include_manager.text_field=Log Block to Include +LumberAxe.log_include_manager.toggle=Add +LumberAxe.log_include_manager.add_success_log=§aBlock named '%1' was successfully added to §oIncluded Choppable Blocks. +LumberAxe.log_include_manager.remove_success_log=§cBlock named '%1' was successfully removed from §oIncluded Choppable Blocks. +LumberAxe.log_include_manager.update_success_log=§eBlock named '%1' from §oIncluded Choppable Blocks§r §ewas successfully updated to '%2' in §oIncluded Choppable Blocks. + +## Exclude Manager +LumberAxe.log_exclude_manager.drop_down=Excluded Choppable Logs +LumberAxe.log_exclude_manager.text_field=Log Block to Exclude +LumberAxe.log_exclude_manager.toggle=Add +LumberAxe.log_exclude_manager.add_success_log=§aBlock named '%1' was successfully added to §oExcluded Choppable Blocks. +LumberAxe.log_exclude_manager.remove_success_log=§cBlock named '%1' was successfully removed from §oExcluded Choppable Blocks. +LumberAxe.log_exclude_manager.update_success_log=§eBlock named '%1' from §oExcluded Choppable Blocks§r §ewas successfully updated to '%2' in §oExcluded Choppable Blocks. + +## Instructions +LumberAxe.log_include_manager.info=How to include custom log: %1-Add: To add a custom log entry to be chopped with lumber axe, you will need to select the dropdown to "Empty" (default). Then, press the "Add" toggle, and start putting custom log identifier in the textfield provided, and submit. %1Example: %1korbon:redwood_log.%1-Delete: To delete an entry, you just need to select an entry within the dropdown, and don't press the "Add" toggle. Then upon submission, it will be deleted. %1-Update: To update an existing entry, similar to adding/inserting it, but instead of making the dropdown selection to be "Empty", select other entry to update with, and submit. %1%1Note: Including a custom log is ignored, if this custom log is already existing in the excluded logs OR it starts with 'stripped_'. +LumberAxe.log_exclude_manager.info=How to exclude custom log: %1-Add: To add a custom log entry to be ignored with lumber axe functionality, you will need to select the dropdown to "Empty" (default). Then, press the "Add" toggle, and start putting custom log identifier in the textfield provided, and submit. %1Example: %1korbon:redwood_log.%1-Delete: To delete an entry, you just need to select an entry within the dropdown, and don't press the "Add" toggle. Then upon submission, it will be deleted. %1-Update: To update an existing entry, similar to adding/inserting it, but instead of making the dropdown selection to be "Empty", select other entry to update with, and submit. %1%1Note: Excluding a custom log is prioritized over including. diff --git a/RP/texts/en_US.lang b/RP/texts/en_US.lang index 6614651..d6447d9 100644 --- a/RP/texts/en_US.lang +++ b/RP/texts/en_US.lang @@ -1,4 +1,4 @@ -## Items +## Items item.yn:wooden_lumber_axe.name=Wooden Lumber Axe item.yn:stone_lumber_axe.name=Stone Lumber Axe item.yn:golden_lumber_axe.name=Golden Lumber Axe @@ -6,15 +6,55 @@ item.yn:iron_lumber_axe.name=Iron Lumber Axe item.yn:diamond_lumber_axe.name=Diamond Lumber Axe item.yn:netherite_lumber_axe.name=Netherite Lumber Axe +## Don't Change +LumberAxe.form.canBeChopped.text=Choppable +LumberAxe.form.cannotBeChopped.text=Cannot be Chopped -## Form-UI -LumberAxe.form.title.text=LOG INFORMATION -LumberAxe.form.treeSizeAbrev.text=HAS -LumberAxe.form.treeSizeAbrevLogs.text=LOG/S -LumberAxe.form.durabilityAbrev.text=DMG -LumberAxe.form.maxDurabilityAbrev.text=MAX -LumberAxe.form.canBeChopped.text=§aChoppable -LumberAxe.form.cannotBeChopped.text=§cCannot be chopped +## Form-UI +LumberAxe.form.title.text=Tree Status Overview +LumberAxe.form.treeSizeAbrev.text=Has +LumberAxe.form.treeSizeAbrevLogs.text=Log/s +LumberAxe.form.trunkHeightAbrev.text=Block/s High -## Error Message -LumberAxe.watchdogError.hang.text=Scripting Error: Try chopping or inspecting smaller trees or different angle. \ No newline at end of file +## Log Messages +LumberAxe.on_load_message=§a§lLumber Axe Addon§r §ahas been loaded successfully. Created by Adr-hyng. +LumberAxe.on_caught_invalid_command=§cInvalid Usage Format. %1 +LumberAxe.on_database_empty=§4No configuration record found in database.§r +LumberAxe.on_database_reset=§aThe database has been reset.§r +LumberAxe.show_database=Database ID: §e%1§r%2%3 +### LumberAxe.show_help_message= -> Not Implemented +LumberAxe.on_caught_command_404=§cError while fetching %1 command: %2 +LumberAxe.on_caught_main_command_not_found=§cInvalid Command: §l%1§r%2§cCheck If The Command Actually Exists. Use /scriptevent %3 help +LumberAxe.server.invalid_log_amount_limitation=Cannot chop the whole tree due to server configuration limitation of %1. Ask server operator to increase / adjust. + +## Server Options Details +LumberAxe.server.durability_damage_per_block=Durability loss per block +LumberAxe.server.chop_limit=Tree Logs Limitation +LumberAxe.server.immersive_chopping=Enable Immersive Mode +LumberAxe.server.immersive_delay=Immersive Mode Delay Time + +## Configuration Options +LumberAxe.configuration.title=%1 Settings +LumberAxe.configuration.general=General Options +LumberAxe.configuration.log_include_manager=Included Logs Manager +LumberAxe.configuration.log_exclude_manager=Excluded Logs Manager + +## Include Manager +LumberAxe.log_include_manager.drop_down=Included Choppable Logs +LumberAxe.log_include_manager.text_field=Log Block to Include +LumberAxe.log_include_manager.toggle=Add +LumberAxe.log_include_manager.add_success_log=§aBlock named '%1' was successfully added to §oIncluded Choppable Blocks. +LumberAxe.log_include_manager.remove_success_log=§cBlock named '%1' was successfully removed from §oIncluded Choppable Blocks. +LumberAxe.log_include_manager.update_success_log=§eBlock named '%1' from §oIncluded Choppable Blocks§r §ewas successfully updated to '%2' in §oIncluded Choppable Blocks. + +## Exclude Manager +LumberAxe.log_exclude_manager.drop_down=Excluded Choppable Logs +LumberAxe.log_exclude_manager.text_field=Log Block to Exclude +LumberAxe.log_exclude_manager.toggle=Add +LumberAxe.log_exclude_manager.add_success_log=§aBlock named '%1' was successfully added to §oExcluded Choppable Blocks. +LumberAxe.log_exclude_manager.remove_success_log=§cBlock named '%1' was successfully removed from §oExcluded Choppable Blocks. +LumberAxe.log_exclude_manager.update_success_log=§eBlock named '%1' from §oExcluded Choppable Blocks§r §ewas successfully updated to '%2' in §oExcluded Choppable Blocks. + +## Instructions +LumberAxe.log_include_manager.info=How to include custom log: %1-Add: To add a custom log entry to be chopped with lumber axe, you will need to select the dropdown to "Empty" (default). Then, press the "Add" toggle, and start putting custom log identifier in the textfield provided, and submit. %1Example: %1korbon:redwood_log.%1-Delete: To delete an entry, you just need to select an entry within the dropdown, and don't press the "Add" toggle. Then upon submission, it will be deleted. %1-Update: To update an existing entry, similar to adding/inserting it, but instead of making the dropdown selection to be "Empty", select other entry to update with, and submit. %1%1Note: Including a custom log is ignored, if this custom log is already existing in the excluded logs OR it starts with 'stripped_'. +LumberAxe.log_exclude_manager.info=How to exclude custom log: %1-Add: To add a custom log entry to be ignored with lumber axe functionality, you will need to select the dropdown to "Empty" (default). Then, press the "Add" toggle, and start putting custom log identifier in the textfield provided, and submit. %1Example: %1korbon:redwood_log.%1-Delete: To delete an entry, you just need to select an entry within the dropdown, and don't press the "Add" toggle. Then upon submission, it will be deleted. %1-Update: To update an existing entry, similar to adding/inserting it, but instead of making the dropdown selection to be "Empty", select other entry to update with, and submit. %1%1Note: Excluding a custom log is prioritized over including. diff --git a/RP/texts/es_ES.lang b/RP/texts/es_ES.lang index 0c5c614..4e74746 100644 --- a/RP/texts/es_ES.lang +++ b/RP/texts/es_ES.lang @@ -6,15 +6,82 @@ item.yn:iron_lumber_axe.name=Hacha de madera de hierro item.yn:diamond_lumber_axe.name=Hacha de madera de diamante item.yn:netherite_lumber_axe.name=Hacha de madera inframundita +LumberAxe.form.canBeChopped.text=ael árbol se puede cortar +LumberAxe.form.cannotBeChopped.text=cel árbol no se puede cortar -## Form-UI -LumberAxe.form.title.text=Información del árbol -LumberAxe.form.treeSizeAbrev.text=TIENE -LumberAxe.form.treeSizeAbrevLogs.text=roncos de árboles -LumberAxe.form.durabilityAbrev.text=DMG -LumberAxe.form.maxDurabilityAbrev.text=MAX -LumberAxe.form.canBeChopped.text=§ael árbol se puede cortar -LumberAxe.form.cannotBeChopped.text=§cel árbol no se puede cortar - -## Error Message -LumberAxe.watchdogError.hang.text=Error de secuencias de comandos: intente cortar o inspeccionar árboles más pequeños o en un ángulo diferente. \ No newline at end of file +## Form-UI +LumberAxe.form.title.text=Resumen del Estado del Árbol +LumberAxe.form.treeSizeAbrev.text=Tiene +LumberAxe.form.treeSizeAbrevLogs.text=Tronco/s +LumberAxe.form.trunkHeightAbrev.text=Bloque/s de Altura + +<<<<<<< HEAD +LumberAxe.on_load_message=§a§lLumber Axe Addon§r §aha sido cargado correctamente. Creado por Adr-hyng. +LumberAxe.on_caught_invalid_command=§cFormato de uso inválido. %1 +LumberAxe.on_database_empty=§4No se encontró ningún registro de configuración en la base de datos.§r +LumberAxe.on_database_reset=§aLa base de datos ha sido restablecida.§r +LumberAxe.show_database=ID de la base de datos: §e%1§r%2%3 +LumberAxe.on_caught_command_404=§cError al obtener el comando %1: %2 +LumberAxe.on_caught_main_command_not_found=§cComando inválido: §l%1§r%2§cVerifica si el comando realmente existe. Usa /scriptevent %3 help + +LumberAxe.server.show_message_on_join=Mostrar mensaje al unirse +LumberAxe.server.durability_damage_per_block=Pérdida de durabilidad por bloque +LumberAxe.server.chop_limit=Límite de troncos de árbol + +LumberAxe.configuration.title=Configuración de %1 +LumberAxe.configuration.general=Opciones generales +LumberAxe.configuration.log_include_manager=Gestor de troncos incluidos +LumberAxe.configuration.log_exclude_manager=Gestor de troncos excluidos + +LumberAxe.log_include_manager.drop_down=Troncos cortables incluidos +LumberAxe.log_include_manager.text_field=Bloque de tronco a incluir +LumberAxe.log_include_manager.toggle=Insertar +LumberAxe.log_include_manager.add_success_log=§aEl bloque llamado '%1' se ha añadido correctamente a §oTroncos cortables incluidos. +LumberAxe.log_include_manager.remove_success_log=§cEl bloque llamado '%1' se ha eliminado correctamente de §oTroncos cortables incluidos. +LumberAxe.log_include_manager.update_success_log=§eEl bloque llamado '%1' de §oTroncos cortables incluidos§r §ese ha actualizado correctamente a '%2' en §oTroncos cortables incluidos. + +LumberAxe.log_exclude_manager.drop_down=Troncos cortables excluidos +LumberAxe.log_exclude_manager.text_field=Bloque de tronco a excluir +LumberAxe.log_exclude_manager.toggle=Insertar +LumberAxe.log_exclude_manager.add_success_log=§aEl bloque llamado '%1' se ha añadido correctamente a §oTroncos cortables excluidos. +LumberAxe.log_exclude_manager.remove_success_log=§cEl bloque llamado '%1' se ha eliminado correctamente de §oTroncos cortables excluidos. +LumberAxe.log_exclude_manager.update_success_log=§eEl bloque llamado '%1' de §oTroncos cortables excluidos§r §ese ha actualizado correctamente a '%2' en §oTroncos cortables excluidos. +======= +## Log Messages +LumberAxe.on_load_message=§a§lLumber Axe Addon§r §ase ha cargado con éxito. Creado por Adr-hyng. +LumberAxe.on_caught_invalid_command=§cFormato de uso incorrecto. %1 +LumberAxe.on_database_empty=§4No se encontraron registros de configuración en la base de datos.§r +LumberAxe.on_database_reset=§aLa base de datos ha sido reiniciada.§r +LumberAxe.show_database=ID de la base de datos: §e%1§r%2%3 +LumberAxe.on_caught_command_404=§cError al buscar el comando %1: %2 +LumberAxe.on_caught_main_command_not_found=§cComando inválido: §l%1§r%2§cVerifica si el comando realmente existe. Usa /scriptevent %3 ayuda +LumberAxe.server.invalid_log_amount_limitation=No se puede talar todo el árbol debido a la limitación de configuración del servidor de %1. Solicita al operador del servidor que lo ajuste. + +## Server Options Details +LumberAxe.server.durability_damage_per_block=Pérdida de durabilidad por bloque +LumberAxe.server.chop_limit=Límite de troncos +LumberAxe.server.immersive_chopping=Habilitar Modo Inmersivo +LumberAxe.server.immersive_delay=Tiempo de retraso del Modo Inmersivo + +## Configuration Options +LumberAxe.configuration.title=Ajustes de %1 +LumberAxe.configuration.general=Opciones Generales +LumberAxe.configuration.log_include_manager=Administrador de Troncos Incluidos +LumberAxe.configuration.log_exclude_manager=Administrador de Troncos Excluidos + +## Include Manager +LumberAxe.log_include_manager.drop_down=Troncos Incluidos Talar +LumberAxe.log_include_manager.text_field=Bloque de Tronco a Incluir +LumberAxe.log_include_manager.toggle=Añadir +LumberAxe.log_include_manager.add_success_log=§aEl bloque llamado '%1' se ha añadido con éxito a §oBloques de Troncos Incluidos. +LumberAxe.log_include_manager.remove_success_log=§cEl bloque llamado '%1' se ha eliminado con éxito de §oBloques de Troncos Incluidos. +LumberAxe.log_include_manager.update_success_log=§eEl bloque llamado '%1' de §oBloques de Troncos Incluidos§r §ese ha actualizado con éxito a '%2' en §oBloques de Troncos Incluidos. + +## Exclude Manager +LumberAxe.log_exclude_manager.drop_down=Troncos Excluidos Talar +LumberAxe.log_exclude_manager.text_field=Bloque de Tronco a Excluir +LumberAxe.log_exclude_manager.toggle=Añadir +LumberAxe.log_exclude_manager.add_success_log=§aEl bloque llamado '%1' se ha añadido con éxito a §oBloques de Troncos Excluidos. +LumberAxe.log_exclude_manager.remove_success_log=§cEl bloque llamado '%1' se ha eliminado con éxito de §oBloques de Troncos Excluidos. +LumberAxe.log_exclude_manager.update_success_log=§eEl bloque llamado '%1' de §oBloques de Troncos Excluidos§r §ese ha actualizado con éxito a '%2' en §oBloques de Troncos Excluidos. +>>>>>>> itemUseOnUpdate diff --git a/RP/texts/es_MX.lang b/RP/texts/es_MX.lang index 0c5c614..b78247d 100644 --- a/RP/texts/es_MX.lang +++ b/RP/texts/es_MX.lang @@ -8,13 +8,82 @@ item.yn:netherite_lumber_axe.name=Hacha de madera inframundita ## Form-UI -LumberAxe.form.title.text=Información del árbol -LumberAxe.form.treeSizeAbrev.text=TIENE -LumberAxe.form.treeSizeAbrevLogs.text=roncos de árboles -LumberAxe.form.durabilityAbrev.text=DMG -LumberAxe.form.maxDurabilityAbrev.text=MAX -LumberAxe.form.canBeChopped.text=§ael árbol se puede cortar -LumberAxe.form.cannotBeChopped.text=§cel árbol no se puede cortar - -## Error Message -LumberAxe.watchdogError.hang.text=Error de secuencias de comandos: intente cortar o inspeccionar árboles más pequeños o en un ángulo diferente. \ No newline at end of file +LumberAxe.form.canBeChopped.text=ael árbol se puede cortar +LumberAxe.form.cannotBeChopped.text=cel árbol no se puede cortar + +<<<<<<< HEAD +LumberAxe.on_load_message=§a§lLumber Axe Addon§r §aha sido cargado con éxito. Creado por Adr-hyng. +LumberAxe.on_caught_invalid_command=§cFormato de uso incorrecto. %1 +LumberAxe.on_database_empty=§4No se encontró ningún registro de configuración en la base de datos.§r +LumberAxe.on_database_reset=§aLa base de datos ha sido reiniciada.§r +LumberAxe.show_database=ID de la base de datos: §e%1§r%2%3 +LumberAxe.on_caught_command_404=§cError al recuperar el comando %1: %2 +LumberAxe.on_caught_main_command_not_found=§cComando inválido: §l%1§r%2§cVerifica si el comando realmente existe. Usa /scriptevent %3 help + +LumberAxe.server.show_message_on_join=Mostrar mensaje al unirse +LumberAxe.server.durability_damage_per_block=Pérdida de durabilidad por bloque +LumberAxe.server.chop_limit=Límite de troncos de árbol + +LumberAxe.configuration.title=Configuración de %1 +LumberAxe.configuration.general=Opciones generales +LumberAxe.configuration.log_include_manager=Administrador de troncos incluidos +LumberAxe.configuration.log_exclude_manager=Administrador de troncos excluidos + +LumberAxe.log_include_manager.drop_down=Troncos cortables incluidos +LumberAxe.log_include_manager.text_field=Bloque de tronco a incluir +LumberAxe.log_include_manager.toggle=Insertar +LumberAxe.log_include_manager.add_success_log=§aEl bloque llamado '%1' se ha agregado exitosamente a §oTroncos cortables incluidos. +LumberAxe.log_include_manager.remove_success_log=§cEl bloque llamado '%1' se ha eliminado exitosamente de §oTroncos cortables incluidos. +LumberAxe.log_include_manager.update_success_log=§eEl bloque llamado '%1' de §oTroncos cortables incluidos§r §se ha actualizado exitosamente a '%2' en §oTroncos cortables incluidos. + +LumberAxe.log_exclude_manager.drop_down=Troncos cortables excluidos +LumberAxe.log_exclude_manager.text_field=Bloque de tronco a excluir +LumberAxe.log_exclude_manager.toggle=Insertar +LumberAxe.log_exclude_manager.add_success_log=§aEl bloque llamado '%1' se ha agregado exitosamente a §oTroncos cortables excluidos. +LumberAxe.log_exclude_manager.remove_success_log=§cEl bloque llamado '%1' se ha eliminado exitosamente de §oTroncos cortables excluidos. +LumberAxe.log_exclude_manager.update_success_log=§eEl bloque llamado '%1' de §oTroncos cortables excluidos§r §se ha actualizado exitosamente a '%2' en §oTroncos cortables excluidos. +======= +## Form-UI +LumberAxe.form.title.text=Vista General del Estado del Árbol +LumberAxe.form.treeSizeAbrev.text=Tiene +LumberAxe.form.treeSizeAbrevLogs.text=Tronco/s +LumberAxe.form.trunkHeightAbrev.text=Bloque/s de Altura + +## Log Messages +LumberAxe.on_load_message=§a§lLumber Axe Addon§r §ase ha cargado con éxito. Creado por Adr-hyng. +LumberAxe.on_caught_invalid_command=§cFormato de uso inválido. %1 +LumberAxe.on_database_empty=§4No se encontró registro de configuración en la base de datos.§r +LumberAxe.on_database_reset=§aLa base de datos ha sido reiniciada.§r +LumberAxe.show_database=ID de la base de datos: §e%1§r%2%3 +LumberAxe.on_caught_command_404=§cError al buscar el comando %1: %2 +LumberAxe.on_caught_main_command_not_found=§cComando Inválido: §l%1§r%2§cVerifica si el comando realmente existe. Usa /scriptevent %3 ayuda +LumberAxe.server.invalid_log_amount_limitation=No se puede talar todo el árbol debido a la limitación de configuración del servidor de %1. Pide al operador del servidor que lo ajuste. + +## Server Options Details +LumberAxe.server.durability_damage_per_block=Pérdida de durabilidad por bloque +LumberAxe.server.chop_limit=Límite de Troncos +LumberAxe.server.immersive_chopping=Activar Modo Inmersivo +LumberAxe.server.immersive_delay=Retraso del Modo Inmersivo + +## Configuration Options +LumberAxe.configuration.title=Configuración de %1 +LumberAxe.configuration.general=Opciones Generales +LumberAxe.configuration.log_include_manager=Administrador de Troncos Incluidos +LumberAxe.configuration.log_exclude_manager=Administrador de Troncos Excluidos + +## Include Manager +LumberAxe.log_include_manager.drop_down=Troncos Incluidos para Talar +LumberAxe.log_include_manager.text_field=Bloque de Tronco a Incluir +LumberAxe.log_include_manager.toggle=Añadir +LumberAxe.log_include_manager.add_success_log=§aEl bloque llamado '%1' se ha añadido con éxito a §oTroncos Incluidos para Talar. +LumberAxe.log_include_manager.remove_success_log=§cEl bloque llamado '%1' se ha eliminado con éxito de §oTroncos Incluidos para Talar. +LumberAxe.log_include_manager.update_success_log=§eEl bloque llamado '%1' de §oTroncos Incluidos para Talar§r §eha sido actualizado exitosamente a '%2' en §oTroncos Incluidos para Talar. + +## Exclude Manager +LumberAxe.log_exclude_manager.drop_down=Troncos Excluidos para Talar +LumberAxe.log_exclude_manager.text_field=Bloque de Tronco a Excluir +LumberAxe.log_exclude_manager.toggle=Añadir +LumberAxe.log_exclude_manager.add_success_log=§aEl bloque llamado '%1' se ha añadido con éxito a §oTroncos Excluidos para Talar. +LumberAxe.log_exclude_manager.remove_success_log=§cEl bloque llamado '%1' se ha eliminado con éxito de §oTroncos Excluidos para Talar. +LumberAxe.log_exclude_manager.update_success_log=§eEl bloque llamado '%1' de §oTroncos Excluidos para Talar§r §eha sido actualizado exitosamente a '%2' en §oTroncos Excluidos para Talar. +>>>>>>> itemUseOnUpdate diff --git a/RP/texts/fr_FR.lang b/RP/texts/fr_FR.lang index b802341..5edc63f 100644 --- a/RP/texts/fr_FR.lang +++ b/RP/texts/fr_FR.lang @@ -6,15 +6,83 @@ item.yn:iron_lumber_axe.name=Hache de bûcheron en fer item.yn:diamond_lumber_axe.name=Hache de bûcheron en diamant item.yn:netherite_lumber_axe.name=Hache de Netherite de bûcheron - ## Form-UI -LumberAxe.form.title.text=Informations sur l'arbre -LumberAxe.form.treeSizeAbrev.text=l'arbre a -LumberAxe.form.treeSizeAbrevLogs.text=bûches d'arbre -LumberAxe.form.durabilityAbrev.text=DMG -LumberAxe.form.maxDurabilityAbrev.text=MAX -LumberAxe.form.canBeChopped.text=§aL'arbre peut être coupé -LumberAxe.form.cannotBeChopped.text=§cL'arbre ne peut pas être coupé - -## Error Message -LumberAxe.watchdogError.hang.text=Erreur de script: essayez de couper ou d'inspecter des arbres plus petits ou sous un angle différent. \ No newline at end of file +LumberAxe.form.canBeChopped.text=L'arbre peut être coupé +LumberAxe.form.cannotBeChopped.text=L'arbre ne peut pas être coupé + +<<<<<<< HEAD +LumberAxe.on_load_message=§a§lLumber Axe Addon§r §aa été chargé avec succès. Créé par Adr-hyng. +LumberAxe.on_caught_invalid_command=§cFormat d'utilisation invalide. %1 +LumberAxe.on_database_empty=§4Aucun enregistrement de configuration trouvé dans la base de données.§r +LumberAxe.on_database_reset=§aLa base de données a été réinitialisée.§r +LumberAxe.show_database=ID de la base de données: §e%1§r%2%3 +LumberAxe.on_caught_command_404=§cErreur lors de la récupération de la commande %1 : %2 +LumberAxe.on_caught_main_command_not_found=§cCommande invalide : §l%1§r%2§cVérifiez si la commande existe réellement. Utilisez /scriptevent %3 help + +LumberAxe.server.show_message_on_join=Afficher le message à la connexion +LumberAxe.server.durability_damage_per_block=Perte de durabilité par bloc +LumberAxe.server.chop_limit=Limite de blocs de tronc d'arbre + +LumberAxe.configuration.title=Paramètres de %1 +LumberAxe.configuration.general=Options générales +LumberAxe.configuration.log_include_manager=Gestionnaire de troncs inclus +LumberAxe.configuration.log_exclude_manager=Gestionnaire de troncs exclus + +LumberAxe.log_include_manager.drop_down=Troncs coupables inclus +LumberAxe.log_include_manager.text_field=Bloc de tronc à inclure +LumberAxe.log_include_manager.toggle=Insérer +LumberAxe.log_include_manager.add_success_log=§aLe bloc nommé '%1' a été ajouté avec succès à §oTroncs coupables inclus. +LumberAxe.log_include_manager.remove_success_log=§cLe bloc nommé '%1' a été retiré avec succès de §oTroncs coupables inclus. +LumberAxe.log_include_manager.update_success_log=§eLe bloc nommé '%1' de §oTroncs coupables inclus§r §ea été mis à jour avec succès en '%2' dans §oTroncs coupables inclus. + +LumberAxe.log_exclude_manager.drop_down=Troncs coupables exclus +LumberAxe.log_exclude_manager.text_field=Bloc de tronc à exclure +LumberAxe.log_exclude_manager.toggle=Insérer +LumberAxe.log_exclude_manager.add_success_log=§aLe bloc nommé '%1' a été ajouté avec succès à §oTroncs coupables exclus. +LumberAxe.log_exclude_manager.remove_success_log=§cLe bloc nommé '%1' a été retiré avec succès de §oTroncs coupables exclus. +LumberAxe.log_exclude_manager.update_success_log=§eLe bloc nommé '%1' de §oTroncs coupables exclus§r §ea été mis à jour avec succès en '%2' dans §oTroncs coupables exclus. +======= +## Form-UI +LumberAxe.form.title.text=Vue d'ensemble de l'état de l'arbre +LumberAxe.form.treeSizeAbrev.text=A +LumberAxe.form.treeSizeAbrevLogs.text=Bûche/s +LumberAxe.form.trunkHeightAbrev.text=Bloc/s de Haut + +## Log Messages +LumberAxe.on_load_message=§a§lAddon Hache de Bûcheron§r §aété chargé avec succès. Créé par Adr-hyng. +LumberAxe.on_caught_invalid_command=§cFormat d'utilisation incorrect. %1 +LumberAxe.on_database_empty=§4Aucun enregistrement de configuration trouvé dans la base de données.§r +LumberAxe.on_database_reset=§aLa base de données a été réinitialisée.§r +LumberAxe.show_database=ID de la base de données : §e%1§r%2%3 +LumberAxe.on_caught_command_404=§cErreur lors de l'exécution de la commande %1 : %2 +LumberAxe.on_caught_main_command_not_found=§cCommande incorrecte : §l%1§r%2§cVérifiez si la commande existe vraiment. Utilisez /scriptevent %3 help +LumberAxe.server.invalid_log_amount_limitation=Impossible d'abattre tout l'arbre en raison de la limitation de configuration du serveur à %1. Demandez à l'opérateur du serveur d'augmenter/ajuster. + +## Server Options Details +LumberAxe.server.durability_damage_per_block=Perte de durabilité par bloc +LumberAxe.server.chop_limit=Limitation des bûches de l'arbre +LumberAxe.server.immersive_chopping=Activer le Mode Immersif +LumberAxe.server.immersive_delay=Délai du Mode Immersif + +## Configuration Options +LumberAxe.configuration.title=Paramètres de %1 +LumberAxe.configuration.general=Options Générales +LumberAxe.configuration.log_include_manager=Gestionnaire de Bûches Incluses +LumberAxe.configuration.log_exclude_manager=Gestionnaire de Bûches Exclues + +## Include Manager +LumberAxe.log_include_manager.drop_down=Bûches Incluses pour la Coupe +LumberAxe.log_include_manager.text_field=Bûche à Inclure +LumberAxe.log_include_manager.toggle=Ajouter +LumberAxe.log_include_manager.add_success_log=§aBloc nommé '%1' ajouté avec succès à §oBûches Incluses pour la Coupe. +LumberAxe.log_include_manager.remove_success_log=§cBloc nommé '%1' supprimé avec succès de §oBûches Incluses pour la Coupe. +LumberAxe.log_include_manager.update_success_log=§eBloc nommé '%1' de §oBûches Incluses pour la Coupe§r §ea été mis à jour avec succès à '%2' dans §oBûches Incluses pour la Coupe. + +## Exclude Manager +LumberAxe.log_exclude_manager.drop_down=Bûches Exclues pour la Coupe +LumberAxe.log_exclude_manager.text_field=Bûche à Exclure +LumberAxe.log_exclude_manager.toggle=Ajouter +LumberAxe.log_exclude_manager.add_success_log=§aBloc nommé '%1' ajouté avec succès à §oBûches Exclues pour la Coupe. +LumberAxe.log_exclude_manager.remove_success_log=§cBloc nommé '%1' supprimé avec succès de §oBûches Exclues pour la Coupe. +LumberAxe.log_exclude_manager.update_success_log=§eBloc nommé '%1' de §oBûches Exclues pour la Coupe§r §ea été mis à jour avec succès à '%2' dans §oBûches Exclues pour la Coupe. +>>>>>>> itemUseOnUpdate diff --git a/RP/texts/ja_JP.lang b/RP/texts/ja_JP.lang new file mode 100644 index 0000000..5e79d47 --- /dev/null +++ b/RP/texts/ja_JP.lang @@ -0,0 +1,53 @@ +## Items +item.yn:wooden_lumber_axe.name=木製の木こりの斧 ### +item.yn:stone_lumber_axe.name=石製の木こりの斧 ### +item.yn:golden_lumber_axe.name=金製の木こりの斧 ### +item.yn:iron_lumber_axe.name=鉄製の木こりの斧 ### +item.yn:diamond_lumber_axe.name=ダイヤモンドの木こりの斧 ### +item.yn:netherite_lumber_axe.name=ネザライトの木こりの斧 ### + +## Form-UI +LumberAxe.form.title.text=木の状態確認 ### +LumberAxe.form.treeSizeAbrev.text=含む ### +LumberAxe.form.treeSizeAbrevLogs.text=ログ ### +LumberAxe.form.trunkHeightAbrev.text=ブロックの高さ ### +LumberAxe.form.canBeChopped.text=切り倒せる ### +LumberAxe.form.cannotBeChopped.text=切り倒せない ### + +## Log Messages +LumberAxe.on_load_message=§a§l木こりの斧アドオン§r §aが正常に読み込まれました。 作成者:Adr-hyng。 ### +LumberAxe.on_caught_invalid_command=§c無効な使用フォーマット。%1 ### +LumberAxe.on_database_empty=§4データベースに構成レコードが見つかりませんでした。§r ### +LumberAxe.on_database_reset=§aデータベースがリセットされました。§r ### +LumberAxe.show_database=データベースID:§e%1§r%2%3 ### +LumberAxe.on_caught_command_404=§c%1 コマンドを取得中にエラーが発生しました: %2 ### +LumberAxe.on_caught_main_command_not_found=§c無効なコマンド: §l%1§r%2§cコマンドが存在するか確認してください。/scriptevent %3 helpを使用 ### +LumberAxe.server.invalid_log_amount_limitation=サーバーの設定制限 %1 により木を全部切り倒すことができません。 サーバーオペレーターに調整を依頼してください。 ### + +## Server Options Details +LumberAxe.server.durability_damage_per_block=ブロックごとの耐久度減少 ### +LumberAxe.server.chop_limit=木のログの制限 ### +LumberAxe.server.immersive_chopping=没入モードを有効化 ### +LumberAxe.server.immersive_delay=没入モード遅延時間 ### + +## Configuration Options +LumberAxe.configuration.title=%1 設定 ### +LumberAxe.configuration.general=一般オプション ### +LumberAxe.configuration.log_include_manager=含まれるログマネージャー ### +LumberAxe.configuration.log_exclude_manager=除外されたログマネージャー ### + +## Include Manager +LumberAxe.log_include_manager.drop_down=含まれる切り倒せるログ ### +LumberAxe.log_include_manager.text_field=含めるログブロック ### +LumberAxe.log_include_manager.toggle=追加 ### +LumberAxe.log_include_manager.add_success_log=§aブロック名 '%1' が正常に §o含まれる切り倒せるブロック に追加されました。 ### +LumberAxe.log_include_manager.remove_success_log=§cブロック名 '%1' が正常に §o含まれる切り倒せるブロック から削除されました。 ### +LumberAxe.log_include_manager.update_success_log=§eブロック名 '%1' が §o含まれる切り倒せるブロック§r に正常に '%2' に更新されました。 ### + +## Exclude Manager +LumberAxe.log_exclude_manager.drop_down=除外された切り倒せるログ ### +LumberAxe.log_exclude_manager.text_field=除外するログブロック ### +LumberAxe.log_exclude_manager.toggle=追加 ### +LumberAxe.log_exclude_manager.add_success_log=§aブロック名 '%1' が正常に §o除外された切り倒せるブロック に追加されました。 ### +LumberAxe.log_exclude_manager.remove_success_log=§cブロック名 '%1' が正常に §o除外された切り倒せるブロック から削除されました。 ### +LumberAxe.log_exclude_manager.update_success_log=§eブロック名 '%1' が §o除外された切り倒せるブロック§r に正常に '%2' に更新されました。 ### diff --git a/RP/texts/languages.json b/RP/texts/languages.json index 8e5ec9a..6a82466 100644 --- a/RP/texts/languages.json +++ b/RP/texts/languages.json @@ -1,9 +1,15 @@ [ "en_US", + "en_GB", "es_ES", "es_MX", "pt_BR", + "pt_PT", "ru_RU", - "fr_FR" + "fr_FR", + "ja_JP", + "zh_CN", + "zh_TW", + "id_ID" ] \ No newline at end of file diff --git a/RP/texts/pt_BR.lang b/RP/texts/pt_BR.lang index 3423c1a..39e4c63 100644 --- a/RP/texts/pt_BR.lang +++ b/RP/texts/pt_BR.lang @@ -8,13 +8,81 @@ item.yn:netherite_lumber_axe.name=machado de lenhador netherita ## Form-UI -LumberAxe.form.title.text=Informação da árvore -LumberAxe.form.treeSizeAbrev.text=TEM -LumberAxe.form.treeSizeAbrevLogs.text=toras de árvore -LumberAxe.form.durabilityAbrev.text=DMG -LumberAxe.form.maxDurabilityAbrev.text=MAX -LumberAxe.form.canBeChopped.text=§aÁrvore pode ser cortada -LumberAxe.form.cannotBeChopped.text=§cÁrvore não pode ser cortada - -## Error Message -LumberAxe.watchdogError.hang.text=Erro de script: tente cortar ou inspecionar árvores pequenas ou ângulos diferentes. \ No newline at end of file +LumberAxe.form.canBeChopped.text=Árvore pode ser cortada +LumberAxe.form.cannotBeChopped.text=Árvore não pode ser cortada +## Form-UI +LumberAxe.form.title.text=Visão Geral do Status da Árvore +LumberAxe.form.treeSizeAbrev.text=Tem +LumberAxe.form.treeSizeAbrevLogs.text=Tora/s +LumberAxe.form.trunkHeightAbrev.text=Bloco/s de Altura + +<<<<<<< HEAD +LumberAxe.on_load_message=§a§lLumber Axe Addon§r §afoi carregado com sucesso. Criado por Adr-hyng. +LumberAxe.on_caught_invalid_command=§cFormato de uso inválido. %1 +LumberAxe.on_database_empty=§4Nenhum registro de configuração encontrado no banco de dados.§r +LumberAxe.on_database_reset=§aO banco de dados foi reiniciado.§r +LumberAxe.show_database=ID do banco de dados: §e%1§r%2%3 +LumberAxe.on_caught_command_404=§cErro ao buscar o comando %1: %2 +LumberAxe.on_caught_main_command_not_found=§cComando inválido: §l%1§r%2§cVerifique se o comando realmente existe. Use /scriptevent %3 help + +LumberAxe.server.show_message_on_join=Mostrar mensagem ao entrar +LumberAxe.server.durability_damage_per_block=Perda de durabilidade por bloco +LumberAxe.server.chop_limit=Limite de troncos de árvore + +LumberAxe.configuration.title=Configurações de %1 +LumberAxe.configuration.general=Opções gerais +LumberAxe.configuration.log_include_manager=Gerenciador de troncos incluídos +LumberAxe.configuration.log_exclude_manager=Gerenciador de troncos excluídos + +LumberAxe.log_include_manager.drop_down=Troncos incluídos cortáveis +LumberAxe.log_include_manager.text_field=Bloco de tronco para incluir +LumberAxe.log_include_manager.toggle=Inserir +LumberAxe.log_include_manager.add_success_log=§aO bloco chamado '%1' foi adicionado com sucesso a §oTroncos incluídos cortáveis. +LumberAxe.log_include_manager.remove_success_log=§cO bloco chamado '%1' foi removido com sucesso de §oTroncos incluídos cortáveis. +LumberAxe.log_include_manager.update_success_log=§eO bloco chamado '%1' de §oTroncos incluídos cortáveis§r §efoi atualizado com sucesso para '%2' em §oTroncos incluídos cortáveis. + +LumberAxe.log_exclude_manager.drop_down=Troncos excluídos cortáveis +LumberAxe.log_exclude_manager.text_field=Bloco de tronco para excluir +LumberAxe.log_exclude_manager.toggle=Inserir +LumberAxe.log_exclude_manager.add_success_log=§aO bloco chamado '%1' foi adicionado com sucesso a §oTroncos excluídos cortáveis. +LumberAxe.log_exclude_manager.remove_success_log=§cO bloco chamado '%1' foi removido com sucesso de §oTroncos excluídos cortáveis. +LumberAxe.log_exclude_manager.update_success_log=§eO bloco chamado '%1' de §oTroncos excluídos cortáveis§r §efoi atualizado com sucesso para '%2' em §oTroncos excluídos cortáveis. +======= +## Log Messages +LumberAxe.on_load_message=§a§lAddon Machado de Lenhador§r §afoi carregado com sucesso. Criado por Adr-hyng. +LumberAxe.on_caught_invalid_command=§cFormato de Uso Inválido. %1 +LumberAxe.on_database_empty=§4Nenhum registro de configuração encontrado no banco de dados.§r +LumberAxe.on_database_reset=§aO banco de dados foi redefinido.§r +LumberAxe.show_database=ID do Banco de Dados: §e%1§r%2%3 +LumberAxe.on_caught_command_404=§cErro ao buscar o comando %1: %2 +LumberAxe.on_caught_main_command_not_found=§cComando Inválido: §l%1§r%2§cVerifique se o comando realmente existe. Use /scriptevent %3 help +LumberAxe.server.invalid_log_amount_limitation=Não é possível cortar toda a árvore devido à limitação de configuração do servidor de %1. Solicite ao operador do servidor para aumentar/ajustar. + +## Server Options Details +LumberAxe.server.durability_damage_per_block=Perda de durabilidade por bloco +LumberAxe.server.chop_limit=Limitação de Toras da Árvore +LumberAxe.server.immersive_chopping=Ativar Modo Imersivo +LumberAxe.server.immersive_delay=Tempo de Atraso do Modo Imersivo + +## Configuration Options +LumberAxe.configuration.title=Configurações de %1 +LumberAxe.configuration.general=Opções Gerais +LumberAxe.configuration.log_include_manager=Gerenciador de Toras Incluídas +LumberAxe.configuration.log_exclude_manager=Gerenciador de Toras Excluídas + +## Include Manager +LumberAxe.log_include_manager.drop_down=Toras Incluídas para Corte +LumberAxe.log_include_manager.text_field=Tora a Incluir +LumberAxe.log_include_manager.toggle=Adicionar +LumberAxe.log_include_manager.add_success_log=§aBloco com o nome '%1' foi adicionado com sucesso a §oToras Incluídas para Corte. +LumberAxe.log_include_manager.remove_success_log=§cBloco com o nome '%1' foi removido com sucesso de §oToras Incluídas para Corte. +LumberAxe.log_include_manager.update_success_log=§eBloco com o nome '%1' de §oToras Incluídas para Corte§r §efoi atualizado com sucesso para '%2' em §oToras Incluídas para Corte. + +## Exclude Manager +LumberAxe.log_exclude_manager.drop_down=Toras Excluídas para Corte +LumberAxe.log_exclude_manager.text_field=Tora a Excluir +LumberAxe.log_exclude_manager.toggle=Adicionar +LumberAxe.log_exclude_manager.add_success_log=§aBloco com o nome '%1' foi adicionado com sucesso a §oToras Excluídas para Corte. +LumberAxe.log_exclude_manager.remove_success_log=§cBloco com o nome '%1' foi removido com sucesso de §oToras Excluídas para Corte. +LumberAxe.log_exclude_manager.update_success_log=§eBloco com o nome '%1' de §oToras Excluídas para Corte§r §efoi atualizado com sucesso para '%2' em §oToras Excluídas para Corte. +>>>>>>> itemUseOnUpdate diff --git a/RP/texts/pt_PT.lang b/RP/texts/pt_PT.lang new file mode 100644 index 0000000..9f48b36 --- /dev/null +++ b/RP/texts/pt_PT.lang @@ -0,0 +1,51 @@ +## Items +item.yn:wooden_lumber_axe.name=Machado de Lenhador de Madeira ### +item.yn:stone_lumber_axe.name=Machado de Lenhador de Pedra ### +item.yn:golden_lumber_axe.name=Machado de Lenhador de Ouro ### +item.yn:iron_lumber_axe.name=Machado de Lenhador de Ferro ### +item.yn:diamond_lumber_axe.name=Machado de Lenhador de Diamante ### +item.yn:netherite_lumber_axe.name=Machado de Lenhador de Netherite ### + +## Form-UI +LumberAxe.form.title.text=Visão Geral do Estado da Árvore ### +LumberAxe.form.treeSizeAbrev.text=Contém ### +LumberAxe.form.treeSizeAbrevLogs.text=Troncos ### +LumberAxe.form.trunkHeightAbrev.text=Altura em Blocos ### +LumberAxe.form.canBeChopped.text=Pode Ser Cortada ### +LumberAxe.form.cannotBeChopped.text=Não Pode Ser Cortada ### + +## Log Messages +LumberAxe.on_load_message=§a§lAdd-on de Machado de Lenhador§r §aCarregado com sucesso. Criado por Adr-hyng. ### +LumberAxe.on_caught_invalid_command=§cFormato de uso inválido. %1 ### +LumberAxe.on_database_empty=§4Nenhum registo de configuração encontrado na base de dados. §r ### +LumberAxe.on_database_reset=§aA base de dados foi reiniciada. §r ### +LumberAxe.show_database=ID da Base de Dados: §e%1§r%2%3 ### +LumberAxe.on_caught_command_404=§cErro ao obter comando %1: %2 ### +LumberAxe.on_caught_main_command_not_found=§cComando inválido: §l%1§r%2§cVerifique se o comando existe. Utilize /scriptevent %3 help ### +LumberAxe.server.invalid_log_amount_limitation=Não é possível cortar a árvore inteira devido à limitação de configuração do servidor %1. Solicite ao operador do servidor que ajuste. ### + +## Server Options Details +LumberAxe.server.durability_damage_per_block=Dano de durabilidade por bloco ### +LumberAxe.server.chop_limit=Limite de troncos da árvore ### +LumberAxe.server.immersive_chopping=Ativar modo imersivo ### +LumberAxe.server.immersive_delay=Atraso do modo imersivo ### + +## Configuration Options +LumberAxe.configuration.title=Configurações de %1 ### +LumberAxe.configuration.general=Opções Gerais ### +LumberAxe.configuration.log_include_manager=Gestor de Troncos Incluídos ### +LumberAxe.configuration.log_exclude_manager=Gestor de Troncos Excluídos ### + +## Include Manager +LumberAxe.log_include_manager.drop_down=Troncos Incluídos no Corte ### +LumberAxe.log_include_manager.text_field=Bloco de Tronco a Incluir ### +LumberAxe.log_include_manager.toggle=Adicionar ### +LumberAxe.log_include_manager.add_success_log=§aBloco '%1' adicionado com sucesso a §oTroncos Incluídos no Corte. ### +LumberAxe.log_include_manager.remove_success_log=§cBloco '%1' removido com sucesso de §oTroncos Incluídos no Corte. ### +LumberAxe.log_include_manager.update_success_log=§eBloco '%1' atualizado com sucesso para '%2' em §oTroncos Incluídos no Corte§r. ### + +## Exclude Manager +LumberAxe.log_exclude_manager.drop_down=Troncos Excluídos no Corte ### +LumberAxe.log_exclude_manager.text_field=Bloco de Tronco a Excluir ### +LumberAxe.log_exclude_manager.toggle=Adicionar ### +LumberAxe.log_exclude_manager.add_success_log=§aBloco '%1' adicionado com sucesso a §oTroncos Excluídos no Corte. ### diff --git a/RP/texts/ru_RU.lang b/RP/texts/ru_RU.lang index 25ad193..277ce34 100644 --- a/RP/texts/ru_RU.lang +++ b/RP/texts/ru_RU.lang @@ -8,13 +8,82 @@ item.yn:netherite_lumber_axe.name=Незеритовый топор дровос ## Form-UI -LumberAxe.form.title.text=Информация о дереве -LumberAxe.form.treeSizeAbrev.text=имеет -LumberAxe.form.treeSizeAbrevLogs.text=бревен дерева -LumberAxe.form.durabilityAbrev.text=DMG -LumberAxe.form.maxDurabilityAbrev.text=MAX -LumberAxe.form.canBeChopped.text=§aДерево можно срубить -LumberAxe.form.cannotBeChopped.text=§cДерево нельзя срубить - -## Error Message -LumberAxe.watchdogError.hang.text=Ошибка сценария: попробуйте рубить или осматривать деревья меньшего размера или под другим углом. +LumberAxe.form.canBeChopped.text=Дерево можно срубить +LumberAxe.form.cannotBeChopped.text=Дерево нельзя срубить + +<<<<<<< HEAD +LumberAxe.on_load_message=§a§lLumber Axe Addon§r §aуспешно загружен. Создатель: Adr-hyng. +LumberAxe.on_caught_invalid_command=§cНеверный формат использования. %1 +LumberAxe.on_database_empty=§4В базе данных не найдено ни одной записи конфигурации.§r +LumberAxe.on_database_reset=§aБаза данных была сброшена.§r +LumberAxe.show_database=ID базы данных: §e%1§r%2%3 +LumberAxe.on_caught_command_404=§cОшибка при получении команды %1: %2 +LumberAxe.on_caught_main_command_not_found=§cНеверная команда: §l%1§r%2§cПроверьте, существует ли команда. Используйте /scriptevent %3 help + +LumberAxe.server.show_message_on_join=Показывать сообщение при входе +LumberAxe.server.durability_damage_per_block=Потеря прочности за блок +LumberAxe.server.chop_limit=Лимит блоков дерева + +LumberAxe.configuration.title=Настройки %1 +LumberAxe.configuration.general=Общие параметры +LumberAxe.configuration.log_include_manager=Менеджер включённых брёвен +LumberAxe.configuration.log_exclude_manager=Менеджер исключённых брёвен + +LumberAxe.log_include_manager.drop_down=Включённые рубимые брёвна +LumberAxe.log_include_manager.text_field=Блок брёвен для включения +LumberAxe.log_include_manager.toggle=Добавить +LumberAxe.log_include_manager.add_success_log=§aБлок под названием '%1' успешно добавлен в §oВключённые рубимые блоки. +LumberAxe.log_include_manager.remove_success_log=§cБлок под названием '%1' успешно удалён из §oВключённых рубимых блоков. +LumberAxe.log_include_manager.update_success_log=§eБлок под названием '%1' из §oВключённых рубимых блоков§r §eуспешно обновлён на '%2' в §oВключённых рубимых блоках. + +LumberAxe.log_exclude_manager.drop_down=Исключённые рубимые брёвна +LumberAxe.log_exclude_manager.text_field=Блок брёвен для исключения +LumberAxe.log_exclude_manager.toggle=Добавить +LumberAxe.log_exclude_manager.add_success_log=§aБлок под названием '%1' успешно добавлен в §oИсключённые рубимые блоки. +LumberAxe.log_exclude_manager.remove_success_log=§cБлок под названием '%1' успешно удалён из §oИсключённых рубимых блоков. +LumberAxe.log_exclude_manager.update_success_log=§eБлок под названием '%1' из §oИсключённых рубимых блоков§r §eуспешно обновлён на '%2' в §oИсключённых рубимых блоках. +======= +## Form-UI +LumberAxe.form.title.text=Обзор состояния дерева +LumberAxe.form.treeSizeAbrev.text=Имеет +LumberAxe.form.treeSizeAbrevLogs.text=Бревно/а +LumberAxe.form.trunkHeightAbrev.text=Блок/ов Высота + +## Log Messages +LumberAxe.on_load_message=§a§lАддон Топор Лесоруба§r §aуспешно загружен. Создано Adr-hyng. +LumberAxe.on_caught_invalid_command=§cНеверный формат использования. %1 +LumberAxe.on_database_empty=§4В базе данных не найдено записей конфигурации.§r +LumberAxe.on_database_reset=§aБаза данных была сброшена.§r +LumberAxe.show_database=ID базы данных: §e%1§r%2%3 +LumberAxe.on_caught_command_404=§cОшибка при выполнении команды %1: %2 +LumberAxe.on_caught_main_command_not_found=§cНеверная команда: §l%1§r%2§cПроверьте, существует ли команда. Используйте /scriptevent %3 help +LumberAxe.server.invalid_log_amount_limitation=Невозможно срубить всё дерево из-за ограничения конфигурации сервера в %1. Обратитесь к администратору для увеличения/настройки. + +## Server Options Details +LumberAxe.server.durability_damage_per_block=Потеря прочности за блок +LumberAxe.server.chop_limit=Ограничение количества бревен дерева +LumberAxe.server.immersive_chopping=Включить Иммерсивный Режим +LumberAxe.server.immersive_delay=Время Задержки Иммерсивного Режима + +## Configuration Options +LumberAxe.configuration.title=Настройки %1 +LumberAxe.configuration.general=Основные Настройки +LumberAxe.configuration.log_include_manager=Менеджер Включенных Бревен +LumberAxe.configuration.log_exclude_manager=Менеджер Исключенных Бревен + +## Include Manager +LumberAxe.log_include_manager.drop_down=Включенные Бревна для Рубки +LumberAxe.log_include_manager.text_field=Бревно для Включения +LumberAxe.log_include_manager.toggle=Добавить +LumberAxe.log_include_manager.add_success_log=§aБлок с именем '%1' успешно добавлен в §oВключенные Бревна для Рубки. +LumberAxe.log_include_manager.remove_success_log=§cБлок с именем '%1' успешно удален из §oВключенные Бревна для Рубки. +LumberAxe.log_include_manager.update_success_log=§eБлок с именем '%1' из §oВключенные Бревна для Рубки§r §eбыл успешно обновлен на '%2' в §oВключенные Бревна для Рубки. + +## Exclude Manager +LumberAxe.log_exclude_manager.drop_down=Исключенные Бревна для Рубки +LumberAxe.log_exclude_manager.text_field=Бревно для Исключения +LumberAxe.log_exclude_manager.toggle=Добавить +LumberAxe.log_exclude_manager.add_success_log=§aБлок с именем '%1' успешно добавлен в §oИсключенные Бревна для Рубки. +LumberAxe.log_exclude_manager.remove_success_log=§cБлок с именем '%1' успешно удален из §oИсключенные Бревна для Рубки. +LumberAxe.log_exclude_manager.update_success_log=§eБлок с именем '%1' из §oИсключенные Бревна для Рубки§r §eбыл успешно обновлен на '%2' в §oИсключенные Бревна для Рубки. +>>>>>>> itemUseOnUpdate diff --git a/RP/texts/zh_CN.lang b/RP/texts/zh_CN.lang new file mode 100644 index 0000000..6dde9b2 --- /dev/null +++ b/RP/texts/zh_CN.lang @@ -0,0 +1,53 @@ +## Items +item.yn:wooden_lumber_axe.name=木制伐木斧 ### +item.yn:stone_lumber_axe.name=石制伐木斧 ### +item.yn:golden_lumber_axe.name=金制伐木斧 ### +item.yn:iron_lumber_axe.name=铁制伐木斧 ### +item.yn:diamond_lumber_axe.name=钻石伐木斧 ### +item.yn:netherite_lumber_axe.name=下界合金伐木斧 ### + +## Form-UI +LumberAxe.form.title.text=树木状态概览 ### +LumberAxe.form.treeSizeAbrev.text=含有 ### +LumberAxe.form.treeSizeAbrevLogs.text=原木 ### +LumberAxe.form.trunkHeightAbrev.text=方块高 ### +LumberAxe.form.canBeChopped.text=可砍伐 ### +LumberAxe.form.cannotBeChopped.text=不可砍伐 ### + +## Log Messages +LumberAxe.on_load_message=§a§l伐木斧模组§r §a已成功加载。由 Adr-hyng 创建。 ### +LumberAxe.on_caught_invalid_command=§c无效的使用格式。%1 ### +LumberAxe.on_database_empty=§4数据库中未找到配置记录。§r ### +LumberAxe.on_database_reset=§a数据库已重置。§r ### +LumberAxe.show_database=数据库 ID:§e%1§r%2%3 ### +LumberAxe.on_caught_command_404=§c获取 %1 命令时出错: %2 ### +LumberAxe.on_caught_main_command_not_found=§c无效命令:§l%1§r%2§c请检查命令是否存在。使用 /scriptevent %3 help ### +LumberAxe.server.invalid_log_amount_limitation=由于服务器配置限制为 %1,无法砍伐整棵树。请向服务器管理员请求调整。 ### + +## Server Options Details +LumberAxe.server.durability_damage_per_block=每个方块的耐久损耗 ### +LumberAxe.server.chop_limit=树木原木限制 ### +LumberAxe.server.immersive_chopping=启用沉浸模式 ### +LumberAxe.server.immersive_delay=沉浸模式延迟时间 ### + +## Configuration Options +LumberAxe.configuration.title=%1 设置 ### +LumberAxe.configuration.general=常规选项 ### +LumberAxe.configuration.log_include_manager=包含的原木管理器 ### +LumberAxe.configuration.log_exclude_manager=排除的原木管理器 ### + +## Include Manager +LumberAxe.log_include_manager.drop_down=包含的可砍伐原木 ### +LumberAxe.log_include_manager.text_field=要包含的原木方块 ### +LumberAxe.log_include_manager.toggle=添加 ### +LumberAxe.log_include_manager.add_success_log=§a名为 '%1' 的方块已成功添加到 §o包含的可砍伐原木 中。 ### +LumberAxe.log_include_manager.remove_success_log=§c名为 '%1' 的方块已成功从 §o包含的可砍伐原木 中移除。 ### +LumberAxe.log_include_manager.update_success_log=§e名为 '%1' 的方块已从 §o包含的可砍伐原木§r §e成功更新为 '%2'。 ### + +## Exclude Manager +LumberAxe.log_exclude_manager.drop_down=排除的可砍伐原木 ### +LumberAxe.log_exclude_manager.text_field=要排除的原木方块 ### +LumberAxe.log_exclude_manager.toggle=添加 ### +LumberAxe.log_exclude_manager.add_success_log=§a名为 '%1' 的方块已成功添加到 §o排除的可砍伐原木 中。 ### +LumberAxe.log_exclude_manager.remove_success_log=§c名为 '%1' 的方块已成功从 §o排除的可砍伐原木 中移除。 ### +LumberAxe.log_exclude_manager.update_success_log=§e名为 '%1' 的方块已从 §o排除的可砍伐原木§r §e成功更新为 '%2'。 ### diff --git a/RP/texts/zh_TW.lang b/RP/texts/zh_TW.lang new file mode 100644 index 0000000..788c780 --- /dev/null +++ b/RP/texts/zh_TW.lang @@ -0,0 +1,53 @@ +## Items +item.yn:wooden_lumber_axe.name=木製伐木斧 ### +item.yn:stone_lumber_axe.name=石製伐木斧 ### +item.yn:golden_lumber_axe.name=金製伐木斧 ### +item.yn:iron_lumber_axe.name=鐵製伐木斧 ### +item.yn:diamond_lumber_axe.name=鑽石伐木斧 ### +item.yn:netherite_lumber_axe.name=下界合金伐木斧 ### + +## Form-UI +LumberAxe.form.title.text=樹木狀態概覽 ### +LumberAxe.form.treeSizeAbrev.text=包含 ### +LumberAxe.form.treeSizeAbrevLogs.text=原木 ### +LumberAxe.form.trunkHeightAbrev.text=方塊高度 ### +LumberAxe.form.canBeChopped.text=可砍伐 ### +LumberAxe.form.cannotBeChopped.text=不可砍伐 ### + +## Log Messages +LumberAxe.on_load_message=§a§l伐木斧模組§r §a已成功加載。由 Adr-hyng 創建。 ### +LumberAxe.on_caught_invalid_command=§c無效的使用格式。%1 ### +LumberAxe.on_database_empty=§4資料庫中未找到配置記錄。§r ### +LumberAxe.on_database_reset=§a資料庫已重置。§r ### +LumberAxe.show_database=資料庫 ID:§e%1§r%2%3 ### +LumberAxe.on_caught_command_404=§c獲取 %1 命令時出錯: %2 ### +LumberAxe.on_caught_main_command_not_found=§c無效命令:§l%1§r%2§c請檢查命令是否存在。使用 /scriptevent %3 help ### +LumberAxe.server.invalid_log_amount_limitation=由於伺服器設置限制為 %1,無法砍伐整棵樹。請聯繫伺服器管理員進行調整。 ### + +## Server Options Details +LumberAxe.server.durability_damage_per_block=每方塊耐久損耗 ### +LumberAxe.server.chop_limit=樹木原木限制 ### +LumberAxe.server.immersive_chopping=啟用沉浸模式 ### +LumberAxe.server.immersive_delay=沉浸模式延遲時間 ### + +## Configuration Options +LumberAxe.configuration.title=%1 設置 ### +LumberAxe.configuration.general=一般選項 ### +LumberAxe.configuration.log_include_manager=包含的原木管理器 ### +LumberAxe.configuration.log_exclude_manager=排除的原木管理器 ### + +## Include Manager +LumberAxe.log_include_manager.drop_down=包含的可砍伐原木 ### +LumberAxe.log_include_manager.text_field=要包含的原木方塊 ### +LumberAxe.log_include_manager.toggle=添加 ### +LumberAxe.log_include_manager.add_success_log=§a方塊名 '%1' 已成功添加到 §o包含的可砍伐原木 中。 ### +LumberAxe.log_include_manager.remove_success_log=§c方塊名 '%1' 已成功從 §o包含的可砍伐原木 中移除。 ### +LumberAxe.log_include_manager.update_success_log=§e方塊名 '%1' 已從 §o包含的可砍伐原木§r §e成功更新為 '%2'。 ### + +## Exclude Manager +LumberAxe.log_exclude_manager.drop_down=排除的可砍伐原木 ### +LumberAxe.log_exclude_manager.text_field=要排除的原木方塊 ### +LumberAxe.log_exclude_manager.toggle=添加 ### +LumberAxe.log_exclude_manager.add_success_log=§a方塊名 '%1' 已成功添加到 §o排除的可砍伐原木 中。 ### +LumberAxe.log_exclude_manager.remove_success_log=§c方塊名 '%1' 已成功從 §o排除的可砍伐原木 中移除。 ### +LumberAxe.log_exclude_manager.update_success_log=§e方塊名 '%1' 已從 §o排除的可砍伐原木§r §e成功更新為 '%2'。 ### diff --git a/RP/textures/InfoUI/axe_durability.png b/RP/textures/InfoUI/axe_durability.png deleted file mode 100644 index 2f808f0..0000000 Binary files a/RP/textures/InfoUI/axe_durability.png and /dev/null differ diff --git a/RP/textures/InfoUI/blocks.png b/RP/textures/InfoUI/blocks.png deleted file mode 100644 index 56d24f0..0000000 Binary files a/RP/textures/InfoUI/blocks.png and /dev/null differ diff --git a/RP/textures/InfoUI/canBeCut.png b/RP/textures/InfoUI/canBeCut.png deleted file mode 100644 index a6b5666..0000000 Binary files a/RP/textures/InfoUI/canBeCut.png and /dev/null differ diff --git a/RP/textures/InfoUI/can_be_cut.png b/RP/textures/InfoUI/can_be_cut.png new file mode 100644 index 0000000..8c320f4 Binary files /dev/null and b/RP/textures/InfoUI/can_be_cut.png differ diff --git a/RP/textures/InfoUI/cannot_be_cut.png b/RP/textures/InfoUI/cannot_be_cut.png new file mode 100644 index 0000000..3963018 Binary files /dev/null and b/RP/textures/InfoUI/cannot_be_cut.png differ diff --git a/RP/textures/InfoUI/lumber_deficit.png b/RP/textures/InfoUI/lumber_deficit.png new file mode 100644 index 0000000..e94e8bf Binary files /dev/null and b/RP/textures/InfoUI/lumber_deficit.png differ diff --git a/RP/textures/InfoUI/lumber_surplus.png b/RP/textures/InfoUI/lumber_surplus.png new file mode 100644 index 0000000..386bfe8 Binary files /dev/null and b/RP/textures/InfoUI/lumber_surplus.png differ diff --git a/RP/textures/InfoUI/required_durability.png b/RP/textures/InfoUI/required_durability.png deleted file mode 100644 index 8f6c045..0000000 Binary files a/RP/textures/InfoUI/required_durability.png and /dev/null differ diff --git a/RP/textures/InfoUI/total_lumber.png b/RP/textures/InfoUI/total_lumber.png new file mode 100644 index 0000000..ab11284 Binary files /dev/null and b/RP/textures/InfoUI/total_lumber.png differ diff --git a/RP/textures/InfoUI/tree_height.png b/RP/textures/InfoUI/tree_height.png new file mode 100644 index 0000000..62694b9 Binary files /dev/null and b/RP/textures/InfoUI/tree_height.png differ diff --git a/RP/textures/entity/example.png b/RP/textures/entity/example.png new file mode 100644 index 0000000..6f84eea Binary files /dev/null and b/RP/textures/entity/example.png differ diff --git a/RP/textures/items/Lumber Axes/diamond_lumber_axe.png b/RP/textures/items/Lumber Axes/diamond_lumber_axe.png index e15bcbf..34a47b9 100644 Binary files a/RP/textures/items/Lumber Axes/diamond_lumber_axe.png and b/RP/textures/items/Lumber Axes/diamond_lumber_axe.png differ diff --git a/RP/textures/items/Lumber Axes/golden_lumber_axe.png b/RP/textures/items/Lumber Axes/golden_lumber_axe.png index cc36928..15db1ac 100644 Binary files a/RP/textures/items/Lumber Axes/golden_lumber_axe.png and b/RP/textures/items/Lumber Axes/golden_lumber_axe.png differ diff --git a/RP/textures/items/Lumber Axes/iron_lumber_axe.png b/RP/textures/items/Lumber Axes/iron_lumber_axe.png index 8723861..de26b90 100644 Binary files a/RP/textures/items/Lumber Axes/iron_lumber_axe.png and b/RP/textures/items/Lumber Axes/iron_lumber_axe.png differ diff --git a/RP/textures/items/Lumber Axes/netherite_lumber_axe.png b/RP/textures/items/Lumber Axes/netherite_lumber_axe.png index 0d61f37..2575ee4 100644 Binary files a/RP/textures/items/Lumber Axes/netherite_lumber_axe.png and b/RP/textures/items/Lumber Axes/netherite_lumber_axe.png differ diff --git a/RP/textures/items/Lumber Axes/stone_lumber_axe.png b/RP/textures/items/Lumber Axes/stone_lumber_axe.png index 056c612..b69383e 100644 Binary files a/RP/textures/items/Lumber Axes/stone_lumber_axe.png and b/RP/textures/items/Lumber Axes/stone_lumber_axe.png differ diff --git a/RP/textures/items/Lumber Axes/wooden_lumber_axe.png b/RP/textures/items/Lumber Axes/wooden_lumber_axe.png index 6f7e587..acb8bf4 100644 Binary files a/RP/textures/items/Lumber Axes/wooden_lumber_axe.png and b/RP/textures/items/Lumber Axes/wooden_lumber_axe.png differ diff --git a/RP/textures/particles/dust.png b/RP/textures/particles/dust.png new file mode 100644 index 0000000..5e16ec0 Binary files /dev/null and b/RP/textures/particles/dust.png differ diff --git a/RP/textures/particles/indicator.png b/RP/textures/particles/indicator.png new file mode 100644 index 0000000..255ab54 Binary files /dev/null and b/RP/textures/particles/indicator.png differ diff --git a/guidelines/item-recipes/diamond_lumber_axe.png b/guidelines/item-recipes/diamond_lumber_axe.png index c229b59..dfef23b 100644 Binary files a/guidelines/item-recipes/diamond_lumber_axe.png and b/guidelines/item-recipes/diamond_lumber_axe.png differ diff --git a/guidelines/item-recipes/golden_lumber_axe.png b/guidelines/item-recipes/golden_lumber_axe.png index 1973dfe..a0faaf1 100644 Binary files a/guidelines/item-recipes/golden_lumber_axe.png and b/guidelines/item-recipes/golden_lumber_axe.png differ diff --git a/guidelines/item-recipes/iron_lumber_axe.png b/guidelines/item-recipes/iron_lumber_axe.png index 391ddc7..7cebefc 100644 Binary files a/guidelines/item-recipes/iron_lumber_axe.png and b/guidelines/item-recipes/iron_lumber_axe.png differ diff --git a/guidelines/item-recipes/netherite_lumber_axe.png b/guidelines/item-recipes/netherite_lumber_axe.png index 01fe631..29ea42c 100644 Binary files a/guidelines/item-recipes/netherite_lumber_axe.png and b/guidelines/item-recipes/netherite_lumber_axe.png differ diff --git a/guidelines/item-recipes/smithing_recipe.png b/guidelines/item-recipes/smithing_recipe.png index 90d974e..98224fc 100644 Binary files a/guidelines/item-recipes/smithing_recipe.png and b/guidelines/item-recipes/smithing_recipe.png differ diff --git a/guidelines/item-recipes/stone_lumber_axe.png b/guidelines/item-recipes/stone_lumber_axe.png index 77ecadd..92f7694 100644 Binary files a/guidelines/item-recipes/stone_lumber_axe.png and b/guidelines/item-recipes/stone_lumber_axe.png differ diff --git a/guidelines/item-recipes/wooden_lumber_axe.png b/guidelines/item-recipes/wooden_lumber_axe.png index 30255d4..ad3a7fd 100644 Binary files a/guidelines/item-recipes/wooden_lumber_axe.png and b/guidelines/item-recipes/wooden_lumber_axe.png differ diff --git a/guidelines/thumbnails/author.bbmodel b/guidelines/thumbnails/author.bbmodel new file mode 100644 index 0000000..b1f4c29 --- /dev/null +++ b/guidelines/thumbnails/author.bbmodel @@ -0,0 +1 @@ +{"meta":{"format_version":"4.10","model_format":"minecraft_title","box_uv":false},"name":"author","visible_box":[1,1,0],"variable_placeholders":"","variable_placeholder_buttons":[],"unhandled_root_fields":{},"resolution":{"width":1000,"height":320},"elements":[{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[84,2,-11],"to":[112,42,11],"autouv":0,"color":4,"origin":[0,0,0],"faces":{"north":{"uv":[0,22,28,62],"texture":0},"east":{"uv":[0,0,0,0],"texture":null},"south":{"uv":[28,22,0,62],"texture":0},"west":{"uv":[0,0,0,0],"texture":null},"up":{"uv":[28,22,0,0],"texture":0},"down":{"uv":[28,84,0,62],"texture":0}},"type":"cube","uuid":"4b0f7ec6-8260-1002-5fcc-81a4666217f1"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[114,44,13],"to":[82,0,-13],"autouv":0,"color":6,"origin":[0,0,0],"faces":{"north":{"uv":[0,266,1,267],"texture":0},"east":{"uv":[0,266,1,267],"texture":0},"south":{"uv":[0,266,1,267],"texture":0},"west":{"uv":[0,266,1,267],"texture":0},"up":{"uv":[0,266,1,267],"texture":0},"down":{"uv":[0,266,1,267],"texture":0}},"type":"cube","uuid":"5a2a15c2-7f14-c195-fcea-0ec472daf6cf"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[52,2,-11],"to":[80,42,11],"autouv":0,"color":8,"origin":[0,0,0],"faces":{"north":{"uv":[120,22,148,62],"texture":0},"east":{"uv":[0,0,0,0],"texture":null},"south":{"uv":[148,22,120,62],"texture":0},"west":{"uv":[0,0,0,0],"texture":null},"up":{"uv":[148,22,120,0],"texture":0},"down":{"uv":[148,84,120,62],"texture":0}},"type":"cube","uuid":"5f9c8cf0-ebc8-8863-18c6-7809d7b158a5"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[54,40,13],"to":[50,4,-13],"autouv":0,"color":2,"origin":[0,0,0],"faces":{"north":{"uv":[0,266,1,267],"texture":0},"east":{"uv":[0,266,1,267],"texture":0},"south":{"uv":[0,266,1,267],"texture":0},"west":{"uv":[0,0,0,0],"texture":null},"up":{"uv":[0,266,1,267],"texture":0},"down":{"uv":[0,266,1,267],"texture":0}},"type":"cube","uuid":"120bb2f0-f22c-7f38-a757-a618f69bbb30"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[82,44,13],"to":[54,0,-13],"autouv":0,"color":6,"origin":[0,0,0],"faces":{"north":{"uv":[0,266,1,267],"texture":0},"east":{"uv":[2,266,3,277],"texture":0},"south":{"uv":[0,266,1,267],"texture":0},"west":{"uv":[0,266,1,267],"texture":0},"up":{"uv":[0,266,1,267],"texture":0},"down":{"uv":[0,266,1,267],"texture":0}},"type":"cube","uuid":"79937187-1919-2346-3d02-8f1c723d2f76"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[20,2,-11],"to":[48,42,11],"autouv":0,"color":9,"origin":[0,0,0],"faces":{"north":{"uv":[550,22,578,62],"texture":0},"east":{"uv":[0,0,0,0],"texture":null},"south":{"uv":[578,22,550,62],"texture":0},"west":{"uv":[0,0,0,0],"texture":null},"up":{"uv":[578,22,550,0],"texture":0},"down":{"uv":[578,84,550,62],"texture":0}},"type":"cube","uuid":"cbe40e78-37ba-2b5a-c46c-5c2da2daf878"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[50,44,13],"to":[18,0,-13],"autouv":0,"color":0,"origin":[0,0,0],"faces":{"north":{"uv":[0,266,1,267],"texture":0},"east":{"uv":[0,266,1,267],"texture":0},"south":{"uv":[0,266,1,267],"texture":0},"west":{"uv":[0,266,1,267],"texture":0},"up":{"uv":[0,266,1,267],"texture":0},"down":{"uv":[0,266,1,267],"texture":0}},"type":"cube","uuid":"b67b90ed-1167-c509-16eb-e409641eebaf"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-12,2,-11],"to":[16,42,11],"autouv":0,"color":2,"origin":[0,0,0],"faces":{"north":{"uv":[240,22,268,62],"texture":0},"east":{"uv":[0,0,0,0],"texture":null},"south":{"uv":[268,22,240,62],"texture":0},"west":{"uv":[0,0,0,0],"texture":null},"up":{"uv":[268,22,240,0],"texture":0},"down":{"uv":[268,84,240,62],"texture":0}},"type":"cube","uuid":"40811b00-943c-64f3-63b7-77620d986be5"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[18,44,13],"to":[-14,0,-13],"autouv":0,"color":7,"origin":[0,0,0],"faces":{"north":{"uv":[0,266,1,267],"texture":0},"east":{"uv":[0,266,1,267],"texture":0},"south":{"uv":[0,266,1,267],"texture":0},"west":{"uv":[0,266,1,267],"texture":0},"up":{"uv":[0,266,1,267],"texture":0},"down":{"uv":[0,266,1,267],"texture":0}},"type":"cube","uuid":"bc37ee41-0580-2185-95e0-e7257ca0fcef"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-44,2,-11],"to":[-16,42,11],"autouv":0,"color":8,"origin":[0,0,0],"faces":{"north":{"uv":[782,22,810,62],"texture":0},"east":{"uv":[0,0,0,0],"texture":null},"south":{"uv":[810,22,782,62],"texture":0},"west":{"uv":[0,0,0,0],"texture":null},"up":{"uv":[810,22,782,0],"texture":0},"down":{"uv":[810,84,782,62],"texture":0}},"type":"cube","uuid":"d36e6959-c28c-1787-b305-695c9e707dd2"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-46,0,-13],"to":[-38,20,13],"autouv":0,"color":4,"origin":[0,0,0],"faces":{"north":{"uv":[0,0,0,0],"texture":null},"east":{"uv":[0,266,1,267],"texture":0},"south":{"uv":[0,0,0,0],"texture":null},"west":{"uv":[0,0,0,0],"texture":null},"up":{"uv":[0,266,1,267],"texture":0},"down":{"uv":[0,0,0,0],"texture":null}},"type":"cube","uuid":"15bc8bc2-69cd-0dc2-c2a4-01d382bcee2d"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-22,0,-13],"to":[-14,20,13],"autouv":0,"color":1,"origin":[0,0,0],"faces":{"north":{"uv":[0,0,0,0],"texture":null},"east":{"uv":[0,0,0,0],"texture":null},"south":{"uv":[0,0,0,0],"texture":null},"west":{"uv":[0,266,1,267],"texture":0},"up":{"uv":[0,266,1,267],"texture":0},"down":{"uv":[0,0,0,0],"texture":null}},"type":"cube","uuid":"e314fe8f-26e2-3749-37f1-3a90b349d837"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-14,44,13],"to":[-46,0,-13],"autouv":0,"color":5,"origin":[0,0,0],"faces":{"north":{"uv":[14,288,18,277],"texture":0},"east":{"uv":[14,288,15,277],"texture":0},"south":{"uv":[14,288,18,277],"texture":0},"west":{"uv":[14,288,15,277],"texture":0},"up":{"uv":[14,287,18,288],"texture":0},"down":{"uv":[0,266,1,267],"texture":0}},"type":"cube","uuid":"26ef6960-69a2-80e9-386e-7511649dd4ae"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-80,2,-11],"to":[-48,42,11],"autouv":0,"color":0,"origin":[0,0,0],"faces":{"north":{"uv":[422,22,454,62],"texture":0},"east":{"uv":[0,0,0,0],"texture":null},"south":{"uv":[454,22,422,62],"texture":0},"west":{"uv":[0,0,0,0],"texture":null},"up":{"uv":[454,22,422,0],"texture":0},"down":{"uv":[454,84,422,62],"texture":0}},"type":"cube","uuid":"cfc8c94c-0f39-e14f-c12f-e5b7407f2ff0"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-66,36,-13],"to":[-62,44,13],"autouv":0,"color":8,"origin":[0,0,0],"faces":{"north":{"uv":[0,0,0,0],"texture":null},"east":{"uv":[0,266,1,267],"texture":0},"south":{"uv":[0,0,0,0],"texture":null},"west":{"uv":[0,266,1,267],"texture":0},"up":{"uv":[0,0,0,0],"texture":null},"down":{"uv":[0,266,1,267],"texture":0}},"type":"cube","uuid":"562b8346-c5e5-44c9-8394-3426e0220808"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-66,0,-13],"to":[-62,10,13],"autouv":0,"color":7,"origin":[0,0,0],"faces":{"north":{"uv":[0,0,0,0],"texture":null},"east":{"uv":[0,266,1,267],"texture":0},"south":{"uv":[0,0,0,0],"texture":null},"west":{"uv":[0,266,1,267],"texture":0},"up":{"uv":[0,266,1,267],"texture":0},"down":{"uv":[0,0,0,0],"texture":null}},"type":"cube","uuid":"7f95eafb-b38c-0ab8-c592-b197fab282ec"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-46,44,13],"to":[-82,0,-13],"autouv":0,"color":2,"origin":[0,0,0],"faces":{"north":{"uv":[19,288,28,266],"texture":0},"east":{"uv":[0,266,1,267],"texture":0},"south":{"uv":[19,288,28,266],"texture":0},"west":{"uv":[0,266,1,267],"texture":0},"up":{"uv":[19,266,28,267],"texture":0},"down":{"uv":[19,266,28,267],"texture":0}},"type":"cube","uuid":"349a54ef-1c8f-5dcb-bb16-b042837b2578"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-112,2,-11],"to":[-84,42,11],"autouv":0,"color":2,"origin":[0,0,0],"faces":{"north":{"uv":[210,22,238,62],"texture":0},"east":{"uv":[0,0,0,0],"texture":null},"south":{"uv":[238,22,210,62],"texture":0},"west":{"uv":[0,0,0,0],"texture":null},"up":{"uv":[238,22,210,0],"texture":0},"down":{"uv":[238,84,210,62],"texture":0}},"type":"cube","uuid":"32ea36f4-690e-8bb0-2754-cfdc3d19f762"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-114,28,-13],"to":[-98,32,13],"autouv":0,"color":1,"origin":[0,0,0],"faces":{"north":{"uv":[0,0,0,0],"texture":null},"east":{"uv":[0,266,1,267],"texture":0},"south":{"uv":[0,0,0,0],"texture":null},"west":{"uv":[0,0,0,0],"texture":null},"up":{"uv":[0,266,1,267],"texture":0},"down":{"uv":[0,266,1,267],"texture":0}},"type":"cube","uuid":"3a8ed0bc-d6f2-76db-2700-3266574d102b"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-82,44,13],"to":[-114,0,-13],"autouv":0,"color":2,"origin":[0,0,0],"faces":{"north":{"uv":[6,277,8,266],"texture":0},"east":{"uv":[7,277,6,266],"texture":0},"south":{"uv":[8,277,6,266],"texture":0},"west":{"uv":[0,266,1,267],"texture":0},"up":{"uv":[0,266,1,267],"texture":0},"down":{"uv":[0,266,1,267],"texture":0}},"type":"cube","uuid":"af8ef236-a13b-d9c9-5e96-bbe66ad4bc4b"}],"outliner":[{"name":"adrhyng","origin":[0,0,0],"color":0,"uuid":"dc477901-0804-6da1-d979-caa9de236537","export":true,"mirror_uv":false,"isOpen":false,"locked":false,"visibility":true,"autouv":0,"children":[{"name":"a","origin":[0,0,0],"color":0,"uuid":"ac40c266-8910-8e74-63a3-ce2ae53cae59","export":true,"mirror_uv":false,"isOpen":false,"locked":false,"visibility":true,"autouv":0,"children":["4b0f7ec6-8260-1002-5fcc-81a4666217f1","5a2a15c2-7f14-c195-fcea-0ec472daf6cf"]},{"name":"d","origin":[0,0,0],"color":0,"uuid":"ac818b49-53c8-6e82-cb7c-71c47d6bff6e","export":true,"mirror_uv":false,"isOpen":false,"locked":false,"visibility":true,"autouv":0,"children":["5f9c8cf0-ebc8-8863-18c6-7809d7b158a5","120bb2f0-f22c-7f38-a757-a618f69bbb30","79937187-1919-2346-3d02-8f1c723d2f76"]},{"name":"r","origin":[0,0,0],"color":0,"uuid":"f06f58a6-8d9f-6c7c-436f-3026e101a97f","export":true,"mirror_uv":false,"isOpen":false,"locked":false,"visibility":true,"autouv":0,"children":["cbe40e78-37ba-2b5a-c46c-5c2da2daf878","b67b90ed-1167-c509-16eb-e409641eebaf"]},{"name":"h","origin":[0,0,0],"color":0,"uuid":"3c4ca18a-a8a0-3c6d-a31f-d38ad5f32077","export":true,"mirror_uv":false,"isOpen":false,"locked":false,"visibility":true,"autouv":0,"children":["40811b00-943c-64f3-63b7-77620d986be5","bc37ee41-0580-2185-95e0-e7257ca0fcef"]},{"name":"y","origin":[0,0,0],"color":0,"uuid":"b7017296-5285-5860-5b3f-6db130592273","export":true,"mirror_uv":false,"isOpen":false,"locked":false,"visibility":true,"autouv":0,"children":["d36e6959-c28c-1787-b305-695c9e707dd2","15bc8bc2-69cd-0dc2-c2a4-01d382bcee2d","e314fe8f-26e2-3749-37f1-3a90b349d837","26ef6960-69a2-80e9-386e-7511649dd4ae"]},{"name":"n","origin":[0,0,0],"color":0,"uuid":"65f05170-9548-1ff7-f6a2-3065c8c19cd2","export":true,"mirror_uv":false,"isOpen":false,"locked":false,"visibility":true,"autouv":0,"children":["cfc8c94c-0f39-e14f-c12f-e5b7407f2ff0","562b8346-c5e5-44c9-8394-3426e0220808","7f95eafb-b38c-0ab8-c592-b197fab282ec","349a54ef-1c8f-5dcb-bb16-b042837b2578"]},{"name":"g","origin":[0,0,0],"color":0,"uuid":"1cf31f3e-c520-6f55-6591-37c2d26f47d9","export":true,"mirror_uv":false,"isOpen":false,"locked":false,"visibility":true,"autouv":0,"children":["32ea36f4-690e-8bb0-2754-cfdc3d19f762","3a8ed0bc-d6f2-76db-2700-3266574d102b","af8ef236-a13b-d9c9-5e96-bbe66ad4bc4b"]}]}],"textures":[{"path":"","name":"adrhyng.png","folder":"","namespace":"","id":"0","width":4000,"height":1280,"uv_width":1000,"uv_height":320,"particle":false,"use_as_default":false,"layers_enabled":false,"sync_to_project":"","render_mode":"default","render_sides":"front","frame_time":1,"frame_order_type":"loop","frame_order":"","frame_interpolate":false,"visible":true,"internal":true,"saved":false,"uuid":"cc4702e6-4940-10bb-ab84-037c257b5510","source":""}]} \ No newline at end of file diff --git a/guidelines/thumbnails/new_thumbnail.aseprite b/guidelines/thumbnails/new_thumbnail.aseprite new file mode 100644 index 0000000..1cab8c5 Binary files /dev/null and b/guidelines/thumbnails/new_thumbnail.aseprite differ diff --git a/guidelines/thumbnails/new_thumbnail.png b/guidelines/thumbnails/new_thumbnail.png new file mode 100644 index 0000000..df11f4b Binary files /dev/null and b/guidelines/thumbnails/new_thumbnail.png differ diff --git a/guidelines/thumbnails/new_thumbnail_1x1.png b/guidelines/thumbnails/new_thumbnail_1x1.png new file mode 100644 index 0000000..884c739 Binary files /dev/null and b/guidelines/thumbnails/new_thumbnail_1x1.png differ diff --git a/guidelines/thumbnails/pack_icon.aseprite b/guidelines/thumbnails/pack_icon.aseprite new file mode 100644 index 0000000..d13cde6 Binary files /dev/null and b/guidelines/thumbnails/pack_icon.aseprite differ diff --git a/guidelines/thumbnails/title_part1.bbmodel b/guidelines/thumbnails/title_part1.bbmodel new file mode 100644 index 0000000..0b29a15 --- /dev/null +++ b/guidelines/thumbnails/title_part1.bbmodel @@ -0,0 +1 @@ +{"meta":{"format_version":"4.10","model_format":"minecraft_title","box_uv":false},"name":"title_part1","visible_box":[1,1,0],"variable_placeholders":"","variable_placeholder_buttons":[],"unhandled_root_fields":{},"resolution":{"width":1000,"height":320},"elements":[{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[74,2,-11],"to":[98,42,11],"autouv":0,"color":6,"origin":[0,0,0],"faces":{"north":{"uv":[344,22,368,62],"texture":0},"east":{"uv":[0,0,0,0],"texture":null},"south":{"uv":[368,22,344,62],"texture":0},"west":{"uv":[0,0,0,0],"texture":null},"up":{"uv":[368,22,344,0],"texture":0},"down":{"uv":[368,84,344,62],"texture":0}},"type":"cube","uuid":"0b188439-c6bd-8995-01a7-605729892913"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[72,12,-13],"to":[84,44,13],"autouv":0,"color":1,"origin":[0,0,0],"faces":{"north":{"uv":[0,0,0,0],"texture":null},"east":{"uv":[0,266,1,267],"texture":0},"south":{"uv":[0,0,0,0],"texture":null},"west":{"uv":[0,0,0,0],"texture":null},"up":{"uv":[0,0,0,0],"texture":null},"down":{"uv":[0,266,1,267],"texture":0}},"type":"cube","uuid":"b1de3e70-83d3-5ac7-3911-df7f9dd0656f"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[100,44,13],"to":[72,0,-13],"autouv":0,"color":6,"origin":[0,0,0],"faces":{"north":{"uv":[9,277,16,266],"texture":0},"east":{"uv":[9,277,10,266],"texture":0},"south":{"uv":[16,277,9,266],"texture":0},"west":{"uv":[0,266,1,267],"texture":0},"up":{"uv":[0,266,1,267],"texture":0},"down":{"uv":[16,266,9,267],"texture":0}},"type":"cube","uuid":"31eb9125-d73d-ac61-1a77-a17f838b63e8"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[42,2,-11],"to":[70,42,11],"autouv":0,"color":0,"origin":[0,0,0],"faces":{"north":{"uv":[640,22,668,62],"texture":0},"east":{"uv":[0,0,0,0],"texture":null},"south":{"uv":[668,22,640,62],"texture":0},"west":{"uv":[0,0,0,0],"texture":null},"up":{"uv":[668,22,640,0],"texture":0},"down":{"uv":[668,84,640,62],"texture":0}},"type":"cube","uuid":"4032eef4-14d6-8c06-637f-c0283d5f78f8"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[72,44,13],"to":[40,0,-13],"autouv":0,"color":1,"origin":[0,0,0],"faces":{"north":{"uv":[0,266,1,267],"texture":0},"east":{"uv":[0,266,1,267],"texture":0},"south":{"uv":[0,266,1,267],"texture":0},"west":{"uv":[0,266,1,267],"texture":0},"up":{"uv":[0,266,1,267],"texture":0},"down":{"uv":[0,266,1,267],"texture":0}},"type":"cube","uuid":"b5e89e8b-9f40-8c95-acb4-e37c73b00de1"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-2,2,-11],"to":[38,42,11],"autouv":0,"color":3,"origin":[0,0,0],"faces":{"north":{"uv":[370,22,410,62],"texture":0},"east":{"uv":[2,268,3,266],"texture":0},"south":{"uv":[410,22,370,62],"texture":0},"west":{"uv":[2,268,3,266],"texture":0},"up":{"uv":[410,22,370,0],"texture":0},"down":{"uv":[410,84,370,62],"texture":0}},"type":"cube","uuid":"a7803ba6-8778-0fa2-6cc5-7b6aa55f88a1"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[14,10,-11],"to":[22,10,11],"autouv":0,"color":7,"origin":[0,0,0],"faces":{"north":{"uv":[0,0,0,0],"texture":null},"east":{"uv":[0,0,0,0],"texture":null},"south":{"uv":[0,0,0,0],"texture":null},"west":{"uv":[0,0,0,0],"texture":null},"up":{"uv":[0,0,0,0],"texture":null},"down":{"uv":[420,84,412,62],"texture":0}},"type":"cube","uuid":"0ac6e37d-0192-1338-787d-bf8d9a773803"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[12,36,-13],"to":[24,44,13],"autouv":0,"color":6,"origin":[0,0,0],"faces":{"north":{"uv":[0,0,0,0],"texture":null},"east":{"uv":[0,266,1,267],"texture":0},"south":{"uv":[0,0,0,0],"texture":null},"west":{"uv":[0,266,1,267],"texture":0},"up":{"uv":[0,0,0,0],"texture":null},"down":{"uv":[1,267,4,268],"texture":0}},"type":"cube","uuid":"a6db6ea2-fe8b-d440-6d68-4b14ce8fbfce"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[16,28,-13],"to":[20,36,13],"autouv":0,"color":2,"origin":[0,0,0],"faces":{"north":{"uv":[0,0,0,0],"texture":null},"east":{"uv":[0,266,1,267],"texture":0},"south":{"uv":[0,0,0,0],"texture":null},"west":{"uv":[0,266,1,267],"texture":0},"up":{"uv":[0,0,0,0],"texture":null},"down":{"uv":[0,266,1,267],"texture":0}},"type":"cube","uuid":"cf3fedf8-aa23-d412-6fb2-439aa0c3b69b"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[12,0,-13],"to":[24,8,13],"autouv":0,"color":3,"origin":[0,0,0],"faces":{"north":{"uv":[0,0,0,0],"texture":null},"east":{"uv":[0,266,1,267],"texture":0},"south":{"uv":[0,0,0,0],"texture":null},"west":{"uv":[0,266,1,267],"texture":0},"up":{"uv":[0,266,1,267],"texture":0},"down":{"uv":[0,0,0,0],"texture":null}},"type":"cube","uuid":"01d09160-692b-616b-98e6-bf1b063939a9"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[40,44,13],"to":[-4,0,-13],"autouv":0,"color":2,"origin":[0,0,0],"faces":{"north":{"uv":[12,277,23,266],"texture":0},"east":{"uv":[0,266,1,267],"texture":0},"south":{"uv":[12,277,23,266],"texture":0},"west":{"uv":[0,266,1,267],"texture":0},"up":{"uv":[12,266,23,267],"texture":0},"down":{"uv":[12,266,23,267],"texture":0}},"type":"cube","uuid":"1bf910fa-ffa2-22aa-9b89-55a0e7733ca7"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-34,2,-11],"to":[-6,42,11],"autouv":0,"color":9,"origin":[0,0,0],"faces":{"north":{"uv":[60,22,88,62],"texture":0},"east":{"uv":[0,0,0,0],"texture":null},"south":{"uv":[88,22,60,62],"texture":0},"west":{"uv":[0,0,0,0],"texture":null},"up":{"uv":[88,22,60,0],"texture":0},"down":{"uv":[88,84,60,62],"texture":0}},"type":"cube","uuid":"0137f6a9-e3c0-80a4-e511-cccaeca9edb6"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-4,44,13],"to":[-36,0,-13],"autouv":0,"color":1,"origin":[0,0,0],"faces":{"north":{"uv":[0,266,1,267],"texture":0},"east":{"uv":[0,266,1,267],"texture":0},"south":{"uv":[0,266,1,267],"texture":0},"west":{"uv":[0,266,1,267],"texture":0},"up":{"uv":[0,266,1,267],"texture":0},"down":{"uv":[0,266,1,267],"texture":0}},"type":"cube","uuid":"013809c7-42e7-77b9-73e1-3e9dbd7a339f"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-66,2,-11],"to":[-38,42,11],"autouv":0,"color":4,"origin":[0,0,0],"faces":{"north":{"uv":[150,22,178,62],"texture":0},"east":{"uv":[0,0,0,0],"texture":null},"south":{"uv":[178,22,150,62],"texture":0},"west":{"uv":[0,0,0,0],"texture":null},"up":{"uv":[178,22,150,0],"texture":0},"down":{"uv":[178,84,150,62],"texture":0}},"type":"cube","uuid":"e9a852a4-29a3-4068-9f43-f0e92a8f7ef5"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-68,28,-13],"to":[-52,32,13],"autouv":0,"color":8,"origin":[0,0,0],"faces":{"north":{"uv":[0,0,0,0],"texture":null},"east":{"uv":[0,266,1,267],"texture":0},"south":{"uv":[0,0,0,0],"texture":null},"west":{"uv":[0,0,0,0],"texture":null},"up":{"uv":[0,266,1,267],"texture":0},"down":{"uv":[0,266,1,267],"texture":0}},"type":"cube","uuid":"412c849d-1e73-9ef1-a671-596be4fd99fc"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-68,12,-13],"to":[-52,16,13],"autouv":0,"color":0,"origin":[0,0,0],"faces":{"north":{"uv":[0,0,0,0],"texture":null},"east":{"uv":[0,266,1,267],"texture":0},"south":{"uv":[0,0,0,0],"texture":null},"west":{"uv":[0,0,0,0],"texture":null},"up":{"uv":[0,266,1,267],"texture":0},"down":{"uv":[0,266,1,267],"texture":0}},"type":"cube","uuid":"ad49ab74-cb23-79bc-e819-88ec8b29a8d0"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-36,44,13],"to":[-68,0,-13],"autouv":0,"color":5,"origin":[0,0,0],"faces":{"north":{"uv":[3,266,5,277],"texture":0},"east":{"uv":[4,266,3,277],"texture":0},"south":{"uv":[5,266,3,277],"texture":0},"west":{"uv":[0,266,1,267],"texture":0},"up":{"uv":[0,266,1,267],"texture":0},"down":{"uv":[0,266,1,267],"texture":0}},"type":"cube","uuid":"83e2c1f3-3e97-2831-eb16-53d95dcc3561"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-98,2,-11],"to":[-70,42,11],"autouv":0,"color":0,"origin":[0,0,0],"faces":{"north":{"uv":[550,22,578,62],"texture":0},"east":{"uv":[0,0,0,0],"texture":null},"south":{"uv":[578,22,550,62],"texture":0},"west":{"uv":[0,0,0,0],"texture":null},"up":{"uv":[578,22,550,0],"texture":0},"down":{"uv":[578,84,550,62],"texture":0}},"type":"cube","uuid":"d77e343d-f1dd-b715-e655-70c32b3ae059"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-68,44,13],"to":[-100,0,-13],"autouv":0,"color":4,"origin":[0,0,0],"faces":{"north":{"uv":[0,266,1,267],"texture":0},"east":{"uv":[0,266,1,267],"texture":0},"south":{"uv":[0,266,1,267],"texture":0},"west":{"uv":[0,266,1,267],"texture":0},"up":{"uv":[0,266,1,267],"texture":0},"down":{"uv":[0,266,1,267],"texture":0}},"type":"cube","uuid":"5667a5d0-edb9-2bc2-2064-900aab5079d5"}],"outliner":[{"name":"lumber","origin":[0,0,0],"color":0,"uuid":"cf7710ca-0c31-21fb-081e-0aeec8aab109","export":true,"mirror_uv":false,"isOpen":false,"locked":false,"visibility":true,"autouv":0,"children":[{"name":"l","origin":[0,0,0],"color":0,"uuid":"801c5550-454c-06ef-ccf5-f8cb08046353","export":true,"mirror_uv":false,"isOpen":false,"locked":false,"visibility":true,"autouv":0,"children":["0b188439-c6bd-8995-01a7-605729892913","b1de3e70-83d3-5ac7-3911-df7f9dd0656f","31eb9125-d73d-ac61-1a77-a17f838b63e8"]},{"name":"u","origin":[0,0,0],"color":0,"uuid":"f8f83b88-7fa8-0e43-fc03-f7361e7f0af5","export":true,"mirror_uv":false,"isOpen":false,"locked":false,"visibility":true,"autouv":0,"children":["4032eef4-14d6-8c06-637f-c0283d5f78f8","b5e89e8b-9f40-8c95-acb4-e37c73b00de1"]},{"name":"m","origin":[0,0,0],"color":0,"uuid":"a32defc5-0c3a-a9a0-a878-c5e5edb4b43a","export":true,"mirror_uv":false,"isOpen":false,"locked":false,"visibility":true,"autouv":0,"children":["a7803ba6-8778-0fa2-6cc5-7b6aa55f88a1","0ac6e37d-0192-1338-787d-bf8d9a773803","a6db6ea2-fe8b-d440-6d68-4b14ce8fbfce","cf3fedf8-aa23-d412-6fb2-439aa0c3b69b","01d09160-692b-616b-98e6-bf1b063939a9","1bf910fa-ffa2-22aa-9b89-55a0e7733ca7"]},{"name":"b","origin":[0,0,0],"color":0,"uuid":"3db69e22-1fc5-3e94-e3d5-a9c59cf7d527","export":true,"mirror_uv":false,"isOpen":false,"locked":false,"visibility":true,"autouv":0,"children":["0137f6a9-e3c0-80a4-e511-cccaeca9edb6","013809c7-42e7-77b9-73e1-3e9dbd7a339f"]},{"name":"e","origin":[0,0,0],"color":0,"uuid":"07ea002f-1a9f-0b86-17b2-5ef11cd080ab","export":true,"mirror_uv":false,"isOpen":false,"locked":false,"visibility":true,"autouv":0,"children":["e9a852a4-29a3-4068-9f43-f0e92a8f7ef5","412c849d-1e73-9ef1-a671-596be4fd99fc","ad49ab74-cb23-79bc-e819-88ec8b29a8d0","83e2c1f3-3e97-2831-eb16-53d95dcc3561"]},{"name":"r","origin":[0,0,0],"color":0,"uuid":"e53b4af6-ab7f-d791-2fb8-f6061a704a47","export":true,"mirror_uv":false,"isOpen":false,"locked":false,"visibility":true,"autouv":0,"children":["d77e343d-f1dd-b715-e655-70c32b3ae059","5667a5d0-edb9-2bc2-2064-900aab5079d5"]}]}],"textures":[{"path":"","name":"lumber.png","folder":"","namespace":"","id":"0","width":4000,"height":1280,"uv_width":1000,"uv_height":320,"particle":false,"use_as_default":false,"layers_enabled":false,"sync_to_project":"","render_mode":"default","render_sides":"front","frame_time":1,"frame_order_type":"loop","frame_order":"","frame_interpolate":false,"visible":true,"internal":true,"saved":false,"uuid":"8dbdaced-4081-4860-946a-920285aef6f8","source":""}]} \ No newline at end of file diff --git a/guidelines/thumbnails/title_part1.png b/guidelines/thumbnails/title_part1.png new file mode 100644 index 0000000..88f5f42 Binary files /dev/null and b/guidelines/thumbnails/title_part1.png differ diff --git a/guidelines/thumbnails/title_part2.bbmodel b/guidelines/thumbnails/title_part2.bbmodel new file mode 100644 index 0000000..e62ca99 --- /dev/null +++ b/guidelines/thumbnails/title_part2.bbmodel @@ -0,0 +1 @@ +{"meta":{"format_version":"4.10","model_format":"minecraft_title","box_uv":false},"name":"title_part2","visible_box":[1,1,0],"variable_placeholders":"","variable_placeholder_buttons":[],"unhandled_root_fields":{},"resolution":{"width":1000,"height":320},"elements":[{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[18,2,-11],"to":[46,42,11],"autouv":0,"color":8,"origin":[0,0,0],"faces":{"north":{"uv":[0,22,28,62],"texture":0},"east":{"uv":[0,0,0,0],"texture":null},"south":{"uv":[28,22,0,62],"texture":0},"west":{"uv":[0,0,0,0],"texture":null},"up":{"uv":[28,22,0,0],"texture":0},"down":{"uv":[28,84,0,62],"texture":0}},"type":"cube","uuid":"a19027f8-0948-b856-d7a5-a5d95d641531"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[48,44,13],"to":[16,0,-13],"autouv":0,"color":9,"origin":[0,0,0],"faces":{"north":{"uv":[0,266,1,267],"texture":0},"east":{"uv":[0,266,1,267],"texture":0},"south":{"uv":[0,266,1,267],"texture":0},"west":{"uv":[0,266,1,267],"texture":0},"up":{"uv":[0,266,1,267],"texture":0},"down":{"uv":[0,266,1,267],"texture":0}},"type":"cube","uuid":"a12bd1d9-1ff4-db18-7b2b-5225cf43e664"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-14,2,-11],"to":[14,42,11],"autouv":0,"color":3,"origin":[0,0,0],"faces":{"north":{"uv":[752,22,780,62],"texture":0},"east":{"uv":[0,0,0,0],"texture":null},"south":{"uv":[780,22,752,62],"texture":0},"west":{"uv":[0,0,0,0],"texture":null},"up":{"uv":[780,22,752,0],"texture":0},"down":{"uv":[780,84,752,62],"texture":0}},"type":"cube","uuid":"2a5d4d0b-6422-a957-2a41-1e19c9bf2421"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[16,44,13],"to":[-16,0,-13],"autouv":0,"color":7,"origin":[0,0,0],"faces":{"north":{"uv":[0,266,1,267],"texture":0},"east":{"uv":[0,266,1,267],"texture":0},"south":{"uv":[0,266,1,267],"texture":0},"west":{"uv":[0,266,1,267],"texture":0},"up":{"uv":[0,266,1,267],"texture":0},"down":{"uv":[0,266,1,267],"texture":0}},"type":"cube","uuid":"b82aa407-9a9f-9c7b-68a5-9fad0275ef4a"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-46,2,-11],"to":[-18,42,11],"autouv":0,"color":8,"origin":[0,0,0],"faces":{"north":{"uv":[150,22,178,62],"texture":0},"east":{"uv":[0,0,0,0],"texture":null},"south":{"uv":[178,22,150,62],"texture":0},"west":{"uv":[0,0,0,0],"texture":null},"up":{"uv":[178,22,150,0],"texture":0},"down":{"uv":[178,84,150,62],"texture":0}},"type":"cube","uuid":"659a046d-9f4f-a2c7-237a-1de0dbb7a553"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-48,28,-13],"to":[-32,32,13],"autouv":0,"color":4,"origin":[0,0,0],"faces":{"north":{"uv":[0,0,0,0],"texture":null},"east":{"uv":[0,266,1,267],"texture":0},"south":{"uv":[0,0,0,0],"texture":null},"west":{"uv":[0,0,0,0],"texture":null},"up":{"uv":[0,266,1,267],"texture":0},"down":{"uv":[0,266,1,267],"texture":0}},"type":"cube","uuid":"7c7df06e-6ddc-e08a-7520-63a50206a49b"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-48,12,-13],"to":[-32,16,13],"autouv":0,"color":3,"origin":[0,0,0],"faces":{"north":{"uv":[0,0,0,0],"texture":null},"east":{"uv":[0,266,1,267],"texture":0},"south":{"uv":[0,0,0,0],"texture":null},"west":{"uv":[0,0,0,0],"texture":null},"up":{"uv":[0,266,1,267],"texture":0},"down":{"uv":[0,266,1,267],"texture":0}},"type":"cube","uuid":"c864e211-55cf-1365-5ce8-67e847a4995f"},{"name":"cube","box_uv":false,"rescale":false,"locked":false,"render_order":"default","allow_mirror_modeling":true,"from":[-16,44,13],"to":[-48,0,-13],"autouv":0,"color":0,"origin":[0,0,0],"faces":{"north":{"uv":[3,266,5,277],"texture":0},"east":{"uv":[4,266,3,277],"texture":0},"south":{"uv":[5,266,3,277],"texture":0},"west":{"uv":[0,266,1,267],"texture":0},"up":{"uv":[0,266,1,267],"texture":0},"down":{"uv":[0,266,1,267],"texture":0}},"type":"cube","uuid":"cdd7483e-f1f5-d39c-253b-acd1589e75d6"}],"outliner":[{"name":"axe","origin":[0,0,0],"color":0,"uuid":"0db5d727-94bb-5a6a-7051-7169720a0fe8","export":true,"mirror_uv":false,"isOpen":false,"locked":false,"visibility":true,"autouv":0,"children":[{"name":"a","origin":[0,0,0],"color":0,"uuid":"9bab9aa9-cd30-894f-5da1-5846c4006682","export":true,"mirror_uv":false,"isOpen":false,"locked":false,"visibility":true,"autouv":0,"children":["a19027f8-0948-b856-d7a5-a5d95d641531","a12bd1d9-1ff4-db18-7b2b-5225cf43e664"]},{"name":"x","origin":[0,0,0],"color":0,"uuid":"749cf6e7-7274-4afe-bb90-9031abbb6ec7","export":true,"mirror_uv":false,"isOpen":false,"locked":false,"visibility":true,"autouv":0,"children":["2a5d4d0b-6422-a957-2a41-1e19c9bf2421","b82aa407-9a9f-9c7b-68a5-9fad0275ef4a"]},{"name":"e","origin":[0,0,0],"color":0,"uuid":"b6e2c70e-377d-26f8-4032-f868e0d6d46a","export":true,"mirror_uv":false,"isOpen":false,"locked":false,"visibility":true,"autouv":0,"children":["659a046d-9f4f-a2c7-237a-1de0dbb7a553","7c7df06e-6ddc-e08a-7520-63a50206a49b","c864e211-55cf-1365-5ce8-67e847a4995f","cdd7483e-f1f5-d39c-253b-acd1589e75d6"]}]}],"textures":[{"path":"","name":"axe.png","folder":"","namespace":"","id":"0","width":1000,"height":320,"uv_width":1000,"uv_height":320,"particle":false,"use_as_default":false,"layers_enabled":false,"sync_to_project":"","render_mode":"default","render_sides":"front","frame_time":1,"frame_order_type":"loop","frame_order":"","frame_interpolate":false,"visible":true,"internal":true,"saved":false,"uuid":"c5ae31d9-4a26-129d-9225-deda159af354","source":""}]} \ No newline at end of file diff --git a/guidelines/thumbnails/title_part2.png b/guidelines/thumbnails/title_part2.png new file mode 100644 index 0000000..aa9f83b Binary files /dev/null and b/guidelines/thumbnails/title_part2.png differ diff --git a/mcip b/mcip index e4d3489..4d80004 100644 --- a/mcip +++ b/mcip @@ -6,9 +6,10 @@ watch="" target="debug" clean=false package_only=false +dev=false # Read the options -TEMP=`getopt -o hiw:t:cp --long help,init,watch:,target:,clean,package-only -n 'build_script' -- "$@"` +TEMP=`getopt -o hiw:t:cpd --long help,init,watch:,target:,clean,package-only,dev -n 'build_script' -- "$@"` eval set -- "$TEMP" # Extract options and their arguments into variables @@ -22,28 +23,36 @@ while true ; do echo -e "[--target] -> Whether to build the addon in debug or release mode. Default is 'debug'. \nCan Generate Config out of settings.json" echo "[--clean | -c] -> Clean 'BP/scripts' folder before building." echo "[--package-only | -p] -> Only package what's already there." + echo "[--dev | -d] -> The usual watch and target debug" + echo "-----------------------------------------------------------" + echo "Debugging: ./mcip --watch \"stable\" --target \"debug\" " + echo "Building: ./mcip --target \"release\" " exit 0 ;; -i|--init) init=true ; shift ;; -w|--watch) watch=$2 ; shift 2 ;; --target) - if ! $init ; then - target=$2 - fi - shift 2 ;; + target=$2 ; shift 2 ;; -c|--clean) clean=true ; shift ;; -p|--package-only) package_only=true ; shift ;; + -d|--dev) + dev=true ; shift ;; --) shift ; break ;; *) echo "Internal error!" ; exit 1 ;; esac done +# Set default values for dev option +if $dev; then + watch="stable" + target="debug" +fi + # Build the command command="python tools/build.py" -command2="python tools/process_config.py --generateConfigTS" if $init; then command+=" --init" @@ -53,9 +62,7 @@ if [[ ! -z "$watch" ]]; then command+=" --watch $watch" fi -if ! $init; then - command+=" --target $target" -fi +command+=" --target $target" if $clean; then command+=" --clean" @@ -66,11 +73,4 @@ if $package_only; then fi # Execute the command - eval $command - -# Execute command2 if --init was called -if $init; then - command="python tools/process_manifest.py --init" - eval $command2 -fi diff --git a/package.json b/package.json index 7d0c2e3..42fe6fb 100644 --- a/package.json +++ b/package.json @@ -9,8 +9,12 @@ "keywords": [], "author": "", "license": "ISC", + "devDependencies": { + "typescript": "^5.5.3" + }, "dependencies": { - "@minecraft/server": "^1.9.0-beta.1.20.60-stable", - "@minecraft/server-ui": "^1.2.0-beta.1.20.30-preview.25" + "@minecraft/common": "^1.2.0", + "@minecraft/server": "^1.13.0", + "@minecraft/server-ui": "^1.2.0" } -} +} \ No newline at end of file diff --git a/setup/mc_manifest.json b/setup/mc_manifest.json index a690673..c785a56 100644 --- a/setup/mc_manifest.json +++ b/setup/mc_manifest.json @@ -1,15 +1,13 @@ { "format_version": 2, "header": { - "bp_name": "Lumber Axe BP: Chopping made easy!", - "rp_name": "Lumber Axe RP: Chopping made easy!", - - "description": "Cut the whole tree with easy as a lumberjack's pie! We create addons/datapack that will ease your survival plays. \n @Created By: @h_YanG_0A & @Brilliant", - - "bp_uuid": "4538bab6-a78b-4710-89e5-a2e103e97a6a", + "bp_name": "Lumber Axe BP", + "rp_name": "Lumber Axe RP", + "description": "An addon that gives you a lumberjack axes, which can preview the tree's status, and chopdown trees. \n @Created By: @Adr-hyng", + "bp_uuid": "4538bab6-a78b-4710-89e5-a2e103e97a6a", "rp_uuid": "62b4cab4-fa42-405b-bff6-1f47b6958dd6", - "version": [1, 0, 5], - "min_engine_version": [ 1, 20, 10 ] + "version": [2, 0, 0], + "min_engine_version": [ 1, 21, 0 ] }, "bp_modules": [ { @@ -23,7 +21,7 @@ "type": "script", "language": "javascript", "uuid": "9717780a-c69f-4679-bbe6-dbf6dfcdd7de", - "version": [ 1, 0, 5 ], + "version": [ 2, 0, 0 ], "entry": "scripts/main.js" } ], @@ -38,23 +36,19 @@ "bp_dependencies": [ { "module_name": "@minecraft/server", - "version": "1.9.0-beta" + "version": "1.13.0" }, { "module_name": "@minecraft/server-ui", - "version": "1.2.0-beta" - } - ], - "bp_server_dependencies": [ - { - "module_name": "@minecraft/server-admin", - "version": "1.0.0-beta" + "version": "1.2.0" } ], + "bp_server_dependencies": [], "metadata": { "authors": [ "@h_YanG_0A", - "@Brilliant" + "Contributor-@Brilliant", + "Contributor-@Dal4y" ], "license": "GPL-3.0-or-later", "url": "https://twitter.com/h_YanG_0A" diff --git a/setup/pack_icon.png b/setup/pack_icon.png index a7674c9..d29dae1 100644 Binary files a/setup/pack_icon.png and b/setup/pack_icon.png differ diff --git a/src/.config_hashes b/src/.config_hashes new file mode 100644 index 0000000..a4c92d1 --- /dev/null +++ b/src/.config_hashes @@ -0,0 +1,3 @@ +b247084a7b5b5a837137f07c567f42eb69d52a569c2c8111f376856df97a254f +741d92373a0ce9a321ab6625c1fd8fbe815e722eccd10985a987541893812b1c +741d92373a0ce9a321ab6625c1fd8fbe815e722eccd10985a987541893812b1c diff --git a/src/classes/entity_override.ts b/src/classes/entity_override.ts new file mode 100644 index 0000000..f9cd391 --- /dev/null +++ b/src/classes/entity_override.ts @@ -0,0 +1,11 @@ +import { Entity } from "@minecraft/server"; +import { OverTakes } from "./partial_overtakes"; + +declare module "@minecraft/server" { + interface Entity { + lastLocation: Vector3; + } +} + +OverTakes(Entity.prototype, { +}); \ No newline at end of file diff --git a/src/classes/item_equippable.ts b/src/classes/item_equippable.ts new file mode 100644 index 0000000..3fcab67 --- /dev/null +++ b/src/classes/item_equippable.ts @@ -0,0 +1,52 @@ +import { ItemComponentTypes, ItemDurabilityComponent, ItemEnchantableComponent, ItemStack, Player } from "@minecraft/server"; +import { EntityEquippableComponent, EquipmentSlot } from "@minecraft/server"; +import { MinecraftEnchantmentTypes, MinecraftItemTypes} from "modules/vanilla-types/index"; +import { OverTakes } from "./partial_overtakes"; + +declare module "@minecraft/server" { + interface EntityEquippableComponent { + get equipment(): ItemStack; + get isEquipped(): boolean; + /** + * This function damage a durability, then returns if item just broke due to low durability, or it did not broke. + * @param equipment tool/equipment to reduce durability. + * @param damageApplied amount of durability deducted to the tool/equipment. + * @returns {boolean} + */ + damageDurability(damageApplied: number): boolean; + } +} + +OverTakes(EntityEquippableComponent.prototype, { + get equipment() { + return this.getEquipment(EquipmentSlot.Mainhand); + }, + get isEquipped() { + return (this.equipment?.typeId === MinecraftItemTypes.FishingRod); + }, + damageDurability(this, damageApplied: number): boolean { + const equipmentToDamage: ItemStack = this.getEquipment(EquipmentSlot.Mainhand) as ItemStack; + if(!equipmentToDamage) return false; + const player = this.entity as Player; + if(!player.isSurvival()) return false; + if(!equipmentToDamage?.hasComponent(ItemComponentTypes.Durability)) throw "Item doesn't have durability to damage with"; + let level: number = 0; + const itemDurability: ItemDurabilityComponent = (equipmentToDamage.getComponent(ItemComponentTypes.Durability) as ItemDurabilityComponent); + if(equipmentToDamage.hasComponent(ItemComponentTypes.Enchantable)) { + const enchantment = equipmentToDamage.getComponent(ItemComponentTypes.Enchantable) as ItemEnchantableComponent; + if(enchantment.hasEnchantment(MinecraftEnchantmentTypes.Unbreaking)) level = enchantment.getEnchantment(MinecraftEnchantmentTypes.Unbreaking).level; + } + const unbreakingMultiplier: number = (100 / (level + 1)) / 100; + const unbreakingDamage: number = damageApplied * unbreakingMultiplier; + if(itemDurability.damage + unbreakingDamage >= itemDurability.maxDurability){ + this.setEquipment(EquipmentSlot.Mainhand, undefined); + player.playSound("random.break"); + return true; + } else if(itemDurability.damage + unbreakingDamage < itemDurability.maxDurability){ + (equipmentToDamage.getComponent(ItemComponentTypes.Durability) as ItemDurabilityComponent).damage += unbreakingDamage; + this.setEquipment(EquipmentSlot.Mainhand, equipmentToDamage); + return false; + } + } + } +); diff --git a/src/classes/itemstack_override.ts b/src/classes/itemstack_override.ts new file mode 100644 index 0000000..e344317 --- /dev/null +++ b/src/classes/itemstack_override.ts @@ -0,0 +1,16 @@ +// import { GameMode, ItemStack } from "@minecraft/server"; +// import { OverTakes } from "./partial_overtakes"; +// import { Configuration } from "configuration/configuration_screen"; + +// declare module "@minecraft/server" { +// interface ItemStack { +// getID(): string; +// } +// } + +// OverTakes(ItemStack.prototype, { +// getID(): string { +// this.getDynamicProperty(); +// return ""; +// } +// }); \ No newline at end of file diff --git a/src/classes/partial_overtakes.ts b/src/classes/partial_overtakes.ts new file mode 100644 index 0000000..e720819 --- /dev/null +++ b/src/classes/partial_overtakes.ts @@ -0,0 +1,9 @@ +type PartialParts = { + [P in keyof b]?: b[P] extends (...param: infer param)=>infer ret?((this: thisArg,...param:param)=>ret):b[P] +}; +export function OverTakes(prototype: b, object: PartialParts): b{ + const prototypeOrigin = Object.setPrototypeOf(Object.defineProperties({},Object.getOwnPropertyDescriptors(prototype)),Object.getPrototypeOf(prototype)); + Object.setPrototypeOf(object, prototypeOrigin); + Object.defineProperties(prototype, Object.getOwnPropertyDescriptors(object)); + return prototypeOrigin; +} \ No newline at end of file diff --git a/src/classes/player.ts b/src/classes/player.ts index c6583d8..0fc43f8 100644 --- a/src/classes/player.ts +++ b/src/classes/player.ts @@ -1,11 +1,23 @@ import { GameMode, Player } from "@minecraft/server"; +import { OverTakes } from "./partial_overtakes"; +import { Configuration } from "configuration/configuration_screen"; declare module "@minecraft/server" { interface Player { - isSurvival(this: Player): boolean; + configuration: Configuration; + isSurvival(): boolean; } } -Player.prototype.isSurvival = function(): boolean { - return this.dimension.getPlayers({ gameMode: GameMode.survival, name: this.name, location: this.location, maxDistance: 1, closest: 1 }).length > 0; -} \ No newline at end of file +const screenConfigs = new WeakMap(); + +OverTakes(Player.prototype, { + isSurvival(): boolean { + return this.getGameMode() === GameMode.survival; + }, + get configuration() { + let sc = screenConfigs.get(this); + if(!sc) screenConfigs.set(this, sc = new Configuration(this)); + return sc; + } +}); \ No newline at end of file diff --git a/src/commands/ICommandBase.ts b/src/commands/ICommandBase.ts new file mode 100644 index 0000000..c489696 --- /dev/null +++ b/src/commands/ICommandBase.ts @@ -0,0 +1,8 @@ +import { Player, ScriptEventCommandMessageAfterEvent } from "@minecraft/server"; +export interface ICommandBase { + name: string, + description: string, + format: string, + usage(): string, + execute(player: Player, args: string[]): void | Promise; +} \ No newline at end of file diff --git a/src/commands/command_handler.ts b/src/commands/command_handler.ts new file mode 100644 index 0000000..bc0eccd --- /dev/null +++ b/src/commands/command_handler.ts @@ -0,0 +1,13 @@ +// Credits to: https://github.com/RohanDaCoder/CommandHandler + +import { ADDON_IDENTIFIER } from "constant"; + +export const CommandHandler = { + prefix: `/scriptevent ${ADDON_IDENTIFIER} `, + commands: [ + 'help', + 'config', + 'database', + 'dev_helper' + ] +}; \ No newline at end of file diff --git a/src/commands/config.ts b/src/commands/config.ts new file mode 100644 index 0000000..7e0a56c --- /dev/null +++ b/src/commands/config.ts @@ -0,0 +1,59 @@ +import { system } from "@minecraft/server"; +import { CommandHandler } from "commands/command_handler"; +import { ICommandBase} from "./ICommandBase"; +import { SendMessageTo } from "utils/utilities"; +import "classes/player"; + +enum REQUIRED_PARAMETER { + SHOW = "show", + RESET = "reset" +} +enum OPTIONAL_PARAMETER { + CLIENT = "client", + SERVER = "server" +} + +const command: ICommandBase = { + name: 'config', + description: 'Show or reset configuration settings.', + format: `[${Object.values(REQUIRED_PARAMETER).join('|')}] [<${Object.values(OPTIONAL_PARAMETER).join('|')}>?]`, + usage() { + return (` + Format: + > ${CommandHandler.prefix}${this.name} ${this.format} + Usage: + > ${CommandHandler.prefix}${this.name} ${REQUIRED_PARAMETER.SHOW} = Shows config + > ${CommandHandler.prefix}${this.name} ${REQUIRED_PARAMETER.RESET} ${OPTIONAL_PARAMETER.CLIENT} = Reset caller client config + > ${CommandHandler.prefix}${this.name} ${REQUIRED_PARAMETER.RESET} ${OPTIONAL_PARAMETER.SERVER} = Reset world server config (Admin) + `).replaceAll(" ", ""); + }, + async execute(player, args) { + if (args && args.length) { + const requiredParams: string[] = (`[${Object.values(REQUIRED_PARAMETER).join('|')}]`).slice(1, -1).split('|').map(command => command.trim()); + const selectedReqParam: string = args[0].toLowerCase(); + const isShow: boolean = REQUIRED_PARAMETER.SHOW === selectedReqParam; + if(!requiredParams.includes(selectedReqParam)) return SendMessageTo( + player, { + rawtext: [ + { + translate: "LumberAxe.on_caught_invalid_command", + with: [command.usage()] + }, + ] + } + ); + if(isShow) { + system.run(() => player.configuration.showServerScreen()); + } + else { + const optionalParams: string[] = (`[${Object.values(OPTIONAL_PARAMETER).join('|')}]`).slice(1, -1).split('|').map(command => command.trim()); + const selectedOptParam: string = args[1]?.toLowerCase(); + let shouldResetClient: boolean = OPTIONAL_PARAMETER.CLIENT === selectedOptParam; + if(!optionalParams.includes(selectedOptParam)) shouldResetClient = true; + if(shouldResetClient) player.configuration.reset("CLIENT"); + } + } + } +}; + +export default command \ No newline at end of file diff --git a/src/commands/database.ts b/src/commands/database.ts new file mode 100644 index 0000000..02f5b48 --- /dev/null +++ b/src/commands/database.ts @@ -0,0 +1,82 @@ +import { Player, world } from "@minecraft/server"; +import { CommandHandler } from "commands/command_handler"; +import { originalDatabase, ADDON_NAME } from "constant"; +import { ICommandBase} from "./ICommandBase"; +import { SendMessageTo } from "utils/utilities"; + +enum REQUIRED_PARAMETER { + SHOW = "show", + RESET = "reset" +} + +const command: ICommandBase = { + name: 'database', + description: 'Inspect or reset a database.', + format: `[${Object.values(REQUIRED_PARAMETER).join('|')}]`, + usage() { + return (` + Format: + > ${CommandHandler.prefix}${this.name} ${this.format} + Usage: + > ${CommandHandler.prefix}${this.name} ${REQUIRED_PARAMETER.SHOW} = Display database content. + > ${CommandHandler.prefix}${this.name} ${REQUIRED_PARAMETER.RESET} = Reset database content. + `).replaceAll(" ", ""); + }, + execute(player, args) { + if (args && args.length) { + const requiredParams: string[] = (`[${Object.values(REQUIRED_PARAMETER).join('|')}]`).slice(1, -1).split('|').map(command => command.trim()); + const selectedReqParam: string = args[0].toLowerCase(); + const isShow: boolean = REQUIRED_PARAMETER.SHOW === selectedReqParam; + if(!requiredParams.includes(selectedReqParam)) return SendMessageTo( + player, { + rawtext: [ + { + translate: "LumberAxe.on_caught_invalid_command", + with: [command.usage()] + }, + ] + } + ); + if(isShow) { + if(originalDatabase.size === 0) return SendMessageTo( + player, { + rawtext: [ + { + translate: "LumberAxe.on_database_empty" + }, + ] + } + ); + let collections: string = ""; + let i = 1; + for(const key of originalDatabase.keys()) { + const t: string[] = (key as string).split("|"); + const player: Player = world.getEntity(t[1]) as Player; + collections += `${i++}. ${player.nameTag}: ${JSON.stringify(t)}\n`; + } + SendMessageTo(player, { + rawtext: [ + { + translate: "LumberAxe.show_database", + with: [ADDON_NAME, "\n", collections] + }, + ] + }); + } else { + SendMessageTo(player, { + rawtext: [ + { + translate: "LumberAxe.on_database_reset" + }, + ] + }); + player.configuration.reset("CLIENT"); + player.configuration.reset("SERVER"); + originalDatabase.clear(); + if(!originalDatabase.isDisposed) originalDatabase.dispose(); + } + } + } +}; + +export default command \ No newline at end of file diff --git a/src/commands/dev_helper.ts b/src/commands/dev_helper.ts new file mode 100644 index 0000000..8ff8112 --- /dev/null +++ b/src/commands/dev_helper.ts @@ -0,0 +1,62 @@ +import { EntityComponentTypes, EntityInventoryComponent, ItemStack, MolangVariableMap } from "@minecraft/server"; +import { CommandHandler } from "commands/command_handler"; +import { ICommandBase} from "./ICommandBase"; +import { SendMessageTo} from "utils/utilities"; +import { axeEquipments, originalDatabase, resetOriginalDatabase, visitedLogs } from "constant"; + +// Automate this, the values should be the description. +enum REQUIRED_PARAMETER { + GET = "get", + TEST = "test", + RELOAD = "reload", +} + +const command: ICommandBase = { + name: 'dev_helper', + description: 'Developer Utility Command', + format: `[${Object.values(REQUIRED_PARAMETER).join('|')}]`, + usage() { + //? It should be Automatic + return (` + Format: + > ${CommandHandler.prefix}${this.name} ${this.format} + Usage: + > ${CommandHandler.prefix}${this.name} ${REQUIRED_PARAMETER.GET} = GETS an enchanted fishing rod for development. + > ${CommandHandler.prefix}${this.name} ${REQUIRED_PARAMETER.TEST} = TEST a Working-in-progress features. + > ${CommandHandler.prefix}${this.name} ${REQUIRED_PARAMETER.RELOAD} = Reloads the addon. + `).replaceAll(" ", ""); + }, + execute(player, args) { + if (!(args && args.length)) return; + const requiredParams: string[] = (`[${Object.values(REQUIRED_PARAMETER).join('|')}]`).slice(1, -1).split('|').map(command => command.trim()); + const selectedReqParam: string = args[0].toLowerCase(); + if(!requiredParams.includes(selectedReqParam)) return SendMessageTo( + player, { + rawtext: [ + { + translate: "LumberAxe.on_caught_invalid_command", + with: [command.usage()] + }, + ] + } + ); + switch(selectedReqParam) { + case REQUIRED_PARAMETER.GET: + for(const axe of axeEquipments) { + (player.getComponent(EntityComponentTypes.Inventory) as EntityInventoryComponent).container.addItem(new ItemStack(axe, 1)); + } + break; + case REQUIRED_PARAMETER.TEST: + console.warn(originalDatabase.size, visitedLogs.length); + break; + case REQUIRED_PARAMETER.RELOAD: + originalDatabase.clear(); + resetOriginalDatabase(); + console.warn(originalDatabase.isValid(), originalDatabase.size); + break; + default: + break; + } + } +}; +export default command \ No newline at end of file diff --git a/src/commands/help.ts b/src/commands/help.ts new file mode 100644 index 0000000..21b184a --- /dev/null +++ b/src/commands/help.ts @@ -0,0 +1,85 @@ +import { Player } from '@minecraft/server'; +import { CommandHandler} from './command_handler'; +import { ICommandBase} from "./ICommandBase"; +import { ADDON_NAME } from 'constant'; +import { SendMessageTo } from 'utils/utilities'; + +const importCommand = async (player: Player, commandName: string): Promise => { + try { + const importedCommandModule = await import(`./${commandName}.js`); + return importedCommandModule.default; + } catch (error) { + SendMessageTo(player, { + rawtext: [ + { + translate: "LumberAxe.on_caught_command_404", + with: [ commandName, error.message] + } + ] + }); + return null; + } +}; + +const details = { + __addonName__: ADDON_NAME, + __name__: 'help', + __description__: 'Displays the help message.', + __format__: '[?]', +} + +const command: ICommandBase = { + name: details.__name__, + description: details.__description__, + format: details.__format__, + usage(): string { + return (`Format: + > ${CommandHandler.prefix}${this.name} ${this.format} + Usage: + > ${CommandHandler.prefix}${this.name} + > ${CommandHandler.prefix}${this.name} config + `).replaceAll(" ", ""); + }, + async execute(player, args) { + if (!args || args.length === 0) { + let helpMessage: string = `\n§aCommands available @ ${details.__addonName__}: \n`; + for (const commandName of CommandHandler.commands) { + const importedCommand = await importCommand(player, commandName); + if (importedCommand) helpMessage += `§e${CommandHandler.prefix}${commandName}§r${importedCommand.format.length ? " " + importedCommand.format : ""} - ${importedCommand.description}\n`; + } + SendMessageTo(player, { + rawtext: [ + { + text: helpMessage + } + ] + }); + } else { + const specifiedCommand = args[0].toLowerCase(); + if(!CommandHandler.commands.includes(specifiedCommand)) return SendMessageTo( + player, { + rawtext: [ + { + translate: "LumberAxe.on_caught_invalid_command", + with: [command.usage()] + }, + ] + } + ); + if (CommandHandler.commands.includes(specifiedCommand)) { + const importedCommand = await importCommand(player, specifiedCommand); + if (importedCommand) { + SendMessageTo(player, { + rawtext: [ + { + text: `\n§e${CommandHandler.prefix}${specifiedCommand}: \n${importedCommand.description}§r ${importedCommand.usage()}` + } + ] + }); + } + } + } + } +} + +export default command; \ No newline at end of file diff --git a/src/config.ts b/src/config.ts deleted file mode 100644 index 931e423..0000000 --- a/src/config.ts +++ /dev/null @@ -1,34 +0,0 @@ -export default { - /** - * Enables debug messages to content logs. - */ - debug: true, - /** - * Lumber Axe durability damage per log destroyed. - */ - durabilityDamagePerBlock: 3, - /** - * 1500 above is not recommended. It Does work but it's not recommended. - */ - chopLimit: 300, - /** - * Included blocks for custom logs, but any custom or vanilla logs also work as long as the block identifier ends with "*_log". - * Check: https://github.com/mcbe-mods/Cut-tree-one-click by Lete114. - */ - includedLog: [], - /** - * Excluded blocks for block logs you don't want to be included in being chopped. - * - * Tip: - * - excludedLog is prioritized over includedLog. - * - It's unnecessary to include log blocks that have "*_log" in their block id. - */ - excludedLog: [], - /** - * Disables the watchDogTerminate Log message. If true, it will only show a warning message when you enable content-ui log in the Minecraft settings. - */ - disableWatchDogTerminateLog: true, -}; - -// version (do not change) -export const VERSION = "1.0.5"; \ No newline at end of file diff --git a/src/configuration/configuration_handler.ts b/src/configuration/configuration_handler.ts new file mode 100644 index 0000000..90ef703 --- /dev/null +++ b/src/configuration/configuration_handler.ts @@ -0,0 +1,26 @@ +import { Player } from "@minecraft/server"; +import { ADDON_NAME } from "constant"; +import { FormBuilder } from "utils/form_builder"; + +export type ConfigurationTypes = "SERVER" | "CLIENT"; +export const ConfigurationCollections_DB = (player: Player, configType: ConfigurationTypes = "CLIENT") => `${ADDON_NAME}|${player.id}|${configType}`; + +export function cloneConfiguration>>(config: T): T { + let clonedConfig = {} as T; + for (const [key, _formBuilder] of Object.entries(config)) { + const formBuilder = >_formBuilder; + const isArrayEmpty = formBuilder.values.length > 0; + const newFormBuilder = new FormBuilder(formBuilder.name); + if (typeof formBuilder.defaultValue === "string" && isArrayEmpty) { + newFormBuilder.createDropdown(formBuilder.values, formBuilder.defaultValue); + } + else if (typeof formBuilder.defaultValue === "string" && !isArrayEmpty) { + newFormBuilder.createTextField(formBuilder.defaultValue); + } + else if (typeof formBuilder.defaultValue === "boolean") { + newFormBuilder.createToggle(formBuilder.defaultValue as boolean); + } + clonedConfig[key as keyof T] = newFormBuilder as T[keyof T]; + } + return clonedConfig; +} \ No newline at end of file diff --git a/src/configuration/configuration_screen.ts b/src/configuration/configuration_screen.ts new file mode 100644 index 0000000..f161612 --- /dev/null +++ b/src/configuration/configuration_screen.ts @@ -0,0 +1,252 @@ +import { Player } from "@minecraft/server"; +import { ActionFormData, ActionFormResponse, FormCancelationReason, ModalFormData, ModalFormResponse } from "@minecraft/server-ui"; +import { ConfigurationCollections_DB, ConfigurationTypes} from "./configuration_handler"; +import {ADDON_NAME, originalDatabase, resetOriginalDatabase} from "constant"; +import { FormBuilder } from "utils/form_builder"; +import { resetServerConfiguration, serverConfigurationCopy, setServerConfiguration } from "./server_configuration"; +import { SendMessageTo } from "utils/utilities"; + +export class Configuration { + private player: Player; + private SERVER_CONFIGURATION_DB: string; + private CLIENT_CONFIGURATION_DB: string + + isConfigurationSettingsOpen: boolean; + constructor(player: Player) { + this.player = player; + this.isConfigurationSettingsOpen = false; + this.CLIENT_CONFIGURATION_DB = ConfigurationCollections_DB(this.player, "CLIENT"); + this.SERVER_CONFIGURATION_DB = ConfigurationCollections_DB(this.player, "SERVER"); + } + reset(configurationType: ConfigurationTypes) { + if(originalDatabase.isValid()) { + if(configurationType === "SERVER") { + resetServerConfiguration(); + originalDatabase.set(this.SERVER_CONFIGURATION_DB, serverConfigurationCopy); + } + } + else throw new Error("Database not found"); + } + saveServer() { + setServerConfiguration(serverConfigurationCopy); + if (originalDatabase.isValid()) originalDatabase.set(this.SERVER_CONFIGURATION_DB, serverConfigurationCopy); + else { + resetOriginalDatabase(); + } + } + loadServer() { + if (originalDatabase?.isValid()) { + if (originalDatabase.has(this.SERVER_CONFIGURATION_DB)) { + setServerConfiguration( originalDatabase.get(this.SERVER_CONFIGURATION_DB) ); + } else { + originalDatabase.set(this.SERVER_CONFIGURATION_DB, serverConfigurationCopy); + } + } + } + showServerScreen() { + const parsedAddonTitle = ADDON_NAME.toLowerCase().replace(/_/g, ' ').replace(/\b\w/g, char => char.toUpperCase()); + const form = new ActionFormData() + .title({rawtext: [ + {translate: "LumberAxe.configuration.title", with: [parsedAddonTitle]} + ]}) + .button({rawtext: [ + {translate: "LumberAxe.configuration.general"} + ]}) + .button({rawtext: [ + {translate: "LumberAxe.configuration.log_include_manager"} + ]}) + .button({rawtext: [ + {translate: "LumberAxe.configuration.log_exclude_manager"} + ]}) + form.show(this.player).then( (response: ActionFormResponse) => { + if (response.canceled || response.cancelationReason === FormCancelationReason.UserClosed || response.cancelationReason === FormCancelationReason.UserBusy) return; + switch(response.selection) { + case 0: return this.showGeneralOptions(); + case 1: return this.showIncludeManager(); + case 2: return this.showExcludeManager(); + default: + break; + } + return; + }); + } + showGeneralOptions() { + const form: ModalFormData = new ModalFormData().title({rawtext: [ + {translate: "LumberAxe.configuration.general"} + ]}); + + this.loadServer(); + + const cachedConfigurationValues: Array<{result: number | boolean | string, index: number}> = []; + + // Only good for read-only Dropdowns + Object.values(serverConfigurationCopy).forEach((builder, index) => { + const isNotDropdown = (builder.values.length === 0); + if (typeof builder.defaultValue === "boolean" && isNotDropdown) { + cachedConfigurationValues.push({result: builder.defaultValue, index}); + form.toggle({rawtext: [{translate: builder.name}]}, builder.defaultValue as boolean); + } + else if (typeof builder.defaultValue === "string" && isNotDropdown) { + cachedConfigurationValues.push({result: builder.defaultValue, index}); + form.textField({rawtext: [{translate: builder.name}]}, builder.defaultValue, builder.defaultValue); + } + }); + + form.show(this.player).then((result: ModalFormResponse) => { + if (!result.formValues) return; + const hadChanges: boolean = !cachedConfigurationValues.every(({result: element}, i) => element === result.formValues[i]); + if (result.canceled || result.cancelationReason === FormCancelationReason.UserClosed || result.cancelationReason === FormCancelationReason.UserBusy) { + return; + } + if (hadChanges) { + result.formValues.forEach((newValue, formIndex) => { + const index = cachedConfigurationValues[formIndex].index; + const key = Object.keys(serverConfigurationCopy)[index]; + const builder = serverConfigurationCopy[key] as FormBuilder; + switch (typeof newValue) { + case "boolean": + builder.defaultValue = newValue; + break; + case "number": + builder.defaultValue = builder.values[newValue]; + break; + case "string": + builder.defaultValue = newValue; + break; + default: + break; + } + serverConfigurationCopy[key] = builder; + }); + this.saveServer(); + } + return this.showServerScreen(); + }); + } + showIncludeManager() { + this.loadServer(); + const preResultFlags: Array = []; + let index = 0; + preResultFlags[index] = 0; index++; + preResultFlags[index] = ""; index++; + preResultFlags[index] = false; index++; + const form = new ModalFormData() + .title({rawtext: [ + {translate: "LumberAxe.configuration.log_include_manager"} + ]}) + .dropdown({rawtext: [ + {translate: "LumberAxe.log_include_manager.drop_down"} + ]}, [...serverConfigurationCopy.includedLog.values], 0) + .textField({rawtext: [ + {translate: "LumberAxe.log_include_manager.text_field"} + ]}, "myaddon:custom_log", preResultFlags[1] as string) + .toggle({rawtext: [ + {translate: "LumberAxe.log_include_manager.toggle"} + ]}, preResultFlags[2] as boolean); + form.show(this.player).then((response: ModalFormResponse) => { + if(!response.formValues) return; + if (response.canceled || response.cancelationReason === FormCancelationReason.UserClosed || response.cancelationReason === FormCancelationReason.UserBusy) return; + const hadChanges: boolean = !preResultFlags.every((element, index) => element === response.formValues[index]); + const selectedIndex: number = (response.formValues[0] as number); + let canUpdate: boolean = response.formValues[2] as boolean; + const dropDownContent: string = response.formValues[1] as string; + const dropDownSelected: number = (selectedIndex as number); + const isEmpty: boolean = !dropDownContent.length; + if (!hadChanges) return this.showServerScreen(); + if(isEmpty) canUpdate = false; + if(canUpdate){ + if(selectedIndex === 0) { + serverConfigurationCopy.includedLog.values.push(dropDownContent); + SendMessageTo(this.player, {rawtext: [ + {text: `§aLumber Axe: `}, + {translate: "LumberAxe.log_include_manager.add_success_log", with: [dropDownContent]} + ]}); + } + else { + if(serverConfigurationCopy.includedLog.values.length) { + const prevValue = serverConfigurationCopy.includedLog.values[dropDownSelected]; + serverConfigurationCopy.includedLog.values[dropDownSelected] = dropDownContent; + SendMessageTo(this.player, {rawtext: [ + {text: `§eLumber Axe: `}, + {translate: "LumberAxe.log_include_manager.update_success_log", with: [prevValue, dropDownContent]} + ]}); + } + } + } else { + if(serverConfigurationCopy.includedLog.values.length && selectedIndex !== 0) { + const itemDeleted = serverConfigurationCopy.includedLog.values.splice(dropDownSelected, 1)[0]; + SendMessageTo(this.player, {rawtext: [ + {text: `§cLumber Axe: `}, + {translate: "LumberAxe.log_include_manager.remove_success_log", with: [itemDeleted]} + ]}); + } + } + this.saveServer(); + return this.showServerScreen(); + }); + } + showExcludeManager() { + this.loadServer(); + const preResultFlags: Array = []; + let index = 0; + preResultFlags[index] = 0; index++; + preResultFlags[index] = ""; index++; + preResultFlags[index] = false; index++; + const form = new ModalFormData() + .title({rawtext: [ + {translate: "LumberAxe.configuration.log_exclude_manager"} + ]}) + .dropdown({rawtext: [ + {translate: "LumberAxe.log_exclude_manager.drop_down"} + ]}, [...serverConfigurationCopy.excludedLog.values], 0) + .textField({rawtext: [ + {translate: "LumberAxe.log_exclude_manager.text_field"} + ]}, "myaddon:custom_log", preResultFlags[1] as string) + .toggle({rawtext: [ + {translate: "LumberAxe.log_exclude_manager.toggle"} + ]}, preResultFlags[2] as boolean); + form.show(this.player).then((response: ModalFormResponse) => { + if(!response.formValues) return; + if (response.canceled || response.cancelationReason === FormCancelationReason.UserClosed || response.cancelationReason === FormCancelationReason.UserBusy) return; + const hadChanges: boolean = !preResultFlags.every((element, index) => element === response.formValues[index]); + const selectedIndex: number = (response.formValues[0] as number); + let canUpdate: boolean = response.formValues[2] as boolean; + const dropDownContent: string = response.formValues[1] as string; + const dropDownSelected: number = (selectedIndex as number); + const isEmpty: boolean = !dropDownContent.length; + if (!hadChanges) return this.showServerScreen(); + if(isEmpty) canUpdate = false; + if(canUpdate){ + if(selectedIndex === 0) { + serverConfigurationCopy.excludedLog.values.push(dropDownContent); + SendMessageTo(this.player, {rawtext: [ + {text: `§aLumber Axe: `}, + {translate: "LumberAxe.log_exclude_manager.add_success_log", with: [dropDownContent]} + ]}); + } + else { + if(serverConfigurationCopy.excludedLog.values.length) { + const prevValue = serverConfigurationCopy.includedLog.values[dropDownSelected]; + serverConfigurationCopy.excludedLog.values[dropDownSelected] = dropDownContent; + SendMessageTo(this.player, {rawtext: [ + {text: `§eLumber Axe: `}, + {translate: "LumberAxe.log_exclude_manager.update_success_log", with: [prevValue, dropDownContent]} + ]}); + } + } + } else { + if(serverConfigurationCopy.excludedLog.values.length && selectedIndex !== 0) { + const itemDeleted = serverConfigurationCopy.excludedLog.values.splice(dropDownSelected, 1)[0]; + SendMessageTo(this.player, {rawtext: [ + {text: `§cLumber Axe: `}, + {translate: "LumberAxe.log_exclude_manager.remove_success_log", with: [itemDeleted]} + ]}); + } + } + this.saveServer(); + return this.showServerScreen(); + }); + } +} + + diff --git a/src/configuration/server_configuration.ts b/src/configuration/server_configuration.ts new file mode 100644 index 0000000..e67cd24 --- /dev/null +++ b/src/configuration/server_configuration.ts @@ -0,0 +1,45 @@ +import { FormBuilder } from "utils/form_builder"; +import { cloneConfiguration } from "./configuration_handler"; + +export const serverConfiguration = { + /** + * Lumber Axe durability damage per log destroyed. + */ + durabilityDamagePerBlock: new FormBuilder("LumberAxe.server.durability_damage_per_block").createTextField("3"), + /** + * Delay for immersive mode. + */ + immersiveModeDelay: new FormBuilder("LumberAxe.server.immersive_delay").createTextField("5"), + /** + * Tree chop limitation for control purposes. + */ + chopLimit: new FormBuilder("LumberAxe.server.chop_limit").createTextField("1000"), + /** + * Included blocks for custom logs, but any custom or vanilla logs also work as long as the block identifier ends with "*_log". + * Check: https://github.com/mcbe-mods/Cut-tree-one-click by Lete114. + */ + includedLog: new FormBuilder("LumberAxe.configuration.log_include_manager").createDropdown(['Empty'], "Empty"), + /** + * Excluded blocks for block logs you don't want to be included in being chopped. + * + * Tip: + * - excludedLog is prioritized over includedLog. + * - It's unnecessary to include log blocks that have "*_log" in their block id. + */ + excludedLog: new FormBuilder("LumberAxe.configuration.log_exclude_manager").createDropdown(['Empty'], "Empty"), + /** + * Enable/Disable Progressive Chopping, which makes you chop trees slightly longer, but nice to see. + */ + immersiveMode: new FormBuilder("LumberAxe.server.immersive_chopping").createToggle(false), + /** + * Enables debug messages to content logs. + */ + debug: new FormBuilder("Debug Mode").createToggle(false), +}; + +export let serverConfigurationCopy = cloneConfiguration(serverConfiguration); +export let setServerConfiguration = (newServerConfig) => serverConfigurationCopy = newServerConfig; +export let resetServerConfiguration = () => serverConfigurationCopy = cloneConfiguration(serverConfiguration); + +// version (do not change) +export const VERSION = "2.0.0"; \ No newline at end of file diff --git a/src/configuration_settings.json b/src/configuration_settings.json index e9d5c61..5ab0b65 100644 --- a/src/configuration_settings.json +++ b/src/configuration_settings.json @@ -1,27 +1,33 @@ { - "durabilityDamagePerBlock": { - "description": "Lumber Axe durability damage per log destroyed.", - "default": 3 - }, - - "chopLimit": { - "description": "1500 above is not recommended. It Does work but it's not recommended.", - "default": 300 - }, - - "includedLog": { - "description": "Included blocks for custom logs, but any custom or vanilla logs also work as long as the block identifier ends with \"*_log\".\nCheck: https://github.com/mcbe-mods/Cut-tree-one-click by Lete114.", - "default": [] - }, - - "excludedLog": { - "description": "Excluded blocks for block logs you don't want to be included in being chopped.\n\nTip:\n- excludedLog is prioritized over includedLog.\n- It's unnecessary to include log blocks that have \"*_log\" in their block id.", - "default": [] - }, - - "disableWatchDogTerminateLog": { - "description": "Disables the watchDogTerminate Log message. If true, it will only show a warning message when you enable content-ui log in the Minecraft settings.", - "default": true - } + "durabilityDamagePerBlock": { + "description": "Lumber Axe durability damage per log destroyed.", + "name": "LumberAxe.server.durability_damage_per_block", + "default": 3 + }, + "immersiveModeDelay": { + "description": "Delay for immersive mode.", + "name": "LumberAxe.server.immersive_delay", + "default": 5 + }, + "chopLimit": { + "description": "Tree chop limitation for control purposes.", + "name": "LumberAxe.server.chop_limit", + "default": 1000 + }, + "includedLog": { + "description": "Included blocks for custom logs, but any custom or vanilla logs also work as long as the block identifier ends with \"*_log\".\nCheck: https://github.com/mcbe-mods/Cut-tree-one-click by Lete114.", + "name": "LumberAxe.configuration.log_include_manager", + "default": ["Empty"] + }, + "excludedLog": { + "description": "Excluded blocks for block logs you don't want to be included in being chopped.\n\nTip:\n- excludedLog is prioritized over includedLog.\n- It's unnecessary to include log blocks that have \"*_log\" in their block id.", + "name": "LumberAxe.configuration.log_exclude_manager", + "default": ["Empty"] + }, + "immersiveMode": { + "description": "Enable/Disable Progressive Chopping, which makes you chop trees slightly longer, but nice to see.", + "name": "LumberAxe.server.immersive_chopping", + "default": false } +} \ No newline at end of file diff --git a/src/constant.ts b/src/constant.ts new file mode 100644 index 0000000..241f0a6 --- /dev/null +++ b/src/constant.ts @@ -0,0 +1,39 @@ +import { InteractedTreeResult } from "index"; +import { JsonDatabase } from "./utils/Database/con-database"; +import { MyCustomItemTypes } from 'items/CustomItemTypes'; +import { system } from "@minecraft/server"; + +export const ADDON_NAMESPACE: string = "yn" +export const ADDON_NAME: string = "Lumber_Axe"; +export const ADDON_IDENTIFIER: string = `${ADDON_NAMESPACE}:lumber`; +export let originalDatabase = new JsonDatabase(ADDON_NAME); +export const resetOriginalDatabase = (): void => { + originalDatabase = new JsonDatabase(ADDON_NAME); +} + +export const playerInteractedTimeLogMap: Map = new Map(); +export const axeEquipments: string[] = Object.values(MyCustomItemTypes); +export const visitedLogs: InteractedTreeResult[] = []; + + +/** + * + * @param player Player + * @param result Interacted Tree to despawn the block outlines later. + * @param instantDespawn To instantly remove the outlines without shifting the visitedLogs. + * @returns + */ +export function resetOutlinedTrees(result: InteractedTreeResult, instantDespawn: boolean = false) { + if(result.isDone) return; + result.isDone = true; + if(!instantDespawn) visitedLogs?.shift(); + const t = system.runJob((function*(){ + for(const blockOutline of result.visitedLogs.blockOutlines) { + if(blockOutline?.isValid()) { + blockOutline.triggerEvent('despawn'); + } + yield; + } + system.clearJob(t); + })()); +} \ No newline at end of file diff --git a/src/functions/tree_utils.ts b/src/functions/tree_utils.ts index 5c4c91a..ce1f8e0 100644 --- a/src/functions/tree_utils.ts +++ b/src/functions/tree_utils.ts @@ -1,163 +1,225 @@ -import { Block, Dimension, EntityEquippableComponent, EquipmentSlot, ItemDurabilityComponent, ItemEnchantableComponent, ItemLockMode, ItemStack, Player, System, Vector3, system } from "@minecraft/server"; -import { MinecraftBlockTypes, MinecraftEnchantmentTypes} from "../modules/vanilla-types/index"; - -import { validLogBlocks, axeEquipments, stackDistribution, durabilityDamagePerBlock, excludedLog, includedLog, chopLimit } from "../index"; - - -async function treeCut(player: Player, dimension: Dimension, location: Vector3, blockTypeId: string): Promise { - // Modified Version - // Author: Lete114 - // Project: https://github.com/mcbe-mods/Cut-tree-one-click - - //! Make Lumberjack (extends Player) Interface / class for this. - const equipment = player.getComponent(EntityEquippableComponent.componentId) as EntityEquippableComponent; - const currentHeldAxe = equipment.getEquipment(EquipmentSlot.Mainhand); - if (!axeEquipments.includes(currentHeldAxe?.typeId)) return; - if (!isLogIncluded(blockTypeId)) return; - - if (!player.isSurvival()) return; - if (player.isSurvival()) currentHeldAxe.lockMode = ItemLockMode.slot; - - //! MAKE THIS D-R-Y - const itemDurability: ItemDurabilityComponent = currentHeldAxe.getComponent(ItemDurabilityComponent.componentId) as ItemDurabilityComponent; - const enchantments: ItemEnchantableComponent = (currentHeldAxe.getComponent(ItemEnchantableComponent.componentId)); - const level: number = enchantments.getEnchantment(MinecraftEnchantmentTypes.Unbreaking)?.level | 0; - const unbreakingMultiplier: number = (100 / (level + 1)) / 100; - const unbreakingDamage: number = durabilityDamagePerBlock * unbreakingMultiplier; - - const visited: Set = await getTreeLogs(dimension, location, blockTypeId, (itemDurability.maxDurability - itemDurability.damage) / unbreakingDamage); - - const totalDamage: number = visited.size * unbreakingDamage; - const postDamagedDurability: number = itemDurability.damage + totalDamage; - - //! Put this to Durability interface - // Check if durabiliy is exact that can chop the tree but broke the axe, then broke it. - if (postDamagedDurability + 1 === itemDurability.maxDurability) { - equipment.setEquipment(EquipmentSlot.Mainhand, undefined); - // Check if the durability is not enough to chop the tree. Then don't apply the 3 damage. - } else if (postDamagedDurability > itemDurability.maxDurability) { - currentHeldAxe.lockMode = ItemLockMode.none; - return; - // Check if total durability will consume is still enough and not near the max durability - } else if (postDamagedDurability < itemDurability.maxDurability){ - itemDurability.damage = itemDurability.damage + totalDamage; - currentHeldAxe.lockMode = ItemLockMode.none; - equipment.setEquipment(EquipmentSlot.Mainhand, currentHeldAxe.clone()); - } - - //! IDK where to put this. - for (const group of groupAdjacentBlocks(visited)) { - const firstElement = JSON.parse(group[0]); - const lastElement = JSON.parse(group[group.length - 1]); - if (firstElement === lastElement) { - await new Promise((resolve) => { - dimension.getBlock(firstElement).setType(MinecraftBlockTypes.Air); - resolve(); - }); - continue; - } else { - await new Promise((resolve) => { - dimension.fillBlocks(firstElement, lastElement, MinecraftBlockTypes.Air); - resolve(); - }); - } - } - - system.runTimeout( async () => { - for (const group of stackDistribution(visited.size)) { - await new Promise((resolve) => { - dimension.spawnItem(new ItemStack(blockTypeId, group), location); - resolve(); - }); - } - }, 5); - -} +import { Block, Dimension, Entity, Vector3, VectorXZ, system, world } from "@minecraft/server"; + +import { serverConfigurationCopy, VisitedBlockResult, TrunkBlockResult, originalDatabase, hashBlock } from "../index"; +import { Graph } from "utils/graph"; +import { Vec3 } from "utils/VectorUtils"; -function isLogIncluded(blockTypeId: string): boolean { - if(excludedLog.includes(blockTypeId) || blockTypeId.includes('stripped_')) return false; - if(includedLog.includes(blockTypeId) || validLogBlocks.test(blockTypeId)) return true; +export function isLogIncluded(rootBlockTypeId: string, blockTypeId: string): boolean { + const validLogBlocks: RegExp = /(_log|_wood|crimson_stem|warped_stem|(?:brown|red_)?mushroom_block)$/; + function extractLogFamily(blockTypeId: string): string { + // Remove the namespace by splitting on the colon (':') and taking the second part + const [, cleanedBlockTypeId] = blockTypeId.split(':'); + + // Split the remaining string by underscores + const parts = cleanedBlockTypeId.split('_'); + + // Remove the last part (e.g., 'log', 'wood', 'stem') + return parts.slice(0, -1).join('_'); + } + if(serverConfigurationCopy.excludedLog.values.includes(blockTypeId) || blockTypeId.includes('stripped_')) return false; + const extractedLogFamily = extractLogFamily(rootBlockTypeId); + const blockFamily = extractLogFamily(blockTypeId); + const isSameFamily = blockFamily === extractedLogFamily; + if((serverConfigurationCopy.includedLog.values.includes(blockTypeId) || + validLogBlocks.test(blockTypeId)) && isSameFamily ) return true; return false; } -function getTreeLogs(dimension: Dimension, location: Vector3, blockTypeId: string, maxNeeded: number): Promise> { - // Modified Version - // Author: Lete114 - // Project: https://github.com/mcbe-mods/Cut-tree-one-click - return new Promise>((resolve) => { - const traversingTreeInterval: number = system.runInterval(() => { - const visited: Set = new Set(); - let queue: Block[] = getBlockNear(dimension, location); + +export async function getTreeLogs( + dimension: Dimension, + location: Vector3, + blockTypeId: string, + maxNeeded: number, + isInspectingTree: boolean = true +): Promise { + const firstBlock = dimension.getBlock(location); + const visitedTree = await new Promise((resolve) => { + const graph = new Graph(); + const visitedTypeIDs: Map = new Map(); + const queue: Block[] = [firstBlock]; + const yOffsets: Map = new Map(); + const visited: Set = new Set([JSON.stringify(firstBlock.location)]); + visitedTypeIDs.set(blockTypeId, 0); + const traversingTreeInterval: number = system.runJob(function* () { + graph.addNode(firstBlock); + originalDatabase.set(`visited_${hashBlock(firstBlock)}`, isInspectingTree); + + // Should spawn outline is indicator for inspection or breaking tree. + // Inspection = True + // Breaking = False + while (queue.length > 0) { - if(visited.size >= chopLimit || visited.size >= maxNeeded) { - system.clearRun(traversingTreeInterval); - resolve(visited); + const size = graph.getSize(); + + // Check termination conditions + if (size >= parseInt(serverConfigurationCopy.chopLimit.defaultValue + "") || size >= maxNeeded) { + break; + } + + const block: Block = queue.shift(); + const mainNode = graph.getNode(block); + if (!mainNode) continue; + yOffsets.set(block.y, false); + + // First, gather all valid neighbors + for (const neighborBlock of getBlockNear(blockTypeId, block)) { + const serializedLocation = JSON.stringify(neighborBlock.location); + let neighborNode = graph.getNode(neighborBlock) ?? graph.addNode(neighborBlock); + originalDatabase.set(`visited_${hashBlock(neighborBlock)}`, isInspectingTree); + + // It should check if this neighbor of main node is already a neighbor, if yes, then continue. + if(mainNode.neighbors.has(neighborNode)) continue; + + // Connect the current node to its neighbor + mainNode.addNeighbor(neighborNode); + neighborNode.addNeighbor(mainNode); + + // Check if the neighbor node has already been visited + if (visited.has(serializedLocation)) continue; + + // Mark this neighbor as visited and add to the queue for further processing + visited.add(serializedLocation); + queue.push(neighborBlock); + + let currentAmount = visitedTypeIDs.get(neighborBlock.typeId) ?? 0; + currentAmount += 1; + visitedTypeIDs.set(neighborBlock.typeId, currentAmount); + yield; + } + yield; + } + + system.clearJob(traversingTreeInterval); + resolve({ + source: graph, + blockOutlines: [], + yOffsets, + typeIds: visitedTypeIDs, + trunk: { + size: 0, + center: {x: 0, z: 0} + } + }); + }()); + }); + + const blockOutlines: Entity[] = []; + const trunk = await getTreeTrunkSize(firstBlock, blockTypeId); + return new Promise((resolve) => { + const t = system.runJob((function*(){ + // Create Block Entity based on the trunk. + for(const yOffset of visitedTree.yOffsets.keys()) { + const outline = dimension.spawnEntity('yn:block_outline', { + x: trunk.center.x, + y: yOffset, + z: trunk.center.z + }); + outline.lastLocation = JSON.parse(JSON.stringify(outline.location)); + blockOutlines.push(outline); + yield; + } + // After all is traversed, start timer. + for(const blockOutline of blockOutlines) { + if(blockOutline?.isValid()) { + blockOutline.triggerEvent('not_persistent'); } - const _block: Block = queue.shift(); - if (!_block || !isLogIncluded(_block?.typeId)) continue; - if (_block.typeId !== blockTypeId) continue; - const pos: string = JSON.stringify(_block.location); - if (visited.has(pos)) continue; - visited.add(pos); - queue.push(...getBlockNear(dimension, _block.location)); + yield; } - queue = []; - system.clearRun(traversingTreeInterval); - resolve(visited); - }, 1); + system.clearJob(t); + resolve({ + typeIds: visitedTree.typeIds, + source: visitedTree.source, + blockOutlines: blockOutlines, + trunk: trunk, + yOffsets: visitedTree.yOffsets + }); + return; + })()); }); } -function getBlockNear(dimension: Dimension, location: Vector3, radius: number = 1): Block[] { - // Modified Version - // Author: Lete114 - // Project: https://github.com/mcbe-mods/Cut-tree-one-click - const centerX: number = location.x; - const centerY: number = location.y; - const centerZ: number = location.z; - const positions: Block[] = []; + + +function* getBlockNear(initialBlockTypeID: string, initialBlock: Block, radius: number = 1): Generator { + const centerX: number = 0; + const centerY: number = 0; + const centerZ: number = 0; let _block: Block; for (let x = centerX - radius; x <= centerX + radius; x++) { for (let y = centerY - radius; y <= centerY + radius; y++) { for (let z = centerZ - radius; z <= centerZ + radius; z++) { - if(centerX === x && centerY === y && centerZ === z) continue; - _block = dimension.getBlock({ x, y, z }); - if(_block.isAir) continue; - positions.push(_block); + if (centerX === x && centerY === y && centerZ === z) continue; + _block = initialBlock.offset({x, y, z}); + if (!_block?.isValid() || !isLogIncluded(initialBlockTypeID, _block?.typeId)) continue; + yield _block; } } } - return positions; } -// Gets all the visited blocks and groups them together. -function groupAdjacentBlocks(visited: Set): string[][] { - // Author: Adr-hyng - // Project: https://github.com/Adr-hyng-OSS/Lumber-Axe - // Convert Set to Array and parse each string to JSON object - const array = Array.from(visited).map(item => JSON.parse(item)); - - // Sort the array based on "x", "z", and "y" - array.sort((a, b) => a.x - b.x || a.z - b.z || a.y - b.y); - - const groups: string[][] = []; - let currentGroup: string[] = []; - - for (let i = 0; i < array.length; i++) { - // If it's the first element or "x" and "z" didn't change and "y" difference is less or equal to 2, add it to the current group - if (i === 0 || (array[i].x === array[i - 1].x && array[i].z === array[i - 1].z && Math.abs(array[i].y - JSON.parse(currentGroup[currentGroup.length - 1]).y) <= 2)) { - currentGroup.push(JSON.stringify(array[i])); - } else { - // Otherwise, add the current group to the groups array and start a new group - groups.push(currentGroup); - currentGroup = [JSON.stringify(array[i])]; - } - } - // Add the last group to the groups array - if (currentGroup.length > 0) { - groups.push(currentGroup); - } - return groups; -} +export function getTreeTrunkSize(blockInteracted: Block, blockTypeId: string): Promise { + return new Promise((fetchedTrunkSizeResolved) => { + let i = 0; + let centroidLog: VectorXZ = { + x: 0, + z: 0 + }; + -export {treeCut, isLogIncluded, getTreeLogs} \ No newline at end of file + // (TODO) Possible to accurately get the trunk size, use a hashset to collect X and Z axis, + // (TODO) Possible to accurately get the trunk height + const visited = new Set(); // To avoid revisiting blocks + const queue: Block[] = [blockInteracted]; // Queue for the floodfill process + const originalY = blockInteracted.y; // Store the original Y position + + const t = system.runJob((function* () { + while (queue.length > 0) { + const currentBlock = queue.shift(); + if ((!currentBlock || !currentBlock.isValid()) && !Vec3.equals(blockInteracted, currentBlock)) continue; + const blockKey = JSON.stringify({x: currentBlock.x, z: currentBlock.z} as VectorXZ); + if (visited.has(blockKey)) continue; + visited.add(blockKey); + + // Accumulate the log coordinates to calculate the centroid + centroidLog.x += currentBlock.x; + centroidLog.z += currentBlock.z; + i++; + + // Add the neighboring blocks within radius 1 (cardinal + diagonal) but limit Y within +2 and -2 range + for (let y = -1; y <= 1; y++) { + const newY = currentBlock.y + y; + if (newY < originalY - 2 || newY > originalY + 2) continue; // Skip if out of allowed y-range + + for (let x = -1; x <= 1; x++) { + for (let z = -1; z <= 1; z++) { + if (x === 0 && z === 0 && y === 0) continue; // Skip the current block itself + const neighborBlock = currentBlock.offset({ x: x, y: y, z: z }); + const neighborLoc = JSON.stringify({x: neighborBlock.x, z: neighborBlock.z} as VectorXZ); + if (!neighborBlock?.isValid() || visited.has(neighborLoc) || !isLogIncluded(blockTypeId, neighborBlock.typeId)) continue; + queue.push(neighborBlock); + yield; + } + yield; + } + yield; + } + yield; + } + + // If only one block found, the centroid is the original block's location + if (i <= 1) { + i = 1; + centroidLog = blockInteracted.center(); + } else { + // Compute the average position of the logs + centroidLog.x = (centroidLog.x / i) + 0.5; + centroidLog.z = (centroidLog.z / i) + 0.5; + } + + system.clearJob(t); + fetchedTrunkSizeResolved({ center: centroidLog, size: i }); + return; + })()); + }); +} diff --git a/src/functions/utils.ts b/src/functions/utils.ts deleted file mode 100644 index 08c3f27..0000000 --- a/src/functions/utils.ts +++ /dev/null @@ -1,35 +0,0 @@ -import { Player, system } from "@minecraft/server"; -import { ActionFormData, ActionFormResponse, FormCancelationReason } from "@minecraft/server-ui"; - -// Calculates the amount of items to be dropped in each stack. O(1) -function stackDistribution(number: number, groupSize: number = 64): number[] { - // Author: Adr-hyng - // Project: https://github.com/Adr-hyng-OSS/Lumber-Axe - const fullGroupsCount = Math.floor(number / groupSize); - const remainder = number % groupSize; - // Create an array with the size of each full group - const groups = new Array(fullGroupsCount).fill(groupSize); - // If there's a remainder, add it as the last group - if (remainder > 0) { - groups.push(remainder); - } - - return groups; -} - -async function forceShow(player: Player, form: ActionFormData, timeout: number = Infinity): Promise { - // Script example for ScriptAPI - // Author: Jayly#1397 - // Worldwidebrine#9037 - // Project: https://github.com/JaylyDev/ScriptAPI - const startTick: number = system.currentTick; - while ((system.currentTick - startTick) < timeout) { - const response: ActionFormResponse = await (form.show(player)).catch(er=>console.error(er,er.stack)) as ActionFormResponse; - if (response.cancelationReason !== FormCancelationReason.UserBusy) { - return response; - } - }; - throw new Error(`Timed out after ${timeout} ticks`); -} - -export {stackDistribution, forceShow} \ No newline at end of file diff --git a/src/index.ts b/src/index.ts index 5f5dbeb..1bde9ec 100644 --- a/src/index.ts +++ b/src/index.ts @@ -1,11 +1,32 @@ -export * from './functions/utils'; +import { Entity, VectorXZ } from '@minecraft/server'; +import { Graph } from 'utils/graph'; + +export * from './utils/utilities'; export * from './functions/tree_utils'; export * from './classes/player'; +export * from './classes/entity_override'; +export * from './classes/item_equippable'; + +export * from "configuration/server_configuration"; +export * from "constant"; +export * from "items/axes"; -import Configuration from "./config"; -const { durabilityDamagePerBlock, chopLimit, includedLog, excludedLog, disableWatchDogTerminateLog } = Configuration; -export { durabilityDamagePerBlock, chopLimit, includedLog, excludedLog, disableWatchDogTerminateLog}; +export type TrunkBlockResult = { + size: number; + center: VectorXZ; +} -export const validLogBlocks: RegExp = /(_log|crimson_stem|warped_stem)$/; +export type VisitedBlockResult = { + source: Graph; + blockOutlines: Entity[]; + yOffsets: Map; + trunk: TrunkBlockResult; + typeIds: Map; +} -export const axeEquipments: string[] = [ "yn:wooden_lumber_axe", "yn:stone_lumber_axe", "yn:iron_lumber_axe", "yn:diamond_lumber_axe", "yn:golden_lumber_axe", "yn:netherite_lumber_axe" ]; \ No newline at end of file +export type InteractedTreeResult = { + initialSize: number; + visitedLogs: VisitedBlockResult; + isDone: boolean; + isBeingChopped: boolean; +} diff --git a/src/items/CustomItemTypes.ts b/src/items/CustomItemTypes.ts new file mode 100644 index 0000000..8caa9a0 --- /dev/null +++ b/src/items/CustomItemTypes.ts @@ -0,0 +1,9 @@ +export enum MyCustomItemTypes { + WoodenLumberAxe = "yn:wooden_lumber_axe", + StoneLumberAxe = "yn:stone_lumber_axe", + IronLumberAxe = "yn:iron_lumber_axe", + DiamondLumberAxe = "yn:diamond_lumber_axe", + GoldenLumberAxe = "yn:golden_lumber_axe", + NetheriteLumberAxe = "yn:netherite_lumber_axe", +} + diff --git a/src/items/axes.ts b/src/items/axes.ts new file mode 100644 index 0000000..614f414 --- /dev/null +++ b/src/items/axes.ts @@ -0,0 +1,23 @@ +import { EntityEquippableComponent, Player, world } from "@minecraft/server"; +import "classes/player"; +import { isLogIncluded } from "functions/tree_utils"; + +world.beforeEvents.worldInitialize.subscribe((registry) => { + registry.itemComponentRegistry.registerCustomComponent('yn:tool_durability', { + onHitEntity(arg) { + if(!(arg.attackingEntity instanceof Player)) return; + const player: Player = arg.attackingEntity; + if(!player.isSurvival()) return; + const axe = (player.getComponent(EntityEquippableComponent.componentId) as EntityEquippableComponent); + axe.damageDurability(1); + }, + onUseOn(arg) { + + }, + onMineBlock(arg) { + const player = arg.source; + const axe = (player.getComponent(EntityEquippableComponent.componentId) as EntityEquippableComponent); + axe.damageDurability(2); + }, + }) +}); \ No newline at end of file diff --git a/src/main.ts b/src/main.ts index 86b2849..33028ae 100644 --- a/src/main.ts +++ b/src/main.ts @@ -1,123 +1,637 @@ -import { world, ItemStack, system, Block, BlockPermutation, Player, ItemDurabilityComponent, ItemEnchantableComponent, ItemUseOnBeforeEvent, WatchdogTerminateBeforeEvent, WatchdogTerminateReason, PlayerLeaveAfterEvent, PlayerBreakBlockAfterEvent, EnchantmentType, EnchantmentTypes } from '@minecraft/server'; -import { FormCancelationReason, ActionFormData, ActionFormResponse} from "@minecraft/server-ui"; -import { disableWatchDogTerminateLog, durabilityDamagePerBlock ,axeEquipments, forceShow, getTreeLogs, isLogIncluded, treeCut} from "./index" -import { MinecraftEnchantmentTypes } from './modules/vanilla-types/index'; +import { world, ScriptEventCommandMessageAfterEvent, system, ScriptEventSource, Player, BlockPermutation, EntityEquippableComponent, EntityInventoryComponent, ItemDurabilityComponent, ItemEnchantableComponent, ItemLockMode, ItemStack, MolangVariableMap, Block, TicksPerSecond, Entity, ItemCooldownComponent } from '@minecraft/server'; +import { ADDON_IDENTIFIER, axeEquipments, originalDatabase, forceShow, getTreeLogs, getTreeTrunkSize, hashBlock, InteractedTreeResult, isLogIncluded, playerInteractedTimeLogMap, resetOutlinedTrees, SendMessageTo, serverConfigurationCopy, stackDistribution, VisitedBlockResult, visitedLogs} from "./index" +import { Logger } from 'utils/logger'; +import './items/axes'; +import { MinecraftEnchantmentTypes, MinecraftBlockTypes } from 'modules/vanilla-types/index'; +import { Graph } from 'utils/graph'; +import { Vec3 } from 'utils/VectorUtils'; +import { ActionFormData, ActionFormResponse, FormCancelationReason } from '@minecraft/server-ui'; -const logMap: Map = new Map(); -const playerInteractionMap: Map = new Map(); +const BLOCK_OUTLINES_DESPAWN_TIMER = 5; -system.beforeEvents.watchdogTerminate.subscribe((e: WatchdogTerminateBeforeEvent) => { - e.cancel = true; - if(e.terminateReason === WatchdogTerminateReason.Hang){ - for(const key of playerInteractionMap.keys()) { - playerInteractionMap.set(key, false); +world.afterEvents.playerSpawn.subscribe((e) => { + if(!e.initialSpawn) return; + e.player.configuration.loadServer(); + if(!originalDatabase.has(`playerFirstJoined-${e.player.id}`)) { + originalDatabase.set(`playerFirstJoined-${e.player.id}`, false); + } + if(!originalDatabase.get(`playerFirstJoined-${e.player.id}`)) { + originalDatabase.set(`playerFirstJoined-${e.player.id}`, true); + SendMessageTo(e.player, { + rawtext: [ + { + translate: "LumberAxe.on_load_message" } - if(!disableWatchDogTerminateLog) world.sendMessage({ - rawtext: [ - { - translate: "LumberAxe.watchdogError.hang.text" - } - ]}); - if(disableWatchDogTerminateLog) console.warn(`Scripting Error: Try chopping or inspecting smaller trees or different angle.`); - } - console.warn(`Watchdog Error: ${(e.terminateReason as WatchdogTerminateReason)}`) -}); - -world.afterEvents.playerLeave.subscribe((e: PlayerLeaveAfterEvent) => { - playerInteractionMap.set(e.playerId, false); -}); - -world.afterEvents.playerBreakBlock.subscribe(async (e: PlayerBreakBlockAfterEvent) => { - const { dimension, player, block } = e; - const currentBreakBlock: BlockPermutation = e.brokenBlockPermutation; - const blockTypeId: string = currentBreakBlock.type.id; - treeCut(player, dimension, block.location, blockTypeId); + ] + }); + } }); + +world.beforeEvents.playerBreakBlock.subscribe((arg) => { + const player: Player = arg.player; + const dimension = player.dimension; + const blockInteracted = arg.block; + const location = blockInteracted.location; + const currentHeldAxe = arg.itemStack; + const currentHeldAxeSlot = player.selectedSlotIndex; + const currentBreakBlock: BlockPermutation = arg.block.permutation; + const blockTypeId: string = currentBreakBlock.type.id; + if(!player.isSurvival()) return; + if (!isLogIncluded(blockTypeId, blockTypeId)) return; + if(originalDatabase.has(`visited_${hashBlock(blockInteracted)}`) && !originalDatabase.get(`visited_${hashBlock(blockInteracted)}`)) { + arg.cancel = true; + return; + } + if(!axeEquipments.includes(currentHeldAxe.typeId)) return; + // /execute positioned ~~~ run fill ~1 ~ ~1 ~-1 ~20 ~-1 jungle_log -world.beforeEvents.itemUseOn.subscribe((e: ItemUseOnBeforeEvent) => { - const currentHeldAxe: ItemStack = e.itemStack; - const blockInteracted: Block = e.block; //! NEEDED - const player: Player = e.source as Player; //! NEEDED - - const oldLog: number = logMap.get(player.name); - logMap.set(player.name, Date.now()); - if ((oldLog + 1_000) >= Date.now()) return; - if (!axeEquipments.includes(currentHeldAxe.typeId) || !isLogIncluded(blockInteracted.typeId)) return; - if(playerInteractionMap.get(player.id)) return; - playerInteractionMap.set(player.id, true); + // Getting the cache, if it has, to remove the particle. + // Filter by getting the graph that has this node. + const possibleVisitedLogs: {result: InteractedTreeResult, index: number}[] = []; + for(let i = 0; i < visitedLogs.length; i++) { + const currentInspectedTree = visitedLogs[i]; + const interactedTreeNode = currentInspectedTree.visitedLogs.source.getNode(blockInteracted); + if(interactedTreeNode) { + possibleVisitedLogs.push({result: currentInspectedTree, index: i}); + } + } - //! MAKE THIS D-R-Y + let initialTreeInspection: InteractedTreeResult; + if(possibleVisitedLogs.length) { + // After filtering check get that tree that this player has inspected, get the latest one. + const latestPossibleInspectedTree = possibleVisitedLogs[possibleVisitedLogs.length - 1]; + const index = latestPossibleInspectedTree.index; + initialTreeInspection = latestPossibleInspectedTree.result; + if(initialTreeInspection.isBeingChopped) { + arg.cancel = true; + return; + } + visitedLogs!.splice(index, 1); + initialTreeInspection!.isDone = true; + } + player.configuration.loadServer(); + system.run(async () => { + currentHeldAxe.lockMode = ItemLockMode.slot; + const inventory = (player.getComponent(EntityInventoryComponent.componentId) as EntityInventoryComponent).container; + inventory.setItem(currentHeldAxeSlot, currentHeldAxe); const itemDurability: ItemDurabilityComponent = currentHeldAxe.getComponent(ItemDurabilityComponent.componentId) as ItemDurabilityComponent; const enchantments: ItemEnchantableComponent = (currentHeldAxe.getComponent(ItemEnchantableComponent.componentId) as ItemEnchantableComponent); const level: number = enchantments.getEnchantment(MinecraftEnchantmentTypes.Unbreaking)?.level | 0; - const currentDurability = itemDurability.damage; - const maxDurability = itemDurability.maxDurability; const unbreakingMultiplier: number = (100 / (level + 1)) / 100; - const unbreakingDamage: number = durabilityDamagePerBlock * unbreakingMultiplier; - const reachableLogs = (maxDurability - currentDurability) / unbreakingDamage; - getTreeLogs(player.dimension, blockInteracted.location, blockInteracted.typeId, reachableLogs + 1).then( (treeCollected: Set) => { - const totalDamage: number = (treeCollected.size) * unbreakingDamage; - const totalDurabilityConsumed: number = currentDurability + totalDamage; - const canBeChopped: boolean = (totalDurabilityConsumed === maxDurability) || (totalDurabilityConsumed < maxDurability); - - const inspectionForm: ActionFormData = new ActionFormData() - .title({ - rawtext: [ - { - translate: "LumberAxe.form.title.text" + const unbreakingDamage: number = +serverConfigurationCopy.durabilityDamagePerBlock.defaultValue * unbreakingMultiplier; + let visited: Graph; + + // This should be the temporary container where it doesn't copy the reference from the original player's visitedNodes. + let destroyedTree :InteractedTreeResult = { + isBeingChopped: true, + initialSize: 0, + isDone: false, + visitedLogs: { + typeIds: new Map(), + blockOutlines: [], + source: new Graph(), + yOffsets: new Map(), + trunk: { + center: { + x: 0, + z: 0 + }, + size: 0 + } + } + }; + const molang = new MolangVariableMap(); + let isTreeDoneTraversing = false; + const brokenTreeTrunk = await getTreeTrunkSize(blockInteracted, blockTypeId); + const topMostBlock = blockInteracted.dimension.getTopmostBlock(brokenTreeTrunk.center); + const bottomMostBlock = await new Promise((getBottomMostBlockResolved) => { + let _bottom = blockInteracted.below(); + const _t = system.runInterval(() => { + if(!isLogIncluded(blockInteracted.typeId, _bottom.typeId)) { + system.clearRun(_t); + getBottomMostBlockResolved(_bottom); + return; + } + _bottom = _bottom.below(); + }); + }); + const mainTreeTrunkHeight = (topMostBlock.y - bottomMostBlock.y); + const isValidVerticalTree = mainTreeTrunkHeight > 2; + + if(isValidVerticalTree) { + let dustRadius = 1; + molang.setFloat('trunk_size', dustRadius); + player.playSound('hit.stem'); + dimension.spawnParticle('yn:tree_dust', {x: brokenTreeTrunk.center.x, y: blockInteracted.y, z: brokenTreeTrunk.center.z}, molang); + const t = system.runInterval(() => { + // Get the first block, and based on that it will get the height. + molang.setFloat('trunk_size', dustRadius += 0.25); + if(isTreeDoneTraversing) { + system.clearRun(t); + return; + }; + player.playSound('hit.stem'); + dimension.spawnParticle('yn:tree_dust', {x: brokenTreeTrunk.center.x, y: blockInteracted.y, z: brokenTreeTrunk.center.z}, molang); + }, 12); + } + const choppedTree = initialTreeInspection === undefined ? (await getTreeLogs( + dimension, location, blockTypeId, + (itemDurability.maxDurability - itemDurability.damage) / unbreakingDamage, + false + ) as VisitedBlockResult) : initialTreeInspection.visitedLogs; + isTreeDoneTraversing = true; + destroyedTree.visitedLogs = choppedTree; + visited = choppedTree.source; + const initialSize = visited.getSize() - 1; + visitedLogs.push(destroyedTree); + + if(!visited) return; + if(initialSize >= +serverConfigurationCopy.chopLimit.defaultValue) { + currentHeldAxe.lockMode = ItemLockMode.none; + inventory.setItem(currentHeldAxeSlot, currentHeldAxe); + SendMessageTo(player, {rawtext: [{translate: `LumberAxe.server.invalid_log_amount_limitation`, with: [serverConfigurationCopy.chopLimit.defaultValue]}]}); + return await new Promise((resolve) => { + system.runJob((function*() { + for (const node of destroyedTree.visitedLogs.source.traverseIterative(blockInteracted, "BFS")) { + yield; + if (!node) continue; + // Reset the temporary permutation for block being destroyed. + originalDatabase.delete(`visited_${hashBlock(node.block)}`); + yield; + } + resetOutlinedTrees(destroyedTree); + resolve(); + return; + })()); + }); + } + + const totalDamage: number = initialSize * unbreakingDamage; + const postDamagedDurability: number = itemDurability.damage + totalDamage; + if (postDamagedDurability + 1 === itemDurability.maxDurability) { + player.playSound("random.break"); + inventory.setItem(currentHeldAxeSlot, undefined); + } else if (postDamagedDurability > itemDurability.maxDurability) { + currentHeldAxe.lockMode = ItemLockMode.none; + inventory.setItem(currentHeldAxeSlot, currentHeldAxe); + return; + } else if (postDamagedDurability < itemDurability.maxDurability) { + itemDurability.damage = itemDurability.damage + totalDamage; + const heldTemp = currentHeldAxe.clone(); + heldTemp.lockMode = ItemLockMode.none; + inventory.setItem(currentHeldAxeSlot, heldTemp); + } + + // Dust Particle (VFX) + const treeDustParseMap = { + 0: 1, // 3.25 + 1: 3.25, // 3.25 + 2: 4, + 3: 4, + 4: 4, + 5: 7, + 6: 7, + 7: 7, + 8: 7, + 9: 7 + } + const trunkYCoordinates = Array.from(destroyedTree.visitedLogs.yOffsets.keys()).sort((a, b) => a - b); + const getTreeDustValue = (key: number) => key > 9 ? 7 : treeDustParseMap[key]; + molang.setFloat('trunk_size', getTreeDustValue(brokenTreeTrunk.size)); + let currentBlockOffset = 0; + + // Currently, it doesn't spawn destroy particle with redwood tree in expansive biomes + if(serverConfigurationCopy.immersiveMode.defaultValue && isValidVerticalTree){ + for(const yOffset of trunkYCoordinates) { + if(currentBlockOffset % 2 === 0) { + await system.waitTicks(+serverConfigurationCopy.immersiveModeDelay.defaultValue); + const loc = {x: destroyedTree.visitedLogs.trunk.center.x, y: yOffset, z: destroyedTree.visitedLogs.trunk.center.z}; + player.playSound('mob.irongolem.crack', {location: loc}); + const molang = new MolangVariableMap(); + molang.setFloat('trunk_size', getTreeDustValue(destroyedTree.visitedLogs.trunk.size)); + dimension.spawnParticle('yn:tree_dust', loc, molang); + } + destroyedTree.visitedLogs.yOffsets.set(yOffset, true); + currentBlockOffset++; + } + } + // /execute positioned -14462 84 11333 run fill ~1 ~ ~1 ~-1 ~10 ~-1 oak_log + let size = 0; + system.runJob( (function* () { + // Dust + if(!(serverConfigurationCopy.immersiveMode.defaultValue) && isValidVerticalTree) { + for(const yOffset of trunkYCoordinates) { + if(currentBlockOffset % 2 === 0) { + const molang = new MolangVariableMap(); + molang.setFloat('trunk_size', getTreeDustValue(destroyedTree.visitedLogs.trunk.size)); + dimension.spawnParticle('yn:tree_dust', {x: destroyedTree.visitedLogs.trunk.center.x, y: yOffset, z: destroyedTree.visitedLogs.trunk.center.z}, molang); + } + currentBlockOffset++; + yield; + } + } + + // Destroy particle + const blockOutlineIterator = destroyedTree.visitedLogs.blockOutlines[Symbol.iterator](); + let blockOutlineIterResult = blockOutlineIterator.next(); + while(!blockOutlineIterResult.done) { + const blockOutline: Entity = blockOutlineIterResult.value; + if(blockOutline?.isValid()) { + blockOutline.setProperty('yn:trunk_size', destroyedTree.visitedLogs.trunk.size); + } + blockOutlineIterResult = blockOutlineIterator.next(); + yield; + } + for (const node of destroyedTree.visitedLogs.source.traverseIterative(blockInteracted, "BFS")) { + yield; + if(Vec3.equals(node.block, blockInteracted.location)) continue; + if (!node) continue; + + // If there's setDestroy that cancels the dropped item, just use that instead of this. + // Custom Destroy Particle + if(isLogIncluded(blockTypeId, node.block.typeId)) { + size++; + system.waitTicks(5).then(() => { + dimension.setBlockType(node.block.location, MinecraftBlockTypes.Air); + }); + } else { + destroyedTree.visitedLogs.source.removeNode(node.block); + break; + } + yield; + } + player.playSound('dig.cave_vines'); + + // For each typeIDs create stackDistribution + for(const [typeIDs, typeIDSize] of choppedTree.typeIds.entries()) { + for (const stackedAmount of stackDistribution(typeIDSize)) { + dimension.spawnItem(new ItemStack(typeIDs, stackedAmount), location); + yield; + } + yield; + } + return; + })()); + await system.waitTicks(3); + system.runJob((function*() { + for (const node of destroyedTree.visitedLogs.source.traverseIterative(blockInteracted, "BFS")) { + // Reset the temporary permutation for block being destroyed. + if (node) { + originalDatabase.delete(`visited_${hashBlock(node.block)}`); + } + yield; + } + if(!destroyedTree?.isDone) resetOutlinedTrees(destroyedTree); + return; + })()); + }); +}); + +world.beforeEvents.itemUseOn.subscribe(async (arg) => { + const currentHeldAxe: ItemStack = arg.itemStack; + const blockInteracted: Block = arg.block; + const player: Player = arg.source as Player; + + if (!axeEquipments.includes(currentHeldAxe.typeId) || !isLogIncluded(blockInteracted.typeId, blockInteracted.typeId)) return; + const oldLog = playerInteractedTimeLogMap.get(player.id); + playerInteractedTimeLogMap.set(player.id, system.currentTick); + if ((oldLog + 10) >= system.currentTick) return; + player.configuration.loadServer(); + const itemDurability: ItemDurabilityComponent = currentHeldAxe.getComponent(ItemDurabilityComponent.componentId) as ItemDurabilityComponent; + const enchantments: ItemEnchantableComponent = (currentHeldAxe.getComponent(ItemEnchantableComponent.componentId) as ItemEnchantableComponent); + const level: number = enchantments.getEnchantment(MinecraftEnchantmentTypes.Unbreaking)?.level | 0; + const currentDurability = itemDurability.damage; + const maxDurability = itemDurability.maxDurability; + const unbreakingMultiplier: number = (100 / (level + 1)) / 100; + const unbreakingDamage: number = +serverConfigurationCopy.durabilityDamagePerBlock.defaultValue * unbreakingMultiplier; + const reachableLogs = (maxDurability - currentDurability) / unbreakingDamage; + + const cooldown = (currentHeldAxe.getComponent(ItemCooldownComponent.componentId)); + + let BLOCK_OUTLINES_DESPAWN_CD = BLOCK_OUTLINES_DESPAWN_TIMER * TicksPerSecond; + try { + // Check also, if this tree is already being interacted. By checking this current blockOutline (node), if it's being interacted. + if(!visitedLogs) return; + const tempResult = await new Promise<{result: VisitedBlockResult, index: number}>((inspectTreePromiseResolve) => { + const tMain = system.runJob((function*(inspectTreePromiseResolve: (inspectedTreeResult: {result: VisitedBlockResult, index: number} | PromiseLike<{result: VisitedBlockResult, index: number}>) => void){ + // Filter by getting the graph that has this node. + const possibleVisitedLogs: {result: InteractedTreeResult, index: number}[] = []; + for(let i = 0; i < visitedLogs.length; i++) { + const currentInspectedTree = visitedLogs[i]; + const interactedTreeNode = currentInspectedTree.visitedLogs.source.getNode(blockInteracted); + if(interactedTreeNode) { + possibleVisitedLogs.push({result: currentInspectedTree, index: i}); + } + } + + if(!possibleVisitedLogs.length) { + if(originalDatabase.has(`visited_${hashBlock(blockInteracted)}`)) { + inspectTreePromiseResolve({result: null, index: -100}); + return system.clearJob(tMain); + } + inspectTreePromiseResolve({result: null, index: -1}); + return system.clearJob(tMain); + } + + // After filtering check get that tree that this player has inspected, get the latest one. + const latestPossibleInspectedTree = possibleVisitedLogs[possibleVisitedLogs.length - 1]; + const index = latestPossibleInspectedTree.index; + const initialTreeInspection = latestPossibleInspectedTree.result; + + if(initialTreeInspection.isBeingChopped) { + inspectTreePromiseResolve({result: null, index: -100}); + return system.clearJob(tMain); + } + + // Remove some nodes in the graph that is not existing anymore. So, it can update its branches or neighbors + for(const node of initialTreeInspection.visitedLogs.source.traverseIterative(blockInteracted, "BFS")) { + if(!node.block?.isValid() || !isLogIncluded(blockInteracted.typeId, node.block.typeId)) { + initialTreeInspection.visitedLogs.source.removeNode(node.block); + originalDatabase.delete(`visited_${hashBlock(node.block)}`); + } + yield; + } + + if(initialTreeInspection.initialSize === initialTreeInspection.visitedLogs.source.getSize()) { + system.clearJob(tMain); + inspectTreePromiseResolve({result: initialTreeInspection.visitedLogs, index: index}); + } + + const finalizedTreeInspection: VisitedBlockResult = { + blockOutlines: [], + typeIds: new Map(), + source: new Graph(), + yOffsets: new Map(), + trunk: { + center: { + x: 0, + z: 0 + }, + size: 0 + } + }; + + // Traverse the interacted block to validate the remaining nodes, if something was removed. O(n) + for(const node of initialTreeInspection.visitedLogs.source.traverseIterative(blockInteracted, "BFS")) { + if(node.block?.isValid()) { + finalizedTreeInspection.blockOutlines.push(initialTreeInspection.visitedLogs.blockOutlines[node.index]); + finalizedTreeInspection.source.addNode(node); + finalizedTreeInspection.yOffsets.set(node.block.location.y, false); + } + yield; + } + + // Just appending the sub-tree as a separate tree. + const newInspectedSubTree: InteractedTreeResult = { + isBeingChopped: false, + initialSize: finalizedTreeInspection.source.getSize(), + isDone: false, + visitedLogs: finalizedTreeInspection + }; + // if this newly inspected tree is just the main inspected tree, then just update, else add this new result, since it has changed. + const currentChangedIndex = visitedLogs.findIndex((result) => newInspectedSubTree.visitedLogs.source.isEqual(initialTreeInspection.visitedLogs.source) && !result.isDone); + if(currentChangedIndex === -1) { + if(newInspectedSubTree.initialSize > 0) visitedLogs.push(newInspectedSubTree); + system.waitTicks(BLOCK_OUTLINES_DESPAWN_CD).then(async (_) => { + system.runJob((function*() { + for(const node of newInspectedSubTree.visitedLogs.source.traverseIterative(blockInteracted, "BFS")) { + originalDatabase.delete(`visited_${hashBlock(node.block)}`); + yield; } - ]}) - .button( - { - rawtext: [ - { - translate: `LumberAxe.form.treeSizeAbrev.text` - }, - { - text: ` ${treeCollected.size !== 0 ? treeCollected.size : 1}${canBeChopped ? "" : "+" } ` - }, - { - translate: `LumberAxe.form.treeSizeAbrevLogs.text` - } - ]}, "textures/InfoUI/blocks.png") - .button( - { - rawtext: [ - { - translate: `LumberAxe.form.durabilityAbrev.text` - }, - { - text: ` ${currentDurability}` - } - ]}, "textures/InfoUI/axe_durability.png") - .button( - { - rawtext: [ - { - translate: `LumberAxe.form.maxDurabilityAbrev.text` - }, - { - text: ` ${maxDurability}` - } - ]}, "textures/InfoUI/required_durability.png") - .button( - { - rawtext: [ - { - text: "§l" - }, - { - translate: `${canBeChopped ? "LumberAxe.form.canBeChopped.text": "LumberAxe.form.cannotBeChopped.text"}` - } - ]}, "textures/InfoUI/canBeCut.png"); - forceShow(player, inspectionForm).then((response: ActionFormResponse) => { - playerInteractionMap.set(player.id, false); - if(response.canceled || response.selection === undefined || response.cancelationReason === FormCancelationReason.UserClosed) return; - }).catch((error: Error) => { - console.warn("Form Error: ", error, error.stack); + if(!visitedLogs[index]) return; + if(!visitedLogs[index].isDone) resetOutlinedTrees(newInspectedSubTree); + return; + })()); + }); + } else { + visitedLogs[index] = newInspectedSubTree; + } + system.clearJob(tMain); + inspectTreePromiseResolve({result: finalizedTreeInspection, index: index}); + })(inspectTreePromiseResolve)); + }); + + if(tempResult.index === -1) { + if(cooldown.getCooldownTicksRemaining(player) !== 0) return; + cooldown.startCooldown(player); + const molangVariable = new MolangVariableMap(); + // Get the bottom most log (TODO) + let isTreeDoneTraversing = false; + let treeOffsets: number[] = []; + let result: InteractedTreeResult = { + isBeingChopped: false, + visitedLogs: { + typeIds: new Map(), + blockOutlines: [], + source: new Graph(), + trunk: { + center: { x: 0, z: 0}, + size: 0 + }, + yOffsets: new Map() + }, + isDone: false, + initialSize: 0, + }; + let interactedTreeTrunk = await getTreeTrunkSize(blockInteracted, blockInteracted.typeId); + const topMostBlock = blockInteracted.dimension.getTopmostBlock(interactedTreeTrunk.center); + const bottomMostBlock = await new Promise((getBottomMostBlockResolved) => { + let _bottom = blockInteracted.below(); + const _t = system.runInterval(() => { + if(!isLogIncluded(blockInteracted.typeId, _bottom.typeId)) { + system.clearRun(_t); + getBottomMostBlockResolved(_bottom); + return; + } + _bottom = _bottom.below(); }); - }).catch((error: Error) => { - console.warn("Tree Error: ", error, error.stack); - playerInteractionMap.set(player.id, false); + }); + const trunkSizeToParticleRadiusParser = { + 1: 1.5, + 2: 2.5, + 3: 2.5, + 4: 2.5, + 5: 3.5, + 6: 3.5, + 7: 3.5, + 8: 3.5, + 9: 3.5 + } + let treeCollectedResult: VisitedBlockResult = null; + const trunkHeight = (topMostBlock.y - (bottomMostBlock.y + 1)); + const isValidVerticalTree = trunkHeight > 2; + + if(isValidVerticalTree) { + const {x: centerX, z: centerZ} = interactedTreeTrunk.center; + const centerBlockErrorCatch = blockInteracted.dimension.getBlock({x: centerX, y: blockInteracted.y, z: centerZ}); + + // (TODO) Only increase when it's 1 blocks away from center, so in total of 9 spaces. + if(!isLogIncluded(blockInteracted.typeId, centerBlockErrorCatch.typeId)) { + interactedTreeTrunk.size++; + } + const it = system.runInterval(() => { + // Get the first block, and based on that it will get the height. + if(result.isDone) { + system.clearRun(it); + return; + } + if(isTreeDoneTraversing) { + molangVariable.setFloat('radius', trunkSizeToParticleRadiusParser[treeCollectedResult.trunk.size]); + molangVariable.setFloat('height', treeOffsets.length); + molangVariable.setFloat('max_age', 1); + molangVariable.setColorRGB('color', {red: 0.0, green: 1.0, blue: 0.0}); + } else { + molangVariable.setFloat('radius', trunkSizeToParticleRadiusParser[interactedTreeTrunk.size]); + molangVariable.setFloat('height', trunkHeight); + molangVariable.setFloat('max_age', 1); + molangVariable.setColorRGB('color', {red: 1.0, green: 1.0, blue: 1.0}); // Change color based on property?? + } + player.dimension.spawnParticle('yn:inspecting_indicator', { + x: interactedTreeTrunk.center.x, + y: bottomMostBlock.y + 1, + z: interactedTreeTrunk.center.z + }, molangVariable); + }, 5); + } + treeCollectedResult = await getTreeLogs(player.dimension, blockInteracted.location, blockInteracted.typeId, +serverConfigurationCopy.chopLimit.defaultValue); + isTreeDoneTraversing = true; + if(isValidVerticalTree) { + treeOffsets = Array.from(treeCollectedResult.yOffsets.keys()).sort((a, b) => a - b); + // If center is empty, then just make the blockOutlines be the position of blockInteract and make radius increase by 1 + // Possible error caught in Expansive Biomes Redwood Forest + const {x: centerX, z: centerZ} = treeCollectedResult.trunk.center; + const centerBlockErrorCatch = blockInteracted.dimension.getBlock({x: centerX, y: blockInteracted.y, z: centerZ}); + if(!isLogIncluded(blockInteracted.typeId, centerBlockErrorCatch.typeId)) { + treeCollectedResult.trunk.size++; + } + } else { + const t = system.runJob((function*() { + for(const node of treeCollectedResult.source.traverseIterative(blockInteracted, "BFS")) { + molangVariable.setFloat('radius', 1.1); + molangVariable.setFloat('height', 0.97); + molangVariable.setFloat('max_age', BLOCK_OUTLINES_DESPAWN_CD / TicksPerSecond); + molangVariable.setColorRGB('color', {red: 0.0, green: 1.0, blue: 0.0}); // Change color based on property?? + player.dimension.spawnParticle('yn:inspecting_indicator', {x: node.block.bottomCenter().x, y: node.block.y, z: node.block.bottomCenter().z}, molangVariable); + yield; + } + system.clearJob(t); + })()); + } + result = { + isBeingChopped: false, + visitedLogs: treeCollectedResult, + isDone: false, + initialSize: treeCollectedResult.source.getSize(), + }; + if(result.initialSize > 0) visitedLogs.push(result); + system.runTimeout(() => { + // Reset temporarily permutations to block using dynamic property. + system.runJob((function*() { + for(const node of treeCollectedResult.source.traverseIterative(blockInteracted, "BFS")) { + originalDatabase.delete(`visited_${hashBlock(node.block)}`); + yield; + } + if(!result?.isDone) resetOutlinedTrees(result); + return; + })()); + }, BLOCK_OUTLINES_DESPAWN_CD); + } else if (tempResult.index >= 0) { + const size = tempResult.result.source.getSize(); + const totalDamage: number = size * unbreakingDamage; + const totalDurabilityConsumed: number = currentDurability + totalDamage; + const canBeChopped: boolean = ((totalDurabilityConsumed === maxDurability) || (totalDurabilityConsumed < maxDurability)) && (size <= +serverConfigurationCopy.chopLimit.defaultValue); + + const inspectionForm: ActionFormData = new ActionFormData() + .title({ + rawtext: [ + { + translate: "LumberAxe.form.title.text" + } + ]}) + .button( + { + rawtext: [ + { + translate: `LumberAxe.form.treeSizeAbrev.text` + }, + { + text: ` ${size !== 0 ? Math.round(canBeChopped ? size : reachableLogs + 1) : 1}${canBeChopped ? "" : "+" } ` + }, + { + translate: `LumberAxe.form.treeSizeAbrevLogs.text` + } + ]}, "textures/InfoUI/total_lumber.png") + .button( + { + rawtext: [ + { + text: `${tempResult.result.yOffsets.size} ` // Get the height of the trunk excluding the branches. + }, + { + translate: `LumberAxe.form.trunkHeightAbrev.text` + } + ]}, "textures/InfoUI/tree_height.png") + .button( + { + rawtext: [ + { + text: `${(maxDurability - totalDurabilityConsumed) > 0 ? '+' : ''}${Math.round(maxDurability - totalDurabilityConsumed)} ` + }, + { + translate: "LumberAxe.form.treeSizeAbrevLogs.text" + } + ]}, (maxDurability - totalDurabilityConsumed) > 0 ? "textures/InfoUI/lumber_surplus.png" : "textures/InfoUI/lumber_deficit.png") + .button( + { + rawtext: [ + { + translate: `${canBeChopped ? "LumberAxe.form.canBeChopped.text": "LumberAxe.form.cannotBeChopped.text"}` + } + ]}, canBeChopped ? "textures/InfoUI/can_be_cut.png" : "textures/InfoUI/cannot_be_cut.png"); + forceShow(player, inspectionForm).then((response: ActionFormResponse) => { + if(response.canceled || response.selection === undefined || response.cancelationReason === FormCancelationReason.UserClosed) { + return; + } + }).catch((error: Error) => { + Logger.error("Form Error: ", error, error.stack); + }); + } + } catch (e) { + console.warn(e, e.stack); + } +}); + +system.afterEvents.scriptEventReceive.subscribe((event: ScriptEventCommandMessageAfterEvent) => { + if(event.sourceType !== ScriptEventSource.Entity) return; + if(!(event.sourceEntity instanceof Player)) return; + if(event.id !== ADDON_IDENTIFIER) return; + const player = event.sourceEntity as Player; + const message = event.message; + const args = message.trim().split(/ +/g); + const cmd = args.shift().toLowerCase(); + system.run(async () => { + try { + const { + default: CommandObject + } = await import(`./commands/${cmd}.js`); + CommandObject.execute(player, args); + } catch (err) { + if (err instanceof ReferenceError) { + SendMessageTo(player, { + rawtext: [ + { + translate: "yn:fishing_got_reel.on_caught_main_command_not_found", + with: [ + cmd, + "\n", + ADDON_IDENTIFIER + ] + } + ] + }); + } else { + Logger.error(err, err.stack); + } + } }); }); \ No newline at end of file diff --git a/src/utils/Database/con-database.d.ts b/src/utils/Database/con-database.d.ts new file mode 100644 index 0000000..6311654 --- /dev/null +++ b/src/utils/Database/con-database.d.ts @@ -0,0 +1,121 @@ +import { World, Entity } from "@minecraft/server"; + +declare class DynamicSource { + readonly source: World | Entity; + constructor(source: World | Entity); + getIds(): string[]; + get(key: string): any; + set(key: string, value: any): void; + delete(key: string): boolean | undefined; + isValid(): boolean; +} + +declare class DynamicDatabase extends Map { + constructor( + source: DynamicSource, + id: string, + kind: string, + parser: { stringify(object: any): string; parse(raw: string): any } + ); + isValid(): boolean; + dispose(): void; + readonly isDisposed: boolean; +} + +declare class JsonDatabase extends DynamicDatabase { + constructor(id: string, source?: World | Entity); +} + +interface DynamicProxyConstructor { + new (id: string, source?: World | Entity): { [k: string]: any | undefined }; +} +declare const DynamicProxy: DynamicProxyConstructor; + +declare enum SerializableKinds { + Boolean = "c0211201-0001-4002-8001-4f90af596647", + Number = "c0211201-0001-4002-8002-4f90af596647", + String = "c0211201-0001-4002-8003-4f90af596647", + Object = "c0211201-0001-4002-8004-4f90af596647", + DynamicTable = "c0211201-0001-4002-8101-4f90af596647", +} + +interface Deserializer extends Generator { + continue(): T; + readonly source: DynamicSource; + readonly length: number; + readonly rootKey: string; + readonly kind: string; +} + +declare namespace Serializer { + function isSerializable(object: any): boolean; + function getSerializerKind(object: any): string | undefined; + function isRegistredKind(kind: string): boolean; + function setSerializableKind(object: object, kind: string): boolean; + function registrySerializer( + kind: K, + serializer: (object: any) => Generator, + deserializer: (source: Deserializer) => void + ): K; + function getSerializer(kind: string): null | ((object: any) => Generator); + function getDeserializer( + kind: string + ): ((source: Deserializer) => void) | null; + function setSerializableClass( + construct: new (...any: any[]) => T, + kind: string, + serializer: (object: T) => Generator, + deserializer: (source: Deserializer) => T + ): void; + function getKindFromClass( + construct: new (...any: any[]) => any + ): string | null; + function getSerializerKinds(): IterableIterator; + function getSerializers(kind: string): { + serializer: (object: any) => Generator; + deserializer: (source: Deserializer) => void; + }; + function overrideSerializers( + kind: K, + serializer: (object: any) => Generator, + deserializer: (source: Deserializer) => void + ): K; +} + +declare class DynamicTable extends Map { + static readonly KIND: string; + static OpenCreate(id: string): DynamicTable; + static ClearAll(): void; + static getTableIds(): IterableIterator; + static DeleteTable(id: string): boolean; + + readonly tableId: string; + private constructor(); + isValid(): boolean; +} + +declare class DataCoruptionError extends ReferenceError { + constructor(source: DynamicSource, rootKey: string, message: string); + remove(): void; +} + +export { + JsonDatabase, + DynamicProxy, + DynamicTable, + Serializer, + DataCoruptionError, + SerializableKinds, +}; + +export const registryAPISerializers: () => void; + +export declare enum APISerializableKinds { + BlockType = "c0211201-0001-4002-8201-4f90af596647", + EntityType = "c0211201-0001-4002-8202-4f90af596647", + ItemType = "c0211201-0001-4002-8203-4f90af596647", + BlockPermutation = "c0211201-0001-4002-8204-4f90af596647", + ItemStack = "c0211201-0001-4002-8205-4f90af596647", + Vector = "c0211201-0001-4002-8206-4f90af596647", +} + \ No newline at end of file diff --git a/src/utils/Database/con-database.js b/src/utils/Database/con-database.js new file mode 100644 index 0000000..158e05f --- /dev/null +++ b/src/utils/Database/con-database.js @@ -0,0 +1,821 @@ +import { world, World, Entity, system } from "@minecraft/server"; +import * as mc from "@minecraft/server"; +const mc_world = world; +const {setDynamicProperty: wSDP, getDynamicProperty: wGDP, getDynamicPropertyIds: wGDPI} = World.prototype; +let {isValid: isValidEntity, setDynamicProperty: eSDP, getDynamicProperty: eGDP, getDynamicPropertyIds: eGDPI} = Entity.prototype; +const DYNAMIC_DB_PREFIX = "\u1221\u2112"; +const ROOT_CONTENT_TABLE_UUID = "c0211201-0001-4001-8001-4f90af596647"; +const STRING_LIMIT = 32e3; +const TABLE_STRING_LENGTH = 31e3; +const GENERATOR_DESERIALIZER_SYMBOL = Symbol("DESERIALIZER"); +const eP = { + gDP: eGDP, + sDP: eSDP, + gDPI: eGDPI +}; +const wP = { + gDP: wGDP, + sDP: wSDP, + gDPI: wGDPI +}; +class DynamicSource { + /**@readonly @type {World | Entity} */ + source; + /**@param {World | Entity} source */ + constructor(source){ + this.source = source; + if(SOURCE_INSTANCES.has(source)) return SOURCE_INSTANCES.get(source); + if(source === mc_world) Object.assign(this, wP); else if (isValidEntity.call(source)) Object.assign(this, eP); + else throw new ReferenceError("Invald source type: " + source); + SOURCE_INSTANCES.set(source, this); + } + /**@returns {string[]} */ + getIds(){return this.gDPI.call(this.source);} + /**@param {string} key @returns {number | boolean | string | import("@minecraft/server").Vector3}*/ + get(key){return this.gDP.call(this.source,key);} + /**@param {string} key */ + set(key,value){ this.sDP.call(this.source, key, value);} + /**@param {string} key @returns {boolean} */ + delete(key){this.sDP.call(this.source, key, undefined); return true;} + /**@returns {boolean} */ + isValid(){ return this.source === world || isValidEntity.call(this.source); } +} +/////////////////////////////////////////////////////////////////////////////////////////////////////// +////////////////////////////////////////////////////////////////////////////////// DYNAMIC DATABASE +/////////////////////////////////////////////////////////////////////////////////////////////////////// +const SOURCE_INSTANCES = new WeakMap(); +const DDB_SUBINSTANCES = new WeakMap(); +class DynamicDatabase extends Map{ + /**@readonly @private @type {DynamicSource} */ + _source; + /**@readonly @private @type {string} */ + _prefix; + /**@readonly @private @type {string} */ + _prefixLength; + /**@readonly @private */ + _STRINGIFY; + /**@readonly @private*/ + _PARSE; + /** @private*/ + _notDisposed; + /**@param {World | Entity} source @param {string} id @param {string} kind */ + constructor(source, id, kind, parser){ + super(); + this._source = new DynamicSource(source); + const PRE = `${kind}${DYNAMIC_DB_PREFIX}${id}${DYNAMIC_DB_PREFIX}`, LENGTH = PRE.length, SOURCE = this._source, PARSE = parser.parse; + const MAP_INSTANCES = DDB_SUBINSTANCES.get(SOURCE)??new Map; + if(MAP_INSTANCES.has(PRE)) return MAP_INSTANCES.get(PRE); + MAP_INSTANCES.set(PRE, this); DDB_SUBINSTANCES.set(SOURCE, MAP_INSTANCES); + if(!SOURCE.isValid()) throw new ReferenceError("Source is no longer valid: " + SOURCE.source); + this._prefix = PRE; + this._prefixLength = LENGTH; + this._STRINGIFY = parser.stringify; + //this._PARSE = PARSE; + this._notDisposed = true; + for (const K of SOURCE.getIds()) if(K.startsWith(PRE)) { + const key = K.substring(LENGTH); + const value = SOURCE.get(K); + if(typeof value === "string") super.set(key, PARSE(value)); + } + } + /**@param {string} key @param {any} value */ + set(key, value){ + if(!this.isValid()) throw new ReferenceError("This database instance is no longer valid"); + if(key.length + this._prefixLength > STRING_LIMIT) throw new TypeError("Key is too long: " + key.length); + if(value === undefined) { + this.delete(key); + return this; + } + const data = this._STRINGIFY(value); + if(data.length > STRING_LIMIT) throw new TypeError("Size of data in string is too long: " + data.length); + this._source.set(this._prefix + key, data); + return super.set(key,value); + } + /**@param {string} key */ + delete(key){ + if(!this.isValid()) throw new ReferenceError("This database instance is no longer valid"); + if(!this.has(key)) return false; + this._source.delete(this._prefix + key); + return super.delete(key); + } + clear(){ + if(!this.isValid()) throw new ReferenceError("This database instance is no longer valid"); + const P = this._prefix; + const s = this._source; + for(const key of this.keys()) s.delete(P + key); + return super.clear(); + } + /**@returns {boolean} */ + isValid(){return this._source.isValid() && this._notDisposed;} + dispose(){ + this._notDisposed = false; + DDB_SUBINSTANCES.get(this._source)?.delete?.(this._prefix); + super.clear(); + } + /**@readonly @type {boolean} */ + get isDisposed(){return !this._notDisposed;} +} +class DynamicWrapper { + + /**@readonly @private @type {Entity | World | ItemStack} */ + _source; + /**@readonly @private @type {string} */ + _prefix; + /**@readonly @private @type {string} */ + _prefixLength; + /**@readonly @private */ + _STRINGIFY; + /**@readonly @private*/ + _PARSE; + /**@param {World | Entity} source @param {string} id @param {string} kind */ + constructor(source, id, kind, parser){ + this._source = source; + const PRE = `${kind}${DYNAMIC_DB_PREFIX}${id}${DYNAMIC_DB_PREFIX}`, LENGTH = PRE.length; + this._prefix = PRE; + this._prefixLength = LENGTH; + this._STRINGIFY = parser.stringify; + this._PARSE = parser.parse; + } + clear(){ for( const k of this.__getKeys()) this._source.set(k, undefined); }; + /** + * @returns true if an element in the Map existed and has been removed, or false if the element does not exist. + */ + delete(key) { + const has = this.has(key); + this._source.set(this._prefix + key, undefined); + return has; + }; + /** + * Executes a provided function once per each key/value pair in the Map, in insertion order. + * @param {(value: any, key: string, map: DynamicWrapper) => void} callbackfn + * @param {any} thisArg + */ + forEach(callbackfn, thisArg = null){ + for (const k of this.keys()) { + try { + callbackfn.call(thisArg??null, k, this.get(k), this); + } catch (error) { + + } + } + } + /** + * Returns a specified element from the Map object. If the value that is associated to the provided key is an object, then you will get a reference to that object and any change made to that object will effectively modify it inside the Map. + * @returns Returns the element associated with the specified key. If no element is associated with the specified key, undefined is returned. + */ + get(key){ const a = this._source.get(this._prefix + key); typeof a === "string"?this._PARSE(a):a; }; + /** + * @returns boolean indicating whether an element with the specified key exists or not. + */ + has(key){ return this._source.get(this._prefix + key)!==undefined; } + /** + * Adds a new element with a specified key and value to the Map. If an element with the same key already exists, the element will be updated. + */ + set(key, value){ + this._source.set(this._prefix + key, this._STRINGIFY(value)); + return this; + }; + /** + * @returns the number of elements in the Map. + */ + get size(){return [...this.__getKeys()].length;}; + /** Returns an iterable of entries in the map. */ + [Symbol.iterator](){return this.entries();} + /** + * Returns an iterable of key, value pairs for every entry in the map. + */ + *entries(){for( const k of this.__getKeys()) yield [k.substring(this._prefixLength),this._PARSE(this._source.get(k))]}; + /** + * Returns an iterable of keys in the map + */ + *keys(){for( const k of this.__getKeys()) yield k.substring(this._prefixLength);}; + /** + * Returns an iterable of values in the map + */ + *values(){for( const k of this.__getKeys()) yield this._PARSE(this._source.get(k));} + *__getKeys(){ for (const K of this._source.getIds()) if(K.startsWith(this._prefix)) yield K; } +} +class JsonDatabase extends DynamicDatabase{ constructor(id, source = world){ super(source, id, "JSON", JSON); } } +class JSONDynamicWrapper extends DynamicWrapper{ constructor(id, source = world){ super(source, id, "JSON", JSON); } } + +class DynamicProxy extends JsonDatabase{ + constructor(id, source = world){ + super(id, source); + return new Proxy(this,{ + defineProperty(t,p,att){ + if(att.value && typeof p === "string"){ + t.set(p,att.value); + return true; + } + return false; + }, + deleteProperty(t,p){ + if(typeof p === "string") return t.delete(p); + return false; + }, + set(t, p, newValue){ + if(typeof p === "string") { + t.set(p, newValue); + return true; + } + return false; + }, + get(t, p){ + if(typeof p === "string") { + return t.get(p)??Object.prototype[p]; + } + return false; + }, + getPrototypeOf(t){return Object.prototype;}, + isExtensible(t){return true;}, + setPrototypeOf(t){return false;}, + has(t,k){return t.has(k);}, + preventExtensions(t){return false;}, + ownKeys(t){ return [...t.keys()]; }, + getOwnPropertyDescriptor(t,k){ + if(t.has(k)){ + return {value:t.get(k), enumerable: true, configurable: true, writable: true}; + } + } + }); + } +} +/////////////////////////////////////////////////////////////////////////////////////////////////////// +////////////////////////////////////////////////////////////////////////////////// COMPLEX DATABASE +/////////////////////////////////////////////////////////////////////////////////////////////////////// +const PARSER_SYMBOL = Symbol("SERIALIZEABLE"); +const SERIALIZERS = new Map(); +const DESERIALIZER_INFO = new WeakMap(); +const ROOT_KEY = "root::" + ROOT_CONTENT_TABLE_UUID; +const TABLE_SOURCES = new WeakMap(); +const TABLE_ID = new WeakMap(); +const ID_TABLE = new WeakMap(); +const TABLE_VALIDS = new WeakSet(); +let isNativeCall = false; +let RootTable; +function getRootTable(){ + if(RootTable) return RootTable; + return RootTable = world.getDynamicProperty(ROOT_KEY)?DATABASE_MANAGER.deserialize(ROOT_KEY, new DynamicSource(world)):(()=>{ + const source = new DynamicSource(world); + isNativeCall = true; + const value = new DynamicTable(); + isNativeCall = false; + TABLE_SOURCES.set(value, source); + TABLE_ID.set(value, ROOT_KEY); + SetTable(source, ROOT_KEY, value); + TABLE_VALIDS.add(value); + DATABASE_MANAGER.serialize(ROOT_KEY, source, value); + return value; + })(); +} + +const SerializableKinds = { + Boolean:"c0211201-0001-4002-8001-4f90af596647", + Number:"c0211201-0001-4002-8002-4f90af596647", + String:"c0211201-0001-4002-8003-4f90af596647", + Object:"c0211201-0001-4002-8004-4f90af596647", + DynamicTable: "c0211201-0001-4002-8101-4f90af596647" +}; +SerializableKinds[SerializableKinds.Boolean] = "Boolean"; +SerializableKinds[SerializableKinds.Number] = "Number"; +SerializableKinds[SerializableKinds.String] = "String"; +SerializableKinds[SerializableKinds.Object] = "Object"; +SerializableKinds[SerializableKinds.DynamicTable] = "DynamicTable"; +const Serializer = { + isSerializable(object){ return object[PARSER_SYMBOL] != undefined; }, + getSerializerKind(object){ return object[PARSER_SYMBOL];}, + isRegistredKind(kind){return SERIALIZERS.has(kind);}, + setSerializableKind(object, kind){ + if(SERIALIZERS.has(kind)){ object[PARSER_SYMBOL] = kind; return true; } + return false; + }, + registrySerializer(kind, serializer, deserializer){ + if(SERIALIZERS.has(kind)) throw new ReferenceError("Duplicate serialization kind: " + kind); + if(typeof kind != "string") throw new TypeError("Kind must be type of string."); + if(typeof serializer != "function" || typeof deserializer != "function") throw new TypeError("serializer or deserializer is not a function"); + SERIALIZERS.set(kind,{serializer,deserializer}); + return kind; + }, + getSerializer(kind){ + return SERIALIZERS.get(kind)?.serializer??null; + }, + getDeserializer(kind){ + return SERIALIZERS.get(kind)?.deserializer??null; + }, + getSerializers(kind){ + const data = SERIALIZERS.get(kind); + if(!data) return null; + return {...data}; + }, + setSerializableClass(construct, kind, serializer, deserializer){ + if(typeof serializer !== "function" || typeof deserializer !== "function") throw new TypeError("Serializer or deserializer is not a function"); + Serializer.registrySerializer(kind, function(obj){ + if(obj == null) throw new TypeError("Null or Undefined is not possible to serialize."); + return serializer(obj); + }, function(obj){ + if(obj[GENERATOR_DESERIALIZER_SYMBOL] !== true) throw new TypeError("Null or Undefined is not possible to serialize."); + return deserializer(obj); + }); + Serializer.setSerializableKind(construct.prototype,kind); + }, + getKindFromClass(construct){ + return construct?.prototype?.[PARSER_SYMBOL]??null; + }, + getSerializerKinds(){return SERIALIZERS.keys();}, + overrideSerializers(kind, serializer, deserializer){ + if(typeof kind != "string") throw new TypeError("Kind must be type of string."); + if(typeof serializer != "function" || typeof deserializer != "function") throw new TypeError("serializer or deserializer is not a function"); + SERIALIZERS.set(kind,{serializer,deserializer}); + return kind; + } +} +const DATABASE_MANAGER = { + getHeader(rootRef, source){ + const data = source.get(rootRef); + if(typeof data != "string") return null; + return JSONReadable(data); + }, + serialize(rootRef, source, object){ + if(!Serializer.isRegistredKind(Serializer.getSerializerKind(object))) throw new TypeError("object is not serializeable."); + const kind = Serializer.getSerializerKind(object) + const serializer = Serializer.getSerializer(kind); + if(!serializer) throw new ReferenceError("No serializer for " + kind); + return this.serializationResolver( + serializer(object, {kind,source,rootRef}) + ,rootRef,source, kind + ); + }, + /**@param {Generator} gen */ + serializationResolver(gen, rootRef, source, kind){ + const oldHeader = this.getHeader(rootRef, source); + const prefix = rootRef + "::"; + let oldLength = 0, newLength = 0; + if(oldHeader){ + const [data] = oldHeader; + oldLength = parseInt(data["length"],36); + } + try { + let genNext = gen.next(); + if(!genNext.done) { + const headerData = genNext.value + ""; + if(headerData.length > TABLE_STRING_LENGTH) gen.throw(new RangeError("Yielded stirng is too big: " + headerData.length)); + genNext = gen.next(); + while (!genNext.done) { + const key = prefix + newLength; + try { + source.set(key, genNext.value + ""); + newLength++; + } catch (error) { + gen.throw(error); + } + genNext = gen.next(); + } + source.set(rootRef, JSONWritable({length:newLength.toString(36),kind},headerData)); + } + return newLength; + } + catch(er){ + Object.setPrototypeOf(er, DataCoruptionError.prototype); + er.source = source; + er.rootKey = rootRef; + throw er; + } + finally { + for (let i = newLength; i < oldLength; i++) source.delete(prefix + i); + } + }, + deserialize(rootRef, source, header = undefined){ + try { + const oldHeader = header??this.getHeader(rootRef, source); + if(!oldHeader) return null; + const prefix = rootRef + "::"; + const [{length:le,kind}, data] = oldHeader; + let length = parseInt(le,36); + if(!Serializer.isRegistredKind(kind)) throw new ReferenceError("Unknown parser kind: " + kind); + const deserializeResolver = Serializer.getDeserializer(kind); + if(!deserializeResolver) throw new ReferenceError("No deserializer for: " + kind); + const deserializer = this.deserializer(source, rootRef, prefix, length, data); + DESERIALIZER_INFO.set(deserializer, { + source, + rootRef, + kind, + deserializeResolver, + oldHeader, + length, + }); + return deserializeResolver(deserializer); + } catch (error) { + error.rootKey = rootRef; + error.source = source; + throw Object.setPrototypeOf(error, DataCoruptionError); + } + }, + *deserializer(source, root, prefix, length, initial){ + yield initial; + let i = 0; + while(i < length){ + const data = source.get(prefix + i); + if(!data) throw new DataCoruptionError(source, root, "No continual data at index of " + i); + yield data; + i++; + } + }, + removeTree(rootRef, source){ + const oldHeader = this.getHeader(rootRef, source); + if(!oldHeader) return false; + const prefix = rootRef + "::"; + const [{length:le}] = oldHeader; + let length = parseInt(le,36); + if(!isFinite(length)) return false; + for (let i = 0; i < length; i++) source.delete(prefix + i); + source.delete(rootRef); + return true; + } +} + +Object.defineProperties(DATABASE_MANAGER.deserializer.prototype,Object.getOwnPropertyDescriptors({ + [GENERATOR_DESERIALIZER_SYMBOL]: true, + return(){ + return {done:true}; + }, + continue(){ + return this.next(...arguments).value; + }, + get source(){ + if(!DESERIALIZER_INFO.has(this)) throw new ReferenceError("Object bound to prototype does not exist."); + return DESERIALIZER_INFO.get(this).source; + }, + get rootKey(){ + if(!DESERIALIZER_INFO.has(this)) throw new ReferenceError("Object bound to prototype does not exist."); + return DESERIALIZER_INFO.get(this).rootRef; + }, + get length(){ + if(!DESERIALIZER_INFO.has(this)) throw new ReferenceError("Object bound to prototype does not exist."); + return DESERIALIZER_INFO.get(this).length; + }, + get kind(){ + if(!DESERIALIZER_INFO.has(this)) throw new ReferenceError("Object bound to prototype does not exist."); + return DESERIALIZER_INFO.get(this).kind; + } +})); +class DynamicTable extends Map{ + /**@readonly */ + static get KIND(){return "c0211201-0001-4002-8101-4f90af596647";} + /**@readonly @type {string} */ + get tableId(){return TABLE_ID.get(this);} + constructor(){ + if(!isNativeCall) throw new ReferenceError("No constructor for " + DynamicTable.name); + super(); + } + get(key){ + if(!this.isValid()) throw new ReferenceError("Object bound to prototype doesn't not exist at [DynamicTable::get()]."); + if(!this.has(key)) return; + const source = TABLE_SOURCES.get(this); + const dataId = super.get(key); + return DATABASE_MANAGER.deserialize(dataId, source); + } + set(key, value){ + if(!this.isValid()) throw new ReferenceError("Object bound to prototype doesn't not exist at [DynamicTable::get()]."); + if(value == null) throw new ReferenceError("You can not assign property to null or undefined"); + if(!Serializer.isRegistredKind(Serializer.getSerializerKind(value))) throw new TypeError("value is not serializeable."); + if(value instanceof DynamicTable) throw new TypeError("You can't set value as DynamicTable please use AddTable"); + const has = this.has(key); + + const source = TABLE_SOURCES.get(this); + + let newKey; + if(has){ + newKey = super.get(key); + const header = DATABASE_MANAGER.getHeader(newKey, source); + if(header?.[0]?.kind === DynamicTable.KIND) { + const a = DATABASE_MANAGER.deserialize(newKey, source, header); + a.clear(); + TABLE_VALIDS.delete(a); + } + }else{ + newKey = "k:" + v4uuid() + super.set(key, newKey); + SaveState(this); + } + DATABASE_MANAGER.serialize(newKey, source, value); + return this; + } + clear(){ + if(!this.isValid()) throw new ReferenceError("Object bound to prototype doesn't not exist at [DynamicTable::clear()]."); + const source = TABLE_SOURCES.get(this); + const KIND = DynamicTable.KIND; + for (const k of super.keys()) { + const dataId = super.get(k); + const header = DATABASE_MANAGER.getHeader(dataId,source); + if(header?.[0]?.kind === KIND) { + const a = DATABASE_MANAGER.deserialize(dataId, source, header); + a.clear(); + TABLE_VALIDS.delete(a); + } + DATABASE_MANAGER.removeTree(dataId, source); + } + SaveState(this); + super.clear(); + } + delete(key){ + if(!this.isValid()) throw new ReferenceError("Object bound to prototype doesn't not exist at [DynamicTable::delete()]."); + const source = TABLE_SOURCES.get(this); + if(!this.has(key)) return false; + const dataId = super.get(key); + const header = DATABASE_MANAGER.getHeader(dataId,source); + if(header?.[0]?.kind === DynamicTable.KIND) { + const a = DATABASE_MANAGER.deserialize(dataId, source, header); + a.clear(); + TABLE_VALIDS.delete(a); + } + DATABASE_MANAGER.removeTree(dataId, source); + SaveState(this); + return super.delete(); + } + *entries(){ + if(!this.isValid()) throw new ReferenceError("Object bound to prototype doesn't not exist at [DynamicTable::entries()]."); + for (const [k,v] of super.entries()) yield [k, this.get(k)]; + } + [Symbol.iterator](){return this.entries();} + *values(){ + if(!this.isValid()) throw new ReferenceError("Object bound to prototype doesn't not exist at [DynamicTable::values()]."); + for (const k of super.keys()) yield this.get(k); + } + isValid(){ + return !!(TABLE_VALIDS.has(this) && TABLE_SOURCES.get(this)?.isValid?.()); + } + /**@returns {DynamicTable} */ + static OpenCreate(id){ + let fromTable = getRootTable(); + let a = fromTable.get(id); + if(a === undefined) { + if(!fromTable.isValid()) throw new ReferenceError("Object bound to prototype doesn't not exist at [DynamicTable::get()]."); + if(Map.prototype.has.call(fromTable,id)) throw new ReferenceError("Value of this key already exists"); + const source = TABLE_SOURCES.get(fromTable); + let newKey = "t" + v4uuid(); + isNativeCall = true; + const value = new DynamicTable(); + isNativeCall = false; + Map.prototype.set.call(fromTable, id, newKey); + SaveState(fromTable); + DATABASE_MANAGER.serialize(newKey, source, value); + TABLE_SOURCES.set(value, source); + TABLE_ID.set(value, newKey); + SetTable(source, newKey, value); + TABLE_VALIDS.add(value); + a = value; + } + else if(!(a instanceof DynamicTable)) throw new TypeError(`Value saved in ${id} is not a dynamic table.`); + return a; + } + static ClearAll(){ getRootTable().clear(); } + static getTableIds(){ return getRootTable().keys(); } + static DeleteTable(key){ return getRootTable().delete(key);} +} +function SaveState(table){ + if(table._task === undefined) { + table._task = system.run(()=>{ + table._task = undefined; + if(table.isValid()){ + DATABASE_MANAGER.serialize(table.tableId, TABLE_SOURCES.get(table), table); + } + }); + } +} +function GetTable(source, rootRef){ return ID_TABLE.get(source)?.get(rootRef); } +function SetTable(source, rootRef, table){ + if(!ID_TABLE.has(source)) ID_TABLE.set(source,new Map()); + ID_TABLE.get(source).set(rootRef, table); +} +class DataCoruptionError extends ReferenceError{ + constructor(source, rootKey, message){ + super(message); + this.rootKey = rootKey; + this.source = source; + } + remove(){ + if(!this.source.isValid()) throw new ReferenceError("Source is no longer valid"); + DATABASE_MANAGER.removeTree(this.rootKey, this.source); + } +} + +Serializer.setSerializableClass(DynamicTable, DynamicTable.KIND, + function*(table){ + let obj = {}, i = 0; + const get = Map.prototype.get, maxSize = 300; + yield Math.ceil(table.size / maxSize); + for(const key of table.keys()) { + if(++i >= maxSize){ + yield JSON.stringify(obj); + i = 0, obj = {}; + } + obj[key] = get.call(table,key); + } + if(i) yield JSON.stringify(obj); + }, + function(n){ + if(GetTable(n.source,n.rootKey)) return GetTable(n.source,n.rootKey); + isNativeCall = true; + const table = new DynamicTable(); + isNativeCall = false; + TABLE_SOURCES.set(table, n.source); + TABLE_ID.set(table, n.rootKey); + SetTable(n.source, n.rootKey, table); + TABLE_VALIDS.add(table); + const set = Map.prototype.set; + const length = Number(n.continue()); + for (let i = 0; i < length; i++) { + const data = n.continue(); + if(!data) throw new DataCoruptionError(n.source,n.rootKey,"Data for this dynamic table are corupted."); + const obj = JSON.parse(data); + for(const k of Object.getOwnPropertyNames(obj)) set.call(table, k, obj[k]); + } + return table; + } +); +Serializer.setSerializableClass(Boolean, SerializableKinds.Boolean, function*(n){yield n;}, function(n){for(const a of n) return a==="true";}); +Serializer.setSerializableClass(Number, SerializableKinds.Number, function*(n){yield n;}, function(n){for(const a of n) return Number(a);}); +Serializer.setSerializableClass(String, SerializableKinds.String, + function*(n){ + let length = n.length; + let cursor = 0; + let i = 0; + yield Math.ceil(length / TABLE_STRING_LENGTH); + while(length > 0){ + const s = n.substring(cursor,cursor + TABLE_STRING_LENGTH); + const l = s.length; + if(l <= 0) return; + length -= l, cursor += l; + yield s; + i++; + } + }, + function(n){ + const count = Number(n.continue()); + const l = new Array(count); + for (let i = 0; i < count; i++) { + l[i] = n.continue(); + } + return l.join(""); + } +); +Serializer.setSerializableClass(Object, SerializableKinds.Object, + function(n){ return Serializer.getSerializer(SerializableKinds.String)(JSON.stringify(n)); }, + function(n){ return JSON.parse(Serializer.getDeserializer(SerializableKinds.String)(n)); } +); +function v4uuid(timestamp = Date.now()){ + const {random,floor} = Math; + const uuid = 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, (c) =>{ + let r = (timestamp + random() * 16) % 16 | 0; + timestamp = floor(timestamp / 16); + return (c == 'x' ? r : (r & 0x3 | 0x8)).toString(16); + }); + return uuid; +} +function Readable(text){ + const size = text.charCodeAt(0); + const info = text.substring(1,1+size); + const data = text.substring(1+size); + return [info, data, size]; +} +function JSONReadable(text){ + const [info, data, size] = Readable(text); + return [JSON.parse(info), data, size]; +} +function Writable(json, text){ + return `${String.fromCharCode(json.length)}${json}${text}`; +} +function JSONWritable(json, text){ return Writable(JSON.stringify(json),text); }; + +export {JsonDatabase, DynamicProxy, DynamicTable, Serializer, DataCoruptionError, SerializableKinds}; + +export const APISerializableKinds = { + BlockType: "c0211201-0001-4002-8201-4f90af596647", + EntityType: "c0211201-0001-4002-8202-4f90af596647", + ItemType: "c0211201-0001-4002-8203-4f90af596647", + BlockPermutation: "c0211201-0001-4002-8204-4f90af596647", + ItemStack: "c0211201-0001-4002-8205-4f90af596647", + Vector: "c0211201-0001-4002-8206-4f90af596647", + "c0211201-0001-4002-8201-4f90af596647": "BlockType", + "c0211201-0001-4002-8202-4f90af596647": "EntityType", + "c0211201-0001-4002-8203-4f90af596647": "ItemType", + "c0211201-0001-4002-8204-4f90af596647": "BlockPermutation", + "c0211201-0001-4002-8205-4f90af596647": "ItemStack", + "c0211201-0001-4002-8206-4f90af596647": "Vector", +} +export const registryAPISerializers = ()=>{ + const { + BlockType, BlockTypes, BlockPermutation, + EntityType, EntityTypes, + ItemType, ItemTypes, ItemStack, Vector + } = mc; + const ItemStackSupportLevel = { + dynamicProperties: ItemStack.prototype.getDynamicProperty, + canPlaceOn: ItemStack.prototype.getCanPlaceOn, + canDestory: ItemStack.prototype.getCanDestroy, + lore: ItemStack.prototype.getLore, + lockMode: mc.ItemLockMode, + keepOnDeath: "keepOnDeath" in ItemStack.prototype, + components: ItemStack.prototype.getComponents, + enchantable: mc.ItemEnchantableComponent, + durability: mc.ItemDurabilityComponent, + } + for (const key in ItemStackSupportLevel) { + if (Object.hasOwnProperty.call(ItemStackSupportLevel, key)) { + const element = ItemStackSupportLevel[key]; + } + } + const ItemStackComponentManager = { + serializers:{ }, + deserializers:{ } + } + const { + serializers: ItemComponentSerializers, + deserializers: ItemComponentDeserializers + } = ItemStackComponentManager; + if(ItemStackSupportLevel.durability) { + ItemComponentSerializers[ItemStackSupportLevel.durability.componentId] = function(component){ return component.damage;}; + ItemComponentDeserializers[ItemStackSupportLevel.durability.componentId] = function(component,v){ component.damage = v;}; + } + if(ItemStackSupportLevel.enchantable){ + ItemComponentSerializers[ItemStackSupportLevel.enchantable.componentId] = function(component){ return component.getEnchantments().map(e=>({t:e.type.id,l:e.level}));}; + ItemComponentDeserializers[ItemStackSupportLevel.enchantable.componentId] = function(component,v){ component.addEnchantments(v.map(e=>({type:e.t,level:e.l})));}; + } + + if(BlockTypes) Serializer.setSerializableClass(BlockType, APISerializableKinds.BlockType, function*(n){yield n.id;}, function(n){for(const a of n) return BlockTypes.get(a);}); + if(EntityTypes) Serializer.setSerializableClass(EntityType, APISerializableKinds.EntityType, function*(n){yield n.id;}, function(n){for(const a of n) return EntityTypes.get(a);}); + if(ItemTypes) Serializer.setSerializableClass(ItemType, APISerializableKinds.ItemType, function*(n){yield n.id;}, function(n){for(const a of n) return ItemTypes.get(a);}); + + if("type" in BlockPermutation.prototype) Serializer.setSerializableClass(BlockPermutation, APISerializableKinds.BlockPermutation, + function*(n){ + yield n.type.id; + yield JSON.stringify(n.getAllStates()); + }, + function(n){ + const [typeId, states] = n; + return BlockPermutation.resolve(typeId, JSON.parse(states)); + } + ); + Serializer.setSerializableClass(ItemStack, APISerializableKinds.ItemStack, + function*(n){ + const components = ItemStackSupportLevel.components?[...n.getComponents()].filter(e=>e && (e.typeId in ItemComponentSerializers)):[]; + const canPlaceOn = ItemStackSupportLevel.canPlaceOn?n.getCanPlaceOn():[]; + const canDestroy = ItemStackSupportLevel.canDestory?n.getCanDestroy():[]; + const dynamicProperties = ItemStackSupportLevel.dynamicProperties?n.getDynamicPropertyIds():[]; + yield JSON.stringify([ + n.typeId, + n.amount, + ItemStackSupportLevel.keepOnDeath?n.keepOnDeath:false, + ItemStackSupportLevel.lockMode?n.lockMode:"", + typeof n.nameTag === "string", + components.length, + canPlaceOn.length, + canDestroy.length, + dynamicProperties.length + ]); + n.nameTag?yield n.nameTag:null; + yield JSON.stringify(ItemStackSupportLevel.lore?n.getLore():[]); + for (const com of components) yield JSON.stringify([com.typeId, ItemComponentSerializers[com.typeId](com)]); + yield * canPlaceOn; + yield * canDestroy; + for(const k of dynamicProperties) { + const data = JSON.stringify(n.getDynamicProperty(k)); + if((data.length + k.length) > TABLE_STRING_LENGTH) throw new TypeError(`Dynamic property '${k}' of this item is too large'${data.length}'`); + yield Writable(k,data); + } + }, + function(n){ + const [ + typeId, amount, keepOnDeath, lockMode, hasNameTag, + componentsCount, canPlaceOnCount, canDestroyCount, dynamicPropertiesCount + ] = JSON.parse(n.continue()); + const item = new ItemStack(typeId, amount); + if(ItemStackSupportLevel.keepOnDeath) item.keepOnDeath = keepOnDeath; + if(ItemStackSupportLevel.lockMode) item.lockMode = lockMode; + if(hasNameTag) item.nameTag = n.continue(); + const lore = JSON.parse(n.continue()); + if(ItemStackSupportLevel.lore) item.setLore(lore); + let i = componentsCount; + while(i--) { + const [id,data] = JSON.parse(n.continue()); + ItemComponentDeserializers[id](item.getComponent(id), data); + } + i = canPlaceOnCount; + const canPlaceOn = []; + while(i--) canPlaceOn.push(n.continue()); + i = canDestroyCount; + const canDestroy = []; + while(i--) canDestroy.push(n.continue()); + item.setCanPlaceOn(canPlaceOn); + item.setCanDestroy(canDestroy); + i = dynamicPropertiesCount; + while(i--) { + const [k,sata] = Readable(n.continue()); + item.setDynamicProperty(k,JSON.parse(sata)); + } + return item; + } + ); + + if(Vector) Serializer.setSerializableClass(Vector, APISerializableKinds.Vector, function(s){const {x,y,z} = s; return Object.Serialize({x,y,z});}, function(n){const {x,y,z} = Object.Deserialize(n);return new Vector(x,y,z);}) +}; \ No newline at end of file diff --git a/src/utils/VectorUtils.ts b/src/utils/VectorUtils.ts new file mode 100644 index 0000000..7cc616d --- /dev/null +++ b/src/utils/VectorUtils.ts @@ -0,0 +1,1022 @@ +import { Block, Vector3 } from "@minecraft/server"; + +/** + * Represents a 3D vector as an array of three numbers [x, y, z]. + */ +export type Vector3Array = [number, number, number]; + +/** + * Represents a 3D vector with x, y, and z components. + */ +export class Vec3 { + static EPSILON: number = 1e-8; + + static back = new Vec3(0, 0, -1); + static down = new Vec3(0, -1, 0); + static forward = new Vec3(0, 0, 1); + static left = new Vec3(-1, 0, 0); + static one = new Vec3(1, 1, 1); + static right = new Vec3(1, 0, 0); + static up = new Vec3(0, 1, 0); + static zero = new Vec3(0, 0, 0); + + x: number; + y: number; + z: number; + + constructor(x: number, y: number, z: number); + constructor(other: Vec3); + constructor(other: Vector3); + constructor(other: Block); + constructor(xOrOther: number | Vec3 | Vector3 | Block, y?: number, z?: number) { + if (typeof xOrOther === "number" && y !== undefined && z !== undefined) { + // Handle the case where three numbers are passed + this.x = xOrOther; + this.y = y; + this.z = z; + } else if (xOrOther instanceof Vec3 || xOrOther instanceof Block || typeof xOrOther === 'object') { + // Handle the case where another Vec3 is passed + this.x = xOrOther.x; + this.y = xOrOther.y; + this.z = xOrOther.z; + } else { + throw new Error("Invalid constructor arguments"); + } + } + /** + * Creates a Vec3 instance from an array of numbers. + * @param array The array containing three numbers representing x, y, and z. + * @returns A new Vec3 instance. + */ + static fromArray(array: Vector3Array): Vec3 { + return new Vec3(array[0], array[1], array[2]); + } + + /** + * Creates a Vec3 instance from a string in the format "x, y, z". + * @param str The stringified vector. + * @returns A new Vec3 instance. + */ + static fromString(str: string): Vec3 { + const parts = str.trim().split(',').map(part => part.trim()); + if (parts.length !== 3) { + throw new Error('Invalid vector format. Expected "x, y, z".'); + } + const [x, y, z] = parts.map(part => parseFloat(part)); + if (isNaN(x) || isNaN(y) || isNaN(z)) { + throw new Error('Invalid vector components. Ensure all components are numbers.'); + } + return new Vec3(x, y, z); + } + + /** + * Calculates the length of the vector. + * @returns The length of the vector. + */ + length(): number { + return Math.hypot(this.x, this.y, this.z); + } + + /** + * Calculates the squared length of the vector. + * @returns The squared length of the vector. + */ + lengthSquared(): number { + return Math.pow(this.x, 2) + Math.pow(this.y, 2) + Math.pow(this.z, 2); + } + + /** + * Normalizes the vector. + * @returns The normalized vector. + */ + normalize(): Vec3 { + const length = this.length(); + return length === 0 ? this : this.scale(1 / length); + } + + /** + * Adds another vector to this vector. + * @param other The vector to add. + * @returns The resulting vector after addition. + */ + add(other: Vec3 | Vector3): Vec3 { + return new Vec3(this.x + other.x, this.y + other.y, this.z + other.z); + } + static add(a: Vec3 | Vector3, b: Vec3 | Vector3): Vec3 { + return new Vec3(b.x + a.x, b.y + a.y, b.z + a.z); + } + + /** + * Subtracts another vector from this vector. + * @param other The vector to subtract. + * @returns The resulting vector after subtraction. + */ + sub(other: Vec3 | Vector3): Vec3 { + return new Vec3(this.x - other.x, this.y - other.y, this.z - other.z); + } + static sub(a: Vec3 | Vector3, b: Vec3 | Vector3): Vec3 { + return new Vec3(a.x - b.x, a.y - b.y, a.z - b.z); + } + + /** + * Multiplies this vector component-wise with another vector. + * @param other The vector to multiply with. + * @returns The resulting vector after multiplication. + */ + mul(other: Vec3 | Vector3): Vec3 { + return new Vec3(this.x * other.x, this.y * other.y, this.z * other.z); + } + static mul(a: Vec3 | Vector3, b: Vec3 | Vector3): Vec3 { + return new Vec3(a.x * b.x, a.y * b.y, a.z * b.z); + } + + /** + * Divides this vector component-wise by another vector. + * @param other The vector to divide by. + * @returns The resulting vector after division. + */ + div(other: Vec3 | Vector3): Vec3 { + return new Vec3(this.x / other.x, this.y / other.y, this.z / other.z); + } + static div(a: Vec3 | Vector3, b: Vec3 | Vector3): Vec3 { + return new Vec3(a.x / b.x, a.y / b.y, a.z / b.z); + } + + /** + * Scales this vector by a scalar value. + * @param scalar The scalar value to scale by. + * @returns The resulting scaled vector. + */ + scale(scalar: number): Vec3 { + return new Vec3(this.x * scalar, this.y * scalar, this.z * scalar); + } + + /** + * Calculates the dot product of this vector with another vector. + * @param other The other vector. + * @returns The dot product. + */ + dot(other: Vec3 | Vector3): number { + return this.x * other.x + this.y * other.y + this.z * other.z; + } + + /** + * Calculates the cross product of this vector with another vector. + * @param other The other vector. + * @returns The cross product vector. + */ + cross(other: Vec3 | Vector3): Vec3 { + const x = this.y * other.z - this.z * other.y; + const y = this.z * other.x - this.x * other.z; + const z = this.x * other.y - this.y * other.x; + return new Vec3(x, y, z); + } + + /** + * Checks if this Vec3 object is equal to another Vec3 object. + * @param other The other Vec3 object to compare with. + * @returns Returns true if the two Vec3 objects are equal, false otherwise. + */ + equals(other: Vec3 | Vector3 | Block): boolean { + return this.x === other.x && this.y === other.y && this.z === other.z; + } + static equals(a: Vec3 | Vector3 | Block, b: Vec3 | Vector3 | Block): boolean { + return b.x === a.x && b.y === a.y && b.z === a.z; + } + + /** + * Checks if the current Vec3 is approximately equal to another Vec3 within a given tolerance. + * @param other The other Vec3 to compare with. + * @param tolerance The tolerance value for the comparison. Defaults to Vec3.EPSILON. + * @returns True if the Vec3 is approximately equal to the other Vec3 within the tolerance, false otherwise. + */ + equalsEpsilon(other: Vec3, tolerance: number = Vec3.EPSILON): boolean { + return ( + Math.abs(this.x - other.x) <= tolerance && + Math.abs(this.y - other.y) <= tolerance && + Math.abs(this.z - other.z) <= tolerance + ); + } + + /** + * Checks if this Vec3 is approximately equal to another Vec3. + * @param other The other Vec3 to compare with. + * @returns True if the Vec3 is approximately equal, false otherwise. + */ + equalsApprox(other: Vec3): boolean { + return this.equalsEpsilon(other, Vec3.EPSILON); + } + + /** + * Calculates the triple scalar product of this vector with two other vectors. + * The triple scalar product is defined as the dot product of the cross product of two vectors with this vector. + * @param b The second vector. + * @param c The third vector. + * @returns The triple scalar product of the three vectors. + */ + tripleScalar(b: Vec3, c: Vec3): number { + return this.dot(b.cross(c)); + } + + /** + * Calculates the barycentric coordinates of a point within a triangle defined by three vertices. + * @param v1 The first vertex of the triangle. + * @param v2 The second vertex of the triangle. + * @param v3 The third vertex of the triangle. + * @returns The barycentric coordinates of the point. + */ + barycentricCoordinates(v1: Vec3, v2: Vec3, v3: Vec3): Vec3 { + const v2MinusV1 = v2.sub(v1); + const v3MinusV1 = v3.sub(v1); + const pointMinusV1 = this.sub(v1); + const dot00 = v2MinusV1.dot(v2MinusV1); + const dot01 = v2MinusV1.dot(v3MinusV1); + const dot02 = v2MinusV1.dot(pointMinusV1); + const dot11 = v3MinusV1.dot(v3MinusV1); + const dot12 = v3MinusV1.dot(pointMinusV1); + const invDenom = 1 / (dot00 * dot11 - dot01 * dot01); + const u = (dot11 * dot02 - dot01 * dot12) * invDenom; + const v = (dot00 * dot12 - dot01 * dot02) * invDenom; + return new Vec3(1 - u - v, u, v); + } + + /** + * Rotates the vector by the specified angle around the given axis. + * @param angle The angle of rotation in radians. + * @param axis The axis of rotation represented as a Vec3 object. + * @returns The rotated vector. + */ + rotate(angle: number, axis: Vec3): Vec3 { + const cosAngle = Math.cos(angle); + const sinAngle = Math.sin(angle); + const oneMinusCos = 1 - cosAngle; + const rotationMatrix = [ + [ + oneMinusCos * axis.x * axis.x + cosAngle, + oneMinusCos * axis.x * axis.y - sinAngle * axis.z, + oneMinusCos * axis.x * axis.z + sinAngle * axis.y + ], + [ + oneMinusCos * axis.y * axis.x + sinAngle * axis.z, + oneMinusCos * axis.y * axis.y + cosAngle, + oneMinusCos * axis.y * axis.z - sinAngle * axis.x + ], + [ + oneMinusCos * axis.z * axis.x - sinAngle * axis.y, + oneMinusCos * axis.z * axis.y + sinAngle * axis.x, + oneMinusCos * axis.z * axis.z + cosAngle + ] + ]; + return this.matrixProduct(rotationMatrix); + } + + /** + * Calculates the product of this vector and a given matrix. + * @param matrix The matrix to multiply with. + * @returns The resulting vector after the matrix multiplication. + * @throws Error if the matrix dimensions are invalid. + */ + matrixProduct(matrix: number[][]): Vec3 { + if ( + matrix.length !== 3 || + matrix[0].length !== 3 || + matrix[1].length !== 3 || + matrix[2].length !== 3 + ) { + throw new Error('Invalid matrix dimensions'); + } + const x = this.x * matrix[0][0] + this.y * matrix[0][1] + this.z * matrix[0][2]; + const y = this.x * matrix[1][0] + this.y * matrix[1][1] + this.z * matrix[1][2]; + const z = this.x * matrix[2][0] + this.y * matrix[2][1] + this.z * matrix[2][2]; + return new Vec3(x, y, z); + } + + /** + * Returns a new Vec3 with the absolute values of each component. + * @returns A new Vec3 containing the absolute values of this Vec3's components. + */ + abs(): Vec3 { + return new Vec3(Math.abs(this.x), Math.abs(this.y), Math.abs(this.z)); + } + + /** + * Calculates the distance between this Vec3 and another Vec3. + * @param other The other Vec3. + * @returns The distance between this Vec3 and the other Vec3. + */ + distance(other: Vec3 | Vector3): number { + return this.sub(other).length(); + } + static distance(a: Vec3 | Vector3 | Block, b: Vec3 | Vector3 | Block): number { + const dx = b.x - a.x; + const dy = b.y - a.y; + const dz = b.z - a.z; + const distance = Math.hypot(dx, dy, dz); + + return distance; + } + + /** + * Calculates the squared distance between this Vec3 and another Vec3. + * @param other The other Vec3. + * @returns The squared distance between this Vec3 and the other Vec3. + */ + distanceSquared(other: Vec3): number { + return this.sub(other).lengthSquared(); + } + + /** + * Calculates the angle between this Vec3 and another Vec3 in radians. + * @param other The other Vec3. + * @returns The angle between this Vec3 and the other Vec3 in radians. + */ + angle(other: Vec3): number { + return Math.acos(this.dot(other) / (this.length() * other.length())); + } + + /** + * Projects this Vec3 onto another Vec3. + * @param other The Vec3 onto which to project. + * @returns The projection of this Vec3 onto the other Vec3. + * @throws Error if the other Vec3 is a zero vector. + */ + projectOnto(other: Vec3): Vec3 { + const lengthSquared = other.lengthSquared(); + if (lengthSquared === 0) { + throw new Error('Cannot project onto a zero vector'); + } + return other.scale(this.dot(other) / lengthSquared); + } + + /** + * Rejects this Vec3 from another Vec3. + * @param other The Vec3 from which to reject. + * @returns The rejection of this Vec3 from the other Vec3. + */ + rejectFrom(other: Vec3): Vec3 { + return this.sub(this.projectOnto(other)); + } + + /** + * Reflects this Vec3 across another Vec3. + * @param other The Vec3 across which to reflect. + * @returns The reflection of this Vec3 across the other Vec3. + */ + reflect(other: Vec3): Vec3 { + return this.sub(this.projectOnto(other).scale(2)); + } + + /** + * Refracts this Vec3 through a surface with a given normal and refraction index. + * @param normal The surface normal. + * @param eta The refraction index. + * @returns The refraction of this Vec3 through the surface. + */ + refract(normal: Vec3, eta: number): Vec3 { + const dot = this.dot(normal); + const k = 1 - eta * eta * (1 - dot * dot); + return k < 0 ? new Vec3(0, 0, 0) : this.scale(eta).sub(normal.scale(eta * dot + Math.sqrt(k))); + } + + /** + * Performs linear interpolation between this Vec3 and another Vec3. + * @param other The other Vec3. + * @param t The interpolation parameter. + * @returns The interpolated Vec3. + */ + lerp(other: Vec3, t: number): Vec3 { + return this.add(other.sub(this).scale(t)); + } + static lerp(a: Vec3 | Vector3 | Block, b: Vec3 | Vector3 | Block, t: number): Vec3 { + const dest = { x: a.x, y: a.y, z: a.z}; + dest.x += (b.x - a.x) * t; + dest.y += (b.y - a.y) * t; + dest.z += (b.z - a.z) * t; + return new Vec3(dest); + } + + /** + * Performs spherical linear interpolation between this Vec3 and another Vec3. + * @param other The other Vec3. + * @param t The interpolation parameter. + * @returns The interpolated Vec3. + */ + slerp(other: Vec3, t: number): Vec3 { + const dot = this.dot(other); + const theta = Math.acos(dot); + const sinTheta = Math.sin(theta); + const scale1 = Math.sin((1 - t) * theta) / sinTheta; + const scale2 = Math.sin(t * theta) / sinTheta; + return this.scale(scale1).add(other.scale(scale2)); + } + + /** + * Performs Hermite interpolation between this Vec3 and another Vec3 with given tangents. + * @param other The other Vec3. + * @param t The interpolation parameter. + * @param tangent1 The tangent at the start. + * @param tangent2 The tangent at the end. + * @returns The interpolated Vec3. + */ + hermite(other: Vec3, t: number, tangent1: Vec3, tangent2: Vec3): Vec3 { + const t2 = t * t; + const t3 = t2 * t; + const h1 = 2 * t3 - 3 * t2 + 1; + const h2 = -2 * t3 + 3 * t2; + const h3 = t3 - 2 * t2 + t; + const h4 = t3 - t2; + return this.scale(h1).add(other.scale(h2)).add(tangent1.scale(h3)).add(tangent2.scale(h4)); + } + + /** + * Calculates the quadratic bezier curve of a vector based on current step. + * @param start starting vector + * @param control control vector + * @param end end vector + * @param t current steps from start to end + * @returns + */ + static quadracticBezier(start: Vector3, control: Vector3, end: Vector3, t: number): Vector3 { + return { + x: (1 - t) * (1 - t) * start.x + 2 * (1 - t) * t * control.x + t * t * end.x, + y: (1 - t) * (1 - t) * start.y + 2 * (1 - t) * t * control.y + t * t * end.y, + z: (1 - t) * (1 - t) * start.z + 2 * (1 - t) * t * control.z + t * t * end.z + }; + } + /** + * Calculates a point on the Bezier curve defined by control points. + * @param controlPoints The control points of the Bezier curve. + * @param t The parameter determining the point on the curve. + * @returns The point on the Bezier curve. + */ + bezier(controlPoints: Vec3[], t: number): Vec3 { + const n = controlPoints.length; + let result = new Vec3(0, 0, 0); + for (let i = 0; i < n; i++) { + const coefficient = + this.binomialCoefficient(n - 1, i) * Math.pow(1 - t, n - 1 - i) * Math.pow(t, i); + result = result.add(controlPoints[i].scale(coefficient)); + } + return result; + } + + /** + * Calculates the binomial coefficient (n choose k). + * @param n The total number of items. + * @param k The number of items to choose. + * @returns The binomial coefficient (n choose k). + */ + binomialCoefficient(n: number, k: number): number { + if (k < 0 || k > n) { + return 0; + } + if (k === 0 || k === n) { + return 1; + } + return this.binomialCoefficient(n - 1, k - 1) + this.binomialCoefficient(n - 1, k); + } + + /** + * Calculates Catmull-Rom interpolation. + * @param controlPoints The control points array. + * @param t The interpolation parameter. + * @param alpha The tension parameter (default is 0.5). + * @returns The interpolated vector. + */ + catmullRom(controlPoints: Vec3[], t: number, alpha: number = 0.5): Vec3 { + const p0 = controlPoints[0]; + const p1 = controlPoints[1]; + const p2 = controlPoints[2]; + const p3 = controlPoints[3]; + const t2 = t * t; + const t3 = t2 * t; + const h1 = -alpha * t3 + 2 * alpha * t2 - alpha * t; + const h2 = (2 - alpha) * t3 + (alpha - 3) * t2 + 1; + const h3 = (alpha - 2) * t3 + (3 - 2 * alpha) * t2 + alpha * t; + const h4 = alpha * t3 - alpha * t2; + return p0.scale(h1).add(p1.scale(h2)).add(p2.scale(h3)).add(p3.scale(h4)); + } + + /** + * Returns the component-wise minimum of this vector and another vector. + * @param other The other vector. + * @returns The component-wise minimum vector. + */ + min(other: Vec3): Vec3 { + return new Vec3(Math.min(this.x, other.x), Math.min(this.y, other.y), Math.min(this.z, other.z)); + } + + /** + * Returns the component-wise maximum of this vector and another vector. + * @param other The other vector. + * @returns The component-wise maximum vector. + */ + max(other: Vec3): Vec3 { + return new Vec3(Math.max(this.x, other.x), Math.max(this.y, other.y), Math.max(this.z, other.z)); + } + + /** + * Clamps this vector to a specified range. + * @param min The minimum range vector. + * @param max The maximum range vector. + * @returns The clamped vector. + */ + clamp(min: Vec3, max: Vec3): Vec3 { + return this.max(min).min(max); + } + + /** + * Rounds each component of this vector downwards to the nearest integer. + * @returns The vector with rounded components. + */ + floor(): Vec3 { + return new Vec3(Math.floor(this.x), Math.floor(this.y), Math.floor(this.z)); + } + + /** + * Rounds each component of this vector upwards to the nearest integer. + * @returns The vector with rounded components. + */ + ceil(): Vec3 { + return new Vec3(Math.ceil(this.x), Math.ceil(this.y), Math.ceil(this.z)); + } + + /** + * Rounds each component of this vector to the nearest integer. + * @returns The vector with rounded components. + */ + round(): Vec3 { + return new Vec3(Math.round(this.x), Math.round(this.y), Math.round(this.z)); + } + + /** + * Calculates the square root of each component of this vector. + * @returns The vector with square root of components. + */ + sqrt(): Vec3 { + return new Vec3(Math.sqrt(this.x), Math.sqrt(this.y), Math.sqrt(this.z)); + } + + /** + * Raises each component of this vector to the power of a specified exponent. + * @param exponent The exponent to raise each component to. + * @returns The vector with components raised to the power of exponent. + */ + pow(exponent: number): Vec3 { + return new Vec3(Math.pow(this.x, exponent), Math.pow(this.y, exponent), Math.pow(this.z, exponent)); + } + + /** + * Calculates the exponential of each component of this vector (e^x). + * @returns The vector with exponential components. + */ + exp(): Vec3 { + return new Vec3(Math.exp(this.x), Math.exp(this.y), Math.exp(this.z)); + } + + /** + * Returns a new Vec3 where each component is the natural logarithm of the corresponding component of the original Vec3. + * @returns A new Vec3 containing the natural logarithms of the components of the original Vec3. + */ + log(): Vec3 { + return new Vec3(Math.log(this.x), Math.log(this.y), Math.log(this.z)); + } + + /** + * Returns a new Vec3 where each component is the sine of the corresponding component of the original Vec3. + * @returns A new Vec3 containing the sine values of the components of the original Vec3. + */ + sin(): Vec3 { + return new Vec3(Math.sin(this.x), Math.sin(this.y), Math.sin(this.z)); + } + + /** + * Returns a new Vec3 where each component is the cosine of the corresponding component of the original Vec3. + * @returns A new Vec3 containing the cosine values of the components of the original Vec3. + */ + cos(): Vec3 { + return new Vec3(Math.cos(this.x), Math.cos(this.y), Math.cos(this.z)); + } + + /** + * Returns a new Vec3 where each component is the tangent of the corresponding component of the original Vec3. + * @returns A new Vec3 containing the tangent values of the components of the original Vec3. + */ + tan(): Vec3 { + return new Vec3(Math.tan(this.x), Math.tan(this.y), Math.tan(this.z)); + } + + /** + * Returns a new Vec3 where each component is the arcsine (in radians) of the corresponding component of the original Vec3. + * @returns A new Vec3 containing the arcsine values (in radians) of the components of the original Vec3. + */ + asin(): Vec3 { + return new Vec3(Math.asin(this.x), Math.asin(this.y), Math.asin(this.z)); + } + + /** + * Returns a new Vec3 where each component is the arccosine (in radians) of the corresponding component of the original Vec3. + * @returns A new Vec3 containing the arccosine values (in radians) of the components of the original Vec3. + */ + acos(): Vec3 { + return new Vec3(Math.acos(this.x), Math.acos(this.y), Math.acos(this.z)); + } + + /** + * Returns a new Vec3 where each component is the arctangent (in radians) of the corresponding component of the original Vec3. + * @returns A new Vec3 containing the arctangent values (in radians) of the components of the original Vec3. + */ + atan(): Vec3 { + return new Vec3(Math.atan(this.x), Math.atan(this.y), Math.atan(this.z)); + } + + /** + * Returns a new Vec3 where each component is the hyperbolic sine of the corresponding component of the original Vec3. + * @returns A new Vec3 containing the hyperbolic sine values of the components of the original Vec3. + */ + sinh(): Vec3 { + return new Vec3(Math.sinh(this.x), Math.sinh(this.y), Math.sinh(this.z)); + } + + /** + * Returns a new Vec3 where each component is the hyperbolic cosine of the corresponding component of the original Vec3. + * @returns A new Vec3 containing the hyperbolic cosine values of the components of the original Vec3. + */ + cosh(): Vec3 { + return new Vec3(Math.cosh(this.x), Math.cosh(this.y), Math.cosh(this.z)); + } + + /** + * Returns a new Vec3 where each component is the hyperbolic tangent of the corresponding component of the original Vec3. + * @returns A new Vec3 containing the hyperbolic tangent values of the components of the original Vec3. + */ + tanh(): Vec3 { + return new Vec3(Math.tanh(this.x), Math.tanh(this.y), Math.tanh(this.z)); + } + + /** + * Returns a new Vec3 where each component is the inverse hyperbolic sine of the corresponding component of the original Vec3. + * @returns A new Vec3 containing the inverse hyperbolic sine values of the components of the original Vec3. + */ + asinh(): Vec3 { + return new Vec3(Math.asinh(this.x), Math.asinh(this.y), Math.asinh(this.z)); + } + + /** + * Returns a new Vec3 where each component is the inverse hyperbolic cosine of the corresponding component of the original Vec3. + * @returns A new Vec3 containing the inverse hyperbolic cosine values of the components of the original Vec3. + */ + acosh(): Vec3 { + return new Vec3(Math.acosh(this.x), Math.acosh(this.y), Math.acosh(this.z)); + } + + /** + * Returns a new Vec3 where each component is the inverse hyperbolic tangent of the corresponding component of the original Vec3. + * @returns A new Vec3 containing the inverse hyperbolic tangent values of the components of the original Vec3. + */ + atanh(): Vec3 { + return new Vec3(Math.atanh(this.x), Math.atanh(this.y), Math.atanh(this.z)); + } + + /** + * Returns a new Vec3 where each component is the sign of the corresponding component of the original Vec3. + * @returns A new Vec3 containing the sign values of the components of the original Vec3. + */ + sign(): Vec3 { + return new Vec3(Math.sign(this.x), Math.sign(this.y), Math.sign(this.z)); + } + + /** + * Returns a new Vec3 where each component is the fractional part of the corresponding component of the original Vec3. + * @returns A new Vec3 containing the fractional parts of the components of the original Vec3. + */ + fract(): Vec3 { + return new Vec3(this.x - Math.floor(this.x), this.y - Math.floor(this.y), this.z - Math.floor(this.z)); + } + + /** + * Performs modulo operation element-wise with another Vec3. + * @param other The Vec3 to perform modulo with. + * @returns A new Vec3 containing the result of the modulo operation. + */ + mod(other: Vec3): Vec3 { + return new Vec3(this.x % other.x, this.y % other.y, this.z % other.z); + } + + /** + * Determines whether each component of this Vec3 is less than the corresponding component of the provided edge Vec3. + * @param edge The edge Vec3 to compare against. + * @returns A new Vec3 where each component is 0 if less than the edge, or 1 otherwise. + */ + step(edge: Vec3): Vec3 { + return new Vec3(this.x < edge.x ? 0 : 1, this.y < edge.y ? 0 : 1, this.z < edge.z ? 0 : 1); + } + + /** + * Performs smoothstep interpolation between two edge Vec3s. + * @param edge0 The starting edge Vec3. + * @param edge1 The ending edge Vec3. + * @returns A new Vec3 containing the result of smoothstep interpolation. + */ + smoothstep(edge0: Vec3, edge1: Vec3): Vec3 { + const t = this.sub(edge0).div(edge1.sub(edge0)).clamp(Vec3.zero, Vec3.one); + return t.mul(t).mul(new Vec3(3, 3, 3).sub(t.scale(2))); + } + + /** + * Transforms this Vec3 from world space to tangent space using the provided normal and tangent vectors. + * @param normal The normal vector in tangent space. + * @param tangent The tangent vector in tangent space. + * @returns A new Vec3 transformed to tangent space. + */ + toTangentSpace(normal: Vec3, tangent: Vec3): Vec3 { + const binormal = this.cross(normal); + const tangentMatrix = [ + [tangent.x, binormal.x, normal.x], + [tangent.y, binormal.y, normal.y], + [tangent.z, binormal.z, normal.z] + ]; + return this.matrixProduct(tangentMatrix); + } + + /** + * Generates Perlin noise for this Vec3 using the specified seed. + * @param seed The seed for generating Perlin noise. Default is 0. + * @returns A new Vec3 containing the Perlin noise values. + */ + perlinNoise(seed: number = 0): Vec3 { + const permutation = new Array(256); + for (let i = 0; i < 256; i++) { + permutation[i] = (seed + i) % 256; + } + const gradients = [ + new Vec3(1, 1, 0), + new Vec3(-1, 1, 0), + new Vec3(1, -1, 0), + new Vec3(-1, -1, 0), + new Vec3(1, 0, 1), + new Vec3(-1, 0, 1), + new Vec3(1, 0, -1), + new Vec3(-1, 0, -1), + new Vec3(0, 1, 1), + new Vec3(0, -1, 1), + new Vec3(0, 1, -1), + new Vec3(0, -1, -1), + new Vec3(1, 1, 0), + new Vec3(0, -1, 1), + new Vec3(-1, 1, 0), + new Vec3(0, -1, -1) + ]; + const fade = (t: number) => t * t * t * (t * (t * 6 - 15) + 10); + const dotProduct = (grad: Vec3, x: number, y: number, z: number) => + grad.x * x + grad.y * y + grad.z * z; + const unitX = Math.floor(this.x) & 255; + const unitY = Math.floor(this.y) & 255; + const unitZ = Math.floor(this.z) & 255; + const relX = this.x - Math.floor(this.x); + const relY = this.y - Math.floor(this.y); + const relZ = this.z - Math.floor(this.z); + const u = fade(relX); + const v = fade(relY); + const w = fade(relZ); + const A = permutation[unitX] + unitY; + const AA = permutation[A] + unitZ; + const AB = permutation[A + 1] + unitZ; + const B = permutation[unitX + 1] + unitY; + const BA = permutation[B] + unitZ; + const BB = permutation[B + 1] + unitZ; + const gradAA = gradients[permutation[AA] % 16]; + const gradAB = gradients[permutation[AB] % 16]; + const gradBA = gradients[permutation[BA] % 16]; + const gradBB = gradients[permutation[BB] % 16]; + const lerpX1 = dotProduct(gradAA, relX, relY, relZ); + const lerpX2 = dotProduct(gradBA, relX - 1, relY, relZ); + const lerpX3 = dotProduct(gradAB, relX, relY - 1, relZ); + const lerpX4 = dotProduct(gradBB, relX - 1, relY - 1, relZ); + const lerpX5 = dotProduct(gradAA, relX, relY, relZ - 1); + const lerpX6 = dotProduct(gradBA, relX - 1, relY, relZ - 1); + const lerpX7 = dotProduct(gradAB, relX, relY - 1, relZ - 1); + const lerpX8 = dotProduct(gradBB, relX - 1, relY - 1, relZ - 1); + const lerpY1 = this.lerp(new Vec3(lerpX1, lerpX2, lerpX3), v); + const lerpY2 = this.lerp(new Vec3(lerpX4, lerpX5, lerpX6), v); + const lerpY3 = this.lerp(new Vec3(lerpX7, lerpX8, lerpX1), v); + const lerpY4 = this.lerp(new Vec3(lerpX2, lerpX3, lerpX4), v); + const lerpZ1 = this.customLerp(lerpY1, lerpY2, w); + const lerpZ2 = this.customLerp(lerpY3, lerpY4, w); + const finalNoiseValue = this.customLerp(lerpZ1, lerpZ2, w); + return finalNoiseValue; + } + + /** + * Performs a custom linear interpolation between two Vec3 objects. + * @param a The starting Vec3 object. + * @param b The ending Vec3 object. + * @param t The interpolation factor between 0 and 1. + * @returns The interpolated Vec3 object. + */ + customLerp(a: Vec3, b: Vec3, t: number): Vec3 { + const x = a.x + t * (b.x - a.x); + const y = a.y + t * (b.y - a.y); + const z = a.z + t * (b.z - a.z); + return new Vec3(x, y, z); + } + + /** + * Calculates the geodesic distance between this Vec3 and another Vec3 assuming they are points on the surface of a unit sphere. + * @param other The other Vec3 to calculate the distance to. + * @returns The geodesic distance between this Vec3 and the other Vec3. + */ + geodesicDistance(other: Vec3): number { + const radius = 1; + const angle = this.angle(other); + const distance = radius * angle; + return distance; + } + + /** + * Calculates a point on a Catmull-Rom spline given four control points and a parameter 't'. + * @param p0 The first control point. + * @param p1 The second control point. + * @param p2 The third control point. + * @param p3 The fourth control point. + * @param t The parameter controlling the position along the spline (0 <= t <= 1). + * @returns The point on the Catmull-Rom spline corresponding to parameter 't'. + */ + catmullRomSpline(p0: Vec3, p1: Vec3, p2: Vec3, p3: Vec3, t: number): Vec3 { + const t2 = t * t; + const t3 = t2 * t; + const h1 = -0.5 * t3 + t2 - 0.5 * t; + const h2 = 1.5 * t3 - 2.5 * t2 + 1.0; + const h3 = -1.5 * t3 + 2.0 * t2 + 0.5 * t; + const h4 = 0.5 * t3 - 0.5 * t2; + return p0.scale(h1).add(p1.scale(h2)).add(p2.scale(h3)).add(p3.scale(h4)); + } + + /** + * Calculates the spherical angle between this vector and another vector. + * @param other The other vector. + * @returns The spherical angle between this vector and the 'other' vector in radians. + */ + sphericalAngle(other: Vec3): number { + const dotProduct = this.dot(other); + const angle = Math.acos(dotProduct / (this.length() * other.length())); + return angle; + } + + /** + * Calculates the complex conjugate of this vector. + * @returns The complex conjugate of this vector. + */ + complexConjugate(): Vec3 { + return new Vec3(this.x, -this.y, -this.z); + } + + /** + * Scales the vector non-uniformly by the provided scaling factors. + * @param scalingFactors The scaling factors for each axis (x, y, z). + * @returns The resulting vector after non-uniform scaling. + */ + nonUniformScale(scalingFactors: Vec3): Vec3 { + return new Vec3(this.x * scalingFactors.x, this.y * scalingFactors.y, this.z * scalingFactors.z); + } + + /** + * Calculates the surface normal of a parametric surface at the specified (u, v) coordinates. + * @param u The u parameter. + * @param v The v parameter. + * @returns The surface normal vector at the specified (u, v) coordinates. + */ + surfaceNormal(u: number, v: number): Vec3 { + const tangentU = this.partialDerivativeU(u, v); + const tangentV = this.partialDerivativeV(u, v); + const normal = tangentU.cross(tangentV).normalize(); + return normal; + } + + /** + * Calculates the partial derivative of the parametric surface with respect to the u parameter. + * @param u The u parameter. + * @param v The v parameter. + * @returns The partial derivative with respect to u at the specified (u, v) coordinates. + * @private + */ + private partialDerivativeU(u: number, v: number): Vec3 { + const deltaU = 0.0001; + const point1 = this.evaluateParametricSurface(u - deltaU, v); + const point2 = this.evaluateParametricSurface(u + deltaU, v); + const tangentU = point2.sub(point1).scale(1 / (2 * deltaU)); + return tangentU; + } + + /** + * Calculates the partial derivative of the parametric surface with respect to the v parameter. + * @param u The u parameter. + * @param v The v parameter. + * @returns The partial derivative with respect to v at the specified (u, v) coordinates. + * @private + */ + private partialDerivativeV(u: number, v: number): Vec3 { + const deltaV = 0.0001; + const point1 = this.evaluateParametricSurface(u, v - deltaV); + const point2 = this.evaluateParametricSurface(u, v + deltaV); + const tangentV = point2.sub(point1).scale(1 / (2 * deltaV)); + return tangentV; + } + + /** + * Evaluates the parametric surface at the specified (u, v) coordinates. + * @param u The u parameter. + * @param v The v parameter. + * @returns The point on the parametric surface corresponding to the specified (u, v) coordinates. + * @private + */ + private evaluateParametricSurface(u: number, v: number): Vec3 { + const radius = 1.0; + const x = radius * Math.cos(u) * Math.sin(v); + const y = radius * Math.sin(u) * Math.sin(v); + const z = radius * Math.cos(v); + return new Vec3(x, y, z); + } + + /** + * Calculates the divergence of the vector. + * The divergence is the sum of the differences between each component of the vector and 0. + * @returns The divergence value. + */ + divergence(): number { + const dx = this.x - 0; + const dy = this.y - 0; + const dz = this.z - 0; + return dx + dy + dz; + } + + /** + * Calculates the curl of the vector. + * The curl is a vector that represents the rotation of a vector field. + * @returns The curl vector. + */ + curl(): Vec3 { + const i = new Vec3(1, 0, 0); + const j = new Vec3(0, 1, 0); + const k = new Vec3(0, 0, 1); + const f_z = this.add(k).z; + const b_z = this.sub(k).z; + const e_y = this.add(j).y; + const c_y = this.sub(j).y; + const d_x = this.add(i).x; + const f_x = this.sub(i).x; + const curl_x = f_z - b_z - (e_y - c_y); + const curl_y = d_x - f_x - (f_z - b_z); + const curl_z = e_y - this.y - (this.x - d_x); + return new Vec3(curl_x, curl_y, curl_z); + } + + /** + * Calculates the gradient of a scalar field at the current position. + * @param scalarField The scalar field function. + * @param epsilon The small value used for numerical differentiation. Default is 1e-6. + * @returns The gradient vector. + */ + gradient(scalarField: (position: Vec3) => number, epsilon: number = 1e-6): Vec3 { + const dx = + (scalarField(this.add(new Vec3(epsilon, 0, 0))) - + scalarField(this.sub(new Vec3(epsilon, 0, 0)))) / + (2 * epsilon); + const dy = + (scalarField(this.add(new Vec3(0, epsilon, 0))) - + scalarField(this.sub(new Vec3(0, epsilon, 0)))) / + (2 * epsilon); + const dz = + (scalarField(this.add(new Vec3(0, 0, epsilon))) - + scalarField(this.sub(new Vec3(0, 0, epsilon)))) / + (2 * epsilon); + return new Vec3(dx, dy, dz); + } + + /** + * Converts the Vec3 object to an array of numbers. + * The array contains the x, y, and z values of the Vec3 object in that order. + * @returns An array of numbers representing the x, y, and z values of the Vec3 object. + */ + toArray(): Vector3Array { + return [this.x, this.y, this.z]; + } + + /** + * Returns a string representation of the Vec3 object. + * The string is formatted as "(x, y, z)". + * @returns A string representation of the Vec3 object. + */ + toString(): string { + return `${this.x}, ${this.y}, ${this.z}`; + } + + /** + * Creates a string representation of a Vec3. + * @param other The Vec3 object. + * @returns A string representing the vector. + */ + static toString(other: Vec3 | Vector3 | Block): string { + return `${other.x}, ${other.y}, ${other.z}`; + } +} diff --git a/src/utils/form_builder.ts b/src/utils/form_builder.ts new file mode 100644 index 0000000..41c0b6b --- /dev/null +++ b/src/utils/form_builder.ts @@ -0,0 +1,26 @@ +export class FormBuilder { + values?: string[]; + defaultValue: T; + name: string; + + constructor(name: string) { + this.name = name; + this.values = []; + } + + createToggle(defaultValue: boolean): this { + this.defaultValue = defaultValue as T; + return this; + } + + createTextField(defaultValue: string): this { + this.defaultValue = defaultValue as T; + return this; + } + + createDropdown(dropDownOptions: string[], defaultValue: string): this { + this.defaultValue = defaultValue as T; + this.values = dropDownOptions as string[]; + return this; + } +} \ No newline at end of file diff --git a/src/utils/graph.ts b/src/utils/graph.ts new file mode 100644 index 0000000..5a754e7 --- /dev/null +++ b/src/utils/graph.ts @@ -0,0 +1,172 @@ +import { Block, IPlayerSpawnAfterEventSignal, Vector3 } from "@minecraft/server"; +import { Vec3 } from "./VectorUtils"; + +/** + * Hash function for a Block based on its location (Vector3). + */ +function hashBlock(block: Block): number { + const prime = 31; + let hash = 1; + + // Convert the block's coordinates to integers and apply the prime multiplier + hash = prime * hash + Math.imul(block.x | 0, prime); + hash = prime * hash + Math.imul(block.y | 0, prime); + hash = prime * hash + Math.imul(block.z | 0, prime); + + // Apply a bitwise shift to further mix the bits + hash ^= (hash << 13); + hash ^= (hash >> 7); + hash ^= (hash << 17); + + return hash >>> 0; // Ensure it's an unsigned 32-bit integer +} + +/** + * GraphNode class now stores a Block instead of just a Vector3. + */ +export class GraphNode { + public block: Block; + public neighbors: Set; + public index: number = 0; + + constructor(block: Block) { + this.block = block; + this.neighbors = new Set(); + } + + addNeighbor(node: GraphNode) { + this.neighbors.add(node); + } + + removeNeighbor(node: GraphNode) { + this.neighbors.delete(node); + } +} + +type GraphTraversalType = "DFS" | "BFS"; + +/** + * Graph class now stores Block objects in the nodes and uses block location for hashing and lookup. + */ +export class Graph { + private nodes: Map; + private hashes: number[]; + + constructor() { + this.nodes = new Map(); + this.hashes = []; + } + + getNode(param: Block): GraphNode | undefined; + getNode(param: Vec3): GraphNode | undefined; + getNode(param: Block | Vec3): GraphNode | undefined { + let node; + if(param instanceof Vec3) { + node = this.nodes.get(Vec3.toString(param)); + } else { + node = this.nodes.get(this.serializeLocation((param).location)); + } + return node; + } + + addNode(block: Block): GraphNode; + addNode(node: GraphNode): void; + addNode(param: Block | GraphNode): GraphNode | void { + if (param instanceof GraphNode) { + this.hashes.push(hashBlock(param.block)); + const key = this.serializeLocation(param.block.location); + param.index = this.nodes.size; + this.nodes.set(key, param); + return; // Since it's a GraphNode, you don't return anything + } else { + const key = this.serializeLocation(param.location); + let node = this.nodes.get(key); + if (!node) { + node = new GraphNode(param); + this.nodes.set(key, node); + } + this.hashes.push(hashBlock(node.block)); + node.index = this.nodes.size - 1; + return node; // Return the newly created or retrieved node + } + } + + removeNode(block: Block) { + const key = this.serializeLocation(block.location); + const node = this.nodes.get(key); + if (!node) return; + // Remove the node from its neighbors' adjacency lists + node.neighbors.forEach(neighbor => { + neighbor.removeNeighbor(node); + node.removeNeighbor(neighbor); + }); + this.hashes.splice(this.hashes.lastIndexOf(hashBlock(block)), 1); + this.nodes.delete(key); + } + + serializeLocation(location: Vector3): string { + return JSON.stringify(location); + } + + getSize(): number { + return this.nodes.size; + } + + traverse(startBlock: Block, traversalType: GraphTraversalType = "DFS", visit: (node: GraphNode) => void) { + const startNode = this.getNode(startBlock); + if (!startNode) { + return; + } + + const visited = new Set(); + const toVisit: GraphNode[] = [startNode]; + + while (toVisit.length > 0) { + const node = traversalType === "DFS" ? toVisit.pop()! : toVisit.shift()!; + + if (!visited.has(node)) { + visit(node); + visited.add(node); + + node.neighbors.forEach(neighbor => { + if (!visited.has(neighbor)) { + toVisit.push(neighbor); + } + }); + } + } + } + + hash(): number { + return this.hashes.reduce((accumulator, currentValue) => accumulator + currentValue, 0); + } + + *traverseIterative(startBlock: Block, traversalType: GraphTraversalType = "DFS"): Generator { + const startNode = this.getNode(startBlock); + if (!startNode) { + return; + } + + const visited = new Set(); + const toVisit: GraphNode[] = [startNode]; + + while (toVisit.length > 0) { + const node = traversalType === "DFS" ? toVisit.pop()! : toVisit.shift()!; + + if (!visited.has(node)) { + yield node; + visited.add(node); + + node.neighbors.forEach(neighbor => { + if (!visited.has(neighbor)) { + toVisit.push(neighbor); + } + }); + } + } + } + + isEqual(otherGraph: Graph): boolean { + return this.hash() === otherGraph.hash(); + } +} diff --git a/src/utils/logger.ts b/src/utils/logger.ts new file mode 100644 index 0000000..d622844 --- /dev/null +++ b/src/utils/logger.ts @@ -0,0 +1,54 @@ +// Import your configuration if necessary + +import { system } from "@minecraft/server"; +import { serverConfigurationCopy } from "index"; + +export enum LogLevel { + DEBUG = 'DEBUG', + INFO = 'INFO', + ERROR = 'ERROR', +} + +export class Logger { + private static level: LogLevel = serverConfigurationCopy.debug.defaultValue ? LogLevel.DEBUG : LogLevel.INFO; // Default log level + + static setLogLevel(level: LogLevel): void { + Logger.level = level; + } + + private static log(level: LogLevel, ...message: any[]): void { + const levels: LogLevel[] = [LogLevel.DEBUG, LogLevel.INFO, LogLevel.ERROR]; + const currentLevelIndex = levels.indexOf(Logger.level); + const logLevelIndex = levels.indexOf(level); + + if (logLevelIndex >= currentLevelIndex && serverConfigurationCopy.debug.defaultValue) { + // const timestamp = Date.now(); + const timestamp = system.currentTick; + const formattedMessage = `[${timestamp}] [${level}] - ${message}`; + + switch (level) { + case LogLevel.DEBUG: + console.warn(formattedMessage); + break; + case LogLevel.INFO: + console.log(formattedMessage); + break; + case LogLevel.ERROR: + console.error(formattedMessage); + break; + } + } + } + + static debug(...message: any[]): void { + if(serverConfigurationCopy.debug.defaultValue) Logger.log(LogLevel.DEBUG, message); + } + + static info(...message: any[]): void { + if(serverConfigurationCopy.debug.defaultValue) Logger.log(LogLevel.INFO, message); + } + + static error(...message: any[]): void { + if(serverConfigurationCopy.debug.defaultValue) Logger.log(LogLevel.ERROR, message); + } +} \ No newline at end of file diff --git a/src/utils/utilities.ts b/src/utils/utilities.ts new file mode 100644 index 0000000..ff0ae5b --- /dev/null +++ b/src/utils/utilities.ts @@ -0,0 +1,93 @@ +import { Block, Player, RawMessage, system } from "@minecraft/server"; +import { ActionFormData, ActionFormResponse, FormCancelationReason } from "@minecraft/server-ui"; + + +/** + * sleep + * @param {number} ticks Amount of time, in ticks, before the timeouts will be + * called. + * @returns {Promise} + */ +export function sleep(ticks: number): Promise { + // Script example for ScriptAPI + // Author: stackoverflow + // Project: https://github.com/JaylyDev/ScriptAPI + return new Promise((resolve) => { + system.runTimeout(resolve, ticks); + }); +}; + +/** + * Generates a random 16-character UUID. + * @returns {string} - A 16-character UUID. +*/ +export function generateUUID16(): string { + const characters = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789'; + let uuid = ''; + for (let i = 0; i < 16; i++) { + const randomIndex = Math.floor(Math.random() * characters.length); + uuid += characters[randomIndex]; + } + return uuid; +} + +/** + * IDK What to call this, but returns boolean if for every X amount of ticks + * @param tick Minecraft Ticks + * @returns + */ +export function ExecuteAtGivenTick(tick: number) { + return (system.currentTick % tick) === 0; +} + +export function SendMessageTo(executor: Player, rawMessage: RawMessage = { rawtext: [ {text: "Not Implemented Yet"} ] }) { + const formattedRawMessage = JSON.stringify(rawMessage); + executor.runCommandAsync(`tellraw ${executor.name} ` + formattedRawMessage); +} + +// Calculates the amount of items to be dropped in each stack. O(1) +function stackDistribution(number: number, groupSize: number = 64): number[] { + // Author: Adr-hyng + // Project: https://github.com/Adr-hyng-OSS/Lumber-Axe + const fullGroupsCount = Math.floor(number / groupSize); + const remainder = number % groupSize; + // Create an array with the size of each full group + const groups = new Array(fullGroupsCount).fill(groupSize); + // If there's a remainder, add it as the last group + if (remainder > 0) { + groups.push(remainder); + } + + return groups; +} + +export function hashBlock(block: Block): string { + const inputString = `${block.dimension.id}_${block.x}-${block.y}-${block.z}`; + + // Simple hash function (djb2) + let hash = 5381; + for (let i = 0; i < inputString.length; i++) { + hash = (hash * 33) ^ inputString.charCodeAt(i); + } + + // Convert hash to an unsigned 32-bit integer, and then to hexadecimal + return (hash >>> 0).toString(16); +} + + +async function forceShow(player: Player, form: ActionFormData, timeout: number = Infinity): Promise { + // Script example for ScriptAPI + // Author: Jayly#1397 + // Worldwidebrine#9037 + // Project: https://github.com/JaylyDev/ScriptAPI + const startTick: number = system.currentTick; + while ((system.currentTick - startTick) < timeout) { + const response: ActionFormResponse = await (form.show(player)).catch(er=>console.error(er,er.stack)) as ActionFormResponse; + if (response.cancelationReason !== FormCancelationReason.UserBusy) { + return response; + } + }; + throw new Error(`Timed out after ${timeout} ticks`); +} + +export {stackDistribution, forceShow} \ No newline at end of file diff --git a/tools/build.py b/tools/build.py index f531428..1a0ad8b 100644 --- a/tools/build.py +++ b/tools/build.py @@ -12,7 +12,22 @@ addon_name = json.loads(open('setup/mc_manifest.json', 'r').read()).get("header").get("bp_name") build_pack_name = addon_name[:addon_name.rfind(" BP")] -version_tag = 'v1.20.6x-110' +version_tag = 'v' + '.'.join(map(str, json.loads(open('setup/mc_manifest.json', 'r').read()).get("header").get("version"))) + + +def check_tsc_compiler(): + try: + # Run the command and capture the output + result = subprocess.Popen(f"where tsc", shell=True, stdout=subprocess.PIPE, stderr=subprocess.PIPE) + output, error = result.communicate() + if(error): raise Exception("Error from check_tsc_compiler") + paths = output.strip().splitlines() + if paths: + return paths[0] + except subprocess.CalledProcessError: + return None + +tsc_path = check_tsc_compiler() def handleError(err): if err: exit(err) @@ -65,7 +80,9 @@ def regExpSub(regEx, replace, file): subprocess.call([sys.executable, 'tools/sync2com-mojang.py', f'--dest={args.watch}'], stdout=subprocess.DEVNULL) print('Watch mode: press control-C to stop.') - tsc = subprocess.Popen('tsc -w', shell=True) + # tsc = subprocess.Popen('tsc -w', shell=True) + tsc = subprocess.Popen([tsc_path, '-w'], shell=True) + # Build settings file process_cfg = subprocess.Popen([sys.executable, 'tools/process_config.py', '-w', f'--target={watch_target}'], stdout=subprocess.DEVNULL) # Sync to com.mojang @@ -82,7 +99,7 @@ def regExpSub(regEx, replace, file): exit() else: print('building scripts...') - handleError(subprocess.call(['tsc', '-b'], shell=True)) + handleError(subprocess.call([tsc_path, '-b'], shell=True)) # Build manifests if args.init: @@ -131,12 +148,13 @@ def zipWriteDir(zip, dirname, arcname): """ -Add: -- [--init | -i] make it have choices of ["beh", "res", "all"] to select what to init -- version should be auto-generated from scripts module. -- [--module | -m] to select what module of script to use for BP-stable. choices: ["v1.20.0", "v1.20.1", "v1.20.10"] - -Bugs: -- Make it update the mc_manifest.json file, after init. +Commands: +--init (reinitialize bp / rp, creating new uuid and stuffs) +--watch (sync dev) +--target (create a debug or release version of addon) +--clean (clean bp/scripts) +--package-only (idk) +Add: +- [-c | --create ] - creates new uuid (bp, rp, res, script) for mc_manifest """ \ No newline at end of file diff --git a/tools/process_config.py b/tools/process_config.py index a5d3b06..c292562 100644 --- a/tools/process_config.py +++ b/tools/process_config.py @@ -16,10 +16,12 @@ settings = { "debug": { "description": "Enables debug messages to content logs.", + "name": "Debug Mode", "default": args.target == 'debug' } } version_str = '' +configuration_path = "configuration"; def compute_hash(filename): with open(filename, 'rb') as f: @@ -33,18 +35,26 @@ def generateScript(isServer): result = '' if isServer: result += 'import { variables } from "@minecraft/server-admin";\n\n' - - result += 'export default {\n' + result += 'import { FormBuilder } from "utils/form_builder";\n' + result += 'import { cloneConfiguration } from "./configuration_handler";\n\n' + result += 'export const serverConfiguration = {\n' for name, data in settings.items(): if isServer: result += f' {name}: variables.get("{name}"),\n' else: value = data["default"] + form_name = data["name"] - if type(value) is str: - value = f'"{value}"' + if type(value) is str or type(value) is int: + value = f'new FormBuilder("{form_name}").createTextField("{value}")' elif type(value) is bool: - value = "true" if value else "false" + value = f'new FormBuilder("{form_name}").createToggle(true)' if value else f'new FormBuilder("{form_name}").createToggle(false)' + elif type(value) is list: + if len(value) > 0: + list_representation = repr(value) + value = f'new FormBuilder("{form_name}").createDropdown({list_representation}, "{value[0]}")' + else: + value = f'new FormBuilder("{form_name}").createDropdown(["Empty"], "")' result += ' /**\n' for line in data['description'].splitlines(): @@ -53,6 +63,9 @@ def generateScript(isServer): result += f' {name}: {value},\n' result += '};\n\n' + result += 'export let serverConfigurationCopy = cloneConfiguration(serverConfiguration);\n' + result += 'export let setServerConfiguration = (newServerConfig) => serverConfigurationCopy = newServerConfig;\n' + result += 'export let resetServerConfiguration = () => serverConfigurationCopy = cloneConfiguration(serverConfiguration);\n\n' result += '\n'.join([ '// version (do not change)', f'export const VERSION = "{version_str}";' @@ -84,22 +97,22 @@ def update(): # Load settings from configuration_settings.json before updating load_settings() - with open('src/config.ts', 'w') as file: + with open(f'src/{configuration_path}/server_configuration.ts', 'w') as file: file.write(generateScript(False)) time.sleep(0.5) - with open('BP/scripts/config.js', 'w') as file: + with open(f'BP/scripts/{configuration_path}/server_configuration.js', 'w') as file: prevResult = generateScript(args.target == 'server') file.write(prevResult) def check_for_changes(): - settings_hash = compute_hash('src/configuration_settings.json') - config_js_hash = compute_hash('BP/scripts/config.js') - config_ts_hash = compute_hash('src/config.ts') + settings_hash = compute_hash(f'src/configuration_settings.json') + config_js_hash = compute_hash(f'BP/scripts/{configuration_path}/server_configuration.js') + config_ts_hash = compute_hash(f'src/{configuration_path}/server_configuration.ts') try: - with open('src/.config_hashes', 'r') as f: + with open(f'src/.config_hashes', 'r') as f: data = f.read().splitlines() except FileNotFoundError: data = ['', '', ''] @@ -107,20 +120,20 @@ def check_for_changes(): if data[0] == settings_hash and data[1] == config_js_hash: return False else: - with open('src/.config_hashes', 'w') as f: + with open(f'src/.config_hashes', 'w') as f: f.write(f"{settings_hash}\n{config_js_hash}\n{config_ts_hash}\n") # Update config.js update() # Copy config.js to BP/scripts if not already there - if not os.path.exists('BP/scripts/config.js'): - shutil.copyfile('BP/scripts/config.js', 'BP/scripts/config.js') + if not os.path.exists(f'BP/scripts/{configuration_path}/server_configuration.js'): + shutil.copyfile(f'BP/scripts/{configuration_path}/server_configuration.js', f'BP/scripts/{configuration_path}/server_configuration.js') # Update config.ts - with open('src/config.ts', 'r') as f: + with open(f'src/{configuration_path}/server_configuration.ts', 'r') as f: config_ts_content = f.read() - with open('src/config.ts', 'w') as f: + with open(f'src/{configuration_path}/server_configuration.ts', 'w') as f: f.write(re.sub(r"const VERSION = .+;", f"const VERSION = \"{version_str}\";", config_ts_content)) return True @@ -129,9 +142,9 @@ def check_for_changes(): def load_settings(): global settings try: - os.utime('src/configuration_settings.json', None) - with open('src/configuration_settings.json', 'r') as file: - settings = {**settings, **json.load(file)} + os.utime(f'src/configuration_settings.json', None) + with open(f'src/configuration_settings.json', 'r') as file: + settings = {**json.load(file), **settings} except (FileNotFoundError, json.JSONDecodeError): # Handle the case where the file is empty or not valid JSON. @@ -155,7 +168,7 @@ def load_settings(): # Generate src/config.ts if args.generateConfigTS: - with open('src/config.ts', 'w') as file: + with open(f'src/{configuration_path}/server_configuration.ts', 'w') as file: file.write(generateScript(False)) exit(0) @@ -169,9 +182,10 @@ def load_settings(): from watchdog.observers import Observer from watchdog.events import FileSystemEventHandler + class MyHandler(FileSystemEventHandler): def on_modified(self, ev): - if ev.src_path in ['src\configuration_settings.json']: + if ev.src_path in ['src\\configuration_settings.json']: if check_for_changes(): print("Settings changed! Updating...") @@ -180,7 +194,7 @@ def on_modified(self, ev): obsSettings.start() obsConfigJS = Observer() - obsConfigJS.schedule(MyHandler(), path='BP\scripts') + obsConfigJS.schedule(MyHandler(), path='BP\\scripts') obsConfigJS.start() try: diff --git a/tools/process_manifest.py b/tools/process_manifest.py index 49345f3..1dd536f 100644 --- a/tools/process_manifest.py +++ b/tools/process_manifest.py @@ -1,9 +1,9 @@ -import json -import argparse +import json, argparse import uuid import shutil import os + parser = argparse.ArgumentParser(description='Build manifest files from \'mc_manifest.json\'.') parser.add_argument('--init', '-i', action='store_true', help='Initialize UUID and versions.') parser.add_argument('--target', choices=['release', 'debug', 'server'], default='debug', help='Whether to build the addon in debug or release mode or for servers.') @@ -12,16 +12,16 @@ bp_manifest = {} rp_manifest = {} -def processJsonElement(element, bp_element, rp_element, init, current_key=None): +def processJsonElement(element, bp_element, rp_element): def process(key, value): if isinstance(value, dict): bp_element[key] = {} rp_element[key] = {} - processJsonElement(value, bp_element[key], rp_element[key], init, key) + processJsonElement(value, bp_element[key], rp_element[key]) elif isinstance(value, list): bp_element[key] = [] rp_element[key] = [] - processJsonElement(value, bp_element[key], rp_element[key], init, key) + processJsonElement(value, bp_element[key], rp_element[key]) else: if isinstance(bp_element, list): bp_element.append(value) @@ -30,8 +30,9 @@ def process(key, value): bp_element[key] = value rp_element[key] = value + if isinstance(element, dict): - for key, value in element.items(): + for [key, value] in element.items(): if key.startswith('bp_'): if key.startswith('bp_server_'): sub = bp_element[key[10:]] @@ -41,14 +42,8 @@ def process(key, value): bp_element[key[10:]] = { **sub, **value } else: bp_element[key[3:]] = value - if init and key == 'bp_modules': - for module in value: - module['uuid'] = str(uuid.uuid4()) elif key.startswith('rp_'): rp_element[key[3:]] = value - if init and key == 'rp_modules': - for module in value: - module['uuid'] = str(uuid.uuid4()) else: process(key, value) elif isinstance(element, list): @@ -58,10 +53,9 @@ def process(key, value): i = i + 1 # load base manifest -with open('setup/mc_manifest.json', 'r') as file: +with open('setup/mc_manifest.json', 'r', encoding='utf-8') as file: manifest = json.load(file) - processJsonElement(manifest, bp_manifest, rp_manifest, args.init) - + processJsonElement(manifest, bp_manifest, rp_manifest) # Generate UUIDv4 for bp_uuid and rp_uuid and set default versions if args.init: @@ -85,15 +79,15 @@ def process(key, value): shutil.copy('setup/pack_icon.png', 'RP/') version = manifest['header']['version'] -bp_manifest['header']['name'] += ' ' + '.'.join(map(str, version)) -rp_manifest['header']['name'] += ' ' + '.'.join(map(str, version)) +bp_manifest['header']['name'] += ' §8(' + '.'.join(map(str, version)) + ')' +rp_manifest['header']['name'] += ' §8(' + '.'.join(map(str, version)) + ')' -if not isinstance(version, str): +if not type(version) is str: version = version[:3] bp_manifest['header']['version'] = version rp_manifest['header']['version'] = version -if 'dependencies' not in bp_manifest: +if not 'dependencies' in bp_manifest: bp_manifest['dependencies'] = [] bp_manifest['dependencies'].append({ 'uuid': rp_manifest['header']['uuid'], @@ -105,7 +99,7 @@ def process(key, value): rp_manifest['header']['name'] += ' [DEBUG]' # export behaviour and resource manifests -with open('BP/manifest.json', 'w') as file: - json.dump(bp_manifest, file, indent=4) -with open('RP/manifest.json', 'w') as file: - json.dump(rp_manifest, file, indent=4) +with open('BP/manifest.json', 'w', encoding='utf-8') as file: + json.dump(bp_manifest, file, indent=4, ensure_ascii=False) +with open('RP/manifest.json', 'w', encoding='utf-8') as file: + json.dump(rp_manifest, file, indent=4, ensure_ascii=False) \ No newline at end of file diff --git a/tools/sync2com-mojang.py b/tools/sync2com-mojang.py index a6fd01a..dbff429 100644 --- a/tools/sync2com-mojang.py +++ b/tools/sync2com-mojang.py @@ -21,8 +21,8 @@ com_mojang = os.path.expandvars(SERVER_LOCATION) -behaviour_pack = com_mojang + f'\\development_behavior_packs\\{pack_folder} BP' -resource_pack = com_mojang + f'\\development_resource_packs\\{pack_folder} RP' +behaviour_pack = os.path.join(com_mojang, 'development_behavior_packs', f'{pack_folder} BP') +resource_pack = os.path.join(com_mojang, 'development_resource_packs', f'{pack_folder} RP') def sync_file(path, from_root, to_root): from_file = Path(path).relative_to(from_root) diff --git a/tsconfig.json b/tsconfig.json index 211acb4..a02fdc0 100644 --- a/tsconfig.json +++ b/tsconfig.json @@ -1,15 +1,16 @@ { - "compilerOptions": { - "module": "ES2020", - "target": "ES2021", - "moduleResolution": "Node", - "allowSyntheticDefaultImports": true, - "baseUrl": "./src", - "rootDir": "./src", - "removeComments": true, - "outDir": "./BP/scripts/" - }, - "exclude": [ "node_modules" ], - "include": [ "src" ] - } - \ No newline at end of file + "compilerOptions": { + "module": "ES2020", + "target": "ES2021", + "moduleResolution": "Node", + "removeComments": true, + "allowJs": true, + "checkJs": false, + "allowSyntheticDefaultImports": true, + "baseUrl": "./src", + "rootDir": "./src", + "outDir": "./BP/scripts" + }, + "exclude": [ "node_modules"], + "include": [ "src/**/*"] +} \ No newline at end of file