您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Create a vertical and ultra-smooth Pong game on the Drawaria canvas
// ==UserScript== // @name Drawaria Pong Engine // @namespace http://tampermonkey.net/ // @version 1.4 // @description Create a vertical and ultra-smooth Pong game on the Drawaria canvas // @author YouTubeDrawaria // @match https://drawaria.online/* // @grant none // @icon  // @license MIT // ==/UserScript== (function() { 'use strict'; /* ---------- SHARED SYSTEM COMPONENTS (Optimized for Drawaria) ---------- */ let drawariaSocket = null; let drawariaCanvas = null; let drawariaCtx = null; // Cola de comandos optimizada con agrupamiento inteligente const commandQueue = []; let batchProcessor = null; const BATCH_SIZE = 10; // Aumentado ligeramente para más fluidez const BATCH_INTERVAL = 40; // Reducido el intervalo para enviar a ~25 FPS de red. // Intercepta WebSocket para capturar el socket del juego const originalWebSocketSend = WebSocket.prototype.send; WebSocket.prototype.send = function(...args) { if (!drawariaSocket && this.url && this.url.includes('drawaria')) { drawariaSocket = this; console.log('🔗 Drawaria WebSocket capturado para el juego de Pong.'); startBatchProcessor(); } return originalWebSocketSend.apply(this, args); }; function startBatchProcessor() { if (batchProcessor) return; batchProcessor = setInterval(() => { if (!drawariaSocket || drawariaSocket.readyState !== WebSocket.OPEN || commandQueue.length === 0) { return; } const batch = commandQueue.splice(0, BATCH_SIZE); batch.forEach(cmd => { try { drawariaSocket.send(cmd); } catch (e) { console.warn('⚠️ Fallo al enviar el comando:', e); } }); }, BATCH_INTERVAL); } /** * Función unificada para encolar comandos de dibujo (Filled Shapes). * @param {number} x1 - Coordenada X inicial (centro de la forma) * @param {number} y1 - Coordenada Y inicial (centro de la forma) * @param {string} color - Color del objeto (ej. '#FFFFFF') * @param {number} thickness - Grosor efectivo (negativo para relleno en el servidor) * @param {boolean} isServerCommand - Si se añade a la cola del servidor. */ function enqueueDrawCommand(x1, y1, color, thickness, isServerCommand = true) { if (!drawariaCanvas) return; const thickAbs = Math.abs(thickness); // Renderizado local para retroalimentación visual inmediata (MAX FPS) if (drawariaCtx) { drawariaCtx.fillStyle = color; // Usar fillStyle para mejor representación de formas llenas // Simulación de círculo (para la pelota) if (thickAbs < 100) { // Asumimos que grosor pequeño es la pelota drawariaCtx.beginPath(); drawariaCtx.arc(x1, y1, thickAbs / 2.5, 0, Math.PI * 2); drawariaCtx.fill(); } else { // Simulación de rectángulo para la paleta (aunque Drawaria lo dibuje como un óvalo) // Para la paleta, dibujamos el óvalo centrado con el grosor de la paleta. drawariaCtx.strokeStyle = color; drawariaCtx.lineWidth = thickAbs; drawariaCtx.lineCap = 'butt'; // Mejor para rectángulos simulados drawariaCtx.beginPath(); drawariaCtx.moveTo(x1, y1 - thickAbs / 2 + 0.1); // Pequeña línea vertical drawariaCtx.lineTo(x1, y1 + thickAbs / 2 - 0.1); drawariaCtx.stroke(); } } // SERVER COMMAND (para otros jugadores) if (isServerCommand && drawariaSocket) { const normX1 = (x1 / drawariaCanvas.width).toFixed(4); const normY1 = (y1 / drawariaCanvas.height).toFixed(4); // Usamos un desplazamiento mínimo para simular un "punto" para la forma rellena const normX2 = (x1 / drawariaCanvas.width + 0.0001).toFixed(4); const normY2 = (y1 / drawariaCanvas.height + 0.0001).toFixed(4); // La clave: grosor negativo para forma rellena const cmd = `42["drawcmd",0,[${normX1},${normY1},${normX2},${normY2},false,${-thickAbs},"${color}",0,0,{}]]`; commandQueue.push(cmd); } } /* ---------- PONG GAME LOGIC AND PHYSICS ---------- */ const PONG_CONSTANTS = { PADDLE_WIDTH: 15, PADDLE_HEIGHT: 120, PADDLE_SPEED: 400, BALL_RADIUS: 20, BALL_INITIAL_SPEED: 350, TIMESTEP: 1 / 60, // Física a 60 Hz MAX_VELOCITY: 800, WALL_OFFSET: 30, BACKGROUND_COLOR: '#000000', // Tinted black BOT_PADDLE_COLOR: '#FF0000', // Red PLAYER_PADDLE_COLOR: '#0000FF', // Blue BALL_COLOR: '#FFFFFF', // White RENDER_INTERVAL: 1000 / 30, // Red a 30 FPS (para comandos) // Grosor estandarizado (es el grosor negativo que se enviará al servidor) BALL_THICKNESS: 20 * 2.5, // ~50 PADDLE_THICKNESS: 120 * 1.05, // ~126 (ligeramente más que la altura) }; class VerticalPongGame { constructor() { this.isActive = false; this.lastTime = 0; this.renderTime = 0; this.score = { player: 0, bot: 0 }; this.input = { up: false, down: false }; this.player = null; this.bot = null; this.ball = null; this.W = 0; this.H = 0; this.init(); } init() { const checkGameReady = () => { const gameCanvas = document.getElementById('canvas'); if (gameCanvas) { drawariaCanvas = gameCanvas; drawariaCtx = gameCanvas.getContext('2d'); this.W = drawariaCanvas.width; this.H = drawariaCanvas.height; this.createGamePanel(); this.setupEventListeners(); // Llama a dibujar el fondo negro una única vez al inicio this.drawBackground(); console.log('✅ Drawaria Pong Game inicializado.'); } else { setTimeout(checkGameReady, 100); } }; checkGameReady(); } drawBackground() { // Dibuja el fondo negro de forma PERMANENTE en el servidor. enqueueDrawCommand( this.W / 2, this.H / 2, PONG_CONSTANTS.BACKGROUND_COLOR, Math.max(this.W, this.H) * 2, true // Enviar al servidor ); } // ... (createGamePanel, setupEventListeners, toggleGame, pauseGame remain the same) createGamePanel() { const existingPanel = document.getElementById('pong-game-panel'); if (existingPanel) existingPanel.remove(); const panel = document.createElement('div'); panel.id = 'pong-game-panel'; panel.style.cssText = ` position: fixed !important; top: 250px !important; right: 20px !important; width: 250px !important; z-index: 2147483647 !important; background: linear-gradient(135deg, #001f3f, #000a14) !important; border: 2px solid #0000FF !important; border-radius: 12px !important; color: white !important; font-family: 'Segoe UI', Arial, sans-serif !important; box-shadow: 0 0 20px rgba(0, 0, 255, 0.3) !important; padding: 15px !important; text-align: center !important; `; panel.innerHTML = ` <h3 style="margin-top: 0; color: #0000FF;">🔵 Vertical Pong Engine 🔴</h3> <div style="margin-bottom: 10px; font-size: 1.2em; font-weight: bold;"> Score: <span id="player-score">0</span> - <span id="bot-score">0</span> </div> <button id="toggle-game" style=" width: 100%; padding: 10px; background: #0000FF; color: white; border: none; border-radius: 8px; cursor: pointer; font-size: 14px; font-weight: bold; ">▶️ Start Pong</button> <div id="game-message" style=" margin-top: 10px; color: #FFD700; font-weight: bold; ">Ready to Play!</div> <div style=" margin-top: 15px; font-size: 10px; color: rgba(255,255,255,0.6); text-align: left; "> <p style="margin: 3px 0;">Control Player Paddle (Blue):</p> <ul style="padding-left: 20px; margin: 3px 0;"> <li>ArrowUp or W: Move Up</li> <li>ArrowDown or S: Move Down</li> </ul> </div> `; document.body.appendChild(panel); this.makePanelDraggable(panel); } setupEventListeners() { document.getElementById('toggle-game').addEventListener('click', () => this.toggleGame()); document.addEventListener('keydown', (e) => this.handleKeyInput(e, true)); document.addEventListener('keyup', (e) => this.handleKeyInput(e, false)); } toggleGame() { if (!this.isActive) { this.startGame(); document.getElementById('toggle-game').textContent = '⏸️ Pause Game'; document.getElementById('toggle-game').style.background = '#FFD700'; } else { this.pauseGame(); document.getElementById('toggle-game').textContent = '▶️ Resume Game'; document.getElementById('toggle-game').style.background = '#0000FF'; } } startGame() { if (this.isActive || !drawariaCanvas) return; this.isActive = true; this.score = { player: 0, bot: 0 }; this.resetGameObjects(); this.updateScoreDisplay(); // Re-draw background just in case was overdrawn this.drawBackground(); this.gameLoop(performance.now()); console.log('🚀 Pong Game Started!'); } pauseGame() { this.isActive = false; document.getElementById('game-message').textContent = 'Game Paused!'; console.log('🛑 Pong Game Paused.'); } resetGameObjects() { const W = this.W; const H = this.H; const P = PONG_CONSTANTS; // Initialize Player Paddle (Blue, Right Side) this.player = { id: 'player', color: P.PLAYER_PADDLE_COLOR, thickness: P.PADDLE_THICKNESS, x: W - P.WALL_OFFSET - P.PADDLE_WIDTH / 2, y: H / 2, W: P.PADDLE_WIDTH, H: P.PADDLE_HEIGHT, lastRenderX: W - P.WALL_OFFSET - P.PADDLE_WIDTH / 2, lastRenderY: H / 2 }; // Initialize Bot Paddle (Red, Left Side) this.bot = { id: 'bot', color: P.BOT_PADDLE_COLOR, thickness: P.PADDLE_THICKNESS, x: P.WALL_OFFSET + P.PADDLE_WIDTH / 2, y: H / 2, W: P.PADDLE_WIDTH, H: P.PADDLE_HEIGHT, lastRenderX: P.WALL_OFFSET + P.PADDLE_WIDTH / 2, lastRenderY: H / 2 }; this.resetBall(); } resetBall(direction = 1) { const W = this.W; const H = this.H; const P = PONG_CONSTANTS; // Reset ball to center this.ball = { id: 'ball', color: P.BALL_COLOR, thickness: P.BALL_THICKNESS, x: W / 2, y: H / 2, radius: P.BALL_RADIUS, lastRenderX: W / 2, lastRenderY: H / 2, // Initial velocity: Random vertical, fixed horizontal (direction * speed) vx: direction * P.BALL_INITIAL_SPEED, vy: (Math.random() * 2 - 1) * P.BALL_INITIAL_SPEED * 0.5 }; } gameLoop(currentTime) { if (!this.isActive) return; const dt = PONG_CONSTANTS.TIMESTEP; const timeElapsed = (currentTime - (this.lastTime || currentTime)) / 1000; this.lastTime = currentTime; // --- 1. LOCAL CLEAR (For local ultra-fluidity) --- if (drawariaCtx) { drawariaCtx.clearRect(0, 0, this.W, this.H); } // --- 2. Update Physics --- this.updatePhysics(dt); // --- 3. LOCAL RENDER (Draws locally at maximum FPS via enqueueDrawCommand) --- this.localRender(); // --- 4. NETWORK RENDER (Sends commands to server at fixed rate) --- if (currentTime - this.renderTime >= PONG_CONSTANTS.RENDER_INTERVAL) { this.networkRender(); this.renderTime = currentTime; } requestAnimationFrame((t) => this.gameLoop(t)); } localRender() { // Draw background locally (optional, but ensures black background on local client) enqueueDrawCommand( this.W / 2, this.H / 2, PONG_CONSTANTS.BACKGROUND_COLOR, Math.max(this.W, this.H) * 2, false // NO enviar al servidor ); this.drawObject(this.player, false); this.drawObject(this.bot, false); this.drawObject(this.ball, false); } networkRender() { const objects = [this.player, this.bot, this.ball]; objects.forEach(obj => { const isMoved = Math.abs(obj.x - obj.lastRenderX) > 1 || Math.abs(obj.y - obj.lastRenderY) > 1; if (isMoved || obj.lastRenderX === -9999) { // Step 1: ERASE (Dirty Erase) at old position using background color if (obj.lastRenderX !== -9999) { this.drawObject(obj, true, obj.lastRenderX, obj.lastRenderY, PONG_CONSTANTS.BACKGROUND_COLOR); } // Step 2: DRAW at new position this.drawObject(obj, true, obj.x, obj.y, obj.color); // Update last render position obj.lastRenderX = obj.x; obj.lastRenderY = obj.y; } }); } drawObject(obj, isServer, x, y, color) { const drawX = x !== undefined ? x : obj.x; const drawY = y !== undefined ? y : obj.y; const drawColor = color !== undefined ? color : obj.color; enqueueDrawCommand(drawX, drawY, drawColor, obj.thickness, isServer); } updatePhysics(dt) { if (!this.ball || !this.player || !this.bot) return; // --- 1. Update Player Paddle Position --- let playerNewY = this.player.y; if (this.input.up) { playerNewY -= PONG_CONSTANTS.PADDLE_SPEED * dt; } if (this.input.down) { playerNewY += PONG_CONSTANTS.PADDLE_SPEED * dt; } // Clamp player paddle position const playerMinY = PONG_CONSTANTS.PADDLE_HEIGHT / 2; const playerMaxY = this.H - PONG_CONSTANTS.PADDLE_HEIGHT / 2; this.player.y = Math.max(playerMinY, Math.min(playerMaxY, playerNewY)); // --- 2. Update Bot Paddle Position (Simple AI) --- const botCenter = this.bot.y; const ballCenter = this.ball.y; const deltaY = ballCenter - botCenter; const botSpeed = PONG_CONSTANTS.PADDLE_SPEED * 0.7; if (Math.abs(deltaY) > 20) { this.bot.y += Math.sign(deltaY) * Math.min(Math.abs(deltaY), botSpeed * dt); } // Clamp bot paddle position const botMinY = PONG_CONSTANTS.PADDLE_HEIGHT / 2; const botMaxY = this.H - PONG_CONSTANTS.PADDLE_HEIGHT / 2; this.bot.y = Math.max(botMinY, Math.min(botMaxY, this.bot.y)); // --- 3. Update Ball Position --- this.ball.x += this.ball.vx * dt; this.ball.y += this.ball.vy * dt; // --- 4. Ball Wall Collision (Top/Bottom) --- const minH = this.ball.radius; const maxH = this.H - this.ball.radius; if (this.ball.y < minH || this.ball.y > maxH) { this.ball.vy *= -1; this.ball.y = this.ball.y < minH ? minH : maxH; } // --- 5. Ball Paddle Collision (Player & Bot) --- this.checkPaddleCollision(this.player); this.checkPaddleCollision(this.bot); // --- 6. Ball Goal Check (Left/Right) --- this.checkGoal(); } checkPaddleCollision(paddle) { const ball = this.ball; const paddleHalfH = paddle.H / 2; // Horizontal intersection point (closest edge of the paddle to the ball) let paddleHitX; if (paddle === this.player) { paddleHitX = paddle.x - paddle.W / 2; // Left edge of player paddle } else { // bot paddleHitX = paddle.x + paddle.W / 2; // Right edge of bot paddle } // Vertical check: within paddle Y-bounds const isCollidingY = ball.y + ball.radius > paddle.y - paddleHalfH && ball.y - ball.radius < paddle.y + paddleHalfH; // Horizontal check: ball hits the side of the paddle const isCollidingX = (ball.vx > 0 && ball.x + ball.radius >= paddleHitX && paddle === this.player) || (ball.vx < 0 && ball.x - ball.radius <= paddleHitX && paddle === this.bot); if (isCollidingX && isCollidingY) { // Positional correction: Push ball out to prevent multi-hit if (paddle === this.player) { ball.x = paddleHitX - ball.radius; } else { ball.x = paddleHitX + ball.radius; } // 1. Reverse horizontal direction ball.vx *= -1; // 2. Angle/Vertical speed adjustment const relativeIntersectY = (ball.y - paddle.y); const normalizedRelativeIntersectionY = relativeIntersectY / paddleHalfH; // -1 to 1 const bounceAngle = normalizedRelativeIntersectionY * (Math.PI / 4); const speed = Math.sqrt(ball.vx * ball.vx + ball.vy * ball.vy) * 1.05; ball.vx = Math.cos(bounceAngle) * speed * Math.sign(ball.vx); ball.vy = Math.sin(bounceAngle) * speed; // 3. Limit max speed const currentSpeed = Math.sqrt(ball.vx * ball.vx + ball.vy * ball.vy); if (currentSpeed > PONG_CONSTANTS.MAX_VELOCITY) { ball.vx = (ball.vx / currentSpeed) * PONG_CONSTANTS.MAX_VELOCITY; ball.vy = (ball.vy / currentSpeed) * PONG_CONSTANTS.MAX_VELOCITY; } } } checkGoal() { const W = this.W; const ballX = this.ball.x; if (ballX < 0) { // Goal scored by Player (passed left side) this.score.player++; this.updateScoreDisplay(); document.getElementById('game-message').textContent = 'GOAL! Player Scores!'; this.resetBall(-1); // Start ball towards the bot (left) } else if (ballX > W) { // Goal scored by Bot (passed right side) this.score.bot++; this.updateScoreDisplay(); document.getElementById('game-message').textContent = 'GOAL! Bot Scores!'; this.resetBall(1); // Start ball towards the player (right) } } // ... (handleKeyInput, updateScoreDisplay, makePanelDraggable remain the same) handleKeyInput(e, isKeyDown) { const key = e.key.toLowerCase(); if (key === 'arrowup' || key === 'w') { this.input.up = isKeyDown; e.preventDefault(); } if (key === 'arrowdown' || key === 's') { this.input.down = isKeyDown; e.preventDefault(); } } updateScoreDisplay() { document.getElementById('player-score').textContent = this.score.player; document.getElementById('bot-score').textContent = this.score.bot; document.getElementById('game-message').textContent = `Player: ${this.score.player} | Bot: ${this.score.bot}`; } makePanelDraggable(panel) { let isDragging = false; let currentX, currentY, initialX, initialY, xOffset = 0, yOffset = 0; const header = panel.querySelector('h3'); if (!header) return; const dragStart = (e) => { initialX = e.clientX - xOffset; initialY = e.clientY - yOffset; isDragging = true; panel.style.cursor = 'grabbing'; }; const dragEnd = () => { initialX = currentX; initialY = currentY; isDragging = false; panel.style.cursor = 'grab'; }; const drag = (e) => { if (isDragging) { e.preventDefault(); currentX = e.clientX - initialX; currentY = e.clientY - initialY; xOffset = currentX; yOffset = currentY; panel.style.transform = `translate3d(${currentX}px, ${currentY}px, 0)`; } }; header.style.cursor = 'grab'; header.addEventListener("mousedown", dragStart); document.addEventListener("mouseup", dragEnd); document.addEventListener("mousemove", drag); } } // Initialization of the Pong Game const initPongGame = () => { const pongGame = new VerticalPongGame(); }; if (document.readyState === 'loading') { document.addEventListener('DOMContentLoaded', initPongGame); } else { setTimeout(initPongGame, 500); } })();