Jump to content

[Solved] Craft Result Disappears After Left-Click


Lumby

Recommended Posts

I have a block that works sort of like a crafting table in that it uses no tileEntities and instantly crafts something out of InventoryCrafting slots. After the crafting result appears, however, although I can shift click the result into my inventory without problems, I am unable to left click the craft result Itemstack as it just vanishes. Is this a result of the code being on the wrong side? If so, should the code in onCraftMatrixChanged() be on the server or client side? Thanks in advance!

 

Container class:

public class ContainerArkenstoneTable extends Container{
    public InventoryCrafting inputInventory = new InventoryCrafting(this, 3, 1);
    public int inputSlotNumber;
    public InventoryArkenstoneResult outputInventory = new InventoryArkenstoneResult();
    public ArkenstoneRecipeHandler arkenstoneRecipeHandler;
    private final World world;
    private final BlockPos pos;
    private final InventoryPlayer playerInventory;
    
    
    public ContainerArkenstoneTable(InventoryPlayer playerInventory, World worldIn, BlockPos posIn){
        this.world = worldIn;
        this.pos = posIn;
        this.playerInventory = playerInventory;
        
        arkenstoneRecipeHandler = new ArkenstoneRecipeHandler();
        
        this.addSlotToContainer(new NoInputSlot(outputInventory, 0, 124, 31));
        this.addSlotToContainer(new DiamondOnlySlot(inputInventory, 0, 26 + 0 * 18, 9 + 18));
        this.addSlotToContainer(new Slot(inputInventory, 1, 44 + 1 * 18, 9 + 18));
        this.addSlotToContainer(new Slot(inputInventory, 2, 44 + 1 * 18, 13 + 2*18));
        

        for (int k = 0; k < 3; ++k){
            for (int i1 = 0; i1 < 9; ++i1){
                this.addSlotToContainer(new Slot(playerInventory, i1 + k * 9 + 9, 8 + i1 * 18, 84 + k * 18));
            }
        }

        for (int l = 0; l < 9; ++l){
            this.addSlotToContainer(new Slot(playerInventory, l, 8 + l * 18, 142));
        }
    }
    
    /**
     * Callback for when the crafting matrix is changed.
     */
    public void onCraftMatrixChanged(IInventory inventoryIn){

		//if the inventory selected is actually input inventory
    	if(inventoryIn == inputInventory){
    		//if this thing is empty, stahp
            while(!inputInventory.isEmpty()) {
                ItemStack outputItemStack = arkenstoneRecipeHandler.getArkenstoneResults(inputInventory);
                
                //if this input crafts nothing
                if (outputItemStack == ItemStack.EMPTY ){
                    return;
                }else {//if this input can craft something
            		//if something is already lingering in outputInventory from before
            	  	if(!outputInventory.isEmpty()){
        	  			//if what is supposed to be in output is NOT the same as the thing actually in output
        	  			if(!outputItemStack.isItemEqual(outputInventory.getStackInSlot(0))){
        	  				//Add what is supposed to go into output inventory to player inventory 
        	  				//automatically, or drop it on the ground
        	  				if(!playerInventory.addItemStackToInventory(outputItemStack)){
        	  					EntityItem entityItem = playerInventory.player.entityDropItem(outputItemStack, 0.5f);
        	  					entityItem.posX = playerInventory.player.posX;
        	  					entityItem.posY = playerInventory.player.posY;
        	  					entityItem.posZ = playerInventory.player.posZ;
        	  				}
        	  				//clear the output inventory
        	  				outputInventory.setInventorySlotContents(0, ItemStack.EMPTY);
        	  			}
        	  		}//at this point output inventory of a different yield should be cleared
            	  	
            	  	//get current item stack in case it's the same item causing the segment before to not clear it
            	  	ItemStack currentStack = outputInventory.getStackInSlot(0);
        	  		//Test to see if different type of dye by getting item damage
        	  		int metadata = outputItemStack.getItemDamage();
        	  		if(metadata == 32767){
        	  			metadata = 0;
        	  		}
        	  		ItemStack newStack = ItemStack.EMPTY;
        	  		//Output Slot already has the item we're trying to craft, just tack on more of it
        	  		if(!currentStack.isEmpty() && 1 + currentStack.getCount() <= outputItemStack.getMaxStackSize()){
        	  			newStack = new ItemStack(outputItemStack.getItem(), 1 + currentStack.getCount(), metadata);
        	  		}else{ //if output slot is already empty OR output slot can't handle the amount we're crafting
        	  			//if output slot already has the item we're trying to craft, but just couldn't fit so much
        	  			if(!currentStack.isEmpty() && !playerInventory.addItemStackToInventory(currentStack)){
        	  				EntityItem entityItem = playerInventory.player.entityDropItem(currentStack, 0.5f);
        	  				entityItem.posX = playerInventory.player.posX;
        	  				entityItem.posY = playerInventory.player.posY;
        	  				entityItem.posZ = playerInventory.player.posZ;
        	  			}else{
        	  				newStack = new ItemStack(outputItemStack.getItem(), 1, metadata);
        	  			}
        	  		}
        	  		outputInventory.setInventorySlotContents(0, newStack);
            	  	
            	  	//removes the contents from input inventory after crafting is completed
            	  	inputInventory.decrStackSize(0,1);
            	  	inputInventory.decrStackSize(1,1);
            	  	inputInventory.decrStackSize(2,1);

                }
            }//while loop ends

    	}
    }


    

