4de0f7257a
This covers the "recents" section and rough design exclusively. It is known that the Field does nothing and that there's a bunch of missing functionality - this is to be iterated upon in future PRs. Labs flag is to aide development and should be removed in a very near future PR. Also, this is focusing on DMs and not user lists in general because I misinterpreted the scope. I'll fix this in a future PR and instead make this the best DM invite dialog it can be. Closes https://github.com/vector-im/riot-web/issues/11197
247 lines
9.6 KiB
JavaScript
247 lines
9.6 KiB
JavaScript
/*
|
|
Copyright 2016 OpenMarket Ltd
|
|
Copyright 2017, 2018 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 from 'react';
|
|
import MatrixClientPeg from './MatrixClientPeg';
|
|
import MultiInviter from './utils/MultiInviter';
|
|
import Modal from './Modal';
|
|
import { getAddressType } from './UserAddress';
|
|
import createRoom from './createRoom';
|
|
import sdk from './';
|
|
import dis from './dispatcher';
|
|
import DMRoomMap from './utils/DMRoomMap';
|
|
import { _t } from './languageHandler';
|
|
import SettingsStore from "./settings/SettingsStore";
|
|
|
|
/**
|
|
* Invites multiple addresses to a room
|
|
* Simpler interface to utils/MultiInviter but with
|
|
* no option to cancel.
|
|
*
|
|
* @param {string} roomId The ID of the room to invite to
|
|
* @param {string[]} addrs Array of strings of addresses to invite. May be matrix IDs or 3pids.
|
|
* @returns {Promise} Promise
|
|
*/
|
|
function inviteMultipleToRoom(roomId, addrs) {
|
|
const inviter = new MultiInviter(roomId);
|
|
return inviter.invite(addrs).then(states => Promise.resolve({states, inviter}));
|
|
}
|
|
|
|
export function showStartChatInviteDialog() {
|
|
if (SettingsStore.isFeatureEnabled("feature_ftue_dms")) {
|
|
const DMInviteDialog = sdk.getComponent("dialogs.DMInviteDialog");
|
|
Modal.createTrackedDialog('Start DM', '', DMInviteDialog, {
|
|
onFinished: (inviteIds) => {
|
|
// TODO: Replace _onStartDmFinished with less hacks
|
|
if (inviteIds.length > 0) _onStartDmFinished(true, inviteIds.map(i => ({address: i})));
|
|
// else ignore and just do nothing
|
|
},
|
|
}, /*className=*/null, /*isPriority=*/false, /*isStatic=*/true);
|
|
return;
|
|
}
|
|
|
|
const AddressPickerDialog = sdk.getComponent("dialogs.AddressPickerDialog");
|
|
|
|
Modal.createTrackedDialog('Start a chat', '', AddressPickerDialog, {
|
|
title: _t('Start a chat'),
|
|
description: _t("Who would you like to communicate with?"),
|
|
placeholder: (validAddressTypes) => {
|
|
// The set of valid address type can be mutated inside the dialog
|
|
// when you first have no IS but agree to use one in the dialog.
|
|
if (validAddressTypes.includes('email')) {
|
|
return _t("Email, name or Matrix ID");
|
|
}
|
|
return _t("Name or Matrix ID");
|
|
},
|
|
validAddressTypes: ['mx-user-id', 'email'],
|
|
button: _t("Start Chat"),
|
|
onFinished: _onStartDmFinished,
|
|
}, /*className=*/null, /*isPriority=*/false, /*isStatic=*/true);
|
|
}
|
|
|
|
export function showRoomInviteDialog(roomId) {
|
|
const AddressPickerDialog = sdk.getComponent("dialogs.AddressPickerDialog");
|
|
|
|
Modal.createTrackedDialog('Chat Invite', '', AddressPickerDialog, {
|
|
title: _t('Invite new room members'),
|
|
button: _t('Send Invites'),
|
|
placeholder: (validAddressTypes) => {
|
|
// The set of valid address type can be mutated inside the dialog
|
|
// when you first have no IS but agree to use one in the dialog.
|
|
if (validAddressTypes.includes('email')) {
|
|
return _t("Email, name or Matrix ID");
|
|
}
|
|
return _t("Name or Matrix ID");
|
|
},
|
|
validAddressTypes: ['mx-user-id', 'email'],
|
|
onFinished: (shouldInvite, addrs) => {
|
|
_onRoomInviteFinished(roomId, shouldInvite, addrs);
|
|
},
|
|
}, /*className=*/null, /*isPriority=*/false, /*isStatic=*/true);
|
|
}
|
|
|
|
/**
|
|
* Checks if the given MatrixEvent is a valid 3rd party user invite.
|
|
* @param {MatrixEvent} event The event to check
|
|
* @returns {boolean} True if valid, false otherwise
|
|
*/
|
|
export function isValid3pidInvite(event) {
|
|
if (!event || event.getType() !== "m.room.third_party_invite") return false;
|
|
|
|
// any events without these keys are not valid 3pid invites, so we ignore them
|
|
const requiredKeys = ['key_validity_url', 'public_key', 'display_name'];
|
|
for (let i = 0; i < requiredKeys.length; ++i) {
|
|
if (!event.getContent()[requiredKeys[i]]) return false;
|
|
}
|
|
|
|
// Valid enough by our standards
|
|
return true;
|
|
}
|
|
|
|
// TODO: Canonical DMs replaces this
|
|
function _onStartDmFinished(shouldInvite, addrs) {
|
|
if (!shouldInvite) return;
|
|
|
|
const addrTexts = addrs.map((addr) => addr.address);
|
|
|
|
if (_isDmChat(addrTexts)) {
|
|
const rooms = _getDirectMessageRooms(addrTexts[0]);
|
|
if (rooms.length > 0) {
|
|
// A Direct Message room already exists for this user, so reuse it
|
|
dis.dispatch({
|
|
action: 'view_room',
|
|
room_id: rooms[0],
|
|
should_peek: false,
|
|
joining: false,
|
|
});
|
|
} else {
|
|
// Start a new DM chat
|
|
createRoom({dmUserId: addrTexts[0]}).catch((err) => {
|
|
const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
|
|
Modal.createTrackedDialog('Failed to start chat', '', ErrorDialog, {
|
|
title: _t("Failed to start chat"),
|
|
description: ((err && err.message) ? err.message : _t("Operation failed")),
|
|
});
|
|
});
|
|
}
|
|
} else if (addrTexts.length === 1) {
|
|
// Start a new DM chat
|
|
createRoom({dmUserId: addrTexts[0]}).catch((err) => {
|
|
const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
|
|
Modal.createTrackedDialog('Failed to start chat', '', ErrorDialog, {
|
|
title: _t("Failed to start chat"),
|
|
description: ((err && err.message) ? err.message : _t("Operation failed")),
|
|
});
|
|
});
|
|
} else {
|
|
// Start multi user chat
|
|
let room;
|
|
createRoom().then((roomId) => {
|
|
room = MatrixClientPeg.get().getRoom(roomId);
|
|
return inviteMultipleToRoom(roomId, addrTexts);
|
|
}).then((result) => {
|
|
return _showAnyInviteErrors(result.states, room, result.inviter);
|
|
}).catch((err) => {
|
|
console.error(err.stack);
|
|
const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
|
|
Modal.createTrackedDialog('Failed to invite', '', ErrorDialog, {
|
|
title: _t("Failed to invite"),
|
|
description: ((err && err.message) ? err.message : _t("Operation failed")),
|
|
});
|
|
});
|
|
}
|
|
}
|
|
|
|
export function inviteUsersToRoom(roomId, userIds) {
|
|
return inviteMultipleToRoom(roomId, userIds).then((result) => {
|
|
const room = MatrixClientPeg.get().getRoom(roomId);
|
|
return _showAnyInviteErrors(result.states, room, result.inviter);
|
|
}).catch((err) => {
|
|
console.error(err.stack);
|
|
const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
|
|
Modal.createTrackedDialog('Failed to invite', '', ErrorDialog, {
|
|
title: _t("Failed to invite"),
|
|
description: ((err && err.message) ? err.message : _t("Operation failed")),
|
|
});
|
|
});
|
|
}
|
|
|
|
function _onRoomInviteFinished(roomId, shouldInvite, addrs) {
|
|
if (!shouldInvite) return;
|
|
|
|
const addrTexts = addrs.map((addr) => addr.address);
|
|
|
|
// Invite new users to a room
|
|
inviteUsersToRoom(roomId, addrTexts);
|
|
}
|
|
|
|
// TODO: Immutable DMs replaces this
|
|
function _isDmChat(addrTexts) {
|
|
if (addrTexts.length === 1 && getAddressType(addrTexts[0]) === 'mx-user-id') {
|
|
return true;
|
|
} else {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
function _showAnyInviteErrors(addrs, room, inviter) {
|
|
// Show user any errors
|
|
const failedUsers = Object.keys(addrs).filter(a => addrs[a] === 'error');
|
|
if (failedUsers.length === 1 && inviter.fatal) {
|
|
// Just get the first message because there was a fatal problem on the first
|
|
// user. This usually means that no other users were attempted, making it
|
|
// pointless for us to list who failed exactly.
|
|
const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
|
|
Modal.createTrackedDialog('Failed to invite users to the room', '', ErrorDialog, {
|
|
title: _t("Failed to invite users to the room:", {roomName: room.name}),
|
|
description: inviter.getErrorText(failedUsers[0]),
|
|
});
|
|
} else {
|
|
const errorList = [];
|
|
for (const addr of failedUsers) {
|
|
if (addrs[addr] === "error") {
|
|
const reason = inviter.getErrorText(addr);
|
|
errorList.push(addr + ": " + reason);
|
|
}
|
|
}
|
|
|
|
if (errorList.length > 0) {
|
|
// React 16 doesn't let us use `errorList.join(<br />)` anymore, so this is our solution
|
|
const description = <div>{errorList.map(e => <div key={e}>{e}</div>)}</div>;
|
|
|
|
const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
|
|
Modal.createTrackedDialog('Failed to invite the following users to the room', '', ErrorDialog, {
|
|
title: _t("Failed to invite the following users to the %(roomName)s room:", {roomName: room.name}),
|
|
description,
|
|
});
|
|
}
|
|
}
|
|
|
|
return addrs;
|
|
}
|
|
|
|
function _getDirectMessageRooms(addr) {
|
|
const dmRoomMap = new DMRoomMap(MatrixClientPeg.get());
|
|
const dmRooms = dmRoomMap.getDMRoomsForUserId(addr);
|
|
const rooms = dmRooms.filter((dmRoom) => {
|
|
const room = MatrixClientPeg.get().getRoom(dmRoom);
|
|
if (room) {
|
|
return room.getMyMembership() === 'join';
|
|
}
|
|
});
|
|
return rooms;
|
|
}
|