From f078f8eca48c1c51f53100efb2c443cdc197b4c0 Mon Sep 17 00:00:00 2001 From: Serhii Kulykov Date: Fri, 15 Nov 2024 11:22:13 +0200 Subject: [PATCH] refactor: extract avatar-group logic into reusable mixin (#8148) --- packages/avatar-group/package.json | 1 + .../src/vaadin-avatar-group-mixin.d.ts | 99 ++++ .../src/vaadin-avatar-group-mixin.js | 536 ++++++++++++++++++ .../avatar-group/src/vaadin-avatar-group.d.ts | 90 +-- .../avatar-group/src/vaadin-avatar-group.js | 531 +---------------- .../test/typings/avatar-group.types.ts | 2 + 6 files changed, 646 insertions(+), 613 deletions(-) create mode 100644 packages/avatar-group/src/vaadin-avatar-group-mixin.d.ts create mode 100644 packages/avatar-group/src/vaadin-avatar-group-mixin.js diff --git a/packages/avatar-group/package.json b/packages/avatar-group/package.json index 22896b9d9c..f1c4507e90 100644 --- a/packages/avatar-group/package.json +++ b/packages/avatar-group/package.json @@ -36,6 +36,7 @@ "polymer" ], "dependencies": { + "@open-wc/dedupe-mixin": "^1.3.0", "@polymer/polymer": "^3.0.0", "@vaadin/a11y-base": "24.6.0-alpha8", "@vaadin/avatar": "24.6.0-alpha8", diff --git a/packages/avatar-group/src/vaadin-avatar-group-mixin.d.ts b/packages/avatar-group/src/vaadin-avatar-group-mixin.d.ts new file mode 100644 index 0000000000..91707e202c --- /dev/null +++ b/packages/avatar-group/src/vaadin-avatar-group-mixin.d.ts @@ -0,0 +1,99 @@ +/** + * @license + * Copyright (c) 2020 - 2024 Vaadin Ltd. + * This program is available under Apache License Version 2.0, available at https://vaadin.com/license/ + */ +import type { Constructor } from '@open-wc/dedupe-mixin'; +import type { AvatarI18n } from '@vaadin/avatar/src/vaadin-avatar.js'; +import type { OverlayClassMixinClass } from '@vaadin/component-base/src/overlay-class-mixin.js'; +import type { ResizeMixinClass } from '@vaadin/component-base/src/resize-mixin.js'; + +export interface AvatarGroupI18n extends AvatarI18n { + activeUsers: { + one: string; + many: string; + }; + joined: string; + left: string; +} + +export interface AvatarGroupItem { + name?: string; + abbr?: string; + img?: string; + colorIndex?: number; + className?: string; +} + +/** + * A mixin providing common avatar functionality. + */ +export declare function AvatarGroupMixin>( + base: T, +): Constructor & Constructor & Constructor & T; + +export declare class AvatarGroupMixinClass { + /** + * An array containing the items which will be stamped as avatars. + * + * The items objects allow to configure [`name`](#/elements/vaadin-avatar#property-name), + * [`abbr`](#/elements/vaadin-avatar#property-abbr), [`img`](#/elements/vaadin-avatar#property-img) + * and [`colorIndex`](#/elements/vaadin-avatar#property-colorIndex) properties on the + * stamped avatars, and set `className` to provide CSS class names. + * + * #### Example + * + * ```js + * group.items = [ + * { + * name: 'User name', + * img: 'url-to-image.png', + * className: 'even' + * }, + * { + * abbr: 'JD', + * colorIndex: 1, + * className: 'odd' + * }, + * ]; + * ``` + */ + items: AvatarGroupItem[] | undefined; + + /** + * The maximum number of avatars to display. By default, all the avatars are displayed. + * When _maxItemsVisible_ is set, the overflowing avatars are grouped into one avatar with + * a dropdown. Setting 0 or 1 has no effect so there are always at least two avatars visible. + * @attr {number} max-items-visible + */ + maxItemsVisible: number | null | undefined; + + /** + * The object used to localize this component. + * To change the default localization, replace the entire + * _i18n_ object or just the property you want to modify. + * + * The object has the following JSON structure and default values: + * ``` + * { + * // Translation of the anonymous user avatar tooltip. + * anonymous: 'anonymous', + * // Translation of the avatar group accessible label. + * // {count} is replaced with the actual count of users. + * activeUsers: { + * one: 'Currently one active user', + * many: 'Currently {count} active users' + * }, + * // Screen reader announcement when user joins group. + * // {user} is replaced with the name or abbreviation. + * // When neither is set, "anonymous" is used instead. + * joined: '{user} joined', + * // Screen reader announcement when user leaves group. + * // {user} is replaced with the name or abbreviation. + * // When neither is set, "anonymous" is used instead. + * left: '{user} left' + * } + * ``` + */ + i18n: AvatarGroupI18n; +} diff --git a/packages/avatar-group/src/vaadin-avatar-group-mixin.js b/packages/avatar-group/src/vaadin-avatar-group-mixin.js new file mode 100644 index 0000000000..938e5e362b --- /dev/null +++ b/packages/avatar-group/src/vaadin-avatar-group-mixin.js @@ -0,0 +1,536 @@ +/** + * @license + * Copyright (c) 2020 - 2024 Vaadin Ltd. + * This program is available under Apache License Version 2.0, available at https://vaadin.com/license/ + */ +import { afterNextRender } from '@polymer/polymer/lib/utils/render-status.js'; +import { html, render } from 'lit'; +import { ifDefined } from 'lit/directives/if-defined.js'; +import { announce } from '@vaadin/a11y-base/src/announce.js'; +import { OverlayClassMixin } from '@vaadin/component-base/src/overlay-class-mixin.js'; +import { ResizeMixin } from '@vaadin/component-base/src/resize-mixin.js'; +import { SlotController } from '@vaadin/component-base/src/slot-controller.js'; + +const MINIMUM_DISPLAYED_AVATARS = 2; + +/** + * A mixin providing common avatar group functionality. + * + * @polymerMixin + * @mixes ResizeMixin + * @mixes OverlayClassMixin + */ +export const AvatarGroupMixin = (superClass) => + class AvatarGroupMixinClass extends ResizeMixin(OverlayClassMixin(superClass)) { + static get properties() { + return { + /** + * An array containing the items which will be stamped as avatars. + * + * The items objects allow to configure [`name`](#/elements/vaadin-avatar#property-name), + * [`abbr`](#/elements/vaadin-avatar#property-abbr), [`img`](#/elements/vaadin-avatar#property-img) + * and [`colorIndex`](#/elements/vaadin-avatar#property-colorIndex) properties on the + * stamped avatars, and set `className` to provide CSS class names. + * + * #### Example + * + * ```js + * group.items = [ + * { + * name: 'User name', + * img: 'url-to-image.png', + * className: 'even' + * }, + * { + * abbr: 'JD', + * colorIndex: 1, + * className: 'odd' + * }, + * ]; + * ``` + * + * @type {!Array | undefined} + */ + items: { + type: Array, + observer: '__itemsChanged', + }, + + /** + * The maximum number of avatars to display. By default, all the avatars are displayed. + * When _maxItemsVisible_ is set, the overflowing avatars are grouped into one avatar with + * a dropdown. Setting 0 or 1 has no effect so there are always at least two avatars visible. + * @attr {number} max-items-visible + */ + maxItemsVisible: { + type: Number, + }, + + /** + * The object used to localize this component. + * To change the default localization, replace the entire + * _i18n_ object or just the property you want to modify. + * + * The object has the following JSON structure and default values: + * ``` + * { + * // Translation of the anonymous user avatar tooltip. + * anonymous: 'anonymous', + * // Translation of the avatar group accessible label. + * // {count} is replaced with the actual count of users. + * activeUsers: { + * one: 'Currently one active user', + * many: 'Currently {count} active users' + * }, + * // Screen reader announcement when user joins group. + * // {user} is replaced with the name or abbreviation. + * // When neither is set, "anonymous" is used instead. + * joined: '{user} joined', + * // Screen reader announcement when user leaves group. + * // {user} is replaced with the name or abbreviation. + * // When neither is set, "anonymous" is used instead. + * left: '{user} left' + * } + * ``` + * @type {!AvatarGroupI18n} + * @default {English/US} + */ + i18n: { + type: Object, + value: () => { + return { + anonymous: 'anonymous', + activeUsers: { + one: 'Currently one active user', + many: 'Currently {count} active users', + }, + joined: '{user} joined', + left: '{user} left', + }; + }, + }, + + /** @private */ + _avatars: { + type: Array, + value: () => [], + }, + + /** @private */ + __itemsInView: { + type: Number, + value: null, + }, + + /** @private */ + _overflow: { + type: Object, + }, + + /** @private */ + _overflowItems: { + type: Array, + observer: '__overflowItemsChanged', + computed: '__computeOverflowItems(items, __itemsInView, maxItemsVisible)', + }, + + /** @private */ + _overflowTooltip: { + type: Object, + }, + + /** @private */ + _opened: { + type: Boolean, + observer: '__openedChanged', + }, + }; + } + + static get observers() { + return [ + '__i18nItemsChanged(i18n, items)', + '__updateAvatarsTheme(_overflow, _avatars, _theme)', + '__updateAvatars(items, __itemsInView, maxItemsVisible, _overflow, i18n)', + '__updateOverflowAvatar(_overflow, items, __itemsInView, maxItemsVisible)', + '__updateOverflowTooltip(_overflowTooltip, items, __itemsInView, maxItemsVisible)', + ]; + } + + /** @protected */ + ready() { + super.ready(); + + this._overflowController = new SlotController(this, 'overflow', 'vaadin-avatar', { + initializer: (overflow) => { + overflow.setAttribute('aria-haspopup', 'menu'); + overflow.setAttribute('aria-expanded', 'false'); + overflow.addEventListener('click', (e) => this._onOverflowClick(e)); + overflow.addEventListener('keydown', (e) => this._onOverflowKeyDown(e)); + + const tooltip = document.createElement('vaadin-tooltip'); + tooltip.setAttribute('slot', 'tooltip'); + overflow.appendChild(tooltip); + + this._overflow = overflow; + this._overflowTooltip = tooltip; + }, + }); + this.addController(this._overflowController); + + const overlay = this.$.overlay; + overlay.renderer = this.__overlayRenderer.bind(this); + this._overlayElement = overlay; + + afterNextRender(this, () => { + this.__setItemsInView(); + }); + } + + /** @protected */ + disconnectedCallback() { + super.disconnectedCallback(); + + this._opened = false; + } + + /** @private */ + __getMessage(user, action) { + return action.replace('{user}', user.name || user.abbr || this.i18n.anonymous); + } + + /** + * Renders items when they are provided by the `items` property and clears the content otherwise. + * @param {!HTMLElement} root + * @param {!Select} _select + * @private + */ + __overlayRenderer(root) { + let menu = root.firstElementChild; + if (!menu) { + menu = document.createElement('vaadin-avatar-group-menu'); + menu.addEventListener('keydown', (event) => this._onListKeyDown(event)); + root.appendChild(menu); + } + + menu.textContent = ''; + + if (!this._overflowItems) { + return; + } + + this._overflowItems.forEach((item) => { + menu.appendChild(this.__createItemElement(item)); + }); + } + + /** @private */ + __createItemElement(item) { + const itemElement = document.createElement('vaadin-avatar-group-menu-item'); + + const avatar = document.createElement('vaadin-avatar'); + itemElement.appendChild(avatar); + + avatar.setAttribute('aria-hidden', 'true'); + avatar.setAttribute('tabindex', '-1'); + avatar.i18n = this.i18n; + + if (this._theme) { + avatar.setAttribute('theme', this._theme); + } + + avatar.name = item.name; + avatar.abbr = item.abbr; + avatar.img = item.img; + avatar.colorIndex = item.colorIndex; + if (item.className) { + avatar.className = item.className; + } + + if (item.name) { + const text = document.createTextNode(item.name); + itemElement.appendChild(text); + } + + return itemElement; + } + + /** @private */ + _onOverflowClick(e) { + e.stopPropagation(); + if (this._opened) { + this.$.overlay.close(); + } else if (!e.defaultPrevented) { + this._opened = true; + } + } + + /** @private */ + _onOverflowKeyDown(e) { + if (!this._opened) { + if (/^(Enter|SpaceBar|\s)$/u.test(e.key)) { + e.preventDefault(); + this._opened = true; + } + } + } + + /** @private */ + _onListKeyDown(event) { + if (event.key === 'Escape' || event.key === 'Tab') { + this._opened = false; + } + } + + /** + * @protected + * @override + */ + _onResize() { + this.__setItemsInView(); + } + + /** @private */ + _onVaadinOverlayClose(e) { + if (e.detail.sourceEvent && e.detail.sourceEvent.composedPath().includes(this)) { + e.preventDefault(); + } + } + + /** @private */ + __renderAvatars(items) { + render( + html` + ${items.map( + (item) => html` + + `, + )} + `, + this, + { renderBefore: this._overflow }, + ); + } + + /** @private */ + __updateAvatars(items, itemsInView, maxItemsVisible, overflow) { + if (!overflow || !Array.isArray(items)) { + return; + } + + const limit = this.__getLimit(items.length, itemsInView, maxItemsVisible); + + this.__renderAvatars(limit ? items.slice(0, limit) : items); + + this._avatars = [...this.querySelectorAll('vaadin-avatar')]; + } + + /** @private */ + __computeOverflowItems(items, itemsInView, maxItemsVisible) { + const count = Array.isArray(items) ? items.length : 0; + const limit = this.__getLimit(count, itemsInView, maxItemsVisible); + return limit ? items.slice(limit) : []; + } + + /** @private */ + __updateOverflowAvatar(overflow, items, itemsInView, maxItemsVisible) { + if (overflow) { + const count = Array.isArray(items) ? items.length : 0; + const maxReached = maxItemsVisible != null && count > this.__getMax(maxItemsVisible); + + overflow.abbr = `+${count - this.__getLimit(count, itemsInView, maxItemsVisible)}`; + overflow.toggleAttribute('hidden', !maxReached && !(itemsInView && itemsInView < count)); + } + } + + /** @private */ + __updateAvatarsTheme(overflow, avatars, theme) { + if (overflow) { + [overflow, ...avatars].forEach((avatar) => { + if (theme) { + avatar.setAttribute('theme', theme); + } else { + avatar.removeAttribute('theme'); + } + }); + } + } + + /** @private */ + __updateOverflowTooltip(tooltip, items, itemsInView, maxItemsVisible) { + if (!tooltip || !Array.isArray(items)) { + return; + } + + const limit = this.__getLimit(items.length, itemsInView, maxItemsVisible); + if (limit == null) { + return; + } + + const result = []; + for (let i = limit; i < items.length; i++) { + const item = items[i]; + if (item) { + result.push(item.name || item.abbr || 'anonymous'); + } + } + + tooltip.text = result.join('\n'); + } + + /** @private */ + __getLimit(items, itemsInView, maxItemsVisible) { + let limit = null; + // Handle max set to 0 or 1 + const adjustedMax = this.__getMax(maxItemsVisible); + if (maxItemsVisible != null && adjustedMax < items) { + limit = adjustedMax - 1; + } else if (itemsInView && itemsInView < items) { + limit = itemsInView; + } + + return Math.min(limit, this.__calculateAvatarsFitWidth()); + } + + /** @private */ + __getMax(maxItemsVisible) { + return Math.max(maxItemsVisible, MINIMUM_DISPLAYED_AVATARS); + } + + /** @private */ + __itemsChanged(items, oldItems) { + this.__setItemsInView(); + + let added = []; + let removed = []; + + const hasNewItems = Array.isArray(items); + const hasOldItems = Array.isArray(oldItems); + + if (hasOldItems) { + removed = oldItems.filter((item) => hasNewItems && !items.includes(item)); + } + + if (hasNewItems) { + added = items.filter((item) => hasOldItems && !oldItems.includes(item)); + } + + this.__announceItemsChange(added, removed); + } + + /** @private */ + __announceItemsChange(added, removed) { + let addedMsg = []; + let removedMsg = []; + if (added) { + addedMsg = added.map((user) => this.__getMessage(user, this.i18n.joined || '{user} joined')); + } + + if (removed) { + removedMsg = removed.map((user) => this.__getMessage(user, this.i18n.left || '{user} left')); + } + + const messages = removedMsg.concat(addedMsg); + if (messages.length > 0) { + announce(messages.join(', ')); + } + } + + /** @private */ + __i18nItemsChanged(i18n, items) { + if (i18n && i18n.activeUsers) { + const count = Array.isArray(items) ? items.length : 0; + const field = count === 1 ? 'one' : 'many'; + if (i18n.activeUsers[field]) { + this.setAttribute('aria-label', i18n.activeUsers[field].replace('{count}', count || 0)); + } + + this._avatars.forEach((avatar) => { + avatar.i18n = i18n; + }); + } + } + + /** @private */ + __openedChanged(opened, wasOpened) { + if (opened) { + if (!this._menuElement) { + this._menuElement = this.$.overlay.querySelector('vaadin-avatar-group-menu'); + } + + this._openedWithFocusRing = this._overflow.hasAttribute('focus-ring'); + + this._menuElement.focus(); + } else if (wasOpened) { + this._overflow.focus(); + if (this._openedWithFocusRing) { + this._overflow.setAttribute('focus-ring', ''); + } + } + this._overflow.setAttribute('aria-expanded', opened === true); + } + + /** @private */ + __overflowItemsChanged(items, oldItems) { + if (items || oldItems) { + this.$.overlay.requestContentUpdate(); + } + } + + /** @private */ + __setItemsInView() { + const avatars = this._avatars; + const items = this.items; + + // Always show at least two avatars + if (!items || !avatars || avatars.length < 3) { + return; + } + + let result = this.__calculateAvatarsFitWidth(); + + // Only show overlay if two or more avatars don't fit + if (result === items.length - 1) { + result = items.length; + } + + // Close overlay if all avatars become visible + if (result >= items.length && this._opened) { + this.$.overlay.close(); + // FIXME: hack to avoid jump before closing + this.$.overlay._flushAnimation('closing'); + } + + // Reserve space for overflow avatar + this.__itemsInView = result; + } + + /** @private */ + __calculateAvatarsFitWidth() { + if (!this.shadowRoot || this._avatars.length < MINIMUM_DISPLAYED_AVATARS) { + return MINIMUM_DISPLAYED_AVATARS; + } + + const avatars = this._avatars; + + // Assume all the avatars have the same width + const avatarWidth = avatars[0].clientWidth; + + // Take negative margin into account + const { marginLeft, marginRight } = getComputedStyle(avatars[1]); + + const offset = this.__isRTL + ? parseInt(marginRight, 0) - parseInt(marginLeft, 0) + : parseInt(marginLeft, 0) - parseInt(marginRight, 0); + + return Math.floor((this.$.container.offsetWidth - avatarWidth) / (avatarWidth + offset)); + } + }; diff --git a/packages/avatar-group/src/vaadin-avatar-group.d.ts b/packages/avatar-group/src/vaadin-avatar-group.d.ts index 811048caee..8e6d58a812 100644 --- a/packages/avatar-group/src/vaadin-avatar-group.d.ts +++ b/packages/avatar-group/src/vaadin-avatar-group.d.ts @@ -6,28 +6,10 @@ import { AvatarI18n } from '@vaadin/avatar/src/vaadin-avatar.js'; import { ControllerMixin } from '@vaadin/component-base/src/controller-mixin.js'; import { ElementMixin } from '@vaadin/component-base/src/element-mixin.js'; -import { OverlayClassMixin } from '@vaadin/component-base/src/overlay-class-mixin.js'; -import { ResizeMixin } from '@vaadin/component-base/src/resize-mixin.js'; import { ThemableMixin } from '@vaadin/vaadin-themable-mixin/vaadin-themable-mixin.js'; +import { AvatarGroupI18n, AvatarGroupItem, AvatarGroupMixin } from './vaadin-avatar-group-mixin.js'; -export { AvatarI18n }; - -export interface AvatarGroupI18n extends AvatarI18n { - activeUsers: { - one: string; - many: string; - }; - joined: string; - left: string; -} - -export interface AvatarGroupItem { - name?: string; - abbr?: string; - img?: string; - colorIndex?: number; - className?: string; -} +export { AvatarGroupI18n, AvatarGroupItem, AvatarI18n }; /** * `` is a Web Component providing avatar group displaying functionality. @@ -69,73 +51,7 @@ export interface AvatarGroupItem { * - `` - has the same API as [``](#/elements/vaadin-list-box). * - `` - has the same API as [``](#/elements/vaadin-item). */ -declare class AvatarGroup extends ResizeMixin( - OverlayClassMixin(ElementMixin(ThemableMixin(ControllerMixin(HTMLElement)))), -) { - /** - * An array containing the items which will be stamped as avatars. - * - * The items objects allow to configure [`name`](#/elements/vaadin-avatar#property-name), - * [`abbr`](#/elements/vaadin-avatar#property-abbr), [`img`](#/elements/vaadin-avatar#property-img) - * and [`colorIndex`](#/elements/vaadin-avatar#property-colorIndex) properties on the - * stamped avatars, and set `className` to provide CSS class names. - * - * #### Example - * - * ```js - * group.items = [ - * { - * name: 'User name', - * img: 'url-to-image.png', - * className: 'even' - * }, - * { - * abbr: 'JD', - * colorIndex: 1, - * className: 'odd' - * }, - * ]; - * ``` - */ - items: AvatarGroupItem[] | undefined; - - /** - * The maximum number of avatars to display. By default, all the avatars are displayed. - * When _maxItemsVisible_ is set, the overflowing avatars are grouped into one avatar with - * a dropdown. Setting 0 or 1 has no effect so there are always at least two avatars visible. - * @attr {number} max-items-visible - */ - maxItemsVisible: number | null | undefined; - - /** - * The object used to localize this component. - * To change the default localization, replace the entire - * _i18n_ object or just the property you want to modify. - * - * The object has the following JSON structure and default values: - * ``` - * { - * // Translation of the anonymous user avatar tooltip. - * anonymous: 'anonymous', - * // Translation of the avatar group accessible label. - * // {count} is replaced with the actual count of users. - * activeUsers: { - * one: 'Currently one active user', - * many: 'Currently {count} active users' - * }, - * // Screen reader announcement when user joins group. - * // {user} is replaced with the name or abbreviation. - * // When neither is set, "anonymous" is used instead. - * joined: '{user} joined', - * // Screen reader announcement when user leaves group. - * // {user} is replaced with the name or abbreviation. - * // When neither is set, "anonymous" is used instead. - * left: '{user} left' - * } - * ``` - */ - i18n: AvatarGroupI18n; -} +declare class AvatarGroup extends AvatarGroupMixin(ElementMixin(ThemableMixin(ControllerMixin(HTMLElement)))) {} declare global { interface HTMLElementTagNameMap { diff --git a/packages/avatar-group/src/vaadin-avatar-group.js b/packages/avatar-group/src/vaadin-avatar-group.js index 65e364225a..c0ac00964e 100644 --- a/packages/avatar-group/src/vaadin-avatar-group.js +++ b/packages/avatar-group/src/vaadin-avatar-group.js @@ -7,24 +7,16 @@ import '@vaadin/avatar/src/vaadin-avatar.js'; import './vaadin-avatar-group-menu.js'; import './vaadin-avatar-group-menu-item.js'; import './vaadin-avatar-group-overlay.js'; -import { afterNextRender } from '@polymer/polymer/lib/utils/render-status.js'; -import { html as legacyHtml, PolymerElement } from '@polymer/polymer/polymer-element.js'; -import { html, render } from 'lit'; -import { ifDefined } from 'lit/directives/if-defined.js'; -import { announce } from '@vaadin/a11y-base/src/announce.js'; +import { html, PolymerElement } from '@polymer/polymer/polymer-element.js'; import { ControllerMixin } from '@vaadin/component-base/src/controller-mixin.js'; import { defineCustomElement } from '@vaadin/component-base/src/define.js'; import { ElementMixin } from '@vaadin/component-base/src/element-mixin.js'; -import { OverlayClassMixin } from '@vaadin/component-base/src/overlay-class-mixin.js'; -import { ResizeMixin } from '@vaadin/component-base/src/resize-mixin.js'; -import { SlotController } from '@vaadin/component-base/src/slot-controller.js'; import { registerStyles, ThemableMixin } from '@vaadin/vaadin-themable-mixin/vaadin-themable-mixin.js'; +import { AvatarGroupMixin } from './vaadin-avatar-group-mixin.js'; import { avatarGroupStyles } from './vaadin-avatar-group-styles.js'; registerStyles('vaadin-avatar-group', avatarGroupStyles, { moduleId: 'vaadin-avatar-group-styles' }); -const MINIMUM_DISPLAYED_AVATARS = 2; - /** * `` is a Web Component providing avatar group displaying functionality. * @@ -69,13 +61,12 @@ const MINIMUM_DISPLAYED_AVATARS = 2; * @extends HTMLElement * @mixes ControllerMixin * @mixes ElementMixin - * @mixes OverlayClassMixin + * @mixes AvatarGroupMixin * @mixes ThemableMixin - * @mixes ResizeMixin */ -class AvatarGroup extends ResizeMixin(OverlayClassMixin(ElementMixin(ThemableMixin(ControllerMixin(PolymerElement))))) { +class AvatarGroup extends AvatarGroupMixin(ElementMixin(ThemableMixin(ControllerMixin(PolymerElement)))) { static get template() { - return legacyHtml` + return html`
@@ -93,518 +84,6 @@ class AvatarGroup extends ResizeMixin(OverlayClassMixin(ElementMixin(ThemableMix static get is() { return 'vaadin-avatar-group'; } - - static get properties() { - return { - /** - * An array containing the items which will be stamped as avatars. - * - * The items objects allow to configure [`name`](#/elements/vaadin-avatar#property-name), - * [`abbr`](#/elements/vaadin-avatar#property-abbr), [`img`](#/elements/vaadin-avatar#property-img) - * and [`colorIndex`](#/elements/vaadin-avatar#property-colorIndex) properties on the - * stamped avatars, and set `className` to provide CSS class names. - * - * #### Example - * - * ```js - * group.items = [ - * { - * name: 'User name', - * img: 'url-to-image.png', - * className: 'even' - * }, - * { - * abbr: 'JD', - * colorIndex: 1, - * className: 'odd' - * }, - * ]; - * ``` - * - * @type {!Array | undefined} - */ - items: { - type: Array, - observer: '__itemsChanged', - }, - - /** - * The maximum number of avatars to display. By default, all the avatars are displayed. - * When _maxItemsVisible_ is set, the overflowing avatars are grouped into one avatar with - * a dropdown. Setting 0 or 1 has no effect so there are always at least two avatars visible. - * @attr {number} max-items-visible - */ - maxItemsVisible: { - type: Number, - }, - - /** - * The object used to localize this component. - * To change the default localization, replace the entire - * _i18n_ object or just the property you want to modify. - * - * The object has the following JSON structure and default values: - * ``` - * { - * // Translation of the anonymous user avatar tooltip. - * anonymous: 'anonymous', - * // Translation of the avatar group accessible label. - * // {count} is replaced with the actual count of users. - * activeUsers: { - * one: 'Currently one active user', - * many: 'Currently {count} active users' - * }, - * // Screen reader announcement when user joins group. - * // {user} is replaced with the name or abbreviation. - * // When neither is set, "anonymous" is used instead. - * joined: '{user} joined', - * // Screen reader announcement when user leaves group. - * // {user} is replaced with the name or abbreviation. - * // When neither is set, "anonymous" is used instead. - * left: '{user} left' - * } - * ``` - * @type {!AvatarGroupI18n} - * @default {English/US} - */ - i18n: { - type: Object, - value: () => { - return { - anonymous: 'anonymous', - activeUsers: { - one: 'Currently one active user', - many: 'Currently {count} active users', - }, - joined: '{user} joined', - left: '{user} left', - }; - }, - }, - - /** @private */ - _avatars: { - type: Array, - value: () => [], - }, - - /** @private */ - __itemsInView: { - type: Number, - value: null, - }, - - /** @private */ - _overflow: { - type: Object, - }, - - /** @private */ - _overflowItems: { - type: Array, - observer: '__overflowItemsChanged', - computed: '__computeOverflowItems(items, __itemsInView, maxItemsVisible)', - }, - - /** @private */ - _overflowTooltip: { - type: Object, - }, - - /** @private */ - _opened: { - type: Boolean, - observer: '__openedChanged', - }, - }; - } - - static get observers() { - return [ - '__i18nItemsChanged(i18n, items)', - '__updateAvatarsTheme(_overflow, _avatars, _theme)', - '__updateAvatars(items, __itemsInView, maxItemsVisible, _overflow, i18n)', - '__updateOverflowAvatar(_overflow, items, __itemsInView, maxItemsVisible)', - '__updateOverflowTooltip(_overflowTooltip, items, __itemsInView, maxItemsVisible)', - ]; - } - - /** @protected */ - ready() { - super.ready(); - - this._overflowController = new SlotController(this, 'overflow', 'vaadin-avatar', { - initializer: (overflow) => { - overflow.setAttribute('aria-haspopup', 'menu'); - overflow.setAttribute('aria-expanded', 'false'); - overflow.addEventListener('click', (e) => this._onOverflowClick(e)); - overflow.addEventListener('keydown', (e) => this._onOverflowKeyDown(e)); - - const tooltip = document.createElement('vaadin-tooltip'); - tooltip.setAttribute('slot', 'tooltip'); - overflow.appendChild(tooltip); - - this._overflow = overflow; - this._overflowTooltip = tooltip; - }, - }); - this.addController(this._overflowController); - - const overlay = this.$.overlay; - overlay.renderer = this.__overlayRenderer.bind(this); - this._overlayElement = overlay; - - afterNextRender(this, () => { - this.__setItemsInView(); - }); - } - - /** @protected */ - disconnectedCallback() { - super.disconnectedCallback(); - - this._opened = false; - } - - /** @private */ - __getMessage(user, action) { - return action.replace('{user}', user.name || user.abbr || this.i18n.anonymous); - } - - /** - * Renders items when they are provided by the `items` property and clears the content otherwise. - * @param {!HTMLElement} root - * @param {!Select} _select - * @private - */ - __overlayRenderer(root) { - let menu = root.firstElementChild; - if (!menu) { - menu = document.createElement('vaadin-avatar-group-menu'); - menu.addEventListener('keydown', (event) => this._onListKeyDown(event)); - root.appendChild(menu); - } - - menu.textContent = ''; - - if (!this._overflowItems) { - return; - } - - this._overflowItems.forEach((item) => { - menu.appendChild(this.__createItemElement(item)); - }); - } - - /** @private */ - __createItemElement(item) { - const itemElement = document.createElement('vaadin-avatar-group-menu-item'); - - const avatar = document.createElement('vaadin-avatar'); - itemElement.appendChild(avatar); - - avatar.setAttribute('aria-hidden', 'true'); - avatar.setAttribute('tabindex', '-1'); - avatar.i18n = this.i18n; - - if (this._theme) { - avatar.setAttribute('theme', this._theme); - } - - avatar.name = item.name; - avatar.abbr = item.abbr; - avatar.img = item.img; - avatar.colorIndex = item.colorIndex; - if (item.className) { - avatar.className = item.className; - } - - if (item.name) { - const text = document.createTextNode(item.name); - itemElement.appendChild(text); - } - - return itemElement; - } - - /** @private */ - _onOverflowClick(e) { - e.stopPropagation(); - if (this._opened) { - this.$.overlay.close(); - } else if (!e.defaultPrevented) { - this._opened = true; - } - } - - /** @private */ - _onOverflowKeyDown(e) { - if (!this._opened) { - if (/^(Enter|SpaceBar|\s)$/u.test(e.key)) { - e.preventDefault(); - this._opened = true; - } - } - } - - /** @private */ - _onListKeyDown(event) { - if (event.key === 'Escape' || event.key === 'Tab') { - this._opened = false; - } - } - - /** - * @protected - * @override - */ - _onResize() { - this.__setItemsInView(); - } - - /** @private */ - _onVaadinOverlayClose(e) { - if (e.detail.sourceEvent && e.detail.sourceEvent.composedPath().includes(this)) { - e.preventDefault(); - } - } - - /** @private */ - __renderAvatars(items) { - render( - html` - ${items.map( - (item) => html` - - `, - )} - `, - this, - { renderBefore: this._overflow }, - ); - } - - /** @private */ - __updateAvatars(items, itemsInView, maxItemsVisible, overflow) { - if (!overflow || !Array.isArray(items)) { - return; - } - - const limit = this.__getLimit(items.length, itemsInView, maxItemsVisible); - - this.__renderAvatars(limit ? items.slice(0, limit) : items); - - this._avatars = [...this.querySelectorAll('vaadin-avatar')]; - } - - /** @private */ - __computeOverflowItems(items, itemsInView, maxItemsVisible) { - const count = Array.isArray(items) ? items.length : 0; - const limit = this.__getLimit(count, itemsInView, maxItemsVisible); - return limit ? items.slice(limit) : []; - } - - /** @private */ - __updateOverflowAvatar(overflow, items, itemsInView, maxItemsVisible) { - if (overflow) { - const count = Array.isArray(items) ? items.length : 0; - const maxReached = maxItemsVisible != null && count > this.__getMax(maxItemsVisible); - - overflow.abbr = `+${count - this.__getLimit(count, itemsInView, maxItemsVisible)}`; - overflow.toggleAttribute('hidden', !maxReached && !(itemsInView && itemsInView < count)); - } - } - - /** @private */ - __updateAvatarsTheme(overflow, avatars, theme) { - if (overflow) { - [overflow, ...avatars].forEach((avatar) => { - if (theme) { - avatar.setAttribute('theme', theme); - } else { - avatar.removeAttribute('theme'); - } - }); - } - } - - /** @private */ - __updateOverflowTooltip(tooltip, items, itemsInView, maxItemsVisible) { - if (!tooltip || !Array.isArray(items)) { - return; - } - - const limit = this.__getLimit(items.length, itemsInView, maxItemsVisible); - if (limit == null) { - return; - } - - const result = []; - for (let i = limit; i < items.length; i++) { - const item = items[i]; - if (item) { - result.push(item.name || item.abbr || 'anonymous'); - } - } - - tooltip.text = result.join('\n'); - } - - /** @private */ - __getLimit(items, itemsInView, maxItemsVisible) { - let limit = null; - // Handle max set to 0 or 1 - const adjustedMax = this.__getMax(maxItemsVisible); - if (maxItemsVisible != null && adjustedMax < items) { - limit = adjustedMax - 1; - } else if (itemsInView && itemsInView < items) { - limit = itemsInView; - } - - return Math.min(limit, this.__calculateAvatarsFitWidth()); - } - - /** @private */ - __getMax(maxItemsVisible) { - return Math.max(maxItemsVisible, MINIMUM_DISPLAYED_AVATARS); - } - - /** @private */ - __itemsChanged(items, oldItems) { - this.__setItemsInView(); - - let added = []; - let removed = []; - - const hasNewItems = Array.isArray(items); - const hasOldItems = Array.isArray(oldItems); - - if (hasOldItems) { - removed = oldItems.filter((item) => hasNewItems && !items.includes(item)); - } - - if (hasNewItems) { - added = items.filter((item) => hasOldItems && !oldItems.includes(item)); - } - - this.__announceItemsChange(added, removed); - } - - /** @private */ - __announceItemsChange(added, removed) { - let addedMsg = []; - let removedMsg = []; - if (added) { - addedMsg = added.map((user) => this.__getMessage(user, this.i18n.joined || '{user} joined')); - } - - if (removed) { - removedMsg = removed.map((user) => this.__getMessage(user, this.i18n.left || '{user} left')); - } - - const messages = removedMsg.concat(addedMsg); - if (messages.length > 0) { - announce(messages.join(', ')); - } - } - - /** @private */ - __i18nItemsChanged(i18n, items) { - if (i18n && i18n.activeUsers) { - const count = Array.isArray(items) ? items.length : 0; - const field = count === 1 ? 'one' : 'many'; - if (i18n.activeUsers[field]) { - this.setAttribute('aria-label', i18n.activeUsers[field].replace('{count}', count || 0)); - } - - this._avatars.forEach((avatar) => { - avatar.i18n = i18n; - }); - } - } - - /** @private */ - __openedChanged(opened, wasOpened) { - if (opened) { - if (!this._menuElement) { - this._menuElement = this.$.overlay.querySelector('vaadin-avatar-group-menu'); - } - - this._openedWithFocusRing = this._overflow.hasAttribute('focus-ring'); - - this._menuElement.focus(); - } else if (wasOpened) { - this._overflow.focus(); - if (this._openedWithFocusRing) { - this._overflow.setAttribute('focus-ring', ''); - } - } - this._overflow.setAttribute('aria-expanded', opened === true); - } - - /** @private */ - __overflowItemsChanged(items, oldItems) { - if (items || oldItems) { - this.$.overlay.requestContentUpdate(); - } - } - - /** @private */ - __setItemsInView() { - const avatars = this._avatars; - const items = this.items; - - // Always show at least two avatars - if (!items || !avatars || avatars.length < 3) { - return; - } - - let result = this.__calculateAvatarsFitWidth(); - - // Only show overlay if two or more avatars don't fit - if (result === items.length - 1) { - result = items.length; - } - - // Close overlay if all avatars become visible - if (result >= items.length && this._opened) { - this.$.overlay.close(); - // FIXME: hack to avoid jump before closing - this.$.overlay._flushAnimation('closing'); - } - - // Reserve space for overflow avatar - this.__itemsInView = result; - } - - /** @private */ - __calculateAvatarsFitWidth() { - if (!this.shadowRoot || this._avatars.length < MINIMUM_DISPLAYED_AVATARS) { - return MINIMUM_DISPLAYED_AVATARS; - } - - const avatars = this._avatars; - - // Assume all the avatars have the same width - const avatarWidth = avatars[0].clientWidth; - - // Take negative margin into account - const { marginLeft, marginRight } = getComputedStyle(avatars[1]); - - const offset = this.__isRTL - ? parseInt(marginRight, 0) - parseInt(marginLeft, 0) - : parseInt(marginLeft, 0) - parseInt(marginRight, 0); - - return Math.floor((this.$.container.offsetWidth - avatarWidth) / (avatarWidth + offset)); - } } defineCustomElement(AvatarGroup); diff --git a/packages/avatar-group/test/typings/avatar-group.types.ts b/packages/avatar-group/test/typings/avatar-group.types.ts index f943e3b4c6..330c5ab4da 100644 --- a/packages/avatar-group/test/typings/avatar-group.types.ts +++ b/packages/avatar-group/test/typings/avatar-group.types.ts @@ -4,6 +4,7 @@ import type { ElementMixinClass } from '@vaadin/component-base/src/element-mixin import type { OverlayClassMixinClass } from '@vaadin/component-base/src/overlay-class-mixin.js'; import type { ResizeMixinClass } from '@vaadin/component-base/src/resize-mixin.js'; import type { ThemableMixinClass } from '@vaadin/vaadin-themable-mixin/vaadin-themable-mixin.js'; +import type { AvatarGroupMixinClass } from '../../src/vaadin-avatar-group-mixin.js'; import type { AvatarGroupI18n, AvatarGroupItem } from '../../vaadin-avatar-group.js'; const assertType = (value: TExpected) => value; @@ -25,6 +26,7 @@ assertType(item.colorIndex); assertType(item.className); // Mixins +assertType(group); assertType(group); assertType(group); assertType(group);