IdentifiantMot de passe
Loading...
Mot de passe oublié ?Je m'inscris ! (gratuit)
Navigation

Inscrivez-vous gratuitement
pour pouvoir participer, suivre les réponses en temps réel, voter pour les messages, poser vos propres questions et recevoir la newsletter

Langage Delphi Discussion :

rtti et procédure de classe


Sujet :

Langage Delphi

  1. #1
    Membre émérite

    Homme Profil pro
    Développeur informatique
    Inscrit en
    Novembre 2007
    Messages
    3 387
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 62
    Localisation : France, Rhône (Rhône Alpes)

    Informations professionnelles :
    Activité : Développeur informatique

    Informations forums :
    Inscription : Novembre 2007
    Messages : 3 387
    Points : 2 999
    Points
    2 999
    Par défaut rtti et procédure de classe
    Bonjour

    Dans un projet, j'ai de nombreuses TForm qui ont une procédure de classe nommée Initialise.
    Elle crée une instance de la TForm si elle n'existe pas encore et appelle ShowModal.
    Dans la TForm principale, je cherche à faire une procédure qui retrouve toutes les TForm qui implémente cette procédure, à l'exécuter puis à refermer la TForm.
    Pour retrouver les TForm en question je fais ce qui suit et qui me ramène bien les noms de classe.
    Là où je sèche, c'est comment appeler la méthode de classe dans la boucle et refermer ensuite la form qui a été créée.

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
     
    var
      aClass: TClass;
      context: TRttiContext;
      types: TArray<TRttiType>;
      aType: TRttiType;
      Mth: TRttiMethod;
    begin
      context := TRttiContext.Create;
      types := context.GetTypes;
      for aType in types do
      begin
        if aType.TypeKind = tkClass then
        begin
          aClass := aType.AsInstance.MetaclassType;
          if (aClass <> TForm) and aClass.InheritsFrom(TForm) then
          begin
            Mth := aType.GetMethod('Initialise');
            if Assigned(Mth) then
            begin
     
                //   ICI, je ne vois pas comment appeler la méthode à part Mth.Invoke(.....) mais que mettre comme paramètre ?
     
     
               // Target.Add(aClass.ClassName);  << Là je l'ajoutais à une liste.
            end;
          end;
        end;
      end;
    end;

  2. #2
    Expert éminent sénior
    Avatar de Paul TOTH
    Homme Profil pro
    Freelance
    Inscrit en
    Novembre 2002
    Messages
    8 964
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 54
    Localisation : France, Paris (Île de France)

    Informations professionnelles :
    Activité : Freelance
    Secteur : High Tech - Éditeur de logiciels

    Informations forums :
    Inscription : Novembre 2002
    Messages : 8 964
    Points : 28 445
    Points
    28 445
    Par défaut
    alors, premières chose qui me vient à l'esprit, c'est de créer une classe TInitForm qui possède une méthode de classe virtuelle Initialise...car tu pourrais alors simplement vérifier que l'instance est de type TInitForm et invoqué sa méthode plus facilement

    sinon il faut passer par http://docwiki.embarcadero.com/Libra...iMethod.Invoke
    Developpez.com: Mes articles, forum FlashPascal
    Entreprise: Execute SARL
    Le Store Excute Store

  3. #3
    Expert éminent sénior
    Avatar de ShaiLeTroll
    Homme Profil pro
    Développeur C++\Delphi
    Inscrit en
    Juillet 2006
    Messages
    13 459
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 43
    Localisation : France, Seine Saint Denis (Île de France)

    Informations professionnelles :
    Activité : Développeur C++\Delphi
    Secteur : High Tech - Éditeur de logiciels

    Informations forums :
    Inscription : Juillet 2006
    Messages : 13 459
    Points : 24 873
    Points
    24 873
    Par défaut
    TForm VCL hérite de TComponent qui implémente IInterface sans la gestion du compteur de référence mais il est tout a fait possible d'utiliser Supports pour récupérer une interface précise dessus

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    type
      IInitializableForm = interface 
        ['{00000000-0000-0000-0000-000000000000}'] // CRTL + MAJ + G
     
        function Initialize(AParam: TTypeOfParam): Boolean;
      end;
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    type
      TModuleTrucUneChoseForm = class(TForm, IInitializableForm)
      private
        { Implémentation de IInitializableForm }
        function Initialize(AParam: TTypeOfParam): Boolean;
      end;
    Pour l'appel

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    var
      MaForm: TForm; // ou TModuleTrucUneChoseForm
      MaInitializableForm: IInitializableForm; 
    begin
      if Supports(MaForm, IInitializableForm, MaInitializableForm) then
        MaInitializableForm.Initialize(MonParam);

    Je l'ai fait en XE2 aussi bien en Delphi que C++Builder, cela fonctionnait parfaitement pour gérer des fenêtres Container pouvant recevoir des fenêtes Dockables qui ne se "connaissaient" mutuellement que via les différentes interfaces

    cela donnait ce genre de déclaration pour le Container
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    TModuleReverseReceptionObjetForm = class(TForm, IModuleReverseReceptionObjet, IModuleReverseReceptionObjetContenantsOut, IModuleReverseReceptionObjetScanFilter)
    Pour les Dockées qui n'offraient pas forcément les mêmes possibilités
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    TModuleReverseDecisionTriMagasinForm = class(TForm, IModuleReverseReceptionObjetDecision, IModuleReverseReceptionObjetDecisionRappel, IRestorableContenantConciliationByActionHandler)
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    TModuleReverseExpeditionFonteControleForm = class(TForm, IModuleReverseReceptionObjetDecision, IRestorableContenantConciliationByActionHandler)
    Aide via F1 - FAQ - Guide du développeur Delphi devant un problème - Pensez-y !
    Attention Troll Méchant !
    "Quand un homme a faim, mieux vaut lui apprendre à pêcher que de lui donner un poisson" Confucius
    Mieux vaut se taire et paraître idiot, Que l'ouvrir et de le confirmer !
    L'ignorance n'excuse pas la médiocrité !

    L'expérience, c'est le nom que chacun donne à ses erreurs. (Oscar Wilde)
    Il faut avoir le courage de se tromper et d'apprendre de ses erreurs

  4. #4
    Membre émérite

    Homme Profil pro
    Développeur informatique
    Inscrit en
    Novembre 2007
    Messages
    3 387
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 62
    Localisation : France, Rhône (Rhône Alpes)

    Informations professionnelles :
    Activité : Développeur informatique

    Informations forums :
    Inscription : Novembre 2007
    Messages : 3 387
    Points : 2 999
    Points
    2 999
    Par défaut
    J'ai bien pensé à ajouter une interface mais ça me fait pas mal de modif dans l'existant.
    D'où l'utilisation des rtti qui fonctionne bien.
    Le problème est que la fonction Initialise est une

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
     
    class procedure TFormAR.initialise;
    begin
      if Assigned(FormAR) = False then
        FormAR := TFormAR.Create(Application.MainForm);
      FormAR.Show;
    end;
    Et donc je ne l'appelle pas sur une instance de la classe.

    En attendant j'ai complété comme ça

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
     
    var
      aClass: TClass;
      context: TRttiContext;
      types: TArray<TRttiType>;
      aType: TRttiType;
      Mth: TRttiMethod;
     
      T : TRttiInstanceType;
      V : TValue;
     
    begin
      context := TRttiContext.Create;
      types := context.GetTypes;
      for aType in types do
      begin
        if aType.TypeKind = tkClass then
        begin
          aClass := aType.AsInstance.MetaclassType;
          if (aClass <> TForm) and aClass.InheritsFrom(TForm) then
          begin
            Mth := aType.GetMethod('Initialise');
            if Assigned(Mth) then
            begin
              // ShowMessage(aClass.QualifiedClassName);
              T := Context.FindType( aClass.QualifiedClassName) as TRttiInstanceType;
              Mth.Invoke(T.metaClassType,[]);
            end;
          end;
        end;
      end;
     
    ça fonctionne mais j'aurais voulu détruire l'objet crée par l'appel de la procédure. Et là, je sais pas comment faire.
    Peut-être que ce n'est pas possible ...
    Pour être précis, je voulais cette procédure pour ouvrir toutes les TForm de l'application qui ont cette procédure pour voir les erreurs restantes après la migration.
    Donc, en faisant ça, je m'évitais de coder chaque ouverture et fermeture pour chacune.

  5. #5
    Membre du Club Avatar de hector94
    Profil pro
    Inscrit en
    Octobre 2007
    Messages
    56
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Octobre 2007
    Messages : 56
    Points : 55
    Points
    55
    Par défaut
    Si initialize renvoi l'objet tform créé, il doit être possible de le détruire ?

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
     
    class function TFormAR.initialise : TForm;
    begin
      if Assigned(FormAR) = False then
        FormAR := TFormAR.Create(Application.MainForm);
      FormAR.Show;
    Result := FormAR;
    end;

  6. #6
    Rédacteur/Modérateur
    Avatar de Andnotor
    Inscrit en
    Septembre 2008
    Messages
    5 693
    Détails du profil
    Informations personnelles :
    Localisation : Autre

    Informations forums :
    Inscription : Septembre 2008
    Messages : 5 693
    Points : 13 128
    Points
    13 128
    Par défaut
    Je rejoins Paul sur l'héritage.

    Mais la méthode n'a même pas besoin d'être virtuelle si tu passes la fiche en paramètre.
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    class procedure TInitForm.Initialise(var aForm :TInitForm);
    begin
      if not Assigned(aForm) then
        aForm := Self.Create(Application);       // Self représente la classe
    end;

  7. #7
    Expert éminent sénior
    Avatar de ShaiLeTroll
    Homme Profil pro
    Développeur C++\Delphi
    Inscrit en
    Juillet 2006
    Messages
    13 459
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 43
    Localisation : France, Seine Saint Denis (Île de France)

    Informations professionnelles :
    Activité : Développeur C++\Delphi
    Secteur : High Tech - Éditeur de logiciels

    Informations forums :
    Inscription : Juillet 2006
    Messages : 13 459
    Points : 24 873
    Points
    24 873
    Par défaut
    Citation Envoyé par Papy214 Voir le message
    Et donc je ne l'appelle pas sur une instance de la classe.
    Dans le cas d'une construction par MetaClass je rejoins Andnotor et en fait, cela s'appelle du coup une Factory

    Attention à la méthode d'appelle pour le Initialise de Andnotor

    il faut bien faire TFormAR.Initialize(MaVarInitForm); pour que le Self soit bien TFormAR et non TInitForm

    Tu peux coupler cela à une TList<TInitFormClass> pour écrire un code ultra simple pour ouvrir d'un coup toutes les fenêtres concernées.

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    var
      MaList: TList<TInitFormClass>; // TInitFormClass = class of TInitForm;
      MaClass: TInitFormClass;
      MaVar: TInitForm;
    begin
      MaList := TList<TInitFormClass>.Create(); // ça peut être une class var de TInitForm et instancié dans le class constructor de TInitForm 
      MaList.Add(TFormAR); // ça voir si tu fais une liste en Singleton et une méthode RegisterMe dans la sectione Initialization dans chaque unité de Form héritée de TInitForm
     
      for MaClass in MaList do
         MaClass.Initialize(MaVar);
    end;
    Aide via F1 - FAQ - Guide du développeur Delphi devant un problème - Pensez-y !
    Attention Troll Méchant !
    "Quand un homme a faim, mieux vaut lui apprendre à pêcher que de lui donner un poisson" Confucius
    Mieux vaut se taire et paraître idiot, Que l'ouvrir et de le confirmer !
    L'ignorance n'excuse pas la médiocrité !

    L'expérience, c'est le nom que chacun donne à ses erreurs. (Oscar Wilde)
    Il faut avoir le courage de se tromper et d'apprendre de ses erreurs

  8. #8
    Expert éminent sénior
    Avatar de ShaiLeTroll
    Homme Profil pro
    Développeur C++\Delphi
    Inscrit en
    Juillet 2006
    Messages
    13 459
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 43
    Localisation : France, Seine Saint Denis (Île de France)

    Informations professionnelles :
    Activité : Développeur C++\Delphi
    Secteur : High Tech - Éditeur de logiciels

    Informations forums :
    Inscription : Juillet 2006
    Messages : 13 459
    Points : 24 873
    Points
    24 873
    Par défaut
    Si l'on reste dans les RTTI

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
          aClass := aType.AsInstance.MetaclassType;
          if aClass.InheritsFrom(TInitForm) then
            TInitFormClass(aClass).Initialize(); // à toi  de voir si tu as besoin des instances retournées, utilise une TObjectList<TInitForm>
    Pour fermer les forms d'un coup une fois toutes ouvertes

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    for i := Screen.FormCount - 1 downto 0 do
      if Screen.Forms[i] is TInitForm then
        Screen.Forms[i].Release(); // Ou Close si caFree dans le OnClose
    Sinon tu peux simplement libérer TObjectList<TInitForm> et via le AOwnsObjects, tous les fênetres seront libérés
    Aide via F1 - FAQ - Guide du développeur Delphi devant un problème - Pensez-y !
    Attention Troll Méchant !
    "Quand un homme a faim, mieux vaut lui apprendre à pêcher que de lui donner un poisson" Confucius
    Mieux vaut se taire et paraître idiot, Que l'ouvrir et de le confirmer !
    L'ignorance n'excuse pas la médiocrité !

    L'expérience, c'est le nom que chacun donne à ses erreurs. (Oscar Wilde)
    Il faut avoir le courage de se tromper et d'apprendre de ses erreurs

  9. #9
    Membre émérite

    Homme Profil pro
    Développeur informatique
    Inscrit en
    Novembre 2007
    Messages
    3 387
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 62
    Localisation : France, Rhône (Rhône Alpes)

    Informations professionnelles :
    Activité : Développeur informatique

    Informations forums :
    Inscription : Novembre 2007
    Messages : 3 387
    Points : 2 999
    Points
    2 999
    Par défaut
    Bonne idée ! Comme quoi la cbhose la plus simple n'est pas forcément celle qu'on voit tout de suite.

    Citation Envoyé par hector94 Voir le message
    Si initialize renvoi l'objet tform créé, il doit être possible de le détruire ?

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
     
    class function TFormAR.initialise : TForm;
    begin
      if Assigned(FormAR) = False then
        FormAR := TFormAR.Create(Application.MainForm);
      FormAR.Show;
    Result := FormAR;
    end;

  10. #10
    Membre émérite

    Homme Profil pro
    Développeur informatique
    Inscrit en
    Novembre 2007
    Messages
    3 387
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 62
    Localisation : France, Rhône (Rhône Alpes)

    Informations professionnelles :
    Activité : Développeur informatique

    Informations forums :
    Inscription : Novembre 2007
    Messages : 3 387
    Points : 2 999
    Points
    2 999
    Par défaut
    Je préfère effectivement cette solution à base de Rtti.
    Et le stockage des fenêtres ouvertes dans une TList également.

    Merci à tous pour vos idées et propositions :-)

    Citation Envoyé par ShaiLeTroll Voir le message
    Si l'on reste dans les RTTI

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
          aClass := aType.AsInstance.MetaclassType;
          if aClass.InheritsFrom(TInitForm) then
            TInitFormClass(aClass).Initialize(); // à toi  de voir si tu as besoin des instances retournées, utilise une TObjectList<TInitForm>
    Pour fermer les forms d'un coup une fois toutes ouvertes

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    for i := Screen.FormCount - 1 downto 0 do
      if Screen.Forms[i] is TInitForm then
        Screen.Forms[i].Release(); // Ou Close si caFree dans le OnClose
    Sinon tu peux simplement libérer TObjectList<TInitForm> et via le AOwnsObjects, tous les fênetres seront libérés

+ Répondre à la discussion
Cette discussion est résolue.

Discussions similaires

  1. Réponses: 2
    Dernier message: 23/04/2008, 11h00
  2. Procédure de classe virtual, override.
    Par cretindezalpes dans le forum Langage
    Réponses: 2
    Dernier message: 22/04/2008, 11h44
  3. Réponses: 4
    Dernier message: 19/04/2006, 15h10
  4. [Débutant] Lancer procédure depuis une autre classe
    Par pugnator dans le forum Langage
    Réponses: 5
    Dernier message: 31/10/2005, 14h50
  5. Utiliser des procédures au lieu des classes
    Par ahage4x4 dans le forum ASP
    Réponses: 5
    Dernier message: 29/06/2005, 10h53

Partager

Partager
  • Envoyer la discussion sur Viadeo
  • Envoyer la discussion sur Twitter
  • Envoyer la discussion sur Google
  • Envoyer la discussion sur Facebook
  • Envoyer la discussion sur Digg
  • Envoyer la discussion sur Delicious
  • Envoyer la discussion sur MySpace
  • Envoyer la discussion sur Yahoo