mirror of
https://github.com/bitwarden/browser
synced 2025-12-18 09:13:33 +00:00
* rough-in passphrase validation failure handling * trigger valid change from settings * fix `max` constraint enforcement * add taps for generator validation monitoring/debugging * HTML constraints validation rises like a phoenix * remove min/max boundaries to fix chrome display issue * bind settings components as view children of options components * remove defunct `okSettings$` * extend validationless generator to passwords * extend validationless generator to catchall emails * extend validationless generator to forwarder emails * extend validationless generator to subaddress emails * extend validationless generator to usernames * fix observable cycle * disable generate button when no algorithm is selected * prevent duplicate algorithm emissions * add constraints that assign email address defaults
70 lines
2.0 KiB
TypeScript
70 lines
2.0 KiB
TypeScript
import { ValidatorFn, Validators } from "@angular/forms";
|
|
import { distinctUntilChanged, map, pairwise, pipe, skipWhile, startWith, takeWhile } from "rxjs";
|
|
|
|
import { AnyConstraint, Constraints } from "@bitwarden/common/tools/types";
|
|
import { UserId } from "@bitwarden/common/types/guid";
|
|
import { CredentialGeneratorConfiguration } from "@bitwarden/generator-core";
|
|
|
|
export function completeOnAccountSwitch() {
|
|
return pipe(
|
|
map(({ id }: { id: UserId | null }) => id),
|
|
skipWhile((id) => !id),
|
|
startWith(null as UserId),
|
|
pairwise(),
|
|
takeWhile(([prev, next]) => (prev ?? next) === next),
|
|
map(([_, id]) => id),
|
|
distinctUntilChanged(),
|
|
);
|
|
}
|
|
|
|
export function toValidators<Policy, Settings>(
|
|
target: keyof Settings,
|
|
configuration: CredentialGeneratorConfiguration<Settings, Policy>,
|
|
policy?: Constraints<Settings>,
|
|
) {
|
|
const validators: Array<ValidatorFn> = [];
|
|
|
|
// widen the types to avoid typecheck issues
|
|
const config: AnyConstraint = configuration.settings.constraints[target];
|
|
const runtime: AnyConstraint = policy[target];
|
|
|
|
const required = getConstraint("required", config, runtime) ?? false;
|
|
if (required) {
|
|
validators.push(Validators.required);
|
|
}
|
|
|
|
const maxLength = getConstraint("maxLength", config, runtime);
|
|
if (maxLength !== undefined) {
|
|
validators.push(Validators.maxLength(maxLength));
|
|
}
|
|
|
|
const minLength = getConstraint("minLength", config, runtime);
|
|
if (minLength !== undefined) {
|
|
validators.push(Validators.minLength(config.minLength));
|
|
}
|
|
|
|
const min = getConstraint("min", config, runtime);
|
|
if (min !== undefined) {
|
|
validators.push(Validators.min(min));
|
|
}
|
|
|
|
const max = getConstraint("max", config, runtime);
|
|
if (max !== undefined) {
|
|
validators.push(Validators.max(max));
|
|
}
|
|
|
|
return validators;
|
|
}
|
|
|
|
function getConstraint<Key extends keyof AnyConstraint>(
|
|
key: Key,
|
|
config: AnyConstraint,
|
|
policy?: AnyConstraint,
|
|
) {
|
|
if (policy && key in policy) {
|
|
return policy[key] ?? config[key];
|
|
} else if (config && key in config) {
|
|
return config[key];
|
|
}
|
|
}
|