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