Module pyspecan.controller.tkGUI.swept.psd

Functions

def define_args(parser: argparse.ArgumentParser)
Expand source code
def define_args(parser: argparse.ArgumentParser):
    parser.add_argument("-rl", "--ref_level", default=PlotConfig.ref_level, type=float, help="ref Level")
    parser.add_argument("-sd", "--scale_div", default=PlotConfig.scale_div, type=float, help="scale per division")
    parser.add_argument("-vb", "--vbw", default=PlotConfig.vbw, type=float, help="video bandwidth")
    parser.add_argument("-w", "--window", default=PlotConfig.window, choices=[k for k in WindowLUT.keys()], help="FFT window function")

Classes

class PSD (parent,
pane: Panel,
**kwargs)
Expand source code
class PSD(FreqPlotController):
    def __init__(self, parent, pane: Panel, **kwargs):
        pane.master.config(text="PSD")
        self.psd_min = None
        self.psd_max = None
        self.fmin = None
        self.fmax = None
        if not "ref_level" in kwargs:
            kwargs["ref_level"] = PlotConfig.ref_level
        if not "scale_div" in kwargs:
            kwargs["scale_div"] = PlotConfig.scale_div
        if not "vbw" in kwargs:
            kwargs["vbw"] = PlotConfig.vbw
        if not "window" in kwargs:
            kwargs["vbw"] = PlotConfig.vbw
        super().__init__(parent, pane, **kwargs)
        fig = plt.figure(figsize=(5,5), layout="constrained")

        fig.canvas = FigureCanvasTkAgg(fig, master=self.fr_canv)
        self.plotter = BlitPlot(fig) # type: ignore
        self.plotter.canvas.draw()
        self.plotter.canvas.get_tk_widget().pack(side=tk.LEFT, fill=tk.BOTH, expand=True) # type: ignore
        self.plotter.add_ax("psd", fig.add_subplot())

        self.plotter.ax("psd").ax.set_autoscale_on(False)
        self.plotter.ax("psd").ax.locator_params(axis="x", nbins=5)
        self.plotter.ax("psd").ax.locator_params(axis="y", nbins=10)
        self.plotter.ax("psd").ax.grid(True, alpha=0.2)

        self.update_f(self.parent.dispatch.last_f)
        if not self.ref_level == "auto":
            self.set_y()
        self.update()

    def reset(self):
        self.psd_min = None
        self.psd_max = None

    def _plot(self, samps):
        psd = self.psd(samps)
        if self.ref_level == "auto":
            self.ref_level = self._calc_ref_level(psd)
            self.set_ref_level(self.ref_level)
            self.set_y()
        if self.pane.sets["show_max"].get() == 1:
            if self.psd_max is None:
                self.psd_max = np.repeat(-np.inf, len(psd))
            self.psd_max[psd > self.psd_max] = psd[psd > self.psd_max]
            line_max = self.plotter.ax("psd").plot(self.parent.model.f, self.psd_max, name="psd_max", color="r")
        else:
            line_max = None
        if self.pane.sets["show_min"].get() == 1:
            if self.psd_min is None:
                self.psd_min = np.repeat(np.inf, len(psd))
            self.psd_min[psd < self.psd_min] = psd[psd < self.psd_min]
            line_min = self.plotter.ax("psd").plot(self.parent.model.f, self.psd_min, name="psd_min", color="b")
        else:
            line_min = None
        line_psd = self.plotter.ax("psd").plot(self.parent.model.f, psd, name="psd", color="y")

        self._show_y_location(psd)
        self.update()

    def update_f(self, f):
        fmin, fmax, fnum = f
        if self.fmin == fmin and self.fmax == fmax:
            return
        else:
            self.fmin = fmin
            self.fmax = fmax
        psd_tick = np.linspace(fmin, fmax, 5)
        psd_text = [str(Frequency.get(f)) for f in psd_tick]
        self.plotter.ax("psd").set_xlim(fmin, fmax)
        self.plotter.ax("psd").ax.set_xticks(psd_tick, psd_text)
        self.update()

    def update_nfft(self, nfft):
        self.reset()

    def set_y(self):
        """Set plot ylimits"""
        self.plotter.ax("psd").set_ylim(self.y_btm, self.y_top)

    def draw_settings(self, row=0):
        row = super().draw_settings(row)
        var_psd_min = tk.IntVar(self.pane.settings, 1)
        chk_show_min = ttk.Checkbutton(self.pane.settings, onvalue=1, offvalue=0,variable=var_psd_min)
        chk_show_min.configure(command=self.toggle_psd_min)
        var_psd_max = tk.IntVar(self.pane.settings, 1)
        chk_show_max = ttk.Checkbutton(self.pane.settings, onvalue=1, offvalue=0, variable=var_psd_max)
        chk_show_max.configure(command=self.toggle_psd_max)

        self.pane.wgts["show_min"] = chk_show_min
        self.pane.sets["show_min"] = var_psd_min
        self.pane.wgts["show_max"] = chk_show_max
        self.pane.sets["show_max"] = var_psd_max

        ttk.Separator(self.pane.settings, orient=tk.HORIZONTAL).grid(row=row,column=0,columnspan=3, pady=5, sticky=tk.EW)
        row += 1
        ttk.Label(self.pane.settings, text="PSD").grid(row=row, column=0,columnspan=2)
        row += 1
        ttk.Label(self.pane.settings, text="Max Hold").grid(row=row, column=0)
        chk_show_max.grid(row=row, column=1)
        row += 1
        ttk.Label(self.pane.settings, text="Min Hold").grid(row=row, column=0)
        chk_show_min.grid(row=row, column=1)
        row += 1
        return row

    # --- GUI bind events and setters --- #
    def set_scale(self, scale):
        prev = float(self.scale)
        super().set_scale(scale)
        if not prev == self.scale:
            self.set_y()

    def set_ref_level(self, ref):
        if self.ref_level == "auto":
            return
        prev = float(self.ref_level)
        super().set_ref_level(ref)
        if not prev == self.ref_level:
            self.set_y()

    def set_vbw(self, smooth):
        prev = float(self.vbw)
        super().set_vbw(smooth)
        if not prev == self.vbw:
            self.psd_min = None
            self.psd_max = None

    def toggle_psd_min(self):
        """Toggle PSD min-hold visibility"""
        art = self.plotter.ax("psd").art("psd_min")
        if art is None:
            return
        if self.pane.sets["show_min"].get() == 0:
            self.psd_max = None
            art.set_visible(False)
        else:
            art.set_visible(True)
        self.update()

    def toggle_psd_max(self):
        """Toggle PSD max-hold visibility"""
        art = self.plotter.ax("psd").art("psd_max")
        if art is None:
            return
        if self.pane.sets["show_max"].get() == 0:
            self.psd_min = None
            art.set_visible(False)
        else:
            art.set_visible(True)
        self.update()

