2024-07-10 14:52:09 +08:00
|
|
|
from pytec.gui.view.zero_limits_warning import ZeroLimitsWarningView
|
|
|
|
from pytec.gui.view.thermostat_ctrl_menu import ThermostatCtrlMenu
|
|
|
|
from pytec.gui.view.conn_menu import ConnMenu
|
|
|
|
from pytec.gui.view.plot_options_menu import PlotOptionsMenu
|
|
|
|
from pytec.gui.view.live_plot_view import LiveDataPlotter
|
|
|
|
from pytec.gui.view.ctrl_panel import CtrlPanel
|
|
|
|
from pytec.gui.view.info_box import InfoBox
|
|
|
|
from pytec.gui.model.pid_autotuner import PIDAutoTuner
|
2024-08-23 11:56:02 +08:00
|
|
|
from pytec.gui.model.thermostat import Thermostat, ThermostatConnectionState
|
2024-05-13 10:35:21 +08:00
|
|
|
import json
|
|
|
|
from autotune import PIDAutotuneState
|
2023-06-27 17:34:39 +08:00
|
|
|
from qasync import asyncSlot, asyncClose
|
2024-05-13 10:35:21 +08:00
|
|
|
import qasync
|
|
|
|
import asyncio
|
|
|
|
import logging
|
|
|
|
import argparse
|
|
|
|
from PyQt6 import QtWidgets, QtGui, uic
|
|
|
|
from PyQt6.QtCore import QSignalBlocker, pyqtSlot
|
|
|
|
import pyqtgraph as pg
|
|
|
|
from functools import partial
|
|
|
|
import importlib.resources
|
2023-05-19 11:23:39 +08:00
|
|
|
|
|
|
|
|
2024-05-13 10:35:21 +08:00
|
|
|
def get_argparser():
|
|
|
|
parser = argparse.ArgumentParser(description="Thermostat Control Panel")
|
|
|
|
|
|
|
|
parser.add_argument(
|
|
|
|
"--connect",
|
|
|
|
default=None,
|
|
|
|
action="store_true",
|
2024-08-23 15:43:02 +08:00
|
|
|
help="Automatically connect to the specified Thermostat in host:port format",
|
2024-05-13 10:35:21 +08:00
|
|
|
)
|
2024-08-23 15:43:02 +08:00
|
|
|
parser.add_argument("HOST", metavar="host", default=None, nargs="?")
|
2024-05-13 10:35:21 +08:00
|
|
|
parser.add_argument("PORT", metavar="port", default=None, nargs="?")
|
|
|
|
parser.add_argument(
|
|
|
|
"-l",
|
|
|
|
"--log",
|
|
|
|
dest="logLevel",
|
|
|
|
choices=["DEBUG", "INFO", "WARNING", "ERROR", "CRITICAL"],
|
|
|
|
help="Set the logging level",
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
"-p",
|
|
|
|
"--param_tree",
|
2024-07-10 14:52:09 +08:00
|
|
|
default=importlib.resources.files("pytec.gui.view").joinpath("param_tree.json"),
|
2024-05-13 10:35:21 +08:00
|
|
|
help="Param Tree Description JSON File",
|
|
|
|
)
|
2023-05-19 11:23:39 +08:00
|
|
|
|
|
|
|
return parser
|
|
|
|
|
|
|
|
|
2024-05-13 10:35:21 +08:00
|
|
|
class MainWindow(QtWidgets.QMainWindow):
|
|
|
|
NUM_CHANNELS = 2
|
2023-08-16 17:35:13 +08:00
|
|
|
|
2024-05-13 10:35:21 +08:00
|
|
|
def __init__(self, args):
|
|
|
|
super(MainWindow, self).__init__()
|
2023-08-16 17:35:13 +08:00
|
|
|
|
2024-07-10 14:52:09 +08:00
|
|
|
ui_file_path = importlib.resources.files("pytec.gui.view").joinpath("tec_qt.ui")
|
2024-05-13 10:35:21 +08:00
|
|
|
uic.loadUi(ui_file_path, self)
|
2023-08-16 17:35:13 +08:00
|
|
|
|
2024-06-20 17:08:07 +08:00
|
|
|
self.info_box = InfoBox()
|
2023-08-16 17:35:13 +08:00
|
|
|
|
2024-07-05 17:21:39 +08:00
|
|
|
self.thermostat = Thermostat(
|
2024-07-08 11:55:09 +08:00
|
|
|
self, self.report_refresh_spin.value()
|
2024-07-05 17:21:39 +08:00
|
|
|
)
|
2024-08-21 18:10:19 +08:00
|
|
|
self._connecting_task = None
|
2024-06-20 15:25:04 +08:00
|
|
|
|
|
|
|
def handle_connection_error():
|
|
|
|
self.info_box.display_info_box(
|
|
|
|
"Connection Error", "Thermostat connection lost. Is it unplugged?"
|
|
|
|
)
|
|
|
|
|
2024-07-05 17:21:39 +08:00
|
|
|
self.thermostat.connection_error.connect(handle_connection_error)
|
2023-08-16 17:35:13 +08:00
|
|
|
|
2024-07-08 11:55:09 +08:00
|
|
|
self.thermostat.connection_error.connect(self.thermostat.timed_out)
|
2024-08-26 17:09:29 +08:00
|
|
|
self.thermostat.connection_error.connect(self.thermostat.end_session)
|
|
|
|
|
|
|
|
self.thermostat.connection_state_changed.connect(self._on_connection_changed)
|
2024-07-09 13:04:19 +08:00
|
|
|
|
2024-07-08 11:55:09 +08:00
|
|
|
self.autotuners = PIDAutoTuner(self, self.thermostat, 2)
|
2024-08-27 16:46:48 +08:00
|
|
|
self.autotuners.autotune_state_changed.connect(self.pid_autotune_handler)
|
2024-05-13 10:35:21 +08:00
|
|
|
|
|
|
|
def get_ctrl_panel_config(args):
|
2024-08-14 11:54:15 +08:00
|
|
|
with open(args.param_tree, "r", encoding="utf-8") as f:
|
2024-05-13 10:35:21 +08:00
|
|
|
return json.load(f)["ctrl_panel"]
|
|
|
|
|
2024-06-20 17:08:07 +08:00
|
|
|
self.ctrl_panel_view = CtrlPanel(
|
2024-08-26 13:49:56 +08:00
|
|
|
self.thermostat,
|
|
|
|
self.autotuners,
|
2024-08-27 13:17:25 +08:00
|
|
|
self.info_box,
|
2024-05-13 10:35:21 +08:00
|
|
|
[self.ch0_tree, self.ch1_tree],
|
|
|
|
get_ctrl_panel_config(args),
|
|
|
|
)
|
2024-06-28 10:02:43 +08:00
|
|
|
|
|
|
|
self.zero_limits_warning = ZeroLimitsWarningView(
|
|
|
|
self.style(), self.limits_warning
|
|
|
|
)
|
2024-05-13 10:35:21 +08:00
|
|
|
self.ctrl_panel_view.set_zero_limits_warning_sig.connect(
|
|
|
|
self.zero_limits_warning.set_limits_warning
|
|
|
|
)
|
2023-06-30 11:27:31 +08:00
|
|
|
|
2024-05-13 10:35:21 +08:00
|
|
|
self.report_apply_btn.clicked.connect(
|
|
|
|
lambda: self.thermostat.set_update_s(self.report_refresh_spin.value())
|
|
|
|
)
|
2023-06-30 11:27:31 +08:00
|
|
|
|
2024-05-13 10:35:21 +08:00
|
|
|
self.channel_graphs = LiveDataPlotter(
|
2024-08-27 15:43:19 +08:00
|
|
|
self.thermostat,
|
2024-05-13 10:35:21 +08:00
|
|
|
[
|
|
|
|
[getattr(self, f"ch{ch}_t_graph"), getattr(self, f"ch{ch}_i_graph")]
|
|
|
|
for ch in range(self.NUM_CHANNELS)
|
|
|
|
]
|
|
|
|
)
|
2023-06-30 11:27:31 +08:00
|
|
|
|
2024-08-27 15:40:59 +08:00
|
|
|
self.plot_options_menu = PlotOptionsMenu(self.channel_graphs)
|
2024-05-13 10:35:21 +08:00
|
|
|
self.plot_settings.setMenu(self.plot_options_menu)
|
2023-06-30 11:27:31 +08:00
|
|
|
|
2024-06-20 17:08:07 +08:00
|
|
|
self.conn_menu = ConnMenu()
|
2024-05-13 10:35:21 +08:00
|
|
|
self.connect_btn.setMenu(self.conn_menu)
|
2023-08-16 17:35:13 +08:00
|
|
|
|
2024-08-27 17:00:16 +08:00
|
|
|
self.thermostat_ctrl_menu = ThermostatCtrlMenu(
|
|
|
|
self.thermostat, self.info_box, self.style()
|
|
|
|
)
|
2024-05-13 10:35:21 +08:00
|
|
|
self.thermostat_settings.setMenu(self.thermostat_ctrl_menu)
|
2023-08-16 17:35:13 +08:00
|
|
|
|
|
|
|
self.loading_spinner.hide()
|
|
|
|
|
2023-06-30 11:27:31 +08:00
|
|
|
if args.connect:
|
|
|
|
if args.IP:
|
2023-08-16 17:35:13 +08:00
|
|
|
self.host_set_line.setText(args.IP)
|
2023-06-30 11:27:31 +08:00
|
|
|
if args.PORT:
|
|
|
|
self.port_set_spin.setValue(int(args.PORT))
|
|
|
|
self.connect_btn.click()
|
|
|
|
|
2024-08-26 17:09:29 +08:00
|
|
|
@asyncSlot(ThermostatConnectionState)
|
2023-06-30 11:27:31 +08:00
|
|
|
async def _on_connection_changed(self, result):
|
2024-08-28 10:24:25 +08:00
|
|
|
self.graph_group.setEnabled(result == ThermostatConnectionState.CONNECTED)
|
|
|
|
self.report_group.setEnabled(result == ThermostatConnectionState.CONNECTED)
|
|
|
|
self.thermostat_settings.setEnabled(
|
|
|
|
result == ThermostatConnectionState.CONNECTED
|
|
|
|
)
|
|
|
|
self.conn_menu.host_set_line.setEnabled(
|
|
|
|
result != ThermostatConnectionState.DISCONNECTED
|
|
|
|
)
|
|
|
|
self.conn_menu.port_set_spin.setEnabled(
|
|
|
|
result != ThermostatConnectionState.DISCONNECTED
|
|
|
|
)
|
|
|
|
|
2024-08-23 11:56:02 +08:00
|
|
|
match result:
|
|
|
|
case ThermostatConnectionState.CONNECTED:
|
|
|
|
self.connect_btn.setText("Disconnect")
|
2024-08-27 18:01:27 +08:00
|
|
|
hw_rev_d = await self.thermostat.get_hw_rev()
|
|
|
|
self.status_lbl.setText(
|
|
|
|
f"Connected to Thermostat v{hw_rev_d['rev']['major']}.{hw_rev_d['rev']['minor']}"
|
|
|
|
)
|
2024-08-23 11:56:02 +08:00
|
|
|
|
|
|
|
case ThermostatConnectionState.CONNECTING:
|
|
|
|
self.status_lbl.setText("Connecting...")
|
|
|
|
self.connect_btn.setText("Stop")
|
|
|
|
|
|
|
|
case ThermostatConnectionState.DISCONNECTED:
|
|
|
|
self.connect_btn.setText("Connect")
|
|
|
|
|
|
|
|
self.status_lbl.setText("Disconnected")
|
|
|
|
self.background_task_lbl.setText("Ready.")
|
|
|
|
self.loading_spinner.hide()
|
|
|
|
self.loading_spinner.stop()
|
|
|
|
self.thermostat_ctrl_menu.fan_pwm_warning.setPixmap(QtGui.QPixmap())
|
|
|
|
self.thermostat_ctrl_menu.fan_pwm_warning.setToolTip("")
|
2024-08-27 15:40:59 +08:00
|
|
|
self.channel_graphs.clear_graphs()
|
2024-08-23 11:56:02 +08:00
|
|
|
self.report_box.setChecked(False)
|
|
|
|
for ch in range(self.NUM_CHANNELS):
|
|
|
|
if self.autotuners.get_state(ch) != PIDAutotuneState.STATE_OFF:
|
|
|
|
if self.thermostat.connection_errored:
|
|
|
|
# Don't send any commands, just reset local state
|
|
|
|
self.autotuners.autotuners[ch].setOff()
|
|
|
|
else:
|
|
|
|
await self.autotuners.stop_pid_from_running(ch)
|
2023-06-30 11:27:31 +08:00
|
|
|
|
2023-08-16 17:35:13 +08:00
|
|
|
@asyncSlot(int)
|
|
|
|
async def on_report_box_stateChanged(self, enabled):
|
2024-05-13 10:35:21 +08:00
|
|
|
await self.thermostat.set_report_mode(enabled)
|
2023-06-30 11:27:31 +08:00
|
|
|
|
|
|
|
@asyncClose
|
2024-08-14 11:54:21 +08:00
|
|
|
async def closeEvent(self, _event):
|
2024-05-13 10:35:21 +08:00
|
|
|
try:
|
2024-08-26 17:09:29 +08:00
|
|
|
await self.thermostat.end_session()
|
2024-05-13 10:35:21 +08:00
|
|
|
except:
|
|
|
|
pass
|
2023-06-30 11:27:31 +08:00
|
|
|
|
|
|
|
@asyncSlot()
|
|
|
|
async def on_connect_btn_clicked(self):
|
2024-08-21 18:10:19 +08:00
|
|
|
if (self._connecting_task is None) and (not self.thermostat.connected()):
|
|
|
|
self._connecting_task = asyncio.create_task(
|
2024-08-23 13:10:51 +08:00
|
|
|
self.thermostat.start_session(
|
|
|
|
host=self.conn_menu.host_set_line.text(),
|
|
|
|
port=self.conn_menu.port_set_spin.value(),
|
|
|
|
)
|
2024-08-21 18:10:19 +08:00
|
|
|
)
|
|
|
|
try:
|
2024-08-27 10:49:06 +08:00
|
|
|
await self._connecting_task
|
2024-08-22 16:51:58 +08:00
|
|
|
except (OSError, asyncio.CancelledError) as exc:
|
2024-08-26 17:09:29 +08:00
|
|
|
await self.thermostat.end_session()
|
2024-08-21 18:10:19 +08:00
|
|
|
if isinstance(exc, asyncio.CancelledError):
|
2023-06-30 11:27:31 +08:00
|
|
|
return
|
2024-08-21 18:10:19 +08:00
|
|
|
raise
|
|
|
|
finally:
|
|
|
|
self._connecting_task = None
|
2023-06-30 11:27:31 +08:00
|
|
|
|
2024-08-21 18:10:19 +08:00
|
|
|
elif self._connecting_task is not None:
|
|
|
|
self._connecting_task.cancel()
|
|
|
|
else:
|
2024-08-26 17:09:29 +08:00
|
|
|
await self.thermostat.end_session()
|
2023-06-27 17:34:39 +08:00
|
|
|
|
2024-08-27 16:46:48 +08:00
|
|
|
@asyncSlot(int, PIDAutotuneState)
|
|
|
|
async def pid_autotune_handler(self, _ch, _state):
|
2024-05-13 10:35:21 +08:00
|
|
|
ch_tuning = []
|
|
|
|
for ch in range(self.NUM_CHANNELS):
|
2024-08-27 18:24:56 +08:00
|
|
|
if self.autotuners.get_state(ch) in {
|
|
|
|
PIDAutotuneState.STATE_READY,
|
|
|
|
PIDAutotuneState.STATE_RELAY_STEP_UP,
|
|
|
|
PIDAutotuneState.STATE_RELAY_STEP_DOWN,
|
|
|
|
}:
|
|
|
|
ch_tuning.append(ch)
|
2024-05-13 10:35:21 +08:00
|
|
|
|
|
|
|
if len(ch_tuning) == 0:
|
|
|
|
self.background_task_lbl.setText("Ready.")
|
|
|
|
self.loading_spinner.hide()
|
|
|
|
self.loading_spinner.stop()
|
|
|
|
else:
|
|
|
|
self.background_task_lbl.setText(
|
|
|
|
"Autotuning channel {ch}...".format(ch=ch_tuning)
|
|
|
|
)
|
|
|
|
self.loading_spinner.start()
|
|
|
|
self.loading_spinner.show()
|
|
|
|
|
2023-06-27 17:34:39 +08:00
|
|
|
|
|
|
|
async def coro_main():
|
2023-05-19 11:23:39 +08:00
|
|
|
args = get_argparser().parse_args()
|
|
|
|
if args.logLevel:
|
|
|
|
logging.basicConfig(level=getattr(logging, args.logLevel))
|
|
|
|
|
2023-06-27 17:34:39 +08:00
|
|
|
app_quit_event = asyncio.Event()
|
2023-06-26 10:20:48 +08:00
|
|
|
|
2023-06-27 17:34:39 +08:00
|
|
|
app = QtWidgets.QApplication.instance()
|
|
|
|
app.aboutToQuit.connect(app_quit_event.set)
|
2024-05-13 10:35:21 +08:00
|
|
|
app.setWindowIcon(
|
2024-07-03 14:40:13 +08:00
|
|
|
QtGui.QIcon(
|
|
|
|
str(importlib.resources.files("pytec.gui.resources").joinpath("artiq.ico"))
|
|
|
|
)
|
2024-05-13 10:35:21 +08:00
|
|
|
)
|
2023-06-26 10:20:48 +08:00
|
|
|
|
2023-06-30 11:27:31 +08:00
|
|
|
main_window = MainWindow(args)
|
2023-05-19 11:23:39 +08:00
|
|
|
main_window.show()
|
2023-06-26 10:20:48 +08:00
|
|
|
|
2023-06-27 17:34:39 +08:00
|
|
|
await app_quit_event.wait()
|
|
|
|
|
|
|
|
|
|
|
|
def main():
|
|
|
|
qasync.run(coro_main())
|
2023-05-19 11:23:39 +08:00
|
|
|
|
|
|
|
|
2024-05-13 10:35:21 +08:00
|
|
|
if __name__ == "__main__":
|
2023-05-19 11:23:39 +08:00
|
|
|
main()
|