diff --git a/packages/mgt-chat/src/components/Chat/Chat.tsx b/packages/mgt-chat/src/components/Chat/Chat.tsx index f6918853ab..b9067b8c29 100644 --- a/packages/mgt-chat/src/components/Chat/Chat.tsx +++ b/packages/mgt-chat/src/components/Chat/Chat.tsx @@ -134,8 +134,8 @@ export const Chat = ({ chatId }: IMgtChatProps) => { ); }} diff --git a/packages/mgt-components/src/components/mgt-person/mgt-person.tests.ts b/packages/mgt-components/src/components/mgt-person/mgt-person.tests.ts index 35d807338e..1149fa539e 100644 --- a/packages/mgt-components/src/components/mgt-person/mgt-person.tests.ts +++ b/packages/mgt-components/src/components/mgt-person/mgt-person.tests.ts @@ -7,11 +7,13 @@ import { fixture, html, expect, waitUntil } from '@open-wc/testing'; import { MockProvider, Providers } from '@microsoft/mgt-element'; import { registerMgtPersonComponent } from './mgt-person'; +import { useConfig } from '../../graph/graph.presence.mock'; +import { PresenceService } from '../../utils/PresenceService'; describe('mgt-person - tests', () => { before(() => { registerMgtPersonComponent(); - Providers.globalProvider = new MockProvider(true); + Providers.globalProvider = new MockProvider(true, [{ id: '48d31887-5fad-4d73-a9f5-3c356e68a038' }]); }); it('should render', async () => { @@ -31,6 +33,19 @@ describe('mgt-person - tests', () => { ); }); + it('unknown user should render with a default icon', async () => { + // purposely throws browser error "Error: Invalid userId" + const person = await fixture( + html`` + ); + await waitUntil(() => person.shadowRoot.querySelector('svg'), 'no svg was populated'); + await expect(person).shadowDom.to.equal( + ` + + ` + ); + }); + it('should pop up a flyout on click', async () => { const person = await fixture(html``); await waitUntil(() => person.shadowRoot.querySelector('img'), 'mgt-person did not update'); @@ -176,4 +191,97 @@ describe('mgt-person - tests', () => { })}' view="twoLines">`); await expect(person.shadowRoot.querySelector('span.initials')).lightDom.to.equal('FV'); }); + + it('should support a change in presence', async () => { + useConfig({ + default: 'Available', + 3: 'Busy', + 4: 'Busy', + 5: 'Busy' + }); + + PresenceService.config.initial = 100; + PresenceService.config.refresh = 200; + + const person = await fixture( + html`` + ); + + const match = (status: string) => `
+
+ Photo for Megan Bowen + + +
+
+ + +
+
`; + + // starts as Offline + await waitUntil(() => person.shadowRoot.querySelector('span.presence-wrapper'), 'no presence span'); + await expect(person).shadowDom.to.equal(match('Offline'), { ignoreAttributes: ['src'] }); + + // changes to Available on initial + await waitUntil(() => person.shadowRoot.querySelector('span[title="Available"]'), 'did not update to available', { + timeout: 4000 + }); + await expect(person).shadowDom.to.equal(match('Available'), { ignoreAttributes: ['src'] }); + + // changes to Busy on refresh + await waitUntil(() => person.shadowRoot.querySelector('span[title="Busy"]'), 'did not update to busy', { + timeout: 4000 + }); + await expect(person).shadowDom.to.equal(match('Busy'), { ignoreAttributes: ['src'] }); + }); + + it('should not update presence on error', async () => { + useConfig({ + default: 'Available', + 0: new Error('purposeful error'), + 1: new Error('purposeful error'), + 2: new Error('purposeful error') + }); + + PresenceService.config.initial = 1000; + PresenceService.config.refresh = 2000; + + const person = await fixture( + html`` + ); + await waitUntil(() => person.shadowRoot.querySelector('img'), 'mgt-person did not update'); + + const match = (status: string) => `
+
+ Photo for Megan Bowen + + +
+
+ + +
+
`; + + // starts Offline during errors + await waitUntil(() => person.shadowRoot.querySelector('span.presence-wrapper'), 'no presence span'); + await expect(person).shadowDom.to.equal(match('Offline'), { ignoreAttributes: ['src'] }); + + // changes to Available eventually + await waitUntil(() => person.shadowRoot.querySelector('span[title="Available"]'), 'did not update to available', { + timeout: 4000 + }); + await expect(person).shadowDom.to.equal(match('Available'), { ignoreAttributes: ['src'] }); + }); }); diff --git a/packages/mgt-components/src/components/mgt-person/mgt-person.ts b/packages/mgt-components/src/components/mgt-person/mgt-person.ts index 5f87c166c8..d5c2f55f65 100644 --- a/packages/mgt-components/src/components/mgt-person/mgt-person.ts +++ b/packages/mgt-components/src/components/mgt-person/mgt-person.ts @@ -28,6 +28,7 @@ import { isUser, isContact } from '../../graph/entityType'; import { ifDefined } from 'lit/directives/if-defined.js'; import { buildComponentName, registerComponent } from '@microsoft/mgt-element'; import { IExpandable, IHistoryClearer } from '../mgt-person-card/types'; +import { PresenceService, PresenceAwareComponent } from '../../utils/PresenceService'; export { PersonCardInteraction } from '../PersonCardInteraction'; @@ -107,7 +108,7 @@ export const registerMgtPersonComponent = () => { * * @cssprop --person-details-wrapper-width - {Length} the minimum width of the details section. Default is 168px. */ -export class MgtPerson extends MgtTemplatedComponent { +export class MgtPerson extends MgtTemplatedComponent implements PresenceAwareComponent { /** * Array of styles to apply to the element. The styles should be defined * using the `css` tag function. @@ -240,6 +241,17 @@ export class MgtPerson extends MgtTemplatedComponent { }) public showPresence: boolean; + /** + * determines if person component refreshes presence + * + * @type {boolean} + */ + @property({ + attribute: 'disable-presence-refresh', + type: Boolean + }) + public disablePresenceRefresh; + /** * determines person component avatar size and apply presence badge accordingly. * Default is "auto". When you set the view > 1, it will default to "auto". @@ -563,6 +575,19 @@ export class MgtPerson extends MgtTemplatedComponent { this.verticalLayout = false; } + /** + * Unregisters from presence service if necessary. Note, it does not cause an error + * if the component was not registered. + * + * @memberof MgtAgenda + */ + public disconnectedCallback() { + if (this.showPresence && !this.disablePresenceRefresh) { + PresenceService.unregister(this); + } + super.disconnectedCallback(); + } + /** * Invoked on each update to perform rendering tasks. This method must return * a lit-html TemplateResult. Setting properties inside this method will *not* @@ -639,7 +664,22 @@ export class MgtPerson extends MgtTemplatedComponent { * @memberof MgtPerson */ protected renderLoading(): TemplateResult { - return this.renderTemplate('loading', null) || html``; + const loadingTemplate = this.renderTemplate('loading', null); + if (loadingTemplate) { + return loadingTemplate; + } + + const avatarClasses = { + 'avatar-icon': true, + vertical: this.isVertical(), + small: !this.isLargeAvatar(), + threeLines: this.isThreeLines(), + fourLines: this.isFourLines() + }; + + return html` + ${this.renderLoadingIcon()} + `; } /** @@ -677,8 +717,19 @@ export class MgtPerson extends MgtTemplatedComponent { }; return html` - - `; + ${this.renderPersonIcon()} + `; + } + + /** + * Render a loading icon. + * + * @protected + * @returns + * @memberof MgtPerson + */ + protected renderLoadingIcon() { + return getSvg(SvgIcon.Loading); } /** @@ -1167,14 +1218,16 @@ export class MgtPerson extends MgtTemplatedComponent { details = this.personDetailsInternal || this.personDetails || this.fallbackDetails; - // populate presence const defaultPresence: Presence = { activity: 'Offline', availability: 'Offline', id: null }; - if (this.showPresence && !this.personPresence && !this._fetchedPresence) { + if (this.showPresence && !this.disablePresenceRefresh) { + this._fetchedPresence = defaultPresence; + PresenceService.register(this); + } else if (this.showPresence && !this.personPresence && !this._fetchedPresence) { try { if (details) { // setting userId to 'me' ensures only the presence.read permission is required @@ -1375,4 +1428,26 @@ export class MgtPerson extends MgtTemplatedComponent { flyout.open(); } }; + + /** + * gets the id of the person that presence updates are needed for + * + * @memberof MgtPerson + * @implements {PresenceAwareComponent} + * @returns {string | undefined} + **/ + public get presenceId(): string | undefined { + return this.personDetailsInternal?.id || this.personDetails?.id || this.fallbackDetails?.id; + } + + /** + * fires when the presence for the user is changed + * + * @memberof MgtPerson + * @implements {PresenceAwareComponent} + * @param {Presence} [presence] + **/ + public onPresenceChange(presence: Presence): void { + this.personPresence = presence; + } } diff --git a/packages/mgt-components/src/exports.ts b/packages/mgt-components/src/exports.ts index fe9e325d0c..3b96f13624 100644 --- a/packages/mgt-components/src/exports.ts +++ b/packages/mgt-components/src/exports.ts @@ -10,6 +10,7 @@ export * from './components/preview'; export * from './graph/types'; export * from './graph/graph.userWithPhoto'; export * from './graph/graph.photos'; +export * from './graph/graph.presence'; export * from './graph/cacheStores'; export * from './styles/theme-manager'; export * from './graph/entityType'; diff --git a/packages/mgt-components/src/graph/graph.presence.mock.ts b/packages/mgt-components/src/graph/graph.presence.mock.ts new file mode 100644 index 0000000000..a753fe97a7 --- /dev/null +++ b/packages/mgt-components/src/graph/graph.presence.mock.ts @@ -0,0 +1,55 @@ +/** + * ------------------------------------------------------------------------------------------- + * Copyright (c) Microsoft Corporation. All Rights Reserved. Licensed under the MIT License. + * See License in the project root for license information. + * ------------------------------------------------------------------------------------------- + */ + +import { IGraph } from '@microsoft/mgt-element'; +import { Presence } from '@microsoft/microsoft-graph-types'; +import { IDynamicPerson } from './types'; + +const globalObject = typeof window !== 'undefined' ? window : global; + +type status = 'Available' | 'Busy' | 'Away' | 'DoNotDisturb' | Error; + +type CallConfig = { + default: status; + [key: string]: status; +} & { + calls?: number; +}; + +export const useConfig = (config: CallConfig | status = 'Available') => { + if (typeof config === 'object') { + globalObject['unit-test:presence-config'] = config; + } else if (typeof config === 'string') { + globalObject['unit-test:presence-config'] = { default: config as status }; + } else { + throw new Error('Invalid config'); + } +}; + +export const getUserPresence = async (_graph: IGraph, _userId?: string): Promise => { + return Promise.reject(new Error()); +}; + +export const getUsersPresenceByPeople = async (graph: IGraph, people?: IDynamicPerson[], _ = true) => { + const config = (globalObject['unit-test:presence-config'] as CallConfig) || { default: 'Available' }; + const index = config.calls || 0; + const value = config[index] || config.default; + config.calls = index + 1; + if (value instanceof Error) { + return Promise.reject(value); + } else { + const peoplePresence: Record = {}; + for (const person of people || []) { + peoplePresence[person.id] = { + id: person.id, + availability: value, + activity: value + }; + } + return Promise.resolve(peoplePresence); + } +}; diff --git a/packages/mgt-components/src/graph/graph.presence.ts b/packages/mgt-components/src/graph/graph.presence.ts index 0feb78aa07..c428f6f07c 100644 --- a/packages/mgt-components/src/graph/graph.presence.ts +++ b/packages/mgt-components/src/graph/graph.presence.ts @@ -66,12 +66,13 @@ export const getUserPresence = async (graph: IGraph, userId?: string): Promise

{ +export const getUsersPresenceByPeople = async (graph: IGraph, people?: IDynamicPerson[], bypassCacheRead = false) => { if (!people || people.length === 0) { return {}; } @@ -90,10 +91,15 @@ export const getUsersPresenceByPeople = async (graph: IGraph, people?: IDynamicP const id = person.id; peoplePresence[id] = null; let presence: CachePresence; - if (getIsPresenceCacheEnabled()) { + if (!bypassCacheRead && getIsPresenceCacheEnabled()) { presence = await cache.getValue(id); } - if (getIsPresenceCacheEnabled() && presence && getPresenceInvalidationTime() > Date.now() - presence.timeCached) { + if ( + !bypassCacheRead && + getIsPresenceCacheEnabled() && + presence && + getPresenceInvalidationTime() > Date.now() - presence.timeCached + ) { peoplePresence[id] = JSON.parse(presence.presence) as Presence; } else { peoplePresenceToQuery.push(id); diff --git a/packages/mgt-components/src/graph/graph.userWithPhoto.mock.ts b/packages/mgt-components/src/graph/graph.userWithPhoto.mock.ts index 5c3f3d6127..f552989a62 100644 --- a/packages/mgt-components/src/graph/graph.userWithPhoto.mock.ts +++ b/packages/mgt-components/src/graph/graph.userWithPhoto.mock.ts @@ -21,23 +21,30 @@ export const getUserWithPhoto = async ( _userId?: string, _requestedProps?: string[] ): Promise => { - return Promise.resolve({ - '@odata.context': - 'https://graph.microsoft.com/v1.0/$metadata#users(businessPhones,displayName,givenName,jobTitle,department,mail,mobilePhone,officeLocation,preferredLanguage,surname,userPrincipalName,id,userType)/$entity', - businessPhones: ['+1 412 555 0109'], - displayName: 'Megan Bowen', - givenName: 'Megan', - jobTitle: 'Auditor', - department: 'Finance', - mail: 'MeganB@M365x214355.onmicrosoft.com', - mobilePhone: null, - officeLocation: '12/1110', - preferredLanguage: 'en-US', - surname: 'Bowen', - userPrincipalName: 'MeganB@M365x214355.onmicrosoft.com', - id: '48d31887-5fad-4d73-a9f5-3c356e68a038', - userType: 'Member', - personImage: - '' - }); + switch (_userId) { + case null: + case undefined: + case '48d31887-5fad-4d73-a9f5-3c356e68a038': + return Promise.resolve({ + '@odata.context': + 'https://graph.microsoft.com/v1.0/$metadata#users(businessPhones,displayName,givenName,jobTitle,department,mail,mobilePhone,officeLocation,preferredLanguage,surname,userPrincipalName,id,userType)/$entity', + businessPhones: ['+1 412 555 0109'], + displayName: 'Megan Bowen', + givenName: 'Megan', + jobTitle: 'Auditor', + department: 'Finance', + mail: 'MeganB@M365x214355.onmicrosoft.com', + mobilePhone: null, + officeLocation: '12/1110', + preferredLanguage: 'en-US', + surname: 'Bowen', + userPrincipalName: 'MeganB@M365x214355.onmicrosoft.com', + id: '48d31887-5fad-4d73-a9f5-3c356e68a038', + userType: 'Member', + personImage: + '' + }); + default: + return Promise.reject(new Error('Invalid userId')); + } }; diff --git a/packages/mgt-components/src/utils/PresenceService.ts b/packages/mgt-components/src/utils/PresenceService.ts new file mode 100644 index 0000000000..2fb2bbf4f8 --- /dev/null +++ b/packages/mgt-components/src/utils/PresenceService.ts @@ -0,0 +1,191 @@ +/** + * ------------------------------------------------------------------------------------------- + * Copyright (c) Microsoft Corporation. All Rights Reserved. Licensed under the MIT License. + * See License in the project root for license information. + * ------------------------------------------------------------------------------------------- + */ + +import { ProviderState, Providers, log, error } from '@microsoft/mgt-element'; +import { getUsersPresenceByPeople } from '../graph/graph.presence'; +import { Presence } from '@microsoft/microsoft-graph-types'; + +/** + * Holds the presence options. + * + * @export + * @interface PresenceConfig + */ +export interface PresenceConfig { + /** + * The maximum time limit before the presence is polled again. + * + * @type {number} + * @memberof PresenceConfig + */ + refresh: number; + + /** + * The maximum time limit after a component is registered for it to be queried. + * + * @type {number} + * @memberof PresenceConfig + */ + initial: number; +} + +export interface PresenceAwareComponent { + get presenceId(): string | undefined; + onPresenceChange(presence: Presence): void; +} + +/** + * Class in charge of managing presence across all users and components. + * + * @export + * @class PresenceService + */ +export class PresenceService { + private static readonly components = new Set(); + private static initPromise: Promise | null = null; + private static registerPromise: (value: unknown) => void; + private static isStopped = false; + + private static readonly presenceConfig: PresenceConfig = { + initial: 1000, + refresh: 30000 + }; + + /** + * Returns the presenceConfig object. + * + * @readonly + * @static + * @type {PresenceConfig} + * @memberof PresenceService + */ + public static get config(): PresenceConfig { + return this.presenceConfig; + } + + /** + * Registers a component with the presence service so that it can receive updates. + * + * @static + * @param {MgtPerson} component + * @memberof PresenceService + */ + public static register(component: PresenceAwareComponent) { + if (this.initPromise === null || this.initPromise === undefined) { + this.initPromise = this.init(); + } + + // after init, add component + this.initPromise.then( + () => { + setTimeout(() => { + if (this.registerPromise) { + this.registerPromise(null); + } + }, this.config.initial); + this.components.add(component); + }, + e => error('the PresenceService could not be initialized; presence will not be updated', e) + ); + } + + /** + * Unregisters a component with the presence service so that it no longer receives updates. + * There is no error if the component was not registered. + * + * @static + * @param {MgtPerson} component + * @memberof PresenceService + */ + public static unregister(component: PresenceAwareComponent) { + this.components.delete(component); + } + + /* + * Stops the presence service. + * + * @static + * @memberof PresenceService + */ + public static stop() { + this.isStopped = true; + } + + /** + * Starts a timer for notifying the components of presence updates. + * + * @private + * @static + * @memberof PresenceService + */ + private static async init(): Promise { + const loop = async () => { + while (!this.isStopped) { + // wait for next interval + const registerPromise = new Promise(resolve => (this.registerPromise = resolve)); + const timeoutPromise = new Promise(resolve => setTimeout(resolve, this.config.refresh)); + await Promise.race([registerPromise, timeoutPromise]); + + // get a valid graph provider + const provider = Providers.globalProvider; + if (!provider || provider.state === ProviderState.Loading || provider.state === ProviderState.SignedOut) { + continue; + } + + // log attempt + log(`updating presence for ${this.components.size} components.`); + const presenceByUserId = new Map(); + + // get full list of users + for (const component of this.components) { + const id = component.presenceId; + if (id && !presenceByUserId.has(id)) { + presenceByUserId.set(id, undefined); + } + } + + // get updated presence for all users + const listOfIds: string[] = []; + try { + const presences = await getUsersPresenceByPeople( + provider.graph, + Array.from(presenceByUserId.keys()).map(userId => ({ id: userId })), + true // bypassCacheRead + ); + if (presences) { + for (const presence of Object.values(presences)) { + if (presence.id) { + presenceByUserId.set(presence.id, presence); + listOfIds.push(`${presence.id}=${presence.availability}/${presence.activity}`); + } + } + } + } catch (e) { + error(`could not update presence`, e); + continue; + } + + // log results + log(`updated presence for ${listOfIds.length} users.`, listOfIds); + + // update components + for (const component of this.components) { + const id = component.presenceId; + if (id) { + const presence = presenceByUserId.get(id); + if (presence) { + component.onPresenceChange(presence); + } + } + } + } + }; + + void loop(); // start loop without waiting for end + return Promise.resolve(); + } +} diff --git a/packages/mgt-components/src/utils/SvgHelper.ts b/packages/mgt-components/src/utils/SvgHelper.ts index 67be44d881..693ad10831 100644 --- a/packages/mgt-components/src/utils/SvgHelper.ts +++ b/packages/mgt-components/src/utils/SvgHelper.ts @@ -299,7 +299,12 @@ export enum SvgIcon { PresenceAway, PresenceOofAway, PresenceOffline, - PresenceStatusUnknown + PresenceStatusUnknown, + + /** + * Loading icon + */ + Loading } import { html } from 'lit'; @@ -772,5 +777,11 @@ export const getSvg = (svgIcon: SvgIcon, color?: string) => { `; + + case SvgIcon.Loading: + return html` + `; } }; diff --git a/packages/mgt-react/src/generated/person.ts b/packages/mgt-react/src/generated/person.ts index 7537ab1c62..cae3e7bae1 100644 --- a/packages/mgt-react/src/generated/person.ts +++ b/packages/mgt-react/src/generated/person.ts @@ -15,6 +15,7 @@ export type PersonProps = { userId?: string; usage?: string; showPresence?: boolean; + disablePresenceRefresh?: boolean; avatarSize?: AvatarSize; personDetails?: IDynamicPerson; personImage?: string; @@ -29,6 +30,7 @@ export type PersonProps = { line3Property?: string; line4Property?: string; view?: ViewType | PersonViewType; + presenceId?: string | undefined; templateContext?: TemplateContext; mediaQuery?: ComponentMediaQuery; line1clicked?: (e: CustomEvent) => void; diff --git a/stories/components/person/person.properties.stories.js b/stories/components/person/person.properties.stories.js index b150e19a25..19da739347 100644 --- a/stories/components/person/person.properties.stories.js +++ b/stories/components/person/person.properties.stories.js @@ -171,6 +171,11 @@ export const personPresence = () => html` > `; +export const myPresence = () => html` +

Change your presence in Teams to see the presence reflected here...
+ +`; + export const personPresenceDisplayAll = () => html` +
Refreshing presence
+ +
+
Refreshing presence (fake)
+ +
+
Refreshing presence disabled
+ + `; + export const moreExamples = () => html`