397 lines
		
	
	
		
			No EOL
		
	
	
		
			12 KiB
		
	
	
	
		
			HTML
		
	
	
	
	
	
			
		
		
	
	
			397 lines
		
	
	
		
			No EOL
		
	
	
		
			12 KiB
		
	
	
	
		
			HTML
		
	
	
	
	
	
<!DOCTYPE html>
 | 
						|
<html lang="sv">
 | 
						|
<head>
 | 
						|
    <meta charset="UTF-8">
 | 
						|
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
 | 
						|
    <title>Arbetsgrupp - Deltagarformulär</title>
 | 
						|
    <script src="/socket.io/socket.io.js"></script>
 | 
						|
    <style>
 | 
						|
        * {
 | 
						|
            box-sizing: border-box;
 | 
						|
            margin: 0;
 | 
						|
            padding: 0;
 | 
						|
        }
 | 
						|
 | 
						|
        body {
 | 
						|
            font-family: 'Segoe UI', Tahoma, Geneva, Verdana, sans-serif;
 | 
						|
            background: linear-gradient(135deg, #667eea 0%, #764ba2 100%);
 | 
						|
            min-height: 100vh;
 | 
						|
            display: flex;
 | 
						|
            align-items: center;
 | 
						|
            justify-content: center;
 | 
						|
            padding: 20px;
 | 
						|
        }
 | 
						|
 | 
						|
        .container {
 | 
						|
            background: rgba(255, 255, 255, 0.95);
 | 
						|
            border-radius: 20px;
 | 
						|
            padding: 40px;
 | 
						|
            max-width: 600px;
 | 
						|
            width: 100%;
 | 
						|
            box-shadow: 0 20px 60px rgba(0, 0, 0, 0.2);
 | 
						|
            backdrop-filter: blur(10px);
 | 
						|
        }
 | 
						|
 | 
						|
        .logo {
 | 
						|
            display: block;
 | 
						|
            margin: 0 auto 20px;
 | 
						|
            height: auto;
 | 
						|
            max-width: 100%;
 | 
						|
            width: 7em;
 | 
						|
        }
 | 
						|
 | 
						|
        h1 {
 | 
						|
            color: #2c3e50;
 | 
						|
            text-align: center;
 | 
						|
            margin-bottom: 30px;
 | 
						|
            font-size: 2rem;
 | 
						|
            font-weight: 600;
 | 
						|
        }
 | 
						|
 | 
						|
        .question {
 | 
						|
            background: #f8f9fa;
 | 
						|
            padding: 20px;
 | 
						|
            border-radius: 12px;
 | 
						|
            margin-bottom: 30px;
 | 
						|
            border-left: 4px solid #667eea;
 | 
						|
        }
 | 
						|
 | 
						|
        .question h2 {
 | 
						|
            color: #2c3e50;
 | 
						|
            font-size: 1.3rem;
 | 
						|
            margin-bottom: 10px;
 | 
						|
        }
 | 
						|
 | 
						|
        .options {
 | 
						|
            display: grid;
 | 
						|
            gap: 15px;
 | 
						|
            margin-bottom: 25px;
 | 
						|
        }
 | 
						|
 | 
						|
        .option {
 | 
						|
            display: flex;
 | 
						|
            align-items: center;
 | 
						|
            padding: 15px;
 | 
						|
            background: white;
 | 
						|
            border: 2px solid #e9ecef;
 | 
						|
            border-radius: 10px;
 | 
						|
            cursor: pointer;
 | 
						|
            transition: all 0.3s ease;
 | 
						|
            position: relative;
 | 
						|
        }
 | 
						|
 | 
						|
        .option:hover {
 | 
						|
            border-color: #667eea;
 | 
						|
            transform: translateY(-2px);
 | 
						|
            box-shadow: 0 4px 15px rgba(102, 126, 234, 0.2);
 | 
						|
        }
 | 
						|
 | 
						|
        .option.selected {
 | 
						|
            border-color: #667eea;
 | 
						|
            background: linear-gradient(45deg, #667eea, #764ba2);
 | 
						|
            color: white;
 | 
						|
        }
 | 
						|
 | 
						|
        .option input[type="checkbox"] {
 | 
						|
            margin-right: 12px;
 | 
						|
            transform: scale(1.2);
 | 
						|
        }
 | 
						|
 | 
						|
        .option label {
 | 
						|
            flex: 1;
 | 
						|
            cursor: pointer;
 | 
						|
            font-weight: 500;
 | 
						|
        }
 | 
						|
 | 
						|
        .custom-input-container {
 | 
						|
            margin-top: 20px;
 | 
						|
        }
 | 
						|
 | 
						|
        .custom-input {
 | 
						|
            width: 100%;
 | 
						|
            padding: 15px;
 | 
						|
            border: 2px solid #e9ecef;
 | 
						|
            border-radius: 10px;
 | 
						|
            font-size: 16px;
 | 
						|
            transition: border-color 0.3s ease;
 | 
						|
        }
 | 
						|
 | 
						|
        .custom-input:focus {
 | 
						|
            outline: none;
 | 
						|
            border-color: #667eea;
 | 
						|
            box-shadow: 0 0 0 3px rgba(102, 126, 234, 0.1);
 | 
						|
        }
 | 
						|
 | 
						|
        .add-custom {
 | 
						|
            margin-top: 10px;
 | 
						|
            background: #28a745;
 | 
						|
            color: white;
 | 
						|
            border: none;
 | 
						|
            padding: 10px 20px;
 | 
						|
            border-radius: 8px;
 | 
						|
            cursor: pointer;
 | 
						|
            font-size: 14px;
 | 
						|
            transition: background-color 0.3s ease;
 | 
						|
        }
 | 
						|
 | 
						|
        .add-custom:hover {
 | 
						|
            background: #218838;
 | 
						|
        }
 | 
						|
 | 
						|
        .submit-btn {
 | 
						|
            width: 100%;
 | 
						|
            background: linear-gradient(45deg, #667eea, #764ba2);
 | 
						|
            color: white;
 | 
						|
            border: none;
 | 
						|
            padding: 18px;
 | 
						|
            border-radius: 12px;
 | 
						|
            font-size: 18px;
 | 
						|
            font-weight: 600;
 | 
						|
            cursor: pointer;
 | 
						|
            transition: all 0.3s ease;
 | 
						|
            margin-top: 20px;
 | 
						|
        }
 | 
						|
 | 
						|
        .submit-btn:hover {
 | 
						|
            transform: translateY(-2px);
 | 
						|
            box-shadow: 0 8px 25px rgba(102, 126, 234, 0.3);
 | 
						|
        }
 | 
						|
 | 
						|
        .submit-btn:disabled {
 | 
						|
            background: #6c757d;
 | 
						|
            cursor: not-allowed;
 | 
						|
            transform: none;
 | 
						|
            box-shadow: none;
 | 
						|
        }
 | 
						|
 | 
						|
        .success-message {
 | 
						|
            background: #d4edda;
 | 
						|
            color: #155724;
 | 
						|
            padding: 15px;
 | 
						|
            border-radius: 10px;
 | 
						|
            margin-bottom: 20px;
 | 
						|
            text-align: center;
 | 
						|
            font-weight: 500;
 | 
						|
        }
 | 
						|
 | 
						|
        .error-message {
 | 
						|
            background: #f8d7da;
 | 
						|
            color: #721c24;
 | 
						|
            padding: 15px;
 | 
						|
            border-radius: 10px;
 | 
						|
            margin-bottom: 20px;
 | 
						|
            text-align: center;
 | 
						|
            font-weight: 500;
 | 
						|
        }
 | 
						|
 | 
						|
        @media (max-width: 768px) {
 | 
						|
            .container {
 | 
						|
                padding: 25px;
 | 
						|
                margin: 10px;
 | 
						|
            }
 | 
						|
            
 | 
						|
            h1 {
 | 
						|
                font-size: 1.5rem;
 | 
						|
            }
 | 
						|
        }
 | 
						|
    </style>
 | 
						|
</head>
 | 
						|
<body>
 | 
						|
    <div class="container">
 | 
						|
        <img src="/aeroklubben.png" alt="Aeroklubben i Göteborg" class="logo" />
 | 
						|
        <h1>Vi behöver DIG!</h1>
 | 
						|
        
 | 
						|
        <div id="message"></div>
 | 
						|
        
 | 
						|
        <form id="responseForm">
 | 
						|
            <div class="question">
 | 
						|
                <h2>Vilken arbetsgrupp skulle du kunna tänka dig delta i?</h2>
 | 
						|
                <p style="color: #6c757d; margin-top: 5px;">Du kan välja flera alternativ</p>
 | 
						|
            </div>
 | 
						|
 | 
						|
            <div style="margin-bottom: 25px;">
 | 
						|
                <input 
 | 
						|
                    type="text" 
 | 
						|
                    id="participantName" 
 | 
						|
                    class="custom-input" 
 | 
						|
                    placeholder="Ditt namn *" 
 | 
						|
                    required
 | 
						|
                    style="margin-bottom: 20px;"
 | 
						|
                >
 | 
						|
            </div>
 | 
						|
            
 | 
						|
            <div class="options" id="options">
 | 
						|
                <!-- Options will be populated dynamically -->
 | 
						|
            </div>
 | 
						|
            
 | 
						|
            <div class="custom-input-container">
 | 
						|
                <input 
 | 
						|
                    type="text" 
 | 
						|
                    id="customInput" 
 | 
						|
                    class="custom-input" 
 | 
						|
                    placeholder="Eller skriv ditt eget alternativ..."
 | 
						|
                >
 | 
						|
                <button type="button" id="addCustom" class="add-custom">Lägg till</button>
 | 
						|
            </div>
 | 
						|
            
 | 
						|
            <button type="submit" id="submitBtn" class="submit-btn">Skicka svar</button>
 | 
						|
        </form>
 | 
						|
    </div>
 | 
						|
 | 
						|
    <script>
 | 
						|
        const socket = io();
 | 
						|
        const form = document.getElementById('responseForm');
 | 
						|
        const optionsContainer = document.getElementById('options');
 | 
						|
        const customInput = document.getElementById('customInput');
 | 
						|
        const addCustomBtn = document.getElementById('addCustom');
 | 
						|
        const submitBtn = document.getElementById('submitBtn');
 | 
						|
        const messageDiv = document.getElementById('message');
 | 
						|
        const participantNameInput = document.getElementById('participantName');
 | 
						|
 | 
						|
        let availableOptions = [];
 | 
						|
 | 
						|
        function showMessage(text, type = 'success') {
 | 
						|
            messageDiv.innerHTML = `<div class="${type}-message">${text}</div>`;
 | 
						|
            scrollTo(0, 0);
 | 
						|
 | 
						|
            setTimeout(() => {
 | 
						|
                messageDiv.innerHTML = '';
 | 
						|
            }, 5000);
 | 
						|
        }
 | 
						|
 | 
						|
        function renderOptions() {
 | 
						|
            // Store currently selected options before re-rendering
 | 
						|
            const currentlySelected = Array.from(document.querySelectorAll('input[type="checkbox"]:checked'))
 | 
						|
                .map(checkbox => checkbox.value);
 | 
						|
            
 | 
						|
            optionsContainer.innerHTML = '';
 | 
						|
            availableOptions.forEach((option, index) => {
 | 
						|
                const optionDiv = document.createElement('div');
 | 
						|
                optionDiv.className = 'option';
 | 
						|
                optionDiv.innerHTML = `
 | 
						|
                    <input type="checkbox" id="option${index}" value="${option}">
 | 
						|
                    <label for="option${index}">${option}</label>
 | 
						|
                `;
 | 
						|
                
 | 
						|
                const checkbox = optionDiv.querySelector('input');
 | 
						|
                const label = optionDiv.querySelector('label');
 | 
						|
                label.addEventListener('click', (e) => {
 | 
						|
                    e.stopPropagation(); // Prevent triggering the div click event
 | 
						|
                });
 | 
						|
                
 | 
						|
                // Restore selection state if this option was previously selected
 | 
						|
                if (currentlySelected.includes(option)) {
 | 
						|
                    checkbox.checked = true;
 | 
						|
                    optionDiv.classList.add('selected');
 | 
						|
                }
 | 
						|
                
 | 
						|
                checkbox.addEventListener('change', () => {
 | 
						|
                    if (checkbox.checked) {
 | 
						|
                        optionDiv.classList.add('selected');
 | 
						|
                    } else {
 | 
						|
                        optionDiv.classList.remove('selected');
 | 
						|
                    }
 | 
						|
                });
 | 
						|
                
 | 
						|
                optionDiv.addEventListener('click', (e) => {
 | 
						|
                    if (e.target !== checkbox) {
 | 
						|
                        checkbox.checked = !checkbox.checked;
 | 
						|
                        checkbox.dispatchEvent(new Event('change'));
 | 
						|
                    }
 | 
						|
                });
 | 
						|
                
 | 
						|
                optionsContainer.appendChild(optionDiv);
 | 
						|
            });
 | 
						|
        }
 | 
						|
 | 
						|
        function addCustomOption() {
 | 
						|
            const customValue = customInput.value.trim();
 | 
						|
            if (customValue && !availableOptions.includes(customValue)) {
 | 
						|
                availableOptions.push(customValue);
 | 
						|
                renderOptions();
 | 
						|
                customInput.value = '';
 | 
						|
                
 | 
						|
                // Auto-select the newly added option
 | 
						|
                const newCheckbox = document.querySelector(`input[value="${customValue}"]`);
 | 
						|
                if (newCheckbox) {
 | 
						|
                    newCheckbox.checked = true;
 | 
						|
                    newCheckbox.dispatchEvent(new Event('change'));
 | 
						|
                }
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        addCustomBtn.addEventListener('click', addCustomOption);
 | 
						|
 | 
						|
        customInput.addEventListener('keypress', (e) => {
 | 
						|
            if (e.key === 'Enter') {
 | 
						|
                e.preventDefault();
 | 
						|
                addCustomOption();
 | 
						|
            }
 | 
						|
        });
 | 
						|
 | 
						|
        form.addEventListener('submit', async (e) => {
 | 
						|
            e.preventDefault();
 | 
						|
            
 | 
						|
            const participantName = participantNameInput.value.trim();
 | 
						|
            const selectedOptions = Array.from(document.querySelectorAll('input[type="checkbox"]:checked'))
 | 
						|
                .map(checkbox => checkbox.value);
 | 
						|
            
 | 
						|
            if (!participantName) {
 | 
						|
                showMessage('Ange ditt namn', 'error');
 | 
						|
                return;
 | 
						|
            }
 | 
						|
            
 | 
						|
            if (selectedOptions.length === 0) {
 | 
						|
                showMessage('Välj minst ett alternativ', 'error');
 | 
						|
                return;
 | 
						|
            }
 | 
						|
 | 
						|
            submitBtn.disabled = true;
 | 
						|
            submitBtn.textContent = 'Skickar...';
 | 
						|
 | 
						|
            try {
 | 
						|
                const response = await fetch('/api/submit', {
 | 
						|
                    method: 'POST',
 | 
						|
                    headers: {
 | 
						|
                        'Content-Type': 'application/json',
 | 
						|
                    },
 | 
						|
                    body: JSON.stringify({ 
 | 
						|
                        responses: selectedOptions,
 | 
						|
                        participantName: participantName
 | 
						|
                    }),
 | 
						|
                });
 | 
						|
 | 
						|
                if (response.ok) {
 | 
						|
                    showMessage('Tack för ditt svar!');
 | 
						|
                    form.reset();
 | 
						|
                    document.querySelectorAll('.option').forEach(opt => opt.classList.remove('selected'));
 | 
						|
                } else {
 | 
						|
                    throw new Error('Server error');
 | 
						|
                }
 | 
						|
            } catch (error) {
 | 
						|
                showMessage('Ett fel uppstod. Försök igen.', 'error');
 | 
						|
            } finally {
 | 
						|
                submitBtn.disabled = false;
 | 
						|
                submitBtn.textContent = 'Skicka svar';
 | 
						|
            }
 | 
						|
        });
 | 
						|
 | 
						|
        // Socket.IO event listeners
 | 
						|
        socket.on('optionsUpdated', (options) => {
 | 
						|
            availableOptions = options;
 | 
						|
            renderOptions();
 | 
						|
        });
 | 
						|
 | 
						|
        // Initial load
 | 
						|
        fetch('/api/options')
 | 
						|
            .then(response => response.json())
 | 
						|
            .then(options => {
 | 
						|
                availableOptions = options;
 | 
						|
                renderOptions();
 | 
						|
            })
 | 
						|
            .catch(error => {
 | 
						|
                console.error('Error loading options:', error);
 | 
						|
            });
 | 
						|
    </script>
 | 
						|
</body>
 | 
						|
</html> |