Viewer.py 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554
  1. '''
  2. '''
  3. from PyQt5 import uic
  4. from PyQt5.Qt import Qt, QEvent, QGraphicsScene, QPointF, QFileDialog, \
  5. QApplication, QMessageBox, QTreeWidgetItem, \
  6. QGraphicsTextItem, QGraphicsItem, QGraphicsRectItem, \
  7. QBrush, QColor, QGraphicsLineItem, QLineF, \
  8. QPen, QPainter, QSvgGenerator, QSize, QRect, QGraphicsItemGroup, \
  9. QGraphicsColorizeEffect, QFont, QDialog, QTableWidgetItem
  10. from PyQt5.QtWidgets import QMainWindow, QGraphicsView
  11. from path import Path
  12. import core
  13. Ui_window, _ = uic.loadUiType(Path(__file__).parent / 'qt_viewer.ui')
  14. Ui_details, _ = uic.loadUiType(Path(__file__).parent / 'qt_details.ui')
  15. palette = {
  16. "Table": QColor(240, 240, 20),
  17. "Query": QColor(210, 50, 210),
  18. "Module": QColor(220, 10, 33),
  19. "Relation": QColor(122, 50, 209),
  20. "Report": QColor(25, 10, 220),
  21. "Form": QColor(10, 180, 220),
  22. "Macro": QColor(40, 220, 10),
  23. }
  24. v_spacing = 120
  25. cell_width = 150
  26. cell_spacing = 25
  27. class GraphicsObject(QGraphicsItemGroup):
  28. items = []
  29. def __init__(self, obj, parent=None):
  30. super(GraphicsObject, self).__init__(parent=parent)
  31. self.obj = obj
  32. self.links = []
  33. self._x = 0
  34. self._y = 0
  35. self.setFlag(QGraphicsItem.ItemIsMovable, True)
  36. self.setFlag(QGraphicsItem.ItemIsSelectable, True)
  37. self.setFlag(QGraphicsItem.ItemIsFocusable, True)
  38. self.setAcceptHoverEvents(True)
  39. self.label = QGraphicsTextItem()
  40. self.label.setTextWidth(cell_width)
  41. self.label.setZValue(2)
  42. self.setText()
  43. self.addToGroup(self.label)
  44. pen = QPen(palette[self.obj.type_].darker(150))
  45. pen.setWidth(2)
  46. self.rect = QGraphicsRectItem(self.label.boundingRect())
  47. self.rect.setPen(pen)
  48. self.rect.setBrush(palette[self.obj.type_].lighter(150))
  49. self.rect.setZValue(0)
  50. self.addToGroup(self.rect)
  51. self.topAnchorCoords = ((self.boundingRect().topLeft().x() + self.boundingRect().topRight().x() / 2), self.boundingRect().topLeft().y())
  52. self.bottomAnchorCoords = ((self.boundingRect().topLeft().x() + self.boundingRect().topRight().x() / 2), self.boundingRect().bottomLeft().y())
  53. self.top_anchor = QGraphicsRectItem(self.topAnchorCoords[0] - 3, self.topAnchorCoords[1] - 3, 6, 6)
  54. self.bottom_anchor = QGraphicsRectItem(self.bottomAnchorCoords[0] - 3, self.bottomAnchorCoords[1] - 3, 6, 6)
  55. for anchor in (self.top_anchor, self.bottom_anchor):
  56. anchor.setBrush(QBrush(QColor(255, 153, 51)))
  57. anchor.setZValue(1)
  58. self.addToGroup(anchor)
  59. effect = QGraphicsColorizeEffect()
  60. effect.setColor(QColor(255, 255, 255))
  61. effect.setStrength(0.35)
  62. effect.setEnabled(False)
  63. self.rect.setGraphicsEffect(effect)
  64. self.setToolTip("[{}] '{}'".format(self.obj.type_, self.obj.name_))
  65. GraphicsObject.items.append(self)
  66. def setText(self):
  67. self.label.setHtml(self.html())
  68. def html(self):
  69. return "[{}]<br/><b>{}</b>".format(self.obj.type_,
  70. self.obj.name_)
  71. def topAnchorCenter(self):
  72. return self.mapToScene(QPointF(*self.topAnchorCoords))
  73. def bottomAnchorCenter(self):
  74. return self.mapToScene(QPointF(*self.bottomAnchorCoords))
  75. def update(self):
  76. if self.pos() is not QPointF(self._x, self._y):
  77. self.setPos(QPointF(self._x, self._y))
  78. for l in self.links:
  79. l.update()
  80. def paint(self, *args, **kwargs):
  81. super(GraphicsObject, self).paint(*args, **kwargs)
  82. self._x, self._y = self.pos().x(), self.pos().y()
  83. self.update()
  84. def setShining(self, active):
  85. self.rect.graphicsEffect().setEnabled(active)
  86. def hoverEnterEvent(self, *args, **kwargs):
  87. self.setShining(True)
  88. def hoverLeaveEvent(self, *args, **kwargs):
  89. self.setShining(False)
  90. class GraphicsRootObject(GraphicsObject):
  91. def __init__(self, obj, parent=None):
  92. super(GraphicsRootObject, self).__init__(obj, parent=parent)
  93. self.level = 0
  94. self.deps = []
  95. self.refs = []
  96. self._dep_emprise = 0
  97. self._ref_emprise = 0
  98. pen = QPen(QColor("red"))
  99. pen.setWidth(2)
  100. self.rect.setPen(pen)
  101. def xleft(self):
  102. return 0
  103. def compute_coords(self):
  104. return 0, 0
  105. def dep_emprise(self):
  106. if not self._dep_emprise:
  107. self._dep_emprise = sum([d.dep_emprise() for d in self.deps]) if self.deps else (cell_width + 2 * cell_spacing)
  108. return self._dep_emprise
  109. def ref_emprise(self):
  110. if not self._ref_emprise:
  111. self._ref_emprise = sum([r.ref_emprise() for r in self.refs]) if self.refs else (cell_width + 2 * cell_spacing)
  112. return self._ref_emprise
  113. class GraphicsDepObject(GraphicsObject):
  114. def __init__(self, obj, parentItem, parent=None):
  115. super(GraphicsDepObject, self).__init__(obj, parent=parent)
  116. self.deps = []
  117. self.parentItem = parentItem
  118. if parentItem:
  119. parentItem.deps.append(self)
  120. self.level = parentItem.level + 1
  121. self._dep_emprise = 0
  122. def xleft(self):
  123. x0 = sum([n.dep_emprise() for n in self.parentItem.deps[0:self.parentItem.deps.index(self)]])
  124. return self.parentItem.xleft() + x0
  125. def compute_coords(self):
  126. x0 = sum([n.dep_emprise() for n in self.parentItem.deps[0:self.parentItem.deps.index(self)]])
  127. x = self.parentItem.xleft() + (0.5 * self.dep_emprise() - (cell_width / 2 + cell_spacing)) + x0
  128. y = v_spacing * self.level
  129. return x, y
  130. def dep_emprise(self):
  131. if not self._dep_emprise:
  132. self._dep_emprise = sum([d.dep_emprise() for d in self.deps]) if self.deps else (cell_width + 2 * cell_spacing)
  133. return self._dep_emprise
  134. class GraphicsCloneDepObject(GraphicsDepObject):
  135. def __init__(self, obj, childItem, parent=None):
  136. super(GraphicsCloneDepObject, self).__init__(obj, childItem, parent)
  137. self.bottom_anchor.setBrush(QColor("red"))
  138. self.clone_of = next((item for item in GraphicsObject.items if item.obj is obj
  139. and (type(item) is GraphicsDepObject or type(item) is GraphicsRefObject)))
  140. def html(self):
  141. return "* Clone *<br/>" + super(GraphicsCloneDepObject, self).html()
  142. @property
  143. def deps(self):
  144. return []
  145. @deps.setter
  146. def deps(self, value):
  147. pass
  148. def hoverEnterEvent(self, *args, **kwargs):
  149. self.setShining(True)
  150. self.clone_of.setShining(True)
  151. def hoverLeaveEvent(self, *args, **kwargs):
  152. self.setShining(False)
  153. self.clone_of.setShining(False)
  154. class GraphicsRefObject(GraphicsObject):
  155. def __init__(self, obj, childItem, parent=None):
  156. super(GraphicsRefObject, self).__init__(obj, parent=parent)
  157. self.refs = []
  158. self.childItem = childItem
  159. if childItem:
  160. childItem.refs.append(self)
  161. self.level = childItem.level - 1
  162. self._ref_emprise = 0
  163. def xleft(self):
  164. x0 = sum([n.ref_emprise() for n in self.childItem.refs[0:self.childItem.refs.index(self)]])
  165. return self.childItem.xleft() + x0
  166. def compute_coords(self):
  167. x0 = sum([n.ref_emprise() for n in self.childItem.refs[0:self.childItem.refs.index(self)]])
  168. return self.childItem.xleft() + (0.5 * self.ref_emprise() - (cell_width / 2 + cell_spacing)) + x0, v_spacing * self.level
  169. def ref_emprise(self):
  170. if not self._ref_emprise:
  171. self._ref_emprise = sum([d.ref_emprise() for d in self.refs]) if self.refs else (cell_width + 2 * cell_spacing)
  172. return self._ref_emprise
  173. class GraphicsCloneRefObject(GraphicsRefObject):
  174. def __init__(self, obj, childItem, parent=None):
  175. super(GraphicsCloneRefObject, self).__init__(obj, childItem, parent)
  176. self.top_anchor.setBrush(QColor("red"))
  177. self.clone_of = next((item for item in GraphicsObject.items if item.obj is obj
  178. and (type(item) is GraphicsDepObject or type(item) is GraphicsRefObject)))
  179. def html(self):
  180. return "* Clone *<br/>" + super(GraphicsCloneRefObject, self).html()
  181. @property
  182. def refs(self):
  183. return []
  184. @refs.setter
  185. def refs(self, value):
  186. pass
  187. def hoverEnterEvent(self, *args, **kwargs):
  188. self.setShining(True)
  189. self.clone_of.setShining(True)
  190. def hoverLeaveEvent(self, *args, **kwargs):
  191. self.setShining(False)
  192. self.clone_of.setShining(False)
  193. class GraphicsLink(QGraphicsLineItem):
  194. items = []
  195. def __init__(self, topGraphicsObject, bottomGraphicsObject, parent=None):
  196. self.topGraphicsObject = topGraphicsObject
  197. self.topGraphicsObject.links.append(self)
  198. self.bottomGraphicsObject = bottomGraphicsObject
  199. self.bottomGraphicsObject.links.append(self)
  200. super(QGraphicsLineItem, self).__init__(parent=parent)
  201. self.update()
  202. def update(self):
  203. line = QLineF(self.mapToScene(self.topGraphicsObject.bottomAnchorCenter()), self.mapToScene(self.bottomGraphicsObject.topAnchorCenter()))
  204. self.setLine(line)
  205. class Viewer(QMainWindow):
  206. def __init__(self):
  207. super (Viewer, self).__init__()
  208. self.createWidgets()
  209. def createWidgets(self):
  210. self.ui = Ui_window()
  211. self.ui.setupUi(self)
  212. self.ui.progressBar.setVisible(False)
  213. self.ui.stackedWidget.setCurrentIndex(0)
  214. self.ui.btn_test.clicked.connect(self.test)
  215. self.ui.btn_select.clicked.connect(self.selectSourceDir)
  216. self.ui.btn_zoom_plus.clicked.connect(self.zoom_plus)
  217. self.ui.btn_zoom_minus.clicked.connect(self.zoom_minus)
  218. self.ui.btn_zoom_view.clicked.connect(self.fit_in_view)
  219. self.ui.btn_svg.clicked.connect(self.to_png)
  220. self.ui.treeWidget.itemClicked.connect(self.treeItemSelected)
  221. self.ui.btn_edit_item.clicked.connect(self.edit_selected_item)
  222. self._title = "<unknown>"
  223. core.Analyse.report = self.update_progression
  224. self.ui.view.setViewportUpdateMode(QGraphicsView.BoundingRectViewportUpdate)
  225. self.ui.view.setTransformationAnchor(QGraphicsView.AnchorUnderMouse)
  226. self.ui.view.viewport().installEventFilter(self)
  227. self._scene = QGraphicsScene()
  228. self._scene.setItemIndexMethod(QGraphicsScene.BspTreeIndex)
  229. self.ui.view.setScene(self._scene)
  230. helpLabel = QGraphicsTextItem()
  231. helpLabel.setPlainText("Sélectionnez le répertoire contenant les sources de la base à analyser,\npuis sélectionnez dans la liste de droite un objet à partir duquel afficher les dépendances...")
  232. helpLabel.setTextWidth(600)
  233. font = QFont()
  234. font.setItalic(True)
  235. font.setWeight(63)
  236. helpLabel.setFont(font)
  237. self._scene.addItem(helpLabel)
  238. def eventFilter(self, _, event):
  239. if event.type() == QEvent.Wheel:
  240. if event.angleDelta().y() > 0:
  241. self.zoom_plus()
  242. elif event.angleDelta().y() < 0:
  243. self.zoom_minus()
  244. return True
  245. return False
  246. def fit_in_view(self):
  247. rect = self._scene.itemsBoundingRect()
  248. rect.adjust(-50, -50, 50, 50)
  249. self._scene.setSceneRect(rect)
  250. self.ui.view.fitInView(self._scene.sceneRect(), Qt.KeepAspectRatio)
  251. def zoom_plus(self):
  252. self.ui.view.scale(1.2, 1.2)
  253. def zoom_minus(self):
  254. self.ui.view.scale(0.8, 0.8)
  255. def to_png(self):
  256. fileName, _ = QFileDialog.getSaveFileName(self, "Save File", "", "Svg File (*.svg)")
  257. if not fileName:
  258. return
  259. gen = QSvgGenerator()
  260. gen.setFileName(fileName)
  261. gen.setSize(QSize(1000, 1000))
  262. gen.setViewBox(QRect(0, 0, 1000, 1000))
  263. gen.setTitle("Access Analyser")
  264. gen.setDescription("Access Analyser Report for {}".format(self._title))
  265. painter = QPainter(gen)
  266. self._scene.render(painter)
  267. painter.end()
  268. def update_progression(self, i, total, msg=""):
  269. self.ui.progressBar.setMaximum(total)
  270. self.ui.progressBar.setValue(i)
  271. if msg:
  272. self.ui.txtPanel.append(msg)
  273. QApplication.processEvents()
  274. def selectSourceDir(self):
  275. source_dir = QFileDialog.getExistingDirectory(self, "Sélectionner le répertoire des sources", "", QFileDialog.ShowDirsOnly)
  276. if not source_dir:
  277. return
  278. self.run(source_dir)
  279. def run(self, source_dir):
  280. source_dir = Path(source_dir)
  281. self._title = source_dir
  282. self.ui.progressBar.setVisible(True)
  283. self.ui.lbl_repertoire.setText(source_dir)
  284. self.ui.txtPanel.clear()
  285. self.ui.treeWidget.clear()
  286. QApplication.setOverrideCursor(Qt.WaitCursor)
  287. core.Analyse.run(source_dir)
  288. QApplication.restoreOverrideCursor()
  289. QMessageBox.information(self, "Analyse", "Analyse terminée: {} objets chargés en mémoire".format(len(core.Analyse.objects)))
  290. if core.Analyse.duplicates():
  291. QMessageBox.warning(self, "Risque d'instabilités", "Attention! Des doublons ont été trouvés dans les noms des objets")
  292. self.ui.progressBar.setVisible(False)
  293. self.ui.stackedWidget.setCurrentIndex(1)
  294. self.ui.txtPanel.clear()
  295. topitem = QTreeWidgetItem()
  296. topitem.setText(0, "Objets")
  297. self.ui.treeWidget.addTopLevelItem(topitem)
  298. groupes = {}
  299. for index, obj in enumerate(core.Analyse.objects):
  300. if not obj.type_ in groupes:
  301. item = QTreeWidgetItem()
  302. item.setText(0, obj.type_)
  303. groupes[obj.type_] = item
  304. topitem.addChild(item)
  305. item = QTreeWidgetItem()
  306. item.setText(0, obj.name_)
  307. item.setData(1, 0, index)
  308. groupes[obj.type_].addChild(item)
  309. self.ui.treeWidget.setColumnHidden(1, True)
  310. self.ui.treeWidget.expandToDepth(1)
  311. def clear_scene(self):
  312. self._scene.clear()
  313. GraphicsObject.items = []
  314. def maj_scene_with(self, root_object):
  315. self.clear_scene()
  316. root = GraphicsRootObject(root_object)
  317. self._scene.addItem(root)
  318. def _addDeps(go):
  319. for dep in go.obj.deps:
  320. if dep not in [go.obj for go in GraphicsObject.items]:
  321. gdep = GraphicsDepObject(dep, go)
  322. if dep.deps:
  323. _addDeps(gdep)
  324. else:
  325. gdep = GraphicsCloneDepObject(dep, go)
  326. self._scene.addItem(gdep)
  327. link = GraphicsLink(go, gdep)
  328. self._scene.addItem(link)
  329. _addDeps(root)
  330. def _addRefs(go):
  331. for ref in go.obj.refs:
  332. if ref not in [go.obj for go in GraphicsObject.items]:
  333. gref = GraphicsRefObject(ref, go)
  334. if ref.refs:
  335. _addRefs(gref)
  336. else:
  337. gref = GraphicsCloneRefObject(ref, go)
  338. self._scene.addItem(gref)
  339. link = GraphicsLink(gref, go)
  340. self._scene.addItem(link)
  341. _addRefs(root)
  342. for item in GraphicsObject.items:
  343. item._x, item._y = item.compute_coords()
  344. if isinstance(item, GraphicsDepObject):
  345. item._x -= (root.dep_emprise() - (cell_width + 2 * cell_spacing)) / 2
  346. if isinstance(item, GraphicsRefObject):
  347. item._x -= (root.ref_emprise() - (cell_width + 2 * cell_spacing)) / 2
  348. item.update()
  349. self.ui.btn_save.setEnabled(True)
  350. self.fit_in_view()
  351. def treeItemSelected(self, item):
  352. index = item.data(1, 0)
  353. if index is None:
  354. self.ui.btn_edit_item.setEnabled(False)
  355. return
  356. obj = core.Analyse.objects[index]
  357. self.maj_scene_with(obj)
  358. self.ui.btn_edit_item.setEnabled(True)
  359. def edit_selected_item(self):
  360. index = self.ui.treeWidget.currentItem().data(1, 0)
  361. obj = core.Analyse.objects[index]
  362. dlg = DetailsDialog(obj, self)
  363. dlg.show()
  364. r = dlg.exec_()
  365. if r:
  366. pass
  367. def test(self):
  368. self.run(Path(__file__).parent / r"test\source")
  369. def keyPressEvent(self, e):
  370. if e.key() == Qt.Key_Control:
  371. self.ui.view.setDragMode(QGraphicsView.ScrollHandDrag)
  372. e.accept()
  373. def keyReleaseEvent(self, e):
  374. if e.key() == Qt.Key_Control:
  375. self.ui.view.setDragMode(QGraphicsView.RubberBandDrag)
  376. class DetailsDialog(QDialog):
  377. def __init__(self, obj, parent=None):
  378. self.obj = obj
  379. super (DetailsDialog, self).__init__(parent)
  380. self.createWidgets()
  381. def createWidgets(self):
  382. self.ui = Ui_details()
  383. self.ui.setupUi(self)
  384. self.ui.tbl_mentions.itemClicked.connect(self.itemClicked)
  385. self.ui.btn_add.clicked.connect(self.add_mention)
  386. self.ui.btn_edit.clicked.connect(self.edit_mention)
  387. self.ui.btn_del.clicked.connect(self.del_mention)
  388. self.ui.lbl_title.setText("{}: {}".format(self.obj.type_, self.obj.name_))
  389. self.ui.tbl_mentions.hideColumn(0)
  390. self.ui.tbl_mentions.setColumnWidth(1, 50)
  391. self.ui.tbl_mentions.setColumnWidth(2, 200)
  392. self.ui.tbl_mentions.setColumnWidth(3, 70)
  393. self.load_table()
  394. def load_table(self):
  395. self.ui.tbl_mentions.setSortingEnabled(False)
  396. self.ui.tbl_mentions.clearContents()
  397. self.ui.tbl_mentions.setRowCount(len(self.obj.mentions))
  398. for index, mention in enumerate(self.obj.mentions):
  399. item = QTableWidgetItem("")
  400. item.setData(0, index)
  401. self.ui.tbl_mentions.setItem(index, 0, item)
  402. item = QTableWidgetItem("{}".format(mention.line))
  403. self.ui.tbl_mentions.setItem(index, 1, item)
  404. item = QTableWidgetItem("{}".format(mention.objname))
  405. self.ui.tbl_mentions.setItem(index, 2, item)
  406. item = QTableWidgetItem("{}".format(mention.obj.type_))
  407. self.ui.tbl_mentions.setItem(index, 3, item)
  408. item = QTableWidgetItem("{}".format(mention.quote))
  409. item.setFont(QFont("Consolas", 8))
  410. self.ui.tbl_mentions.setItem(index, 4, item)
  411. self.ui.tbl_mentions.setSortingEnabled(True)
  412. def itemClicked(self):
  413. self.ui.btn_edit.setEnabled(True)
  414. self.ui.btn_del.setEnabled(True)
  415. def add_mention(self):
  416. pass
  417. def selected_index(self):
  418. return self.ui.tbl_mentions.item(self.ui.tbl_mentions.currentRow(), 0).data(0)
  419. def edit_mention(self):
  420. pass
  421. def del_mention(self):
  422. index = self.selected_index()
  423. if not QMessageBox.question(self, "Confirmation", "Etes-vous sûr de vouloir supprimer la référence sélectionnée?") == QMessageBox.Yes:
  424. return
  425. del self.obj.mentions[index]
  426. self.load_table()