python:Tkinter 开发邮件客户端,能编写邮件,发送邮件带附件

news2025/6/9 15:22:53

Python Tkinter 邮件客户端

下面是一个使用 Python Tkinter 开发的简单邮件客户端,支持编写邮件和发送邮件功能:

功能说明

这个邮件客户端包含以下功能:

  1. 邮件编写功能

    • 收件人地址输入
    • 抄送地址输入
    • 邮件主题输入
    • 邮件正文编辑区(支持多行文本)
  2. 邮件发送功能

    • 通过SMTP协议发送邮件
    • 支持SSL/TLS加密
    • 显示发送状态
  3. SMTP配置

    • SMTP服务器地址
    • SMTP端口号
    • 发件人邮箱地址
    • 邮箱密码(授权码)
  4. 界面功能

    • 标签页布局(写邮件和配置)
    • 清空表单按钮
    • 状态栏显示操作状态

使用说明

  1. 在"配置"标签页中设置您的SMTP服务器信息:

    • SMTP服务器:例如 smtp.gmail.com (Gmail)
    • 端口:例如 587 (TLS) 或 465 (SSL)
    • 邮箱地址:您的完整邮箱地址
    • 密码:您的邮箱密码或应用专用密码
  2. 在"写邮件"标签页中:

    • 填写收件人地址(多个地址用逗号分隔)
    • 填写抄送地址(可选)
    • 输入邮件主题
    • 编写邮件正文
  3. 点击"发送"按钮发送邮件

注意事项

  • 大多数邮箱服务商需要使用应用专用密码而不是邮箱登录密码
  • Gmail用户需要在账户设置中启用"安全性较低的应用的访问权限"
  • 某些邮箱服务商可能需要使用不同的端口号(如465用于SSL)
  • 程序使用了SSL加密传输,保证通信安全

扩展建议

如果需要进一步完善此邮件客户端,可以考虑添加以下功能:

  1. 添加邮件附件支持
  2. 实现邮件接收功能(POP3/IMAP)
  3. 添加联系人管理
  4. 增加邮件草稿保存功能
  5. 实现邮件模板功能

要运行此程序,您需要安装 Python和 Tkinter(通常Python标准安装包含Tkinter)。
运行 python eMailClient_1.py
eMail

Python Tkinter 邮件客户端(带附件功能)

下面是在原有邮件客户端基础上增加了附件功能的完整代码:

# -*- coding: utf-8 -*-
""" 用 python Tkinter 开发一个邮件客户端,能编写邮件,发送邮件及附件 """
import os
import ssl
import tkinter as tk
from tkinter import ttk, messagebox, scrolledtext, filedialog
import smtplib
from email.mime.text import MIMEText
from email.mime.multipart import MIMEMultipart
from email.mime.application import MIMEApplication

