Jump to content

Recommended Posts

Posted (edited)

Hello, my friend and I recently started modding because we thought it would be fun, so far it has!

Unfortunately i have been hard-stuck on this particular problem and i really don't know why.

Basically i have made a custom recipe called a "grinding recipe" for my block the grinder (called firstblock for now) and i would like when you put a firstblock item into the top slot for it to "grind it" and transform it into a firstitem. This is really just a proof of concept before i implement it further with actual items. Once it works i can replicate the success easily (hopefully)

The problem code is this:

                       IRecipe<?> irecipe = this.world.getRecipeManager().getRecipe(Objects.requireNonNull((ModItems.FIRSTITEM).getRegistryName())/*(IRecipeType<AbstractGrindingRecipe>)this.recipeType,this,this.world*/).orElse(null);

getRecipe always returns null no matter what i give it. As you can see by the commented out section i have also tried the other method of getRecipe. 

 

My question is, is it not enough that i have registered my recipe like this

package orebesity.common.recipes;

import net.minecraft.item.crafting.IRecipeSerializer;
import net.minecraftforge.registries.ObjectHolder;

public class ModRecipes {
    @ObjectHolder("orebesity:grinding")
    public static IRecipeSerializer<GrindingRecipe> GRINDING;// = IRecipeSerializer.register("grinding",new GrindingRecipeSerializer<>(GrindingRecipe::new,100));

}

 

and made my recipes item file like this?

 

{
  "type": "orebesity:grinding",
  "ingredient": {
    "item": "orebesity:firstblock"
  },
  "result": "orebesity:firstitem",
  "experience": 0.7,
  "grindingtime": 200
}

 

I have no idea how forge adds recipes to the RecipeManager (or if i even have to) but it seems like minecraft doesn't recognize that my firstitem CAN be grinded from my firstblock. Here is my firstblock tile entity code if that is the problem.

class FirstBlockTile extends TileEntity implements ITickableTileEntity, INamedContainerProvider, IRecipeHolder, ISidedInventory {
    private LazyOptional<IItemHandler> handler = LazyOptional.of(this::getHandler);
    public FirstBlockTile() {
        super(FIRSTBLOCK_TILE);
        recipeType=null;
    }
    public FirstBlockTile(IRecipeType<? extends AbstractGrindingRecipe> recipeType) {
        super(FIRSTBLOCK_TILE);
        this.recipeType = recipeType;
    }
    protected final IRecipeType<? extends AbstractGrindingRecipe> recipeType;
    private int grindingTime=0;
    @Override
    public void tick() {
        if(!this.world.isRemote){

           handler.ifPresent(h->{
               boolean isOn=false;
               ItemStack stack = h.getStackInSlot(0);
               if(stack.getItem()==ModBlocks.FIRSTBLOCK.asItem()){
                   isOn=true;
                   if(!stack.isEmpty()){
                       IRecipe<?> irecipe = this.world.getRecipeManager().getRecipe(Objects.requireNonNull((ModItems.FIRSTITEM).getRegistryName())/*(IRecipeType<AbstractGrindingRecipe>)this.recipeType,this,this.world*/).orElse(null);
                       System.out.println("alligator"+ModItems.FIRSTITEM.getRegistryName());
                       System.out.println("crocodile"+irecipe);
                       if(this.canYouGrind(irecipe)){
                           if(stack.hasContainerItem()){

                           }
                           else{
                               if(!stack.isEmpty()){
                                   Item item = stack.getItem();
                                   stack.shrink(1);
                                   if(stack.isEmpty()){

                                   }
                               }
                           }
                       }

                       if(canYouGrind(irecipe)){
                           ++this.grindingTime;
                           if(this.grindingTime==getDefaultGrindTime(stack.getItem())){
                               this.grindingTime=0;
                               this.itemGrinded(irecipe);
                           }
                       }else{
                           this.grindingTime=0;
                       }

                   }


               }
               if(isOn){
                   this.world.setBlockState(pos,this.world.getBlockState(this.pos).with(FirstBlock.ON,Boolean.valueOf("true")));
               }
               else{
                   this.world.setBlockState(pos,this.world.getBlockState(this.pos).with(FirstBlock.ON,Boolean.valueOf("false")));
               }


           });

        }
    }

