Bonjour,
je voudrais qu'au passage de la souris au dessus d'un noeud apparaisse une image ou à défaut du texte. Quelqu'un serait-il faire cela ?
Toute info. est la bienvenue.
Bonjour,
je voudrais qu'au passage de la souris au dessus d'un noeud apparaisse une image ou à défaut du texte. Quelqu'un serait-il faire cela ?
Toute info. est la bienvenue.
J'ai un peu pensé a ton idée. Ca ne me parait pas tres simple a faire.
Cette partie de la représentation de l'arbre dépend directement du QTreeView, je dirais qu'un moyen (simple ?) c'est d'utiliser un Delegate dans lequel tu surcharge la méthode paint() pour qu'elle rajoute une QFrame au dessus du '+' avec une tooltip. Il ne faut pas que cette Frame n'intercepte les événements pour ne pas troubler le comportement de QTreeView dessous.
Je ne sais pas si je me suis bien fais comprendre. Enfin c'est comme ca que j'essaierai de faire. Sinon il faut comprendre en détail comment est dessiné le QTree et changer l'affichage directement dedans pour ajouter ce que tu veux.
Hum je te suis difficilement. Je n'ai pas eu pour le moment à devoir faire ce type de choses.
En fait, je me disais que ce post devrait donner une méthode consistant à passer par la gestion de l'évènement "Menu Contextuel" (on n'aurait plus de menu contextuel pour le TreeWidget mais bon...). La méthode renvoyant un QPoint, il faudrait ensuite se servir de ce QPoint pour afficher une vignette qui pourrait contenir du texte avec une barre de défilement si besoin ou bien une image de taille maximum fixée. Cela ne doit pas être dur mais j'ai la flemme (je l'avoue).
Si quelqu'un sait faire cela je serais preneur.
Voilà une première ébauche de solution qui est loin d'être parfaite. Mais bon c'est un début...
L'idée est de connecter le slot pour le menu contextuel à la fonction suivante :
Code : Sélectionner tout - Visualiser dans une fenêtre à part
1
2
3
4
5
6
7
8
9 def afficheImage(self, point): # Idée inspirée par le livre de Mark Summerfield. pixmap = QtGui.QPixmap("MonImage.jpg") splash = QtGui.QSplashScreen(pixmap) splash.setMask(pixmap.mask()) # this is usefull if the splashscreen is not a regular ractangle... splash.show() splash.showMessage(u'') # make sure Qt really display the splash screen t = time.sleep(1)
==========
Voici pour tester
==========
Code de dial_Projet3.py
Voici le code de l'application elle-même.
Code : Sélectionner tout - Visualiser dans une fenêtre à part
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216 # -*- coding: utf-8 -*- # Form implementation generated from reading ui file 'C:\Program Files\EasyPHP 2.0b1\www\Python 07-2008\DebuterAvecPythonEtPyQT\CodesProjets\03-Proj3_VueArborescente\dial_Projet3.ui ' # # Created: Tue Jul 29 23:37:30 2008 # by: PyQt4 UI code generator 4.3.3 # # WARNING! All changes made in this file will be lost! from PyQt4 import QtCore, QtGui class Ui_dial_Projet3(object): def setupUi(self, dial_Projet3): dial_Projet3.setObjectName("dial_Projet3") dial_Projet3.resize(QtCore.QSize(QtCore.QRect(0,0,520,390).size()).expandedTo(dial_Projet3.minimumSizeHint())) dial_Projet3.setAutoFillBackground(True) self.treeWidget = QtGui.QTreeWidget(dial_Projet3) self.treeWidget.setGeometry(QtCore.QRect(270,40,240,340)) self.treeWidget.setObjectName("treeWidget") self.textUtilisateur = QtGui.QTextEdit(dial_Projet3) self.textUtilisateur.setGeometry(QtCore.QRect(10,40,240,340)) self.textUtilisateur.setObjectName("textUtilisateur") self.lineTitre_1 = QtGui.QLineEdit(dial_Projet3) self.lineTitre_1.setEnabled(False) self.lineTitre_1.setGeometry(QtCore.QRect(10,10,241,20)) palette = QtGui.QPalette() brush = QtGui.QBrush(QtGui.QColor(0,0,0)) brush.setStyle(QtCore.Qt.SolidPattern) palette.setBrush(QtGui.QPalette.Active,QtGui.QPalette.WindowText,brush) brush = QtGui.QBrush(QtGui.QColor(0,0,0)) brush.setStyle(QtCore.Qt.SolidPattern) palette.setBrush(QtGui.QPalette.Active,QtGui.QPalette.Dark,brush) brush = QtGui.QBrush(QtGui.QColor(0,0,0)) brush.setStyle(QtCore.Qt.SolidPattern) palette.setBrush(QtGui.QPalette.Active,QtGui.QPalette.Text,brush) brush = QtGui.QBrush(QtGui.QColor(0,0,0)) brush.setStyle(QtCore.Qt.SolidPattern) palette.setBrush(QtGui.QPalette.Active,QtGui.QPalette.ButtonText,brush) brush = QtGui.QBrush(QtGui.QColor(255,255,255)) brush.setStyle(QtCore.Qt.SolidPattern) palette.setBrush(QtGui.QPalette.Active,QtGui.QPalette.Base,brush) brush = QtGui.QBrush(QtGui.QColor(255,255,255)) brush.setStyle(QtCore.Qt.SolidPattern) palette.setBrush(QtGui.QPalette.Active,QtGui.QPalette.Window,brush) brush = QtGui.QBrush(QtGui.QColor(0,0,0)) brush.setStyle(QtCore.Qt.SolidPattern) palette.setBrush(QtGui.QPalette.Inactive,QtGui.QPalette.WindowText,brush) brush = QtGui.QBrush(QtGui.QColor(0,0,0)) brush.setStyle(QtCore.Qt.SolidPattern) palette.setBrush(QtGui.QPalette.Inactive,QtGui.QPalette.Dark,brush) brush = QtGui.QBrush(QtGui.QColor(0,0,0)) brush.setStyle(QtCore.Qt.SolidPattern) palette.setBrush(QtGui.QPalette.Inactive,QtGui.QPalette.Text,brush) brush = QtGui.QBrush(QtGui.QColor(0,0,0)) brush.setStyle(QtCore.Qt.SolidPattern) palette.setBrush(QtGui.QPalette.Inactive,QtGui.QPalette.ButtonText,brush) brush = QtGui.QBrush(QtGui.QColor(255,255,255)) brush.setStyle(QtCore.Qt.SolidPattern) palette.setBrush(QtGui.QPalette.Inactive,QtGui.QPalette.Base,brush) brush = QtGui.QBrush(QtGui.QColor(255,255,255)) brush.setStyle(QtCore.Qt.SolidPattern) palette.setBrush(QtGui.QPalette.Inactive,QtGui.QPalette.Window,brush) brush = QtGui.QBrush(QtGui.QColor(0,0,0)) brush.setStyle(QtCore.Qt.SolidPattern) palette.setBrush(QtGui.QPalette.Disabled,QtGui.QPalette.WindowText,brush) brush = QtGui.QBrush(QtGui.QColor(0,0,0)) brush.setStyle(QtCore.Qt.SolidPattern) palette.setBrush(QtGui.QPalette.Disabled,QtGui.QPalette.Dark,brush) brush = QtGui.QBrush(QtGui.QColor(0,0,0)) brush.setStyle(QtCore.Qt.SolidPattern) palette.setBrush(QtGui.QPalette.Disabled,QtGui.QPalette.Text,brush) brush = QtGui.QBrush(QtGui.QColor(0,0,0)) brush.setStyle(QtCore.Qt.SolidPattern) palette.setBrush(QtGui.QPalette.Disabled,QtGui.QPalette.ButtonText,brush) brush = QtGui.QBrush(QtGui.QColor(255,255,255)) brush.setStyle(QtCore.Qt.SolidPattern) palette.setBrush(QtGui.QPalette.Disabled,QtGui.QPalette.Base,brush) brush = QtGui.QBrush(QtGui.QColor(255,255,255)) brush.setStyle(QtCore.Qt.SolidPattern) palette.setBrush(QtGui.QPalette.Disabled,QtGui.QPalette.Window,brush) self.lineTitre_1.setPalette(palette) font = QtGui.QFont() font.setWeight(50) font.setBold(False) self.lineTitre_1.setFont(font) self.lineTitre_1.setAlignment(QtCore.Qt.AlignCenter) self.lineTitre_1.setObjectName("lineTitre_1") self.lineTitre_2 = QtGui.QLineEdit(dial_Projet3) self.lineTitre_2.setEnabled(False) self.lineTitre_2.setGeometry(QtCore.QRect(270,10,241,20)) palette = QtGui.QPalette() brush = QtGui.QBrush(QtGui.QColor(0,0,0)) brush.setStyle(QtCore.Qt.SolidPattern) palette.setBrush(QtGui.QPalette.Active,QtGui.QPalette.WindowText,brush) brush = QtGui.QBrush(QtGui.QColor(0,0,0)) brush.setStyle(QtCore.Qt.SolidPattern) palette.setBrush(QtGui.QPalette.Active,QtGui.QPalette.Dark,brush) brush = QtGui.QBrush(QtGui.QColor(0,0,0)) brush.setStyle(QtCore.Qt.SolidPattern) palette.setBrush(QtGui.QPalette.Active,QtGui.QPalette.Text,brush) brush = QtGui.QBrush(QtGui.QColor(0,0,0)) brush.setStyle(QtCore.Qt.SolidPattern) palette.setBrush(QtGui.QPalette.Active,QtGui.QPalette.ButtonText,brush) brush = QtGui.QBrush(QtGui.QColor(255,255,255)) brush.setStyle(QtCore.Qt.SolidPattern) palette.setBrush(QtGui.QPalette.Active,QtGui.QPalette.Base,brush) brush = QtGui.QBrush(QtGui.QColor(255,255,255)) brush.setStyle(QtCore.Qt.SolidPattern) palette.setBrush(QtGui.QPalette.Active,QtGui.QPalette.Window,brush) brush = QtGui.QBrush(QtGui.QColor(0,0,0)) brush.setStyle(QtCore.Qt.SolidPattern) palette.setBrush(QtGui.QPalette.Inactive,QtGui.QPalette.WindowText,brush) brush = QtGui.QBrush(QtGui.QColor(0,0,0)) brush.setStyle(QtCore.Qt.SolidPattern) palette.setBrush(QtGui.QPalette.Inactive,QtGui.QPalette.Dark,brush) brush = QtGui.QBrush(QtGui.QColor(0,0,0)) brush.setStyle(QtCore.Qt.SolidPattern) palette.setBrush(QtGui.QPalette.Inactive,QtGui.QPalette.Text,brush) brush = QtGui.QBrush(QtGui.QColor(0,0,0)) brush.setStyle(QtCore.Qt.SolidPattern) palette.setBrush(QtGui.QPalette.Inactive,QtGui.QPalette.ButtonText,brush) brush = QtGui.QBrush(QtGui.QColor(255,255,255)) brush.setStyle(QtCore.Qt.SolidPattern) palette.setBrush(QtGui.QPalette.Inactive,QtGui.QPalette.Base,brush) brush = QtGui.QBrush(QtGui.QColor(255,255,255)) brush.setStyle(QtCore.Qt.SolidPattern) palette.setBrush(QtGui.QPalette.Inactive,QtGui.QPalette.Window,brush) brush = QtGui.QBrush(QtGui.QColor(0,0,0)) brush.setStyle(QtCore.Qt.SolidPattern) palette.setBrush(QtGui.QPalette.Disabled,QtGui.QPalette.WindowText,brush) brush = QtGui.QBrush(QtGui.QColor(0,0,0)) brush.setStyle(QtCore.Qt.SolidPattern) palette.setBrush(QtGui.QPalette.Disabled,QtGui.QPalette.Dark,brush) brush = QtGui.QBrush(QtGui.QColor(0,0,0)) brush.setStyle(QtCore.Qt.SolidPattern) palette.setBrush(QtGui.QPalette.Disabled,QtGui.QPalette.Text,brush) brush = QtGui.QBrush(QtGui.QColor(0,0,0)) brush.setStyle(QtCore.Qt.SolidPattern) palette.setBrush(QtGui.QPalette.Disabled,QtGui.QPalette.ButtonText,brush) brush = QtGui.QBrush(QtGui.QColor(255,255,255)) brush.setStyle(QtCore.Qt.SolidPattern) palette.setBrush(QtGui.QPalette.Disabled,QtGui.QPalette.Base,brush) brush = QtGui.QBrush(QtGui.QColor(255,255,255)) brush.setStyle(QtCore.Qt.SolidPattern) palette.setBrush(QtGui.QPalette.Disabled,QtGui.QPalette.Window,brush) self.lineTitre_2.setPalette(palette) font = QtGui.QFont() font.setWeight(50) font.setBold(False) self.lineTitre_2.setFont(font) self.lineTitre_2.setAlignment(QtCore.Qt.AlignCenter) self.lineTitre_2.setObjectName("lineTitre_2") self.retranslateUi(dial_Projet3) QtCore.QMetaObject.connectSlotsByName(dial_Projet3) def retranslateUi(self, dial_Projet3): dial_Projet3.setWindowTitle(QtGui.QApplication.translate("dial_Projet3", "Projet n°3", None, QtGui.QApplication.UnicodeUTF8)) self.treeWidget.headerItem().setText(0,QtGui.QApplication.translate("dial_Projet3", "1", None, QtGui.QApplication.UnicodeUTF8)) self.lineTitre_1.setText(QtGui.QApplication.translate("dial_Projet3", "Votre table des matières avec les balises", None, QtGui.QApplication.UnicodeUTF8)) self.lineTitre_2.setText(QtGui.QApplication.translate("dial_Projet3", "Arbre de la table des matières", None, QtGui.QApplication.UnicodeUTF8)) if __name__ == "__main__": import sys app = QtGui.QApplication(sys.argv) dial_Projet3 = QtGui.QDialog() ui = Ui_dial_Projet3() ui.setupUi(dial_Projet3) dial_Projet3.show() sys.exit(app.exec_())
Code : Sélectionner tout - Visualiser dans une fenêtre à part
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304 # -*- coding: utf-8 -*- #!/usr/bin/env python # On importe les bibliothèques que nous allons utiliser. import sys from PyQt4 import QtCore, QtGui # On importe notre boîte de dialogue. from dial_Projet3 import Ui_dial_Projet3 import time # # # # # # # # # # # # # # # # # # # # # # # Comportement de la boîte de dialogue. DEBUT class dial_Projet3(QtGui.QDialog, Ui_dial_Projet3): def __init__(self): QtGui.QDialog.__init__(self) Ui_dial_Projet3.__init__(self) self.setupUi(self) # On met un texte brut type qui aidera l'utilisateur à faire ses propres tables tout seul. self.texte_Type = """Intro Chap I >Para 1 >>Partie A >>Partie B >Para 2 Chap II Conclusion""" self.textUtilisateur.setText(self.texte_Type) # On crée à la main le tableau pour garder en mémoire la table des matières. self.tablo_tabMat_Memo = [] self.tablo_tabMat_Memo.append('Intro') tablo_Para = [] tablo_Para.append('Chap I') tablo_Partie = [] tablo_Partie.append('Para 1') tablo_Partie.append('Partie A') tablo_Partie.append('Partie B') tablo_Para.append(tablo_Partie) tablo_Para.append('Para 2') self.tablo_tabMat_Memo.append(tablo_Para) self.tablo_tabMat_Memo.append('Chap II') self.tablo_tabMat_Memo.append('Conclusion') # On continue à remplir l'arbre à la main. # PETITE NOUVEAUTE : nous allons faire en sorte que l'arbre soit totalement déployé dès l'affichage(voir le commentaire ci-dessous). self.treeWidget.clear() self.treeWidget.setColumnCount(1) self.treeWidget.setHeaderLabels(["Sommaire"]) self.treeWidget.setItemsExpandable(True) tabNiv_1 = QtGui.QTreeWidgetItem(self.treeWidget, ["Intro"]) tabNiv_1.setIcon(0, QtGui.QIcon('Images/im_1.png')) tabNiv_1 = QtGui.QTreeWidgetItem(self.treeWidget, ["Chap I"]) tabNiv_1.setIcon(0, QtGui.QIcon('Images/im_1.png')) # La ligne suivante permet "d'ouvrir" le noeud Para 1. On fait de même avec tous les noeuds de Chap I, puis avec Chap I lui-même. self.treeWidget.expandItem(tabNiv_1) tabNiv_2 = QtGui.QTreeWidgetItem(tabNiv_1, ["Para 1"]) tabNiv_2.setIcon(0, QtGui.QIcon('Images/im_2.png')) self.treeWidget.expandItem(tabNiv_2) tabNiv_3 = QtGui.QTreeWidgetItem(tabNiv_2, ["Partie A"]) tabNiv_3.setIcon(0, QtGui.QIcon('Images/im_3.png')) self.treeWidget.expandItem(tabNiv_3) tabNiv_3 = QtGui.QTreeWidgetItem(tabNiv_2, ["Partie B"]) tabNiv_3.setIcon(0, QtGui.QIcon('Images/im_3.png')) self.treeWidget.expandItem(tabNiv_3) tabNiv_2 = QtGui.QTreeWidgetItem(tabNiv_1, ["Para 2"]) tabNiv_2.setIcon(0, QtGui.QIcon('Images/im_2.png')) self.treeWidget.expandItem(tabNiv_2) tabNiv_1 = QtGui.QTreeWidgetItem(self.treeWidget, ["Chap II"]) tabNiv_1.setIcon(0, QtGui.QIcon('Images/im_1.png')) tabNiv_1 = QtGui.QTreeWidgetItem(self.treeWidget, ["Conclusion"]) tabNiv_1.setIcon(0, QtGui.QIcon('Images/im_1.png')) # On définit la connection liée au changement du contenu du QTextEdit. self.connect(self.textUtilisateur, QtCore.SIGNAL("textChanged()"), self.chgtTexte) self.connect(self.treeWidget, QtCore.SIGNAL("itemClicked(QTreeWidgetItem*,int)"), self.clicNoeud) # On définit la connection liée au menu contextuel. self.treeWidget.setContextMenuPolicy(QtCore.Qt.CustomContextMenu) self.connect(self.treeWidget, QtCore.SIGNAL("customContextMenuRequested(const QPoint &)"), self.afficheImage) # Menu Contextuel # Cf. bas de la page : # http://www.nabble.com/Context-menu-on-items-in-QTreeView-td19981837.html # # http://www.riverbankcomputing.com/static/Docs/PyQt4/html/qwidget.html#setContextMenuPolicy # http://www.riverbankcomputing.com/static/Docs/PyQt4/html/qt.html#ContextMenuPolicy-enum # http://www.riverbankcomputing.com/static/Docs/PyQt4/html/qwidget.html#customContextMenuRequested def afficheImage(self, point): # On récupère les infos sur l'item sélectionné. index = self.treeWidget.indexAt(point) if not index.isValid(): return # print index.text(0) item = self.treeWidget.itemAt(point) name = item.text(0) # Nom du noeaud # Idée inspirée par le livre de Mark Summerfield. pixmap = QtGui.QPixmap("MonImage.jpg") splash = QtGui.QSplashScreen(pixmap) splash.setMask(pixmap.mask()) # this is usefull if the splashscreen is not a regular ractangle... splash.show() splash.showMessage(u'') # make sure Qt really display the splash screen t = time.sleep(1) # Info récupérée sur le forum suivant : . def clicNoeud(self,noeudClic): # La méthode ci-dessous a ses limites car il faut connaître la profondeur de l'arbre. par_1='' par_2='' test_1 = noeudClic.parent() if test_1 != None: par_1 = str(noeudClic.parent().text(0)) test_2 = test_1.parent() if test_2 != None: par_2 = str(test_2.parent()) if par_2!='': print "Clic sur une partie-Titre de la partie :" print " " + str(noeudClic.text(0)) elif par_1!='': print "Clic sur un paragraphe-Titre du paragraphe :" print " " + str(noeudClic.text(0)) else: print "Clic sur un chapitre-Titre du chapitre :" print " " + str(noeudClic.text(0)) print'' # Comportement de la boîte de dialogue. FIN # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # Analyse brutale du changement de contenu. DEBUT def chgtTexte(self): text_TabMatBrut = self.textUtilisateur.toPlainText() # Traduction du contenu sous forme d'un tableau. self.tablo_tabMat = [] niv = 0 maj = True lignes_tabMat_Sale = text_TabMatBrut.split("\n") lignes_tabMat = [] for i in range(len(lignes_tabMat_Sale)): # On nettoie les lignes. uneLignePropre = str(lignes_tabMat_Sale[i]).strip() if uneLignePropre <> '' and uneLignePropre <> '>' and uneLignePropre <> '>>' : lignes_tabMat.append(uneLignePropre) # On parcourt les lignes en les analysant. for i in range(len(lignes_tabMat)): uneLignePropre = lignes_tabMat[i] # Repérage d'une PARTIE if uneLignePropre[:2] == '>>': if niv < 2: maj = False print u"Un problème de structure logique (sûrement temporaire) a été repéré.\nLa 1ère balise défectueuse est : \"" + uneLignePropre + '".\n' break else: uneLignePropre = uneLignePropre[2:].strip() if i == len(lignes_tabMat) - 1: tablo_Partie.append(uneLignePropre) tablo_Para.append(tablo_Partie) self.tablo_tabMat.append(tablo_Para) else: lignePropreSuivante = lignes_tabMat[i + 1] if lignePropreSuivante[:2] == '>>': tablo_Partie.append(uneLignePropre) elif lignePropreSuivante[0] == '>': tablo_Partie.append(uneLignePropre) tablo_Para.append(tablo_Partie) else : tablo_Partie.append(uneLignePropre) tablo_Para.append(tablo_Partie) self.tablo_tabMat.append(tablo_Para) niv = 3 # Repérage d'un PARAGRAPHE elif uneLignePropre[0] == '>': if niv == 0: maj = False print u"Un problème de structure logique (sûrement temporaire) a été repéré.\nLa 1ère balise défectueuse est : \"" + uneLignePropre + '".\n' break else: uneLignePropre = uneLignePropre[1:].strip() tablo_Partie = [] if i == len(lignes_tabMat) - 1: tablo_Para.append(uneLignePropre) self.tablo_tabMat.append(tablo_Para) else: lignePropreSuivante = lignes_tabMat[i + 1] if lignePropreSuivante[:2] == '>>': tablo_Partie.append(uneLignePropre) elif lignePropreSuivante[0] == '>': tablo_Para.append(uneLignePropre) else: tablo_Para.append(uneLignePropre) self.tablo_tabMat.append(tablo_Para) niv = 2 # Repérage d'un CHAPITRE else: niv = 1 tablo_Para = [] tablo_Partie = [] if i == len(lignes_tabMat) - 1: self.tablo_tabMat.append(uneLignePropre) else: lignePropreSuivante = lignes_tabMat[i + 1] if lignePropreSuivante[0] <> '>': self.tablo_tabMat.append(uneLignePropre) else: tablo_Para.append(uneLignePropre) # Mise à jour de l'arbre de la table (la technique est similaire à celle utilisée pour l'affichage HTML de la Table des Matières). if maj: if self.tablo_tabMat <> self.tablo_tabMat_Memo: self.lineTitre_2.setText(QtGui.QApplication.translate("dial_TestAnaBalise", "Arbre de la table des matières", None, QtGui.QApplication.UnicodeUTF8)) self.tablo_tabMat_Memo = self.tablo_tabMat self.treeWidget.clear() self.treeWidget.setColumnCount(1) self.treeWidget.setHeaderLabels(["Sommaire"]) self.treeWidget.setItemsExpandable(True) for i_Chap in range(len(self.tablo_tabMat)): try: # ATTENTION ! IL faut donner un tableau avec une seule chaîne pour nourrir l'arbre. # Donc tabNiv_1 = QtGui.QTreeWidgetItem(self.treeWidget, [self.tablo_tabMat[i_Chap]]) ne marchera # que si self.tablo_tabMat[i_Chap] est une chaîne de caractères. # Si ce n'est pas le cas, c'est que le chapitre contient des paragraphes. tabNiv_1 = QtGui.QTreeWidgetItem(self.treeWidget, [self.tablo_tabMat[i_Chap]]) tabNiv_1.setIcon(0, QtGui.QIcon('Images/im_1.png')) except: tabloPara = self.tablo_tabMat[i_Chap] for i_Para in range(len(tabloPara)): if i_Para == 0: tabNiv_1 = QtGui.QTreeWidgetItem(self.treeWidget, [tabloPara[0]]) tabNiv_1.setIcon(0, QtGui.QIcon('Images/im_1.png')) self.treeWidget.expandItem(tabNiv_1) else: try: tabNiv_2 = QtGui.QTreeWidgetItem(tabNiv_1, [tabloPara[i_Para]]) tabNiv_2.setIcon(0, QtGui.QIcon('Images/im_2.png')) self.treeWidget.expandItem(tabNiv_2) except: tabloPartie = tabloPara[i_Para] for i_Partie in range(len(tabloPartie)): if i_Partie == 0: tabNiv_2 = QtGui.QTreeWidgetItem(tabNiv_1, [tabloPartie[0]]) tabNiv_2.setIcon(0, QtGui.QIcon('Images/im_2.png')) self.treeWidget.expandItem(tabNiv_2) else: tabNiv_3 = QtGui.QTreeWidgetItem(tabNiv_2, [tabloPartie[i_Partie]]) tabNiv_3.setIcon(0, QtGui.QIcon('Images/im_3.png')) self.treeWidget.expandItem(tabNiv_3) # Pas de mise à jour possible. else : self.tablo_tabMat_Memo = self.tablo_tabMat self.lineTitre_2.setText(QtGui.QApplication.translate("dial_TestAnaBalise", "Mise à jour impossible", None, QtGui.QApplication.UnicodeUTF8)) # Analyse brutale du changement de contenu. FIN # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # Lancement de l'application. if __name__ == "__main__": app = QtGui.QApplication(sys.argv) Projet3 = dial_Projet3() Projet3.show() sys.exit(app.exec_())
Voici une méthode qui affiche une image pour une durée de 2 secondes sans bloquer l'utilisation du logiciel. Lors d'un clic droit dessus, elle disparait automatiquement. L'inconvénient de cette méthode est que l'image est forcément placée au centre de l'écran.
Code : Sélectionner tout - Visualiser dans une fenêtre à part
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20 def afficheImage(self, point): # On récupère les infos sur l'item sélectionné. index = self.treeWidget.indexAt(point) if not index.isValid(): return # print index.text(0) item = self.treeWidget.itemAt(point) name = item.text(0) # Nom du noeaud # Idée inspirée par le livre de Mark Summerfield. pixmap = QtGui.QPixmap("Collines.jpg") # pixmap = QtGui.QPixmap("im_ronde.jpg") BUG splash = QtGui.QSplashScreen(pixmap) splash.setMask(pixmap.mask()) splash.show() splash.showMessage(u'') t = time.time() while time.time() < t + 2: app.processEvents() # On redonne la main à l'application.
========
Code complet
========
Code : Sélectionner tout - Visualiser dans une fenêtre à part
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308 # -*- coding: utf-8 -*- #!/usr/bin/env python # On importe les bibliothèques que nous allons utiliser. import sys from PyQt4 import QtCore, QtGui # On importe notre boîte de dialogue. from dial_Projet3 import Ui_dial_Projet3 import time # # # # # # # # # # # # # # # # # # # # # # # Comportement de la boîte de dialogue. DEBUT class dial_Projet3(QtGui.QDialog, Ui_dial_Projet3): def __init__(self): QtGui.QDialog.__init__(self) Ui_dial_Projet3.__init__(self) self.setupUi(self) # On met un texte brut type qui aidera l'utilisateur à faire ses propres tables tout seul. self.texte_Type = """Intro Chap I >Para 1 >>Partie A >>Partie B >Para 2 Chap II Conclusion""" self.textUtilisateur.setText(self.texte_Type) # On crée à la main le tableau pour garder en mémoire la table des matières. self.tablo_tabMat_Memo = [] self.tablo_tabMat_Memo.append('Intro') tablo_Para = [] tablo_Para.append('Chap I') tablo_Partie = [] tablo_Partie.append('Para 1') tablo_Partie.append('Partie A') tablo_Partie.append('Partie B') tablo_Para.append(tablo_Partie) tablo_Para.append('Para 2') self.tablo_tabMat_Memo.append(tablo_Para) self.tablo_tabMat_Memo.append('Chap II') self.tablo_tabMat_Memo.append('Conclusion') # On continue à remplir l'arbre à la main. # PETITE NOUVEAUTE : nous allons faire en sorte que l'arbre soit totalement déployé dès l'affichage(voir le commentaire ci-dessous). self.treeWidget.clear() self.treeWidget.setColumnCount(1) self.treeWidget.setHeaderLabels(["Sommaire"]) self.treeWidget.setItemsExpandable(True) tabNiv_1 = QtGui.QTreeWidgetItem(self.treeWidget, ["Intro"]) tabNiv_1.setIcon(0, QtGui.QIcon('Images/im_1.png')) tabNiv_1 = QtGui.QTreeWidgetItem(self.treeWidget, ["Chap I"]) tabNiv_1.setIcon(0, QtGui.QIcon('Images/im_1.png')) # La ligne suivante permet "d'ouvrir" le noeud Para 1. On fait de même avec tous les noeuds de Chap I, puis avec Chap I lui-même. self.treeWidget.expandItem(tabNiv_1) tabNiv_2 = QtGui.QTreeWidgetItem(tabNiv_1, ["Para 1"]) tabNiv_2.setIcon(0, QtGui.QIcon('Images/im_2.png')) self.treeWidget.expandItem(tabNiv_2) tabNiv_3 = QtGui.QTreeWidgetItem(tabNiv_2, ["Partie A"]) tabNiv_3.setIcon(0, QtGui.QIcon('Images/im_3.png')) self.treeWidget.expandItem(tabNiv_3) tabNiv_3 = QtGui.QTreeWidgetItem(tabNiv_2, ["Partie B"]) tabNiv_3.setIcon(0, QtGui.QIcon('Images/im_3.png')) self.treeWidget.expandItem(tabNiv_3) tabNiv_2 = QtGui.QTreeWidgetItem(tabNiv_1, ["Para 2"]) tabNiv_2.setIcon(0, QtGui.QIcon('Images/im_2.png')) self.treeWidget.expandItem(tabNiv_2) tabNiv_1 = QtGui.QTreeWidgetItem(self.treeWidget, ["Chap II"]) tabNiv_1.setIcon(0, QtGui.QIcon('Images/im_1.png')) tabNiv_1 = QtGui.QTreeWidgetItem(self.treeWidget, ["Conclusion"]) tabNiv_1.setIcon(0, QtGui.QIcon('Images/im_1.png')) # On définit la connection liée au changement du contenu du QTextEdit. self.connect(self.textUtilisateur, QtCore.SIGNAL("textChanged()"), self.chgtTexte) self.connect(self.treeWidget, QtCore.SIGNAL("itemClicked(QTreeWidgetItem*,int)"), self.clicNoeud) # On définit la connection liée au menu contextuel. self.treeWidget.setContextMenuPolicy(QtCore.Qt.CustomContextMenu) self.connect(self.treeWidget, QtCore.SIGNAL("customContextMenuRequested(const QPoint &)"), self.afficheImage) # Menu Contextuel # Cf. bas de la page : # http://www.nabble.com/Context-menu-on-items-in-QTreeView-td19981837.html # # http://www.riverbankcomputing.com/static/Docs/PyQt4/html/qwidget.html#setContextMenuPolicy # http://www.riverbankcomputing.com/static/Docs/PyQt4/html/qt.html#ContextMenuPolicy-enum # http://www.riverbankcomputing.com/static/Docs/PyQt4/html/qwidget.html#customContextMenuRequested def afficheImage(self, point): # On récupère les infos sur l'item sélectionné. index = self.treeWidget.indexAt(point) if not index.isValid(): return # print index.text(0) item = self.treeWidget.itemAt(point) name = item.text(0) # Nom du noeaud # Idée inspirée par le livre de Mark Summerfield. pixmap = QtGui.QPixmap("Collines.jpg") # pixmap = QtGui.QPixmap("im_ronde.jpg") BUG splash = QtGui.QSplashScreen(pixmap) splash.setMask(pixmap.mask()) splash.show() splash.showMessage(u'') t = time.time() while time.time() < t + 2: app.processEvents() # On redonne la main à l'application. # Info récupérée sur le forum suivant : . def clicNoeud(self,noeudClic): # La méthode ci-dessous a ses limites car il faut connaître la profondeur de l'arbre. par_1='' par_2='' test_1 = noeudClic.parent() if test_1 != None: par_1 = str(noeudClic.parent().text(0)) test_2 = test_1.parent() if test_2 != None: par_2 = str(test_2.parent()) if par_2!='': print "Clic sur une partie-Titre de la partie :" print " " + str(noeudClic.text(0)) elif par_1!='': print "Clic sur un paragraphe-Titre du paragraphe :" print " " + str(noeudClic.text(0)) else: print "Clic sur un chapitre-Titre du chapitre :" print " " + str(noeudClic.text(0)) print'' # Comportement de la boîte de dialogue. FIN # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # Analyse brutale du changement de contenu. DEBUT def chgtTexte(self): text_TabMatBrut = self.textUtilisateur.toPlainText() # Traduction du contenu sous forme d'un tableau. self.tablo_tabMat = [] niv = 0 maj = True lignes_tabMat_Sale = text_TabMatBrut.split("\n") lignes_tabMat = [] for i in range(len(lignes_tabMat_Sale)): # On nettoie les lignes. uneLignePropre = str(lignes_tabMat_Sale[i]).strip() if uneLignePropre <> '' and uneLignePropre <> '>' and uneLignePropre <> '>>' : lignes_tabMat.append(uneLignePropre) # On parcourt les lignes en les analysant. for i in range(len(lignes_tabMat)): uneLignePropre = lignes_tabMat[i] # Repérage d'une PARTIE if uneLignePropre[:2] == '>>': if niv < 2: maj = False print u"Un problème de structure logique (sûrement temporaire) a été repéré.\nLa 1ère balise défectueuse est : \"" + uneLignePropre + '".\n' break else: uneLignePropre = uneLignePropre[2:].strip() if i == len(lignes_tabMat) - 1: tablo_Partie.append(uneLignePropre) tablo_Para.append(tablo_Partie) self.tablo_tabMat.append(tablo_Para) else: lignePropreSuivante = lignes_tabMat[i + 1] if lignePropreSuivante[:2] == '>>': tablo_Partie.append(uneLignePropre) elif lignePropreSuivante[0] == '>': tablo_Partie.append(uneLignePropre) tablo_Para.append(tablo_Partie) else : tablo_Partie.append(uneLignePropre) tablo_Para.append(tablo_Partie) self.tablo_tabMat.append(tablo_Para) niv = 3 # Repérage d'un PARAGRAPHE elif uneLignePropre[0] == '>': if niv == 0: maj = False print u"Un problème de structure logique (sûrement temporaire) a été repéré.\nLa 1ère balise défectueuse est : \"" + uneLignePropre + '".\n' break else: uneLignePropre = uneLignePropre[1:].strip() tablo_Partie = [] if i == len(lignes_tabMat) - 1: tablo_Para.append(uneLignePropre) self.tablo_tabMat.append(tablo_Para) else: lignePropreSuivante = lignes_tabMat[i + 1] if lignePropreSuivante[:2] == '>>': tablo_Partie.append(uneLignePropre) elif lignePropreSuivante[0] == '>': tablo_Para.append(uneLignePropre) else: tablo_Para.append(uneLignePropre) self.tablo_tabMat.append(tablo_Para) niv = 2 # Repérage d'un CHAPITRE else: niv = 1 tablo_Para = [] tablo_Partie = [] if i == len(lignes_tabMat) - 1: self.tablo_tabMat.append(uneLignePropre) else: lignePropreSuivante = lignes_tabMat[i + 1] if lignePropreSuivante[0] <> '>': self.tablo_tabMat.append(uneLignePropre) else: tablo_Para.append(uneLignePropre) # Mise à jour de l'arbre de la table (la technique est similaire à celle utilisée pour l'affichage HTML de la Table des Matières). if maj: if self.tablo_tabMat <> self.tablo_tabMat_Memo: self.lineTitre_2.setText(QtGui.QApplication.translate("dial_TestAnaBalise", "Arbre de la table des matières", None, QtGui.QApplication.UnicodeUTF8)) self.tablo_tabMat_Memo = self.tablo_tabMat self.treeWidget.clear() self.treeWidget.setColumnCount(1) self.treeWidget.setHeaderLabels(["Sommaire"]) self.treeWidget.setItemsExpandable(True) for i_Chap in range(len(self.tablo_tabMat)): try: # ATTENTION ! IL faut donner un tableau avec une seule chaîne pour nourrir l'arbre. # Donc tabNiv_1 = QtGui.QTreeWidgetItem(self.treeWidget, [self.tablo_tabMat[i_Chap]]) ne marchera # que si self.tablo_tabMat[i_Chap] est une chaîne de caractères. # Si ce n'est pas le cas, c'est que le chapitre contient des paragraphes. tabNiv_1 = QtGui.QTreeWidgetItem(self.treeWidget, [self.tablo_tabMat[i_Chap]]) tabNiv_1.setIcon(0, QtGui.QIcon('Images/im_1.png')) except: tabloPara = self.tablo_tabMat[i_Chap] for i_Para in range(len(tabloPara)): if i_Para == 0: tabNiv_1 = QtGui.QTreeWidgetItem(self.treeWidget, [tabloPara[0]]) tabNiv_1.setIcon(0, QtGui.QIcon('Images/im_1.png')) self.treeWidget.expandItem(tabNiv_1) else: try: tabNiv_2 = QtGui.QTreeWidgetItem(tabNiv_1, [tabloPara[i_Para]]) tabNiv_2.setIcon(0, QtGui.QIcon('Images/im_2.png')) self.treeWidget.expandItem(tabNiv_2) except: tabloPartie = tabloPara[i_Para] for i_Partie in range(len(tabloPartie)): if i_Partie == 0: tabNiv_2 = QtGui.QTreeWidgetItem(tabNiv_1, [tabloPartie[0]]) tabNiv_2.setIcon(0, QtGui.QIcon('Images/im_2.png')) self.treeWidget.expandItem(tabNiv_2) else: tabNiv_3 = QtGui.QTreeWidgetItem(tabNiv_2, [tabloPartie[i_Partie]]) tabNiv_3.setIcon(0, QtGui.QIcon('Images/im_3.png')) self.treeWidget.expandItem(tabNiv_3) # Pas de mise à jour possible. else : self.tablo_tabMat_Memo = self.tablo_tabMat self.lineTitre_2.setText(QtGui.QApplication.translate("dial_TestAnaBalise", "Mise à jour impossible", None, QtGui.QApplication.UnicodeUTF8)) # Analyse brutale du changement de contenu. FIN # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # Lancement de l'application. if __name__ == "__main__": app = QtGui.QApplication(sys.argv) Projet3 = dial_Projet3() Projet3.show() sys.exit(app.exec_())
Grâce à la liste de PyQt j'ai pu arriver à la méthode suivante :
Ce qui est nouveau ce sont les deux parties suivantes :
Code : Sélectionner tout - Visualiser dans une fenêtre à part
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325 # -*- coding: utf-8 -*- #!/usr/bin/env python # On importe les bibliothèques que nous allons utiliser. import sys from PyQt4 import QtCore, QtGui # On importe notre boîte de dialogue. from dial_Projet3 import Ui_dial_Projet3 import time # # # # # # # # # # # # # # # # # # # # # # # Comportement de la boîte de dialogue. DEBUT class MyFrame(QtGui.QFrame): # Squelette donné par Brian Kelley sur la liste de PyQt. def __init__(self, parent, titre, pixmapfile, position): QtGui.QFrame.__init__(self, parent, QtCore.Qt.WindowStaysOnTopHint | QtCore.Qt.Tool) # QtCore.Qt.X11BypassWindowManagerHint | QtCore.Qt.FramelessWindowHint | QtCore.Qt.WindowStaysOnTopHint | QtCore.Qt.Tool self.setWindowTitle(titre) # rect=QtCore.QRect(position, QtCore.QPoint(1000,1000)) # self.setFrameRect(rect) label = QtGui.QLabel(self) label.setPixmap(pixmapfile) label.show() self.timer = QtCore.QTimer() self.connect( self.timer, QtCore.SIGNAL("timeout()"), self, QtCore.SLOT("hide()") ) def showEvent(self, evt): self.timer.start(10000) # 10 seconds QtGui.QFrame.showEvent(self, evt) class dial_Projet3(QtGui.QDialog, Ui_dial_Projet3): def __init__(self): QtGui.QDialog.__init__(self) Ui_dial_Projet3.__init__(self) self.setupUi(self) # On met un texte brut type qui aidera l'utilisateur à faire ses propres tables tout seul. self.texte_Type = """Intro Chap I >Para 1 >>Partie A >>Partie B >Para 2 Chap II Conclusion""" self.textUtilisateur.setText(self.texte_Type) # On crée à la main le tableau pour garder en mémoire la table des matières. self.tablo_tabMat_Memo = [] self.tablo_tabMat_Memo.append('Intro') tablo_Para = [] tablo_Para.append('Chap I') tablo_Partie = [] tablo_Partie.append('Para 1') tablo_Partie.append('Partie A') tablo_Partie.append('Partie B') tablo_Para.append(tablo_Partie) tablo_Para.append('Para 2') self.tablo_tabMat_Memo.append(tablo_Para) self.tablo_tabMat_Memo.append('Chap II') self.tablo_tabMat_Memo.append('Conclusion') # On continue à remplir l'arbre à la main. # PETITE NOUVEAUTE : nous allons faire en sorte que l'arbre soit totalement déployé dès l'affichage(voir le commentaire ci-dessous). self.treeWidget.clear() self.treeWidget.setColumnCount(1) self.treeWidget.setHeaderLabels(["Sommaire"]) self.treeWidget.setItemsExpandable(True) tabNiv_1 = QtGui.QTreeWidgetItem(self.treeWidget, ["Intro"]) tabNiv_1.setIcon(0, QtGui.QIcon('Images/im_1.png')) tabNiv_1 = QtGui.QTreeWidgetItem(self.treeWidget, ["Chap I"]) tabNiv_1.setIcon(0, QtGui.QIcon('Images/im_1.png')) # La ligne suivante permet "d'ouvrir" le noeud Para 1. On fait de même avec tous les noeuds de Chap I, puis avec Chap I lui-même. self.treeWidget.expandItem(tabNiv_1) tabNiv_2 = QtGui.QTreeWidgetItem(tabNiv_1, ["Para 1"]) tabNiv_2.setIcon(0, QtGui.QIcon('Images/im_2.png')) self.treeWidget.expandItem(tabNiv_2) tabNiv_3 = QtGui.QTreeWidgetItem(tabNiv_2, ["Partie A"]) tabNiv_3.setIcon(0, QtGui.QIcon('Images/im_3.png')) self.treeWidget.expandItem(tabNiv_3) tabNiv_3 = QtGui.QTreeWidgetItem(tabNiv_2, ["Partie B"]) tabNiv_3.setIcon(0, QtGui.QIcon('Images/im_3.png')) self.treeWidget.expandItem(tabNiv_3) tabNiv_2 = QtGui.QTreeWidgetItem(tabNiv_1, ["Para 2"]) tabNiv_2.setIcon(0, QtGui.QIcon('Images/im_2.png')) self.treeWidget.expandItem(tabNiv_2) tabNiv_1 = QtGui.QTreeWidgetItem(self.treeWidget, ["Chap II"]) tabNiv_1.setIcon(0, QtGui.QIcon('Images/im_1.png')) tabNiv_1 = QtGui.QTreeWidgetItem(self.treeWidget, ["Conclusion"]) tabNiv_1.setIcon(0, QtGui.QIcon('Images/im_1.png')) # On définit la connection liée au changement du contenu du QTextEdit. self.connect(self.textUtilisateur, QtCore.SIGNAL("textChanged()"), self.chgtTexte) self.connect(self.treeWidget, QtCore.SIGNAL("itemClicked(QTreeWidgetItem*,int)"), self.clicNoeud) # On définit la connection liée au menu contextuel. self.treeWidget.setContextMenuPolicy(QtCore.Qt.CustomContextMenu) self.connect(self.treeWidget, QtCore.SIGNAL("customContextMenuRequested(const QPoint &)"), self.afficheImage) # Menu Contextuel # Cf. bas de la page : # http://www.nabble.com/Context-menu-on-items-in-QTreeView-td19981837.html # # http://www.riverbankcomputing.com/static/Docs/PyQt4/html/qwidget.html#setContextMenuPolicy # http://www.riverbankcomputing.com/static/Docs/PyQt4/html/qt.html#ContextMenuPolicy-enum # http://www.riverbankcomputing.com/static/Docs/PyQt4/html/qwidget.html#customContextMenuRequested def afficheImage(self, point): # On récupère les infos sur l'item sélectionné. index = self.treeWidget.indexAt(point) if not index.isValid(): return # print index.text(0) item = self.treeWidget.itemAt(point) name = item.text(0) # Nom du noeaud # Problème des grands images à gérer par exemple en les redimensionnant ou en utilisant un GraphicsView et des scrollbars. pixmap = QtGui.QPixmap("normale.jpg") # pixmap = QtGui.QPixmap("tropGrande.jpg") frame = MyFrame(self, name, pixmap, self.mapToGlobal(QtGui.QCursor.pos())) frame.show() # Info récupérée sur le forum suivant : . def clicNoeud(self,noeudClic): # La méthode ci-dessous a ses limites car il faut connaître la profondeur de l'arbre. par_1='' par_2='' test_1 = noeudClic.parent() if test_1 != None: par_1 = str(noeudClic.parent().text(0)) test_2 = test_1.parent() if test_2 != None: par_2 = str(test_2.parent()) if par_2!='': print "Clic sur une partie-Titre de la partie :" print " " + str(noeudClic.text(0)) elif par_1!='': print "Clic sur un paragraphe-Titre du paragraphe :" print " " + str(noeudClic.text(0)) else: print "Clic sur un chapitre-Titre du chapitre :" print " " + str(noeudClic.text(0)) print'' # Comportement de la boîte de dialogue. FIN # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # Analyse brutale du changement de contenu. DEBUT def chgtTexte(self): text_TabMatBrut = self.textUtilisateur.toPlainText() # Traduction du contenu sous forme d'un tableau. self.tablo_tabMat = [] niv = 0 maj = True lignes_tabMat_Sale = text_TabMatBrut.split("\n") lignes_tabMat = [] for i in range(len(lignes_tabMat_Sale)): # On nettoie les lignes. uneLignePropre = str(lignes_tabMat_Sale[i]).strip() if uneLignePropre <> '' and uneLignePropre <> '>' and uneLignePropre <> '>>' : lignes_tabMat.append(uneLignePropre) # On parcourt les lignes en les analysant. for i in range(len(lignes_tabMat)): uneLignePropre = lignes_tabMat[i] # Repérage d'une PARTIE if uneLignePropre[:2] == '>>': if niv < 2: maj = False print u"Un problème de structure logique (sûrement temporaire) a été repéré.\nLa 1ère balise défectueuse est : \"" + uneLignePropre + '".\n' break else: uneLignePropre = uneLignePropre[2:].strip() if i == len(lignes_tabMat) - 1: tablo_Partie.append(uneLignePropre) tablo_Para.append(tablo_Partie) self.tablo_tabMat.append(tablo_Para) else: lignePropreSuivante = lignes_tabMat[i + 1] if lignePropreSuivante[:2] == '>>': tablo_Partie.append(uneLignePropre) elif lignePropreSuivante[0] == '>': tablo_Partie.append(uneLignePropre) tablo_Para.append(tablo_Partie) else : tablo_Partie.append(uneLignePropre) tablo_Para.append(tablo_Partie) self.tablo_tabMat.append(tablo_Para) niv = 3 # Repérage d'un PARAGRAPHE elif uneLignePropre[0] == '>': if niv == 0: maj = False print u"Un problème de structure logique (sûrement temporaire) a été repéré.\nLa 1ère balise défectueuse est : \"" + uneLignePropre + '".\n' break else: uneLignePropre = uneLignePropre[1:].strip() tablo_Partie = [] if i == len(lignes_tabMat) - 1: tablo_Para.append(uneLignePropre) self.tablo_tabMat.append(tablo_Para) else: lignePropreSuivante = lignes_tabMat[i + 1] if lignePropreSuivante[:2] == '>>': tablo_Partie.append(uneLignePropre) elif lignePropreSuivante[0] == '>': tablo_Para.append(uneLignePropre) else: tablo_Para.append(uneLignePropre) self.tablo_tabMat.append(tablo_Para) niv = 2 # Repérage d'un CHAPITRE else: niv = 1 tablo_Para = [] tablo_Partie = [] if i == len(lignes_tabMat) - 1: self.tablo_tabMat.append(uneLignePropre) else: lignePropreSuivante = lignes_tabMat[i + 1] if lignePropreSuivante[0] <> '>': self.tablo_tabMat.append(uneLignePropre) else: tablo_Para.append(uneLignePropre) # Mise à jour de l'arbre de la table (la technique est similaire à celle utilisée pour l'affichage HTML de la Table des Matières). if maj: if self.tablo_tabMat <> self.tablo_tabMat_Memo: self.lineTitre_2.setText(QtGui.QApplication.translate("dial_TestAnaBalise", "Arbre de la table des matières", None, QtGui.QApplication.UnicodeUTF8)) self.tablo_tabMat_Memo = self.tablo_tabMat self.treeWidget.clear() self.treeWidget.setColumnCount(1) self.treeWidget.setHeaderLabels(["Sommaire"]) self.treeWidget.setItemsExpandable(True) for i_Chap in range(len(self.tablo_tabMat)): try: # ATTENTION ! IL faut donner un tableau avec une seule chaîne pour nourrir l'arbre. # Donc tabNiv_1 = QtGui.QTreeWidgetItem(self.treeWidget, [self.tablo_tabMat[i_Chap]]) ne marchera # que si self.tablo_tabMat[i_Chap] est une chaîne de caractères. # Si ce n'est pas le cas, c'est que le chapitre contient des paragraphes. tabNiv_1 = QtGui.QTreeWidgetItem(self.treeWidget, [self.tablo_tabMat[i_Chap]]) tabNiv_1.setIcon(0, QtGui.QIcon('Images/im_1.png')) except: tabloPara = self.tablo_tabMat[i_Chap] for i_Para in range(len(tabloPara)): if i_Para == 0: tabNiv_1 = QtGui.QTreeWidgetItem(self.treeWidget, [tabloPara[0]]) tabNiv_1.setIcon(0, QtGui.QIcon('Images/im_1.png')) self.treeWidget.expandItem(tabNiv_1) else: try: tabNiv_2 = QtGui.QTreeWidgetItem(tabNiv_1, [tabloPara[i_Para]]) tabNiv_2.setIcon(0, QtGui.QIcon('Images/im_2.png')) self.treeWidget.expandItem(tabNiv_2) except: tabloPartie = tabloPara[i_Para] for i_Partie in range(len(tabloPartie)): if i_Partie == 0: tabNiv_2 = QtGui.QTreeWidgetItem(tabNiv_1, [tabloPartie[0]]) tabNiv_2.setIcon(0, QtGui.QIcon('Images/im_2.png')) self.treeWidget.expandItem(tabNiv_2) else: tabNiv_3 = QtGui.QTreeWidgetItem(tabNiv_2, [tabloPartie[i_Partie]]) tabNiv_3.setIcon(0, QtGui.QIcon('Images/im_3.png')) self.treeWidget.expandItem(tabNiv_3) # Pas de mise à jour possible. else : self.tablo_tabMat_Memo = self.tablo_tabMat self.lineTitre_2.setText(QtGui.QApplication.translate("dial_TestAnaBalise", "Mise à jour impossible", None, QtGui.QApplication.UnicodeUTF8)) # Analyse brutale du changement de contenu. FIN # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # Lancement de l'application. if __name__ == "__main__": app = QtGui.QApplication(sys.argv) Projet3 = dial_Projet3() Projet3.show() sys.exit(app.exec_())
Code : Sélectionner tout - Visualiser dans une fenêtre à part
1
2
3
4
5
6
7
8
9
10
11
12
13
14 def afficheImage(self, point): # On récupère les infos sur l'item sélectionné. index = self.treeWidget.indexAt(point) if not index.isValid(): return # print index.text(0) item = self.treeWidget.itemAt(point) name = item.text(0) # Nom du noeaud # Problème des grands images à gérer par exemple en les redimensionnant ou en utilisant un GraphicsView et des scrollbars. pixmap = QtGui.QPixmap("normale.jpg") # pixmap = QtGui.QPixmap("tropGrande.jpg") frame = MyFrame(self, name, pixmap, self.mapToGlobal(QtGui.QCursor.pos())) frame.show()Il reste un énorme problème que je pose dans le post suivant : voir ici . Une solution y est proposée.
Code : Sélectionner tout - Visualiser dans une fenêtre à part
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21 class MyFrame(QtGui.QFrame): # Squelette donné par Brian Kelley sur la liste de PyQt. def __init__(self, parent, titre, pixmapfile, position): QtGui.QFrame.__init__(self, parent, QtCore.Qt.WindowStaysOnTopHint | QtCore.Qt.Tool) # QtCore.Qt.X11BypassWindowManagerHint | QtCore.Qt.FramelessWindowHint | QtCore.Qt.WindowStaysOnTopHint | QtCore.Qt.Tool self.setWindowTitle(titre) # rect=QtCore.QRect(position, QtCore.QPoint(1000,1000)) # self.setFrameRect(rect) label = QtGui.QLabel(self) label.setPixmap(pixmapfile) label.show() self.timer = QtCore.QTimer() self.connect( self.timer, QtCore.SIGNAL("timeout()"), self, QtCore.SLOT("hide()") ) def showEvent(self, evt): self.timer.start(10000) # 10 seconds QtGui.QFrame.showEvent(self, evt)
Merci à shadowsam car l'utilisation d'une frame permet de faire des menus contextuels très évolués.Cela va me rendre de grands services.
![]()
Vous avez un bloqueur de publicités installé.
Le Club Developpez.com n'affiche que des publicités IT, discrètes et non intrusives.
Afin que nous puissions continuer à vous fournir gratuitement du contenu de qualité, merci de nous soutenir en désactivant votre bloqueur de publicités sur Developpez.com.
Partager