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.
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.
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.
20 # ======================================================================
26 myrepr.maxstring = 100
29 from Noyau.N_utils import repr_float
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
46 from Extensions import parametre
47 from Extensions import param2
51 class MCSIMP(I_OBJECT.OBJECT):
53 def GetNomConcept(self):
57 nomconcept=p.get_sdname()
61 nomconcept= p.object.get_sdname()
70 Retourne le texte à afficher dans l'arbre représentant la valeur de l'objet
74 if self.valeur == None :
76 elif type(self.valeur) == types.FloatType :
77 # Traitement d'un flottant isolé
78 # txt = repr_float(self.valeur)
79 # Normalement str fait un travail correct
80 txt = str(self.valeur)
81 clefobj=self.GetNomConcept()
82 if self.jdc.appli.dict_reels.has_key(clefobj):
83 if self.jdc.appli.dict_reels[clefobj].has_key(self.valeur):
84 txt=self.jdc.appli.dict_reels[clefobj][self.valeur]
85 elif type(self.valeur) in (types.ListType,types.TupleType) :
86 # Traitement des listes
89 for val in self.valeur:
90 if type(val) == types.FloatType :
91 # CCAR : Normalement str fait un travail correct
92 #txt=txt + i*',' + repr_float(val)
93 clefobj=self.GetNomConcept()
94 if self.jdc.appli.dict_reels.has_key(clefobj):
95 if self.jdc.appli.dict_reels[clefobj].has_key(val):
96 txt=txt + i*',' +self.jdc.appli.dict_reels[clefobj][val]
98 txt=txt + i*',' + str(val)
100 txt=txt + i*',' + str(val)
101 elif isinstance(val,ASSD):
102 txt = txt + i*',' + val.get_name()
105 elif type(val) == types.InstanceType and val.__class__.__name__ in ('PARAMETRE','PARAMETRE_EVAL'):
106 txt = txt + i*','+ str(val)
108 txt = txt + i*','+ myrepr.repr(val)
111 elif isinstance(self.valeur,ASSD):
114 elif type(self.valeur) == types.InstanceType and self.valeur.__class__.__name__ in ('PARAMETRE','PARAMETRE_EVAL'):
118 # Traitement des autres cas
119 txt = myrepr.repr(self.valeur)
121 # txt peut etre une longue chaine sur plusieurs lignes.
122 # Il est possible de tronquer cette chaine au premier \n et
123 # de limiter la longueur de la chaine a 30 caracteres. Cependant
124 # ceci provoque une perte d'information pour l'utilisateur
125 # Pour le moment on retourne la chaine telle que
130 Retourne une chaîne de caractère représentant la valeur de self
133 if type(val) == types.FloatType :
134 clefobj=self.GetNomConcept()
135 if self.jdc.appli.dict_reels.has_key(clefobj):
136 if self.jdc.appli.dict_reels[clefobj].has_key(val):
137 return self.jdc.appli.dict_reels[clefobj][val]
138 if type(val) != types.TupleType :
140 return val.get_name()
147 s=s+item.get_name()+','
155 Méthode booléenne qui retourne 1 si l'objet attend un objet ASSD
156 qui n'existe pas encore (type CO()), 0 sinon
158 for typ in self.definition.type:
159 if type(typ) == types.ClassType :
160 if issubclass(typ,CO) :
166 Méthode booléenne qui retourne 1 si le MCS attend un objet de type ASSD
169 for typ in self.definition.type:
170 if type(typ) == types.ClassType :
171 if issubclass(typ,ASSD) and not issubclass(typ,GEOM):
175 def wait_assd_or_geom(self):
177 Retourne 1 si le mot-clé simple attend un objet de type
178 assd, ASSD, geom ou GEOM
179 Retourne 0 dans le cas contraire
181 for typ in self.definition.type:
182 if type(typ) == types.ClassType :
183 if typ.__name__ in ("GEOM","ASSD","geom","assd") or issubclass(typ,GEOM) :
189 Retourne 1 si le mot-clé simple attend un objet de type GEOM
190 Retourne 0 dans le cas contraire
192 for typ in self.definition.type:
193 if type(typ) == types.ClassType :
194 if issubclass(typ,GEOM) : return 1
199 Retourne 1 si le mot-clé simple attend un objet de type TXM
200 Retourne 0 dans le cas contraire
202 for typ in self.definition.type:
203 if typ == 'TXM' :return 1
206 def get_liste_valeurs(self):
209 if self.valeur == None:
211 elif type(self.valeur) == types.TupleType:
212 return list(self.valeur)
213 elif type(self.valeur) == types.ListType:
219 return self.definition.statut=='o'
221 def valid_valeur(self,new_valeur):
223 Verifie que la valeur passee en argument (new_valeur) est valide
224 sans modifier la valeur courante (evite d'utiliser set_valeur et est plus performant)
226 old_valeur=self.valeur
228 self.valeur = new_valeur
229 self.val = new_valeur
230 self.state="modified"
231 validite=self.isvalid()
232 self.valeur = old_valeur
233 self.val = old_valeur
234 self.state="modified"
238 def valid_valeur_partielle(self,new_valeur):
240 Verifie que la valeur passee en argument (new_valeur) est partiellement valide
241 sans modifier la valeur courante (evite d'utiliser set_valeur et est plus performant)
243 old_valeur=self.valeur
246 self.valeur = new_valeur
247 self.val = new_valeur
248 self.state="modified"
252 elif self.definition.validators :
253 validite=self.definition.validators.valide_liste_partielle(new_valeur)
256 min,max=self.get_min_max()
257 if len(new_valeur) < min :
260 self.valeur = old_valeur
261 self.val = old_valeur
262 self.state="modified"
266 def update_condition_bloc(self):
267 """ Met a jour les blocs conditionnels dependant du mot cle simple self
269 if self.definition.position == 'global' :
270 self.etape.deep_update_condition_bloc()
271 elif self.definition.position == 'global_jdc' :
272 self.jdc.deep_update_condition_bloc()
274 self.parent.update_condition_bloc()
276 def set_valeur(self,new_valeur,evaluation='oui'):
277 #print "set_valeur",new_valeur
279 self.valeur = new_valeur
280 self.val = new_valeur
281 self.update_condition_bloc()
285 def eval_valeur(self,new_valeur):
287 Essaie d'évaluer new_valeur comme une SD, une déclaration Python
288 ou un EVAL: Retourne la valeur évaluée (ou None) et le test de réussite (1 ou 0)
290 sd = self.jdc.get_sd_avant_etape(new_valeur,self.etape)
291 #sd = self.jdc.get_contexte_avant(self.etape).get(new_valeur,None)
295 lsd = self.jdc.cherche_list_avant(self.etape,new_valeur)
300 # On veut EVAL avec tous ses comportements. On utilise Accas. Perfs ??
303 objet = eval(new_valeur,d)
306 itparam=self.cherche_item_parametre(new_valeur)
310 object=eval(new_valeur.valeur,d)
313 if CONTEXT.debug : traceback.print_exc()
316 def cherche_item_parametre (self,new_valeur):
318 nomparam=new_valeur[0:new_valeur.find("[")]
319 indice=new_valeur[new_valeur.find("[")+1:new_valeur.find("]")]
320 for p in self.jdc.params:
321 if p.nom == nomparam :
322 if int(indice) < len(p.get_valeurs()):
323 itparam=parametre.ITEM_PARAMETRE(p,int(indice))
329 def update_concept(self,sd):
330 if type(self.valeur) in (types.ListType,types.TupleType) :
331 if sd in self.valeur:self.fin_modif()
333 if sd == self.valeur:self.fin_modif()
335 def delete_concept(self,sd):
340 Met a jour la valeur du mot cle simple suite à la disparition
343 #print "delete_concept",sd
344 if type(self.valeur) == types.TupleType :
345 if sd in self.valeur:
347 self.valeur=list(self.valeur)
348 self.valeur.remove(sd)
350 elif type(self.valeur) == types.ListType:
351 if sd in self.valeur:
353 self.valeur.remove(sd)
356 if self.valeur == sd:
362 def replace_concept(self,old_sd,sd):
365 - old_sd=concept remplacé
368 Met a jour la valeur du mot cle simple suite au remplacement
371 #print "replace_concept",old_sd,sd
372 if type(self.valeur) == types.TupleType :
373 if old_sd in self.valeur:
375 self.valeur=list(self.valeur)
376 i=self.valeur.index(old_sd)
379 elif type(self.valeur) == types.ListType:
380 if old_sd in self.valeur:
382 i=self.valeur.index(old_sd)
386 if self.valeur == old_sd:
392 def set_valeur_co(self,nom_co):
394 Affecte à self l'objet de type CO et de nom nom_co
396 #print "set_valeur_co",nom_co
397 step=self.etape.parent
398 if nom_co == None or nom_co == '':
401 # Avant de créer un concept il faut s'assurer du contexte : step
403 sd= step.get_sd_autour_etape(nom_co,self.etape,avec='oui')
405 # Si un concept du meme nom existe deja dans la portée de l'étape
406 # on ne crée pas le concept
407 return 0,"un concept de meme nom existe deja"
408 # Il n'existe pas de concept de meme nom. On peut donc le créer
409 # Il faut néanmoins que la méthode NommerSdProd de step gère les
410 # contextes en mode editeur
411 # Normalement la méthode du Noyau doit etre surchargée
412 # On déclare l'étape du mot clé comme etape courante pour NommerSdprod
413 cs= CONTEXT.get_current_step()
414 CONTEXT.unset_current_step()
415 CONTEXT.set_current_step(step)
416 step.set_etape_context(self.etape)
417 new_objet = Accas.CO(nom_co)
418 CONTEXT.unset_current_step()
419 CONTEXT.set_current_step(cs)
421 self.valeur = new_objet
425 # On force l'enregistrement de new_objet en tant que concept produit
426 # de la macro en appelant get_type_produit avec force=1
427 self.etape.get_type_produit(force=1)
428 #print "set_valeur_co",new_objet
429 return 1,"Concept créé"
431 def verif_existence_sd(self):
433 Vérifie que les structures de données utilisées dans self existent bien dans le contexte
434 avant étape, sinon enlève la référence à ces concepts
436 #print "verif_existence_sd"
437 # Attention : possible probleme avec include
438 l_sd_avant_etape = self.jdc.get_contexte_avant(self.etape).values()
439 if type(self.valeur) in (types.TupleType,types.ListType) :
442 for sd in self.valeur:
443 if isinstance(sd,ASSD) :
444 if sd in l_sd_avant_etape :
451 if isinstance(self.valeur,ASSD) :
452 if self.valeur not in l_sd_avant_etape :
457 def get_min_max(self):
459 Retourne les valeurs min et max admissibles pour la valeur de self
461 return self.definition.min,self.definition.max
466 Retourne le type attendu par le mot-clé simple
468 return self.definition.type
470 def delete_mc_global(self):
471 """ Retire self des declarations globales
473 if self.definition.position == 'global' :
474 etape = self.get_etape()
476 del etape.mc_globaux[self.nom]
477 elif self.definition.position == 'global_jdc' :
478 del self.jdc.mc_globaux[self.nom]
480 def update_mc_global(self):
482 Met a jour les mots cles globaux enregistrés dans l'étape parente
483 et dans le jdc parent.
484 Un mot cle simple peut etre global.
486 if self.definition.position == 'global' :
487 etape = self.get_etape()
489 etape.mc_globaux[self.nom]=self
490 elif self.definition.position == 'global_jdc' :
492 self.jdc.mc_globaux[self.nom]=self
494 def nbrColonnes(self):
495 genea = self.get_genealogie()
496 if "VALE_C" in genea and "DEFI_FONCTION" in genea : return 3
497 if "VALE" in genea and "DEFI_FONCTION" in genea : return 2
501 #--------------------------------------------------------------------------------
503 #ATTENTION SURCHARGE : toutes les methodes ci apres sont des surcharges du Noyau et de Validation
504 # Elles doivent etre reintegrees des que possible
507 def isvalid(self,cr='non'):
509 Cette méthode retourne un indicateur de validité de l'objet de type MCSIMP
511 - 0 si l'objet est invalide
512 - 1 si l'objet est valide
514 Le paramètre cr permet de paramétrer le traitement. Si cr == 'oui'
515 la méthode construit également un comte-rendu de validation
516 dans self.cr qui doit avoir été créé préalablement.
518 if self.state == 'unchanged':
523 # verification presence
524 if self.isoblig() and v == None :
526 self.cr.fatal(string.join(("Mot-clé : ",self.nom," obligatoire non valorisé")))
532 self.cr.fatal("None n'est pas une valeur autorisée")
535 #PN ??? je n ose pas y toucher ???
536 #if v.__class__.__name__ in ('PARAMETRE','EVAL', 'ITEM_PARAMETRE','PARAMETRE_EVAL'):
537 if ((issubclass(v.__class__,param2.Formula)) or
538 (v.__class__.__name__ in ('EVAL', 'ITEM_PARAMETRE','PARAMETRE_EVAL'))):
539 verif_type=self.verif_typeihm(v)
541 verif_type=self.verif_type(val=v,cr=None)
542 # cas des tuples avec un ITEM_PARAMETRE
544 if type(v) == types.TupleType :
547 if ((issubclass(i.__class__,param2.Formula)) or
548 (i.__class__.__name__ in ('EVAL', 'ITEM_PARAMETRE','PARAMETRE_EVAL'))):
549 if self.verif_typeihm(val=i,cr=cr) == 0:
555 verif_type=self.verif_type(val=new_val,cr=cr)
557 # Cas d une liste de paramétre
558 verif_type=self.verif_typeliste(val=v,cr=cr)
560 verif_type=self.verif_type(val=v,cr=cr)
561 valid = verif_type*self.verif_into(cr=cr)*self.verif_card(cr=cr)
563 # On verifie les validateurs s'il y en a et si necessaire (valid == 1)
565 if valid and self.definition.validators and not self.definition.validators.verif(self.valeur):
567 self.cr.fatal(string.join(("Mot-clé : ",self.nom,"devrait avoir ",self.definition.validators.info())))
569 # fin des validateurs
571 # cas d un item Parametre
572 if self.valeur.__class__.__name__ == 'ITEM_PARAMETRE':
573 valid=self.valeur.isvalid()
576 self.cr.fatal(string.join( repr (self.valeur), " a un indice incorrect"))
578 self.set_valid(valid)
582 def verif_typeihm(self,val,cr='non'):
588 return self.verif_type(val,cr)
590 def verif_typeliste(self,val,cr='non') :
593 verif=verif+self.verif_typeihm(v,cr)