pyqtgraph/examples/PaletteEditor.py
2021-06-13 16:57:44 +09:00

422 lines
17 KiB
Python

#!/usr/bin/python
# -*- coding: utf-8 -*-
"""
Adjust color palettes and apply them to a running plot
"""
## Add path to library (just for examples; you do not need this)
import initExample
import numpy as np
from pyqtgraph.Qt import mkQApp, QtCore, QtGui, QtWidgets
from pyqtgraph.ptime import time
import pyqtgraph as pg
class MainWindow(QtWidgets.QMainWindow):
""" example application main window """
def __init__(self, *args, **kwargs):
super(MainWindow, self).__init__(*args, **kwargs)
pg.setConfigOption('imageAxisOrder', 'row-major')
force_dark = False # start in forced dark mode?
self.setWindowTitle('pyqtgraph example: Palette editor')
self.resize(600,600)
self.palette_options = (
('system (reapply to update)', 'system', []),
('legacy', 'legacy', []),
('relaxed (dark)' , 'relaxed_dark', []),
('relaxed (light)', 'relaxed_light', []),
('pastels (light)', 'pastels', []),
('mono green', 'monochrome', ['green']),
('mono amber', 'monochrome', ['amber']),
('mono blue' , 'monochrome', ['blue' ]),
('synthwave' , 'synthwave', []),
)
self.colormap_options = (
'CET-C1', 'CET-C2','CET-C6','CET-C7', 'CET-R2', 'CET-R4',
'CET-L8', 'CET-L16', 'CET-CBC1', 'CET-CBC2', 'none'
)
app = QtWidgets.QApplication.instance()
self.q_palette = {
'system' : app.palette(),
'dark' : self.make_dark_QPalette()
}
app.setStyle("Fusion")
self.ui = self.prepare_ui() # relocate long-winded window layout
# dictionary self.ui contains references to UI elements
if force_dark:
self.ui['dark'].setChecked(True)
self.handle_dark_button(True)
self.open_palette = pg.palette.get('system')
self.open_palette.apply()
self.update_color_fields( self.open_palette )
self.num_points = 30
# configure overview plot with four colors:
plt = self.ui['plot1']
plt.enableAutoRange(False)
plt.setYRange( 0, 4, padding=0 )
plt.setXRange( 0, self.num_points, padding=0 )
for key in ('left','right','top','bottom'):
ax = plt.getAxis(key)
ax.show()
ax.setZValue(0.1)
self.curves = []
curve = pg.PlotCurveItem(pen='p0', brush=('p0',127))
curve.setFillLevel(0)
self.curves.append( (1, 1, curve) ) # dataset 1, vertical offset 3
plt.addItem(curve)
curve = pg.ScatterPlotItem(
symbol='o', size=5, pen='p0', brush=('p0',127),
hoverable=True, hoverPen='gr_acc', hoverBrush='p0')
self.curves.append( (1, 1, curve) ) # dataset 1, vertical offset 2
plt.addItem(curve)
pen_list = ['p2', 'p4', 'p6'] # add three more plots
for idx, pen in enumerate( pen_list ):
curve = pg.PlotCurveItem()
curve.setPen(pen, width=5)
self.curves.append( (3+2*idx, 1.5+0.8*idx, curve) ) # datasets 2+, vertical offset 3+
plt.addItem(curve)
# configure tall plot with eight colors and region overlay:
plt = self.ui['plot2']
plt.enableAutoRange(False)
plt.setYRange( -0.6, 7.6, padding=0 )
plt.setXRange( 0, self.num_points, padding=0 )
plt.getAxis('bottom').hide()
plt.getAxis('left').setLabel('plot color')
plt.getAxis('left').setGrid(0.5) # 63)
pen_list = [('p0',2),('p1',2),('p2',2),('p3',2),('p4',2),('p5',2),('p6',2),('p7',2)] # add right-side plots for each main color
for idx, pen in enumerate( pen_list ):
curve = pg.PlotCurveItem(pen=pen)
self.curves.append( (1+idx, idx, curve) ) # datasets 2+, vertical offset by index
plt.addItem(curve)
item = pg.LinearRegionItem( values=(4, 8), orientation='vertical' )
plt.addItem(item)
# show off some color text:
col_list = [
('k','black'),('b','blue'),('c','cyan'),('g','green'),('y','yellow'),
('r','red'),('m','magenta'),('w','white')]
for idx in range(10):
name = 'm{:d}'.format(idx)
col_list.append( (name, '## '+name+' ##') )
for idx, (color,text) in enumerate( col_list ):
text_item = pg.TextItem(text, color=color, anchor=(1,0))
text_item.setPos( self.num_points, 7.7 - idx/3 )
plt.addItem(text_item)
self.show()
# prepare for continuous updates and frame rate measurement
self.last_time = time()
self.fps = None
self.timer = QtCore.QTimer(singleShot=False)
self.timer.timeout.connect( self.timed_update )
# prepare initial data and display in plots
self.data = np.zeros((9, self.num_points ))
self.data[0,:] = np.arange( self.data.shape[1] ) # used as x data
self.phases = np.zeros(9)
self.timed_update()
### handle GUI interaction ###############################################
def update_color_fields(self, pal):
""" update line edit fields for selected palette """
if pal is None:
print('palette is None!')
return
for key in self.ui['widget_from_color_key']:
wid = self.ui['widget_from_color_key'][key]
qcol = pal[key]
if wid is not None:
wid.setText( qcol.name() )
def handle_palette_select(self, idx):
""" user selected a palette in dropdown menu """
text, identifier, args = self.palette_options[idx]
del text # not needed here
self.open_palette = pg.palette.get(identifier, *args)
print('loaded palette:', identifier, args)
if identifier in pg.palette.PALETTE_DEFINITIONS:
info = pg.palette.PALETTE_DEFINITIONS[identifier]
colormap_sampling = info['colormap_sampling']
if colormap_sampling is None:
identifier, start, step = 'none', 0.000, 0.125
else:
identifier, start, step = colormap_sampling
self.ui['sample_start'].setText('{:+.3f}'.format(start))
self.ui['sample_step' ].setText('{:+.3f}'.format(step) )
for idx, map_id in enumerate( self.colormap_options ):
if map_id == identifier:
# print('found colormap at idx',idx)
self.ui['colormaps'].setCurrentIndex(idx)
self.update_color_fields(self.open_palette)
self.open_palette.apply()
def handle_colormap_select(self, param=None):
""" user selected a colormap in dropdown menu or changed start / step vales """
del param # drop index sent by QComboBox
identifier = self.ui['colormaps'].currentText()
if identifier == 'none':
return
start = self.ui['sample_start'].text()
step = self.ui['sample_step' ].text()
try:
start = float(start)
except ValueError:
start = 0.0
self.ui['sample_start'].setText('{:+.3f}'.format(start) )
try:
step = float(step)
except ValueError:
step = 0.125
self.ui['sample_step'].setText('{:+.3f}'.format(step) )
self.open_palette.sampleColorMap( cmap=identifier, start=start, step=step )
# print('applied color map {:s} starting at {:.3f} with step {:3f}'.format(identifier, start, step) )
self.update_color_fields(self.open_palette)
self.open_palette.apply()
def handle_color_update(self):
""" figure out what color field was updated """
source = self.sender()
key = self.ui['color_key_from_widget'][source]
requested = source.text()
if len(requested) < 1:
value = 0x808080
else:
if requested[0] == '#':
requested = requested[1:]
try:
value = int(requested,16)
except ValueError:
value = 0x808080
color = '#{:06x}'.format(value)
print('color value is',color)
# source.setText(color)
self.open_palette[key] = color
self.update_color_fields(self.open_palette)
self.open_palette.apply()
print('color update requested for',key)
def handle_update_button(self, active):
""" start/stop timer """
if active:
self.timer.start(1)
else:
self.timer.stop()
def handle_dark_button(self, active):
""" manually switch to dark palette to test on windows """
app = QtWidgets.QApplication.instance()
if active:
app.setPalette( self.q_palette['dark'] ) # apply dark QPalette
else:
app.setPalette( self.q_palette['system'] ) # reapply QPalette stored at start-up
def timed_update(self):
""" update loop, called by timer """
self.speed = np.linspace(0.01, 0.06, 9)
self.phases += self.speed * np.random.normal(1, 1, size=9)
for idx in range(1, self.data.shape[0]):
self.data[idx, :-1] = self.data[idx, 1:] # roll
self.data[1:, -1] = 0.5 * np.sin( self.phases[1:] )
xdata = self.data[0,:]
for idx, offset, curve in self.curves:
curve.setData( x=xdata, y=( offset + self.data[idx,:] ) )
now = time()
dt = now - self.last_time
self.last_time = now
if self.fps is None:
self.fps = 1.0/dt
else:
s = np.clip(dt*3., 0, 1)
self.fps = self.fps * (1-s) + (1.0/dt) * s
self.ui['plot2'].setTitle('%0.1f fps' % self.fps)
QtWidgets.QApplication.processEvents() ## force complete redraw for every plot
### Qt color definitions for dark palette on Windows #####################
def make_dark_QPalette(self):
""" manually define a dark mode palette """
BLACK = QtGui.QColor('#000000')
BG_LIGHT = QtGui.QColor('#505354')
BG_NORMAL = QtGui.QColor('#2e3132')
BG_DARK = QtGui.QColor('#0e1112')
FG_LIGHT = QtGui.QColor('#f0f4f5')
FG_NORMAL = QtGui.QColor('#d4d8d9')
FG_DARK = QtGui.QColor('#b8bcbd')
SEL_LIGHT = QtGui.QColor('#148CD2')
SEL_NORMAL = QtGui.QColor('#1464A0')
SEL_DARK = QtGui.QColor('#14506E')
qpal = QtGui.QPalette( QtGui.QColor(BG_DARK) )
for ptype in ( QtGui.QPalette.ColorGroup.Active, QtGui.QPalette.ColorGroup.Inactive ):
qpal.setColor( ptype, QtGui.QPalette.ColorRole.Window, BG_NORMAL )
qpal.setColor( ptype, QtGui.QPalette.ColorRole.WindowText, FG_LIGHT ) # or white?
qpal.setColor( ptype, QtGui.QPalette.ColorRole.Base, BG_DARK )
qpal.setColor( ptype, QtGui.QPalette.ColorRole.Text, FG_LIGHT )
qpal.setColor( ptype, QtGui.QPalette.ColorRole.AlternateBase, BG_DARK )
qpal.setColor( ptype, QtGui.QPalette.ColorRole.ToolTipBase, BG_LIGHT )
qpal.setColor( ptype, QtGui.QPalette.ColorRole.ToolTipText, FG_LIGHT )
qpal.setColor( ptype, QtGui.QPalette.ColorRole.Button, BG_NORMAL )
qpal.setColor( ptype, QtGui.QPalette.ColorRole.ButtonText, FG_LIGHT )
qpal.setColor( ptype, QtGui.QPalette.ColorRole.Link, SEL_NORMAL )
qpal.setColor( ptype, QtGui.QPalette.ColorRole.LinkVisited, FG_NORMAL )
qpal.setColor( ptype, QtGui.QPalette.ColorRole.Highlight, SEL_LIGHT )
qpal.setColor( ptype, QtGui.QPalette.ColorRole.HighlightedText, BLACK )
qpal.setColor( QtGui.QPalette.ColorGroup.Disabled, QtGui.QPalette.ColorRole.Button, BG_NORMAL )
qpal.setColor( QtGui.QPalette.ColorGroup.Disabled, QtGui.QPalette.ColorRole.ButtonText, FG_DARK )
qpal.setColor( QtGui.QPalette.ColorGroup.Disabled, QtGui.QPalette.ColorRole.WindowText, FG_DARK )
return qpal
##########################################################################
def prepare_ui(self):
""" Boring Qt window layout code is implemented here """
ui = {}
main_wid = QtWidgets.QWidget()
self.setCentralWidget(main_wid)
color_fields = (
# key, description, reference to line edit field)
['gr_bg' , (0,0), 'graph background' ],
['gr_fg' , (1,0), 'graph foreground' ],
['gr_txt', (2,0), 'graph text' ],
['gr_reg', (3,0), 'graph region' ],
['gr_acc', (4,0), 'graphical accent' ],
['gr_hlt', (5,0), 'graphical highlight'],
['p0', (0,1), ' plot 0'], ['p1', (1,1), ' plot 1'],
['p2', (2,1), ' plot 2'], ['p3', (3,1), ' plot 3'],
['p4', (4,1), ' plot 4'], ['p5', (5,1), ' plot 5'],
['p6', (6,1), ' plot 6'], ['p7', (7,1), ' plot 7']
)
gr_wid1 = pg.GraphicsLayoutWidget(show=True)
ui['plot1'] = gr_wid1.addPlot()
gr_wid2 = pg.GraphicsLayoutWidget(show=True)
ui['plot2'] = gr_wid2.addPlot()
main_layout = QtWidgets.QHBoxLayout( main_wid )
l_wid = QtWidgets.QWidget()
main_layout.addWidget(l_wid)
main_layout.addWidget(gr_wid2)
l_layout = QtWidgets.QGridLayout( l_wid )
l_layout.setContentsMargins(0,0,0,0)
l_layout.setSpacing(1)
row_idx = 0
label = QtWidgets.QLabel('Override system style:')
l_layout.addWidget( label, row_idx,0, 1,2 )
label = QtWidgets.QLabel('Select a palette:')
l_layout.addWidget( label, row_idx,2, 1,2 )
row_idx += 1
btn = QtWidgets.QPushButton('Apply dark GUI')
btn.setCheckable(True)
btn.setChecked(False)
btn.clicked.connect(self.handle_dark_button)
box = QtWidgets.QComboBox()
for text, identifier, args in self.palette_options:
del identifier, args # not needed here
box.addItem(text)
box.activated.connect(self.handle_palette_select)
l_layout.addWidget( box, row_idx,2, 1,2 )
l_layout.addWidget( btn, row_idx,0, 1,2 )
ui['dark'] = btn
row_idx += 1
label = QtWidgets.QLabel('Sampled color map:')
l_layout.addWidget( label, row_idx,0, 1,2 )
label = QtWidgets.QLabel('start')
l_layout.addWidget( label, row_idx,2, 1,1 )
label = QtWidgets.QLabel('step')
l_layout.addWidget( label, row_idx,3, 1,1 )
row_idx += 1
box = QtWidgets.QComboBox()
for identifier in self.colormap_options:
box.addItem(identifier)
ui['colormaps'] = box
ui['colormaps'].activated.connect(self.handle_colormap_select)
ui['sample_start'] = QtWidgets.QLineEdit(' 0.000')
ui['sample_start'].editingFinished.connect(self.handle_colormap_select)
ui['sample_step' ] = QtWidgets.QLineEdit('+0.125')
ui['sample_step' ].editingFinished.connect(self.handle_colormap_select)
l_layout.addWidget( box, row_idx,0, 1,2 )
l_layout.addWidget( ui['sample_start'], row_idx,2, 1,1 )
l_layout.addWidget( ui['sample_step' ], row_idx,3, 1,1 )
row_idx += 1
spacer = QtWidgets.QWidget()
spacer.setFixedHeight(10)
l_layout.addWidget( spacer, row_idx,0, 1,2 )
row_idx += 1
label = QtWidgets.QLabel('Functional colors:')
l_layout.addWidget( label, row_idx,0, 1,2 )
row_idx += 1
row = 0
ui['widget_from_color_key'] = {} # look-up for color editing fields
ui['color_key_from_widget'] = {} # reverse look-up for color editing fields
for field_list in color_fields:
key, pos, text = field_list
lab = QtWidgets.QLabel(text)
lab.setAlignment(QtCore.Qt.AlignmentFlag.AlignCenter)
edt = QtWidgets.QLineEdit()
edt.editingFinished.connect(self.handle_color_update)
row = row_idx + pos[0]
col = 2 * pos[1] # 0 or 2
l_layout.addWidget( lab, row,col+0, 1,1 )
l_layout.addWidget( edt, row,col+1, 1,1 )
ui['color_key_from_widget'][edt] = key
ui['widget_from_color_key'][key] = edt
row_idx = row
btn = QtWidgets.QPushButton('generate continuous data')
btn.setCheckable(True)
btn.setChecked(False)
btn.clicked.connect(self.handle_update_button)
l_layout.addWidget( btn, row_idx,0, 1,2 )
row_idx += 1
spacer = QtWidgets.QWidget()
spacer.setFixedHeight(10)
l_layout.addWidget( spacer, row_idx,0, 1,2 )
row_idx += 1
label = QtWidgets.QLabel('Overview:')
l_layout.addWidget( label, row_idx,0, 1,4 )
row_idx += 1
l_layout.addWidget( gr_wid1, row_idx,0, 1,4 )
row_idx += 1
return ui
mkQApp("Palette editor")
main_window = MainWindow()
## Start Qt event loop
if __name__ == '__main__':
pg.exec()