隐藏主要是 :  withdraw()函数。        重新显示出来主要是:      update()和deiconify()函数。

来源:http://www.blog.pythonlibrary.org/2012/07/26/tkinter-how-to-show-hide-a-window/

Today we’re going to take a look at Tkinter! I was curious about how one would go about hiding a frame and then re-showing it using Tkinter and I kept finding threads (like this one) that talked about using withdraw() and deiconify() but didn’t really provide any usable code. In wxPython, I did this sort of thing using pubsub. We’ll go over three different versions of how to hide and show the root frame.

My First Example

A lot of example code for Tkinter is not very object oriented. What I mean by that is that the code I see isn’t in classes. To each their own, but I find GUI code easier to follow in a class. Anyway, that how I ended up creating my first example:

import Tkinter as Tk

########################################################################
class MyApp(object):
"""""" #----------------------------------------------------------------------
def __init__(self, parent):
"""Constructor"""
self.root = parent
self.root.title("Main frame")
self.frame = Tk.Frame(parent)
self.frame.pack() btn = Tk.Button(self.frame, text="Open Frame", command=self.openFrame)
btn.pack() #----------------------------------------------------------------------
def hide(self):
""""""
self.root.withdraw() #----------------------------------------------------------------------
def openFrame(self):
""""""
self.hide()
otherFrame = Tk.Toplevel()
otherFrame.geometry("400x300")
otherFrame.title("otherFrame")
handler = lambda: self.onCloseOtherFrame(otherFrame)
btn = Tk.Button(otherFrame, text="Close", command=handler)
btn.pack() #----------------------------------------------------------------------
def onCloseOtherFrame(self, otherFrame):
""""""
otherFrame.destroy()
self.show() #----------------------------------------------------------------------
def show(self):
""""""
self.root.update()
self.root.deiconify() #----------------------------------------------------------------------
if __name__ == "__main__":
root = Tk.Tk()
root.geometry("800x600")
app = MyApp(root)
root.mainloop()
Let’s take a few moments to break this down a little. We have a simple class where we pass in a “root” object (Tk.Tk()) as the top-level parent. This in turn is used as the parent of the Tk.Frame. The pack() command is one of the geometry managers that Tkinter comes with. It allows you to “pack” widgets into columns or rows and has various options like fill, expand and side. Next we create a Tk.Button and pack it. If you don’t call pack (or one of the other geometry managers) then your widgets won’t appear at all. In the Button instantiation process, we pass it a parent, a string for its label and a command to be run when the button is clicked. When the user clicks the button, we create another Top level window and give it a different title, size and a close button. We use the lambda anonymous method to create the callback as we need to pass the otherFrame instance to the handler so we can close it. We could have just created the otherFrame as a class property (i.e. self.otherFrame) too and skipped the lambda, but if you do much with Tkinter, then you really need to get used to seeing that kind of callback setup. When the close button is called, it destroys the otherFrame and calls the show method, which shows the original frame. Some examples say that you need to call the update() method before you call the deiconify() one, however if you comment out the update() call, you’ll see that it works fine. At least it did on Windows with Python 2.6. Now let’s try splitting the second frame into its own class! Splitting the Second Frame into a Class
Putting the second frame into it’s very own class promotes code re-use and better organization of your code, especially if the second frame were to be really complex. Here’s one way to do it: import Tkinter as Tk ########################################################################
class OtherFrame(Tk.Toplevel):
"""""" #----------------------------------------------------------------------
def __init__(self):
"""Constructor"""
Tk.Toplevel.__init__(self)
self.geometry("400x300")
self.title("otherFrame") ########################################################################
class MyApp(object):
"""""" #----------------------------------------------------------------------
def __init__(self, parent):
"""Constructor"""
self.root = parent
self.root.title("Main frame")
self.frame = Tk.Frame(parent)
self.frame.pack() btn = Tk.Button(self.frame, text="Open Frame", command=self.openFrame)
btn.pack() #----------------------------------------------------------------------
def hide(self):
""""""
self.root.withdraw() #----------------------------------------------------------------------
def openFrame(self):
""""""
self.hide()
subFrame = OtherFrame()
handler = lambda: self.onCloseOtherFrame(subFrame)
btn = Tk.Button(subFrame, text="Close", command=handler)
btn.pack() #----------------------------------------------------------------------
def onCloseOtherFrame(self, otherFrame):
""""""
otherFrame.destroy()
self.show() #----------------------------------------------------------------------
def show(self):
""""""
self.root.update()
self.root.deiconify() #----------------------------------------------------------------------
if __name__ == "__main__":
root = Tk.Tk()
root.geometry("800x600")
app = MyApp(root)
root.mainloop()
Now this is mostly the same as the first version of the code. It would be really nice to create the second frame’s button in the second frame’s class, but if we do that then it becomes hard to tell the original frame to deiconify. Still for completeness, let’s see how that would look: import Tkinter as Tk ########################################################################
class OtherFrame(Tk.Toplevel):
"""""" #----------------------------------------------------------------------
def __init__(self, original):
"""Constructor"""
self.original_frame = original
Tk.Toplevel.__init__(self)
self.geometry("400x300")
self.title("otherFrame") btn = Tk.Button(self, text="Close", command=self.onClose)
btn.pack() #----------------------------------------------------------------------
def onClose(self):
""""""
self.destroy()
self.original_frame.show() ########################################################################
class MyApp(object):
"""""" #----------------------------------------------------------------------
def __init__(self, parent):
"""Constructor"""
self.root = parent
self.root.title("Main frame")
self.frame = Tk.Frame(parent)
self.frame.pack() btn = Tk.Button(self.frame, text="Open Frame", command=self.openFrame)
btn.pack() #----------------------------------------------------------------------
def hide(self):
""""""
self.root.withdraw() #----------------------------------------------------------------------
def openFrame(self):
""""""
self.hide()
subFrame = OtherFrame(self) #----------------------------------------------------------------------
def show(self):
""""""
self.root.update()
self.root.deiconify() #----------------------------------------------------------------------
if __name__ == "__main__":
root = Tk.Tk()
root.geometry("800x600")
app = MyApp(root)
root.mainloop()
Note that in this version, we have to pass the instance of the MyApp class to the other frame so we can call its show method. You can also see that we no longer need the lambda function since we don’t need to pass the other frame instance to the handler any more. That makes things simpler. Still this is a fragile way of doing things. Why? Well if you decide to change the main frame’s show method to showFrame or anything else, then you have to remember to change it in the other class too or it breaks. This can get tedious very quickly if you are passing instances around to multiple classes. Fortunately there’s a simple solution and it’s called pubsub! Using pubsub to Communicate Between Tkinter Windows
You’ll need to go to pubsub’s website and install the package as it’s not included with Python. It IS included with wxPython, although I don’t think you can really use that version outside of wxPython very easily. Anyway, once you have it, you can follow along with this code: from pubsub import pub
import Tkinter as Tk ########################################################################
class OtherFrame(Tk.Toplevel):
"""""" #----------------------------------------------------------------------
def __init__(self):
"""Constructor"""
Tk.Toplevel.__init__(self)
self.geometry("400x300")
self.title("otherFrame") # create the button
btn = Tk.Button(self, text="Close", command=self.onClose)
btn.pack() #----------------------------------------------------------------------
def onClose(self):
"""
closes the frame and sends a message to the main frame
"""
self.destroy()
pub.sendMessage("otherFrameClosed", arg1="data") ########################################################################
class MyApp(object):
"""""" #----------------------------------------------------------------------
def __init__(self, parent):
"""Constructor"""
self.root = parent
self.root.title("Main frame")
self.frame = Tk.Frame(parent)
self.frame.pack() btn = Tk.Button(self.frame, text="Open Frame", command=self.openFrame)
btn.pack() pub.subscribe(self.listener, "otherFrameClosed") #----------------------------------------------------------------------
def listener(self, arg1, arg2=None):
"""
pubsub listener - opens main frame when otherFrame closes
"""
self.show() #----------------------------------------------------------------------
def hide(self):
"""
hides main frame
"""
self.root.withdraw() #----------------------------------------------------------------------
def openFrame(self):
"""
opens other frame and hides main frame
"""
self.hide()
subFrame = OtherFrame() #----------------------------------------------------------------------
def show(self):
"""
shows main frame
"""
self.root.update()
self.root.deiconify() #----------------------------------------------------------------------
if __name__ == "__main__":
root = Tk.Tk()
root.geometry("800x600")
app = MyApp(root)
root.mainloop()
As might be expected, this code just integrate the pubsub stuff. We create a listener method in our main frame and we “register” it by calling pub.subscribe(self.listener, "otherFrameClosed")
The “signature” is otherFrameClosed. So if we publish a message with that signature, then the main frame and any other class that has subscribed to that signature will call their respective methods. In the other frame, we add a pub.sendMessage call to the end of our close method where we publish to that aforementioned signature and we pass along a dummy argument. You don’t have to do that, but I thought it would be better if you knew how to pass information between classes. You can pass pretty much any Python object / type that you want to. Wrapping Up
Now you know a little bit about Tkinter and a few of its top-level methods. You can make your frames disappear and reappear on command! You have also gotten a taste of the power of pubsub. Go forth and code with this new knowledge! Additional Resources
Tkinter Toplevel information from effbot
zetcode’s Tkinter tutorial
Python lambda
Fredrik Lundh’s intro to Tkinter

