Tkinter制作登录界面以及登陆后页面切换

news2025/1/16 1:35:32

Tkinter制作登录界面以及登陆后页面切换

  • 前言
  • 序言
    • 1. 由来
    • 2. 思路
    • 3. 项目结构描述
    • 4. 项目实战
      • 1. 登录界面实现(代码)
      • 2. 首页界面实现(代码)
      • 3. 打包build.py(与main.py同级目录)
      • 4. 打包安装包

前言

本帖子,默认您已了解Tkinter的基础操作,以及原理,文中仅会对部分逻辑描述,不会对Tkinter讲解.

序言

1. 由来

入职以来,很长时间都是在做网页版的后台,用到最多的就是Java语言,框架主要是:Spring Boot,Spring Cloud 。中间件大多用:Mq,Redis,Nacos等,但是随着项目的发展,网页版已经无法满足客户的需求以及业务生态的发展,需要增加客户端应用,由于公司没有此类开发经验,重担就留给了俺,一个小趴菜(报头痛哭)…好了,废话不多说,开搞。说到客户端第一时间想到的就是python的PyQt5,因为以前了解过。但是随着深入了解,好是好,功能也齐全,但是由于作者比较笨,也比较懒实在不想去烧脑学习了,就盯上了较为简单快捷的Tkinter。

2. 思路

制作客户端主要流程被我分为(不考虑后台服务情况下):明白需求(了解业务)、设计架构(主要是使用的语言以及怎么模块化开发,方便后续升级管理)、打包部署(Inno setup 打包)、测试调优(测试人员)。

3. 项目结构描述

下文中出现的如: login/ui.py 这样的文件名称时则代表的是,在包login下的ui.py文件,后续可以自行修改。

4. 项目实战

