Salome HOME
CCAR: mise a jour proc distrib
[tools/eficas.git] / Editeur / panels.py
1 #            CONFIGURATION MANAGEMENT OF EDF VERSION
2 # ======================================================================
3 # COPYRIGHT (C) 1991 - 2002  EDF R&D                  WWW.CODE-ASTER.ORG
4 # THIS PROGRAM IS FREE SOFTWARE; YOU CAN REDISTRIBUTE IT AND/OR MODIFY
5 # IT UNDER THE TERMS OF THE GNU GENERAL PUBLIC LICENSE AS PUBLISHED BY
6 # THE FREE SOFTWARE FOUNDATION; EITHER VERSION 2 OF THE LICENSE, OR
7 # (AT YOUR OPTION) ANY LATER VERSION.
8 #
9 # THIS PROGRAM IS DISTRIBUTED IN THE HOPE THAT IT WILL BE USEFUL, BUT
10 # WITHOUT ANY WARRANTY; WITHOUT EVEN THE IMPLIED WARRANTY OF
11 # MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE. SEE THE GNU
12 # GENERAL PUBLIC LICENSE FOR MORE DETAILS.
13 #
14 # YOU SHOULD HAVE RECEIVED A COPY OF THE GNU GENERAL PUBLIC LICENSE
15 # ALONG WITH THIS PROGRAM; IF NOT, WRITE TO EDF R&D CODE_ASTER,
16 #    1 AVENUE DU GENERAL DE GAULLE, 92141 CLAMART CEDEX, FRANCE.
17 #
18 #
19 # ======================================================================
20 import string
21 import os
22 from Tkinter import *
23 import Pmw
24 import time
25
26 from widgets import ListeChoix
27 from widgets import ListeChoixParGroupes
28 import prefs
29 import options
30
31 SEPARATEUR = '-'*30
32
33 class Panel(Frame) :
34   """
35   Classe servant de classe mère à toutes celles représentant les
36   panneaux à afficher en fonction de la nature de l'objet en cours
37   Elle est toujours dérivée.
38   """
39   def __init__(self,parent,panneau,node) :
40       self.parent=parent
41       self.panneau = panneau
42       self.node=node
43       Frame.__init__(self,self.panneau)
44       self.place(x=0,y=0,relheight=1,relwidth=1)
45       self.creer_boutons()
46       self.init()
47
48   def destroy(self):
49       Frame.destroy(self)
50       self.panneau=None
51       self.parent=None
52       # Because on herite de Frame
53       self.master=None
54       # On supprime explicitement les references aux objets Tk
55       self.nb=None
56       self.fr_but=None
57       self.bouton_cata=None
58       self.bouton_doc=None
59       self.bouton_com=None
60       self.bouton_sup=None
61       self.frame_eval=None
62       self.label=None
63       self.frame_boutons=None
64       self.frame_comment=None
65       self.frame_param=None
66       # On termine la suppression de facon brutale (objets Tk et non Tk)
67       for k in self.__dict__.keys():
68          # il est plus prudent de ne pas détruire le lien sur le Node
69          # si on voulait mettre l'attribut node à None, il faudrait
70          # que tous les appels à node.parent.select() apparaissent après
71          # toutes les autres actions liées au panel (node.item.isglobal(), ...)
72          if k != 'node' : setattr(self,k,None)
73
74   def creer_boutons(self):
75       """
76       Méthode créant les boutons se trouvant dans la partie contextuelle d'EFICAS
77       (à droite sous les onglets )
78       """
79       self.fr_but = Frame(self,height=30)
80       self.fr_but.pack(side='bottom',fill='x')
81       self.bouton_com = Button(self.fr_but,
82                                text = 'Commentariser',
83                                command = self.ajout_commentaire,
84                                width=14)
85       self.bouton_sup = Button(self.fr_but,
86                                text = "Supprimer",
87                                command=self.supprimer,
88                                width=14)
89       self.bouton_doc = Button(self.fr_but,
90                                text="Documentation",
91                                command=self.visu_doc,
92                                width=14)
93       self.bouton_cata = Button(self.fr_but,
94                                 text = "Catalogue",
95                                 command = self.show_catalogue,
96                                 width=14)
97       if self.parent.appli.CONFIGURATION.isdeveloppeur == 'OUI':
98           self.bouton_sup.place(relx=0.25,rely = 0.5,relheight = 0.8,anchor='center')
99           self.bouton_cata.place(relx=0.5,rely = 0.5,relheight = 0.8,anchor='center')
100           self.bouton_doc.place(relx=0.75,rely = 0.5,relheight = 0.8,anchor='center')
101       else:
102           self.bouton_sup.place(relx=0.3,rely = 0.5,relheight = 0.8,anchor='center')
103           self.bouton_doc.place(relx=0.7,rely = 0.5,relheight = 0.8,anchor='center')
104
105   def show_catalogue(self):
106       try:
107           genea = self.node.item.get_genealogie()
108           self.parent.appli.browser_catalogue_objet(genea)
109       except Exception,e:
110           traceback.print_exc()
111       
112   def efface(self):
113       self.node.efface()
114
115 # ------------------------------------------------------------------------
116 #     Méthodes permettant d'ajouter des commentaires, des paramètres
117 #                     et des objets EVAL.
118 #       Ces méthodes sont utilisées par les panneaux des JDC,ETAPE,
119 #                 COMMENTAIRE et PARAMETRE
120 # ------------------------------------------------------------------------
121
122   def ajout_commentaire(self,ind='after'):
123       """
124       Ajoute un commentaire à l'intérieur du JDC :
125       - si ind='after'  : l'ajoute après l'objet courant
126       - si ind='before' : l'ajoute avant.
127       """
128       if self.parent.modified == 'n' : self.parent.init_modif()
129       return self.node.append_brother("COMMENTAIRE",ind)
130     
131   def ajout_commentaire_first(self):
132       """
133       Ajoute un commentaire en début de JDC
134       """
135       if self.parent.modified == 'n' : self.parent.init_modif()
136       return self.node.append_child("COMMENTAIRE",'first')
137
138   def ajout_parametre(self,ind='after'):
139       """
140       Ajoute un parametre à l'intérieur du JDC :
141       - si ind='after'  : l'ajoute après l'objet courant
142       - si ind='before' : l'ajoute avant.
143       """
144       if self.parent.modified == 'n' : self.parent.init_modif()
145       return self.node.append_brother("PARAMETRE",ind)
146     
147   def ajout_parametre_first(self):
148       """
149       Ajoute un parametre en début de JDC
150       """
151       if self.parent.modified == 'n' : self.parent.init_modif()
152       return self.node.append_child("PARAMETRE",'first')
153
154   def ajout_parametre_eval(self,ind='after'):
155       """
156       Ajoute un paramètre EVAL à l'intérieur du JDC :
157       - si ind='after'  : l'ajoute après l'objet courant
158       - si ind='before' : l'ajoute avant.
159       """
160       if self.parent.modified == 'n' : self.parent.init_modif()
161       return self.node.append_brother("PARAMETRE_EVAL",ind)
162     
163   def ajout_parametre_eval_first(self):
164       """
165       Ajoute un paramètre EVAL en début de JDC
166       """
167       if self.parent.modified == 'n' : self.parent.init_modif()
168       return self.node.append_child("PARAMETRE_EVAL",'first')
169     
170 # ------------------------------------------------------------------------
171    
172   def visu_doc(self):
173       """ Permet d'ouvrir le fichier doc U de la commande au format pdf avec Acrobat Reader
174         - Ne fonctionne pas sous UNIX (chemin d'accès Acrobat Reader)
175         - indication du chemin d'accès aux fichiers pdf à revoir : trop statique"""
176       cle_doc = self.node.item.get_docu()
177       if cle_doc == None : return
178       cle_doc = string.replace(cle_doc,'.','')
179       cle_doc = string.replace(cle_doc,'-','')
180       commande = self.parent.appli.CONFIGURATION.exec_acrobat
181       nom_fichier = cle_doc+".pdf"
182       fichier = os.path.abspath(os.path.join(self.parent.appli.CONFIGURATION.path_doc,
183                                        nom_fichier))
184       if os.name == 'nt':
185           os.spawnv(os.P_NOWAIT,commande,(commande,fichier,))
186       elif os.name == 'posix':
187           script ="#!/usr/bin/sh \n%s %s&" %(commande,fichier)
188           pid = os.system(script)
189       
190   def supprimer(self):
191       """
192       Suppression du noeud courant
193       """
194       if self.parent.modified == 'n' : self.parent.init_modif()
195       pere = self.node.parent
196       self.node.delete()
197       
198   def affiche(self):
199       """ Force l'affichage des fenêtres en cours """
200       self.tkraise()
201
202   def selectMC(self,name):
203       """ On retrouve le mot-clé sous le curseur pour affichage du fr """
204       cmd=self.node.item.get_definition()
205       texte_infos = ''
206       for e in cmd.entites.keys() :
207           if e == name :
208               texte_infos=getattr(cmd.entites[e],prefs.lang)
209               break
210       if texte_infos == '' : texte_infos="Pas d'infos disponibles"
211       self.parent.appli.affiche_infos(texte_infos)
212
213   def defMC(self,name):
214       """ On ajoute un mot-clé à la commande : subnode """
215       if name == SEPARATEUR:return
216       if self.parent.modified == 'n' : self.parent.init_modif()
217       if name != "COMMENTAIRE":
218           self.node.append_child(name)
219       else :
220           self.ajout_commentaire()    
221
222   def selectCmd(self,name):
223       """ On retrouve la commande sous le curseur pour affichage du fr """
224       if name != 'COMMENTAIRE' and name != SEPARATEUR:
225           texte_infos=getattr(self.parent.jdc.get_cmd(name),prefs.lang)
226           self.parent.appli.affiche_infos(texte_infos)
227           
228   def defCmd(self,name):
229       """
230       On ajoute une commande après la commande selectionnée : after
231       ou bien on ajoute un commentaire
232       """
233       if name == SEPARATEUR:return
234       if self.parent.modified == 'n' : self.parent.init_modif()
235       if name != "COMMENTAIRE":
236           new_node = self.node.append_brother(name,'after')
237       else :
238           new_node = self.ajout_commentaire()
239
240   def defCmdFirst(self,name):
241       """ On ajoute une commande ou un commentaire au début du fichier de commandes """
242       if name == SEPARATEUR:return
243       if self.parent.modified == 'n' : self.parent.init_modif()
244       if name != "COMMENTAIRE":
245           new_node = self.node.append_child(name,'first')
246       else :
247           new_node = self.ajout_commentaire_first()
248         
249 class OngletPanel(Panel) :
250   """ Cette classe est virtuelle et doit être dérivée
251       Elle contient les principales méthodes d'affichage des différents onglets"""
252
253   def raisecmd(self,page):
254       self.nb.page(page).focus_set()
255       if page == 'Concept':
256           try:
257               self._any.focus()
258           except:
259               pass
260
261   def affiche(self):
262       page=self.nb.getcurselection()
263       self.nb.page(page).focus_set()
264       if page == 'Concept':self._any.component('entry').focus_set()
265       self.tkraise()
266
267 # ------------------------------------------------------------------------
268 #     Méthodes permettant d'afficher des pages partagées par différents
269 #           types d'objets (règles,mots-clés,concept,...)
270 # ------------------------------------------------------------------------
271
272   def makeConceptPage(self,page):
273       """
274       Crée la page de saisie du nom du concept
275       """
276       self.label = Label(page,text='Nom du concept :')
277       self.label.place(relx=0.1,rely=0.4)
278       self._any = Entry(page,relief='sunken')
279       self._any.place(relx=0.35,rely=0.4,relwidth=0.5)
280       self._any.bind("<Return>",lambda e,s=self:s.execConcept())
281       self._any.insert(0,self.node.item.GetText())
282       type_sd = self.node.item.get_type_sd_prod()
283       if type_sd :
284           txt = "L'opérateur courant retourne un objet de type %s" %type_sd
285           self.label = Label(page, text = txt)
286           self.label.place(relx=0.5,rely=0.55,anchor='n')
287       self._any.focus()
288       # aide associée au panneau
289       bulle_aide="""Tapez dans la zone de saisie le nom que vous voulez donner
290       au concept retoruné par l'opérateur courant et pressez <Return> pour valider"""
291       page.bind("<Button-3>", lambda e,s=self,a=bulle_aide : s.parent.appli.affiche_aide(e,a))
292       page.bind("<ButtonRelease-3>",self.parent.appli.efface_aide)
293         
294   def makeMoclesPage(self,page):
295       """
296       Crée la page qui affiche la liste des mots-clés que l'on peut
297       encore ajouter
298       """
299       genea =self.node.item.get_genealogie()
300       jdc = self.node.item.get_jdc()
301       liste_mc=self.node.item.get_liste_mc_ordonnee(genea,jdc.cata_ordonne_dico)
302       liste_commandes = (("<Enter>",self.selectMC),
303                          ("<Leave>",self.deselectMC),
304                          ("<Double-Button-1>",self.defMC))
305       Liste = ListeChoix(self,page,liste_mc,liste_commandes = liste_commandes,titre = "Mots-clés permis")
306       Liste.affiche_liste()
307       # aide associée au panneau
308       bulle_aide="""Double-cliquez sur le mot-clé que vous voulez ajouter à
309       la commande en cours d'édition"""
310       Liste.MCbox.bind("<Button-3>", lambda e,s=self,a=bulle_aide : s.parent.appli.affiche_aide(e,a))
311       Liste.MCbox.bind("<ButtonRelease-3>",self.parent.appli.efface_aide)
312
313   def makeCommentairePage(self,page):
314       label = Label(page,text = "Insérer un commentaire :")
315       label.grid(column = 0, row = 2)
316       but_avant = Button(page,text = "AVANT",command = lambda s=self :s.ajout_commentaire(ind = 'before'))
317       but_apres = Button(page,text = "APRES",command = self.ajout_commentaire)
318       but_avant.grid(column = 1,row =2)
319       but_apres.grid(column = 1,row =3)
320       
321   def makeCommandePage(self,page):
322       """
323          Cree l'onglet
324       """
325       frame1 = Frame(page,height = 20)
326       frame1.pack(side='top',fill='x')
327       label = Label(frame1,text ="La commande choisie sera ajoutée\n APRES la commande courante")
328       label.pack(side='top')
329       frame2 = Frame(page)
330       frame2.pack(side='top',fill='both',expand=1)
331       liste_commandes = (("<Enter>",self.selectCmd),
332                          ("<Leave>",self.deselectCmd),
333                          ("<Double-Button-1>",self.defCmd))
334       if options.affichage_commandes == "alphabetic":
335          liste_cmd = self.get_liste_cmd()
336          Liste = ListeChoix(self,frame2,liste_cmd,liste_commandes = liste_commandes,
337                                    filtre='oui',titre = "Commandes")
338       else:
339          liste_groupes=self.node.item.object.niveau.definition.liste_groupes
340          dict_groupes=self.node.item.object.niveau.definition.dict_groupes
341          Liste = ListeChoixParGroupes(self,frame2,liste_groupes,dict_groupes,
342                                       liste_commandes = liste_commandes,
343                                       filtre='oui',titre = "Commandes")
344       Liste.affiche_liste()
345       # aide associée au panneau
346       bulle_aide="""Double-cliquez sur la commande que vous voulez ajouter au jeu de commandes"""
347       Liste.MCbox.bind("<Button-3>", lambda e,s=self,a=bulle_aide : s.parent.appli.affiche_aide(e,a))
348       Liste.MCbox.bind("<ButtonRelease-3>",self.parent.appli.efface_aide)
349
350   def makeJDCPage(self,page):
351       """
352       Crée la page correspondant à un objet de type JDC
353       """
354       liste_commandes = (("<Enter>",self.selectCmd),
355                          ("<Leave>",self.deselectCmd),
356                          ("<Double-Button-1>",self.defCmdFirst))
357       if options.affichage_commandes == "alphabetic":
358          liste_cmd = self.get_liste_cmd()
359          Liste = ListeChoix(self,page,liste_cmd,liste_commandes = liste_commandes,
360                             filtre='oui',titre = "Commandes")
361       else:
362          liste_groupes=self.node.item.object.niveau.definition.liste_groupes
363          dict_groupes=self.node.item.object.niveau.definition.dict_groupes
364          Liste = ListeChoixParGroupes(self,page,liste_groupes,dict_groupes,
365                                       liste_commandes = liste_commandes,
366                                       filtre='oui',titre = "Commandes")
367       Liste.affiche_liste()
368        # aide associée au panneau
369       bulle_aide="""Double-cliquez sur la commande que vous voulez ajouter au jeu de commandes"""
370       Liste.MCbox.bind("<Button-3>", lambda e,s=self,a=bulle_aide : s.parent.appli.affiche_aide(e,a))
371       Liste.MCbox.bind("<ButtonRelease-3>",self.parent.appli.efface_aide)
372
373   def makeReglesPage(self,page) :
374       """
375       Crée la page qui affiche la liste des règles avec celle qui ne sont
376       pas respectées en rouge
377       """
378       regles = []
379       regles = self.node.item.get_regles()
380       dictionnaire = self.node.item.get_mc_presents()
381       texte_regles = []
382       l_regles_en_defaut=[]
383       if len(regles) > 0:
384         i = 0
385         for regle in regles :
386           texte_regles.append(regle.gettext())
387           texte,test = regle.verif(dictionnaire)
388           if test == 0 : l_regles_en_defaut.append(i)
389           i = i+1
390       Liste = ListeChoix(self,page,texte_regles,liste_marques=l_regles_en_defaut,active='non',titre="Règles")
391       Liste.affiche_liste()
392       # aide associée au panneau
393       bulle_aide="""Ce panneau contient la liste des règles qui s'appliquent à l'objet
394       en cours d'édition.
395       - en noir : règles valides
396       - en rouge : règles violées"""
397       Liste.MCbox.bind("<Button-3>", lambda e,s=self,a=bulle_aide : s.parent.appli.affiche_aide(e,a))
398       Liste.MCbox.bind("<ButtonRelease-3>",self.parent.appli.efface_aide)
399
400   def makeParamCommentPage_for_etape(self,page):
401       """
402       Crée la page qui offre le choix à l'utilisateur d'ajouter un commentaire
403       ou un paramètre, avant ou après le noeud courant dans l'arbre.
404       Cette page est destinée aux objets de niveau ETAPE cad à toutes les CMD,
405       les commentaires inter commandes et les paramètres
406       """
407       # les frame ...
408       self.frame_comment = Frame(page,bd=1,relief='raised')
409       self.frame_param   = Frame(page,bd=1,relief='raised')
410       self.frame_eval    = Frame(page,bd=1,relief='raised')
411       self.frame_boutons = Frame(page,bd=1,relief='raised')
412       self.frame_comment.place(relx=0,rely=0,relwidth=1,relheight=0.28)
413       self.frame_param.place(relx=0,rely=0.28,relwidth=1,relheight=0.28)
414       self.frame_eval.place(relx=0,rely=0.56,relwidth=1,relheight=0.28)
415       self.frame_boutons.place(relx=0,rely=0.84,relwidth=1,relheight=0.16)
416       # remplissage de la frame commentaire
417       Label(self.frame_comment,text = "Insérer un commentaire :").place(relx=0.1,rely=0.5,anchor='w')
418       but_comment_avant = Button(self.frame_comment,
419                                  text = "AVANT "+self.node.item.get_nom(),
420                                  command = lambda s=self :s.ajout_commentaire(ind = 'before'))
421       but_comment_apres = Button(self.frame_comment,
422                                  text = "APRES "+self.node.item.get_nom(),
423                                  command = self.ajout_commentaire)
424       but_comment_avant.place(relx=0.6,rely=0.3,anchor='w',relwidth=0.3)
425       but_comment_apres.place(relx=0.6,rely=0.7,anchor='w',relwidth=0.3)
426       # remplissage de la frame paramètre
427       Label(self.frame_param,text = "Insérer un paramètre :").place(relx=0.1,rely=0.5,anchor='w')
428       but_param_avant = Button(self.frame_param,
429                                  text = "AVANT "+self.node.item.get_nom(),
430                                  command = lambda s=self :s.ajout_parametre(ind = 'before'))
431       but_param_apres = Button(self.frame_param,
432                                  text = "APRES "+self.node.item.get_nom(),
433                                  command = self.ajout_parametre)
434       but_param_avant.place(relx=0.6,rely=0.3,anchor='w',relwidth=0.3)
435       but_param_apres.place(relx=0.6,rely=0.7,anchor='w',relwidth=0.3)
436       # remplissage de la frame eval
437       Label(self.frame_eval,text="Insérer un paramètre EVAL :").place(relx=0.1,rely=0.5,anchor='w')
438           #Label(self.frame_eval,text='Non encore disponible').place(relx=0.6,rely=0.5,anchor='w')
439       but_eval_avant = Button(self.frame_eval,
440                               text = "AVANT "+self.node.item.get_nom(),
441                               command = lambda s=self :s.ajout_parametre_eval(ind = 'before'))
442       but_eval_apres = Button(self.frame_eval,
443                               text = "APRES "+self.node.item.get_nom(),
444                               command = self.ajout_parametre_eval)
445       but_eval_avant.place(relx=0.6,rely=0.3,anchor='w',relwidth=0.3)
446       but_eval_apres.place(relx=0.6,rely=0.7,anchor='w',relwidth=0.3)      
447       # remplissage de la frame boutons
448       Button(self.frame_boutons,
449              text="Commentariser toute la commande",
450              command = self.comment_commande).place(relx=0.5,rely=0.5,anchor='center')
451     
452   def deselectMC(self,name):
453       #EFICAS.affiche_infos('')
454       self.parent.appli.affiche_infos('')
455     
456   def get_liste_cmd_old(self):
457       listeCmd = self.cata.listCmd()
458       return listeCmd
459
460   def get_liste_cmd(self):
461       listeCmd = self.node.item.object.niveau.definition.get_liste_cmd()
462       return listeCmd
463
464   def deselectCmd(self,name):
465       #EFICAS.affiche_infos('')
466       self.parent.appli.affiche_infos('')
467     
468   def execConcept(self):
469       """
470       Nomme le concept SD retourné par l'étape
471       """
472       if not hasattr(self,'valeur_choisie'):
473           nom = self._any.get()
474       else:
475           nom = self.valeur_choisie.get()
476       nom = string.strip(nom)
477       if nom == '' : return # si pas de nom, on ressort sans rien faire ...
478       if self.parent.modified == 'n' : self.parent.init_modif()
479       # Pourquoi node.etape ???
480       #test,mess = self.node.etape.item.nomme_sd(nom)
481       test,mess = self.node.item.nomme_sd(nom)
482       #EFICAS.affiche_infos(mess)
483       self.parent.appli.affiche_infos(mess)
484       self.node.racine.update()
485   
486   def changed(self):
487       pass
488
489   def comment_commande(self):
490     """
491     Cette méthode a pour but de commentariser la commande pointée par self.node
492     """
493     commande_comment = self.node.item.get_objet_commentarise()
494     #XXX il faudrait ici aussi eviter l'appel à EFICAS
495     self.parent.appli.bureau.JDCDisplay_courant.ReplaceObjectNode(self.node,commande_comment,None)
496     #EFICAS.JDCDisplay_courant.ReplaceObjectNode(self.node,commande_comment,None)
497       
498 class Panel_Inactif(Panel):
499   """
500      Cette classe sert à définir un panneau dans lequel on dit que le noeud 
501      sélectionné n'est pas actif
502   """
503   def __init__(self,parent,panneau,node) :
504       self.parent=parent
505       self.panneau = panneau
506       self.node=node
507       Frame.__init__(self,self.panneau)
508       self.place(x=0,y=0,relheight=1,relwidth=1)
509       self.creer_texte()
510
511   def creer_texte(self):
512       texte = "Le noeud sélectionné ne correspond pas à un objet actif\n"
513       texte = texte + "Seules les commandes placées entre \nDEBUT/POURSUITE et FIN sont actives"
514       longueur = int(self.panneau.winfo_width()*0.8)
515       self.label = Label(self,text=texte,wraplength=longueur,justify='center')
516       self.label.place(relx=0.5,rely=0.4,relwidth=0.8,anchor='center')
517       self.bouton_sup = Button(self,
518                                text = "Supprimer",
519                                command=self.supprimer,
520                                width=14)
521       self.bouton_sup.place(relx=0.5,rely=0.8,anchor='center')
522
523