tkinter 弹出窗口 传值回到 主窗口

简介: 有些时候,我们需要使用弹出窗口,对程序的运行参数进行设置。有两种选择一、标准窗口如果只对一个参数进行设置(或者说从弹出窗口取回一个值),那么可以使用simpledialog,导入方法:from tkinter.

有些时候,我们需要使用弹出窗口,对程序的运行参数进行设置。有两种选择

一、标准窗口

如果只对一个参数进行设置(或者说从弹出窗口取回一个值),那么可以使用simpledialog,导入方法:

from tkinter.simpledialog import askstring, askinteger, askfloat
AI 代码解读
完整例子
import tkinter as tk
from tkinter.simpledialog import askstring, askinteger, askfloat

# 接收一个整数
def print_integer():
    res = askinteger("Spam", "Egg count", initialvalue=12*12)
    print(res)
    
# 接收一个浮点数
def print_float():
    res = askfloat("Spam", "Egg weight\n(in tons)", minvalue=1, maxvalue=100)
    print(res)

# 接收一个字符串
def print_string():
    res = askstring("Spam", "Egg label")
    print(res)
    
    
root = tk.Tk()
    
tk.Button(root, text='取一个字符串', command=print_string).pack()
tk.Button(root, text='取一个整数', command=print_integer).pack()
tk.Button(root, text='取一个浮点数', command=print_float).pack()

root.mainloop()
AI 代码解读

二、自定义窗口

如果要设置的参数个数超过两个,那么tkinter提供的标准窗口就处理不了了。

这就需要自定义一个窗口,那么问题来了:怎样将自定义窗口中的数据传回主窗口?

百度查询,不外乎两种方法:全局变量(global)、对象变量([]、{}等),都不是我想要的。

然后,去 stackoverflow 逛了一下,综合了个问题的答案,得到两个本人比较满意的方案。

一种是松耦合,另一种是紧耦合

1)松耦合

说明

  • 主窗类,继承了 tk.Tk
  • 弹窗类,继承了 tk.Toplevel

要点

  • 弹窗,将多个数据,打包,放入一个名为 username 的私有 list 对象,销毁弹窗
  • 主窗,待弹窗运行后,通过wait_window方法,取得弹窗的名为 username 私有变量

完整代码:

import tkinter as tk

'''松耦合'''

# 弹窗
class MyDialog(tk.Toplevel):
    def __init__(self):
        super().__init__()
        self.title('设置用户信息')
        
        # 弹窗界面
        self.setup_UI()
        
        
    def setup_UI(self):
        # 第一行(两列)
        row1 = tk.Frame(self)
        row1.pack(fill="x")
        tk.Label(row1, text='姓名:', width=8).pack(side=tk.LEFT)
        self.name = tk.StringVar()
        tk.Entry(row1, textvariable=self.name, width=20).pack(side=tk.LEFT)
        
        # 第二行
        row2 = tk.Frame(self)
        row2.pack(fill="x", ipadx=1, ipady=1)
        tk.Label(row2, text='年龄:', width=8).pack(side=tk.LEFT)
        self.age = tk.IntVar()
        tk.Entry(row2, textvariable=self.age, width=20).pack(side=tk.LEFT)
        
        # 第三行
        row3 = tk.Frame(self)
        row3.pack(fill="x")
        tk.Button(row3, text="取消", command=self.cancel).pack(side=tk.RIGHT)
        tk.Button(row3, text="确定", command=self.ok).pack(side=tk.RIGHT)
        

    def ok(self):
        self.userinfo = [self.name.get(), self.age.get()] # 设置数据
        self.destroy() # 销毁窗口
        
    def cancel(self):
        self.userinfo = None # 空!
        self.destroy()

        



