您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Adds comets with falling hearts and particle effects along with beautiful SVG decorations to drawaria.online.
// ==UserScript== // @name Drawaria Hearts Love Mod // @namespace http://tampermonkey.net/ // @version 1.0 // @description Adds comets with falling hearts and particle effects along with beautiful SVG decorations to drawaria.online. // @author YouTubeDrawaria // @match https://drawaria.online/* // @icon  // @grant none // @license MIT // ==/UserScript== (function() { 'use strict'; // Create a canvas overlay covering the page. const canvas = document.createElement('canvas'); canvas.id = 'heartsCanvas'; canvas.style.position = 'fixed'; canvas.style.top = '0'; canvas.style.left = '0'; canvas.style.width = '100%'; canvas.style.height = '100%'; canvas.style.pointerEvents = 'none'; canvas.style.zIndex = '9999'; document.body.appendChild(canvas); const ctx = canvas.getContext('2d'); // Resize canvas to match window size. function resizeCanvas() { canvas.width = window.innerWidth; canvas.height = window.innerHeight; } window.addEventListener('resize', resizeCanvas); resizeCanvas(); // Arrays to hold comet and heart particles. const comets = []; const hearts = []; // Utility: random value between min and max. function rand(min, max) { return Math.random() * (max - min) + min; } // Define a comet that moves diagonally across the canvas. class Comet { constructor() { // Start at a random position at the top of the screen. this.x = rand(-100, canvas.width); this.y = rand(-50, 0); // Velocity moving diagonally down and right. this.vx = rand(2, 5); this.vy = rand(2, 4); // Lifespan for when to remove the comet. this.life = rand(150, 250); this.spawnRate = rand(5, 10); // How many frames between spawning hearts. this.frameCounter = 0; } update() { this.x += this.vx; this.y += this.vy; this.life--; this.frameCounter++; // Periodically spawn a heart particle. if (this.frameCounter >= this.spawnRate) { this.frameCounter = 0; hearts.push(new Heart(this.x, this.y)); } } draw(ctx) { // Draw comet core (a glowing circle) ctx.save(); ctx.shadowBlur = 20; ctx.shadowColor = 'rgba(255,255,255,0.8)'; ctx.fillStyle = 'rgba(255, 200, 50, 0.9)'; ctx.beginPath(); ctx.arc(this.x, this.y, 8, 0, Math.PI * 2); ctx.fill(); ctx.restore(); // Draw comet tail (fading streak) ctx.save(); const grad = ctx.createLinearGradient(this.x, this.y, this.x - this.vx * 10, this.y - this.vy * 10); grad.addColorStop(0, 'rgba(255,200,50,0.8)'); grad.addColorStop(1, 'rgba(255,200,50,0)'); ctx.strokeStyle = grad; ctx.lineWidth = 4; ctx.beginPath(); ctx.moveTo(this.x, this.y); ctx.lineTo(this.x - this.vx * 10, this.y - this.vy * 10); ctx.stroke(); ctx.restore(); } } // Define a heart particle that falls and rotates. class Heart { constructor(x, y) { // Start position is given by the comet. this.x = x; this.y = y; // Heart particle falls gradually with a slight drift. this.vx = rand(-1, 1); this.vy = rand(1, 3); this.rotation = rand(0, Math.PI * 2); this.rotationSpeed = rand(-0.1, 0.1); this.size = rand(10, 20); this.life = 100; // Frames to live this.opacity = 1; // Random color for flair. const r = Math.floor(rand(150, 255)); const g = Math.floor(rand(50, 150)); const b = Math.floor(rand(150, 255)); this.color = `rgba(${r}, ${g}, ${b},`; } update() { this.x += this.vx; this.y += this.vy; this.rotation += this.rotationSpeed; this.life--; this.opacity = this.life / 100; } draw(ctx) { ctx.save(); ctx.translate(this.x, this.y); ctx.rotate(this.rotation); ctx.scale(this.size/25, this.size/25); ctx.beginPath(); // Draw a heart shape using bezier curves. ctx.moveTo(0, 0); ctx.bezierCurveTo(0, -3, -5, -15, -25, -15); ctx.bezierCurveTo(-55, -15, -55, 22.5, -55, 22.5); ctx.bezierCurveTo(-55, 40, -35, 62, 0, 80); ctx.bezierCurveTo(35, 62, 55, 40, 55, 22.5); ctx.bezierCurveTo(55, 22.5, 55, -15, 25, -15); ctx.bezierCurveTo(10, -15, 0, -3, 0, 0); ctx.fillStyle = this.color + this.opacity + ')'; ctx.fill(); ctx.restore(); } } // Create decorative background SVG elements. function addSVGDecorations() { const svgNS = "http://www.w3.org/2000/svg"; const svg = document.createElementNS(svgNS, 'svg'); svg.setAttribute('width', '100%'); svg.setAttribute('height', '100%'); svg.style.position = 'fixed'; svg.style.top = 0; svg.style.left = 0; svg.style.zIndex = '0'; svg.style.pointerEvents = 'none'; svg.innerHTML = ` <defs> <radialGradient id="grad1" cx="50%" cy="50%" r="50%"> <stop offset="0%" style="stop-color:rgba(255,255,255,0.8);stop-opacity:1"/> <stop offset="100%" style="stop-color:rgba(0,150,255,0.2);stop-opacity:0"/> </radialGradient> </defs> <!-- Star-like circles --> <circle cx="10%" cy="20%" r="50" fill="url(#grad1)" /> <circle cx="80%" cy="30%" r="70" fill="url(#grad1)" /> <circle cx="30%" cy="70%" r="40" fill="url(#grad1)" /> <circle cx="90%" cy="80%" r="60" fill="url(#grad1)" /> <!-- Decorative paths --> <path d="M0,100 Q50,150 100,100 T200,100" stroke="rgba(255, 255, 255, 0.2)" stroke-width="4" fill="none"/> `; document.body.insertBefore(svg, canvas); } addSVGDecorations(); // Main animation loop. function animate() { ctx.clearRect(0, 0, canvas.width, canvas.height); // Update and draw comets. for (let i = comets.length - 1; i >= 0; i--) { const comet = comets[i]; comet.update(); comet.draw(ctx); // Remove comet if its lifespan is over or it goes out of view. if (comet.life <= 0 || comet.x > canvas.width + 50 || comet.y > canvas.height + 50) { comets.splice(i, 1); } } // Update and draw heart particles. for (let i = hearts.length - 1; i >= 0; i--) { const heart = hearts[i]; heart.update(); heart.draw(ctx); if (heart.life <= 0) { hearts.splice(i, 1); } } // Randomly add new comets. if (Math.random() < 0.02) { // adjust spawn rate here comets.push(new Comet()); } requestAnimationFrame(animate); } animate(); })();