Skip to content

Commit 7060ff3

Browse files
committed
feat(ui5-li): add text wrapping support with expandable text
The standard list item now supports content wrapping through a new "wrapping" property. When enabled, long text content (title and description) will wrap to multiple lines instead of truncating with an ellipsis. Key features: - Responsive behavior based on screen size: - On mobile (size S): Text truncates after 100 characters - On tablets/desktop (size M/L): Text truncates after 300 characters - "Show More/Less" functionality for very long content using ui5-expandable-text - Works for both title and description fields - CSS styling to ensure proper layout in different states Implementation details: - Added "wrapping" boolean property to ListItemStandard component - Conditionally render content using ExpandableText component when wrapping is enabled - Added _maxCharacters getter to determine truncation point based on media range - Updated ListItemStandardTemplate to handle wrapped content rendering - Added CSS styles to support proper wrapping behavior - Updated HTML test page with examples Documentation: - Added new section in List.mdx explaining the wrapping behavior - Created detailed sample in WrappingBehavior folder - Added JSDoc descriptions for the new property - Added note about deprecated usage of default slot in favor of text property Tests: - Added desktop test for 300 character limit - Added mobile-specific test for 100 character limit - Added test for toggling wrapping property NOTE: This change also promotes the use of the "text" property (added in v2.9.0) instead of the default slot content for setting list item text. The default slot usage is now deprecated and will be removed in a future version.
1 parent 967ef56 commit 7060ff3

File tree

13 files changed

+555
-52
lines changed

13 files changed

+555
-52
lines changed

packages/main/cypress/specs/List.cy.tsx

