mirror of https://github.com/TZGyn/shortener
added demo login page
parent
60d0a07eba
commit
d97c790057
@ -0,0 +1,25 @@
|
|||||||
|
import Root from "./input.svelte";
|
||||||
|
|
||||||
|
type FormInputEvent<T extends Event = Event> = T & {
|
||||||
|
currentTarget: EventTarget & HTMLInputElement;
|
||||||
|
};
|
||||||
|
export type InputEvents = {
|
||||||
|
blur: FormInputEvent<FocusEvent>;
|
||||||
|
change: FormInputEvent<Event>;
|
||||||
|
click: FormInputEvent<MouseEvent>;
|
||||||
|
focus: FormInputEvent<FocusEvent>;
|
||||||
|
keydown: FormInputEvent<KeyboardEvent>;
|
||||||
|
keypress: FormInputEvent<KeyboardEvent>;
|
||||||
|
keyup: FormInputEvent<KeyboardEvent>;
|
||||||
|
mouseover: FormInputEvent<MouseEvent>;
|
||||||
|
mouseenter: FormInputEvent<MouseEvent>;
|
||||||
|
mouseleave: FormInputEvent<MouseEvent>;
|
||||||
|
paste: FormInputEvent<ClipboardEvent>;
|
||||||
|
input: FormInputEvent<InputEvent>;
|
||||||
|
};
|
||||||
|
|
||||||
|
export {
|
||||||
|
Root,
|
||||||
|
//
|
||||||
|
Root as Input
|
||||||
|
};
|
||||||
@ -0,0 +1,33 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import type { HTMLInputAttributes } from "svelte/elements";
|
||||||
|
import { cn } from "$lib/utils";
|
||||||
|
import type { InputEvents } from ".";
|
||||||
|
|
||||||
|
type $$Props = HTMLInputAttributes;
|
||||||
|
type $$Events = InputEvents;
|
||||||
|
|
||||||
|
let className: $$Props["class"] = undefined;
|
||||||
|
export let value: $$Props["value"] = undefined;
|
||||||
|
export { className as class };
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<input
|
||||||
|
class={cn(
|
||||||
|
"flex h-10 w-full rounded-md border border-input bg-transparent px-3 py-2 text-sm ring-offset-background file:border-0 file:bg-transparent file:text-foreground file:text-sm file:font-medium placeholder:text-muted-foreground focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:cursor-not-allowed disabled:opacity-50",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
bind:value
|
||||||
|
on:blur
|
||||||
|
on:change
|
||||||
|
on:click
|
||||||
|
on:focus
|
||||||
|
on:keydown
|
||||||
|
on:keypress
|
||||||
|
on:keyup
|
||||||
|
on:mouseover
|
||||||
|
on:mouseenter
|
||||||
|
on:mouseleave
|
||||||
|
on:paste
|
||||||
|
on:input
|
||||||
|
{...$$restProps}
|
||||||
|
/>
|
||||||
@ -0,0 +1,7 @@
|
|||||||
|
import Root from "./label.svelte";
|
||||||
|
|
||||||
|
export {
|
||||||
|
Root,
|
||||||
|
//
|
||||||
|
Root as Label
|
||||||
|
};
|
||||||
@ -0,0 +1,21 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import { Label as LabelPrimitive } from "bits-ui";
|
||||||
|
import { cn } from "$lib/utils";
|
||||||
|
|
||||||
|
type $$Props = LabelPrimitive.Props;
|
||||||
|
type $$Events = LabelPrimitive.Events;
|
||||||
|
|
||||||
|
let className: $$Props["class"] = undefined;
|
||||||
|
export { className as class };
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<LabelPrimitive.Root
|
||||||
|
class={cn(
|
||||||
|
"text-sm font-medium leading-none peer-disabled:cursor-not-allowed peer-disabled:opacity-70",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...$$restProps}
|
||||||
|
on:mousedown
|
||||||
|
>
|
||||||
|
<slot />
|
||||||
|
</LabelPrimitive.Root>
|
||||||
@ -0,0 +1,6 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import ThemeToggle from '$lib/components/theme-toggle.svelte';
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<ThemeToggle />
|
||||||
|
<slot />
|
||||||
@ -0,0 +1,61 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import { Button } from '$lib/components/ui/button';
|
||||||
|
import { Input } from '$lib/components/ui/input';
|
||||||
|
import { Label } from '$lib/components/ui/label';
|
||||||
|
import { Github, LoaderIcon } from 'lucide-svelte';
|
||||||
|
import { cn } from '$lib/utils';
|
||||||
|
|
||||||
|
let className: string | undefined | null = undefined;
|
||||||
|
export { className as class };
|
||||||
|
|
||||||
|
let isLoading = false;
|
||||||
|
async function onSubmit() {
|
||||||
|
isLoading = true;
|
||||||
|
|
||||||
|
setTimeout(() => {
|
||||||
|
isLoading = false;
|
||||||
|
}, 3000);
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<div class={cn('grid gap-6', className)} {...$$restProps}>
|
||||||
|
<form on:submit|preventDefault={onSubmit}>
|
||||||
|
<div class="grid gap-2">
|
||||||
|
<div class="grid gap-1">
|
||||||
|
<Label class="sr-only" for="email">Email</Label>
|
||||||
|
<Input
|
||||||
|
id="email"
|
||||||
|
placeholder="name@example.com"
|
||||||
|
type="email"
|
||||||
|
autocapitalize="none"
|
||||||
|
autocomplete="email"
|
||||||
|
autocorrect="off"
|
||||||
|
disabled={isLoading}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<Button disabled={isLoading}>
|
||||||
|
{#if isLoading}
|
||||||
|
<LoaderIcon />
|
||||||
|
{/if}
|
||||||
|
Sign In with Email
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
<div class="relative">
|
||||||
|
<div class="absolute inset-0 flex items-center">
|
||||||
|
<span class="w-full border-t" />
|
||||||
|
</div>
|
||||||
|
<div class="relative flex justify-center text-xs uppercase">
|
||||||
|
<span class="bg-background px-2 text-muted-foreground"> Or continue with </span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<Button variant="outline" type="button" disabled={isLoading}>
|
||||||
|
{#if isLoading}
|
||||||
|
<LoaderIcon />
|
||||||
|
{:else}
|
||||||
|
<Github />
|
||||||
|
{/if}
|
||||||
|
{' '}
|
||||||
|
Github
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
@ -0,0 +1,73 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import { Button } from '$lib/components/ui/button';
|
||||||
|
import UserAuthForm from './(components)/user-auth-form.svelte';
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<div class="md:hidden">
|
||||||
|
<img
|
||||||
|
src="/examples/authentication-light.png"
|
||||||
|
width={1280}
|
||||||
|
height={843}
|
||||||
|
alt="Authentication"
|
||||||
|
class="block dark:hidden"
|
||||||
|
/>
|
||||||
|
<img
|
||||||
|
src="/examples/authentication-dark.png"
|
||||||
|
width={1280}
|
||||||
|
height={843}
|
||||||
|
alt="Authentication"
|
||||||
|
class="hidden dark:block"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="container relative hidden h-[800px] flex-col items-center justify-center md:grid lg:max-w-none lg:grid-cols-2 lg:px-0"
|
||||||
|
>
|
||||||
|
<Button
|
||||||
|
href="/examples/authentication"
|
||||||
|
variant="ghost"
|
||||||
|
class="absolute right-4 top-4 md:right-8 md:top-8"
|
||||||
|
>
|
||||||
|
Login
|
||||||
|
</Button>
|
||||||
|
<div class="relative hidden h-full flex-col bg-muted p-10 text-white dark:border-r lg:flex">
|
||||||
|
<div
|
||||||
|
class="absolute inset-0 bg-cover"
|
||||||
|
style="
|
||||||
|
background-image:
|
||||||
|
url(https://images.unsplash.com/photo-1590069261209-f8e9b8642343?ixlib=rb-4.0.3&ixid=MnwxMjA3fDB8MHxwaG90by1wYWdlfHx8fGVufDB8fHx8&auto=format&fit=crop&w=1376&q=80);"
|
||||||
|
/>
|
||||||
|
<div class="relative z-20 flex items-center text-lg font-medium">
|
||||||
|
<!-- <Command class="mr-2 h-6 w-6" /> -->
|
||||||
|
Acme Inc
|
||||||
|
</div>
|
||||||
|
<div class="relative z-20 mt-auto">
|
||||||
|
<blockquote class="space-y-2">
|
||||||
|
<p class="text-lg">
|
||||||
|
“This library has saved me countless hours of work and helped me deliver stunning
|
||||||
|
designs to my clients faster than ever before. Highly recommended!”
|
||||||
|
</p>
|
||||||
|
<footer class="text-sm">Sofia Davis</footer>
|
||||||
|
</blockquote>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="lg:p-8">
|
||||||
|
<div class="mx-auto flex w-full flex-col justify-center space-y-6 sm:w-[350px]">
|
||||||
|
<div class="flex flex-col space-y-2 text-center">
|
||||||
|
<h1 class="text-2xl font-semibold tracking-tight">Create an account</h1>
|
||||||
|
<p class="text-sm text-muted-foreground">Enter your email below to create your account</p>
|
||||||
|
</div>
|
||||||
|
<UserAuthForm />
|
||||||
|
<p class="px-8 text-center text-sm text-muted-foreground">
|
||||||
|
By clicking continue, you agree to our{' '}
|
||||||
|
<a href="/terms" class="underline underline-offset-4 hover:text-primary">
|
||||||
|
Terms of Service
|
||||||
|
</a>{' '}
|
||||||
|
and{' '}
|
||||||
|
<a href="/privacy" class="underline underline-offset-4 hover:text-primary">
|
||||||
|
Privacy Policy
|
||||||
|
</a>
|
||||||
|
.
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
@ -0,0 +1,7 @@
|
|||||||
|
import type { PageLoad } from './$types';
|
||||||
|
|
||||||
|
export const load: PageLoad = async () => {
|
||||||
|
return {
|
||||||
|
title: 'Authentication Example'
|
||||||
|
};
|
||||||
|
};
|
||||||
@ -1,9 +1,7 @@
|
|||||||
<script>
|
<script>
|
||||||
import '../app.postcss';
|
import '../app.postcss';
|
||||||
import { ModeWatcher } from 'mode-watcher';
|
import { ModeWatcher } from 'mode-watcher';
|
||||||
import ThemeToggle from '$lib/components/theme-toggle.svelte';
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<ModeWatcher />
|
<ModeWatcher />
|
||||||
<ThemeToggle />
|
|
||||||
<slot />
|
<slot />
|
||||||
|
|||||||
Loading…
Reference in New Issue