    /**
     * Called when the container is closed.
     */
    @Override
    public void onContainerClosed(EntityPlayer playerIn)
    {
    	InventoryPlayer inventoryplayer = playerIn.inventory;
        if(!inventoryplayer.getItemStack().isEmpty())
        {
            playerIn.dropItem(inventoryplayer.getItemStack(), false);
            inventoryplayer.setItemStack(ItemStack.EMPTY);
        }
        if(!world.isRemote) {
        	ItemStack itemStack = outputInventory.getStackInSlot(0);
            if(!itemStack.isEmpty()){
            	if(!inventoryplayer.addItemStackToInventory(itemStack)) {
            		playerIn.dropItem(itemStack, false);
            	}
            }
            
            for(int i = 0; i < inputInventory.getSizeInventory(); i++ ){
                itemStack = inputInventory.getStackInSlot(i);
                if(!itemStack.isEmpty()){
                	if(!inventoryplayer.addItemStackToInventory(itemStack)) {
                		playerIn.dropItem(itemStack, false);
                	}
                }
            }
        }
    }

    /**
     * Determines whether supplied player can use this container
     */
    @Override
    public boolean canInteractWith(EntityPlayer playerIn)
    {
        if (this.world.getBlockState(this.pos).getBlock() != ModBlocks.ArkenstoneTableBlock)
        {
            return false;
        }
        else
        {
            return playerIn.getDistanceSq((double)this.pos.getX() + 0.5D, (double)this.pos.getY() + 0.5D, (double)this.pos.getZ() + 0.5D) <= 64.0D;
        }
    }

    /**
     * Handle when the stack in slot {@code index} is shift-clicked. Normally this moves the stack between the player
     * inventory and the other inventory(s).
     */
	@Override
	public ItemStack transferStackInSlot(final EntityPlayer player, final int index) {
		return ModUtils.transferStackInSlot(player, index, this);
	}
 

    /**
     * Called to determine if the current slot is valid for the stack merging (double-click) code. The stack passed in
     * is null for the initial slot that was double-clicked.
     */
    public boolean canMergeSlot(ItemStack stack, Slot slotIn){
        return slotIn.inventory != this.outputInventory && super.canMergeSlot(stack, slotIn);
    }
    


}

 

ModUtils class with methods that the above class references:

