Beispiel #1
0
 def __init__(self):
     wx.Frame.__init__(self, None, -1, self.title)
     
     self.datagen = DataGen()
     self.datav = [self.datagen.next()[0]]
     self.datac = [self.datagen.next()[1]]
     self.paused = False
     self.ocv = True
     exited = False
     self.create_menu()
     self.create_status_bar()
     self.create_main_panel()
Beispiel #2
0
class GraphFrame(wx.Frame):
    """ The main frame of the application
    """
    title = 'Demo: dynamic matplotlib graph'
    
    def __init__(self):
        wx.Frame.__init__(self, None, -1, self.title)
        
        self.datagen = DataGen()
        self.datav = [self.datagen.next()[0]]
        self.datac = [self.datagen.next()[1]]
        self.paused = False
        self.ocv = True
        exited = False
        self.create_menu()
        self.create_status_bar()
        self.create_main_panel()
        
        #self.redraw_timer = wx.Timer(self)
        #self.Bind(wx.EVT_TIMER, self.on_redraw_timer, self.redraw_timer)        
        #self.redraw_timer.Start(100)

    def create_menu(self):
        self.menubar = wx.MenuBar()
        
        menu_file = wx.Menu()
        m_expt = menu_file.Append(-1, "&Save plot\tCtrl-S", "Save plot to file")
        self.Bind(wx.EVT_MENU, self.on_save_plot, m_expt)
        menu_file.AppendSeparator()
        m_exit = menu_file.Append(-1, "E&xit\tCtrl-X", "Exit")
        self.Bind(wx.EVT_MENU, self.on_exit, m_exit)
                
        self.menubar.Append(menu_file, "&File")
        self.SetMenuBar(self.menubar)

    def create_main_panel(self):
        self.panel = wx.Panel(self)

        self.init_plot()
        self.canvas = FigCanvas(self.panel, -1, self.fig)

        self.xmin_control = BoundControlBox(self.panel, -1, "X min", 0)
        self.xmax_control = BoundControlBox(self.panel, -1, "X max", 50)
        self.ymin_control = BoundControlBox(self.panel, -1, "Y min", 0)
        self.ymax_control = BoundControlBox(self.panel, -1, "Y max", 100)
        
        self.pause_button = wx.Button(self.panel, -1, "Pause")
        self.ocv_button = wx.Button(self.panel, -1, "Start")
        self.Bind(wx.EVT_BUTTON, self.on_pause_button, self.pause_button)
        self.Bind(wx.EVT_BUTTON, self.on_ocv_button, self.ocv_button)
        self.Bind(wx.EVT_UPDATE_UI, self.on_update_pause_button, self.pause_button)
        self.Bind(wx.EVT_UPDATE_UI, self.on_update_ocv_button, self.ocv_button)
        
        self.cb_grid = wx.CheckBox(self.panel, -1, 
            "Show Grid",
            style=wx.ALIGN_RIGHT)
        self.Bind(wx.EVT_CHECKBOX, self.on_cb_grid, self.cb_grid)
        self.cb_grid.SetValue(True)
        
        self.cb_xlab = wx.CheckBox(self.panel, -1, 
            "Show X labels",
            style=wx.ALIGN_RIGHT)
        self.Bind(wx.EVT_CHECKBOX, self.on_cb_xlab, self.cb_xlab)        
        self.cb_xlab.SetValue(True)
        
        self.hbox1 = wx.BoxSizer(wx.HORIZONTAL)
        self.hbox1.Add(self.pause_button, border=5, flag=wx.ALL | wx.ALIGN_CENTER_VERTICAL)
        self.hbox1.AddSpacer(20)
        self.hbox1.Add(self.ocv_button, border=5, flag=wx.ALL | wx.ALIGN_CENTER_VERTICAL)
        self.hbox1.AddSpacer(20)
        self.hbox1.Add(self.cb_grid, border=5, flag=wx.ALL | wx.ALIGN_CENTER_VERTICAL)
        self.hbox1.AddSpacer(10)
        self.hbox1.Add(self.cb_xlab, border=5, flag=wx.ALL | wx.ALIGN_CENTER_VERTICAL)
        
        self.hbox2 = wx.BoxSizer(wx.HORIZONTAL)
        self.hbox2.Add(self.xmin_control, border=5, flag=wx.ALL)
        self.hbox2.Add(self.xmax_control, border=5, flag=wx.ALL)
        self.hbox2.AddSpacer(24)
        self.hbox2.Add(self.ymin_control, border=5, flag=wx.ALL)
        self.hbox2.Add(self.ymax_control, border=5, flag=wx.ALL)
        
        self.vbox = wx.BoxSizer(wx.VERTICAL)
        self.vbox.Add(self.canvas, 1, flag=wx.LEFT | wx.TOP | wx.GROW)        
        self.vbox.Add(self.hbox1, 0, flag=wx.ALIGN_LEFT | wx.TOP)
        self.vbox.Add(self.hbox2, 0, flag=wx.ALIGN_LEFT | wx.TOP)
        
        self.panel.SetSizer(self.vbox)
        self.vbox.Fit(self)
    
    def create_status_bar(self):
        self.statusbar = self.CreateStatusBar()

    def init_plot(self):
        self.dpi = 100
        self.fig = Figure((3.0, 5.0), dpi=self.dpi)

        self.axesv = self.fig.add_subplot(211)
        self.axesc = self.fig.add_subplot(212)
        self.axesv.set_axis_bgcolor('black')
        self.axesv.set_title('Voltage (V)', size=12)

        self.axesc.set_axis_bgcolor('black')
        self.axesc.set_title('Current (mA)', size=12)
        
        pylab.setp(self.axesv.get_xticklabels(), fontsize=8)
        pylab.setp(self.axesv.get_yticklabels(), fontsize=8)

        pylab.setp(self.axesc.get_xticklabels(), fontsize=8)
        pylab.setp(self.axesc.get_yticklabels(), fontsize=8)

        # plot the data as a line series, and save the reference 
        # to the plotted line series
        #
        self.plot_datav = self.axesv.plot(
            self.datav, 
            linewidth=1,
            color=(1, 1, 0),
            )[0]
        
        self.plot_datac = self.axesc.plot(
            self.datac, 
            linewidth=1,
            color=(0, 0, 1),
            )[0]
        

    def draw_plot(self):
        """ Redraws the plot
        """
        # when xmin is on auto, it "follows" xmax to produce a 
        # sliding window effect. therefore, xmin is assigned after
        # xmax.
        #
        if self.xmax_control.is_auto():
            xmaxv = xmaxc = len(self.datav) if len(self.datav) > 50 else 50
        else:
            xmaxv = float(self.xmax_control.manual_value())
            xmaxc = float(self.xmax_control.manual_value())
            
        if self.xmin_control.is_auto():            
            xminv = xminc = xmaxv - 50
        else:
            xminv = float(self.xmin_control.manual_value())
            xminc = float(self.xmin_control.manual_value())

        # for ymin and ymax, find the minimal and maximal values
        # in the data set and add a mininal margin.
        # 
        # note that it's easy to change this scheme to the 
        # minimal/maximal value in the current display, and not
        # the whole data set.
        #
        if self.ymin_control.is_auto():
            yminv = min(self.datav[int(xminv):int(xmaxv)])
            yminc = min(self.datac[int(xminc):int(xmaxc)])
        else:
            yminv = float(self.ymin_control.manual_value())
            yminc = float(self.ymin_control.manual_value())
        if self.ymax_control.is_auto():
            ymaxv = max(self.datav[int(xminv):int(xmaxv)])
            ymaxc = max(self.datac[int(xminc):int(xmaxc)])
        else:
            ymaxv = float(self.ymax_control.manual_value())
            ymaxc = float(self.ymax_control.manual_value())

        self.axesv.set_xbound(lower=xminv, upper=xmaxv)
        self.axesv.set_ybound(lower=yminv, upper=ymaxv)

        self.axesc.set_xbound(lower=xminc, upper=xmaxc)
        self.axesc.set_ybound(lower=yminc, upper=ymaxc)
        
        # anecdote: axes.grid assumes b=True if any other flag is
        # given even if b is set to False.
        # so just passing the flag into the first statement won't
        # work.
        #
        if self.cb_grid.IsChecked():
            self.axesv.grid(True, color='gray')
            self.axesc.grid(True, color='gray')
        else:
            self.axesv.grid(False)
            self.axesc.grid(False)

        # Using setp here is convenient, because get_xticklabels
        # returns a list over which one needs to explicitly 
        # iterate, and setp already handles this.
        #  
        pylab.setp(self.axesv.get_xticklabels(), 
            visible=self.cb_xlab.IsChecked())

        pylab.setp(self.axesc.get_xticklabels(), 
            visible=self.cb_xlab.IsChecked())
        
        self.plot_datav.set_xdata(np.arange(len(self.datav)))
        self.plot_datav.set_ydata(np.array(self.datav))

        
        self.plot_datac.set_xdata(np.arange(len(self.datac)))
        self.plot_datac.set_ydata(np.array(self.datac))
        
        
        self.canvas.draw()
    
    def on_pause_button(self, event):
        self.paused = not self.paused

    def on_ocv_button(self, event):
        self.ocv = not self.ocv
    
    def on_update_pause_button(self, event):
        label = "Resume" if self.paused else "Pause"
        self.pause_button.SetLabel(label)

    def on_update_ocv_button(self, event):
        label = "Start" if self.ocv else "OCV"
        self.ocv_button.SetLabel(label)
    
    def on_cb_grid(self, event):
        self.draw_plot()
    
    def on_cb_xlab(self, event):
        self.draw_plot()
    
    def on_save_plot(self, event):
        """
        file_choices = "PNG (*.png)|*.png"
        
        dlg = wx.FileDialog(
            self, 
            message="Save plot as...",
            defaultDir=os.getcwd(),
            defaultFile="plot.png",
            wildcard=file_choices,
            style=wx.SAVE)
        
        if dlg.ShowModal() == wx.ID_OK:
            path = dlg.GetPath()
            self.canvas.print_figure(path, dpi=self.dpi)
            self.flash_status_message("Saved to %s" % path)

        """
        #file_choices = "PICKLE (*.pickle)|*.pickle"
        dlg = wx.FileDialog(
            self,
            message="Save plot as...",
            defaultDir=os.getcwd(),
            defaultFile="plot",
            #wildcard=file_choices,
            style=wx.SAVE)
        if dlg.ShowModal() == wx.ID_OK:
            path = dlg.GetPath() + "current.pickle"
            outfile = open(path,"w")
            pickle.dump(self.datac, outfile)
            outfile.close()
            path = dlg.GetPath() + "voltage.pickle"
            outfile = open(path,"w")
            pickle.dump(self.datav, outfile)
            outfile.close()
            path = dlg.GetPath()
            outfile = open(path,"w")
            outfile.close()
            path = dlg.GetPath()
            self.canvas.print_figure(path, dpi=self.dpi)
            outfile.close()
            path = dlg.GetPath()
            path = path + "log.txt"
            outfile = open(path, "w")
            i = 0
            for voltage, current in zip(self.datav, self.datac):
                outfile.write("Sample: " + str(i) + ", Voltage: " + str(voltage) + ", Current(mA): " + str(current) + "\n")
                i += 1
            outfile.close()

    def on_load_plot(self):
        self.paused = True
        file_choices = "PICKLE (*.pickle)|*.pickle"
        dlg = wx.FileDialog(
            self,
            message="Load plot...",
            defaultDir=os.getcwd(),
            #wildcard=file_choices,
            style=wx.OPEN)
        if dlg.ShowModal() == wx.ID_OK:
            path = dlg.GetPath() + "current.pickle"
            infile = open(path,"r")
            self.datac = pickle.load(infile)
            infile.close()
            path = dlg.GetPath() + "voltage.pickle"
            infile = open(path,"r")
            self.datav = pickle.load(infile)
            infile.close()
        while True:
            app.frame.on_redraw_timer(False)
            time.sleep(.1)
    
    def on_redraw_timer(self, pause):
        # if paused do not add data, but still redraw the plot
        # (to respond to scale modifications, grid change, etc.)
        #
        if not self.paused and not pause:
            self.datav.append(self.datagen.next()[0])
            self.datac.append(self.datagen.next()[1])
        
        self.draw_plot()
    
    def on_exit(self, event):
        exited = True
        self.datagen.erase()
        self.Destroy()
    
    def flash_status_message(self, msg, flash_len_ms=1500):
        self.statusbar.SetStatusText(msg)
        self.timeroff = wx.Timer(self)
        self.Bind(
            wx.EVT_TIMER, 
            self.on_flash_status_off, 
            self.timeroff)
        self.timeroff.Start(flash_len_ms, oneShot=True)
    
    def on_flash_status_off(self, event):
        self.statusbar.SetStatusText('')