Skip to content

Commit

Permalink
feat(ui5-combobox/multi-combobox): physical items (#10051)
Browse files Browse the repository at this point in the history
  • Loading branch information
MapTo0 authored Nov 7, 2024
1 parent 950441b commit 73f82ab
Show file tree
Hide file tree
Showing 25 changed files with 554 additions and 561 deletions.
39 changes: 23 additions & 16 deletions packages/main/cypress/specs/LitKeyFunction.cy.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
import { html } from "lit";
import "../../src/MultiComboBox.js";
import "../../src/MultiComboBoxItem.js";
import type List from "../../src/List.js";

describe("Lit HTML key function for #each", () => {
it("LIT HTML does not mess up keys when looping over lists", () => {
Expand All @@ -21,30 +22,36 @@ describe("Lit HTML key function for #each", () => {

cy.get("@mcb")
.shadow()
.find(".ui5-multi-combobox-all-items-responsive-popover")
.as("popover");
.find("[ui5-responsive-popover]")
.as("rpo");

cy.get("@popover")
.find(".ui5-multi-combobox-all-items-list > ui5-li")
.as("items");
cy.get("@rpo")
.find("[ui5-list]")
.as("list");

cy.get("@items")
.eq(0)
cy.get("@list")
.then($el => {
return ($el[0] as List).getSlottedNodes("items");
})
.realClick();

cy.get("@mcb")
.shadow()
.find(".inputIcon")
.realClick();

// cy.get("@items")
// .eq(0)
// .should("contain.text", "<empty>")
// .should("not.have.attr", "selected");

// cy.get("@items")
// .eq(3)
// .should("contain.text", "USA")
// .should("have.attr", "selected");
cy.get("@list")
.then($el => {
return ($el[0] as List).getSlottedNodes("items")[0];
})
.invoke("attr", "text", "<empty>")
.should("not.have.attr", "selected");

cy.get("@list")
.then($el => {
return ($el[0] as List).getSlottedNodes("items")[3];
})
.invoke("attr", "text", "USA")
.should("have.attr", "selected");
});
});
3 changes: 1 addition & 2 deletions packages/main/cypress/specs/base/Events.cy.ts
Original file line number Diff line number Diff line change
Expand Up @@ -206,8 +206,7 @@ describe("Event bubbling", () => {
.realClick();

cy.get("@multiCombobox")
.shadow()
.find("[ui5-responsive-popover]")
.find("[ui5-mcb-item]")
.should("be.visible");

cy.get("@multiComboboxIcon")
Expand Down
29 changes: 10 additions & 19 deletions packages/main/src/ComboBox.ts
Original file line number Diff line number Diff line change
Expand Up @@ -98,18 +98,13 @@ interface IComboBoxItem extends UI5Element {
isGroupItem?: boolean,
selected?: boolean,
additionalText?: string,
stableDomRef: string,
_isVisible?: boolean,
items?: Array<IComboBoxItem>
}

type ValueStateAnnouncement = Record<Exclude<ValueState, ValueState.None>, string>;
type ValueStateTypeAnnouncement = Record<Exclude<ValueState, ValueState.None>, string>;

type ComboBoxListItem = ListItemStandard & {
mappedItem: ComboBoxItem
};

enum ValueStateIconMapping {
Negative = "error",
Critical = "alert",
Expand Down Expand Up @@ -398,7 +393,12 @@ class ComboBox extends UI5Element implements IFormInputElement {
* Defines the component items.
* @public
*/
@slot({ type: HTMLElement, "default": true, invalidateOnChildChange: true })
@slot({
type: HTMLElement,
"default": true,
individualSlots: true,
invalidateOnChildChange: true,
})
items!: Array<IComboBoxItem>;

/**
Expand Down Expand Up @@ -504,10 +504,6 @@ class ComboBox extends UI5Element implements IFormInputElement {
}

this.storeResponsivePopoverWidth();

this.items.forEach(item => {
item._getRealDomRef = () => this._getPicker().querySelector(`*[data-ui5-stable=${item.stableDomRef}]`)!;
});
}

_focusin(e: FocusEvent) {
Expand Down Expand Up @@ -1129,9 +1125,9 @@ class ComboBox extends UI5Element implements IFormInputElement {
}

_selectItem(e: CustomEvent<ListItemClickEventDetail>) {
const listItem = e.detail.item as ComboBoxListItem;
const item = e.detail.item as ComboBoxItem;

this._selectedItemText = listItem.mappedItem.text || "";
this._selectedItemText = item.text || "";
this._selectionPerformed = true;

const sameItemSelected = this.value === this._selectedItemText;
Expand All @@ -1144,17 +1140,12 @@ class ComboBox extends UI5Element implements IFormInputElement {

this.value = this._selectedItemText;

if (!listItem.mappedItem.selected) {
if (!item.selected) {
this.fireDecoratorEvent<ComboBoxSelectionChangeEventDetail>("selection-change", {
item: listItem.mappedItem,
item,
});
}

this._filteredItems.map(item => {
item.selected = (item === listItem.mappedItem && !item.isGroupItem);
return item;
});

this._fireChangeEvent();
this._closeRespPopover();

Expand Down
18 changes: 18 additions & 0 deletions packages/main/src/ComboBoxItem.hbs
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
{{>include "./ListItemBase.hbs"}}

{{#*inline "listItemContent"}}
<div part="content" id="content" class="ui5-li-content">
<div class="ui5-li-text-wrapper">
<span part="title" class="ui5-li-title">
{{{text}}}
</span>
{{#if additionalText}}
<span part="additional-text" class="ui5-li-additional-text">{{additionalText}}</span>
{{/if}}
</div>
</div>
{{/inline}}

{{#*inline "listItemAttributes"}}
role="option"
{{/inline}}
27 changes: 19 additions & 8 deletions packages/main/src/ComboBoxItem.ts
Original file line number Diff line number Diff line change
@@ -1,19 +1,24 @@
import customElement from "@ui5/webcomponents-base/dist/decorators/customElement.js";
import property from "@ui5/webcomponents-base/dist/decorators/property.js";
import UI5Element from "@ui5/webcomponents-base/dist/UI5Element.js";
import type { IComboBoxItem } from "./ComboBox.js";
import ListItemBase from "./ListItemBase.js";

import ComboBoxItemTemplate from "./generated/templates/ComboBoxItemTemplate.lit.js";
import ComboboxItemCss from "./generated/themes/ComboBoxItem.css.js";
/**
* @class
* The `ui5-cb-item` represents the item for a `ui5-combobox`.
* @constructor
* @extends UI5Element
* @abstract
* @extends ListItemBase
* @implements {IComboBoxItem}
* @public
*/
@customElement("ui5-cb-item")
class ComboBoxItem extends UI5Element implements IComboBoxItem {
@customElement({
tag: "ui5-cb-item",
template: ComboBoxItemTemplate,
styles: [ListItemBase.styles, ComboboxItemCss],
})
class ComboBoxItem extends ListItemBase implements IComboBoxItem {
/**
* Defines the text of the component.
* @default undefined
Expand Down Expand Up @@ -52,9 +57,15 @@ class ComboBoxItem extends UI5Element implements IComboBoxItem {
@property({ type: Boolean })
selected = false;

get stableDomRef() {
return this.getAttribute("stable-dom-ref") || `${this._id}-stable-dom-ref`;
}
/**
* Defines the markup text that will be displayed as suggestion.
* Used for highlighting the matching parts of the text.
*
* @since 2.4.0
* @private
*/
@property()
markupText = "";
}

ComboBoxItem.define();
Expand Down
9 changes: 9 additions & 0 deletions packages/main/src/ComboBoxItemGroup.hbs
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
{{>include "./ListItemGroup.hbs"}}

{{#*inline "items"}}
{{#each items}}
{{#if _isVisible}}
<slot name="{{this._individualSlot}}"></slot>
{{/if}}
{{/each}}
{{/inline}}
74 changes: 28 additions & 46 deletions packages/main/src/ComboBoxItemGroup.ts
Original file line number Diff line number Diff line change
@@ -1,63 +1,45 @@
import customElement from "@ui5/webcomponents-base/dist/decorators/customElement.js";
import property from "@ui5/webcomponents-base/dist/decorators/property.js";
import slot from "@ui5/webcomponents-base/dist/decorators/slot.js";
import UI5Element from "@ui5/webcomponents-base/dist/UI5Element.js";
import type { IComboBoxItem } from "./ComboBox.js";
import ListItemGroup from "./ListItemGroup.js";
import type ComboBoxItem from "./ComboBoxItem.js";
import ComboBoxItemGroupTemplate from "./generated/templates/ComboBoxItemGroupTemplate.lit.js";

/**
* @class
* The `ui5-cb-group-item` is type of suggestion item,
* that can be used to split the `ui5-combobox` suggestions into groups.
* @constructor
* @extends UI5Element
* @extends ListItemGroup
* @abstract
* @public
* @implements {IComboBoxItem}
* @since 1.0.0-rc.15
*/
@customElement("ui5-cb-item-group")
class ComboBoxItemGroup extends UI5Element implements IComboBoxItem {
/**
* Defines the text of the component.
* @default undefined
* @public
*/
@property()
headerText?: string;

/**
* Indicates whether the item is focused
* @protected
*/
@property({ type: Boolean })
focused = false

/**
* Defines the items of the <code>ui5-cb-item-group</code>.
* @public
*/
@slot({
"default": true,
invalidateOnChildChange: true,
type: HTMLElement,
})
items!: Array<IComboBoxItem>;

/**
* Used to avoid tag name checks
* @protected
*/
get isGroupItem(): boolean {
return true;
}

get stableDomRef() {
return this.getAttribute("stable-dom-ref") || `${this._id}-stable-dom-ref`;
}

get _isVisible() {
return this.items.some(item => item._isVisible);
}
@customElement({
tag: "ui5-cb-item-group",
template: ComboBoxItemGroupTemplate,
})
class ComboBoxItemGroup extends ListItemGroup implements IComboBoxItem {
/**
* Defines the items of the <code>ui5-cb-item-group</code>.
* @public
*/
@slot({
"default": true,
invalidateOnChildChange: true,
individualSlots: true,
type: HTMLElement,
})
items!: Array<ComboBoxItem>;

get isGroupItem(): boolean {
return true;
}

get _isVisible() {
return this.items.some(item => item._isVisible);
}
}

ComboBoxItemGroup.define();
Expand Down
33 changes: 2 additions & 31 deletions packages/main/src/ComboBoxPopover.hbs
Original file line number Diff line number Diff line change
Expand Up @@ -85,20 +85,7 @@
selection-mode="Single"
>
{{#each _filteredItems}}
{{#if isGroupItem}}
{{#if _isVisible}}
<ui5-li-group header-text="{{this.headerText}}" ?focused={{this.focused}}>
{{#each this.items}}
{{#if _isVisible}}
{{> listItem}}
{{/if}}
{{/each}}
</ui5-li-group>
{{/if}}
{{else}}
{{> listItem}}
{{/if}}

<slot name="{{this._individualSlot}}"></slot>
{{/each}}
</ui5-list>

Expand Down Expand Up @@ -140,20 +127,4 @@
{{else}}
<slot name="valueStateMessage"></slot>
{{/if}}
{{/inline}}

{{#*inline "listItem"}}
<ui5-li
accessible-role="Option"
type="Active"
wrapping-type="Normal"
additional-text={{this.additionalText}}
group-name={{this.groupName}}
.mappedItem={{this}}
?selected={{this.selected}}
?focused={{this.focused}}
data-ui5-stable="{{this.stableDomRef}}"
>
{{this.text}}
</ui5-li>
{{/inline}}
{{/inline}}
4 changes: 2 additions & 2 deletions packages/main/src/List.ts
Original file line number Diff line number Diff line change
Expand Up @@ -872,10 +872,10 @@ class List extends UI5Element {

slottedItems.forEach(item => {
if (isInstanceOfListItemGroup(item)) {
const groupItems = [item.groupHeaderItem, ...item.items].filter(Boolean);
const groupItems = [item.groupHeaderItem, ...item.items.filter(listItem => listItem.assignedSlot)].filter(Boolean);
items.push(...groupItems);
} else {
items.push(item);
item.assignedSlot && items.push(item);
}
});

Expand Down
Loading

0 comments on commit 73f82ab

Please sign in to comment.