Tkinter隐藏窗口再让他显示出来的例子的更多相关文章

  1. 怎样让窗口不显示在任务栏和ALT+TAB中(隐藏窗口再嵌套,几乎是万能的办法)

    之前想弄个像QQ旋风那样的悬浮窗口,就研究了下怎么让窗口不显示在任务栏中,方法其实很简单就是将窗口的扩张属性设置成WS_EX_TOOLWINDOW,MSDN中对该属性有详细介绍,代码如下: ::Set ...

  2. MFC无闪烁隐藏窗口

    今天需要用到将窗口的程序隐藏,但是如果在OnInitDialog()中,直接写: ShowWindow(SW_HIDE); 是无效的,因为这本身就是个初始化窗口函数.也就是说,窗口在并没有显示出来的时 ...

  3. 在非UI线程中更改UI(Delphi使用隐藏窗口来处理,QT使用信号槽)

    在Delphi里我记得是使用TThread.Synchronize(TThreadMethod),原理是利用了一个隐藏窗口来处理. 在QT Debug模式一下,碰到了同样的问题,显示错误: canno ...

  4. 将VirtualBox里安装的虚拟机在后台运行方法(在状态栏隐藏窗口)

    由于工作和学习需要,经常要开一个虚拟机开测试和开发,虚拟机我选择Oracle公司的VirtualBox(用了几年了,感觉不错的一款产品),经常开着这个窗口感觉有些浪费资源,这样隐藏窗口就在需求了. 将 ...

  5. 在Form Load中设置showInTaskBar =false 或 隐藏窗口 this.Hide()时会导致注册的全局快捷键无效

    在Form Load中设置showInTaskBar =false   或 隐藏窗口 this.Hide() 会导致注册的全局快捷键无效.  反正是其中一个,有点记不清了. 在Form Shown中s ...

  6. C# 实现窗口程序winform像QQ一样靠近桌面边缘自动隐藏窗口

    实现原理: 实现这个功能的原理步骤如下: 1.判断窗体程序是否靠近桌面边缘: 2.获取桌面屏幕大小与窗体程序大小: 3.把窗体程序显示在桌面以外隐藏起来,预留部分窗体方便用户拉出程序: 4.判断鼠标是 ...

  7. 实现:调用API函数ShowWindow()来隐藏窗口

    只需要将相应代码复制即可. 代码如下: #include <iostream> #include <windows.h> int main() { HWND hDos; //声 ...

  8. electron开发 - mac关闭和隐藏窗口

    针对mac平台的app let willQuitApp = false; // 控制退出方式 mainWindow.on('close', (e) => { if (willQuitApp) { ...

  9. C# 显示、隐藏窗口对应的任务栏

    WPF中全屏窗口,会自动隐藏任务栏. 那非全屏窗口如何隐藏任务栏?甚至有没有一种场景,隐藏任务后自定义一套系统任务栏来显示? 以下会分阶段讲述一些概念 1. 主屏任务栏 任务栏,其实也是一个窗口,主屏 ...

