2019-04-16 18:52:31 +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 classNames from 'classnames';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates a validation function from a set of rules describing what to validate.
|
|
|
|
*
|
2019-04-17 13:39:11 +03:00
|
|
|
* @param {Function} description
|
|
|
|
* Function that returns a string summary of the kind of value that will
|
|
|
|
* meet the validation rules. Shown at the top of the validation feedback.
|
2019-04-16 18:52:31 +03:00
|
|
|
* @param {Object} rules
|
|
|
|
* An array of rules describing how to check to input value. Each rule in an object
|
|
|
|
* and may have the following properties:
|
|
|
|
* - `key`: A unique ID for the rule. Required.
|
2019-04-18 23:33:37 +03:00
|
|
|
* - `test`: A function used to determine the rule's current validity. Required.
|
2019-04-17 13:39:11 +03:00
|
|
|
* - `valid`: Function returning text to show when the rule is valid. Only shown if set.
|
|
|
|
* - `invalid`: Function returning text to show when the rule is invalid. Only shown if set.
|
2019-04-16 18:52:31 +03:00
|
|
|
* @returns {Function}
|
|
|
|
* A validation function that takes in the current input value and returns
|
|
|
|
* the overall validity and a feedback UI that can be rendered for more detail.
|
|
|
|
*/
|
|
|
|
export default function withValidation({ description, rules }) {
|
2019-04-18 23:33:37 +03:00
|
|
|
return function onValidate({ value, focused, allowEmpty = true }) {
|
2019-04-16 18:52:31 +03:00
|
|
|
// TODO: Re-run only after ~200ms of inactivity
|
2019-04-18 23:33:37 +03:00
|
|
|
if (!value && allowEmpty) {
|
2019-04-16 18:52:31 +03:00
|
|
|
return {
|
|
|
|
valid: null,
|
|
|
|
feedback: null,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
const results = [];
|
|
|
|
let valid = true;
|
|
|
|
if (rules && rules.length) {
|
|
|
|
for (const rule of rules) {
|
2019-04-18 23:33:37 +03:00
|
|
|
if (!rule.key || !rule.test) {
|
2019-04-16 18:52:31 +03:00
|
|
|
continue;
|
|
|
|
}
|
2019-04-18 23:33:37 +03:00
|
|
|
const ruleValid = rule.test({ value, allowEmpty });
|
2019-04-16 18:52:31 +03:00
|
|
|
valid = valid && ruleValid;
|
|
|
|
if (ruleValid && rule.valid) {
|
|
|
|
// If the rule's result is valid and has text to show for
|
|
|
|
// the valid state, show it.
|
|
|
|
results.push({
|
|
|
|
key: rule.key,
|
|
|
|
valid: true,
|
2019-04-17 13:39:11 +03:00
|
|
|
text: rule.valid(),
|
2019-04-16 18:52:31 +03:00
|
|
|
});
|
|
|
|
} else if (!ruleValid && rule.invalid) {
|
|
|
|
// If the rule's result is invalid and has text to show for
|
|
|
|
// the invalid state, show it.
|
|
|
|
results.push({
|
|
|
|
key: rule.key,
|
|
|
|
valid: false,
|
2019-04-17 13:39:11 +03:00
|
|
|
text: rule.invalid(),
|
2019-04-16 18:52:31 +03:00
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-16 20:12:13 +03:00
|
|
|
// Hide feedback when not focused
|
|
|
|
if (!focused) {
|
|
|
|
return {
|
|
|
|
valid,
|
|
|
|
feedback: null,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2019-04-16 18:52:31 +03:00
|
|
|
let details;
|
|
|
|
if (results && results.length) {
|
|
|
|
details = <ul className="mx_Validation_details">
|
|
|
|
{results.map(result => {
|
|
|
|
const classes = classNames({
|
|
|
|
"mx_Validation_detail": true,
|
|
|
|
"mx_Validation_valid": result.valid,
|
|
|
|
"mx_Validation_invalid": !result.valid,
|
|
|
|
});
|
|
|
|
return <li key={result.key} className={classes}>
|
|
|
|
{result.text}
|
|
|
|
</li>;
|
|
|
|
})}
|
|
|
|
</ul>;
|
|
|
|
}
|
|
|
|
|
2019-04-17 13:39:11 +03:00
|
|
|
let summary;
|
|
|
|
if (description) {
|
|
|
|
summary = <div className="mx_Validation_description">{description()}</div>;
|
|
|
|
}
|
|
|
|
|
2019-04-16 18:52:31 +03:00
|
|
|
const feedback = <div className="mx_Validation">
|
2019-04-17 13:39:11 +03:00
|
|
|
{summary}
|
2019-04-16 18:52:31 +03:00
|
|
|
{details}
|
|
|
|
</div>;
|
|
|
|
|
|
|
|
return {
|
|
|
|
valid,
|
|
|
|
feedback,
|
|
|
|
};
|
|
|
|
};
|
|
|
|
}
|