Salome HOME
PN
[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 update_condition_bloc(self):
266       """ Met a jour les blocs conditionnels dependant du mot cle simple self
267       """
268       if self.definition.position == 'global' : 
269          self.etape.deep_update_condition_bloc()
270       elif self.definition.position == 'global_jdc' :
271          self.jdc.deep_update_condition_bloc()
272       else:
273          self.parent.update_condition_bloc()
274
275   def set_valeur(self,new_valeur,evaluation='oui'):
276         #print "set_valeur",new_valeur
277         self.init_modif()
278         self.valeur = new_valeur
279         self.val = new_valeur
280         self.update_condition_bloc()
281         self.fin_modif()
282         return 1
283
284   def eval_valeur(self,new_valeur):
285     """
286         Essaie d'évaluer new_valeur comme une SD, une déclaration Python 
287         ou un EVAL: Retourne la valeur évaluée (ou None) et le test de réussite (1 ou 0)
288     """
289     sd = self.jdc.get_sd_avant_etape(new_valeur,self.etape)
290     #sd = self.jdc.get_contexte_avant(self.etape).get(new_valeur,None)
291     #print sd
292     if sd :
293       return sd,1
294     lsd = self.jdc.cherche_list_avant(self.etape,new_valeur) 
295     if lsd :
296       return lsd,1
297     else:
298       d={}
299       # On veut EVAL avec tous ses comportements. On utilise Accas. Perfs ??
300       d['EVAL']=Accas.EVAL
301       try :
302         objet = eval(new_valeur,d)
303         return objet,1
304       except Exception:
305         itparam=self.cherche_item_parametre(new_valeur)
306         if itparam:
307              return itparam,1
308         try :
309              object=eval(new_valeur.valeur,d)
310         except :
311              pass
312         if CONTEXT.debug : traceback.print_exc()
313         return None,0
314
315   def cherche_item_parametre (self,new_valeur):
316         try:
317           nomparam=new_valeur[0:new_valeur.find("[")]
318           indice=new_valeur[new_valeur.find("[")+1:new_valeur.find("]")]
319           for p in self.jdc.params:
320              if p.nom == nomparam :
321                 if int(indice) < len(p.get_valeurs()):
322                    itparam=parametre.ITEM_PARAMETRE(p,int(indice))
323                    return itparam
324           return None
325         except:
326           return None
327
328   def update_concept(self,sd):
329     if type(self.valeur) in (types.ListType,types.TupleType) :
330        if sd in self.valeur:self.fin_modif()
331     else:
332        if sd == self.valeur:self.fin_modif()
333
334   def delete_concept(self,sd):
335     """ 
336         Inputs :
337            - sd=concept detruit
338         Fonction :
339         Met a jour la valeur du mot cle simple suite à la disparition 
340         du concept sd
341     """
342     #print "delete_concept",sd
343     if type(self.valeur) == types.TupleType :
344       if sd in self.valeur:
345         self.init_modif()
346         self.valeur=list(self.valeur)
347         self.valeur.remove(sd)
348         self.fin_modif()
349     elif type(self.valeur) == types.ListType:
350       if sd in self.valeur:
351         self.init_modif()
352         self.valeur.remove(sd)
353         self.fin_modif()
354     else:
355       if self.valeur == sd:
356         self.init_modif()
357         self.valeur=None
358         self.val=None
359         self.fin_modif()
360
361   def replace_concept(self,old_sd,sd):
362     """
363         Inputs :
364            - old_sd=concept remplacé
365            - sd=nouveau concept
366         Fonction :
367         Met a jour la valeur du mot cle simple suite au remplacement 
368         du concept old_sd
369     """
370     #print "replace_concept",old_sd,sd
371     if type(self.valeur) == types.TupleType :
372       if old_sd in self.valeur:
373         self.init_modif()
374         self.valeur=list(self.valeur)
375         i=self.valeur.index(old_sd)
376         self.valeur[i]=sd
377         self.fin_modif()
378     elif type(self.valeur) == types.ListType:
379       if old_sd in self.valeur:
380         self.init_modif()
381         i=self.valeur.index(old_sd)
382         self.valeur[i]=sd
383         self.fin_modif()
384     else:
385       if self.valeur == old_sd:
386         self.init_modif()
387         self.valeur=sd
388         self.val=sd
389         self.fin_modif()
390
391   def set_valeur_co(self,nom_co):
392       """
393           Affecte à self l'objet de type CO et de nom nom_co
394       """
395       #print "set_valeur_co",nom_co
396       step=self.etape.parent
397       if nom_co == None or nom_co == '':
398          new_objet=None
399       else:
400          # Avant de créer un concept il faut s'assurer du contexte : step 
401          # courant
402          sd= step.get_sd_autour_etape(nom_co,self.etape,avec='oui')
403          if sd:
404             # Si un concept du meme nom existe deja dans la portée de l'étape
405             # on ne crée pas le concept
406             return 0,"un concept de meme nom existe deja"
407          # Il n'existe pas de concept de meme nom. On peut donc le créer 
408          # Il faut néanmoins que la méthode NommerSdProd de step gère les 
409          # contextes en mode editeur
410          # Normalement la méthode  du Noyau doit etre surchargée
411          # On déclare l'étape du mot clé comme etape courante pour NommerSdprod
412          cs= CONTEXT.get_current_step()
413          CONTEXT.unset_current_step()
414          CONTEXT.set_current_step(step)
415          step.set_etape_context(self.etape)
416          new_objet = Accas.CO(nom_co)
417          CONTEXT.unset_current_step()
418          CONTEXT.set_current_step(cs)
419       self.init_modif()
420       self.valeur = new_objet
421       self.val = new_objet
422       self.fin_modif()
423       step.reset_context()
424       # On force l'enregistrement de new_objet en tant que concept produit 
425       # de la macro en appelant get_type_produit avec force=1
426       self.etape.get_type_produit(force=1)
427       #print "set_valeur_co",new_objet
428       return 1,"Concept créé"
429         
430   def verif_existence_sd(self):
431      """
432         Vérifie que les structures de données utilisées dans self existent bien dans le contexte
433         avant étape, sinon enlève la référence à ces concepts
434      """
435      #print "verif_existence_sd"
436      # Attention : possible probleme avec include
437      l_sd_avant_etape = self.jdc.get_contexte_avant(self.etape).values()  
438      if type(self.valeur) in (types.TupleType,types.ListType) :
439        l=[]
440        self.init_modif()
441        for sd in self.valeur:
442          if isinstance(sd,ASSD) :
443             if sd in l_sd_avant_etape :
444                l.append(sd)
445          else:
446             l.append(sd)
447        self.valeur=tuple(l)
448        self.fin_modif()
449      else:
450        if isinstance(self.valeur,ASSD) :
451           if self.valeur not in l_sd_avant_etape :
452              self.init_modif()
453              self.valeur = None
454              self.fin_modif()
455  
456   def get_min_max(self):
457      """
458      Retourne les valeurs min et max admissibles pour la valeur de self
459      """
460      return self.definition.min,self.definition.max
461
462
463   def get_type(self):
464      """
465      Retourne le type attendu par le mot-clé simple
466      """
467      return self.definition.type
468
469   def delete_mc_global(self):
470       """ Retire self des declarations globales
471       """
472       if self.definition.position == 'global' : 
473          etape = self.get_etape()
474          if etape :
475             del etape.mc_globaux[self.nom]
476       elif self.definition.position == 'global_jdc' :
477          del self.jdc.mc_globaux[self.nom]
478
479   def update_mc_global(self):
480      """
481         Met a jour les mots cles globaux enregistrés dans l'étape parente
482         et dans le jdc parent.
483         Un mot cle simple peut etre global. 
484      """
485      if self.definition.position == 'global' :
486         etape = self.get_etape()
487         if etape :
488            etape.mc_globaux[self.nom]=self
489      elif self.definition.position == 'global_jdc' :
490         if self.jdc:
491            self.jdc.mc_globaux[self.nom]=self
492
493 #--------------------------------------------------------------------------------
494  
495 #ATTENTION SURCHARGE : toutes les methodes ci apres sont des surcharges du Noyau et de Validation
496 # Elles doivent etre reintegrees des que possible
497
498
499   def isvalid(self,cr='non'):
500       """
501          Cette méthode retourne un indicateur de validité de l'objet de type MCSIMP
502
503            - 0 si l'objet est invalide
504            - 1 si l'objet est valide
505
506          Le paramètre cr permet de paramétrer le traitement. Si cr == 'oui'
507          la méthode construit également un comte-rendu de validation
508          dans self.cr qui doit avoir été créé préalablement.
509       """
510       if self.state == 'unchanged':
511         return self.valid
512       else:
513         valid = 1
514         v=self.valeur
515         #  verification presence
516         if self.isoblig() and v == None :
517           if cr == 'oui' :
518             self.cr.fatal(string.join(("Mot-clé : ",self.nom," obligatoire non valorisé")))
519           valid = 0
520
521         if v is None:
522            valid=0
523            if cr == 'oui' :
524               self.cr.fatal("None n'est pas une valeur autorisée")
525         else:
526            # type,into ...
527            if v.__class__.__name__ in ('PARAMETRE' , 'EVAL', 'ITEM_PARAMETRE','PARAMETRE_EVAL'):
528               verif_type=1
529            else:
530               verif_type=self.verif_type(val=v,cr=None)
531               # cas des tuples avec un ITEM_PARAMETRE
532               if verif_type == 0:
533                  if type(v) == types.TupleType :
534                    new_val=[]
535                    for i in v:
536                      if i.__class__.__name__  not in ('PARAMETRE','EVAL', 'ITEM_PARAMETRE','PARAMETRE_EVAL'): 
537                           new_val.append(i)
538                    if new_val != [] :
539                      verif_type=self.verif_type(val=new_val,cr=cr)
540                    else :
541                      # Cas d une liste de paramétre
542                      verif_type= 1
543                  else:
544                      verif_type=self.verif_type(val=v,cr=cr)
545            valid = verif_type*self.verif_into(cr=cr)*self.verif_card(cr=cr)
546            #
547            # On verifie les validateurs s'il y en a et si necessaire (valid == 1)
548            #
549            if valid and self.definition.validators and not self.definition.validators.verif(self.valeur):
550               if cr == 'oui' :
551                  self.cr.fatal(string.join(("Mot-clé : ",self.nom,"devrait avoir ",self.definition.validators.info())))
552               valid=0
553            # fin des validateurs
554            #
555         # cas d un item Parametre
556         if self.valeur.__class__.__name__ == 'ITEM_PARAMETRE':
557            valid=self.valeur.isvalid()
558            if valid == 0:
559               if cr == 'oui' :
560                  self.cr.fatal(string.join( repr (self.valeur), " a un indice incorrect"))
561
562         self.set_valid(valid)
563         return self.valid
564
565
566   def verif_typeihm(self,val,cr='non'):
567       try :
568          self.val.eval()
569          return 1
570       except :
571         pass
572       return self.verif_type(val,cr)
573
574   def verif_typeliste(self,val,cr='non') :
575       verif=0
576       for v in val :
577         verif=verif+self.verif_typeihm(v,cr)
578       return verif
579