From b7bed44a61b3635fe97237254d0a94a23f3fc008 Mon Sep 17 00:00:00 2001 From: simibubi <31564874+simibubi@users.noreply.github.com> Date: Mon, 19 Oct 2020 22:07:05 +0200 Subject: [PATCH 1/2] Basic Brewing - Refactored JEI plugin to use a builder for categories - The mixer can now process vanilla potion recipes - JEI support for fluid potion mixing - Refined some categories to not show obsolete information - Compacting and automatic vanilla packing are now in separate tabs - Fluids can now be manually picked up from a basin - Basins and Spouts can now extract from and fill vanilla potion items - Improved the filtering for fluid tests, such as the basin recipe and smart pipes - Fixed CombinedTankWrapper losing nbt data of transferred fluidstacks --- src/generated/resources/.cache/cache | 26 +- .../resources/assets/create/lang/en_ud.json | 1 + .../resources/assets/create/lang/en_us.json | 27 +- .../assets/create/lang/unfinished/de_de.json | 29 +- .../assets/create/lang/unfinished/fr_fr.json | 25 +- .../assets/create/lang/unfinished/it_it.json | 25 +- .../assets/create/lang/unfinished/ja_jp.json | 25 +- .../assets/create/lang/unfinished/ko_kr.json | 25 +- .../assets/create/lang/unfinished/nl_nl.json | 29 +- .../assets/create/lang/unfinished/pt_br.json | 29 +- .../assets/create/lang/unfinished/ru_ru.json | 29 +- .../assets/create/lang/unfinished/zh_cn.json | 25 +- .../create/recipes/emptying/water_bottle.json | 18 - .../create/recipes/filling/water_bottle.json | 21 - .../create/recipes/mixing/temp_cobble.json | 6 +- .../java/com/simibubi/create/AllFluids.java | 21 + src/main/java/com/simibubi/create/Create.java | 1 + .../simibubi/create/compat/jei/CreateJEI.java | 408 +++++++++++------- .../compat/jei/category/BasinCategory.java | 34 +- .../jei/category/BlockCuttingCategory.java | 2 +- .../category/BlockzapperUpgradeCategory.java | 27 +- .../jei/category/CreateRecipeCategory.java | 42 +- .../compat/jei/category/CrushingCategory.java | 10 +- ...Category.java => FanBlastingCategory.java} | 12 +- ...nCategory.java => FanSmokingCategory.java} | 6 +- ...gCategory.java => FanWashingCategory.java} | 6 +- .../category/MechanicalCraftingCategory.java | 218 +++++----- .../compat/jei/category/MillingCategory.java | 3 +- .../compat/jei/category/MixingCategory.java | 32 +- .../MysteriousItemConversionCategory.java | 4 +- .../compat/jei/category/PackingCategory.java | 35 +- .../jei/category/PolishingCategory.java | 15 +- .../compat/jei/category/PressingCategory.java | 13 +- .../category/ProcessingViaFanCategory.java | 14 +- .../compat/jei/category/SawingCategory.java | 2 +- .../mixer/MechanicalMixerTileEntity.java | 35 +- .../fluids/CombinedFluidHandler.java | 138 ------ .../contraptions/fluids/FluidNetworkFlow.java | 2 +- .../fluids/actors/FillingBySpout.java | 46 +- .../fluids/actors/GenericItemFilling.java | 82 ++++ .../fluids/potion/PotionFluid.java | 111 +++++ .../fluids/potion/PotionFluidHandler.java | 171 ++++++++ .../potion/PotionMixingRecipeManager.java | 158 +++++++ .../contraptions/processing/BasinBlock.java | 66 ++- .../contraptions/processing/BasinRecipe.java | 25 +- .../processing/EmptyingByBasin.java | 8 + .../processing/ProcessingRecipeBuilder.java | 6 +- .../logistics/item/filter/FilterItem.java | 20 +- .../simibubi/create/events/CommonEvents.java | 15 +- .../foundation/data/CreateRegistrate.java | 23 +- .../foundation/data/VirtualFluidBuilder.java | 35 ++ .../data/recipe/EmptyingRecipeGen.java | 26 +- .../data/recipe/FillingRecipeGen.java | 21 +- .../data/recipe/MixingRecipeGen.java | 5 +- .../foundation/fluid/CombinedTankWrapper.java | 45 +- .../foundation/fluid/FluidIngredient.java | 19 +- .../resources/META-INF/accesstransformer.cfg | 5 +- .../assets/create/lang/default/messages.json | 31 +- .../create/textures/fluid/potion_flow.png | Bin 0 -> 133632 bytes .../textures/fluid/potion_flow.png.mcmeta | 3 + .../create/textures/fluid/potion_still.png | Bin 0 -> 66280 bytes .../textures/fluid/potion_still.png.mcmeta | 5 + 62 files changed, 1533 insertions(+), 813 deletions(-) delete mode 100644 src/generated/resources/data/create/recipes/emptying/water_bottle.json delete mode 100644 src/generated/resources/data/create/recipes/filling/water_bottle.json create mode 100644 src/main/java/com/simibubi/create/AllFluids.java rename src/main/java/com/simibubi/create/compat/jei/category/{BlastingViaFanCategory.java => FanBlastingCategory.java} (68%) rename src/main/java/com/simibubi/create/compat/jei/category/{SmokingViaFanCategory.java => FanSmokingCategory.java} (71%) rename src/main/java/com/simibubi/create/compat/jei/category/{SplashingCategory.java => FanWashingCategory.java} (93%) delete mode 100644 src/main/java/com/simibubi/create/content/contraptions/fluids/CombinedFluidHandler.java create mode 100644 src/main/java/com/simibubi/create/content/contraptions/fluids/actors/GenericItemFilling.java create mode 100644 src/main/java/com/simibubi/create/content/contraptions/fluids/potion/PotionFluid.java create mode 100644 src/main/java/com/simibubi/create/content/contraptions/fluids/potion/PotionFluidHandler.java create mode 100644 src/main/java/com/simibubi/create/content/contraptions/fluids/potion/PotionMixingRecipeManager.java create mode 100644 src/main/java/com/simibubi/create/foundation/data/VirtualFluidBuilder.java create mode 100644 src/main/resources/assets/create/textures/fluid/potion_flow.png create mode 100644 src/main/resources/assets/create/textures/fluid/potion_flow.png.mcmeta create mode 100644 src/main/resources/assets/create/textures/fluid/potion_still.png create mode 100644 src/main/resources/assets/create/textures/fluid/potion_still.png.mcmeta diff --git a/src/generated/resources/.cache/cache b/src/generated/resources/.cache/cache index 3c3ebbe8e..c2daa081e 100644 --- a/src/generated/resources/.cache/cache +++ b/src/generated/resources/.cache/cache @@ -368,17 +368,17 @@ a3a11524cd3515fc01d905767b4b7ea782adaf03 assets/create/blockstates/yellow_seat.j 6801fa1f466f172700e573e5b8ee8ee5f9ca4583 assets/create/blockstates/yellow_valve_handle.json 7f39521b211441f5c3e06d60c5978cebe16cacfb assets/create/blockstates/zinc_block.json b7181bcd8182b2f17088e5aa881f374c9c65470c assets/create/blockstates/zinc_ore.json -ec6eb28b76106e24adb255e49943da3f53196bfb assets/create/lang/en_ud.json -f719ad41c8be237bbbafa8fabd8fda2b47288a31 assets/create/lang/en_us.json -5f8522c527cbbe50e33a7774316988fa6077d78d assets/create/lang/unfinished/de_de.json -0473893175500c6fd3f5410b575fc6ffc3abbb28 assets/create/lang/unfinished/fr_fr.json -c9fb5f2769f3e9be347bcd34b21a6c961e9a2324 assets/create/lang/unfinished/it_it.json -9bf0b4231a4f3d58b31fdb8d9876c8fdb40c5df5 assets/create/lang/unfinished/ja_jp.json -d7bd07920ecca15c2fe0728b141b3a9230ab545f assets/create/lang/unfinished/ko_kr.json -71c6d13fd8b300c851a591b76b32eff442acf586 assets/create/lang/unfinished/nl_nl.json -1668581de8b7ff54bb8cf2d705faa07936d2d8fd assets/create/lang/unfinished/pt_br.json -891d5a4abd250d7901a3a3df4907722e8ef864e9 assets/create/lang/unfinished/ru_ru.json -ef6159365f4746e8cc979527801893c92f19e6da assets/create/lang/unfinished/zh_cn.json +33f621fe1a09a46426af59c9352c33e97c2bc00d assets/create/lang/en_ud.json +f68da0c370026f416745b46445b53c63457cdeac assets/create/lang/en_us.json +1fa1a84b7dae8c7636a667e094883646ad6fa1e2 assets/create/lang/unfinished/de_de.json +4b2501a9d7730b4fb469d241afd72483171ce36b assets/create/lang/unfinished/fr_fr.json +efff9eee3f434fa04b3a688e8e861ab7d36fb45f assets/create/lang/unfinished/it_it.json +6db6ebd43ec38e11ad81f4b997b5765857bdf8da assets/create/lang/unfinished/ja_jp.json +3975b1184f38e7d7150ba2be0179bd3cf36ed7ca assets/create/lang/unfinished/ko_kr.json +84c44b3b4805925d0b489e295b38a72dbf2703e1 assets/create/lang/unfinished/nl_nl.json +f82f1e6250dd9a5ca9a7b92078bf1f7429d14d00 assets/create/lang/unfinished/pt_br.json +72d5dcbaa454de0ff33ff0cb2108938c2a5e677a assets/create/lang/unfinished/ru_ru.json +fdc2007ea9e3f8d8b2a6f8dedc434b591a0bbf0a assets/create/lang/unfinished/zh_cn.json 846200eb548d3bfa2e77b41039de159b4b6cfb45 assets/create/models/block/acacia_window.json 1930fa3a3c98d53dd19e4ee7f55bc27fd47aa281 assets/create/models/block/acacia_window_pane_noside.json 1763ea2c9b981d187f5031ba608f3d5d3be3986a assets/create/models/block/acacia_window_pane_noside_alt.json @@ -2723,7 +2723,6 @@ ddda28bb6efc43b7e3149756daf53e1664187283 data/create/recipes/dolomite_cobbleston 500ecdfdcf34e9d26256948e206aab4f0b79e659 data/create/recipes/dolomite_cobblestone_wall_from_dolomite_cobblestone_stonecutting.json ff39e629b242ae91e23aec86b0a1f757dd938305 data/create/recipes/dolomite_pillar.json b4a8d14d9a20e812e0acb691b5b511a87e8b0576 data/create/recipes/dolomite_pillar_from_dolomite_stonecutting.json -ae6698363e49f7cb5f2ed52c6b8805bebed31fa2 data/create/recipes/emptying/water_bottle.json 0e11aa1accb71ed62e212f23a7069b7b7b4e8119 data/create/recipes/fancy_andesite_bricks_from_andesite_stonecutting.json 8b86fc9a9416adeaab3f26192a73a481887675c3 data/create/recipes/fancy_andesite_bricks_slab.json c7b762b25c7a6705dba3e922e981be851ac4f36b data/create/recipes/fancy_andesite_bricks_slab_from_fancy_andesite_bricks_stonecutting.json @@ -2787,7 +2786,6 @@ bd355332d17adcb0460b1d43146ca288efb78395 data/create/recipes/fancy_weathered_lim d2ab9ce73636773165564506580f2ec13bd1fc50 data/create/recipes/fancy_weathered_limestone_bricks_stairs_from_fancy_weathered_limestone_bricks_stonecutting.json 36947f27d2b2e57b00440fd5acd06a7554e5a387 data/create/recipes/fancy_weathered_limestone_bricks_wall.json 1d0e41ca98e48073c72adf4077610c96e592f9a5 data/create/recipes/fancy_weathered_limestone_bricks_wall_from_fancy_weathered_limestone_bricks_stonecutting.json -3196d3eda9e67771e86e9af7026d4388765a8a73 data/create/recipes/filling/water_bottle.json 5b8bbde7f8b270ab75fac18d6858f2fadbc0efa3 data/create/recipes/framed_glass_from_glass_colorless_stonecutting.json d697de0c9b706ca4e18da7a2d769e7e5fe8d769d data/create/recipes/framed_glass_pane.json a0dae50faaa1b7142bb4309675e3084c68daa547 data/create/recipes/gabbro_bricks_from_gabbro_stonecutting.json @@ -2909,7 +2907,7 @@ e7bfaa806d57573d060fac0a5e7a84f345b8bb85 data/create/recipes/mixing/andesite_all 76939e4d3e5b615ae21d14c0ff7b917a622bcf80 data/create/recipes/mixing/chromatic_compound.json d9a3dff1288d675ab812eef1eb73cb27dcc71bd2 data/create/recipes/mixing/crushed_brass.json 00b165ea38d834c7955440e87062004a8182c3f8 data/create/recipes/mixing/gunpowder.json -743c2d62c94a4989b88046c9cdf0fa024cd09fa8 data/create/recipes/mixing/temp_cobble.json +cf2a741be46fd2069f242afe74eb7c11f3139c46 data/create/recipes/mixing/temp_cobble.json 3295a2195707f952a83deb3bed10b43570b215e5 data/create/recipes/mixing/temp_lava.json 1998c6f84f871d6da58ec29d729401d18f8f1aa1 data/create/recipes/mossy_andesite_from_andesite_stonecutting.json 89929d9cb11b5c589b2ecfa821c61add1ef7b62b data/create/recipes/mossy_dark_scoria_from_dark_scoria_stonecutting.json diff --git a/src/generated/resources/assets/create/lang/en_ud.json b/src/generated/resources/assets/create/lang/en_ud.json index 49e40242a..90d627d05 100644 --- a/src/generated/resources/assets/create/lang/en_ud.json +++ b/src/generated/resources/assets/create/lang/en_ud.json @@ -373,6 +373,7 @@ "entity.create.seat": "\u0287\u0250\u01DDS", "entity.create.stationary_contraption": "uo\u0131\u0287d\u0250\u0279\u0287uo\u0186 \u028E\u0279\u0250uo\u0131\u0287\u0250\u0287S", "entity.create.super_glue": "\u01DDn\u05DF\u2141 \u0279\u01DDdnS", + "fluid.create.flowing_potion": "uo\u0131\u0287o\u0500", "item.create.andesite_alloy": "\u028Eo\u05DF\u05DF\u2C6F \u01DD\u0287\u0131s\u01DDpu\u2C6F", "item.create.attribute_filter": "\u0279\u01DD\u0287\u05DF\u0131\u2132 \u01DD\u0287nq\u0131\u0279\u0287\u0287\u2C6F", "item.create.belt_connector": "\u0287\u05DF\u01DD\u15FA \u05DF\u0250\u0254\u0131u\u0250\u0265\u0254\u01DDW", diff --git a/src/generated/resources/assets/create/lang/en_us.json b/src/generated/resources/assets/create/lang/en_us.json index a178af162..0c3d53528 100644 --- a/src/generated/resources/assets/create/lang/en_us.json +++ b/src/generated/resources/assets/create/lang/en_us.json @@ -378,6 +378,8 @@ "entity.create.stationary_contraption": "Stationary Contraption", "entity.create.super_glue": "Super Glue", + "fluid.create.flowing_potion": "Potion", + "item.create.andesite_alloy": "Andesite Alloy", "item.create.attribute_filter": "Attribute Filter", "item.create.belt_connector": "Mechanical Belt", @@ -540,26 +542,26 @@ "create.recipe.crushing": "Crushing", "create.recipe.milling": "Milling", - "create.recipe.splashing": "Bulk Washing", - "create.recipe.splashing.fan": "Fan behind Flowing Water", - "create.recipe.smoking_via_fan": "Bulk Smoking", - "create.recipe.smoking_via_fan.fan": "Fan behind Fire", - "create.recipe.blasting_via_fan": "Bulk Smelting", - "create.recipe.blasting_via_fan.fan": "Fan behind Lava", + "create.recipe.fan_washing": "Bulk Washing", + "create.recipe.fan_washing.fan": "Fan behind Flowing Water", + "create.recipe.fan_smoking": "Bulk Smoking", + "create.recipe.fan_smoking.fan": "Fan behind Fire", + "create.recipe.fan_blasting": "Bulk Blasting", + "create.recipe.fan_blasting.fan": "Fan behind Lava", "create.recipe.pressing": "Pressing", "create.recipe.mixing": "Mixing", - "create.recipe.shapeless_mixing": "Shapeless Crafting", + "create.recipe.automatic_shapeless": "Automated Shapeless Crafting", + "create.recipe.automatic_brewing": "Automated Brewing", "create.recipe.packing": "Compacting", + "create.recipe.automatic_packing": "Automated Packing", "create.recipe.sawing": "Sawing", - "create.recipe.mechanical_crafting": "Crafting", - "create.recipe.mechanical_crafting_exclusive": "Mechanical Crafting", + "create.recipe.mechanical_crafting": "Mechanical Crafting", + "create.recipe.automatic_shaped": "Automated Shaped Crafting", "create.recipe.block_cutting": "Block Cutting", "create.recipe.blockzapper_upgrade": "Handheld Blockzapper", "create.recipe.sandpaper_polishing": "Sandpaper Polishing", - "create.recipe.mystery_conversion": "Chromatic Metamorphosis", - "create.recipe.processing.catalyst": "Catalyst", + "create.recipe.mystery_conversion": "Mysterious Conversion", "create.recipe.processing.chance": "%1$s%% Chance", - "create.recipe.processing.chanceToReturn": "%1$s%% Chance to Return", "create.recipe.heat_requirement.none": "No Heating Required", "create.recipe.heat_requirement.heated": "Heated", "create.recipe.heat_requirement.superheated": "Super-Heated", @@ -577,7 +579,6 @@ "create.generic.unit.rpm": "RPM", "create.generic.unit.stress": "su", "create.generic.unit.degrees": "°", - "create.generic.unit.bucket": "1 Bucket", "create.generic.unit.millibuckets": "%1$smB", "create.action.scroll": "Scroll", diff --git a/src/generated/resources/assets/create/lang/unfinished/de_de.json b/src/generated/resources/assets/create/lang/unfinished/de_de.json index 4f0561656..e8f05621e 100644 --- a/src/generated/resources/assets/create/lang/unfinished/de_de.json +++ b/src/generated/resources/assets/create/lang/unfinished/de_de.json @@ -1,5 +1,5 @@ { - "_": "Missing Localizations: 922", + "_": "Missing Localizations: 928", "_": "->------------------------] Game Elements [------------------------<-", @@ -379,6 +379,8 @@ "entity.create.stationary_contraption": "UNLOCALIZED: Stationary Contraption", "entity.create.super_glue": "UNLOCALIZED: Super Glue", + "fluid.create.flowing_potion": "UNLOCALIZED: Potion", + "item.create.andesite_alloy": "UNLOCALIZED: Andesite Alloy", "item.create.attribute_filter": "UNLOCALIZED: Attribute Filter", "item.create.belt_connector": "Mechanischer Riemen", @@ -541,26 +543,26 @@ "create.recipe.crushing": "Mahlen", "create.recipe.milling": "UNLOCALIZED: Milling", - "create.recipe.splashing": "Waschen", - "create.recipe.splashing.fan": "Propeller hinter fließendem Wasser", - "create.recipe.smoking_via_fan": "Räuchern", - "create.recipe.smoking_via_fan.fan": "Propeller hinter Feuer", - "create.recipe.blasting_via_fan": "Schmelzen", - "create.recipe.blasting_via_fan.fan": "Propeller hinter Lava", + "create.recipe.fan_washing": "UNLOCALIZED: Bulk Washing", + "create.recipe.fan_washing.fan": "UNLOCALIZED: Fan behind Flowing Water", + "create.recipe.fan_smoking": "UNLOCALIZED: Bulk Smoking", + "create.recipe.fan_smoking.fan": "UNLOCALIZED: Fan behind Fire", + "create.recipe.fan_blasting": "UNLOCALIZED: Bulk Blasting", + "create.recipe.fan_blasting.fan": "UNLOCALIZED: Fan behind Lava", "create.recipe.pressing": "Mechanische Presse", "create.recipe.mixing": "UNLOCALIZED: Mixing", - "create.recipe.shapeless_mixing": "UNLOCALIZED: Shapeless Crafting", + "create.recipe.automatic_shapeless": "UNLOCALIZED: Automated Shapeless Crafting", + "create.recipe.automatic_brewing": "UNLOCALIZED: Automated Brewing", "create.recipe.packing": "UNLOCALIZED: Compacting", + "create.recipe.automatic_packing": "UNLOCALIZED: Automated Packing", "create.recipe.sawing": "UNLOCALIZED: Sawing", - "create.recipe.mechanical_crafting": "UNLOCALIZED: Crafting", - "create.recipe.mechanical_crafting_exclusive": "UNLOCALIZED: Mechanical Crafting", + "create.recipe.mechanical_crafting": "UNLOCALIZED: Mechanical Crafting", + "create.recipe.automatic_shaped": "UNLOCALIZED: Automated Shaped Crafting", "create.recipe.block_cutting": "UNLOCALIZED: Block Cutting", "create.recipe.blockzapper_upgrade": "Blockpistole", "create.recipe.sandpaper_polishing": "UNLOCALIZED: Sandpaper Polishing", - "create.recipe.mystery_conversion": "UNLOCALIZED: Chromatic Metamorphosis", - "create.recipe.processing.catalyst": "UNLOCALIZED: Catalyst", + "create.recipe.mystery_conversion": "UNLOCALIZED: Mysterious Conversion", "create.recipe.processing.chance": "Chance: %1$s%%", - "create.recipe.processing.chanceToReturn": "UNLOCALIZED: %1$s%% Chance to Return", "create.recipe.heat_requirement.none": "UNLOCALIZED: No Heating Required", "create.recipe.heat_requirement.heated": "UNLOCALIZED: Heated", "create.recipe.heat_requirement.superheated": "UNLOCALIZED: Super-Heated", @@ -578,7 +580,6 @@ "create.generic.unit.rpm": "UNLOCALIZED: RPM", "create.generic.unit.stress": "UNLOCALIZED: su", "create.generic.unit.degrees": "UNLOCALIZED: °", - "create.generic.unit.bucket": "UNLOCALIZED: 1 Bucket", "create.generic.unit.millibuckets": "UNLOCALIZED: %1$smB", "create.action.scroll": "Wechseln", diff --git a/src/generated/resources/assets/create/lang/unfinished/fr_fr.json b/src/generated/resources/assets/create/lang/unfinished/fr_fr.json index cb26a7b57..274921286 100644 --- a/src/generated/resources/assets/create/lang/unfinished/fr_fr.json +++ b/src/generated/resources/assets/create/lang/unfinished/fr_fr.json @@ -1,5 +1,5 @@ { - "_": "Missing Localizations: 547", + "_": "Missing Localizations: 555", "_": "->------------------------] Game Elements [------------------------<-", @@ -379,6 +379,8 @@ "entity.create.stationary_contraption": "UNLOCALIZED: Stationary Contraption", "entity.create.super_glue": "UNLOCALIZED: Super Glue", + "fluid.create.flowing_potion": "UNLOCALIZED: Potion", + "item.create.andesite_alloy": "Alliage d'andésite", "item.create.attribute_filter": "Filtre d'attribut", "item.create.belt_connector": "Tapis roulant", @@ -541,26 +543,26 @@ "create.recipe.crushing": "Ecrasement", "create.recipe.milling": "UNLOCALIZED: Milling", - "create.recipe.splashing": "Lavage en vrac", - "create.recipe.splashing.fan": "Ventilateur derrière de l'eau qui coule", - "create.recipe.smoking_via_fan": "Fumer en vrac", - "create.recipe.smoking_via_fan.fan": "Fan behind Fire", - "create.recipe.blasting_via_fan": "Ventilateur derrière du feu", - "create.recipe.blasting_via_fan.fan": "Ventilateur derrière de la lave", + "create.recipe.fan_washing": "UNLOCALIZED: Bulk Washing", + "create.recipe.fan_washing.fan": "UNLOCALIZED: Fan behind Flowing Water", + "create.recipe.fan_smoking": "UNLOCALIZED: Bulk Smoking", + "create.recipe.fan_smoking.fan": "UNLOCALIZED: Fan behind Fire", + "create.recipe.fan_blasting": "UNLOCALIZED: Bulk Blasting", + "create.recipe.fan_blasting.fan": "UNLOCALIZED: Fan behind Lava", "create.recipe.pressing": "Pressage", "create.recipe.mixing": "Mixage", - "create.recipe.shapeless_mixing": "UNLOCALIZED: Shapeless Crafting", + "create.recipe.automatic_shapeless": "UNLOCALIZED: Automated Shapeless Crafting", + "create.recipe.automatic_brewing": "UNLOCALIZED: Automated Brewing", "create.recipe.packing": "Compactage", + "create.recipe.automatic_packing": "UNLOCALIZED: Automated Packing", "create.recipe.sawing": "UNLOCALIZED: Sawing", "create.recipe.mechanical_crafting": "Fabrication mécanique", - "create.recipe.mechanical_crafting_exclusive": "UNLOCALIZED: Mechanical Crafting", + "create.recipe.automatic_shaped": "UNLOCALIZED: Automated Shaped Crafting", "create.recipe.block_cutting": "Coupe de bloc", "create.recipe.blockzapper_upgrade": "Blockzappeur portable", "create.recipe.sandpaper_polishing": "Polissage au papier de verre", "create.recipe.mystery_conversion": "Métamorphose chromatique", - "create.recipe.processing.catalyst": "Catalyseur", "create.recipe.processing.chance": "%1$s%% de chance", - "create.recipe.processing.chanceToReturn": "%1$s%% de chance de retour", "create.recipe.heat_requirement.none": "UNLOCALIZED: No Heating Required", "create.recipe.heat_requirement.heated": "UNLOCALIZED: Heated", "create.recipe.heat_requirement.superheated": "UNLOCALIZED: Super-Heated", @@ -578,7 +580,6 @@ "create.generic.unit.rpm": "tr/min", "create.generic.unit.stress": "us", "create.generic.unit.degrees": "°", - "create.generic.unit.bucket": "UNLOCALIZED: 1 Bucket", "create.generic.unit.millibuckets": "UNLOCALIZED: %1$smB", "create.action.scroll": "Faire défiler", diff --git a/src/generated/resources/assets/create/lang/unfinished/it_it.json b/src/generated/resources/assets/create/lang/unfinished/it_it.json index 9efa7360c..f5730a023 100644 --- a/src/generated/resources/assets/create/lang/unfinished/it_it.json +++ b/src/generated/resources/assets/create/lang/unfinished/it_it.json @@ -1,5 +1,5 @@ { - "_": "Missing Localizations: 535", + "_": "Missing Localizations: 539", "_": "->------------------------] Game Elements [------------------------<-", @@ -379,6 +379,8 @@ "entity.create.stationary_contraption": "UNLOCALIZED: Stationary Contraption", "entity.create.super_glue": "UNLOCALIZED: Super Glue", + "fluid.create.flowing_potion": "UNLOCALIZED: Potion", + "item.create.andesite_alloy": "Lega di Andesite", "item.create.attribute_filter": "Filtro Attributi", "item.create.belt_connector": "Nastro Meccanico", @@ -541,26 +543,26 @@ "create.recipe.crushing": "Frantumazione", "create.recipe.milling": "Fresatura", - "create.recipe.splashing": "Lavaggio alla Rinfusa", - "create.recipe.splashing.fan": "Ventilatore Dietro l'Acqua Corrente", - "create.recipe.smoking_via_fan": "UNLOCALIZED: Bulk Smoking", - "create.recipe.smoking_via_fan.fan": "UNLOCALIZED: Fan behind Fire", - "create.recipe.blasting_via_fan": "UNLOCALIZED: Bulk Smelting", - "create.recipe.blasting_via_fan.fan": "UNLOCALIZED: Fan behind Lava", + "create.recipe.fan_washing": "UNLOCALIZED: Bulk Washing", + "create.recipe.fan_washing.fan": "UNLOCALIZED: Fan behind Flowing Water", + "create.recipe.fan_smoking": "UNLOCALIZED: Bulk Smoking", + "create.recipe.fan_smoking.fan": "UNLOCALIZED: Fan behind Fire", + "create.recipe.fan_blasting": "UNLOCALIZED: Bulk Blasting", + "create.recipe.fan_blasting.fan": "UNLOCALIZED: Fan behind Lava", "create.recipe.pressing": "Pressatura", "create.recipe.mixing": "Mescolamento", - "create.recipe.shapeless_mixing": "UNLOCALIZED: Shapeless Crafting", + "create.recipe.automatic_shapeless": "UNLOCALIZED: Automated Shapeless Crafting", + "create.recipe.automatic_brewing": "UNLOCALIZED: Automated Brewing", "create.recipe.packing": "Compattazione", + "create.recipe.automatic_packing": "UNLOCALIZED: Automated Packing", "create.recipe.sawing": "UNLOCALIZED: Sawing", "create.recipe.mechanical_crafting": "Creazione Meccanico", - "create.recipe.mechanical_crafting_exclusive": "UNLOCALIZED: Mechanical Crafting", + "create.recipe.automatic_shaped": "UNLOCALIZED: Automated Shaped Crafting", "create.recipe.block_cutting": "Taglio Blocco", "create.recipe.blockzapper_upgrade": "UNLOCALIZED: Handheld Blockzapper", "create.recipe.sandpaper_polishing": "Carta Vetrata Levigata", "create.recipe.mystery_conversion": "Metamorfosi Cromatica", - "create.recipe.processing.catalyst": "Catalizzatore", "create.recipe.processing.chance": "%1$s%% Possibilità", - "create.recipe.processing.chanceToReturn": "%1$s%% Possibilità di Ritorno", "create.recipe.heat_requirement.none": "UNLOCALIZED: No Heating Required", "create.recipe.heat_requirement.heated": "UNLOCALIZED: Heated", "create.recipe.heat_requirement.superheated": "UNLOCALIZED: Super-Heated", @@ -578,7 +580,6 @@ "create.generic.unit.rpm": "RPM", "create.generic.unit.stress": "su", "create.generic.unit.degrees": "°", - "create.generic.unit.bucket": "UNLOCALIZED: 1 Bucket", "create.generic.unit.millibuckets": "UNLOCALIZED: %1$smB", "create.action.scroll": "Scorri", diff --git a/src/generated/resources/assets/create/lang/unfinished/ja_jp.json b/src/generated/resources/assets/create/lang/unfinished/ja_jp.json index f0976e969..72ed9f0d7 100644 --- a/src/generated/resources/assets/create/lang/unfinished/ja_jp.json +++ b/src/generated/resources/assets/create/lang/unfinished/ja_jp.json @@ -1,5 +1,5 @@ { - "_": "Missing Localizations: 530", + "_": "Missing Localizations: 538", "_": "->------------------------] Game Elements [------------------------<-", @@ -379,6 +379,8 @@ "entity.create.stationary_contraption": "UNLOCALIZED: Stationary Contraption", "entity.create.super_glue": "UNLOCALIZED: Super Glue", + "fluid.create.flowing_potion": "UNLOCALIZED: Potion", + "item.create.andesite_alloy": "安山岩合金", "item.create.attribute_filter": "属性フィルター", "item.create.belt_connector": "メカニカルベルト", @@ -541,26 +543,26 @@ "create.recipe.crushing": "粉砕", "create.recipe.milling": "製粉", - "create.recipe.splashing": "一括洗浄", - "create.recipe.splashing.fan": "流れる水の後ろにファンを置く", - "create.recipe.smoking_via_fan": "一括燻製", - "create.recipe.smoking_via_fan.fan": "炎の後ろにファンを置く", - "create.recipe.blasting_via_fan": "一括製錬", - "create.recipe.blasting_via_fan.fan": "溶岩の後ろにファンを置く", + "create.recipe.fan_washing": "UNLOCALIZED: Bulk Washing", + "create.recipe.fan_washing.fan": "UNLOCALIZED: Fan behind Flowing Water", + "create.recipe.fan_smoking": "UNLOCALIZED: Bulk Smoking", + "create.recipe.fan_smoking.fan": "UNLOCALIZED: Fan behind Fire", + "create.recipe.fan_blasting": "UNLOCALIZED: Bulk Blasting", + "create.recipe.fan_blasting.fan": "UNLOCALIZED: Fan behind Lava", "create.recipe.pressing": "押しつぶし", "create.recipe.mixing": "混合", - "create.recipe.shapeless_mixing": "UNLOCALIZED: Shapeless Crafting", + "create.recipe.automatic_shapeless": "UNLOCALIZED: Automated Shapeless Crafting", + "create.recipe.automatic_brewing": "UNLOCALIZED: Automated Brewing", "create.recipe.packing": "圧縮", + "create.recipe.automatic_packing": "UNLOCALIZED: Automated Packing", "create.recipe.sawing": "UNLOCALIZED: Sawing", "create.recipe.mechanical_crafting": "メカニカルクラフト", - "create.recipe.mechanical_crafting_exclusive": "UNLOCALIZED: Mechanical Crafting", + "create.recipe.automatic_shaped": "UNLOCALIZED: Automated Shaped Crafting", "create.recipe.block_cutting": "ブロックカット", "create.recipe.blockzapper_upgrade": "携帯型ブロックザッパー", "create.recipe.sandpaper_polishing": "紙やすりでの研磨", "create.recipe.mystery_conversion": "色彩変態", - "create.recipe.processing.catalyst": "触媒", "create.recipe.processing.chance": "%1$s%%チャンス", - "create.recipe.processing.chanceToReturn": "%1$s%%チャンスで戻る", "create.recipe.heat_requirement.none": "UNLOCALIZED: No Heating Required", "create.recipe.heat_requirement.heated": "UNLOCALIZED: Heated", "create.recipe.heat_requirement.superheated": "UNLOCALIZED: Super-Heated", @@ -578,7 +580,6 @@ "create.generic.unit.rpm": "RPM", "create.generic.unit.stress": "su", "create.generic.unit.degrees": "°", - "create.generic.unit.bucket": "UNLOCALIZED: 1 Bucket", "create.generic.unit.millibuckets": "UNLOCALIZED: %1$smB", "create.action.scroll": "スクロール", diff --git a/src/generated/resources/assets/create/lang/unfinished/ko_kr.json b/src/generated/resources/assets/create/lang/unfinished/ko_kr.json index 887eb9a0e..1159a98f7 100644 --- a/src/generated/resources/assets/create/lang/unfinished/ko_kr.json +++ b/src/generated/resources/assets/create/lang/unfinished/ko_kr.json @@ -1,5 +1,5 @@ { - "_": "Missing Localizations: 535", + "_": "Missing Localizations: 539", "_": "->------------------------] Game Elements [------------------------<-", @@ -379,6 +379,8 @@ "entity.create.stationary_contraption": "UNLOCALIZED: Stationary Contraption", "entity.create.super_glue": "UNLOCALIZED: Super Glue", + "fluid.create.flowing_potion": "UNLOCALIZED: Potion", + "item.create.andesite_alloy": "안산암 합금", "item.create.attribute_filter": "속성 필터 틀", "item.create.belt_connector": "기계식 벨트", @@ -541,26 +543,26 @@ "create.recipe.crushing": "분쇄", "create.recipe.milling": "맷돌질", - "create.recipe.splashing": "세척", - "create.recipe.splashing.fan": "물과 환풍기", - "create.recipe.smoking_via_fan": "UNLOCALIZED: Bulk Smoking", - "create.recipe.smoking_via_fan.fan": "UNLOCALIZED: Fan behind Fire", - "create.recipe.blasting_via_fan": "UNLOCALIZED: Bulk Smelting", - "create.recipe.blasting_via_fan.fan": "UNLOCALIZED: Fan behind Lava", + "create.recipe.fan_washing": "UNLOCALIZED: Bulk Washing", + "create.recipe.fan_washing.fan": "UNLOCALIZED: Fan behind Flowing Water", + "create.recipe.fan_smoking": "UNLOCALIZED: Bulk Smoking", + "create.recipe.fan_smoking.fan": "UNLOCALIZED: Fan behind Fire", + "create.recipe.fan_blasting": "UNLOCALIZED: Bulk Blasting", + "create.recipe.fan_blasting.fan": "UNLOCALIZED: Fan behind Lava", "create.recipe.pressing": "압착", "create.recipe.mixing": "혼합", - "create.recipe.shapeless_mixing": "UNLOCALIZED: Shapeless Crafting", + "create.recipe.automatic_shapeless": "UNLOCALIZED: Automated Shapeless Crafting", + "create.recipe.automatic_brewing": "UNLOCALIZED: Automated Brewing", "create.recipe.packing": "압축", + "create.recipe.automatic_packing": "UNLOCALIZED: Automated Packing", "create.recipe.sawing": "UNLOCALIZED: Sawing", "create.recipe.mechanical_crafting": "기계 조합", - "create.recipe.mechanical_crafting_exclusive": "UNLOCALIZED: Mechanical Crafting", + "create.recipe.automatic_shaped": "UNLOCALIZED: Automated Shaped Crafting", "create.recipe.block_cutting": "블럭 절단", "create.recipe.blockzapper_upgrade": "UNLOCALIZED: Handheld Blockzapper", "create.recipe.sandpaper_polishing": "사포 연마", "create.recipe.mystery_conversion": "?", - "create.recipe.processing.catalyst": "촉매", "create.recipe.processing.chance": "%1$s%% 확률", - "create.recipe.processing.chanceToReturn": "%1$s%% 확률로 반환", "create.recipe.heat_requirement.none": "UNLOCALIZED: No Heating Required", "create.recipe.heat_requirement.heated": "UNLOCALIZED: Heated", "create.recipe.heat_requirement.superheated": "UNLOCALIZED: Super-Heated", @@ -578,7 +580,6 @@ "create.generic.unit.rpm": "RPM", "create.generic.unit.stress": "su", "create.generic.unit.degrees": "°", - "create.generic.unit.bucket": "UNLOCALIZED: 1 Bucket", "create.generic.unit.millibuckets": "UNLOCALIZED: %1$smB", "create.action.scroll": "스크롤하세요", diff --git a/src/generated/resources/assets/create/lang/unfinished/nl_nl.json b/src/generated/resources/assets/create/lang/unfinished/nl_nl.json index 87943968c..82fca5cd5 100644 --- a/src/generated/resources/assets/create/lang/unfinished/nl_nl.json +++ b/src/generated/resources/assets/create/lang/unfinished/nl_nl.json @@ -1,5 +1,5 @@ { - "_": "Missing Localizations: 861", + "_": "Missing Localizations: 867", "_": "->------------------------] Game Elements [------------------------<-", @@ -379,6 +379,8 @@ "entity.create.stationary_contraption": "UNLOCALIZED: Stationary Contraption", "entity.create.super_glue": "UNLOCALIZED: Super Glue", + "fluid.create.flowing_potion": "UNLOCALIZED: Potion", + "item.create.andesite_alloy": "UNLOCALIZED: Andesite Alloy", "item.create.attribute_filter": "UNLOCALIZED: Attribute Filter", "item.create.belt_connector": "Mechanische Transportband", @@ -541,26 +543,26 @@ "create.recipe.crushing": "Verpulveren", "create.recipe.milling": "UNLOCALIZED: Milling", - "create.recipe.splashing": "Bulk Wassen", - "create.recipe.splashing.fan": "Ventilator achter vloeiend water", - "create.recipe.smoking_via_fan": "Bulk Rook", - "create.recipe.smoking_via_fan.fan": "Ventilator achter vuur", - "create.recipe.blasting_via_fan": "Bulk Smelten", - "create.recipe.blasting_via_fan.fan": "Ventilator achter Lava", + "create.recipe.fan_washing": "UNLOCALIZED: Bulk Washing", + "create.recipe.fan_washing.fan": "UNLOCALIZED: Fan behind Flowing Water", + "create.recipe.fan_smoking": "UNLOCALIZED: Bulk Smoking", + "create.recipe.fan_smoking.fan": "UNLOCALIZED: Fan behind Fire", + "create.recipe.fan_blasting": "UNLOCALIZED: Bulk Blasting", + "create.recipe.fan_blasting.fan": "UNLOCALIZED: Fan behind Lava", "create.recipe.pressing": "Persen", "create.recipe.mixing": "Mengen", - "create.recipe.shapeless_mixing": "UNLOCALIZED: Shapeless Crafting", + "create.recipe.automatic_shapeless": "UNLOCALIZED: Automated Shapeless Crafting", + "create.recipe.automatic_brewing": "UNLOCALIZED: Automated Brewing", "create.recipe.packing": "Compressen", + "create.recipe.automatic_packing": "UNLOCALIZED: Automated Packing", "create.recipe.sawing": "UNLOCALIZED: Sawing", - "create.recipe.mechanical_crafting": "UNLOCALIZED: Crafting", - "create.recipe.mechanical_crafting_exclusive": "UNLOCALIZED: Mechanical Crafting", + "create.recipe.mechanical_crafting": "UNLOCALIZED: Mechanical Crafting", + "create.recipe.automatic_shaped": "UNLOCALIZED: Automated Shaped Crafting", "create.recipe.block_cutting": "Blok Zagen", "create.recipe.blockzapper_upgrade": "Blokzapper", "create.recipe.sandpaper_polishing": "UNLOCALIZED: Sandpaper Polishing", - "create.recipe.mystery_conversion": "UNLOCALIZED: Chromatic Metamorphosis", - "create.recipe.processing.catalyst": "UNLOCALIZED: Catalyst", + "create.recipe.mystery_conversion": "UNLOCALIZED: Mysterious Conversion", "create.recipe.processing.chance": "%1$s%% Kans", - "create.recipe.processing.chanceToReturn": "UNLOCALIZED: %1$s%% Chance to Return", "create.recipe.heat_requirement.none": "UNLOCALIZED: No Heating Required", "create.recipe.heat_requirement.heated": "UNLOCALIZED: Heated", "create.recipe.heat_requirement.superheated": "UNLOCALIZED: Super-Heated", @@ -578,7 +580,6 @@ "create.generic.unit.rpm": "UNLOCALIZED: RPM", "create.generic.unit.stress": "UNLOCALIZED: su", "create.generic.unit.degrees": "UNLOCALIZED: °", - "create.generic.unit.bucket": "UNLOCALIZED: 1 Bucket", "create.generic.unit.millibuckets": "UNLOCALIZED: %1$smB", "create.action.scroll": "Scroll", diff --git a/src/generated/resources/assets/create/lang/unfinished/pt_br.json b/src/generated/resources/assets/create/lang/unfinished/pt_br.json index be9d0d07d..477382187 100644 --- a/src/generated/resources/assets/create/lang/unfinished/pt_br.json +++ b/src/generated/resources/assets/create/lang/unfinished/pt_br.json @@ -1,5 +1,5 @@ { - "_": "Missing Localizations: 929", + "_": "Missing Localizations: 935", "_": "->------------------------] Game Elements [------------------------<-", @@ -379,6 +379,8 @@ "entity.create.stationary_contraption": "UNLOCALIZED: Stationary Contraption", "entity.create.super_glue": "UNLOCALIZED: Super Glue", + "fluid.create.flowing_potion": "UNLOCALIZED: Potion", + "item.create.andesite_alloy": "UNLOCALIZED: Andesite Alloy", "item.create.attribute_filter": "UNLOCALIZED: Attribute Filter", "item.create.belt_connector": "Esteira Mecânica", @@ -541,26 +543,26 @@ "create.recipe.crushing": "Moendo", "create.recipe.milling": "UNLOCALIZED: Milling", - "create.recipe.splashing": "Lavando em Massa", - "create.recipe.splashing.fan": "Ventilador atras de Água corrente", - "create.recipe.smoking_via_fan": "Fumaceando em Massa", - "create.recipe.smoking_via_fan.fan": "Ventilador atras de Fogo", - "create.recipe.blasting_via_fan": "Fundindo em Massa", - "create.recipe.blasting_via_fan.fan": "Ventilador atras de Lava", + "create.recipe.fan_washing": "UNLOCALIZED: Bulk Washing", + "create.recipe.fan_washing.fan": "UNLOCALIZED: Fan behind Flowing Water", + "create.recipe.fan_smoking": "UNLOCALIZED: Bulk Smoking", + "create.recipe.fan_smoking.fan": "UNLOCALIZED: Fan behind Fire", + "create.recipe.fan_blasting": "UNLOCALIZED: Bulk Blasting", + "create.recipe.fan_blasting.fan": "UNLOCALIZED: Fan behind Lava", "create.recipe.pressing": "Prensa Mecânica", "create.recipe.mixing": "UNLOCALIZED: Mixing", - "create.recipe.shapeless_mixing": "UNLOCALIZED: Shapeless Crafting", + "create.recipe.automatic_shapeless": "UNLOCALIZED: Automated Shapeless Crafting", + "create.recipe.automatic_brewing": "UNLOCALIZED: Automated Brewing", "create.recipe.packing": "UNLOCALIZED: Compacting", + "create.recipe.automatic_packing": "UNLOCALIZED: Automated Packing", "create.recipe.sawing": "UNLOCALIZED: Sawing", - "create.recipe.mechanical_crafting": "UNLOCALIZED: Crafting", - "create.recipe.mechanical_crafting_exclusive": "UNLOCALIZED: Mechanical Crafting", + "create.recipe.mechanical_crafting": "UNLOCALIZED: Mechanical Crafting", + "create.recipe.automatic_shaped": "UNLOCALIZED: Automated Shaped Crafting", "create.recipe.block_cutting": "UNLOCALIZED: Block Cutting", "create.recipe.blockzapper_upgrade": "Blockzapper Portátil", "create.recipe.sandpaper_polishing": "UNLOCALIZED: Sandpaper Polishing", - "create.recipe.mystery_conversion": "UNLOCALIZED: Chromatic Metamorphosis", - "create.recipe.processing.catalyst": "UNLOCALIZED: Catalyst", + "create.recipe.mystery_conversion": "UNLOCALIZED: Mysterious Conversion", "create.recipe.processing.chance": "%1$s%% de chance", - "create.recipe.processing.chanceToReturn": "UNLOCALIZED: %1$s%% Chance to Return", "create.recipe.heat_requirement.none": "UNLOCALIZED: No Heating Required", "create.recipe.heat_requirement.heated": "UNLOCALIZED: Heated", "create.recipe.heat_requirement.superheated": "UNLOCALIZED: Super-Heated", @@ -578,7 +580,6 @@ "create.generic.unit.rpm": "UNLOCALIZED: RPM", "create.generic.unit.stress": "UNLOCALIZED: su", "create.generic.unit.degrees": "UNLOCALIZED: °", - "create.generic.unit.bucket": "UNLOCALIZED: 1 Bucket", "create.generic.unit.millibuckets": "UNLOCALIZED: %1$smB", "create.action.scroll": "Rolar", diff --git a/src/generated/resources/assets/create/lang/unfinished/ru_ru.json b/src/generated/resources/assets/create/lang/unfinished/ru_ru.json index c9e93e431..d31f1b1ef 100644 --- a/src/generated/resources/assets/create/lang/unfinished/ru_ru.json +++ b/src/generated/resources/assets/create/lang/unfinished/ru_ru.json @@ -1,5 +1,5 @@ { - "_": "Missing Localizations: 923", + "_": "Missing Localizations: 929", "_": "->------------------------] Game Elements [------------------------<-", @@ -379,6 +379,8 @@ "entity.create.stationary_contraption": "UNLOCALIZED: Stationary Contraption", "entity.create.super_glue": "UNLOCALIZED: Super Glue", + "fluid.create.flowing_potion": "UNLOCALIZED: Potion", + "item.create.andesite_alloy": "UNLOCALIZED: Andesite Alloy", "item.create.attribute_filter": "UNLOCALIZED: Attribute Filter", "item.create.belt_connector": "Механическая лента", @@ -541,26 +543,26 @@ "create.recipe.crushing": "Дробление", "create.recipe.milling": "UNLOCALIZED: Milling", - "create.recipe.splashing": "Промывка вентилятором", - "create.recipe.splashing.fan": "Вентилятор за проточной водой", - "create.recipe.smoking_via_fan": "Копчение вентилятором", - "create.recipe.smoking_via_fan.fan": "Вентилятор за огнём", - "create.recipe.blasting_via_fan": "Плавление вентилятором", - "create.recipe.blasting_via_fan.fan": "Вентелятор за лавой", + "create.recipe.fan_washing": "UNLOCALIZED: Bulk Washing", + "create.recipe.fan_washing.fan": "UNLOCALIZED: Fan behind Flowing Water", + "create.recipe.fan_smoking": "UNLOCALIZED: Bulk Smoking", + "create.recipe.fan_smoking.fan": "UNLOCALIZED: Fan behind Fire", + "create.recipe.fan_blasting": "UNLOCALIZED: Bulk Blasting", + "create.recipe.fan_blasting.fan": "UNLOCALIZED: Fan behind Lava", "create.recipe.pressing": "Механический пресс", "create.recipe.mixing": "UNLOCALIZED: Mixing", - "create.recipe.shapeless_mixing": "UNLOCALIZED: Shapeless Crafting", + "create.recipe.automatic_shapeless": "UNLOCALIZED: Automated Shapeless Crafting", + "create.recipe.automatic_brewing": "UNLOCALIZED: Automated Brewing", "create.recipe.packing": "UNLOCALIZED: Compacting", + "create.recipe.automatic_packing": "UNLOCALIZED: Automated Packing", "create.recipe.sawing": "UNLOCALIZED: Sawing", - "create.recipe.mechanical_crafting": "UNLOCALIZED: Crafting", - "create.recipe.mechanical_crafting_exclusive": "UNLOCALIZED: Mechanical Crafting", + "create.recipe.mechanical_crafting": "UNLOCALIZED: Mechanical Crafting", + "create.recipe.automatic_shaped": "UNLOCALIZED: Automated Shaped Crafting", "create.recipe.block_cutting": "UNLOCALIZED: Block Cutting", "create.recipe.blockzapper_upgrade": "Портативный размещатель блоков", "create.recipe.sandpaper_polishing": "UNLOCALIZED: Sandpaper Polishing", - "create.recipe.mystery_conversion": "UNLOCALIZED: Chromatic Metamorphosis", - "create.recipe.processing.catalyst": "UNLOCALIZED: Catalyst", + "create.recipe.mystery_conversion": "UNLOCALIZED: Mysterious Conversion", "create.recipe.processing.chance": "%1$s%% шанс выпадения", - "create.recipe.processing.chanceToReturn": "UNLOCALIZED: %1$s%% Chance to Return", "create.recipe.heat_requirement.none": "UNLOCALIZED: No Heating Required", "create.recipe.heat_requirement.heated": "UNLOCALIZED: Heated", "create.recipe.heat_requirement.superheated": "UNLOCALIZED: Super-Heated", @@ -578,7 +580,6 @@ "create.generic.unit.rpm": "UNLOCALIZED: RPM", "create.generic.unit.stress": "UNLOCALIZED: su", "create.generic.unit.degrees": "UNLOCALIZED: °", - "create.generic.unit.bucket": "UNLOCALIZED: 1 Bucket", "create.generic.unit.millibuckets": "UNLOCALIZED: %1$smB", "create.action.scroll": "КолМыши", diff --git a/src/generated/resources/assets/create/lang/unfinished/zh_cn.json b/src/generated/resources/assets/create/lang/unfinished/zh_cn.json index abc133ef5..1a3be748a 100644 --- a/src/generated/resources/assets/create/lang/unfinished/zh_cn.json +++ b/src/generated/resources/assets/create/lang/unfinished/zh_cn.json @@ -1,5 +1,5 @@ { - "_": "Missing Localizations: 217", + "_": "Missing Localizations: 221", "_": "->------------------------] Game Elements [------------------------<-", @@ -379,6 +379,8 @@ "entity.create.stationary_contraption": "固定结构", "entity.create.super_glue": "强力胶", + "fluid.create.flowing_potion": "UNLOCALIZED: Potion", + "item.create.andesite_alloy": "安山合金", "item.create.attribute_filter": "属性过滤器", "item.create.belt_connector": "传送带", @@ -541,26 +543,26 @@ "create.recipe.crushing": "粉碎", "create.recipe.milling": "研磨", - "create.recipe.splashing": "洗涤", - "create.recipe.splashing.fan": "在鼓风机前方倒水", - "create.recipe.smoking_via_fan": "UNLOCALIZED: Bulk Smoking", - "create.recipe.smoking_via_fan.fan": "UNLOCALIZED: Fan behind Fire", - "create.recipe.blasting_via_fan": "UNLOCALIZED: Bulk Smelting", - "create.recipe.blasting_via_fan.fan": "UNLOCALIZED: Fan behind Lava", + "create.recipe.fan_washing": "UNLOCALIZED: Bulk Washing", + "create.recipe.fan_washing.fan": "UNLOCALIZED: Fan behind Flowing Water", + "create.recipe.fan_smoking": "UNLOCALIZED: Bulk Smoking", + "create.recipe.fan_smoking.fan": "UNLOCALIZED: Fan behind Fire", + "create.recipe.fan_blasting": "UNLOCALIZED: Bulk Blasting", + "create.recipe.fan_blasting.fan": "UNLOCALIZED: Fan behind Lava", "create.recipe.pressing": "金属压片", "create.recipe.mixing": "混合搅拌", - "create.recipe.shapeless_mixing": "UNLOCALIZED: Shapeless Crafting", + "create.recipe.automatic_shapeless": "UNLOCALIZED: Automated Shapeless Crafting", + "create.recipe.automatic_brewing": "UNLOCALIZED: Automated Brewing", "create.recipe.packing": "压块塑形", + "create.recipe.automatic_packing": "UNLOCALIZED: Automated Packing", "create.recipe.sawing": "UNLOCALIZED: Sawing", "create.recipe.mechanical_crafting": "自动合成", - "create.recipe.mechanical_crafting_exclusive": "UNLOCALIZED: Mechanical Crafting", + "create.recipe.automatic_shaped": "UNLOCALIZED: Automated Shaped Crafting", "create.recipe.block_cutting": "方块切割", "create.recipe.blockzapper_upgrade": "UNLOCALIZED: Handheld Blockzapper", "create.recipe.sandpaper_polishing": "砂纸抛光", "create.recipe.mystery_conversion": "化合物变异", - "create.recipe.processing.catalyst": "催化剂", "create.recipe.processing.chance": "%1$s%% 概率", - "create.recipe.processing.chanceToReturn": "%1$s%% 概率返还", "create.recipe.heat_requirement.none": "UNLOCALIZED: No Heating Required", "create.recipe.heat_requirement.heated": "UNLOCALIZED: Heated", "create.recipe.heat_requirement.superheated": "UNLOCALIZED: Super-Heated", @@ -578,7 +580,6 @@ "create.generic.unit.rpm": "RPM", "create.generic.unit.stress": "su", "create.generic.unit.degrees": "°", - "create.generic.unit.bucket": "UNLOCALIZED: 1 Bucket", "create.generic.unit.millibuckets": "UNLOCALIZED: %1$smB", "create.action.scroll": "滚轮", diff --git a/src/generated/resources/data/create/recipes/emptying/water_bottle.json b/src/generated/resources/data/create/recipes/emptying/water_bottle.json deleted file mode 100644 index 1a34fd570..000000000 --- a/src/generated/resources/data/create/recipes/emptying/water_bottle.json +++ /dev/null @@ -1,18 +0,0 @@ -{ - "type": "create:emptying", - "ingredients": [ - { - "item": "minecraft:potion" - } - ], - "results": [ - { - "item": "minecraft:glass_bottle", - "count": 1 - }, - { - "fluid": "minecraft:water", - "amount": 250 - } - ] -} \ No newline at end of file diff --git a/src/generated/resources/data/create/recipes/filling/water_bottle.json b/src/generated/resources/data/create/recipes/filling/water_bottle.json deleted file mode 100644 index 4f99990bd..000000000 --- a/src/generated/resources/data/create/recipes/filling/water_bottle.json +++ /dev/null @@ -1,21 +0,0 @@ -{ - "type": "create:filling", - "ingredients": [ - { - "item": "minecraft:glass_bottle" - }, - { - "fluidTag": "minecraft:water", - "amount": 250 - } - ], - "results": [ - { - "item": "minecraft:potion", - "count": 1, - "nbt": { - "Potion": "minecraft:water" - } - } - ] -} \ No newline at end of file diff --git a/src/generated/resources/data/create/recipes/mixing/temp_cobble.json b/src/generated/resources/data/create/recipes/mixing/temp_cobble.json index 897f6d99c..def4cc8c2 100644 --- a/src/generated/resources/data/create/recipes/mixing/temp_cobble.json +++ b/src/generated/resources/data/create/recipes/mixing/temp_cobble.json @@ -2,11 +2,13 @@ "type": "create:mixing", "ingredients": [ { - "fluidTag": "minecraft:water", + "fluid": "minecraft:water", + "nbt": {}, "amount": 250 }, { - "fluidTag": "minecraft:lava", + "fluid": "minecraft:lava", + "nbt": {}, "amount": 25 } ], diff --git a/src/main/java/com/simibubi/create/AllFluids.java b/src/main/java/com/simibubi/create/AllFluids.java new file mode 100644 index 000000000..dba2a7919 --- /dev/null +++ b/src/main/java/com/simibubi/create/AllFluids.java @@ -0,0 +1,21 @@ +package com.simibubi.create; + +import com.simibubi.create.content.contraptions.fluids.potion.PotionFluid; +import com.simibubi.create.content.contraptions.fluids.potion.PotionFluid.PotionFluidAttributes; +import com.simibubi.create.foundation.data.CreateRegistrate; +import com.tterrag.registrate.util.entry.RegistryEntry; + +public class AllFluids { + + private static final CreateRegistrate REGISTRATE = Create.registrate(); + + public static RegistryEntry POTION = + REGISTRATE.virtualFluid("potion", PotionFluidAttributes::new, PotionFluid::new) + .lang(f -> "fluid.create.flowing_potion", "Potion") + .register(); + + // Load this class + + public static void register() {} + +} diff --git a/src/main/java/com/simibubi/create/Create.java b/src/main/java/com/simibubi/create/Create.java index e9e9f5753..26e6eb038 100644 --- a/src/main/java/com/simibubi/create/Create.java +++ b/src/main/java/com/simibubi/create/Create.java @@ -75,6 +75,7 @@ public class Create { AllBlocks.register(); AllItems.register(); + AllFluids.register(); AllTags.register(); AllPaletteBlocks.register(); AllEntityTypes.register(); diff --git a/src/main/java/com/simibubi/create/compat/jei/CreateJEI.java b/src/main/java/com/simibubi/create/compat/jei/CreateJEI.java index 5f8bf6fb0..a90208b6a 100644 --- a/src/main/java/com/simibubi/create/compat/jei/CreateJEI.java +++ b/src/main/java/com/simibubi/create/compat/jei/CreateJEI.java @@ -1,91 +1,178 @@ package com.simibubi.create.compat.jei; +import java.util.ArrayList; +import java.util.Collections; +import java.util.List; +import java.util.function.Function; +import java.util.function.Predicate; +import java.util.function.Supplier; +import java.util.stream.Collectors; + +import javax.annotation.Nonnull; + +import com.google.common.base.Predicates; import com.simibubi.create.AllBlocks; import com.simibubi.create.AllItems; import com.simibubi.create.AllRecipeTypes; import com.simibubi.create.Create; -import com.simibubi.create.compat.jei.category.*; +import com.simibubi.create.compat.jei.category.BlockCuttingCategory; import com.simibubi.create.compat.jei.category.BlockCuttingCategory.CondensedBlockCuttingRecipe; +import com.simibubi.create.compat.jei.category.BlockzapperUpgradeCategory; +import com.simibubi.create.compat.jei.category.CreateRecipeCategory; +import com.simibubi.create.compat.jei.category.CrushingCategory; +import com.simibubi.create.compat.jei.category.FanBlastingCategory; +import com.simibubi.create.compat.jei.category.FanSmokingCategory; +import com.simibubi.create.compat.jei.category.FanWashingCategory; +import com.simibubi.create.compat.jei.category.MechanicalCraftingCategory; +import com.simibubi.create.compat.jei.category.MillingCategory; +import com.simibubi.create.compat.jei.category.MixingCategory; +import com.simibubi.create.compat.jei.category.MysteriousItemConversionCategory; +import com.simibubi.create.compat.jei.category.PackingCategory; +import com.simibubi.create.compat.jei.category.PolishingCategory; +import com.simibubi.create.compat.jei.category.PressingCategory; +import com.simibubi.create.compat.jei.category.ProcessingViaFanCategory; +import com.simibubi.create.compat.jei.category.SawingCategory; import com.simibubi.create.content.contraptions.components.press.MechanicalPressTileEntity; +import com.simibubi.create.content.contraptions.fluids.potion.PotionMixingRecipeManager; import com.simibubi.create.content.contraptions.processing.BasinRecipe; import com.simibubi.create.content.logistics.block.inventories.AdjustableCrateScreen; import com.simibubi.create.content.logistics.item.filter.AbstractFilterScreen; import com.simibubi.create.content.schematics.block.SchematicannonScreen; import com.simibubi.create.foundation.config.AllConfigs; import com.simibubi.create.foundation.config.CRecipes; -import com.simibubi.create.foundation.fluid.FluidIngredient; -import com.simibubi.create.foundation.utility.Lang; +import com.simibubi.create.foundation.config.ConfigBase.ConfigBool; + import mezz.jei.api.IModPlugin; import mezz.jei.api.JeiPlugin; -import mezz.jei.api.ingredients.IIngredientType; -import mezz.jei.api.registration.*; -import net.minecraft.block.Blocks; +import mezz.jei.api.registration.IGuiHandlerRegistration; +import mezz.jei.api.registration.IRecipeCatalystRegistration; +import mezz.jei.api.registration.IRecipeCategoryRegistration; +import mezz.jei.api.registration.IRecipeRegistration; +import mezz.jei.api.registration.ISubtypeRegistration; import net.minecraft.client.Minecraft; import net.minecraft.item.ItemStack; -import net.minecraft.item.crafting.*; +import net.minecraft.item.crafting.ICraftingRecipe; +import net.minecraft.item.crafting.IRecipe; +import net.minecraft.item.crafting.IRecipeSerializer; +import net.minecraft.item.crafting.IRecipeType; +import net.minecraft.item.crafting.ShapedRecipe; +import net.minecraft.util.IItemProvider; import net.minecraft.util.ResourceLocation; -import net.minecraft.util.text.StringTextComponent; -import net.minecraft.util.text.TextFormatting; - -import javax.annotation.Nonnull; -import java.util.List; -import java.util.function.Predicate; -import java.util.stream.Collectors; @JeiPlugin public class CreateJEI implements IModPlugin { private static final ResourceLocation ID = new ResourceLocation(Create.ID, "jei_plugin"); - public static final IIngredientType FLUIDS = new IIngredientType() { - - @Override - @Nonnull - public Class getIngredientClass() { - return FluidIngredient.class; - } - }; - - private final MillingCategory millingCategory; - private final CrushingCategory crushingCategory; - private final SplashingCategory splashingCategory; - private final SmokingViaFanCategory smokingCategory; - private final PressingCategory pressingCategory; - private final BlastingViaFanCategory blastingCategory; - private final BlockzapperUpgradeCategory blockzapperCategory; - private final MixingCategory shapelessMixingCategory; - private final MixingCategory mixingCategory; - private final SawingCategory sawingCategory; - private final BlockCuttingCategory blockCuttingCategory; - private final PackingCategory packingCategory; - private final PolishingCategory polishingCategory; - private final MysteriousItemConversionCategory mysteryConversionCategory; - private final MechanicalCraftingCategory mechanicalCraftingCategory; - private final MechanicalCraftingCategory mechanicalCraftingExclusiveCategory; - @Override @Nonnull public ResourceLocation getPluginUid() { return ID; } - public CreateJEI() { - millingCategory = new MillingCategory(); - crushingCategory = new CrushingCategory(); - splashingCategory = new SplashingCategory(); - pressingCategory = new PressingCategory(); - smokingCategory = new SmokingViaFanCategory(); - blastingCategory = new BlastingViaFanCategory(); - blockzapperCategory = new BlockzapperUpgradeCategory(); - shapelessMixingCategory = new MixingCategory(true); - mixingCategory = new MixingCategory(false); - sawingCategory = new SawingCategory(); - blockCuttingCategory = new BlockCuttingCategory(); - packingCategory = new PackingCategory(); - polishingCategory = new PolishingCategory(); - mysteryConversionCategory = new MysteriousItemConversionCategory(); - mechanicalCraftingCategory = new MechanicalCraftingCategory(true); - mechanicalCraftingExclusiveCategory = new MechanicalCraftingCategory(false); + final List> ALL = new ArrayList<>(); + final CreateRecipeCategory + + milling = register("milling", MillingCategory::new).recipes(AllRecipeTypes.MILLING) + .catalyst(AllBlocks.MILLSTONE::get) + .build(), + + crushing = register("crushing", CrushingCategory::new).recipes(AllRecipeTypes.CRUSHING) + .recipesExcluding(AllRecipeTypes.MILLING::getType, AllRecipeTypes.CRUSHING::getType) + .catalyst(AllBlocks.CRUSHING_WHEEL::get) + .build(), + + pressing = register("pressing", PressingCategory::new).recipes(AllRecipeTypes.PRESSING) + .catalyst(AllBlocks.MECHANICAL_PRESS::get) + .build(), + + washing = register("fan_washing", FanWashingCategory::new).recipes(AllRecipeTypes.SPLASHING) + .catalystStack(ProcessingViaFanCategory.getFan("fan_washing")) + .build(), + + smoking = register("fan_smoking", FanSmokingCategory::new).recipes(() -> IRecipeType.SMOKING) + .catalystStack(ProcessingViaFanCategory.getFan("fan_smoking")) + .build(), + + blasting = register("fan_blasting", FanBlastingCategory::new) + .recipesExcluding(() -> IRecipeType.SMELTING, () -> IRecipeType.SMOKING) + .catalystStack(ProcessingViaFanCategory.getFan("fan_blasting")) + .build(), + + blockzapper = register("blockzapper_upgrade", BlockzapperUpgradeCategory::new) + .recipes(AllRecipeTypes.BLOCKZAPPER_UPGRADE.serializer.getRegistryName()) + .catalyst(AllItems.BLOCKZAPPER::get) + .build(), + + mixing = register("mixing", MixingCategory::standard).recipes(AllRecipeTypes.MIXING::getType) + .catalyst(AllBlocks.MECHANICAL_MIXER::get) + .catalyst(AllBlocks.BASIN::get) + .build(), + + autoShapeless = register("automatic_shapeless", MixingCategory::autoShapeless) + .recipes(r -> r.getSerializer() == IRecipeSerializer.CRAFTING_SHAPELESS + && !MechanicalPressTileEntity.canCompress(r.getIngredients()), BasinRecipe::convertShapeless) + .catalyst(AllBlocks.MECHANICAL_MIXER::get) + .catalyst(AllBlocks.BASIN::get) + .enableWhen(c -> c.allowShapelessInMixer) + .build(), + + brewing = register("automatic_brewing", MixingCategory::autoBrewing) + .recipeList(PotionMixingRecipeManager::getAllBrewingRecipes) + .catalyst(AllBlocks.MECHANICAL_MIXER::get) + .catalyst(AllBlocks.BASIN::get) + .build(), + + sawing = register("sawing", SawingCategory::new).recipes(AllRecipeTypes.CUTTING) + .catalyst(AllBlocks.MECHANICAL_SAW::get) + .build(), + + blockCutting = register("block_cutting", BlockCuttingCategory::new) + .recipeList(() -> CondensedBlockCuttingRecipe.condenseRecipes(findRecipesByType(IRecipeType.STONECUTTING))) + .catalyst(AllBlocks.MECHANICAL_SAW::get) + .enableWhen(c -> c.allowStonecuttingOnSaw) + .build(), + + packing = register("packing", PackingCategory::standard).recipes(AllRecipeTypes.COMPACTING) + .catalyst(AllBlocks.MECHANICAL_PRESS::get) + .catalyst(AllBlocks.BASIN::get) + .build(), + + autoSquare = register("automatic_packing", PackingCategory::autoSquare) + .recipes(r -> (r instanceof ICraftingRecipe) && MechanicalPressTileEntity.canCompress(r.getIngredients()), + BasinRecipe::convertShapeless) + .catalyst(AllBlocks.MECHANICAL_PRESS::get) + .catalyst(AllBlocks.BASIN::get) + .enableWhen(c -> c.allowShapedSquareInPress) + .build(), + + polishing = register("sandpaper_polishing", PolishingCategory::new).recipes(AllRecipeTypes.SANDPAPER_POLISHING) + .catalyst(AllItems.SAND_PAPER::get) + .catalyst(AllItems.RED_SAND_PAPER::get) + .build(), + + mysteryConversion = register("mystery_conversion", MysteriousItemConversionCategory::new) + .recipeList(MysteriousItemConversionCategory::getRecipes) + .build(), + + autoShaped = register("automatic_shaped", MechanicalCraftingCategory::new) + .recipes( + r -> (r.getType() == IRecipeType.CRAFTING && r.getType() != AllRecipeTypes.MECHANICAL_CRAFTING.type) + && (r instanceof ShapedRecipe)) + .catalyst(AllBlocks.MECHANICAL_CRAFTER::get) + .enableWhen(c -> c.allowRegularCraftingInCrafter) + .build(), + + mechanicalCrafting = + register("mechanical_crafting", MechanicalCraftingCategory::new).recipes(AllRecipeTypes.MECHANICAL_CRAFTING) + .catalyst(AllBlocks.MECHANICAL_CRAFTER::get) + .build() + + ; + + private > CategoryBuilder register(String name, + Supplier> supplier) { + return new CategoryBuilder(name, supplier); } @Override @@ -95,136 +182,135 @@ public class CreateJEI implements IModPlugin { @Override public void registerCategories(IRecipeCategoryRegistration registration) { - registration.addRecipeCategories(millingCategory, crushingCategory, splashingCategory, pressingCategory, smokingCategory, blastingCategory, blockzapperCategory, mixingCategory, shapelessMixingCategory, sawingCategory, blockCuttingCategory, packingCategory, polishingCategory, mysteryConversionCategory, mechanicalCraftingCategory, mechanicalCraftingExclusiveCategory); + ALL.forEach(registration::addRecipeCategories); } @Override public void registerRecipes(IRecipeRegistration registration) { - CRecipes recipeConfig = AllConfigs.SERVER.recipes; - - registration.addRecipes(findRecipes(AllRecipeTypes.MILLING), millingCategory.getUid()); - registration.addRecipes(findRecipes(AllRecipeTypes.CRUSHING), crushingCategory.getUid()); - registration.addRecipes(findRecipesByTypeExcluding(AllRecipeTypes.MILLING.getType(), AllRecipeTypes.CRUSHING.getType()), crushingCategory.getUid()); - registration.addRecipes(findRecipes(AllRecipeTypes.SPLASHING), splashingCategory.getUid()); - registration.addRecipes(findRecipes(AllRecipeTypes.PRESSING), pressingCategory.getUid()); - registration.addRecipes(findRecipesById(AllRecipeTypes.BLOCKZAPPER_UPGRADE.serializer.getRegistryName()), blockzapperCategory.getUid()); - registration.addRecipes(findRecipesByType(IRecipeType.SMOKING), smokingCategory.getUid()); - registration.addRecipes(findRecipesByTypeExcluding(IRecipeType.SMELTING, IRecipeType.SMOKING), blastingCategory.getUid()); - registration.addRecipes(findRecipes(AllRecipeTypes.MIXING), mixingCategory.getUid()); - - if (recipeConfig.allowShapelessInMixer.get()) - registration.addRecipes( - findRecipes(r -> r.getSerializer() == IRecipeSerializer.CRAFTING_SHAPELESS && !MechanicalPressTileEntity.canCompress(r.getIngredients())) - .stream() - .map(BasinRecipe::convert) - .collect(Collectors.toList()), - shapelessMixingCategory.getUid()); - - registration.addRecipes(findRecipes(AllRecipeTypes.CUTTING), sawingCategory.getUid()); - - if (recipeConfig.allowStonecuttingOnSaw.get()) - registration.addRecipes(CondensedBlockCuttingRecipe.condenseRecipes(findRecipesByType(IRecipeType.STONECUTTING)), blockCuttingCategory.getUid()); - - registration.addRecipes(findRecipes(AllRecipeTypes.COMPACTING), packingCategory.getUid()); - - if (recipeConfig.allowShapedSquareInPress.get()) - registration.addRecipes( - findRecipes(r -> (r instanceof ICraftingRecipe) && MechanicalPressTileEntity.canCompress(r.getIngredients())) - .stream() - .map(BasinRecipe::convert) - .collect(Collectors.toList()), - packingCategory.getUid()); - - registration.addRecipes(findRecipes(AllRecipeTypes.SANDPAPER_POLISHING), polishingCategory.getUid()); - registration.addRecipes(MysteriousItemConversionCategory.getRecipes(), mysteryConversionCategory.getUid()); - registration.addRecipes(findRecipes(r -> (r.getType() == AllRecipeTypes.MECHANICAL_CRAFTING.type)), mechanicalCraftingExclusiveCategory.getUid()); - - if (recipeConfig.allowRegularCraftingInCrafter.get()) - registration.addRecipes( - findRecipes(r -> (r.getType() == IRecipeType.CRAFTING && r.getType() != AllRecipeTypes.MECHANICAL_CRAFTING.type) && (r instanceof ShapedRecipe)), - mechanicalCraftingCategory.getUid()); + ALL.forEach(c -> c.recipes.forEach(s -> registration.addRecipes(s.get(), c.getUid()))); } @Override public void registerRecipeCatalysts(IRecipeCatalystRegistration registration) { - ItemStack fan = new ItemStack(AllBlocks.ENCASED_FAN.get()); - - ItemStack splashingFan = fan.copy().setDisplayName(new StringTextComponent(TextFormatting.RESET + Lang.translate("recipe.splashing.fan"))); - ItemStack smokingFan = fan.copy().setDisplayName(new StringTextComponent(TextFormatting.RESET + Lang.translate("recipe.smokingViaFan.fan"))); - ItemStack blastingFan = fan.copy().setDisplayName(new StringTextComponent(TextFormatting.RESET + Lang.translate("recipe.blastingViaFan.fan"))); - - registration.addRecipeCatalyst(new ItemStack(AllBlocks.MILLSTONE.get()), millingCategory.getUid()); - - registration.addRecipeCatalyst(new ItemStack(AllBlocks.CRUSHING_WHEEL.get()), crushingCategory.getUid()); - - registration.addRecipeCatalyst(splashingFan, splashingCategory.getUid()); - - registration.addRecipeCatalyst(smokingFan, smokingCategory.getUid()); - - registration.addRecipeCatalyst(blastingFan, blastingCategory.getUid()); - - registration.addRecipeCatalyst(new ItemStack(AllBlocks.MECHANICAL_PRESS.get()), pressingCategory.getUid()); - - registration.addRecipeCatalyst(new ItemStack(AllItems.BLOCKZAPPER.get()), blockzapperCategory.getUid()); - - registration.addRecipeCatalyst(new ItemStack(AllBlocks.MECHANICAL_MIXER.get()), shapelessMixingCategory.getUid()); - registration.addRecipeCatalyst(new ItemStack(AllBlocks.BASIN.get()), shapelessMixingCategory.getUid()); - - registration.addRecipeCatalyst(new ItemStack(AllBlocks.MECHANICAL_MIXER.get()), mixingCategory.getUid()); - registration.addRecipeCatalyst(new ItemStack(AllBlocks.BASIN.get()), mixingCategory.getUid()); - - registration.addRecipeCatalyst(new ItemStack(AllBlocks.MECHANICAL_SAW.get()), sawingCategory.getUid()); - - registration.addRecipeCatalyst(new ItemStack(AllBlocks.MECHANICAL_SAW.get()), blockCuttingCategory.getUid()); - registration.addRecipeCatalyst(new ItemStack(Blocks.STONECUTTER), blockCuttingCategory.getUid()); - - registration.addRecipeCatalyst(new ItemStack(AllBlocks.MECHANICAL_PRESS.get()), packingCategory.getUid()); - registration.addRecipeCatalyst(new ItemStack(AllBlocks.BASIN.get()), packingCategory.getUid()); - - registration.addRecipeCatalyst(AllItems.SAND_PAPER.asStack(), polishingCategory.getUid()); - registration.addRecipeCatalyst(AllItems.RED_SAND_PAPER.asStack(), polishingCategory.getUid()); - - registration.addRecipeCatalyst(new ItemStack(AllBlocks.MECHANICAL_CRAFTER.get()), mechanicalCraftingCategory.getUid()); - - registration.addRecipeCatalyst(new ItemStack(AllBlocks.MECHANICAL_CRAFTER.get()), mechanicalCraftingExclusiveCategory.getUid()); + ALL.forEach(c -> c.recipeCatalysts.forEach(s -> registration.addRecipeCatalyst(s.get(), c.getUid()))); } @Override public void registerGuiHandlers(IGuiHandlerRegistration registration) { registration.addGuiContainerHandler(AdjustableCrateScreen.class, new SlotMover<>()); registration.addGuiContainerHandler(SchematicannonScreen.class, new SlotMover<>()); - registration.addGhostIngredientHandler(AbstractFilterScreen.class, new FilterGhostIngredientHandler()); } - private static List> findRecipes(AllRecipeTypes recipe) { - return findRecipesByType(recipe.type); + private class CategoryBuilder> { + CreateRecipeCategory category; + private Predicate pred; + + CategoryBuilder(String name, Supplier> category) { + this.category = category.get(); + this.category.setCategoryId(name); + this.pred = Predicates.alwaysTrue(); + } + + CategoryBuilder catalyst(Supplier supplier) { + return catalystStack(() -> new ItemStack(supplier.get().asItem())); + } + + CategoryBuilder catalystStack(Supplier supplier) { + category.recipeCatalysts.add(supplier); + return this; + } + + CategoryBuilder recipes(AllRecipeTypes recipeTypeEntry) { + return recipes(recipeTypeEntry::getType); + } + + CategoryBuilder recipes(Supplier> recipeType) { + return recipes(r -> r.getType() == recipeType.get()); + } + + CategoryBuilder recipes(ResourceLocation serializer) { + return recipes(r -> r.getSerializer() + .getRegistryName() + .equals(serializer)); + } + + CategoryBuilder recipes(Predicate> pred) { + return recipeList(() -> findRecipes(pred)); + } + + CategoryBuilder recipes(Predicate> pred, Function, T> converter) { + return recipeList(() -> findRecipes(pred), converter); + } + + CategoryBuilder recipeList(Supplier>> list) { + return recipeList(list, null); + } + + CategoryBuilder recipeList(Supplier>> list, Function, T> converter) { + category.recipes.add(() -> { + if (!this.pred.test(AllConfigs.SERVER.recipes)) + return Collections.emptyList(); + if (converter != null) + return list.get() + .stream() + .map(converter) + .collect(Collectors.toList()); + return list.get(); + }); + return this; + } + + CategoryBuilder recipesExcluding(Supplier> recipeType, + Supplier> excluded) { + category.recipes.add(() -> { + if (!this.pred.test(AllConfigs.SERVER.recipes)) + return Collections.emptyList(); + return findRecipesByTypeExcluding(recipeType.get(), excluded.get()); + }); + return this; + } + + CategoryBuilder enableWhen(Function configValue) { + this.pred = c -> configValue.apply(c) + .get() + .booleanValue(); + return this; + } + + CreateRecipeCategory build() { + ALL.add(category); + return category; + } + } - private static List> findRecipes(Predicate> predicate) { - return Minecraft.getInstance().world.getRecipeManager() - .getRecipes() - .stream() - .filter(predicate) - .collect(Collectors.toList()); - } - - private static List> findRecipesByType(IRecipeType type) { + static List> findRecipesByType(IRecipeType type) { return findRecipes(r -> r.getType() == type); } - private static List> findRecipesById(ResourceLocation id) { - return findRecipes(r -> r.getSerializer().getRegistryName().equals(id)); + static List> findRecipes(Predicate> predicate) { + return Minecraft.getInstance().world.getRecipeManager() + .getRecipes() + .stream() + .filter(predicate) + .collect(Collectors.toList()); } - private static List> findRecipesByTypeExcluding(IRecipeType type, IRecipeType excludingType) { - List> byType = findRecipesByType(type); - List> byExcludingType = findRecipesByType(excludingType); + static List> findRecipesByTypeExcluding(IRecipeType type, IRecipeType excludingType) { + List> byType = findRecipes(r -> r.getType() == type); + List> byExcludingType = findRecipes(r -> r.getType() == excludingType); byType.removeIf(recipe -> { for (IRecipe r : byExcludingType) { - ItemStack[] matchingStacks = recipe.getIngredients().get(0).getMatchingStacks(); - if (matchingStacks.length == 0) return true; - if (r.getIngredients().get(0).test(matchingStacks[0])) return true; + ItemStack[] matchingStacks = recipe.getIngredients() + .get(0) + .getMatchingStacks(); + if (matchingStacks.length == 0) + return true; + if (r.getIngredients() + .get(0) + .test(matchingStacks[0])) + return true; } return false; }); diff --git a/src/main/java/com/simibubi/create/compat/jei/category/BasinCategory.java b/src/main/java/com/simibubi/create/compat/jei/category/BasinCategory.java index af117b494..3f82a09a3 100644 --- a/src/main/java/com/simibubi/create/compat/jei/category/BasinCategory.java +++ b/src/main/java/com/simibubi/create/compat/jei/category/BasinCategory.java @@ -32,8 +32,11 @@ import net.minecraftforge.fluids.FluidStack; public class BasinCategory extends CreateRecipeCategory { - public BasinCategory(String id, IDrawable icon, IDrawable background) { - super(id, icon, background); + private boolean needsHeating; + + public BasinCategory(boolean needsHeating, IDrawable icon, IDrawable background) { + super(icon, background); + this.needsHeating = needsHeating; } @Override @@ -44,13 +47,13 @@ public class BasinCategory extends CreateRecipeCategory { @Override public void setIngredients(BasinRecipe recipe, IIngredients ingredients) { List itemIngredients = new ArrayList<>(recipe.getIngredients()); - + HeatCondition requiredHeat = recipe.getRequiredHeat(); if (!requiredHeat.testBlazeBurner(HeatLevel.NONE)) itemIngredients.add(Ingredient.fromItems(AllBlocks.BLAZE_BURNER.get())); if (!requiredHeat.testBlazeBurner(HeatLevel.KINDLED)) itemIngredients.add(Ingredient.fromItems(AllItems.BLAZE_CAKE.get())); - + ingredients.setInputIngredients(itemIngredients); ingredients.setInputLists(VanillaTypes.FLUID, recipe.getFluidIngredients() .stream() @@ -108,7 +111,7 @@ public class BasinCategory extends CreateRecipeCategory { fluidStacks.init(j, true, 17 + xOffset + (i2 % 3) * 19, 51 - (i2 / 3) * 19 + yOffset); List stacks = fluidIngredients.get(j) .getMatchingFluidStacks(); - fluidStacks.set(j, stacks); + fluidStacks.set(j, withImprovedVisibility(stacks)); } if (!itemOutput.isEmpty()) { @@ -120,11 +123,11 @@ public class BasinCategory extends CreateRecipeCategory { if (!fluidOutput.isEmpty()) { fluidStacks.init(j, false, 142, 51 + yOffset); - fluidStacks.set(j, fluidOutput); + fluidStacks.set(j, withImprovedVisibility(fluidOutput)); } addFluidTooltip(fluidStacks, fluidIngredients, ImmutableList.of(fluidOutput)); - + HeatCondition requiredHeat = recipe.getRequiredHeat(); if (!requiredHeat.testBlazeBurner(HeatLevel.NONE)) { itemStacks.init(++i, true, 133, 80); @@ -136,6 +139,19 @@ public class BasinCategory extends CreateRecipeCategory { } } + public List withImprovedVisibility(List stacks) { + return stacks.stream() + .map(this::withImprovedVisibility) + .collect(Collectors.toList()); + } + + public FluidStack withImprovedVisibility(FluidStack stack) { + FluidStack display = stack.copy(); + int displayedAmount = (int) (stack.getAmount() * .75f) + 250; + display.setAmount(displayedAmount); + return display; + } + @Override public void draw(BasinRecipe recipe, double mouseX, double mouseY) { List> actualIngredients = ItemHelper.condenseIngredients(recipe.getIngredients()); @@ -156,9 +172,11 @@ public class BasinCategory extends CreateRecipeCategory { AllGuiTextures shadow = noHeat ? AllGuiTextures.JEI_SHADOW : AllGuiTextures.JEI_LIGHT; shadow.draw(81, 58 + (noHeat ? 10 : 30)); + if (!needsHeating) + return; + AllGuiTextures heatBar = noHeat ? AllGuiTextures.JEI_NO_HEAT_BAR : AllGuiTextures.JEI_HEAT_BAR; heatBar.draw(4, 80); - Minecraft.getInstance().fontRenderer.drawStringWithShadow(Lang.translate(requiredHeat.getTranslationKey()), 9, 85, requiredHeat.getColor()); } diff --git a/src/main/java/com/simibubi/create/compat/jei/category/BlockCuttingCategory.java b/src/main/java/com/simibubi/create/compat/jei/category/BlockCuttingCategory.java index bb181f51f..6326ed9ab 100644 --- a/src/main/java/com/simibubi/create/compat/jei/category/BlockCuttingCategory.java +++ b/src/main/java/com/simibubi/create/compat/jei/category/BlockCuttingCategory.java @@ -26,7 +26,7 @@ public class BlockCuttingCategory extends CreateRecipeCategory { public BlockzapperUpgradeCategory() { - super("blockzapper_upgrade", itemIcon(AllItems.BLOCKZAPPER.get()), - new ScreenResourceWrapper(BLOCKZAPPER_UPGRADE_RECIPE)); + super(itemIcon(AllItems.BLOCKZAPPER.get()), new ScreenResourceWrapper(BLOCKZAPPER_UPGRADE_RECIPE)); } @Override @@ -57,7 +56,8 @@ public class BlockzapperUpgradeCategory extends CreateRecipeCategory 91 + 52 || mouseY < 1 || mouseY > 53) return list; list.addAll(recipe.getRecipeOutput() - .getTooltip(Minecraft.getInstance().player, - Minecraft.getInstance().gameSettings.advancedItemTooltips ? ITooltipFlag.TooltipFlags.ADVANCED - : ITooltipFlag.TooltipFlags.NORMAL) - .stream().map(ITextComponent::getFormattedText).collect(Collectors.toList())); + .getTooltip(Minecraft.getInstance().player, + Minecraft.getInstance().gameSettings.advancedItemTooltips ? ITooltipFlag.TooltipFlags.ADVANCED + : ITooltipFlag.TooltipFlags.NORMAL) + .stream() + .map(ITextComponent::getFormattedText) + .collect(Collectors.toList())); return list; } @Override public void draw(BlockzapperUpgradeRecipe recipe, double mouseX, double mouseY) { FontRenderer font = Minecraft.getInstance().fontRenderer; - String componentName = - Lang.translate("blockzapper.component." + Lang.asId(recipe.getUpgradedComponent().name())); + String componentName = Lang.translate("blockzapper.component." + Lang.asId(recipe.getUpgradedComponent() + .name())); String text = "+ " + recipe.getTier().color + componentName; font.drawStringWithShadow(text, (BLOCKZAPPER_UPGRADE_RECIPE.width - font.getStringWidth(text)) / 2, 57, - 0x8B8B8B); - + 0x8B8B8B); + RenderSystem.pushMatrix(); RenderSystem.translated(126, 0, 0); RenderSystem.scaled(3.5, 3.5, 3.5); RenderSystem.translated(-10, 0, 0); RenderSystem.color3f(1, 1, 1); - GuiGameElement.of(recipe.getRecipeOutput()).render(); + GuiGameElement.of(recipe.getRecipeOutput()) + .render(); RenderSystem.popMatrix(); } } \ No newline at end of file diff --git a/src/main/java/com/simibubi/create/compat/jei/category/CreateRecipeCategory.java b/src/main/java/com/simibubi/create/compat/jei/category/CreateRecipeCategory.java index 7ae9784f8..60eaa0855 100644 --- a/src/main/java/com/simibubi/create/compat/jei/category/CreateRecipeCategory.java +++ b/src/main/java/com/simibubi/create/compat/jei/category/CreateRecipeCategory.java @@ -3,10 +3,14 @@ package com.simibubi.create.compat.jei.category; import java.util.ArrayList; import java.util.List; import java.util.Map; +import java.util.function.Supplier; +import java.util.stream.Collectors; +import com.simibubi.create.AllFluids; import com.simibubi.create.Create; import com.simibubi.create.compat.jei.DoubleItemIcon; import com.simibubi.create.compat.jei.EmptyBackground; +import com.simibubi.create.content.contraptions.fluids.potion.PotionFluidHandler; import com.simibubi.create.content.contraptions.processing.ProcessingOutput; import com.simibubi.create.content.contraptions.processing.ProcessingRecipe; import com.simibubi.create.foundation.fluid.FluidIngredient; @@ -21,23 +25,30 @@ import net.minecraft.item.ItemStack; import net.minecraft.item.crafting.IRecipe; import net.minecraft.util.IItemProvider; import net.minecraft.util.ResourceLocation; +import net.minecraft.util.text.ITextComponent; import net.minecraft.util.text.TextFormatting; import net.minecraftforge.fluids.FluidStack; public abstract class CreateRecipeCategory> implements IRecipeCategory { - private ResourceLocation uid; - private String name; + public List> recipeCatalysts = new ArrayList<>(); + public List>>> recipes = new ArrayList<>(); + public ResourceLocation uid; + + protected String name; private IDrawable icon; private IDrawable background; - public CreateRecipeCategory(String id, IDrawable icon, IDrawable background) { - uid = new ResourceLocation(Create.ID, id); - name = id; + public CreateRecipeCategory(IDrawable icon, IDrawable background) { this.background = background; this.icon = icon; } + public void setCategoryId(String name) { + this.uid = new ResourceLocation(Create.ID, name); + this.name = name; + } + @Override public IDrawable getIcon() { return icon; @@ -102,10 +113,25 @@ public abstract class CreateRecipeCategory> implements IRec inputs.forEach(f -> amounts.add(f.getRequiredAmount())); outputs.forEach(f -> amounts.add(f.getAmount())); - fluidStacks.addTooltipCallback((slotIndex, input, ingredient, tooltip) -> { + fluidStacks.addTooltipCallback((slotIndex, input, fluid, tooltip) -> { + if (fluid.getFluid() + .isEquivalentTo(AllFluids.POTION.get())) { + String name = PotionFluidHandler.getPotionName(fluid) + .getFormattedText(); + if (tooltip.isEmpty()) + tooltip.add(0, name); + else + tooltip.set(0, name); + + ArrayList potionTooltip = new ArrayList<>(); + PotionFluidHandler.addPotionTooltip(fluid, potionTooltip, 1); + tooltip.addAll(1, potionTooltip.stream() + .map(ITextComponent::getFormattedText) + .collect(Collectors.toList())); + } + int amount = amounts.get(slotIndex); - String text = TextFormatting.GOLD + (amount == 1000 ? Lang.translate("generic.unit.bucket") - : Lang.translate("generic.unit.millibuckets", amount)); + String text = TextFormatting.GOLD + Lang.translate("generic.unit.millibuckets", amount); if (tooltip.isEmpty()) tooltip.add(0, text); else diff --git a/src/main/java/com/simibubi/create/compat/jei/category/CrushingCategory.java b/src/main/java/com/simibubi/create/compat/jei/category/CrushingCategory.java index 64129fc4c..b4ceb8ad4 100644 --- a/src/main/java/com/simibubi/create/compat/jei/category/CrushingCategory.java +++ b/src/main/java/com/simibubi/create/compat/jei/category/CrushingCategory.java @@ -20,8 +20,7 @@ public class CrushingCategory extends CreateRecipeCategory results = recipe.getRollableResults(); int size = results.size(); int offset = -size * 19 / 2; for (int outputIndex = 0; outputIndex < size; outputIndex++) { itemStacks.init(outputIndex + 1, false, getBackground().getWidth() / 2 + offset + 19 * outputIndex, 78); - itemStacks.set(outputIndex + 1, results.get(outputIndex).getStack()); + itemStacks.set(outputIndex + 1, results.get(outputIndex) + .getStack()); } addStochasticTooltip(itemStacks, results); diff --git a/src/main/java/com/simibubi/create/compat/jei/category/BlastingViaFanCategory.java b/src/main/java/com/simibubi/create/compat/jei/category/FanBlastingCategory.java similarity index 68% rename from src/main/java/com/simibubi/create/compat/jei/category/BlastingViaFanCategory.java rename to src/main/java/com/simibubi/create/compat/jei/category/FanBlastingCategory.java index 96b526d40..4e48ac500 100644 --- a/src/main/java/com/simibubi/create/compat/jei/category/BlastingViaFanCategory.java +++ b/src/main/java/com/simibubi/create/compat/jei/category/FanBlastingCategory.java @@ -8,10 +8,10 @@ import net.minecraft.fluid.Fluids; import net.minecraft.item.Items; import net.minecraft.item.crafting.AbstractCookingRecipe; -public class BlastingViaFanCategory extends ProcessingViaFanCategory { +public class FanBlastingCategory extends ProcessingViaFanCategory { - public BlastingViaFanCategory() { - super("blasting_via_fan", doubleItemIcon(AllItems.PROPELLER.get(), Items.LAVA_BUCKET)); + public FanBlastingCategory() { + super(doubleItemIcon(AllItems.PROPELLER.get(), Items.LAVA_BUCKET)); } @Override @@ -24,9 +24,9 @@ public class BlastingViaFanCategory extends ProcessingViaFanCategory { +public class FanSmokingCategory extends ProcessingViaFanCategory { - public SmokingViaFanCategory() { - super("smoking_via_fan", doubleItemIcon(AllItems.PROPELLER.get(), Items.BLAZE_POWDER)); + public FanSmokingCategory() { + super(doubleItemIcon(AllItems.PROPELLER.get(), Items.BLAZE_POWDER)); } @Override diff --git a/src/main/java/com/simibubi/create/compat/jei/category/SplashingCategory.java b/src/main/java/com/simibubi/create/compat/jei/category/FanWashingCategory.java similarity index 93% rename from src/main/java/com/simibubi/create/compat/jei/category/SplashingCategory.java rename to src/main/java/com/simibubi/create/compat/jei/category/FanWashingCategory.java index c986affbd..7779b3047 100644 --- a/src/main/java/com/simibubi/create/compat/jei/category/SplashingCategory.java +++ b/src/main/java/com/simibubi/create/compat/jei/category/FanWashingCategory.java @@ -17,10 +17,10 @@ import mezz.jei.api.ingredients.IIngredients; import net.minecraft.fluid.Fluids; import net.minecraft.item.Items; -public class SplashingCategory extends ProcessingViaFanCategory { +public class FanWashingCategory extends ProcessingViaFanCategory { - public SplashingCategory() { - super("splashing", doubleItemIcon(AllItems.PROPELLER.get(), Items.WATER_BUCKET)); + public FanWashingCategory() { + super(doubleItemIcon(AllItems.PROPELLER.get(), Items.WATER_BUCKET)); } @Override diff --git a/src/main/java/com/simibubi/create/compat/jei/category/MechanicalCraftingCategory.java b/src/main/java/com/simibubi/create/compat/jei/category/MechanicalCraftingCategory.java index 3b38e6d70..7b731b1e4 100644 --- a/src/main/java/com/simibubi/create/compat/jei/category/MechanicalCraftingCategory.java +++ b/src/main/java/com/simibubi/create/compat/jei/category/MechanicalCraftingCategory.java @@ -1,9 +1,15 @@ package com.simibubi.create.compat.jei.category; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.List; +import java.util.stream.Collectors; + import com.mojang.blaze3d.systems.RenderSystem; import com.simibubi.create.AllBlocks; import com.simibubi.create.compat.jei.category.animations.AnimatedCrafter; import com.simibubi.create.foundation.gui.AllGuiTextures; + import mezz.jei.api.constants.VanillaTypes; import mezz.jei.api.gui.IRecipeLayout; import mezz.jei.api.gui.ingredient.IGuiItemStackGroup; @@ -23,13 +29,107 @@ import net.minecraft.util.NonNullList; import net.minecraft.util.text.ITextComponent; import net.minecraft.util.text.TextFormatting; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.List; -import java.util.stream.Collectors; - public class MechanicalCraftingCategory extends CreateRecipeCategory { + private final AnimatedCrafter crafter = new AnimatedCrafter(); + + public MechanicalCraftingCategory() { + super(itemIcon(AllBlocks.MECHANICAL_CRAFTER.get()), emptyBackground(177, 107)); + } + + @Override + public void setIngredients(ShapedRecipe recipe, IIngredients ingredients) { + ingredients.setInputIngredients(recipe.getIngredients()); + ingredients.setOutput(VanillaTypes.ITEM, recipe.getRecipeOutput()); + } + + @Override + public void setRecipe(IRecipeLayout recipeLayout, ShapedRecipe recipe, IIngredients ingredients) { + IGuiItemStackGroup itemStacks = recipeLayout.getItemStacks(); + NonNullList recipeIngredients = recipe.getIngredients(); + + itemStacks.init(0, false, 133, 80); + itemStacks.set(0, recipe.getRecipeOutput() + .getStack()); + + int x = getXPadding(recipe); + int y = getYPadding(recipe); + float scale = getScale(recipe); + int size = recipeIngredients.size(); + IIngredientRenderer renderer = new CrafterIngredientRenderer(recipe); + + for (int i = 0; i < size; i++) { + float f = 19 * scale; + int slotSize = (int) (16 * scale); + int xPosition = (int) (x + 1 + (i % recipe.getWidth()) * f); + int yPosition = (int) (y + 1 + (i / recipe.getWidth()) * f); + itemStacks.init(i + 1, true, renderer, xPosition, yPosition, slotSize, slotSize, 0, 0); + itemStacks.set(i + 1, Arrays.asList(recipeIngredients.get(i) + .getMatchingStacks())); + } + + } + + static int maxSize = 100; + + public static float getScale(ShapedRecipe recipe) { + int w = recipe.getWidth(); + int h = recipe.getHeight(); + return Math.min(1, maxSize / (19f * Math.max(w, h))); + } + + public static int getYPadding(ShapedRecipe recipe) { + return 3 + 50 - (int) (getScale(recipe) * recipe.getHeight() * 19 * .5); + } + + public static int getXPadding(ShapedRecipe recipe) { + return 3 + 50 - (int) (getScale(recipe) * recipe.getWidth() * 19 * .5); + } + + @Override + public void draw(ShapedRecipe recipe, double mouseX, double mouseY) { + RenderSystem.pushMatrix(); + float scale = getScale(recipe); + RenderSystem.translated(getXPadding(recipe), getYPadding(recipe), 0); + + for (int row = 0; row < recipe.getHeight(); row++) + for (int col = 0; col < recipe.getWidth(); col++) + if (!recipe.getIngredients() + .get(row * recipe.getWidth() + col) + .hasNoMatchingItems()) { + RenderSystem.pushMatrix(); + RenderSystem.translated(col * 19 * scale, row * 19 * scale, 0); + RenderSystem.scaled(scale, scale, scale); + AllGuiTextures.JEI_SLOT.draw(0, 0); + RenderSystem.popMatrix(); + } + + RenderSystem.popMatrix(); + + AllGuiTextures.JEI_SLOT.draw(133, 80); + AllGuiTextures.JEI_DOWN_ARROW.draw(128, 59); + crafter.draw(129, 25); + + RenderSystem.pushMatrix(); + RenderSystem.translated(0, 0, 300); + + RenderHelper.disableStandardItemLighting(); + int amount = 0; + for (Ingredient ingredient : recipe.getIngredients()) { + if (Ingredient.EMPTY == ingredient) + continue; + amount++; + } + + Minecraft.getInstance().fontRenderer.drawStringWithShadow(amount + "", 142, 39, 0xFFFFFF); + RenderSystem.popMatrix(); + } + + @Override + public Class getRecipeClass() { + return ShapedRecipe.class; + } + private static final class CrafterIngredientRenderer implements IIngredientRenderer { private final ShapedRecipe recipe; @@ -66,11 +166,10 @@ public class MechanicalCraftingCategory extends CreateRecipeCategory list; try { - list = ingredient - .getTooltip(player, tooltipFlag) - .stream() - .map(ITextComponent::getFormattedText) - .collect(Collectors.toList()); + list = ingredient.getTooltip(player, tooltipFlag) + .stream() + .map(ITextComponent::getFormattedText) + .collect(Collectors.toList()); } catch (RuntimeException | LinkageError e) { return new ArrayList<>(); } @@ -94,103 +193,4 @@ public class MechanicalCraftingCategory extends CreateRecipeCategory recipeIngredients = recipe.getIngredients(); - - itemStacks.init(0, false, 133, 80); - itemStacks.set(0, recipe.getRecipeOutput().getStack()); - - int x = getXPadding(recipe); - int y = getYPadding(recipe); - float scale = getScale(recipe); - int size = recipeIngredients.size(); - IIngredientRenderer renderer = new CrafterIngredientRenderer(recipe); - - for (int i = 0; i < size; i++) { - float f = 19 * scale; - int slotSize = (int) (16 * scale); - int xPosition = (int) (x + 1 + (i % recipe.getWidth()) * f); - int yPosition = (int) (y + 1 + (i / recipe.getWidth()) * f); - itemStacks.init(i + 1, true, renderer, xPosition, yPosition, slotSize, slotSize, 0, 0); - itemStacks.set(i + 1, Arrays.asList(recipeIngredients.get(i).getMatchingStacks())); - } - - } - - static int maxSize = 100; - - public static float getScale(ShapedRecipe recipe) { - int w = recipe.getWidth(); - int h = recipe.getHeight(); - return Math.min(1, maxSize / (19f * Math.max(w, h))); - } - - public static int getYPadding(ShapedRecipe recipe) { - return 3 + 50 - (int) (getScale(recipe) * recipe.getHeight() * 19 * .5); - } - - public static int getXPadding(ShapedRecipe recipe) { - return 3 + 50 - (int) (getScale(recipe) * recipe.getWidth() * 19 * .5); - } - - @Override - public void draw(ShapedRecipe recipe, double mouseX, double mouseY) { - RenderSystem.pushMatrix(); - float scale = getScale(recipe); - RenderSystem.translated(getXPadding(recipe), getYPadding(recipe), 0); - - for (int row = 0; row < recipe.getHeight(); row++) - for (int col = 0; col < recipe.getWidth(); col++) - if (!recipe.getIngredients().get(row * recipe.getWidth() + col).hasNoMatchingItems()) { - RenderSystem.pushMatrix(); - RenderSystem.translated(col * 19 * scale, row * 19 * scale, 0); - RenderSystem.scaled(scale, scale, scale); - AllGuiTextures.JEI_SLOT.draw(0, 0); - RenderSystem.popMatrix(); - } - - RenderSystem.popMatrix(); - - AllGuiTextures.JEI_SLOT.draw(133, 80); - AllGuiTextures.JEI_DOWN_ARROW.draw(128, 59); - crafter.draw(129, 25); - - RenderSystem.pushMatrix(); - RenderSystem.translated(0, 0, 300); - - RenderHelper.disableStandardItemLighting(); - int amount = 0; - for (Ingredient ingredient : recipe.getIngredients()) { - if (Ingredient.EMPTY == ingredient) - continue; - amount++; - } - - Minecraft.getInstance().fontRenderer - .drawStringWithShadow(amount + "", 142, 39, 0xFFFFFF); - RenderSystem.popMatrix(); - } - - @Override - public Class getRecipeClass() { - return ShapedRecipe.class; - } - } diff --git a/src/main/java/com/simibubi/create/compat/jei/category/MillingCategory.java b/src/main/java/com/simibubi/create/compat/jei/category/MillingCategory.java index 78be3a3ca..a3dc27e04 100644 --- a/src/main/java/com/simibubi/create/compat/jei/category/MillingCategory.java +++ b/src/main/java/com/simibubi/create/compat/jei/category/MillingCategory.java @@ -20,8 +20,7 @@ public class MillingCategory extends CreateRecipeCategory getRecipes() { List recipes = new ArrayList<>(); + recipes.add(ConversionRecipe.create(AllItems.EMPTY_BLAZE_BURNER.asStack(), AllBlocks.BLAZE_BURNER.asStack())); recipes.add(ConversionRecipe.create(AllItems.CHROMATIC_COMPOUND.asStack(), AllItems.SHADOW_STEEL.asStack())); recipes.add(ConversionRecipe.create(AllItems.CHROMATIC_COMPOUND.asStack(), AllItems.REFINED_RADIANCE.asStack())); return recipes; } public MysteriousItemConversionCategory() { - super("mystery_conversion", itemIcon(AllItems.CHROMATIC_COMPOUND.get()), emptyBackground(177, 50)); + super(itemIcon(AllItems.CHROMATIC_COMPOUND.get()), emptyBackground(177, 50)); } @Override diff --git a/src/main/java/com/simibubi/create/compat/jei/category/PackingCategory.java b/src/main/java/com/simibubi/create/compat/jei/category/PackingCategory.java index e3b4b24a9..e4c654888 100644 --- a/src/main/java/com/simibubi/create/compat/jei/category/PackingCategory.java +++ b/src/main/java/com/simibubi/create/compat/jei/category/PackingCategory.java @@ -10,25 +10,41 @@ import com.simibubi.create.foundation.gui.AllGuiTextures; import mezz.jei.api.gui.IRecipeLayout; import mezz.jei.api.gui.ingredient.IGuiItemStackGroup; import mezz.jei.api.ingredients.IIngredients; +import net.minecraft.block.Blocks; import net.minecraft.item.crafting.Ingredient; +import net.minecraft.util.IItemProvider; import net.minecraft.util.NonNullList; public class PackingCategory extends BasinCategory { private AnimatedPress press = new AnimatedPress(true); + private PackingType type; - public PackingCategory() { - super("packing", doubleItemIcon(AllBlocks.MECHANICAL_PRESS.get(), AllBlocks.BASIN.get()), - emptyBackground(177, 103)); + enum PackingType { + AUTO_SQUARE, COMPACTING; + } + + public static PackingCategory standard() { + return new PackingCategory(PackingType.COMPACTING, AllBlocks.BASIN.get(), 103); + } + + public static PackingCategory autoSquare() { + return new PackingCategory(PackingType.AUTO_SQUARE, Blocks.CRAFTING_TABLE, 85); + } + + protected PackingCategory(PackingType type, IItemProvider icon, int height) { + super(type != PackingType.AUTO_SQUARE, doubleItemIcon(AllBlocks.MECHANICAL_PRESS.get(), icon), + emptyBackground(177, height)); + this.type = type; } @Override public void setRecipe(IRecipeLayout recipeLayout, BasinRecipe recipe, IIngredients ingredients) { - if (!recipe.convertedRecipe) { + if (type == PackingType.COMPACTING) { super.setRecipe(recipeLayout, recipe, ingredients); return; } - + IGuiItemStackGroup itemStacks = recipeLayout.getItemStacks(); int i = 0; @@ -48,21 +64,20 @@ public class PackingCategory extends BasinCategory { @Override public void draw(BasinRecipe recipe, double mouseX, double mouseY) { - if (!recipe.convertedRecipe) { + if (type == PackingType.COMPACTING) { super.draw(recipe, mouseX, mouseY); - + } else { NonNullList ingredients2 = recipe.getIngredients(); int size = ingredients2.size(); int rows = size == 4 ? 2 : 3; - for (int i = 0; i < size; i++) + for (int i = 0; i < size; i++) AllGuiTextures.JEI_SLOT.draw((rows == 2 ? 26 : 17) + (i % rows) * 19, 50 - (i / rows) * 19); AllGuiTextures.JEI_SLOT.draw(141, 50); AllGuiTextures.JEI_DOWN_ARROW.draw(136, 32); AllGuiTextures.JEI_SHADOW.draw(81, 68); - AllGuiTextures.JEI_NO_HEAT_BAR.draw(4, 80); } - + press.draw(getBackground().getWidth() / 2 + 6, 40); } diff --git a/src/main/java/com/simibubi/create/compat/jei/category/PolishingCategory.java b/src/main/java/com/simibubi/create/compat/jei/category/PolishingCategory.java index 9f78bd908..5ca1af5de 100644 --- a/src/main/java/com/simibubi/create/compat/jei/category/PolishingCategory.java +++ b/src/main/java/com/simibubi/create/compat/jei/category/PolishingCategory.java @@ -25,7 +25,7 @@ public class PolishingCategory extends CreateRecipeCategory results = recipe.getRollableResults(); itemStacks.init(0, true, 26, 28); - itemStacks.set(0, Arrays.asList(recipe.getIngredients().get(0).getMatchingStacks())); + itemStacks.set(0, Arrays.asList(recipe.getIngredients() + .get(0) + .getMatchingStacks())); itemStacks.init(1, false, 131, 28); - itemStacks.set(1, results.get(0).getStack()); + itemStacks.set(1, results.get(0) + .getStack()); addStochasticTooltip(itemStacks, results); } @@ -61,7 +64,8 @@ public class PolishingCategory extends CreateRecipeCategory ingredients = recipe.getIngredients(); - ItemStack[] matchingStacks = ingredients.get(0).getMatchingStacks(); + ItemStack[] matchingStacks = ingredients.get(0) + .getMatchingStacks(); if (matchingStacks.length == 0) return; @@ -69,7 +73,8 @@ public class PolishingCategory extends CreateRecipeCategory { private AnimatedPress press = new AnimatedPress(false); public PressingCategory() { - super("pressing", doubleItemIcon(AllBlocks.MECHANICAL_PRESS.get(), AllItems.IRON_SHEET.get()), - emptyBackground(177, 70)); + super(doubleItemIcon(AllBlocks.MECHANICAL_PRESS.get(), AllItems.IRON_SHEET.get()), emptyBackground(177, 70)); } @Override @@ -39,12 +38,15 @@ public class PressingCategory extends CreateRecipeCategory { public void setRecipe(IRecipeLayout recipeLayout, PressingRecipe recipe, IIngredients ingredients) { IGuiItemStackGroup itemStacks = recipeLayout.getItemStacks(); itemStacks.init(0, true, 26, 50); - itemStacks.set(0, Arrays.asList(recipe.getIngredients().get(0).getMatchingStacks())); + itemStacks.set(0, Arrays.asList(recipe.getIngredients() + .get(0) + .getMatchingStacks())); List results = recipe.getRollableResults(); for (int outputIndex = 0; outputIndex < results.size(); outputIndex++) { itemStacks.init(outputIndex + 1, false, 131 + 19 * outputIndex, 50); - itemStacks.set(outputIndex + 1, results.get(outputIndex).getStack()); + itemStacks.set(outputIndex + 1, results.get(outputIndex) + .getStack()); } addStochasticTooltip(itemStacks, results); @@ -54,7 +56,8 @@ public class PressingCategory extends CreateRecipeCategory { public void draw(PressingRecipe recipe, double mouseX, double mouseY) { AllGuiTextures.JEI_SLOT.draw(26, 50); getRenderedSlot(recipe, 0).draw(131, 50); - if (recipe.getRollableResults().size() > 1) + if (recipe.getRollableResults() + .size() > 1) getRenderedSlot(recipe, 1).draw(131 + 19, 50); AllGuiTextures.JEI_SHADOW.draw(61, 41); AllGuiTextures.JEI_LONG_ARROW.draw(52, 54); diff --git a/src/main/java/com/simibubi/create/compat/jei/category/ProcessingViaFanCategory.java b/src/main/java/com/simibubi/create/compat/jei/category/ProcessingViaFanCategory.java index 4778bf058..ea6afdac2 100644 --- a/src/main/java/com/simibubi/create/compat/jei/category/ProcessingViaFanCategory.java +++ b/src/main/java/com/simibubi/create/compat/jei/category/ProcessingViaFanCategory.java @@ -1,6 +1,7 @@ package com.simibubi.create.compat.jei.category; import java.util.Arrays; +import java.util.function.Supplier; import com.mojang.blaze3d.systems.RenderSystem; import com.simibubi.create.AllBlockPartials; @@ -8,18 +9,22 @@ import com.simibubi.create.AllBlocks; import com.simibubi.create.compat.jei.category.animations.AnimatedKinetics; import com.simibubi.create.foundation.gui.AllGuiTextures; import com.simibubi.create.foundation.gui.GuiGameElement; +import com.simibubi.create.foundation.utility.Lang; import mezz.jei.api.constants.VanillaTypes; import mezz.jei.api.gui.IRecipeLayout; import mezz.jei.api.gui.drawable.IDrawable; import mezz.jei.api.gui.ingredient.IGuiItemStackGroup; import mezz.jei.api.ingredients.IIngredients; +import net.minecraft.item.ItemStack; import net.minecraft.item.crafting.IRecipe; +import net.minecraft.util.text.StringTextComponent; +import net.minecraft.util.text.TextFormatting; public abstract class ProcessingViaFanCategory> extends CreateRecipeCategory { - public ProcessingViaFanCategory(String name, IDrawable icon) { - super(name, icon, emptyBackground(177, 70)); + public ProcessingViaFanCategory(IDrawable icon) { + super(icon, emptyBackground(177, 70)); } @Override @@ -28,6 +33,11 @@ public abstract class ProcessingViaFanCategory> extends Cre ingredients.setOutput(VanillaTypes.ITEM, recipe.getRecipeOutput()); } + public static Supplier getFan(String name) { + return () -> AllBlocks.ENCASED_FAN.asStack() + .setDisplayName(new StringTextComponent(TextFormatting.RESET + Lang.translate("recipe." + name + ".fan"))); + } + @Override public void setRecipe(IRecipeLayout recipeLayout, T recipe, IIngredients ingredients) { IGuiItemStackGroup itemStacks = recipeLayout.getItemStacks(); diff --git a/src/main/java/com/simibubi/create/compat/jei/category/SawingCategory.java b/src/main/java/com/simibubi/create/compat/jei/category/SawingCategory.java index e80686a35..86b8785f6 100644 --- a/src/main/java/com/simibubi/create/compat/jei/category/SawingCategory.java +++ b/src/main/java/com/simibubi/create/compat/jei/category/SawingCategory.java @@ -20,7 +20,7 @@ public class SawingCategory extends CreateRecipeCategory { private AnimatedSaw saw = new AnimatedSaw(); public SawingCategory() { - super("sawing", doubleItemIcon(AllBlocks.MECHANICAL_SAW.get(), Items.OAK_LOG), emptyBackground(177, 70)); + super(doubleItemIcon(AllBlocks.MECHANICAL_SAW.get(), Items.OAK_LOG), emptyBackground(177, 70)); } @Override diff --git a/src/main/java/com/simibubi/create/content/contraptions/components/mixer/MechanicalMixerTileEntity.java b/src/main/java/com/simibubi/create/content/contraptions/components/mixer/MechanicalMixerTileEntity.java index 17bbf9fce..7177d4fac 100644 --- a/src/main/java/com/simibubi/create/content/contraptions/components/mixer/MechanicalMixerTileEntity.java +++ b/src/main/java/com/simibubi/create/content/contraptions/components/mixer/MechanicalMixerTileEntity.java @@ -1,9 +1,11 @@ package com.simibubi.create.content.contraptions.components.mixer; +import java.util.List; import java.util.Optional; import com.simibubi.create.AllRecipeTypes; import com.simibubi.create.content.contraptions.components.press.MechanicalPressTileEntity; +import com.simibubi.create.content.contraptions.fluids.potion.PotionMixingRecipeManager; import com.simibubi.create.content.contraptions.processing.BasinOperatingTileEntity; import com.simibubi.create.content.contraptions.processing.BasinTileEntity; import com.simibubi.create.foundation.advancement.AllTriggers; @@ -24,6 +26,8 @@ import net.minecraft.util.Direction.Axis; import net.minecraft.util.math.AxisAlignedBB; import net.minecraft.util.math.MathHelper; import net.minecraft.util.math.Vec3d; +import net.minecraftforge.items.CapabilityItemHandler; +import net.minecraftforge.items.IItemHandler; public class MechanicalMixerTileEntity extends BasinOperatingTileEntity { @@ -151,6 +155,35 @@ public class MechanicalMixerTileEntity extends BasinOperatingTileEntity { } } + @Override + protected List> getMatchingRecipes() { + List> matchingRecipes = super.getMatchingRecipes(); + + Optional basin = getBasin(); + if (!basin.isPresent()) + return matchingRecipes; + IItemHandler availableItems = basin.get() + .getCapability(CapabilityItemHandler.ITEM_HANDLER_CAPABILITY) + .orElse(null); + if (availableItems == null) + return matchingRecipes; + + for (int i = 0; i < availableItems.getSlots(); i++) { + ItemStack stack = availableItems.getStackInSlot(i); + if (stack.isEmpty()) + continue; + + List list = PotionMixingRecipeManager.ALL.get(stack.getItem()); + if (list == null) + continue; + for (MixingRecipe mixingRecipe : list) + if (matchBasinRecipe(mixingRecipe)) + matchingRecipes.add(mixingRecipe); + } + + return matchingRecipes; + } + @Override protected boolean matchStaticFilters(IRecipe r) { return ((r.getSerializer() == IRecipeSerializer.CRAFTING_SHAPELESS @@ -190,7 +223,7 @@ public class MechanicalMixerTileEntity extends BasinOperatingTileEntity { protected boolean isRunning() { return running; } - + @Override protected Optional getProcessedRecipeTrigger() { return Optional.of(AllTriggers.MIXER_MIX); diff --git a/src/main/java/com/simibubi/create/content/contraptions/fluids/CombinedFluidHandler.java b/src/main/java/com/simibubi/create/content/contraptions/fluids/CombinedFluidHandler.java deleted file mode 100644 index 9b03c90d2..000000000 --- a/src/main/java/com/simibubi/create/content/contraptions/fluids/CombinedFluidHandler.java +++ /dev/null @@ -1,138 +0,0 @@ -package com.simibubi.create.content.contraptions.fluids; - -import java.util.Arrays; -import java.util.stream.Collectors; -import java.util.stream.IntStream; - -import javax.annotation.Nonnull; - -import net.minecraft.nbt.CompoundNBT; -import net.minecraft.nbt.ListNBT; -import net.minecraft.util.math.MathHelper; -import net.minecraftforge.common.util.NonNullConsumer; -import net.minecraftforge.fluids.FluidStack; -import net.minecraftforge.fluids.capability.IFluidHandler; - -public class CombinedFluidHandler implements IFluidHandler { - private final int capacity; - private final FluidStack[] tanks; - - public CombinedFluidHandler(int tankNumber, int capacity) { - this.capacity = capacity; - this.tanks = new FluidStack[tankNumber]; - Arrays.fill(tanks, FluidStack.EMPTY); - } - - @Override - public int getTanks() { - return tanks.length; - } - - @Nonnull - @Override - public FluidStack getFluidInTank(int tank) { - if (tank < 0 || tank >= tanks.length) - return FluidStack.EMPTY; - return tanks[tank]; - } - - @Override - public int getTankCapacity(int tank) { - return capacity; - } - - @Override - public boolean isFluidValid(int tank, @Nonnull FluidStack stack) { - return (!stack.isEmpty()) && (tanks[tank].isEmpty() || tanks[tank].isFluidEqual(stack)) - && tanks[tank].getAmount() < capacity; - } - - @Override - public int fill(FluidStack resource, FluidAction action) { - int tankIndex; - int amount = resource.getAmount(); - while ((tankIndex = getFittingFluidSlot(resource)) != -1) { - int newAmount = MathHelper.clamp(amount - capacity - tanks[tankIndex].getAmount(), 0, Integer.MAX_VALUE); - if (action == FluidAction.EXECUTE) - if (tanks[tankIndex].isEmpty()) - tanks[tankIndex] = new FluidStack(resource.getFluid(), amount - newAmount); - else - tanks[tankIndex].grow(amount - newAmount); - amount = newAmount; - if (amount == 0) - return 0; - } - return amount; - } - - @Nonnull - @Override - public FluidStack drain(FluidStack resource, FluidAction action) { - if (resource.isEmpty()) - return FluidStack.EMPTY; - - FluidStack stack = new FluidStack(resource, 0); - - for (int i = 0; i < tanks.length; i++) { - if (tanks[i].isFluidEqual(resource)) { - stack.grow(tanks[i].getAmount()); - if (action == FluidAction.EXECUTE) - tanks[i] = FluidStack.EMPTY; - } - } - - return stack.isEmpty() ? FluidStack.EMPTY : stack; - } - - @Nonnull - @Override - public FluidStack drain(int maxDrain, FluidAction action) { - FluidStack stack = new FluidStack(tanks[0].getFluid(), 0); - - for (int i = 0; i < tanks.length; i++) { - if (stack.isEmpty() || tanks[i].isFluidEqual(stack)) { - int newDrainAmount = MathHelper.clamp(stack.getAmount() + tanks[i].getAmount(), 0, maxDrain); - if (action == FluidAction.EXECUTE) { - tanks[i].shrink(newDrainAmount - stack.getAmount()); - if (tanks[i].isEmpty()) - tanks[i] = FluidStack.EMPTY; - } - if (stack.isEmpty()) - stack = tanks[i].copy(); - if (stack.isEmpty()) - continue; - stack.setAmount(newDrainAmount); - } - } - - return stack.isEmpty() ? FluidStack.EMPTY : stack; - } - - private int getFittingFluidSlot(FluidStack fluidStack) { - return IntStream.range(0, tanks.length) - .filter(i -> isFluidValid(i, fluidStack)) - .findFirst() - .orElse(-1); - } - - private void setFluid(FluidStack fluid, int tank) { - tanks[tank] = fluid; - } - - public CombinedFluidHandler readFromNBT(ListNBT fluidNBTs) { - for (int i = 0; i < Math.min(tanks.length, fluidNBTs.size()); i++) - setFluid(FluidStack.loadFluidStackFromNBT(fluidNBTs.getCompound(i)), i); - return this; - } - - public ListNBT getListNBT() { - return Arrays.stream(tanks) - .map(fluid -> fluid.writeToNBT(new CompoundNBT())) - .collect(Collectors.toCollection(ListNBT::new)); - } - - public void forEachTank(NonNullConsumer fluidStackConsumer) { - Arrays.stream(tanks) - .forEach(fluidStackConsumer::accept); - } -} diff --git a/src/main/java/com/simibubi/create/content/contraptions/fluids/FluidNetworkFlow.java b/src/main/java/com/simibubi/create/content/contraptions/fluids/FluidNetworkFlow.java index 7e630a56f..1324ee41c 100644 --- a/src/main/java/com/simibubi/create/content/contraptions/fluids/FluidNetworkFlow.java +++ b/src/main/java/com/simibubi/create/content/contraptions/fluids/FluidNetworkFlow.java @@ -114,7 +114,7 @@ class FluidNetworkFlow { return; } - fluidStack = provideFluid; + fluidStack = provideFluid.copy(); // There is currently no unfinished flow being followed if (flowPointers.isEmpty()) { diff --git a/src/main/java/com/simibubi/create/content/contraptions/fluids/actors/FillingBySpout.java b/src/main/java/com/simibubi/create/content/contraptions/fluids/actors/FillingBySpout.java index 4afe6f0d6..799fef49f 100644 --- a/src/main/java/com/simibubi/create/content/contraptions/fluids/actors/FillingBySpout.java +++ b/src/main/java/com/simibubi/create/content/contraptions/fluids/actors/FillingBySpout.java @@ -9,12 +9,7 @@ import com.simibubi.create.foundation.fluid.FluidIngredient; import net.minecraft.item.ItemStack; import net.minecraft.item.crafting.IRecipe; import net.minecraft.world.World; -import net.minecraftforge.common.util.LazyOptional; import net.minecraftforge.fluids.FluidStack; -import net.minecraftforge.fluids.capability.CapabilityFluidHandler; -import net.minecraftforge.fluids.capability.IFluidHandler.FluidAction; -import net.minecraftforge.fluids.capability.IFluidHandlerItem; -import net.minecraftforge.fluids.capability.wrappers.FluidBucketWrapper; import net.minecraftforge.items.ItemStackHandler; import net.minecraftforge.items.wrapper.RecipeWrapper; @@ -27,18 +22,7 @@ public class FillingBySpout { if (AllRecipeTypes.FILLING.find(wrapper, world) .isPresent()) return true; - - LazyOptional capability = - stack.getCapability(CapabilityFluidHandler.FLUID_HANDLER_ITEM_CAPABILITY); - IFluidHandlerItem tank = capability.orElse(null); - if (tank == null) - return false; - for (int i = 0; i < tank.getTanks(); i++) { - if (tank.getFluidInTank(i) - .getAmount() < tank.getTankCapacity(i)) - return true; - } - return false; + return GenericItemFilling.canItemBeFilled(world, stack); } public static int getRequiredAmountForItem(World world, ItemStack stack, FluidStack availableFluid) { @@ -50,23 +34,12 @@ public class FillingBySpout { if (requiredFluid.test(availableFluid)) return requiredFluid.getRequiredAmount(); } - - LazyOptional capability = - stack.getCapability(CapabilityFluidHandler.FLUID_HANDLER_ITEM_CAPABILITY); - IFluidHandlerItem tank = capability.orElse(null); - if (tank == null) - return -1; - if (tank instanceof FluidBucketWrapper) - return 1000; - - int filled = tank.fill(availableFluid, FluidAction.SIMULATE); - return filled == 0 ? -1 : filled; + return GenericItemFilling.getRequiredAmountForItem(world, stack, availableFluid); } public static ItemStack fillItem(World world, int requiredAmount, ItemStack stack, FluidStack availableFluid) { FluidStack toFill = availableFluid.copy(); toFill.setAmount(requiredAmount); - availableFluid.shrink(requiredAmount); wrapper.setInventorySlotContents(0, stack); Optional> recipe = AllRecipeTypes.FILLING.find(wrapper, world); @@ -75,23 +48,12 @@ public class FillingBySpout { FluidIngredient requiredFluid = fillingRecipe.getRequiredFluid(); if (requiredFluid.test(toFill)) { List results = fillingRecipe.rollResults(); + availableFluid.shrink(requiredAmount); stack.shrink(1); return results.isEmpty() ? ItemStack.EMPTY : results.get(0); } } - - ItemStack split = stack.copy(); - split.setCount(1); - LazyOptional capability = - split.getCapability(CapabilityFluidHandler.FLUID_HANDLER_ITEM_CAPABILITY); - IFluidHandlerItem tank = capability.orElse(null); - if (tank == null) - return ItemStack.EMPTY; - tank.fill(toFill, FluidAction.EXECUTE); - ItemStack container = tank.getContainer() - .copy(); - stack.shrink(1); - return container; + return GenericItemFilling.fillItem(world, requiredAmount, stack, availableFluid); } } diff --git a/src/main/java/com/simibubi/create/content/contraptions/fluids/actors/GenericItemFilling.java b/src/main/java/com/simibubi/create/content/contraptions/fluids/actors/GenericItemFilling.java new file mode 100644 index 000000000..e6b030e59 --- /dev/null +++ b/src/main/java/com/simibubi/create/content/contraptions/fluids/actors/GenericItemFilling.java @@ -0,0 +1,82 @@ +package com.simibubi.create.content.contraptions.fluids.actors; + +import com.simibubi.create.content.contraptions.fluids.potion.PotionFluidHandler; +import com.simibubi.create.foundation.fluid.FluidHelper; + +import net.minecraft.item.ItemStack; +import net.minecraft.item.Items; +import net.minecraft.potion.PotionUtils; +import net.minecraft.potion.Potions; +import net.minecraft.world.World; +import net.minecraftforge.common.util.LazyOptional; +import net.minecraftforge.fluids.FluidStack; +import net.minecraftforge.fluids.capability.CapabilityFluidHandler; +import net.minecraftforge.fluids.capability.IFluidHandler.FluidAction; +import net.minecraftforge.fluids.capability.IFluidHandlerItem; +import net.minecraftforge.fluids.capability.wrappers.FluidBucketWrapper; + +public class GenericItemFilling { + + public static boolean canItemBeFilled(World world, ItemStack stack) { + if (stack.getItem() == Items.GLASS_BOTTLE) + return true; + + LazyOptional capability = + stack.getCapability(CapabilityFluidHandler.FLUID_HANDLER_ITEM_CAPABILITY); + IFluidHandlerItem tank = capability.orElse(null); + if (tank == null) + return false; + for (int i = 0; i < tank.getTanks(); i++) { + if (tank.getFluidInTank(i) + .getAmount() < tank.getTankCapacity(i)) + return true; + } + return false; + } + + public static int getRequiredAmountForItem(World world, ItemStack stack, FluidStack availableFluid) { + if (stack.getItem() == Items.GLASS_BOTTLE) + return PotionFluidHandler.getRequiredAmountForFilledBottle(stack, availableFluid); + + LazyOptional capability = + stack.getCapability(CapabilityFluidHandler.FLUID_HANDLER_ITEM_CAPABILITY); + IFluidHandlerItem tank = capability.orElse(null); + if (tank == null) + return -1; + if (tank instanceof FluidBucketWrapper) + return 1000; + + int filled = tank.fill(availableFluid, FluidAction.SIMULATE); + return filled == 0 ? -1 : filled; + } + + public static ItemStack fillItem(World world, int requiredAmount, ItemStack stack, FluidStack availableFluid) { + FluidStack toFill = availableFluid.copy(); + toFill.setAmount(requiredAmount); + availableFluid.shrink(requiredAmount); + + if (stack.getItem() == Items.GLASS_BOTTLE) { + ItemStack fillBottle = ItemStack.EMPTY; + if (FluidHelper.isWater(toFill.getFluid())) + fillBottle = PotionUtils.addPotionToItemStack(new ItemStack(Items.POTION), Potions.WATER); + else + fillBottle = PotionFluidHandler.fillBottle(stack, toFill); + stack.shrink(1); + return fillBottle; + } + + ItemStack split = stack.copy(); + split.setCount(1); + LazyOptional capability = + split.getCapability(CapabilityFluidHandler.FLUID_HANDLER_ITEM_CAPABILITY); + IFluidHandlerItem tank = capability.orElse(null); + if (tank == null) + return ItemStack.EMPTY; + tank.fill(toFill, FluidAction.EXECUTE); + ItemStack container = tank.getContainer() + .copy(); + stack.shrink(1); + return container; + } + +} diff --git a/src/main/java/com/simibubi/create/content/contraptions/fluids/potion/PotionFluid.java b/src/main/java/com/simibubi/create/content/contraptions/fluids/potion/PotionFluid.java new file mode 100644 index 000000000..d1d54ff75 --- /dev/null +++ b/src/main/java/com/simibubi/create/content/contraptions/fluids/potion/PotionFluid.java @@ -0,0 +1,111 @@ +package com.simibubi.create.content.contraptions.fluids.potion; + +import java.util.Collection; +import java.util.List; + +import com.simibubi.create.AllFluids; + +import net.minecraft.block.BlockState; +import net.minecraft.block.Blocks; +import net.minecraft.fluid.Fluid; +import net.minecraft.fluid.IFluidState; +import net.minecraft.item.Item; +import net.minecraft.item.Items; +import net.minecraft.nbt.CompoundNBT; +import net.minecraft.nbt.ListNBT; +import net.minecraft.potion.EffectInstance; +import net.minecraft.potion.Potion; +import net.minecraft.potion.PotionUtils; +import net.minecraft.potion.Potions; +import net.minecraft.util.ResourceLocation; +import net.minecraftforge.fluids.FluidAttributes; +import net.minecraftforge.fluids.FluidStack; +import net.minecraftforge.fluids.ForgeFlowingFluid; +import net.minecraftforge.registries.ForgeRegistries; + +public class PotionFluid extends ForgeFlowingFluid { + + public enum BottleType { + REGULAR, SPLASH, LINGERING; + } + + public PotionFluid(Properties properties) { + super(properties); + } + + public static FluidStack withEffects(int amount, Potion potion, List customEffects) { + FluidStack fluidStack = new FluidStack(AllFluids.POTION.get() + .getStillFluid(), amount); + addPotionToFluidStack(fluidStack, potion); + appendEffects(fluidStack, customEffects); + return fluidStack; + } + + @Override + public Fluid getStillFluid() { + return this; + } + + @Override + public Fluid getFlowingFluid() { + return this; + } + + @Override + public Item getFilledBucket() { + return Items.AIR; + } + + @Override + protected BlockState getBlockState(IFluidState state) { + return Blocks.AIR.getDefaultState(); + } + + @Override + public boolean isSource(IFluidState p_207193_1_) { + return false; + } + + @Override + public int getLevel(IFluidState p_207192_1_) { + return 0; + } + + public static class PotionFluidAttributes extends FluidAttributes { + + public PotionFluidAttributes(Builder builder, Fluid fluid) { + super(builder, fluid); + } + + @Override + public int getColor(FluidStack stack) { + CompoundNBT tag = stack.getOrCreateTag(); + int color = PotionUtils.getPotionColorFromEffectList(PotionUtils.getEffectsFromTag(tag)) | 0xff000000; + return color; + } + + } + + public static FluidStack addPotionToFluidStack(FluidStack fs, Potion potion) { + ResourceLocation resourcelocation = ForgeRegistries.POTION_TYPES.getKey(potion); + if (potion == Potions.EMPTY) { + fs.removeChildTag("Potion"); + return fs; + } + fs.getOrCreateTag() + .putString("Potion", resourcelocation.toString()); + return fs; + } + + public static FluidStack appendEffects(FluidStack fs, Collection customEffects) { + if (customEffects.isEmpty()) + return fs; + CompoundNBT compoundnbt = fs.getOrCreateTag(); + ListNBT listnbt = compoundnbt.getList("CustomPotionEffects", 9); + for (EffectInstance effectinstance : customEffects) + listnbt.add(effectinstance.write(new CompoundNBT())); + compoundnbt.put("CustomPotionEffects", listnbt); + return fs; + } + +} diff --git a/src/main/java/com/simibubi/create/content/contraptions/fluids/potion/PotionFluidHandler.java b/src/main/java/com/simibubi/create/content/contraptions/fluids/potion/PotionFluidHandler.java new file mode 100644 index 000000000..aea7da59d --- /dev/null +++ b/src/main/java/com/simibubi/create/content/contraptions/fluids/potion/PotionFluidHandler.java @@ -0,0 +1,171 @@ +package com.simibubi.create.content.contraptions.fluids.potion; + +import java.util.List; +import java.util.Map; +import java.util.Map.Entry; + +import com.google.common.collect.Lists; +import com.simibubi.create.content.contraptions.fluids.potion.PotionFluid.BottleType; +import com.simibubi.create.foundation.utility.NBTHelper; +import com.simibubi.create.foundation.utility.Pair; + +import net.minecraft.entity.ai.attributes.AttributeModifier; +import net.minecraft.entity.ai.attributes.IAttribute; +import net.minecraft.fluid.Fluids; +import net.minecraft.item.Item; +import net.minecraft.item.ItemStack; +import net.minecraft.item.Items; +import net.minecraft.nbt.CompoundNBT; +import net.minecraft.potion.Effect; +import net.minecraft.potion.EffectInstance; +import net.minecraft.potion.EffectUtils; +import net.minecraft.potion.Potion; +import net.minecraft.potion.PotionUtils; +import net.minecraft.potion.Potions; +import net.minecraft.util.IItemProvider; +import net.minecraft.util.Tuple; +import net.minecraft.util.text.ITextComponent; +import net.minecraft.util.text.StringTextComponent; +import net.minecraft.util.text.TextFormatting; +import net.minecraft.util.text.TranslationTextComponent; +import net.minecraftforge.api.distmarker.Dist; +import net.minecraftforge.api.distmarker.OnlyIn; +import net.minecraftforge.fluids.FluidStack; + +public class PotionFluidHandler { + + public static Pair emptyPotion(ItemStack stack, boolean simulate) { + FluidStack fluid = getFluidFromPotionItem(stack); + if (!simulate) + stack.shrink(1); + return Pair.of(fluid, new ItemStack(Items.GLASS_BOTTLE)); + } + + public static FluidStack getFluidFromPotionItem(ItemStack stack) { + Potion potion = PotionUtils.getPotionFromItem(stack); + List list = PotionUtils.getFullEffectsFromItem(stack); + FluidStack fluid = PotionFluid.withEffects(250, potion, list); + BottleType bottleTypeFromItem = bottleTypeFromItem(stack); + if (potion == Potions.WATER && list.isEmpty() && bottleTypeFromItem == BottleType.REGULAR) + return new FluidStack(Fluids.WATER, fluid.getAmount()); + NBTHelper.writeEnum(fluid.getOrCreateTag(), "Bottle", bottleTypeFromItem); + return fluid; + } + + public static BottleType bottleTypeFromItem(ItemStack stack) { + Item item = stack.getItem(); + if (item == Items.LINGERING_POTION) + return BottleType.LINGERING; + if (item == Items.SPLASH_POTION) + return BottleType.SPLASH; + return BottleType.REGULAR; + } + + public static IItemProvider itemFromBottleType(BottleType type) { + switch (type) { + case LINGERING: + return Items.LINGERING_POTION; + case SPLASH: + return Items.SPLASH_POTION; + case REGULAR: + default: + return Items.POTION; + } + } + + public static int getRequiredAmountForFilledBottle(ItemStack stack, FluidStack availableFluid) { + return 250; + } + + public static ItemStack fillBottle(ItemStack stack, FluidStack availableFluid) { + CompoundNBT tag = availableFluid.getOrCreateTag(); + ItemStack potionStack = new ItemStack(itemFromBottleType(NBTHelper.readEnum(tag, "Bottle", BottleType.class))); + PotionUtils.addPotionToItemStack(potionStack, PotionUtils.getPotionTypeFromNBT(tag)); + PotionUtils.appendEffects(potionStack, PotionUtils.getFullEffectsFromTag(tag)); + return potionStack; + } + + public static ITextComponent getPotionName(FluidStack fs) { + CompoundNBT tag = fs.getOrCreateTag(); + IItemProvider itemFromBottleType = itemFromBottleType(NBTHelper.readEnum(tag, "Bottle", BottleType.class)); + return new TranslationTextComponent(PotionUtils.getPotionTypeFromNBT(tag) + .getNamePrefixed(itemFromBottleType.asItem() + .getTranslationKey() + ".effect.")); + } + + // Modified version of PotionUtils#addPotionTooltip + @OnlyIn(Dist.CLIENT) + public static void addPotionTooltip(FluidStack fs, List tooltip, float p_185182_2_) { + List list = PotionUtils.getEffectsFromTag(fs.getOrCreateTag()); + List> list1 = Lists.newArrayList(); + if (list.isEmpty()) { + tooltip.add((new TranslationTextComponent("effect.none")).applyTextStyle(TextFormatting.GRAY)); + } else { + for (EffectInstance effectinstance : list) { + ITextComponent itextcomponent = new TranslationTextComponent(effectinstance.getEffectName()); + Effect effect = effectinstance.getPotion(); + Map map = effect.getAttributeModifierMap(); + if (!map.isEmpty()) { + for (Entry entry : map.entrySet()) { + AttributeModifier attributemodifier = entry.getValue(); + AttributeModifier attributemodifier1 = new AttributeModifier(attributemodifier.getName(), + effect.getAttributeModifierAmount(effectinstance.getAmplifier(), attributemodifier), + attributemodifier.getOperation()); + list1.add(new Tuple<>(entry.getKey() + .getName(), attributemodifier1)); + } + } + + if (effectinstance.getAmplifier() > 0) { + itextcomponent.appendText(" ") + .appendSibling(new TranslationTextComponent("potion.potency." + effectinstance.getAmplifier())); + } + + if (effectinstance.getDuration() > 20) { + itextcomponent.appendText(" (") + .appendText(EffectUtils.getPotionDurationString(effectinstance, p_185182_2_)) + .appendText(")"); + } + + tooltip.add(itextcomponent.applyTextStyle(effect.getEffectType() + .getColor())); + } + } + + if (!list1.isEmpty()) { + tooltip.add(new StringTextComponent("")); + tooltip.add((new TranslationTextComponent("potion.whenDrank")).applyTextStyle(TextFormatting.DARK_PURPLE)); + + for (Tuple tuple : list1) { + AttributeModifier attributemodifier2 = tuple.getB(); + double d0 = attributemodifier2.getAmount(); + double d1; + if (attributemodifier2.getOperation() != AttributeModifier.Operation.MULTIPLY_BASE + && attributemodifier2.getOperation() != AttributeModifier.Operation.MULTIPLY_TOTAL) { + d1 = attributemodifier2.getAmount(); + } else { + d1 = attributemodifier2.getAmount() * 100.0D; + } + + if (d0 > 0.0D) { + tooltip.add((new TranslationTextComponent( + "attribute.modifier.plus." + attributemodifier2.getOperation() + .getId(), + ItemStack.DECIMALFORMAT.format(d1), + new TranslationTextComponent("attribute.name." + (String) tuple.getA()))) + .applyTextStyle(TextFormatting.BLUE)); + } else if (d0 < 0.0D) { + d1 = d1 * -1.0D; + tooltip.add((new TranslationTextComponent( + "attribute.modifier.take." + attributemodifier2.getOperation() + .getId(), + ItemStack.DECIMALFORMAT.format(d1), + new TranslationTextComponent("attribute.name." + (String) tuple.getA()))) + .applyTextStyle(TextFormatting.RED)); + } + } + } + + } + +} diff --git a/src/main/java/com/simibubi/create/content/contraptions/fluids/potion/PotionMixingRecipeManager.java b/src/main/java/com/simibubi/create/content/contraptions/fluids/potion/PotionMixingRecipeManager.java new file mode 100644 index 000000000..2fac429f0 --- /dev/null +++ b/src/main/java/com/simibubi/create/content/contraptions/fluids/potion/PotionMixingRecipeManager.java @@ -0,0 +1,158 @@ +package com.simibubi.create.content.contraptions.fluids.potion; + +import java.util.ArrayList; +import java.util.Collection; +import java.util.HashMap; +import java.util.HashSet; +import java.util.List; +import java.util.Map; +import java.util.Set; +import java.util.stream.Collectors; + +import com.simibubi.create.Create; +import com.simibubi.create.content.contraptions.components.mixer.MixingRecipe; +import com.simibubi.create.content.contraptions.processing.HeatCondition; +import com.simibubi.create.content.contraptions.processing.ProcessingRecipeBuilder; +import com.simibubi.create.foundation.fluid.FluidIngredient; + +import net.minecraft.client.resources.ReloadListener; +import net.minecraft.item.Item; +import net.minecraft.item.ItemStack; +import net.minecraft.item.crafting.Ingredient; +import net.minecraft.potion.Potion; +import net.minecraft.potion.PotionBrewing; +import net.minecraft.potion.PotionUtils; +import net.minecraft.potion.Potions; +import net.minecraft.profiler.IProfiler; +import net.minecraft.resources.IResourceManager; +import net.minecraftforge.common.brewing.BrewingRecipeRegistry; +import net.minecraftforge.common.brewing.IBrewingRecipe; +import net.minecraftforge.common.brewing.VanillaBrewingRecipe; +import net.minecraftforge.fluids.FluidStack; +import net.minecraftforge.registries.ForgeRegistries; + +public class PotionMixingRecipeManager { + + public static Map> ALL = new HashMap<>(); + + public static List getAllBrewingRecipes() { + List mixingRecipes = new ArrayList<>(); + + // Vanilla + for (IBrewingRecipe iBrewingRecipe : BrewingRecipeRegistry.getRecipes()) { + if (!(iBrewingRecipe instanceof VanillaBrewingRecipe)) + continue; + + List bottles = new ArrayList<>(); + PotionBrewing.POTION_ITEMS.forEach(i -> { + for (ItemStack itemStack : i.getMatchingStacks()) + bottles.add(itemStack); + }); + + Collection reagents = getAllReagents(iBrewingRecipe); + + Set basicPotions = new HashSet<>(); + for (Potion potion : ForgeRegistries.POTION_TYPES.getValues()) { + if (potion == Potions.EMPTY) + continue; + for (ItemStack stack : bottles) + basicPotions.add(PotionUtils.addPotionToItemStack(stack.copy(), potion)); + } + + Set uniqueKeys = new HashSet<>(); + List potionFrontier = new ArrayList<>(); + List newPotions = new ArrayList<>(); + potionFrontier.addAll(basicPotions); + + int recipeIndex = 0; + + while (!potionFrontier.isEmpty()) { + newPotions.clear(); + + for (ItemStack inputPotionStack : potionFrontier) { + Potion inputPotion = PotionUtils.getPotionFromItem(inputPotionStack); + + for (ItemStack potionReagent : reagents) { + ItemStack outputPotionStack = iBrewingRecipe.getOutput(inputPotionStack.copy(), potionReagent); + if (outputPotionStack.isEmpty()) + continue; + + String uniqueKey = potionReagent.getItem() + .getRegistryName() + .toString() + "_" + + inputPotion.getRegistryName() + .toString() + + "_" + inputPotionStack.getItem() + .getRegistryName() + .toString(); + + if (!uniqueKeys.add(uniqueKey)) + continue; + + if (inputPotionStack.getItem() == outputPotionStack.getItem()) { + Potion outputPotion = PotionUtils.getPotionFromItem(outputPotionStack); + if (outputPotion == Potions.WATER) + continue; + } + + FluidStack fluidFromPotionItem = PotionFluidHandler.getFluidFromPotionItem(inputPotionStack); + FluidStack fluidFromPotionItem2 = PotionFluidHandler.getFluidFromPotionItem(outputPotionStack); + fluidFromPotionItem.setAmount(1000); + fluidFromPotionItem2.setAmount(1000); + + MixingRecipe mixingRecipe = new ProcessingRecipeBuilder<>(MixingRecipe::new, + Create.asResource("potion_" + recipeIndex++)).require(Ingredient.fromStacks(potionReagent)) + .require(FluidIngredient.fromFluidStack(fluidFromPotionItem)) + .output(fluidFromPotionItem2) + .requiresHeat(HeatCondition.HEATED) + .build(); + + mixingRecipes.add(mixingRecipe); + newPotions.add(outputPotionStack); + } + } + + potionFrontier.clear(); + potionFrontier.addAll(newPotions); + } + + break; + } + + // TODO Modded brewing recipes? + + return mixingRecipes; + } + + public static Collection getAllReagents(IBrewingRecipe recipe) { + return ForgeRegistries.ITEMS.getValues() + .stream() + .map(ItemStack::new) + .filter(recipe::isIngredient) + .collect(Collectors.toList()); + } + + public static final ReloadListener LISTENER = new ReloadListener() { + + @Override + protected Object prepare(IResourceManager p_212854_1_, IProfiler p_212854_2_) { + return new Object(); + } + + @Override + protected void apply(Object p_212853_1_, IResourceManager p_212853_2_, IProfiler p_212853_3_) { + ALL.clear(); + getAllBrewingRecipes().forEach(recipe -> { + for (Ingredient ingredient : recipe.getIngredients()) { + for (ItemStack itemStack : ingredient.getMatchingStacks()) { + ALL.computeIfAbsent(itemStack.getItem(), t -> new ArrayList<>()) + .add(recipe); + return; + } + } + }); + } + + }; + +} diff --git a/src/main/java/com/simibubi/create/content/contraptions/processing/BasinBlock.java b/src/main/java/com/simibubi/create/content/contraptions/processing/BasinBlock.java index 6a57ebbb1..6925f7c67 100644 --- a/src/main/java/com/simibubi/create/content/contraptions/processing/BasinBlock.java +++ b/src/main/java/com/simibubi/create/content/contraptions/processing/BasinBlock.java @@ -3,6 +3,7 @@ package com.simibubi.create.content.contraptions.processing; import com.simibubi.create.AllBlocks; import com.simibubi.create.AllShapes; import com.simibubi.create.AllTileEntities; +import com.simibubi.create.content.contraptions.fluids.actors.GenericItemFilling; import com.simibubi.create.content.contraptions.wrench.IWrenchable; import com.simibubi.create.foundation.advancement.AllTriggers; import com.simibubi.create.foundation.block.ITE; @@ -81,7 +82,9 @@ public class BasinBlock extends Block implements ITE, IWrenchab if (!heldItem.isEmpty()) { if (tryEmptyItemIntoBasin(worldIn, player, handIn, heldItem, te)) return ActionResultType.SUCCESS; - return ActionResultType.PASS; + if (tryFillItemFromBasin(worldIn, player, handIn, heldItem, te)) + return ActionResultType.SUCCESS; + return ActionResultType.SUCCESS; } IItemHandlerModifiable inv = te.itemCapability.orElse(new ItemStackHandler(1)); @@ -101,25 +104,72 @@ public class BasinBlock extends Block implements ITE, IWrenchab if (!EmptyingByBasin.canItemBeEmptied(worldIn, heldItem)) return false; - Pair emptyItem = EmptyingByBasin.emptyItem(worldIn, heldItem, true); + Pair emptyingResult = EmptyingByBasin.emptyItem(worldIn, heldItem, true); LazyOptional capability = te.getCapability(CapabilityFluidHandler.FLUID_HANDLER_CAPABILITY); IFluidHandler tank = capability.orElse(null); - FluidStack fluidStack = emptyItem.getFirst(); + FluidStack fluidStack = emptyingResult.getFirst(); if (tank == null || fluidStack.getAmount() != tank.fill(fluidStack, FluidAction.SIMULATE)) return false; if (worldIn.isRemote) return true; - EmptyingByBasin.emptyItem(worldIn, heldItem, false); + ItemStack copyOfHeld = heldItem.copy(); + emptyingResult = EmptyingByBasin.emptyItem(worldIn, copyOfHeld, false); tank.fill(fluidStack, FluidAction.EXECUTE); - if (heldItem.isEmpty()) - player.setHeldItem(handIn, emptyItem.getSecond()); - else - player.inventory.placeItemBackInInventory(worldIn, emptyItem.getSecond()); + + if (!player.isCreative()) { + if (copyOfHeld.isEmpty()) + player.setHeldItem(handIn, emptyingResult.getSecond()); + else { + player.setHeldItem(handIn, copyOfHeld); + player.inventory.placeItemBackInInventory(worldIn, emptyingResult.getSecond()); + } + } return true; } + protected boolean tryFillItemFromBasin(World world, PlayerEntity player, Hand handIn, ItemStack heldItem, + BasinTileEntity te) { + if (!GenericItemFilling.canItemBeFilled(world, heldItem)) + return false; + + LazyOptional capability = te.getCapability(CapabilityFluidHandler.FLUID_HANDLER_CAPABILITY); + IFluidHandler tank = capability.orElse(null); + + if (tank == null) + return false; + + for (int i = 0; i < tank.getTanks(); i++) { + FluidStack fluid = tank.getFluidInTank(i); + if (fluid.isEmpty()) + continue; + int requiredAmountForItem = GenericItemFilling.getRequiredAmountForItem(world, heldItem, fluid.copy()); + if (requiredAmountForItem == -1) + continue; + if (requiredAmountForItem > fluid.getAmount()) + continue; + + if (world.isRemote) + return true; + + if (player.isCreative()) + heldItem = heldItem.copy(); + ItemStack out = GenericItemFilling.fillItem(world, requiredAmountForItem, heldItem, fluid.copy()); + + FluidStack copy = fluid.copy(); + copy.setAmount(requiredAmountForItem); + tank.drain(copy, FluidAction.EXECUTE); + + if (!player.isCreative()) + player.inventory.placeItemBackInInventory(world, out); + te.notifyUpdate(); + return true; + } + + return false; + } + @Override public void onLanded(IBlockReader worldIn, Entity entityIn) { super.onLanded(worldIn, entityIn); diff --git a/src/main/java/com/simibubi/create/content/contraptions/processing/BasinRecipe.java b/src/main/java/com/simibubi/create/content/contraptions/processing/BasinRecipe.java index d258d6adc..32ed3a75a 100644 --- a/src/main/java/com/simibubi/create/content/contraptions/processing/BasinRecipe.java +++ b/src/main/java/com/simibubi/create/content/contraptions/processing/BasinRecipe.java @@ -32,8 +32,23 @@ public class BasinRecipe extends ProcessingRecipe { public static boolean match(BasinTileEntity basin, IRecipe recipe) { FilteringBehaviour filter = basin.getFilter(); - if (filter == null || !filter.test(recipe.getRecipeOutput())) + if (filter == null) return false; + + boolean filterTest = filter.test(recipe.getRecipeOutput()); + if (recipe instanceof BasinRecipe) { + BasinRecipe basinRecipe = (BasinRecipe) recipe; + if (basinRecipe.getRollableResults() + .isEmpty() + && !basinRecipe.getFluidResults() + .isEmpty()) + filterTest = filter.test(basinRecipe.getFluidResults() + .get(0)); + } + + if (!filterTest) + return false; + return apply(basin, recipe, true); } @@ -146,17 +161,11 @@ public class BasinRecipe extends ProcessingRecipe { return true; } - /** - * For JEI purposes only - */ - public boolean convertedRecipe; - - public static BasinRecipe convert(IRecipe recipe) { + public static BasinRecipe convertShapeless(IRecipe recipe) { BasinRecipe basinRecipe = new ProcessingRecipeBuilder<>(BasinRecipe::new, recipe.getId()).withItemIngredients(recipe.getIngredients()) .withSingleItemOutput(recipe.getRecipeOutput()) .build(); - basinRecipe.convertedRecipe = true; return basinRecipe; } diff --git a/src/main/java/com/simibubi/create/content/contraptions/processing/EmptyingByBasin.java b/src/main/java/com/simibubi/create/content/contraptions/processing/EmptyingByBasin.java index 041647651..8545a81d1 100644 --- a/src/main/java/com/simibubi/create/content/contraptions/processing/EmptyingByBasin.java +++ b/src/main/java/com/simibubi/create/content/contraptions/processing/EmptyingByBasin.java @@ -4,9 +4,11 @@ import java.util.List; import java.util.Optional; import com.simibubi.create.AllRecipeTypes; +import com.simibubi.create.content.contraptions.fluids.potion.PotionFluidHandler; import com.simibubi.create.foundation.utility.Pair; import net.minecraft.item.ItemStack; +import net.minecraft.item.PotionItem; import net.minecraft.item.crafting.IRecipe; import net.minecraft.world.World; import net.minecraftforge.common.util.LazyOptional; @@ -22,6 +24,9 @@ public class EmptyingByBasin { static RecipeWrapper wrapper = new RecipeWrapper(new ItemStackHandler(1)); public static boolean canItemBeEmptied(World world, ItemStack stack) { + if (stack.getItem() instanceof PotionItem) + return true; + wrapper.setInventorySlotContents(0, stack); if (AllRecipeTypes.EMPTYING.find(wrapper, world) .isPresent()) @@ -44,6 +49,9 @@ public class EmptyingByBasin { FluidStack resultingFluid = FluidStack.EMPTY; ItemStack resultingItem = ItemStack.EMPTY; + if (stack.getItem() instanceof PotionItem) + return PotionFluidHandler.emptyPotion(stack, simulate); + wrapper.setInventorySlotContents(0, stack); Optional> recipe = AllRecipeTypes.EMPTYING.find(wrapper, world); if (recipe.isPresent()) { diff --git a/src/main/java/com/simibubi/create/content/contraptions/processing/ProcessingRecipeBuilder.java b/src/main/java/com/simibubi/create/content/contraptions/processing/ProcessingRecipeBuilder.java index d33278705..c0b1deb2e 100644 --- a/src/main/java/com/simibubi/create/content/contraptions/processing/ProcessingRecipeBuilder.java +++ b/src/main/java/com/simibubi/create/content/contraptions/processing/ProcessingRecipeBuilder.java @@ -155,7 +155,11 @@ public class ProcessingRecipeBuilder> { } public ProcessingRecipeBuilder output(Fluid fluid, int amount) { - params.fluidResults.add(new FluidStack(fluid, amount)); + return output(new FluidStack(fluid, amount)); + } + + public ProcessingRecipeBuilder output(FluidStack fluidStack) { + params.fluidResults.add(fluidStack); return this; } diff --git a/src/main/java/com/simibubi/create/content/logistics/item/filter/FilterItem.java b/src/main/java/com/simibubi/create/content/logistics/item/filter/FilterItem.java index d66741e7c..abc56ce43 100644 --- a/src/main/java/com/simibubi/create/content/logistics/item/filter/FilterItem.java +++ b/src/main/java/com/simibubi/create/content/logistics/item/filter/FilterItem.java @@ -8,6 +8,7 @@ import javax.annotation.Nonnull; import com.simibubi.create.AllItems; import com.simibubi.create.AllKeys; +import com.simibubi.create.content.contraptions.processing.EmptyingByBasin; import com.simibubi.create.content.logistics.item.filter.AttributeFilterContainer.WhitelistMode; import com.simibubi.create.foundation.item.ItemDescription; import com.simibubi.create.foundation.utility.Lang; @@ -35,7 +36,6 @@ import net.minecraftforge.api.distmarker.Dist; import net.minecraftforge.api.distmarker.OnlyIn; import net.minecraftforge.common.util.Constants.NBT; import net.minecraftforge.fluids.FluidStack; -import net.minecraftforge.fluids.capability.CapabilityFluidHandler; import net.minecraftforge.fml.network.NetworkHooks; import net.minecraftforge.items.ItemHandlerHelper; import net.minecraftforge.items.ItemStackHandler; @@ -185,7 +185,7 @@ public class FilterItem extends Item implements INamedContainerProvider { } public static boolean test(World world, FluidStack stack, ItemStack filter) { - return test(world, stack, filter, false); + return test(world, stack, filter, true); } private static boolean test(World world, ItemStack stack, ItemStack filter, boolean matchNBT) { @@ -264,13 +264,17 @@ public class FilterItem extends Item implements INamedContainerProvider { return false; if (!(filter.getItem() instanceof FilterItem)) { + if (!EmptyingByBasin.canItemBeEmptied(world, filter)) + return false; + FluidStack fluidInFilter = EmptyingByBasin.emptyItem(world, filter, true) + .getFirst(); + if (fluidInFilter == null) + return false; if (!matchNBT) - return filter.getCapability(CapabilityFluidHandler.FLUID_HANDLER_ITEM_CAPABILITY) - .filter(ifh -> ifh.getTanks() > 0) - .map(ifh -> ifh.getFluidInTank(0) - .getFluid() == stack.getFluid()) - .orElse(false); - return stack.isFluidEqual(filter); + return fluidInFilter.getFluid() + .isEquivalentTo(stack.getFluid()); + boolean fluidEqual = fluidInFilter.isFluidEqual(stack); + return fluidEqual; } if (AllItems.FILTER.get() == filter.getItem()) { diff --git a/src/main/java/com/simibubi/create/events/CommonEvents.java b/src/main/java/com/simibubi/create/events/CommonEvents.java index 03f1f59e2..d99961017 100644 --- a/src/main/java/com/simibubi/create/events/CommonEvents.java +++ b/src/main/java/com/simibubi/create/events/CommonEvents.java @@ -4,6 +4,7 @@ import com.simibubi.create.Create; import com.simibubi.create.content.contraptions.components.structureMovement.ContraptionHandler; import com.simibubi.create.content.contraptions.components.structureMovement.train.CouplingPhysics; import com.simibubi.create.content.contraptions.components.structureMovement.train.capability.CapabilityMinecartController; +import com.simibubi.create.content.contraptions.fluids.potion.PotionMixingRecipeManager; import com.simibubi.create.content.contraptions.wrench.WrenchItem; import com.simibubi.create.content.schematics.ServerSchematicLoader; import com.simibubi.create.foundation.command.AllCommands; @@ -13,6 +14,7 @@ import com.simibubi.create.foundation.utility.recipe.RecipeFinder; import net.minecraft.entity.Entity; import net.minecraft.entity.LivingEntity; +import net.minecraft.resources.IReloadableResourceManager; import net.minecraft.world.IWorld; import net.minecraft.world.World; import net.minecraftforge.event.AttachCapabilitiesEvent; @@ -43,7 +45,7 @@ public class CommonEvents { Create.lagger.tick(); ServerSpeedProvider.serverTick(); } - + @SubscribeEvent public static void onChunkUnloaded(ChunkEvent.Unload event) { CapabilityMinecartController.onChunkUnloaded(event); @@ -74,7 +76,7 @@ public class CommonEvents { World world = event.getWorld(); ContraptionHandler.addSpawnedContraptionsToCollisionList(entity, world); } - + @SubscribeEvent public static void onEntityAttackedByPlayer(AttackEntityEvent event) { WrenchItem.wrenchInstaKillsMinecarts(event); @@ -87,9 +89,10 @@ public class CommonEvents { @SubscribeEvent public static void serverAboutToStart(FMLServerAboutToStartEvent event) { - event.getServer() - .getResourceManager() - .addReloadListener(RecipeFinder.LISTENER); + IReloadableResourceManager resourceManager = event.getServer() + .getResourceManager(); + resourceManager.addReloadListener(RecipeFinder.LISTENER); + resourceManager.addReloadListener(PotionMixingRecipeManager.LISTENER); } @SubscribeEvent @@ -111,7 +114,7 @@ public class CommonEvents { Create.torquePropagator.onUnloadWorld(world); WorldAttached.invalidateWorld(world); } - + @SubscribeEvent public static void attachCapabilities(AttachCapabilitiesEvent event) { CapabilityMinecartController.attach(event); diff --git a/src/main/java/com/simibubi/create/foundation/data/CreateRegistrate.java b/src/main/java/com/simibubi/create/foundation/data/CreateRegistrate.java index 47390a57f..f30e4c089 100644 --- a/src/main/java/com/simibubi/create/foundation/data/CreateRegistrate.java +++ b/src/main/java/com/simibubi/create/foundation/data/CreateRegistrate.java @@ -4,9 +4,11 @@ import java.util.Collection; import java.util.IdentityHashMap; import java.util.Map; import java.util.Map.Entry; +import java.util.function.BiFunction; import java.util.function.Supplier; import java.util.stream.Collectors; +import com.simibubi.create.Create; import com.simibubi.create.CreateClient; import com.simibubi.create.content.AllSections; import com.simibubi.create.foundation.block.IBlockVertexColor; @@ -16,6 +18,7 @@ import com.simibubi.create.foundation.block.render.CustomRenderedItemModel; import com.tterrag.registrate.AbstractRegistrate; import com.tterrag.registrate.builders.BlockBuilder; import com.tterrag.registrate.builders.Builder; +import com.tterrag.registrate.builders.FluidBuilder; import com.tterrag.registrate.builders.ItemBuilder; import com.tterrag.registrate.util.NonNullLazyValue; import com.tterrag.registrate.util.entry.RegistryEntry; @@ -29,10 +32,13 @@ import net.minecraft.block.Block.Properties; import net.minecraft.client.renderer.color.IBlockColor; import net.minecraft.client.renderer.color.IItemColor; import net.minecraft.client.renderer.model.IBakedModel; +import net.minecraft.fluid.Fluid; import net.minecraft.item.Item; import net.minecraft.util.IItemProvider; import net.minecraftforge.api.distmarker.Dist; import net.minecraftforge.api.distmarker.OnlyIn; +import net.minecraftforge.fluids.FluidAttributes; +import net.minecraftforge.fluids.ForgeFlowingFluid; import net.minecraftforge.fml.DistExecutor; import net.minecraftforge.fml.RegistryObject; import net.minecraftforge.fml.javafmlmod.FMLJavaModLoadingContext; @@ -114,11 +120,24 @@ public class CreateRegistrate extends AbstractRegistrate { .simpleItem(); } + /* Fluids */ + + public FluidBuilder virtualFluid(String name, + BiFunction attributesFactory, + NonNullFunction factory) { + return entry(name, + c -> new VirtualFluidBuilder<>(self(), self(), name, c, Create.asResource("fluid/" + name + "_still"), + Create.asResource("fluid/" + name + "_flow"), attributesFactory, factory)); + } + + /* Util */ + public static NonNullConsumer connectedTextures(ConnectedTextureBehaviour behavior) { return entry -> onClient(() -> () -> registerCTBehviour(entry, behavior)); } - - public static NonNullConsumer blockModel(Supplier> func) { + + public static NonNullConsumer blockModel( + Supplier> func) { return entry -> onClient(() -> () -> registerBlockModel(entry, func)); } diff --git a/src/main/java/com/simibubi/create/foundation/data/VirtualFluidBuilder.java b/src/main/java/com/simibubi/create/foundation/data/VirtualFluidBuilder.java new file mode 100644 index 000000000..916f3eeb9 --- /dev/null +++ b/src/main/java/com/simibubi/create/foundation/data/VirtualFluidBuilder.java @@ -0,0 +1,35 @@ +package com.simibubi.create.foundation.data; + +import java.util.function.BiFunction; + +import com.tterrag.registrate.AbstractRegistrate; +import com.tterrag.registrate.builders.BuilderCallback; +import com.tterrag.registrate.builders.FluidBuilder; +import com.tterrag.registrate.util.nullness.NonNullFunction; +import com.tterrag.registrate.util.nullness.NonNullSupplier; + +import net.minecraft.fluid.Fluid; +import net.minecraft.util.ResourceLocation; +import net.minecraftforge.fluids.FluidAttributes; +import net.minecraftforge.fluids.ForgeFlowingFluid; +import net.minecraftforge.fluids.ForgeFlowingFluid.Properties; + +/** + * For registering fluids with no buckets/blocks + */ +public class VirtualFluidBuilder extends FluidBuilder { + + public VirtualFluidBuilder(AbstractRegistrate owner, P parent, String name, BuilderCallback callback, + ResourceLocation stillTexture, ResourceLocation flowingTexture, + BiFunction attributesFactory, + NonNullFunction factory) { + super(owner, parent, name, callback, stillTexture, flowingTexture, attributesFactory, factory); + source(factory); + } + + @Override + public NonNullSupplier asSupplier() { + return this::getEntry; + } + +} diff --git a/src/main/java/com/simibubi/create/foundation/data/recipe/EmptyingRecipeGen.java b/src/main/java/com/simibubi/create/foundation/data/recipe/EmptyingRecipeGen.java index 9106e54a8..e49369f0e 100644 --- a/src/main/java/com/simibubi/create/foundation/data/recipe/EmptyingRecipeGen.java +++ b/src/main/java/com/simibubi/create/foundation/data/recipe/EmptyingRecipeGen.java @@ -3,23 +3,21 @@ package com.simibubi.create.foundation.data.recipe; import com.simibubi.create.AllRecipeTypes; import net.minecraft.data.DataGenerator; -import net.minecraft.fluid.Fluids; -import net.minecraft.item.ItemStack; -import net.minecraft.item.Items; -import net.minecraft.potion.PotionUtils; -import net.minecraft.potion.Potions; -import net.minecraftforge.common.crafting.NBTIngredient; public class EmptyingRecipeGen extends ProcessingRecipeGen { - GeneratedRecipe - - WATER_BOTTLE = create("water_bottle", b -> b - .require(NBTIngredient.fromStacks(PotionUtils.addPotionToItemStack(new ItemStack(Items.POTION), Potions.WATER))) - .output(Fluids.WATER, 250) - .output(Items.GLASS_BOTTLE)) - - ; + /* + * potion/bottles are handled internally now. keeping this builder for reference + */ + +// GeneratedRecipe +// +// WATER_BOTTLE = create("water_bottle", b -> b +// .require(NBTIngredient.fromStacks(PotionUtils.addPotionToItemStack(new ItemStack(Items.POTION), Potions.WATER))) +// .output(Fluids.WATER, 250) +// .output(Items.GLASS_BOTTLE)) +// +// ; public EmptyingRecipeGen(DataGenerator p_i48262_1_) { super(p_i48262_1_); diff --git a/src/main/java/com/simibubi/create/foundation/data/recipe/FillingRecipeGen.java b/src/main/java/com/simibubi/create/foundation/data/recipe/FillingRecipeGen.java index 1f2a26e01..480d741ec 100644 --- a/src/main/java/com/simibubi/create/foundation/data/recipe/FillingRecipeGen.java +++ b/src/main/java/com/simibubi/create/foundation/data/recipe/FillingRecipeGen.java @@ -3,21 +3,20 @@ package com.simibubi.create.foundation.data.recipe; import com.simibubi.create.AllRecipeTypes; import net.minecraft.data.DataGenerator; -import net.minecraft.item.ItemStack; -import net.minecraft.item.Items; -import net.minecraft.potion.PotionUtils; -import net.minecraft.potion.Potions; -import net.minecraft.tags.FluidTags; public class FillingRecipeGen extends ProcessingRecipeGen { - GeneratedRecipe + /* + * potion/bottles are handled internally now. keeping this builder for reference + */ - WATER_BOTTLE = create("water_bottle", b -> b.require(Items.GLASS_BOTTLE) - .require(FluidTags.WATER, 250) - .output(PotionUtils.addPotionToItemStack(new ItemStack(Items.POTION), Potions.WATER))) - - ; +// GeneratedRecipe +// +// WATER_BOTTLE = create("water_bottle", b -> b.require(Items.GLASS_BOTTLE) +// .require(FluidTags.WATER, 250) +// .output(PotionUtils.addPotionToItemStack(new ItemStack(Items.POTION), Potions.WATER))) +// +// ; public FillingRecipeGen(DataGenerator p_i48262_1_) { super(p_i48262_1_); diff --git a/src/main/java/com/simibubi/create/foundation/data/recipe/MixingRecipeGen.java b/src/main/java/com/simibubi/create/foundation/data/recipe/MixingRecipeGen.java index 4b0e3a7de..39fb993cf 100644 --- a/src/main/java/com/simibubi/create/foundation/data/recipe/MixingRecipeGen.java +++ b/src/main/java/com/simibubi/create/foundation/data/recipe/MixingRecipeGen.java @@ -9,7 +9,6 @@ import net.minecraft.block.Blocks; import net.minecraft.data.DataGenerator; import net.minecraft.fluid.Fluids; import net.minecraft.item.Items; -import net.minecraft.tags.FluidTags; import net.minecraft.tags.ItemTags; import net.minecraftforge.common.Tags; @@ -17,8 +16,8 @@ public class MixingRecipeGen extends ProcessingRecipeGen { GeneratedRecipe - TEMPCOBBLE = create("temp_cobble", b -> b.require(FluidTags.WATER, 250) - .require(FluidTags.LAVA, 25) + TEMPCOBBLE = create("temp_cobble", b -> b.require(Fluids.WATER, 250) + .require(Fluids.LAVA, 25) .output(Blocks.COBBLESTONE, 1)), TEMP_LAVA = create("temp_lava", b -> b.require(Tags.Items.COBBLESTONE) diff --git a/src/main/java/com/simibubi/create/foundation/fluid/CombinedTankWrapper.java b/src/main/java/com/simibubi/create/foundation/fluid/CombinedTankWrapper.java index 67fce3854..7429764d7 100644 --- a/src/main/java/com/simibubi/create/foundation/fluid/CombinedTankWrapper.java +++ b/src/main/java/com/simibubi/create/foundation/fluid/CombinedTankWrapper.java @@ -1,5 +1,7 @@ package com.simibubi.create.foundation.fluid; +import com.simibubi.create.foundation.utility.Iterate; + import net.minecraftforge.fluids.FluidStack; import net.minecraftforge.fluids.capability.IFluidHandler; import net.minecraftforge.items.wrapper.EmptyHandler; @@ -64,23 +66,29 @@ public class CombinedTankWrapper implements IFluidHandler { public int fill(FluidStack resource, FluidAction action) { if (resource.isEmpty()) return 0; - + int filled = 0; resource = resource.copy(); - for (IFluidHandler iFluidHandler : itemHandler) { - boolean skipRest = false; - int filledIntoCurrent = iFluidHandler.fill(resource, action); - - for (int i = 0; i < iFluidHandler.getTanks(); i++) - if (iFluidHandler.getFluidInTank(i).isFluidEqual(resource) && enforceVariety) - skipRest = true; - - resource.shrink(filledIntoCurrent); - filled += filledIntoCurrent; + boolean fittingHandlerFound = false; + Outer: for (boolean searchPass : Iterate.trueAndFalse) { + for (IFluidHandler iFluidHandler : itemHandler) { - if (resource.isEmpty() || skipRest) - break; + for (int i = 0; i < iFluidHandler.getTanks(); i++) + if (searchPass && iFluidHandler.getFluidInTank(i) + .isFluidEqual(resource)) + fittingHandlerFound = true; + + if (searchPass && !fittingHandlerFound) + continue; + + int filledIntoCurrent = iFluidHandler.fill(resource, action); + resource.shrink(filledIntoCurrent); + filled += filledIntoCurrent; + + if (resource.isEmpty() || fittingHandlerFound || enforceVariety && filledIntoCurrent != 0) + break Outer; + } } return filled; @@ -90,7 +98,7 @@ public class CombinedTankWrapper implements IFluidHandler { public FluidStack drain(FluidStack resource, FluidAction action) { if (resource.isEmpty()) return resource; - + FluidStack drained = FluidStack.EMPTY; resource = resource.copy(); @@ -99,8 +107,8 @@ public class CombinedTankWrapper implements IFluidHandler { int amount = drainedFromCurrent.getAmount(); resource.shrink(amount); - if (!drainedFromCurrent.isEmpty()) - drained = new FluidStack(drainedFromCurrent.getFluid(), amount + drained.getAmount()); + if (!drainedFromCurrent.isEmpty() && (drained.isEmpty() || drainedFromCurrent.isFluidEqual(drained))) + drained = new FluidStack(drainedFromCurrent.getFluid(), amount + drained.getAmount(), drained.getTag()); if (resource.isEmpty()) break; } @@ -117,8 +125,9 @@ public class CombinedTankWrapper implements IFluidHandler { int amount = drainedFromCurrent.getAmount(); maxDrain -= amount; - if (!drainedFromCurrent.isEmpty()) - drained = new FluidStack(drainedFromCurrent.getFluid(), amount + drained.getAmount()); + if (!drainedFromCurrent.isEmpty() && (drained.isEmpty() || drainedFromCurrent.isFluidEqual(drained))) + drained = new FluidStack(drainedFromCurrent.getFluid(), amount + drained.getAmount(), + drainedFromCurrent.getTag()); if (maxDrain == 0) break; } diff --git a/src/main/java/com/simibubi/create/foundation/fluid/FluidIngredient.java b/src/main/java/com/simibubi/create/foundation/fluid/FluidIngredient.java index 2ca8293f1..d7b0e50fa 100644 --- a/src/main/java/com/simibubi/create/foundation/fluid/FluidIngredient.java +++ b/src/main/java/com/simibubi/create/foundation/fluid/FluidIngredient.java @@ -42,6 +42,15 @@ public abstract class FluidIngredient implements Predicate { return ingredient; } + public static FluidIngredient fromFluidStack(FluidStack fluidStack) { + FluidStackIngredient ingredient = new FluidStackIngredient(); + ingredient.fluid = fluidStack.getFluid(); + ingredient.amountRequired = fluidStack.getAmount(); + if (fluidStack.hasTag()) + ingredient.tagToMatch = fluidStack.getTag(); + return ingredient; + } + protected int amountRequired; protected abstract boolean testInternal(FluidStack t); @@ -135,10 +144,12 @@ public abstract class FluidIngredient implements Predicate { if (!t.getFluid() .isEquivalentTo(fluid)) return false; - CompoundNBT tag = t.getTag() - .copy(); - return tag.merge(tagToMatch) - .equals(t.getTag()); + if (tagToMatch.isEmpty()) + return true; + CompoundNBT tag = t.getOrCreateTag(); + return tag.copy() + .merge(tagToMatch) + .equals(tag); } @Override diff --git a/src/main/resources/META-INF/accesstransformer.cfg b/src/main/resources/META-INF/accesstransformer.cfg index 549553b12..2069e9221 100644 --- a/src/main/resources/META-INF/accesstransformer.cfg +++ b/src/main/resources/META-INF/accesstransformer.cfg @@ -13,4 +13,7 @@ public net.minecraft.world.server.ChunkManager field_219253_g #chunksToUnload # ChunkStatus public-f net.minecraft.world.chunk.ChunkStatus field_222617_m #FULL public net.minecraft.world.chunk.ChunkStatus$IGenerationWorker -public net.minecraft.world.chunk.ChunkStatus$ILoadingWorker \ No newline at end of file +public net.minecraft.world.chunk.ChunkStatus$ILoadingWorker + +# PotionBrewing +public net.minecraft.potion.PotionBrewing field_185215_c # POTION_ITEMS diff --git a/src/main/resources/assets/create/lang/default/messages.json b/src/main/resources/assets/create/lang/default/messages.json index b61b73f9f..57d78d5bb 100644 --- a/src/main/resources/assets/create/lang/default/messages.json +++ b/src/main/resources/assets/create/lang/default/messages.json @@ -14,31 +14,31 @@ "create.recipe.crushing": "Crushing", "create.recipe.milling": "Milling", - "create.recipe.splashing": "Bulk Washing", - "create.recipe.splashing.fan": "Fan behind Flowing Water", - "create.recipe.smoking_via_fan": "Bulk Smoking", - "create.recipe.smoking_via_fan.fan": "Fan behind Fire", - "create.recipe.blasting_via_fan": "Bulk Smelting", - "create.recipe.blasting_via_fan.fan": "Fan behind Lava", + "create.recipe.fan_washing": "Bulk Washing", + "create.recipe.fan_washing.fan": "Fan behind Flowing Water", + "create.recipe.fan_smoking": "Bulk Smoking", + "create.recipe.fan_smoking.fan": "Fan behind Fire", + "create.recipe.fan_blasting": "Bulk Blasting", + "create.recipe.fan_blasting.fan": "Fan behind Lava", "create.recipe.pressing": "Pressing", "create.recipe.mixing": "Mixing", - "create.recipe.shapeless_mixing": "Shapeless Crafting", + "create.recipe.automatic_shapeless": "Automated Shapeless Crafting", + "create.recipe.automatic_brewing": "Automated Brewing", "create.recipe.packing": "Compacting", + "create.recipe.automatic_packing": "Automated Packing", "create.recipe.sawing": "Sawing", - "create.recipe.mechanical_crafting": "Crafting", - "create.recipe.mechanical_crafting_exclusive": "Mechanical Crafting", + "create.recipe.mechanical_crafting": "Mechanical Crafting", + "create.recipe.automatic_shaped": "Automated Shaped Crafting", "create.recipe.block_cutting": "Block Cutting", "create.recipe.blockzapper_upgrade": "Handheld Blockzapper", "create.recipe.sandpaper_polishing": "Sandpaper Polishing", - "create.recipe.mystery_conversion": "Chromatic Metamorphosis", - "create.recipe.processing.catalyst": "Catalyst", - "create.recipe.processing.chance": "%1$s%% Chance", - "create.recipe.processing.chanceToReturn": "%1$s%% Chance to Return", - + "create.recipe.mystery_conversion": "Mysterious Conversion", + + "create.recipe.processing.chance": "%1$s%% Chance", "create.recipe.heat_requirement.none": "No Heating Required", "create.recipe.heat_requirement.heated": "Heated", "create.recipe.heat_requirement.superheated": "Super-Heated", - + "create.generic.range": "Range", "create.generic.radius": "Radius", "create.generic.width": "Width", @@ -52,7 +52,6 @@ "create.generic.unit.rpm": "RPM", "create.generic.unit.stress": "su", "create.generic.unit.degrees": "\u00B0", - "create.generic.unit.bucket": "1 Bucket", "create.generic.unit.millibuckets": "%1$smB", "create.action.scroll": "Scroll", diff --git a/src/main/resources/assets/create/textures/fluid/potion_flow.png b/src/main/resources/assets/create/textures/fluid/potion_flow.png new file mode 100644 index 0000000000000000000000000000000000000000..09184febb5adc99f4ad7d18418d88788864d20e1 GIT binary patch literal 133632 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 H6*K=gsC{w` literal 0 HcmV?d00001 diff --git a/src/main/resources/assets/create/textures/fluid/potion_still.png.mcmeta b/src/main/resources/assets/create/textures/fluid/potion_still.png.mcmeta new file mode 100644 index 000000000..24f9c2fae --- /dev/null +++ b/src/main/resources/assets/create/textures/fluid/potion_still.png.mcmeta @@ -0,0 +1,5 @@ +{ + "animation": { + "frametime": 1 + } +} From 42f97477cfe64bed419232ad2d4e80d287adf742 Mon Sep 17 00:00:00 2001 From: Kryppers <65094918+Kryppers@users.noreply.github.com> Date: Mon, 19 Oct 2020 23:16:33 +0100 Subject: [PATCH 2/2] Up-tip the tools. Another couple of tooltip updates; I've still left the smart pipe untouched until I've understood the filtering properly. --- .../assets/create/lang/default/tooltips.json | 15 +++++++++++---- 1 file changed, 11 insertions(+), 4 deletions(-) diff --git a/src/main/resources/assets/create/lang/default/tooltips.json b/src/main/resources/assets/create/lang/default/tooltips.json index 55f496a26..fb1eafaef 100644 --- a/src/main/resources/assets/create/lang/default/tooltips.json +++ b/src/main/resources/assets/create/lang/default/tooltips.json @@ -92,7 +92,7 @@ "block.create.fluid_tank.tooltip": "FLUID TANK", "block.create.fluid_tank.tooltip.summary": "_Stores_ all your favourite _fluids_.", "block.create.fluid_tank.tooltip.control1": "Right-clicked with Wrench", - "block.create.fluid_tank.tooltip.action1": "changes the optional window", + "block.create.fluid_tank.tooltip.action1": "Changes the optional window", "block.create.fluid_valve.tooltip": "FLUID VALVE", "block.create.fluid_valve.tooltip.summary": "Halts the flow of fluid down a pipe.", @@ -108,8 +108,6 @@ "block.create.smart_fluid_pipe.tooltip": "SMART FLUID PIPE", "block.create.smart_fluid_pipe.tooltip.summary": "A _fluid_ _pipe_ with a filter. Can specify which _fluids_ pass through.", - "block.create.smart_fluid_pipe.tooltip.condition1": " ~ ", - "block.create.smart_fluid_pipe.tooltip.behaviour1": " ~ ", "block.create.smart_fluid_pipe.tooltip.control1": " ~ ", "block.create.smart_fluid_pipe.tooltip.action1": " ~ ", @@ -120,6 +118,15 @@ "block.create.spout.tooltip.condition2": "Fluid Automation", "block.create.spout.tooltip.behaviour2": "The spout placed above a _belt_ or _depot_ will react automatically with a _fluid_ _container_ _item_ that passes beneath it.", + "block.create.mechanical_arm.tooltip": "MECHANICAL ARM", + "block.create.mechanical_arm.tooltip.summary": "Advanced contraption for re-locating _items_.", + "block.create.mechanical_arm.tooltip.condition1": "Item Transfer", + "block.create.mechanical_arm.tooltip.behaviour1": "Can take or place items into any _accessible_ _inventory_, such as _Belts_, _Depots_, _Funnels_ and _Mechanical_ _Crafters_.", + "block.create.mechanical_arm.tooltip.control1": "While in Hand", + "block.create.mechanical_arm.tooltip.action1": "Right-Click an _accessible_ _item_ _inventory_ to set it as a _source_ for the _Mechanical_ _Arm_. Right-click twice to set it as the _destination_.", + "block.create.mechanical_arm.tooltip.control2": "Scroll with Wrench", + "block.create.mechanical_arm.tooltip.action2": "Sets the ordering behaviour for _items_ output by the _mechanical_ _Arm_.", + "item.create.wand_of_symmetry.tooltip": "SYMMETRY WAND", "item.create.wand_of_symmetry.tooltip.summary": "Perfectly mirrors Block placement across configured planes.", "item.create.wand_of_symmetry.tooltip.condition1": "When in Hotbar", @@ -363,7 +370,7 @@ "block.create.portable_storage_interface.tooltip": "PORTABLE STORAGE INTERFACE", "block.create.portable_storage_interface.tooltip.summary": "A portable interchange point for _moving_ _items_ to and from a _structure_ moved by a piston, bearing, minecart, or pulley.", "block.create.portable_storage_interface.tooltip.condition1": "While Moving", - "block.create.portable_storage_interface.tooltip.behaviour1": "Interacts with stationary _transposers_ such that transposers _facing_ _away_ from the interface _pull_ _items_, and transposers targeting the interface will _insert_ _items_ from attached inventory. The contraption will briefly stall as items are exchanged.", + "block.create.portable_storage_interface.tooltip.behaviour1": "Interacts with stationary _portable_ _storage_ _interfaces_ to transfer items to or from the contraption. Direction of transfer is dependent on the blocks attached to the _Interface_. The contraption will briefly stall as items are exchanged.", "block.create.rotation_speed_controller.tooltip": "ROTATION SPEED CONTROLLER", "block.create.rotation_speed_controller.tooltip.summary": "A _configurable_ _relay_ able to speed up or slow down the target component to any desired speed.",