# 主窗
class MyApp(tk.Tk):
    
    def __init__(self):
        super().__init__()
        #self.pack() # 若继承 tk.Frame ,此句必须有!
        self.title('用户信息')
        
        # 程序参数/数据
        self.name = '张三'
        self.age = 30
        
        # 程序界面
        self.setupUI()

    
    def setupUI(self):
        # 第一行(两列)
        row1 = tk.Frame(self)
        row1.pack(fill="x")
        tk.Label(row1, text='姓名:', width=8).pack(side=tk.LEFT)
        self.l1 = tk.Label(row1, text=self.name, width=20)
        self.l1.pack(side=tk.LEFT)
        
        # 第二行
        row2 = tk.Frame(self)
        row2.pack(fill="x")
        tk.Label(row2, text='年龄:', width=8).pack(side=tk.LEFT)
        self.l2 = tk.Label(row2, text=self.age, width=20)
        self.l2.pack(side=tk.LEFT)
        
        # 第三行
        row3 = tk.Frame(self)
        row3.pack(fill="x")
        tk.Button(row3, text="设置", command=self.setup_config).pack(side=tk.RIGHT)
        
        
    # 设置参数
    def setup_config(self):
        # 接收弹窗的数据
        res = self.ask_userinfo()
        #print(res)
        if res is None: return
        
        # 更改参数
        self.name, self.age = res
        
        # 更新界面
        self.l1.config(text=self.name)
        self.l2.config(text=self.age)
    
    
    # 弹窗
    def ask_userinfo(self):
        inputDialog = MyDialog()
    
        self.wait_window(inputDialog) # 这一句很重要!!!
        
        return inputDialog.userinfo
        
        
if __name__ == '__main__':
    app = MyApp()
    app.mainloop()
    
AI 代码解读

2)紧耦合

说明

  • 主窗类,继承了 tk.Tk
  • 弹窗类,继承了 tk.Toplevel

要点

  • 弹窗,显式地保存父窗口,显式地修改父窗口数据,显式地更新父窗口部件,最后销毁弹窗
  • 主窗,待弹窗运行后,通过wait_window方法,返回 None

完整代码:

import tkinter as tk

'''紧耦合'''

# 弹窗
class PopupDialog(tk.Toplevel):
    def __init__(self, parent):
        super().__init__()
        self.title('设置用户信息')
        
        self.parent = parent # 显式地保留父窗口
        
        # 第一行(两列)
        row1 = tk.Frame(self)
        row1.pack(fill="x")
        tk.Label(row1, text='姓名:', width=8).pack(side=tk.LEFT)
        self.name = tk.StringVar()
        tk.Entry(row1, textvariable=self.name, width=20).pack(side=tk.LEFT)
        
        # 第二行
        row2 = tk.Frame(self)
        row2.pack(fill="x", ipadx=1, ipady=1)
        tk.Label(row2, text='年龄:', width=8).pack(side=tk.LEFT)
        self.age = tk.IntVar()
        tk.Entry(row2, textvariable=self.age, width=20).pack(side=tk.LEFT)
        
        # 第三行
        row3 = tk.Frame(self)
        row3.pack(fill="x")
        tk.Button(row3, text="取消", command=self.cancel).pack(side=tk.RIGHT)
        tk.Button(row3, text="确定", command=self.ok).pack(side=tk.RIGHT)
        

    def ok(self):
        # 显式地更改父窗口参数
        self.parent.name = self.name.get()
        self.parent.age = self.age.get()
        
        # 显式地更新父窗口界面
        self.parent.l1.config(text=self.parent.name)
        self.parent.l2.config(text=self.parent.age)
        
        self.destroy() # 销毁窗口
        
    def cancel(self):
        self.destroy()

        



# 主窗
class MyApp(tk.Tk):
    
    def __init__(self):
        super().__init__()
        # self.pack() # 若继承 tk.Frame,此句必须有!!!
        self.title('用户信息')
        
        # 程序参数
        self.name = '张三'
        self.age = 30
        
        # 程序界面
        self.setupUI()

    
    def setupUI(self):
        # 第一行(两列)
        row1 = tk.Frame(self)
        row1.pack(fill="x")
        tk.Label(row1, text='姓名:', width=8).pack(side=tk.LEFT)
        self.l1 = tk.Label(row1, text=self.name, width=20)
        self.l1.pack(side=tk.LEFT)
        
        # 第二行
        row2 = tk.Frame(self)
        row2.pack(fill="x")
        tk.Label(row2, text='年龄:', width=8).pack(side=tk.LEFT)
        self.l2 = tk.Label(row2, text=self.age, width=20)
        self.l2.pack(side=tk.LEFT)
        
        # 第三行
        row3 = tk.Frame(self)
        row3.pack(fill="x")
        tk.Button(row3, text="设置", command=self.setup_config).pack(side=tk.RIGHT)
        
    # 设置参数
    def setup_config(self):
        pw = PopupDialog(self)
        self.wait_window(pw) # 这一句很重要!!!
        
        return
        
        
