Merged from Ingo B. with changes

Implements single axis scaling by mouse wheel over tick marks
This commit is contained in:
Luke Campagnola 2011-04-05 12:53:05 -04:00
commit a5668a1a26
12 changed files with 156 additions and 20 deletions

83
ColorButton.py Normal file
View File

@ -0,0 +1,83 @@
# -*- coding: utf-8 -*-
from PyQt4 import QtGui, QtCore
if not hasattr(QtCore, 'Signal'):
QtCore.Signal = QtCore.pyqtSignal
import functions
class ColorButton(QtGui.QPushButton):
sigColorChanging = QtCore.Signal(object) ## emitted whenever a new color is picked in the color dialog
sigColorChanged = QtCore.Signal(object) ## emitted when the selected color is accepted (user clicks OK)
def __init__(self, color=(128,128,128)):
QtGui.QPushButton.__init__(self)
self.setColor(color)
self.colorDialog = QtGui.QColorDialog()
self.colorDialog.setOption(QtGui.QColorDialog.ShowAlphaChannel, True)
self.colorDialog.setOption(QtGui.QColorDialog.DontUseNativeDialog, True)
self.colorDialog.currentColorChanged.connect(self.dialogColorChanged)
self.colorDialog.rejected.connect(self.colorRejected)
self.colorDialog.colorSelected.connect(self.colorSelected)
#QtCore.QObject.connect(self.colorDialog, QtCore.SIGNAL('currentColorChanged(const QColor&)'), self.currentColorChanged)
#QtCore.QObject.connect(self.colorDialog, QtCore.SIGNAL('rejected()'), self.currentColorRejected)
self.clicked.connect(self.selectColor)
self.setMinimumHeight(15)
self.setMinimumWidth(15)
def paintEvent(self, ev):
QtGui.QPushButton.paintEvent(self, ev)
p = QtGui.QPainter(self)
p.setBrush(functions.mkBrush(self._color))
p.drawRect(self.rect().adjusted(5, 5, -5, -5))
p.end()
def setColor(self, color, finished=True):
self._color = functions.mkColor(color)
if finished:
self.sigColorChanged.emit(self)
else:
self.sigColorChanging.emit(self)
self.update()
def selectColor(self):
self.origColor = self.color()
self.colorDialog.setCurrentColor(self.color())
self.colorDialog.open()
def dialogColorChanged(self, color):
if color.isValid():
self.setColor(color, finished=False)
def colorRejected(self):
self.setColor(self.origColor, finished=False)
def colorSelected(self, color):
self.setColor(self._color, finished=True)
def saveState(self):
return functions.colorTuple(self._color)
def restoreState(self, state):
self.setColor(state)
def color(self):
return functions.mkColor(self._color)
def widgetGroupInterface(self):
return (self.sigColorChanged, ColorButton.saveState, ColorButton.restoreState)
if __name__ == '__main__':
app = QtGui.QApplication([])
win = QtGui.QMainWindow()
btn = ColorButton()
win.setCentralWidget(btn)
win.show()
def change(btn):
print "change", btn.color()
def done(btn):
print "done", btn.color()
btn.sigColorChanging.connect(change)
btn.sigColorChanged.connect(done)

View File

@ -254,15 +254,14 @@ class GraphicsView(QtGui.QGraphicsView):
def wheelEvent(self, ev): def wheelEvent(self, ev):
QtGui.QGraphicsView.wheelEvent(self, ev)
if not self.mouseEnabled: if not self.mouseEnabled:
return return
QtGui.QGraphicsView.wheelEvent(self, ev)
sc = 1.001 ** ev.delta() sc = 1.001 ** ev.delta()
#self.scale *= sc #self.scale *= sc
#self.updateMatrix() #self.updateMatrix()
self.scale(sc, sc) self.scale(sc, sc)
def setAspectLocked(self, s): def setAspectLocked(self, s):
self.aspectLocked = s self.aspectLocked = s

View File

@ -1037,6 +1037,10 @@ class PlotItem(QtGui.QGraphicsWidget):
mode = False mode = False
return mode return mode
#def wheelEvent(self, ev):
# disables panning the whole scene by mousewheel
#ev.accept()
def resizeEvent(self, ev): def resizeEvent(self, ev):
self.ctrlBtn.move(0, self.size().height() - self.ctrlBtn.size().height()) self.ctrlBtn.move(0, self.size().height() - self.ctrlBtn.size().height())
self.autoBtn.move(self.ctrlBtn.width(), self.size().height() - self.autoBtn.size().height()) self.autoBtn.move(self.ctrlBtn.width(), self.size().height() - self.autoBtn.size().height())
@ -1088,7 +1092,7 @@ class PlotItem(QtGui.QGraphicsWidget):
if arr.ndim != 1: if arr.ndim != 1:
raise Exception("Array must be 1D to plot (shape is %s)" % arr.shape) raise Exception("Array must be 1D to plot (shape is %s)" % arr.shape)
if x is None: if x is None:
x = arange(arr.shape[0]) x = np.arange(arr.shape[0])
if x.ndim != 1: if x.ndim != 1:
raise Exception("X array must be 1D to plot (shape is %s)" % x.shape) raise Exception("X array must be 1D to plot (shape is %s)" % x.shape)
c = PlotCurveItem(arr, x=x) c = PlotCurveItem(arr, x=x)