class MailClient:
    def __init__(self, root):
        self.root = root
        self.root.title("邮件客户端 (带附件)")
        self.root.geometry("1000x650")
        self.root.resizable(True, True)
        
        # 设置主题
        self.style = ttk.Style()
        self.style.theme_use('clam')
        
        # 创建主框架
        self.main_frame = ttk.Frame(root)
        self.main_frame.pack(fill=tk.BOTH, expand=True, padx=10, pady=10)
        
        # 创建标签页
        self.tab_control = ttk.Notebook(self.main_frame)
        self.tab1 = ttk.Frame(self.tab_control)
        self.tab2 = ttk.Frame(self.tab_control)
        
        self.tab_control.add(self.tab1, text='写邮件')
        self.tab_control.add(self.tab2, text='配置')
        self.tab_control.pack(fill=tk.BOTH, expand=True)
        
        # 写邮件标签页
        self.create_compose_tab()
        
        # 配置标签页
        self.create_config_tab()
        
        # 状态栏
        self.status_var = tk.StringVar()
        self.status_var.set("就绪")
        self.status_bar = ttk.Label(root, textvariable=self.status_var, relief=tk.SUNKEN, anchor=tk.W)
        self.status_bar.pack(side=tk.BOTTOM, fill=tk.X)
        
        # 初始化配置
        self.smtp_server = "smtp.example.com"
        self.smtp_port = 587
        self.email_address = "your_email@example.com"
        self.email_password = "your_password"
        
        # 附件相关
        self.attachment_path = None
        self.attachment_name = None
        
    def create_compose_tab(self):
        # 收件人
        ttk.Label(self.tab1, text="收件人:").grid(row=0, column=0, sticky=tk.W, padx=5, pady=5)
        self.to_entry = ttk.Entry(self.tab1, width=70)
        self.to_entry.grid(row=0, column=1, columnspan=2, sticky=tk.W+tk.E, padx=5, pady=5)
        
        # 抄送
        ttk.Label(self.tab1, text="抄送:").grid(row=1, column=0, sticky=tk.W, padx=5, pady=5)
        self.cc_entry = ttk.Entry(self.tab1, width=70)
        self.cc_entry.grid(row=1, column=1, columnspan=2, sticky=tk.W+tk.E, padx=5, pady=5)
        
        # 主题
        ttk.Label(self.tab1, text="主题:").grid(row=2, column=0, sticky=tk.W, padx=5, pady=5)
        self.subject_entry = ttk.Entry(self.tab1, width=70)
        self.subject_entry.grid(row=2, column=1, columnspan=2, sticky=tk.W+tk.E, padx=5, pady=5)
        
        # 附件
        ttk.Label(self.tab1, text="附件:").grid(row=3, column=0, sticky=tk.W, padx=5, pady=5)
        
        self.attachment_frame = ttk.Frame(self.tab1)
        self.attachment_frame.grid(row=3, column=1, columnspan=2, sticky=tk.W+tk.E, padx=5, pady=5)
        
        self.attachment_label = ttk.Label(self.attachment_frame, text="未选择文件", width=60)
        self.attachment_label.pack(side=tk.LEFT, fill=tk.X, expand=True)
        
        self.add_attachment_btn = ttk.Button(self.attachment_frame, text="添加附件", 
                                            width=10, command=self.add_attachment)
        self.add_attachment_btn.pack(side=tk.RIGHT, padx=(5, 0))
        
        self.remove_attachment_btn = ttk.Button(self.attachment_frame, text="移除", 
                                               width=6, command=self.remove_attachment, 
                                               state=tk.DISABLED)
        self.remove_attachment_btn.pack(side=tk.RIGHT)
        
        # 邮件正文
        ttk.Label(self.tab1, text="正文:").grid(row=4, column=0, sticky=tk.NW, padx=5, pady=5)
        self.body_text = scrolledtext.ScrolledText(self.tab1, width=80, height=18)
        self.body_text.grid(row=4, column=1, columnspan=2, sticky=tk.W+tk.E+tk.N+tk.S, padx=5, pady=5)
        
        # 按钮框架
        button_frame = ttk.Frame(self.tab1)
        button_frame.grid(row=5, column=1, columnspan=2, sticky=tk.E, padx=5, pady=10)
        
        self.send_button = ttk.Button(button_frame, text="发送", command=self.send_email)
        self.send_button.pack(side=tk.RIGHT, padx=5)
        
        self.clear_button = ttk.Button(button_frame, text="清空", command=self.clear_form)
        self.clear_button.pack(side=tk.RIGHT, padx=5)
        
        # 配置网格权重
        self.tab1.grid_columnconfigure(1, weight=1)
        self.tab1.grid_rowconfigure(4, weight=1)
    
    def create_config_tab(self):
        # 配置说明
        ttk.Label(self.tab2, text="邮件服务器配置", font=("Arial", 10, "bold")).grid(row=0, column=0, columnspan=2, sticky=tk.W, padx=5, pady=10)
        
        # SMTP服务器
        ttk.Label(self.tab2, text="SMTP服务器:").grid(row=1, column=0, sticky=tk.W, padx=5, pady=5)
        self.smtp_entry = ttk.Entry(self.tab2, width=40)
        self.smtp_entry.grid(row=1, column=1, sticky=tk.W+tk.E, padx=5, pady=5)
        self.smtp_entry.insert(0, "smtp.example.com")
        
        # 端口
        ttk.Label(self.tab2, text="端口:").grid(row=2, column=0, sticky=tk.W, padx=5, pady=5)
        self.port_entry = ttk.Entry(self.tab2, width=10)
        self.port_entry.grid(row=2, column=1, sticky=tk.W, padx=5, pady=5)
        self.port_entry.insert(0, "587")
        
        # 邮箱地址
        ttk.Label(self.tab2, text="邮箱地址:").grid(row=3, column=0, sticky=tk.W, padx=5, pady=5)
        self.email_entry = ttk.Entry(self.tab2, width=40)
        self.email_entry.grid(row=3, column=1, sticky=tk.W+tk.E, padx=5, pady=5)
        self.email_entry.insert(0, "your_email@example.com")
        
        # 密码
        ttk.Label(self.tab2, text="密码:").grid(row=4, column=0, sticky=tk.W, padx=5, pady=5)
        self.password_entry = ttk.Entry(self.tab2, width=40, show="*")
        self.password_entry.grid(row=4, column=1, sticky=tk.W+tk.E, padx=5, pady=5)
        self.password_entry.insert(0, "your_password")
        
        # 保存配置按钮
        ttk.Button(self.tab2, text="保存配置", command=self.save_config).grid(row=5, column=1, sticky=tk.E, padx=5, pady=10)
        
        # 配置说明
        note_frame = ttk.LabelFrame(self.tab2, text="配置说明")
        note_frame.grid(row=6, column=0, columnspan=2, sticky=tk.W+tk.E, padx=5, pady=10)
        
        note_text = """
常用邮箱配置参考:
- Gmail: 
  SMTP服务器: smtp.gmail.com
  端口: 587 (TLS) 或 465 (SSL)
  需要启用"安全性较低的应用的访问权限"
  
- 网易邮箱:
  SMTP服务器: smtp.163.com
  端口: 465 (SSL) 或 994 (SSL)
  
- QQ邮箱:
  SMTP服务器: smtp.qq.com
  端口: 465 (SSL) 或 587 (TLS)
  需要生成授权码代替密码
        """
        ttk.Label(note_frame, text=note_text, justify=tk.LEFT).pack(padx=10, pady=10, fill=tk.BOTH)
        
        # 配置网格权重
        self.tab2.grid_columnconfigure(1, weight=1)
    
    def add_attachment(self):
        file_path = filedialog.askopenfilename(
            title="选择附件",
            filetypes=[("所有文件", "*.*"), 
                      ("文档", "*.doc *.docx *.pdf *.txt"), 
                      ("图片", "*.jpg *.jpeg *.png *.gif"),
                      ("压缩文件", "*.zip *.rar")]
        )
        if file_path:
            self.attachment_path = file_path
            self.attachment_name = os.path.basename(file_path)
            self.attachment_label.config(text=self.attachment_name)
            self.remove_attachment_btn.config(state=tk.NORMAL)
            self.status_var.set(f"已添加附件: {self.attachment_name}")
    
    def remove_attachment(self):
        self.attachment_path = None
        self.attachment_name = None
        self.attachment_label.config(text="未选择文件")
        self.remove_attachment_btn.config(state=tk.DISABLED)
        self.status_var.set("已移除附件")
    
    def save_config(self):
        self.smtp_server = self.smtp_entry.get()
        self.smtp_port = int(self.port_entry.get())
        self.email_address = self.email_entry.get()
        self.email_password = self.password_entry.get()
        
        messagebox.showinfo("配置保存", "邮件配置已成功保存!")
        self.status_var.set("配置已保存")
    
    def clear_form(self):
        self.to_entry.delete(0, tk.END)
        self.cc_entry.delete(0, tk.END)
        self.subject_entry.delete(0, tk.END)
        self.body_text.delete("1.0", tk.END)
        self.remove_attachment()
        self.status_var.set("表单已清空")
    
    def send_email(self):
        to_email = self.to_entry.get()
        cc_email = self.cc_entry.get()
        subject = self.subject_entry.get()
        body = self.body_text.get("1.0", tk.END)
        
        if not to_email:
            messagebox.showerror("错误", "请填写收件人地址")
            return
        
        try:
            # 创建邮件
            msg = MIMEMultipart()
            msg['From'] = self.email_address
            msg['To'] = to_email
            if cc_email:
                msg['Cc'] = cc_email
            msg['Subject'] = subject
            
            # 添加邮件正文
            msg.attach(MIMEText(body, 'plain'))
            
            # 添加附件
            if self.attachment_path:
                self.status_var.set("正在添加附件...")
                with open(self.attachment_path, "rb") as attachment:
                    part = MIMEApplication(attachment.read(), Name=self.attachment_name)
                part['Content-Disposition'] = f'attachment; filename="{self.attachment_name}"'
                msg.attach(part)
            
            # 连接SMTP服务器
            self.status_var.set("正在连接SMTP服务器...")
            context = ssl.create_default_context()
            
            with smtplib.SMTP(self.smtp_server, self.smtp_port) as server:
                server.starttls(context=context)
                self.status_var.set("正在登录邮箱...")
                server.login(self.email_address, self.email_password)
                
                # 发送邮件
                self.status_var.set("正在发送邮件...")
                recipients = [to_email]
                if cc_email:
                    recipients.extend(cc_email.split(","))
                
                server.sendmail(self.email_address, recipients, msg.as_string())
                self.status_var.set("邮件发送成功!")
                messagebox.showinfo("成功", "邮件已成功发送!")
                self.clear_form()
        
        except smtplib.SMTPAuthenticationError:
            self.status_var.set("发送失败")
            messagebox.showerror("认证失败", "邮箱地址或密码错误,请检查配置!")
        except smtplib.SMTPException as e:
            self.status_var.set("发送失败")
            messagebox.showerror("SMTP错误", f"发送邮件时出错:\n{str(e)}")
        except Exception as e:
            self.status_var.set("发送失败")
            messagebox.showerror("错误", f"发生错误:\n{str(e)}")