if __name__ == '__main__':
    app = MyApp()
    app.mainloop()
    
AI 代码解读

效果图

目录
打赏
0
0
0
0
12
分享
相关文章
阿里云网络安全体系解析:如何构建数字时代的"安全盾牌"
在数字经济时代,阿里云作为亚太地区最大的云服务提供商,构建了行业领先的网络安全体系。本文解析其网络安全架构的三大核心维度:基础架构安全、核心技术防护和安全管理体系。通过技术创新与体系化防御,阿里云为企业数字化转型提供坚实的安全屏障,确保数据安全与业务连续性。案例显示,某金融客户借助阿里云成功拦截3200万次攻击,降低运维成本40%,响应时间缩短至8分钟。未来,阿里云将继续推进自适应安全架构,助力企业提升核心竞争力。
NVIDIA Triton系列02-功能与架构简介
本文介绍了NVIDIA Triton推理服务器的功能与架构,强调其不仅适用于大型服务类应用,还能广泛应用于各类推理场景。Triton支持多种模型格式、查询类型和部署方式,具备高效的模型管理和优化能力,确保高性能和系统稳定性。文章详细解析了Triton的主从架构,包括模型仓库、客户端应用、通信协议和推理服务器的核心功能模块。
359 1
NVIDIA Triton系列02-功能与架构简介
在 Vue 中使用自定义指令
【10月更文挑战第14天】通过合理地使用自定义指令,可以为 Vue 应用带来更多的灵活性和扩展性,提高开发效率和用户体验。
MySQL——如何查看MySQL登录日志
MySQL——如何查看MySQL登录日志
774 0
解决方案评测:通义万相 AI 绘画创作
通义万相 AI 绘画创作工具在功能、使用体验等方面表现出色,为用户提供了一种便捷、高效的绘画创作方式。虽然存在一些不足之处,但随着技术的不断发展和优化,相信其性能和表现会不断提升。对于艺术家、设计师、创意工作者以及普通爱好者来说,通义万相都是一款值得尝试和探索的 AI 绘画工具。
837 4
解决方案评测:通义万相 AI 绘画创作
Java一分钟之-高级集合框架:并发集合(Collections.synchronizedXXX)
【5月更文挑战第18天】Java集合框架的`Collections.synchronizedXXX`方法可将普通集合转为线程安全,但使用时需注意常见问题和易错点。错误的同步范围(仅同步单个操作而非迭代)可能导致并发修改异常;错误地同步整个集合类可能引起死锁;并发遍历和修改集合需使用`Iterator`避免`ConcurrentModificationException`。示例代码展示了正确使用同步集合的方法。在复杂并发场景下,推荐使用`java.util.concurrent`包中的并发集合以提高性能。
163 3
[UDS] --- SecurityAccess 0x27 service
[UDS] --- SecurityAccess 0x27 service
556 0
使用mybatis-generator-plugin自动生成代码整合SpringBoot+MyBatis
使用mybatis-generator-plugin自动生成代码整合SpringBoot+MyBatis
530 0
使用mybatis-generator-plugin自动生成代码整合SpringBoot+MyBatis
Vue系列教程(18)- 集成UI框架(ElementUI)
Vue系列教程(18)- 集成UI框架(ElementUI)
499 1
AI助理

你好,我是AI助理

可以解答问题、推荐解决方案等

登录插画

登录以查看您的控制台资源

管理云资源
状态一览
快捷访问