1. 登录界面实现(代码)

  1. 图片展示:
    在这里插入图片描述

  2. login/ui.py

    from tkinter import *
    from tkinter.ttk import *
    
    
    class WinGUI(Tk):
        def __init__(self):
            super().__init__()
            self.__win()
            self.iconbitmap('favicon.ico')
            self.tk_input_username_label = Entry(self, )
            self.tk_input_username_label.place(relx=0.5367, rely=0.2471, relwidth=0.4000, relheight=0.1471)
            self.tk_input_password_label = Entry(self, show='*')
            self.tk_input_password_label.place(relx=0.5367, rely=0.4441, relwidth=0.4000, relheight=0.1471)
            self.tk_button_login_button = Button(self, text="立即登录", takefocus=False, )
            self.tk_button_login_button.place(relx=0.5367, rely=0.6824, relwidth=0.4000, relheight=0.1471)
            self.tk_label_register = Label(self, text="还没有账号?立即注册", anchor="center", )
            self.tk_label_register.place(relx=0.6933, rely=0.8676, relwidth=0.2250, relheight=0.0882)
            self.tk_canvas_login_canvas = Canvas(self, )
            self.tk_canvas_login_canvas.place(relx=0.0000, rely=0.0000, relwidth=0.4667, relheight=1.0000)
            self.tk_label_welcome = Label(self, text="项目介绍(自定义)", anchor="center", )
            self.tk_label_welcome.place(relx=0.4667, rely=0.0588, relwidth=0.5333, relheight=0.1471)
    
        def __win(self):
            self.title("登录界面")
            width, height = 600, 340
            screenwidth = self.winfo_screenwidth()
            screenheight = self.winfo_screenheight()
            geometry = '%dx%d+%d+%d' % (width, height, (screenwidth - width) / 2, (screenheight - height) / 2)
            self.geometry(geometry)
            self.minsize(width=width, height=height)
    
        def login(self):
            username = self.tk_input_username_label.get()
            password = self.tk_input_password_label.get()
            return username, password
    
    
    class Win(WinGUI):
        def __init__(self, controller):
            self.ctl = controller
            super().__init__()
            self.__event_bind()
            self.__style_config()
            self.ctl.init(self)
    
        def __event_bind(self):
            self.tk_input_password_label.bind('<Return>', self.ctl.login_submit)
            self.tk_button_login_button.bind('<Button-1>', self.ctl.login_submit)
            self.tk_button_login_button.bind('<Return>', self.ctl.login_submit)
            self.tk_label_register.bind('<Button-1>', self.ctl.register)
            pass
    
        def __style_config(self):
            pass
    
  3. login/control.py

    import tkinter.messagebox
    
    from login.ui import WinGUI
    
    
    class Controller:
        # 导入UI类后,替换以下的 object 类型,将获得 IDE 属性提示功能
        ui: WinGUI
    
        def __init__(self):
            self.url = "登录地址"
    
        def init(self, ui):
            """
            得到UI实例,对组件进行初始化配置
            """
            self.ui = ui
    
        def close_windows(self):
            print("点击了菜单")
            if self.ui:
                self.ui.destroy()
    
        def version(self):
            print("点击了菜单")
            tkinter.messagebox.showinfo("版本信息", "当前版本: V1.0.0")
    
        def login_submit(self, evt):
            u, p = self.ui.login()
            if u is None or len(u) == 0:
                tkinter.messagebox.showinfo("登录提示", "请输入登录用户名!")
            elif p is None or len(p) == 0:
                tkinter.messagebox.showinfo("登录提示", "请输入登录密码!")
            print(
                f"触发了登录操作,地址:{self.url},账号:{u},密码:{p}")
            # 模拟登录操作
            if u == 'admin' and p == 'admin':
                print(f'登录成功')
                # 关闭登录窗口
                self.ui.destroy()
                from home.control import Controller as HomeUIController
                from home.ui import Win as MainWin
                # 这里创建首页界面,并进入GUI循环,可以将登陆后的Token统一管理,不想管理的可以当作参数传过去不过比较麻烦
                app = MainWin(HomeUIController())
                app.mainloop()
            else:
                tkinter.messagebox.showinfo("登录提示", "登陆失败了,请检查账号密码是否正确.")
    
        def register(self, evt):
            print(f"触发了注册操作,地址:{self.url}")
    
    
  4. 启动类main.py

    # 导入窗口控制器
    from login.control import Controller as MainUIController
    # 导入布局文件
    from login.ui import Win as MainWin
    
    if __name__ == "__main__":
        app = MainWin(MainUIController())
        app.iconbitmap('favicon.ico')
        # 启动
        app.mainloop()
    

