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