public class ModUtils {
	
	
	public static ItemStack transferStackInSlot(final EntityPlayer player, final int index, final Container container) {
		ItemStack itemstack = ItemStack.EMPTY;
		final Slot slot = container.inventorySlots.get(index);
		if ((slot != null) && slot.getHasStack()) {
			final ItemStack itemstack1 = slot.getStack();
			itemstack = itemstack1.copy();

			final int containerSlots = container.inventorySlots.size() - player.inventory.mainInventory.size();
			if (index < containerSlots) {
				if (!mergeItemStack(itemstack1, containerSlots, container.inventorySlots.size(), true, container)) {
					return ItemStack.EMPTY;
				}
			} else if (!mergeItemStack(itemstack1, 0, containerSlots, false, container)) {
				return ItemStack.EMPTY;
			}
			if (itemstack1.getCount() == 0) {
				slot.putStack(ItemStack.EMPTY);
			} else {
				slot.onSlotChanged();
			}
			if (itemstack1.getCount() == itemstack.getCount()) {
				return ItemStack.EMPTY;
			}
			slot.onTake(player, itemstack1);
		}
		if(container instanceof ContainerArkenstoneTable) {
			container.onCraftMatrixChanged(((ContainerArkenstoneTable) container).inputInventory);
		}
		return itemstack;
	}
   
   public static boolean mergeItemStack(final ItemStack stack, final int startIndex, final int endIndex, final boolean reverseDirection, final Container container) {
		boolean flag = false;
		int i = startIndex;

		if (reverseDirection) {
			i = endIndex - 1;
		}

		if (stack.isStackable()) {
			while (!stack.isEmpty()) {
				if (reverseDirection) {
					if (i < startIndex) {
						break;
					}
				} else if (i >= endIndex) {
					break;
				}

				final Slot slot = container.inventorySlots.get(i);
				final ItemStack itemstack = slot.getStack();

				if (!itemstack.isEmpty() && (itemstack.getItem() == stack.getItem()) && (!stack.getHasSubtypes() || (stack.getMetadata() == itemstack.getMetadata())) && ItemStack.areItemStackTagsEqual(stack, itemstack)) {
					final int j = itemstack.getCount() + stack.getCount();
					final int maxSize = Math.min(slot.getSlotStackLimit(), stack.getMaxStackSize());

					if (j <= maxSize) {
						stack.setCount(0);
						itemstack.setCount(j);
						slot.onSlotChanged();
						flag = true;
					} else if (itemstack.getCount() < maxSize) {
						stack.shrink(maxSize - itemstack.getCount());
						itemstack.setCount(maxSize);
						slot.onSlotChanged();
						flag = true;
					}
				}

				if (reverseDirection) {
					--i;
				} else {
					++i;
				}
			}
		}

		if (!stack.isEmpty()) {
			if (reverseDirection) {
				i = endIndex - 1;
			} else {
				i = startIndex;
			}

			while (true) {
				if (reverseDirection) {
					if (i < startIndex) {
						break;
					}
				} else if (i >= endIndex) {
					break;
				}

				final Slot slot1 = container.inventorySlots.get(i);
				final ItemStack itemstack1 = slot1.getStack();

				if (itemstack1.isEmpty() && slot1.isItemValid(stack)) {
					if (stack.getCount() > slot1.getSlotStackLimit()) {
						slot1.putStack(stack.splitStack(slot1.getSlotStackLimit()));
					} else {
						slot1.putStack(stack.splitStack(stack.getCount()));
					}

					slot1.onSlotChanged();
					flag = true;
					break;
				}

				if (reverseDirection) {
					--i;
				} else {
					++i;
				}
			}
		}

		return flag;
	}
}

 

ArkenstoneRecipeHandler class

package com.crumbletheundead.blocks.machines.ArkenstoneTable;

import net.minecraft.init.Items;
import net.minecraft.inventory.InventoryCrafting;
import net.minecraft.item.Item;
import net.minecraft.item.ItemStack;

public class ArkenstoneRecipeHandler {
	
	//items put into the crafting grid
	private boolean hasEmpty, hasDiamond, hasBone;
	
	public ArkenstoneRecipeHandler() {
	}
	
	public ItemStack getArkenstoneResults(InventoryCrafting inputInventory) {
		hasEmpty = false;
		hasDiamond = false;
		hasBone = false;
		//test if first slot has diamonds
		if(inputInventory.getStackInSlot(0).getItem().equals(Items.DIAMOND)) {
			hasDiamond = true;
		}
		
		//go through the other two slots to see if they have any items required
        for(int i = 1; i<= inputInventory.getSizeInventory()-1; i++) {
        	if(inputInventory.getStackInSlot(i).isEmpty()) {
        		hasEmpty = true;
        	}else if(inputInventory.getStackInSlot(i).getItem().equals(Items.BONE)) {
        		hasBone = true;
        	}
        }
        
        if(hasDiamond && hasBone && hasEmpty) {
        	return new ItemStack(Items.APPLE);
        }else {
        	return ItemStack.EMPTY;
        }
	}
	
}

 

