2 ## This file is part of the sigrok-meter project.
4 ## Copyright (C) 2015 Jens Steinhauser <jens.steinhauser@gmail.com>
6 ## This program is free software; you can redistribute it and/or modify
7 ## it under the terms of the GNU General Public License as published by
8 ## the Free Software Foundation; either version 2 of the License, or
9 ## (at your option) any later version.
11 ## This program is distributed in the hope that it will be useful,
12 ## but WITHOUT ANY WARRANTY; without even the implied warranty of
13 ## MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 ## GNU General Public License for more details.
16 ## You should have received a copy of the GNU General Public License
17 ## along with this program; if not, see <http://www.gnu.org/licenses/>.
22 QtCore
= qtcompat
.QtCore
23 QtGui
= qtcompat
.QtGui
24 pyqtgraph
= qtcompat
.pyqtgraph
26 # Black foreground on white background.
27 pyqtgraph
.setConfigOption('background', 'w')
28 pyqtgraph
.setConfigOption('foreground', 'k')
31 '''Helper class to keep all graphics items of a plot together.'''
33 def __init__(self
, view
, xaxis
, yaxis
):
39 class MultiPlotItem(pyqtgraph
.GraphicsWidget
):
41 # Emitted when a plot is shown.
42 plotShown
= QtCore
.Signal()
44 # Emitted when a plot is hidden by the user via the context menu.
45 plotHidden
= QtCore
.Signal(Plot
)
47 def __init__(self
, parent
=None):
48 pyqtgraph
.GraphicsWidget
.__init
__(self
, parent
)
50 self
.setLayout(QtGui
.QGraphicsGridLayout())
51 self
.layout().setContentsMargins(10, 10, 10, 1)
52 self
.layout().setHorizontalSpacing(0)
53 self
.layout().setVerticalSpacing(0)
56 self
.layout().setColumnPreferredWidth(i
, 0)
57 self
.layout().setColumnMinimumWidth(i
, 0)
58 self
.layout().setColumnSpacing(i
, 0)
60 self
.layout().setColumnStretchFactor(0, 0)
61 self
.layout().setColumnStretchFactor(1, 100)
63 # List of 'Plot' objects that are shown.
66 self
._hideActions
= {}
69 '''Adds and returns a new plot.'''
71 row
= self
.layout().rowCount()
73 view
= pyqtgraph
.ViewBox(parent
=self
)
75 # If this is not the first plot, link to the axis of the previous one.
77 view
.setXLink(self
._plots
[-1].view
)
79 yaxis
= pyqtgraph
.AxisItem(parent
=self
, orientation
='left')
80 yaxis
.linkToView(view
)
83 xaxis
= pyqtgraph
.AxisItem(parent
=self
, orientation
='bottom')
84 xaxis
.linkToView(view
)
87 plot
= Plot(view
, xaxis
, yaxis
)
88 self
._plots
.append(plot
)
92 # Create a separate action object for each plots context menu, so that
93 # we can later find out which plot should be hidden by looking at
94 # 'self._hideActions'.
95 hideAction
= QtGui
.QAction('Hide', self
)
96 hideAction
.triggered
.connect(self
._onHideActionTriggered
)
97 self
._hideActions
[id(hideAction
)] = plot
98 view
.menu
.insertAction(view
.menu
.actions()[0], hideAction
)
102 def _rowNumber(self
, plot
):
103 '''Returns the number of the first row a plot occupies.'''
105 # Every plot takes up two rows.
106 return 2 * self
._plots
.index(plot
)
109 def _onHideActionTriggered(self
, checked
=False):
110 # The plot that we want to hide.
111 plot
= self
._hideActions
[id(self
.sender())]
114 def hidePlot(self
, plot
):
117 # Only hiding wouldn't give up the space occupied by the items,
118 # we have to remove them from the layout.
119 self
.layout().removeItem(plot
.view
)
120 self
.layout().removeItem(plot
.xaxis
)
121 self
.layout().removeItem(plot
.yaxis
)
127 row
= self
._rowNumber
(plot
)
128 self
.layout().setRowStretchFactor(row
, 0)
129 self
.layout().setRowStretchFactor(row
+ 1, 0)
132 self
.plotHidden
.emit(plot
)
134 def showPlot(self
, plot
):
135 '''Adds the items of the plot to the scene's layout and makes
141 row
= self
._rowNumber
(plot
)
142 self
.layout().addItem(plot
.yaxis
, row
, 0, QtCore
.Qt
.AlignRight
)
143 self
.layout().addItem(plot
.view
, row
, 1)
144 self
.layout().addItem(plot
.xaxis
, row
+ 1, 1)
150 for i
in range(row
, row
+ 2):
151 self
.layout().setRowPreferredHeight(i
, 0)
152 self
.layout().setRowMinimumHeight(i
, 0)
153 self
.layout().setRowSpacing(i
, 0)
155 self
.layout().setRowStretchFactor(row
, 100)
156 self
.layout().setRowStretchFactor(row
+ 1, 0)
159 self
.plotShown
.emit()
161 class MultiPlotWidget(pyqtgraph
.GraphicsView
):
162 '''Widget that aligns multiple plots on top of each other.
164 (The built in classes fail at doing this correctly when the axis grow,
165 just try zooming in the "GraphicsLayout" or the "Linked View" examples.)'''
167 def __init__(self
, parent
=None):
168 pyqtgraph
.GraphicsView
.__init
__(self
, parent
)
170 self
.multiPlotItem
= MultiPlotItem()
171 self
.setCentralItem(self
.multiPlotItem
)
178 setattr(self
, m
, getattr(self
.multiPlotItem
, m
))
180 self
.multiPlotItem
.plotShown
.connect(self
._on
_plotShown
)
182 # Expose the signal of the plot item.
183 self
.plotHidden
= self
.multiPlotItem
.plotHidden
185 def _on_plotShown(self
):
186 # This call is needed if only one plot exists and it was hidden,
187 # without it the layout would start acting weird and not make the
188 # MultiPlotItem fill the view widget after showing the plot again.
189 self
.resizeEvent(None)