diff --git a/.changeset/sixty-numbers-tan.md b/.changeset/sixty-numbers-tan.md new file mode 100644 index 0000000000000..d0c6d9367cf85 --- /dev/null +++ b/.changeset/sixty-numbers-tan.md @@ -0,0 +1,6 @@ +--- +"@rocket.chat/i18n": minor +"@rocket.chat/ui-voip": minor +--- + +Introduces native desktop notifications for voice calls when using the desktop app. diff --git a/packages/i18n/src/locales/en.i18n.json b/packages/i18n/src/locales/en.i18n.json index 640ef7f49df6f..0984f1b3c9bd2 100644 --- a/packages/i18n/src/locales/en.i18n.json +++ b/packages/i18n/src/locales/en.i18n.json @@ -2560,6 +2560,7 @@ "Incoming_Livechats": "Queued chats", "Incoming_WebHook": "Incoming WebHook", "Incoming_call": "Incoming call", + "Incoming_call_ellipsis": "Incoming call...", "Incoming_call_from": "Incoming call from", "Incoming_call_from__roomName__": "Incoming call from {{roomName}}", "Incoming_call_transfer": "Incoming call transfer", diff --git a/packages/ui-voip/package.json b/packages/ui-voip/package.json index df20769403453..7fdb5ec514721 100644 --- a/packages/ui-voip/package.json +++ b/packages/ui-voip/package.json @@ -19,6 +19,7 @@ "typecheck": "tsc --noEmit --skipLibCheck -p tsconfig.json" }, "dependencies": { + "@rocket.chat/desktop-api": "workspace:^", "@rocket.chat/emitter": "~0.31.25", "@rocket.chat/media-signaling": "workspace:~", "@tanstack/react-query": "~5.65.1", diff --git a/packages/ui-voip/src/global.d.ts b/packages/ui-voip/src/global.d.ts new file mode 100644 index 0000000000000..51c30e263ccad --- /dev/null +++ b/packages/ui-voip/src/global.d.ts @@ -0,0 +1,7 @@ +import type { IRocketChatDesktop } from '@rocket.chat/desktop-api'; + +declare global { + interface Window { + RocketChatDesktop?: IRocketChatDesktop; + } +} diff --git a/packages/ui-voip/src/v2/MediaCallProvider.tsx b/packages/ui-voip/src/v2/MediaCallProvider.tsx index 069c13b307d02..e1a80109e2daa 100644 --- a/packages/ui-voip/src/v2/MediaCallProvider.tsx +++ b/packages/ui-voip/src/v2/MediaCallProvider.tsx @@ -11,7 +11,7 @@ import { useToastMessageDispatch, useSetting, } from '@rocket.chat/ui-contexts'; -import { useCallback, useEffect } from 'react'; +import { ReactNode, useCallback, useEffect } from 'react'; import { createPortal } from 'react-dom'; import { useTranslation } from 'react-i18next'; @@ -19,13 +19,18 @@ import MediaCallContext, { PeerInfo } from './MediaCallContext'; import MediaCallWidget from './MediaCallWidget'; import TransferModal from './TransferModal'; import { useCallSounds } from './useCallSounds'; +import { useDesktopNotifications } from './useDesktopNotifications'; import { getExtensionFromPeerInfo, useMediaSession } from './useMediaSession'; import { useMediaSessionInstance } from './useMediaSessionInstance'; import useMediaStream from './useMediaStream'; import { isValidTone, useTonePlayer } from './useTonePlayer'; import { stopTracks, useDevicePermissionPrompt2, PermissionRequestCancelledCallRejectedError } from '../hooks/useDevicePermissionPrompt'; -const MediaCallProvider = ({ children }: { children: React.ReactNode }) => { +type MediaCallProviderProps = { + children: ReactNode; +}; + +const MediaCallProvider = ({ children }: MediaCallProviderProps) => { const user = useUser(); const { t } = useTranslation(); const dispatchToastMessage = useToastMessageDispatch(); @@ -37,6 +42,8 @@ const MediaCallProvider = ({ children }: { children: React.ReactNode }) => { const instance = useMediaSessionInstance(userId ?? undefined); const session = useMediaSession(instance); + useDesktopNotifications(session); + const [remoteStreamRefCallback, audioElement] = useMediaStream(instance); const setOutputMediaDevice = useSetOutputMediaDevice(); diff --git a/packages/ui-voip/src/v2/MediaCallWidget.stories.tsx b/packages/ui-voip/src/v2/MediaCallWidget.stories.tsx index 84c6d34bc46d1..4fe83d06c602e 100644 --- a/packages/ui-voip/src/v2/MediaCallWidget.stories.tsx +++ b/packages/ui-voip/src/v2/MediaCallWidget.stories.tsx @@ -4,7 +4,7 @@ import type { Meta, StoryFn, StoryObj } from '@storybook/react'; import { useMediaCallContext } from './MediaCallContext'; import MediaCallWidget from './MediaCallWidget'; -import MediaCallProviderMock from './MockedMediaCallProvider'; +import MockedMediaCallProvider from './MockedMediaCallProvider'; const mockedContexts = mockAppRoot() .withTranslations('en', 'core', { @@ -26,9 +26,9 @@ const meta = { decorators: [ mockedContexts, (Story, options) => ( - + - + ), ], } satisfies Meta; diff --git a/packages/ui-voip/src/v2/MockedMediaCallProvider.tsx b/packages/ui-voip/src/v2/MockedMediaCallProvider.tsx index 1da93f4a3206e..b49dc37bf74d1 100644 --- a/packages/ui-voip/src/v2/MockedMediaCallProvider.tsx +++ b/packages/ui-voip/src/v2/MockedMediaCallProvider.tsx @@ -1,5 +1,5 @@ import { UserStatus } from '@rocket.chat/core-typings'; -import { useState } from 'react'; +import { ReactNode, useState } from 'react'; import MediaCallContext from './MediaCallContext'; import type { State, PeerInfo } from './MediaCallContext'; @@ -7,7 +7,17 @@ import type { State, PeerInfo } from './MediaCallContext'; const avatarUrl = ``; const myData: any[] = Array.from({ length: 100 }, (_, i) => ({ value: `user-${i}`, label: `User ${i}`, identifier: `000${i}`, avatarUrl })); -const MediaCallProviderMock = ({ +type MockedMediaCallProviderProps = { + children: ReactNode; + state?: State; + transferredBy?: string; + remoteMuted?: boolean; + remoteHeld?: boolean; + muted?: boolean; + held?: boolean; +}; + +const MockedMediaCallProvider = ({ children, state = 'closed', transferredBy = undefined, @@ -15,15 +25,7 @@ const MediaCallProviderMock = ({ remoteHeld = false, muted = false, held = false, -}: { - children: React.ReactNode; - state?: State; - transferredBy?: string; - remoteMuted?: boolean; - remoteHeld?: boolean; - muted?: boolean; - held?: boolean; -}) => { +}: MockedMediaCallProviderProps) => { const [peerInfo, setPeerInfo] = useState({ displayName: 'John Doe', userId: '1234567890', @@ -147,4 +149,4 @@ const MediaCallProviderMock = ({ return {children}; }; -export default MediaCallProviderMock; +export default MockedMediaCallProvider; diff --git a/packages/ui-voip/src/v2/components/Widget/Widget.tsx b/packages/ui-voip/src/v2/components/Widget/Widget.tsx index 3b87d06a04026..5ba332a4be626 100644 --- a/packages/ui-voip/src/v2/components/Widget/Widget.tsx +++ b/packages/ui-voip/src/v2/components/Widget/Widget.tsx @@ -1,6 +1,6 @@ import { Palette } from '@rocket.chat/fuselage'; import styled from '@rocket.chat/styled'; -import { ComponentProps, useLayoutEffect } from 'react'; +import { ComponentProps, ReactNode, useLayoutEffect } from 'react'; import { FocusScope } from 'react-aria'; import { DragContext } from './WidgetDraggableContext'; @@ -28,7 +28,7 @@ const WidgetBase = styled('article')` `; type WidgetProps = { - children: React.ReactNode; + children: ReactNode; } & ComponentProps; const Widget = ({ children, ...props }: WidgetProps) => { diff --git a/packages/ui-voip/src/v2/useDesktopNotifications.ts b/packages/ui-voip/src/v2/useDesktopNotifications.ts new file mode 100644 index 0000000000000..2038c54f5d715 --- /dev/null +++ b/packages/ui-voip/src/v2/useDesktopNotifications.ts @@ -0,0 +1,75 @@ +import { useEffect, useRef } from 'react'; +import { useTranslation } from 'react-i18next'; + +import { PeerInfo } from './MediaCallContext'; +import { SessionInfo } from './useMediaSessionInstance'; +import { convertAvatarUrlToPng } from './utils/convertAvatarUrlToPng'; + +const getDisplayInfo = (peerInfo?: PeerInfo) => { + if (!peerInfo) { + return undefined; + } + + if ('number' in peerInfo) { + return { title: peerInfo.number }; + } + if ('displayName' in peerInfo) { + return { title: peerInfo.displayName, avatar: peerInfo.avatarUrl }; + } + return undefined; +}; + +export const useDesktopNotifications = (sessionInfo: SessionInfo) => { + const previousCallId = useRef(undefined); + const { t } = useTranslation(); + + const displayInfo = getDisplayInfo(sessionInfo.peerInfo); + useEffect(() => { + if ( + typeof window.RocketChatDesktop?.dispatchCustomNotification !== 'function' || + typeof window.RocketChatDesktop?.closeCustomNotification !== 'function' + ) { + return; + } + + let isMounted = true; + + if (sessionInfo.state !== 'ringing') { + if (previousCallId.current) { + window.RocketChatDesktop.closeCustomNotification(previousCallId.current); + previousCallId.current = undefined; + } + return; + } + + if (!displayInfo?.title) { + return; + } + + const notifyDesktop = async () => { + const avatarAsPng = await convertAvatarUrlToPng(displayInfo.avatar); + + if (!isMounted) { + return; + } + + window.RocketChatDesktop?.dispatchCustomNotification({ + type: 'voice', + id: sessionInfo.callId, + payload: { + title: displayInfo.title, + body: t('Incoming_call_ellipsis'), + avatar: avatarAsPng || undefined, + requireInteraction: true, + }, + }); + }; + + notifyDesktop(); + previousCallId.current = sessionInfo.callId; + + return () => { + isMounted = false; + }; + }, [displayInfo?.avatar, displayInfo?.title, sessionInfo.callId, sessionInfo.state, t]); +}; diff --git a/packages/ui-voip/src/v2/useKeypad.tsx b/packages/ui-voip/src/v2/useKeypad.tsx index 159cd6726496a..230776edacaaf 100644 --- a/packages/ui-voip/src/v2/useKeypad.tsx +++ b/packages/ui-voip/src/v2/useKeypad.tsx @@ -1,11 +1,11 @@ import { Divider, Box, TextInput, Field, FieldRow } from '@rocket.chat/fuselage'; -import { useState } from 'react'; +import { ReactNode, useState } from 'react'; import { useTranslation } from 'react-i18next'; import { Keypad } from './components'; type UseKeypad = { - element: React.ReactNode; + element: ReactNode; buttonProps: { title: string; onClick: () => void; diff --git a/packages/ui-voip/src/v2/useMediaSession.ts b/packages/ui-voip/src/v2/useMediaSession.ts index 062f30e07fec0..ab2bfe206b7ce 100644 --- a/packages/ui-voip/src/v2/useMediaSession.ts +++ b/packages/ui-voip/src/v2/useMediaSession.ts @@ -8,6 +8,7 @@ import type { SessionInfo } from './useMediaSessionInstance'; const defaultSessionInfo: SessionInfo = { state: 'closed' as const, + callId: undefined, connectionState: 'CONNECTING' as const, peerInfo: undefined, transferredBy: undefined, @@ -48,7 +49,10 @@ export const getExtensionFromPeerInfo = (peerInfo: PeerInfo): string | undefined return undefined; }; -const deriveWidgetStateFromCallState = (callState: CallState, callRole: CallRole): State | undefined => { +const deriveWidgetStateFromCallState = ( + callState: CallState, + callRole: CallRole, +): Extract | undefined => { switch (callState) { case 'active': case 'accepted': @@ -143,7 +147,18 @@ export const useMediaSession = (instance?: MediaSignalingSession): MediaSession return; } - const { contact, transferredBy: callTransferredBy, state: callState, role, muted, held, hidden, remoteHeld, remoteMute } = mainCall; + const { + contact, + transferredBy: callTransferredBy, + state: callState, + role, + muted, + held, + hidden, + remoteHeld, + remoteMute, + callId, + } = mainCall; const state = deriveWidgetStateFromCallState(callState, role); const connectionState = deriveConnectionStateFromCallState(callState); @@ -162,6 +177,7 @@ export const useMediaSession = (instance?: MediaSignalingSession): MediaSession hidden, remoteHeld, remoteMuted: remoteMute, + callId, }, }); return; @@ -189,7 +205,7 @@ export const useMediaSession = (instance?: MediaSignalingSession): MediaSession dispatch({ type: 'instance_updated', - payload: { state, peerInfo, transferredBy, muted, held, connectionState, hidden, remoteHeld, remoteMuted: remoteMute }, + payload: { state, peerInfo, transferredBy, muted, held, connectionState, hidden, remoteHeld, remoteMuted: remoteMute, callId }, }); }; diff --git a/packages/ui-voip/src/v2/useMediaSessionInstance.ts b/packages/ui-voip/src/v2/useMediaSessionInstance.ts index d726191fa7cc4..c9c0da0db687f 100644 --- a/packages/ui-voip/src/v2/useMediaSessionInstance.ts +++ b/packages/ui-voip/src/v2/useMediaSessionInstance.ts @@ -23,10 +23,12 @@ interface BaseSession { interface EmptySession extends BaseSession { state: Extract; + callId: undefined; } interface CallSession extends BaseSession { state: Extract; + callId: string; peerInfo: PeerInfo; } diff --git a/packages/ui-voip/src/v2/utils/convertAvatarUrlToPng.ts b/packages/ui-voip/src/v2/utils/convertAvatarUrlToPng.ts new file mode 100644 index 0000000000000..45da47cdcd55a --- /dev/null +++ b/packages/ui-voip/src/v2/utils/convertAvatarUrlToPng.ts @@ -0,0 +1,36 @@ +export const convertAvatarUrlToPng = (avatarUrl: string | undefined): Promise => { + return new Promise((resolve) => { + if (!avatarUrl) { + resolve(''); + return; + } + + const image = new Image(); + + const onLoad = (): void => { + const canvas = document.createElement('canvas'); + canvas.width = image.width; + canvas.height = image.height; + const context = canvas.getContext('2d'); + + if (!context) { + resolve(''); + return; + } + + context.drawImage(image, 0, 0); + try { + resolve(canvas.toDataURL('image/png')); + } catch (e) { + resolve(''); + } + }; + + const onError = (): void => resolve(''); + + image.onload = onLoad; + image.onerror = onError; + image.crossOrigin = 'anonymous'; + image.src = avatarUrl; + }); +}; diff --git a/packages/ui-voip/src/v2/views/IncomingCall.stories.tsx b/packages/ui-voip/src/v2/views/IncomingCall.stories.tsx index 698e5936334dc..ffc34e625c3c6 100644 --- a/packages/ui-voip/src/v2/views/IncomingCall.stories.tsx +++ b/packages/ui-voip/src/v2/views/IncomingCall.stories.tsx @@ -2,7 +2,7 @@ import { mockAppRoot } from '@rocket.chat/mock-providers'; import type { Meta, StoryFn } from '@storybook/react'; import IncomingCall from './IncomingCall'; -import MediaCallProviderMock from '../MockedMediaCallProvider'; +import MockedMediaCallProvider from '../MockedMediaCallProvider'; const mockedContexts = mockAppRoot() .withTranslations('en', 'core', { @@ -18,9 +18,9 @@ export default { decorators: [ mockedContexts, (Story) => ( - + - + ), ], } satisfies Meta; diff --git a/packages/ui-voip/src/v2/views/IncomingCallTransfer.stories.tsx b/packages/ui-voip/src/v2/views/IncomingCallTransfer.stories.tsx index b639b1e708b40..a0a48f3de288c 100644 --- a/packages/ui-voip/src/v2/views/IncomingCallTransfer.stories.tsx +++ b/packages/ui-voip/src/v2/views/IncomingCallTransfer.stories.tsx @@ -2,7 +2,7 @@ import { mockAppRoot } from '@rocket.chat/mock-providers'; import type { Meta, StoryFn } from '@storybook/react'; import IncomingCallTransfer from './IncomingCallTransfer'; -import MediaCallProviderMock from '../MockedMediaCallProvider'; +import MockedMediaCallProvider from '../MockedMediaCallProvider'; const mockedContexts = mockAppRoot() .withTranslations('en', 'core', { @@ -19,9 +19,9 @@ export default { decorators: [ mockedContexts, (Story) => ( - + - + ), ], } satisfies Meta; diff --git a/packages/ui-voip/src/v2/views/NewCall.stories.tsx b/packages/ui-voip/src/v2/views/NewCall.stories.tsx index 736e729780a6f..953485f7c4921 100644 --- a/packages/ui-voip/src/v2/views/NewCall.stories.tsx +++ b/packages/ui-voip/src/v2/views/NewCall.stories.tsx @@ -2,7 +2,7 @@ import { mockAppRoot } from '@rocket.chat/mock-providers'; import type { Meta, StoryFn } from '@storybook/react'; import NewCall from './NewCall'; -import MediaCallProviderMock from '../MockedMediaCallProvider'; +import MockedMediaCallProvider from '../MockedMediaCallProvider'; const mockedContexts = mockAppRoot() .withTranslations('en', 'core', { @@ -18,9 +18,9 @@ export default { decorators: [ mockedContexts, (Story) => ( - + - + ), ], } satisfies Meta; diff --git a/packages/ui-voip/src/v2/views/OngoingCall.stories.tsx b/packages/ui-voip/src/v2/views/OngoingCall.stories.tsx index 68944f2128420..a8158cfb388a4 100644 --- a/packages/ui-voip/src/v2/views/OngoingCall.stories.tsx +++ b/packages/ui-voip/src/v2/views/OngoingCall.stories.tsx @@ -2,7 +2,7 @@ import { mockAppRoot } from '@rocket.chat/mock-providers'; import type { Meta, StoryFn } from '@storybook/react'; import OngoingCall from './OngoingCall'; -import MediaCallProviderMock from '../MockedMediaCallProvider'; +import MockedMediaCallProvider from '../MockedMediaCallProvider'; const mockedContexts = mockAppRoot().buildStoryDecorator(); @@ -12,9 +12,9 @@ export default { decorators: [ mockedContexts, (Story) => ( - + - + ), ], } satisfies Meta; @@ -25,40 +25,40 @@ export const OngoingCallStory: StoryFn = () => { export const OngoingCallWithSlots: StoryFn = () => { return ( - + - + ); }; export const OngoingCallWithRemoteStatus: StoryFn = () => { return ( - + - + ); }; export const OngoingCallWithRemoteStatusMuted: StoryFn = () => { return ( - + - + ); }; export const OngoingCallWithRemoteStatusHeld: StoryFn = () => { return ( - + - + ); }; export const OngoingCallWithSlotsAndRemoteStatus: StoryFn = () => { return ( - + - + ); }; diff --git a/packages/ui-voip/src/v2/views/OutgoingCall.stories.tsx b/packages/ui-voip/src/v2/views/OutgoingCall.stories.tsx index 74e7f0ecfaf51..fb02940f146c0 100644 --- a/packages/ui-voip/src/v2/views/OutgoingCall.stories.tsx +++ b/packages/ui-voip/src/v2/views/OutgoingCall.stories.tsx @@ -2,7 +2,7 @@ import { mockAppRoot } from '@rocket.chat/mock-providers'; import type { Meta, StoryFn } from '@storybook/react'; import OutgoingCall from './OutgoingCall'; -import MediaCallProviderMock from '../MockedMediaCallProvider'; +import MockedMediaCallProvider from '../MockedMediaCallProvider'; const mockedContexts = mockAppRoot() .withTranslations('en', 'core', { @@ -17,9 +17,9 @@ export default { decorators: [ mockedContexts, (Story) => ( - + - + ), ], } satisfies Meta; diff --git a/packages/ui-voip/src/v2/views/OutgoingCallTransfer.stories.tsx b/packages/ui-voip/src/v2/views/OutgoingCallTransfer.stories.tsx index c2a01b54cfd2e..f64fe038b7305 100644 --- a/packages/ui-voip/src/v2/views/OutgoingCallTransfer.stories.tsx +++ b/packages/ui-voip/src/v2/views/OutgoingCallTransfer.stories.tsx @@ -2,7 +2,7 @@ import { mockAppRoot } from '@rocket.chat/mock-providers'; import type { Meta, StoryFn } from '@storybook/react'; import OutgoingCallTransfer from './OutgoingCallTransfer'; -import MediaCallProviderMock from '../MockedMediaCallProvider'; +import MockedMediaCallProvider from '../MockedMediaCallProvider'; const mockedContexts = mockAppRoot() .withTranslations('en', 'core', { @@ -18,9 +18,9 @@ export default { decorators: [ mockedContexts, (Story) => ( - + - + ), ], } satisfies Meta; diff --git a/yarn.lock b/yarn.lock index a5a6d0b6dc198..ef60b63ad81ee 100644 --- a/yarn.lock +++ b/yarn.lock @@ -8474,7 +8474,7 @@ __metadata: languageName: unknown linkType: soft -"@rocket.chat/desktop-api@workspace:packages/desktop-api, @rocket.chat/desktop-api@workspace:~": +"@rocket.chat/desktop-api@workspace:^, @rocket.chat/desktop-api@workspace:packages/desktop-api, @rocket.chat/desktop-api@workspace:~": version: 0.0.0-use.local resolution: "@rocket.chat/desktop-api@workspace:packages/desktop-api" dependencies: @@ -10601,6 +10601,7 @@ __metadata: "@playwright/test": "npm:^1.52.0" "@react-spectrum/test-utils": "npm:~1.0.0-alpha.8" "@rocket.chat/css-in-js": "npm:~0.31.25" + "@rocket.chat/desktop-api": "workspace:^" "@rocket.chat/emitter": "npm:~0.31.25" "@rocket.chat/eslint-config": "workspace:^" "@rocket.chat/fuselage": "npm:^0.68.1"