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;
|
|
|
|
}
|