tkinter内嵌Matplotlib系列(二)之函数曲线绘制

目录

前言

前一章节,我们解读了tkinter内嵌Matplotlib的教程,了解其内嵌的原理,就是在tkinter创建matplotlib的画布控件,再利用其返回的画布对象进行绘图,其他附加功能,使用tkinter控件实现。

(一)对matplotlib画布的封装:

(1)说明:

我们希望对官方的实例代码进行封装成一个函数,并返回一个画布对象,外部再调用该函数,并获取画布对象,进行绘制操作。

(2)封装后的代码:

"""
    画布文件,实现绘图区域的显示,并返回画布的对象。
"""
import tkinter as tk

# 创建画布需要的库
from matplotlib.backends.backend_tkagg import FigureCanvasTkAgg
# 创建工具栏需要的库
from matplotlib.backends.backend_tkagg import NavigationToolbar2Tk
# 快捷键需要的库
from matplotlib.backend_bases import key_press_handler
# 导入画图常用的库
from matplotlib.figure import Figure

def plot_fun(root):
    """
        该函数实现的是内嵌画布,不负责画图,返回画布对象。
    :param root:父亲控件对象, 一般是容器控件或者窗体
    :return: 画布对象
    """
    # 画布的大小和分别率
    fig = Figure(dpi=100)
    axs = fig.add_subplot(111)

    # 创建画布
    canvas = FigureCanvasTkAgg(fig, master=root)  # A tk.DrawingArea.
    canvas.draw()
    # 显示画布
    canvas.get_tk_widget().pack()

    # 创建工具条
    toolbar = NavigationToolbar2Tk(canvas, root)
    toolbar.update()
    # 显示工具条
    canvas.get_tk_widget().pack()

    # 调用快捷键
    def on_key_press(event):
        key_press_handler(event, canvas, toolbar)

    canvas.mpl_connect("key_press_event", on_key_press)

    # 返回画布的对象
    return axs

(二)思路分析:

1.需求说明:

(1)背景:

作为学生的我们,你是否有那么一个场景,唉……,这个数学函数好难求哦,要是知道它的图像这么画就好了。

(2)需求:

给出数学表达式,绘制出该数学表达式的函数曲线,一来可以观察函数的变化趋势,二来可以根据两条曲线的交点,来求解出方程的大致结果。

2.框架的设置:

(1)说明:

分模块编程,向来是众人所提倡的,再python里更是很好的实现。

再动手敲代码之前,我们先来大致的设置一下,小项目的框架。

(2)框架图解:

01.png

3.文件说明:

(1)main.py

主程序文件,负责程序的启动与结束和窗体的大致设置。

(2)widget.py

控件文件,负责程序控件的创建与布局

(3)figure.py

画布文件,实现绘图区域的显示,并返回画布的对象。

(4)plot.py

绘图文件,负责函数曲线的绘制

(三)各文件的源代码

1.main.py

"""
    主程序文件,负责程序的启动与结束和窗体的大致设置。
"""

import tkinter as tk
import widget


def win_w_h(root):
    """
        控制窗口的大小和出现的位置
    :param root:
    :return: 窗口的大小和出现的位置
    """
    # 设置标题:
    win.title("数学函数绘图")

    # 绘图区标签
    label_plot = tk.Label(root, text="绘     图       区",
                          font=("微软雅黑", 10), fg="blue")
    label_plot.place(relx=0.26, rely=0)

    label_func = tk.Label(root, text="功     能       区",
                          font=("微软雅黑", 10), fg="blue")
    label_func.place(relx=0.75, rely=0)
    # 获取屏幕的大小;
    screen_height = root.winfo_screenheight()
    screen_width = root.winfo_screenwidth()
    # 窗体的大小
    win_width = 0.8 * screen_width
    win_height = 0.8 * screen_height
    # 窗体出现的位置:控制的是左上角的坐标
    show_width = (screen_width - win_width) / 2
    show_height = (screen_height - win_height) / 2

    # 返回窗体 坐标
    return win_width, win_height, show_width, show_height


win = tk.Tk()
# 大小 位置
win.geometry("%dx%d+%d+%d" % (win_w_h(win)))


# 创建一个容器, 没有画布时的背景
frame1 = tk.Frame(win, bg="#c0c0c0")
frame1.place(relx=0.00, rely=0.05, relwidth=0.62, relheight=0.89)


# 控件区
frame2 = tk.Frame(win, bg="#808080")
frame2.place(relx=0.62, rely=0.05, relwidth=0.38, relheight=0.89)

# 调用控件模块
widget.widget_main(win, frame2)
win.mainloop()

2.widget.py

"""
    控件文件,负责程序控件的创建与布局
"""
import tkinter as tk
# 对话框所需的库
import tkinter.messagebox as mb
# 画布文件
import figure
# 绘图文件
import plot


