Merge remote-tracking branch 'upstream/develop' into develop

This commit is contained in:
Nigel Mansell 2020-09-05 16:51:13 -04:00
commit 590e2b70f9
216 changed files with 4346 additions and 5445 deletions

View file

@ -1,3 +1,72 @@
Changes in [3.3.0](https://github.com/matrix-org/matrix-react-sdk/releases/tag/v3.3.0) (2020-09-01)
===================================================================================================
[Full Changelog](https://github.com/matrix-org/matrix-react-sdk/compare/v3.3.0-rc.1...v3.3.0)
* Upgrade to JS SDK 8.2.0
Changes in [3.3.0-rc.1](https://github.com/matrix-org/matrix-react-sdk/releases/tag/v3.3.0-rc.1) (2020-08-26)
=============================================================================================================
[Full Changelog](https://github.com/matrix-org/matrix-react-sdk/compare/v3.2.0...v3.3.0-rc.1)
* Upgrade to JS SDK 8.2.0-rc.1
* Update from Weblate
[\#5146](https://github.com/matrix-org/matrix-react-sdk/pull/5146)
* BaseAvatar avoid initial render with default avatar
[\#5142](https://github.com/matrix-org/matrix-react-sdk/pull/5142)
* Enforce Secure Backup completion when requested by HS
[\#5130](https://github.com/matrix-org/matrix-react-sdk/pull/5130)
* Communities v2 prototype: Explore rooms, global state, and default room
[\#5139](https://github.com/matrix-org/matrix-react-sdk/pull/5139)
* Add communities v2 prototyping feature flag + initial tag panel prototypes
[\#5133](https://github.com/matrix-org/matrix-react-sdk/pull/5133)
* Remove some unused components
[\#5134](https://github.com/matrix-org/matrix-react-sdk/pull/5134)
* Allow avatar image view for 1:1 rooms
[\#5137](https://github.com/matrix-org/matrix-react-sdk/pull/5137)
* Send mx_local_settings in rageshake
[\#5136](https://github.com/matrix-org/matrix-react-sdk/pull/5136)
* Run all room leaving behaviour through a single function
[\#5132](https://github.com/matrix-org/matrix-react-sdk/pull/5132)
* Add clarifying comment in media device selection
[\#5131](https://github.com/matrix-org/matrix-react-sdk/pull/5131)
* Settings v3: Feature flag changes
[\#5124](https://github.com/matrix-org/matrix-react-sdk/pull/5124)
* Clear url previews if they all get edited out of the event
[\#5129](https://github.com/matrix-org/matrix-react-sdk/pull/5129)
* Consider tab completions as modifications for editing purposes to unlock
sending
[\#5128](https://github.com/matrix-org/matrix-react-sdk/pull/5128)
* Use matrix-doc for SAS emoji translations
[\#5125](https://github.com/matrix-org/matrix-react-sdk/pull/5125)
* Add a rageshake function to download the logs locally
[\#3849](https://github.com/matrix-org/matrix-react-sdk/pull/3849)
* Room List filtering visual tweaks
[\#5123](https://github.com/matrix-org/matrix-react-sdk/pull/5123)
* Make reply preview not an overlay so you can see new messages
[\#5072](https://github.com/matrix-org/matrix-react-sdk/pull/5072)
* Allow room tile context menu when minimized using right click
[\#5113](https://github.com/matrix-org/matrix-react-sdk/pull/5113)
* Add null guard to group inviter for corrupted groups
[\#5121](https://github.com/matrix-org/matrix-react-sdk/pull/5121)
* Room List styling tweaks
[\#5118](https://github.com/matrix-org/matrix-react-sdk/pull/5118)
* Fix corner rounding on images not always affecting right side
[\#5120](https://github.com/matrix-org/matrix-react-sdk/pull/5120)
* Change add room action for rooms to context menu
[\#5108](https://github.com/matrix-org/matrix-react-sdk/pull/5108)
* Switch out the globe icon and colour it depending on theme
[\#5106](https://github.com/matrix-org/matrix-react-sdk/pull/5106)
* Message Action Bar watch for event send changes
[\#5115](https://github.com/matrix-org/matrix-react-sdk/pull/5115)
* Put message previews for Emoji behind Labs
[\#5110](https://github.com/matrix-org/matrix-react-sdk/pull/5110)
* Fix styling for selected community marker
[\#5107](https://github.com/matrix-org/matrix-react-sdk/pull/5107)
* Fix action bar safe area regression
[\#5111](https://github.com/matrix-org/matrix-react-sdk/pull/5111)
* Fix /op slash command
[\#5109](https://github.com/matrix-org/matrix-react-sdk/pull/5109)
Changes in [3.2.0](https://github.com/matrix-org/matrix-react-sdk/releases/tag/v3.2.0) (2020-08-17) Changes in [3.2.0](https://github.com/matrix-org/matrix-react-sdk/releases/tag/v3.2.0) (2020-08-17)
=================================================================================================== ===================================================================================================
[Full Changelog](https://github.com/matrix-org/matrix-react-sdk/compare/v3.2.0-rc.1...v3.2.0) [Full Changelog](https://github.com/matrix-org/matrix-react-sdk/compare/v3.2.0-rc.1...v3.2.0)

View file

@ -1,6 +1,6 @@
{ {
"name": "matrix-react-sdk", "name": "matrix-react-sdk",
"version": "3.2.0", "version": "3.3.0",
"description": "SDK for matrix.org using React", "description": "SDK for matrix.org using React",
"author": "matrix.org", "author": "matrix.org",
"repository": { "repository": {
@ -61,7 +61,6 @@
"classnames": "^2.2.6", "classnames": "^2.2.6",
"commonmark": "^0.29.1", "commonmark": "^0.29.1",
"counterpart": "^0.18.6", "counterpart": "^0.18.6",
"create-react-class": "^15.6.3",
"diff-dom": "^4.1.6", "diff-dom": "^4.1.6",
"diff-match-patch": "^1.0.5", "diff-match-patch": "^1.0.5",
"emojibase-data": "^5.0.1", "emojibase-data": "^5.0.1",
@ -163,9 +162,7 @@
"stylelint-config-standard": "^18.3.0", "stylelint-config-standard": "^18.3.0",
"stylelint-scss": "^3.18.0", "stylelint-scss": "^3.18.0",
"typescript": "^3.9.7", "typescript": "^3.9.7",
"walk": "^2.3.14", "walk": "^2.3.14"
"webpack": "^4.43.0",
"webpack-cli": "^3.3.12"
}, },
"jest": { "jest": {
"testMatch": [ "testMatch": [

View file

@ -68,6 +68,7 @@
@import "./views/dialogs/_CreateRoomDialog.scss"; @import "./views/dialogs/_CreateRoomDialog.scss";
@import "./views/dialogs/_DeactivateAccountDialog.scss"; @import "./views/dialogs/_DeactivateAccountDialog.scss";
@import "./views/dialogs/_DevtoolsDialog.scss"; @import "./views/dialogs/_DevtoolsDialog.scss";
@import "./views/dialogs/_EditCommunityPrototypeDialog.scss";
@import "./views/dialogs/_GroupAddressPicker.scss"; @import "./views/dialogs/_GroupAddressPicker.scss";
@import "./views/dialogs/_IncomingSasDialog.scss"; @import "./views/dialogs/_IncomingSasDialog.scss";
@import "./views/dialogs/_InviteDialog.scss"; @import "./views/dialogs/_InviteDialog.scss";

View file

@ -16,9 +16,33 @@ limitations under the License.
.mx_UserMenu { .mx_UserMenu {
// to make the ... button sort of aligned with the explore button below // to make the menu button sort of aligned with the explore button below
padding-right: 6px; padding-right: 6px;
&.mx_UserMenu_prototype {
// The margin & padding combination between here and the ::after is to
// align the border line with the tag panel.
margin-bottom: 6px;
padding-right: 0; // make the right edge line up with the explore button
.mx_UserMenu_headerButtons {
// considering we've eliminated right padding on the menu itself, we need to
// push the chevron in slightly (roughly lining up with the center of the
// plus buttons)
margin-right: 2px;
}
// we cheat opacity on the theme colour with an after selector here
&::after {
content: '';
border-bottom: 1px solid $primary-fg-color; // XXX: Variable abuse
opacity: 0.2;
display: block;
padding-top: 8px;
}
}
.mx_UserMenu_headerButtons { .mx_UserMenu_headerButtons {
width: 16px; width: 16px;
height: 16px; height: 16px;
@ -36,7 +60,7 @@ limitations under the License.
mask-size: contain; mask-size: contain;
mask-repeat: no-repeat; mask-repeat: no-repeat;
background: $primary-fg-color; background: $primary-fg-color;
mask-image: url('$(res)/img/element-icons/context-menu.svg'); mask-image: url('$(res)/img/feather-customised/chevron-down.svg');
} }
} }
@ -56,6 +80,28 @@ limitations under the License.
} }
} }
.mx_UserMenu_doubleName {
flex: 1;
min-width: 0; // make flexbox aware that it can crush this to a tiny width
.mx_UserMenu_userName,
.mx_UserMenu_subUserName {
display: block;
}
.mx_UserMenu_subUserName {
color: $muted-fg-color;
font-size: $font-13px;
line-height: $font-18px;
flex: 1;
// Ellipsize any text overflow
text-overflow: ellipsis;
overflow: hidden;
white-space: nowrap;
}
}
.mx_UserMenu_userName { .mx_UserMenu_userName {
font-weight: 600; font-weight: 600;
font-size: $font-15px; font-size: $font-15px;
@ -89,6 +135,44 @@ limitations under the License.
.mx_UserMenu_contextMenu { .mx_UserMenu_contextMenu {
width: 247px; width: 247px;
// These override the styles already present on the user menu rather than try to
// define a new menu. They are specifically for the stacked menu when a community
// is being represented as a prototype.
&.mx_UserMenu_contextMenu_prototype {
padding-bottom: 16px;
.mx_UserMenu_contextMenu_header {
padding-bottom: 0;
padding-top: 16px;
&:nth-child(n + 2) {
padding-top: 8px;
}
}
hr {
width: 85%;
opacity: 0.2;
border: none;
border-bottom: 1px solid $primary-fg-color; // XXX: Variable abuse
}
&.mx_IconizedContextMenu {
> .mx_IconizedContextMenu_optionList {
margin-top: 4px;
&::before {
border: none;
}
> .mx_AccessibleButton {
padding-top: 2px;
padding-bottom: 2px;
}
}
}
}
&.mx_IconizedContextMenu .mx_IconizedContextMenu_optionList_red { &.mx_IconizedContextMenu .mx_IconizedContextMenu_optionList_red {
.mx_AccessibleButton { .mx_AccessibleButton {
padding-top: 16px; padding-top: 16px;
@ -193,4 +277,12 @@ limitations under the License.
.mx_UserMenu_iconSignOut::before { .mx_UserMenu_iconSignOut::before {
mask-image: url('$(res)/img/element-icons/leave.svg'); mask-image: url('$(res)/img/element-icons/leave.svg');
} }
.mx_UserMenu_iconMembers::before {
mask-image: url('$(res)/img/element-icons/room/members.svg');
}
.mx_UserMenu_iconInvite::before {
mask-image: url('$(res)/img/element-icons/room/invite.svg');
}
} }

View file

@ -0,0 +1,77 @@
/*
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.
*/
// XXX: many of these styles are shared with the create dialog
.mx_EditCommunityPrototypeDialog {
&.mx_Dialog_fixedWidth {
width: 360px;
}
.mx_Dialog_content {
margin-bottom: 12px;
.mx_AccessibleButton.mx_AccessibleButton_kind_primary {
display: block;
height: 32px;
font-size: $font-16px;
line-height: 32px;
}
.mx_EditCommunityPrototypeDialog_rowAvatar {
display: flex;
flex-direction: row;
align-items: center;
}
.mx_EditCommunityPrototypeDialog_avatarContainer {
margin-top: 20px;
margin-bottom: 20px;
.mx_EditCommunityPrototypeDialog_avatar,
.mx_EditCommunityPrototypeDialog_placeholderAvatar {
width: 96px;
height: 96px;
border-radius: 96px;
}
.mx_EditCommunityPrototypeDialog_placeholderAvatar {
background-color: #368bd6; // hardcoded for both themes
&::before {
display: inline-block;
background-color: #fff; // hardcoded because the background is
mask-repeat: no-repeat;
mask-size: 96px;
width: 96px;
height: 96px;
mask-position: center;
content: '';
vertical-align: middle;
mask-image: url('$(res)/img/element-icons/add-photo.svg');
}
}
}
.mx_EditCommunityPrototypeDialog_tip {
margin-left: 20px;
& > b, & > span {
display: block;
color: $muted-fg-color;
}
}
}
}

View file

@ -89,6 +89,13 @@ limitations under the License.
font-weight: bold; font-weight: bold;
text-transform: uppercase; text-transform: uppercase;
} }
.mx_InviteDialog_subname {
margin-bottom: 10px;
margin-top: -10px; // HACK: Positioning with margins is bad
font-size: $font-12px;
color: $muted-fg-color;
}
} }
.mx_InviteDialog_roomTile { .mx_InviteDialog_roomTile {
@ -226,3 +233,7 @@ limitations under the License.
.mx_InviteDialog_addressBar { .mx_InviteDialog_addressBar {
margin-right: 45px; margin-right: 45px;
} }
.mx_InviteDialog_helpText .mx_AccessibleButton_kind_link {
padding: 0;
}

View file

@ -1,5 +1,5 @@
/* /*
Copyright 2019 New Vector Ltd Copyright 2019, 2020 New Vector Ltd
Licensed under the Apache License, Version 2.0 (the "License"); Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License. you may not use this file except in compliance with the License.
@ -22,6 +22,7 @@ limitations under the License.
font-size: $font-20px; font-size: $font-20px;
font-weight: 600; font-weight: 600;
color: $primary-fg-color; color: $primary-fg-color;
margin-bottom: 10px;
} }
.mx_SettingsTab_heading:nth-child(n + 2) { .mx_SettingsTab_heading:nth-child(n + 2) {

View file

@ -28,6 +28,7 @@ import SettingsStore from "../settings/SettingsStore";
import {ActiveRoomObserver} from "../ActiveRoomObserver"; import {ActiveRoomObserver} from "../ActiveRoomObserver";
import {Notifier} from "../Notifier"; import {Notifier} from "../Notifier";
import type {Renderer} from "react-dom"; import type {Renderer} from "react-dom";
import RightPanelStore from "../stores/RightPanelStore";
declare global { declare global {
interface Window { interface Window {
@ -49,6 +50,7 @@ declare global {
singletonModalManager: ModalManager; singletonModalManager: ModalManager;
mxSettingsStore: SettingsStore; mxSettingsStore: SettingsStore;
mxNotifier: typeof Notifier; mxNotifier: typeof Notifier;
mxRightPanelStore: RightPanelStore;
} }
interface Document { interface Document {

View file

@ -15,7 +15,7 @@ See the License for the specific language governing permissions and
limitations under the License. limitations under the License.
*/ */
import createReactClass from 'create-react-class'; import React from "react";
import * as sdk from './index'; import * as sdk from './index';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import { _t } from './languageHandler'; import { _t } from './languageHandler';
@ -24,21 +24,19 @@ import { _t } from './languageHandler';
* Wrap an asynchronous loader function with a react component which shows a * Wrap an asynchronous loader function with a react component which shows a
* spinner until the real component loads. * spinner until the real component loads.
*/ */
export default createReactClass({ export default class AsyncWrapper extends React.Component {
propTypes: { static propTypes = {
/** A promise which resolves with the real component /** A promise which resolves with the real component
*/ */
prom: PropTypes.object.isRequired, prom: PropTypes.object.isRequired,
}, };
getInitialState: function() { state = {
return {
component: null, component: null,
error: null, error: null,
}; };
},
componentDidMount: function() { componentDidMount() {
this._unmounted = false; this._unmounted = false;
// XXX: temporary logging to try to diagnose // XXX: temporary logging to try to diagnose
// https://github.com/vector-im/element-web/issues/3148 // https://github.com/vector-im/element-web/issues/3148
@ -56,17 +54,17 @@ export default createReactClass({
console.warn('AsyncWrapper promise failed', e); console.warn('AsyncWrapper promise failed', e);
this.setState({error: e}); this.setState({error: e});
}); });
}, }
componentWillUnmount: function() { componentWillUnmount() {
this._unmounted = true; this._unmounted = true;
}, }
_onWrapperCancelClick: function() { _onWrapperCancelClick = () => {
this.props.onFinished(false); this.props.onFinished(false);
}, };
render: function() { render() {
if (this.state.component) { if (this.state.component) {
const Component = this.state.component; const Component = this.state.component;
return <Component {...this.props} />; return <Component {...this.props} />;
@ -87,6 +85,6 @@ export default createReactClass({
const Spinner = sdk.getComponent("elements.Spinner"); const Spinner = sdk.getComponent("elements.Spinner");
return <Spinner />; return <Spinner />;
} }
}, }
}); }

View file

@ -70,6 +70,7 @@ interface IContent {
interface IThumbnail { interface IThumbnail {
info: { info: {
// eslint-disable-next-line camelcase
thumbnail_info: { thumbnail_info: {
w: number; w: number;
h: number; h: number;
@ -104,7 +105,12 @@ interface IAbortablePromise<T> extends Promise<T> {
* @return {Promise} A promise that resolves with an object with an info key * @return {Promise} A promise that resolves with an object with an info key
* and a thumbnail key. * and a thumbnail key.
*/ */
function createThumbnail(element: ThumbnailableElement, inputWidth: number, inputHeight: number, mimeType: string): Promise<IThumbnail> { function createThumbnail(
element: ThumbnailableElement,
inputWidth: number,
inputHeight: number,
mimeType: string,
): Promise<IThumbnail> {
return new Promise((resolve) => { return new Promise((resolve) => {
let targetWidth = inputWidth; let targetWidth = inputWidth;
let targetHeight = inputHeight; let targetHeight = inputHeight;
@ -437,11 +443,13 @@ export default class ContentMessages {
for (let i = 0; i < okFiles.length; ++i) { for (let i = 0; i < okFiles.length; ++i) {
const file = okFiles[i]; const file = okFiles[i];
if (!uploadAll) { if (!uploadAll) {
const {finished} = Modal.createTrackedDialog<[boolean, boolean]>('Upload Files confirmation', '', UploadConfirmDialog, { const {finished} = Modal.createTrackedDialog<[boolean, boolean]>('Upload Files confirmation',
'', UploadConfirmDialog, {
file, file,
currentIndex: i, currentIndex: i,
totalFiles: okFiles.length, totalFiles: okFiles.length,
}); },
);
const [shouldContinue, shouldUploadAll] = await finished; const [shouldContinue, shouldUploadAll] = await finished;
if (!shouldContinue) break; if (!shouldContinue) break;
if (shouldUploadAll) { if (shouldUploadAll) {

View file

@ -30,7 +30,7 @@ import {
showToast as showUnverifiedSessionsToast, showToast as showUnverifiedSessionsToast,
} from "./toasts/UnverifiedSessionToast"; } from "./toasts/UnverifiedSessionToast";
import { privateShouldBeEncrypted } from "./createRoom"; import { privateShouldBeEncrypted } from "./createRoom";
import { isSecretStorageBeingAccessed, accessSecretStorage } from "./CrossSigningManager"; import { isSecretStorageBeingAccessed, accessSecretStorage } from "./SecurityManager";
import { isSecureBackupRequired } from './utils/WellKnownUtils'; import { isSecureBackupRequired } from './utils/WellKnownUtils';
import { isLoggedIn } from './components/structures/MatrixChat'; import { isLoggedIn } from './components/structures/MatrixChat';
@ -220,7 +220,10 @@ export default class DeviceListener {
await cli.downloadKeys([cli.getUserId()]); await cli.downloadKeys([cli.getUserId()]);
// cross signing isn't enabled - nag to enable it // cross signing isn't enabled - nag to enable it
// There are 3 different toasts for: // There are 3 different toasts for:
if (cli.getStoredCrossSigningForUser(cli.getUserId())) { if (
!cli.getCrossSigningId() &&
cli.getStoredCrossSigningForUser(cli.getUserId())
) {
// Cross-signing on account but this device doesn't trust the master key (verify this session) // Cross-signing on account but this device doesn't trust the master key (verify this session)
showSetupEncryptionToast(SetupKind.VERIFY_THIS_SESSION); showSetupEncryptionToast(SetupKind.VERIFY_THIS_SESSION);
} else { } else {

View file

@ -339,33 +339,9 @@ class HtmlHighlighter extends BaseHighlighter<string> {
} }
} }
class TextHighlighter extends BaseHighlighter<React.ReactNode> {
private key = 0;
/* create a <span> node to hold the given content
*
* snippet: content of the span
* highlight: true to highlight as a search match
*
* returns a React node
*/
protected processSnippet(snippet: string, highlight: boolean): React.ReactNode {
const key = this.key++;
let node = <span key={key} className={highlight ? this.highlightClass : null}>
{ snippet }
</span>;
if (highlight && this.highlightLink) {
node = <a key={key} href={this.highlightLink}>{ node }</a>;
}
return node;
}
}
interface IContent { interface IContent {
format?: string; format?: string;
// eslint-disable-next-line camelcase
formatted_body?: string; formatted_body?: string;
body: string; body: string;
} }
@ -474,8 +450,13 @@ export function bodyToHtml(content: IContent, highlights: string[], opts: IOpts
}); });
return isDisplayedWithHtml ? return isDisplayedWithHtml ?
<span key="body" ref={opts.ref} className={className} dangerouslySetInnerHTML={{ __html: safeBody }} dir="auto" /> : <span
<span key="body" ref={opts.ref} className={className} dir="auto">{ strippedBody }</span>; key="body"
ref={opts.ref}
className={className}
dangerouslySetInnerHTML={{ __html: safeBody }}
dir="auto"
/> : <span key="body" ref={opts.ref} className={className} dir="auto">{ strippedBody }</span>;
} }
/** /**

View file

@ -15,7 +15,7 @@ limitations under the License.
*/ */
import commonmark from 'commonmark'; import commonmark from 'commonmark';
import escape from 'lodash/escape'; import {escape} from "lodash";
const ALLOWED_HTML_TAGS = ['sub', 'sup', 'del', 'u']; const ALLOWED_HTML_TAGS = ['sub', 'sup', 'del', 'u'];

View file

@ -31,7 +31,7 @@ import {verificationMethods} from 'matrix-js-sdk/src/crypto';
import MatrixClientBackedSettingsHandler from "./settings/handlers/MatrixClientBackedSettingsHandler"; import MatrixClientBackedSettingsHandler from "./settings/handlers/MatrixClientBackedSettingsHandler";
import * as StorageManager from './utils/StorageManager'; import * as StorageManager from './utils/StorageManager';
import IdentityAuthClient from './IdentityAuthClient'; import IdentityAuthClient from './IdentityAuthClient';
import { crossSigningCallbacks } from './CrossSigningManager'; import { crossSigningCallbacks } from './SecurityManager';
import {SHOW_QR_CODE_METHOD} from "matrix-js-sdk/src/crypto/verification/QRCode"; import {SHOW_QR_CODE_METHOD} from "matrix-js-sdk/src/crypto/verification/QRCode";
export interface IMatrixClientCreds { export interface IMatrixClientCreds {

View file

@ -151,7 +151,7 @@ export class ModalManager {
prom: Promise<React.ComponentType>, prom: Promise<React.ComponentType>,
props?: IProps<T>, props?: IProps<T>,
className?: string, className?: string,
options?: IOptions<T> options?: IOptions<T>,
) { ) {
const modal: IModal<T> = { const modal: IModal<T> = {
onFinished: props ? props.onFinished : null, onFinished: props ? props.onFinished : null,
@ -182,7 +182,7 @@ export class ModalManager {
private getCloseFn<T extends any[]>( private getCloseFn<T extends any[]>(
modal: IModal<T>, modal: IModal<T>,
props: IProps<T> props: IProps<T>,
): [IHandle<T>["close"], IHandle<T>["finished"]] { ): [IHandle<T>["close"], IHandle<T>["finished"]] {
const deferred = defer<T>(); const deferred = defer<T>();
return [async (...args: T) => { return [async (...args: T) => {
@ -264,7 +264,7 @@ export class ModalManager {
className?: string, className?: string,
isPriorityModal = false, isPriorityModal = false,
isStaticModal = false, isStaticModal = false,
options: IOptions<T> = {} options: IOptions<T> = {},
): IHandle<T> { ): IHandle<T> {
const {modal, closeDialog, onFinishedProm} = this.buildModal<T>(prom, props, className, options); const {modal, closeDialog, onFinishedProm} = this.buildModal<T>(prom, props, className, options);
if (isPriorityModal) { if (isPriorityModal) {
@ -287,7 +287,7 @@ export class ModalManager {
private appendDialogAsync<T extends any[]>( private appendDialogAsync<T extends any[]>(
prom: Promise<React.ComponentType>, prom: Promise<React.ComponentType>,
props?: IProps<T>, props?: IProps<T>,
className?: string className?: string,
): IHandle<T> { ): IHandle<T> {
const {modal, closeDialog, onFinishedProm} = this.buildModal<T>(prom, props, className, {}); const {modal, closeDialog, onFinishedProm} = this.buildModal<T>(prom, props, className, {});

View file

@ -24,6 +24,7 @@ import * as sdk from './';
import { _t } from './languageHandler'; import { _t } from './languageHandler';
import {KIND_DM, KIND_INVITE} from "./components/views/dialogs/InviteDialog"; import {KIND_DM, KIND_INVITE} from "./components/views/dialogs/InviteDialog";
import CommunityPrototypeInviteDialog from "./components/views/dialogs/CommunityPrototypeInviteDialog"; import CommunityPrototypeInviteDialog from "./components/views/dialogs/CommunityPrototypeInviteDialog";
import {CommunityPrototypeStore} from "./stores/CommunityPrototypeStore";
/** /**
* Invites multiple addresses to a room * Invites multiple addresses to a room
@ -64,6 +65,16 @@ export function showCommunityRoomInviteDialog(roomId, communityName) {
); );
} }
export function showCommunityInviteDialog(communityId) {
const chat = CommunityPrototypeStore.instance.getGeneralChat(communityId);
if (chat) {
const name = CommunityPrototypeStore.instance.getCommunityName(communityId);
showCommunityRoomInviteDialog(chat.roomId, name);
} else {
throw new Error("Failed to locate appropriate room to start an invite in");
}
}
/** /**
* Checks if the given MatrixEvent is a valid 3rd party user invite. * Checks if the given MatrixEvent is a valid 3rd party user invite.
* @param {MatrixEvent} event The event to check * @param {MatrixEvent} event The event to check

View file

@ -1,5 +1,5 @@
/* /*
Copyright 2019 The Matrix.org Foundation C.I.C. Copyright 2019, 2020 The Matrix.org Foundation C.I.C.
Licensed under the Apache License, Version 2.0 (the "License"); Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License. you may not use this file except in compliance with the License.
@ -142,7 +142,7 @@ const onSecretRequested = async function({
return; return;
} }
if (!deviceTrust || !deviceTrust.isVerified()) { if (!deviceTrust || !deviceTrust.isVerified()) {
console.log(`CrossSigningManager: Ignoring request from untrusted device ${deviceId}`); console.log(`Ignoring secret request from untrusted device ${deviceId}`);
return; return;
} }
if ( if (

View file

@ -15,7 +15,7 @@ See the License for the specific language governing permissions and
limitations under the License. limitations under the License.
*/ */
import _clamp from 'lodash/clamp'; import {clamp} from "lodash";
export default class SendHistoryManager { export default class SendHistoryManager {
history: Array<HistoryItem> = []; history: Array<HistoryItem> = [];
@ -54,7 +54,7 @@ export default class SendHistoryManager {
} }
getItem(offset: number): ?HistoryItem { getItem(offset: number): ?HistoryItem {
this.currentIndex = _clamp(this.currentIndex + offset, 0, this.history.length - 1); this.currentIndex = clamp(this.currentIndex + offset, 0, this.history.length - 1);
return this.history[this.currentIndex]; return this.history[this.currentIndex];
} }
} }

View file

@ -168,7 +168,7 @@ const shortcuts: Record<Categories, IShortcut[]> = {
key: Key.U, key: Key.U,
}], }],
description: _td("Upload a file"), description: _td("Upload a file"),
} },
], ],
[Categories.ROOM_LIST]: [ [Categories.ROOM_LIST]: [

View file

@ -190,7 +190,7 @@ export const RovingTabIndexProvider: React.FC<IProps> = ({children, handleHomeEn
ev.preventDefault(); ev.preventDefault();
ev.stopPropagation(); ev.stopPropagation();
} else if (onKeyDown) { } else if (onKeyDown) {
return onKeyDown(ev, state); return onKeyDown(ev, context.state);
} }
}, [context.state, onKeyDown, handleHomeEnd]); }, [context.state, onKeyDown, handleHomeEnd]);

View file

@ -30,6 +30,7 @@ const Toolbar: React.FC<IProps> = ({children, ...props}) => {
const target = ev.target as HTMLElement; const target = ev.target as HTMLElement;
let handled = true; let handled = true;
// HOME and END are handled by RovingTabIndexProvider
switch (ev.key) { switch (ev.key) {
case Key.ARROW_UP: case Key.ARROW_UP:
case Key.ARROW_DOWN: case Key.ARROW_DOWN:
@ -47,8 +48,6 @@ const Toolbar: React.FC<IProps> = ({children, ...props}) => {
} }
break; break;
// HOME and END are handled by RovingTabIndexProvider
default: default:
handled = false; handled = false;
} }

View file

@ -26,8 +26,9 @@ interface IProps extends React.ComponentProps<typeof AccessibleButton> {
// Semantic component for representing a role=menuitem // Semantic component for representing a role=menuitem
export const MenuItem: React.FC<IProps> = ({children, label, ...props}) => { export const MenuItem: React.FC<IProps> = ({children, label, ...props}) => {
const ariaLabel = props["aria-label"] || label;
return ( return (
<AccessibleButton {...props} role="menuitem" tabIndex={-1} aria-label={label}> <AccessibleButton {...props} role="menuitem" tabIndex={-1} aria-label={ariaLabel}>
{ children } { children }
</AccessibleButton> </AccessibleButton>
); );

View file

@ -20,7 +20,8 @@ import AccessibleTooltipButton from "../../components/views/elements/AccessibleT
import {useRovingTabIndex} from "../RovingTabIndex"; import {useRovingTabIndex} from "../RovingTabIndex";
import {Ref} from "./types"; import {Ref} from "./types";
interface IProps extends Omit<React.ComponentProps<typeof AccessibleTooltipButton>, "onFocus" | "inputRef" | "tabIndex"> { type ATBProps = React.ComponentProps<typeof AccessibleTooltipButton>;
interface IProps extends Omit<ATBProps, "onFocus" | "inputRef" | "tabIndex"> {
inputRef?: Ref; inputRef?: Ref;
} }

View file

@ -16,7 +16,6 @@ limitations under the License.
import React from "react"; import React from "react";
import AccessibleButton from "../../components/views/elements/AccessibleButton";
import {useRovingTabIndex} from "../RovingTabIndex"; import {useRovingTabIndex} from "../RovingTabIndex";
import {FocusHandler, Ref} from "./types"; import {FocusHandler, Ref} from "./types";

View file

@ -17,7 +17,6 @@ limitations under the License.
import FileSaver from 'file-saver'; import FileSaver from 'file-saver';
import React, {createRef} from 'react'; import React, {createRef} from 'react';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import createReactClass from 'create-react-class';
import { _t } from '../../../languageHandler'; import { _t } from '../../../languageHandler';
import { MatrixClient } from 'matrix-js-sdk'; import { MatrixClient } from 'matrix-js-sdk';
@ -27,34 +26,31 @@ import * as sdk from '../../../index';
const PHASE_EDIT = 1; const PHASE_EDIT = 1;
const PHASE_EXPORTING = 2; const PHASE_EXPORTING = 2;
export default createReactClass({ export default class ExportE2eKeysDialog extends React.Component {
displayName: 'ExportE2eKeysDialog', static propTypes = {
propTypes: {
matrixClient: PropTypes.instanceOf(MatrixClient).isRequired, matrixClient: PropTypes.instanceOf(MatrixClient).isRequired,
onFinished: PropTypes.func.isRequired, onFinished: PropTypes.func.isRequired,
},
getInitialState: function() {
return {
phase: PHASE_EDIT,
errStr: null,
}; };
},
// TODO: [REACT-WARNING] Replace component with real class, use constructor for refs constructor(props) {
UNSAFE_componentWillMount: function() { super(props);
this._unmounted = false; this._unmounted = false;
this._passphrase1 = createRef(); this._passphrase1 = createRef();
this._passphrase2 = createRef(); this._passphrase2 = createRef();
},
componentWillUnmount: function() { this.state = {
phase: PHASE_EDIT,
errStr: null,
};
}
componentWillUnmount() {
this._unmounted = true; this._unmounted = true;
}, }
_onPassphraseFormSubmit: function(ev) { _onPassphraseFormSubmit = (ev) => {
ev.preventDefault(); ev.preventDefault();
const passphrase = this._passphrase1.current.value; const passphrase = this._passphrase1.current.value;
@ -69,9 +65,9 @@ export default createReactClass({
this._startExport(passphrase); this._startExport(passphrase);
return false; return false;
}, };
_startExport: function(passphrase) { _startExport(passphrase) {
// extra Promise.resolve() to turn synchronous exceptions into // extra Promise.resolve() to turn synchronous exceptions into
// asynchronous ones. // asynchronous ones.
Promise.resolve().then(() => { Promise.resolve().then(() => {
@ -102,15 +98,15 @@ export default createReactClass({
errStr: null, errStr: null,
phase: PHASE_EXPORTING, phase: PHASE_EXPORTING,
}); });
}, }
_onCancelClick: function(ev) { _onCancelClick = (ev) => {
ev.preventDefault(); ev.preventDefault();
this.props.onFinished(false); this.props.onFinished(false);
return false; return false;
}, };
render: function() { render() {
const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog'); const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog');
const disableForm = (this.state.phase === PHASE_EXPORTING); const disableForm = (this.state.phase === PHASE_EXPORTING);
@ -184,5 +180,5 @@ export default createReactClass({
</form> </form>
</BaseDialog> </BaseDialog>
); );
}, }
}); }

View file

@ -16,7 +16,6 @@ limitations under the License.
import React, {createRef} from 'react'; import React, {createRef} from 'react';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import createReactClass from 'create-react-class';
import { MatrixClient } from 'matrix-js-sdk'; import { MatrixClient } from 'matrix-js-sdk';
import * as MegolmExportEncryption from '../../../utils/MegolmExportEncryption'; import * as MegolmExportEncryption from '../../../utils/MegolmExportEncryption';
@ -38,48 +37,45 @@ function readFileAsArrayBuffer(file) {
const PHASE_EDIT = 1; const PHASE_EDIT = 1;
const PHASE_IMPORTING = 2; const PHASE_IMPORTING = 2;
export default createReactClass({ export default class ImportE2eKeysDialog extends React.Component {
displayName: 'ImportE2eKeysDialog', static propTypes = {
propTypes: {
matrixClient: PropTypes.instanceOf(MatrixClient).isRequired, matrixClient: PropTypes.instanceOf(MatrixClient).isRequired,
onFinished: PropTypes.func.isRequired, onFinished: PropTypes.func.isRequired,
},
getInitialState: function() {
return {
enableSubmit: false,
phase: PHASE_EDIT,
errStr: null,
}; };
},
// TODO: [REACT-WARNING] Replace component with real class, use constructor for refs constructor(props) {
UNSAFE_componentWillMount: function() { super(props);
this._unmounted = false; this._unmounted = false;
this._file = createRef(); this._file = createRef();
this._passphrase = createRef(); this._passphrase = createRef();
},
componentWillUnmount: function() { this.state = {
enableSubmit: false,
phase: PHASE_EDIT,
errStr: null,
};
}
componentWillUnmount() {
this._unmounted = true; this._unmounted = true;
}, }
_onFormChange: function(ev) { _onFormChange = (ev) => {
const files = this._file.current.files || []; const files = this._file.current.files || [];
this.setState({ this.setState({
enableSubmit: (this._passphrase.current.value !== "" && files.length > 0), enableSubmit: (this._passphrase.current.value !== "" && files.length > 0),
}); });
}, };
_onFormSubmit: function(ev) { _onFormSubmit = (ev) => {
ev.preventDefault(); ev.preventDefault();
this._startImport(this._file.current.files[0], this._passphrase.current.value); this._startImport(this._file.current.files[0], this._passphrase.current.value);
return false; return false;
}, };
_startImport: function(file, passphrase) { _startImport(file, passphrase) {
this.setState({ this.setState({
errStr: null, errStr: null,
phase: PHASE_IMPORTING, phase: PHASE_IMPORTING,
@ -105,15 +101,15 @@ export default createReactClass({
phase: PHASE_EDIT, phase: PHASE_EDIT,
}); });
}); });
}, }
_onCancelClick: function(ev) { _onCancelClick = (ev) => {
ev.preventDefault(); ev.preventDefault();
this.props.onFinished(false); this.props.onFinished(false);
return false; return false;
}, };
render: function() { render() {
const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog'); const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog');
const disableForm = (this.state.phase !== PHASE_EDIT); const disableForm = (this.state.phase !== PHASE_EDIT);
@ -188,5 +184,5 @@ export default createReactClass({
</form> </form>
</BaseDialog> </BaseDialog>
); );
}, }
}); }

View file

@ -21,7 +21,7 @@ import * as sdk from '../../../../index';
import {MatrixClientPeg} from '../../../../MatrixClientPeg'; import {MatrixClientPeg} from '../../../../MatrixClientPeg';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import {_t, _td} from '../../../../languageHandler'; import {_t, _td} from '../../../../languageHandler';
import { accessSecretStorage } from '../../../../CrossSigningManager'; import { accessSecretStorage } from '../../../../SecurityManager';
import AccessibleButton from "../../../../components/views/elements/AccessibleButton"; import AccessibleButton from "../../../../components/views/elements/AccessibleButton";
import {copyNode} from "../../../../utils/strings"; import {copyNode} from "../../../../utils/strings";
import PassphraseField from "../../../../components/views/auth/PassphraseField"; import PassphraseField from "../../../../components/views/auth/PassphraseField";

View file

@ -22,7 +22,7 @@ import {MatrixClientPeg} from '../../../../MatrixClientPeg';
import FileSaver from 'file-saver'; import FileSaver from 'file-saver';
import {_t, _td} from '../../../../languageHandler'; import {_t, _td} from '../../../../languageHandler';
import Modal from '../../../../Modal'; import Modal from '../../../../Modal';
import { promptForBackupPassphrase } from '../../../../CrossSigningManager'; import { promptForBackupPassphrase } from '../../../../SecurityManager';
import {copyNode} from "../../../../utils/strings"; import {copyNode} from "../../../../utils/strings";
import {SSOAuthEntry} from "../../../../components/views/auth/InteractiveAuthEntryComponents"; import {SSOAuthEntry} from "../../../../components/views/auth/InteractiveAuthEntryComponents";
import PassphraseField from "../../../../components/views/auth/PassphraseField"; import PassphraseField from "../../../../components/views/auth/PassphraseField";

View file

@ -89,7 +89,11 @@ export default class CommandProvider extends AutocompleteProvider {
renderCompletions(completions: React.ReactNode[]): React.ReactNode { renderCompletions(completions: React.ReactNode[]): React.ReactNode {
return ( return (
<div className="mx_Autocomplete_Completion_container_block" role="listbox" aria-label={_t("Command Autocomplete")}> <div
className="mx_Autocomplete_Completion_container_block"
role="listbox"
aria-label={_t("Command Autocomplete")}
>
{ completions } { completions }
</div> </div>
); );

View file

@ -23,7 +23,7 @@ import {MatrixClientPeg} from '../MatrixClientPeg';
import QueryMatcher from './QueryMatcher'; import QueryMatcher from './QueryMatcher';
import {PillCompletion} from './Components'; import {PillCompletion} from './Components';
import * as sdk from '../index'; import * as sdk from '../index';
import _sortBy from 'lodash/sortBy'; import {sortBy} from "lodash";
import {makeGroupPermalink} from "../utils/permalinks/Permalinks"; import {makeGroupPermalink} from "../utils/permalinks/Permalinks";
import {ICompletion, ISelectionRange} from "./Autocompleter"; import {ICompletion, ISelectionRange} from "./Autocompleter";
import FlairStore from "../stores/FlairStore"; import FlairStore from "../stores/FlairStore";
@ -81,7 +81,7 @@ export default class CommunityProvider extends AutocompleteProvider {
const matchedString = command[0]; const matchedString = command[0];
completions = this.matcher.match(matchedString); completions = this.matcher.match(matchedString);
completions = _sortBy(completions, [ completions = sortBy(completions, [
(c) => score(matchedString, c.groupId), (c) => score(matchedString, c.groupId),
(c) => c.groupId.length, (c) => c.groupId.length,
]).map(({avatarUrl, groupId, name}) => ({ ]).map(({avatarUrl, groupId, name}) => ({
@ -91,15 +91,15 @@ export default class CommunityProvider extends AutocompleteProvider {
href: makeGroupPermalink(groupId), href: makeGroupPermalink(groupId),
component: ( component: (
<PillCompletion title={name} description={groupId}> <PillCompletion title={name} description={groupId}>
<BaseAvatar name={name || groupId} <BaseAvatar
name={name || groupId}
width={24} width={24}
height={24} height={24}
url={avatarUrl ? cli.mxcUrlToHttp(avatarUrl, 24, 24) : null} /> url={avatarUrl ? cli.mxcUrlToHttp(avatarUrl, 24, 24) : null} />
</PillCompletion> </PillCompletion>
), ),
range, range,
})) })).slice(0, 4);
.slice(0, 4);
} }
return completions; return completions;
} }

View file

@ -23,8 +23,7 @@ import AutocompleteProvider from './AutocompleteProvider';
import QueryMatcher from './QueryMatcher'; import QueryMatcher from './QueryMatcher';
import {PillCompletion} from './Components'; import {PillCompletion} from './Components';
import {ICompletion, ISelectionRange} from './Autocompleter'; import {ICompletion, ISelectionRange} from './Autocompleter';
import _uniq from 'lodash/uniq'; import {uniq, sortBy} from 'lodash';
import _sortBy from 'lodash/sortBy';
import SettingsStore from "../settings/SettingsStore"; import SettingsStore from "../settings/SettingsStore";
import { shortcodeToUnicode } from '../HtmlUtils'; import { shortcodeToUnicode } from '../HtmlUtils';
import { EMOJI, IEmoji } from '../emoji'; import { EMOJI, IEmoji } from '../emoji';
@ -115,7 +114,7 @@ export default class EmojiProvider extends AutocompleteProvider {
} }
// Finally, sort by original ordering // Finally, sort by original ordering
sorters.push((c) => c._orderBy); sorters.push((c) => c._orderBy);
completions = _sortBy(_uniq(completions), sorters); completions = sortBy(uniq(completions), sorters);
completions = completions.map(({shortname}) => { completions = completions.map(({shortname}) => {
const unicode = shortcodeToUnicode(shortname); const unicode = shortcodeToUnicode(shortname);
@ -139,7 +138,11 @@ export default class EmojiProvider extends AutocompleteProvider {
renderCompletions(completions: React.ReactNode[]): React.ReactNode { renderCompletions(completions: React.ReactNode[]): React.ReactNode {
return ( return (
<div className="mx_Autocomplete_Completion_container_pill" role="listbox" aria-label={_t("Emoji Autocomplete")}> <div
className="mx_Autocomplete_Completion_container_pill"
role="listbox"
aria-label={_t("Emoji Autocomplete")}
>
{ completions } { completions }
</div> </div>
); );

View file

@ -16,8 +16,7 @@ See the License for the specific language governing permissions and
limitations under the License. limitations under the License.
*/ */
import _at from 'lodash/at'; import {at, uniq} from 'lodash';
import _uniq from 'lodash/uniq';
import {removeHiddenChars} from "matrix-js-sdk/src/utils"; import {removeHiddenChars} from "matrix-js-sdk/src/utils";
interface IOptions<T extends {}> { interface IOptions<T extends {}> {
@ -73,7 +72,7 @@ export default class QueryMatcher<T extends Object> {
// type for their values. We assume that those values who's keys have // type for their values. We assume that those values who's keys have
// been specified will be string. Also, we cannot infer all the // been specified will be string. Also, we cannot infer all the
// types of the keys of the objects at compile. // types of the keys of the objects at compile.
const keyValues = _at<string>(<any>object, this._options.keys); const keyValues = at<string>(<any>object, this._options.keys);
if (this._options.funcs) { if (this._options.funcs) {
for (const f of this._options.funcs) { for (const f of this._options.funcs) {
@ -137,7 +136,7 @@ export default class QueryMatcher<T extends Object> {
}); });
// Now map the keys to the result objects. Also remove any duplicates. // Now map the keys to the result objects. Also remove any duplicates.
return _uniq(matches.map((match) => match.object)); return uniq(matches.map((match) => match.object));
} }
private processQuery(query: string): string { private processQuery(query: string): string {

View file

@ -27,7 +27,7 @@ import {PillCompletion} from './Components';
import * as sdk from '../index'; import * as sdk from '../index';
import {makeRoomPermalink} from "../utils/permalinks/Permalinks"; import {makeRoomPermalink} from "../utils/permalinks/Permalinks";
import {ICompletion, ISelectionRange} from "./Autocompleter"; import {ICompletion, ISelectionRange} from "./Autocompleter";
import { uniqBy, sortBy } from 'lodash'; import {uniqBy, sortBy} from "lodash";
const ROOM_REGEX = /\B#\S*/g; const ROOM_REGEX = /\B#\S*/g;
@ -110,9 +110,7 @@ export default class RoomProvider extends AutocompleteProvider {
), ),
range, range,
}; };
}) }).filter((completion) => !!completion.completion && completion.completion.length > 0).slice(0, 4);
.filter((completion) => !!completion.completion && completion.completion.length > 0)
.slice(0, 4);
} }
return completions; return completions;
} }

View file

@ -23,7 +23,7 @@ import AutocompleteProvider from './AutocompleteProvider';
import {PillCompletion} from './Components'; import {PillCompletion} from './Components';
import * as sdk from '../index'; import * as sdk from '../index';
import QueryMatcher from './QueryMatcher'; import QueryMatcher from './QueryMatcher';
import _sortBy from 'lodash/sortBy'; import {sortBy} from 'lodash';
import {MatrixClientPeg} from '../MatrixClientPeg'; import {MatrixClientPeg} from '../MatrixClientPeg';
import MatrixEvent from "matrix-js-sdk/src/models/event"; import MatrixEvent from "matrix-js-sdk/src/models/event";
@ -71,8 +71,13 @@ export default class UserProvider extends AutocompleteProvider {
} }
} }
private onRoomTimeline = (ev: MatrixEvent, room: Room, toStartOfTimeline: boolean, removed: boolean, private onRoomTimeline = (
data: IRoomTimelineData) => { ev: MatrixEvent,
room: Room,
toStartOfTimeline: boolean,
removed: boolean,
data: IRoomTimelineData,
) => {
if (!room) return; if (!room) return;
if (removed) return; if (removed) return;
if (room.roomId !== this.room.roomId) return; if (room.roomId !== this.room.roomId) return;
@ -151,7 +156,7 @@ export default class UserProvider extends AutocompleteProvider {
const currentUserId = MatrixClientPeg.get().credentials.userId; const currentUserId = MatrixClientPeg.get().credentials.userId;
this.users = this.room.getJoinedMembers().filter(({userId}) => userId !== currentUserId); this.users = this.room.getJoinedMembers().filter(({userId}) => userId !== currentUserId);
this.users = _sortBy(this.users, (member) => 1E20 - lastSpoken[member.userId] || 1E20); this.users = sortBy(this.users, (member) => 1E20 - lastSpoken[member.userId] || 1E20);
this.matcher.setObjects(this.users); this.matcher.setObjects(this.users);
} }
@ -171,7 +176,11 @@ export default class UserProvider extends AutocompleteProvider {
renderCompletions(completions: React.ReactNode[]): React.ReactNode { renderCompletions(completions: React.ReactNode[]): React.ReactNode {
return ( return (
<div className="mx_Autocomplete_Completion_container_pill" role="listbox" aria-label={_t("User Autocomplete")}> <div
className="mx_Autocomplete_Completion_container_pill"
role="listbox"
aria-label={_t("User Autocomplete")}
>
{ completions } { completions }
</div> </div>
); );

View file

@ -1,90 +0,0 @@
/*
Copyright 2015, 2016 OpenMarket Ltd
Copyright 2019 Michael Telatynski <7t3chguy@gmail.com>
Copyright 2019, 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 React from 'react';
import createReactClass from 'create-react-class';
import PropTypes from 'prop-types';
import { _t } from '../../languageHandler';
import SdkConfig from '../../SdkConfig';
export default createReactClass({
displayName: 'CompatibilityPage',
propTypes: {
onAccept: PropTypes.func,
},
getDefaultProps: function() {
return {
onAccept: function() {}, // NOP
};
},
onAccept: function() {
this.props.onAccept();
},
render: function() {
const brand = SdkConfig.get().brand;
return (
<div className="mx_CompatibilityPage">
<div className="mx_CompatibilityPage_box">
<p>{_t(
"Sorry, your browser is <b>not</b> able to run %(brand)s.",
{
brand,
},
{
'b': (sub) => <b>{sub}</b>,
})
}</p>
<p>
{ _t(
"%(brand)s uses many advanced browser features, some of which are not available " +
"or experimental in your current browser.",
{ brand },
) }
</p>
<p>
{ _t(
'Please install <chromeLink>Chrome</chromeLink>, <firefoxLink>Firefox</firefoxLink>, ' +
'or <safariLink>Safari</safariLink> for the best experience.',
{},
{
'chromeLink': (sub) => <a href="https://www.google.com/chrome">{sub}</a>,
'firefoxLink': (sub) => <a href="https://firefox.com">{sub}</a>,
'safariLink': (sub) => <a href="https://apple.com/safari">{sub}</a>,
},
)}
</p>
<p>
{ _t(
"With your current browser, the look and feel of the application may be " +
"completely incorrect, and some or all features may not function. " +
"If you want to try it anyway you can continue, but you are on your own in terms " +
"of any issues you may encounter!",
) }
</p>
<button onClick={this.onAccept}>
{ _t("I understand the risks and wish to continue") }
</button>
</div>
</div>
);
},
});

View file

@ -233,8 +233,7 @@ export class ContextMenu extends React.PureComponent<IProps, IState> {
switch (ev.key) { switch (ev.key) {
case Key.TAB: case Key.TAB:
case Key.ESCAPE: case Key.ESCAPE:
// close on left and right arrows too for when it is a context menu on a <Toolbar /> case Key.ARROW_LEFT: // close on left and right arrows too for when it is a context menu on a <Toolbar />
case Key.ARROW_LEFT:
case Key.ARROW_RIGHT: case Key.ARROW_RIGHT:
this.props.onFinished(); this.props.onFinished();
break; break;

View file

@ -43,8 +43,8 @@ export default class EmbeddedPage extends React.PureComponent {
static contextType = MatrixClientContext; static contextType = MatrixClientContext;
constructor(props) { constructor(props, context) {
super(props); super(props, context);
this._dispatcherRef = null; this._dispatcherRef = null;

View file

@ -16,7 +16,6 @@ limitations under the License.
*/ */
import React from 'react'; import React from 'react';
import createReactClass from 'create-react-class';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import {Filter} from 'matrix-js-sdk'; import {Filter} from 'matrix-js-sdk';
@ -28,23 +27,20 @@ import { _t } from '../../languageHandler';
/* /*
* Component which shows the filtered file using a TimelinePanel * Component which shows the filtered file using a TimelinePanel
*/ */
const FilePanel = createReactClass({ class FilePanel extends React.Component {
displayName: 'FilePanel', static propTypes = {
roomId: PropTypes.string.isRequired,
};
// This is used to track if a decrypted event was a live event and should be // This is used to track if a decrypted event was a live event and should be
// added to the timeline. // added to the timeline.
decryptingEvents: new Set(), decryptingEvents = new Set();
propTypes: { state = {
roomId: PropTypes.string.isRequired,
},
getInitialState: function() {
return {
timelineSet: null, timelineSet: null,
}; };
},
onRoomTimeline(ev, room, toStartOfTimeline, removed, data) { onRoomTimeline = (ev, room, toStartOfTimeline, removed, data) => {
if (room.roomId !== this.props.roomId) return; if (room.roomId !== this.props.roomId) return;
if (toStartOfTimeline || !data || !data.liveEvent || ev.isRedacted()) return; if (toStartOfTimeline || !data || !data.liveEvent || ev.isRedacted()) return;
@ -53,9 +49,9 @@ const FilePanel = createReactClass({
} else { } else {
this.addEncryptedLiveEvent(ev); this.addEncryptedLiveEvent(ev);
} }
}, };
onEventDecrypted(ev, err) { onEventDecrypted = (ev, err) => {
if (ev.getRoomId() !== this.props.roomId) return; if (ev.getRoomId() !== this.props.roomId) return;
const eventId = ev.getId(); const eventId = ev.getId();
@ -63,7 +59,7 @@ const FilePanel = createReactClass({
if (err) return; if (err) return;
this.addEncryptedLiveEvent(ev); this.addEncryptedLiveEvent(ev);
}, };
addEncryptedLiveEvent(ev, toStartOfTimeline) { addEncryptedLiveEvent(ev, toStartOfTimeline) {
if (!this.state.timelineSet) return; if (!this.state.timelineSet) return;
@ -77,7 +73,7 @@ const FilePanel = createReactClass({
if (!this.state.timelineSet.eventIdToTimeline(ev.getId())) { if (!this.state.timelineSet.eventIdToTimeline(ev.getId())) {
this.state.timelineSet.addEventToTimeline(ev, timeline, false); this.state.timelineSet.addEventToTimeline(ev, timeline, false);
} }
}, }
async componentDidMount() { async componentDidMount() {
const client = MatrixClientPeg.get(); const client = MatrixClientPeg.get();
@ -98,7 +94,7 @@ const FilePanel = createReactClass({
client.on('Room.timeline', this.onRoomTimeline); client.on('Room.timeline', this.onRoomTimeline);
client.on('Event.decrypted', this.onEventDecrypted); client.on('Event.decrypted', this.onEventDecrypted);
} }
}, }
componentWillUnmount() { componentWillUnmount() {
const client = MatrixClientPeg.get(); const client = MatrixClientPeg.get();
@ -110,7 +106,7 @@ const FilePanel = createReactClass({
client.removeListener('Room.timeline', this.onRoomTimeline); client.removeListener('Room.timeline', this.onRoomTimeline);
client.removeListener('Event.decrypted', this.onEventDecrypted); client.removeListener('Event.decrypted', this.onEventDecrypted);
} }
}, }
async fetchFileEventsServer(room) { async fetchFileEventsServer(room) {
const client = MatrixClientPeg.get(); const client = MatrixClientPeg.get();
@ -134,9 +130,9 @@ const FilePanel = createReactClass({
const timelineSet = room.getOrCreateFilteredTimelineSet(filter); const timelineSet = room.getOrCreateFilteredTimelineSet(filter);
return timelineSet; return timelineSet;
}, }
onPaginationRequest(timelineWindow, direction, limit) { onPaginationRequest = (timelineWindow, direction, limit) => {
const client = MatrixClientPeg.get(); const client = MatrixClientPeg.get();
const eventIndex = EventIndexPeg.get(); const eventIndex = EventIndexPeg.get();
const roomId = this.props.roomId; const roomId = this.props.roomId;
@ -152,7 +148,7 @@ const FilePanel = createReactClass({
} else { } else {
return timelineWindow.paginate(direction, limit); return timelineWindow.paginate(direction, limit);
} }
}, };
async updateTimelineSet(roomId: string) { async updateTimelineSet(roomId: string) {
const client = MatrixClientPeg.get(); const client = MatrixClientPeg.get();
@ -188,9 +184,9 @@ const FilePanel = createReactClass({
} else { } else {
console.error("Failed to add filtered timelineSet for FilePanel as no room!"); console.error("Failed to add filtered timelineSet for FilePanel as no room!");
} }
}, }
render: function() { render() {
if (MatrixClientPeg.get().isGuest()) { if (MatrixClientPeg.get().isGuest()) {
return <div className="mx_FilePanel mx_RoomView_messageListWrapper"> return <div className="mx_FilePanel mx_RoomView_messageListWrapper">
<div className="mx_RoomView_empty"> <div className="mx_RoomView_empty">
@ -220,7 +216,7 @@ const FilePanel = createReactClass({
// "(" + this.state.timelineSet._timelines.join(", ") + ")" + " with key " + this.props.roomId); // "(" + this.state.timelineSet._timelines.join(", ") + ")" + " with key " + this.props.roomId);
return ( return (
<div className="mx_FilePanel" role="tabpanel"> <div className="mx_FilePanel" role="tabpanel">
<TimelinePanel key={"filepanel_" + this.props.roomId} <TimelinePanel
manageReadReceipts={false} manageReadReceipts={false}
manageReadMarkers={false} manageReadMarkers={false}
timelineSet={this.state.timelineSet} timelineSet={this.state.timelineSet}
@ -239,7 +235,7 @@ const FilePanel = createReactClass({
</div> </div>
); );
} }
}, }
}); }
export default FilePanel; export default FilePanel;

View file

@ -17,7 +17,6 @@ limitations under the License.
*/ */
import React from 'react'; import React from 'react';
import createReactClass from 'create-react-class';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import {MatrixClientPeg} from '../../MatrixClientPeg'; import {MatrixClientPeg} from '../../MatrixClientPeg';
import * as sdk from '../../index'; import * as sdk from '../../index';
@ -70,10 +69,8 @@ const UserSummaryType = PropTypes.shape({
}).isRequired, }).isRequired,
}); });
const CategoryRoomList = createReactClass({ class CategoryRoomList extends React.Component {
displayName: 'CategoryRoomList', static propTypes = {
props: {
rooms: PropTypes.arrayOf(RoomSummaryType).isRequired, rooms: PropTypes.arrayOf(RoomSummaryType).isRequired,
category: PropTypes.shape({ category: PropTypes.shape({
profile: PropTypes.shape({ profile: PropTypes.shape({
@ -84,9 +81,9 @@ const CategoryRoomList = createReactClass({
// Whether the list should be editable // Whether the list should be editable
editing: PropTypes.bool.isRequired, editing: PropTypes.bool.isRequired,
}, };
onAddRoomsToSummaryClicked: function(ev) { onAddRoomsToSummaryClicked = (ev) => {
ev.preventDefault(); ev.preventDefault();
const AddressPickerDialog = sdk.getComponent("dialogs.AddressPickerDialog"); const AddressPickerDialog = sdk.getComponent("dialogs.AddressPickerDialog");
Modal.createTrackedDialog('Add Rooms to Group Summary', '', AddressPickerDialog, { Modal.createTrackedDialog('Add Rooms to Group Summary', '', AddressPickerDialog, {
@ -122,9 +119,9 @@ const CategoryRoomList = createReactClass({
}); });
}, },
}, /*className=*/null, /*isPriority=*/false, /*isStatic=*/true); }, /*className=*/null, /*isPriority=*/false, /*isStatic=*/true);
}, };
render: function() { render() {
const TintableSvg = sdk.getComponent("elements.TintableSvg"); const TintableSvg = sdk.getComponent("elements.TintableSvg");
const addButton = this.props.editing ? const addButton = this.props.editing ?
(<AccessibleButton className="mx_GroupView_featuredThings_addButton" (<AccessibleButton className="mx_GroupView_featuredThings_addButton"
@ -155,19 +152,17 @@ const CategoryRoomList = createReactClass({
{ roomNodes } { roomNodes }
{ addButton } { addButton }
</div>; </div>;
}, }
}); }
const FeaturedRoom = createReactClass({ class FeaturedRoom extends React.Component {
displayName: 'FeaturedRoom', static propTypes = {
props: {
summaryInfo: RoomSummaryType.isRequired, summaryInfo: RoomSummaryType.isRequired,
editing: PropTypes.bool.isRequired, editing: PropTypes.bool.isRequired,
groupId: PropTypes.string.isRequired, groupId: PropTypes.string.isRequired,
}, };
onClick: function(e) { onClick = (e) => {
e.preventDefault(); e.preventDefault();
e.stopPropagation(); e.stopPropagation();
@ -176,9 +171,9 @@ const FeaturedRoom = createReactClass({
room_alias: this.props.summaryInfo.profile.canonical_alias, room_alias: this.props.summaryInfo.profile.canonical_alias,
room_id: this.props.summaryInfo.room_id, room_id: this.props.summaryInfo.room_id,
}); });
}, };
onDeleteClicked: function(e) { onDeleteClicked = (e) => {
e.preventDefault(); e.preventDefault();
e.stopPropagation(); e.stopPropagation();
GroupStore.removeRoomFromGroupSummary( GroupStore.removeRoomFromGroupSummary(
@ -201,9 +196,9 @@ const FeaturedRoom = createReactClass({
description: _t("The room '%(roomName)s' could not be removed from the summary.", {roomName}), description: _t("The room '%(roomName)s' could not be removed from the summary.", {roomName}),
}); });
}); });
}, };
render: function() { render() {
const RoomAvatar = sdk.getComponent("avatars.RoomAvatar"); const RoomAvatar = sdk.getComponent("avatars.RoomAvatar");
const roomName = this.props.summaryInfo.profile.name || const roomName = this.props.summaryInfo.profile.name ||
@ -243,13 +238,11 @@ const FeaturedRoom = createReactClass({
<div className="mx_GroupView_featuredThing_name">{ roomNameNode }</div> <div className="mx_GroupView_featuredThing_name">{ roomNameNode }</div>
{ deleteButton } { deleteButton }
</AccessibleButton>; </AccessibleButton>;
}, }
}); }
const RoleUserList = createReactClass({ class RoleUserList extends React.Component {
displayName: 'RoleUserList', static propTypes = {
props: {
users: PropTypes.arrayOf(UserSummaryType).isRequired, users: PropTypes.arrayOf(UserSummaryType).isRequired,
role: PropTypes.shape({ role: PropTypes.shape({
profile: PropTypes.shape({ profile: PropTypes.shape({
@ -260,9 +253,9 @@ const RoleUserList = createReactClass({
// Whether the list should be editable // Whether the list should be editable
editing: PropTypes.bool.isRequired, editing: PropTypes.bool.isRequired,
}, };
onAddUsersClicked: function(ev) { onAddUsersClicked = (ev) => {
ev.preventDefault(); ev.preventDefault();
const AddressPickerDialog = sdk.getComponent("dialogs.AddressPickerDialog"); const AddressPickerDialog = sdk.getComponent("dialogs.AddressPickerDialog");
Modal.createTrackedDialog('Add Users to Group Summary', '', AddressPickerDialog, { Modal.createTrackedDialog('Add Users to Group Summary', '', AddressPickerDialog, {
@ -298,9 +291,9 @@ const RoleUserList = createReactClass({
}); });
}, },
}, /*className=*/null, /*isPriority=*/false, /*isStatic=*/true); }, /*className=*/null, /*isPriority=*/false, /*isStatic=*/true);
}, };
render: function() { render() {
const TintableSvg = sdk.getComponent("elements.TintableSvg"); const TintableSvg = sdk.getComponent("elements.TintableSvg");
const addButton = this.props.editing ? const addButton = this.props.editing ?
(<AccessibleButton className="mx_GroupView_featuredThings_addButton" onClick={this.onAddUsersClicked}> (<AccessibleButton className="mx_GroupView_featuredThings_addButton" onClick={this.onAddUsersClicked}>
@ -325,19 +318,17 @@ const RoleUserList = createReactClass({
{ userNodes } { userNodes }
{ addButton } { addButton }
</div>; </div>;
}, }
}); }
const FeaturedUser = createReactClass({ class FeaturedUser extends React.Component {
displayName: 'FeaturedUser', static propTypes = {
props: {
summaryInfo: UserSummaryType.isRequired, summaryInfo: UserSummaryType.isRequired,
editing: PropTypes.bool.isRequired, editing: PropTypes.bool.isRequired,
groupId: PropTypes.string.isRequired, groupId: PropTypes.string.isRequired,
}, };
onClick: function(e) { onClick = (e) => {
e.preventDefault(); e.preventDefault();
e.stopPropagation(); e.stopPropagation();
@ -345,9 +336,9 @@ const FeaturedUser = createReactClass({
action: 'view_start_chat_or_reuse', action: 'view_start_chat_or_reuse',
user_id: this.props.summaryInfo.user_id, user_id: this.props.summaryInfo.user_id,
}); });
}, };
onDeleteClicked: function(e) { onDeleteClicked = (e) => {
e.preventDefault(); e.preventDefault();
e.stopPropagation(); e.stopPropagation();
GroupStore.removeUserFromGroupSummary( GroupStore.removeUserFromGroupSummary(
@ -368,9 +359,9 @@ const FeaturedUser = createReactClass({
description: _t("The user '%(displayName)s' could not be removed from the summary.", {displayName}), description: _t("The user '%(displayName)s' could not be removed from the summary.", {displayName}),
}); });
}); });
}, };
render: function() { render() {
const BaseAvatar = sdk.getComponent("avatars.BaseAvatar"); const BaseAvatar = sdk.getComponent("avatars.BaseAvatar");
const name = this.props.summaryInfo.displayname || this.props.summaryInfo.user_id; const name = this.props.summaryInfo.displayname || this.props.summaryInfo.user_id;
@ -394,23 +385,20 @@ const FeaturedUser = createReactClass({
<div className="mx_GroupView_featuredThing_name">{ userNameNode }</div> <div className="mx_GroupView_featuredThing_name">{ userNameNode }</div>
{ deleteButton } { deleteButton }
</AccessibleButton>; </AccessibleButton>;
}, }
}); }
const GROUP_JOINPOLICY_OPEN = "open"; const GROUP_JOINPOLICY_OPEN = "open";
const GROUP_JOINPOLICY_INVITE = "invite"; const GROUP_JOINPOLICY_INVITE = "invite";
export default createReactClass({ export default class GroupView extends React.Component {
displayName: 'GroupView', static propTypes = {
propTypes: {
groupId: PropTypes.string.isRequired, groupId: PropTypes.string.isRequired,
// Whether this is the first time the group admin is viewing the group // Whether this is the first time the group admin is viewing the group
groupIsNew: PropTypes.bool, groupIsNew: PropTypes.bool,
}, };
getInitialState: function() { state = {
return {
summary: null, summary: null,
isGroupPublicised: null, isGroupPublicised: null,
isUserPrivileged: null, isUserPrivileged: null,
@ -426,9 +414,8 @@ export default createReactClass({
inviterProfile: null, inviterProfile: null,
showRightPanel: RightPanelStore.getSharedInstance().isOpenForGroup, showRightPanel: RightPanelStore.getSharedInstance().isOpenForGroup,
}; };
},
componentDidMount: function() { componentDidMount() {
this._unmounted = false; this._unmounted = false;
this._matrixClient = MatrixClientPeg.get(); this._matrixClient = MatrixClientPeg.get();
this._matrixClient.on("Group.myMembership", this._onGroupMyMembership); this._matrixClient.on("Group.myMembership", this._onGroupMyMembership);
@ -437,9 +424,9 @@ export default createReactClass({
this._dispatcherRef = dis.register(this._onAction); this._dispatcherRef = dis.register(this._onAction);
this._rightPanelStoreToken = RightPanelStore.getSharedInstance().addListener(this._onRightPanelStoreUpdate); this._rightPanelStoreToken = RightPanelStore.getSharedInstance().addListener(this._onRightPanelStoreUpdate);
}, }
componentWillUnmount: function() { componentWillUnmount() {
this._unmounted = true; this._unmounted = true;
this._matrixClient.removeListener("Group.myMembership", this._onGroupMyMembership); this._matrixClient.removeListener("Group.myMembership", this._onGroupMyMembership);
dis.unregister(this._dispatcherRef); dis.unregister(this._dispatcherRef);
@ -448,10 +435,11 @@ export default createReactClass({
if (this._rightPanelStoreToken) { if (this._rightPanelStoreToken) {
this._rightPanelStoreToken.remove(); this._rightPanelStoreToken.remove();
} }
}, }
// TODO: [REACT-WARNING] Replace with appropriate lifecycle event // TODO: [REACT-WARNING] Replace with appropriate lifecycle event
UNSAFE_componentWillReceiveProps: function(newProps) { // eslint-disable-next-line camelcase
UNSAFE_componentWillReceiveProps(newProps) {
if (this.props.groupId !== newProps.groupId) { if (this.props.groupId !== newProps.groupId) {
this.setState({ this.setState({
summary: null, summary: null,
@ -460,24 +448,24 @@ export default createReactClass({
this._initGroupStore(newProps.groupId); this._initGroupStore(newProps.groupId);
}); });
} }
}, }
_onRightPanelStoreUpdate: function() { _onRightPanelStoreUpdate = () => {
this.setState({ this.setState({
showRightPanel: RightPanelStore.getSharedInstance().isOpenForGroup, showRightPanel: RightPanelStore.getSharedInstance().isOpenForGroup,
}); });
}, };
_onGroupMyMembership: function(group) { _onGroupMyMembership = (group) => {
if (this._unmounted || group.groupId !== this.props.groupId) return; if (this._unmounted || group.groupId !== this.props.groupId) return;
if (group.myMembership === 'leave') { if (group.myMembership === 'leave') {
// Leave settings - the user might have clicked the "Leave" button // Leave settings - the user might have clicked the "Leave" button
this._closeSettings(); this._closeSettings();
} }
this.setState({membershipBusy: false}); this.setState({membershipBusy: false});
}, };
_initGroupStore: function(groupId, firstInit) { _initGroupStore(groupId, firstInit) {
const group = this._matrixClient.getGroup(groupId); const group = this._matrixClient.getGroup(groupId);
if (group && group.inviter && group.inviter.userId) { if (group && group.inviter && group.inviter.userId) {
this._fetchInviterProfile(group.inviter.userId); this._fetchInviterProfile(group.inviter.userId);
@ -506,9 +494,9 @@ export default createReactClass({
}); });
} }
}); });
}, }
onGroupStoreUpdated(firstInit) { onGroupStoreUpdated = (firstInit) => {
if (this._unmounted) return; if (this._unmounted) return;
const summary = GroupStore.getSummary(this.props.groupId); const summary = GroupStore.getSummary(this.props.groupId);
if (summary.profile) { if (summary.profile) {
@ -533,7 +521,7 @@ export default createReactClass({
if (this.props.groupIsNew && firstInit) { if (this.props.groupIsNew && firstInit) {
this._onEditClick(); this._onEditClick();
} }
}, };
_fetchInviterProfile(userId) { _fetchInviterProfile(userId) {
this.setState({ this.setState({
@ -555,9 +543,9 @@ export default createReactClass({
inviterProfileBusy: false, inviterProfileBusy: false,
}); });
}); });
}, }
_onEditClick: function() { _onEditClick = () => {
this.setState({ this.setState({
editing: true, editing: true,
profileForm: Object.assign({}, this.state.summary.profile), profileForm: Object.assign({}, this.state.summary.profile),
@ -568,20 +556,20 @@ export default createReactClass({
GROUP_JOINPOLICY_INVITE, GROUP_JOINPOLICY_INVITE,
}, },
}); });
}, };
_onShareClick: function() { _onShareClick = () => {
const ShareDialog = sdk.getComponent("dialogs.ShareDialog"); const ShareDialog = sdk.getComponent("dialogs.ShareDialog");
Modal.createTrackedDialog('share community dialog', '', ShareDialog, { Modal.createTrackedDialog('share community dialog', '', ShareDialog, {
target: this._matrixClient.getGroup(this.props.groupId) || new Group(this.props.groupId), target: this._matrixClient.getGroup(this.props.groupId) || new Group(this.props.groupId),
}); });
}, };
_onCancelClick: function() { _onCancelClick = () => {
this._closeSettings(); this._closeSettings();
}, };
_onAction(payload) { _onAction = (payload) => {
switch (payload.action) { switch (payload.action) {
// NOTE: close_settings is an app-wide dispatch; as it is dispatched from MatrixChat // NOTE: close_settings is an app-wide dispatch; as it is dispatched from MatrixChat
case 'close_settings': case 'close_settings':
@ -593,34 +581,34 @@ export default createReactClass({
default: default:
break; break;
} }
}, };
_closeSettings() { _closeSettings = () => {
dis.dispatch({action: 'close_settings'}); dis.dispatch({action: 'close_settings'});
}, };
_onNameChange: function(value) { _onNameChange = (value) => {
const newProfileForm = Object.assign(this.state.profileForm, { name: value }); const newProfileForm = Object.assign(this.state.profileForm, { name: value });
this.setState({ this.setState({
profileForm: newProfileForm, profileForm: newProfileForm,
}); });
}, };
_onShortDescChange: function(value) { _onShortDescChange = (value) => {
const newProfileForm = Object.assign(this.state.profileForm, { short_description: value }); const newProfileForm = Object.assign(this.state.profileForm, { short_description: value });
this.setState({ this.setState({
profileForm: newProfileForm, profileForm: newProfileForm,
}); });
}, };
_onLongDescChange: function(e) { _onLongDescChange = (e) => {
const newProfileForm = Object.assign(this.state.profileForm, { long_description: e.target.value }); const newProfileForm = Object.assign(this.state.profileForm, { long_description: e.target.value });
this.setState({ this.setState({
profileForm: newProfileForm, profileForm: newProfileForm,
}); });
}, };
_onAvatarSelected: function(ev) { _onAvatarSelected = ev => {
const file = ev.target.files[0]; const file = ev.target.files[0];
if (!file) return; if (!file) return;
@ -644,15 +632,15 @@ export default createReactClass({
description: _t('Failed to upload image'), description: _t('Failed to upload image'),
}); });
}); });
}, };
_onJoinableChange: function(ev) { _onJoinableChange = ev => {
this.setState({ this.setState({
joinableForm: { policyType: ev.target.value }, joinableForm: { policyType: ev.target.value },
}); });
}, };
_onSaveClick: function() { _onSaveClick = () => {
this.setState({saving: true}); this.setState({saving: true});
const savePromise = this.state.isUserPrivileged ? this._saveGroup() : Promise.resolve(); const savePromise = this.state.isUserPrivileged ? this._saveGroup() : Promise.resolve();
savePromise.then((result) => { savePromise.then((result) => {
@ -683,16 +671,16 @@ export default createReactClass({
avatarChanged: false, avatarChanged: false,
}); });
}); });
}, };
_saveGroup: async function() { async _saveGroup() {
await this._matrixClient.setGroupProfile(this.props.groupId, this.state.profileForm); await this._matrixClient.setGroupProfile(this.props.groupId, this.state.profileForm);
await this._matrixClient.setGroupJoinPolicy(this.props.groupId, { await this._matrixClient.setGroupJoinPolicy(this.props.groupId, {
type: this.state.joinableForm.policyType, type: this.state.joinableForm.policyType,
}); });
}, }
_onAcceptInviteClick: async function() { _onAcceptInviteClick = async () => {
this.setState({membershipBusy: true}); this.setState({membershipBusy: true});
// Wait 500ms to prevent flashing. Do this before sending a request otherwise we risk the // Wait 500ms to prevent flashing. Do this before sending a request otherwise we risk the
@ -709,9 +697,9 @@ export default createReactClass({
description: _t("Unable to accept invite"), description: _t("Unable to accept invite"),
}); });
}); });
}, };
_onRejectInviteClick: async function() { _onRejectInviteClick = async () => {
this.setState({membershipBusy: true}); this.setState({membershipBusy: true});
// Wait 500ms to prevent flashing. Do this before sending a request otherwise we risk the // Wait 500ms to prevent flashing. Do this before sending a request otherwise we risk the
@ -728,9 +716,9 @@ export default createReactClass({
description: _t("Unable to reject invite"), description: _t("Unable to reject invite"),
}); });
}); });
}, };
_onJoinClick: async function() { _onJoinClick = async () => {
if (this._matrixClient.isGuest()) { if (this._matrixClient.isGuest()) {
dis.dispatch({action: 'require_registration', screen_after: {screen: `group/${this.props.groupId}`}}); dis.dispatch({action: 'require_registration', screen_after: {screen: `group/${this.props.groupId}`}});
return; return;
@ -752,9 +740,9 @@ export default createReactClass({
description: _t("Unable to join community"), description: _t("Unable to join community"),
}); });
}); });
}, };
_leaveGroupWarnings: function() { _leaveGroupWarnings() {
const warnings = []; const warnings = [];
if (this.state.isUserPrivileged) { if (this.state.isUserPrivileged) {
@ -768,10 +756,9 @@ export default createReactClass({
} }
return warnings; return warnings;
}, }
_onLeaveClick = () => {
_onLeaveClick: function() {
const QuestionDialog = sdk.getComponent("dialogs.QuestionDialog"); const QuestionDialog = sdk.getComponent("dialogs.QuestionDialog");
const warnings = this._leaveGroupWarnings(); const warnings = this._leaveGroupWarnings();
@ -806,13 +793,13 @@ export default createReactClass({
}); });
}, },
}); });
}, };
_onAddRoomsClick: function() { _onAddRoomsClick = () => {
showGroupAddRoomDialog(this.props.groupId); showGroupAddRoomDialog(this.props.groupId);
}, };
_getGroupSection: function() { _getGroupSection() {
const groupSettingsSectionClasses = classnames({ const groupSettingsSectionClasses = classnames({
"mx_GroupView_group": this.state.editing, "mx_GroupView_group": this.state.editing,
"mx_GroupView_group_disabled": this.state.editing && !this.state.isUserPrivileged, "mx_GroupView_group_disabled": this.state.editing && !this.state.isUserPrivileged,
@ -856,9 +843,9 @@ export default createReactClass({
{ this._getLongDescriptionNode() } { this._getLongDescriptionNode() }
{ this._getRoomsNode() } { this._getRoomsNode() }
</div>; </div>;
}, }
_getRoomsNode: function() { _getRoomsNode() {
const RoomDetailList = sdk.getComponent('rooms.RoomDetailList'); const RoomDetailList = sdk.getComponent('rooms.RoomDetailList');
const AccessibleButton = sdk.getComponent('elements.AccessibleButton'); const AccessibleButton = sdk.getComponent('elements.AccessibleButton');
const TintableSvg = sdk.getComponent('elements.TintableSvg'); const TintableSvg = sdk.getComponent('elements.TintableSvg');
@ -902,9 +889,9 @@ export default createReactClass({
className={roomDetailListClassName} /> className={roomDetailListClassName} />
} }
</div>; </div>;
}, }
_getFeaturedRoomsNode: function() { _getFeaturedRoomsNode() {
const summary = this.state.summary; const summary = this.state.summary;
const defaultCategoryRooms = []; const defaultCategoryRooms = [];
@ -943,9 +930,9 @@ export default createReactClass({
{ defaultCategoryNode } { defaultCategoryNode }
{ categoryRoomNodes } { categoryRoomNodes }
</div>; </div>;
}, }
_getFeaturedUsersNode: function() { _getFeaturedUsersNode() {
const summary = this.state.summary; const summary = this.state.summary;
const noRoleUsers = []; const noRoleUsers = [];
@ -984,9 +971,9 @@ export default createReactClass({
{ noRoleNode } { noRoleNode }
{ roleUserNodes } { roleUserNodes }
</div>; </div>;
}, }
_getMembershipSection: function() { _getMembershipSection() {
const Spinner = sdk.getComponent("elements.Spinner"); const Spinner = sdk.getComponent("elements.Spinner");
const BaseAvatar = sdk.getComponent("avatars.BaseAvatar"); const BaseAvatar = sdk.getComponent("avatars.BaseAvatar");
@ -1100,9 +1087,9 @@ export default createReactClass({
</div> </div>
</div> </div>
</div>; </div>;
}, }
_getJoinableNode: function() { _getJoinableNode() {
const InlineSpinner = sdk.getComponent('elements.InlineSpinner'); const InlineSpinner = sdk.getComponent('elements.InlineSpinner');
return this.state.editing ? <div> return this.state.editing ? <div>
<h3> <h3>
@ -1136,9 +1123,9 @@ export default createReactClass({
</label> </label>
</div> </div>
</div> : null; </div> : null;
}, }
_getLongDescriptionNode: function() { _getLongDescriptionNode() {
const summary = this.state.summary; const summary = this.state.summary;
let description = null; let description = null;
if (summary.profile && summary.profile.long_description) { if (summary.profile && summary.profile.long_description) {
@ -1175,9 +1162,9 @@ export default createReactClass({
<div className="mx_GroupView_groupDesc"> <div className="mx_GroupView_groupDesc">
{ description } { description }
</div>; </div>;
}, }
render: function() { render() {
const GroupAvatar = sdk.getComponent("avatars.GroupAvatar"); const GroupAvatar = sdk.getComponent("avatars.GroupAvatar");
const Spinner = sdk.getComponent("elements.Spinner"); const Spinner = sdk.getComponent("elements.Spinner");
@ -1366,5 +1353,5 @@ export default createReactClass({
console.error("Invalid state for GroupView"); console.error("Invalid state for GroupView");
return <div />; return <div />;
} }
}, }
}); }

View file

@ -17,7 +17,6 @@ limitations under the License.
import {InteractiveAuth} from "matrix-js-sdk"; import {InteractiveAuth} from "matrix-js-sdk";
import React, {createRef} from 'react'; import React, {createRef} from 'react';
import createReactClass from 'create-react-class';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import getEntryComponentForLoginType from '../views/auth/InteractiveAuthEntryComponents'; import getEntryComponentForLoginType from '../views/auth/InteractiveAuthEntryComponents';
@ -26,10 +25,8 @@ import * as sdk from '../../index';
export const ERROR_USER_CANCELLED = new Error("User cancelled auth session"); export const ERROR_USER_CANCELLED = new Error("User cancelled auth session");
export default createReactClass({ export default class InteractiveAuthComponent extends React.Component {
displayName: 'InteractiveAuth', static propTypes = {
propTypes: {
// matrix client to use for UI auth requests // matrix client to use for UI auth requests
matrixClient: PropTypes.object.isRequired, matrixClient: PropTypes.object.isRequired,
@ -86,20 +83,19 @@ export default createReactClass({
// continueText and continueKind are passed straight through to the AuthEntryComponent. // continueText and continueKind are passed straight through to the AuthEntryComponent.
continueText: PropTypes.string, continueText: PropTypes.string,
continueKind: PropTypes.string, continueKind: PropTypes.string,
}, };
getInitialState: function() { constructor(props) {
return { super(props);
this.state = {
authStage: null, authStage: null,
busy: false, busy: false,
errorText: null, errorText: null,
stageErrorText: null, stageErrorText: null,
submitButtonEnabled: false, submitButtonEnabled: false,
}; };
},
// TODO: [REACT-WARNING] Replace component with real class, use constructor for refs
UNSAFE_componentWillMount: function() {
this._unmounted = false; this._unmounted = false;
this._authLogic = new InteractiveAuth({ this._authLogic = new InteractiveAuth({
authData: this.props.authData, authData: this.props.authData,
@ -114,6 +110,18 @@ export default createReactClass({
requestEmailToken: this._requestEmailToken, requestEmailToken: this._requestEmailToken,
}); });
this._intervalId = null;
if (this.props.poll) {
this._intervalId = setInterval(() => {
this._authLogic.poll();
}, 2000);
}
this._stageComponent = createRef();
}
// TODO: [REACT-WARNING] Replace component with real class, use constructor for refs
UNSAFE_componentWillMount() { // eslint-disable-line camelcase
this._authLogic.attemptAuth().then((result) => { this._authLogic.attemptAuth().then((result) => {
const extra = { const extra = {
emailSid: this._authLogic.getEmailSid(), emailSid: this._authLogic.getEmailSid(),
@ -132,26 +140,17 @@ export default createReactClass({
errorText: msg, errorText: msg,
}); });
}); });
this._intervalId = null;
if (this.props.poll) {
this._intervalId = setInterval(() => {
this._authLogic.poll();
}, 2000);
} }
this._stageComponent = createRef(); componentWillUnmount() {
},
componentWillUnmount: function() {
this._unmounted = true; this._unmounted = true;
if (this._intervalId !== null) { if (this._intervalId !== null) {
clearInterval(this._intervalId); clearInterval(this._intervalId);
} }
}, }
_requestEmailToken: async function(...args) { _requestEmailToken = async (...args) => {
this.setState({ this.setState({
busy: true, busy: true,
}); });
@ -162,15 +161,15 @@ export default createReactClass({
busy: false, busy: false,
}); });
} }
}, };
tryContinue: function() { tryContinue = () => {
if (this._stageComponent.current && this._stageComponent.current.tryContinue) { if (this._stageComponent.current && this._stageComponent.current.tryContinue) {
this._stageComponent.current.tryContinue(); this._stageComponent.current.tryContinue();
} }
}, };
_authStateUpdated: function(stageType, stageState) { _authStateUpdated = (stageType, stageState) => {
const oldStage = this.state.authStage; const oldStage = this.state.authStage;
this.setState({ this.setState({
busy: false, busy: false,
@ -180,16 +179,16 @@ export default createReactClass({
}, () => { }, () => {
if (oldStage != stageType) this._setFocus(); if (oldStage != stageType) this._setFocus();
}); });
}, };
_requestCallback: function(auth) { _requestCallback = (auth) => {
// This wrapper just exists because the js-sdk passes a second // This wrapper just exists because the js-sdk passes a second
// 'busy' param for backwards compat. This throws the tests off // 'busy' param for backwards compat. This throws the tests off
// so discard it here. // so discard it here.
return this.props.makeRequest(auth); return this.props.makeRequest(auth);
}, };
_onBusyChanged: function(busy) { _onBusyChanged = (busy) => {
// if we've started doing stuff, reset the error messages // if we've started doing stuff, reset the error messages
if (busy) { if (busy) {
this.setState({ this.setState({
@ -204,29 +203,29 @@ export default createReactClass({
// there's a new screen to show the user. This is implemented by setting // there's a new screen to show the user. This is implemented by setting
// `busy: false` in `_authStateUpdated`. // `busy: false` in `_authStateUpdated`.
// See also https://github.com/vector-im/element-web/issues/12546 // See also https://github.com/vector-im/element-web/issues/12546
}, };
_setFocus: function() { _setFocus() {
if (this._stageComponent.current && this._stageComponent.current.focus) { if (this._stageComponent.current && this._stageComponent.current.focus) {
this._stageComponent.current.focus(); this._stageComponent.current.focus();
} }
}, }
_submitAuthDict: function(authData) { _submitAuthDict = authData => {
this._authLogic.submitAuthDict(authData); this._authLogic.submitAuthDict(authData);
}, };
_onPhaseChange: function(newPhase) { _onPhaseChange = newPhase => {
if (this.props.onStagePhaseChange) { if (this.props.onStagePhaseChange) {
this.props.onStagePhaseChange(this.state.authStage, newPhase || 0); this.props.onStagePhaseChange(this.state.authStage, newPhase || 0);
} }
}, };
_onStageCancel: function() { _onStageCancel = () => {
this.props.onAuthFinished(false, ERROR_USER_CANCELLED); this.props.onAuthFinished(false, ERROR_USER_CANCELLED);
}, };
_renderCurrentStage: function() { _renderCurrentStage() {
const stage = this.state.authStage; const stage = this.state.authStage;
if (!stage) { if (!stage) {
if (this.state.busy) { if (this.state.busy) {
@ -260,16 +259,17 @@ export default createReactClass({
onCancel={this._onStageCancel} onCancel={this._onStageCancel}
/> />
); );
}, }
_onAuthStageFailed: function(e) { _onAuthStageFailed = e => {
this.props.onAuthFinished(false, e); this.props.onAuthFinished(false, e);
}, };
_setEmailSid: function(sid) {
this._authLogic.setEmailSid(sid);
},
render: function() { _setEmailSid = sid => {
this._authLogic.setEmailSid(sid);
};
render() {
let error = null; let error = null;
if (this.state.errorText) { if (this.state.errorText) {
error = ( error = (
@ -287,5 +287,5 @@ export default createReactClass({
</div> </div>
</div> </div>
); );
}, }
}); }

View file

@ -43,11 +43,11 @@ import PlatformPeg from "../../PlatformPeg";
import { DefaultTagID } from "../../stores/room-list/models"; import { DefaultTagID } from "../../stores/room-list/models";
import { import {
showToast as showSetPasswordToast, showToast as showSetPasswordToast,
hideToast as hideSetPasswordToast hideToast as hideSetPasswordToast,
} from "../../toasts/SetPasswordToast"; } from "../../toasts/SetPasswordToast";
import { import {
showToast as showServerLimitToast, showToast as showServerLimitToast,
hideToast as hideServerLimitToast hideToast as hideServerLimitToast,
} from "../../toasts/ServerLimitToast"; } from "../../toasts/ServerLimitToast";
import { Action } from "../../dispatcher/actions"; import { Action } from "../../dispatcher/actions";
import LeftPanel from "./LeftPanel"; import LeftPanel from "./LeftPanel";
@ -79,6 +79,7 @@ interface IProps {
initialEventPixelOffset: number; initialEventPixelOffset: number;
leftDisabled: boolean; leftDisabled: boolean;
rightDisabled: boolean; rightDisabled: boolean;
// eslint-disable-next-line camelcase
page_type: string; page_type: string;
autoJoin: boolean; autoJoin: boolean;
thirdPartyInvite?: object; thirdPartyInvite?: object;
@ -98,7 +99,9 @@ interface IProps {
} }
interface IUsageLimit { interface IUsageLimit {
// eslint-disable-next-line camelcase
limit_type: "monthly_active_user" | string; limit_type: "monthly_active_user" | string;
// eslint-disable-next-line camelcase
admin_contact?: string; admin_contact?: string;
} }
@ -316,10 +319,10 @@ class LoggedInView extends React.Component<IProps, IState> {
} }
}; };
_calculateServerLimitToast(syncErrorData: IState["syncErrorData"], usageLimitEventContent?: IUsageLimit) { _calculateServerLimitToast(syncError: IState["syncErrorData"], usageLimitEventContent?: IUsageLimit) {
const error = syncErrorData && syncErrorData.error && syncErrorData.error.errcode === "M_RESOURCE_LIMIT_EXCEEDED"; const error = syncError && syncError.error && syncError.error.errcode === "M_RESOURCE_LIMIT_EXCEEDED";
if (error) { if (error) {
usageLimitEventContent = syncErrorData.error.data; usageLimitEventContent = syncError.error.data;
} }
if (usageLimitEventContent) { if (usageLimitEventContent) {

View file

@ -69,7 +69,7 @@ import { ViewUserPayload } from "../../dispatcher/payloads/ViewUserPayload";
import { Action } from "../../dispatcher/actions"; import { Action } from "../../dispatcher/actions";
import { import {
showToast as showAnalyticsToast, showToast as showAnalyticsToast,
hideToast as hideAnalyticsToast hideToast as hideAnalyticsToast,
} from "../../toasts/AnalyticsToast"; } from "../../toasts/AnalyticsToast";
import {showToast as showNotificationsToast} from "../../toasts/DesktopNotificationsToast"; import {showToast as showNotificationsToast} from "../../toasts/DesktopNotificationsToast";
import { OpenToTabPayload } from "../../dispatcher/payloads/OpenToTabPayload"; import { OpenToTabPayload } from "../../dispatcher/payloads/OpenToTabPayload";
@ -129,6 +129,7 @@ interface IScreen {
params?: object; params?: object;
} }
/* eslint-disable camelcase */
interface IRoomInfo { interface IRoomInfo {
room_id?: string; room_id?: string;
room_alias?: string; room_alias?: string;
@ -140,6 +141,7 @@ interface IRoomInfo {
oob_data?: object; oob_data?: object;
via_servers?: string[]; via_servers?: string[];
} }
/* eslint-enable camelcase */
interface IProps { // TODO type things better interface IProps { // TODO type things better
config: Record<string, any>; config: Record<string, any>;
@ -165,6 +167,7 @@ interface IState {
// the master view we are showing. // the master view we are showing.
view: Views; view: Views;
// What the LoggedInView would be showing if visible // What the LoggedInView would be showing if visible
// eslint-disable-next-line camelcase
page_type?: PageTypes; page_type?: PageTypes;
// The ID of the room we're viewing. This is either populated directly // The ID of the room we're viewing. This is either populated directly
// in the case where we view a room by ID or by RoomView when it resolves // in the case where we view a room by ID or by RoomView when it resolves
@ -180,8 +183,11 @@ interface IState {
middleDisabled: boolean; middleDisabled: boolean;
// the right panel's disabled state is tracked in its store. // the right panel's disabled state is tracked in its store.
// Parameters used in the registration dance with the IS // Parameters used in the registration dance with the IS
// eslint-disable-next-line camelcase
register_client_secret?: string; register_client_secret?: string;
// eslint-disable-next-line camelcase
register_session_id?: string; register_session_id?: string;
// eslint-disable-next-line camelcase
register_id_sid?: string; register_id_sid?: string;
// When showing Modal dialogs we need to set aria-hidden on the root app element // When showing Modal dialogs we need to set aria-hidden on the root app element
// and disable it when there are no dialogs // and disable it when there are no dialogs
@ -341,6 +347,7 @@ export default class MatrixChat extends React.PureComponent<IProps, IState> {
} }
// TODO: [REACT-WARNING] Replace with appropriate lifecycle stage // TODO: [REACT-WARNING] Replace with appropriate lifecycle stage
// eslint-disable-next-line camelcase
UNSAFE_componentWillUpdate(props, state) { UNSAFE_componentWillUpdate(props, state) {
if (this.shouldTrackPageChange(this.state, state)) { if (this.shouldTrackPageChange(this.state, state)) {
this.startPageChangeTimer(); this.startPageChangeTimer();
@ -610,8 +617,7 @@ export default class MatrixChat extends React.PureComponent<IProps, IState> {
const UserSettingsDialog = sdk.getComponent("dialogs.UserSettingsDialog"); const UserSettingsDialog = sdk.getComponent("dialogs.UserSettingsDialog");
Modal.createTrackedDialog('User settings', '', UserSettingsDialog, Modal.createTrackedDialog('User settings', '', UserSettingsDialog,
{initialTabId: tabPayload.initialTabId}, {initialTabId: tabPayload.initialTabId},
/*className=*/null, /*isPriority=*/false, /*isStatic=*/true /*className=*/null, /*isPriority=*/false, /*isStatic=*/true);
);
// View the welcome or home page if we need something to look at // View the welcome or home page if we need something to look at
this.viewSomethingBehindModal(); this.viewSomethingBehindModal();
@ -1433,7 +1439,6 @@ export default class MatrixChat extends React.PureComponent<IProps, IState> {
cli.on("crypto.warning", (type) => { cli.on("crypto.warning", (type) => {
switch (type) { switch (type) {
case 'CRYPTO_WARNING_OLD_VERSION_DETECTED': case 'CRYPTO_WARNING_OLD_VERSION_DETECTED':
const brand = SdkConfig.get().brand;
Modal.createTrackedDialog('Crypto migrated', '', ErrorDialog, { Modal.createTrackedDialog('Crypto migrated', '', ErrorDialog, {
title: _t('Old cryptography data detected'), title: _t('Old cryptography data detected'),
description: _t( description: _t(
@ -1444,7 +1449,7 @@ export default class MatrixChat extends React.PureComponent<IProps, IState> {
"in this version. This may also cause messages exchanged with this " + "in this version. This may also cause messages exchanged with this " +
"version to fail. If you experience problems, log out and back in " + "version to fail. If you experience problems, log out and back in " +
"again. To retain message history, export and re-import your keys.", "again. To retain message history, export and re-import your keys.",
{ brand }, { brand: SdkConfig.get().brand },
), ),
}); });
break; break;

View file

@ -17,7 +17,6 @@ limitations under the License.
*/ */
import React from 'react'; import React from 'react';
import createReactClass from 'create-react-class';
import * as sdk from '../../index'; import * as sdk from '../../index';
import { _t } from '../../languageHandler'; import { _t } from '../../languageHandler';
import SdkConfig from '../../SdkConfig'; import SdkConfig from '../../SdkConfig';
@ -26,29 +25,23 @@ import AccessibleButton from '../views/elements/AccessibleButton';
import MatrixClientContext from "../../contexts/MatrixClientContext"; import MatrixClientContext from "../../contexts/MatrixClientContext";
import AutoHideScrollbar from "./AutoHideScrollbar"; import AutoHideScrollbar from "./AutoHideScrollbar";
export default createReactClass({ export default class MyGroups extends React.Component {
displayName: 'MyGroups', static contextType = MatrixClientContext;
getInitialState: function() { state = {
return {
groups: null, groups: null,
error: null, error: null,
}; };
},
statics: { componentDidMount() {
contextType: MatrixClientContext,
},
componentDidMount: function() {
this._fetch(); this._fetch();
}, }
_onCreateGroupClick: function() { _onCreateGroupClick = () => {
dis.dispatch({action: 'view_create_group'}); dis.dispatch({action: 'view_create_group'});
}, };
_fetch: function() { _fetch() {
this.context.getJoinedGroups().then((result) => { this.context.getJoinedGroups().then((result) => {
this.setState({groups: result.groups, error: null}); this.setState({groups: result.groups, error: null});
}, (err) => { }, (err) => {
@ -59,9 +52,9 @@ export default createReactClass({
} }
this.setState({groups: null, error: err}); this.setState({groups: null, error: err});
}); });
}, }
render: function() { render() {
const brand = SdkConfig.get().brand; const brand = SdkConfig.get().brand;
const Loader = sdk.getComponent("elements.Spinner"); const Loader = sdk.getComponent("elements.Spinner");
const SimpleRoomHeader = sdk.getComponent('rooms.SimpleRoomHeader'); const SimpleRoomHeader = sdk.getComponent('rooms.SimpleRoomHeader');
@ -149,5 +142,5 @@ export default createReactClass({
{ content } { content }
</div> </div>
</div>; </div>;
}, }
}); }

View file

@ -17,7 +17,6 @@ limitations under the License.
*/ */
import React from 'react'; import React from 'react';
import createReactClass from 'create-react-class';
import { _t } from '../../languageHandler'; import { _t } from '../../languageHandler';
import {MatrixClientPeg} from "../../MatrixClientPeg"; import {MatrixClientPeg} from "../../MatrixClientPeg";
import * as sdk from "../../index"; import * as sdk from "../../index";
@ -25,13 +24,8 @@ import * as sdk from "../../index";
/* /*
* Component which shows the global notification list using a TimelinePanel * Component which shows the global notification list using a TimelinePanel
*/ */
const NotificationPanel = createReactClass({ class NotificationPanel extends React.Component {
displayName: 'NotificationPanel', render() {
propTypes: {
},
render: function() {
// wrap a TimelinePanel with the jump-to-event bits turned off. // wrap a TimelinePanel with the jump-to-event bits turned off.
const TimelinePanel = sdk.getComponent("structures.TimelinePanel"); const TimelinePanel = sdk.getComponent("structures.TimelinePanel");
const Loader = sdk.getComponent("elements.Spinner"); const Loader = sdk.getComponent("elements.Spinner");
@ -45,7 +39,7 @@ const NotificationPanel = createReactClass({
if (timelineSet) { if (timelineSet) {
return ( return (
<div className="mx_NotificationPanel" role="tabpanel"> <div className="mx_NotificationPanel" role="tabpanel">
<TimelinePanel key={"NotificationPanel_" + this.props.roomId} <TimelinePanel
manageReadReceipts={false} manageReadReceipts={false}
manageReadMarkers={false} manageReadMarkers={false}
timelineSet={timelineSet} timelineSet={timelineSet}
@ -63,7 +57,7 @@ const NotificationPanel = createReactClass({
</div> </div>
); );
} }
}, }
}); }
export default NotificationPanel; export default NotificationPanel;

View file

@ -21,6 +21,8 @@ limitations under the License.
import React from 'react'; import React from 'react';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import classNames from 'classnames'; import classNames from 'classnames';
import {Room} from "matrix-js-sdk/src/models/room";
import * as sdk from '../../index'; import * as sdk from '../../index';
import dis from '../../dispatcher/dispatcher'; import dis from '../../dispatcher/dispatcher';
import RateLimitedFunc from '../../ratelimitedfunc'; import RateLimitedFunc from '../../ratelimitedfunc';
@ -34,7 +36,7 @@ import {Action} from "../../dispatcher/actions";
export default class RightPanel extends React.Component { export default class RightPanel extends React.Component {
static get propTypes() { static get propTypes() {
return { return {
roomId: PropTypes.string, // if showing panels for a given room, this is set room: PropTypes.instanceOf(Room), // if showing panels for a given room, this is set
groupId: PropTypes.string, // if showing panels for a given group, this is set groupId: PropTypes.string, // if showing panels for a given group, this is set
user: PropTypes.object, // used if we know the user ahead of opening the panel user: PropTypes.object, // used if we know the user ahead of opening the panel
}; };
@ -42,8 +44,8 @@ export default class RightPanel extends React.Component {
static contextType = MatrixClientContext; static contextType = MatrixClientContext;
constructor(props) { constructor(props, context) {
super(props); super(props, context);
this.state = { this.state = {
phase: this._getPhaseFromProps(), phase: this._getPhaseFromProps(),
isUserPrivilegedInGroup: null, isUserPrivilegedInGroup: null,
@ -161,13 +163,13 @@ export default class RightPanel extends React.Component {
} }
onRoomStateMember(ev, state, member) { onRoomStateMember(ev, state, member) {
if (member.roomId !== this.props.roomId) { if (member.roomId !== this.props.room.roomId) {
return; return;
} }
// redraw the badge on the membership list // redraw the badge on the membership list
if (this.state.phase === RightPanelPhases.RoomMemberList && member.roomId === this.props.roomId) { if (this.state.phase === RightPanelPhases.RoomMemberList && member.roomId === this.props.room.roomId) {
this._delayedUpdate(); this._delayedUpdate();
} else if (this.state.phase === RightPanelPhases.RoomMemberInfo && member.roomId === this.props.roomId && } else if (this.state.phase === RightPanelPhases.RoomMemberInfo && member.roomId === this.props.room.roomId &&
member.userId === this.state.member.userId) { member.userId === this.state.member.userId) {
// refresh the member info (e.g. new power level) // refresh the member info (e.g. new power level)
this._delayedUpdate(); this._delayedUpdate();
@ -226,8 +228,8 @@ export default class RightPanel extends React.Component {
switch (this.state.phase) { switch (this.state.phase) {
case RightPanelPhases.RoomMemberList: case RightPanelPhases.RoomMemberList:
if (this.props.roomId) { if (this.props.room.roomId) {
panel = <MemberList roomId={this.props.roomId} key={this.props.roomId} />; panel = <MemberList roomId={this.props.room.roomId} key={this.props.room.roomId} />;
} }
break; break;
case RightPanelPhases.GroupMemberList: case RightPanelPhases.GroupMemberList:
@ -242,8 +244,8 @@ export default class RightPanel extends React.Component {
case RightPanelPhases.EncryptionPanel: case RightPanelPhases.EncryptionPanel:
panel = <UserInfo panel = <UserInfo
user={this.state.member} user={this.state.member}
roomId={this.props.roomId} roomId={this.props.room.roomId}
key={this.props.roomId || this.state.member.userId} key={this.props.room.roomId || this.state.member.userId}
onClose={this.onCloseUserInfo} onClose={this.onCloseUserInfo}
phase={this.state.phase} phase={this.state.phase}
verificationRequest={this.state.verificationRequest} verificationRequest={this.state.verificationRequest}
@ -251,7 +253,7 @@ export default class RightPanel extends React.Component {
/>; />;
break; break;
case RightPanelPhases.Room3pidMemberInfo: case RightPanelPhases.Room3pidMemberInfo:
panel = <ThirdPartyMemberInfo event={this.state.event} key={this.props.roomId} />; panel = <ThirdPartyMemberInfo event={this.state.event} key={this.props.room.roomId} />;
break; break;
case RightPanelPhases.GroupMemberInfo: case RightPanelPhases.GroupMemberInfo:
panel = <UserInfo panel = <UserInfo
@ -270,7 +272,7 @@ export default class RightPanel extends React.Component {
panel = <NotificationPanel />; panel = <NotificationPanel />;
break; break;
case RightPanelPhases.FilePanel: case RightPanelPhases.FilePanel:
panel = <FilePanel roomId={this.props.roomId} resizeNotifier={this.props.resizeNotifier} />; panel = <FilePanel roomId={this.props.room.roomId} resizeNotifier={this.props.resizeNotifier} />;
break; break;
} }

View file

@ -17,7 +17,6 @@ limitations under the License.
*/ */
import React from 'react'; import React from 'react';
import createReactClass from 'create-react-class';
import {MatrixClientPeg} from "../../MatrixClientPeg"; import {MatrixClientPeg} from "../../MatrixClientPeg";
import * as sdk from "../../index"; import * as sdk from "../../index";
import dis from "../../dispatcher/dispatcher"; import dis from "../../dispatcher/dispatcher";
@ -42,16 +41,16 @@ function track(action) {
Analytics.trackEvent('RoomDirectory', action); Analytics.trackEvent('RoomDirectory', action);
} }
export default createReactClass({ export default class RoomDirectory extends React.Component {
displayName: 'RoomDirectory', static propTypes = {
propTypes: {
onFinished: PropTypes.func.isRequired, onFinished: PropTypes.func.isRequired,
}, };
constructor(props) {
super(props);
getInitialState: function() {
const selectedCommunityId = TagOrderStore.getSelectedTags()[0]; const selectedCommunityId = TagOrderStore.getSelectedTags()[0];
return { this.state = {
publicRooms: [], publicRooms: [],
loading: true, loading: true,
protocolsLoading: true, protocolsLoading: true,
@ -64,10 +63,7 @@ export default createReactClass({
: null, : null,
communityName: null, communityName: null,
}; };
},
// TODO: [REACT-WARNING] Move this to constructor
UNSAFE_componentWillMount: function() {
this._unmounted = false; this._unmounted = false;
this.nextBatch = null; this.nextBatch = null;
this.filterTimeout = null; this.filterTimeout = null;
@ -115,16 +111,16 @@ export default createReactClass({
} }
this.refreshRoomList(); this.refreshRoomList();
}, }
componentWillUnmount: function() { componentWillUnmount() {
if (this.filterTimeout) { if (this.filterTimeout) {
clearTimeout(this.filterTimeout); clearTimeout(this.filterTimeout);
} }
this._unmounted = true; this._unmounted = true;
}, }
refreshRoomList: function() { refreshRoomList = () => {
if (this.state.selectedCommunityId) { if (this.state.selectedCommunityId) {
this.setState({ this.setState({
publicRooms: GroupStore.getGroupRooms(this.state.selectedCommunityId).map(r => { publicRooms: GroupStore.getGroupRooms(this.state.selectedCommunityId).map(r => {
@ -158,9 +154,9 @@ export default createReactClass({
loading: true, loading: true,
}); });
this.getMoreRooms(); this.getMoreRooms();
}, };
getMoreRooms: function() { getMoreRooms() {
if (this.state.selectedCommunityId) return Promise.resolve(); // no more rooms if (this.state.selectedCommunityId) return Promise.resolve(); // no more rooms
if (!MatrixClientPeg.get()) return Promise.resolve(); if (!MatrixClientPeg.get()) return Promise.resolve();
@ -233,7 +229,7 @@ export default createReactClass({
), ),
}); });
}); });
}, }
/** /**
* A limited interface for removing rooms from the directory. * A limited interface for removing rooms from the directory.
@ -242,7 +238,7 @@ export default createReactClass({
* HS admins to do this through the RoomSettings interface, but * HS admins to do this through the RoomSettings interface, but
* this needs SPEC-417. * this needs SPEC-417.
*/ */
removeFromDirectory: function(room) { removeFromDirectory(room) {
const alias = get_display_alias_for_room(room); const alias = get_display_alias_for_room(room);
const name = room.name || alias || _t('Unnamed room'); const name = room.name || alias || _t('Unnamed room');
@ -284,18 +280,18 @@ export default createReactClass({
}); });
}, },
}); });
}, }
onRoomClicked: function(room, ev) { onRoomClicked = (room, ev) => {
if (ev.shiftKey && !this.state.selectedCommunityId) { if (ev.shiftKey && !this.state.selectedCommunityId) {
ev.preventDefault(); ev.preventDefault();
this.removeFromDirectory(room); this.removeFromDirectory(room);
} else { } else {
this.showRoom(room); this.showRoom(room);
} }
}, };
onOptionChange: function(server, instanceId) { onOptionChange = (server, instanceId) => {
// clear next batch so we don't try to load more rooms // clear next batch so we don't try to load more rooms
this.nextBatch = null; this.nextBatch = null;
this.setState({ this.setState({
@ -313,15 +309,15 @@ export default createReactClass({
// find the five gitter ones, at which point we do not want // find the five gitter ones, at which point we do not want
// to render all those rooms when switching back to 'all networks'. // to render all those rooms when switching back to 'all networks'.
// Easiest to just blow away the state & re-fetch. // Easiest to just blow away the state & re-fetch.
}, };
onFillRequest: function(backwards) { onFillRequest = (backwards) => {
if (backwards || !this.nextBatch) return Promise.resolve(false); if (backwards || !this.nextBatch) return Promise.resolve(false);
return this.getMoreRooms(); return this.getMoreRooms();
}, };
onFilterChange: function(alias) { onFilterChange = (alias) => {
this.setState({ this.setState({
filterString: alias || null, filterString: alias || null,
}); });
@ -337,9 +333,9 @@ export default createReactClass({
this.filterTimeout = null; this.filterTimeout = null;
this.refreshRoomList(); this.refreshRoomList();
}, 700); }, 700);
}, };
onFilterClear: function() { onFilterClear = () => {
// update immediately // update immediately
this.setState({ this.setState({
filterString: null, filterString: null,
@ -348,9 +344,9 @@ export default createReactClass({
if (this.filterTimeout) { if (this.filterTimeout) {
clearTimeout(this.filterTimeout); clearTimeout(this.filterTimeout);
} }
}, };
onJoinFromSearchClick: function(alias) { onJoinFromSearchClick = (alias) => {
// If we don't have a particular instance id selected, just show that rooms alias // If we don't have a particular instance id selected, just show that rooms alias
if (!this.state.instanceId || this.state.instanceId === ALL_ROOMS) { if (!this.state.instanceId || this.state.instanceId === ALL_ROOMS) {
// If the user specified an alias without a domain, add on whichever server is selected // If the user specified an alias without a domain, add on whichever server is selected
@ -391,9 +387,9 @@ export default createReactClass({
}); });
}); });
} }
}, };
onPreviewClick: function(ev, room) { onPreviewClick = (ev, room) => {
this.props.onFinished(); this.props.onFinished();
dis.dispatch({ dis.dispatch({
action: 'view_room', action: 'view_room',
@ -401,9 +397,9 @@ export default createReactClass({
should_peek: true, should_peek: true,
}); });
ev.stopPropagation(); ev.stopPropagation();
}, };
onViewClick: function(ev, room) { onViewClick = (ev, room) => {
this.props.onFinished(); this.props.onFinished();
dis.dispatch({ dis.dispatch({
action: 'view_room', action: 'view_room',
@ -411,26 +407,26 @@ export default createReactClass({
should_peek: false, should_peek: false,
}); });
ev.stopPropagation(); ev.stopPropagation();
}, };
onJoinClick: function(ev, room) { onJoinClick = (ev, room) => {
this.showRoom(room, null, true); this.showRoom(room, null, true);
ev.stopPropagation(); ev.stopPropagation();
}, };
onCreateRoomClick: function(room) { onCreateRoomClick = room => {
this.props.onFinished(); this.props.onFinished();
dis.dispatch({ dis.dispatch({
action: 'view_create_room', action: 'view_create_room',
public: true, public: true,
}); });
}, };
showRoomAlias: function(alias, autoJoin=false) { showRoomAlias(alias, autoJoin=false) {
this.showRoom(null, alias, autoJoin); this.showRoom(null, alias, autoJoin);
}, }
showRoom: function(room, room_alias, autoJoin=false) { showRoom(room, room_alias, autoJoin=false) {
this.props.onFinished(); this.props.onFinished();
const payload = { const payload = {
action: 'view_room', action: 'view_room',
@ -474,7 +470,7 @@ export default createReactClass({
payload.room_id = room.room_id; payload.room_id = room.room_id;
} }
dis.dispatch(payload); dis.dispatch(payload);
}, }
getRow(room) { getRow(room) {
const client = MatrixClientPeg.get(); const client = MatrixClientPeg.get();
@ -540,22 +536,22 @@ export default createReactClass({
<td className="mx_RoomDirectory_join">{joinOrViewButton}</td> <td className="mx_RoomDirectory_join">{joinOrViewButton}</td>
</tr> </tr>
); );
}, }
collectScrollPanel: function(element) { collectScrollPanel = (element) => {
this.scrollPanel = element; this.scrollPanel = element;
}, };
_stringLooksLikeId: function(s, field_type) { _stringLooksLikeId(s, field_type) {
let pat = /^#[^\s]+:[^\s]/; let pat = /^#[^\s]+:[^\s]/;
if (field_type && field_type.regexp) { if (field_type && field_type.regexp) {
pat = new RegExp(field_type.regexp); pat = new RegExp(field_type.regexp);
} }
return pat.test(s); return pat.test(s);
}, }
_getFieldsForThirdPartyLocation: function(userInput, protocol, instance) { _getFieldsForThirdPartyLocation(userInput, protocol, instance) {
// make an object with the fields specified by that protocol. We // make an object with the fields specified by that protocol. We
// require that the values of all but the last field come from the // require that the values of all but the last field come from the
// instance. The last is the user input. // instance. The last is the user input.
@ -569,20 +565,20 @@ export default createReactClass({
} }
fields[requiredFields[requiredFields.length - 1]] = userInput; fields[requiredFields[requiredFields.length - 1]] = userInput;
return fields; return fields;
}, }
/** /**
* called by the parent component when PageUp/Down/etc is pressed. * called by the parent component when PageUp/Down/etc is pressed.
* *
* We pass it down to the scroll panel. * We pass it down to the scroll panel.
*/ */
handleScrollKey: function(ev) { handleScrollKey = ev => {
if (this.scrollPanel) { if (this.scrollPanel) {
this.scrollPanel.handleScrollKey(ev); this.scrollPanel.handleScrollKey(ev);
} }
}, };
render: function() { render() {
const Loader = sdk.getComponent("elements.Spinner"); const Loader = sdk.getComponent("elements.Spinner");
const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog'); const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog');
const AccessibleButton = sdk.getComponent('elements.AccessibleButton'); const AccessibleButton = sdk.getComponent('elements.AccessibleButton');
@ -712,8 +708,8 @@ export default createReactClass({
</div> </div>
</BaseDialog> </BaseDialog>
); );
}, }
}); }
// Similar to matrix-react-sdk's MatrixTools.getDisplayAliasForRoom // Similar to matrix-react-sdk's MatrixTools.getDisplayAliasForRoom
// but works with the objects we get from the public room list // but works with the objects we get from the public room list

View file

@ -20,7 +20,6 @@ import classNames from "classnames";
import defaultDispatcher from "../../dispatcher/dispatcher"; import defaultDispatcher from "../../dispatcher/dispatcher";
import { _t } from "../../languageHandler"; import { _t } from "../../languageHandler";
import { ActionPayload } from "../../dispatcher/payloads"; import { ActionPayload } from "../../dispatcher/payloads";
import { throttle } from 'lodash';
import { Key } from "../../Keyboard"; import { Key } from "../../Keyboard";
import AccessibleButton from "../views/elements/AccessibleButton"; import AccessibleButton from "../views/elements/AccessibleButton";
import { Action } from "../../dispatcher/actions"; import { Action } from "../../dispatcher/actions";

View file

@ -17,7 +17,6 @@ limitations under the License.
*/ */
import React from 'react'; import React from 'react';
import createReactClass from 'create-react-class';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import Matrix from 'matrix-js-sdk'; import Matrix from 'matrix-js-sdk';
import { _t, _td } from '../../languageHandler'; import { _t, _td } from '../../languageHandler';
@ -39,10 +38,8 @@ function getUnsentMessages(room) {
}); });
} }
export default createReactClass({ export default class RoomStatusBar extends React.Component {
displayName: 'RoomStatusBar', static propTypes = {
propTypes: {
// the room this statusbar is representing. // the room this statusbar is representing.
room: PropTypes.object.isRequired, room: PropTypes.object.isRequired,
// This is true when the user is alone in the room, but has also sent a message. // This is true when the user is alone in the room, but has also sent a message.
@ -86,37 +83,35 @@ export default createReactClass({
// callback for when the status bar is displaying something and should // callback for when the status bar is displaying something and should
// be visible // be visible
onVisible: PropTypes.func, onVisible: PropTypes.func,
}, };
getInitialState: function() { state = {
return {
syncState: MatrixClientPeg.get().getSyncState(), syncState: MatrixClientPeg.get().getSyncState(),
syncStateData: MatrixClientPeg.get().getSyncStateData(), syncStateData: MatrixClientPeg.get().getSyncStateData(),
unsentMessages: getUnsentMessages(this.props.room), unsentMessages: getUnsentMessages(this.props.room),
}; };
},
componentDidMount: function() { componentDidMount() {
MatrixClientPeg.get().on("sync", this.onSyncStateChange); MatrixClientPeg.get().on("sync", this.onSyncStateChange);
MatrixClientPeg.get().on("Room.localEchoUpdated", this._onRoomLocalEchoUpdated); MatrixClientPeg.get().on("Room.localEchoUpdated", this._onRoomLocalEchoUpdated);
this._checkSize(); this._checkSize();
}, }
componentDidUpdate: function() { componentDidUpdate() {
this._checkSize(); this._checkSize();
}, }
componentWillUnmount: function() { componentWillUnmount() {
// we may have entirely lost our client as we're logging out before clicking login on the guest bar... // we may have entirely lost our client as we're logging out before clicking login on the guest bar...
const client = MatrixClientPeg.get(); const client = MatrixClientPeg.get();
if (client) { if (client) {
client.removeListener("sync", this.onSyncStateChange); client.removeListener("sync", this.onSyncStateChange);
client.removeListener("Room.localEchoUpdated", this._onRoomLocalEchoUpdated); client.removeListener("Room.localEchoUpdated", this._onRoomLocalEchoUpdated);
} }
}, }
onSyncStateChange: function(state, prevState, data) { onSyncStateChange = (state, prevState, data) => {
if (state === "SYNCING" && prevState === "SYNCING") { if (state === "SYNCING" && prevState === "SYNCING") {
return; return;
} }
@ -124,39 +119,39 @@ export default createReactClass({
syncState: state, syncState: state,
syncStateData: data, syncStateData: data,
}); });
}, };
_onResendAllClick: function() { _onResendAllClick = () => {
Resend.resendUnsentEvents(this.props.room); Resend.resendUnsentEvents(this.props.room);
dis.fire(Action.FocusComposer); dis.fire(Action.FocusComposer);
}, };
_onCancelAllClick: function() { _onCancelAllClick = () => {
Resend.cancelUnsentEvents(this.props.room); Resend.cancelUnsentEvents(this.props.room);
dis.fire(Action.FocusComposer); dis.fire(Action.FocusComposer);
}, };
_onRoomLocalEchoUpdated: function(event, room, oldEventId, oldStatus) { _onRoomLocalEchoUpdated = (event, room, oldEventId, oldStatus) => {
if (room.roomId !== this.props.room.roomId) return; if (room.roomId !== this.props.room.roomId) return;
this.setState({ this.setState({
unsentMessages: getUnsentMessages(this.props.room), unsentMessages: getUnsentMessages(this.props.room),
}); });
}, };
// Check whether current size is greater than 0, if yes call props.onVisible // Check whether current size is greater than 0, if yes call props.onVisible
_checkSize: function() { _checkSize() {
if (this._getSize()) { if (this._getSize()) {
if (this.props.onVisible) this.props.onVisible(); if (this.props.onVisible) this.props.onVisible();
} else { } else {
if (this.props.onHidden) this.props.onHidden(); if (this.props.onHidden) this.props.onHidden();
} }
}, }
// We don't need the actual height - just whether it is likely to have // We don't need the actual height - just whether it is likely to have
// changed - so we use '0' to indicate normal size, and other values to // changed - so we use '0' to indicate normal size, and other values to
// indicate other sizes. // indicate other sizes.
_getSize: function() { _getSize() {
if (this._shouldShowConnectionError() || if (this._shouldShowConnectionError() ||
this.props.hasActiveCall || this.props.hasActiveCall ||
this.props.sentMessageAndIsAlone this.props.sentMessageAndIsAlone
@ -166,10 +161,10 @@ export default createReactClass({
return STATUS_BAR_EXPANDED_LARGE; return STATUS_BAR_EXPANDED_LARGE;
} }
return STATUS_BAR_HIDDEN; return STATUS_BAR_HIDDEN;
}, }
// return suitable content for the image on the left of the status bar. // return suitable content for the image on the left of the status bar.
_getIndicator: function() { _getIndicator() {
if (this.props.hasActiveCall) { if (this.props.hasActiveCall) {
const TintableSvg = sdk.getComponent("elements.TintableSvg"); const TintableSvg = sdk.getComponent("elements.TintableSvg");
return ( return (
@ -182,9 +177,9 @@ export default createReactClass({
} }
return null; return null;
}, }
_shouldShowConnectionError: function() { _shouldShowConnectionError() {
// no conn bar trumps the "some not sent" msg since you can't resend without // no conn bar trumps the "some not sent" msg since you can't resend without
// a connection! // a connection!
// There's one situation in which we don't show this 'no connection' bar, and that's // There's one situation in which we don't show this 'no connection' bar, and that's
@ -195,9 +190,9 @@ export default createReactClass({
this.state.syncStateData.error.errcode === 'M_RESOURCE_LIMIT_EXCEEDED', this.state.syncStateData.error.errcode === 'M_RESOURCE_LIMIT_EXCEEDED',
); );
return this.state.syncState === "ERROR" && !errorIsMauError; return this.state.syncState === "ERROR" && !errorIsMauError;
}, }
_getUnsentMessageContent: function() { _getUnsentMessageContent() {
const unsentMessages = this.state.unsentMessages; const unsentMessages = this.state.unsentMessages;
if (!unsentMessages.length) return null; if (!unsentMessages.length) return null;
@ -272,10 +267,10 @@ export default createReactClass({
</div> </div>
</div> </div>
</div>; </div>;
}, }
// return suitable content for the main (text) part of the status bar. // return suitable content for the main (text) part of the status bar.
_getContent: function() { _getContent() {
if (this._shouldShowConnectionError()) { if (this._shouldShowConnectionError()) {
return ( return (
<div className="mx_RoomStatusBar_connectionLostBar"> <div className="mx_RoomStatusBar_connectionLostBar">
@ -323,9 +318,9 @@ export default createReactClass({
} }
return null; return null;
}, }
render: function() { render() {
const content = this._getContent(); const content = this._getContent();
const indicator = this._getIndicator(); const indicator = this._getIndicator();
@ -339,5 +334,5 @@ export default createReactClass({
</div> </div>
</div> </div>
); );
}, }
}); }

View file

@ -24,7 +24,6 @@ limitations under the License.
import shouldHideEvent from '../../shouldHideEvent'; import shouldHideEvent from '../../shouldHideEvent';
import React, {createRef} from 'react'; import React, {createRef} from 'react';
import createReactClass from 'create-react-class';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import classNames from 'classnames'; import classNames from 'classnames';
import { _t } from '../../languageHandler'; import { _t } from '../../languageHandler';
@ -68,9 +67,8 @@ if (DEBUG) {
debuglog = console.log.bind(console); debuglog = console.log.bind(console);
} }
export default createReactClass({ export default class RoomView extends React.Component {
displayName: 'RoomView', static propTypes = {
propTypes: {
ConferenceHandler: PropTypes.any, ConferenceHandler: PropTypes.any,
// Called with the credentials of a registered user (if they were a ROU that // Called with the credentials of a registered user (if they were a ROU that
@ -97,15 +95,15 @@ export default createReactClass({
// Servers the RoomView can use to try and assist joins // Servers the RoomView can use to try and assist joins
viaServers: PropTypes.arrayOf(PropTypes.string), viaServers: PropTypes.arrayOf(PropTypes.string),
}, };
statics: { static contextType = MatrixClientContext;
contextType: MatrixClientContext,
}, constructor(props, context) {
super(props, context);
getInitialState: function() {
const llMembers = this.context.hasLazyLoadMembersEnabled(); const llMembers = this.context.hasLazyLoadMembersEnabled();
return { this.state = {
room: null, room: null,
roomId: null, roomId: null,
roomLoading: true, roomLoading: true,
@ -171,10 +169,7 @@ export default createReactClass({
matrixClientIsReady: this.context && this.context.isInitialSyncComplete(), matrixClientIsReady: this.context && this.context.isInitialSyncComplete(),
}; };
},
// TODO: [REACT-WARNING] Replace component with real class, use constructor for refs
UNSAFE_componentWillMount: function() {
this.dispatcherRef = dis.register(this.onAction); this.dispatcherRef = dis.register(this.onAction);
this.context.on("Room", this.onRoom); this.context.on("Room", this.onRoom);
this.context.on("Room.timeline", this.onRoomTimeline); this.context.on("Room.timeline", this.onRoomTimeline);
@ -191,7 +186,6 @@ export default createReactClass({
// Start listening for RoomViewStore updates // Start listening for RoomViewStore updates
this._roomStoreToken = RoomViewStore.addListener(this._onRoomViewStoreUpdate); this._roomStoreToken = RoomViewStore.addListener(this._onRoomViewStoreUpdate);
this._rightPanelStoreToken = RightPanelStore.getSharedInstance().addListener(this._onRightPanelStoreUpdate); this._rightPanelStoreToken = RightPanelStore.getSharedInstance().addListener(this._onRightPanelStoreUpdate);
this._onRoomViewStoreUpdate(true);
WidgetEchoStore.on('update', this._onWidgetEchoStoreUpdate); WidgetEchoStore.on('update', this._onWidgetEchoStoreUpdate);
this._showReadReceiptsWatchRef = SettingsStore.watchSetting("showReadReceipts", null, this._showReadReceiptsWatchRef = SettingsStore.watchSetting("showReadReceipts", null,
@ -201,15 +195,20 @@ export default createReactClass({
this._searchResultsPanel = createRef(); this._searchResultsPanel = createRef();
this._layoutWatcherRef = SettingsStore.watchSetting("useIRCLayout", null, this.onLayoutChange); this._layoutWatcherRef = SettingsStore.watchSetting("useIRCLayout", null, this.onLayoutChange);
}, }
_onReadReceiptsChange: function() { // TODO: [REACT-WARNING] Move into constructor
UNSAFE_componentWillMount() {
this._onRoomViewStoreUpdate(true);
}
_onReadReceiptsChange = () => {
this.setState({ this.setState({
showReadReceipts: SettingsStore.getValue("showReadReceipts", this.state.roomId), showReadReceipts: SettingsStore.getValue("showReadReceipts", this.state.roomId),
}); });
}, };
_onRoomViewStoreUpdate: function(initial) { _onRoomViewStoreUpdate = initial => {
if (this.unmounted) { if (this.unmounted) {
return; return;
} }
@ -303,7 +302,7 @@ export default createReactClass({
if (initial) { if (initial) {
this._setupRoom(newState.room, newState.roomId, newState.joining, newState.shouldPeek); this._setupRoom(newState.room, newState.roomId, newState.joining, newState.shouldPeek);
} }
}, };
_getRoomId() { _getRoomId() {
// According to `_onRoomViewStoreUpdate`, `state.roomId` can be null // According to `_onRoomViewStoreUpdate`, `state.roomId` can be null
@ -312,9 +311,9 @@ export default createReactClass({
// the bare room ID. (We may want to update `state.roomId` after // the bare room ID. (We may want to update `state.roomId` after
// resolving aliases, so we could always trust it.) // resolving aliases, so we could always trust it.)
return this.state.room ? this.state.room.roomId : this.state.roomId; return this.state.room ? this.state.room.roomId : this.state.roomId;
}, }
_getPermalinkCreatorForRoom: function(room) { _getPermalinkCreatorForRoom(room) {
if (!this._permalinkCreators) this._permalinkCreators = {}; if (!this._permalinkCreators) this._permalinkCreators = {};
if (this._permalinkCreators[room.roomId]) return this._permalinkCreators[room.roomId]; if (this._permalinkCreators[room.roomId]) return this._permalinkCreators[room.roomId];
@ -327,22 +326,22 @@ export default createReactClass({
this._permalinkCreators[room.roomId].load(); this._permalinkCreators[room.roomId].load();
} }
return this._permalinkCreators[room.roomId]; return this._permalinkCreators[room.roomId];
}, }
_stopAllPermalinkCreators: function() { _stopAllPermalinkCreators() {
if (!this._permalinkCreators) return; if (!this._permalinkCreators) return;
for (const roomId of Object.keys(this._permalinkCreators)) { for (const roomId of Object.keys(this._permalinkCreators)) {
this._permalinkCreators[roomId].stop(); this._permalinkCreators[roomId].stop();
} }
}, }
_onWidgetEchoStoreUpdate: function() { _onWidgetEchoStoreUpdate = () => {
this.setState({ this.setState({
showApps: this._shouldShowApps(this.state.room), showApps: this._shouldShowApps(this.state.room),
}); });
}, };
_setupRoom: function(room, roomId, joining, shouldPeek) { _setupRoom(room, roomId, joining, shouldPeek) {
// if this is an unknown room then we're in one of three states: // if this is an unknown room then we're in one of three states:
// - This is a room we can peek into (search engine) (we can /peek) // - This is a room we can peek into (search engine) (we can /peek)
// - This is a room we can publicly join or were invited to. (we can /join) // - This is a room we can publicly join or were invited to. (we can /join)
@ -404,9 +403,9 @@ export default createReactClass({
this.setState({isPeeking: false}); this.setState({isPeeking: false});
} }
} }
}, }
_shouldShowApps: function(room) { _shouldShowApps(room) {
if (!BROWSER_SUPPORTS_SANDBOX) return false; if (!BROWSER_SUPPORTS_SANDBOX) return false;
// Check if user has previously chosen to hide the app drawer for this // Check if user has previously chosen to hide the app drawer for this
@ -417,9 +416,9 @@ export default createReactClass({
// This is confusing, but it means to say that we default to the tray being // This is confusing, but it means to say that we default to the tray being
// hidden unless the user clicked to open it. // hidden unless the user clicked to open it.
return hideWidgetDrawer === "false"; return hideWidgetDrawer === "false";
}, }
componentDidMount: function() { componentDidMount() {
const call = this._getCallForRoom(); const call = this._getCallForRoom();
const callState = call ? call.call_state : "ended"; const callState = call ? call.call_state : "ended";
this.setState({ this.setState({
@ -435,14 +434,14 @@ export default createReactClass({
this.onResize(); this.onResize();
document.addEventListener("keydown", this.onNativeKeyDown); document.addEventListener("keydown", this.onNativeKeyDown);
}, }
shouldComponentUpdate: function(nextProps, nextState) { shouldComponentUpdate(nextProps, nextState) {
return (!ObjectUtils.shallowEqual(this.props, nextProps) || return (!ObjectUtils.shallowEqual(this.props, nextProps) ||
!ObjectUtils.shallowEqual(this.state, nextState)); !ObjectUtils.shallowEqual(this.state, nextState));
}, }
componentDidUpdate: function() { componentDidUpdate() {
if (this._roomView.current) { if (this._roomView.current) {
const roomView = this._roomView.current; const roomView = this._roomView.current;
if (!roomView.ondrop) { if (!roomView.ondrop) {
@ -464,9 +463,9 @@ export default createReactClass({
atEndOfLiveTimeline: this._messagePanel.isAtEndOfLiveTimeline(), atEndOfLiveTimeline: this._messagePanel.isAtEndOfLiveTimeline(),
}); });
} }
}, }
componentWillUnmount: function() { componentWillUnmount() {
// set a boolean to say we've been unmounted, which any pending // set a boolean to say we've been unmounted, which any pending
// promises can use to throw away their results. // promises can use to throw away their results.
// //
@ -543,21 +542,21 @@ export default createReactClass({
// Tinter.tint(); // reset colourscheme // Tinter.tint(); // reset colourscheme
SettingsStore.unwatchSetting(this._layoutWatcherRef); SettingsStore.unwatchSetting(this._layoutWatcherRef);
}, }
onLayoutChange: function() { onLayoutChange = () => {
this.setState({ this.setState({
useIRCLayout: SettingsStore.getValue("useIRCLayout"), useIRCLayout: SettingsStore.getValue("useIRCLayout"),
}); });
}, };
_onRightPanelStoreUpdate: function() { _onRightPanelStoreUpdate = () => {
this.setState({ this.setState({
showRightPanel: RightPanelStore.getSharedInstance().isOpenForRoom, showRightPanel: RightPanelStore.getSharedInstance().isOpenForRoom,
}); });
}, };
onPageUnload(event) { onPageUnload = event => {
if (ContentMessages.sharedInstance().getCurrentUploads().length > 0) { if (ContentMessages.sharedInstance().getCurrentUploads().length > 0) {
return event.returnValue = return event.returnValue =
_t("You seem to be uploading files, are you sure you want to quit?"); _t("You seem to be uploading files, are you sure you want to quit?");
@ -565,10 +564,10 @@ export default createReactClass({
return event.returnValue = return event.returnValue =
_t("You seem to be in a call, are you sure you want to quit?"); _t("You seem to be in a call, are you sure you want to quit?");
} }
}, };
// we register global shortcuts here, they *must not conflict* with local shortcuts elsewhere or both will fire // we register global shortcuts here, they *must not conflict* with local shortcuts elsewhere or both will fire
onNativeKeyDown: function(ev) { onNativeKeyDown = ev => {
let handled = false; let handled = false;
const ctrlCmdOnly = isOnlyCtrlOrCmdKeyEvent(ev); const ctrlCmdOnly = isOnlyCtrlOrCmdKeyEvent(ev);
@ -592,9 +591,9 @@ export default createReactClass({
ev.stopPropagation(); ev.stopPropagation();
ev.preventDefault(); ev.preventDefault();
} }
}, };
onReactKeyDown: function(ev) { onReactKeyDown = ev => {
let handled = false; let handled = false;
switch (ev.key) { switch (ev.key) {
@ -613,7 +612,7 @@ export default createReactClass({
break; break;
case Key.U.toUpperCase(): case Key.U.toUpperCase():
if (isOnlyCtrlOrCmdIgnoreShiftKeyEvent(ev) && ev.shiftKey) { if (isOnlyCtrlOrCmdIgnoreShiftKeyEvent(ev) && ev.shiftKey) {
dis.dispatch({ action: "upload_file" }) dis.dispatch({ action: "upload_file" });
handled = true; handled = true;
} }
break; break;
@ -623,9 +622,9 @@ export default createReactClass({
ev.stopPropagation(); ev.stopPropagation();
ev.preventDefault(); ev.preventDefault();
} }
}, };
onAction: function(payload) { onAction = payload => {
switch (payload.action) { switch (payload.action) {
case 'message_send_failed': case 'message_send_failed':
case 'message_sent': case 'message_sent':
@ -709,9 +708,9 @@ export default createReactClass({
} }
break; break;
} }
}, };
onRoomTimeline: function(ev, room, toStartOfTimeline, removed, data) { onRoomTimeline = (ev, room, toStartOfTimeline, removed, data) => {
if (this.unmounted) return; if (this.unmounted) return;
// ignore events for other rooms // ignore events for other rooms
@ -747,51 +746,51 @@ export default createReactClass({
}); });
} }
} }
}, };
onRoomName: function(room) { onRoomName = room => {
if (this.state.room && room.roomId == this.state.room.roomId) { if (this.state.room && room.roomId == this.state.room.roomId) {
this.forceUpdate(); this.forceUpdate();
} }
}, };
onRoomRecoveryReminderDontAskAgain: function() { onRoomRecoveryReminderDontAskAgain = () => {
// Called when the option to not ask again is set: // Called when the option to not ask again is set:
// force an update to hide the recovery reminder // force an update to hide the recovery reminder
this.forceUpdate(); this.forceUpdate();
}, };
onKeyBackupStatus() { onKeyBackupStatus = () => {
// Key backup status changes affect whether the in-room recovery // Key backup status changes affect whether the in-room recovery
// reminder is displayed. // reminder is displayed.
this.forceUpdate(); this.forceUpdate();
}, };
canResetTimeline: function() { canResetTimeline = () => {
if (!this._messagePanel) { if (!this._messagePanel) {
return true; return true;
} }
return this._messagePanel.canResetTimeline(); return this._messagePanel.canResetTimeline();
}, };
// called when state.room is first initialised (either at initial load, // called when state.room is first initialised (either at initial load,
// after a successful peek, or after we join the room). // after a successful peek, or after we join the room).
_onRoomLoaded: function(room) { _onRoomLoaded = room => {
this._calculatePeekRules(room); this._calculatePeekRules(room);
this._updatePreviewUrlVisibility(room); this._updatePreviewUrlVisibility(room);
this._loadMembersIfJoined(room); this._loadMembersIfJoined(room);
this._calculateRecommendedVersion(room); this._calculateRecommendedVersion(room);
this._updateE2EStatus(room); this._updateE2EStatus(room);
this._updatePermissions(room); this._updatePermissions(room);
}, };
_calculateRecommendedVersion: async function(room) { async _calculateRecommendedVersion(room) {
this.setState({ this.setState({
upgradeRecommendation: await room.getRecommendedVersion(), upgradeRecommendation: await room.getRecommendedVersion(),
}); });
}, }
_loadMembersIfJoined: async function(room) { async _loadMembersIfJoined(room) {
// lazy load members if enabled // lazy load members if enabled
if (this.context.hasLazyLoadMembersEnabled()) { if (this.context.hasLazyLoadMembersEnabled()) {
if (room && room.getMyMembership() === 'join') { if (room && room.getMyMembership() === 'join') {
@ -808,9 +807,9 @@ export default createReactClass({
} }
} }
} }
}, }
_calculatePeekRules: function(room) { _calculatePeekRules(room) {
const guestAccessEvent = room.currentState.getStateEvents("m.room.guest_access", ""); const guestAccessEvent = room.currentState.getStateEvents("m.room.guest_access", "");
if (guestAccessEvent && guestAccessEvent.getContent().guest_access === "can_join") { if (guestAccessEvent && guestAccessEvent.getContent().guest_access === "can_join") {
this.setState({ this.setState({
@ -824,17 +823,17 @@ export default createReactClass({
canPeek: true, canPeek: true,
}); });
} }
}, }
_updatePreviewUrlVisibility: function({roomId}) { _updatePreviewUrlVisibility({roomId}) {
// URL Previews in E2EE rooms can be a privacy leak so use a different setting which is per-room explicit // URL Previews in E2EE rooms can be a privacy leak so use a different setting which is per-room explicit
const key = this.context.isRoomEncrypted(roomId) ? 'urlPreviewsEnabled_e2ee' : 'urlPreviewsEnabled'; const key = this.context.isRoomEncrypted(roomId) ? 'urlPreviewsEnabled_e2ee' : 'urlPreviewsEnabled';
this.setState({ this.setState({
showUrlPreview: SettingsStore.getValue(key, roomId), showUrlPreview: SettingsStore.getValue(key, roomId),
}); });
}, }
onRoom: function(room) { onRoom = room => {
if (!room || room.roomId !== this.state.roomId) { if (!room || room.roomId !== this.state.roomId) {
return; return;
} }
@ -843,32 +842,32 @@ export default createReactClass({
}, () => { }, () => {
this._onRoomLoaded(room); this._onRoomLoaded(room);
}); });
}, };
onDeviceVerificationChanged: function(userId, device) { onDeviceVerificationChanged = (userId, device) => {
const room = this.state.room; const room = this.state.room;
if (!room.currentState.getMember(userId)) { if (!room.currentState.getMember(userId)) {
return; return;
} }
this._updateE2EStatus(room); this._updateE2EStatus(room);
}, };
onUserVerificationChanged: function(userId, _trustStatus) { onUserVerificationChanged = (userId, _trustStatus) => {
const room = this.state.room; const room = this.state.room;
if (!room || !room.currentState.getMember(userId)) { if (!room || !room.currentState.getMember(userId)) {
return; return;
} }
this._updateE2EStatus(room); this._updateE2EStatus(room);
}, };
onCrossSigningKeysChanged: function() { onCrossSigningKeysChanged = () => {
const room = this.state.room; const room = this.state.room;
if (room) { if (room) {
this._updateE2EStatus(room); this._updateE2EStatus(room);
} }
}, };
_updateE2EStatus: async function(room) { async _updateE2EStatus(room) {
if (!this.context.isRoomEncrypted(room.roomId)) { if (!this.context.isRoomEncrypted(room.roomId)) {
return; return;
} }
@ -886,26 +885,26 @@ export default createReactClass({
this.setState({ this.setState({
e2eStatus: await shieldStatusForRoom(this.context, room), e2eStatus: await shieldStatusForRoom(this.context, room),
}); });
}, }
updateTint: function() { updateTint() {
const room = this.state.room; const room = this.state.room;
if (!room) return; if (!room) return;
console.log("Tinter.tint from updateTint"); console.log("Tinter.tint from updateTint");
const colorScheme = SettingsStore.getValue("roomColor", room.roomId); const colorScheme = SettingsStore.getValue("roomColor", room.roomId);
Tinter.tint(colorScheme.primary_color, colorScheme.secondary_color); Tinter.tint(colorScheme.primary_color, colorScheme.secondary_color);
}, }
onAccountData: function(event) { onAccountData = event => {
const type = event.getType(); const type = event.getType();
if ((type === "org.matrix.preview_urls" || type === "im.vector.web.settings") && this.state.room) { if ((type === "org.matrix.preview_urls" || type === "im.vector.web.settings") && this.state.room) {
// non-e2ee url previews are stored in legacy event type `org.matrix.room.preview_urls` // non-e2ee url previews are stored in legacy event type `org.matrix.room.preview_urls`
this._updatePreviewUrlVisibility(this.state.room); this._updatePreviewUrlVisibility(this.state.room);
} }
}, };
onRoomAccountData: function(event, room) { onRoomAccountData = (event, room) => {
if (room.roomId == this.state.roomId) { if (room.roomId == this.state.roomId) {
const type = event.getType(); const type = event.getType();
if (type === "org.matrix.room.color_scheme") { if (type === "org.matrix.room.color_scheme") {
@ -918,18 +917,18 @@ export default createReactClass({
this._updatePreviewUrlVisibility(room); this._updatePreviewUrlVisibility(room);
} }
} }
}, };
onRoomStateEvents: function(ev, state) { onRoomStateEvents = (ev, state) => {
// ignore if we don't have a room yet // ignore if we don't have a room yet
if (!this.state.room || this.state.room.roomId !== state.roomId) { if (!this.state.room || this.state.room.roomId !== state.roomId) {
return; return;
} }
this._updatePermissions(this.state.room); this._updatePermissions(this.state.room);
}, };
onRoomStateMember: function(ev, state, member) { onRoomStateMember = (ev, state, member) => {
// ignore if we don't have a room yet // ignore if we don't have a room yet
if (!this.state.room) { if (!this.state.room) {
return; return;
@ -941,17 +940,17 @@ export default createReactClass({
} }
this._updateRoomMembers(member); this._updateRoomMembers(member);
}, };
onMyMembership: function(room, membership, oldMembership) { onMyMembership = (room, membership, oldMembership) => {
if (room.roomId === this.state.roomId) { if (room.roomId === this.state.roomId) {
this.forceUpdate(); this.forceUpdate();
this._loadMembersIfJoined(room); this._loadMembersIfJoined(room);
this._updatePermissions(room); this._updatePermissions(room);
} }
}, };
_updatePermissions: function(room) { _updatePermissions(room) {
if (room) { if (room) {
const me = this.context.getUserId(); const me = this.context.getUserId();
const canReact = room.getMyMembership() === "join" && room.currentState.maySendEvent("m.reaction", me); const canReact = room.getMyMembership() === "join" && room.currentState.maySendEvent("m.reaction", me);
@ -959,11 +958,11 @@ export default createReactClass({
this.setState({canReact, canReply}); this.setState({canReact, canReply});
} }
}, }
// rate limited because a power level change will emit an event for every // rate limited because a power level change will emit an event for every
// member in the room. // member in the room.
_updateRoomMembers: rate_limited_func(function(dueToMember) { _updateRoomMembers = rate_limited_func((dueToMember) => {
// a member state changed in this room // a member state changed in this room
// refresh the conf call notification state // refresh the conf call notification state
this._updateConfCallNotification(); this._updateConfCallNotification();
@ -978,9 +977,9 @@ export default createReactClass({
this._checkIfAlone(this.state.room, memberCountInfluence); this._checkIfAlone(this.state.room, memberCountInfluence);
this._updateE2EStatus(this.state.room); this._updateE2EStatus(this.state.room);
}, 500), }, 500);
_checkIfAlone: function(room, countInfluence) { _checkIfAlone(room, countInfluence) {
let warnedAboutLonelyRoom = false; let warnedAboutLonelyRoom = false;
if (localStorage) { if (localStorage) {
warnedAboutLonelyRoom = localStorage.getItem('mx_user_alone_warned_' + this.state.room.roomId); warnedAboutLonelyRoom = localStorage.getItem('mx_user_alone_warned_' + this.state.room.roomId);
@ -993,9 +992,9 @@ export default createReactClass({
let joinedOrInvitedMemberCount = room.getJoinedMemberCount() + room.getInvitedMemberCount(); let joinedOrInvitedMemberCount = room.getJoinedMemberCount() + room.getInvitedMemberCount();
if (countInfluence) joinedOrInvitedMemberCount += countInfluence; if (countInfluence) joinedOrInvitedMemberCount += countInfluence;
this.setState({isAlone: joinedOrInvitedMemberCount === 1}); this.setState({isAlone: joinedOrInvitedMemberCount === 1});
}, }
_updateConfCallNotification: function() { _updateConfCallNotification() {
const room = this.state.room; const room = this.state.room;
if (!room || !this.props.ConferenceHandler) { if (!room || !this.props.ConferenceHandler) {
return; return;
@ -1017,7 +1016,7 @@ export default createReactClass({
confMember.membership === "join" confMember.membership === "join"
), ),
}); });
}, }
_updateDMState() { _updateDMState() {
const room = this.state.room; const room = this.state.room;
@ -1028,9 +1027,9 @@ export default createReactClass({
if (dmInviter) { if (dmInviter) {
Rooms.setDMRoom(room.roomId, dmInviter); Rooms.setDMRoom(room.roomId, dmInviter);
} }
}, }
onSearchResultsFillRequest: function(backwards) { onSearchResultsFillRequest = backwards => {
if (!backwards) { if (!backwards) {
return Promise.resolve(false); return Promise.resolve(false);
} }
@ -1043,25 +1042,25 @@ export default createReactClass({
debuglog("no more search results"); debuglog("no more search results");
return Promise.resolve(false); return Promise.resolve(false);
} }
}, };
onInviteButtonClick: function() { onInviteButtonClick = () => {
// call AddressPickerDialog // call AddressPickerDialog
dis.dispatch({ dis.dispatch({
action: 'view_invite', action: 'view_invite',
roomId: this.state.room.roomId, roomId: this.state.room.roomId,
}); });
this.setState({isAlone: false}); // there's a good chance they'll invite someone this.setState({isAlone: false}); // there's a good chance they'll invite someone
}, };
onStopAloneWarningClick: function() { onStopAloneWarningClick = () => {
if (localStorage) { if (localStorage) {
localStorage.setItem('mx_user_alone_warned_' + this.state.room.roomId, true); localStorage.setItem('mx_user_alone_warned_' + this.state.room.roomId, true);
} }
this.setState({isAlone: false}); this.setState({isAlone: false});
}, };
onJoinButtonClicked: function(ev) { onJoinButtonClicked = ev => {
// If the user is a ROU, allow them to transition to a PWLU // If the user is a ROU, allow them to transition to a PWLU
if (this.context && this.context.isGuest()) { if (this.context && this.context.isGuest()) {
// Join this room once the user has registered and logged in // Join this room once the user has registered and logged in
@ -1120,10 +1119,9 @@ export default createReactClass({
return Promise.resolve(); return Promise.resolve();
}); });
} }
};
}, onMessageListScroll = ev => {
onMessageListScroll: function(ev) {
if (this._messagePanel.isAtEndOfLiveTimeline()) { if (this._messagePanel.isAtEndOfLiveTimeline()) {
this.setState({ this.setState({
numUnreadMessages: 0, numUnreadMessages: 0,
@ -1135,9 +1133,9 @@ export default createReactClass({
}); });
} }
this._updateTopUnreadMessagesBar(); this._updateTopUnreadMessagesBar();
}, };
onDragOver: function(ev) { onDragOver = ev => {
ev.stopPropagation(); ev.stopPropagation();
ev.preventDefault(); ev.preventDefault();
@ -1154,9 +1152,9 @@ export default createReactClass({
ev.dataTransfer.dropEffect = 'copy'; ev.dataTransfer.dropEffect = 'copy';
} }
} }
}, };
onDrop: function(ev) { onDrop = ev => {
ev.stopPropagation(); ev.stopPropagation();
ev.preventDefault(); ev.preventDefault();
ContentMessages.sharedInstance().sendContentListToRoom( ContentMessages.sharedInstance().sendContentListToRoom(
@ -1164,15 +1162,15 @@ export default createReactClass({
); );
this.setState({ draggingFile: false }); this.setState({ draggingFile: false });
dis.fire(Action.FocusComposer); dis.fire(Action.FocusComposer);
}, };
onDragLeaveOrEnd: function(ev) { onDragLeaveOrEnd = ev => {
ev.stopPropagation(); ev.stopPropagation();
ev.preventDefault(); ev.preventDefault();
this.setState({ draggingFile: false }); this.setState({ draggingFile: false });
}, };
injectSticker: function(url, info, text) { injectSticker(url, info, text) {
if (this.context.isGuest()) { if (this.context.isGuest()) {
dis.dispatch({action: 'require_registration'}); dis.dispatch({action: 'require_registration'});
return; return;
@ -1185,9 +1183,9 @@ export default createReactClass({
return; return;
} }
}); });
}, }
onSearch: function(term, scope) { onSearch = (term, scope) => {
this.setState({ this.setState({
searchTerm: term, searchTerm: term,
searchScope: scope, searchScope: scope,
@ -1213,9 +1211,9 @@ export default createReactClass({
debuglog("sending search request"); debuglog("sending search request");
const searchPromise = eventSearch(term, roomId); const searchPromise = eventSearch(term, roomId);
this._handleSearchResult(searchPromise); this._handleSearchResult(searchPromise);
}, };
_handleSearchResult: function(searchPromise) { _handleSearchResult(searchPromise) {
const self = this; const self = this;
// keep a record of the current search id, so that if the search terms // keep a record of the current search id, so that if the search terms
@ -1266,9 +1264,9 @@ export default createReactClass({
searchInProgress: false, searchInProgress: false,
}); });
}); });
}, }
getSearchResultTiles: function() { getSearchResultTiles() {
const SearchResultTile = sdk.getComponent('rooms.SearchResultTile'); const SearchResultTile = sdk.getComponent('rooms.SearchResultTile');
const Spinner = sdk.getComponent("elements.Spinner"); const Spinner = sdk.getComponent("elements.Spinner");
@ -1348,20 +1346,20 @@ export default createReactClass({
onHeightChanged={onHeightChanged} />); onHeightChanged={onHeightChanged} />);
} }
return ret; return ret;
}, }
onPinnedClick: function() { onPinnedClick = () => {
const nowShowingPinned = !this.state.showingPinned; const nowShowingPinned = !this.state.showingPinned;
const roomId = this.state.room.roomId; const roomId = this.state.room.roomId;
this.setState({showingPinned: nowShowingPinned, searching: false}); this.setState({showingPinned: nowShowingPinned, searching: false});
SettingsStore.setValue("PinnedEvents.isOpen", roomId, SettingLevel.ROOM_DEVICE, nowShowingPinned); SettingsStore.setValue("PinnedEvents.isOpen", roomId, SettingLevel.ROOM_DEVICE, nowShowingPinned);
}, };
onSettingsClick: function() { onSettingsClick = () => {
dis.dispatch({ action: 'open_room_settings' }); dis.dispatch({ action: 'open_room_settings' });
}, };
onCancelClick: function() { onCancelClick = () => {
console.log("updateTint from onCancelClick"); console.log("updateTint from onCancelClick");
this.updateTint(); this.updateTint();
if (this.state.forwardingEvent) { if (this.state.forwardingEvent) {
@ -1371,23 +1369,23 @@ export default createReactClass({
}); });
} }
dis.fire(Action.FocusComposer); dis.fire(Action.FocusComposer);
}, };
onLeaveClick: function() { onLeaveClick = () => {
dis.dispatch({ dis.dispatch({
action: 'leave_room', action: 'leave_room',
room_id: this.state.room.roomId, room_id: this.state.room.roomId,
}); });
}, };
onForgetClick: function() { onForgetClick = () => {
dis.dispatch({ dis.dispatch({
action: 'forget_room', action: 'forget_room',
room_id: this.state.room.roomId, room_id: this.state.room.roomId,
}); });
}, };
onRejectButtonClicked: function(ev) { onRejectButtonClicked = ev => {
const self = this; const self = this;
this.setState({ this.setState({
rejecting: true, rejecting: true,
@ -1412,9 +1410,9 @@ export default createReactClass({
rejectError: error, rejectError: error,
}); });
}); });
}, };
onRejectAndIgnoreClick: async function() { onRejectAndIgnoreClick = async () => {
this.setState({ this.setState({
rejecting: true, rejecting: true,
}); });
@ -1446,49 +1444,49 @@ export default createReactClass({
rejectError: error, rejectError: error,
}); });
} }
}, };
onRejectThreepidInviteButtonClicked: function(ev) { onRejectThreepidInviteButtonClicked = ev => {
// We can reject 3pid invites in the same way that we accept them, // We can reject 3pid invites in the same way that we accept them,
// using /leave rather than /join. In the short term though, we // using /leave rather than /join. In the short term though, we
// just ignore them. // just ignore them.
// https://github.com/vector-im/vector-web/issues/1134 // https://github.com/vector-im/vector-web/issues/1134
dis.fire(Action.ViewRoomDirectory); dis.fire(Action.ViewRoomDirectory);
}, };
onSearchClick: function() { onSearchClick = () => {
this.setState({ this.setState({
searching: !this.state.searching, searching: !this.state.searching,
showingPinned: false, showingPinned: false,
}); });
}, };
onCancelSearchClick: function() { onCancelSearchClick = () => {
this.setState({ this.setState({
searching: false, searching: false,
searchResults: null, searchResults: null,
}); });
}, };
// jump down to the bottom of this room, where new events are arriving // jump down to the bottom of this room, where new events are arriving
jumpToLiveTimeline: function() { jumpToLiveTimeline = () => {
this._messagePanel.jumpToLiveTimeline(); this._messagePanel.jumpToLiveTimeline();
dis.fire(Action.FocusComposer); dis.fire(Action.FocusComposer);
}, };
// jump up to wherever our read marker is // jump up to wherever our read marker is
jumpToReadMarker: function() { jumpToReadMarker = () => {
this._messagePanel.jumpToReadMarker(); this._messagePanel.jumpToReadMarker();
}, };
// update the read marker to match the read-receipt // update the read marker to match the read-receipt
forgetReadMarker: function(ev) { forgetReadMarker = ev => {
ev.stopPropagation(); ev.stopPropagation();
this._messagePanel.forgetReadMarker(); this._messagePanel.forgetReadMarker();
}, };
// decide whether or not the top 'unread messages' bar should be shown // decide whether or not the top 'unread messages' bar should be shown
_updateTopUnreadMessagesBar: function() { _updateTopUnreadMessagesBar = () => {
if (!this._messagePanel) { if (!this._messagePanel) {
return; return;
} }
@ -1497,12 +1495,12 @@ export default createReactClass({
if (this.state.showTopUnreadMessagesBar != showBar) { if (this.state.showTopUnreadMessagesBar != showBar) {
this.setState({showTopUnreadMessagesBar: showBar}); this.setState({showTopUnreadMessagesBar: showBar});
} }
}, };
// get the current scroll position of the room, so that it can be // get the current scroll position of the room, so that it can be
// restored when we switch back to it. // restored when we switch back to it.
// //
_getScrollState: function() { _getScrollState() {
const messagePanel = this._messagePanel; const messagePanel = this._messagePanel;
if (!messagePanel) return null; if (!messagePanel) return null;
@ -1537,9 +1535,9 @@ export default createReactClass({
focussedEvent: scrollState.trackedScrollToken, focussedEvent: scrollState.trackedScrollToken,
pixelOffset: scrollState.pixelOffset, pixelOffset: scrollState.pixelOffset,
}; };
}, }
onResize: function() { onResize = () => {
// It seems flexbox doesn't give us a way to constrain the auxPanel height to have // It seems flexbox doesn't give us a way to constrain the auxPanel height to have
// a minimum of the height of the video element, whilst also capping it from pushing out the page // a minimum of the height of the video element, whilst also capping it from pushing out the page
// so we have to do it via JS instead. In this implementation we cap the height by putting // so we have to do it via JS instead. In this implementation we cap the height by putting
@ -1557,16 +1555,16 @@ export default createReactClass({
if (auxPanelMaxHeight < 50) auxPanelMaxHeight = 50; if (auxPanelMaxHeight < 50) auxPanelMaxHeight = 50;
this.setState({auxPanelMaxHeight: auxPanelMaxHeight}); this.setState({auxPanelMaxHeight: auxPanelMaxHeight});
}, };
onFullscreenClick: function() { onFullscreenClick = () => {
dis.dispatch({ dis.dispatch({
action: 'video_fullscreen', action: 'video_fullscreen',
fullscreen: true, fullscreen: true,
}, true); }, true);
}, };
onMuteAudioClick: function() { onMuteAudioClick = () => {
const call = this._getCallForRoom(); const call = this._getCallForRoom();
if (!call) { if (!call) {
return; return;
@ -1574,9 +1572,9 @@ export default createReactClass({
const newState = !call.isMicrophoneMuted(); const newState = !call.isMicrophoneMuted();
call.setMicrophoneMuted(newState); call.setMicrophoneMuted(newState);
this.forceUpdate(); // TODO: just update the voip buttons this.forceUpdate(); // TODO: just update the voip buttons
}, };
onMuteVideoClick: function() { onMuteVideoClick = () => {
const call = this._getCallForRoom(); const call = this._getCallForRoom();
if (!call) { if (!call) {
return; return;
@ -1584,29 +1582,29 @@ export default createReactClass({
const newState = !call.isLocalVideoMuted(); const newState = !call.isLocalVideoMuted();
call.setLocalVideoMuted(newState); call.setLocalVideoMuted(newState);
this.forceUpdate(); // TODO: just update the voip buttons this.forceUpdate(); // TODO: just update the voip buttons
}, };
onStatusBarVisible: function() { onStatusBarVisible = () => {
if (this.unmounted) return; if (this.unmounted) return;
this.setState({ this.setState({
statusBarVisible: true, statusBarVisible: true,
}); });
}, };
onStatusBarHidden: function() { onStatusBarHidden = () => {
// This is currently not desired as it is annoying if it keeps expanding and collapsing // This is currently not desired as it is annoying if it keeps expanding and collapsing
if (this.unmounted) return; if (this.unmounted) return;
this.setState({ this.setState({
statusBarVisible: false, statusBarVisible: false,
}); });
}, };
/** /**
* called by the parent component when PageUp/Down/etc is pressed. * called by the parent component when PageUp/Down/etc is pressed.
* *
* We pass it down to the scroll panel. * We pass it down to the scroll panel.
*/ */
handleScrollKey: function(ev) { handleScrollKey = ev => {
let panel; let panel;
if (this._searchResultsPanel.current) { if (this._searchResultsPanel.current) {
panel = this._searchResultsPanel.current; panel = this._searchResultsPanel.current;
@ -1617,48 +1615,48 @@ export default createReactClass({
if (panel) { if (panel) {
panel.handleScrollKey(ev); panel.handleScrollKey(ev);
} }
}, };
/** /**
* get any current call for this room * get any current call for this room
*/ */
_getCallForRoom: function() { _getCallForRoom() {
if (!this.state.room) { if (!this.state.room) {
return null; return null;
} }
return CallHandler.getCallForRoom(this.state.room.roomId); return CallHandler.getCallForRoom(this.state.room.roomId);
}, }
// this has to be a proper method rather than an unnamed function, // this has to be a proper method rather than an unnamed function,
// otherwise react calls it with null on each update. // otherwise react calls it with null on each update.
_gatherTimelinePanelRef: function(r) { _gatherTimelinePanelRef = r => {
this._messagePanel = r; this._messagePanel = r;
if (r) { if (r) {
console.log("updateTint from RoomView._gatherTimelinePanelRef"); console.log("updateTint from RoomView._gatherTimelinePanelRef");
this.updateTint(); this.updateTint();
} }
}, };
_getOldRoom: function() { _getOldRoom() {
const createEvent = this.state.room.currentState.getStateEvents("m.room.create", ""); const createEvent = this.state.room.currentState.getStateEvents("m.room.create", "");
if (!createEvent || !createEvent.getContent()['predecessor']) return null; if (!createEvent || !createEvent.getContent()['predecessor']) return null;
return this.context.getRoom(createEvent.getContent()['predecessor']['room_id']); return this.context.getRoom(createEvent.getContent()['predecessor']['room_id']);
}, }
_getHiddenHighlightCount: function() { _getHiddenHighlightCount() {
const oldRoom = this._getOldRoom(); const oldRoom = this._getOldRoom();
if (!oldRoom) return 0; if (!oldRoom) return 0;
return oldRoom.getUnreadNotificationCount('highlight'); return oldRoom.getUnreadNotificationCount('highlight');
}, }
_onHiddenHighlightsClick: function() { _onHiddenHighlightsClick = () => {
const oldRoom = this._getOldRoom(); const oldRoom = this._getOldRoom();
if (!oldRoom) return; if (!oldRoom) return;
dis.dispatch({action: "view_room", room_id: oldRoom.roomId}); dis.dispatch({action: "view_room", room_id: oldRoom.roomId});
}, };
render: function() { render() {
const RoomHeader = sdk.getComponent('rooms.RoomHeader'); const RoomHeader = sdk.getComponent('rooms.RoomHeader');
const ForwardMessage = sdk.getComponent("rooms.ForwardMessage"); const ForwardMessage = sdk.getComponent("rooms.ForwardMessage");
const AuxPanel = sdk.getComponent("rooms.AuxPanel"); const AuxPanel = sdk.getComponent("rooms.AuxPanel");
@ -2064,7 +2062,7 @@ export default createReactClass({
const showRightPanel = !forceHideRightPanel && this.state.room && this.state.showRightPanel; const showRightPanel = !forceHideRightPanel && this.state.room && this.state.showRightPanel;
const rightPanel = showRightPanel const rightPanel = showRightPanel
? <RightPanel roomId={this.state.room.roomId} resizeNotifier={this.props.resizeNotifier} /> ? <RightPanel room={this.state.room} resizeNotifier={this.props.resizeNotifier} />
: null; : null;
const timelineClasses = classNames("mx_RoomView_timeline", { const timelineClasses = classNames("mx_RoomView_timeline", {
@ -2118,5 +2116,5 @@ export default createReactClass({
</main> </main>
</RoomContext.Provider> </RoomContext.Provider>
); );
}, }
}); }

View file

@ -15,7 +15,6 @@ limitations under the License.
*/ */
import React, {createRef} from "react"; import React, {createRef} from "react";
import createReactClass from 'create-react-class';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import { Key } from '../../Keyboard'; import { Key } from '../../Keyboard';
import Timer from '../../utils/Timer'; import Timer from '../../utils/Timer';
@ -84,10 +83,8 @@ if (DEBUG_SCROLL) {
* offset as normal. * offset as normal.
*/ */
export default createReactClass({ export default class ScrollPanel extends React.Component {
displayName: 'ScrollPanel', static propTypes = {
propTypes: {
/* stickyBottom: if set to true, then once the user hits the bottom of /* stickyBottom: if set to true, then once the user hits the bottom of
* the list, any new children added to the list will cause the list to * the list, any new children added to the list will cause the list to
* scroll down to show the new element, rather than preserving the * scroll down to show the new element, rather than preserving the
@ -97,7 +94,7 @@ export default createReactClass({
/* startAtBottom: if set to true, the view is assumed to start /* startAtBottom: if set to true, the view is assumed to start
* scrolled to the bottom. * scrolled to the bottom.
* XXX: It's likley this is unecessary and can be derived from * XXX: It's likely this is unnecessary and can be derived from
* stickyBottom, but I'm adding an extra parameter to ensure * stickyBottom, but I'm adding an extra parameter to ensure
* behaviour stays the same for other uses of ScrollPanel. * behaviour stays the same for other uses of ScrollPanel.
* If so, let's remove this parameter down the line. * If so, let's remove this parameter down the line.
@ -141,6 +138,7 @@ export default createReactClass({
/* style: styles to add to the top-level div /* style: styles to add to the top-level div
*/ */
style: PropTypes.object, style: PropTypes.object,
/* resizeNotifier: ResizeNotifier to know when middle column has changed size /* resizeNotifier: ResizeNotifier to know when middle column has changed size
*/ */
resizeNotifier: PropTypes.object, resizeNotifier: PropTypes.object,
@ -149,20 +147,19 @@ export default createReactClass({
* of the wrapper * of the wrapper
*/ */
fixedChildren: PropTypes.node, fixedChildren: PropTypes.node,
}, };
getDefaultProps: function() { static defaultProps = {
return {
stickyBottom: true, stickyBottom: true,
startAtBottom: true, startAtBottom: true,
onFillRequest: function(backwards) { return Promise.resolve(false); }, onFillRequest: function(backwards) { return Promise.resolve(false); },
onUnfillRequest: function(backwards, scrollToken) {}, onUnfillRequest: function(backwards, scrollToken) {},
onScroll: function() {}, onScroll: function() {},
}; };
},
// TODO: [REACT-WARNING] Replace component with real class, use constructor for refs constructor(props) {
UNSAFE_componentWillMount: function() { super(props);
this._pendingFillRequests = {b: null, f: null}; this._pendingFillRequests = {b: null, f: null};
if (this.props.resizeNotifier) { if (this.props.resizeNotifier) {
@ -172,13 +169,13 @@ export default createReactClass({
this.resetScrollState(); this.resetScrollState();
this._itemlist = createRef(); this._itemlist = createRef();
}, }
componentDidMount: function() { componentDidMount() {
this.checkScroll(); this.checkScroll();
}, }
componentDidUpdate: function() { componentDidUpdate() {
// after adding event tiles, we may need to tweak the scroll (either to // after adding event tiles, we may need to tweak the scroll (either to
// keep at the bottom of the timeline, or to maintain the view after // keep at the bottom of the timeline, or to maintain the view after
// adding events to the top). // adding events to the top).
@ -186,9 +183,9 @@ export default createReactClass({
// This will also re-check the fill state, in case the paginate was inadequate // This will also re-check the fill state, in case the paginate was inadequate
this.checkScroll(); this.checkScroll();
this.updatePreventShrinking(); this.updatePreventShrinking();
}, }
componentWillUnmount: function() { componentWillUnmount() {
// set a boolean to say we've been unmounted, which any pending // set a boolean to say we've been unmounted, which any pending
// promises can use to throw away their results. // promises can use to throw away their results.
// //
@ -198,41 +195,41 @@ export default createReactClass({
if (this.props.resizeNotifier) { if (this.props.resizeNotifier) {
this.props.resizeNotifier.removeListener("middlePanelResized", this.onResize); this.props.resizeNotifier.removeListener("middlePanelResized", this.onResize);
} }
}, }
onScroll: function(ev) { onScroll = ev => {
debuglog("onScroll", this._getScrollNode().scrollTop); debuglog("onScroll", this._getScrollNode().scrollTop);
this._scrollTimeout.restart(); this._scrollTimeout.restart();
this._saveScrollState(); this._saveScrollState();
this.updatePreventShrinking(); this.updatePreventShrinking();
this.props.onScroll(ev); this.props.onScroll(ev);
this.checkFillState(); this.checkFillState();
}, };
onResize: function() { onResize = () => {
this.checkScroll(); this.checkScroll();
// update preventShrinkingState if present // update preventShrinkingState if present
if (this.preventShrinkingState) { if (this.preventShrinkingState) {
this.preventShrinking(); this.preventShrinking();
} }
}, };
// after an update to the contents of the panel, check that the scroll is // after an update to the contents of the panel, check that the scroll is
// where it ought to be, and set off pagination requests if necessary. // where it ought to be, and set off pagination requests if necessary.
checkScroll: function() { checkScroll = () => {
if (this.unmounted) { if (this.unmounted) {
return; return;
} }
this._restoreSavedScrollState(); this._restoreSavedScrollState();
this.checkFillState(); this.checkFillState();
}, };
// return true if the content is fully scrolled down right now; else false. // return true if the content is fully scrolled down right now; else false.
// //
// note that this is independent of the 'stuckAtBottom' state - it is simply // note that this is independent of the 'stuckAtBottom' state - it is simply
// about whether the content is scrolled down right now, irrespective of // about whether the content is scrolled down right now, irrespective of
// whether it will stay that way when the children update. // whether it will stay that way when the children update.
isAtBottom: function() { isAtBottom = () => {
const sn = this._getScrollNode(); const sn = this._getScrollNode();
// fractional values (both too big and too small) // fractional values (both too big and too small)
// for scrollTop happen on certain browsers/platforms // for scrollTop happen on certain browsers/platforms
@ -240,7 +237,7 @@ export default createReactClass({
// so check difference <= 1; // so check difference <= 1;
return Math.abs(sn.scrollHeight - (sn.scrollTop + sn.clientHeight)) <= 1; return Math.abs(sn.scrollHeight - (sn.scrollTop + sn.clientHeight)) <= 1;
}, };
// returns the vertical height in the given direction that can be removed from // returns the vertical height in the given direction that can be removed from
// the content box (which has a height of scrollHeight, see checkFillState) without // the content box (which has a height of scrollHeight, see checkFillState) without
@ -273,7 +270,7 @@ export default createReactClass({
// |#########| - | // |#########| - |
// |#########| | // |#########| |
// `---------' - // `---------' -
_getExcessHeight: function(backwards) { _getExcessHeight(backwards) {
const sn = this._getScrollNode(); const sn = this._getScrollNode();
const contentHeight = this._getMessagesHeight(); const contentHeight = this._getMessagesHeight();
const listHeight = this._getListHeight(); const listHeight = this._getListHeight();
@ -285,10 +282,10 @@ export default createReactClass({
} else { } else {
return contentHeight - (unclippedScrollTop + 2*sn.clientHeight) - UNPAGINATION_PADDING; return contentHeight - (unclippedScrollTop + 2*sn.clientHeight) - UNPAGINATION_PADDING;
} }
}, }
// check the scroll state and send out backfill requests if necessary. // check the scroll state and send out backfill requests if necessary.
checkFillState: async function(depth=0) { checkFillState = async (depth=0) => {
if (this.unmounted) { if (this.unmounted) {
return; return;
} }
@ -368,10 +365,10 @@ export default createReactClass({
this._fillRequestWhileRunning = false; this._fillRequestWhileRunning = false;
this.checkFillState(); this.checkFillState();
} }
}, };
// check if unfilling is possible and send an unfill request if necessary // check if unfilling is possible and send an unfill request if necessary
_checkUnfillState: function(backwards) { _checkUnfillState(backwards) {
let excessHeight = this._getExcessHeight(backwards); let excessHeight = this._getExcessHeight(backwards);
if (excessHeight <= 0) { if (excessHeight <= 0) {
return; return;
@ -417,10 +414,10 @@ export default createReactClass({
this.props.onUnfillRequest(backwards, markerScrollToken); this.props.onUnfillRequest(backwards, markerScrollToken);
}, UNFILL_REQUEST_DEBOUNCE_MS); }, UNFILL_REQUEST_DEBOUNCE_MS);
} }
}, }
// check if there is already a pending fill request. If not, set one off. // check if there is already a pending fill request. If not, set one off.
_maybeFill: function(depth, backwards) { _maybeFill(depth, backwards) {
const dir = backwards ? 'b' : 'f'; const dir = backwards ? 'b' : 'f';
if (this._pendingFillRequests[dir]) { if (this._pendingFillRequests[dir]) {
debuglog("Already a "+dir+" fill in progress - not starting another"); debuglog("Already a "+dir+" fill in progress - not starting another");
@ -456,7 +453,7 @@ export default createReactClass({
return this.checkFillState(depth + 1); return this.checkFillState(depth + 1);
} }
}); });
}, }
/* get the current scroll state. This returns an object with the following /* get the current scroll state. This returns an object with the following
* properties: * properties:
@ -472,9 +469,7 @@ export default createReactClass({
* the number of pixels the bottom of the tracked child is above the * the number of pixels the bottom of the tracked child is above the
* bottom of the scroll panel. * bottom of the scroll panel.
*/ */
getScrollState: function() { getScrollState = () => this.scrollState;
return this.scrollState;
},
/* reset the saved scroll state. /* reset the saved scroll state.
* *
@ -488,7 +483,7 @@ export default createReactClass({
* no use if no children exist yet, or if you are about to replace the * no use if no children exist yet, or if you are about to replace the
* child list.) * child list.)
*/ */
resetScrollState: function() { resetScrollState = () => {
this.scrollState = { this.scrollState = {
stuckAtBottom: this.props.startAtBottom, stuckAtBottom: this.props.startAtBottom,
}; };
@ -496,20 +491,20 @@ export default createReactClass({
this._pages = 0; this._pages = 0;
this._scrollTimeout = new Timer(100); this._scrollTimeout = new Timer(100);
this._heightUpdateInProgress = false; this._heightUpdateInProgress = false;
}, };
/** /**
* jump to the top of the content. * jump to the top of the content.
*/ */
scrollToTop: function() { scrollToTop = () => {
this._getScrollNode().scrollTop = 0; this._getScrollNode().scrollTop = 0;
this._saveScrollState(); this._saveScrollState();
}, };
/** /**
* jump to the bottom of the content. * jump to the bottom of the content.
*/ */
scrollToBottom: function() { scrollToBottom = () => {
// the easiest way to make sure that the scroll state is correctly // the easiest way to make sure that the scroll state is correctly
// saved is to do the scroll, then save the updated state. (Calculating // saved is to do the scroll, then save the updated state. (Calculating
// it ourselves is hard, and we can't rely on an onScroll callback // it ourselves is hard, and we can't rely on an onScroll callback
@ -517,25 +512,25 @@ export default createReactClass({
const sn = this._getScrollNode(); const sn = this._getScrollNode();
sn.scrollTop = sn.scrollHeight; sn.scrollTop = sn.scrollHeight;
this._saveScrollState(); this._saveScrollState();
}, };
/** /**
* Page up/down. * Page up/down.
* *
* @param {number} mult: -1 to page up, +1 to page down * @param {number} mult: -1 to page up, +1 to page down
*/ */
scrollRelative: function(mult) { scrollRelative = mult => {
const scrollNode = this._getScrollNode(); const scrollNode = this._getScrollNode();
const delta = mult * scrollNode.clientHeight * 0.5; const delta = mult * scrollNode.clientHeight * 0.5;
scrollNode.scrollBy(0, delta); scrollNode.scrollBy(0, delta);
this._saveScrollState(); this._saveScrollState();
}, };
/** /**
* Scroll up/down in response to a scroll key * Scroll up/down in response to a scroll key
* @param {object} ev the keyboard event * @param {object} ev the keyboard event
*/ */
handleScrollKey: function(ev) { handleScrollKey = ev => {
switch (ev.key) { switch (ev.key) {
case Key.PAGE_UP: case Key.PAGE_UP:
if (!ev.ctrlKey && !ev.shiftKey && !ev.altKey && !ev.metaKey) { if (!ev.ctrlKey && !ev.shiftKey && !ev.altKey && !ev.metaKey) {
@ -561,7 +556,7 @@ export default createReactClass({
} }
break; break;
} }
}, };
/* Scroll the panel to bring the DOM node with the scroll token /* Scroll the panel to bring the DOM node with the scroll token
* `scrollToken` into view. * `scrollToken` into view.
@ -574,7 +569,7 @@ export default createReactClass({
* node (specifically, the bottom of it) will be positioned. If omitted, it * node (specifically, the bottom of it) will be positioned. If omitted, it
* defaults to 0. * defaults to 0.
*/ */
scrollToToken: function(scrollToken, pixelOffset, offsetBase) { scrollToToken = (scrollToken, pixelOffset, offsetBase) => {
pixelOffset = pixelOffset || 0; pixelOffset = pixelOffset || 0;
offsetBase = offsetBase || 0; offsetBase = offsetBase || 0;
@ -596,9 +591,9 @@ export default createReactClass({
scrollNode.scrollTop = (trackedNode.offsetTop - (scrollNode.clientHeight * offsetBase)) + pixelOffset; scrollNode.scrollTop = (trackedNode.offsetTop - (scrollNode.clientHeight * offsetBase)) + pixelOffset;
this._saveScrollState(); this._saveScrollState();
} }
}, };
_saveScrollState: function() { _saveScrollState() {
if (this.props.stickyBottom && this.isAtBottom()) { if (this.props.stickyBottom && this.isAtBottom()) {
this.scrollState = { stuckAtBottom: true }; this.scrollState = { stuckAtBottom: true };
debuglog("saved stuckAtBottom state"); debuglog("saved stuckAtBottom state");
@ -641,9 +636,9 @@ export default createReactClass({
bottomOffset: bottomOffset, bottomOffset: bottomOffset,
pixelOffset: bottomOffset - viewportBottom, //needed for restoring the scroll position when coming back to the room pixelOffset: bottomOffset - viewportBottom, //needed for restoring the scroll position when coming back to the room
}; };
}, }
_restoreSavedScrollState: async function() { async _restoreSavedScrollState() {
const scrollState = this.scrollState; const scrollState = this.scrollState;
if (scrollState.stuckAtBottom) { if (scrollState.stuckAtBottom) {
@ -676,7 +671,8 @@ export default createReactClass({
} else { } else {
debuglog("not updating height because request already in progress"); debuglog("not updating height because request already in progress");
} }
}, }
// need a better name that also indicates this will change scrollTop? Rebalance height? Reveal content? // need a better name that also indicates this will change scrollTop? Rebalance height? Reveal content?
async _updateHeight() { async _updateHeight() {
// wait until user has stopped scrolling // wait until user has stopped scrolling
@ -731,7 +727,7 @@ export default createReactClass({
debuglog("updateHeight to", {newHeight, topDiff}); debuglog("updateHeight to", {newHeight, topDiff});
} }
} }
}, }
_getTrackedNode() { _getTrackedNode() {
const scrollState = this.scrollState; const scrollState = this.scrollState;
@ -764,11 +760,11 @@ export default createReactClass({
} }
return scrollState.trackedNode; return scrollState.trackedNode;
}, }
_getListHeight() { _getListHeight() {
return this._bottomGrowth + (this._pages * PAGE_SIZE); return this._bottomGrowth + (this._pages * PAGE_SIZE);
}, }
_getMessagesHeight() { _getMessagesHeight() {
const itemlist = this._itemlist.current; const itemlist = this._itemlist.current;
@ -777,17 +773,17 @@ export default createReactClass({
const firstNodeTop = itemlist.firstElementChild ? itemlist.firstElementChild.offsetTop : 0; const firstNodeTop = itemlist.firstElementChild ? itemlist.firstElementChild.offsetTop : 0;
// 18 is itemlist padding // 18 is itemlist padding
return lastNodeBottom - firstNodeTop + (18 * 2); return lastNodeBottom - firstNodeTop + (18 * 2);
}, }
_topFromBottom(node) { _topFromBottom(node) {
// current capped height - distance from top = distance from bottom of container to top of tracked element // current capped height - distance from top = distance from bottom of container to top of tracked element
return this._itemlist.current.clientHeight - node.offsetTop; return this._itemlist.current.clientHeight - node.offsetTop;
}, }
/* get the DOM node which has the scrollTop property we care about for our /* get the DOM node which has the scrollTop property we care about for our
* message panel. * message panel.
*/ */
_getScrollNode: function() { _getScrollNode() {
if (this.unmounted) { if (this.unmounted) {
// this shouldn't happen, but when it does, turn the NPE into // this shouldn't happen, but when it does, turn the NPE into
// something more meaningful. // something more meaningful.
@ -801,18 +797,18 @@ export default createReactClass({
} }
return this._divScroll; return this._divScroll;
}, }
_collectScroll: function(divScroll) { _collectScroll = divScroll => {
this._divScroll = divScroll; this._divScroll = divScroll;
}, };
/** /**
Mark the bottom offset of the last tile so we can balance it out when Mark the bottom offset of the last tile so we can balance it out when
anything below it changes, by calling updatePreventShrinking, to keep anything below it changes, by calling updatePreventShrinking, to keep
the same minimum bottom offset, effectively preventing the timeline to shrink. the same minimum bottom offset, effectively preventing the timeline to shrink.
*/ */
preventShrinking: function() { preventShrinking = () => {
const messageList = this._itemlist.current; const messageList = this._itemlist.current;
const tiles = messageList && messageList.children; const tiles = messageList && messageList.children;
if (!messageList) { if (!messageList) {
@ -836,16 +832,16 @@ export default createReactClass({
offsetNode: lastTileNode, offsetNode: lastTileNode,
}; };
debuglog("prevent shrinking, last tile ", offsetFromBottom, "px from bottom"); debuglog("prevent shrinking, last tile ", offsetFromBottom, "px from bottom");
}, };
/** Clear shrinking prevention. Used internally, and when the timeline is reloaded. */ /** Clear shrinking prevention. Used internally, and when the timeline is reloaded. */
clearPreventShrinking: function() { clearPreventShrinking = () => {
const messageList = this._itemlist.current; const messageList = this._itemlist.current;
const balanceElement = messageList && messageList.parentElement; const balanceElement = messageList && messageList.parentElement;
if (balanceElement) balanceElement.style.paddingBottom = null; if (balanceElement) balanceElement.style.paddingBottom = null;
this.preventShrinkingState = null; this.preventShrinkingState = null;
debuglog("prevent shrinking cleared"); debuglog("prevent shrinking cleared");
}, };
/** /**
update the container padding to balance update the container padding to balance
@ -855,7 +851,7 @@ export default createReactClass({
from the bottom of the marked tile grows larger than from the bottom of the marked tile grows larger than
what it was when marking. what it was when marking.
*/ */
updatePreventShrinking: function() { updatePreventShrinking = () => {
if (this.preventShrinkingState) { if (this.preventShrinkingState) {
const sn = this._getScrollNode(); const sn = this._getScrollNode();
const scrollState = this.scrollState; const scrollState = this.scrollState;
@ -885,9 +881,9 @@ export default createReactClass({
this.clearPreventShrinking(); this.clearPreventShrinking();
} }
} }
}, };
render: function() { render() {
// TODO: the classnames on the div and ol could do with being updated to // TODO: the classnames on the div and ol could do with being updated to
// reflect the fact that we don't necessarily contain a list of messages. // reflect the fact that we don't necessarily contain a list of messages.
// it's not obvious why we have a separate div and ol anyway. // it's not obvious why we have a separate div and ol anyway.
@ -905,5 +901,5 @@ export default createReactClass({
</div> </div>
</AutoHideScrollbar> </AutoHideScrollbar>
); );
}, }
}); }

View file

@ -16,7 +16,6 @@ limitations under the License.
*/ */
import React, {createRef} from 'react'; import React, {createRef} from 'react';
import createReactClass from 'create-react-class';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import { Key } from '../../Keyboard'; import { Key } from '../../Keyboard';
import dis from '../../dispatcher/dispatcher'; import dis from '../../dispatcher/dispatcher';
@ -24,10 +23,8 @@ import { throttle } from 'lodash';
import AccessibleButton from '../../components/views/elements/AccessibleButton'; import AccessibleButton from '../../components/views/elements/AccessibleButton';
import classNames from 'classnames'; import classNames from 'classnames';
export default createReactClass({ export default class SearchBox extends React.Component {
displayName: 'SearchBox', static propTypes = {
propTypes: {
onSearch: PropTypes.func, onSearch: PropTypes.func,
onCleared: PropTypes.func, onCleared: PropTypes.func,
onKeyDown: PropTypes.func, onKeyDown: PropTypes.func,
@ -38,35 +35,32 @@ export default createReactClass({
// on room search focus action (it would be nicer to take // on room search focus action (it would be nicer to take
// this functionality out, but not obvious how that would work) // this functionality out, but not obvious how that would work)
enableRoomSearchFocus: PropTypes.bool, enableRoomSearchFocus: PropTypes.bool,
}, };
getDefaultProps: function() { static defaultProps = {
return {
enableRoomSearchFocus: false, enableRoomSearchFocus: false,
}; };
},
getInitialState: function() { constructor(props) {
return { super(props);
this._search = createRef();
this.state = {
searchTerm: "", searchTerm: "",
blurred: true, blurred: true,
}; };
}, }
// TODO: [REACT-WARNING] Replace component with real class, use constructor for refs componentDidMount() {
UNSAFE_componentWillMount: function() {
this._search = createRef();
},
componentDidMount: function() {
this.dispatcherRef = dis.register(this.onAction); this.dispatcherRef = dis.register(this.onAction);
}, }
componentWillUnmount: function() { componentWillUnmount() {
dis.unregister(this.dispatcherRef); dis.unregister(this.dispatcherRef);
}, }
onAction: function(payload) { onAction = payload => {
if (!this.props.enableRoomSearchFocus) return; if (!this.props.enableRoomSearchFocus) return;
switch (payload.action) { switch (payload.action) {
@ -81,51 +75,51 @@ export default createReactClass({
} }
break; break;
} }
}, };
onChange: function() { onChange = () => {
if (!this._search.current) return; if (!this._search.current) return;
this.setState({ searchTerm: this._search.current.value }); this.setState({ searchTerm: this._search.current.value });
this.onSearch(); this.onSearch();
}, };
onSearch: throttle(function() { onSearch = throttle(() => {
this.props.onSearch(this._search.current.value); this.props.onSearch(this._search.current.value);
}, 200, {trailing: true, leading: true}), }, 200, {trailing: true, leading: true});
_onKeyDown: function(ev) { _onKeyDown = ev => {
switch (ev.key) { switch (ev.key) {
case Key.ESCAPE: case Key.ESCAPE:
this._clearSearch("keyboard"); this._clearSearch("keyboard");
break; break;
} }
if (this.props.onKeyDown) this.props.onKeyDown(ev); if (this.props.onKeyDown) this.props.onKeyDown(ev);
}, };
_onFocus: function(ev) { _onFocus = ev => {
this.setState({blurred: false}); this.setState({blurred: false});
ev.target.select(); ev.target.select();
if (this.props.onFocus) { if (this.props.onFocus) {
this.props.onFocus(ev); this.props.onFocus(ev);
} }
}, };
_onBlur: function(ev) { _onBlur = ev => {
this.setState({blurred: true}); this.setState({blurred: true});
if (this.props.onBlur) { if (this.props.onBlur) {
this.props.onBlur(ev); this.props.onBlur(ev);
} }
}, };
_clearSearch: function(source) { _clearSearch(source) {
this._search.current.value = ""; this._search.current.value = "";
this.onChange(); this.onChange();
if (this.props.onCleared) { if (this.props.onCleared) {
this.props.onCleared(source); this.props.onCleared(source);
} }
}, }
render: function() { render() {
// check for collapsed here and // check for collapsed here and
// not at parent so we keep // not at parent so we keep
// searchTerm in our state // searchTerm in our state
@ -166,5 +160,5 @@ export default createReactClass({
{ clearButton } { clearButton }
</div> </div>
); );
}, }
}); }

View file

@ -18,7 +18,6 @@ limitations under the License.
import * as React from "react"; import * as React from "react";
import {_t} from '../../languageHandler'; import {_t} from '../../languageHandler';
import * as PropTypes from "prop-types";
import * as sdk from "../../index"; import * as sdk from "../../index";
import AutoHideScrollbar from './AutoHideScrollbar'; import AutoHideScrollbar from './AutoHideScrollbar';
import { ReactNode } from "react"; import { ReactNode } from "react";

View file

@ -16,7 +16,6 @@ limitations under the License.
*/ */
import React from 'react'; import React from 'react';
import createReactClass from 'create-react-class';
import TagOrderStore from '../../stores/TagOrderStore'; import TagOrderStore from '../../stores/TagOrderStore';
import GroupActions from '../../actions/GroupActions'; import GroupActions from '../../actions/GroupActions';
@ -32,21 +31,15 @@ import AutoHideScrollbar from "./AutoHideScrollbar";
import SettingsStore from "../../settings/SettingsStore"; import SettingsStore from "../../settings/SettingsStore";
import UserTagTile from "../views/elements/UserTagTile"; import UserTagTile from "../views/elements/UserTagTile";
const TagPanel = createReactClass({ class TagPanel extends React.Component {
displayName: 'TagPanel', static contextType = MatrixClientContext;
statics: { state = {
contextType: MatrixClientContext,
},
getInitialState() {
return {
orderedTags: [], orderedTags: [],
selectedTags: [], selectedTags: [],
}; };
},
componentDidMount: function() { componentDidMount() {
this.unmounted = false; this.unmounted = false;
this.context.on("Group.myMembership", this._onGroupMyMembership); this.context.on("Group.myMembership", this._onGroupMyMembership);
this.context.on("sync", this._onClientSync); this.context.on("sync", this._onClientSync);
@ -62,7 +55,7 @@ const TagPanel = createReactClass({
}); });
// This could be done by anything with a matrix client // This could be done by anything with a matrix client
dis.dispatch(GroupActions.fetchJoinedGroups(this.context)); dis.dispatch(GroupActions.fetchJoinedGroups(this.context));
}, }
componentWillUnmount() { componentWillUnmount() {
this.unmounted = true; this.unmounted = true;
@ -71,14 +64,14 @@ const TagPanel = createReactClass({
if (this._tagOrderStoreToken) { if (this._tagOrderStoreToken) {
this._tagOrderStoreToken.remove(); this._tagOrderStoreToken.remove();
} }
}, }
_onGroupMyMembership() { _onGroupMyMembership = () => {
if (this.unmounted) return; if (this.unmounted) return;
dis.dispatch(GroupActions.fetchJoinedGroups(this.context)); dis.dispatch(GroupActions.fetchJoinedGroups(this.context));
}, };
_onClientSync(syncState, prevState) { _onClientSync = (syncState, prevState) => {
// Consider the client reconnected if there is no error with syncing. // Consider the client reconnected if there is no error with syncing.
// This means the state could be RECONNECTING, SYNCING, PREPARED or CATCHUP. // This means the state could be RECONNECTING, SYNCING, PREPARED or CATCHUP.
const reconnected = syncState !== "ERROR" && prevState !== syncState; const reconnected = syncState !== "ERROR" && prevState !== syncState;
@ -86,18 +79,18 @@ const TagPanel = createReactClass({
// Load joined groups // Load joined groups
dis.dispatch(GroupActions.fetchJoinedGroups(this.context)); dis.dispatch(GroupActions.fetchJoinedGroups(this.context));
} }
}, };
onMouseDown(e) { onMouseDown = e => {
// only dispatch if its not a no-op // only dispatch if its not a no-op
if (this.state.selectedTags.length > 0) { if (this.state.selectedTags.length > 0) {
dis.dispatch({action: 'deselect_tags'}); dis.dispatch({action: 'deselect_tags'});
} }
}, };
onClearFilterClick(ev) { onClearFilterClick = ev => {
dis.dispatch({action: 'deselect_tags'}); dis.dispatch({action: 'deselect_tags'});
}, };
renderGlobalIcon() { renderGlobalIcon() {
if (!SettingsStore.getValue("feature_communities_v2_prototypes")) return null; if (!SettingsStore.getValue("feature_communities_v2_prototypes")) return null;
@ -108,7 +101,7 @@ const TagPanel = createReactClass({
<hr className="mx_TagPanel_divider" /> <hr className="mx_TagPanel_divider" />
</div> </div>
); );
}, }
render() { render() {
const DNDTagTile = sdk.getComponent('elements.DNDTagTile'); const DNDTagTile = sdk.getComponent('elements.DNDTagTile');
@ -173,6 +166,6 @@ const TagPanel = createReactClass({
</Droppable> </Droppable>
</AutoHideScrollbar> </AutoHideScrollbar>
</div>; </div>;
}, }
}); }
export default TagPanel; export default TagPanel;

View file

@ -19,7 +19,6 @@ limitations under the License.
import SettingsStore from "../../settings/SettingsStore"; import SettingsStore from "../../settings/SettingsStore";
import React, {createRef} from 'react'; import React, {createRef} from 'react';
import createReactClass from 'create-react-class';
import ReactDOM from "react-dom"; import ReactDOM from "react-dom";
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import {EventTimeline} from "matrix-js-sdk"; import {EventTimeline} from "matrix-js-sdk";
@ -54,10 +53,8 @@ if (DEBUG) {
* *
* Also responsible for handling and sending read receipts. * Also responsible for handling and sending read receipts.
*/ */
const TimelinePanel = createReactClass({ class TimelinePanel extends React.Component {
displayName: 'TimelinePanel', static propTypes = {
propTypes: {
// The js-sdk EventTimelineSet object for the timeline sequence we are // The js-sdk EventTimelineSet object for the timeline sequence we are
// representing. This may or may not have a room, depending on what it's // representing. This may or may not have a room, depending on what it's
// a timeline representing. If it has a room, we maintain RRs etc for // a timeline representing. If it has a room, we maintain RRs etc for
@ -115,23 +112,28 @@ const TimelinePanel = createReactClass({
// whether to use the irc layout // whether to use the irc layout
useIRCLayout: PropTypes.bool, useIRCLayout: PropTypes.bool,
}, }
statics: {
// a map from room id to read marker event timestamp // a map from room id to read marker event timestamp
roomReadMarkerTsMap: {}, static roomReadMarkerTsMap = {};
},
getDefaultProps: function() { static defaultProps = {
return {
// By default, disable the timelineCap in favour of unpaginating based on // By default, disable the timelineCap in favour of unpaginating based on
// event tile heights. (See _unpaginateEvents) // event tile heights. (See _unpaginateEvents)
timelineCap: Number.MAX_VALUE, timelineCap: Number.MAX_VALUE,
className: 'mx_RoomView_messagePanel', className: 'mx_RoomView_messagePanel',
}; };
},
getInitialState: function() { constructor(props) {
super(props);
debuglog("TimelinePanel: mounting");
this.lastRRSentEventId = undefined;
this.lastRMSentEventId = undefined;
this._messagePanel = createRef();
// XXX: we could track RM per TimelineSet rather than per Room. // XXX: we could track RM per TimelineSet rather than per Room.
// but for now we just do it per room for simplicity. // but for now we just do it per room for simplicity.
let initialReadMarker = null; let initialReadMarker = null;
@ -144,7 +146,7 @@ const TimelinePanel = createReactClass({
} }
} }
return { this.state = {
events: [], events: [],
liveEvents: [], liveEvents: [],
timelineLoading: true, // track whether our room timeline is loading timelineLoading: true, // track whether our room timeline is loading
@ -203,24 +205,6 @@ const TimelinePanel = createReactClass({
// how long to show the RM for when it's scrolled off-screen // how long to show the RM for when it's scrolled off-screen
readMarkerOutOfViewThresholdMs: SettingsStore.getValue("readMarkerOutOfViewThresholdMs"), readMarkerOutOfViewThresholdMs: SettingsStore.getValue("readMarkerOutOfViewThresholdMs"),
}; };
},
// TODO: [REACT-WARNING] Replace component with real class, use constructor for refs
UNSAFE_componentWillMount: function() {
debuglog("TimelinePanel: mounting");
this.lastRRSentEventId = undefined;
this.lastRMSentEventId = undefined;
this._messagePanel = createRef();
if (this.props.manageReadReceipts) {
this.updateReadReceiptOnUserActivity();
}
if (this.props.manageReadMarkers) {
this.updateReadMarkerOnUserActivity();
}
this.dispatcherRef = dis.register(this.onAction); this.dispatcherRef = dis.register(this.onAction);
MatrixClientPeg.get().on("Room.timeline", this.onRoomTimeline); MatrixClientPeg.get().on("Room.timeline", this.onRoomTimeline);
@ -234,12 +218,24 @@ const TimelinePanel = createReactClass({
MatrixClientPeg.get().on("Event.decrypted", this.onEventDecrypted); MatrixClientPeg.get().on("Event.decrypted", this.onEventDecrypted);
MatrixClientPeg.get().on("Event.replaced", this.onEventReplaced); MatrixClientPeg.get().on("Event.replaced", this.onEventReplaced);
MatrixClientPeg.get().on("sync", this.onSync); MatrixClientPeg.get().on("sync", this.onSync);
}
// TODO: [REACT-WARNING] Move into constructor
// eslint-disable-next-line camelcase
UNSAFE_componentWillMount() {
if (this.props.manageReadReceipts) {
this.updateReadReceiptOnUserActivity();
}
if (this.props.manageReadMarkers) {
this.updateReadMarkerOnUserActivity();
}
this._initTimeline(this.props); this._initTimeline(this.props);
}, }
// TODO: [REACT-WARNING] Replace with appropriate lifecycle event // TODO: [REACT-WARNING] Replace with appropriate lifecycle event
UNSAFE_componentWillReceiveProps: function(newProps) { // eslint-disable-next-line camelcase
UNSAFE_componentWillReceiveProps(newProps) {
if (newProps.timelineSet !== this.props.timelineSet) { if (newProps.timelineSet !== this.props.timelineSet) {
// throw new Error("changing timelineSet on a TimelinePanel is not supported"); // throw new Error("changing timelineSet on a TimelinePanel is not supported");
@ -260,9 +256,9 @@ const TimelinePanel = createReactClass({
" (was " + this.props.eventId + ")"); " (was " + this.props.eventId + ")");
return this._initTimeline(newProps); return this._initTimeline(newProps);
} }
}, }
shouldComponentUpdate: function(nextProps, nextState) { shouldComponentUpdate(nextProps, nextState) {
if (!ObjectUtils.shallowEqual(this.props, nextProps)) { if (!ObjectUtils.shallowEqual(this.props, nextProps)) {
if (DEBUG) { if (DEBUG) {
console.group("Timeline.shouldComponentUpdate: props change"); console.group("Timeline.shouldComponentUpdate: props change");
@ -284,9 +280,9 @@ const TimelinePanel = createReactClass({
} }
return false; return false;
}, }
componentWillUnmount: function() { componentWillUnmount() {
// set a boolean to say we've been unmounted, which any pending // set a boolean to say we've been unmounted, which any pending
// promises can use to throw away their results. // promises can use to throw away their results.
// //
@ -316,9 +312,9 @@ const TimelinePanel = createReactClass({
client.removeListener("Event.replaced", this.onEventReplaced); client.removeListener("Event.replaced", this.onEventReplaced);
client.removeListener("sync", this.onSync); client.removeListener("sync", this.onSync);
} }
}, }
onMessageListUnfillRequest: function(backwards, scrollToken) { onMessageListUnfillRequest = (backwards, scrollToken) => {
// If backwards, unpaginate from the back (i.e. the start of the timeline) // If backwards, unpaginate from the back (i.e. the start of the timeline)
const dir = backwards ? EventTimeline.BACKWARDS : EventTimeline.FORWARDS; const dir = backwards ? EventTimeline.BACKWARDS : EventTimeline.FORWARDS;
debuglog("TimelinePanel: unpaginating events in direction", dir); debuglog("TimelinePanel: unpaginating events in direction", dir);
@ -349,18 +345,18 @@ const TimelinePanel = createReactClass({
firstVisibleEventIndex, firstVisibleEventIndex,
}); });
} }
}, };
onPaginationRequest(timelineWindow, direction, size) { onPaginationRequest = (timelineWindow, direction, size) => {
if (this.props.onPaginationRequest) { if (this.props.onPaginationRequest) {
return this.props.onPaginationRequest(timelineWindow, direction, size); return this.props.onPaginationRequest(timelineWindow, direction, size);
} else { } else {
return timelineWindow.paginate(direction, size); return timelineWindow.paginate(direction, size);
} }
}, };
// set off a pagination request. // set off a pagination request.
onMessageListFillRequest: function(backwards) { onMessageListFillRequest = backwards => {
if (!this._shouldPaginate()) return Promise.resolve(false); if (!this._shouldPaginate()) return Promise.resolve(false);
const dir = backwards ? EventTimeline.BACKWARDS : EventTimeline.FORWARDS; const dir = backwards ? EventTimeline.BACKWARDS : EventTimeline.FORWARDS;
@ -425,9 +421,9 @@ const TimelinePanel = createReactClass({
}); });
}); });
}); });
}, };
onMessageListScroll: function(e) { onMessageListScroll = e => {
if (this.props.onScroll) { if (this.props.onScroll) {
this.props.onScroll(e); this.props.onScroll(e);
} }
@ -447,9 +443,9 @@ const TimelinePanel = createReactClass({
// NO-OP when timeout already has set to the given value // NO-OP when timeout already has set to the given value
this._readMarkerActivityTimer.changeTimeout(timeout); this._readMarkerActivityTimer.changeTimeout(timeout);
} }
}, };
onAction: function(payload) { onAction = payload => {
if (payload.action === 'ignore_state_changed') { if (payload.action === 'ignore_state_changed') {
this.forceUpdate(); this.forceUpdate();
} }
@ -463,9 +459,9 @@ const TimelinePanel = createReactClass({
} }
}); });
} }
}, };
onRoomTimeline: function(ev, room, toStartOfTimeline, removed, data) { onRoomTimeline = (ev, room, toStartOfTimeline, removed, data) => {
// ignore events for other timeline sets // ignore events for other timeline sets
if (data.timeline.getTimelineSet() !== this.props.timelineSet) return; if (data.timeline.getTimelineSet() !== this.props.timelineSet) return;
@ -537,21 +533,19 @@ const TimelinePanel = createReactClass({
} }
}); });
}); });
}, };
onRoomTimelineReset: function(room, timelineSet) { onRoomTimelineReset = (room, timelineSet) => {
if (timelineSet !== this.props.timelineSet) return; if (timelineSet !== this.props.timelineSet) return;
if (this._messagePanel.current && this._messagePanel.current.isAtBottom()) { if (this._messagePanel.current && this._messagePanel.current.isAtBottom()) {
this._loadTimeline(); this._loadTimeline();
} }
}, };
canResetTimeline: function() { canResetTimeline = () => this._messagePanel.current && this._messagePanel.current.isAtBottom();
return this._messagePanel.current && this._messagePanel.current.isAtBottom();
},
onRoomRedaction: function(ev, room) { onRoomRedaction = (ev, room) => {
if (this.unmounted) return; if (this.unmounted) return;
// ignore events for other rooms // ignore events for other rooms
@ -560,9 +554,9 @@ const TimelinePanel = createReactClass({
// we could skip an update if the event isn't in our timeline, // we could skip an update if the event isn't in our timeline,
// but that's probably an early optimisation. // but that's probably an early optimisation.
this.forceUpdate(); this.forceUpdate();
}, };
onEventReplaced: function(replacedEvent, room) { onEventReplaced = (replacedEvent, room) => {
if (this.unmounted) return; if (this.unmounted) return;
// ignore events for other rooms // ignore events for other rooms
@ -571,27 +565,27 @@ const TimelinePanel = createReactClass({
// we could skip an update if the event isn't in our timeline, // we could skip an update if the event isn't in our timeline,
// but that's probably an early optimisation. // but that's probably an early optimisation.
this.forceUpdate(); this.forceUpdate();
}, };
onRoomReceipt: function(ev, room) { onRoomReceipt = (ev, room) => {
if (this.unmounted) return; if (this.unmounted) return;
// ignore events for other rooms // ignore events for other rooms
if (room !== this.props.timelineSet.room) return; if (room !== this.props.timelineSet.room) return;
this.forceUpdate(); this.forceUpdate();
}, };
onLocalEchoUpdated: function(ev, room, oldEventId) { onLocalEchoUpdated = (ev, room, oldEventId) => {
if (this.unmounted) return; if (this.unmounted) return;
// ignore events for other rooms // ignore events for other rooms
if (room !== this.props.timelineSet.room) return; if (room !== this.props.timelineSet.room) return;
this._reloadEvents(); this._reloadEvents();
}, };
onAccountData: function(ev, room) { onAccountData = (ev, room) => {
if (this.unmounted) return; if (this.unmounted) return;
// ignore events for other rooms // ignore events for other rooms
@ -605,9 +599,9 @@ const TimelinePanel = createReactClass({
this.setState({ this.setState({
readMarkerEventId: ev.getContent().event_id, readMarkerEventId: ev.getContent().event_id,
}, this.props.onReadMarkerUpdated); }, this.props.onReadMarkerUpdated);
}, };
onEventDecrypted: function(ev) { onEventDecrypted = ev => {
// Can be null for the notification timeline, etc. // Can be null for the notification timeline, etc.
if (!this.props.timelineSet.room) return; if (!this.props.timelineSet.room) return;
@ -620,19 +614,19 @@ const TimelinePanel = createReactClass({
if (ev.getRoomId() === this.props.timelineSet.room.roomId) { if (ev.getRoomId() === this.props.timelineSet.room.roomId) {
this.forceUpdate(); this.forceUpdate();
} }
}, };
onSync: function(state, prevState, data) { onSync = (state, prevState, data) => {
this.setState({clientSyncState: state}); this.setState({clientSyncState: state});
}, };
_readMarkerTimeout(readMarkerPosition) { _readMarkerTimeout(readMarkerPosition) {
return readMarkerPosition === 0 ? return readMarkerPosition === 0 ?
this.state.readMarkerInViewThresholdMs : this.state.readMarkerInViewThresholdMs :
this.state.readMarkerOutOfViewThresholdMs; this.state.readMarkerOutOfViewThresholdMs;
}, }
updateReadMarkerOnUserActivity: async function() { async updateReadMarkerOnUserActivity() {
const initialTimeout = this._readMarkerTimeout(this.getReadMarkerPosition()); const initialTimeout = this._readMarkerTimeout(this.getReadMarkerPosition());
this._readMarkerActivityTimer = new Timer(initialTimeout); this._readMarkerActivityTimer = new Timer(initialTimeout);
@ -644,9 +638,9 @@ const TimelinePanel = createReactClass({
// outside of try/catch to not swallow errors // outside of try/catch to not swallow errors
this.updateReadMarker(); this.updateReadMarker();
} }
}, }
updateReadReceiptOnUserActivity: async function() { async updateReadReceiptOnUserActivity() {
this._readReceiptActivityTimer = new Timer(READ_RECEIPT_INTERVAL_MS); this._readReceiptActivityTimer = new Timer(READ_RECEIPT_INTERVAL_MS);
while (this._readReceiptActivityTimer) { //unset on unmount while (this._readReceiptActivityTimer) { //unset on unmount
UserActivity.sharedInstance().timeWhileActiveNow(this._readReceiptActivityTimer); UserActivity.sharedInstance().timeWhileActiveNow(this._readReceiptActivityTimer);
@ -656,9 +650,9 @@ const TimelinePanel = createReactClass({
// outside of try/catch to not swallow errors // outside of try/catch to not swallow errors
this.sendReadReceipt(); this.sendReadReceipt();
} }
}, }
sendReadReceipt: function() { sendReadReceipt = () => {
if (SettingsStore.getValue("lowBandwidth")) return; if (SettingsStore.getValue("lowBandwidth")) return;
if (!this._messagePanel.current) return; if (!this._messagePanel.current) return;
@ -766,11 +760,11 @@ const TimelinePanel = createReactClass({
}); });
} }
} }
}, };
// if the read marker is on the screen, we can now assume we've caught up to the end // if the read marker is on the screen, we can now assume we've caught up to the end
// of the screen, so move the marker down to the bottom of the screen. // of the screen, so move the marker down to the bottom of the screen.
updateReadMarker: function() { updateReadMarker = () => {
if (!this.props.manageReadMarkers) return; if (!this.props.manageReadMarkers) return;
if (this.getReadMarkerPosition() === 1) { if (this.getReadMarkerPosition() === 1) {
// the read marker is at an event below the viewport, // the read marker is at an event below the viewport,
@ -801,11 +795,11 @@ const TimelinePanel = createReactClass({
// Send the updated read marker (along with read receipt) to the server // Send the updated read marker (along with read receipt) to the server
this.sendReadReceipt(); this.sendReadReceipt();
}, };
// advance the read marker past any events we sent ourselves. // advance the read marker past any events we sent ourselves.
_advanceReadMarkerPastMyEvents: function() { _advanceReadMarkerPastMyEvents() {
if (!this.props.manageReadMarkers) return; if (!this.props.manageReadMarkers) return;
// we call `_timelineWindow.getEvents()` rather than using // we call `_timelineWindow.getEvents()` rather than using
@ -837,11 +831,11 @@ const TimelinePanel = createReactClass({
const ev = events[i]; const ev = events[i];
this._setReadMarker(ev.getId(), ev.getTs()); this._setReadMarker(ev.getId(), ev.getTs());
}, }
/* jump down to the bottom of this room, where new events are arriving /* jump down to the bottom of this room, where new events are arriving
*/ */
jumpToLiveTimeline: function() { jumpToLiveTimeline = () => {
// if we can't forward-paginate the existing timeline, then there // if we can't forward-paginate the existing timeline, then there
// is no point reloading it - just jump straight to the bottom. // is no point reloading it - just jump straight to the bottom.
// //
@ -854,12 +848,12 @@ const TimelinePanel = createReactClass({
this._messagePanel.current.scrollToBottom(); this._messagePanel.current.scrollToBottom();
} }
} }
}, };
/* scroll to show the read-up-to marker. We put it 1/3 of the way down /* scroll to show the read-up-to marker. We put it 1/3 of the way down
* the container. * the container.
*/ */
jumpToReadMarker: function() { jumpToReadMarker = () => {
if (!this.props.manageReadMarkers) return; if (!this.props.manageReadMarkers) return;
if (!this._messagePanel.current) return; if (!this._messagePanel.current) return;
if (!this.state.readMarkerEventId) return; if (!this.state.readMarkerEventId) return;
@ -883,11 +877,11 @@ const TimelinePanel = createReactClass({
// As with jumpToLiveTimeline, we want to reload the timeline around the // As with jumpToLiveTimeline, we want to reload the timeline around the
// read-marker. // read-marker.
this._loadTimeline(this.state.readMarkerEventId, 0, 1/3); this._loadTimeline(this.state.readMarkerEventId, 0, 1/3);
}, };
/* update the read-up-to marker to match the read receipt /* update the read-up-to marker to match the read receipt
*/ */
forgetReadMarker: function() { forgetReadMarker = () => {
if (!this.props.manageReadMarkers) return; if (!this.props.manageReadMarkers) return;
const rmId = this._getCurrentReadReceipt(); const rmId = this._getCurrentReadReceipt();
@ -903,17 +897,17 @@ const TimelinePanel = createReactClass({
} }
this._setReadMarker(rmId, rmTs); this._setReadMarker(rmId, rmTs);
}, };
/* return true if the content is fully scrolled down and we are /* return true if the content is fully scrolled down and we are
* at the end of the live timeline. * at the end of the live timeline.
*/ */
isAtEndOfLiveTimeline: function() { isAtEndOfLiveTimeline = () => {
return this._messagePanel.current return this._messagePanel.current
&& this._messagePanel.current.isAtBottom() && this._messagePanel.current.isAtBottom()
&& this._timelineWindow && this._timelineWindow
&& !this._timelineWindow.canPaginate(EventTimeline.FORWARDS); && !this._timelineWindow.canPaginate(EventTimeline.FORWARDS);
}, }
/* get the current scroll state. See ScrollPanel.getScrollState for /* get the current scroll state. See ScrollPanel.getScrollState for
@ -921,10 +915,10 @@ const TimelinePanel = createReactClass({
* *
* returns null if we are not mounted. * returns null if we are not mounted.
*/ */
getScrollState: function() { getScrollState = () => {
if (!this._messagePanel.current) { return null; } if (!this._messagePanel.current) { return null; }
return this._messagePanel.current.getScrollState(); return this._messagePanel.current.getScrollState();
}, };
// returns one of: // returns one of:
// //
@ -932,7 +926,7 @@ const TimelinePanel = createReactClass({
// -1: read marker is above the window // -1: read marker is above the window
// 0: read marker is visible // 0: read marker is visible
// +1: read marker is below the window // +1: read marker is below the window
getReadMarkerPosition: function() { getReadMarkerPosition = () => {
if (!this.props.manageReadMarkers) return null; if (!this.props.manageReadMarkers) return null;
if (!this._messagePanel.current) return null; if (!this._messagePanel.current) return null;
@ -953,9 +947,9 @@ const TimelinePanel = createReactClass({
} }
return null; return null;
}, };
canJumpToReadMarker: function() { canJumpToReadMarker = () => {
// 1. Do not show jump bar if neither the RM nor the RR are set. // 1. Do not show jump bar if neither the RM nor the RR are set.
// 3. We want to show the bar if the read-marker is off the top of the screen. // 3. We want to show the bar if the read-marker is off the top of the screen.
// 4. Also, if pos === null, the event might not be paginated - show the unread bar // 4. Also, if pos === null, the event might not be paginated - show the unread bar
@ -963,14 +957,14 @@ const TimelinePanel = createReactClass({
const ret = this.state.readMarkerEventId !== null && // 1. const ret = this.state.readMarkerEventId !== null && // 1.
(pos < 0 || pos === null); // 3., 4. (pos < 0 || pos === null); // 3., 4.
return ret; return ret;
}, };
/* /*
* called by the parent component when PageUp/Down/etc is pressed. * called by the parent component when PageUp/Down/etc is pressed.
* *
* We pass it down to the scroll panel. * We pass it down to the scroll panel.
*/ */
handleScrollKey: function(ev) { handleScrollKey = ev => {
if (!this._messagePanel.current) { return; } if (!this._messagePanel.current) { return; }
// jump to the live timeline on ctrl-end, rather than the end of the // jump to the live timeline on ctrl-end, rather than the end of the
@ -980,9 +974,9 @@ const TimelinePanel = createReactClass({
} else { } else {
this._messagePanel.current.handleScrollKey(ev); this._messagePanel.current.handleScrollKey(ev);
} }
}, };
_initTimeline: function(props) { _initTimeline(props) {
const initialEvent = props.eventId; const initialEvent = props.eventId;
const pixelOffset = props.eventPixelOffset; const pixelOffset = props.eventPixelOffset;
@ -994,7 +988,7 @@ const TimelinePanel = createReactClass({
} }
return this._loadTimeline(initialEvent, pixelOffset, offsetBase); return this._loadTimeline(initialEvent, pixelOffset, offsetBase);
}, }
/** /**
* (re)-load the event timeline, and initialise the scroll state, centered * (re)-load the event timeline, and initialise the scroll state, centered
@ -1012,7 +1006,7 @@ const TimelinePanel = createReactClass({
* *
* returns a promise which will resolve when the load completes. * returns a promise which will resolve when the load completes.
*/ */
_loadTimeline: function(eventId, pixelOffset, offsetBase) { _loadTimeline(eventId, pixelOffset, offsetBase) {
this._timelineWindow = new Matrix.TimelineWindow( this._timelineWindow = new Matrix.TimelineWindow(
MatrixClientPeg.get(), this.props.timelineSet, MatrixClientPeg.get(), this.props.timelineSet,
{windowLimit: this.props.timelineCap}); {windowLimit: this.props.timelineCap});
@ -1122,21 +1116,21 @@ const TimelinePanel = createReactClass({
}); });
prom.then(onLoaded, onError); prom.then(onLoaded, onError);
} }
}, }
// handle the completion of a timeline load or localEchoUpdate, by // handle the completion of a timeline load or localEchoUpdate, by
// reloading the events from the timelinewindow and pending event list into // reloading the events from the timelinewindow and pending event list into
// the state. // the state.
_reloadEvents: function() { _reloadEvents() {
// we might have switched rooms since the load started - just bin // we might have switched rooms since the load started - just bin
// the results if so. // the results if so.
if (this.unmounted) return; if (this.unmounted) return;
this.setState(this._getEvents()); this.setState(this._getEvents());
}, }
// get the list of events from the timeline window and the pending event list // get the list of events from the timeline window and the pending event list
_getEvents: function() { _getEvents() {
const events = this._timelineWindow.getEvents(); const events = this._timelineWindow.getEvents();
const firstVisibleEventIndex = this._checkForPreJoinUISI(events); const firstVisibleEventIndex = this._checkForPreJoinUISI(events);
@ -1154,7 +1148,7 @@ const TimelinePanel = createReactClass({
liveEvents, liveEvents,
firstVisibleEventIndex, firstVisibleEventIndex,
}; };
}, }
/** /**
* Check for undecryptable messages that were sent while the user was not in * Check for undecryptable messages that were sent while the user was not in
@ -1166,7 +1160,7 @@ const TimelinePanel = createReactClass({
* undecryptable event that was sent while the user was not in the room. If no * undecryptable event that was sent while the user was not in the room. If no
* such events were found, then it returns 0. * such events were found, then it returns 0.
*/ */
_checkForPreJoinUISI: function(events) { _checkForPreJoinUISI(events) {
const room = this.props.timelineSet.room; const room = this.props.timelineSet.room;
if (events.length === 0 || !room || if (events.length === 0 || !room ||
@ -1228,18 +1222,18 @@ const TimelinePanel = createReactClass({
} }
} }
return 0; return 0;
}, }
_indexForEventId: function(evId) { _indexForEventId(evId) {
for (let i = 0; i < this.state.events.length; ++i) { for (let i = 0; i < this.state.events.length; ++i) {
if (evId == this.state.events[i].getId()) { if (evId == this.state.events[i].getId()) {
return i; return i;
} }
} }
return null; return null;
}, }
_getLastDisplayedEventIndex: function(opts) { _getLastDisplayedEventIndex(opts) {
opts = opts || {}; opts = opts || {};
const ignoreOwn = opts.ignoreOwn || false; const ignoreOwn = opts.ignoreOwn || false;
const allowPartial = opts.allowPartial || false; const allowPartial = opts.allowPartial || false;
@ -1313,7 +1307,7 @@ const TimelinePanel = createReactClass({
} }
return null; return null;
}, }
/** /**
* Get the id of the event corresponding to our user's latest read-receipt. * Get the id of the event corresponding to our user's latest read-receipt.
@ -1324,7 +1318,7 @@ const TimelinePanel = createReactClass({
* SDK. * SDK.
* @return {String} the event ID * @return {String} the event ID
*/ */
_getCurrentReadReceipt: function(ignoreSynthesized) { _getCurrentReadReceipt(ignoreSynthesized) {
const client = MatrixClientPeg.get(); const client = MatrixClientPeg.get();
// the client can be null on logout // the client can be null on logout
if (client == null) { if (client == null) {
@ -1333,9 +1327,9 @@ const TimelinePanel = createReactClass({
const myUserId = client.credentials.userId; const myUserId = client.credentials.userId;
return this.props.timelineSet.room.getEventReadUpTo(myUserId, ignoreSynthesized); return this.props.timelineSet.room.getEventReadUpTo(myUserId, ignoreSynthesized);
}, }
_setReadMarker: function(eventId, eventTs, inhibitSetState) { _setReadMarker(eventId, eventTs, inhibitSetState) {
const roomId = this.props.timelineSet.room.roomId; const roomId = this.props.timelineSet.room.roomId;
// don't update the state (and cause a re-render) if there is // don't update the state (and cause a re-render) if there is
@ -1358,9 +1352,9 @@ const TimelinePanel = createReactClass({
this.setState({ this.setState({
readMarkerEventId: eventId, readMarkerEventId: eventId,
}, this.props.onReadMarkerUpdated); }, this.props.onReadMarkerUpdated);
}, }
_shouldPaginate: function() { _shouldPaginate() {
// don't try to paginate while events in the timeline are // don't try to paginate while events in the timeline are
// still being decrypted. We don't render events while they're // still being decrypted. We don't render events while they're
// being decrypted, so they don't take up space in the timeline. // being decrypted, so they don't take up space in the timeline.
@ -1369,13 +1363,11 @@ const TimelinePanel = createReactClass({
return !this.state.events.some((e) => { return !this.state.events.some((e) => {
return e.isBeingDecrypted(); return e.isBeingDecrypted();
}); });
}, }
getRelationsForEvent(...args) { getRelationsForEvent = (...args) => this.props.timelineSet.getRelationsForEvent(...args);
return this.props.timelineSet.getRelationsForEvent(...args);
},
render: function() { render() {
const MessagePanel = sdk.getComponent("structures.MessagePanel"); const MessagePanel = sdk.getComponent("structures.MessagePanel");
const Loader = sdk.getComponent("elements.Spinner"); const Loader = sdk.getComponent("elements.Spinner");
@ -1456,7 +1448,7 @@ const TimelinePanel = createReactClass({
useIRCLayout={this.props.useIRCLayout} useIRCLayout={this.props.useIRCLayout}
/> />
); );
}, }
}); }
export default TimelinePanel; export default TimelinePanel;

View file

@ -16,30 +16,28 @@ limitations under the License.
*/ */
import React from 'react'; import React from 'react';
import createReactClass from 'create-react-class';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import ContentMessages from '../../ContentMessages'; import ContentMessages from '../../ContentMessages';
import dis from "../../dispatcher/dispatcher"; import dis from "../../dispatcher/dispatcher";
import filesize from "filesize"; import filesize from "filesize";
import { _t } from '../../languageHandler'; import { _t } from '../../languageHandler';
export default createReactClass({ export default class UploadBar extends React.Component {
displayName: 'UploadBar', static propTypes = {
propTypes: {
room: PropTypes.object, room: PropTypes.object,
}, };
componentDidMount: function() { componentDidMount() {
this.dispatcherRef = dis.register(this.onAction); this.dispatcherRef = dis.register(this.onAction);
this.mounted = true; this.mounted = true;
}, }
componentWillUnmount: function() { componentWillUnmount() {
this.mounted = false; this.mounted = false;
dis.unregister(this.dispatcherRef); dis.unregister(this.dispatcherRef);
}, }
onAction: function(payload) { onAction = payload => {
switch (payload.action) { switch (payload.action) {
case 'upload_progress': case 'upload_progress':
case 'upload_finished': case 'upload_finished':
@ -48,9 +46,9 @@ export default createReactClass({
if (this.mounted) this.forceUpdate(); if (this.mounted) this.forceUpdate();
break; break;
} }
}, };
render: function() { render() {
const uploads = ContentMessages.sharedInstance().getCurrentUploads(); const uploads = ContentMessages.sharedInstance().getCurrentUploads();
// for testing UI... - also fix up the ContentMessages.getCurrentUploads().length // for testing UI... - also fix up the ContentMessages.getCurrentUploads().length
@ -105,5 +103,5 @@ export default createReactClass({
<div className="mx_UploadBar_uploadFilename">{ uploadText }</div> <div className="mx_UploadBar_uploadFilename">{ uploadText }</div>
</div> </div>
); );
}, }
}); }

View file

@ -40,8 +40,16 @@ import AccessibleTooltipButton from "../views/elements/AccessibleTooltipButton";
import { SettingLevel } from "../../settings/SettingLevel"; import { SettingLevel } from "../../settings/SettingLevel";
import IconizedContextMenu, { import IconizedContextMenu, {
IconizedContextMenuOption, IconizedContextMenuOption,
IconizedContextMenuOptionList IconizedContextMenuOptionList,
} from "../views/context_menus/IconizedContextMenu"; } from "../views/context_menus/IconizedContextMenu";
import { CommunityPrototypeStore } from "../../stores/CommunityPrototypeStore";
import * as fbEmitter from "fbemitter";
import TagOrderStore from "../../stores/TagOrderStore";
import { showCommunityInviteDialog } from "../../RoomInvite";
import dis from "../../dispatcher/dispatcher";
import { RightPanelPhases } from "../../stores/RightPanelStorePhases";
import ErrorDialog from "../views/dialogs/ErrorDialog";
import EditCommunityPrototypeDialog from "../views/dialogs/EditCommunityPrototypeDialog";
interface IProps { interface IProps {
isMinimized: boolean; isMinimized: boolean;
@ -58,6 +66,7 @@ export default class UserMenu extends React.Component<IProps, IState> {
private dispatcherRef: string; private dispatcherRef: string;
private themeWatcherRef: string; private themeWatcherRef: string;
private buttonRef: React.RefObject<HTMLButtonElement> = createRef(); private buttonRef: React.RefObject<HTMLButtonElement> = createRef();
private tagStoreRef: fbEmitter.EventSubscription;
constructor(props: IProps) { constructor(props: IProps) {
super(props); super(props);
@ -77,14 +86,20 @@ export default class UserMenu extends React.Component<IProps, IState> {
public componentDidMount() { public componentDidMount() {
this.dispatcherRef = defaultDispatcher.register(this.onAction); this.dispatcherRef = defaultDispatcher.register(this.onAction);
this.themeWatcherRef = SettingsStore.watchSetting("theme", null, this.onThemeChanged); this.themeWatcherRef = SettingsStore.watchSetting("theme", null, this.onThemeChanged);
this.tagStoreRef = TagOrderStore.addListener(this.onTagStoreUpdate);
} }
public componentWillUnmount() { public componentWillUnmount() {
if (this.themeWatcherRef) SettingsStore.unwatchSetting(this.themeWatcherRef); if (this.themeWatcherRef) SettingsStore.unwatchSetting(this.themeWatcherRef);
if (this.dispatcherRef) defaultDispatcher.unregister(this.dispatcherRef); if (this.dispatcherRef) defaultDispatcher.unregister(this.dispatcherRef);
OwnProfileStore.instance.off(UPDATE_EVENT, this.onProfileUpdate); OwnProfileStore.instance.off(UPDATE_EVENT, this.onProfileUpdate);
this.tagStoreRef.remove();
} }
private onTagStoreUpdate = () => {
this.forceUpdate(); // we don't have anything useful in state to update
};
private isUserOnDarkTheme(): boolean { private isUserOnDarkTheme(): boolean {
const theme = SettingsStore.getValue("theme"); const theme = SettingsStore.getValue("theme");
if (theme.startsWith("custom-")) { if (theme.startsWith("custom-")) {
@ -189,9 +204,54 @@ export default class UserMenu extends React.Component<IProps, IState> {
defaultDispatcher.dispatch({action: 'view_home_page'}); defaultDispatcher.dispatch({action: 'view_home_page'});
}; };
private onCommunitySettingsClick = (ev: ButtonEvent) => {
ev.preventDefault();
ev.stopPropagation();
Modal.createTrackedDialog('Edit Community', '', EditCommunityPrototypeDialog, {
communityId: CommunityPrototypeStore.instance.getSelectedCommunityId(),
});
this.setState({contextMenuPosition: null}); // also close the menu
};
private onCommunityMembersClick = (ev: ButtonEvent) => {
ev.preventDefault();
ev.stopPropagation();
// We'd ideally just pop open a right panel with the member list, but the current
// way the right panel is structured makes this exceedingly difficult. Instead, we'll
// switch to the general room and open the member list there as it should be in sync
// anyways.
const chat = CommunityPrototypeStore.instance.getSelectedCommunityGeneralChat();
if (chat) {
dis.dispatch({
action: 'view_room',
room_id: chat.roomId,
}, true);
dis.dispatch({action: Action.SetRightPanelPhase, phase: RightPanelPhases.RoomMemberList});
} else {
// "This should never happen" clauses go here for the prototype.
Modal.createTrackedDialog('Failed to find general chat', '', ErrorDialog, {
title: _t('Failed to find the general chat for this community'),
description: _t("Failed to find the general chat for this community"),
});
}
this.setState({contextMenuPosition: null}); // also close the menu
};
private onCommunityInviteClick = (ev: ButtonEvent) => {
ev.preventDefault();
ev.stopPropagation();
showCommunityInviteDialog(CommunityPrototypeStore.instance.getSelectedCommunityId());
this.setState({contextMenuPosition: null}); // also close the menu
};
private renderContextMenu = (): React.ReactNode => { private renderContextMenu = (): React.ReactNode => {
if (!this.state.contextMenuPosition) return null; if (!this.state.contextMenuPosition) return null;
const prototypeCommunityName = CommunityPrototypeStore.instance.getSelectedCommunityName();
let hostingLink; let hostingLink;
const signupLink = getHostingLink("user-context-menu"); const signupLink = getHostingLink("user-context-menu");
if (signupLink) { if (signupLink) {
@ -225,14 +285,7 @@ export default class UserMenu extends React.Component<IProps, IState> {
); );
} }
return <IconizedContextMenu let primaryHeader = (
// -20 to overlap the context menu by just over the width of the `...` icon and make it look connected
left={this.state.contextMenuPosition.width + this.state.contextMenuPosition.left - 20}
top={this.state.contextMenuPosition.top + this.state.contextMenuPosition.height}
onFinished={this.onCloseMenu}
className="mx_UserMenu_contextMenu"
>
<div className="mx_UserMenu_contextMenu_header">
<div className="mx_UserMenu_contextMenu_name"> <div className="mx_UserMenu_contextMenu_name">
<span className="mx_UserMenu_contextMenu_displayName"> <span className="mx_UserMenu_contextMenu_displayName">
{OwnProfileStore.instance.displayName} {OwnProfileStore.instance.displayName}
@ -241,19 +294,9 @@ export default class UserMenu extends React.Component<IProps, IState> {
{MatrixClientPeg.get().getUserId()} {MatrixClientPeg.get().getUserId()}
</span> </span>
</div> </div>
<AccessibleTooltipButton );
className="mx_UserMenu_contextMenu_themeButton" let primaryOptionList = (
onClick={this.onSwitchThemeClick} <React.Fragment>
title={this.state.isDarkTheme ? _t("Switch to light mode") : _t("Switch to dark mode")}
>
<img
src={require("../../../res/img/element-icons/roomlist/dark-light-mode.svg")}
alt={_t("Switch theme")}
width={16}
/>
</AccessibleTooltipButton>
</div>
{hostingLink}
<IconizedContextMenuOptionList> <IconizedContextMenuOptionList>
{homeButton} {homeButton}
<IconizedContextMenuOption <IconizedContextMenuOption
@ -289,6 +332,105 @@ export default class UserMenu extends React.Component<IProps, IState> {
onClick={this.onSignOutClick} onClick={this.onSignOutClick}
/> />
</IconizedContextMenuOptionList> </IconizedContextMenuOptionList>
</React.Fragment>
);
let secondarySection = null;
if (prototypeCommunityName) {
primaryHeader = (
<div className="mx_UserMenu_contextMenu_name">
<span className="mx_UserMenu_contextMenu_displayName">
{prototypeCommunityName}
</span>
</div>
);
primaryOptionList = (
<IconizedContextMenuOptionList>
<IconizedContextMenuOption
iconClassName="mx_UserMenu_iconSettings"
label={_t("Settings")}
aria-label={_t("Community settings")}
onClick={this.onCommunitySettingsClick}
/>
<IconizedContextMenuOption
iconClassName="mx_UserMenu_iconMembers"
label={_t("Members")}
onClick={this.onCommunityMembersClick}
/>
<IconizedContextMenuOption
iconClassName="mx_UserMenu_iconInvite"
label={_t("Invite")}
onClick={this.onCommunityInviteClick}
/>
</IconizedContextMenuOptionList>
);
secondarySection = (
<React.Fragment>
<hr />
<div className="mx_UserMenu_contextMenu_header">
<div className="mx_UserMenu_contextMenu_name">
<span className="mx_UserMenu_contextMenu_displayName">
{OwnProfileStore.instance.displayName}
</span>
<span className="mx_UserMenu_contextMenu_userId">
{MatrixClientPeg.get().getUserId()}
</span>
</div>
</div>
<IconizedContextMenuOptionList>
<IconizedContextMenuOption
iconClassName="mx_UserMenu_iconSettings"
label={_t("Settings")}
aria-label={_t("User settings")}
onClick={(e) => this.onSettingsOpen(e, null)}
/>
<IconizedContextMenuOption
iconClassName="mx_UserMenu_iconMessage"
label={_t("Feedback")}
onClick={this.onProvideFeedback}
/>
</IconizedContextMenuOptionList>
<IconizedContextMenuOptionList red>
<IconizedContextMenuOption
iconClassName="mx_UserMenu_iconSignOut"
label={_t("Sign out")}
onClick={this.onSignOutClick}
/>
</IconizedContextMenuOptionList>
</React.Fragment>
)
}
const classes = classNames({
"mx_UserMenu_contextMenu": true,
"mx_UserMenu_contextMenu_prototype": !!prototypeCommunityName,
});
return <IconizedContextMenu
// numerical adjustments to overlap the context menu by just over the width of the
// menu icon and make it look connected
left={this.state.contextMenuPosition.width + this.state.contextMenuPosition.left - 10}
top={this.state.contextMenuPosition.top + this.state.contextMenuPosition.height + 8}
onFinished={this.onCloseMenu}
className={classes}
>
<div className="mx_UserMenu_contextMenu_header">
{primaryHeader}
<AccessibleTooltipButton
className="mx_UserMenu_contextMenu_themeButton"
onClick={this.onSwitchThemeClick}
title={this.state.isDarkTheme ? _t("Switch to light mode") : _t("Switch to dark mode")}
>
<img
src={require("../../../res/img/element-icons/roomlist/dark-light-mode.svg")}
alt={_t("Switch theme")}
width={16}
/>
</AccessibleTooltipButton>
</div>
{hostingLink}
{primaryOptionList}
{secondarySection}
</IconizedContextMenu>; </IconizedContextMenu>;
}; };
@ -298,12 +440,34 @@ export default class UserMenu extends React.Component<IProps, IState> {
const displayName = OwnProfileStore.instance.displayName || MatrixClientPeg.get().getUserId(); const displayName = OwnProfileStore.instance.displayName || MatrixClientPeg.get().getUserId();
const avatarUrl = OwnProfileStore.instance.getHttpAvatarUrl(avatarSize); const avatarUrl = OwnProfileStore.instance.getHttpAvatarUrl(avatarSize);
const prototypeCommunityName = CommunityPrototypeStore.instance.getSelectedCommunityName();
let isPrototype = false;
let menuName = _t("User menu");
let name = <span className="mx_UserMenu_userName">{displayName}</span>; let name = <span className="mx_UserMenu_userName">{displayName}</span>;
let buttons = ( let buttons = (
<span className="mx_UserMenu_headerButtons"> <span className="mx_UserMenu_headerButtons">
{/* masked image in CSS */} {/* masked image in CSS */}
</span> </span>
); );
if (prototypeCommunityName) {
name = (
<div className="mx_UserMenu_doubleName">
<span className="mx_UserMenu_userName">{prototypeCommunityName}</span>
<span className="mx_UserMenu_subUserName">{displayName}</span>
</div>
);
menuName = _t("Community and user menu");
isPrototype = true;
} else if (SettingsStore.getValue("feature_communities_v2_prototypes")) {
name = (
<div className="mx_UserMenu_doubleName">
<span className="mx_UserMenu_userName">{_t("Home")}</span>
<span className="mx_UserMenu_subUserName">{displayName}</span>
</div>
);
isPrototype = true;
}
if (this.props.isMinimized) { if (this.props.isMinimized) {
name = null; name = null;
buttons = null; buttons = null;
@ -312,6 +476,7 @@ export default class UserMenu extends React.Component<IProps, IState> {
const classes = classNames({ const classes = classNames({
'mx_UserMenu': true, 'mx_UserMenu': true,
'mx_UserMenu_minimized': this.props.isMinimized, 'mx_UserMenu_minimized': this.props.isMinimized,
'mx_UserMenu_prototype': isPrototype,
}); });
return ( return (
@ -320,7 +485,7 @@ export default class UserMenu extends React.Component<IProps, IState> {
className={classes} className={classes}
onClick={this.onOpenMenuClick} onClick={this.onOpenMenuClick}
inputRef={this.buttonRef} inputRef={this.buttonRef}
label={_t("User menu")} label={menuName}
isExpanded={!!this.state.contextMenuPosition} isExpanded={!!this.state.contextMenuPosition}
onContextMenu={this.onContextMenu} onContextMenu={this.onContextMenu}
> >

View file

@ -17,24 +17,21 @@ limitations under the License.
*/ */
import React from 'react'; import React from 'react';
import createReactClass from 'create-react-class';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import SyntaxHighlight from '../views/elements/SyntaxHighlight'; import SyntaxHighlight from '../views/elements/SyntaxHighlight';
import {_t} from "../../languageHandler"; import {_t} from "../../languageHandler";
import * as sdk from "../../index"; import * as sdk from "../../index";
export default createReactClass({ export default class ViewSource extends React.Component {
displayName: 'ViewSource', static propTypes = {
propTypes: {
content: PropTypes.object.isRequired, content: PropTypes.object.isRequired,
onFinished: PropTypes.func.isRequired, onFinished: PropTypes.func.isRequired,
roomId: PropTypes.string.isRequired, roomId: PropTypes.string.isRequired,
eventId: PropTypes.string.isRequired, eventId: PropTypes.string.isRequired,
}, };
render: function() { render() {
const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog'); const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog');
return ( return (
<BaseDialog className="mx_ViewSource" onFinished={this.props.onFinished} title={_t('View Source')}> <BaseDialog className="mx_ViewSource" onFinished={this.props.onFinished} title={_t('View Source')}>
@ -49,5 +46,5 @@ export default createReactClass({
</div> </div>
</BaseDialog> </BaseDialog>
); );
}, }
}); }

View file

@ -17,7 +17,6 @@ limitations under the License.
*/ */
import React from 'react'; import React from 'react';
import createReactClass from 'create-react-class';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import { _t } from '../../../languageHandler'; import { _t } from '../../../languageHandler';
import * as sdk from '../../../index'; import * as sdk from '../../../index';
@ -40,18 +39,15 @@ const PHASE_EMAIL_SENT = 3;
// User has clicked the link in email and completed reset // User has clicked the link in email and completed reset
const PHASE_DONE = 4; const PHASE_DONE = 4;
export default createReactClass({ export default class ForgotPassword extends React.Component {
displayName: 'ForgotPassword', static propTypes = {
propTypes: {
serverConfig: PropTypes.instanceOf(ValidatedServerConfig).isRequired, serverConfig: PropTypes.instanceOf(ValidatedServerConfig).isRequired,
onServerConfigChange: PropTypes.func.isRequired, onServerConfigChange: PropTypes.func.isRequired,
onLoginClick: PropTypes.func, onLoginClick: PropTypes.func,
onComplete: PropTypes.func.isRequired, onComplete: PropTypes.func.isRequired,
}, };
getInitialState: function() { state = {
return {
phase: PHASE_FORGOT, phase: PHASE_FORGOT,
email: "", email: "",
password: "", password: "",
@ -67,23 +63,23 @@ export default createReactClass({
serverDeadError: "", serverDeadError: "",
serverRequiresIdServer: null, serverRequiresIdServer: null,
}; };
},
componentDidMount: function() { componentDidMount() {
this.reset = null; this.reset = null;
this._checkServerLiveliness(this.props.serverConfig); this._checkServerLiveliness(this.props.serverConfig);
}, }
// TODO: [REACT-WARNING] Replace with appropriate lifecycle event // TODO: [REACT-WARNING] Replace with appropriate lifecycle event
UNSAFE_componentWillReceiveProps: function(newProps) { // eslint-disable-next-line camelcase
UNSAFE_componentWillReceiveProps(newProps) {
if (newProps.serverConfig.hsUrl === this.props.serverConfig.hsUrl && if (newProps.serverConfig.hsUrl === this.props.serverConfig.hsUrl &&
newProps.serverConfig.isUrl === this.props.serverConfig.isUrl) return; newProps.serverConfig.isUrl === this.props.serverConfig.isUrl) return;
// Do a liveliness check on the new URLs // Do a liveliness check on the new URLs
this._checkServerLiveliness(newProps.serverConfig); this._checkServerLiveliness(newProps.serverConfig);
}, }
_checkServerLiveliness: async function(serverConfig) { async _checkServerLiveliness(serverConfig) {
try { try {
await AutoDiscoveryUtils.validateServerConfigWithStaticUrls( await AutoDiscoveryUtils.validateServerConfigWithStaticUrls(
serverConfig.hsUrl, serverConfig.hsUrl,
@ -100,9 +96,9 @@ export default createReactClass({
} catch (e) { } catch (e) {
this.setState(AutoDiscoveryUtils.authComponentStateForError(e, "forgot_password")); this.setState(AutoDiscoveryUtils.authComponentStateForError(e, "forgot_password"));
} }
}, }
submitPasswordReset: function(email, password) { submitPasswordReset(email, password) {
this.setState({ this.setState({
phase: PHASE_SENDING_EMAIL, phase: PHASE_SENDING_EMAIL,
}); });
@ -117,9 +113,9 @@ export default createReactClass({
phase: PHASE_FORGOT, phase: PHASE_FORGOT,
}); });
}); });
}, }
onVerify: async function(ev) { onVerify = async ev => {
ev.preventDefault(); ev.preventDefault();
if (!this.reset) { if (!this.reset) {
console.error("onVerify called before submitPasswordReset!"); console.error("onVerify called before submitPasswordReset!");
@ -131,9 +127,9 @@ export default createReactClass({
} catch (err) { } catch (err) {
this.showErrorDialog(err.message); this.showErrorDialog(err.message);
} }
}, };
onSubmitForm: async function(ev) { onSubmitForm = async ev => {
ev.preventDefault(); ev.preventDefault();
// refresh the server errors, just in case the server came back online // refresh the server errors, just in case the server came back online
@ -166,41 +162,41 @@ export default createReactClass({
}, },
}); });
} }
}, };
onInputChanged: function(stateKey, ev) { onInputChanged = (stateKey, ev) => {
this.setState({ this.setState({
[stateKey]: ev.target.value, [stateKey]: ev.target.value,
}); });
}, };
async onServerDetailsNextPhaseClick() { onServerDetailsNextPhaseClick = async () => {
this.setState({ this.setState({
phase: PHASE_FORGOT, phase: PHASE_FORGOT,
}); });
}, };
onEditServerDetailsClick(ev) { onEditServerDetailsClick = ev => {
ev.preventDefault(); ev.preventDefault();
ev.stopPropagation(); ev.stopPropagation();
this.setState({ this.setState({
phase: PHASE_SERVER_DETAILS, phase: PHASE_SERVER_DETAILS,
}); });
}, };
onLoginClick: function(ev) { onLoginClick = ev => {
ev.preventDefault(); ev.preventDefault();
ev.stopPropagation(); ev.stopPropagation();
this.props.onLoginClick(); this.props.onLoginClick();
}, };
showErrorDialog: function(body, title) { showErrorDialog(body, title) {
const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog"); const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
Modal.createTrackedDialog('Forgot Password Error', '', ErrorDialog, { Modal.createTrackedDialog('Forgot Password Error', '', ErrorDialog, {
title: title, title: title,
description: body, description: body,
}); });
}, }
renderServerDetails() { renderServerDetails() {
const ServerConfig = sdk.getComponent("auth.ServerConfig"); const ServerConfig = sdk.getComponent("auth.ServerConfig");
@ -218,7 +214,7 @@ export default createReactClass({
submitText={_t("Next")} submitText={_t("Next")}
submitClass="mx_Login_submit" submitClass="mx_Login_submit"
/>; />;
}, }
renderForgot() { renderForgot() {
const Field = sdk.getComponent('elements.Field'); const Field = sdk.getComponent('elements.Field');
@ -335,12 +331,12 @@ export default createReactClass({
{_t('Sign in instead')} {_t('Sign in instead')}
</a> </a>
</div>; </div>;
}, }
renderSendingEmail() { renderSendingEmail() {
const Spinner = sdk.getComponent("elements.Spinner"); const Spinner = sdk.getComponent("elements.Spinner");
return <Spinner />; return <Spinner />;
}, }
renderEmailSent() { renderEmailSent() {
return <div> return <div>
@ -350,7 +346,7 @@ export default createReactClass({
<input className="mx_Login_submit" type="button" onClick={this.onVerify} <input className="mx_Login_submit" type="button" onClick={this.onVerify}
value={_t('I have verified my email address')} /> value={_t('I have verified my email address')} />
</div>; </div>;
}, }
renderDone() { renderDone() {
return <div> return <div>
@ -363,9 +359,9 @@ export default createReactClass({
<input className="mx_Login_submit" type="button" onClick={this.props.onComplete} <input className="mx_Login_submit" type="button" onClick={this.props.onComplete}
value={_t('Return to login screen')} /> value={_t('Return to login screen')} />
</div>; </div>;
}, }
render: function() { render() {
const AuthHeader = sdk.getComponent("auth.AuthHeader"); const AuthHeader = sdk.getComponent("auth.AuthHeader");
const AuthBody = sdk.getComponent("auth.AuthBody"); const AuthBody = sdk.getComponent("auth.AuthBody");
@ -397,5 +393,5 @@ export default createReactClass({
</AuthBody> </AuthBody>
</AuthPage> </AuthPage>
); );
}, }
}); }

View file

@ -17,7 +17,6 @@ limitations under the License.
*/ */
import React from 'react'; import React from 'react';
import createReactClass from 'create-react-class';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import {_t, _td} from '../../../languageHandler'; import {_t, _td} from '../../../languageHandler';
import * as sdk from '../../../index'; import * as sdk from '../../../index';
@ -53,13 +52,11 @@ _td("Invalid base_url for m.identity_server");
_td("Identity server URL does not appear to be a valid identity server"); _td("Identity server URL does not appear to be a valid identity server");
_td("General failure"); _td("General failure");
/** /*
* A wire component which glues together login UI components and Login logic * A wire component which glues together login UI components and Login logic
*/ */
export default createReactClass({ export default class LoginComponent extends React.Component {
displayName: 'Login', static propTypes = {
propTypes: {
// Called when the user has logged in. Params: // Called when the user has logged in. Params:
// - The object returned by the login API // - The object returned by the login API
// - The user's password, if applicable, (may be cached in memory for a // - The user's password, if applicable, (may be cached in memory for a
@ -85,10 +82,14 @@ export default createReactClass({
serverConfig: PropTypes.instanceOf(ValidatedServerConfig).isRequired, serverConfig: PropTypes.instanceOf(ValidatedServerConfig).isRequired,
isSyncing: PropTypes.bool, isSyncing: PropTypes.bool,
}, };
getInitialState: function() { constructor(props) {
return { super(props);
this._unmounted = false;
this.state = {
busy: false, busy: false,
busyLoggingIn: null, busyLoggingIn: null,
errorText: null, errorText: null,
@ -113,11 +114,6 @@ export default createReactClass({
serverErrorIsFatal: false, serverErrorIsFatal: false,
serverDeadError: "", serverDeadError: "",
}; };
},
// TODO: [REACT-WARNING] Move this to constructor
UNSAFE_componentWillMount: function() {
this._unmounted = false;
// map from login step type to a function which will render a control // map from login step type to a function which will render a control
// letting you do that login type // letting you do that login type
@ -130,33 +126,32 @@ export default createReactClass({
}; };
this._initLoginLogic(); this._initLoginLogic();
}, }
componentWillUnmount: function() { componentWillUnmount() {
this._unmounted = true; this._unmounted = true;
}, }
// TODO: [REACT-WARNING] Replace with appropriate lifecycle event // TODO: [REACT-WARNING] Replace with appropriate lifecycle event
// eslint-disable-next-line camelcase
UNSAFE_componentWillReceiveProps(newProps) { UNSAFE_componentWillReceiveProps(newProps) {
if (newProps.serverConfig.hsUrl === this.props.serverConfig.hsUrl && if (newProps.serverConfig.hsUrl === this.props.serverConfig.hsUrl &&
newProps.serverConfig.isUrl === this.props.serverConfig.isUrl) return; newProps.serverConfig.isUrl === this.props.serverConfig.isUrl) return;
// Ensure that we end up actually logging in to the right place // Ensure that we end up actually logging in to the right place
this._initLoginLogic(newProps.serverConfig.hsUrl, newProps.serverConfig.isUrl); this._initLoginLogic(newProps.serverConfig.hsUrl, newProps.serverConfig.isUrl);
}, }
onPasswordLoginError: function(errorText) { onPasswordLoginError = errorText => {
this.setState({ this.setState({
errorText, errorText,
loginIncorrect: Boolean(errorText), loginIncorrect: Boolean(errorText),
}); });
}, };
isBusy: function() { isBusy = () => this.state.busy || this.props.busy;
return this.state.busy || this.props.busy;
},
onPasswordLogin: async function(username, phoneCountry, phoneNumber, password) { onPasswordLogin = async (username, phoneCountry, phoneNumber, password) => {
if (!this.state.serverIsAlive) { if (!this.state.serverIsAlive) {
this.setState({busy: true}); this.setState({busy: true});
// Do a quick liveliness check on the URLs // Do a quick liveliness check on the URLs
@ -263,13 +258,13 @@ export default createReactClass({
loginIncorrect: error.httpStatus === 401 || error.httpStatus === 403, loginIncorrect: error.httpStatus === 401 || error.httpStatus === 403,
}); });
}); });
}, };
onUsernameChanged: function(username) { onUsernameChanged = username => {
this.setState({ username: username }); this.setState({ username: username });
}, };
onUsernameBlur: async function(username) { onUsernameBlur = async username => {
const doWellknownLookup = username[0] === "@"; const doWellknownLookup = username[0] === "@";
this.setState({ this.setState({
username: username, username: username,
@ -314,19 +309,19 @@ export default createReactClass({
}); });
} }
} }
}, };
onPhoneCountryChanged: function(phoneCountry) { onPhoneCountryChanged = phoneCountry => {
this.setState({ phoneCountry: phoneCountry }); this.setState({ phoneCountry: phoneCountry });
}, };
onPhoneNumberChanged: function(phoneNumber) { onPhoneNumberChanged = phoneNumber => {
this.setState({ this.setState({
phoneNumber: phoneNumber, phoneNumber: phoneNumber,
}); });
}, };
onPhoneNumberBlur: function(phoneNumber) { onPhoneNumberBlur = phoneNumber => {
// Validate the phone number entered // Validate the phone number entered
if (!PHONE_NUMBER_REGEX.test(phoneNumber)) { if (!PHONE_NUMBER_REGEX.test(phoneNumber)) {
this.setState({ this.setState({
@ -339,15 +334,15 @@ export default createReactClass({
canTryLogin: true, canTryLogin: true,
}); });
} }
}, };
onRegisterClick: function(ev) { onRegisterClick = ev => {
ev.preventDefault(); ev.preventDefault();
ev.stopPropagation(); ev.stopPropagation();
this.props.onRegisterClick(); this.props.onRegisterClick();
}, };
onTryRegisterClick: function(ev) { onTryRegisterClick = ev => {
const step = this._getCurrentFlowStep(); const step = this._getCurrentFlowStep();
if (step === 'm.login.sso' || step === 'm.login.cas') { if (step === 'm.login.sso' || step === 'm.login.cas') {
// If we're showing SSO it means that registration is also probably disabled, // If we're showing SSO it means that registration is also probably disabled,
@ -361,23 +356,23 @@ export default createReactClass({
// Don't intercept - just go through to the register page // Don't intercept - just go through to the register page
this.onRegisterClick(ev); this.onRegisterClick(ev);
} }
}, };
async onServerDetailsNextPhaseClick() { onServerDetailsNextPhaseClick = () => {
this.setState({ this.setState({
phase: PHASE_LOGIN, phase: PHASE_LOGIN,
}); });
}, };
onEditServerDetailsClick(ev) { onEditServerDetailsClick = ev => {
ev.preventDefault(); ev.preventDefault();
ev.stopPropagation(); ev.stopPropagation();
this.setState({ this.setState({
phase: PHASE_SERVER_DETAILS, phase: PHASE_SERVER_DETAILS,
}); });
}, };
_initLoginLogic: async function(hsUrl, isUrl) { async _initLoginLogic(hsUrl, isUrl) {
hsUrl = hsUrl || this.props.serverConfig.hsUrl; hsUrl = hsUrl || this.props.serverConfig.hsUrl;
isUrl = isUrl || this.props.serverConfig.isUrl; isUrl = isUrl || this.props.serverConfig.isUrl;
@ -465,9 +460,9 @@ export default createReactClass({
busy: false, busy: false,
}); });
}); });
}, }
_isSupportedFlow: function(flow) { _isSupportedFlow(flow) {
// technically the flow can have multiple steps, but no one does this // technically the flow can have multiple steps, but no one does this
// for login and loginLogic doesn't support it so we can ignore it. // for login and loginLogic doesn't support it so we can ignore it.
if (!this._stepRendererMap[flow.type]) { if (!this._stepRendererMap[flow.type]) {
@ -475,11 +470,11 @@ export default createReactClass({
return false; return false;
} }
return true; return true;
}, }
_getCurrentFlowStep: function() { _getCurrentFlowStep() {
return this._loginLogic ? this._loginLogic.getCurrentFlowStep() : null; return this._loginLogic ? this._loginLogic.getCurrentFlowStep() : null;
}, }
_errorTextFromError(err) { _errorTextFromError(err) {
let errCode = err.errcode; let errCode = err.errcode;
@ -526,7 +521,7 @@ export default createReactClass({
} }
return errorText; return errorText;
}, }
renderServerComponent() { renderServerComponent() {
const ServerConfig = sdk.getComponent("auth.ServerConfig"); const ServerConfig = sdk.getComponent("auth.ServerConfig");
@ -552,7 +547,7 @@ export default createReactClass({
delayTimeMs={250} delayTimeMs={250}
{...serverDetailsProps} {...serverDetailsProps}
/>; />;
}, }
renderLoginComponentForStep() { renderLoginComponentForStep() {
if (PHASES_ENABLED && this.state.phase !== PHASE_LOGIN) { if (PHASES_ENABLED && this.state.phase !== PHASE_LOGIN) {
@ -572,9 +567,9 @@ export default createReactClass({
} }
return null; return null;
}, }
_renderPasswordStep: function() { _renderPasswordStep = () => {
const PasswordLogin = sdk.getComponent('auth.PasswordLogin'); const PasswordLogin = sdk.getComponent('auth.PasswordLogin');
let onEditServerDetailsClick = null; let onEditServerDetailsClick = null;
@ -603,9 +598,9 @@ export default createReactClass({
busy={this.props.isSyncing || this.state.busyLoggingIn} busy={this.props.isSyncing || this.state.busyLoggingIn}
/> />
); );
}, };
_renderSsoStep: function(loginType) { _renderSsoStep = loginType => {
const SignInToText = sdk.getComponent('views.auth.SignInToText'); const SignInToText = sdk.getComponent('views.auth.SignInToText');
let onEditServerDetailsClick = null; let onEditServerDetailsClick = null;
@ -634,9 +629,9 @@ export default createReactClass({
/> />
</div> </div>
); );
}, };
render: function() { render() {
const Loader = sdk.getComponent("elements.Spinner"); const Loader = sdk.getComponent("elements.Spinner");
const InlineSpinner = sdk.getComponent("elements.InlineSpinner"); const InlineSpinner = sdk.getComponent("elements.InlineSpinner");
const AuthHeader = sdk.getComponent("auth.AuthHeader"); const AuthHeader = sdk.getComponent("auth.AuthHeader");
@ -704,5 +699,5 @@ export default createReactClass({
</AuthBody> </AuthBody>
</AuthPage> </AuthPage>
); );
}, }
}); }

View file

@ -15,29 +15,24 @@ limitations under the License.
*/ */
import React from 'react'; import React from 'react';
import createReactClass from 'create-react-class';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import * as sdk from '../../../index'; import * as sdk from '../../../index';
import {MatrixClientPeg} from '../../../MatrixClientPeg'; import {MatrixClientPeg} from '../../../MatrixClientPeg';
import { _t } from '../../../languageHandler'; import { _t } from '../../../languageHandler';
import AuthPage from "../../views/auth/AuthPage"; import AuthPage from "../../views/auth/AuthPage";
export default createReactClass({ export default class PostRegistration extends React.Component {
displayName: 'PostRegistration', static propTypes = {
propTypes: {
onComplete: PropTypes.func.isRequired, onComplete: PropTypes.func.isRequired,
}, };
getInitialState: function() { state = {
return {
avatarUrl: null, avatarUrl: null,
errorString: null, errorString: null,
busy: false, busy: false,
}; };
},
componentDidMount: function() { componentDidMount() {
// There is some assymetry between ChangeDisplayName and ChangeAvatar, // There is some assymetry between ChangeDisplayName and ChangeAvatar,
// as ChangeDisplayName will auto-get the name but ChangeAvatar expects // as ChangeDisplayName will auto-get the name but ChangeAvatar expects
// the URL to be passed to you (because it's also used for room avatars). // the URL to be passed to you (because it's also used for room avatars).
@ -55,9 +50,9 @@ export default createReactClass({
busy: false, busy: false,
}); });
}); });
}, }
render: function() { render() {
const ChangeDisplayName = sdk.getComponent('settings.ChangeDisplayName'); const ChangeDisplayName = sdk.getComponent('settings.ChangeDisplayName');
const ChangeAvatar = sdk.getComponent('settings.ChangeAvatar'); const ChangeAvatar = sdk.getComponent('settings.ChangeAvatar');
const AuthHeader = sdk.getComponent('auth.AuthHeader'); const AuthHeader = sdk.getComponent('auth.AuthHeader');
@ -78,5 +73,5 @@ export default createReactClass({
</AuthBody> </AuthBody>
</AuthPage> </AuthPage>
); );
}, }
}); }

View file

@ -19,7 +19,6 @@ limitations under the License.
import Matrix from 'matrix-js-sdk'; import Matrix from 'matrix-js-sdk';
import React from 'react'; import React from 'react';
import createReactClass from 'create-react-class';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import * as sdk from '../../../index'; import * as sdk from '../../../index';
import { _t, _td } from '../../../languageHandler'; import { _t, _td } from '../../../languageHandler';
@ -43,10 +42,8 @@ const PHASE_REGISTRATION = 1;
// Enable phases for registration // Enable phases for registration
const PHASES_ENABLED = true; const PHASES_ENABLED = true;
export default createReactClass({ export default class Registration extends React.Component {
displayName: 'Registration', static propTypes = {
propTypes: {
// Called when the user has logged in. Params: // Called when the user has logged in. Params:
// - object with userId, deviceId, homeserverUrl, identityServerUrl, accessToken // - object with userId, deviceId, homeserverUrl, identityServerUrl, accessToken
// - The user's password, if available and applicable (may be cached in memory // - The user's password, if available and applicable (may be cached in memory
@ -65,12 +62,13 @@ export default createReactClass({
onLoginClick: PropTypes.func.isRequired, onLoginClick: PropTypes.func.isRequired,
onServerConfigChange: PropTypes.func.isRequired, onServerConfigChange: PropTypes.func.isRequired,
defaultDeviceDisplayName: PropTypes.string, defaultDeviceDisplayName: PropTypes.string,
}, };
constructor(props) {
super(props);
getInitialState: function() {
const serverType = ServerType.getTypeFromServerConfig(this.props.serverConfig); const serverType = ServerType.getTypeFromServerConfig(this.props.serverConfig);
this.state = {
return {
busy: false, busy: false,
errorText: null, errorText: null,
// We remember the values entered by the user because // We remember the values entered by the user because
@ -118,14 +116,15 @@ export default createReactClass({
// this is the user ID that's logged in. // this is the user ID that's logged in.
differentLoggedInUserId: null, differentLoggedInUserId: null,
}; };
}, }
componentDidMount: function() { componentDidMount() {
this._unmounted = false; this._unmounted = false;
this._replaceClient(); this._replaceClient();
}, }
// TODO: [REACT-WARNING] Replace with appropriate lifecycle event // TODO: [REACT-WARNING] Replace with appropriate lifecycle event
// eslint-disable-next-line camelcase
UNSAFE_componentWillReceiveProps(newProps) { UNSAFE_componentWillReceiveProps(newProps) {
if (newProps.serverConfig.hsUrl === this.props.serverConfig.hsUrl && if (newProps.serverConfig.hsUrl === this.props.serverConfig.hsUrl &&
newProps.serverConfig.isUrl === this.props.serverConfig.isUrl) return; newProps.serverConfig.isUrl === this.props.serverConfig.isUrl) return;
@ -142,7 +141,7 @@ export default createReactClass({
phase: this.getDefaultPhaseForServerType(serverType), phase: this.getDefaultPhaseForServerType(serverType),
}); });
} }
}, }
getDefaultPhaseForServerType(type) { getDefaultPhaseForServerType(type) {
switch (type) { switch (type) {
@ -155,9 +154,9 @@ export default createReactClass({
case ServerType.ADVANCED: case ServerType.ADVANCED:
return PHASE_SERVER_DETAILS; return PHASE_SERVER_DETAILS;
} }
}, }
onServerTypeChange(type) { onServerTypeChange = type => {
this.setState({ this.setState({
serverType: type, serverType: type,
}); });
@ -184,9 +183,9 @@ export default createReactClass({
this.setState({ this.setState({
phase: this.getDefaultPhaseForServerType(type), phase: this.getDefaultPhaseForServerType(type),
}); });
}, };
_replaceClient: async function(serverConfig) { async _replaceClient(serverConfig) {
this.setState({ this.setState({
errorText: null, errorText: null,
serverDeadError: null, serverDeadError: null,
@ -286,18 +285,18 @@ export default createReactClass({
showGenericError(e); showGenericError(e);
} }
} }
}, }
onFormSubmit: function(formVals) { onFormSubmit = formVals => {
this.setState({ this.setState({
errorText: "", errorText: "",
busy: true, busy: true,
formVals: formVals, formVals: formVals,
doingUIAuth: true, doingUIAuth: true,
}); });
}, };
_requestEmailToken: function(emailAddress, clientSecret, sendAttempt, sessionId) { _requestEmailToken = (emailAddress, clientSecret, sendAttempt, sessionId) => {
return this.state.matrixClient.requestRegisterEmailToken( return this.state.matrixClient.requestRegisterEmailToken(
emailAddress, emailAddress,
clientSecret, clientSecret,
@ -309,9 +308,9 @@ export default createReactClass({
session_id: sessionId, session_id: sessionId,
}), }),
); );
}, }
_onUIAuthFinished: async function(success, response, extra) { _onUIAuthFinished = async (success, response, extra) => {
if (!success) { if (!success) {
let msg = response.message || response.toString(); let msg = response.message || response.toString();
// can we give a better error message? // can we give a better error message?
@ -395,9 +394,9 @@ export default createReactClass({
} }
this.setState(newState); this.setState(newState);
}, };
_setupPushers: function() { _setupPushers() {
if (!this.props.brand) { if (!this.props.brand) {
return Promise.resolve(); return Promise.resolve();
} }
@ -418,15 +417,15 @@ export default createReactClass({
}, (error) => { }, (error) => {
console.error("Couldn't get pushers: " + error); console.error("Couldn't get pushers: " + error);
}); });
}, }
onLoginClick: function(ev) { onLoginClick = ev => {
ev.preventDefault(); ev.preventDefault();
ev.stopPropagation(); ev.stopPropagation();
this.props.onLoginClick(); this.props.onLoginClick();
}, };
onGoToFormClicked(ev) { onGoToFormClicked = ev => {
ev.preventDefault(); ev.preventDefault();
ev.stopPropagation(); ev.stopPropagation();
this._replaceClient(); this._replaceClient();
@ -435,23 +434,23 @@ export default createReactClass({
doingUIAuth: false, doingUIAuth: false,
phase: PHASE_REGISTRATION, phase: PHASE_REGISTRATION,
}); });
}, };
async onServerDetailsNextPhaseClick() { onServerDetailsNextPhaseClick = async () => {
this.setState({ this.setState({
phase: PHASE_REGISTRATION, phase: PHASE_REGISTRATION,
}); });
}, };
onEditServerDetailsClick(ev) { onEditServerDetailsClick = ev => {
ev.preventDefault(); ev.preventDefault();
ev.stopPropagation(); ev.stopPropagation();
this.setState({ this.setState({
phase: PHASE_SERVER_DETAILS, phase: PHASE_SERVER_DETAILS,
}); });
}, };
_makeRegisterRequest: function(auth) { _makeRegisterRequest = auth => {
// We inhibit login if we're trying to register with an email address: this // We inhibit login if we're trying to register with an email address: this
// avoids a lot of complex race conditions that can occur if we try to log // avoids a lot of complex race conditions that can occur if we try to log
// the user in one one or both of the tabs they might end up with after // the user in one one or both of the tabs they might end up with after
@ -471,20 +470,20 @@ export default createReactClass({
if (auth) registerParams.auth = auth; if (auth) registerParams.auth = auth;
if (inhibitLogin !== undefined && inhibitLogin !== null) registerParams.inhibit_login = inhibitLogin; if (inhibitLogin !== undefined && inhibitLogin !== null) registerParams.inhibit_login = inhibitLogin;
return this.state.matrixClient.registerRequest(registerParams); return this.state.matrixClient.registerRequest(registerParams);
}, };
_getUIAuthInputs: function() { _getUIAuthInputs() {
return { return {
emailAddress: this.state.formVals.email, emailAddress: this.state.formVals.email,
phoneCountry: this.state.formVals.phoneCountry, phoneCountry: this.state.formVals.phoneCountry,
phoneNumber: this.state.formVals.phoneNumber, phoneNumber: this.state.formVals.phoneNumber,
}; };
}, }
// Links to the login page shown after registration is completed are routed through this // Links to the login page shown after registration is completed are routed through this
// which checks the user hasn't already logged in somewhere else (perhaps we should do // which checks the user hasn't already logged in somewhere else (perhaps we should do
// this more generally?) // this more generally?)
_onLoginClickWithCheck: async function(ev) { _onLoginClickWithCheck = async ev => {
ev.preventDefault(); ev.preventDefault();
const sessionLoaded = await Lifecycle.loadSession({ignoreGuest: true}); const sessionLoaded = await Lifecycle.loadSession({ignoreGuest: true});
@ -492,7 +491,7 @@ export default createReactClass({
// ok fine, there's still no session: really go to the login page // ok fine, there's still no session: really go to the login page
this.props.onLoginClick(); this.props.onLoginClick();
} }
}, };
renderServerComponent() { renderServerComponent() {
const ServerTypeSelector = sdk.getComponent("auth.ServerTypeSelector"); const ServerTypeSelector = sdk.getComponent("auth.ServerTypeSelector");
@ -553,7 +552,7 @@ export default createReactClass({
/> />
{serverDetails} {serverDetails}
</div>; </div>;
}, }
renderRegisterComponent() { renderRegisterComponent() {
if (PHASES_ENABLED && this.state.phase !== PHASE_REGISTRATION) { if (PHASES_ENABLED && this.state.phase !== PHASE_REGISTRATION) {
@ -608,9 +607,9 @@ export default createReactClass({
serverRequiresIdServer={this.state.serverRequiresIdServer} serverRequiresIdServer={this.state.serverRequiresIdServer}
/>; />;
} }
}, }
render: function() { render() {
const AuthHeader = sdk.getComponent('auth.AuthHeader'); const AuthHeader = sdk.getComponent('auth.AuthHeader');
const AuthBody = sdk.getComponent("auth.AuthBody"); const AuthBody = sdk.getComponent("auth.AuthBody");
const AccessibleButton = sdk.getComponent('elements.AccessibleButton'); const AccessibleButton = sdk.getComponent('elements.AccessibleButton');
@ -706,5 +705,5 @@ export default createReactClass({
</AuthBody> </AuthBody>
</AuthPage> </AuthPage>
); );
}, }
}); }

View file

@ -18,16 +18,13 @@ limitations under the License.
import { _t } from '../../../languageHandler'; import { _t } from '../../../languageHandler';
import React from 'react'; import React from 'react';
import createReactClass from 'create-react-class';
export default createReactClass({ export default class AuthFooter extends React.Component {
displayName: 'AuthFooter', render() {
render: function() {
return ( return (
<div className="mx_AuthFooter"> <div className="mx_AuthFooter">
<a href="https://matrix.org" target="_blank" rel="noreferrer noopener">{ _t("powered by Matrix") }</a> <a href="https://matrix.org" target="_blank" rel="noreferrer noopener">{ _t("powered by Matrix") }</a>
</div> </div>
); );
}, }
}); }

View file

@ -17,17 +17,14 @@ limitations under the License.
import React from 'react'; import React from 'react';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import createReactClass from 'create-react-class';
import * as sdk from '../../../index'; import * as sdk from '../../../index';
export default createReactClass({ export default class AuthHeader extends React.Component {
displayName: 'AuthHeader', static propTypes = {
propTypes: {
disableLanguageSelector: PropTypes.bool, disableLanguageSelector: PropTypes.bool,
}, };
render: function() { render() {
const AuthHeaderLogo = sdk.getComponent('auth.AuthHeaderLogo'); const AuthHeaderLogo = sdk.getComponent('auth.AuthHeaderLogo');
const LanguageSelector = sdk.getComponent('views.auth.LanguageSelector'); const LanguageSelector = sdk.getComponent('views.auth.LanguageSelector');
@ -37,5 +34,5 @@ export default createReactClass({
<LanguageSelector disabled={this.props.disableLanguageSelector} /> <LanguageSelector disabled={this.props.disableLanguageSelector} />
</div> </div>
); );
}, }
}); }

View file

@ -15,7 +15,6 @@ limitations under the License.
*/ */
import React, {createRef} from 'react'; import React, {createRef} from 'react';
import createReactClass from 'create-react-class';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import { _t } from '../../../languageHandler'; import { _t } from '../../../languageHandler';
@ -24,36 +23,31 @@ const DIV_ID = 'mx_recaptcha';
/** /**
* A pure UI component which displays a captcha form. * A pure UI component which displays a captcha form.
*/ */
export default createReactClass({ export default class CaptchaForm extends React.Component {
displayName: 'CaptchaForm', static propTypes = {
propTypes: {
sitePublicKey: PropTypes.string, sitePublicKey: PropTypes.string,
// called with the captcha response // called with the captcha response
onCaptchaResponse: PropTypes.func, onCaptchaResponse: PropTypes.func,
}, };
getDefaultProps: function() { static defaultProps = {
return {
onCaptchaResponse: () => {}, onCaptchaResponse: () => {},
}; };
},
getInitialState: function() { constructor(props) {
return { super(props);
this.state = {
errorText: null, errorText: null,
}; };
},
// TODO: [REACT-WARNING] Replace component with real class, use constructor for refs
UNSAFE_componentWillMount: function() {
this._captchaWidgetId = null; this._captchaWidgetId = null;
this._recaptchaContainer = createRef(); this._recaptchaContainer = createRef();
}, }
componentDidMount: function() { componentDidMount() {
// Just putting a script tag into the returned jsx doesn't work, annoyingly, // Just putting a script tag into the returned jsx doesn't work, annoyingly,
// so we do this instead. // so we do this instead.
if (global.grecaptcha) { if (global.grecaptcha) {
@ -68,13 +62,13 @@ export default createReactClass({
); );
this._recaptchaContainer.current.appendChild(scriptTag); this._recaptchaContainer.current.appendChild(scriptTag);
} }
}, }
componentWillUnmount: function() { componentWillUnmount() {
this._resetRecaptcha(); this._resetRecaptcha();
}, }
_renderRecaptcha: function(divId) { _renderRecaptcha(divId) {
if (!global.grecaptcha) { if (!global.grecaptcha) {
console.error("grecaptcha not loaded!"); console.error("grecaptcha not loaded!");
throw new Error("Recaptcha did not load successfully"); throw new Error("Recaptcha did not load successfully");
@ -93,15 +87,15 @@ export default createReactClass({
sitekey: publicKey, sitekey: publicKey,
callback: this.props.onCaptchaResponse, callback: this.props.onCaptchaResponse,
}); });
}, }
_resetRecaptcha: function() { _resetRecaptcha() {
if (this._captchaWidgetId !== null) { if (this._captchaWidgetId !== null) {
global.grecaptcha.reset(this._captchaWidgetId); global.grecaptcha.reset(this._captchaWidgetId);
} }
}, }
_onCaptchaLoaded: function() { _onCaptchaLoaded() {
console.log("Loaded recaptcha script."); console.log("Loaded recaptcha script.");
try { try {
this._renderRecaptcha(DIV_ID); this._renderRecaptcha(DIV_ID);
@ -110,9 +104,9 @@ export default createReactClass({
errorText: e.toString(), errorText: e.toString(),
}); });
} }
}, }
render: function() { render() {
let error = null; let error = null;
if (this.state.errorText) { if (this.state.errorText) {
error = ( error = (
@ -131,5 +125,5 @@ export default createReactClass({
{ error } { error }
</div> </div>
); );
}, }
}); }

View file

@ -16,14 +16,11 @@ limitations under the License.
*/ */
import React from 'react'; import React from 'react';
import createReactClass from 'create-react-class';
import { _t } from '../../../languageHandler'; import { _t } from '../../../languageHandler';
import SdkConfig from '../../../SdkConfig'; import SdkConfig from '../../../SdkConfig';
export default createReactClass({ export default class CustomServerDialog extends React.Component {
displayName: 'CustomServerDialog', render() {
render: function() {
const brand = SdkConfig.get().brand; const brand = SdkConfig.get().brand;
return ( return (
<div className="mx_ErrorDialog"> <div className="mx_ErrorDialog">
@ -46,5 +43,5 @@ export default createReactClass({
</div> </div>
</div> </div>
); );
}, }
}); }

View file

@ -17,7 +17,6 @@ limitations under the License.
*/ */
import React, {createRef} from 'react'; import React, {createRef} from 'react';
import createReactClass from 'create-react-class';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import url from 'url'; import url from 'url';
import classnames from 'classnames'; import classnames from 'classnames';
@ -75,14 +74,10 @@ import AccessibleButton from "../elements/AccessibleButton";
export const DEFAULT_PHASE = 0; export const DEFAULT_PHASE = 0;
export const PasswordAuthEntry = createReactClass({ export class PasswordAuthEntry extends React.Component {
displayName: 'PasswordAuthEntry', static LOGIN_TYPE = "m.login.password";
statics: { static propTypes = {
LOGIN_TYPE: "m.login.password",
},
propTypes: {
matrixClient: PropTypes.object.isRequired, matrixClient: PropTypes.object.isRequired,
submitAuthDict: PropTypes.func.isRequired, submitAuthDict: PropTypes.func.isRequired,
errorText: PropTypes.string, errorText: PropTypes.string,
@ -90,19 +85,17 @@ export const PasswordAuthEntry = createReactClass({
// happen? // happen?
busy: PropTypes.bool, busy: PropTypes.bool,
onPhaseChange: PropTypes.func.isRequired, onPhaseChange: PropTypes.func.isRequired,
}, };
componentDidMount: function() { componentDidMount() {
this.props.onPhaseChange(DEFAULT_PHASE); this.props.onPhaseChange(DEFAULT_PHASE);
}, }
getInitialState: function() { state = {
return {
password: "", password: "",
}; };
},
_onSubmit: function(e) { _onSubmit = e => {
e.preventDefault(); e.preventDefault();
if (this.props.busy) return; if (this.props.busy) return;
@ -117,16 +110,16 @@ export const PasswordAuthEntry = createReactClass({
}, },
password: this.state.password, password: this.state.password,
}); });
}, };
_onPasswordFieldChange: function(ev) { _onPasswordFieldChange = ev => {
// enable the submit button iff the password is non-empty // enable the submit button iff the password is non-empty
this.setState({ this.setState({
password: ev.target.value, password: ev.target.value,
}); });
}, };
render: function() { render() {
const passwordBoxClass = classnames({ const passwordBoxClass = classnames({
"error": this.props.errorText, "error": this.props.errorText,
}); });
@ -176,36 +169,32 @@ export const PasswordAuthEntry = createReactClass({
{ errorSection } { errorSection }
</div> </div>
); );
}, }
}); }
export const RecaptchaAuthEntry = createReactClass({ export class RecaptchaAuthEntry extends React.Component {
displayName: 'RecaptchaAuthEntry', static LOGIN_TYPE = "m.login.recaptcha";
statics: { static propTypes = {
LOGIN_TYPE: "m.login.recaptcha",
},
propTypes: {
submitAuthDict: PropTypes.func.isRequired, submitAuthDict: PropTypes.func.isRequired,
stageParams: PropTypes.object.isRequired, stageParams: PropTypes.object.isRequired,
errorText: PropTypes.string, errorText: PropTypes.string,
busy: PropTypes.bool, busy: PropTypes.bool,
onPhaseChange: PropTypes.func.isRequired, onPhaseChange: PropTypes.func.isRequired,
}, };
componentDidMount: function() { componentDidMount() {
this.props.onPhaseChange(DEFAULT_PHASE); this.props.onPhaseChange(DEFAULT_PHASE);
}, }
_onCaptchaResponse: function(response) { _onCaptchaResponse = response => {
this.props.submitAuthDict({ this.props.submitAuthDict({
type: RecaptchaAuthEntry.LOGIN_TYPE, type: RecaptchaAuthEntry.LOGIN_TYPE,
response: response, response: response,
}); });
}, };
render: function() { render() {
if (this.props.busy) { if (this.props.busy) {
const Loader = sdk.getComponent("elements.Spinner"); const Loader = sdk.getComponent("elements.Spinner");
return <Loader />; return <Loader />;
@ -241,31 +230,24 @@ export const RecaptchaAuthEntry = createReactClass({
{ errorSection } { errorSection }
</div> </div>
); );
}, }
}); }
export const TermsAuthEntry = createReactClass({ export class TermsAuthEntry extends React.Component {
displayName: 'TermsAuthEntry', static LOGIN_TYPE = "m.login.terms";
statics: { static propTypes = {
LOGIN_TYPE: "m.login.terms",
},
propTypes: {
submitAuthDict: PropTypes.func.isRequired, submitAuthDict: PropTypes.func.isRequired,
stageParams: PropTypes.object.isRequired, stageParams: PropTypes.object.isRequired,
errorText: PropTypes.string, errorText: PropTypes.string,
busy: PropTypes.bool, busy: PropTypes.bool,
showContinue: PropTypes.bool, showContinue: PropTypes.bool,
onPhaseChange: PropTypes.func.isRequired, onPhaseChange: PropTypes.func.isRequired,
}, };
componentDidMount: function() { constructor(props) {
this.props.onPhaseChange(DEFAULT_PHASE); super(props);
},
// TODO: [REACT-WARNING] Move this to constructor
componentWillMount: function() {
// example stageParams: // example stageParams:
// //
// { // {
@ -310,17 +292,22 @@ export const TermsAuthEntry = createReactClass({
pickedPolicies.push(langPolicy); pickedPolicies.push(langPolicy);
} }
this.setState({ this.state = {
"toggledPolicies": initToggles, toggledPolicies: initToggles,
"policies": pickedPolicies, policies: pickedPolicies,
}); };
}, }
tryContinue: function() {
componentDidMount() {
this.props.onPhaseChange(DEFAULT_PHASE);
}
tryContinue = () => {
this._trySubmit(); this._trySubmit();
}, };
_togglePolicy: function(policyId) { _togglePolicy(policyId) {
const newToggles = {}; const newToggles = {};
for (const policy of this.state.policies) { for (const policy of this.state.policies) {
let checked = this.state.toggledPolicies[policy.id]; let checked = this.state.toggledPolicies[policy.id];
@ -329,9 +316,9 @@ export const TermsAuthEntry = createReactClass({
newToggles[policy.id] = checked; newToggles[policy.id] = checked;
} }
this.setState({"toggledPolicies": newToggles}); this.setState({"toggledPolicies": newToggles});
}, }
_trySubmit: function() { _trySubmit = () => {
let allChecked = true; let allChecked = true;
for (const policy of this.state.policies) { for (const policy of this.state.policies) {
const checked = this.state.toggledPolicies[policy.id]; const checked = this.state.toggledPolicies[policy.id];
@ -340,9 +327,9 @@ export const TermsAuthEntry = createReactClass({
if (allChecked) this.props.submitAuthDict({type: TermsAuthEntry.LOGIN_TYPE}); if (allChecked) this.props.submitAuthDict({type: TermsAuthEntry.LOGIN_TYPE});
else this.setState({errorText: _t("Please review and accept all of the homeserver's policies")}); else this.setState({errorText: _t("Please review and accept all of the homeserver's policies")});
}, };
render: function() { render() {
if (this.props.busy) { if (this.props.busy) {
const Loader = sdk.getComponent("elements.Spinner"); const Loader = sdk.getComponent("elements.Spinner");
return <Loader />; return <Loader />;
@ -387,17 +374,13 @@ export const TermsAuthEntry = createReactClass({
{ submitButton } { submitButton }
</div> </div>
); );
}, }
}); }
export const EmailIdentityAuthEntry = createReactClass({ export class EmailIdentityAuthEntry extends React.Component {
displayName: 'EmailIdentityAuthEntry', static LOGIN_TYPE = "m.login.email.identity";
statics: { static propTypes = {
LOGIN_TYPE: "m.login.email.identity",
},
propTypes: {
matrixClient: PropTypes.object.isRequired, matrixClient: PropTypes.object.isRequired,
submitAuthDict: PropTypes.func.isRequired, submitAuthDict: PropTypes.func.isRequired,
authSessionId: PropTypes.string.isRequired, authSessionId: PropTypes.string.isRequired,
@ -407,13 +390,13 @@ export const EmailIdentityAuthEntry = createReactClass({
fail: PropTypes.func.isRequired, fail: PropTypes.func.isRequired,
setEmailSid: PropTypes.func.isRequired, setEmailSid: PropTypes.func.isRequired,
onPhaseChange: PropTypes.func.isRequired, onPhaseChange: PropTypes.func.isRequired,
}, };
componentDidMount: function() { componentDidMount() {
this.props.onPhaseChange(DEFAULT_PHASE); this.props.onPhaseChange(DEFAULT_PHASE);
}, }
render: function() { render() {
// This component is now only displayed once the token has been requested, // This component is now only displayed once the token has been requested,
// so we know the email has been sent. It can also get loaded after the user // so we know the email has been sent. It can also get loaded after the user
// has clicked the validation link if the server takes a while to propagate // has clicked the validation link if the server takes a while to propagate
@ -434,17 +417,13 @@ export const EmailIdentityAuthEntry = createReactClass({
</div> </div>
); );
} }
}, }
}); }
export const MsisdnAuthEntry = createReactClass({ export class MsisdnAuthEntry extends React.Component {
displayName: 'MsisdnAuthEntry', static LOGIN_TYPE = "m.login.msisdn";
statics: { static propTypes = {
LOGIN_TYPE: "m.login.msisdn",
},
propTypes: {
inputs: PropTypes.shape({ inputs: PropTypes.shape({
phoneCountry: PropTypes.string, phoneCountry: PropTypes.string,
phoneNumber: PropTypes.string, phoneNumber: PropTypes.string,
@ -454,16 +433,14 @@ export const MsisdnAuthEntry = createReactClass({
submitAuthDict: PropTypes.func.isRequired, submitAuthDict: PropTypes.func.isRequired,
matrixClient: PropTypes.object, matrixClient: PropTypes.object,
onPhaseChange: PropTypes.func.isRequired, onPhaseChange: PropTypes.func.isRequired,
}, };
getInitialState: function() { state = {
return {
token: '', token: '',
requestingToken: false, requestingToken: false,
}; };
},
componentDidMount: function() { componentDidMount() {
this.props.onPhaseChange(DEFAULT_PHASE); this.props.onPhaseChange(DEFAULT_PHASE);
this._submitUrl = null; this._submitUrl = null;
@ -477,12 +454,12 @@ export const MsisdnAuthEntry = createReactClass({
}).finally(() => { }).finally(() => {
this.setState({requestingToken: false}); this.setState({requestingToken: false});
}); });
}, }
/* /*
* Requests a verification token by SMS. * Requests a verification token by SMS.
*/ */
_requestMsisdnToken: function() { _requestMsisdnToken() {
return this.props.matrixClient.requestRegisterMsisdnToken( return this.props.matrixClient.requestRegisterMsisdnToken(
this.props.inputs.phoneCountry, this.props.inputs.phoneCountry,
this.props.inputs.phoneNumber, this.props.inputs.phoneNumber,
@ -493,15 +470,15 @@ export const MsisdnAuthEntry = createReactClass({
this._sid = result.sid; this._sid = result.sid;
this._msisdn = result.msisdn; this._msisdn = result.msisdn;
}); });
}, }
_onTokenChange: function(e) { _onTokenChange = e => {
this.setState({ this.setState({
token: e.target.value, token: e.target.value,
}); });
}, };
_onFormSubmit: async function(e) { _onFormSubmit = async e => {
e.preventDefault(); e.preventDefault();
if (this.state.token == '') return; if (this.state.token == '') return;
@ -552,9 +529,9 @@ export const MsisdnAuthEntry = createReactClass({
this.props.fail(e); this.props.fail(e);
console.log("Failed to submit msisdn token"); console.log("Failed to submit msisdn token");
} }
}, };
render: function() { render() {
if (this.state.requestingToken) { if (this.state.requestingToken) {
const Loader = sdk.getComponent("elements.Spinner"); const Loader = sdk.getComponent("elements.Spinner");
return <Loader />; return <Loader />;
@ -598,8 +575,8 @@ export const MsisdnAuthEntry = createReactClass({
</div> </div>
); );
} }
}, }
}); }
export class SSOAuthEntry extends React.Component { export class SSOAuthEntry extends React.Component {
static propTypes = { static propTypes = {
@ -686,46 +663,46 @@ export class SSOAuthEntry extends React.Component {
} }
} }
export const FallbackAuthEntry = createReactClass({ export class FallbackAuthEntry extends React.Component {
displayName: 'FallbackAuthEntry', static propTypes = {
propTypes: {
matrixClient: PropTypes.object.isRequired, matrixClient: PropTypes.object.isRequired,
authSessionId: PropTypes.string.isRequired, authSessionId: PropTypes.string.isRequired,
loginType: PropTypes.string.isRequired, loginType: PropTypes.string.isRequired,
submitAuthDict: PropTypes.func.isRequired, submitAuthDict: PropTypes.func.isRequired,
errorText: PropTypes.string, errorText: PropTypes.string,
onPhaseChange: PropTypes.func.isRequired, onPhaseChange: PropTypes.func.isRequired,
}, };
componentDidMount: function() { constructor(props) {
this.props.onPhaseChange(DEFAULT_PHASE); super(props);
},
// TODO: [REACT-WARNING] Replace component with real class, use constructor for refs
UNSAFE_componentWillMount: function() {
// we have to make the user click a button, as browsers will block // we have to make the user click a button, as browsers will block
// the popup if we open it immediately. // the popup if we open it immediately.
this._popupWindow = null; this._popupWindow = null;
window.addEventListener("message", this._onReceiveMessage); window.addEventListener("message", this._onReceiveMessage);
this._fallbackButton = createRef(); this._fallbackButton = createRef();
}, }
componentWillUnmount: function() {
componentDidMount() {
this.props.onPhaseChange(DEFAULT_PHASE);
}
componentWillUnmount() {
window.removeEventListener("message", this._onReceiveMessage); window.removeEventListener("message", this._onReceiveMessage);
if (this._popupWindow) { if (this._popupWindow) {
this._popupWindow.close(); this._popupWindow.close();
} }
}, }
focus: function() { focus = () => {
if (this._fallbackButton.current) { if (this._fallbackButton.current) {
this._fallbackButton.current.focus(); this._fallbackButton.current.focus();
} }
}, };
_onShowFallbackClick: function(e) { _onShowFallbackClick = e => {
e.preventDefault(); e.preventDefault();
e.stopPropagation(); e.stopPropagation();
@ -735,18 +712,18 @@ export const FallbackAuthEntry = createReactClass({
); );
this._popupWindow = window.open(url); this._popupWindow = window.open(url);
this._popupWindow.opener = null; this._popupWindow.opener = null;
}, };
_onReceiveMessage: function(event) { _onReceiveMessage = event => {
if ( if (
event.data === "authDone" && event.data === "authDone" &&
event.origin === this.props.matrixClient.getHomeserverUrl() event.origin === this.props.matrixClient.getHomeserverUrl()
) { ) {
this.props.submitAuthDict({}); this.props.submitAuthDict({});
} }
}, };
render: function() { render() {
let errorSection; let errorSection;
if (this.props.errorText) { if (this.props.errorText) {
errorSection = ( errorSection = (
@ -761,8 +738,8 @@ export const FallbackAuthEntry = createReactClass({
{errorSection} {errorSection}
</div> </div>
); );
}, }
}); }
const AuthEntryComponents = [ const AuthEntryComponents = [
PasswordAuthEntry, PasswordAuthEntry,

View file

@ -18,7 +18,6 @@ limitations under the License.
*/ */
import React from 'react'; import React from 'react';
import createReactClass from 'create-react-class';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import * as sdk from '../../../index'; import * as sdk from '../../../index';
import * as Email from '../../../email'; import * as Email from '../../../email';
@ -39,13 +38,11 @@ const FIELD_PASSWORD_CONFIRM = 'field_password_confirm';
const PASSWORD_MIN_SCORE = 3; // safely unguessable: moderate protection from offline slow-hash scenario. const PASSWORD_MIN_SCORE = 3; // safely unguessable: moderate protection from offline slow-hash scenario.
/** /*
* A pure UI component which displays a registration form. * A pure UI component which displays a registration form.
*/ */
export default createReactClass({ export default class RegistrationForm extends React.Component {
displayName: 'RegistrationForm', static propTypes = {
propTypes: {
// Values pre-filled in the input boxes when the component loads // Values pre-filled in the input boxes when the component loads
defaultEmail: PropTypes.string, defaultEmail: PropTypes.string,
defaultPhoneCountry: PropTypes.string, defaultPhoneCountry: PropTypes.string,
@ -58,17 +55,17 @@ export default createReactClass({
serverConfig: PropTypes.instanceOf(ValidatedServerConfig).isRequired, serverConfig: PropTypes.instanceOf(ValidatedServerConfig).isRequired,
canSubmit: PropTypes.bool, canSubmit: PropTypes.bool,
serverRequiresIdServer: PropTypes.bool, serverRequiresIdServer: PropTypes.bool,
}, };
getDefaultProps: function() { static defaultProps = {
return {
onValidationChange: console.error, onValidationChange: console.error,
canSubmit: true, canSubmit: true,
}; };
},
getInitialState: function() { constructor(props) {
return { super(props);
this.state = {
// Field error codes by field ID // Field error codes by field ID
fieldValid: {}, fieldValid: {},
// The ISO2 country code selected in the phone number entry // The ISO2 country code selected in the phone number entry
@ -80,9 +77,9 @@ export default createReactClass({
passwordConfirm: this.props.defaultPassword || "", passwordConfirm: this.props.defaultPassword || "",
passwordComplexity: null, passwordComplexity: null,
}; };
}, }
onSubmit: async function(ev) { onSubmit = async ev => {
ev.preventDefault(); ev.preventDefault();
if (!this.props.canSubmit) return; if (!this.props.canSubmit) return;
@ -118,7 +115,7 @@ export default createReactClass({
title: _t("Warning!"), title: _t("Warning!"),
description: desc, description: desc,
button: _t("Continue"), button: _t("Continue"),
onFinished: function(confirmed) { onFinished(confirmed) {
if (confirmed) { if (confirmed) {
self._doSubmit(ev); self._doSubmit(ev);
} }
@ -127,9 +124,9 @@ export default createReactClass({
} else { } else {
self._doSubmit(ev); self._doSubmit(ev);
} }
}, };
_doSubmit: function(ev) { _doSubmit(ev) {
const email = this.state.email.trim(); const email = this.state.email.trim();
const promise = this.props.onRegisterClick({ const promise = this.props.onRegisterClick({
username: this.state.username.trim(), username: this.state.username.trim(),
@ -145,7 +142,7 @@ export default createReactClass({
ev.target.disabled = false; ev.target.disabled = false;
}); });
} }
}, }
async verifyFieldsBeforeSubmit() { async verifyFieldsBeforeSubmit() {
// Blur the active element if any, so we first run its blur validation, // Blur the active element if any, so we first run its blur validation,
@ -196,12 +193,12 @@ export default createReactClass({
invalidField.focus(); invalidField.focus();
invalidField.validate({ allowEmpty: false, focused: true }); invalidField.validate({ allowEmpty: false, focused: true });
return false; return false;
}, }
/** /**
* @returns {boolean} true if all fields were valid last time they were validated. * @returns {boolean} true if all fields were valid last time they were validated.
*/ */
allFieldsValid: function() { allFieldsValid() {
const keys = Object.keys(this.state.fieldValid); const keys = Object.keys(this.state.fieldValid);
for (let i = 0; i < keys.length; ++i) { for (let i = 0; i < keys.length; ++i) {
if (!this.state.fieldValid[keys[i]]) { if (!this.state.fieldValid[keys[i]]) {
@ -209,7 +206,7 @@ export default createReactClass({
} }
} }
return true; return true;
}, }
findFirstInvalidField(fieldIDs) { findFirstInvalidField(fieldIDs) {
for (const fieldID of fieldIDs) { for (const fieldID of fieldIDs) {
@ -218,34 +215,34 @@ export default createReactClass({
} }
} }
return null; return null;
}, }
markFieldValid: function(fieldID, valid) { markFieldValid(fieldID, valid) {
const { fieldValid } = this.state; const { fieldValid } = this.state;
fieldValid[fieldID] = valid; fieldValid[fieldID] = valid;
this.setState({ this.setState({
fieldValid, fieldValid,
}); });
}, }
onEmailChange(ev) { onEmailChange = ev => {
this.setState({ this.setState({
email: ev.target.value, email: ev.target.value,
}); });
}, };
async onEmailValidate(fieldState) { onEmailValidate = async fieldState => {
const result = await this.validateEmailRules(fieldState); const result = await this.validateEmailRules(fieldState);
this.markFieldValid(FIELD_EMAIL, result.valid); this.markFieldValid(FIELD_EMAIL, result.valid);
return result; return result;
}, };
validateEmailRules: withValidation({ validateEmailRules = withValidation({
description: () => _t("Use an email address to recover your account"), description: () => _t("Use an email address to recover your account"),
rules: [ rules: [
{ {
key: "required", key: "required",
test: function({ value, allowEmpty }) { test({ value, allowEmpty }) {
return allowEmpty || !this._authStepIsRequired('m.login.email.identity') || !!value; return allowEmpty || !this._authStepIsRequired('m.login.email.identity') || !!value;
}, },
invalid: () => _t("Enter email address (required on this homeserver)"), invalid: () => _t("Enter email address (required on this homeserver)"),
@ -256,31 +253,31 @@ export default createReactClass({
invalid: () => _t("Doesn't look like a valid email address"), invalid: () => _t("Doesn't look like a valid email address"),
}, },
], ],
}), });
onPasswordChange(ev) { onPasswordChange = ev => {
this.setState({ this.setState({
password: ev.target.value, password: ev.target.value,
}); });
}, };
onPasswordValidate(result) { onPasswordValidate = result => {
this.markFieldValid(FIELD_PASSWORD, result.valid); this.markFieldValid(FIELD_PASSWORD, result.valid);
}, };
onPasswordConfirmChange(ev) { onPasswordConfirmChange = ev => {
this.setState({ this.setState({
passwordConfirm: ev.target.value, passwordConfirm: ev.target.value,
}); });
}, };
async onPasswordConfirmValidate(fieldState) { onPasswordConfirmValidate = async fieldState => {
const result = await this.validatePasswordConfirmRules(fieldState); const result = await this.validatePasswordConfirmRules(fieldState);
this.markFieldValid(FIELD_PASSWORD_CONFIRM, result.valid); this.markFieldValid(FIELD_PASSWORD_CONFIRM, result.valid);
return result; return result;
}, };
validatePasswordConfirmRules: withValidation({ validatePasswordConfirmRules = withValidation({
rules: [ rules: [
{ {
key: "required", key: "required",
@ -289,39 +286,39 @@ export default createReactClass({
}, },
{ {
key: "match", key: "match",
test: function({ value }) { test({ value }) {
return !value || value === this.state.password; return !value || value === this.state.password;
}, },
invalid: () => _t("Passwords don't match"), invalid: () => _t("Passwords don't match"),
}, },
], ],
}), });
onPhoneCountryChange(newVal) { onPhoneCountryChange = newVal => {
this.setState({ this.setState({
phoneCountry: newVal.iso2, phoneCountry: newVal.iso2,
phonePrefix: newVal.prefix, phonePrefix: newVal.prefix,
}); });
}, };
onPhoneNumberChange(ev) { onPhoneNumberChange = ev => {
this.setState({ this.setState({
phoneNumber: ev.target.value, phoneNumber: ev.target.value,
}); });
}, };
async onPhoneNumberValidate(fieldState) { onPhoneNumberValidate = async fieldState => {
const result = await this.validatePhoneNumberRules(fieldState); const result = await this.validatePhoneNumberRules(fieldState);
this.markFieldValid(FIELD_PHONE_NUMBER, result.valid); this.markFieldValid(FIELD_PHONE_NUMBER, result.valid);
return result; return result;
}, };
validatePhoneNumberRules: withValidation({ validatePhoneNumberRules = withValidation({
description: () => _t("Other users can invite you to rooms using your contact details"), description: () => _t("Other users can invite you to rooms using your contact details"),
rules: [ rules: [
{ {
key: "required", key: "required",
test: function({ value, allowEmpty }) { test({ value, allowEmpty }) {
return allowEmpty || !this._authStepIsRequired('m.login.msisdn') || !!value; return allowEmpty || !this._authStepIsRequired('m.login.msisdn') || !!value;
}, },
invalid: () => _t("Enter phone number (required on this homeserver)"), invalid: () => _t("Enter phone number (required on this homeserver)"),
@ -332,21 +329,21 @@ export default createReactClass({
invalid: () => _t("Doesn't look like a valid phone number"), invalid: () => _t("Doesn't look like a valid phone number"),
}, },
], ],
}), });
onUsernameChange(ev) { onUsernameChange = ev => {
this.setState({ this.setState({
username: ev.target.value, username: ev.target.value,
}); });
}, };
async onUsernameValidate(fieldState) { onUsernameValidate = async fieldState => {
const result = await this.validateUsernameRules(fieldState); const result = await this.validateUsernameRules(fieldState);
this.markFieldValid(FIELD_USERNAME, result.valid); this.markFieldValid(FIELD_USERNAME, result.valid);
return result; return result;
}, };
validateUsernameRules: withValidation({ validateUsernameRules = withValidation({
description: () => _t("Use lowercase letters, numbers, dashes and underscores only"), description: () => _t("Use lowercase letters, numbers, dashes and underscores only"),
rules: [ rules: [
{ {
@ -360,7 +357,7 @@ export default createReactClass({
invalid: () => _t("Some characters not allowed"), invalid: () => _t("Some characters not allowed"),
}, },
], ],
}), });
/** /**
* A step is required if all flows include that step. * A step is required if all flows include that step.
@ -372,7 +369,7 @@ export default createReactClass({
return this.props.flows.every((flow) => { return this.props.flows.every((flow) => {
return flow.stages.includes(step); return flow.stages.includes(step);
}); });
}, }
/** /**
* A step is used if any flows include that step. * A step is used if any flows include that step.
@ -384,7 +381,7 @@ export default createReactClass({
return this.props.flows.some((flow) => { return this.props.flows.some((flow) => {
return flow.stages.includes(step); return flow.stages.includes(step);
}); });
}, }
_showEmail() { _showEmail() {
const haveIs = Boolean(this.props.serverConfig.isUrl); const haveIs = Boolean(this.props.serverConfig.isUrl);
@ -395,7 +392,7 @@ export default createReactClass({
return false; return false;
} }
return true; return true;
}, }
_showPhoneNumber() { _showPhoneNumber() {
const threePidLogin = !SdkConfig.get().disable_3pid_login; const threePidLogin = !SdkConfig.get().disable_3pid_login;
@ -408,7 +405,7 @@ export default createReactClass({
return false; return false;
} }
return true; return true;
}, }
renderEmail() { renderEmail() {
if (!this._showEmail()) { if (!this._showEmail()) {
@ -426,7 +423,7 @@ export default createReactClass({
onChange={this.onEmailChange} onChange={this.onEmailChange}
onValidate={this.onEmailValidate} onValidate={this.onEmailValidate}
/>; />;
}, }
renderPassword() { renderPassword() {
return <PassphraseField return <PassphraseField
@ -437,7 +434,7 @@ export default createReactClass({
onChange={this.onPasswordChange} onChange={this.onPasswordChange}
onValidate={this.onPasswordValidate} onValidate={this.onPasswordValidate}
/>; />;
}, }
renderPasswordConfirm() { renderPasswordConfirm() {
const Field = sdk.getComponent('elements.Field'); const Field = sdk.getComponent('elements.Field');
@ -451,7 +448,7 @@ export default createReactClass({
onChange={this.onPasswordConfirmChange} onChange={this.onPasswordConfirmChange}
onValidate={this.onPasswordConfirmValidate} onValidate={this.onPasswordConfirmValidate}
/>; />;
}, }
renderPhoneNumber() { renderPhoneNumber() {
if (!this._showPhoneNumber()) { if (!this._showPhoneNumber()) {
@ -477,7 +474,7 @@ export default createReactClass({
onChange={this.onPhoneNumberChange} onChange={this.onPhoneNumberChange}
onValidate={this.onPhoneNumberValidate} onValidate={this.onPhoneNumberValidate}
/>; />;
}, }
renderUsername() { renderUsername() {
const Field = sdk.getComponent('elements.Field'); const Field = sdk.getComponent('elements.Field');
@ -491,9 +488,9 @@ export default createReactClass({
onChange={this.onUsernameChange} onChange={this.onUsernameChange}
onValidate={this.onUsernameValidate} onValidate={this.onUsernameValidate}
/>; />;
}, }
render: function() { render() {
let yourMatrixAccountText = _t('Create your Matrix account on %(serverName)s', { let yourMatrixAccountText = _t('Create your Matrix account on %(serverName)s', {
serverName: this.props.serverConfig.hsName, serverName: this.props.serverConfig.hsName,
}); });
@ -578,5 +575,5 @@ export default createReactClass({
</form> </form>
</div> </div>
); );
}, }
}); }

View file

@ -17,7 +17,7 @@ See the License for the specific language governing permissions and
limitations under the License. limitations under the License.
*/ */
import React, {useCallback, useContext, useEffect, useMemo, useState} from 'react'; import React, {useCallback, useContext, useEffect, useState} from 'react';
import classNames from 'classnames'; import classNames from 'classnames';
import * as AvatarLogic from '../../../Avatar'; import * as AvatarLogic from '../../../Avatar';
import SettingsStore from "../../../settings/SettingsStore"; import SettingsStore from "../../../settings/SettingsStore";
@ -96,7 +96,7 @@ const BaseAvatar = (props: IProps) => {
urls, urls,
width = 40, width = 40,
height = 40, height = 40,
resizeMethod = "crop", // eslint-disable-line no-unused-vars resizeMethod = "crop", // eslint-disable-line @typescript-eslint/no-unused-vars
defaultToInitialLetter = true, defaultToInitialLetter = true,
onClick, onClick,
inputRef, inputRef,

View file

@ -126,7 +126,7 @@ export default class DecoratedRoomAvatar extends React.PureComponent<IProps, ISt
private onPresenceUpdate = () => { private onPresenceUpdate = () => {
if (this.isUnmounted) return; if (this.isUnmounted) return;
let newIcon = this.getPresenceIcon(); const newIcon = this.getPresenceIcon();
if (newIcon !== this.state.icon) this.setState({icon: newIcon}); if (newIcon !== this.state.icon) this.setState({icon: newIcon});
}; };

View file

@ -47,7 +47,7 @@ export default class GroupAvatar extends React.Component<IProps> {
render() { render() {
// extract the props we use from props so we can pass any others through // extract the props we use from props so we can pass any others through
// should consider adding this as a global rule in js-sdk? // should consider adding this as a global rule in js-sdk?
/*eslint no-unused-vars: ["error", { "ignoreRestSiblings": true }]*/ /* eslint @typescript-eslint/no-unused-vars: ["error", { "ignoreRestSiblings": true }] */
const {groupId, groupAvatarUrl, groupName, ...otherProps} = this.props; const {groupId, groupAvatarUrl, groupName, ...otherProps} = this.props;
return ( return (

View file

@ -16,23 +16,24 @@ limitations under the License.
*/ */
import React from 'react'; import React from 'react';
import {RoomMember} from "matrix-js-sdk/src/models/room-member";
import dis from "../../../dispatcher/dispatcher"; import dis from "../../../dispatcher/dispatcher";
import {Action} from "../../../dispatcher/actions"; import {Action} from "../../../dispatcher/actions";
import {MatrixClientPeg} from "../../../MatrixClientPeg"; import {MatrixClientPeg} from "../../../MatrixClientPeg";
import BaseAvatar from "./BaseAvatar"; import BaseAvatar from "./BaseAvatar";
interface IProps { interface IProps {
// TODO: replace with correct type member: RoomMember;
member: any; fallbackUserId?: string;
fallbackUserId: string;
width: number; width: number;
height: number; height: number;
resizeMethod: string; resizeMethod?: string;
// The onClick to give the avatar // The onClick to give the avatar
onClick: React.MouseEventHandler; onClick?: React.MouseEventHandler;
// Whether the onClick of the avatar should be overriden to dispatch `Action.ViewUser` // Whether the onClick of the avatar should be overriden to dispatch `Action.ViewUser`
viewUserOnClick: boolean; viewUserOnClick?: boolean;
title: string; title?: string;
} }
interface IState { interface IState {

View file

@ -19,7 +19,6 @@ limitations under the License.
import React from 'react'; import React from 'react';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import createReactClass from 'create-react-class';
import {EventStatus} from 'matrix-js-sdk'; import {EventStatus} from 'matrix-js-sdk';
import {MatrixClientPeg} from '../../../MatrixClientPeg'; import {MatrixClientPeg} from '../../../MatrixClientPeg';
@ -37,10 +36,8 @@ function canCancel(eventStatus) {
return eventStatus === EventStatus.QUEUED || eventStatus === EventStatus.NOT_SENT; return eventStatus === EventStatus.QUEUED || eventStatus === EventStatus.NOT_SENT;
} }
export default createReactClass({ export default class MessageContextMenu extends React.Component {
displayName: 'MessageContextMenu', static propTypes = {
propTypes: {
/* the MatrixEvent associated with the context menu */ /* the MatrixEvent associated with the context menu */
mxEvent: PropTypes.object.isRequired, mxEvent: PropTypes.object.isRequired,
@ -52,28 +49,26 @@ export default createReactClass({
/* callback called when the menu is dismissed */ /* callback called when the menu is dismissed */
onFinished: PropTypes.func, onFinished: PropTypes.func,
}, };
getInitialState: function() { state = {
return {
canRedact: false, canRedact: false,
canPin: false, canPin: false,
}; };
},
componentDidMount: function() { componentDidMount() {
MatrixClientPeg.get().on('RoomMember.powerLevel', this._checkPermissions); MatrixClientPeg.get().on('RoomMember.powerLevel', this._checkPermissions);
this._checkPermissions(); this._checkPermissions();
}, }
componentWillUnmount: function() { componentWillUnmount() {
const cli = MatrixClientPeg.get(); const cli = MatrixClientPeg.get();
if (cli) { if (cli) {
cli.removeListener('RoomMember.powerLevel', this._checkPermissions); cli.removeListener('RoomMember.powerLevel', this._checkPermissions);
} }
}, }
_checkPermissions: function() { _checkPermissions = () => {
const cli = MatrixClientPeg.get(); const cli = MatrixClientPeg.get();
const room = cli.getRoom(this.props.mxEvent.getRoomId()); const room = cli.getRoom(this.props.mxEvent.getRoomId());
@ -84,47 +79,47 @@ export default createReactClass({
if (!SettingsStore.getValue("feature_pinning")) canPin = false; if (!SettingsStore.getValue("feature_pinning")) canPin = false;
this.setState({canRedact, canPin}); this.setState({canRedact, canPin});
}, };
_isPinned: function() { _isPinned() {
const room = MatrixClientPeg.get().getRoom(this.props.mxEvent.getRoomId()); const room = MatrixClientPeg.get().getRoom(this.props.mxEvent.getRoomId());
const pinnedEvent = room.currentState.getStateEvents('m.room.pinned_events', ''); const pinnedEvent = room.currentState.getStateEvents('m.room.pinned_events', '');
if (!pinnedEvent) return false; if (!pinnedEvent) return false;
const content = pinnedEvent.getContent(); const content = pinnedEvent.getContent();
return content.pinned && Array.isArray(content.pinned) && content.pinned.includes(this.props.mxEvent.getId()); return content.pinned && Array.isArray(content.pinned) && content.pinned.includes(this.props.mxEvent.getId());
}, }
onResendClick: function() { onResendClick = () => {
Resend.resend(this.props.mxEvent); Resend.resend(this.props.mxEvent);
this.closeMenu(); this.closeMenu();
}, };
onResendEditClick: function() { onResendEditClick = () => {
Resend.resend(this.props.mxEvent.replacingEvent()); Resend.resend(this.props.mxEvent.replacingEvent());
this.closeMenu(); this.closeMenu();
}, };
onResendRedactionClick: function() { onResendRedactionClick = () => {
Resend.resend(this.props.mxEvent.localRedactionEvent()); Resend.resend(this.props.mxEvent.localRedactionEvent());
this.closeMenu(); this.closeMenu();
}, };
onResendReactionsClick: function() { onResendReactionsClick = () => {
for (const reaction of this._getUnsentReactions()) { for (const reaction of this._getUnsentReactions()) {
Resend.resend(reaction); Resend.resend(reaction);
} }
this.closeMenu(); this.closeMenu();
}, };
onReportEventClick: function() { onReportEventClick = () => {
const ReportEventDialog = sdk.getComponent("dialogs.ReportEventDialog"); const ReportEventDialog = sdk.getComponent("dialogs.ReportEventDialog");
Modal.createTrackedDialog('Report Event', '', ReportEventDialog, { Modal.createTrackedDialog('Report Event', '', ReportEventDialog, {
mxEvent: this.props.mxEvent, mxEvent: this.props.mxEvent,
}, 'mx_Dialog_reportEvent'); }, 'mx_Dialog_reportEvent');
this.closeMenu(); this.closeMenu();
}, };
onViewSourceClick: function() { onViewSourceClick = () => {
const ev = this.props.mxEvent.replacingEvent() || this.props.mxEvent; const ev = this.props.mxEvent.replacingEvent() || this.props.mxEvent;
const ViewSource = sdk.getComponent('structures.ViewSource'); const ViewSource = sdk.getComponent('structures.ViewSource');
Modal.createTrackedDialog('View Event Source', '', ViewSource, { Modal.createTrackedDialog('View Event Source', '', ViewSource, {
@ -133,9 +128,9 @@ export default createReactClass({
content: ev.event, content: ev.event,
}, 'mx_Dialog_viewsource'); }, 'mx_Dialog_viewsource');
this.closeMenu(); this.closeMenu();
}, };
onViewClearSourceClick: function() { onViewClearSourceClick = () => {
const ev = this.props.mxEvent.replacingEvent() || this.props.mxEvent; const ev = this.props.mxEvent.replacingEvent() || this.props.mxEvent;
const ViewSource = sdk.getComponent('structures.ViewSource'); const ViewSource = sdk.getComponent('structures.ViewSource');
Modal.createTrackedDialog('View Clear Event Source', '', ViewSource, { Modal.createTrackedDialog('View Clear Event Source', '', ViewSource, {
@ -145,9 +140,9 @@ export default createReactClass({
content: ev._clearEvent, content: ev._clearEvent,
}, 'mx_Dialog_viewsource'); }, 'mx_Dialog_viewsource');
this.closeMenu(); this.closeMenu();
}, };
onRedactClick: function() { onRedactClick = () => {
const ConfirmRedactDialog = sdk.getComponent("dialogs.ConfirmRedactDialog"); const ConfirmRedactDialog = sdk.getComponent("dialogs.ConfirmRedactDialog");
Modal.createTrackedDialog('Confirm Redact Dialog', '', ConfirmRedactDialog, { Modal.createTrackedDialog('Confirm Redact Dialog', '', ConfirmRedactDialog, {
onFinished: async (proceed) => { onFinished: async (proceed) => {
@ -176,9 +171,9 @@ export default createReactClass({
}, },
}, 'mx_Dialog_confirmredact'); }, 'mx_Dialog_confirmredact');
this.closeMenu(); this.closeMenu();
}, };
onCancelSendClick: function() { onCancelSendClick = () => {
const mxEvent = this.props.mxEvent; const mxEvent = this.props.mxEvent;
const editEvent = mxEvent.replacingEvent(); const editEvent = mxEvent.replacingEvent();
const redactEvent = mxEvent.localRedactionEvent(); const redactEvent = mxEvent.localRedactionEvent();
@ -199,17 +194,17 @@ export default createReactClass({
Resend.removeFromQueue(this.props.mxEvent); Resend.removeFromQueue(this.props.mxEvent);
} }
this.closeMenu(); this.closeMenu();
}, };
onForwardClick: function() { onForwardClick = () => {
dis.dispatch({ dis.dispatch({
action: 'forward_event', action: 'forward_event',
event: this.props.mxEvent, event: this.props.mxEvent,
}); });
this.closeMenu(); this.closeMenu();
}, };
onPinClick: function() { onPinClick = () => {
MatrixClientPeg.get().getStateEvent(this.props.mxEvent.getRoomId(), 'm.room.pinned_events', '') MatrixClientPeg.get().getStateEvent(this.props.mxEvent.getRoomId(), 'm.room.pinned_events', '')
.catch((e) => { .catch((e) => {
// Intercept the Event Not Found error and fall through the promise chain with no event. // Intercept the Event Not Found error and fall through the promise chain with no event.
@ -230,28 +225,28 @@ export default createReactClass({
cli.sendStateEvent(this.props.mxEvent.getRoomId(), 'm.room.pinned_events', {pinned: eventIds}, ''); cli.sendStateEvent(this.props.mxEvent.getRoomId(), 'm.room.pinned_events', {pinned: eventIds}, '');
}); });
this.closeMenu(); this.closeMenu();
}, };
closeMenu: function() { closeMenu = () => {
if (this.props.onFinished) this.props.onFinished(); if (this.props.onFinished) this.props.onFinished();
}, };
onUnhidePreviewClick: function() { onUnhidePreviewClick = () => {
if (this.props.eventTileOps) { if (this.props.eventTileOps) {
this.props.eventTileOps.unhideWidget(); this.props.eventTileOps.unhideWidget();
} }
this.closeMenu(); this.closeMenu();
}, };
onQuoteClick: function() { onQuoteClick = () => {
dis.dispatch({ dis.dispatch({
action: 'quote', action: 'quote',
event: this.props.mxEvent, event: this.props.mxEvent,
}); });
this.closeMenu(); this.closeMenu();
}, };
onPermalinkClick: function(e: Event) { onPermalinkClick = (e: Event) => {
e.preventDefault(); e.preventDefault();
const ShareDialog = sdk.getComponent("dialogs.ShareDialog"); const ShareDialog = sdk.getComponent("dialogs.ShareDialog");
Modal.createTrackedDialog('share room message dialog', '', ShareDialog, { Modal.createTrackedDialog('share room message dialog', '', ShareDialog, {
@ -259,12 +254,12 @@ export default createReactClass({
permalinkCreator: this.props.permalinkCreator, permalinkCreator: this.props.permalinkCreator,
}); });
this.closeMenu(); this.closeMenu();
}, };
onCollapseReplyThreadClick: function() { onCollapseReplyThreadClick = () => {
this.props.collapseReplyThread(); this.props.collapseReplyThread();
this.closeMenu(); this.closeMenu();
}, };
_getReactions(filter) { _getReactions(filter) {
const cli = MatrixClientPeg.get(); const cli = MatrixClientPeg.get();
@ -277,17 +272,17 @@ export default createReactClass({
relation.event_id === eventId && relation.event_id === eventId &&
filter(e); filter(e);
}); });
}, }
_getPendingReactions() { _getPendingReactions() {
return this._getReactions(e => canCancel(e.status)); return this._getReactions(e => canCancel(e.status));
}, }
_getUnsentReactions() { _getUnsentReactions() {
return this._getReactions(e => e.status === EventStatus.NOT_SENT); return this._getReactions(e => e.status === EventStatus.NOT_SENT);
}, }
render: function() { render() {
const cli = MatrixClientPeg.get(); const cli = MatrixClientPeg.get();
const me = cli.getUserId(); const me = cli.getUserId();
const mxEvent = this.props.mxEvent; const mxEvent = this.props.mxEvent;
@ -489,5 +484,5 @@ export default createReactClass({
{ reportEventButton } { reportEventButton }
</div> </div>
); );
}, }
}); }

View file

@ -1,57 +0,0 @@
/*
Copyright 2015, 2016 OpenMarket Ltd
Copyright 2019 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 PropTypes from 'prop-types';
import createReactClass from 'create-react-class';
import { _t } from '../../../languageHandler';
const Presets = {
PrivateChat: "private_chat",
PublicChat: "public_chat",
Custom: "custom",
};
export default createReactClass({
displayName: 'CreateRoomPresets',
propTypes: {
onChange: PropTypes.func,
preset: PropTypes.string,
},
Presets: Presets,
getDefaultProps: function() {
return {
onChange: function() {},
};
},
onValueChanged: function(ev) {
this.props.onChange(ev.target.value);
},
render: function() {
return (
<select className="mx_Presets" onChange={this.onValueChanged} value={this.props.preset}>
<option value={this.Presets.PrivateChat}>{ _t("Private Chat") }</option>
<option value={this.Presets.PublicChat}>{ _t("Public Chat") }</option>
<option value={this.Presets.Custom}>{ _t("Custom") }</option>
</select>
);
},
});

View file

@ -1,106 +0,0 @@
/*
Copyright 2015, 2016 OpenMarket Ltd
Copyright 2019 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 PropTypes from 'prop-types';
import createReactClass from 'create-react-class';
import { _t } from '../../../languageHandler';
export default createReactClass({
displayName: 'RoomAlias',
propTypes: {
// Specifying a homeserver will make magical things happen when you,
// e.g. start typing in the room alias box.
homeserver: PropTypes.string,
alias: PropTypes.string,
onChange: PropTypes.func,
},
getDefaultProps: function() {
return {
onChange: function() {},
alias: '',
};
},
getAliasLocalpart: function() {
let room_alias = this.props.alias;
if (room_alias && this.props.homeserver) {
const suffix = ":" + this.props.homeserver;
if (room_alias.startsWith("#") && room_alias.endsWith(suffix)) {
room_alias = room_alias.slice(1, -suffix.length);
}
}
return room_alias;
},
onValueChanged: function(ev) {
this.props.onChange(ev.target.value);
},
onFocus: function(ev) {
const target = ev.target;
const curr_val = ev.target.value;
if (this.props.homeserver) {
if (curr_val == "") {
const self = this;
setTimeout(function() {
target.value = "#:" + self.props.homeserver;
target.setSelectionRange(1, 1);
}, 0);
} else {
const suffix = ":" + this.props.homeserver;
setTimeout(function() {
target.setSelectionRange(
curr_val.startsWith("#") ? 1 : 0,
curr_val.endsWith(suffix) ? (target.value.length - suffix.length) : target.value.length,
);
}, 0);
}
}
},
onBlur: function(ev) {
const curr_val = ev.target.value;
if (this.props.homeserver) {
if (curr_val == "#:" + this.props.homeserver) {
ev.target.value = "";
return;
}
if (curr_val != "") {
let new_val = ev.target.value;
const suffix = ":" + this.props.homeserver;
if (!curr_val.startsWith("#")) new_val = "#" + new_val;
if (!curr_val.endsWith(suffix)) new_val = new_val + suffix;
ev.target.value = new_val;
}
}
},
render: function() {
return (
<input type="text" className="mx_RoomAlias" placeholder={_t("Address (optional)")}
onChange={this.onValueChanged} onFocus={this.onFocus} onBlur={this.onBlur}
value={this.props.alias} />
);
},
});

View file

@ -19,7 +19,6 @@ limitations under the License.
import React, {createRef} from 'react'; import React, {createRef} from 'react';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import createReactClass from 'create-react-class';
import { _t, _td } from '../../../languageHandler'; import { _t, _td } from '../../../languageHandler';
import * as sdk from '../../../index'; import * as sdk from '../../../index';
@ -45,10 +44,8 @@ const addressTypeName = {
}; };
export default createReactClass({ export default class AddressPickerDialog extends React.Component {
displayName: "AddressPickerDialog", static propTypes = {
propTypes: {
title: PropTypes.string.isRequired, title: PropTypes.string.isRequired,
description: PropTypes.node, description: PropTypes.node,
// Extra node inserted after picker input, dropdown and errors // Extra node inserted after picker input, dropdown and errors
@ -66,26 +63,28 @@ export default createReactClass({
// Whether the current user should be included in the addresses returned. Only // Whether the current user should be included in the addresses returned. Only
// applicable when pickerType is `user`. Default: false. // applicable when pickerType is `user`. Default: false.
includeSelf: PropTypes.bool, includeSelf: PropTypes.bool,
}, };
getDefaultProps: function() { static defaultProps = {
return {
value: "", value: "",
focus: true, focus: true,
validAddressTypes: addressTypes, validAddressTypes: addressTypes,
pickerType: 'user', pickerType: 'user',
includeSelf: false, includeSelf: false,
}; };
},
getInitialState: function() { constructor(props) {
super(props);
this._textinput = createRef();
let validAddressTypes = this.props.validAddressTypes; let validAddressTypes = this.props.validAddressTypes;
// Remove email from validAddressTypes if no IS is configured. It may be added at a later stage by the user // Remove email from validAddressTypes if no IS is configured. It may be added at a later stage by the user
if (!MatrixClientPeg.get().getIdentityServerUrl() && validAddressTypes.includes("email")) { if (!MatrixClientPeg.get().getIdentityServerUrl() && validAddressTypes.includes("email")) {
validAddressTypes = validAddressTypes.filter(type => type !== "email"); validAddressTypes = validAddressTypes.filter(type => type !== "email");
} }
return { this.state = {
// Whether to show an error message because of an invalid address // Whether to show an error message because of an invalid address
invalidAddressError: false, invalidAddressError: false,
// List of UserAddressType objects representing // List of UserAddressType objects representing
@ -106,19 +105,14 @@ export default createReactClass({
// dialog is open and represents the supported list of address types at this time. // dialog is open and represents the supported list of address types at this time.
validAddressTypes, validAddressTypes,
}; };
}, }
// TODO: [REACT-WARNING] Replace component with real class, use constructor for refs componentDidMount() {
UNSAFE_componentWillMount: function() {
this._textinput = createRef();
},
componentDidMount: function() {
if (this.props.focus) { if (this.props.focus) {
// Set the cursor at the end of the text input // Set the cursor at the end of the text input
this._textinput.current.value = this.props.value; this._textinput.current.value = this.props.value;
} }
}, }
getPlaceholder() { getPlaceholder() {
const { placeholder } = this.props; const { placeholder } = this.props;
@ -127,9 +121,9 @@ export default createReactClass({
} }
// Otherwise it's a function, as checked by prop types. // Otherwise it's a function, as checked by prop types.
return placeholder(this.state.validAddressTypes); return placeholder(this.state.validAddressTypes);
}, }
onButtonClick: function() { onButtonClick = () => {
let selectedList = this.state.selectedList.slice(); let selectedList = this.state.selectedList.slice();
// Check the text input field to see if user has an unconverted address // Check the text input field to see if user has an unconverted address
// If there is and it's valid add it to the local selectedList // If there is and it's valid add it to the local selectedList
@ -138,13 +132,13 @@ export default createReactClass({
if (selectedList === null) return; if (selectedList === null) return;
} }
this.props.onFinished(true, selectedList); this.props.onFinished(true, selectedList);
}, };
onCancel: function() { onCancel = () => {
this.props.onFinished(false); this.props.onFinished(false);
}, };
onKeyDown: function(e) { onKeyDown = e => {
const textInput = this._textinput.current ? this._textinput.current.value : undefined; const textInput = this._textinput.current ? this._textinput.current.value : undefined;
if (e.key === Key.ESCAPE) { if (e.key === Key.ESCAPE) {
@ -181,9 +175,9 @@ export default createReactClass({
e.preventDefault(); e.preventDefault();
this._addAddressesToList([textInput]); this._addAddressesToList([textInput]);
} }
}, };
onQueryChanged: function(ev) { onQueryChanged = ev => {
const query = ev.target.value; const query = ev.target.value;
if (this.queryChangedDebouncer) { if (this.queryChangedDebouncer) {
clearTimeout(this.queryChangedDebouncer); clearTimeout(this.queryChangedDebouncer);
@ -216,10 +210,9 @@ export default createReactClass({
searchError: null, searchError: null,
}); });
} }
}, };
onDismissed: function(index) { onDismissed = index => () => {
return () => {
const selectedList = this.state.selectedList.slice(); const selectedList = this.state.selectedList.slice();
selectedList.splice(index, 1); selectedList.splice(index, 1);
this.setState({ this.setState({
@ -229,15 +222,12 @@ export default createReactClass({
}); });
if (this._cancelThreepidLookup) this._cancelThreepidLookup(); if (this._cancelThreepidLookup) this._cancelThreepidLookup();
}; };
},
onClick: function(index) { onClick = index => () => {
return () => {
this.onSelected(index); this.onSelected(index);
}; };
},
onSelected: function(index) { onSelected = index => {
const selectedList = this.state.selectedList.slice(); const selectedList = this.state.selectedList.slice();
selectedList.push(this._getFilteredSuggestions()[index]); selectedList.push(this._getFilteredSuggestions()[index]);
this.setState({ this.setState({
@ -246,9 +236,9 @@ export default createReactClass({
query: "", query: "",
}); });
if (this._cancelThreepidLookup) this._cancelThreepidLookup(); if (this._cancelThreepidLookup) this._cancelThreepidLookup();
}, };
_doNaiveGroupSearch: function(query) { _doNaiveGroupSearch(query) {
const lowerCaseQuery = query.toLowerCase(); const lowerCaseQuery = query.toLowerCase();
this.setState({ this.setState({
busy: true, busy: true,
@ -280,9 +270,9 @@ export default createReactClass({
busy: false, busy: false,
}); });
}); });
}, }
_doNaiveGroupRoomSearch: function(query) { _doNaiveGroupRoomSearch(query) {
const lowerCaseQuery = query.toLowerCase(); const lowerCaseQuery = query.toLowerCase();
const results = []; const results = [];
GroupStore.getGroupRooms(this.props.groupId).forEach((r) => { GroupStore.getGroupRooms(this.props.groupId).forEach((r) => {
@ -302,9 +292,9 @@ export default createReactClass({
this.setState({ this.setState({
busy: false, busy: false,
}); });
}, }
_doRoomSearch: function(query) { _doRoomSearch(query) {
const lowerCaseQuery = query.toLowerCase(); const lowerCaseQuery = query.toLowerCase();
const rooms = MatrixClientPeg.get().getRooms(); const rooms = MatrixClientPeg.get().getRooms();
const results = []; const results = [];
@ -359,9 +349,9 @@ export default createReactClass({
this.setState({ this.setState({
busy: false, busy: false,
}); });
}, }
_doUserDirectorySearch: function(query) { _doUserDirectorySearch(query) {
this.setState({ this.setState({
busy: true, busy: true,
query, query,
@ -393,9 +383,9 @@ export default createReactClass({
busy: false, busy: false,
}); });
}); });
}, }
_doLocalSearch: function(query) { _doLocalSearch(query) {
this.setState({ this.setState({
query, query,
searchError: null, searchError: null,
@ -417,9 +407,9 @@ export default createReactClass({
}); });
}); });
this._processResults(results, query); this._processResults(results, query);
}, }
_processResults: function(results, query) { _processResults(results, query) {
const suggestedList = []; const suggestedList = [];
results.forEach((result) => { results.forEach((result) => {
if (result.room_id) { if (result.room_id) {
@ -485,9 +475,9 @@ export default createReactClass({
}, () => { }, () => {
if (this.addressSelector) this.addressSelector.moveSelectionTop(); if (this.addressSelector) this.addressSelector.moveSelectionTop();
}); });
}, }
_addAddressesToList: function(addressTexts) { _addAddressesToList(addressTexts) {
const selectedList = this.state.selectedList.slice(); const selectedList = this.state.selectedList.slice();
let hasError = false; let hasError = false;
@ -529,9 +519,9 @@ export default createReactClass({
}); });
if (this._cancelThreepidLookup) this._cancelThreepidLookup(); if (this._cancelThreepidLookup) this._cancelThreepidLookup();
return hasError ? null : selectedList; return hasError ? null : selectedList;
}, }
_lookupThreepid: async function(medium, address) { async _lookupThreepid(medium, address) {
let cancelled = false; let cancelled = false;
// Note that we can't safely remove this after we're done // Note that we can't safely remove this after we're done
// because we don't know that it's the same one, so we just // because we don't know that it's the same one, so we just
@ -577,9 +567,9 @@ export default createReactClass({
searchError: _t('Something went wrong!'), searchError: _t('Something went wrong!'),
}); });
} }
}, }
_getFilteredSuggestions: function() { _getFilteredSuggestions() {
// map addressType => set of addresses to avoid O(n*m) operation // map addressType => set of addresses to avoid O(n*m) operation
const selectedAddresses = {}; const selectedAddresses = {};
this.state.selectedList.forEach(({address, addressType}) => { this.state.selectedList.forEach(({address, addressType}) => {
@ -591,17 +581,17 @@ export default createReactClass({
return this.state.suggestedList.filter(({address, addressType}) => { return this.state.suggestedList.filter(({address, addressType}) => {
return !(selectedAddresses[addressType] && selectedAddresses[addressType].has(address)); return !(selectedAddresses[addressType] && selectedAddresses[addressType].has(address));
}); });
}, }
_onPaste: function(e) { _onPaste = e => {
// Prevent the text being pasted into the textarea // Prevent the text being pasted into the textarea
e.preventDefault(); e.preventDefault();
const text = e.clipboardData.getData("text"); const text = e.clipboardData.getData("text");
// Process it as a list of addresses to add instead // Process it as a list of addresses to add instead
this._addAddressesToList(text.split(/[\s,]+/)); this._addAddressesToList(text.split(/[\s,]+/));
}, };
onUseDefaultIdentityServerClick(e) { onUseDefaultIdentityServerClick = e => {
e.preventDefault(); e.preventDefault();
// Update the IS in account data. Actually using it may trigger terms. // Update the IS in account data. Actually using it may trigger terms.
@ -612,15 +602,15 @@ export default createReactClass({
const { validAddressTypes } = this.state; const { validAddressTypes } = this.state;
validAddressTypes.push('email'); validAddressTypes.push('email');
this.setState({ validAddressTypes }); this.setState({ validAddressTypes });
}, };
onManageSettingsClick(e) { onManageSettingsClick = e => {
e.preventDefault(); e.preventDefault();
dis.fire(Action.ViewUserSettings); dis.fire(Action.ViewUserSettings);
this.onCancel(); this.onCancel();
}, };
render: function() { render() {
const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog'); const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog');
const DialogButtons = sdk.getComponent('views.elements.DialogButtons'); const DialogButtons = sdk.getComponent('views.elements.DialogButtons');
const AddressSelector = sdk.getComponent("elements.AddressSelector"); const AddressSelector = sdk.getComponent("elements.AddressSelector");
@ -738,5 +728,5 @@ export default createReactClass({
onCancel={this.onCancel} /> onCancel={this.onCancel} />
</BaseDialog> </BaseDialog>
); );
}, }
}); }

View file

@ -16,37 +16,36 @@ limitations under the License.
import React from 'react'; import React from 'react';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import createReactClass from 'create-react-class';
import * as sdk from '../../../index'; import * as sdk from '../../../index';
import { _t } from '../../../languageHandler'; import { _t } from '../../../languageHandler';
import SettingsStore from "../../../settings/SettingsStore"; import SettingsStore from "../../../settings/SettingsStore";
import {SettingLevel} from "../../../settings/SettingLevel"; import {SettingLevel} from "../../../settings/SettingLevel";
export default createReactClass({ export default class AskInviteAnywayDialog extends React.Component {
propTypes: { static propTypes = {
unknownProfileUsers: PropTypes.array.isRequired, // [ {userId, errorText}... ] unknownProfileUsers: PropTypes.array.isRequired, // [ {userId, errorText}... ]
onInviteAnyways: PropTypes.func.isRequired, onInviteAnyways: PropTypes.func.isRequired,
onGiveUp: PropTypes.func.isRequired, onGiveUp: PropTypes.func.isRequired,
onFinished: PropTypes.func.isRequired, onFinished: PropTypes.func.isRequired,
}, };
_onInviteClicked: function() { _onInviteClicked = () => {
this.props.onInviteAnyways(); this.props.onInviteAnyways();
this.props.onFinished(true); this.props.onFinished(true);
}, };
_onInviteNeverWarnClicked: function() { _onInviteNeverWarnClicked = () => {
SettingsStore.setValue("promptBeforeInviteUnknownUsers", null, SettingLevel.ACCOUNT, false); SettingsStore.setValue("promptBeforeInviteUnknownUsers", null, SettingLevel.ACCOUNT, false);
this.props.onInviteAnyways(); this.props.onInviteAnyways();
this.props.onFinished(true); this.props.onFinished(true);
}, };
_onGiveUpClicked: function() { _onGiveUpClicked = () => {
this.props.onGiveUp(); this.props.onGiveUp();
this.props.onFinished(false); this.props.onFinished(false);
}, };
render: function() { render() {
const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog'); const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog');
const errorList = this.props.unknownProfileUsers const errorList = this.props.unknownProfileUsers
@ -78,5 +77,5 @@ export default createReactClass({
</div> </div>
</BaseDialog> </BaseDialog>
); );
}, }
}); }

View file

@ -17,7 +17,6 @@ limitations under the License.
*/ */
import React from 'react'; import React from 'react';
import createReactClass from 'create-react-class';
import FocusLock from 'react-focus-lock'; import FocusLock from 'react-focus-lock';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import classNames from 'classnames'; import classNames from 'classnames';
@ -28,16 +27,14 @@ import {MatrixClientPeg} from '../../../MatrixClientPeg';
import { _t } from "../../../languageHandler"; import { _t } from "../../../languageHandler";
import MatrixClientContext from "../../../contexts/MatrixClientContext"; import MatrixClientContext from "../../../contexts/MatrixClientContext";
/** /*
* Basic container for modal dialogs. * Basic container for modal dialogs.
* *
* Includes a div for the title, and a keypress handler which cancels the * Includes a div for the title, and a keypress handler which cancels the
* dialog on escape. * dialog on escape.
*/ */
export default createReactClass({ export default class BaseDialog extends React.Component {
displayName: 'BaseDialog', static propTypes = {
propTypes: {
// onFinished callback to call when Escape is pressed // onFinished callback to call when Escape is pressed
// Take a boolean which is true if the dialog was dismissed // Take a boolean which is true if the dialog was dismissed
// with a positive / confirm action or false if it was // with a positive / confirm action or false if it was
@ -81,21 +78,20 @@ export default createReactClass({
PropTypes.object, PropTypes.object,
PropTypes.arrayOf(PropTypes.string), PropTypes.arrayOf(PropTypes.string),
]), ]),
}, };
getDefaultProps: function() { static defaultProps = {
return {
hasCancel: true, hasCancel: true,
fixedWidth: true, fixedWidth: true,
}; };
},
// TODO: [REACT-WARNING] Move this to constructor constructor(props) {
UNSAFE_componentWillMount() { super(props);
this._matrixClient = MatrixClientPeg.get(); this._matrixClient = MatrixClientPeg.get();
}, }
_onKeyDown: function(e) { _onKeyDown = (e) => {
if (this.props.onKeyDown) { if (this.props.onKeyDown) {
this.props.onKeyDown(e); this.props.onKeyDown(e);
} }
@ -104,13 +100,13 @@ export default createReactClass({
e.preventDefault(); e.preventDefault();
this.props.onFinished(false); this.props.onFinished(false);
} }
}, };
_onCancelClick: function(e) { _onCancelClick = (e) => {
this.props.onFinished(false); this.props.onFinished(false);
}, };
render: function() { render() {
let cancelButton; let cancelButton;
if (this.props.hasCancel) { if (this.props.hasCancel) {
cancelButton = ( cancelButton = (
@ -161,5 +157,5 @@ export default createReactClass({
</FocusLock> </FocusLock>
</MatrixClientContext.Provider> </MatrixClientContext.Provider>
); );
}, }
}); }

View file

@ -21,9 +21,6 @@ import { IDialogProps } from "./IDialogProps";
import Field from "../elements/Field"; import Field from "../elements/Field";
import AccessibleButton from "../elements/AccessibleButton"; import AccessibleButton from "../elements/AccessibleButton";
import { MatrixClientPeg } from "../../../MatrixClientPeg"; import { MatrixClientPeg } from "../../../MatrixClientPeg";
import InfoTooltip from "../elements/InfoTooltip";
import dis from "../../../dispatcher/dispatcher";
import {showCommunityRoomInviteDialog} from "../../../RoomInvite";
import { arrayFastClone } from "../../../utils/arrays"; import { arrayFastClone } from "../../../utils/arrays";
import SdkConfig from "../../../SdkConfig"; import SdkConfig from "../../../SdkConfig";
import { RoomMember } from "matrix-js-sdk/src/models/room-member"; import { RoomMember } from "matrix-js-sdk/src/models/room-member";
@ -31,7 +28,6 @@ import InviteDialog from "./InviteDialog";
import BaseAvatar from "../avatars/BaseAvatar"; import BaseAvatar from "../avatars/BaseAvatar";
import {getHttpUriForMxc} from "matrix-js-sdk/src/content-repo"; import {getHttpUriForMxc} from "matrix-js-sdk/src/content-repo";
import {inviteMultipleToRoom, showAnyInviteErrors} from "../../../RoomInvite"; import {inviteMultipleToRoom, showAnyInviteErrors} from "../../../RoomInvite";
import {humanizeTime} from "../../../utils/humanize";
import StyledCheckbox from "../elements/StyledCheckbox"; import StyledCheckbox from "../elements/StyledCheckbox";
import Modal from "../../../Modal"; import Modal from "../../../Modal";
import ErrorDialog from "./ErrorDialog"; import ErrorDialog from "./ErrorDialog";
@ -171,7 +167,7 @@ export default class CommunityPrototypeInviteDialog extends React.PureComponent<
public render() { public render() {
const emailAddresses = []; const emailAddresses = [];
this.state.emailTargets.forEach((address, i) => { this.state.emailTargets.forEach((address, i) => {
emailAddresses.push( emailAddresses.push((
<Field <Field
key={i} key={i}
value={address} value={address}
@ -180,11 +176,11 @@ export default class CommunityPrototypeInviteDialog extends React.PureComponent<
placeholder={_t("Email address")} placeholder={_t("Email address")}
onBlur={() => this.onAddressBlur(i)} onBlur={() => this.onAddressBlur(i)}
/> />
); ));
}); });
// Push a clean input // Push a clean input
emailAddresses.push( emailAddresses.push((
<Field <Field
key={emailAddresses.length} key={emailAddresses.length}
value={""} value={""}
@ -192,23 +188,23 @@ export default class CommunityPrototypeInviteDialog extends React.PureComponent<
label={emailAddresses.length > 0 ? _t("Add another email") : _t("Email address")} label={emailAddresses.length > 0 ? _t("Add another email") : _t("Email address")}
placeholder={emailAddresses.length > 0 ? _t("Add another email") : _t("Email address")} placeholder={emailAddresses.length > 0 ? _t("Add another email") : _t("Email address")}
/> />
); ));
let peopleIntro = null; let peopleIntro = null;
let people = []; const people = [];
if (this.state.showPeople) { if (this.state.showPeople) {
const humansToPresent = this.state.people.slice(0, this.state.numPeople); const humansToPresent = this.state.people.slice(0, this.state.numPeople);
humansToPresent.forEach((person, i) => { humansToPresent.forEach((person, i) => {
people.push(this.renderPerson(person, i)); people.push(this.renderPerson(person, i));
}); });
if (humansToPresent.length < this.state.people.length) { if (humansToPresent.length < this.state.people.length) {
people.push( people.push((
<AccessibleButton <AccessibleButton
onClick={this.onShowMorePeople} onClick={this.onShowMorePeople}
kind="link" key="more" kind="link" key="more"
className="mx_CommunityPrototypeInviteDialog_morePeople" className="mx_CommunityPrototypeInviteDialog_morePeople"
>{_t("Show more")}</AccessibleButton> >{_t("Show more")}</AccessibleButton>
); ));
} }
} }
if (this.state.people.length > 0) { if (this.state.people.length > 0) {

View file

@ -15,17 +15,14 @@ limitations under the License.
*/ */
import React from 'react'; import React from 'react';
import createReactClass from 'create-react-class';
import * as sdk from '../../../index'; import * as sdk from '../../../index';
import { _t } from '../../../languageHandler'; import { _t } from '../../../languageHandler';
/* /*
* A dialog for confirming a redaction. * A dialog for confirming a redaction.
*/ */
export default createReactClass({ export default class ConfirmRedactDialog extends React.Component {
displayName: 'ConfirmRedactDialog', render() {
render: function() {
const QuestionDialog = sdk.getComponent('views.dialogs.QuestionDialog'); const QuestionDialog = sdk.getComponent('views.dialogs.QuestionDialog');
return ( return (
<QuestionDialog onFinished={this.props.onFinished} <QuestionDialog onFinished={this.props.onFinished}
@ -36,5 +33,5 @@ export default createReactClass({
button={_t("Remove")}> button={_t("Remove")}>
</QuestionDialog> </QuestionDialog>
); );
}, }
}); }

View file

@ -15,7 +15,6 @@ limitations under the License.
*/ */
import React from 'react'; import React from 'react';
import createReactClass from 'create-react-class';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import { MatrixClient } from 'matrix-js-sdk'; import { MatrixClient } from 'matrix-js-sdk';
import * as sdk from '../../../index'; import * as sdk from '../../../index';
@ -30,9 +29,8 @@ import { GroupMemberType } from '../../../groups';
* to make it obvious what is going to happen. * to make it obvious what is going to happen.
* Also tweaks the style for 'dangerous' actions (albeit only with colour) * Also tweaks the style for 'dangerous' actions (albeit only with colour)
*/ */
export default createReactClass({ export default class ConfirmUserActionDialog extends React.Component {
displayName: 'ConfirmUserActionDialog', static propTypes = {
propTypes: {
// matrix-js-sdk (room) member object. Supply either this or 'groupMember' // matrix-js-sdk (room) member object. Supply either this or 'groupMember'
member: PropTypes.object, member: PropTypes.object,
// group member object. Supply either this or 'member' // group member object. Supply either this or 'member'
@ -48,35 +46,36 @@ export default createReactClass({
askReason: PropTypes.bool, askReason: PropTypes.bool,
danger: PropTypes.bool, danger: PropTypes.bool,
onFinished: PropTypes.func.isRequired, onFinished: PropTypes.func.isRequired,
}, };
getDefaultProps: () => ({ static defaultProps = {
danger: false, danger: false,
askReason: false, askReason: false,
}), };
constructor(props) {
super(props);
// TODO: [REACT-WARNING] Move this to constructor
UNSAFE_componentWillMount: function() {
this._reasonField = null; this._reasonField = null;
}, }
onOk: function() { onOk = () => {
let reason; let reason;
if (this._reasonField) { if (this._reasonField) {
reason = this._reasonField.value; reason = this._reasonField.value;
} }
this.props.onFinished(true, reason); this.props.onFinished(true, reason);
}, };
onCancel: function() { onCancel = () => {
this.props.onFinished(false); this.props.onFinished(false);
}, };
_collectReasonField: function(e) { _collectReasonField = e => {
this._reasonField = e; this._reasonField = e;
}, };
render: function() { render() {
const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog'); const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog');
const DialogButtons = sdk.getComponent('views.elements.DialogButtons'); const DialogButtons = sdk.getComponent('views.elements.DialogButtons');
const MemberAvatar = sdk.getComponent("views.avatars.MemberAvatar"); const MemberAvatar = sdk.getComponent("views.avatars.MemberAvatar");
@ -134,5 +133,5 @@ export default createReactClass({
onCancel={this.onCancel} /> onCancel={this.onCancel} />
</BaseDialog> </BaseDialog>
); );
}, }
}); }

View file

@ -163,8 +163,9 @@ export default class CreateCommunityPrototypeDialog extends React.PureComponent<
</span> </span>
); );
if (this.state.error) { if (this.state.error) {
const classes = "mx_CreateCommunityPrototypeDialog_subtext mx_CreateCommunityPrototypeDialog_subtext_error";
helpText = ( helpText = (
<span className="mx_CreateCommunityPrototypeDialog_subtext mx_CreateCommunityPrototypeDialog_subtext_error"> <span className={classes}>
{this.state.error} {this.state.error}
</span> </span>
); );
@ -205,7 +206,10 @@ export default class CreateCommunityPrototypeDialog extends React.PureComponent<
ref={this.avatarUploadRef} accept="image/*" ref={this.avatarUploadRef} accept="image/*"
onChange={this.onAvatarChanged} onChange={this.onAvatarChanged}
/> />
<AccessibleButton onClick={this.onChangeAvatar} className="mx_CreateCommunityPrototypeDialog_avatarContainer"> <AccessibleButton
onClick={this.onChangeAvatar}
className="mx_CreateCommunityPrototypeDialog_avatarContainer"
>
{preview} {preview}
</AccessibleButton> </AccessibleButton>
<div className="mx_CreateCommunityPrototypeDialog_tip"> <div className="mx_CreateCommunityPrototypeDialog_tip">

View file

@ -15,46 +15,42 @@ limitations under the License.
*/ */
import React from 'react'; import React from 'react';
import createReactClass from 'create-react-class';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import * as sdk from '../../../index'; import * as sdk from '../../../index';
import dis from '../../../dispatcher/dispatcher'; import dis from '../../../dispatcher/dispatcher';
import { _t } from '../../../languageHandler'; import { _t } from '../../../languageHandler';
import {MatrixClientPeg} from '../../../MatrixClientPeg'; import {MatrixClientPeg} from '../../../MatrixClientPeg';
export default createReactClass({ export default class CreateGroupDialog extends React.Component {
displayName: 'CreateGroupDialog', static propTypes = {
propTypes: {
onFinished: PropTypes.func.isRequired, onFinished: PropTypes.func.isRequired,
}, };
getInitialState: function() { state = {
return {
groupName: '', groupName: '',
groupId: '', groupId: '',
groupError: null, groupError: null,
creating: false, creating: false,
createError: null, createError: null,
}; };
},
_onGroupNameChange: function(e) { _onGroupNameChange = e => {
this.setState({ this.setState({
groupName: e.target.value, groupName: e.target.value,
}); });
}, };
_onGroupIdChange: function(e) { _onGroupIdChange = e => {
this.setState({ this.setState({
groupId: e.target.value, groupId: e.target.value,
}); });
}, };
_onGroupIdBlur: function(e) { _onGroupIdBlur = e => {
this._checkGroupId(); this._checkGroupId();
}, };
_checkGroupId: function(e) { _checkGroupId(e) {
let error = null; let error = null;
if (!this.state.groupId) { if (!this.state.groupId) {
error = _t("Community IDs cannot be empty."); error = _t("Community IDs cannot be empty.");
@ -67,9 +63,9 @@ export default createReactClass({
createError: null, createError: null,
}); });
return error; return error;
}, }
_onFormSubmit: function(e) { _onFormSubmit = e => {
e.preventDefault(); e.preventDefault();
if (this._checkGroupId()) return; if (this._checkGroupId()) return;
@ -94,13 +90,13 @@ export default createReactClass({
}).finally(() => { }).finally(() => {
this.setState({creating: false}); this.setState({creating: false});
}); });
}, };
_onCancel: function() { _onCancel = () => {
this.props.onFinished(false); this.props.onFinished(false);
}, };
render: function() { render() {
const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog'); const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog');
const Spinner = sdk.getComponent('elements.Spinner'); const Spinner = sdk.getComponent('elements.Spinner');
@ -171,5 +167,5 @@ export default createReactClass({
</form> </form>
</BaseDialog> </BaseDialog>
); );
}, }
}); }

View file

@ -16,7 +16,6 @@ limitations under the License.
*/ */
import React from 'react'; import React from 'react';
import createReactClass from 'create-react-class';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import * as sdk from '../../../index'; import * as sdk from '../../../index';
import SdkConfig from '../../../SdkConfig'; import SdkConfig from '../../../SdkConfig';
@ -25,19 +24,19 @@ import { _t } from '../../../languageHandler';
import {MatrixClientPeg} from '../../../MatrixClientPeg'; import {MatrixClientPeg} from '../../../MatrixClientPeg';
import {Key} from "../../../Keyboard"; import {Key} from "../../../Keyboard";
import {privateShouldBeEncrypted} from "../../../createRoom"; import {privateShouldBeEncrypted} from "../../../createRoom";
import TagOrderStore from "../../../stores/TagOrderStore"; import {CommunityPrototypeStore} from "../../../stores/CommunityPrototypeStore";
import GroupStore from "../../../stores/GroupStore";
export default createReactClass({ export default class CreateRoomDialog extends React.Component {
displayName: 'CreateRoomDialog', static propTypes = {
propTypes: {
onFinished: PropTypes.func.isRequired, onFinished: PropTypes.func.isRequired,
defaultPublic: PropTypes.bool, defaultPublic: PropTypes.bool,
}, };
constructor(props) {
super(props);
getInitialState() {
const config = SdkConfig.get(); const config = SdkConfig.get();
return { this.state = {
isPublic: this.props.defaultPublic || false, isPublic: this.props.defaultPublic || false,
isEncrypted: privateShouldBeEncrypted(), isEncrypted: privateShouldBeEncrypted(),
name: "", name: "",
@ -47,7 +46,7 @@ export default createReactClass({
noFederate: config.default_federate === false, noFederate: config.default_federate === false,
nameIsValid: false, nameIsValid: false,
}; };
}, }
_roomCreateOptions() { _roomCreateOptions() {
const opts = {}; const opts = {};
@ -72,32 +71,32 @@ export default createReactClass({
opts.encryption = this.state.isEncrypted; opts.encryption = this.state.isEncrypted;
} }
if (TagOrderStore.getSelectedPrototypeTag()) { if (CommunityPrototypeStore.instance.getSelectedCommunityId()) {
opts.associatedWithCommunity = TagOrderStore.getSelectedPrototypeTag(); opts.associatedWithCommunity = CommunityPrototypeStore.instance.getSelectedCommunityId();
} }
return opts; return opts;
}, }
componentDidMount() { componentDidMount() {
this._detailsRef.addEventListener("toggle", this.onDetailsToggled); this._detailsRef.addEventListener("toggle", this.onDetailsToggled);
// move focus to first field when showing dialog // move focus to first field when showing dialog
this._nameFieldRef.focus(); this._nameFieldRef.focus();
}, }
componentWillUnmount() { componentWillUnmount() {
this._detailsRef.removeEventListener("toggle", this.onDetailsToggled); this._detailsRef.removeEventListener("toggle", this.onDetailsToggled);
}, }
_onKeyDown: function(event) { _onKeyDown = event => {
if (event.key === Key.ENTER) { if (event.key === Key.ENTER) {
this.onOk(); this.onOk();
event.preventDefault(); event.preventDefault();
event.stopPropagation(); event.stopPropagation();
} }
}, };
onOk: async function() { onOk = async () => {
const activeElement = document.activeElement; const activeElement = document.activeElement;
if (activeElement) { if (activeElement) {
activeElement.blur(); activeElement.blur();
@ -123,51 +122,51 @@ export default createReactClass({
field.validate({ allowEmpty: false, focused: true }); field.validate({ allowEmpty: false, focused: true });
} }
} }
}, };
onCancel: function() { onCancel = () => {
this.props.onFinished(false); this.props.onFinished(false);
}, };
onNameChange(ev) { onNameChange = ev => {
this.setState({name: ev.target.value}); this.setState({name: ev.target.value});
}, };
onTopicChange(ev) { onTopicChange = ev => {
this.setState({topic: ev.target.value}); this.setState({topic: ev.target.value});
}, };
onPublicChange(isPublic) { onPublicChange = isPublic => {
this.setState({isPublic}); this.setState({isPublic});
}, };
onEncryptedChange(isEncrypted) { onEncryptedChange = isEncrypted => {
this.setState({isEncrypted}); this.setState({isEncrypted});
}, };
onAliasChange(alias) { onAliasChange = alias => {
this.setState({alias}); this.setState({alias});
}, };
onDetailsToggled(ev) { onDetailsToggled = ev => {
this.setState({detailsOpen: ev.target.open}); this.setState({detailsOpen: ev.target.open});
}, };
onNoFederateChange(noFederate) { onNoFederateChange = noFederate => {
this.setState({noFederate}); this.setState({noFederate});
}, };
collectDetailsRef(ref) { collectDetailsRef = ref => {
this._detailsRef = ref; this._detailsRef = ref;
}, };
async onNameValidate(fieldState) { onNameValidate = async fieldState => {
const result = await this._validateRoomName(fieldState); const result = await CreateRoomDialog._validateRoomName(fieldState);
this.setState({nameIsValid: result.valid}); this.setState({nameIsValid: result.valid});
return result; return result;
}, };
_validateRoomName: withValidation({ static _validateRoomName = withValidation({
rules: [ rules: [
{ {
key: "required", key: "required",
@ -175,9 +174,9 @@ export default createReactClass({
invalid: () => _t("Please enter a name for the room"), invalid: () => _t("Please enter a name for the room"),
}, },
], ],
}), });
render: function() { render() {
const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog'); const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog');
const DialogButtons = sdk.getComponent('views.elements.DialogButtons'); const DialogButtons = sdk.getComponent('views.elements.DialogButtons');
const Field = sdk.getComponent('views.elements.Field'); const Field = sdk.getComponent('views.elements.Field');
@ -198,7 +197,7 @@ export default createReactClass({
"Private rooms can be found and joined by invitation only. Public rooms can be " + "Private rooms can be found and joined by invitation only. Public rooms can be " +
"found and joined by anyone.", "found and joined by anyone.",
)}</p>; )}</p>;
if (TagOrderStore.getSelectedPrototypeTag()) { if (CommunityPrototypeStore.instance.getSelectedCommunityId()) {
publicPrivateLabel = <p>{_t( publicPrivateLabel = <p>{_t(
"Private rooms can be found and joined by invitation only. Public rooms can be " + "Private rooms can be found and joined by invitation only. Public rooms can be " +
"found and joined by anyone in this community.", "found and joined by anyone in this community.",
@ -239,9 +238,8 @@ export default createReactClass({
} }
let title = this.state.isPublic ? _t('Create a public room') : _t('Create a private room'); let title = this.state.isPublic ? _t('Create a public room') : _t('Create a private room');
if (TagOrderStore.getSelectedPrototypeTag()) { if (CommunityPrototypeStore.instance.getSelectedCommunityId()) {
const summary = GroupStore.getSummary(TagOrderStore.getSelectedPrototypeTag()); const name = CommunityPrototypeStore.instance.getSelectedCommunityName();
const name = summary?.profile?.name || TagOrderStore.getSelectedPrototypeTag();
title = _t("Create a room in %(communityName)s", {communityName: name}); title = _t("Create a room in %(communityName)s", {communityName: name});
} }
return ( return (
@ -275,5 +273,5 @@ export default createReactClass({
onCancel={this.onCancel} /> onCancel={this.onCancel} />
</BaseDialog> </BaseDialog>
); );
}, }
}); }

View file

@ -0,0 +1,167 @@
/*
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 React, { ChangeEvent } from 'react';
import BaseDialog from "./BaseDialog";
import { _t } from "../../../languageHandler";
import { IDialogProps } from "./IDialogProps";
import Field from "../elements/Field";
import AccessibleButton from "../elements/AccessibleButton";
import { MatrixClientPeg } from "../../../MatrixClientPeg";
import { CommunityPrototypeStore } from "../../../stores/CommunityPrototypeStore";
import FlairStore from "../../../stores/FlairStore";
interface IProps extends IDialogProps {
communityId: string;
}
interface IState {
name: string;
error: string;
busy: boolean;
currentAvatarUrl: string;
avatarFile: File;
avatarPreview: string;
}
// XXX: This is a lot of duplication from the create dialog, just in a different shape
export default class EditCommunityPrototypeDialog extends React.PureComponent<IProps, IState> {
private avatarUploadRef: React.RefObject<HTMLInputElement> = React.createRef();
constructor(props: IProps) {
super(props);
const profile = CommunityPrototypeStore.instance.getCommunityProfile(props.communityId);
this.state = {
name: profile?.name || "",
error: null,
busy: false,
avatarFile: null,
avatarPreview: null,
currentAvatarUrl: profile?.avatarUrl,
};
}
private onNameChange = (ev: ChangeEvent<HTMLInputElement>) => {
this.setState({name: ev.target.value});
};
private onSubmit = async (ev) => {
ev.preventDefault();
ev.stopPropagation();
if (this.state.busy) return;
// We'll create the community now to see if it's taken, leaving it active in
// the background for the user to look at while they invite people.
this.setState({busy: true});
try {
let avatarUrl = this.state.currentAvatarUrl || ""; // must be a string for synapse to accept it
if (this.state.avatarFile) {
avatarUrl = await MatrixClientPeg.get().uploadContent(this.state.avatarFile);
}
await MatrixClientPeg.get().setGroupProfile(this.props.communityId, {
name: this.state.name,
avatar_url: avatarUrl,
});
// ask the flair store to update the profile too
await FlairStore.refreshGroupProfile(MatrixClientPeg.get(), this.props.communityId);
// we did it, so close the dialog
this.props.onFinished(true);
} catch (e) {
console.error(e);
this.setState({
busy: false,
error: _t("There was an error updating your community. The server is unable to process your request."),
});
}
};
private onAvatarChanged = (e: ChangeEvent<HTMLInputElement>) => {
if (!e.target.files || !e.target.files.length) {
this.setState({avatarFile: null});
} else {
this.setState({busy: true});
const file = e.target.files[0];
const reader = new FileReader();
reader.onload = (ev: ProgressEvent<FileReader>) => {
this.setState({avatarFile: file, busy: false, avatarPreview: ev.target.result as string});
};
reader.readAsDataURL(file);
}
};
private onChangeAvatar = () => {
if (this.avatarUploadRef.current) this.avatarUploadRef.current.click();
};
public render() {
let preview = <img src={this.state.avatarPreview} className="mx_EditCommunityPrototypeDialog_avatar" />;
if (!this.state.avatarPreview) {
if (this.state.currentAvatarUrl) {
const url = MatrixClientPeg.get().mxcUrlToHttp(this.state.currentAvatarUrl);
preview = <img src={url} className="mx_EditCommunityPrototypeDialog_avatar" />;
} else {
preview = <div className="mx_EditCommunityPrototypeDialog_placeholderAvatar" />
}
}
return (
<BaseDialog
className="mx_EditCommunityPrototypeDialog"
onFinished={this.props.onFinished}
title={_t("Update community")}
>
<form onSubmit={this.onSubmit}>
<div className="mx_Dialog_content">
<div className="mx_EditCommunityPrototypeDialog_rowName">
<Field
value={this.state.name}
onChange={this.onNameChange}
placeholder={_t("Enter name")}
label={_t("Enter name")}
/>
</div>
<div className="mx_EditCommunityPrototypeDialog_rowAvatar">
<input
type="file" style={{display: "none"}}
ref={this.avatarUploadRef} accept="image/*"
onChange={this.onAvatarChanged}
/>
<AccessibleButton
onClick={this.onChangeAvatar}
className="mx_EditCommunityPrototypeDialog_avatarContainer"
>{preview}</AccessibleButton>
<div className="mx_EditCommunityPrototypeDialog_tip">
<b>{_t("Add image (optional)")}</b>
<span>
{_t("An image will help people identify your community.")}
</span>
</div>
</div>
<AccessibleButton kind="primary" onClick={this.onSubmit} disabled={this.state.busy}>
{_t("Save")}
</AccessibleButton>
</div>
</form>
</BaseDialog>
);
}
}

View file

@ -26,14 +26,12 @@ limitations under the License.
*/ */
import React from 'react'; import React from 'react';
import createReactClass from 'create-react-class';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import * as sdk from '../../../index'; import * as sdk from '../../../index';
import { _t } from '../../../languageHandler'; import { _t } from '../../../languageHandler';
export default createReactClass({ export default class ErrorDialog extends React.Component {
displayName: 'ErrorDialog', static propTypes = {
propTypes: {
title: PropTypes.string, title: PropTypes.string,
description: PropTypes.oneOfType([ description: PropTypes.oneOfType([
PropTypes.element, PropTypes.element,
@ -43,18 +41,16 @@ export default createReactClass({
focus: PropTypes.bool, focus: PropTypes.bool,
onFinished: PropTypes.func.isRequired, onFinished: PropTypes.func.isRequired,
headerImage: PropTypes.string, headerImage: PropTypes.string,
}, };
getDefaultProps: function() { static defaultProps = {
return {
focus: true, focus: true,
title: null, title: null,
description: null, description: null,
button: null, button: null,
}; };
},
render: function() { render() {
const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog'); const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog');
return ( return (
<BaseDialog <BaseDialog
@ -74,5 +70,5 @@ export default createReactClass({
</div> </div>
</BaseDialog> </BaseDialog>
); );
}, }
}); }

View file

@ -17,15 +17,13 @@ limitations under the License.
*/ */
import React from 'react'; import React from 'react';
import createReactClass from 'create-react-class';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import * as sdk from '../../../index'; import * as sdk from '../../../index';
import { _t } from '../../../languageHandler'; import { _t } from '../../../languageHandler';
import classNames from "classnames"; import classNames from "classnames";
export default createReactClass({ export default class InfoDialog extends React.Component {
displayName: 'InfoDialog', static propTypes = {
propTypes: {
className: PropTypes.string, className: PropTypes.string,
title: PropTypes.string, title: PropTypes.string,
description: PropTypes.node, description: PropTypes.node,
@ -33,21 +31,19 @@ export default createReactClass({
onFinished: PropTypes.func, onFinished: PropTypes.func,
hasCloseButton: PropTypes.bool, hasCloseButton: PropTypes.bool,
onKeyDown: PropTypes.func, onKeyDown: PropTypes.func,
}, };
getDefaultProps: function() { static defaultProps = {
return {
title: '', title: '',
description: '', description: '',
hasCloseButton: false, hasCloseButton: false,
}; };
},
onFinished: function() { onFinished = () => {
this.props.onFinished(); this.props.onFinished();
}, };
render: function() { render() {
const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog'); const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog');
const DialogButtons = sdk.getComponent('views.elements.DialogButtons'); const DialogButtons = sdk.getComponent('views.elements.DialogButtons');
return ( return (
@ -69,5 +65,5 @@ export default createReactClass({
</DialogButtons> </DialogButtons>
</BaseDialog> </BaseDialog>
); );
}, }
}); }

View file

@ -17,7 +17,6 @@ limitations under the License.
*/ */
import React from 'react'; import React from 'react';
import createReactClass from 'create-react-class';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import * as sdk from '../../../index'; import * as sdk from '../../../index';
@ -27,10 +26,8 @@ import AccessibleButton from '../elements/AccessibleButton';
import {ERROR_USER_CANCELLED} from "../../structures/InteractiveAuth"; import {ERROR_USER_CANCELLED} from "../../structures/InteractiveAuth";
import {SSOAuthEntry} from "../auth/InteractiveAuthEntryComponents"; import {SSOAuthEntry} from "../auth/InteractiveAuthEntryComponents";
export default createReactClass({ export default class InteractiveAuthDialog extends React.Component {
displayName: 'InteractiveAuthDialog', static propTypes = {
propTypes: {
// matrix client to use for UI auth requests // matrix client to use for UI auth requests
matrixClient: PropTypes.object.isRequired, matrixClient: PropTypes.object.isRequired,
@ -70,19 +67,17 @@ export default createReactClass({
// //
// Default is defined in _getDefaultDialogAesthetics() // Default is defined in _getDefaultDialogAesthetics()
aestheticsForStagePhases: PropTypes.object, aestheticsForStagePhases: PropTypes.object,
}, };
getInitialState: function() { state = {
return {
authError: null, authError: null,
// See _onUpdateStagePhase() // See _onUpdateStagePhase()
uiaStage: null, uiaStage: null,
uiaStagePhase: null, uiaStagePhase: null,
}; };
},
_getDefaultDialogAesthetics: function() { _getDefaultDialogAesthetics() {
const ssoAesthetics = { const ssoAesthetics = {
[SSOAuthEntry.PHASE_PREAUTH]: { [SSOAuthEntry.PHASE_PREAUTH]: {
title: _t("Use Single Sign On to continue"), title: _t("Use Single Sign On to continue"),
@ -102,9 +97,9 @@ export default createReactClass({
[SSOAuthEntry.LOGIN_TYPE]: ssoAesthetics, [SSOAuthEntry.LOGIN_TYPE]: ssoAesthetics,
[SSOAuthEntry.UNSTABLE_LOGIN_TYPE]: ssoAesthetics, [SSOAuthEntry.UNSTABLE_LOGIN_TYPE]: ssoAesthetics,
}; };
}, }
_onAuthFinished: function(success, result) { _onAuthFinished = (success, result) => {
if (success) { if (success) {
this.props.onFinished(true, result); this.props.onFinished(true, result);
} else { } else {
@ -116,18 +111,18 @@ export default createReactClass({
}); });
} }
} }
}, };
_onUpdateStagePhase: function(newStage, newPhase) { _onUpdateStagePhase = (newStage, newPhase) => {
// We copy the stage and stage phase params into state for title selection in render() // We copy the stage and stage phase params into state for title selection in render()
this.setState({uiaStage: newStage, uiaStagePhase: newPhase}); this.setState({uiaStage: newStage, uiaStagePhase: newPhase});
}, };
_onDismissClick: function() { _onDismissClick = () => {
this.props.onFinished(false); this.props.onFinished(false);
}, };
render: function() { render() {
const InteractiveAuth = sdk.getComponent("structures.InteractiveAuth"); const InteractiveAuth = sdk.getComponent("structures.InteractiveAuth");
const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog'); const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog');
@ -190,5 +185,5 @@ export default createReactClass({
{ content } { content }
</BaseDialog> </BaseDialog>
); );
}, }
}); }

View file

@ -32,11 +32,12 @@ import IdentityAuthClient from "../../../IdentityAuthClient";
import Modal from "../../../Modal"; import Modal from "../../../Modal";
import {humanizeTime} from "../../../utils/humanize"; import {humanizeTime} from "../../../utils/humanize";
import createRoom, {canEncryptToAllUsers, privateShouldBeEncrypted} from "../../../createRoom"; import createRoom, {canEncryptToAllUsers, privateShouldBeEncrypted} from "../../../createRoom";
import {inviteMultipleToRoom} from "../../../RoomInvite"; import {inviteMultipleToRoom, showCommunityInviteDialog} from "../../../RoomInvite";
import {Key} from "../../../Keyboard"; import {Key} from "../../../Keyboard";
import {Action} from "../../../dispatcher/actions"; import {Action} from "../../../dispatcher/actions";
import {DefaultTagID} from "../../../stores/room-list/models"; import {DefaultTagID} from "../../../stores/room-list/models";
import RoomListStore from "../../../stores/room-list/RoomListStore"; import RoomListStore from "../../../stores/room-list/RoomListStore";
import {CommunityPrototypeStore} from "../../../stores/CommunityPrototypeStore";
// we have a number of types defined from the Matrix spec which can't reasonably be altered here. // we have a number of types defined from the Matrix spec which can't reasonably be altered here.
/* eslint-disable camelcase */ /* eslint-disable camelcase */
@ -909,12 +910,23 @@ export default class InviteDialog extends React.PureComponent {
this.props.onFinished(); this.props.onFinished();
}; };
_onCommunityInviteClick = (e) => {
this.props.onFinished();
showCommunityInviteDialog(CommunityPrototypeStore.instance.getSelectedCommunityId());
};
_renderSection(kind: "recents"|"suggestions") { _renderSection(kind: "recents"|"suggestions") {
let sourceMembers = kind === 'recents' ? this.state.recents : this.state.suggestions; let sourceMembers = kind === 'recents' ? this.state.recents : this.state.suggestions;
let showNum = kind === 'recents' ? this.state.numRecentsShown : this.state.numSuggestionsShown; let showNum = kind === 'recents' ? this.state.numRecentsShown : this.state.numSuggestionsShown;
const showMoreFn = kind === 'recents' ? this._showMoreRecents.bind(this) : this._showMoreSuggestions.bind(this); const showMoreFn = kind === 'recents' ? this._showMoreRecents.bind(this) : this._showMoreSuggestions.bind(this);
const lastActive = (m) => kind === 'recents' ? m.lastActive : null; const lastActive = (m) => kind === 'recents' ? m.lastActive : null;
let sectionName = kind === 'recents' ? _t("Recent Conversations") : _t("Suggestions"); let sectionName = kind === 'recents' ? _t("Recent Conversations") : _t("Suggestions");
let sectionSubname = null;
if (kind === 'suggestions' && CommunityPrototypeStore.instance.getSelectedCommunityId()) {
const communityName = CommunityPrototypeStore.instance.getSelectedCommunityName();
sectionSubname = _t("May include members not in %(communityName)s", {communityName});
}
if (this.props.kind === KIND_INVITE) { if (this.props.kind === KIND_INVITE) {
sectionName = kind === 'recents' ? _t("Recently Direct Messaged") : _t("Suggestions"); sectionName = kind === 'recents' ? _t("Recently Direct Messaged") : _t("Suggestions");
@ -993,6 +1005,7 @@ export default class InviteDialog extends React.PureComponent {
return ( return (
<div className='mx_InviteDialog_section'> <div className='mx_InviteDialog_section'>
<h3>{sectionName}</h3> <h3>{sectionName}</h3>
{sectionSubname ? <p className="mx_InviteDialog_subname">{sectionSubname}</p> : null}
{tiles} {tiles}
{showMore} {showMore}
</div> </div>
@ -1083,6 +1096,33 @@ export default class InviteDialog extends React.PureComponent {
return <a href={makeUserPermalink(userId)} rel="noreferrer noopener" target="_blank">{userId}</a>; return <a href={makeUserPermalink(userId)} rel="noreferrer noopener" target="_blank">{userId}</a>;
}}, }},
); );
if (CommunityPrototypeStore.instance.getSelectedCommunityId()) {
const communityName = CommunityPrototypeStore.instance.getSelectedCommunityName();
helpText = _t(
"Start a conversation with someone using their name, username (like <userId/>) or email address. " +
"This won't invite them to %(communityName)s. To invite someone to %(communityName)s, click " +
"<a>here</a>.",
{communityName}, {
userId: () => {
return (
<a
href={makeUserPermalink(userId)}
rel="noreferrer noopener"
target="_blank"
>{userId}</a>
);
},
a: (sub) => {
return (
<AccessibleButton
kind="link"
onClick={this._onCommunityInviteClick}
>{sub}</AccessibleButton>
);
},
},
);
}
buttonText = _t("Go"); buttonText = _t("Go");
goButtonFn = this._startDm; goButtonFn = this._startDm;
} else { // KIND_INVITE } else { // KIND_INVITE

View file

@ -16,14 +16,12 @@ limitations under the License.
*/ */
import React from 'react'; import React from 'react';
import createReactClass from 'create-react-class';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import * as sdk from '../../../index'; import * as sdk from '../../../index';
import { _t } from '../../../languageHandler'; import { _t } from '../../../languageHandler';
export default createReactClass({ export default class QuestionDialog extends React.Component {
displayName: 'QuestionDialog', static propTypes = {
propTypes: {
title: PropTypes.string, title: PropTypes.string,
description: PropTypes.node, description: PropTypes.node,
extraButtons: PropTypes.node, extraButtons: PropTypes.node,
@ -34,10 +32,9 @@ export default createReactClass({
headerImage: PropTypes.string, headerImage: PropTypes.string,
quitOnly: PropTypes.bool, // quitOnly doesn't show the cancel button just the quit [x]. quitOnly: PropTypes.bool, // quitOnly doesn't show the cancel button just the quit [x].
fixedWidth: PropTypes.bool, fixedWidth: PropTypes.bool,
}, };
getDefaultProps: function() { static defaultProps = {
return {
title: "", title: "",
description: "", description: "",
extraButtons: null, extraButtons: null,
@ -46,17 +43,16 @@ export default createReactClass({
danger: false, danger: false,
quitOnly: false, quitOnly: false,
}; };
},
onOk: function() { onOk = () => {
this.props.onFinished(true); this.props.onFinished(true);
}, };
onCancel: function() { onCancel = () => {
this.props.onFinished(false); this.props.onFinished(false);
}, };
render: function() { render() {
const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog'); const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog');
const DialogButtons = sdk.getComponent('views.elements.DialogButtons'); const DialogButtons = sdk.getComponent('views.elements.DialogButtons');
let primaryButtonClass = ""; let primaryButtonClass = "";
@ -88,5 +84,5 @@ export default createReactClass({
</DialogButtons> </DialogButtons>
</BaseDialog> </BaseDialog>
); );
}, }
}); }

View file

@ -15,38 +15,33 @@ limitations under the License.
*/ */
import React from 'react'; import React from 'react';
import createReactClass from 'create-react-class';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import * as sdk from '../../../index'; import * as sdk from '../../../index';
import {MatrixClientPeg} from '../../../MatrixClientPeg'; import {MatrixClientPeg} from '../../../MatrixClientPeg';
import Modal from '../../../Modal'; import Modal from '../../../Modal';
import { _t } from '../../../languageHandler'; import { _t } from '../../../languageHandler';
export default createReactClass({ export default class RoomUpgradeDialog extends React.Component {
displayName: 'RoomUpgradeDialog', static propTypes = {
propTypes: {
room: PropTypes.object.isRequired, room: PropTypes.object.isRequired,
onFinished: PropTypes.func.isRequired, onFinished: PropTypes.func.isRequired,
}, };
componentDidMount: async function() { state = {
busy: true,
};
async componentDidMount() {
const recommended = await this.props.room.getRecommendedVersion(); const recommended = await this.props.room.getRecommendedVersion();
this._targetVersion = recommended.version; this._targetVersion = recommended.version;
this.setState({busy: false}); this.setState({busy: false});
}, }
getInitialState: function() { _onCancelClick = () => {
return {
busy: true,
};
},
_onCancelClick: function() {
this.props.onFinished(false); this.props.onFinished(false);
}, };
_onUpgradeClick: function() { _onUpgradeClick = () => {
this.setState({busy: true}); this.setState({busy: true});
MatrixClientPeg.get().upgradeRoom(this.props.room.roomId, this._targetVersion).then(() => { MatrixClientPeg.get().upgradeRoom(this.props.room.roomId, this._targetVersion).then(() => {
this.props.onFinished(true); this.props.onFinished(true);
@ -59,9 +54,9 @@ export default createReactClass({
}).finally(() => { }).finally(() => {
this.setState({busy: false}); this.setState({busy: false});
}); });
}, };
render: function() { render() {
const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog'); const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog');
const DialogButtons = sdk.getComponent('views.elements.DialogButtons'); const DialogButtons = sdk.getComponent('views.elements.DialogButtons');
const Spinner = sdk.getComponent('views.elements.Spinner'); const Spinner = sdk.getComponent('views.elements.Spinner');
@ -106,5 +101,5 @@ export default createReactClass({
{buttons} {buttons}
</BaseDialog> </BaseDialog>
); );
}, }
}); }

View file

@ -17,7 +17,6 @@ limitations under the License.
*/ */
import React from 'react'; import React from 'react';
import createReactClass from 'create-react-class';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import * as sdk from '../../../index'; import * as sdk from '../../../index';
import SdkConfig from '../../../SdkConfig'; import SdkConfig from '../../../SdkConfig';
@ -25,20 +24,18 @@ import Modal from '../../../Modal';
import { _t } from '../../../languageHandler'; import { _t } from '../../../languageHandler';
export default createReactClass({ export default class SessionRestoreErrorDialog extends React.Component {
displayName: 'SessionRestoreErrorDialog', static propTypes = {
propTypes: {
error: PropTypes.string.isRequired, error: PropTypes.string.isRequired,
onFinished: PropTypes.func.isRequired, onFinished: PropTypes.func.isRequired,
}, };
_sendBugReport: function() { _sendBugReport = () => {
const BugReportDialog = sdk.getComponent("dialogs.BugReportDialog"); const BugReportDialog = sdk.getComponent("dialogs.BugReportDialog");
Modal.createTrackedDialog('Session Restore Error', 'Send Bug Report Dialog', BugReportDialog, {}); Modal.createTrackedDialog('Session Restore Error', 'Send Bug Report Dialog', BugReportDialog, {});
}, };
_onClearStorageClick: function() { _onClearStorageClick = () => {
const QuestionDialog = sdk.getComponent("dialogs.QuestionDialog"); const QuestionDialog = sdk.getComponent("dialogs.QuestionDialog");
Modal.createTrackedDialog('Session Restore Confirm Logout', '', QuestionDialog, { Modal.createTrackedDialog('Session Restore Confirm Logout', '', QuestionDialog, {
title: _t("Sign out"), title: _t("Sign out"),
@ -48,15 +45,15 @@ export default createReactClass({
danger: true, danger: true,
onFinished: this.props.onFinished, onFinished: this.props.onFinished,
}); });
}, };
_onRefreshClick: function() { _onRefreshClick = () => {
// Is this likely to help? Probably not, but giving only one button // Is this likely to help? Probably not, but giving only one button
// that clears your storage seems awful. // that clears your storage seems awful.
window.location.reload(true); window.location.reload(true);
}, };
render: function() { render() {
const brand = SdkConfig.get().brand; const brand = SdkConfig.get().brand;
const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog'); const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog');
const DialogButtons = sdk.getComponent('views.elements.DialogButtons'); const DialogButtons = sdk.getComponent('views.elements.DialogButtons');
@ -110,5 +107,5 @@ export default createReactClass({
{ dialogButtons } { dialogButtons }
</BaseDialog> </BaseDialog>
); );
}, }
}); }

View file

@ -16,7 +16,6 @@ limitations under the License.
*/ */
import React from 'react'; import React from 'react';
import createReactClass from 'create-react-class';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import * as sdk from '../../../index'; import * as sdk from '../../../index';
import * as Email from '../../../email'; import * as Email from '../../../email';
@ -25,31 +24,28 @@ import { _t } from '../../../languageHandler';
import Modal from '../../../Modal'; import Modal from '../../../Modal';
/** /*
* Prompt the user to set an email address. * Prompt the user to set an email address.
* *
* On success, `onFinished(true)` is called. * On success, `onFinished(true)` is called.
*/ */
export default createReactClass({ export default class SetEmailDialog extends React.Component {
displayName: 'SetEmailDialog', static propTypes = {
propTypes: {
onFinished: PropTypes.func.isRequired, onFinished: PropTypes.func.isRequired,
}, };
getInitialState: function() { state = {
return {
emailAddress: '', emailAddress: '',
emailBusy: false, emailBusy: false,
}; };
},
onEmailAddressChanged: function(value) { onEmailAddressChanged = value => {
this.setState({ this.setState({
emailAddress: value, emailAddress: value,
}); });
}, };
onSubmit: function() { onSubmit = () => {
const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog"); const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
const QuestionDialog = sdk.getComponent("dialogs.QuestionDialog"); const QuestionDialog = sdk.getComponent("dialogs.QuestionDialog");
@ -81,21 +77,21 @@ export default createReactClass({
}); });
}); });
this.setState({emailBusy: true}); this.setState({emailBusy: true});
}, };
onCancelled: function() { onCancelled = () => {
this.props.onFinished(false); this.props.onFinished(false);
}, };
onEmailDialogFinished: function(ok) { onEmailDialogFinished = ok => {
if (ok) { if (ok) {
this.verifyEmailAddress(); this.verifyEmailAddress();
} else { } else {
this.setState({emailBusy: false}); this.setState({emailBusy: false});
} }
}, };
verifyEmailAddress: function() { verifyEmailAddress() {
this._addThreepid.checkEmailLinkClicked().then(() => { this._addThreepid.checkEmailLinkClicked().then(() => {
this.props.onFinished(true); this.props.onFinished(true);
}, (err) => { }, (err) => {
@ -119,9 +115,9 @@ export default createReactClass({
}); });
} }
}); });
}, }
render: function() { render() {
const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog'); const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog');
const Spinner = sdk.getComponent('elements.Spinner'); const Spinner = sdk.getComponent('elements.Spinner');
const EditableText = sdk.getComponent('elements.EditableText'); const EditableText = sdk.getComponent('elements.EditableText');
@ -161,5 +157,5 @@ export default createReactClass({
</div> </div>
</BaseDialog> </BaseDialog>
); );
}, }
}); }

View file

@ -16,7 +16,6 @@ limitations under the License.
*/ */
import React, {createRef} from 'react'; import React, {createRef} from 'react';
import createReactClass from 'create-react-class';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import * as sdk from '../../../index'; import * as sdk from '../../../index';
import {MatrixClientPeg} from '../../../MatrixClientPeg'; import {MatrixClientPeg} from '../../../MatrixClientPeg';
@ -29,23 +28,27 @@ import { SAFE_LOCALPART_REGEX } from '../../../Registration';
// sending a request to the server // sending a request to the server
const USERNAME_CHECK_DEBOUNCE_MS = 250; const USERNAME_CHECK_DEBOUNCE_MS = 250;
/** /*
* Prompt the user to set a display name. * Prompt the user to set a display name.
* *
* On success, `onFinished(true, newDisplayName)` is called. * On success, `onFinished(true, newDisplayName)` is called.
*/ */
export default createReactClass({ export default class SetMxIdDialog extends React.Component {
displayName: 'SetMxIdDialog', static propTypes = {
propTypes: {
onFinished: PropTypes.func.isRequired, onFinished: PropTypes.func.isRequired,
// Called when the user requests to register with a different homeserver // Called when the user requests to register with a different homeserver
onDifferentServerClicked: PropTypes.func.isRequired, onDifferentServerClicked: PropTypes.func.isRequired,
// Called if the user wants to switch to login instead // Called if the user wants to switch to login instead
onLoginClick: PropTypes.func.isRequired, onLoginClick: PropTypes.func.isRequired,
}, };
getInitialState: function() { constructor(props) {
return { super(props);
this._input_value = createRef();
this._uiAuth = createRef();
this.state = {
// The entered username // The entered username
username: '', username: '',
// Indicate ongoing work on the username // Indicate ongoing work on the username
@ -60,21 +63,15 @@ export default createReactClass({
// Indicate error with auth // Indicate error with auth
authError: '', authError: '',
}; };
}, }
// TODO: [REACT-WARNING] Replace component with real class, use constructor for refs componentDidMount() {
UNSAFE_componentWillMount: function() {
this._input_value = createRef();
this._uiAuth = createRef();
},
componentDidMount: function() {
this._input_value.current.select(); this._input_value.current.select();
this._matrixClient = MatrixClientPeg.get(); this._matrixClient = MatrixClientPeg.get();
}, }
onValueChange: function(ev) { onValueChange = ev => {
this.setState({ this.setState({
username: ev.target.value, username: ev.target.value,
usernameBusy: true, usernameBusy: true,
@ -99,24 +96,24 @@ export default createReactClass({
}); });
}, USERNAME_CHECK_DEBOUNCE_MS); }, USERNAME_CHECK_DEBOUNCE_MS);
}); });
}, };
onKeyUp: function(ev) { onKeyUp = ev => {
if (ev.key === Key.ENTER) { if (ev.key === Key.ENTER) {
this.onSubmit(); this.onSubmit();
} }
}, };
onSubmit: function(ev) { onSubmit = ev => {
if (this._uiAuth.current) { if (this._uiAuth.current) {
this._uiAuth.current.tryContinue(); this._uiAuth.current.tryContinue();
} }
this.setState({ this.setState({
doingUIAuth: true, doingUIAuth: true,
}); });
}, };
_doUsernameCheck: function() { _doUsernameCheck() {
// We do a quick check ahead of the username availability API to ensure the // We do a quick check ahead of the username availability API to ensure the
// user ID roughly looks okay from a Matrix perspective. // user ID roughly looks okay from a Matrix perspective.
if (!SAFE_LOCALPART_REGEX.test(this.state.username)) { if (!SAFE_LOCALPART_REGEX.test(this.state.username)) {
@ -167,13 +164,13 @@ export default createReactClass({
this.setState(newState); this.setState(newState);
}, },
); );
}, }
_generatePassword: function() { _generatePassword() {
return Math.random().toString(36).slice(2); return Math.random().toString(36).slice(2);
}, }
_makeRegisterRequest: function(auth) { _makeRegisterRequest = auth => {
// Not upgrading - changing mxids // Not upgrading - changing mxids
const guestAccessToken = null; const guestAccessToken = null;
if (!this._generatedPassword) { if (!this._generatedPassword) {
@ -187,9 +184,9 @@ export default createReactClass({
{}, {},
guestAccessToken, guestAccessToken,
); );
}, };
_onUIAuthFinished: function(success, response) { _onUIAuthFinished = (success, response) => {
this.setState({ this.setState({
doingUIAuth: false, doingUIAuth: false,
}); });
@ -207,9 +204,9 @@ export default createReactClass({
accessToken: response.access_token, accessToken: response.access_token,
password: this._generatedPassword, password: this._generatedPassword,
}); });
}, };
render: function() { render() {
const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog'); const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog');
const InteractiveAuth = sdk.getComponent('structures.InteractiveAuth'); const InteractiveAuth = sdk.getComponent('structures.InteractiveAuth');
@ -303,5 +300,5 @@ export default createReactClass({
</div> </div>
</BaseDialog> </BaseDialog>
); );
}, }
}); }

View file

@ -17,7 +17,6 @@ limitations under the License.
*/ */
import React from 'react'; import React from 'react';
import createReactClass from 'create-react-class';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import * as sdk from '../../../index'; import * as sdk from '../../../index';
import { _t } from '../../../languageHandler'; import { _t } from '../../../languageHandler';
@ -63,32 +62,25 @@ const WarmFuzzy = function(props) {
* *
* On success, `onFinished()` when finished * On success, `onFinished()` when finished
*/ */
export default createReactClass({ export default class SetPasswordDialog extends React.Component {
displayName: 'SetPasswordDialog', static propTypes = {
propTypes: {
onFinished: PropTypes.func.isRequired, onFinished: PropTypes.func.isRequired,
}, };
getInitialState: function() { state = {
return {
error: null, error: null,
}; };
},
componentDidMount: function() { _onPasswordChanged = res => {
console.info('SetPasswordDialog component did mount');
},
_onPasswordChanged: function(res) {
Modal.createDialog(WarmFuzzy, { Modal.createDialog(WarmFuzzy, {
didSetEmail: res.didSetEmail, didSetEmail: res.didSetEmail,
onFinished: () => { onFinished: () => {
this.props.onFinished(); this.props.onFinished();
}, },
}); });
}, };
_onPasswordChangeError: function(err) { _onPasswordChangeError = err => {
let errMsg = err.error || ""; let errMsg = err.error || "";
if (err.httpStatus === 403) { if (err.httpStatus === 403) {
errMsg = _t('Failed to change password. Is your password correct?'); errMsg = _t('Failed to change password. Is your password correct?');
@ -101,9 +93,9 @@ export default createReactClass({
this.setState({ this.setState({
error: errMsg, error: errMsg,
}); });
}, };
render: function() { render() {
const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog'); const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog');
const ChangePassword = sdk.getComponent('views.settings.ChangePassword'); const ChangePassword = sdk.getComponent('views.settings.ChangePassword');
@ -132,5 +124,5 @@ export default createReactClass({
</div> </div>
</BaseDialog> </BaseDialog>
); );
}, }
}); }

View file

@ -198,14 +198,16 @@ export default class ShareDialog extends React.PureComponent<IProps, IState> {
const encodedUrl = encodeURIComponent(matrixToUrl); const encodedUrl = encodeURIComponent(matrixToUrl);
const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog'); const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog');
return <BaseDialog title={title} return <BaseDialog
title={title}
className='mx_ShareDialog' className='mx_ShareDialog'
contentId='mx_Dialog_content' contentId='mx_Dialog_content'
onFinished={this.props.onFinished} onFinished={this.props.onFinished}
> >
<div className="mx_ShareDialog_content"> <div className="mx_ShareDialog_content">
<div className="mx_ShareDialog_matrixto"> <div className="mx_ShareDialog_matrixto">
<a href={matrixToUrl} <a
href={matrixToUrl}
onClick={ShareDialog.onLinkClick} onClick={ShareDialog.onLinkClick}
className="mx_ShareDialog_matrixto_link" className="mx_ShareDialog_matrixto_link"
> >

View file

@ -15,14 +15,12 @@ limitations under the License.
*/ */
import React, {createRef} from 'react'; import React, {createRef} from 'react';
import createReactClass from 'create-react-class';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import * as sdk from '../../../index'; import * as sdk from '../../../index';
import Field from "../elements/Field"; import Field from "../elements/Field";
export default createReactClass({ export default class TextInputDialog extends React.Component {
displayName: 'TextInputDialog', static propTypes = {
propTypes: {
title: PropTypes.string, title: PropTypes.string,
description: PropTypes.oneOfType([ description: PropTypes.oneOfType([
PropTypes.element, PropTypes.element,
@ -36,39 +34,36 @@ export default createReactClass({
hasCancel: PropTypes.bool, hasCancel: PropTypes.bool,
validator: PropTypes.func, // result of withValidation validator: PropTypes.func, // result of withValidation
fixedWidth: PropTypes.bool, fixedWidth: PropTypes.bool,
}, };
getDefaultProps: function() { static defaultProps = {
return {
title: "", title: "",
value: "", value: "",
description: "", description: "",
focus: true, focus: true,
hasCancel: true, hasCancel: true,
}; };
},
getInitialState: function() { constructor(props) {
return { super(props);
this._field = createRef();
this.state = {
value: this.props.value, value: this.props.value,
valid: false, valid: false,
}; };
}, }
// TODO: [REACT-WARNING] Replace component with real class, use constructor for refs componentDidMount() {
UNSAFE_componentWillMount: function() {
this._field = createRef();
},
componentDidMount: function() {
if (this.props.focus) { if (this.props.focus) {
// Set the cursor at the end of the text input // Set the cursor at the end of the text input
// this._field.current.value = this.props.value; // this._field.current.value = this.props.value;
this._field.current.focus(); this._field.current.focus();
} }
}, }
onOk: async function(ev) { onOk = async ev => {
ev.preventDefault(); ev.preventDefault();
if (this.props.validator) { if (this.props.validator) {
await this._field.current.validate({ allowEmpty: false }); await this._field.current.validate({ allowEmpty: false });
@ -80,27 +75,27 @@ export default createReactClass({
} }
} }
this.props.onFinished(true, this.state.value); this.props.onFinished(true, this.state.value);
}, };
onCancel: function() { onCancel = () => {
this.props.onFinished(false); this.props.onFinished(false);
}, };
onChange: function(ev) { onChange = ev => {
this.setState({ this.setState({
value: ev.target.value, value: ev.target.value,
}); });
}, };
onValidate: async function(fieldState) { onValidate = async fieldState => {
const result = await this.props.validator(fieldState); const result = await this.props.validator(fieldState);
this.setState({ this.setState({
valid: result.valid, valid: result.valid,
}); });
return result; return result;
}, };
render: function() { render() {
const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog'); const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog');
const DialogButtons = sdk.getComponent('views.elements.DialogButtons'); const DialogButtons = sdk.getComponent('views.elements.DialogButtons');
return ( return (
@ -137,5 +132,5 @@ export default createReactClass({
/> />
</BaseDialog> </BaseDialog>
); );
}, }
}); }

View file

@ -21,7 +21,7 @@ import * as sdk from '../../../../index';
import {MatrixClientPeg} from '../../../../MatrixClientPeg'; import {MatrixClientPeg} from '../../../../MatrixClientPeg';
import { MatrixClient } from 'matrix-js-sdk'; import { MatrixClient } from 'matrix-js-sdk';
import { _t } from '../../../../languageHandler'; import { _t } from '../../../../languageHandler';
import { accessSecretStorage } from '../../../../CrossSigningManager'; import { accessSecretStorage } from '../../../../SecurityManager';
const RESTORE_TYPE_PASSPHRASE = 0; const RESTORE_TYPE_PASSPHRASE = 0;
const RESTORE_TYPE_RECOVERYKEY = 1; const RESTORE_TYPE_RECOVERYKEY = 1;

View file

@ -15,7 +15,7 @@ See the License for the specific language governing permissions and
limitations under the License. limitations under the License.
*/ */
import { debounce } from 'lodash'; import {debounce} from "lodash";
import classNames from 'classnames'; import classNames from 'classnames';
import React from 'react'; import React from 'react';
import PropTypes from "prop-types"; import PropTypes from "prop-types";

View file

@ -16,16 +16,13 @@ limitations under the License.
import React from 'react'; import React from 'react';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import createReactClass from 'create-react-class';
import AccessibleButton from './AccessibleButton'; import AccessibleButton from './AccessibleButton';
import dis from '../../../dispatcher/dispatcher'; import dis from '../../../dispatcher/dispatcher';
import * as sdk from '../../../index'; import * as sdk from '../../../index';
import Analytics from '../../../Analytics'; import Analytics from '../../../Analytics';
export default createReactClass({ export default class ActionButton extends React.Component {
displayName: 'RoleButton', static propTypes = {
propTypes: {
size: PropTypes.string, size: PropTypes.string,
tooltip: PropTypes.bool, tooltip: PropTypes.bool,
action: PropTypes.string.isRequired, action: PropTypes.string.isRequired,
@ -33,39 +30,35 @@ export default createReactClass({
label: PropTypes.string.isRequired, label: PropTypes.string.isRequired,
iconPath: PropTypes.string, iconPath: PropTypes.string,
className: PropTypes.string, className: PropTypes.string,
}, };
getDefaultProps: function() { static defaultProps = {
return {
size: "25", size: "25",
tooltip: false, tooltip: false,
}; };
},
getInitialState: function() { state = {
return {
showTooltip: false, showTooltip: false,
}; };
},
_onClick: function(ev) { _onClick = (ev) => {
ev.stopPropagation(); ev.stopPropagation();
Analytics.trackEvent('Action Button', 'click', this.props.action); Analytics.trackEvent('Action Button', 'click', this.props.action);
dis.dispatch({action: this.props.action}); dis.dispatch({action: this.props.action});
}, };
_onMouseEnter: function() { _onMouseEnter = () => {
if (this.props.tooltip) this.setState({showTooltip: true}); if (this.props.tooltip) this.setState({showTooltip: true});
if (this.props.mouseOverAction) { if (this.props.mouseOverAction) {
dis.dispatch({action: this.props.mouseOverAction}); dis.dispatch({action: this.props.mouseOverAction});
} }
}, };
_onMouseLeave: function() { _onMouseLeave = () => {
this.setState({showTooltip: false}); this.setState({showTooltip: false});
}, };
render: function() { render() {
const TintableSvg = sdk.getComponent("elements.TintableSvg"); const TintableSvg = sdk.getComponent("elements.TintableSvg");
let tooltip; let tooltip;
@ -94,5 +87,5 @@ export default createReactClass({
{ tooltip } { tooltip }
</AccessibleButton> </AccessibleButton>
); );
}, }
}); }

View file

@ -17,15 +17,12 @@ limitations under the License.
import React from 'react'; import React from 'react';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import createReactClass from 'create-react-class';
import * as sdk from '../../../index'; import * as sdk from '../../../index';
import classNames from 'classnames'; import classNames from 'classnames';
import { UserAddressType } from '../../../UserAddress'; import { UserAddressType } from '../../../UserAddress';
export default createReactClass({ export default class AddressSelector extends React.Component {
displayName: 'AddressSelector', static propTypes = {
propTypes: {
onSelected: PropTypes.func.isRequired, onSelected: PropTypes.func.isRequired,
// List of the addresses to display // List of the addresses to display
@ -37,90 +34,91 @@ export default createReactClass({
// Element to put as a header on top of the list // Element to put as a header on top of the list
header: PropTypes.node, header: PropTypes.node,
}, };
getInitialState: function() { constructor(props) {
return { super(props);
this.state = {
selected: this.props.selected === undefined ? 0 : this.props.selected, selected: this.props.selected === undefined ? 0 : this.props.selected,
hover: false, hover: false,
}; };
}, }
// TODO: [REACT-WARNING] Replace with appropriate lifecycle event // TODO: [REACT-WARNING] Replace with appropriate lifecycle event
UNSAFE_componentWillReceiveProps: function(props) { UNSAFE_componentWillReceiveProps(props) {
// Make sure the selected item isn't outside the list bounds // Make sure the selected item isn't outside the list bounds
const selected = this.state.selected; const selected = this.state.selected;
const maxSelected = this._maxSelected(props.addressList); const maxSelected = this._maxSelected(props.addressList);
if (selected > maxSelected) { if (selected > maxSelected) {
this.setState({ selected: maxSelected }); this.setState({ selected: maxSelected });
} }
}, }
componentDidUpdate: function() { componentDidUpdate() {
// As the user scrolls with the arrow keys keep the selected item // As the user scrolls with the arrow keys keep the selected item
// at the top of the window. // at the top of the window.
if (this.scrollElement && this.props.addressList.length > 0 && !this.state.hover) { if (this.scrollElement && this.props.addressList.length > 0 && !this.state.hover) {
const elementHeight = this.addressListElement.getBoundingClientRect().height; const elementHeight = this.addressListElement.getBoundingClientRect().height;
this.scrollElement.scrollTop = (this.state.selected * elementHeight) - elementHeight; this.scrollElement.scrollTop = (this.state.selected * elementHeight) - elementHeight;
} }
}, }
moveSelectionTop: function() { moveSelectionTop = () => {
if (this.state.selected > 0) { if (this.state.selected > 0) {
this.setState({ this.setState({
selected: 0, selected: 0,
hover: false, hover: false,
}); });
} }
}, };
moveSelectionUp: function() { moveSelectionUp = () => {
if (this.state.selected > 0) { if (this.state.selected > 0) {
this.setState({ this.setState({
selected: this.state.selected - 1, selected: this.state.selected - 1,
hover: false, hover: false,
}); });
} }
}, };
moveSelectionDown: function() { moveSelectionDown = () => {
if (this.state.selected < this._maxSelected(this.props.addressList)) { if (this.state.selected < this._maxSelected(this.props.addressList)) {
this.setState({ this.setState({
selected: this.state.selected + 1, selected: this.state.selected + 1,
hover: false, hover: false,
}); });
} }
}, };
chooseSelection: function() { chooseSelection = () => {
this.selectAddress(this.state.selected); this.selectAddress(this.state.selected);
}, };
onClick: function(index) { onClick = index => {
this.selectAddress(index); this.selectAddress(index);
}, };
onMouseEnter: function(index) { onMouseEnter = index => {
this.setState({ this.setState({
selected: index, selected: index,
hover: true, hover: true,
}); });
}, };
onMouseLeave: function() { onMouseLeave = () => {
this.setState({ hover: false }); this.setState({ hover: false });
}, };
selectAddress: function(index) { selectAddress = index => {
// Only try to select an address if one exists // Only try to select an address if one exists
if (this.props.addressList.length !== 0) { if (this.props.addressList.length !== 0) {
this.props.onSelected(index); this.props.onSelected(index);
this.setState({ hover: false }); this.setState({ hover: false });
} }
}, };
createAddressListTiles: function() { createAddressListTiles() {
const self = this;
const AddressTile = sdk.getComponent("elements.AddressTile"); const AddressTile = sdk.getComponent("elements.AddressTile");
const maxSelected = this._maxSelected(this.props.addressList); const maxSelected = this._maxSelected(this.props.addressList);
const addressList = []; const addressList = [];
@ -157,15 +155,15 @@ export default createReactClass({
} }
} }
return addressList; return addressList;
}, }
_maxSelected: function(list) { _maxSelected(list) {
const listSize = list.length === 0 ? 0 : list.length - 1; const listSize = list.length === 0 ? 0 : list.length - 1;
const maxSelected = listSize > (this.props.truncateAt - 1) ? (this.props.truncateAt - 1) : listSize; const maxSelected = listSize > (this.props.truncateAt - 1) ? (this.props.truncateAt - 1) : listSize;
return maxSelected; return maxSelected;
}, }
render: function() { render() {
const classes = classNames({ const classes = classNames({
"mx_AddressSelector": true, "mx_AddressSelector": true,
"mx_AddressSelector_empty": this.props.addressList.length === 0, "mx_AddressSelector_empty": this.props.addressList.length === 0,
@ -177,5 +175,5 @@ export default createReactClass({
{ this.createAddressListTiles() } { this.createAddressListTiles() }
</div> </div>
); );
}, }
}); }

Some files were not shown because too many files have changed in this diff Show more