FlexManager.java 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190
  1. package ui.controller;
  2. import java.util.ArrayList;
  3. import java.util.HashMap;
  4. import java.util.List;
  5. import java.util.stream.Collectors;
  6. import classes.AbstractCanvasObject;
  7. import classes.GroupNode;
  8. import classes.Flexibility;
  9. import classes.HolonElement;
  10. import classes.HolonObject;
  11. import ui.model.Model;
  12. /**
  13. * Class to Manage to flexibilities.
  14. * To order, view Flexibilities.
  15. * @author tom
  16. *
  17. */
  18. public class FlexManager {
  19. private int timeStep;
  20. private List<Flexibility> allFlexModels;
  21. private List<Flexibility> allFlexesOrderedThisTimeStep = new ArrayList<Flexibility>();
  22. private HashMap<HolonElement, List<FlexWrapper>> accessOtherFlex = new HashMap<HolonElement, List<FlexWrapper>>();
  23. private HashMap<Flexibility, FlexWrapper> accessFlexMap = new HashMap<Flexibility, FlexWrapper>();
  24. public FlexManager(Model model, int timeStep, FlexManager timestepBefore){
  25. //Thread.dumpStack();
  26. this.timeStep = timeStep;
  27. allFlexModels = getAllFlexFromModel(model);
  28. //fill accessFlexMap
  29. allFlexModels.stream().map(flex -> new FlexWrapper(flex, timeStep, (timestepBefore != null)?timestepBefore.getFlexWrapper(flex):null)).forEach(flexWrapper -> {
  30. accessFlexMap.put(flexWrapper.getFlex(), flexWrapper);
  31. HolonElement ele = flexWrapper.getFlex().getElement();
  32. if(accessOtherFlex.containsKey(ele)) {
  33. accessOtherFlex.get(ele).add(flexWrapper);
  34. }else {
  35. ArrayList<FlexWrapper> toAdd = new ArrayList<FlexWrapper>();
  36. toAdd.add(flexWrapper);
  37. accessOtherFlex.put(ele, new ArrayList<FlexWrapper>(toAdd));
  38. }
  39. });
  40. //because when added not all flexes can see others
  41. accessFlexMap.values().stream().forEach(flexWrapper -> flexWrapper.revalidateState());
  42. }
  43. private FlexWrapper getFlexWrapper(Flexibility flex) {
  44. return accessFlexMap.getOrDefault(flex, null);
  45. }
  46. public List<FlexWrapper> getAllFlexWrapper() {
  47. return accessFlexMap.values().stream().collect(Collectors.toList());
  48. }
  49. public List<FlexWrapper> getAllFlexWrapperWithState(FlexState state) {
  50. return accessFlexMap.values().stream().filter(flexWrapper -> (flexWrapper.getState() == state)).collect(Collectors.toList());
  51. }
  52. private List<Flexibility> getAllFlexFromModel(Model model) {
  53. return createListOfHolonObjects(model.getObjectsOnCanvas()).stream().flatMap(hObject -> hObject.getElements().stream()).flatMap(hElement -> hElement.flexList.stream()).collect(Collectors.toList());
  54. }
  55. private List<HolonObject> createListOfHolonObjects(List<AbstractCanvasObject> objectsOnCanvas) {
  56. List<HolonObject> list = new ArrayList<HolonObject>();
  57. for(AbstractCanvasObject aCps : objectsOnCanvas) {
  58. if(aCps instanceof HolonObject) list.add((HolonObject) aCps);
  59. else if(aCps instanceof GroupNode)list.addAll(createListOfHolonObjects(((GroupNode)aCps).getNodes()));
  60. }
  61. return list;
  62. }
  63. public int getTimeStep() {
  64. return timeStep;
  65. }
  66. public List<Flexibility> getAllFlexesOrderedThisTimeStep() {
  67. return allFlexesOrderedThisTimeStep;
  68. }
  69. public void orderFlexFromList(List<Flexibility> flexList) {
  70. flexList.stream().forEach(flex -> {
  71. FlexWrapper flexToOrder = accessFlexMap.get(flex);
  72. if(flexToOrder!=null)flexToOrder.order();
  73. });
  74. }
  75. public void orderFlex(Flexibility flex) {
  76. FlexWrapper flexToOrder = accessFlexMap.get(flex);
  77. if(flexToOrder!=null)flexToOrder.order();
  78. }
  79. public boolean isAFlexInUseOfHolonElement(HolonElement ele) {
  80. return ele.flexList.stream().filter(flex -> this.accessFlexMap.containsKey(flex)).anyMatch(flex -> (this.accessFlexMap.get(flex).getState() == FlexState.IN_USE));
  81. }
  82. /**
  83. * Or Return null
  84. * @param flex
  85. * @return
  86. */
  87. public FlexWrapper getFlexWrapperFromFlexibility(Flexibility flex) {
  88. return accessFlexMap.get(flex);
  89. }
  90. public static enum FlexState{
  91. IN_USE, ON_COOLDOWN, OFFERED, NOT_OFFERED, UNAVAILABLE
  92. }
  93. //Classes
  94. public class FlexWrapper{
  95. private Flexibility flex;
  96. private FlexState state;
  97. int timeStep;
  98. int durationEndTime = -1;
  99. int coolDownEndTime = -1;
  100. public FlexWrapper(Flexibility flex, int timeStep, FlexWrapper old) {
  101. this.flex = flex;
  102. this.timeStep = timeStep;
  103. if(old == null) {
  104. state = flex.fulfillsConstrains()?(flex.offered?FlexState.OFFERED:FlexState.NOT_OFFERED):FlexState.UNAVAILABLE;
  105. }else {
  106. durationEndTime = old.durationEndTime;
  107. coolDownEndTime = old.coolDownEndTime;
  108. revalidateState();
  109. }
  110. }
  111. public void revalidateState() {
  112. if(remainingTimeTillActivation() == 0) state = (flex.fulfillsConstrains() && !otherFlexInUseOrOnCooldown())?(flex.offered?FlexState.OFFERED:FlexState.NOT_OFFERED):FlexState.UNAVAILABLE;
  113. else if(remainingDuration()== 0) state = FlexState.ON_COOLDOWN;
  114. else state = FlexState.IN_USE;
  115. }
  116. public Flexibility getFlex() {
  117. return flex;
  118. }
  119. public FlexState getState() {
  120. return state;
  121. }
  122. public boolean canOrder() {
  123. return (state == FlexState.OFFERED) && //Right state
  124. !otherFlexInUseOrOnCooldown(); //No other flex of this ele in use
  125. }
  126. private boolean otherFlexInUseOrOnCooldown() {
  127. if(accessOtherFlex.get(this.getFlex().getElement()) == null) return false;
  128. return accessOtherFlex.get(this.getFlex().getElement()).stream().anyMatch(flexWrapper -> flexWrapper != this && (flexWrapper.getState() == FlexState.IN_USE || flexWrapper.getState() == FlexState.ON_COOLDOWN));
  129. }
  130. public boolean order() {
  131. if(canOrder()) {
  132. state=FlexState.IN_USE;
  133. allFlexesOrderedThisTimeStep.add(flex);
  134. durationEndTime = timeStep + flex.getDuration();
  135. coolDownEndTime = durationEndTime + flex.getCooldown();
  136. //accessFlexMap.values().stream().filter(flexWrapper -> (flexWrapper.getFlex().getElement() == flex.getElement() && flexWrapper != this)).forEach(otherFlex -> otherFlex.revalidateState());
  137. accessOtherFlex.get(this.getFlex().getElement()).stream().filter(flexWrapper -> (flexWrapper != this)).forEach(otherFlex -> otherFlex.revalidateState());
  138. return true;
  139. }
  140. return false;
  141. }
  142. public boolean cancel() {
  143. if(allFlexesOrderedThisTimeStep.contains(flex)) {
  144. state=FlexState.OFFERED;
  145. durationEndTime = -1;
  146. coolDownEndTime = -1;
  147. allFlexesOrderedThisTimeStep.remove(flex);
  148. accessOtherFlex.get(this.getFlex().getElement()).stream().filter(flexWrapper -> (flexWrapper != this)).forEach(otherFlex -> otherFlex.revalidateState());
  149. return true;
  150. }
  151. return false;
  152. }
  153. public int remainingTimeTillActivation() {
  154. return Math.max(0, coolDownEndTime - timeStep);
  155. }
  156. public int remainingDuration() {
  157. return Math.max(0, durationEndTime - timeStep);
  158. }
  159. }
  160. public void reset() {
  161. getAllFlexWrapper().forEach(flexWrapper -> flexWrapper.cancel());
  162. }
  163. }