help@rskworld.in +91 93305 39277
RSK World
  • Home
  • Development
    • Web Development
    • Mobile Apps
    • Software
    • Games
    • Project
  • Technologies
    • Data Science
    • AI Development
    • Cloud Development
    • Blockchain
    • Cyber Security
    • Dev Tools
    • Testing Tools
  • About
  • Contact

Theme Settings

Color Scheme
Display Options
Font Size
100%
Back to Project
RSK World
star-pattern-maker
/
js
RSK World
star-pattern-maker
Star Pattern Maker - HTML5 Canvas + 3D Rendering + Physics Simulation + AI Patterns + Generative Audio + Modern UI + Glassmorphism Design
js
  • 3d-engine.js3.8 KB
  • ai-patterns.js5.6 KB
  • animations.js1.4 KB
  • audio.js1.3 KB
  • controls.js12.2 KB
  • export.js3.9 KB
  • filters.js3.5 KB
  • fractals.js3.5 KB
  • gallery.js2.5 KB
  • main.js1.9 KB
  • particles.js1.4 KB
  • physics.js1.7 KB
  • presets.js2.8 KB
  • randomizer.js1.4 KB
  • renderer.js7 KB
  • shaders.js6.5 KB
  • shapes.js1.7 KB
  • sound-gen.js4.4 KB
  • state.js1.9 KB
  • symmetry.js3.9 KB
  • themes.js1.4 KB
  • timeline.js4.2 KB
  • utils.js546 B
controls.js
js/controls.js
Raw Download
Find: Go to:
/* V7 Controls Logic */
import { state } from './state.js';
import { exportPNG, toggleVideoRecording, recordGIF, exportSVG } from './export.js';
import { initAudio } from './audio.js';
import { savePattern } from './gallery.js';
import { THEMES, applyTheme } from './themes.js';
import { randomizeState } from './randomizer.js';
import { PRESETS, applyPreset } from './presets.js';
import { updateAnimations } from './animations.js';

