mirror of
https://github.com/yyhuni/xingrin.git
synced 2026-01-31 11:46:16 +08:00
- Replace traditional card-based login form with immersive terminal-style interface - Add PixelBlast animated background component for cyberpunk aesthetic - Implement TerminalLogin component with typewriter and terminal effects - Add new animation components: FaultyTerminal, PixelBlast, Shuffle with CSS modules - Add gravity-stars background animation component from animate-ui - Add terminal cursor blink animation to global styles - Update login page translations to support terminal UI prompts and messages - Replace Lottie animation with dynamic WebGL-based PixelBlast component - Add dynamic imports to prevent SSR issues with WebGL rendering - Update component registry to include @magicui and @react-bits registries - Refactor login form state management to use async/await pattern - Add fingerprint meta tag for search engine identification (FOFA/Shodan) - Improve visual hierarchy with relative z-index layering for background and content
384 lines
11 KiB
CSS
384 lines
11 KiB
CSS
@import "tailwindcss";
|
|
@import "tw-animate-css";
|
|
@import "@xterm/xterm/css/xterm.css";
|
|
@import "../styles/themes/index.css";
|
|
|
|
@custom-variant dark (&:is(.dark *));
|
|
|
|
@theme inline {
|
|
--color-background: var(--background);
|
|
--color-foreground: var(--foreground);
|
|
--color-sidebar-ring: var(--sidebar-ring);
|
|
--color-sidebar-border: var(--sidebar-border);
|
|
--color-sidebar-accent-foreground: var(--sidebar-accent-foreground);
|
|
--color-sidebar-accent: var(--sidebar-accent);
|
|
--color-sidebar-primary-foreground: var(--sidebar-primary-foreground);
|
|
--color-sidebar-primary: var(--sidebar-primary);
|
|
--color-sidebar-foreground: var(--sidebar-foreground);
|
|
--color-sidebar: var(--sidebar);
|
|
--color-chart-5: var(--chart-5);
|
|
--color-chart-4: var(--chart-4);
|
|
--color-chart-3: var(--chart-3);
|
|
--color-chart-2: var(--chart-2);
|
|
--color-chart-1: var(--chart-1);
|
|
--color-ring: var(--ring);
|
|
--color-input: var(--input);
|
|
--color-border: var(--border);
|
|
--color-destructive: var(--destructive);
|
|
--color-accent-foreground: var(--accent-foreground);
|
|
--color-accent: var(--accent);
|
|
--color-muted-foreground: var(--muted-foreground);
|
|
--color-muted: var(--muted);
|
|
--color-secondary-foreground: var(--secondary-foreground);
|
|
--color-secondary: var(--secondary);
|
|
--color-primary-foreground: var(--primary-foreground);
|
|
--color-primary: var(--primary);
|
|
--color-popover-foreground: var(--popover-foreground);
|
|
--color-popover: var(--popover);
|
|
--color-card-foreground: var(--card-foreground);
|
|
--color-card: var(--card);
|
|
--radius-sm: calc(var(--radius) - 4px);
|
|
--radius-md: calc(var(--radius) - 2px);
|
|
--radius-lg: var(--radius);
|
|
--radius-xl: calc(var(--radius) + 4px);
|
|
--font-sans: 'Noto Sans SC', system-ui, -apple-system, PingFang SC, sans-serif;
|
|
--font-mono: 'JetBrains Mono', 'Fira Code', Consolas, monospace;
|
|
--font-serif: Georgia, 'Noto Serif SC', serif;
|
|
--tracking-tighter: calc(var(--tracking-normal) - 0.05em);
|
|
--tracking-tight: calc(var(--tracking-normal) - 0.025em);
|
|
--tracking-wide: calc(var(--tracking-normal) + 0.025em);
|
|
--tracking-wider: calc(var(--tracking-normal) + 0.05em);
|
|
--tracking-widest: calc(var(--tracking-normal) + 0.1em);
|
|
--tracking-normal: var(--tracking-normal);
|
|
--shadow-2xl: var(--shadow-2xl);
|
|
--shadow-xl: var(--shadow-xl);
|
|
--shadow-lg: var(--shadow-lg);
|
|
--shadow-md: var(--shadow-md);
|
|
--shadow: var(--shadow);
|
|
--shadow-sm: var(--shadow-sm);
|
|
--shadow-xs: var(--shadow-xs);
|
|
--shadow-2xs: var(--shadow-2xs);
|
|
--spacing: var(--spacing);
|
|
--letter-spacing: var(--letter-spacing);
|
|
--shadow-offset-y: var(--shadow-offset-y);
|
|
--shadow-offset-x: var(--shadow-offset-x);
|
|
--shadow-spread: var(--shadow-spread);
|
|
--shadow-blur: var(--shadow-blur);
|
|
--shadow-opacity: var(--shadow-opacity);
|
|
--color-shadow-color: var(--shadow-color);
|
|
--color-destructive-foreground: var(--destructive-foreground);
|
|
}
|
|
|
|
/* 基础主题 - Vercel 风格 (黑白灰) */
|
|
/* 只在没有设置 data-theme 时应用默认样式 */
|
|
:root:not([data-theme]),
|
|
[data-theme="vercel"] {
|
|
--radius: 0.5rem;
|
|
--background: oklch(0.9900 0 0);
|
|
--foreground: oklch(0 0 0);
|
|
--card: oklch(1 0 0);
|
|
--card-foreground: oklch(0 0 0);
|
|
--popover: oklch(0.9900 0 0);
|
|
--popover-foreground: oklch(0 0 0);
|
|
--primary: oklch(0 0 0);
|
|
--primary-foreground: oklch(1 0 0);
|
|
--secondary: oklch(0.9400 0 0);
|
|
--secondary-foreground: oklch(0 0 0);
|
|
--muted: oklch(0.9700 0 0);
|
|
--muted-foreground: oklch(0.4400 0 0);
|
|
--accent: oklch(0.9400 0 0);
|
|
--accent-foreground: oklch(0 0 0);
|
|
--destructive: oklch(0.6300 0.1900 23.0300);
|
|
--destructive-foreground: oklch(1 0 0);
|
|
--border: oklch(0.9200 0 0);
|
|
--input: oklch(0.9400 0 0);
|
|
--ring: oklch(0 0 0);
|
|
--chart-1: oklch(0.8100 0.1700 75.3500);
|
|
--chart-2: oklch(0.5500 0.2200 264.5300);
|
|
--chart-3: oklch(0.7200 0 0);
|
|
--chart-4: oklch(0.9200 0 0);
|
|
--chart-5: oklch(0.5600 0 0);
|
|
--sidebar: oklch(0.9900 0 0);
|
|
--sidebar-foreground: oklch(0 0 0);
|
|
--sidebar-primary: oklch(0 0 0);
|
|
--sidebar-primary-foreground: oklch(1 0 0);
|
|
--sidebar-accent: oklch(0.9400 0 0);
|
|
--sidebar-accent-foreground: oklch(0 0 0);
|
|
--sidebar-border: oklch(0.9400 0 0);
|
|
--sidebar-ring: oklch(0 0 0);
|
|
--font-sans: 'Noto Sans SC', system-ui, -apple-system, PingFang SC, Hiragino Sans GB, Microsoft YaHei, sans-serif;
|
|
--font-serif: Georgia, serif;
|
|
--font-mono: 'JetBrains Mono', 'Fira Code', Consolas, monospace;
|
|
--shadow-color: hsl(0 0% 0%);
|
|
--shadow-opacity: 0.18;
|
|
--shadow-blur: 2px;
|
|
--shadow-spread: 0px;
|
|
--shadow-offset-x: 0px;
|
|
--shadow-offset-y: 1px;
|
|
--letter-spacing: 0em;
|
|
--spacing: 0.25rem;
|
|
--shadow-2xs: 0px 1px 2px 0px hsl(0 0% 0% / 0.09);
|
|
--shadow-xs: 0px 1px 2px 0px hsl(0 0% 0% / 0.09);
|
|
--shadow-sm: 0px 1px 2px 0px hsl(0 0% 0% / 0.18), 0px 1px 2px -1px hsl(0 0% 0% / 0.18);
|
|
--shadow: 0px 1px 2px 0px hsl(0 0% 0% / 0.18), 0px 1px 2px -1px hsl(0 0% 0% / 0.18);
|
|
--shadow-md: 0px 1px 2px 0px hsl(0 0% 0% / 0.18), 0px 2px 4px -1px hsl(0 0% 0% / 0.18);
|
|
--shadow-lg: 0px 1px 2px 0px hsl(0 0% 0% / 0.18), 0px 4px 6px -1px hsl(0 0% 0% / 0.18);
|
|
--shadow-xl: 0px 1px 2px 0px hsl(0 0% 0% / 0.18), 0px 8px 10px -1px hsl(0 0% 0% / 0.18);
|
|
--shadow-2xl: 0px 1px 2px 0px hsl(0 0% 0% / 0.45);
|
|
--tracking-normal: 0em;
|
|
}
|
|
|
|
/* 基础主题 - 暗色模式 (Vercel 风格) */
|
|
[data-theme="vercel"].dark,
|
|
:root:not([data-theme]).dark {
|
|
--background: oklch(0 0 0);
|
|
--foreground: oklch(1 0 0);
|
|
--card: oklch(0.1400 0 0);
|
|
--card-foreground: oklch(1 0 0);
|
|
--popover: oklch(0.1800 0 0);
|
|
--popover-foreground: oklch(1 0 0);
|
|
--primary: oklch(1 0 0);
|
|
--primary-foreground: oklch(0 0 0);
|
|
--secondary: oklch(0.2500 0 0);
|
|
--secondary-foreground: oklch(1 0 0);
|
|
--muted: oklch(0.2300 0 0);
|
|
--muted-foreground: oklch(0.7200 0 0);
|
|
--accent: oklch(0.3200 0 0);
|
|
--accent-foreground: oklch(1 0 0);
|
|
--destructive: oklch(0.6900 0.2000 23.9100);
|
|
--destructive-foreground: oklch(0 0 0);
|
|
--border: oklch(0.2600 0 0);
|
|
--input: oklch(0.3200 0 0);
|
|
--ring: oklch(0.7200 0 0);
|
|
--chart-1: oklch(0.8100 0.1700 75.3500);
|
|
--chart-2: oklch(0.5800 0.2100 260.8400);
|
|
--chart-3: oklch(0.5600 0 0);
|
|
--chart-4: oklch(0.4400 0 0);
|
|
--chart-5: oklch(0.9200 0 0);
|
|
--sidebar: oklch(0.1800 0 0);
|
|
--sidebar-foreground: oklch(1 0 0);
|
|
--sidebar-primary: oklch(1 0 0);
|
|
--sidebar-primary-foreground: oklch(0 0 0);
|
|
--sidebar-accent: oklch(0.3200 0 0);
|
|
--sidebar-accent-foreground: oklch(1 0 0);
|
|
--sidebar-border: oklch(0.3200 0 0);
|
|
--sidebar-ring: oklch(0.7200 0 0);
|
|
}
|
|
|
|
@layer base {
|
|
|
|
html,
|
|
body {
|
|
height: 100%;
|
|
}
|
|
|
|
* {
|
|
@apply border-border outline-ring/50;
|
|
}
|
|
|
|
body {
|
|
@apply bg-background text-foreground;
|
|
/* 禁止页面级滚动,滚动交给主内容容器 */
|
|
overflow: hidden;
|
|
letter-spacing: var(--tracking-normal);
|
|
}
|
|
|
|
/* 全局滚动条样式 - Webkit 浏览器 (Chrome, Safari, Edge) */
|
|
::-webkit-scrollbar {
|
|
width: 10px;
|
|
height: 10px;
|
|
}
|
|
|
|
::-webkit-scrollbar-track {
|
|
background: transparent;
|
|
}
|
|
|
|
::-webkit-scrollbar-thumb {
|
|
background: oklch(0.708 0 0 / 0.3);
|
|
border-radius: 5px;
|
|
border: 2px solid transparent;
|
|
background-clip: padding-box;
|
|
}
|
|
|
|
::-webkit-scrollbar-thumb:hover {
|
|
background: oklch(0.708 0 0 / 0.5);
|
|
border-radius: 5px;
|
|
border: 2px solid transparent;
|
|
background-clip: padding-box;
|
|
}
|
|
|
|
/* 暗色主题下的滚动条 */
|
|
.dark ::-webkit-scrollbar-thumb {
|
|
background: oklch(0.556 0 0 / 0.4);
|
|
border-radius: 5px;
|
|
border: 2px solid transparent;
|
|
background-clip: padding-box;
|
|
}
|
|
|
|
.dark ::-webkit-scrollbar-thumb:hover {
|
|
background: oklch(0.556 0 0 / 0.6);
|
|
border-radius: 5px;
|
|
border: 2px solid transparent;
|
|
background-clip: padding-box;
|
|
}
|
|
|
|
/* Firefox 滚动条样式 */
|
|
* {
|
|
scrollbar-width: thin;
|
|
scrollbar-color: oklch(0.708 0 0 / 0.3) transparent;
|
|
}
|
|
|
|
.dark * {
|
|
scrollbar-color: oklch(0.556 0 0 / 0.4) transparent;
|
|
}
|
|
|
|
/* 隐藏滚动条但保持可滚动 */
|
|
.scrollbar-hide {
|
|
-ms-overflow-style: none;
|
|
/* IE and Edge */
|
|
scrollbar-width: none;
|
|
/* Firefox */
|
|
}
|
|
|
|
.scrollbar-hide::-webkit-scrollbar {
|
|
display: none;
|
|
/* Chrome, Safari and Opera */
|
|
}
|
|
|
|
}
|
|
|
|
/* 登录页背景 - 使用主题色适配亮暗模式 */
|
|
.login-bg {
|
|
position: relative;
|
|
background-color: var(--background);
|
|
}
|
|
|
|
.login-bg::before {
|
|
content: '';
|
|
position: absolute;
|
|
inset: 0;
|
|
background-color: var(--primary);
|
|
opacity: 0.04;
|
|
mask-image: url("data:image/svg+xml,%3Csvg width='60' height='60' viewBox='0 0 60 60' xmlns='http://www.w3.org/2000/svg'%3E%3Cg fill='none' fill-rule='evenodd'%3E%3Cg fill='%23000' fill-opacity='1'%3E%3Cpath d='M36 34v-4h-2v4h-4v2h4v4h2v-4h4v-2h-4zm0-30V0h-2v4h-4v2h4v4h2V6h4V4h-4zM6 34v-4H4v4H0v2h4v4h2v-4h4v-2H6zM6 4V0H4v4H0v2h4v4h2V6h4V4H6z'/%3E%3C/g%3E%3C/g%3E%3C/svg%3E");
|
|
-webkit-mask-image: url("data:image/svg+xml,%3Csvg width='60' height='60' viewBox='0 0 60 60' xmlns='http://www.w3.org/2000/svg'%3E%3Cg fill='none' fill-rule='evenodd'%3E%3Cg fill='%23000' fill-opacity='1'%3E%3Cpath d='M36 34v-4h-2v4h-4v2h4v4h2v-4h4v-2h-4zm0-30V0h-2v4h-4v2h4v4h2V6h4V4h-4zM6 34v-4H4v4H0v2h4v4h2v-4h4v-2H6zM6 4V0H4v4H0v2h4v4h2V6h4V4H6z'/%3E%3C/g%3E%3C/g%3E%3C/svg%3E");
|
|
mask-size: 60px 60px;
|
|
-webkit-mask-size: 60px 60px;
|
|
pointer-events: none;
|
|
z-index: 0;
|
|
}
|
|
|
|
.login-bg > * {
|
|
position: relative;
|
|
z-index: 1;
|
|
}
|
|
|
|
/* 终端光标闪烁动画 */
|
|
@keyframes blink {
|
|
0%, 50% {
|
|
opacity: 1;
|
|
}
|
|
51%, 100% {
|
|
opacity: 0;
|
|
}
|
|
}
|
|
|
|
.animate-blink {
|
|
animation: blink 1s step-end infinite;
|
|
}
|
|
|
|
/* 通知铃铛摇晃动画 */
|
|
@keyframes wiggle {
|
|
0%, 100% {
|
|
transform: rotate(0deg);
|
|
}
|
|
15% {
|
|
transform: rotate(15deg);
|
|
}
|
|
30% {
|
|
transform: rotate(-12deg);
|
|
}
|
|
45% {
|
|
transform: rotate(8deg);
|
|
}
|
|
60% {
|
|
transform: rotate(-5deg);
|
|
}
|
|
75% {
|
|
transform: rotate(2deg);
|
|
}
|
|
}
|
|
|
|
/* 进度条条纹动画 */
|
|
@keyframes progress-stripes {
|
|
from {
|
|
background-position: 1rem 0;
|
|
}
|
|
to {
|
|
background-position: 0 0;
|
|
}
|
|
}
|
|
|
|
.progress-striped {
|
|
background-image: linear-gradient(
|
|
45deg,
|
|
rgba(255, 255, 255, 0.15) 25%,
|
|
transparent 25%,
|
|
transparent 50%,
|
|
rgba(255, 255, 255, 0.15) 50%,
|
|
rgba(255, 255, 255, 0.15) 75%,
|
|
transparent 75%,
|
|
transparent
|
|
);
|
|
background-size: 1rem 1rem;
|
|
animation: progress-stripes 1s linear infinite;
|
|
}
|
|
|
|
/* 闪电闪烁动画 - 快速扫描按钮 */
|
|
@keyframes flash {
|
|
0%, 90%, 100% {
|
|
opacity: 1;
|
|
transform: scale(1);
|
|
filter: drop-shadow(0 0 2px rgba(250, 204, 21, 0.4));
|
|
}
|
|
93% {
|
|
opacity: 1;
|
|
transform: scale(1.3);
|
|
filter: drop-shadow(0 0 8px rgba(250, 204, 21, 0.8));
|
|
}
|
|
96% {
|
|
opacity: 0.6;
|
|
transform: scale(1);
|
|
filter: drop-shadow(0 0 2px rgba(250, 204, 21, 0.4));
|
|
}
|
|
}
|
|
|
|
/* 按钮整体发光动画 */
|
|
@keyframes glow {
|
|
0%, 85%, 100% {
|
|
box-shadow: 0 0 0 transparent;
|
|
}
|
|
90% {
|
|
box-shadow: 0 0 12px oklch(from var(--primary) l c h / 0.5), 0 0 24px oklch(from var(--primary) l c h / 0.3);
|
|
}
|
|
95% {
|
|
box-shadow: 0 0 4px oklch(from var(--primary) l c h / 0.2);
|
|
}
|
|
}
|
|
|
|
.animate-glow {
|
|
animation: glow 3s ease-in-out infinite;
|
|
}
|
|
|
|
/* 边框流光动画 */
|
|
@keyframes border-flow {
|
|
0% {
|
|
transform: translateX(-100%) rotate(0deg);
|
|
}
|
|
100% {
|
|
transform: translateX(100%) rotate(0deg);
|
|
}
|
|
}
|
|
|
|
.animate-border-flow {
|
|
animation: border-flow 2s linear infinite;
|
|
} |