def widget_main(win, root):
    """
        负责程序控件的创建与布局
    :param win: 主窗体的对象。
    :param root: 绘图区的容器对象。
    :return: 无
    """
    # 控件区的容器对象
    frame1 = None


# ===========功能区============================
    # 绘图的功能函数
    def plot_f():
        string = entry.get()
        # 判断输入框是否为空
        if string == "":
            mb.showerror("提示", "没有输入值,请重新输入:")
        else:
            # 判断是否已经创建画布
            if frame1==None:
                mb.showerror("提示", "没有创建画布,不能画图,请先创建画布")
            else:
                axs = figure.plot_fun(frame1)
                plot.plot_main(string, axs)

    # 清除的功能函数
    def clear():
        nonlocal frame1
        entry.delete(0, "end")
        if frame1==None:
            mb.showerror("提示", "已经没有画布,无法清除画布")
        else:
            frame1.destroy()
            frame1 = None

    # 创建画布的功能函数
    def create():
        nonlocal frame1
        if frame1 != None:
            mb.showerror("提示", "画布已经存在,请不要重复创建画布")
        else:
            frame1 = tk.LabelFrame(win, bg="#80ff80", text="画-----布", labelanchor="n", fg="green")
            frame1.place(relx=0.00, rely=0.05, relwidth=0.62, relheight=0.95)


# =============控件区======================
    #  标签控件
    label = tk.Label(root,
                     text="请输入含x的数学公式:",
                     font=("微软雅黑", 18),
                     fg="blue")
    label.place(relx=0.2, rely=0.1)

    # 输入框
    entry = tk.Entry(root, font=("华文楷体", 15))
    entry.place(relx=0.1, rely=0.2, relwidth=0.8)

    # 创建画布区
    btn_draw = tk.Button(root,
                         text="创建",
                         cursor="hand2",
                         width=10,
                         bg="orange",
                         relief="raised",
                         command=create
                         )
    btn_draw.place(relx=0.1, rely=0.3)

    # 绘图按钮
    btn_draw = tk.Button(root,
                         text="绘图",
                         cursor="hand2",
                         width=10,
                         bg="green",
                         relief="raised",
                         command=plot_f
                         )
    btn_draw.place(relx=0.4, rely=0.3)

    # 清除按钮
    btn_clear = tk.Button(root,
                          text="清除",
                          cursor="hand2",
                          width=10,
                          bg="yellow",
                          relief="raised",
                          command=clear
                          )
    btn_clear.place(relx=0.7, rely=0.3)

3.figure.py

"""
    画布文件,实现绘图区域的显示,并返回画布的对象。
"""
import tkinter as tk

# 创建画布需要的库
from matplotlib.backends.backend_tkagg import FigureCanvasTkAgg
# 创建工具栏需要的库
from matplotlib.backends.backend_tkagg import NavigationToolbar2Tk
# 快捷键需要的库
from matplotlib.backend_bases import key_press_handler
# 导入画图常用的库
from matplotlib.figure import Figure

def plot_fun(root):
    """
        该函数实现的是内嵌画布,不负责画图,返回画布对象。
    :param root:父亲控件对象, 一般是容器控件或者窗体
    :return: 画布对象
    """
    # 画布的大小和分别率
    fig = Figure(dpi=100)
    axs = fig.add_subplot(111)

    # 创建画布
    canvas = FigureCanvasTkAgg(fig, master=root)  # A tk.DrawingArea.
    canvas.draw()
    # 显示画布
    canvas.get_tk_widget().pack()

    # 创建工具条
    toolbar = NavigationToolbar2Tk(canvas, root)
    toolbar.update()
    # 显示工具条
    canvas.get_tk_widget().pack()

    # 调用快捷键
    def on_key_press(event):
        key_press_handler(event, canvas, toolbar)

    canvas.mpl_connect("key_press_event", on_key_press)

    # 返回画布的对象
    return axs

4.plot.py

"""
    绘图文件,负责函数曲线的绘制
"""
import numpy as np


def plot_main(string, plt):
    """
        负责函数曲线的绘制
    :param string: 数学表达式
    :param plt: 画布的对象
    :return: 无
    """
    list_expr = []
    list_expr = string.split(",")
    string1 = []
    for sub_expr in list_expr:
        string1.append(sub_expr)
    x = np.linspace(-10, 10, 100)
    y = []
    num = string.count('x')
    for i in x:
        t = (i, ) * num
        string = string.replace("x", "(%f)")
        i = eval(string % t)
        y.append(i)
    plt.plot(x, y)
    plt.grid(True)
    plt.legend(labels=string1)

(四)文件结构

四个文件均处于同一个文件夹下,用main.py来运行。

02.png

(五)项目下载:

百度网盘下载

链接:https://pan.baidu.com/s/13G_hWqagxqHRkdHaYcxiQQ
提取码:codq

作者:Mark

日期:2019/02/15 周五

posted @ 2019-02-15 19:23  梦并不遥远  阅读(7824)  评论(0编辑  收藏  举报