HolonCanvas.java 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516
  1. package ui.view;
  2. import java.awt.Color;
  3. import java.awt.Dimension;
  4. import java.awt.Graphics;
  5. import java.awt.Graphics2D;
  6. import java.awt.RenderingHints;
  7. import java.awt.event.ActionEvent;
  8. import java.awt.event.ActionListener;
  9. import java.awt.event.MouseEvent;
  10. import java.awt.event.MouseListener;
  11. import java.awt.event.MouseMotionListener;
  12. import java.awt.event.MouseWheelEvent;
  13. import java.awt.event.MouseWheelListener;
  14. import java.util.ArrayList;
  15. import javax.swing.JCheckBox;
  16. import javax.swing.JComboBox;
  17. import javax.swing.JLabel;
  18. import javax.swing.JPanel;
  19. import javax.swing.event.ChangeEvent;
  20. import javax.swing.event.ChangeListener;
  21. import javax.swing.plaf.BorderUIResource.BevelBorderUIResource;
  22. import classes.AbstractCpsObject;
  23. import classes.Constants;
  24. import classes.CpsUpperNode;
  25. import classes.HolonBody;
  26. import classes.HolonElement;
  27. import classes.HolonObject;
  28. import classes.SubNet;
  29. import classes.Vector2d;
  30. import ui.controller.Control;
  31. import ui.model.Model;
  32. public class HolonCanvas extends JPanel implements MouseWheelListener, MouseListener, MouseMotionListener {
  33. /**
  34. *
  35. */
  36. private static final long serialVersionUID = 1L;
  37. // Rendering
  38. private Graphics2D g2;
  39. // Ball objects
  40. private ArrayList<HolonBody> bodies = new ArrayList<>();
  41. private int subCount;
  42. // Frames
  43. private int currentFrameRate;
  44. private long previousTime = System.currentTimeMillis();
  45. private long currentTime = previousTime;
  46. private long elapsedTime;
  47. private long totalElapsedTime = 0;
  48. private int frameCount = 0;
  49. private Dimension center;
  50. private ArrayList<SubNet> subnets;
  51. private Control controller;
  52. private Model model;
  53. private int maxX;
  54. private int maxY;
  55. private JComboBox<String> combo = new JComboBox<>();
  56. private int comboChoice = 0;
  57. private String info;
  58. private int bodyNr;
  59. private final JLabel lblBodyInfo = new JLabel("Holon Info:");
  60. private JCheckBox chckbxSort = new JCheckBox("sort by size");
  61. private ArrayList<HolonBody> sortedSize = new ArrayList<>();
  62. private ArrayList<HolonBody> sortedDist = new ArrayList<>();
  63. private boolean sizeChange = false;
  64. private HolonBody toDrag;
  65. private boolean beingDragged = false;
  66. public HolonCanvas(Model mod, Control control) {
  67. // Wire up Events
  68. this.controller = control;
  69. this.model = mod;
  70. add(lblBodyInfo);
  71. this.add(combo);
  72. subnets = controller.getSimManager().getSubNets();
  73. subCount = subnets.size();
  74. previousTime = System.currentTimeMillis();
  75. currentTime = previousTime;
  76. totalElapsedTime = 0;
  77. frameCount = 0;
  78. this.addMouseWheelListener(this);
  79. combo.addItem("ID");
  80. combo.addItem("Nr. of Objects");
  81. combo.addItem("Nr. of Edges");
  82. combo.addItem("Nr. of Switches");
  83. combo.addItem("Total Production");
  84. combo.addItem("Total Consumption");
  85. combo.addItem("Nr. of Elements");
  86. combo.addItem("Nr. of Producers");
  87. combo.addItem("Nr. of active Elements");
  88. combo.addActionListener(new ActionListener() {
  89. @Override
  90. public void actionPerformed(ActionEvent e) {
  91. comboChoice = combo.getSelectedIndex();
  92. }
  93. });
  94. this.addMouseListener(this);
  95. this.addMouseMotionListener(this);
  96. chckbxSort.addChangeListener(new ChangeListener() {
  97. @Override
  98. public void stateChanged(ChangeEvent e) {
  99. sizeChange = true;
  100. }
  101. });
  102. add(chckbxSort);
  103. }
  104. // Start Render and Update Threads
  105. public void paintComponent(Graphics g) {
  106. super.paintComponent(g);
  107. // add new colors if necessary
  108. for (int i = 0; i < controller.getSimManager().getSubNets().size(); i++) {
  109. if (model.getSubNetColors().size() - 1 < i) {
  110. controller.addSubNetColor(new Color((int) (Math.random() * 255), (int) (Math.random() * 255),
  111. (int) (Math.random() * 255)));
  112. }
  113. }
  114. // check if subnets have changed
  115. if (!controller.getSimManager().getSubNets().equals(subnets)) {
  116. subnets = controller.getSimManager().getSubNets();
  117. subCount = subnets.size();
  118. calcCenter();
  119. maxX = center.width;
  120. maxY = center.height;
  121. addNewBodies(subCount);
  122. sizeChange = true;
  123. }
  124. currentTime = System.currentTimeMillis();
  125. elapsedTime = (currentTime - previousTime); // elapsed time in seconds
  126. totalElapsedTime += elapsedTime;
  127. if (totalElapsedTime > 1000) {
  128. currentFrameRate = frameCount;
  129. frameCount = 0;
  130. totalElapsedTime = 0;
  131. }
  132. // check if HolonBodys should bes sorted after size
  133. if (chckbxSort.isSelected() && sizeChange) {
  134. rearrangeAfterSize();
  135. }
  136. updateBodies(elapsedTime / 1000f);
  137. render(g);
  138. try {
  139. // Thread.sleep(getFpsDelay(maxFrameRate));
  140. Thread.sleep(5);
  141. } catch (Exception e) {
  142. e.printStackTrace();
  143. }
  144. previousTime = currentTime;
  145. frameCount++;
  146. repaint();
  147. }
  148. // sort the HolonBodys after size
  149. private void rearrangeAfterSize() {
  150. sizeChange = false;
  151. HolonBody tmp = null;
  152. int j = 0;
  153. sortedSize.addAll(bodies);
  154. insertionSizeSort(sortedSize);
  155. sortedDist.addAll(bodies);
  156. ArrayList<Vector2d> pos = insertionDistSort(sortedDist);
  157. for (int i = 0; i < subCount; i++) {
  158. for (j = 0; j < subCount; j++) {
  159. if (sortedSize.get(subCount - 1 - i).getId() == bodies.get(j).getId()) {
  160. bodies.get(j).position = pos.get(i);
  161. break;
  162. }
  163. }
  164. }
  165. }
  166. // updates the bodies according to the changes of subnets
  167. private void addNewBodies(int subCount) {
  168. // find correct color for existing HolonBodys
  169. ArrayList<HolonBody> newBodies = new ArrayList<>();
  170. for (int i = 0; i < subCount; i++) {
  171. for (int j = 0; j < bodies.size(); j++) {
  172. if (model.getSubNetColors().get(i) == bodies.get(j).getColor()) {
  173. bodies.get(j).setRadius(
  174. (subnets.get(i).getObjects().size() * 5 + 10) * controller.getHolonBodyScale() / 100);
  175. bodies.get(j).setMass((float) Math.pow((subnets.get(i).getObjects().size() + 1) * 5, 3));
  176. newBodies.add(bodies.get(j));
  177. newBodies.get(i).setId(i);
  178. if (newBodies.get(i).position.getX() > maxX)
  179. maxX = (int) newBodies.get(i).position.getX();
  180. if (newBodies.get(i).position.getY() > maxY)
  181. maxY = (int) newBodies.get(i).position.getY();
  182. break;
  183. }
  184. }
  185. }
  186. bodies = newBodies;
  187. // adding new HolonBodys
  188. for (int i = bodies.size(); i < subCount; i++) {
  189. float radius = (subnets.get(i).getObjects().size() * 5 + 10) * controller.getHolonBodyScale() / 100;
  190. HolonBody temp = new HolonBody(maxX + 1, maxY + 1, radius,
  191. (float) Math.pow((subnets.get(i).getObjects().size() + 1) * 5, 3), model.getSubNetColors().get(i));
  192. temp.setId(i);
  193. bodies.add(temp);
  194. if (bodies.get(i).position.getX() > maxX)
  195. maxX = (int) newBodies.get(i).position.getX();
  196. if (bodies.get(i).position.getY() > maxY)
  197. maxY = (int) newBodies.get(i).position.getY();
  198. }
  199. }
  200. public void render(Graphics g) {
  201. this.g2 = (Graphics2D) g;
  202. this.g2.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
  203. // Render Background
  204. this.g2.setColor(Color.WHITE);
  205. this.g2.fillRect(0, 0, getWidth(), getHeight());
  206. // Render Game Objects
  207. for (int i = 0; i < subCount; i++) {
  208. bodyNr = bodies.get(i).getId();
  209. switch (comboChoice) {
  210. case 0:
  211. info = "" + bodyNr;
  212. break;
  213. case 1:
  214. info = "" + subnets.get(bodyNr).getObjects().size();
  215. break;
  216. case 2:
  217. info = "" + subnets.get(bodyNr).getEdges().size();
  218. break;
  219. case 3:
  220. info = "" + subnets.get(bodyNr).getSwitches().size();
  221. break;
  222. case 4:
  223. info = "" + getTotalProduction(new ArrayList<AbstractCpsObject>(subnets.get(bodyNr).getObjects()));
  224. break;
  225. case 5:
  226. info = "" + getTotalConsumption(new ArrayList<AbstractCpsObject>(subnets.get(bodyNr).getObjects()));
  227. break;
  228. case 6:
  229. info = "" + getTotalElements(new ArrayList<AbstractCpsObject>(subnets.get(bodyNr).getObjects()));
  230. break;
  231. case 7:
  232. info = "" + getTotalProducers(new ArrayList<AbstractCpsObject>(subnets.get(bodyNr).getObjects()));
  233. break;
  234. case 8:
  235. info = "" + getActiveElements(new ArrayList<AbstractCpsObject>(subnets.get(bodyNr).getObjects()));
  236. break;
  237. default:
  238. info = "" + bodyNr;
  239. break;
  240. }
  241. bodies.get(i).setRadius((subnets.get(bodies.get(i).getId()).getObjects().size() * 5 + 10)
  242. * controller.getHolonBodyScale() / 100);
  243. bodies.get(i).draw(this.g2, info);
  244. }
  245. }
  246. private int getActiveElements(ArrayList<AbstractCpsObject> objects) {
  247. int val = 0;
  248. for (AbstractCpsObject obj : objects) {
  249. if (obj instanceof HolonObject) {
  250. for (HolonElement ele : ((HolonObject) obj).getElements()) {
  251. if (ele.getActive()) {
  252. val += 1;
  253. }
  254. }
  255. } else if (obj instanceof CpsUpperNode) {
  256. val += getTotalProduction(((CpsUpperNode) obj).getNodes());
  257. }
  258. }
  259. return val;
  260. }
  261. private int getTotalProducers(ArrayList<AbstractCpsObject> objects) {
  262. float val = 0;
  263. int prod = 0;
  264. int tStep = model.getCurIteration();
  265. for (AbstractCpsObject obj : objects) {
  266. if (obj instanceof HolonObject) {
  267. for (HolonElement ele : ((HolonObject) obj).getElements()) {
  268. if (ele.getEnergyAt()[tStep] > 0 && ele.getActive()) {
  269. val += ele.getEnergyAt()[tStep] * ele.getAmount();
  270. }
  271. }
  272. if (val > 0)
  273. prod += 1;
  274. } else if (obj instanceof CpsUpperNode) {
  275. val += getTotalProduction(((CpsUpperNode) obj).getNodes());
  276. }
  277. }
  278. return prod;
  279. }
  280. private int getTotalElements(ArrayList<AbstractCpsObject> objects) {
  281. int val = 0;
  282. for (AbstractCpsObject obj : objects) {
  283. if (obj instanceof HolonObject) {
  284. val += ((HolonObject) obj).getElements().size();
  285. } else if (obj instanceof CpsUpperNode) {
  286. val += getTotalConsumption(((CpsUpperNode) obj).getNodes());
  287. }
  288. }
  289. return val;
  290. }
  291. private float getTotalConsumption(ArrayList<AbstractCpsObject> objects) {
  292. float val = 0;
  293. int tStep = model.getCurIteration();
  294. for (AbstractCpsObject obj : objects) {
  295. if (obj instanceof HolonObject) {
  296. for (HolonElement ele : ((HolonObject) obj).getElements()) {
  297. if (ele.getEnergyAt()[tStep] < 0 && ele.getActive()) {
  298. val += ele.getEnergyAt()[tStep] * ele.getAmount();
  299. }
  300. }
  301. } else if (obj instanceof CpsUpperNode) {
  302. val += getTotalConsumption(((CpsUpperNode) obj).getNodes());
  303. }
  304. }
  305. return val;
  306. }
  307. private float getTotalProduction(ArrayList<AbstractCpsObject> objects) {
  308. float val = 0;
  309. int tStep = model.getCurIteration();
  310. for (AbstractCpsObject obj : objects) {
  311. if (obj instanceof HolonObject) {
  312. for (HolonElement ele : ((HolonObject) obj).getElements()) {
  313. if (ele.getEnergyAt()[tStep] > 0 && ele.getActive()) {
  314. val += ele.getEnergyAt()[tStep] * ele.getAmount();
  315. }
  316. }
  317. } else if (obj instanceof CpsUpperNode) {
  318. val += getTotalProduction(((CpsUpperNode) obj).getNodes());
  319. }
  320. }
  321. return val;
  322. }
  323. public void updateBodies(float elapsedSeconds) {
  324. // step the position of movable objects based off their velocity/gravity
  325. // and elapsedTime
  326. calcCenter();
  327. for (int i = 0; i < subCount; i++) {
  328. if (!bodies.get(i).equals(toDrag) || (bodies.get(i).equals(toDrag)&& !beingDragged)) {
  329. bodies.get(i).position.setX(
  330. (float) (bodies.get(i).position.getX() + (bodies.get(i).velocity.getX() * (elapsedSeconds))
  331. - ((bodies.get(i).position.getX() - center.getWidth()) / (50 + subCount))));
  332. bodies.get(i).position.setY(
  333. (float) (bodies.get(i).position.getY() + (bodies.get(i).velocity.getY() * (elapsedSeconds))
  334. - ((bodies.get(i).position.getY() - center.getHeight()) / (50 + subCount))));
  335. if (Math.abs(bodies.get(i).velocity.getX()) < Constants.epsilon)
  336. bodies.get(i).velocity.setX(0);
  337. if (Math.abs(bodies.get(i).velocity.getY()) < Constants.epsilon)
  338. bodies.get(i).velocity.setY(0);
  339. }
  340. }
  341. checkCollisions();
  342. }
  343. // Insertion sort for Sweep and Prune
  344. public void insertionSort(ArrayList<HolonBody> a) {
  345. for (int p = 1; p < subCount; p++) {
  346. Comparable<HolonBody> tmp = a.get(p);
  347. int j = p;
  348. for (; j > 0 && tmp.compareTo(a.get(j - 1)) < 0; j--)
  349. a.set(j, a.get(j - 1));
  350. a.set(j, (HolonBody) tmp);
  351. }
  352. }
  353. // Insertion sort for subnet size
  354. private void insertionSizeSort(ArrayList<HolonBody> a) {
  355. for (int p = 1; p < subCount; p++) {
  356. HolonBody tmp = a.get(p);
  357. int j = p;
  358. for (; j > 0 && tmp.compareSizeTo(a.get(j - 1)) < 0; j--)
  359. a.set(j, a.get(j - 1));
  360. a.set(j, (HolonBody) tmp);
  361. }
  362. }
  363. // Insertion sort for HolonBody distance
  364. private ArrayList<Vector2d> insertionDistSort(ArrayList<HolonBody> a) {
  365. ArrayList<Vector2d> pos = new ArrayList<>();
  366. for (int p = 1; p < subCount; p++) {
  367. HolonBody tmp = a.get(p);
  368. int j = p;
  369. for (; j > 0 && tmp.compareDistTo(a.get(j - 1), center) < 0; j--)
  370. a.set(j, a.get(j - 1));
  371. a.set(j, (HolonBody) tmp);
  372. }
  373. for (int i = 0; i < subCount; i++)
  374. pos.add(a.get(i).position);
  375. return pos;
  376. }
  377. public void checkCollisions() {
  378. insertionSort(bodies);
  379. for (int i = 0; i < subCount; i++) {
  380. // Ball to Ball collision
  381. for (int j = i + 1; j < subCount; j++) {
  382. if ((bodies.get(i).position.getX() + bodies.get(i).getRadius()) < (bodies.get(j).position.getX()
  383. - bodies.get(j).getRadius()))
  384. break;
  385. if ((bodies.get(i).position.getY() + bodies.get(i).getRadius()) < (bodies.get(j).position.getY()
  386. - bodies.get(j).getRadius())
  387. || (bodies.get(j).position.getY() + bodies.get(j).getRadius()) < (bodies.get(i).position.getY()
  388. - bodies.get(i).getRadius()))
  389. continue;
  390. bodies.get(i).resolveCollision(bodies.get(j));
  391. }
  392. }
  393. }
  394. // calc the center of the canvas
  395. public void calcCenter() {
  396. center = this.getSize();
  397. center.height /= 2;
  398. center.width /= 2;
  399. }
  400. @Override
  401. public void mouseWheelMoved(MouseWheelEvent e) {
  402. controller.setHolonBodyScale(model.getHolonBodyScale() + (-e.getScrollAmount() * e.getWheelRotation()));
  403. }
  404. @Override
  405. public void mouseClicked(MouseEvent e) {
  406. // TODO Auto-generated method stub
  407. }
  408. @Override
  409. public void mouseEntered(MouseEvent e) {
  410. // TODO Auto-generated method stub
  411. }
  412. @Override
  413. public void mouseExited(MouseEvent e) {
  414. // TODO Auto-generated method stub
  415. }
  416. @Override
  417. public void mousePressed(MouseEvent e) {
  418. // Body Selection
  419. for (int i = 0; i < subCount; i++) {
  420. float cx = bodies.get(i).position.getX();
  421. float cy = bodies.get(i).position.getY();
  422. float cr = bodies.get(i).getRadius();
  423. if (e.getX() - cr <= cx && e.getY() - cr <= cy && e.getX() + cr >= cx && e.getY() + cr >= cy) {
  424. controller.addSelectedHolonBody(bodies.get(i).getId());
  425. toDrag = bodies.get(i);
  426. break;
  427. } else {
  428. controller.addSelectedHolonBody(-1);
  429. toDrag = null;
  430. }
  431. }
  432. }
  433. @Override
  434. public void mouseReleased(MouseEvent e) {
  435. beingDragged = false;
  436. }
  437. @Override
  438. public void mouseDragged(MouseEvent e) {
  439. beingDragged = true;
  440. if (toDrag != null) {
  441. toDrag.position = new Vector2d(e.getX(), e.getY());
  442. }
  443. repaint();
  444. }
  445. @Override
  446. public void mouseMoved(MouseEvent e) {
  447. // TODO Auto-generated method stub
  448. }
  449. }