primevue-mirror/pages/validation/index.vue

573 lines
23 KiB
Vue

<template>
<div>
<div class="content-section introduction">
<div class="feature-intro">
<h1>Vuelidate</h1>
<p>PrimeVue components can be easily used/integrated with <a href="https://vuelidate-next.netlify.app/">Vuelidate</a>. In this example, a register panel is simulated using Vuelidate.</p>
</div>
<AppDemoActions />
</div>
<div class="content-section implementation form-demo">
<Dialog v-model:visible="showMessage" :breakpoints="{ '960px': '80vw' }" :style="{ width: '30vw' }" position="top">
<div class="flex align-items-center flex-column pt-6 px-3">
<i class="pi pi-check-circle" :style="{ fontSize: '5rem', color: 'var(--green-500)' }"></i>
<h5>Registration Successful!</h5>
<p :style="{ lineHeight: 1.5, textIndent: '1rem' }">
Your account is registered under name <b>{{ name }}</b> ; it'll be valid next 30 days without activation. Please check <b>{{ email }}</b> for activation instructions.
</p>
</div>
<template #footer>
<div class="flex justify-content-center">
<Button label="OK" @click="toggleDialog" class="p-button-text" />
</div>
</template>
</Dialog>
<div class="flex justify-content-center">
<div class="card">
<h5 class="text-center">Register</h5>
<form @submit.prevent="handleSubmit(!v$.$invalid)" class="p-fluid">
<div class="field">
<div class="p-float-label">
<InputText id="name" v-model="v$.name.$model" :class="{ 'p-invalid': v$.name.$invalid && submitted }" />
<label for="name" :class="{ 'p-error': v$.name.$invalid && submitted }">Name*</label>
</div>
<small v-if="(v$.name.$invalid && submitted) || v$.name.$pending.$response" class="p-error">{{ v$.name.required.$message.replace('Value', 'Name') }}</small>
</div>
<div class="field">
<div class="p-float-label p-input-icon-right">
<i class="pi pi-envelope" />
<InputText id="email" v-model="v$.email.$model" :class="{ 'p-invalid': v$.email.$invalid && submitted }" aria-describedby="email-error" />
<label for="email" :class="{ 'p-error': v$.email.$invalid && submitted }">Email*</label>
</div>
<span v-if="v$.email.$error && submitted">
<span id="email-error" v-for="(error, index) of v$.email.$errors" :key="index">
<small class="p-error">{{ error.$message }}</small>
</span>
</span>
<small v-else-if="(v$.email.$invalid && submitted) || v$.email.$pending.$response" class="p-error">{{ v$.email.required.$message.replace('Value', 'Email') }}</small>
</div>
<div class="field">
<div class="p-float-label">
<Password id="password" v-model="v$.password.$model" :class="{ 'p-invalid': v$.password.$invalid && submitted }" toggleMask>
<template #header>
<h6>Pick a password</h6>
</template>
<template #footer="sp">
{{ sp.level }}
<Divider />
<p class="mt-2">Suggestions</p>
<ul class="pl-2 ml-2 mt-0" style="line-height: 1.5">
<li>At least one lowercase</li>
<li>At least one uppercase</li>
<li>At least one numeric</li>
<li>Minimum 8 characters</li>
</ul>
</template>
</Password>
<label for="password" :class="{ 'p-error': v$.password.$invalid && submitted }">Password*</label>
</div>
<small v-if="(v$.password.$invalid && submitted) || v$.password.$pending.$response" class="p-error">{{ v$.password.required.$message.replace('Value', 'Password') }}</small>
</div>
<div class="field">
<div class="p-float-label">
<Calendar id="date" v-model="date" :showIcon="true" />
<label for="date">Birthday</label>
</div>
</div>
<div class="field">
<div class="p-float-label">
<Dropdown id="country" v-model="country" :options="countries" optionLabel="name" />
<label for="country">Country</label>
</div>
</div>
<div class="field-checkbox">
<Checkbox id="accept" name="accept" value="Accept" v-model="v$.accept.$model" :class="{ 'p-invalid': v$.accept.$invalid && submitted }" />
<label for="accept" :class="{ 'p-error': v$.accept.$invalid && submitted }">I agree to the terms and conditions*</label>
</div>
<Button type="submit" label="Submit" class="mt-2" />
</form>
</div>
</div>
</div>
<ClientOnly>
<AppDoc
name="VuelidateFormDemo"
:sources="sources"
:service="['CountryService']"
:data="['countries']"
github="validation/VuelidateFormDemo.vue"
:dependencies="{ '@vuelidate/core': '^2.0.0-alpha.14', '@vuelidate/validators': '^2.0.0-alpha.12' }"
/>
</ClientOnly>
</div>
</template>
<script>
import { email, required } from '@vuelidate/validators';
import { useVuelidate } from '@vuelidate/core';
import CountryService from '../../service/CountryService';
export default {
setup: () => ({ v$: useVuelidate() }),
data() {
return {
name: '',
email: '',
password: '',
date: null,
country: null,
accept: null,
submitted: false,
countries: null,
showMessage: false,
sources: {
'options-api': {
tabName: 'Options API Source',
content: `
<template>
<div class="form-demo">
<Dialog v-model:visible="showMessage" :breakpoints="{ '960px': '80vw' }" :style="{ width: '30vw' }" position="top">
<div class="flex align-items-center flex-column pt-6 px-3">
<i class="pi pi-check-circle" :style="{fontSize: '5rem', color: 'var(--green-500)' }"></i>
<h5>Registration Successful!</h5>
<p :style="{lineHeight: 1.5, textIndent: '1rem'}">
Your account is registered under name <b>{{name}}</b> ; it'll be valid next 30 days without activation. Please check <b>{{email}}</b> for activation instructions.
</p>
</div>
<template #footer>
<div class="flex justify-content-center">
<Button label="OK" @click="toggleDialog" class="p-button-text" />
</div>
</template>
</Dialog>
<div class="flex justify-content-center">
<div class="card">
<h5 class="text-center">Register</h5>
<form @submit.prevent="handleSubmit(!v$.$invalid)" class="p-fluid">
<div class="field">
<div class="p-float-label">
<InputText id="name" v-model="v$.name.$model" :class="{'p-invalid':v$.name.$invalid && submitted}" />
<label for="name" :class="{'p-error':v$.name.$invalid && submitted}">Name*</label>
</div>
<small v-if="(v$.name.$invalid && submitted) || v$.name.$pending.$response" class="p-error">{{v$.name.required.$message.replace('Value', 'Name')}}</small>
</div>
<div class="field">
<div class="p-float-label p-input-icon-right">
<i class="pi pi-envelope" />
<InputText id="email" v-model="v$.email.$model" :class="{'p-invalid':v$.email.$invalid && submitted}" aria-describedby="email-error"/>
<label for="email" :class="{'p-error':v$.email.$invalid && submitted}">Email*</label>
</div>
<span v-if="v$.email.$error && submitted">
<span id="email-error" v-for="(error, index) of v$.email.$errors" :key="index">
<small class="p-error">{{error.$message}}</small>
</span>
</span>
<small v-else-if="(v$.email.$invalid && submitted) || v$.email.$pending.$response" class="p-error">{{v$.email.required.$message.replace('Value', 'Email')}}</small>
</div>
<div class="field">
<div class="p-float-label">
<Password id="password" v-model="v$.password.$model" :class="{'p-invalid':v$.password.$invalid && submitted}" toggleMask>
<template #header>
<h6>Pick a password</h6>
</template>
<template #footer="sp">
{{sp.level}}
<Divider />
<p class="mt-2">Suggestions</p>
<ul class="pl-2 ml-2 mt-0" style="line-height: 1.5">
<li>At least one lowercase</li>
<li>At least one uppercase</li>
<li>At least one numeric</li>
<li>Minimum 8 characters</li>
</ul>
</template>
</Password>
<label for="password" :class="{'p-error':v$.password.$invalid && submitted}">Password*</label>
</div>
<small v-if="(v$.password.$invalid && submitted) || v$.password.$pending.$response" class="p-error">{{v$.password.required.$message.replace('Value', 'Password')}}</small>
</div>
<div class="field">
<div class="p-float-label">
<Calendar id="date" v-model="date" :showIcon="true" />
<label for="date">Birthday</label>
</div>
</div>
<div class="field">
<div class="p-float-label">
<Dropdown id="country" v-model="country" :options="countries" optionLabel="name" />
<label for="country">Country</label>
</div>
</div>
<div class="field-checkbox">
<Checkbox id="accept" name="accept" value="Accept" v-model="v$.accept.$model" :class="{'p-invalid':v$.accept.$invalid && submitted}" />
<label for="accept" :class="{'p-error': v$.accept.$invalid && submitted}">I agree to the terms and conditions*</label>
</div>
<Button type="submit" label="Submit" class="mt-2" />
</form>
</div>
</div>
</div>
</template>
<script>
import { email, required } from "@vuelidate/validators";
import { useVuelidate } from "@vuelidate/core";
import CountryService from './service/CountryService';
export default {
setup: () => ({ v$: useVuelidate() }),
data() {
return {
name: '',
email: '',
password: '',
date: null,
country: null,
accept: null,
submitted: false,
countries: null,
showMessage: false
}
},
countryService: null,
validations() {
return {
name: {
required
},
email: {
required,
email
},
password: {
required
},
accept: {
required
}
}
},
created() {
this.countryService = new CountryService();
},
mounted() {
this.countryService.getCountries().then(data => this.countries = data);
},
methods: {
handleSubmit(isFormValid) {
this.submitted = true;
if (!isFormValid) {
return;
}
this.toggleDialog();
},
toggleDialog() {
this.showMessage = !this.showMessage;
if(!this.showMessage) {
this.resetForm();
}
},
resetForm() {
this.name = '';
this.email = '';
this.password = '';
this.date = null;
this.country = null;
this.accept = null;
this.submitted = false;
}
}
}
<\\/script>
<style lang="scss" scoped>
.form-demo {
.card {
min-width: 450px;
form {
margin-top: 2rem;
}
.field {
margin-bottom: 1.5rem;
}
}
@media screen and (max-width: 960px) {
.card {
width: 80%;
}
}
}
</style>
`
},
'composition-api': {
tabName: 'Composition API Source',
content: `
<template>
<div class="form-demo">
<Dialog v-model:visible="showMessage" :breakpoints="{ '960px': '80vw' }" :style="{ width: '30vw' }" position="top">
<div class="flex align-items-center flex-column pt-6 px-3">
<i class="pi pi-check-circle" :style="{fontSize: '5rem', color: 'var(--green-500)' }"></i>
<h5>Registration Successful!</h5>
<p :style="{lineHeight: 1.5, textIndent: '1rem'}">
Your account is registered under name <b>{{state.name}}</b> ; it'll be valid next 30 days without activation. Please check <b>{{state.email}}</b> for activation instructions.
</p>
</div>
<template #footer>
<div class="flex justify-content-center">
<Button label="OK" @click="toggleDialog" class="p-button-text" />
</div>
</template>
</Dialog>
<div class="flex justify-content-center">
<div class="card">
<h5 class="text-center">Register</h5>
<form @submit.prevent="handleSubmit(!v$.$invalid)" class="p-fluid">
<div class="field">
<div class="p-float-label">
<InputText id="name" v-model="v$.name.$model" :class="{'p-invalid':v$.name.$invalid && submitted}" />
<label for="name" :class="{'p-error':v$.name.$invalid && submitted}">Name*</label>
</div>
<small v-if="(v$.name.$invalid && submitted) || v$.name.$pending.$response" class="p-error">{{v$.name.required.$message.replace('Value', 'Name')}}</small>
</div>
<div class="field">
<div class="p-float-label p-input-icon-right">
<i class="pi pi-envelope" />
<InputText id="email" v-model="v$.email.$model" :class="{'p-invalid':v$.email.$invalid && submitted}" aria-describedby="email-error"/>
<label for="email" :class="{'p-error':v$.email.$invalid && submitted}">Email*</label>
</div>
<span v-if="v$.email.$error && submitted">
<span id="email-error" v-for="(error, index) of v$.email.$errors" :key="index">
<small class="p-error">{{error.$message}}</small>
</span>
</span>
<small v-else-if="(v$.email.$invalid && submitted) || v$.email.$pending.$response" class="p-error">{{v$.email.required.$message.replace('Value', 'Email')}}</small>
</div>
<div class="field">
<div class="p-float-label">
<Password id="password" v-model="v$.password.$model" :class="{'p-invalid':v$.password.$invalid && submitted}" toggleMask>
<template #header>
<h6>Pick a password</h6>
</template>
<template #footer="sp">
{{sp.level}}
<Divider />
<p class="mt-2">Suggestions</p>
<ul class="pl-2 ml-2 mt-0" style="line-height: 1.5">
<li>At least one lowercase</li>
<li>At least one uppercase</li>
<li>At least one numeric</li>
<li>Minimum 8 characters</li>
</ul>
</template>
</Password>
<label for="password" :class="{'p-error':v$.password.$invalid && submitted}">Password*</label>
</div>
<small v-if="(v$.password.$invalid && submitted) || v$.password.$pending.$response" class="p-error">{{v$.password.required.$message.replace('Value', 'Password')}}</small>
</div>
<div class="field">
<div class="p-float-label">
<Calendar id="date" v-model="date" :showIcon="true" />
<label for="date">Birthday</label>
</div>
</div>
<div class="field">
<div class="p-float-label">
<Dropdown id="country" v-model="country" :options="countries" optionLabel="name" />
<label for="country">Country</label>
</div>
</div>
<div class="field-checkbox">
<Checkbox id="accept" name="accept" value="Accept" v-model="v$.accept.$model" :class="{'p-invalid':v$.accept.$invalid && submitted}" />
<label for="accept" :class="{'p-error': v$.accept.$invalid && submitted}">I agree to the terms and conditions*</label>
</div>
<Button type="submit" label="Submit" class="mt-2" />
</form>
</div>
</div>
</div>
</template>
<script>
import { reactive, ref, onMounted } from 'vue';
import { email, required } from "@vuelidate/validators";
import { useVuelidate } from "@vuelidate/core";
import CountryService from './service/CountryService';
export default {
setup() {
onMounted(() => {
countryService.value.getCountries().then(data => countries.value = data);
})
const state = reactive({
name: '',
email: '',
password: '',
accept: null
});
const rules = {
name: { required },
email: { required, email },
password: { required },
accept: { required }
};
const countryService = ref(new CountryService());
const submitted = ref(false);
const countries = ref();
const showMessage = ref(false);
const date = ref();
const country = ref();
const v$ = useVuelidate(rules, state);
const handleSubmit = (isFormValid) => {
submitted.value = true;
if (!isFormValid) {
return;
}
toggleDialog();
}
const toggleDialog = () => {
showMessage.value = !showMessage.value;
if(!showMessage.value) {
resetForm();
}
}
const resetForm = () => {
state.name = '';
state.email = '';
state.password = '';
state.date = null;
state.country = null;
state.accept = null;
submitted.value = false;
}
return { state, v$, handleSubmit, toggleDialog, submitted, countries, showMessage, date, country }
}
}
<\\/script>
<style lang="scss" scoped>
.form-demo {
.card {
min-width: 450px;
form {
margin-top: 2rem;
}
.field {
margin-bottom: 1.5rem;
}
}
@media screen and (max-width: 960px) {
.card {
width: 80%;
}
}
}
</style>
`
}
}
};
},
countryService: null,
validations() {
return {
name: {
required
},
email: {
required,
email
},
password: {
required
},
accept: {
required
}
};
},
created() {
this.countryService = new CountryService();
},
mounted() {
this.countryService.getCountries().then((data) => (this.countries = data));
},
methods: {
handleSubmit(isFormValid) {
this.submitted = true;
if (!isFormValid) {
return;
}
this.toggleDialog();
},
toggleDialog() {
this.showMessage = !this.showMessage;
if (!this.showMessage) {
this.resetForm();
}
},
resetForm() {
this.name = '';
this.email = '';
this.password = '';
this.date = null;
this.country = null;
this.accept = null;
this.submitted = false;
}
}
};
</script>
<style lang="scss" scoped>
.form-demo {
.card {
min-width: 450px;
form {
margin-top: 2rem;
}
.field {
margin-bottom: 1.5rem;
}
}
@media screen and (max-width: 960px) {
.card {
width: 80%;
}
}
}
</style>