28
examples/test_Arrow.py Normal file
View File

@ -0,0 +1,28 @@
# -*- coding: utf-8 -*-
## Add path to library (just for examples; you do not need this)
import sys, os
sys.path.insert(0, os.path.join(os.path.dirname(__file__), '..', '..'))
import numpy as np
from PyQt4 import QtGui, QtCore
import pyqtgraph as pg
app = QtGui.QApplication([])
mw = QtGui.QMainWindow()
p = pg.PlotWidget()
mw.setCentralWidget(p)
c = p.plot(x=np.sin(np.linspace(0, 2*np.pi, 100)), y=np.cos(np.linspace(0, 2*np.pi, 100)))
a = pg.CurveArrow(c)
p.addItem(a)
mw.show()
anim = a.makeAnimation(loop=-1)
anim.start()
## Start Qt event loop unless running in interactive mode.
if sys.flags.interactive != 1:
app.exec_()

View File

@ -1,7 +1,7 @@
# -*- coding: utf-8 -*- # -*- coding: utf-8 -*-
## Add path to library (just for examples; you do not need this) ## Add path to library (just for examples; you do not need this)
import sys, os import sys, os
sys.path.append(os.path.join(os.path.dirname(__file__), '..', '..')) sys.path.insert(0, os.path.join(os.path.dirname(__file__), '..', '..'))
from PyQt4 import QtCore, QtGui from PyQt4 import QtCore, QtGui

View File

@ -1,7 +1,7 @@
# -*- coding: utf-8 -*- # -*- coding: utf-8 -*-
## Add path to library (just for examples; you do not need this) ## Add path to library (just for examples; you do not need this)
import sys, os import sys, os
sys.path.append(os.path.join(os.path.dirname(__file__), '..', '..')) sys.path.insert(0, os.path.join(os.path.dirname(__file__), '..', '..'))
import numpy as np import numpy as np

View File

@ -2,7 +2,7 @@
# -*- coding: utf-8 -*- # -*- coding: utf-8 -*-
## Add path to library (just for examples; you do not need this) ## Add path to library (just for examples; you do not need this)
import sys, os import sys, os
sys.path.append(os.path.join(os.path.dirname(__file__), '..', '..')) sys.path.insert(0, os.path.join(os.path.dirname(__file__), '..', '..'))
from scipy import random from scipy import random

2
examples/test_PlotWidget.py Executable file → Normal file
View File

@ -2,7 +2,7 @@
# -*- coding: utf-8 -*- # -*- coding: utf-8 -*-
## Add path to library (just for examples; you do not need this) ## Add path to library (just for examples; you do not need this)
import sys, os import sys, os
sys.path.append(os.path.join(os.path.dirname(__file__), '..', '..')) sys.path.insert(0, os.path.join(os.path.dirname(__file__), '..', '..'))
from PyQt4 import QtGui, QtCore from PyQt4 import QtGui, QtCore

View File

@ -2,7 +2,7 @@
# -*- coding: utf-8 -*- # -*- coding: utf-8 -*-
## Add path to library (just for examples; you do not need this) ## Add path to library (just for examples; you do not need this)
import sys, os import sys, os
sys.path.append(os.path.join(os.path.dirname(__file__), '..', '..')) sys.path.insert(0, os.path.join(os.path.dirname(__file__), '..', '..'))
from PyQt4 import QtCore, QtGui from PyQt4 import QtCore, QtGui

View File

@ -1,7 +1,7 @@
# -*- coding: utf-8 -*- # -*- coding: utf-8 -*-
## Add path to library (just for examples; you do not need this) ## Add path to library (just for examples; you do not need this)
import sys, os import sys, os
sys.path = [os.path.join(os.path.dirname(__file__), '..', '..')] + sys.path sys.path.insert(0, os.path.join(os.path.dirname(__file__), '..', '..'))
from PyQt4 import QtCore, QtGui from PyQt4 import QtCore, QtGui

View File

@ -2,7 +2,7 @@
# -*- coding: utf-8 -*- # -*- coding: utf-8 -*-
## Add path to library (just for examples; you do not need this) ## Add path to library (just for examples; you do not need this)
import sys, os import sys, os
sys.path.append(os.path.join(os.path.dirname(__file__), '..', '..')) sys.path.insert(0, os.path.join(os.path.dirname(__file__), '..', '..'))
## This example uses a ViewBox to create a PlotWidget-like interface ## This example uses a ViewBox to create a PlotWidget-like interface

