Как изменить в классе 1 свойства виджета (метки) в классе 2 (Tkinter)? - PullRequest
0 голосов
/ 29 апреля 2019

Я пытаюсь использовать кнопку recordBn в классе MainMenu, которая изменяет текст метки dateLb в классе RecordTime.Но я получаю сообщение об ошибке, как показано ниже.

AttributeError: 'RecordTime' object has no attribute 'dateLb'

Использование empcbxlist() из GFPTime Класс

Я пытался решить эту проблему из этого post и несколько других сообщений с таким же сообщением об ошибке (AttributeError), но не смогли найти решение.Помогите мне решить мою ошибку

#!/usr/bin/python
import os.path
import sys
import tkinter as tk
from tkinter import messagebox
from tkinter import ttk
# from PIL import Image, ImageTk
# from dbfunctions import *
import datetime

class GFPTime(tk.Tk):
    def __init__(self,*args,**kwargs):

        tk.Tk.__init__(self,*args,**kwargs)

        tk.Tk.wm_title(self,"GFP Employee Timecard System")
        tk.Tk.wm_geometry(self,"800x480+0+0")

        container =tk.Frame(self)
        container.pack(side='top',fill='both',expand= True)

        container.grid_rowconfigure(0,weight=1)
        container.grid_columnconfigure(0,weight=1)

        self.frames = {}

        for F in (MainMenu,RecordTime):

            frame = F(container,self)

            self.frames[F] = frame
            frame.grid(row=0,column=0,stick='nsew')

        self.show_frame(MainMenu)
        # print(self.frames.values())
    def show_frame(self,cont):
        frame = self.frames[cont]
        frame.tkraise()

    def empcbxlist(self, widget_name, criteria, output):
        # db = TimeCardDB()
        # cbvalue = db.listActiveEmployees()
        # db.dbclose()

        widget = getattr(RecordTime,'dateLb')
        widget[criteria] = output

###
class MainMenu(tk.Frame):

    def __init__(self,parent,controller):
        tk.Frame.__init__(self,parent)
        #tk.Frame.configure(self,background='red')
        font9 = "-family {Minion Pro} -size 14 -weight bold"
        self.controller = controller

        recordBn = tk.Button(self,command=lambda: [self.controller.show_frame(RecordTime), self.controller.empcbxlist('self.dateLb','text','yessss')])
        recordBn.place(relx=0.269, rely=0.646, height=50, width=180)
        recordBn.configure(activebackground="#ececec")
        recordBn.configure(activeforeground="#000000")
        recordBn.configure(background="#d9d9d9")
        recordBn.configure(disabledforeground="#a3a3a3")
        recordBn.configure(font=font9)
        recordBn.configure(foreground="#000000")
        recordBn.configure(highlightbackground="#d9d9d9")
        recordBn.configure(highlightcolor="black")
        recordBn.configure(pady="0")
        recordBn.configure(text='''Record Time''')#!/usr/bin/python


class RecordTime(tk.Frame):

    def __init__(self,parent,controller):
        tk.Frame.__init__(self,parent)
        self.controller = controller
        dateLb = tk.Label(self)
        dateLb.place(relx=0.213, rely=0.021, height=38, width=144)
        dateLb.configure(activebackground="#f9f9f9")
        dateLb.configure(activeforeground="black")
        dateLb.configure(disabledforeground="#a3a3a3")
        dateLb.configure(font="-family {Minion Pro} -size 14 -weight bold")
        dateLb.configure(foreground="#000000")
        dateLb.configure(highlightbackground="#d9d9d9")
        dateLb.configure(highlightcolor="black")
        dateLb.configure(text='''SHOW DATE''')



GFPTime().mainloop()

Ответы [ 2 ]

0 голосов
/ 29 апреля 2019

Ваш код немного грязный и слишком сложный.Я уменьшил его до более удобоваримого размера - украшения только добавляли к беспорядку кода.

Пожалуйста, смотрите на него, пока не поймете, что и зачем делают элементы, и, может быть, восстановите оттуда.

import tkinter as tk
import datetime


class GFPTime(tk.Tk):

    def __init__(self):
        super().__init__()
        super().wm_title("GFP Employee Timecard System")
        self.container = tk.Frame(self)
        self.container.pack(side='top',fill='both',expand= True)
        self.frames = {}
        for F in (MainMenu, RecordTime):    
            self.frames[F] = F(self.container, self)
            self.frames[F].grid(row=0,column=0,stick='nsew')
        self.show_frame(MainMenu)

    def show_frame(self, container):
        self.frames[container].tkraise()

    def empcbxlist(self, text_value):
        self.show_frame(RecordTime)
        self.frames[RecordTime].date_lbl.configure(text=text_value)


class MainMenu(tk.Frame):

    def __init__(self, parent, controller):
        super().__init__(parent)
        self.controller = controller
        record_btn = tk.Button(self, text='Record Time', 
                             command=lambda text_value=str(datetime.date.today()): self.controller.empcbxlist(text_value))
        record_btn.pack()


