All Downloads are FREE. Search and download functionalities are using the official Maven repository.

mmb.content.stn.network.STNNetworkProcessing Maven / Gradle / Ivy

Go to download

Dependency for the MultiMachineBuilder, a voxel game about building an industrial empire in a finite world. THIS RELEASE IS NOT PLAYABLE. To play the game, donwload from >ITCH.IO LINK HERE< or >GH releases link here<

The newest version!
/**
 * 
 */
package mmb.content.stn.network;

import java.util.Collections;
import java.util.Map.Entry;
import java.util.Set;

import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.node.ArrayNode;
import com.fasterxml.jackson.databind.node.ObjectNode;
import com.pploder.events.Event;

import it.unimi.dsi.fastutil.objects.Object2IntMap;
import mmb.NN;
import mmb.Nil;
import mmb.beans.Saver;
import mmb.content.ContentsBlocks;
import mmb.content.agro.AgroRecipeGroup.AgroProcessingRecipe;
import mmb.content.craft.CraftingRecipeGroup.CraftingRecipe;
import mmb.content.ditems.Stencil;
import mmb.content.stn.block.STNBaseMachine;
import mmb.engine.CatchingEvent;
import mmb.engine.block.Blocks;
import mmb.engine.debug.Debugger;
import mmb.engine.inv.storage.SetInventory;
import mmb.engine.item.ItemEntry;
import mmb.engine.json.JsonTool;
import mmb.engine.recipe.ItemLists;
import mmb.engine.recipe.RecipeOutput;
import monniasza.collects.Collects;
import monniasza.collects.Identifiable;
import monniasza.collects.indexar.Database;
import monniasza.collects.indexar.ManyToManyIndex;
import monniasza.collects.indexar.OneToOneIndex;
import monniasza.collects.selfset.HashSelfSet;
import monniasza.collects.selfset.SelfSet;

/**
 * The implementation of the Auto-Crafting component of the STN
 * TODO crafting
 * TODO procurement
 * @author oskar
 */
public class STNNetworkProcessing implements Saver{
	@NN private final Debugger debug = new Debugger("STN crafting and procurement");
	
	//Processing recipe
	//recipe counter (to avoid confusion)
	int rgrouprcounter = 0;
	/**
	 * A recipe tag used by STN machines
	 * @author oskar
	 */
	public class STNRGroupTag implements Identifiable, Saver{
		/** The icon shown in lists */
		@NN public ItemEntry icon = Blocks.blockVoid;
		/** The name of the crafting group */
		@NN public final String name;
		/**
		 * Creates a recipe tag
		 * @param icon the recipe tag's icon
		 * @param s name of the recipe tag
		 */
		STNRGroupTag(ItemEntry icon, String s) {
			this.icon = icon;
			this.name = s;
		}
		
		//Equality checks for data structures
		@Override
		public int hashCode() {
			final int prime = 31;
			int result = 1;
			result = prime * result + getEnclosingInstance().hashCode();
			result = prime * result + name.hashCode();
			return result;
		}
		@Override
		public boolean equals(@Nil Object obj) {
			if (this == obj)
				return true;
			if (obj == null)
				return false;
			if (getClass() != obj.getClass())
				return false;
			STNRGroupTag other = (STNRGroupTag) obj;
			if (!getEnclosingInstance().equals(other.getEnclosingInstance()))
				return false;
			return name.equals(other.name);
		}
		private STNNetworkProcessing getEnclosingInstance() {
			return STNNetworkProcessing.this;
		}
		@Override
		public String id() {
			return name;
		}
				
		//Recipe indexing
		/**
		 * A processing recipe.
		 * @author oskar
		 */
		public class STNPRecipe{
			/** The ingredients required to craft the recipe*/
			@NN public final RecipeOutput in;
			/** The results of this recipe */
			@NN public final RecipeOutput out;
			/** The recipe counter (unique ID) */
			public final int count;
			
