mirror of https://github.com/Nezreka/SoulSync.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
275 lines
10 KiB
275 lines
10 KiB
#!/usr/bin/env python3
|
|
|
|
from PyQt6.QtWidgets import (QDialog, QVBoxLayout, QHBoxLayout, QLabel,
|
|
QPushButton, QFrame, QScrollArea, QWidget)
|
|
from PyQt6.QtCore import Qt
|
|
from PyQt6.QtGui import QFont
|
|
from utils.logging_config import get_logger
|
|
|
|
logger = get_logger("version_info_modal")
|
|
|
|
class VersionInfoModal(QDialog):
|
|
"""Modal displaying recent changes and version information"""
|
|
|
|
def __init__(self, parent=None):
|
|
super().__init__(parent)
|
|
self.setWindowTitle("What's New in SoulSync v0.6")
|
|
self.setModal(True)
|
|
self.setFixedSize(600, 500)
|
|
self.setup_ui()
|
|
|
|
def setup_ui(self):
|
|
self.setStyleSheet("""
|
|
VersionInfoModal {
|
|
background: #1a1a1a;
|
|
border-radius: 12px;
|
|
}
|
|
""")
|
|
|
|
layout = QVBoxLayout(self)
|
|
layout.setContentsMargins(0, 0, 0, 0)
|
|
layout.setSpacing(0)
|
|
|
|
# Header
|
|
header = self.create_header()
|
|
layout.addWidget(header)
|
|
|
|
# Content area with scroll
|
|
content_area = self.create_content_area()
|
|
layout.addWidget(content_area)
|
|
|
|
# Footer with close button
|
|
footer = self.create_footer()
|
|
layout.addWidget(footer)
|
|
|
|
def create_header(self):
|
|
header = QFrame()
|
|
header.setFixedHeight(80)
|
|
header.setStyleSheet("""
|
|
QFrame {
|
|
background: #1a1a1a;
|
|
border-top-left-radius: 12px;
|
|
border-top-right-radius: 12px;
|
|
border-bottom: 1px solid rgba(255, 255, 255, 0.1);
|
|
}
|
|
""")
|
|
|
|
layout = QVBoxLayout(header)
|
|
layout.setContentsMargins(30, 20, 30, 15)
|
|
layout.setSpacing(5)
|
|
|
|
# Title
|
|
title = QLabel("What's New in SoulSync")
|
|
title.setFont(QFont("SF Pro Display", 18, QFont.Weight.Bold))
|
|
title.setStyleSheet("""
|
|
color: #ffffff;
|
|
letter-spacing: -0.5px;
|
|
font-weight: 700;
|
|
""")
|
|
|
|
# Version subtitle
|
|
version_subtitle = QLabel("Version 0.6 - Complete Multi-Server Support")
|
|
version_subtitle.setFont(QFont("SF Pro Text", 11, QFont.Weight.Medium))
|
|
version_subtitle.setStyleSheet("""
|
|
color: rgba(255, 255, 255, 0.7);
|
|
letter-spacing: 0.1px;
|
|
margin-top: 2px;
|
|
""")
|
|
|
|
layout.addWidget(title)
|
|
layout.addWidget(version_subtitle)
|
|
|
|
return header
|
|
|
|
def create_content_area(self):
|
|
# Scroll area for content
|
|
scroll_area = QScrollArea()
|
|
scroll_area.setWidgetResizable(True)
|
|
scroll_area.setHorizontalScrollBarPolicy(Qt.ScrollBarPolicy.ScrollBarAlwaysOff)
|
|
scroll_area.setVerticalScrollBarPolicy(Qt.ScrollBarPolicy.ScrollBarAsNeeded)
|
|
scroll_area.setStyleSheet("""
|
|
QScrollArea {
|
|
border: none;
|
|
background: #1a1a1a;
|
|
}
|
|
QScrollBar:vertical {
|
|
background: #2a2a2a;
|
|
width: 8px;
|
|
border-radius: 4px;
|
|
}
|
|
QScrollBar::handle:vertical {
|
|
background: #555555;
|
|
border-radius: 4px;
|
|
}
|
|
QScrollBar::handle:vertical:hover {
|
|
background: #666666;
|
|
}
|
|
""")
|
|
|
|
# Content widget
|
|
content_widget = QWidget()
|
|
content_layout = QVBoxLayout(content_widget)
|
|
content_layout.setContentsMargins(30, 25, 30, 25)
|
|
content_layout.setSpacing(25)
|
|
|
|
# Multi-Server Support
|
|
multiserver_section = self.create_feature_section(
|
|
"🔀 Complete Multi-Server Support",
|
|
"Full Plex and Jellyfin compatibility across all features and workflows",
|
|
[
|
|
"• Added native Jellyfin client with full API integration for artists, albums, and tracks",
|
|
"• Automatic server detection and dynamic client switching throughout the application",
|
|
"• All sync tools now work seamlessly with both Plex and Jellyfin servers",
|
|
"• Database filtering by server source ensures clean separation of library data",
|
|
"• Updated all download and analysis workers for multi-server compatibility",
|
|
"• Server-specific UI adaptations (e.g., metadata updater hidden for Jellyfin)",
|
|
"• Unified media client architecture for consistent cross-server functionality"
|
|
],
|
|
"Switch between Plex and Jellyfin in Settings → Server Configuration - all features automatically adapt!"
|
|
)
|
|
content_layout.addWidget(multiserver_section)
|
|
|
|
# Enhanced Compatibility
|
|
compatibility_section = self.create_feature_section(
|
|
"⚙️ Enhanced Cross-Server Features",
|
|
"All major workflows updated for seamless multi-server operation",
|
|
[
|
|
"• Playlist sync (Spotify & YouTube) now supports both Plex and Jellyfin",
|
|
"• Download Missing Tracks modals work with active server automatically",
|
|
"• Artist browsing and album analysis adapt to your selected media server",
|
|
"• Database updates and library scans respect server-specific configurations",
|
|
"• Playlist track analysis workers dynamically use the correct server API",
|
|
"• All toast notifications and UI feedback are server-aware"
|
|
]
|
|
)
|
|
content_layout.addWidget(compatibility_section)
|
|
|
|
# Technical Improvements
|
|
technical_section = self.create_feature_section(
|
|
"🔧 Technical Architecture Updates",
|
|
"Robust foundation for reliable multi-server operation",
|
|
[
|
|
"• Implemented server-agnostic database schema with source tracking",
|
|
"• Added unified configuration management for multiple media servers",
|
|
"• Created wrapper classes for consistent API interfaces across servers",
|
|
"• Enhanced error handling with server-specific messaging",
|
|
"• Optimized performance with intelligent client caching and connection pooling",
|
|
"• Future-ready architecture for additional media server integrations"
|
|
]
|
|
)
|
|
content_layout.addWidget(technical_section)
|
|
|
|
scroll_area.setWidget(content_widget)
|
|
return scroll_area
|
|
|
|
def create_feature_section(self, title, description, features, usage_note=None):
|
|
section = QFrame()
|
|
section.setStyleSheet("""
|
|
QFrame {
|
|
background: transparent;
|
|
border: none;
|
|
border-left: 3px solid rgba(29, 185, 84, 0.4);
|
|
border-radius: 0px;
|
|
padding: 0px;
|
|
margin-left: 5px;
|
|
}
|
|
""")
|
|
|
|
layout = QVBoxLayout(section)
|
|
layout.setContentsMargins(20, 18, 20, 18)
|
|
layout.setSpacing(12)
|
|
|
|
# Section title
|
|
title_label = QLabel(title)
|
|
title_label.setFont(QFont("SF Pro Text", 14, QFont.Weight.Bold))
|
|
title_label.setStyleSheet("""
|
|
color: #1ed760;
|
|
font-weight: 600;
|
|
letter-spacing: -0.2px;
|
|
margin-bottom: 3px;
|
|
""")
|
|
layout.addWidget(title_label)
|
|
|
|
# Description
|
|
desc_label = QLabel(description)
|
|
desc_label.setFont(QFont("SF Pro Text", 11))
|
|
desc_label.setStyleSheet("""
|
|
color: rgba(255, 255, 255, 0.8);
|
|
line-height: 1.4;
|
|
margin-bottom: 8px;
|
|
""")
|
|
desc_label.setWordWrap(True)
|
|
layout.addWidget(desc_label)
|
|
|
|
# Features list
|
|
for feature in features:
|
|
feature_label = QLabel(feature)
|
|
feature_label.setFont(QFont("SF Pro Text", 10))
|
|
feature_label.setStyleSheet("""
|
|
color: rgba(255, 255, 255, 0.7);
|
|
line-height: 1.5;
|
|
padding-left: 8px;
|
|
margin: 2px 0px;
|
|
""")
|
|
feature_label.setWordWrap(True)
|
|
layout.addWidget(feature_label)
|
|
|
|
# Usage note if provided
|
|
if usage_note:
|
|
usage_label = QLabel(f"💡 {usage_note}")
|
|
usage_label.setFont(QFont("SF Pro Text", 10))
|
|
usage_label.setStyleSheet("""
|
|
color: #1ed760;
|
|
background: transparent;
|
|
border: none;
|
|
padding: 8px 0px;
|
|
margin-top: 8px;
|
|
line-height: 1.4;
|
|
font-style: italic;
|
|
""")
|
|
usage_label.setWordWrap(True)
|
|
layout.addWidget(usage_label)
|
|
|
|
return section
|
|
|
|
def create_footer(self):
|
|
footer = QFrame()
|
|
footer.setFixedHeight(65)
|
|
footer.setStyleSheet("""
|
|
QFrame {
|
|
background: rgba(255, 255, 255, 0.02);
|
|
border-top: 1px solid rgba(255, 255, 255, 0.08);
|
|
border-bottom-left-radius: 12px;
|
|
border-bottom-right-radius: 12px;
|
|
}
|
|
""")
|
|
|
|
layout = QHBoxLayout(footer)
|
|
layout.setContentsMargins(30, 15, 30, 15)
|
|
|
|
# Close button
|
|
close_button = QPushButton("Close")
|
|
close_button.setFixedSize(100, 35)
|
|
close_button.setFont(QFont("SF Pro Text", 10, QFont.Weight.Medium))
|
|
close_button.setStyleSheet("""
|
|
QPushButton {
|
|
background: #1db954;
|
|
color: white;
|
|
border: none;
|
|
border-radius: 6px;
|
|
font-weight: 500;
|
|
letter-spacing: 0.1px;
|
|
}
|
|
QPushButton:hover {
|
|
background: #1ed760;
|
|
}
|
|
QPushButton:pressed {
|
|
background: #169c46;
|
|
}
|
|
""")
|
|
close_button.clicked.connect(self.accept)
|
|
|
|
layout.addStretch()
|
|
layout.addWidget(close_button)
|
|
|
|
return footer |