website refactor

This commit is contained in:
2026-01-17 15:46:55 +01:00
parent 4d5ce9bfd6
commit 72a626ce71
346 changed files with 19308 additions and 8605 deletions

View File

@@ -1,27 +1,18 @@
'use client';
import React from 'react';
import {
Lock,
Eye,
EyeOff,
AlertCircle,
Flag,
Shield,
CheckCircle2,
ArrowLeft,
} from 'lucide-react';
import { Card } from '@/ui/Card';
import { Shield, CheckCircle2, ArrowLeft, AlertCircle } from 'lucide-react';
import { Button } from '@/ui/Button';
import { Input } from '@/ui/Input';
import { Heading } from '@/ui/Heading';
import { Box } from '@/ui/Box';
import { Stack } from '@/ui/Stack';
import { Text } from '@/ui/Text';
import { Link } from '@/ui/Link';
import { Surface } from '@/ui/Surface';
import { Icon } from '@/ui/Icon';
import { LoadingSpinner } from '@/ui/LoadingSpinner';
import { Box } from '@/ui/Box';
import { Stack } from '@/ui/Stack';
import { Icon } from '@/ui/Icon';
import { AuthCard } from '@/components/auth/AuthCard';
import { AuthForm } from '@/components/auth/AuthForm';
import { PasswordField } from '@/ui/PasswordField';
import { AuthFooterLinks } from '@/components/auth/AuthFooterLinks';
import { routes } from '@/lib/routing/RouteConfig';
import { ResetPasswordViewData } from '@/lib/builders/view-data/types/ResetPasswordViewData';
@@ -50,194 +41,102 @@ export function ResetPasswordTemplate({
uiState,
mutationState,
}: ResetPasswordTemplateProps) {
const isSubmitting = mutationState.isPending;
return (
<Box as="main" minHeight="100vh" display="flex" alignItems="center" justifyContent="center" position="relative">
{/* Background Pattern */}
<Box position="absolute" inset="0" bg="linear-gradient(to bottom right, rgba(59, 130, 246, 0.05), transparent, rgba(147, 51, 234, 0.05))" />
<Box position="relative" w="full" maxWidth="28rem" px={4}>
{/* Header */}
<Box textAlign="center" mb={8}>
<Surface variant="muted" rounded="2xl" border padding={4} w="4rem" h="4rem" display="flex" alignItems="center" justifyContent="center" mx="auto" mb={4}>
<Icon icon={Flag} size={8} color="#3b82f6" />
</Surface>
<Heading level={1}>Reset Password</Heading>
<Text color="text-gray-400" block mt={2}>
Create a new secure password for your account
</Text>
</Box>
<AuthCard
title="Reset Password"
description={viewData.showSuccess ? undefined : "Create a new secure password for your account"}
>
{!viewData.showSuccess ? (
<AuthForm onSubmit={formActions.handleSubmit}>
<Stack gap={4}>
<PasswordField
label="New Password"
id="newPassword"
name="newPassword"
value={viewData.formState.fields.newPassword.value}
onChange={formActions.handleChange}
errorMessage={viewData.formState.fields.newPassword.error}
placeholder="••••••••"
disabled={isSubmitting}
autoComplete="new-password"
showPassword={uiState.showPassword}
onTogglePassword={() => formActions.setShowPassword(!uiState.showPassword)}
/>
<Card position="relative" overflow="hidden">
{/* Background accent */}
<Box position="absolute" top="0" right="0" w="8rem" h="8rem" bg="linear-gradient(to bottom left, rgba(59, 130, 246, 0.1), transparent)" />
<PasswordField
label="Confirm Password"
id="confirmPassword"
name="confirmPassword"
value={viewData.formState.fields.confirmPassword.value}
onChange={formActions.handleChange}
errorMessage={viewData.formState.fields.confirmPassword.error}
placeholder="••••••••"
disabled={isSubmitting}
autoComplete="new-password"
showPassword={uiState.showConfirmPassword}
onTogglePassword={() => formActions.setShowConfirmPassword(!uiState.showConfirmPassword)}
/>
</Stack>
{!viewData.showSuccess ? (
<Box as="form" onSubmit={formActions.handleSubmit}>
<Stack gap={5} position="relative">
{/* New Password */}
<Box>
<Text size="sm" weight="medium" color="text-gray-300" block mb={2}>
New Password
</Text>
<Box position="relative">
<Box position="absolute" left="3" top="50%" zIndex={10}>
<Icon icon={Lock} size={4} color="#6b7280" />
</Box>
<Input
id="newPassword"
name="newPassword"
type={uiState.showPassword ? 'text' : 'password'}
value={viewData.formState.fields.newPassword.value}
onChange={formActions.handleChange}
variant={viewData.formState.fields.newPassword.error ? 'error' : 'default'}
placeholder="••••••••"
disabled={mutationState.isPending}
autoComplete="new-password"
/>
<Box
as="button"
type="button"
onClick={() => formActions.setShowPassword(!uiState.showPassword)}
position="absolute"
right="3"
top="50%"
zIndex={10}
bg="transparent"
borderStyle="none"
cursor="pointer"
>
<Icon icon={uiState.showPassword ? EyeOff : Eye} size={4} color="#6b7280" />
</Box>
</Box>
{viewData.formState.fields.newPassword.error && (
<Text size="xs" color="text-error-red" block mt={1}>
{viewData.formState.fields.newPassword.error}
</Text>
)}
</Box>
{/* Confirm Password */}
<Box>
<Text size="sm" weight="medium" color="text-gray-300" block mb={2}>
Confirm Password
</Text>
<Box position="relative">
<Box position="absolute" left="3" top="50%" zIndex={10}>
<Icon icon={Lock} size={4} color="#6b7280" />
</Box>
<Input
id="confirmPassword"
name="confirmPassword"
type={uiState.showConfirmPassword ? 'text' : 'password'}
value={viewData.formState.fields.confirmPassword.value}
onChange={formActions.handleChange}
variant={viewData.formState.fields.confirmPassword.error ? 'error' : 'default'}
placeholder="••••••••"
disabled={mutationState.isPending}
autoComplete="new-password"
/>
<Box
as="button"
type="button"
onClick={() => formActions.setShowConfirmPassword(!uiState.showConfirmPassword)}
position="absolute"
right="3"
top="50%"
zIndex={10}
bg="transparent"
borderStyle="none"
cursor="pointer"
>
<Icon icon={uiState.showConfirmPassword ? EyeOff : Eye} size={4} color="#6b7280" />
</Box>
</Box>
{viewData.formState.fields.confirmPassword.error && (
<Text size="xs" color="text-error-red" block mt={1}>
{viewData.formState.fields.confirmPassword.error}
</Text>
)}
</Box>
{/* Error Message */}
{mutationState.error && (
<Surface variant="muted" rounded="lg" border padding={3} bg="bg-red-500/10" borderColor="border-red-500/30">
<Stack direction="row" align="start" gap={3}>
<Icon icon={AlertCircle} size={5} color="#ef4444" />
<Text size="sm" color="text-error-red">{mutationState.error}</Text>
</Stack>
</Surface>
)}
{/* Submit Button */}
<Button
type="submit"
variant="primary"
disabled={mutationState.isPending}
fullWidth
icon={mutationState.isPending ? <LoadingSpinner size={4} color="white" /> : <Icon icon={Shield} size={4} />}
>
{mutationState.isPending ? 'Resetting...' : 'Reset Password'}
</Button>
{/* Back to Login */}
<Box textAlign="center">
<Link href={routes.auth.login}>
<Stack direction="row" align="center" justify="center" gap={1}>
<Icon icon={ArrowLeft} size={4} color="#3b82f6" />
<Text size="sm" color="text-primary-blue">Back to Login</Text>
</Stack>
</Link>
</Box>
{mutationState.error && (
<Box p={4} bg="critical-red/10" border borderColor="critical-red/30" rounded="md">
<Stack direction="row" align="start" gap={3}>
<Icon icon={AlertCircle} size={4.5} color="var(--color-critical)" />
<Text size="sm" color="text-critical-red">{mutationState.error}</Text>
</Stack>
</Box>
) : (
<Stack gap={4} position="relative">
<Surface variant="muted" rounded="lg" border padding={4} bg="bg-green-500/10" borderColor="border-green-500/30">
<Stack direction="row" align="start" gap={3}>
<Icon icon={CheckCircle2} size={6} color="#10b981" />
<Box>
<Text size="sm" color="text-performance-green" weight="medium" block>{viewData.successMessage}</Text>
<Text size="xs" color="text-gray-400" block mt={1}>
Your password has been successfully reset
</Text>
</Box>
</Stack>
</Surface>
<Button
type="button"
variant="secondary"
onClick={() => window.location.href = '/auth/login'}
fullWidth
>
Return to Login
</Button>
</Stack>
)}
</Card>
{/* Trust Indicators */}
<Stack direction="row" align="center" justify="center" gap={6} mt={6}>
<Stack direction="row" align="center" gap={2}>
<Icon icon={Shield} size={4} color="#737373" />
<Text size="sm" color="text-gray-500">Secure password reset</Text>
</Stack>
<Stack direction="row" align="center" gap={2}>
<Icon icon={CheckCircle2} size={4} color="#737373" />
<Text size="sm" color="text-gray-500">Encrypted transmission</Text>
</Stack>
</Stack>
<Button
type="submit"
variant="primary"
disabled={isSubmitting}
fullWidth
icon={isSubmitting ? <LoadingSpinner size={4} /> : <Shield size={16} />}
>
{isSubmitting ? 'Resetting...' : 'Reset Password'}
</Button>
{/* Footer */}
<Box mt={6} textAlign="center">
<Text size="xs" color="text-gray-500">
Need help?{' '}
<Link href="/support">
<Text color="text-gray-400">Contact support</Text>
<Box textAlign="center">
<Link href={routes.auth.login}>
<Stack direction="row" align="center" justify="center" gap={2} group>
<Icon icon={ArrowLeft} size={3.5} color="var(--color-primary)" groupHoverScale />
<Text size="sm" weight="bold" color="text-primary-accent">Back to Login</Text>
</Stack>
</Link>
</Text>
</Box>
</Box>
</Box>
</Box>
</AuthForm>
) : (
<Stack gap={6}>
<Box p={4} bg="success-green/10" border borderColor="success-green/30" rounded="md">
<Stack direction="row" align="start" gap={3}>
<Icon icon={CheckCircle2} size={5} color="var(--color-success)" />
<Box>
<Text size="sm" color="text-success-green" weight="bold" block>Password Reset</Text>
<Text size="xs" color="text-gray-400" block mt={1}>{viewData.successMessage}</Text>
</Box>
</Stack>
</Box>
<Button
type="button"
variant="secondary"
onClick={() => window.location.href = '/auth/login'}
fullWidth
>
Return to Login
</Button>
</Stack>
)}
<AuthFooterLinks>
<Text size="xs" color="text-gray-600">
Need help?{' '}
<Link href="/support">Contact support</Link>
</Text>
</AuthFooterLinks>
</AuthCard>
);
}