Salome HOME
*** empty log message ***
[tools/eficas.git] / Editeur / composimp.py
1 # -*- coding: utf-8 -*-
2 #            CONFIGURATION MANAGEMENT OF EDF VERSION
3 # ======================================================================
4 # COPYRIGHT (C) 1991 - 2002  EDF R&D                  WWW.CODE-ASTER.ORG
5 # THIS PROGRAM IS FREE SOFTWARE; YOU CAN REDISTRIBUTE IT AND/OR MODIFY
6 # IT UNDER THE TERMS OF THE GNU GENERAL PUBLIC LICENSE AS PUBLISHED BY
7 # THE FREE SOFTWARE FOUNDATION; EITHER VERSION 2 OF THE LICENSE, OR
8 # (AT YOUR OPTION) ANY LATER VERSION.
9 #
10 # THIS PROGRAM IS DISTRIBUTED IN THE HOPE THAT IT WILL BE USEFUL, BUT
11 # WITHOUT ANY WARRANTY; WITHOUT EVEN THE IMPLIED WARRANTY OF
12 # MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE. SEE THE GNU
13 # GENERAL PUBLIC LICENSE FOR MORE DETAILS.
14 #
15 # YOU SHOULD HAVE RECEIVED A COPY OF THE GNU GENERAL PUBLIC LICENSE
16 # ALONG WITH THIS PROGRAM; IF NOT, WRITE TO EDF R&D CODE_ASTER,
17 #    1 AVENUE DU GENERAL DE GAULLE, 92141 CLAMART CEDEX, FRANCE.
18 #
19 #
20 # ======================================================================
21 # Modules Python
22 import string,types,os
23 from Tkinter import *
24 import Pmw
25 from copy import copy,deepcopy
26 import traceback
27
28 # Modules Eficas
29 import Objecttreeitem
30 import prefs
31 import panels
32 import images
33 from widgets import ListeChoix
34 from widgets import FenetreDeSelection
35
36 from Noyau.N_CR import justify_text
37 from utils import substract_list
38
39       
40 class SIMPTreeItem(Objecttreeitem.AtomicObjectTreeItem):
41   from newsimppanel import newSIMPPanel
42   panel = newSIMPPanel
43
44   def init(self) :
45       self.expandable = 0
46       self.affect_panel()
47
48
49   def affect_panel(self):
50       """
51       Cette méthode attribue le panel à l'objet pointé par self en fonction de la
52       nature de la valeur demandée pour cet objet
53       """
54       from uniquepanel import UNIQUE_Panel
55       from plusieurspanel import PLUSIEURS_Panel
56
57       #print "affect_panel : ",self.nom,self.is_list(),self.has_into(), self.get_into(None)
58
59       if self.wait_shell():
60           # l'objet attend un shell
61           from shellpanel import SHELLPanel
62           self.panel = SHELLPanel
63       elif self.has_into():
64           # l'objet prend sa (ses) valeur(s) dans un ensemble discret de valeurs
65           if self.is_list() :
66              from plusieursintopanel import PLUSIEURS_INTO_Panel
67              self.panel = PLUSIEURS_INTO_Panel
68           else:
69              from uniqueintopanel import UNIQUE_INTO_Panel
70              self.panel = UNIQUE_INTO_Panel
71       else:
72           # l'objet prend une ou des valeurs à priori quelconques
73           if self.is_list() :
74               # on attend une liste de valeurs mais de quel type ?
75               if self.wait_assd():
76                   # on attend une liste de SD
77                   from plusieursassdpanel import PLUSIEURS_ASSD_Panel
78                   self.panel = PLUSIEURS_ASSD_Panel
79               else:
80                   # on attend une liste de valeurs de types debase (entiers, réels,...)
81                   from plusieursbasepanel import PLUSIEURS_BASE_Panel
82                   self.panel = PLUSIEURS_BASE_Panel
83           else:
84               # on n'attend qu'une seule valeur mais de quel type ?
85               if self.wait_co():
86                   # on attend une SD ou un objet de la classe CO (qui n'existe pas encore)
87                   from uniquesdcopanel import UNIQUE_SDCO_Panel
88                   self.panel = UNIQUE_SDCO_Panel
89               elif self.wait_assd():
90                   # on attend une SD
91                   from uniqueassdpanel import UNIQUE_ASSD_Panel
92                   from uniqueassdpanel import UNIQUE_ASSD_Panel_Reel
93                   if 'R' in self.GetType():
94                      self.panel = UNIQUE_ASSD_Panel_Reel
95                   else :
96                      self.panel = UNIQUE_ASSD_Panel
97               else:
98                   # on attend une valeur d'un type de base (entier,réel,...)
99                   if self.wait_complex():
100                       # on attend un complexe
101                       from uniquecomppanel import UNIQUE_COMP_Panel
102                       self.panel = UNIQUE_COMP_Panel
103                   else:
104                       # on attend un entier, un réel ou une string
105                       from uniquebasepanel import UNIQUE_BASE_Panel
106                       self.panel = UNIQUE_BASE_Panel
107       # cas particulier des fonctions
108       genea = self.get_genealogie()
109       if "VALE" in genea or "VALE_C" in genea:
110          if "DEFI_FONCTION" in genea :
111             from fonctionpanel import FONCTION_Panel
112             self.panel=FONCTION_Panel
113       #---------------------------------------------------------
114       # PN ajout pour lancement de Salome
115       #---------------------------------------------------------
116       if  self.appli.salome != 0 :
117           import panelsSalome
118
119           self.select_noeud_maille=0
120           self.clef_fonction="SALOME"
121           for i in range(0,len( genea )) :
122              self.clef_fonction=self.clef_fonction+"_"+ genea[i]
123              if genea[i] == "GROUP_NO" or genea[i] == "GROUP_MA":
124                 self.select_noeud_maille=1
125
126           recherche=panelsSalome.dict_classes_salome[self.panel]
127           if hasattr(recherche,self.clef_fonction):
128              self.panel=recherche
129           if self.select_noeud_maille==1 :
130              self.panel=recherche
131
132
133   #-----------------------------------------------
134   #
135   # Methodes liees aux informations sur le Panel
136   # ou au mot-clef simple
137   #
138   #-----------------------------------------------
139   # is_list
140   # get_into            a priori inutile --> commentee
141   # has_into
142   # wait_into           a priori inutile --> commentee
143   # GetMinMax
144   # GetMultiplicite
145   # GetIntervalle
146   # GetListeValeurs
147   # get_liste_possible
148
149   def is_list(self):
150       """
151           Cette méthode indique si le mot cle simple attend une liste (valeur de retour 1)
152           ou s'il n'en attend pas (valeur de retour 0)
153
154           Deux cas principaux peuvent se presenter : avec validateurs ou bien sans.
155           Dans le cas sans validateur, l'information est donnée par l'attribut max
156           de la definition du mot cle.
157           Dans le cas avec validateur, il faut combiner l'information précédente avec
158           celle issue de l'appel de la méthode is_list sur le validateur.On utilisera
159           l'operateur ET pour effectuer cette combinaison (AndVal).
160       """
161       is_a_list=0
162       min,max = self.GetMinMax()
163       assert (min <= max)
164       if max > 1 :
165                 is_a_list=1
166       # Dans le cas avec validateurs, pour que le mot cle soit considéré
167       # comme acceptant une liste, il faut que max soit supérieur a 1
168       # ET que la méthode is_list du validateur retourne 1. Dans les autres cas
169       # on retournera 0 (n'attend pas de liste)
170       if self.definition.validators :
171          is_a_list= self.definition.validators.is_list() * is_a_list
172       return is_a_list 
173
174   #def get_into(self,liste_courante=None):
175   #    """
176   #        Cette méthode retourne la liste de choix proposée par le mot cle. Si le mot cle ne propose
177   #        pas de liste de choix, la méthode retourne None.
178   #        L'argument d'entrée liste_courante, s'il est différent de None, donne la liste des choix déjà
179   #        effectués par l'utilisateur. Dans ce cas, la méthode get_into doit calculer la liste des choix
180   #        en en tenant compte.
181   #        Cette méthode part du principe que la relation entre into du mot clé et les validateurs est
182   #        une relation de type ET (AndVal).
183   #    """
184   #    if not self.object.definition.validators :
185   #       return self.object.definition.into
186   #    else:
187   #       return self.object.definition.validators.get_into(liste_courante,self.definition.into)
188
189   def has_into(self):
190       """
191           Cette méthode indique si le mot cle simple propose un choix (valeur de retour 1)
192           ou s'il n'en propose pas (valeur de retour 0)
193
194           Deux cas principaux peuvent se presenter : avec validateurs ou bien sans.
195           Dans le cas sans validateur, l'information est donnée par l'attribut into
196           de la definition du mot cle.
197           Dans le cas avec validateurs, pour que le mot cle soit considéré
198           comme proposant un choix, il faut que into soit présent OU
199           que la méthode has_into du validateur retourne 1. Dans les autres cas
200           on retournera 0 (ne propose pas de choix)
201       """
202       has_an_into=0
203       if self.definition.into:
204          has_an_into=1
205       elif self.definition.validators :
206          has_an_into= self.definition.validators.has_into()
207       return has_an_into
208
209 #  def wait_into(self):
210 #      """ Méthode booléenne qui retourne 1 si l'objet pointé par self
211 #      prend ses valeurs dans un ensemble discret (into), 0 sinon """
212 #      if self.object.definition.into != None :
213 #          return 1
214 #      else:
215 #          return 0
216
217   def GetMinMax(self):
218       """ Retourne les valeurs min et max de la définition de object """
219       return self.object.get_min_max()
220
221   def GetMultiplicite(self):
222       """ A préciser.
223           Retourne la multiplicité des valeurs affectées à l'objet
224           représenté par l'item. Pour le moment retourne invariablement 1.
225       """
226       return 1
227
228   def GetIntervalle(self):
229       """ 
230            Retourne le domaine de valeur attendu par l'objet représenté 
231            par l'item.
232       """
233       return self.object.getintervalle()
234
235   def GetListeValeurs(self) :
236       """ Retourne la liste des valeurs de object """
237       valeurs=self.object.get_liste_valeurs()
238       try :
239         if "R" in self.object.definition.type:
240            clef=self.object.GetNomConcept()
241            if self.appli.dict_reels.has_key(clef):
242               if type(valeurs) == types.TupleType:
243                  valeurs_reelles=[]
244                  for val in valeurs :
245                     if self.appli.dict_reels[clef].has_key(val) : 
246                        valeurs_reelles.append(self.appli.dict_reels[clef][val])
247                     else :
248                        valeurs_reelles.append(val)
249               else :
250                  if self.appli.dict_reels[clef].has_key(valeurs):
251                     valeurs_reelles=self.appli.dict_reels[clef][valeurs]
252               valeurs=valeurs_reelles
253       except :
254         pass
255       return valeurs
256     
257   def get_liste_possible(self,listeActuelle=[]):
258       if hasattr(self.definition.validators,'into'):
259          self.get_definition().into=self.definition.validators.into 
260       valeurspossibles = self.get_definition().into
261       # CCAR : Ne serait-il pas preferable d'appeler get_into ?
262       #valeurspossibles=self.get_into(listeActuelle)
263
264       listevalideitem=[]
265       for item in valeurspossibles:
266           encorevalide=self.valide_item(item)
267           if encorevalide :
268              listevalideitem.append(item)
269       # on ne verifie pas la liste des choix si max = 1 
270       # (sinon cela enleve tous les choix possibles)
271       min,max=self.GetMinMax()
272       if max != 1 : 
273          listevalideliste=[]
274          for item in listevalideitem:
275              listetravail=[]
276              for item2 in listeActuelle : listetravail.append(item2)
277              encorevalide=self.valide_liste_partielle(item,listetravail)
278              if encorevalide :
279                 listevalideliste.append(item)
280       else :
281          listevalideliste=listevalideitem
282       return listevalideliste
283
284   def get_liste_param_possible(self):
285   # 
286       liste_param=[]
287       for param in self.object.jdc.params:
288           encorevalide=self.valide_item(param.valeur)
289           if encorevalide:
290              type_param=param.valeur.__class__.__name__
291              for typ in self.definition.type:
292                  if typ=='R':
293                      liste_param.append(param)
294                  if typ=='I' and type_param=='int':
295                      liste_param.append(param)
296                  if typ=='TXM' and type_param=='str':
297                      liste_param.append(repr(param))
298       return liste_param
299
300   #--------------------------------------------------
301   #
302   # Methodes liees a la validite des valeurs saisies
303   #
304   #---------------------------------------------------
305   # valide_item
306   # valide_liste_partielle
307   # valide_liste_complete
308   # info_erreur_item
309   # info_erreur_liste
310   # IsInIntervalle
311   # isvalid
312
313   def valide_item(self,item):
314       """
315         On fait un try except pour les erreurs de type (exple
316         on rentre 1 pour une chaine de caracteres
317       """
318       valide=1
319       if self.definition.validators :
320          try :
321             valide=self.definition.validators.verif_item(item)
322          except :
323             valide = 0
324       return valide
325      
326   def valide_liste_partielle(self,item,listecourante):
327       valeur=listecourante
328       valeur.append(item)
329       valeur = tuple(valeur)
330       return self.object.valid_valeur_partielle(valeur)
331
332   def valide_liste_partielle_BAK(self,item,listecourante):
333       valeuravant=self.object.valeur
334       valeur=listecourante
335       valeur.append(item)
336       valeur = tuple(valeur)
337       retour=self.object.set_valeur(valeur)
338       validite=0
339       if self.object.isvalid():
340          validite=1
341       elif self.definition.validators :
342          validite=self.definition.validators.valide_liste_partielle(valeur)
343
344       if validite==0:
345          min,max=self.GetMinMax()
346          if len(valeur) < min :
347             validite=1
348       retour=self.object.set_valeur(valeuravant)
349       return validite 
350
351   def valide_liste_complete (self,valeur):
352       return self.object.valid_valeur(valeur)
353
354   def valide_liste_complete_BAK (self,valeur):
355       valeuravant=self.object.valeur
356       retour=self.object.set_valeur(valeur)
357       validite=0
358       if self.object.isvalid():
359          validite=1
360       retour=self.object.set_valeur(valeuravant)
361       return validite
362      
363   def info_erreur_item(self) :
364       commentaire=""
365       if self.definition.validators :
366          commentaire=self.definition.validators.info_erreur_item()
367       return commentaire
368       
369   def aide(self) :
370       commentaire=""
371       if self.definition.validators :
372          commentaire=self.definition.validators.aide()
373       return commentaire
374
375   def info_erreur_liste(self) :
376       commentaire=""
377       if self.definition.validators :
378          commentaire=self.definition.validators.info_erreur_liste()
379       return commentaire
380
381   def IsInIntervalle(self,valeur):
382       """ 
383           Retourne 1 si la valeur est dans l'intervalle permis par
384           l'objet représenté par l'item.
385       """
386       return self.object.isinintervalle(valeur)
387
388   def isvalid(self):
389     valide=self.object.isvalid()
390     return valide
391
392   #--------------------------------------------------
393   #
394   # Autres ...
395   #
396   #---------------------------------------------------
397   # SetText     a priori inutilisee --> commentee
398   # GetIconName
399   # GetText
400   # getval     a  priori inutilisee --> commentee
401   # set_valeur_co
402   # get_sd_avant_du_bon_type
403   # verif       a  priori inutilisee --> commentee
404   # delete_valeur_co
405
406   #def SetText(self, text):
407   #  try:
408   #    value = eval(text)
409   #    self.object.setval(value)
410   #  except:
411   #    pass
412
413   def GetIconName(self):
414     if self.isvalid():
415       return "ast-green-ball"
416     elif self.object.isoblig():
417       return "ast-red-ball"
418     else:
419       return "ast-yel-ball"
420
421   def GetText(self):
422     """
423     Classe SIMPTreeItem
424     Retourne le texte à afficher dans l'arbre représentant la valeur de l'objet
425     pointé par self 
426     """
427     text= self.object.GetText()
428     return text
429     
430   #def getval(self):
431   #    return self.object.getval()
432
433   def set_valeur_co(self,nom_co):
434       """
435       Affecte au MCS pointé par self l'objet de type CO et de nom nom_co
436       """
437       ret = self.object.set_valeur_co(nom_co)
438       #print "set_valeur_co",ret
439       return ret
440       
441   def get_sd_avant_du_bon_type(self):
442       """
443       Retourne la liste des noms des SD présentes avant l'étape qui contient
444       le MCS pointé par self et du type requis par ce MCS
445       """
446       a=self.object.etape.parent.get_sd_avant_du_bon_type(self.object.etape,self.object.definition.type)
447       return a
448
449   #def verif(self):
450   #    pass
451
452   def delete_valeur_co(self,valeur=None):
453       """
454            Supprime la valeur du mot cle (de type CO)
455            il faut propager la destruction aux autres etapes
456       """
457       if not valeur : valeur=self.object.valeur
458       # XXX faut il vraiment appeler del_sdprod ???
459       #self.object.etape.parent.del_sdprod(valeur)
460       self.object.etape.parent.delete_concept(valeur)
461
462   #-----------------------------------------------
463   #
464   # Methodes liees au type de l objet attendu
465   #
466   #-----------------------------------------------
467   # wait_co 
468   # wait_geom
469   # wait_complex
470   # wait_reel
471   # wait_shell
472   # wait_assd
473   # GetType
474
475   def wait_co(self):
476       """
477       Méthode booléenne qui retourne 1 si l'objet pointé par self
478       attend un objet de type ASSD qui n'existe pas encore (type CO()),
479       0 sinon
480       """
481       return self.object.wait_co()
482
483   def wait_geom(self):
484       """
485       Méthode booléenne qui retourne 1 si l'objet pointé par self
486       attend un objet GEOM, 0 sinon
487       """
488       return self.object.wait_geom()
489     
490   def wait_complex(self):
491       """ Méthode booléenne qui retourne 1 si l'objet pointé par self
492       attend un complexe, 0 sinon """
493       if 'C' in self.object.definition.type:
494           return 1
495       else:
496           return 0
497
498   def wait_reel(self):
499       """ Méthode booléenne qui retourne 1 si l'objet pointé par self
500       attend un réel, 0 sinon """
501       if 'R' in self.object.definition.type:
502           return 1
503       else:
504           return 0
505         
506   def wait_shell(self):
507       """ Méthode booléenne qui retourne 1 si l'objet pointé par self
508       attend un shell, 0 sinon """
509       if 'shell' in self.object.definition.type:
510           return 1
511       else:
512           return 0
513
514   def wait_assd(self):
515       """Méthode booléenne qui retourne 1 si l'objet pointé par self
516       attend un objet de type ASSD ou dérivé, 0 sinon """
517       return self.object.wait_assd()
518     
519   def GetType(self):
520       """ 
521           Retourne le type de valeur attendu par l'objet représenté par l'item.
522       """
523       return self.object.get_type()
524
525   #-----------------------------------------------------
526   #
527   # Methodes liees  a l evaluation de la valeur saisie
528   #
529   #-----------------------------------------------------
530   # eval_valeur
531   # eval_valeur_item
532   # is_CO
533   # traite_reel
534
535   def eval_valeur(self,valeur):
536       """ Lance l'interprétation de 'valeur' (chaîne de caractères) comme valeur
537       de l'objet pointé par self :
538         - retourne l'objet associé si on a pu interpréter (entier, réel, ASSD,...)
539         - retourne 'valeur' (chaîne de caractères) sinon
540         - retourne None en cas d invalidite
541         - retourne invalide si 1 des objets du tuple l est
542       """
543       validite=1
544       if type(valeur) in (types.ListType,types.TupleType) :
545          valeurretour=[]
546          for item in valeur :
547              newvaleur,validiteitem=self.eval_valeur_item(item)
548              valeurretour.append(newvaleur)
549              if validiteitem == 0:
550                 validite=0
551       else :
552          valeurretour,validite= self.eval_valeur_item(valeur)
553       if validite == 0 :
554          valeurretour = None
555       return valeurretour,validite
556
557   def eval_valeur_item(self,valeur):
558       """ Lance l'interprétation de 'valeur' qui doit ne pas etre un tuple 
559           - va retourner la valeur de retour et la validite
560             selon le type de l objet attendu
561           - traite les reels et les parametres 
562       """ 
563       #print "eval_valeur_item",valeur
564       if valeur==None or valeur == "" :
565          return None,0
566       validite=1
567       if self.wait_reel():
568              valeurinter = self.traite_reel(valeur)
569              valeurretour,validite= self.object.eval_valeur(valeurinter)
570       elif self.wait_geom():
571              valeurretour,validite = valeur,1
572       else :
573              valeurretour,validite= self.object.eval_valeur(valeur)
574       #print "eval_valeur_item",valeurretour,validite
575
576       if validite == 0:
577          if type(valeur) == types.StringType and self.object.wait_TXM():
578             essai_valeur="'" + valeur + "'"
579             valeurretour,validite= self.object.eval_valeur(essai_valeur)
580
581       if hasattr(valeurretour,'__class__'):
582          #if valeurretour.__class__.__name__ in ('PARAMETRE','PARAMETRE_EVAL'):
583          if valeurretour.__class__.__name__ in ('PARAMETRE',):
584             validite=1
585
586       #if self.wait_co():
587          # CCAR : il ne faut pas essayer de creer un concept
588          # il faut simplement en chercher un existant ce qui a du etre fait par self.object.eval_valeur(valeur)
589          #try:
590             #valeurretour=Accas.CO(valeur)
591          #except:
592             #valeurretour=None
593             #validite=0
594       # on est dans le cas où on a évalué et où on n'aurait pas du
595       if self.object.wait_TXM() :
596           if type(valeurretour) != types.StringType:
597              valeurretour=str(valeur)
598              validite=1
599       return valeurretour,validite
600       
601   def is_CO(self,valeur=None):
602       """
603          Indique si valeur est un concept produit de la macro
604          Cette méthode n'a de sens que pour un MCSIMP d'une MACRO
605          Si valeur vaut None on teste la valeur du mot cle
606       """
607       # Pour savoir si un concept est un nouveau concept de macro
608       # on regarde s'il est présent dans l'attribut sdprods de l'étape
609       # ou si son nom de classe est CO.
610       # Il faut faire les 2 tests car une macro non valide peut etre
611       # dans un etat pas tres catholique avec des CO pas encore types
612       # et donc pas dans sdprods (resultat d'une exception dans type_sdprod)
613       if not valeur:valeur=self.object.valeur
614       if valeur in self.object.etape.sdprods:return 1
615       if type(valeur) is not types.InstanceType:return 0
616       if valeur.__class__.__name__ == 'CO':return 1
617       return 0
618
619   def is_param(self,valeur) :
620       for param in self.jdc.params:
621           if (repr(param) == valeur):
622              return 1
623       return 0
624
625   def traite_reel(self,valeur):
626       """
627       Cette fonction a pour but de rajouter le '.' en fin de chaîne pour un réel
628       ou de détecter si on fait référence à un concept produit par DEFI_VALEUR
629       ou un EVAL ...
630       """
631       valeur = string.strip(valeur)
632       liste_reels = self.get_sd_avant_du_bon_type()
633       if valeur in liste_reels:
634           return valeur
635       if len(valeur) >= 3 :
636           if valeur[0:4] == 'EVAL' :
637               # on a trouvé un EVAL --> on retourne directement la valeur
638               return valeur
639       if string.find(valeur,'.') == -1 :
640           # aucun '.' n'a été trouvé dans valeur --> on en rajoute un à la fin
641           if (self.is_param(valeur)):
642               return valeur
643           else:
644               if string.find(valeur,'e') != -1:
645                  # Notation scientifique ?
646                  try :
647                     r=eval(valeur)
648                     return valeur
649                  except :
650                     return None
651               else :
652                  return valeur+'.'
653       else:
654           return valeur
655         
656
657 import Accas
658 treeitem = SIMPTreeItem
659 objet = Accas.MCSIMP
660