reset password form UI

This commit is contained in:
Hayden 2024-04-28 12:05:17 -05:00
parent c83d178182
commit 0c968de1fb
No known key found for this signature in database
GPG key ID: 17CF79474E257545
4 changed files with 245 additions and 172 deletions

View file

@ -0,0 +1,66 @@
<script setup lang="ts">
import MdiGithub from "~icons/mdi/github";
import MdiTwitter from "~icons/mdi/twitter";
import MdiDiscord from "~icons/mdi/discord";
import MdiFolder from "~icons/mdi/folder";
const api = usePublicApi();
const { data: status } = useAsyncData(async () => {
const { data } = await api.status();
return data;
});
</script>
<template>
<div>
<AppToast />
<div class="flex flex-col min-h-screen">
<div class="fill-primary min-w-full absolute top-0 z-[-1]">
<div class="bg-primary flex-col flex min-h-[20vh]" />
<svg
class="fill-primary drop-shadow-xl"
xmlns="http://www.w3.org/2000/svg"
viewBox="0 0 1440 320"
preserveAspectRatio="none"
>
<path
fill-opacity="1"
d="M0,32L80,69.3C160,107,320,181,480,181.3C640,181,800,107,960,117.3C1120,128,1280,224,1360,272L1440,320L1440,0L1360,0C1280,0,1120,0,960,0C800,0,640,0,480,0C320,0,160,0,80,0L0,0Z"
></path>
</svg>
</div>
<div>
<header class="p-4 sm:px-6 lg:p-14 sm:py-6 sm:flex sm:items-end mx-auto">
<div>
<h2 class="mt-1 text-4xl font-bold tracking-tight text-neutral-content sm:text-5xl lg:text-6xl flex">
HomeB
<AppLogo class="w-12 -mb-4" />
x
</h2>
<p class="ml-1 text-lg text-base-content/50">Track, Organize, and Manage your Things.</p>
</div>
<div class="flex mt-6 sm:mt-0 gap-4 ml-auto text-neutral-content">
<a class="tooltip" data-tip="Project Github" href="https://github.com/hay-kot/homebox" target="_blank">
<MdiGithub class="h-8 w-8" />
</a>
<a href="https://twitter.com/haybytes" class="tooltip" data-tip="Follow The Developer" target="_blank">
<MdiTwitter class="h-8 w-8" />
</a>
<a href="https://discord.gg/tuncmNrE4z" class="tooltip" data-tip="Join The Discord" target="_blank">
<MdiDiscord class="h-8 w-8" />
</a>
<a href="https://hay-kot.github.io/homebox/" class="tooltip" data-tip="Read The Docs" target="_blank">
<MdiFolder class="h-8 w-8" />
</a>
</div>
</header>
<div class="grid p-6 sm:place-items-center min-h-[50vh]">
<slot :status="status" />
</div>
</div>
<footer v-if="status" class="mt-auto text-center w-full bottom-0 pb-4">
<p class="text-center text-sm">Version: {{ status.build.version }} ~ Build: {{ status.build.commit }}</p>
</footer>
</div>
</div>
</template>

View file

@ -1,4 +1,3 @@
<script setup lang="ts"></script>
<template> <template>
<div> <div>
<AppToast /> <AppToast />

View file