2. 首页界面实现(代码)

  1. 图片展示
    在这里插入图片描述

  2. home/ui.py

    from tkinter import *
    from tkinter.ttk import *
    
    
    class WinGUI(Tk):
        def __init__(self):
            super().__init__()
            self.__win()
            # 当前页数
            self.index = 1
            self.total = 10
            self.data = []
            self.tk_table_m1ef1meg = self.__tk_table_m1ef1meg(self)
            self.tk_label_m1ef4id7 = self.__tk_label_m1ef4id7(self)
            self.tk_input_query = self.__tk_input_query(self)
            self.tk_button_query = self.__tk_button_query(self)
            self.tk_button_head = self.__tk_button_head(self)
            self.tk_button_previous__page = self.__tk_button_previous__page(self)
            self.tk_button_next_page = self.__tk_button_next_page(self)
            self.tk_button_last = self.__tk_button_last(self)
    
        def __win(self):
            self.title("项目名称")
            # 设置窗口大小、居中
            width = 800
            height = 600
            screenwidth = self.winfo_screenwidth()
            screenheight = self.winfo_screenheight()
            geometry = '%dx%d+%d+%d' % (width, height, (screenwidth - width) / 2, (screenheight - height) / 2)
            self.iconbitmap('favicon.ico')
            self.geometry(geometry)
    
            self.minsize(width=width, height=height)
    
        def scrollbar_autohide(self, vbar, hbar, widget):
            """自动隐藏滚动条"""
    
            def show():
                if vbar: vbar.lift(widget)
                if hbar: hbar.lift(widget)
    
            def hide():
                if vbar: vbar.lower(widget)
                if hbar: hbar.lower(widget)
    
            hide()
            widget.bind("<Enter>", lambda e: show())
            if vbar: vbar.bind("<Enter>", lambda e: show())
            if vbar: vbar.bind("<Leave>", lambda e: hide())
            if hbar: hbar.bind("<Enter>", lambda e: show())
            if hbar: hbar.bind("<Leave>", lambda e: hide())
            widget.bind("<Leave>", lambda e: hide())
    
        def v_scrollbar(self, vbar, widget, x, y, w, h, pw, ph):
            widget.configure(yscrollcommand=vbar.set)
            vbar.config(command=widget.yview)
            vbar.place(relx=(w + x) / pw, rely=y / ph, relheight=h / ph, anchor='ne')
    
        def h_scrollbar(self, hbar, widget, x, y, w, h, pw, ph):
            widget.configure(xscrollcommand=hbar.set)
            hbar.config(command=widget.xview)
            hbar.place(relx=x / pw, rely=(y + h) / ph, relwidth=w / pw, anchor='sw')
    
        def create_bar(self, master, widget, is_vbar, is_hbar, x, y, w, h, pw, ph):
            vbar, hbar = None, None
            if is_vbar:
                vbar = Scrollbar(master)
                self.v_scrollbar(vbar, widget, x, y, w, h, pw, ph)
            if is_hbar:
                hbar = Scrollbar(master, orient="horizontal")
                self.h_scrollbar(hbar, widget, x, y, w, h, pw, ph)
            self.scrollbar_autohide(vbar, hbar, widget)
    
        def __tk_table_m1ef1meg(self, parent):
            # 表头字段 表头宽度
            columns = {"序列": 79, "姓名": 239, "民族": 79, "年龄": 79, "简介": 319}
            tk_table = Treeview(parent, show="headings", columns=list(columns), )
            for text, width in columns.items():  # 批量设置列属性
                tk_table.heading(text, text=text, anchor='center')
                tk_table.column(text, anchor='center', width=width, stretch=False)  # stretch 不自动拉伸
    
            tk_table.place(relx=0.0000, rely=0.1817, relwidth=0.9988, relheight=0.7500)
            return tk_table
    
        def __tk_button_next_page(self, parent):
            btn = Button(parent, text="下一页", takefocus=False, )
            btn.place(relx=0.8313, rely=0.9500, relwidth=0.0625, relheight=0.0500)
            return btn
    
        def __tk_button_previous__page(self, parent):
            btn = Button(parent, text="上一页", takefocus=False, )
            btn.place(relx=0.7462, rely=0.9500, relwidth=0.0625, relheight=0.0500)
            return btn
    
        def __tk_label_m1ef4id7(self, parent):
            label = Label(parent, text="当前第3页/总130页", anchor="center", )
            label.place(relx=0.4625, rely=0.9500, relwidth=0.1938, relheight=0.0500)
            return label
    
        def __tk_input_query(self, parent):
            ipt = Entry(parent, )
            ipt.place(relx=0.0000, rely=0.1183, relwidth=0.2500, relheight=0.0500)
            return ipt
    
        def __tk_button_query(self, parent):
            btn = Button(parent, text="搜索", takefocus=False, )
            btn.place(relx=0.2612, rely=0.1183, relwidth=0.0788, relheight=0.0500)
            return btn
    
        def __tk_button_head(self, parent):
            btn = Button(parent, text="首页", takefocus=False, )
            btn.place(relx=0.6675, rely=0.9500, relwidth=0.0625, relheight=0.0500)
            return btn
    
        def __tk_button_last(self, parent):
            btn = Button(parent, text="尾页", takefocus=False, )
            btn.place(relx=0.9187, rely=0.9500, relwidth=0.0625, relheight=0.0500)
            return btn
    
        def page_add_index(self):
            # 下一页
            if self.index < self.total:
                self.index += 1
            return self.index
    
        def page_sub_index(self):
            # 上一页
            if self.index > 1:
                self.index -= 1
            return self.index
    
        def update_table(self):
            if self.data and len(self.data) > 0:
                if self.tk_table_m1ef1meg.get_children():
                    for get_child in self.tk_table_m1ef1meg.get_children():
                        self.tk_table_m1ef1meg.delete(get_child)
                for index, datum in enumerate(self.data):
                    self.tk_table_m1ef1meg.insert("", "end", text='',
                                                  values=(
                                                      index + 1, datum.get('name', ''), datum.get('nation', ''),
                                                      datum.get('age', ''), datum.get('info', '')))
    
    
    class Win(WinGUI):
    
        def __init__(self, controller):
            self.ctl = controller
            super().__init__()
            self.__event_bind()
            self.__style_config()
            self.config(menu=self.create_menu())
            self.ctl.init(self)
    
        def create_menu(self):
        	"""
        	创建自定义的菜单,可以不要
        	"""
            menu = Menu(self, tearoff=False)
            menu.add_cascade(label="设置", menu=self.menu_m1ecaoyu(menu))
            menu.add_cascade(label="帮助", menu=self.menu_m1eccdqt(menu))
            return menu
    
        def menu_m1ecaoyu(self, parent):
            menu = Menu(parent, tearoff=False)
            menu.add_command(label="关闭界面", command=self.ctl.close_windows)
            return menu
    
        def menu_m1eccdqt(self, parent):
            menu = Menu(parent, tearoff=False)
            menu.add_command(label="版本信息", command=self.ctl.version)
            return menu
    
        def __event_bind(self):
        	"""
        	此处对组件绑定事件
        	"""
        	# 搜索框绑定回车事件,回车直接搜索
            self.tk_input_query.bind('<Return>', self.ctl.select_member_data)
            # 搜索按钮绑定点击事件
            self.tk_button_query.bind('<Button-1>', self.ctl.select_member_data)
            # 首页事件
            self.tk_button_head.bind('<Button-1>', lambda event: self.ctl.load_member_data_next(self.index))
            self.tk_button_last.bind('<Button-1>', lambda event: self.ctl.load_member_data_next(self.total))
            self.tk_button_previous__page.bind('<Button-1>',
                                               lambda event: self.ctl.load_member_data_next(self.page_sub_index()))
            self.tk_button_next_page.bind('<Button-1>', lambda event: self.ctl.load_member_data_next(self.page_add_index()))
    
            pass
    
        def __style_config(self):
            pass
    

