diff --git a/res/css/_components.scss b/res/css/_components.scss index d8bc238db5..9041eef13f 100644 --- a/res/css/_components.scss +++ b/res/css/_components.scss @@ -236,4 +236,6 @@ @import "./views/verification/_VerificationShowSas.scss"; @import "./views/voip/_CallContainer.scss"; @import "./views/voip/_CallView.scss"; +@import "./views/voip/_DialPad.scss"; +@import "./views/voip/_DialPadModal.scss"; @import "./views/voip/_VideoFeed.scss"; diff --git a/res/css/views/rooms/_RoomList.scss b/res/css/views/rooms/_RoomList.scss index b7759d265f..66e1b827d0 100644 --- a/res/css/views/rooms/_RoomList.scss +++ b/res/css/views/rooms/_RoomList.scss @@ -24,6 +24,9 @@ limitations under the License. .mx_RoomList_iconExplore::before { mask-image: url('$(res)/img/element-icons/roomlist/explore.svg'); } +.mx_RoomList_iconDialpad::before { + mask-image: url('$(res)/img/element-icons/roomlist/dialpad.svg'); +} .mx_RoomList_explorePrompt { margin: 4px 12px 4px; diff --git a/res/css/views/voip/_DialPad.scss b/res/css/views/voip/_DialPad.scss new file mode 100644 index 0000000000..0c7bff0ce8 --- /dev/null +++ b/res/css/views/voip/_DialPad.scss @@ -0,0 +1,62 @@ +/* +Copyright 2020 The Matrix.org Foundation C.I.C. + +Licensed under the Apache License, Version 2.0 (the "License"); +you may not use this file except in compliance with the License. +You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +*/ + +.mx_DialPad { + display: grid; + grid-template-columns: repeat(3, 1fr); + gap: 16px; +} + +.mx_DialPad_button { + width: 40px; + height: 40px; + background-color: $theme-button-bg-color; + border-radius: 40px; + font-size: 18px; + font-weight: 600; + text-align: center; + vertical-align: middle; + line-height: 40px; +} + +.mx_DialPad_deleteButton, .mx_DialPad_dialButton { + &::before { + content: ''; + display: inline-block; + height: 40px; + width: 40px; + vertical-align: middle; + mask-repeat: no-repeat; + mask-size: 20px; + mask-position: center; + background-color: $primary-bg-color; + } +} + +.mx_DialPad_deleteButton { + background-color: $notice-primary-color; + &::before { + mask-image: url('$(res)/img/element-icons/call/delete.svg'); + mask-position: 9px; // delete icon is right-heavy so have to be slightly to the left to look centered + } +} + +.mx_DialPad_dialButton { + background-color: $accent-color; + &::before { + mask-image: url('$(res)/img/element-icons/call/voice-call.svg'); + } +} diff --git a/res/css/views/voip/_DialPadModal.scss b/res/css/views/voip/_DialPadModal.scss new file mode 100644 index 0000000000..f9d7673a38 --- /dev/null +++ b/res/css/views/voip/_DialPadModal.scss @@ -0,0 +1,74 @@ +/* +Copyright 2020 The Matrix.org Foundation C.I.C. + +Licensed under the Apache License, Version 2.0 (the "License"); +you may not use this file except in compliance with the License. +You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +*/ + +.mx_Dialog_dialPadWrapper .mx_Dialog { + padding: 0px; +} + +.mx_DialPadModal { + width: 192px; + height: 368px; +} + +.mx_DialPadModal_header { + margin-top: 12px; + margin-left: 12px; + margin-right: 12px; +} + +.mx_DialPadModal_title { + color: $muted-fg-color; + font-size: 12px; + font-weight: 600; +} + +.mx_DialPadModal_cancel { + float: right; + mask: url('$(res)/img/feather-customised/cancel.svg'); + mask-repeat: no-repeat; + mask-position: center; + mask-size: cover; + width: 14px; + height: 14px; + background-color: $dialog-close-fg-color; + cursor: pointer; +} + +.mx_DialPadModal_field { + border: none; + margin: 0px; +} + +.mx_DialPadModal_field input { + font-size: 18px; + font-weight: 600; +} + +.mx_DialPadModal_dialPad { + margin-left: 16px; + margin-right: 16px; + margin-top: 16px; +} + +.mx_DialPadModal_horizSep { + position: relative; + &::before { + content: ''; + position: absolute; + width: 100%; + border-bottom: 1px solid $input-darker-bg-color; + } +} diff --git a/res/img/element-icons/call/delete.svg b/res/img/element-icons/call/delete.svg new file mode 100644 index 0000000000..133bdad4ca --- /dev/null +++ b/res/img/element-icons/call/delete.svg @@ -0,0 +1,10 @@ + + + + + + + + + + diff --git a/res/img/element-icons/roomlist/dialpad.svg b/res/img/element-icons/roomlist/dialpad.svg new file mode 100644 index 0000000000..b51d4a4dc9 --- /dev/null +++ b/res/img/element-icons/roomlist/dialpad.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/CallHandler.tsx b/src/CallHandler.tsx index 504dae5c84..bcb2042f84 100644 --- a/src/CallHandler.tsx +++ b/src/CallHandler.tsx @@ -82,6 +82,9 @@ import CountlyAnalytics from "./CountlyAnalytics"; import {UIFeature} from "./settings/UIFeature"; import { CallError } from "matrix-js-sdk/src/webrtc/call"; import { logger } from 'matrix-js-sdk/src/logger'; +import { Action } from './dispatcher/actions'; + +const CHECK_PSTN_SUPPORT_ATTEMPTS = 3; enum AudioID { Ring = 'ringAudio', @@ -119,6 +122,8 @@ export default class CallHandler { private calls = new Map(); // roomId -> call private audioPromises = new Map>(); private dispatcherRef: string = null; + private supportsPstnProtocol = null; + private pstnSupportCheckTimer: NodeJS.Timeout; // number actually because we're in the browser static sharedInstance() { if (!window.mxCallHandler) { @@ -145,6 +150,8 @@ export default class CallHandler { if (SettingsStore.getValue(UIFeature.Voip)) { MatrixClientPeg.get().on('Call.incoming', this.onCallIncoming); } + + this.checkForPstnSupport(CHECK_PSTN_SUPPORT_ATTEMPTS); } stop() { @@ -158,6 +165,33 @@ export default class CallHandler { } } + private async checkForPstnSupport(maxTries) { + try { + const protocols = await MatrixClientPeg.get().getThirdpartyProtocols(); + if (protocols['im.vector.protocol.pstn'] !== undefined) { + this.supportsPstnProtocol = protocols['im.vector.protocol.pstn']; + } else if (protocols['m.protocol.pstn'] !== undefined) { + this.supportsPstnProtocol = protocols['m.protocol.pstn']; + } else { + this.supportsPstnProtocol = null; + } + dis.dispatch({action: Action.PstnSupportUpdated}); + } catch (e) { + if (maxTries === 1) { + console.log("Failed to check for pstn protocol support and no retries remain: assuming no support", e); + } else { + console.log("Failed to check for pstn protocol support: will retry", e); + this.pstnSupportCheckTimer = setTimeout(() => { + this.checkForPstnSupport(maxTries - 1); + }, 10000); + } + } + } + + getSupportsPstnProtocol() { + return this.supportsPstnProtocol; + } + private onCallIncoming = (call) => { // we dispatch this synchronously to make sure that the event // handlers on the call are set up immediately (so that if diff --git a/src/components/structures/MatrixChat.tsx b/src/components/structures/MatrixChat.tsx index 4a8d3cc718..62c729c422 100644 --- a/src/components/structures/MatrixChat.tsx +++ b/src/components/structures/MatrixChat.tsx @@ -80,6 +80,7 @@ import CreateCommunityPrototypeDialog from "../views/dialogs/CreateCommunityProt import ThreepidInviteStore, { IThreepidInvite, IThreepidInviteWireFormat } from "../../stores/ThreepidInviteStore"; import {UIFeature} from "../../settings/UIFeature"; import { CommunityPrototypeStore } from "../../stores/CommunityPrototypeStore"; +import DialPadModal from "../views/voip/DialPadModal"; /** constants for MatrixChat.state.view */ export enum Views { @@ -703,6 +704,9 @@ export default class MatrixChat extends React.PureComponent { this.state.resizeNotifier.notifyLeftHandleResized(); }); break; + case Action.OpenDialPad: + Modal.createTrackedDialog('Dial pad', '', DialPadModal, {}, "mx_Dialog_dialPadWrapper"); + break; case 'on_logged_in': if ( !Lifecycle.isSoftLogout() && diff --git a/src/components/views/rooms/RoomList.tsx b/src/components/views/rooms/RoomList.tsx index 6e677f2b01..4a4afbc5ac 100644 --- a/src/components/views/rooms/RoomList.tsx +++ b/src/components/views/rooms/RoomList.tsx @@ -46,6 +46,7 @@ import { objectShallowClone, objectWithOnly } from "../../../utils/objects"; import { IconizedContextMenuOption, IconizedContextMenuOptionList } from "../context_menus/IconizedContextMenu"; import AccessibleButton from "../elements/AccessibleButton"; import { CommunityPrototypeStore } from "../../../stores/CommunityPrototypeStore"; +import CallHandler from "../../../CallHandler"; interface IProps { onKeyDown: (ev: React.KeyboardEvent) => void; @@ -89,10 +90,44 @@ interface ITagAesthetics { defaultHidden: boolean; } -const TAG_AESTHETICS: { +interface ITagAestheticsMap { // @ts-ignore - TS wants this to be a string but we know better [tagId: TagID]: ITagAesthetics; -} = { +} + +// If we have no dialer support, we just show the create chat dialog +const dmOnAddRoom = (dispatcher?: Dispatcher) => { + (dispatcher || defaultDispatcher).dispatch({action: 'view_create_chat'}); +}; + +// If we have dialer support, show a context menu so the user can pick between +// the dialer and the create chat dialog +const dmAddRoomContextMenu = (onFinished: () => void) => { + return + { + e.preventDefault(); + e.stopPropagation(); + onFinished(); + defaultDispatcher.dispatch({action: "view_create_chat"}); + }} + /> + { + e.preventDefault(); + e.stopPropagation(); + onFinished(); + defaultDispatcher.fire(Action.OpenDialPad); + }} + /> + ; +}; + +const TAG_AESTHETICS: ITagAestheticsMap = { [DefaultTagID.Invite]: { sectionLabel: _td("Invites"), isInvite: true, @@ -108,9 +143,8 @@ const TAG_AESTHETICS: { isInvite: false, defaultHidden: false, addRoomLabel: _td("Start chat"), - onAddRoom: (dispatcher?: Dispatcher) => { - (dispatcher || defaultDispatcher).dispatch({action: 'view_create_chat'}); - }, + // Either onAddRoom or addRoomContextMenu are set depending on whether we + // have dialer support. }, [DefaultTagID.Untagged]: { sectionLabel: _td("Rooms"), @@ -178,6 +212,7 @@ function customTagAesthetics(tagId: TagID): ITagAesthetics { export default class RoomList extends React.PureComponent { private dispatcherRef; private customTagStoreRef; + private tagAesthetics: ITagAestheticsMap; constructor(props: IProps) { super(props); @@ -187,6 +222,10 @@ export default class RoomList extends React.PureComponent { isNameFiltering: !!RoomListStore.instance.getFirstNameFilterCondition(), }; + // shallow-copy from the template as we need to make modifications to it + this.tagAesthetics = Object.assign({}, TAG_AESTHETICS); + this.updateDmAddRoomAction(); + this.dispatcherRef = defaultDispatcher.register(this.onAction); } @@ -202,6 +241,17 @@ export default class RoomList extends React.PureComponent { if (this.customTagStoreRef) this.customTagStoreRef.remove(); } + private updateDmAddRoomAction() { + const dmTagAesthetics = Object.assign({}, TAG_AESTHETICS[DefaultTagID.DM]); + if (CallHandler.sharedInstance().getSupportsPstnProtocol()) { + dmTagAesthetics.addRoomContextMenu = dmAddRoomContextMenu; + } else { + dmTagAesthetics.onAddRoom = dmOnAddRoom; + } + + this.tagAesthetics[DefaultTagID.DM] = dmTagAesthetics; + } + private onAction = (payload: ActionPayload) => { if (payload.action === Action.ViewRoomDelta) { const viewRoomDeltaPayload = payload as ViewRoomDeltaPayload; @@ -214,6 +264,9 @@ export default class RoomList extends React.PureComponent { show_room_tile: true, // to make sure the room gets scrolled into view }); } + } else if (payload.action === Action.PstnSupportUpdated) { + this.updateDmAddRoomAction(); + this.updateLists(); } }; @@ -355,7 +408,7 @@ export default class RoomList extends React.PureComponent { const aesthetics: ITagAesthetics = isCustomTag(orderedTagId) ? customTagAesthetics(orderedTagId) - : TAG_AESTHETICS[orderedTagId]; + : this.tagAesthetics[orderedTagId]; if (!aesthetics) throw new Error(`Tag ${orderedTagId} does not have aesthetics`); components.push( void; +} + +class DialPadButton extends React.PureComponent { + onClick = () => { + this.props.onButtonPress(this.props.digit); + } + + render() { + switch (this.props.kind) { + case DialPadButtonKind.Digit: + return + {this.props.digit} + ; + case DialPadButtonKind.Delete: + return ; + case DialPadButtonKind.Dial: + return ; + } + } +} + +interface IProps { + onDigitPress: (string) => void; + onDeletePress: (string) => void; + onDialPress: (string) => void; +} + +export default class Dialpad extends React.PureComponent { + render() { + const buttonNodes = []; + + for (const button of BUTTONS) { + buttonNodes.push(); + } + + buttonNodes.push(); + buttonNodes.push(); + + return
+ {buttonNodes} +
; + } +} diff --git a/src/components/views/voip/DialPadModal.tsx b/src/components/views/voip/DialPadModal.tsx new file mode 100644 index 0000000000..9f7e4140c9 --- /dev/null +++ b/src/components/views/voip/DialPadModal.tsx @@ -0,0 +1,111 @@ +/* +Copyright 2020 The Matrix.org Foundation C.I.C. + +Licensed under the Apache License, Version 2.0 (the "License"); +you may not use this file except in compliance with the License. +You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. +*/ + +import * as React from "react"; +import { ensureDMExists } from "../../../createRoom"; +import { _t } from "../../../languageHandler"; +import { MatrixClientPeg } from "../../../MatrixClientPeg"; +import AccessibleButton from "../elements/AccessibleButton"; +import Field from "../elements/Field"; +import DialPad from './DialPad'; +import dis from '../../../dispatcher/dispatcher'; +import Modal from "../../../Modal"; +import ErrorDialog from "../../views/dialogs/ErrorDialog"; + +interface IProps { + onFinished: (boolean) => void; +} + +interface IState { + value: string; +} + +export default class DialpadModal extends React.PureComponent { + constructor(props) { + super(props); + this.state = { + value: '', + } + } + + onCancelClick = () => { + this.props.onFinished(false); + } + + onChange = (ev) => { + this.setState({value: ev.target.value}); + } + + onFormSubmit = (ev) => { + ev.preventDefault(); + this.onDialPress(); + } + + onDigitPress = (digit) => { + this.setState({value: this.state.value + digit}); + } + + onDeletePress = () => { + if (this.state.value.length === 0) return; + this.setState({value: this.state.value.slice(0, -1)}); + } + + onDialPress = async () => { + const results = await MatrixClientPeg.get().getThirdpartyUser('im.vector.protocol.pstn', { + 'm.id.phone': this.state.value, + }); + if (!results || results.length === 0 || !results[0].userid) { + Modal.createTrackedDialog('', '', ErrorDialog, { + title: _t("Unable to look up phone number"), + description: _t("There was an error looking up the phone number"), + }); + } + const userId = results[0].userid; + + const roomId = await ensureDMExists(MatrixClientPeg.get(), userId); + + dis.dispatch({ + action: 'view_room', + room_id: roomId, + }); + + this.props.onFinished(true); + } + + render() { + return
+
+
+ {_t("Dial pad")} + +
+
+ + +
+
+
+ +
+
; + } +} diff --git a/src/dispatcher/actions.ts b/src/dispatcher/actions.ts index 6fb71df30d..be292774de 100644 --- a/src/dispatcher/actions.ts +++ b/src/dispatcher/actions.ts @@ -94,4 +94,15 @@ export enum Action { * Trigged after the phase of the right panel is set. Should be used with AfterRightPanelPhaseChangePayload. */ AfterRightPanelPhaseChange = "after_right_panel_phase_change", + + /** + * Opens the modal dial pad + */ + OpenDialPad = "open_dial_pad", + + /** + * Fired when CallHandler has checked for PSTN protocol support + * XXX: Is an action the right thing for this? + */ + PstnSupportUpdated = "pstn_support_updated", } diff --git a/src/i18n/strings/en_EN.json b/src/i18n/strings/en_EN.json index 507f3e071f..be2a7b3dbe 100644 --- a/src/i18n/strings/en_EN.json +++ b/src/i18n/strings/en_EN.json @@ -861,6 +861,9 @@ "Fill Screen": "Fill Screen", "Return to call": "Return to call", "%(name)s on hold": "%(name)s on hold", + "Unable to look up phone number": "Unable to look up phone number", + "There was an error looking up the phone number": "There was an error looking up the phone number", + "Dial pad": "Dial pad", "Unknown caller": "Unknown caller", "Incoming voice call": "Incoming voice call", "Incoming video call": "Incoming video call", @@ -1459,6 +1462,8 @@ "Hide Widgets": "Hide Widgets", "Show Widgets": "Show Widgets", "Search": "Search", + "Start a Conversation": "Start a Conversation", + "Open dial pad": "Open dial pad", "Invites": "Invites", "Favourites": "Favourites", "People": "People",