    protected int getDefaultGrindTime(Item item){
        if(item==ModBlocks.FIRSTBLOCK.asItem())
            return 100;
        return 0;
    }
    private void itemGrinded(@Nullable IRecipe<?> recipe) {
        if (recipe != null && this.canYouGrind(recipe)) {
            ItemStack itemstack1 = recipe.getRecipeOutput();
            handler.ifPresent(h-> {
                ItemStack itemstack = h.getStackInSlot(0);
                ItemStack itemstack2= h.getStackInSlot(1);
                if (itemstack2.isEmpty()) {
                    itemstack2=itemstack1;
                } else if (itemstack2.getItem() == itemstack1.getItem()) {
                    itemstack2.grow(itemstack1.getCount());
                }

                if (!this.world.isRemote) {
                    this.setRecipeUsed(recipe);
                }
                itemstack.shrink(1);
            });
        }
    }
    protected boolean canYouGrind(@Nullable IRecipe<?> recipeType){

        ItemStack stack = recipeType.getRecipeOutput();
        if(stack.isEmpty()){
            return true;
        }else{
            AtomicReference<ItemStack> resultSlot = null;
            handler.ifPresent((IItemHandler h) -> resultSlot.set(h.getStackInSlot(1)));
            if(resultSlot.get().isEmpty())
            {
                return true;
            }else if (!resultSlot.get().isItemEqual(resultSlot.get())) {
                return false;
            } else if (resultSlot.get().getCount() + resultSlot.get().getCount() <= this.getInventoryStackLimit() && resultSlot.get().getCount() + resultSlot.get().getCount() <= resultSlot.get().getMaxStackSize()) { // Forge fix: make furnace respect stack sizes in furnace recipes
                return true;
            } else {
                return resultSlot.get().getCount() + resultSlot.get().getCount() <= resultSlot.get().getMaxStackSize(); // Forge fix: make furnace respect stack sizes in furnace recipes
            }
        }

    }
    @Override
    public void read(CompoundNBT compound) {
        CompoundNBT invTag = compound.getCompound("inv");

        handler.ifPresent(new NonNullConsumer<IItemHandler>() {
            @Override
            public void accept(@Nonnull IItemHandler h) {
                ((INBTSerializable<CompoundNBT>) h).deserializeNBT(invTag);

            }
        });
        super.read(compound);
    }

    @Override
    public CompoundNBT write(CompoundNBT compound) {

        handler.ifPresent(new NonNullConsumer<IItemHandler>() {
            @Override
            public void accept(@Nonnull IItemHandler h) {
                CompoundNBT c = ((INBTSerializable<CompoundNBT>) h).serializeNBT();
                compound.put("inv", c);
            }
        });

        return super.write(compound);
    }
    private IItemHandler getHandler(){
        return new ItemStackHandler(2){
            @Override
            protected void onContentsChanged(int slot) {
                markDirty();
            }
        };
    }
    @Nonnull
    @Override
    public <T> LazyOptional<T> getCapability(@Nonnull Capability<T> cap, @Nullable Direction side) {
        if(cap == CapabilityItemHandler.ITEM_HANDLER_CAPABILITY) {
            return handler.cast();
        }
        return super.getCapability(cap, side);
    }

    @Override
    public ITextComponent getDisplayName() {
        return new StringTextComponent(getType().getRegistryName().getPath());
    }

    @Nullable
    @Override
    public Container createMenu(int i, PlayerInventory playerInventory, PlayerEntity playerEntity) {
        return new FirstBlockContainer(i,pos,world,playerInventory,playerEntity);
    }

    @Override
    public void setRecipeUsed(@Nullable IRecipe<?> recipe) {

    }

    @Nullable
    @Override
    public IRecipe<?> getRecipeUsed() {
        return null;
    }

    @Override
    public int[] getSlotsForFace(Direction side) {
        return new int[0];
    }

    @Override
    public boolean canInsertItem(int index, ItemStack itemStackIn, @Nullable Direction direction) {
        return false;
    }

    @Override
    public boolean canExtractItem(int index, ItemStack stack, Direction direction) {
        return false;
    }

    @Override
    public int getSizeInventory() {
        return 0;
    }

    @Override
    public boolean isEmpty() {
        return false;
    }

    @Override
    public ItemStack getStackInSlot(int index) {
        return null;
    }

    @Override
    public ItemStack decrStackSize(int index, int count) {
        return null;
    }

    @Override
    public ItemStack removeStackFromSlot(int index) {
        return null;
    }

    @Override
    public void setInventorySlotContents(int index, ItemStack stack) {

    }

    @Override
    public boolean isUsableByPlayer(PlayerEntity player) {
        return false;
    }

    @Override
    public void clear() {

    }
}

 

Thanks for the help ❤️

Edited by pancax
Posted

You haven't actually registered your recipeSerializer.

@ObjectHolder is just a reference to the registered serializer, it doesn't register anything. You need to subscribe to the registry event (Just like you do for items/blocks)

Posted

Hello, Thanks for the replay AlpVax! Turns out i have already registered it ( i just forgot to include it in the original post) Here is how i registered it

@SubscribeEvent
        public static void registerRecipes(final RegistryEvent.Register<IRecipeSerializer<?>> event){
            event.getRegistry().register(new GrindingRecipeSerializer<>(GrindingRecipe::new,100));
            LOGGER.info("hello from recipes");

        }

