2020-06-03 01:26:07 +00:00
|
|
|
/*
|
|
|
|
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-06-24 02:59:26 +00:00
|
|
|
import { createRef } from "react";
|
2020-06-03 01:26:07 +00:00
|
|
|
import TagPanel from "./TagPanel";
|
|
|
|
import classNames from "classnames";
|
|
|
|
import dis from "../../dispatcher/dispatcher";
|
|
|
|
import { _t } from "../../languageHandler";
|
|
|
|
import RoomList2 from "../views/rooms/RoomList2";
|
2020-07-08 13:11:47 +00:00
|
|
|
import { HEADER_HEIGHT } from "../views/rooms/RoomSublist2";
|
2020-06-03 01:26:07 +00:00
|
|
|
import { Action } from "../../dispatcher/actions";
|
2020-06-26 01:38:11 +00:00
|
|
|
import UserMenu from "./UserMenu";
|
2020-06-09 02:33:21 +00:00
|
|
|
import RoomSearch from "./RoomSearch";
|
|
|
|
import AccessibleButton from "../views/elements/AccessibleButton";
|
2020-06-08 23:11:58 +00:00
|
|
|
import RoomBreadcrumbs2 from "../views/rooms/RoomBreadcrumbs2";
|
|
|
|
import { BreadcrumbsStore } from "../../stores/BreadcrumbsStore";
|
|
|
|
import { UPDATE_EVENT } from "../../stores/AsyncStore";
|
2020-06-22 19:09:42 +00:00
|
|
|
import ResizeNotifier from "../../utils/ResizeNotifier";
|
2020-06-26 02:35:40 +00:00
|
|
|
import SettingsStore from "../../settings/SettingsStore";
|
2020-07-02 21:21:10 +00:00
|
|
|
import RoomListStore, { LISTS_UPDATE_EVENT } from "../../stores/room-list/RoomListStore2";
|
|
|
|
import {Key} from "../../Keyboard";
|
2020-07-06 20:32:46 +00:00
|
|
|
import IndicatorScrollbar from "../structures/IndicatorScrollbar";
|
2020-06-03 01:26:07 +00:00
|
|
|
|
2020-06-29 02:03:04 +00: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-06-03 01:26:07 +00: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. *
|
|
|
|
*******************************************************************/
|
|
|
|
|
|
|
|
interface IProps {
|
2020-06-11 20:39:28 +00:00
|
|
|
isMinimized: boolean;
|
2020-06-22 19:09:42 +00:00
|
|
|
resizeNotifier: ResizeNotifier;
|
2020-06-03 01:26:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
interface IState {
|
2020-06-29 02:03:04 +00:00
|
|
|
searchFilter: string;
|
2020-06-08 23:11:58 +00:00
|
|
|
showBreadcrumbs: boolean;
|
2020-06-26 02:35:40 +00:00
|
|
|
showTagPanel: boolean;
|
2020-06-03 01:26:07 +00:00
|
|
|
}
|
|
|
|
|
2020-07-06 00:14:02 +00:00
|
|
|
// List of CSS classes which should be included in keyboard navigation within the room list
|
|
|
|
const cssClasses = [
|
|
|
|
"mx_RoomSearch_input",
|
|
|
|
"mx_RoomSearch_icon", // minimized <RoomSearch />
|
|
|
|
"mx_RoomSublist2_headerText",
|
|
|
|
"mx_RoomTile2",
|
|
|
|
"mx_RoomSublist2_showNButton",
|
|
|
|
];
|
|
|
|
|
2020-06-03 01:26:07 +00:00
|
|
|
export default class LeftPanel2 extends React.Component<IProps, IState> {
|
2020-06-22 19:09:42 +00:00
|
|
|
private listContainerRef: React.RefObject<HTMLDivElement> = createRef();
|
2020-06-26 02:35:40 +00:00
|
|
|
private tagPanelWatcherRef: string;
|
2020-07-02 21:21:10 +00:00
|
|
|
private focusedElement = null;
|
2020-07-07 17:33:32 +00:00
|
|
|
private isDoingStickyHeaders = false;
|
2020-06-22 19:09:42 +00:00
|
|
|
|
2020-06-03 01:26:07 +00:00
|
|
|
constructor(props: IProps) {
|
|
|
|
super(props);
|
|
|
|
|
|
|
|
this.state = {
|
|
|
|
searchFilter: "",
|
2020-06-08 23:11:58 +00:00
|
|
|
showBreadcrumbs: BreadcrumbsStore.instance.visible,
|
2020-06-26 02:35:40 +00:00
|
|
|
showTagPanel: SettingsStore.getValue('TagPanel.enableTagPanel'),
|
2020-06-03 01:26:07 +00:00
|
|
|
};
|
2020-06-08 23:11:58 +00:00
|
|
|
|
|
|
|
BreadcrumbsStore.instance.on(UPDATE_EVENT, this.onBreadcrumbsUpdate);
|
2020-07-01 15:05:27 +00:00
|
|
|
RoomListStore.instance.on(LISTS_UPDATE_EVENT, this.onBreadcrumbsUpdate);
|
2020-06-26 02:35:40 +00:00
|
|
|
this.tagPanelWatcherRef = SettingsStore.watchSetting("TagPanel.enableTagPanel", null, () => {
|
|
|
|
this.setState({showTagPanel: SettingsStore.getValue("TagPanel.enableTagPanel")});
|
|
|
|
});
|
2020-06-22 19:09:42 +00:00
|
|
|
|
|
|
|
// We watch the middle panel because we don't actually get resized, the middle panel does.
|
|
|
|
// We listen to the noisy channel to avoid choppy reaction times.
|
|
|
|
this.props.resizeNotifier.on("middlePanelResizedNoisy", this.onResize);
|
2020-06-08 23:11:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public componentWillUnmount() {
|
2020-06-26 02:35:40 +00:00
|
|
|
SettingsStore.unwatchSetting(this.tagPanelWatcherRef);
|
2020-06-08 23:11:58 +00:00
|
|
|
BreadcrumbsStore.instance.off(UPDATE_EVENT, this.onBreadcrumbsUpdate);
|
2020-07-01 15:05:27 +00:00
|
|
|
RoomListStore.instance.off(LISTS_UPDATE_EVENT, this.onBreadcrumbsUpdate);
|
2020-06-22 19:09:42 +00:00
|
|
|
this.props.resizeNotifier.off("middlePanelResizedNoisy", this.onResize);
|
2020-06-03 01:26:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
private onSearch = (term: string): void => {
|
|
|
|
this.setState({searchFilter: term});
|
|
|
|
};
|
|
|
|
|
2020-06-09 02:33:21 +00:00
|
|
|
private onExplore = () => {
|
|
|
|
dis.fire(Action.ViewRoomDirectory);
|
2020-06-03 01:26:07 +00:00
|
|
|
};
|
|
|
|
|
2020-06-08 23:11:58 +00:00
|
|
|
private onBreadcrumbsUpdate = () => {
|
|
|
|
const newVal = BreadcrumbsStore.instance.visible;
|
|
|
|
if (newVal !== this.state.showBreadcrumbs) {
|
|
|
|
this.setState({showBreadcrumbs: newVal});
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2020-06-22 19:09:42 +00:00
|
|
|
private handleStickyHeaders(list: HTMLDivElement) {
|
2020-07-07 17:33:32 +00:00
|
|
|
if (this.isDoingStickyHeaders) return;
|
|
|
|
this.isDoingStickyHeaders = true;
|
2020-07-08 18:17:51 +00:00
|
|
|
window.requestAnimationFrame(() => {
|
2020-07-07 17:33:32 +00:00
|
|
|
this.doStickyHeaders(list);
|
|
|
|
this.isDoingStickyHeaders = false;
|
2020-07-08 18:17:51 +00:00
|
|
|
});
|
2020-07-07 17:33:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
private doStickyHeaders(list: HTMLDivElement) {
|
2020-06-13 17:54:40 +00:00
|
|
|
const rlRect = list.getBoundingClientRect();
|
|
|
|
const bottom = rlRect.bottom;
|
|
|
|
const top = rlRect.top;
|
|
|
|
const sublists = list.querySelectorAll<HTMLDivElement>(".mx_RoomSublist2");
|
|
|
|
const headerRightMargin = 24; // calculated from margins and widths to align with non-sticky tiles
|
|
|
|
|
|
|
|
const headerStickyWidth = rlRect.width - headerRightMargin;
|
|
|
|
|
|
|
|
let gotBottom = false;
|
2020-07-03 10:17:54 +00:00
|
|
|
let lastTopHeader;
|
2020-06-13 17:54:40 +00:00
|
|
|
for (const sublist of sublists) {
|
|
|
|
const slRect = sublist.getBoundingClientRect();
|
|
|
|
|
2020-06-14 01:07:19 +00:00
|
|
|
const header = sublist.querySelector<HTMLDivElement>(".mx_RoomSublist2_stickable");
|
2020-07-08 08:21:33 +00:00
|
|
|
header.style.removeProperty("display"); // always clear display:none first
|
2020-06-13 17:54:40 +00:00
|
|
|
|
2020-07-08 13:11:47 +00:00
|
|
|
if (slRect.top + HEADER_HEIGHT > bottom && !gotBottom) {
|
2020-06-13 17:54:40 +00:00
|
|
|
header.classList.add("mx_RoomSublist2_headerContainer_sticky");
|
|
|
|
header.classList.add("mx_RoomSublist2_headerContainer_stickyBottom");
|
|
|
|
header.style.width = `${headerStickyWidth}px`;
|
2020-07-03 14:52:52 +00:00
|
|
|
header.style.removeProperty("top");
|
2020-06-13 17:54:40 +00:00
|
|
|
gotBottom = true;
|
2020-07-08 13:11:47 +00:00
|
|
|
} else if (((slRect.top - (HEADER_HEIGHT * 0.6) + HEADER_HEIGHT) < top) || sublist === sublists[0]) {
|
2020-07-08 12:49:38 +00:00
|
|
|
// the header should become sticky once it is 60% or less out of view at the top.
|
2020-07-08 13:11:47 +00:00
|
|
|
// We also add HEADER_HEIGHT because the sticky header is put above the scrollable area,
|
2020-07-08 12:49:38 +00:00
|
|
|
// into the padding of .mx_LeftPanel2_roomListWrapper,
|
2020-07-08 13:11:47 +00:00
|
|
|
// by subtracting HEADER_HEIGHT from the top below.
|
2020-07-08 12:49:38 +00:00
|
|
|
// We also always try to make the first sublist header sticky.
|
2020-06-13 17:54:40 +00:00
|
|
|
header.classList.add("mx_RoomSublist2_headerContainer_sticky");
|
|
|
|
header.classList.add("mx_RoomSublist2_headerContainer_stickyTop");
|
|
|
|
header.style.width = `${headerStickyWidth}px`;
|
2020-07-08 13:11:47 +00:00
|
|
|
header.style.top = `${rlRect.top - HEADER_HEIGHT}px`;
|
2020-07-03 10:17:54 +00:00
|
|
|
if (lastTopHeader) {
|
|
|
|
lastTopHeader.style.display = "none";
|
|
|
|
}
|
|
|
|
lastTopHeader = header;
|
2020-06-13 17:54:40 +00:00
|
|
|
} else {
|
|
|
|
header.classList.remove("mx_RoomSublist2_headerContainer_sticky");
|
|
|
|
header.classList.remove("mx_RoomSublist2_headerContainer_stickyTop");
|
|
|
|
header.classList.remove("mx_RoomSublist2_headerContainer_stickyBottom");
|
2020-07-06 21:23:20 +00:00
|
|
|
header.style.removeProperty("width");
|
2020-07-03 14:52:52 +00:00
|
|
|
header.style.removeProperty("top");
|
2020-06-13 17:54:40 +00:00
|
|
|
}
|
|
|
|
}
|
2020-07-08 12:49:38 +00:00
|
|
|
|
|
|
|
// add appropriate sticky classes to wrapper so it has
|
|
|
|
// the necessary top/bottom padding to put the sticky header in
|
|
|
|
const listWrapper = list.parentElement;
|
|
|
|
if (gotBottom) {
|
|
|
|
listWrapper.classList.add("stickyBottom");
|
|
|
|
} else {
|
|
|
|
listWrapper.classList.remove("stickyBottom");
|
|
|
|
}
|
|
|
|
if (lastTopHeader) {
|
|
|
|
listWrapper.classList.add("stickyTop");
|
|
|
|
} else {
|
|
|
|
listWrapper.classList.remove("stickyTop");
|
|
|
|
}
|
|
|
|
|
|
|
|
// ensure scroll doesn't go above the gap left by the header of
|
|
|
|
// the first sublist always being sticky if no other header is sticky
|
2020-07-08 13:11:47 +00:00
|
|
|
if (list.scrollTop < HEADER_HEIGHT) {
|
|
|
|
list.scrollTop = HEADER_HEIGHT;
|
2020-07-08 12:49:38 +00:00
|
|
|
}
|
2020-06-22 19:09:42 +00:00
|
|
|
}
|
|
|
|
|
2020-06-29 02:03:04 +00:00
|
|
|
// TODO: Improve header reliability: https://github.com/vector-im/riot-web/issues/14232
|
2020-06-22 19:09:42 +00:00
|
|
|
private onScroll = (ev: React.MouseEvent<HTMLDivElement>) => {
|
|
|
|
const list = ev.target as HTMLDivElement;
|
|
|
|
this.handleStickyHeaders(list);
|
|
|
|
};
|
|
|
|
|
|
|
|
private onResize = () => {
|
|
|
|
if (!this.listContainerRef.current) return; // ignore: no headers to sticky
|
|
|
|
this.handleStickyHeaders(this.listContainerRef.current);
|
2020-06-13 17:54:40 +00:00
|
|
|
};
|
|
|
|
|
2020-07-02 21:21:10 +00:00
|
|
|
private onFocus = (ev: React.FocusEvent) => {
|
|
|
|
this.focusedElement = ev.target;
|
|
|
|
};
|
|
|
|
|
|
|
|
private onBlur = () => {
|
|
|
|
this.focusedElement = null;
|
|
|
|
};
|
|
|
|
|
|
|
|
private onKeyDown = (ev: React.KeyboardEvent) => {
|
|
|
|
if (!this.focusedElement) return;
|
|
|
|
|
|
|
|
switch (ev.key) {
|
|
|
|
case Key.ARROW_UP:
|
|
|
|
case Key.ARROW_DOWN:
|
2020-07-03 13:49:25 +00:00
|
|
|
ev.stopPropagation();
|
|
|
|
ev.preventDefault();
|
|
|
|
this.onMoveFocus(ev.key === Key.ARROW_UP);
|
2020-07-02 21:21:10 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2020-07-03 13:49:25 +00:00
|
|
|
private onMoveFocus = (up: boolean) => {
|
2020-07-02 21:21:10 +00:00
|
|
|
let element = this.focusedElement;
|
|
|
|
|
|
|
|
let descending = false; // are we currently descending or ascending through the DOM tree?
|
2020-07-03 13:49:25 +00:00
|
|
|
let classes: DOMTokenList;
|
2020-07-02 21:21:10 +00:00
|
|
|
|
|
|
|
do {
|
|
|
|
const child = up ? element.lastElementChild : element.firstElementChild;
|
|
|
|
const sibling = up ? element.previousElementSibling : element.nextElementSibling;
|
|
|
|
|
|
|
|
if (descending) {
|
|
|
|
if (child) {
|
|
|
|
element = child;
|
|
|
|
} else if (sibling) {
|
|
|
|
element = sibling;
|
|
|
|
} else {
|
|
|
|
descending = false;
|
|
|
|
element = element.parentElement;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (sibling) {
|
|
|
|
element = sibling;
|
|
|
|
descending = true;
|
|
|
|
} else {
|
|
|
|
element = element.parentElement;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (element) {
|
|
|
|
classes = element.classList;
|
|
|
|
}
|
2020-07-06 00:14:02 +00:00
|
|
|
} while (element && !cssClasses.some(c => classes.contains(c)));
|
2020-07-02 21:21:10 +00:00
|
|
|
|
|
|
|
if (element) {
|
|
|
|
element.focus();
|
|
|
|
this.focusedElement = element;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2020-06-04 22:34:04 +00:00
|
|
|
private renderHeader(): React.ReactNode {
|
2020-06-08 23:11:58 +00:00
|
|
|
let breadcrumbs;
|
2020-07-06 20:32:46 +00:00
|
|
|
if (this.state.showBreadcrumbs && !this.props.isMinimized) {
|
2020-06-08 23:11:58 +00:00
|
|
|
breadcrumbs = (
|
2020-07-06 20:32:46 +00:00
|
|
|
<IndicatorScrollbar
|
|
|
|
className="mx_LeftPanel2_headerRow mx_LeftPanel2_breadcrumbsContainer mx_AutoHideScrollbar"
|
|
|
|
verticalScrollsHorizontally={true}
|
|
|
|
>
|
|
|
|
<RoomBreadcrumbs2 />
|
|
|
|
</IndicatorScrollbar>
|
2020-06-08 23:11:58 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-06-04 22:34:04 +00:00
|
|
|
return (
|
|
|
|
<div className="mx_LeftPanel2_userHeader">
|
2020-06-26 01:38:11 +00:00
|
|
|
<UserMenu isMinimized={this.props.isMinimized} />
|
2020-06-08 23:11:58 +00:00
|
|
|
{breadcrumbs}
|
2020-06-03 01:26:07 +00:00
|
|
|
</div>
|
|
|
|
);
|
2020-06-04 22:34:04 +00:00
|
|
|
}
|
2020-06-03 01:26:07 +00:00
|
|
|
|
2020-06-09 02:33:21 +00:00
|
|
|
private renderSearchExplore(): React.ReactNode {
|
|
|
|
return (
|
2020-07-05 00:07:46 +00:00
|
|
|
<div
|
|
|
|
className="mx_LeftPanel2_filterContainer"
|
|
|
|
onFocus={this.onFocus}
|
|
|
|
onBlur={this.onBlur}
|
|
|
|
onKeyDown={this.onKeyDown}
|
|
|
|
>
|
2020-07-02 21:21:10 +00:00
|
|
|
<RoomSearch
|
|
|
|
onQueryUpdate={this.onSearch}
|
|
|
|
isMinimized={this.props.isMinimized}
|
|
|
|
onVerticalArrow={this.onKeyDown}
|
|
|
|
/>
|
2020-06-09 02:33:21 +00:00
|
|
|
<AccessibleButton
|
2020-07-03 13:43:02 +00:00
|
|
|
className="mx_LeftPanel2_exploreButton"
|
2020-06-09 02:33:21 +00:00
|
|
|
onClick={this.onExplore}
|
2020-07-05 00:07:46 +00:00
|
|
|
title={_t("Explore rooms")}
|
2020-06-09 02:33:21 +00:00
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-06-04 22:34:04 +00:00
|
|
|
public render(): React.ReactNode {
|
2020-06-26 02:35:40 +00:00
|
|
|
const tagPanel = !this.state.showTagPanel ? null : (
|
2020-06-04 22:34:04 +00:00
|
|
|
<div className="mx_LeftPanel2_tagPanelContainer">
|
|
|
|
<TagPanel/>
|
2020-06-03 01:26:07 +00:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
|
|
|
|
const roomList = <RoomList2
|
2020-07-02 21:21:10 +00:00
|
|
|
onKeyDown={this.onKeyDown}
|
2020-06-03 01:26:07 +00:00
|
|
|
resizeNotifier={null}
|
|
|
|
collapsed={false}
|
|
|
|
searchFilter={this.state.searchFilter}
|
2020-07-02 21:21:10 +00:00
|
|
|
onFocus={this.onFocus}
|
|
|
|
onBlur={this.onBlur}
|
2020-06-11 20:39:28 +00:00
|
|
|
isMinimized={this.props.isMinimized}
|
2020-07-06 20:05:06 +00:00
|
|
|
onResize={this.onResize}
|
2020-06-03 01:26:07 +00:00
|
|
|
/>;
|
|
|
|
|
2020-06-29 02:03:04 +00:00
|
|
|
// TODO: Conference handling / calls: https://github.com/vector-im/riot-web/issues/14177
|
2020-06-03 01:26:07 +00:00
|
|
|
|
|
|
|
const containerClasses = classNames({
|
2020-06-04 22:34:04 +00:00
|
|
|
"mx_LeftPanel2": true,
|
2020-06-26 02:35:40 +00:00
|
|
|
"mx_LeftPanel2_hasTagPanel": !!tagPanel,
|
2020-06-11 20:39:28 +00:00
|
|
|
"mx_LeftPanel2_minimized": this.props.isMinimized,
|
2020-06-03 01:26:07 +00:00
|
|
|
});
|
|
|
|
|
2020-07-01 14:15:18 +00:00
|
|
|
const roomListClasses = classNames(
|
2020-06-30 22:52:13 +00:00
|
|
|
"mx_LeftPanel2_actualRoomListContainer",
|
|
|
|
"mx_AutoHideScrollbar",
|
|
|
|
);
|
|
|
|
|
2020-06-03 01:26:07 +00:00
|
|
|
return (
|
|
|
|
<div className={containerClasses}>
|
|
|
|
{tagPanel}
|
2020-06-04 22:34:04 +00:00
|
|
|
<aside className="mx_LeftPanel2_roomListContainer">
|
|
|
|
{this.renderHeader()}
|
2020-06-09 02:33:21 +00:00
|
|
|
{this.renderSearchExplore()}
|
2020-07-08 12:49:04 +00:00
|
|
|
<div className="mx_LeftPanel2_roomListWrapper">
|
|
|
|
<div
|
|
|
|
className={roomListClasses}
|
|
|
|
onScroll={this.onScroll}
|
|
|
|
ref={this.listContainerRef}
|
|
|
|
// Firefox sometimes makes this element focusable due to
|
|
|
|
// overflow:scroll;, so force it out of tab order.
|
|
|
|
tabIndex={-1}
|
|
|
|
>
|
|
|
|
{roomList}
|
|
|
|
</div>
|
2020-07-02 21:21:10 +00:00
|
|
|
</div>
|
2020-06-03 01:26:07 +00:00
|
|
|
</aside>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|