			STNPRecipe(RecipeOutput in, RecipeOutput out, int count){
				this.in = in;
				this.out = out;
				this.count = count;
			}

			@Override
			public int hashCode() {
				final int prime = 31;
				int result = 1;
				result = prime * result + getEnclosingInstance().hashCode();
				result = prime * result + (in.hashCode());
				result = prime * result + (out.hashCode());
				return result;
			}

			@Override
			public boolean equals(@Nil Object obj) {
				if (this == obj)
					return true;
				if (obj == null)
					return false;
				if (getClass() != obj.getClass())
					return false;
				STNPRecipe other = (STNPRecipe) obj;
				if (!getEnclosingInstance().equals(other.getEnclosingInstance()))
					return false;
				if (!in.equals(other.in))
					return false;
				if (!out.equals(other.out))
					return false;
				return true;
			}

			private STNRGroupTag getEnclosingInstance() {
				return STNRGroupTag.this;
			}
		}
		/** Index of recipes by input, for this recipe tag*/
		@NN public final ManyToManyIndex inputIndex =
				new ManyToManyIndex<>(recipe -> recipe.in.items());
		/** Index of recipes by output, for this recipe tag*/
		@NN public final ManyToManyIndex outputIndex =
				new ManyToManyIndex<>(recipe -> recipe.out.items());
		/** Index of recipes by input, for this recipe tag*/
		@NN public final OneToOneIndex countIndex =
				new OneToOneIndex<>(recipe -> Integer.valueOf(recipe.count));
		@NN final Database recipes0 =
				new Database<>(STNPRecipe.class)
				.addIndex(inputIndex)
				.addIndex(outputIndex);
		/** The index of recipes for this tag*/
		@NN public final Set recipes = Collections.unmodifiableSet(recipes0);
		
		//Recipe manipulation
		/**
		 * Produces a processing recipe
		 * @param in incoming items
		 * @param out outgoing items
		 * @return a new recipe
		 */
		public STNPRecipe produceRecipe(RecipeOutput in, RecipeOutput out) {
			return produceRecipe(in, out, rgrouprcounter++);
		}
		//like its smaller cousin, but takes an ID for (de)serialization purposes
		STNPRecipe produceRecipe(RecipeOutput in, RecipeOutput out, int count) {
			STNPRecipe recipe = new STNPRecipe(in, out, count);
			recipeProduced.trigger(recipe);
			precipeProduced.trigger(recipe);
			recipes0.add(recipe);
			return recipe;
		}
		
		void kill() {
			//Kill all recipes
			for(STNPRecipe recipe: recipes) {
				precipeKilled.trigger(recipe);
				recipeKilled.trigger(recipe);
			}
		}
		
		//Events
		/** Invoked when a processing recipe is produced */
		@NN public final Event recipeProduced = new CatchingEvent<>(debug, "Failed to run a processing recipe produced event");
		/** Invoked when a processing recipe is killed */
		@NN public final Event recipeKilled = new CatchingEvent<>(debug, "Failed to run a processing recipe killed event");
		
		@Override
		public @Nil JsonNode save() {
			ObjectNode node = JsonTool.newObjectNode();
			
			//Save the icon
			JsonNode nodeIcon = ItemEntry.saveItem(icon);
			node.set("icon", nodeIcon);
			
			//Save recipes
			ObjectNode nodeRecipes = JsonTool.newObjectNode();
			for(STNPRecipe recipe: recipes) {
				int id = recipe.count;
				JsonNode nodeIn = ItemLists.save(recipe.in);
				JsonNode nodeOut = ItemLists.save(recipe.out);
				ArrayNode nodeArray = JsonTool.newArrayNode().add(nodeIn).add(nodeOut);
				nodeRecipes.set(Integer.toHexString(id), nodeArray);
			}
			node.set("recipes", nodeRecipes);
			
			return node;
		}

