Source code for edifice.base_components.table_grid_view

from __future__ import annotations

import logging
from typing import TYPE_CHECKING

from edifice.qt import QT_VERSION

if QT_VERSION == "PyQt6" and not TYPE_CHECKING:
    from PyQt6.QtWidgets import QGridLayout, QWidget
else:
    from PySide6.QtWidgets import QGridLayout, QWidget

from .base_components import CommandType, Element, PropsDict, QtWidgetElement, _get_widget_children, _WidgetTree

logger = logging.getLogger("Edifice")


[docs] class TableGridRow(QtWidgetElement): """ Row Element of a :class:`TableGridView`. This Element’s parent must be a :class:`TableGridView`. This Element may have children of any type of :class:`Element`. This Element has no Qt underlying, and no props. """
[docs] def __init__(self): super().__init__()
def _qt_update_commands( self, widget_trees: dict[Element, _WidgetTree], # noqa: ARG002 newprops: PropsDict, # noqa: ARG002 ) -> list[CommandType]: # This element has no Qt underlying so it has nothing to do except store # the children of the row. # The _qt_update_commands for TableGridView will render the children. return []
[docs] class TableGridView(QtWidgetElement[QWidget]): """Table-style grid layout displays its children as aligned rows of columns. .. highlights:: - Underlying Qt Layout `QGridLayout <https://doc.qt.io/qtforpython-6/PySide6/QtWidgets/QGridLayout.html>`_ This component has similar behavior to an `HTML table <https://developer.mozilla.org/en-US/docs/Web/HTML/Element/table>`_. Each column will be the width of its widest child. Each row will be the height of its tallest child. .. rubric:: Props All **props** for :class:`QtWidgetElement` plus: Args: row_stretch: *n*:sup:`th` row stretch size in proportion to the *n*:sup:`th` :code:`int` in this list. See `setRowStretch <https://doc.qt.io/qtforpython-6/PySide6/QtWidgets/QGridLayout.html#PySide6.QtWidgets.QGridLayout.setRowStretch>`_ column_stretch: *n*:sup:`th` column stretch size in proportion to the *n*:sup:`th` :code:`int` in this list. See `setColumnStretch <https://doc.qt.io/qtforpython-6/PySide6/QtWidgets/QGridLayout.html#PySide6.QtWidgets.QGridLayout.setColumnStretch>`_ row_minheight: *n*:sup:`th` row minimum height is the *n*:sup:`th` :code:`int` in this list. See `setRowMinimumHeight <https://doc.qt.io/qtforpython-6/PySide6/QtWidgets/QGridLayout.html#PySide6.QtWidgets.QGridLayout.setRowMinimumHeight>`_ column_minwidth: *n*:sup:`th` column minimum width is the *n*:sup:`th` :code:`int` in this list. See `setColumnMinimumWidth <https://doc.qt.io/qtforpython-6/PySide6/QtWidgets/QGridLayout.html#PySide6.QtWidgets.QGridLayout.setColumnMinimumWidth>`_ .. rubric:: Usage The only type of child :class:`Element` allowed in a :class:`TableGridView` is :class:`TableGridRow`. Each :class:`TableGridRow` establishes a row in the table, and may have children of any type of :class:`Element`. .. code-block:: python with TableGridView(): with TableGridRow(): Label(text="row 0 column 0") Label(text="row 0 column 1") with TableGridRow(): Label(text="row 1 column 0") with VBoxView(): Label(text="row 1 column 1") If the :class:`TableGridRow` s are added and removed dynamically then it’s a good idea to :func:`Element.set_key` each row. """
[docs] def __init__( self, row_stretch: tuple[int, ...] = (), column_stretch: tuple[int, ...] = (), row_minheight: tuple[int, ...] = (), column_minwidth: tuple[int, ...] = (), **kwargs, ): super().__init__(**kwargs) self._register_props( { "row_stretch": row_stretch, "column_stretch": column_stretch, "row_minheight": row_minheight, "column_minwidth": column_minwidth, }, ) self.underlying = None self._old_children: dict[QtWidgetElement, tuple[int, int]] = {} """Like _LinearView._widget_children""" self._row_stretch = row_stretch self._column_stretch = column_stretch self._row_minheight = row_minheight self._column_minwidth = column_minwidth
def _initialize(self): self.underlying = QWidget() self.underlying_layout = QGridLayout() self.underlying.setObjectName(str(id(self))) self.underlying.setLayout(self.underlying_layout) self.underlying_layout.setContentsMargins(0, 0, 0, 0) self.underlying_layout.setSpacing(0) def _clear(self): while self.underlying_layout.takeAt(0) is not None: pass def _add_child(self, child_component: QtWidgetElement, row: int, column: int): # https://doc.qt.io/qtforpython-6/PySide6/QtWidgets/QGridLayout.html#PySide6.QtWidgets.QGridLayout.addWidget assert child_component.underlying is not None self.underlying_layout.addWidget(child_component.underlying, row, column) if len(self._row_stretch) > row: self.underlying_layout.setRowStretch(row, self._row_stretch[row]) if len(self._column_stretch) > column: self.underlying_layout.setColumnStretch(column, self._column_stretch[column]) if len(self._row_minheight) > row: self.underlying_layout.setRowMinimumHeight(row, self._row_minheight[row]) if len(self._column_minwidth) > column: self.underlying_layout.setColumnMinimumWidth(column, self._column_minwidth[column]) def _delete_child(self, child_component: QtWidgetElement, row: int, column: int): if self.underlying_layout is None: logger.warning("_delete_child No underlying_layout " + str(self)) else: if (layoutitem := self.underlying_layout.itemAtPosition(row, column)) is None: logger.warning("_delete_child itemAtPosition failed " + str((row, column)) + " " + str(self)) else: if (w := layoutitem.widget()) is None: logger.warning("_delete_child widget is None " + str((row, column)) + " " + str(self)) else: w.deleteLater() self.underlying_layout.removeItem(layoutitem) def _soft_delete_child(self, child_component: QtWidgetElement, row: int, column: int): if self.underlying_layout is None: logger.warning("_delete_child No underlying_layout " + str(self)) else: if (layoutitem := self.underlying_layout.itemAtPosition(row, column)) is None: logger.warning("_delete_child itemAtPosition failed " + str((row, column)) + " " + str(self)) else: self.underlying_layout.removeItem(layoutitem) def _qt_update_commands( self, widget_trees: dict[Element, _WidgetTree], newprops, ): if self.underlying is None: self._initialize() assert self.underlying is not None # The following is equivalent to _LinearView._recompute_children(). # # The direct children of this Element are TableGridRow, but # the TableGridRow doesn't have a Qt instantiation so we # want to treat the TableGridRow children as the children of # the TableGridView. children = _get_widget_children(widget_trees, self) new_children: dict[QtWidgetElement, tuple[int, int]] = {} for row, c in enumerate(children): children_of_row = _get_widget_children(widget_trees, c) for col, child in enumerate(children_of_row): new_children[child] = (row, col) old_deletions = self._old_children.items() - new_children.items() new_additions = new_children.items() - self._old_children.items() commands: list[CommandType] = [] for w, (row, column) in old_deletions: if w in new_children: # TODO this condition is never hit commands.append(CommandType(self._soft_delete_child, w, row, column)) else: commands.append(CommandType(self._delete_child, w, row, column)) for w, (row, column) in new_additions: commands.append(CommandType(self._add_child, w, row, column)) self._old_children = new_children if "row_stretch" in newprops: commands.append(CommandType(self._set_row_stretch, newprops.row_stretch)) if "column_stretch" in newprops: commands.append(CommandType(self._set_column_stretch, newprops.column_stretch)) if "row_minheight" in newprops: commands.append(CommandType(self._set_row_minheight, newprops.row_minheight)) if "column_minwidth" in newprops: commands.append(CommandType(self._set_column_minwidth, newprops.column_minwidth)) commands.extend( super()._qt_update_commands_super(widget_trees, newprops, self.underlying, self.underlying_layout), ) return commands def _set_row_stretch(self, row_stretch: tuple[int, ...]): self._row_stretch = row_stretch for i, stretch in enumerate(row_stretch[: self.underlying_layout.rowCount()]): self.underlying_layout.setRowStretch(i, stretch) def _set_column_stretch(self, column_stretch: tuple[int, ...]): self._column_stretch = column_stretch for i, stretch in enumerate(column_stretch[: self.underlying_layout.columnCount()]): self.underlying_layout.setColumnStretch(i, stretch) def _set_row_minheight(self, row_minheight: tuple[int, ...]): self._row_minheight = row_minheight for i, minheight in enumerate(row_minheight[: self.underlying_layout.rowCount()]): self.underlying_layout.setRowMinimumHeight(i, minheight) def _set_column_minwidth(self, column_minwidth: tuple[int, ...]): self._column_minwidth = column_minwidth for i, minwidth in enumerate(column_minwidth[: self.underlying_layout.columnCount()]): self.underlying_layout.setColumnMinimumWidth(i, minwidth)