ViewBox overhaul. Cleaned up code pathway:
- setRange now only affects target range - updateViewRange only affects view range - updateMatrix only affects childGroup transform - updateMatrix is only called before a render Pathway now looks like: setRange -> updateViewRange -> matrix dirty -> ... -> sigRangeChanged ... -> prepareForPaint -> updateAutoRange, updateMatrix if dirty
This commit is contained in:
commit
58ed8ee7bd
@ -27,7 +27,7 @@ arr += np.random.normal(size=(100,100))
|
|||||||
|
|
||||||
## create GUI
|
## create GUI
|
||||||
app = QtGui.QApplication([])
|
app = QtGui.QApplication([])
|
||||||
w = pg.GraphicsWindow(size=(800,800), border=True)
|
w = pg.GraphicsWindow(size=(1000,800), border=True)
|
||||||
w.setWindowTitle('pyqtgraph example: ROI Examples')
|
w.setWindowTitle('pyqtgraph example: ROI Examples')
|
||||||
|
|
||||||
text = """Data Selection From Image.<br>\n
|
text = """Data Selection From Image.<br>\n
|
||||||
|
@ -13,7 +13,8 @@ from pyqtgraph.widgets.RemoteGraphicsView import RemoteGraphicsView
|
|||||||
app = pg.mkQApp()
|
app = pg.mkQApp()
|
||||||
|
|
||||||
## Create the widget
|
## Create the widget
|
||||||
v = RemoteGraphicsView(debug=False)
|
v = RemoteGraphicsView(debug=False) # setting debug=True causes both processes to print information
|
||||||
|
# about interprocess communication
|
||||||
v.show()
|
v.show()
|
||||||
v.setWindowTitle('pyqtgraph example: RemoteGraphicsView')
|
v.setWindowTitle('pyqtgraph example: RemoteGraphicsView')
|
||||||
|
|
||||||
|
@ -23,7 +23,9 @@ p2 = win.addPlot(row=2, col=0)
|
|||||||
|
|
||||||
region = pg.LinearRegionItem()
|
region = pg.LinearRegionItem()
|
||||||
region.setZValue(10)
|
region.setZValue(10)
|
||||||
p2.addItem(region)
|
# Add the LinearRegionItem to the ViewBox, but tell the ViewBox to exclude this
|
||||||
|
# item when doing auto-range calculations.
|
||||||
|
p2.addItem(region, ignoreBounds=True)
|
||||||
|
|
||||||
#pg.dbg()
|
#pg.dbg()
|
||||||
p1.setAutoVisible(y=True)
|
p1.setAutoVisible(y=True)
|
||||||
|
@ -139,7 +139,7 @@ def importModules(path, globals, locals, excludes=()):
|
|||||||
d = os.path.join(os.path.split(globals['__file__'])[0], path)
|
d = os.path.join(os.path.split(globals['__file__'])[0], path)
|
||||||
files = set()
|
files = set()
|
||||||
for f in frozenSupport.listdir(d):
|
for f in frozenSupport.listdir(d):
|
||||||
if frozenSupport.isdir(os.path.join(d, f)) and f != '__pycache__':
|
if frozenSupport.isdir(os.path.join(d, f)) and f not in ['__pycache__', 'tests']:
|
||||||
files.add(f)
|
files.add(f)
|
||||||
elif f[-3:] == '.py' and f != '__init__.py':
|
elif f[-3:] == '.py' and f != '__init__.py':
|
||||||
files.add(f[:-3])
|
files.add(f[:-3])
|
||||||
@ -311,13 +311,15 @@ def image(*args, **kargs):
|
|||||||
return w
|
return w
|
||||||
show = image ## for backward compatibility
|
show = image ## for backward compatibility
|
||||||
|
|
||||||
def dbg():
|
def dbg(*args, **kwds):
|
||||||
"""
|
"""
|
||||||
Create a console window and begin watching for exceptions.
|
Create a console window and begin watching for exceptions.
|
||||||
|
|
||||||
|
All arguments are passed to :func:`ConsoleWidget.__init__() <pyqtgraph.console.ConsoleWidget.__init__>`.
|
||||||
"""
|
"""
|
||||||
mkQApp()
|
mkQApp()
|
||||||
from . import console
|
from . import console
|
||||||
c = console.ConsoleWidget()
|
c = console.ConsoleWidget(*args, **kwds)
|
||||||
c.catchAllExceptions()
|
c.catchAllExceptions()
|
||||||
c.show()
|
c.show()
|
||||||
global consoles
|
global consoles
|
||||||
|
@ -12,6 +12,7 @@ class GraphicsObject(GraphicsItem, QtGui.QGraphicsObject):
|
|||||||
"""
|
"""
|
||||||
_qtBaseClass = QtGui.QGraphicsObject
|
_qtBaseClass = QtGui.QGraphicsObject
|
||||||
def __init__(self, *args):
|
def __init__(self, *args):
|
||||||
|
self.__inform_view_on_changes = True
|
||||||
QtGui.QGraphicsObject.__init__(self, *args)
|
QtGui.QGraphicsObject.__init__(self, *args)
|
||||||
self.setFlag(self.ItemSendsGeometryChanges)
|
self.setFlag(self.ItemSendsGeometryChanges)
|
||||||
GraphicsItem.__init__(self)
|
GraphicsItem.__init__(self)
|
||||||
@ -20,7 +21,7 @@ class GraphicsObject(GraphicsItem, QtGui.QGraphicsObject):
|
|||||||
ret = QtGui.QGraphicsObject.itemChange(self, change, value)
|
ret = QtGui.QGraphicsObject.itemChange(self, change, value)
|
||||||
if change in [self.ItemParentHasChanged, self.ItemSceneHasChanged]:
|
if change in [self.ItemParentHasChanged, self.ItemSceneHasChanged]:
|
||||||
self.parentChanged()
|
self.parentChanged()
|
||||||
if change in [self.ItemPositionHasChanged, self.ItemTransformHasChanged]:
|
if self.__inform_view_on_changes and change in [self.ItemPositionHasChanged, self.ItemTransformHasChanged]:
|
||||||
self.informViewBoundsChanged()
|
self.informViewBoundsChanged()
|
||||||
|
|
||||||
## workaround for pyqt bug:
|
## workaround for pyqt bug:
|
||||||
|
@ -1320,7 +1320,6 @@ class Handle(UIGraphicsItem):
|
|||||||
## determine rotation of transform
|
## determine rotation of transform
|
||||||
#m = self.sceneTransform() ## Qt bug: do not access sceneTransform() until we know this object has a scene.
|
#m = self.sceneTransform() ## Qt bug: do not access sceneTransform() until we know this object has a scene.
|
||||||
#mi = m.inverted()[0]
|
#mi = m.inverted()[0]
|
||||||
|
|
||||||
dt = self.deviceTransform()
|
dt = self.deviceTransform()
|
||||||
|
|
||||||
if dt is None:
|
if dt is None:
|
||||||
@ -1339,10 +1338,10 @@ class Handle(UIGraphicsItem):
|
|||||||
return dti.map(tr.map(self.path))
|
return dti.map(tr.map(self.path))
|
||||||
|
|
||||||
|
|
||||||
def viewRangeChanged(self):
|
def viewTransformChanged(self):
|
||||||
GraphicsObject.viewRangeChanged(self)
|
GraphicsObject.viewTransformChanged(self)
|
||||||
self._shape = None ## invalidate shape, recompute later if requested.
|
self._shape = None ## invalidate shape, recompute later if requested.
|
||||||
#self.updateShape()
|
self.update()
|
||||||
|
|
||||||
#def itemChange(self, change, value):
|
#def itemChange(self, change, value):
|
||||||
#if change == self.ItemScenePositionHasChanged:
|
#if change == self.ItemScenePositionHasChanged:
|
||||||
|
@ -17,6 +17,10 @@ __all__ = ['ViewBox']
|
|||||||
class ChildGroup(ItemGroup):
|
class ChildGroup(ItemGroup):
|
||||||
|
|
||||||
sigItemsChanged = QtCore.Signal()
|
sigItemsChanged = QtCore.Signal()
|
||||||
|
def __init__(self, parent):
|
||||||
|
ItemGroup.__init__(self, parent)
|
||||||
|
# excempt from telling view when transform changes
|
||||||
|
self._GraphicsObject__inform_view_on_change = False
|
||||||
|
|
||||||
def itemChange(self, change, value):
|
def itemChange(self, change, value):
|
||||||
ret = ItemGroup.itemChange(self, change, value)
|
ret = ItemGroup.itemChange(self, change, value)
|
||||||
@ -87,7 +91,8 @@ class ViewBox(GraphicsWidget):
|
|||||||
self.addedItems = []
|
self.addedItems = []
|
||||||
#self.gView = view
|
#self.gView = view
|
||||||
#self.showGrid = showGrid
|
#self.showGrid = showGrid
|
||||||
self.matrixNeedsUpdate = True ## indicates that range has changed, but matrix update was deferred
|
self._matrixNeedsUpdate = True ## indicates that range has changed, but matrix update was deferred
|
||||||
|
self._autoRangeNeedsUpdate = True ## indicates auto-range needs to be recomputed.
|
||||||
|
|
||||||
self.state = {
|
self.state = {
|
||||||
|
|
||||||
@ -183,7 +188,7 @@ class ViewBox(GraphicsWidget):
|
|||||||
|
|
||||||
def unregister(self):
|
def unregister(self):
|
||||||
"""
|
"""
|
||||||
Remove this ViewBox forom the list of linkable views. (see :func:`register() <pyqtgraph.ViewBox.register>`)
|
Remove this ViewBox from the list of linkable views. (see :func:`register() <pyqtgraph.ViewBox.register>`)
|
||||||
"""
|
"""
|
||||||
del ViewBox.AllViews[self]
|
del ViewBox.AllViews[self]
|
||||||
if self.name is not None:
|
if self.name is not None:
|
||||||
@ -195,6 +200,25 @@ class ViewBox(GraphicsWidget):
|
|||||||
def implements(self, interface):
|
def implements(self, interface):
|
||||||
return interface == 'ViewBox'
|
return interface == 'ViewBox'
|
||||||
|
|
||||||
|
def itemChange(self, change, value):
|
||||||
|
ret = GraphicsWidget.itemChange(self, change, value)
|
||||||
|
if change == self.ItemSceneChange:
|
||||||
|
scene = self.scene()
|
||||||
|
if scene is not None:
|
||||||
|
scene.sigPrepareForPaint.disconnect(self.prepareForPaint)
|
||||||
|
elif change == self.ItemSceneHasChanged:
|
||||||
|
scene = self.scene()
|
||||||
|
if scene is not None:
|
||||||
|
scene.sigPrepareForPaint.connect(self.prepareForPaint)
|
||||||
|
return ret
|
||||||
|
|
||||||
|
def prepareForPaint(self):
|
||||||
|
#autoRangeEnabled = (self.state['autoRange'][0] is not False) or (self.state['autoRange'][1] is not False)
|
||||||
|
# don't check whether auto range is enabled here--only check when setting dirty flag.
|
||||||
|
if self._autoRangeNeedsUpdate: # and autoRangeEnabled:
|
||||||
|
self.updateAutoRange()
|
||||||
|
if self._matrixNeedsUpdate:
|
||||||
|
self.updateMatrix()
|
||||||
|
|
||||||
def getState(self, copy=True):
|
def getState(self, copy=True):
|
||||||
"""Return the current state of the ViewBox.
|
"""Return the current state of the ViewBox.
|
||||||
@ -223,7 +247,8 @@ class ViewBox(GraphicsWidget):
|
|||||||
del state['linkedViews']
|
del state['linkedViews']
|
||||||
|
|
||||||
self.state.update(state)
|
self.state.update(state)
|
||||||
self.updateMatrix()
|
#self.updateMatrix()
|
||||||
|
self.updateViewRange()
|
||||||
self.sigStateChanged.emit(self)
|
self.sigStateChanged.emit(self)
|
||||||
|
|
||||||
|
|
||||||
@ -308,16 +333,15 @@ class ViewBox(GraphicsWidget):
|
|||||||
ch.setParentItem(None)
|
ch.setParentItem(None)
|
||||||
|
|
||||||
def resizeEvent(self, ev):
|
def resizeEvent(self, ev):
|
||||||
#self.setRange(self.range, padding=0)
|
self.linkedXChanged()
|
||||||
|
self.linkedYChanged()
|
||||||
self.updateAutoRange()
|
self.updateAutoRange()
|
||||||
self.updateMatrix()
|
self.updateViewRange()
|
||||||
self.sigStateChanged.emit(self)
|
self.sigStateChanged.emit(self)
|
||||||
self.background.setRect(self.rect())
|
self.background.setRect(self.rect())
|
||||||
#self._itemBoundsCache.clear()
|
|
||||||
#self.linkedXChanged()
|
|
||||||
#self.linkedYChanged()
|
|
||||||
self.sigResized.emit(self)
|
self.sigResized.emit(self)
|
||||||
|
|
||||||
|
|
||||||
def viewRange(self):
|
def viewRange(self):
|
||||||
"""Return a the view's visible range as a list: [[xmin, xmax], [ymin, ymax]]"""
|
"""Return a the view's visible range as a list: [[xmin, xmax], [ymin, ymax]]"""
|
||||||
return [x[:] for x in self.state['viewRange']] ## return copy
|
return [x[:] for x in self.state['viewRange']] ## return copy
|
||||||
@ -352,9 +376,9 @@ class ViewBox(GraphicsWidget):
|
|||||||
def setRange(self, rect=None, xRange=None, yRange=None, padding=None, update=True, disableAutoRange=True):
|
def setRange(self, rect=None, xRange=None, yRange=None, padding=None, update=True, disableAutoRange=True):
|
||||||
"""
|
"""
|
||||||
Set the visible range of the ViewBox.
|
Set the visible range of the ViewBox.
|
||||||
Must specify at least one of *range*, *xRange*, or *yRange*.
|
Must specify at least one of *rect*, *xRange*, or *yRange*.
|
||||||
|
|
||||||
============= =====================================================================
|
================== =====================================================================
|
||||||
**Arguments**
|
**Arguments**
|
||||||
*rect* (QRectF) The full range that should be visible in the view box.
|
*rect* (QRectF) The full range that should be visible in the view box.
|
||||||
*xRange* (min,max) The range that should be visible along the x-axis.
|
*xRange* (min,max) The range that should be visible along the x-axis.
|
||||||
@ -362,68 +386,76 @@ class ViewBox(GraphicsWidget):
|
|||||||
*padding* (float) Expand the view by a fraction of the requested range.
|
*padding* (float) Expand the view by a fraction of the requested range.
|
||||||
By default, this value is set between 0.02 and 0.1 depending on
|
By default, this value is set between 0.02 and 0.1 depending on
|
||||||
the size of the ViewBox.
|
the size of the ViewBox.
|
||||||
============= =====================================================================
|
*update* (bool) If True, update the range of the ViewBox immediately.
|
||||||
|
Otherwise, the update is deferred until before the next render.
|
||||||
|
*disableAutoRange* (bool) If True, auto-ranging is diabled. Otherwise, it is left
|
||||||
|
unchanged.
|
||||||
|
================== =====================================================================
|
||||||
|
|
||||||
"""
|
"""
|
||||||
|
#print self.name, "ViewBox.setRange", rect, xRange, yRange, padding
|
||||||
|
|
||||||
changes = {}
|
changes = {} # axes
|
||||||
|
setRequested = [False, False]
|
||||||
|
|
||||||
if rect is not None:
|
if rect is not None:
|
||||||
changes = {0: [rect.left(), rect.right()], 1: [rect.top(), rect.bottom()]}
|
changes = {0: [rect.left(), rect.right()], 1: [rect.top(), rect.bottom()]}
|
||||||
|
setRequested = [True, True]
|
||||||
if xRange is not None:
|
if xRange is not None:
|
||||||
changes[0] = xRange
|
changes[0] = xRange
|
||||||
|
setRequested[0] = True
|
||||||
if yRange is not None:
|
if yRange is not None:
|
||||||
changes[1] = yRange
|
changes[1] = yRange
|
||||||
|
setRequested[1] = True
|
||||||
|
|
||||||
if len(changes) == 0:
|
if len(changes) == 0:
|
||||||
print(rect)
|
print(rect)
|
||||||
raise Exception("Must specify at least one of rect, xRange, or yRange. (gave rect=%s)" % str(type(rect)))
|
raise Exception("Must specify at least one of rect, xRange, or yRange. (gave rect=%s)" % str(type(rect)))
|
||||||
|
|
||||||
|
# Update axes one at a time
|
||||||
changed = [False, False]
|
changed = [False, False]
|
||||||
for ax, range in changes.items():
|
for ax, range in changes.items():
|
||||||
if padding is None:
|
|
||||||
xpad = self.suggestPadding(ax)
|
|
||||||
else:
|
|
||||||
xpad = padding
|
|
||||||
mn = min(range)
|
mn = min(range)
|
||||||
mx = max(range)
|
mx = max(range)
|
||||||
if mn == mx: ## If we requested 0 range, try to preserve previous scale. Otherwise just pick an arbitrary scale.
|
|
||||||
|
# If we requested 0 range, try to preserve previous scale.
|
||||||
|
# Otherwise just pick an arbitrary scale.
|
||||||
|
if mn == mx:
|
||||||
dy = self.state['viewRange'][ax][1] - self.state['viewRange'][ax][0]
|
dy = self.state['viewRange'][ax][1] - self.state['viewRange'][ax][0]
|
||||||
if dy == 0:
|
if dy == 0:
|
||||||
dy = 1
|
dy = 1
|
||||||
mn -= dy*0.5
|
mn -= dy*0.5
|
||||||
mx += dy*0.5
|
mx += dy*0.5
|
||||||
xpad = 0.0
|
xpad = 0.0
|
||||||
if any(np.isnan([mn, mx])) or any(np.isinf([mn, mx])):
|
|
||||||
raise Exception("Not setting range [%s, %s]" % (str(mn), str(mx)))
|
|
||||||
|
|
||||||
|
# Make sure no nan/inf get through
|
||||||
|
if not all(np.isfinite([mn, mx])):
|
||||||
|
raise Exception("Cannot set range [%s, %s]" % (str(mn), str(mx)))
|
||||||
|
|
||||||
|
# Apply padding
|
||||||
|
if padding is None:
|
||||||
|
xpad = self.suggestPadding(ax)
|
||||||
|
else:
|
||||||
|
xpad = padding
|
||||||
p = (mx-mn) * xpad
|
p = (mx-mn) * xpad
|
||||||
mn -= p
|
mn -= p
|
||||||
mx += p
|
mx += p
|
||||||
|
|
||||||
|
# Set target range
|
||||||
if self.state['targetRange'][ax] != [mn, mx]:
|
if self.state['targetRange'][ax] != [mn, mx]:
|
||||||
self.state['targetRange'][ax] = [mn, mx]
|
self.state['targetRange'][ax] = [mn, mx]
|
||||||
changed[ax] = True
|
changed[ax] = True
|
||||||
|
|
||||||
aspect = self.state['aspectLocked'] # size ratio / view ratio
|
# Update viewRange to match targetRange as closely as possible while
|
||||||
if aspect is not False and len(changes) == 1:
|
# accounting for aspect ratio constraint
|
||||||
## need to adjust orthogonal target range to match
|
lockX, lockY = setRequested
|
||||||
size = [self.width(), self.height()]
|
if lockX and lockY:
|
||||||
tr1 = self.state['targetRange'][ax]
|
lockX = False
|
||||||
tr2 = self.state['targetRange'][1-ax]
|
lockY = False
|
||||||
if size[1] == 0 or aspect == 0:
|
self.updateViewRange(lockX, lockY)
|
||||||
ratio = 1.0
|
|
||||||
else:
|
|
||||||
ratio = (size[0] / float(size[1])) / aspect
|
|
||||||
if ax == 0:
|
|
||||||
ratio = 1.0 / ratio
|
|
||||||
w = (tr1[1]-tr1[0]) * ratio
|
|
||||||
d = 0.5 * (w - (tr2[1]-tr2[0]))
|
|
||||||
self.state['targetRange'][1-ax] = [tr2[0]-d, tr2[1]+d]
|
|
||||||
|
|
||||||
|
# Disable auto-range if needed
|
||||||
|
if disableAutoRange:
|
||||||
|
|
||||||
if any(changed) and disableAutoRange:
|
|
||||||
if all(changed):
|
if all(changed):
|
||||||
ax = ViewBox.XYAxes
|
ax = ViewBox.XYAxes
|
||||||
elif changed[0]:
|
elif changed[0]:
|
||||||
@ -431,27 +463,43 @@ class ViewBox(GraphicsWidget):
|
|||||||
elif changed[1]:
|
elif changed[1]:
|
||||||
ax = ViewBox.YAxis
|
ax = ViewBox.YAxis
|
||||||
self.enableAutoRange(ax, False)
|
self.enableAutoRange(ax, False)
|
||||||
|
changed.append(True)
|
||||||
|
|
||||||
|
# If nothing has changed, we are done.
|
||||||
|
if any(changed):
|
||||||
|
#if update and self.matrixNeedsUpdate:
|
||||||
|
#self.updateMatrix(changed)
|
||||||
|
#return
|
||||||
|
|
||||||
self.sigStateChanged.emit(self)
|
self.sigStateChanged.emit(self)
|
||||||
|
|
||||||
|
# Update target rect for debugging
|
||||||
|
if self.target.isVisible():
|
||||||
self.target.setRect(self.mapRectFromItem(self.childGroup, self.targetRect()))
|
self.target.setRect(self.mapRectFromItem(self.childGroup, self.targetRect()))
|
||||||
|
|
||||||
if update and (any(changed) or self.matrixNeedsUpdate):
|
# If ortho axes have auto-visible-only, update them now
|
||||||
self.updateMatrix(changed)
|
# Note that aspect ratio constraints and auto-visible probably do not work together..
|
||||||
|
if changed[0] and self.state['autoVisibleOnly'][1] and (self.state['autoRange'][0] is not False):
|
||||||
|
self._autoRangeNeedsUpdate = True
|
||||||
|
#self.updateAutoRange() ## Maybe just indicate that auto range needs to be updated?
|
||||||
|
elif changed[1] and self.state['autoVisibleOnly'][0] and (self.state['autoRange'][1] is not False):
|
||||||
|
self._autoRangeNeedsUpdate = True
|
||||||
|
#self.updateAutoRange()
|
||||||
|
|
||||||
if not update and any(changed):
|
## Update view matrix only if requested
|
||||||
self.matrixNeedsUpdate = True
|
#if update:
|
||||||
|
#self.updateMatrix(changed)
|
||||||
|
## Otherwise, indicate that the matrix needs to be updated
|
||||||
|
#else:
|
||||||
|
#self.matrixNeedsUpdate = True
|
||||||
|
|
||||||
|
## Inform linked views that the range has changed <<This should be moved>>
|
||||||
|
#for ax, range in changes.items():
|
||||||
|
#link = self.linkedView(ax)
|
||||||
|
#if link is not None:
|
||||||
|
#link.linkedViewChanged(self, ax)
|
||||||
|
|
||||||
for ax, range in changes.items():
|
|
||||||
link = self.linkedView(ax)
|
|
||||||
if link is not None:
|
|
||||||
link.linkedViewChanged(self, ax)
|
|
||||||
|
|
||||||
if changed[0] and self.state['autoVisibleOnly'][1]:
|
|
||||||
self.updateAutoRange()
|
|
||||||
elif changed[1] and self.state['autoVisibleOnly'][0]:
|
|
||||||
self.updateAutoRange()
|
|
||||||
|
|
||||||
def setYRange(self, min, max, padding=None, update=True):
|
def setYRange(self, min, max, padding=None, update=True):
|
||||||
"""
|
"""
|
||||||
@ -560,15 +608,14 @@ class ViewBox(GraphicsWidget):
|
|||||||
self.setRange(vr.translated(t), padding=0)
|
self.setRange(vr.translated(t), padding=0)
|
||||||
else:
|
else:
|
||||||
if x is not None:
|
if x is not None:
|
||||||
x1, x2 = vr.left()+x, vr.right()+x
|
x = vr.left()+x, vr.right()+x
|
||||||
self.setXRange(x1, x2, padding=0)
|
|
||||||
if y is not None:
|
if y is not None:
|
||||||
y1, y2 = vr.top()+y, vr.bottom()+y
|
y = vr.top()+y, vr.bottom()+y
|
||||||
self.setYRange(y1, y2, padding=0)
|
self.setRange(xRange=x, yRange=y, padding=0)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
def enableAutoRange(self, axis=None, enable=True):
|
def enableAutoRange(self, axis=None, enable=True, x=None, y=None):
|
||||||
"""
|
"""
|
||||||
Enable (or disable) auto-range for *axis*, which may be ViewBox.XAxis, ViewBox.YAxis, or ViewBox.XYAxes for both
|
Enable (or disable) auto-range for *axis*, which may be ViewBox.XAxis, ViewBox.YAxis, or ViewBox.XYAxes for both
|
||||||
(if *axis* is omitted, both axes will be changed).
|
(if *axis* is omitted, both axes will be changed).
|
||||||
@ -581,24 +628,46 @@ class ViewBox(GraphicsWidget):
|
|||||||
#import traceback
|
#import traceback
|
||||||
#traceback.print_stack()
|
#traceback.print_stack()
|
||||||
|
|
||||||
|
# support simpler interface:
|
||||||
|
if x is not None or y is not None:
|
||||||
|
if x is not None:
|
||||||
|
self.enableAutoRange(ViewBox.XAxis, x)
|
||||||
|
if y is not None:
|
||||||
|
self.enableAutoRange(ViewBox.YAxis, y)
|
||||||
|
return
|
||||||
|
|
||||||
if enable is True:
|
if enable is True:
|
||||||
enable = 1.0
|
enable = 1.0
|
||||||
|
|
||||||
if axis is None:
|
if axis is None:
|
||||||
axis = ViewBox.XYAxes
|
axis = ViewBox.XYAxes
|
||||||
|
|
||||||
|
needAutoRangeUpdate = False
|
||||||
|
|
||||||
if axis == ViewBox.XYAxes or axis == 'xy':
|
if axis == ViewBox.XYAxes or axis == 'xy':
|
||||||
self.state['autoRange'][0] = enable
|
axes = [0, 1]
|
||||||
self.state['autoRange'][1] = enable
|
|
||||||
elif axis == ViewBox.XAxis or axis == 'x':
|
elif axis == ViewBox.XAxis or axis == 'x':
|
||||||
self.state['autoRange'][0] = enable
|
axes = [0]
|
||||||
elif axis == ViewBox.YAxis or axis == 'y':
|
elif axis == ViewBox.YAxis or axis == 'y':
|
||||||
self.state['autoRange'][1] = enable
|
axes = [1]
|
||||||
else:
|
else:
|
||||||
raise Exception('axis argument must be ViewBox.XAxis, ViewBox.YAxis, or ViewBox.XYAxes.')
|
raise Exception('axis argument must be ViewBox.XAxis, ViewBox.YAxis, or ViewBox.XYAxes.')
|
||||||
|
|
||||||
if enable:
|
for ax in axes:
|
||||||
|
if self.state['autoRange'][ax] != enable:
|
||||||
|
# If we are disabling, do one last auto-range to make sure that
|
||||||
|
# previously scheduled auto-range changes are enacted
|
||||||
|
if enable is False and self._autoRangeNeedsUpdate:
|
||||||
self.updateAutoRange()
|
self.updateAutoRange()
|
||||||
|
|
||||||
|
self.state['autoRange'][ax] = enable
|
||||||
|
self._autoRangeNeedsUpdate |= (enable is not False)
|
||||||
|
self.update()
|
||||||
|
|
||||||
|
|
||||||
|
#if needAutoRangeUpdate:
|
||||||
|
# self.updateAutoRange()
|
||||||
|
|
||||||
self.sigStateChanged.emit(self)
|
self.sigStateChanged.emit(self)
|
||||||
|
|
||||||
def disableAutoRange(self, axis=None):
|
def disableAutoRange(self, axis=None):
|
||||||
@ -686,6 +755,7 @@ class ViewBox(GraphicsWidget):
|
|||||||
args['disableAutoRange'] = False
|
args['disableAutoRange'] = False
|
||||||
self.setRange(**args)
|
self.setRange(**args)
|
||||||
finally:
|
finally:
|
||||||
|
self._autoRangeNeedsUpdate = False
|
||||||
self._updatingRange = False
|
self._updatingRange = False
|
||||||
|
|
||||||
def setXLink(self, view):
|
def setXLink(self, view):
|
||||||
@ -724,6 +794,7 @@ class ViewBox(GraphicsWidget):
|
|||||||
if oldLink is not None:
|
if oldLink is not None:
|
||||||
try:
|
try:
|
||||||
getattr(oldLink, signal).disconnect(slot)
|
getattr(oldLink, signal).disconnect(slot)
|
||||||
|
oldLink.sigResized.disconnect(slot)
|
||||||
except TypeError:
|
except TypeError:
|
||||||
## This can occur if the view has been deleted already
|
## This can occur if the view has been deleted already
|
||||||
pass
|
pass
|
||||||
@ -734,6 +805,7 @@ class ViewBox(GraphicsWidget):
|
|||||||
else:
|
else:
|
||||||
self.state['linkedViews'][axis] = weakref.ref(view)
|
self.state['linkedViews'][axis] = weakref.ref(view)
|
||||||
getattr(view, signal).connect(slot)
|
getattr(view, signal).connect(slot)
|
||||||
|
view.sigResized.connect(slot)
|
||||||
if view.autoRangeEnabled()[axis] is not False:
|
if view.autoRangeEnabled()[axis] is not False:
|
||||||
self.enableAutoRange(axis, False)
|
self.enableAutoRange(axis, False)
|
||||||
slot()
|
slot()
|
||||||
@ -770,6 +842,7 @@ class ViewBox(GraphicsWidget):
|
|||||||
if self.linksBlocked or view is None:
|
if self.linksBlocked or view is None:
|
||||||
return
|
return
|
||||||
|
|
||||||
|
#print self.name, "ViewBox.linkedViewChanged", axis, view.viewRange()[axis]
|
||||||
vr = view.viewRect()
|
vr = view.viewRect()
|
||||||
vg = view.screenGeometry()
|
vg = view.screenGeometry()
|
||||||
sg = self.screenGeometry()
|
sg = self.screenGeometry()
|
||||||
@ -828,14 +901,18 @@ class ViewBox(GraphicsWidget):
|
|||||||
|
|
||||||
def itemBoundsChanged(self, item):
|
def itemBoundsChanged(self, item):
|
||||||
self._itemBoundsCache.pop(item, None)
|
self._itemBoundsCache.pop(item, None)
|
||||||
self.updateAutoRange()
|
if (self.state['autoRange'][0] is not False) or (self.state['autoRange'][1] is not False):
|
||||||
|
self._autoRangeNeedsUpdate = True
|
||||||
|
self.update()
|
||||||
|
#self.updateAutoRange()
|
||||||
|
|
||||||
def invertY(self, b=True):
|
def invertY(self, b=True):
|
||||||
"""
|
"""
|
||||||
By default, the positive y-axis points upward on the screen. Use invertY(True) to reverse the y-axis.
|
By default, the positive y-axis points upward on the screen. Use invertY(True) to reverse the y-axis.
|
||||||
"""
|
"""
|
||||||
self.state['yInverted'] = b
|
self.state['yInverted'] = b
|
||||||
self.updateMatrix(changed=(False, True))
|
#self.updateMatrix(changed=(False, True))
|
||||||
|
self.updateViewRange()
|
||||||
self.sigStateChanged.emit(self)
|
self.sigStateChanged.emit(self)
|
||||||
|
|
||||||
def yInverted(self):
|
def yInverted(self):
|
||||||
@ -861,7 +938,7 @@ class ViewBox(GraphicsWidget):
|
|||||||
self.state['aspectLocked'] = ratio
|
self.state['aspectLocked'] = ratio
|
||||||
if ratio != currentRatio: ## If this would change the current range, do that now
|
if ratio != currentRatio: ## If this would change the current range, do that now
|
||||||
#self.setRange(0, self.state['viewRange'][0][0], self.state['viewRange'][0][1])
|
#self.setRange(0, self.state['viewRange'][0][0], self.state['viewRange'][0][1])
|
||||||
self.updateMatrix()
|
self.updateViewRange()
|
||||||
self.sigStateChanged.emit(self)
|
self.sigStateChanged.emit(self)
|
||||||
|
|
||||||
def childTransform(self):
|
def childTransform(self):
|
||||||
@ -1228,47 +1305,116 @@ class ViewBox(GraphicsWidget):
|
|||||||
bounds = QtCore.QRectF(range[0][0], range[1][0], range[0][1]-range[0][0], range[1][1]-range[1][0])
|
bounds = QtCore.QRectF(range[0][0], range[1][0], range[0][1]-range[0][0], range[1][1]-range[1][0])
|
||||||
return bounds
|
return bounds
|
||||||
|
|
||||||
|
def updateViewRange(self, forceX=False, forceY=False):
|
||||||
|
## Update viewRange to match targetRange as closely as possible, given
|
||||||
|
## aspect ratio constraints. The *force* arguments are used to indicate
|
||||||
|
## which axis (if any) should be unchanged when applying constraints.
|
||||||
|
|
||||||
|
viewRange = [self.state['targetRange'][0][:], self.state['targetRange'][1][:]]
|
||||||
def updateMatrix(self, changed=None):
|
|
||||||
## Make the childGroup's transform match the requested range.
|
|
||||||
|
|
||||||
if changed is None:
|
|
||||||
changed = [False, False]
|
changed = [False, False]
|
||||||
changed = list(changed)
|
|
||||||
|
# Make correction for aspect ratio constraint
|
||||||
|
|
||||||
|
## aspect is (widget w/h) / (view range w/h)
|
||||||
|
aspect = self.state['aspectLocked'] # size ratio / view ratio
|
||||||
tr = self.targetRect()
|
tr = self.targetRect()
|
||||||
bounds = self.rect()
|
bounds = self.rect()
|
||||||
|
if aspect is not False and tr.width() != 0 and bounds.width() != 0:
|
||||||
## set viewRect, given targetRect and possibly aspect ratio constraint
|
|
||||||
aspect = self.state['aspectLocked']
|
|
||||||
if aspect is False or bounds.height() == 0:
|
|
||||||
self.state['viewRange'] = [self.state['targetRange'][0][:], self.state['targetRange'][1][:]]
|
|
||||||
else:
|
|
||||||
## aspect is (widget w/h) / (view range w/h)
|
|
||||||
|
|
||||||
## This is the view range aspect ratio we have requested
|
## This is the view range aspect ratio we have requested
|
||||||
targetRatio = tr.width() / tr.height()
|
targetRatio = tr.width() / tr.height()
|
||||||
## This is the view range aspect ratio we need to obey aspect constraint
|
## This is the view range aspect ratio we need to obey aspect constraint
|
||||||
viewRatio = (bounds.width() / bounds.height()) / aspect
|
viewRatio = (bounds.width() / bounds.height()) / aspect
|
||||||
|
|
||||||
if targetRatio > viewRatio:
|
# Decide which range to keep unchanged
|
||||||
|
#print self.name, "aspect:", aspect, "changed:", changed, "auto:", self.state['autoRange']
|
||||||
|
if forceX:
|
||||||
|
ax = 0
|
||||||
|
elif forceY:
|
||||||
|
ax = 1
|
||||||
|
else:
|
||||||
|
# if we are not required to keep a particular axis unchanged,
|
||||||
|
# then make the entire target range visible
|
||||||
|
ax = 0 if targetRatio > viewRatio else 1
|
||||||
|
|
||||||
|
#### these should affect viewRange, not targetRange!
|
||||||
|
if ax == 0:
|
||||||
## view range needs to be taller than target
|
## view range needs to be taller than target
|
||||||
dy = 0.5 * (tr.width() / viewRatio - tr.height())
|
dy = 0.5 * (tr.width() / viewRatio - tr.height())
|
||||||
if dy != 0:
|
if dy != 0:
|
||||||
changed[1] = True
|
changed[1] = True
|
||||||
self.state['viewRange'] = [
|
viewRange[1] = [self.state['targetRange'][1][0] - dy, self.state['targetRange'][1][1] + dy]
|
||||||
self.state['targetRange'][0][:],
|
|
||||||
[self.state['targetRange'][1][0] - dy, self.state['targetRange'][1][1] + dy]
|
|
||||||
]
|
|
||||||
else:
|
else:
|
||||||
## view range needs to be wider than target
|
## view range needs to be wider than target
|
||||||
dx = 0.5 * (tr.height() * viewRatio - tr.width())
|
dx = 0.5 * (tr.height() * viewRatio - tr.width())
|
||||||
if dx != 0:
|
if dx != 0:
|
||||||
changed[0] = True
|
changed[0] = True
|
||||||
self.state['viewRange'] = [
|
viewRange[0] = [self.state['targetRange'][0][0] - dx, self.state['targetRange'][0][1] + dx]
|
||||||
[self.state['targetRange'][0][0] - dx, self.state['targetRange'][0][1] + dx],
|
|
||||||
self.state['targetRange'][1][:]
|
changed = [(viewRange[i][0] != self.state['viewRange'][i][0]) and (viewRange[i][1] != self.state['viewRange'][i][1]) for i in (0,1)]
|
||||||
]
|
self.state['viewRange'] = viewRange
|
||||||
|
|
||||||
|
# emit range change signals
|
||||||
|
if changed[0]:
|
||||||
|
self.sigXRangeChanged.emit(self, tuple(self.state['viewRange'][0]))
|
||||||
|
if changed[1]:
|
||||||
|
self.sigYRangeChanged.emit(self, tuple(self.state['viewRange'][1]))
|
||||||
|
|
||||||
|
if any(changed):
|
||||||
|
self.sigRangeChanged.emit(self, self.state['viewRange'])
|
||||||
|
self.update()
|
||||||
|
|
||||||
|
# Inform linked views that the range has changed
|
||||||
|
for ax in [0, 1]:
|
||||||
|
if not changed[ax]:
|
||||||
|
continue
|
||||||
|
link = self.linkedView(ax)
|
||||||
|
if link is not None:
|
||||||
|
link.linkedViewChanged(self, ax)
|
||||||
|
|
||||||
|
self._matrixNeedsUpdate = True
|
||||||
|
|
||||||
|
def updateMatrix(self, changed=None):
|
||||||
|
## Make the childGroup's transform match the requested viewRange.
|
||||||
|
|
||||||
|
#if changed is None:
|
||||||
|
#changed = [False, False]
|
||||||
|
#changed = list(changed)
|
||||||
|
#tr = self.targetRect()
|
||||||
|
bounds = self.rect()
|
||||||
|
|
||||||
|
## set viewRect, given targetRect and possibly aspect ratio constraint
|
||||||
|
#self.state['viewRange'] = [self.state['targetRange'][0][:], self.state['targetRange'][1][:]]
|
||||||
|
|
||||||
|
#aspect = self.state['aspectLocked']
|
||||||
|
#if aspect is False or bounds.height() == 0:
|
||||||
|
#self.state['viewRange'] = [self.state['targetRange'][0][:], self.state['targetRange'][1][:]]
|
||||||
|
#else:
|
||||||
|
### aspect is (widget w/h) / (view range w/h)
|
||||||
|
|
||||||
|
### This is the view range aspect ratio we have requested
|
||||||
|
#targetRatio = tr.width() / tr.height()
|
||||||
|
### This is the view range aspect ratio we need to obey aspect constraint
|
||||||
|
#viewRatio = (bounds.width() / bounds.height()) / aspect
|
||||||
|
|
||||||
|
#if targetRatio > viewRatio:
|
||||||
|
### view range needs to be taller than target
|
||||||
|
#dy = 0.5 * (tr.width() / viewRatio - tr.height())
|
||||||
|
#if dy != 0:
|
||||||
|
#changed[1] = True
|
||||||
|
#self.state['viewRange'] = [
|
||||||
|
#self.state['targetRange'][0][:],
|
||||||
|
#[self.state['targetRange'][1][0] - dy, self.state['targetRange'][1][1] + dy]
|
||||||
|
#]
|
||||||
|
#else:
|
||||||
|
### view range needs to be wider than target
|
||||||
|
#dx = 0.5 * (tr.height() * viewRatio - tr.width())
|
||||||
|
#if dx != 0:
|
||||||
|
#changed[0] = True
|
||||||
|
#self.state['viewRange'] = [
|
||||||
|
#[self.state['targetRange'][0][0] - dx, self.state['targetRange'][0][1] + dx],
|
||||||
|
#self.state['targetRange'][1][:]
|
||||||
|
#]
|
||||||
|
|
||||||
vr = self.viewRect()
|
vr = self.viewRect()
|
||||||
if vr.height() == 0 or vr.width() == 0:
|
if vr.height() == 0 or vr.width() == 0:
|
||||||
@ -1289,15 +1435,16 @@ class ViewBox(GraphicsWidget):
|
|||||||
|
|
||||||
self.childGroup.setTransform(m)
|
self.childGroup.setTransform(m)
|
||||||
|
|
||||||
if changed[0]:
|
# moved to viewRangeChanged
|
||||||
self.sigXRangeChanged.emit(self, tuple(self.state['viewRange'][0]))
|
#if changed[0]:
|
||||||
if changed[1]:
|
#self.sigXRangeChanged.emit(self, tuple(self.state['viewRange'][0]))
|
||||||
self.sigYRangeChanged.emit(self, tuple(self.state['viewRange'][1]))
|
#if changed[1]:
|
||||||
if any(changed):
|
#self.sigYRangeChanged.emit(self, tuple(self.state['viewRange'][1]))
|
||||||
self.sigRangeChanged.emit(self, self.state['viewRange'])
|
#if any(changed):
|
||||||
|
#self.sigRangeChanged.emit(self, self.state['viewRange'])
|
||||||
|
|
||||||
self.sigTransformChanged.emit(self) ## segfaults here: 1
|
self.sigTransformChanged.emit(self) ## segfaults here: 1
|
||||||
self.matrixNeedsUpdate = False
|
self._matrixNeedsUpdate = False
|
||||||
|
|
||||||
def paint(self, p, opt, widget):
|
def paint(self, p, opt, widget):
|
||||||
if self.border is not None:
|
if self.border is not None:
|
||||||
|
95
pyqtgraph/graphicsItems/tests/ViewBox.py
Normal file
95
pyqtgraph/graphicsItems/tests/ViewBox.py
Normal file
@ -0,0 +1,95 @@
|
|||||||
|
"""
|
||||||
|
ViewBox test cases:
|
||||||
|
|
||||||
|
* call setRange then resize; requested range must be fully visible
|
||||||
|
* lockAspect works correctly for arbitrary aspect ratio
|
||||||
|
* autoRange works correctly with aspect locked
|
||||||
|
* call setRange with aspect locked, then resize
|
||||||
|
* AutoRange with all the bells and whistles
|
||||||
|
* item moves / changes transformation / changes bounds
|
||||||
|
* pan only
|
||||||
|
* fractional range
|
||||||
|
|
||||||
|
|
||||||
|
"""
|
||||||
|
|
||||||
|
import pyqtgraph as pg
|
||||||
|
app = pg.mkQApp()
|
||||||
|
|
||||||
|
imgData = pg.np.zeros((10, 10))
|
||||||
|
imgData[0] = 3
|
||||||
|
imgData[-1] = 3
|
||||||
|
imgData[:,0] = 3
|
||||||
|
imgData[:,-1] = 3
|
||||||
|
|
||||||
|
def testLinkWithAspectLock():
|
||||||
|
global win, vb
|
||||||
|
win = pg.GraphicsWindow()
|
||||||
|
vb = win.addViewBox(name="image view")
|
||||||
|
vb.setAspectLocked()
|
||||||
|
vb.enableAutoRange(x=False, y=False)
|
||||||
|
p1 = win.addPlot(name="plot 1")
|
||||||
|
p2 = win.addPlot(name="plot 2", row=1, col=0)
|
||||||
|
win.ci.layout.setRowFixedHeight(1, 150)
|
||||||
|
win.ci.layout.setColumnFixedWidth(1, 150)
|
||||||
|
|
||||||
|
def viewsMatch():
|
||||||
|
r0 = pg.np.array(vb.viewRange())
|
||||||
|
r1 = pg.np.array(p1.vb.viewRange()[1])
|
||||||
|
r2 = pg.np.array(p2.vb.viewRange()[1])
|
||||||
|
match = (abs(r0[1]-r1) <= (abs(r1) * 0.001)).all() and (abs(r0[0]-r2) <= (abs(r2) * 0.001)).all()
|
||||||
|
return match
|
||||||
|
|
||||||
|
p1.setYLink(vb)
|
||||||
|
p2.setXLink(vb)
|
||||||
|
print "link views match:", viewsMatch()
|
||||||
|
win.show()
|
||||||
|
print "show views match:", viewsMatch()
|
||||||
|
img = pg.ImageItem(imgData)
|
||||||
|
vb.addItem(img)
|
||||||
|
vb.autoRange()
|
||||||
|
p1.plot(x=imgData.sum(axis=0), y=range(10))
|
||||||
|
p2.plot(x=range(10), y=imgData.sum(axis=1))
|
||||||
|
print "add items views match:", viewsMatch()
|
||||||
|
#p1.setAspectLocked()
|
||||||
|
#grid = pg.GridItem()
|
||||||
|
#vb.addItem(grid)
|
||||||
|
pg.QtGui.QApplication.processEvents()
|
||||||
|
pg.QtGui.QApplication.processEvents()
|
||||||
|
#win.resize(801, 600)
|
||||||
|
|
||||||
|
def testAspectLock():
|
||||||
|
global win, vb
|
||||||
|
win = pg.GraphicsWindow()
|
||||||
|
vb = win.addViewBox(name="image view")
|
||||||
|
vb.setAspectLocked()
|
||||||
|
img = pg.ImageItem(imgData)
|
||||||
|
vb.addItem(img)
|
||||||
|
|
||||||
|
|
||||||
|
#app.processEvents()
|
||||||
|
#print "init views match:", viewsMatch()
|
||||||
|
#p2.setYRange(-300, 300)
|
||||||
|
#print "setRange views match:", viewsMatch()
|
||||||
|
#app.processEvents()
|
||||||
|
#print "setRange views match (after update):", viewsMatch()
|
||||||
|
|
||||||
|
#print "--lock aspect--"
|
||||||
|
#p1.setAspectLocked(True)
|
||||||
|
#print "lockAspect views match:", viewsMatch()
|
||||||
|
#p2.setYRange(-200, 200)
|
||||||
|
#print "setRange views match:", viewsMatch()
|
||||||
|
#app.processEvents()
|
||||||
|
#print "setRange views match (after update):", viewsMatch()
|
||||||
|
|
||||||
|
#win.resize(100, 600)
|
||||||
|
#app.processEvents()
|
||||||
|
#vb.setRange(xRange=[-10, 10], padding=0)
|
||||||
|
#app.processEvents()
|
||||||
|
#win.resize(600, 100)
|
||||||
|
#app.processEvents()
|
||||||
|
#print vb.viewRange()
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
testLinkWithAspectLock()
|
@ -20,10 +20,8 @@ if __name__ == '__main__':
|
|||||||
|
|
||||||
if opts.pop('pyside', False):
|
if opts.pop('pyside', False):
|
||||||
import PySide
|
import PySide
|
||||||
#import pyqtgraph
|
|
||||||
#import pyqtgraph.multiprocess.processes
|
|
||||||
targetStr = opts.pop('targetStr')
|
targetStr = opts.pop('targetStr')
|
||||||
target = pickle.loads(targetStr) ## unpickling the target should import everything we need
|
target = pickle.loads(targetStr) ## unpickling the target should import everything we need
|
||||||
#target(name, port, authkey, ppid)
|
|
||||||
target(**opts) ## Send all other options to the target function
|
target(**opts) ## Send all other options to the target function
|
||||||
sys.exit(0)
|
sys.exit(0)
|
||||||
|
@ -48,9 +48,10 @@ class Process(RemoteEventHandler):
|
|||||||
it must be picklable (bound methods are not).
|
it must be picklable (bound methods are not).
|
||||||
copySysPath If True, copy the contents of sys.path to the remote process
|
copySysPath If True, copy the contents of sys.path to the remote process
|
||||||
debug If True, print detailed information about communication
|
debug If True, print detailed information about communication
|
||||||
with the child process.
|
with the child process. Note that this option may cause
|
||||||
|
strange behavior on some systems due to a python bug:
|
||||||
|
http://bugs.python.org/issue3905
|
||||||
============ =============================================================
|
============ =============================================================
|
||||||
|
|
||||||
"""
|
"""
|
||||||
if target is None:
|
if target is None:
|
||||||
target = startEventLoop
|
target = startEventLoop
|
||||||
@ -82,7 +83,13 @@ class Process(RemoteEventHandler):
|
|||||||
|
|
||||||
## note: we need all three streams to have their own PIPE due to this bug:
|
## note: we need all three streams to have their own PIPE due to this bug:
|
||||||
## http://bugs.python.org/issue3905
|
## http://bugs.python.org/issue3905
|
||||||
self.proc = subprocess.Popen((executable, bootstrap), stdin=subprocess.PIPE, stdout=subprocess.PIPE, stderr=subprocess.PIPE)
|
if debug is True: # when debugging, we need to keep the usual stdout
|
||||||
|
stdout = sys.stdout
|
||||||
|
stderr = sys.stderr
|
||||||
|
else:
|
||||||
|
stdout = subprocess.PIPE
|
||||||
|
stderr = subprocess.PIPE
|
||||||
|
self.proc = subprocess.Popen((executable, bootstrap), stdin=subprocess.PIPE, stdout=stdout, stderr=stderr)
|
||||||
|
|
||||||
targetStr = pickle.dumps(target) ## double-pickle target so that child has a chance to
|
targetStr = pickle.dumps(target) ## double-pickle target so that child has a chance to
|
||||||
## set its sys.path properly before unpickling the target
|
## set its sys.path properly before unpickling the target
|
||||||
|
@ -147,6 +147,11 @@ class GraphicsView(QtGui.QGraphicsView):
|
|||||||
#print "GV: paint", ev.rect()
|
#print "GV: paint", ev.rect()
|
||||||
return QtGui.QGraphicsView.paintEvent(self, ev)
|
return QtGui.QGraphicsView.paintEvent(self, ev)
|
||||||
|
|
||||||
|
def render(self, *args, **kwds):
|
||||||
|
self.scene().prepareForPaint()
|
||||||
|
return QtGui.QGraphicsView.render(self, *args, **kwds)
|
||||||
|
|
||||||
|
|
||||||
def close(self):
|
def close(self):
|
||||||
self.centralWidget = None
|
self.centralWidget = None
|
||||||
self.scene().clear()
|
self.scene().clear()
|
||||||
|
@ -18,12 +18,15 @@ class RemoteGraphicsView(QtGui.QWidget):
|
|||||||
|
|
||||||
"""
|
"""
|
||||||
def __init__(self, parent=None, *args, **kwds):
|
def __init__(self, parent=None, *args, **kwds):
|
||||||
|
"""
|
||||||
|
The keyword arguments 'debug' and 'name', if specified, are passed to QtProcess.__init__().
|
||||||
|
"""
|
||||||
self._img = None
|
self._img = None
|
||||||
self._imgReq = None
|
self._imgReq = None
|
||||||
self._sizeHint = (640,480) ## no clue why this is needed, but it seems to be the default sizeHint for GraphicsView.
|
self._sizeHint = (640,480) ## no clue why this is needed, but it seems to be the default sizeHint for GraphicsView.
|
||||||
## without it, the widget will not compete for space against another GraphicsView.
|
## without it, the widget will not compete for space against another GraphicsView.
|
||||||
QtGui.QWidget.__init__(self)
|
QtGui.QWidget.__init__(self)
|
||||||
self._proc = mp.QtProcess(debug=kwds.pop('debug', False))
|
self._proc = mp.QtProcess(debug=kwds.pop('debug', False), name=kwds.pop('name', None))
|
||||||
self.pg = self._proc._import('pyqtgraph')
|
self.pg = self._proc._import('pyqtgraph')
|
||||||
self.pg.setConfigOptions(**self.pg.CONFIG_OPTIONS)
|
self.pg.setConfigOptions(**self.pg.CONFIG_OPTIONS)
|
||||||
rpgRemote = self._proc._import('pyqtgraph.widgets.RemoteGraphicsView')
|
rpgRemote = self._proc._import('pyqtgraph.widgets.RemoteGraphicsView')
|
||||||
@ -123,6 +126,7 @@ class Renderer(GraphicsView):
|
|||||||
|
|
||||||
def __init__(self, *args, **kwds):
|
def __init__(self, *args, **kwds):
|
||||||
## Create shared memory for rendered image
|
## Create shared memory for rendered image
|
||||||
|
#pg.dbg(namespace={'r': self})
|
||||||
if sys.platform.startswith('win'):
|
if sys.platform.startswith('win'):
|
||||||
self.shmtag = "pyqtgraph_shmem_" + ''.join([chr((random.getrandbits(20)%25) + 97) for i in range(20)])
|
self.shmtag = "pyqtgraph_shmem_" + ''.join([chr((random.getrandbits(20)%25) + 97) for i in range(20)])
|
||||||
self.shm = mmap.mmap(-1, mmap.PAGESIZE, self.shmtag) # use anonymous mmap on windows
|
self.shm = mmap.mmap(-1, mmap.PAGESIZE, self.shmtag) # use anonymous mmap on windows
|
||||||
@ -184,7 +188,11 @@ class Renderer(GraphicsView):
|
|||||||
self.img = QtGui.QImage(ch, self.width(), self.height(), QtGui.QImage.Format_ARGB32)
|
self.img = QtGui.QImage(ch, self.width(), self.height(), QtGui.QImage.Format_ARGB32)
|
||||||
else:
|
else:
|
||||||
address = ctypes.addressof(ctypes.c_char.from_buffer(self.shm, 0))
|
address = ctypes.addressof(ctypes.c_char.from_buffer(self.shm, 0))
|
||||||
|
try:
|
||||||
self.img = QtGui.QImage(sip.voidptr(address), self.width(), self.height(), QtGui.QImage.Format_ARGB32)
|
self.img = QtGui.QImage(sip.voidptr(address), self.width(), self.height(), QtGui.QImage.Format_ARGB32)
|
||||||
|
except TypeError:
|
||||||
|
# different versions of pyqt have different requirements here..
|
||||||
|
self.img = QtGui.QImage(memoryview(buffer(self.shm)), self.width(), self.height(), QtGui.QImage.Format_ARGB32)
|
||||||
self.img.fill(0xffffffff)
|
self.img.fill(0xffffffff)
|
||||||
p = QtGui.QPainter(self.img)
|
p = QtGui.QPainter(self.img)
|
||||||
self.render(p, self.viewRect(), self.rect())
|
self.render(p, self.viewRect(), self.rect())
|
||||||
|
Loading…
Reference in New Issue
Block a user