artiq/artiq/gui/applets.py

282 lines
11 KiB
Python
Raw Normal View History

2016-01-07 20:23:17 +08:00
import logging
import asyncio
2016-01-16 01:15:06 +08:00
import sys
import shlex
from functools import partial
2016-01-07 20:23:17 +08:00
from quamash import QtCore, QtGui, QtWidgets
from pyqtgraph import dockarea
2016-02-08 16:59:15 +08:00
from artiq.protocols.pipe_ipc import AsyncioParentComm
2016-02-08 21:35:37 +08:00
from artiq.protocols import pyon
2016-02-08 16:59:15 +08:00
2016-01-07 20:23:17 +08:00
logger = logging.getLogger(__name__)
2016-02-08 16:59:15 +08:00
class AppletIPCServer(AsyncioParentComm):
def write_pyon(self, obj):
self.write(pyon.encode(obj).encode() + b"\n")
async def read_pyon(self):
line = await self.readline()
return pyon.decode(line.decode())
2016-02-08 21:35:37 +08:00
async def serve(self, embed_cb):
try:
while True:
obj = await self.read_pyon()
try:
action = obj["action"]
if action == "embed":
embed_cb(obj["win_id"])
self.write_pyon({"action": "embed_done"})
elif action == "subscribe":
print("applet subscribed: ", obj["datasets"])
else:
raise ValueError("unknown action in applet request")
except:
logger.warning("error processing applet request",
exc_info=True)
self.write_pyon({"action": "error"})
except asyncio.CancelledError:
pass
except:
logger.error("error processing data from applet, "
"server stopped", exc_info=True)
def start(self, embed_cb):
self.server_task = asyncio.ensure_future(self.serve(embed_cb))
async def stop(self):
self.server_task.cancel()
await asyncio.wait([self.server_task])
2016-02-08 16:59:15 +08:00
2016-01-07 20:23:17 +08:00
class AppletDock(dockarea.Dock):
2016-02-08 16:59:15 +08:00
def __init__(self, name, command):
2016-02-08 21:35:37 +08:00
dockarea.Dock.__init__(self, "applet" + str(id(self)), # TODO
2016-01-07 20:23:17 +08:00
label="Applet: " + name,
closable=True)
self.setMinimumSize(QtCore.QSize(500, 400))
self.applet_name = name
self.command = command
def rename(self, name):
self.applet_name = name
self.label.setText("Applet: " + name)
async def start(self):
2016-02-08 21:35:37 +08:00
self.ipc = AppletIPCServer()
2016-01-16 01:15:06 +08:00
command = self.command.format(python=sys.executable,
2016-02-08 16:59:15 +08:00
ipc_address=self.ipc.get_address())
logger.debug("starting command %s for %s", command, self.applet_name)
try:
2016-02-08 16:59:15 +08:00
await self.ipc.create_subprocess(*shlex.split(command))
except:
logger.warning("Applet %s failed to start", self.applet_name,
exc_info=True)
2016-02-08 21:35:37 +08:00
self.ipc.start(self.embed)
2016-01-07 20:23:17 +08:00
2016-02-08 21:35:37 +08:00
def embed(self, win_id):
logger.debug("capturing window 0x%x for %s", win_id, self.applet_name)
2016-02-08 21:35:37 +08:00
embed_window = QtGui.QWindow.fromWinId(win_id)
embed_widget = QtWidgets.QWidget.createWindowContainer(embed_window)
self.addWidget(embed_widget)
2016-01-07 20:23:17 +08:00
async def terminate(self):
2016-02-08 21:35:37 +08:00
if hasattr(self, "ipc"):
await self.ipc.stop()
self.ipc.write_pyon({"action": "terminate"})
try:
2016-02-08 16:59:15 +08:00
await asyncio.wait_for(self.ipc.process.wait(), 2.0)
except:
logger.warning("Applet %s failed to exit, killing",
self.applet_name)
try:
2016-02-08 16:59:15 +08:00
self.ipc.process.kill()
except ProcessLookupError:
pass
2016-02-08 16:59:15 +08:00
await self.ipc.process.wait()
del self.ipc
async def restart(self):
await self.terminate()
await self.start()
2016-01-07 20:23:17 +08:00
2016-01-16 01:15:06 +08:00
_templates = [
("Big number", "{python} -m artiq.applets.big_number "
2016-02-08 16:59:15 +08:00
"embedded {ipc_address} NUMBER_DATASET"),
2016-01-16 01:15:06 +08:00
("Histogram", "{python} -m artiq.applets.plot_hist "
2016-02-08 16:59:15 +08:00
"embedded {ipc_address} COUNTS_DATASET "
2016-01-16 01:15:06 +08:00
"--x BIN_BOUNDARIES_DATASET"),
("XY", "{python} -m artiq.applets.plot_xy "
2016-02-08 16:59:15 +08:00
"embedded {ipc_address} Y_DATASET --x X_DATASET "
2016-01-16 01:15:06 +08:00
"--error ERROR_DATASET --fit FIT_DATASET"),
("XY + Histogram", "{python} -m artiq.applets.plot_xy_hist "
2016-02-08 16:59:15 +08:00
"embedded {ipc_address} X_DATASET "
2016-01-16 01:15:06 +08:00
"HIST_BIN_BOUNDARIES_DATASET "
"HISTS_COUNTS_DATASET"),
]
2016-01-07 20:23:17 +08:00
class AppletsDock(dockarea.Dock):
2016-02-08 16:59:15 +08:00
def __init__(self, dock_area):
self.dock_area = dock_area
self.dock_to_checkbox = dict()
self.workaround_pyqtgraph_bug = False
2016-01-07 20:23:17 +08:00
dockarea.Dock.__init__(self, "Applets")
self.setMinimumSize(QtCore.QSize(850, 450))
self.table = QtWidgets.QTableWidget(0, 3)
self.table.setHorizontalHeaderLabels(["Enable", "Name", "Command"])
self.table.setSelectionBehavior(QtGui.QAbstractItemView.SelectRows)
self.table.setSelectionMode(QtGui.QAbstractItemView.SingleSelection)
self.table.horizontalHeader().setStretchLastSection(True)
self.table.horizontalHeader().setResizeMode(
QtGui.QHeaderView.ResizeToContents)
self.table.verticalHeader().setResizeMode(
QtGui.QHeaderView.ResizeToContents)
self.table.verticalHeader().hide()
self.table.setTextElideMode(QtCore.Qt.ElideNone)
self.addWidget(self.table)
self.table.setContextMenuPolicy(QtCore.Qt.ActionsContextMenu)
new_action = QtGui.QAction("New applet", self.table)
new_action.triggered.connect(self.new)
self.table.addAction(new_action)
2016-01-16 01:15:06 +08:00
templates_menu = QtGui.QMenu()
for name, template in _templates:
action = QtGui.QAction(name, self.table)
action.triggered.connect(partial(self.new_template, template))
templates_menu.addAction(action)
restart_action = QtGui.QAction("New applet from template", self.table)
restart_action.setMenu(templates_menu)
self.table.addAction(restart_action)
2016-01-07 20:23:17 +08:00
restart_action = QtGui.QAction("Restart selected applet", self.table)
2016-01-17 04:12:16 +08:00
restart_action.setShortcut("CTRL+R")
restart_action.setShortcutContext(QtCore.Qt.WidgetShortcut)
restart_action.triggered.connect(self.restart)
2016-01-07 20:23:17 +08:00
self.table.addAction(restart_action)
delete_action = QtGui.QAction("Delete selected applet", self.table)
2016-01-17 04:12:16 +08:00
delete_action.setShortcut("DELETE")
delete_action.setShortcutContext(QtCore.Qt.WidgetShortcut)
2016-01-07 20:23:17 +08:00
delete_action.triggered.connect(self.delete)
self.table.addAction(delete_action)
self.table.cellChanged.connect(self.cell_changed)
2016-02-08 16:59:15 +08:00
def create(self, name, command):
dock = AppletDock(name, command)
# If a dock is floated and then dock state is restored, pyqtgraph
# leaves a "phantom" window open.
if self.workaround_pyqtgraph_bug:
self.dock_area.addDock(dock)
else:
self.dock_area.floatDock(dock)
asyncio.ensure_future(dock.start())
dock.sigClosed.connect(partial(self.on_dock_closed, dock))
return dock
2016-01-07 20:23:17 +08:00
def cell_changed(self, row, column):
if column == 0:
item = self.table.item(row, column)
if item.checkState() == QtCore.Qt.Checked:
command = self.table.item(row, 2)
if command:
command = command.text()
name = self.table.item(row, 1)
if name is None:
name = ""
else:
name = name.text()
2016-02-08 16:59:15 +08:00
dock = self.create(name, command)
item.applet_dock = dock
self.dock_to_checkbox[dock] = item
2016-01-07 20:23:17 +08:00
else:
2016-02-08 16:59:15 +08:00
dock = getattr(item, "applet_dock", None)
if dock is not None:
# This calls self.on_dock_closed
dock.close()
elif column == 1 or column == 2:
new_value = self.table.item(row, column).text()
2016-02-08 16:59:15 +08:00
dock = getattr(self.table.item(row, 0), "applet_dock", None)
if dock is not None:
if column == 1:
2016-02-08 16:59:15 +08:00
dock.rename(new_value)
else:
2016-02-08 16:59:15 +08:00
dock.command = new_value
2016-02-08 16:59:15 +08:00
def on_dock_closed(self, dock):
asyncio.ensure_future(dock.terminate())
checkbox_item = self.dock_to_checkbox[dock]
checkbox_item.applet_dock = None
del self.dock_to_checkbox[dock]
checkbox_item.setCheckState(QtCore.Qt.Unchecked)
2016-01-07 20:23:17 +08:00
def new(self):
row = self.table.rowCount()
self.table.insertRow(row)
checkbox = QtWidgets.QTableWidgetItem()
checkbox.setFlags(QtCore.Qt.ItemIsSelectable |
QtCore.Qt.ItemIsUserCheckable |
QtCore.Qt.ItemIsEnabled)
checkbox.setCheckState(QtCore.Qt.Unchecked)
self.table.setItem(row, 0, checkbox)
2016-01-16 01:15:06 +08:00
self.table.setItem(row, 1, QtWidgets.QTableWidgetItem())
self.table.setItem(row, 2, QtWidgets.QTableWidgetItem())
2016-01-09 19:29:20 +08:00
return row
2016-01-07 20:23:17 +08:00
2016-01-16 01:15:06 +08:00
def new_template(self, template):
row = self.new()
self.table.item(row, 2).setText(template)
def restart(self):
selection = self.table.selectedRanges()
if selection:
row = selection[0].topRow()
2016-02-08 16:59:15 +08:00
dock = getattr(self.table.item(row, 0), "applet_dock", None)
if dock is not None:
asyncio.ensure_future(dock.restart())
2016-01-07 20:23:17 +08:00
def delete(self):
selection = self.table.selectedRanges()
if selection:
row = selection[0].topRow()
2016-02-08 16:59:15 +08:00
dock = getattr(self.table.item(row, 0), "applet_dock", None)
if dock is not None:
# This calls self.on_dock_closed
dock.close()
2016-01-16 01:11:13 +08:00
self.table.removeRow(row)
2016-01-07 20:23:17 +08:00
2016-02-08 16:59:15 +08:00
async def stop(self):
for row in range(self.table.rowCount()):
dock = getattr(self.table.item(row, 0), "applet_dock", None)
if dock is not None:
await dock.terminate()
2016-01-09 19:29:20 +08:00
def save_state(self):
state = []
for row in range(self.table.rowCount()):
enabled = self.table.item(row, 0).checkState() == QtCore.Qt.Checked
name = self.table.item(row, 1).text()
command = self.table.item(row, 2).text()
state.append((enabled, name, command))
return state
def restore_state(self, state):
2016-02-08 16:59:15 +08:00
self.workaround_pyqtgraph_bug = True
2016-01-09 19:29:20 +08:00
for enabled, name, command in state:
row = self.new()
item = QtWidgets.QTableWidgetItem()
item.setText(name)
self.table.setItem(row, 1, item)
item = QtWidgets.QTableWidgetItem()
item.setText(command)
self.table.setItem(row, 2, item)
if enabled:
self.table.item(row, 0).setCheckState(QtCore.Qt.Checked)
self.workaround_pyqtgraph_bug = False