Panneau virtuel de contrôle en python (*Py*thon *V*irtual *C*ontrol *P*anel)
1. Introduction
Le panneau de contrôle virtuel pyVCP (python Virtual Control Panel) a été créé pour donner à l’intégrateur la possibilité de personnaliser l’interface graphique AXIS avec des boutons et des indicateurs destinés aux tâches spéciales.
Le coût d’un panneau de contrôle physique est très élevé et il peut utiliser un grand nombre de broches d’entrées/sorties. C’est là que le panneau virtuel prends l’avantage car il ne coûte rien d’utiliser pyVCP.
Les panneaux virtuels de contrôle peuvent être utilisés pour tester ou monitorer le matériel, les entrées/sorties et remplacer temporairement d’autres matériels d’entrées/sorties pendant le déboguage d’une logique ladder, ou pour simuler un panneau physique avant de le construire et de le câbler vers les cartes électroniques.
L’image suivante montre plusieurs widgets pyVCP.
2. pyVCP
La disposition d’un panneau pyVCP est spécifiée avec un fichier XML qui contient les balises des widgets entre <pyvcp> et </pyvcp>. Par exemple:
<pyvcp>
Si vous placez ce texte dans un fichier nommé tiny.xml et que vous le lancez avec:
pyvcp -c panneau tiny.xml
pyVCP va créer le panneau pour vous, il y inclut deux widgets, un Label avec le texte “Ceci est un indicateur à LED” et une LED rouge, utilisée pour afficher l'état d’un signal HAL de type BIT. Il va aussi créer un composant HAL nommé “panneau” (tous les widgets dans ce panneau sont connectés aux pins qui démarrent avec “panneau”). Comme aucune balise <halpin> n'était présente à l’intérieur de la balise <led>, pyVCP nomme automatiquement la pin HAL pour le widget LED panneau.led.0
Pour obtenir la liste des widgets, leurs balises et options, consultez la documentation des widgets: [sec:Documentation-des-widgets]
Un fois que vous avez créé votre panneau, connecter les signaux HAL de la forme à la pin pyVCP se fait avec la commande halcmd linksp habituelle. Si vous débutez avec HAL, le tutoriel de HAL [cha:Tutoriel-HAL] est vivement recommendé.
3. Sécurité avec pyVCP
Certaines parties de pyVCP sont évaluées comme du code Python, elles peuvent donc exécuter n’importe quelle action disponible dans les programmes Python. N’utilisez que des fichiers pyVCP en .xml à partir d’une source de confiance.
4. Utiliser pyVCP avec AXIS
Puisque AXIS utilise le même environnement graphique et les même outils (Tkinter) que pyVCP, il est possible d’inclure un panneau pyVCP sur le côté droit de l’interface utilisateur normale d’AXIS. Un exemple typique est présenté ci-dessous.
Placer le fichier pyVCP XML décrivant le panneau dans le même répertoire que le fichier .ini. Nous voulons afficher la vitesse courante de la broche sur un widget barre de progression. Copier le code XML suivant dans un fichier appelé broche.xml:
<pyvcp>
Ici nous avons fait un panneau avec un label “Vitesse broche:” et un widget barre de progression. Nous avons spécifié que la pin HAL connectée à la barre de progression devait s’appeler “spindle-speed” et réglé la valeur maximum de la barre à 5000 (se reporter à la documentation des widgets, plus loin, pour toutes les options disponibles). Pour faire connaître ce fichier à AXIS et qu’il l’appelle au démarrage, nous devons préciser ce qui suit dans la section [DISPLAY] du fichier .ini:
PYVCP = broche.xml
Pour que notre widget affiche réellement la vitesse de la broche “spindle-speed”, il doit être raccordé au signal approprié de HAL. Le fichier .hal qui sera exécuté quand AXIS et pyVCP démarreront doit être spécifié, de la manière suivante, dans la section [HAL] du fichier .ini:
POSTGUI_HALFILE = broche_vers_pyvcp.hal
Ce changement lancera la commande HAL spécifiée dans “broche_vers_pyvcp.hal”. Dans notre exemple, ce fichier contiendra juste la commande suivante:
net spindle-rpm-filtered => pyvcp.spindle-speed
ce qui suppose que le signal appelé “spindle-rpm-filtered” existe aussi. Noter que lors de l’exécution avec AXIS, toutes les pins des widgets de pyVCP ont des noms commençant par “pyvcp.”.
Voila à quoi ressemble le panneau pyVCP que nous venons de créer, incorporé à AXIS. La configuration sim/lathe fournie en exemple, est configurée de cette manière.
5. Documentation des widgets de pyVCP
Les signaux de HAL existent en deux variantes, BIT et FLOAT. pyVCP peut afficher la valeur d’un signal avec un widget indicateur, ou modifier la valeur d’un signal avec un widget de contrôle. Ainsi, il y a quatre classes de widgets pyVCP connectables aux signaux de HAL. Une cinquième classe de widgets d’aide permet d’organiser et d’appliquer des labels aux panneaux.
-
Widgets de signalisation, signaux BIT: led, rectled
-
Widgets de contrôle, signaux BIT: button, checkbutton, radiobutton
-
Widgets de signalisation, signaux FLOAT: number, s32, u32, bar, meter
-
Widgets de contrôle, signaux FLOAT: spinbox, scale, jogwheel
-
Widgets d’aide: hbox, vbox, table, label, labelframe
5.0.1. Syntaxe
Chaque widget est décrit brièvement, suivi par la forme d'écriture utilisée et d’une capture d'écran. Toutes les balises contenues dans la balise du widget principal, sont optionnelles.
5.0.2. Notes générales
à l’heure actuelle, les deux syntaxes, basée sur les balises et basée sur les attributs, sont supportées. Par exemple, les deux fragments de code XML suivants sont traités de manière identique:
<led halpin="ma-led"/>
et
<led><halpin>"ma-led"</halpin></led>
Quand la syntaxe basée sur les attributs est utilisée, les règles suivantes sont utilisées pour convertir les valeurs des attributs en valeurs Python:
-
Si le premier caractère de l’attribut est un des suivants: {(["' , Il est évalué comme une expression Python.
-
Si la chaine est acceptée par int(), la valeur est traitée comme un entier.
-
Si la chaine est acceptée par float(), la valeur est traitée comme un flottant.
-
Autrement, la chaine est acceptée comme une chaine.
Quand la syntaxe basée sur les balises est utilisée, le texte entre les balises est toujours évalué comme un expression Python.
Les exemples ci-dessous montrent un mélange des deux formats.
5.0.3. Commentaires
Pour ajouter un commentaire utiliser la syntaxe de xml.
<!--Mon commentaire-->
5.0.4. Editer un fichier XML
Editer le fichier XML avec un éditeur de texte. La plupart du temps un double click sur le nom de fichier permet de choisir "ouvrir avec l’editeur de texte" ou similaire.
5.0.5. Couleurs
Les couleurs peuvent être spécifiées en utilisant les couleurs RGB de X11 soit par le nom, par exemple: "gray75" ou soit en hexa décimal, par exemple: "#0000ff”. Une liste complète est consultable ici: http://sedition.com/perl/rgb.html.
Couleurs les plus courantes (les numéros suivant la couleur indiquent la nuance de la couleur)
-
white
-
black
-
blue and blue1 - 4
-
cyan and cyan1 - 4
-
green and green1 - 4
-
yellow and yellow1 - 4
-
red and red1 - 4
-
purple and purple1 - 4
-
gray and gray0 - 100
5.1. LED
Une LED est utilisée pour indiquer l'état d’un signal BIT. La couleur de la LED sera on_color quand le signal BIT est vrai et off_color autrement.
<led>
<halpin> définit le nom de la pin, par défaut: “led.n”, où n est un
entier
<size> définit la taille de la led, par défaut: 20
<on_color> définit la couleur de la led LED quand la pin est vraie,
par défaut: “green”
<off_color> définit la couleur de la LED quand la pin est fausse, par
défaut: “ref”
5.2. LED rectangulaire (rectled)
C’est une variante du widget "led".
<vbox> + <relief>RIDGE</relief> + <bd>6</bd> + <rectled> + <halpin>"ma-led-rect"</halpin> + <height>"50"</height> + <width>"100"</width> + <on_color>"green"</on_color> + <off_color>"red"</off_color> + </rectled> + </vbox>
Le code ci-dessus produit cet exemple.
Il affiche également un relief autour de la boîte.
5.3. Bouton (button)
Un bouton permet de contrôler une pin BIT. La pin sera mise vraie quand le bouton sera pressé et maintenu enfoncé, elle sera mise fausse quand le bouton sera relâché.
Les boutons peuvent suivre le format suivant:
-
<padx>n</padx> où "n" est le nombre d’espaces horizontaux supplémentaires
-
<pady>n</pady> où "n" est le nombre d’espaces verticaux supplémentaires
-
<activebackground>"color"</activebackground> Couleur au survol du curseur
-
<bg>"color"</bg> Couleur du bouton
5.3.1. Bouton avec texte (Text Button)
<button> <button>
5.3.2. Case à cocher (checkbutton)
Une case à cocher contrôle une pin BIT. La pin sera mise vraie quand la case sera cochée et fausse si la case est décochée.
<checkbutton>
Une case non cochée: et une case cochée:
5.3.3. Bouton radio (radiobutton)
Un bouton radio placera une seule des pins BIT vraie. Les autres seront mises fausses.
<radiobutton>
Noter que dans cet exemple, les pins de HAL seront nommées mon-bouton-radio.un, mon-bouton-radio.deux et mon-bouton-radio.trois. Dans la capture d'écran, la valeur “trois” est sélectionnée.
5.4. Affichage d’un nombre (number)
L’affichage d’un nombre peux suivre le format suivant:
-
<font>("Font Name",n)</font> où "n" est la taille de la police
-
<width>n</width> où "n" est la largeur totale utilisée
-
<justify>pos</justify> où "pos" peut être LEFT, CENTER ou RIGHT (devrait marcher)
-
<padx>n</padx> où "n" est le nombre d’espaces horizontaux supplémentaires
-
<pady>n</pady> où "n" est le nombre d’espaces verticaux supplémentaires
5.4.1. Number
Le widget number affiche la valeur d’un signal de type float.
<number> + <halpin>"number"</halpin> + <font>("Helvetica",24)</font> + <format>"+4.4f"</format> + </number>
5.4.2. Flottant
Le widget nombre affiche la valeur d’un signal FLOAT.
<number>
<font> est une police de caractères de type Tkinter avec la spécification de sa taille. Noter que sous Ubuntu 6.06 Helvetica n’est pas disponible en taille supérieure à 40 ou 50. Une police qui peut être agrandie jusqu'à la taille 200 est la police courier 10 pitch, que vous pouvez spécifier de la manière suivante, pour afficher des chiffres réellement grands:
<font>('courier 10 pitch',100)</font>
<format> est un format style C, spécifié pour définir le format d’affichage du nombre.
5.4.3. Nombre s32
Le widget s32 affiche la valeur d’un nombre s32. La syntaxe est la même que celle de "number" excepté le nom qui est <s32>. Il faut prévoir une largeur suffisante pour afficher le nombre dans sa totalité.
<s32> + <halpin>"simple-number"</halpin> + <font>("Helvetica",24)</font> + <format>"6d"</format> + <width>6</width> + </s32>
5.4.4. Nombre u32
Le widget u32 affiche la valeur d’un nombre u32. La syntaxe est la même que celle de "number" excepté le nom qui est <u32>.
5.5. Affichage d’images
Seul l’affichage d’images au format gif est possible. Toutes les images doivent avoir la même taille. Les images doivent être toutes dans le même répertoire que le fichier ini (ou dans le répertoire courant pour un fonctionnement en ligne de commande avec halrun/halcmd).
5.5.1. Image Bit
La bascule "image_bit" bascule entre deux images selon la position vraie ou fausse de halpin.
<pyvcp> + <image name='fwd' file='fwd.gif'/> + <image name='rev' file='rev.gif'/> + <vbox> + <image_bit halpin='selectimage' images='fwd rev'/> + </vbox> + </pyvcp>
En utilisant les deux images fwd.gif et rev.gif.
FWD est affiché quand "selectimage" est fausse
et REV est affiché quand "selectimage" est vraie.
5.5.2. Image u32
La bascule "image_u32" est la même que "image_bit" excepté que le nombre d’images n’est pratiquement plus limité, il suffit de "selectionner" l’image en ajustant halpin à une valeur entière commençant à 0 pour la première image de la liste, à 1 pour la seconde image etc.
<pyvcp> + <image name='stb' file='stb.gif'/> + <image name='fwd' file='fwd.gif'/> + <image name='rev' file='rev.gif'/> + <vbox> + <image_u32 halpin='selectimage' images='stb fwd rev'/> + </vbox> + </pyvcp>
Même résultat mais en ajoutant l’image stb.gif.
5.6. Barre de progression (bar)
Le widget barre de progression affiche la valeur d’un signal FLOAT, graphiquement dans une barre de progression et simultanément, en numérique.
<bar>
5.7. Galvanomètre (meter)
Le galvanomètre affiche la valeur d’un signal FLOAT dans un affichage à aiguille “à l’ancienne”.
<meter>
5.8. Roue codeuse (spinbox)
La roue codeuse contrôle une pin FLOAT. La valeur de la pin est augmentée ou diminuée de la valeur de resolution, à chaque pression sur une flèche, ou en positionnant la souris sur le nombre puis en tournant la molette de la souris.
<spinbox>
5.9. Curseur (scale)
Le curseur contrôle une pin FLOAT. La valeur de la pin est augmentée ou diminuée en déplaçant le curseur, ou en positionnant la souris sur le curseur puis en tournant la molette de la souris.
<scale>
Noter que par défaut c’est min qui est affiché même si il est supérieur à max, à moins que min ne soit négatif.
5.10. Bouton tournant (dial)
Le bouton tournant imite le fonctionnement d’un vrai bouton tournant, en sortant sur un FLOAT HAL la valeur sur laquelle est positionné le curseur, que ce soit en le faisant tourner avec un mouvement circulaire, ou en tournant la molette de la souris. Un double click gauche augmente la résolution et un double click droit la diminue d’un digit. La sortie est limitée par les valeurs min et max. La variable cpr fixe le nombre de graduations sur le pourtour du cadran (prudence avec les grands nombres).
<dial> + <size>200</size> + <cpr>100</cpr> + <min_>-15</min_> + <max_>15</max_> + <text>"Dial"</text> + <init>0</init> + <resolution>0.001</resolution> + <halpin>"anaout"</halpin> + <dialcolor>"yellow"</dialcolor> + <edgecolor>"green"</edgecolor> + <dotcolor>"black"</dotcolor> + </dial>
5.11. Bouton tournant (jogwheel)
Le bouton tournant imite le fonctionnement d’un vrai bouton tournant, en sortant sur une pin FLOAT la valeur sur laquelle est positionné le curseur, que ce soit en le faisant tourner avec un mouvement circulaire, ou en tournant la molette de la souris.
<jogwheel>
6. Documentation des containers de pyVCP
Les containers sont des widgets qui contiennent d’autres widgets.
6.1. Bordures
Le container bordure est spécifié avec deux tags utilisés ensembles. Le tag <relief> spécifie le type de bordure et le tag <bd> spécifie la largeur de la bordure.
- <relief>type</relief>
-
La valeur de "type" peut être: FLAT, SUNKEN, RAISED, GROOVE, ou RIDGE
- <bd>n</bd>
-
La valeur de "n" fixe la largeur de la bordure.
<hbox> + <button> + <relief>FLAT</relief> + <text>"FLAT"</text> + <bd>3</bd> + </button>
<button> + <relief>SUNKEN</relief> + <text>"SUNKEN"</text> + <bd>3</bd> + </button>
<button> + <relief>RAISED</relief> + <text>"RAISED"</text> + <bd>3</bd> + </button>
<button> + <relief>GROOVE</relief> + <text>"GROOVE"</text> + <bd>3</bd> + </button>
<button> + <relief>RIDGE</relief> + <text>"RIDGE"</text> + <bd>3</bd> + </button> + </hbox>
6.2. Hbox
Utilisez une Hbox lorsque vous voulez aligner les widgets, horizontalement, les uns à côtés des autres.
<hbox>
à l’intérieur d’une Hbox, vous pouvez utiliser les balises <boxfill fill=""/>, <boxanchor anchor=""/> et <boxexpand expand=""/> pour choisir le comportement des éléments contenus dans la boîte, lors d’un redimensionnement de la fenêtre. Pour des détails sur le comportement de fill, anchor, et expand, référez vous au manuel du pack Tk, pack(3tk). Par défaut, fill='y', anchor='center', expand='yes'.
6.3. Vbox
Utilisez une Vbox lorsque vous voulez aligner les widgets verticalement, les uns au dessus des autres.
<vbox>
à l’intérieur d’une Hbox, vous pouvez utiliser les balises <boxfill fill=""/>, <boxanchor anchor=""/> et <boxexpand expand=""/> pour choisir le comportement des éléments contenus dans la boîte, lors d’un redimensionnement de la fenêtre. Pour des détails sur le comportement de fill, anchor, et expand, référez vous au manuel du pack Tk, pack(3tk). Par défaut, fill='y', anchor='center', expand='yes'.
6.4. Label
Un label est un texte qui s’affiche sur le panneau.
<label>
6.5. Labelframe
Un labelframe est un cadre entouré d’un sillon et un label en haut à gauche.
<labelframe text="Label: Leds groupées">
6.6. Table
Une table est un container qui permet d'écrire dans une grille de lignes et de colonnes. Chaque ligne débute avec la balise <tablerow/> . Un widget container peut être en lignes ou en colonnes par l’utilisation de la balise <tablespan rows= cols=/> . Les bordures des cellules contenant les widgets “sticky” peuvent être réglées grâce à l’utilisation de la balise <tablesticky sticky=/>. Une table peut s'étirer sur ses lignes et colonnes flexibles (sticky).
Exemple:
<table flexible_rows="[2]" flexible_columns="[1,4]">
image:images/pyvcp_table.png[]
6.7. Onglets (Tabs)
Une interface à onglets peut économiser énormément d’espace.
<tabs> + <names> ["spindle","green eggs"]</names> + </tabs> + <tabs> + <names>["Spindle", "Green Eggs", "Ham"]</names> + <vbox> + <label> + <text>"Spindle speed:"</text> + </label> + <bar> + <halpin>"spindle-speed"</halpin> + <max_>5000</max_> + </bar> + </vbox> + <vbox> + <label> + <text>"(this is the green eggs tab)"</text> + </label> + </vbox> + <vbox> + <label> + <text>"(this tab has nothing on it)"</text> + </label> + </vbox> + </tabs>