Drawaria Pong Engine

Create a vertical and ultra-smooth Pong game on the Drawaria canvas

您需要先安裝使用者腳本管理器擴展,如 TampermonkeyGreasemonkeyViolentmonkey 之後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyUserscripts 後才能安裝該腳本。

你需要先安裝一款使用者腳本管理器擴展,比如 Tampermonkey,才能安裝此腳本

您需要先安裝使用者腳本管理器擴充功能後才能安裝該腳本。

(我已經安裝了使用者腳本管理器,讓我安裝!)

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

(我已經安裝了使用者樣式管理器,讓我安裝!)

// ==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);
    }

})();