HolonCanvas.java 15 KB

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