""" BreakPilot Studio - Content Feed Modul Funktionen: - Educational Content durchsuchen & entdecken - Matrix Feed Integration - Search & Filter (Kategorie, Alter, Lizenz, Tags) - Content Preview & Download - Rating System (5 Sterne + Kommentare) - Favoriten & Collections """ class ContentFeedModule: """Modul fΓΌr Content Feed & Discovery.""" @staticmethod def get_css() -> str: """CSS fΓΌr das Content Feed Modul.""" return """ /* ============================================= CONTENT FEED MODULE ============================================= */ .panel-content-feed { display: flex; flex-direction: column; height: 100%; background: var(--bp-bg); overflow-y: auto; } /* Header with Search */ .content-feed-header { padding: 24px 40px; background: var(--bp-surface); border-bottom: 1px solid var(--bp-border); position: sticky; top: 0; z-index: 10; } .feed-header-top { display: flex; align-items: center; justify-content: space-between; margin-bottom: 20px; } .feed-header-top h2 { font-size: 24px; font-weight: 700; color: var(--bp-text); } /* Search Bar */ .feed-search-bar { position: relative; } .feed-search-input { width: 100%; padding: 14px 48px 14px 20px; background: var(--bp-bg); border: 1px solid var(--bp-border); border-radius: 999px; color: var(--bp-text); font-size: 15px; transition: all 0.2s; } .feed-search-input:focus { outline: none; border-color: var(--bp-primary); box-shadow: 0 0 0 3px var(--bp-primary-soft); } .feed-search-icon { position: absolute; right: 20px; top: 50%; transform: translateY(-50%); color: var(--bp-text-muted); font-size: 18px; } /* Filters */ .feed-filters { display: flex; gap: 12px; flex-wrap: wrap; margin-top: 16px; } .filter-group { display: flex; gap: 8px; align-items: center; } .filter-label { font-size: 13px; font-weight: 500; color: var(--bp-text-muted); } .filter-select { padding: 8px 16px; background: var(--bp-bg); border: 1px solid var(--bp-border); border-radius: 8px; color: var(--bp-text); font-size: 13px; cursor: pointer; transition: all 0.2s; } .filter-select:hover { border-color: var(--bp-primary); } .filter-chip { padding: 8px 16px; background: var(--bp-bg); border: 1px solid var(--bp-border); border-radius: 999px; font-size: 13px; color: var(--bp-text); cursor: pointer; transition: all 0.2s; } .filter-chip:hover { background: var(--bp-primary-soft); border-color: var(--bp-primary); } .filter-chip.active { background: var(--bp-primary); border-color: var(--bp-primary); color: white; } /* View Toggle */ .view-toggle { display: flex; gap: 4px; padding: 4px; background: var(--bp-bg); border-radius: 8px; } .view-btn { padding: 8px 12px; background: transparent; border: none; border-radius: 6px; color: var(--bp-text-muted); cursor: pointer; transition: all 0.2s; } .view-btn:hover { background: var(--bp-surface-elevated); } .view-btn.active { background: var(--bp-primary); color: white; } /* Content Area */ .content-feed-content { padding: 32px 40px; flex: 1; } /* Feed Grid View */ .feed-grid { display: grid; grid-template-columns: repeat(auto-fill, minmax(340px, 1fr)); gap: 24px; } /* Feed List View */ .feed-list { display: flex; flex-direction: column; gap: 16px; } /* Feed Card (Grid View) */ .feed-card { background: var(--bp-surface); border: 1px solid var(--bp-border); border-radius: 12px; overflow: hidden; transition: all 0.3s; cursor: pointer; } .feed-card:hover { transform: translateY(-4px); box-shadow: 0 12px 24px rgba(0, 0, 0, 0.15); border-color: var(--bp-primary); } .feed-card-thumbnail { width: 100%; height: 200px; background: linear-gradient(135deg, #667eea 0%, #764ba2 100%); display: flex; align-items: center; justify-content: center; font-size: 64px; color: white; position: relative; } .feed-card-badge { position: absolute; top: 12px; right: 12px; padding: 6px 12px; background: rgba(0, 0, 0, 0.7); backdrop-filter: blur(10px); border-radius: 6px; font-size: 11px; font-weight: 600; color: white; text-transform: uppercase; } .feed-card-body { padding: 20px; } .feed-card-title { font-size: 17px; font-weight: 600; color: var(--bp-text); margin-bottom: 8px; line-height: 1.4; } .feed-card-description { font-size: 14px; color: var(--bp-text-muted); margin-bottom: 16px; line-height: 1.5; display: -webkit-box; -webkit-line-clamp: 2; -webkit-box-orient: vertical; overflow: hidden; } .feed-card-meta { display: flex; flex-wrap: wrap; gap: 8px; margin-bottom: 16px; } .feed-meta-badge { padding: 4px 10px; background: var(--bp-bg); border-radius: 4px; font-size: 11px; color: var(--bp-text-muted); display: inline-flex; align-items: center; gap: 4px; } .feed-card-footer { display: flex; align-items: center; justify-content: space-between; padding-top: 16px; border-top: 1px solid var(--bp-border); } .feed-rating { display: flex; align-items: center; gap: 6px; font-size: 13px; color: var(--bp-text-muted); } .feed-stars { color: var(--bp-gold); } .feed-actions { display: flex; gap: 8px; } .feed-action-btn { padding: 8px 16px; background: transparent; border: 1px solid var(--bp-border); border-radius: 6px; color: var(--bp-text); font-size: 12px; cursor: pointer; transition: all 0.2s; } .feed-action-btn:hover { background: var(--bp-primary); border-color: var(--bp-primary); color: white; } .feed-action-btn-primary { background: var(--bp-primary); border-color: var(--bp-primary); color: white; } .feed-action-btn-primary:hover { background: var(--bp-primary-hover); } /* Feed List Card */ .feed-list-card { display: flex; gap: 20px; background: var(--bp-surface); border: 1px solid var(--bp-border); border-radius: 12px; padding: 20px; transition: all 0.3s; cursor: pointer; } .feed-list-card:hover { border-color: var(--bp-primary); box-shadow: 0 4px 12px rgba(0, 0, 0, 0.1); } .feed-list-thumbnail { width: 120px; height: 120px; flex-shrink: 0; background: linear-gradient(135deg, #667eea 0%, #764ba2 100%); border-radius: 8px; display: flex; align-items: center; justify-content: center; font-size: 48px; color: white; } .feed-list-body { flex: 1; } /* Content Modal */ .content-modal-overlay { position: fixed; top: 0; left: 0; right: 0; bottom: 0; background: rgba(0, 0, 0, 0.7); backdrop-filter: blur(5px); display: flex; align-items: center; justify-content: center; z-index: 1000; padding: 20px; } .content-modal { background: var(--bp-surface); border-radius: 16px; max-width: 900px; width: 100%; max-height: 90vh; overflow-y: auto; box-shadow: 0 20px 60px rgba(0, 0, 0, 0.3); } .content-modal-header { padding: 32px; border-bottom: 1px solid var(--bp-border); position: sticky; top: 0; background: var(--bp-surface); z-index: 1; } .modal-close { float: right; width: 32px; height: 32px; display: flex; align-items: center; justify-content: center; background: transparent; border: 1px solid var(--bp-border); border-radius: 6px; color: var(--bp-text-muted); cursor: pointer; font-size: 18px; transition: all 0.2s; } .modal-close:hover { background: var(--bp-danger); border-color: var(--bp-danger); color: white; } .content-modal-body { padding: 32px; } .modal-content-preview { width: 100%; max-height: 500px; background: var(--bp-bg); border-radius: 12px; margin-bottom: 24px; overflow: hidden; } .modal-content-preview iframe, .modal-content-preview video, .modal-content-preview img { width: 100%; height: 100%; border: none; } .modal-content-info { margin-top: 24px; } .modal-section-title { font-size: 14px; font-weight: 600; color: var(--bp-text-muted); text-transform: uppercase; letter-spacing: 0.5px; margin-bottom: 12px; } .license-info { display: flex; align-items: center; gap: 12px; padding: 16px; background: var(--bp-bg); border: 1px solid var(--bp-border); border-radius: 8px; margin-bottom: 24px; } .license-badge-large { font-size: 32px; } .license-text { flex: 1; } .license-name-large { font-size: 16px; font-weight: 600; color: var(--bp-text); margin-bottom: 4px; } .license-link { font-size: 13px; color: var(--bp-primary); text-decoration: none; } .license-link:hover { text-decoration: underline; } /* Rating Section */ .rating-section { margin-top: 24px; padding-top: 24px; border-top: 1px solid var(--bp-border); } .rating-stars-interactive { display: flex; gap: 8px; font-size: 32px; margin-bottom: 16px; } .star-btn { background: none; border: none; color: var(--bp-border); cursor: pointer; transition: all 0.2s; padding: 0; } .star-btn:hover, .star-btn.active { color: var(--bp-gold); transform: scale(1.1); } .rating-comment { width: 100%; padding: 12px; background: var(--bp-bg); border: 1px solid var(--bp-border); border-radius: 8px; color: var(--bp-text); font-family: inherit; font-size: 14px; resize: vertical; min-height: 80px; } .rating-submit { margin-top: 12px; padding: 12px 24px; background: var(--bp-primary); border: none; border-radius: 8px; color: white; font-weight: 600; cursor: pointer; transition: all 0.2s; } .rating-submit:hover { background: var(--bp-primary-hover); } /* Empty State */ .feed-empty { text-align: center; padding: 80px 20px; color: var(--bp-text-muted); } .feed-empty-icon { font-size: 80px; margin-bottom: 24px; } .feed-empty-title { font-size: 20px; font-weight: 600; color: var(--bp-text); margin-bottom: 12px; } .feed-empty-text { font-size: 15px; line-height: 1.6; } /* Loading State */ .feed-loading { text-align: center; padding: 60px 20px; color: var(--bp-text-muted); } .spinner { width: 48px; height: 48px; border: 4px solid var(--bp-border); border-top-color: var(--bp-primary); border-radius: 50%; animation: spin 1s linear infinite; margin: 0 auto 16px; } @keyframes spin { to { transform: rotate(360deg); } } """ @staticmethod def get_html() -> str: """HTML fΓΌr das Content Feed Modul.""" return """
""" @staticmethod def get_js() -> str: """JavaScript fΓΌr das Content Feed Modul.""" return """ // ============================================= // CONTENT FEED MODULE // ============================================= (function() { 'use strict'; const ContentFeed = { currentView: 'grid', allContent: [], filteredContent: [], currentFilters: { search: '', category: '', type: '', age: '', preset: 'all' }, init() { this.bindViewToggle(); this.bindSearch(); this.bindFilters(); this.loadContent(); }, bindViewToggle() { document.querySelectorAll('.view-btn').forEach(btn => { btn.addEventListener('click', () => { const view = btn.dataset.view; document.querySelectorAll('.view-btn').forEach(b => b.classList.remove('active')); btn.classList.add('active'); this.currentView = view; this.renderContent(); }); }); }, bindSearch() { const searchInput = document.getElementById('feed-search'); let searchTimeout; searchInput.addEventListener('input', (e) => { clearTimeout(searchTimeout); searchTimeout = setTimeout(() => { this.currentFilters.search = e.target.value.toLowerCase(); this.filterContent(); }, 300); }); }, bindFilters() { // Select filters ['category', 'type', 'age'].forEach(filter => { document.getElementById(`filter-${filter}`).addEventListener('change', (e) => { this.currentFilters[filter] = e.target.value; this.filterContent(); }); }); // Chip filters document.querySelectorAll('.filter-chip').forEach(chip => { chip.addEventListener('click', () => { document.querySelectorAll('.filter-chip').forEach(c => c.classList.remove('active')); chip.classList.add('active'); this.currentFilters.preset = chip.dataset.filter; this.filterContent(); }); }); }, async loadContent() { document.getElementById('feed-loading').style.display = 'block'; document.getElementById('feed-grid').style.display = 'none'; document.getElementById('feed-list').style.display = 'none'; document.getElementById('feed-empty').style.display = 'none'; try { const response = await fetch('http://localhost:8002/api/v1/content?limit=50'); if (!response.ok) throw new Error('Failed to load content'); const data = await response.json(); this.allContent = data.content || []; this.filteredContent = this.allContent; this.renderContent(); } catch (error) { console.error('Error loading content:', error); document.getElementById('feed-loading').innerHTML = `