gpt4 book ai didi

python - 向 tk.Tk 和 tk.Toplevel 添加方法

转载 作者:太空宇宙 更新时间:2023-11-04 01:47:56 25 4
gpt4 key购买 nike

我想向 tk.Tktk.Toplevel 的实例添加 2 个方法。前者作为应用程序根窗口存在,后者由用户创建任意次数。每个 tk.Toplevel 代表应用程序的不同功能,目前有 13 种变体。该应用程序已配置为在任何时候只能创建每个变体的 1 个实例。

解决这个问题的一种方法可能是对这两个类进行子类化:

class RootWindow(tk.Tk):
def method_1(self):
...
def method_2(self):
...

class TopWindow(tk.Toplevel):
def method_1(self):
...
def method_2(self):
...

因为我只想添加 2 个功能方法(它们是相同的,因为它们使用这些类的方法,例如 winfo_height()),子类化似乎过分而且多余,因为相同的代码是写了两次。因此,另一种方法可能是使用setattr(),但我觉得这违反了类设计的开闭原则:

def method_1(self):
...
def method_2(self):
...

class RootWindow(tk.Tk):
def __init__(self):
setattr(self, 'method_1', method_1)
setattr(self, 'method_2', method_2)

class TopWindow(tk.Toplevel):
def __init__(self):
setattr(self, 'method_1', method_1)
setattr(self, 'method_2', method_2)

现在我只写了一次方法并修改了类;本质上,这浓缩了第一种方法并且实际上是相同的。

鉴于 tk.Tk 是一个窗口,而 tk.Toplevel 是一个窗口,有没有办法通过 tkinter 我在哪里可以定义所有基于“窗口”的小部件都可以访问的方法吗?如果所有窗口都来自单个 tkinter 类,这将不是问题,但是 tk.Tk 的根窗口与 tk.Toplevel 的根窗口不同(例如事件绑定(bind),不一定是视觉行为)。

编辑:

看看@BryanOakleys 的解决方案,我发现 linters 真的不喜欢这样;他们指向未解析的属性引用,我知道子类可以访问这些属性,但不能访问混合类。请考虑以下事项:

class Mixin:
def method(self):
width = self.winfo_width()
height = self.winfo_height()
print(f'The window is {width} x {height}')

IDE linters 抛出警告,因为 self 找不到定义的 winfo_width。然而,我知道它会被 tk.Tktk.Toplevel 的子类的实例调用,但滥用会导致未定义的行为,例如示例:

class OtherWindow(Window):
pass

o = OtherWindow()
o.method()
# Attribute error

这有问题吗?提议的解决方案有效,但感觉它是被迫工作的,因为“我知道它将如何使用”。一个明显的解决方案是键入提示:

class Mixin:
def method(self: Union[tk.Tk, tk.Toplevel]):
width = self.winfo_width()
height = self.winfo_height()
print(f'The window is {width} x {height}')

因此,linters 得到了满足,代码对任何人都是可读的(假设他们不需要查找 Union)。同样,这是使用混合类(从未尝试过)时的预期行为吗?

最佳答案

我会使用混合类。保留 TopWindowRootWindow 的自定义类是创建基础 ToplevelTk 类的特化的正确解决方案,mixin解决了不想重复代码的问题。

例如:

class CustomMixin():
def method_1(self):
...
def method_2(self):
...

class TopWindow(CustomMixin, tk.Toplevel):
pass

class RootWindow(CustomMixin, tk.Tk):
pass

关于python - 向 tk.Tk 和 tk.Toplevel 添加方法,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/58664587/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com