lazylabel-gui 1.1.0__py3-none-any.whl → 1.1.2__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- lazylabel/__init__.py +8 -8
- lazylabel/config/__init__.py +6 -6
- lazylabel/config/hotkeys.py +168 -168
- lazylabel/config/paths.py +40 -40
- lazylabel/config/settings.py +65 -65
- lazylabel/core/__init__.py +6 -6
- lazylabel/core/file_manager.py +105 -105
- lazylabel/core/model_manager.py +97 -94
- lazylabel/core/segment_manager.py +171 -140
- lazylabel/main.py +36 -22
- lazylabel/models/__init__.py +4 -4
- lazylabel/models/sam_model.py +195 -154
- lazylabel/ui/__init__.py +7 -7
- lazylabel/ui/control_panel.py +241 -220
- lazylabel/ui/editable_vertex.py +64 -51
- lazylabel/ui/hotkey_dialog.py +383 -383
- lazylabel/ui/hoverable_pixelmap_item.py +22 -22
- lazylabel/ui/hoverable_polygon_item.py +39 -39
- lazylabel/ui/main_window.py +1659 -1264
- lazylabel/ui/numeric_table_widget_item.py +9 -9
- lazylabel/ui/photo_viewer.py +54 -54
- lazylabel/ui/reorderable_class_table.py +61 -61
- lazylabel/ui/right_panel.py +315 -239
- lazylabel/ui/widgets/__init__.py +8 -7
- lazylabel/ui/widgets/adjustments_widget.py +108 -107
- lazylabel/ui/widgets/model_selection_widget.py +93 -93
- lazylabel/ui/widgets/settings_widget.py +105 -105
- lazylabel/ui/widgets/status_bar.py +109 -0
- lazylabel/utils/__init__.py +5 -5
- lazylabel/utils/custom_file_system_model.py +132 -132
- lazylabel/utils/utils.py +12 -12
- {lazylabel_gui-1.1.0.dist-info → lazylabel_gui-1.1.2.dist-info}/METADATA +197 -197
- lazylabel_gui-1.1.2.dist-info/RECORD +37 -0
- {lazylabel_gui-1.1.0.dist-info → lazylabel_gui-1.1.2.dist-info}/licenses/LICENSE +21 -21
- lazylabel_gui-1.1.0.dist-info/RECORD +0 -36
- {lazylabel_gui-1.1.0.dist-info → lazylabel_gui-1.1.2.dist-info}/WHEEL +0 -0
- {lazylabel_gui-1.1.0.dist-info → lazylabel_gui-1.1.2.dist-info}/entry_points.txt +0 -0
- {lazylabel_gui-1.1.0.dist-info → lazylabel_gui-1.1.2.dist-info}/top_level.txt +0 -0
lazylabel/ui/right_panel.py
CHANGED
@@ -1,239 +1,315 @@
|
|
1
|
-
"""Right panel with file explorer and segment management."""
|
2
|
-
|
3
|
-
from PyQt6.QtWidgets import (
|
4
|
-
QWidget,
|
5
|
-
|
6
|
-
|
7
|
-
|
8
|
-
|
9
|
-
|
10
|
-
|
11
|
-
|
12
|
-
|
13
|
-
|
14
|
-
|
15
|
-
|
16
|
-
|
17
|
-
|
18
|
-
|
19
|
-
|
20
|
-
|
21
|
-
|
22
|
-
|
23
|
-
|
24
|
-
|
25
|
-
|
26
|
-
|
27
|
-
|
28
|
-
|
29
|
-
|
30
|
-
|
31
|
-
|
32
|
-
|
33
|
-
|
34
|
-
|
35
|
-
|
36
|
-
|
37
|
-
|
38
|
-
|
39
|
-
|
40
|
-
self.
|
41
|
-
self.
|
42
|
-
|
43
|
-
self.
|
44
|
-
|
45
|
-
|
46
|
-
|
47
|
-
|
48
|
-
|
49
|
-
|
50
|
-
|
51
|
-
|
52
|
-
|
53
|
-
|
54
|
-
self.
|
55
|
-
|
56
|
-
|
57
|
-
|
58
|
-
|
59
|
-
|
60
|
-
|
61
|
-
|
62
|
-
|
63
|
-
|
64
|
-
|
65
|
-
|
66
|
-
|
67
|
-
|
68
|
-
|
69
|
-
|
70
|
-
|
71
|
-
|
72
|
-
|
73
|
-
|
74
|
-
|
75
|
-
|
76
|
-
|
77
|
-
|
78
|
-
|
79
|
-
|
80
|
-
|
81
|
-
self.
|
82
|
-
|
83
|
-
|
84
|
-
|
85
|
-
|
86
|
-
|
87
|
-
|
88
|
-
|
89
|
-
|
90
|
-
layout
|
91
|
-
|
92
|
-
|
93
|
-
|
94
|
-
|
95
|
-
self.
|
96
|
-
|
97
|
-
|
98
|
-
layout.
|
99
|
-
|
100
|
-
|
101
|
-
|
102
|
-
|
103
|
-
|
104
|
-
|
105
|
-
|
106
|
-
|
107
|
-
|
108
|
-
|
109
|
-
|
110
|
-
|
111
|
-
|
112
|
-
self.
|
113
|
-
self.
|
114
|
-
|
115
|
-
|
116
|
-
|
117
|
-
|
118
|
-
|
119
|
-
|
120
|
-
|
121
|
-
|
122
|
-
|
123
|
-
|
124
|
-
|
125
|
-
|
126
|
-
|
127
|
-
|
128
|
-
|
129
|
-
|
130
|
-
self.
|
131
|
-
|
132
|
-
|
133
|
-
)
|
134
|
-
self.
|
135
|
-
self.
|
136
|
-
|
137
|
-
|
138
|
-
self.
|
139
|
-
|
140
|
-
|
141
|
-
|
142
|
-
self.
|
143
|
-
|
144
|
-
)
|
145
|
-
|
146
|
-
|
147
|
-
|
148
|
-
|
149
|
-
|
150
|
-
|
151
|
-
|
152
|
-
|
153
|
-
|
154
|
-
|
155
|
-
|
156
|
-
self.class_table
|
157
|
-
self.
|
158
|
-
|
159
|
-
|
160
|
-
|
161
|
-
"""
|
162
|
-
|
163
|
-
|
164
|
-
|
165
|
-
|
166
|
-
|
167
|
-
|
168
|
-
|
169
|
-
|
170
|
-
|
171
|
-
|
172
|
-
|
173
|
-
|
174
|
-
|
175
|
-
|
176
|
-
|
177
|
-
|
178
|
-
|
179
|
-
def
|
180
|
-
"""
|
181
|
-
self.
|
182
|
-
|
183
|
-
|
184
|
-
|
185
|
-
|
186
|
-
self.
|
187
|
-
|
188
|
-
|
189
|
-
|
190
|
-
|
191
|
-
|
192
|
-
|
193
|
-
|
194
|
-
|
195
|
-
|
196
|
-
|
197
|
-
|
198
|
-
|
199
|
-
|
200
|
-
|
201
|
-
|
202
|
-
|
203
|
-
|
204
|
-
|
205
|
-
|
206
|
-
|
207
|
-
|
208
|
-
|
209
|
-
|
210
|
-
|
211
|
-
|
212
|
-
|
213
|
-
|
214
|
-
|
215
|
-
for
|
216
|
-
|
217
|
-
|
218
|
-
|
219
|
-
|
220
|
-
|
221
|
-
|
222
|
-
|
223
|
-
|
224
|
-
|
225
|
-
|
226
|
-
|
227
|
-
|
228
|
-
|
229
|
-
|
230
|
-
|
231
|
-
|
232
|
-
|
233
|
-
|
234
|
-
|
235
|
-
|
236
|
-
|
237
|
-
|
238
|
-
|
239
|
-
|
1
|
+
"""Right panel with file explorer and segment management."""
|
2
|
+
|
3
|
+
from PyQt6.QtWidgets import (
|
4
|
+
QWidget,
|
5
|
+
QVBoxLayout,
|
6
|
+
QPushButton,
|
7
|
+
QLabel,
|
8
|
+
QHBoxLayout,
|
9
|
+
QTableWidget,
|
10
|
+
QTreeView,
|
11
|
+
QComboBox,
|
12
|
+
QSplitter,
|
13
|
+
QSpacerItem,
|
14
|
+
QHeaderView,
|
15
|
+
)
|
16
|
+
from PyQt6.QtCore import Qt, pyqtSignal
|
17
|
+
from PyQt6.QtGui import QBrush, QColor
|
18
|
+
|
19
|
+
from .reorderable_class_table import ReorderableClassTable
|
20
|
+
from .numeric_table_widget_item import NumericTableWidgetItem
|
21
|
+
|
22
|
+
|
23
|
+
class RightPanel(QWidget):
|
24
|
+
"""Right panel with file explorer and segment management."""
|
25
|
+
|
26
|
+
# Signals
|
27
|
+
open_folder_requested = pyqtSignal()
|
28
|
+
image_selected = pyqtSignal("QModelIndex")
|
29
|
+
merge_selection_requested = pyqtSignal()
|
30
|
+
delete_selection_requested = pyqtSignal()
|
31
|
+
segments_selection_changed = pyqtSignal()
|
32
|
+
class_alias_changed = pyqtSignal(int, str) # class_id, alias
|
33
|
+
reassign_classes_requested = pyqtSignal()
|
34
|
+
class_filter_changed = pyqtSignal()
|
35
|
+
class_toggled = pyqtSignal(int) # class_id
|
36
|
+
pop_out_requested = pyqtSignal()
|
37
|
+
|
38
|
+
def __init__(self, parent=None):
|
39
|
+
super().__init__(parent)
|
40
|
+
self.setMinimumWidth(50) # Allow collapsing but maintain minimum
|
41
|
+
self.preferred_width = 350 # Store preferred width for expansion
|
42
|
+
self._setup_ui()
|
43
|
+
self._connect_signals()
|
44
|
+
|
45
|
+
def _setup_ui(self):
|
46
|
+
"""Setup the UI layout."""
|
47
|
+
self.v_layout = QVBoxLayout(self)
|
48
|
+
|
49
|
+
# Top button row
|
50
|
+
toggle_layout = QHBoxLayout()
|
51
|
+
|
52
|
+
self.btn_popout = QPushButton("⋯")
|
53
|
+
self.btn_popout.setToolTip("Pop out panel to separate window")
|
54
|
+
self.btn_popout.setMaximumWidth(30)
|
55
|
+
toggle_layout.addWidget(self.btn_popout)
|
56
|
+
|
57
|
+
toggle_layout.addStretch()
|
58
|
+
|
59
|
+
self.v_layout.addLayout(toggle_layout)
|
60
|
+
|
61
|
+
# Main controls widget
|
62
|
+
self.main_controls_widget = QWidget()
|
63
|
+
main_layout = QVBoxLayout(self.main_controls_widget)
|
64
|
+
main_layout.setContentsMargins(0, 0, 0, 0)
|
65
|
+
|
66
|
+
# Vertical splitter for sections
|
67
|
+
v_splitter = QSplitter(Qt.Orientation.Vertical)
|
68
|
+
|
69
|
+
# File explorer section
|
70
|
+
self._setup_file_explorer(v_splitter)
|
71
|
+
|
72
|
+
# Segment management section
|
73
|
+
self._setup_segment_management(v_splitter)
|
74
|
+
|
75
|
+
# Class management section
|
76
|
+
self._setup_class_management(v_splitter)
|
77
|
+
|
78
|
+
main_layout.addWidget(v_splitter)
|
79
|
+
|
80
|
+
# Status label
|
81
|
+
self.status_label = QLabel("")
|
82
|
+
self.status_label.setAlignment(Qt.AlignmentFlag.AlignCenter)
|
83
|
+
main_layout.addWidget(self.status_label)
|
84
|
+
|
85
|
+
self.v_layout.addWidget(self.main_controls_widget)
|
86
|
+
|
87
|
+
def _setup_file_explorer(self, splitter):
|
88
|
+
"""Setup file explorer section."""
|
89
|
+
file_explorer_widget = QWidget()
|
90
|
+
layout = QVBoxLayout(file_explorer_widget)
|
91
|
+
layout.setContentsMargins(0, 0, 0, 0)
|
92
|
+
|
93
|
+
self.btn_open_folder = QPushButton("Open Image Folder")
|
94
|
+
self.btn_open_folder.setToolTip("Open a directory of images")
|
95
|
+
layout.addWidget(self.btn_open_folder)
|
96
|
+
|
97
|
+
self.file_tree = QTreeView()
|
98
|
+
layout.addWidget(self.file_tree)
|
99
|
+
|
100
|
+
splitter.addWidget(file_explorer_widget)
|
101
|
+
|
102
|
+
def _setup_segment_management(self, splitter):
|
103
|
+
"""Setup segment management section."""
|
104
|
+
segment_widget = QWidget()
|
105
|
+
layout = QVBoxLayout(segment_widget)
|
106
|
+
layout.setContentsMargins(0, 0, 0, 0)
|
107
|
+
|
108
|
+
# Class filter
|
109
|
+
filter_layout = QHBoxLayout()
|
110
|
+
filter_layout.addWidget(QLabel("Filter Class:"))
|
111
|
+
self.class_filter_combo = QComboBox()
|
112
|
+
self.class_filter_combo.setToolTip("Filter segments list by class")
|
113
|
+
filter_layout.addWidget(self.class_filter_combo)
|
114
|
+
layout.addLayout(filter_layout)
|
115
|
+
|
116
|
+
# Segment table
|
117
|
+
self.segment_table = QTableWidget()
|
118
|
+
self.segment_table.setColumnCount(3)
|
119
|
+
self.segment_table.setHorizontalHeaderLabels(
|
120
|
+
["Segment ID", "Class ID", "Alias"]
|
121
|
+
)
|
122
|
+
self.segment_table.horizontalHeader().setSectionResizeMode(
|
123
|
+
QHeaderView.ResizeMode.Stretch
|
124
|
+
)
|
125
|
+
self.segment_table.setSelectionBehavior(
|
126
|
+
QTableWidget.SelectionBehavior.SelectRows
|
127
|
+
)
|
128
|
+
self.segment_table.setSortingEnabled(True)
|
129
|
+
self.segment_table.setEditTriggers(QTableWidget.EditTrigger.NoEditTriggers)
|
130
|
+
layout.addWidget(self.segment_table)
|
131
|
+
|
132
|
+
# Action buttons
|
133
|
+
action_layout = QHBoxLayout()
|
134
|
+
self.btn_merge_selection = QPushButton("Merge to Class")
|
135
|
+
self.btn_merge_selection.setToolTip(
|
136
|
+
"Merge selected segments into a single class (M)"
|
137
|
+
)
|
138
|
+
self.btn_delete_selection = QPushButton("Delete")
|
139
|
+
self.btn_delete_selection.setToolTip(
|
140
|
+
"Delete selected segments (Delete/Backspace)"
|
141
|
+
)
|
142
|
+
action_layout.addWidget(self.btn_merge_selection)
|
143
|
+
action_layout.addWidget(self.btn_delete_selection)
|
144
|
+
layout.addLayout(action_layout)
|
145
|
+
|
146
|
+
splitter.addWidget(segment_widget)
|
147
|
+
|
148
|
+
def _setup_class_management(self, splitter):
|
149
|
+
"""Setup class management section."""
|
150
|
+
class_widget = QWidget()
|
151
|
+
layout = QVBoxLayout(class_widget)
|
152
|
+
layout.setContentsMargins(0, 0, 0, 0)
|
153
|
+
|
154
|
+
layout.addWidget(QLabel("Class Order:"))
|
155
|
+
|
156
|
+
self.class_table = ReorderableClassTable()
|
157
|
+
self.class_table.setToolTip(
|
158
|
+
"Double-click to set class aliases and drag to reorder channels for saving.\nClick once to toggle as active class for new segments."
|
159
|
+
)
|
160
|
+
self.class_table.setColumnCount(2)
|
161
|
+
self.class_table.setHorizontalHeaderLabels(["Alias", "Class ID"])
|
162
|
+
self.class_table.horizontalHeader().setSectionResizeMode(
|
163
|
+
0, QHeaderView.ResizeMode.Stretch
|
164
|
+
)
|
165
|
+
self.class_table.horizontalHeader().setSectionResizeMode(
|
166
|
+
1, QHeaderView.ResizeMode.ResizeToContents
|
167
|
+
)
|
168
|
+
self.class_table.setEditTriggers(QTableWidget.EditTrigger.DoubleClicked)
|
169
|
+
layout.addWidget(self.class_table)
|
170
|
+
|
171
|
+
self.btn_reassign_classes = QPushButton("Reassign Class IDs")
|
172
|
+
self.btn_reassign_classes.setToolTip(
|
173
|
+
"Re-index class channels based on the current order in this table"
|
174
|
+
)
|
175
|
+
layout.addWidget(self.btn_reassign_classes)
|
176
|
+
|
177
|
+
splitter.addWidget(class_widget)
|
178
|
+
|
179
|
+
def _connect_signals(self):
|
180
|
+
"""Connect internal signals."""
|
181
|
+
self.btn_open_folder.clicked.connect(self.open_folder_requested)
|
182
|
+
self.file_tree.doubleClicked.connect(self.image_selected)
|
183
|
+
self.btn_merge_selection.clicked.connect(self.merge_selection_requested)
|
184
|
+
self.btn_delete_selection.clicked.connect(self.delete_selection_requested)
|
185
|
+
self.segment_table.itemSelectionChanged.connect(self.segments_selection_changed)
|
186
|
+
self.class_table.itemChanged.connect(self._handle_class_alias_change)
|
187
|
+
self.class_table.cellClicked.connect(self._handle_class_toggle)
|
188
|
+
self.btn_reassign_classes.clicked.connect(self.reassign_classes_requested)
|
189
|
+
self.class_filter_combo.currentIndexChanged.connect(self.class_filter_changed)
|
190
|
+
self.btn_popout.clicked.connect(self.pop_out_requested)
|
191
|
+
|
192
|
+
def mouseDoubleClickEvent(self, event):
|
193
|
+
"""Handle double-click to expand collapsed panel."""
|
194
|
+
if self.width() < 50: # If panel is collapsed
|
195
|
+
# Request expansion by calling parent method
|
196
|
+
if self.parent() and hasattr(self.parent(), "_expand_right_panel"):
|
197
|
+
self.parent()._expand_right_panel()
|
198
|
+
super().mouseDoubleClickEvent(event)
|
199
|
+
|
200
|
+
def _handle_class_alias_change(self, item):
|
201
|
+
"""Handle class alias change in table."""
|
202
|
+
if item.column() != 0: # Only handle alias column
|
203
|
+
return
|
204
|
+
|
205
|
+
class_table = self.class_table
|
206
|
+
id_item = class_table.item(item.row(), 1)
|
207
|
+
if id_item:
|
208
|
+
try:
|
209
|
+
class_id = int(id_item.text())
|
210
|
+
self.class_alias_changed.emit(class_id, item.text())
|
211
|
+
except (ValueError, AttributeError):
|
212
|
+
pass
|
213
|
+
|
214
|
+
def _handle_class_toggle(self, row, column):
|
215
|
+
"""Handle class table cell click for toggling active class."""
|
216
|
+
# Get the class ID from the clicked row
|
217
|
+
id_item = self.class_table.item(row, 1)
|
218
|
+
if id_item:
|
219
|
+
try:
|
220
|
+
class_id = int(id_item.text())
|
221
|
+
self.class_toggled.emit(class_id)
|
222
|
+
except (ValueError, AttributeError):
|
223
|
+
pass
|
224
|
+
|
225
|
+
def update_active_class_display(self, active_class_id):
|
226
|
+
"""Update the visual display to show which class is active."""
|
227
|
+
# Block signals to prevent triggering change events during update
|
228
|
+
self.class_table.blockSignals(True)
|
229
|
+
|
230
|
+
for row in range(self.class_table.rowCount()):
|
231
|
+
id_item = self.class_table.item(row, 1)
|
232
|
+
alias_item = self.class_table.item(row, 0)
|
233
|
+
if id_item and alias_item:
|
234
|
+
try:
|
235
|
+
class_id = int(id_item.text())
|
236
|
+
if class_id == active_class_id:
|
237
|
+
# Make active class bold and add indicator
|
238
|
+
font = alias_item.font()
|
239
|
+
font.setBold(True)
|
240
|
+
alias_item.setFont(font)
|
241
|
+
id_item.setFont(font)
|
242
|
+
# Add visual indicator
|
243
|
+
if not alias_item.text().startswith("🔸 "):
|
244
|
+
alias_item.setText(f"🔸 {alias_item.text()}")
|
245
|
+
else:
|
246
|
+
# Make inactive classes normal
|
247
|
+
font = alias_item.font()
|
248
|
+
font.setBold(False)
|
249
|
+
alias_item.setFont(font)
|
250
|
+
id_item.setFont(font)
|
251
|
+
# Remove visual indicator
|
252
|
+
if alias_item.text().startswith("🔸 "):
|
253
|
+
alias_item.setText(alias_item.text()[2:])
|
254
|
+
except (ValueError, AttributeError):
|
255
|
+
pass
|
256
|
+
|
257
|
+
# Re-enable signals
|
258
|
+
self.class_table.blockSignals(False)
|
259
|
+
|
260
|
+
def setup_file_model(self, file_model):
|
261
|
+
"""Setup the file model for the tree view."""
|
262
|
+
self.file_tree.setModel(file_model)
|
263
|
+
self.file_tree.setColumnWidth(0, 200)
|
264
|
+
|
265
|
+
def set_folder(self, folder_path, file_model):
|
266
|
+
"""Set the folder for file browsing."""
|
267
|
+
self.file_tree.setRootIndex(file_model.setRootPath(folder_path))
|
268
|
+
|
269
|
+
def get_selected_segment_indices(self):
|
270
|
+
"""Get indices of selected segments."""
|
271
|
+
selected_items = self.segment_table.selectedItems()
|
272
|
+
selected_rows = sorted(list({item.row() for item in selected_items}))
|
273
|
+
return [
|
274
|
+
self.segment_table.item(row, 0).data(Qt.ItemDataRole.UserRole)
|
275
|
+
for row in selected_rows
|
276
|
+
if self.segment_table.item(row, 0)
|
277
|
+
]
|
278
|
+
|
279
|
+
def get_class_order(self):
|
280
|
+
"""Get the current class order from the class table."""
|
281
|
+
ordered_ids = []
|
282
|
+
for row in range(self.class_table.rowCount()):
|
283
|
+
id_item = self.class_table.item(row, 1)
|
284
|
+
if id_item and id_item.text():
|
285
|
+
try:
|
286
|
+
ordered_ids.append(int(id_item.text()))
|
287
|
+
except ValueError:
|
288
|
+
continue
|
289
|
+
return ordered_ids
|
290
|
+
|
291
|
+
def clear_selections(self):
|
292
|
+
"""Clear all selections."""
|
293
|
+
self.segment_table.clearSelection()
|
294
|
+
self.class_table.clearSelection()
|
295
|
+
|
296
|
+
def select_all_segments(self):
|
297
|
+
"""Select all segments."""
|
298
|
+
self.segment_table.selectAll()
|
299
|
+
|
300
|
+
def set_status(self, message):
|
301
|
+
"""Set status message."""
|
302
|
+
self.status_label.setText(message)
|
303
|
+
|
304
|
+
def clear_status(self):
|
305
|
+
"""Clear status message."""
|
306
|
+
self.status_label.clear()
|
307
|
+
|
308
|
+
def set_popout_mode(self, is_popped_out: bool):
|
309
|
+
"""Update the pop-out button based on panel state."""
|
310
|
+
if is_popped_out:
|
311
|
+
self.btn_popout.setText("⇤")
|
312
|
+
self.btn_popout.setToolTip("Return panel to main window")
|
313
|
+
else:
|
314
|
+
self.btn_popout.setText("⋯")
|
315
|
+
self.btn_popout.setToolTip("Pop out panel to separate window")
|
lazylabel/ui/widgets/__init__.py
CHANGED
@@ -1,7 +1,8 @@
|
|
1
|
-
"""UI widgets for LazyLabel."""
|
2
|
-
|
3
|
-
from .model_selection_widget import ModelSelectionWidget
|
4
|
-
from .settings_widget import SettingsWidget
|
5
|
-
from .adjustments_widget import AdjustmentsWidget
|
6
|
-
|
7
|
-
|
1
|
+
"""UI widgets for LazyLabel."""
|
2
|
+
|
3
|
+
from .model_selection_widget import ModelSelectionWidget
|
4
|
+
from .settings_widget import SettingsWidget
|
5
|
+
from .adjustments_widget import AdjustmentsWidget
|
6
|
+
from .status_bar import StatusBar
|
7
|
+
|
8
|
+
__all__ = ["ModelSelectionWidget", "SettingsWidget", "AdjustmentsWidget", "StatusBar"]
|