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

    • So am trying to make a custom 1.19.2 modpack and everything works until I add Oculus. I have tried Oculus+Embedium and Oculus+Rubdium and by themselves they work but as soon as I add anything it crashes no matter what it is. The modpack works fine with just Embedium and Rubdium. Can you help me to see if this is something i can fix or do i just have to deal with not having shaders. Here is the crash log. Thank you for your time. https://paste.ee/p/WXfNZ24K
    • New users at Temureceive a 40 Off discount on orders over 40 Off Use the code [{acx318439}]] during checkout to get TemuDiscount 40 Off For New Users. You n save 40 Off off your first order with the Promo Code available for a limited time only. Extra 30% off for new and existing customers + Up to $40 Off % off & more. Temu Promo Codes for New users- [{acx318439}]] Temudiscount code for New customers- [{acx318439}]] Temu $40 Off Promo Code- [{acx318439}]] what are Temu codes- acx318439 does Temu give you $40 Off - [{acx318439}]] Yes Verified Temu Promo Code january 2025- {acx318439} TemuNew customer offer {acx318439} Temudiscount codejanuary 2025 {acx318439} 40 off Promo Code Temu {acx318439} Temu 40% off any order {acx318439} 40 dollar off Temu code {acx318439} TemuCoupon $40 Off off for New customers There are a number of discounts and deals shoppers n take advantage of with the Teemu Coupon Bundle [{acx318439}]]. TemuCoupon $40 Off off for New customers [{acx318439}]] will save you $40 Off on your order. To get a discount, click on the item to purchase and enter the code. You n think of it as a supercharged savings pack for all your shopping needs Temu Promo Code 80% off – [{acx318439}]] Free Temu codes 50% off – [{acx318439}]] TemuCoupon $40 Off off – [{acx318439}]] Temu buy to get ₱39 – [{acx318439}]] Temu 129 coupon bundle – [{acx318439}]] Temu buy 3 to get €99 – [{acx318439}]] Exclusive $40 Off Off TemuDiscount Code Temu $40 Off Off Promo Code : (acx318439) Temu Discount Code $40 Off Bundle acx318439) acx318439 Temu $40 Off off Promo Code for Exsting users : acx318439) Temu Promo Code $40 Off off Temu 40 Off coupon code (acx318439) will save you 40 Off on your order. To get a discount, click on the item to purchase and enter the code. Yes, Temu offers 40 Off Coupon Code “acx318439” for Existing Customers.  You can get a 40 Off bonus plus 30% off any purchase at Temu with the 40 Off Coupon Bundle at Temu if you sign up with the referral code [{acx318439}]] and make a first purchase of $40 Off or more. Temu Promo Code 40 off-{acx318439} Temu Promo Code -{acx318439} Temu Promo Code $40 Off off-{acx318439} kubonus code -{acx318439} Get ready to unlock a world of savings with our free Temu UK coupons! We’ve got you covered with a wide range of Temu UK coupon code options that will help you maximize your shopping experience.30% Off Temu UK Coupons, Promo Codes + 25% Cash Back [ acx318439]   Yes, Temu offers 40 off coupon code {acx318439} for first-time users. You can get a $40 bonus plus 40% off any purchase at Temu with the $40 Coupon Bundle if you sign up with the referral code [{acx318439}]] and make a first purchase of $40 or more. If you are who wish to join Temu, then you should use this exclusive TemuCoupon code 40 off (acx318439) and get 40 off on your purchase with Temu. You can get a 40% discount with TemuCoupon code {acx318439}. This exclusive offer is for existing customers and can be used for a 40 reduction on your total purchase. Enter coupon code {acx318439} at checkout to avail of the discount. You can use the code {acx318439} to get a 40 off TemuCoupon as a new customer. Apply this TemuCoupon code $40 off (acx318439) to get a $40 discount on your shopping with Temu. If you’re a first-time user and looking for a TemuCoupon code $40 first time user(acx318439) then using this code will give you a flat $40 Off and a 90% discount on your Temu shopping.     •    acx318439: Enjoy flat 40% off on your first Temu order.     •    acx318439: Download the Temu app and get an additional 40% off.     •    acx318439: Celebrate spring with up to 90% discount on selected items.     •    acx318439: Score up to 90% off on clearance items.     •    acx318439: Beat the heat with hot summer savings of up to 90% off.     •    acx318439: Temu UK Coupon Code to 40% off on Appliances at Temu. How to Apply Temu Coupon Code? Using the TemuCoupon code $40 off is a breeze. All you need to do is follow these simple steps:     1    Visit the Temu website or app and browse through the vast collection of products.     2    Once you’ve added the items you wish to purchase to your cart, proceed to the checkout page.     3    During the checkout process, you’ll be prompted to enter a coupon code or promo code.     4    Type in the coupon code: [{acx318439}]] and click “Apply.”     5    Voila! You’ll instantly see the $40 discount reflected in your total purchase amount. Temu New User Coupon: Up To 90% OFF For Existing Customers Temu Existing customer’s coupon codes are designed just for new customers, offering the biggest discounts 90% and the best deals currently available on Temu. To maximize your savings, download the Temu app and apply our Temu new user coupon during checkout.     •    acx318439: New users can get up to 80% extra off.     •    acx318439: Get a massive 40% off your first order!     •    acx318439: Get 20% off on your first order; no minimum spending required.     •    acx318439: Take an extra 15% off your first order on top of existing discounts.     •    acx318439: Temu UK Enjoy a 40% discount on your entire first purchase.  
    • What do I do now when it says "1 error"?
    • Hello everyone new here how are you all?
    • I haven't tested it but under https://minecraft.wiki/w/Items_model_definition it says now:   So I guess the resource location must have changed with 1.24.4, which means you need to move your models/item/ to the new source. But as I said I haven't tested this so it also may be that this wont work. Nevertheless give it a try      EDIT (important) So now I tested it and found out how it works   Let the model files (e.g. the .json from blockbench) within "assets/<your_mod_id>/models/item" In addition to that do the following: Every model you added will need a new file under "assets/<your_mod_id>/items" That file is also a JSON and looks like this: { "model": { "type": "minecraft:model", "model": "your_mod_id:item/custom_item" } } - "type" can be minecraft:model, minecraft:composite, minecraft:condition, minecraft:select, minecraft:range_dispatch, minecraft:empty, minecraft:bundle/selected_item or minecraft:special. (In most cases you would need minecraft:model) - "model" is the path to your actual model for this item. For example the value above would point to "assets/your_mod_id/models/item/custom_item"
  • Topics

×
×
  • Create New...

Important Information

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