Files
Obsidian-Main/20.01. Programming/Python/tkinter.md

3.5 KiB
Raw Blame History

matplotlib包裝成獨立視窗

class Plot2D(Frame):
    def __init__(self, parent, dataCollector, **kwargs):
        Frame.__init__(self, parent.mainWindow, **kwargs)

        self.parent = parent
        self.mainWindows = Toplevel(parent.mainWindow)
        self.mainWindows.title("AF State")
        self.figure = plt.Figure(figsize=(9,5), dpi=100)
        self.figure.suptitle('AF value plot', fontsize=16)
        self.ax = self.figure.add_subplot(111)
        self.canvas = FigureCanvasTkAgg(self.figure, master=self.mainWindows)
        self.canvas.get_tk_widget().pack(fill='both')
        self.axline = None

        self.dataCollector = dataCollector
        self.dataCollector.start()

    def close(self):
        print("Plot2D close")
        self.mainWindows.destroy()
        self.dataCollector.stop()
        self.dataCollector = None

    def draw(self):
        if self.dataCollector:
            datax, datay = self.dataCollector.getPlotData()
            self.ax.clear()
            self.ax.set_xlabel('Last {} datas'.format(self.dataCollector.getDataLength()))
            self.axline, = self.ax.plot(datax, datay)
            self.canvas.draw()

    def getWindow(self):
        return self.mainWindows

    def getLastData(self):
        return self.dataCollector.getLastData()

其中這一行:

self.mainWindows = Toplevel(parent.mainWindow)

是用來開一個新的視窗,其中的parent.mainWindow就是用tk.TK()所產生出來的root。

因為需要一直更新資料,所以需要的一個DataCollector來提供資料,DataCollector會提供畫圖需要的list

datax, datay = self.dataCollector.getPlotData()

DataCollector的定義如下:

class AfStateCollector(threading.Thread):
    def __init__(self, dataLength=100, pollingInterval=0.033):
        threading.Thread.__init__(self)
        self.dataLength = dataLength
        self.pollingInterval = pollingInterval
        self.stopEvent = threading.Event()
        self.data = []
        self.xdata = []

    def run(self):
        while True:
            if self.stopEvent.is_set():
                break

            afValue = self.readAf()
            self.data.append(afValue)
            self.xdata.append(len(self.xdata))
            if len(self.data) > self.dataLength:
                self.data = self.data[-self.dataLength:]
                self.xdata = list(range(self.dataLength))

            # print(f'afValue = {afValue}')
            time.sleep(self.pollingInterval)

        print("AfStateCollector stopped.")

    def readAf(self):
        ReadTestXUreg_cmd = "lvreg testxu read 10"
        ReadTestXUreg_cmd_process = subprocess.Popen(ReadTestXUreg_cmd, stdout=subprocess.PIPE, stderr=subprocess.PIPE)

        outstring, err = ReadTestXUreg_cmd_process.communicate()
        outstring = outstring.strip().decode('utf-8')
        outstring = int(outstring, 16)
        outstring_H = (outstring & 0xFF00) / 256
        outstring_L = outstring & 0xFF
        outAFStat = int(outstring_L * 256 + outstring_H)

        return outAFStat