SimulationManager.java 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593
  1. package ui.controller;
  2. import classes.*;
  3. import ui.model.Model;
  4. import ui.view.FlexiblePane;
  5. import ui.view.MyCanvas;
  6. import java.util.ArrayList;
  7. import java.util.HashMap;
  8. /**
  9. * Controller for Simulation.
  10. *
  11. * @author Gruppe14
  12. */
  13. public class SimulationManager {
  14. private Model model;
  15. private ArrayList<AbstractCpsObject> objectsToHandle;
  16. // private ArrayList<CpsEdge> allConnections;
  17. private ArrayList<SubNet> subNets;
  18. private ArrayList<CpsEdge> brokenEdges;
  19. private MyCanvas canvas;
  20. private int timeStep;
  21. private HashMap<Integer, Float> tagTable = new HashMap<>();
  22. private FlexiblePane flexPane;
  23. /**
  24. * Constructor.
  25. *
  26. * @param m Model
  27. */
  28. public SimulationManager(Model m) {
  29. canvas = null;
  30. model = m;
  31. subNets = new ArrayList<>();
  32. brokenEdges = new ArrayList<>();
  33. }
  34. /**
  35. * calculates the flow of the edges and the supply for objects.
  36. *
  37. * @param x current Iteration
  38. */
  39. public void calculateStateForTimeStep(int x) {
  40. reset();
  41. timeStep = x;
  42. searchForSubNets();
  43. for (SubNet singleSubNet : subNets) {
  44. resetConnections(singleSubNet.getObjects().get(0), new ArrayList<>(), new ArrayList<>());
  45. }
  46. for (SubNet singleSubNet : subNets) {
  47. float production = calculateEnergy("prod", singleSubNet, timeStep);
  48. float consumption = calculateEnergy("cons", singleSubNet, timeStep);
  49. float wastedEnergy = production - consumption;
  50. float minConsumption = calculateMinimumEnergy(singleSubNet, timeStep);
  51. setFlowSimulation(singleSubNet);
  52. for (HolonObject hl : singleSubNet.getObjects()) {
  53. if (!(hl.getState() == 0) && !(hl.getState() == 3)) {
  54. for (int i = 0; i < hl.getConnections().size(); i++) {
  55. CpsEdge edge = hl.getConnectedTo().get(i);
  56. if (edge.isWorking() && edge.getFlow() > 0
  57. || edge.getCapacity() == CpsEdge.CAPACITY_INFINITE) {
  58. // 0 = no energy, 1 = not supplied, 2 = supplied, 3
  59. // = producer, 4 = partially supplied
  60. if ((production + consumption) >= 0) {
  61. hl.setState(HolonObject.SUPPLIED);
  62. }
  63. if ((production + consumption) < 0) {
  64. if ((production + minConsumption) >= 0) {
  65. hl.setState(HolonObject.PARTIALLY_SUPPLIED);
  66. } else if (hl.checkIfPartiallySupplied(timeStep)) {
  67. hl.setState(HolonObject.PARTIALLY_SUPPLIED);
  68. } else {
  69. hl.setState(HolonObject.NOT_SUPPLIED);
  70. }
  71. }
  72. break;
  73. }
  74. hl.setState(HolonObject.NOT_SUPPLIED);
  75. }
  76. if (hl.checkIfPartiallySupplied(timeStep) && !(hl.getState() == 2)) {
  77. hl.setState(HolonObject.PARTIALLY_SUPPLIED);
  78. }
  79. }
  80. }
  81. }
  82. canvas.repaint();
  83. // printNetToConsole();
  84. flexPane.recalculate();
  85. }
  86. /**
  87. * Set Flow Simulation.
  88. *
  89. * @param sN Subnet
  90. */
  91. public void setFlowSimulation(SubNet sN) {
  92. ArrayList<AbstractCpsObject> producers = new ArrayList<>();
  93. AbstractCpsObject tmp = null;
  94. tagTable = new HashMap<>();
  95. // traverse all objects in this subnet
  96. for (HolonObject hl : sN.getObjects()) {
  97. float energy = hl.getCurrentEnergyAtTimeStep(timeStep);
  98. // if their production is higher than their consumption
  99. if (energy > 0) {
  100. tagTable.put(hl.getId(), energy);
  101. hl.addTag(hl.getId());
  102. for (CpsEdge edge : hl.getConnections()) {
  103. if (edge.isWorking()) {
  104. // set other end of edge as tmp-object
  105. // and add this end to the other end's tag-list
  106. AbstractCpsObject a = edge.getA();
  107. AbstractCpsObject b = edge.getB();
  108. if (a.getId() == hl.getId()) {
  109. b.addTag(hl.getId());
  110. tmp = b;
  111. }
  112. if (b.getId() == hl.getId()) {
  113. a.addTag(hl.getId());
  114. tmp = a;
  115. }
  116. edge.setFlow(edge.getFlow() + energy);
  117. edge.calculateState();
  118. edge.addTag(hl.getId());
  119. if (edge.isWorking() && !producers.contains(tmp)) {
  120. if (tmp instanceof HolonSwitch) {
  121. if (((HolonSwitch) tmp).getState(timeStep)) {
  122. producers.add(tmp);
  123. }
  124. } else if (!(tmp instanceof CpsUpperNode)) {
  125. producers.add(tmp);
  126. }
  127. }
  128. }
  129. }
  130. }
  131. }
  132. setFlowSimRec(producers, 0);
  133. }
  134. /**
  135. * Set Flow Simulation Rec.
  136. *
  137. * @param nodes the nodes
  138. * @param iter the Iteration
  139. */
  140. public void setFlowSimRec(ArrayList<AbstractCpsObject> nodes, int iter) {
  141. ArrayList<AbstractCpsObject> newNodes = new ArrayList<>();
  142. ArrayList<CpsEdge> changedEdges = new ArrayList<>();
  143. AbstractCpsObject tmp;
  144. if (nodes.size() != 0) {
  145. for (AbstractCpsObject cps : nodes) {
  146. // check whether the cps is in a legit state if it is a switch
  147. if (legitState(cps)) {
  148. for (CpsEdge edge : cps.getConnections()) {
  149. // is edge working
  150. // and does the edge's tag-list not (yet) contain all tags of the cps
  151. if (edge.isWorking()
  152. && (!(edge.containsTags(edge.getTags(), cps.getTag())))) {
  153. if (edge.getA().getId() == cps.getId()) {
  154. tmp = edge.getB();
  155. } else {
  156. tmp = edge.getA();
  157. }
  158. for (Integer tag : cps.getTag()) {
  159. if (!(edge.getTags().contains(tag))
  160. && !(edge.getPseudoTags().contains(tag))) {
  161. edge.setFlow(edge.getFlow() + tagTable.get(tag));
  162. edge.addTag(tag);
  163. }
  164. }
  165. // uppernodes do not spread energy
  166. if (!(tmp instanceof CpsUpperNode)) {
  167. for (Integer tag : tmp.getTag()) {
  168. if (!(edge.getTags().contains(tag))
  169. && tagTable.get(tag) != null
  170. && !(edge.getPseudoTags().contains(tag))) {
  171. edge.setFlow(edge.getFlow() + tagTable.get(tag));
  172. edge.addPseudoTag(tag);
  173. changedEdges.add(edge);
  174. }
  175. }
  176. }
  177. edge.calculateState();
  178. if (edge.isWorking()
  179. && !(tmp instanceof CpsUpperNode)) {
  180. tmp.addAllPseudoTags(cps.getTag());
  181. if (!newNodes.contains(tmp)) {
  182. newNodes.add(tmp);
  183. }
  184. }
  185. }
  186. }
  187. }
  188. }
  189. setPseudoTags(newNodes, changedEdges);
  190. setFlowSimRec(newNodes, iter + 1);
  191. }
  192. }
  193. /**
  194. * Set the Pseudo Tags.
  195. *
  196. * @param nodes Array of AbstractCpsObjects
  197. */
  198. public void setPseudoTags(ArrayList<AbstractCpsObject> nodes, ArrayList<CpsEdge> edges) {
  199. for (AbstractCpsObject node : nodes) {
  200. node.recalculateTags();
  201. node.setPseudoTags(new ArrayList<>());
  202. }
  203. for (CpsEdge edge : edges) {
  204. edge.recalculateTags();
  205. edge.setPseudoTag(new ArrayList<>());
  206. }
  207. }
  208. // /**
  209. // * Print the nodes.
  210. // *
  211. // * @param nodes
  212. // * Array of AbstractCpsObject
  213. // */
  214. // public void printNodes(ArrayList<AbstractCpsObject> nodes) {
  215. // for (AbstractCpsObject node : nodes) {
  216. // System.out.println(node.getId());
  217. // }
  218. // }
  219. // /**
  220. // * Get String.
  221. // *
  222. // * @param tags
  223. // * the tags
  224. // * @return the String
  225. // */
  226. // public String getString(ArrayList<Integer> tags) {
  227. // String result = "";
  228. // for (Integer i : tags) {
  229. // result = result + ", " + i;
  230. // }
  231. // return result;
  232. // }
  233. // /**
  234. // * Merge the Lists.
  235. // *
  236. // * @param a
  237. // * first liust
  238. // * @param b
  239. // * second list
  240. // * @return the Result
  241. // */
  242. // public ArrayList<Integer> mergeLists(ArrayList<Integer> a, ArrayList<Integer> b) {
  243. // ArrayList<Integer> result = new ArrayList<>();
  244. // for (Integer i : a) {
  245. // if (!(result.contains(i))) {
  246. // result.add(i);
  247. // }
  248. // }
  249. // for (Integer j : b) {
  250. // if (!(result.contains(j))) {
  251. // result.add(j);
  252. // }
  253. // }
  254. // return result;
  255. // }
  256. /**
  257. * Reset the Connection.
  258. *
  259. * @param cps CpsObject
  260. * @param visitedObj the visited Objects
  261. * @param visitedEdges the visited Edges
  262. */
  263. public void resetConnections(AbstractCpsObject cps, ArrayList<Integer> visitedObj,
  264. ArrayList<CpsEdge> visitedEdges) {
  265. visitedObj.add(cps.getId());
  266. cps.resetTags();
  267. for (CpsEdge e : cps.getConnections()) {
  268. if (!(visitedEdges.contains(e))) {
  269. e.setFlow(0);
  270. e.calculateState();
  271. e.setTags(new ArrayList<>());
  272. visitedEdges.add(e);
  273. if (!(visitedObj.contains(e.getA().getId()))) {
  274. resetConnections(e.getA(), visitedObj, visitedEdges);
  275. e.getA().resetTags();
  276. }
  277. if (!(visitedObj.contains(e.getB().getId()))) {
  278. resetConnections(e.getB(), visitedObj, visitedEdges);
  279. e.getB().resetTags();
  280. }
  281. }
  282. }
  283. }
  284. /**
  285. * calculates the energy of either all producers or consumers.
  286. *
  287. * @param type Type
  288. * @param sN Subnet
  289. * @param x Integer
  290. * @return The Energy
  291. */
  292. public float calculateEnergy(String type, SubNet sN, int x) {
  293. float energy = 0;
  294. for (HolonObject hl : sN.getObjects()) {
  295. if (type.equals("prod")) {
  296. if (hl.getCurrentEnergyAtTimeStep(x) > 0) {
  297. energy = energy + hl.getCurrentEnergyAtTimeStep(x);
  298. hl.setState(HolonObject.PRODUCER);
  299. }
  300. }
  301. if (type.equals("cons")) {
  302. if (hl.getCurrentEnergyAtTimeStep(x) < 0) {
  303. energy = energy + hl.getCurrentEnergyAtTimeStep(x);
  304. hl.setState(HolonObject.NOT_SUPPLIED);
  305. }
  306. }
  307. if (hl.getCurrentEnergyAtTimeStep(x) == 0) {
  308. hl.setState(HolonObject.NO_ENERGY);
  309. }
  310. }
  311. return energy;
  312. }
  313. /**
  314. * Calculate the Minimum Energy.
  315. *
  316. * @param sN Subnet
  317. * @param x Integer
  318. * @return the Calculated minimum Energy
  319. */
  320. public float calculateMinimumEnergy(SubNet sN, int x) {
  321. float min = 0;
  322. float minElement = 0;
  323. for (HolonObject hl : sN.getObjects()) {
  324. if (hl.getElements().size() > 0 && hl.getElements().get(0).getTotalEnergyAtTimeStep(x) < 0) {
  325. minElement = hl.getElements().get(0).getTotalEnergyAtTimeStep(x);
  326. }
  327. for (HolonElement he : hl.getElements()) {
  328. if (minElement < he.getTotalEnergyAtTimeStep(x) && he.getTotalEnergyAtTimeStep(x) < 0) {
  329. minElement = he.getTotalEnergyAtTimeStep(x);
  330. }
  331. }
  332. min = min + minElement;
  333. }
  334. return min;
  335. }
  336. /**
  337. * generates all subNets from all objectsToHandle.
  338. */
  339. public void searchForSubNets() {
  340. subNets = new ArrayList<>();
  341. brokenEdges.clear();
  342. boolean end = false;
  343. int i = 0;
  344. AbstractCpsObject cps;
  345. if (objectsToHandle.size() > 0) {
  346. while (!end) {
  347. cps = objectsToHandle.get(i);
  348. SubNet singleSubNet = new SubNet(new ArrayList<>(), new ArrayList<>(),
  349. new ArrayList<>());
  350. singleSubNet = buildSubNet(cps, new ArrayList<>(), singleSubNet);
  351. if (singleSubNet.getObjects().size() != 0) {
  352. subNets.add(singleSubNet);
  353. }
  354. if (0 == objectsToHandle.size()) {
  355. end = true;
  356. }
  357. }
  358. }
  359. }
  360. /**
  361. * recursivly generates a subnet of all objects, that one specific object is
  362. * connected to.
  363. *
  364. * @param cps AbstractCpsObject
  365. * @param visited visited Array of Integer
  366. * @param sN Subnets
  367. * @return Subnet
  368. */
  369. public SubNet buildSubNet(AbstractCpsObject cps, ArrayList<Integer> visited, SubNet sN) {
  370. visited.add(cps.getId());
  371. if (cps instanceof HolonObject) {
  372. sN.getObjects().add((HolonObject) cps);
  373. }
  374. if (cps instanceof HolonSwitch) {
  375. sN.getSwitches().add((HolonSwitch) cps);
  376. }
  377. removeFromToHandle(cps.getId());
  378. AbstractCpsObject a;
  379. AbstractCpsObject b;
  380. for (CpsEdge edge : cps.getConnections()) {
  381. if (edge.isWorking()) {
  382. a = edge.getA();
  383. b = edge.getB();
  384. if (!(cps instanceof HolonSwitch)) {
  385. if (!(sN.getEdges().contains(edge))) {
  386. sN.getEdges().add(edge);
  387. }
  388. }
  389. if (cps instanceof HolonSwitch && ((HolonSwitch) cps).getState(timeStep)) {
  390. if (!(sN.getEdges().contains(edge))) {
  391. sN.getEdges().add(edge);
  392. }
  393. }
  394. if (!visited.contains(a.getId()) && legitState(cps) && !(a instanceof CpsUpperNode)) {
  395. sN = buildSubNet(a, visited, sN);
  396. }
  397. if (!visited.contains(b.getId()) && legitState(cps) && !(b instanceof CpsUpperNode)) {
  398. sN = buildSubNet(b, visited, sN);
  399. }
  400. if (a instanceof CpsUpperNode && a.getId() != cps.getId()) {
  401. edge.setConnected(CpsEdge.CON_UPPER_NODE_NOT_INSIDE);
  402. checkForConnectedStates(b, (CpsUpperNode) a, edge);
  403. }
  404. if (b instanceof CpsUpperNode && b.getId() != cps.getId()) {
  405. edge.setConnected(CpsEdge.CON_UPPER_NODE_NOT_INSIDE);
  406. checkForConnectedStates(a, (CpsUpperNode) b, edge);
  407. }
  408. } else {
  409. brokenEdges.add(edge);
  410. }
  411. }
  412. return sN;
  413. }
  414. /**
  415. * is the Switch in a legitimate State.
  416. *
  417. * @param current AbstractCpsObject
  418. * @return boolean
  419. */
  420. public boolean legitState(AbstractCpsObject current) {
  421. if (current instanceof HolonSwitch) {
  422. return ((HolonSwitch) current).getState(timeStep);
  423. }
  424. return true;
  425. }
  426. /**
  427. * removes an Object that already has been handled with.
  428. *
  429. * @param id the Object ID
  430. */
  431. public void removeFromToHandle(int id) {
  432. for (int i = 0; i < objectsToHandle.size(); i++) {
  433. if (objectsToHandle.get(i).getId() == id) {
  434. objectsToHandle.remove(i);
  435. }
  436. }
  437. }
  438. // /**
  439. // * ensures that objectsToHandle only contains HolonObjects.
  440. // */
  441. // public void cleanObjectsToHandle() {
  442. // for (int i = 0; i < objectsToHandle.size(); i++) {
  443. // if (!(objectsToHandle.get(i) instanceof HolonObject)) {
  444. // objectsToHandle.remove(i);
  445. // }
  446. // }
  447. // }
  448. /**
  449. * copies the data of an array of Objects.
  450. *
  451. * @param toCopy the ArrayList of CpsObjects co Copy
  452. */
  453. public void copyObjects(ArrayList<AbstractCpsObject> toCopy) {
  454. for (AbstractCpsObject cps : toCopy) {
  455. if (cps instanceof CpsUpperNode) {
  456. copyObjects(((CpsUpperNode) cps).getNodes());
  457. } else {
  458. objectsToHandle.add(cps);
  459. }
  460. }
  461. }
  462. /**
  463. * Prints the Components auf all subnets.
  464. */
  465. public void printNetToConsole() {
  466. for (int i = 0; i < subNets.size(); i++) {
  467. System.out.println("SUBNET NR:" + i);
  468. System.out.println(" Objects:");
  469. for (int j = 0; j < subNets.get(i).getObjects().size(); j++) {
  470. HolonObject hl = subNets.get(i).getObjects().get(j);
  471. System.out.println(" " + hl.getName() + " " + hl.getId());
  472. }
  473. System.out.println(" Edges:");
  474. for (int j = 0; j < subNets.get(i).getEdges().size(); j++) {
  475. CpsEdge edge = subNets.get(i).getEdges().get(j);
  476. System.out.println(" " + edge.getA().getName() + " connected To " + edge.getB().getName());
  477. }
  478. System.out.println(" Switches:");
  479. for (int j = 0; j < subNets.get(i).getSwitches().size(); j++) {
  480. HolonSwitch sw = subNets.get(i).getSwitches().get(j);
  481. System.out.println(" " + sw.getName() + " " + sw.getId() + " State:" + sw.getActiveAt()[timeStep]);
  482. }
  483. }
  484. }
  485. /**
  486. * Set the Canvas.
  487. *
  488. * @param can the Canvas
  489. */
  490. public void setCanvas(MyCanvas can) {
  491. canvas = can;
  492. }
  493. /**
  494. * Reset all Data to the current state of the Model.
  495. */
  496. public void reset() {
  497. objectsToHandle = new ArrayList<>();
  498. copyObjects(model.getObjectsOnCanvas());
  499. }
  500. /**
  501. * Resets the State of all Edges
  502. */
  503. public void resetEdges() {
  504. for (CpsEdge e : brokenEdges) {
  505. e.setWorkingState(true);
  506. }
  507. }
  508. /**
  509. * Resets the State for the whole Simulation Model
  510. */
  511. public void resetSimulation() {
  512. reset();
  513. resetEdges();
  514. }
  515. /**
  516. * Get all Subnets.
  517. *
  518. * @return all Subnets
  519. */
  520. public ArrayList<SubNet> getSubNets() {
  521. return subNets;
  522. }
  523. /**
  524. * Get broken Edges
  525. */
  526. public ArrayList<CpsEdge> getBrokenEdges() {
  527. return brokenEdges;
  528. }
  529. /**
  530. * checks whether a given object is connected to an object inside the upperNode.
  531. * if yes, the state for the edge is changed in "connected" or "not connected"
  532. *
  533. * @param cps
  534. * @param cUNode
  535. */
  536. public void checkForConnectedStates(AbstractCpsObject cps, CpsUpperNode cUNode, CpsEdge theEdge) {
  537. AbstractCpsObject tmp;
  538. for (CpsEdge edge : cps.getConnections()) {
  539. if (edge.getA().getId() == cps.getId()) {
  540. tmp = edge.getB();
  541. } else {
  542. tmp = edge.getA();
  543. }
  544. if (cUNode.getNodes().contains(tmp)) {
  545. if (tmp instanceof CpsUpperNode) {
  546. checkForConnectedStates(cps, (CpsUpperNode) tmp, theEdge);
  547. } else {
  548. theEdge.setConnected(CpsEdge.CON_UPPER_NODE_AND_INSIDE);
  549. break;
  550. }
  551. }
  552. }
  553. }
  554. public FlexiblePane getFlexiblePane() {
  555. return flexPane;
  556. }
  557. public void setFlexiblePane(FlexiblePane fp) {
  558. flexPane = fp;
  559. }
  560. }