* Add labs flag for sliding sync; add sliding_sync_proxy_url to config.json * Disable the labs toggle if sliding_sync_proxy_url is not set * Do validation checks on the sliding sync proxy URL before enabling it in Labs * Enable sliding sync and add SlidingSyncManager * Get room subscriptions working * Hijack renderSublists in sliding sync mode * Add support for sorting alphabetically/recency and room name filters * Filter out tombstoned rooms; start adding show more logic list ranges update but the UI doesn't * update the UI when the list is updated * bugfix: make sure the list sorts numerically * Get invites transitioning correctly * Force enable sliding sync and labs for now * Linting * Disable spotlight search * Initial cypress plugins for Sliding Sync Proxy * Use --rm when running Synapse in Docker for Cypress tests * Update src/MatrixClientPeg.ts Co-authored-by: Travis Ralston <travisr@matrix.org> * Update src/components/views/rooms/RoomSublist.tsx Co-authored-by: Travis Ralston <travisr@matrix.org> * Update src/settings/controllers/SlidingSyncController.ts Co-authored-by: Travis Ralston <travisr@matrix.org> * Update src/components/views/rooms/RoomSublist.tsx Co-authored-by: Travis Ralston <travisr@matrix.org> * WIP add room searching to spotlight search * Only read sliding sync results when there is a result, else use the local cache * Use feature_sliding_sync not slidingSync * Some review comments * More review comments * Use RoomViewStore to set room subscriptions * Comment why any * Update src/components/views/rooms/RoomSublist.tsx Co-authored-by: Michael Telatynski <7t3chguy@gmail.com> * Fix cypress docker abstraction * Iterate sliding sync proxy support * Stash mostly functional test * Update sliding sync proxy image * i18n * Add support for spaces; use list ID -> index mappings - Mappings are more reusable and easier to understand than racing for index positions. - Register for all spaces immediately on startup. * When the active space is updated, update the list registration * Set spaces filter in the correct place * Skeleton placeholder whilst loading the space * Filter out spaces from the room list * Use the new txn_id promises * Ensure we actually resolve list registrations * Fix matrix-org/sliding-sync#30: don't show tombstoned search results * Remove unused imports * Add SYNCV3_SECRET to proxy to ensure it starts up; correct aliases for SS test * Add another basic sliding sync e2e test * Unbreak netlify * Add more logging for debugging duplicate rooms * If sliding sync is enabled, always use the rooms result even if it's empty * Drop-in copy of RoomListStore for sliding sync * Remove conditionals from RoomListStore - we have SlidingRoomListStore now * WIP SlidingRoomListStore * Add most sliding sync logic to SlidingRoomListStore Still lots of logic in RoomSublist. Broken things: - Join count is wrong completely. - No skeleton placeholder when switching spaces. * Migrate joined count to SS RLS * Reinstate the skeleton UI when the list is loading * linting * Add support for sticky rooms based on the currently active room * Add a bunch of passing SS E2E tests; some WIP * Unbreak build from git merge * Suppress unread indicators in sliding sync mode * Add regression test for https://github.com/matrix-org/sliding-sync/issues/28 * Add invite test flows; show the invite list The refactor to SS RLS removed the invite list entirely. * Remove show more click as it wasn't the bug * Linting and i18n * only enable SS by default on netlify * Jest fixes; merge conflict fixes; remove debug logging; use right sort enum values * Actually fix jest tests * Add support for favourites and low priority * Bump sliding sync version * Update sliding sync labs to be user configurable * delint * To disable SS or change proxy URL the user has to log out * Review comments * Linting * Apply suggestions from code review Co-authored-by: Travis Ralston <travisr@matrix.org> * Update src/stores/room-list/SlidingRoomListStore.ts Co-authored-by: Travis Ralston <travisr@matrix.org> * Review comments * Add issue link for TODO markers * Linting * Apply suggestions from code review Co-authored-by: Michael Telatynski <7t3chguy@gmail.com> * More review comments * More review comments * stricter types Co-authored-by: Michael Telatynski <7t3chguy@gmail.com> Co-authored-by: Travis Ralston <travisr@matrix.org>
145 lines
4.7 KiB
TypeScript
145 lines
4.7 KiB
TypeScript
/*
|
|
Copyright 2015, 2016 OpenMarket 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, { ChangeEvent, createRef } from 'react';
|
|
|
|
import Field from "../elements/Field";
|
|
import { _t, _td } from '../../../languageHandler';
|
|
import { IFieldState, IValidationResult } from "../elements/Validation";
|
|
import BaseDialog from "./BaseDialog";
|
|
import DialogButtons from "../elements/DialogButtons";
|
|
import { IDialogProps } from "./IDialogProps";
|
|
|
|
interface IProps extends IDialogProps {
|
|
title?: string;
|
|
description?: React.ReactNode;
|
|
value?: string;
|
|
placeholder?: string;
|
|
button?: string;
|
|
busyMessage?: string; // pass _td string
|
|
focus?: boolean;
|
|
hasCancel?: boolean;
|
|
validator?: (fieldState: IFieldState) => Promise<IValidationResult>; // result of withValidation
|
|
fixedWidth?: boolean;
|
|
}
|
|
|
|
interface IState {
|
|
value: string;
|
|
busy: boolean;
|
|
valid: boolean;
|
|
}
|
|
|
|
export default class TextInputDialog extends React.Component<IProps, IState> {
|
|
private field = createRef<Field>();
|
|
|
|
public static defaultProps = {
|
|
title: "",
|
|
value: "",
|
|
description: "",
|
|
busyMessage: _td("Loading..."),
|
|
focus: true,
|
|
hasCancel: true,
|
|
};
|
|
|
|
constructor(props: IProps) {
|
|
super(props);
|
|
|
|
this.state = {
|
|
value: this.props.value,
|
|
busy: false,
|
|
valid: false,
|
|
};
|
|
}
|
|
|
|
public componentDidMount(): void {
|
|
if (this.props.focus) {
|
|
// Set the cursor at the end of the text input
|
|
// this._field.current.value = this.props.value;
|
|
this.field.current.focus();
|
|
}
|
|
}
|
|
|
|
private onOk = async (ev: React.FormEvent): Promise<void> => {
|
|
ev.preventDefault();
|
|
if (this.props.validator) {
|
|
this.setState({ busy: true });
|
|
await this.field.current.validate({ allowEmpty: false });
|
|
|
|
if (!this.field.current.state.valid) {
|
|
this.field.current.focus();
|
|
this.field.current.validate({ allowEmpty: false, focused: true });
|
|
this.setState({ busy: false });
|
|
return;
|
|
}
|
|
}
|
|
this.props.onFinished(true, this.state.value);
|
|
};
|
|
|
|
private onCancel = (): void => {
|
|
this.props.onFinished(false);
|
|
};
|
|
|
|
private onChange = (ev: ChangeEvent<HTMLInputElement>): void => {
|
|
this.setState({
|
|
value: ev.target.value,
|
|
});
|
|
};
|
|
|
|
private onValidate = async (fieldState: IFieldState): Promise<IValidationResult> => {
|
|
const result = await this.props.validator(fieldState);
|
|
this.setState({
|
|
valid: result.valid,
|
|
});
|
|
return result;
|
|
};
|
|
|
|
public render(): JSX.Element {
|
|
return (
|
|
<BaseDialog
|
|
className="mx_TextInputDialog"
|
|
onFinished={this.props.onFinished}
|
|
title={this.props.title}
|
|
fixedWidth={this.props.fixedWidth}
|
|
>
|
|
<form onSubmit={this.onOk}>
|
|
<div className="mx_Dialog_content">
|
|
<div className="mx_TextInputDialog_label">
|
|
<label htmlFor="textinput"> { this.props.description } </label>
|
|
</div>
|
|
<div>
|
|
<Field
|
|
className="mx_TextInputDialog_input"
|
|
ref={this.field}
|
|
type="text"
|
|
label={this.props.placeholder}
|
|
value={this.state.value}
|
|
onChange={this.onChange}
|
|
onValidate={this.props.validator ? this.onValidate : undefined}
|
|
/>
|
|
</div>
|
|
</div>
|
|
</form>
|
|
<DialogButtons
|
|
primaryButton={this.state.busy ? _t(this.props.busyMessage) : this.props.button}
|
|
disabled={this.state.busy}
|
|
onPrimaryButtonClick={this.onOk}
|
|
onCancel={this.onCancel}
|
|
hasCancel={this.props.hasCancel}
|
|
/>
|
|
</BaseDialog>
|
|
);
|
|
}
|
|
}
|