it-tools/src/composable/validation.ts

39 lines
784 B
TypeScript
Raw Normal View History

2022-04-11 22:47:05 +02:00
import { reactive, watch, type Ref } from 'vue';
type UseValidationRule<T> = {
2022-04-11 23:08:50 +02:00
validator: (value: T) => boolean;
message: string;
};
2022-04-11 22:47:05 +02:00
2022-05-09 17:40:29 +02:00
function isFalsyOrHasThrown(cb: () => boolean) {
try {
return !cb();
} catch (_) {
return true;
}
}
2022-04-11 22:47:05 +02:00
export function useValidation<T>({ source, rules }: { source: Ref<T>; rules: UseValidationRule<T>[] }) {
const state = reactive<{
2022-04-11 23:08:50 +02:00
message: string;
status: undefined | 'error';
2022-04-11 22:47:05 +02:00
}>({
message: '',
2022-04-11 23:08:50 +02:00
status: undefined,
});
2022-04-11 22:47:05 +02:00
watch([source], () => {
2022-04-11 23:08:50 +02:00
state.message = '';
state.status = undefined;
for (const rule of rules) {
2022-05-09 17:40:29 +02:00
if (isFalsyOrHasThrown(() => rule.validator(source.value))) {
2022-04-11 23:08:50 +02:00
state.message = rule.message;
state.status = 'error';
2022-04-11 22:47:05 +02:00
}
}
2022-04-11 23:08:50 +02:00
});
2022-04-11 22:47:05 +02:00
return state;
}