Tkinter利用Frame实现界面切换


最近有一个需要,用python搭建一个测试软件。需要三种测试功能,三种功能所需要的参数还不一样,所以操作界面也不一样。在网上看了一些教程,感觉自己水平不够,看不太懂,就自己琢磨了一下,实现了三个界面互相切换的功能,在这里记录一下。(Python3做的)

我的主界面只有输入参数的界面需要切换,其他界面不需要改变。因此我把这个界面所有的组件都放在一个Frame中,利用destroy方法摧毁一个,再新放置另一个Frame,从而实现界面的切换。下面是我截取的界面部分代码,可能多少有点格式的问题,需要小修改。另外我已经把所有按键的触发函数去掉了。后面简单的分析一下代码。

import tkinter as tk
#---------主窗口初始化---------
def main_window_init(size = '900x556'):
    my_window = tk.Tk() #新建一个主窗口
    my_window.title('探针测试系统') #窗口名
    my_window.geometry(size) #窗口大小
    my_window.resizable(width = False,height = False) #是否可以拉伸
    #my_window.iconbitmap('001.ico') #图标
    return my_window

#---------创建IV测试参数输入界面---------
def IV_input_window_init():

    #----------参数输入窗口---------
    Frame_input = tk.Frame(main_window) #放置所有组件的框架

    tk.Label(Frame_input,text='开始电压(V):',bg='#DDDDDD',width=12).grid(row=0,column=0) #起始电压文本
    Entry_v_start = tk.Entry(Frame_input,bg='#DDDDDD',width=12) #起始电压输入控件
    Entry_v_start.grid(row=0,column=1)

    tk.Label(Frame_input,text='结束电压(V):',bg='#DDDDDD',width=12).grid(row=1,column=0) #结束电压文本
    Entry_v_stop = tk.Entry(Frame_input,bg='#DDDDDD',width=12) #结束电压输入控件
    Entry_v_stop.grid(row=1,column=1)

    tk.Label(Frame_input,text='步长(V):',bg='#DDDDDD',width=12).grid(row=2,column=0) #步长
    Entry_step = tk.Entry(Frame_input,bg='#DDDDDD',width=12) #步长输入控件
    Entry_step.grid(row=2,column=1)

    tk.Label(Frame_input,text='延时(ms):',bg='#DDDDDD',width=12).grid(row=3,column=0) #延时
    Entry_delay = tk.Entry(Frame_input,bg='#DDDDDD',width=12) #延时
    Entry_delay.grid(row=3,column=1)

    tk.Label(Frame_input,text='限流(A):',bg='#DDDDDD',width=12).grid(row=4,column=0) #限流文本
    Entry_compliance = tk.Entry(Frame_input,bg='#DDDDDD',width=12) #限流输入控件
    Entry_compliance.grid(row=4,column=1)

    tk.Label(Frame_input,text='点数(个):',bg='#DDDDDD',width=12).grid(row=5,column=0) #文本
    tk.Label(Frame_input,textvariable=num_of_data,
       bg='#DDDDDD',width=12,relief='ridge').grid(row=5,column=1) #点数

    #Rbuttn_sweep_direction = tk.Checkbutton(Frame_input,text='单扫')

    tk.Button(Frame_input,text='保存',width=12, #保存参数按键
        bg='#87CEFA').grid(row=6,column=0)

    tk.Button(Frame_input,text='运行',width=12, #启动按键
        bg='#87CEFA').grid(row=6,column=1)

    tk.Button(Frame_input,text='清除图像',width=12, #清空
    bg='#87CEFA').grid(row=7,column=0)

    tk.Button(Frame_input,text='退出',width=12, #退出
    command=main_window.quit,bg='#87CEFA').grid(row=7,column=1)

    #Frame_input.place(x=60,y=200) #放置框架
    return Frame_input

#---------创建IV_t测试参数输入界面---------
def IVt_input_window_init():

    #----------参数输入窗口---------
    Frame_input = tk.Frame(main_window) #放置所有组件的框架

    tk.Label(Frame_input,text='偏置电压(V):').grid(row=0,column=0) #起始电压文本
    Entry_v_bias = tk.Entry(Frame_input,bg='#DDDDDD') #起始电压输入控件
    Entry_v_bias.grid(row=0,column=1)

    tk.Label(Frame_input,text='扫描点数:').grid(row=1,column=0) #结束电压文本
    Entry_v_stop = tk.Entry(Frame_input,bg='#DDDDDD') #结束电压输入控件
    Entry_v_stop.grid(row=1,column=1)

    tk.Label(Frame_input,text='步长(V):').grid(row=2,column=0) #步长
    Entry_step = tk.Entry(Frame_input,bg='#DDDDDD') #步长输入控件
    Entry_step.grid(row=2,column=1)

    tk.Label(Frame_input,text='延时(ms):').grid(row=3,column=0) #延时
    Entry_delay = tk.Entry(Frame_input,bg='#DDDDDD') #延时
    Entry_delay.grid(row=3,column=1)

    tk.Button(Frame_input,text='保存',width=12,height=3, #保存参数按键
        bg='#0baef4').grid(row=4,column=0)

    tk.Button(Frame_input,text='运行',width=12,height=3, #启动按键
        bg='#0baef4').grid(row=4,column=1)

    tk.Button(Frame_input,text='CLR',width=12,height=3, #清空
    bg='#0baef4').grid(row=5,column=0)

    tk.Button(Frame_input,text='退出',width=12,height=3, #退出
    command=main_window.quit,bg='#0baef4').grid(row=5,column=1)

    return Frame_input

