2012-03-01 21:55:32 -05:00
|
|
|
from pyqtgraph.Qt import QtGui, QtCore
|
2012-07-12 15:35:58 -04:00
|
|
|
from pyqtgraph.python2_3 import asUnicode
|
2012-03-01 21:55:32 -05:00
|
|
|
import numpy as np
|
|
|
|
from pyqtgraph.Point import Point
|
|
|
|
import pyqtgraph.debug as debug
|
|
|
|
import weakref
|
|
|
|
import pyqtgraph.functions as fn
|
2012-06-29 14:39:27 -04:00
|
|
|
import pyqtgraph as pg
|
2012-05-11 18:05:41 -04:00
|
|
|
from .GraphicsWidget import GraphicsWidget
|
2012-03-01 21:55:32 -05:00
|
|
|
|
|
|
|
__all__ = ['AxisItem']
|
|
|
|
class AxisItem(GraphicsWidget):
|
2012-04-28 16:00:42 -04:00
|
|
|
"""
|
|
|
|
GraphicsItem showing a single plot axis with ticks, values, and label.
|
|
|
|
Can be configured to fit on any side of a plot, and can automatically synchronize its displayed scale with ViewBox items.
|
|
|
|
Ticks can be extended to draw a grid.
|
|
|
|
If maxTickLength is negative, ticks point into the plot.
|
|
|
|
"""
|
|
|
|
|
2012-03-01 21:55:32 -05:00
|
|
|
def __init__(self, orientation, pen=None, linkView=None, parent=None, maxTickLength=-5, showValues=True):
|
|
|
|
"""
|
2012-04-28 16:00:42 -04:00
|
|
|
============== ===============================================================
|
|
|
|
**Arguments:**
|
|
|
|
orientation one of 'left', 'right', 'top', or 'bottom'
|
|
|
|
maxTickLength (px) maximum length of ticks to draw. Negative values draw
|
|
|
|
into the plot, positive values draw outward.
|
|
|
|
linkView (ViewBox) causes the range of values displayed in the axis
|
|
|
|
to be linked to the visible range of a ViewBox.
|
|
|
|
showValues (bool) Whether to display values adjacent to ticks
|
|
|
|
pen (QPen) Pen used when drawing ticks.
|
|
|
|
============== ===============================================================
|
2012-03-01 21:55:32 -05:00
|
|
|
"""
|
|
|
|
|
|
|
|
GraphicsWidget.__init__(self, parent)
|
|
|
|
self.label = QtGui.QGraphicsTextItem(self)
|
|
|
|
self.showValues = showValues
|
2012-03-12 10:04:59 -04:00
|
|
|
self.picture = None
|
2012-03-01 21:55:32 -05:00
|
|
|
self.orientation = orientation
|
|
|
|
if orientation not in ['left', 'right', 'top', 'bottom']:
|
|
|
|
raise Exception("Orientation argument must be one of 'left', 'right', 'top', or 'bottom'.")
|
|
|
|
if orientation in ['left', 'right']:
|
|
|
|
#self.setMinimumWidth(25)
|
|
|
|
#self.setSizePolicy(QtGui.QSizePolicy(
|
|
|
|
#QtGui.QSizePolicy.Minimum,
|
|
|
|
#QtGui.QSizePolicy.Expanding
|
|
|
|
#))
|
|
|
|
self.label.rotate(-90)
|
|
|
|
#else:
|
|
|
|
#self.setMinimumHeight(50)
|
|
|
|
#self.setSizePolicy(QtGui.QSizePolicy(
|
|
|
|
#QtGui.QSizePolicy.Expanding,
|
|
|
|
#QtGui.QSizePolicy.Minimum
|
|
|
|
#))
|
|
|
|
#self.drawLabel = False
|
|
|
|
|
|
|
|
self.labelText = ''
|
|
|
|
self.labelUnits = ''
|
|
|
|
self.labelUnitPrefix=''
|
|
|
|
self.labelStyle = {'color': '#CCC'}
|
2012-04-21 15:57:47 -04:00
|
|
|
self.logMode = False
|
2012-03-01 21:55:32 -05:00
|
|
|
|
|
|
|
self.textHeight = 18
|
|
|
|
self.tickLength = maxTickLength
|
2012-05-21 17:31:09 -04:00
|
|
|
self._tickLevels = None ## used to override the automatic ticking system with explicit ticks
|
2012-03-01 21:55:32 -05:00
|
|
|
self.scale = 1.0
|
|
|
|
self.autoScale = True
|
|
|
|
|
|
|
|
self.setRange(0, 1)
|
|
|
|
|
|
|
|
self.setPen(pen)
|
|
|
|
|
2012-03-12 10:04:59 -04:00
|
|
|
self._linkedView = None
|
2012-03-01 21:55:32 -05:00
|
|
|
if linkView is not None:
|
|
|
|
self.linkToView(linkView)
|
|
|
|
|
|
|
|
self.showLabel(False)
|
|
|
|
|
|
|
|
self.grid = False
|
|
|
|
#self.setCacheMode(self.DeviceCoordinateCache)
|
|
|
|
|
|
|
|
def close(self):
|
|
|
|
self.scene().removeItem(self.label)
|
|
|
|
self.label = None
|
|
|
|
self.scene().removeItem(self)
|
|
|
|
|
|
|
|
def setGrid(self, grid):
|
|
|
|
"""Set the alpha value for the grid, or False to disable."""
|
|
|
|
self.grid = grid
|
2012-03-12 10:04:59 -04:00
|
|
|
self.picture = None
|
|
|
|
self.prepareGeometryChange()
|
2012-03-01 21:55:32 -05:00
|
|
|
self.update()
|
|
|
|
|
2012-04-21 15:57:47 -04:00
|
|
|
def setLogMode(self, log):
|
2012-04-28 16:00:42 -04:00
|
|
|
"""
|
|
|
|
If *log* is True, then ticks are displayed on a logarithmic scale and values
|
|
|
|
are adjusted accordingly. (This is usually accessed by changing the log mode
|
|
|
|
of a :func:`PlotItem <pyqtgraph.PlotItem.setLogMode>`)
|
|
|
|
"""
|
2012-04-21 15:57:47 -04:00
|
|
|
self.logMode = log
|
|
|
|
self.picture = None
|
|
|
|
self.update()
|
2012-03-01 21:55:32 -05:00
|
|
|
|
|
|
|
def resizeEvent(self, ev=None):
|
|
|
|
#s = self.size()
|
|
|
|
|
|
|
|
## Set the position of the label
|
|
|
|
nudge = 5
|
|
|
|
br = self.label.boundingRect()
|
|
|
|
p = QtCore.QPointF(0, 0)
|
|
|
|
if self.orientation == 'left':
|
|
|
|
p.setY(int(self.size().height()/2 + br.width()/2))
|
|
|
|
p.setX(-nudge)
|
|
|
|
#s.setWidth(10)
|
|
|
|
elif self.orientation == 'right':
|
|
|
|
#s.setWidth(10)
|
|
|
|
p.setY(int(self.size().height()/2 + br.width()/2))
|
|
|
|
p.setX(int(self.size().width()-br.height()+nudge))
|
|
|
|
elif self.orientation == 'top':
|
|
|
|
#s.setHeight(10)
|
|
|
|
p.setY(-nudge)
|
|
|
|
p.setX(int(self.size().width()/2. - br.width()/2.))
|
|
|
|
elif self.orientation == 'bottom':
|
|
|
|
p.setX(int(self.size().width()/2. - br.width()/2.))
|
|
|
|
#s.setHeight(10)
|
|
|
|
p.setY(int(self.size().height()-br.height()+nudge))
|
|
|
|
#self.label.resize(s)
|
|
|
|
self.label.setPos(p)
|
2012-03-12 10:04:59 -04:00
|
|
|
self.picture = None
|
2012-03-01 21:55:32 -05:00
|
|
|
|
|
|
|
def showLabel(self, show=True):
|
2012-04-28 16:00:42 -04:00
|
|
|
"""Show/hide the label text for this axis."""
|
2012-03-01 21:55:32 -05:00
|
|
|
#self.drawLabel = show
|
|
|
|
self.label.setVisible(show)
|
|
|
|
if self.orientation in ['left', 'right']:
|
|
|
|
self.setWidth()
|
|
|
|
else:
|
|
|
|
self.setHeight()
|
|
|
|
if self.autoScale:
|
|
|
|
self.setScale()
|
|
|
|
|
|
|
|
def setLabel(self, text=None, units=None, unitPrefix=None, **args):
|
2012-04-28 16:00:42 -04:00
|
|
|
"""Set the text displayed adjacent to the axis."""
|
2012-03-01 21:55:32 -05:00
|
|
|
if text is not None:
|
|
|
|
self.labelText = text
|
|
|
|
self.showLabel()
|
|
|
|
if units is not None:
|
|
|
|
self.labelUnits = units
|
|
|
|
self.showLabel()
|
|
|
|
if unitPrefix is not None:
|
|
|
|
self.labelUnitPrefix = unitPrefix
|
|
|
|
if len(args) > 0:
|
|
|
|
self.labelStyle = args
|
|
|
|
self.label.setHtml(self.labelString())
|
|
|
|
self.resizeEvent()
|
2012-03-12 10:04:59 -04:00
|
|
|
self.picture = None
|
2012-03-01 21:55:32 -05:00
|
|
|
self.update()
|
|
|
|
|
|
|
|
def labelString(self):
|
|
|
|
if self.labelUnits == '':
|
|
|
|
if self.scale == 1.0:
|
|
|
|
units = ''
|
|
|
|
else:
|
2012-05-11 18:05:41 -04:00
|
|
|
units = asUnicode('(x%g)') % (1.0/self.scale)
|
2012-03-01 21:55:32 -05:00
|
|
|
else:
|
|
|
|
#print repr(self.labelUnitPrefix), repr(self.labelUnits)
|
2012-05-11 18:05:41 -04:00
|
|
|
units = asUnicode('(%s%s)') % (self.labelUnitPrefix, self.labelUnits)
|
2012-03-01 21:55:32 -05:00
|
|
|
|
2012-05-11 18:05:41 -04:00
|
|
|
s = asUnicode('%s %s') % (self.labelText, units)
|
2012-03-01 21:55:32 -05:00
|
|
|
|
|
|
|
style = ';'.join(['%s: "%s"' % (k, self.labelStyle[k]) for k in self.labelStyle])
|
|
|
|
|
2012-05-11 18:05:41 -04:00
|
|
|
return asUnicode("<span style='%s'>%s</span>") % (style, s)
|
2012-03-01 21:55:32 -05:00
|
|
|
|
|
|
|
def setHeight(self, h=None):
|
|
|
|
if h is None:
|
|
|
|
h = self.textHeight + max(0, self.tickLength)
|
|
|
|
if self.label.isVisible():
|
|
|
|
h += self.textHeight
|
|
|
|
self.setMaximumHeight(h)
|
|
|
|
self.setMinimumHeight(h)
|
2012-03-12 10:04:59 -04:00
|
|
|
self.picture = None
|
2012-03-01 21:55:32 -05:00
|
|
|
|
|
|
|
|
|
|
|
def setWidth(self, w=None):
|
|
|
|
if w is None:
|
|
|
|
w = max(0, self.tickLength) + 40
|
|
|
|
if self.label.isVisible():
|
|
|
|
w += self.textHeight
|
|
|
|
self.setMaximumWidth(w)
|
|
|
|
self.setMinimumWidth(w)
|
|
|
|
|
2012-06-29 14:39:27 -04:00
|
|
|
def pen(self):
|
|
|
|
if self._pen is None:
|
|
|
|
return fn.mkPen(pg.getConfigOption('foreground'))
|
|
|
|
return self._pen
|
|
|
|
|
2012-03-01 21:55:32 -05:00
|
|
|
def setPen(self, pen):
|
2012-06-29 14:39:27 -04:00
|
|
|
"""
|
|
|
|
Set the pen used for drawing text, axes, ticks, and grid lines.
|
|
|
|
if pen == None, the default will be used (see :func:`setConfigOption
|
|
|
|
<pyqtgraph.setConfigOption>`)
|
|
|
|
"""
|
|
|
|
self._pen = pen
|
2012-03-12 10:04:59 -04:00
|
|
|
self.picture = None
|
2012-03-01 21:55:32 -05:00
|
|
|
self.update()
|
|
|
|
|
|
|
|
def setScale(self, scale=None):
|
|
|
|
"""
|
2012-07-09 14:41:10 -04:00
|
|
|
Set the value scaling for this axis. Values on the axis are multiplied
|
|
|
|
by this scale factor before being displayed as text. By default,
|
|
|
|
this scaling value is automatically determined based on the visible range
|
|
|
|
and the axis units are updated to reflect the chosen scale factor.
|
2012-04-28 16:00:42 -04:00
|
|
|
|
|
|
|
For example: If the axis spans values from -0.1 to 0.1 and has units set
|
|
|
|
to 'V' then a scale of 1000 would cause the axis to display values -100 to 100
|
|
|
|
and the units would appear as 'mV'
|
2012-03-01 21:55:32 -05:00
|
|
|
"""
|
|
|
|
if scale is None:
|
|
|
|
#if self.drawLabel: ## If there is a label, then we are free to rescale the values
|
|
|
|
if self.label.isVisible():
|
2012-06-18 13:50:44 -04:00
|
|
|
#d = self.range[1] - self.range[0]
|
2012-03-12 12:23:25 -04:00
|
|
|
#(scale, prefix) = fn.siScale(d / 2.)
|
|
|
|
(scale, prefix) = fn.siScale(max(abs(self.range[0]), abs(self.range[1])))
|
2012-03-01 21:55:32 -05:00
|
|
|
if self.labelUnits == '' and prefix in ['k', 'm']: ## If we are not showing units, wait until 1e6 before scaling.
|
|
|
|
scale = 1.0
|
|
|
|
prefix = ''
|
|
|
|
self.setLabel(unitPrefix=prefix)
|
|
|
|
else:
|
|
|
|
scale = 1.0
|
2012-07-09 14:41:10 -04:00
|
|
|
else:
|
|
|
|
self.setLabel(unitPrefix='')
|
|
|
|
self.autoScale = False
|
|
|
|
|
2012-03-01 21:55:32 -05:00
|
|
|
if scale != self.scale:
|
|
|
|
self.scale = scale
|
|
|
|
self.setLabel()
|
2012-03-12 10:04:59 -04:00
|
|
|
self.picture = None
|
2012-03-01 21:55:32 -05:00
|
|
|
self.update()
|
|
|
|
|
|
|
|
def setRange(self, mn, mx):
|
2012-05-21 17:31:09 -04:00
|
|
|
"""Set the range of values displayed by the axis.
|
|
|
|
Usually this is handled automatically by linking the axis to a ViewBox with :func:`linkToView <pyqtgraph.AxisItem.linkToView>`"""
|
2012-06-18 13:50:44 -04:00
|
|
|
if any(np.isinf((mn, mx))) or any(np.isnan((mn, mx))):
|
2012-03-01 21:55:32 -05:00
|
|
|
raise Exception("Not setting range to [%s, %s]" % (str(mn), str(mx)))
|
|
|
|
self.range = [mn, mx]
|
|
|
|
if self.autoScale:
|
|
|
|
self.setScale()
|
2012-03-12 10:04:59 -04:00
|
|
|
self.picture = None
|
2012-03-01 21:55:32 -05:00
|
|
|
self.update()
|
|
|
|
|
2012-03-12 10:04:59 -04:00
|
|
|
def linkedView(self):
|
|
|
|
"""Return the ViewBox this axis is linked to"""
|
|
|
|
if self._linkedView is None:
|
|
|
|
return None
|
|
|
|
else:
|
|
|
|
return self._linkedView()
|
|
|
|
|
2012-03-01 21:55:32 -05:00
|
|
|
def linkToView(self, view):
|
2012-04-28 16:00:42 -04:00
|
|
|
"""Link this axis to a ViewBox, causing its displayed range to match the visible range of the view."""
|
2012-03-12 10:04:59 -04:00
|
|
|
oldView = self.linkedView()
|
|
|
|
self._linkedView = weakref.ref(view)
|
2012-03-01 21:55:32 -05:00
|
|
|
if self.orientation in ['right', 'left']:
|
2012-03-12 10:04:59 -04:00
|
|
|
if oldView is not None:
|
|
|
|
oldView.sigYRangeChanged.disconnect(self.linkedViewChanged)
|
2012-03-01 21:55:32 -05:00
|
|
|
view.sigYRangeChanged.connect(self.linkedViewChanged)
|
|
|
|
else:
|
2012-03-12 10:04:59 -04:00
|
|
|
if oldView is not None:
|
|
|
|
oldView.sigXRangeChanged.disconnect(self.linkedViewChanged)
|
2012-03-01 21:55:32 -05:00
|
|
|
view.sigXRangeChanged.connect(self.linkedViewChanged)
|
|
|
|
|
|
|
|
def linkedViewChanged(self, view, newRange):
|
2012-06-18 13:50:44 -04:00
|
|
|
if self.orientation in ['right', 'left'] and view.yInverted():
|
|
|
|
self.setRange(*newRange[::-1])
|
|
|
|
else:
|
|
|
|
self.setRange(*newRange)
|
2012-03-01 21:55:32 -05:00
|
|
|
|
|
|
|
def boundingRect(self):
|
2012-03-12 10:04:59 -04:00
|
|
|
linkedView = self.linkedView()
|
|
|
|
if linkedView is None or self.grid is False:
|
2012-03-01 21:55:32 -05:00
|
|
|
rect = self.mapRectFromParent(self.geometry())
|
|
|
|
## extend rect if ticks go in negative direction
|
|
|
|
if self.orientation == 'left':
|
|
|
|
rect.setRight(rect.right() - min(0,self.tickLength))
|
|
|
|
elif self.orientation == 'right':
|
|
|
|
rect.setLeft(rect.left() + min(0,self.tickLength))
|
|
|
|
elif self.orientation == 'top':
|
|
|
|
rect.setBottom(rect.bottom() - min(0,self.tickLength))
|
|
|
|
elif self.orientation == 'bottom':
|
|
|
|
rect.setTop(rect.top() + min(0,self.tickLength))
|
|
|
|
return rect
|
|
|
|
else:
|
2012-03-12 10:04:59 -04:00
|
|
|
return self.mapRectFromParent(self.geometry()) | linkedView.mapRectToItem(self, linkedView.boundingRect())
|
2012-03-01 21:55:32 -05:00
|
|
|
|
|
|
|
def paint(self, p, opt, widget):
|
2012-03-12 10:04:59 -04:00
|
|
|
if self.picture is None:
|
|
|
|
self.picture = QtGui.QPicture()
|
|
|
|
painter = QtGui.QPainter(self.picture)
|
|
|
|
try:
|
|
|
|
self.drawPicture(painter)
|
|
|
|
finally:
|
|
|
|
painter.end()
|
2012-05-21 17:31:09 -04:00
|
|
|
p.setRenderHint(p.Antialiasing, False)
|
|
|
|
p.setRenderHint(p.TextAntialiasing, True)
|
2012-03-12 10:04:59 -04:00
|
|
|
self.picture.play(p)
|
|
|
|
|
2012-04-04 09:31:58 -04:00
|
|
|
|
2012-05-21 17:31:09 -04:00
|
|
|
def setTicks(self, ticks):
|
|
|
|
"""Explicitly determine which ticks to display.
|
|
|
|
This overrides the behavior specified by tickSpacing(), tickValues(), and tickStrings()
|
|
|
|
The format for *ticks* looks like::
|
2012-04-04 09:31:58 -04:00
|
|
|
|
2012-05-21 17:31:09 -04:00
|
|
|
[
|
|
|
|
[ (majorTickValue1, majorTickString1), (majorTickValue2, majorTickString2), ... ],
|
|
|
|
[ (minorTickValue1, minorTickString1), (minorTickValue2, minorTickString2), ... ],
|
|
|
|
...
|
|
|
|
]
|
|
|
|
|
|
|
|
If *ticks* is None, then the default tick system will be used instead.
|
|
|
|
"""
|
|
|
|
self._tickLevels = ticks
|
|
|
|
self.picture = None
|
|
|
|
self.update()
|
|
|
|
|
2012-04-04 09:31:58 -04:00
|
|
|
def tickSpacing(self, minVal, maxVal, size):
|
|
|
|
"""Return values describing the desired spacing and offset of ticks.
|
|
|
|
|
|
|
|
This method is called whenever the axis needs to be redrawn and is a
|
|
|
|
good method to override in subclasses that require control over tick locations.
|
|
|
|
|
2012-04-28 16:00:42 -04:00
|
|
|
The return value must be a list of three tuples::
|
|
|
|
|
2012-04-04 09:31:58 -04:00
|
|
|
[
|
|
|
|
(major tick spacing, offset),
|
|
|
|
(minor tick spacing, offset),
|
|
|
|
(sub-minor tick spacing, offset),
|
|
|
|
...
|
|
|
|
]
|
|
|
|
"""
|
|
|
|
dif = abs(maxVal - minVal)
|
|
|
|
if dif == 0:
|
|
|
|
return []
|
|
|
|
|
|
|
|
## decide optimal minor tick spacing in pixels (this is just aesthetics)
|
|
|
|
pixelSpacing = np.log(size+10) * 5
|
|
|
|
optimalTickCount = size / pixelSpacing
|
|
|
|
if optimalTickCount < 1:
|
|
|
|
optimalTickCount = 1
|
|
|
|
|
|
|
|
## optimal minor tick spacing
|
|
|
|
optimalSpacing = dif / optimalTickCount
|
|
|
|
|
|
|
|
## the largest power-of-10 spacing which is smaller than optimal
|
|
|
|
p10unit = 10 ** np.floor(np.log10(optimalSpacing))
|
|
|
|
|
|
|
|
## Determine major/minor tick spacings which flank the optimal spacing.
|
|
|
|
intervals = np.array([1., 2., 10., 20., 100.]) * p10unit
|
|
|
|
minorIndex = 0
|
|
|
|
while intervals[minorIndex+1] <= optimalSpacing:
|
|
|
|
minorIndex += 1
|
|
|
|
|
|
|
|
return [
|
|
|
|
(intervals[minorIndex+2], 0),
|
|
|
|
(intervals[minorIndex+1], 0),
|
|
|
|
(intervals[minorIndex], 0)
|
|
|
|
]
|
|
|
|
|
2012-07-09 08:38:30 -04:00
|
|
|
##### This does not work -- switching between 2/5 confuses the automatic text-level-selection
|
|
|
|
### Determine major/minor tick spacings which flank the optimal spacing.
|
|
|
|
#intervals = np.array([1., 2., 5., 10., 20., 50., 100.]) * p10unit
|
|
|
|
#minorIndex = 0
|
|
|
|
#while intervals[minorIndex+1] <= optimalSpacing:
|
|
|
|
#minorIndex += 1
|
|
|
|
|
|
|
|
### make sure we never see 5 and 2 at the same time
|
|
|
|
#intIndexes = [
|
|
|
|
#[0,1,3],
|
|
|
|
#[0,2,3],
|
|
|
|
#[2,3,4],
|
|
|
|
#[3,4,6],
|
|
|
|
#[3,5,6],
|
|
|
|
#][minorIndex]
|
|
|
|
|
|
|
|
#return [
|
|
|
|
#(intervals[intIndexes[2]], 0),
|
|
|
|
#(intervals[intIndexes[1]], 0),
|
|
|
|
#(intervals[intIndexes[0]], 0)
|
|
|
|
#]
|
|
|
|
|
|
|
|
|
2012-04-04 09:31:58 -04:00
|
|
|
|
|
|
|
def tickValues(self, minVal, maxVal, size):
|
|
|
|
"""
|
2012-04-28 16:00:42 -04:00
|
|
|
Return the values and spacing of ticks to draw::
|
|
|
|
|
|
|
|
[
|
|
|
|
(spacing, [major ticks]),
|
|
|
|
(spacing, [minor ticks]),
|
|
|
|
...
|
|
|
|
]
|
2012-04-04 09:31:58 -04:00
|
|
|
|
|
|
|
By default, this method calls tickSpacing to determine the correct tick locations.
|
|
|
|
This is a good method to override in subclasses.
|
|
|
|
"""
|
2012-06-18 13:50:44 -04:00
|
|
|
minVal, maxVal = sorted((minVal, maxVal))
|
|
|
|
|
2012-04-21 15:57:47 -04:00
|
|
|
|
2012-04-04 09:31:58 -04:00
|
|
|
ticks = []
|
|
|
|
tickLevels = self.tickSpacing(minVal, maxVal, size)
|
2012-06-18 13:50:44 -04:00
|
|
|
allValues = np.array([])
|
2012-04-04 09:31:58 -04:00
|
|
|
for i in range(len(tickLevels)):
|
|
|
|
spacing, offset = tickLevels[i]
|
|
|
|
|
|
|
|
## determine starting tick
|
|
|
|
start = (np.ceil((minVal-offset) / spacing) * spacing) + offset
|
|
|
|
|
|
|
|
## determine number of ticks
|
|
|
|
num = int((maxVal-start) / spacing) + 1
|
2012-05-21 17:31:09 -04:00
|
|
|
values = np.arange(num) * spacing + start
|
2012-06-18 13:50:44 -04:00
|
|
|
## remove any ticks that were present in higher levels
|
|
|
|
## we assume here that if the difference between a tick value and a previously seen tick value
|
|
|
|
## is less than spacing/100, then they are 'equal' and we can ignore the new tick.
|
2012-07-09 08:38:30 -04:00
|
|
|
values = list(filter(lambda x: all(np.abs(allValues-x) > spacing*0.01), values) )
|
2012-06-18 13:50:44 -04:00
|
|
|
allValues = np.concatenate([allValues, values])
|
2012-05-21 17:31:09 -04:00
|
|
|
ticks.append((spacing, values))
|
2012-06-29 14:39:27 -04:00
|
|
|
|
|
|
|
if self.logMode:
|
|
|
|
return self.logTickValues(minVal, maxVal, size, ticks)
|
|
|
|
|
2012-04-04 09:31:58 -04:00
|
|
|
return ticks
|
|
|
|
|
2012-06-29 14:39:27 -04:00
|
|
|
def logTickValues(self, minVal, maxVal, size, stdTicks):
|
|
|
|
|
|
|
|
## start with the tick spacing given by tickValues().
|
|
|
|
## Any level whose spacing is < 1 needs to be converted to log scale
|
|
|
|
|
|
|
|
ticks = []
|
|
|
|
for (spacing, t) in stdTicks:
|
|
|
|
if spacing >= 1.0:
|
|
|
|
ticks.append((spacing, t))
|
|
|
|
|
|
|
|
if len(ticks) < 3:
|
|
|
|
v1 = int(np.floor(minVal))
|
|
|
|
v2 = int(np.ceil(maxVal))
|
|
|
|
#major = list(range(v1+1, v2))
|
|
|
|
|
|
|
|
minor = []
|
|
|
|
for v in range(v1, v2):
|
|
|
|
minor.extend(v + np.log10(np.arange(1, 10)))
|
|
|
|
minor = [x for x in minor if x>minVal and x<maxVal]
|
|
|
|
ticks.append((None, minor))
|
|
|
|
return ticks
|
2012-04-04 09:31:58 -04:00
|
|
|
|
|
|
|
def tickStrings(self, values, scale, spacing):
|
|
|
|
"""Return the strings that should be placed next to ticks. This method is called
|
|
|
|
when redrawing the axis and is a good method to override in subclasses.
|
|
|
|
The method is called with a list of tick values, a scaling factor (see below), and the
|
|
|
|
spacing between ticks (this is required since, in some instances, there may be only
|
|
|
|
one tick and thus no other way to determine the tick spacing)
|
|
|
|
|
|
|
|
The scale argument is used when the axis label is displaying units which may have an SI scaling prefix.
|
|
|
|
When determining the text to display, use value*scale to correctly account for this prefix.
|
|
|
|
For example, if the axis label's units are set to 'V', then a tick value of 0.001 might
|
|
|
|
be accompanied by a scale value of 1000. This indicates that the label is displaying 'mV', and
|
|
|
|
thus the tick should display 0.001 * 1000 = 1.
|
|
|
|
"""
|
2012-04-21 15:57:47 -04:00
|
|
|
if self.logMode:
|
|
|
|
return self.logTickStrings(values, scale, spacing)
|
|
|
|
|
2012-04-04 09:31:58 -04:00
|
|
|
places = max(0, np.ceil(-np.log10(spacing*scale)))
|
|
|
|
strings = []
|
|
|
|
for v in values:
|
|
|
|
vs = v * scale
|
|
|
|
if abs(vs) < .001 or abs(vs) >= 10000:
|
|
|
|
vstr = "%g" % vs
|
|
|
|
else:
|
|
|
|
vstr = ("%%0.%df" % places) % vs
|
|
|
|
strings.append(vstr)
|
|
|
|
return strings
|
2012-03-12 10:04:59 -04:00
|
|
|
|
2012-04-21 15:57:47 -04:00
|
|
|
def logTickStrings(self, values, scale, spacing):
|
|
|
|
return ["%0.1g"%x for x in 10 ** np.array(values).astype(float)]
|
|
|
|
|
2012-03-12 10:04:59 -04:00
|
|
|
def drawPicture(self, p):
|
|
|
|
|
2012-03-23 02:41:10 -04:00
|
|
|
p.setRenderHint(p.Antialiasing, False)
|
|
|
|
p.setRenderHint(p.TextAntialiasing, True)
|
|
|
|
|
2012-03-01 21:55:32 -05:00
|
|
|
prof = debug.Profiler("AxisItem.paint", disabled=True)
|
|
|
|
|
|
|
|
#bounds = self.boundingRect()
|
|
|
|
bounds = self.mapRectFromParent(self.geometry())
|
|
|
|
|
2012-03-12 10:04:59 -04:00
|
|
|
linkedView = self.linkedView()
|
|
|
|
if linkedView is None or self.grid is False:
|
2012-04-04 09:31:58 -04:00
|
|
|
tickBounds = bounds
|
2012-03-01 21:55:32 -05:00
|
|
|
else:
|
2012-04-04 09:31:58 -04:00
|
|
|
tickBounds = linkedView.mapRectToItem(self, linkedView.boundingRect())
|
2012-03-01 21:55:32 -05:00
|
|
|
|
|
|
|
if self.orientation == 'left':
|
|
|
|
span = (bounds.topRight(), bounds.bottomRight())
|
2012-04-04 09:31:58 -04:00
|
|
|
tickStart = tickBounds.right()
|
2012-03-01 21:55:32 -05:00
|
|
|
tickStop = bounds.right()
|
|
|
|
tickDir = -1
|
|
|
|
axis = 0
|
|
|
|
elif self.orientation == 'right':
|
|
|
|
span = (bounds.topLeft(), bounds.bottomLeft())
|
2012-04-04 09:31:58 -04:00
|
|
|
tickStart = tickBounds.left()
|
2012-03-01 21:55:32 -05:00
|
|
|
tickStop = bounds.left()
|
|
|
|
tickDir = 1
|
|
|
|
axis = 0
|
|
|
|
elif self.orientation == 'top':
|
|
|
|
span = (bounds.bottomLeft(), bounds.bottomRight())
|
2012-04-04 09:31:58 -04:00
|
|
|
tickStart = tickBounds.bottom()
|
2012-03-01 21:55:32 -05:00
|
|
|
tickStop = bounds.bottom()
|
|
|
|
tickDir = -1
|
|
|
|
axis = 1
|
|
|
|
elif self.orientation == 'bottom':
|
|
|
|
span = (bounds.topLeft(), bounds.topRight())
|
2012-04-04 09:31:58 -04:00
|
|
|
tickStart = tickBounds.top()
|
2012-03-01 21:55:32 -05:00
|
|
|
tickStop = bounds.top()
|
|
|
|
tickDir = 1
|
|
|
|
axis = 1
|
2012-04-04 09:31:58 -04:00
|
|
|
#print tickStart, tickStop, span
|
|
|
|
|
2012-03-01 21:55:32 -05:00
|
|
|
## draw long line along axis
|
2012-06-29 14:39:27 -04:00
|
|
|
p.setPen(self.pen())
|
2012-03-01 21:55:32 -05:00
|
|
|
p.drawLine(*span)
|
2012-04-04 09:31:58 -04:00
|
|
|
p.translate(0.5,0) ## resolves some damn pixel ambiguity
|
2012-03-01 21:55:32 -05:00
|
|
|
|
|
|
|
## determine size of this item in pixels
|
2012-05-11 18:05:41 -04:00
|
|
|
points = list(map(self.mapToDevice, span))
|
2012-06-21 21:52:34 -04:00
|
|
|
if None in points:
|
|
|
|
return
|
2012-03-01 21:55:32 -05:00
|
|
|
lengthInPixels = Point(points[1] - points[0]).length()
|
2012-04-04 09:31:58 -04:00
|
|
|
if lengthInPixels == 0:
|
|
|
|
return
|
2012-03-01 21:55:32 -05:00
|
|
|
|
2012-05-21 17:31:09 -04:00
|
|
|
if self._tickLevels is None:
|
|
|
|
tickLevels = self.tickValues(self.range[0], self.range[1], lengthInPixels)
|
|
|
|
tickStrings = None
|
|
|
|
else:
|
|
|
|
## parse self.tickLevels into the formats returned by tickLevels() and tickStrings()
|
|
|
|
tickLevels = []
|
|
|
|
tickStrings = []
|
|
|
|
for level in self._tickLevels:
|
|
|
|
values = []
|
|
|
|
strings = []
|
|
|
|
tickLevels.append((None, values))
|
|
|
|
tickStrings.append(strings)
|
|
|
|
for val, strn in level:
|
|
|
|
values.append(val)
|
|
|
|
strings.append(strn)
|
2012-03-01 21:55:32 -05:00
|
|
|
|
2012-03-12 12:23:25 -04:00
|
|
|
textLevel = 1 ## draw text at this scale level
|
2012-03-01 21:55:32 -05:00
|
|
|
|
2012-04-04 09:31:58 -04:00
|
|
|
## determine mapping between tick values and local coordinates
|
|
|
|
dif = self.range[1] - self.range[0]
|
2012-03-01 21:55:32 -05:00
|
|
|
if axis == 0:
|
2012-04-04 09:31:58 -04:00
|
|
|
xScale = -bounds.height() / dif
|
|
|
|
offset = self.range[0] * xScale - bounds.height()
|
2012-03-01 21:55:32 -05:00
|
|
|
else:
|
2012-04-04 09:31:58 -04:00
|
|
|
xScale = bounds.width() / dif
|
|
|
|
offset = self.range[0] * xScale
|
2012-06-30 23:32:26 -04:00
|
|
|
|
|
|
|
xRange = [x * xScale - offset for x in self.range]
|
|
|
|
xMin = min(xRange)
|
|
|
|
xMax = max(xRange)
|
2012-03-01 21:55:32 -05:00
|
|
|
|
|
|
|
prof.mark('init')
|
|
|
|
|
2012-04-04 09:31:58 -04:00
|
|
|
tickPositions = [] # remembers positions of previously drawn ticks
|
|
|
|
|
|
|
|
## draw ticks
|
2012-03-01 21:55:32 -05:00
|
|
|
## (to improve performance, we do not interleave line and text drawing, since this causes unnecessary pipeline switching)
|
|
|
|
## draw three different intervals, long ticks first
|
2012-06-30 23:32:26 -04:00
|
|
|
|
2012-04-04 09:31:58 -04:00
|
|
|
for i in range(len(tickLevels)):
|
|
|
|
tickPositions.append([])
|
|
|
|
ticks = tickLevels[i][1]
|
2012-03-01 21:55:32 -05:00
|
|
|
|
|
|
|
## length of tick
|
2012-04-04 09:31:58 -04:00
|
|
|
tickLength = self.tickLength / ((i*1.0)+1.0)
|
2012-03-12 10:04:59 -04:00
|
|
|
|
2012-04-04 09:31:58 -04:00
|
|
|
lineAlpha = 255 / (i+1)
|
2012-03-12 10:04:59 -04:00
|
|
|
if self.grid is not False:
|
2012-07-03 14:44:07 -04:00
|
|
|
lineAlpha *= self.grid/255. * np.clip((0.05 * lengthInPixels / (len(ticks)+1)), 0., 1.)
|
2012-03-01 21:55:32 -05:00
|
|
|
|
2012-04-04 09:31:58 -04:00
|
|
|
for v in ticks:
|
2012-06-30 23:32:26 -04:00
|
|
|
## determine actual position to draw this tick
|
2012-04-04 09:31:58 -04:00
|
|
|
x = (v * xScale) - offset
|
2012-06-30 23:32:26 -04:00
|
|
|
if x < xMin or x > xMax: ## last check to make sure no out-of-bounds ticks are drawn
|
|
|
|
tickPositions[i].append(None)
|
|
|
|
continue
|
|
|
|
tickPositions[i].append(x)
|
|
|
|
|
2012-04-04 09:31:58 -04:00
|
|
|
p1 = [x, x]
|
|
|
|
p2 = [x, x]
|
2012-03-01 21:55:32 -05:00
|
|
|
p1[axis] = tickStart
|
2012-03-12 10:04:59 -04:00
|
|
|
p2[axis] = tickStop
|
|
|
|
if self.grid is False:
|
2012-04-04 09:31:58 -04:00
|
|
|
p2[axis] += tickLength*tickDir
|
2012-06-29 14:39:27 -04:00
|
|
|
tickPen = self.pen()
|
|
|
|
color = tickPen.color()
|
|
|
|
color.setAlpha(lineAlpha)
|
|
|
|
tickPen.setColor(color)
|
|
|
|
p.setPen(tickPen)
|
2012-03-12 12:23:25 -04:00
|
|
|
p.drawLine(Point(p1), Point(p2))
|
2012-03-01 21:55:32 -05:00
|
|
|
prof.mark('draw ticks')
|
2012-05-21 17:31:09 -04:00
|
|
|
|
|
|
|
## Draw text until there is no more room (or no more text)
|
|
|
|
textRects = []
|
2012-04-04 09:31:58 -04:00
|
|
|
for i in range(len(tickLevels)):
|
2012-05-21 17:31:09 -04:00
|
|
|
## Get the list of strings to display for this level
|
|
|
|
if tickStrings is None:
|
|
|
|
spacing, values = tickLevels[i]
|
|
|
|
strings = self.tickStrings(values, self.scale, spacing)
|
2012-04-04 09:31:58 -04:00
|
|
|
else:
|
2012-05-21 17:31:09 -04:00
|
|
|
strings = tickStrings[i]
|
2012-04-04 09:31:58 -04:00
|
|
|
|
2012-05-21 17:31:09 -04:00
|
|
|
if len(strings) == 0:
|
2012-04-04 09:31:58 -04:00
|
|
|
continue
|
2012-06-30 23:32:26 -04:00
|
|
|
|
|
|
|
## ignore strings belonging to ticks that were previously ignored
|
|
|
|
for j in range(len(strings)):
|
|
|
|
if tickPositions[i][j] is None:
|
|
|
|
strings[j] = None
|
2012-05-21 17:31:09 -04:00
|
|
|
|
2012-07-09 08:38:30 -04:00
|
|
|
textRects.extend([p.boundingRect(QtCore.QRectF(0, 0, 100, 100), QtCore.Qt.AlignCenter, s) for s in strings if s is not None])
|
2012-05-21 17:31:09 -04:00
|
|
|
if i > 0: ## always draw top level
|
|
|
|
## measure all text, make sure there's enough room
|
|
|
|
if axis == 0:
|
|
|
|
textSize = np.sum([r.height() for r in textRects])
|
|
|
|
else:
|
|
|
|
textSize = np.sum([r.width() for r in textRects])
|
|
|
|
|
|
|
|
## If the strings are too crowded, stop drawing text now
|
|
|
|
textFillRatio = float(textSize) / lengthInPixels
|
|
|
|
if textFillRatio > 0.7:
|
|
|
|
break
|
|
|
|
#spacing, values = tickLevels[best]
|
|
|
|
#strings = self.tickStrings(values, self.scale, spacing)
|
|
|
|
for j in range(len(strings)):
|
|
|
|
vstr = strings[j]
|
2012-06-30 23:32:26 -04:00
|
|
|
if vstr is None:## this tick was ignored because it is out of bounds
|
|
|
|
continue
|
2012-05-21 17:31:09 -04:00
|
|
|
x = tickPositions[i][j]
|
|
|
|
textRect = p.boundingRect(QtCore.QRectF(0, 0, 100, 100), QtCore.Qt.AlignCenter, vstr)
|
|
|
|
height = textRect.height()
|
|
|
|
self.textHeight = height
|
|
|
|
if self.orientation == 'left':
|
|
|
|
textFlags = QtCore.Qt.AlignRight|QtCore.Qt.AlignVCenter
|
|
|
|
rect = QtCore.QRectF(tickStop-100, x-(height/2), 99-max(0,self.tickLength), height)
|
|
|
|
elif self.orientation == 'right':
|
|
|
|
textFlags = QtCore.Qt.AlignLeft|QtCore.Qt.AlignVCenter
|
|
|
|
rect = QtCore.QRectF(tickStop+max(0,self.tickLength)+1, x-(height/2), 100-max(0,self.tickLength), height)
|
|
|
|
elif self.orientation == 'top':
|
|
|
|
textFlags = QtCore.Qt.AlignCenter|QtCore.Qt.AlignBottom
|
|
|
|
rect = QtCore.QRectF(x-100, tickStop-max(0,self.tickLength)-height, 200, height)
|
|
|
|
elif self.orientation == 'bottom':
|
|
|
|
textFlags = QtCore.Qt.AlignCenter|QtCore.Qt.AlignTop
|
|
|
|
rect = QtCore.QRectF(x-100, tickStop+max(0,self.tickLength), 200, height)
|
|
|
|
|
2012-06-29 14:39:27 -04:00
|
|
|
p.setPen(self.pen())
|
2012-05-21 17:31:09 -04:00
|
|
|
p.drawText(rect, textFlags, vstr)
|
2012-03-01 21:55:32 -05:00
|
|
|
prof.mark('draw text')
|
|
|
|
prof.finish()
|
|
|
|
|
|
|
|
def show(self):
|
|
|
|
|
|
|
|
if self.orientation in ['left', 'right']:
|
|
|
|
self.setWidth()
|
|
|
|
else:
|
|
|
|
self.setHeight()
|
|
|
|
GraphicsWidget.show(self)
|
|
|
|
|
|
|
|
def hide(self):
|
|
|
|
if self.orientation in ['left', 'right']:
|
|
|
|
self.setWidth(0)
|
|
|
|
else:
|
|
|
|
self.setHeight(0)
|
|
|
|
GraphicsWidget.hide(self)
|
|
|
|
|
|
|
|
def wheelEvent(self, ev):
|
2012-03-12 10:04:59 -04:00
|
|
|
if self.linkedView() is None:
|
|
|
|
return
|
2012-03-01 21:55:32 -05:00
|
|
|
if self.orientation in ['left', 'right']:
|
|
|
|
self.linkedView().wheelEvent(ev, axis=1)
|
|
|
|
else:
|
|
|
|
self.linkedView().wheelEvent(ev, axis=0)
|
|
|
|
ev.accept()
|
2012-03-12 10:04:59 -04:00
|
|
|
|
|
|
|
def mouseDragEvent(self, event):
|
|
|
|
if self.linkedView() is None:
|
|
|
|
return
|
|
|
|
if self.orientation in ['left', 'right']:
|
|
|
|
return self.linkedView().mouseDragEvent(event, axis=1)
|
|
|
|
else:
|
|
|
|
return self.linkedView().mouseDragEvent(event, axis=0)
|
|
|
|
|
|
|
|
def mouseClickEvent(self, event):
|
|
|
|
if self.linkedView() is None:
|
|
|
|
return
|
|
|
|
return self.linkedView().mouseClickEvent(event)
|