226 lines
5.8 KiB
CSS
226 lines
5.8 KiB
CSS
@import "tailwindcss";
|
|
|
|
@theme {
|
|
/* Deep ocean palette — distinct from NoFi's harsh void */
|
|
--color-deep: #050d1a;
|
|
--color-ocean: #0c1e3a;
|
|
--color-current: #0f2847;
|
|
--color-surface: #143561;
|
|
|
|
/* Flow accents */
|
|
--color-flow: #2dd4bf;
|
|
--color-flow-light: #5eead4;
|
|
--color-glow: #a7f3d0;
|
|
--color-anti-green: #e94560;
|
|
|
|
/* Text colors — cooler than NoFi's beige */
|
|
--color-foam: #e0f2f1;
|
|
--color-mist: #94a3b8;
|
|
|
|
/* Typography — sans-serif, not monospace */
|
|
--font-family-sans: var(--font-outfit), "Outfit", system-ui, sans-serif;
|
|
--font-family-caveat: var(--font-caveat), "Caveat", cursive;
|
|
}
|
|
|
|
/* ============================================
|
|
FLOWING ANIMATIONS — organic, not mechanical
|
|
============================================ */
|
|
|
|
/* Gentle wave undulation */
|
|
@keyframes wave {
|
|
0%, 100% { transform: translateY(0) rotate(0deg); }
|
|
25% { transform: translateY(-3px) rotate(0.3deg); }
|
|
75% { transform: translateY(3px) rotate(-0.3deg); }
|
|
}
|
|
|
|
/* Slow horizontal drift like a current */
|
|
@keyframes current {
|
|
0%, 100% { transform: translateX(0); }
|
|
50% { transform: translateX(12px); }
|
|
}
|
|
|
|
/* Breathing scale — organic pulse */
|
|
@keyframes breathe {
|
|
0%, 100% { transform: scale(1); opacity: 1; }
|
|
50% { transform: scale(1.015); opacity: 0.95; }
|
|
}
|
|
|
|
/* Soft glow pulse */
|
|
@keyframes glow-pulse {
|
|
0%, 100% { opacity: 0.4; filter: blur(0px); }
|
|
50% { opacity: 0.7; filter: blur(1px); }
|
|
}
|
|
|
|
/* Flowing gradient background */
|
|
@keyframes flow-gradient {
|
|
0% { background-position: 0% 50%; }
|
|
50% { background-position: 100% 50%; }
|
|
100% { background-position: 0% 50%; }
|
|
}
|
|
|
|
/* Ripple expand */
|
|
@keyframes ripple {
|
|
0% { transform: scale(0.95); opacity: 0; }
|
|
50% { opacity: 1; }
|
|
100% { transform: scale(1); opacity: 1; }
|
|
}
|
|
|
|
/* Rising bubbles */
|
|
@keyframes rise {
|
|
0% { transform: translateY(0); opacity: 0.6; }
|
|
100% { transform: translateY(-60px); opacity: 0; }
|
|
}
|
|
|
|
.animate-wave { animation: wave 6s ease-in-out infinite; }
|
|
.animate-current { animation: current 10s ease-in-out infinite; }
|
|
.animate-breathe { animation: breathe 6s ease-in-out infinite; }
|
|
.animate-glow-pulse { animation: glow-pulse 4s ease-in-out infinite; }
|
|
.animate-ripple { animation: ripple 0.8s ease-out forwards; }
|
|
.animate-flow-gradient {
|
|
background-size: 200% 200%;
|
|
animation: flow-gradient 8s ease-in-out infinite;
|
|
}
|
|
|
|
/* ============================================
|
|
PIPE FLOW — core Sankey animation
|
|
============================================ */
|
|
|
|
@keyframes pipe-flow {
|
|
0% { stroke-dashoffset: 24; }
|
|
100% { stroke-dashoffset: 0; }
|
|
}
|
|
|
|
.animate-pipe-flow {
|
|
stroke-dasharray: 8 4;
|
|
animation: pipe-flow 1.5s linear infinite;
|
|
}
|
|
|
|
.animate-pipe-flow-fast {
|
|
stroke-dasharray: 8 4;
|
|
animation: pipe-flow 0.8s linear infinite;
|
|
}
|
|
|
|
.animate-pipe-flow-slow {
|
|
stroke-dasharray: 8 4;
|
|
animation: pipe-flow 2.5s linear infinite;
|
|
}
|
|
|
|
/* Particle travel along path */
|
|
@keyframes travel-path {
|
|
0% { offset-distance: 0%; opacity: 0; }
|
|
5% { opacity: 1; }
|
|
95% { opacity: 1; }
|
|
100% { offset-distance: 100%; opacity: 0; }
|
|
}
|
|
|
|
.animate-particle {
|
|
offset-rotate: 0deg;
|
|
animation: travel-path var(--duration, 3s) linear infinite;
|
|
animation-delay: var(--delay, 0s);
|
|
}
|
|
|
|
/* ============================================
|
|
FLOWING UNDERLINE — SVG wave, not scrawl
|
|
============================================ */
|
|
|
|
.flow-underline {
|
|
position: relative;
|
|
display: inline-block;
|
|
}
|
|
.flow-underline::after {
|
|
content: '';
|
|
position: absolute;
|
|
left: -2%;
|
|
bottom: -6px;
|
|
width: 104%;
|
|
height: 6px;
|
|
background: linear-gradient(90deg, transparent, var(--color-flow), transparent);
|
|
border-radius: 3px;
|
|
opacity: 0.6;
|
|
mask-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 120 8'%3E%3Cpath d='M0,4 Q15,0 30,4 Q45,8 60,4 Q75,0 90,4 Q105,8 120,4' fill='none' stroke='white' stroke-width='6'/%3E%3C/svg%3E");
|
|
mask-size: 100% 100%;
|
|
-webkit-mask-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 120 8'%3E%3Cpath d='M0,4 Q15,0 30,4 Q45,8 60,4 Q75,0 90,4 Q105,8 120,4' fill='none' stroke='white' stroke-width='6'/%3E%3C/svg%3E");
|
|
-webkit-mask-size: 100% 100%;
|
|
}
|
|
|
|
/* ============================================
|
|
SECTION DIVIDERS — flowing wave transitions
|
|
============================================ */
|
|
|
|
.wave-divider {
|
|
position: absolute;
|
|
bottom: -1px;
|
|
left: 0;
|
|
width: 100%;
|
|
overflow: hidden;
|
|
line-height: 0;
|
|
}
|
|
|
|
.wave-divider svg {
|
|
position: relative;
|
|
display: block;
|
|
width: calc(100% + 1.3px);
|
|
height: 60px;
|
|
}
|
|
|
|
/* ============================================
|
|
COMPONENT STYLES
|
|
============================================ */
|
|
|
|
/* Sandbox nodes — rounded, glowing */
|
|
.flow-node {
|
|
border: 1.5px solid rgba(45, 212, 191, 0.2);
|
|
border-radius: 16px;
|
|
padding: 14px;
|
|
background: rgba(12, 30, 58, 0.85);
|
|
backdrop-filter: blur(8px);
|
|
min-width: 120px;
|
|
transition: all 0.4s ease;
|
|
box-shadow: 0 0 20px rgba(45, 212, 191, 0.05);
|
|
}
|
|
|
|
.flow-node:hover {
|
|
border-color: rgba(45, 212, 191, 0.5);
|
|
box-shadow: 0 0 30px rgba(45, 212, 191, 0.15);
|
|
}
|
|
|
|
/* Disable feTurbulence on mobile for performance */
|
|
@media (max-width: 768px) {
|
|
.water-distortion {
|
|
filter: none !important;
|
|
}
|
|
}
|
|
|
|
/* React Flow overrides */
|
|
.react-flow__background {
|
|
background-color: var(--color-deep) !important;
|
|
}
|
|
|
|
.react-flow__controls button {
|
|
background-color: var(--color-ocean) !important;
|
|
border-color: rgba(45, 212, 191, 0.15) !important;
|
|
color: var(--color-foam) !important;
|
|
fill: var(--color-foam) !important;
|
|
border-radius: 8px !important;
|
|
}
|
|
|
|
.react-flow__controls button:hover {
|
|
background-color: var(--color-current) !important;
|
|
}
|
|
|
|
.react-flow__minimap {
|
|
background-color: var(--color-deep) !important;
|
|
border-radius: 12px !important;
|
|
}
|
|
|
|
/* Smooth scrolling */
|
|
html {
|
|
scroll-behavior: smooth;
|
|
}
|
|
|
|
/* Selection color */
|
|
::selection {
|
|
background: rgba(45, 212, 191, 0.3);
|
|
color: var(--color-foam);
|
|
}
|