GUI.java 39 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160
  1. package ui.view;
  2. import java.awt.BorderLayout;
  3. import java.awt.Color;
  4. import java.awt.Component;
  5. import java.awt.Cursor;
  6. import java.awt.Dimension;
  7. import java.awt.Image;
  8. import java.awt.Point;
  9. import java.awt.Toolkit;
  10. import java.awt.event.ActionEvent;
  11. import java.awt.event.ActionListener;
  12. import java.awt.event.MouseAdapter;
  13. import java.awt.event.MouseEvent;
  14. import java.beans.PropertyChangeEvent;
  15. import java.beans.PropertyChangeListener;
  16. import java.io.File;
  17. import java.io.IOException;
  18. import java.util.ArrayList;
  19. import javax.swing.BoxLayout;
  20. import javax.swing.DefaultComboBoxModel;
  21. import javax.swing.ImageIcon;
  22. import javax.swing.JButton;
  23. import javax.swing.JCheckBoxMenuItem;
  24. import javax.swing.JComboBox;
  25. import javax.swing.JEditorPane;
  26. import javax.swing.JFileChooser;
  27. import javax.swing.JFrame;
  28. import javax.swing.JLabel;
  29. import javax.swing.JMenu;
  30. import javax.swing.JMenuBar;
  31. import javax.swing.JMenuItem;
  32. import javax.swing.JOptionPane;
  33. import javax.swing.JPanel;
  34. import javax.swing.JPopupMenu;
  35. import javax.swing.JScrollPane;
  36. import javax.swing.JSlider;
  37. import javax.swing.JSplitPane;
  38. import javax.swing.JTabbedPane;
  39. import javax.swing.JTable;
  40. import javax.swing.JTextField;
  41. import javax.swing.JToolBar;
  42. import javax.swing.JTree;
  43. import javax.swing.SwingUtilities;
  44. import javax.swing.border.LineBorder;
  45. import javax.swing.event.ChangeEvent;
  46. import javax.swing.event.ChangeListener;
  47. import javax.swing.filechooser.FileNameExtensionFilter;
  48. import javax.swing.table.DefaultTableModel;
  49. import javax.swing.tree.DefaultMutableTreeNode;
  50. import javax.swing.tree.DefaultTreeModel;
  51. import javax.swing.tree.TreeCellRenderer;
  52. import Interfaces.CategoryListener;
  53. import classes.Category;
  54. import classes.CpsEdge;
  55. import classes.CpsObject;
  56. import classes.HolonElement;
  57. import classes.HolonObject;
  58. import classes.HolonSwitch;
  59. import classes.HolonTransformer;
  60. import ui.controller.Control;
  61. import ui.controller.SimulationManager;
  62. import ui.model.Model;;
  63. public class GUI<E> implements CategoryListener {
  64. private JFrame frmCyberPhysical;
  65. private final JMenuBar menuBar = new JMenuBar();
  66. private final JMenu mnNewMenu = new JMenu("File");
  67. private final JMenu mnNewMenu_1 = new JMenu("Edit");
  68. private final JMenu mnNewMenu_2 = new JMenu("Options");
  69. private final JMenu mnNewMenu_3 = new JMenu("View");
  70. private final JMenu mnHelp = new JMenu("Help");
  71. private final JMenuItem mntmOpen = new JMenuItem("Open");
  72. private final JMenuItem mntmNew = new JMenuItem("New");
  73. private final JMenuItem mntmSave = new JMenuItem("Save");
  74. private final JMenuItem aboutUs = new JMenuItem("About Us");
  75. private final JMenuItem canvasSize = new JMenuItem("View Size");
  76. private final JSplitPane splitPane = new JSplitPane();
  77. private final JSplitPane splitPane_1 = new JSplitPane();
  78. private final JScrollPane scrollPane_1 = new JScrollPane();
  79. private final JTabbedPane tabbedPane = new JTabbedPane(JTabbedPane.TOP);
  80. private final JScrollPane scrollPane_2 = new JScrollPane();
  81. private JPopupMenu popmenuEdit = new JPopupMenu();
  82. private JMenuItem editItem = new JMenuItem("Edit Object");
  83. private String catOfObjToBeEdited;
  84. private final JLabel maxGraph = new JLabel("100%");
  85. private final JLabel medGraph = new JLabel("50%");
  86. private final JLabel minGraph = new JLabel("0%");
  87. private final JLabel elementGraph = new JLabel("None ");
  88. private final ArrayList<HolonElement> selectedElements = new ArrayList<HolonElement>();
  89. private String holonEleNamesDisplayed = "None ";
  90. private final JTree tree = new JTree();
  91. private final JEditorPane dtrpnHereWillBe = new JEditorPane();
  92. /******************************************
  93. ************* Right Container*************
  94. ******************************************/
  95. // Right Container: here comes the information about the HolonObject, such
  96. // as HolonElements Information, Properties and Consumption/Production graph
  97. private final JSplitPane split_HolonEl_Pro = new JSplitPane(JSplitPane.VERTICAL_SPLIT);
  98. private final JSplitPane split_Graph_HolonEl = new JSplitPane(JSplitPane.VERTICAL_SPLIT);
  99. // In this section are all the Holonelements that correspond to the clicked
  100. // HolonObject with consumption/production, name and amount.
  101. private JTable tableHolonElement = new JTable();
  102. private PropertyTable tableModelHolonElement = new PropertyTable();
  103. private final JPanel scrollElements = new JPanel();
  104. private JScrollPane tableHolonElementScrollPane = new JScrollPane();
  105. // In this section are all the properties that correspond to the clicked
  106. // HolonObject, such as connections, name, Type, etc.
  107. private JTable tableProperties = new JTable();
  108. private JPanel graphLabel = new JPanel();
  109. private DefaulTable tableModelProperties;
  110. private final JScrollPane scrollProperties = new JScrollPane();
  111. // In this section is the graph for the selected HolonElement of the clicked
  112. // HolonObject
  113. private JTable tableGraph = new JTable();
  114. private DefaultTableModel tableModelGraph = new DefaultTableModel();
  115. private final JScrollPane scrollGraph = new JScrollPane();
  116. private Model model;
  117. private final Control controller;
  118. // Pop up Windows
  119. private AddObjectPopUp addObjectPopUP;
  120. private AboutUsPopUp aboutUsPopUp;
  121. private AddElementPopUp addElementPopUp;
  122. private final JPanel panel = new JPanel();
  123. private final JPanel panel_HolonEl = new JPanel();
  124. private final JComboBox comboBox = new JComboBox();
  125. // private final JComboBox comboBoxGraph = new JComboBox();
  126. // Buttons
  127. private final JButton btnAdd = new JButton("+");
  128. private final JButton btnDel = new JButton("-");
  129. private final JButton btnAddHolEL = new JButton("+");
  130. private final JButton btnDelHolEL = new JButton("-");
  131. private final JButton resetGraphBtn = new JButton("Reset");
  132. private final JToolBar toolBar = new JToolBar();
  133. private final JToolBar toolBarHolonEl = new JToolBar();
  134. private final JToolBar toolBarGraph = new JToolBar();
  135. // variables
  136. private boolean dragging = false;
  137. private String actualObjectClicked;
  138. private Image img = null;
  139. private CpsObject tempCps = null;
  140. private HolonElement tempElement = null;
  141. private int yValueElements = 0;
  142. private MyCanvas canvas;
  143. private UnitGraph unitGraph; // for testing, remove later
  144. private final JSplitPane splitPane_3 = new JSplitPane();
  145. private final JSlider sizeSlider = new JSlider();
  146. private final JLabel lblImageSize = new JLabel("Image Size");
  147. // Time Stuff
  148. private TimePanel timePanel;
  149. private final JMenu mnAlgorithm = new JMenu("Algorithm");
  150. private final JCheckBoxMenuItem chckbxmntmUseAlgorithm = new JCheckBoxMenuItem("Use Algorithm");
  151. private final JSplitPane splitPane_2 = new JSplitPane();
  152. private final JLabel lblSelect = new JLabel("Select");
  153. private final JComboBox comboBoxAlgo = new JComboBox();
  154. private int yTHIS;
  155. private int xTHIS;
  156. private int yBTHIS;
  157. private int xBTHIS;
  158. private CpsObject temp = null;
  159. private final JButton btnTest = new JButton("test");
  160. /**
  161. * Create the application.
  162. */
  163. public GUI(Control control) {
  164. this.controller = control;
  165. this.model = control.getModel();
  166. this.canvas = new MyCanvas(model, control);
  167. this.unitGraph = new UnitGraph(model, control);
  168. control.initListener(this);
  169. initialize();
  170. updateCategories(model.getCategories());
  171. }
  172. /**
  173. * Initialize the contents of the frame.
  174. */
  175. @SuppressWarnings({ "serial", "unchecked" })
  176. private void initialize() {
  177. frmCyberPhysical = new JFrame();
  178. frmCyberPhysical.setTitle("Cyber Physical Systems Model");
  179. frmCyberPhysical.setBounds(100, 100, 1000, 800);
  180. frmCyberPhysical.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
  181. frmCyberPhysical.setJMenuBar(menuBar);
  182. frmCyberPhysical.setIconImage(new ImageIcon(this.getClass().getResource("/Images/Dummy_House.png")).getImage()
  183. .getScaledInstance(30, 30, Image.SCALE_SMOOTH));
  184. menuBar.add(mnNewMenu);
  185. mnNewMenu.add(mntmNew);
  186. mnNewMenu.add(mntmOpen);
  187. mnNewMenu.add(mntmSave);
  188. menuBar.add(mnNewMenu_1);
  189. menuBar.add(mnNewMenu_2);
  190. menuBar.add(mnNewMenu_3);
  191. mnNewMenu_3.add(canvasSize);
  192. canvasSize.addActionListener(new ActionListener() {
  193. @Override
  194. public void actionPerformed(ActionEvent e) {
  195. JPanel myPanel = new JPanel();
  196. JTextField field1 = new JTextField("" + canvas.getWidth() + "");
  197. JTextField field2 = new JTextField("" + canvas.getHeight() + "");
  198. myPanel.add(field1);
  199. myPanel.add(field2);
  200. JOptionPane.showMessageDialog(null, myPanel);
  201. canvas.setPreferredSize(
  202. new Dimension(Integer.parseInt(field1.getText()), Integer.parseInt(field2.getText())));
  203. canvas.repaint();
  204. }
  205. });
  206. mnNewMenu_3.add(splitPane_3);
  207. sizeSlider.setMinimum(15);
  208. sizeSlider.setMaximum(115);
  209. sizeSlider.addChangeListener(new ChangeListener() {
  210. @Override
  211. public void stateChanged(ChangeEvent e) {
  212. controller.setScale(sizeSlider.getValue());
  213. tree.setRowHeight(model.getScale());
  214. canvas.objectSelectionHighlighting();
  215. canvas.repaint();
  216. }
  217. });
  218. splitPane_3.setRightComponent(sizeSlider);
  219. splitPane_3.setLeftComponent(lblImageSize);
  220. menuBar.add(mnHelp);
  221. mnHelp.add(aboutUs);
  222. menuBar.add(mnAlgorithm);
  223. mnAlgorithm.add(chckbxmntmUseAlgorithm);
  224. mnAlgorithm.add(splitPane_2);
  225. splitPane_2.setLeftComponent(lblSelect);
  226. splitPane_2.setRightComponent(comboBoxAlgo);
  227. canvas.setBackground(Color.WHITE);
  228. canvas.setPreferredSize(new Dimension(1000, 1000));
  229. JScrollPane canvasSP = new JScrollPane(canvas);
  230. canvasSP.setBorder(null);
  231. tabbedPane.addTab("Modeling", new ImageIcon(new ImageIcon(this.getClass().getResource("/Images/home.png"))
  232. .getImage().getScaledInstance(30, 30, Image.SCALE_SMOOTH)), canvasSP, "Model a CPS");
  233. tabbedPane.addTab("Simulation",
  234. new ImageIcon(new ImageIcon(this.getClass().getResource("/Images/Dummy_House.png")).getImage()
  235. .getScaledInstance(30, 30, Image.SCALE_SMOOTH)),
  236. scrollPane_2, "Simulate the CPS");
  237. dtrpnHereWillBe.setText("Here will be the Simulation");
  238. scrollPane_2.setViewportView(dtrpnHereWillBe);
  239. /********************
  240. * RIGHT CONTAINER (INFORMATION)
  241. **********************/
  242. // Set up of the HolonElements section
  243. Object[] columnNames = { "Device", "Energy", "Quantity", "Activated" };
  244. tableModelHolonElement.setColumnIdentifiers(columnNames);
  245. tableHolonElement.setBorder(null);
  246. tableHolonElement.setModel(tableModelHolonElement);
  247. tableHolonElement.setFillsViewportHeight(true);
  248. tableHolonElement.setCellSelectionEnabled(true);
  249. tableHolonElement.setColumnSelectionAllowed(true);
  250. tableHolonElementScrollPane.setViewportView(tableHolonElement);
  251. scrollElements.setLayout(new BorderLayout(0, 0));
  252. scrollElements.add(panel_HolonEl, BorderLayout.NORTH);
  253. scrollElements.add(tableHolonElementScrollPane);
  254. panel_HolonEl.setLayout(new BoxLayout(panel_HolonEl, BoxLayout.X_AXIS));
  255. toolBarHolonEl.add(btnAddHolEL);
  256. toolBarHolonEl.add(btnDelHolEL);
  257. toolBarHolonEl.setFloatable(false);
  258. panel_HolonEl.add(toolBarHolonEl);
  259. // Set up of the Properties section
  260. Object[] colNames = { "Field", "Information" };
  261. tableModelProperties = new DefaulTable(100, colNames.length);
  262. tableModelProperties.setColumnIdentifiers(colNames);
  263. tableProperties.setModel(tableModelProperties);
  264. tableProperties.setFillsViewportHeight(true);
  265. tableProperties.setCellSelectionEnabled(true);
  266. tableProperties.setColumnSelectionAllowed(true);
  267. scrollProperties.setViewportView(tableProperties);
  268. // Set up of the Graph section
  269. Object[] tempText = { "Here comes the graph for each clicked HolonElement" };
  270. tableModelGraph.setColumnIdentifiers(tempText);
  271. tableGraph.setModel(tableModelGraph);
  272. tableGraph.setFillsViewportHeight(true);
  273. tableGraph.setCellSelectionEnabled(true);
  274. tableGraph.setColumnSelectionAllowed(true);
  275. scrollGraph.setViewportView(unitGraph);
  276. graphLabel.setLayout(new BorderLayout(0, 10));
  277. graphLabel.add(maxGraph, BorderLayout.NORTH);
  278. graphLabel.add(medGraph, BorderLayout.CENTER);
  279. graphLabel.add(minGraph, BorderLayout.SOUTH);
  280. toolBarGraph.add(elementGraph);
  281. toolBarGraph.add(resetGraphBtn);
  282. toolBarGraph.setFloatable(false);
  283. scrollGraph.setRowHeaderView(graphLabel);
  284. scrollGraph.setColumnHeaderView(toolBarGraph);
  285. /***********************
  286. * HolonElement Table Actions
  287. **********************/
  288. /*
  289. * Add HolonElement to given HolonObject
  290. */
  291. btnAddHolEL.addActionListener(new ActionListener() {
  292. public void actionPerformed(ActionEvent arg0) {
  293. CpsObject tempCpsObject = getActualCps();
  294. if (tempCpsObject != null && tempCpsObject.getClass() == HolonObject.class
  295. && tempCpsObject.getID() != 0) {
  296. addElementPopUp = new AddElementPopUp();
  297. addElementPopUp.setActualCps(getActualCps());
  298. addElementPopUp.setVisible(true);
  299. HolonElement ele = addElementPopUp.getElement();
  300. controller.addElementCanvasObject(tempCpsObject.getID(), ele.getEleName(), ele.getAmount(),
  301. ele.getEnergy());
  302. refreshTableHolonElement();
  303. refreshTableProperties();
  304. }
  305. }
  306. });
  307. /*
  308. * Delete the choosen HolonElement of the selected HolonObject
  309. */
  310. btnDelHolEL.addActionListener(new ActionListener() {
  311. public void actionPerformed(ActionEvent arg0) {
  312. if (getActualCps().getClass() == HolonObject.class) {
  313. HolonObject obj = (HolonObject) getActualCps();
  314. tempElement = getActualHolonElement(obj, yValueElements);
  315. if (tempElement != null && obj.getClass() == HolonObject.class && obj.getID() != 0) {
  316. controller.deleteElementCanvas(obj.getID(), tempElement.getEleName());
  317. refreshTableHolonElement();
  318. refreshTableProperties();
  319. }
  320. }
  321. }
  322. });
  323. /*
  324. * Communication between HolonElement Table and displayed Graph
  325. */
  326. tableHolonElement.addMouseListener(new MouseAdapter() {
  327. public void mousePressed(MouseEvent e) {
  328. HolonObject obj = (HolonObject) getActualCps();
  329. yValueElements = e.getY();
  330. HolonElement ele = getActualHolonElement(obj, yValueElements);
  331. if (e.isControlDown() && ele != null) {
  332. if (!selectedElements.contains(ele)) {
  333. selectedElements.add(ele);
  334. if (!holonEleNamesDisplayed.equals("None ")) {
  335. holonEleNamesDisplayed = holonEleNamesDisplayed + "; " + ele.getEleName() + " ";
  336. } else {
  337. holonEleNamesDisplayed = ele.getEleName() + " ";
  338. }
  339. unitGraph.repaintWithNewElement(selectedElements);
  340. }
  341. } else if (ele != null) {
  342. selectedElements.clear();
  343. selectedElements.add(ele);
  344. holonEleNamesDisplayed = ele.getEleName() + " ";
  345. unitGraph.repaintWithNewElement(selectedElements);
  346. } else {
  347. elementGraph.setText("None ");
  348. unitGraph.empty();
  349. }
  350. // if any HolonElement is double-clicked --> Edit instance of
  351. // selected HolonElement
  352. if (e.getClickCount() == 2) {
  353. yTHIS = e.getY();
  354. xTHIS = e.getX();
  355. }
  356. if (e.getClickCount() == 1 && ele == null) {
  357. selectedElements.clear();
  358. holonEleNamesDisplayed = "None ";
  359. }
  360. for (int i = 0; i < selectedElements.size(); i++) {
  361. if (i == 0) {
  362. System.out.println("Selected Items: " + selectedElements.get(i).getEleName());
  363. } else {
  364. System.out.println(selectedElements.get(i).getEleName());
  365. }
  366. }
  367. elementGraph.setText(holonEleNamesDisplayed);
  368. yBTHIS = e.getY();
  369. xBTHIS = e.getX();
  370. }
  371. });
  372. /*
  373. * If the HolonElement Table enters to editing instance, than is the
  374. * propertyChangeListener triggered
  375. */
  376. tableHolonElement.addPropertyChangeListener(new PropertyChangeListener() {
  377. @Override
  378. public void propertyChange(PropertyChangeEvent evt) {
  379. try {
  380. int yMouse = yTHIS;
  381. int yBMouse = yBTHIS;
  382. int selectedValueX = (int) Math.floor(xTHIS / (tableHolonElement.getWidth() / 4));
  383. int selectedValueY = (int) Math.floor(yMouse / 16);
  384. int selectedValueBX = (int) Math.floor(xBTHIS / (tableHolonElement.getWidth() / 4));
  385. int selectedValueBY = (int) Math.floor(yBMouse / 16);
  386. if (getActualCps() != null && getActualCps().getClass() == HolonObject.class) {
  387. if (selectedValueBX == 3) {
  388. HolonElement eleBTemp = getActualHolonElement((HolonObject) getActualCps(), yBMouse);
  389. String newBStuff = tableModelHolonElement.getValueAt(selectedValueBY, selectedValueBX)
  390. .toString();
  391. Boolean bTemp = Boolean.parseBoolean(newBStuff);
  392. eleBTemp.setActive(bTemp);
  393. } else {
  394. HolonElement eleTemp = getActualHolonElement((HolonObject) getActualCps(), yMouse);
  395. String newStuff = tableModelHolonElement.getValueAt(selectedValueY, selectedValueX)
  396. .toString();
  397. if (selectedValueX == 0) {
  398. eleTemp.setEleName(newStuff);
  399. } else if (selectedValueX == 1) {
  400. Float ftemp = Float.parseFloat(newStuff);
  401. eleTemp.setEnergy(ftemp);
  402. } else if (selectedValueX == 2) {
  403. Integer iTemp = Integer.parseInt(newStuff);
  404. eleTemp.setAmount(iTemp);
  405. }
  406. }
  407. refreshTableProperties();
  408. tableModelHolonElement.fireTableDataChanged();
  409. }
  410. } catch (Exception e) {
  411. }
  412. }
  413. });
  414. /***********************
  415. * HolonElement Properties Actions
  416. **********************/
  417. /*
  418. * If any value at the Properties Table enters to editing instance, than
  419. * is PropertyChangeListener triggered (For HolonObject, the only value
  420. * to be edited is the name and for CpsEdge the Max.flow)
  421. */
  422. tableProperties.addPropertyChangeListener(new PropertyChangeListener() {
  423. @Override
  424. public void propertyChange(PropertyChangeEvent evt) {
  425. try {
  426. Object temp;
  427. if (getActualCps() != null) {
  428. if (getActualCps() instanceof HolonObject) {
  429. temp = tableModelProperties.getValueAt(0, 1);
  430. getActualCps().setName(temp.toString());
  431. } else if (getActualCps() instanceof HolonTransformer) {
  432. // get Info of the Properties for Transformer
  433. }
  434. } else {
  435. temp = tableModelProperties.getValueAt(2, 1);
  436. Float ftemp = Float.parseFloat(temp.toString());
  437. model.getSelectedEdge().setCapacity(ftemp);
  438. }
  439. canvas.repaint();
  440. } catch (Exception e) {
  441. }
  442. }
  443. });
  444. /***********************
  445. * HolonElement Graph Actions
  446. **********************/
  447. /*
  448. * Reset the graph of the selected HolonElement
  449. */
  450. resetGraphBtn.setBorder(new LineBorder(Color.BLACK));
  451. resetGraphBtn.addActionListener(new ActionListener() {
  452. public void actionPerformed(ActionEvent arg0) {
  453. unitGraph.reset();
  454. }
  455. });
  456. /*****************************
  457. * RIGHT CONTAINER DONE
  458. *****************************/
  459. frmCyberPhysical.getContentPane().setLayout(new BorderLayout(0, 0));
  460. TreeCellRenderer customRenderer = new TreeCellRenderer() {
  461. @Override
  462. public Component getTreeCellRendererComponent(JTree tree, Object value, boolean selected, boolean expanded,
  463. boolean leaf, int row, boolean hasFocus) {
  464. JLabel label = new JLabel();
  465. Image imgR = null;
  466. if (leaf) {
  467. for (Category cat : model.getCategories()) {
  468. for (CpsObject cps : cat.getObjects()) {
  469. if (value.toString().compareTo(cps.getObjName()) == 0) {
  470. File checkPath = new File(cps.getImage());
  471. if (checkPath.exists()) {
  472. imgR = new ImageIcon(cps.getImage()).getImage().getScaledInstance(
  473. controller.getScale(), controller.getScale(), java.awt.Image.SCALE_SMOOTH);
  474. } else {
  475. imgR = new ImageIcon(this.getClass().getResource(cps.getImage())).getImage()
  476. .getScaledInstance(controller.getScale(), controller.getScale(),
  477. java.awt.Image.SCALE_SMOOTH);
  478. }
  479. if (imgR != null) {
  480. label.setIcon(new ImageIcon(imgR));
  481. }
  482. label.setText(cps.getName());
  483. }
  484. }
  485. }
  486. }
  487. tree.setRowHeight(model.getScale());
  488. if (hasFocus) {
  489. label.setForeground(new Color(0, 0, 255));
  490. label.setOpaque(true);
  491. }
  492. if (label.getText().length() == 0) {
  493. label.setText(value.toString());
  494. if (value.toString().compareTo("Categories") != 0) {
  495. label.setIcon(new ImageIcon(this.getClass().getResource("/Images/folder.png")));
  496. }
  497. }
  498. return label;
  499. }
  500. };
  501. tree.setCellRenderer(customRenderer);
  502. tree.addMouseListener(new MouseAdapter() {
  503. public void mouseReleased(MouseEvent e) {
  504. try {
  505. if (dragging) {
  506. int x = (int) canvas.getMousePosition().getX();
  507. int y = (int) canvas.getMousePosition().getY();
  508. CpsObject h = null;
  509. if (tempCps.getClass() == HolonObject.class) {
  510. h = new HolonObject(tempCps);
  511. }
  512. if (tempCps.getClass() == HolonSwitch.class) {
  513. h = new HolonSwitch(tempCps);
  514. }
  515. if (tempCps.getClass() == HolonTransformer.class) {
  516. h = new HolonTransformer(tempCps);
  517. }
  518. h.setPosition(x, y);
  519. controller.addObjectCanvas(h);
  520. canvas.repaint();
  521. dragging = false;
  522. }
  523. } catch (Exception e2) {
  524. }
  525. frmCyberPhysical.setCursor(new Cursor(Cursor.DEFAULT_CURSOR));
  526. }
  527. });
  528. popmenuEdit.add(editItem);
  529. editItem.setEnabled(false);
  530. editItem.addActionListener(new ActionListener() {
  531. @Override
  532. public void actionPerformed(ActionEvent e) {
  533. }
  534. });
  535. tree.addMouseListener(new MouseAdapter() {
  536. public void mousePressed(MouseEvent e) {
  537. try {
  538. actualObjectClicked = tree.getPathForLocation(e.getX(), e.getY()).getLastPathComponent().toString();
  539. // if an Object was selected, the porperties are shown in
  540. // the table
  541. DefaultMutableTreeNode selectedNode = (DefaultMutableTreeNode) tree
  542. .getPathForLocation(e.getX(), e.getY()).getLastPathComponent();
  543. if (selectedNode.getLevel() == 2) {
  544. CpsObject selected = controller.searchCategoryObject(selectedNode.getParent().toString(),
  545. selectedNode.toString());
  546. deleteRows();
  547. // if (selected instanceof HolonObject && selected !=
  548. // null) {
  549. // selected = (HolonObject) selected;
  550. // fillElementTable(((HolonObject)
  551. // selected).getElements());
  552. // }
  553. }
  554. if (SwingUtilities.isRightMouseButton(e)) {
  555. for (Category cat : model.getCategories()) {
  556. for (CpsObject cps : cat.getObjects()) {
  557. if (actualObjectClicked.compareTo(cps.getObjName()) == 0
  558. && !(cps instanceof HolonSwitch)) {
  559. editItem.setEnabled(true);
  560. popmenuEdit.show(e.getComponent(), e.getX(), e.getY());
  561. catOfObjToBeEdited = selectedNode.getParent().toString();
  562. tempCps = cps;
  563. }
  564. }
  565. }
  566. } else {
  567. for (Category cat : model.getCategories()) {
  568. for (CpsObject cps : cat.getObjects()) {
  569. if (actualObjectClicked.compareTo(cps.getObjName()) == 0) {
  570. File checkPath = new File(cps.getImage());
  571. if (checkPath.exists()) {
  572. img = new ImageIcon(cps.getImage()).getImage().getScaledInstance(
  573. controller.getScale(), controller.getScale(),
  574. java.awt.Image.SCALE_SMOOTH);
  575. } else {
  576. img = new ImageIcon(this.getClass().getResource(cps.getImage())).getImage()
  577. .getScaledInstance(controller.getScale(), controller.getScale(),
  578. java.awt.Image.SCALE_SMOOTH);
  579. }
  580. tempCps = cps;
  581. dragging = true;
  582. Cursor cursor = Toolkit.getDefaultToolkit().createCustomCursor(img, new Point(0, 0),
  583. "Image");
  584. frmCyberPhysical.setCursor(cursor);
  585. }
  586. }
  587. }
  588. }
  589. } catch (Exception e2) {
  590. }
  591. }
  592. });
  593. editItem.addActionListener(new ActionListener() {
  594. @Override
  595. public void actionPerformed(ActionEvent e) {
  596. // Remove the selected Object object
  597. addObjectPopUP = new AddObjectPopUp(true, tempCps, catOfObjToBeEdited);
  598. addObjectPopUP.setCategory(catOfObjToBeEdited);
  599. addObjectPopUP.setController(controller);
  600. addObjectPopUP.setVisible(true);
  601. }
  602. });
  603. scrollPane_1.setViewportView(tree);
  604. scrollPane_1.setColumnHeaderView(panel);
  605. panel.setLayout(new BoxLayout(panel, BoxLayout.Y_AXIS));
  606. toolBar.setFloatable(false);
  607. panel.add(toolBar);
  608. toolBar.add(comboBox);
  609. comboBox.setModel(new DefaultComboBoxModel(new String[] { "Category", "Object", "Transformer", "Switch" }));
  610. // Add Button
  611. btnAdd.addActionListener(new ActionListener() {
  612. public void actionPerformed(ActionEvent arg0) {
  613. Object nodeInfo = tree.getLastSelectedPathComponent();
  614. String selectedOption = comboBox.getSelectedItem().toString();
  615. DefaultMutableTreeNode selectedNode = (DefaultMutableTreeNode) nodeInfo;
  616. switch (selectedOption) {
  617. case "Category":
  618. String catName = JOptionPane.showInputDialog("Please enter a Name for Category ");
  619. if (catName.length() != 0) {
  620. controller.addCategory(catName);
  621. }
  622. break;
  623. case "Object":
  624. if (selectedNode == null) {
  625. JOptionPane.showMessageDialog(new JFrame(),
  626. "Please select a Category first before adding " + selectedOption + ".");
  627. }
  628. if (selectedNode.getLevel() == 1) {
  629. addObjectPopUP = new AddObjectPopUp(false, null, null);
  630. addObjectPopUP.setVisible(true);
  631. addObjectPopUP.setController(controller);
  632. addObjectPopUP.setCategory(selectedNode.toString());
  633. }
  634. break;
  635. default:
  636. addObjectAction(selectedOption, selectedNode);
  637. break;
  638. }
  639. tree.repaint();
  640. }
  641. });
  642. /**
  643. * Pop up - About Us with some important information about the
  644. * developers, source and programming stuff
  645. */
  646. aboutUs.addMouseListener(new MouseAdapter() {
  647. @Override
  648. public void mousePressed(MouseEvent e) {
  649. aboutUsPopUp = new AboutUsPopUp();
  650. aboutUsPopUp.setVisible(true);
  651. }
  652. });
  653. /**
  654. * Update of every interaction between the user and the canvas (only on
  655. * the canvas). Basically the update of all the information concerning
  656. * the clicked HolonObject
  657. */
  658. canvas.addMouseListener(new MouseAdapter() {
  659. @Override
  660. public void mousePressed(MouseEvent e) {
  661. if (temp == null || temp.getID() != model.getSelectedObjectID()) {
  662. unitGraph.empty();
  663. elementGraph.setText("None ");
  664. }
  665. temp = getActualCps();
  666. // Update of the Information about the HolonElements - only for
  667. // if (temp instanceof HolonObject) {
  668. // refreshTableHolonElement();
  669. // }
  670. // Update of the Information about the Properties - only for
  671. // CpsObjects
  672. // Erase old data
  673. if (tableModelProperties.getRowCount() > 0) {
  674. for (int i = tableModelProperties.getRowCount() - 1; i > -1; i--) {
  675. tableModelProperties.removeRow(i);
  676. }
  677. }
  678. // Write new data
  679. if (temp != null) {
  680. Object[] tempName = { "Name", temp.getName() };
  681. tableModelProperties.addRow(tempName);
  682. Object[] tempId = { "ID", temp.getID() };
  683. tableModelProperties.addRow(tempId);
  684. if (temp instanceof HolonObject) {
  685. refreshTableHolonElement();
  686. Object[] tempEnergy = { "Total Energy", ((HolonObject) temp).getCurrentEnergy() };
  687. tableModelProperties.addRow(tempEnergy);
  688. } else if (temp instanceof HolonTransformer) {
  689. deleteRows();
  690. Object[] tempRatioPerc = { "Ratio Type", true };
  691. tableModelProperties.addRow(tempRatioPerc);
  692. } else if (temp instanceof HolonSwitch) {
  693. deleteRows();
  694. Object[] tempActive = { "Active", ((HolonSwitch) temp).getActiveAt()[model.getCurIteration()] };
  695. tableModelProperties.addRow(tempActive);
  696. unitGraph.repaintWithNewSwitch((HolonSwitch) temp);
  697. elementGraph.setText(temp.getName());
  698. } else {
  699. deleteRows();
  700. }
  701. tableModelProperties.setCellEditable(0, 1, true);
  702. tableModelProperties.setCellEditable(2, 1, false);
  703. ArrayList<CpsEdge> temp_array = temp.getConnections();
  704. if (!temp_array.isEmpty()) {
  705. boolean first = true;
  706. for (CpsEdge temp2 : temp_array) {
  707. if (first) {
  708. first = false;
  709. if (temp.getID() != temp2.getA().getID()) {
  710. Object[] tempConnection = { temp.getName() + " is connected to",
  711. temp2.getA().getName() + " with ID: " + temp2.getA().getID() };
  712. tableModelProperties.addRow(tempConnection);
  713. } else {
  714. Object[] tempConnection = { temp.getName() + " is connected to",
  715. temp2.getB().getName() + " with ID: " + temp2.getB().getID() };
  716. tableModelProperties.addRow(tempConnection);
  717. }
  718. } else {
  719. if (temp.getID() != temp2.getA().getID()) {
  720. Object[] tempConnection = { "",
  721. temp2.getA().getName() + " with ID: " + temp2.getA().getID() };
  722. tableModelProperties.addRow(tempConnection);
  723. } else {
  724. Object[] tempConnection = { "",
  725. temp2.getB().getName() + " with ID: " + temp2.getB().getID() };
  726. tableModelProperties.addRow(tempConnection);
  727. }
  728. }
  729. }
  730. }
  731. } else if (model.getSelectedEdge() != null) {
  732. Object[] tempName = { "Name", "Edge: " + model.getSelectedEdge().getA().getName() + " to "
  733. + model.getSelectedEdge().getB().getName() };
  734. tableModelProperties.addRow(tempName);
  735. Object[] tempFlow = { "Current flow", model.getSelectedEdge().getFlow() };
  736. tableModelProperties.addRow(tempFlow);
  737. Object[] tempCapacity = { "Max. Capacity", model.getSelectedEdge().getCapacity() };
  738. tableModelProperties.addRow(tempCapacity);
  739. tableModelProperties.setCellEditable(0, 1, false);
  740. tableModelProperties.setCellEditable(2, 1, true);
  741. } else if (getActualCps() == null) {
  742. deleteRows();
  743. }
  744. }
  745. });
  746. toolBar.add(btnAdd);
  747. // Del Button
  748. btnDel.addActionListener(new ActionListener() {
  749. public void actionPerformed(ActionEvent arg0) {
  750. Object nodeInfo = tree.getLastSelectedPathComponent();
  751. if (nodeInfo != null) {
  752. DefaultMutableTreeNode selectedNode = (DefaultMutableTreeNode) nodeInfo;
  753. String nodeName = selectedNode.getUserObject().toString();
  754. int depthOfNode = selectedNode.getLevel();
  755. switch (depthOfNode) {
  756. case 1:
  757. int dialogResult = JOptionPane.showConfirmDialog(null,
  758. "Do you realy want to delete the Category " + nodeName + "?", "Warning",
  759. JOptionPane.YES_NO_OPTION);
  760. if (dialogResult == JOptionPane.YES_OPTION) {
  761. controller.deleteCategory(nodeName);
  762. }
  763. break;
  764. case 2:
  765. DefaultMutableTreeNode parent = (DefaultMutableTreeNode) selectedNode.getParent();
  766. controller.delObjectCategory(parent.getUserObject().toString(), nodeName);
  767. break;
  768. default:
  769. JOptionPane.showMessageDialog(new JFrame(),
  770. "Please select a Category or an Object in order to delete something.");
  771. }
  772. } else {
  773. JOptionPane.showMessageDialog(new JFrame(),
  774. "Please select a Category or an Object in order to delete something.");
  775. }
  776. tree.repaint();
  777. }
  778. });
  779. toolBar.add(btnDel);
  780. frmCyberPhysical.getContentPane().add(splitPane);
  781. mntmNew.addActionListener(new ActionListener() {
  782. @Override
  783. public void actionPerformed(ActionEvent arg0) {
  784. model.getEdgesOnCanvas().removeAll(model.getEdgesOnCanvas());
  785. model.getObjectsOnCanvas().removeAll(model.getObjectsOnCanvas());
  786. controller.setSelectedObjectID(0);
  787. controller.setSelecteEdge(null);
  788. controller.setCurIteration(0);
  789. unitGraph.empty();
  790. elementGraph.setText("None ");
  791. canvas.tempCps = null;
  792. canvas.objectSelectionHighlighting();
  793. canvas.repaint();
  794. }
  795. });
  796. mntmOpen.addActionListener(new java.awt.event.ActionListener() {
  797. @Override
  798. public void actionPerformed(java.awt.event.ActionEvent evt) {
  799. menuFileExitActionPerformed(evt);
  800. }
  801. private void menuFileExitActionPerformed(java.awt.event.ActionEvent evt) {
  802. JFileChooser fileChooser = new JFileChooser();
  803. JFrame test = new JFrame();
  804. FileNameExtensionFilter jsonFilter = new FileNameExtensionFilter("*.json", "json");
  805. fileChooser.setFileFilter(jsonFilter);
  806. if (fileChooser.showOpenDialog(test) == JFileChooser.APPROVE_OPTION) {
  807. File file = fileChooser.getSelectedFile();
  808. try {
  809. controller.loadFile(file.getAbsolutePath());
  810. canvas.repaint();
  811. tree.repaint();
  812. } catch (IOException e) {
  813. // TODO Auto-generated catch block
  814. e.printStackTrace();
  815. }
  816. }
  817. }
  818. });
  819. mntmSave.addActionListener(new java.awt.event.ActionListener() {
  820. @Override
  821. public void actionPerformed(java.awt.event.ActionEvent evt) {
  822. menuSaveActionPerformed(evt);
  823. }
  824. private void menuSaveActionPerformed(java.awt.event.ActionEvent evt) {
  825. JFileChooser fileChooser = new JFileChooser();
  826. JFrame test = new JFrame();
  827. FileNameExtensionFilter jsonFilter = new FileNameExtensionFilter("*.json", "json");
  828. fileChooser.setFileFilter(jsonFilter);
  829. if (fileChooser.showSaveDialog(test) == JFileChooser.APPROVE_OPTION) {
  830. String file = fileChooser.getSelectedFile().getPath();
  831. if (!file.endsWith(".json"))
  832. file += ".json";
  833. try {
  834. controller.saveFile(new File(file).getAbsolutePath());
  835. } catch (IOException e) {
  836. // TODO Auto-generated catch block
  837. e.printStackTrace();
  838. }
  839. }
  840. }
  841. });
  842. timePanel = new TimePanel(model, controller);
  843. timePanel.setBorder(null);
  844. ((JSlider) (timePanel.getComponent(1))).addChangeListener(new ChangeListener() {
  845. @Override
  846. public void stateChanged(ChangeEvent e) {
  847. unitGraph.repaint();
  848. }
  849. });
  850. splitPane.setRightComponent(splitPane_1);
  851. splitPane.setDividerLocation(200);
  852. splitPane_1.setDividerLocation(500);
  853. splitPane.setLeftComponent(scrollPane_1);
  854. splitPane_1.setLeftComponent(tabbedPane);
  855. splitPane_1.setRightComponent(split_HolonEl_Pro);
  856. split_HolonEl_Pro.setDividerLocation(400);
  857. split_HolonEl_Pro.setTopComponent(split_Graph_HolonEl);
  858. split_HolonEl_Pro.setBottomComponent(scrollProperties);
  859. split_Graph_HolonEl.setDividerLocation(150);
  860. split_Graph_HolonEl.setTopComponent(scrollGraph);
  861. split_Graph_HolonEl.setBottomComponent(scrollElements);
  862. tabbedPane.setBorder(null);
  863. scrollProperties.setBorder(null);
  864. scrollGraph.setBorder(null);
  865. scrollElements.setBorder(null);
  866. splitPane.setBorder(null);
  867. splitPane_1.setBorder(null);
  868. split_HolonEl_Pro.setBorder(null);
  869. split_Graph_HolonEl.setBorder(null);
  870. scrollPane_2.setBorder(null);
  871. panel_HolonEl.setBorder(null);
  872. btnTest.addActionListener(new ActionListener() {
  873. public void actionPerformed(ActionEvent arg0) {
  874. SimulationManager sm = new SimulationManager(model);
  875. sm.calculateStateForTimeStep();
  876. }
  877. });
  878. panel_HolonEl.add(btnTest);
  879. tableHolonElementScrollPane.setBorder(null);
  880. frmCyberPhysical.getContentPane().add(timePanel, BorderLayout.SOUTH);
  881. }
  882. /*
  883. * adds a specific object type to selected Category also handles input
  884. * windows and illegal inputs
  885. */
  886. public void addObjectAction(String objType, DefaultMutableTreeNode selectedNode) {
  887. if (selectedNode == null) {
  888. JOptionPane.showMessageDialog(new JFrame(),
  889. "Please select a Category first before adding " + objType + ".");
  890. }
  891. // if selected node is a directory for Categories
  892. else {
  893. if (selectedNode.getLevel() == 1) {
  894. String objname = JOptionPane.showInputDialog("Please enter a Name for the " + objType);
  895. Category cat = controller.searchCategory(selectedNode.getUserObject().toString());
  896. if (objname.length() != 0) {
  897. switch (objType) {
  898. case "Switch":
  899. controller.addSwitch(cat, objname);
  900. break;
  901. case "Transformer":
  902. controller.addTransformer(cat, objname);
  903. break;
  904. }
  905. }
  906. } else {
  907. JOptionPane.showMessageDialog(new JFrame(),
  908. "Objects can not be added to Objects. Please select a Category.");
  909. }
  910. }
  911. }
  912. /**
  913. * reloads the Categories from Model
  914. */
  915. public void updateCategories(final ArrayList<Category> categories) {
  916. tree.setModel(new DefaultTreeModel(new DefaultMutableTreeNode("Categories") {
  917. {
  918. DefaultMutableTreeNode node_1;
  919. for (Category c : categories) {
  920. node_1 = new DefaultMutableTreeNode(c.getName());
  921. // kann eventuell umgeändert werden
  922. for (CpsObject obj : c.getObjects()) {
  923. node_1.add(new DefaultMutableTreeNode(obj.getObjName()));
  924. }
  925. add(node_1);
  926. }
  927. }
  928. }));
  929. }
  930. /**
  931. *
  932. */
  933. public void onChange(ArrayList<Category> categories) {
  934. DefaultTreeModel model = (DefaultTreeModel) tree.getModel();
  935. updateCategories(categories);
  936. model.reload();
  937. }
  938. /**
  939. *
  940. * @return
  941. */
  942. public JFrame getFrmCyberPhysical() {
  943. return frmCyberPhysical;
  944. }
  945. /**
  946. * Getter for selected CpsObject
  947. *
  948. * @return selected CpsObject
  949. */
  950. public CpsObject getActualCps() {
  951. int tempID = model.getSelectedObjectID();
  952. CpsObject tempCps = controller.searchByID(tempID);
  953. return tempCps;
  954. }
  955. /**
  956. * Search for actual selected HolonElement
  957. *
  958. * @param obj
  959. * selected HolonObject
  960. * @param yValue
  961. * Y-Coord in the HolonElementsTable
  962. * @return the selected HolonElement
  963. */
  964. public HolonElement getActualHolonElement(HolonObject obj, int yValue) {
  965. final int yTemp = (int) Math.floor(yValue / 16);
  966. int rowsTotal = tableModelHolonElement.getRowCount();
  967. if (rowsTotal != 0 && rowsTotal > yTemp) {
  968. model.setSelectedHolonElement(obj.getElements().get(yTemp));
  969. return obj.getElements().get(yTemp);
  970. } else {
  971. model.setSelectedHolonElement(null);
  972. return null;
  973. }
  974. }
  975. /**
  976. * Update the HolonElement Table, that means erase all rows and add the new
  977. * rows with new data
  978. */
  979. public void refreshTableHolonElement() {
  980. // Update of the Information about the HolonElements - only for
  981. // HolonObjects
  982. deleteRows();
  983. if (getActualCps() != null) {
  984. fillElementTable(((HolonObject) getActualCps()).getElements());
  985. }
  986. tableModelHolonElement.fireTableDataChanged();
  987. }
  988. /**
  989. * Erase all information of the HolonElement Model
  990. */
  991. public void deleteRows() {
  992. if (tableModelHolonElement.getRowCount() > 0) {
  993. for (int i = tableModelHolonElement.getRowCount() - 1; i > -1; i--) {
  994. tableModelHolonElement.removeRow(i);
  995. }
  996. }
  997. }
  998. /**
  999. * Add the Information of the given ArrayList in the HolonElement Model as
  1000. * Name,Energy and Amount
  1001. *
  1002. * @param elements
  1003. * ArrayList to be displayed
  1004. */
  1005. public void fillElementTable(ArrayList<HolonElement> elements) {
  1006. for (HolonElement he : elements) {
  1007. Object[] temp = { he.getEleName(), he.getEnergy(), he.getAmount(), he.getActive() };
  1008. tableModelHolonElement.addRow(temp);
  1009. }
  1010. }
  1011. /**
  1012. * Update the information concerning properties of the selected CpsObject
  1013. */
  1014. public void refreshTableProperties() {
  1015. CpsObject tempCps = getActualCps();
  1016. if (tempCps != null && tempCps.getClass() == HolonObject.class) {
  1017. tableModelProperties.removeRow(2);
  1018. Object[] tempEnergy = { "Total Energy", ((HolonObject) tempCps).getCurrentEnergy() };
  1019. tableModelProperties.insertRow(2, tempEnergy);
  1020. } else if (tempCps instanceof HolonTransformer) {
  1021. // Refresh Transformer Info
  1022. }
  1023. }
  1024. private static void addPopup(Component component, final JPopupMenu popup) {
  1025. component.addMouseListener(new MouseAdapter() {
  1026. public void mousePressed(MouseEvent e) {
  1027. if (e.isPopupTrigger()) {
  1028. showMenu(e);
  1029. }
  1030. }
  1031. public void mouseReleased(MouseEvent e) {
  1032. if (e.isPopupTrigger()) {
  1033. showMenu(e);
  1034. }
  1035. }
  1036. private void showMenu(MouseEvent e) {
  1037. popup.show(e.getComponent(), e.getX(), e.getY());
  1038. }
  1039. });
  1040. }
  1041. }