HolonCanvasController.java 7.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268
  1. package ui.controller;
  2. import java.awt.Dimension;
  3. import java.util.ArrayList;
  4. import classes.AbstractCpsObject;
  5. import classes.Constants;
  6. import classes.CpsUpperNode;
  7. import classes.HolonBody;
  8. import classes.HolonElement;
  9. import classes.HolonObject;
  10. import classes.SubNet;
  11. import classes.Vector2d;
  12. import ui.model.Model;
  13. public class HolonCanvasController {
  14. // Ball objects
  15. private ArrayList<HolonBody> bodies = new ArrayList<>();
  16. private int subCount;
  17. private Dimension center;
  18. private Model model;
  19. private ArrayList<HolonBody> sortedSize = new ArrayList<>();
  20. private ArrayList<HolonBody> sortedDist = new ArrayList<>();
  21. private int toDrag;
  22. private boolean beingDragged;
  23. /**
  24. * Constructor.
  25. *
  26. * @param model
  27. * the Model
  28. * @param mp
  29. * the MultipurposeController
  30. */
  31. public HolonCanvasController(Model model) {
  32. this.model = model;
  33. }
  34. // sort the HolonBodys after size
  35. public void rearrangeAfterSize() {
  36. int j = 0;
  37. sortedSize.addAll(bodies);
  38. insertionSizeSort(sortedSize);
  39. sortedDist.addAll(bodies);
  40. ArrayList<Vector2d> pos = insertionDistSort(sortedDist);
  41. for (int i = 0; i < subCount; i++) {
  42. for (j = 0; j < subCount; j++) {
  43. if (sortedSize.get(subCount - 1 - i).getId() == bodies.get(j).getId()) {
  44. bodies.get(j).position = pos.get(i);
  45. break;
  46. }
  47. }
  48. }
  49. }
  50. // updates the bodies according to the changes of subnets
  51. public void addNewBodies(int subCount, ArrayList<SubNet> subnets, Dimension center, int HolonBodyScale) {
  52. this.subCount = subCount;
  53. this.center = center;
  54. // find correct color for existing HolonBodys
  55. ArrayList<HolonBody> newBodies = new ArrayList<>();
  56. for (int i = 0; i < subCount; i++) {
  57. for (int j = 0; j < bodies.size(); j++) {
  58. if (model.getSubNetColors().get(i) == bodies.get(j).getColor()) {
  59. bodies.get(j).setRadius((subnets.get(i).getObjects().size() * 5 + 10) * HolonBodyScale / 100);
  60. bodies.get(j).setMass((float) Math.pow((subnets.get(i).getObjects().size() + 1) * 5, 3));
  61. newBodies.add(bodies.get(j));
  62. newBodies.get(i).setId(i);
  63. break;
  64. }
  65. }
  66. }
  67. bodies = newBodies;
  68. // adding new HolonBodys
  69. for (int i = bodies.size(); i < subCount; i++) {
  70. float radius = (subnets.get(i).getObjects().size() * 5 + 10) * HolonBodyScale / 100;
  71. HolonBody temp = new HolonBody((float) (center.width + Math.pow(-1, i)), center.height + 1, radius,
  72. (float) Math.pow((subnets.get(i).getObjects().size() + 1) * 5, 3), model.getSubNetColors().get(i));
  73. temp.setId(i);
  74. bodies.add(temp);
  75. }
  76. }
  77. public int getActiveElements(ArrayList<AbstractCpsObject> objects) {
  78. int val = 0;
  79. for (AbstractCpsObject obj : objects) {
  80. if (obj instanceof HolonObject) {
  81. for (HolonElement ele : ((HolonObject) obj).getElements()) {
  82. if (ele.getActive()) {
  83. val += 1;
  84. }
  85. }
  86. } else if (obj instanceof CpsUpperNode) {
  87. val += getTotalProduction(((CpsUpperNode) obj).getNodes());
  88. }
  89. }
  90. return val;
  91. }
  92. public int getTotalProducers(ArrayList<AbstractCpsObject> objects) {
  93. float val = 0;
  94. int prod = 0;
  95. int tStep = model.getCurIteration();
  96. for (AbstractCpsObject obj : objects) {
  97. if (obj instanceof HolonObject) {
  98. for (HolonElement ele : ((HolonObject) obj).getElements()) {
  99. if (ele.getEnergyAt()[tStep] > 0 && ele.getActive()) {
  100. val += ele.getEnergyAt()[tStep] * ele.getAmount();
  101. }
  102. }
  103. if (val > 0)
  104. prod += 1;
  105. } else if (obj instanceof CpsUpperNode) {
  106. val += getTotalProduction(((CpsUpperNode) obj).getNodes());
  107. }
  108. }
  109. return prod;
  110. }
  111. public int getTotalElements(ArrayList<AbstractCpsObject> objects) {
  112. int val = 0;
  113. for (AbstractCpsObject obj : objects) {
  114. if (obj instanceof HolonObject) {
  115. val += ((HolonObject) obj).getElements().size();
  116. } else if (obj instanceof CpsUpperNode) {
  117. val += getTotalConsumption(((CpsUpperNode) obj).getNodes());
  118. }
  119. }
  120. return val;
  121. }
  122. public float getTotalConsumption(ArrayList<AbstractCpsObject> objects) {
  123. float val = 0;
  124. int tStep = model.getCurIteration();
  125. for (AbstractCpsObject obj : objects) {
  126. if (obj instanceof HolonObject) {
  127. for (HolonElement ele : ((HolonObject) obj).getElements()) {
  128. if (ele.getEnergyAt()[tStep] < 0 && ele.getActive()) {
  129. val += ele.getEnergyAt()[tStep] * ele.getAmount();
  130. }
  131. }
  132. } else if (obj instanceof CpsUpperNode) {
  133. val += getTotalConsumption(((CpsUpperNode) obj).getNodes());
  134. }
  135. }
  136. return val;
  137. }
  138. public float getTotalProduction(ArrayList<AbstractCpsObject> objects) {
  139. float val = 0;
  140. int tStep = model.getCurIteration();
  141. for (AbstractCpsObject obj : objects) {
  142. if (obj instanceof HolonObject) {
  143. for (HolonElement ele : ((HolonObject) obj).getElements()) {
  144. if (ele.getEnergyAt()[tStep] > 0 && ele.getActive()) {
  145. val += ele.getEnergyAt()[tStep] * ele.getAmount();
  146. }
  147. }
  148. } else if (obj instanceof CpsUpperNode) {
  149. val += getTotalProduction(((CpsUpperNode) obj).getNodes());
  150. }
  151. }
  152. return val;
  153. }
  154. public void updateBodies(float elapsedSeconds) {
  155. // step the position of movable objects based off their velocity/gravity
  156. // and elapsedTime
  157. for (int i = 0; i < subCount; i++) {
  158. if (toDrag != bodies.get(i).getId() || !beingDragged) {
  159. bodies.get(i).position.setX(
  160. (float) (bodies.get(i).position.getX() + (bodies.get(i).velocity.getX() * (elapsedSeconds))
  161. - ((bodies.get(i).position.getX() - center.getWidth()) / (50 + subCount))));
  162. bodies.get(i).position.setY(
  163. (float) (bodies.get(i).position.getY() + (bodies.get(i).velocity.getY() * (elapsedSeconds))
  164. - ((bodies.get(i).position.getY() - center.getHeight()) / (50 + subCount))));
  165. if (Math.abs(bodies.get(i).velocity.getX()) < Constants.epsilon)
  166. bodies.get(i).velocity.setX(0);
  167. if (Math.abs(bodies.get(i).velocity.getY()) < Constants.epsilon)
  168. bodies.get(i).velocity.setY(0);
  169. }
  170. }
  171. checkCollisions();
  172. }
  173. // Insertion sort for Sweep and Prune
  174. public void insertionSort(ArrayList<HolonBody> a) {
  175. for (int p = 1; p < subCount; p++) {
  176. Comparable<HolonBody> tmp = a.get(p);
  177. int j = p;
  178. for (; j > 0 && tmp.compareTo(a.get(j - 1)) < 0; j--)
  179. a.set(j, a.get(j - 1));
  180. a.set(j, (HolonBody) tmp);
  181. }
  182. }
  183. // Insertion sort for subnet size
  184. private void insertionSizeSort(ArrayList<HolonBody> a) {
  185. for (int p = 1; p < subCount; p++) {
  186. HolonBody tmp = a.get(p);
  187. int j = p;
  188. for (; j > 0 && tmp.compareSizeTo(a.get(j - 1)) < 0; j--)
  189. a.set(j, a.get(j - 1));
  190. a.set(j, (HolonBody) tmp);
  191. }
  192. }
  193. // Insertion sort for HolonBody distance
  194. private ArrayList<Vector2d> insertionDistSort(ArrayList<HolonBody> a) {
  195. ArrayList<Vector2d> pos = new ArrayList<>();
  196. for (int p = 1; p < subCount; p++) {
  197. HolonBody tmp = a.get(p);
  198. int j = p;
  199. for (; j > 0 && tmp.compareDistTo(a.get(j - 1), center) < 0; j--)
  200. a.set(j, a.get(j - 1));
  201. a.set(j, (HolonBody) tmp);
  202. }
  203. for (int i = 0; i < subCount; i++)
  204. pos.add(a.get(i).position);
  205. return pos;
  206. }
  207. public void checkCollisions() {
  208. insertionSort(bodies);
  209. for (int i = 0; i < subCount; i++) {
  210. // Ball to Ball collision
  211. for (int j = i + 1; j < subCount; j++) {
  212. if ((bodies.get(i).position.getX() + bodies.get(i).getRadius()) < (bodies.get(j).position.getX()
  213. - bodies.get(j).getRadius()))
  214. break;
  215. if ((bodies.get(i).position.getY() + bodies.get(i).getRadius()) < (bodies.get(j).position.getY()
  216. - bodies.get(j).getRadius())
  217. || (bodies.get(j).position.getY() + bodies.get(j).getRadius()) < (bodies.get(i).position.getY()
  218. - bodies.get(i).getRadius()))
  219. continue;
  220. bodies.get(i).resolveCollision(bodies.get(j));
  221. }
  222. }
  223. }
  224. public ArrayList<HolonBody> getBodies() {
  225. return bodies;
  226. }
  227. public void setDrag(boolean beingDragged) {
  228. this.beingDragged = beingDragged;
  229. }
  230. public void setBodyToDrag(int i) {
  231. this.toDrag = i;
  232. }
  233. }