InventoryArkenstone class

package com.crumbletheundead.blocks.machines.ArkenstoneTable;

import javax.annotation.Nullable;

import net.minecraft.entity.player.EntityPlayer;
import net.minecraft.inventory.IInventory;
import net.minecraft.inventory.ItemStackHelper;
import net.minecraft.item.ItemStack;
import net.minecraft.item.crafting.IRecipe;
import net.minecraft.util.NonNullList;
import net.minecraft.util.text.ITextComponent;
import net.minecraft.util.text.TextComponentString;
import net.minecraft.util.text.TextComponentTranslation;

public class InventoryArkenstoneResult implements IInventory{
	private final NonNullList<ItemStack> stackResult = NonNullList.<ItemStack>withSize(1, ItemStack.EMPTY);
	
    /**
     * Returns the number of slots in the inventory.
     */
    @Override
    public int getSizeInventory()
    {
        return 1;
    }

    /**
     * Returns the stack in slot i
     */
    @Override
    public ItemStack getStackInSlot(int index)
    {
        return this.stackResult.get(0);
    }

    
    @Override
    public boolean isEmpty()
    {
        for (ItemStack itemstack : this.stackResult)
        {
            if (!itemstack.isEmpty())
            {
                return false;
            }
        }

        return true;
    }
    
    @Override
    public String getName()
    {
        return "Arkenstone Result";
    }
    
    @Override
    public boolean hasCustomName()
    {
        return false;
    }
    
    @Override
    public ITextComponent getDisplayName()
    {
        return new TextComponentTranslation(this.getName(), new Object[0]);
    }
    
    @Override
    public ItemStack decrStackSize(int index, int count)
    {
        if(this.stackResult.get(index) != null)
        {
            ItemStack itemstack = this.stackResult.get(index);
            this.stackResult.get(index).setCount(0);;
            return itemstack;
        }
        else
        {
            return ItemStack.EMPTY;
        }
    }
    
    
    @Override
    public ItemStack removeStackFromSlot(int index)
    {
        return ItemStackHelper.getAndRemove(this.stackResult, 0);
    }
    @Override
    public void setInventorySlotContents(int index, ItemStack stack)
    {
        this.stackResult.set(0, stack);
    }
    @Override
    public int getInventoryStackLimit()
    {
        return 64;
    }
    @Override
    public boolean isUsableByPlayer(EntityPlayer player)
    {
        return true;
    }
    @Override
    public void openInventory(EntityPlayer player)
    {
    }
    @Override
    public void closeInventory(EntityPlayer player)
    {
    }
    @Override
    public boolean isItemValidForSlot(int index, ItemStack stack)
    {
        return false;
    }
    public int getField(int id)
    {
        return 0;
    }

    public void setField(int id, int value)
    {
    }

    public int getFieldCount()
    {
        return 0;
    }

    public void clear()
    {
        this.stackResult.clear();
    }

	@Override
	public void markDirty() {
		
		
	}
}

 

Edited by Lumby
solved
Link to comment
Share on other sites

My initial guess was that since items automatically crafted, when the output slot was clicked it checked the inputInventory again to find nothing and hence reset the output to ItemStack.Empty. After an afternoon in the debug perspective, however, I feel like the problem isn't within the onCraftingMatrixChanged() method, as it isn't called when I take items out of the slot. Any ideas what else may be a possible cause?

Link to comment
Share on other sites

I've found out the problem. The problem was that my InventoryArkenstoneResult class had an outdated decrStackSize, since I was following a tutorial from older forge versions. The new one can pulled from the InventoryCraftingResult class if anyone else encounters this problem in the future. 

    public ItemStack decrStackSize(int index, int count)
    {
        return ItemStackHelper.getAndRemove(this.stackResult, 0);
    }

 

