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.
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.
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.
19 # ======================================================================
25 from widgets import ListeChoix
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.
35 def __init__(self,parent,panneau,node) :
37 self.panneau = panneau
39 Frame.__init__(self,self.panneau)
40 self.place(x=0,y=0,relheight=1,relwidth=1)
48 # Because on herite de Frame
50 # On supprime explicitement les references aux objets Tk
59 self.frame_boutons=None
60 self.frame_comment=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)
70 def creer_boutons(self):
72 Méthode créant les boutons se trouvant dans la partie contextuelle d'EFICAS
73 (à droite sous les onglets )
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,
81 self.bouton_sup = Button(self.fr_but,
83 command=self.supprimer,
85 self.bouton_doc = Button(self.fr_but,
87 command=self.visu_doc,
89 self.bouton_cata = Button(self.fr_but,
91 command = self.show_catalogue,
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')
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')
101 def show_catalogue(self):
103 genea = self.node.item.get_genealogie()
104 self.parent.appli.browser_catalogue_objet(genea)
106 traceback.print_exc()
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 # ------------------------------------------------------------------------
118 def ajout_commentaire(self,ind='after'):
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.
124 if self.parent.modified == 'n' : self.parent.init_modif()
125 return self.node.append_brother("COMMENTAIRE",ind)
127 def ajout_commentaire_first(self):
129 Ajoute un commentaire en début de JDC
131 if self.parent.modified == 'n' : self.parent.init_modif()
132 return self.node.append_child("COMMENTAIRE",'first')
134 def ajout_parametre(self,ind='after'):
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.
140 if self.parent.modified == 'n' : self.parent.init_modif()
141 return self.node.append_brother("PARAMETRE",ind)
143 def ajout_parametre_first(self):
145 Ajoute un parametre en début de JDC
147 if self.parent.modified == 'n' : self.parent.init_modif()
148 return self.node.append_child("PARAMETRE",'first')
150 def ajout_parametre_eval(self,ind='after'):
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.
156 if self.parent.modified == 'n' : self.parent.init_modif()
157 return self.node.append_brother("PARAMETRE_EVAL",ind)
159 def ajout_parametre_eval_first(self):
161 Ajoute un paramètre EVAL en début de JDC
163 if self.parent.modified == 'n' : self.parent.init_modif()
164 return self.node.append_child("PARAMETRE_EVAL",'first')
166 # ------------------------------------------------------------------------
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,
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)
188 Suppression du noeud courant
190 if self.parent.modified == 'n' : self.parent.init_modif()
191 pere = self.node.parent
195 """ Force l'affichage des fenêtres en cours """
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()
202 for e in cmd.entites.keys() :
204 texte_infos=cmd.entites[e].fr
206 if texte_infos == '' : texte_infos="Pas d'infos disponibles"
207 #EFICAS.affiche_infos(texte_infos)
208 self.parent.appli.affiche_infos(texte_infos)
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)
217 self.ajout_commentaire()
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)
226 def defCmd(self,name):
228 On ajoute une commande après la commande selectionnée : after
229 ou bien on ajoute un commentaire
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')
236 new_node = self.ajout_commentaire()
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')
245 new_node = self.ajout_commentaire_first()
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"""
251 def raisecmd(self,page):
252 self.nb.page(page).focus_set()
253 if page == 'Concept':
260 page=self.nb.getcurselection()
261 self.nb.page(page).focus_set()
262 if page == 'Concept':self._any.component('entry').focus_set()
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 # ------------------------------------------------------------------------
270 def makeConceptPage(self,page):
272 Crée la page de saisie du nom du concept
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()
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')
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)
292 def makeMoclesPage(self,page):
294 Crée la page qui affiche la liste des mots-clés que l'on peut
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)
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)
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')
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)
337 def makeJDCPage(self,page):
339 Crée la page correspondant à un objet de type JDC
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)
352 def makeReglesPage(self,page) :
354 Crée la page qui affiche la liste des règles avec celle qui ne sont
355 pas respectées en rouge
358 regles = self.node.item.get_regles()
359 dictionnaire = self.node.item.get_mc_presents()
361 l_regles_en_defaut=[]
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)
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
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)
379 def makeParamCommentPage_for_etape(self,page):
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
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')
431 def deselectMC(self,name):
432 #EFICAS.affiche_infos('')
433 self.parent.appli.affiche_infos('')
435 def get_liste_cmd_old(self):
436 listeCmd = self.cata.listCmd()
439 def get_liste_cmd(self):
440 listeCmd = self.node.item.object.niveau.definition.get_liste_cmd()
443 def deselectCmd(self,name):
444 #EFICAS.affiche_infos('')
445 self.parent.appli.affiche_infos('')
447 def execConcept(self):
449 Nomme le concept SD retourné par l'étape
451 if not hasattr(self,'valeur_choisie'):
452 nom = self._any.get()
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()
468 def comment_commande(self):
470 Cette méthode a pour but de commentariser la commande pointée par self.node
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)
477 class Panel_Inactif(Panel):
479 Cette classe sert à définir un panneau dans lequel on dit que le noeud
480 sélectionné n'est pas actif
482 def __init__(self,parent,panneau,node) :
484 self.panneau = panneau
486 Frame.__init__(self,self.panneau)
487 self.place(x=0,y=0,relheight=1,relwidth=1)
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,
498 command=self.supprimer,
500 self.bouton_sup.place(relx=0.5,rely=0.8,anchor='center')