Salome HOME
pour tenir compte des LASSD
[tools/eficas.git] / Ihm / I_MCSIMP.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 import types,string
22 import traceback
23 from copy import copy
24 from repr import Repr
25 myrepr = Repr()
26 myrepr.maxstring = 100
27 myrepr.maxother = 100
28
29 from Noyau.N_utils import repr_float
30
31 # Attention : les classes ASSD,.... peuvent etre surchargées
32 # dans le package Accas. Il faut donc prendre des précautions si
33 # on utilise les classes du Noyau pour faire des tests (isxxxx, ...)
34 # Si on veut créer des objets comme des CO avec les classes du noyau
35 # ils n'auront pas les conportements des autres packages (pb!!!)
36 # Il vaut mieux les importer d'Accas mais problème d'import circulaire,
37 # on ne peut pas les importer au début.
38 # On fait donc un import local quand c'est nécessaire (peut occasionner
39 # des pbs de prformance).
40 from Noyau.N_ASSD import ASSD,assd
41 from Noyau.N_GEOM import GEOM,geom
42 from Noyau.N_CO import CO
43 import Accas
44 # fin attention
45
46 from Extensions import parametre
47 import I_OBJECT
48 import CONNECTOR
49
50 class MCSIMP(I_OBJECT.OBJECT):
51
52   def GetNomConcept(self):
53       p=self
54       while p.parent :
55          try :
56             nomconcept=p.get_sdname()
57             return nomconcept
58          except:
59             try :
60                nomconcept= p.object.get_sdname()
61                return nomconcept
62             except :
63                pass
64          p=p.parent
65       return ""
66
67   def GetText(self):
68     """
69         Retourne le texte à afficher dans l'arbre représentant la valeur de l'objet
70         pointé par self
71     """
72
73     if self.valeur == None : 
74       return None
75     elif type(self.valeur) == types.FloatType : 
76       # Traitement d'un flottant isolé
77       # txt = repr_float(self.valeur)
78       # Normalement str fait un travail correct
79       txt = str(self.valeur)
80       clefobj=self.GetNomConcept()
81       if self.jdc.appli.dict_reels.has_key(clefobj):
82         if self.jdc.appli.dict_reels[clefobj].has_key(self.valeur):
83            txt=self.jdc.appli.dict_reels[clefobj][self.valeur]
84     elif type(self.valeur) in (types.ListType,types.TupleType) :
85       # Traitement des listes
86       txt='('
87       i=0
88       for val in self.valeur:
89         if type(val) == types.FloatType : 
90            # CCAR : Normalement str fait un travail correct
91            #txt=txt + i*',' + repr_float(val)
92            clefobj=self.GetNomConcept()
93            if self.jdc.appli.dict_reels.has_key(clefobj):
94               if self.jdc.appli.dict_reels[clefobj].has_key(val):
95                  txt=txt + i*',' +self.jdc.appli.dict_reels[clefobj][val]
96               else :
97                  txt=txt + i*',' + str(val)
98            else :
99               txt=txt + i*',' + str(val)
100         elif isinstance(val,ASSD): 
101            txt = txt + i*',' + val.get_name()
102     #PN
103     # ajout du elif
104         elif type(val) == types.InstanceType and val.__class__.__name__ in  ('PARAMETRE','PARAMETRE_EVAL'):
105            txt = txt + i*','+ str(val) 
106         else: 
107            txt = txt + i*','+ myrepr.repr(val)
108         i=1
109       txt=txt+')'
110     elif isinstance(self.valeur,ASSD): 
111       # Cas des ASSD
112       txt=self.getval()
113     elif type(self.valeur) == types.InstanceType and self.valeur.__class__.__name__ in  ('PARAMETRE','PARAMETRE_EVAL'):
114       # Cas des PARAMETRES
115       txt=str(self.valeur)
116     else:
117       # Traitement des autres cas
118       txt = myrepr.repr(self.valeur)
119
120     # txt peut etre une longue chaine sur plusieurs lignes.
121     # Il est possible de tronquer cette chaine au premier \n et 
122     # de limiter la longueur de la chaine a 30 caracteres. Cependant
123     # ceci provoque une perte d'information pour l'utilisateur
124     # Pour le moment on retourne la chaine telle que
125     return txt
126
127   def getval(self):
128     """ 
129        Retourne une chaîne de caractère représentant la valeur de self 
130     """
131     val=self.valeur
132     if type(val) == types.FloatType : 
133       clefobj=self.GetNomConcept()
134       if self.jdc.appli.dict_reels.has_key(clefobj):
135         if self.jdc.appli.dict_reels[clefobj].has_key(val):
136            return self.jdc.appli.dict_reels[clefobj][val]
137     if type(val) != types.TupleType :
138       try:
139         return val.get_name()
140       except:
141         return val
142     else :
143       s='( '
144       for item in val :
145         try :
146           s=s+item.get_name()+','
147         except:
148           s=s+`item`+','
149       s=s+' )'
150       return s
151
152   def wait_co(self):
153     """
154         Méthode booléenne qui retourne 1 si l'objet attend un objet ASSD 
155         qui n'existe pas encore (type CO()), 0 sinon
156     """
157     for typ in self.definition.type:
158       if type(typ) == types.ClassType :
159         if issubclass(typ,CO) :
160            return 1
161     return 0
162
163   def wait_assd(self):
164     """ 
165         Méthode booléenne qui retourne 1 si le MCS attend un objet de type ASSD 
166         ou dérivé, 0 sinon
167     """
168     for typ in self.definition.type:
169       if type(typ) == types.ClassType :
170         if issubclass(typ,ASSD) and not issubclass(typ,GEOM):
171           return 1
172     return 0
173
174   def wait_assd_or_geom(self):
175     """ 
176          Retourne 1 si le mot-clé simple attend un objet de type
177          assd, ASSD, geom ou GEOM
178          Retourne 0 dans le cas contraire
179     """
180     for typ in self.definition.type:
181       if type(typ) == types.ClassType :
182         if typ.__name__ in ("GEOM","ASSD","geom","assd") or issubclass(typ,GEOM) :
183           return 1
184     return 0
185
186   def wait_geom(self):
187     """ 
188          Retourne 1 si le mot-clé simple attend un objet de type GEOM
189          Retourne 0 dans le cas contraire
190     """
191     for typ in self.definition.type:
192       if type(typ) == types.ClassType :
193         if issubclass(typ,GEOM) : return 1
194     return 0
195
196   def wait_TXM(self):
197     """ 
198          Retourne 1 si le mot-clé simple attend un objet de type TXM
199          Retourne 0 dans le cas contraire
200     """
201     for typ in self.definition.type:
202       if typ == 'TXM' :return 1
203     return 0
204
205   def get_liste_valeurs(self):
206     """
207     """
208     if self.valeur == None:
209       return []
210     elif type(self.valeur) == types.TupleType:
211       return list(self.valeur)
212     elif type(self.valeur) == types.ListType:
213       return self.valeur
214     else:
215       return [self.valeur]
216
217   def isoblig(self):
218     return self.definition.statut=='o'
219
220   def valid_valeur(self,new_valeur):
221       """
222         Verifie que la valeur passee en argument (new_valeur) est valide
223         sans modifier la valeur courante (evite d'utiliser set_valeur et est plus performant)
224       """
225       old_valeur=self.valeur
226       old_val=self.val
227       self.valeur = new_valeur
228       self.val = new_valeur
229       self.state="modified"
230       validite=self.isvalid()
231       self.valeur = old_valeur
232       self.val = old_valeur
233       self.state="modified"
234       self.isvalid()
235       return validite
236
237   def valid_valeur_partielle(self,new_valeur):
238       """
239         Verifie que la valeur passee en argument (new_valeur) est partiellement valide
240         sans modifier la valeur courante (evite d'utiliser set_valeur et est plus performant)
241       """
242       old_valeur=self.valeur
243       old_val=self.val
244
245       self.valeur = new_valeur
246       self.val = new_valeur
247       self.state="modified"
248       validite=0
249       if self.isvalid():
250          validite=1
251       elif self.definition.validators :
252          validite=self.definition.validators.valide_liste_partielle(new_valeur)
253
254       if validite==0:
255          min,max=self.get_min_max()
256          if len(new_valeur) < min :
257             validite=1
258
259       self.valeur = old_valeur
260       self.val = old_valeur
261       self.state="modified"
262       self.isvalid()
263       return validite
264
265   def set_valeur(self,new_valeur,evaluation='oui'):
266         #print "set_valeur",new_valeur
267         self.init_modif()
268         self.valeur = new_valeur
269         self.val = new_valeur
270         if self.definition.position == 'global' : 
271            self.etape.deep_update_condition_bloc()
272         elif self.definition.position == 'global_jdc' :
273            self.jdc.deep_update_condition_bloc()
274         else:
275            self.parent.update_condition_bloc()
276         self.fin_modif()
277         return 1
278
279   def eval_valeur(self,new_valeur):
280     """
281         Essaie d'évaluer new_valeur comme une SD, une déclaration Python 
282         ou un EVAL: Retourne la valeur évaluée (ou None) et le test de réussite (1 ou 0)
283     """
284     sd = self.jdc.get_sd_avant_etape(new_valeur,self.etape)
285     #sd = self.jdc.get_contexte_avant(self.etape).get(new_valeur,None)
286     #print sd
287     if sd :
288       return sd,1
289     lsd = self.jdc.cherche_list_avant(self.etape,new_valeur) 
290     if lsd :
291       return lsd,1
292     else:
293       d={}
294       # On veut EVAL avec tous ses comportements. On utilise Accas. Perfs ??
295       d['EVAL']=Accas.EVAL
296       try :
297         objet = eval(new_valeur,d)
298         return objet,1
299       except Exception:
300         itparam=self.cherche_item_parametre(new_valeur)
301         if itparam:
302              return itparam,1
303         if CONTEXT.debug : traceback.print_exc()
304         return None,0
305
306   def cherche_item_parametre (self,new_valeur):
307         try:
308           nomparam=new_valeur[0:new_valeur.find("[")]
309           indice=new_valeur[new_valeur.find("[")+1:new_valeur.find("]")]
310           for p in self.jdc.params:
311              if p.nom == nomparam :
312                 if int(indice) < len(p.get_valeurs()):
313                    itparam=parametre.ITEM_PARAMETRE(p,int(indice))
314                    return itparam
315           return None
316         except:
317           return None
318
319   def update_concept(self,sd):
320     if type(self.valeur) in (types.ListType,types.TupleType) :
321        if sd in self.valeur:self.fin_modif()
322     else:
323        if sd == self.valeur:self.fin_modif()
324
325   def delete_concept(self,sd):
326     """ 
327         Inputs :
328            - sd=concept detruit
329         Fonction :
330         Met a jour la valeur du mot cle simple suite à la disparition 
331         du concept sd
332     """
333     #print "delete_concept",sd
334     if type(self.valeur) == types.TupleType :
335       if sd in self.valeur:
336         self.init_modif()
337         self.valeur=list(self.valeur)
338         self.valeur.remove(sd)
339         self.fin_modif()
340     elif type(self.valeur) == types.ListType:
341       if sd in self.valeur:
342         self.init_modif()
343         self.valeur.remove(sd)
344         self.fin_modif()
345     else:
346       if self.valeur == sd:
347         self.init_modif()
348         self.valeur=None
349         self.val=None
350         self.fin_modif()
351
352   def replace_concept(self,old_sd,sd):
353     """
354         Inputs :
355            - old_sd=concept remplacé
356            - sd=nouveau concept
357         Fonction :
358         Met a jour la valeur du mot cle simple suite au remplacement 
359         du concept old_sd
360     """
361     #print "replace_concept",old_sd,sd
362     if type(self.valeur) == types.TupleType :
363       if old_sd in self.valeur:
364         self.init_modif()
365         self.valeur=list(self.valeur)
366         i=self.valeur.index(old_sd)
367         self.valeur[i]=sd
368         self.fin_modif()
369     elif type(self.valeur) == types.ListType:
370       if old_sd in self.valeur:
371         self.init_modif()
372         i=self.valeur.index(old_sd)
373         self.valeur[i]=sd
374         self.fin_modif()
375     else:
376       if self.valeur == old_sd:
377         self.init_modif()
378         self.valeur=sd
379         self.val=sd
380         self.fin_modif()
381
382   def set_valeur_co(self,nom_co):
383       """
384           Affecte à self l'objet de type CO et de nom nom_co
385       """
386       #print "set_valeur_co",nom_co
387       step=self.etape.parent
388       if nom_co == None or nom_co == '':
389          new_objet=None
390       else:
391          # Avant de créer un concept il faut s'assurer du contexte : step 
392          # courant
393          sd= step.get_sd_autour_etape(nom_co,self.etape,avec='oui')
394          if sd:
395             # Si un concept du meme nom existe deja dans la portée de l'étape
396             # on ne crée pas le concept
397             return 0,"un concept de meme nom existe deja"
398          # Il n'existe pas de concept de meme nom. On peut donc le créer 
399          # Il faut néanmoins que la méthode NommerSdProd de step gère les 
400          # contextes en mode editeur
401          # Normalement la méthode  du Noyau doit etre surchargée
402          # On déclare l'étape du mot clé comme etape courante pour NommerSdprod
403          cs= CONTEXT.get_current_step()
404          CONTEXT.unset_current_step()
405          CONTEXT.set_current_step(step)
406          step.set_etape_context(self.etape)
407          new_objet = Accas.CO(nom_co)
408          CONTEXT.unset_current_step()
409          CONTEXT.set_current_step(cs)
410       self.init_modif()
411       self.valeur = new_objet
412       self.val = new_objet
413       self.fin_modif()
414       step.reset_context()
415       # On force l'enregistrement de new_objet en tant que concept produit 
416       # de la macro en appelant get_type_produit avec force=1
417       self.etape.get_type_produit(force=1)
418       #print "set_valeur_co",new_objet
419       return 1,"Concept créé"
420         
421   def verif_existence_sd(self):
422      """
423         Vérifie que les structures de données utilisées dans self existent bien dans le contexte
424         avant étape, sinon enlève la référence à ces concepts
425      """
426      #print "verif_existence_sd"
427      # Attention : possible probleme avec include
428      l_sd_avant_etape = self.jdc.get_contexte_avant(self.etape).values()  
429      if type(self.valeur) in (types.TupleType,types.ListType) :
430        l=[]
431        self.init_modif()
432        for sd in self.valeur:
433          if isinstance(sd,ASSD) :
434             if sd in l_sd_avant_etape :
435                l.append(sd)
436          else:
437             l.append(sd)
438        self.valeur=tuple(l)
439        self.fin_modif()
440      else:
441        if isinstance(self.valeur,ASSD) :
442           if self.valeur not in l_sd_avant_etape :
443              self.init_modif()
444              self.valeur = None
445              self.fin_modif()
446  
447   def get_min_max(self):
448      """
449      Retourne les valeurs min et max admissibles pour la valeur de self
450      """
451      return self.definition.min,self.definition.max
452
453
454   def get_type(self):
455      """
456      Retourne le type attendu par le mot-clé simple
457      """
458      return self.definition.type
459
460 #--------------------------------------------------------------------------------
461 # PN : ajout pour Salome des methodes suivantes (jusqu aux méthodes surchargees)
462 #--------------------------------------------------------------------------------
463   def get_salome_valeurs(self):
464        l=[]
465        if not hasattr(self,'list_salome_valeurs'):
466            self.list_salome_valeurs=[]
467        if self.list_salome_valeurs != [] :
468            for val in self.list_salome_valeurs:
469                 l.append(val)
470        return l
471
472   def put_salome_valeurs(self,list):
473        self.list_salome_valeurs=[]
474        for val in list:
475            self.list_salome_valeurs.append(val)
476
477   def add_salome_valeurs(self,val):
478       if not hasattr(self,'list_salome_valeurs'):
479            self.list_salome_valeurs=[]
480       try:
481            self.list_salome_valeurs.append(val)
482       except :
483            try:
484               for uneval in val :
485                   self.list_salome_valeurs.append(uneval)
486            except :
487               pass
488
489   def has_salome_valeurs(self):
490       if not hasattr(self,'list_salome_valeurs'):
491            self.list_salome_valeurs=[]
492       if self.list_salome_valeurs != []:
493            return true
494       else:
495            return false
496
497 #--------------------------------------------------------------------------------
498 # PN : fin ajout pour Salome 
499 #--------------------------------------------------------------------------------
500  
501 #ATTENTION SURCHARGE : toutes les methodes ci apres sont des surcharges du Noyau et de Validation
502 # Elles doivent etre reintegrees des que possible
503
504
505   def isvalid(self,cr='non'):
506       """
507          Cette méthode retourne un indicateur de validité de l'objet de type MCSIMP
508
509            - 0 si l'objet est invalide
510            - 1 si l'objet est valide
511
512          Le paramètre cr permet de paramétrer le traitement. Si cr == 'oui'
513          la méthode construit également un comte-rendu de validation
514          dans self.cr qui doit avoir été créé préalablement.
515       """
516       if self.state == 'unchanged':
517         return self.valid
518       else:
519         valid = 1
520         v=self.valeur
521         #  verification presence
522         if self.isoblig() and v == None :
523           if cr == 'oui' :
524             self.cr.fatal(string.join(("Mot-clé : ",self.nom," obligatoire non valorisé")))
525           valid = 0
526
527         if v is None:
528            valid=0
529            if cr == 'oui' :
530               self.cr.fatal("None n'est pas une valeur autorisée")
531         else:
532            # type,into ...
533            if v.__class__.__name__ in ('PARAMETRE' , 'EVAL', 'ITEM_PARAMETRE','PARAMETRE_EVAL'):
534               verif_type=1
535            else:
536               verif_type=self.verif_type(val=v,cr=None)
537               # cas des tuples avec un ITEM_PARAMETRE
538               if verif_type == 0:
539                  if type(v) == types.TupleType :
540                    new_val=[]
541                    for i in v:
542                      if i.__class__.__name__  not in ('PARAMETRE','EVAL', 'ITEM_PARAMETRE','PARAMETRE_EVAL'): 
543                           new_val.append(i)
544                    if new_val != [] :
545                      verif_type=self.verif_type(val=new_val,cr=cr)
546                    else :
547                      # Cas d une liste de paramétre
548                      verif_type= 1
549                  else:
550                      verif_type=self.verif_type(val=v,cr=cr)
551            valid = verif_type*self.verif_into(cr=cr)*self.verif_card(cr=cr)
552            #
553            # On verifie les validateurs s'il y en a et si necessaire (valid == 1)
554            #
555            if valid and self.definition.validators and not self.definition.validators.verif(self.valeur):
556               if cr == 'oui' :
557                  self.cr.fatal(string.join(("Mot-clé : ",self.nom,"devrait avoir ",self.definition.validators.info())))
558               valid=0
559            # fin des validateurs
560            #
561         # cas d un item Parametre
562         if self.valeur.__class__.__name__ == 'ITEM_PARAMETRE':
563            valid=self.valeur.isvalid()
564            if valid == 0:
565               if cr == 'oui' :
566                  self.cr.fatal(string.join( repr (self.valeur), " a un indice incorrect"))
567
568         self.set_valid(valid)
569         return self.valid