/* * JaamSim Discrete Event Simulation * Copyright (C) 2014 Ausenco Engineering Canada Inc. * Copyright (C) 2016 JaamSim Software Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package com.jaamsim.ProcessFlow; import com.jaamsim.Graphics.DisplayEntity; import com.jaamsim.Samples.SampleConstant; import com.jaamsim.Samples.SampleInput; import com.jaamsim.basicsim.Entity; import com.jaamsim.input.EntityInput; import com.jaamsim.input.Keyword; import com.jaamsim.input.Output; import com.jaamsim.states.StateEntity; import com.jaamsim.units.DimensionlessUnit; import com.jaamsim.units.TimeUnit; public class Pack extends LinkedService { @Keyword(description = "The prototype for EntityContainers to be generated.\n" + "The generated EntityContainers will be copies of this entity.", exampleList = {"EntityContainer1"}) protected final EntityInput<EntityContainer> prototypeEntityContainer; @Keyword(description = "The number of entities to pack into the container.", exampleList = {"2", "DiscreteDistribution1", "'1 + [TimeSeries1].PresentValue'"}) protected final SampleInput numberOfEntities; @Keyword(description = "The service time required to pack each entity in the container.", exampleList = { "3.0 h", "ExponentialDistribution1", "'1[s] + 0.5*[TimeSeries1].PresentValue'" }) private final SampleInput serviceTime; protected EntityContainer container; // the generated EntityContainer private int numberGenerated; // Number of EntityContainers generated so far private int numberInserted; // Number of entities inserted to the EntityContainer private int numberToInsert; // Number of entities to insert in the present EntityContainer private boolean startedPacking; // True if the packing process has already started private DisplayEntity packedEntity; // the entity being packed { prototypeEntityContainer = new EntityInput<>(EntityContainer.class, "PrototypeEntityContainer", "Key Inputs", null); prototypeEntityContainer.setRequired(true); this.addInput(prototypeEntityContainer); numberOfEntities = new SampleInput("NumberOfEntities", "Key Inputs", new SampleConstant(1.0)); numberOfEntities.setUnitType(DimensionlessUnit.class); numberOfEntities.setEntity(this); numberOfEntities.setValidRange(0, Double.POSITIVE_INFINITY); this.addInput(numberOfEntities); serviceTime = new SampleInput("ServiceTime", "Key Inputs", new SampleConstant(TimeUnit.class, 0.0)); serviceTime.setUnitType(TimeUnit.class); serviceTime.setEntity(this); serviceTime.setValidRange(0, Double.POSITIVE_INFINITY); this.addInput(serviceTime); } @Override public void earlyInit() { super.earlyInit(); container = null; numberGenerated = 0; numberInserted = 0; startedPacking = false; packedEntity = null; } protected EntityContainer getNextContainer() { numberGenerated++; EntityContainer proto = prototypeEntityContainer.getValue(); StringBuilder sb = new StringBuilder(); sb.append(this.getName()).append("_").append(numberGenerated); EntityContainer ret = Entity.fastCopy(proto, sb.toString()); ret.earlyInit(); return ret; } @Override protected boolean startProcessing(double simTime) { // If necessary, get a new container if (container == null) { container = this.getNextContainer(); numberInserted = 0; // Set the state for the container and its contents if (!stateAssignment.getValue().isEmpty()) container.setPresentState(stateAssignment.getValue()); // Position the container over the pack object this.moveToProcessPosition(container); } // Are there sufficient entities in the queue to start packing? if (!startedPacking) { Integer m = this.getNextMatchValue(simTime); numberToInsert = this.getNumberToInsert(simTime); if (waitQueue.getValue().getMatchCount(m) < numberToInsert) { return false; } startedPacking = true; this.setMatchValue(m); } // Select the next entity to pack and set its state if (numberInserted < numberToInsert) { packedEntity = this.getNextEntityForMatch(getMatchValue()); if (!stateAssignment.getValue().isEmpty() && packedEntity instanceof StateEntity) ((StateEntity)packedEntity).setPresentState(stateAssignment.getValue()); // Move the entity into position for processing this.moveToProcessPosition(packedEntity); } return true; } @Override protected boolean processStep(double simTime) { // Remove the next entity from the queue and pack the container if (packedEntity != null) { container.addEntity(packedEntity); packedEntity = null; numberInserted++; } // If the container is full, send it to the next component if (numberInserted >= numberToInsert) { this.sendToNextComponent(container); container = null; numberInserted = 0; startedPacking = false; } return true; } protected int getNumberToInsert(double simTime) { int ret = (int)numberOfEntities.getValue().getNextSample(simTime); ret = Math.max(ret, 1); return ret; } @Override protected double getStepDuration(double simTime) { return serviceTime.getValue().getNextSample(simTime); } @Output(name = "Container", description = "The EntityContainer that is being filled.") public DisplayEntity getContainer(double simTime) { return container; } }