mirror of
https://github.com/element-hq/element-web.git
synced 2024-12-17 16:12:13 +03:00
1ba601ba22
Conflicts: src/components/structures/SpaceRoomView.tsx src/components/views/dialogs/BetaFeedbackDialog.tsx
366 lines
14 KiB
TypeScript
366 lines
14 KiB
TypeScript
/*
|
|
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.
|
|
*/
|
|
|
|
import React, { ReactNode, useContext, useMemo, useState } from "react";
|
|
import classNames from "classnames";
|
|
import { Room } from "matrix-js-sdk/src/models/room";
|
|
import { sleep } from "matrix-js-sdk/src/utils";
|
|
|
|
import { _t } from '../../../languageHandler';
|
|
import { IDialogProps } from "./IDialogProps";
|
|
import BaseDialog from "./BaseDialog";
|
|
import Dropdown from "../elements/Dropdown";
|
|
import SearchBox from "../../structures/SearchBox";
|
|
import SpaceStore from "../../../stores/SpaceStore";
|
|
import RoomAvatar from "../avatars/RoomAvatar";
|
|
import { getDisplayAliasForRoom } from "../../../Rooms";
|
|
import AccessibleButton from "../elements/AccessibleButton";
|
|
import AutoHideScrollbar from "../../structures/AutoHideScrollbar";
|
|
import DMRoomMap from "../../../utils/DMRoomMap";
|
|
import { calculateRoomVia } from "../../../utils/permalinks/Permalinks";
|
|
import StyledCheckbox from "../elements/StyledCheckbox";
|
|
import MatrixClientContext from "../../../contexts/MatrixClientContext";
|
|
import { sortRooms } from "../../../stores/room-list/algorithms/tag-sorting/RecentAlgorithm";
|
|
import ProgressBar from "../elements/ProgressBar";
|
|
import DecoratedRoomAvatar from "../avatars/DecoratedRoomAvatar";
|
|
import QueryMatcher from "../../../autocomplete/QueryMatcher";
|
|
import TruncatedList from "../elements/TruncatedList";
|
|
import EntityTile from "../rooms/EntityTile";
|
|
import BaseAvatar from "../avatars/BaseAvatar";
|
|
|
|
interface IProps extends IDialogProps {
|
|
space: Room;
|
|
onCreateRoomClick(space: Room): void;
|
|
}
|
|
|
|
const Entry = ({ room, checked, onChange }) => {
|
|
return <label className="mx_AddExistingToSpace_entry">
|
|
{ room?.isSpaceRoom()
|
|
? <RoomAvatar room={room} height={32} width={32} />
|
|
: <DecoratedRoomAvatar room={room} avatarSize={32} />
|
|
}
|
|
<span className="mx_AddExistingToSpace_entry_name">{ room.name }</span>
|
|
<StyledCheckbox
|
|
onChange={onChange ? (e) => onChange(e.target.checked) : null}
|
|
checked={checked}
|
|
disabled={!onChange}
|
|
/>
|
|
</label>;
|
|
};
|
|
|
|
interface IAddExistingToSpaceProps {
|
|
space: Room;
|
|
footerPrompt?: ReactNode;
|
|
emptySelectionButton?: ReactNode;
|
|
onFinished(added: boolean): void;
|
|
}
|
|
|
|
export const AddExistingToSpace: React.FC<IAddExistingToSpaceProps> = ({
|
|
space,
|
|
footerPrompt,
|
|
emptySelectionButton,
|
|
onFinished,
|
|
}) => {
|
|
const cli = useContext(MatrixClientContext);
|
|
const visibleRooms = useMemo(() => cli.getVisibleRooms().filter(r => r.getMyMembership() === "join"), [cli]);
|
|
|
|
const [selectedToAdd, setSelectedToAdd] = useState(new Set<Room>());
|
|
const [progress, setProgress] = useState<number>(null);
|
|
const [error, setError] = useState<Error>(null);
|
|
const [query, setQuery] = useState("");
|
|
const lcQuery = query.toLowerCase().trim();
|
|
|
|
const existingSubspacesSet = useMemo(() => new Set(SpaceStore.instance.getChildSpaces(space.roomId)), [space]);
|
|
const existingRoomsSet = useMemo(() => new Set(SpaceStore.instance.getChildRooms(space.roomId)), [space]);
|
|
|
|
const [spaces, rooms, dms] = useMemo(() => {
|
|
let rooms = visibleRooms;
|
|
|
|
if (lcQuery) {
|
|
const matcher = new QueryMatcher<Room>(visibleRooms, {
|
|
keys: ["name"],
|
|
funcs: [r => [r.getCanonicalAlias(), ...r.getAltAliases()].filter(Boolean)],
|
|
shouldMatchWordsOnly: false,
|
|
});
|
|
|
|
rooms = matcher.match(lcQuery);
|
|
}
|
|
|
|
const joinRule = space.getJoinRule();
|
|
return sortRooms(rooms).reduce((arr, room) => {
|
|
if (room.isSpaceRoom()) {
|
|
if (room !== space && !existingSubspacesSet.has(room)) {
|
|
arr[0].push(room);
|
|
}
|
|
} else if (!existingRoomsSet.has(room)) {
|
|
if (!DMRoomMap.shared().getUserIdForRoomId(room.roomId)) {
|
|
arr[1].push(room);
|
|
} else if (joinRule !== "public") {
|
|
// Only show DMs for non-public spaces as they make very little sense in spaces other than "Just Me" ones.
|
|
arr[2].push(room);
|
|
}
|
|
}
|
|
return arr;
|
|
}, [[], [], []]);
|
|
}, [visibleRooms, space, lcQuery, existingRoomsSet, existingSubspacesSet]);
|
|
|
|
const addRooms = async () => {
|
|
setError(null);
|
|
setProgress(0);
|
|
|
|
let error;
|
|
|
|
for (const room of selectedToAdd) {
|
|
const via = calculateRoomVia(room);
|
|
try {
|
|
await SpaceStore.instance.addRoomToSpace(space, room.roomId, via).catch(async e => {
|
|
if (e.errcode === "M_LIMIT_EXCEEDED") {
|
|
await sleep(e.data.retry_after_ms);
|
|
return SpaceStore.instance.addRoomToSpace(space, room.roomId, via); // retry
|
|
}
|
|
|
|
throw e;
|
|
});
|
|
setProgress(i => i + 1);
|
|
} catch (e) {
|
|
console.error("Failed to add rooms to space", e);
|
|
setError(error = e);
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!error) {
|
|
onFinished(true);
|
|
}
|
|
};
|
|
|
|
const busy = progress !== null;
|
|
|
|
let footer;
|
|
if (error) {
|
|
footer = <>
|
|
<img
|
|
src={require("../../../../res/img/element-icons/warning-badge.svg")}
|
|
height="24"
|
|
width="24"
|
|
alt=""
|
|
/>
|
|
|
|
<span className="mx_AddExistingToSpaceDialog_error">
|
|
<div className="mx_AddExistingToSpaceDialog_errorHeading">{ _t("Not all selected were added") }</div>
|
|
<div className="mx_AddExistingToSpaceDialog_errorCaption">{ _t("Try again") }</div>
|
|
</span>
|
|
|
|
<AccessibleButton className="mx_AddExistingToSpaceDialog_retryButton" onClick={addRooms}>
|
|
{ _t("Retry") }
|
|
</AccessibleButton>
|
|
</>;
|
|
} else if (busy) {
|
|
footer = <span>
|
|
<ProgressBar value={progress} max={selectedToAdd.size} />
|
|
<div className="mx_AddExistingToSpaceDialog_progressText">
|
|
{ _t("Adding rooms... (%(progress)s out of %(count)s)", {
|
|
count: selectedToAdd.size,
|
|
progress,
|
|
}) }
|
|
</div>
|
|
</span>;
|
|
} else {
|
|
let button = emptySelectionButton;
|
|
if (!button || selectedToAdd.size > 0) {
|
|
button = <AccessibleButton kind="primary" disabled={selectedToAdd.size < 1} onClick={addRooms}>
|
|
{ _t("Add") }
|
|
</AccessibleButton>;
|
|
}
|
|
|
|
footer = <>
|
|
<span>
|
|
{ footerPrompt }
|
|
</span>
|
|
|
|
{ button }
|
|
</>;
|
|
}
|
|
|
|
const onChange = !busy && !error ? (checked, room) => {
|
|
if (checked) {
|
|
selectedToAdd.add(room);
|
|
} else {
|
|
selectedToAdd.delete(room);
|
|
}
|
|
setSelectedToAdd(new Set(selectedToAdd));
|
|
} : null;
|
|
|
|
const [truncateAt, setTruncateAt] = useState(20);
|
|
function overflowTile(overflowCount, totalCount) {
|
|
const text = _t("and %(count)s others...", { count: overflowCount });
|
|
return (
|
|
<EntityTile
|
|
className="mx_EntityTile_ellipsis"
|
|
avatarJsx={
|
|
<BaseAvatar url={require("../../../../res/img/ellipsis.svg")} name="..." width={36} height={36} />
|
|
}
|
|
name={text}
|
|
presenceState="online"
|
|
suppressOnHover={true}
|
|
onClick={() => setTruncateAt(totalCount)}
|
|
/>
|
|
);
|
|
}
|
|
|
|
return <div className="mx_AddExistingToSpace">
|
|
<SearchBox
|
|
className="mx_textinput_icon mx_textinput_search"
|
|
placeholder={_t("Filter your rooms and spaces")}
|
|
onSearch={setQuery}
|
|
autoComplete={true}
|
|
autoFocus={true}
|
|
/>
|
|
<AutoHideScrollbar className="mx_AddExistingToSpace_content">
|
|
{ rooms.length > 0 ? (
|
|
<div className="mx_AddExistingToSpace_section">
|
|
<h3>{ _t("Rooms") }</h3>
|
|
<TruncatedList
|
|
truncateAt={truncateAt}
|
|
createOverflowElement={overflowTile}
|
|
getChildren={(start, end) => rooms.slice(start, end).map(room =>
|
|
<Entry
|
|
key={room.roomId}
|
|
room={room}
|
|
checked={selectedToAdd.has(room)}
|
|
onChange={onChange ? (checked) => {
|
|
onChange(checked, room);
|
|
} : null}
|
|
/>,
|
|
)}
|
|
getChildCount={() => rooms.length}
|
|
/>
|
|
</div>
|
|
) : undefined }
|
|
|
|
{ spaces.length > 0 ? (
|
|
<div className="mx_AddExistingToSpace_section mx_AddExistingToSpace_section_spaces">
|
|
<h3>{ _t("Spaces") }</h3>
|
|
<div className="mx_AddExistingToSpace_section_experimental">
|
|
<div>{ _t("Feeling experimental?") }</div>
|
|
<div>{ _t("You can add existing spaces to a space.") }</div>
|
|
</div>
|
|
{ spaces.map(space => {
|
|
return <Entry
|
|
key={space.roomId}
|
|
room={space}
|
|
checked={selectedToAdd.has(space)}
|
|
onChange={onChange ? (checked) => {
|
|
onChange(checked, space);
|
|
} : null}
|
|
/>;
|
|
}) }
|
|
</div>
|
|
) : null }
|
|
|
|
{ dms.length > 0 ? (
|
|
<div className="mx_AddExistingToSpace_section">
|
|
<h3>{ _t("Direct Messages") }</h3>
|
|
{ dms.map(room => {
|
|
return <Entry
|
|
key={room.roomId}
|
|
room={room}
|
|
checked={selectedToAdd.has(room)}
|
|
onChange={onChange ? (checked) => {
|
|
onChange(checked, room);
|
|
} : null}
|
|
/>;
|
|
}) }
|
|
</div>
|
|
) : null }
|
|
|
|
{ spaces.length + rooms.length + dms.length < 1 ? <span className="mx_AddExistingToSpace_noResults">
|
|
{ _t("No results") }
|
|
</span> : undefined }
|
|
</AutoHideScrollbar>
|
|
|
|
<div className="mx_AddExistingToSpace_footer">
|
|
{ footer }
|
|
</div>
|
|
</div>;
|
|
};
|
|
|
|
const AddExistingToSpaceDialog: React.FC<IProps> = ({ space, onCreateRoomClick, onFinished }) => {
|
|
const [selectedSpace, setSelectedSpace] = useState(space);
|
|
const existingSubspaces = SpaceStore.instance.getChildSpaces(space.roomId);
|
|
|
|
let spaceOptionSection;
|
|
if (existingSubspaces.length > 0) {
|
|
const options = [space, ...existingSubspaces].map((space) => {
|
|
const classes = classNames("mx_AddExistingToSpaceDialog_dropdownOption", {
|
|
mx_AddExistingToSpaceDialog_dropdownOptionActive: space === selectedSpace,
|
|
});
|
|
return <div key={space.roomId} className={classes}>
|
|
<RoomAvatar room={space} width={24} height={24} />
|
|
{ space.name || getDisplayAliasForRoom(space) || space.roomId }
|
|
</div>;
|
|
});
|
|
|
|
spaceOptionSection = (
|
|
<Dropdown
|
|
id="mx_SpaceSelectDropdown"
|
|
onOptionChange={(key: string) => {
|
|
setSelectedSpace(existingSubspaces.find(space => space.roomId === key) || space);
|
|
}}
|
|
value={selectedSpace.roomId}
|
|
label={_t("Space selection")}
|
|
>
|
|
{ options }
|
|
</Dropdown>
|
|
);
|
|
} else {
|
|
spaceOptionSection = <div className="mx_AddExistingToSpaceDialog_onlySpace">
|
|
{ space.name || getDisplayAliasForRoom(space) || space.roomId }
|
|
</div>;
|
|
}
|
|
|
|
const title = <React.Fragment>
|
|
<RoomAvatar room={selectedSpace} height={40} width={40} />
|
|
<div>
|
|
<h1>{ _t("Add existing rooms") }</h1>
|
|
{ spaceOptionSection }
|
|
</div>
|
|
</React.Fragment>;
|
|
|
|
return <BaseDialog
|
|
title={title}
|
|
className="mx_AddExistingToSpaceDialog"
|
|
contentId="mx_AddExistingToSpace"
|
|
onFinished={onFinished}
|
|
fixedWidth={false}
|
|
>
|
|
<MatrixClientContext.Provider value={space.client}>
|
|
<AddExistingToSpace
|
|
space={space}
|
|
onFinished={onFinished}
|
|
footerPrompt={<>
|
|
<div>{ _t("Want to add a new room instead?") }</div>
|
|
<AccessibleButton onClick={() => onCreateRoomClick(space)} kind="link">
|
|
{ _t("Create a new room") }
|
|
</AccessibleButton>
|
|
</>}
|
|
/>
|
|
</MatrixClientContext.Provider>
|
|
</BaseDialog>;
|
|
};
|
|
|
|
export default AddExistingToSpaceDialog;
|
|
|