Spawns multiple bouncy 3D objects with shape selection, corrected collision, and a settings UI.
// ==UserScript== // @name Bouncy Shaded Cube Final Release // @namespace http://tampermonkey.net/ // @version 3.0 // @description Spawns multiple bouncy 3D objects with shape selection, corrected collision, and a settings UI. // @author alisteveman12 // @match *://*/* // @grant GM_addStyle // @grant GM_setValue // @grant GM_getValue // @require https://cdnjs.cloudflare.com/ajax/libs/three.js/r128/three.min.js // @require https://cdn.jsdelivr.net/npm/[email protected]/examples/js/loaders/GLTFLoader.js // @run-at document-idle // @icon  // ==/UserScript== (function( ) { 'use strict'; // Check if THREE is loaded. If not, wait and check again. if (typeof window.THREE === 'undefined') { window.setTimeout(arguments.callee, 100); return; } // --- Configuration (Stored in GM_values) --- const DEFAULT_SETTINGS = { SHAPE: 'Cube', // New setting CUBE_SIZE: 50, SPEED: 2, BOUNCE_HEIGHT: 10, BOUNCE_SPEED: 0.05, MAX_CUBES: 10, COLLISION_ENABLED: true, CUBE_COLOR: 0x00ff00 // Default green color }; let settings = {}; // --- Global State Variables --- let cubeIsRunning = false; let animationFrameId = null; let canvas = null; let renderer = null; let scene = null; let camera = null; let cubes = []; let cubeButton = null; let spawnMoreButton = null; let deleteCubeButton = null; let settingsButton = null; let settingsPanel = null; let width = window.innerWidth; let height = window.innerHeight; // --- Cube Class (to manage multiple cubes) --- class BouncyCube { constructor(color) { let geometry; const size = settings.CUBE_SIZE; switch (settings.SHAPE) { case 'Sphere': geometry = new THREE.SphereGeometry(size / 2, 32, 32); break; case 'Tetrahedron': geometry = new THREE.TetrahedronGeometry(size / 2); break; case 'Custom': // --- Placeholder for Custom Model Loading --- console.warn("Custom Model selected, but not yet implemented. Defaulting to Cube."); geometry = new THREE.BoxGeometry(size, size, size); break; case 'Cube': default: geometry = new THREE.BoxGeometry(size, size, size); break; } this.geometry = geometry; const objectColor = color || Math.random() * 0xffffff; this.material = new THREE.MeshPhongMaterial({ color: objectColor, shininess: 30 }); this.mesh = new THREE.Mesh(this.geometry, this.material); scene.add(this.mesh); // Initial random position and velocity this.cubeX = Math.random() * width; this.cubeY = Math.random() * height; this.velocityX = (Math.random() > 0.5 ? 1 : -1) * (settings.SPEED + Math.random() * 1); this.velocityY = (Math.random() > 0.5 ? 1 : -1) * (settings.SPEED + Math.random() * 1); this.timeOffset = Math.random() * 100; // Set initial world position const worldPos = screenToWorld(this.cubeX, this.cubeY); this.mesh.position.x = worldPos.x; this.mesh.position.y = worldPos.y; } update() { // 1. Update position based on screen coordinates this.cubeX += this.velocityX; this.cubeY += this.velocityY; this.timeOffset += settings.BOUNCE_SPEED; // 2. Collision detection (on screen edges) const halfSize = settings.CUBE_SIZE / 2; if (this.cubeX + halfSize > width || this.cubeX - halfSize < 0) { this.velocityX = -this.velocityX; this.cubeX = Math.max(halfSize, Math.min(width - halfSize, this.cubeX)); } if (this.cubeY + halfSize > height || this.cubeY - halfSize < 0) { this.velocityY = -this.velocityY; this.cubeY = Math.max(halfSize, Math.min(height - halfSize, this.cubeY)); } // 3. Convert screen position to world position for rendering const worldPos = screenToWorld(this.cubeX, this.cubeY); this.mesh.position.x = worldPos.x; this.mesh.position.y = worldPos.y; // 4. "Bounce" effect (up and down along the Z-axis) this.mesh.position.z = Math.sin(this.timeOffset) * settings.BOUNCE_HEIGHT; // 5. Rotation for visual interest this.mesh.rotation.x += 0.01; this.mesh.rotation.y += 0.01; } dispose() { scene.remove(this.mesh); this.geometry.dispose(); this.material.dispose(); } } // --- Utility Functions --- function loadSettings() { settings = GM_getValue('bouncyCubeSettings', DEFAULT_SETTINGS); // Ensure all default keys exist in case of new settings settings = { ...DEFAULT_SETTINGS, ...settings }; } function saveSettings() { GM_setValue('bouncyCubeSettings', settings); } // Function to convert screen coordinates to Three.js world coordinates function screenToWorld(x, y) { const vector = new THREE.Vector3(); vector.set( (x / width) * 2 - 1, -(y / height) * 2 + 1, 0.5 ); vector.unproject(camera); const dir = vector.sub(camera.position).normalize(); const distance = -camera.position.z / dir.z; const pos = camera.position.clone().add(dir.multiplyScalar(distance)); return pos; } function onWindowResize() { if (!cubeIsRunning) return; width = window.innerWidth; height = window.innerHeight; // Update canvas size canvas.width = width; canvas.height = height; // Update camera aspect ratio camera.aspect = width / height; camera.updateProjectionMatrix(); // Update renderer size renderer.setSize(width, height); } function checkCollisions() { if (!settings.COLLISION_ENABLED) return; for (let i = 0; i < cubes.length; i++) { for (let j = i + 1; j < cubes.length; j++) { const cubeA = cubes[i]; const cubeB = cubes[j]; const dx = cubeA.cubeX - cubeB.cubeX; const dy = cubeA.cubeY - cubeB.cubeY; const distance = Math.sqrt(dx * dx + dy * dy); const minDistance = settings.CUBE_SIZE; if (distance < minDistance) { // Collision detected const angle = Math.atan2(dy, dx); const sin = Math.sin(angle); const cos = Math.cos(angle); // Rotate velocities to the collision axis const vA = { x: cubeA.velocityX, y: cubeA.velocityY }; const vB = { x: cubeB.velocityX, y: cubeB.velocityY }; const uA = { x: vA.x * cos + vA.y * sin, y: vA.y * cos - vA.x * sin }; const uB = { x: vB.x * cos + vB.y * sin, y: vB.y * cos - vB.x * sin }; // Swap x velocities (perfect elastic collision) const vFinalA = { x: uB.x, y: uA.y }; const vFinalB = { x: uA.x, y: uB.y }; // Rotate back to original coordinate system cubeA.velocityX = vFinalA.x * cos - vFinalA.y * sin; cubeA.velocityY = vFinalA.y * cos + vFinalA.x * sin; cubeB.velocityX = vFinalB.x * cos - vFinalB.y * sin; cubeB.velocityY = vFinalB.y * cos + vFinalB.x * sin; // Separation to prevent sticking (crucial fix) const overlap = minDistance - distance; const separationX = overlap * cos; const separationY = overlap * sin; // Move both cubes apart by half the overlap cubeA.cubeX += separationX * 0.5; cubeA.cubeY += separationY * 0.5; cubeB.cubeX -= separationX * 0.5; cubeB.cubeY -= separationY * 0.5; } } } } function animate() { animationFrameId = requestAnimationFrame(animate); // Update all cubes cubes.forEach(cube => cube.update()); // Check for collisions checkCollisions(); // Render the scene renderer.render(scene, camera); } function updateButtonVisibility() { const hasCubes = cubes.length > 0; if (spawnMoreButton) { spawnMoreButton.style.display = hasCubes ? 'block' : 'none'; spawnMoreButton.textContent = `Spawn More (${cubes.length}/${settings.MAX_CUBES})`; } if (deleteCubeButton) { deleteCubeButton.style.display = hasCubes ? 'block' : 'none'; deleteCubeButton.textContent = `Delete (${cubes.length})`; } if (settingsButton) { settingsButton.style.display = hasCubes ? 'block' : 'none'; // Also hide the settings panel if the button is hidden if (!hasCubes && settingsPanel) { settingsPanel.style.display = 'none'; } } // Update main toggle button text cubeButton.textContent = hasCubes ? 'Stop Objects' : 'Start Objects'; } function spawnCube() { if (cubes.length >= settings.MAX_CUBES) { alert(`Maximum number of objects (${settings.MAX_CUBES}) reached!`); return; } cubes.push(new BouncyCube(settings.CUBE_COLOR)); updateButtonVisibility(); } function deleteCube() { if (cubes.length === 0) { return; } // Remove the last spawned cube const cubeToRemove = cubes.pop(); cubeToRemove.dispose(); updateButtonVisibility(); // If all cubes are gone, stop the whole animation if (cubes.length === 0) { stopCube(); } } // --- Main Control Functions --- function startCube() { if (cubeIsRunning) return; // 1. Setup Canvas canvas = document.createElement('canvas'); canvas.id = 'bouncy-cube-canvas'; canvas.style.cssText = 'position: fixed; top: 0; left: 0; pointer-events: none; z-index: 99999;'; document.body.appendChild(canvas); width = window.innerWidth; height = window.innerHeight; canvas.width = width; canvas.height = height; // 2. Three.js Setup scene = new THREE.Scene(); camera = new THREE.PerspectiveCamera(75, width / height, 0.1, 1000); renderer = new THREE.WebGLRenderer({ canvas: canvas, alpha: true }); renderer.setSize(width, height); camera.position.z = 100; // 3. Lighting Setup const ambientLight = new THREE.AmbientLight(0x404040, 2); scene.add(ambientLight); const directionalLight = new THREE.DirectionalLight(0xffffff, 1); directionalLight.position.set(1, 1, 1).normalize(); scene.add(directionalLight); // 4. Spawn the initial cube cubes = []; spawnCube(); // 5. Start Animation and Listeners window.addEventListener('resize', onWindowResize, false); animate(); cubeIsRunning = true; updateButtonVisibility(); } function stopCube() { if (!cubeIsRunning) return; // 1. Stop Animation cancelAnimationFrame(animationFrameId); // 2. Remove Canvas and Listeners if (canvas && canvas.parentNode) { canvas.parentNode.removeChild(canvas); } window.removeEventListener('resize', onWindowResize, false); // 3. Dispose of all cubes cubes.forEach(cube => cube.dispose()); cubes = []; // 4. Cleanup Three.js objects renderer.dispose(); scene = null; camera = null; renderer = null; canvas = null; cubeIsRunning = false; updateButtonVisibility(); } function toggleCube() { if (cubeIsRunning) { stopCube(); } else { startCube(); } } function toggleSettingsPanel() { if (settingsPanel.style.display === 'block') { settingsPanel.style.display = 'none'; } else { settingsPanel.style.display = 'block'; populateSettingsPanel(); } } function handleSettingChange(event) { const key = event.target.id.replace('setting-', ''); let value; if (event.target.type === 'checkbox') { value = event.target.checked; } else if (event.target.type === 'color') { // Convert hex color string to number value = parseInt(event.target.value.substring(1), 16); } else if (event.target.tagName === 'SELECT') { value = event.target.value; } else { value = parseFloat(event.target.value); if (key === 'MAX_CUBES') { value = parseInt(event.target.value); } } settings[key] = value; saveSettings(); // Special case: If a geometry-affecting setting changes, re-initialize cubes if (cubeIsRunning && (key === 'CUBE_SIZE' || key === 'CUBE_COLOR' || key === 'SHAPE')) { // Stop and restart to apply new geometry/material stopCube(); startCube(); } updateButtonVisibility(); } function populateSettingsPanel() { const shapeOptions = [ { value: 'Cube', text: 'Cube' }, { value: 'Sphere', text: 'Sphere' }, { value: 'Tetrahedron', text: 'Tetrahedron (Triangle)' }, { value: 'Custom', text: 'Custom Model (Coming Soon)', disabled: true } ]; let shapeSelectOptions = shapeOptions.map(opt => `<option value="${opt.value}" ${settings.SHAPE === opt.value ? 'selected' : ''} ${opt.disabled ? 'disabled' : ''}>${opt.text}</option>` ).join(''); settingsPanel.innerHTML = ` <h3>Object Settings</h3> <div class="setting-group"> <label for="setting-SHAPE">Shape:</label> <select id="setting-SHAPE"> ${shapeSelectOptions} </select> </div> <div class="setting-group"> <label for="setting-CUBE_SIZE">Size:</label> <input type="range" id="setting-CUBE_SIZE" min="10" max="100" step="5" value="${settings.CUBE_SIZE}"> <span>${settings.CUBE_SIZE}</span> </div> <div class="setting-group"> <label for="setting-SPEED">Speed:</label> <input type="range" id="setting-SPEED" min="0.5" max="5" step="0.5" value="${settings.SPEED}"> <span>${settings.SPEED}</span> </div> <div class="setting-group"> <label for="setting-BOUNCE_HEIGHT">Bounce Height:</label> <input type="range" id="setting-BOUNCE_HEIGHT" min="0" max="50" step="5" value="${settings.BOUNCE_HEIGHT}"> <span>${settings.BOUNCE_HEIGHT}</span> </div> <div class="setting-group"> <label for="setting-BOUNCE_SPEED">Bounce Speed:</label> <input type="range" id="setting-BOUNCE_SPEED" min="0.01" max="0.2" step="0.01" value="${settings.BOUNCE_SPEED}"> <span>${settings.BOUNCE_SPEED}</span> </div> <div class="setting-group"> <label for="setting-MAX_CUBES">Max Objects:</label> <input type="number" id="setting-MAX_CUBES" min="1" max="50" value="${settings.MAX_CUBES}"> </div> <div class="setting-group"> <label for="setting-COLLISION_ENABLED">Collisions:</label> <input type="checkbox" id="setting-COLLISION_ENABLED" ${settings.COLLISION_ENABLED ? 'checked' : ''}> </div> <div class="setting-group"> <label for="setting-CUBE_COLOR">Color:</label> <input type="color" id="setting-CUBE_COLOR" value="#${settings.CUBE_COLOR.toString(16).padStart(6, '0')}"> </div> <button id="reset-settings">Reset to Default</button> `; // Add event listeners to all inputs settingsPanel.querySelectorAll('input, select').forEach(input => { input.addEventListener('input', handleSettingChange); // Update range span dynamically if (input.type === 'range') { input.addEventListener('input', (e) => { e.target.nextElementSibling.textContent = e.target.value; }); } }); // Add event listener for reset button settingsPanel.querySelector('#reset-settings').addEventListener('click', () => { if (confirm('Are you sure you want to reset all settings to default?')) { settings = DEFAULT_SETTINGS; saveSettings(); populateSettingsPanel(); // Re-populate with default values // Stop and restart to apply all defaults if (cubeIsRunning) { stopCube(); startCube(); } } }); } // --- Menu Button and Panel Setup --- function createMenuButtons() { // Consolidated CSS for all buttons and the settings panel GM_addStyle("#bouncy-cube-toggle, #spawn-more-cubes, #delete-cube, #settings-button { position: fixed; padding: 10px 15px; background-color: #333; color: white; border: none; border-radius: 5px; cursor: pointer; z-index: 100000; font-family: sans-serif; font-size: 14px; box-shadow: 0 2px 5px rgba(0,0,0,0.5); transition: background-color 0.3s; } #bouncy-cube-toggle:hover, #spawn-more-cubes:hover, #delete-cube:hover, #settings-button:hover { background-color: #555; } #bouncy-cube-toggle { bottom: 10px; right: 10px; } #spawn-more-cubes { bottom: 60px; right: 10px; } #delete-cube { bottom: 110px; right: 10px; background-color: #d9534f; } #delete-cube:hover { background-color: #c9302c; } #settings-button { bottom: 160px; right: 10px; background-color: #f0ad4e; } #settings-button:hover { background-color: #ec971f; } #bouncy-cube-settings-panel { position: fixed; bottom: 10px; right: 180px; width: 300px; background-color: #222; color: #fff; border: 1px solid #555; border-radius: 8px; padding: 15px; box-shadow: 0 4px 12px rgba(0,0,0,0.5); z-index: 99999; display: none; font-family: sans-serif; } #bouncy-cube-settings-panel h3 { margin-top: 0; border-bottom: 1px solid #555; padding-bottom: 10px; } #bouncy-cube-settings-panel .setting-group { display: flex; justify-content: space-between; align-items: center; margin-bottom: 10px; } #bouncy-cube-settings-panel input[type=\"range\"] { flex-grow: 1; margin: 0 10px; } #bouncy-cube-settings-panel input[type=\"number\"], #bouncy-cube-settings-panel input[type=\"color\"], #bouncy-cube-settings-panel select { width: 60px; color: #000; } #bouncy-cube-settings-panel button { width: 100%; margin-top: 10px; background-color: #5bc0de; color: white; } #bouncy-cube-settings-panel button:hover { background-color: #31b0d5; }"); // 1. Start/Stop Cube Button (Bottom) cubeButton = document.createElement('button'); cubeButton.id = 'bouncy-cube-toggle'; cubeButton.textContent = 'Start Objects'; cubeButton.addEventListener('click', toggleCube); document.body.appendChild(cubeButton); // 2. Spawn More Button (Middle-Bottom) spawnMoreButton = document.createElement('button'); spawnMoreButton.id = 'spawn-more-cubes'; spawnMoreButton.textContent = 'Spawn More'; spawnMoreButton.style.display = 'none'; spawnMoreButton.addEventListener('click', spawnCube); document.body.appendChild(spawnMoreButton); // 3. Delete Cube Button (Middle-Top) deleteCubeButton = document.createElement('button'); deleteCubeButton.id = 'delete-cube'; deleteCubeButton.textContent = 'Delete'; deleteCubeButton.style.display = 'none'; deleteCubeButton.addEventListener('click', deleteCube); document.body.appendChild(deleteCubeButton); // 4. Settings Button (Top) settingsButton = document.createElement('button'); settingsButton.id = 'settings-button'; settingsButton.textContent = 'Settings'; settingsButton.style.display = 'none'; // Initially hidden settingsButton.addEventListener('click', toggleSettingsPanel); document.body.appendChild(settingsButton); // 5. Settings Panel settingsPanel = document.createElement('div'); settingsPanel.id = 'bouncy-cube-settings-panel'; document.body.appendChild(settingsPanel); } // --- Initialization --- loadSettings(); createMenuButtons(); })();