		@Override
		public void load(@Nil JsonNode data) {
			if(data == null) return;
			
			//Load the icon
			JsonNode nodeIcon = data.get("icon");
			icon = ItemEntry.loadFromJson(nodeIcon);
			
			//Load recipes
			JsonNode nodeRecipes = data.get("recipes");
			for(Entry recipe: Collects.iter(nodeRecipes.fields())) {
				String numberString = recipe.getKey();
				JsonNode nodeArray = recipe.getValue();
				
				JsonNode nodeIn = nodeArray.get(0);
				JsonNode nodeOut = nodeArray.get(1);
				
				RecipeOutput in = ItemLists.read(nodeIn);
				RecipeOutput out = ItemLists.read(nodeOut);
				int id = Integer.parseInt(numberString, 16);
				
				produceRecipe(in, out, id);
			}
		}
	}
	
	/**
	 * @param dataLayerSTN
	 */
	public STNNetworkProcessing(DataLayerSTN dataLayerSTN) {
		debug.id = "STN crafting and procurement @"+dataLayerSTN.id().getName();
	}

	//Crafting system
	void theProcessingMachineIsGone(STNBaseMachine m) {
		/*if(m instanceof STNProcessor) {
			
		}else if(m instanceof STNCrafter){
		
		}*/
	}
	
	//TODO procurer index
	
	
	//TODO crafter index
	
	//craft recipe index
	/** Index of stencil inputs */
	@NN public final ManyToManyIndex stencil2InIndex = new ManyToManyIndex<>(s -> s.in().items());
	/** Index of stencil outputs */
	@NN public final ManyToManyIndex stencil2OutIndex = new ManyToManyIndex<>(s -> s.out().items());
	@NN private final Database<@NN Stencil> stencils0 = new Database<>(Stencil.class).addIndex(stencil2OutIndex).addIndex(stencil2InIndex);
	/** 
	 * The inventory for stencils 
	 * @apiNote DO NOT BULK INSERT
	 */
	@NN public final SetInventory stencilinv = new SetInventory<>(stencils0, Stencil.class).setCapacity(Double.POSITIVE_INFINITY);
	/** The set of stencils */
	@NN public final Set stencils = Collections.unmodifiableSet(stencils0);
	
	//TODO processor index
	
	//process recipe index
	/** Index of recipes by input, for this world*/
	@NN public final ManyToManyIndex<@NN STNRGroupTag.STNPRecipe, ItemEntry> processRecipe2InIndex =
			new ManyToManyIndex<>(recipe -> recipe.in.items());
	/** Index of recipes by output, for this world*/
	@NN public final ManyToManyIndex<@NN STNRGroupTag.STNPRecipe, ItemEntry> processRecipe2OutIndex =
			new ManyToManyIndex<>(recipe -> recipe.out.items());
	/** Index of recipes by input, for this recipe tag*/
	@NN public final OneToOneIndex<@NN STNRGroupTag.STNPRecipe, Integer> processCountIndex =
			new OneToOneIndex<>(recipe -> Integer.valueOf(recipe.count));
	/** The full index of processing recipes */
	@NN private final Database<@NN STNRGroupTag.STNPRecipe> precipes0 =
			new Database<>(STNRGroupTag.STNPRecipe.class)
			.addIndex(processRecipe2InIndex)
			.addIndex(processRecipe2OutIndex)
			.addIndex(processCountIndex);
	/** The full index of recipes*/
	@NN public final Set precipes = Collections.unmodifiableSet(precipes0);
	
