Module pyspecan.controller.tkGUI.rt.spg

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 PlotConfig
Expand source code
class PlotConfig:
    ref_level = 0
    scale_div = 10.0
    vbw = 10.0
    window = "blackman"

Class variables

var ref_level
var scale_div
var vbw
var window
class SPG (parent,
pane: Panel,
**kwargs)
Expand source code
class SPG(FreqPlotControllerRT):
    def __init__(self, parent, pane: Panel, **kwargs):
        pane.master.config(text="Spectrogram")
        self.max_count = 256
        self.psds: np.ndarray = None # type: ignore
        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)
        self.reset()
        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("spg", fig.add_subplot())

        self.plotter.ax("spg").ax.set_autoscale_on(False)
        self.plotter.ax("spg").ax.locator_params(axis="x", nbins=5)
        self.plotter.ax("spg").ax.locator_params(axis="y", nbins=5)

        self.update_f(self.parent.dispatch.last_f)
        self.set_y()
        self.update()

    def reset(self):
        self.psds = np.zeros((self.max_count, self.parent.model.get_nfft()), dtype=np.float32)
        self.psds[:,:] = -np.inf

    def _plot(self, samps):
        psd = self.psd(samps)
        psd = np.sum(psd, axis=1) / psd.shape[1]
        psd = np.clip(psd, self.y_btm, self.y_top)
        y_rng = abs(self.y_btm - self.y_top)
        psd += (0 - self.y_btm)
        psd = psd / y_rng

        self.psds = np.roll(self.psds, 1, axis=0)
        self.psds[0,:] = psd
        # print(self.psds.shape)
        im = self.plotter.ax("spg").imshow(
            self.psds, name="spg",
            aspect="auto", origin="upper", cmap="magma",
            vmin=0.0, vmax=1.0,
            interpolation="nearest", resample=False, rasterized=True
        )
        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]

        spg_tick = np.linspace(0, fnum+1, 5)
        spg_text = psd_text
        self.plotter.ax("spg").ax.set_xlim(0, fnum)
        self.plotter.ax("spg").ax.set_xticks(spg_tick, spg_text)

    def update_nfft(self, nfft):
        self.psds = np.zeros((self.max_count, nfft), dtype=np.float32)
        self.psds[:,:] = -np.inf
        self.reset()

    def set_y(self):
        """Set plot ylimits"""
        self.plotter.ax("spg").set_ylim(self.max_count, 0)

    def draw_settings(self, row=0):
        row = super().draw_settings(row)
        var_max_count = tk.StringVar(self.pane.settings, str(self.max_count))
        ent_max_count = ttk.Entry(self.pane.settings, textvariable=var_max_count, width=10)
        ent_max_count.bind("<Return>", self.handle_event)

        self.pane.wgts["max_count"] = ent_max_count
        self.pane.sets["max_count"] = var_max_count

        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="Spectrogram").grid(row=row, column=0,columnspan=2)
        row += 1
        ttk.Label(self.pane.settings, text="Max Count").grid(row=row, column=0)
        ent_max_count.grid(row=row, column=1)
        row += 1
        return row

    # --- GUI bind events and setters --- #
    def handle_event(self, event):
        if event.widget == self.pane.wgts["max_count"]:
            self.set_count(self.pane.sets["max_count"].get())
        else:
            super().handle_event(event)

    def set_count(self, count):
        prev = int(self.max_count)
        try:
            count = int(count)
            self.max_count = count
        except ValueError:
            count = self.max_count
        self.pane.sets["count"].set(str(self.max_count))
        if not prev == self.max_count:
            self.reset()

    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):
        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

Controller for view.tkGUI frequency-domain plots

Ancestors

Methods

def handle_event(self, event)
Expand source code
def handle_event(self, event):
    if event.widget == self.pane.wgts["max_count"]:
        self.set_count(self.pane.sets["max_count"].get())
    else:
        super().handle_event(event)
def reset(self)
Expand source code
def reset(self):
    self.psds = np.zeros((self.max_count, self.parent.model.get_nfft()), dtype=np.float32)
    self.psds[:,:] = -np.inf

Reset plot view

def set_count(self, count)
Expand source code
def set_count(self, count):
    prev = int(self.max_count)
    try:
        count = int(count)
        self.max_count = count
    except ValueError:
        count = self.max_count
    self.pane.sets["count"].set(str(self.max_count))
    if not prev == self.max_count:
        self.reset()
def set_y(self)
Expand source code
def set_y(self):
    """Set plot ylimits"""
    self.plotter.ax("spg").set_ylim(self.max_count, 0)

Set plot ylimits

Inherited members