#---------创建Endurance测试参数输入界面---------
def Endurance_input_window_init():

    #----------参数输入窗口---------
    Frame_input = tk.Frame(main_window) #放置所有组件的框架

    tk.Label(Frame_input,text='开始电压(V):').grid(row=0,column=0) #起始电压文本
    Entry_v_start = tk.Entry(Frame_input,bg='#DDDDDD') #起始电压输入控件
    Entry_v_start.grid(row=0,column=1)

    tk.Label(Frame_input,text='结束电压(V):').grid(row=1,column=0) #结束电压文本
    Entry_v_stop = tk.Entry(Frame_input,bg='#DDDDDD') #结束电压输入控件
    Entry_v_stop.grid(row=1,column=1)

    return Frame_input


#---------创建模式选择界面-----------
def testmode_switch_init():
    Frame_testmode = tk.Frame(main_window,width=900,height=20) #放置所有组件的框架

    tk.Button(Frame_testmode,text='I-V测试',width=33,height=2, #I-V测试
       command=switch_to_iv,bg='#696969',fg='#FFFFFF',
       font=("Arial",11)).grid(row=0,column=0)

    tk.Button(Frame_testmode,text='I-V-t测试',width=33,height=2, #I-V-t测试
       command=switch_to_iv_t,bg='#696969',fg='#FFFFFF',
       font=("Arial",11)).grid(row=0,column=1)

    tk.Button(Frame_testmode,text='endurance测试',width=33,height=2, #endurrance测试
       command=switch_to_endurance,bg='#696969',fg='#FFFFFF',
       font=("Arial",11)).grid(row=0,column=2)

    Frame_testmode.place(x=0,y=3)


#--------进入IV测试界面----------
def switch_to_iv():
    global intput_win1,intput_win2,intput_win3
    intput_win1.destroy()
    intput_win2.destroy()
    intput_win3.destroy()

    intput_win1 = IV_input_window_init() #输入界面初始化
    intput_win1.place(x=60,y=200) #放置框架

    print('iv')
    
#--------进入IV-t测试界面----------
def switch_to_iv_t():
    global intput_win1,intput_win2,intput_win3
    intput_win1.destroy()
    intput_win2.destroy()
    intput_win3.destroy()

    intput_win2 = IVt_input_window_init()
    intput_win2.place(x=60,y=200) #放置框架

#--------进入endurance测试界面----------
def switch_to_endurance():
    global intput_win1,intput_win2,intput_win3
    intput_win1.destroy()
    intput_win2.destroy()
    intput_win3.destroy()

    intput_win3 = Endurance_input_window_init()
    intput_win3.place(x=60,y=200) #放置框架


#----------创建matplotlib画布----------
def data_canvas_init():
    fig = plt.figure(figsize=(6,4),dpi=70) #图像大小和比例
    plt.grid()
    canvas_spice = FigureCanvasTkAgg(fig,main_window)
    canvas_spice.get_tk_widget().place(x=300,y=100) #放置位置
    canvas_spice.draw()
    return canvas_spice


#Test_parameter = parameter_class() #存储测试参数
#keithley_237 = open_instrument() #打开keithley237设备
main_window = main_window_init() #创建主窗口

#num_of_data = tk.IntVar() #保存扫描点数的变量,可以随时更新

#----------背景------------
#设置背景一定要在主函数里,否则不会显示
img_bg = tk.PhotoImage(file="pic/bg_1.gif") #打开图片
canvas_bg = tk.Canvas(main_window,width=900,height=556) #新建一个画布
canvas_bg.create_image(440,278,image = img_bg) #放置背景图片
canvas_bg.place(x=-2,y=0) #放置画布

#global intput_win1,intput_win2,intput_win3
intput_win1 = IV_input_window_init() #输入界面初始化
intput_win2 = IVt_input_window_init()
intput_win3 = Endurance_input_window_init()

intput_win1.place(x=60,y=200) #放置框架

testmode_switch_init() #测试模式选择界面初始化
canvas_data = data_canvas_init() #图形化数据部分初始化

main_window.mainloop() #开始主循环

 代码里三个Frame的布局是分别在 IV_input_window_init()、IVt_input_window_init()、Endurance_input_window_init()中完成的,布局完成后并没有放置Frame,而是返回了一个Frame。所以我在模式切换的界面,testmode_switch_init()中设置了三个按钮,分别对应了switch_to_iv()、switch_to_iv_t()和switch_to_endurance()三个响应事件。接下来是重点。

 我在主函数里用了三个全局变量intput_win1、intput_win2和intput_win3,分别对应三个Frame,开始的时候默认放置intput_win1,所以用了intput_win1.place()函数。在切换界面按键按下以后,响应函数依次摧毁intput_win1、intput_win2和intput_win3三个Frame。然后重新对着三个全局变量赋值,赋的是三个新的Frame,最后放置对应的一个Frame就可以了。这里有一点需要注意,在函数中对全局变量赋值的时候需要先用global声明一下,否则会报错。

 最后展示一下实际的效果,这个是开始的界面

 

 

按下IVT测试按键后,切换到另一个界面

 

 

 按下endurane测试按键后切换到第三个界面。

 


免责声明!

本站转载的文章为个人学习借鉴使用,本站对版权不负任何法律责任。如果侵犯了您的隐私权益,请联系本站邮箱yoyou2525@163.com删除。



 
粤ICP备18138465号  © 2018-2025 CODEPRJ.COM