From ae411b94011ce1a74a771f458c55d8b064d279b1 Mon Sep 17 00:00:00 2001 From: Dariusz Niemczyk Date: Tue, 3 Aug 2021 12:40:37 +0200 Subject: [PATCH 01/24] Refactor Call components into smaller pieces --- src/components/views/voip/CallPreview.tsx | 178 +-------------- src/components/views/voip/CallView.tsx | 118 +--------- .../views/voip/CallView/CallViewHeader.tsx | 135 ++++++++++++ .../views/voip/PictureInPictureDragger.tsx | 208 ++++++++++++++++++ 4 files changed, 363 insertions(+), 276 deletions(-) create mode 100644 src/components/views/voip/CallView/CallViewHeader.tsx create mode 100644 src/components/views/voip/PictureInPictureDragger.tsx diff --git a/src/components/views/voip/CallPreview.tsx b/src/components/views/voip/CallPreview.tsx index 6261b9965f..565e4657b9 100644 --- a/src/components/views/voip/CallPreview.tsx +++ b/src/components/views/voip/CallPreview.tsx @@ -15,7 +15,7 @@ See the License for the specific language governing permissions and limitations under the License. */ -import React, { createRef } from 'react'; +import React from 'react'; import CallView from "./CallView"; import RoomViewStore from '../../../stores/RoomViewStore'; @@ -27,23 +27,8 @@ import SettingsStore from "../../../settings/SettingsStore"; import { CallEvent, CallState, MatrixCall } from 'matrix-js-sdk/src/webrtc/call'; import { MatrixClientPeg } from '../../../MatrixClientPeg'; import { replaceableComponent } from "../../../utils/replaceableComponent"; -import UIStore from '../../../stores/UIStore'; -import { lerp } from '../../../utils/AnimationUtils'; -import { MarkedExecution } from '../../../utils/MarkedExecution'; import { EventSubscription } from 'fbemitter'; - -const PIP_VIEW_WIDTH = 336; -const PIP_VIEW_HEIGHT = 232; - -const MOVING_AMT = 0.2; -const SNAPPING_AMT = 0.05; - -const PADDING = { - top: 58, - bottom: 58, - left: 76, - right: 8, -}; +import { PictureInPictureDragger } from './PictureInPictureDragger'; const SHOW_CALL_IN_STATES = [ CallState.Connected, @@ -66,10 +51,6 @@ interface IState { // Any other call we're displaying: only if the user is on two calls and not viewing either of the rooms // they belong to secondaryCall: MatrixCall; - - // Position of the CallPreview - translationX: number; - translationY: number; } // Splits a list of calls into one 'primary' one and a list @@ -112,16 +93,6 @@ export default class CallPreview extends React.Component { private roomStoreToken: EventSubscription; private dispatcherRef: string; private settingsWatcherRef: string; - private callViewWrapper = createRef(); - private initX = 0; - private initY = 0; - private desiredTranslationX = UIStore.instance.windowWidth - PADDING.right - PIP_VIEW_WIDTH; - private desiredTranslationY = UIStore.instance.windowHeight - PADDING.bottom - PIP_VIEW_WIDTH; - private moving = false; - private scheduledUpdate = new MarkedExecution( - () => this.animationCallback(), - () => requestAnimationFrame(() => this.scheduledUpdate.trigger()), - ); constructor(props: IProps) { super(props); @@ -136,17 +107,12 @@ export default class CallPreview extends React.Component { roomId, primaryCall: primaryCall, secondaryCall: secondaryCalls[0], - translationX: UIStore.instance.windowWidth - PADDING.right - PIP_VIEW_WIDTH, - translationY: UIStore.instance.windowHeight - PADDING.bottom - PIP_VIEW_WIDTH, }; } public componentDidMount() { CallHandler.sharedInstance().addListener(CallHandlerEvent.CallChangeRoom, this.updateCalls); this.roomStoreToken = RoomViewStore.addListener(this.onRoomViewStoreUpdate); - document.addEventListener("mousemove", this.onMoving); - document.addEventListener("mouseup", this.onEndMoving); - window.addEventListener("resize", this.onResize); this.dispatcherRef = dis.register(this.onAction); MatrixClientPeg.get().on(CallEvent.RemoteHoldUnhold, this.onCallRemoteHold); } @@ -154,9 +120,6 @@ export default class CallPreview extends React.Component { public componentWillUnmount() { CallHandler.sharedInstance().removeListener(CallHandlerEvent.CallChangeRoom, this.updateCalls); MatrixClientPeg.get().removeListener(CallEvent.RemoteHoldUnhold, this.onCallRemoteHold); - document.removeEventListener("mousemove", this.onMoving); - document.removeEventListener("mouseup", this.onEndMoving); - window.removeEventListener("resize", this.onResize); if (this.roomStoreToken) { this.roomStoreToken.remove(); } @@ -164,94 +127,6 @@ export default class CallPreview extends React.Component { SettingsStore.unwatchSetting(this.settingsWatcherRef); } - private onResize = (): void => { - this.snap(false); - }; - - private animationCallback = () => { - // If the PiP isn't being dragged and there is only a tiny difference in - // the desiredTranslation and translation, quit the animationCallback - // loop. If that is the case, it means the PiP has snapped into its - // position and there is nothing to do. Not doing this would cause an - // infinite loop - if ( - !this.moving && - Math.abs(this.state.translationX - this.desiredTranslationX) <= 1 && - Math.abs(this.state.translationY - this.desiredTranslationY) <= 1 - ) return; - - const amt = this.moving ? MOVING_AMT : SNAPPING_AMT; - this.setState({ - translationX: lerp(this.state.translationX, this.desiredTranslationX, amt), - translationY: lerp(this.state.translationY, this.desiredTranslationY, amt), - }); - this.scheduledUpdate.mark(); - }; - - private setTranslation(inTranslationX: number, inTranslationY: number) { - const width = this.callViewWrapper.current?.clientWidth || PIP_VIEW_WIDTH; - const height = this.callViewWrapper.current?.clientHeight || PIP_VIEW_HEIGHT; - - // Avoid overflow on the x axis - if (inTranslationX + width >= UIStore.instance.windowWidth) { - this.desiredTranslationX = UIStore.instance.windowWidth - width; - } else if (inTranslationX <= 0) { - this.desiredTranslationX = 0; - } else { - this.desiredTranslationX = inTranslationX; - } - - // Avoid overflow on the y axis - if (inTranslationY + height >= UIStore.instance.windowHeight) { - this.desiredTranslationY = UIStore.instance.windowHeight - height; - } else if (inTranslationY <= 0) { - this.desiredTranslationY = 0; - } else { - this.desiredTranslationY = inTranslationY; - } - } - - private snap(animate?: boolean): void { - const translationX = this.desiredTranslationX; - const translationY = this.desiredTranslationY; - // We subtract the PiP size from the window size in order to calculate - // the position to snap to from the PiP center and not its top-left - // corner - const windowWidth = ( - UIStore.instance.windowWidth - - (this.callViewWrapper.current?.clientWidth || PIP_VIEW_WIDTH) - ); - const windowHeight = ( - UIStore.instance.windowHeight - - (this.callViewWrapper.current?.clientHeight || PIP_VIEW_HEIGHT) - ); - - if (translationX >= windowWidth / 2 && translationY >= windowHeight / 2) { - this.desiredTranslationX = windowWidth - PADDING.right; - this.desiredTranslationY = windowHeight - PADDING.bottom; - } else if (translationX >= windowWidth / 2 && translationY <= windowHeight / 2) { - this.desiredTranslationX = windowWidth - PADDING.right; - this.desiredTranslationY = PADDING.top; - } else if (translationX <= windowWidth / 2 && translationY >= windowHeight / 2) { - this.desiredTranslationX = PADDING.left; - this.desiredTranslationY = windowHeight - PADDING.bottom; - } else { - this.desiredTranslationX = PADDING.left; - this.desiredTranslationY = PADDING.top; - } - - if (animate) { - // We start animating here because we want the PiP to move when we're - // resizing the window - this.scheduledUpdate.mark(); - } else { - this.setState({ - translationX: this.desiredTranslationX, - translationY: this.desiredTranslationY, - }); - } - } - private onRoomViewStoreUpdate = () => { if (RoomViewStore.getRoomId() === this.state.roomId) return; @@ -300,53 +175,22 @@ export default class CallPreview extends React.Component { }); }; - private onStartMoving = (event: React.MouseEvent) => { - event.preventDefault(); - event.stopPropagation(); - - this.moving = true; - this.initX = event.pageX - this.desiredTranslationX; - this.initY = event.pageY - this.desiredTranslationY; - this.scheduledUpdate.mark(); - }; - - private onMoving = (event: React.MouseEvent | MouseEvent) => { - if (!this.moving) return; - - event.preventDefault(); - event.stopPropagation(); - - this.setTranslation(event.pageX - this.initX, event.pageY - this.initY); - }; - - private onEndMoving = () => { - this.moving = false; - this.snap(true); - }; - public render() { + const pipMode = true; if (this.state.primaryCall) { - const translatePixelsX = this.state.translationX + "px"; - const translatePixelsY = this.state.translationY + "px"; - const style = { - transform: `translateX(${translatePixelsX}) - translateY(${translatePixelsY})`, - }; - return ( -
- -
+ pipMode={pipMode} + /> } + + ); } diff --git a/src/components/views/voip/CallView.tsx b/src/components/views/voip/CallView.tsx index 356e642d65..096cd8a6a9 100644 --- a/src/components/views/voip/CallView.tsx +++ b/src/components/views/voip/CallView.tsx @@ -37,6 +37,7 @@ import DesktopCapturerSourcePicker from "../elements/DesktopCapturerSourcePicker import Modal from '../../../Modal'; import { SDPStreamMetadataPurpose } from 'matrix-js-sdk/src/webrtc/callEventTypes'; import CallViewSidebar from './CallViewSidebar'; +import { CallViewHeader } from './CallView/CallViewHeader'; interface IProps { // The call for us to display @@ -56,7 +57,7 @@ interface IProps { pipMode?: boolean; // Used for dragging the PiP CallView - onMouseDownOnHeader?: (event: React.MouseEvent) => void; + onMouseDownOnHeader?: (event: React.MouseEvent) => void; } interface IState { @@ -115,7 +116,6 @@ export default class CallView extends React.Component { private controlsHideTimer: number = null; private dialpadButton = createRef(); private contextMenuButton = createRef(); - private contextMenu = createRef(); constructor(props: IProps) { super(props); @@ -231,21 +231,6 @@ export default class CallView extends React.Component { }); }; - private onFullscreenClick = () => { - dis.dispatch({ - action: 'video_fullscreen', - fullscreen: true, - }); - }; - - private onExpandClick = () => { - const userFacingRoomId = CallHandler.sharedInstance().roomIdForCall(this.props.call); - dis.dispatch({ - action: 'view_room', - room_id: userFacingRoomId, - }); - }; - private onControlsHideTimer = () => { if (this.state.hoveringControls || this.state.showDialpad || this.state.showMoreMenu) return; this.controlsHideTimer = null; @@ -389,23 +374,6 @@ export default class CallView extends React.Component { this.setState({ hoveringControls: false }); }; - private onRoomAvatarClick = (): void => { - const userFacingRoomId = CallHandler.sharedInstance().roomIdForCall(this.props.call); - dis.dispatch({ - action: 'view_room', - room_id: userFacingRoomId, - }); - }; - - private onSecondaryRoomAvatarClick = (): void => { - const userFacingRoomId = CallHandler.sharedInstance().roomIdForCall(this.props.secondaryCall); - - dis.dispatch({ - action: 'view_room', - room_id: userFacingRoomId, - }); - }; - private onCallResumeClick = (): void => { const userFacingRoomId = CallHandler.sharedInstance().roomIdForCall(this.props.call); CallHandler.sharedInstance().setActiveCallRoomId(userFacingRoomId); @@ -814,83 +782,15 @@ export default class CallView extends React.Component { ); } - const callTypeText = isVideoCall ? _t("Video Call") : _t("Voice Call"); - let myClassName; - - let fullScreenButton; - if (!this.props.pipMode) { - fullScreenButton = ( -
- ); - } - - let expandButton; - if (this.props.pipMode) { - expandButton =
; - } - - const headerControls =
- { fullScreenButton } - { expandButton } -
; - - const callTypeIconClassName = classNames("mx_CallView_header_callTypeIcon", { - "mx_CallView_header_callTypeIcon_voice": !isVideoCall, - "mx_CallView_header_callTypeIcon_video": isVideoCall, - }); - - let header: React.ReactNode; - if (!this.props.pipMode) { - header =
-
- { callTypeText } - { headerControls } -
; - myClassName = 'mx_CallView_large'; - } else { - let secondaryCallInfo; - if (this.props.secondaryCall) { - secondaryCallInfo = - - - - { _t("%(name)s on hold", { name: secCallRoom.name }) } - - - ; - } - - header = ( -
- - - -
-
{ callRoom.name }
-
- { callTypeText } - { secondaryCallInfo } -
-
- { headerControls } -
- ); - myClassName = 'mx_CallView_pip'; - } + const myClassName = this.props.pipMode ? 'mx_CallView_pip' : 'mx_CallView_large'; return
- { header } + { contentView }
; } diff --git a/src/components/views/voip/CallView/CallViewHeader.tsx b/src/components/views/voip/CallView/CallViewHeader.tsx new file mode 100644 index 0000000000..5df45c90a0 --- /dev/null +++ b/src/components/views/voip/CallView/CallViewHeader.tsx @@ -0,0 +1,135 @@ +/* +Copyright 2021 New Vector Ltd + +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 { CallType } from 'matrix-js-sdk/src/webrtc/call'; +import { Room } from 'matrix-js-sdk/src/models/room'; +import React from 'react'; +import { isUndefined } from 'lodash'; +import { _t } from '../../../../languageHandler'; +import RoomAvatar from '../../avatars/RoomAvatar'; +import AccessibleButton from '../../elements/AccessibleButton'; +import dis from '../../../../dispatcher/dispatcher'; +import WidgetAvatar from '../../avatars/WidgetAvatar'; +import { IApp } from '../../../../stores/WidgetStore'; +import WidgetUtils from '../../../../utils/WidgetUtils'; + +const callTypeTranslationByType: Record string> = { + [CallType.Video]: () => _t("Video Call"), + [CallType.Voice]: () => _t("Audio Call"), + 'widget': (app: IApp) => WidgetUtils.getWidgetName(app), +}; + +interface CallViewHeaderProps { + pipMode: boolean; + type: CallType | 'widget'; + callRooms?: Room[]; + app?: IApp; + onPipMouseDown: (event: React.MouseEvent) => void; +} + +const onRoomAvatarClick = (roomId: string) => { + dis.dispatch({ + action: 'view_room', + room_id: roomId, + }); +}; + +const onFullscreenClick = () => { + dis.dispatch({ + action: 'video_fullscreen', + fullscreen: true, + }); +}; + +const onExpandClick = (roomId: string) => { + dis.dispatch({ + action: 'view_room', + room_id: roomId, + }); +}; + +type CallControlsProps = Pick & { + roomId: string; +}; +function CallControls({ pipMode = false, type, roomId }: CallControlsProps) { + return
+ { (pipMode && type === CallType.Video) && +
} + { pipMode &&
onExpandClick(roomId)} + title={_t("Return to call")} + /> } +
; +} +function SecondaryCallInfo({ callRoom }: {callRoom: Room}) { + return + onRoomAvatarClick(callRoom.roomId)}> + + + { _t("%(name)s on hold", { name: callRoom.name }) } + + + ; +} + +function getAvatarBasedOnRoomType(roomOrWidget: Room | IApp) { + if (roomOrWidget instanceof Room) { + return ; + } else if (!isUndefined(roomOrWidget)) { + return ; + } + return null; +} + +export function CallViewHeader({ + type, + pipMode = false, + callRooms = [], + app, + onPipMouseDown, +}: CallViewHeaderProps) { + const [callRoom, onHoldCallRoom] = callRooms; + const callTypeText = callTypeTranslationByType[type](app); + const avatar = getAvatarBasedOnRoomType(callRoom ?? app); + const callRoomName = type === 'widget' ? callTypeText : callRoom.name; + const roomId = app ? app.roomId : callRoom.roomId; + if (!pipMode) { + return
+
+ { callTypeText } + +
; + } + return (
+ onRoomAvatarClick(roomId)}> + { avatar } + +
+
{ callRoomName }
+
+ { callTypeText } + { onHoldCallRoom && } +
+
+ +
+ ); +} diff --git a/src/components/views/voip/PictureInPictureDragger.tsx b/src/components/views/voip/PictureInPictureDragger.tsx new file mode 100644 index 0000000000..14a2a88939 --- /dev/null +++ b/src/components/views/voip/PictureInPictureDragger.tsx @@ -0,0 +1,208 @@ +/* +Copyright 2021 New Vector Ltd + +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 React, { createRef } from 'react'; +import UIStore from '../../../stores/UIStore'; +import { IApp } from '../../../stores/WidgetStore'; +import { lerp } from '../../../utils/AnimationUtils'; +import { MarkedExecution } from '../../../utils/MarkedExecution'; +import { replaceableComponent } from '../../../utils/replaceableComponent'; + +const PIP_VIEW_WIDTH = 336; +const PIP_VIEW_HEIGHT = 232; + +const MOVING_AMT = 0.2; +const SNAPPING_AMT = 0.05; + +const PADDING = { + top: 58, + bottom: 58, + left: 76, + right: 8, +}; + +interface IProps { + className?: string; + children: (event: MouseEvent) => React.ReactNode; + draggable: boolean; + app?: IApp; +} + +interface IState { + // Position of the PictureInPictureDragger + translationX: number; + translationY: number; +} + +/** + * PictureInPictureDragger shows a small version of CallView hovering over the UI in 'picture-in-picture' + * (PiP mode). It displays the call(s) which is *not* in the room the user is currently viewing. + */ +@replaceableComponent("views.voip.PictureInPictureDragger") +export class PictureInPictureDragger extends React.Component { + private callViewWrapper = createRef(); + private initX = 0; + private initY = 0; + private desiredTranslationX = UIStore.instance.windowWidth - PADDING.right - PIP_VIEW_WIDTH; + private desiredTranslationY = UIStore.instance.windowHeight - PADDING.bottom - PIP_VIEW_WIDTH; + private moving = false; + private scheduledUpdate = new MarkedExecution( + () => this.animationCallback(), + () => requestAnimationFrame(() => this.scheduledUpdate.trigger()), + ); + + constructor(props: IProps) { + super(props); + + this.state = { + translationX: UIStore.instance.windowWidth - PADDING.right - PIP_VIEW_WIDTH, + translationY: UIStore.instance.windowHeight - PADDING.bottom - PIP_VIEW_WIDTH, + }; + } + + public componentDidMount() { + document.addEventListener("mousemove", this.onMoving); + document.addEventListener("mouseup", this.onEndMoving); + window.addEventListener("resize", this.snap); + } + + public componentWillUnmount() { + document.removeEventListener("mousemove", this.onMoving); + document.removeEventListener("mouseup", this.onEndMoving); + window.removeEventListener("resize", this.snap); + } + + private animationCallback = () => { + // If the PiP isn't being dragged and there is only a tiny difference in + // the desiredTranslation and translation, quit the animationCallback + // loop. If that is the case, it means the PiP has snapped into its + // position and there is nothing to do. Not doing this would cause an + // infinite loop + if ( + !this.moving && + Math.abs(this.state.translationX - this.desiredTranslationX) <= 1 && + Math.abs(this.state.translationY - this.desiredTranslationY) <= 1 + ) return; + + const amt = this.moving ? MOVING_AMT : SNAPPING_AMT; + this.setState({ + translationX: lerp(this.state.translationX, this.desiredTranslationX, amt), + translationY: lerp(this.state.translationY, this.desiredTranslationY, amt), + }); + this.scheduledUpdate.mark(); + }; + + private setTranslation(inTranslationX: number, inTranslationY: number) { + const width = this.callViewWrapper.current?.clientWidth || PIP_VIEW_WIDTH; + const height = this.callViewWrapper.current?.clientHeight || PIP_VIEW_HEIGHT; + + // Avoid overflow on the x axis + if (inTranslationX + width >= UIStore.instance.windowWidth) { + this.desiredTranslationX = UIStore.instance.windowWidth - width; + } else if (inTranslationX <= 0) { + this.desiredTranslationX = 0; + } else { + this.desiredTranslationX = inTranslationX; + } + + // Avoid overflow on the y axis + if (inTranslationY + height >= UIStore.instance.windowHeight) { + this.desiredTranslationY = UIStore.instance.windowHeight - height; + } else if (inTranslationY <= 0) { + this.desiredTranslationY = 0; + } else { + this.desiredTranslationY = inTranslationY; + } + } + + private snap = () => { + const translationX = this.desiredTranslationX; + const translationY = this.desiredTranslationY; + // We subtract the PiP size from the window size in order to calculate + // the position to snap to from the PiP center and not its top-left + // corner + const windowWidth = ( + UIStore.instance.windowWidth - + (this.callViewWrapper.current?.clientWidth || PIP_VIEW_WIDTH) + ); + const windowHeight = ( + UIStore.instance.windowHeight - + (this.callViewWrapper.current?.clientHeight || PIP_VIEW_HEIGHT) + ); + + if (translationX >= windowWidth / 2 && translationY >= windowHeight / 2) { + this.desiredTranslationX = windowWidth - PADDING.right; + this.desiredTranslationY = windowHeight - PADDING.bottom; + } else if (translationX >= windowWidth / 2 && translationY <= windowHeight / 2) { + this.desiredTranslationX = windowWidth - PADDING.right; + this.desiredTranslationY = PADDING.top; + } else if (translationX <= windowWidth / 2 && translationY >= windowHeight / 2) { + this.desiredTranslationX = PADDING.left; + this.desiredTranslationY = windowHeight - PADDING.bottom; + } else { + this.desiredTranslationX = PADDING.left; + this.desiredTranslationY = PADDING.top; + } + + // We start animating here because we want the PiP to move when we're + // resizing the window + this.scheduledUpdate.mark(); + }; + + private onStartMoving = (event: React.MouseEvent | MouseEvent) => { + event.preventDefault(); + event.stopPropagation(); + + this.moving = true; + this.initX = event.pageX - this.desiredTranslationX; + this.initY = event.pageY - this.desiredTranslationY; + this.scheduledUpdate.mark(); + }; + + private onMoving = (event: React.MouseEvent | MouseEvent) => { + if (!this.moving) return; + + event.preventDefault(); + event.stopPropagation(); + + this.setTranslation(event.pageX - this.initX, event.pageY - this.initY); + }; + + private onEndMoving = () => { + this.moving = false; + this.snap(); + }; + + public render() { + const translatePixelsX = this.state.translationX + "px"; + const translatePixelsY = this.state.translationY + "px"; + const style = { + transform: `translateX(${translatePixelsX}) + translateY(${translatePixelsY})`, + }; + return ( +
+ <> + { this.props.children(this.onStartMoving) } + +
+ ); + } +} + From 36efa448b2ef673d8c282ad87bc3da6bb4d01cf2 Mon Sep 17 00:00:00 2001 From: Dariusz Niemczyk Date: Tue, 3 Aug 2021 13:24:41 +0200 Subject: [PATCH 02/24] Fix TS types --- src/components/views/voip/CallView.tsx | 2 +- src/components/views/voip/PictureInPictureDragger.tsx | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/src/components/views/voip/CallView.tsx b/src/components/views/voip/CallView.tsx index 096cd8a6a9..3fa8d602da 100644 --- a/src/components/views/voip/CallView.tsx +++ b/src/components/views/voip/CallView.tsx @@ -57,7 +57,7 @@ interface IProps { pipMode?: boolean; // Used for dragging the PiP CallView - onMouseDownOnHeader?: (event: React.MouseEvent) => void; + onMouseDownOnHeader?: (event: MouseEvent) => void; } interface IState { diff --git a/src/components/views/voip/PictureInPictureDragger.tsx b/src/components/views/voip/PictureInPictureDragger.tsx index 14a2a88939..1a3c3d8828 100644 --- a/src/components/views/voip/PictureInPictureDragger.tsx +++ b/src/components/views/voip/PictureInPictureDragger.tsx @@ -35,7 +35,7 @@ const PADDING = { interface IProps { className?: string; - children: (event: MouseEvent) => React.ReactNode; + children: (startMovingEventHandler: (event: MouseEvent) => void) => React.ReactNode; draggable: boolean; app?: IApp; } From 29e9db44a319bf6b7fb7438809e64d2df3e849a4 Mon Sep 17 00:00:00 2001 From: Dariusz Niemczyk Date: Tue, 3 Aug 2021 13:30:14 +0200 Subject: [PATCH 03/24] Another fix to the TS types --- src/components/views/voip/CallView.tsx | 2 +- src/components/views/voip/PictureInPictureDragger.tsx | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/src/components/views/voip/CallView.tsx b/src/components/views/voip/CallView.tsx index 3fa8d602da..084efa66d1 100644 --- a/src/components/views/voip/CallView.tsx +++ b/src/components/views/voip/CallView.tsx @@ -57,7 +57,7 @@ interface IProps { pipMode?: boolean; // Used for dragging the PiP CallView - onMouseDownOnHeader?: (event: MouseEvent) => void; + onMouseDownOnHeader?: (event: React.MouseEvent) => void; } interface IState { diff --git a/src/components/views/voip/PictureInPictureDragger.tsx b/src/components/views/voip/PictureInPictureDragger.tsx index 1a3c3d8828..9445ee68b6 100644 --- a/src/components/views/voip/PictureInPictureDragger.tsx +++ b/src/components/views/voip/PictureInPictureDragger.tsx @@ -35,7 +35,7 @@ const PADDING = { interface IProps { className?: string; - children: (startMovingEventHandler: (event: MouseEvent) => void) => React.ReactNode; + children: (startMovingEventHandler: (event: React.MouseEvent) => void) => React.ReactNode; draggable: boolean; app?: IApp; } From 90c2eb0b92e1eb36fbc6bcbfced6cf62c5a20377 Mon Sep 17 00:00:00 2001 From: Dariusz Niemczyk <3636685+Palid@users.noreply.github.com> Date: Tue, 3 Aug 2021 13:50:00 +0200 Subject: [PATCH 04/24] Update src/components/views/voip/CallView/CallViewHeader.tsx MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Co-authored-by: Šimon Brandner --- src/components/views/voip/CallView/CallViewHeader.tsx | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/components/views/voip/CallView/CallViewHeader.tsx b/src/components/views/voip/CallView/CallViewHeader.tsx index 5df45c90a0..954982b9b3 100644 --- a/src/components/views/voip/CallView/CallViewHeader.tsx +++ b/src/components/views/voip/CallView/CallViewHeader.tsx @@ -27,7 +27,7 @@ import WidgetUtils from '../../../../utils/WidgetUtils'; const callTypeTranslationByType: Record string> = { [CallType.Video]: () => _t("Video Call"), - [CallType.Voice]: () => _t("Audio Call"), + [CallType.Voice]: () => _t("Voice Call"), 'widget': (app: IApp) => WidgetUtils.getWidgetName(app), }; From 838d9105c9747b28eb32fd535664ec6f2007ce65 Mon Sep 17 00:00:00 2001 From: Dariusz Niemczyk <3636685+Palid@users.noreply.github.com> Date: Tue, 3 Aug 2021 13:53:49 +0200 Subject: [PATCH 05/24] Update src/components/views/voip/CallView/CallViewHeader.tsx MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Co-authored-by: Šimon Brandner --- src/components/views/voip/CallView/CallViewHeader.tsx | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/components/views/voip/CallView/CallViewHeader.tsx b/src/components/views/voip/CallView/CallViewHeader.tsx index 954982b9b3..e5ee6bff96 100644 --- a/src/components/views/voip/CallView/CallViewHeader.tsx +++ b/src/components/views/voip/CallView/CallViewHeader.tsx @@ -96,13 +96,13 @@ function getAvatarBasedOnRoomType(roomOrWidget: Room | IApp) { return null; } -export function CallViewHeader({ +export const CallViewHeader: React.FC = ({ type, pipMode = false, callRooms = [], app, onPipMouseDown, -}: CallViewHeaderProps) { +}) { const [callRoom, onHoldCallRoom] = callRooms; const callTypeText = callTypeTranslationByType[type](app); const avatar = getAvatarBasedOnRoomType(callRoom ?? app); From a0b0a91d08ce9c14ce3e7fdf22a084847f3b48da Mon Sep 17 00:00:00 2001 From: Dariusz Niemczyk Date: Tue, 3 Aug 2021 14:03:35 +0200 Subject: [PATCH 06/24] Fix ident --- src/components/views/voip/CallPreview.tsx | 4 +- src/components/views/voip/CallView.tsx | 26 +- .../views/voip/CallView/CallViewHeader.tsx | 42 +-- .../views/voip/PictureInPictureDragger.tsx | 276 +++++++++--------- 4 files changed, 174 insertions(+), 174 deletions(-) diff --git a/src/components/views/voip/CallPreview.tsx b/src/components/views/voip/CallPreview.tsx index 565e4657b9..0607215e0c 100644 --- a/src/components/views/voip/CallPreview.tsx +++ b/src/components/views/voip/CallPreview.tsx @@ -144,8 +144,8 @@ export default class CallPreview extends React.Component { private onAction = (payload: ActionPayload) => { switch (payload.action) { - // listen for call state changes to prod the render method, which - // may hide the global CallView if the call it is tracking is dead + // listen for call state changes to prod the render method, which + // may hide the global CallView if the call it is tracking is dead case 'call_state': { this.updateCalls(); break; diff --git a/src/components/views/voip/CallView.tsx b/src/components/views/voip/CallView.tsx index 084efa66d1..c7297afb95 100644 --- a/src/components/views/voip/CallView.tsx +++ b/src/components/views/voip/CallView.tsx @@ -40,23 +40,23 @@ import CallViewSidebar from './CallViewSidebar'; import { CallViewHeader } from './CallView/CallViewHeader'; interface IProps { - // The call for us to display - call: MatrixCall; + // The call for us to display + call: MatrixCall; - // Another ongoing call to display information about - secondaryCall?: MatrixCall; + // Another ongoing call to display information about + secondaryCall?: MatrixCall; - // a callback which is called when the content in the CallView changes - // in a way that is likely to cause a resize. - onResize?: any; + // a callback which is called when the content in the CallView changes + // in a way that is likely to cause a resize. + onResize?: any; - // Whether this call view is for picture-in-picture mode - // otherwise, it's the larger call view when viewing the room the call is in. - // This is sort of a proxy for a number of things but we currently have no - // need to control those things separately, so this is simpler. - pipMode?: boolean; + // Whether this call view is for picture-in-picture mode + // otherwise, it's the larger call view when viewing the room the call is in. + // This is sort of a proxy for a number of things but we currently have no + // need to control those things separately, so this is simpler. + pipMode?: boolean; - // Used for dragging the PiP CallView + // Used for dragging the PiP CallView onMouseDownOnHeader?: (event: React.MouseEvent) => void; } diff --git a/src/components/views/voip/CallView/CallViewHeader.tsx b/src/components/views/voip/CallView/CallViewHeader.tsx index e5ee6bff96..4650e65853 100644 --- a/src/components/views/voip/CallView/CallViewHeader.tsx +++ b/src/components/views/voip/CallView/CallViewHeader.tsx @@ -26,37 +26,37 @@ import { IApp } from '../../../../stores/WidgetStore'; import WidgetUtils from '../../../../utils/WidgetUtils'; const callTypeTranslationByType: Record string> = { - [CallType.Video]: () => _t("Video Call"), - [CallType.Voice]: () => _t("Voice Call"), - 'widget': (app: IApp) => WidgetUtils.getWidgetName(app), + [CallType.Video]: () => _t("Video Call"), + [CallType.Voice]: () => _t("Voice Call"), + 'widget': (app: IApp) => WidgetUtils.getWidgetName(app), }; interface CallViewHeaderProps { - pipMode: boolean; - type: CallType | 'widget'; - callRooms?: Room[]; - app?: IApp; - onPipMouseDown: (event: React.MouseEvent) => void; + pipMode: boolean; + type: CallType | 'widget'; + callRooms?: Room[]; + app?: IApp; + onPipMouseDown: (event: React.MouseEvent) => void; } const onRoomAvatarClick = (roomId: string) => { dis.dispatch({ - action: 'view_room', - room_id: roomId, + action: 'view_room', + room_id: roomId, }); }; const onFullscreenClick = () => { dis.dispatch({ - action: 'video_fullscreen', - fullscreen: true, + action: 'video_fullscreen', + fullscreen: true, }); }; const onExpandClick = (roomId: string) => { dis.dispatch({ - action: 'view_room', - room_id: roomId, + action: 'view_room', + room_id: roomId, }); }; @@ -97,12 +97,12 @@ function getAvatarBasedOnRoomType(roomOrWidget: Room | IApp) { } export const CallViewHeader: React.FC = ({ - type, - pipMode = false, - callRooms = [], - app, - onPipMouseDown, -}) { + type, + pipMode = false, + callRooms = [], + app, + onPipMouseDown, +}) => { const [callRoom, onHoldCallRoom] = callRooms; const callTypeText = callTypeTranslationByType[type](app); const avatar = getAvatarBasedOnRoomType(callRoom ?? app); @@ -132,4 +132,4 @@ export const CallViewHeader: React.FC = ({
); -} +}; diff --git a/src/components/views/voip/PictureInPictureDragger.tsx b/src/components/views/voip/PictureInPictureDragger.tsx index 9445ee68b6..1ef7867992 100644 --- a/src/components/views/voip/PictureInPictureDragger.tsx +++ b/src/components/views/voip/PictureInPictureDragger.tsx @@ -27,23 +27,23 @@ const MOVING_AMT = 0.2; const SNAPPING_AMT = 0.05; const PADDING = { - top: 58, - bottom: 58, - left: 76, - right: 8, + top: 58, + bottom: 58, + left: 76, + right: 8, }; interface IProps { - className?: string; + className?: string; children: (startMovingEventHandler: (event: React.MouseEvent) => void) => React.ReactNode; - draggable: boolean; - app?: IApp; + draggable: boolean; + app?: IApp; } interface IState { - // Position of the PictureInPictureDragger - translationX: number; - translationY: number; + // Position of the PictureInPictureDragger + translationX: number; + translationY: number; } /** @@ -52,157 +52,157 @@ interface IState { */ @replaceableComponent("views.voip.PictureInPictureDragger") export class PictureInPictureDragger extends React.Component { - private callViewWrapper = createRef(); - private initX = 0; - private initY = 0; - private desiredTranslationX = UIStore.instance.windowWidth - PADDING.right - PIP_VIEW_WIDTH; - private desiredTranslationY = UIStore.instance.windowHeight - PADDING.bottom - PIP_VIEW_WIDTH; - private moving = false; - private scheduledUpdate = new MarkedExecution( - () => this.animationCallback(), - () => requestAnimationFrame(() => this.scheduledUpdate.trigger()), - ); + private callViewWrapper = createRef(); + private initX = 0; + private initY = 0; + private desiredTranslationX = UIStore.instance.windowWidth - PADDING.right - PIP_VIEW_WIDTH; + private desiredTranslationY = UIStore.instance.windowHeight - PADDING.bottom - PIP_VIEW_WIDTH; + private moving = false; + private scheduledUpdate = new MarkedExecution( + () => this.animationCallback(), + () => requestAnimationFrame(() => this.scheduledUpdate.trigger()), + ); - constructor(props: IProps) { - super(props); + constructor(props: IProps) { + super(props); - this.state = { - translationX: UIStore.instance.windowWidth - PADDING.right - PIP_VIEW_WIDTH, - translationY: UIStore.instance.windowHeight - PADDING.bottom - PIP_VIEW_WIDTH, - }; - } + this.state = { + translationX: UIStore.instance.windowWidth - PADDING.right - PIP_VIEW_WIDTH, + translationY: UIStore.instance.windowHeight - PADDING.bottom - PIP_VIEW_WIDTH, + }; + } - public componentDidMount() { - document.addEventListener("mousemove", this.onMoving); - document.addEventListener("mouseup", this.onEndMoving); - window.addEventListener("resize", this.snap); - } + public componentDidMount() { + document.addEventListener("mousemove", this.onMoving); + document.addEventListener("mouseup", this.onEndMoving); + window.addEventListener("resize", this.snap); + } - public componentWillUnmount() { - document.removeEventListener("mousemove", this.onMoving); - document.removeEventListener("mouseup", this.onEndMoving); - window.removeEventListener("resize", this.snap); - } + public componentWillUnmount() { + document.removeEventListener("mousemove", this.onMoving); + document.removeEventListener("mouseup", this.onEndMoving); + window.removeEventListener("resize", this.snap); + } - private animationCallback = () => { - // If the PiP isn't being dragged and there is only a tiny difference in - // the desiredTranslation and translation, quit the animationCallback - // loop. If that is the case, it means the PiP has snapped into its - // position and there is nothing to do. Not doing this would cause an - // infinite loop - if ( - !this.moving && + private animationCallback = () => { + // If the PiP isn't being dragged and there is only a tiny difference in + // the desiredTranslation and translation, quit the animationCallback + // loop. If that is the case, it means the PiP has snapped into its + // position and there is nothing to do. Not doing this would cause an + // infinite loop + if ( + !this.moving && Math.abs(this.state.translationX - this.desiredTranslationX) <= 1 && Math.abs(this.state.translationY - this.desiredTranslationY) <= 1 - ) return; + ) return; - const amt = this.moving ? MOVING_AMT : SNAPPING_AMT; - this.setState({ - translationX: lerp(this.state.translationX, this.desiredTranslationX, amt), - translationY: lerp(this.state.translationY, this.desiredTranslationY, amt), - }); - this.scheduledUpdate.mark(); - }; + const amt = this.moving ? MOVING_AMT : SNAPPING_AMT; + this.setState({ + translationX: lerp(this.state.translationX, this.desiredTranslationX, amt), + translationY: lerp(this.state.translationY, this.desiredTranslationY, amt), + }); + this.scheduledUpdate.mark(); + }; - private setTranslation(inTranslationX: number, inTranslationY: number) { - const width = this.callViewWrapper.current?.clientWidth || PIP_VIEW_WIDTH; - const height = this.callViewWrapper.current?.clientHeight || PIP_VIEW_HEIGHT; + private setTranslation(inTranslationX: number, inTranslationY: number) { + const width = this.callViewWrapper.current?.clientWidth || PIP_VIEW_WIDTH; + const height = this.callViewWrapper.current?.clientHeight || PIP_VIEW_HEIGHT; - // Avoid overflow on the x axis - if (inTranslationX + width >= UIStore.instance.windowWidth) { - this.desiredTranslationX = UIStore.instance.windowWidth - width; - } else if (inTranslationX <= 0) { - this.desiredTranslationX = 0; - } else { - this.desiredTranslationX = inTranslationX; - } + // Avoid overflow on the x axis + if (inTranslationX + width >= UIStore.instance.windowWidth) { + this.desiredTranslationX = UIStore.instance.windowWidth - width; + } else if (inTranslationX <= 0) { + this.desiredTranslationX = 0; + } else { + this.desiredTranslationX = inTranslationX; + } - // Avoid overflow on the y axis - if (inTranslationY + height >= UIStore.instance.windowHeight) { - this.desiredTranslationY = UIStore.instance.windowHeight - height; - } else if (inTranslationY <= 0) { - this.desiredTranslationY = 0; - } else { - this.desiredTranslationY = inTranslationY; - } - } + // Avoid overflow on the y axis + if (inTranslationY + height >= UIStore.instance.windowHeight) { + this.desiredTranslationY = UIStore.instance.windowHeight - height; + } else if (inTranslationY <= 0) { + this.desiredTranslationY = 0; + } else { + this.desiredTranslationY = inTranslationY; + } + } - private snap = () => { - const translationX = this.desiredTranslationX; - const translationY = this.desiredTranslationY; - // We subtract the PiP size from the window size in order to calculate - // the position to snap to from the PiP center and not its top-left - // corner - const windowWidth = ( - UIStore.instance.windowWidth - + private snap = () => { + const translationX = this.desiredTranslationX; + const translationY = this.desiredTranslationY; + // We subtract the PiP size from the window size in order to calculate + // the position to snap to from the PiP center and not its top-left + // corner + const windowWidth = ( + UIStore.instance.windowWidth - (this.callViewWrapper.current?.clientWidth || PIP_VIEW_WIDTH) - ); - const windowHeight = ( - UIStore.instance.windowHeight - + ); + const windowHeight = ( + UIStore.instance.windowHeight - (this.callViewWrapper.current?.clientHeight || PIP_VIEW_HEIGHT) - ); + ); - if (translationX >= windowWidth / 2 && translationY >= windowHeight / 2) { - this.desiredTranslationX = windowWidth - PADDING.right; - this.desiredTranslationY = windowHeight - PADDING.bottom; - } else if (translationX >= windowWidth / 2 && translationY <= windowHeight / 2) { - this.desiredTranslationX = windowWidth - PADDING.right; - this.desiredTranslationY = PADDING.top; - } else if (translationX <= windowWidth / 2 && translationY >= windowHeight / 2) { - this.desiredTranslationX = PADDING.left; - this.desiredTranslationY = windowHeight - PADDING.bottom; - } else { - this.desiredTranslationX = PADDING.left; - this.desiredTranslationY = PADDING.top; - } + if (translationX >= windowWidth / 2 && translationY >= windowHeight / 2) { + this.desiredTranslationX = windowWidth - PADDING.right; + this.desiredTranslationY = windowHeight - PADDING.bottom; + } else if (translationX >= windowWidth / 2 && translationY <= windowHeight / 2) { + this.desiredTranslationX = windowWidth - PADDING.right; + this.desiredTranslationY = PADDING.top; + } else if (translationX <= windowWidth / 2 && translationY >= windowHeight / 2) { + this.desiredTranslationX = PADDING.left; + this.desiredTranslationY = windowHeight - PADDING.bottom; + } else { + this.desiredTranslationX = PADDING.left; + this.desiredTranslationY = PADDING.top; + } - // We start animating here because we want the PiP to move when we're - // resizing the window - this.scheduledUpdate.mark(); - }; + // We start animating here because we want the PiP to move when we're + // resizing the window + this.scheduledUpdate.mark(); + }; - private onStartMoving = (event: React.MouseEvent | MouseEvent) => { - event.preventDefault(); - event.stopPropagation(); + private onStartMoving = (event: React.MouseEvent | MouseEvent) => { + event.preventDefault(); + event.stopPropagation(); - this.moving = true; - this.initX = event.pageX - this.desiredTranslationX; - this.initY = event.pageY - this.desiredTranslationY; - this.scheduledUpdate.mark(); - }; + this.moving = true; + this.initX = event.pageX - this.desiredTranslationX; + this.initY = event.pageY - this.desiredTranslationY; + this.scheduledUpdate.mark(); + }; - private onMoving = (event: React.MouseEvent | MouseEvent) => { - if (!this.moving) return; + private onMoving = (event: React.MouseEvent | MouseEvent) => { + if (!this.moving) return; - event.preventDefault(); - event.stopPropagation(); + event.preventDefault(); + event.stopPropagation(); - this.setTranslation(event.pageX - this.initX, event.pageY - this.initY); - }; + this.setTranslation(event.pageX - this.initX, event.pageY - this.initY); + }; - private onEndMoving = () => { - this.moving = false; - this.snap(); - }; + private onEndMoving = () => { + this.moving = false; + this.snap(); + }; - public render() { - const translatePixelsX = this.state.translationX + "px"; - const translatePixelsY = this.state.translationY + "px"; - const style = { - transform: `translateX(${translatePixelsX}) + public render() { + const translatePixelsX = this.state.translationX + "px"; + const translatePixelsY = this.state.translationY + "px"; + const style = { + transform: `translateX(${translatePixelsX}) translateY(${translatePixelsY})`, - }; - return ( -
- <> - { this.props.children(this.onStartMoving) } - -
- ); - } + }; + return ( +
+ <> + { this.props.children(this.onStartMoving) } + +
+ ); + } } From f592d37f3931625effd62280ec70bea0de3f9a2a Mon Sep 17 00:00:00 2001 From: Dariusz Niemczyk Date: Tue, 3 Aug 2021 14:15:14 +0200 Subject: [PATCH 07/24] Remove widget support for CallViewHeader --- .../views/voip/CallView/CallViewHeader.tsx | 69 +++++++++---------- .../views/voip/PictureInPictureDragger.tsx | 1 + 2 files changed, 32 insertions(+), 38 deletions(-) diff --git a/src/components/views/voip/CallView/CallViewHeader.tsx b/src/components/views/voip/CallView/CallViewHeader.tsx index 4650e65853..d721fe30e3 100644 --- a/src/components/views/voip/CallView/CallViewHeader.tsx +++ b/src/components/views/voip/CallView/CallViewHeader.tsx @@ -16,26 +16,20 @@ limitations under the License. import { CallType } from 'matrix-js-sdk/src/webrtc/call'; import { Room } from 'matrix-js-sdk/src/models/room'; import React from 'react'; -import { isUndefined } from 'lodash'; import { _t } from '../../../../languageHandler'; import RoomAvatar from '../../avatars/RoomAvatar'; import AccessibleButton from '../../elements/AccessibleButton'; import dis from '../../../../dispatcher/dispatcher'; -import WidgetAvatar from '../../avatars/WidgetAvatar'; -import { IApp } from '../../../../stores/WidgetStore'; -import WidgetUtils from '../../../../utils/WidgetUtils'; -const callTypeTranslationByType: Record string> = { +const callTypeTranslationByType: Record string> = { [CallType.Video]: () => _t("Video Call"), [CallType.Voice]: () => _t("Voice Call"), - 'widget': (app: IApp) => WidgetUtils.getWidgetName(app), }; interface CallViewHeaderProps { pipMode: boolean; - type: CallType | 'widget'; + type: CallType; callRooms?: Room[]; - app?: IApp; onPipMouseDown: (event: React.MouseEvent) => void; } @@ -63,7 +57,7 @@ const onExpandClick = (roomId: string) => { type CallControlsProps = Pick & { roomId: string; }; -function CallControls({ pipMode = false, type, roomId }: CallControlsProps) { +function CallViewHeaderControls({ pipMode = false, type, roomId }: CallControlsProps): JSX.Element { return
{ (pipMode && type === CallType.Video) &&
}
; } -function SecondaryCallInfo({ callRoom }: {callRoom: Room}) { +function SecondaryCallInfo({ callRoom }: {callRoom: Room}): JSX.Element { return onRoomAvatarClick(callRoom.roomId)}> @@ -87,49 +81,48 @@ function SecondaryCallInfo({ callRoom }: {callRoom: Room}) { ; } -function getAvatarBasedOnRoomType(roomOrWidget: Room | IApp) { - if (roomOrWidget instanceof Room) { - return ; - } else if (!isUndefined(roomOrWidget)) { - return ; - } - return null; +function CallTypeIcon({ type }: {type: CallType}) { + const classes = { + [CallType.Video]: 'mx_CallView_header_callTypeIcon_video', + [CallType.Voice]: 'mx_CallView_header_callTypeIcon_voice', + }; + const iconClass = classes[type] ?? ''; + return
; } export const CallViewHeader: React.FC = ({ type, pipMode = false, callRooms = [], - app, onPipMouseDown, }) => { const [callRoom, onHoldCallRoom] = callRooms; - const callTypeText = callTypeTranslationByType[type](app); - const avatar = getAvatarBasedOnRoomType(callRoom ?? app); - const callRoomName = type === 'widget' ? callTypeText : callRoom.name; - const roomId = app ? app.roomId : callRoom.roomId; + const callTypeText = callTypeTranslationByType[type]; + const callRoomName = callRoom.name; + const { roomId } = callRoom; if (!pipMode) { return
-
+ { callTypeText } - +
; } - return (
- onRoomAvatarClick(roomId)}> - { avatar } - -
-
{ callRoomName }
-
- { callTypeText } - { onHoldCallRoom && } + return ( +
+ onRoomAvatarClick(roomId)}> + ; + +
+
{ callRoomName }
+
+ { callTypeText } + { onHoldCallRoom && } +
+
- -
); }; diff --git a/src/components/views/voip/PictureInPictureDragger.tsx b/src/components/views/voip/PictureInPictureDragger.tsx index 1ef7867992..d55d431e8f 100644 --- a/src/components/views/voip/PictureInPictureDragger.tsx +++ b/src/components/views/voip/PictureInPictureDragger.tsx @@ -13,6 +13,7 @@ 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 React, { createRef } from 'react'; import UIStore from '../../../stores/UIStore'; import { IApp } from '../../../stores/WidgetStore'; From 466151a10c5e92d2543212e187a5763b3a09e96b Mon Sep 17 00:00:00 2001 From: Dariusz Niemczyk <3636685+Palid@users.noreply.github.com> Date: Tue, 3 Aug 2021 15:00:55 +0200 Subject: [PATCH 08/24] Update src/components/views/voip/CallPreview.tsx MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Co-authored-by: Šimon Brandner --- src/components/views/voip/CallPreview.tsx | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/src/components/views/voip/CallPreview.tsx b/src/components/views/voip/CallPreview.tsx index 0607215e0c..e68e9fd148 100644 --- a/src/components/views/voip/CallPreview.tsx +++ b/src/components/views/voip/CallPreview.tsx @@ -144,9 +144,10 @@ export default class CallPreview extends React.Component { private onAction = (payload: ActionPayload) => { switch (payload.action) { - // listen for call state changes to prod the render method, which - // may hide the global CallView if the call it is tracking is dead case 'call_state': { + // listen for call state changes to prod the render method, which + // may hide the global CallView if the call it is tracking is dead + this.updateCalls(); break; } @@ -197,4 +198,3 @@ export default class CallPreview extends React.Component { return ; } } - From 389d0b2d4ae4022e4eb36bd491a2f9dedc6793b1 Mon Sep 17 00:00:00 2001 From: Dariusz Niemczyk Date: Tue, 3 Aug 2021 14:59:35 +0200 Subject: [PATCH 09/24] Another fix for indentation --- src/components/views/voip/CallView.tsx | 4 ++-- .../views/voip/CallView/CallViewHeader.tsx | 2 +- .../views/voip/PictureInPictureDragger.tsx | 18 +++++++++--------- 3 files changed, 12 insertions(+), 12 deletions(-) diff --git a/src/components/views/voip/CallView.tsx b/src/components/views/voip/CallView.tsx index c7297afb95..5f755b74d0 100644 --- a/src/components/views/voip/CallView.tsx +++ b/src/components/views/voip/CallView.tsx @@ -660,7 +660,7 @@ export default class CallView extends React.Component { let onHoldBackground = null; const backgroundStyle: CSSProperties = {}; const backgroundAvatarUrl = avatarUrlForMember( - // is it worth getting the size of the div to pass here? + // is it worth getting the size of the div to pass here? this.props.call.getOpponentMember(), 1024, 1024, 'crop', ); backgroundStyle.backgroundImage = 'url(' + backgroundAvatarUrl + ')'; @@ -680,7 +680,7 @@ export default class CallView extends React.Component { mx_CallView_voice_hold: isOnHold, }); - contentView =( + contentView = (
{ }; type CallControlsProps = Pick & { - roomId: string; + roomId: string; }; function CallViewHeaderControls({ pipMode = false, type, roomId }: CallControlsProps): JSX.Element { return
diff --git a/src/components/views/voip/PictureInPictureDragger.tsx b/src/components/views/voip/PictureInPictureDragger.tsx index d55d431e8f..8fbe88f2ce 100644 --- a/src/components/views/voip/PictureInPictureDragger.tsx +++ b/src/components/views/voip/PictureInPictureDragger.tsx @@ -68,8 +68,8 @@ export class PictureInPictureDragger extends React.Component { super(props); this.state = { - translationX: UIStore.instance.windowWidth - PADDING.right - PIP_VIEW_WIDTH, - translationY: UIStore.instance.windowHeight - PADDING.bottom - PIP_VIEW_WIDTH, + translationX: UIStore.instance.windowWidth - PADDING.right - PIP_VIEW_WIDTH, + translationY: UIStore.instance.windowHeight - PADDING.bottom - PIP_VIEW_WIDTH, }; } @@ -93,14 +93,14 @@ export class PictureInPictureDragger extends React.Component { // infinite loop if ( !this.moving && - Math.abs(this.state.translationX - this.desiredTranslationX) <= 1 && - Math.abs(this.state.translationY - this.desiredTranslationY) <= 1 + Math.abs(this.state.translationX - this.desiredTranslationX) <= 1 && + Math.abs(this.state.translationY - this.desiredTranslationY) <= 1 ) return; const amt = this.moving ? MOVING_AMT : SNAPPING_AMT; this.setState({ - translationX: lerp(this.state.translationX, this.desiredTranslationX, amt), - translationY: lerp(this.state.translationY, this.desiredTranslationY, amt), + translationX: lerp(this.state.translationX, this.desiredTranslationX, amt), + translationY: lerp(this.state.translationY, this.desiredTranslationY, amt), }); this.scheduledUpdate.mark(); }; @@ -136,11 +136,11 @@ export class PictureInPictureDragger extends React.Component { // corner const windowWidth = ( UIStore.instance.windowWidth - - (this.callViewWrapper.current?.clientWidth || PIP_VIEW_WIDTH) + (this.callViewWrapper.current?.clientWidth || PIP_VIEW_WIDTH) ); const windowHeight = ( UIStore.instance.windowHeight - - (this.callViewWrapper.current?.clientHeight || PIP_VIEW_HEIGHT) + (this.callViewWrapper.current?.clientHeight || PIP_VIEW_HEIGHT) ); if (translationX >= windowWidth / 2 && translationY >= windowHeight / 2) { @@ -190,7 +190,7 @@ export class PictureInPictureDragger extends React.Component { const translatePixelsX = this.state.translationX + "px"; const translatePixelsY = this.state.translationY + "px"; const style = { - transform: `translateX(${translatePixelsX}) + transform: `translateX(${translatePixelsX}) translateY(${translatePixelsY})`, }; return ( From 3ab54bce6ea710342f3e12c023d0608077886996 Mon Sep 17 00:00:00 2001 From: Dariusz Niemczyk Date: Tue, 3 Aug 2021 15:00:11 +0200 Subject: [PATCH 10/24] Move CallViewHeader css to separate file --- res/css/_components.scss | 1 + res/css/views/voip/_CallView.scss | 113 --------------------- res/css/views/voip/_CallViewHeader.scss | 128 ++++++++++++++++++++++++ 3 files changed, 129 insertions(+), 113 deletions(-) create mode 100644 res/css/views/voip/_CallViewHeader.scss diff --git a/res/css/_components.scss b/res/css/_components.scss index 76551b51f8..174dc76d7f 100644 --- a/res/css/_components.scss +++ b/res/css/_components.scss @@ -273,6 +273,7 @@ @import "./views/voip/_CallView.scss"; @import "./views/voip/_CallViewForRoom.scss"; @import "./views/voip/_CallViewSidebar.scss"; +@import "./views/voip/_CallViewHeader.scss"; @import "./views/voip/_DialPad.scss"; @import "./views/voip/_DialPadContextMenu.scss"; @import "./views/voip/_DialPadModal.scss"; diff --git a/res/css/views/voip/_CallView.scss b/res/css/views/voip/_CallView.scss index eff865f20c..6505bbfcbd 100644 --- a/res/css/views/voip/_CallView.scss +++ b/res/css/views/voip/_CallView.scss @@ -201,119 +201,6 @@ limitations under the License. } } -.mx_CallView_header { - height: 44px; - display: flex; - flex-direction: row; - align-items: center; - justify-content: left; - flex-shrink: 0; -} - -.mx_CallView_header_callType { - font-size: 1.2rem; - font-weight: bold; - vertical-align: middle; -} - -.mx_CallView_header_secondaryCallInfo { - &::before { - content: '·'; - margin-left: 6px; - margin-right: 6px; - } -} - -.mx_CallView_header_controls { - margin-left: auto; -} - -.mx_CallView_header_button { - display: inline-block; - vertical-align: middle; - cursor: pointer; - - &::before { - content: ''; - display: inline-block; - height: 20px; - width: 20px; - vertical-align: middle; - background-color: $secondary-fg-color; - mask-repeat: no-repeat; - mask-size: contain; - mask-position: center; - } -} - -.mx_CallView_header_button_fullscreen { - &::before { - mask-image: url('$(res)/img/element-icons/call/fullscreen.svg'); - } -} - -.mx_CallView_header_button_expand { - &::before { - mask-image: url('$(res)/img/element-icons/call/expand.svg'); - } -} - -.mx_CallView_header_callInfo { - margin-left: 12px; - margin-right: 16px; -} - -.mx_CallView_header_roomName { - font-weight: bold; - font-size: 12px; - line-height: initial; - height: 15px; -} - -.mx_CallView_secondaryCall_roomName { - margin-left: 4px; -} - -.mx_CallView_header_callTypeSmall { - font-size: 12px; - color: $secondary-fg-color; - line-height: initial; - height: 15px; - overflow: hidden; - white-space: nowrap; - text-overflow: ellipsis; - max-width: 240px; -} - -.mx_CallView_header_callTypeIcon { - display: inline-block; - margin-right: 6px; - height: 16px; - width: 16px; - vertical-align: middle; - - &::before { - content: ''; - display: inline-block; - vertical-align: top; - - height: 16px; - width: 16px; - background-color: $secondary-fg-color; - mask-repeat: no-repeat; - mask-size: contain; - mask-position: center; - } - - &.mx_CallView_header_callTypeIcon_voice::before { - mask-image: url('$(res)/img/element-icons/call/voice-call.svg'); - } - - &.mx_CallView_header_callTypeIcon_video::before { - mask-image: url('$(res)/img/element-icons/call/video-call.svg'); - } -} - .mx_CallView_callControls { position: absolute; display: flex; diff --git a/res/css/views/voip/_CallViewHeader.scss b/res/css/views/voip/_CallViewHeader.scss new file mode 100644 index 0000000000..9dff07090f --- /dev/null +++ b/res/css/views/voip/_CallViewHeader.scss @@ -0,0 +1,128 @@ +/* +Copyright 2021 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_CallView_header { + height: 44px; + display: flex; + flex-direction: row; + align-items: center; + justify-content: left; + flex-shrink: 0; +} + +.mx_CallView_header_callType { + font-size: 1.2rem; + font-weight: bold; + vertical-align: middle; +} + +.mx_CallView_header_secondaryCallInfo { + &::before { + content: '·'; + margin-left: 6px; + margin-right: 6px; + } +} + +.mx_CallView_header_controls { + margin-left: auto; +} + +.mx_CallView_header_button { + display: inline-block; + vertical-align: middle; + cursor: pointer; + + &::before { + content: ''; + display: inline-block; + height: 20px; + width: 20px; + vertical-align: middle; + background-color: $secondary-fg-color; + mask-repeat: no-repeat; + mask-size: contain; + mask-position: center; + } +} + +.mx_CallView_header_button_fullscreen { + &::before { + mask-image: url('$(res)/img/element-icons/call/fullscreen.svg'); + } +} + +.mx_CallView_header_button_expand { + &::before { + mask-image: url('$(res)/img/element-icons/call/expand.svg'); + } +} + +.mx_CallView_header_callInfo { + margin-left: 12px; + margin-right: 16px; +} + +.mx_CallView_header_roomName { + font-weight: bold; + font-size: 12px; + line-height: initial; + height: 15px; +} + +.mx_CallView_secondaryCall_roomName { + margin-left: 4px; +} + +.mx_CallView_header_callTypeSmall { + font-size: 12px; + color: $secondary-fg-color; + line-height: initial; + height: 15px; + overflow: hidden; + white-space: nowrap; + text-overflow: ellipsis; + max-width: 240px; +} + +.mx_CallView_header_callTypeIcon { + display: inline-block; + margin-right: 6px; + height: 16px; + width: 16px; + vertical-align: middle; + + &::before { + content: ''; + display: inline-block; + vertical-align: top; + + height: 16px; + width: 16px; + background-color: $secondary-fg-color; + mask-repeat: no-repeat; + mask-size: contain; + mask-position: center; + } + + &.mx_CallView_header_callTypeIcon_voice::before { + mask-image: url('$(res)/img/element-icons/call/voice-call.svg'); + } + + &.mx_CallView_header_callTypeIcon_video::before { + mask-image: url('$(res)/img/element-icons/call/video-call.svg'); + } +} From 08b27a7c8284e0c5358f452d330eabf5ae4a9471 Mon Sep 17 00:00:00 2001 From: Dariusz Niemczyk <3636685+Palid@users.noreply.github.com> Date: Tue, 3 Aug 2021 15:15:30 +0200 Subject: [PATCH 11/24] Update src/components/views/voip/PictureInPictureDragger.tsx MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Co-authored-by: Šimon Brandner --- src/components/views/voip/PictureInPictureDragger.tsx | 2 -- 1 file changed, 2 deletions(-) diff --git a/src/components/views/voip/PictureInPictureDragger.tsx b/src/components/views/voip/PictureInPictureDragger.tsx index 8fbe88f2ce..a066ee1fc8 100644 --- a/src/components/views/voip/PictureInPictureDragger.tsx +++ b/src/components/views/voip/PictureInPictureDragger.tsx @@ -38,7 +38,6 @@ interface IProps { className?: string; children: (startMovingEventHandler: (event: React.MouseEvent) => void) => React.ReactNode; draggable: boolean; - app?: IApp; } interface IState { @@ -206,4 +205,3 @@ export class PictureInPictureDragger extends React.Component { ); } } - From 8a1def1d26b01ccaaa224a31c2942c45bb0c4bb5 Mon Sep 17 00:00:00 2001 From: Dariusz Niemczyk <3636685+Palid@users.noreply.github.com> Date: Tue, 3 Aug 2021 15:15:35 +0200 Subject: [PATCH 12/24] Update src/components/views/voip/CallView/CallViewHeader.tsx MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Co-authored-by: Šimon Brandner --- src/components/views/voip/CallView/CallViewHeader.tsx | 1 + 1 file changed, 1 insertion(+) diff --git a/src/components/views/voip/CallView/CallViewHeader.tsx b/src/components/views/voip/CallView/CallViewHeader.tsx index 018089d5d5..117dab0b96 100644 --- a/src/components/views/voip/CallView/CallViewHeader.tsx +++ b/src/components/views/voip/CallView/CallViewHeader.tsx @@ -13,6 +13,7 @@ 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 { CallType } from 'matrix-js-sdk/src/webrtc/call'; import { Room } from 'matrix-js-sdk/src/models/room'; import React from 'react'; From b18d03be329b18becd1e61c78bf6b90c410a8626 Mon Sep 17 00:00:00 2001 From: Dariusz Niemczyk Date: Tue, 3 Aug 2021 15:20:55 +0200 Subject: [PATCH 13/24] Fix final style issues --- .../views/voip/CallView/CallViewHeader.tsx | 25 ++++++++++--------- 1 file changed, 13 insertions(+), 12 deletions(-) diff --git a/src/components/views/voip/CallView/CallViewHeader.tsx b/src/components/views/voip/CallView/CallViewHeader.tsx index 117dab0b96..a5f9de3b7d 100644 --- a/src/components/views/voip/CallView/CallViewHeader.tsx +++ b/src/components/views/voip/CallView/CallViewHeader.tsx @@ -21,6 +21,7 @@ import { _t } from '../../../../languageHandler'; import RoomAvatar from '../../avatars/RoomAvatar'; import AccessibleButton from '../../elements/AccessibleButton'; import dis from '../../../../dispatcher/dispatcher'; +import classNames from 'classnames'; const callTypeTranslationByType: Record string> = { [CallType.Video]: () => _t("Video Call"), @@ -58,7 +59,7 @@ const onExpandClick = (roomId: string) => { type CallControlsProps = Pick & { roomId: string; }; -function CallViewHeaderControls({ pipMode = false, type, roomId }: CallControlsProps): JSX.Element { +const CallViewHeaderControls: React.FC = ({ pipMode = false, type, roomId }) => { return
{ (pipMode && type === CallType.Video) &&
}
; -} -function SecondaryCallInfo({ callRoom }: {callRoom: Room}): JSX.Element { +}; +const SecondaryCallInfo: React.FC<{ callRoom: Room }> = ({ callRoom }) => { return onRoomAvatarClick(callRoom.roomId)}> @@ -80,16 +81,16 @@ function SecondaryCallInfo({ callRoom }: {callRoom: Room}): JSX.Element { ; -} +}; -function CallTypeIcon({ type }: {type: CallType}) { - const classes = { - [CallType.Video]: 'mx_CallView_header_callTypeIcon_video', - [CallType.Voice]: 'mx_CallView_header_callTypeIcon_voice', - }; - const iconClass = classes[type] ?? ''; - return
; -} +const CallTypeIcon: React.FC<{ type: CallType }> = ({ type }) => { + const classes = classNames({ + 'mx_CallView_header_callTypeIcon': true, + 'mx_CallView_header_callTypeIcon_video': type === CallType.Video, + 'mx_CallView_header_callTypeIcon_voice': type === CallType.Voice, + }); + return
; +}; export const CallViewHeader: React.FC = ({ type, From 7487636f90f53306ac9fcca40876b7f0b97dea1c Mon Sep 17 00:00:00 2001 From: Dariusz Niemczyk Date: Tue, 3 Aug 2021 18:00:11 +0200 Subject: [PATCH 14/24] Fix linter again --- src/components/views/voip/PictureInPictureDragger.tsx | 1 - 1 file changed, 1 deletion(-) diff --git a/src/components/views/voip/PictureInPictureDragger.tsx b/src/components/views/voip/PictureInPictureDragger.tsx index a066ee1fc8..d3b4c8a5d1 100644 --- a/src/components/views/voip/PictureInPictureDragger.tsx +++ b/src/components/views/voip/PictureInPictureDragger.tsx @@ -16,7 +16,6 @@ limitations under the License. import React, { createRef } from 'react'; import UIStore from '../../../stores/UIStore'; -import { IApp } from '../../../stores/WidgetStore'; import { lerp } from '../../../utils/AnimationUtils'; import { MarkedExecution } from '../../../utils/MarkedExecution'; import { replaceableComponent } from '../../../utils/replaceableComponent'; From 806b36856dbd69e331624f9d4260cfcde34b0c98 Mon Sep 17 00:00:00 2001 From: Dariusz Niemczyk Date: Tue, 3 Aug 2021 18:01:34 +0200 Subject: [PATCH 15/24] Fix i18n --- src/i18n/strings/en_EN.json | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/src/i18n/strings/en_EN.json b/src/i18n/strings/en_EN.json index 3ad8daa85c..235701c86a 100644 --- a/src/i18n/strings/en_EN.json +++ b/src/i18n/strings/en_EN.json @@ -907,11 +907,6 @@ "%(sharerName)s is presenting": "%(sharerName)s is presenting", "Your camera is turned off": "Your camera is turned off", "Your camera is still enabled": "Your camera is still enabled", - "Video Call": "Video Call", - "Voice Call": "Voice Call", - "Fill Screen": "Fill Screen", - "Return to call": "Return to call", - "%(name)s on hold": "%(name)s on hold", "Unknown caller": "Unknown caller", "Incoming voice call": "Incoming voice call", "Incoming video call": "Incoming video call", @@ -920,6 +915,11 @@ "Silence call": "Silence call", "Decline": "Decline", "Accept": "Accept", + "Video Call": "Video Call", + "Voice Call": "Voice Call", + "Fill Screen": "Fill Screen", + "Return to call": "Return to call", + "%(name)s on hold": "%(name)s on hold", "The other party cancelled the verification.": "The other party cancelled the verification.", "Verified!": "Verified!", "You've successfully verified this user.": "You've successfully verified this user.", From 35386078a497ce0e8f2cac956c0ce719569f557e Mon Sep 17 00:00:00 2001 From: Dariusz Niemczyk Date: Fri, 6 Aug 2021 13:25:27 +0200 Subject: [PATCH 16/24] Post-merge changes --- res/css/views/voip/_CallViewHeader.scss | 1 + src/components/views/voip/PictureInPictureDragger.tsx | 2 +- 2 files changed, 2 insertions(+), 1 deletion(-) diff --git a/res/css/views/voip/_CallViewHeader.scss b/res/css/views/voip/_CallViewHeader.scss index 9dff07090f..1ae3c6a273 100644 --- a/res/css/views/voip/_CallViewHeader.scss +++ b/res/css/views/voip/_CallViewHeader.scss @@ -21,6 +21,7 @@ limitations under the License. align-items: center; justify-content: left; flex-shrink: 0; + cursor: pointer; } .mx_CallView_header_callType { diff --git a/src/components/views/voip/PictureInPictureDragger.tsx b/src/components/views/voip/PictureInPictureDragger.tsx index d3b4c8a5d1..298bb6d536 100644 --- a/src/components/views/voip/PictureInPictureDragger.tsx +++ b/src/components/views/voip/PictureInPictureDragger.tsx @@ -24,7 +24,7 @@ const PIP_VIEW_WIDTH = 336; const PIP_VIEW_HEIGHT = 232; const MOVING_AMT = 0.2; -const SNAPPING_AMT = 0.05; +const SNAPPING_AMT = 0.1; const PADDING = { top: 58, From e8762fbe5b8b3e579b31fe99fd8f567af4c7b902 Mon Sep 17 00:00:00 2001 From: Dariusz Niemczyk Date: Fri, 6 Aug 2021 13:28:53 +0200 Subject: [PATCH 17/24] Fix merge conflicts --- src/components/views/voip/CallView/CallViewHeader.tsx | 11 ++++++----- 1 file changed, 6 insertions(+), 5 deletions(-) diff --git a/src/components/views/voip/CallView/CallViewHeader.tsx b/src/components/views/voip/CallView/CallViewHeader.tsx index a5f9de3b7d..710a0c4372 100644 --- a/src/components/views/voip/CallView/CallViewHeader.tsx +++ b/src/components/views/voip/CallView/CallViewHeader.tsx @@ -22,6 +22,7 @@ import RoomAvatar from '../../avatars/RoomAvatar'; import AccessibleButton from '../../elements/AccessibleButton'; import dis from '../../../../dispatcher/dispatcher'; import classNames from 'classnames'; +import AccessibleTooltipButton from '../../elements/AccessibleTooltipButton'; const callTypeTranslationByType: Record string> = { [CallType.Video]: () => _t("Video Call"), @@ -62,11 +63,11 @@ type CallControlsProps = Pick & { const CallViewHeaderControls: React.FC = ({ pipMode = false, type, roomId }) => { return
{ (pipMode && type === CallType.Video) && -
} - { pipMode &&
} + { pipMode && onExpandClick(roomId)} title={_t("Return to call")} /> } From f6c4412b5fb3a57492974f4d66e106e991117912 Mon Sep 17 00:00:00 2001 From: Dariusz Niemczyk Date: Fri, 6 Aug 2021 13:30:20 +0200 Subject: [PATCH 18/24] Remove old translations --- src/i18n/strings/en_EN.json | 3 --- 1 file changed, 3 deletions(-) diff --git a/src/i18n/strings/en_EN.json b/src/i18n/strings/en_EN.json index 6b8d6eeef5..9f30776f44 100644 --- a/src/i18n/strings/en_EN.json +++ b/src/i18n/strings/en_EN.json @@ -925,9 +925,6 @@ "%(sharerName)s is presenting": "%(sharerName)s is presenting", "Your camera is turned off": "Your camera is turned off", "Your camera is still enabled": "Your camera is still enabled", - "Incoming voice call": "Incoming voice call", - "Incoming video call": "Incoming video call", - "Incoming call": "Incoming call", "Video Call": "Video Call", "Voice Call": "Voice Call", "Fill Screen": "Fill Screen", From 8ac82457e8deeb902f93f8c33915ba65d04edcc8 Mon Sep 17 00:00:00 2001 From: Dariusz Niemczyk <3636685+Palid@users.noreply.github.com> Date: Fri, 6 Aug 2021 14:10:59 +0200 Subject: [PATCH 19/24] Update src/components/views/voip/PictureInPictureDragger.tsx MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Co-authored-by: Šimon Brandner --- src/components/views/voip/PictureInPictureDragger.tsx | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/components/views/voip/PictureInPictureDragger.tsx b/src/components/views/voip/PictureInPictureDragger.tsx index 298bb6d536..244ecddee4 100644 --- a/src/components/views/voip/PictureInPictureDragger.tsx +++ b/src/components/views/voip/PictureInPictureDragger.tsx @@ -189,7 +189,7 @@ export class PictureInPictureDragger extends React.Component { const translatePixelsY = this.state.translationY + "px"; const style = { transform: `translateX(${translatePixelsX}) - translateY(${translatePixelsY})`, + translateY(${translatePixelsY})`, }; return (
Date: Fri, 6 Aug 2021 14:17:39 +0200 Subject: [PATCH 20/24] Rename css --- res/css/views/voip/_CallViewHeader.scss | 26 +++++++++---------- .../views/voip/CallView/CallViewHeader.tsx | 26 +++++++++---------- 2 files changed, 26 insertions(+), 26 deletions(-) diff --git a/res/css/views/voip/_CallViewHeader.scss b/res/css/views/voip/_CallViewHeader.scss index 1ae3c6a273..014cfce478 100644 --- a/res/css/views/voip/_CallViewHeader.scss +++ b/res/css/views/voip/_CallViewHeader.scss @@ -14,7 +14,7 @@ See the License for the specific language governing permissions and limitations under the License. */ -.mx_CallView_header { +.mx_CallViewHeader { height: 44px; display: flex; flex-direction: row; @@ -24,13 +24,13 @@ limitations under the License. cursor: pointer; } -.mx_CallView_header_callType { +.mx_CallViewHeader_callType { font-size: 1.2rem; font-weight: bold; vertical-align: middle; } -.mx_CallView_header_secondaryCallInfo { +.mx_CallViewHeader_secondaryCallInfo { &::before { content: '·'; margin-left: 6px; @@ -38,11 +38,11 @@ limitations under the License. } } -.mx_CallView_header_controls { +.mx_CallViewHeader_controls { margin-left: auto; } -.mx_CallView_header_button { +.mx_CallViewHeader_button { display: inline-block; vertical-align: middle; cursor: pointer; @@ -60,24 +60,24 @@ limitations under the License. } } -.mx_CallView_header_button_fullscreen { +.mx_CallViewHeader_button_fullscreen { &::before { mask-image: url('$(res)/img/element-icons/call/fullscreen.svg'); } } -.mx_CallView_header_button_expand { +.mx_CallViewHeader_button_expand { &::before { mask-image: url('$(res)/img/element-icons/call/expand.svg'); } } -.mx_CallView_header_callInfo { +.mx_CallViewHeader_callInfo { margin-left: 12px; margin-right: 16px; } -.mx_CallView_header_roomName { +.mx_CallViewHeader_roomName { font-weight: bold; font-size: 12px; line-height: initial; @@ -88,7 +88,7 @@ limitations under the License. margin-left: 4px; } -.mx_CallView_header_callTypeSmall { +.mx_CallViewHeader_callTypeSmall { font-size: 12px; color: $secondary-fg-color; line-height: initial; @@ -99,7 +99,7 @@ limitations under the License. max-width: 240px; } -.mx_CallView_header_callTypeIcon { +.mx_CallViewHeader_callTypeIcon { display: inline-block; margin-right: 6px; height: 16px; @@ -119,11 +119,11 @@ limitations under the License. mask-position: center; } - &.mx_CallView_header_callTypeIcon_voice::before { + &.mx_CallViewHeader_callTypeIcon_voice::before { mask-image: url('$(res)/img/element-icons/call/voice-call.svg'); } - &.mx_CallView_header_callTypeIcon_video::before { + &.mx_CallViewHeader_callTypeIcon_video::before { mask-image: url('$(res)/img/element-icons/call/video-call.svg'); } } diff --git a/src/components/views/voip/CallView/CallViewHeader.tsx b/src/components/views/voip/CallView/CallViewHeader.tsx index 710a0c4372..8fc3720ece 100644 --- a/src/components/views/voip/CallView/CallViewHeader.tsx +++ b/src/components/views/voip/CallView/CallViewHeader.tsx @@ -61,20 +61,20 @@ type CallControlsProps = Pick & { roomId: string; }; const CallViewHeaderControls: React.FC = ({ pipMode = false, type, roomId }) => { - return
+ return
{ (pipMode && type === CallType.Video) && - } - { pipMode && onExpandClick(roomId)} title={_t("Return to call")} /> }
; }; const SecondaryCallInfo: React.FC<{ callRoom: Room }> = ({ callRoom }) => { - return + return onRoomAvatarClick(callRoom.roomId)}> @@ -86,9 +86,9 @@ const SecondaryCallInfo: React.FC<{ callRoom: Room }> = ({ callRoom }) => { const CallTypeIcon: React.FC<{ type: CallType }> = ({ type }) => { const classes = classNames({ - 'mx_CallView_header_callTypeIcon': true, - 'mx_CallView_header_callTypeIcon_video': type === CallType.Video, - 'mx_CallView_header_callTypeIcon_voice': type === CallType.Voice, + 'mx_CallViewHeader_callTypeIcon': true, + 'mx_CallViewHeader_callTypeIcon_video': type === CallType.Video, + 'mx_CallViewHeader_callTypeIcon_voice': type === CallType.Voice, }); return
; }; @@ -104,23 +104,23 @@ export const CallViewHeader: React.FC = ({ const callRoomName = callRoom.name; const { roomId } = callRoom; if (!pipMode) { - return
+ return
- { callTypeText } + { callTypeText }
; } return (
onRoomAvatarClick(roomId)}> ; -
-
{ callRoomName }
-
+
+
{ callRoomName }
+
{ callTypeText } { onHoldCallRoom && }
From cba87f433ba1d43c437b636bfb0300db45cc49f1 Mon Sep 17 00:00:00 2001 From: Dariusz Niemczyk Date: Fri, 6 Aug 2021 16:29:23 +0200 Subject: [PATCH 21/24] Fix merge conflicts --- src/components/views/voip/CallPreview.tsx | 3 +- src/components/views/voip/CallView.tsx | 2 +- .../views/voip/PictureInPictureDragger.tsx | 30 +++++++++++++++---- 3 files changed, 28 insertions(+), 7 deletions(-) diff --git a/src/components/views/voip/CallPreview.tsx b/src/components/views/voip/CallPreview.tsx index e68e9fd148..f958a8d6f6 100644 --- a/src/components/views/voip/CallPreview.tsx +++ b/src/components/views/voip/CallPreview.tsx @@ -184,11 +184,12 @@ export default class CallPreview extends React.Component { className="mx_CallPreview" draggable={pipMode} > - { (onMouseDownOnHeader) => } diff --git a/src/components/views/voip/CallView.tsx b/src/components/views/voip/CallView.tsx index 43bcfd348b..259cf065b6 100644 --- a/src/components/views/voip/CallView.tsx +++ b/src/components/views/voip/CallView.tsx @@ -55,7 +55,7 @@ interface IProps { // a callback which is called when the content in the CallView changes // in a way that is likely to cause a resize. - onResize?: any; + onResize?: () => void; // Whether this call view is for picture-in-picture mode // otherwise, it's the larger call view when viewing the room the call is in. diff --git a/src/components/views/voip/PictureInPictureDragger.tsx b/src/components/views/voip/PictureInPictureDragger.tsx index 244ecddee4..bf5a9ef61a 100644 --- a/src/components/views/voip/PictureInPictureDragger.tsx +++ b/src/components/views/voip/PictureInPictureDragger.tsx @@ -33,9 +33,14 @@ const PADDING = { right: 8, }; +interface IChildrenOptions { + onStartMoving: (event: React.MouseEvent) => void; + onResize: (event: React.MouseEvent) => void; +} + interface IProps { className?: string; - children: (startMovingEventHandler: (event: React.MouseEvent) => void) => React.ReactNode; + children: ({ onStartMoving, onResize }: IChildrenOptions) => React.ReactNode; draggable: boolean; } @@ -74,13 +79,13 @@ export class PictureInPictureDragger extends React.Component { public componentDidMount() { document.addEventListener("mousemove", this.onMoving); document.addEventListener("mouseup", this.onEndMoving); - window.addEventListener("resize", this.snap); + window.addEventListener("resize", this.onResize); } public componentWillUnmount() { document.removeEventListener("mousemove", this.onMoving); document.removeEventListener("mouseup", this.onEndMoving); - window.removeEventListener("resize", this.snap); + window.removeEventListener("resize", this.onResize); } private animationCallback = () => { @@ -126,7 +131,11 @@ export class PictureInPictureDragger extends React.Component { } } - private snap = () => { + private onResize = (): void => { + this.snap(false); + }; + + private snap = (animate = false) => { const translationX = this.desiredTranslationX; const translationY = this.desiredTranslationY; // We subtract the PiP size from the window size in order to calculate @@ -158,6 +167,17 @@ export class PictureInPictureDragger extends React.Component { // We start animating here because we want the PiP to move when we're // resizing the window this.scheduledUpdate.mark(); + + if (animate) { + // We start animating here because we want the PiP to move when we're + // resizing the window + this.scheduledUpdate.mark(); + } else { + this.setState({ + translationX: this.desiredTranslationX, + translationY: this.desiredTranslationY, + }); + } }; private onStartMoving = (event: React.MouseEvent | MouseEvent) => { @@ -181,7 +201,7 @@ export class PictureInPictureDragger extends React.Component { private onEndMoving = () => { this.moving = false; - this.snap(); + this.snap(true); }; public render() { From 7513f5ba7cac8a24db074c6079c454302d845a44 Mon Sep 17 00:00:00 2001 From: Dariusz Niemczyk Date: Fri, 6 Aug 2021 16:43:20 +0200 Subject: [PATCH 22/24] Fix types --- src/components/views/voip/CallPreview.tsx | 4 ++-- src/components/views/voip/CallView.tsx | 2 +- src/components/views/voip/PictureInPictureDragger.tsx | 7 +++++-- 3 files changed, 8 insertions(+), 5 deletions(-) diff --git a/src/components/views/voip/CallPreview.tsx b/src/components/views/voip/CallPreview.tsx index f958a8d6f6..37ac621116 100644 --- a/src/components/views/voip/CallPreview.tsx +++ b/src/components/views/voip/CallPreview.tsx @@ -184,8 +184,8 @@ export default class CallPreview extends React.Component { className="mx_CallPreview" draggable={pipMode} > - { (onMouseDownOnHeader, onResizeHandler) => void; + onResize?: (event: Event) => void; // Whether this call view is for picture-in-picture mode // otherwise, it's the larger call view when viewing the room the call is in. diff --git a/src/components/views/voip/PictureInPictureDragger.tsx b/src/components/views/voip/PictureInPictureDragger.tsx index bf5a9ef61a..e8617e09f2 100644 --- a/src/components/views/voip/PictureInPictureDragger.tsx +++ b/src/components/views/voip/PictureInPictureDragger.tsx @@ -35,7 +35,7 @@ const PADDING = { interface IChildrenOptions { onStartMoving: (event: React.MouseEvent) => void; - onResize: (event: React.MouseEvent) => void; + onResize: (event: Event) => void; } interface IProps { @@ -218,7 +218,10 @@ export class PictureInPictureDragger extends React.Component { ref={this.callViewWrapper} > <> - { this.props.children(this.onStartMoving) } + { this.props.children({ + onStartMoving: this.onStartMoving, + onResize: this.onResize, + }) }
); From ebee78acc8647c0f567e47cc0c352897b8dfb929 Mon Sep 17 00:00:00 2001 From: Dariusz Niemczyk Date: Fri, 6 Aug 2021 17:45:14 +0200 Subject: [PATCH 23/24] Fix dangling ';' --- src/components/views/voip/CallView/CallViewHeader.tsx | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/components/views/voip/CallView/CallViewHeader.tsx b/src/components/views/voip/CallView/CallViewHeader.tsx index 8fc3720ece..85e47adb78 100644 --- a/src/components/views/voip/CallView/CallViewHeader.tsx +++ b/src/components/views/voip/CallView/CallViewHeader.tsx @@ -116,7 +116,7 @@ export const CallViewHeader: React.FC = ({ onMouseDown={onPipMouseDown} > onRoomAvatarClick(roomId)}> - ; +
{ callRoomName }
From b73c73f07c6fa6aba6c5a4ab9e07770f3ae9489b Mon Sep 17 00:00:00 2001 From: Dariusz Niemczyk Date: Fri, 6 Aug 2021 17:47:59 +0200 Subject: [PATCH 24/24] Fix missing string --- src/components/views/voip/CallView/CallViewHeader.tsx | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/src/components/views/voip/CallView/CallViewHeader.tsx b/src/components/views/voip/CallView/CallViewHeader.tsx index 85e47adb78..acc577c5d9 100644 --- a/src/components/views/voip/CallView/CallViewHeader.tsx +++ b/src/components/views/voip/CallView/CallViewHeader.tsx @@ -24,9 +24,9 @@ import dis from '../../../../dispatcher/dispatcher'; import classNames from 'classnames'; import AccessibleTooltipButton from '../../elements/AccessibleTooltipButton'; -const callTypeTranslationByType: Record string> = { - [CallType.Video]: () => _t("Video Call"), - [CallType.Voice]: () => _t("Voice Call"), +const callTypeTranslationByType: Record = { + [CallType.Video]: _t("Video Call"), + [CallType.Voice]: _t("Voice Call"), }; interface CallViewHeaderProps { @@ -103,6 +103,7 @@ export const CallViewHeader: React.FC = ({ const callTypeText = callTypeTranslationByType[type]; const callRoomName = callRoom.name; const { roomId } = callRoom; + if (!pipMode) { return