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