-
Notifications
You must be signed in to change notification settings - Fork 5
feat(ui5): Add table best practices skill #78
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Open
FelixSchubert-001
wants to merge
3
commits into
UI5:main
Choose a base branch
from
FelixSchubert-001:feat/ui5-best-practices-tables
base: main
Could not load branches
Branch not found: {{ refName }}
Loading
Could not load tags
Nothing to show
Loading
Are you sure you want to change the base?
Some commits from the old base branch may be removed from the timeline,
and old review comments may become outdated.
+702
−0
Open
Changes from 2 commits
Commits
Show all changes
3 commits
Select commit
Hold shift + click to select a range
File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| Original file line number | Diff line number | Diff line change |
|---|---|---|
| @@ -0,0 +1,144 @@ | ||
| --- | ||
| name: ui5-best-practices-tables | ||
| description: | | ||
| UI5 table best practices and authoritative development guidelines. Use when the user asks to "create a table", "add a table", "which table should I use", "implement sap.m.Table", "implement sap.ui.table", "GridTable", "ResponsiveTable", "TreeTable", "SmartTable", "MDC table", "sap.ui.mdc.Table", "sap.ui.comp.smarttable", "table not showing data", "table binding", "table selection", "drag and drop table", "table personalization", "copy paste table", "table export", "table sticky header", "table growing", "table pop-in", "table performance", "table accessibility", "table items binding", or is writing any UI5 freestyle application that includes a table control. Covers: control selection matrix, mandatory rules, common error patterns, and per-control API guidance for sap.m.Table, sap.ui.table.Table, sap.ui.table.TreeTable, sap.ui.comp.smarttable.SmartTable, and sap.ui.mdc.Table. | ||
|
|
||
| Keywords: sap.m.Table, sap.ui.table, GridTable, ResponsiveTable, TreeTable, SmartTable, sap.ui.mdc.Table, table binding, rows aggregation, items aggregation, rowMode, growing, sticky, personalization, drag and drop, DragDropInfo, p13n.Engine, OData V4, OData V2, cell templates, ariaLabelledBy | ||
| --- | ||
|
|
||
| # UI5 Table Best Practices | ||
|
|
||
| Apply these guidelines whenever generating, reviewing, or troubleshooting UI5 table code in freestyle applications. | ||
|
|
||
| **UI5 version baseline:** SAPUI5 1.136+ LTS. All features mentioned are available from this version unless noted. | ||
|
|
||
| ## When to load each reference | ||
|
|
||
| | Trigger | Load | | ||
| |---|---| | ||
| | Working on or planning a `sap.m.Table` (ResponsiveTable) | [`references/sap-m-table.md`](references/sap-m-table.md) | | ||
| | Working on or planning a `sap.ui.table.Table` (GridTable) | [`references/grid-table.md`](references/grid-table.md) | | ||
| | Working on or planning a `sap.ui.table.TreeTable` | [`references/tree-table.md`](references/tree-table.md) | | ||
| | Working on or planning a `sap.ui.comp.smarttable.SmartTable` | [`references/smart-table.md`](references/smart-table.md) | | ||
| | Working on or planning a `sap.ui.mdc.Table` | [`references/mdc-table.md`](references/mdc-table.md) | | ||
| | Adding drag-and-drop to any table | [`references/drag-and-drop.md`](references/drag-and-drop.md) | | ||
| | Adding column personalization | [`references/personalization.md`](references/personalization.md) | | ||
| | Choosing cell templates, alignment, or data type binding | [`references/cell-templates.md`](references/cell-templates.md) | | ||
|
|
||
| Load before producing any output. Do not work from memory. | ||
|
|
||
| --- | ||
|
|
||
| ## Core Rules | ||
|
|
||
| ### Mandatory | ||
|
|
||
| - Choose the table type using the [Selection Matrix](#selection-matrix) before writing any code. | ||
| - Use the `rows` aggregation for `sap.ui.table.*` and `items` for `sap.m.Table`. Never swap them. | ||
| - Use `sap.m.p13n.Engine` for personalization. Never build custom personalization dialogs. | ||
| - Set `ariaLabelledBy` on every table, referencing the table title control. | ||
| - Align cell content by data type: numbers and dates right (`hAlign="End"`), text and links left. | ||
| - Use appropriate cell templates: `sap.m.Text` for display, `sap.m.ObjectNumber` for numbers, `sap.m.Link` for navigation. | ||
| - Request `$count=true` from the back end for `sap.ui.table.*` when a total count is required. | ||
| - Use the `rowMode` aggregation (not the deprecated `visibleRowCountMode` property) for `sap.ui.table.*` (UI5 1.119+). | ||
|
|
||
| ### Prohibitions | ||
|
|
||
| - Do not use global variables. Use `sap.ui.define` AMD modules or ES6 imports. | ||
| - Do not enable text wrapping in `sap.ui.table.*` cells — it breaks virtualization. | ||
| - Do not assume `sap.ui.export.Spreadsheet` is available. Detect the library before use. | ||
| - Do not use `sap.ui.table.Table` for mobile-first scenarios. Use `sap.m.Table`. | ||
| - Do not use `sap.m.Table` for datasets with 1000+ rows that require virtualization. | ||
| - Do not place multiple interactive elements in one `sap.ui.table.Table` cell. | ||
| - Do not return enum objects from formatters. Return string literals or primitive values. | ||
| - Do not use formatters for `ColumnListItem` `highlight`. Use direct data binding. | ||
| - Do not access models without checking availability — causes "Cannot read properties of undefined". | ||
| - Do not mix type namespaces: never use `sap.ui.model.odata.type.*` with a JSON model, or `sap.ui.model.type.*` with OData. | ||
|
|
||
| ### OData V4 policy | ||
|
|
||
| Prefer SAP Fiori elements building blocks over freestyle tables for OData V4. Use `sap.ui.mdc.Table` only when Fiori elements is out of scope. | ||
|
|
||
| --- | ||
|
|
||
| ## Selection Matrix | ||
|
|
||
| | Table type | Use when | Do not use when | | ||
| |---|---|---| | ||
| | `sap.m.Table` | Mobile/responsive, pop-in behavior, JSON models, fewer than 100 rows | 1000+ rows, virtualization required, desktop-only, cell selection needed | | ||
| | `sap.ui.table.Table` | Desktop, 1000+ rows, virtualization, fixed columns, dense data | Mobile-first, pop-in required, text wrapping, small datasets | | ||
| | `sap.ui.table.TreeTable` | Hierarchical data, expand/collapse, parent-child relationships | Flat data, mobile-first, grouping (not hierarchy) | | ||
| | `SmartTable` | OData V2, annotations, automatic columns, smart filtering | JSON-only, precise control required, OData V4 | | ||
| | `sap.ui.mdc.Table` | OData V4 freestyle (when Fiori elements is ruled out), delegate pattern | JSON-only, simple apps, OData V2 | | ||
|
|
||
| ### Dataset size guide | ||
|
|
||
| | Rows | Recommended table | Strategy | | ||
| |---|---|---| | ||
| | < 100 | `sap.m.Table` | Simple binding, `growing` optional | | ||
| | 100–1000 | `sap.ui.table.Table` | Virtualization, `threshold=100` | | ||
| | 1000+ | `sap.ui.table.Table` | Virtualization, `threshold=100–500`, `$count=true` when needed | | ||
|
|
||
| --- | ||
|
|
||
| ## Common Errors | ||
|
|
||
| | Symptom | Cause | Fix | | ||
| |---|---|---| | ||
| | No data displayed | Incorrect binding path or missing model | Verify `bindRows`/`bindItems` path and model attachment. | | ||
| | Rows not scrolling (`sap.ui.table.*`) | Count not requested | Set `$count=true` for OData when a total count is required. | | ||
| | Selection not working (`sap.ui.table.*`) | Plugin conflict | Do not call the table selection API when a selection plugin is applied; use the plugin API instead. | | ||
| | Text wrapping issues | Wrapping enabled in `sap.ui.table.*` | Use fixed-height content or switch to `sap.m.Table`. | | ||
| | Copy/paste not working | Plugin not attached or wrong namespace | Attach the correct plugin (see Drag & Drop section). | | ||
| | Personalization not persisting | Engine not configured | Verify `sap.m.p13n.Engine` registration. | | ||
| | `CopyProvider` error | `extractData` not defined | Implement `extractData` on the plugin. | | ||
| | Table not visible | Invalid container structure | Use a valid container (see Container Structures). | | ||
| | OData types on JSON model | Wrong type namespace | Match the type namespace to the model: `sap.ui.model.type.*` for JSON, `sap.ui.model.odata.type.*` for OData V2, `sap.ui.model.odata.v4.type.*` for OData V4. | | ||
| | Excel export fails silently | Library not loaded or invalid `extractData` | Detect the library, return a 2D array from `extractData`, ensure `dataSource` binding. | | ||
| | "No Data Available" | Model not set before binding | Set the model in `Component.init()` before router initialization. | | ||
|
|
||
| --- | ||
|
|
||
| ## Container Structures | ||
|
|
||
| ### Valid | ||
|
|
||
| | Structure | Use case | | ||
| |---|---| | ||
| | `View > Page > content > Table` | Standard page | | ||
| | `View > Page > content > Panel > Table` | Grouped content | | ||
| | `View > Page > content > IconTabBar > items > IconTabFilter > Table` | Tabs | | ||
| | `View > SplitApp > detailPages > Page > Table` | Master-detail | | ||
| | `View > Dialog > content > Table` | Modal | | ||
| | `View > Table` | Standalone | | ||
|
|
||
| ### Invalid (and why) | ||
|
|
||
| | Structure | Issue | | ||
| |---|---| | ||
| | `Page > content > VBox > Table` | `VBox` needs explicit height — table becomes invisible | | ||
| | `Page > VBox > Table` | `VBox` not in `content` — table not rendered | | ||
| | `Page > content > FlexBox > Table` | Sizing conflict — table collapses | | ||
| | `Page > content > ScrollContainer > Table` | Double scrolling — virtualization breaks | | ||
|
|
||
| --- | ||
|
|
||
| ## Performance & Accessibility | ||
|
|
||
| ### Anti-patterns to avoid | ||
|
|
||
| - Handcrafted personalization dialogs (use `sap.m.p13n.Engine`). | ||
| - Text wrapping in `sap.ui.table.Table` cells. | ||
| - Multiple interactive elements in one GridTable cell. | ||
| - Mixing OData V2 `SmartTable` with V4 services. | ||
| - Deep nesting in cell templates. | ||
| - Fixed `threshold` without load testing. | ||
| - Unconditional `$count=true` (request count only when needed). | ||
|
|
||
| ### Accessibility checklist | ||
|
|
||
| - Set `ariaLabelledBy` on every table, referencing the visible title. | ||
| - Use `sap.m.Text` (not raw text nodes) as cell templates. | ||
| - For `sap.ui.table.Table`: test keyboard navigation (Tab, arrow keys, Space/Enter for selection). | ||
| - Verify that the personalization dialog is keyboard-operable. | ||
| - Test with a screen reader (NVDA/JAWS on Windows, VoiceOver on macOS) before shipping. |
44 changes: 44 additions & 0 deletions
44
plugins/ui5/skills/ui5-best-practices-tables/references/cell-templates.md
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| Original file line number | Diff line number | Diff line change |
|---|---|---|
| @@ -0,0 +1,44 @@ | ||
| # Cell Templates & Alignment | ||
|
|
||
| ## Alignment by data type | ||
|
|
||
| | Data type | Alignment | Property | | ||
| |---|---|---| | ||
| | Text | Left | default | | ||
| | Numbers | Right | `hAlign="End"` | | ||
| | Dates | Right | `hAlign="End"` | | ||
| | Boolean | Left | default | | ||
| | Links | Left | default | | ||
|
|
||
| Set `hAlign` on the `Column` control, not on the cell template. | ||
|
|
||
| ## Cell template selection | ||
|
|
||
| | Content type | Template control | | ||
| |---|---| | ||
| | Plain text | `sap.m.Text` | | ||
| | Formatted number / amount | `sap.m.ObjectNumber` | | ||
| | Navigation | `sap.m.Link` | | ||
| | Status | `sap.m.ObjectStatus` | | ||
| | Icon | `sap.ui.core.Icon` | | ||
|
|
||
| ## Model type binding | ||
|
|
||
| NEVER mix type namespaces. Always match the type namespace to the model: | ||
|
|
||
| ```xml | ||
| <!-- JSON Model --> | ||
| <Text text="{path: 'price', type: 'sap.ui.model.type.Float'}"/> | ||
|
|
||
| <!-- OData V2 --> | ||
| <Text text="{path: 'Price', type: 'sap.ui.model.odata.type.Decimal'}"/> | ||
|
|
||
| <!-- OData V4 --> | ||
| <Text text="{path: 'Price', type: 'sap.ui.model.odata.v4.type.Decimal'}"/> | ||
| ``` | ||
|
|
||
| | Model | Type namespace | | ||
| |---|---| | ||
| | JSON | `sap.ui.model.type.*` | | ||
| | OData V2 | `sap.ui.model.odata.type.*` | | ||
| | OData V4 | `sap.ui.model.odata.v4.type.*` | |
50 changes: 50 additions & 0 deletions
50
plugins/ui5/skills/ui5-best-practices-tables/references/drag-and-drop.md
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| Original file line number | Diff line number | Diff line change |
|---|---|---|
| @@ -0,0 +1,50 @@ | ||
| # Drag & Drop | ||
|
|
||
| Configure drag-and-drop on the **table**, not on individual items or cells. | ||
|
|
||
| **❌ Wrong — DnD on the item:** | ||
| ```xml | ||
| <Table items="{/products}"> | ||
| <items> | ||
| <ColumnListItem> | ||
| <dragDropConfig> | ||
| <dnd:DragInfo sourceAggregation="items"/> | ||
| </dragDropConfig> | ||
| </ColumnListItem> | ||
| </items> | ||
| </Table> | ||
| ``` | ||
|
|
||
| **✅ Correct — reordering within the same table using `DragInfo` + `DropInfo` with matching `groupName`:** | ||
| ```xml | ||
| <Table items="{/products}"> | ||
| <items> | ||
| <ColumnListItem><cells>...</cells></ColumnListItem> | ||
| </items> | ||
| <dragDropConfig> | ||
| <dnd:DragInfo sourceAggregation="items" groupName="reorder"/> | ||
| <dnd:DropInfo targetAggregation="items" | ||
| groupName="reorder" | ||
| dropPosition="Between" | ||
| drop=".onDrop"/> | ||
| </dragDropConfig> | ||
| </Table> | ||
| ``` | ||
|
|
||
| **✅ Alternative — using `DragDropInfo` (no `groupName` needed for same-table reorder):** | ||
| ```xml | ||
| <Table items="{/products}"> | ||
| <dragDropConfig> | ||
| <dnd:DragDropInfo sourceAggregation="items" | ||
| targetAggregation="items" | ||
| dropPosition="Between" | ||
| drop=".onDrop"/> | ||
| </dragDropConfig> | ||
| </Table> | ||
| ``` | ||
|
|
||
| ## Key rules | ||
|
|
||
| - For reordering within the same table: use `DragDropInfo`, or set a matching `groupName` on both `DragInfo` and `DropInfo`. | ||
| - For drag between different tables: use matching `groupName` values on both tables. | ||
| - Update the bound model in the `drop` handler to reflect the new order. |
79 changes: 79 additions & 0 deletions
79
plugins/ui5/skills/ui5-best-practices-tables/references/grid-table.md
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| Original file line number | Diff line number | Diff line change |
|---|---|---|
| @@ -0,0 +1,79 @@ | ||
| # sap.ui.table.Table (GridTable) | ||
|
|
||
| API: https://ui5.sap.com/1.136.0/api/sap.ui.table.Table | ||
|
|
||
| ## Minimal complete example | ||
|
|
||
| ```xml | ||
| <mvc:View xmlns:mvc="sap.ui.core.mvc" xmlns="sap.m" xmlns:table="sap.ui.table" | ||
| controllerName="my.app.controller.Main"> | ||
| <Page title="Products"> | ||
| <table:Table id="gridTable" rows="{/products}" selectionMode="MultiToggle" | ||
| ariaLabelledBy="gridTableTitle" threshold="100"> | ||
| <table:extension> | ||
| <OverflowToolbar> | ||
| <Title id="gridTableTitle" text="Products" level="H2"/> | ||
| <ToolbarSpacer/> | ||
| <Button icon="sap-icon:action-settings" press=".onPersonalize"/> | ||
| </OverflowToolbar> | ||
| </table:extension> | ||
| <table:rowMode> | ||
| <table:rowmodes:Fixed rowCount="10"/> | ||
| </table:rowMode> | ||
| <table:columns> | ||
| <table:Column> | ||
| <Label text="Name"/> | ||
| <table:template><Text text="{name}" wrapping="false"/></table:template> | ||
| </table:Column> | ||
| <table:Column hAlign="End"> | ||
| <Label text="Price"/> | ||
| <table:template><ObjectNumber number="{price}" unit="{currency}"/></table:template> | ||
| </table:Column> | ||
| </table:columns> | ||
| </table:Table> | ||
| </Page> | ||
| </mvc:View> | ||
| ``` | ||
|
|
||
| ## Key properties | ||
|
|
||
| | Property | Type | Default | Since | Notes | | ||
| |---|---|---|---|---| | ||
| | `selectionBehavior` | SelectionBehavior | RowSelector | 1.0 | `Row`, `RowSelector`, `RowOnly`. | | ||
| | `columnHeaderVisible` | boolean | true | 1.0 | Show/hide column headers. | | ||
| | `showNoData` | boolean | true | 1.0 | Show "No data" text. | | ||
| | `noData` | string / Control | — | 1.0 | Custom no-data content. | | ||
| | `showOverlay` | boolean | false | 1.21 | Block interaction with overlay. | | ||
| | `threshold` | int | 100 | 1.0 | Prefetch buffer for virtualization. | | ||
|
|
||
| ## rowMode aggregation (UI5 1.119+) | ||
|
|
||
| Use the `rowMode` aggregation instead of the deprecated `visibleRowCountMode` property. | ||
|
|
||
| ```xml | ||
| <table:rowMode> | ||
| <table:rowmodes:Fixed rowCount="10"/> | ||
| </table:rowMode> | ||
| ``` | ||
|
|
||
| Available row modes: `Fixed`, `Auto`, `Interactive`. | ||
|
|
||
| ## Selection behavior | ||
|
|
||
| - `Row`: selection changed anywhere in the row, including the selector column. | ||
| - `RowSelector`: selection changed only via the selector column; row clicks do not select. | ||
| - `RowOnly`: selection changed only via row clicks; the selector column is hidden. | ||
|
|
||
| Do not call the table's selection API when a `SelectionPlugin` is attached. Use the plugin API instead. | ||
|
|
||
| ## No-data customization | ||
|
|
||
| ```xml | ||
| <table:Table> | ||
| <table:noData> | ||
| <IllustratedMessage illustrationType="NoData" title="No Products"> | ||
| <Button text="Add Product" press=".onAddProduct"/> | ||
| </IllustratedMessage> | ||
| </table:noData> | ||
| </table:Table> | ||
| ``` |
48 changes: 48 additions & 0 deletions
48
plugins/ui5/skills/ui5-best-practices-tables/references/mdc-table.md
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| Original file line number | Diff line number | Diff line change |
|---|---|---|
| @@ -0,0 +1,48 @@ | ||
| # sap.ui.mdc.Table | ||
|
|
||
| API: https://ui5.sap.com/1.136.0/api/sap.ui.mdc.Table | ||
|
|
||
| ## Delegate pattern | ||
|
|
||
| `sap.ui.mdc.Table` uses a delegate for metadata and data operations. Implement both `fetchProperties` and `updateBindingInfo`. | ||
|
|
||
| Minimal delegate: | ||
| ```javascript | ||
| sap.ui.define(["sap/ui/mdc/odata/v4/TableDelegate"], function(TableDelegate) { | ||
| const MyDelegate = Object.assign({}, TableDelegate); | ||
|
|
||
| MyDelegate.fetchProperties = function(oTable) { | ||
| return Promise.resolve([ | ||
| { | ||
| key: "name", label: "Name", | ||
| dataType: "sap.ui.model.type.String", | ||
| sortable: true, filterable: true | ||
| }, | ||
| { | ||
| key: "price", label: "Price", | ||
| dataType: "sap.ui.model.type.Float", | ||
| sortable: true, filterable: true | ||
| } | ||
| ]); | ||
| }; | ||
|
|
||
| return MyDelegate; | ||
| }); | ||
| ``` | ||
|
|
||
| ## MDC table usage | ||
|
|
||
| ```xml | ||
| <mdc:Table id="mdcTable" header="Products" | ||
| delegate="{name: 'my/app/delegate/TableDelegate', payload: {entitySet: 'Products'}}" | ||
| p13nMode="Column,Sort,Filter" autoBindOnInit="true"> | ||
| <mdc:columns> | ||
| <mdc:Column header="Name" propertyKey="name"> | ||
| <Text text="{name}"/> | ||
| </mdc:Column> | ||
| <mdc:Column header="Price" propertyKey="price"> | ||
| <ObjectNumber number="{price}"/> | ||
| </mdc:Column> | ||
| </mdc:columns> | ||
| </mdc:Table> | ||
| ``` |
Oops, something went wrong.
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
Uh oh!
There was an error while loading. Please reload this page.