1081 lines
62 KiB
HTML
1081 lines
62 KiB
HTML
<html><head><title>Dialog creation/fr</title><meta http-equiv="Content-Type" content="text/html; charset=utf-8"><link type='text/css' href='wiki.css' rel='stylesheet'></head><body><h1>Dialog creation/fr</h1></div>
|
||
|
||
<div id="mw-content-text" lang="fr" dir="ltr" class="mw-content-ltr"><hr/><div class="mw-parser-output"><p>Dans cette page nous allons vous montrer comment construire une simple boîte de dialogue avec <a rel="nofollow" class="external text" href="http://qt-project.org/doc/qt-4.8/designer-manual.html">Qt Designer</a>, Qt Designer, est l'outil officiel de Qt pour la conception d'interfaces (Gui), puis de le convertir en code Python, et l'utiliser à l'intérieur de FreeCAD.<br />
|
||
Je vais supposer, que pour l'exemple, vous savez déjà comment modifier et exécuter un script Python, et que vous pouvez travailler avec des choses simples dans une fenêtre de terminal tel que se déplacer, etc . . Bien sûr, vous devez également avoir installé <b><a href="https://www.freecadweb.org/wiki/index.php?title=PySide/fr" title="PySide/fr">PySide</a></b>.
|
||
</p>
|
||
<div id="toc" class="toc"><div class="toctitle"><h2>Contents</h2></div>
|
||
<ul>
|
||
<li class="toclevel-1 tocsection-1"><a href="#Construire_une_bo.C3.AEte_de_dialogue"><span class="tocnumber">1</span> <span class="toctext">Construire une boîte de dialogue</span></a></li>
|
||
<li class="toclevel-1 tocsection-2"><a href="#Conversion_de_notre_bo.C3.AEte_de_dialogue_en_code_Python_avec_.22pyuic.22"><span class="tocnumber">2</span> <span class="toctext">Conversion de notre boîte de dialogue en code Python avec "pyuic"</span></a></li>
|
||
<li class="toclevel-1 tocsection-3"><a href="#Faire_quelque_chose_avec_notre_bo.C3.AEte_de_dialogue"><span class="tocnumber">3</span> <span class="toctext">Faire quelque chose avec notre boîte de dialogue</span></a></li>
|
||
<li class="toclevel-1 tocsection-4"><a href="#Le_script_complet"><span class="tocnumber">4</span> <span class="toctext">Le script complet</span></a></li>
|
||
<li class="toclevel-1 tocsection-5"><a href="#Cr.C3.A9ation_d.27une_bo.C3.AEte_de_dialogue_avec_ses_boutons"><span class="tocnumber">5</span> <span class="toctext">Création d'une boîte de dialogue avec ses boutons</span></a>
|
||
<ul>
|
||
<li class="toclevel-2 tocsection-6"><a href="#M.C3.A9thode_1"><span class="tocnumber">5.1</span> <span class="toctext">Méthode 1</span></a></li>
|
||
<li class="toclevel-2 tocsection-7"><a href="#M.C3.A9thode_2"><span class="tocnumber">5.2</span> <span class="toctext">Méthode 2</span></a></li>
|
||
</ul>
|
||
</li>
|
||
<li class="toclevel-1 tocsection-8"><a href="#Ic.C3.B4nes_personnalis.C3.A9s_dans_la_Vue_combin.C3.A9e"><span class="tocnumber">6</span> <span class="toctext">Icônes personnalisés dans la Vue combinée</span></a></li>
|
||
<li class="toclevel-1 tocsection-9"><a href="#Utiliser_QFileDialog_for_.C3.A9crire_dans_un_fichier"><span class="tocnumber">7</span> <span class="toctext">Utiliser QFileDialog for écrire dans un fichier</span></a></li>
|
||
<li class="toclevel-1 tocsection-10"><a href="#Utiliser_QFileDialog_pour_lire_un_fichier"><span class="tocnumber">8</span> <span class="toctext">Utiliser QFileDialog pour lire un fichier</span></a></li>
|
||
<li class="toclevel-1 tocsection-11"><a href="#Utiliser_QColorDialog_pour_utiliser_une_couleur"><span class="tocnumber">9</span> <span class="toctext">Utiliser QColorDialog pour utiliser une couleur</span></a></li>
|
||
<li class="toclevel-1 tocsection-12"><a href="#Quelques_commandes_utiles"><span class="tocnumber">10</span> <span class="toctext">Quelques commandes utiles</span></a></li>
|
||
</ul>
|
||
</div>
|
||
|
||
<h2><span class="mw-headline" id="Construire_une_bo.C3.AEte_de_dialogue">Construire une boîte de dialogue</span></h2>
|
||
<p>Dans les applications de CAO, bien concevoir une <b>UI</b> (interface utilisateur) est très important.<br />
|
||
Tout ce que l'utilisateur fera, se fera à travers un outil de l'interface: la lecture des boîtes de dialogue, appuyer sur les boutons, le choix entre les icônes, etc . . <br />
|
||
Il est donc très important de réfléchir attentivement à la conception de votre boîte de dialogue, comment vous voulez que l'utilisateur se comporter avec la boîte, et comment sera le flux de travail de votre action.
|
||
</p><p>Il y a une deux choses à savoir lors de la conception de l'interface:<br />
|
||
</p>
|
||
<ul><li> Boîtes de dialogue <a rel="nofollow" class="external text" href="http://fr.wikipedia.org/wiki/Fenêtre_modale">modales ou non-modale</a> :
|
||
<ul><li> Une boîte de dialogue <b>modale</b> apparaît en face de votre écran et, arrête l'action de la fenêtre principale, forçant l'utilisateur à répondre à la boîte de dialogue.</li>
|
||
<li> Une boîte de dialogue <b>non modale</b> ne vous empêche pas de travailler sur la Fenêtre principale, vous pouvez travailler sur les deux fenêtres.<br /></li></ul></li></ul>
|
||
<p>Dans certains cas, le premier est préférable, dans d'autres cas non.
|
||
</p>
|
||
<ul><li> Identifier ce qui est nécessaire et ce qui est optionnel:<br />
|
||
<ul><li> Assurez-vous que l'utilisateur sait ce qu'il doit faire. Prévoyez des étiquettes avec des descriptions appropriées, des info-bulles d'utilisation, etc . . <br /></li></ul></li>
|
||
<li> Séparez les commandes à partir de paramètres:<br />
|
||
<ul><li> Cela se fait habituellement avec des boutons et des champs de saisie de texte.<br /></li>
|
||
<li> L'utilisateur sait que cliquer sur un bouton va produire une action, tout en changeant une valeur dans un champ de texte, va changer un paramètre quelque part. Cependant, aujourd'hui, les utilisateurs savent généralement bien ce qu'est un bouton, ce qu'est un champ de saisie, etc . . .<br /></li></ul></li></ul>
|
||
<p>La boîte à outils de l'interface <b>Qt</b> que nous utilisons, est une boîte à outils <b>state-of-the-art</b> (interface graphique avancée), et nous n'aurons pas beaucoup d'inquiétudes pour rendre les choses claires, car elles sont déjà très claires par elles-mêmes.
|
||
</p><p>Donc, maintenant que nous avons bien défini ce que nous ferons, il est temps d'ouvrir <b>Qt Designer</b>.<br />
|
||
Nous allons concevoir très facilement une simple boîte de dialogue, comme ceci:
|
||
</p><p><a href="https://www.freecadweb.org/wiki/index.php?title=File:Qttestdialog.jpg" class="image"><img alt="Qttestdialog.jpg" src="Qttestdialog.jpg" width="210" height="216" /></a>
|
||
</p><p>Nous allons ensuite utiliser cette boîte de dialogue dans FreeCAD pour produire une belle surface plane rectangulaire.<br />
|
||
Vous ne trouverez peut-être pas très utile de produire de beaux plans rectangulaires, mais il sera facile de le changer plus tard et de faire des choses plus complexes.<br /> Lorsque vous l'ouvrez, Qt Designer ressemble à ceci:
|
||
</p><p><a href="https://www.freecadweb.org/wiki/index.php?title=File:Qtdesigner-screenshot.jpg" class="image"><img alt="Qtdesigner-screenshot.jpg" src="Qtdesigner-screenshot.jpg" width="800" height="486" /></a>
|
||
</p><p>Il est très simple à utiliser. Sur la barre de gauche vous avez des éléments qui peuvent être glissés sur votre widget (tous les outils). Sur le côté droit vous avez des panneaux d'affichage de propriétés de toutes sortes, des propriétés de certains éléments modifiables.<br />
|
||
Donc, commencez par créer un nouveau widget. Sélectionnez "<b>Dialog without buttons</b>", car nous ne voulons pas de boutons par défaut Ok/Annuler. Ensuite, faites glisser sur votre widget <b>3 labels</b>, un pour le titre, un pour l'écriture "Height" (Hauteur) et l'autre pour l'écriture "Width" (Largeur).<br /><br />
|
||
Les <b>labels</b> (étiquettes) sont de simples textes qui apparaissent sur votre widget, il servent a informer l'utilisateur.<br />
|
||
Si vous sélectionnez un <b>label</b>, sur le côté droit apparaîssent plusieurs propriétés que vous pouvez modifier, comme le style de police, taille, etc . . .
|
||
</p><p>Ensuite, ajoutez 2 <b>LineEdits</b> , qui sont des champs texte que l'utilisateur peut remplir, un pour la hauteur et l'autre pour la largeur.<br /><br />
|
||
Ici aussi, nous pouvons modifier les propriétés. Par exemple, pourquoi ne pas définir une valeur par défaut ? Par exemple 1,00 pour chacun d'eux.<br />
|
||
De cette façon, lorsque l'utilisateur verra la boîte de dialogue, les deux valeurs seront déjà remplies et si les valeurs conviennent, il peut directement appuyer sur le bouton, gain de temps précieux.<br />
|
||
Ensuite, ajoutez un <b>PushButton</b> , qui est le bouton, que l'utilisateur devra appuyer après avoir rempli les 2 champs.
|
||
</p><p>Notez que j'ai choisi ici des contrôles très simples, mais <b>Qt</b> a beaucoup plus d'options, par exemple, vous pouvez utiliser <b>spinbox</b> au lieu de <b>LineEdits</b>, etc .. <br />
|
||
Regardez tout ce qui est disponible, vous aurez sûrement d'autres idées.
|
||
</p><p>C'est à peu près tout ce que nous devons faire dans Qt Designer.<br />
|
||
Une dernière chose, nous allons renommer tous nos éléments avec des noms faciles, de sorte qu'il sera plus facile de les identifier dans nos scripts:
|
||
</p><p><a href="https://www.freecadweb.org/wiki/index.php?title=File:Qtpropeditor.jpg" class="image"><img alt="Qtpropeditor.jpg" src="Qtpropeditor.jpg" width="348" height="321" /></a>
|
||
</p>
|
||
<h2><span class="mw-headline" id="Conversion_de_notre_bo.C3.AEte_de_dialogue_en_code_Python_avec_.22pyuic.22">Conversion de notre boîte de dialogue en code Python avec "pyuic"</span></h2>
|
||
<p>Maintenant, nous allons sauver notre widget quelque part. Il sera sauvegardé dans un fichier <b>.Ui</b>, que nous allons facilement convertir en script Python avec <b>pyuic</b>.<br />
|
||
Dans windows, le programme est livré avec <b>pyuic pyqt</b> (à vérifier), sur Linux, vous aurez probablement besoin de l'installer séparément à partir de votre gestionnaire de paquets (sur debian-systèmes basés sur, il fait partie du paquet pyqt4-dev-tools).<br />
|
||
Pour faire la conversion, vous aurez besoin d'ouvrir une fenêtre de terminal (ou une fenêtre d'invite de commandes), accédez à l'endroit où vous avez enregistré votre fichier <b>ui</b> :
|
||
</p>
|
||
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre>pyuic mywidget.ui > mywidget.py</pre></div>
|
||
<p>Dans Windows pyuic.py est présent dans "C:\Python27\Lib\site-packages\PyQt4\uic\pyuic.py"
|
||
Créez un fichier batch "compQt4.bat" pour automatiser la tâche:
|
||
</p>
|
||
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre>@"C:\Python27\python" "C:\Python27\Lib\site-packages\PyQt4\uic\pyuic.py" -x %1.ui > %1.py</pre></div>
|
||
<p>Dans la console Dos tapez sans extension
|
||
</p>
|
||
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre>compQt4 myUiFile</pre></div>
|
||
<p>Dans Linux : à venir
|
||
</p><p>Depuis la version 0.13, FreeCAD migre progressivement de PyQt à <a rel="nofollow" class="external text" href="http://qt-project.org/wiki/PySide">PySide</a> (Choisissez votre installateur PySide suivant votre système <a rel="nofollow" class="external text" href="http://pyside.readthedocs.org/en/latest/building/">building PySide</a>),
|
||
</p>
|
||
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre>pyside-uic mywidget.ui -o mywidget.py</pre></div>
|
||
<p>Dans Windows le fichier uic.py se trouve dans "C:\Python27\Lib\site-packages\PySide\scripts\uic.py"
|
||
Créez un fichier batch "compSide.bat" pour automatiser la tâche:
|
||
</p>
|
||
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre>@"C:\Python27\python" "C:\Python27\Lib\site-packages\PySide\scripts\uic.py" %1.ui > %1.py</pre></div>
|
||
<p>Dans la console Python tapez sans extension:
|
||
</p>
|
||
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre>compSide myUiFile</pre></div>
|
||
<p>Into Linux : to do
|
||
</p><p><br />
|
||
Sur certains systèmes, le programme est appelé pyuic4 lieu de pyuic. Il sert uniquement de convertisseur du fichier .ui pour l'utiliser dans un script Python. Si nous ouvrons le fichier mywidget.py, son contenu est très facile à comprendre:
|
||
</p>
|
||
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre>from PySide import QtCore, QtGui
|
||
|
||
class Ui_Dialog(object):
|
||
def setupUi(self, Dialog):
|
||
Dialog.setObjectName("Dialog")
|
||
Dialog.resize(187, 178)
|
||
self.title = QtGui.QLabel(Dialog)
|
||
self.title.setGeometry(QtCore.QRect(10, 10, 271, 16))
|
||
self.title.setObjectName("title")
|
||
self.label_width = QtGui.QLabel(Dialog)
|
||
...
|
||
|
||
self.retranslateUi(Dialog)
|
||
QtCore.QMetaObject.connectSlotsByName(Dialog)
|
||
|
||
def retranslateUi(self, Dialog):
|
||
Dialog.setWindowTitle(QtGui.QApplication.translate("Dialog", "Dialog", None, QtGui.QApplication.UnicodeUTF8))
|
||
self.title.setText(QtGui.QApplication.translate("Dialog", "Plane-O-Matic", None, QtGui.QApplication.UnicodeUTF8))
|
||
...</pre></div>
|
||
<p>Comme vous voyez, il a une structure très simple: une classe nommée <b>Ui_Dialog</b> est créé, qui stocke les éléments de l'interface de notre widget.<br />
|
||
Cette classe dispose de deux méthodes, une pour la mise en place du widget, et l'autre pour traduire son contenu, qui fait partie du mécanisme général de Qt pour la traduction des éléments d'interface.<br />
|
||
La méthode de configuration, crée simplement, un par un, les widgets tels que nous les avons définis dans Qt Designer, et définit leurs options aussi comme nous avons décidé plus tôt.<br /><br />
|
||
Puis, toute l'interface est traduite, et enfin, les "slots" se connectent (nous en reparlerons plus tard).
|
||
</p><p>Nous pouvons maintenant créer un nouveau widget, et utiliser cette classe pour créer son interface.<br />
|
||
Nous pouvons déjà voir notre widget en action, en mettant notre fichier mywidget.py dans un endroit où FreeCAD la trouvera (dans le répertoire bin FreeCAD, ou dans l'un des sous-répertoires Mod), et, dans l'interpréteur Python de FreeCAD, faisons:
|
||
</p>
|
||
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre>from PySide import QtGui
|
||
import mywidget
|
||
d = QtGui.QWidget()
|
||
d.ui = mywidget.Ui_Dialog()
|
||
d.ui.setupUi(d)
|
||
d.show()</pre></div>
|
||
<p>Et notre boîte de dialogue apparaîtra! Notez que notre interpréteur Python fonctionne toujours, nous avons une boîte de dialogue non modale.<br />
|
||
Donc, pour la fermer, nous pouvons (à part cliquer sur son icône, bien sûr) faire:
|
||
</p>
|
||
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre>d.hide()</pre></div>
|
||
<h2><span class="mw-headline" id="Faire_quelque_chose_avec_notre_bo.C3.AEte_de_dialogue">Faire quelque chose avec notre boîte de dialogue</span></h2>
|
||
<p>Maintenant que nous pouvons afficher et masquer notre boîte de dialogue, nous avons juste besoin d'ajouter la dernière partie, pour en faire quelque chose !<br />
|
||
Si vous explorez un peu Qt Designer, vous découvrirez rapidement toute une section appelée "<b>signaux et slots</b>".<br />
|
||
Fondamentalement, cela fonctionne comme ceci, ce sont les éléments sur vos widgets (dans la terminologie de Qt, ces éléments sont eux-mêmes des widgets) qui peuvent envoyer des signaux.<br /><br />
|
||
Ces signaux diffèrent selon le type de widget. Par exemple, un bouton peut envoyer un signal quand il est pressé et quand il est relâché.<br />
|
||
Ces signaux peuvent être connectés à des créneaux, qui peuvent être des fonctionnalités spéciales d'autres widgets (par exemple une boîte de dialogue a un bouton "Fermer" sur lequel vous pouvez connecter le signal à partir d'un autre bouton "Fermer"), ou, peuvent être des fonctions personnalisées.<br />
|
||
<a rel="nofollow" class="external text" href="http://www.riverbankcomputing.co.uk/static/Docs/PyQt4/html/classes.html">La documentation de référence PyQt</a> répertorie tous les widgets Qt, ce qu'ils peuvent faire, ce qu'ils signalent, ce qu'ils peuvent envoyer, etc . . .
|
||
</p><p>Ce que nous allons faire ici, c'est créer une nouvelle fonction qui permettra de créer une surface plane basée sur la hauteur et la largeur, et, relier cette fonction au bouton "Create!".<br />
|
||
Donc, nous allons commencer par importer nos modules FreeCAD, en mettant la ligne suivante en haut du script, où nous importons déjà <b>QtCore</b> et <b>QtGui</b>:
|
||
</p>
|
||
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre>import FreeCAD, Part</pre></div>
|
||
<p>Ensuite, nous allons ajouter une nouvelle fonction à notre classe <b>Ui_Dialog</b>:
|
||
</p>
|
||
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre>def createPlane(self):
|
||
try:
|
||
# first we check if valid numbers have been entered
|
||
w = float(self.width.text())
|
||
h = float(self.height.text())
|
||
except ValueError:
|
||
print "Error! Width and Height values must be valid numbers!"
|
||
else:
|
||
# create a face from 4 points
|
||
p1 = FreeCAD.Vector(0,0,0)
|
||
p2 = FreeCAD.Vector(w,0,0)
|
||
p3 = FreeCAD.Vector(w,h,0)
|
||
p4 = FreeCAD.Vector(0,h,0)
|
||
pointslist = [p1,p2,p3,p4,p1]
|
||
mywire = Part.makePolygon(pointslist)
|
||
myface = Part.Face(mywire)
|
||
Part.show(myface)
|
||
self.hide()</pre></div>
|
||
<p>Puis, nous avons besoin d'informer Qt pour qu'il se connecte sur le bouton de la fonction, en plaçant la ligne suivante juste avant <b>QtCore.QMetaObject.connectSlotsByName(Dialog)</b>:
|
||
</p>
|
||
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre>QtCore.QObject.connect(self.create,QtCore.SIGNAL("pressed()"),self.createPlane)</pre></div>
|
||
<p>Il s'agit, comme vous le voyez, de relier le signal du bouton enfoncé de l'objet a créer (<b>"Create!" Bouton</b>), à un emplacement nommé <b>createPlane</b>, dont nous venons de définir.<br />
|
||
Ça y est ! Maintenant, la touche finale, nous pouvons ajouter une petite fonction, pour créer la boîte de dialogue, elle sera plus facile a appeler.<br />
|
||
En dehors de la classe <b>Ui_Dialog class</b>, nous allons ajouter le code suivant:
|
||
</p>
|
||
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre>class plane():
|
||
def __init__(self):
|
||
self.d = QtGui.QWidget()
|
||
self.ui = Ui_Dialog()
|
||
self.ui.setupUi(self.d)
|
||
self.d.show()</pre></div>
|
||
<p>(Rappel sur Python : la méthode <b>__init__</b> est une classe qui s'exécute automatiquement chaque fois qu'un nouvel objet est créé ! )
|
||
</p><p>Puis, à partir de FreeCAD, nous avons seulement besoin de faire:
|
||
</p>
|
||
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre>import mywidget
|
||
myDialog = mywidget.plane()</pre></div>
|
||
<p>Voilà, c'est tout ...<br />
|
||
Maintenant, vous pouvez essayer toutes sortes de choses, comme par exemple l'insertion de votre widget dans l'interface FreeCAD (voir la page <a href="https://www.freecadweb.org/wiki/index.php?title=Code_snippets/fr" title="Code snippets/fr">Code snippets</a>), ou, faire des outils personnalisés beaucoup plus avancés, en utilisant d'autres éléments dans votre widget.
|
||
</p>
|
||
<h2><span class="mw-headline" id="Le_script_complet">Le script complet</span></h2>
|
||
<p>Ceci est le script de référence complet:
|
||
</p>
|
||
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre># -*- coding: utf-8 -*-
|
||
|
||
# Form implementation generated from reading ui file 'mywidget.ui'
|
||
#
|
||
# Created: Mon Jun 1 19:09:10 2009
|
||
# by: PyQt4 UI code generator 4.4.4
|
||
# Modified for PySide 16:02:2015
|
||
# WARNING! All changes made in this file will be lost!
|
||
|
||
from PySide import QtCore, QtGui
|
||
import FreeCAD, Part
|
||
|
||
class Ui_Dialog(object):
|
||
def setupUi(self, Dialog):
|
||
Dialog.setObjectName("Dialog")
|
||
Dialog.resize(187, 178)
|
||
self.title = QtGui.QLabel(Dialog)
|
||
self.title.setGeometry(QtCore.QRect(10, 10, 271, 16))
|
||
self.title.setObjectName("title")
|
||
self.label_width = QtGui.QLabel(Dialog)
|
||
self.label_width.setGeometry(QtCore.QRect(10, 50, 57, 16))
|
||
self.label_width.setObjectName("label_width")
|
||
self.label_height = QtGui.QLabel(Dialog)
|
||
self.label_height.setGeometry(QtCore.QRect(10, 90, 57, 16))
|
||
self.label_height.setObjectName("label_height")
|
||
self.width = QtGui.QLineEdit(Dialog)
|
||
self.width.setGeometry(QtCore.QRect(60, 40, 111, 26))
|
||
self.width.setObjectName("width")
|
||
self.height = QtGui.QLineEdit(Dialog)
|
||
self.height.setGeometry(QtCore.QRect(60, 80, 111, 26))
|
||
self.height.setObjectName("height")
|
||
self.create = QtGui.QPushButton(Dialog)
|
||
self.create.setGeometry(QtCore.QRect(50, 140, 83, 26))
|
||
self.create.setObjectName("create")
|
||
|
||
self.retranslateUi(Dialog)
|
||
QtCore.QObject.connect(self.create,QtCore.SIGNAL("pressed()"),self.createPlane)
|
||
QtCore.QMetaObject.connectSlotsByName(Dialog)
|
||
|
||
def retranslateUi(self, Dialog):
|
||
Dialog.setWindowTitle(QtGui.QApplication.translate("Dialog", "Dialog", None, QtGui.QApplication.UnicodeUTF8))
|
||
self.title.setText(QtGui.QApplication.translate("Dialog", "Plane-O-Matic", None, QtGui.QApplication.UnicodeUTF8))
|
||
self.label_width.setText(QtGui.QApplication.translate("Dialog", "Width", None, QtGui.QApplication.UnicodeUTF8))
|
||
self.label_height.setText(QtGui.QApplication.translate("Dialog", "Height", None, QtGui.QApplication.UnicodeUTF8))
|
||
self.create.setText(QtGui.QApplication.translate("Dialog", "Create!", None, QtGui.QApplication.UnicodeUTF8))
|
||
|
||
def createPlane(self):
|
||
try:
|
||
# first we check if valid numbers have been entered
|
||
w = float(self.width.text())
|
||
h = float(self.height.text())
|
||
except ValueError:
|
||
print "Error! Width and Height values must be valid numbers!"
|
||
else:
|
||
# create a face from 4 points
|
||
p1 = FreeCAD.Vector(0,0,0)
|
||
p2 = FreeCAD.Vector(w,0,0)
|
||
p3 = FreeCAD.Vector(w,h,0)
|
||
p4 = FreeCAD.Vector(0,h,0)
|
||
pointslist = [p1,p2,p3,p4,p1]
|
||
mywire = Part.makePolygon(pointslist)
|
||
myface = Part.Face(mywire)
|
||
Part.show(myface)
|
||
|
||
class plane():
|
||
def __init__(self):
|
||
self.d = QtGui.QWidget()
|
||
self.ui = Ui_Dialog()
|
||
self.ui.setupUi(self.d)
|
||
self.d.show()</pre></div>
|
||
<h2><span class="mw-headline" id="Cr.C3.A9ation_d.27une_bo.C3.AEte_de_dialogue_avec_ses_boutons">Création d'une boîte de dialogue avec ses boutons</span></h2>
|
||
<h3><span class="mw-headline" id="M.C3.A9thode_1">Méthode 1</span></h3>
|
||
<p>Un exemple d'une boîte de dialogue complète avec ses connections.
|
||
</p>
|
||
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre># -*- coding: utf-8 -*-
|
||
# Create by flachyjoe
|
||
|
||
from PySide import QtCore, QtGui
|
||
|
||
try:
|
||
_fromUtf8 = QtCore.QString.fromUtf8
|
||
except AttributeError:
|
||
def _fromUtf8(s):
|
||
return s
|
||
|
||
try:
|
||
_encoding = QtGui.QApplication.UnicodeUTF8
|
||
def _translate(context, text, disambig):
|
||
return QtGui.QApplication.translate(context, text, disambig, _encoding)
|
||
except AttributeError:
|
||
def _translate(context, text, disambig):
|
||
return QtGui.QApplication.translate(context, text, disambig)
|
||
|
||
|
||
class Ui_MainWindow(object):
|
||
|
||
def __init__(self, MainWindow):
|
||
self.window = MainWindow
|
||
|
||
MainWindow.setObjectName(_fromUtf8("MainWindow"))
|
||
MainWindow.resize(400, 300)
|
||
self.centralWidget = QtGui.QWidget(MainWindow)
|
||
self.centralWidget.setObjectName(_fromUtf8("centralWidget"))
|
||
|
||
self.pushButton = QtGui.QPushButton(self.centralWidget)
|
||
self.pushButton.setGeometry(QtCore.QRect(30, 170, 93, 28))
|
||
self.pushButton.setObjectName(_fromUtf8("pushButton"))
|
||
self.pushButton.clicked.connect(self.on_pushButton_clicked) #connection pushButton
|
||
|
||
self.lineEdit = QtGui.QLineEdit(self.centralWidget)
|
||
self.lineEdit.setGeometry(QtCore.QRect(30, 40, 211, 22))
|
||
self.lineEdit.setObjectName(_fromUtf8("lineEdit"))
|
||
self.lineEdit.returnPressed.connect(self.on_lineEdit_clicked) #connection lineEdit
|
||
|
||
self.checkBox = QtGui.QCheckBox(self.centralWidget)
|
||
self.checkBox.setGeometry(QtCore.QRect(30, 90, 81, 20))
|
||
self.checkBox.setChecked(True)
|
||
self.checkBox.setObjectName(_fromUtf8("checkBoxON"))
|
||
self.checkBox.clicked.connect(self.on_checkBox_clicked) #connection checkBox
|
||
|
||
self.radioButton = QtGui.QRadioButton(self.centralWidget)
|
||
self.radioButton.setGeometry(QtCore.QRect(30, 130, 95, 20))
|
||
self.radioButton.setObjectName(_fromUtf8("radioButton"))
|
||
self.radioButton.clicked.connect(self.on_radioButton_clicked) #connection radioButton
|
||
|
||
MainWindow.setCentralWidget(self.centralWidget)
|
||
|
||
self.menuBar = QtGui.QMenuBar(MainWindow)
|
||
self.menuBar.setGeometry(QtCore.QRect(0, 0, 400, 26))
|
||
self.menuBar.setObjectName(_fromUtf8("menuBar"))
|
||
MainWindow.setMenuBar(self.menuBar)
|
||
|
||
self.mainToolBar = QtGui.QToolBar(MainWindow)
|
||
self.mainToolBar.setObjectName(_fromUtf8("mainToolBar"))
|
||
MainWindow.addToolBar(QtCore.Qt.TopToolBarArea, self.mainToolBar)
|
||
|
||
self.statusBar = QtGui.QStatusBar(MainWindow)
|
||
self.statusBar.setObjectName(_fromUtf8("statusBar"))
|
||
MainWindow.setStatusBar(self.statusBar)
|
||
|
||
self.retranslateUi(MainWindow)
|
||
|
||
def retranslateUi(self, MainWindow):
|
||
MainWindow.setWindowTitle(_translate("MainWindow", "MainWindow", None))
|
||
self.pushButton.setText(_translate("MainWindow", "OK", None))
|
||
self.lineEdit.setText(_translate("MainWindow", "tyty", None))
|
||
self.checkBox.setText(_translate("MainWindow", "CheckBox", None))
|
||
self.radioButton.setText(_translate("MainWindow", "RadioButton", None))
|
||
|
||
def on_checkBox_clicked(self):
|
||
if self.checkBox.checkState()==0:
|
||
App.Console.PrintMessage(str(self.checkBox.checkState())+" CheckBox KO\r\n")
|
||
else:
|
||
App.Console.PrintMessage(str(self.checkBox.checkState())+" CheckBox OK\r\n")
|
||
# App.Console.PrintMessage(str(self.lineEdit.setText("tititi"))+" LineEdit\r\n") #write text to the lineEdit window !
|
||
# str(self.lineEdit.setText("tititi")) #écrit le texte dans la fenêtre lineEdit
|
||
App.Console.PrintMessage(str(self.lineEdit.displayText())+" LineEdit\r\n")
|
||
|
||
def on_radioButton_clicked(self):
|
||
if self.radioButton.isChecked():
|
||
App.Console.PrintMessage(str(self.radioButton.isChecked())+" Radio OK\r\n")
|
||
else:
|
||
App.Console.PrintMessage(str(self.radioButton.isChecked())+" Radio KO\r\n")
|
||
|
||
def on_lineEdit_clicked(self):
|
||
# if self.lineEdit.textChanged():
|
||
App.Console.PrintMessage(str(self.lineEdit.displayText())+" LineEdit Display\r\n")
|
||
|
||
def on_pushButton_clicked(self):
|
||
App.Console.PrintMessage("Terminé\r\n")
|
||
self.window.hide()
|
||
|
||
MainWindow = QtGui.QMainWindow()
|
||
ui = Ui_MainWindow(MainWindow)
|
||
MainWindow.show()</pre></div>
|
||
<p>Ici la même fenêtre mais avec un icône sur chaque bouton.
|
||
</p><p>Téléchargez les icônes associés (Faites clic droit sur l'icône "Enregistrer l'image sous ...)"
|
||
</p><p><a href="https://www.freecadweb.org/wiki/index.php?title=File:Icone01.png" class="image"><img alt="Icone01.png" src="Icone01.png" width="64" height="64" /></a> <a href="https://www.freecadweb.org/wiki/index.php?title=File:Icone02.png" class="image"><img alt="Icone02.png" src="Icone02.png" width="64" height="64" /></a> <a href="https://www.freecadweb.org/wiki/index.php?title=File:Icone03.png" class="image"><img alt="Icone03.png" src="Icone03.png" width="64" height="64" /></a>
|
||
</p>
|
||
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre># -*- coding: utf-8 -*-
|
||
|
||
from PySide import QtCore, QtGui
|
||
|
||
try:
|
||
_fromUtf8 = QtCore.QString.fromUtf8
|
||
except AttributeError:
|
||
def _fromUtf8(s):
|
||
return s
|
||
|
||
try:
|
||
_encoding = QtGui.QApplication.UnicodeUTF8
|
||
def _translate(context, text, disambig):
|
||
return QtGui.QApplication.translate(context, text, disambig, _encoding)
|
||
except AttributeError:
|
||
def _translate(context, text, disambig):
|
||
return QtGui.QApplication.translate(context, text, disambig)
|
||
|
||
|
||
class Ui_MainWindow(object):
|
||
|
||
def __init__(self, MainWindow):
|
||
self.window = MainWindow
|
||
path = FreeCAD.ConfigGet("UserAppData")
|
||
# path = FreeCAD.ConfigGet("AppHomePath")
|
||
|
||
MainWindow.setObjectName(_fromUtf8("MainWindow"))
|
||
MainWindow.resize(400, 300)
|
||
self.centralWidget = QtGui.QWidget(MainWindow)
|
||
self.centralWidget.setObjectName(_fromUtf8("centralWidget"))
|
||
|
||
self.pushButton = QtGui.QPushButton(self.centralWidget)
|
||
self.pushButton.setGeometry(QtCore.QRect(30, 170, 93, 28))
|
||
self.pushButton.setObjectName(_fromUtf8("pushButton"))
|
||
self.pushButton.clicked.connect(self.on_pushButton_clicked) #connection pushButton
|
||
|
||
self.lineEdit = QtGui.QLineEdit(self.centralWidget)
|
||
self.lineEdit.setGeometry(QtCore.QRect(30, 40, 211, 22))
|
||
self.lineEdit.setObjectName(_fromUtf8("lineEdit"))
|
||
self.lineEdit.returnPressed.connect(self.on_lineEdit_clicked) #connection lineEdit
|
||
|
||
self.checkBox = QtGui.QCheckBox(self.centralWidget)
|
||
self.checkBox.setGeometry(QtCore.QRect(30, 90, 100, 20))
|
||
self.checkBox.setChecked(True)
|
||
self.checkBox.setObjectName(_fromUtf8("checkBoxON"))
|
||
self.checkBox.clicked.connect(self.on_checkBox_clicked) #connection checkBox
|
||
|
||
self.radioButton = QtGui.QRadioButton(self.centralWidget)
|
||
self.radioButton.setGeometry(QtCore.QRect(30, 130, 95, 20))
|
||
self.radioButton.setObjectName(_fromUtf8("radioButton"))
|
||
self.radioButton.clicked.connect(self.on_radioButton_clicked) #connection radioButton
|
||
|
||
MainWindow.setCentralWidget(self.centralWidget)
|
||
|
||
self.menuBar = QtGui.QMenuBar(MainWindow)
|
||
self.menuBar.setGeometry(QtCore.QRect(0, 0, 400, 26))
|
||
self.menuBar.setObjectName(_fromUtf8("menuBar"))
|
||
MainWindow.setMenuBar(self.menuBar)
|
||
|
||
self.mainToolBar = QtGui.QToolBar(MainWindow)
|
||
self.mainToolBar.setObjectName(_fromUtf8("mainToolBar"))
|
||
MainWindow.addToolBar(QtCore.Qt.TopToolBarArea, self.mainToolBar)
|
||
|
||
self.statusBar = QtGui.QStatusBar(MainWindow)
|
||
self.statusBar.setObjectName(_fromUtf8("statusBar"))
|
||
MainWindow.setStatusBar(self.statusBar)
|
||
|
||
self.retranslateUi(MainWindow)
|
||
|
||
# Affiche un icone sur le bouton PushButton
|
||
# self.image_01 = "C:\Program Files\FreeCAD0.13\Icone01.png" # adapt the icon name
|
||
self.image_01 = path+"Icone01.png" # adapt the name of the icon
|
||
icon01 = QtGui.QIcon()
|
||
icon01.addPixmap(QtGui.QPixmap(self.image_01),QtGui.QIcon.Normal, QtGui.QIcon.Off)
|
||
self.pushButton.setIcon(icon01)
|
||
self.pushButton.setLayoutDirection(QtCore.Qt.RightToLeft) # This command reverses the direction of the button
|
||
|
||
# Affiche un icone sur le bouton RadioButton
|
||
# self.image_02 = "C:\Program Files\FreeCAD0.13\Icone02.png" # adapt the name of the icon
|
||
self.image_02 = path+"Icone02.png" # adapter le nom de l'icone
|
||
icon02 = QtGui.QIcon()
|
||
icon02.addPixmap(QtGui.QPixmap(self.image_02),QtGui.QIcon.Normal, QtGui.QIcon.Off)
|
||
self.radioButton.setIcon(icon02)
|
||
# self.radioButton.setLayoutDirection(QtCore.Qt.RightToLeft) # This command reverses the direction of the button
|
||
|
||
# Affiche un icone sur le bouton CheckBox
|
||
# self.image_03 = "C:\Program Files\FreeCAD0.13\Icone03.png" # the name of the icon
|
||
self.image_03 = path+"Icone03.png" # adapter le nom de l'icone
|
||
icon03 = QtGui.QIcon()
|
||
icon03.addPixmap(QtGui.QPixmap(self.image_03),QtGui.QIcon.Normal, QtGui.QIcon.Off)
|
||
self.checkBox.setIcon(icon03)
|
||
# self.checkBox.setLayoutDirection(QtCore.Qt.RightToLeft) # This command reverses the direction of the button
|
||
|
||
|
||
def retranslateUi(self, MainWindow):
|
||
MainWindow.setWindowTitle(_translate("MainWindow", "FreeCAD", None))
|
||
self.pushButton.setText(_translate("MainWindow", "OK", None))
|
||
self.lineEdit.setText(_translate("MainWindow", "tyty", None))
|
||
self.checkBox.setText(_translate("MainWindow", "CheckBox", None))
|
||
self.radioButton.setText(_translate("MainWindow", "RadioButton", None))
|
||
|
||
def on_checkBox_clicked(self):
|
||
if self.checkBox.checkState()==0:
|
||
App.Console.PrintMessage(str(self.checkBox.checkState())+" CheckBox KO\r\n")
|
||
else:
|
||
App.Console.PrintMessage(str(self.checkBox.checkState())+" CheckBox OK\r\n")
|
||
# App.Console.PrintMessage(str(self.lineEdit.setText("tititi"))+" LineEdit\r\n") # write text to the lineEdit window !
|
||
# str(self.lineEdit.setText("tititi")) #écrit le texte dans la fenêtre lineEdit
|
||
App.Console.PrintMessage(str(self.lineEdit.displayText())+" LineEdit\r\n")
|
||
|
||
def on_radioButton_clicked(self):
|
||
if self.radioButton.isChecked():
|
||
App.Console.PrintMessage(str(self.radioButton.isChecked())+" Radio OK\r\n")
|
||
else:
|
||
App.Console.PrintMessage(str(self.radioButton.isChecked())+" Radio KO\r\n")
|
||
|
||
def on_lineEdit_clicked(self):
|
||
# if self.lineEdit.textChanged():
|
||
App.Console.PrintMessage(str(self.lineEdit.displayText())+" LineEdit Display\r\n")
|
||
|
||
def on_pushButton_clicked(self):
|
||
App.Console.PrintMessage("Terminé\r\n")
|
||
self.window.hide()
|
||
|
||
MainWindow = QtGui.QMainWindow()
|
||
ui = Ui_MainWindow(MainWindow)
|
||
MainWindow.show()</pre></div>
|
||
<p>Ici le code pour afficher l'icône sur le <b>pushButton</b>, modifiez le nom pour un autre bouton, (<b>radioButton, checkBox</b>) ainsi que le chemin de l'icône.
|
||
</p>
|
||
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre> # Affiche un icône sur le bouton PushButton
|
||
# self.image_01 = "C:\Program Files\FreeCAD0.13\icone01.png" # the name of the icon
|
||
self.image_01 = path+"icone01.png" # the name of the icon
|
||
icon01 = QtGui.QIcon()
|
||
icon01.addPixmap(QtGui.QPixmap(self.image_01),QtGui.QIcon.Normal, QtGui.QIcon.Off)
|
||
self.pushButton.setIcon(icon01)
|
||
self.pushButton.setLayoutDirection(QtCore.Qt.RightToLeft) # This command reverses the direction of the button</pre></div>
|
||
<p>La commande
|
||
<b>UserAppData</b> donne le chemin utilisateur
|
||
<b>AppHomePath</b> donne le chemin d'installation de FreeCAD
|
||
</p>
|
||
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre># path = FreeCAD.ConfigGet("UserAppData")
|
||
path = FreeCAD.ConfigGet("AppHomePath")</pre></div>
|
||
<p>Cette commande inverse le sens horizontal du bouton, droite à gauche
|
||
</p>
|
||
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre>self.pushButton.setLayoutDirection(QtCore.Qt.RightToLeft) # This command reverses the direction of the button</pre></div>
|
||
<h3><span class="mw-headline" id="M.C3.A9thode_2">Méthode 2</span></h3>
|
||
<p>Une autre méthode pour afficher une fenêtre, ici en créant un fichier <b>QtForm.py</b> qui renferme l'entête du programme (module appelé avec <b>import QtForm</b>), et d'un deuxième module qui renferme le code de la fenêtre tous ces accessoires, et votre code (le module appelant).
|
||
</p><p>Cette méthode nécessite 2 fichiers distincts, mais permet de raccourcir votre programme, en utilisant le fichier <b>QtForm.py</b> en import. Il faut alors distribuer les deux fichiers ensemble, ils sont indissociables.
|
||
</p><p>Le fichier <b>QtForm.py</b>
|
||
</p>
|
||
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre># -*- coding: utf-8 -*-
|
||
# Create by flachyjoe
|
||
from PySide import QtCore, QtGui
|
||
|
||
try:
|
||
_fromUtf8 = QtCore.QString.fromUtf8
|
||
except AttributeError:
|
||
def _fromUtf8(s):
|
||
return s
|
||
|
||
try:
|
||
_encoding = QtGui.QApplication.UnicodeUTF8
|
||
def _translate(context, text, disambig):
|
||
return QtGui.QApplication.translate(context, text, disambig, _encoding)
|
||
except AttributeError:
|
||
def _translate(context, text, disambig):
|
||
return QtGui.QApplication.translate(context, text, disambig)
|
||
|
||
class Form(object):
|
||
def __init__(self, title, width, height):
|
||
self.window = QtGui.QMainWindow()
|
||
self.title=title
|
||
self.window.setObjectName(_fromUtf8(title))
|
||
self.window.setWindowTitle(_translate(self.title, self.title, None))
|
||
self.window.resize(width, height)
|
||
|
||
def show(self):
|
||
self.createUI()
|
||
self.retranslateUI()
|
||
self.window.show()
|
||
|
||
def setText(self, control, text):
|
||
control.setText(_translate(self.title, text, None))</pre></div>
|
||
<p>Le fichier appelant, qui contient la fenêtre et votre code.
|
||
</p><p>Le fichier <b>mon_fichier.py</b>
|
||
</p><p>Les connections sont à faire, un bon exercice.
|
||
</p>
|
||
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre># -*- coding: utf-8 -*-
|
||
# Create by flachyjoe
|
||
from PySide import QtCore, QtGui
|
||
import QtForm
|
||
|
||
class myForm(QtForm.Form):
|
||
def createUI(self):
|
||
self.centralWidget = QtGui.QWidget(self.window)
|
||
self.window.setCentralWidget(self.centralWidget)
|
||
|
||
self.pushButton = QtGui.QPushButton(self.centralWidget)
|
||
self.pushButton.setGeometry(QtCore.QRect(30, 170, 93, 28))
|
||
self.pushButton.clicked.connect(self.on_pushButton_clicked)
|
||
|
||
self.lineEdit = QtGui.QLineEdit(self.centralWidget)
|
||
self.lineEdit.setGeometry(QtCore.QRect(30, 40, 211, 22))
|
||
|
||
self.checkBox = QtGui.QCheckBox(self.centralWidget)
|
||
self.checkBox.setGeometry(QtCore.QRect(30, 90, 81, 20))
|
||
self.checkBox.setChecked(True)
|
||
|
||
self.radioButton = QtGui.QRadioButton(self.centralWidget)
|
||
self.radioButton.setGeometry(QtCore.QRect(30, 130, 95, 20))
|
||
|
||
def retranslateUI(self):
|
||
self.setText(self.pushButton, "Fermer")
|
||
self.setText(self.lineEdit, "essai de texte")
|
||
self.setText(self.checkBox, "CheckBox")
|
||
self.setText(self.radioButton, "RadioButton")
|
||
|
||
def on_pushButton_clicked(self):
|
||
self.window.hide()
|
||
|
||
myWindow=myForm("Fenetre de test",400,300)
|
||
myWindow.show()</pre></div>
|
||
<p><b>Autre exemple</b>
|
||
</p>
|
||
<center>
|
||
<ul class="gallery mw-gallery-traditional">
|
||
<li class="gallerybox" style="width: 435px"><div style="width: 435px">
|
||
<div class="thumb" style="width: 430px;"><div style="margin:15px auto;"><a href="https://www.freecadweb.org/wiki/index.php?title=File:Qt_Example_00.png" class="image"><img alt="" src="167px-Qt_Example_00.png" width="167" height="200" srcset="/wiki/images/thumb/f/fe/Qt_Example_00.png/250px-Qt_Example_00.png 1.5x, /wiki/images/thumb/f/fe/Qt_Example_00.png/333px-Qt_Example_00.png 2x" /></a></div></div>
|
||
<div class="gallerytext">
|
||
<p>Qt example 1
|
||
</p>
|
||
</div>
|
||
</div></li>
|
||
<li class="gallerybox" style="width: 435px"><div style="width: 435px">
|
||
<div class="thumb" style="width: 430px;"><div style="margin:15px auto;"><a href="https://www.freecadweb.org/wiki/index.php?title=File:Qt_Example_01.png" class="image"><img alt="" src="167px-Qt_Example_01.png" width="167" height="200" srcset="/wiki/images/thumb/d/d4/Qt_Example_01.png/250px-Qt_Example_01.png 1.5x, /wiki/images/thumb/d/d4/Qt_Example_01.png/333px-Qt_Example_01.png 2x" /></a></div></div>
|
||
<div class="gallerytext">
|
||
<p>Qt example details
|
||
</p>
|
||
</div>
|
||
</div></li>
|
||
</ul>
|
||
</center>
|
||
<div style="clear:both"></div>
|
||
<p><br />
|
||
Sont traités :
|
||
</p>
|
||
<ol><li> <b>icon for window</b> : l'icône affiché sur le coin supérieur gauche de la fenêtre principale</li>
|
||
<li> <b>horizontalSlider</b> : horizontal slider sa connexion et extraction / affectation de données</li>
|
||
<li> <b>progressBar horizontal</b> : progress bar horizontal sa connexion et extraction / affectation de données</li>
|
||
<li> <b>verticalSlider</b> : vertical slider sa connexion et extraction / affectation de données</li>
|
||
<li> <b>progressBar vertical</b> : progress bar verticale sa connexion et extraction / affectation de données</li>
|
||
<li> <b>lineEdit</b> : line edit sa connexion et extraction / affectation de données</li>
|
||
<li> <b>lineEdit</b> : </li>
|
||
<li> <b>doubleSpinBox</b> : double spinbox sa connexion et extraction / affectation de données</li>
|
||
<li> <b>doubleSpinBox</b> : </li>
|
||
<li> <b>doubleSpinBox</b> :</li>
|
||
<li> <b>buttom</b> : buttom et sa connexion</li>
|
||
<li> <b>buttom</b> :</li>
|
||
<li> <b>radioButtom</b> : radio button avec icône sa connexion checked</li>
|
||
<li> <b>checkBox</b> : checkbox with avec icône sa connexion checked and unchecked</li>
|
||
<li> <b>textEdit</b> : text edit sa connexion et extraction / affectation de données</li>
|
||
<li> <b>graphicsView</b> : graphic view avec 2 images et la méthode pour changer d'image</li></ol>
|
||
<p>La page de code et les icônes <a href="https://www.freecadweb.org/wiki/index.php?title=Qt_Example/fr" title="Qt Example/fr">Qt_Example</a>
|
||
</p>
|
||
<h2><span class="mw-headline" id="Ic.C3.B4nes_personnalis.C3.A9s_dans_la_Vue_combin.C3.A9e">Icônes personnalisés dans la Vue combinée</span></h2>
|
||
<p>Ici un exemple de création d'un objet avec quelques propriétés associé à un icône personnel.
|
||
</p><p>Téléchargez l'icône exemple et placez le dans votre répertoire de macro <a href="https://www.freecadweb.org/wiki/index.php?title=File:FreeCADIco.png" class="image" title="icône Exemple pour la macro"><img alt="icône Exemple pour la macro" src="24px-FreeCADIco.png" width="24" height="22" srcset="/wiki/images/thumb/8/86/FreeCADIco.png/36px-FreeCADIco.png 1.5x, /wiki/images/thumb/8/86/FreeCADIco.png/48px-FreeCADIco.png 2x" /></a>.
|
||
</p><p>Trois méthodes d'affectation d'un icône à un objet sont traitées, un icône fichier au format .png présent sur le disque, un icône sauvé au format .xpm inclus dans la macro elle même et un icône disponible dans les ressources FreeCAD.
|
||
</p><p><a href="https://www.freecadweb.org/wiki/index.php?title=File:Qt_Example_02.png" class="image" title="icon personalized"><img alt="icon personalized" src="Qt_Example_02.png" width="338" height="490" /></a>
|
||
</p>
|
||
<div style="clear:both"></div>
|
||
<p><br />
|
||
</p>
|
||
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre>import PySide
|
||
import FreeCAD, FreeCADGui, Part
|
||
from pivy import coin
|
||
from PySide import QtGui ,QtCore
|
||
from PySide.QtGui import *
|
||
from PySide.QtCore import *
|
||
import Draft
|
||
|
||
global path
|
||
param = FreeCAD.ParamGet("User parameter:BaseApp/Preferences/Macro")# macro path in FreeCAD preferences
|
||
path = param.GetString("MacroPath","") + "/" # macro path
|
||
path = path.replace("\\","/") # convert the "\" to "/"
|
||
|
||
|
||
class IconViewProviderToFile: # Class ViewProvider create Property view of object
|
||
def __init__( self, obj, icon):
|
||
self.icone = icon
|
||
|
||
def getIcon(self): # GetIcon
|
||
return self.icone
|
||
|
||
def attach(self, obj): # Property view of object
|
||
self.modes = []
|
||
self.modes.append("Flat Lines")
|
||
self.modes.append("Shaded")
|
||
self.modes.append("Wireframe")
|
||
self.modes.append("Points")
|
||
obj.addDisplayMode( coin.SoGroup(),"Flat Lines" ) # Display Mode
|
||
obj.addDisplayMode( coin.SoGroup(),"Shaded" )
|
||
obj.addDisplayMode( coin.SoGroup(),"Wireframe" )
|
||
obj.addDisplayMode( coin.SoGroup(),"Points" )
|
||
return self.modes
|
||
|
||
def getDisplayModes(self,obj):
|
||
return self.modes
|
||
|
||
#####################################################
|
||
########## Example with icon to file # begin ########
|
||
#####################################################
|
||
|
||
object1 = FreeCAD.ActiveDocument.addObject("App::FeaturePython", "Icon_In_File_Disk") # create your object
|
||
object1.addProperty("App::PropertyString","Identity", "ExampleTitle0", "Identity of object").Identity = "FCSpring" # Identity of object
|
||
object1.addProperty("App::PropertyFloat" ,"Pitch", "ExampleTitle0", "Pitch betwen 2 heads").Pitch = 2.0 # other Property Data
|
||
object1.addProperty("App::PropertyBool" ,"View", "ExampleTitle1", "Hello world").View = True # ...
|
||
object1.addProperty("App::PropertyColor" ,"LineColor","ExampleTitle2", "Color to choice").LineColor = (0.13,0.15,0.37) # ...
|
||
#...other Property Data
|
||
#...other Property Data
|
||
#
|
||
object1.ViewObject.Proxy = IconViewProviderToFile( object1, path + "FreeCADIco.png") # icon download to file
|
||
App.ActiveDocument.recompute()
|
||
#
|
||
#__Detail__:
|
||
# FreeCAD.ActiveDocument.addObject( = create now object personalized
|
||
# "App::FeaturePython", = object as FeaturePython
|
||
# "Icon_In_File_Disk") = internal name of your object
|
||
#
|
||
#
|
||
# "App::PropertyString", = type of Property , availlable : PropertyString, PropertyFloat, PropertyBool, PropertyColor
|
||
# "Identity", = name of the feature
|
||
# "ExampleTitle0", = title of the "section"
|
||
# "Identity of object") = tooltip displayed on mouse
|
||
# .Identity = variable (same of name of the feature)
|
||
# object1.ViewObject.Proxy = create the view object and gives the icon
|
||
#
|
||
########## example with icon to file end
|
||
|
||
|
||
|
||
#####################################################
|
||
########## Example with icon in macro # begin #######
|
||
#####################################################
|
||
|
||
def setIconInMacro(self): # def contener the icon in format .xpm
|
||
# File format XPM created by Gimp "https://www.gimp.org/"
|
||
# Choice palette Tango
|
||
# Create your masterwork ...
|
||
# For export the image in XPM format
|
||
# Menu File > Export as > .xpm
|
||
# (For convert image true color in Tango color palette :
|
||
# Menu Image > Mode > Indexed ... > Use custom palette > Tango Icon Theme > Convert)
|
||
return """
|
||
/* XPM */
|
||
static char * XPM[] = {
|
||
"22 24 5 1",
|
||
" c None",
|
||
".c #CE5C00",
|
||
"+c #EDD400",
|
||
"@c #F57900",
|
||
"#c #8F5902",
|
||
" ",
|
||
" ",
|
||
" .... ",
|
||
" ..@@@@.. ",
|
||
" . ...@...... ",
|
||
" .+++++++++... ",
|
||
" . ....++... ",
|
||
" .@..@@@@@@.+++++.. ",
|
||
" .@@@@@..# ++++ .. ",
|
||
" . ++++ .@.. ",
|
||
" .++++++++ .@@@.+. ",
|
||
" . ..@@@@@. ++. ",
|
||
" ..@@@@@@@@@. +++ . ",
|
||
" ....@...# +++++ @.. ",
|
||
" . ++++++++ .@. . ",
|
||
" .++++++++ .@@@@ . ",
|
||
" . #....@@@@. ++. ",
|
||
" .@@@@@@@@@.. +++ . ",
|
||
" ........ +++++... ",
|
||
" ... ..+++++ ..@.. ",
|
||
" ...... .@@@ +. ",
|
||
" ......++. ",
|
||
" ... ",
|
||
" "};
|
||
"""
|
||
|
||
object2 = FreeCAD.ActiveDocument.addObject("App::FeaturePython", "Icon_XPM_In_Macro") #
|
||
object2.addProperty("App::PropertyString","Identity","ExampleTitle","Identity of object").Identity = "FCSpring"
|
||
#...other Property Data
|
||
#...other Property Data
|
||
#
|
||
object2.ViewObject.Proxy = IconViewProviderToFile( object2, setIconInMacro("")) # icon in macro (.XPM)
|
||
App.ActiveDocument.recompute()
|
||
########## example with icon in macro end
|
||
|
||
|
||
|
||
####################################################################
|
||
########## Example with icon to FreeCAD ressource # begin ##########
|
||
####################################################################
|
||
|
||
object3 = FreeCAD.ActiveDocument.addObject("App::FeaturePython", "Icon_Ressource_FreeCAD") #
|
||
object3.addProperty("App::PropertyString","Identity","ExampleTitle","Identity of object").Identity = "FCSpring"
|
||
#...other Property Data
|
||
#...other Property Data
|
||
#
|
||
object3.ViewObject.Proxy = IconViewProviderToFile( object3, ":/icons/Draft_Draft.svg") # icon to FreeCAD ressource
|
||
App.ActiveDocument.recompute()
|
||
########## example with icon to FreeCAD ressource end</pre></div>
|
||
<p>Exemple complet de création d'un cube avec son icône
|
||
</p>
|
||
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre>#https://forum.freecadweb.org/viewtopic.php?t=10255#p83319
|
||
import FreeCAD, Part, math
|
||
from FreeCAD import Base
|
||
from PySide import QtGui
|
||
|
||
global path
|
||
param = FreeCAD.ParamGet("User parameter:BaseApp/Preferences/Macro")# macro path in FreeCAD preferences
|
||
path = param.GetString("MacroPath","") + "/" # macro path
|
||
path = path.replace("\\","/") # convert the "\" to "/"
|
||
|
||
def setIconInMacro(self):
|
||
return """
|
||
/* XPM */
|
||
static char * xpm[] = {
|
||
"22 22 12 1",
|
||
" c None",
|
||
".c #A40000",
|
||
"+c #2E3436",
|
||
"@c #CE5C00",
|
||
"#c #F57900",
|
||
"$c #FCAF3E",
|
||
"%c #5C3566",
|
||
"&c #204A87",
|
||
"*c #555753",
|
||
"=c #3465A4",
|
||
"-c #4E9A06",
|
||
";c #729FCF",
|
||
" ",
|
||
" ",
|
||
" ",
|
||
" .. .. ",
|
||
" +@#+++.$$ ",
|
||
" +.#+%..$$ ",
|
||
" &*$ &*#* ",
|
||
" & =&= = ",
|
||
" ++& +.== %= ",
|
||
" ++$@ ..$ %= & ",
|
||
" ..-&%.#$$ &## +=$ ",
|
||
" .# ..$ ..#%%.#$$ ",
|
||
" ; =+=## %-$# ",
|
||
" &= ;& %= ",
|
||
" ;+ &=; %= ",
|
||
" ++$- +*$- ",
|
||
" .#&&+.@$$ ",
|
||
" ..$# ..$# ",
|
||
" .. .. ",
|
||
" ",
|
||
" ",
|
||
" "};
|
||
"""
|
||
|
||
class PartFeature:
|
||
def __init__(self, obj):
|
||
obj.Proxy = self
|
||
|
||
class Box(PartFeature):
|
||
def __init__(self, obj):
|
||
PartFeature.__init__(self, obj)
|
||
obj.addProperty("App::PropertyLength", "Length", "Box", "Length of the box").Length = 1.0
|
||
obj.addProperty("App::PropertyLength", "Width", "Box", "Width of the box" ).Width = 1.0
|
||
obj.addProperty("App::PropertyLength", "Height", "Box", "Height of the box").Height = 1.0
|
||
|
||
def onChanged(self, fp, prop):
|
||
try:
|
||
if prop == "Length" or prop == "Width" or prop == "Height":
|
||
fp.Shape = Part.makeBox(fp.Length,fp.Width,fp.Height)
|
||
except:
|
||
pass
|
||
|
||
def execute(self, fp):
|
||
fp.Shape = Part.makeBox(fp.Length,fp.Width,fp.Height)
|
||
|
||
class ViewProviderBox:
|
||
def __init__(self, obj, icon):
|
||
obj.Proxy = self
|
||
self.icone = icon
|
||
|
||
def getIcon(self):
|
||
return self.icone
|
||
|
||
def attach(self, obj):
|
||
return
|
||
|
||
def setupContextMenu(self, obj, menu):
|
||
action = menu.addAction("Set default height")
|
||
action.triggered.connect(lambda f=self.setDefaultHeight, arg=obj:f(arg))
|
||
|
||
action = menu.addAction("Hello World")
|
||
action.triggered.connect(self.showHelloWorld)
|
||
|
||
def setDefaultHeight(self, view):
|
||
view.Object.Height = 15.0
|
||
|
||
def showHelloWorld(self):
|
||
QtGui.QMessageBox.information(None, "Hi there", "Hello World")
|
||
|
||
def makeBox():
|
||
FreeCAD.newDocument()
|
||
a=FreeCAD.ActiveDocument.addObject("Part::FeaturePython","Box")
|
||
Box(a)
|
||
# ViewProviderBox(a.ViewObject, path + "FreeCADIco.png") # icon download to file
|
||
# ViewProviderBox(a.ViewObject, ":/icons/Draft_Draft.svg") # icon to FreeCAD ressource
|
||
ViewProviderBox(a.ViewObject, setIconInMacro("")) # icon in macro (.XPM)
|
||
App.ActiveDocument.recompute()
|
||
|
||
makeBox()</pre></div>
|
||
<h2><span class="mw-headline" id="Utiliser_QFileDialog_for_.C3.A9crire_dans_un_fichier">Utiliser QFileDialog for écrire dans un fichier</span></h2>
|
||
<p>Code complet :
|
||
</p>
|
||
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre># -*- coding: utf-8 -*-
|
||
import PySide
|
||
from PySide import QtGui ,QtCore
|
||
from PySide.QtGui import *
|
||
from PySide.QtCore import *
|
||
path = FreeCAD.ConfigGet("UserAppData")
|
||
|
||
try:
|
||
SaveName = QFileDialog.getSaveFileName(None,QString.fromLocal8Bit("Save a file txt"),path, "*.txt") # PyQt4
|
||
# "here the text displayed on windows" "here the filter (extension)"
|
||
except Exception:
|
||
SaveName, Filter = PySide.QtGui.QFileDialog.getSaveFileName(None, "Save a file txt", path, "*.txt") # PySide
|
||
# "here the text displayed on windows" "here the filter (extension)"
|
||
if SaveName == "": # if the name file are not selected then Abord process
|
||
App.Console.PrintMessage("Process aborted"+"\n")
|
||
else: # if the name file are selected or created then
|
||
App.Console.PrintMessage("Registration of "+SaveName+"\n") # text displayed to Report view (Menu > View > Report view checked)
|
||
try: # detect error ...
|
||
file = open(SaveName, 'w') # open the file selected to write (w)
|
||
try: # if error detected to write ...
|
||
# here your code
|
||
print "here your code"
|
||
file.write(str(1)+"\n") # write the number convert in text with (str())
|
||
file.write("FreeCAD the best") # write the the text with (" ")
|
||
except Exception: # if error detected to write
|
||
App.Console.PrintError("Error write file "+"\n") # detect error ... display the text in red (PrintError)
|
||
finally: # if error detected to write ... or not the file is closed
|
||
file.close() # if error detected to write ... or not the file is closed
|
||
except Exception:
|
||
App.Console.PrintError("Error Open file "+SaveName+"\n") # detect error ... display the text in red (PrintError)</pre></div>
|
||
<h2><span class="mw-headline" id="Utiliser_QFileDialog_pour_lire_un_fichier">Utiliser QFileDialog pour lire un fichier</span></h2>
|
||
<p>Code complet:
|
||
</p>
|
||
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre># -*- coding: utf-8 -*-
|
||
import PySide
|
||
from PySide import QtGui ,QtCore
|
||
from PySide.QtGui import *
|
||
from PySide.QtCore import *
|
||
path = FreeCAD.ConfigGet("UserAppData")
|
||
|
||
OpenName = ""
|
||
try:
|
||
OpenName = QFileDialog.getOpenFileName(None,QString.fromLocal8Bit("Read a file txt"),path, "*.txt") # PyQt4
|
||
# "here the text displayed on windows" "here the filter (extension)"
|
||
except Exception:
|
||
OpenName, Filter = PySide.QtGui.QFileDialog.getOpenFileName(None, "Read a file txt", path, "*.txt") #PySide
|
||
# "here the text displayed on windows" "here the filter (extension)"
|
||
if OpenName == "": # if the name file are not selected then Abord process
|
||
App.Console.PrintMessage("Process aborted"+"\n")
|
||
else:
|
||
App.Console.PrintMessage("Read "+OpenName+"\n") # text displayed to Report view (Menu > View > Report view checked)
|
||
try: # detect error to read file
|
||
file = open(OpenName, "r") # open the file selected to read (r) # (rb is binary)
|
||
try: # detect error ...
|
||
# here your code
|
||
print "here your code"
|
||
op = OpenName.split("/") # decode the path
|
||
op2 = op[-1].split(".") # decode the file name
|
||
nomF = op2[0] # the file name are isolated
|
||
|
||
App.Console.PrintMessage(str(nomF)+"\n") # the file name are displayed
|
||
|
||
for ligne in file: # read the file
|
||
X = ligne.rstrip('\n\r') #.split() # decode the line
|
||
print X # print the line in report view other method
|
||
# (Menu > Edit > preferences... > Output window > Redirect internal Python output (and errors) to report view checked)
|
||
except Exception: # if error detected to read
|
||
App.Console.PrintError("Error read file "+"\n") # detect error ... display the text in red (PrintError)
|
||
finally: # if error detected to read ... or not error the file is closed
|
||
file.close() # if error detected to read ... or not error the file is closed
|
||
except Exception: # if one error detected to read file
|
||
App.Console.PrintError("Error in Open the file "+OpenName+"\n") # if one error detected ... display the text in red (PrintError)</pre></div>
|
||
<h2><span class="mw-headline" id="Utiliser_QColorDialog_pour_utiliser_une_couleur">Utiliser QColorDialog pour utiliser une couleur</span></h2>
|
||
<p>Code complet:
|
||
</p>
|
||
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre># -*- coding: utf-8 -*-
|
||
# https://deptinfo-ensip.univ-poitiers.fr/ENS/pyside-docs/PySide/QtGui/QColor.html
|
||
import PySide
|
||
from PySide import QtGui ,QtCore
|
||
from PySide.QtGui import *
|
||
from PySide.QtCore import *
|
||
path = FreeCAD.ConfigGet("UserAppData")
|
||
|
||
couleur = QtGui.QColorDialog.getColor()
|
||
if couleur.isValid():
|
||
red = int(str(couleur.name()[1:3]),16) # decode hexadecimal to int()
|
||
green = int(str(couleur.name()[3:5]),16) # decode hexadecimal to int()
|
||
blue = int(str(couleur.name()[5:7]),16) # decode hexadecimal to int()
|
||
|
||
print couleur #
|
||
print "hexadecimal ",couleur.name() # color format hexadecimal mode 16
|
||
print "Red color ",red # color format decimal
|
||
print "Green color ",green # color format decimal
|
||
print "Blue color ",blue # color format decimal</pre></div>
|
||
<h2><span class="mw-headline" id="Quelques_commandes_utiles">Quelques commandes utiles</span></h2>
|
||
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre># Here the code to display the icon on the '''pushButton''',
|
||
# change the name to another button, ('''radioButton, checkBox''') as well as the path to the icon,
|
||
|
||
# Displays an icon on the button PushButton
|
||
# self.image_01 = "C:\Program Files\FreeCAD0.13\icone01.png" # he name of the icon
|
||
self.image_01 = path+"icone01.png" # the name of the icon
|
||
icon01 = QtGui.QIcon()
|
||
icon01.addPixmap(QtGui.QPixmap(self.image_01),QtGui.QIcon.Normal, QtGui.QIcon.Off)
|
||
self.pushButton.setIcon(icon01)
|
||
self.pushButton.setLayoutDirection(QtCore.Qt.RightToLeft) # This command reverses the direction of the button
|
||
|
||
|
||
# path = FreeCAD.ConfigGet("UserAppData") # gives the user path
|
||
path = FreeCAD.ConfigGet("AppHomePath") # gives the installation path of FreeCAD
|
||
|
||
# This command reverses the horizontal button, right to left
|
||
self.pushButton.setLayoutDirection(QtCore.Qt.RightToLeft) # This command reverses the horizontal button
|
||
|
||
# Displays an info button
|
||
self.pushButton.setToolTip(_translate("MainWindow", "Quitter la fonction", None)) # Displays an info button
|
||
|
||
# This function gives a color button
|
||
self.pushButton.setStyleSheet("background-color: red") # This function gives a color button
|
||
|
||
# This function gives a color to the text of the button
|
||
self.pushButton.setStyleSheet("color : #ff0000") # This function gives a color to the text of the button
|
||
|
||
# combinaison des deux, bouton et texte
|
||
self.pushButton.setStyleSheet("color : #ff0000; background-color : #0000ff;" ) # combination of the two, button, and text
|
||
|
||
# replace the icon in the main window
|
||
MainWindow.setWindowIcon(QtGui.QIcon('C:\Program Files\FreeCAD0.13\View-C3P.png'))
|
||
|
||
# connects a lineEdit on execute
|
||
self.lineEdit.returnPressed.connect(self.execute) # connects a lineEdit on "def execute" after validation on enter
|
||
# self.lineEdit.textChanged.connect(self.execute) # connects a lineEdit on "def execute" with each keystroke on the keyboard
|
||
|
||
# display text in a lineEdit
|
||
self.lineEdit.setText(str(val_X)) # Displays the value in the lineEdit (convert to string)
|
||
|
||
# extract the string contained in a lineEdit
|
||
val_X = self.lineEdit.text() # extract the (string) string contained in lineEdit
|
||
val_X = float(val_X0) # converted the string to an floating
|
||
val_X = int(val_X0) # convert the string to an integer
|
||
|
||
# This code allows you to change the font and its attributes
|
||
font = QtGui.QFont()
|
||
font.setFamily("Times New Roman")
|
||
font.setPointSize(10)
|
||
font.setWeight(10)
|
||
font.setBold(True) # same result with tags "<b>your text</b>" (in quotes)
|
||
self.label_6.setFont(font)
|
||
self.label_6.setObjectName("label_6")
|
||
self.label_6.setStyleSheet("color : #ff0000") # This function gives a color to the text
|
||
self.label_6.setText(_translate("MainWindow", "Select a view", None))</pre></div>
|
||
<p>En utilisant les caractères accentués, dans le cas ou vous obtenez les erreurs suivantes :
|
||
</p><p>plusieurs méthodes sont possibles.
|
||
</p><p><font color="#FF0000"><b>UnicodeDecodeError: 'utf8' codec can't decode bytes in position 0-2: invalid data</b></font>
|
||
</p>
|
||
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre># conversion from a lineEdit
|
||
App.activeDocument().CopyRight.Text = str(unicode(self.lineEdit_20.text() , 'ISO-8859-1').encode('UTF-8'))
|
||
DESIGNED_BY = unicode(self.lineEdit_01.text(), 'ISO-8859-1').encode('UTF-8')</pre></div>
|
||
<p>ou avec la procédure
|
||
</p>
|
||
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre>def utf8(unio):
|
||
return unicode(unio).encode('UTF8')</pre></div>
|
||
<p><font color="#FF0000"><b>UnicodeEncodeError: 'ascii' codec can't encode character u'\xe9' in position 9: ordinal not in range(128)</b></font>
|
||
</p>
|
||
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre># conversion
|
||
a = u"Nom de l'élément : "
|
||
f.write('''a.encode('iso-8859-1')'''+str(element_)+"\n")</pre></div>
|
||
<p>ou avec la procédure
|
||
</p>
|
||
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre>def iso8859(encoder):
|
||
return unicode(encoder).encode('iso-8859-1')</pre></div>
|
||
<p>ou
|
||
</p>
|
||
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre>iso8859(unichr(176))</pre></div>
|
||
<p>ou
|
||
</p>
|
||
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre>unichr(ord(176))</pre></div>
|
||
<p>ou
|
||
</p>
|
||
<div class="mw-highlight mw-content-ltr" dir="ltr"><pre>uniteSs = "mm"+iso8859(unichr(178))
|
||
print unicode(uniteSs, 'iso8859')</pre></div>
|
||
<p><br />
|
||
</p>
|
||
|
||
<div style="clear:both"></div>
|
||
</div>
|
||
|
||
|
||
|
||
</div>
|
||
|
||
</div><div class="printfooter">
|
||
Online version: "<a dir="ltr" href="https://www.freecadweb.org/wiki/index.php?title=Dialog_creation/fr&oldid=245367">http://www.freecadweb.org/wiki/index.php?title=Dialog_creation/fr&oldid=245367</a>"</div>
|
||
<div id="catlinks" class="catlinks" data-mw="interface"></div><div class="visualClear"></div>
|
||
</div>
|
||
</div>
|
||
<div id="mw-navigation">
|
||
<h2>Navigation menu</h2>
|
||
|
||
</body></html> |