<script lang="ts" setup> import { computed, ref, watch } from 'vue' import { useUserStore } from '@/stores/userStore' import ToolTip from '@/components/ToolTip.vue' const firstname = ref<string>('') const lastname = ref<string>('') const email = ref<string>('') const username = ref<string>('') const password = ref<string>('') const confirm = ref<string>('') const showPassword = ref<boolean>(false) const errorMessage = ref<string>('') const userStore = useUserStore() const nameRegex = /^[æÆøØåÅa-zA-Z,.'-][æÆøØåÅa-zA-Z ,.'-]{0,29}$/ const emailRegex = /^[æÆøØåÅa-zA-Z0-9_+&*-]+(?:\.[æÆøØåÅa-zA-Z0-9_+&*-]+)*@(?:[æÆøØåÅa-zA-Z0-9-]+\.)+[æÆøØåÅa-zA-Z]{2,7}$/ const usernameRegex = /^[ÆØÅæøåA-Za-z][æÆøØåÅA-Za-z0-9_]{2,29}$/ const passwordRegex = /^(?=.*[0-9])(?=.*[a-zæøå])(?=.*[ÆØÅA-Z])(?=.*[@#$%^&+=!])(?=\S+$).{8,30}$/ const isFirstNameValid = computed(() => nameRegex.test(firstname.value) && firstname.value) const isLastNameValid = computed(() => nameRegex.test(lastname.value) && lastname.value) const isEmailValid = computed(() => emailRegex.test(email.value)) const isUsernameValid = computed(() => usernameRegex.test(username.value)) const isPasswordValid = computed(() => passwordRegex.test(password.value)) const isFormInvalid = computed( () => [isFirstNameValid, isLastNameValid, isEmailValid, isUsernameValid, isPasswordValid].some( (v) => !v.value ) || password.value !== confirm.value ) const submitForm = () => { userStore.register(firstname.value, lastname.value, email.value, username.value, password.value) } const toggleShowPassword = () => { showPassword.value = !showPassword.value } watch( () => userStore.errorMessage, (newValue: string) => { errorMessage.value = newValue } ) </script> <template> <div class="flex flex-col justify-center gap-5 w-full"> <div class="flex flex-col"> <div class="flex flex-row justify-between mx-4"> <p>Fornavn*</p> <ToolTip :message="'Must include only letters, spaces, commas, apostrophes, periods, and hyphens. 1-30 characters long'" /> </div> <input v-model="firstname" name="firstname" :class="{ 'border-2 border-lime-400': isFirstNameValid }" placeholder="Skriv inn fornavn" type="text" /> </div> <div class="flex flex-col"> <div class="flex flex-row justify-between mx-4"> <p>Etternavn*</p> <ToolTip :message="'Must include only letters, spaces, commas, apostrophes, periods, and hyphens. 1-30 characters long'" /> </div> <input v-model="lastname" name="lastname" :class="{ 'border-2 border-lime-400': isLastNameValid }" placeholder="Skriv inn etternavn" type="text" /> </div> <div class="flex flex-col"> <div class="flex flex-row justify-between mx-4"> <p>E-post*</p> <ToolTip :message="'Valid email: Starts with Norwegian letters, numbers, or special characters. Includes \@\ followed by a domain. Ends with 2-7 letters.'" /> </div> <input v-model="email" name="email" :class="{ 'border-2 border-lime-400': isEmailValid }" placeholder="Skriv inn e-post" type="text" /> </div> <div class="flex flex-col"> <div class="flex flex-row justify-between mx-4"> <p>Brukernavn*</p> <ToolTip :message="'Must start with a letter and can include numbers and underscores. 3-30 characters long.'" /> </div> <input v-model="username" name="username" placeholder="Skriv inn brukernavn" type="text" :class="{ 'border-2 border-lime-400': isUsernameValid }" /> </div> <div class="flex flex-col"> <div class="flex flex-row justify-between mx-4"> <p>Passord*</p> <ToolTip :message="'Must be at least 8 characters, including at least one number, one lowercase letter, one uppercase letter, one special character (@#$%^&+=!), and no spaces.'" /> </div> <div class="relative"> <input name="password" v-model="password" :type="showPassword ? 'text' : 'password'" placeholder="Skriv inn passord" class="w-full" :class="{ 'border-2 border-lime-400': isPasswordValid }" /> <button class="absolute right-0 top-1 bg-transparent hover:bg-transparent mr-4 mt-1" @click="toggleShowPassword" > {{ showPassword ? '🔓' : '🔒' }} </button> </div> <input v-model="confirm" :class="{ 'border-2 border-lime-400': password == confirm && '' !== confirm.valueOf() }" class="mt-2" name="confirm" placeholder="Bekreft passord" type="password" /> </div> <div class="flex flex-row gap-5"> <button :disabled="isFormInvalid" class="grow-0 primary" name="submit" @click="submitForm" > Registrer deg </button> <p>{{ errorMessage }}</p> </div> </div> </template>