if __name__ == "__main__":
    root = tk.Tk()
    app = MailClient(root)
    root.mainloop()

新增功能说明

1. 附件功能

  • 添加附件:通过"添加附件"按钮可以选择本地文件作为邮件附件
  • 附件显示:显示已添加的附件文件名
  • 移除附件:可以随时移除已选择的附件
  • 附件类型:支持所有文件类型,并在文件选择对话框中分类显示常见文件类型

2. 界面改进

  • 附件区域位于主题和正文之间,清晰可见
  • 添加了附件状态显示(未选择文件/已选择文件)
  • 优化了状态栏信息,显示附件操作状态

3. 邮件发送处理

  • 使用MIMEApplication处理附件文件
  • 正确设置附件的Content-Disposition头部
  • 在发送过程中显示附件处理状态

4. 错误处理增强

  • 针对SMTP认证失败提供特定错误提示
  • 捕获并显示所有可能的异常
  • 提供更详细的错误信息

5. 配置说明

  • 添加了常用邮箱的配置说明(Gmail、网易邮箱、QQ邮箱)
  • 包含端口号和特殊要求提示

使用说明

  1. 配置邮件服务器

    • 在"配置"标签页填写您的SMTP服务器信息
    • 参考提供的常用邮箱配置示例
    • 点击"保存配置"按钮保存设置
  2. 添加附件

    • 在"写邮件"标签页点击"添加附件"按钮
    • 选择您要附加的文件
    • 文件名将显示在附件区域
    • 如需移除,点击"移除"按钮
  3. 发送邮件

    • 填写收件人、主题和邮件正文
    • 点击"发送"按钮发送邮件
    • 发送状态将显示在底部状态栏