	//Process tag index
	@NN private final SelfSet processingTagsIndex0 = HashSelfSet.createNonnull(STNRGroupTag.class);
	/** The self-set of tags */
	@NN public final SelfSet processingTagsIndex = Collects.unmodifiableSelfSet(processingTagsIndex0);
	/**
	 * Produces a new tag
	 * @param icon item to be used as an icon
	 * @param name name of the tag
	 * @return a new tag
	 */
	public STNRGroupTag newTag(ItemEntry icon, String name) {
		STNRGroupTag tag = new STNRGroupTag(icon, name);
		processingTagsIndex0.add(tag); //index this tag
		tagProduced.trigger(tag);
		return tag;
	}
	/**
	 * Kills the tag (removes everything related to it)
	 * @param tag
	 */
	public void killTag(STNRGroupTag tag) {
		//Eliminate recipes from DB
		Set recipes2go = tag.recipes;
		precipes0.removeAll(recipes2go);
		tag.kill();
		tagKilled.trigger(tag);
	}
	
	//Events
	/** Invoked when tag is produced */
	@NN public final Event tagProduced = new CatchingEvent<>(debug, "Failed to run a tag produced event");
	/** Invoked when tag is killed */
	@NN public final Event tagKilled = new CatchingEvent<>(debug, "Failed to run a tag killed event");
	/** Invoked when a processing recipe is produced */
	@NN public final Event precipeProduced = new CatchingEvent<>(debug, "Failed to run a processing recipe produced event");
	/** Invoked when a processing recipe is killed */
	@NN public final Event precipeKilled = new CatchingEvent<>(debug, "Failed to run a processing recipe killed event");
	
	//Serialization
	@Override
	public JsonNode save() {
		ObjectNode node = JsonTool.newObjectNode();
		
		//Save processing recipes
		ObjectNode procrNode = JsonTool.newObjectNode();
		for(STNRGroupTag tag: processingTagsIndex) {
			//Save each tag individually
			JsonNode savedTag = tag.save();
			String tagId = tag.id();
			procrNode.set(tagId, savedTag);
		}
		node.set("recipesProcessing", procrNode);
		
		//Save crafting recipes
		JsonNode craftrNode = stencilinv.save();
		node.set("recipesCrafting", craftrNode);
		
		//TODO Save procurers
		
		return node;
	}

	@Override
	public void load(@Nil JsonNode data) {
		if(data == null) return;
		
		//Load processing recipes
		JsonNode procrNode = data.get("recipesProcessing");
		for(Entry entry: Collects.iter(procrNode.fields())) {
			String id = entry.getKey();
			JsonNode nodeTag = entry.getValue();
			STNRGroupTag ttag = newTag(ContentsBlocks.CHEST, id);
			ttag.load(nodeTag);
		}
		
		//Load crafting recipes
		JsonNode craftrNode = data.get("recipesCrafting");
		stencilinv.load(craftrNode);
		stencilinv.setCapacity(Double.POSITIVE_INFINITY);
		
		//Load procurers
		
	}
	
	/**
	 * Checks if the item is potentially producible
	 * (procurable, craftable or processable to)
	 * @param entry item to test
	 * @return does given item entry have any valid crafting/process/procurement?
	 */
	public boolean isEverProducible(ItemEntry entry) {
		//TODO procurement
		if(stencil2OutIndex.containsTo(entry)) return true;
		if(processRecipe2OutIndex.containsTo(entry)) return true;
		return false;
	}
	/**
	 * Checks if the item is potentially obtainable
	 * (the item is planned with a negative amount, stored in the inventory, or producible ({@link #isEverProducible(ItemEntry)})
	 * @param entry item to test
	 * @param inv remaining items in inventory map
	 * @param queue queue from which items may be withdrawn
	 * @return is the item either in the inventory or producible
	 */
	public boolean isEverObtainable(ItemEntry entry, Object2IntMap inv, Object2IntMap queue) {
		return queue.getOrDefault(entry, 0) < 0 || inv.getOrDefault(entry, 0) > 0 || isEverProducible(entry);
	}
	
	public boolean isAllObtainable(Set entries, Object2IntMap inv, Object2IntMap queue) {
		return Collects.isAll(entries, item -> isEverObtainable(item, inv, queue));
	}
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy