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

Tkinter Python Discussion :

Fermer une application multifenêtres tkinter


Sujet :

Tkinter Python

  1. #1
    P.G
    P.G est déconnecté
    Membre confirmé
    Profil pro
    Inscrit en
    Octobre 2004
    Messages
    158
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Octobre 2004
    Messages : 158
    Par défaut Fermer une application multifenêtres tkinter
    Bonjour à tous,

    Travaillant sur une GUI à plusieurs fenêtres avec tkinter en compilant plusieurs codes je cherche dans l'exemple ci-dessous à fermer toutes les fenêtres avec l'appui sur le bouton quit.

    Hélas le message invariablement obtenu est le suivant :
    Exception in Tkinter callback
    Traceback (most recent call last):
    File "C:\Users\Patrick\Desktop\WPy64-3760\python-3.7.6.amd64\lib\tkinter\__init__.py", line 1705, in __call__
    return self.func(*args)
    File "R:/TPRO/NAS PRO/USB/NSI/NSI 1ERE/100300 NSI IA PROJET/AGENT WUMPUS/AGENT_WUMPUS_TRAVAUX_2022/python3/chap14/multi_fen_pg_essai_1.py", line 95, in quit
    self.root.destroy()
    AttributeError: 'Demo' object has no attribute 'root'

    Voilà le code (issu au départ de swinnen)
    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
    35
    36
    37
    38
    39
    40
    41
    42
    43
    44
    45
    46
    47
    48
    49
    50
    51
    52
    53
    54
    55
    56
    57
    58
    59
    60
    61
    62
    63
    64
    65
    66
    67
    68
    69
    70
    71
    72
    73
    74
    75
    76
    77
    78
    79
    80
    81
    82
    83
    84
    85
    86
    87
    88
    89
    90
    91
    92
    93
    94
    95
    96
    97
    98
    99
    100
    101
    102
    103
    104
    105
    106
     
    #! /usr/bin/env python
    # -*- coding:Utf8 -*-
     
    from tkinter import *
     
    class FunnyButton(Button):
        "Bouton de fantaisie : vert virant au rouge quand on l'actionne"
        def __init__(self, boss, **Arguments):
            Button.__init__(self, boss,  bg ="dark green", fg ="white", bd =5,
                            activebackground ="red", activeforeground ="yellow",
                            font =('Helvetica', 12, 'bold'), **Arguments)
     
    class SpinBox(Frame):
        "widget composite comportant un Label et 2 boutons 'up' & 'down'"
        def __init__(self, boss, largC=5, largB =2, vList=[0], liInd=0, orient =Y):
            Frame.__init__(self, boss)
            self.vList =vList           # liste des valeurs à présenter
            self.liInd =liInd           # index de la valeur à montrer par défaut
            if orient ==Y:
                s, augm, dimi = TOP, "^", "v"      # Orientation 'verticale'
            else:
                s, augm, dimi = RIGHT, ">", "<"    # Orientation 'horizontale'
            Button(self, text =augm, width =largB, command =self.up).pack(side =s)
            self.champ = Label(self, bg ='white', width =largC,
                               text =str(vList[liInd]), relief =SUNKEN)
            self.champ.pack(pady =3, side =s)
            Button(self, text=dimi, width=largB, command =self.down).pack(side =s)
     
        def up(self):
            if self.liInd < len(self.vList) -1:
                self.liInd += 1
            else:
                self.bell()       # émission d'un bip
            self.champ.configure(text =str(self.vList[self.liInd]))
     
        def down(self):
            if self.liInd > 0:
                self.liInd -= 1
            else:
                self.bell()       # émission d'un bip
            self.champ.configure(text =str(self.vList[self.liInd]))
     
        def get(self):
            return self.vList[self.liInd]
     
    class FenDessin(Toplevel):
        "Fenêtre satellite (modale) contenant un simple canevas"
        def __init__(self, **Arguments):
            Toplevel.__init__(self, **Arguments)
            self.geometry("250x200+100+240")
            self.overrideredirect(1)            # => fenêtre sans bordure ni bandeau
            #self.transient(self.master)         # => fenêtre 'modale'
            self.can =Canvas(self, bg="ivory", width =200, height =150)
            self.img = PhotoImage(file ="papillon2.gif")
            self.can.create_image(90, 80, image =self.img)        
            self.can.pack(padx =20, pady =20)
     
    class FenControle(Toplevel):
        "Fenêtre satellite contenant des contrôles de redimensionnement"
        def __init__(self, boss, **Arguments):
            Toplevel.__init__(self, boss, **Arguments)
            self.geometry("250x200+400+230")
            self.resizable(width =0, height =0)    # => empêche le redimensionnement
            p =(10, 30, 60, 90, 120, 150, 180, 210, 240, 270, 300)
            self.spX =SpinBox(self, largC=5,largB =1,vList =p,liInd=5,orient =X)
            self.spX.pack(pady =5)
            self.spY =SpinBox(self, largC=5,largB =1,vList =p,liInd=5,orient =Y)
            self.spY.pack(pady =5)
            FunnyButton(self, text ="Dimensionner le canevas",
                        command =boss.redimF1).pack(pady =5)
     
    class Demo(Frame):
        "Démo. de quelques caractéristiques du widget Toplevel"
        def __init__(self, parent):
            Frame.__init__(self, parent)
            self.parent = parent        
            self.master.geometry("400x300+200+200")
            self.master.config(bg ="cadet blue")
            FunnyButton(self, text ="Top 1", command =self.top1).pack(side =LEFT)
            FunnyButton(self, text ="Top 2", command =self.top2).pack(side =LEFT)
            FunnyButton(self, text ="Quitter", command =self.quit).pack()
            self.pack(side =BOTTOM, padx =10, pady =10)
     
        def top1(self):
            self.fen1 =FenDessin(bg ="grey")
     
        def top2(self):
            self.fen2 =FenControle(self, bg ="khaki")
     
        def redimF1(self):
            dimX, dimY = self.fen2.spX.get(), self.fen2.spY.get()
            self.fen1.can.config(width =dimX, height =dimY)
     
        def quit(self):
            self.root.destroy()        
     
    def main():
        root = Tk()
        root.geometry("330x300+300+300")
        app = Demo(root)
        root.mainloop()
     
     
    if __name__ == '__main__':
        main()
    Sous winpython WPy64-3760

    Merci du coup de main.

    Patrick

  2. #2
    Membre Expert
    Avatar de MPython Alaplancha
    Homme Profil pro
    Paysan à 3 francs six sous
    Inscrit en
    Juin 2018
    Messages
    914
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Pyrénées Orientales (Languedoc Roussillon)

    Informations professionnelles :
    Activité : Paysan à 3 francs six sous
    Secteur : Agroalimentaire - Agriculture

    Informations forums :
    Inscription : Juin 2018
    Messages : 914
    Billets dans le blog
    7
    Par défaut
    Bonjour,
    L'exception dit que la classe Demo n'a pas d'attribut nommé root
    self.root.destroy()
    AttributeError: 'Demo' object has no attribute 'root'
    Effectivement, en y regardant de plus près on voit que l'attribut convoité s'appelle self.parent et non self.root:
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    class Demo(Frame):
        "Démo. de quelques caractéristiques du widget Toplevel"
        def __init__(self, parent):
            Frame.__init__(self, parent)
            self.parent = parent

  3. #3
    P.G
    P.G est déconnecté
    Membre confirmé
    Profil pro
    Inscrit en
    Octobre 2004
    Messages
    158
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Octobre 2004
    Messages : 158
    Par défaut
    Bonjour Hominidé

    Oui effectivement je pensais que le mot root était incorporé à l'ensemble tkinter.
    Une fois corrigé cela fonctionne merci beaucoup.

    Une question complémentaire peut-on tester l'existence d'un attribut dans une classe pour faire un code du genre
    si la fenêtre fille n'est pas créée alors envoyé un message.

    Cela éviterai ce message et permettrai de s'adapter à la situation :
    Exemple de message :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
        self.fen1.can.config(width =dimX, height =dimY)
    AttributeError: 'Demo' object has no attribute 'fen1'
    (puisque la sous fenêtre n'a pas encore été créée )

    Merci encore

    P.G

  4. #4
    Membre Expert
    Avatar de MPython Alaplancha
    Homme Profil pro
    Paysan à 3 francs six sous
    Inscrit en
    Juin 2018
    Messages
    914
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Pyrénées Orientales (Languedoc Roussillon)

    Informations professionnelles :
    Activité : Paysan à 3 francs six sous
    Secteur : Agroalimentaire - Agriculture

    Informations forums :
    Inscription : Juin 2018
    Messages : 914
    Billets dans le blog
    7
    Par défaut
    Bonjour,
    Citation Envoyé par P.G Voir le message
    Une question complémentaire peut-on tester l'existence d'un attribut dans une classe pour faire un code du genre
    si la fenêtre fille n'est pas créée alors
    Oui. Tu peux tester son existence avec un if ...
    (tu pourrais aussi mettre en place une clause try/except AttributeError: si cela est pertinant...)

  5. #5
    Expert éminent
    Homme Profil pro
    Architecte technique retraité
    Inscrit en
    Juin 2008
    Messages
    21 681
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Manche (Basse Normandie)

    Informations professionnelles :
    Activité : Architecte technique retraité
    Secteur : Industrie

    Informations forums :
    Inscription : Juin 2008
    Messages : 21 681
    Par défaut
    Citation Envoyé par P.G Voir le message
    Oui effectivement je pensais que le mot root était incorporé à l'ensemble tkinter.
    root/self.root sont des variables/attributs qui n'existent qu'après leur avoir assigné un objet (root = Tk() par exemple).
    Evitez quand vous le pouvez de détruire la fenêtre principale, sortez plutôt de "mainloop" via un "quit":
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    import tkinter as tk
     
    label = tk.Label(text='123')
    label.pack()
    btn = tk.Button(text='quit', command=label.quit).pack()
    tk.mainloop()
    print('on sort...')
    Citation Envoyé par P.G Voir le message
    Une question complémentaire peut-on tester l'existence d'un attribut dans une classe pour faire un code du genre
    hasattr ou try....except sont là pour ça mais c'est vous qui avez fabriqué la classe... initialiser self.fen1 à None et tester qu'il est différent de None avant d'accéder à self.fen1.can... est la solution la plus logique/simple.

    Après il faudrait peut être revoir les relations entre vos classes: quel est l'intérêt d'avoir 2 fenêtres qui échangent des messages via un tiers plutôt que directement (demo crée fen2 qui crée fen1 => demo contrôle fen2 qui contrôle fen1...).

    - W
    Architectures post-modernes.
    Python sur DVP c'est aussi des FAQs, des cours et tutoriels

  6. #6
    Invité de passage
    Homme Profil pro
    Consultant informatique
    Inscrit en
    Mars 2023
    Messages
    1
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Landes (Aquitaine)

    Informations professionnelles :
    Activité : Consultant informatique
    Secteur : Santé

    Informations forums :
    Inscription : Mars 2023
    Messages : 1
    Par défaut resising root from a Toplevel window
    Bonjour

    J'ai une fenêtre root definie comme suit : root.geometry("480x580") qui fonctionne parfaitement que je souhaite reformater depuis une classe Toplevel comme suit:

    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
     
    class Resize_desk(Toplevel):
         def __init__(self,parent)
              global librairie, options
              super.__init__(parent)
              self.parent-parent
              self.title("Choisissez une taille")
             ....
             ....
         if taille==1:
            self.parent.geometry("480x240")
         elif taille == 2:
            self.parent.geometry("480x360")
         elif taille ==3:
            self.parent.geometry("480x480")
         ....
         ....
    Tout le reste fonctionne correctement MAIS l'exécution, quelle que soit la taille choisie, renvoie le message d'erreur suivant;

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
     
         self.parent.geometry("480x360")
       Type Error: 'str' object is not callable
    Et l'option root.geometry("480x360") depuis la classe Resize_desk ne marche pas.

    Quelle peut être la raison de cette erreur?
    3 jours de recherche sur le web n'ont rien donné.
    Auriez vous une idée?

  7. #7
    Expert éminent
    Homme Profil pro
    Architecte technique retraité
    Inscrit en
    Juin 2008
    Messages
    21 681
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Manche (Basse Normandie)

    Informations professionnelles :
    Activité : Architecte technique retraité
    Secteur : Industrie

    Informations forums :
    Inscription : Juin 2008
    Messages : 21 681
    Par défaut
    Salut,

    Citation Envoyé par Alphée Voir le message
    Quelle peut être la raison de cette erreur?
    3 jours de recherche sur le web n'ont rien donné.
    Auriez vous une idée?
    Pourquoi ne pas ouvrir une discussion plutôt que de cannibaliser celle de quelqu'un d'autre qui a eu d'autres problèmes?
    Le message d'erreur dit juste que l'objet self.parent.geometry étant un str(ing), on ne peut l'utiliser comme fonction.
    Après par quelle magie self.parent.geometry est devenu string dans l'exécution de votre code? C'est pas avec le peu que vous en montrez qu'on pourra dire quoi que ce soit... D'où l'intérêt d'ouvrir une discussion en y ajoutant des informations complémentaires.
    - W
    Architectures post-modernes.
    Python sur DVP c'est aussi des FAQs, des cours et tutoriels

Discussions similaires

  1. Fermer une application Windows
    Par telecnop dans le forum Langage
    Réponses: 20
    Dernier message: 28/06/2006, 21h15
  2. Fermer une application externe
    Par jean tof dans le forum C++Builder
    Réponses: 2
    Dernier message: 02/05/2006, 16h18
  3. Fermer une application exterieure
    Par P.B dans le forum Windows
    Réponses: 4
    Dernier message: 01/12/2005, 15h17
  4. [VB6]Fermer une application avec VB
    Par Mylou dans le forum VB 6 et antérieur
    Réponses: 3
    Dernier message: 04/04/2003, 21h32
  5. Fermer une application à l'aide de OnIdle
    Par Thierry Rapp dans le forum Composants VCL
    Réponses: 2
    Dernier message: 29/08/2002, 12h44

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