And here is my Grinding Recipe Serializer

 

public class GrindingRecipeSerializer<T extends AbstractGrindingRecipe> extends net.minecraftforge.registries.ForgeRegistryEntry<IRecipeSerializer<?>> implements IRecipeSerializer<T> {
        private final int i;
        private final GrindingRecipeSerializer.IFactory<T> factory;

        public GrindingRecipeSerializer(GrindingRecipeSerializer.IFactory<T> factory, int i){
            this.factory=factory;
            this.i=i;
            setRegistryName("orebesity:grinding");
        }
        @Override
        public T read(ResourceLocation recipeId, JsonObject json) {
            String s = JSONUtils.getString(json, "group", "");
            JsonElement jsonelement = (JsonElement)(JSONUtils.isJsonArray(json, "ingredient") ? JSONUtils.getJsonArray(json, "ingredient") : JSONUtils.getJsonObject(json, "ingredient"));
            Ingredient ingredient = Ingredient.deserialize(jsonelement);
            String s1 = JSONUtils.getString(json, "result");
            ResourceLocation resourcelocation = new ResourceLocation(s1);
            ItemStack itemstack = new ItemStack(Registry.ITEM.getValue(resourcelocation).orElseThrow(() -> new IllegalStateException("Item: " + s1 + " does not exist")));
            float f = JSONUtils.getFloat(json, "experience", 0.0F);
            int i = JSONUtils.getInt(json, "grindingtime", this.i);
            return this.factory.create(recipeId, s, ingredient, itemstack, f, i);
        }

        @Override
        public T read(ResourceLocation recipeId, PacketBuffer buffer) {
            String s = buffer.readString(32767);
            Ingredient ingredient = Ingredient.read(buffer);
            ItemStack itemstack = buffer.readItemStack();
            float f = buffer.readFloat();
            int i = buffer.readVarInt();
            return this.factory.create(recipeId, s, ingredient, itemstack, f, i);
        }

        @Override
        public void write(PacketBuffer buffer, T recipe) {
            buffer.writeString(recipe.group);
            recipe.ingredient.write(buffer);
            buffer.writeItemStack(recipe.result);
            buffer.writeFloat(recipe.experience);
            buffer.writeVarInt(recipe.grindTime);
        }

        public interface IFactory<T extends AbstractGrindingRecipe>{
            T create(ResourceLocation resourcelocation, String name, Ingredient ingredient, ItemStack stack, float f , int i);
        }

}

 

I was wondering if i need to register anything else as well?

Join the conversation

You can post now and register later. If you have an account, sign in now to post with your account.
Note: Your post will require moderator approval before it will be visible.

Guest
Unfortunately, your content contains terms that we do not allow. Please edit your content to remove the highlighted words below.
Reply to this topic...

×   Pasted as rich text.   Restore formatting

  Only 75 emoji are allowed.

×   Your link has been automatically embedded.   Display as a link instead

×   Your previous content has been restored.   Clear editor

×   You cannot paste images directly. Upload or insert images from URL.