+91-1
Original file line numberDiff line numberDiff line change
@@ -20,7 +20,7 @@ describe("List Tests", () => {
2020

2121
cy.get<List>("@list")
2222
.then(list => {
23-
list.get(0).addEventListener("ui5-load-more", cy.stub().as("loadMore"));
23+
list.get(0)?.addEventListener("ui5-load-more", cy.stub().as("loadMore"));
2424
})
2525
.shadow()
2626
.find(".ui5-list-scroll-container")
@@ -177,3 +177,93 @@ describe("List - Accessibility", () => {
177177
});
178178
});
179179
});
180+
181+
describe("List - Wrapping Behavior", () => {
182+
it("renders list items with wrapping functionality", () => {
183+
const longText = "This is a very long text that should demonstrate the wrapping functionality of ListItemStandard components; This is a very long text that should demonstrate the wrapping functionality of ListItemStandard components; This is a very long text that should demonstrate the wrapping functionality of ListItemStandard components; This is a very long text that should demonstrate the wrapping functionality of ListItemStandard components; This is a very long text that should demonstrate the wrapping functionality of ListItemStandard components";
184+
const longDescription = "This is an even longer description text to verify that wrapping works correctly for the description part of the list item as well; This is an even longer description text to verify that wrapping works correctly for the description part of the list item as well; This is an even longer description text to verify that wrapping works correctly for the description part of the list item as well; This is an even longer description text to verify that wrapping works correctly for the description part of the list item as well; This is an even longer description text to verify that wrapping works correctly for the description part of the list item as well; This is an even longer description text to verify that wrapping works correctly for the description part of the list item as well";
185+
186+
cy.mount(
187+
<List>
188+
<ListItemStandard id="wrapping-item" wrapping text={longText} description={longDescription}></ListItemStandard>
189+
</List>
190+
);
191+
192+
// Check wrapping attributes are set correctly
193+
cy.get("#wrapping-item")
194+
.should("have.attr", "wrapping");
195+
196+
cy.get("#wrapping-item")
197+
.should("have.attr", "wrapping-type", "Normal");
198+
199+
// Check that ExpandableText components are present in the wrapping item
200+
cy.get("#wrapping-item")
201+
.shadow()
202+
.find("ui5-expandable-text")
203+
.should("exist")
204+
.and("have.length", 2);
205+
});
206+
207+
it("uses maxCharacters of 300 on desktop viewport for wrapping list items", () => {
208+
const longText = "This is a very long text that exceeds 100 characters but is less than 300 characters. This sentence is just to add more text to ensure we pass the 100 character threshold. And now we're adding even more text to be extra certain that we have enough content to demonstrate the behavior properly. And now we're adding even more text to be extra certain that we have enough content to demonstrate the behavior properly. And now we're adding even more text to be extra certain that we have enough content to demonstrate the behavior properly.";
209+
210+
cy.mount(
211+
<List>
212+
<ListItemStandard id="wrapping-item" wrapping text={longText}></ListItemStandard>
213+
</List>
214+
);
215+
216+
// Check that ExpandableText is created with maxCharacters prop of 300
217+
cy.get("#wrapping-item")
218+
.shadow()
219+
.find("ui5-expandable-text")
220+
.first()
221+
.invoke('prop', 'maxCharacters')
222+
.should('eq', 300);
223+
});
224+
225+
it("should switch wrapping type when wrapping prop is toggled", () => {
226+
const longText = "This is a very long text that should be wrapped when the wrapping prop is enabled, and truncated when it's disabled. This is a very long text that should be wrapped when the wrapping prop is enabled, and truncated when it's disabled. This is a very long text that should be wrapped when the wrapping prop is enabled, and truncated when it's disabled. This is a very long text that should be wrapped when the wrapping prop is enabled, and truncated when it's disabled. This is a very long text that should be wrapped when the wrapping prop is enabled, and truncated when it's disabled. And now we're adding even more text to be extra certain that we have enough content to demonstrate the behavior properly.";
227+
228+
// First render with wrapping enabled
229+
cy.mount(
230+
<List>
231+
<ListItemStandard id="wrapping-item" wrapping text={longText}></ListItemStandard>
232+
</List>
233+
);
234+
235+
// Check that wrapping attribute is set
236+
cy.get("#wrapping-item")
237+
.should("have.attr", "wrapping");
238+
239+
// Check that wrapping-type attribute is set to Normal
240+
cy.get("#wrapping-item")
241+
.should("have.attr", "wrapping-type", "Normal");
242+
243+
// Should have expandable text component when wrapping is enabled
244+
cy.get("#wrapping-item")
245+
.shadow()
246+
.find("ui5-expandable-text")
247+
.should("exist");
248+
249+
// Remove the wrapping attribute from the existing component
250+
cy.get("#wrapping-item")
251+
.then($el => {
252+
$el[0].removeAttribute("wrapping");
253+
});
254+
255+
// Check that wrapping attribute is removed
256+
cy.get("#wrapping-item")
257+
.should("not.have.attr", "wrapping");
258+
259+
// Now check the wrapping-type attribute in a separate command
260+
cy.get("#wrapping-item")
261+
.should("have.attr", "wrapping-type", "None");
262+
263+
// Should not have expandable text component when wrapping is disabled
264+
cy.get("#wrapping-item")
265+
.shadow()
266+
.find("ui5-expandable-text")
267+
.should("not.exist");
268+
});
269+
});
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,31 @@
1+
import List from "../../src/List.js";
2+
import ListItemStandard from "../../src/ListItemStandard.js";
3+
4+
describe("List Mobile Tests", () => {
5+
before(() => {
6+
cy.ui5SimulateDevice("phone");
7+
});
8+
9+
it("adjusts maxCharacters based on viewport size for wrapping list items", () => {
10+
const longText = "This is a very long text that exceeds 100 characters but is less than 300 characters. This sentence is just to add more text to ensure we pass the 100 character threshold. And now we're adding even more text to be extra certain.";
11+
12+
cy.mount(
13+
<List>
14+
<ListItemStandard id="wrapping-item" wrapping text={longText}></ListItemStandard>
15+
</List>
16+
);
17+
18+
// Get the list item and check its media range
19+
cy.get("#wrapping-item")
20+
.invoke('prop', 'mediaRange')
21+
.should('eq', 'S');
22+
23+
// Check that ExpandableText is created with maxCharacters prop of 100
24+
cy.get("#wrapping-item")
25+
.shadow()
26+
.find("ui5-expandable-text")
27+
.first()
28+
.invoke('prop', 'maxCharacters')
29+
.should('eq', 100);
30+
});
31+
});

packages/main/src/List.ts

+19-1
Original file line numberDiff line numberDiff line change
@@ -55,6 +55,7 @@ import type {
5555
SelectionRequestEventDetail,
5656
} from "./ListItem.js";
5757
import ListSeparator from "./types/ListSeparator.js";
58+
import MediaRange from "@ui5/webcomponents-base/dist/MediaRange.js";
5859

