= {};
+ 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`