Viewer.py 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679
  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. QListWidgetItem, QInputDialog, QIcon, QPixmap
  11. from PyQt5.QtWidgets import QMainWindow, QGraphicsView
  12. from path import Path
  13. from core import Mention
  14. import core
  15. Ui_window, _ = uic.loadUiType(Path(__file__).parent / 'qt_viewer.ui')
  16. Ui_details, _ = uic.loadUiType(Path(__file__).parent / 'qt_details.ui')
  17. Ui_select, _ = uic.loadUiType(Path(__file__).parent / 'qt_select_object.ui')
  18. palette = {
  19. "Table": QColor(240, 240, 20),
  20. "Query": QColor(210, 50, 210),
  21. "Module": QColor(220, 10, 33),
  22. "Relation": QColor(122, 50, 209),
  23. "Report": QColor(25, 10, 220),
  24. "Form": QColor(10, 180, 220),
  25. "Macro": QColor(40, 220, 10),
  26. }
  27. v_spacing = 120
  28. cell_width = 150
  29. cell_spacing = 25
  30. class GraphicsObject(QGraphicsItemGroup):
  31. items = []
  32. def __init__(self, obj, parent=None):
  33. super(GraphicsObject, self).__init__(parent=parent)
  34. self.obj = obj
  35. self.links = []
  36. self._x = 0
  37. self._y = 0
  38. self.setFlag(QGraphicsItem.ItemIsMovable, True)
  39. self.setFlag(QGraphicsItem.ItemIsSelectable, True)
  40. self.setFlag(QGraphicsItem.ItemIsFocusable, True)
  41. self.setAcceptHoverEvents(True)
  42. self.label = QGraphicsTextItem()
  43. self.label.setTextWidth(cell_width)
  44. self.label.setZValue(2)
  45. self.setText()
  46. self.addToGroup(self.label)
  47. pen = QPen(palette[self.obj.type_].darker(150))
  48. pen.setWidth(2)
  49. self.rect = QGraphicsRectItem(self.label.boundingRect())
  50. self.rect.setPen(pen)
  51. self.rect.setBrush(palette[self.obj.type_].lighter(150))
  52. self.rect.setZValue(0)
  53. self.addToGroup(self.rect)
  54. self.topAnchorCoords = ((self.boundingRect().topLeft().x() + self.boundingRect().topRight().x() / 2), self.boundingRect().topLeft().y())
  55. self.bottomAnchorCoords = ((self.boundingRect().topLeft().x() + self.boundingRect().topRight().x() / 2), self.boundingRect().bottomLeft().y())
  56. self.top_anchor = QGraphicsRectItem(self.topAnchorCoords[0] - 3, self.topAnchorCoords[1] - 3, 6, 6)
  57. self.bottom_anchor = QGraphicsRectItem(self.bottomAnchorCoords[0] - 3, self.bottomAnchorCoords[1] - 3, 6, 6)
  58. for anchor in (self.top_anchor, self.bottom_anchor):
  59. anchor.setBrush(QBrush(QColor(255, 153, 51)))
  60. anchor.setZValue(1)
  61. self.addToGroup(anchor)
  62. effect = QGraphicsColorizeEffect()
  63. effect.setColor(QColor(255, 255, 255))
  64. effect.setStrength(0.35)
  65. effect.setEnabled(False)
  66. self.rect.setGraphicsEffect(effect)
  67. self.setToolTip("[{}] '{}'".format(self.obj.type_, self.obj.name_))
  68. GraphicsObject.items.append(self)
  69. def setText(self):
  70. self.label.setHtml(self.html())
  71. def html(self):
  72. return "[{}]<br/><b>{}</b>".format(self.obj.type_,
  73. self.obj.name_)
  74. def topAnchorCenter(self):
  75. return self.mapToScene(QPointF(*self.topAnchorCoords))
  76. def bottomAnchorCenter(self):
  77. return self.mapToScene(QPointF(*self.bottomAnchorCoords))
  78. def update(self):
  79. if self.pos() is not QPointF(self._x, self._y):
  80. self.setPos(QPointF(self._x, self._y))
  81. for l in self.links:
  82. l.update()
  83. def paint(self, *args, **kwargs):
  84. super(GraphicsObject, self).paint(*args, **kwargs)
  85. self._x, self._y = self.pos().x(), self.pos().y()
  86. self.update()
  87. def mouseReleaseEvent(self, _):
  88. self.obj._custom_position = (self.pos().x(), self.pos().y())
  89. def setShining(self, active):
  90. self.rect.graphicsEffect().setEnabled(active)
  91. def hoverEnterEvent(self, _):
  92. self.setShining(True)
  93. def hoverLeaveEvent(self, _):
  94. self.setShining(False)
  95. class GraphicsRootObject(GraphicsObject):
  96. def __init__(self, obj, parent=None):
  97. super(GraphicsRootObject, self).__init__(obj, parent=parent)
  98. self.level = 0
  99. self.deps = []
  100. self.refs = []
  101. self._dep_emprise = 0
  102. self._ref_emprise = 0
  103. pen = QPen(QColor("red"))
  104. pen.setWidth(2)
  105. self.rect.setPen(pen)
  106. def xleft(self):
  107. return 0
  108. def compute_coords(self):
  109. return 0, 0
  110. def dep_emprise(self):
  111. if not self._dep_emprise:
  112. self._dep_emprise = sum([d.dep_emprise() for d in self.deps]) if self.deps else (cell_width + 2 * cell_spacing)
  113. return self._dep_emprise
  114. def ref_emprise(self):
  115. if not self._ref_emprise:
  116. self._ref_emprise = sum([r.ref_emprise() for r in self.refs]) if self.refs else (cell_width + 2 * cell_spacing)
  117. return self._ref_emprise
  118. class GraphicsDepObject(GraphicsObject):
  119. def __init__(self, obj, parentItem, parent=None):
  120. super(GraphicsDepObject, self).__init__(obj, parent=parent)
  121. self.deps = []
  122. self.parentItem = parentItem
  123. if parentItem:
  124. parentItem.deps.append(self)
  125. self.level = parentItem.level + 1
  126. self._dep_emprise = 0
  127. def xleft(self):
  128. x0 = sum([n.dep_emprise() for n in self.parentItem.deps[0:self.parentItem.deps.index(self)]])
  129. return self.parentItem.xleft() + x0
  130. def compute_coords(self):
  131. x0 = sum([n.dep_emprise() for n in self.parentItem.deps[0:self.parentItem.deps.index(self)]])
  132. x = self.parentItem.xleft() + (0.5 * self.dep_emprise() - (cell_width / 2 + cell_spacing)) + x0
  133. y = v_spacing * self.level
  134. return x, y
  135. def dep_emprise(self):
  136. if not self._dep_emprise:
  137. self._dep_emprise = sum([d.dep_emprise() for d in self.deps]) if self.deps else (cell_width + 2 * cell_spacing)
  138. return self._dep_emprise
  139. class GraphicsCloneDepObject(GraphicsDepObject):
  140. def __init__(self, obj, childItem, parent=None):
  141. super(GraphicsCloneDepObject, self).__init__(obj, childItem, parent)
  142. self.bottom_anchor.setBrush(QColor("red"))
  143. self.clone_of = next((item for item in GraphicsObject.items if item.obj is obj
  144. and (type(item) is GraphicsDepObject or type(item) is GraphicsRefObject)))
  145. self.setToolTip("L'objet [{}] '{}' a déjà été placé sur l'arbre".format(self.obj.type_, self.obj.name_))
  146. def html(self):
  147. return "* Clone *<br/>" + super(GraphicsCloneDepObject, self).html()
  148. @property
  149. def deps(self):
  150. return []
  151. @deps.setter
  152. def deps(self, value):
  153. pass
  154. def hoverEnterEvent(self, _):
  155. self.setShining(True)
  156. self.clone_of.setShining(True)
  157. def hoverLeaveEvent(self, _):
  158. self.setShining(False)
  159. self.clone_of.setShining(False)
  160. class GraphicsRefObject(GraphicsObject):
  161. def __init__(self, obj, childItem, parent=None):
  162. super(GraphicsRefObject, self).__init__(obj, parent=parent)
  163. self.refs = []
  164. self.childItem = childItem
  165. if childItem:
  166. childItem.refs.append(self)
  167. self.level = childItem.level - 1
  168. self._ref_emprise = 0
  169. def xleft(self):
  170. x0 = sum([n.ref_emprise() for n in self.childItem.refs[0:self.childItem.refs.index(self)]])
  171. return self.childItem.xleft() + x0
  172. def compute_coords(self):
  173. x0 = sum([n.ref_emprise() for n in self.childItem.refs[0:self.childItem.refs.index(self)]])
  174. return self.childItem.xleft() + (0.5 * self.ref_emprise() - (cell_width / 2 + cell_spacing)) + x0, v_spacing * self.level
  175. def ref_emprise(self):
  176. if not self._ref_emprise:
  177. self._ref_emprise = sum([d.ref_emprise() for d in self.refs]) if self.refs else (cell_width + 2 * cell_spacing)
  178. return self._ref_emprise
  179. class GraphicsCloneRefObject(GraphicsRefObject):
  180. def __init__(self, obj, childItem, parent=None):
  181. super(GraphicsCloneRefObject, self).__init__(obj, childItem, parent)
  182. self.top_anchor.setBrush(QColor("red"))
  183. self.clone_of = next((item for item in GraphicsObject.items if item.obj is obj
  184. and (type(item) is GraphicsDepObject or type(item) is GraphicsRefObject)))
  185. self.setToolTip("L'objet [{}] '{}' a déjà été placé sur l'arbre".format(self.obj.type_, self.obj.name_))
  186. def html(self):
  187. return "* Clone *<br/>" + super(GraphicsCloneRefObject, self).html()
  188. @property
  189. def refs(self):
  190. return []
  191. @refs.setter
  192. def refs(self, value):
  193. pass
  194. def hoverEnterEvent(self, _):
  195. self.setShining(True)
  196. self.clone_of.setShining(True)
  197. def hoverLeaveEvent(self, _):
  198. self.setShining(False)
  199. self.clone_of.setShining(False)
  200. class GraphicsLink(QGraphicsLineItem):
  201. items = []
  202. def __init__(self, topGraphicsObject, bottomGraphicsObject, parent=None):
  203. self.topGraphicsObject = topGraphicsObject
  204. self.topGraphicsObject.links.append(self)
  205. self.bottomGraphicsObject = bottomGraphicsObject
  206. self.bottomGraphicsObject.links.append(self)
  207. super(QGraphicsLineItem, self).__init__(parent=parent)
  208. self.update()
  209. def update(self):
  210. line = QLineF(self.mapToScene(self.topGraphicsObject.bottomAnchorCenter()), self.mapToScene(self.bottomGraphicsObject.topAnchorCenter()))
  211. self.setLine(line)
  212. class Viewer(QMainWindow):
  213. def __init__(self):
  214. super (Viewer, self).__init__()
  215. self.createWidgets()
  216. def createWidgets(self):
  217. self.ui = Ui_window()
  218. self.ui.setupUi(self)
  219. self.ui.progressBar.setVisible(False)
  220. self.ui.stackedWidget.setCurrentIndex(0)
  221. self.ui.btn_test.clicked.connect(self.test)
  222. self.ui.btn_select.clicked.connect(self.selectSourceDir)
  223. self.ui.btn_zoom_plus.clicked.connect(self.zoom_plus)
  224. self.ui.btn_zoom_minus.clicked.connect(self.zoom_minus)
  225. self.ui.btn_zoom_view.clicked.connect(self.fit_in_view)
  226. self.ui.btn_svg.clicked.connect(self.to_svg)
  227. self.ui.btn_save.clicked.connect(self.save)
  228. self.ui.btn_load.clicked.connect(self.open)
  229. self.ui.treeWidget.itemClicked.connect(self.treeItemSelected)
  230. self.ui.btn_edit_item.clicked.connect(self.edit_selected_item)
  231. self._title = "<unknown>"
  232. core.Analyse.report = self.update_progression
  233. self.ui.view.setViewportUpdateMode(QGraphicsView.BoundingRectViewportUpdate)
  234. self.ui.view.setTransformationAnchor(QGraphicsView.AnchorUnderMouse)
  235. self.ui.view.viewport().installEventFilter(self)
  236. self._scene = QGraphicsScene()
  237. self._scene.setItemIndexMethod(QGraphicsScene.BspTreeIndex)
  238. self.ui.view.setScene(self._scene)
  239. helpLabel = QGraphicsTextItem()
  240. 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...")
  241. helpLabel.setTextWidth(600)
  242. font = QFont()
  243. font.setItalic(True)
  244. font.setWeight(63)
  245. helpLabel.setFont(font)
  246. self._scene.addItem(helpLabel)
  247. def eventFilter(self, _, event):
  248. if event.type() == QEvent.Wheel:
  249. if event.angleDelta().y() > 0:
  250. self.zoom_plus()
  251. elif event.angleDelta().y() < 0:
  252. self.zoom_minus()
  253. return True
  254. return False
  255. def fit_in_view(self):
  256. rect = self._scene.itemsBoundingRect()
  257. rect.adjust(-50, -50, 50, 50)
  258. self._scene.setSceneRect(rect)
  259. self.ui.view.fitInView(self._scene.sceneRect(), Qt.KeepAspectRatio)
  260. def zoom_plus(self):
  261. self.ui.view.scale(1.2, 1.2)
  262. def zoom_minus(self):
  263. self.ui.view.scale(0.8, 0.8)
  264. def to_svg(self):
  265. fileName, _ = QFileDialog.getSaveFileName(self, "Save File", "", "Svg File (*.svg)")
  266. if not fileName:
  267. return
  268. gen = QSvgGenerator()
  269. gen.setFileName(fileName)
  270. gen.setSize(QSize(1000, 1000))
  271. gen.setViewBox(QRect(0, 0, 1000, 1000))
  272. gen.setTitle("Access Analyser")
  273. gen.setDescription("Access Analyser Report for {}".format(self._title))
  274. painter = QPainter(gen)
  275. self._scene.render(painter)
  276. painter.end()
  277. def save(self):
  278. filepath, _ = QFileDialog.getSaveFileName(self, 'Enregistrer sous', '', "Access Analyser Files (*.acan)")
  279. if filepath:
  280. core.Analyse.dump_to(filepath)
  281. def open(self):
  282. filepath, _ = QFileDialog.getOpenFileName(self, 'Charger les données de', '', "Access Analyser Files (*.acan)")
  283. if filepath:
  284. core.Analyse.load_from(filepath)
  285. self.load()
  286. def update_progression(self, i, total, msg=""):
  287. self.ui.progressBar.setMaximum(total)
  288. self.ui.progressBar.setValue(i)
  289. if msg:
  290. self.ui.txtPanel.append(msg)
  291. QApplication.processEvents()
  292. def selectSourceDir(self):
  293. source_dir = QFileDialog.getExistingDirectory(self, "Sélectionner le répertoire des sources", "", QFileDialog.ShowDirsOnly)
  294. if not source_dir:
  295. return
  296. self.run(source_dir)
  297. def run(self, source_dir):
  298. source_dir = Path(source_dir)
  299. self._title = source_dir
  300. self.ui.progressBar.setVisible(True)
  301. self.ui.lbl_repertoire.setText(source_dir)
  302. self.ui.txtPanel.clear()
  303. QApplication.setOverrideCursor(Qt.WaitCursor)
  304. core.Analyse.run(source_dir)
  305. QApplication.restoreOverrideCursor()
  306. QMessageBox.information(self, "Analyse", "Analyse terminée: {} objets chargés en mémoire".format(len(core.Analyse.objects)))
  307. if core.Analyse.duplicates():
  308. QMessageBox.warning(self, "Risque d'instabilités", "Attention! Des doublons ont été trouvés dans les noms des objets")
  309. self.ui.progressBar.setVisible(False)
  310. self.ui.txtPanel.clear()
  311. self.load()
  312. def load(self):
  313. self.clear_scene()
  314. self.ui.treeWidget.clear()
  315. self.ui.stackedWidget.setCurrentIndex(1)
  316. topitem = QTreeWidgetItem()
  317. topitem.setText(0, "Objets")
  318. self.ui.treeWidget.addTopLevelItem(topitem)
  319. groupes = {}
  320. for index, obj in enumerate(core.Analyse.objects):
  321. if not obj.type_ in groupes:
  322. item = QTreeWidgetItem()
  323. item.setText(0, obj.type_)
  324. groupes[obj.type_] = item
  325. topitem.addChild(item)
  326. item = QTreeWidgetItem()
  327. item.setText(0, obj.name_)
  328. item.setData(1, 0, index)
  329. groupes[obj.type_].addChild(item)
  330. self.ui.treeWidget.setColumnHidden(1, True)
  331. self.ui.treeWidget.expandToDepth(1)
  332. self.ui.btn_save.setEnabled(True)
  333. def clear_scene(self):
  334. self._scene.clear()
  335. GraphicsObject.items = []
  336. def maj_scene_with(self, root_object):
  337. self.clear_scene()
  338. root = GraphicsRootObject(root_object)
  339. self._scene.addItem(root)
  340. def _addDeps(go):
  341. for dep in go.obj.deps:
  342. if dep not in [go.obj for go in GraphicsObject.items]:
  343. gdep = GraphicsDepObject(dep, go)
  344. if dep.deps:
  345. _addDeps(gdep)
  346. else:
  347. gdep = GraphicsCloneDepObject(dep, go)
  348. self._scene.addItem(gdep)
  349. link = GraphicsLink(go, gdep)
  350. self._scene.addItem(link)
  351. _addDeps(root)
  352. def _addRefs(go):
  353. for ref in go.obj.refs:
  354. if ref not in [go.obj for go in GraphicsObject.items]:
  355. gref = GraphicsRefObject(ref, go)
  356. if ref.refs:
  357. _addRefs(gref)
  358. else:
  359. gref = GraphicsCloneRefObject(ref, go)
  360. self._scene.addItem(gref)
  361. link = GraphicsLink(gref, go)
  362. self._scene.addItem(link)
  363. _addRefs(root)
  364. for item in GraphicsObject.items:
  365. item._x, item._y = item.compute_coords()
  366. if isinstance(item, GraphicsDepObject):
  367. item._x -= (root.dep_emprise() - (cell_width + 2 * cell_spacing)) / 2
  368. if isinstance(item, GraphicsRefObject):
  369. item._x -= (root.ref_emprise() - (cell_width + 2 * cell_spacing)) / 2
  370. item.update()
  371. # # on applique les positions customisees dans un deuxieme temps: plus propre en cas d'erreur
  372. # for item in GraphicsObject.items:
  373. # if hasattr(item.obj, "_custom_position"):
  374. # item._x, item._y = item.obj._custom_position
  375. # item.update()
  376. self.fit_in_view()
  377. def treeItemSelected(self, item):
  378. index = item.data(1, 0)
  379. if index is None:
  380. self.ui.btn_edit_item.setEnabled(False)
  381. return
  382. obj = core.Analyse.objects[index]
  383. self.maj_scene_with(obj)
  384. self.ui.btn_edit_item.setEnabled(True)
  385. def edit_selected_item(self):
  386. index = self.ui.treeWidget.currentItem().data(1, 0)
  387. obj = core.Analyse.objects[index]
  388. dlg = DetailsDialog(obj, self)
  389. dlg.show()
  390. r = dlg.exec_()
  391. if r:
  392. core.Analyse.build_trees()
  393. self.maj_scene_with(obj)
  394. def test(self):
  395. self.run(Path(__file__).parent / r"test\source")
  396. def keyPressEvent(self, e):
  397. if e.key() == Qt.Key_Control:
  398. self.ui.view.setDragMode(QGraphicsView.ScrollHandDrag)
  399. e.accept()
  400. def keyReleaseEvent(self, e):
  401. if e.key() == Qt.Key_Control:
  402. self.ui.view.setDragMode(QGraphicsView.RubberBandDrag)
  403. class DetailsDialog(QDialog):
  404. def __init__(self, obj, parent=None):
  405. self.obj = obj
  406. super (DetailsDialog, self).__init__(parent)
  407. self.createWidgets()
  408. def createWidgets(self):
  409. self.ui = Ui_details()
  410. self.ui.setupUi(self)
  411. self.ui.tbl_mentions.itemClicked.connect(self.itemClicked)
  412. self.ui.btn_add.clicked.connect(self.add_mention)
  413. self.ui.btn_edit.clicked.connect(self.edit_mention)
  414. self.ui.btn_del.clicked.connect(self.del_mention)
  415. self.ui.btn_ok.clicked.connect(self.ok)
  416. self.ui.lbl_title.setText("{}: {}".format(self.obj.type_, self.obj.name_))
  417. self.ui.tbl_mentions.hideColumn(0)
  418. self.ui.tbl_mentions.setColumnWidth(1, 50)
  419. self.ui.tbl_mentions.setColumnWidth(2, 70)
  420. self.ui.tbl_mentions.setColumnWidth(3, 150)
  421. self.load_table()
  422. def load_table(self):
  423. self.ui.tbl_mentions.setSortingEnabled(False)
  424. self.ui.tbl_mentions.clearContents()
  425. self.ui.tbl_mentions.setRowCount(len(self.obj.mentions))
  426. for index, mention in enumerate(self.obj.mentions):
  427. item = QTableWidgetItem("")
  428. if mention.warning:
  429. item.setIcon(QIcon(QPixmap(core.here / "rsc\\warning_16.png")))
  430. item.setToolTip(mention.warning)
  431. self.ui.tbl_mentions.setVerticalHeaderItem(index, item)
  432. item = QTableWidgetItem("")
  433. item.setData(0, index)
  434. self.ui.tbl_mentions.setItem(index, 0, item)
  435. item = QTableWidgetItem("{}".format(mention.line))
  436. self.ui.tbl_mentions.setItem(index, 1, item)
  437. item = QTableWidgetItem("{}".format(mention.obj.type_))
  438. self.ui.tbl_mentions.setItem(index, 2, item)
  439. item = QTableWidgetItem("{}".format(mention.objname))
  440. self.ui.tbl_mentions.setItem(index, 3, item)
  441. item = QTableWidgetItem("{}".format(mention.quote))
  442. item.setFont(QFont("Consolas", 8))
  443. self.ui.tbl_mentions.setItem(index, 4, item)
  444. self.ui.tbl_mentions.setSortingEnabled(True)
  445. def itemClicked(self):
  446. self.ui.btn_edit.setEnabled(True)
  447. self.ui.btn_del.setEnabled(True)
  448. def add_mention(self):
  449. dlg = ObjectSelectorDialog(parent=self)
  450. dlg.show()
  451. r = dlg.exec_()
  452. if not r:
  453. return
  454. mention = Mention(0, dlg.obj.name_, "(ajouté manuellement)", dlg.obj)
  455. linenum, _ = QInputDialog.getInt(self, "Ligne", "Numéro de ligne correspondant dans le code-source? (facultatif)", 0)
  456. if linenum > 0:
  457. mention.line = linenum
  458. try:
  459. with open(self.obj.sourcefile, "r", encoding='utf-8') as f:
  460. mention.quote = next((line.strip() for i, line in enumerate(f) if i == linenum - 1)).strip()
  461. except FileNotFoundError:
  462. pass
  463. self.obj.mentions.append(mention)
  464. self.load_table()
  465. def selected_index(self):
  466. row = self.ui.tbl_mentions.currentRow()
  467. if row < 0:
  468. return None
  469. return self.ui.tbl_mentions.item(row, 0).data(0)
  470. def edit_mention(self):
  471. row = self.ui.tbl_mentions.currentRow()
  472. if row < 0:
  473. return
  474. item = QTableWidgetItem("")
  475. item.setIcon(QIcon(QPixmap(core.here / "rsc\\edit_16.png")))
  476. self.ui.tbl_mentions.setVerticalHeaderItem(row, item)
  477. index = self.ui.tbl_mentions.item(row, 0).data(0)
  478. mention = self.obj.mentions[index]
  479. dlg = ObjectSelectorDialog(filterStr=mention.objname, parent=self)
  480. dlg.show()
  481. r = dlg.exec_()
  482. if r:
  483. mention.obj = dlg.obj
  484. mention.warning = ""
  485. self.load_table()
  486. def del_mention(self):
  487. index = self.selected_index()
  488. if not QMessageBox.question(self, "Confirmation", "Etes-vous sûr de vouloir supprimer la référence sélectionnée?") == QMessageBox.Yes:
  489. return
  490. del self.obj.mentions[index]
  491. self.load_table()
  492. def ok(self):
  493. self.done(1)
  494. class ObjectSelectorDialog(QDialog):
  495. def __init__(self, filterStr="", parent=None):
  496. self.obj = None
  497. super (ObjectSelectorDialog, self).__init__(parent)
  498. self.createWidgets()
  499. self.ui.searchBox.setText(filterStr)
  500. def createWidgets(self):
  501. self.ui = Ui_select()
  502. self.ui.setupUi(self)
  503. for obj in core.Analyse.objects:
  504. item = QListWidgetItem("{} - {}".format(obj.type_, obj.name_))
  505. item.obj = obj
  506. self.ui.listWidget.addItem(item)
  507. self.ui.searchBox.textChanged.connect(self.filter)
  508. self.ui.listWidget.itemClicked.connect(self.itemClicked)
  509. self.ui.listWidget.itemDoubleClicked.connect(self.itemDoubleClicked)
  510. self.ui.btn_ok.clicked.connect(self.ok)
  511. def itemClicked(self, item):
  512. self.obj = item.obj
  513. self.ui.btn_ok.setEnabled(True)
  514. def itemDoubleClicked(self, item):
  515. self.itemClicked(item)
  516. self.ok()
  517. def filter(self, filterStr):
  518. for i in range(self.ui.listWidget.count()):
  519. item = self.ui.listWidget.item(i)
  520. item.setHidden(filterStr.lower() not in item.text().lower())
  521. def ok(self):
  522. self.done(1)
  523. def done(self, status):
  524. if not status:
  525. self.obj = None
  526. super(ObjectSelectorDialog, self).done(status)