diff --git a/res/css/_components.scss b/res/css/_components.scss
index 4c253c7d9a..261b35690e 100644
--- a/res/css/_components.scss
+++ b/res/css/_components.scss
@@ -141,6 +141,7 @@
@import "./views/messages/_MEmoteBody.scss";
@import "./views/messages/_MFileBody.scss";
@import "./views/messages/_MImageBody.scss";
+@import "./views/messages/_MJitsiWidgetEvent.scss";
@import "./views/messages/_MNoticeBody.scss";
@import "./views/messages/_MStickerBody.scss";
@import "./views/messages/_MTextBody.scss";
diff --git a/res/css/views/messages/_MJitsiWidgetEvent.scss b/res/css/views/messages/_MJitsiWidgetEvent.scss
new file mode 100644
index 0000000000..3e51e89744
--- /dev/null
+++ b/res/css/views/messages/_MJitsiWidgetEvent.scss
@@ -0,0 +1,55 @@
+/*
+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_MJitsiWidgetEvent {
+ display: grid;
+ grid-template-columns: 24px minmax(0, 1fr) min-content;
+
+ &::before {
+ grid-column: 1;
+ grid-row: 1 / 3;
+ width: 16px;
+ height: 16px;
+ content: "";
+ top: 0;
+ bottom: 0;
+ left: 0;
+ right: 0;
+ mask-repeat: no-repeat;
+ mask-position: center;
+ mask-size: contain;
+ background-color: $composer-e2e-icon-color; // XXX: Variable abuse
+ margin-top: 4px;
+ mask-image: url('$(res)/img/element-icons/call/video-call.svg');
+ }
+
+ .mx_MJitsiWidgetEvent_title {
+ font-weight: 600;
+ font-size: $font-15px;
+ grid-column: 2;
+ grid-row: 1;
+ }
+
+ .mx_MJitsiWidgetEvent_subtitle {
+ grid-column: 2;
+ grid-row: 2;
+ }
+
+ .mx_MJitsiWidgetEvent_title,
+ .mx_MJitsiWidgetEvent_subtitle {
+ overflow-wrap: break-word;
+ }
+}
diff --git a/res/css/views/rooms/_AppsDrawer.scss b/res/css/views/rooms/_AppsDrawer.scss
index fee3d61153..244e88ca3e 100644
--- a/res/css/views/rooms/_AppsDrawer.scss
+++ b/res/css/views/rooms/_AppsDrawer.scss
@@ -15,7 +15,7 @@ See the License for the specific language governing permissions and
limitations under the License.
*/
-$MiniAppTileHeight: 114px;
+$MiniAppTileHeight: 200px;
.mx_AppsDrawer {
margin: 5px 5px 5px 18px;
@@ -220,9 +220,10 @@ $MiniAppTileHeight: 114px;
}
.mx_AppTileBody_mini {
- height: 112px;
+ height: $MiniAppTileHeight;
width: 100%;
overflow: hidden;
+ border-radius: 8px;
}
.mx_AppTile .mx_AppTileBody,
diff --git a/res/css/views/rooms/_MessageComposer.scss b/res/css/views/rooms/_MessageComposer.scss
index a403a8dc4c..71c0db947e 100644
--- a/res/css/views/rooms/_MessageComposer.scss
+++ b/res/css/views/rooms/_MessageComposer.scss
@@ -217,7 +217,7 @@ limitations under the License.
}
}
- &.mx_MessageComposer_hangup::before {
+ &.mx_MessageComposer_hangup:not(.mx_AccessibleButton_disabled)::before {
background-color: $warning-color;
}
}
diff --git a/res/css/views/voip/_CallContainer.scss b/res/css/views/voip/_CallContainer.scss
index 4d26d8a312..650302b7e1 100644
--- a/res/css/views/voip/_CallContainer.scss
+++ b/res/css/views/voip/_CallContainer.scss
@@ -23,9 +23,16 @@ limitations under the License.
z-index: 100;
box-shadow: 0px 14px 24px rgba(0, 0, 0, 0.08);
- cursor: pointer;
+ // Disable pointer events for Jitsi widgets to function. Direct
+ // calls have their own cursor and behaviour, but we need to make
+ // sure the cursor hits the iframe for Jitsi which will be at a
+ // different level.
+ pointer-events: none;
.mx_CallPreview {
+ pointer-events: initial; // restore pointer events so the user can leave/interact
+ cursor: pointer;
+
.mx_VideoView {
width: 350px;
}
@@ -37,7 +44,7 @@ limitations under the License.
}
.mx_AppTile_persistedWrapper div {
- min-width: 300px;
+ min-width: 350px;
}
.mx_IncomingCallBox {
@@ -45,6 +52,9 @@ limitations under the License.
background-color: $primary-bg-color;
padding: 8px;
+ pointer-events: initial; // restore pointer events so the user can accept/decline
+ cursor: pointer;
+
.mx_IncomingCallBox_CallerInfo {
display: flex;
direction: row;
diff --git a/src/CallHandler.tsx b/src/CallHandler.tsx
index 62b91f938b..2ff018d4d6 100644
--- a/src/CallHandler.tsx
+++ b/src/CallHandler.tsx
@@ -74,6 +74,8 @@ import {base32} from "rfc4648";
import QuestionDialog from "./components/views/dialogs/QuestionDialog";
import ErrorDialog from "./components/views/dialogs/ErrorDialog";
+import WidgetStore from "./stores/WidgetStore";
+import ActiveWidgetStore from "./stores/ActiveWidgetStore";
// until we ts-ify the js-sdk voip code
type Call = any;
@@ -351,6 +353,14 @@ export default class CallHandler {
console.info("Place conference call in %s", payload.room_id);
this.startCallApp(payload.room_id, payload.type);
break;
+ case 'end_conference':
+ console.info("Terminating conference call in %s", payload.room_id);
+ this.terminateCallApp(payload.room_id);
+ break;
+ case 'hangup_conference':
+ console.info("Leaving conference call in %s", payload.room_id);
+ this.hangupCallApp(payload.room_id);
+ break;
case 'incoming_call':
{
if (this.getAnyActiveCall()) {
@@ -398,10 +408,12 @@ export default class CallHandler {
show: true,
});
+ // prevent double clicking the call button
const room = MatrixClientPeg.get().getRoom(roomId);
const currentJitsiWidgets = WidgetUtils.getRoomWidgetsOfType(room, WidgetType.JITSI);
-
- if (WidgetEchoStore.roomHasPendingWidgetsOfType(roomId, currentJitsiWidgets, WidgetType.JITSI)) {
+ const hasJitsi = currentJitsiWidgets.length > 0
+ || WidgetEchoStore.roomHasPendingWidgetsOfType(roomId, currentJitsiWidgets, WidgetType.JITSI);
+ if (hasJitsi) {
Modal.createTrackedDialog('Call already in progress', '', ErrorDialog, {
title: _t('Call in Progress'),
description: _t('A call is currently being placed!'),
@@ -409,33 +421,6 @@ export default class CallHandler {
return;
}
- if (currentJitsiWidgets.length > 0) {
- console.warn(
- "Refusing to start conference call widget in " + roomId +
- " a conference call widget is already present",
- );
-
- if (WidgetUtils.canUserModifyWidgets(roomId)) {
- Modal.createTrackedDialog('Already have Jitsi Widget', '', QuestionDialog, {
- title: _t('End Call'),
- description: _t('Remove the group call from the room?'),
- button: _t('End Call'),
- cancelButton: _t('Cancel'),
- onFinished: (endCall) => {
- if (endCall) {
- WidgetUtils.setRoomWidget(roomId, currentJitsiWidgets[0].getContent()['id']);
- }
- },
- });
- } else {
- Modal.createTrackedDialog('Already have Jitsi Widget', '', ErrorDialog, {
- title: _t('Call in Progress'),
- description: _t("You don't have permission to remove the call from the room"),
- });
- }
- return;
- }
-
const jitsiDomain = Jitsi.getInstance().preferredDomain;
const jitsiAuth = await Jitsi.getInstance().getJitsiAuth();
let confId;
@@ -484,4 +469,38 @@ export default class CallHandler {
console.error(e);
});
}
+
+ private terminateCallApp(roomId: string) {
+ Modal.createTrackedDialog('Confirm Jitsi Terminate', '', QuestionDialog, {
+ hasCancelButton: true,
+ title: _t("End conference"),
+ description: _t("This will end the conference for everyone. Continue?"),
+ button: _t("End conference"),
+ onFinished: (proceed) => {
+ if (!proceed) return;
+
+ // We'll just obliterate them all. There should only ever be one, but might as well
+ // be safe.
+ const roomInfo = WidgetStore.instance.getRoom(roomId);
+ const jitsiWidgets = roomInfo.widgets.filter(w => WidgetType.JITSI.matches(w.type));
+ jitsiWidgets.forEach(w => {
+ // setting invalid content removes it
+ WidgetUtils.setRoomWidget(roomId, w.id);
+ });
+ },
+ });
+ }
+
+ private hangupCallApp(roomId: string) {
+ const roomInfo = WidgetStore.instance.getRoom(roomId);
+ if (!roomInfo) return; // "should never happen" clauses go here
+
+ const jitsiWidgets = roomInfo.widgets.filter(w => WidgetType.JITSI.matches(w.type));
+ jitsiWidgets.forEach(w => {
+ const messaging = ActiveWidgetStore.getWidgetMessaging(w.id);
+ if (!messaging) return; // more "should never happen" words
+
+ messaging.hangup();
+ });
+ }
}
diff --git a/src/TextForEvent.js b/src/TextForEvent.js
index f9cda23650..34d40bf1fd 100644
--- a/src/TextForEvent.js
+++ b/src/TextForEvent.js
@@ -18,7 +18,6 @@ import { _t } from './languageHandler';
import * as Roles from './Roles';
import {isValid3pidInvite} from "./RoomInvite";
import SettingsStore from "./settings/SettingsStore";
-import {WidgetType} from "./widgets/WidgetType";
import {ALL_RULE_TYPES, ROOM_RULE_TYPES, SERVER_RULE_TYPES, USER_RULE_TYPES} from "./mjolnir/BanList";
function textForMemberEvent(ev) {
@@ -464,10 +463,6 @@ function textForWidgetEvent(event) {
const {name: prevName, type: prevType, url: prevUrl} = event.getPrevContent();
const {name, type, url} = event.getContent() || {};
- if (WidgetType.JITSI.matches(type) || WidgetType.JITSI.matches(prevType)) {
- return textForJitsiWidgetEvent(event, senderName, url, prevUrl);
- }
-
let widgetName = name || prevName || type || prevType || '';
// Apply sentence case to widget name
if (widgetName && widgetName.length > 0) {
@@ -493,24 +488,6 @@ function textForWidgetEvent(event) {
}
}
-function textForJitsiWidgetEvent(event, senderName, url, prevUrl) {
- if (url) {
- if (prevUrl) {
- return _t('Group call modified by %(senderName)s', {
- senderName,
- });
- } else {
- return _t('Group call started by %(senderName)s', {
- senderName,
- });
- }
- } else {
- return _t('Group call ended by %(senderName)s', {
- senderName,
- });
- }
-}
-
function textForMjolnirEvent(event) {
const senderName = event.getSender();
const {entity: prevEntity} = event.getPrevContent();
diff --git a/src/WidgetMessaging.js b/src/WidgetMessaging.js
index c68e926ac1..9394abf025 100644
--- a/src/WidgetMessaging.js
+++ b/src/WidgetMessaging.js
@@ -107,6 +107,17 @@ export default class WidgetMessaging {
});
}
+ /**
+ * Tells the widget to hang up on its call.
+ * @returns {Promise<*>} Resolves when the widget has acknowledged the message.
+ */
+ hangup() {
+ return this.messageToWidget({
+ api: OUTBOUND_API_NAME,
+ action: KnownWidgetActions.Hangup,
+ });
+ }
+
/**
* Request a screenshot from a widget
* @return {Promise} To be resolved with screenshot data when it has been generated
diff --git a/src/components/views/dialogs/SetPasswordDialog.js b/src/components/views/dialogs/SetPasswordDialog.js
index 3649190ac9..f2d5a96b4c 100644
--- a/src/components/views/dialogs/SetPasswordDialog.js
+++ b/src/components/views/dialogs/SetPasswordDialog.js
@@ -117,7 +117,9 @@ export default class SetPasswordDialog extends React.Component {
autoFocusNewPasswordInput={true}
shouldAskForEmail={true}
onError={this._onPasswordChangeError}
- onFinished={this._onPasswordChanged} />
+ onFinished={this._onPasswordChanged}
+ buttonLabel={_t("Set Password")}
+ />
{ this.state.error }
diff --git a/src/components/views/elements/PersistentApp.js b/src/components/views/elements/PersistentApp.js
index 686739a9f7..a3e413151a 100644
--- a/src/components/views/elements/PersistentApp.js
+++ b/src/components/views/elements/PersistentApp.js
@@ -82,6 +82,7 @@ export default class PersistentApp extends React.Component {
showDelete={false}
showMinimise={false}
miniMode={true}
+ showMenubar={false}
/>;
}
}
diff --git a/src/components/views/messages/MJitsiWidgetEvent.tsx b/src/components/views/messages/MJitsiWidgetEvent.tsx
new file mode 100644
index 0000000000..3d191209f9
--- /dev/null
+++ b/src/components/views/messages/MJitsiWidgetEvent.tsx
@@ -0,0 +1,76 @@
+/*
+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 React from 'react';
+import { MatrixEvent } from "matrix-js-sdk/src/models/event";
+import { _t } from "../../../languageHandler";
+import WidgetStore from "../../../stores/WidgetStore";
+
+interface IProps {
+ mxEvent: MatrixEvent;
+}
+
+export default class MJitsiWidgetEvent extends React.PureComponent {
+ constructor(props) {
+ super(props);
+ }
+
+ render() {
+ const url = this.props.mxEvent.getContent()['url'];
+ const prevUrl = this.props.mxEvent.getPrevContent()['url'];
+ const senderName = this.props.mxEvent.sender?.name || this.props.mxEvent.getSender();
+
+ let joinCopy = _t('Join the conference at the top of this room');
+ if (!WidgetStore.instance.isPinned(this.props.mxEvent.getStateKey())) {
+ joinCopy = _t('Join the conference from the room information card on the right');
+ }
+
+ if (!url) {
+ // removed
+ return (
+
+
+ {_t('Video conference ended by %(senderName)s', {senderName})}
+
+
+ );
+ } else if (prevUrl) {
+ // modified
+ return (
+
+
+ {_t('Video conference updated by %(senderName)s', {senderName})}
+
+
+ {joinCopy}
+
+
+ );
+ } else {
+ // assume added
+ return (
+
+
+ {_t("Video conference started by %(senderName)s", {senderName})}
+
+
+ {joinCopy}
+
+
+ );
+ }
+ }
+}
diff --git a/src/components/views/rooms/BasicMessageComposer.tsx b/src/components/views/rooms/BasicMessageComposer.tsx
index 7c2eb83a94..d9b34b93ef 100644
--- a/src/components/views/rooms/BasicMessageComposer.tsx
+++ b/src/components/views/rooms/BasicMessageComposer.tsx
@@ -619,13 +619,14 @@ export default class BasicMessageEditor extends React.Component
}
private onFormatAction = (action: Formatting) => {
- const range = getRangeForSelection(
- this.editorRef.current,
- this.props.model,
- document.getSelection());
+ const range = getRangeForSelection(this.editorRef.current, this.props.model, document.getSelection());
+ // trim the range as we want it to exclude leading/trailing spaces
+ range.trim();
+
if (range.length === 0) {
return;
}
+
this.historyManager.ensureLastChangesPushed(this.props.model);
this.modifiedFlag = true;
switch (action) {
diff --git a/src/components/views/rooms/EventTile.js b/src/components/views/rooms/EventTile.js
index a1cc681a4c..81034cf07b 100644
--- a/src/components/views/rooms/EventTile.js
+++ b/src/components/views/rooms/EventTile.js
@@ -34,6 +34,7 @@ import * as ObjectUtils from "../../../ObjectUtils";
import MatrixClientContext from "../../../contexts/MatrixClientContext";
import {E2E_STATE} from "./E2EIcon";
import {toRem} from "../../../utils/units";
+import {WidgetType} from "../../../widgets/WidgetType";
import RoomAvatar from "../avatars/RoomAvatar";
const eventTileTypes = {
@@ -111,6 +112,19 @@ export function getHandlerTile(ev) {
}
}
+ // TODO: Enable support for m.widget event type (https://github.com/vector-im/element-web/issues/13111)
+ if (type === "im.vector.modular.widgets") {
+ let type = ev.getContent()['type'];
+ if (!type) {
+ // deleted/invalid widget - try the past widget type
+ type = ev.getPrevContent()['type'];
+ }
+
+ if (WidgetType.JITSI.matches(type)) {
+ return "messages.MJitsiWidgetEvent";
+ }
+ }
+
return ev.isState() ? stateEventTileTypes[type] : eventTileTypes[type];
}
@@ -627,16 +641,18 @@ export default class EventTile extends React.Component {
const msgtype = content.msgtype;
const eventType = this.props.mxEvent.getType();
+ let tileHandler = getHandlerTile(this.props.mxEvent);
+
// Info messages are basically information about commands processed on a room
const isBubbleMessage = eventType.startsWith("m.key.verification") ||
(eventType === "m.room.message" && msgtype && msgtype.startsWith("m.key.verification")) ||
- (eventType === "m.room.encryption");
+ (eventType === "m.room.encryption") ||
+ (tileHandler === "messages.MJitsiWidgetEvent");
let isInfoMessage = (
!isBubbleMessage && eventType !== 'm.room.message' &&
eventType !== 'm.sticker' && eventType !== 'm.room.create'
);
- let tileHandler = getHandlerTile(this.props.mxEvent);
// If we're showing hidden events in the timeline, we should use the
// source tile when there's no regular tile for an event and also for
// replace relations (which otherwise would display as a confusing
diff --git a/src/components/views/rooms/MessageComposer.js b/src/components/views/rooms/MessageComposer.js
index e6cd686e3c..71999fb04f 100644
--- a/src/components/views/rooms/MessageComposer.js
+++ b/src/components/views/rooms/MessageComposer.js
@@ -1,6 +1,7 @@
/*
Copyright 2015, 2016 OpenMarket Ltd
Copyright 2017, 2018 New Vector Ltd
+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.
@@ -32,6 +33,10 @@ import {aboveLeftOf, ContextMenu, ContextMenuTooltipButton, useContextMenu} from
import AccessibleTooltipButton from "../elements/AccessibleTooltipButton";
import ReplyPreview from "./ReplyPreview";
import {UIFeature} from "../../../settings/UIFeature";
+import WidgetStore from "../../../stores/WidgetStore";
+import WidgetUtils from "../../../utils/WidgetUtils";
+import {UPDATE_EVENT} from "../../../stores/AsyncStore";
+import ActiveWidgetStore from "../../../stores/ActiveWidgetStore";
function ComposerAvatar(props) {
const MemberStatusMessageAvatar = sdk.getComponent('avatars.MemberStatusMessageAvatar');
@@ -85,8 +90,15 @@ VideoCallButton.propTypes = {
};
function HangupButton(props) {
- const AccessibleButton = sdk.getComponent('elements.AccessibleButton');
const onHangupClick = () => {
+ if (props.isConference) {
+ dis.dispatch({
+ action: props.canEndConference ? 'end_conference' : 'hangup_conference',
+ room_id: props.roomId,
+ });
+ return;
+ }
+
const call = CallHandler.sharedInstance().getCallForRoom(props.roomId);
if (!call) {
return;
@@ -98,14 +110,28 @@ function HangupButton(props) {
room_id: call.roomId,
});
};
- return ();
+ title={tooltip}
+ disabled={!canLeaveConference}
+ />
+ );
}
HangupButton.propTypes = {
roomId: PropTypes.string.isRequired,
+ isConference: PropTypes.bool.isRequired,
+ canEndConference: PropTypes.bool,
+ isInConference: PropTypes.bool,
};
const EmojiButton = ({addEmoji}) => {
@@ -226,12 +252,17 @@ export default class MessageComposer extends React.Component {
this._onRoomViewStoreUpdate = this._onRoomViewStoreUpdate.bind(this);
this._onTombstoneClick = this._onTombstoneClick.bind(this);
this.renderPlaceholderText = this.renderPlaceholderText.bind(this);
+ WidgetStore.instance.on(UPDATE_EVENT, this._onWidgetUpdate);
+ ActiveWidgetStore.on('update', this._onActiveWidgetUpdate);
this._dispatcherRef = null;
+
this.state = {
isQuoting: Boolean(RoomViewStore.getQuotingEvent()),
tombstone: this._getRoomTombstone(),
canSendMessages: this.props.room.maySendMessage(),
showCallButtons: SettingsStore.getValue("showCallButtonsInComposer"),
+ hasConference: WidgetStore.instance.doesRoomHaveConference(this.props.room),
+ joinedConference: WidgetStore.instance.isJoinedToConferenceIn(this.props.room),
};
}
@@ -247,6 +278,14 @@ export default class MessageComposer extends React.Component {
}
};
+ _onWidgetUpdate = () => {
+ this.setState({hasConference: WidgetStore.instance.doesRoomHaveConference(this.props.room)});
+ };
+
+ _onActiveWidgetUpdate = () => {
+ this.setState({joinedConference: WidgetStore.instance.isJoinedToConferenceIn(this.props.room)});
+ };
+
componentDidMount() {
this.dispatcherRef = dis.register(this.onAction);
MatrixClientPeg.get().on("RoomState.events", this._onRoomStateEvents);
@@ -277,6 +316,8 @@ export default class MessageComposer extends React.Component {
if (this._roomStoreToken) {
this._roomStoreToken.remove();
}
+ WidgetStore.instance.removeListener(UPDATE_EVENT, this._onWidgetUpdate);
+ ActiveWidgetStore.removeListener('update', this._onActiveWidgetUpdate);
dis.unregister(this.dispatcherRef);
}
@@ -392,9 +433,19 @@ export default class MessageComposer extends React.Component {
}
if (this.state.showCallButtons) {
- if (callInProgress) {
+ if (this.state.hasConference) {
+ const canEndConf = WidgetUtils.canUserModifyWidgets(this.props.room.roomId);
controls.push(
- ,
+ ,
+ );
+ } else if (callInProgress) {
+ controls.push(
+ ,
);
} else {
controls.push(
diff --git a/src/components/views/settings/ChangePassword.js b/src/components/views/settings/ChangePassword.js
index 0b62f1fa81..8ae000f087 100644
--- a/src/components/views/settings/ChangePassword.js
+++ b/src/components/views/settings/ChangePassword.js
@@ -35,6 +35,7 @@ export default class ChangePassword extends React.Component {
rowClassName: PropTypes.string,
buttonClassName: PropTypes.string,
buttonKind: PropTypes.string,
+ buttonLabel: PropTypes.string,
confirm: PropTypes.bool,
// Whether to autoFocus the new password input
autoFocusNewPasswordInput: PropTypes.bool,
@@ -271,7 +272,7 @@ export default class ChangePassword extends React.Component {
/>
- { _t('Change Password') }
+ { this.props.buttonLabel || _t('Change Password') }
);
diff --git a/src/editor/range.ts b/src/editor/range.ts
index 27f59f34a9..838dfd8b98 100644
--- a/src/editor/range.ts
+++ b/src/editor/range.ts
@@ -18,6 +18,10 @@ import EditorModel from "./model";
import DocumentPosition, {Predicate} from "./position";
import {Part} from "./parts";
+const whitespacePredicate: Predicate = (index, offset, part) => {
+ return part.text[offset].trim() === "";
+};
+
export default class Range {
private _start: DocumentPosition;
private _end: DocumentPosition;
@@ -35,6 +39,11 @@ export default class Range {
});
}
+ trim() {
+ this._start = this._start.forwardsWhile(this.model, whitespacePredicate);
+ this._end = this._end.backwardsWhile(this.model, whitespacePredicate);
+ }
+
expandBackwardsWhile(predicate: Predicate) {
this._start = this._start.backwardsWhile(this.model, predicate);
}
diff --git a/src/i18n/strings/en_EN.json b/src/i18n/strings/en_EN.json
index 65374ea3ec..a377663570 100644
--- a/src/i18n/strings/en_EN.json
+++ b/src/i18n/strings/en_EN.json
@@ -50,12 +50,10 @@
"You cannot place a call with yourself.": "You cannot place a call with yourself.",
"Call in Progress": "Call in Progress",
"A call is currently being placed!": "A call is currently being placed!",
- "End Call": "End Call",
- "Remove the group call from the room?": "Remove the group call from the room?",
- "Cancel": "Cancel",
- "You don't have permission to remove the call from the room": "You don't have permission to remove the call from the room",
"Permission Required": "Permission Required",
"You do not have permission to start a conference call in this room": "You do not have permission to start a conference call in this room",
+ "End conference": "End conference",
+ "This will end the conference for everyone. Continue?": "This will end the conference for everyone. Continue?",
"Replying With Files": "Replying With Files",
"At this time it is not possible to reply with a file. Would you like to upload this file without replying?": "At this time it is not possible to reply with a file. Would you like to upload this file without replying?",
"Continue": "Continue",
@@ -143,6 +141,7 @@
"Cancel entering passphrase?": "Cancel entering passphrase?",
"Are you sure you want to cancel entering passphrase?": "Are you sure you want to cancel entering passphrase?",
"Go Back": "Go Back",
+ "Cancel": "Cancel",
"Setting up keys": "Setting up keys",
"Messages": "Messages",
"Actions": "Actions",
@@ -277,9 +276,6 @@
"%(widgetName)s widget modified by %(senderName)s": "%(widgetName)s widget modified by %(senderName)s",
"%(widgetName)s widget added by %(senderName)s": "%(widgetName)s widget added by %(senderName)s",
"%(widgetName)s widget removed by %(senderName)s": "%(widgetName)s widget removed by %(senderName)s",
- "Group call modified by %(senderName)s": "Group call modified by %(senderName)s",
- "Group call started by %(senderName)s": "Group call started by %(senderName)s",
- "Group call ended by %(senderName)s": "Group call ended by %(senderName)s",
"%(senderName)s removed the rule banning users matching %(glob)s": "%(senderName)s removed the rule banning users matching %(glob)s",
"%(senderName)s removed the rule banning rooms matching %(glob)s": "%(senderName)s removed the rule banning rooms matching %(glob)s",
"%(senderName)s removed the rule banning servers matching %(glob)s": "%(senderName)s removed the rule banning servers matching %(glob)s",
@@ -1393,6 +1389,11 @@
"Invalid file%(extra)s": "Invalid file%(extra)s",
"Error decrypting image": "Error decrypting image",
"Show image": "Show image",
+ "Join the conference at the top of this room": "Join the conference at the top of this room",
+ "Join the conference from the room information card on the right": "Join the conference from the room information card on the right",
+ "Video conference ended by %(senderName)s": "Video conference ended by %(senderName)s",
+ "Video conference updated by %(senderName)s": "Video conference updated by %(senderName)s",
+ "Video conference started by %(senderName)s": "Video conference started by %(senderName)s",
"You have ignored this user, so their message is hidden. Show anyways.": "You have ignored this user, so their message is hidden. Show anyways.",
"You verified %(name)s": "You verified %(name)s",
"You cancelled verifying %(name)s": "You cancelled verifying %(name)s",
diff --git a/src/stores/WidgetStore.ts b/src/stores/WidgetStore.ts
index 10327ce4e9..996c169e94 100644
--- a/src/stores/WidgetStore.ts
+++ b/src/stores/WidgetStore.ts
@@ -22,6 +22,7 @@ import { AsyncStoreWithClient } from "./AsyncStoreWithClient";
import defaultDispatcher from "../dispatcher/dispatcher";
import SettingsStore from "../settings/SettingsStore";
import WidgetEchoStore from "../stores/WidgetEchoStore";
+import ActiveWidgetStore from "../stores/ActiveWidgetStore";
import WidgetUtils from "../utils/WidgetUtils";
import {SettingLevel} from "../settings/SettingLevel";
import {WidgetType} from "../widgets/WidgetType";
@@ -158,7 +159,8 @@ export default class WidgetStore extends AsyncStoreWithClient {
let pinned = roomInfo && roomInfo.pinned[widgetId];
// Jitsi widgets should be pinned by default
- if (pinned === undefined && WidgetType.JITSI.matches(this.widgetMap.get(widgetId).type)) pinned = true;
+ const widget = this.widgetMap.get(widgetId);
+ if (pinned === undefined && WidgetType.JITSI.matches(widget?.type)) pinned = true;
return pinned;
}
@@ -206,6 +208,24 @@ export default class WidgetStore extends AsyncStoreWithClient {
}
return roomInfo.widgets;
}
+
+ public doesRoomHaveConference(room: Room): boolean {
+ const roomInfo = this.getRoom(room.roomId);
+ if (!roomInfo) return false;
+
+ const currentWidgets = roomInfo.widgets.filter(w => WidgetType.JITSI.matches(w.type));
+ const hasPendingWidgets = WidgetEchoStore.roomHasPendingWidgetsOfType(room.roomId, [], WidgetType.JITSI);
+ return currentWidgets.length > 0 || hasPendingWidgets;
+ }
+
+ public isJoinedToConferenceIn(room: Room): boolean {
+ const roomInfo = this.getRoom(room.roomId);
+ if (!roomInfo) return false;
+
+ // A persistent conference widget indicates that we're participating
+ const widgets = roomInfo.widgets.filter(w => WidgetType.JITSI.matches(w.type));
+ return widgets.some(w => ActiveWidgetStore.getWidgetPersistence(w.id));
+ }
}
window.mxWidgetStore = WidgetStore.instance;
diff --git a/src/widgets/WidgetApi.ts b/src/widgets/WidgetApi.ts
index 672cbf2a56..c25d607948 100644
--- a/src/widgets/WidgetApi.ts
+++ b/src/widgets/WidgetApi.ts
@@ -39,6 +39,7 @@ export enum KnownWidgetActions {
SetAlwaysOnScreen = "set_always_on_screen",
ClientReady = "im.vector.ready",
Terminate = "im.vector.terminate",
+ Hangup = "im.vector.hangup",
}
export type WidgetAction = KnownWidgetActions | string;
@@ -119,13 +120,15 @@ export class WidgetApi extends EventEmitter {
// Automatically acknowledge so we can move on
this.replyToRequest(payload, {});
- } else if (payload.action === KnownWidgetActions.Terminate) {
+ } else if (payload.action === KnownWidgetActions.Terminate
+ || payload.action === KnownWidgetActions.Hangup) {
// Finalization needs to be async, so postpone with a promise
let finalizePromise = Promise.resolve();
const wait = (promise) => {
finalizePromise = finalizePromise.then(() => promise);
};
- this.emit('terminate', wait);
+ const emitName = payload.action === KnownWidgetActions.Terminate ? 'terminate' : 'hangup';
+ this.emit(emitName, wait);
Promise.resolve(finalizePromise).then(() => {
// Acknowledge that we're shut down now
this.replyToRequest(payload, {});
diff --git a/test/editor/range-test.js b/test/editor/range-test.js
index b69ed9eb53..60055af824 100644
--- a/test/editor/range-test.js
+++ b/test/editor/range-test.js
@@ -88,4 +88,19 @@ describe('editor/range', function() {
expect(model.parts[1].text).toBe("man");
expect(model.parts.length).toBe(2);
});
+ it('range trim spaces off both ends', () => {
+ const renderer = createRenderer();
+ const pc = createPartCreator();
+ const model = new EditorModel([
+ pc.plain("abc abc abc"),
+ ], pc, renderer);
+ const range = model.startRange(
+ model.positionForOffset(3, false), // at end of first `abc`
+ model.positionForOffset(8, false), // at start of last `abc`
+ );
+
+ expect(range.parts[0].text).toBe(" abc ");
+ range.trim();
+ expect(range.parts[0].text).toBe("abc");
+ });
});