210 lines
		
	
	
		
			8.3 KiB
		
	
	
	
		
			TypeScript
		
	
	
			
		
		
	
	
			210 lines
		
	
	
		
			8.3 KiB
		
	
	
	
		
			TypeScript
		
	
	
| /*
 | |
| Copyright 2016 - 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.
 | |
| */
 | |
| 
 | |
| import React from "react";
 | |
| import { Room } from "matrix-js-sdk/src/models/room";
 | |
| import { MatrixEvent } from "matrix-js-sdk/src/models/event";
 | |
| import { User } from "matrix-js-sdk/src/models/user";
 | |
| import { logger } from "matrix-js-sdk/src/logger";
 | |
| import { EventType } from "matrix-js-sdk/src/@types/event";
 | |
| 
 | |
| import { MatrixClientPeg } from "./MatrixClientPeg";
 | |
| import MultiInviter, { CompletionStates } from "./utils/MultiInviter";
 | |
| import Modal from "./Modal";
 | |
| import { _t } from "./languageHandler";
 | |
| import InviteDialog from "./components/views/dialogs/InviteDialog";
 | |
| import BaseAvatar from "./components/views/avatars/BaseAvatar";
 | |
| import { mediaFromMxc } from "./customisations/Media";
 | |
| import ErrorDialog from "./components/views/dialogs/ErrorDialog";
 | |
| import { KIND_DM, KIND_INVITE } from "./components/views/dialogs/InviteDialogTypes";
 | |
| import { Member } from "./utils/direct-messages";
 | |
| 
 | |
| export interface IInviteResult {
 | |
|     states: CompletionStates;
 | |
|     inviter: MultiInviter;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * 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[]} addresses Array of strings of addresses to invite. May be matrix IDs or 3pids.
 | |
|  * @param {boolean} sendSharedHistoryKeys whether to share e2ee keys with the invitees if applicable.
 | |
|  * @param {function} progressCallback optional callback, fired after each invite.
 | |
|  * @returns {Promise} Promise
 | |
|  */
 | |
| export function inviteMultipleToRoom(
 | |
|     roomId: string,
 | |
|     addresses: string[],
 | |
|     sendSharedHistoryKeys = false,
 | |
|     progressCallback?: () => void,
 | |
| ): Promise<IInviteResult> {
 | |
|     const inviter = new MultiInviter(roomId, progressCallback);
 | |
|     return inviter
 | |
|         .invite(addresses, undefined, sendSharedHistoryKeys)
 | |
|         .then((states) => Promise.resolve({ states, inviter }));
 | |
| }
 | |
| 
 | |
| export function showStartChatInviteDialog(initialText = ""): void {
 | |
|     // This dialog handles the room creation internally - we don't need to worry about it.
 | |
|     Modal.createDialog(
 | |
|         InviteDialog,
 | |
|         { kind: KIND_DM, initialText },
 | |
|         /*className=*/ "mx_InviteDialog_flexWrapper",
 | |
|         /*isPriority=*/ false,
 | |
|         /*isStatic=*/ true,
 | |
|     );
 | |
| }
 | |
| 
 | |