5960
// Template
6061
import ListTemplate from "./ListTemplate.js";
@@ -464,6 +465,13 @@ class List extends UI5Element {
464465
@property({ type: Boolean })
465466
_loadMoreActive = false;
466467

468+
/**
469+
* Defines the current media query size.
470+
* @private
471+
*/
472+
@property({ type: String })
473+
mediaRange!: string;
474+
467475
/**
468476
* Defines the items of the component.
469477
*
@@ -494,6 +502,7 @@ class List extends UI5Element {
494502
resizeListenerAttached: boolean;
495503
listEndObserved: boolean;
496504
_handleResize: ResizeObserverCallback;
505+
_handleMediaRangeUpdateBound: ResizeObserverCallback;
497506
initialIntersection: boolean;
498507
_selectionRequested?: boolean;
499508
_groupCount: number;
@@ -530,7 +539,7 @@ class List extends UI5Element {
530539

531540
this._handleResize = this.checkListInViewport.bind(this);
532541

533-
this._handleResize = this.checkListInViewport.bind(this);
542+
this._handleMediaRangeUpdateBound = this._handleMediaRangeUpdate.bind(this);
534543

535544
// Indicates the List bottom most part has been detected by the IntersectionObserver
536545
// for the first time.
@@ -562,13 +571,15 @@ class List extends UI5Element {
562571
onEnterDOM() {
563572
registerUI5Element(this, this._updateAssociatedLabelsTexts.bind(this));
564573
DragRegistry.subscribe(this);
574+
ResizeHandler.register(this.getDomRef()!, this._handleMediaRangeUpdateBound);
565575
}
566576

567577
onExitDOM() {
568578
deregisterUI5Element(this);
569579
this.unobserveListEnd();
570580
this.resizeListenerAttached = false;
571581
ResizeHandler.deregister(this.getDomRef()!, this._handleResize);
582+
ResizeHandler.deregister(this.getDomRef()!, this._handleMediaRangeUpdateBound);
572583
DragRegistry.unsubscribe(this);
573584
}
574585

@@ -776,6 +787,8 @@ class List extends UI5Element {
776787
(item as ListItem)._selectionMode = this.selectionMode;
777788
}
778789
item.hasBorder = showBottomBorder;
790+
791+
(item as ListItem).mediaRange = this.mediaRange;
779792
});
780793
}
781794

@@ -1065,6 +1078,11 @@ class List extends UI5Element {
10651078
}
10661079
}
10671080

1081+
_handleMediaRangeUpdate() {
1082+
const width = this.getBoundingClientRect().width;
1083+
this.mediaRange = MediaRange.getCurrentRange(MediaRange.RANGESETS.RANGE_4STEPS, width);
1084+
}
1085+
10681086
/*
10691087
* KEYBOARD SUPPORT
10701088
*/

packages/main/src/ListItem.ts

+7
Original file line numberDiff line numberDiff line change
@@ -191,6 +191,13 @@ abstract class ListItem extends ListItemBase {
191191
@property()
192192
_selectionMode: `${ListSelectionMode}` = "None";
193193

194+
/**
195+
* Defines the current media query size.
196+
* @private
197+
*/
198+
@property({ type: String })
199+
mediaRange!: string;
200+
194201
/**
195202
* Defines the delete button, displayed in "Delete" mode.
196203
* **Note:** While the slot allows custom buttons, to match

packages/main/src/ListItemStandard.ts

+73-27
Original file line numberDiff line numberDiff line change
@@ -28,7 +28,8 @@ import ListItemStandardTemplate from "./ListItemStandardTemplate.js";
2828
* @csspart checkbox - Used to style the checkbox rendered when the list item is in multiple selection mode
2929
* @slot {Node[]} default - Defines the text of the component.
3030
*
31-
* **Note:** Although this slot accepts HTML Elements, it is strongly recommended that you only use text in order to preserve the intended design.
31+
* **Note:** Although this slot accepts HTML Elements, it is strongly recommended that you only use text in order to preserve the intended design. <br/>
32+
* **Note:** Deprecated since version `2.9.0`. Use the `text` property instead.
3233
* @constructor
3334
* @extends ListItem
3435
* @public
@@ -40,37 +41,26 @@ import ListItemStandardTemplate from "./ListItemStandardTemplate.js";
4041
})
4142
class ListItemStandard extends ListItem implements IAccessibleListItem {
4243
/**
43-
* Defines the description displayed right under the item text, if such is present.
44+
* Defines the text of the component.
45+
*
4446
* @default undefined
4547
* @public
46-
* @since 0.8.0
48+
* @since 2.9.0
4749
*/
4850
@property()
49-
description?: string;
51+
text?: string;
5052

5153
/**
52-
* Defines the `icon` source URI.
53-
*
54-
* **Note:**
55-
* SAP-icons font provides numerous built-in icons. To find all the available icons, see the
56-
* [Icon Explorer](https://sdk.openui5.org/test-resources/sap/m/demokit/iconExplorer/webapp/index.html).
54+
* Defines the description displayed right under the item text, if such is present.
5755
* @default undefined
5856
* @public
57+
* @since 0.8.0
5958
*/
6059
@property()
61-
icon?: string;
62-
63-
/**
64-
* Defines whether the `icon` should be displayed in the beginning of the list item or in the end.
65-
*
66-
* @default false
67-
* @public
68-
*/
69-
@property({ type: Boolean })
70-
iconEnd = false;
60+
description?: string;
7161

7262
/**
73-
* Defines the `additionalText`, displayed in the end of the list item.
63+
* Defines the additional text, displayed in the end of the list item.
7464
* @default undefined
7565
* @public
7666
* @since 1.0.0-rc.15
@@ -89,6 +79,27 @@ class ListItemStandard extends ListItem implements IAccessibleListItem {
8979
@property()
9080
additionalTextState: `${ValueState}` = "None";
9181

82+
/**
83+
* Defines the `icon` source URI.
84+
*
85+
* **Note:**
86+
* SAP-icons font provides numerous built-in icons. To find all the available icons, see the
87+
* [Icon Explorer](https://sdk.openui5.org/test-resources/sap/m/demokit/iconExplorer/webapp/index.html).
88+
* @default undefined
89+
* @public
90+
*/
91+
@property()
92+
icon?: string;
93+
94+
/**
95+
* Defines whether the `icon` should be displayed in the beginning of the list item or in the end.
96+
*
97+
* @default false
98+
* @public
99+
*/
100+
@property({ type: Boolean })
101+
iconEnd = false;
102+
92103
/**
93104
* Defines whether the item is movable.
94105
* @default false
@@ -99,14 +110,20 @@ class ListItemStandard extends ListItem implements IAccessibleListItem {
99110
movable = false;
100111

101112
/**
102-
* Defines the text alternative of the component.
103-
* Note: If not provided a default text alternative will be set, if present.
104-
* @default undefined
113+
* Defines whether the content of the list item should wrap when it's too long.
114+
* When set to true, the content (title, description) will be wrapped
115+
* using the `ui5-expandable-text` component.<br/>
116+
*
117+
* The text can wrap up to 100 characters on small screens (size S) and
118+
* up to 300 characters on larger screens (size M and above). When text exceeds
119+
* these limits, it truncates with an ellipsis followed by a text expansion trigger.
120+
*
121+
* @default false
105122
* @public
106-
* @since 1.0.0-rc.15
123+
* @since 2.9.0
107124
*/
108-
@property()
109-
declare accessibleName?: string;
125+
@property({ type: Boolean })
126+
wrapping = false;
110127

111128
/**
112129
* Defines if the text of the component should wrap, they truncate by default.
@@ -119,6 +136,16 @@ class ListItemStandard extends ListItem implements IAccessibleListItem {
119136
@property()
120137
wrappingType: `${WrappingType}` = "None";
121138

139+
/**
140+
* Defines the text alternative of the component.
141+
* **Note:** If not provided a default text alternative will be set, if present.
142+
* @default undefined
143+
* @public
144+
* @since 1.0.0-rc.15
145+
*/
146+
@property()
147+
accessibleName?: string;
148+
122149
/**
123150
* Indicates if the list item has text content.
124151
* @private
@@ -143,8 +170,27 @@ class ListItemStandard extends ListItem implements IAccessibleListItem {
143170

144171
onBeforeRendering() {
145172
super.onBeforeRendering();
146-
this.hasTitle = !!this.textContent;
173+
this.hasTitle = !!(this.text || this.textContent);
147174
this._hasImage = this.hasImage;
175+
this.wrappingType = this.wrapping ? "Normal" : "None";
176+
}
177+
178+
/**
179+
* Returns the content text, either from text property or from the default slot
180+
* @private
181+
*/
182+
get _textContent(): string {
183+
return this.text || this.textContent || "";
184+
}
185+
186+
/**
187+
* Determines the maximum characters to display based on the current media range.
188+
* - Size S: 100 characters
189+
* - Size M and larger: 300 characters
190+
* @private
191+
*/
192+
get _maxCharacters(): number {
193+
return this.mediaRange === "S" ? 100 : 300;
148194
}
149195

150196
get displayIconBegin(): boolean {

0 commit comments

Comments
 (0)