3. 打包build.py(与main.py同级目录)

import subprocess


def main_windows_package():
    # 定义 PyInstaller 命令
    pyinstaller_command = [
        'pyinstaller', '--onefile', '-w',
        '--icon', 'favicon.ico', 'main.py']
    try:
        subprocess.run(pyinstaller_command, check=True)
    except subprocess.CalledProcessError as e:
        print(f"An error occurred while running PyInstaller: {e}")


if __name__ == '__main__':
    main_windows_package()

4. 打包安装包

各位可以自行选择工具,由于此处项目过于简单,即开即用没有做打包的脚本,后续做到需要打包时会在后面写出来,当然有需要的也可以留言,看到后可以义务帮助。

执行了 build.py 后同级会出现一个dist目录,里边会有main.exe可执行文件,这个就是程序打包后的啦,好啦,到此结束。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2166109.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【nrm】npm 注册表管理器

nrm是什么 nrm&#xff08;NPM Registry Manager&#xff09;是一个用于管理 Node.js 包管理器&#xff08;如 npm 和 Yarn&#xff09;的注册表工具。它可以帮助用户快速切换不同的 npm 源&#xff0c;以便于提高包安装的速度和效率&#xff0c;特别是在中国大陆地区&#xf…

Ubuntu23.10下处理libncurses5-dev包的安装问题

