| | <!DOCTYPE html>
|
| | <html lang="zh-CN">
|
| | <head>
|
| | <meta charset="UTF-8">
|
| | <title>Impact & Bound Square - Enhanced</title>
|
| | <style>
|
| | body {
|
| | margin: 0;
|
| | overflow: hidden;
|
| | background-color: #000;
|
| | font-family: 'Inter', -apple-system, sans-serif;
|
| | }
|
| | canvas { display: block; }
|
| |
|
| | #loginForm {
|
| | position: absolute;
|
| | top: 50%;
|
| | left: 50%;
|
| | transform: translate(-50%, -40%);
|
| | opacity: 0;
|
| | visibility: hidden;
|
| | transition: all 0.8s cubic-bezier(0.16, 1, 0.3, 1);
|
| | z-index: 10;
|
| | width: 280px;
|
| | padding: 45px;
|
| | background: rgba(255, 255, 255, 0.02);
|
| | backdrop-filter: blur(20px);
|
| | -webkit-backdrop-filter: blur(20px);
|
| | border-radius: 40px;
|
| | border: 1px solid rgba(255, 255, 255, 0.08);
|
| | }
|
| |
|
| | #loginForm.visible {
|
| | opacity: 1;
|
| | visibility: visible;
|
| | transform: translate(-50%, -50%);
|
| | }
|
| |
|
| | .label {
|
| | color: rgba(255, 255, 255, 0.3);
|
| | font-size: 10px;
|
| | letter-spacing: 4px;
|
| | margin-bottom: 8px;
|
| | text-transform: uppercase;
|
| | }
|
| |
|
| | .form-group {
|
| | position: relative;
|
| | margin-bottom: 35px;
|
| | }
|
| |
|
| | .form-group input {
|
| | width: 100%;
|
| | padding: 12px 0;
|
| | background: transparent;
|
| | border: none;
|
| | border-bottom: 1px solid rgba(255, 255, 255, 0.1);
|
| | color: #fff;
|
| | font-size: 14px;
|
| | outline: none;
|
| | }
|
| |
|
| | .form-group::after {
|
| | content: '';
|
| | position: absolute;
|
| | bottom: 0; left: 0;
|
| | width: 0; height: 1px;
|
| | background: #fff;
|
| | transition: width 0.6s ease;
|
| | }
|
| |
|
| | .form-group input:focus ~ ::after { width: 100%; }
|
| |
|
| | #loginButton {
|
| | width: 100%;
|
| | padding: 16px;
|
| | background: #fff;
|
| | border: none;
|
| | border-radius: 50px;
|
| | color: #000;
|
| | font-size: 11px;
|
| | font-weight: 800;
|
| | letter-spacing: 6px;
|
| | cursor: pointer;
|
| | transition: all 0.4s;
|
| | margin-top: 10px;
|
| | }
|
| |
|
| | #loginButton:hover {
|
| | transform: scale(1.05);
|
| | box-shadow: 0 0 40px rgba(255, 255, 255, 0.4);
|
| | }
|
| |
|
| | .timestamp {
|
| | position: absolute;
|
| | top: 40px;
|
| | right: 40px;
|
| | color: rgba(255, 255, 255, 0.15);
|
| | font-size: 10px;
|
| | font-family: monospace;
|
| | }
|
| | </style>
|
| | </head>
|
| | <body>
|
| | <div class="timestamp" id="timer"></div>
|
| | <canvas id="sandCanvas"></canvas>
|
| |
|
| | <div id="loginForm">
|
| | <div class="form-group">
|
| | <div class="label">Identity</div>
|
| | <input type="text" id="username" placeholder=" " autocomplete="off">
|
| | </div>
|
| | <div class="form-group">
|
| | <div class="label">Access Code</div>
|
| | <input type="password" id="password" placeholder=" " autocomplete="off">
|
| | </div>
|
| | <button id="loginButton">CONNECT</button>
|
| | </div>
|
| |
|
| | <script>
|
| | const canvas = document.getElementById('sandCanvas');
|
| | const ctx = canvas.getContext('2d');
|
| | const loginForm = document.getElementById('loginForm');
|
| | const timerEl = document.getElementById('timer');
|
| |
|
| | let width, height, particles = [];
|
| | const particleCount = 4500;
|
| | const mouseThreshold = 220;
|
| | let isHovering = false;
|
| | let noiseTimer = 0;
|
| |
|
| | function updateTimer() {
|
| | const now = new Date();
|
| | timerEl.innerText = now.toLocaleString('en-GB').toUpperCase();
|
| | }
|
| | setInterval(updateTimer, 1000);
|
| |
|
| | window.addEventListener('resize', init);
|
| | window.addEventListener('mousemove', (e) => {
|
| | const dx = e.clientX - width / 2;
|
| | const dy = e.clientY - height / 2;
|
| | isHovering = Math.sqrt(dx*dx + dy*dy) < mouseThreshold;
|
| | });
|
| |
|
| | function isInsideRoundedSquare(px, py, halfSide, radius) {
|
| | const dx = Math.abs(px - width / 2);
|
| | const dy = Math.abs(py - height / 2);
|
| | if (dx > halfSide || dy > halfSide) return false;
|
| | if (dx > halfSide - radius && dy > halfSide - radius) {
|
| | const cx = dx - (halfSide - radius);
|
| | const cy = dy - (halfSide - radius);
|
| | return (cx * cx + cy * cy <= radius * radius);
|
| | }
|
| | return true;
|
| | }
|
| |
|
| | class Particle {
|
| | constructor() {
|
| | this.init();
|
| | }
|
| |
|
| | init() {
|
| | const angle = Math.random() * Math.PI * 2;
|
| |
|
| | const r = 5 + Math.random() * 15;
|
| | this.x = width / 2 + Math.cos(angle) * r;
|
| | this.y = height / 2 + Math.sin(angle) * r;
|
| | this.vx = Math.cos(angle) * 2;
|
| | this.vy = Math.sin(angle) * 2;
|
| | this.size = Math.random() * 1.6;
|
| | this.alpha = Math.random() * 0.5 + 0.2;
|
| | this.isEscaping = false;
|
| | }
|
| |
|
| | update(breath) {
|
| | const dx = this.x - width / 2;
|
| | const dy = this.y - height / 2;
|
| | const dist = Math.sqrt(dx * dx + dy * dy) || 1;
|
| |
|
| | if (isHovering) {
|
| | this.vx += (Math.random() - 0.5) * 4;
|
| | this.vy += (Math.random() - 0.5) * 4;
|
| | } else {
|
| |
|
| | const repulsionRadius = 45;
|
| | if (dist < repulsionRadius) {
|
| | const force = (repulsionRadius - dist) / repulsionRadius;
|
| | this.vx += (dx / dist) * force * 6;
|
| | this.vy += (dy / dist) * force * 6;
|
| | }
|
| |
|
| |
|
| | const pushBase = Math.pow(breath, 5) * 14;
|
| | const randomScatter = (Math.random() - 0.5) * pushBase * 0.5;
|
| | this.vx += (dx / dist) * pushBase + randomScatter;
|
| | this.vy += (dy / dist) * pushBase + randomScatter;
|
| |
|
| |
|
| | if (breath > 0.88 && Math.random() > 0.98) {
|
| | this.isEscaping = true;
|
| | }
|
| |
|
| |
|
| | const side = 180;
|
| | const cornerR = 80;
|
| |
|
| | if (!this.isEscaping) {
|
| | if (!isInsideRoundedSquare(this.x + this.vx, this.y + this.vy, side, cornerR)) {
|
| | this.vx *= -0.4;
|
| | this.vy *= -0.4;
|
| | this.vx += (Math.random() - 0.5) * 2;
|
| | this.vy += (Math.random() - 0.5) * 2;
|
| | }
|
| |
|
| |
|
| | const targetR = 140;
|
| | const pull = (targetR - dist) * 0.012;
|
| | this.vx += (dx / dist) * pull;
|
| | this.vy += (dy / dist) * pull;
|
| | }
|
| | }
|
| |
|
| | this.x += this.vx;
|
| | this.y += this.vy;
|
| |
|
| |
|
| | const friction = this.isEscaping ? 0.98 : 0.86;
|
| | this.vx *= friction;
|
| | this.vy *= friction;
|
| |
|
| |
|
| | if (dist > width * 0.6 || (this.isEscaping && Math.abs(this.vx) < 0.05)) {
|
| | this.init();
|
| | }
|
| | }
|
| |
|
| | draw(breath) {
|
| | const b = 180 + breath * 75;
|
| | const finalAlpha = this.isEscaping ? this.alpha * 0.4 : this.alpha;
|
| | ctx.fillStyle = `rgba(${b}, ${b}, ${b}, ${finalAlpha})`;
|
| | ctx.fillRect(this.x, this.y, this.size, this.size);
|
| | }
|
| | }
|
| |
|
| | function init() {
|
| | width = canvas.width = window.innerWidth;
|
| | height = canvas.height = window.innerHeight;
|
| | particles = [];
|
| | for (let i = 0; i < particleCount; i++) particles.push(new Particle());
|
| | }
|
| |
|
| | function drawCore(breath) {
|
| | if (isHovering) return;
|
| | const size = 10 + breath * 4;
|
| | ctx.save();
|
| | ctx.shadowBlur = 40 * breath;
|
| | ctx.shadowColor = '#fff';
|
| | ctx.fillStyle = '#fff';
|
| | ctx.beginPath();
|
| | ctx.arc(width / 2, height / 2, size, 0, Math.PI * 2);
|
| | ctx.fill();
|
| | ctx.restore();
|
| | }
|
| |
|
| | function animate() {
|
| | ctx.fillStyle = 'rgba(0, 0, 0, 0.25)';
|
| | ctx.fillRect(0, 0, width, height);
|
| |
|
| | noiseTimer += 0.04;
|
| | const breath = Math.pow((Math.sin(noiseTimer) + 1) / 2, 2);
|
| |
|
| | if (isHovering) {
|
| | loginForm.classList.add('visible');
|
| | } else {
|
| | loginForm.classList.remove('visible');
|
| | }
|
| |
|
| | particles.forEach(p => {
|
| | p.update(breath);
|
| | p.draw(breath);
|
| | });
|
| |
|
| | drawCore(breath);
|
| | requestAnimationFrame(animate);
|
| | }
|
| |
|
| | init();
|
| | updateTimer();
|
| | animate();
|
| | </script>
|
| | </body>
|
| | </html> |