Controller for view.tkGUI frequency-domain plots

Ancestors

Methods

def reset(self)
Expand source code
def reset(self):
    self.psd_min = None
    self.psd_max = None

Reset plot view

def set_y(self)
Expand source code
def set_y(self):
    """Set plot ylimits"""
    self.plotter.ax("psd").set_ylim(self.y_btm, self.y_top)

Set plot ylimits

def toggle_psd_max(self)
Expand source code
def toggle_psd_max(self):
    """Toggle PSD max-hold visibility"""
    art = self.plotter.ax("psd").art("psd_max")
    if art is None:
        return
    if self.pane.sets["show_max"].get() == 0:
        self.psd_min = None
        art.set_visible(False)
    else:
        art.set_visible(True)
    self.update()

Toggle PSD max-hold visibility

def toggle_psd_min(self)
Expand source code
def toggle_psd_min(self):
    """Toggle PSD min-hold visibility"""
    art = self.plotter.ax("psd").art("psd_min")
    if art is None:
        return
    if self.pane.sets["show_min"].get() == 0:
        self.psd_max = None
        art.set_visible(False)
    else:
        art.set_visible(True)
    self.update()

Toggle PSD min-hold visibility

Inherited members

class PlotConfig
Expand source code
class PlotConfig:
    ref_level = "auto"
    scale_div = 10.0
    vbw = 10.0
    window = "blackman"

Class variables

var ref_level
var scale_div
var vbw
var window