diff --git a/frontend/taipy-gui/.eslintrc.js b/frontend/taipy-gui/.eslintrc.js index 8c9af50c4a..30b459d814 100644 --- a/frontend/taipy-gui/.eslintrc.js +++ b/frontend/taipy-gui/.eslintrc.js @@ -11,36 +11,33 @@ * specific language governing permissions and limitations under the License. */ -module.exports = { - parser: '@typescript-eslint/parser', // Specifies the ESLint parser - extends: [ - 'plugin:react/recommended', // Uses the recommended rules from @eslint-plugin-react - 'plugin:@typescript-eslint/recommended', // Uses the recommended rules from @typescript-eslint/eslint-plugin +module.exports = { + parser: "@typescript-eslint/parser", // Specifies the ESLint parser + extends: [ + "plugin:react/recommended", // Uses the recommended rules from @eslint-plugin-react + "plugin:@typescript-eslint/recommended", // Uses the recommended rules from @typescript-eslint/eslint-plugin +// "plugin:react/jsx-runtime", //using the new JSX transform from React 17 ], - plugins: [ - "@typescript-eslint", - "react-hooks", - "eslint-plugin-tsdoc" - ], - parserOptions: { - ecmaVersion: 2018, // Allows for the parsing of modern ECMAScript features - sourceType: 'module', // Allows for the use of imports - ecmaFeatures: { - jsx: true, // Allows for the parsing of JSX - }, + plugins: ["@typescript-eslint", "react-hooks", "eslint-plugin-tsdoc"], + parserOptions: { + ecmaVersion: 2018, // Allows for the parsing of modern ECMAScript features + sourceType: "module", // Allows for the use of imports + ecmaFeatures: { + jsx: true, // Allows for the parsing of JSX + }, }, - rules: { - // Place to specify ESLint rules. Can be used to overwrite rules specified from the extended configs - "@typescript-eslint/explicit-function-return-type": "off", - "@typescript-eslint/explicit-module-boundary-types": "off", - "@typescript-eslint/no-unused-expressions": "off", // allows a && b() - "react-hooks/rules-of-hooks": "error", // Checks rules of Hooks - "react-hooks/exhaustive-deps": "error", // Checks effect dependencies - "tsdoc/syntax": "off", // "warn" to check tsdoc syntax + rules: { + // Place to specify ESLint rules. Can be used to overwrite rules specified from the extended configs + "@typescript-eslint/explicit-function-return-type": "off", + "@typescript-eslint/explicit-module-boundary-types": "off", + "@typescript-eslint/no-unused-expressions": "off", // allows a && b() + "react-hooks/rules-of-hooks": "error", // Checks rules of Hooks + "react-hooks/exhaustive-deps": "error", // Checks effect dependencies + "tsdoc/syntax": "off", // "warn" to check tsdoc syntax }, - settings: { - react: { - version: 'detect', // Tells eslint-plugin-react to automatically detect the version of React to use - }, + settings: { + react: { + version: "detect", // Tells eslint-plugin-react to automatically detect the version of React to use + }, }, - }; +}; diff --git a/frontend/taipy-gui/src/components/Taipy/AutoLoadingTable.tsx b/frontend/taipy-gui/src/components/Taipy/AutoLoadingTable.tsx index a7e8a11a59..25053e30dd 100644 --- a/frontend/taipy-gui/src/components/Taipy/AutoLoadingTable.tsx +++ b/frontend/taipy-gui/src/components/Taipy/AutoLoadingTable.tsx @@ -604,7 +604,7 @@ const AutoLoadingTable = (props: TaipyTableProps) => { {columns[col].dfid === EDIT_COL ? ( [ diff --git a/frontend/taipy-gui/src/components/Taipy/PaginatedTable.tsx b/frontend/taipy-gui/src/components/Taipy/PaginatedTable.tsx index e93b4798c6..2f5a750a8c 100644 --- a/frontend/taipy-gui/src/components/Taipy/PaginatedTable.tsx +++ b/frontend/taipy-gui/src/components/Taipy/PaginatedTable.tsx @@ -12,16 +12,24 @@ */ import React, { - useState, - useEffect, - useCallback, - useRef, - useMemo, - CSSProperties, ChangeEvent, + CSSProperties, MouseEvent, + useCallback, + useEffect, + useMemo, + useRef, + useState, } from "react"; + +import AddIcon from "@mui/icons-material/Add"; +import DataSaverOff from "@mui/icons-material/DataSaverOff"; +import DataSaverOn from "@mui/icons-material/DataSaverOn"; +import Download from "@mui/icons-material/Download"; import Box from "@mui/material/Box"; +import IconButton from "@mui/material/IconButton"; +import Paper from "@mui/material/Paper"; +import Skeleton from "@mui/material/Skeleton"; import Table from "@mui/material/Table"; import TableBody from "@mui/material/TableBody"; import TableCell from "@mui/material/TableCell"; @@ -30,62 +38,55 @@ import TableHead from "@mui/material/TableHead"; import TablePagination from "@mui/material/TablePagination"; import TableRow from "@mui/material/TableRow"; import TableSortLabel from "@mui/material/TableSortLabel"; -import Paper from "@mui/material/Paper"; -import Skeleton from "@mui/material/Skeleton"; -import Typography from "@mui/material/Typography"; import Tooltip from "@mui/material/Tooltip"; +import Typography from "@mui/material/Typography"; import { visuallyHidden } from "@mui/utils"; -import IconButton from "@mui/material/IconButton"; -import AddIcon from "@mui/icons-material/Add"; -import DataSaverOn from "@mui/icons-material/DataSaverOn"; -import DataSaverOff from "@mui/icons-material/DataSaverOff"; -import Download from "@mui/icons-material/Download"; import { createRequestTableUpdateAction, createSendActionNameAction } from "../../context/taipyReducers"; +import { emptyArray } from "../../utils"; +import { + useClassNames, + useDispatch, + useDispatchRequestUpdateOnFirstRender, + useDynamicJsonProperty, + useDynamicProperty, + useFormatConfig, + useModule, +} from "../../utils/hooks"; +import TableFilter from "./TableFilter"; import { addActionColumn, baseBoxSx, + ColumnDesc, + DEFAULT_SIZE, defaultColumns, - EditableCell, + DownloadAction, EDIT_COL, + EditableCell, + FilterDesc, getClassName, + getFormatFn, + getPageKey, + getRowIndex, getSortByIndex, + getTooltip, headBoxSx, - ROW_CLASS_NAME, + iconInRowSx, OnCellValidation, + OnRowClick, OnRowDeletion, + OnRowSelection, Order, PageSizeOptionsType, paperSx, + ROW_CLASS_NAME, RowType, RowValue, tableSx, TaipyPaginatedTableProps, - ColumnDesc, - iconInRowSx, - DEFAULT_SIZE, - OnRowSelection, - getRowIndex, - getTooltip, - OnRowClick, - DownloadAction, - getFormatFn, - getPageKey, - FilterDesc, } from "./tableUtils"; -import { - useClassNames, - useDispatch, - useDispatchRequestUpdateOnFirstRender, - useDynamicJsonProperty, - useDynamicProperty, - useFormatConfig, - useModule, -} from "../../utils/hooks"; -import TableFilter from "./TableFilter"; -import { getSuffixedClassNames, getUpdateVar } from "./utils"; -import { emptyArray } from "../../utils"; import { getComponentClassName } from "./TaipyStyle"; +import { getSuffixedClassNames, getUpdateVar } from "./utils"; const loadingStyle: CSSProperties = { width: "100%", height: "3em", textAlign: "right", verticalAlign: "center" }; const skeletonSx = { width: "100%", height: "3em" }; @@ -136,98 +137,99 @@ const PaginatedTable = (props: TaipyPaginatedTableProps) => { const hover = useDynamicProperty(props.hoverText, props.defaultHoverText, undefined); const baseColumns = useDynamicJsonProperty(props.columns, props.defaultColumns, defaultColumns); - const [colsOrder, columns, cellClassNames, tooltips, formats, handleNan, filter, partialEditable, nbWidth] = useMemo(() => { - let hNan = !!props.nanValue; - if (baseColumns) { - try { - let filter = false; - let partialEditable = editable; - const newCols: Record = {}; - Object.entries(baseColumns).forEach(([cId, cDesc]) => { - const nDesc = (newCols[cId] = { ...cDesc }); - if (typeof nDesc.filter != "boolean") { - nDesc.filter = !!props.filter; - } - filter = filter || nDesc.filter; - if (typeof nDesc.notEditable == "boolean") { - partialEditable = partialEditable || !nDesc.notEditable; - } else { - nDesc.notEditable = !editable; - } - if (nDesc.tooltip === undefined) { - nDesc.tooltip = props.tooltip; - } - }); - addActionColumn( - (active && partialEditable && (onAdd || onDelete) ? 1 : 0) + - (active && filter ? 1 : 0) + - (active && downloadable ? 1 : 0), - newCols - ); - const colsOrder = Object.keys(newCols).sort(getSortByIndex(newCols)); - let nbWidth = 0; - const functions = colsOrder.reduce>>((pv, col) => { - if (newCols[col].className) { - pv.classNames = pv.classNames || {}; - pv.classNames[newCols[col].dfid] = newCols[col].className; - } - hNan = hNan || !!newCols[col].nanValue; - if (newCols[col].tooltip) { - pv.tooltips = pv.tooltips || {}; - pv.tooltips[newCols[col].dfid] = newCols[col].tooltip; - } - if (newCols[col].formatFn) { - pv.formats = pv.formats || {}; - pv.formats[newCols[col].dfid] = newCols[col].formatFn; + const [colsOrder, columns, cellClassNames, tooltips, formats, handleNan, filter, partialEditable, nbWidth] = + useMemo(() => { + let hNan = !!props.nanValue; + if (baseColumns) { + try { + let filter = false; + let partialEditable = editable; + const newCols: Record = {}; + Object.entries(baseColumns).forEach(([cId, cDesc]) => { + const nDesc = (newCols[cId] = { ...cDesc }); + if (typeof nDesc.filter != "boolean") { + nDesc.filter = !!props.filter; + } + filter = filter || nDesc.filter; + if (typeof nDesc.notEditable == "boolean") { + partialEditable = partialEditable || !nDesc.notEditable; + } else { + nDesc.notEditable = !editable; + } + if (nDesc.tooltip === undefined) { + nDesc.tooltip = props.tooltip; + } + }); + addActionColumn( + (active && partialEditable && (onAdd || onDelete) ? 1 : 0) + + (active && filter ? 1 : 0) + + (active && downloadable ? 1 : 0), + newCols + ); + const colsOrder = Object.keys(newCols).sort(getSortByIndex(newCols)); + let nbWidth = 0; + const functions = colsOrder.reduce>>((pv, col) => { + if (newCols[col].className) { + pv.classNames = pv.classNames || {}; + pv.classNames[newCols[col].dfid] = newCols[col].className; + } + hNan = hNan || !!newCols[col].nanValue; + if (newCols[col].tooltip) { + pv.tooltips = pv.tooltips || {}; + pv.tooltips[newCols[col].dfid] = newCols[col].tooltip; + } + if (newCols[col].formatFn) { + pv.formats = pv.formats || {}; + pv.formats[newCols[col].dfid] = newCols[col].formatFn; + } + if (newCols[col].width !== undefined) { + nbWidth++; + } + return pv; + }, {}); + nbWidth = nbWidth ? colsOrder.length - nbWidth : 0; + if (props.rowClassName) { + functions.classNames = functions.classNames || {}; + functions.classNames[ROW_CLASS_NAME] = props.rowClassName; } - if (newCols[col].width !== undefined) { - nbWidth++; - } - return pv; - }, {}); - nbWidth = colsOrder.length - nbWidth; - if (props.rowClassName) { - functions.classNames = functions.classNames || {}; - functions.classNames[ROW_CLASS_NAME] = props.rowClassName; + return [ + colsOrder, + newCols, + functions.classNames, + functions.tooltips, + functions.formats, + hNan, + filter, + partialEditable, + nbWidth, + ]; + } catch (e) { + console.info("PaginatedTable.columns: ", (e as Error).message || e); } - return [ - colsOrder, - newCols, - functions.classNames, - functions.tooltips, - functions.formats, - hNan, - filter, - partialEditable, - nbWidth, - ]; - } catch (e) { - console.info("PaginatedTable.columns: ", (e as Error).message || e); } - } - return [ - [] as string[], - {} as Record, - {} as Record, - {} as Record, - {} as Record, - hNan, - false, - false, - 0, - ]; - }, [ - active, - editable, - onAdd, - onDelete, - baseColumns, - props.rowClassName, - props.tooltip, - props.nanValue, - props.filter, - downloadable, - ]); + return [ + [] as string[], + {} as Record, + {} as Record, + {} as Record, + {} as Record, + hNan, + false, + false, + 0, + ]; + }, [ + active, + editable, + onAdd, + onDelete, + baseColumns, + props.rowClassName, + props.tooltip, + props.nanValue, + props.filter, + downloadable, + ]); useDispatchRequestUpdateOnFirstRender(dispatch, id, module, updateVars); @@ -256,7 +258,18 @@ const PaginatedTable = (props: TaipyPaginatedTableProps) => { const endIndex = showAll ? -1 : startIndex + rowsPerPage - 1; const cols = colsOrder.map((col) => columns[col].dfid).filter((c) => c != EDIT_COL); const afs = appliedFilters.filter((fd) => Object.values(columns).some((cd) => cd.dfid === fd.col)); - pageKey.current = getPageKey(columns, `${startIndex}-${endIndex}`, cols, orderBy, order, afs, aggregates, cellClassNames, tooltips, formats); + pageKey.current = getPageKey( + columns, + `${startIndex}-${endIndex}`, + cols, + orderBy, + order, + afs, + aggregates, + cellClassNames, + tooltips, + formats + ); if (refresh || !props.data || props.data[pageKey.current] === undefined) { setLoading(true); const applies = aggregates.length @@ -491,7 +504,13 @@ const PaginatedTable = (props: TaipyPaginatedTableProps) => { const boxSx = useMemo(() => ({ ...baseBoxSx, width: width }), [width]); return ( - + @@ -506,7 +525,7 @@ const PaginatedTable = (props: TaipyPaginatedTableProps) => { columns[col].width ? { width: columns[col].width } : nbWidth - ? { width: `${100 / nbWidth}%`, maxWidth: 0 } + ? { minWidth: `${100 / nbWidth}%` } : undefined } > diff --git a/taipy/common/config/__init__.py b/taipy/common/config/__init__.py index 8f4b760447..79efd4dec7 100644 --- a/taipy/common/config/__init__.py +++ b/taipy/common/config/__init__.py @@ -9,7 +9,7 @@ # an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the # specific language governing permissions and limitations under the License. -""" The `taipy.common.config` package provides features to configure a Taipy application. +"""The `taipy.common.config` package provides features to configure a Taipy application. Its main class is the `Config^` singleton. It exposes various static methods and attributes to configure the Taipy application and retrieve the configuration values. @@ -45,6 +45,8 @@ """ +import os +from inspect import signature from typing import List from ._init import Config @@ -55,40 +57,55 @@ from .unique_section import UniqueSection -def _config_doc(func): - def func_with_doc(section, attr_name, default, configuration_methods, add_to_unconflicted_sections=False): - import os - - if os.environ.get("GENERATING_TAIPY_DOC", None) and os.environ["GENERATING_TAIPY_DOC"] == "true": - with (open("config_doc.txt", "a") as f): - from inspect import signature - - # Add the documentation for configure methods - for exposed_configuration_method, configuration_method in configuration_methods: - annotation = " @staticmethod\n" - sign = " def " + exposed_configuration_method + str(signature(configuration_method)) + ":\n" - doc = ' """' + configuration_method.__doc__ + '"""\n' - content = " pass\n\n" - f.write(annotation + sign + doc + content) - - # Add the documentation for the attribute - annotation = ' @property\n' - sign = f" def {attr_name} (self) -> {section.__name__}:\n" - if issubclass(section, UniqueSection): - doc = f' """The configured {section.__name__} section."""\n' - elif issubclass(section, Section): - doc = f' """The configured {section.__name__} sections ."""\n' - else: - print(f" ERROR - Invalid section class: {section.__name__}") # noqa: T201 - return +def __write_method_to_doc(configuration_methods): + if os.environ.get("GENERATING_TAIPY_DOC", None) and os.environ["GENERATING_TAIPY_DOC"] == "true": + with open("config_doc.txt", "a") as f: + from inspect import signature + + # Add the documentation for configure methods + for exposed_configuration_method, configuration_method in configuration_methods: + annotation = " @staticmethod\n" + sign = " def " + exposed_configuration_method + str(signature(configuration_method)) + ":\n" + doc = ' """' + configuration_method.__doc__ + '"""\n' content = " pass\n\n" f.write(annotation + sign + doc + content) - return func(section, attr_name, default, configuration_methods, add_to_unconflicted_sections) + + +def __write_section_to_doc(section, attr_name): + if os.environ.get("GENERATING_TAIPY_DOC", None) and os.environ["GENERATING_TAIPY_DOC"] == "true": + with open("config_doc.txt", "a") as f: + # Add the documentation for the attribute + annotation = " @property\n" + sign = f" def {attr_name} (self) -> {section.__name__}:\n" + if issubclass(section, UniqueSection): + doc = f' """The configured {section.__name__} section."""\n' + elif issubclass(section, Section): + doc = f' """The configured {section.__name__} sections ."""\n' + else: + print(f" ERROR - Invalid section class: {section.__name__}") # noqa: T201 + return + content = " pass\n\n" + f.write(annotation + sign + doc + content) + + +def _config_doc_for_section(func): + def func_with_doc(section, attribute_name, default, configuration_methods, add_to_unconflicted_sections=False): + __write_section_to_doc(section, attribute_name) + __write_method_to_doc(configuration_methods) + return func(section, attribute_name, default, configuration_methods, add_to_unconflicted_sections) return func_with_doc -@_config_doc +def _config_doc_for_method(func): + def func_with_doc(configuration_methods): + __write_method_to_doc(configuration_methods) + return func(configuration_methods) + + return func_with_doc + + +@_config_doc_for_section def _inject_section( section_clazz, attribute_name: str, @@ -110,3 +127,9 @@ def _inject_section( for exposed_configuration_method, configuration_method in configuration_methods: setattr(Config, exposed_configuration_method, configuration_method) + + +@_config_doc_for_method +def _inject_method(configuration_methods: List[tuple]): + for exposed_configuration_method, configuration_method in configuration_methods: + setattr(Config, exposed_configuration_method, configuration_method)