注意事项

  1. 安全考虑

    • 密码字段使用掩码显示
    • 使用SSL/TLS加密传输
    • 建议使用应用专用密码而不是邮箱登录密码
  2. 附件限制

    • 大附件可能因邮件服务器限制而发送失败
    • 某些邮箱服务商有附件大小限制(如25MB)
  3. 特殊要求

    • Gmail用户需要启用"安全性较低的应用的访问权限"
    • QQ邮箱需要使用授权码代替密码
  4. 错误处理

    • 程序会捕获并显示详细的错误信息
    • 认证失败会提供特定提示
    • 发送过程中会显示状态更新

这个增强版的邮件客户端现在可以满足基本的邮件发送需求,包括添加附件功能,使用起来更加实用和便捷。
在这里插入图片描述

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

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

相关文章

深入解析Java21核心新特性(虚拟线程,分代 ZGC,记录模式模式匹配增强)

文章目录 前言一、虚拟线程 (Virtual Threads - JEP 444) - 并发的革命1.1 解决的核心问题🎯1.2 工作原理与核心机制⚙️1.3 使用详解与最佳实践🛠️1.4 注意事项⚠️1.5 总结 📚 二、分代 ZGC (Generational ZGC - JEP 439) - 低延迟新高度2…

免费批量去水印工具 - 针对文心一言生成图片