export function initControls(canvas, ctx) {
    const controls = {
        spikes: document.getElementById('spikes'),
        outerRadius: document.getElementById('outerRadius'),
        innerRadius: document.getElementById('innerRadius'),
        curve: document.getElementById('curve'),
        twist: document.getElementById('twist'),
        lineWidth: document.getElementById('lineWidth'),
        glow: document.getElementById('glow'),
        jitter: document.getElementById('jitter'),
        speed: document.getElementById('speed'),

        wallpaperMode: document.getElementById('wallpaperMode'),
        cols: document.getElementById('cols'),
        rows: document.getElementById('rows'),

        kaleidoscopeMode: document.getElementById('kaleidoscopeMode'),
        segments: document.getElementById('segments'),

        audioMode: document.getElementById('audioMode'),

        fillColor: document.getElementById('fillColor'),
        fillColor2: document.getElementById('fillColor2'),
        useGradient: document.getElementById('useGradient'),
        strokeColor: document.getElementById('strokeColor'),
        bgColor: document.getElementById('bgColor'),

        // New Particles
        particlesEnabled: document.getElementById('particlesEnabled'),

        // V8 FX
        rainbowMode: document.getElementById('rainbowMode'),
        pulseEnabled: document.getElementById('pulseEnabled'),
        pulseEnabled: document.getElementById('pulseEnabled'),
        parallaxEnabled: document.getElementById('parallaxEnabled'),

        // V9
        fractalMode: document.getElementById('fractalMode'),
        fractalDepth: document.getElementById('fractalDepth'),

        // V11
        trailEnabled: document.getElementById('trailEnabled'),
        echoEnabled: document.getElementById('echoEnabled'),

        // V12
        gravityEnabled: document.getElementById('gravityEnabled'),
        orbitEnabled: document.getElementById('orbitEnabled'),
        chromaticAberration: document.getElementById('chromaticAberration'),
        holographicMode: document.getElementById('holographicMode'),

        // V13
        glitchEnabled: document.getElementById('glitchEnabled'),
        scanlinesEnabled: document.getElementById('scanlinesEnabled'),
        vignetteEnabled: document.getElementById('vignetteEnabled'),
        bloomEnabled: document.getElementById('bloomEnabled'),
        morphEnabled: document.getElementById('morphEnabled'),
        colorShiftEnabled: document.getElementById('colorShiftEnabled'),
        mandalaMode: document.getElementById('mandalaMode'),
        spiralMode: document.getElementById('spiralMode')
    };

    const displays = {
        spikes: document.getElementById('spikesVal'),
        outerRadius: document.getElementById('outerRadiusVal'),
        innerRadius: document.getElementById('innerRadiusVal'),
        curve: document.getElementById('curveVal'),
        twist: document.getElementById('twistVal'),
        lineWidth: document.getElementById('lineWidthVal'),
        glow: document.getElementById('glowVal'),
        jitter: document.getElementById('jitterVal'),
        speed: document.getElementById('speedVal'),
        cols: document.getElementById('colsVal'),
        rows: document.getElementById('rowsVal'),
        segments: document.getElementById('segmentsVal'),
        fractalDepth: document.getElementById('depthVal')
    };

    function updateState(key, value) {
        if (['useGradient', 'wallpaperMode', 'kaleidoscopeMode', 'audioMode', 'particlesEnabled', 'rainbowMode', 'pulseEnabled', 'parallaxEnabled', 'fractalMode', 'trailEnabled', 'echoEnabled', 'gravityEnabled', 'orbitEnabled', 'chromaticAberration', 'holographicMode', 'glitchEnabled', 'scanlinesEnabled', 'vignetteEnabled', 'bloomEnabled', 'morphEnabled', 'colorShiftEnabled', 'mandalaMode', 'spiralMode'].includes(key)) {
            state[key] = value;
        } else {
            state[key] = (key.includes('Color')) ? value : parseInt(value);
        }

        if (displays[key]) displays[key].textContent = state[key];

        if (key === 'audioMode' && value === true) initAudio();

        // Trigger draw if window._triggerDraw exists
        if (window._triggerDraw) window._triggerDraw();

        // Start loop if needed
        const needsLoop = state.speed > 0 || state.audioMode || state.particlesEnabled;
        if (needsLoop && window._triggerLoop) window._triggerLoop();
    }

    // Attach Listeners
    Object.keys(controls).forEach(key => {
        const input = controls[key];
        if (!input) return;

        const eventType = (input.type === 'checkbox' || input.type.includes('color')) ? 'input' : 'input';

        input.addEventListener(eventType, (e) => {
            const val = (input.type === 'checkbox') ? input.checked : input.value;
            updateState(key, val);
        });
    });

    // Buttons
    document.getElementById('exportBtn').addEventListener('click', () => exportPNG(canvas));
    document.getElementById('recordBtn').addEventListener('click', (e) => toggleVideoRecording(canvas, e.target));
    document.getElementById('recordGifBtn').addEventListener('click', (e) => recordGIF(canvas, e.target));
    document.getElementById('exportSvgBtn').addEventListener('click', () => exportSVG(canvas));

    // Gallery Save
    // Gallery Save (V10 - Auto Name)
    function triggerSave() {
        const name = `Pattern ${new Date().toLocaleString()}`;
        savePattern(name, canvas.toDataURL('image/png', 0.2));

        // Simple Toast Feedback (Console for now, or UI)
        const btn = document.getElementById('savePatternBtn');
        const originalText = btn.innerHTML;
        btn.innerHTML = '<i class="fas fa-check"></i> Saved!';
        setTimeout(() => btn.innerHTML = originalText, 1500);
    }
    document.getElementById('savePatternBtn').addEventListener('click', triggerSave);

    // Randomize (V10)
    function triggerRandom() {
        randomizeState(state);
        // Sync UI toggles/inputs
        Object.keys(controls).forEach(k => {
            if (controls[k]) {
                if (controls[k].type === 'checkbox') controls[k].checked = state[k];
                else controls[k].value = state[k];
                if (displays[k]) displays[k].textContent = state[k];
            }
        });
        if (window._triggerDraw) window._triggerDraw();
    }
    // Bind if button exists (we need to add it to HTML)
    const rndBtn = document.getElementById('randomizeBtn');
    if (rndBtn) rndBtn.addEventListener('click', triggerRandom);

    // Keyboard Shortcuts (V10)
    window.addEventListener('keydown', (e) => {
        // Ignore if typing in an input
        if (e.target.tagName === 'INPUT') return;

        switch (e.key.toLowerCase()) {
            case ' ': // Space: Toggle Animation
                state.speed = state.speed > 0 ? 0 : 2;
                if (controls.speed) controls.speed.value = state.speed;
                if (displays.speed) displays.speed.textContent = state.speed;
                if (state.speed > 0 && window._triggerLoop) window._triggerLoop();
                e.preventDefault();
                break;
            case 'r': // R: Randomize
                triggerRandom();
                break;
            case 's': // S: Save
                if (e.ctrlKey || e.metaKey) { // Ctrl+S usually saves page, let's override? 
                    e.preventDefault();
                    triggerSave();
                } else {
                    triggerSave();
                }
                break;
        }
    });

    // Listen for Load Event
    window.addEventListener('loadPattern', (e) => {
        const id = e.detail;
        // logic is handled in gallery.js loading to State, but we need to update UI controls
        // This is tricky. Reactivity usually handles this.
        // We need to sync Controls -> State.
        // Let's reload page? No.
        // Let's manually require a sync function.
        import('./gallery.js').then(({ loadPattern }) => {
            if (loadPattern(id)) {
                // Sync UI
                Object.keys(controls).forEach(k => {
                    if (controls[k]) {
                        if (controls[k].type === 'checkbox') controls[k].checked = state[k];
                        else controls[k].value = state[k];
                        if (displays[k]) displays[k].textContent = state[k];
                    }
                });
                if (window._triggerDraw) window._triggerDraw();
            }
        });
    });

    // Parallax Mouse Tracking (V8)
    const previewArea = document.getElementById('previewContainer');
    if (previewArea) {
        previewArea.addEventListener('mousemove', (e) => {
            if (!state.parallaxEnabled) return;

            const rect = previewArea.getBoundingClientRect();
            // Normalize -1 to 1
            const x = (e.clientX - rect.left) / rect.width;
            const y = (e.clientY - rect.top) / rect.height;

            state.mouseX = (x - 0.5) * 2;
            state.mouseY = (y - 0.5) * 2;

            if (window._triggerDraw) window._triggerDraw();
        });

        previewArea.addEventListener('mouseleave', () => {
            state.mouseX = 0;
            state.mouseY = 0;
            if (window._triggerDraw) window._triggerDraw();
        });
    }

    // Generate Themes (V9)
    const themeGrid = document.getElementById('themeGrid');
    if (themeGrid) {
        THEMES.forEach((t, i) => {
            const btn = document.createElement('div');
            btn.style.width = '30px';
            btn.style.height = '30px';
            btn.style.borderRadius = '50%';
            btn.style.background = `linear-gradient(135deg, ${t.fillColor}, ${t.fillColor2})`;
            btn.style.border = '2px solid rgba(255,255,255,0.2)';
            btn.style.cursor = 'pointer';
            btn.title = t.name;
            btn.onclick = () => applyTheme(i);
            themeGrid.appendChild(btn);
        });
    }

    // Listen for Theme Change to Sync UI
    window.addEventListener('themeChanged', (e) => {
        const t = e.detail;
        if (controls.fillColor) controls.fillColor.value = t.fillColor;
        if (controls.fillColor2) controls.fillColor2.value = t.fillColor2;
        if (controls.strokeColor) controls.strokeColor.value = t.strokeColor;
        if (controls.bgColor) controls.bgColor.value = t.bgColor;
        if (controls.glow) controls.glow.value = t.glow;
        if (displays.glow) displays.glow.textContent = t.glow;

        if (window._triggerDraw) window._triggerDraw();
    });

    // Generate Presets (V12)
    const presetGrid = document.getElementById('presetGrid');
    if (presetGrid) {
        PRESETS.forEach((p, i) => {
            const btn = document.createElement('button');
            btn.className = 'btn btn-outline';
            btn.style.width = '100%';
            btn.style.padding = '0.5rem';
            btn.style.fontSize = '0.85rem';
            btn.textContent = p.name;
            btn.onclick = () => {
                applyPreset(i);
                // Sync UI
                Object.keys(controls).forEach(k => {
                    if (controls[k] && state.hasOwnProperty(k)) {
                        if (controls[k].type === 'checkbox') controls[k].checked = state[k];
                        else if (controls[k].type === 'range' || controls[k].type === 'color') controls[k].value = state[k];
                        if (displays[k]) displays[k].textContent = state[k];
                    }
                });
                if (window._triggerDraw) window._triggerDraw();
            };
            presetGrid.appendChild(btn);
        });
    }
}
287 lines•12.2 KB
javascript

About RSK World

Founded by Molla Samser, with Designer & Tester Rima Khatun, RSK World is your one-stop destination for free programming resources, source code, and development tools.

Founder: Molla Samser
Designer & Tester: Rima Khatun

Development

  • Game Development
  • Web Development
  • Mobile Development
  • AI Development
  • Development Tools

Legal

  • Terms & Conditions
  • Privacy Policy
  • Disclaimer

Contact Info

Nutanhat, Mongolkote
Purba Burdwan, West Bengal
India, 713147

+91 93305 39277

hello@rskworld.in
support@rskworld.in

© 2026 RSK World. All rights reserved.

Content used for educational purposes only. View Disclaimer