@@ -148,6 +148,8 @@ def __init__(self, parent: QWidget | None = None):
148
148
class _QLUTWidget (QWidget ):
149
149
def __init__ (self , parent : QWidget | None = None ) -> None :
150
150
super ().__init__ (parent )
151
+
152
+ # -- WIDGETS -- #
151
153
self .visible = QCheckBox ()
152
154
153
155
self .cmap = _CmapCombo ()
@@ -162,54 +164,58 @@ def __init__(self, parent: QWidget | None = None) -> None:
162
164
QLabeledRangeSlider .LabelPosition .LabelsOnHandle
163
165
)
164
166
self .clims .setEdgeLabelMode (QLabeledRangeSlider .EdgeLabelMode .NoLabel )
165
- self .clims .setRange (0 , 2 ** 16 ) # TODO: expose
167
+ self .clims .setRange (0 , 2 ** 16 )
166
168
167
169
self .auto_clim = QPushButton ("Auto" )
168
170
self .auto_clim .setMaximumWidth (42 )
169
171
self .auto_clim .setCheckable (True )
170
172
171
- # TODO: Consider other options here
172
173
add_histogram_icon = QIconifyIcon ("foundation:graph-bar" )
173
174
self .histogram_btn = QPushButton (add_histogram_icon , "" )
174
175
self .histogram_btn .setCheckable (True )
175
176
176
- top = QHBoxLayout ()
177
- top .setSpacing (5 )
178
- top .setContentsMargins (0 , 0 , 0 , 0 )
179
- top .addWidget (self .visible )
180
- top .addWidget (self .cmap )
181
- top .addWidget (self .clims )
182
- top .addWidget (self .auto_clim )
183
- top .addWidget (self .histogram_btn )
184
-
185
- # TODO: Consider a container for this...
186
- self ._histogram_layout = QHBoxLayout ()
187
- histogram_ctrls = QVBoxLayout ()
177
+ set_range_icon = QIconifyIcon ("fluent:full-screen-maximize-24-filled" )
178
+ self .hist_range = QPushButton (set_range_icon , "" , self )
179
+ self .hist_range .setVisible (False )
180
+
181
+ log_icon = QIconifyIcon ("mdi:math-log" )
182
+ self .hist_log = QPushButton (log_icon , "" , self )
183
+ self .hist_log .setToolTip ("log (base 10, count+1)" )
184
+ self .hist_log .setCheckable (True )
185
+ self .hist_log .setVisible (False )
186
+
187
+ # -- LAYOUT -- #
188
+
189
+ # "main" lut controls (always visible)
190
+ lut_layout = QHBoxLayout ()
191
+ lut_layout .setSpacing (5 )
192
+ lut_layout .setContentsMargins (0 , 0 , 0 , 0 )
193
+ lut_layout .addWidget (self .visible )
194
+ lut_layout .addWidget (self .cmap )
195
+ lut_layout .addWidget (self .clims )
196
+ lut_layout .addWidget (self .auto_clim )
197
+ lut_layout .addWidget (self .histogram_btn )
198
+
199
+ # histogram controls go in their own layout
200
+ hist_ctrls_layout = QVBoxLayout ()
188
201
# Add a vertical spacer that expands to take up available space
189
202
# This is the key component that pushes everything down
190
- spacer = QSpacerItem (
191
- 0 , 0 , QSizePolicy .Policy .Minimum , QSizePolicy .Policy .Expanding
203
+ hist_ctrls_layout . addItem (
204
+ QSpacerItem ( 0 , 0 , QSizePolicy .Policy .Minimum , QSizePolicy .Policy .Expanding )
192
205
)
193
- histogram_ctrls .addItem (spacer )
194
- set_range_icon = QIconifyIcon ("fluent:full-screen-maximize-24-filled" )
195
- self .set_hist_range_btn = QPushButton (set_range_icon , "" , self )
196
- self .set_hist_range_btn .setVisible (False )
197
- log_icon = QIconifyIcon ("mdi:math-log" )
198
- self .log_btn = QPushButton (log_icon , "" , self )
199
- self .log_btn .setToolTip ("log (base 10, count+1)" )
200
- self .log_btn .setCheckable (True )
201
- self .log_btn .setVisible (False )
202
- histogram_ctrls .addWidget (self .log_btn )
203
- histogram_ctrls .addWidget (self .set_hist_range_btn )
204
- self .histogram : HistogramCanvas | None = None
205
- self ._histogram_layout .addLayout (histogram_ctrls )
206
+ hist_ctrls_layout .addWidget (self .hist_log )
207
+ hist_ctrls_layout .addWidget (self .hist_range )
208
+
209
+ # histogram layout contains controls + a histogram (which is added later)
210
+ self .hist_layout = QHBoxLayout ()
211
+ self .hist_layout .addLayout (hist_ctrls_layout )
206
212
207
- # Retain a reference to this layout so we can add self._histogram later
213
+ # Overall layout
208
214
self ._layout = QVBoxLayout (self )
209
215
self ._layout .setSpacing (0 )
210
216
self ._layout .setContentsMargins (0 , 0 , 0 , 0 )
211
- self ._layout .addLayout (top )
212
- self ._layout .addLayout (self ._histogram_layout )
217
+ self ._layout .addLayout (lut_layout )
218
+ self ._layout .addLayout (self .hist_layout )
213
219
214
220
215
221
class QLutView (LutView ):
@@ -220,12 +226,11 @@ def __init__(self, channel: ChannelKey = None) -> None:
220
226
super ().__init__ ()
221
227
self ._qwidget = _QLUTWidget ()
222
228
self ._channel = channel
229
+ self .histogram : HistogramCanvas | None = None
223
230
# TODO: use emit_fast
224
231
self ._qwidget .histogram_btn .toggled .connect (self ._on_q_histogram_toggled )
225
- self ._qwidget .log_btn .toggled .connect (self ._on_log_btn_toggled )
226
- self ._qwidget .set_hist_range_btn .clicked .connect (
227
- self ._on_set_histogram_range_clicked
228
- )
232
+ self ._qwidget .hist_log .toggled .connect (self ._on_log_btn_toggled )
233
+ self ._qwidget .hist_range .clicked .connect (self ._on_set_histogram_range_clicked )
229
234
self ._qwidget .visible .toggled .connect (self ._on_q_visibility_changed )
230
235
self ._qwidget .cmap .currentColormapChanged .connect (self ._on_q_cmap_changed )
231
236
self ._qwidget .clims .valueChanged .connect (self ._on_q_clims_changed )
@@ -291,26 +296,29 @@ def _on_q_auto_changed(self, autoscale: bool) -> None:
291
296
self ._model .clims = ClimsManual (min = clims [0 ], max = clims [1 ])
292
297
293
298
def _on_q_histogram_toggled (self , toggled : bool ) -> None :
294
- def set_visible_in (layout : QLayout ) -> None :
295
- if count := layout .count ():
296
- for i in range (count ):
297
- item = layout .itemAt (i )
298
- if item and item .widget () and isinstance (item .widget (), QWidget ):
299
- item .widget ().setVisible (toggled )
300
- elif item and isinstance (item .layout (), QLayout ):
301
- set_visible_in (item .layout ())
302
-
303
- set_visible_in (self ._qwidget ._histogram_layout )
304
- if self ._qwidget .histogram is None :
299
+ # Recursively show/hide hist_layout
300
+ self ._set_layout_visibility (toggled , self ._qwidget .hist_layout )
301
+ # Request histogram if not created yet
302
+ if self .histogram is None :
305
303
self .histogramRequested .emit (self ._channel )
306
304
305
+ def _set_layout_visibility (self , toggled : bool , layout : QLayout ) -> None :
306
+ for i in range (layout .count ()):
307
+ item = layout .itemAt (i )
308
+ if item is None :
309
+ continue
310
+ if wdg := item .widget ():
311
+ wdg .setVisible (toggled )
312
+ elif child_layout := item .layout ():
313
+ self ._set_layout_visibility (toggled , child_layout )
314
+
307
315
def _on_log_btn_toggled (self , toggled : bool ) -> None :
308
- if hist := self ._qwidget . histogram :
316
+ if hist := self .histogram :
309
317
hist .set_log_base (10 if toggled else None )
310
318
311
319
def _on_set_histogram_range_clicked (self ) -> None :
312
- self ._qwidget .log_btn .setChecked (False )
313
- if hist := self ._qwidget . histogram :
320
+ self ._qwidget .hist_log .setChecked (False )
321
+ if hist := self .histogram :
314
322
hist .set_range ()
315
323
316
324
@@ -569,10 +577,10 @@ def add_histogram(self, channel: ChannelKey, histogram: HistogramCanvas) -> None
569
577
QSize (lut ._qwidget .width (), lut ._qwidget .height () + 100 )
570
578
)
571
579
# Add widget to view
572
- lut ._qwidget . histogram = histogram
580
+ lut .histogram = histogram
573
581
widget = cast (QWidget , histogram .frontend_widget ())
574
582
widget .resize (QSize (lut ._qwidget .width (), 100 ))
575
- lut ._qwidget ._histogram_layout .addWidget (widget , 1 )
583
+ lut ._qwidget .hist_layout .addWidget (widget , 1 )
576
584
577
585
def remove_histogram (self , widget : QWidget ) -> None :
578
586
widget .setParent (None )
0 commit comments