mirror of
https://github.com/Alvin-Zilverstand/narrow_casting_system.git
synced 2026-03-06 21:29:47 +01:00
🎿 Complete SnowWorld Narrowcasting System - MBO Challenge 18
✅ Full-stack narrowcasting platform implementation ✅ Real-time WebSocket communication for instant updates ✅ Zone-specific content distribution (reception, restaurant, skislope, lockers, shop) ✅ Professional admin dashboard with content management interface ✅ Beautiful client display with winter/snow theme matching SnowWorld branding ✅ Comprehensive technical documentation and test suite ✅ Docker deployment support with CI/CD pipeline ✅ All system tests passing successfully 🏗️ Technical Implementation: - Backend: Node.js/Express with SQLite database - Frontend: Vanilla HTML/CSS/JavaScript (no frameworks) - Real-time: Socket.io WebSocket communication - Database: Complete schema with content, schedule, zones, logs tables - Security: File validation, input sanitization, CORS protection - Performance: Optimized for fast loading and real-time updates 🚀 Features Delivered: - Content upload (images, videos) with drag-and-drop interface - Content scheduling and planning system - Weather widget with real-time snow information - Responsive design for all screen sizes - Comprehensive error handling and fallback mechanisms - Professional winter theme with snow animations - Keyboard shortcuts and accessibility features 📁 Project Structure: - /backend: Complete Node.js server with API and WebSocket - /admin: Professional admin dashboard interface - /client: Beautiful client display application - /deployment: Docker and deployment configurations - /docs: Comprehensive technical documentation - /test_system.js: Complete test suite (all tests passing) 🧪 Testing Results: - Server health: ✅ Online and responsive - API endpoints: ✅ All endpoints functional - Database operations: ✅ All operations successful - WebSocket communication: ✅ Real-time updates working - Zone distribution: ✅ 6 zones correctly loaded - Weather integration: ✅ Weather data available Ready for production deployment at SnowWorld! 🎿❄️
This commit is contained in:
567
admin/js/ui.js
Normal file
567
admin/js/ui.js
Normal file
@@ -0,0 +1,567 @@
|
||||
// UI Management for SnowWorld Admin Dashboard
|
||||
class UIManager {
|
||||
constructor() {
|
||||
this.currentTab = 'content';
|
||||
this.contentCache = new Map();
|
||||
this.zonesCache = null;
|
||||
this.init();
|
||||
}
|
||||
|
||||
init() {
|
||||
this.setupEventListeners();
|
||||
this.loadZones();
|
||||
this.loadInitialData();
|
||||
}
|
||||
|
||||
setupEventListeners() {
|
||||
// Tab navigation
|
||||
document.querySelectorAll('.nav-tab').forEach(tab => {
|
||||
tab.addEventListener('click', (e) => {
|
||||
this.switchTab(e.target.dataset.tab);
|
||||
});
|
||||
});
|
||||
|
||||
// Content upload
|
||||
document.getElementById('addContentBtn')?.addEventListener('click', () => {
|
||||
this.openContentModal();
|
||||
});
|
||||
|
||||
document.getElementById('contentUploadForm')?.addEventListener('submit', (e) => {
|
||||
e.preventDefault();
|
||||
this.uploadContent();
|
||||
});
|
||||
|
||||
// Schedule management
|
||||
document.getElementById('addScheduleBtn')?.addEventListener('click', () => {
|
||||
this.openScheduleModal();
|
||||
});
|
||||
|
||||
document.getElementById('scheduleForm')?.addEventListener('submit', (e) => {
|
||||
e.preventDefault();
|
||||
this.createSchedule();
|
||||
});
|
||||
|
||||
// Filters
|
||||
document.getElementById('applyFilters')?.addEventListener('click', () => {
|
||||
this.applyContentFilters();
|
||||
});
|
||||
|
||||
// Modal controls
|
||||
document.querySelectorAll('.close-btn').forEach(btn => {
|
||||
btn.addEventListener('click', () => {
|
||||
this.closeModals();
|
||||
});
|
||||
});
|
||||
|
||||
// Refresh button
|
||||
document.getElementById('refreshBtn')?.addEventListener('click', () => {
|
||||
this.refreshData();
|
||||
});
|
||||
|
||||
// File input preview
|
||||
document.getElementById('contentFile')?.addEventListener('change', (e) => {
|
||||
this.previewFile(e.target.files[0]);
|
||||
});
|
||||
}
|
||||
|
||||
// Tab Management
|
||||
switchTab(tabName) {
|
||||
// Update active tab
|
||||
document.querySelectorAll('.nav-tab').forEach(tab => {
|
||||
tab.classList.remove('active');
|
||||
});
|
||||
document.querySelector(`[data-tab="${tabName}"]`).classList.add('active');
|
||||
|
||||
// Update tab content
|
||||
document.querySelectorAll('.tab-content').forEach(content => {
|
||||
content.classList.remove('active');
|
||||
});
|
||||
document.getElementById(`${tabName}-tab`).classList.add('active');
|
||||
|
||||
this.currentTab = tabName;
|
||||
this.loadTabData(tabName);
|
||||
}
|
||||
|
||||
async loadTabData(tabName) {
|
||||
try {
|
||||
switch (tabName) {
|
||||
case 'content':
|
||||
await this.loadContent();
|
||||
break;
|
||||
case 'schedule':
|
||||
await this.loadSchedule();
|
||||
break;
|
||||
case 'zones':
|
||||
await this.loadZonesOverview();
|
||||
break;
|
||||
case 'analytics':
|
||||
await this.loadAnalytics();
|
||||
break;
|
||||
}
|
||||
} catch (error) {
|
||||
console.error(`Error loading ${tabName} data:`, error);
|
||||
this.showToast(`Fout bij het laden van ${tabName} data`, 'error');
|
||||
}
|
||||
}
|
||||
|
||||
// Content Management
|
||||
async loadContent(zone = null, type = null) {
|
||||
const cacheKey = `${zone || 'all'}-${type || 'all'}`;
|
||||
|
||||
if (this.contentCache.has(cacheKey)) {
|
||||
this.renderContent(this.contentCache.get(cacheKey));
|
||||
return;
|
||||
}
|
||||
|
||||
const content = await api.getContent(zone, type);
|
||||
this.contentCache.set(cacheKey, content);
|
||||
this.renderContent(content);
|
||||
}
|
||||
|
||||
renderContent(content) {
|
||||
const grid = document.getElementById('contentGrid');
|
||||
if (!grid) return;
|
||||
|
||||
if (content.length === 0) {
|
||||
grid.innerHTML = `
|
||||
<div class="empty-state">
|
||||
<i class="fas fa-photo-video fa-3x"></i>
|
||||
<h3>Geen content gevonden</h3>
|
||||
<p>Begin met het toevoegen van content voor uw narrowcasting systeem.</p>
|
||||
</div>
|
||||
`;
|
||||
return;
|
||||
}
|
||||
|
||||
grid.innerHTML = content.map(item => this.createContentCard(item)).join('');
|
||||
|
||||
// Add event listeners to content cards
|
||||
grid.querySelectorAll('.delete-content').forEach(btn => {
|
||||
btn.addEventListener('click', (e) => {
|
||||
const contentId = e.target.dataset.contentId;
|
||||
this.deleteContent(contentId);
|
||||
});
|
||||
});
|
||||
}
|
||||
|
||||
createContentCard(item) {
|
||||
const typeIcon = {
|
||||
'image': 'fa-image',
|
||||
'video': 'fa-video',
|
||||
'livestream': 'fa-broadcast-tower'
|
||||
}[item.type] || 'fa-file';
|
||||
|
||||
const typeLabel = {
|
||||
'image': 'Afbeelding',
|
||||
'video': 'Video',
|
||||
'livestream': 'Livestream'
|
||||
}[item.type] || 'Bestand';
|
||||
|
||||
return `
|
||||
<div class="content-item" data-content-id="${item.id}">
|
||||
<div class="content-preview ${item.type}">
|
||||
${item.type === 'image' ?
|
||||
`<img src="${item.url}" alt="${item.title}" onerror="this.src='https://via.placeholder.com/300x200?text=Afbeelding'">` :
|
||||
`<i class="fas ${typeIcon} fa-3x"></i>`
|
||||
}
|
||||
</div>
|
||||
<div class="content-info">
|
||||
<h3 class="content-title">${item.title}</h3>
|
||||
<div class="content-meta">
|
||||
<span><i class="fas ${typeIcon}"></i> ${typeLabel}</span>
|
||||
<span><i class="fas fa-map-marker-alt"></i> Zone: ${item.zone}</span>
|
||||
<span><i class="fas fa-clock"></i> Duur: ${item.duration}s</span>
|
||||
<span><i class="fas fa-calendar"></i> ${new Date(item.createdAt).toLocaleDateString('nl-NL')}</span>
|
||||
</div>
|
||||
<div class="content-actions">
|
||||
<button class="btn btn-danger btn-small delete-content" data-content-id="${item.id}">
|
||||
<i class="fas fa-trash"></i> Verwijderen
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
`;
|
||||
}
|
||||
|
||||
// Modal Management
|
||||
openContentModal() {
|
||||
const modal = document.getElementById('contentModal');
|
||||
modal.classList.add('active');
|
||||
this.loadZonesSelect('contentZone');
|
||||
}
|
||||
|
||||
openScheduleModal() {
|
||||
const modal = document.getElementById('scheduleModal');
|
||||
modal.classList.add('active');
|
||||
this.loadContentSelect();
|
||||
this.loadZonesSelect('scheduleZone');
|
||||
this.setDefaultScheduleTimes();
|
||||
}
|
||||
|
||||
closeModals() {
|
||||
document.querySelectorAll('.modal').forEach(modal => {
|
||||
modal.classList.remove('active');
|
||||
});
|
||||
|
||||
// Reset forms
|
||||
document.getElementById('contentUploadForm')?.reset();
|
||||
document.getElementById('scheduleForm')?.reset();
|
||||
document.getElementById('fileInfo').innerHTML = '';
|
||||
}
|
||||
|
||||
// Content Upload
|
||||
previewFile(file) {
|
||||
if (!file) return;
|
||||
|
||||
const fileInfo = document.getElementById('fileInfo');
|
||||
const fileSize = (file.size / (1024 * 1024)).toFixed(2);
|
||||
|
||||
fileInfo.innerHTML = `
|
||||
<div class="file-details">
|
||||
<strong>Bestand:</strong> ${file.name}<br>
|
||||
<strong>Grootte:</strong> ${fileSize} MB<br>
|
||||
<strong>Type:</strong> ${file.type}
|
||||
</div>
|
||||
`;
|
||||
|
||||
// Auto-detect content type
|
||||
if (file.type.startsWith('image/')) {
|
||||
document.getElementById('contentType').value = 'image';
|
||||
} else if (file.type.startsWith('video/')) {
|
||||
document.getElementById('contentType').value = 'video';
|
||||
}
|
||||
}
|
||||
|
||||
async uploadContent() {
|
||||
const form = document.getElementById('contentUploadForm');
|
||||
const formData = new FormData();
|
||||
|
||||
const fileInput = document.getElementById('contentFile');
|
||||
const title = document.getElementById('contentTitle').value;
|
||||
const type = document.getElementById('contentType').value;
|
||||
const zone = document.getElementById('contentZone').value;
|
||||
const duration = document.getElementById('contentDuration').value;
|
||||
|
||||
if (!fileInput.files[0]) {
|
||||
this.showToast('Selecteer een bestand', 'error');
|
||||
return;
|
||||
}
|
||||
|
||||
formData.append('content', fileInput.files[0]);
|
||||
formData.append('title', title);
|
||||
formData.append('type', type);
|
||||
formData.append('zone', zone);
|
||||
formData.append('duration', duration);
|
||||
|
||||
try {
|
||||
this.showLoading('Bezig met uploaden...');
|
||||
const result = await api.uploadContent(formData);
|
||||
|
||||
this.closeModals();
|
||||
this.clearContentCache();
|
||||
await this.loadContent();
|
||||
|
||||
this.showToast('Content succesvol geüpload!', 'success');
|
||||
} catch (error) {
|
||||
console.error('Upload error:', error);
|
||||
this.showToast('Upload mislukt: ' + error.message, 'error');
|
||||
} finally {
|
||||
this.hideLoading();
|
||||
}
|
||||
}
|
||||
|
||||
async deleteContent(contentId) {
|
||||
if (!confirm('Weet u zeker dat u deze content wilt verwijderen?')) {
|
||||
return;
|
||||
}
|
||||
|
||||
try {
|
||||
this.showLoading('Bezig met verwijderen...');
|
||||
await api.deleteContent(contentId);
|
||||
|
||||
this.clearContentCache();
|
||||
await this.loadContent();
|
||||
|
||||
this.showToast('Content succesvol verwijderd', 'success');
|
||||
} catch (error) {
|
||||
console.error('Delete error:', error);
|
||||
this.showToast('Verwijderen mislukt: ' + error.message, 'error');
|
||||
} finally {
|
||||
this.hideLoading();
|
||||
}
|
||||
}
|
||||
|
||||
// Schedule Management
|
||||
async loadSchedule() {
|
||||
const zoneSelect = document.getElementById('scheduleZoneSelect');
|
||||
const selectedZone = zoneSelect?.value || 'reception';
|
||||
|
||||
try {
|
||||
const schedule = await api.getSchedule(selectedZone);
|
||||
this.renderSchedule(schedule);
|
||||
} catch (error) {
|
||||
console.error('Error loading schedule:', error);
|
||||
this.showToast('Fout bij het laden van planning', 'error');
|
||||
}
|
||||
}
|
||||
|
||||
renderSchedule(schedule) {
|
||||
const timeline = document.getElementById('scheduleTimeline');
|
||||
if (!timeline) return;
|
||||
|
||||
if (schedule.length === 0) {
|
||||
timeline.innerHTML = `
|
||||
<div class="empty-state">
|
||||
<i class="fas fa-calendar-times fa-3x"></i>
|
||||
<h3>Geen actieve planning</h3>
|
||||
<p>Er is momenteel geen geplande content voor deze zone.</p>
|
||||
</div>
|
||||
`;
|
||||
return;
|
||||
}
|
||||
|
||||
timeline.innerHTML = schedule.map(item => `
|
||||
<div class="schedule-item">
|
||||
<div class="schedule-time">
|
||||
${new Date(item.startTime).toLocaleTimeString('nl-NL', {hour: '2-digit', minute: '2-digit'})} -
|
||||
${new Date(item.endTime).toLocaleTimeString('nl-NL', {hour: '2-digit', minute: '2-digit'})}
|
||||
</div>
|
||||
<div class="schedule-content">
|
||||
<h4>${item.title}</h4>
|
||||
<p>Type: ${item.type} | Duur: ${item.duration}s</p>
|
||||
</div>
|
||||
</div>
|
||||
`).join('');
|
||||
}
|
||||
|
||||
async createSchedule() {
|
||||
const formData = {
|
||||
contentId: document.getElementById('scheduleContent').value,
|
||||
zone: document.getElementById('scheduleZone').value,
|
||||
startTime: document.getElementById('scheduleStart').value,
|
||||
endTime: document.getElementById('scheduleEnd').value,
|
||||
priority: parseInt(document.getElementById('schedulePriority').value)
|
||||
};
|
||||
|
||||
try {
|
||||
this.showLoading('Bezig met plannen...');
|
||||
await api.createSchedule(formData);
|
||||
|
||||
this.closeModals();
|
||||
await this.loadSchedule();
|
||||
|
||||
this.showToast('Planning succesvol aangemaakt!', 'success');
|
||||
} catch (error) {
|
||||
console.error('Schedule creation error:', error);
|
||||
this.showToast('Planning mislukt: ' + error.message, 'error');
|
||||
} finally {
|
||||
this.hideLoading();
|
||||
}
|
||||
}
|
||||
|
||||
setDefaultScheduleTimes() {
|
||||
const now = new Date();
|
||||
const startTime = new Date(now.getTime() + 60 * 60 * 1000); // 1 hour from now
|
||||
const endTime = new Date(startTime.getTime() + 60 * 60 * 1000); // 1 hour duration
|
||||
|
||||
document.getElementById('scheduleStart').value = startTime.toISOString().slice(0, 16);
|
||||
document.getElementById('scheduleEnd').value = endTime.toISOString().slice(0, 16);
|
||||
}
|
||||
|
||||
// Zones Management
|
||||
async loadZones() {
|
||||
if (this.zonesCache) return this.zonesCache;
|
||||
|
||||
try {
|
||||
this.zonesCache = await api.getZones();
|
||||
return this.zonesCache;
|
||||
} catch (error) {
|
||||
console.error('Error loading zones:', error);
|
||||
return [];
|
||||
}
|
||||
}
|
||||
|
||||
async loadZonesSelect(selectId) {
|
||||
const zones = await this.loadZones();
|
||||
const select = document.getElementById(selectId);
|
||||
if (!select) return;
|
||||
|
||||
select.innerHTML = zones.map(zone =>
|
||||
`<option value="${zone.id}">${zone.name}</option>`
|
||||
).join('');
|
||||
}
|
||||
|
||||
async loadContentSelect() {
|
||||
try {
|
||||
const content = await api.getContent();
|
||||
const select = document.getElementById('scheduleContent');
|
||||
if (!select) return;
|
||||
|
||||
select.innerHTML = content.map(item =>
|
||||
`<option value="${item.id}">${item.title} (${item.type})</option>`
|
||||
).join('');
|
||||
} catch (error) {
|
||||
console.error('Error loading content select:', error);
|
||||
}
|
||||
}
|
||||
|
||||
async loadZonesOverview() {
|
||||
const zones = await this.loadZones();
|
||||
const grid = document.getElementById('zonesGrid');
|
||||
if (!grid) return;
|
||||
|
||||
const zoneIcons = {
|
||||
'reception': 'fa-door-open',
|
||||
'restaurant': 'fa-utensils',
|
||||
'skislope': 'fa-skiing',
|
||||
'lockers': 'fa-locker',
|
||||
'shop': 'fa-shopping-bag',
|
||||
'all': 'fa-globe'
|
||||
};
|
||||
|
||||
grid.innerHTML = zones.map(zone => `
|
||||
<div class="zone-card">
|
||||
<div class="zone-icon">
|
||||
<i class="fas ${zoneIcons[zone.id] || 'fa-map-marker-alt'} fa-3x"></i>
|
||||
</div>
|
||||
<h3 class="zone-name">${zone.name}</h3>
|
||||
<p class="zone-description">${zone.description}</p>
|
||||
</div>
|
||||
`).join('');
|
||||
}
|
||||
|
||||
// Analytics
|
||||
async loadAnalytics() {
|
||||
try {
|
||||
const contentStats = await api.getContentStats();
|
||||
const scheduleStats = await api.getScheduleStats();
|
||||
const zones = await this.loadZones();
|
||||
|
||||
this.renderContentStats(contentStats);
|
||||
this.renderScheduleStats(scheduleStats);
|
||||
this.renderZoneStats(zones);
|
||||
} catch (error) {
|
||||
console.error('Error loading analytics:', error);
|
||||
this.showToast('Fout bij het laden van analytics', 'error');
|
||||
}
|
||||
}
|
||||
|
||||
renderContentStats(stats) {
|
||||
const container = document.getElementById('contentStats');
|
||||
if (!container) return;
|
||||
|
||||
container.innerHTML = `
|
||||
<div class="stat-item">
|
||||
<span class="stat-label">Totaal Content</span>
|
||||
<span class="stat-value">${stats.total}</span>
|
||||
</div>
|
||||
${Object.entries(stats.byType).map(([type, count]) => `
|
||||
<div class="stat-item">
|
||||
<span class="stat-label">${type.charAt(0).toUpperCase() + type.slice(1)}</span>
|
||||
<span class="stat-value">${count}</span>
|
||||
</div>
|
||||
`).join('')}
|
||||
`;
|
||||
}
|
||||
|
||||
renderScheduleStats(stats) {
|
||||
const container = document.getElementById('scheduleStats');
|
||||
if (!container) return;
|
||||
|
||||
container.innerHTML = `
|
||||
<div class="stat-item">
|
||||
<span class="stat-label">Totaal Planningen</span>
|
||||
<span class="stat-value">${stats.total}</span>
|
||||
</div>
|
||||
<div class="stat-item">
|
||||
<span class="stat-label">Actief</span>
|
||||
<span class="stat-value">${stats.active}</span>
|
||||
</div>
|
||||
<div class="stat-item">
|
||||
<span class="stat-label">Aankomend</span>
|
||||
<span class="stat-value">${stats.upcoming}</span>
|
||||
</div>
|
||||
`;
|
||||
}
|
||||
|
||||
renderZoneStats(zones) {
|
||||
const container = document.getElementById('zoneStats');
|
||||
if (!container) return;
|
||||
|
||||
container.innerHTML = zones.map(zone => `
|
||||
<div class="stat-item">
|
||||
<span class="stat-label">${zone.name}</span>
|
||||
<span class="stat-value">${zone.description}</span>
|
||||
</div>
|
||||
`).join('');
|
||||
}
|
||||
|
||||
// Utility Methods
|
||||
showToast(message, type = 'info') {
|
||||
const container = document.getElementById('toastContainer');
|
||||
const toast = document.createElement('div');
|
||||
toast.className = `toast ${type}`;
|
||||
toast.innerHTML = `
|
||||
<div class="toast-message">${message}</div>
|
||||
<button class="toast-close" onclick="this.parentElement.remove()">×</button>
|
||||
`;
|
||||
|
||||
container.appendChild(toast);
|
||||
|
||||
// Auto remove after 5 seconds
|
||||
setTimeout(() => {
|
||||
if (toast.parentElement) {
|
||||
toast.remove();
|
||||
}
|
||||
}, 5000);
|
||||
}
|
||||
|
||||
showLoading(message = 'Bezig...') {
|
||||
const loading = document.createElement('div');
|
||||
loading.id = 'globalLoading';
|
||||
loading.className = 'loading-overlay';
|
||||
loading.innerHTML = `
|
||||
<div class="loading-content">
|
||||
<div class="spinner"></div>
|
||||
<p>${message}</p>
|
||||
</div>
|
||||
`;
|
||||
document.body.appendChild(loading);
|
||||
}
|
||||
|
||||
hideLoading() {
|
||||
const loading = document.getElementById('globalLoading');
|
||||
if (loading) {
|
||||
loading.remove();
|
||||
}
|
||||
}
|
||||
|
||||
clearContentCache() {
|
||||
this.contentCache.clear();
|
||||
}
|
||||
|
||||
async refreshData() {
|
||||
this.clearContentCache();
|
||||
await this.loadTabData(this.currentTab);
|
||||
this.showToast('Data ververst!', 'success');
|
||||
}
|
||||
|
||||
async loadInitialData() {
|
||||
try {
|
||||
await this.loadZones();
|
||||
await this.loadContent();
|
||||
} catch (error) {
|
||||
console.error('Error loading initial data:', error);
|
||||
this.showToast('Fout bij het laden van initiële data', 'error');
|
||||
}
|
||||
}
|
||||
|
||||
applyContentFilters() {
|
||||
const zone = document.getElementById('zoneFilter').value;
|
||||
const type = document.getElementById('typeFilter').value;
|
||||
this.loadContent(zone || null, type || null);
|
||||
}
|
||||
}
|
||||
|
||||
// Create global UI instance
|
||||
window.ui = new UIManager();
|
||||
Reference in New Issue
Block a user