import sys import os import threading import tkinter as tk from tkinter import ttk import RPi.GPIO as GPIO from matplotlib.figure import Figure from matplotlib.backends.backend_tkagg import FigureCanvasTkAgg import time # Add the parent directory to the module search path sys.path.append(os.path.abspath(os.path.join(os.path.dirname(__file__), ".."))) from interface_board_pins import * # Import pin assignments # Adjusted Constants DIGITAL_PLOT_UPDATE_INTERVAL = 50 # 50ms (20 FPS) is smooth enough NOT_ACTIVE_CHECK_INTERVAL = 2000 # Check inactive tabs less frequently MAX_HISTORY = 100 # Keep last 100 values for scrolling updating_enabled = False # Track if the tab is active input_channels = list(range(8)) data = {ch: [0] * MAX_HISTORY for ch in input_channels} # Preallocate data time_data = list(range(-MAX_HISTORY, 0)) # Simulated time axis def create_digital_plot_tab(notebook): frame = ttk.Frame(notebook) notebook.add(frame, text="Digital Inputs") figure = Figure(figsize=(8, 5), dpi=100) ax = figure.add_subplot(1, 1, 1) ax.set_ylim(-0.2, 1.2) ax.set_xlim(-MAX_HISTORY, 0) # Keep time axis fixed canvas = FigureCanvasTkAgg(figure, master=frame) canvas.get_tk_widget().pack(fill=tk.BOTH, expand=True) # Initialize lines for fast updates lines = {ch: ax.step(time_data, data[ch], where="post")[0] for ch in input_channels} def update_plot(): if not updating_enabled: frame.after(NOT_ACTIVE_CHECK_INTERVAL, update_plot) return # Shift existing data left for ch in input_channels: data[ch].pop(0) data[ch].append(GPIO.input(GPIO_DIGITAL_INPUTS[ch])) # Update only the y-data for efficiency for ch in input_channels: lines[ch].set_ydata(data[ch]) canvas.draw_idle() # More efficient than draw() frame.after(DIGITAL_PLOT_UPDATE_INTERVAL, update_plot) update_plot() def set_updating_enabled(is_active): global updating_enabled updating_enabled = is_active print(f"digital_plot tab: set updating_enabled to {updating_enabled}")