Announcements



  • Recently Browsing

    • No registered users viewing this page.
  • Posts

    • But your Launcher does not find it   Java path is: /run/user/1000/doc/3f910b8/java Checking Java version... Java checker returned some invalid data we don't understand: Check the Azul Zulu site and select your OS and download the latest Java 8 build for Linux https://www.azul.com/downloads/?version=java-8-lts&package=jre#zulu After installation, check the path and put this path into your Launcher Java settings (Java Executable)
    • Try other builds of pehkui and origins++ until you find a working combination
    • Some Create addons are only compatible with Create 6 or Create 5 - so not both versions at the same time Try older builds of Create Stuff and Additions This is the last build before the Create 6 update: https://www.curseforge.com/minecraft/mc-mods/create-stuff-additions/files/6168370
    • ✅ Crobo Coupon Code: 51k3b0je — Get Your \$25 Amazon Gift Card Bonus! If you’re new to Crobo and want to make the most out of your first transaction, you’ve come to the right place. The **Crobo coupon code: 51k3b0je** is a fantastic way to get started. By using this code, you can unlock an exclusive **\$25 Amazon gift card** after completing your first eligible transfer. Let’s dive deep into how the **Crobo coupon code: 51k3b0je** works, why you should use it, and how to claim your reward. --- 🌟 What is Crobo? Crobo is a trusted, modern platform designed for **international money transfers**. It offers fast, secure, and low-cost transactions, making it a favorite choice for individuals and businesses alike. Crobo is committed to transparency, low fees, and competitive exchange rates. And with promo deals like the **Crobo coupon code: 51k3b0je**, it becomes even more attractive. Crobo focuses on providing customers with: * Quick transfer speeds * Minimal fees * Safe, encrypted transactions * Great referral and promo code rewards When you choose Crobo, you’re choosing a platform that values your time, money, and loyalty. And now with the **Crobo coupon code: 51k3b0je**, you can start your Crobo journey with a **bonus reward**! ---# 💥 What is the Crobo Coupon Code: 51k3b0je? The **Crobo coupon code: 51k3b0je** is a **special promotional code** designed for new users. By entering this code during signup, you’ll be eligible for: ✅ A **\$25 Amazon gift card** after your first qualifying transfer. ✅ Access to Crobo’s referral system to earn more rewards. ✅ The ability to combine with future seasonal Crobo discounts. Unlike generic promo codes that just offer small fee reductions, the **Crobo coupon code: 51k3b0je** directly gives you a tangible, valuable reward — perfect for online shopping or gifting. --- ### 🎯 Why Use Crobo Coupon Code: 51k3b0je? There are many reasons why users choose to apply the **Crobo coupon code: 51k3b0je**: 🌟 **Free bonus reward** — Your first transfer can instantly earn you a \$25 Amazon gift card. 🌟 **Trusted platform** — Crobo is known for secure, fast, and affordable transfers. 🌟 **Easy to apply** — Simply enter **Crobo coupon code: 51k3b0je** at signup — no complicated steps. 🌟 **Referral opportunities** — Once you’ve used **Crobo coupon code: 51k3b0je**, you can invite friends and earn more rewards. 🌟 **Stackable savings** — Pair **Crobo coupon code: 51k3b0je** with Crobo’s ongoing offers or holiday deals for even more benefits. --- ### 📝 How to Use Crobo Coupon Code: 51k3b0je Getting started with **Crobo coupon code: 51k3b0je** is quick and easy. Just follow these steps: 1️⃣ **Download the Crobo app** (available on Google Play Store and Apple App Store) or visit the official Crobo website. 2️⃣ **Start the sign-up process** by entering your basic details (name, email, phone number, etc.). 3️⃣ When prompted, enter **Crobo coupon code: 51k3b0je** in the promo code or coupon code field. 4️⃣ Complete your first transaction — be sure to meet the minimum amount required to qualify for the reward (usually specified in Crobo’s promo terms). 5️⃣ After the transaction is verified, receive your **\$25 Amazon gift card** directly via email or within your Crobo account. --- ### 💡 Tips to Maximize Your Crobo Coupon Code: 51k3b0je Bonus 👉 **Transfer the minimum qualifying amount or more** — this ensures you meet the conditions for the gift card. 👉 **Refer friends after your signup** — Crobo allows users who’ve signed up with codes like **Crobo coupon code: 51k3b0je** to share their own code for extra bonuses. 👉 **Check for additional Crobo promotions** — sometimes Crobo offers seasonal or regional deals that stack with the coupon code. 👉 **Complete your transaction soon after signup** — many bonuses have time limits, so act quickly! --- ### 🚀 Frequently Asked Questions about Crobo Coupon Code: 51k3b0je **Q: Can I use Crobo coupon code: 51k3b0je if I already have a Crobo account?** A: No — the **Crobo coupon code: 51k3b0je** is intended for **new users only**. It must be applied during the initial registration process. --- **Q: How long does it take to get the \$25 Amazon gift card after using Crobo coupon code: 51k3b0je?** A: Typically, the gift card is sent **within a few business days** after your first qualifying transfer is completed and verified. --- **Q: Are there hidden fees when using Crobo coupon code: 51k3b0je?** A: No — Crobo is transparent about its fees. The **Crobo coupon code: 51k3b0je** simply adds a bonus reward without increasing your costs. --- **Q: Can I combine Crobo coupon code: 51k3b0je with other promo codes?** A: The **Crobo coupon code: 51k3b0je** is generally applied as a standalone signup bonus. However, Crobo often offers **ongoing promotions** that may apply to future transactions. ---  📌 Reference Crobo promo code: {51k3b0je} Crobo discount code: {51k3b0je} --- # 🌍 Final Thoughts If you want to enjoy safe, fast, and affordable money transfers with an added bonus, **Crobo coupon code: 51k3b0je** is your best option. Not only will you experience excellent service, but you’ll also earn a **\$25 Amazon gift card** — a reward that you can use immediately for shopping or gifts. 👉 **Don’t wait — sign up today using Crobo coupon code: 51k3b0je and claim your bonus!**
    • Does this schematic contain stuff from the mod prettypipes? Looks like Forgematica has issues to load it Try to load the schematic with worldedit - remove the pipes, save it and test it again
  • Topics

×
×
  • Create New...

Important Information

By using this site, you agree to our Terms of Use.