2019-09-20 18:45:14 +03:00
|
|
|
/*
|
|
|
|
Copyright 2019 New Vector Ltd
|
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
import { _t } from '../../../languageHandler';
|
|
|
|
import React from 'react';
|
|
|
|
import PropTypes from 'prop-types';
|
2019-12-20 04:19:56 +03:00
|
|
|
import * as sdk from '../../../index';
|
2019-09-20 18:45:14 +03:00
|
|
|
import withValidation from './Validation';
|
2019-12-21 00:13:46 +03:00
|
|
|
import {MatrixClientPeg} from '../../../MatrixClientPeg';
|
2019-09-20 18:45:14 +03:00
|
|
|
|
2019-12-26 21:52:57 +03:00
|
|
|
// Controlled form component wrapping Field for inputting a room alias scoped to a given domain
|
2019-09-20 18:45:14 +03:00
|
|
|
export default class RoomAliasField extends React.PureComponent {
|
|
|
|
static propTypes = {
|
|
|
|
id: PropTypes.string.isRequired,
|
|
|
|
domain: PropTypes.string.isRequired,
|
|
|
|
onChange: PropTypes.func,
|
2019-12-26 21:52:57 +03:00
|
|
|
value: PropTypes.string.isRequired,
|
2019-09-20 18:45:14 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
constructor(props) {
|
|
|
|
super(props);
|
|
|
|
this.state = {isValid: true};
|
|
|
|
}
|
|
|
|
|
|
|
|
_asFullAlias(localpart) {
|
|
|
|
return `#${localpart}:${this.props.domain}`;
|
|
|
|
}
|
|
|
|
|
2020-02-29 03:36:01 +03:00
|
|
|
_isValid(value) {
|
|
|
|
const fullAlias = this._asFullAlias(value);
|
|
|
|
// XXX: FIXME https://github.com/matrix-org/matrix-doc/issues/668
|
|
|
|
return !value.includes("#") && !value.includes(":") && !value.includes(",") &&
|
|
|
|
encodeURI(fullAlias) === fullAlias;
|
|
|
|
}
|
|
|
|
|
2019-09-20 18:45:14 +03:00
|
|
|
render() {
|
|
|
|
const Field = sdk.getComponent('views.elements.Field');
|
|
|
|
const poundSign = (<span>#</span>);
|
|
|
|
const aliasPostfix = ":" + this.props.domain;
|
|
|
|
const domain = (<span title={aliasPostfix}>{aliasPostfix}</span>);
|
|
|
|
const maxlength = 255 - this.props.domain.length - 2; // 2 for # and :
|
|
|
|
return (
|
|
|
|
<Field
|
|
|
|
label={_t("Room alias")}
|
|
|
|
className="mx_RoomAliasField"
|
|
|
|
prefix={poundSign}
|
|
|
|
postfix={domain}
|
|
|
|
id={this.props.id}
|
|
|
|
ref={ref => this._fieldRef = ref}
|
|
|
|
onValidate={this._onValidate}
|
|
|
|
placeholder={_t("e.g. my-room")}
|
|
|
|
onChange={this._onChange}
|
2019-12-26 21:52:57 +03:00
|
|
|
value={this.props.value.substring(1, this.props.value.length - this.props.domain.length - 1)}
|
2019-09-20 18:45:14 +03:00
|
|
|
maxLength={maxlength} />
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
_onChange = (ev) => {
|
|
|
|
if (this.props.onChange) {
|
|
|
|
this.props.onChange(this._asFullAlias(ev.target.value));
|
|
|
|
}
|
2019-12-26 21:52:57 +03:00
|
|
|
};
|
2019-09-20 18:45:14 +03:00
|
|
|
|
|
|
|
_onValidate = async (fieldState) => {
|
|
|
|
const result = await this._validationRules(fieldState);
|
|
|
|
this.setState({isValid: result.valid});
|
|
|
|
return result;
|
|
|
|
};
|
|
|
|
|
|
|
|
_validationRules = withValidation({
|
|
|
|
rules: [
|
|
|
|
{
|
|
|
|
key: "safeLocalpart",
|
|
|
|
test: async ({ value }) => {
|
|
|
|
if (!value) {
|
|
|
|
return true;
|
|
|
|
}
|
2020-02-29 03:36:01 +03:00
|
|
|
return this._isValid(value);
|
2019-09-20 18:45:14 +03:00
|
|
|
},
|
|
|
|
invalid: () => _t("Some characters not allowed"),
|
|
|
|
}, {
|
|
|
|
key: "required",
|
|
|
|
test: async ({ value, allowEmpty }) => allowEmpty || !!value,
|
|
|
|
invalid: () => _t("Please provide a room alias"),
|
|
|
|
}, {
|
|
|
|
key: "taken",
|
2020-02-29 03:36:01 +03:00
|
|
|
skip: ({value}) => !value || !this._isValid(value),
|
2019-09-20 18:45:14 +03:00
|
|
|
test: async ({value}) => {
|
|
|
|
if (!value) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
const client = MatrixClientPeg.get();
|
|
|
|
try {
|
|
|
|
await client.getRoomIdForAlias(this._asFullAlias(value));
|
|
|
|
// we got a room id, so the alias is taken
|
|
|
|
return false;
|
|
|
|
} catch (err) {
|
|
|
|
// any server error code will do,
|
|
|
|
// either it M_NOT_FOUND or the alias is invalid somehow,
|
|
|
|
// in which case we don't want to show the invalid message
|
|
|
|
return !!err.errcode;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
valid: () => _t("This alias is available to use"),
|
|
|
|
invalid: () => _t("This alias is already in use"),
|
|
|
|
},
|
|
|
|
],
|
|
|
|
});
|
|
|
|
|
|
|
|
get isValid() {
|
|
|
|
return this.state.isValid;
|
|
|
|
}
|
|
|
|
|
|
|
|
validate(options) {
|
|
|
|
return this._fieldRef.validate(options);
|
|
|
|
}
|
|
|
|
|
|
|
|
focus() {
|
|
|
|
this._fieldRef.focus();
|
|
|
|
}
|
|
|
|
}
|