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