make views combineable
This commit is contained in:
@@ -10,7 +10,7 @@ from python_toolkit.serial_reader import SerialReader
|
|||||||
from python_toolkit.serial_mock import SerialMock
|
from python_toolkit.serial_mock import SerialMock
|
||||||
from python_toolkit.gui.connect import ConnectFrame
|
from python_toolkit.gui.connect import ConnectFrame
|
||||||
from .config import DEFAULT_CALIB_WEIGHT, DEFAULT_CALIB, DISPLAY_TYPES, MOV_AVG_DEFAULTS
|
from .config import DEFAULT_CALIB_WEIGHT, DEFAULT_CALIB, DISPLAY_TYPES, MOV_AVG_DEFAULTS
|
||||||
from .views import *
|
from .views import CombinedView
|
||||||
|
|
||||||
class WeightApp(tk.Tk):
|
class WeightApp(tk.Tk):
|
||||||
def __init__(self, weight_reader: SerialReader):
|
def __init__(self, weight_reader: SerialReader):
|
||||||
@@ -19,7 +19,7 @@ class WeightApp(tk.Tk):
|
|||||||
self.record_start = None
|
self.record_start = None
|
||||||
self.record_window = []
|
self.record_window = []
|
||||||
self.weight_reader = weight_reader
|
self.weight_reader = weight_reader
|
||||||
self.view = None
|
self.view = None
|
||||||
|
|
||||||
self.toolbar = tk.Frame(self, padx=10)
|
self.toolbar = tk.Frame(self, padx=10)
|
||||||
self.toolbar.pack(side=tk.LEFT)
|
self.toolbar.pack(side=tk.LEFT)
|
||||||
@@ -50,10 +50,18 @@ class WeightApp(tk.Tk):
|
|||||||
self.view_type = tk.Frame(self.toolbar, pady=20)
|
self.view_type = tk.Frame(self.toolbar, pady=20)
|
||||||
self.view_type.pack()
|
self.view_type.pack()
|
||||||
self.view_type_label = ttk.Label(self.view_type, text="Visual:")
|
self.view_type_label = ttk.Label(self.view_type, text="Visual:")
|
||||||
self.view_type_label.pack(side=tk.LEFT)
|
self.view_type_label.pack()
|
||||||
self.view_type_select = ttk.Combobox(self.view_type, values=[t.value for t in DISPLAY_TYPES])
|
|
||||||
self.view_type_select.set(DISPLAY_TYPES.NUMBER.value)
|
# Checkboxes for multiple view selection
|
||||||
self.view_type_select.pack(side=tk.LEFT)
|
self.view_vars = {}
|
||||||
|
for display_type in DISPLAY_TYPES:
|
||||||
|
var = tk.BooleanVar()
|
||||||
|
if display_type == DISPLAY_TYPES.NUMBER:
|
||||||
|
var.set(True) # Default to NUMBER view
|
||||||
|
checkbox = ttk.Checkbutton(self.view_type, text=display_type.name, variable=var)
|
||||||
|
checkbox.pack()
|
||||||
|
self.view_vars[display_type] = var
|
||||||
|
|
||||||
self.view_type_update = ttk.Button(self.view_type, text="Refresh", command=self.update_view)
|
self.view_type_update = ttk.Button(self.view_type, text="Refresh", command=self.update_view)
|
||||||
self.view_type_update.pack()
|
self.view_type_update.pack()
|
||||||
|
|
||||||
@@ -90,6 +98,7 @@ class WeightApp(tk.Tk):
|
|||||||
self.view_type.pack_forget()
|
self.view_type.pack_forget()
|
||||||
self.recording_frame.pack_forget()
|
self.recording_frame.pack_forget()
|
||||||
self.view.pack_forget()
|
self.view.pack_forget()
|
||||||
|
|
||||||
self.connection_settings.pack()
|
self.connection_settings.pack()
|
||||||
|
|
||||||
|
|
||||||
@@ -111,32 +120,30 @@ class WeightApp(tk.Tk):
|
|||||||
self.weight_reader.calib_factor = float(self.calib_weight.get()) / float(self.calib_measurements.get())
|
self.weight_reader.calib_factor = float(self.calib_weight.get()) / float(self.calib_measurements.get())
|
||||||
|
|
||||||
def update_view(self):
|
def update_view(self):
|
||||||
selected_view = self.view_type_select.get()
|
# Get selected display types using logical AND operator
|
||||||
|
selected_types = DISPLAY_TYPES(0) # Start with empty flags
|
||||||
if self.view is not None:
|
for display_type, var in self.view_vars.items():
|
||||||
self.view.pack_forget()
|
if var.get(): # If checkbox is checked
|
||||||
|
selected_types |= display_type # Combine using bitwise OR
|
||||||
|
|
||||||
|
# Remove existing views
|
||||||
if selected_view == DISPLAY_TYPES.NUMBER.value:
|
if self.view is not None:
|
||||||
self.view = NumberView(self,
|
self.view.update_views(selected_types)
|
||||||
tare_command=self.weight_reader.tare,
|
|
||||||
calibrate_command=self.calibrate,
|
|
||||||
padx=50)
|
|
||||||
self.view.pack(side=tk.RIGHT)
|
|
||||||
elif selected_view == DISPLAY_TYPES.CIRCLE.value:
|
|
||||||
self.view = CircleView(self,
|
|
||||||
tare_command=self.weight_reader.tare,
|
|
||||||
calibrate_command=self.calibrate,
|
|
||||||
padx=50)
|
|
||||||
self.view.pack(side=tk.RIGHT)
|
|
||||||
else:
|
else:
|
||||||
raise Exception(f"View {selected_view} not found.")
|
self.view = CombinedView(self,
|
||||||
|
tare_command=self.weight_reader.tare,
|
||||||
|
calibrate_command=self.calibrate)
|
||||||
|
|
||||||
|
self.update_weight_display()
|
||||||
|
|
||||||
|
|
||||||
def update_weight_display(self):
|
def update_weight_display(self):
|
||||||
weight = self.weight_reader.value
|
weight = self.weight_reader.value
|
||||||
if self.recording:
|
if self.recording:
|
||||||
self.record_window.append((time() - self.record_start, weight))
|
self.record_window.append((time() - self.record_start, weight))
|
||||||
|
|
||||||
self.view.refresh(weight)
|
self.view.refresh(weight)
|
||||||
|
|
||||||
self.after(20, self.update_weight_display)
|
self.after(20, self.update_weight_display)
|
||||||
|
|
||||||
def calibrate(self):
|
def calibrate(self):
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
from enum import Enum
|
from enum import Flag
|
||||||
|
|
||||||
# DEFAULT_CALIB = 307333.83
|
# DEFAULT_CALIB = 307333.83
|
||||||
DEFAULT_CALIB = -105030.71880199667
|
DEFAULT_CALIB = -105030.71880199667
|
||||||
@@ -11,6 +11,6 @@ MOV_AVG_DEFAULTS = {
|
|||||||
"ignore_samples": 2
|
"ignore_samples": 2
|
||||||
}
|
}
|
||||||
|
|
||||||
class DISPLAY_TYPES(Enum):
|
class DISPLAY_TYPES(Flag):
|
||||||
NUMBER = 'number'
|
NUMBER = 1
|
||||||
CIRCLE = 'circle'
|
CIRCLE = 2
|
||||||
@@ -1,2 +1,3 @@
|
|||||||
from .number import NumberView
|
from .number import NumberView
|
||||||
from .circle import CircleView
|
from .circle import CircleView
|
||||||
|
from .combined_view import CombinedView
|
||||||
@@ -1,32 +1,18 @@
|
|||||||
from tkinter import ttk, Frame, Canvas
|
from PIL import Image
|
||||||
from PIL import Image, ImageDraw
|
class View:
|
||||||
from tkinter import PhotoImage
|
|
||||||
import io
|
|
||||||
|
|
||||||
class View(Frame):
|
def __init__(self, parent, size, center, **kwargs):
|
||||||
|
self.size = size
|
||||||
def __init__(self, *args, tare_command=None, calibrate_command=None, **kwargs):
|
self.center = center
|
||||||
super().__init__(*args, **kwargs)
|
|
||||||
|
|
||||||
self.actions = Frame(self)
|
|
||||||
self.actions.pack()
|
|
||||||
self.calibrate_button = ttk.Button(self.actions, text="Calibrate", command=calibrate_command)
|
|
||||||
self.calibrate_button.pack()
|
|
||||||
self.tare_button = ttk.Button(self.actions, text="Tare", command=tare_command)
|
|
||||||
self.tare_button.pack()
|
|
||||||
|
|
||||||
self.size = (168, 144)
|
|
||||||
self.center = (168 // 2, 144 // 2)
|
|
||||||
self.bkg_im = self._init_im()
|
self.bkg_im = self._init_im()
|
||||||
self._init_ui()
|
|
||||||
|
|
||||||
self.canvas = Canvas(self, width=168, height=144, background='white',
|
self.ui = None
|
||||||
highlightthickness=1, highlightbackground="black")
|
self.init_ui(parent)
|
||||||
self.canvas.pack()
|
|
||||||
|
self.update_weight(0.0)
|
||||||
|
|
||||||
self.refresh(0.0)
|
|
||||||
|
|
||||||
def _init_ui(self):
|
def init_ui(self, parent):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
def _init_im(self):
|
def _init_im(self):
|
||||||
@@ -35,17 +21,3 @@ class View(Frame):
|
|||||||
def update_weight(self,
|
def update_weight(self,
|
||||||
weight: float) -> None:
|
weight: float) -> None:
|
||||||
raise NotImplementedError()
|
raise NotImplementedError()
|
||||||
|
|
||||||
def refresh(self, weight: float):
|
|
||||||
# draw weight on bkg_im
|
|
||||||
im = self.update_weight(weight)
|
|
||||||
|
|
||||||
# Convert PIL image to bytes
|
|
||||||
buffer = io.BytesIO()
|
|
||||||
im.save(buffer, format='PNG')
|
|
||||||
buffer.seek(0)
|
|
||||||
|
|
||||||
# Load into PhotoImage and display on canvas
|
|
||||||
self.photo = PhotoImage(data=buffer.getvalue())
|
|
||||||
self.canvas.delete("all")
|
|
||||||
self.canvas.create_image(0, 0, anchor="nw", image=self.photo)
|
|
||||||
|
|||||||
@@ -6,12 +6,12 @@ from .base import View
|
|||||||
|
|
||||||
class CircleView(View):
|
class CircleView(View):
|
||||||
|
|
||||||
def _init_ui(self):
|
def init_ui(self, parent):
|
||||||
self.target_frame = tk.Frame(self.actions)
|
self.ui = tk.Frame(parent)
|
||||||
self.target_frame.pack()
|
self.ui.pack()
|
||||||
self.target_label = ttk.Label(self.target_frame, text="Target (g)")
|
self.target_label = ttk.Label(self.ui, text="Target (g)")
|
||||||
self.target_label.pack(side=tk.LEFT)
|
self.target_label.pack(side=tk.LEFT)
|
||||||
self.target = ttk.Entry(self.target_frame)
|
self.target = ttk.Entry(self.ui)
|
||||||
self.target.insert(0, 100.0)
|
self.target.insert(0, 100.0)
|
||||||
self.target.pack(side=tk.LEFT)
|
self.target.pack(side=tk.LEFT)
|
||||||
|
|
||||||
|
|||||||
69
frontend/views/combined_view.py
Normal file
69
frontend/views/combined_view.py
Normal file
@@ -0,0 +1,69 @@
|
|||||||
|
import io
|
||||||
|
import tkinter as tk
|
||||||
|
from tkinter import Frame, Canvas, ttk, PhotoImage
|
||||||
|
|
||||||
|
from PIL import Image, ImageChops
|
||||||
|
|
||||||
|
from ..config import DISPLAY_TYPES
|
||||||
|
from . import NumberView, CircleView
|
||||||
|
|
||||||
|
class CombinedView(tk.Frame):
|
||||||
|
def __init__(self, parent,
|
||||||
|
tare_command=None, calibrate_command=None,
|
||||||
|
**kwargs):
|
||||||
|
super().__init__(parent, **kwargs)
|
||||||
|
self.views = []
|
||||||
|
self.tare_command = tare_command
|
||||||
|
self.calibrate_command = calibrate_command
|
||||||
|
|
||||||
|
self.actions = Frame(self)
|
||||||
|
self.actions.pack()
|
||||||
|
self.calibrate_button = ttk.Button(self.actions, text="Calibrate", command=calibrate_command)
|
||||||
|
self.calibrate_button.pack()
|
||||||
|
self.tare_button = ttk.Button(self.actions, text="Tare", command=tare_command)
|
||||||
|
self.tare_button.pack()
|
||||||
|
|
||||||
|
self.im_size = (168, 144)
|
||||||
|
self.center = (168 // 2, 144 // 2)
|
||||||
|
|
||||||
|
self.canvas = Canvas(self, width=168, height=144, background='white',
|
||||||
|
highlightthickness=1, highlightbackground="black")
|
||||||
|
self.canvas.pack()
|
||||||
|
|
||||||
|
|
||||||
|
def update_views(self, selected_types: DISPLAY_TYPES):
|
||||||
|
for v in self.views:
|
||||||
|
if v.ui is not None:
|
||||||
|
v.ui.destroy()
|
||||||
|
self.views.clear()
|
||||||
|
|
||||||
|
if selected_types & DISPLAY_TYPES.NUMBER:
|
||||||
|
number_view = NumberView(self.actions, self.im_size, self.center)
|
||||||
|
self.views.append(number_view)
|
||||||
|
|
||||||
|
if selected_types & DISPLAY_TYPES.CIRCLE:
|
||||||
|
circle_view = CircleView(self.actions, self.im_size, self.center)
|
||||||
|
self.views.append(circle_view)
|
||||||
|
|
||||||
|
|
||||||
|
def refresh(self, weight: float):
|
||||||
|
ims = []
|
||||||
|
for view in self.views:
|
||||||
|
im = view.update_weight(weight)
|
||||||
|
ims.append(im)
|
||||||
|
|
||||||
|
self.canvas.delete("all")
|
||||||
|
# Combine images by logical_and
|
||||||
|
if ims:
|
||||||
|
combined_im = ims[0]
|
||||||
|
for im in ims[1:]:
|
||||||
|
combined_im = ImageChops.logical_and(combined_im, im)
|
||||||
|
|
||||||
|
# Convert PIL image to bytes
|
||||||
|
buffer = io.BytesIO()
|
||||||
|
combined_im.save(buffer, format='PNG')
|
||||||
|
buffer.seek(0)
|
||||||
|
|
||||||
|
# Load into PhotoImage and display on canvas
|
||||||
|
self.photo = PhotoImage(data=buffer.getvalue())
|
||||||
|
self.canvas.create_image(0, 0, anchor="nw", image=self.photo)
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
from PIL import Image, ImageDraw
|
from PIL import ImageDraw
|
||||||
|
|
||||||
from .base import View
|
from .base import View
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user