免费批量去水印工具 - 针对文心一言生成图片 工具介绍 这是一款免费的批量去水印工具,专门针对文心一言生成的图片进行处理。通过简单的操作,您可以快速去除图片中的水印。 下载链接 您可以通过以下网盘链接下载工具: 链接: https://pa…

在WPS中如何启用宏VBA wps.vba.exe下载和安装

首先我们点击导航栏中的【工具】,点击左侧 运行宏,根据提示 点击 立即加载。加载卡在50%时间比较长,耐心等待。 关闭wps重新打开后, word和xls表格都可以使用了。 如果电脑无法联网,需要提前下载 WPS VBA插件 WPS VB…

Hardware-Efficient Attention for Fast Decoding

TL;DR 2025 年普林斯顿大学提出的硬件友好 attention 设计,在 MQA/GQA 与 deepseek 提出的 MLA 基础之上继续优化,提出 Grouped-Tied Attention (GTA) 和 Grouped Latent Attention (GLA),实现更高推理效率的同时也能保持较好的模型效果。 …

LLMs 系列科普文(13)

十三、AlphaGO 提到强化学习的历史,不得不提到 alphago,如果你不记得这是什么了,那你是否还曾记得,早些年 AI 已经可以在围棋中击败人类选手了。 AlphaGO 系统又 DeepMind 公司开发,你可以在网络上找到当初人机大战的…

element-plus 单选组件 el-radio,选不上,又没报错,直接复制官网也不行解决方案

在使用 Vue 框架开发项目时,Element UI 是常用的组件库。最近在开发中遇到了 Element 单选框组件el-radio的双向绑定问题,直接复制element官网上的的案例下来也是不得,经过调试和探索,终于找到了解决方案,特此记录分享…

idea 启动jar程序并调试

