HolonCanvas.java 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377
  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.MouseWheelEvent;
  12. import java.awt.event.MouseWheelListener;
  13. import java.util.ArrayList;
  14. import javax.swing.JCheckBox;
  15. import javax.swing.JComboBox;
  16. import javax.swing.JLabel;
  17. import javax.swing.JPanel;
  18. import classes.Constants;
  19. import classes.HolonBody;
  20. import classes.SubNet;
  21. import classes.Vector2d;
  22. import ui.controller.Control;
  23. import ui.model.Model;
  24. public class HolonCanvas extends JPanel implements MouseWheelListener, MouseListener {
  25. /**
  26. *
  27. */
  28. private static final long serialVersionUID = 1L;
  29. // Rendering
  30. private Graphics2D g2;
  31. // Ball objects
  32. private ArrayList<HolonBody> bodies = new ArrayList<>();
  33. private int subCount;
  34. // Frames
  35. private int currentFrameRate;
  36. private long previousTime = System.currentTimeMillis();
  37. private long currentTime = previousTime;
  38. private long elapsedTime;
  39. private long totalElapsedTime = 0;
  40. private int frameCount = 0;
  41. private Dimension center;
  42. private ArrayList<SubNet> subnets;
  43. private Control controller;
  44. private Model model;
  45. private int maxX;
  46. private int maxY;
  47. private JComboBox<String> combo = new JComboBox<>();
  48. private int comboChoice = 0;
  49. private String info;
  50. private int bodyNr;
  51. private final JLabel lblBodyInfo = new JLabel("Holon Info:");
  52. private JCheckBox chckbxSort = new JCheckBox("sort by size");
  53. private ArrayList<HolonBody> sortedSize = new ArrayList<>();
  54. private ArrayList<HolonBody> sortedDist = new ArrayList<>();
  55. private boolean sizeChange = false;
  56. public HolonCanvas(Model mod, Control control) {
  57. // Wire up Events
  58. this.controller = control;
  59. this.model = mod;
  60. add(lblBodyInfo);
  61. this.add(combo);
  62. subnets = controller.getSimManager().getSubNets();
  63. subCount = subnets.size();
  64. previousTime = System.currentTimeMillis();
  65. currentTime = previousTime;
  66. totalElapsedTime = 0;
  67. frameCount = 0;
  68. this.addMouseWheelListener(this);
  69. combo.addItem("ID");
  70. combo.addItem("Nr. of Objects");
  71. combo.addItem("Nr. of Edges");
  72. combo.addItem("Nr. of Switches");
  73. combo.addActionListener(new ActionListener() {
  74. @Override
  75. public void actionPerformed(ActionEvent e) {
  76. comboChoice = combo.getSelectedIndex();
  77. }
  78. });
  79. this.addMouseListener(this);
  80. add(chckbxSort);
  81. }
  82. // Start Render and Update Threads
  83. public void paintComponent(Graphics g) {
  84. super.paintComponent(g);
  85. // add new colors if necessary
  86. for (int i = 0; i < controller.getSimManager().getSubNets().size(); i++) {
  87. if (model.getSubNetColors().size() - 1 < i) {
  88. controller.addSubNetColor(new Color((int) (Math.random() * 255), (int) (Math.random() * 255),
  89. (int) (Math.random() * 255)));
  90. }
  91. }
  92. // check if subnets have changed
  93. if (!controller.getSimManager().getSubNets().equals(subnets)) {
  94. subnets = controller.getSimManager().getSubNets();
  95. subCount = subnets.size();
  96. calcCenter();
  97. maxX = center.width;
  98. maxY = center.height;
  99. addNewBodies(subCount);
  100. sizeChange = true;
  101. }
  102. currentTime = System.currentTimeMillis();
  103. elapsedTime = (currentTime - previousTime); // elapsed time in seconds
  104. totalElapsedTime += elapsedTime;
  105. if (totalElapsedTime > 1000) {
  106. currentFrameRate = frameCount;
  107. frameCount = 0;
  108. totalElapsedTime = 0;
  109. }
  110. // check if HolonBodys should bes sorted after size
  111. if (chckbxSort.isSelected() && sizeChange) {
  112. rearrangeAfterSize();
  113. }
  114. updateBodies(elapsedTime / 1000f);
  115. render(g);
  116. try {
  117. // Thread.sleep(getFpsDelay(maxFrameRate));
  118. Thread.sleep(5);
  119. } catch (Exception e) {
  120. e.printStackTrace();
  121. }
  122. previousTime = currentTime;
  123. frameCount++;
  124. repaint();
  125. }
  126. // sort the HolonBodys after size
  127. private void rearrangeAfterSize() {
  128. sizeChange = false;
  129. HolonBody tmp = null;
  130. int j = 0;
  131. sortedSize.addAll(bodies);
  132. insertionSizeSort(sortedSize);
  133. sortedDist.addAll(bodies);
  134. ArrayList<Vector2d> pos = insertionDistSort(sortedDist);
  135. for (int i = 0; i < subCount; i++) {
  136. for (j = 0; j < subCount; j++) {
  137. if (sortedSize.get(subCount - 1 - i).getId() == bodies.get(j).getId()) {
  138. bodies.get(j).position = pos.get(i);
  139. break;
  140. }
  141. }
  142. }
  143. }
  144. // updates the bodies according to the changes of subnets
  145. private void addNewBodies(int subCount) {
  146. // find correct color for existing HolonBodys
  147. ArrayList<HolonBody> newBodies = new ArrayList<>();
  148. for (int i = 0; i < subCount; i++) {
  149. for (int j = 0; j < bodies.size(); j++) {
  150. if (model.getSubNetColors().get(i) == bodies.get(j).getColor()) {
  151. bodies.get(j).setRadius(
  152. (subnets.get(i).getObjects().size() * 5 + 10) * controller.getHolonBodyScale() / 100);
  153. bodies.get(j).setMass((float) Math.pow((subnets.get(i).getObjects().size() + 1) * 5, 3));
  154. newBodies.add(bodies.get(j));
  155. newBodies.get(i).setId(i);
  156. if (newBodies.get(i).position.getX() > maxX)
  157. maxX = (int) newBodies.get(i).position.getX();
  158. if (newBodies.get(i).position.getY() > maxY)
  159. maxY = (int) newBodies.get(i).position.getY();
  160. break;
  161. }
  162. }
  163. }
  164. bodies = newBodies;
  165. // adding new HolonBodys
  166. for (int i = bodies.size(); i < subCount; i++) {
  167. float radius = (subnets.get(i).getObjects().size() * 5 + 10) * controller.getHolonBodyScale() / 100;
  168. HolonBody temp = new HolonBody(maxX + 1, maxY + 1, radius,
  169. (float) Math.pow((subnets.get(i).getObjects().size() + 1) * 5, 3), model.getSubNetColors().get(i));
  170. temp.setId(i);
  171. bodies.add(temp);
  172. if (bodies.get(i).position.getX() > maxX)
  173. maxX = (int) newBodies.get(i).position.getX();
  174. if (bodies.get(i).position.getY() > maxY)
  175. maxY = (int) newBodies.get(i).position.getY();
  176. }
  177. }
  178. public void render(Graphics g) {
  179. this.g2 = (Graphics2D) g;
  180. this.g2.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
  181. // Render Background
  182. this.g2.setColor(Color.WHITE);
  183. this.g2.fillRect(0, 0, getWidth(), getHeight());
  184. // Render Game Objects
  185. for (int i = 0; i < subCount; i++) {
  186. bodyNr = bodies.get(i).getId();
  187. switch (comboChoice) {
  188. case 0:
  189. info = "" + bodyNr;
  190. break;
  191. case 1:
  192. info = "" + subnets.get(bodyNr).getObjects().size();
  193. break;
  194. case 2:
  195. info = "" + subnets.get(bodyNr).getEdges().size();
  196. break;
  197. case 3:
  198. info = "" + subnets.get(bodyNr).getSwitches().size();
  199. break;
  200. default:
  201. info = "" + bodies.get(i).getId();
  202. break;
  203. }
  204. bodies.get(i).setRadius((subnets.get(bodies.get(i).getId()).getObjects().size() * 5 + 10)
  205. * controller.getHolonBodyScale() / 100);
  206. bodies.get(i).draw(this.g2, info);
  207. }
  208. }
  209. public void updateBodies(float elapsedSeconds) {
  210. // step the position of movable objects based off their velocity/gravity
  211. // and elapsedTime
  212. calcCenter();
  213. for (int i = 0; i < subCount; i++) {
  214. bodies.get(i).position
  215. .setX((float) (bodies.get(i).position.getX() + (bodies.get(i).velocity.getX() * (elapsedSeconds))
  216. - ((bodies.get(i).position.getX() - center.getWidth()) / (50 + subCount))));
  217. bodies.get(i).position
  218. .setY((float) (bodies.get(i).position.getY() + (bodies.get(i).velocity.getY() * (elapsedSeconds))
  219. - ((bodies.get(i).position.getY() - center.getHeight()) / (50 + subCount))));
  220. if (Math.abs(bodies.get(i).velocity.getX()) < Constants.epsilon)
  221. bodies.get(i).velocity.setX(0);
  222. if (Math.abs(bodies.get(i).velocity.getY()) < Constants.epsilon)
  223. bodies.get(i).velocity.setY(0);
  224. }
  225. checkCollisions();
  226. }
  227. // Insertion sort for Sweep and Prune
  228. public void insertionSort(ArrayList<HolonBody> a) {
  229. for (int p = 1; p < subCount; p++) {
  230. Comparable<HolonBody> tmp = a.get(p);
  231. int j = p;
  232. for (; j > 0 && tmp.compareTo(a.get(j - 1)) < 0; j--)
  233. a.set(j, a.get(j - 1));
  234. a.set(j, (HolonBody) tmp);
  235. }
  236. }
  237. // Insertion sort for subnet size
  238. private void insertionSizeSort(ArrayList<HolonBody> a) {
  239. for (int p = 1; p < subCount; p++) {
  240. HolonBody tmp = a.get(p);
  241. int j = p;
  242. for (; j > 0 && tmp.compareSizeTo(a.get(j - 1)) < 0; j--)
  243. a.set(j, a.get(j - 1));
  244. a.set(j, (HolonBody) tmp);
  245. }
  246. }
  247. // Insertion sort for HolonBody distance
  248. private ArrayList<Vector2d> insertionDistSort(ArrayList<HolonBody> a) {
  249. ArrayList<Vector2d> pos = new ArrayList<>();
  250. for (int p = 1; p < subCount; p++) {
  251. HolonBody tmp = a.get(p);
  252. int j = p;
  253. for (; j > 0 && tmp.compareDistTo(a.get(j - 1), center) < 0; j--)
  254. a.set(j, a.get(j - 1));
  255. a.set(j, (HolonBody) tmp);
  256. }
  257. for (int i = 0; i < subCount; i++)
  258. pos.add(a.get(i).position);
  259. return pos;
  260. }
  261. public void checkCollisions() {
  262. insertionSort(bodies);
  263. for (int i = 0; i < subCount; i++) {
  264. // Ball to Ball collision
  265. for (int j = i + 1; j < subCount; j++) {
  266. if ((bodies.get(i).position.getX() + bodies.get(i).getRadius()) < (bodies.get(j).position.getX()
  267. - bodies.get(j).getRadius()))
  268. break;
  269. if ((bodies.get(i).position.getY() + bodies.get(i).getRadius()) < (bodies.get(j).position.getY()
  270. - bodies.get(j).getRadius())
  271. || (bodies.get(j).position.getY() + bodies.get(j).getRadius()) < (bodies.get(i).position.getY()
  272. - bodies.get(i).getRadius()))
  273. continue;
  274. bodies.get(i).resolveCollision(bodies.get(j));
  275. }
  276. }
  277. }
  278. // calc the center of the canvas
  279. public void calcCenter() {
  280. center = this.getSize();
  281. center.height /= 2;
  282. center.width /= 2;
  283. }
  284. @Override
  285. public void mouseWheelMoved(MouseWheelEvent e) {
  286. controller.setHolonBodyScale(model.getHolonBodyScale() + (-e.getScrollAmount() * e.getWheelRotation()));
  287. }
  288. @Override
  289. public void mouseClicked(MouseEvent e) {
  290. // TODO Auto-generated method stub
  291. }
  292. @Override
  293. public void mouseEntered(MouseEvent e) {
  294. // TODO Auto-generated method stub
  295. }
  296. @Override
  297. public void mouseExited(MouseEvent e) {
  298. // TODO Auto-generated method stub
  299. }
  300. @Override
  301. public void mousePressed(MouseEvent e) {
  302. // Body Selection
  303. for (int i = 0; i < subCount; i++) {
  304. float cx = bodies.get(i).position.getX();
  305. float cy = bodies.get(i).position.getY();
  306. float cr = bodies.get(i).getRadius();
  307. if (e.getX() - cr <= cx && e.getY() - cr <= cy && e.getX() + cr >= cx && e.getY() + cr >= cy) {
  308. controller.addSelectedHolonBody(bodies.get(i).getId());
  309. break;
  310. } else {
  311. controller.addSelectedHolonBody(-1);
  312. }
  313. }
  314. }
  315. @Override
  316. public void mouseReleased(MouseEvent e) {
  317. // TODO Auto-generated method stub
  318. }
  319. }