随机推荐

  1. C#开发中遇到问题常问题

    1.C# decimal 赋值null decimal?是可空类型就是可以将值设置为Null,decimal 不能设置为null 2.var 定义变量 VAR 是3.5新出的一个定义变量的类型其实也就 ...

  2. linux设备驱动程序注冊过程具体解释

    Linux的驱动程序注冊过程,大致分为两个步骤: 模块初始化 驱动程序注冊 以下以内核提供的演示样例代码pci-skeleton.c,具体说明一个pci设备驱动程序的注冊过程.其它设备的驱动代码注冊过 ...

  3. Swift - 多线程实现方式(3) - Grand Central Dispatch(GCD)

    1,Swift继续使用Object-C原有的一套线程,包括三种多线程编程技术: (1)NSThread (2)Cocoa NSOperation(NSOperation和NSOperationQueu ...

  4. 十天学习PHP之第三天

    1)按右边的结构:查看改动表结构  2)按右边的浏览:查看表中的数据  3)按右边的SQL:执行SQL语句  4)按右边的插入:插入一行记录  5)按右边的清空:删除表中全部记录  6)按右边的删除: ...

  5. iOS开发技巧 -- 复用代码片段

    如果你是一位开发人员在开发过程中会发现有些代码无论是在同一个工程中还是在不同工程中使用率会很高,有经验的人会直接封装在一个类里,或者写成一个宏定义或者把这些代码收集起来,下次直接使用,或者放到xcod ...

  6. C# - 重写虚方法

    项目目录: 创建教师类(Teacher),虚方法有Teach(); 创建学生类(Student),重写的方法是Teach(); 教师类: · 加上关键字 Virtual 就是声明可以重写此方法. us ...

  7. HTML5 实现拖拽

    如图 可以从第一个方框拖拽花色到第二个方框中. 也可以再拖动回来. 具体代码实现 index.html <!DOCTYPE HTML> <html> <head> ...

  8. Codeforces 282E Sausage Maximization(字典树)

    题目链接:282E Sausage Maximization 题目大意:给定一个序列A.要求从中选取一个前缀,一个后缀,能够为空,当时不能重叠.亦或和最大. 解题思路:预处理出前缀后缀亦或和,然后在字 ...

  9. java中文排序问题(转)

    在Java中,对一个数组或列表(在本文中统称为集合)中的元素排序,是一个很经常的事情.好在Sun公司在Java库中实现了大部分功能.如果集合中的元素实现了Comparable接口,调用以下的静态(st ...

  10. ME21N增强提示警告消息

    在ME21N增强中,可以使用message的方法提示错误的消息,但警告消息使用message则提示不了,需要使用系统宏mmpur_message 提示. data:begin of lw_equp, ...