Ubuntu23.10下处理libncurses5-dev包的安装问题 导语环境准备问题和解决方案总结参考文献 导语 使用Ubuntu23.10的时候&#xff0c;遇到需要termios的场景&#xff0c;结果发现无论是codeblocks还是系统本身的gcc都无法找到term.h和curse.h&#xff0c;网上找了很多解决方案都…

了解云计算工作负载保护的重要性,确保数据和应用程序安全

云计算de小白 云计算技术的快速发展使数据和应用程序安全成为一种关键需求&#xff0c;而不仅仅是一种偏好。随着越来越多的客户公司将业务迁移到云端&#xff0c;保护他们的云工作负载&#xff08;指所有部署的应用程序和服务&#xff09;变得越来越重要。云工作负载保护&…

【stm32】TIM定时器输出比较-PWM驱动LED呼吸灯/舵机/直流电机

TIM定时器输出比较 一、输出比较简介1、OC&#xff08;Output Compare&#xff09;输出比较2、PWM简介3、输出比较通道(高级)4、输出比较通道(通用)5、输出比较模式6、PWM基本结构配置步骤&#xff1a;程序代码&#xff1a;PWM驱动LED呼吸灯 7、参数计算8、舵机简介程序代码&am…

nginx 安装(Centos)

nginx 安装-适用于 Centos 7.x [rootiZhp35weqb4z7gvuh357fbZ ~]# lsb_release -a LSB Version: :core-4.1-amd64:core-4.1-noarch Distributor ID: CentOS Description: CentOS Linux release 7.9.2009 (Core) Release: 7.9.2009 Codename: Core# 创建文件…

大模型训练:K8s 环境中数千节点存储最佳实践

今天这篇博客来自全栈工程师朱唯唯&#xff0c;她在前不久举办的 KubeCon 中国大会上进行了该主题分享。 Kubernetes 已经成为事实的应用编排标准&#xff0c;越来越多的应用在不断的向云原生靠拢。与此同时&#xff0c;人工智能技术的迅速发展&#xff0c;尤其是大型语言模型&…

智慧政务助力实现服务民生新突破

在数字化转型的浪潮中&#xff0c;中国移动紧密结合人工智能&#xff08;AI&#xff09;技术&#xff0c;推动政务服务的智能化升级。近日&#xff0c;中国移动正式发布政务大模型3.0版本&#xff0c;以科技创新提升政务效率&#xff0c;实现服务民生的新突破。 为什么…

【Android】页面启动耗时统计流程梳理

文章基于Android 11 写在前面&#xff1a; 最近的文章都会放流程图&#xff0c;时序图之类的图片&#xff0c;解释下为什么这么做&#xff1a; 图片的好处&#xff1a; 流程清晰&#xff0c;一目了然很多代码&#xff0c;如同老太太的裹脚布&#xff0c;又臭又长。影响理解&a…

Python的Pandas库学习指南

应用场景 Pandas库因其强大的数据处理和分析能力&#xff0c;在数据科学和数据分析领域有着广泛的应用。 1. 数据清洗 处理缺失数据&#xff1a;通过删除或填充缺失值来清洗数据。数据类型转换&#xff1a;将数据转换为合适的格式&#xff0c;例如将日期字符串转换为日期类型…

k8s_资源管理介绍

资源管理介绍 在k8s中&#xff0c;所有内容都抽象成资源&#xff0c;用户需要通过操作资源来管理k8s k8s本身就是一个集群系统&#xff0c;用户可以在集群中部署服务&#xff0c;在k8s集群中运行一个个的容器&#xff0c;将指定的程序部署到容器中 k8s最小的管理单元是pod&…

AI-Talk开发板之wifi scan

