Skip to content

Commit

Permalink
further click interaction correction
Browse files Browse the repository at this point in the history
  • Loading branch information
nielslyngsoe committed Nov 20, 2024
1 parent c5bf804 commit 8d99634
Show file tree
Hide file tree
Showing 2 changed files with 39 additions and 26 deletions.
52 changes: 26 additions & 26 deletions packages/uui-base/lib/mixins/SelectableMixin.ts
Original file line number Diff line number Diff line change
Expand Up @@ -75,25 +75,45 @@ export const SelectableMixin = <T extends Constructor<LitElement>>(

constructor(...args: any[]) {
super(...args);
this.addEventListener('click', this._handleClick);
this.addEventListener('keydown', this.handleSelectKeydown);
this.addEventListener('click', this.#onClick);
this.addEventListener('keydown', this.#onKeydown);
}

private handleSelectKeydown = (e: KeyboardEvent) => {
#onKeydown = (e: KeyboardEvent) => {
const composePath = e.composedPath();
if (
(this._selectable || (this.deselectable && this.selected)) &&
composePath.indexOf(this.selectableTarget) === 0
) {
if (this.selectableTarget === this) {
if (e.code !== 'Space' && e.code !== 'Enter') return;
this._toggleSelect();
this.#toggleSelect();
e.preventDefault();
}
}
};

private _select() {
#onClick = (e: Event) => {
const composePath = e.composedPath();
if (
(this._selectable || (this.deselectable && this.selected)) &&
composePath.indexOf(this.selectableTarget) === 0
) {
this.#toggleSelect();
}
};

#toggleSelect() {
// Only allow for select-interaction if selectable is true. Deselectable is ignored in this case, we do not want a DX where only deselection is a possibility..
if (!this.selectable) return;
if (this.deselectable === false) {
this.#select();
} else {
this.selected ? this.#deselect() : this.#select();
}
}

#select() {
if (!this.selectable) return;
const selectEvent = new UUISelectableEvent(UUISelectableEvent.SELECTED);
this.dispatchEvent(selectEvent);
Expand All @@ -102,34 +122,14 @@ export const SelectableMixin = <T extends Constructor<LitElement>>(
this.selected = true;
}

private _deselect() {
#deselect() {
if (!this.deselectable) return;
const selectEvent = new UUISelectableEvent(UUISelectableEvent.DESELECTED);
this.dispatchEvent(selectEvent);
if (selectEvent.defaultPrevented) return;

this.selected = false;
}

private _handleClick(e: Event) {
const composePath = e.composedPath();
if (
(this._selectable || (this.deselectable && this.selected)) &&
composePath.indexOf(this.selectableTarget) === 0
) {
this._toggleSelect();
}
}

private _toggleSelect() {
// Only allow for select-interaction if selectable is true. Deselectable is ignored in this case, we do not want a DX where only deselection is a possibility..
if (!this.selectable) return;
if (this.deselectable === false) {
this._select();
} else {
this.selected ? this._deselect() : this._select();
}
}
}
// prettier-ignore
return (SelectableMixinClass as unknown) as Constructor<SelectableMixinInterface> & T;
Expand Down
13 changes: 13 additions & 0 deletions packages/uui-card-media/lib/uui-card-media.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -109,6 +109,19 @@ describe('UUICardMediaElement', () => {
it('do not react to a click event on other parts', async () => {
element.selectable = true;
await elementUpdated(element);
const listener = oneEvent(element, UUICardEvent.OPEN);
element
.shadowRoot!.querySelector<HTMLAnchorElement>('#open-part')!
.click();
const event = await listener;
expect(event).to.exist;
expect(event.type).to.equal(UUICardEvent.OPEN);
expect(element.selected).to.be.false;
});
it('do not react to a click event on other parts as href', async () => {
element.selectable = true;
element.href = '#hello';
await elementUpdated(element);
element
.shadowRoot!.querySelector<HTMLAnchorElement>('#open-part')!
.click();
Expand Down

0 comments on commit 8d99634

Please sign in to comment.