feat: PURO AI landing + auth + docs redesign #1

Merged
purovps merged 20 commits from feat/design-landing-auth into main 2026-04-19 13:58:25 +00:00
Showing only changes of commit 9ee99d17fd - Show all commits

View File

@@ -1,69 +1,45 @@
<template>
<div class="relative flex min-h-screen items-center justify-center overflow-hidden p-4">
<!-- Background -->
<div
class="absolute inset-0 bg-gradient-to-br from-gray-50 via-primary-50/30 to-gray-100 dark:from-dark-950 dark:via-dark-900 dark:to-dark-950"
></div>
<div class="auth-shell" :class="{ 'auth-shell-split': hasNarrative }">
<div class="bg-glow soft"></div>
<!-- Decorative Elements -->
<div class="pointer-events-none absolute inset-0 overflow-hidden">
<!-- Gradient Orbs -->
<div
class="absolute -right-40 -top-40 h-80 w-80 rounded-full bg-primary-400/20 blur-3xl"
></div>
<div
class="absolute -bottom-40 -left-40 h-80 w-80 rounded-full bg-primary-500/15 blur-3xl"
></div>
<div
class="absolute left-1/2 top-1/2 h-96 w-96 -translate-x-1/2 -translate-y-1/2 rounded-full bg-primary-300/10 blur-3xl"
></div>
<!-- LEFT: Narrative (split mode only, hidden on mobile) -->
<aside v-if="hasNarrative" class="auth-narrative">
<slot name="narrative"></slot>
</aside>
<!-- Grid Pattern -->
<div
class="absolute inset-0 bg-[linear-gradient(rgba(20,184,166,0.03)_1px,transparent_1px),linear-gradient(90deg,rgba(20,184,166,0.03)_1px,transparent_1px)] bg-[size:64px_64px]"
></div>
</div>
<!-- Content Container -->
<div class="relative z-10 w-full max-w-md">
<!-- Logo/Brand -->
<div class="mb-8 text-center">
<!-- Custom Logo or Default Logo -->
<template v-if="settingsLoaded">
<!-- RIGHT: Form -->
<main class="auth-main">
<div class="auth-main-inner">
<!-- Legacy centered-card header (only when no narrative slot) -->
<div class="mb-8 text-center" v-if="!hasNarrative && settingsLoaded">
<div
class="mb-4 inline-flex h-16 w-16 items-center justify-center overflow-hidden rounded-2xl shadow-lg shadow-primary-500/30"
class="mb-4 inline-flex h-14 w-14 items-center justify-center overflow-hidden rounded-2xl"
>
<img :src="siteLogo || '/logo.png'" alt="Logo" class="h-full w-full object-contain" />
</div>
<h1 class="text-gradient mb-2 text-3xl font-bold">
{{ siteName }}
</h1>
<p class="text-sm text-gray-500 dark:text-dark-400">
{{ siteSubtitle }}
</p>
</template>
</div>
<h1 class="text-2xl font-bold">{{ siteName }}</h1>
<p class="text-sm text-gray-500 dark:text-dark-400" v-if="siteSubtitle">{{ siteSubtitle }}</p>
</div>
<!-- Card Container -->
<div class="card-glass rounded-2xl p-8 shadow-glass">
<!-- Form content -->
<slot />
</div>
<!-- Footer Links -->
<div class="mt-6 text-center text-sm">
<slot name="footer" />
</div>
<!-- Footer link slot (e.g., "没有账户?注册") -->
<div class="mt-6 text-center text-sm">
<slot name="footer" />
</div>
<!-- Copyright -->
<div class="mt-8 text-center text-xs text-gray-400 dark:text-dark-500">
&copy; {{ currentYear }} {{ siteName }}. All rights reserved.
<!-- Copyright (legacy mode only) -->
<div class="mt-8 text-center text-xs text-gray-400 dark:text-dark-500" v-if="!hasNarrative">
&copy; {{ currentYear }} {{ siteName }}. All rights reserved.
</div>
</div>
</div>
</main>
</div>
</template>
<script setup lang="ts">
import { computed, onMounted } from 'vue'
import { computed, onMounted, useSlots } from 'vue'
import { useAppStore } from '@/stores'
import { sanitizeUrl } from '@/utils/url'
@@ -76,6 +52,9 @@ const settingsLoaded = computed(() => appStore.publicSettingsLoaded)
const currentYear = computed(() => new Date().getFullYear())
const slots = useSlots()
const hasNarrative = computed(() => !!slots.narrative)
onMounted(() => {
appStore.fetchPublicSettings()
})
@@ -85,4 +64,57 @@ onMounted(() => {
.text-gradient {
@apply bg-gradient-to-r from-primary-600 to-primary-500 bg-clip-text text-transparent;
}
.auth-shell {
min-height: 100vh;
position: relative;
overflow: hidden;
}
.auth-shell-split {
display: grid;
grid-template-columns: 1fr 1fr;
}
@media (max-width: 900px) {
.auth-shell-split {
grid-template-columns: 1fr;
}
.auth-narrative {
display: none;
}
}
.auth-narrative {
position: relative;
padding: 48px 56px;
background: linear-gradient(135deg, #0a0e1a 0%, #1e1b4b 100%);
border-right: 1px solid var(--border);
display: flex;
flex-direction: column;
justify-content: space-between;
color: var(--text-0);
font-family: var(--font-sans);
}
.auth-main {
display: flex;
align-items: center;
justify-content: center;
padding: 40px 24px;
position: relative;
z-index: 2;
}
.auth-main-inner {
width: 100%;
max-width: 420px;
}
/* Legacy-mode (no narrative slot) background — keep existing gradient decorative look */
.auth-shell:not(.auth-shell-split) {
display: flex;
align-items: center;
justify-content: center;
padding: 1rem;
background: linear-gradient(to bottom right, #f9fafb, rgba(240,253,250,0.3), #f3f4f6);
}
:global(.dark) .auth-shell:not(.auth-shell-split) {
background: linear-gradient(to bottom right, #020617, #0f172a, #020617);
}
</style>