View File

@ -1524,8 +1524,10 @@ class ScaleItem(QtGui.QGraphicsWidget):
else: else:
xs = bounds.width() / dif xs = bounds.width() / dif
tickPositions = set() # remembers positions of previously drawn ticks
## draw ticks and text ## draw ticks and text
for i in [i1, i1+1, i1+2]: ## draw three different intervals ## draw three different intervals, long ticks first
for i in reversed([i1, i1+1, i1+2]):
if i > len(intervals): if i > len(intervals):
continue continue
## spacing for this interval ## spacing for this interval
@ -1569,7 +1571,11 @@ class ScaleItem(QtGui.QGraphicsWidget):
if p1[1-axis] < 0: if p1[1-axis] < 0:
continue continue
p.setPen(QtGui.QPen(QtGui.QColor(100, 100, 100, a))) p.setPen(QtGui.QPen(QtGui.QColor(100, 100, 100, a)))
p.drawLine(Point(p1), Point(p2)) # draw tick only if there is none
tickPos = p1[1-axis]
if tickPos not in tickPositions:
p.drawLine(Point(p1), Point(p2))
tickPositions.add(tickPos)
if i == textLevel: if i == textLevel:
if abs(v) < .001 or abs(v) >= 10000: if abs(v) < .001 or abs(v) >= 10000:
vstr = "%g" % (v * self.scale) vstr = "%g" % (v * self.scale)
@ -1635,11 +1641,15 @@ class ScaleItem(QtGui.QGraphicsWidget):
else: else:
self.setHeight(0) self.setHeight(0)
QtGui.QGraphicsWidget.hide(self) QtGui.QGraphicsWidget.hide(self)
def wheelEvent(self, ev):
if self.linkedView is None or self.linkedView() is None: return
if self.orientation in ['left', 'right']:
self.linkedView().wheelEvent(ev, axis=1)
else:
self.linkedView().wheelEvent(ev, axis=0)
ev.accept()
class ViewBox(QtGui.QGraphicsWidget): class ViewBox(QtGui.QGraphicsWidget):
@ -1655,7 +1665,7 @@ class ViewBox(QtGui.QGraphicsWidget):
#self.gView = view #self.gView = view
#self.showGrid = showGrid #self.showGrid = showGrid
self.range = [[0,1], [0,1]] ## child coord. range visible [[xmin, xmax], [ymin, ymax]] self.range = [[0,1], [0,1]] ## child coord. range visible [[xmin, xmax], [ymin, ymax]]
self.wheelScaleFactor = -1.0 / 8.0
self.aspectLocked = False self.aspectLocked = False
self.setFlag(QtGui.QGraphicsItem.ItemClipsChildrenToShape) self.setFlag(QtGui.QGraphicsItem.ItemClipsChildrenToShape)
#self.setFlag(QtGui.QGraphicsItem.ItemClipsToShape) #self.setFlag(QtGui.QGraphicsItem.ItemClipsToShape)
@ -1800,7 +1810,19 @@ class ViewBox(QtGui.QGraphicsWidget):
#self.replot(autoRange=False) #self.replot(autoRange=False)
#self.updateMatrix() #self.updateMatrix()
def wheelEvent(self, ev, axis=None):
mask = np.array(self.mouseEnabled, dtype=np.float)
if axis is not None and axis >= 0 and axis < len(mask):
mv = mask[axis]
mask[:] = 0
mask[axis] = mv
s = ((mask * 0.02) + 1) ** (ev.delta() * self.wheelScaleFactor) # actual scaling factor
# scale 'around' mouse cursor position
center = Point(self.childGroup.transform().inverted()[0].map(ev.pos()))
self.scaleBy(s, center)
self.emit(QtCore.SIGNAL('rangeChangedManually'), self.mouseEnabled)
ev.accept()
def mouseMoveEvent(self, ev): def mouseMoveEvent(self, ev):
QtGui.QGraphicsWidget.mouseMoveEvent(self, ev) QtGui.QGraphicsWidget.mouseMoveEvent(self, ev)
pos = np.array([ev.pos().x(), ev.pos().y()]) pos = np.array([ev.pos().x(), ev.pos().y()])
@ -1812,7 +1834,7 @@ class ViewBox(QtGui.QGraphicsWidget):
mask = np.array(self.mouseEnabled, dtype=np.float) mask = np.array(self.mouseEnabled, dtype=np.float)
## Scale or translate based on mouse button ## Scale or translate based on mouse button
if ev.buttons() & QtCore.Qt.LeftButton: if ev.buttons() & (QtCore.Qt.LeftButton | QtCore.Qt.MidButton):
if not self.yInverted: if not self.yInverted:
mask *= np.array([1, -1]) mask *= np.array([1, -1])
tr = dif*mask tr = dif*mask