im_mainwindow.py 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262
  1. from PyQt5.QtWidgets import QMainWindow, QGraphicsItem, QAction, QActionGroup, QGraphicsView, QTableWidgetItem
  2. from PyQt5.QtGui import QIcon
  3. from PyQt5.QtCore import QStateMachine, QState
  4. from PyQt5.Qt import QApplication, Qt
  5. from sketchUI.ui import Ui_MainWindow
  6. from sketchUI.im_scene import CustomScene, Mode
  7. from sketchUI import mvops
  8. from sketchUI.graphics_node_item import GraphicsNodeItem
  9. from sketchUI.graphics_edge_item import GraphicsEdgeItem
  10. from wrappers.modelverse import element_list_nice
  11. from verifier import Verifier
  12. import commons
  13. class IMMainWindow(QMainWindow, Ui_MainWindow):
  14. def __init__(self, model):
  15. QMainWindow.__init__(self)
  16. self.setupUi(self)
  17. self.setWindowTitle(model)
  18. self._cur_model = model
  19. self._scene = CustomScene(model, self)
  20. self._scene.set_mode(Mode.SELECT)
  21. self._scene.setSceneRect(0, 0, 200, 200)
  22. self.graphicsView.setScene(self._scene)
  23. self.listWidget.addItems(commons.get_available_types())
  24. self.listWidget.itemDoubleClicked.connect(self._on_list_item_clicked)
  25. self.setup_toolbar()
  26. self.setup_state_machine()
  27. self.menuActionVerify.triggered.connect(self._on_verify_clicked)
  28. # load the model
  29. self._load_model()
  30. # setup log viewer
  31. self.plainTextEdit.setReadOnly(True)
  32. # setup table view for attributes
  33. self.tableWidget.setColumnCount(2)
  34. self.tableWidget.setHorizontalHeaderLabels(["Key", "Value"])
  35. self.tableWidget.horizontalHeader().setStretchLastSection(True)
  36. self.tableWidget.itemChanged.connect(self._on_attribute_edited)
  37. #lastly, start the state machine
  38. self._statemachine.start()
  39. def setup_toolbar(self):
  40. self.select_action = QAction("Select", self)
  41. self.select_action.setIcon(QIcon("sketchUI/icons/select.png"))
  42. self.select_action.setCheckable(True)
  43. self.select_action.setChecked(True)
  44. self.connect_action = QAction("Connect", self)
  45. self.connect_action.setIcon(QIcon("sketchUI/icons/connect.png"))
  46. self.connect_action.setCheckable(True)
  47. action_group = QActionGroup(self)
  48. action_group.setExclusive(True)
  49. action_group.addAction(self.select_action)
  50. action_group.addAction(self.connect_action)
  51. for item in action_group.actions():
  52. self.toolBar.addAction(item)
  53. def setup_state_machine(self):
  54. self._statemachine = QStateMachine()
  55. state_select = QState()
  56. state_connect = QState()
  57. state_select.addTransition(self.connect_action.triggered, state_connect)
  58. state_connect.addTransition(self.select_action.triggered, state_select)
  59. state_connect.entered.connect(self._state_connect_entered)
  60. state_select.entered.connect(self._state_select_entered)
  61. self._statemachine.addState(state_select)
  62. self._statemachine.addState(state_connect)
  63. self._statemachine.setInitialState(state_select)
  64. def _load_model(self):
  65. """
  66. Load the model from the mv and render to screen using the concrete syntaxes stored in the mv
  67. """
  68. model = element_list_nice(self._cur_model)
  69. if not model:
  70. # empty model
  71. return
  72. for item in model:
  73. typ = item["type"]
  74. if typ == "Node":
  75. # first, draw all nodes
  76. node_type = item["typeID"]
  77. self._add_node_to_scene(item["id"], node_type)
  78. for item in model:
  79. # now the edges
  80. typ = item["type"]
  81. if typ == "Edge":
  82. target = item["__target"]
  83. src = item["__source"]
  84. self._add_edge_to_scene(src, target, item["id"])
  85. def _state_connect_entered(self):
  86. self._scene.set_mode(Mode.CONNECT)
  87. self._make_items_movable(False)
  88. self._enable_box_select(False)
  89. self._enable_list_widget(False)
  90. def _state_select_entered(self):
  91. self._scene.set_mode(Mode.SELECT)
  92. self._make_items_movable(True)
  93. self._enable_box_select(True)
  94. self._enable_list_widget(True)
  95. def _enable_list_widget(self, enabled):
  96. self.listWidget.setEnabled(enabled)
  97. def _make_items_movable(self, movable):
  98. for item in self._scene.items():
  99. if isinstance(item, GraphicsEdgeItem):
  100. # edges are selectable (for delete) but never movable
  101. item.setFlag(QGraphicsItem.ItemIsMovable, False)
  102. item.setFlag(QGraphicsItem.ItemIsSelectable, True)
  103. else:
  104. item.setFlag(QGraphicsItem.ItemIsMovable, movable)
  105. item.setFlag(QGraphicsItem.ItemIsSelectable, movable)
  106. def _enable_box_select(self, enable):
  107. if enable:
  108. self.graphicsView.setDragMode(QGraphicsView.RubberBandDrag)
  109. else:
  110. self.graphicsView.setDragMode(QGraphicsView.NoDrag)
  111. def _add_node_to_scene(self, node_id, node_type, x=0, y=0):
  112. """
  113. Render a node with id and type to the canvas by getting its concrete syntax
  114. from the modelverse.
  115. """
  116. consyn = mvops.get_consyn_of(node_type)
  117. item = GraphicsNodeItem(node_id, node_type, consyn)
  118. item.setPos(x, y)
  119. item.setFlag(QGraphicsItem.ItemIsMovable, True)
  120. item.setFlag(QGraphicsItem.ItemIsSelectable, True)
  121. self._scene.addItem(item)
  122. def _add_edge_to_scene(self, from_id, to_id, edge_id):
  123. from_item = None
  124. to_item = None
  125. for item in self._scene.items():
  126. try:
  127. node_id = item.node_id
  128. except AttributeError:
  129. # no node item, continue
  130. continue
  131. if node_id == from_id:
  132. from_item = item
  133. continue
  134. if node_id == to_id:
  135. to_item = item
  136. self._scene.draw_edge(from_item, to_item, is_new=False, edge_id=edge_id)
  137. def _on_list_item_clicked(self, event):
  138. # add new node to model in mv
  139. node_id = mvops.add_node(self._cur_model, event.text())
  140. # render to scene
  141. self._add_node_to_scene(node_id, event.text())
  142. self.plainTextEdit.appendPlainText("Added node of type {} to model".format(event.text()))
  143. def _on_verify_clicked(self, event):
  144. self.plainTextEdit.appendPlainText("Verifying instance model against example models ...")
  145. self.plainTextEdit.repaint()
  146. QApplication.setOverrideCursor(Qt.WaitCursor)
  147. verify = Verifier(self._cur_model)
  148. verify.init()
  149. self.plainTextEdit.appendPlainText("Verify: Checking node typing ...")
  150. self.plainTextEdit.repaint()
  151. ret = verify.verify_node_typing()
  152. if not ret["OK"]:
  153. self.plainTextEdit.appendPlainText("Error: {}".format(str(ret["error"])))
  154. QApplication.restoreOverrideCursor()
  155. return
  156. else:
  157. self.plainTextEdit.appendPlainText("OK")
  158. self.plainTextEdit.repaint()
  159. self.plainTextEdit.appendPlainText("Verify: Checking node multiplicity ...")
  160. self.plainTextEdit.repaint()
  161. ret = verify.verify_node_multiplicity()
  162. if not ret["OK"]:
  163. self.plainTextEdit.appendPlainText("Error: {}".format(str(ret["error"])))
  164. QApplication.restoreOverrideCursor()
  165. return
  166. else:
  167. self.plainTextEdit.appendPlainText("OK")
  168. self.plainTextEdit.repaint()
  169. self.plainTextEdit.appendPlainText("Verify: Checking attributes ...")
  170. self.plainTextEdit.repaint()
  171. ret = verify.verify_attributes()
  172. if not ret["OK"]:
  173. self.plainTextEdit.appendPlainText("Error: {}".format(str(ret["error"])))
  174. QApplication.restoreOverrideCursor()
  175. return
  176. else:
  177. self.plainTextEdit.appendPlainText("OK")
  178. self.plainTextEdit.repaint()
  179. self.plainTextEdit.appendPlainText("Verify: Checking edges ...")
  180. self.plainTextEdit.repaint()
  181. ret = verify.verify_associations()
  182. if not ret["OK"]:
  183. self.plainTextEdit.appendPlainText("Error: {}".format(str(ret["error"])))
  184. QApplication.restoreOverrideCursor()
  185. return
  186. else:
  187. self.plainTextEdit.appendPlainText("OK")
  188. self.plainTextEdit.repaint()
  189. self.plainTextEdit.appendPlainText("Verify OK")
  190. QApplication.restoreOverrideCursor()
  191. def _on_attribute_edited(self, item):
  192. # type: (QTableWidgetItem) -> None
  193. """ An attribute was edited, change it in the model but do not check (too expensive so
  194. checking is done by verify method on demand).
  195. If the new entered value is empty, delete the attribute.
  196. """
  197. row = self.tableWidget.row(item)
  198. attr_key = self.tableWidget.item(row, 0).text()
  199. attr_val = self.tableWidget.item(row, 1).text()
  200. node = self._scene.selectedItems()[0]
  201. if not attr_val:
  202. self.plainTextEdit.appendPlainText("Deleting attribute {}".format(attr_key))
  203. mvops.delete_attribute_from_node(self._cur_model, node.node_id, attr_key)
  204. self.tableWidget.removeRow(row)
  205. else:
  206. self.plainTextEdit.appendPlainText("Updating value of attribute {} to {}".format(attr_key, attr_val))
  207. mvops.update_attribute_val(self._cur_model, node.node_id, attr_key, attr_val)
  208. def add_new_attribute(self, key, val="unknown"):
  209. """
  210. Adds a new attribute to the view with key "key" and optional val. Also adds this attribute to the modelverse
  211. model.
  212. """
  213. selected_node = self._scene.selectedItems()[0]
  214. self.plainTextEdit.appendPlainText("Adding new attribute with key {} to node {}".format(key, selected_node.get_type()))
  215. self.tableWidget.blockSignals(True)
  216. table_item_key = QTableWidgetItem(key)
  217. table_item_key.setFlags(Qt.ItemIsSelectable | Qt.ItemIsEnabled)
  218. table_item_val = QTableWidgetItem(val)
  219. cur_row_cnt = self.tableWidget.rowCount()
  220. self.tableWidget.insertRow(cur_row_cnt)
  221. self.tableWidget.setItem(cur_row_cnt, 0, table_item_key)
  222. self.tableWidget.setItem(cur_row_cnt, 1, table_item_val)
  223. self.tableWidget.blockSignals(False)
  224. # add to modelverse
  225. mvops.add_attribute(self._cur_model, selected_node.node_id, key, val)