| export function showRoomInviteDialog(roomId: string, initialText = ""): void {
 | |
|     // This dialog handles the room creation internally - we don't need to worry about it.
 | |
|     Modal.createDialog(
 | |
|         InviteDialog,
 | |
|         {
 | |
|             kind: KIND_INVITE,
 | |
|             initialText,
 | |
|             roomId,
 | |
|         },
 | |
|         /*className=*/ "mx_InviteDialog_flexWrapper",
 | |
|         /*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: MatrixEvent): boolean {
 | |
|     if (!event || event.getType() !== EventType.RoomThirdPartyInvite) 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"];
 | |
|     if (requiredKeys.some((key) => !event.getContent()[key])) {
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     // Valid enough by our standards
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| export function inviteUsersToRoom(
 | |
|     roomId: string,
 | |
|     userIds: string[],
 | |
|     sendSharedHistoryKeys = false,
 | |
|     progressCallback?: () => void,
 | |
| ): Promise<void> {
 | |
|     return inviteMultipleToRoom(roomId, userIds, sendSharedHistoryKeys, progressCallback)
 | |
|         .then((result) => {
 | |
|             const room = MatrixClientPeg.get().getRoom(roomId)!;
 | |
|             showAnyInviteErrors(result.states, room, result.inviter);
 | |
|         })
 | |
|         .catch((err) => {
 | |
|             logger.error(err.stack);
 | |
|             Modal.createDialog(ErrorDialog, {
 | |
|                 title: _t("Failed to invite"),
 | |
|                 description: err && err.message ? err.message : _t("Operation failed"),
 | |
|             });
 | |
|         });
 | |
| }
 | |
| 
 | |
| export function showAnyInviteErrors(
 | |
|     states: CompletionStates,
 | |
|     room: Room,
 | |
|     inviter: MultiInviter,
 | |
|     userMap?: Map<string, Member>,
 | |
| ): boolean {
 | |
|     // Show user any errors
 | |
|     const failedUsers = Object.keys(states).filter((a) => states[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.
 | |
|         Modal.createDialog(ErrorDialog, {
 | |
|             title: _t("Failed to invite users to %(roomName)s", { roomName: room.name }),
 | |
|             description: inviter.getErrorText(failedUsers[0]),
 | |
|         });
 | |
|         return false;
 | |
|     } else {
 | |
|         const errorList: string[] = [];
 | |
|         for (const addr of failedUsers) {
 | |
|             if (states[addr] === "error") {
 | |
|                 const reason = inviter.getErrorText(addr);
 | |
|                 errorList.push(addr + ": " + reason);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         const cli = MatrixClientPeg.get();
 | |
|         if (errorList.length > 0) {
 | |
|             // React 16 doesn't let us use `errorList.join(<br />)` anymore, so this is our solution
 | |
|             const description = (
 | |
|                 <div className="mx_InviteDialog_multiInviterError">
 | |
|                     <h4>
 | |
|                         {_t(
 | |
|                             "We sent the others, but the below people couldn't be invited to <RoomName/>",
 | |
|                             {},
 | |
|                             {
 | |
|                                 RoomName: () => <b>{room.name}</b>,
 | |
|                             },
 | |
|                         )}
 | |
|                     </h4>
 | |
|                     <div>
 | |
|                         {failedUsers.map((addr) => {
 | |
|                             const user = userMap?.get(addr) || cli.getUser(addr);
 | |
|                             const name = (user as Member).name || (user as User).rawDisplayName;
 | |
|                             const avatarUrl = (user as Member).getMxcAvatarUrl?.() || (user as User).avatarUrl;
 | |
|                             return (
 | |
|                                 <div key={addr} className="mx_InviteDialog_tile mx_InviteDialog_tile--inviterError">
 | |
|                                     <div className="mx_InviteDialog_tile_avatarStack">
 | |
|                                         <BaseAvatar
 | |
|                                             url={
 | |
|                                                 (avatarUrl && mediaFromMxc(avatarUrl).getSquareThumbnailHttp(24)) ??
 | |
|                                                 undefined
 | |
|                                             }
 | |
|                                             name={name!}
 | |
|                                             idName={user?.userId}
 | |
|                                             width={36}
 | |
|                                             height={36}
 | |
|                                         />
 | |
|                                     </div>
 | |
|                                     <div className="mx_InviteDialog_tile_nameStack">
 | |
|                                         <span className="mx_InviteDialog_tile_nameStack_name">{name}</span>
 | |
|                                         <span className="mx_InviteDialog_tile_nameStack_userId">{user?.userId}</span>
 | |
|                                     </div>
 | |
|                                     <div className="mx_InviteDialog_tile--inviterError_errorText">
 | |
|                                         {inviter.getErrorText(addr)}
 | |
|                                     </div>
 | |
|                                 </div>
 | |
|                             );
 | |
|                         })}
 | |
|                     </div>
 | |
|                 </div>
 | |
|             );
 | |
| 
 | |
|             Modal.createDialog(ErrorDialog, {
 | |
|                 title: _t("Some invites couldn't be sent"),
 | |
|                 description,
 | |
|             });
 | |
|             return false;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return true;
 | |
| }
 |