doc: update IRC reference to Libera.Chat
[sigrok-meter/gsi.git] / multiplotwidget.py
blob603b2f7b2c409a8253e95b42998052d9ef586383
1 ##
2 ## This file is part of the sigrok-meter project.
3 ##
4 ## Copyright (C) 2015 Jens Steinhauser <jens.steinhauser@gmail.com>
5 ##
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/>.
20 import qtcompat
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')
30 class Plot(object):
31 '''Helper class to keep all graphics items of a plot together.'''
33 def __init__(self, view, xaxis, yaxis):
34 self.view = view
35 self.xaxis = xaxis
36 self.yaxis = yaxis
37 self.visible = False
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)
55 for i in range(2):
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.
64 self._plots = []
66 self._hideActions = {}
68 def addPlot(self):
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.
76 if self._plots:
77 view.setXLink(self._plots[-1].view)
79 yaxis = pyqtgraph.AxisItem(parent=self, orientation='left')
80 yaxis.linkToView(view)
81 yaxis.setGrid(255)
83 xaxis = pyqtgraph.AxisItem(parent=self, orientation='bottom')
84 xaxis.linkToView(view)
85 xaxis.setGrid(255)
87 plot = Plot(view, xaxis, yaxis)
88 self._plots.append(plot)
90 self.showPlot(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)
100 return plot
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)
108 @QtCore.Slot()
109 def _onHideActionTriggered(self, checked=False):
110 # The plot that we want to hide.
111 plot = self._hideActions[id(self.sender())]
112 self.hidePlot(plot)
114 def hidePlot(self, plot):
115 '''Hides '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)
123 plot.view.hide()
124 plot.xaxis.hide()
125 plot.yaxis.hide()
127 row = self._rowNumber(plot)
128 self.layout().setRowStretchFactor(row, 0)
129 self.layout().setRowStretchFactor(row + 1, 0)
131 plot.visible = False
132 self.plotHidden.emit(plot)
134 def showPlot(self, plot):
135 '''Adds the items of the plot to the scene's layout and makes
136 them visible.'''
138 if plot.visible:
139 return
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)
146 plot.view.show()
147 plot.xaxis.show()
148 plot.yaxis.show()
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)
158 plot.visible = True
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)
173 for m in [
174 'addPlot',
175 'hidePlot',
176 'showPlot'
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)