class RecordTime(tk.Frame):

    def __init__(self, parent, controller):
        super().__init__(parent)
        self.controller = controller
        self.date_lbl = tk.Label(self, text='SHOW DATE')
        self.date_lbl.pack()


GFPTime().mainloop()
0 голосов
/ 29 апреля 2019

Есть несколько проблем с вашим кодом.

  1. Когда вы создаете словарь self.frames и сохраняете объект каждого фрейма со своим именем класса, используйте словарь для получения их методов и атрибутов, а не для использования имени класса, такого как widget = getattr(RecordTime,'dateLb') их вместо RecordTime используйте класс, который вы сохранили в словаре.

  2. Не указывайте имя виджета self.dateLb в лямбда-функции, просто достаточно dateLb.

    Изменение:

    self.controller.empcbxlist('self.dateLb','text','yessss')

    К

    self.controller.empcbxlist('dateLb','text','yessss')

  3. Как сказал @ aw1668 "Заменить dateLb на self.dateLb в RecordTime классе" Так что глобализируйте dateLb метку, изменив ее на self.dateLb.

Я изменил блок кода внутри empcbxlist() на этот.

def empcbxlist(self, widget_name, criteria, output):
    # db = TimeCardDB()
    # cbvalue = db.listActiveEmployees()
    # db.dbclose()

    # Get the frame. 
    ob = self.frames.get( RecordTime ) 
    widget = getattr(ob,widget_name)
    widget[criteria] = output

    # can use this to change the text to be on safe side.
    # for i, j  in self.frames.items():
    #     if i.__name__ == RecordTime.__name__:
    #         widget = getattr(j,widget_name)
    #         widget[criteria] = output

Полный код:

#!/usr/bin/python
import os.path
import sys
import tkinter as tk
from tkinter import messagebox
from tkinter import ttk
# from PIL import Image, ImageTk
# from dbfunctions import *
import datetime

class GFPTime(tk.Tk):
    def __init__(self,*args,**kwargs):

        tk.Tk.__init__(self,*args,**kwargs)

        tk.Tk.wm_title(self,"GFP Employee Timecard System")
        tk.Tk.wm_geometry(self,"800x480+0+0")

        container =tk.Frame(self)
        container.pack(side='top',fill='both',expand= True)

        container.grid_rowconfigure(0,weight=1)
        container.grid_columnconfigure(0,weight=1)

        self.frames = {}

        for F in (MainMenu,RecordTime):

            frame = F(container,self)

            self.frames[F] = frame
            frame.grid(row=0,column=0,stick='nsew')

        self.show_frame(MainMenu)
        # print(self.frames.values())
    def show_frame(self,cont):
        frame = self.frames[cont]
        frame.tkraise()

    def empcbxlist(self, widget_name, criteria, output):
        # db = TimeCardDB()
        # cbvalue = db.listActiveEmployees()
        # db.dbclose()

        ob = self.frames.get( RecordTime ) 
        widget = getattr(ob,widget_name)
        widget[criteria] = output

###
class MainMenu(tk.Frame):

    def __init__(self,parent,controller):
        tk.Frame.__init__(self,parent)
        #tk.Frame.configure(self,background='red')
        font9 = "-family {Minion Pro} -size 14 -weight bold"
        self.controller = controller

        recordBn = tk.Button(self,command=lambda: [self.controller.show_frame(RecordTime), self.controller.empcbxlist('dateLb','text','yessss')])
        recordBn.place(relx=0.269, rely=0.646, height=50, width=180)
        recordBn.configure(activebackground="#ececec")
        recordBn.configure(activeforeground="#000000")
        recordBn.configure(background="#d9d9d9")
        recordBn.configure(disabledforeground="#a3a3a3")
        recordBn.configure(font=font9)
        recordBn.configure(foreground="#000000")
        recordBn.configure(highlightbackground="#d9d9d9")
        recordBn.configure(highlightcolor="black")
        recordBn.configure(pady="0")
        recordBn.configure(text='''Record Time''')#!/usr/bin/python


class RecordTime(tk.Frame):

    def __init__(self,parent,controller):
        tk.Frame.__init__(self,parent)
        self.controller = controller
        self.dateLb = tk.Label(self)
        self.dateLb.place(relx=0.213, rely=0.021, height=38, width=144)
        self.dateLb.configure(activebackground="#f9f9f9")
        self.dateLb.configure(activeforeground="black")
        self.dateLb.configure(disabledforeground="#a3a3a3")
        self.dateLb.configure(font="-family {Minion Pro} -size 14 -weight bold")
        self.dateLb.configure(foreground="#000000")
        self.dateLb.configure(highlightbackground="#d9d9d9")
        self.dateLb.configure(highlightcolor="black")
        self.dateLb.configure(text='''SHOW DATE''')



GFPTime().mainloop()

Надеюсь, это поможет.

...