一、说明 AI-Talk开发板使用ESP32-C3扩展WIFI通信功能&#xff0c;与CSK6011A通过SPI接口通信。 与处理器的信号连接&#xff1a; ESP32-C3需要烧录hosted固件&#xff0c;参考&#xff1a;AI-Talk开发板更新ESP32固件_esp32 固件-CSDN博客 二、工程 1、创建项目 进入exampl…

本地编译安装|编译安装最新版postgis3.4.3版本指南

一、本地编译安装步骤介绍 本地编译&#xff0c;指的是在本地环境编译安装某个软件&#xff0c;例如&#xff0c;本文所述的最新版postgis3.4.3&#xff0c;本地是什么cpu架构&#xff0c;编译完成后&#xff0c;编译产出物就可以在其它的同cpu架构的服务器上直接适用了&#…

关于JAVA中Scanner和sout读取超时问题

1.Scanner与System.out为什么慢 Scanner读取速度慢的原因&#xff1a; 在ACM模式下&#xff0c;输入数据和输出数据时&#xff0c;他会将数据放在一个文件里面 在new上一个Scanner时&#xff0c;去调用一个next&#xff08;&#xff09;方法&#xff0c;他会去访问IO设备&…

【mac开发入坑指南】能让你的终端好用一万倍的神仙组合iTerm2 + oh-my-zsh

介绍 iTerm2 iTerm2是默认终端的替代品&#xff0c;也是目前Mac系统下最好用的终端工具&#xff0c;集颜值和效率于一身。 Oh-My-Zsh Oh My Zsh 是一款社区驱动的命令行工具&#xff0c;正如它的主页上说的&#xff0c;Oh My Zsh 是一种生活方式。 它基于Zsh 命令行&#xff0c…

docker-文件复制(docker ps:用于在Docker主机和容器之间拷贝文件或目录)

文章目录 1、把宿主机的文件复制到容器内部1.1、查询 宿主机 root 下的文件1.2、docker cp /root/anaconda-ks.cfg spzx-redis:/root1.3、查看 spzx-redis 容器 中/root目录下是否有 anaconda-ks.cfg 文件 2、把容器中的文件 复制 到宿主机中2.1、查看 spzx-redis 容器 / 下的文…

怎么批量制作文本或链接静态码?批量静态码在线的生成技巧

怎么将文本或者链接做成静态二维码呢&#xff1f;有很多的场景都会使用静态二维码&#xff0c;在需要制作大量静态二维码时&#xff0c;有什么方法能够快速提高二维码制作效率呢&#xff1f;在网上能够找到在线二维码生成器的功能来快速生成二维码&#xff0c;下面通过这篇文章…

【shell脚本8】Shell脚本学习--其他

目录 ​编辑 Shell输入输出重定向 重定向深入讲解 Here Document Shell输入输出重定向 Unix 命令默认从标准输入设备(stdin)获取输入&#xff0c;将结果输出到标准输出设备(stdout)显示。一般情况下&#xff0c;标准输入设备就是键盘&#xff0c;标准输出设备就是终端&…

JAVA-StringBuilder和StringBuffer

一、认识String类 1.认识 String在Java中是字符串类型&#xff0c;但与其他类型不同。它是一个类&#xff0c;可以创建对象的类。与int、char等自待类型有些许不同。但它仍然是java提供的一种类型。 类中有4个属性&#xff0c;这里主要认识一下value属性。它是实际存放字符串…

AI 赋能大模型:从 ChatGPT 到国产大模型的角逐与发展契机

在当今科技飞速发展的时代&#xff0c;大模型作为人工智能领域的关键技术&#xff0c;正引发着深刻的变革。它们在自然语言处理、计算机视觉、语音识别等众多领域展现出了惊人的潜力&#xff0c;为各行各业带来了前所未有的机遇和挑战。本文将深入剖析大模型的技术原理、市场态…

字符串的join和os.path.join()

>>> key "".join("I love China.")>>> keyI love China.>>> key.join("--xwf")-I love China.-I love China.xI love China.wI love China.f为什么执行key.join("--xwf")的结果不是“I love China.--xwf”…