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