2018-04-12 01:58:04 +03:00
|
|
|
/*
|
|
|
|
Copyright 2017 Michael Telatynski <7t3chguy@gmail.com>
|
|
|
|
|
|
|
|
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.
|
|
|
|
*/
|
|
|
|
|
2020-02-21 20:15:53 +03:00
|
|
|
import React, {useState, useEffect} from 'react';
|
2018-04-12 01:58:04 +03:00
|
|
|
import PropTypes from 'prop-types';
|
2019-12-20 04:19:56 +03:00
|
|
|
import * as sdk from '../../../index';
|
2018-04-12 01:58:04 +03:00
|
|
|
import SyntaxHighlight from '../elements/SyntaxHighlight';
|
2018-04-13 02:43:44 +03:00
|
|
|
import { _t } from '../../../languageHandler';
|
2020-07-16 22:00:02 +03:00
|
|
|
import { Room, MatrixEvent } from "matrix-js-sdk";
|
2019-02-24 07:28:42 +03:00
|
|
|
import Field from "../elements/Field";
|
2019-12-17 20:26:12 +03:00
|
|
|
import MatrixClientContext from "../../../contexts/MatrixClientContext";
|
2020-02-24 16:44:04 +03:00
|
|
|
import {useEventEmitter} from "../../../hooks/useEventEmitter";
|
2018-04-12 01:58:04 +03:00
|
|
|
|
2020-02-21 20:15:53 +03:00
|
|
|
import {
|
|
|
|
PHASE_UNSENT,
|
|
|
|
PHASE_REQUESTED,
|
|
|
|
PHASE_READY,
|
|
|
|
PHASE_DONE,
|
|
|
|
PHASE_STARTED,
|
|
|
|
PHASE_CANCELLED,
|
2020-02-24 13:17:33 +03:00
|
|
|
} from "matrix-js-sdk/src/crypto/verification/request/VerificationRequest";
|
2020-12-30 01:30:22 +03:00
|
|
|
import WidgetStore from "../../../stores/WidgetStore";
|
|
|
|
import {UPDATE_EVENT} from "../../../stores/AsyncStore";
|
2020-02-21 20:15:53 +03:00
|
|
|
|
2019-12-17 20:26:12 +03:00
|
|
|
class GenericEditor extends React.PureComponent {
|
2018-04-12 01:58:04 +03:00
|
|
|
// static propTypes = {onBack: PropTypes.func.isRequired};
|
|
|
|
|
2019-12-17 20:26:12 +03:00
|
|
|
constructor(props) {
|
|
|
|
super(props);
|
2018-04-12 01:58:04 +03:00
|
|
|
this._onChange = this._onChange.bind(this);
|
|
|
|
this.onBack = this.onBack.bind(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
onBack() {
|
|
|
|
if (this.state.message) {
|
|
|
|
this.setState({ message: null });
|
|
|
|
} else {
|
|
|
|
this.props.onBack();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
_onChange(e) {
|
|
|
|
this.setState({[e.target.id]: e.target.type === 'checkbox' ? e.target.checked : e.target.value});
|
|
|
|
}
|
|
|
|
|
|
|
|
_buttons() {
|
|
|
|
return <div className="mx_Dialog_buttons">
|
|
|
|
<button onClick={this.onBack}>{ _t('Back') }</button>
|
|
|
|
{ !this.state.message && <button onClick={this._send}>{ _t('Send') }</button> }
|
|
|
|
</div>;
|
|
|
|
}
|
|
|
|
|
|
|
|
textInput(id, label) {
|
2019-02-26 01:38:33 +03:00
|
|
|
return <Field id={id} label={label} size="42" autoFocus={true} type="text" autoComplete="on"
|
2019-02-24 07:28:42 +03:00
|
|
|
value={this.state[id]} onChange={this._onChange} />;
|
2018-04-12 01:58:04 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class SendCustomEvent extends GenericEditor {
|
|
|
|
static getLabel() { return _t('Send Custom Event'); }
|
|
|
|
|
|
|
|
static propTypes = {
|
|
|
|
onBack: PropTypes.func.isRequired,
|
2019-12-17 20:26:12 +03:00
|
|
|
room: PropTypes.instanceOf(Room).isRequired,
|
2018-04-12 01:58:04 +03:00
|
|
|
forceStateEvent: PropTypes.bool,
|
|
|
|
inputs: PropTypes.object,
|
|
|
|
};
|
|
|
|
|
2019-12-17 20:26:12 +03:00
|
|
|
static contextType = MatrixClientContext;
|
|
|
|
|
|
|
|
constructor(props) {
|
|
|
|
super(props);
|
2018-04-12 01:58:04 +03:00
|
|
|
this._send = this._send.bind(this);
|
|
|
|
|
|
|
|
const {eventType, stateKey, evContent} = Object.assign({
|
|
|
|
eventType: '',
|
|
|
|
stateKey: '',
|
|
|
|
evContent: '{\n\n}',
|
|
|
|
}, this.props.inputs);
|
|
|
|
|
|
|
|
this.state = {
|
|
|
|
isStateEvent: Boolean(this.props.forceStateEvent),
|
|
|
|
|
|
|
|
eventType,
|
|
|
|
stateKey,
|
|
|
|
evContent,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
send(content) {
|
2019-12-17 20:26:12 +03:00
|
|
|
const cli = this.context;
|
2018-04-12 01:58:04 +03:00
|
|
|
if (this.state.isStateEvent) {
|
2019-12-17 20:26:12 +03:00
|
|
|
return cli.sendStateEvent(this.props.room.roomId, this.state.eventType, content, this.state.stateKey);
|
2018-04-12 01:58:04 +03:00
|
|
|
} else {
|
2019-12-17 20:26:12 +03:00
|
|
|
return cli.sendEvent(this.props.room.roomId, this.state.eventType, content);
|
2018-04-12 01:58:04 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async _send() {
|
|
|
|
if (this.state.eventType === '') {
|
|
|
|
this.setState({ message: _t('You must specify an event type!') });
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
let message;
|
|
|
|
try {
|
|
|
|
const content = JSON.parse(this.state.evContent);
|
|
|
|
await this.send(content);
|
|
|
|
message = _t('Event sent!');
|
|
|
|
} catch (e) {
|
|
|
|
message = _t('Failed to send custom event.') + ' (' + e.toString() + ')';
|
|
|
|
}
|
|
|
|
this.setState({ message });
|
|
|
|
}
|
|
|
|
|
|
|
|
render() {
|
|
|
|
if (this.state.message) {
|
|
|
|
return <div>
|
|
|
|
<div className="mx_Dialog_content">
|
|
|
|
{ this.state.message }
|
|
|
|
</div>
|
|
|
|
{ this._buttons() }
|
|
|
|
</div>;
|
|
|
|
}
|
|
|
|
|
|
|
|
return <div>
|
2018-05-23 19:18:33 +03:00
|
|
|
<div className="mx_DevTools_content">
|
2019-05-16 23:55:17 +03:00
|
|
|
<div className="mx_DevTools_eventTypeStateKeyGroup">
|
|
|
|
{ this.textInput('eventType', _t('Event Type')) }
|
|
|
|
{ this.state.isStateEvent && this.textInput('stateKey', _t('State Key')) }
|
|
|
|
</div>
|
2018-04-12 01:58:04 +03:00
|
|
|
|
|
|
|
<br />
|
|
|
|
|
2019-02-24 07:28:42 +03:00
|
|
|
<Field id="evContent" label={_t("Event Content")} type="text" className="mx_DevTools_textarea"
|
|
|
|
autoComplete="off" value={this.state.evContent} onChange={this._onChange} element="textarea" />
|
2018-04-12 01:58:04 +03:00
|
|
|
</div>
|
|
|
|
<div className="mx_Dialog_buttons">
|
|
|
|
<button onClick={this.onBack}>{ _t('Back') }</button>
|
|
|
|
{ !this.state.message && <button onClick={this._send}>{ _t('Send') }</button> }
|
|
|
|
{ !this.state.message && !this.props.forceStateEvent && <div style={{float: "right"}}>
|
|
|
|
<input id="isStateEvent" className="mx_DevTools_tgl mx_DevTools_tgl-flip" type="checkbox" onChange={this._onChange} checked={this.state.isStateEvent} />
|
|
|
|
<label className="mx_DevTools_tgl-btn" data-tg-off="Event" data-tg-on="State Event" htmlFor="isStateEvent" />
|
|
|
|
</div> }
|
|
|
|
</div>
|
|
|
|
</div>;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class SendAccountData extends GenericEditor {
|
|
|
|
static getLabel() { return _t('Send Account Data'); }
|
|
|
|
|
|
|
|
static propTypes = {
|
2019-12-17 20:26:12 +03:00
|
|
|
room: PropTypes.instanceOf(Room).isRequired,
|
2018-04-12 01:58:04 +03:00
|
|
|
isRoomAccountData: PropTypes.bool,
|
|
|
|
forceMode: PropTypes.bool,
|
|
|
|
inputs: PropTypes.object,
|
|
|
|
};
|
|
|
|
|
2019-12-17 20:26:12 +03:00
|
|
|
static contextType = MatrixClientContext;
|
|
|
|
|
|
|
|
constructor(props) {
|
|
|
|
super(props);
|
2018-04-12 01:58:04 +03:00
|
|
|
this._send = this._send.bind(this);
|
|
|
|
|
|
|
|
const {eventType, evContent} = Object.assign({
|
|
|
|
eventType: '',
|
|
|
|
evContent: '{\n\n}',
|
|
|
|
}, this.props.inputs);
|
|
|
|
|
|
|
|
this.state = {
|
|
|
|
isRoomAccountData: Boolean(this.props.isRoomAccountData),
|
|
|
|
|
|
|
|
eventType,
|
|
|
|
evContent,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
send(content) {
|
2019-12-17 20:26:12 +03:00
|
|
|
const cli = this.context;
|
2018-04-12 01:58:04 +03:00
|
|
|
if (this.state.isRoomAccountData) {
|
2019-12-17 20:26:12 +03:00
|
|
|
return cli.setRoomAccountData(this.props.room.roomId, this.state.eventType, content);
|
2018-04-12 01:58:04 +03:00
|
|
|
}
|
|
|
|
return cli.setAccountData(this.state.eventType, content);
|
|
|
|
}
|
|
|
|
|
|
|
|
async _send() {
|
|
|
|
if (this.state.eventType === '') {
|
|
|
|
this.setState({ message: _t('You must specify an event type!') });
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
let message;
|
|
|
|
try {
|
|
|
|
const content = JSON.parse(this.state.evContent);
|
|
|
|
await this.send(content);
|
|
|
|
message = _t('Event sent!');
|
|
|
|
} catch (e) {
|
|
|
|
message = _t('Failed to send custom event.') + ' (' + e.toString() + ')';
|
|
|
|
}
|
|
|
|
this.setState({ message });
|
|
|
|
}
|
|
|
|
|
|
|
|
render() {
|
|
|
|
if (this.state.message) {
|
|
|
|
return <div>
|
|
|
|
<div className="mx_Dialog_content">
|
|
|
|
{ this.state.message }
|
|
|
|
</div>
|
|
|
|
{ this._buttons() }
|
|
|
|
</div>;
|
|
|
|
}
|
|
|
|
|
|
|
|
return <div>
|
2018-05-23 19:40:15 +03:00
|
|
|
<div className="mx_DevTools_content">
|
2018-04-12 01:58:04 +03:00
|
|
|
{ this.textInput('eventType', _t('Event Type')) }
|
|
|
|
<br />
|
|
|
|
|
2019-02-26 01:12:06 +03:00
|
|
|
<Field id="evContent" label={_t("Event Content")} type="text" className="mx_DevTools_textarea"
|
|
|
|
autoComplete="off" value={this.state.evContent} onChange={this._onChange} element="textarea" />
|
2018-04-12 01:58:04 +03:00
|
|
|
</div>
|
|
|
|
<div className="mx_Dialog_buttons">
|
|
|
|
<button onClick={this.onBack}>{ _t('Back') }</button>
|
|
|
|
{ !this.state.message && <button onClick={this._send}>{ _t('Send') }</button> }
|
|
|
|
{ !this.state.message && <div style={{float: "right"}}>
|
|
|
|
<input id="isRoomAccountData" className="mx_DevTools_tgl mx_DevTools_tgl-flip" type="checkbox" onChange={this._onChange} checked={this.state.isRoomAccountData} disabled={this.props.forceMode} />
|
|
|
|
<label className="mx_DevTools_tgl-btn" data-tg-off="Account Data" data-tg-on="Room Data" htmlFor="isRoomAccountData" />
|
|
|
|
</div> }
|
|
|
|
</div>
|
|
|
|
</div>;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-04 14:57:22 +03:00
|
|
|
const INITIAL_LOAD_TILES = 20;
|
2018-07-04 15:07:57 +03:00
|
|
|
const LOAD_TILES_STEP_SIZE = 50;
|
2018-07-04 14:57:22 +03:00
|
|
|
|
2019-12-17 20:26:12 +03:00
|
|
|
class FilteredList extends React.PureComponent {
|
2018-04-12 01:58:04 +03:00
|
|
|
static propTypes = {
|
|
|
|
children: PropTypes.any,
|
|
|
|
query: PropTypes.string,
|
|
|
|
onChange: PropTypes.func,
|
|
|
|
};
|
|
|
|
|
2018-07-04 14:57:22 +03:00
|
|
|
static filterChildren(children, query) {
|
|
|
|
if (!query) return children;
|
|
|
|
const lcQuery = query.toLowerCase();
|
|
|
|
return children.filter((child) => child.key.toLowerCase().includes(lcQuery));
|
|
|
|
}
|
|
|
|
|
2019-12-17 20:26:12 +03:00
|
|
|
constructor(props) {
|
|
|
|
super(props);
|
2018-04-12 01:58:04 +03:00
|
|
|
|
2018-07-04 14:57:22 +03:00
|
|
|
this.state = {
|
|
|
|
filteredChildren: FilteredList.filterChildren(this.props.children, this.props.query),
|
|
|
|
truncateAt: INITIAL_LOAD_TILES,
|
|
|
|
};
|
2018-04-12 01:58:04 +03:00
|
|
|
}
|
|
|
|
|
2020-04-01 23:35:39 +03:00
|
|
|
// TODO: [REACT-WARNING] Replace with appropriate lifecycle event
|
2020-04-01 23:45:54 +03:00
|
|
|
UNSAFE_componentWillReceiveProps(nextProps) { // eslint-disable-line camelcase
|
2018-07-04 14:57:22 +03:00
|
|
|
if (this.props.children === nextProps.children && this.props.query === nextProps.query) return;
|
|
|
|
this.setState({
|
|
|
|
filteredChildren: FilteredList.filterChildren(nextProps.children, nextProps.query),
|
|
|
|
truncateAt: INITIAL_LOAD_TILES,
|
|
|
|
});
|
2018-04-12 01:58:04 +03:00
|
|
|
}
|
|
|
|
|
2018-07-04 14:57:22 +03:00
|
|
|
showAll = () => {
|
|
|
|
this.setState({
|
2018-07-04 15:07:57 +03:00
|
|
|
truncateAt: this.state.truncateAt + LOAD_TILES_STEP_SIZE,
|
2018-07-04 14:57:22 +03:00
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
createOverflowElement = (overflowCount: number, totalCount: number) => {
|
|
|
|
return <button className="mx_DevTools_RoomStateExplorer_button" onClick={this.showAll}>
|
|
|
|
{ _t("and %(count)s others...", { count: overflowCount }) }
|
|
|
|
</button>;
|
|
|
|
};
|
|
|
|
|
|
|
|
onQuery = (ev) => {
|
|
|
|
if (this.props.onChange) this.props.onChange(ev.target.value);
|
|
|
|
};
|
|
|
|
|
|
|
|
getChildren = (start: number, end: number) => {
|
|
|
|
return this.state.filteredChildren.slice(start, end);
|
|
|
|
};
|
|
|
|
|
|
|
|
getChildCount = (): number => {
|
|
|
|
return this.state.filteredChildren.length;
|
|
|
|
};
|
|
|
|
|
2018-04-12 01:58:04 +03:00
|
|
|
render() {
|
2018-07-04 14:57:22 +03:00
|
|
|
const TruncatedList = sdk.getComponent("elements.TruncatedList");
|
2018-04-12 01:58:04 +03:00
|
|
|
return <div>
|
2020-03-30 00:59:15 +03:00
|
|
|
<Field label={_t('Filter results')} autoFocus={true} size={64}
|
2019-02-24 07:28:42 +03:00
|
|
|
type="text" autoComplete="off" value={this.props.query} onChange={this.onQuery}
|
2018-07-26 20:17:45 +03:00
|
|
|
className="mx_TextInputDialog_input mx_DevTools_RoomStateExplorer_query"
|
|
|
|
// force re-render so that autoFocus is applied when this component is re-used
|
|
|
|
key={this.props.children[0] ? this.props.children[0].key : ''} />
|
2019-02-24 07:28:42 +03:00
|
|
|
|
2018-07-04 14:57:22 +03:00
|
|
|
<TruncatedList getChildren={this.getChildren}
|
|
|
|
getChildCount={this.getChildCount}
|
|
|
|
truncateAt={this.state.truncateAt}
|
|
|
|
createOverflowElement={this.createOverflowElement} />
|
2018-04-12 01:58:04 +03:00
|
|
|
</div>;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-17 20:26:12 +03:00
|
|
|
class RoomStateExplorer extends React.PureComponent {
|
2018-04-12 01:58:04 +03:00
|
|
|
static getLabel() { return _t('Explore Room State'); }
|
|
|
|
|
|
|
|
static propTypes = {
|
|
|
|
onBack: PropTypes.func.isRequired,
|
2019-12-17 20:26:12 +03:00
|
|
|
room: PropTypes.instanceOf(Room).isRequired,
|
2018-04-12 01:58:04 +03:00
|
|
|
};
|
|
|
|
|
2019-12-17 20:26:12 +03:00
|
|
|
static contextType = MatrixClientContext;
|
|
|
|
|
2020-07-16 22:00:02 +03:00
|
|
|
roomStateEvents: Map<string, Map<string, MatrixEvent>>;
|
|
|
|
|
2019-12-17 20:26:12 +03:00
|
|
|
constructor(props) {
|
|
|
|
super(props);
|
2018-04-12 01:58:04 +03:00
|
|
|
|
2019-12-17 20:26:12 +03:00
|
|
|
this.roomStateEvents = this.props.room.currentState.events;
|
2018-04-12 01:58:04 +03:00
|
|
|
|
|
|
|
this.onBack = this.onBack.bind(this);
|
|
|
|
this.editEv = this.editEv.bind(this);
|
|
|
|
this.onQueryEventType = this.onQueryEventType.bind(this);
|
|
|
|
this.onQueryStateKey = this.onQueryStateKey.bind(this);
|
|
|
|
|
|
|
|
this.state = {
|
|
|
|
eventType: null,
|
|
|
|
event: null,
|
|
|
|
editing: false,
|
|
|
|
|
|
|
|
queryEventType: '',
|
|
|
|
queryStateKey: '',
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
browseEventType(eventType) {
|
|
|
|
return () => {
|
|
|
|
this.setState({ eventType });
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
onViewSourceClick(event) {
|
|
|
|
return () => {
|
|
|
|
this.setState({ event });
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
onBack() {
|
|
|
|
if (this.state.editing) {
|
|
|
|
this.setState({ editing: false });
|
|
|
|
} else if (this.state.event) {
|
|
|
|
this.setState({ event: null });
|
|
|
|
} else if (this.state.eventType) {
|
|
|
|
this.setState({ eventType: null });
|
|
|
|
} else {
|
|
|
|
this.props.onBack();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
editEv() {
|
|
|
|
this.setState({ editing: true });
|
|
|
|
}
|
|
|
|
|
|
|
|
onQueryEventType(filterEventType) {
|
|
|
|
this.setState({ queryEventType: filterEventType });
|
|
|
|
}
|
|
|
|
|
|
|
|
onQueryStateKey(filterStateKey) {
|
|
|
|
this.setState({ queryStateKey: filterStateKey });
|
|
|
|
}
|
|
|
|
|
|
|
|
render() {
|
|
|
|
if (this.state.event) {
|
|
|
|
if (this.state.editing) {
|
2019-12-17 20:26:12 +03:00
|
|
|
return <SendCustomEvent room={this.props.room} forceStateEvent={true} onBack={this.onBack} inputs={{
|
2018-04-12 01:58:04 +03:00
|
|
|
eventType: this.state.event.getType(),
|
|
|
|
evContent: JSON.stringify(this.state.event.getContent(), null, '\t'),
|
|
|
|
stateKey: this.state.event.getStateKey(),
|
|
|
|
}} />;
|
|
|
|
}
|
|
|
|
|
|
|
|
return <div className="mx_ViewSource">
|
|
|
|
<div className="mx_Dialog_content">
|
|
|
|
<SyntaxHighlight className="json">
|
|
|
|
{ JSON.stringify(this.state.event.event, null, 2) }
|
|
|
|
</SyntaxHighlight>
|
|
|
|
</div>
|
|
|
|
<div className="mx_Dialog_buttons">
|
|
|
|
<button onClick={this.onBack}>{ _t('Back') }</button>
|
|
|
|
<button onClick={this.editEv}>{ _t('Edit') }</button>
|
|
|
|
</div>
|
|
|
|
</div>;
|
|
|
|
}
|
|
|
|
|
|
|
|
let list = null;
|
|
|
|
|
|
|
|
const classes = 'mx_DevTools_RoomStateExplorer_button';
|
|
|
|
if (this.state.eventType === null) {
|
|
|
|
list = <FilteredList query={this.state.queryEventType} onChange={this.onQueryEventType}>
|
|
|
|
{
|
2020-07-16 22:00:02 +03:00
|
|
|
Array.from(this.roomStateEvents.entries()).map(([eventType, allStateKeys]) => {
|
2018-04-12 01:58:04 +03:00
|
|
|
let onClickFn;
|
2020-07-17 03:34:04 +03:00
|
|
|
if (allStateKeys.size === 1 && allStateKeys.has("")) {
|
2020-07-16 22:00:02 +03:00
|
|
|
onClickFn = this.onViewSourceClick(allStateKeys.get(""));
|
2018-07-26 20:17:45 +03:00
|
|
|
} else {
|
2020-07-16 22:00:02 +03:00
|
|
|
onClickFn = this.browseEventType(eventType);
|
2018-04-12 01:58:04 +03:00
|
|
|
}
|
|
|
|
|
2020-07-16 22:00:02 +03:00
|
|
|
return <button className={classes} key={eventType} onClick={onClickFn}>
|
|
|
|
{eventType}
|
2018-04-12 01:58:04 +03:00
|
|
|
</button>;
|
|
|
|
})
|
|
|
|
}
|
|
|
|
</FilteredList>;
|
|
|
|
} else {
|
2020-07-09 07:23:51 +03:00
|
|
|
const stateGroup = this.roomStateEvents.get(this.state.eventType);
|
2018-04-12 01:58:04 +03:00
|
|
|
|
|
|
|
list = <FilteredList query={this.state.queryStateKey} onChange={this.onQueryStateKey}>
|
|
|
|
{
|
2020-07-16 22:00:02 +03:00
|
|
|
Array.from(stateGroup.entries()).map(([stateKey, ev]) => {
|
2018-04-12 01:58:04 +03:00
|
|
|
return <button className={classes} key={stateKey} onClick={this.onViewSourceClick(ev)}>
|
|
|
|
{ stateKey }
|
|
|
|
</button>;
|
|
|
|
})
|
|
|
|
}
|
|
|
|
</FilteredList>;
|
|
|
|
}
|
|
|
|
|
|
|
|
return <div>
|
|
|
|
<div className="mx_Dialog_content">
|
|
|
|
{ list }
|
|
|
|
</div>
|
|
|
|
<div className="mx_Dialog_buttons">
|
|
|
|
<button onClick={this.onBack}>{ _t('Back') }</button>
|
|
|
|
</div>
|
|
|
|
</div>;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-17 20:26:12 +03:00
|
|
|
class AccountDataExplorer extends React.PureComponent {
|
2018-04-12 01:58:04 +03:00
|
|
|
static getLabel() { return _t('Explore Account Data'); }
|
|
|
|
|
|
|
|
static propTypes = {
|
|
|
|
onBack: PropTypes.func.isRequired,
|
2019-12-17 20:26:12 +03:00
|
|
|
room: PropTypes.instanceOf(Room).isRequired,
|
2018-04-12 01:58:04 +03:00
|
|
|
};
|
|
|
|
|
2019-12-17 20:26:12 +03:00
|
|
|
static contextType = MatrixClientContext;
|
|
|
|
|
|
|
|
constructor(props) {
|
|
|
|
super(props);
|
2018-04-12 01:58:04 +03:00
|
|
|
|
|
|
|
this.onBack = this.onBack.bind(this);
|
|
|
|
this.editEv = this.editEv.bind(this);
|
|
|
|
this._onChange = this._onChange.bind(this);
|
|
|
|
this.onQueryEventType = this.onQueryEventType.bind(this);
|
|
|
|
|
|
|
|
this.state = {
|
|
|
|
isRoomAccountData: false,
|
|
|
|
event: null,
|
|
|
|
editing: false,
|
|
|
|
|
|
|
|
queryEventType: '',
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
getData() {
|
|
|
|
if (this.state.isRoomAccountData) {
|
2019-12-17 20:26:12 +03:00
|
|
|
return this.props.room.accountData;
|
2018-04-12 01:58:04 +03:00
|
|
|
}
|
2019-12-17 20:26:12 +03:00
|
|
|
return this.context.store.accountData;
|
2018-04-12 01:58:04 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
onViewSourceClick(event) {
|
|
|
|
return () => {
|
|
|
|
this.setState({ event });
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
onBack() {
|
|
|
|
if (this.state.editing) {
|
|
|
|
this.setState({ editing: false });
|
|
|
|
} else if (this.state.event) {
|
|
|
|
this.setState({ event: null });
|
|
|
|
} else {
|
|
|
|
this.props.onBack();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
_onChange(e) {
|
|
|
|
this.setState({[e.target.id]: e.target.type === 'checkbox' ? e.target.checked : e.target.value});
|
|
|
|
}
|
|
|
|
|
|
|
|
editEv() {
|
|
|
|
this.setState({ editing: true });
|
|
|
|
}
|
|
|
|
|
|
|
|
onQueryEventType(queryEventType) {
|
|
|
|
this.setState({ queryEventType });
|
|
|
|
}
|
|
|
|
|
|
|
|
render() {
|
|
|
|
if (this.state.event) {
|
|
|
|
if (this.state.editing) {
|
2019-12-17 20:26:12 +03:00
|
|
|
return <SendAccountData
|
|
|
|
room={this.props.room}
|
|
|
|
isRoomAccountData={this.state.isRoomAccountData}
|
|
|
|
onBack={this.onBack}
|
|
|
|
inputs={{
|
|
|
|
eventType: this.state.event.getType(),
|
|
|
|
evContent: JSON.stringify(this.state.event.getContent(), null, '\t'),
|
|
|
|
}} forceMode={true} />;
|
2018-04-12 01:58:04 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
return <div className="mx_ViewSource">
|
2018-05-23 19:18:33 +03:00
|
|
|
<div className="mx_DevTools_content">
|
2018-04-12 01:58:04 +03:00
|
|
|
<SyntaxHighlight className="json">
|
|
|
|
{ JSON.stringify(this.state.event.event, null, 2) }
|
|
|
|
</SyntaxHighlight>
|
|
|
|
</div>
|
|
|
|
<div className="mx_Dialog_buttons">
|
|
|
|
<button onClick={this.onBack}>{ _t('Back') }</button>
|
|
|
|
<button onClick={this.editEv}>{ _t('Edit') }</button>
|
|
|
|
</div>
|
|
|
|
</div>;
|
|
|
|
}
|
|
|
|
|
|
|
|
const rows = [];
|
|
|
|
|
|
|
|
const classes = 'mx_DevTools_RoomStateExplorer_button';
|
|
|
|
|
|
|
|
const data = this.getData();
|
|
|
|
Object.keys(data).forEach((evType) => {
|
|
|
|
const ev = data[evType];
|
|
|
|
rows.push(<button className={classes} key={evType} onClick={this.onViewSourceClick(ev)}>
|
|
|
|
{ evType }
|
|
|
|
</button>);
|
|
|
|
});
|
|
|
|
|
|
|
|
return <div>
|
|
|
|
<div className="mx_Dialog_content">
|
|
|
|
<FilteredList query={this.state.queryEventType} onChange={this.onQueryEventType}>
|
|
|
|
{ rows }
|
|
|
|
</FilteredList>
|
|
|
|
</div>
|
|
|
|
<div className="mx_Dialog_buttons">
|
|
|
|
<button onClick={this.onBack}>{ _t('Back') }</button>
|
|
|
|
{ !this.state.message && <div style={{float: "right"}}>
|
|
|
|
<input id="isRoomAccountData" className="mx_DevTools_tgl mx_DevTools_tgl-flip" type="checkbox" onChange={this._onChange} checked={this.state.isRoomAccountData} />
|
|
|
|
<label className="mx_DevTools_tgl-btn" data-tg-off="Account Data" data-tg-on="Room Data" htmlFor="isRoomAccountData" />
|
|
|
|
</div> }
|
|
|
|
</div>
|
|
|
|
</div>;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-17 20:26:12 +03:00
|
|
|
class ServersInRoomList extends React.PureComponent {
|
2019-03-19 02:21:21 +03:00
|
|
|
static getLabel() { return _t('View Servers in Room'); }
|
|
|
|
|
|
|
|
static propTypes = {
|
|
|
|
onBack: PropTypes.func.isRequired,
|
2019-12-17 20:26:12 +03:00
|
|
|
room: PropTypes.instanceOf(Room).isRequired,
|
2019-03-19 02:21:21 +03:00
|
|
|
};
|
|
|
|
|
2019-12-17 20:26:12 +03:00
|
|
|
static contextType = MatrixClientContext;
|
|
|
|
|
|
|
|
constructor(props) {
|
|
|
|
super(props);
|
2019-03-19 02:21:21 +03:00
|
|
|
|
2019-12-17 20:26:12 +03:00
|
|
|
const room = this.props.room;
|
2019-03-19 02:21:21 +03:00
|
|
|
const servers = new Set();
|
|
|
|
room.currentState.getStateEvents("m.room.member").forEach(ev => servers.add(ev.getSender().split(":")[1]));
|
|
|
|
this.servers = Array.from(servers).map(s =>
|
|
|
|
<button key={s} className="mx_DevTools_ServersInRoomList_button">
|
|
|
|
{ s }
|
|
|
|
</button>);
|
|
|
|
|
|
|
|
this.state = {
|
|
|
|
query: '',
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2019-05-10 00:30:45 +03:00
|
|
|
onQuery = (query) => {
|
2019-03-19 02:21:21 +03:00
|
|
|
this.setState({ query });
|
|
|
|
}
|
|
|
|
|
|
|
|
render() {
|
|
|
|
return <div>
|
|
|
|
<div className="mx_Dialog_content">
|
|
|
|
<FilteredList query={this.state.query} onChange={this.onQuery}>
|
|
|
|
{ this.servers }
|
|
|
|
</FilteredList>
|
|
|
|
</div>
|
|
|
|
<div className="mx_Dialog_buttons">
|
|
|
|
<button onClick={this.props.onBack}>{ _t('Back') }</button>
|
|
|
|
</div>
|
|
|
|
</div>;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-21 20:15:53 +03:00
|
|
|
const PHASE_MAP = {
|
|
|
|
[PHASE_UNSENT]: "unsent",
|
|
|
|
[PHASE_REQUESTED]: "requested",
|
|
|
|
[PHASE_READY]: "ready",
|
|
|
|
[PHASE_DONE]: "done",
|
|
|
|
[PHASE_STARTED]: "started",
|
|
|
|
[PHASE_CANCELLED]: "cancelled",
|
2020-02-24 13:17:33 +03:00
|
|
|
};
|
2020-02-21 20:15:53 +03:00
|
|
|
|
|
|
|
function VerificationRequest({txnId, request}) {
|
|
|
|
const [, updateState] = useState();
|
2020-02-24 14:53:12 +03:00
|
|
|
const [timeout, setRequestTimeout] = useState(request.timeout);
|
2020-02-21 20:15:53 +03:00
|
|
|
|
|
|
|
/* Re-render if something changes state */
|
2020-02-24 16:44:04 +03:00
|
|
|
useEventEmitter(request, "change", updateState);
|
2020-02-21 20:15:53 +03:00
|
|
|
|
|
|
|
/* Keep re-rendering if there's a timeout */
|
|
|
|
useEffect(() => {
|
2020-02-24 13:17:33 +03:00
|
|
|
if (request.timeout == 0) return;
|
2020-02-21 20:15:53 +03:00
|
|
|
|
2020-02-24 14:33:49 +03:00
|
|
|
/* Note that request.timeout is a getter, so its value changes */
|
2020-02-21 20:15:53 +03:00
|
|
|
const id = setInterval(() => {
|
2020-02-24 14:53:12 +03:00
|
|
|
setRequestTimeout(request.timeout);
|
2020-02-21 20:15:53 +03:00
|
|
|
}, 500);
|
|
|
|
|
2020-02-24 13:17:33 +03:00
|
|
|
return () => { clearInterval(id); };
|
|
|
|
}, [request]);
|
2020-02-21 20:15:53 +03:00
|
|
|
|
|
|
|
return (<div className="mx_DevTools_VerificationRequest">
|
|
|
|
<dl>
|
|
|
|
<dt>Transaction</dt>
|
|
|
|
<dd>{txnId}</dd>
|
|
|
|
<dt>Phase</dt>
|
|
|
|
<dd>{PHASE_MAP[request.phase] || request.phase}</dd>
|
|
|
|
<dt>Timeout</dt>
|
|
|
|
<dd>{Math.floor(timeout / 1000)}</dd>
|
|
|
|
<dt>Methods</dt>
|
|
|
|
<dd>{request.methods && request.methods.join(", ")}</dd>
|
|
|
|
<dt>requestingUserId</dt>
|
|
|
|
<dd>{request.requestingUserId}</dd>
|
2020-02-24 17:17:34 +03:00
|
|
|
<dt>observeOnly</dt>
|
|
|
|
<dd>{JSON.stringify(request.observeOnly)}</dd>
|
2020-02-21 20:15:53 +03:00
|
|
|
</dl>
|
|
|
|
</div>);
|
|
|
|
}
|
|
|
|
|
2020-02-24 13:17:33 +03:00
|
|
|
class VerificationExplorer extends React.Component {
|
2020-02-21 20:15:53 +03:00
|
|
|
static getLabel() {
|
|
|
|
return _t("Verification Requests");
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Ensure this.context is the cli */
|
|
|
|
static contextType = MatrixClientContext;
|
|
|
|
|
2020-02-24 13:17:33 +03:00
|
|
|
onNewRequest = () => {
|
|
|
|
this.forceUpdate();
|
|
|
|
}
|
|
|
|
|
|
|
|
componentDidMount() {
|
|
|
|
const cli = this.context;
|
|
|
|
cli.on("crypto.verification.request", this.onNewRequest);
|
|
|
|
}
|
|
|
|
|
|
|
|
componentWillUnmount() {
|
|
|
|
const cli = this.context;
|
|
|
|
cli.off("crypto.verification.request", this.onNewRequest);
|
|
|
|
}
|
|
|
|
|
2020-02-21 20:15:53 +03:00
|
|
|
render() {
|
|
|
|
const cli = this.context;
|
|
|
|
const room = this.props.room;
|
|
|
|
const inRoomChannel = cli._crypto._inRoomVerificationRequests;
|
|
|
|
const inRoomRequests = (inRoomChannel._requestsByRoomId || new Map()).get(room.roomId) || new Map();
|
|
|
|
|
|
|
|
return (<div>
|
|
|
|
<div className="mx_Dialog_content">
|
|
|
|
{Array.from(inRoomRequests.entries()).reverse().map(([txnId, request]) =>
|
2020-02-24 13:17:33 +03:00
|
|
|
<VerificationRequest txnId={txnId} request={request} key={txnId} />,
|
2020-02-21 20:15:53 +03:00
|
|
|
)}
|
|
|
|
</div>
|
2020-04-21 18:27:31 +03:00
|
|
|
<div className="mx_Dialog_buttons">
|
|
|
|
<button onClick={this.props.onBack}>{_t("Back")}</button>
|
|
|
|
</div>
|
2020-02-21 20:15:53 +03:00
|
|
|
</div>);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-12-30 01:30:22 +03:00
|
|
|
class WidgetExplorer extends React.Component {
|
|
|
|
static getLabel() {
|
|
|
|
return _t("Active Widgets");
|
|
|
|
}
|
|
|
|
|
|
|
|
constructor(props) {
|
|
|
|
super(props);
|
|
|
|
|
|
|
|
this.state = {
|
|
|
|
query: '',
|
|
|
|
editWidget: null, // set to an IApp when editing
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
onWidgetStoreUpdate = () => {
|
|
|
|
this.forceUpdate();
|
|
|
|
};
|
|
|
|
|
|
|
|
onQueryChange = (query) => {
|
|
|
|
this.setState({query});
|
|
|
|
};
|
|
|
|
|
|
|
|
onEditWidget = (widget) => {
|
|
|
|
this.setState({editWidget: widget});
|
|
|
|
};
|
|
|
|
|
|
|
|
onBack = () => {
|
|
|
|
const widgets = WidgetStore.instance.getApps(this.props.room.roomId);
|
|
|
|
if (this.state.editWidget && widgets.includes(this.state.editWidget)) {
|
|
|
|
this.setState({editWidget: null});
|
|
|
|
} else {
|
|
|
|
this.props.onBack();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
componentDidMount() {
|
|
|
|
WidgetStore.instance.on(UPDATE_EVENT, this.onWidgetStoreUpdate);
|
|
|
|
}
|
|
|
|
|
|
|
|
componentWillUnmount() {
|
|
|
|
WidgetStore.instance.off(UPDATE_EVENT, this.onWidgetStoreUpdate);
|
|
|
|
}
|
|
|
|
|
|
|
|
render() {
|
|
|
|
const room = this.props.room;
|
|
|
|
|
|
|
|
const editWidget = this.state.editWidget;
|
|
|
|
const widgets = WidgetStore.instance.getApps(room.roomId);
|
|
|
|
if (editWidget && widgets.includes(editWidget)) {
|
|
|
|
const allState = Array.from(Array.from(room.currentState.events.values()).map(e => e.values()))
|
|
|
|
.reduce((p, c) => {p.push(...c); return p;}, []);
|
|
|
|
const stateEv = allState.find(ev => ev.getId() === editWidget.eventId);
|
|
|
|
if (!stateEv) { // "should never happen"
|
|
|
|
return <div>
|
|
|
|
{_t("There was an error finding this widget.")}
|
|
|
|
<div className="mx_Dialog_buttons">
|
|
|
|
<button onClick={this.onBack}>{_t("Back")}</button>
|
|
|
|
</div>
|
|
|
|
</div>;
|
|
|
|
}
|
|
|
|
return <SendCustomEvent
|
|
|
|
onBack={this.onBack}
|
|
|
|
room={room}
|
|
|
|
forceStateEvent={true}
|
|
|
|
inputs={{
|
|
|
|
eventType: stateEv.getType(),
|
|
|
|
evContent: JSON.stringify(stateEv.getContent(), null, '\t'),
|
|
|
|
stateKey: stateEv.getStateKey(),
|
|
|
|
}}
|
|
|
|
/>;
|
|
|
|
}
|
|
|
|
|
|
|
|
return (<div>
|
|
|
|
<div className="mx_Dialog_content">
|
|
|
|
<FilteredList query={this.state.query} onChange={this.onQueryChange}>
|
|
|
|
{widgets.map(w =>
|
|
|
|
<button
|
|
|
|
className='mx_DevTools_RoomStateExplorer_button'
|
|
|
|
key={w.url + w.eventId}
|
|
|
|
onClick={() => this.onEditWidget(w)}
|
|
|
|
>{w.url}</button>
|
|
|
|
)}
|
|
|
|
</FilteredList>
|
|
|
|
</div>
|
|
|
|
<div className="mx_Dialog_buttons">
|
|
|
|
<button onClick={this.onBack}>{_t("Back")}</button>
|
|
|
|
</div>
|
|
|
|
</div>);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-12 01:58:04 +03:00
|
|
|
const Entries = [
|
|
|
|
SendCustomEvent,
|
|
|
|
RoomStateExplorer,
|
|
|
|
SendAccountData,
|
|
|
|
AccountDataExplorer,
|
2019-03-19 02:21:21 +03:00
|
|
|
ServersInRoomList,
|
2020-02-21 20:15:53 +03:00
|
|
|
VerificationExplorer,
|
2020-12-30 01:30:22 +03:00
|
|
|
WidgetExplorer,
|
2018-04-12 01:58:04 +03:00
|
|
|
];
|
|
|
|
|
2019-12-17 20:26:12 +03:00
|
|
|
export default class DevtoolsDialog extends React.PureComponent {
|
2018-04-12 01:58:04 +03:00
|
|
|
static propTypes = {
|
|
|
|
roomId: PropTypes.string.isRequired,
|
|
|
|
onFinished: PropTypes.func.isRequired,
|
|
|
|
};
|
|
|
|
|
2019-12-17 20:26:12 +03:00
|
|
|
constructor(props) {
|
|
|
|
super(props);
|
2018-04-12 01:58:04 +03:00
|
|
|
this.onBack = this.onBack.bind(this);
|
|
|
|
this.onCancel = this.onCancel.bind(this);
|
|
|
|
|
|
|
|
this.state = {
|
|
|
|
mode: null,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
componentWillUnmount() {
|
|
|
|
this._unmounted = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
_setMode(mode) {
|
|
|
|
return () => {
|
|
|
|
this.setState({ mode });
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
onBack() {
|
|
|
|
if (this.prevMode) {
|
|
|
|
this.setState({ mode: this.prevMode });
|
|
|
|
this.prevMode = null;
|
|
|
|
} else {
|
|
|
|
this.setState({ mode: null });
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
onCancel() {
|
|
|
|
this.props.onFinished(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
render() {
|
|
|
|
let body;
|
|
|
|
|
|
|
|
if (this.state.mode) {
|
2019-12-17 20:26:12 +03:00
|
|
|
body = <MatrixClientContext.Consumer>
|
|
|
|
{(cli) => <React.Fragment>
|
|
|
|
<div className="mx_DevTools_label_left">{ this.state.mode.getLabel() }</div>
|
|
|
|
<div className="mx_DevTools_label_right">Room ID: { this.props.roomId }</div>
|
|
|
|
<div className="mx_DevTools_label_bottom" />
|
|
|
|
<this.state.mode onBack={this.onBack} room={cli.getRoom(this.props.roomId)} />
|
|
|
|
</React.Fragment>}
|
|
|
|
</MatrixClientContext.Consumer>;
|
2018-04-12 01:58:04 +03:00
|
|
|
} else {
|
|
|
|
const classes = "mx_DevTools_RoomStateExplorer_button";
|
2019-12-17 20:26:12 +03:00
|
|
|
body = <React.Fragment>
|
2018-10-26 15:15:16 +03:00
|
|
|
<div>
|
2018-04-12 01:58:04 +03:00
|
|
|
<div className="mx_DevTools_label_left">{ _t('Toolbox') }</div>
|
|
|
|
<div className="mx_DevTools_label_right">Room ID: { this.props.roomId }</div>
|
|
|
|
<div className="mx_DevTools_label_bottom" />
|
|
|
|
|
|
|
|
<div className="mx_Dialog_content">
|
|
|
|
{ Entries.map((Entry) => {
|
|
|
|
const label = Entry.getLabel();
|
|
|
|
const onClick = this._setMode(Entry);
|
|
|
|
return <button className={classes} key={label} onClick={onClick}>{ label }</button>;
|
|
|
|
}) }
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
<div className="mx_Dialog_buttons">
|
|
|
|
<button onClick={this.onCancel}>{ _t('Cancel') }</button>
|
|
|
|
</div>
|
2019-12-17 20:26:12 +03:00
|
|
|
</React.Fragment>;
|
2018-04-12 01:58:04 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
const BaseDialog = sdk.getComponent('views.dialogs.BaseDialog');
|
|
|
|
return (
|
|
|
|
<BaseDialog className="mx_QuestionDialog" onFinished={this.props.onFinished} title={_t('Developer Tools')}>
|
|
|
|
{ body }
|
|
|
|
</BaseDialog>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|