@ -1,9 +1,5 @@
<script setup lang="ts"> <script setup lang="ts">
import { useRouteHash } from "@vueuse/router"; import { useRouteHash } from "@vueuse/router";
import MdiGithub from "~icons/mdi/github";
import MdiTwitter from "~icons/mdi/twitter";
import MdiDiscord from "~icons/mdi/discord";
import MdiFolder from "~icons/mdi/folder";
import MdiAccount from "~icons/mdi/account"; import MdiAccount from "~icons/mdi/account";
import MdiAccountPlus from "~icons/mdi/account-plus"; import MdiAccountPlus from "~icons/mdi/account-plus";
import MdiLogin from "~icons/mdi/login"; import MdiLogin from "~icons/mdi/login";
@ -40,23 +36,6 @@
const pageForm = useRouteHash(PageForms.Login); const pageForm = useRouteHash(PageForms.Login);
const pageFormStr = computed(() => (pageForm.value[0] === "#" ? pageForm.value.slice(1) : pageForm.value)); const pageFormStr = computed(() => (pageForm.value[0] === "#" ? pageForm.value.slice(1) : pageForm.value));
const { data: status } = useAsyncData(async () => {
const { data } = await api.status();
if (data.demo) {
username.value = "demo@example.com";
password.value = "demo";
}
return data;
});
whenever(status, status => {
if (status?.demo) {
email.value = "demo@example.com";
loginPassword.value = "demo";
}
});
const route = useRoute(); const route = useRoute();
const router = useRouter(); const router = useRouter();
@ -148,47 +127,7 @@
</script> </script>
<template> <template>
<div class="flex flex-col min-h-screen"> <NuxtLayout v-slot="{ status }" name="center-card">
<div class="fill-primary min-w-full absolute top-0 z-[-1]">
<div class="bg-primary flex-col flex min-h-[20vh]" />
<svg
class="fill-primary drop-shadow-xl"
xmlns="http://www.w3.org/2000/svg"
viewBox="0 0 1440 320"
preserveAspectRatio="none"
>
<path
fill-opacity="1"
d="M0,32L80,69.3C160,107,320,181,480,181.3C640,181,800,107,960,117.3C1120,128,1280,224,1360,272L1440,320L1440,0L1360,0C1280,0,1120,0,960,0C800,0,640,0,480,0C320,0,160,0,80,0L0,0Z"
></path>
</svg>
</div>
<div>
<header class="p-4 sm:px-6 lg:p-14 sm:py-6 sm:flex sm:items-end mx-auto">
<div>
<h2 class="mt-1 text-4xl font-bold tracking-tight text-neutral-content sm:text-5xl lg:text-6xl flex">
HomeB
<AppLogo class="w-12 -mb-4" />
x
</h2>
<p class="ml-1 text-lg text-base-content/50">Track, Organize, and Manage your Things.</p>
</div>
<div class="flex mt-6 sm:mt-0 gap-4 ml-auto text-neutral-content">
<a class="tooltip" data-tip="Project Github" href="https://github.com/hay-kot/homebox" target="_blank">
<MdiGithub class="h-8 w-8" />
</a>
<a href="https://twitter.com/haybytes" class="tooltip" data-tip="Follow The Developer" target="_blank">
<MdiTwitter class="h-8 w-8" />
</a>
<a href="https://discord.gg/tuncmNrE4z" class="tooltip" data-tip="Join The Discord" target="_blank">
<MdiDiscord class="h-8 w-8" />
</a>
<a href="https://hay-kot.github.io/homebox/" class="tooltip" data-tip="Read The Docs" target="_blank">
<MdiFolder class="h-8 w-8" />
</a>
</div>
</header>
<div class="grid p-6 sm:place-items-center min-h-[50vh]">
<div> <div>
<Transition name="slide-fade"> <Transition name="slide-fade">
<form v-if="pageFormStr === PageForms.Register" @submit.prevent="registerUser"> <form v-if="pageFormStr === PageForms.Register" @submit.prevent="registerUser">
@ -298,12 +237,7 @@
</NuxtLink> </NuxtLink>
</div> </div>
</div> </div>
</div> </NuxtLayout>
</div>
<footer v-if="status" class="mt-auto text-center w-full bottom-0 pb-4">
<p class="text-center text-sm">Version: {{ status.build.version }} ~ Build: {{ status.build.commit }}</p>
</footer>
</div>
</template> </template>
<style lang="css" scoped> <style lang="css" scoped>

View file

@ -0,0 +1,74 @@
<template>
<div>
<Title>Password Reset</Title>
<form @submit.prevent="resetPassword">
<div class="card w-max-[500px] md:w-[500px] bg-base-100 shadow-xl">
<div class="card-body">
<h2 class="card-title text-2xl align-center">
<MdiAccount class="mr-1 w-7 h-7" />
Password Reset
</h2>
<FormPassword v-model="form.password" label="New Password" />
<FormPassword v-model="form.passwordConfirm" label="Confirm Password" />
<PasswordScore v-model:valid="form.requirementsMet" :password="form.password" />
<div class="card-actions justify-end">
<button type="submit" class="btn btn-primary mt-2" :class="loading ? 'loading' : ''">Reset Password</button>
</div>
</div>
</div>
</form>
<div class="grid place-content-center pt-4">
<NuxtLink to="/#login">
<p class="text-xs text-base-content/50 mt-2">Account Login</p>
</NuxtLink>
</div>
</div>
</template>
<script setup lang="ts">
import MdiAccount from "~icons/mdi/account";
const route = useRoute();
definePageMeta({
title: "Password Reset",
layout: "center-card",
middleware: [
() => {
const ctx = useAuthContext();
if (ctx.isAuthorized()) {
return "/home";
}
},
],
});
const toast = useNotifier();
const loading = ref(false);
const token = route.query.token;
const form = reactive({
requirementsMet: false,
password: "",
passwordConfirm: "",
});
function resetPassword() {
if (token === undefined) {
return toast.error("Invalid reset token");
}
if (form.password !== form.passwordConfirm) {
return toast.error("Passwords do not match");
}
if (!form.requirementsMet) {
return toast.error("Password does not meet requirements");
}
loading.value = true;
}
</script>
<style scoped></style>