Link to comment
Share on other sites

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

    • Working with Lee Ultimate Hacker was a transformative experience that not only restored hacked funds but also exemplified a level of professionalism that exceeded expectations. In a world where cryptocurrency theft is a prevalent threat, the need for reliable recovery services is paramount. My testimony delves into my journey with Lee Ultimate Hacker, highlighting the profound impact of hacked funds and the exceptional recovery process provided by the company. From the initial consultation to the execution of the recovery plan, the expertise and dedication of Lee Ultimate Hacker shone through, setting a new standard for crypto recovery services. Within the rapidly evolving realm of cryptocurrencies, there is always a chance of fraud and hacking. Cybercrime affects a lot of people and companies, costing them their hard-earned digital assets. As a result, there is an increasing need for reputable and trustworthy crypto recovery services to assist victims in getting back what is truly theirs. A beam of brilliance amid the swirling storms of cryptocurrency robbery is Lee Ultimate Hacker. Lee Ultimate Hacker is passionate about assisting clients in safety and effectively recovering their monies that have been compromised, and it does so by utilizing state-of-the-art technology and a team of knowledgeable specialists. In the crypto recovery market, they are a well-known brand thanks to their track record of expertise and accomplishments. The repercussions of having your crypto assets hacked can be devastating. Apart from the financial loss, there is also a sense of violation and helplessness that comes with being a victim of cybercrime. The emotional toll and stress of dealing with such a situation can be overwhelming, affecting both personal and professional aspects of life. Upon contacting Lee Ultimate Hacker via LEEULTIMATEHACKER@ AOL. COM   Support @ leeultimatehacker . com.  telegram:LEEULTIMATE   wh@tsapp +1  (715) 314  -  9248  https://leeultimatehacker.com I met with a thorough initial consultation to understand the details of the hack and assess the extent of the damage. This step is crucial in formulating a tailored recovery plan that suits the specific needs of each client. With their extensive knowledge of blockchain technology and forensic investigation abilities, they create a recovery plan that increases the likelihood that the funds will be successfully retrieved.  Lee Ultimate Hacker has an illustrious history of accomplishment. The team's professionalism, knowledge, and commitment to helping people recover from crypto-related issues have been emphasized by the countless clients who have shared their great experiences working with them.
    • Hi, has anyone dealt with gradlew :runClient crashing on start before? I've currently made a forge 1.18.2 mod with the latest mdk and I use IntelliJ IDEA 2024.1 as my IDE, and now whenever I try to run the `:runClient` command with gradle, it works at first, but then the process stops reporting this issue: `Caused by: java.lang.reflect.InvocationTargetException` and `Caused by: java.lang.NoSuchMethodError: 'int net.minecraft.util.Mth.m_14045_(int, int, int)'`. I've pasted the full runClient gradle log in this message. I investigated this issue further on the forge forums to find that not much people had encountered it, and those who did encounter it somehow fixed it with fixes that does not work for me like deleting cache and let gradle redownload the cache or anything and that this issue is caused by a "Corrupted Cache", or whatever the heck that meant. I tried cloning my entire repo (https://github.com/Type-32/PreciseManufacturing) to another directory to "start fresh" but the same issue persists. I created a new project with a clean forge mod 1.18.2 template but the issue persists. I tried all the fixes I can find but none of them worked. even `.\gradlew --refresh-dependencies` didn't work. I am getting desparate for any help now ~~this is so freaking frustrating~~ This is my build.gradle file plugins { id 'eclipse' id 'idea' id 'net.minecraftforge.gradle' version '[6.0.16,6.2)' id 'org.parchmentmc.librarian.forgegradle' version '1.+' } group = mod_group_id version = mod_version base { archivesName = mod_id } java { toolchain.languageVersion = JavaLanguageVersion.of(17) } minecraft { mappings channel: mapping_channel, version: mapping_version copyIdeResources = true runs { // applies to all the run configs below configureEach { workingDirectory project.file('run') property 'forge.logging.markers', 'REGISTRIES' property 'forge.logging.console.level', 'debug' mods { "${mod_id}" { source sourceSets.main } } } client { // Comma-separated list of namespaces to load gametests from. Empty = all namespaces. property 'forge.enabledGameTestNamespaces', mod_id property 'mixin.env.remapRefMap', 'true' property 'mixin.env.refMapRemappingFile', "${projectDir}/build/createSrgToMcp/output.srg" } server { property 'forge.enabledGameTestNamespaces', mod_id args '--nogui' } gameTestServer { property 'forge.enabledGameTestNamespaces', mod_id } data { // example of overriding the workingDirectory set in configureEach above workingDirectory project.file('run-data') // Specify the modid for data generation, where to output the resulting resource, and where to look for existing resources. args '--mod', mod_id, '--all', '--output', file('src/generated/resources/'), '--existing', file('src/main/resources/') } } } jarJar.enable() reobf { jarJar { } } tasks.jarJar.finalizedBy('reobfJarJar') // Include resources generated by data generators. sourceSets.main.resources { srcDir 'src/generated/resources' } repositories { maven { name = 'tterrag maven' url = 'https://maven.tterrag.com/' } mavenLocal() } dependencies { minecraft "net.minecraftforge:forge:${minecraft_version}-${forge_version}" implementation fg.deobf("com.simibubi.create:create-${create_minecraft_version}:${create_version}:slim") { transitive = false } implementation fg.deobf("com.jozufozu.flywheel:flywheel-forge-${flywheel_minecraft_version}:${flywheel_version}") implementation fg.deobf("com.tterrag.registrate:Registrate:${registrate_version}") // [MC<minecraft_version>,MC<next_minecraft_version>) jarJar(group: 'com.tterrag.registrate', name: 'Registrate', version: "[MC1.18.2,MC1.19)") // Example mod dependency with JEI - using fg.deobf() ensures the dependency is remapped to your development mappings // The JEI API is declared for compile time use, while the full JEI artifact is used at runtime // compileOnly fg.deobf("mezz.jei:jei-${mc_version}-common-api:${jei_version}") // compileOnly fg.deobf("mezz.jei:jei-${mc_version}-forge-api:${jei_version}") // runtimeOnly fg.deobf("mezz.jei:jei-${mc_version}-forge:${jei_version}") // Example mod dependency using a mod jar from ./libs with a flat dir repository // This maps to ./libs/coolmod-${mc_version}-${coolmod_version}.jar // The group id is ignored when searching -- in this case, it is "blank" // implementation fg.deobf("blank:coolmod-${mc_version}:${coolmod_version}") } tasks.named('processResources', ProcessResources).configure { var replaceProperties = [ minecraft_version: minecraft_version, minecraft_version_range: minecraft_version_range, forge_version: forge_version, forge_version_range: forge_version_range, loader_version_range: loader_version_range, mod_id: mod_id, mod_name: mod_name, mod_license: mod_license, mod_version: mod_version, mod_authors: mod_authors, mod_description: mod_description, ] inputs.properties replaceProperties filesMatching(['META-INF/mods.toml', 'pack.mcmeta']) { expand replaceProperties + [project: project] }} // Example for how to get properties into the manifest for reading at runtime. tasks.named('jar', Jar).configure { manifest { attributes([ "Specification-Title": mod_id, "Specification-Vendor": mod_authors, "Specification-Version": "1", // We are version 1 of ourselves "Implementation-Title": project.name, "Implementation-Version": project.jar.archiveVersion, "Implementation-Vendor": mod_authors, "Implementation-Timestamp": new Date().format("yyyy-MM-dd'T'HH:mm:ssZ") ]) } // This is the preferred method to reobfuscate your jar file finalizedBy 'reobfJar' } tasks.withType(JavaCompile).configureEach { options.encoding = 'UTF-8' // Use the UTF-8 charset for Java compilation }  
    • pastebin.com and paste.ee couldn't handle the debug and crash log sizes, apologies I have to send it on mega, and I hope it's not a problem Basically, Minecraft turns on normally but when I try to create a world, it goes to 100%, joining world, saving world and crashes   debug log https://mega.nz/file/kP1nGDKZ decryption key: C_VSH-IO6Kpi9IdqUs2Z0KDu0Fpujmen_I3rI1yUyVw crash log https://mega.nz/file/RXVEhRpZ 0r05xiqoGmL8rQEXjYaf8Q8BO-XbJGzpeBDek3aqb0w
  • Topics

×
×
  • Create New...

Important Information

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