2020-03-20 23:38:20 +03:00
|
|
|
/*
|
|
|
|
Copyright 2015, 2016 OpenMarket Ltd
|
|
|
|
Copyright 2017, 2018 Vector Creations 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.
|
|
|
|
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 * as React from "react";
|
2020-04-30 22:21:50 +03:00
|
|
|
import { _t, _td } from "../../../languageHandler";
|
2020-03-20 23:38:20 +03:00
|
|
|
import { RovingTabIndexProvider } from "../../../accessibility/RovingTabIndex";
|
|
|
|
import { ResizeNotifier } from "../../../utils/ResizeNotifier";
|
2020-06-04 00:07:12 +03:00
|
|
|
import RoomListStore, { LISTS_UPDATE_EVENT, RoomListStore2 } from "../../../stores/room-list/RoomListStore2";
|
2020-04-30 22:21:50 +03:00
|
|
|
import { ITagMap } from "../../../stores/room-list/algorithms/models";
|
|
|
|
import { DefaultTagID, TagID } from "../../../stores/room-list/models";
|
|
|
|
import { Dispatcher } from "flux";
|
2020-05-14 22:45:17 +03:00
|
|
|
import dis from "../../../dispatcher/dispatcher";
|
2020-07-02 18:04:38 +03:00
|
|
|
import defaultDispatcher from "../../../dispatcher/dispatcher";
|
2020-05-12 01:12:45 +03:00
|
|
|
import RoomSublist2 from "./RoomSublist2";
|
2020-05-14 22:45:17 +03:00
|
|
|
import { ActionPayload } from "../../../dispatcher/payloads";
|
2020-05-29 16:59:06 +03:00
|
|
|
import { NameFilterCondition } from "../../../stores/room-list/filters/NameFilterCondition";
|
2020-06-04 06:16:53 +03:00
|
|
|
import { ListLayout } from "../../../stores/room-list/ListLayout";
|
2020-07-02 18:04:38 +03:00
|
|
|
import { MatrixClientPeg } from "../../../MatrixClientPeg";
|
|
|
|
import GroupAvatar from "../avatars/GroupAvatar";
|
|
|
|
import TemporaryTile from "./TemporaryTile";
|
|
|
|
import { NotificationColor, StaticNotificationState } from "./NotificationBadge";
|
2020-03-20 23:38:20 +03:00
|
|
|
|
2020-06-29 05:03:04 +03:00
|
|
|
// TODO: Remove banner on launch: https://github.com/vector-im/riot-web/issues/14231
|
|
|
|
// TODO: Rename on launch: https://github.com/vector-im/riot-web/issues/14231
|
|
|
|
|
2020-05-14 21:53:00 +03:00
|
|
|
/*******************************************************************
|
|
|
|
* CAUTION *
|
|
|
|
*******************************************************************
|
|
|
|
* This is a work in progress implementation and isn't complete or *
|
|
|
|
* even useful as a component. Please avoid using it until this *
|
|
|
|
* warning disappears. *
|
2020-05-21 20:53:16 +03:00
|
|
|
*******************************************************************/
|
2020-05-14 21:53:00 +03:00
|
|
|
|
2020-03-20 23:38:20 +03:00
|
|
|
interface IProps {
|
|
|
|
onKeyDown: (ev: React.KeyboardEvent) => void;
|
|
|
|
onFocus: (ev: React.FocusEvent) => void;
|
|
|
|
onBlur: (ev: React.FocusEvent) => void;
|
|
|
|
resizeNotifier: ResizeNotifier;
|
|
|
|
collapsed: boolean;
|
|
|
|
searchFilter: string;
|
2020-06-11 23:39:28 +03:00
|
|
|
isMinimized: boolean;
|
2020-03-20 23:38:20 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
interface IState {
|
2020-04-30 22:21:50 +03:00
|
|
|
sublists: ITagMap;
|
2020-06-04 06:16:53 +03:00
|
|
|
layouts: Map<TagID, ListLayout>;
|
2020-03-20 23:38:20 +03:00
|
|
|
}
|
|
|
|
|
2020-04-30 22:21:50 +03:00
|
|
|
const TAG_ORDER: TagID[] = [
|
|
|
|
// -- Community Invites Placeholder --
|
|
|
|
|
|
|
|
DefaultTagID.Invite,
|
|
|
|
DefaultTagID.Favourite,
|
|
|
|
DefaultTagID.DM,
|
|
|
|
DefaultTagID.Untagged,
|
|
|
|
|
|
|
|
// -- Custom Tags Placeholder --
|
|
|
|
|
|
|
|
DefaultTagID.LowPriority,
|
|
|
|
DefaultTagID.ServerNotice,
|
|
|
|
DefaultTagID.Archived,
|
|
|
|
];
|
|
|
|
const COMMUNITY_TAGS_BEFORE_TAG = DefaultTagID.Invite;
|
|
|
|
const CUSTOM_TAGS_BEFORE_TAG = DefaultTagID.LowPriority;
|
|
|
|
const ALWAYS_VISIBLE_TAGS: TagID[] = [
|
|
|
|
DefaultTagID.DM,
|
|
|
|
DefaultTagID.Untagged,
|
|
|
|
];
|
|
|
|
|
|
|
|
interface ITagAesthetics {
|
|
|
|
sectionLabel: string;
|
|
|
|
addRoomLabel?: string;
|
|
|
|
onAddRoom?: (dispatcher: Dispatcher<ActionPayload>) => void;
|
|
|
|
isInvite: boolean;
|
|
|
|
defaultHidden: boolean;
|
2020-03-20 23:38:20 +03:00
|
|
|
}
|
|
|
|
|
2020-04-30 22:21:50 +03:00
|
|
|
const TAG_AESTHETICS: {
|
|
|
|
// @ts-ignore - TS wants this to be a string but we know better
|
|
|
|
[tagId: TagID]: ITagAesthetics;
|
|
|
|
} = {
|
|
|
|
[DefaultTagID.Invite]: {
|
|
|
|
sectionLabel: _td("Invites"),
|
|
|
|
isInvite: true,
|
|
|
|
defaultHidden: false,
|
|
|
|
},
|
|
|
|
[DefaultTagID.Favourite]: {
|
|
|
|
sectionLabel: _td("Favourites"),
|
|
|
|
isInvite: false,
|
|
|
|
defaultHidden: false,
|
|
|
|
},
|
|
|
|
[DefaultTagID.DM]: {
|
2020-06-05 06:21:04 +03:00
|
|
|
sectionLabel: _td("People"),
|
2020-04-30 22:21:50 +03:00
|
|
|
isInvite: false,
|
|
|
|
defaultHidden: false,
|
|
|
|
addRoomLabel: _td("Start chat"),
|
|
|
|
onAddRoom: (dispatcher: Dispatcher<ActionPayload>) => dispatcher.dispatch({action: 'view_create_chat'}),
|
|
|
|
},
|
|
|
|
[DefaultTagID.Untagged]: {
|
|
|
|
sectionLabel: _td("Rooms"),
|
|
|
|
isInvite: false,
|
|
|
|
defaultHidden: false,
|
|
|
|
addRoomLabel: _td("Create room"),
|
|
|
|
onAddRoom: (dispatcher: Dispatcher<ActionPayload>) => dispatcher.dispatch({action: 'view_create_room'}),
|
|
|
|
},
|
|
|
|
[DefaultTagID.LowPriority]: {
|
|
|
|
sectionLabel: _td("Low priority"),
|
|
|
|
isInvite: false,
|
|
|
|
defaultHidden: false,
|
|
|
|
},
|
|
|
|
[DefaultTagID.ServerNotice]: {
|
|
|
|
sectionLabel: _td("System Alerts"),
|
|
|
|
isInvite: false,
|
|
|
|
defaultHidden: false,
|
|
|
|
},
|
2020-06-29 05:03:04 +03:00
|
|
|
|
|
|
|
// TODO: Replace with archived view: https://github.com/vector-im/riot-web/issues/14038
|
2020-04-30 22:21:50 +03:00
|
|
|
[DefaultTagID.Archived]: {
|
|
|
|
sectionLabel: _td("Historical"),
|
|
|
|
isInvite: false,
|
|
|
|
defaultHidden: true,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2020-03-20 23:38:20 +03:00
|
|
|
export default class RoomList2 extends React.Component<IProps, IState> {
|
2020-05-29 16:59:06 +03:00
|
|
|
private searchFilter: NameFilterCondition = new NameFilterCondition();
|
2020-03-20 23:38:20 +03:00
|
|
|
|
|
|
|
constructor(props: IProps) {
|
|
|
|
super(props);
|
|
|
|
|
2020-06-04 00:07:12 +03:00
|
|
|
this.state = {
|
|
|
|
sublists: {},
|
2020-06-04 06:16:53 +03:00
|
|
|
layouts: new Map<TagID, ListLayout>(),
|
2020-06-04 00:07:12 +03:00
|
|
|
};
|
2020-03-20 23:38:20 +03:00
|
|
|
}
|
|
|
|
|
2020-05-29 16:59:06 +03:00
|
|
|
public componentDidUpdate(prevProps: Readonly<IProps>): void {
|
|
|
|
if (prevProps.searchFilter !== this.props.searchFilter) {
|
|
|
|
const hadSearch = !!this.searchFilter.search.trim();
|
|
|
|
const haveSearch = !!this.props.searchFilter.trim();
|
|
|
|
this.searchFilter.search = this.props.searchFilter;
|
|
|
|
if (!hadSearch && haveSearch) {
|
|
|
|
// started a new filter - add the condition
|
|
|
|
RoomListStore.instance.addFilter(this.searchFilter);
|
|
|
|
} else if (hadSearch && !haveSearch) {
|
|
|
|
// cleared a filter - remove the condition
|
|
|
|
RoomListStore.instance.removeFilter(this.searchFilter);
|
|
|
|
} // else the filter hasn't changed enough for us to care here
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-20 23:38:20 +03:00
|
|
|
public componentDidMount(): void {
|
2020-06-04 00:07:12 +03:00
|
|
|
RoomListStore.instance.on(LISTS_UPDATE_EVENT, (store: RoomListStore2) => {
|
|
|
|
const newLists = store.orderedLists;
|
|
|
|
console.log("new lists", newLists);
|
|
|
|
|
2020-06-04 06:16:53 +03:00
|
|
|
const layoutMap = new Map<TagID, ListLayout>();
|
2020-06-04 00:07:12 +03:00
|
|
|
for (const tagId of Object.keys(newLists)) {
|
2020-06-04 06:16:53 +03:00
|
|
|
layoutMap.set(tagId, new ListLayout(tagId));
|
2020-06-04 00:07:12 +03:00
|
|
|
}
|
|
|
|
|
2020-06-04 06:16:53 +03:00
|
|
|
this.setState({sublists: newLists, layouts: layoutMap});
|
2020-03-20 23:38:20 +03:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2020-07-02 18:04:38 +03:00
|
|
|
private renderCommunityInvites(): React.ReactElement[] {
|
|
|
|
// TODO: Put community invites in a more sensible place (not in the room list)
|
|
|
|
return MatrixClientPeg.get().getGroups().filter(g => {
|
|
|
|
if (g.myMembership !== 'invite') return false;
|
|
|
|
return !this.searchFilter || this.searchFilter.matches(g.name);
|
|
|
|
}).map(g => {
|
|
|
|
const avatar = (
|
|
|
|
<GroupAvatar
|
|
|
|
groupId={g.groupId}
|
|
|
|
groupName={g.name}
|
|
|
|
groupAvatarUrl={g.avatarUrl}
|
|
|
|
width={32} height={32} resizeMethod='crop'
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
const openGroup = () => {
|
|
|
|
defaultDispatcher.dispatch({
|
|
|
|
action: 'view_group',
|
|
|
|
group_id: g.groupId,
|
|
|
|
});
|
|
|
|
};
|
|
|
|
return (
|
|
|
|
<TemporaryTile
|
|
|
|
isMinimized={this.props.isMinimized}
|
|
|
|
isSelected={false}
|
|
|
|
displayName={g.name}
|
|
|
|
avatar={avatar}
|
|
|
|
notificationState={StaticNotificationState.forSymbol("!", NotificationColor.Red)}
|
|
|
|
onClick={openGroup}
|
2020-07-02 18:27:42 +03:00
|
|
|
key={`temporaryGroupTile_${g.groupId}`}
|
2020-07-02 18:04:38 +03:00
|
|
|
/>
|
|
|
|
);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2020-04-30 22:21:50 +03:00
|
|
|
private renderSublists(): React.ReactElement[] {
|
|
|
|
const components: React.ReactElement[] = [];
|
|
|
|
|
|
|
|
for (const orderedTagId of TAG_ORDER) {
|
|
|
|
if (COMMUNITY_TAGS_BEFORE_TAG === orderedTagId) {
|
|
|
|
// Populate community invites if we have the chance
|
2020-06-29 05:03:04 +03:00
|
|
|
// TODO: Community invites: https://github.com/vector-im/riot-web/issues/14179
|
2020-04-30 22:21:50 +03:00
|
|
|
}
|
|
|
|
if (CUSTOM_TAGS_BEFORE_TAG === orderedTagId) {
|
|
|
|
// Populate custom tags if needed
|
2020-06-29 05:03:04 +03:00
|
|
|
// TODO: Custom tags: https://github.com/vector-im/riot-web/issues/14091
|
2020-04-30 22:21:50 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
const orderedRooms = this.state.sublists[orderedTagId] || [];
|
|
|
|
if (orderedRooms.length === 0 && !ALWAYS_VISIBLE_TAGS.includes(orderedTagId)) {
|
|
|
|
continue; // skip tag - not needed
|
|
|
|
}
|
|
|
|
|
|
|
|
const aesthetics: ITagAesthetics = TAG_AESTHETICS[orderedTagId];
|
|
|
|
if (!aesthetics) throw new Error(`Tag ${orderedTagId} does not have aesthetics`);
|
|
|
|
|
2020-05-08 21:53:05 +03:00
|
|
|
const onAddRoomFn = aesthetics.onAddRoom ? () => aesthetics.onAddRoom(dis) : null;
|
2020-07-02 18:04:38 +03:00
|
|
|
const extraTiles = orderedTagId === DefaultTagID.Invite ? this.renderCommunityInvites() : null;
|
2020-06-04 06:16:53 +03:00
|
|
|
components.push(
|
|
|
|
<RoomSublist2
|
|
|
|
key={`sublist-${orderedTagId}`}
|
2020-06-22 23:52:17 +03:00
|
|
|
tagId={orderedTagId}
|
2020-06-04 06:16:53 +03:00
|
|
|
forRooms={true}
|
|
|
|
rooms={orderedRooms}
|
|
|
|
startAsHidden={aesthetics.defaultHidden}
|
|
|
|
label={_t(aesthetics.sectionLabel)}
|
|
|
|
onAddRoom={onAddRoomFn}
|
|
|
|
addRoomLabel={aesthetics.addRoomLabel}
|
|
|
|
isInvite={aesthetics.isInvite}
|
|
|
|
layout={this.state.layouts.get(orderedTagId)}
|
2020-06-11 23:39:28 +03:00
|
|
|
isMinimized={this.props.isMinimized}
|
2020-07-02 18:04:38 +03:00
|
|
|
extraBadTilesThatShouldntExist={extraTiles}
|
2020-06-04 06:16:53 +03:00
|
|
|
/>
|
|
|
|
);
|
2020-04-30 22:21:50 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
return components;
|
|
|
|
}
|
|
|
|
|
2020-03-20 23:38:20 +03:00
|
|
|
public render() {
|
2020-04-30 22:21:50 +03:00
|
|
|
const sublists = this.renderSublists();
|
2020-03-20 23:38:20 +03:00
|
|
|
return (
|
|
|
|
<RovingTabIndexProvider handleHomeEnd={true} onKeyDown={this.props.onKeyDown}>
|
|
|
|
{({onKeyDownHandler}) => (
|
|
|
|
<div
|
|
|
|
onFocus={this.props.onFocus}
|
|
|
|
onBlur={this.props.onBlur}
|
|
|
|
onKeyDown={onKeyDownHandler}
|
2020-06-05 01:34:04 +03:00
|
|
|
className="mx_RoomList2"
|
2020-03-20 23:38:20 +03:00
|
|
|
role="tree"
|
|
|
|
aria-label={_t("Rooms")}
|
|
|
|
// Firefox sometimes makes this element focusable due to
|
|
|
|
// overflow:scroll;, so force it out of tab order.
|
|
|
|
tabIndex={-1}
|
2020-04-30 22:21:50 +03:00
|
|
|
>{sublists}</div>
|
2020-03-20 23:38:20 +03:00
|
|
|
)}
|
|
|
|
</RovingTabIndexProvider>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|