artiq/artiq/gui/scanwidget.py

216 lines
7.1 KiB
Python
Raw Normal View History

import logging
2016-02-26 03:34:04 +08:00
from PyQt5 import QtGui, QtCore, QtWidgets
import numpy as np
2016-02-26 03:34:04 +08:00
from .ticker import Ticker
2016-02-26 03:34:04 +08:00
logger = logging.getLogger(__name__)
2016-03-17 05:08:32 +08:00
class ScanWidget(QtWidgets.QWidget):
startChanged = QtCore.pyqtSignal(float)
stopChanged = QtCore.pyqtSignal(float)
numChanged = QtCore.pyqtSignal(int)
def __init__(self, zoomFactor=1.05, zoomMargin=.1, dynamicRange=1e9):
2016-03-17 05:08:32 +08:00
QtWidgets.QWidget.__init__(self)
self.zoomMargin = zoomMargin
self.dynamicRange = dynamicRange
self.zoomFactor = zoomFactor
2016-02-26 03:34:04 +08:00
self.ticker = Ticker()
self.menu = QtWidgets.QMenu(self)
action = QtWidgets.QAction("&View Range", self)
action.triggered.connect(self.viewRange)
self.menu.addAction(action)
action = QtWidgets.QAction("&Snap Range", self)
action.triggered.connect(self.snapRange)
self.menu.addAction(action)
qfm = QtGui.QFontMetrics(self.font())
self._labelSize = QtCore.QSize(
(self.ticker.precision + 5)*qfm.averageCharWidth(),
qfm.lineSpacing())
self._start, self._stop, self._num = None, None, None
2016-03-17 05:08:32 +08:00
self._axisView = None, None
2016-03-16 22:59:49 +08:00
self._offset, self._pressed, self._dragLeft = None, None, None
2016-02-26 03:34:04 +08:00
def contextMenuEvent(self, ev):
self.menu.popup(ev.globalPos())
2016-03-17 04:13:14 +08:00
def sizeHint(self):
return self.minimumSizeHint()
def minimumSizeHint(self):
return QtCore.QSize(2.5*3*self._labelSize.width(),
4*self._labelSize.height())
2016-03-17 04:13:14 +08:00
def _axisToPixel(self, val):
a, b = self._axisView
return a + val*b
def _pixelToAxis(self, val):
a, b = self._axisView
return (val - a)/b
def _setView(self, axis_left, axis_scale):
self._axisView = axis_left, axis_scale
self.update()
def setStart(self, val):
if self._start == val:
return
self._start = val
self.update()
self.startChanged.emit(val)
def setStop(self, val):
if self._stop == val:
return
self._stop = val
self.update()
self.stopChanged.emit(val)
def setNum(self, val):
if self._num == val:
return
self._num = val
self.update()
self.numChanged.emit(val)
def viewRange(self):
center = (self._stop + self._start)/2
scale = self.width()*(1 - 2*self.zoomMargin)
if self._stop != self._start:
scale /= abs(self._stop - self._start)
if center:
scale = min(scale, self.dynamicRange/abs(center))
else:
scale = self.dynamicRange
if center:
scale /= abs(center)
left = self.width()/2 - center*scale
self._setView(left, scale)
def snapRange(self):
self.setStart(self._pixelToAxis(self.zoomMargin*self.width()))
self.setStop(self._pixelToAxis((1 - self.zoomMargin)*self.width()))
2016-02-26 03:34:04 +08:00
def _hitHandle(self, pos, val):
2016-03-17 05:08:32 +08:00
qfm = QtGui.QFontMetrics(self.font())
val = self._axisToPixel(val)
rect = QtCore.QRect(val - qfm.lineSpacing()/2, 3*qfm.lineSpacing(),
qfm.lineSpacing(), qfm.lineSpacing())
return rect.contains(pos)
2016-02-26 03:34:04 +08:00
def mousePressEvent(self, ev):
if ev.buttons() ^ ev.button():
2016-02-26 03:34:04 +08:00
ev.ignore()
return
if self._hitHandle(ev.pos(), self._stop):
self._pressed = "stop"
2016-03-17 05:08:32 +08:00
self._offset = ev.x() - self._axisToPixel(self._stop)
elif self._hitHandle(ev.pos(), self._start):
self._pressed = "start"
2016-03-17 05:08:32 +08:00
self._offset = ev.x() - self._axisToPixel(self._stop)
else:
2016-03-16 22:59:49 +08:00
self._pressed = "axis"
self._offset = ev.x()
self._dragLeft = self._axisView[0]
2016-02-26 03:34:04 +08:00
def mouseMoveEvent(self, ev):
if not self._pressed:
2016-02-26 03:34:04 +08:00
ev.ignore()
return
if self._pressed == "stop":
2016-03-16 22:59:49 +08:00
self._stop = self._pixelToAxis(ev.x() - self._offset)
self.update()
2016-03-17 05:08:32 +08:00
self.stopChanged.emit(self._stop)
elif self._pressed == "start":
2016-03-16 22:59:49 +08:00
self._start = self._pixelToAxis(ev.x() - self._offset)
self.update()
2016-03-17 05:08:32 +08:00
self.startChanged.emit(self._start)
2016-03-16 22:59:49 +08:00
elif self._pressed == "axis":
self._setView(self._dragLeft + ev.x() - self._offset,
self._axisView[1])
2016-02-26 03:34:04 +08:00
def mouseReleaseEvent(self, ev):
if self._pressed == "start":
self.startChanged.emit(self._start)
elif self._pressed == "stop":
self.stopChanged.emit(self._stop)
self._pressed = None
def _zoom(self, z, x):
a, b = self._axisView
scale = z*b
left = x - z*(x - a)
if z > 1 and abs(left - self.width()/2) > self.dynamicRange:
return
self._setView(left, scale)
def wheelEvent(self, ev):
y = ev.angleDelta().y()/120.
if ev.modifiers() & QtCore.Qt.ShiftModifier:
if y:
self.setNum(max(1, self._num + y))
elif ev.modifiers() & QtCore.Qt.ControlModifier:
if y:
self._zoom(self.zoomFactor**y, ev.x())
else:
ev.ignore()
2016-02-26 03:34:04 +08:00
def resizeEvent(self, ev):
if not ev.oldSize().isValid():
self.viewRange()
return
a, b = self._axisView
scale = b*ev.size().width()/ev.oldSize().width()
center = (self._stop + self._start)/2
if center:
scale = min(scale, self.dynamicRange/abs(center))
left = a*scale/b
self.ticker.min_ticks = int(ev.size().width() /
(2.5*self._labelSize.width()))
self._setView(left, scale)
def paintEvent(self, ev):
painter = QtGui.QPainter(self)
qfm = QtGui.QFontMetrics(painter.font())
avgCharWidth = qfm.averageCharWidth()
lineSpacing = qfm.lineSpacing()
descent = qfm.descent()
2016-03-17 04:13:14 +08:00
painter.translate(0, lineSpacing)
ticks, prefix, labels = self.ticker(self._pixelToAxis(0),
self._pixelToAxis(self.width()))
painter.drawText(0, 0, prefix)
painter.translate(0, lineSpacing)
2016-03-17 04:13:14 +08:00
painter.setPen(QtGui.QPen(QtCore.Qt.black, 2, QtCore.Qt.SolidLine))
for t, l in zip(ticks, labels):
t = self._axisToPixel(t)
painter.drawText(t - len(l)/2*avgCharWidth, 0, l)
2016-03-17 04:13:14 +08:00
painter.drawLine(t, descent, t, (lineSpacing + descent)/2)
painter.translate(0, (lineSpacing + descent)/2)
painter.drawLine(0, 0, self.width(), 0)
for p in np.linspace(self._axisToPixel(self._start),
self._axisToPixel(self._stop),
self._num):
2016-03-17 04:13:14 +08:00
painter.drawLine(p, 0, p, (lineSpacing - descent)/2)
painter.translate(0, (lineSpacing - descent)/2)
for x, c in (self._start, QtCore.Qt.blue), (self._stop, QtCore.Qt.red):
x = self._axisToPixel(x)
2016-03-17 05:08:32 +08:00
painter.setPen(c)
painter.setBrush(c)
painter.drawPolygon(*(QtCore.QPointF(*i) for i in [
(x, 0), (x - lineSpacing/2, lineSpacing),
(x + lineSpacing/2, lineSpacing)]))