diff --git a/tabulous/_qt/_toolbar/_toolbar_widget.py b/tabulous/_qt/_toolbar/_toolbar_widget.py index 332839ca..5fd9f975 100644 --- a/tabulous/_qt/_toolbar/_toolbar_widget.py +++ b/tabulous/_qt/_toolbar/_toolbar_widget.py @@ -104,7 +104,7 @@ def __init__(self, parent: _QtMainWidgetBase): self._child_widgets = weakref.WeakValueDictionary() self.addWidget(self._tab) - self.setMaximumHeight(120) + self.setMaximumHeight(84) self.initToolbar() @property @@ -153,7 +153,10 @@ def registerAction(self, tabname: str, f: Callable, icon: Union[str, Path]): self.addToolBar(tabname) toolbar = self._child_widgets[tabname] qicon = QColoredSVGIcon.fromfile(icon) - fn = lambda: f(self.viewer) + + def fn(): + return f(self.viewer) + fn.__doc__ = f.__doc__ toolbar.appendAction(fn, qicon) return @@ -168,7 +171,9 @@ def registerMenuAction(self, tabname: str, f: Callable, name: str | None = None) if name is None: name = f.__name__.replace("_", " ").capitalize() - fn = lambda: f(self.viewer) + def fn(): + return f(self.viewer) + fn.__doc__ = f.__doc__ toolbar.appendMenuAction(fn, name) return None @@ -182,69 +187,69 @@ def setToolButtonColor(self, color: str): def initToolbar(self): """Add tool buttons""" - self.registerAction("Home", cmds.file.open_table, ICON_DIR / "open_table.svg") - self.registerAction("Home", cmds.file.open_spreadsheet, ICON_DIR / "open_spreadsheet.svg") - self.registerAction("Home", cmds.file.save_table, ICON_DIR / "save_table.svg") + self.registerAction("Home", cmds.file.open_table, ICON_DIR / "open_table.svg") # noqa: E501 + self.registerAction("Home", cmds.file.open_spreadsheet, ICON_DIR / "open_spreadsheet.svg") # noqa: E501 + self.registerAction("Home", cmds.file.save_table, ICON_DIR / "save_table.svg") # noqa: E501 self.addSeparatorToChild("Home") - self.registerAction("Home", cmds.file.open_sample, ICON_DIR / "open_sample.svg") + self.registerAction("Home", cmds.file.open_sample, ICON_DIR / "open_sample.svg") # noqa: E501 self.addSeparatorToChild("Home") - self.registerAction("Home", cmds.window.toggle_console, ICON_DIR / "toggle_console.svg") - self.registerAction("Home", cmds.window.show_command_palette, ICON_DIR / "palette.svg") + self.registerAction("Home", cmds.window.toggle_console, ICON_DIR / "toggle_console.svg") # noqa: E501 + self.registerAction("Home", cmds.window.show_command_palette, ICON_DIR / "palette.svg") # noqa: E501 - self.registerAction("Edit", cmds.selection.copy_data_tab_separated, ICON_DIR / "copy.svg") - self.registerAction("Edit", cmds.selection.paste_data_tab_separated, ICON_DIR / "paste.svg") - self.registerAction("Edit", cmds.selection.cut_data, ICON_DIR / "cut.svg") + self.registerAction("Edit", cmds.selection.copy_data_tab_separated, ICON_DIR / "copy.svg") # noqa: E501 + self.registerAction("Edit", cmds.selection.paste_data_tab_separated, ICON_DIR / "paste.svg") # noqa: E501 + self.registerAction("Edit", cmds.selection.cut_data, ICON_DIR / "cut.svg") # noqa: E501 self.addSeparatorToChild("Edit") - self.registerAction("Edit", cmds.table.undo_table, ICON_DIR / "undo.svg") - self.registerAction("Edit", cmds.table.redo_table, ICON_DIR / "redo.svg") + self.registerAction("Edit", cmds.table.undo_table, ICON_DIR / "undo.svg") # noqa: E501 + self.registerAction("Edit", cmds.table.redo_table, ICON_DIR / "redo.svg") # noqa: E501 - self.registerAction("Table", cmds.table.copy_as_table, ICON_DIR / "copy_as_table.svg") - self.registerAction("Table", cmds.table.copy_as_spreadsheet, ICON_DIR / "copy_as_spreadsheet.svg") + self.registerAction("Table", cmds.table.copy_as_table, ICON_DIR / "copy_as_table.svg") # noqa: E501 + self.registerAction("Table", cmds.table.copy_as_spreadsheet, ICON_DIR / "copy_as_spreadsheet.svg") # noqa: E501 self.addSeparatorToChild("Table") - self.registerAction("Table", cmds.table.groupby, ICON_DIR / "groupby.svg") - self.registerAction("Table", cmds.table.switch_header, ICON_DIR / "switch_header.svg") + self.registerAction("Table", cmds.table.groupby, ICON_DIR / "groupby.svg") # noqa: E501 + self.registerAction("Table", cmds.table.switch_header, ICON_DIR / "switch_header.svg") # noqa: E501 self.registerAction("Table", cmds.table.pivot, ICON_DIR / "pivot.svg") self.registerAction("Table", cmds.table.melt, ICON_DIR / "melt.svg") self.addSeparatorToChild("Table") - self.registerAction("Table", cmds.table.random, ICON_DIR / "random.svg") - self.registerAction("Table", cmds.table.round, ICON_DIR / "round.svg") - - self.registerAction("Analyze", cmds.analysis.summarize_table, ICON_DIR / "summarize_table.svg") - self.registerAction("Analyze", cmds.analysis.show_eval_widget, ICON_DIR / "eval.svg") - self.registerAction("Analyze", cmds.table.show_finder_widget, ICON_DIR / "find_item.svg") - self.registerAction("Analyze", cmds.selection.sort_by_columns, ICON_DIR / "sort_table.svg") - self.registerAction("Analyze", cmds.analysis.show_filter_widget, ICON_DIR / "filter.svg") + self.registerAction("Table", cmds.table.random, ICON_DIR / "random.svg") # noqa: E501 + self.registerAction("Table", cmds.table.round, ICON_DIR / "round.svg") # noqa: E501 + + self.registerAction("Analyze", cmds.analysis.summarize_table, ICON_DIR / "summarize_table.svg") # noqa: E501 + self.registerAction("Analyze", cmds.analysis.show_eval_widget, ICON_DIR / "eval.svg") # noqa: E501 + self.registerAction("Analyze", cmds.table.show_finder_widget, ICON_DIR / "find_item.svg") # noqa: E501 + self.registerAction("Analyze", cmds.selection.sort_by_columns, ICON_DIR / "sort_table.svg") # noqa: E501 + self.registerAction("Analyze", cmds.analysis.show_filter_widget, ICON_DIR / "filter.svg") # noqa: E501 self.addSeparatorToChild("Analyze") - self.registerAction("Analyze", cmds.analysis.show_optimizer_widget, ICON_DIR / "optimize.svg") - self.registerAction("Analyze", cmds.analysis.show_stats_widget, ICON_DIR / "stats_test.svg") - self.registerAction("Analyze", cmds.analysis.show_sklearn_widget, ICON_DIR / "sklearn_analysis.svg") - - self.registerAction("View", cmds.view.set_popup_mode, ICON_DIR / "view_popup.svg") - self.registerAction("View", cmds.view.set_dual_h_mode, ICON_DIR / "view_dual_h.svg") - self.registerAction("View", cmds.view.set_dual_v_mode, ICON_DIR / "view_dual_v.svg") - self.registerAction("View", cmds.view.reset_view_mode, ICON_DIR / "view_reset.svg") + self.registerAction("Analyze", cmds.analysis.show_optimizer_widget, ICON_DIR / "optimize.svg") # noqa: E501 + self.registerAction("Analyze", cmds.analysis.show_stats_widget, ICON_DIR / "stats_test.svg") # noqa: E501 + self.registerAction("Analyze", cmds.analysis.show_sklearn_widget, ICON_DIR / "sklearn_analysis.svg") # noqa: E501 + + self.registerAction("View", cmds.view.set_popup_mode, ICON_DIR / "view_popup.svg") # noqa: E501 + self.registerAction("View", cmds.view.set_dual_h_mode, ICON_DIR / "view_dual_h.svg") # noqa: E501 + self.registerAction("View", cmds.view.set_dual_v_mode, ICON_DIR / "view_dual_v.svg") # noqa: E501 + self.registerAction("View", cmds.view.reset_view_mode, ICON_DIR / "view_reset.svg") # noqa: E501 self.addSeparatorToChild("View") - self.registerAction("View", cmds.tab.tile_tables, ICON_DIR / "tile.svg") - self.registerAction("View", cmds.tab.untile_table, ICON_DIR / "untile.svg") + self.registerAction("View", cmds.tab.tile_tables, ICON_DIR / "tile.svg") # noqa: E501 + self.registerAction("View", cmds.tab.untile_table, ICON_DIR / "untile.svg") # noqa: E501 self.addSeparatorToChild("View") - self.registerAction("View", cmds.table.switch_layout, ICON_DIR / "switch_layout.svg") + self.registerAction("View", cmds.table.switch_layout, ICON_DIR / "switch_layout.svg") # noqa: E501 self.registerAction("Plot", cmds.plot.plot, ICON_DIR / "plot.svg") - self.registerAction("Plot", cmds.plot.scatter, ICON_DIR / "scatter.svg") + self.registerAction("Plot", cmds.plot.scatter, ICON_DIR / "scatter.svg") # noqa: E501 self.registerAction("Plot", cmds.plot.hist, ICON_DIR / "hist.svg") self.registerMenuAction("Plot", cmds.plot.bar, name="Run plt.bar") - self.registerMenuAction("Plot", cmds.plot.errorbar, name="Run plt.errorbar") - self.registerMenuAction("Plot", cmds.plot.fill_between, name="Run plt.fill_between") - self.registerMenuAction("Plot", cmds.plot.fill_betweenx, name="Run plt.fill_betweenx") + self.registerMenuAction("Plot", cmds.plot.errorbar, name="Run plt.errorbar") # noqa: E501 + self.registerMenuAction("Plot", cmds.plot.fill_between, name="Run plt.fill_between") # noqa: E501 + self.registerMenuAction("Plot", cmds.plot.fill_betweenx, name="Run plt.fill_betweenx") # noqa: E501 self.addSeparatorToChild("Plot") - self.registerAction("Plot", cmds.plot.swarmplot, ICON_DIR / "swarmplot.svg") - self.registerAction("Plot", cmds.plot.barplot, ICON_DIR / "barplot.svg") - self.registerAction("Plot", cmds.plot.boxplot, ICON_DIR / "boxplot.svg") - self.registerMenuAction("Plot", cmds.plot.boxenplot, name="Run sns.boxenplot") - self.registerMenuAction("Plot", cmds.plot.stripplot, name="Run sns.stripplot") - self.registerMenuAction("Plot", cmds.plot.violinplot, name="Run sns.violinplot") + self.registerAction("Plot", cmds.plot.swarmplot, ICON_DIR / "swarmplot.svg") # noqa: E501 + self.registerAction("Plot", cmds.plot.barplot, ICON_DIR / "barplot.svg") # noqa: E501 + self.registerAction("Plot", cmds.plot.boxplot, ICON_DIR / "boxplot.svg") # noqa: E501 + self.registerMenuAction("Plot", cmds.plot.boxenplot, name="Run sns.boxenplot") # noqa: E501 + self.registerMenuAction("Plot", cmds.plot.stripplot, name="Run sns.stripplot") # noqa: E501 + self.registerMenuAction("Plot", cmds.plot.violinplot, name="Run sns.violinplot") # noqa: E501 self.addSeparatorToChild("Plot") - self.registerAction("Plot", cmds.plot.new_figure, ICON_DIR / "new_figure.svg") + self.registerAction("Plot", cmds.plot.new_figure, ICON_DIR / "new_figure.svg") # noqa: E501 return # fmt: on diff --git a/tabulous/_qt/_undo.py b/tabulous/_qt/_undo.py index 3d647dcc..110f5f69 100644 --- a/tabulous/_qt/_undo.py +++ b/tabulous/_qt/_undo.py @@ -3,6 +3,7 @@ from collections_undo import UndoManager, fmt from qtpy import QtWidgets as QtW, QtCore, QtGui from qtpy.QtCore import Qt +from qtpy.sip import isdeleted import numpy as np import pandas as pd @@ -45,8 +46,11 @@ def __init__(self, mgr: UndoManager): self.setMinimumHeight(160) def refresh(self, *_): - self._listview.model().updateUndoShape() - self._listview.viewport().update() + if isdeleted(self): + self._mgr.called.remove(self.refresh) + else: + self._listview.model().updateUndoShape() + self._listview.viewport().update() class QUndoStackModel(QtCore.QAbstractListModel): @@ -101,7 +105,7 @@ def __init__(self, maxsize=1e7) -> None: self._widget = None def widget(self) -> QUndoStackViewer: - if self._widget is None: + if self._widget is None or isdeleted(self._widget): self._widget = QUndoStackViewer(self) return self._widget diff --git a/tabulous/_text_formatter.py b/tabulous/_text_formatter.py index 541595b8..8f8086ee 100644 --- a/tabulous/_text_formatter.py +++ b/tabulous/_text_formatter.py @@ -226,11 +226,20 @@ def toFormatter(self, val=None) -> str: if val == self._ENUM.default: fmt = None elif val == self._ENUM.decimal: - fmt = f"{{:.{n}f}}" + + def fmt(x): + return f"{float(x):.{n}f}" + elif val == self._ENUM.exponential: - fmt = f"{{:.{n}g}}" + + def fmt(x): + return f"{float(x):.{n}g}" + elif val == self._ENUM.percent: - fmt = f"{{:.{n}%}}" + + def fmt(x): + return f"{float(x):.{n}%}" + else: raise RuntimeError() return fmt