添加一个JAR 应用程序,填写以下内容: JAR路径:填写你要启动的jar程序的绝对路径 虚拟机选项:-Xmx1G -Xms1G -agentlib:jdwptransportdt_socket,servery,suspendn,address*:5005 程序实参(可选,minecraft专用…

CSS 轮廓(Outline)与边框(Border)的深度解析

在 CSS 中,轮廓(outline)和边框(border)是两个用于装饰元素的重要属性,但它们在功能、渲染机制和应用场景上存在显著差异。下面从多个维度进行详细对比: 一、基础定义与语法差异 边框&#xf…

Docker 部署 Python 的 Flask项目

文章目录 一、构建运行 Docker 容器1. 查找合适镜像2.本地docker 拉取镜像3.项目配置1. python项目下生成 requirements.txt 依赖文件2. 生成Dockerfile文件3.忽略不必要文件4. 构建镜像 4. 运行容器5.测试 二、常见问题与解决方案 一、构建运行 Docker 容器 1. 查找合适镜像 …

Vue入门到实战之第一篇【超基础】

Vue入门到实战之第一篇 学习路线1. Vue 概念1.1 Vue 是什么 2. 创建Vue实例,初始化渲染3. 插值表达式 {{ }}4. Vue响应式特性5. 开发者工具 学习路线 1. Vue 概念 1.1 Vue 是什么 概念: Vue是一个用于 构建用户界面1 的 渐进式2 框架3 1:基…

实时数据分析的技术架构:Lambda vs Kappa架构选择

文章目录 引言:实时数据分析架构的重要性Lambda架构深度解析Kappa架构技术特性架构对比分析维度性能与可扩展性评估技术栈选型指南实际应用场景分析成本效益对比模型混合架构与演进策略企业级决策框架最佳实践与案例研究技术趋势与未来展望引言:实时数据分析架构的重要性 在…

springboot2.x升级springboot3.x

springboot2.x升级springboot3.x 背景升级jdk版本为17以上springboot版本修改javax包更新mybatis-plus升级swagger升级springdocspringdoc配置 背景 当前项目是springboot2.5.9版本的springbootmybatis-plus项目,需要升级到springboot3.5.0项目。 升级jdk版本为17…

Python训练打卡Day43

复习日 1.卷积神经网络的基本概念 2.kaggle找到一个图像数据集,用cnn网络进行训练并且用grad-cam做可视化 进阶:并拆分成多个文件 tips:注册kaggle的注意事项 安装插件:Header Editor 然后打开扩展选项: 输入网址:ht…

227.2018年蓝桥杯国赛 - 交换次数(中等)- 贪心

227. 交换次数(贪心) 1. 2018年蓝桥杯国赛 - 交换次数(中等) 标签:2018 暴力 国赛 1.1 题目描述 IT 产业人才需求节节攀升。业内巨头百度、阿里巴巴、腾讯(简称 BAT )在某海滩进行招聘活动。…

STM32入门学习之系统时钟配置

1. 时钟就是单片机的心脏。单片机根据时钟频率来控制每个部件的工作,时钟是单片机的脉搏,决定了每条命令运行的速率,没有时钟单片机将停止工作。 如何理解“时钟决定了单片机每条命令运行的速率”? 首先需要去理解单片机中的时…

【ArcGIS Pro微课1000例】0072:如何自动保存编辑内容及保存工程?

文章目录 一、自动保存编辑内容二、自动保存工程在使用ArcGIS或者ArcGIS Pro时,经常会遇到以下报错,无论点击【发送报告】,还是【不发送】,软件都会强制退出,这时如果对所操作没有保存,就会前功尽弃。 此时,自动保存工作就显得尤为重要,接下来讲解两种常见的自动保存方…

AU音频软件|Audition 2025网盘下载与安装教程指南

说起AU,有些小伙伴可能第一印象是化学元素金(Aurum)。实际上,本文要介绍的AU,全称是Adobe Audition,是一款专业音频编辑和混音软件‌,广泛应用于音乐制作、广播、电影及视频声音设计等领域。 目…

网络编程(TCP编程)

思维导图 1.基础流程 流程图中是TCP连接的基础步骤&#xff0c;其他操作都是在此基础上进行添加修改。 2.函数接口 2.1 创建套接字&#xff08;socket&#xff09; int socket(int domain, int type, int protocol); 头文件&#xff1a;#include <sys/types.h> …

热成像实例分割电力设备数据集(3类,838张)

在现代电力系统的运维管理中&#xff0c;红外热成像已经成为检测设备隐患、预防故障的重要手段。相比传统可见光图像&#xff0c;红外图像可揭示设备温度分布&#xff0c;从而更直观地反映过热、老化等问题。而在AI赋能下&#xff0c;通过实例分割技术对热成像中的电力设备进行…

用电脑通过USB总线连接控制keysight示波器

通过USB总线控制示波器的优势 在上篇文章我介绍了如何通过网线远程连接keysight示波器&#xff0c;如果连接的距离不是很远&#xff0c;也可以通过USB线将示波器与电脑连接起来&#xff0c;实现对示波器的控制和截图。 在KEYSIGHT示波器DSOX1204A的后端&#xff0c;除了有网口…