Newer
Older
Contains the scheduler_gui MainWindow class for scheduling operations in an SFG.
Start main-window with ``start_gui()``.
import webbrowser
from collections import defaultdict, deque
from copy import deepcopy
from importlib.machinery import SourceFileLoader
from typing import TYPE_CHECKING, Deque, Dict, List, Optional, cast, overload
from qtpy.QtCore import (
QByteArray,
QCoreApplication,
QRectF,
QSettings,
QStandardPaths,
Qt,
Slot,
)
from qtpy.QtGui import QCloseEvent, QColor, QFont, QIcon, QIntValidator, QPalette
QColorDialog,
QDialog,
QDialogButtonBox,
import b_asic.logger as logger
from b_asic._version import __version__
from b_asic.gui_utils.about_window import AboutWindow
from b_asic.gui_utils.color_button import ColorButton
from b_asic.gui_utils.mpl_window import MPLWindow
from b_asic.scheduler_gui._preferences import (
Active_Color,
ColorDataType,
Execution_Time_Color,
Font,
Latency_Color,
Signal_Color,
Signal_Warning_Color,
)
from b_asic.scheduler_gui.axes_item import AxesItem
from b_asic.scheduler_gui.operation_item import OperationItem
from b_asic.scheduler_gui.scheduler_item import SchedulerItem
from b_asic.scheduler_gui.ui_main_window import Ui_MainWindow
log: "Logger" = logger.getLogger(__name__, "scheduler-gui.log")
if __debug__:
# Print some system version information
log.debug(f"Qt version (runtime): {QtCore.qVersion()}")
log.debug(f"Qt version (compile time): {QtCore.__version__}")
log.debug(f"QT_API: {QT_API}")
log.debug(f"PySide version: {PySide6.__version__}")
log.debug(f"PyQt version: {PYQT_VERSION_STR}")
log.debug(f"QtPy version: {qtpy.__version__}")
# The following QCoreApplication values is used for QSettings among others
QCoreApplication.setOrganizationName("Linköping University")
QCoreApplication.setApplicationName("B-ASIC Scheduling GUI")
QCoreApplication.setApplicationVersion(__version__)
class ScheduleMainWindow(QMainWindow, Ui_MainWindow):
_schedule: Optional[Schedule]
_graph: Optional[SchedulerItem]
_debug_rectangles: QGraphicsItemGroup
_color_per_type: Dict[str, QColor] = dict()
converted_colorPerType: Dict[str, str] = dict()
super().__init__()
self._schedule = None
self._graph = None
self._scale = 75.0
self.setupUi(self)
self._read_settings()
self._init_ui()
self._show_incorrect_execution_time = True
self._execution_time_for_variables = None
self._execution_time_plot_dialogs = defaultdict(lambda: None)
self._ports_accesses_for_storage = None
self._color_changed_perType = False
self.changed_operation_colors: Dict[str, QColor] = dict()
# Recent files
self._max_recent_files = 4
self._recent_files_actions: List[QAction] = []
self._recent_file_paths: Deque[str] = deque(maxlen=self._max_recent_files)
self._create_recent_file_actions_and_menus()
self._init_graphics()
def _init_ui(self) -> None:
"""Initialize the ui"""
# Connect signals to slots
self.menu_load_from_file.triggered.connect(self._load_schedule_from_pyfile)
self.menu_load_from_file.setIcon(get_icon('import'))
self.menu_open.setIcon(get_icon('open'))
self.menu_open.triggered.connect(self.open_schedule)
self.menu_close_schedule.triggered.connect(self.close_schedule)
self.menu_close_schedule.setIcon(get_icon('close'))
self.actionPreferences.triggered.connect(self.Preferences_Dialog_clicked)
self.menu_node_info.triggered.connect(self.show_info_table)
self.menu_exit_dialog.triggered.connect(self.hide_exit_dialog)
self.actionReorder.triggered.connect(self._action_reorder)
self.actionReorder.setIcon(get_icon('reorder'))
self.actionStatus_bar.triggered.connect(self._toggle_statusbar)
self.action_incorrect_execution_time.setIcon(get_icon('warning'))
self.action_incorrect_execution_time.triggered.connect(
self._toggle_execution_time_warning
)
self.action_show_port_numbers.setIcon(get_icon('port-numbers'))
self.action_show_port_numbers.triggered.connect(self._toggle_port_number)
self.actionPlot_schedule.setIcon(get_icon('plot-schedule'))
self.actionPlot_schedule.triggered.connect(self._plot_schedule)
self.action_view_variables.triggered.connect(
self._show_execution_times_for_variables
)
self.action_view_port_accesses.triggered.connect(
self._show_ports_accesses_for_storage
)
self.actionZoom_to_fit.setIcon(get_icon('zoom-to-fit'))
self.actionZoom_to_fit.triggered.connect(self._zoom_to_fit)
self.actionToggle_full_screen.setIcon(get_icon('full-screen'))
self.actionToggle_full_screen.triggered.connect(self._toggle_fullscreen)
self.actionUndo.setIcon(get_icon('undo'))
self.actionRedo.setIcon(get_icon('redo'))
self.splitter.splitterMoved.connect(self._splitter_moved)
self.actionDocumentation.triggered.connect(self._open_documentation)
self.actionAbout.triggered.connect(self._open_about_window)
self.actionDecrease_time_resolution.triggered.connect(
self._decrease_time_resolution
)
self.actionDecrease_time_resolution.setIcon(get_icon('decrease-timeresolution'))
self.actionIncrease_time_resolution.triggered.connect(
self._increase_time_resolution
)
self.actionIncrease_time_resolution.setIcon(get_icon('increase-timeresolution'))
# Setup event member functions
self.closeEvent = self._close_event
# Setup info table
self.info_table.setSpan(0, 0, 1, 2) # Span 'Schedule' over 2 columns
self.info_table.setSpan(1, 0, 1, 2) # Span 'Operator' over 2 columns
# Init central-widget splitter
self._splitter_min = self.splitter.minimumSizeHint().height()
self.splitter.setStretchFactor(0, 1)
self.splitter.setStretchFactor(1, 0)
self.splitter.setCollapsible(0, False)
self.splitter.setCollapsible(1, True)
def _init_graphics(self) -> None:
"""Initialize the QGraphics framework"""
self._scene = QGraphicsScene()
self._scene.addRect(0, 0, 0, 0) # dummy rect to be able to setPos() graph
self.view.setScene(self._scene)
self.view.scale(self._scale, self._scale)
OperationItem._scale = self._scale
AxesItem._scale = self._scale
self._scene.sceneRectChanged.connect(self.shrink_scene_to_min_size)
@property
#########
# Slots #
#########
def _plot_schedule(self) -> None:
"""Callback for plotting schedule using Matplotlib."""
@Slot()
def _open_documentation(self) -> None:
"""Callback to open documentation web page."""
webbrowser.open_new_tab("https://da.gitlab-pages.liu.se/B-ASIC/")
@Slot()
def _action_reorder(self) -> None:
"""Callback to reorder all operations vertically based on start time."""
if self.schedule is None:
return
if self._graph is not None:
self._graph._redraw_from_start()
self.update_statusbar("Operations reordered based on start time")
@Slot()
def _increase_time_resolution(self) -> None:
"""Callback for increasing time resolution."""
# Create dialog asking for int
factor, ok = QInputDialog.getInt(
self, "Increase time resolution", "Factor", 1, 1
)
# Check return value
if ok:
if factor > 1:
self.schedule.increase_time_resolution(factor)
self.open(self.schedule)
print(f"schedule.increase_time_resolution({factor})")
self.update_statusbar(f"Time resolution increased by a factor {factor}")
else: # Cancelled
@Slot()
def _decrease_time_resolution(self) -> None:
"""Callback for decreasing time resolution."""
# Get possible factors
vals = [str(v) for v in self.schedule.get_possible_time_resolution_decrements()]
# Create dialog
factor, ok = QInputDialog.getItem(
self, "Decrease time resolution", "Factor", vals, editable=False
)
# Check return value
if ok:
if int(factor) > 1:
self.schedule.decrease_time_resolution(int(factor))
self.open(self.schedule)
print(f"schedule.decrease_time_resolution({factor})")
self.update_statusbar(f"Time resolution decreased by a factor {factor}")
else: # Cancelled
def wheelEvent(self, event) -> None:
"""Zoom in or out using mouse wheel if control is pressed."""
if event.modifiers() == Qt.KeyboardModifier.ControlModifier:
old_zoom = self._zoom
self._zoom += event.angleDelta().y() / 2500
self.view.scale(self._zoom, self._zoom)
self._zoom = old_zoom
@Slot()
def _load_schedule_from_pyfile(self) -> None:
"""
SLOT() for SIGNAL(menu_load_from_file.triggered)
"""
QStandardPaths.standardLocations(QStandardPaths.HomeLocation)[0],
str,
)
if not os.path.exists(last_file): # if filename does not exist
last_file = os.path.dirname(last_file) + "/"
if not os.path.exists(last_file): # if path does not exist
last_file = QStandardPaths.standardLocations(
QStandardPaths.HomeLocation
)[0]
abs_path_filename, accepted = QFileDialog.getOpenFileName(
self,
self.tr("Open python file"),
last_file,
self.tr("Python Files (*.py *.py3)"),
)
if not abs_path_filename or not accepted:
self._load_from_file(abs_path_filename)
def _load_from_file(self, abs_path_filename):
"""
Import from Python-file.
Load a python script as a module and search for a Schedule object. If
found, opens it.
"""
log.debug(f"abs_path_filename = {abs_path_filename}.")
module_name = inspect.getmodulename(abs_path_filename)
log.error(f"Could not load module from file '{abs_path_filename}'.")
module = SourceFileLoader(module_name, abs_path_filename).load_module()
except Exception as e:
"Exception occurred. Could not load module from file"
" '{}'.\n\n{}".format(abs_path_filename, e)
self._add_recent_file(abs_path_filename)
schedule_obj_list = dict(
inspect.getmembers(module, (lambda x: isinstance(x, Schedule)))
)
if not schedule_obj_list: # return if no Schedule objects in script
QMessageBox.warning(
self,
self.tr("File not found"),
self.tr("Cannot find any Schedule object in file '{}'.").format(
os.path.basename(abs_path_filename)
),
"Cannot find any Schedule object in file '{}'.".format(
if len(schedule_obj_list) == 1:
schedule = [val for val in schedule_obj_list.values()][0]
else:
ret_tuple = QInputDialog.getItem(
self,
self.tr("Load object"),
self.tr(
"Found the following Schedule objects in file.\n\n"
"Select an object to proceed:"
),
schedule_obj_list.keys(),
0,
False,
)
if not ret_tuple[1]: # User canceled the operation
log.debug("Load schedule operation: user canceled")
del module
return
schedule = schedule_obj_list[ret_tuple[0]]
self.open(schedule)
settings.setValue("scheduler/last_opened_file", abs_path_filename)
self._file_name = abs_path_filename
self._toggle_file_loaded(True)
SLOT() for SIGNAL(menu_close_schedule.triggered)
"""
self._graph._signals.component_selected.disconnect(
self.info_table_update_component
)
self._graph._signals.schedule_time_changed.disconnect(
self.info_table_update_schedule
)
self._graph._signals.schedule_time_changed.disconnect(
self._schedule_changed
)
self._graph._signals.reopen.disconnect(self._reopen_schedule)
self._graph._signals.execution_time_plot.disconnect(
self._execution_time_plot
)
self._graph.removeSceneEventFilters(self._graph.event_items)
self._scene.removeItem(self._graph)
self.menu_close_schedule.setEnabled(False)
self.menu_save.setEnabled(False)
self.menu_save_as.setEnabled(False)
del self._graph
self._graph = None
del self._schedule
self._schedule = None
self.info_table_clear()
self.action_view_variables.setEnabled(False)
self.action_view_port_accesses.setEnabled(False)
self.menu_view_execution_times.setEnabled(False)
SLOT() for SIGNAL(menu_save.triggered)
"""
if self._file_name is None:
self.save_as()
return
self._schedule._sfg._graph_id_generator = None
with open(self._file_name, 'wb') as f:
pickle.dump(self._schedule, f)
self._add_recent_file(self._file_name)
self.update_statusbar(self.tr("Schedule saved successfully"))
Save current schedule asking for file name.
SLOT() for SIGNAL(menu_save_as.triggered)
"""
filename, extension = QFileDialog.getSaveFileName(
self, 'Save File', '.', filter=self.tr("B-ASIC schedule (*.bsc)")
)
if not filename:
return
if not filename.endswith('.bsc'):
filename += '.bsc'
self._file_name = filename
self._schedule._sfg._graph_id_generator = None
with open(self._file_name, 'wb') as f:
pickle.dump(self._schedule, f)
self._add_recent_file(self._file_name)
self.update_statusbar(self.tr("Schedule saved successfully"))
@Slot()
def open_schedule(self) -> None:
"""
Open a schedule.
SLOT() for SIGNAL(menu_open.triggered)
"""
# TODO: all
last_file = QStandardPaths.standardLocations(QStandardPaths.HomeLocation)[0]
abs_path_filename, accepted = QFileDialog.getOpenFileName(
self,
self.tr("Open schedule file"),
last_file,
self.tr("B-ASIC schedule (*.bsc)"),
)
if not abs_path_filename or not accepted:
return
self._open_schedule_file(abs_path_filename)
def _open_schedule_file(self, abs_path_filename: str):
"""Open a saved schedule (*.bsc-file), which is a pickled Schedule."""
self._file_name = abs_path_filename
self._add_recent_file(abs_path_filename)
with open(self._file_name, 'rb') as f:
schedule = pickle.load(f)
self.open(schedule)
settings = QSettings()
settings.setValue("scheduler/last_opened_file", self._file_name)
self._toggle_file_loaded(True)
self.update_statusbar(self.tr("Schedule loaded successfully"))
@Slot(bool)
def show_info_table(self, checked: bool) -> None:
SLOT(bool) for SIGNAL(menu_node_info.triggered)
Takes in a boolean and hide or show the info table accordingly with
# Note: splitter handler index 0 is a hidden splitter handle far most left,
# use index 1
_, max_ = self.splitter.getRange(1) # tuple(min, max)
if checked:
if self._splitter_pos < self._splitter_min:
self.splitter.moveSplitter(max_ - self._splitter_min, 1)
else:
self.splitter.moveSplitter(max_ - self._splitter_pos, 1)
else:
self.splitter.moveSplitter(max_, 1)
def _toggle_file_loaded(self, enable: bool):
self.menu_save.setEnabled(enable)
self.menu_save_as.setEnabled(enable)
@Slot(bool)
def hide_exit_dialog(self, checked: bool) -> None:
Update state of exit dialog setting.
SLOT(bool) for SIGNAL(menu_exit_dialog.triggered)
Takes in a boolean and stores 'checked' in 'hide_exit_dialog' item in
settings.
"""
settings = QSettings()
settings.setValue("scheduler/hide_exit_dialog", checked)
@Slot(int, int)
def _splitter_moved(self, pos: int, index: int) -> None:
"""
SLOT(int, int) for SIGNAL(splitter.splitterMoved)
Callback method used to check if the right widget (info window)
has collapsed. Update the checkbutton accordingly.
"""
width = self.splitter.sizes()[1]
if width == 0:
if self.menu_node_info.isChecked() is True:
self.menu_node_info.setChecked(False)
else:
if self.menu_node_info.isChecked() is False:
self.menu_node_info.setChecked(True)
self._splitter_pos = width
@Slot(str)
def info_table_update_component(self, graph_id: GraphID) -> None:
Fill the 'Operation' part of the info table.
SLOT(str) for SIGNAL(_graph._signals.component_selected)
Takes in an operator-id, first clears the 'Operator' part of the info
table and then fill in the table with new values from the operator
associated with *graph_id*.
self._info_table_fill_component(graph_id)
@Slot()
def info_table_update_schedule(self) -> None:
Update the 'Schedule' part of the info table.
SLOT() for SIGNAL(_graph._signals.schedule_time_changed)
Updates the 'Schedule' part of the info table.
"""
self.info_table.item(1, 1).setText(str(self.schedule.schedule_time))
@Slot(QRectF)
def shrink_scene_to_min_size(self, rect: QRectF) -> None:
SLOT(QRectF) for SIGNAL(_scene.sceneRectChanged)
Takes in a QRectF (unused) and shrink the scene bounding rectangle to
its minimum size, when the bounding rectangle signals a change in
geometry.
"""
self._scene.setSceneRect(self._scene.itemsBoundingRect())
##########
# Events #
##########
"""
EVENT: Replaces QMainWindow default closeEvent(QCloseEvent) event. Takes
in a QCloseEvent and display an exit dialog, depending on
'hide_exit_dialog' in settings.
"""
hide_dialog = settings.value("scheduler/hide_exit_dialog", True, bool)
ret = QMessageBox.StandardButton.Yes
if not hide_dialog:
box = QMessageBox(self)
box.setWindowTitle(self.tr("Confirm Exit"))
box.setText(
"<h3>"
+ self.tr("Confirm Exit")
+ "</h3><p><br>"
+ self.tr("Are you sure you want to exit?")
+ " <br></p>"
)
box.setIcon(QMessageBox.Question)
box.setStandardButtons(QMessageBox.Yes | QMessageBox.No)
buttons: list[QAbstractButton] = box.buttons()
buttons[0].setText(self.tr("&Exit"))
buttons[1].setText(self.tr("&Cancel"))
checkbox = QCheckBox(self.tr("Do not ask again"))
box.setCheckBox(checkbox)
ret = box.exec_()
if ret == QMessageBox.StandardButton.Yes:
if not hide_dialog:
settings.setValue("scheduler/hide_exit_dialog", checkbox.isChecked())
log.info(f"Exit: {os.path.basename(__file__)}")
if self._ports_accesses_for_storage:
self._ports_accesses_for_storage.close()
if self._execution_time_for_variables:
self._execution_time_for_variables.close()
for dialog in self._execution_time_plot_dialogs.values():
if dialog:
dialog.close()
def _open_about_window(self, event=None):
self.about_page = AboutWindow(self)
self.about_page.show()
###########################
# Helper member functions #
###########################
def _print_button_pressed(self, func_name: str) -> None:
alert.exec_()
def open(self, schedule: Schedule) -> None:
"""Take a Schedule and create a SchedulerItem object."""
self.close_schedule()
self._schedule = deepcopy(schedule)
self.menu_save.setEnabled(True)
self.menu_save_as.setEnabled(True)
self._scene.addItem(self._graph)
self._graph.installSceneEventFilters(self._graph.event_items)
self._graph._signals.component_selected.connect(
self.info_table_update_component
)
self._graph._signals.component_moved.connect(self.info_table_update_component)
self._graph._signals.component_moved.connect(self._schedule_changed)
self._graph._signals.schedule_time_changed.connect(
self.info_table_update_schedule
)
self._graph._signals.schedule_time_changed.connect(self._schedule_changed)
self._graph._signals.redraw_all.connect(self._redraw_all)
self._graph._signals.reopen.connect(self._reopen_schedule)
self._graph._signals.execution_time_plot.connect(self._execution_time_plot)
self.actionPreferences.setEnabled(True)
self.load_preferences()
self.action_view_variables.setEnabled(True)
self.action_view_port_accesses.setEnabled(True)
self.update_statusbar(self.tr("Schedule loaded successfully"))
def _redraw_all(self) -> None:
self._graph._redraw_all()
@Slot()
def _reopen_schedule(self) -> None:
self.open(self._schedule)
"""
Write *msg* to the statusbar with temporarily policy.
Parameters
----------
msg : str
The message to write.
self.statusbar.showMessage(msg)
def _write_settings(self) -> None:
"""Write settings from MainWindow to Settings."""
settings = QSettings()
settings.setValue(
"scheduler/maximized", self.isMaximized()
) # window: maximized, in X11 - always False
settings.setValue("scheduler/pos", self.pos()) # window: pos
settings.setValue("scheduler/size", self.size()) # window: size
settings.setValue("scheduler/menu/node_info", self.menu_node_info.isChecked())
settings.setValue("scheduler/splitter/state", self.splitter.saveState())
settings.setValue("scheduler/splitter/pos", self.splitter.sizes()[1])
settings.beginGroup("scheduler/preferences")
settings.setValue("font", Font.current_font.toString())
settings.setValue("fontSize", Font.size)
settings.setValue("fontColor", Font.color)
settings.setValue("fontBold", Font.current_font.bold())
settings.setValue("fontItalic", Font.current_font.italic())
settings.setValue("fontChanged", Font.changed)
settings.setValue(Signal_Color.name, Signal_Color.current_color.name())
settings.setValue(Active_Color.name, Active_Color.current_color.name())
settings.setValue(
Signal_Warning_Color.name, Signal_Warning_Color.current_color.name()
)
settings.setValue(
Execution_Time_Color.name, Execution_Time_Color.current_color.name()
)
settings.setValue(f"{Signal_Color.name}_changed", Signal_Color.changed)
settings.setValue(f"{Active_Color.name}_changed", Active_Color.changed)
settings.setValue(
f"{Signal_Warning_Color.name}_changed", Signal_Warning_Color.changed
)
self.Save_colortype()
settings.sync()
log.debug(f"Settings written to '{settings.fileName()}'.")
log.warning("Settings cant be saved to file, read-only.")
def _read_settings(self) -> None:
"""Read settings from Settings to MainWindow."""
if settings.value("scheduler/maximized", defaultValue=False, type=bool):
self.move(settings.value("scheduler/pos", self.pos()))
self.resize(settings.value("scheduler/size", self.size()))
self.restoreState(settings.value("scheduler/state", QByteArray()))
settings.value("scheduler/menu/node_info", True, bool)
settings.value("scheduler/splitter/state", QByteArray())
self._splitter_pos = settings.value("scheduler/splitter/pos", 200, int)
settings.value("scheduler/hide_exit_dialog", False, bool)
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
settings.beginGroup("scheduler/preferences")
Font.current_font = QFont(
settings.value("font", defaultValue=Font.DEFAULT.toString(), type=str)
)
Font.size = settings.value(
"fontSize", defaultValue=Font.DEFAULT.pointSizeF(), type=int
)
Font.color = QColor(
settings.value("fontColor", defaultValue=Font.DEFAULT_COLOR, type=str)
)
Font.bold = settings.value(
"fontBold", defaultValue=Font.DEFAULT.bold(), type=bool
)
Font.italic = settings.value(
"fontItalic", defaultValue=Font.DEFAULT.italic(), type=bool
)
Font.changed = settings.value("fontChanged", Font.changed, bool)
Signal_Color.current_color = QColor(
settings.value(
"Signal Color", defaultValue=Signal_Color.DEFAULT.name(), type=str
)
)
Active_Color.current_color = QColor(
settings.value(
"Active Color", defaultValue=Active_Color.DEFAULT.name(), type=str
)
)
Signal_Warning_Color.current_color = QColor(
settings.value(
"Warning Color",
defaultValue=Signal_Warning_Color.DEFAULT.name(),
type=str,
)
)
Latency_Color.current_color = QColor(
settings.value(
"Latency Color", defaultValue=Latency_Color.DEFAULT.name(), type=str
)
)
Execution_Time_Color.current_color = QColor(
settings.value(
"Execution Time Color",
defaultValue=Execution_Time_Color.DEFAULT.name(),
type=str,
)
)
Signal_Color.changed = settings.value(
f"{Signal_Color.name}_changed", False, bool
)
Active_Color.changed = settings.value(
f"{Active_Color.name}_changed", False, bool
)
Signal_Warning_Color.changed = settings.value(
f"{Signal_Warning_Color.name}_changed",
False,
bool,
)
Latency_Color.changed = settings.value(
f"{Latency_Color.name}_changed", False, bool
)
Execution_Time_Color.changed = settings.value(
f"{Execution_Time_Color.name}_changed",
False,
bool,
)
self._color_changed_perType = settings.value(
"_color_changed_perType", False, bool
)
settings.endGroup()
settings.sync()
log.debug(f"Settings read from '{settings.fileName()}'.")
def info_table_fill_schedule(self, schedule: Schedule) -> None:
Fill the 'Schedule' part of the info table.
Parameters
----------
schedule : Schedule
The Schedule to get information from.
self.info_table.insertRow(1)
self.info_table.insertRow(1)
self.info_table.setItem(1, 0, QTableWidgetItem("Schedule Time"))
self.info_table.setItem(2, 0, QTableWidgetItem("Cyclic"))
self.info_table.setItem(1, 1, QTableWidgetItem(str(schedule.schedule_time)))
self.info_table.setItem(2, 1, QTableWidgetItem(str(schedule.cyclic)))
def _info_table_fill_component(self, graph_id: GraphID) -> None:
Fill the 'Operator' part of the info table.
Parameters
----------
graph_id : GraphID
The GraphID of the operator to get information from.
if self.schedule is None:
return
op: GraphComponent = cast(
GraphComponent, self.schedule.sfg.find_by_id(graph_id)
self.info_table.setItem(si, 0, QTableWidgetItem("ID"))
self.info_table.setItem(si, 1, QTableWidgetItem(str(op.graph_id)))
si += 1
if op.name:
self.info_table.insertRow(si)
self.info_table.setItem(si, 0, QTableWidgetItem("Name"))
self.info_table.setItem(si, 1, QTableWidgetItem(str(op.name)))
si += 1
for key, value in op.params.items():
self.info_table.insertRow(si)
self.info_table.setItem(si, 0, QTableWidgetItem(key))
self.info_table.setItem(si, 1, QTableWidgetItem(str(value)))
si += 1
self.info_table.insertRow(si)
self.info_table.setItem(si, 0, QTableWidgetItem("Forward slack"))
self.info_table.setItem(
si, 1, QTableWidgetItem(str(self.schedule.forward_slack(graph_id)))
)
si += 1
self.info_table.insertRow(si)
self.info_table.setItem(si, 0, QTableWidgetItem("Backward slack"))
self.info_table.setItem(
si,
1,
QTableWidgetItem(str(self.schedule.backward_slack(graph_id))),
)
si += 1
self.info_table_clear_component()
self.info_table_clear_schedule()
def info_table_clear_schedule(self) -> None:
"""Clear the schedule part of the info table."""
row = self.info_table.findItems("Operator", Qt.MatchFlag.MatchExactly)
for _ in range(1, row):
self.info_table.removeRow(1)
log.error("'Operator' not found in info table. It may have been renamed.")
"""Exit application."""
log.info("Exiting the application.")
QApplication.quit()
"""Clear the component part of the info table."""
row = self.info_table.findItems("Operator", Qt.MatchFlag.MatchExactly)
if row:
row = row[0].row()
for _ in range(self.info_table.rowCount() - row + 1):
log.error("'Operator' not found in info table. It may have been renamed.")
def _create_recent_file_actions_and_menus(self):
for i in range(self._max_recent_files):
recent_file_action = QAction(self.menu_Recent_Schedule)
recent_file_action.setVisible(False)
recent_file_action.triggered.connect(
lambda b=0, x=recent_file_action: self._open_recent_file(x)
self._recent_files_actions.append(recent_file_action)
self.menu_Recent_Schedule.addAction(recent_file_action)
def _update_operation_types(self):
self.menu_view_execution_times.setEnabled(True)
for action in self.menu_view_execution_times.actions():
self.menu_view_execution_times.removeAction(action)
for type_name in self._schedule.get_used_type_names():
type_action = QAction(self.menu_view_execution_times)
type_action.setText(type_name)
type_action.triggered.connect(
lambda b=0, x=type_name: self._show_execution_times_for_type(x)
)
self.menu_view_execution_times.addAction(type_action)
def Preferences_Dialog_clicked(self):
"""Open the Preferences dialog to customize fonts, colors, and settings"""
dialog = QDialog()
dialog.setWindowTitle("Preferences")
layout = QVBoxLayout()
layout.setSpacing(15)
# Add label for the dialog
label = QLabel("Personalize Your Fonts and Colors")
layout.addWidget(label)
groupbox = QGroupBox()
Hlayout = QHBoxLayout()
label = QLabel("Color Settings:")