diff --git a/.DS_Store b/.DS_Store
index 88788dd..0ec6926 100644
Binary files a/.DS_Store and b/.DS_Store differ
diff --git a/.github/workflows/build.yml b/.github/workflows/build.yml
index 33be1b5..f1342d9 100644
--- a/.github/workflows/build.yml
+++ b/.github/workflows/build.yml
@@ -30,16 +30,16 @@ jobs:
if: github.event_name == 'push'
uses: softprops/action-gh-release@v2
with:
- tag_name: 2.0.0
- name: 2.0.0 Stable
+ tag_name: 2.1.0A6
+ name: 2.1.0 Alpha 4
body: ${{ github.event.head_commit.message }}
- prerelease: false
+ prerelease: true
files: |
./dist/Converter_arm64_darwin.zip
build_intel:
name: Build Intel
runs-on: macos-15-intel
- if: github.repository_owner == 'pyquick'
+ if: github.repository_owner == 'intsant'
permissions:
contents: write
steps:
@@ -58,10 +58,9 @@ jobs:
if: github.event_name == 'push'
uses: softprops/action-gh-release@v2
with:
- tag_name: 2.0.0
- name: 2.0.0 Stable
+ tag_name: 2.1.0A6
+ name: 2.1.0 Alpha 4
body: ${{ github.event.head_commit.message }}
- prerelease: false
+ prerelease: true
files: |
- ./dist/Converter_intel_darwin.zip
-
\ No newline at end of file
+ ./dist/Converter_intel_darwin.zip
\ No newline at end of file
diff --git a/.gitignore b/.gitignore
index 52c80aa..572e1d7 100644
--- a/.gitignore
+++ b/.gitignore
@@ -201,7 +201,7 @@ cython_debug/
# refer to https://docs.cursor.com/context/ignore-files
.cursorignore
.cursorindexingignore
-
+.trae/
# Marimo
marimo/_static/
marimo/_lsp/
diff --git a/CHANGELOG/v2.0.0_new_features.md b/CHANGELOG/v2.0.0_new_features.md
deleted file mode 100644
index 4ff6c81..0000000
--- a/CHANGELOG/v2.0.0_new_features.md
+++ /dev/null
@@ -1,77 +0,0 @@
-# New Features and Code Improvements
-
-## New Features
-
-### 1. Password Protection Support
-- **Password Detection Module**: Implemented password detection functionality for ZIP, RAR, and 7z formats
-- **Secure Password Dialog**: Added secure password input interface with comprehensive error handling
-- **Protected File Handling**: Enhanced capability for processing password-protected archive files
-
-### 2. Update Management System
-- **GitHub PAT Integration**: Implemented secure storage and validation of GitHub Personal Access Tokens (PAT)
-- **Automatic Update Mechanism**: Added automatic update detection, download, and installation workflow
-- **Backup and Recovery**: Implemented seamless backup and recovery functionality during update processes
-
-### 3. Theme System
-- **Automatic Theme Switching**: Implemented system theme detection and synchronization
-- **Dark/Light Mode Support**: Added complete dark and light theme support
-- **Theme Persistence**: User theme selections persist across application restarts
-
-### 4. Settings Management System
-- **Real-time Auto-save**: Implemented real-time automatic saving for settings dialogs
-- **Status Label Display**: Added visual feedback for save status
-- **Settings Validation**: Implemented validation for configuration items
-
-### 5. Archive Management Tools
-- **ZIP Processing Tools**: Added complete ZIP file creation, extraction, and modification functionality
-- **Progress Tracking**: Added detailed progress display for archive operations
-- **Content Listing**: Implemented archive file content viewing functionality
-
-## Code Improvements
-
-### 1. Architecture Refactoring
-- **Modular Design**: Refactored code into a more modular architecture for improved maintainability
-- **Component Separation**: Achieved clear separation between UI components and business logic
-- **Dependency Injection**: Improved dependency relationship management between components
-
-### 2. UI Framework Migration
-- **wxPython to PySide6 Migration**: Completed full migration from wxPython to PySide6 (Qt6)
-- **Modernized UI**: Implemented a more modern, fluid user interface design
-- **Responsive Layout**: Improved interface layout responsiveness and adaptability
-
-### 3. Error Handling and Logging
-- **Comprehensive Error Handling**: Implemented complete exception catching and handling mechanisms
-- **Structured Logging System**: Added structured logging system
-- **User-Friendly Error Messages**: Improved user-friendliness of error messages
-
-### 4. Performance Optimization
-- **Asynchronous Operations**: Converted time-consuming operations to asynchronous execution for better UI responsiveness
-- **Memory Management**: Optimized memory usage and reduced memory leak risks
-- **Resource Loading**: Improved resource loading and management methods
-
-### 5. Security Enhancements
-- **Secure Storage**: Implemented secure credential storage using macOS Keychain
-- **Input Validation**: Added comprehensive input validation and sanitization mechanisms
-- **Permission Management**: Improved application permission management and requests
-
-### 6. Build System Improvements
-- **Dual Architecture Support**: Implemented dual architecture builds for ARM64 and Intel Macs
-- **Dependency Management**: Improved dependency management and version control during build processes
-- **Automated CI/CD**: Enhanced GitHub Actions workflow automation
-
-### 7. Code Quality Enhancement
-- **Type Hints**: Added comprehensive Python type hints
-- **Docstrings**: Improved function and class documentation strings
-- **Code Standards**: Unified code style and naming conventions
-
-## Summary
-
-These new features and code improvements collectively constitute a comprehensive upgrade from version 1.0.0 pre2 to the current 2.0.0 version, not only enhancing application functionality but also significantly improving code quality and user experience.
-
-The most significant changes include:
-- Complete UI framework migration from wxPython to PySide6 (Qt6)
-- Implementation of password protection and secure credential management
-- Addition of automatic update management system
-- Enhanced theme system with automatic switching
-- Improved build system supporting dual architecture (ARM64 and Intel)
-- Comprehensive code refactoring for better maintainability and performance
\ No newline at end of file
diff --git a/Converter.py b/Converter.py
index 7e5a356..bcf801e 100644
--- a/Converter.py
+++ b/Converter.py
@@ -3,8 +3,8 @@
from concurrent.futures import thread
from importlib import reload
import sys
-
import os
+import threading
from PySide6.QtWidgets import (
QApplication,
QWidget,
@@ -16,14 +16,26 @@
QGridLayout,
QSizePolicy,
QGroupBox,
- QDialog
+ QDialog,
+ QListWidget,
+ QListWidgetItem,
+ QMessageBox,
+ QFrame,
+ QStackedWidget
)
-from PySide6.QtGui import QIcon, QPainter, QPixmap, QPalette
-from PySide6.QtCore import QSize, Qt, QSettings, QPropertyAnimation, QEasingCurve, QTimer
+from PySide6.QtGui import QIcon, QPainter, QPixmap, QPalette, QColor
+from PySide6.QtCore import QSize, Qt, QSettings, QPropertyAnimation, QEasingCurve, QTimer, Signal
import multiprocessing
-from qfluentwidgets import Theme, setTheme,qconfig,SystemThemeListener
- # Keep for freeze_support, but remove direct Process usage
-from settings.update_settings_gui import UpdateDialog
+from qfluentwidgets import (
+ HeaderCardWidget, ImageLabel, Theme, setTheme, qconfig, SystemThemeListener,
+ FluentWindow, NavigationItemPosition,
+ CardWidget, PushButton, PrimaryPushButton, IconWidget,
+ BodyLabel, CaptionLabel, SubtitleLabel, TitleLabel, LargeTitleLabel,
+ FluentIcon as FIF, setFont, TransparentToolButton, SegmentedWidget,
+ setCustomStyleSheet, ElevatedCardWidget, ProgressBar, FlowLayout,
+ ScrollArea
+)
+from qfluentwidgets.components.widgets.card_widget import SimpleCardWidget
from settings.settings_gui import SettingsDialog
from con import CON # Import CON instance for theme settings
# Encoding settings have been moved to debug_logger for handling
@@ -55,7 +67,518 @@ def create_placeholder_icon(path: str, color: str, text: str):
return True
return False
-class IconButtonsWindow(QWidget):
+
+class AppCard(CardWidget):
+ """Application card widget"""
+
+ def __init__(self, icon_path, title, content, app_type, parent=None):
+ super().__init__(parent)
+ self.title = title
+ self.content = content
+ self.app_type = app_type
+ self.icon_path = icon_path
+ self.icon_widget = ImageLabel(icon_path, self)
+ self.title_label = BodyLabel(self.title, self)
+ self.content_label = CaptionLabel(self.content, self)
+ self.icon_widget.scaledToHeight(68)
+ self.icon_widget.setFixedSize(48, 48)
+ self.content_label.setTextColor(QColor("#606060"), QColor("#d2d2d2"))
+ self.setFixedHeight(73)
+ self.h_box_layout = QHBoxLayout(self)
+ self.v_box_layout = QVBoxLayout()
+
+
+ # Configure layouts
+ self.h_box_layout.setContentsMargins(20, 11, 11, 11)
+ self.h_box_layout.setSpacing(15)
+ self.v_box_layout.setContentsMargins(0, 0, 0, 0)
+ self.v_box_layout.setSpacing(0)
+
+
+ self.open_button = PrimaryPushButton('Open', self)
+ self.open_button.setFixedWidth(120)
+ self.more_button = TransparentToolButton(FIF.MORE, self)
+ self.more_button.setFixedSize(32, 32)
+
+ # Add components to layouts
+ self.h_box_layout.addWidget(self.icon_widget)
+
+ self.v_box_layout.addWidget(self.title_label, 0, Qt.AlignmentFlag.AlignVCenter)
+ self.v_box_layout.addWidget(self.content_label, 0, Qt.AlignmentFlag.AlignVCenter)
+ self.h_box_layout.addLayout(self.v_box_layout)
+
+ self.h_box_layout.addStretch(1)
+ self.h_box_layout.addWidget(self.open_button, 0, Qt.AlignmentFlag.AlignRight)
+ self.h_box_layout.addWidget(self.more_button, 0, Qt.AlignmentFlag.AlignRight)
+
+ self.open_button.clicked.connect(self.on_open_clicked)
+
+
+ def on_open_clicked(self):
+ """Handle open button clicked event"""
+ if self.app_type == 'image':
+ run_image_app()
+ elif self.app_type == 'arc':
+ run_zip_app()
+
+
+class AppCardTask(CardWidget):
+ """Task card widget for task manager interface"""
+
+ task_cancelled = Signal(str)
+ task_retried = Signal(str)
+ task_removed = Signal(str)
+
+ def __init__(self, task_id, task_type, task_info, parent=None):
+ super().__init__(parent)
+ self.task_id = task_id
+ self.task_type = task_type
+ self.task_info = task_info
+ self.setup_ui()
+ self.setup_connections()
+
+ def setup_ui(self):
+ """Initialize UI components"""
+ self.setFixedHeight(120)
+ self.setFixedWidth(280)
+ self.setBorderRadius(12)
+
+ main_layout = QHBoxLayout(self)
+ main_layout.setContentsMargins(15, 12, 15, 12)
+ main_layout.setSpacing(12)
+
+ icon_layout = QVBoxLayout()
+ icon_layout.setContentsMargins(0, 0, 0, 0)
+ icon_layout.setSpacing(4)
+
+ if self.task_type == "image":
+ task_icon = FIF.PHOTO
+ else:
+ task_icon = FIF.FOLDER
+ self.icon_widget = IconWidget(task_icon, self)
+ self.icon_widget.setFixedSize(40, 40)
+ icon_layout.addWidget(self.icon_widget, 0, Qt.AlignmentFlag.AlignCenter)
+
+ status_badges = {
+ "pending": ("Pending", "#9e9e9e"),
+ "running": ("Running", "#2196f3"),
+ "completed": ("Completed", "#4caf50"),
+ "failed": ("Failed", "#f44336"),
+ "cancelled": ("Cancelled", "#ff9800")
+ }
+ status = self.task_info.get("status", "pending")
+ status_text, status_color = status_badges.get(status, status_badges["pending"])
+
+ self.status_badge = BodyLabel(status_text)
+ self.status_badge.setStyleSheet(f"""
+ BodyLabel {{
+ background-color: {status_color}20;
+ color: {status_color};
+ padding: 2px 8px;
+ border-radius: 10px;
+ font-size: 10px;
+ font-weight: bold;
+ }}
+ """)
+ icon_layout.addWidget(self.status_badge, 0, Qt.AlignmentFlag.AlignCenter)
+
+ main_layout.addLayout(icon_layout)
+
+ content_layout = QVBoxLayout()
+ content_layout.setContentsMargins(0, 0, 0, 0)
+ content_layout.setSpacing(6)
+
+ input_path = self.task_info.get("input_path", "Unknown")
+ filename = os.path.basename(input_path) if input_path else "Unknown"
+ self.title_label = BodyLabel(filename)
+ self.title_label.setTextColor(QColor("#333333"), QColor("#e0e0e0"))
+ self.title_label.setFixedWidth(160)
+ content_layout.addWidget(self.title_label)
+
+ self.status_label = CaptionLabel(self.get_status_text())
+ self.status_label.setTextColor(QColor("#666666"), QColor("#a0a0a0"))
+ self.status_label.setFixedWidth(160)
+ content_layout.addWidget(self.status_label)
+
+ self.progress_bar = ProgressBar()
+ self.progress_bar.setFixedHeight(4)
+ self.progress_bar.setFixedWidth(160)
+ self.progress_bar.setValue(self.task_info.get("progress", 0))
+ content_layout.addWidget(self.progress_bar)
+
+ main_layout.addLayout(content_layout)
+
+ button_layout = QVBoxLayout()
+ button_layout.setContentsMargins(0, 0, 0, 0)
+ button_layout.setSpacing(8)
+ button_layout.addStretch(1)
+
+ self.cancel_button = TransparentToolButton(FIF.CANCEL, self)
+ self.cancel_button.setFixedSize(28, 28)
+ self.cancel_button.setToolTip("Cancel Task")
+ button_layout.addWidget(self.cancel_button, 0, Qt.AlignmentFlag.AlignRight)
+
+ self.retry_button = TransparentToolButton(FIF.RETURN, self)
+ self.retry_button.setFixedSize(28, 28)
+ self.retry_button.setToolTip("Retry Task")
+ self.retry_button.setVisible(False)
+ button_layout.addWidget(self.retry_button, 0, Qt.AlignmentFlag.AlignRight)
+
+ self.remove_button = TransparentToolButton(FIF.DELETE, self)
+ self.remove_button.setFixedSize(28, 28)
+ self.remove_button.setToolTip("Remove Task")
+ button_layout.addWidget(self.remove_button, 0, Qt.AlignmentFlag.AlignRight)
+
+ button_layout.addStretch(1)
+
+ main_layout.addLayout(button_layout)
+
+ def get_status_text(self):
+ """Get status display text"""
+ status = self.task_info.get("status", "pending")
+ if status == "pending":
+ return "Waiting in queue..."
+ elif status == "running":
+ progress = self.task_info.get("progress", 0)
+ return f"Processing... {progress}%"
+ elif status == "completed":
+ return "Task completed successfully"
+ elif status == "failed":
+ error = self.task_info.get("error", "Unknown error")
+ return f"Failed: {error}"
+ elif status == "cancelled":
+ return "Task was cancelled"
+ return "Unknown status"
+
+ def setup_connections(self):
+ """Setup button connections"""
+ self.cancel_button.clicked.connect(lambda: self.task_cancelled.emit(self.task_id))
+ self.retry_button.clicked.connect(lambda: self.task_retried.emit(self.task_id))
+ self.remove_button.clicked.connect(lambda: self.task_removed.emit(self.task_id))
+
+ def update_task(self, task_info):
+ """Update task information and UI"""
+ self.task_info = task_info
+
+ status_badges = {
+ "pending": ("Pending", "#9e9e9e"),
+ "running": ("Running", "#2196f3"),
+ "completed": ("Completed", "#4caf50"),
+ "failed": ("Failed", "#f44336"),
+ "cancelled": ("Cancelled", "#ff9800")
+ }
+ status = task_info.get("status", "pending")
+ status_text, status_color = status_badges.get(status, status_badges["pending"])
+
+ self.status_badge.setText(status_text)
+ self.status_badge.setStyleSheet(f"""
+ BodyLabel {{
+ background-color: {status_color}20;
+ color: {status_color};
+ padding: 2px 8px;
+ border-radius: 10px;
+ font-size: 10px;
+ font-weight: bold;
+ }}
+ """)
+
+ self.status_label.setText(self.get_status_text())
+ progress = task_info.get("progress", 0)
+ self.progress_bar.setValue(progress)
+
+ if status == "running":
+ self.cancel_button.setVisible(True)
+ self.retry_button.setVisible(False)
+ elif status in ["failed", "cancelled"]:
+ self.cancel_button.setVisible(False)
+ self.retry_button.setVisible(True)
+ else:
+ self.cancel_button.setVisible(False)
+ self.retry_button.setVisible(False)
+
+
+class HomeInterface(QFrame):
+ """Home interface showing app cards"""
+
+ def __init__(self, icon_paths, parent=None):
+ super().__init__(parent)
+ self.icon_paths = icon_paths
+ self.setObjectName("home_interface")
+ self.init_ui()
+
+ def init_ui(self):
+ """Initialize UI components"""
+ # Layouts
+ main_layout = QVBoxLayout(self)
+ main_layout.setContentsMargins(40, 35, 40, 35)
+ main_layout.setSpacing(25)
+
+ # Title
+ title_label = LargeTitleLabel("Converter")
+ title_label.setAlignment(Qt.AlignmentFlag.AlignCenter)
+ main_layout.addWidget(title_label)
+
+ # Image Converter card
+ image_card = AppCard(
+ icon_path=self.icon_paths['app_icon_path'],
+ title="Image Converter",
+ content="Convert PNG images to ICNS format for macOS applications",
+ app_type="image"
+ )
+ image_card.setBorderRadius(35)
+ main_layout.addWidget(image_card)
+
+ # Archive Converter card
+ archive_card = AppCard(
+ icon_path=self.icon_paths['zip_icon_path'],
+ title="Archive Converter",
+ content="Create and extract ZIP, RAR, and 7Z archive files",
+ app_type="arc"
+ )
+ main_layout.addWidget(archive_card)
+ archive_card.setBorderRadius(35)
+
+ # Add stretch to push content to top
+ main_layout.addStretch(1)
+
+
+class TaskInterface(ScrollArea):
+ """Task management interface"""
+
+ def __init__(self, parent=None):
+ super().__init__(parent)
+ self.setObjectName("task_interface")
+ self.setWidgetResizable(True)
+ self.task_cards = {}
+ self.init_ui()
+
+ def init_ui(self):
+ """Initialize UI components"""
+ from qfluentwidgets import setCustomStyleSheet
+
+ container_widget = QWidget()
+ container_widget.setObjectName("task_container")
+ self.setWidget(container_widget)
+
+ main_layout = QVBoxLayout(container_widget)
+ main_layout.setContentsMargins(20, 20, 20, 20)
+ main_layout.setSpacing(20)
+
+ title_label = SubtitleLabel("Task Manager")
+ main_layout.addWidget(title_label)
+
+ self.task_container = QWidget()
+ self.task_flow_layout = FlowLayout(self.task_container)
+ self.task_flow_layout.setContentsMargins(10, 10, 10, 10)
+ self.task_flow_layout.setSpacing(15)
+
+ main_layout.addWidget(self.task_container, 1)
+
+ controls_layout = QHBoxLayout()
+
+ self.clear_tasks_button = PrimaryPushButton("Clear Completed")
+ self.clear_tasks_button.setObjectName("clear_tasks_button")
+ self.clear_tasks_button.setFixedHeight(40)
+ controls_layout.addWidget(self.clear_tasks_button)
+
+ controls_layout.addStretch(1)
+
+ main_layout.addLayout(controls_layout)
+
+ task_count_label = CaptionLabel("Tasks will appear here when you start conversions")
+ task_count_label.setTextColor(QColor("#888888"), QColor("#666666"))
+ main_layout.addWidget(task_count_label, 0, Qt.AlignmentFlag.AlignCenter)
+
+ def add_task_card(self, task_id, task_type, task_info):
+ """Add a new task card to the interface"""
+ if task_id in self.task_cards:
+ return
+
+ task_card = AppCardTask(task_id, task_type, task_info, self.task_container)
+ self.task_cards[task_id] = task_card
+ self.task_flow_layout.addWidget(task_card)
+
+ task_card.task_cancelled.connect(self._on_task_cancelled)
+ task_card.task_retried.connect(self._on_task_retried)
+ task_card.task_removed.connect(self._on_task_removed)
+
+ def update_task_card(self, task_id, task_info):
+ """Update an existing task card"""
+ if task_id in self.task_cards:
+ self.task_cards[task_id].update_task(task_info)
+
+ def remove_task_card(self, task_id):
+ """Remove a task card from the interface"""
+ if task_id in self.task_cards:
+ task_card = self.task_cards.pop(task_id)
+ self.task_flow_layout.removeWidget(task_card)
+ task_card.deleteLater()
+
+ def clear_all_cards(self):
+ """Clear all task cards"""
+ for task_id in list(self.task_cards.keys()):
+ self.remove_task_card(task_id)
+
+ def _on_task_cancelled(self, task_id):
+ """Handle task cancellation"""
+ if hasattr(self.parent(), 'task_manager'):
+ self.parent().task_manager.cancel_task(task_id)
+
+ def _on_task_retried(self, task_id):
+ """Handle task retry"""
+ if hasattr(self.parent(), 'task_manager'):
+ self.parent().task_manager.retry_task(task_id)
+
+ def _on_task_removed(self, task_id):
+ """Handle task removal"""
+ self.remove_task_card(task_id)
+
+
+class SettingsInterface(QFrame):
+ """Settings interface"""
+
+ def __init__(self, parent=None):
+ super().__init__(parent)
+ self.setObjectName("settings_interface")
+ self.init_ui()
+ self.load_settings()
+ self._connect_settings_signals()
+
+ def init_ui(self):
+ """Initialize UI components"""
+ # Layouts
+ main_layout = QVBoxLayout(self)
+ main_layout.setContentsMargins(20, 20, 20, 20)
+ main_layout.setSpacing(15)
+
+ # Create SegmentedWidget and QStackedWidget
+ self.segmented_widget = SegmentedWidget(self)
+ setCustomStyleSheet(self.segmented_widget, CON.qss_seg, CON.qss_seg)
+ self.stacked_widget = QStackedWidget(self)
+
+ # General page
+ general_page = QWidget()
+ general_layout = QVBoxLayout(general_page)
+ general_layout.setContentsMargins(15, 15, 15, 15)
+ general_layout.setSpacing(15)
+
+ from settings.general_settings import GeneralSettingsWidget
+ self.general_widget = GeneralSettingsWidget()
+ self.general_widget.setObjectName("general_widget")
+ general_layout.addWidget(self.general_widget)
+ general_layout.addStretch()
+
+ self.stacked_widget.addWidget(general_page)
+
+ # Debug page
+ debug_page = QWidget()
+ debug_layout = QVBoxLayout(debug_page)
+ debug_layout.setContentsMargins(15, 15, 15, 15)
+ debug_layout.setSpacing(15)
+
+ from debug.debug_gui import DebugSettingsWidget
+ self.debug_widget = DebugSettingsWidget()
+ self.debug_widget.setObjectName("debug_widget")
+ debug_layout.addWidget(self.debug_widget)
+ debug_layout.addStretch()
+
+ self.stacked_widget.addWidget(debug_page)
+
+ # Update page
+ update_page = QWidget()
+ update_layout = QVBoxLayout(update_page)
+ update_layout.setContentsMargins(15, 15, 15, 15)
+ update_layout.setSpacing(15)
+
+ from settings.update_settings_gui import UpdateSettingsWidget
+ self.update_group = UpdateSettingsWidget()
+ self.update_group.setObjectName("update_group")
+ update_layout.addWidget(self.update_group)
+ update_layout.addStretch()
+
+ self.stacked_widget.addWidget(update_page)
+
+ # Add tab items
+ self.add_sub_interface(general_page, "general_page", "General")
+ self.add_sub_interface(debug_page, "debug_page", "Debug")
+ self.add_sub_interface(update_page, "update_page", "Update")
+
+ # Connect tab change signal
+ self.stacked_widget.currentChanged.connect(self.on_current_index_changed)
+ self.stacked_widget.setCurrentIndex(0)
+ self.segmented_widget.setCurrentItem("general_page")
+
+ # Add to main layout
+ main_layout.addWidget(self.segmented_widget, 0, Qt.AlignmentFlag.AlignHCenter)
+ main_layout.addWidget(self.stacked_widget, 1)
+
+ def add_sub_interface(self, widget: QWidget, object_name: str, text: str):
+ """Add sub-page to SegmentedWidget and QStackedWidget"""
+ widget.setObjectName(object_name)
+ self.segmented_widget.addItem(
+ routeKey=object_name,
+ text=text,
+ onClick=lambda: self.stacked_widget.setCurrentWidget(widget)
+ )
+
+ def on_current_index_changed(self, index):
+ """Handle current page change"""
+ widget = self.stacked_widget.widget(index)
+ if widget:
+ self.segmented_widget.setCurrentItem(widget.objectName())
+
+ def load_settings(self):
+ """Load settings from QSettings"""
+ settings = QSettings("MyCompany", "ConverterApp")
+ # Theme settings - always set to System Default (index 0)
+ settings.setValue("theme", 0) # Force save System Default
+
+ # Load General settings (includes Image Converter settings)
+ if hasattr(self, 'general_widget'):
+ self.general_widget.load_settings()
+
+ # Debug settings are now handled by the DebugSettingsWidget itself
+
+ def _connect_settings_signals(self):
+ """Connect all settings controls' signals to real-time saving"""
+ # Connect general widget settings
+ if hasattr(self, 'general_widget'):
+ self.general_widget.settings_changed.connect(self.on_settings_changed)
+
+ # Connect debug widget auto-save signals (already handled in DebugSettingsWidget)
+ # Debug settings are now handled by the DebugSettingsWidget itself
+
+ # Connect update dialog settings
+ # Update settings related signal connections have been removed, handled internally by UpdateDialog
+
+ def on_settings_changed(self):
+ """Handle any settings change and trigger auto-save"""
+ self.save_settings_async()
+
+ def save_settings_async(self):
+ """Asynchronously save settings in a separate thread"""
+ def save_thread():
+ settings = QSettings("MyCompany", "ConverterApp")
+ # Theme settings - always System Default
+ settings.setValue("theme", 0)
+
+ # Save General settings
+ if hasattr(self, 'general_widget'):
+ self.general_widget.save_settings()
+
+ # Debug settings are now handled by the DebugSettingsWidget itself
+
+ # Image converter settings are now saved by the general widget
+ # No separate image converter widget exists anymore
+
+ settings.sync() # Ensure settings are written to disk
+
+ # Start separate thread to execute save operation
+ threading.Thread(target=save_thread).start()
+
+
+class MainWindow(FluentWindow):
+ """Main application window"""
def _load_qss_file(self, filename):
"""Load QSS content from external file"""
@@ -84,13 +607,11 @@ def DARK_QSS(self):
def __init__(self, q_app: QApplication):
super().__init__()
self._q_app = q_app # Store QApplication instance
- self.setWindowTitle("Converter")
- # Load theme setting immediately
self.settings = QSettings("MyCompany", "ConverterApp")
self.theme_setting = self.settings.value("theme", 0, type=int)
self.themeListener = SystemThemeListener(self)
- self.path= os.path.dirname(os.path.abspath(__file__))
+ self.path = os.path.dirname(os.path.abspath(__file__))
# Define paths for icon files
self.app_icon_path = os.path.join(self.path,"AppIcon.png")
self.appd_icon_path = os.path.join(self.path,"AppIcond.png")
@@ -113,15 +634,94 @@ def __init__(self, q_app: QApplication):
print("Note: zipd.png file not found. Will try to create a PNG placeholder icon.")
create_placeholder_icon(self.zipd_icon_path, "dimgray", "ZipD")
- self.init_ui()
+ # Icon paths dictionary for home interface
+ self.icon_paths = {
+ 'app_icon_path': self.app_icon_path,
+ 'appd_icon_path': self.appd_icon_path,
+ 'zip_icon_path': self.zip_icon_path,
+ 'zipd_icon_path': self.zipd_icon_path
+ }
+
+ # Initialize interfaces
+ self.init_interfaces()
+
+ # Initialize window
+ self.init_window()
+ self.init_navigation()
+
+ # Initialize task manager
+ self.init_task_manager()
+
+ # Apply theme
setTheme(Theme.AUTO)
self.themeListener.start()
qconfig.themeChanged.connect(self._onThemeChanged)
- # Apply theme based on settings or initial system detection
- self._apply_system_theme_from_settings()
+ self._apply_system_theme_from_settings()
+
+ def init_interfaces(self):
+ """Initialize sub-interfaces"""
+ # Create home interface with app cards
+ self.home_interface = HomeInterface(self.icon_paths)
+
+ # Create task interface for task management
+ self.task_interface = TaskInterface()
+
+ # Create settings interface
+ self.settings_interface = SettingsInterface()
+
+ def init_window(self):
+ """Initialize window properties"""
+ self.setWindowTitle("Converter")
+ self.setWindowIcon(QIcon(self.app_icon_path))
+ self.resize(900, 700)
+
+ def init_navigation(self):
+ """Initialize navigation items"""
+ self.addSubInterface(
+ self.home_interface,
+ FIF.HOME,
+ 'Home'
+ )
+
+ self.addSubInterface(
+ self.task_interface,
+ FIF.HISTORY,
+ 'Task Manager'
+ )
+
+ self.addSubInterface(
+ self.settings_interface,
+ FIF.SETTING,
+ 'Settings',
+ NavigationItemPosition.BOTTOM
+ )
def closeEvent(self, event):
"""窗口关闭事件"""
- # 停止监听器线程
+ # Check if task mode is enabled and if any sub-windows are open
+ task_mode_enabled = self.settings.value("task_mode", False, type=bool)
+ if task_mode_enabled:
+ has_open_windows = False
+ # Get all top level widgets
+ for widget in self._q_app.topLevelWidgets():
+ # Check if there are any image or arc windows open
+ if widget is not self:
+ window_title = widget.windowTitle()
+ if "Image Converter" in window_title or "Archive File Processing Tool" in window_title:
+ has_open_windows = True
+ break
+
+ if has_open_windows:
+ # Show modal dialog explaining why closing is not allowed
+ QMessageBox.information(
+ self,
+ "Cannot Close",
+ "Task Mode is enabled and sub-windows are open. Please close all sub-windows first.",
+ QMessageBox.StandardButton.Ok
+ )
+ event.ignore()
+ return
+
+ # Stop listener thread
if hasattr(self, 'themeListener'):
self.themeListener.terminate()
self.themeListener.deleteLater()
@@ -168,8 +768,59 @@ def update_sub_widgets_theme(self, is_dark_mode):
self._settings_dialog.apply_theme(is_dark_mode)
def init_ui(self):
- # Create main layout
- main_layout = QVBoxLayout()
+ # Create main horizontal layout for sidebar and content
+ main_horizontal_layout = QHBoxLayout()
+
+ # --- Task Sidebar ---
+ # Create sidebar widget
+ from qfluentwidgets import PushButton, setCustomStyleSheet
+
+ # Keep using QWidget as the container
+ self.sidebar_widget = QWidget()
+ self.sidebar_widget.setObjectName("sidebar_widget")
+ self.sidebar_layout = QVBoxLayout(self.sidebar_widget)
+
+ # Sidebar title
+ sidebar_title = QLabel("Task Manager")
+ sidebar_title.setObjectName("sidebar_title")
+ sidebar_title.setAlignment(Qt.AlignmentFlag.AlignCenter)
+ self.sidebar_layout.addWidget(sidebar_title)
+
+ # Task list
+ self.task_list = QListWidget()
+ self.task_list.setObjectName("task_list")
+ from con import CON
+ setCustomStyleSheet(self.task_list, CON.qss_combo, CON.qss_combo)
+ self.sidebar_layout.addWidget(self.task_list)
+
+ # Sidebar control buttons
+ sidebar_controls = QHBoxLayout()
+
+ # Clear completed tasks button
+ self.clear_tasks_button = PushButton("Clear Completed")
+ self.clear_tasks_button.setObjectName("clear_tasks_button")
+ self.clear_tasks_button.setIconSize(QSize(16, 16))
+ setCustomStyleSheet(self.clear_tasks_button, CON.qss, CON.qss)
+ sidebar_controls.addWidget(self.clear_tasks_button)
+
+ # Collapse/Expand button
+ self.toggle_sidebar_button = PushButton("Collapse")
+ self.toggle_sidebar_button.setObjectName("toggle_sidebar_button")
+ self.toggle_sidebar_button.setIconSize(QSize(16, 16))
+ self.toggle_sidebar_button.clicked.connect(self.toggle_sidebar)
+ setCustomStyleSheet(self.toggle_sidebar_button, CON.qss, CON.qss)
+ sidebar_controls.addWidget(self.toggle_sidebar_button)
+
+ self.sidebar_layout.addLayout(sidebar_controls)
+
+ # Add sidebar to main horizontal layout
+ self.sidebar_widget.setFixedWidth(250)
+ main_horizontal_layout.addWidget(self.sidebar_widget)
+
+ # --- Main Content ---
+ # Create main content widget with vertical layout
+ content_widget = QWidget()
+ main_layout = QVBoxLayout(content_widget)
main_layout.setSpacing(25) # Increased spacing for better visual separation
main_layout.setContentsMargins(40, 35, 40, 35) # Better margins
@@ -258,14 +909,183 @@ def init_ui(self):
settings_button_layout.addWidget(settings_button)
settings_button_layout.addStretch()
main_layout.addLayout(settings_button_layout)
-
- # Set the main layout for the window
- self.setLayout(main_layout)
+
+ # Add content widget to main horizontal layout
+ main_horizontal_layout.addWidget(content_widget, 1) # Give content stretch priority
+
+ # Add task count indicator (visible when sidebar is collapsed)
+ self.task_count_indicator = QLabel("0")
+ self.task_count_indicator.setObjectName("task_count_indicator")
+ self.task_count_indicator.setAlignment(Qt.AlignmentFlag.AlignCenter)
+ self.task_count_indicator.setStyleSheet("""
+ background-color: #0078d4;
+ color: white;
+ border-radius: 12px;
+ padding: 5px 10px;
+ font-weight: bold;
+ """)
+ self.task_count_indicator.setFixedSize(30, 30)
+ self.task_count_indicator.hide() # Initially hidden
+
+ # Set the main horizontal layout for the window
+ self.setLayout(main_horizontal_layout)
+
+ # Initialize task manager
+ self.init_task_manager()
def show_settings(self):
settings_dialog = SettingsDialog(self)
self._settings_dialog = settings_dialog # Save dialog reference
settings_dialog.show() # Use show() instead of exec() to keep dialog non-modal
+
+ def toggle_sidebar(self):
+ """Toggle sidebar visibility"""
+ if self.sidebar_widget.isVisible():
+ # Hide sidebar and show task count indicator
+ self.sidebar_widget.hide()
+ self.task_count_indicator.show()
+ self.toggle_sidebar_button.setText("Expand")
+ else:
+ # Show sidebar and hide task count indicator
+ self.sidebar_widget.show()
+ self.task_count_indicator.hide()
+ self.toggle_sidebar_button.setText("Collapse")
+
+ def init_task_manager(self):
+ """Initialize task manager and connect signals"""
+ # Import task manager
+ from support.task_manager import TaskManager
+
+ # Create task manager instance
+ self.task_manager = TaskManager()
+
+ # Connect task manager signals to UI updates
+ self.task_manager.task_added.connect(self._on_task_added)
+ self.task_manager.task_updated.connect(self._on_task_updated)
+ self.task_manager.task_completed.connect(self._on_task_completed)
+ self.task_manager.task_failed.connect(self._on_task_failed)
+ self.task_manager.progress_updated.connect(self._on_task_progress_updated)
+
+ # Connect task interface control buttons
+ self.task_interface.clear_tasks_button.clicked.connect(self._clear_completed_tasks)
+
+ # Start task file monitor
+ self._start_task_monitor()
+
+ def _start_task_monitor(self):
+ """Start task file monitor to check for new or updated tasks"""
+ # Create a timer to check task files every 1 second
+ self.task_monitor_timer = QTimer(self)
+ self.task_monitor_timer.timeout.connect(self._check_task_files)
+ self.task_monitor_timer.start(1000) # Check every 1 second
+
+ # Create task directory if it doesn't exist
+ self.task_dir = os.path.expanduser("~/.converter/tasks")
+ os.makedirs(self.task_dir, exist_ok=True)
+
+ def _check_task_files(self):
+ """Check for new or updated task files"""
+ import json
+
+ # Get all task files in the directory
+ try:
+ task_files = [f for f in os.listdir(self.task_dir) if f.startswith("task_") and f.endswith(".json")]
+
+ for task_file in task_files:
+ task_file_path = os.path.join(self.task_dir, task_file)
+ task_id = task_file[5:-5] # Extract task_id from filename
+
+ # Read task info from file
+ with open(task_file_path, "r") as f:
+ task_info = json.load(f)
+
+ # Check if task already exists in task manager
+ if task_id in self.task_manager.tasks:
+ # Update existing task
+ self._update_existing_task(task_id, task_info)
+ else:
+ # Add new task
+ self._add_new_task(task_id, task_info)
+
+ except Exception as e:
+ print(f"Error checking task files: {e}")
+
+ def _add_new_task(self, task_id, task_info):
+ """Add a new task to task manager"""
+ # Create task using task manager's add_task method
+ self.task_manager.add_task(
+ task_info["task_type"],
+ task_info["input_path"],
+ task_info["output_path"]
+ )
+
+ def _update_existing_task(self, task_id, task_info):
+ """Update an existing task in task manager"""
+ # Update task progress
+ self.task_manager.tasks[task_id].progress = task_info["progress"]
+ self.task_manager.tasks[task_id].status = task_info["status"]
+
+ # Emit progress updated signal
+ self.task_manager.progress_updated.emit(task_id, task_info["progress"])
+
+ # If task is completed or failed, emit appropriate signal
+ if task_info["status"] == "completed":
+ self.task_manager.task_completed.emit(task_id, task_info)
+ elif task_info["status"] == "failed":
+ self.task_manager.task_failed.emit(task_id, task_info, task_info.get("error", "Unknown error"))
+
+ def _on_task_added(self, task_id, task_info):
+ """Handle new task added"""
+ self.task_interface.add_task_card(task_id, task_info['task_type'], task_info)
+ self._update_task_count()
+
+ def _on_task_updated(self, task_id, task_info):
+ """Handle task updated"""
+ self.task_interface.update_task_card(task_id, task_info)
+
+ def _on_task_completed(self, task_id, task_info):
+ """Handle task completed"""
+ self._on_task_updated(task_id, task_info)
+
+ # Send system notification
+ from support.notification import send_notification
+ send_notification(
+ "Task Completed",
+ f"{task_info['task_type'].capitalize()} conversion completed: {os.path.basename(task_info['input_path'])}"
+ )
+
+ def _on_task_failed(self, task_id, task_info, error):
+ """Handle task failed"""
+ # Update task in list
+ self._on_task_updated(task_id, task_info)
+
+ # Send system notification
+ from support.notification import send_notification
+ send_notification(
+ "Task Failed",
+ f"{task_info['task_type'].capitalize()} conversion failed: {os.path.basename(task_info['input_path'])}"
+ )
+
+ def _on_task_progress_updated(self, task_id, progress):
+ """Handle task progress updated"""
+ for task_id_key in self.task_interface.task_cards:
+ if task_id_key == task_id:
+ task_info = self.task_manager.tasks.get(task_id, {})
+ if task_info:
+ task_info['progress'] = progress
+ self.task_interface.update_task_card(task_id, task_info)
+ break
+
+ def _clear_completed_tasks(self):
+ """Clear completed tasks from list"""
+ self.task_manager.clear_completed_tasks()
+ self.task_interface.clear_all_cards()
+ self._update_task_count()
+
+ def _update_task_count(self):
+ """Update task count indicator (no longer needed in FluentWindow)"""
+ # Task count indicator is no longer needed in FluentWindow
+ pass
class AnimatedAppDialog(QDialog):
def __init__(self, parent=None, app_type=""):
@@ -420,16 +1240,25 @@ def run_image_app():
try:
# Get the main window instance
app = QApplication.instance()
+ if app is None:
+ return
main_window = None
for widget in app.topLevelWidgets():
- if isinstance(widget, IconButtonsWindow):
+ if isinstance(widget, MainWindow):
main_window = widget
break
if main_window:
- # Create and show the animation dialog
- dialog = ImageAppDialog(main_window)
- dialog.show()
+ # Check if task mode is enabled
+ task_mode_enabled = main_window.settings.value("task_mode", False, type=bool)
+
+ if task_mode_enabled:
+ # In task mode, run directly in the same process
+ run_image()
+ else:
+ # Create and show the animation dialog
+ dialog = ImageAppDialog(main_window)
+ dialog.show()
else:
# Fallback to multiprocessing if no main window found
multiprocessing.Process(target=run_image).start()
@@ -444,16 +1273,25 @@ def run_zip_app():
try:
# Get the main window instance
app = QApplication.instance()
+ if app is None:
+ return
main_window = None
for widget in app.topLevelWidgets():
- if isinstance(widget, IconButtonsWindow):
+ if isinstance(widget, MainWindow):
main_window = widget
break
if main_window:
- # Create and show the animation dialog
- dialog = ZipAppDialog(main_window)
- dialog.show()
+ # Check if task mode is enabled
+ task_mode_enabled = main_window.settings.value("task_mode", False, type=bool)
+
+ if task_mode_enabled:
+ # In task mode, run directly in the same process
+ run_zip()
+ else:
+ # Create and show the animation dialog
+ dialog = ZipAppDialog(main_window)
+ dialog.show()
else:
# Fallback to multiprocessing if no main window found
multiprocessing.Process(target=run_zip).start()
@@ -481,7 +1319,7 @@ def run_zip_app():
from support.toggle import theme_manager
theme_manager.start()
setTheme(Theme.AUTO)
- window = IconButtonsWindow(q_app=app)
+ window = MainWindow(q_app=app)
window.show()
# Connect to palette changes for real-time theme switching ONLY if setting is System Default
app.paletteChanged.connect(lambda: window._apply_system_theme(app.palette().color(QPalette.ColorRole.Window).lightnessF() < 0.5))
diff --git a/Converter.spec b/Converter.spec
index 0c8a875..c6b098a 100644
--- a/Converter.spec
+++ b/Converter.spec
@@ -261,12 +261,12 @@ app = BUNDLE(
coll,
name='Converter.app',
icon=os.path.join(current_dir, 'AppIcon.icns'),
- bundle_identifier='com.pyquick.converter',
+ bundle_identifier='com.intsant.converter',
info_plist={
'CFBundleDisplayName': 'Converter',
'CFBundleExecutable': 'Converter',
'CFBundleIconFile': 'AppIcon.icns',
- 'CFBundleIdentifier': 'com.pyquick.converter',
+ 'CFBundleIdentifier': 'com.intsant.converter',
'CFBundleInfoDictionaryVersion': '6.0',
'CFBundleName': 'Converter',
'CFBundlePackageType': 'APPL',
@@ -275,5 +275,5 @@ app = BUNDLE(
'NSHighResolutionCapable': True,
'LSMinimumSystemVersion': '11.7',
},
- version='2.0.0',
+ version='2.1.0A6',
)
\ No newline at end of file
diff --git a/LICENSE.txt b/LICENSE.txt
new file mode 100644
index 0000000..53d1f3d
--- /dev/null
+++ b/LICENSE.txt
@@ -0,0 +1,675 @@
+ GNU GENERAL PUBLIC LICENSE
+ Version 3, 29 June 2007
+
+ Copyright (C) 2007 Free Software Foundation, Inc.
+ Everyone is permitted to copy and distribute verbatim copies
+ of this license document, but changing it is not allowed.
+
+ Preamble
+
+ The GNU General Public License is a free, copyleft license for
+software and other kinds of works.
+
+ The licenses for most software and other practical works are designed
+to take away your freedom to share and change the works. By contrast,
+the GNU General Public License is intended to guarantee your freedom to
+share and change all versions of a program--to make sure it remains free
+software for all its users. We, the Free Software Foundation, use the
+GNU General Public License for most of our software; it applies also to
+any other work released this way by its authors. You can apply it to
+your programs, too.
+
+ When we speak of free software, we are referring to freedom, not
+price. Our General Public Licenses are designed to make sure that you
+have the freedom to distribute copies of free software (and charge for
+them if you wish), that you receive source code or can get it if you
+want it, that you can change the software or use pieces of it in new
+free programs, and that you know you can do these things.
+
+ To protect your rights, we need to prevent others from denying you
+these rights or asking you to surrender the rights. Therefore, you have
+certain responsibilities if you distribute copies of the software, or if
+you modify it: responsibilities to respect the freedom of others.
+
+ For example, if you distribute copies of such a program, whether
+gratis or for a fee, you must pass on to the recipients the same
+freedoms that you received. You must make sure that they, too, receive
+or can get the source code. And you must show them these terms so they
+know their rights.
+
+ Developers that use the GNU GPL protect your rights with two steps:
+(1) assert copyright on the software, and (2) offer you this License
+giving you legal permission to copy, distribute and/or modify it.
+
+ For the developers' and authors' protection, the GPL clearly explains
+that there is no warranty for this free software. For both users' and
+authors' sake, the GPL requires that modified versions be marked as
+changed, so that their problems will not be attributed erroneously to
+authors of previous versions.
+
+ Some devices are designed to deny users access to install or run
+modified versions of the software inside them, although the manufacturer
+can do so. This is fundamentally incompatible with the aim of
+protecting users' freedom to change the software. The systematic
+pattern of such abuse occurs in the area of products for individuals to
+use, which is precisely where it is most unacceptable. Therefore, we
+have designed this version of the GPL to prohibit the practice for those
+products. If such problems arise substantially in other domains, we
+stand ready to extend this provision to those domains in future versions
+of the GPL, as needed to protect the freedom of users.
+
+ Finally, every program is threatened constantly by software patents.
+States should not allow patents to restrict development and use of
+software on general-purpose computers, but in those that do, we wish to
+avoid the special danger that patents applied to a free program could
+make it effectively proprietary. To prevent this, the GPL assures that
+patents cannot be used to render the program non-free.
+
+ The precise terms and conditions for copying, distribution and
+modification follow.
+
+ TERMS AND CONDITIONS
+
+ 0. Definitions.
+
+ "This License" refers to version 3 of the GNU General Public License.
+
+ "Copyright" also means copyright-like laws that apply to other kinds of
+works, such as semiconductor masks.
+
+ "The Program" refers to any copyrightable work licensed under this
+License. Each licensee is addressed as "you". "Licensees" and
+"recipients" may be individuals or organizations.
+
+ To "modify" a work means to copy from or adapt all or part of the work
+in a fashion requiring copyright permission, other than the making of an
+exact copy. The resulting work is called a "modified version" of the
+earlier work or a work "based on" the earlier work.
+
+ A "covered work" means either the unmodified Program or a work based
+on the Program.
+
+ To "propagate" a work means to do anything with it that, without
+permission, would make you directly or secondarily liable for
+infringement under applicable copyright law, except executing it on a
+computer or modifying a private copy. Propagation includes copying,
+distribution (with or without modification), making available to the
+public, and in some countries other activities as well.
+
+ To "convey" a work means any kind of propagation that enables other
+parties to make or receive copies. Mere interaction with a user through
+a computer network, with no transfer of a copy, is not conveying.
+
+ An interactive user interface displays "Appropriate Legal Notices"
+to the extent that it includes a convenient and prominently visible
+feature that (1) displays an appropriate copyright notice, and (2)
+tells the user that there is no warranty for the work (except to the
+extent that warranties are provided), that licensees may convey the
+work under this License, and how to view a copy of this License. If
+the interface presents a list of user commands or options, such as a
+menu, a prominent item in the list meets this criterion.
+
+ 1. Source Code.
+
+ The "source code" for a work means the preferred form of the work
+for making modifications to it. "Object code" means any non-source
+form of a work.
+
+ A "Standard Interface" means an interface that either is an official
+standard defined by a recognized standards body, or, in the case of
+interfaces specified for a particular programming language, one that
+is widely used among developers working in that language.
+
+ The "System Libraries" of an executable work include anything, other
+than the work as a whole, that (a) is included in the normal form of
+packaging a Major Component, but which is not part of that Major
+Component, and (b) serves only to enable use of the work with that
+Major Component, or to implement a Standard Interface for which an
+implementation is available to the public in source code form. A
+"Major Component", in this context, means a major essential component
+(kernel, window system, and so on) of the specific operating system
+(if any) on which the executable work runs, or a compiler used to
+produce the work, or an object code interpreter used to run it.
+
+ The "Corresponding Source" for a work in object code form means all
+the source code needed to generate, install, and (for an executable
+work) run the object code and to modify the work, including scripts to
+control those activities. However, it does not include the work's
+System Libraries, or general-purpose tools or generally available free
+programs which are used unmodified in performing those activities but
+which are not part of the work. For example, Corresponding Source
+includes interface definition files associated with source files for
+the work, and the source code for shared libraries and dynamically
+linked subprograms that the work is specifically designed to require,
+such as by intimate data communication or control flow between those
+subprograms and other parts of the work.
+
+ The Corresponding Source need not include anything that users
+can regenerate automatically from other parts of the Corresponding
+Source.
+
+ The Corresponding Source for a work in source code form is that
+same work.
+
+ 2. Basic Permissions.
+
+ All rights granted under this License are granted for the term of
+copyright on the Program, and are irrevocable provided the stated
+conditions are met. This License explicitly affirms your unlimited
+permission to run the unmodified Program. The output from running a
+covered work is covered by this License only if the output, given its
+content, constitutes a covered work. This License acknowledges your
+rights of fair use or other equivalent, as provided by copyright law.
+
+ You may make, run and propagate covered works that you do not
+convey, without conditions so long as your license otherwise remains
+in force. You may convey covered works to others for the sole purpose
+of having them make modifications exclusively for you, or provide you
+with facilities for running those works, provided that you comply with
+the terms of this License in conveying all material for which you do
+not control copyright. Those thus making or running the covered works
+for you must do so exclusively on your behalf, under your direction
+and control, on terms that prohibit them from making any copies of
+your copyrighted material outside their relationship with you.
+
+ Conveying under any other circumstances is permitted solely under
+the conditions stated below. Sublicensing is not allowed; section 10
+makes it unnecessary.
+
+ 3. Protecting Users' Legal Rights From Anti-Circumvention Law.
+
+ No covered work shall be deemed part of an effective technological
+measure under any applicable law fulfilling obligations under article
+11 of the WIPO copyright treaty adopted on 20 December 1996, or
+similar laws prohibiting or restricting circumvention of such
+measures.
+
+ When you convey a covered work, you waive any legal power to forbid
+circumvention of technological measures to the extent such circumvention
+is effected by exercising rights under this License with respect to
+the covered work, and you disclaim any intention to limit operation or
+modification of the work as a means of enforcing, against the work's
+users, your or third parties' legal rights to forbid circumvention of
+technological measures.
+
+ 4. Conveying Verbatim Copies.
+
+ You may convey verbatim copies of the Program's source code as you
+receive it, in any medium, provided that you conspicuously and
+appropriately publish on each copy an appropriate copyright notice;
+keep intact all notices stating that this License and any
+non-permissive terms added in accord with section 7 apply to the code;
+keep intact all notices of the absence of any warranty; and give all
+recipients a copy of this License along with the Program.
+
+ You may charge any price or no price for each copy that you convey,
+and you may offer support or warranty protection for a fee.
+
+ 5. Conveying Modified Source Versions.
+
+ You may convey a work based on the Program, or the modifications to
+produce it from the Program, in the form of source code under the
+terms of section 4, provided that you also meet all of these conditions:
+
+ a) The work must carry prominent notices stating that you modified
+ it, and giving a relevant date.
+
+ b) The work must carry prominent notices stating that it is
+ released under this License and any conditions added under section
+ 7. This requirement modifies the requirement in section 4 to
+ "keep intact all notices".
+
+ c) You must license the entire work, as a whole, under this
+ License to anyone who comes into possession of a copy. This
+ License will therefore apply, along with any applicable section 7
+ additional terms, to the whole of the work, and all its parts,
+ regardless of how they are packaged. This License gives no
+ permission to license the work in any other way, but it does not
+ invalidate such permission if you have separately received it.
+
+ d) If the work has interactive user interfaces, each must display
+ Appropriate Legal Notices; however, if the Program has interactive
+ interfaces that do not display Appropriate Legal Notices, your
+ work need not make them do so.
+
+ A compilation of a covered work with other separate and independent
+works, which are not by their nature extensions of the covered work,
+and which are not combined with it such as to form a larger program,
+in or on a volume of a storage or distribution medium, is called an
+"aggregate" if the compilation and its resulting copyright are not
+used to limit the access or legal rights of the compilation's users
+beyond what the individual works permit. Inclusion of a covered work
+in an aggregate does not cause this License to apply to the other
+parts of the aggregate.
+
+ 6. Conveying Non-Source Forms.
+
+ You may convey a covered work in object code form under the terms
+of sections 4 and 5, provided that you also convey the
+machine-readable Corresponding Source under the terms of this License,
+in one of these ways:
+
+ a) Convey the object code in, or embodied in, a physical product
+ (including a physical distribution medium), accompanied by the
+ Corresponding Source fixed on a durable physical medium
+ customarily used for software interchange.
+
+ b) Convey the object code in, or embodied in, a physical product
+ (including a physical distribution medium), accompanied by a
+ written offer, valid for at least three years and valid for as
+ long as you offer spare parts or customer support for that product
+ model, to give anyone who possesses the object code either (1) a
+ copy of the Corresponding Source for all the software in the
+ product that is covered by this License, on a durable physical
+ medium customarily used for software interchange, for a price no
+ more than your reasonable cost of physically performing this
+ conveying of source, or (2) access to copy the
+ Corresponding Source from a network server at no charge.
+
+ c) Convey individual copies of the object code with a copy of the
+ written offer to provide the Corresponding Source. This
+ alternative is allowed only occasionally and noncommercially, and
+ only if you received the object code with such an offer, in accord
+ with subsection 6b.
+
+ d) Convey the object code by offering access from a designated
+ place (gratis or for a charge), and offer equivalent access to the
+ Corresponding Source in the same way through the same place at no
+ further charge. You need not require recipients to copy the
+ Corresponding Source along with the object code. If the place to
+ copy the object code is a network server, the Corresponding Source
+ may be on a different server (operated by you or a third party)
+ that supports equivalent copying facilities, provided you maintain
+ clear directions next to the object code saying where to find the
+ Corresponding Source. Regardless of what server hosts the
+ Corresponding Source, you remain obligated to ensure that it is
+ available for as long as needed to satisfy these requirements.
+
+ e) Convey the object code using peer-to-peer transmission, provided
+ you inform other peers where the object code and Corresponding
+ Source of the work are being offered to the general public at no
+ charge under subsection 6d.
+
+ A separable portion of the object code, whose source code is excluded
+from the Corresponding Source as a System Library, need not be
+included in conveying the object code work.
+
+ A "User Product" is either (1) a "consumer product", which means any
+tangible personal property which is normally used for personal, family,
+or household purposes, or (2) anything designed or sold for incorporation
+into a dwelling. In determining whether a product is a consumer product,
+doubtful cases shall be resolved in favor of coverage. For a particular
+product received by a particular user, "normally used" refers to a
+typical or common use of that class of product, regardless of the status
+of the particular user or of the way in which the particular user
+actually uses, or expects or is expected to use, the product. A product
+is a consumer product regardless of whether the product has substantial
+commercial, industrial or non-consumer uses, unless such uses represent
+the only significant mode of use of the product.
+
+ "Installation Information" for a User Product means any methods,
+procedures, authorization keys, or other information required to install
+and execute modified versions of a covered work in that User Product from
+a modified version of its Corresponding Source. The information must
+suffice to ensure that the continued functioning of the modified object
+code is in no case prevented or interfered with solely because
+modification has been made.
+
+ If you convey an object code work under this section in, or with, or
+specifically for use in, a User Product, and the conveying occurs as
+part of a transaction in which the right of possession and use of the
+User Product is transferred to the recipient in perpetuity or for a
+fixed term (regardless of how the transaction is characterized), the
+Corresponding Source conveyed under this section must be accompanied
+by the Installation Information. But this requirement does not apply
+if neither you nor any third party retains the ability to install
+modified object code on the User Product (for example, the work has
+been installed in ROM).
+
+ The requirement to provide Installation Information does not include a
+requirement to continue to provide support service, warranty, or updates
+for a work that has been modified or installed by the recipient, or for
+the User Product in which it has been modified or installed. Access to a
+network may be denied when the modification itself materially and
+adversely affects the operation of the network or violates the rules and
+protocols for communication across the network.
+
+ Corresponding Source conveyed, and Installation Information provided,
+in accord with this section must be in a format that is publicly
+documented (and with an implementation available to the public in
+source code form), and must require no special password or key for
+unpacking, reading or copying.
+
+ 7. Additional Terms.
+
+ "Additional permissions" are terms that supplement the terms of this
+License by making exceptions from one or more of its conditions.
+Additional permissions that are applicable to the entire Program shall
+be treated as though they were included in this License, to the extent
+that they are valid under applicable law. If additional permissions
+apply only to part of the Program, that part may be used separately
+under those permissions, but the entire Program remains governed by
+this License without regard to the additional permissions.
+
+ When you convey a copy of a covered work, you may at your option
+remove any additional permissions from that copy, or from any part of
+it. (Additional permissions may be written to require their own
+removal in certain cases when you modify the work.) You may place
+additional permissions on material, added by you to a covered work,
+for which you have or can give appropriate copyright permission.
+
+ Notwithstanding any other provision of this License, for material you
+add to a covered work, you may (if authorized by the copyright holders of
+that material) supplement the terms of this License with terms:
+
+ a) Disclaiming warranty or limiting liability differently from the
+ terms of sections 15 and 16 of this License; or
+
+ b) Requiring preservation of specified reasonable legal notices or
+ author attributions in that material or in the Appropriate Legal
+ Notices displayed by works containing it; or
+
+ c) Prohibiting misrepresentation of the origin of that material, or
+ requiring that modified versions of such material be marked in
+ reasonable ways as different from the original version; or
+
+ d) Limiting the use for publicity purposes of names of licensors or
+ authors of the material; or
+
+ e) Declining to grant rights under trademark law for use of some
+ trade names, trademarks, or service marks; or
+
+ f) Requiring indemnification of licensors and authors of that
+ material by anyone who conveys the material (or modified versions of
+ it) with contractual assumptions of liability to the recipient, for
+ any liability that these contractual assumptions directly impose on
+ those licensors and authors.
+
+ All other non-permissive additional terms are considered "further
+restrictions" within the meaning of section 10. If the Program as you
+received it, or any part of it, contains a notice stating that it is
+governed by this License along with a term that is a further
+restriction, you may remove that term. If a license document contains
+a further restriction but permits relicensing or conveying under this
+License, you may add to a covered work material governed by the terms
+of that license document, provided that the further restriction does
+not survive such relicensing or conveying.
+
+ If you add terms to a covered work in accord with this section, you
+must place, in the relevant source files, a statement of the
+additional terms that apply to those files, or a notice indicating
+where to find the applicable terms.
+
+ Additional terms, permissive or non-permissive, may be stated in the
+form of a separately written license, or stated as exceptions;
+the above requirements apply either way.
+
+ 8. Termination.
+
+ You may not propagate or modify a covered work except as expressly
+provided under this License. Any attempt otherwise to propagate or
+modify it is void, and will automatically terminate your rights under
+this License (including any patent licenses granted under the third
+paragraph of section 11).
+
+ However, if you cease all violation of this License, then your
+license from a particular copyright holder is reinstated (a)
+provisionally, unless and until the copyright holder explicitly and
+finally terminates your license, and (b) permanently, if the copyright
+holder fails to notify you of the violation by some reasonable means
+prior to 60 days after the cessation.
+
+ Moreover, your license from a particular copyright holder is
+reinstated permanently if the copyright holder notifies you of the
+violation by some reasonable means, this is the first time you have
+received notice of violation of this License (for any work) from that
+copyright holder, and you cure the violation prior to 30 days after
+your receipt of the notice.
+
+ Termination of your rights under this section does not terminate the
+licenses of parties who have received copies or rights from you under
+this License. If your rights have been terminated and not permanently
+reinstated, you do not qualify to receive new licenses for the same
+material under section 10.
+
+ 9. Acceptance Not Required for Having Copies.
+
+ You are not required to accept this License in order to receive or
+run a copy of the Program. Ancillary propagation of a covered work
+occurring solely as a consequence of using peer-to-peer transmission
+to receive a copy likewise does not require acceptance. However,
+nothing other than this License grants you permission to propagate or
+modify any covered work. These actions infringe copyright if you do
+not accept this License. Therefore, by modifying or propagating a
+covered work, you indicate your acceptance of this License to do so.
+
+ 10. Automatic Licensing of Downstream Recipients.
+
+ Each time you convey a covered work, the recipient automatically
+receives a license from the original licensors, to run, modify and
+propagate that work, subject to this License. You are not responsible
+for enforcing compliance by third parties with this License.
+
+ An "entity transaction" is a transaction transferring control of an
+organization, or substantially all assets of one, or subdividing an
+organization, or merging organizations. If propagation of a covered
+work results from an entity transaction, each party to that
+transaction who receives a copy of the work also receives whatever
+licenses to the work the party's predecessor in interest had or could
+give under the previous paragraph, plus a right to possession of the
+Corresponding Source of the work from the predecessor in interest, if
+the predecessor has it or can get it with reasonable efforts.
+
+ You may not impose any further restrictions on the exercise of the
+rights granted or affirmed under this License. For example, you may
+not impose a license fee, royalty, or other charge for exercise of
+rights granted under this License, and you may not initiate litigation
+(including a cross-claim or counterclaim in a lawsuit) alleging that
+any patent claim is infringed by making, using, selling, offering for
+sale, or importing the Program or any portion of it.
+
+ 11. Patents.
+
+ A "contributor" is a copyright holder who authorizes use under this
+License of the Program or a work on which the Program is based. The
+work thus licensed is called the contributor's "contributor version".
+
+ A contributor's "essential patent claims" are all patent claims
+owned or controlled by the contributor, whether already acquired or
+hereafter acquired, that would be infringed by some manner, permitted
+by this License, of making, using, or selling its contributor version,
+but do not include claims that would be infringed only as a
+consequence of further modification of the contributor version. For
+purposes of this definition, "control" includes the right to grant
+patent sublicenses in a manner consistent with the requirements of
+this License.
+
+ Each contributor grants you a non-exclusive, worldwide, royalty-free
+patent license under the contributor's essential patent claims, to
+make, use, sell, offer for sale, import and otherwise run, modify and
+propagate the contents of its contributor version.
+
+ In the following three paragraphs, a "patent license" is any express
+agreement or commitment, however denominated, not to enforce a patent
+(such as an express permission to practice a patent or covenant not to
+sue for patent infringement). To "grant" such a patent license to a
+party means to make such an agreement or commitment not to enforce a
+patent against the party.
+
+ If you convey a covered work, knowingly relying on a patent license,
+and the Corresponding Source of the work is not available for anyone
+to copy, free of charge and under the terms of this License, through a
+publicly available network server or other readily accessible means,
+then you must either (1) cause the Corresponding Source to be so
+available, or (2) arrange to deprive yourself of the benefit of the
+patent license for this particular work, or (3) arrange, in a manner
+consistent with the requirements of this License, to extend the patent
+license to downstream recipients. "Knowingly relying" means you have
+actual knowledge that, but for the patent license, your conveying the
+covered work in a country, or your recipient's use of the covered work
+in a country, would infringe one or more identifiable patents in that
+country that you have reason to believe are valid.
+
+ If, pursuant to or in connection with a single transaction or
+arrangement, you convey, or propagate by procuring conveyance of, a
+covered work, and grant a patent license to some of the parties
+receiving the covered work authorizing them to use, propagate, modify
+or convey a specific copy of the covered work, then the patent license
+you grant is automatically extended to all recipients of the covered
+work and works based on it.
+
+ A patent license is "discriminatory" if it does not include within
+the scope of its coverage, prohibits the exercise of, or is
+conditioned on the non-exercise of one or more of the rights that are
+specifically granted under this License. You may not convey a covered
+work if you are a party to an arrangement with a third party that is
+in the business of distributing software, under which you make payment
+to the third party based on the extent of your activity of conveying
+the work, and under which the third party grants, to any of the
+parties who would receive the covered work from you, a discriminatory
+patent license (a) in connection with copies of the covered work
+conveyed by you (or copies made from those copies), or (b) primarily
+for and in connection with specific products or compilations that
+contain the covered work, unless you entered into that arrangement,
+or that patent license was granted, prior to 28 March 2007.
+
+ Nothing in this License shall be construed as excluding or limiting
+any implied license or other defenses to infringement that may
+otherwise be available to you under applicable patent law.
+
+ 12. No Surrender of Others' Freedom.
+
+ If conditions are imposed on you (whether by court order, agreement or
+otherwise) that contradict the conditions of this License, they do not
+excuse you from the conditions of this License. If you cannot convey a
+covered work so as to satisfy simultaneously your obligations under this
+License and any other pertinent obligations, then as a consequence you may
+not convey it at all. For example, if you agree to terms that obligate you
+to collect a royalty for further conveying from those to whom you convey
+the Program, the only way you could satisfy both those terms and this
+License would be to refrain entirely from conveying the Program.
+
+ 13. Use with the GNU Affero General Public License.
+
+ Notwithstanding any other provision of this License, you have
+permission to link or combine any covered work with a work licensed
+under version 3 of the GNU Affero General Public License into a single
+combined work, and to convey the resulting work. The terms of this
+License will continue to apply to the part which is the covered work,
+but the special requirements of the GNU Affero General Public License,
+section 13, concerning interaction through a network will apply to the
+combination as such.
+
+ 14. Revised Versions of this License.
+
+ The Free Software Foundation may publish revised and/or new versions of
+the GNU General Public License from time to time. Such new versions will
+be similar in spirit to the present version, but may differ in detail to
+address new problems or concerns.
+
+ Each version is given a distinguishing version number. If the
+Program specifies that a certain numbered version of the GNU General
+Public License "or any later version" applies to it, you have the
+option of following the terms and conditions either of that numbered
+version or of any later version published by the Free Software
+Foundation. If the Program does not specify a version number of the
+GNU General Public License, you may choose any version ever published
+by the Free Software Foundation.
+
+ If the Program specifies that a proxy can decide which future
+versions of the GNU General Public License can be used, that proxy's
+public statement of acceptance of a version permanently authorizes you
+to choose that version for the Program.
+
+ Later license versions may give you additional or different
+permissions. However, no additional obligations are imposed on any
+author or copyright holder as a result of your choosing to follow a
+later version.
+
+ 15. Disclaimer of Warranty.
+
+ THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY
+APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT
+HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY
+OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO,
+THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
+PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM
+IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF
+ALL NECESSARY SERVICING, REPAIR OR CORRECTION.
+
+ 16. Limitation of Liability.
+
+ IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
+WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS
+THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY
+GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE
+USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF
+DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD
+PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS),
+EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF
+SUCH DAMAGES.
+
+ 17. Interpretation of Sections 15 and 16.
+
+ If the disclaimer of warranty and limitation of liability provided
+above cannot be given local legal effect according to their terms,
+reviewing courts shall apply local law that most closely approximates
+an absolute waiver of all civil liability in connection with the
+Program, unless a warranty or assumption of liability accompanies a
+copy of the Program in return for a fee.
+
+ END OF TERMS AND CONDITIONS
+
+ How to Apply These Terms to Your New Programs
+
+ If you develop a new program, and you want it to be of the greatest
+possible use to the public, the best way to achieve this is to make it
+free software which everyone can redistribute and change under these terms.
+
+ To do so, attach the following notices to the program. It is safest
+to attach them to the start of each source file to most effectively
+state the exclusion of warranty; and each file should have at least
+the "copyright" line and a pointer to where the full notice is found.
+
+
+ Copyright (C)
+
+ This program is free software: you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation, either version 3 of the License, or
+ (at your option) any later version.
+
+ This program is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with this program. If not, see .
+
+Also add information on how to contact you by electronic and paper mail.
+
+ If the program does terminal interaction, make it output a short
+notice like this when it starts in an interactive mode:
+
+ Copyright (C)
+ This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
+ This is free software, and you are welcome to redistribute it
+ under certain conditions; type `show c' for details.
+
+The hypothetical commands `show w' and `show c' should show the appropriate
+parts of the General Public License. Of course, your program's commands
+might be different; for a GUI interface, you would use an "about box".
+
+ You should also get your employer (if you work as a programmer) or school,
+if any, to sign a "copyright disclaimer" for the program, if necessary.
+For more information on this, and how to apply and follow the GNU GPL, see
+.
+
+ The GNU General Public License does not permit incorporating your program
+into proprietary programs. If your program is a subroutine library, you
+may consider it more useful to permit linking proprietary applications with
+the library. If this is what you want to do, use the GNU Lesser General
+Public License instead of this License. But first, please read
+.
+
diff --git a/arc_gui.py b/arc_gui.py
index 6e8f913..fdf01fa 100644
--- a/arc_gui.py
+++ b/arc_gui.py
@@ -5,8 +5,8 @@
import shutil
from pathlib import Path
-from PySide6.QtCore import QThread, Signal, Qt, QTimer, QUrl, QObject
-from PySide6.QtWidgets import (QApplication, QMainWindow, QVBoxLayout, QHBoxLayout,
+from PySide6.QtCore import QThread, Signal, Qt, QTimer, QUrl, QObject, QSize, QSettings
+from PySide6.QtWidgets import (QApplication, QMainWindow, QVBoxLayout, QHBoxLayout, QGridLayout,
QPushButton, QLabel, QLineEdit, QTextEdit, QProgressBar,
QTabWidget, QWidget, QGroupBox, QListWidget, QListWidgetItem,
QFileDialog, QCheckBox, QComboBox, QFrame, QMessageBox, QMenu)
@@ -15,10 +15,12 @@
from con import CON
from support.toggle import ThemeManager
+from support.GUI.arc_support import BatchDropZoneWidget
# Add the current directory to Python path to import convertzip module
sys.path.append(os.path.dirname(os.path.abspath(__file__)))
-from support.archive_manager import create_archive, extract_archive, add_to_archive, list_archive_contents, SUPPORTED_ARCHIVE_FORMATS
-from support.password_detector import PasswordDetector, detect_password_protection
+from support.archive_manager import create_archive, extract_archive, add_to_archive, list_archive_contents, SUPPORTED_ARCHIVE_FORMATS, batch_extract_archives
+from support.password_detector import PasswordDetector
+from password_dialog import PasswordDialog, SimplePasswordDialog
# Remove the problematic reconfigure calls
# sys.stdout.reconfigure(encoding='utf-8')
@@ -28,6 +30,7 @@ class CreateZipWorker(QObject):
finished = Signal()
progress_updated = Signal(str, int)
conversion_error = Signal(str)
+ canceled = Signal() # Add canceled signal
def __init__(self, output_path, sources, archive_format, password=None):
super().__init__()
@@ -35,6 +38,12 @@ def __init__(self, output_path, sources, archive_format, password=None):
self.sources = sources
self.archive_format = archive_format
self.password = password
+ self.is_stopped = False # Add stop flag
+
+ def stop(self):
+ """Stop the archive creation process"""
+ self.is_stopped = True
+ self.progress_updated.emit("Canceling archive creation...", 0)
def run(self):
try:
@@ -51,8 +60,22 @@ def run(self):
if not os.path.exists(source):
raise ValueError(f"Source file does not exist: {source}")
- create_archive(self.output_path, self.sources, self.archive_format, self._update_progress_callback, self.password)
- self.finished.emit()
+ # Update progress callback to check for cancellation
+ def progress_callback(message, percentage):
+ if self.is_stopped:
+ raise RuntimeError("Archive creation canceled by user")
+ self.progress_updated.emit(message, percentage)
+
+ create_archive(self.output_path, self.sources, self.archive_format, progress_callback, self.password)
+ if not self.is_stopped:
+ self.finished.emit()
+ else:
+ self.canceled.emit()
+ except RuntimeError as e:
+ if "canceled" in str(e).lower():
+ self.canceled.emit()
+ else:
+ self.conversion_error.emit(str(e))
except ValueError as e:
# Handle value errors
self.conversion_error.emit(f"Input error: {str(e)}")
@@ -70,9 +93,12 @@ def run(self):
self.conversion_error.emit(str(e))
except Exception as e:
# Handle all other exceptions
- import traceback
- error_msg = f"Unexpected error: {str(e)}\n{traceback.format_exc()}"
- self.conversion_error.emit(error_msg)
+ if self.is_stopped:
+ self.canceled.emit()
+ else:
+ import traceback
+ error_msg = f"Unexpected error: {str(e)}\n{traceback.format_exc()}"
+ self.conversion_error.emit(error_msg)
def _update_progress_callback(self, message, percentage):
self.progress_updated.emit(message, percentage)
@@ -82,25 +108,47 @@ class ExtractZipWorker(QObject):
progress_updated = Signal(str, int)
conversion_error = Signal(str)
password_required = Signal(str) # Emits error message when password is required
+ canceled = Signal() # Add canceled signal
def __init__(self, zip_path, dest_path, password=None):
super().__init__()
self.archive_path = zip_path # Renamed for clarity with generic archive_manager
self.extract_to = dest_path
self.password = password
+ self.is_stopped = False # Add stop flag
+
+ def stop(self):
+ """Stop the archive extraction process"""
+ self.is_stopped = True
+ self.progress_updated.emit("Canceling archive extraction...", 0)
def run(self):
try:
- extract_archive(self.archive_path, self.extract_to, self._update_progress_callback, self.password)
- self.finished.emit()
+ # Update progress callback to check for cancellation
+ def progress_callback(message, percentage):
+ if self.is_stopped:
+ raise RuntimeError("Archive extraction canceled by user")
+ self.progress_updated.emit(message, percentage)
+
+ extract_archive(self.archive_path, self.extract_to, progress_callback, self.password)
+ if not self.is_stopped:
+ self.finished.emit()
+ else:
+ self.canceled.emit()
except RuntimeError as e:
- # Handle password required case
- if "password" in str(e).lower() or "encrypted" in str(e).lower():
- self.password_required.emit(str(e))
+ if "canceled" in str(e).lower():
+ self.canceled.emit()
else:
- self.conversion_error.emit(str(e))
+ # Handle password required case
+ if "password" in str(e).lower() or "encrypted" in str(e).lower():
+ self.password_required.emit(str(e))
+ else:
+ self.conversion_error.emit(str(e))
except Exception as e:
- self.conversion_error.emit(str(e))
+ if self.is_stopped:
+ self.canceled.emit()
+ else:
+ self.conversion_error.emit(str(e))
def _update_progress_callback(self, message, percentage):
self.progress_updated.emit(message, percentage)
@@ -109,26 +157,48 @@ class AddToZipWorker(QObject):
finished = Signal()
progress_updated = Signal(str, int)
conversion_error = Signal(str)
+ canceled = Signal() # Add canceled signal
def __init__(self, zip_path, file_paths):
super().__init__()
self.archive_path = zip_path # Renamed for clarity with generic archive_manager
self.files_to_add = file_paths if isinstance(file_paths, list) else [file_paths]
+ self.is_stopped = False # Add stop flag
+
+ def stop(self):
+ """Stop the add to archive process"""
+ self.is_stopped = True
+ self.progress_updated.emit("Canceling add to archive...", 0)
def run(self):
try:
# Handle multiple files
total_files = len(self.files_to_add)
for i, file_path in enumerate(self.files_to_add):
+ # Check if canceled before processing each file
+ if self.is_stopped:
+ raise RuntimeError("Add to archive canceled by user")
+
self._update_progress_callback(f"Adding file {i+1}/{total_files}: {os.path.basename(file_path)}", (i/total_files)*100)
add_to_archive(self.archive_path, file_path, None) # No individual progress for each file
- self._update_progress_callback(f"Added {total_files} files to archive", 100)
- self.finished.emit()
+ if not self.is_stopped:
+ self._update_progress_callback(f"Added {total_files} files to archive", 100)
+ self.finished.emit()
+ else:
+ self.canceled.emit()
+ except RuntimeError as e:
+ if "canceled" in str(e).lower():
+ self.canceled.emit()
+ else:
+ self.conversion_error.emit(str(e))
except NotImplementedError as e:
self.conversion_error.emit(str(e))
except Exception as e:
- self.conversion_error.emit(str(e))
+ if self.is_stopped:
+ self.canceled.emit()
+ else:
+ self.conversion_error.emit(str(e))
def _update_progress_callback(self, message, percentage):
self.progress_updated.emit(message, percentage)
@@ -137,32 +207,227 @@ class ListZipContentsWorker(QObject):
finished = Signal(list) # Emits list of contents
conversion_error = Signal(str)
password_required = Signal(str) # Emits error message when password is required
+ canceled = Signal() # Add canceled signal
def __init__(self, zip_path, password=None):
super().__init__()
self.archive_path = zip_path # Renamed for clarity with generic archive_manager
self.password = password
self.result = None # Add result attribute to store results
+ self.is_stopped = False # Add stop flag
+
+ def stop(self):
+ """Stop the list contents process"""
+ self.is_stopped = True
+ print(f"[DEBUG] ListZipContentsWorker: Stop requested")
def run(self):
try:
+ # Check if canceled before starting
+ if self.is_stopped:
+ self.canceled.emit()
+ return
+
print(f"[DEBUG] ListZipContentsWorker: Starting to list contents of {self.archive_path}")
+ # This operation is typically fast, but we'll still add a cancel check
contents = list_archive_contents(self.archive_path, password=self.password)
print(f"[DEBUG] ListZipContentsWorker: Got {len(contents) if contents else 0} items")
- self.result = contents # 设置result属性
- self.finished.emit(contents)
+
+ if not self.is_stopped:
+ self.result = contents # 设置result属性
+ self.finished.emit(contents)
+ else:
+ self.canceled.emit()
except RuntimeError as e:
- # Handle password required case
- print(f"[DEBUG] ListZipContentsWorker: RuntimeError - {str(e)}")
- if "password" in str(e).lower() or "encrypted" in str(e).lower():
- self.password_required.emit(str(e))
+ if not self.is_stopped:
+ # Handle password required case
+ print(f"[DEBUG] ListZipContentsWorker: RuntimeError - {str(e)}")
+ if "password" in str(e).lower() or "encrypted" in str(e).lower():
+ self.password_required.emit(str(e))
+ else:
+ self.conversion_error.emit(str(e))
else:
+ self.canceled.emit()
+ except Exception as e:
+ if not self.is_stopped:
+ print(f"[DEBUG] ListZipContentsWorker: Exception - {str(e)}")
+ import traceback
+ traceback.print_exc()
self.conversion_error.emit(str(e))
+ else:
+ self.canceled.emit()
+
+class BatchExtractWorker(QObject):
+ """Worker for batch archive extraction"""
+ finished = Signal(int, int, list, list) # Emits success_count, failed_count, success_files, failed_files
+ progress_updated = Signal(int, int, str, int, int) # processed_count, total_count, current_file, success_count, failed_count
+ conversion_error = Signal(str) # Emits error messages
+ individual_progress = Signal(str, str, int) # Emits archive name, message, percentage
+ status_updated = Signal(str) # Emits status messages
+ canceled = Signal() # Add canceled signal for consistency
+
+ def __init__(self, archive_paths, dest_folder, create_subfolders=True, overwrite_files=False, parent_gui=None):
+ super().__init__()
+ self.archive_paths = archive_paths
+ self.dest_folder = dest_folder
+ self.create_subfolders = create_subfolders
+ self.overwrite_files = overwrite_files
+ self.is_stopped = False
+ self.parent_gui = parent_gui # Reference to main GUI for password dialogs
+
+ # Track detailed statistics
+ self.success_count = 0
+ self.failed_count = 0
+ self.success_files = []
+ self.failed_files = []
+
+ def stop(self):
+ """Stop the batch extraction process"""
+ self.is_stopped = True
+ self.status_updated.emit("Stopping batch extraction...")
+
+ def run(self):
+ """Execute batch extraction"""
+ try:
+ if not self.archive_paths:
+ raise ValueError("No archive files to extract")
+
+ if not self.dest_folder:
+ raise ValueError("Destination folder is not specified")
+
+ # Ensure destination folder exists
+ if not os.path.exists(self.dest_folder):
+ try:
+ os.makedirs(self.dest_folder, exist_ok=True)
+ except Exception as e:
+ raise ValueError(f"Failed to create destination folder: {str(e)}")
+
+ # Initialize password detector
+ password_detector = PasswordDetector()
+
+ # Reset statistics
+ self.success_count = 0
+ self.failed_count = 0
+ self.success_files = []
+ self.failed_files = []
+
+ total_files = len(self.archive_paths)
+ self.status_updated.emit(f"Starting batch extraction of {total_files} archive(s)...")
+
+ def progress_callback(current, total, current_file=""):
+ if self.is_stopped:
+ return # Stop processing if requested
+
+ # Handle different call patterns
+ if isinstance(current, str):
+ # Called with (message, progress_percent) pattern for individual file extraction
+ message = current
+ progress_percent = total
+ archive_name = os.path.basename(current_file) if current_file else ""
+ # Emit individual file progress
+ self.individual_progress.emit(archive_name, message, int(progress_percent))
+ else:
+ # Called with (current, total, current_file) pattern for batch progress
+ current_val = int(current) if isinstance(current, (int, float)) else 0
+ total_val = int(total) if isinstance(total, (int, float)) else total_files
+
+ # Get current file path
+ if isinstance(current, int) and 1 <= current <= total_files:
+ current_file_path = self.archive_paths[current - 1]
+ else:
+ current_file_path = str(current_file) if current_file else ""
+
+ # Calculate overall progress percentage
+ overall_progress = (current_val / total_val * 100) if total_val > 0 else 0
+
+ # Emit batch progress update
+ self.progress_updated.emit(current_val, total_val, current_file_path, self.success_count, self.failed_count)
+
+ def password_callback(archive_path, format_name, is_protected):
+ """Callback to request password from user via GUI"""
+ if self.is_stopped:
+ return None
+ if self.parent_gui and hasattr(self.parent_gui, 'request_password'):
+ try:
+ # Request password from main GUI thread
+ return self.parent_gui.request_password(archive_path, format_name, is_protected)
+ except Exception as e:
+ self.conversion_error.emit(f"Error requesting password: {str(e)}")
+ return None
+ return None
+
+ def error_callback(archive_path, error_message):
+ """Callback for individual archive errors"""
+ self.failed_count += 1 # Increment failed count
+ self.failed_files.append((archive_path, error_message))
+ self.conversion_error.emit(f"Error processing {os.path.basename(archive_path)}: {error_message}")
+ # Update progress after error
+ processed = self.success_count + self.failed_count
+ self.progress_updated.emit(processed, total_files, archive_path, self.success_count, self.failed_count)
+
+ def success_callback(archive_path):
+ """Callback for successful archive extraction"""
+ self.success_count += 1 # Increment success count
+ self.success_files.append(archive_path)
+ # Update progress after success
+ processed = self.success_count + self.failed_count
+ self.progress_updated.emit(processed, total_files, archive_path, self.success_count, self.failed_count)
+
+ # Prepare options for batch extraction
+ options = {
+ 'create_subfolders': self.create_subfolders,
+ 'overwrite_existing': self.overwrite_files,
+ 'progress_callback': progress_callback if not self.is_stopped else None,
+ 'password_callback': password_callback if not self.is_stopped else None,
+ 'password_detector': password_detector,
+ 'error_callback': error_callback,
+ 'success_callback': success_callback
+ }
+
+ if self.is_stopped:
+ self.status_updated.emit("Batch extraction stopped by user")
+ self.finished.emit(self.success_count, self.failed_count, self.success_files, self.failed_files)
+ return
+
+ # Call batch extraction function with password detection
+ result = batch_extract_archives(
+ self.archive_paths,
+ self.dest_folder,
+ **options
+ )
+
+ if not self.is_stopped:
+ # Update final statistics from result
+ self.success_count = result.get('success_count', self.success_count)
+ self.failed_count = result.get('error_count', self.failed_count)
+
+ # Emit final status
+ self.status_updated.emit(f"Batch extraction completed: {self.success_count} successful, {self.failed_count} failed")
+
+ # Emit finished signal with detailed results
+ self.finished.emit(self.success_count, self.failed_count, self.success_files, self.failed_files)
+ else:
+ self.status_updated.emit("Batch extraction stopped by user")
+ # Emit canceled signal instead of finished for consistency with other workers
+ self.canceled.emit()
+ self.finished.emit(self.success_count, self.failed_count, self.success_files, self.failed_files)
+
+ except ValueError as e:
+ error_msg = f"Input error: {str(e)}"
+ self.conversion_error.emit(error_msg)
+ self.status_updated.emit(f"Batch extraction failed: {error_msg}")
+ self.finished.emit(0, total_files if 'total_files' in locals() else 0, [], self.archive_paths)
+ except RuntimeError as e:
+ error_msg = str(e)
+ self.conversion_error.emit(error_msg)
+ self.status_updated.emit(f"Batch extraction failed: {error_msg}")
+ self.finished.emit(0, total_files if 'total_files' in locals() else 0, [], self.archive_paths)
except Exception as e:
- print(f"[DEBUG] ListZipContentsWorker: Exception - {str(e)}")
import traceback
- traceback.print_exc()
- self.conversion_error.emit(str(e))
+ error_msg = f"Unexpected error during batch extraction: {str(e)}\n{traceback.format_exc()}"
+ self.conversion_error.emit(error_msg)
+ self.status_updated.emit(f"Batch extraction failed unexpectedly")
+ self.finished.emit(0, total_files if 'total_files' in locals() else 0, [], self.archive_paths)
class ZipGUI(QMainWindow):
@@ -220,8 +485,8 @@ def DARK_QSS(self):
def __init__(self, initial_dark_mode=False):
super().__init__()
self.setWindowTitle("Archive File Processing Tool")
- self.setGeometry(200, 200, 800, 600)
- self.setMinimumSize(600, 780)
+ self.setGeometry(200, 200, 1200, 900)
+ self.setMinimumSize(1200, 900)
# Enable drag and drop for the main window
self.setAcceptDrops(True)
@@ -235,6 +500,9 @@ def __init__(self, initial_dark_mode=False):
setTheme(Theme.AUTO)
self.themeListener.start()
qconfig.themeChanged.connect(self._onThemeChanged)
+
+ # Load settings
+ self.load_settings()
def closeEvent(self, event):
"""Window close event"""
# Stop listener thread
@@ -245,6 +513,12 @@ def closeEvent(self, event):
def _onThemeChanged(self, theme: Theme):
"""Theme change handling"""
# Update interface to respond to theme changes
+ is_dark_mode = theme == Theme.DARK
+
+ # Update drag and drop area theme
+ if hasattr(self, 'batch_drop_area'):
+ self.batch_drop_area.set_theme(is_dark_mode)
+
self.update()
setTheme(Theme.AUTO)
def init_variables(self):
@@ -255,12 +529,25 @@ def init_variables(self):
self.create_zip_worker_thread = None # Renamed to generic for clarity
self.create_zip_worker = None # Renamed to generic for clarity
+ # Task mode setting
+ self.task_mode = False
+
# Variables for Extract ZIP tab
self.extract_zip_path = ""
self.extract_dest_path = ""
self.extract_zip_worker_thread = None # Renamed to generic for clarity
self.extract_zip_worker = None # Renamed to generic for clarity
+ # Variables for Batch Extract tab
+ self.batch_extract_files = []
+ self.batch_extract_dest_path = ""
+ self.batch_extract_worker = None
+ self.batch_extract_worker_thread = None
+ self.batch_extract_running = False
+ self.batch_extract_success_count = 0
+ self.batch_extract_failed_count = 0
+ self.batch_extract_password = None
+
# Variables for Add to ZIP tab
self.add_zip_path = ""
self.add_file_path = ""
@@ -275,6 +562,35 @@ def init_variables(self):
# Password protection status for archive contents
self.is_password_protected = False
self._current_password = None
+ self._password_dialog = None
+
+ def request_password(self, archive_path, format_name, is_protected):
+ """Request password from user for a protected archive"""
+ archive_name = os.path.basename(archive_path)
+ title = "Password Required"
+
+ if is_protected:
+ content = f"The archive '{archive_name}' ({format_name.upper()}) is password protected.\nPlease enter the password:"
+ else:
+ content = f"Enter password for archive '{archive_name}' ({format_name.upper()}):"
+
+ # Create and show password dialog
+ self._password_dialog = PasswordDialog(
+ parent=self,
+ title=title,
+ content=content,
+ error_message=""
+ )
+
+ # Show dialog and get result
+ if self._password_dialog.exec() == PasswordDialog.DialogCode.Accepted:
+ password = self._password_dialog.get_password()
+ self._password_dialog = None
+ return password
+ else:
+ # User canceled
+ self._password_dialog = None
+ return None
def setup_ui(self):
self.main_widget = QWidget(self)
@@ -307,9 +623,34 @@ def _apply_theme(self, is_dark_mode):
self.setStyleSheet(self.DARK_QSS)
else:
self.setStyleSheet(self.LIGHT_QSS)
+
+ # Update drag and drop area theme
+ if hasattr(self, 'batch_drop_area'):
+ self.batch_drop_area.set_theme(is_dark_mode)
def _apply_system_theme(self, is_dark_mode):
self._apply_theme(is_dark_mode)
+
+ def load_settings(self):
+ """Load settings from QSettings"""
+ settings = QSettings("MyCompany", "ConverterApp")
+
+ # Load task mode setting
+ self.task_mode = settings.value("task_mode", False, type=bool)
+ if hasattr(self, 'task_mode_check'):
+ self.task_mode_check.setChecked(self.task_mode)
+
+ def save_settings(self):
+ """Save settings to QSettings"""
+ settings = QSettings("MyCompany", "ConverterApp")
+
+ # Save task mode setting
+ if hasattr(self, 'task_mode_check'):
+ settings.setValue("task_mode", self.task_mode_check.isChecked())
+ else:
+ settings.setValue("task_mode", self.task_mode)
+
+ settings.sync()
def center_window(self):
qr = self.frameGeometry()
@@ -334,17 +675,36 @@ def create_create_tab(self):
tab_sizer = QVBoxLayout(tab_panel)
self.notebook.addTab(tab_panel, "Create Archive") # Changed tab title
- # Output file selection
+ # Output file selection with ScrollArea inside GroupBox
output_box = QGroupBox("Output Archive File") # Changed group box title
- output_box_sizer = QHBoxLayout(output_box)
+ output_box.setMinimumHeight(200) # Set minimum height
+ output_box_sizer = QVBoxLayout(output_box)
+
+ # Create ScrollArea inside GroupBox
+ output_scroll_area = ScrollArea()
+ output_scroll_area.setWidgetResizable(True)
+ output_scroll_area.setVerticalScrollBarPolicy(Qt.ScrollBarPolicy.ScrollBarAsNeeded)
+ output_scroll_area.setHorizontalScrollBarPolicy(Qt.ScrollBarPolicy.ScrollBarAsNeeded)
+
+ # Create content widget for ScrollArea
+ content_widget = QWidget()
+ content_layout = QHBoxLayout(content_widget)
self.create_output_text = LineEdit()
setCustomStyleSheet(self.create_output_text, CON.qss_line, CON.qss_line)
# self.create_output_text.setReadOnly(True) # Allow users to manually input path
- output_box_sizer.addWidget(self.create_output_text, 1)
+ content_layout.addWidget(self.create_output_text, 1)
output_button = PushButton("Browse...")
output_button.clicked.connect(self.browse_create_output)
- output_box_sizer.addWidget(output_button)
+ content_layout.addWidget(output_button)
+
+ # Set content widget to ScrollArea
+ output_scroll_area.setWidget(content_widget)
+
+ # Add ScrollArea to GroupBox layout
+ output_box_sizer.addWidget(output_scroll_area)
+
+ # Add GroupBox to main layout
tab_sizer.addWidget(output_box)
# Archive Format Selection (new)
@@ -364,13 +724,24 @@ def create_create_tab(self):
format_layout.addWidget(self.create_format_combo, 1)
tab_sizer.addLayout(format_layout)
- # Source files list
+ # Source files list with ScrollArea inside GroupBox
sources_box = QGroupBox("Source Files/Directories")
+ sources_box.setMinimumHeight(200) # Set minimum height
sources_box_sizer = QVBoxLayout(sources_box)
+ # Create ScrollArea inside GroupBox
+ sources_scroll_area = ScrollArea()
+ sources_scroll_area.setWidgetResizable(True)
+ sources_scroll_area.setVerticalScrollBarPolicy(Qt.ScrollBarPolicy.ScrollBarAsNeeded)
+ sources_scroll_area.setHorizontalScrollBarPolicy(Qt.ScrollBarPolicy.ScrollBarAsNeeded)
+
+ # Create content widget for ScrollArea
+ content_widget = QWidget()
+ content_layout = QVBoxLayout(content_widget)
+
self.sources_listbox = ListWidget()
self.sources_listbox.setMinimumHeight(280) # Set minimum height
- sources_box_sizer.addWidget(self.sources_listbox, 1) # Increase stretch weight
+ content_layout.addWidget(self.sources_listbox, 1) # Increase stretch weight
# Set right-click to immediately select
self.sources_listbox.setContextMenuPolicy(Qt.ContextMenuPolicy.CustomContextMenu)
# Context menu functionality removed
@@ -391,9 +762,22 @@ def create_create_tab(self):
button_sizer.addWidget(remove_button)
button_sizer.addStretch(1) # Push buttons to left
- sources_box_sizer.addLayout(button_sizer)
+ content_layout.addLayout(button_sizer)
+
+ # Set content widget to ScrollArea
+ sources_scroll_area.setWidget(content_widget)
+
+ # Add ScrollArea to GroupBox layout
+ sources_box_sizer.addWidget(sources_scroll_area)
+
+ # Add GroupBox to main layout
tab_sizer.addWidget(sources_box, 1) # Give sources box more stretch
+ # Task mode control
+ self.task_mode_check = CheckBox("Enable Task Mode")
+ self.task_mode_check.setChecked(False)
+ tab_sizer.addWidget(self.task_mode_check)
+
# Progress bar
self.create_progress_label = QLabel("")
tab_sizer.addWidget(self.create_progress_label)
@@ -403,11 +787,28 @@ def create_create_tab(self):
self.create_progress.setValue(0)
tab_sizer.addWidget(self.create_progress)
+ # Create button layout with cancel button
+ button_layout = QHBoxLayout()
+ button_layout.setSpacing(10)
+
# Create button
self.create_button = PrimaryPushButton("Create Archive") # Changed button text
-
self.create_button.clicked.connect(self.start_create_archive) # Changed signal
- tab_sizer.addWidget(self.create_button, 0, Qt.AlignmentFlag.AlignCenter)
+ button_layout.addWidget(self.create_button)
+
+ # Cancel button
+ self.create_cancel_button = PushButton("Cancel")
+ self.create_cancel_button.clicked.connect(self.cancel_create_archive)
+ self.create_cancel_button.setEnabled(False) # Initially disabled
+ button_layout.addWidget(self.create_cancel_button)
+
+ # Create a container widget to center the button layout
+ button_container = QWidget()
+ button_container_layout = QHBoxLayout(button_container)
+ button_container_layout.addStretch()
+ button_container_layout.addLayout(button_layout)
+ button_container_layout.addStretch()
+ tab_sizer.addWidget(button_container)
tab_sizer.addStretch(1) # Push content to top
@@ -416,33 +817,86 @@ def create_extract_tab(self):
tab_sizer = QVBoxLayout(tab_panel)
self.notebook.addTab(tab_panel, "Extract Archive") # Changed tab title
- # Archive file selection (changed title)
+ # Tab selector for single/batch extract
+ self.extract_tab_widget = QTabWidget()
+ tab_sizer.addWidget(self.extract_tab_widget)
+
+ # Single Extract Tab
+ self.create_single_extract_tab()
+
+ # Batch Extract Tab
+ self.create_batch_extract_tab()
+
+ tab_sizer.addStretch(1) # Push content to top
+
+ def create_single_extract_tab(self):
+ """Create single archive extraction tab"""
+ single_panel = QWidget()
+ single_sizer = QVBoxLayout(single_panel)
+
+ # Archive file selection with ScrollArea inside GroupBox (changed title)
zip_box = QGroupBox("Archive File to Extract")
- zip_box_sizer = QHBoxLayout(zip_box)
+ zip_box.setMinimumHeight(200) # Set minimum height
+ zip_box_sizer = QVBoxLayout(zip_box)
+ # Create ScrollArea inside GroupBox
+ zip_scroll_area = ScrollArea()
+ zip_scroll_area.setWidgetResizable(True)
+ zip_scroll_area.setVerticalScrollBarPolicy(Qt.ScrollBarPolicy.ScrollBarAsNeeded)
+ zip_scroll_area.setHorizontalScrollBarPolicy(Qt.ScrollBarPolicy.ScrollBarAsNeeded)
+
+ # Create content widget for ScrollArea
+ content_widget = QWidget()
+ content_layout = QHBoxLayout(content_widget)
+
self.extract_zip_text = LineEdit()
setCustomStyleSheet(self.extract_zip_text, CON.qss_line, CON.qss_line)
# self.extract_zip_text.setReadOnly(True) # Allow users to manually input path
- zip_box_sizer.addWidget(self.extract_zip_text, 1)
+ content_layout.addWidget(self.extract_zip_text, 1)
zip_button = PushButton("Browse...")
-
zip_button.clicked.connect(self.browse_extract_archive) # Changed signal
- zip_box_sizer.addWidget(zip_button)
- tab_sizer.addWidget(zip_box)
+ content_layout.addWidget(zip_button)
+
+ # Set content widget to ScrollArea
+ zip_scroll_area.setWidget(content_widget)
+
+ # Add ScrollArea to GroupBox layout
+ zip_box_sizer.addWidget(zip_scroll_area)
+
+ # Add GroupBox to main layout
+ single_sizer.addWidget(zip_box)
- # Destination folder selection
+ # Destination folder selection with ScrollArea inside GroupBox
dest_box = QGroupBox("Destination Folder")
- dest_box_sizer = QHBoxLayout(dest_box)
+ dest_box.setMinimumHeight(200) # Set minimum height
+ dest_box_sizer = QVBoxLayout(dest_box)
+ # Create ScrollArea inside GroupBox
+ dest_scroll_area = ScrollArea()
+ dest_scroll_area.setWidgetResizable(True)
+ dest_scroll_area.setVerticalScrollBarPolicy(Qt.ScrollBarPolicy.ScrollBarAsNeeded)
+ dest_scroll_area.setHorizontalScrollBarPolicy(Qt.ScrollBarPolicy.ScrollBarAsNeeded)
+
+ # Create content widget for ScrollArea
+ content_widget = QWidget()
+ content_layout = QHBoxLayout(content_widget)
+
self.extract_dest_text = LineEdit()
setCustomStyleSheet(self.extract_dest_text, CON.qss_line, CON.qss_line)
# self.extract_dest_text.setReadOnly(True) # Allow users to manually input path
- dest_box_sizer.addWidget(self.extract_dest_text, 1)
+ content_layout.addWidget(self.extract_dest_text, 1)
dest_button = PushButton("Browse...")
-
dest_button.clicked.connect(self.browse_extract_dest)
- dest_box_sizer.addWidget(dest_button)
- tab_sizer.addWidget(dest_box)
+ content_layout.addWidget(dest_button)
+
+ # Set content widget to ScrollArea
+ dest_scroll_area.setWidget(content_widget)
+
+ # Add ScrollArea to GroupBox layout
+ dest_box_sizer.addWidget(dest_scroll_area)
+
+ # Add GroupBox to main layout
+ single_sizer.addWidget(dest_box)
# Password status indicator
password_status_box = QHBoxLayout()
@@ -452,58 +906,358 @@ def create_extract_tab(self):
password_status_box.addWidget(self.extract_password_status_label)
password_status_box.addWidget(self.extract_password_status_icon)
password_status_box.addStretch()
- tab_sizer.addLayout(password_status_box)
+ single_sizer.addLayout(password_status_box)
# Progress bar
self.extract_progress_label = QLabel("")
- tab_sizer.addWidget(self.extract_progress_label)
+ single_sizer.addWidget(self.extract_progress_label)
self.extract_progress = ProgressBar()
self.extract_progress.setRange(0, 100)
self.extract_progress.setValue(0)
- tab_sizer.addWidget(self.extract_progress)
+ single_sizer.addWidget(self.extract_progress)
+ # Extract button layout with cancel button
+ button_layout = QHBoxLayout()
+ button_layout.setSpacing(10)
+
# Extract button
self.extract_button = PrimaryPushButton("Extract Archive") # Changed button text
-
self.extract_button.clicked.connect(self.start_extract_archive) # Changed signal
- tab_sizer.addWidget(self.extract_button, 0, Qt.AlignmentFlag.AlignCenter)
+ button_layout.addWidget(self.extract_button)
+
+ # Cancel button
+ self.extract_cancel_button = PushButton("Cancel")
+ self.extract_cancel_button.clicked.connect(self.cancel_extract_archive)
+ self.extract_cancel_button.setEnabled(False) # Initially disabled
+ button_layout.addWidget(self.extract_cancel_button)
+
+ # Create a container widget to center the button layout
+ button_container = QWidget()
+ button_container_layout = QHBoxLayout(button_container)
+ button_container_layout.addStretch()
+ button_container_layout.addLayout(button_layout)
+ button_container_layout.addStretch()
+ single_sizer.addWidget(button_container)
+
+ self.extract_tab_widget.addTab(single_panel, "Single Extract")
+
+ def create_batch_extract_tab(self):
+ """Create batch archive extraction tab"""
+ batch_panel = QWidget()
+ main_layout = QHBoxLayout(batch_panel)
+
+ # Left side - File management and selection
+ left_panel = QWidget()
+ left_layout = QVBoxLayout(left_panel)
+
+ # File selection group with ScrollArea inside GroupBox
+ file_group = QGroupBox("Batch Archive Files")
+ file_group.setMinimumHeight(200) # Set minimum height
+ file_group_layout = QVBoxLayout(file_group)
+
+ # Create ScrollArea inside GroupBox
+ file_scroll_area = ScrollArea()
+ file_scroll_area.setWidgetResizable(True)
+ file_scroll_area.setVerticalScrollBarPolicy(Qt.ScrollBarPolicy.ScrollBarAsNeeded)
+ file_scroll_area.setHorizontalScrollBarPolicy(Qt.ScrollBarPolicy.ScrollBarAsNeeded)
+
+ # Create content widget for ScrollArea
+ file_content_widget = QWidget()
+ file_content_layout = QVBoxLayout(file_content_widget)
+
+ # Drag and drop area
+ self.batch_drop_area = BatchDropZoneWidget("Drag archive files here\nor click to browse")
+ file_content_layout.addWidget(self.batch_drop_area)
+
+ # File list
+ self.batch_files_listbox = ListWidget()
+ self.batch_files_listbox.setMinimumHeight(200)
+ self.batch_files_listbox.setMinimumWidth(300)
+ file_content_layout.addWidget(self.batch_files_listbox)
+
+ # File management buttons
+ file_buttons_layout = QHBoxLayout()
+
+ self.batch_add_files_btn = PushButton("Add Files")
+ self.batch_add_files_btn.setMinimumWidth(80)
+ self.batch_add_files_btn.clicked.connect(self.browse_batch_archive_files)
+ file_buttons_layout.addWidget(self.batch_add_files_btn)
+
+ self.batch_remove_files_btn = PushButton("Remove Selected")
+ self.batch_remove_files_btn.setMinimumWidth(100)
+ self.batch_remove_files_btn.clicked.connect(self.remove_selected_batch_files)
+ file_buttons_layout.addWidget(self.batch_remove_files_btn)
+
+ self.batch_clear_files_btn = PushButton("Clear All")
+ self.batch_clear_files_btn.setMinimumWidth(80)
+ self.batch_clear_files_btn.clicked.connect(self.clear_batch_files)
+ file_buttons_layout.addWidget(self.batch_clear_files_btn)
+
+ file_buttons_layout.addStretch()
+ file_content_layout.addLayout(file_buttons_layout)
+
+ # Set content widget to ScrollArea
+ file_scroll_area.setWidget(file_content_widget)
+
+ # Add ScrollArea to GroupBox layout
+ file_group_layout.addWidget(file_scroll_area)
+
+ left_layout.addWidget(file_group)
+ left_layout.addStretch(1)
+
+ # Right side - Configuration and progress
+ right_panel = QWidget()
+ right_layout = QVBoxLayout(right_panel)
+
+ # Destination folder group with ScrollArea inside GroupBox
+ dest_group = QGroupBox("Destination Folder")
+ dest_group.setMinimumHeight(200) # Set minimum height
+ dest_group_layout = QVBoxLayout(dest_group)
+
+ # Create ScrollArea inside GroupBox
+ dest_scroll_area = ScrollArea()
+ dest_scroll_area.setWidgetResizable(True)
+ dest_scroll_area.setVerticalScrollBarPolicy(Qt.ScrollBarPolicy.ScrollBarAsNeeded)
+ dest_scroll_area.setHorizontalScrollBarPolicy(Qt.ScrollBarPolicy.ScrollBarAsNeeded)
+
+ # Create content widget for ScrollArea
+ dest_content_widget = QWidget()
+ dest_content_layout = QVBoxLayout(dest_content_widget)
+
+ dest_layout = QHBoxLayout()
+ self.batch_extract_dest_text = LineEdit()
+ setCustomStyleSheet(self.batch_extract_dest_text, CON.qss_line, CON.qss_line)
+ dest_layout.addWidget(self.batch_extract_dest_text, 1)
+
+ batch_dest_button = PushButton("Browse...")
+ batch_dest_button.setMinimumWidth(70)
+ batch_dest_button.clicked.connect(self.browse_batch_extract_dest)
+ dest_layout.addWidget(batch_dest_button)
+
+ dest_content_layout.addLayout(dest_layout)
+
+ # Set content widget to ScrollArea
+ dest_scroll_area.setWidget(dest_content_widget)
+
+ # Add ScrollArea to GroupBox layout
+ dest_group_layout.addWidget(dest_scroll_area)
+
+ right_layout.addWidget(dest_group)
+
+ # Options group with ScrollArea inside GroupBox
+ options_group = QGroupBox("Extract Options")
+ options_group.setMinimumHeight(200) # Set minimum height
+ options_group_layout = QVBoxLayout(options_group)
+
+ # Create ScrollArea inside GroupBox
+ options_scroll_area = ScrollArea()
+ options_scroll_area.setWidgetResizable(True)
+ options_scroll_area.setVerticalScrollBarPolicy(Qt.ScrollBarPolicy.ScrollBarAsNeeded)
+ options_scroll_area.setHorizontalScrollBarPolicy(Qt.ScrollBarPolicy.ScrollBarAsNeeded)
+
+ # Create content widget for ScrollArea
+ options_content_widget = QWidget()
+ options_content_layout = QVBoxLayout(options_content_widget)
+
+ options_layout = QHBoxLayout()
+
+ # Left options column
+ left_options = QVBoxLayout()
+ self.batch_create_subfolders_check = CheckBox("Create subfolder for each archive")
+ self.batch_create_subfolders_check.setChecked(True)
+ left_options.addWidget(self.batch_create_subfolders_check)
+
+ # Overwrite options
+ self.batch_overwrite_files_check = CheckBox("Overwrite existing files")
+ self.batch_overwrite_files_check.setChecked(False)
+ left_options.addWidget(self.batch_overwrite_files_check)
+
+ # Skip existing files option
+ self.batch_skip_existing_files_check = CheckBox("Skip existing files")
+ self.batch_skip_existing_files_check.setChecked(False)
+ left_options.addWidget(self.batch_skip_existing_files_check)
+
+ # Overwrite strategy options
+ self.overwrite_strategy_combo = ModelComboBox()
+ self.overwrite_strategy_combo.addItems([
+ "Overwrite all",
+ "Skip existing",
+ "Rename new",
+ "Overwrite if newer"
+ ])
+ self.overwrite_strategy_combo.setMinimumHeight(30)
+ setCustomStyleSheet(self.overwrite_strategy_combo, CON.qss_combo_2, CON.qss_combo_2)
+ left_options.addWidget(QLabel("Overwrite Strategy:"))
+
+ left_options.addWidget(self.overwrite_strategy_combo)
- tab_sizer.addStretch(1) # Push content to top
+ options_layout.addLayout(left_options)
+ options_layout.addStretch(1)
+
+ options_content_layout.addLayout(options_layout)
+
+ # Set content widget to ScrollArea
+ options_scroll_area.setWidget(options_content_widget)
+
+ # Add ScrollArea to GroupBox layout
+ options_group_layout.addWidget(options_scroll_area)
+
+ right_layout.addWidget(options_group)
+
+ # Progress group with ScrollArea inside GroupBox
+ progress_group = QGroupBox("Progress & Statistics")
+ progress_group.setMinimumHeight(200) # Set minimum height
+ progress_group_layout = QVBoxLayout(progress_group)
+
+ # Create ScrollArea inside GroupBox
+ progress_scroll_area = ScrollArea()
+ progress_scroll_area.setWidgetResizable(True)
+ progress_scroll_area.setVerticalScrollBarPolicy(Qt.ScrollBarPolicy.ScrollBarAsNeeded)
+ progress_scroll_area.setHorizontalScrollBarPolicy(Qt.ScrollBarPolicy.ScrollBarAsNeeded)
+
+ # Create content widget for ScrollArea
+ progress_content_widget = QWidget()
+ progress_content_layout = QVBoxLayout(progress_content_widget)
+
+ # Progress label
+ self.batch_progress_label = QLabel("Ready to extract archives")
+ self.batch_progress_label.setWordWrap(True)
+ progress_content_layout.addWidget(self.batch_progress_label)
+
+ # Progress bar
+ self.batch_progress = ProgressBar()
+ self.batch_progress.setRange(0, 100)
+ self.batch_progress.setValue(0)
+ progress_content_layout.addWidget(self.batch_progress)
+
+ # Statistics in a grid layout
+ stats_widget = QWidget()
+ stats_layout = QGridLayout(stats_widget)
+ stats_layout.setSpacing(10)
+
+ # Statistics labels
+ self.batch_total_count_label = QLabel("Total Archives:")
+ self.batch_total_count_value = QLabel("0")
+ stats_layout.addWidget(self.batch_total_count_label, 0, 0)
+ stats_layout.addWidget(self.batch_total_count_value, 0, 1)
+
+ self.batch_success_count_label = QLabel("Successful:")
+ self.batch_success_count_value = QLabel("0")
+ stats_layout.addWidget(self.batch_success_count_label, 1, 0)
+ stats_layout.addWidget(self.batch_success_count_value, 1, 1)
+
+ self.batch_failed_count_label = QLabel("Failed:")
+ self.batch_failed_count_value = QLabel("0")
+ stats_layout.addWidget(self.batch_failed_count_label, 2, 0)
+ stats_layout.addWidget(self.batch_failed_count_value, 2, 1)
+
+ progress_content_layout.addWidget(stats_widget)
+
+ # Set content widget to ScrollArea
+ progress_scroll_area.setWidget(progress_content_widget)
+
+ # Add ScrollArea to GroupBox layout
+ progress_group_layout.addWidget(progress_scroll_area)
+
+ right_layout.addWidget(progress_group)
+
+ # Control buttons
+ button_widget = QWidget()
+ button_layout = QHBoxLayout(button_widget)
+ button_layout.setSpacing(10)
+
+ self.batch_start_btn = PrimaryPushButton("Start Extract")
+ self.batch_start_btn.setMinimumWidth(100)
+ self.batch_start_btn.clicked.connect(self.start_batch_extract)
+ button_layout.addWidget(self.batch_start_btn)
+
+ self.batch_stop_btn = PushButton("Stop")
+ self.batch_stop_btn.setMinimumWidth(60)
+ self.batch_stop_btn.clicked.connect(self.stop_batch_extract)
+ self.batch_stop_btn.setEnabled(False)
+ button_layout.addWidget(self.batch_stop_btn)
+
+ button_layout.addStretch(1)
+ right_layout.addWidget(button_widget)
+
+ # Add panels to main layout with proportional sizing
+ main_layout.addWidget(left_panel, 2) # 2/4 of width for file management
+ main_layout.addWidget(right_panel, 2) # 2/4 of width for options and progress - equal width for both panels
+
+ self.extract_tab_widget.addTab(batch_panel, "Batch Extract")
+
+ # Connect drop area signals
+ self.batch_drop_area.files_dropped.connect(self.on_batch_files_dropped)
def create_add_tab(self):
tab_panel = QWidget()
tab_sizer = QVBoxLayout(tab_panel)
self.notebook.addTab(tab_panel, "Add to Archive") # Changed tab title
- # Existing Archive file selection
+ # Existing Archive file selection with ScrollArea inside GroupBox
zip_box = QGroupBox("Existing Archive File") # Changed group box title
- zip_box_sizer = QHBoxLayout(zip_box)
+ zip_box.setMinimumHeight(200) # Set minimum height
+ zip_box_sizer = QVBoxLayout(zip_box)
+
+ # Create ScrollArea inside GroupBox
+ zip_scroll_area = ScrollArea()
+ zip_scroll_area.setWidgetResizable(True)
+ zip_scroll_area.setVerticalScrollBarPolicy(Qt.ScrollBarPolicy.ScrollBarAsNeeded)
+ zip_scroll_area.setHorizontalScrollBarPolicy(Qt.ScrollBarPolicy.ScrollBarAsNeeded)
+
+ # Create content widget for ScrollArea
+ zip_content_widget = QWidget()
+ zip_content_layout = QHBoxLayout(zip_content_widget)
self.add_zip_text = LineEdit()
setCustomStyleSheet(self.add_zip_text, CON.qss_line, CON.qss_line)
# self.add_zip_text.setReadOnly(True) # Allow users to manually input path
- zip_box_sizer.addWidget(self.add_zip_text, 1)
+ zip_content_layout.addWidget(self.add_zip_text, 1)
zip_button = PushButton("Browse...")
zip_button.clicked.connect(self.browse_add_archive) # Changed signal
- zip_box_sizer.addWidget(zip_button)
+ zip_content_layout.addWidget(zip_button)
+
+ # Set content widget to ScrollArea
+ zip_scroll_area.setWidget(zip_content_widget)
+
+ # Add ScrollArea to GroupBox layout
+ zip_box_sizer.addWidget(zip_scroll_area)
+
tab_sizer.addWidget(zip_box)
- # File to add selection
+ # File to add selection with ScrollArea inside GroupBox
file_box = QGroupBox("Files to Add")
+ file_box.setMinimumHeight(200) # Set minimum height
file_box_sizer = QVBoxLayout(file_box)
+
+ # Create ScrollArea inside GroupBox
+ file_scroll_area = ScrollArea()
+ file_scroll_area.setWidgetResizable(True)
+ file_scroll_area.setVerticalScrollBarPolicy(Qt.ScrollBarPolicy.ScrollBarAsNeeded)
+ file_scroll_area.setHorizontalScrollBarPolicy(Qt.ScrollBarPolicy.ScrollBarAsNeeded)
+
+ # Create content widget for ScrollArea
+ file_content_widget = QWidget()
+ file_content_layout = QVBoxLayout(file_content_widget)
# File list for multiple files (always visible)
self.add_files_listbox = ListWidget()
self.add_files_listbox.setMinimumHeight(150)
self.add_files_listbox.setVisible(True) # Always visible
- file_box_sizer.addWidget(self.add_files_listbox)
+ file_content_layout.addWidget(self.add_files_listbox)
# Browse button
file_button = PushButton("Browse...")
file_button.clicked.connect(self.browse_add_file)
- file_box_sizer.addWidget(file_button)
+ file_content_layout.addWidget(file_button)
+
+ # Set content widget to ScrollArea
+ file_scroll_area.setWidget(file_content_widget)
+
+ # Add ScrollArea to GroupBox layout
+ file_box_sizer.addWidget(file_scroll_area)
tab_sizer.addWidget(file_box)
@@ -516,11 +1270,28 @@ def create_add_tab(self):
self.add_progress.setValue(0)
tab_sizer.addWidget(self.add_progress)
+ # Add button layout with cancel button
+ button_layout = QHBoxLayout()
+ button_layout.setSpacing(10)
+
# Add button
self.add_button = PrimaryPushButton("Add to Archive") # Changed button text
-
self.add_button.clicked.connect(self.start_add_to_archive) # Changed signal
- tab_sizer.addWidget(self.add_button, 0, Qt.AlignmentFlag.AlignCenter)
+ button_layout.addWidget(self.add_button)
+
+ # Cancel button
+ self.add_cancel_button = PushButton("Cancel")
+ self.add_cancel_button.clicked.connect(self.cancel_add_to_archive)
+ self.add_cancel_button.setEnabled(False) # Initially disabled
+ button_layout.addWidget(self.add_cancel_button)
+
+ # Create a container widget to center the button layout
+ button_container = QWidget()
+ button_container_layout = QHBoxLayout(button_container)
+ button_container_layout.addStretch()
+ button_container_layout.addLayout(button_layout)
+ button_container_layout.addStretch()
+ tab_sizer.addWidget(button_container)
tab_sizer.addStretch(1) # Push content to top
@@ -529,18 +1300,36 @@ def create_list_tab(self):
tab_sizer = QVBoxLayout(tab_panel)
self.notebook.addTab(tab_panel, "List Contents")
- # Archive file selection (changed title)
+ # Archive file selection with ScrollArea inside GroupBox (changed title)
zip_box = QGroupBox("Archive File")
- zip_box_sizer = QHBoxLayout(zip_box)
+ zip_box.setMinimumHeight(200) # Set minimum height
+ zip_box_sizer = QVBoxLayout(zip_box)
+
+ # Create ScrollArea inside GroupBox
+ zip_scroll_area = ScrollArea()
+ zip_scroll_area.setWidgetResizable(True)
+ zip_scroll_area.setVerticalScrollBarPolicy(Qt.ScrollBarPolicy.ScrollBarAsNeeded)
+ zip_scroll_area.setHorizontalScrollBarPolicy(Qt.ScrollBarPolicy.ScrollBarAsNeeded)
+
+ # Create content widget for ScrollArea
+ zip_content_widget = QWidget()
+ zip_content_layout = QHBoxLayout(zip_content_widget)
self.list_zip_text = LineEdit()
setCustomStyleSheet(self.list_zip_text, CON.qss_line, CON.qss_line)
# self.list_zip_text.setReadOnly(True) # Allow users to manually input path
- zip_box_sizer.addWidget(self.list_zip_text, 1)
+ zip_content_layout.addWidget(self.list_zip_text, 1)
zip_button = PushButton("Browse...")
zip_button.clicked.connect(self.browse_list_archive) # Changed signal
- zip_box_sizer.addWidget(zip_button)
+ zip_content_layout.addWidget(zip_button)
+
+ # Set content widget to ScrollArea
+ zip_scroll_area.setWidget(zip_content_widget)
+
+ # Add ScrollArea to GroupBox layout
+ zip_box_sizer.addWidget(zip_scroll_area)
+
tab_sizer.addWidget(zip_box)
# Password status indicator
@@ -553,25 +1342,60 @@ def create_list_tab(self):
password_status_box.addStretch()
tab_sizer.addLayout(password_status_box)
- # Listbox for contents
+ # Listbox for contents with ScrollArea inside GroupBox
contents_box = QGroupBox("Archive Contents") # Changed group box title
+ contents_box.setMinimumHeight(200) # Set minimum height
contents_box_sizer = QVBoxLayout(contents_box)
+
+ # Create ScrollArea inside GroupBox
+ contents_scroll_area = ScrollArea()
+ contents_scroll_area.setWidgetResizable(True)
+ contents_scroll_area.setVerticalScrollBarPolicy(Qt.ScrollBarPolicy.ScrollBarAsNeeded)
+ contents_scroll_area.setHorizontalScrollBarPolicy(Qt.ScrollBarPolicy.ScrollBarAsNeeded)
+
+ # Create content widget for ScrollArea
+ contents_content_widget = QWidget()
+ contents_content_layout = QVBoxLayout(contents_content_widget)
self.contents_listbox = ListWidget()
self.contents_listbox.setMinimumHeight(250) # Set larger minimum height
self.contents_listbox.setDragEnabled(True) # Enable drag functionality
- contents_box_sizer.addWidget(self.contents_listbox, 3) # Increase stretch weight
+ contents_content_layout.addWidget(self.contents_listbox, 3) # Increase stretch weight
# Set right-click menu
self.contents_listbox.setContextMenuPolicy(Qt.ContextMenuPolicy.CustomContextMenu)
# Context menu functionality removed
# self.contents_listbox.customContextMenuRequested.connect(self.show_contents_context_menu)
- tab_sizer.addWidget(contents_box, 2) # Give contents box more stretch
+
+ # Set content widget to ScrollArea
+ contents_scroll_area.setWidget(contents_content_widget)
+
+ # Add ScrollArea to GroupBox layout
+ contents_box_sizer.addWidget(contents_scroll_area)
+
+ tab_sizer.addWidget(contents_box, 2) # Give contents group box more stretch
+ # List button layout with cancel button
+ button_layout = QHBoxLayout()
+ button_layout.setSpacing(10)
+
# List button
self.list_button = PrimaryPushButton("List Contents")
-
self.list_button.clicked.connect(self.start_list_archive_contents) # Changed signal
- tab_sizer.addWidget(self.list_button, 0, Qt.AlignmentFlag.AlignCenter)
+ button_layout.addWidget(self.list_button)
+
+ # Cancel button
+ self.list_cancel_button = PushButton("Cancel")
+ self.list_cancel_button.clicked.connect(self.cancel_list_archive_contents)
+ self.list_cancel_button.setEnabled(False) # Initially disabled
+ button_layout.addWidget(self.list_cancel_button)
+
+ # Create a container widget to center the button layout
+ button_container = QWidget()
+ button_container_layout = QHBoxLayout(button_container)
+ button_container_layout.addStretch()
+ button_container_layout.addLayout(button_layout)
+ button_container_layout.addStretch()
+ tab_sizer.addWidget(button_container)
tab_sizer.addStretch(1) # Push content to top
@@ -699,88 +1523,38 @@ def _verify_password_strength(self, password):
# We don't need to verify it against an existing archive since we're creating a new one
return True
- def on_tab_changed(self, index):
- """Handle tab change with optional slide animation effect based on UI_FLUENT environment variable"""
- import sys
- import os
- sys.path.append(os.path.join(os.path.dirname(__file__), 'support'))
- from support.check_flag import check_flag
-
- # Check if UI_FLUENT environment variable is set to YES using check_flag function
- ui_fluent_enabled = check_flag("UI_FLUENT")
-
- # Skip animation if UI_FLUENT is not enabled
- if not ui_fluent_enabled:
- self._previous_tab_index = index
- # Force layout update when animation is disabled
- self.notebook.currentWidget().updateGeometry()
- if self.notebook.currentWidget().layout():
- self.notebook.currentWidget().layout().update()
- self.notebook.currentWidget().layout().activate()
- return
-
- # Proceed with animation if UI_FLUENT is enabled
- from PySide6.QtCore import QPropertyAnimation, QEasingCurve, QRect
+ def confirm_cancel(self, operation_name):
+ """Show confirmation dialog for canceling an operation
- # Get current tab widget
- current_widget = self.notebook.currentWidget()
- if not current_widget:
- return
-
- # Skip animation during initial startup to prevent layout issues
- if not hasattr(self, '_previous_tab_index') and not self.notebook.isVisible():
- self._previous_tab_index = index
- return
+ Args:
+ operation_name: Name of the operation being canceled
- # Get tab widget dimensions
- tab_width = self.notebook.width()
- tab_height = self.notebook.height()
-
- # Skip animation if window is not yet properly sized
- if tab_width <= 0 or tab_height <= 0:
- self._previous_tab_index = index
- return
-
- # Determine slide direction based on tab index
- if hasattr(self, '_previous_tab_index'):
- if index > self._previous_tab_index:
- # Sliding from right to left - start from 80% of width to prevent going out of bounds
- start_pos = QRect(int(tab_width * 0.8), 0, tab_width, tab_height)
- else:
- # Sliding from left to right - start from -80% of width to prevent going out of bounds
- start_pos = QRect(int(-tab_width * 0.8), 0, tab_width, tab_height)
- else:
- # First time, slide from right - start from 80% of width
- start_pos = QRect(int(tab_width * 0.8), 0, tab_width, tab_height)
-
- # Set initial position
- current_widget.setGeometry(start_pos)
-
- # Create slide animation
- self.slide_animation = QPropertyAnimation(current_widget, b"geometry")
- self.slide_animation.setDuration(300) # 300ms animation for smooth slide
- self.slide_animation.setStartValue(start_pos)
- self.slide_animation.setEndValue(QRect(0, 0, tab_width, tab_height))
- self.slide_animation.setEasingCurve(QEasingCurve.Type.OutCubic)
-
- # Store current tab index for next animation
- self._previous_tab_index = index
-
- # Connect animation finished signal to update layout
- self.slide_animation.finished.connect(lambda: self._update_tab_layout(current_widget))
+ Returns:
+ bool: True if user confirmed cancel, False otherwise
+ """
+ reply = QMessageBox.question(
+ self,
+ "取消操作",
+ f"确定要取消当前的{operation_name}操作吗?",
+ QMessageBox.StandardButton.Yes | QMessageBox.StandardButton.No,
+ QMessageBox.StandardButton.No
+ )
+ return reply == QMessageBox.StandardButton.Yes
+
+ def log_cancel(self, operation_type):
+ """Log a cancel operation
- # Start the animation
- self.slide_animation.start()
+ Args:
+ operation_type: Type of operation that was canceled
+ """
+ import datetime
+ timestamp = datetime.datetime.now().strftime("%Y-%m-%d %H:%M:%S")
+ print(f"[{timestamp}] [CANCEL] {operation_type} operation canceled by user")
- def _update_tab_layout(self, widget):
- """Update widget layout after animation completes"""
- # Force layout update to prevent layout issues
- if widget and widget.layout():
- widget.layout().update()
- widget.layout().activate()
- widget.updateGeometry()
- # Repaint the widget to ensure all elements are properly displayed
- widget.repaint()
+ def on_tab_changed(self, index):
+ """Handle tab change without animation"""
+ # Simply store the previous tab index and return
+ self._previous_tab_index = index
@@ -950,6 +1724,10 @@ def start_create_archive(self):
self.create_progress_label.setText("Starting archive creation...")
self.create_progress.setValue(0)
+ # Enable cancel button and disable create button during operation
+ self.create_button.setEnabled(False)
+ self.create_cancel_button.setEnabled(True)
+
self.create_zip_worker = CreateZipWorker(self.create_output_path, self.create_sources, self.create_archive_format, password)
self.create_zip_worker_thread = QThread()
self.create_zip_worker.moveToThread(self.create_zip_worker_thread)
@@ -957,6 +1735,7 @@ def start_create_archive(self):
self.create_zip_worker.finished.connect(self.on_create_archive_finished)
self.create_zip_worker.progress_updated.connect(self.update_create_progress)
self.create_zip_worker.conversion_error.connect(self.on_create_archive_error)
+ self.create_zip_worker.canceled.connect(self.on_create_archive_canceled)
self.create_zip_worker_thread.started.connect(self.create_zip_worker.run)
self.create_zip_worker_thread.start()
@@ -998,8 +1777,43 @@ def on_create_archive_error(self, error_message):
# Update archive status display
self.update_archive_status(archive_info, False)
+ def cancel_create_archive(self):
+ """Cancel the archive creation process"""
+ if self.confirm_cancel("归档创建"):
+ self.log_cancel("Create Archive")
+ # Stop the worker
+ if self.create_zip_worker:
+ self.create_zip_worker.stop()
+
+ def on_create_archive_canceled(self):
+ """Handle archive creation canceled"""
+ # 使用强制线程清理方法
+ self._force_cleanup_create_thread()
+
+ # Reset button states
+ self.create_button.setEnabled(True)
+ self.create_cancel_button.setEnabled(False)
+
+ # Update progress and status
+ self.create_progress.setValue(0)
+ self.create_progress_label.setText("Archive creation canceled")
+
+ # Show cancel notification
+ self._show_info_bar(
+ title='Canceled',
+ content='Archive creation canceled by user',
+ duration=2000
+ )
+
+ # Update archive status display
+ self.update_archive_status("Archive creation canceled", False)
+
def _force_cleanup_create_thread(self):
"""强制清理创建归档的线程,确保完全终止"""
+ # Reset button states
+ self.create_button.setEnabled(True)
+ self.create_cancel_button.setEnabled(False)
+
if self.create_zip_worker_thread:
if self.create_zip_worker_thread.isRunning():
# 先尝试正常退出
@@ -1056,6 +1870,390 @@ def browse_extract_dest(self):
self.extract_dest_path = dir_dialog.selectedFiles()[0]
self.extract_dest_text.setText(self.extract_dest_path)
+ def browse_batch_extract_dest(self):
+ """Browse for batch extraction destination folder"""
+ dir_dialog = QFileDialog(self)
+ dir_dialog.setFileMode(QFileDialog.FileMode.Directory)
+ dir_dialog.setOption(QFileDialog.Option.ShowDirsOnly, True)
+ if dir_dialog.exec():
+ self.batch_extract_dest_path = dir_dialog.selectedFiles()[0]
+ self.batch_extract_dest_text.setText(self.batch_extract_dest_path)
+
+ def browse_batch_archive_files(self):
+ """Browse for archive files to batch extract"""
+ file_dialog = QFileDialog(self)
+ file_dialog.setFileMode(QFileDialog.FileMode.ExistingFiles)
+ file_dialog.setNameFilters([
+ "Archive files (*.zip *.rar *.7z *.tar *.gz *.bz2 *.xz *.tar.gz *.tar.bz2 *.tar.xz *.tgz *.tbz2)",
+ "All files (*)"
+ ])
+ if file_dialog.exec():
+ file_paths = file_dialog.selectedFiles()
+ self.add_batch_files(file_paths)
+
+ def add_batch_files(self, file_paths):
+ """Add files to batch list"""
+ supported_formats = ('.zip', '.rar', '.7z', '.tar', '.gz', '.bz2', '.xz', '.tar.gz', '.tar.bz2', '.tar.xz', '.tgz', '.tbz2')
+
+ for file_path in file_paths:
+ # Check if file is a supported archive format
+ if os.path.splitext(file_path.lower())[1] in supported_formats:
+ # Avoid duplicates
+ if file_path not in [self.batch_files_listbox.item(i).toolTip() for i in range(self.batch_files_listbox.count())]:
+ item = QListWidgetItem(os.path.basename(file_path))
+ item.setToolTip(file_path)
+ self.batch_files_listbox.addItem(item)
+
+ self.update_batch_stats()
+
+ def remove_selected_batch_files(self):
+ """Remove selected files from batch list"""
+ selected_items = self.batch_files_listbox.selectedItems()
+ if not selected_items:
+ return
+
+ # Remove items in reverse order to maintain correct indices
+ indices = []
+ for item in selected_items:
+ indices.append(self.batch_files_listbox.row(item))
+
+ indices.sort(reverse=True)
+
+ for index in indices:
+ self.batch_files_listbox.takeItem(index)
+
+ self.update_batch_stats()
+
+ def clear_batch_files(self):
+ """Clear all files from batch list"""
+ self.batch_files_listbox.clear()
+ self.update_batch_stats()
+
+ def get_batch_archive_files(self):
+ """Get list of all archive files in batch list"""
+ file_paths = []
+ for i in range(self.batch_files_listbox.count()):
+ item = self.batch_files_listbox.item(i)
+ file_paths.append(item.toolTip())
+ return file_paths
+
+ def on_batch_files_dropped(self, file_paths):
+ """Handle files dropped in batch area"""
+ self.add_batch_files(file_paths)
+
+ def update_batch_stats(self):
+ """Update batch statistics display"""
+ total_count = self.batch_files_listbox.count()
+ self.batch_total_count_label.setText(f"Total: {total_count}")
+ self.batch_success_count_label.setText("Success: 0")
+ self.batch_failed_count_label.setText("Failed: 0")
+
+ def start_batch_extract(self):
+ """Start batch extraction process"""
+ file_paths = self.get_batch_archive_files()
+
+ if not file_paths:
+ self._show_popup(
+ target=self.batch_start_btn,
+ icon=InfoBarIcon.ERROR,
+ title='Error',
+ content='Please add archive files to extract',
+ duration=2000
+ )
+ return
+
+ if not self.batch_extract_dest_path:
+ self._show_popup(
+ target=self.batch_extract_dest_text,
+ icon=InfoBarIcon.ERROR,
+ title='Error',
+ content='Please specify the extraction destination folder',
+ duration=2000
+ )
+ return
+
+ # Validate destination folder
+ if not os.path.exists(self.batch_extract_dest_path):
+ try:
+ os.makedirs(self.batch_extract_dest_path, exist_ok=True)
+ except Exception as e:
+ self._show_popup(
+ target=self.batch_start_btn,
+ icon=InfoBarIcon.ERROR,
+ title='Error',
+ content=f'Failed to create destination folder: {str(e)}',
+ duration=2000
+ )
+ return
+
+ # Create batch extract options
+ create_subfolders = self.batch_create_subfolders_check.isChecked()
+
+ # Get overwrite strategy
+ overwrite_strategy = self.overwrite_strategy_combo.currentText()
+
+ # Determine overwrite behavior based on strategy
+ if overwrite_strategy == "Overwrite all":
+ overwrite_files = True
+ skip_existing = False
+ elif overwrite_strategy == "Skip existing":
+ overwrite_files = False
+ skip_existing = True
+ elif overwrite_strategy == "Rename new":
+ overwrite_files = False
+ skip_existing = False
+ # Note: Rename new functionality would need to be implemented in the archive_manager
+ elif overwrite_strategy == "Overwrite if newer":
+ overwrite_files = True
+ skip_existing = False
+ # Note: Overwrite if newer functionality would need to be implemented in the archive_manager
+ else:
+ # Default behavior
+ overwrite_files = self.batch_overwrite_files_check.isChecked()
+ skip_existing = self.batch_skip_existing_files_check.isChecked()
+
+ # Reset progress and statistics
+ self.batch_progress.setValue(0)
+ self.update_batch_stats()
+ self.batch_progress_label.setText("Preparing for batch extraction...")
+
+ # Disable start button and enable stop button
+ self.batch_start_btn.setEnabled(False)
+ self.batch_stop_btn.setEnabled(True)
+
+ # Create and start batch worker
+ self.batch_extract_worker = BatchExtractWorker(
+ file_paths,
+ self.batch_extract_dest_path,
+ create_subfolders,
+ overwrite_files,
+ parent_gui=self # Pass reference to main GUI for password dialogs
+ )
+ self.batch_extract_worker_thread = QThread()
+ self.batch_extract_worker.moveToThread(self.batch_extract_worker_thread)
+
+ # Connect signals
+ self.batch_extract_worker.finished.connect(self.on_batch_extract_finished)
+ self.batch_extract_worker.progress_updated.connect(self.on_batch_extract_progress)
+ self.batch_extract_worker.conversion_error.connect(self.on_batch_extract_error)
+ self.batch_extract_worker.individual_progress.connect(self.on_batch_individual_progress)
+ self.batch_extract_worker.status_updated.connect(self.on_batch_status_updated)
+ self.batch_extract_worker_thread.started.connect(self.batch_extract_worker.run)
+ self.batch_extract_worker_thread.start()
+
+ def stop_batch_extract(self):
+ """Stop batch extraction process"""
+ if hasattr(self, 'batch_extract_worker'):
+ self.batch_extract_worker.stop()
+ self.batch_progress_label.setText("Stopping batch extraction...")
+ # Disable stop button to prevent multiple stops
+ self.batch_stop_btn.setEnabled(False)
+ # Start a timer to check if thread has stopped after a timeout
+ QTimer.singleShot(2000, self._check_batch_thread_status)
+
+ def _check_batch_thread_status(self):
+ """Check if batch thread has stopped after timeout"""
+ if hasattr(self, 'batch_extract_worker_thread') and self.batch_extract_worker_thread.isRunning():
+ # Thread is still running, try to force stop
+ self.batch_progress_label.setText("Force stopping batch extraction...")
+ self._force_stop_batch_thread()
+ self.on_batch_extract_stopped()
+
+ def _force_stop_batch_thread(self):
+ """Force stop batch extraction thread"""
+ if hasattr(self, 'batch_extract_worker_thread') and self.batch_extract_worker_thread.isRunning():
+ try:
+ # Try to quit gracefully first
+ self.batch_extract_worker_thread.quit()
+ if not self.batch_extract_worker_thread.wait(1000): # Wait 1 second
+ # If graceful quit fails, terminate
+ self.batch_extract_worker_thread.terminate()
+ self.batch_extract_worker_thread.wait(1000) # Wait another second
+ except Exception as e:
+ print(f"Error stopping batch thread: {str(e)}")
+
+ def on_batch_extract_stopped(self):
+ """Handle batch extraction stopped by user"""
+ self.batch_progress_label.setText("Batch extraction stopped by user")
+ self._cleanup_batch_thread()
+ # Update UI
+ self.batch_start_btn.setEnabled(True)
+ self.batch_stop_btn.setEnabled(False)
+
+ def _cleanup_batch_thread(self):
+ """Clean up batch extraction thread and worker resources"""
+ # Clean up worker thread
+ if hasattr(self, 'batch_extract_worker_thread'):
+ if self.batch_extract_worker_thread.isRunning():
+ try:
+ self.batch_extract_worker_thread.quit()
+ self.batch_extract_worker_thread.wait(1000)
+ except Exception as e:
+ print(f"Error cleaning up batch thread: {str(e)}")
+
+ # Delete thread object
+ self.batch_extract_worker_thread.deleteLater()
+ delattr(self, 'batch_extract_worker_thread')
+
+ # Clean up worker
+ if hasattr(self, 'batch_extract_worker'):
+ # Delete worker object
+ self.batch_extract_worker.deleteLater()
+ delattr(self, 'batch_extract_worker')
+
+ def on_batch_extract_progress(self, processed_count, total_count, current_file, success_count, failed_count):
+ """Handle batch extraction progress update"""
+ progress_percentage = int((processed_count / total_count) * 100)
+ self.batch_progress.setValue(progress_percentage)
+
+ # Update statistics
+ self.batch_total_count_value.setText(str(total_count))
+ self.batch_success_count_value.setText(str(success_count))
+ self.batch_failed_count_value.setText(str(failed_count))
+
+ # Update progress label with more detailed information
+ current_file_name = os.path.basename(current_file) if current_file else ""
+ self.batch_progress_label.setText(f"Processing: {current_file_name} ({processed_count}/{total_count}) - {progress_percentage}%")
+
+ def on_batch_individual_progress(self, archive_name, message, progress):
+ """Handle individual archive extraction progress"""
+ # Update status bar with individual file progress
+ self.status_bar.showMessage(f"Extracting {archive_name}: {message} - {progress}%")
+
+ def on_batch_status_updated(self, status_message):
+ """Handle batch extraction status updates"""
+ # Update status bar with general status messages
+ self.status_bar.showMessage(status_message)
+
+ def on_batch_extract_finished(self, success_count, failed_count, success_files=None, failed_files=None):
+ """Handle batch extraction finished"""
+ # Clean up thread
+ self._cleanup_batch_thread()
+
+ # Update final statistics
+ total_count = success_count + failed_count
+ self.batch_total_count_value.setText(str(total_count))
+ self.batch_success_count_value.setText(str(success_count))
+ self.batch_failed_count_value.setText(str(failed_count))
+
+ # Re-enable start button and disable stop button
+ self.batch_start_btn.setEnabled(True)
+ self.batch_stop_btn.setEnabled(False)
+
+ # Show completion message with detailed results
+ result_message = f"Batch extraction completed: {success_count} successful, {failed_count} failed"
+ self.batch_progress_label.setText(result_message)
+
+ # Show appropriate message based on results
+ if failed_count == 0:
+ self._show_info_bar(
+ title='Success',
+ content=f'All {total_count} archives extracted successfully!',
+ duration=3000
+ )
+ elif success_count == 0:
+ self._show_popup(
+ target=self.batch_progress,
+ icon=InfoBarIcon.ERROR,
+ title='Error',
+ content=f'Failed to extract all {total_count} archives.',
+ duration=3000
+ )
+ # Show detailed failures if available
+ if failed_files:
+ self._show_batch_extract_failures(failed_files)
+ else:
+ self._show_info_bar(
+ title='Partially Complete',
+ content=f'Extracted {success_count} out of {total_count} archives successfully.',
+ duration=3000
+ )
+ # Show detailed failures if available
+ if failed_files:
+ self._show_batch_extract_failures(failed_files)
+
+ # Clear status bar
+ self.status_bar.clearMessage()
+
+ def _show_batch_extract_failures(self, failed_files):
+ """Show detailed information about failed extractions"""
+ from qfluentwidgets import MessageBox
+
+ # Create detailed failure message
+ failure_details = "Failed to extract the following archives:\n\n"
+ for file_path, error_msg in failed_files[:10]: # Show first 10 failures
+ file_name = os.path.basename(file_path)
+ failure_details += f"• {file_name}: {error_msg}\n"
+
+ if len(failed_files) > 10:
+ failure_details += f"\n... and {len(failed_files) - 10} more failures."
+
+ # Show message box with failure details
+ msg_box = MessageBox(
+ 'Batch Extraction Failures',
+ failure_details,
+ self
+ )
+ msg_box.yesButton.setText('OK')
+ msg_box.exec()
+
+ def _force_cleanup_batch_thread(self):
+ """Deprecated method, use _cleanup_batch_thread instead"""
+ self._cleanup_batch_thread()
+
+ def on_batch_extract_error(self, error_message):
+ """Handle batch extraction error"""
+ # Clean up thread
+ self._force_cleanup_batch_thread()
+
+ # Re-enable start button and disable stop button
+ self.batch_start_btn.setEnabled(True)
+ self.batch_stop_btn.setEnabled(False)
+
+ self._show_popup(
+ target=self.batch_progress,
+ icon=InfoBarIcon.ERROR,
+ title='Error',
+ content=f'Batch extraction error: {str(error_message)}',
+ duration=3000
+ )
+
+ def on_batch_extract_stopped(self):
+ """Handle batch extraction stopped by user"""
+ # Clean up thread
+ self._force_cleanup_batch_thread()
+
+ # Re-enable start button and disable stop button
+ self.batch_start_btn.setEnabled(True)
+ self.batch_stop_btn.setEnabled(False)
+
+ self.batch_progress_label.setText("Stopped")
+
+ self._show_popup(
+ target=self.batch_progress,
+ icon=InfoBarIcon.WARNING,
+ title='Stopped',
+ content='Batch extraction stopped by user.',
+ duration=2000
+ )
+
+ def _force_cleanup_batch_thread(self):
+ """Force cleanup batch extraction thread"""
+ if hasattr(self, 'batch_extract_worker_thread') and self.batch_extract_worker_thread.isRunning():
+ self.batch_extract_worker_thread.quit()
+ self.batch_extract_worker_thread.wait()
+
+ def reset_batch_ui(self):
+ """Reset batch extraction UI to initial state"""
+ self.batch_files_listbox.clear()
+ self.batch_extract_dest_path = ""
+ self.batch_extract_dest_text.setText("Select destination folder...")
+ self.batch_create_subfolders_check.setChecked(True)
+ self.batch_overwrite_files_check.setChecked(False)
+ self.batch_progress.setValue(0)
+ self.update_batch_stats()
+ self.batch_progress_label.setText("Ready")
+
def auto_set_extract_dest_from_file(self, file_path):
"""Automatically set the extract destination to the file's parent directory"""
try:
@@ -1097,6 +2295,10 @@ def start_extract_archive(self):
self.extract_progress_label.setText("Starting archive extraction...")
self.extract_progress.setValue(0)
+
+ # Enable cancel button and disable extract button during operation
+ self.extract_button.setEnabled(False)
+ self.extract_cancel_button.setEnabled(True)
# Check if archive is password protected by attempting to list contents first
try:
@@ -1148,6 +2350,7 @@ def start_extract_archive(self):
self.extract_zip_worker.progress_updated.connect(self.update_extract_progress)
self.extract_zip_worker.conversion_error.connect(self.on_extract_archive_error)
self.extract_zip_worker.password_required.connect(self.on_extract_archive_error)
+ self.extract_zip_worker.canceled.connect(self.on_extract_archive_canceled)
self.extract_zip_worker_thread.started.connect(self.extract_zip_worker.run)
self.extract_zip_worker_thread.start()
@@ -1206,6 +2409,9 @@ def on_extract_archive_error(self, error_message):
self.extract_zip_worker = ExtractZipWorker(self.extract_zip_path, self.extract_dest_path, password)
self.extract_zip_worker_thread = QThread()
self.extract_zip_worker.moveToThread(self.extract_zip_worker_thread)
+
+ # Connect signals including canceled signal
+ self.extract_zip_worker.canceled.connect(self.on_extract_archive_canceled)
# 连接信号
self.extract_zip_worker.finished.connect(self.on_extract_archive_finished)
@@ -1248,8 +2454,40 @@ def on_extract_archive_error(self, error_message):
)
self.extract_progress_label.setText("Archive extraction failed.")
+ def cancel_extract_archive(self):
+ """Cancel the archive extraction process"""
+ if self.confirm_cancel("归档解压"):
+ self.log_cancel("Extract Archive")
+ # Stop the worker
+ if self.extract_zip_worker:
+ self.extract_zip_worker.stop()
+
+ def on_extract_archive_canceled(self):
+ """Handle archive extraction canceled"""
+ # 使用强制线程清理方法
+ self._force_cleanup_thread()
+
+ # Update progress and status
+ self.extract_progress.setValue(0)
+ self.extract_progress_label.setText("Archive extraction canceled")
+
+ # Show cancel notification
+ self._show_info_bar(
+ title='Canceled',
+ content='Archive extraction canceled by user',
+ duration=2000
+ )
+
+ # Update password status to indicate unknown status
+ self.is_password_protected = False
+ self.update_password_status_extract(False, "Archive Status Unknown")
+
def _force_cleanup_thread(self):
"""强制清理线程,确保完全终止"""
+ # Reset button states
+ self.extract_button.setEnabled(True)
+ self.extract_cancel_button.setEnabled(False)
+
if self.extract_zip_worker_thread:
if self.extract_zip_worker_thread.isRunning():
# 先尝试正常退出
@@ -1339,6 +2577,10 @@ def start_add_to_archive(self):
self.add_progress_label.setText("Starting archive file addition...")
self.add_progress.setValue(0)
+
+ # Enable cancel button and disable add button during operation
+ self.add_button.setEnabled(False)
+ self.add_cancel_button.setEnabled(True)
# Handle multiple files - split by semicolon if contains multiple paths
if isinstance(self.add_file_path, list):
@@ -1357,7 +2599,8 @@ def start_add_to_archive(self):
self.add_to_zip_worker.finished.connect(self.on_add_to_archive_finished)
self.add_to_zip_worker.progress_updated.connect(self.update_add_progress)
- self.add_to_zip_worker.conversion_error.connect(self.on_add_to_archive_error)
+ self.add_to_zip_worker.finished.connect(self.on_add_to_archive_finished)
+ self.add_to_zip_worker.canceled.connect(self.on_add_to_archive_canceled)
self.add_to_zip_worker_thread.started.connect(self.add_to_zip_worker.run)
self.add_to_zip_worker_thread.start()
@@ -1392,8 +2635,36 @@ def on_add_to_archive_error(self, error_message):
)
self.add_progress_label.setText("Archive file addition failed.")
+ def cancel_add_to_archive(self):
+ """Cancel the add to archive process"""
+ if self.confirm_cancel("添加到归档"):
+ self.log_cancel("Add to Archive")
+ # Stop the worker
+ if self.add_to_zip_worker:
+ self.add_to_zip_worker.stop()
+
+ def on_add_to_archive_canceled(self):
+ """Handle add to archive canceled"""
+ # 使用强制线程清理方法
+ self._force_cleanup_add_thread()
+
+ # Update progress and status
+ self.add_progress.setValue(0)
+ self.add_progress_label.setText("Add to archive canceled")
+
+ # Show cancel notification
+ self._show_info_bar(
+ title='Canceled',
+ content='Add to archive canceled by user',
+ duration=2000
+ )
+
def _force_cleanup_add_thread(self):
"""强制清理添加到归档的线程,确保完全终止"""
+ # Reset button states
+ self.add_button.setEnabled(True)
+ self.add_cancel_button.setEnabled(False)
+
if self.add_to_zip_worker_thread:
if self.add_to_zip_worker_thread.isRunning():
# 先尝试正常退出
@@ -1475,6 +2746,10 @@ def start_list_archive_contents(self):
self.contents_listbox.clear()
self.contents_listbox.addItem("Listing contents...")
+
+ # Enable cancel button and disable list button during operation
+ self.list_button.setEnabled(False)
+ self.list_cancel_button.setEnabled(True)
# Reset password protection status
self.is_password_protected = False
@@ -1490,6 +2765,7 @@ def start_list_archive_contents(self):
self.list_zip_worker.finished.connect(self.on_list_zip_finished)
self.list_zip_worker.conversion_error.connect(self.on_list_archive_error)
self.list_zip_worker.password_required.connect(self.on_password_required)
+ self.list_zip_worker.canceled.connect(self.on_list_archive_canceled)
self.list_zip_worker_thread.started.connect(self.list_zip_worker.run)
self.list_zip_worker_thread.start()
@@ -1678,8 +2954,36 @@ def on_list_archive_error(self, error_message):
# Update password status for other errors
self.update_password_status_list(False, "Error Listing Contents")
+ def cancel_list_archive_contents(self):
+ """Cancel the list archive contents process"""
+ if self.confirm_cancel("列出内容"):
+ self.log_cancel("List Contents")
+ # Stop the worker
+ if self.list_zip_worker:
+ self.list_zip_worker.stop()
+
+ def on_list_archive_canceled(self):
+ """Handle list archive contents canceled"""
+ # 使用强制线程清理方法
+ self._force_cleanup_list_thread()
+
+ # Update listbox and status
+ self.contents_listbox.clear()
+ self.contents_listbox.addItem("List contents canceled")
+
+ # Show cancel notification
+ self._show_info_bar(
+ title='Canceled',
+ content='List contents canceled by user',
+ duration=2000
+ )
+
def _force_cleanup_list_thread(self):
"""强制清理列出归档内容的线程,确保完全终止"""
+ # Reset button states
+ self.list_button.setEnabled(True)
+ self.list_cancel_button.setEnabled(False)
+
if self.list_zip_worker_thread:
if self.list_zip_worker_thread.isRunning():
# 先尝试正常退出
diff --git a/build_project_arm64.py b/build_project_arm64.py
index d2127b2..4852cc9 100644
--- a/build_project_arm64.py
+++ b/build_project_arm64.py
@@ -2,7 +2,7 @@
import clean
import buildzip
from setup_ccache import setup_ccache
-target="com.pyquick.converter"
+target="com.intsant.converter"
from patch import enable
from plistedit import add_utf_info
if __name__ == "__main__":
diff --git a/build_project_base.py b/build_project_base.py
index 58570da..9b9bb6b 100644
--- a/build_project_base.py
+++ b/build_project_base.py
@@ -53,7 +53,7 @@ def compile_gui():
"--include-package=image_converter",
"--include-package=support",
"--include-package=update", # 添加update包以确保所有更新功能正常工作
- "--macos-signed-app-name=com.pyquick.converter",
+ "--macos-signed-app-name=com.intsant.converter",
"--enable-plugin=pyside6",
"--prefer-source-code",
"--output-dir=dist", # Output directory
@@ -227,7 +227,7 @@ def create_macos_app_bundle(current_dir, dist_path):
CFBundleIconFile
AppIcon.icns
CFBundleIdentifier
- com.pyquick.converter
+ com.intsant.converter
CFBundleInfoDictionaryVersion
6.0
CFBundleName
diff --git a/debug/debug_gui.py b/debug/debug_gui.py
index 9e049f9..9c47a27 100644
--- a/debug/debug_gui.py
+++ b/debug/debug_gui.py
@@ -5,187 +5,231 @@
import os
import sys
-from PySide6.QtWidgets import (
- QWidget, QVBoxLayout, QHBoxLayout, QLabel,
- QCheckBox, QGroupBox, QSpacerItem, QSizePolicy,
- QPushButton, QTextBrowser
-)
+from PySide6.QtWidgets import QWidget, QVBoxLayout, QHBoxLayout
from PySide6.QtCore import QSettings, Qt
from PySide6.QtGui import QFont
from qfluentwidgets import (
- CheckBox, TextBrowser, IndeterminateProgressBar,
- ProgressBar, PrimaryPushButton
+ SettingCardGroup, SwitchSettingCard, PushSettingCard, PrimaryPushSettingCard,
+ FluentIcon, BodyLabel, CaptionLabel, TextBrowser, InfoBar, InfoBarPosition,
+ setCustomStyleSheet, HeaderCardWidget, SingleDirectionScrollArea
)
import sys
import os
-from qfluentwidgets import *
-# Add support directory to path for debug_logger import
sys.path.insert(0, os.path.join(os.path.dirname(__file__), '..', 'support'))
from support.debug_logger import DebugLogger
-class DebugSettingsWidget(QWidget):
- """Debug Settings GUI Widget"""
-
- def __init__(self):
- super().__init__()
- self.setWindowTitle("Debug Settings")
- self.debug_logger = DebugLogger()
- self.settings = QSettings("MyCompany", "ConverterApp")
+class DebugStatusCard(HeaderCardWidget):
+ """Debug status card using HeaderCardWidget"""
+
+ def __init__(self, parent=None):
+ super().__init__(parent)
+ self.setTitle('Debug Status')
- self.init_ui()
- self.load_settings()
- self.connect_auto_save_signals()
-
- def init_ui(self):
- """Initialize the UI"""
- main_layout = QVBoxLayout(self)
- main_layout.setContentsMargins(25, 25, 25, 25)
- main_layout.setSpacing(20)
-
- # Debug Settings Group
- debug_group = QGroupBox("Debug Settings")
- debug_layout = QVBoxLayout()
- debug_layout.setContentsMargins(25, 25, 25, 25)
- debug_layout.setSpacing(20)
-
- # Add top spacing
- debug_layout.addSpacerItem(QSpacerItem(0, 15, QSizePolicy.Policy.Minimum, QSizePolicy.Policy.Fixed))
-
- # Debug Mode Toggle
- self.debug_enabled_checkbox = CheckBox("Enable Debug Mode")
- self.debug_enabled_checkbox.setMinimumHeight(60)
- # Signal connection will be done in connect_auto_save_signals
- debug_layout.addWidget(self.debug_enabled_checkbox)
-
- # Enhanced Logging
- self.enhanced_logging_checkbox = CheckBox("Enable Enhanced Logging (with module info)")
- self.enhanced_logging_checkbox.setMinimumHeight(60)
- # Signal connection will be done in connect_auto_save_signals
- debug_layout.addWidget(self.enhanced_logging_checkbox)
-
- # Log File Info
- log_info_layout = QHBoxLayout()
- log_info_layout.addWidget(QLabel("Log Directory:"))
- self.log_dir_label = QLabel("~/.converter/log/")
- self.log_dir_label.setWordWrap(True)
- self.log_dir_label.setStyleSheet("QLabel { color: #666; }")
- log_info_layout.addWidget(self.log_dir_label)
- log_info_layout.addStretch()
- debug_layout.addLayout(log_info_layout)
-
- # Status Label
- self.status_label = QLabel("Debug mode is currently disabled.")
- self.status_label.setMinimumHeight(60)
- self.status_label.setMinimumWidth(550)
- self.status_label.setWordWrap(True)
- self.status_label.setStyleSheet("""
- QLabel {
- padding: 8px;
- background-color: #f8f9fa;
- border-radius: 16px;
- border: 1px solid #e9ecef;
- }
- """)
- debug_layout.addWidget(self.status_label)
-
- # Log Preview Area
- self.log_preview_browser = TextBrowser()
- self.log_preview_browser.setMinimumHeight(150)
- self.log_preview_browser.setPlaceholderText("Log preview will appear here when debug mode is enabled...")
- debug_layout.addWidget(self.log_preview_browser)
-
- # Buttons Container
- button_container = QHBoxLayout()
- button_container.setSpacing(15)
- from con import CON
- # Test Debug Button
- self.test_debug_button = PrimaryPushButton("Test Debug Output")
- setCustomStyleSheet(self.test_debug_button, CON.qss_debug, CON.qss_debug)
- self.test_debug_button.setFixedSize(180, 60)
- self.test_debug_button.clicked.connect(self.test_debug_output)
-
- # View Logs Button
- self.view_logs_button = PrimaryPushButton("View Log Directory")
- self.view_logs_button.setFixedSize(180, 60)
- setCustomStyleSheet(self.view_logs_button, CON.qss_debug, CON.qss_debug)
- self.view_logs_button.clicked.connect(self.view_log_directory)
-
- # Clear Logs Button
- self.clear_logs_button = PrimaryPushButton("Clear Logs")
- self.clear_logs_button.setFixedSize(180, 60)
- setCustomStyleSheet(self.clear_logs_button, CON.qss_debug, CON.qss_debug)
- self.clear_logs_button.clicked.connect(self.clear_logs)
-
- button_container.addStretch()
- button_container.addWidget(self.test_debug_button)
- button_container.addWidget(self.view_logs_button)
- button_container.addWidget(self.clear_logs_button)
- button_container.addStretch()
- debug_layout.addLayout(button_container)
-
- # Add bottom spacing
- debug_layout.addSpacerItem(QSpacerItem(0, 15, QSizePolicy.Policy.Minimum, QSizePolicy.Policy.Fixed))
-
- debug_group.setLayout(debug_layout)
- main_layout.addWidget(debug_group)
-
- self.setLayout(main_layout)
-
- # Set button fonts
- font = QFont()
- font.setPointSize(12)
- font.setBold(True)
- self.test_debug_button.setFont(font)
- self.view_logs_button.setFont(font)
- self.clear_logs_button.setFont(font)
-
- def load_settings(self):
- """Load current debug settings"""
- debug_enabled = bool(self.settings.value("debug_enabled", False, type=bool))
- enhanced_logging = bool(self.settings.value("enhanced_logging", True, type=bool))
+ from qfluentwidgets import IconWidget, HyperlinkLabel
- self.debug_enabled_checkbox.setChecked(debug_enabled)
- self.enhanced_logging_checkbox.setChecked(enhanced_logging)
+ # Create status icon
+ self.statusIcon = IconWidget(FluentIcon.INFO, self)
+ self.statusIcon.setFixedSize(16, 16)
- self.update_status_label()
-
- def update_status_label(self):
- """Update the status label based on current settings"""
- debug_enabled = self.debug_enabled_checkbox.isChecked()
- enhanced_logging = self.enhanced_logging_checkbox.isChecked()
+ # Create status label
+ self.statusLabel = BodyLabel('Debug mode is currently disabled.', self)
+
+ # Create detail button
+ self.detailButton = HyperlinkLabel('View Logs', self)
+ self.detailButton.clicked.connect(self.view_logs)
+
+ # Setup layout
+ self.vBoxLayout = QVBoxLayout()
+ self.hBoxLayout = QHBoxLayout()
+
+ self.hBoxLayout.setSpacing(10)
+ self.vBoxLayout.setSpacing(16)
+ self.hBoxLayout.setContentsMargins(0, 0, 0, 0)
+ self.vBoxLayout.setContentsMargins(0, 0, 0, 0)
+
+ self.hBoxLayout.addWidget(self.statusIcon)
+ self.hBoxLayout.addWidget(self.statusLabel)
+ self.vBoxLayout.addLayout(self.hBoxLayout)
+ self.vBoxLayout.addWidget(self.detailButton)
+ self.viewLayout.addLayout(self.vBoxLayout)
+
+ self.update_status(False, False)
+
+ def update_status(self, debug_enabled, enhanced_logging):
+ """Update status display based on debug settings"""
if debug_enabled:
- status_text = "✓ Debug mode is ENABLED"
+ from qfluentwidgets import InfoBarIcon
+ self.statusIcon.setIcon(InfoBarIcon.SUCCESS)
+ status_text = " Debug mode is ENABLED"
if enhanced_logging:
status_text += " with enhanced logging"
status_text += ". All debug output is being logged to ~/.converter/log/"
- self.status_label.setStyleSheet("""
- QLabel {
- padding: 8px;
- background-color: #e8f5e8;
- border-radius: 16px;
- border: 1px solid #d4edda;
+ self.statusLabel.setStyleSheet("""
+ BodyLabel {
color: #155724;
}
""")
else:
+ self.statusIcon.setIcon(FluentIcon.INFO)
status_text = "Debug mode is DISABLED. Only basic console output will be shown."
- self.status_label.setStyleSheet("""
- QLabel {
- padding: 8px;
- background-color: #f8f9fa;
- border-radius: 16px;
- border: 1px solid #e9ecef;
+ self.statusLabel.setStyleSheet("""
+ BodyLabel {
color: #6c757d;
}
""")
- self.status_label.setText(status_text)
-
+ self.statusLabel.setText(status_text)
+
+ def view_logs(self):
+ """View log directory"""
+ log_dir = os.path.expanduser("~/.converter/log")
+
+ if not os.path.exists(log_dir):
+ os.makedirs(log_dir, exist_ok=True)
+
+ try:
+ if sys.platform == "darwin": # macOS
+ os.system(f"open '{log_dir}'")
+ elif sys.platform == "win32": # Windows
+ os.system(f"explorer '{log_dir}'")
+ else: # Linux
+ os.system(f"xdg-open '{log_dir}'")
+ except Exception as e:
+ print(f"Failed to open log directory: {e}")
+class DebugSettingsWidget(QWidget):
+ """Debug settings widget using qfluentwidgets SettingCard components"""
+
+ def __init__(self, parent=None):
+ super().__init__(parent)
+ self.debug_logger = DebugLogger()
+ self.settings = QSettings("MyCompany", "ConverterApp")
+ self.setup_ui()
+ self.connect_signals()
+ self.load_settings()
+
+ def setup_ui(self):
+ """Setup the UI layout"""
+ main_layout = QVBoxLayout(self)
+ main_layout.setContentsMargins(0, 0, 0, 0)
+ main_layout.setSpacing(0)
+
+ # Create scroll area
+ self.scroll_area = SingleDirectionScrollArea(orient=Qt.Orientation.Vertical)
+ self.scroll_area.setWidgetResizable(True)
+ self.scroll_area.enableTransparentBackground()
+
+ # Create scroll content widget
+ scroll_content = QWidget()
+ scroll_content.setObjectName("scroll_content")
+ scroll_layout = QVBoxLayout(scroll_content)
+ scroll_layout.setContentsMargins(30, 30, 30, 30)
+ scroll_layout.setSpacing(20)
+
+ # Create setting card groups
+ self.debug_group = SettingCardGroup("Debug Configuration", scroll_content)
+ self.log_group = SettingCardGroup("Log Management", scroll_content)
+
+ # Create debug setting cards
+ self.debug_enabled_card = SwitchSettingCard(
+ FluentIcon.DEVELOPER_TOOLS,
+ "Enable Debug Mode",
+ "Enable debug logging for troubleshooting",
+ parent=self.debug_group
+ )
+
+ self.enhanced_logging_card = SwitchSettingCard(
+ FluentIcon.DOCUMENT,
+ "Enhanced Logging",
+ "Include module information in debug output",
+ parent=self.debug_group
+ )
+
+ # Create log management cards
+ self.test_debug_card = PrimaryPushSettingCard(
+ "Test Debug Output",
+ FluentIcon.CODE,
+ "Generate test debug messages"
+ )
+ self.test_debug_card.clicked.connect(self.test_debug_output)
+
+ self.view_logs_card = PushSettingCard(
+ "View Log Directory",
+ FluentIcon.FOLDER,
+ "Open log folder in file explorer"
+ )
+ self.view_logs_card.clicked.connect(self.view_log_directory)
+
+ self.clear_logs_card = PushSettingCard(
+ "Clear Logs",
+ FluentIcon.DELETE,
+ "Remove all log files"
+ )
+ self.clear_logs_card.clicked.connect(self.clear_logs)
+
+ # Add cards to groups
+ self.debug_group.addSettingCards([
+ self.debug_enabled_card,
+ self.enhanced_logging_card
+ ])
+
+ self.log_group.addSettingCards([
+ self.test_debug_card,
+ self.view_logs_card,
+ self.clear_logs_card
+ ])
+
+ # Add log info and preview to log group
+ log_info_label = CaptionLabel("Log files are stored in ~/.converter/log/")
+ self.log_group.vBoxLayout.addWidget(log_info_label)
+
+ # Debug status card
+ self.debug_status_card = DebugStatusCard(scroll_content)
+
+ # Log preview area
+ self.log_preview_browser = TextBrowser()
+ self.log_preview_browser.setMinimumHeight(150)
+ self.log_preview_browser.setPlaceholderText("Log preview will appear here when debug mode is enabled...")
+ self.log_group.vBoxLayout.addWidget(self.log_preview_browser)
+
+ # Add groups to scroll layout
+ scroll_layout.addWidget(self.debug_group)
+ scroll_layout.addWidget(self.debug_status_card)
+ scroll_layout.addWidget(self.log_group)
+ scroll_layout.addStretch()
+
+ # Set scroll content
+ self.scroll_area.setWidget(scroll_content)
+
+ # Add scroll area to main layout
+ main_layout.addWidget(self.scroll_area)
+
+ def connect_signals(self):
+ """Connect signals for auto-save"""
+ self.debug_enabled_card.checkedChanged.connect(self.on_debug_setting_changed)
+ self.enhanced_logging_card.checkedChanged.connect(self.on_enhanced_logging_changed)
+
+ def load_settings(self):
+ """Load current debug settings"""
+ self.debug_enabled_card.setChecked(bool(self.settings.value("debug_enabled", False, type=bool)))
+ self.enhanced_logging_card.setChecked(bool(self.settings.value("enhanced_logging", True, type=bool)))
+
+ # Disable enhanced logging checkbox if debug mode is not enabled
+ self.enhanced_logging_card.setEnabled(self.debug_enabled_card.isChecked())
+
+ self.update_status_label()
+
+ def update_status_label(self):
+ """Update the status label based on current settings"""
+ debug_enabled = self.debug_enabled_card.isChecked()
+ enhanced_logging = self.enhanced_logging_card.isChecked()
+
+ self.debug_status_card.update_status(debug_enabled, enhanced_logging)
+
def test_debug_output(self):
"""Test debug output functionality"""
self.debug_logger.log_debug("This is a test debug message from Debug Settings GUI")
@@ -208,9 +252,9 @@ def test_debug_output(self):
self.log_preview_browser.setPlainText(preview)
except Exception as e:
self.log_preview_browser.setPlainText(f"Error reading log file: {e}")
-
+
def view_log_directory(self):
- """Open the log directory in file explorer"""
+ """Open log directory in file explorer"""
log_dir = os.path.expanduser("~/.converter/log")
if not os.path.exists(log_dir):
@@ -225,7 +269,7 @@ def view_log_directory(self):
os.system(f"xdg-open '{log_dir}'")
except Exception as e:
self.debug_logger.log_error(f"Failed to open log directory: {e}")
-
+
def clear_logs(self):
"""Clear all log files"""
log_dir = os.path.expanduser("~/.converter/log")
@@ -241,24 +285,25 @@ def clear_logs(self):
except Exception as e:
self.log_preview_browser.setPlainText(f"Error clearing logs: {e}")
- # Use print instead of log_error to avoid potential recursion
print(f"ERROR: Failed to clear logs: {e}")
else:
self.log_preview_browser.setPlainText("Log directory does not exist.")
- def connect_auto_save_signals(self):
- """Connect all UI controls to auto-save functionality"""
- # Connect checkboxes to auto-save - the toggle methods already call auto_save_settings
- # But we also add a direct connection to ensure the signal is triggered
- self.debug_enabled_checkbox.stateChanged.connect(self.on_debug_setting_changed)
- self.enhanced_logging_checkbox.stateChanged.connect(self.on_enhanced_logging_changed)
-
def on_debug_setting_changed(self):
"""Handle debug setting change and trigger auto-save"""
- debug_enabled = self.debug_enabled_checkbox.isChecked()
+ debug_enabled = self.debug_enabled_card.isChecked()
self.settings.setValue("debug_enabled", debug_enabled)
self.settings.sync()
+ # Enable/disable enhanced logging checkbox based on debug mode
+ self.enhanced_logging_card.setEnabled(debug_enabled)
+
+ # If debug mode is disabled, also disable enhanced logging
+ if not debug_enabled:
+ self.enhanced_logging_card.setChecked(False)
+ self.settings.setValue("enhanced_logging", False)
+ self.settings.sync()
+
# Reinitialize debug logger with new settings
self.debug_logger = DebugLogger()
@@ -268,13 +313,10 @@ def on_debug_setting_changed(self):
self.debug_logger.log_info("Debug mode enabled via GUI (auto-save)")
else:
self.debug_logger.log_info("Debug mode disabled via GUI (auto-save)")
-
- # Emit auto-save signal to parent if exists
- self.auto_save_settings()
def on_enhanced_logging_changed(self):
"""Handle enhanced logging setting change and trigger auto-save"""
- enhanced_logging = self.enhanced_logging_checkbox.isChecked()
+ enhanced_logging = self.enhanced_logging_card.isChecked()
self.settings.setValue("enhanced_logging", enhanced_logging)
self.settings.sync()
@@ -284,17 +326,6 @@ def on_enhanced_logging_changed(self):
self.debug_logger.log_info("Enhanced logging enabled via GUI (auto-save)")
else:
self.debug_logger.log_info("Enhanced logging disabled via GUI (auto-save)")
-
- # Emit auto-save signal to parent if exists
- self.auto_save_settings()
-
- def auto_save_settings(self):
- """Auto-save settings"""
- try:
- # Force sync settings to disk
- self.settings.sync()
- except Exception as e:
- print(f"Error in auto_save_settings: {e}")
if __name__ == "__main__":
@@ -307,4 +338,4 @@ def auto_save_settings(self):
widget.resize(800, 600)
widget.show()
- sys.exit(app.exec())
\ No newline at end of file
+ sys.exit(app.exec())
diff --git a/image_converter.py b/image_converter.py
index 4f77733..fd79a42 100644
--- a/image_converter.py
+++ b/image_converter.py
@@ -1,7 +1,7 @@
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
"""
-PNG to ICNS Converter with wxPython GUI
+PNG to ICNS Converter with PySide6
This script provides a graphical interface for converting PNG images to ICNS format.
"""
@@ -11,12 +11,13 @@
import os
import threading
import subprocess
+from concurrent.futures import ThreadPoolExecutor, as_completed
from PIL import Image
from PySide6.QtWidgets import (
QApplication, QMainWindow, QWidget, QVBoxLayout, QHBoxLayout,
- QLabel,
+ QLabel, QComboBox, QSpinBox, QListWidget,
QFileDialog, QMessageBox, QTabWidget, QGroupBox, QSizePolicy,
- QTreeWidgetItem, QFrame, QScrollArea, QListWidgetItem
+ QTreeWidgetItem, QFrame,QListWidgetItem
)
from PySide6.QtGui import QPixmap, QIcon, QFont, QImage, QPalette
from PySide6.QtCore import Qt, QSize, Signal, QObject, QThread
@@ -28,8 +29,10 @@
# Add the current directory to Python path to import convert module
sys.path.append(os.path.dirname(os.path.abspath(__file__)))
from support import convert
+from support.GUI.image_support import DropZoneWidget, DirectoryDropLineEdit, PreviewTab, ThumbnailGridWidget
from con import CON
+
class ConversionWorker(QObject):
finished = Signal()
progress_updated = Signal(str, int)
@@ -53,6 +56,7 @@ def __init__(self, input_path, output_path, output_format, min_size_param=None,
def run(self):
try:
if self.output_format == "icns":
+ # For icns format, use positional arguments with correct order
convert.convert_image(
self.input_path,
self.output_path,
@@ -63,21 +67,262 @@ def run(self):
progress_callback=self._update_progress_callback
)
else:
+ # For non-icns formats, use keyword arguments for clarity and to avoid parameter order issues
convert.convert_image(
- self.input_path,
- self.output_path,
- self.output_format,
+ input_path=self.input_path,
+ output_path=self.output_path,
+ output_format=self.output_format,
quality=self.quality,
progress_callback=self._update_progress_callback
)
self.finished.emit()
except Exception as e:
- self.conversion_error.emit(str(e))
-
- def _update_progress_callback(self, message, percentage):
+ error_msg = f"Conversion error: {str(e)}"
+ self.conversion_error.emit(error_msg)
+ print(f"[ERROR] ConversionWorker: {error_msg}")
+
+ def _update_progress_callback(self, *args):
+ """Handle variable number of arguments from progress_callback
+ Can be called with:
+ - (message, percentage)
+ - (current, total, message)
+ - (message, percentage, extra)
+ """
+ # Determine the arguments format
+ if len(args) == 2:
+ # Format: (message, percentage)
+ message, percentage = args
+ elif len(args) == 3:
+ # Format: (current, total, message) or (message, percentage, extra)
+ if isinstance(args[0], (int, float)) and isinstance(args[1], (int, float)):
+ # Format: (current, total, message)
+ current, total, message = args
+ percentage = int((current / total) * 100) if total > 0 else 0
+ else:
+ # Format: (message, percentage, extra)
+ message, percentage, _ = args
+ else:
+ # Unexpected format, use default values
+ message = "Processing..." if args else "Unknown progress"
+ percentage = 0
+
self.progress_updated.emit(message, percentage)
+class BatchConversionWorker(QObject):
+ finished = Signal()
+ progress_updated = Signal(int, int, str, int) # current_index, total_count, current_file, percentage
+ file_processed = Signal(str, str, str, bool, str) # filename, input_path, output_path, success, error_message
+ batch_error = Signal(str)
+ total_progress_updated = Signal(int) # overall progress percentage
+
+ def __init__(self, input_paths, output_dir, output_format, min_size_param=None, max_size_param=None, quality_param=None,
+ preserve_folder_structure=False, prefix="", suffix="", auto_detect_max_size=False):
+ super().__init__()
+ self.input_paths = input_paths
+ self.output_dir = output_dir
+ self.output_format = output_format
+ self.quality = int(quality_param) if quality_param is not None else 85
+ self.is_cancelled = False
+ self.preserve_folder_structure = preserve_folder_structure
+ self.prefix = prefix
+ self.suffix = suffix
+ self.auto_detect_max_size = auto_detect_max_size
+
+ if output_format == "icns":
+ self.min_size = int(min_size_param) if min_size_param is not None else 16
+ self.max_size = int(max_size_param) if max_size_param is not None else None
+ else:
+ self.min_size = None
+ self.max_size = None
+
+ def cancel(self):
+ """Cancel the batch conversion process"""
+ self.is_cancelled = True
+
+ def run(self):
+ try:
+ total_files = len(self.input_paths)
+ if total_files == 0:
+ self.finished.emit()
+ return
+
+ # Get common parent directory if preserving folder structure
+ common_parent = None
+ if self.preserve_folder_structure and self.input_paths:
+ # Get all directories
+ directories = [os.path.dirname(path) for path in self.input_paths]
+ if directories:
+ # Find common parent directory
+ common_parent = os.path.commonpath(directories)
+
+ # Calculate optimal number of threads (use CPU cores * 2 for I/O bound tasks)
+ max_workers = min(16, os.cpu_count() * 2)
+
+ # Track processed files and progress
+ processed_files = 0
+
+ # Create a list to hold conversion tasks
+ conversion_tasks = []
+
+ # Prepare all conversion tasks
+ for i, input_path in enumerate(self.input_paths):
+ filename = os.path.basename(input_path)
+ name_without_ext = os.path.splitext(filename)[0]
+
+ # Apply prefix and suffix
+ output_filename = f"{self.prefix}{name_without_ext}{self.suffix}.{self.output_format.lower()}"
+
+ # Determine output path based on folder structure option
+ if self.preserve_folder_structure and common_parent:
+ # Get relative path from common parent
+ relative_dir = os.path.relpath(os.path.dirname(input_path), common_parent)
+ output_path = os.path.join(self.output_dir, relative_dir, output_filename)
+ # Create directories if they don't exist
+ os.makedirs(os.path.dirname(output_path), exist_ok=True)
+ else:
+ # Create "converted" subdirectory in the output directory
+ converted_dir = os.path.join(self.output_dir, "converted")
+ os.makedirs(converted_dir, exist_ok=True)
+ output_path = os.path.join(converted_dir, output_filename)
+
+ # Add to conversion tasks
+ conversion_tasks.append({
+ 'index': i,
+ 'input_path': input_path,
+ 'output_path': output_path,
+ 'filename': filename,
+ 'total_files': total_files
+ })
+
+ # Use ThreadPoolExecutor for concurrent conversion
+ with ThreadPoolExecutor(max_workers=max_workers) as executor:
+ # Submit all tasks
+ future_to_task = {}
+ for task in conversion_tasks:
+ if self.is_cancelled:
+ error_msg = "Batch conversion was cancelled"
+ self.batch_error.emit(error_msg)
+ print(f"[ERROR] BatchConversionWorker: {error_msg}")
+ return
+
+ try:
+ future = executor.submit(self._convert_single_file, task)
+ future_to_task[future] = task
+ except Exception as e:
+ error_msg = f"Error submitting task for {task['filename']}: {str(e)}"
+ self.batch_error.emit(error_msg)
+ print(f"[ERROR] BatchConversionWorker: {error_msg}")
+ # Continue with other tasks instead of failing completely
+
+ # Process completed tasks
+ for future in as_completed(future_to_task):
+ if self.is_cancelled:
+ error_msg = "Batch conversion was cancelled"
+ self.batch_error.emit(error_msg)
+ print(f"[ERROR] BatchConversionWorker: {error_msg}")
+ executor.shutdown(wait=False, cancel_futures=True)
+ return
+
+ task = future_to_task[future]
+ try:
+ success, message = future.result()
+ # Signal that this file was processed
+ self.file_processed.emit(task['filename'], task['input_path'], task['output_path'], success, message if not success else "")
+ except Exception as e:
+ # Signal that this file failed
+ error_msg = f"Error processing {task['filename']}: {str(e)}"
+ self.file_processed.emit(task['filename'], task['input_path'], task['output_path'], False, error_msg)
+ print(f"[ERROR] BatchConversionWorker: {error_msg}")
+
+ # Update processed count and overall progress
+ processed_files += 1
+ overall_progress = int((processed_files / total_files) * 100)
+ self.total_progress_updated.emit(overall_progress)
+
+ # Final progress update
+ self.total_progress_updated.emit(100)
+ self.finished.emit()
+
+ except Exception as e:
+ error_msg = f"Batch conversion error: {str(e)}"
+ self.batch_error.emit(error_msg)
+ print(f"[ERROR] BatchConversionWorker: {error_msg}")
+
+ def _convert_single_file(self, task):
+ """Convert a single file (thread-safe)"""
+ if self.is_cancelled:
+ raise Exception("Batch conversion was cancelled")
+
+ # Update file-specific progress
+ self.progress_updated.emit(task['index']+1, task['total_files'], task['filename'], 0)
+
+ # Create a progress callback for this specific file
+ def progress_callback(*args):
+ """Handle variable number of arguments from progress_callback
+ Can be called with:
+ - (message, percentage)
+ - (current, total, message)
+ - (message, percentage, extra)
+ """
+ # Determine the arguments format
+ if len(args) == 2:
+ # Format: (message, percentage)
+ message, percentage = args
+ elif len(args) == 3:
+ # Format: (current, total, message) or (message, percentage, extra)
+ if isinstance(args[0], (int, float)) and isinstance(args[1], (int, float)):
+ # Format: (current, total, message)
+ current, total, message = args
+ percentage = int((current / total) * 100) if total > 0 else 0
+ else:
+ # Format: (message, percentage, extra)
+ message, percentage, _ = args
+ else:
+ # Unexpected format, use default values
+ message = "Processing..." if args else "Unknown progress"
+ percentage = 0
+
+ # Update the file-specific progress
+ self.progress_updated.emit(task['index']+1, task['total_files'], task['filename'], percentage)
+
+ # Perform the conversion
+ if self.output_format == "icns":
+ # Auto-detect max size if enabled
+ current_max_size = int(self.max_size) if self.max_size is not None else None
+ if self.auto_detect_max_size:
+ try:
+ # Get image dimensions
+ width, height = convert.get_image_info(task['input_path'])
+ # Use the minimum of width and height as the max size (since ICNS uses square sizes)
+ current_max_size = min(width, height)
+ except Exception as e:
+ # Fall back to default if auto-detect fails
+ print(f"[WARNING] Failed to auto-detect max size for {task['filename']}: {str(e)}")
+ current_max_size = int(self.max_size) if self.max_size is not None else None
+
+ success, message = convert.convert_image(
+ task['input_path'],
+ task['output_path'],
+ self.output_format,
+ int(self.min_size) if self.min_size is not None else 16,
+ current_max_size,
+ quality=self.quality,
+ progress_callback=progress_callback
+ )
+ else:
+ # For non-icns formats, use keyword arguments for clarity and to avoid parameter order issues
+ success, message = convert.convert_image(
+ input_path=task['input_path'],
+ output_path=task['output_path'],
+ output_format=self.output_format,
+ quality=self.quality,
+ progress_callback=progress_callback
+ )
+
+ return success, message
+
+
class ICNSConverterGUI(QMainWindow):
def _load_qss_file(self, filename):
@@ -139,12 +384,13 @@ def closeEvent(self, e):
def _apply_theme(self, is_dark_mode):
if is_dark_mode:
self.setStyleSheet(self.DARK_QSS)
-
-
else:
self.setStyleSheet(self.LIGHT_QSS)
-
+ # Update DropZoneWidget theme if it exists
+ if hasattr(self, 'drop_zone') and self.drop_zone:
+ self.drop_zone.set_theme(is_dark_mode)
+
# Update success view theme if it exists and is visible
if hasattr(self, 'success_widget') and self.success_widget and self.success_widget.isVisible():
self._apply_success_theme()
@@ -171,10 +417,12 @@ def load_settings(self):
auto_preview_val = settings.value("image_converter/auto_preview", True, type=bool)
remember_path_val = settings.value("image_converter/remember_path", True, type=bool)
completion_notify_val = settings.value("image_converter/completion_notify", True, type=bool)
+ task_mode_val = settings.value("task_mode", False, type=bool)
self.auto_preview = bool(auto_preview_val) if auto_preview_val is not None else True
self.remember_path = bool(remember_path_val) if remember_path_val is not None else True
self.completion_notify = bool(completion_notify_val) if completion_notify_val is not None else True
+ self.task_mode = bool(task_mode_val) if task_mode_val is not None else False
# Load remembered paths if setting is enabled
if self.remember_path:
@@ -201,8 +449,10 @@ def load_settings(self):
self.icns_method_combo.setCurrentText(str(self.icns_method))
if hasattr(self, 'overwrite_confirm_check'):
self.overwrite_confirm_check.setChecked(bool(self.overwrite_confirm))
+ if hasattr(self, 'task_mode_check'):
+ self.task_mode_check.setChecked(bool(self.task_mode))
- print(f"Settings loaded: min_size={self.min_size}, max_size={self.max_size}, output_format={self.output_format}")
+ print(f"Settings loaded: min_size={self.min_size}, max_size={self.max_size}, output_format={self.output_format}, task_mode={self.task_mode}")
def save_settings(self):
"""Save settings to QSettings"""
@@ -227,6 +477,12 @@ def save_settings(self):
settings.setValue("image_converter/remember_path", self.remember_path)
settings.setValue("image_converter/completion_notify", self.completion_notify)
+ # Save task mode setting
+ if hasattr(self, 'task_mode_check'):
+ settings.setValue("task_mode", self.task_mode_check.isChecked())
+ else:
+ settings.setValue("task_mode", self.task_mode)
+
# Save remembered paths if setting is enabled
if self.remember_path:
if hasattr(self, 'last_input_dir'):
@@ -260,6 +516,18 @@ def init_variables(self, reset_all=False):
self.icns_method = "iconutil (Recommended)"
self.overwrite_confirm = True
+ # Batch conversion variables
+ self.batch_files = []
+ self.batch_converting = False
+ self.batch_worker = None
+ self.batch_thread = None
+ self.batch_current_index = 0
+ self.batch_current_file = ""
+ self.batch_success_count = 0
+ self.batch_failed_count = 0
+ self.batch_canceled = False
+ self.batch_results = [] # Store conversion results
+
# Only reset interface behavior settings if explicitly requested
if reset_all or not hasattr(self, 'auto_preview'):
# Interface behavior settings - prioritize reading from launcher settings
@@ -332,6 +600,7 @@ def setup_ui(self):
self.create_widgets()
self.create_success_view()
+ self.create_batch_success_view()
self.create_history_tab()
self.show_main_view()
@@ -355,11 +624,44 @@ def create_widgets(self):
# Main converter tab
self.converter_tab = QWidget()
- self.tab_widget.addTab(self.converter_tab, "Converter")
+ self.tab_widget.addTab(self.converter_tab, "Signle Converter")
# Setup main converter content
self.setup_converter_tab()
+ # Batch converter tab
+ self.batch_converter_tab = QWidget()
+ self.tab_widget.addTab(self.batch_converter_tab, "Batch Converter")
+
+ # Setup batch converter content
+ self.setup_batch_converter_tab()
+
+ # Preview tab - for image preview functionality
+ self.preview_tab = PreviewTab()
+ self.tab_widget.addTab(self.preview_tab, "Preview")
+
+ # Connect drop signals to update preview tab
+ self._connect_preview_signals()
+
+ def _connect_preview_signals(self):
+ """Connect signals to update preview tab"""
+ # Connect drop zone signals to preview tab
+ self.drop_zone.filesDropped.connect(self.preview_tab.show_multiple_previews)
+ self.drop_zone.folderDropped.connect(self.preview_tab.show_multiple_previews)
+
+ # Connect input text change to preview tab
+ self.input_text.textChanged.connect(self.on_input_text_changed)
+
+ def on_input_text_changed(self):
+ """Handle input text change to update preview"""
+ input_path = self.input_text.text().strip()
+ if input_path and os.path.exists(input_path):
+ # Single file change
+ self.preview_tab.show_single_preview(input_path)
+ else:
+ # Clear preview if invalid path
+ self.preview_tab.clear_previews()
+
def setup_converter_tab(self):
"""Setup the main converter tab content"""
converter_layout = QVBoxLayout(self.converter_tab)
@@ -415,19 +717,6 @@ def setup_converter_tab(self):
output_layout.addWidget(output_button)
file_ops_group_layout.addLayout(output_layout)
- # Image Preview (moved from right panel)
- preview_group_box = QGroupBox("Image Preview")
- preview_group_layout = QVBoxLayout(preview_group_box)
- preview_group_layout.setContentsMargins(10, 25, 10, 10)
- left_layout.addWidget(preview_group_box, 1) # Give preview some stretch on the left
-
- self.preview_label = QLabel()
- self.preview_label.setAlignment(Qt.AlignmentFlag.AlignCenter)
- self.preview_label.setFixedSize(250, 250) # Reduced fixed size for the preview area
- self.preview_label.setSizePolicy(QSizePolicy.Policy.Expanding, QSizePolicy.Policy.Expanding) # Allow it to expand
- self._set_placeholder_preview()
- preview_group_layout.addWidget(self.preview_label, 1, Qt.AlignmentFlag.AlignCenter)
-
# Add stretch to left layout to push content to top
left_layout.addStretch(1)
@@ -453,15 +742,18 @@ def setup_converter_tab(self):
# Conversion Options with TreeWidget for better organization
options_group_box = QGroupBox("Conversion Options")
options_group_layout = QVBoxLayout(options_group_box)
- options_group_layout.setContentsMargins(10, 25, 10, 10) # Reset margins for options group
- right_side_v_layout.addWidget(options_group_box, 6) # Give more stretch to options
+ options_group_layout.setContentsMargins(10, 10, 10, 10) # Reset margins for options group
+ options_group_box.setMinimumSize(250,500)
+ right_side_v_layout.addWidget(options_group_box, 8) # Give even more stretch to options
# Create scroll area for TreeWidget
- scroll_area = QScrollArea()
+ scroll_area = ScrollArea()
scroll_area.setWidgetResizable(True)
+ #scroll_area.setMinimumSize(90,90)
scroll_area.setVerticalScrollBarPolicy(Qt.ScrollBarPolicy.ScrollBarAsNeeded)
scroll_area.setHorizontalScrollBarPolicy(Qt.ScrollBarPolicy.ScrollBarAsNeeded)
- scroll_area.setMinimumHeight(450) # Set minimum height for scroll area
+ # 使用更灵活的高度设置,允许根据内容自动调整
+ scroll_area.setSizePolicy(QSizePolicy.Policy.Expanding, QSizePolicy.Policy.Expanding)
# Create TreeWidget for organized settings
self.options_tree = TreeWidget()
@@ -482,6 +774,12 @@ def setup_converter_tab(self):
progress_group_layout.setContentsMargins(10, 25, 10, 10)
left_layout.addWidget(progress_group_box, 0) # No stretch for progress, compact
+ # Task mode control
+ self.task_mode_check = CheckBox("Enable Task Mode")
+ self.task_mode_check.setChecked(False)
+ progress_group_layout.addWidget(self.task_mode_check)
+ progress_group_layout.addSpacing(10)
+
self.progress_label = QLabel("Ready")
self.progress_label.setAlignment(Qt.AlignmentFlag.AlignCenter)
self.progress = ProgressBar()
@@ -490,7 +788,11 @@ def setup_converter_tab(self):
progress_group_layout.addWidget(self.progress_label)
progress_group_layout.addWidget(self.progress)
- # Convert Button (moved before Progress) - Replace with PrimaryPushButton and apply custom style
+ # Convert and Cancel Button Layout
+ convert_control_layout = QHBoxLayout()
+ convert_control_layout.setSpacing(10)
+
+ # Convert Button - Replace with PrimaryPushButton and apply custom style
self.convert_button = PrimaryPushButton("Convert to ICNS")
self.convert_button.setFixedSize(180, 40)
font = self.convert_button.font()
@@ -500,7 +802,26 @@ def setup_converter_tab(self):
# Apply custom style to convert button
setCustomStyleSheet(self.convert_button, CON.qss, CON.qss)
self.convert_button.clicked.connect(self.on_start_conversion)
- left_layout.addWidget(self.convert_button, 0, Qt.AlignmentFlag.AlignCenter)
+
+ # Cancel Button
+ self.cancel_button = PushButton("Cancel")
+ self.cancel_button.setFixedSize(180, 40)
+ font = self.cancel_button.font()
+ font.setPointSize(font.pointSize() + 1)
+ self.cancel_button.setFont(font)
+ self.cancel_button.setEnabled(False)
+ # Apply custom style to cancel button
+ setCustomStyleSheet(self.cancel_button, CON.qss, CON.qss)
+ self.cancel_button.clicked.connect(self.on_cancel_conversion)
+
+ convert_control_layout.addWidget(self.convert_button)
+ convert_control_layout.addWidget(self.cancel_button)
+ # Create a wrapper widget to center the button layout
+ button_wrapper = QWidget()
+ button_wrapper_layout = QVBoxLayout(button_wrapper)
+ button_wrapper_layout.setAlignment(Qt.AlignmentFlag.AlignCenter)
+ button_wrapper_layout.addLayout(convert_control_layout)
+ left_layout.addWidget(button_wrapper)
@@ -513,6 +834,575 @@ def setup_converter_tab(self):
# Add a stretch to the converter layout to push everything to the top
converter_layout.addStretch(1)
+ def setup_batch_converter_tab(self):
+ """Setup the batch converter tab content"""
+ batch_layout = QVBoxLayout(self.batch_converter_tab)
+
+ # Title for batch converter
+ batch_title_label = QLabel("Batch Image Converter")
+ batch_title_font = QFont()
+ batch_title_font.setPointSize(batch_title_label.font().pointSize() + 6)
+ batch_title_font.setBold(True)
+ batch_title_label.setFont(batch_title_font)
+ batch_title_label.setAlignment(Qt.AlignmentFlag.AlignCenter)
+ batch_layout.addWidget(batch_title_label, 0, Qt.AlignmentFlag.AlignHCenter)
+
+ # Create horizontal layout for main content
+ main_batch_h_layout = QHBoxLayout()
+ main_batch_h_layout.setSpacing(20)
+ batch_layout.addLayout(main_batch_h_layout)
+
+ # Left Panel: File Selection and Preview
+ left_batch_panel = QWidget()
+ left_batch_layout = QVBoxLayout(left_batch_panel)
+ left_batch_layout.setContentsMargins(15, 15, 15, 15)
+ left_batch_layout.setSpacing(15)
+ main_batch_h_layout.addWidget(left_batch_panel, 1)
+
+ # Drop zone for files and folders
+ drop_group_box = QGroupBox("File Selection")
+ drop_group_layout = QVBoxLayout(drop_group_box)
+ drop_group_layout.setContentsMargins(10, 25, 10, 10)
+ drop_group_layout.setSpacing(15)
+ left_batch_layout.addWidget(drop_group_box)
+
+ # Drop zone
+ self.drop_zone = DropZoneWidget()
+ self.drop_zone.setFixedHeight(160)
+ drop_group_layout.addWidget(self.drop_zone)
+
+ # Browse button for manual file selection
+ browse_file_layout = QHBoxLayout()
+ self.browse_file_button = PushButton("Browse...")
+ self.browse_file_button.setText("Browse...")
+ setCustomStyleSheet(self.browse_file_button, CON.qss, CON.qss)
+ self.browse_file_button.clicked.connect(self.on_browse_batch_input)
+
+ browse_file_layout.addStretch()
+ browse_file_layout.addWidget(self.browse_file_button)
+ drop_group_layout.addLayout(browse_file_layout)
+
+ # Connect drop signals
+ self.drop_zone.filesDropped.connect(self.on_batch_files_dropped)
+ self.drop_zone.folderDropped.connect(self.on_batch_folder_dropped)
+
+ # Output Directory Selection (moved here)
+ output_dir_group_box = QGroupBox("Output Directory")
+ output_dir_group_layout = QVBoxLayout(output_dir_group_box)
+ output_dir_group_layout.setContentsMargins(10, 25, 10, 10)
+ output_dir_group_layout.setSpacing(10)
+ left_batch_layout.addWidget(output_dir_group_box, 0)
+
+ # Batch output directory layout with drag-drop support
+ batch_output_layout = QHBoxLayout()
+ batch_output_label = QLabel("📁 Output Directory:")
+ self.batch_output_text = DirectoryDropLineEdit() # 使用新的拖拽支持输入框
+ self.batch_output_text.setPlaceholderText("Same as input files directory")
+ setCustomStyleSheet(self.batch_output_text, CON.qss_line, CON.qss_line)
+
+ batch_output_button = PushButton("Browse...")
+ setCustomStyleSheet(batch_output_button, CON.qss, CON.qss)
+ batch_output_button.clicked.connect(self.on_browse_batch_output)
+
+ batch_output_layout.addWidget(batch_output_label)
+ batch_output_layout.addWidget(self.batch_output_text, 1)
+ batch_output_layout.addWidget(batch_output_button)
+ output_dir_group_layout.addLayout(batch_output_layout)
+
+ # File list with remove functionality
+ file_list_group_box = QGroupBox("Selected Files")
+ file_list_group_layout = QVBoxLayout(file_list_group_box)
+ file_list_group_layout.setContentsMargins(10, 25, 10, 10)
+ left_batch_layout.addWidget(file_list_group_box, 1)
+
+ # File list with scroll area
+ file_scroll_area = ScrollArea()
+ file_scroll_area.setWidgetResizable(True)
+ file_scroll_area.setMinimumHeight(200)
+ file_scroll_area.setMaximumHeight(300)
+
+ self.file_list_widget = ListWidget()
+ file_scroll_area.setWidget(self.file_list_widget)
+ file_list_group_layout.addWidget(file_scroll_area)
+
+ # File management buttons
+ file_mgmt_layout = QHBoxLayout()
+ self.clear_files_btn = PushButton("Clear All")
+ self.remove_selected_btn = PushButton("Remove Selected")
+ setCustomStyleSheet(self.clear_files_btn, CON.qss, CON.qss)
+ setCustomStyleSheet(self.remove_selected_btn, CON.qss, CON.qss)
+
+ self.clear_files_btn.clicked.connect(self.clear_batch_files)
+ self.remove_selected_btn.clicked.connect(self.remove_selected_files)
+
+ file_mgmt_layout.addWidget(self.clear_files_btn)
+ file_mgmt_layout.addWidget(self.remove_selected_btn)
+ file_list_group_layout.addLayout(file_mgmt_layout)
+
+ # Preview area removed - now in dedicated Preview tab
+
+ # Right Panel: Options and Progress
+ right_batch_panel = QWidget()
+ right_batch_layout = QVBoxLayout(right_batch_panel)
+ right_batch_layout.setContentsMargins(15, 15, 15, 15)
+ right_batch_layout.setSpacing(15)
+ main_batch_h_layout.addWidget(right_batch_panel, 1)
+
+ # Batch options (reusing existing options tree)
+ options_group_box = QGroupBox("Conversion Options")
+ options_group_layout = QVBoxLayout(options_group_box)
+ options_group_layout.setContentsMargins(10, 25, 10, 10)
+ right_batch_layout.addWidget(options_group_box, 10)
+
+ # Create scroll area for options
+ batch_scroll_area = ScrollArea()
+ batch_scroll_area.setWidgetResizable(True)
+ # 使用更灵活的尺寸策略
+ batch_scroll_area.setSizePolicy(QSizePolicy.Policy.Expanding, QSizePolicy.Policy.Expanding)
+
+ self.batch_options_tree = TreeWidget()
+ self.batch_options_tree.setHeaderHidden(True)
+ self.batch_options_tree.setRootIsDecorated(True)
+ self.batch_options_tree.setIndentation(20)
+
+ batch_scroll_area.setWidget(self.batch_options_tree)
+ options_group_layout.addWidget(batch_scroll_area)
+
+ # Setup batch options tree (reuse existing option structure)
+ self._setup_batch_options_tree()
+
+ # Batch Progress & Results section
+ progress_results_group_box = QGroupBox("Batch Progress & Results")
+ progress_results_layout = QVBoxLayout(progress_results_group_box)
+ progress_results_layout.setContentsMargins(10, 25, 10, 10)
+ progress_results_layout.setSpacing(15)
+ right_batch_layout.addWidget(progress_results_group_box, 2)
+
+ # Overall progress
+ self.batch_progress_label = QLabel("Ready")
+ self.batch_progress_label.setAlignment(Qt.AlignmentFlag.AlignCenter)
+ self.batch_progress = ProgressBar()
+ self.batch_progress.setRange(0, 100)
+ self.batch_progress.setValue(0)
+
+ progress_results_layout.addWidget(self.batch_progress_label)
+ progress_results_layout.addWidget(self.batch_progress)
+
+ # Current file progress
+ self.current_file_label = QLabel("No file processing")
+ self.current_file_label.setAlignment(Qt.AlignmentFlag.AlignCenter)
+ self.current_file_progress = ProgressBar()
+ self.current_file_progress.setRange(0, 100)
+ self.current_file_progress.setValue(0)
+
+ progress_results_layout.addWidget(self.current_file_label)
+ progress_results_layout.addWidget(self.current_file_progress)
+
+ # Separator line
+ separator = QFrame()
+ separator.setFrameShape(QFrame.HLine)
+ separator.setFrameShadow(QFrame.Sunken)
+ progress_results_layout.addWidget(separator)
+
+ # Results display with statistics
+ results_layout = QVBoxLayout()
+ results_layout.setSpacing(10)
+
+ # Statistics
+ stats_layout = QHBoxLayout()
+ self.success_count_label = QLabel("Success: 0")
+ self.failed_count_label = QLabel("Failed: 0")
+ self.total_count_label = QLabel("Total: 0")
+ stats_layout.addWidget(self.success_count_label)
+ stats_layout.addWidget(self.failed_count_label)
+ stats_layout.addWidget(self.total_count_label)
+ stats_layout.addStretch(1)
+ results_layout.addLayout(stats_layout)
+
+ # Results list with scroll
+ results_scroll_area = ScrollArea()
+ results_scroll_area.setWidgetResizable(True)
+ results_scroll_area.setMinimumHeight(150)
+
+ self.results_list_widget = ListWidget()
+ results_scroll_area.setWidget(self.results_list_widget)
+ results_layout.addWidget(results_scroll_area)
+
+ progress_results_layout.addLayout(results_layout)
+
+ # Batch control buttons
+ batch_control_layout = QHBoxLayout()
+ # Initialize with default format text
+ batch_output_format = self.batch_format_combo.currentText().lower() if hasattr(self, 'batch_format_combo') else 'icns'
+ self.start_batch_btn = PrimaryPushButton(f"Convert to {batch_output_format.upper()}")
+ self.stop_batch_btn = PushButton("Stop")
+ self.stop_batch_btn.setEnabled(False)
+
+ setCustomStyleSheet(self.start_batch_btn, CON.qss, CON.qss)
+ setCustomStyleSheet(self.stop_batch_btn, CON.qss, CON.qss)
+
+ self.start_batch_btn.clicked.connect(self.start_batch_conversion)
+ self.stop_batch_btn.clicked.connect(self.stop_batch_conversion)
+
+ batch_control_layout.addWidget(self.start_batch_btn)
+ batch_control_layout.addWidget(self.stop_batch_btn)
+ right_batch_layout.addLayout(batch_control_layout)
+
+ # Add stretch to push content to top
+ right_batch_layout.addStretch(1)
+
+ # Add stretch to push content to top-left
+ batch_layout.addStretch(1)
+
+ # Initialize batch variables
+ self.batch_files = []
+ self.batch_converting = False
+
+ def _setup_batch_options_tree(self):
+ """Setup the TreeWidget with organized settings for batch conversion (same structure as single conversion)"""
+ # Clear existing items
+ self.batch_options_tree.clear()
+
+ # Create main categories with icons for better visual hierarchy (same as single conversion)
+ basic_item = QTreeWidgetItem(["📋 Basic Options"])
+ processing_item = QTreeWidgetItem(["🎨 Image Processing"])
+ output_item = QTreeWidgetItem(["📤 Output Options"])
+ advanced_item = QTreeWidgetItem(["⚙️ Advanced Settings"])
+
+ # Add to tree
+ self.batch_options_tree.addTopLevelItem(basic_item)
+ self.batch_options_tree.addTopLevelItem(processing_item)
+ self.batch_options_tree.addTopLevelItem(output_item)
+ self.batch_options_tree.addTopLevelItem(advanced_item)
+
+ # Set font for categories to make them stand out (same as single conversion)
+ bold_font = self.batch_options_tree.font()
+ bold_font.setBold(True)
+ bold_font.setPointSize(bold_font.pointSize() + 1)
+
+ basic_item.setFont(0, bold_font)
+ processing_item.setFont(0, bold_font)
+ output_item.setFont(0, bold_font)
+ advanced_item.setFont(0, bold_font)
+
+ # Set background colors for categories (same as single conversion)
+ basic_item.setBackground(0, QColor(240, 248, 255, 80)) # Light blue
+ processing_item.setBackground(0, QColor(240, 255, 240, 80)) # Light green
+ output_item.setBackground(0, QColor(255, 250, 205, 80)) # Light yellow
+ advanced_item.setBackground(0, QColor(255, 248, 240, 80)) # Light orange
+
+ # Basic Options (same structure as single conversion)
+ self._create_batch_basic_options(basic_item)
+
+ # Processing Options (same structure as single conversion)
+ self._create_batch_processing_options(processing_item)
+
+ # Output Options (new)
+ self._create_batch_output_options(output_item)
+
+ # Advanced Options (same structure as single conversion)
+ self._create_batch_advanced_options(advanced_item)
+
+ # Set expansion state (same as single conversion)
+ basic_item.setExpanded(True)
+ processing_item.setExpanded(True)
+ output_item.setExpanded(True)
+ advanced_item.setExpanded(False) # Hidden by default
+
+ # Connect tree signals (same as single conversion)
+ self.batch_options_tree.itemExpanded.connect(self._on_batch_tree_item_expanded)
+ self.batch_options_tree.itemCollapsed.connect(self._on_batch_tree_item_collapsed)
+
+ def _create_batch_basic_options(self, parent_item):
+ """Create basic options widgets for batch conversion with responsive layout"""
+ # Output Format
+ format_widget = QWidget()
+ # 使用更灵活的尺寸策略
+ format_widget.setMinimumHeight(55)
+ format_widget.setSizePolicy(QSizePolicy.Policy.Expanding, QSizePolicy.Policy.Preferred)
+ format_layout = QHBoxLayout(format_widget)
+ format_layout.setContentsMargins(5, 8, 5, 8)
+
+ format_label = QLabel("🗂️ Output Format:")
+ format_label.setMinimumWidth(120) # Ensure consistent width for labels
+
+ self.batch_format_combo = ModelComboBox()
+ self.batch_format_combo.addItems(convert.SUPPORTED_FORMATS)
+ self.batch_format_combo.currentIndexChanged.connect(self.on_batch_format_change)
+ setCustomStyleSheet(self.batch_format_combo, CON.qss_combo, CON.qss_combo)
+
+ format_layout.addWidget(format_label)
+ format_layout.addWidget(self.batch_format_combo, 1) # Give combo box more stretch
+
+ format_item = QTreeWidgetItem()
+ parent_item.addChild(format_item)
+ self.batch_options_tree.setItemWidget(format_item, 0, format_widget)
+
+ # Size Options (grouped in a sub-item)
+ size_item = QTreeWidgetItem(["📏 Size Options"])
+ parent_item.addChild(size_item)
+
+ # Minimum Size
+ min_size_widget = QWidget()
+ min_size_layout = QHBoxLayout(min_size_widget)
+ min_size_layout.setContentsMargins(25, 5, 5, 5) # Indent for sub-item
+ # 使用更灵活的尺寸策略
+ min_size_widget.setMinimumHeight(45)
+ min_size_widget.setSizePolicy(QSizePolicy.Policy.Expanding, QSizePolicy.Policy.Preferred)
+
+ min_size_label = QLabel("Min Size:")
+ min_size_label.setMinimumWidth(80)
+ self.batch_min_spin = SpinBox()
+ setCustomStyleSheet(self.batch_min_spin, CON.qss_spin, CON.qss_spin)
+ self.batch_min_spin.setRange(16, 512)
+ self.batch_min_spin.setValue(self.min_size)
+ self.batch_min_spin.setSuffix(" px") # Add unit suffix
+ self.batch_min_spin.valueChanged.connect(self.on_batch_min_size_change)
+
+ min_size_layout.addWidget(min_size_label)
+ min_size_layout.addWidget(self.batch_min_spin, 1)
+
+ min_size_sub_item = QTreeWidgetItem()
+ size_item.addChild(min_size_sub_item)
+ self.batch_options_tree.setItemWidget(min_size_sub_item, 0, min_size_widget)
+
+ # Maximum Size
+ max_size_widget = QWidget()
+ max_size_layout = QHBoxLayout(max_size_widget)
+ max_size_layout.setContentsMargins(25, 5, 5, 5) # Indent for sub-item
+ # 使用更灵活的尺寸策略
+ max_size_widget.setMinimumHeight(45)
+ max_size_widget.setSizePolicy(QSizePolicy.Policy.Expanding, QSizePolicy.Policy.Preferred)
+
+ max_size_label = QLabel("Max Size:")
+ max_size_label.setMinimumWidth(80)
+ self.batch_max_spin = SpinBox()
+ setCustomStyleSheet(self.batch_max_spin, CON.qss_spin, CON.qss_spin)
+ self.batch_max_spin.setRange(32, 1024)
+ self.batch_max_spin.setValue(self.max_size)
+ self.batch_max_spin.setSuffix(" px") # Add unit suffix
+ self.batch_max_spin.valueChanged.connect(self.on_batch_max_size_change)
+
+ max_size_layout.addWidget(max_size_label)
+ max_size_layout.addWidget(self.batch_max_spin, 1)
+
+ max_size_sub_item = QTreeWidgetItem()
+ size_item.addChild(max_size_sub_item)
+ self.batch_options_tree.setItemWidget(max_size_sub_item, 0, max_size_widget)
+
+ # Auto-detect max size checkbox
+ auto_detect_widget = QWidget()
+ auto_detect_layout = QHBoxLayout(auto_detect_widget)
+ auto_detect_layout.setContentsMargins(25, 5, 5, 5) # Indent for sub-item
+ # 使用更灵活的尺寸策略
+ auto_detect_widget.setMinimumHeight(45)
+ auto_detect_widget.setSizePolicy(QSizePolicy.Policy.Expanding, QSizePolicy.Policy.Preferred)
+
+ self.batch_auto_detect_check = CheckBox("🔍 Auto-detect max size for each image")
+ self.batch_auto_detect_check.stateChanged.connect(self.on_batch_auto_detect_toggle)
+ auto_detect_layout.addWidget(self.batch_auto_detect_check)
+
+ auto_detect_sub_item = QTreeWidgetItem()
+ size_item.addChild(auto_detect_sub_item)
+ self.batch_options_tree.setItemWidget(auto_detect_sub_item, 0, auto_detect_widget)
+
+ # Auto-detect button (disabled for now, will be removed later)
+ # auto_widget = QWidget()
+ # auto_layout = QHBoxLayout(auto_widget)
+ # auto_layout.setContentsMargins(25, 5, 5, 5) # Indent for sub-item
+ # # 使用更灵活的尺寸策略
+ # auto_widget.setMinimumHeight(40)
+ # auto_widget.setSizePolicy(QSizePolicy.Policy.Expanding, QSizePolicy.Policy.Preferred)
+ #
+ # self.batch_auto_button = PrimaryPushButton("🔍 Auto-detect Max Size")
+ # setCustomStyleSheet(self.batch_auto_button, CON.qss_debug, CON.qss_debug)
+ # self.batch_auto_button.clicked.connect(self.on_batch_auto_detect)
+ # auto_layout.addWidget(self.batch_auto_button)
+ #
+ # auto_sub_item = QTreeWidgetItem()
+ # size_item.addChild(auto_sub_item)
+ # self.batch_options_tree.setItemWidget(auto_sub_item, 0, auto_widget)
+
+ # Expand size options by default
+ size_item.setExpanded(True)
+
+ def _create_batch_processing_options(self, parent_item):
+ """Create image processing options widgets for batch conversion (same as single conversion)"""
+ # Keep aspect ratio
+ aspect_widget = QWidget()
+ aspect_layout = QHBoxLayout(aspect_widget)
+ aspect_layout.setContentsMargins(5, 8, 5, 8)
+ # 使用更灵活的尺寸策略
+ aspect_widget.setMinimumHeight(45)
+ aspect_widget.setSizePolicy(QSizePolicy.Policy.Expanding, QSizePolicy.Policy.Preferred)
+ self.batch_keep_aspect_check = CheckBox("📐 Maintain original aspect ratio")
+ self.batch_keep_aspect_check.setChecked(self.keep_aspect_ratio)
+ self.batch_keep_aspect_check.stateChanged.connect(self.on_batch_keep_aspect_changed)
+ aspect_layout.addWidget(self.batch_keep_aspect_check)
+
+ aspect_item = QTreeWidgetItem()
+ parent_item.addChild(aspect_item)
+ self.batch_options_tree.setItemWidget(aspect_item, 0, aspect_widget)
+
+ # Auto crop
+ crop_widget = QWidget()
+ crop_layout = QHBoxLayout(crop_widget)
+ crop_layout.setContentsMargins(5, 8, 5, 8)
+ # 使用更灵活的尺寸策略
+ crop_widget.setMinimumHeight(45)
+ crop_widget.setSizePolicy(QSizePolicy.Policy.Expanding, QSizePolicy.Policy.Preferred)
+ self.batch_auto_crop_check = CheckBox("✂️ Auto-crop non-square to square")
+ self.batch_auto_crop_check.setChecked(self.auto_crop)
+ self.batch_auto_crop_check.stateChanged.connect(self.on_batch_auto_crop_changed)
+ crop_layout.addWidget(self.batch_auto_crop_check)
+ crop_item = QTreeWidgetItem()
+ parent_item.addChild(crop_item)
+ self.batch_options_tree.setItemWidget(crop_item, 0, crop_widget)
+
+ # Quality slider
+ quality_widget = QWidget()
+ quality_layout = QHBoxLayout(quality_widget)
+ quality_layout.setContentsMargins(5, 8, 5, 8)
+ # 使用更灵活的尺寸策略
+ quality_widget.setMinimumHeight(55)
+ quality_widget.setSizePolicy(QSizePolicy.Policy.Expanding, QSizePolicy.Policy.Preferred)
+ quality_label = QLabel("🎨 Quality:")
+ quality_label.setMinimumWidth(100) # Ensure consistent width for labels
+ quality_layout.addWidget(quality_label)
+
+ self.batch_quality_slider = Slider(Qt.Orientation.Horizontal)
+ self.batch_quality_slider.setRange(1, 100)
+ self.batch_quality_slider.setValue(self.quality)
+ self.batch_quality_slider.valueChanged.connect(self.on_batch_quality_changed)
+ quality_layout.addWidget(self.batch_quality_slider, 1) # Give slider more stretch
+
+ # Use read-only QLabel for quality value, same as single conversion
+ self.batch_quality_label = QLabel(str(self.quality))
+ self.batch_quality_label.setMinimumWidth(30) # Ensure consistent width for label
+ self.batch_quality_label.setMaximumWidth(60) # Limit width
+ self.batch_quality_label.setAlignment(Qt.AlignmentFlag.AlignCenter)
+ quality_layout.addWidget(self.batch_quality_label)
+
+ quality_item = QTreeWidgetItem()
+ parent_item.addChild(quality_item)
+ self.batch_options_tree.setItemWidget(quality_item, 0, quality_widget)
+
+ def _create_batch_output_options(self, parent_item):
+ """Create batch output options widgets"""
+ # Output Structure options
+ output_structure_item = QTreeWidgetItem(["📁 Output Structure Options"])
+ parent_item.addChild(output_structure_item)
+
+ # Preserve folder structure
+ preserve_folder_widget = QWidget()
+ preserve_folder_layout = QHBoxLayout(preserve_folder_widget)
+ preserve_folder_layout.setContentsMargins(25, 5, 5, 5) # Indent for sub-item
+ # 使用更灵活的尺寸策略
+ preserve_folder_widget.setMinimumHeight(45)
+ preserve_folder_widget.setSizePolicy(QSizePolicy.Policy.Expanding, QSizePolicy.Policy.Preferred)
+
+ self.batch_preserve_folder_check = CheckBox("Preserve original folder structure")
+ self.batch_preserve_folder_check.setChecked(False)
+ preserve_folder_layout.addWidget(self.batch_preserve_folder_check)
+
+ preserve_folder_sub_item = QTreeWidgetItem()
+ output_structure_item.addChild(preserve_folder_sub_item)
+ self.batch_options_tree.setItemWidget(preserve_folder_sub_item, 0, preserve_folder_widget)
+
+ # Filename modification options
+ filename_mod_item = QTreeWidgetItem(["📝 Filename Modification"])
+ parent_item.addChild(filename_mod_item)
+
+ # Prefix
+ prefix_widget = QWidget()
+ prefix_layout = QHBoxLayout(prefix_widget)
+ prefix_layout.setContentsMargins(25, 5, 5, 5) # Indent for sub-item
+ # 使用更灵活的尺寸策略
+ prefix_widget.setMinimumHeight(45)
+ prefix_widget.setSizePolicy(QSizePolicy.Policy.Expanding, QSizePolicy.Policy.Preferred)
+
+ prefix_label = QLabel("Prefix:")
+ prefix_label.setMinimumWidth(80)
+ self.batch_prefix_edit = LineEdit()
+ self.batch_prefix_edit.setPlaceholderText("Add prefix to filenames")
+ prefix_layout.addWidget(prefix_label)
+ prefix_layout.addWidget(self.batch_prefix_edit, 1)
+
+ prefix_sub_item = QTreeWidgetItem()
+ filename_mod_item.addChild(prefix_sub_item)
+ self.batch_options_tree.setItemWidget(prefix_sub_item, 0, prefix_widget)
+
+ # Suffix
+ suffix_widget = QWidget()
+ suffix_layout = QHBoxLayout(suffix_widget)
+ suffix_layout.setContentsMargins(25, 5, 5, 5) # Indent for sub-item
+ # 使用更灵活的尺寸策略
+ suffix_widget.setMinimumHeight(45)
+ suffix_widget.setSizePolicy(QSizePolicy.Policy.Expanding, QSizePolicy.Policy.Preferred)
+
+ suffix_label = QLabel("Suffix:")
+ suffix_label.setMinimumWidth(80)
+ self.batch_suffix_edit = LineEdit()
+ self.batch_suffix_edit.setPlaceholderText("Add suffix to filenames")
+ suffix_layout.addWidget(suffix_label)
+ suffix_layout.addWidget(self.batch_suffix_edit, 1)
+
+ suffix_sub_item = QTreeWidgetItem()
+ filename_mod_item.addChild(suffix_sub_item)
+ self.batch_options_tree.setItemWidget(suffix_sub_item, 0, suffix_widget)
+
+ # Expand output structure options by default
+ output_structure_item.setExpanded(True)
+ filename_mod_item.setExpanded(True)
+
+ def _create_batch_advanced_options(self, parent_item):
+ """Create advanced options widgets for batch conversion (same as single conversion)"""
+ # ICNS method
+ method_widget = QWidget()
+ method_layout = QHBoxLayout(method_widget)
+ method_layout.setContentsMargins(5, 8, 5, 8)
+ # 使用更灵活的尺寸策略
+ method_widget.setMinimumHeight(55)
+ method_widget.setSizePolicy(QSizePolicy.Policy.Expanding, QSizePolicy.Policy.Preferred)
+
+ method_label = QLabel("⚙️ ICNS method:")
+ method_label.setMinimumWidth(120) # Ensure consistent width for labels
+ method_layout.addWidget(method_label)
+
+ self.batch_icns_method_combo = ModelComboBox()
+ self.batch_icns_method_combo.addItems(["iconutil (Recommended)", "Pillow Fallback"])
+ self.batch_icns_method_combo.setCurrentText(self.icns_method)
+ self.batch_icns_method_combo.currentTextChanged.connect(self.on_batch_icns_method_changed)
+ setCustomStyleSheet(self.batch_icns_method_combo, CON.qss_combo, CON.qss_combo)
+ method_layout.addWidget(self.batch_icns_method_combo, 1) # Give combo box more stretch
+
+ method_item = QTreeWidgetItem()
+ parent_item.addChild(method_item)
+ self.batch_options_tree.setItemWidget(method_item, 0, method_widget)
+
+ # Overwrite confirmation
+ overwrite_widget = QWidget()
+ overwrite_layout = QHBoxLayout(overwrite_widget)
+ overwrite_layout.setContentsMargins(5, 8, 5, 8)
+ # 使用更灵活的尺寸策略
+ overwrite_widget.setMinimumHeight(45)
+ overwrite_widget.setSizePolicy(QSizePolicy.Policy.Expanding, QSizePolicy.Policy.Preferred)
+ self.batch_overwrite_confirm_check = CheckBox("⚠️ Confirm before overwriting files")
+ self.batch_overwrite_confirm_check.setChecked(self.overwrite_confirm)
+ self.batch_overwrite_confirm_check.stateChanged.connect(self.on_batch_overwrite_confirm_changed)
+ overwrite_layout.addWidget(self.batch_overwrite_confirm_check)
+
+ overwrite_item = QTreeWidgetItem()
+ parent_item.addChild(overwrite_item)
+ self.batch_options_tree.setItemWidget(overwrite_item, 0, overwrite_widget)
+
+ def _update_batch_basic_options(self):
+ """Update batch basic options display"""
+ # Find and update the basic options item
+ basic_item = self.batch_options_tree.topLevelItem(0)
+ if basic_item and basic_item.text(0) == "Basic Options":
+ basic_item.setText(0, f"Basic Options (Min: {self.batch_min_spin.value()}px, Max: {self.batch_max_spin.value()}px, Format: {self.batch_format_combo.currentText()})")
+
def create_history_tab(self):
"""Create the history tab for conversion history"""
if not self.remember_path:
@@ -724,10 +1614,12 @@ def _setup_options_tree(self):
self.options_tree.itemCollapsed.connect(self._on_tree_item_collapsed)
def _create_basic_options(self, parent_item):
- """Create basic options widgets"""
+ """Create basic options widgets with responsive layout"""
# Output Format
format_widget = QWidget()
- format_widget.setMinimumSize(300, 55)
+ # 使用更灵活的尺寸策略而不是固定最小尺寸
+ format_widget.setMinimumHeight(55)
+ format_widget.setSizePolicy(QSizePolicy.Policy.Expanding, QSizePolicy.Policy.Preferred)
format_layout = QHBoxLayout(format_widget)
format_layout.setContentsMargins(5, 8, 5, 8)
@@ -811,12 +1703,14 @@ def _create_basic_options(self, parent_item):
size_item.setExpanded(True)
def _create_processing_options(self, parent_item):
- """Create image processing options widgets"""
+ """Create image processing options widgets with responsive layout"""
# Keep aspect ratio
aspect_widget = QWidget()
aspect_layout = QHBoxLayout(aspect_widget)
aspect_layout.setContentsMargins(5, 8, 5, 8)
- aspect_widget.setMinimumSize(300, 45)
+ # 使用更灵活的尺寸策略
+ aspect_widget.setMinimumHeight(45)
+ aspect_widget.setSizePolicy(QSizePolicy.Policy.Expanding, QSizePolicy.Policy.Preferred)
self.keep_aspect_check = CheckBox("📐 Maintain original aspect ratio")
self.keep_aspect_check.stateChanged.connect(self.on_keep_aspect_changed)
aspect_layout.addWidget(self.keep_aspect_check)
@@ -861,12 +1755,14 @@ def _create_processing_options(self, parent_item):
self.options_tree.setItemWidget(quality_item, 0, quality_widget)
def _create_advanced_options(self, parent_item):
- """Create advanced options widgets"""
- # ICNS method
+ """Create advanced settings widgets with responsive layout"""
+ # ICNS method selection
method_widget = QWidget()
method_layout = QHBoxLayout(method_widget)
method_layout.setContentsMargins(5, 8, 5, 8)
- method_widget.setMinimumSize(300, 55)
+ # 使用更灵活的尺寸策略
+ method_widget.setMinimumHeight(55)
+ method_widget.setSizePolicy(QSizePolicy.Policy.Expanding, QSizePolicy.Policy.Preferred)
method_label = QLabel("⚙️ ICNS method:")
method_label.setMinimumWidth(120) # Ensure consistent width for labels
@@ -896,69 +1792,9 @@ def _create_advanced_options(self, parent_item):
self.options_tree.setItemWidget(overwrite_item, 0, overwrite_widget)
def on_tab_changed(self, index):
- """Handle tab change with optional slide animation effect based on UI_FLUENT environment variable"""
- import sys
- import os
- sys.path.append(os.path.join(os.path.dirname(__file__), 'support'))
- from support.check_flag import check_flag
-
- # Check if UI_FLUENT environment variable is set to YES using check_flag function
- ui_fluent_enabled = check_flag("UI_FLUENT")
-
- # Skip animation if UI_FLUENT is not enabled
- if not ui_fluent_enabled:
- self._previous_tab_index = index
- return
-
- # Proceed with animation if UI_FLUENT is enabled
- from PySide6.QtCore import QPropertyAnimation, QEasingCurve, QRect
-
- # Get current tab widget
- current_widget = self.tab_widget.currentWidget()
- if not current_widget:
- return
-
- # Skip animation during initial startup to prevent layout issues
- if not hasattr(self, '_previous_tab_index') and not self.tab_widget.isVisible():
- self._previous_tab_index = index
- return
-
- # Get tab widget dimensions
- tab_width = self.tab_widget.width()
- tab_height = self.tab_widget.height()
-
- # Skip animation if window is not yet properly sized
- if tab_width <= 0 or tab_height <= 0:
- self._previous_tab_index = index
- return
-
- # Determine slide direction based on tab index
- if hasattr(self, '_previous_tab_index'):
- if index > self._previous_tab_index:
- # Sliding from right to left - start from 80% of width to prevent going out of bounds
- start_pos = QRect(int(tab_width * 0.8), 0, tab_width, tab_height)
- else:
- # Sliding from left to right - start from -80% of width to prevent going out of bounds
- start_pos = QRect(int(-tab_width * 0.8), 0, tab_width, tab_height)
- else:
- # First time, slide from right - start from 80% of width
- start_pos = QRect(int(tab_width * 0.8), 0, tab_width, tab_height)
-
- # Set initial position
- current_widget.setGeometry(start_pos)
-
- # Create slide animation
- self.slide_animation = QPropertyAnimation(current_widget, b"geometry")
- self.slide_animation.setDuration(300) # 300ms animation for smooth slide
- self.slide_animation.setStartValue(start_pos)
- self.slide_animation.setEndValue(QRect(0, 0, tab_width, tab_height))
- self.slide_animation.setEasingCurve(QEasingCurve.Type.OutCubic)
-
- # Store current tab index for next animation
+ """Handle tab change without animation"""
+ # Simply store the previous tab index and return
self._previous_tab_index = index
-
- # Start the animation
- self.slide_animation.start()
def _on_tree_item_expanded(self, item):
"""Handle tree item expansion"""
@@ -1048,18 +1884,177 @@ def create_success_view(self):
self.success_widget.hide() # Initially hidden
+ def create_batch_success_view(self):
+ """Create a specialized batch conversion success view"""
+ # Create batch success widget as a top-level overlay
+ self.batch_success_widget = QWidget(self)
+ self.batch_success_widget.setObjectName("success_overlay")
+
+ # Set it to cover the entire window
+ self.batch_success_widget.setGeometry(self.rect())
+
+ # Create layout for batch success widget
+ self.batch_success_layout = QVBoxLayout(self.batch_success_widget)
+ self.batch_success_layout.setContentsMargins(0, 0, 0, 0)
+
+ # Create a semi-transparent overlay background
+ overlay = QWidget()
+ overlay.setObjectName("success_overlay")
+ overlay_layout = QVBoxLayout(overlay)
+ overlay_layout.setContentsMargins(0, 0, 0, 0)
+
+ center_panel = QWidget()
+ center_panel.setObjectName("success_center_panel")
+ center_layout = QVBoxLayout(center_panel)
+ center_layout.setContentsMargins(50, 50, 50, 50) # Increased margins
+ center_layout.setSpacing(20)
+
+ # Add stretch to center the panel vertically
+ overlay_layout.addStretch()
+ overlay_layout.addWidget(center_panel, 0, Qt.AlignmentFlag.AlignCenter)
+ overlay_layout.addStretch()
+
+ self.batch_success_layout.addWidget(overlay)
+
+ center_layout.addStretch()
+
+ # Batch success title
+ title = QLabel("Batch Conversion Complete!")
+ font = title.font()
+ font.setPointSize(font.pointSize() + 8) # Larger title
+ font.setBold(True)
+ title.setFont(font)
+ title.setAlignment(Qt.AlignmentFlag.AlignCenter)
+ title.setObjectName("success_title_label")
+ center_layout.addWidget(title)
+
+ # Batch checkmark with check symbol
+ checkmark = QLabel("✓")
+ checkmark_font = checkmark.font()
+ checkmark_font.setPointSize(checkmark_font.pointSize() + 30) # Larger checkmark
+ checkmark_font.setBold(True)
+ checkmark.setFont(checkmark_font)
+ checkmark.setObjectName("success_checkmark_label")
+ checkmark.setAlignment(Qt.AlignmentFlag.AlignCenter)
+ center_layout.addWidget(checkmark)
+
+ # Dynamic message - will be updated with actual batch stats
+ self.batch_success_message = QLabel("Processing batch conversion results...")
+ self.batch_success_message.setAlignment(Qt.AlignmentFlag.AlignCenter)
+ self.batch_success_message.setObjectName("success_message_label")
+ center_layout.addWidget(self.batch_success_message)
+
+ # Batch statistics display
+ stats_widget = QWidget()
+ stats_layout = QVBoxLayout(stats_widget)
+ stats_layout.setSpacing(10)
+
+ self.batch_file_count_label = QLabel("Files Processed: 0")
+ self.batch_file_count_label.setAlignment(Qt.AlignmentFlag.AlignCenter)
+ self.batch_file_count_label.setObjectName("batch_file_count_label")
+ stats_layout.addWidget(self.batch_file_count_label)
+
+ self.batch_success_count_label = QLabel("Successfully Converted: 0")
+ self.batch_success_count_label.setAlignment(Qt.AlignmentFlag.AlignCenter)
+ self.batch_success_count_label.setObjectName("batch_success_count_label")
+ stats_layout.addWidget(self.batch_success_count_label)
+
+ self.batch_failed_count_label = QLabel("Failed Conversions: 0")
+ self.batch_failed_count_label.setAlignment(Qt.AlignmentFlag.AlignCenter)
+ self.batch_failed_count_label.setObjectName("batch_failed_count_label")
+ stats_layout.addWidget(self.batch_failed_count_label)
+
+ self.batch_output_dir_label = QLabel("Output Directory: ")
+ self.batch_output_dir_label.setAlignment(Qt.AlignmentFlag.AlignCenter)
+ self.batch_output_dir_label.setWordWrap(True)
+ self.batch_output_dir_label.setObjectName("batch_output_dir_label")
+ stats_layout.addWidget(self.batch_output_dir_label)
+
+ center_layout.addWidget(stats_widget)
+
+ # Button layout
+ button_layout = QHBoxLayout()
+
+ # Open output folder button
+ open_folder_btn = PrimaryPushButton("Open Output Folder")
+ open_folder_btn.clicked.connect(self.on_open_batch_output_folder)
+ open_folder_btn.setFixedSize(200, 45) # Larger buttons
+ open_folder_btn.setObjectName("open_converted_file_button") # Object name for QSS
+ setCustomStyleSheet(open_folder_btn, CON.qss, CON.qss)
+ button_layout.addWidget(open_folder_btn, 0, Qt.AlignmentFlag.AlignCenter)
+
+ # Return to converter button
+ return_btn = PushButton("Return to Converter")
+ return_btn.clicked.connect(self.show_main_view)
+ return_btn.setFixedSize(200, 45) # Larger buttons
+ return_btn.setObjectName("return_to_converter_button") # Object name for QSS
+ setCustomStyleSheet(return_btn, CON.qss, CON.qss)
+ button_layout.addWidget(return_btn, 0, Qt.AlignmentFlag.AlignCenter)
+
+ center_layout.addLayout(button_layout)
+ center_layout.addStretch()
+
+ self.batch_success_widget.hide() # Initially hidden
+
+ def show_batch_success_view(self, total_files=0, success_count=0, failed_count=0, output_dir="", format_name="PNG"):
+ """Show the batch success view with dynamic statistics"""
+ # Update the batch success widget geometry to match the window
+ self.batch_success_widget.setGeometry(self.rect())
+
+ # Update dynamic content
+ if hasattr(self, 'batch_success_message'):
+ if failed_count == 0:
+ self.batch_success_message.setText(f"All {total_files} {format_name.upper()} files converted successfully!")
+ elif success_count == 0:
+ self.batch_success_message.setText(f"Batch conversion failed: {failed_count} files could not be converted")
+ else:
+ self.batch_success_message.setText(f"Batch conversion completed: {success_count}/{total_files} files converted successfully")
+
+ if hasattr(self, 'batch_file_count_label'):
+ self.batch_file_count_label.setText(f"Files Processed: {total_files}")
+
+ if hasattr(self, 'batch_success_count_label'):
+ self.batch_success_count_label.setText(f"Successfully Converted: {success_count}")
+
+ if hasattr(self, 'batch_failed_count_label'):
+ if failed_count > 0:
+ self.batch_failed_count_label.setText(f"Failed Conversions: {failed_count}")
+ self.batch_failed_count_label.show()
+ else:
+ # Hide failed count when it's 0
+ self.batch_failed_count_label.hide()
+
+ if hasattr(self, 'batch_output_dir_label'):
+ self.batch_output_dir_label.setText(f"Output Directory: {output_dir}")
+
+ # Show the batch success widget as an overlay
+ self.batch_success_widget.show()
+ self.batch_success_widget.raise_() # Bring to front
+
+ # Apply theme-specific styles
+ self._apply_success_theme()
+
+ def on_open_batch_output_folder(self):
+ """Open the batch output folder in the file explorer"""
+ output_dir = self.output_dir_input.text().strip()
+ if output_dir and os.path.exists(output_dir):
+ try:
+ if sys.platform == "win32":
+ os.startfile(output_dir)
+ elif sys.platform == "darwin":
+ subprocess.run(["open", output_dir])
+ else: # Linux and other POSIX systems
+ subprocess.run(["xdg-open", output_dir])
+ except Exception as e:
+ QMessageBox.warning(self, "Error", f"Could not open folder: {str(e)}")
+ else:
+ QMessageBox.warning(self, "Error", "Output directory not found")
+
def _set_placeholder_preview(self):
- placeholder_text = "Drag and drop image here\nor click 'Browse...' to select file\n🖼️"
- font = QFont()
- font.setPointSize(16) # Larger font for placeholder
- self.preview_label.setFont(font)
- self.preview_label.setText(placeholder_text)
- self.preview_label.setPixmap(QPixmap()) # Clear any previous image
-
- # Enable drag and drop for the preview label
- self.preview_label.setAcceptDrops(True)
- self.preview_label.dragEnterEvent = self.dragEnterEvent
- self.preview_label.dropEvent = self.dropEvent
+ """Set placeholder text in the preview tab"""
+ # Clear existing previews and show default info
+ self.preview_tab.clear_previews()
+
def on_browse_input(self):
file_dialog = QFileDialog(self)
@@ -1131,6 +2126,41 @@ def on_browse_output(self):
# Remember the directory for next time if setting is enabled
if self.remember_path:
self.last_output_dir = os.path.dirname(self.output_path)
+
+ def on_browse_batch_input(self):
+ """Browse for batch input files"""
+ file_dialog = QFileDialog(self)
+ file_dialog.setFileMode(QFileDialog.FileMode.ExistingFiles)
+ file_dialog.setNameFilter("Supported Images (*.png *.jpg *.jpeg *.gif *.bmp *.webp *.ico *.tiff);;All Files (*)")
+ file_dialog.setViewMode(QFileDialog.ViewMode.Detail)
+
+ # Use last directory if available
+ if hasattr(self, 'last_batch_input_dir') and self.last_batch_input_dir:
+ file_dialog.setDirectory(self.last_batch_input_dir)
+
+ if file_dialog.exec() == QFileDialog.DialogCode.Accept:
+ selected_files = file_dialog.selectedFiles()
+ if selected_files:
+ # Remember this directory for future use
+ self.last_batch_input_dir = os.path.dirname(selected_files[0])
+
+ # Add files to batch file list
+ for file_path in selected_files:
+ self._add_file_to_batch_list(file_path)
+
+ def on_browse_batch_output(self):
+ """Browse for batch output directory"""
+ directory = QFileDialog.getExistingDirectory(
+ self,
+ "Select Output Directory for Batch Conversion",
+ self.batch_output_text.text() if self.batch_output_text.text() else os.path.expanduser("~"),
+ QFileDialog.Option.ShowDirsOnly
+ )
+
+ if directory:
+ self.batch_output_text.setText(directory)
+ # Remember this directory for future use
+ self.last_batch_output_dir = directory
def on_format_change(self, index):
self.output_format = self.format_combo.currentText().lower()
@@ -1147,6 +2177,28 @@ def on_format_change(self, index):
self.max_spin.setEnabled(enable_size_options)
self.auto_button.setEnabled(enable_size_options)
+ def on_batch_format_change(self, index):
+ """Handle format change in batch conversion"""
+ batch_output_format = self.batch_format_combo.currentText().lower()
+
+ # Update batch conversion button text to match format
+ if hasattr(self, 'start_batch_btn'):
+ self.start_batch_btn.setText(f"Convert to {batch_output_format.upper()}")
+
+ # Enable/disable auto-detect button based on format
+ if hasattr(self, 'batch_use_auto_detect'):
+ if batch_output_format in ['png', 'jpg', 'jpeg', 'webp']:
+ self.batch_use_auto_detect.setEnabled(True)
+ else:
+ self.batch_use_auto_detect.setEnabled(False)
+ self.batch_use_auto_detect.setChecked(False)
+
+ # Enable/disable min/max spin boxes based on format
+ if hasattr(self, 'batch_min_spin') and hasattr(self, 'batch_max_spin'):
+ enable_size_options = (batch_output_format == "icns")
+ self.batch_min_spin.setEnabled(enable_size_options)
+ self.batch_max_spin.setEnabled(enable_size_options)
+
def auto_set_output(self):
if self.input_path:
base_name = os.path.splitext(os.path.basename(self.input_path))[0]
@@ -1199,30 +2251,17 @@ def on_auto_detect(self):
def show_preview(self):
+ """Show preview in the preview tab"""
if self.input_path and os.path.exists(self.input_path):
try:
+ # Use the preview_tab to display the single preview
+ self.preview_tab.show_single_preview(self.input_path)
img = Image.open(self.input_path)
- # img.thumbnail((350, 350)) # Removed PIL thumbnailing
-
- if img.mode == 'RGBA':
- qimage = QImage(img.tobytes("raw", "RGBA"), img.size[0], img.size[1], QImage.Format.Format_RGBA8888) # Corrected enum
- else:
- qimage = QImage(img.tobytes("raw", "RGB"), img.size[0], img.size[1], QImage.Format.Format_RGB888) # Corrected enum
-
- pixmap = QPixmap.fromImage(qimage)
- # Scale pixmap to fit the label, maintaining aspect ratio
- scaled_pixmap = pixmap.scaled(self.preview_label.size(), Qt.AspectRatioMode.KeepAspectRatio, Qt.TransformationMode.SmoothTransformation)
- self.preview_label.setPixmap(scaled_pixmap)
- # Reset font to default if it was changed by placeholder
- self.preview_label.setFont(QFont())
- self.preview_label.setText("") # Clear placeholder text
self.status_bar.showMessage(f"Loaded: {os.path.basename(self.input_path)} ({img.size[0]}x{img.size[1]})")
except Exception as e:
- self.preview_label.setText("Preview error")
self.status_bar.showMessage("Preview error")
else:
- self.preview_label.clear()
- self._set_placeholder_preview() # Show placeholder when no image selected
+ self.preview_tab.clear_previews()
self.status_bar.showMessage("Ready")
def on_min_size_change(self, value):
@@ -1247,6 +2286,32 @@ def on_quality_changed(self, value):
self.quality_label.setText(str(value))
self.save_settings()
+ def on_batch_auto_detect_toggle(self, state):
+ """Handle auto-detect toggle in batch conversion"""
+ auto_detect_enabled = bool(state)
+ # Enable/disable min and max size spin boxes based on auto-detect setting
+ if hasattr(self, 'batch_min_spin') and hasattr(self, 'batch_max_spin'):
+ self.batch_min_spin.setEnabled(not auto_detect_enabled)
+ self.batch_max_spin.setEnabled(not auto_detect_enabled)
+
+ # Enable/disable auto-detect button if it exists
+ if hasattr(self, 'batch_auto_button'):
+ self.batch_auto_button.setEnabled(not auto_detect_enabled)
+
+ def on_batch_auto_detect(self):
+ """Handle auto-detect in batch conversion"""
+ # This method is deprecated, will be removed in future
+ PopupTeachingTip.create(
+ target=self.batch_auto_button,
+ icon=InfoBarIcon.WARNING,
+ title='Notice',
+ content="Auto-detect is now available through the checkbox option.",
+ isClosable=True,
+ tailPosition=TeachingTipTailPosition.TOP,
+ duration=3000,
+ parent=self
+ )
+
def on_icns_method_changed(self, text):
self.icns_method = text
self.save_settings()
@@ -1254,6 +2319,45 @@ def on_icns_method_changed(self, text):
def on_overwrite_confirm_changed(self, state):
self.overwrite_confirm = bool(state)
self.save_settings()
+
+ # Batch conversion event handlers (mirroring single conversion handlers)
+ def on_batch_min_size_change(self, value):
+ if hasattr(self, 'batch_min_size'):
+ self.batch_min_size = value
+
+ def on_batch_max_size_change(self, value):
+ if hasattr(self, 'batch_max_size'):
+ self.batch_max_size = value
+
+ def on_batch_quality_changed(self, value):
+ if hasattr(self, 'batch_quality'):
+ self.batch_quality = value
+ if hasattr(self, 'batch_quality_label'):
+ self.batch_quality_label.setText(str(value))
+
+ def on_batch_icns_method_changed(self, text):
+ if hasattr(self, 'batch_icns_method'):
+ self.batch_icns_method = text
+
+ def on_batch_overwrite_confirm_changed(self, state):
+ if hasattr(self, 'batch_overwrite_confirm'):
+ self.batch_overwrite_confirm = bool(state)
+
+ def on_batch_keep_aspect_changed(self, state):
+ if hasattr(self, 'batch_keep_aspect_ratio'):
+ self.batch_keep_aspect_ratio = bool(state)
+
+ def on_batch_auto_crop_changed(self, state):
+ if hasattr(self, 'batch_auto_crop'):
+ self.batch_auto_crop = bool(state)
+
+ def _on_batch_tree_item_expanded(self, item):
+ """Handle batch tree item expansion"""
+ pass # Could be enhanced for batch-specific behavior
+
+ def _on_batch_tree_item_collapsed(self, item):
+ """Handle batch tree item collapse"""
+ pass # Could be enhanced for batch-specific behavior
def on_interface_setting_changed(self):
"""Handle changes to interface behavior settings"""
@@ -1371,8 +2475,40 @@ def on_start_conversion(self):
)
return
+ # Check if task mode is enabled
+ settings = QSettings("MyCompany", "ConverterApp")
+ task_mode_enabled = settings.value("task_mode", False, type=bool)
+
+ if task_mode_enabled:
+ # Create a simple task notification file
+ import json
+ import uuid
+ import time
+
+ task_id = str(uuid.uuid4())
+ task_info = {
+ "task_id": task_id,
+ "task_type": "image",
+ "input_path": self.input_path,
+ "output_path": self.output_path,
+ "status": "pending",
+ "progress": 0,
+ "timestamp": time.time()
+ }
+
+ # Write task info to a temporary file
+ task_dir = os.path.expanduser("~/.converter/tasks")
+ os.makedirs(task_dir, exist_ok=True)
+ task_file = os.path.join(task_dir, f"task_{task_id}.json")
+ with open(task_file, "w") as f:
+ json.dump(task_info, f)
+
+ # Store task_id for later updates
+ self.current_task_id = task_id
+
self.converting = True
self.convert_button.setEnabled(False)
+ self.cancel_button.setEnabled(True)
self.progress.setValue(0)
self.progress_label.setText("Starting conversion...")
@@ -1391,12 +2527,37 @@ def on_start_conversion(self):
self._thread.started.connect(self._worker.run)
self._thread.start()
+ def on_cancel_conversion(self):
+ """Cancel the ongoing conversion process"""
+ if not self.converting:
+ return
+
+ # Update UI to show cancellation
+ self.progress_label.setText("Canceling conversion...")
+ self.cancel_button.setEnabled(False)
+
+ # Stop the worker thread safely
+ if hasattr(self, '_thread') and self._thread.isRunning():
+ self._thread.quit()
+ self._thread.wait()
+
+ # Update UI state
+ self.converting = False
+ self.convert_button.setEnabled(True)
+ self.progress.setValue(0)
+ self.progress_label.setText("Conversion canceled")
+ self.status_bar.showMessage("Conversion canceled by user")
+
def on_conversion_finished(self):
self.converting = False
if hasattr(self, '_thread') and self._thread.isRunning():
self._thread.quit()
self._thread.wait()
+ # Reset button states
+ self.convert_button.setEnabled(True)
+ self.cancel_button.setEnabled(False)
+
# Add to history if remember_path is enabled
if self.remember_path:
self.add_to_history(self.input_path, self.output_path, str(self.output_format))
@@ -1421,17 +2582,63 @@ def on_conversion_finished(self):
self.show_success_view()
else:
# If completion notification is disabled, just show a simple status message
- self.convert_button.setEnabled(True)
self.progress.setValue(100)
self.progress_label.setText("Conversion completed successfully!")
self.status_bar.showMessage(f"Conversion completed: {os.path.basename(self.output_path)}")
+ # Auto-reset after conversion completion
+ self._reset_after_conversion()
+
+ def _reset_after_conversion(self):
+ """Reset conversion state after completion to prepare for next conversion"""
+ # Only reset variables if remember_path is disabled
+ if not self.remember_path:
+ # Reset all variables when remember_path is disabled
+ self.init_variables(reset_all=True)
+
+ # Reset UI elements
+ if hasattr(self, 'input_text'):
+ self.input_text.clear()
+ if hasattr(self, 'output_text'):
+ self.output_text.clear()
+ if hasattr(self, 'info_text'):
+ self.info_text.setText("No image selected")
+ if hasattr(self, 'preview_tab'):
+ self.preview_tab.clear_previews() # Clear previews when returning to main view
+
def on_conversion_error(self, error_message):
self.converting = False
self.convert_button.setEnabled(True)
+ self.cancel_button.setEnabled(False) # Reset cancel button state
if hasattr(self, '_thread') and self._thread.isRunning():
self._thread.quit()
self._thread.wait()
+
+ # Update task status if task mode is enabled
+ if hasattr(self, 'current_task_id') and self.current_task_id:
+ try:
+ import json
+ import time
+
+ task_dir = os.path.expanduser("~/.converter/tasks")
+ task_file = os.path.join(task_dir, f"task_{self.current_task_id}.json")
+
+ # Read existing task info
+ if os.path.exists(task_file):
+ with open(task_file, "r") as f:
+ task_info = json.load(f)
+
+ # Update task info
+ task_info["status"] = "failed"
+ task_info["error"] = error_message
+ task_info["timestamp"] = time.time()
+
+ # Write updated task info
+ with open(task_file, "w") as f:
+ json.dump(task_info, f)
+ except Exception as e:
+ print(f"Error updating task status: {e}")
+
PopupTeachingTip.create(
target=self.convert_button,
icon=InfoBarIcon.ERROR,
@@ -1444,6 +2651,370 @@ def on_conversion_error(self, error_message):
)
self.progress_label.setText("Conversion Failed")
+ # Batch conversion callback methods
+ def on_batch_files_dropped(self, file_paths):
+ """Handle multiple files dropped in the batch converter"""
+ # Filter only supported image files
+ supported_formats = ('.png', '.jpg', '.jpeg', '.webp', '.ico', '.tiff', '.tif', '.icns', '.bmp', '.gif', '.svg', '.heic', '.heif', '.avif', '.jxl', '.pdf', '.eps', '.dds', '.exr')
+ new_files = []
+
+ for file_path in file_paths:
+ if os.path.splitext(file_path.lower())[1] in supported_formats:
+ new_files.append(file_path)
+
+ if new_files:
+ self.batch_files.extend(new_files)
+ self.update_batch_file_list()
+ # Update preview tab with multiple previews
+ self.preview_tab.show_multiple_previews(new_files)
+ self.update_batch_statistics()
+
+ def on_batch_folder_dropped(self, folder_path):
+ """Handle folder dropped in the batch converter"""
+ # Scan folder for supported image files
+ supported_formats = ('.png', '.jpg', '.jpeg', '.webp', '.ico', '.tiff', '.tif', '.icns', '.bmp', '.gif', '.svg', '.heic', '.heif', '.avif', '.jxl', '.pdf', '.eps', '.dds', '.exr')
+ found_files = []
+
+ try:
+ for root, dirs, files in os.walk(folder_path):
+ for file in files:
+ if any(file.lower().endswith(ext) for ext in supported_formats):
+ file_path = os.path.join(root, file)
+ found_files.append(file_path)
+ except Exception as e:
+ PopupTeachingTip.create(
+ target=self.drop_zone,
+ icon=InfoBarIcon.ERROR,
+ title='ERROR',
+ content=f"Cannot scan folder: {str(e)}",
+ isClosable=True,
+ tailPosition=TeachingTipTailPosition.TOP,
+ duration=3000,
+ parent=self
+ )
+ return
+
+ if found_files:
+ self.batch_files.extend(found_files)
+ self.update_batch_file_list()
+ # Update preview tab with multiple previews from folder
+ self.preview_tab.show_multiple_previews(found_files)
+ self.update_batch_statistics()
+
+ PopupTeachingTip.create(
+ target=self.drop_zone,
+ icon=InfoBarIcon.SUCCESS,
+ title='SUCCESS',
+ content=f"Added {len(found_files)} files from folder: {os.path.basename(folder_path)}",
+ isClosable=True,
+ tailPosition=TeachingTipTailPosition.TOP,
+ duration=3000,
+ parent=self
+ )
+ else:
+ PopupTeachingTip.create(
+ target=self.drop_zone,
+ icon=InfoBarIcon.WARNING,
+ title='WARNING',
+ content="No supported image files found in the folder",
+ isClosable=True,
+ tailPosition=TeachingTipTailPosition.TOP,
+ duration=3000,
+ parent=self
+ )
+
+ def update_batch_file_list(self):
+ """Update the file list widget with current batch files"""
+ self.file_list_widget.clear()
+ for file_path in self.batch_files:
+ item = QListWidgetItem(os.path.basename(file_path))
+ item.setToolTip(file_path)
+ self.file_list_widget.addItem(item)
+
+ def update_batch_statistics(self):
+ """Update batch conversion statistics"""
+ total_count = len(self.batch_files)
+ self.total_count_label.setText(f"Total: {total_count}")
+
+ def clear_batch_files(self):
+ """Clear all batch files"""
+ self.batch_files.clear()
+ self.update_batch_file_list()
+ # Clear preview tab
+ self.preview_tab.clear_previews()
+ self.update_batch_statistics()
+ self.clear_batch_results()
+
+ def remove_selected_files(self):
+ """Remove selected files from batch"""
+ selected_items = self.file_list_widget.selectedItems()
+ if not selected_items:
+ return
+
+ # Get indices of selected items
+ indices = []
+ for item in selected_items:
+ indices.append(self.file_list_widget.row(item))
+
+ # Sort indices in descending order to remove from end
+ indices.sort(reverse=True)
+
+ # Remove files from batch_files list
+ for index in indices:
+ if 0 <= index < len(self.batch_files):
+ del self.batch_files[index]
+
+ self.update_batch_file_list()
+ # Update preview tab with remaining files
+ self.preview_tab.show_multiple_previews(self.batch_files)
+ self.update_batch_statistics()
+
+ def start_batch_conversion(self):
+ """Start batch conversion process"""
+ if not self.batch_files:
+ PopupTeachingTip.create(
+ target=self.start_batch_btn,
+ icon=InfoBarIcon.WARNING,
+ title='WARNING',
+ content="Please select files to convert",
+ isClosable=True,
+ tailPosition=TeachingTipTailPosition.TOP,
+ duration=2000,
+ parent=self
+ )
+ return
+
+ # Get current conversion options
+ options = self._get_batch_conversion_options()
+ if not options:
+ return
+
+ # Start batch conversion in background thread
+ self.batch_converting = True
+ self.start_batch_btn.setEnabled(False)
+ self.stop_batch_btn.setEnabled(True)
+
+ # Reset progress and results
+ self.batch_progress.setValue(0)
+ self.current_file_progress.setValue(0)
+ self.clear_batch_results()
+
+ # Start batch conversion thread
+ self.batch_thread = QThread()
+ self.batch_worker = BatchConversionWorker(
+ self.batch_files,
+ self.get_batch_output_directory(),
+ options['output_format'],
+ min_size_param=options['min_size'],
+ max_size_param=options['max_size'],
+ quality_param=options['quality'],
+ preserve_folder_structure=options['preserve_folder_structure'],
+ prefix=options['prefix'],
+ suffix=options['suffix'],
+ auto_detect_max_size=options['auto_detect_max_size']
+ )
+ self.batch_worker.moveToThread(self.batch_thread)
+
+ # Connect signals
+ self.batch_worker.total_progress_updated.connect(self.on_batch_progress)
+ self.batch_worker.progress_updated.connect(self.on_batch_file_progress)
+ self.batch_worker.file_processed.connect(self.on_batch_file_processed)
+ self.batch_worker.finished.connect(self.on_batch_finished)
+ self.batch_worker.batch_error.connect(self.on_batch_error)
+
+ # Start the thread
+ self.batch_thread.started.connect(self.batch_worker.run)
+ self.batch_thread.start()
+
+ def stop_batch_conversion(self):
+ """Stop batch conversion process"""
+ if hasattr(self, 'batch_worker') and self.batch_converting:
+ self.batch_worker.cancel()
+ self.on_batch_stopped()
+
+ def on_batch_stopped(self):
+ """Handle batch conversion stopped"""
+ self.batch_converting = False
+ self.start_batch_btn.setEnabled(True)
+ self.stop_batch_btn.setEnabled(False)
+ self.batch_progress_label.setText("Stopped")
+ self.current_file_label.setText("No file processing")
+
+ # Clean up thread
+ if hasattr(self, 'batch_thread') and self.batch_thread.isRunning():
+ self.batch_thread.quit()
+ self.batch_thread.wait()
+
+ def on_batch_progress(self, overall_progress, message=None):
+ """Update batch overall progress"""
+ self.batch_progress.setValue(overall_progress)
+ if message:
+ self.batch_progress_label.setText(message)
+ else:
+ self.batch_progress_label.setText(f"Overall progress: {overall_progress}%")
+
+ def on_batch_file_progress(self, current_index, total_count, current_file, percentage):
+ """Update current file progress"""
+ self.current_file_progress.setValue(percentage)
+ self.current_file_label.setText(f"Processing {current_file} ({current_index}/{total_count})")
+
+ def on_batch_file_processed(self, filename, input_path, output_path, success, error_message):
+ """Handle individual file processing result"""
+ if success:
+ item_text = f"✓ {filename}"
+ item = QListWidgetItem(item_text)
+ item.setForeground(Qt.green)
+ # Add to history if conversion was successful and remember_path is enabled
+ if self.remember_path:
+ format_type = self.batch_format_combo.currentText().lower()
+ self.add_to_history(input_path, output_path, format_type)
+ else:
+ item_text = f"✗ {filename}: {error_message}"
+ item = QListWidgetItem(item_text)
+ item.setForeground(Qt.red)
+
+ self.results_list_widget.addItem(item)
+
+ def on_batch_finished(self):
+ """Handle batch conversion finished"""
+ self.batch_converting = False
+ self.start_batch_btn.setEnabled(True)
+ self.stop_batch_btn.setEnabled(False)
+
+ # Clean up thread
+ if hasattr(self, 'batch_thread') and self.batch_thread.isRunning():
+ self.batch_thread.quit()
+ self.batch_thread.wait()
+
+ # Get statistics from results list
+ success_count = 0
+ failed_count = 0
+
+ for i in range(self.results_list_widget.count()):
+ item = self.results_list_widget.item(i)
+ if item:
+ if "✓" in item.text():
+ success_count += 1
+ else:
+ failed_count += 1
+
+ self.success_count_label.setText(f"Success: {success_count}")
+ self.failed_count_label.setText(f"Failed: {failed_count}")
+
+ # Update statistics label
+ total_files = success_count + failed_count
+ self.batch_progress_label.setText(f"Completed: {total_files} files")
+
+ # Get output directory path
+ output_dir = self.get_batch_output_directory()
+
+ # Show batch success view (even if there were some failures, we still show the results)
+ self.show_batch_success_view(
+ total_files=total_files,
+ success_count=success_count,
+ failed_count=failed_count,
+ output_dir=output_dir,
+ format_name=self.batch_format_combo.currentText()
+ )
+
+ # Auto-reset after batch conversion completion
+ self._reset_batch_after_conversion()
+
+ def _reset_batch_after_conversion(self):
+ """Reset batch conversion state after completion to prepare for next conversion"""
+ # Clear file list and results
+ self.clear_batch_files()
+ self.clear_batch_results()
+
+ # Reset progress bars
+ self.batch_progress.setValue(0)
+ self.current_file_progress.setValue(0)
+ self.batch_progress_label.setText("Ready for new conversion")
+ self.current_file_label.setText("No file processing")
+
+ # Clear preview tab
+ if hasattr(self, 'preview_tab'):
+ self.preview_tab.clear_previews()
+
+ def on_batch_error(self, error_message):
+ """Handle batch conversion error"""
+ self.batch_converting = False
+ self.start_batch_btn.setEnabled(True)
+ self.stop_batch_btn.setEnabled(False)
+
+ PopupTeachingTip.create(
+ target=self.start_batch_btn,
+ icon=InfoBarIcon.ERROR,
+ title='ERROR',
+ content=f"Batch conversion failed: {error_message}",
+ isClosable=True,
+ tailPosition=TeachingTipTailPosition.TOP,
+ duration=5000,
+ parent=self
+ )
+
+ # Clean up thread
+ if hasattr(self, 'batch_thread') and self.batch_thread.isRunning():
+ self.batch_thread.quit()
+ self.batch_thread.wait()
+
+ def clear_batch_results(self):
+ """Clear batch conversion results"""
+ self.results_list_widget.clear()
+ self.success_count_label.setText("Success: 0")
+ self.failed_count_label.setText("Failed: 0")
+
+ def _get_batch_conversion_options(self):
+ """Get conversion options from batch options tree (same structure as single conversion)"""
+ try:
+ options = {
+ 'output_format': self.batch_format_combo.currentText(),
+ 'min_size': self.batch_min_spin.value(),
+ 'max_size': self.batch_max_spin.value(),
+ 'use_auto_detect': getattr(self, 'batch_use_auto_detect', False),
+ 'auto_detect_max_size': self.batch_auto_detect_check.isChecked() if hasattr(self, 'batch_auto_detect_check') else False,
+ 'keep_aspect_ratio': self.batch_keep_aspect_check.isChecked(),
+ 'auto_crop': self.batch_auto_crop_check.isChecked(),
+ 'quality': self.batch_quality_slider.value(),
+ 'icns_method': self.batch_icns_method_combo.currentText(),
+ 'overwrite_confirm': self.batch_overwrite_confirm_check.isChecked(),
+ 'preserve_folder_structure': self.batch_preserve_folder_check.isChecked() if hasattr(self, 'batch_preserve_folder_check') else False,
+ 'prefix': self.batch_prefix_edit.text().strip() if hasattr(self, 'batch_prefix_edit') else "",
+ 'suffix': self.batch_suffix_edit.text().strip() if hasattr(self, 'batch_suffix_edit') else ""
+ }
+ return options
+ except Exception as e:
+ PopupTeachingTip.create(
+ target=self.batch_options_tree,
+ icon=InfoBarIcon.ERROR,
+ title='ERROR',
+ content=f"Invalid conversion options: {str(e)}",
+ isClosable=True,
+ tailPosition=TeachingTipTailPosition.TOP,
+ duration=3000,
+ parent=self
+ )
+ return None
+
+ def get_output_directory(self):
+ """Get output directory for batch conversion"""
+ if hasattr(self, 'output_text') and self.output_text.text():
+ return self.output_text.text()
+ else:
+ # Default to current directory if no output path set
+ return os.path.dirname(self.input_text.text()) if hasattr(self, 'input_text') and self.input_text.text() else os.getcwd()
+
+ def get_batch_output_directory(self):
+ """Get output directory for batch conversion"""
+ if hasattr(self, 'batch_output_text') and self.batch_output_text.text():
+ return self.batch_output_text.text()
+ elif hasattr(self, 'output_text') and self.output_text.text():
+ # Fall back to single conversion output directory
+ return self.output_text.text()
+ else:
+ # Default to current directory if no output path set
+ return os.getcwd()
+
def show_success_view(self):
# Update the success widget geometry to match the window
self.success_widget.setGeometry(self.rect())
@@ -1496,24 +3067,12 @@ def show_main_view(self):
if hasattr(self, 'success_widget'):
self.success_widget.hide()
- # Only reset variables if remember_path is disabled
- if not self.remember_path:
- # Reset all variables when remember_path is disabled
- self.init_variables(reset_all=True)
-
- # Reset UI elements
- if hasattr(self, 'input_text'):
- self.input_text.clear()
- if hasattr(self, 'output_text'):
- self.output_text.clear()
- if hasattr(self, 'info_text'):
- self.info_text.setText("No image selected")
- if hasattr(self, 'preview_label'):
- self.preview_label.clear()
- self._set_placeholder_preview() # Show placeholder when returning to main view
- else:
- # If remember_path is enabled, only reset the UI state but keep the paths
- pass
+ # Hide the batch success widget if it's shown
+ if hasattr(self, 'batch_success_widget'):
+ self.batch_success_widget.hide()
+
+ # Reset conversion state when returning to main view
+ self._reset_after_conversion()
# Always reset these UI elements regardless of remember_path setting
if hasattr(self, 'min_spin'):
@@ -1548,6 +3107,35 @@ def show_main_view(self):
if hasattr(self, 'status_bar'):
self.status_bar.showMessage("Ready")
+ # Reset batch conversion related UI elements when returning to main view
+ if hasattr(self, 'batch_output_text'):
+ self.batch_output_text.clear() # Clear batch output directory
+ if hasattr(self, 'batch_format_combo'):
+ self.batch_format_combo.setCurrentText("icns") # Reset batch output format
+ if hasattr(self, 'batch_min_spin'):
+ self.batch_min_spin.setValue(16) # Reset batch min size
+ if hasattr(self, 'batch_max_spin'):
+ self.batch_max_spin.setValue(1024) # Reset batch max size
+ if hasattr(self, 'batch_options_tree'):
+ self.batch_options_tree.collapseAll() # Reset batch options tree
+ # Expand basic categories for batch options
+ if self.batch_options_tree.topLevelItemCount() >= 3:
+ item0 = self.batch_options_tree.topLevelItem(0)
+ item1 = self.batch_options_tree.topLevelItem(1)
+ item2 = self.batch_options_tree.topLevelItem(2)
+ if item0:
+ item0.setExpanded(True) # Basic Options
+ if item1:
+ item1.setExpanded(True) # Image Processing
+ if item2:
+ item2.setExpanded(False) # Advanced Settings
+ if hasattr(self, 'start_batch_btn'):
+ self.start_batch_btn.setEnabled(True) # Reset batch start button
+ if hasattr(self, 'stop_batch_btn'):
+ self.stop_batch_btn.setEnabled(False) # Reset batch stop button
+ if hasattr(self, 'results_list_widget'):
+ self.results_list_widget.clear() # Clear batch results list
+
# Only update history tab visibility if it's the first time or remember_path setting changed
if not hasattr(self, '_history_tab_initialized'):
self.create_history_tab()
@@ -1557,6 +3145,32 @@ def show_main_view(self):
def update_progress(self, message, percentage):
self.progress_label.setText(message)
self.progress.setValue(percentage)
+
+ # Update task progress file if task mode is enabled
+ if hasattr(self, 'current_task_id') and self.current_task_id:
+ try:
+ import json
+ import time
+
+ task_dir = os.path.expanduser("~/.converter/tasks")
+ task_file = os.path.join(task_dir, f"task_{self.current_task_id}.json")
+
+ # Read existing task info
+ if os.path.exists(task_file):
+ with open(task_file, "r") as f:
+ task_info = json.load(f)
+
+ # Update task info
+ task_info["status"] = "running"
+ task_info["progress"] = percentage
+ task_info["message"] = message
+ task_info["timestamp"] = time.time()
+
+ # Write updated task info
+ with open(task_file, "w") as f:
+ json.dump(task_info, f)
+ except Exception as e:
+ print(f"Error updating task progress: {e}")
def center_window(self):
qr = self.frameGeometry()
@@ -1610,6 +3224,9 @@ def resizeEvent(self, event):
# Update the success widget geometry when the window is resized
if hasattr(self, 'success_widget') and self.success_widget:
self.success_widget.setGeometry(self.rect())
+ # Update the batch success widget geometry when the window is resized
+ if hasattr(self, 'batch_success_widget') and self.batch_success_widget:
+ self.batch_success_widget.setGeometry(self.rect())
class ICNSConverterApp:
diff --git a/qss/converter_dark.qss b/qss/converter_dark.qss
index f14d2c4..d80a149 100644
--- a/qss/converter_dark.qss
+++ b/qss/converter_dark.qss
@@ -106,42 +106,3 @@ QTabBar::tab:!selected:hover {
background: qlineargradient(x1:0, y1:0, x2:0, y2:1, stop:0 #0056b3, stop:1 #004085);
}
-/* Enhanced CheckBox */
-QCheckBox {
- color: #e0e0e0;
- font-weight: 500;
-}
-
-QCheckBox::indicator {
- width: 18px;
- height: 18px;
- border-radius: 4px;
- border: 2px solid rgba(255, 255, 255, 0.2);
- background-color: rgba(45, 45, 45, 0.8);
-}
-
-QCheckBox::indicator:checked {
- background-color: #2196F3;
- border-color: #2196F3;
- image: url();
-}
-
-QCheckBox::indicator:hover {
- border-color: #2196F3;
- background-color: rgba(33, 150, 243, 0.1);
-}
-
-/* Success overlay styles */
-#success_overlay {
- background-color: rgba(0, 0, 0, 180);
-}
-
-#success_center_panel {
- background: qlineargradient(x1:0, y1:0, x2:0, y2:1,
- stop:0 rgba(60, 60, 60, 0.9),
- stop:1 rgba(45, 45, 45, 0.7));
- border-radius: 25px;
- border: 1px solid rgba(255, 255, 255, 0.1);
- padding: 35px;
- min-width: 420px;
-}
\ No newline at end of file
diff --git a/qss/settings_dark.qss b/qss/settings_dark.qss
deleted file mode 100644
index 3443e09..0000000
--- a/qss/settings_dark.qss
+++ /dev/null
@@ -1,259 +0,0 @@
-QDialog {
- background-color: #2b2b2b;
- color: #e0e0e0;
- font-family: Arial, sans-serif;
- border-radius: 20px;
-}
-
-QLabel {
- color: #e0e0e0;
- background-color: transparent;
-}
-
-/* Checkbox Styles */
-QCheckBox {
- color: #e0e0e0;
- background-color: transparent;
- spacing: 8px;
-}
-
-QCheckBox::indicator {
- width: 18px;
- height: 18px;
- background-color: transparent;
- border: 2px solid #5a5a5a;
- border-radius: 16px;
-}
-
-QCheckBox::indicator:checked {
- background-color: #0078d4;
- border-color: #0078d4;
- image: url("data:image/svg+xml;utf8,");
-}
-
-QCheckBox::indicator:hover {
- border-color: #0078d4;
- background-color: rgba(0, 120, 212, 0.1);
-}
-
-QCheckBox::indicator:checked:hover {
- background-color: #106ebe;
- border-color: #106ebe;
-}
-
-QCheckBox:focus {
- outline: none;
- background-color: transparent;
-}
-
-QPushButton {
- background-color: #2196F3;
- border: none;
- color: white;
- padding: 12px 20px;
- border-radius: 18px;
- font-weight: bold;
- font-size: 13px;
-}
-PushButton,ToolButton,PrimaryPushButton,PrimaryToolButton{
- border-radius: 16px
-}
-QPushButton:hover {
- background-color: #1976D2;
-}
-
-QPushButton:pressed {
- background-color: #0D47A1;
-}
-
-QComboBox {
- border: none;
- border-radius: 16px;
- padding: 8px 12px;
- background-color: rgba(45, 45, 45, 0.6);
- color: #ffffff;
- selection-background-color: #4a90e2;
- selection-color: #ffffff;
-}
-
-QComboBox:editable {
- background-color: rgba(45, 45, 45, 0.6);
-}
-
-/* Debug Settings Widget Styles */
-QGroupBox {
- border: none;
- border-radius: 20px;
- margin-top: 15px;
- padding-top: 25px;
- font-weight: bold;
- background-color: rgba(51, 51, 51, 0.4);
- color: #e0e0e0;
-}
-
-QGroupBox::title {
- subcontrol-origin: margin;
- left: 15px;
- padding: 0 8px 0 8px;
- color: #e0e0e0;
- background-color: transparent;
-}
-
-QComboBox:!editable, QComboBox::drop-down:editable {
- background-color: rgba(45, 45, 45, 0.6);
-}
-
-QComboBox:!editable:on, QComboBox::drop-down:editable:on {
- background-color: rgba(45, 45, 45, 0.6);
-}
-
-QComboBox:on {
- background-color: rgba(45, 45, 45, 0.6);
-}
-
-QComboBox::drop-down {
- subcontrol-origin: padding;
- subcontrol-position: top right;
- width: 24px;
- border: none;
- border-top-right-radius: 15px;
- border-bottom-right-radius: 15px;
- background-color: rgba(68, 68, 68, 0.7);
-}
-
-QComboBox::down-arrow {
- image: url("data:image/svg+xml;utf8,");
- width: 16px;
- height: 12px;
-}
-
-QComboBox::down-arrow:on {
- image: url("data:image/svg+xml;utf8,");
-}
-
-QComboBox QAbstractItemView {
- border: none;
- border-radius: 16px;
- background-color: rgba(45, 45, 45, 0.8);
- color: #ffffff;
- selection-background-color: #4a90e2;
- selection-color: #ffffff;
- outline: 0px;
-}
-
-QComboBox QAbstractItemView::item {
- padding: 8px 15px;
- border-radius: 16px;
- margin: 2px;
-}
-
-QComboBox QAbstractItemView::item:selected {
- background-color: #4a90e2;
- color: #ffffff;
-}
-
-QComboBox QAbstractItemView::item:hover {
- background-color: rgba(58, 58, 58, 0.8);
-}
-
-#status_label {
- color: #aaaaaa;
- font-size: 12px;
- font-style: italic;
- background-color: transparent;
-}
-
-#settings_apply_button {
- background-color: #4CAF50; /* Green for apply button */
- border-radius: 18px;
-}
-
-#settings_apply_button:hover {
- background-color: #43A047;
-}
-
-#settings_apply_button:pressed {
- background-color: #2E7D32;
-}
-
-/* Update Button Styles */
-QPushButton#settings_button {
- background-color: #2196F3; /* Blue */
- border: none;
- color: white;
- padding: 15px 20px;
- border-radius: 20px;
- font-weight: bold;
- font-size: 14px;
- text-align: center;
-}
-
-QPushButton#settings_button:hover {
- background-color: #1976D2;
-}
-
-QPushButton#settings_button:pressed {
- background-color: #0D47A1;
-}
-
-/* Text Elements with Transparent Backgrounds */
-QWidget QLabel {
- color: #e0e0e0;
- background-color: transparent;
-}
-
-/* TabWidget with Slide Animation */
-QTabWidget::pane {
- border: 1px solid #5a5a5a;
- border-radius: 20px;
- padding: 5px;
- background-color: rgba(45, 45, 45, 0.6);
-}
-
-QTabBar::tab {
- background: #404040;
- border: 1px solid #5a5a5a;
- border-radius: 4px 4px 0px 0px;
- padding: 12px 20px;
- margin-right: 4px;
- color: #b3b3b3;
- font-weight: 500;
-}
-
-QTabBar::tab:selected {
- background: #0078d4;
- color: #ffffff;
- border-bottom: 2px solid #0078d4;
-}
-
-QTabBar::tab:hover {
- background: qlineargradient(x1:0, y1:0, x2:0, y2:1, stop:0 #4a4a4a, stop:1 #3a3a3a);
- color: #e0e0e0;
-}
-
-QTabBar::tab:!selected:hover {
- background: qlineargradient(x1:0, y1:0, x2:0, y2:1, stop:0 #454545, stop:1 #353535);
- color: #cccccc;
-}
- background-color: transparent;
-}
-
-SegmentedWidget QLabel {
- background-color: transparent;
- color: #e0e0e0;
-}
-
-StackedWidget QLabel {
- background-color: transparent;
- color: #e0e0e0;
-}
-
-/* Ensure all text has transparent background */
-* {
- selection-background-color: rgba(0, 120, 212, 0.3);
-}
-
-*:focus {
- outline: none;
- background-color: transparent;
-}
\ No newline at end of file
diff --git a/qss/settings_light.qss b/qss/settings_light.qss
deleted file mode 100644
index 8b09c06..0000000
--- a/qss/settings_light.qss
+++ /dev/null
@@ -1,259 +0,0 @@
-QDialog {
- background-color: #f0f0f0;
- color: #333;
- font-family: Arial, sans-serif;
- border-radius: 20px;
-}
-
-QLabel {
- color: #333;
- background-color: transparent;
-}
-
-/* Checkbox Styles */
-QCheckBox {
- color: #333;
- background-color: transparent;
- spacing: 8px;
-}
-
-QCheckBox::indicator {
- width: 18px;
- height: 18px;
- background-color: transparent;
- border: 2px solid #bdc3c7;
- border-radius: 16px;
-}
-
-QCheckBox::indicator:checked {
- background-color: #3498db;
- border-color: #3498db;
- image: url("data:image/svg+xml;utf8,");
-}
-
-QCheckBox::indicator:hover {
- border-color: #3498db;
- background-color: rgba(52, 152, 219, 0.1);
-}
-
-QCheckBox::indicator:checked:hover {
- background-color: #2980b9;
- border-color: #2980b9;
-}
-
-QCheckBox:focus {
- outline: none;
- background-color: transparent;
-}
-
-QPushButton {
- background-color: #2196F3;
- border: none;
- color: white;
- padding: 12px 20px;
- border-radius: 18px;
- font-weight: bold;
- font-size: 13px;
-}
-PushButton,ToolButton,PrimaryPushButton,PrimaryToolButton{
- border-radius: 16px
-}
-QPushButton:hover {
- background-color: #1976D2;
-}
-
-QPushButton:pressed {
- background-color: #0D47A1;
-}
-
-QComboBox {
- border: none;
- border-radius: 16px;
- padding: 8px 12px;
- background-color: rgba(255, 255, 255, 0.7);
- color: #000000;
- selection-background-color: #4a90e2;
- selection-color: #ffffff;
-}
-
-QComboBox:editable {
- background-color: rgba(255, 255, 255, 0.7);
-}
-
-QComboBox:!editable, QComboBox::drop-down:editable {
- background-color: rgba(255, 255, 255, 0.7);
-}
-
-/* Debug Settings Widget Styles */
-QGroupBox {
- border: none;
- border-radius: 20px;
- margin-top: 15px;
- padding-top: 25px;
- font-weight: bold;
- background-color: rgba(248, 249, 250, 0.6);
- color: #333333;
-}
-
-QGroupBox::title {
- subcontrol-origin: margin;
- left: 15px;
- padding: 0 8px 0 8px;
- color: #333333;
- background-color: transparent;
-}
-
-QComboBox:!editable:on, QComboBox::drop-down:editable:on {
- background-color: rgba(255, 255, 255, 0.7);
-}
-
-QComboBox:on {
- background-color: rgba(255, 255, 255, 0.7);
-}
-
-QComboBox::drop-down {
- subcontrol-origin: padding;
- subcontrol-position: top right;
- width: 24px;
- border: none;
- border-top-right-radius: 15px;
- border-bottom-right-radius: 15px;
- background-color: rgba(204, 204, 204, 0.7);
-}
-
-QComboBox::down-arrow {
- image: url("data:image/svg+xml;utf8,");
- width: 16px;
- height: 12px;
-}
-
-QComboBox::down-arrow:on {
- image: url("data:image/svg+xml;utf8,");
-}
-
-QComboBox QAbstractItemView {
- border: none;
- border-radius: 16px;
- background-color: rgba(255, 255, 255, 0.9);
- color: #000000;
- selection-background-color: #4a90e2;
- selection-color: #ffffff;
- outline: 0px;
-}
-
-QComboBox QAbstractItemView::item {
- padding: 8px 15px;
- border-radius: 16px;
- margin: 2px;
-}
-
-QComboBox QAbstractItemView::item:selected {
- background-color: #4a90e2;
- color: #ffffff;
-}
-
-QComboBox QAbstractItemView::item:hover {
- background-color: rgba(0, 0, 0, 0.05);
-}
-
-#status_label {
- color: #555;
- font-size: 12px;
- font-style: italic;
- background-color: transparent;
-}
-
-#settings_apply_button {
- background-color: #4CAF50; /* Green for apply button */
- border-radius: 18px;
-}
-
-#settings_apply_button:hover {
- background-color: #43A047;
-}
-
-#settings_apply_button:pressed {
- background-color: #2E7D32;
-}
-
-/* Update Button Styles */
-QPushButton#settings_button {
- background-color: #2196F3; /* Blue */
- border: none;
- color: white;
- padding: 15px 20px;
- border-radius: 20px;
- font-weight: bold;
- font-size: 14px;
- text-align: center;
-}
-
-QPushButton#settings_button:hover {
- background-color: #1976D2;
-}
-
-QPushButton#settings_button:pressed {
- background-color: #0D47A1;
-}
-
-/* Text Elements with Transparent Backgrounds */
-QWidget QLabel {
- color: #333;
- background-color: transparent;
-}
-
-/* TabWidget with Slide Animation */
-QTabWidget::pane {
- border: 1px solid #bdc3c7;
- border-radius: 20px;
- padding: 5px;
- background-color: rgba(255, 255, 255, 0.7);
-}
-
-QTabBar::tab {
- background: #ecf0f1;
- border: 1px solid #bdc3c7;
- border-radius: 4px 4px 0px 0px;
- padding: 12px 20px;
- margin-right: 4px;
- color: #7f8c8d;
- font-weight: 500;
-}
-
-QTabBar::tab:selected {
- background: #3498db;
- color: #ffffff;
- border-bottom: 2px solid #3498db;
-}
-
-QTabBar::tab:hover {
- background: qlineargradient(x1:0, y1:0, x2:0, y2:1, stop:0 #d5dbdb, stop:1 #c5cbcb);
- color: #2c3e50;
-}
-
-QTabBar::tab:!selected:hover {
- background: qlineargradient(x1:0, y1:0, x2:0, y2:1, stop:0 #d0d3d4, stop:1 #c0c3c4);
- color: #34495e;
-}
- background-color: transparent;
-}
-
-SegmentedWidget QLabel {
- background-color: transparent;
- color: #333;
-}
-
-StackedWidget QLabel {
- background-color: transparent;
- color: #333;
-}
-
-/* Ensure all text has transparent background */
-* {
- selection-background-color: rgba(52, 152, 219, 0.3);
-}
-
-*:focus {
- outline: none;
- background-color: transparent;
-}
\ No newline at end of file
diff --git a/requirements.txt b/requirements.txt
index 5b2b4e0..155cf77 100644
--- a/requirements.txt
+++ b/requirements.txt
@@ -5,5 +5,4 @@ rarfile
py7zr
requests
PySide6-Fluent-Widgets[full]
-patool
cryptography
diff --git a/settings/general_settings.py b/settings/general_settings.py
index 5e4a635..aec90b9 100644
--- a/settings/general_settings.py
+++ b/settings/general_settings.py
@@ -2,149 +2,81 @@
General Settings Widget
"""
-from PySide6.QtWidgets import (QWidget, QVBoxLayout, QHBoxLayout, QGroupBox,
- QLabel, QLineEdit)
+from PySide6.QtWidgets import QWidget, QVBoxLayout
from PySide6.QtCore import Qt, Signal, QSettings
-from PySide6.QtGui import QIntValidator
-from qfluentwidgets import *
+from qfluentwidgets import (
+ SettingCardGroup, SwitchSettingCard, PushSettingCard, PrimaryPushSettingCard,
+ FluentIcon, BodyLabel, CaptionLabel, PasswordLineEdit, InfoBar, InfoBarPosition,
+ setCustomStyleSheet, ExpandGroupSettingCard, SingleDirectionScrollArea
+)
import os
import sys
sys.path.append(os.path.dirname(os.path.dirname(os.path.abspath(__file__))))
from utils.security import encrypt_pat, decrypt_pat
-from con import CON
-class GeneralSettingsWidget(QWidget):
- """General settings widget"""
-
- settings_changed = Signal()
+
+
+class GitHubSettingCard(ExpandGroupSettingCard):
+ """GitHub PAT settings card using ExpandGroupSettingCard"""
def __init__(self, parent=None):
- super().__init__(parent)
- self.settings = QSettings("pyquick", "converter")
- self.pat_input = None # Initialize PAT input field reference
- self.setup_ui()
- self.load_settings()
- self.connect_signals()
-
- def setup_ui(self):
- """Setup the UI layout"""
- main_layout = QVBoxLayout(self)
- main_layout.setContentsMargins(15, 15, 15, 15)
- main_layout.setSpacing(15)
+ super().__init__(
+ FluentIcon.GITHUB,
+ "GitHub Settings",
+ "Configure GitHub Personal Access Token for API access",
+ parent
+ )
+ from qfluentwidgets import ComboBox, PushButton
-
-
- # Image Converter settings group
- image_converter_group = QGroupBox("Image Converter Settings")
- image_converter_layout = QVBoxLayout(image_converter_group)
-
- # Interface behavior settings
- behavior_group = QGroupBox("Interface Behavior Settings")
- behavior_layout = QVBoxLayout(behavior_group)
-
- self.auto_preview_check = CheckBox("Auto-show preview after selecting file")
- behavior_layout.addWidget(self.auto_preview_check)
- behavior_layout.addSpacing(10)
- self.remember_path_check = CheckBox("Remember last selected input/output paths")
- behavior_layout.addWidget(self.remember_path_check)
- behavior_layout.addSpacing(10)
- self.completion_notify_check = CheckBox("Show success notification after conversion")
- behavior_layout.addWidget(self.completion_notify_check)
- behavior_layout.addSpacing(10)
-
- image_converter_layout.addWidget(behavior_group)
- image_converter_layout.addStretch()
-
- # GitHub Settings Group
- github_group = QGroupBox("GitHub Settings")
- github_layout = QVBoxLayout()
-
- # GitHub PAT Settings
- pat_layout = QHBoxLayout()
- pat_label = QLabel("GitHub PAT (Personal Access Token):")
self.pat_input = PasswordLineEdit()
self.pat_input.setPlaceholderText("ghp_xxxxxxxxxxxxxxxxxxxx")
self.pat_input.setClearButtonEnabled(True)
- self.pat_input.setFixedHeight(33)
- setCustomStyleSheet(self.pat_input,"PasswordLineEdit{ border-radius: 16px; }","PasswordLineEdit{ border-radius: 16px; }")
- pat_layout.addWidget(pat_label)
- pat_layout.addWidget(self.pat_input)
- github_layout.addLayout(pat_layout)
-
- # PAT Info
- pat_info = QLabel("Used for GitHub API access, recommend only 'repo' permission")
- pat_info.setStyleSheet("color: #666; font-size: 11px;")
- github_layout.addWidget(pat_info)
- self.qss_debug = """PushButton{ border-radius: 12px; }"""
- # PAT Test Button
- test_pat_btn = PushButton("Test PAT")
- setCustomStyleSheet(test_pat_btn,self.qss_debug,self.qss_debug)
- test_pat_btn.clicked.connect(self.test_pat)
- github_layout.addWidget(test_pat_btn)
-
- github_group.setLayout(github_layout)
- main_layout.addWidget(github_group)
- main_layout.addWidget(image_converter_group)
- main_layout.addStretch()
-
- def connect_signals(self):
- """Connect signals for auto-save"""
- # Image converter settings signals
- self.auto_preview_check.stateChanged.connect(self.on_settings_changed)
- self.remember_path_check.stateChanged.connect(self.on_settings_changed)
- self.completion_notify_check.stateChanged.connect(self.on_settings_changed)
-
- # Placeholder for future general settings signals
- pass
-
- def on_settings_changed(self):
- """Emit settings changed signal and save settings"""
- self.save_settings()
- self.settings_changed.emit()
-
- def load_settings(self):
- """Load settings from QSettings"""
- settings = QSettings("MyCompany", "ConverterApp")
+ self.pat_input.setFixedHeight(40)
+ setCustomStyleSheet(self.pat_input, "PasswordLineEdit{ border-radius: 16px; }", "PasswordLineEdit{ border-radius: 16px; }")
- # Image converter settings
- self.auto_preview_check.setChecked(settings.value("image_converter/auto_preview", True, type=bool))
- self.remember_path_check.setChecked(settings.value("image_converter/remember_path", True, type=bool))
- self.completion_notify_check.setChecked(settings.value("image_converter/completion_notify", True, type=bool))
+ self.test_button = PushButton("Test PAT")
+ self.test_button.setFixedWidth(135)
+ self.test_button.setFixedHeight(46)
+ self.test_button.clicked.connect(self.test_pat)
- # GitHub PAT settings
- encrypted_pat = settings.value("general/github_pat", "", type=str)
- if encrypted_pat:
- decrypted_pat = decrypt_pat(encrypted_pat)
- self.pat_input.setText(decrypted_pat)
-
- def save_settings(self):
- """Save settings to QSettings"""
- settings = QSettings("MyCompany", "ConverterApp")
+ self.permission_combo = ComboBox()
+ self.permission_combo.addItems(["repo", "public_repo", "repo:status", "repo_deployment"])
+ self.permission_combo.setFixedWidth(135)
+ self.permission_combo.setFixedHeight(46)
+ self.permission_combo.setCurrentText("repo")
- # Image converter settings
- settings.setValue("image_converter/auto_preview", self.auto_preview_check.isChecked())
- settings.setValue("image_converter/remember_path", self.remember_path_check.isChecked())
- settings.setValue("image_converter/completion_notify", self.completion_notify_check.isChecked())
+ self.viewLayout.setContentsMargins(0, 0, 0, 0)
+ self.viewLayout.setSpacing(0)
- # GitHub PAT settings
- pat_text = self.pat_input.text().strip()
- if pat_text:
- encrypted_pat = encrypt_pat(pat_text)
- settings.setValue("general/github_pat", encrypted_pat)
- else:
- settings.setValue("general/github_pat", "")
-
+ self.addGroup(
+ FluentIcon.SAVE,
+ "Personal Access Token",
+ "Enter your GitHub PAT for API access",
+ self.pat_input
+ )
+ self.addGroup(
+ FluentIcon.CHECKBOX,
+ "Recommended Permission",
+ "Select the permission scope for your PAT",
+ self.permission_combo
+ )
+ self.addGroup(
+ FluentIcon.EDIT,
+ "Test Connection",
+ "Verify your PAT is valid",
+ self.test_button
+ )
+
def test_pat(self):
"""Test GitHub PAT validity"""
pat = self.pat_input.text().strip()
if not pat:
- PopupTeachingTip.create(
- target=self.pat_input,
- icon=InfoBarIcon.WARNING,
+ InfoBar.warning(
title='Warning',
content='Please enter GitHub PAT first',
+ orient=Qt.Orientation.Horizontal,
isClosable=True,
- tailPosition=TeachingTipTailPosition.TOP,
+ position=InfoBarPosition.TOP,
duration=2000,
parent=self
)
@@ -152,7 +84,6 @@ def test_pat(self):
try:
import requests
- # Test PAT API endpoint
response = requests.get(
"https://api.github.com/user",
headers={
@@ -165,61 +96,211 @@ def test_pat(self):
if response.status_code == 200:
user_data = response.json()
username = user_data.get("login", "Unknown")
- PopupTeachingTip.create(
- target=self.pat_input,
- icon=InfoBarIcon.SUCCESS,
+ InfoBar.success(
title='Success',
content=f'PAT verified! Associated user: {username}',
+ orient=Qt.Orientation.Horizontal,
isClosable=True,
- tailPosition=TeachingTipTailPosition.TOP,
+ position=InfoBarPosition.TOP,
duration=3000,
parent=self
)
elif response.status_code == 401:
- PopupTeachingTip.create(
- target=self.pat_input,
- icon=InfoBarIcon.ERROR,
+ InfoBar.error(
title='Error',
content='PAT invalid or expired',
+ orient=Qt.Orientation.Horizontal,
isClosable=True,
- tailPosition=TeachingTipTailPosition.TOP,
+ position=InfoBarPosition.TOP,
duration=3000,
parent=self
)
else:
- PopupTeachingTip.create(
- target=self.pat_input,
- icon=InfoBarIcon.ERROR,
+ InfoBar.error(
title='Error',
content=f'PAT verification failed (status code: {response.status_code})',
+ orient=Qt.Orientation.Horizontal,
isClosable=True,
- tailPosition=TeachingTipTailPosition.TOP,
+ position=InfoBarPosition.TOP,
duration=3000,
parent=self
)
except requests.exceptions.RequestException as e:
- PopupTeachingTip.create(
- target=self.pat_input,
- icon=InfoBarIcon.ERROR,
+ InfoBar.error(
title='Error',
content=f'Network error: {str(e)}',
+ orient=Qt.Orientation.Horizontal,
isClosable=True,
- tailPosition=TeachingTipTailPosition.TOP,
+ position=InfoBarPosition.TOP,
duration=3000,
parent=self
)
except Exception as e:
- PopupTeachingTip.create(
- target=self.pat_input,
- icon=InfoBarIcon.ERROR,
+ InfoBar.error(
title='Error',
content=f'Test failed: {str(e)}',
+ orient=Qt.Orientation.Horizontal,
isClosable=True,
- tailPosition=TeachingTipTailPosition.TOP,
- duration=3000,
- parent=self
- )
+ position=InfoBarPosition.TOP,
+ duration=3000,
+ parent=self
+ )
+
+ def get_pat(self):
+ """Get current PAT text"""
+ return self.pat_input.text().strip()
+
+ def set_pat(self, pat):
+ """Set PAT text"""
+ self.pat_input.setText(pat)
+
+ def get_permission(self):
+ """Get selected permission"""
+ return self.permission_combo.currentText()
+
+ def set_permission(self, permission):
+ """Set permission"""
+ self.permission_combo.setCurrentText(permission)
+
+
+class GeneralSettingsWidget(QWidget):
+ """General settings widget using qfluentwidgets SettingCard components"""
+
+ settings_changed = Signal()
+
+ def __init__(self, parent=None):
+ super().__init__(parent)
+ self.settings = QSettings("MyCompany", "ConverterApp")
+ self.setup_ui()
+ self.connect_signals()
+ self.load_settings()
+
+ def setup_ui(self):
+ """Setup the UI layout"""
+ main_layout = QVBoxLayout(self)
+ main_layout.setContentsMargins(0, 0, 0, 0)
+ main_layout.setSpacing(0)
+
+ self.scroll_area = SingleDirectionScrollArea(orient=Qt.Orientation.Vertical)
+ self.scroll_area.setWidgetResizable(True)
+ self.scroll_area.enableTransparentBackground()
+
+ scroll_content = QWidget()
+ scroll_content.setObjectName("scroll_content")
+ scroll_layout = QVBoxLayout(scroll_content)
+ scroll_layout.setContentsMargins(30, 30, 30, 30)
+ scroll_layout.setSpacing(20)
+
+ self.behavior_group = SettingCardGroup("Interface Behavior", scroll_content)
+
+ self.auto_preview_card = SwitchSettingCard(
+ FluentIcon.VIEW,
+ "Auto-show preview",
+ "Automatically show preview after selecting file",
+ parent=self.behavior_group
+ )
+
+ self.remember_path_card = SwitchSettingCard(
+ FluentIcon.HISTORY,
+ "Remember paths",
+ "Remember last selected input/output paths",
+ parent=self.behavior_group
+ )
+
+ self.completion_notify_card = SwitchSettingCard(
+ FluentIcon.COMPLETED,
+ "Completion notification",
+ "Show success notification after conversion",
+ parent=self.behavior_group
+ )
+
+ self.task_mode_card = SwitchSettingCard(
+ FluentIcon.SETTING,
+ "Task Mode",
+ "Enable task queue for batch processing",
+ parent=self.behavior_group
+ )
+
+ self.github_card = GitHubSettingCard(scroll_content)
+
+ self.behavior_group.addSettingCards([
+ self.auto_preview_card,
+ self.remember_path_card,
+ self.completion_notify_card,
+ self.task_mode_card
+ ])
+
+ scroll_layout.addWidget(self.behavior_group)
+ scroll_layout.addWidget(self.github_card)
+ scroll_layout.addStretch()
+
+ self.scroll_area.setWidget(scroll_content)
+
+ main_layout.addWidget(self.scroll_area)
+
+ def connect_signals(self):
+ """Connect signals for auto-save"""
+ self.auto_preview_card.checkedChanged.connect(self.on_settings_changed)
+ self.remember_path_card.checkedChanged.connect(self.on_settings_changed)
+ self.completion_notify_card.checkedChanged.connect(self.on_settings_changed)
+ self.task_mode_card.checkedChanged.connect(self.on_settings_changed)
+
+ # Connect GitHub settings signals
+ self.github_card.pat_input.textChanged.connect(self.on_settings_changed)
+ self.github_card.permission_combo.currentTextChanged.connect(self.on_settings_changed)
+
+ def on_settings_changed(self):
+ """Emit settings changed signal and save settings"""
+ self.save_settings()
+ self.settings_changed.emit()
+
+ def load_settings(self):
+ """Load settings from QSettings"""
+ # Block signals during loading to prevent auto-save
+ self.auto_preview_card.blockSignals(True)
+ self.remember_path_card.blockSignals(True)
+ self.completion_notify_card.blockSignals(True)
+ self.task_mode_card.blockSignals(True)
+ self.github_card.pat_input.blockSignals(True)
+ self.github_card.permission_combo.blockSignals(True)
+
+ self.auto_preview_card.setChecked(bool(self.settings.value("image_converter/auto_preview", True, type=bool)))
+ self.remember_path_card.setChecked(bool(self.settings.value("image_converter/remember_path", True, type=bool)))
+ self.completion_notify_card.setChecked(bool(self.settings.value("image_converter/completion_notify", True, type=bool)))
+ self.task_mode_card.setChecked(bool(self.settings.value("task_mode", False, type=bool)))
+
+ encrypted_pat = self.settings.value("general/github_pat", "", type=str)
+ if encrypted_pat:
+ decrypted_pat = decrypt_pat(str(encrypted_pat))
+ self.github_card.set_pat(decrypted_pat)
+
+ permission = self.settings.value("general/github_permission", "repo", type=str)
+ self.github_card.set_permission(str(permission))
+
+ # Unblock signals after loading
+ self.auto_preview_card.blockSignals(False)
+ self.remember_path_card.blockSignals(False)
+ self.completion_notify_card.blockSignals(False)
+ self.task_mode_card.blockSignals(False)
+ self.github_card.pat_input.blockSignals(False)
+ self.github_card.permission_combo.blockSignals(False)
+
+ def save_settings(self):
+ """Save settings to QSettings"""
+ self.settings.setValue("image_converter/auto_preview", self.auto_preview_card.isChecked())
+ self.settings.setValue("image_converter/remember_path", self.remember_path_card.isChecked())
+ self.settings.setValue("image_converter/completion_notify", self.completion_notify_card.isChecked())
+ self.settings.setValue("task_mode", self.task_mode_card.isChecked())
+
+ pat_text = self.github_card.get_pat()
+ if pat_text:
+ encrypted_pat = encrypt_pat(pat_text)
+ self.settings.setValue("general/github_pat", encrypted_pat)
+ else:
+ self.settings.setValue("general/github_pat", "")
+
+ self.settings.setValue("general/github_permission", self.github_card.get_permission())
if __name__ == "__main__":
@@ -229,4 +310,4 @@ def test_pat(self):
app = QApplication(sys.argv)
widget = GeneralSettingsWidget()
widget.show()
- sys.exit(app.exec())
\ No newline at end of file
+ sys.exit(app.exec())
diff --git a/settings/settings_gui.py b/settings/settings_gui.py
index 27526bc..3d9671d 100644
--- a/settings/settings_gui.py
+++ b/settings/settings_gui.py
@@ -15,7 +15,7 @@
from PySide6.QtGui import QPalette
from PySide6.QtCore import Qt, QSettings, QPropertyAnimation, QEasingCurve, Signal, Slot
from qfluentwidgets import SegmentedWidget, setCustomStyleSheet
-from .update_settings_gui import UpdateDialog
+from .update_settings_gui import UpdateSettingsWidget
from con import CON
class SettingsDialog(QDialog):
@@ -133,7 +133,7 @@ def init_ui(self):
update_layout.setContentsMargins(15, 15, 15, 15)
update_layout.setSpacing(15)
- self.update_group = UpdateDialog()
+ self.update_group = UpdateSettingsWidget()
self.update_group.setObjectName("update_group")
update_layout.addWidget(self.update_group)
update_layout.addStretch()
diff --git a/settings/update_settings_gui.py b/settings/update_settings_gui.py
index fd3519c..654059b 100644
--- a/settings/update_settings_gui.py
+++ b/settings/update_settings_gui.py
@@ -1,35 +1,21 @@
# -*- coding: utf-8 -*-
-#Please do not change import
-from concurrent.futures import thread
-import multiprocessing
+"""
+Update Settings GUI Widget for Converter application
+"""
+
import sys
import os
-import threading
-
-from PySide6.QtWidgets import (
- QApplication,
- QWidget,
- QPushButton,
- QVBoxLayout,
- QHBoxLayout,
- QLabel,
- QCheckBox,
- QGroupBox,
- QSpacerItem,
- QSizePolicy,
- QProgressBar,
- QMessageBox,
- QAbstractButton
-)
-from PySide6.QtCore import QSettings, Qt, QThread, Signal, QTimer, QTimer
+from PySide6.QtWidgets import QWidget, QVBoxLayout, QHBoxLayout
+from PySide6.QtCore import QSettings, Qt, QThread, Signal, QTimer
from PySide6.QtGui import QFont
+from qfluentwidgets import *
from darkdetect import isDark
from update.update_manager import UpdateManager
from update.download_update import download_and_apply_update
from qframelesswindow.utils import getSystemAccentColor
-from qfluentwidgets import *
-import time
from con import CON
+
+
class CheckUpdateThread(QThread):
check_finished = Signal(dict)
@@ -58,13 +44,12 @@ def __init__(self, download_url, version, include_prerelease=False):
self.include_prerelease = include_prerelease
self.is_cancelled = False
self.progress_callback = None
- self.downloader = None # Initialize downloader attribute
+ self.downloader = None
def run(self):
try:
print(f"DownloadThread started, is_cancelled: {self.is_cancelled}")
- # Check if cancelled before starting download
if self.is_cancelled:
print("DownloadThread: cancelled before download")
result = {
@@ -74,7 +59,6 @@ def run(self):
self.finished.emit(result)
return
- # Create a simple progress callback that emits signals
def progress_callback(progress, downloaded, total_size):
if not self.is_cancelled:
progress_data = {
@@ -84,16 +68,13 @@ def progress_callback(progress, downloaded, total_size):
}
self.progress_updated.emit(progress_data)
- # Set the progress callback
self.progress_callback = progress_callback
- # Create update info dictionary
update_info = {
"download_url": self.download_url,
"latest_version": self.version
}
- # Create UpdateDownloader instance for cancellation
from update.download_update import UpdateDownloader
import tempfile
temp_dir = tempfile.mkdtemp(prefix="update_")
@@ -102,21 +83,18 @@ def progress_callback(progress, downloaded, total_size):
self.downloader = UpdateDownloader(
download_url=self.download_url,
target_directory=temp_dir,
- max_threads=64 # Use 64 threads
+ max_threads=64
)
- # Download update only (don't apply it)
print("DownloadThread: starting download")
result = self.downloader.download_update(self.version, progress_callback)
print(f"DownloadThread: download completed with status: {result.get('status')}")
- # Check if the download was cancelled
if result.get("status") == "cancelled":
print("DownloadThread: result status is cancelled")
self.finished.emit(result)
return
- # Check if cancelled during download
if self.is_cancelled or (self.downloader and hasattr(self.downloader, '_cancelled') and self.downloader._cancelled):
print("DownloadThread: cancelled during download")
result = {
@@ -127,7 +105,6 @@ def progress_callback(progress, downloaded, total_size):
self.finished.emit(result)
return
- # Check again if cancelled
if self.is_cancelled:
print("DownloadThread: cancelled after download")
result = {
@@ -138,7 +115,6 @@ def progress_callback(progress, downloaded, total_size):
self.finished.emit(result)
return
- # Add downloader to result for cleanup
result["downloader"] = self.downloader
result["temp_dir"] = temp_dir
print("DownloadThread: emitting success result")
@@ -146,10 +122,15 @@ def progress_callback(progress, downloaded, total_size):
except Exception as e:
print(f"DownloadThread exception: {e}")
+ import shutil
+ try:
+ shutil.rmtree(temp_dir)
+ except Exception as cleanup_error:
+ print(f"Failed to clean up temp directory: {cleanup_error}")
+
error_result = {
"status": "error",
- "message": f"Download failed: {str(e)}",
- "temp_dir": temp_dir
+ "message": f"Download failed: {str(e)}"
}
self.finished.emit(error_result)
@@ -162,214 +143,192 @@ def progress_callback(progress, downloaded, total_size):
self.finished.emit(error_result)
def cancel(self):
- """Cancel the download"""
+ """Cancel download"""
print("DownloadThread: cancel() called")
self.is_cancelled = True
- # Cancel the downloader
if self.downloader:
print("DownloadThread: cancelling downloader")
self.downloader.cancel()
- # Wait a moment to ensure cancellation is complete
import time
time.sleep(0.1)
- # Request thread exit
self.quit()
- # Wait for thread to completely stop
- if not self.wait(1000): # Wait up to 1 second
+ if not self.wait(1000):
print("Warning: DownloadThread did not stop within 1 second, forcing termination")
self.terminate()
- self.wait(500) # Wait another 500ms to ensure termination is complete
+ self.wait(500)
-class UpdateDialog(QWidget):
- __version__ = "2.0.0RC1"
- def __init__(self):
- super().__init__()
- # Remove SystemThemeListener to avoid thread issues
- # self.themeListener = SystemThemeListener(self)
-
-
- self.setWindowTitle("Update Settings")
+class UpdateSettingsWidget(QWidget):
+ """Update settings widget using qfluentwidgets SettingCard components"""
+
+ __version__ = "2.1.0A6"
+
+ def __init__(self, parent=None):
+ super().__init__(parent)
self.update_manager = UpdateManager(self.__version__)
self.check_thread = None
self.download_thread = None
-
- # Detect current version type
- self._detect_current_version_type()
+ self.current_update_info = None
- self.init_ui()
+ self._detect_current_version_type()
+ self.setup_ui()
self.load_settings()
- self.connect_auto_save_signals()
-
- def _detect_current_version_type(self):
- """检测当前版本类型,如果是Alpha或Deepdev版本,则限制可用通道"""
- try:
- # 解析当前版本
- version_info = self.update_manager._parse_version(self.__version__)
- _, _, _, current_tag, _ = version_info
-
- # 检查是否为Alpha或Deepdev版本
- self.is_alpha_version = (current_tag == 'A')
- self.is_deepdev_version = (current_tag == 'D')
- self.is_internal_version = self.is_alpha_version or self.is_deepdev_version
-
- if self.is_internal_version:
- print(f"检测到内部版本: {self.__version__} (标签: {current_tag})")
-
- except Exception as e:
- print(f"版本检测失败: {e}")
- self.is_internal_version = False
- self.is_alpha_version = False
- self.is_deepdev_version = False
+ self.connect_signals()
-
- def init_ui(self):
-
+ def setup_ui(self):
+ """Setup UI layout"""
main_layout = QVBoxLayout(self)
- main_layout.setContentsMargins(25, 25, 25, 25)
- main_layout.setSpacing(20)
-
- update_group = QGroupBox("Update Settings")
- update_layout = QVBoxLayout()
- update_layout.setContentsMargins(25, 25, 25, 25)
- update_layout.setSpacing(20)
-
- # 添加顶部间距
- update_layout.addSpacerItem(QSpacerItem(0, 15, QSizePolicy.Policy.Minimum, QSizePolicy.Policy.Fixed))
-
- # 添加版本类型选择器
- prerelease_container = QHBoxLayout()
- prerelease_container.setSpacing(10)
-
- self.prerelease_type_label = QLabel("Update channel:")
- prerelease_container.addWidget(self.prerelease_type_label)
-
- self.prerelease_type_combo = ModelComboBox()
- setCustomStyleSheet(self.prerelease_type_combo,CON.qss_combo_2,CON.qss_combo_2)
- # 根据版本类型设置可用通道
+ main_layout.setContentsMargins(0, 0, 0, 0)
+ main_layout.setSpacing(0)
+
+ # Create scroll area
+ self.scroll_area = SingleDirectionScrollArea(orient=Qt.Orientation.Vertical)
+ self.scroll_area.setWidgetResizable(True)
+ self.scroll_area.enableTransparentBackground()
+
+ # Create scroll content widget
+ scroll_content = QWidget()
+ scroll_content.setObjectName("scroll_content")
+ scroll_layout = QVBoxLayout(scroll_content)
+ scroll_layout.setContentsMargins(30, 30, 30, 30)
+ scroll_layout.setSpacing(20)
+
+ # Create setting card groups
+ self.update_group = SettingCardGroup("Update Configuration", scroll_content)
+
+ # Determine available channels based on version type
+ available_channels = ["Stable"]
if self.is_internal_version:
- # 内部版本只显示对应的通道和稳定版
if self.is_alpha_version:
- self.prerelease_type_combo.addItems([ "Alpha"])
+ available_channels = ["Stable", "Alpha"]
print("Alpha版本:只显示Alpha通道")
elif self.is_deepdev_version:
- self.prerelease_type_combo.addItems(["Deepdev"])
+ available_channels = ["Stable", "Deepdev"]
print("Deepdev版本:只显示Deepdev通道")
else:
- # 普通版本显示所有通道
- self.prerelease_type_combo.addItems(["Stable", "RC (Release Candidate)", "Beta", "Deepdev", "Alpha"])
-
- self.prerelease_type_combo.setFixedWidth(200)
- prerelease_container.addWidget(self.prerelease_type_combo)
-
- prerelease_container.addStretch()
- update_layout.addLayout(prerelease_container)
-
- self.update_status_label = QLabel("Ready to check for updates.")
- self.update_status_label.setMinimumHeight(60)
- self.update_status_label.setMinimumWidth(550)
+ available_channels = ["Stable", "RC (Release Candidate)", "Beta", "Deepdev", "Alpha"]
+
+ channel_layout = QHBoxLayout()
+ channel_label = BodyLabel("Update Channel:")
+ channel_label.setFixedWidth(120)
+ self.channel_combo = ComboBox()
+ self.channel_combo.addItems(available_channels)
+ self.channel_combo.setFixedWidth(250)
+ channel_layout.addWidget(channel_label)
+ channel_layout.addWidget(self.channel_combo)
+ channel_layout.addStretch()
+ self.update_group.vBoxLayout.addLayout(channel_layout)
+
+ channel_info = CaptionLabel("Select update channel to receive updates from")
+ self.update_group.vBoxLayout.addWidget(channel_info)
+
+ # Create check for updates card
+ self.check_update_card = PrimaryPushSettingCard(
+ "Check for Updates",
+ FluentIcon.SYNC,
+ "Check for new updates"
+ )
+ self.check_update_card.clicked.connect(self.check_for_updates)
+
+ # Create download update card (initially disabled)
+ self.download_update_card = PrimaryPushSettingCard(
+ "Download Update",
+ FluentIcon.DOWNLOAD,
+ "Download and install latest update"
+ )
+ self.download_update_card.clicked.connect(self.download_update)
+ self.download_update_card.setEnabled(False)
+
+ # Create restart application card (initially disabled)
+ self.restart_card = PrimaryPushSettingCard(
+ "Restart Application",
+ FluentIcon.SYNC,
+ "Restart to apply update"
+ )
+ self.restart_card.clicked.connect(self.restart_application)
+ self.restart_card.setEnabled(False)
+
+ # Add cards to group
+ self.update_group.addSettingCards([
+ self.check_update_card,
+ self.download_update_card,
+ self.restart_card
+ ])
+
+ # Add update status and progress to group
+ self.update_status_label = BodyLabel("Ready to check for updates.")
self.update_status_label.setWordWrap(True)
- #self.update_status_label.setStyleSheet("QLabel { padding: 8px; background-color: #f8f9fa; border-radius: 5px; }")
- update_layout.addWidget(self.update_status_label)
+ self.update_group.vBoxLayout.addWidget(self.update_status_label)
- # 更新内容显示区域
+ # Release content browser
self.release_content_browser = TextBrowser()
self.release_content_browser.setMinimumHeight(150)
self.release_content_browser.setVisible(False)
- update_layout.addWidget(self.release_content_browser)
+ self.update_group.vBoxLayout.addWidget(self.release_content_browser)
- # 下载进度条
+ # Progress bars
self.progress_bar = IndeterminateProgressBar()
self.progress_bar.setVisible(False)
- update_layout.addWidget(self.progress_bar)
+ self.update_group.vBoxLayout.addWidget(self.progress_bar)
- # 实际下载进度条
self.download_progress_bar = ProgressBar()
self.download_progress_bar.setRange(0, 100)
self.download_progress_bar.setValue(0)
self.download_progress_bar.setVisible(False)
- update_layout.addWidget(self.download_progress_bar)
+ self.update_group.vBoxLayout.addWidget(self.download_progress_bar)
- # 进度标签(显示百分比、下载大小、速度等信息)
- self.progress_label = QLabel("0%")
+ # Progress label
+ self.progress_label = BodyLabel("0%")
self.progress_label.setAlignment(Qt.AlignmentFlag.AlignCenter)
font = QFont()
font.setPointSize(10)
font.setBold(True)
self.progress_label.setFont(font)
self.progress_label.setVisible(False)
- update_layout.addWidget(self.progress_label)
-
- # 按钮容器
- button_container = QHBoxLayout()
- button_container.setSpacing(15)
-
- self.update_button = PrimaryPushButton("Check for Updates")
- self.update_button.setFixedSize(180, 60)
- setCustomStyleSheet(self.update_button, CON.qss_debug, CON.qss_debug)
- self.update_button.clicked.connect(self.check_for_updates)
-
- self.download_button =PrimaryPushButton("Download Update")
- self.download_button.setFixedSize(180, 60)
- setCustomStyleSheet(self.download_button, CON.qss_debug, CON.qss_debug)
- self.download_button.clicked.connect(self.download_update)
- self.download_button.setVisible(False)
- self.download_button.setEnabled(False)
-
- self.restart_button = PrimaryPushButton("Restart Application")
- self.restart_button.setFixedSize(180, 60)
- setCustomStyleSheet(self.restart_button, CON.qss_debug, CON.qss_debug)
- self.restart_button.clicked.connect(self.restart_application)
- self.restart_button.setVisible(False)
- self.restart_button.setEnabled(False)
-
- button_container.addStretch()
- button_container.addWidget(self.update_button)
- button_container.addWidget(self.download_button)
- button_container.addWidget(self.restart_button)
- button_container.addStretch()
- update_layout.addLayout(button_container)
-
- # 添加底部间距
- update_layout.addSpacerItem(QSpacerItem(0, 15, QSizePolicy.Policy.Minimum, QSizePolicy.Policy.Fixed))
-
- update_group.setLayout(update_layout)
- main_layout.addWidget(update_group)
-
- self.setLayout(main_layout)
-
-
+ self.update_group.vBoxLayout.addWidget(self.progress_label)
- # 设置按钮字体
- font = QFont()
- font.setPointSize(12)
- font.setBold(True)
- self.update_button.setFont(font)
- self.download_button.setFont(font)
- self.restart_button.setFont(font)
-
- # 设置标签字体
- label_font = QFont()
- label_font.setPointSize(11)
- self.update_status_label.setFont(label_font)
-
- # 设置标签字体
- label_font = QFont()
- label_font.setPointSize(11)
- self.prerelease_type_label.setFont(label_font)
-
-
-
-
-
+ # Add groups to scroll layout
+ scroll_layout.addWidget(self.update_group)
+ scroll_layout.addStretch()
+
+ # Set scroll content
+ self.scroll_area.setWidget(scroll_content)
+
+ # Add scroll area to main layout
+ main_layout.addWidget(self.scroll_area)
+
+ def connect_signals(self):
+ """Connect signals"""
+ self.channel_combo.currentIndexChanged.connect(self.on_update_channel_changed)
+
+ def _detect_current_version_type(self):
+ """Detect current version type"""
+ try:
+ version_info = self.update_manager._parse_version(self.__version__)
+ _, _, _, current_tag, _ = version_info
+
+ self.is_alpha_version = (current_tag == 'A')
+ self.is_deepdev_version = (current_tag == 'D')
+ self.is_internal_version = self.is_alpha_version or self.is_deepdev_version
+
+ if self.is_internal_version:
+ print(f"检测到内部版本: {self.__version__} (标签: {current_tag})")
+
+ except Exception as e:
+ print(f"版本检测失败: {e}")
+ self.is_internal_version = False
+ self.is_alpha_version = False
+ self.is_deepdev_version = False
+
def load_settings(self):
+ """Load settings from QSettings"""
settings = QSettings("MyCompany", "ConverterApp")
prerelease_type = settings.value("update/prerelease_type", "stable", type=str)
- # 设置版本类型选择器
- type_index = 0 # 默认为 "Stable"
+
+ type_index = 0
if prerelease_type == "rc":
type_index = 1
elif prerelease_type == "beta":
@@ -378,16 +337,15 @@ def load_settings(self):
type_index = 3
elif prerelease_type == "alpha":
type_index = 4
- self.prerelease_type_combo.setCurrentIndex(type_index)
-
+
+ self.channel_combo.setCurrentIndex(type_index)
+
def save_settings(self):
- """保存设置"""
+ """Save settings to QSettings"""
try:
- # 保存更新通道设置
- prerelease_type = "stable" # 默认
+ prerelease_type = "stable"
- # 根据选择的索引确定类型
- current_index = self.prerelease_type_combo.currentIndex()
+ current_index = self.channel_combo.currentIndex()
if current_index == 1:
prerelease_type = "rc"
elif current_index == 2:
@@ -404,43 +362,25 @@ def save_settings(self):
except Exception as e:
print(f"Error saving settings: {e}")
- def connect_auto_save_signals(self):
- """Connect UI controls to auto-save functionality"""
- # Connect combo box to auto-save
- self.prerelease_type_combo.currentIndexChanged.connect(self.on_update_channel_changed)
-
- def auto_save_settings(self):
- """Auto-save settings immediately upon change"""
- try:
- # Save settings immediately
- self.save_settings()
- except Exception as e:
- print(f"Error in auto_save_settings: {e}")
-
def on_update_channel_changed(self, index):
- """处理更新通道选择变化"""
- # 自动保存设置
- self.auto_save_settings()
+ """Handle update channel change"""
+ self.save_settings()
def _get_update_check_params(self):
- """获取更新检查参数"""
- prerelease_type = "stable" # 默认稳定版
+ """Get update check parameters"""
+ prerelease_type = "stable"
- current_index = self.prerelease_type_combo.currentIndex()
- current_text = self.prerelease_type_combo.currentText()
+ current_index = self.channel_combo.currentIndex()
+ current_text = self.channel_combo.currentText()
- # 根据版本类型调整索引映射
if self.is_internal_version:
if self.is_alpha_version:
- # Alpha版本:索引0=Stable, 索引1=Alpha
if current_index == 1:
prerelease_type = "alpha"
elif self.is_deepdev_version:
- # Deepdev版本:索引0=Stable, 索引1=Deepdev
if current_index == 1:
prerelease_type = "deepdev"
else:
- # 普通版本:标准索引映射
if current_index == 1:
prerelease_type = "rc"
elif current_index == 2:
@@ -450,42 +390,37 @@ def _get_update_check_params(self):
elif current_index == 4:
prerelease_type = "alpha"
- # 根据通道类型决定是否包含预发布版本
include_prerelease = (prerelease_type != "stable")
return include_prerelease, prerelease_type if include_prerelease else None
-
+
def check_for_updates(self):
- """检查更新"""
+ """Check for updates"""
self.update_status_label.setText("Checking for updates...")
setThemeColor(getSystemAccentColor(), save=False)
- self.update_button.setEnabled(False)
+ self.check_update_card.setEnabled(False)
- # 隐藏TextBrowser、下载按钮和重启按钮
self.release_content_browser.setVisible(False)
- self.download_button.setVisible(False)
- self.restart_button.setVisible(False)
+ self.download_update_card.setEnabled(False)
+ self.restart_card.setEnabled(False)
- # 显示进度条并设置主题色
self.progress_bar.setVisible(True)
self.progress_bar.resume()
self.progress_bar.start()
- QApplication.processEvents() # 确保界面更新
- # 获取更新检查参数
- include_prerelease, prerelease_type = self._get_update_check_params()
+ from PySide6.QtWidgets import QApplication
+ QApplication.processEvents()
- # Auto-save settings immediately when checking for updates
- self.auto_save_settings()
+ include_prerelease, prerelease_type = self._get_update_check_params()
+ self.save_settings()
- # 启动检查更新线程
self.check_thread = CheckUpdateThread(self.update_manager, include_prerelease, prerelease_type if include_prerelease else None)
self.check_thread.check_finished.connect(self.on_check_finished)
self.check_thread.start()
def on_check_finished(self, result):
+ """Handle check finished"""
if result["status"] == "update_available":
- # 获取版本类型信息
version_type = ""
if "version_info" in result:
version_tuple = result["version_info"]
@@ -494,11 +429,9 @@ def on_check_finished(self, result):
version_type = f" ({version_type})"
self.update_status_label.setText(f"✅ {result['message']}\n\nVersion: {result['latest_version']}{version_type}")
- self.download_button.setVisible(True)
- self.download_button.setEnabled(True)
+ self.download_update_card.setEnabled(True)
self.current_update_info = result
- # 显示更新内容
if result.get("release_body"):
self.release_content_browser.setMarkdown(result["release_body"])
self.release_content_browser.setVisible(True)
@@ -506,23 +439,23 @@ def on_check_finished(self, result):
self.release_content_browser.setVisible(False)
elif result["status"] == "error":
self.update_status_label.setText(f"❌ Check failed: {result['message']}")
- self.download_button.setVisible(False)
+ self.download_update_card.setEnabled(False)
self.release_content_browser.setVisible(False)
else:
self.update_status_label.setText(f"ℹ️ {result['message']}")
- self.download_button.setVisible(False)
+ self.download_update_card.setEnabled(False)
self.release_content_browser.setVisible(False)
self.progress_bar.pause()
self.progress_bar.setVisible(False)
- self.update_button.setEnabled(True)
+ self.check_update_card.setEnabled(True)
def download_update(self):
+ """Download update"""
if hasattr(self, 'current_update_info'):
- self.download_button.setEnabled(False)
- self.update_button.setEnabled(False)
+ self.download_update_card.setEnabled(False)
+ self.check_update_card.setEnabled(False)
- # 隐藏TextBrowser
self.release_content_browser.setVisible(False)
self.progress_bar.setVisible(True)
@@ -530,18 +463,14 @@ def download_update(self):
self.download_progress_bar.setVisible(False)
self.update_status_label.setText("Download in progress...")
- # 显示进度标签
self.progress_label.setText("0%")
self.progress_label.setVisible(True)
- # 获取下载URL和版本信息
download_url = self.current_update_info.get("download_url")
latest_version = self.current_update_info.get("latest_version")
- # 获取预发布设置
include_prerelease, prerelease_type = self._get_update_check_params()
- # 重置下载计时器
if hasattr(self, '_download_start_time'):
delattr(self, '_download_start_time')
if hasattr(self, '_last_downloaded'):
@@ -549,388 +478,106 @@ def download_update(self):
if hasattr(self, '_last_time'):
delattr(self, '_last_time')
- # 使用新的DownloadThread类
self.download_thread = DownloadThread(download_url, latest_version, include_prerelease)
self.download_thread.progress_updated.connect(self.on_progress_updated)
self.download_thread.finished.connect(self.on_download_finished)
self.download_thread.start()
- # 将下载按钮改为取消按钮
- self.download_button.setText("Cancel Download")
- self.download_button.setEnabled(True)
- self.download_button.clicked.disconnect()
- self.download_button.clicked.connect(self.cancel_download)
+ self.download_update_card.setText("Cancel Download")
+ self.download_update_card.setEnabled(True)
+ self.download_update_card.clicked.disconnect()
+ self.download_update_card.clicked.connect(self.cancel_download)
def cancel_download(self):
- """取消下载"""
+ """Cancel download"""
if hasattr(self, 'download_thread') and self.download_thread is not None and self.download_thread.isRunning():
self.download_thread.cancel()
- self.download_button.setEnabled(False)
+ self.download_update_card.setEnabled(False)
self.update_status_label.setText("Cancelling download...")
- # 设置一个定时器来检查线程是否已停止
def check_thread_stopped():
- # 检查线程是否已停止
if self.download_thread is None:
- # 线程已被清理,直接返回
return
if not self.download_thread.isRunning():
- # 线程已停止,显示取消状态
self.show_cancelled_state()
else:
- # 线程仍在运行,继续等待
QTimer.singleShot(500, check_thread_stopped)
- # 开始检查线程状态
QTimer.singleShot(500, check_thread_stopped)
- def start_swing_animation(self):
- """启动左右摆动动画"""
- pass
-
- def update_swing_animation(self):
- """更新摆动动画"""
- pass
+ def show_cancelled_state(self):
+ """Show cancelled state"""
+ self.update_status_label.setText("Download cancelled")
+ self.progress_bar.setVisible(False)
+ self.download_progress_bar.setVisible(False)
+ self.progress_label.setVisible(False)
+ self.check_update_card.setEnabled(True)
+ self.download_update_card.setText("Download Update")
+ self.download_update_card.clicked.disconnect()
+ self.download_update_card.clicked.connect(self.download_update)
def on_progress_updated(self, progress_data):
- """处理下载进度更新"""
+ """Handle progress update"""
if not progress_data:
return
-
- # 从progress_data中获取进度信息
+
progress = progress_data.get("progress", 0)
- downloaded_bytes = progress_data.get("downloaded", 0)
- total_bytes = progress_data.get("total", 0)
-
- # 更新进度条显示实际进度
- if progress > 0:
- # 显示实际进度条,隐藏不确定进度条
- self.progress_bar.setVisible(False)
- self.download_progress_bar.setVisible(True)
- self.download_progress_bar.setValue(progress)
-
- # 计算下载速度和时间估计
- current_time = time.time()
- if not hasattr(self, '_download_start_time'):
- self._download_start_time = current_time
- self._last_downloaded = 0
- self._last_time = current_time
-
- # 计算平均速度和剩余时间
- elapsed_time = current_time - self._download_start_time
- if elapsed_time > 1.0: # 至少1秒后再计算速度
- # 计算平均下载速度
- avg_download_speed = downloaded_bytes / elapsed_time # bytes per second
-
- # 计算剩余时间
- remaining_bytes = total_bytes - downloaded_bytes
- if avg_download_speed > 0 and total_bytes > 0:
- remaining_time = remaining_bytes / avg_download_speed
- # 格式化剩余时间
- if remaining_time > 3600:
- eta_str = f"{remaining_time/3600:.1f}h"
- elif remaining_time > 60:
- eta_str = f"{remaining_time/60:.1f}m"
- else:
- eta_str = f"{remaining_time:.0f}s"
- else:
- eta_str = "计算中"
-
- # 格式化文件大小
- def format_size(bytes_size):
- if bytes_size >= 1024*1024*1024:
- return f"{bytes_size/(1024*1024*1024):.2f} GB"
- elif bytes_size >= 1024*1024:
- return f"{bytes_size/(1024*1024):.2f} MB"
- elif bytes_size >= 1024:
- return f"{bytes_size/1024:.2f} KB"
- else:
- return f"{bytes_size} B"
-
- # 更新数字进度标签
- if hasattr(self, 'progress_label'):
- downloaded_str = format_size(downloaded_bytes)
- total_str = format_size(total_bytes)
- speed_str = format_size(avg_download_speed) + "/s"
-
- self.progress_label.setText(
- f"{progress}% - {downloaded_str}/{total_str} - {speed_str} - ETA: {eta_str}"
- )
+ downloaded = progress_data.get("downloaded", 0)
+ total = progress_data.get("total", 0)
- if progress == 100:
- self.progress_bar.pause()
+ self.download_progress_bar.setVisible(True)
+ self.download_progress_bar.setValue(int(progress))
- def on_download_finished(self, result):
- # 停止进度条
- if result["status"] == "success":
- # 下载成功,立即开始应用更新
- self.progress_bar.pause()
- self.progress_bar.setVisible(False)
- self.download_progress_bar.setVisible(False)
-
- # 隐藏数字进度标签
- if hasattr(self, 'progress_label'):
- self.progress_label.setVisible(False)
-
- # 清理下载计时器变量
- if hasattr(self, '_download_start_time'):
- delattr(self, '_download_start_time')
- if hasattr(self, '_last_downloaded'):
- delattr(self, '_last_downloaded')
- if hasattr(self, '_last_time'):
- delattr(self, '_last_time')
-
- # 清理下载线程
- if hasattr(self, 'download_thread') and self.download_thread:
- if self.download_thread.downloader:
- self.download_thread.downloader.cleanup()
-
- # 确保线程完全停止后再设置为None
- if self.download_thread.isRunning():
- self.download_thread.quit()
- self.download_thread.wait(2000) # 最多等待2秒
-
- self.download_thread = None
-
- # 保存更新信息
- self.update_result = result
-
- # 显示应用更新进度
- self.update_status_label.setText("Applying updates...")
-
- # 使用IndeterminateProgressBar显示应用更新进度
- self.progress_bar.setVisible(True)
- self.progress_bar.start()
-
- # 立即执行更新应用
- self.apply_update()
-
+ if total > 0:
+ downloaded_mb = downloaded / (1024 * 1024)
+ total_mb = total / (1024 * 1024)
+ self.progress_label.setText(f"{progress:.1f}% - {downloaded_mb:.1f} MB / {total_mb:.1f} MB")
else:
- # 下载失败或取消
- if result["status"] == "cancelled":
- # 先将进度条设置为100%
- self.download_progress_bar.setValue(100)
- self.progress_label.setText("100%")
- QApplication.processEvents() # 确保界面更新
-
- # 短暂延迟后显示取消状态
- QTimer.singleShot(500, lambda: self.show_cancelled_state())
- else:
- # 确保错误消息正确处理非ASCII字符
- try:
- error_message = str(result['message'])
- self.update_status_label.setText(f"❌ 下载失败: {error_message}")
- except Exception as e:
- # 如果出现编码问题,显示基本错误信息
- self.update_status_label.setText(f"❌ 下载失败: 处理错误消息时出现编码问题")
-
- self.progress_bar.setVisible(False)
- self.download_progress_bar.setVisible(False)
-
- # 隐藏数字进度标签
- if hasattr(self, 'progress_label'):
- self.progress_label.setVisible(False)
-
- # 清理下载计时器变量
- if hasattr(self, '_download_start_time'):
- delattr(self, '_download_start_time')
- if hasattr(self, '_last_downloaded'):
- delattr(self, '_last_downloaded')
- if hasattr(self, '_last_time'):
- delattr(self, '_last_time')
-
- # 恢复下载按钮
- self.download_button.setText("Download Update")
- self.download_button.clicked.disconnect()
- self.download_button.clicked.connect(self.download_update)
- self.download_button.setEnabled(True)
- self.download_button.setVisible(True)
-
- self.update_button.setEnabled(True)
+ self.progress_label.setText(f"{progress:.1f}%")
- def show_cancelled_state(self):
- """显示取消下载的状态,恢复到初始形态"""
- self.update_status_label.setText("Ready to check for updates.")
-
- # 隐藏进度条
+ def on_download_finished(self, result):
+ """Handle download finished"""
self.progress_bar.setVisible(False)
self.download_progress_bar.setVisible(False)
+ self.progress_label.setVisible(False)
- # 隐藏数字进度标签
- if hasattr(self, 'progress_label'):
- self.progress_label.setVisible(False)
-
- # 隐藏发布内容浏览器
- self.release_content_browser.setVisible(False)
-
- # 清理下载计时器变量
- if hasattr(self, '_download_start_time'):
- delattr(self, '_download_start_time')
- if hasattr(self, '_last_downloaded'):
- delattr(self, '_last_downloaded')
- if hasattr(self, '_last_time'):
- delattr(self, '_last_time')
-
- # Clean up downloader and thread
- if hasattr(self, 'download_thread') and self.download_thread:
- if self.download_thread.downloader:
- self.download_thread.downloader.cleanup()
-
- # 确保线程完全停止后再设置为None
- if self.download_thread.isRunning():
- self.download_thread.quit()
- self.download_thread.wait(2000) # 最多等待2秒
-
- self.download_thread = None
-
- # 隐藏下载按钮,显示检查更新按钮
- self.download_button.setVisible(False)
- self.download_button.setEnabled(False)
-
- # 启用检查更新按钮
- self.update_button.setEnabled(True)
-
- def apply_update(self):
- """应用更新"""
- if hasattr(self, 'update_result'):
- # 新的下载流程已经自动应用了更新,所以这里只需要显示成功消息
- if self.update_result.get("status") == "success":
- # 更新状态显示
- self.update_status_label.setText("✅ Update applied successfully! Please restart the application.")
-
- # 显示重启按钮
- self.restart_button.setVisible(True)
- self.restart_button.setEnabled(True)
-
- # 停止进度条
- self.progress_bar.pause()
- self.progress_bar.setVisible(False)
- else:
- # 处理错误情况
- error_message = self.update_result.get("message", "Unknown error")
- self.update_status_label.setText(f"❌ Update failed: {error_message}")
-
- # 停止进度条
- self.progress_bar.pause()
- self.progress_bar.setVisible(False)
- else:
- print("❌ update_result属性不存在")
- self.update_status_label.setText("❌ Update information lost, please restart manually")
-
-
-
- def __del__(self):
- """析构函数,确保所有线程都被正确清理"""
- try:
- # 清理下载线程
- if hasattr(self, 'download_thread') and self.download_thread is not None:
- if self.download_thread.isRunning():
- self.download_thread.quit()
- if not self.download_thread.wait(1000):
- self.download_thread.terminate()
- self.download_thread.wait(500)
-
- # 清理下载器
- if hasattr(self.download_thread, 'downloader') and self.download_thread.downloader:
- self.download_thread.downloader.cleanup()
-
- self.download_thread = None
-
- # 清理检查更新线程
- if hasattr(self, 'check_thread') and self.check_thread is not None:
- if self.check_thread.isRunning():
- self.check_thread.quit()
- if not self.check_thread.wait(1000):
- self.check_thread.terminate()
- self.check_thread.wait(500)
-
- self.check_thread = None
- except:
- # 在析构函数中忽略所有异常
- pass
-
- def closeEvent(self, event):
- """重写窗口关闭事件,下载时阻止关闭"""
- # 检查是否有下载正在进行
- if hasattr(self, 'download_thread') and self.download_thread is not None and self.download_thread.isRunning():
- # 下载正在进行,显示提示并阻止关闭
- reply = QMessageBox.question(
- self,
- "下载进行中",
- "正在下载更新,确定要关闭窗口吗?\n关闭窗口将取消下载。",
- QMessageBox.StandardButton.Yes | QMessageBox.StandardButton.No,
- QMessageBox.StandardButton.No
- )
-
- if reply == QMessageBox.StandardButton.Yes:
- # 用户确认关闭,先取消下载
- self.cancel_download()
-
- # 等待更长时间确保线程完全停止
- max_wait_time = 5000 # 最多等待5秒
- wait_step = 500 # 每次等待500ms
- total_waited = 0
-
- while self.download_thread and self.download_thread.isRunning() and total_waited < max_wait_time:
- QApplication.processEvents() # 处理待处理的事件
- self.download_thread.wait(wait_step) # 等待一小段时间
- total_waited += wait_step
-
- # 如果线程仍在运行,强制终止
- if self.download_thread and self.download_thread.isRunning():
- self.download_thread.terminate()
- self.download_thread.wait(1000) # 再等待1秒
-
- # 清理下载器
- if self.download_thread and self.download_thread.downloader:
- self.download_thread.downloader.cleanup()
-
- self.download_thread = None
-
- # 允许关闭窗口
- event.accept()
- else:
- # 用户取消关闭,阻止窗口关闭
- event.ignore()
+ if result["status"] == "success":
+ self.update_status_label.setText("✅ Download completed successfully!")
+ self.restart_card.setEnabled(True)
+ self.check_update_card.setEnabled(True)
+ self.download_update_card.setText("Download Update")
+ self.download_update_card.clicked.disconnect()
+ self.download_update_card.clicked.connect(self.download_update)
+ elif result["status"] == "cancelled":
+ self.show_cancelled_state()
else:
- # 没有下载在进行,检查是否有检查更新线程
- if hasattr(self, 'check_thread') and self.check_thread is not None and self.check_thread.isRunning():
- # 检查更新正在进行,取消它
- self.check_thread.quit()
- if not self.check_thread.wait(1000): # 等待最多1秒
- self.check_thread.terminate()
- self.check_thread.wait(500) # 再等待500ms
- self.check_thread = None
-
- # 允许正常关闭
- event.accept()
+ self.update_status_label.setText(f"❌ Download failed: {result.get('message', 'Unknown error')}")
+ self.check_update_card.setEnabled(True)
+ self.download_update_card.setText("Download Update")
+ self.download_update_card.clicked.disconnect()
+ self.download_update_card.clicked.connect(self.download_update)
def restart_application(self):
- """重启应用程序"""
+ """Restart application"""
+ self.update_status_label.setText("Restarting application...")
+
+ import subprocess
+ import sys
+
try:
- # 执行重启脚本
- restart_script_path = os.path.join(os.path.dirname(os.path.dirname(__file__)), "update", "restart.command")
-
- if os.path.exists(restart_script_path):
- # 执行重启脚本
- os.system(f"'{restart_script_path}' &")
- print(f"✅ 重启脚本已执行: {restart_script_path}")
-
- # 关闭当前应用程序
- QApplication.quit()
-
- else:
- print(f"❌ 重启脚本不存在: {restart_script_path}")
- self.update_status_label.setText("❌ Restart script not found, please restart manually")
-
+ subprocess.Popen([sys.executable] + sys.argv)
+ from PySide6.QtWidgets import QApplication
+ QApplication.instance().quit()
except Exception as e:
- print(f"❌ 执行重启脚本时出错: {e}")
- self.update_status_label.setText(f"❌ Restart failed: {e}")
+ self.update_status_label.setText(f"Failed to restart: {e}")
-def main():
+
+if __name__ == "__main__":
+ from PySide6.QtWidgets import QApplication
+
app = QApplication(sys.argv)
- window = UpdateDialog()
- window.resize(500, 350) # 扩大窗口大小以容纳更多内容
- window.show()
- sys.exit(app.exec())
\ No newline at end of file
+ widget = UpdateSettingsWidget()
+ widget.resize(800, 600)
+ widget.show()
+ sys.exit(app.exec())
diff --git a/support/GUI/arc_support.py b/support/GUI/arc_support.py
new file mode 100644
index 0000000..9a910cb
--- /dev/null
+++ b/support/GUI/arc_support.py
@@ -0,0 +1,253 @@
+import os
+from PySide6.QtCore import Qt, QSize, Signal
+from PySide6.QtWidgets import QFrame, QVBoxLayout, QLabel, QFileDialog
+from PySide6.QtGui import QDragEnterEvent, QDropEvent, QPalette, QPixmap
+
+
+class BatchDropZoneWidget(QFrame):
+ """Custom widget for drag and drop archive file selection"""
+ files_dropped = Signal(list) # Signal for multiple archive files dropped
+
+ def __init__(self, placeholder_text="Drag archive files here or click to browse", parent=None):
+ super().__init__(parent)
+ self.setAcceptDrops(True)
+ self.setMinimumHeight(100) # 增加到100确保足够空间
+ self.setFixedHeight(100) # 设置固定高度防止缩小
+ self.setMinimumWidth(200) # 设置最小宽度
+ self.is_dark_mode = False # Track current theme mode
+
+ # Define supported archive formats
+ self.supported_formats = {
+ '.zip', '.rar', '.7z', '.tar', '.gz', '.bz2', '.xz', '.cab', '.iso', '.arj', '.ace', '.lzh', '.lha'
+ }
+
+ layout = QVBoxLayout(self)
+ layout.setAlignment(Qt.AlignmentFlag.AlignCenter)
+ layout.setContentsMargins(10, 10, 10, 10) # 设置内边距防止内容紧贴边框
+
+ self.icon_label = QLabel("📁")
+ self.icon_label.setAlignment(Qt.AlignmentFlag.AlignCenter)
+ self.icon_label.setStyleSheet("font-size: 20px;")
+
+ self.text_label = QLabel(placeholder_text)
+ self.text_label.setAlignment(Qt.AlignmentFlag.AlignCenter)
+ self.text_label.setStyleSheet("color: #666; font-size: 12px;")
+ self.text_label.setWordWrap(True)
+
+ layout.addWidget(self.icon_label)
+ layout.addWidget(self.text_label)
+
+ self.drag_over = False
+
+ # Click to browse
+ self.mousePressEvent = self.browse_files
+
+ # Apply initial light theme style after all widgets are created
+ self._apply_light_theme_style()
+
+ def sizeHint(self):
+ """Return fixed size hint to prevent resizing"""
+ return super().sizeHint()
+
+ def minimumSizeHint(self):
+ """Return minimum size hint to prevent shrinking"""
+ return QSize(200, 100)
+
+ def set_theme(self, is_dark_mode):
+ """Update the theme of the drag and drop area"""
+ self.is_dark_mode = is_dark_mode
+ if self.is_dark_mode:
+ self._apply_dark_theme_style()
+ else:
+ self._apply_light_theme_style()
+
+ def _apply_light_theme_style(self):
+ """Apply light theme styles"""
+ self.setStyleSheet("""
+ QFrame {
+ border: 2px dashed #aaa;
+ border-radius: 10px;
+ background-color: #f9f9f9;
+ }
+ QFrame:hover {
+ border-color: #007acc;
+ background-color: #f0f8ff;
+ }
+ QFrame:drop {
+ border-color: #28a745;
+ background-color: #f0fff0;
+ }
+ """)
+ self.text_label.setStyleSheet("color: #666; font-size: 12px;")
+
+ def _apply_dark_theme_style(self):
+ """Apply dark theme styles"""
+ self.setStyleSheet("""
+ QFrame {
+ border: 2px dashed #555;
+ border-radius: 10px;
+ background-color: #2d2d2d;
+ }
+ QFrame:hover {
+ border-color: #007acc;
+ background-color: #1e3a5f;
+ }
+ QFrame:drop {
+ border-color: #28a745;
+ background-color: #1a2f1a;
+ }
+ """)
+ self.text_label.setStyleSheet("color: #aaa; font-size: 12px;")
+
+ def browse_files(self, event):
+ """Open file browser when clicked"""
+ file_dialog = QFileDialog()
+ file_paths, _ = file_dialog.getOpenFileNames(
+ self,
+ "Select Archive Files",
+ "",
+ "Archive Files (*.zip *.rar *.7z *.tar *.gz *.bz2 *.xz *.cab *.iso *.arj *.ace *.lzh *.lha);;All Files (*)"
+ )
+ if file_paths:
+ self.files_dropped.emit(file_paths)
+
+ def dragEnterEvent(self, event):
+ if event.mimeData().hasUrls():
+ # Check if any of the dragged items are supported archive formats
+ has_supported_files = False
+ total_files = 0
+ supported_files = 0
+
+ for url in event.mimeData().urls():
+ if hasattr(url, 'toLocalFile'):
+ path = url.toLocalFile()
+ else:
+ path = url.path() if hasattr(url, 'path') else ""
+
+ if path and os.path.isfile(path):
+ total_files += 1
+ if self._is_supported_archive_file(path):
+ supported_files += 1
+
+ # Accept if there are supported archive files
+ if supported_files > 0:
+ self._set_drag_over_style(True)
+ event.acceptProposedAction()
+ self._update_text_label(total_files, supported_files, True)
+ else:
+ self._set_reject_style()
+ else:
+ event.ignore()
+
+ def dragLeaveEvent(self, event):
+ self._reset_style()
+
+ def dragMoveEvent(self, event):
+ if event.mimeData().hasUrls():
+ self._set_drag_over_style(True)
+ event.acceptProposedAction()
+ else:
+ event.ignore()
+
+ def dropEvent(self, event):
+ self._reset_style()
+
+ files = []
+
+ for url in event.mimeData().urls():
+ path = url.toLocalFile()
+ if os.path.isfile(path) and self._is_supported_archive_file(path):
+ files.append(path)
+
+ # Emit signal for valid archive files
+ if files:
+ self.files_dropped.emit(files)
+
+ def _is_supported_archive_file(self, file_path):
+ """Check if the file has a supported archive format extension"""
+ if not file_path:
+ return False
+ _, ext = os.path.splitext(file_path.lower())
+ return ext in self.supported_formats
+
+ def _set_drag_over_style(self, has_supported=True):
+ """Set style for drag over state"""
+ # Ensure fixed size is maintained during style changes
+ current_width = self.width()
+ current_height = self.height()
+
+ if self.is_dark_mode:
+ self.setStyleSheet("""
+ QFrame {
+ border: 2px solid #28a745;
+ border-radius: 10px;
+ background-color: #1a2f1a;
+ }
+ """)
+ else:
+ self.setStyleSheet("""
+ QFrame {
+ border: 2px solid #28a745;
+ border-radius: 10px;
+ background-color: #f0fff0;
+ }
+ """)
+
+ # Restore fixed size after style change
+ self.setFixedHeight(100)
+ if current_width > 0:
+ self.setMinimumWidth(current_width)
+
+ self.drag_over = True
+
+ def _set_reject_style(self):
+ """Set style for rejected drag items"""
+ # Ensure fixed size is maintained during style changes
+ current_width = self.width()
+
+ if self.is_dark_mode:
+ self.setStyleSheet("""
+ QFrame {
+ border: 2px solid #dc3545;
+ border-radius: 10px;
+ background-color: #2a1a1a;
+ }
+ """)
+ else:
+ self.setStyleSheet("""
+ QFrame {
+ border: 2px solid #dc3545;
+ border-radius: 10px;
+ background-color: #fff5f5;
+ }
+ """)
+
+ # Restore fixed size after style change
+ self.setFixedHeight(100)
+ if current_width > 0:
+ self.setMinimumWidth(current_width)
+
+ def _reset_style(self):
+ """Reset to default style"""
+ # Ensure fixed size is maintained during style changes
+ current_width = self.width()
+
+ if self.is_dark_mode:
+ self._apply_dark_theme_style()
+ else:
+ self._apply_light_theme_style()
+
+ # Restore fixed size after style change
+ self.setFixedHeight(100)
+ if current_width > 0:
+ self.setMinimumWidth(current_width)
+
+ self.drag_over = False
+
+ def _update_text_label(self, total_files, supported_files, has_supported):
+ """Update the text label during drag operations"""
+ if has_supported:
+ if total_files == supported_files:
+ self.text_label.setText(f"Release to add {supported_files} archive file(s)")
+ else:
+ self.text_label.setText(f"Release to add {supported_files} archive file(s)\n({total_files - supported_files} unsupported file(s) will be ignored)")
diff --git a/support/GUI/image_support.py b/support/GUI/image_support.py
new file mode 100644
index 0000000..0dcef90
--- /dev/null
+++ b/support/GUI/image_support.py
@@ -0,0 +1,935 @@
+#!/usr/bin/env python3
+# -*- coding: utf-8 -*-
+
+import os
+from PySide6.QtWidgets import (
+ QFrame, QVBoxLayout, QLabel, QHBoxLayout, QWidget,
+ QFileDialog
+)
+from PySide6.QtCore import QRect
+from PySide6.QtGui import QPixmap, QFont, QPainter, QColor
+from PySide6.QtCore import Qt, QSize, Signal
+from qfluentwidgets import *
+
+
+class DropZoneWidget(QFrame):
+ """Custom widget for drag and drop file/folder selection with support for all image formats"""
+ filesDropped = Signal(list) # Signal for multiple files dropped
+ folderDropped = Signal(str) # Signal for folder dropped
+
+ def __init__(self, parent=None):
+ super().__init__(parent)
+ self.setAcceptDrops(True)
+ self.setMinimumHeight(100)
+ self.setFixedHeight(100) # 设置固定高度防止缩小
+ self.setMinimumWidth(200) # 设置最小宽度
+ self.is_dark_mode = False # Track current theme mode
+
+ # Define supported image formats
+ self.supported_formats = {
+ # Common formats
+ '.png', '.jpg', '.jpeg', '.bmp', '.gif', '.tiff', '.tif', '.ico', '.icns',
+ # Web formats
+ '.webp', '.svg',
+ # High efficiency formats
+ '.heic', '.heif', '.avif', '.jxl',
+ # Other formats
+ '.pdf', '.eps', '.dds', '.exr'
+ }
+
+ # 设置初始状态变量
+ self.is_dark_mode = False # 初始化为浅色主题
+
+ # 创建布局
+ layout = QVBoxLayout(self)
+ layout.setAlignment(Qt.AlignmentFlag.AlignCenter)
+ layout.setContentsMargins(10, 10, 10, 10) # 设置内边距防止内容紧贴边框
+
+ self.icon_label = QLabel("📁")
+ self.icon_label.setAlignment(Qt.AlignmentFlag.AlignCenter)
+ self.icon_label.setStyleSheet("font-size: 24px;")
+
+ self.text_label = QLabel("Drag files or folders here\n(Supports: PNG, JPG, JPEG, BMP, GIF, TIFF, ICO, ICNS, WebP, SVG, HEIC, HEIF, AVIF, JXL, PDF, EPS, DDS, EXR)")
+ self.text_label.setAlignment(Qt.AlignmentFlag.AlignCenter)
+ self.text_label.setStyleSheet("color: #666; font-size: 12px;")
+ self.text_label.setWordWrap(True)
+
+ # 应用初始浅色主题样式(在组件创建之后)
+ self._apply_light_theme_style()
+
+ layout.addWidget(self.icon_label)
+ layout.addWidget(self.text_label)
+
+ self.drag_over = False
+
+ # Click to browse
+ self.mousePressEvent = self.browse_files
+
+ def sizeHint(self):
+ """Return fixed size hint to prevent resizing"""
+ return super().sizeHint()
+
+ def minimumSizeHint(self):
+ """Return minimum size hint to prevent shrinking"""
+ return QSize(200, 100)
+
+ def set_theme(self, is_dark_mode):
+ """Update the theme of the drag and drop area"""
+ self.is_dark_mode = is_dark_mode
+ if self.is_dark_mode:
+ self._apply_dark_theme_style()
+ else:
+ self._apply_light_theme_style()
+
+ def _apply_light_theme_style(self):
+ """Apply light theme styles"""
+ self.setStyleSheet("""
+ QFrame {
+ border: 2px dashed #aaa;
+ border-radius: 10px;
+ background-color: #f9f9f9;
+ }
+ QFrame:hover {
+ border-color: #007acc;
+ background-color: #f0f8ff;
+ }
+ QFrame:drop {
+ border-color: #28a745;
+ background-color: #f0fff0;
+ }
+ """)
+ self.text_label.setStyleSheet("color: #666; font-size: 12px;")
+
+ def _apply_dark_theme_style(self):
+ """Apply dark theme styles"""
+ self.setStyleSheet("""
+ QFrame {
+ border: 2px dashed #555;
+ border-radius: 10px;
+ background-color: #2d2d2d;
+ }
+ QFrame:hover {
+ border-color: #007acc;
+ background-color: #1e3a5f;
+ }
+ QFrame:drop {
+ border-color: #28a745;
+ background-color: #1a2f1a;
+ }
+ """)
+ self.text_label.setStyleSheet("color: #aaa; font-size: 12px;")
+
+ def browse_files(self, event):
+ """Open file browser when clicked"""
+ file_dialog = QFileDialog()
+ file_paths, _ = file_dialog.getOpenFileNames(
+ self,
+ "Select Image Files",
+ "",
+ "Image Files (*.png *.jpg *.jpeg *.bmp *.gif *.tiff *.ico *.webp);;All Files (*)"
+ )
+ if file_paths:
+ self.filesDropped.emit(file_paths)
+
+ def dragEnterEvent(self, event):
+ if event.mimeData().hasUrls():
+ # Check if any of the dragged items are supported formats
+ has_supported_files = False
+ total_files = 0
+ supported_files = 0
+ has_folders = False
+
+ for url in event.mimeData().urls():
+ if hasattr(url, 'toLocalFile'):
+ path = url.toLocalFile()
+ else:
+ path = url.path() if hasattr(url, 'path') else ""
+
+ if path and os.path.isfile(path):
+ total_files += 1
+ if self._is_supported_image_file(path):
+ supported_files += 1
+ elif path and os.path.isdir(path):
+ total_files += 1
+ has_folders = True # Folders count as one item
+
+ # Accept if there are supported image files or folders
+ if supported_files > 0 or has_folders:
+ self._set_drag_over_style(True)
+ event.acceptProposedAction()
+ self._update_text_label(total_files, supported_files, True)
+ else:
+ # Show rejection style
+ self._set_reject_style()
+ else:
+ event.ignore()
+
+ def dragLeaveEvent(self, event):
+ self._reset_style()
+
+ def dragMoveEvent(self, event):
+ if event.mimeData().hasUrls():
+ self._set_drag_over_style(True)
+ event.acceptProposedAction()
+ else:
+ event.ignore()
+
+ def dropEvent(self, event):
+ self._reset_style()
+
+ files = []
+ folders = []
+ rejected_files = []
+
+ for url in event.mimeData().urls():
+ path = url.toLocalFile()
+ if os.path.isdir(path):
+ folders.append(path)
+ elif os.path.isfile(path):
+ if self._is_supported_image_file(path):
+ files.append(path)
+ else:
+ rejected_files.append(path)
+
+ # Emit signals for valid items
+ if folders:
+ # For now, emit first folder (can be extended for multiple folders)
+ self.folderDropped.emit(folders[0])
+
+ if files:
+ self.filesDropped.emit(files)
+
+ # Show result message
+ if rejected_files:
+ self._show_rejected_files_message(rejected_files)
+
+ def _is_supported_image_file(self, file_path):
+ """Check if the file has a supported image format extension"""
+ if not file_path:
+ return False
+ _, ext = os.path.splitext(file_path.lower())
+ return ext in self.supported_formats
+
+ def _set_drag_over_style(self, has_supported=True):
+ """Set style for drag over state"""
+ # Ensure fixed size is maintained during style changes
+ current_width = self.width()
+
+ if has_supported:
+ if self.is_dark_mode:
+ self.setStyleSheet("""
+ QFrame {
+ border: 2px solid #28a745;
+ border-radius: 10px;
+ background-color: #1a2f1a;
+ }
+ """)
+ else:
+ self.setStyleSheet("""
+ QFrame {
+ border: 2px solid #28a745;
+ border-radius: 10px;
+ background-color: #f0fff0;
+ }
+ """)
+ self.drag_over = True
+ else:
+ if self.is_dark_mode:
+ self.setStyleSheet("""
+ QFrame {
+ border: 2px solid #007acc;
+ border-radius: 10px;
+ background-color: #1e3a5f;
+ }
+ """)
+ else:
+ self.setStyleSheet("""
+ QFrame {
+ border: 2px solid #007acc;
+ border-radius: 10px;
+ background-color: #e6f3ff;
+ }
+ """)
+
+ # Restore fixed size after style change
+ self.setFixedHeight(100)
+ if current_width > 0:
+ self.setMinimumWidth(current_width)
+
+ def _set_reject_style(self):
+ """Set style for rejected drag items"""
+ # Ensure fixed size is maintained during style changes
+ current_width = self.width()
+
+ if self.is_dark_mode:
+ self.setStyleSheet("""
+ QFrame {
+ border: 2px solid #dc3545;
+ border-radius: 10px;
+ background-color: #2a1a1a;
+ }
+ """)
+ else:
+ self.setStyleSheet("""
+ QFrame {
+ border: 2px solid #dc3545;
+ border-radius: 10px;
+ background-color: #fff5f5;
+ }
+ """)
+
+ # Restore fixed size after style change
+ self.setFixedHeight(100)
+ if current_width > 0:
+ self.setMinimumWidth(current_width)
+
+ def _reset_style(self):
+ """Reset to default style"""
+ # Ensure fixed size is maintained during style changes
+ current_width = self.width()
+
+ if self.is_dark_mode:
+ self._apply_dark_theme_style()
+ else:
+ self._apply_light_theme_style()
+
+ # Restore fixed size after style change
+ self.setFixedHeight(100)
+ if current_width > 0:
+ self.setMinimumWidth(current_width)
+
+ self.drag_over = False
+
+ def _update_text_label(self, total_files, supported_files, has_supported):
+ """Update the text label during drag operations"""
+ if has_supported:
+ if supported_files > 0:
+ if total_files == supported_files:
+ self.text_label.setText(f"Release to add {supported_files} image file(s)\n(Supported formats: {len(self.supported_formats)} types)")
+ else:
+ self.text_label.setText(f"Release to add {supported_files} image file(s)\n({total_files - supported_files} unsupported file(s) will be ignored)")
+ else:
+ # Check if we're dealing with folders
+ # If total_files is 1 and supported_files is 0, it's likely a folder
+ if total_files == 1:
+ self.text_label.setText("Release to process folder\n(Folder will be scanned for image files)")
+ else:
+ self.text_label.setText("Release to process folders\n(Folders will be scanned for image files)")
+ else:
+ self.text_label.setText("Drag image files or folders here")
+
+ def _show_rejected_files_message(self, rejected_files):
+ """Show message about rejected files"""
+ if rejected_files:
+ rejected_names = [os.path.basename(f) for f in rejected_files[:3]] # Show first 3
+ if len(rejected_files) > 3:
+ rejected_names.append(f"and {len(rejected_files) - 3} more...")
+
+ # You can add a tooltip or status message here
+ print(f"Rejected {len(rejected_files)} unsupported file(s): {', '.join(rejected_names)}")
+
+
+class DirectoryDropLineEdit(LineEdit):
+ """支持文件夹拖拽的输出路径输入框"""
+
+ def __init__(self, parent=None):
+ super().__init__(parent)
+ self.setPlaceholderText("Drag folder here or click Browse...")
+
+ def dragEnterEvent(self, event):
+ """处理拖拽进入事件"""
+ if event.mimeData().hasUrls():
+ urls = event.mimeData().urls()
+ # 检查是否包含文件夹
+ has_folder = False
+ for url in urls:
+ if hasattr(url, 'toLocalFile'):
+ file_path = url.toLocalFile()
+ if file_path and os.path.isdir(file_path):
+ has_folder = True
+ break
+ elif hasattr(url, 'path') and os.path.isdir(url.path()):
+ has_folder = True
+ break
+
+ if has_folder:
+ event.acceptProposedAction()
+ self.setStyleSheet("""
+ LineEdit {
+ border: 2px solid #28a745;
+ border-radius: 4px;
+ background-color: #f0fff0;
+ }
+ """)
+ return
+ event.ignore()
+
+ def dragLeaveEvent(self, event):
+ """处理拖拽离开事件"""
+ self.setStyleSheet("")
+
+ def dragMoveEvent(self, event):
+ """处理拖拽移动事件"""
+ if event.mimeData().hasUrls():
+ urls = event.mimeData().urls()
+ # 检查是否包含文件夹
+ has_folder = False
+ for url in urls:
+ if hasattr(url, 'toLocalFile'):
+ file_path = url.toLocalFile()
+ if file_path and os.path.isdir(file_path):
+ has_folder = True
+ break
+ elif hasattr(url, 'path') and os.path.isdir(url.path()):
+ has_folder = True
+ break
+
+ if has_folder:
+ event.acceptProposedAction()
+ return
+ event.ignore()
+
+ def dropEvent(self, event):
+ """处理放置事件"""
+ self.setStyleSheet("")
+
+ if event.mimeData().hasUrls():
+ urls = event.mimeData().urls()
+ for url in urls:
+ if hasattr(url, 'toLocalFile'):
+ file_path = url.toLocalFile()
+ if file_path and os.path.isdir(file_path):
+ self.setText(file_path)
+ # 发射信号表示路径已设置
+ self.editingFinished.emit()
+ event.acceptProposedAction()
+ return
+ elif hasattr(url, 'path') and os.path.isdir(url.path()):
+ file_path = url.path()
+ self.setText(file_path)
+ self.editingFinished.emit()
+ event.acceptProposedAction()
+ return
+ event.ignore()
+
+
+class PreviewTab(QWidget):
+ """独立的预览标签页,支持单张和多张图片预览"""
+
+ def __init__(self, parent=None):
+ super().__init__(parent)
+ self.current_previews = [] # 当前显示的预览图片
+ self.setup_ui()
+
+ def setup_ui(self):
+ """设置预览标签页界面"""
+ layout = QVBoxLayout(self)
+
+ # 标题
+ title_label = QLabel("Preview")
+ title_font = QFont()
+ title_font.setPointSize(title_label.font().pointSize() + 4)
+ title_font.setBold(True)
+ title_label.setFont(title_font)
+ title_label.setAlignment(Qt.AlignmentFlag.AlignCenter)
+ layout.addWidget(title_label, 0, Qt.AlignmentFlag.AlignHCenter)
+
+ # 信息标签
+ self.info_label = QLabel("Drag image files to the tab above for preview")
+ self.info_label.setAlignment(Qt.AlignmentFlag.AlignCenter)
+ self.info_label.setStyleSheet("color: #666; font-style: italic;")
+ layout.addWidget(self.info_label, 0, Qt.AlignmentFlag.AlignHCenter)
+
+ # 滚动区域用于显示多个预览
+ scroll_area = ScrollArea()
+ scroll_area.setWidgetResizable(True)
+ scroll_area.setVerticalScrollBarPolicy(Qt.ScrollBarPolicy.ScrollBarAsNeeded)
+ scroll_area.setHorizontalScrollBarPolicy(Qt.ScrollBarPolicy.ScrollBarAsNeeded)
+ scroll_area.setMinimumHeight(400)
+
+ # 预览容器
+ self.preview_container = QWidget()
+ self.preview_layout = QVBoxLayout(self.preview_container)
+ self.preview_layout.setAlignment(Qt.AlignmentFlag.AlignCenter)
+ self.preview_layout.setSpacing(20)
+
+ scroll_area.setWidget(self.preview_container)
+ layout.addWidget(scroll_area)
+
+
+
+
+
+
+ def show_single_preview(self, file_path):
+ """显示单张图片预览"""
+ self.clear_previews()
+ self._create_single_preview_widget(file_path)
+ self._update_info_label(1, [file_path])
+
+ def show_multiple_previews(self, file_paths):
+ """显示多张图片预览"""
+ self.clear_previews()
+
+ # Filter out folders and only process files
+ valid_files = []
+ for file_path in file_paths:
+ if os.path.isfile(file_path):
+ valid_files.append(file_path)
+ self._create_single_preview_widget(file_path)
+ else:
+ # Skip folders, they will be handled differently
+ print(f"Skipping folder: {file_path}")
+
+ self._update_info_label(len(valid_files), valid_files)
+
+ def clear_previews(self):
+ """清除所有预览"""
+ # 清除现有的预览widget
+ for i in reversed(range(self.preview_layout.count())):
+ child = self.preview_layout.itemAt(i).widget()
+ if child:
+ child.setParent(None)
+ child.deleteLater()
+
+ self.current_previews = []
+
+ def _create_single_preview_widget(self, file_path):
+ """为单张图片创建预览widget"""
+ if not os.path.exists(file_path):
+ return
+
+ # Skip folders, they are not image files
+ if os.path.isdir(file_path):
+ print(f"Skipping folder preview: {file_path}")
+ return
+
+ # 创建单个预览的容器
+ preview_widget = QWidget()
+ preview_widget.setFixedSize(300, 320) # 固定大小
+ preview_widget.setStyleSheet("""
+ QWidget {
+ border: 1px solid #ddd;
+ border-radius: 8px;
+ background-color: white;
+ }
+ QWidget:hover {
+ border-color: #007acc;
+ background-color: #f8f9fa;
+ }
+ """)
+
+ preview_layout = QVBoxLayout(preview_widget)
+ preview_layout.setContentsMargins(10, 10, 10, 10)
+ preview_layout.setSpacing(10)
+
+ # 图片显示区域
+ image_label = QLabel()
+ image_label.setAlignment(Qt.AlignmentFlag.AlignCenter)
+ image_label.setFixedSize(280, 220)
+ image_label.setStyleSheet("border: 1px solid #eee; border-radius: 4px; background-color: #f5f5f5;")
+
+ # 加载并显示图片
+ if self._can_load_image(file_path):
+ pixmap = None
+ try:
+ pixmap = QPixmap(file_path)
+ if not pixmap.isNull():
+ # 缩放图片以适应显示区域
+ scaled_pixmap = pixmap.scaled(
+ 270, 210,
+ Qt.AspectRatioMode.KeepAspectRatio,
+ Qt.TransformationMode.SmoothTransformation
+ )
+ image_label.setPixmap(scaled_pixmap)
+ else:
+ self._set_thumbnail_placeholder(image_label, file_path)
+ except Exception as e:
+ print(f"Error loading preview for {file_path}: {e}")
+ self._set_thumbnail_placeholder(image_label, file_path)
+ else:
+ self._set_thumbnail_placeholder(image_label, file_path)
+
+ preview_layout.addWidget(image_label, 0, Qt.AlignmentFlag.AlignCenter)
+
+ # 文件信息
+ info_widget = QWidget()
+ info_layout = QVBoxLayout(info_widget)
+ info_layout.setContentsMargins(0, 0, 0, 0)
+ info_layout.setSpacing(2)
+
+ # 文件名
+ filename_label = QLabel(os.path.basename(file_path))
+ filename_label.setAlignment(Qt.AlignmentFlag.AlignCenter)
+ filename_label.setStyleSheet("font-weight: bold; color: #333;")
+ filename_label.setWordWrap(True)
+
+ # 文件大小
+ try:
+ file_size = os.path.getsize(file_path)
+ size_text = self._format_file_size(file_size)
+ except:
+ size_text = "Unknown size"
+
+ size_label = QLabel(size_text)
+ size_label.setAlignment(Qt.AlignmentFlag.AlignCenter)
+ size_label.setStyleSheet("color: #666; font-size: 11px;")
+
+ info_layout.addWidget(filename_label)
+ info_layout.addWidget(size_label)
+ preview_layout.addWidget(info_widget)
+
+ # 添加到容器
+ self.preview_layout.addWidget(preview_widget, 0, Qt.AlignmentFlag.AlignCenter)
+ self.current_previews.append((file_path, preview_widget))
+
+ def _can_load_image(self, file_path):
+ """检查是否能直接加载图片"""
+ if not file_path:
+ return False
+
+ _, ext = os.path.splitext(file_path.lower())
+ direct_load_formats = {'.png', '.jpg', '.jpeg', '.bmp', '.gif', '.tiff', '.ico', '.icns', '.webp'}
+
+ return ext in direct_load_formats
+
+ def _set_thumbnail_placeholder(self, label, file_path):
+ """设置缩略图占位符"""
+ _, ext = os.path.splitext(file_path.lower())
+
+ # 格式图标映射
+ format_icons = {
+ '.svg': '🖼️',
+ '.pdf': '📄',
+ '.eps': '📄',
+ '.heic': '📱',
+ '.heif': '📱',
+ '.avif': '🖼️',
+ '.jxl': '🖼️',
+ '.dds': '🎮',
+ '.exr': '🎬'
+ }
+
+ icon = format_icons.get(ext, '📷')
+
+ # 创建占位符
+ placeholder_pixmap = QPixmap(270, 210)
+ placeholder_pixmap.fill(QColor('#f5f5f5'))
+
+ painter = QPainter(placeholder_pixmap)
+ painter.setPen(QColor('#999'))
+ painter.setFont(QFont('Arial', 48))
+
+ # 绘制图标
+ painter.drawText(placeholder_pixmap.rect(), Qt.AlignmentFlag.AlignCenter, icon)
+
+ # 绘制格式信息
+ painter.setFont(QFont('Arial', 12))
+ painter.drawText(
+ QRect(0, 170, 270, 40),
+ Qt.AlignmentFlag.AlignCenter,
+ f"{ext.upper()[1:]} Format"
+ )
+
+ painter.end()
+
+ label.setPixmap(placeholder_pixmap)
+
+ # 设置tooltip
+ label.setToolTip(f"Preview not available - {ext.upper()[1:]} Format\nWill show converted result")
+
+ def _format_file_size(self, size_bytes):
+ """格式化文件大小"""
+ if size_bytes == 0:
+ return "0 B"
+
+ size_names = ["B", "KB", "MB", "GB"]
+ import math
+ i = int(math.floor(math.log(size_bytes, 1024)))
+ p = math.pow(1024, i)
+ s = round(size_bytes / p, 2)
+ return f"{s} {size_names[i]}"
+
+ def _update_info_label(self, count, file_paths):
+ """更新信息标签"""
+ if count == 0:
+ self.info_label.setText("Drag image files to the tab above for preview")
+ self.info_label.setStyleSheet("color: #666; font-style: italic;")
+ elif count == 1:
+ filename = os.path.basename(file_paths[0])
+ self.info_label.setText(f"Previewing: {filename}")
+ self.info_label.setStyleSheet("color: #007acc; font-weight: bold;")
+ else:
+ self.info_label.setText(f"Previewing {count} images")
+ self.info_label.setStyleSheet("color: #28a745; font-weight: bold;")
+
+
+class ThumbnailGridWidget(QWidget):
+ """Custom widget for displaying thumbnail grid of batch files"""
+ def __init__(self, parent=None):
+ super().__init__(parent)
+ self.setMinimumHeight(150)
+ self.thumbnails = []
+ self.setup_ui()
+
+ def setup_ui(self):
+ layout = QVBoxLayout(self)
+
+ self.scroll_area = ScrollArea()
+ self.scroll_area.setWidgetResizable(True)
+ self.scroll_area.setMaximumHeight(200)
+
+ self.thumbnail_container = QWidget()
+ self.thumbnail_layout = QVBoxLayout(self.thumbnail_container)
+
+ self.scroll_area.setWidget(self.thumbnail_container)
+ layout.addWidget(self.scroll_area)
+
+ # Placeholder text
+ placeholder_label = QLabel("No files selected")
+ placeholder_label.setAlignment(Qt.AlignmentFlag.AlignCenter)
+ placeholder_label.setStyleSheet("color: #999; font-style: italic;")
+ self.thumbnail_layout.addWidget(placeholder_label)
+
+ def add_thumbnails(self, file_paths):
+ """Add thumbnails for the given file paths"""
+ # Clear existing thumbnails
+ self.clear_thumbnails()
+
+ # Define supported image formats
+ supported_formats = {'.png', '.jpg', '.jpeg', '.webp', '.ico', '.bmp', '.tiff', '.tif', '.gif', '.svg', '.psd'}
+
+ for file_path in file_paths:
+ try:
+ if os.path.exists(file_path):
+ _, ext = os.path.splitext(file_path.lower())
+
+ if ext in supported_formats:
+ thumbnail = self.create_thumbnail_widget(file_path, ext)
+ self.thumbnail_layout.addWidget(thumbnail)
+ self.thumbnails.append(thumbnail)
+ else:
+ # Create placeholder for unsupported formats
+ thumbnail = self.create_unsupported_thumbnail(file_path, ext)
+ self.thumbnail_layout.addWidget(thumbnail)
+ self.thumbnails.append(thumbnail)
+ except Exception as e:
+ print(f"Error creating thumbnail for {file_path}: {e}")
+
+ # Remove placeholder if thumbnails were added
+ if self.thumbnails:
+ for i in range(self.thumbnail_layout.count()):
+ item = self.thumbnail_layout.itemAt(i)
+ if item and item.widget() and isinstance(item.widget(), QLabel):
+ if item.widget().text() == "No files selected":
+ widget = item.widget()
+ self.thumbnail_layout.removeWidget(widget)
+ widget.deleteLater()
+ break
+
+ def create_thumbnail_widget(self, file_path, file_ext):
+ """Create a thumbnail widget for a single file"""
+ container = QWidget()
+ container.setMaximumHeight(80)
+ container.setStyleSheet("""
+ QWidget {
+ border: 1px solid #ddd;
+ border-radius: 5px;
+ margin: 2px;
+ padding: 5px;
+ }
+ QWidget:hover {
+ border-color: #007acc;
+ background-color: #f5f5f5;
+ }
+ """)
+
+ layout = QHBoxLayout(container)
+ layout.setContentsMargins(5, 5, 5, 5)
+
+ # Thumbnail image or icon
+ thumbnail_label = QLabel()
+ thumbnail_label.setFixedSize(60, 60)
+ thumbnail_label.setAlignment(Qt.AlignmentFlag.AlignCenter)
+
+ # Try to load image for supported formats
+ if self._can_load_image(file_ext):
+ pixmap = None
+ try:
+ pixmap = QPixmap(file_path)
+ if not pixmap.isNull():
+ scaled_pixmap = pixmap.scaled(60, 60, Qt.AspectRatioMode.KeepAspectRatio,
+ Qt.TransformationMode.SmoothTransformation)
+ thumbnail_label.setPixmap(scaled_pixmap)
+ else:
+ self._set_thumbnail_placeholder(thumbnail_label, file_ext, "Load Error")
+ except Exception as e:
+ print(f"Error loading image {file_path}: {e}")
+ self._set_thumbnail_placeholder(thumbnail_label, file_ext, "Error")
+ else:
+ self._set_thumbnail_placeholder(thumbnail_label, file_ext)
+
+ # File info
+ info_layout = QVBoxLayout()
+ filename_label = QLabel(os.path.basename(file_path))
+ filename_label.setStyleSheet("font-weight: bold; font-size: 12px;")
+
+ # Get file size
+ try:
+ size_bytes = os.path.getsize(file_path)
+ size_str = self.format_file_size(size_bytes)
+ except:
+ size_str = "Unknown size"
+
+ size_label = QLabel(f"Size: {size_str}")
+ size_label.setStyleSheet("color: #666; font-size: 10px;")
+
+ info_layout.addWidget(filename_label)
+ info_layout.addWidget(size_label)
+ info_layout.addStretch()
+
+ layout.addWidget(thumbnail_label)
+ layout.addLayout(info_layout)
+ layout.addStretch()
+
+ return container
+
+ def create_unsupported_thumbnail(self, file_path, file_ext):
+ """Create a thumbnail widget for unsupported file formats"""
+ container = QWidget()
+ container.setMaximumHeight(80)
+ container.setStyleSheet("""
+ QWidget {
+ border: 1px dashed #ccc;
+ border-radius: 5px;
+ margin: 2px;
+ padding: 5px;
+ background-color: #f9f9f9;
+ }
+ QWidget:hover {
+ border-color: #999;
+ background-color: #f0f0f0;
+ }
+ """)
+
+ layout = QHBoxLayout(container)
+ layout.setContentsMargins(5, 5, 5, 5)
+
+ # Icon for unsupported format
+ icon_label = QLabel()
+ icon_label.setFixedSize(60, 60)
+ icon_label.setAlignment(Qt.AlignmentFlag.AlignCenter)
+ self._set_thumbnail_placeholder(icon_label, file_ext, show_format=True)
+
+ # File info
+ info_layout = QVBoxLayout()
+ filename_label = QLabel(os.path.basename(file_path))
+ filename_label.setStyleSheet("font-weight: bold; font-size: 12px; color: #666;")
+
+ # Get file size
+ try:
+ size_bytes = os.path.getsize(file_path)
+ size_str = self.format_file_size(size_bytes)
+ except:
+ size_str = "Unknown size"
+
+ size_label = QLabel(f"Size: {size_str}")
+ size_label.setStyleSheet("color: #999; font-size: 10px;")
+
+ info_layout.addWidget(filename_label)
+ info_layout.addWidget(size_label)
+ info_layout.addStretch()
+
+ layout.addWidget(icon_label)
+ layout.addLayout(info_layout)
+ layout.addStretch()
+
+ return container
+
+ def _can_load_image(self, file_ext):
+ """Check if the file extension can be loaded as an image by Qt"""
+ # Qt supports most common formats, but some require plugins
+ loadable_formats = {'.png', '.jpg', '.jpeg', '.webp', '.ico', '.bmp', '.tif', '.tiff', '.gif'}
+ return file_ext.lower() in loadable_formats
+
+ def _set_thumbnail_placeholder(self, label, file_ext, error_text=None, show_format=False):
+ """Set placeholder text and style for thumbnail label"""
+ if error_text:
+ # Error state
+ label.setText(error_text)
+ label.setStyleSheet("""
+ QLabel {
+ border: 1px solid #ddd;
+ border-radius: 5px;
+ background-color: #ffe6e6;
+ color: #dc3545;
+ font-size: 10px;
+ font-weight: bold;
+ }
+ """)
+ elif show_format:
+ # Show format information for unsupported but recognized formats
+ format_name = file_ext.upper()[1:] if file_ext.startswith('.') else file_ext.upper()
+ icon_map = {
+ 'SVG': '🎨',
+ 'PSD': '🎨',
+ 'BMP': '🖼️',
+ 'TIFF': '🖼️',
+ 'TIF': '🖼️',
+ 'GIF': '🖼️',
+ 'PNG': '🖼️',
+ 'JPG': '🖼️',
+ 'JPEG': '🖼️',
+ 'WEBP': '🖼️',
+ 'ICO': '🖼️'
+ }
+ icon = icon_map.get(format_name, '📄')
+ label.setText(f"{icon}\n{format_name}")
+ label.setStyleSheet("""
+ QLabel {
+ border: 1px solid #ddd;
+ border-radius: 5px;
+ background-color: #f5f5f5;
+ color: #666;
+ font-size: 8px;
+ font-weight: bold;
+ text-align: center;
+ }
+ """)
+ else:
+ # Generic file icon
+ label.setText("📄")
+ label.setStyleSheet("""
+ QLabel {
+ border: 1px solid #ddd;
+ border-radius: 5px;
+ background-color: #f5f5f5;
+ color: #666;
+ font-size: 24px;
+ }
+ """)
+
+ def clear_thumbnails(self):
+ """Clear all thumbnails"""
+ # Remove all thumbnail widgets
+ for thumbnail in self.thumbnails:
+ if thumbnail and thumbnail.parent():
+ thumbnail.setParent(None)
+ thumbnail.deleteLater()
+
+ self.thumbnails.clear()
+
+ # Add placeholder back
+ placeholder_label = QLabel("No files selected")
+ placeholder_label.setAlignment(Qt.AlignmentFlag.AlignCenter)
+ placeholder_label.setStyleSheet("color: #999; font-style: italic;")
+ self.thumbnail_layout.addWidget(placeholder_label)
+
+ def format_file_size(self, size_bytes):
+ """Format file size in human readable format"""
+ if size_bytes < 1024:
+ return f"{size_bytes} B"
+ elif size_bytes < 1024 * 1024:
+ return f"{size_bytes / 1024:.1f} KB"
+ else:
+ return f"{size_bytes / (1024 * 1024):.1f} MB"
diff --git a/support/__pycache__/convert.cpython-314.pyc b/support/__pycache__/convert.cpython-314.pyc
index a5343ad..ed845e0 100644
Binary files a/support/__pycache__/convert.cpython-314.pyc and b/support/__pycache__/convert.cpython-314.pyc differ
diff --git a/support/archive_manager.py b/support/archive_manager.py
index f284b3a..f7842df 100644
--- a/support/archive_manager.py
+++ b/support/archive_manager.py
@@ -122,12 +122,12 @@ def _get_archive_type(file_path):
return "lzh"
return None
-def _run_command_with_timeout(cmd, timeout=2, progress_callback=None):
+def _run_command_with_timeout(cmd, timeout=2, progress_callback=None, cwd=None):
"""Run command with timeout limit"""
start_time = time.time()
try:
- result = subprocess.run(cmd, capture_output=True, text=True, timeout=timeout)
+ result = subprocess.run(cmd, capture_output=True, text=True, timeout=timeout, cwd=cwd)
if progress_callback:
elapsed = time.time() - start_time
@@ -355,13 +355,13 @@ def create_archive(output_path, source_paths, archive_format, progress_callback=
if progress_callback:
progress_callback(f"Starting {archive_format} archive creation...", 0)
- # For password-protected ZIP files, use direct processing without wrapper directory
+ # For password-protected ZIP files, use CLI tool
if archive_format == "zip" and password:
if progress_callback:
- progress_callback(f"Creating password-protected ZIP file directly...", 10)
+ progress_callback(f"Creating password-protected ZIP file with CLI tool...", 10)
- # Use a specialized function for password-protected ZIP files
- success = _create_password_protected_zip(output_path, source_paths, password, progress_callback)
+ # Use CLI tool for password-protected ZIP files
+ success = _create_zip_with_cli(output_path, source_paths, progress_callback, password)
if not success:
raise RuntimeError(f"Failed to create password-protected ZIP archive")
else:
@@ -370,10 +370,14 @@ def create_archive(output_path, source_paths, archive_format, progress_callback=
temp_dir = tempfile.mkdtemp()
wrapper_dir = os.path.join(temp_dir, archive_name)
- # Ensure the wrapper directory doesn't already exist
- if os.path.exists(wrapper_dir):
- shutil.rmtree(wrapper_dir)
- os.makedirs(wrapper_dir)
+ # Create the wrapper directory - ensure it doesn't already exist
+ try:
+ os.makedirs(wrapper_dir, exist_ok=False)
+ except FileExistsError:
+ # If directory already exists, remove it and create a new one
+ if os.path.exists(wrapper_dir):
+ shutil.rmtree(wrapper_dir)
+ os.makedirs(wrapper_dir)
try:
# Copy all source files to the wrapper directory
@@ -405,9 +409,12 @@ def create_archive(output_path, source_paths, archive_format, progress_callback=
progress_callback(f"Creating {archive_format} archive...", 40)
success = False
- if archive_format in ["zip", "tar", "tar.gz", "tar.bz2", "tar.xz", "zipx"]:
- # Use patool for processing
- success = _create_with_patool(output_path, [wrapped_source_path], archive_format, progress_callback, password)
+ if archive_format == "zip":
+ # Use CLI tool for ZIP creation
+ success = _create_zip_with_cli(output_path, [wrapped_source_path], progress_callback, password)
+ elif archive_format in ["tar", "tar.gz", "tar.bz2", "tar.xz", "zipx"]:
+ # Use CLI tool for processing
+ success = _create_tar_with_cli(output_path, [wrapped_source_path], archive_format, progress_callback)
elif archive_format in ["bz2", "xz", "lzma", "gz"]:
# These formats can only compress single files, so need to create tar first, then compress
success = _create_single_file_compression(output_path, [wrapped_source_path], archive_format, progress_callback)
@@ -445,279 +452,213 @@ def create_archive(output_path, source_paths, archive_format, progress_callback=
progress_callback(f"Error creating archive: {str(e)}", -1)
return False
-def _create_with_patool(output_path, source_paths, archive_format, progress_callback=None, password=None):
- """Create archive file using patool"""
- import subprocess
- try:
- import patoolib
- except ImportError:
- raise ImportError("patool is required for this format. Install with: pip install patool")
-
- # Create temporary directory
- temp_dir = tempfile.mkdtemp()
+
+
+def _create_single_file_compression(output_path, source_paths, compression_format, progress_callback=None):
+ """
+ Create single file compression format (bz2, xz, lzma, gz)
+ These formats can only compress single files, so need to create tar first, then compress
+ """
try:
- # Copy all source files to temporary directory
- for source_path in source_paths:
- if os.path.isfile(source_path):
- shutil.copy2(source_path, temp_dir)
- elif os.path.isdir(source_path):
- dest_dir = os.path.join(temp_dir, os.path.basename(source_path))
- # Ensure the destination directory doesn't already exist
- if os.path.exists(dest_dir):
- shutil.rmtree(dest_dir)
- shutil.copytree(source_path, dest_dir)
-
- # Use patool to create archive - fix path issues, use full paths
- temp_files = []
- for item in os.listdir(temp_dir):
- item_path = os.path.join(temp_dir, item)
- temp_files.append(item_path)
-
- if not temp_files:
- raise ValueError("No files found to archive")
+ # Create temporary directory
+ temp_dir = tempfile.mkdtemp()
+ try:
+ # Step 1: Create tar file using tar command
+ base_name = os.path.splitext(os.path.basename(output_path))[0]
+ tar_path = os.path.join(temp_dir, f"{base_name}.tar")
- if progress_callback:
- progress_callback(f"Creating {archive_format} archive with patool...", 50)
-
- # Add debug information
- if progress_callback:
- progress_callback(f"Output path: {output_path}", 55)
- progress_callback(f"Temp files: {temp_files}", 60)
-
- # For ZIP format with password, use Python's zipfile module
- if archive_format == "zip" and password:
if progress_callback:
- progress_callback(f"Creating password-protected ZIP file...", 65)
-
- password_success = False
-
- # First try to use zip command if available (most compatible)
- zip_tool = _get_cli_tool("zip")
- if zip_tool:
- if progress_callback:
- progress_callback(f"Trying to create password-protected ZIP with zip command...", 70)
-
- # Create a temporary directory for zip to work with
- temp_zip_dir = tempfile.mkdtemp()
- try:
- # Ensure output file doesn't exist
- if os.path.exists(output_path):
- os.remove(output_path)
-
- # Copy files to temp directory
- for file_path in temp_files:
- if os.path.isfile(file_path):
- shutil.copy2(file_path, temp_zip_dir)
- elif os.path.isdir(file_path):
- dest_dir = os.path.join(temp_zip_dir, os.path.basename(file_path))
- shutil.copytree(file_path, dest_dir)
-
- # Create password-protected ZIP with zip command
- cmd = [zip_tool, "-r", "-P", password, output_path, "."]
- result = subprocess.run(cmd, cwd=temp_zip_dir, capture_output=True, text=True)
-
- # Check if the command succeeded (return code 0) and the output file exists
- if result.returncode == 0 and os.path.exists(output_path) and os.path.getsize(output_path) > 0:
- password_success = True
- if progress_callback:
- progress_callback(f"Password-protected ZIP created with zip command", 85)
- else:
- if progress_callback:
- progress_callback(f"Failed to create password-protected ZIP with zip command: return code {result.returncode}", 75)
- finally:
- shutil.rmtree(temp_zip_dir)
+ progress_callback(f"Creating intermediate tar file...", 30)
+
+ # Build tar command
+ cmd = ["tar", "-c", "-f", tar_path]
+
+ # Process source paths
+ if len(source_paths) == 1:
+ source_path = source_paths[0]
+ if os.path.isdir(source_path):
+ # For directories, add directory contents
+ work_dir = os.path.dirname(source_path) or "."
+ rel_path = os.path.basename(source_path)
+ cmd.append(rel_path)
+ else:
+ # For files, use directory containing the file
+ work_dir = os.path.dirname(source_path) or "."
+ rel_path = os.path.basename(source_path)
+ cmd.append(rel_path)
else:
- if progress_callback:
- progress_callback(f"zip command not available, trying pyzipper...", 75)
-
- # If zip command failed, try to use pyzipper
- if not password_success:
- try:
- import pyzipper
- if progress_callback:
- progress_callback(f"Creating password-protected ZIP with pyzipper...", 70)
-
- # Create a password-protected ZIP with pyzipper
- with pyzipper.AESZipFile(output_path, 'w', compression=pyzipper.ZIP_DEFLATED, encryption=pyzipper.WZ_AES) as zf:
- zf.setpassword(password.encode())
- for file_path in temp_files:
- if os.path.isfile(file_path):
- # 只使用文件名,不包含路径
- zf.write(file_path, os.path.basename(file_path))
- elif os.path.isdir(file_path):
- # 对于目录,递归添加所有文件,保留目录结构
- for root, dirs, files in os.walk(file_path):
- for file in files:
- file_full_path = os.path.join(root, file)
- # 计算相对于目录的路径,保留子目录结构
- arcname = os.path.relpath(file_full_path, file_path)
- zf.write(file_full_path, arcname)
-
- password_success = True
- if progress_callback:
- progress_callback(f"Password-protected ZIP created with pyzipper", 85)
-
- except ImportError:
- if progress_callback:
- progress_callback(f"pyzipper not available, trying 7zz...", 75)
- except Exception as e:
- if progress_callback:
- progress_callback(f"Failed to create password-protected ZIP with pyzipper: {str(e)}", 75)
+ # Multiple sources - use parent directory as working directory
+ work_dir = os.path.commonpath([os.path.dirname(p) if os.path.isfile(p) else p for p in source_paths])
+ for source_path in source_paths:
+ if os.path.isfile(source_path):
+ cmd.append(os.path.basename(source_path))
+ elif os.path.isdir(source_path):
+ cmd.append(os.path.basename(source_path))
- # If pyzipper failed, try to use 7zz
- if not password_success:
- sevenz_tool = _get_cli_tool("7zz")
- if sevenz_tool:
- if progress_callback:
- progress_callback(f"Trying to create password-protected ZIP with 7zz...", 75)
-
- # Create a temporary directory for 7zz to work with
- temp_7z_dir = tempfile.mkdtemp()
- try:
- # Copy files to temp directory
- for file_path in temp_files:
- if os.path.isfile(file_path):
- shutil.copy2(file_path, temp_7z_dir)
- elif os.path.isdir(file_path):
- dest_dir = os.path.join(temp_7z_dir, os.path.basename(file_path))
- shutil.copytree(file_path, dest_dir)
-
- # Create password-protected ZIP with 7zz
- cmd = [sevenz_tool, "a", f"-p{password}", "-y", output_path, os.path.join(temp_7z_dir, "*")]
- result = subprocess.run(cmd, capture_output=True, text=True)
-
- if result.returncode == 0:
- password_success = True
- if progress_callback:
- progress_callback(f"Password-protected ZIP created with 7zz", 85)
- else:
- if progress_callback:
- progress_callback(f"Failed to create password-protected ZIP with 7zz: {result.stderr}", 80)
- finally:
- shutil.rmtree(temp_7z_dir)
- else:
- if progress_callback:
- progress_callback(f"7zz not available", 75)
+ # Run tar command
+ result = _run_command_with_timeout(cmd, cwd=work_dir, timeout=30)
- # If pyzipper and 7zz both failed, we'll fall back to standard ZIP without password
- if not password_success:
- if progress_callback:
- progress_callback(f"All password protection methods failed", 75)
+ if result.returncode != 0:
+ raise RuntimeError(f"tar creation failed: {result.stderr}")
- # If all methods failed, create a non-password-protected ZIP
- if not password_success:
- if progress_callback:
- progress_callback(f"Warning: Could not create password-protected ZIP, creating standard ZIP", 75)
-
- with zipfile.ZipFile(output_path, 'w', zipfile.ZIP_DEFLATED) as zf:
- for file_path in temp_files:
- if os.path.isfile(file_path):
- # 只使用文件名,不包含路径
- zf.write(file_path, os.path.basename(file_path))
- elif os.path.isdir(file_path):
- # 对于目录,递归添加所有文件,保留目录结构
- for root, dirs, files in os.walk(file_path):
- for file in files:
- file_full_path = os.path.join(root, file)
- # 计算相对于目录的路径,保留子目录结构
- arcname = os.path.relpath(file_full_path, file_path)
- zf.write(file_full_path, arcname)
-
- if progress_callback:
- progress_callback(f"Standard ZIP created (no password protection)", 85)
+ if not os.path.exists(tar_path):
+ raise RuntimeError("Failed to create intermediate tar file")
+ # Step 2: Compress tar file using appropriate command
if progress_callback:
- progress_callback(f"ZIP archive created", 90)
- else:
- # Try different patool calling methods
- try:
- # Method 1: Directly pass file paths
- patoolib.create_archive(output_path, temp_files)
- except Exception as e1:
- # Method 2: Switch to temporary directory and use relative paths
+ progress_callback(f"Compressing tar file with {compression_format}...", 60)
+
+ # Build compression command based on format
+ if compression_format == "gz":
try:
- original_cwd = os.getcwd()
- os.chdir(temp_dir)
- rel_files = [os.path.basename(f) for f in temp_files]
- patoolib.create_archive(output_path, rel_files)
- os.chdir(original_cwd)
- except Exception as e2:
- os.chdir(original_cwd)
-
- # Method 3: Use full paths and patool's --verbose option
+ gzip_tool = _get_cli_tool("gzip")
+ cmd = [gzip_tool, "-c", tar_path]
+ except FileNotFoundError:
+ # Fallback to system gzip
+ cmd = ["gzip", "-c", tar_path]
+ output_file = open(output_path, "wb")
+ result = subprocess.run(cmd, stdout=output_file, stderr=subprocess.PIPE)
+ output_file.close()
+ elif compression_format == "bz2":
+ try:
+ bzip2_tool = _get_cli_tool("bzip2")
+ cmd = [bzip2_tool, "-c", tar_path]
+ except FileNotFoundError:
+ # Fallback to system bzip2
+ cmd = ["bzip2", "-c", tar_path]
+ output_file = open(output_path, "wb")
+ result = subprocess.run(cmd, stdout=output_file, stderr=subprocess.PIPE)
+ output_file.close()
+ elif compression_format == "xz":
+ try:
+ xz_tool = _get_cli_tool("xz")
+ cmd = [xz_tool, "-c", tar_path]
+ output_file = open(output_path, "wb")
+ result = subprocess.run(cmd, stdout=output_file, stderr=subprocess.PIPE)
+ output_file.close()
+ except FileNotFoundError:
+ # Fallback to 7zz tool
try:
- import subprocess
- cmd = ["patool", "create", output_path] + temp_files
+ sevenz_tool = _get_cli_tool("7zz")
+ cmd = [sevenz_tool, "a", "-txz", output_path, tar_path]
result = subprocess.run(cmd, capture_output=True, text=True)
- if result.returncode != 0:
- raise RuntimeError(f"patool command failed: {result.stderr}")
- except Exception as e3:
- raise RuntimeError(f"All patool methods failed: {e1}, {e2}, {e3}")
+ except FileNotFoundError:
+ # Final fallback to system xz
+ cmd = ["xz", "-c", tar_path]
+ output_file = open(output_path, "wb")
+ result = subprocess.run(cmd, stdout=output_file, stderr=subprocess.PIPE)
+ output_file.close()
+ elif compression_format == "lzma":
+ # Use Python's lzma module directly
+ try:
+ import lzma
+ with open(tar_path, 'rb') as f_in:
+ with lzma.open(output_path, 'wb') as f_out:
+ shutil.copyfileobj(f_in, f_out)
+ # Create a dummy successful result
+ class DummyResult:
+ def __init__(self):
+ self.returncode = 0
+ self.stderr = ""
+ result = DummyResult()
+ except Exception as e:
+ raise RuntimeError(f"Failed to create LZMA archive: {str(e)}")
+ else:
+ raise ValueError(f"Unsupported compression format: {compression_format}")
- if progress_callback:
- progress_callback(f"{archive_format} archive created", 90)
+ if result.returncode != 0:
+ raise RuntimeError(f"Compression failed: {result.stderr}")
- return True
+ if progress_callback:
+ progress_callback(f"{compression_format} archive created", 90)
+
+ return True
+ finally:
+ shutil.rmtree(temp_dir)
except Exception as e:
if progress_callback:
- progress_callback(f"Error creating {archive_format} archive: {str(e)}", -1)
+ progress_callback(f"Error creating {compression_format} archive: {str(e)}", -1)
return False
- finally:
- shutil.rmtree(temp_dir)
-def _create_single_file_compression(output_path, source_paths, compression_format, progress_callback=None):
- """
- Create single file compression format (bz2, xz, lzma)
- These formats can only compress single files, so need to create tar first, then compress
- """
+def _create_zip_with_cli(output_path, source_paths, progress_callback=None, password=None):
+ """Create ZIP file using zip CLI tool"""
try:
- import patoolib
- except ImportError:
- raise ImportError("patool is required for this format. Install with: pip install patool")
+ zip_tool = _get_cli_tool("zip")
+ except FileNotFoundError:
+ # If zip tool is not found, raise an error instead of falling back to patool
+ raise RuntimeError("zip tool not found. Please install zip command-line tool")
+
+ # Build zip command - use working directory approach
+ temp_dir = None # Initialize temp_dir to None
+ if len(source_paths) == 1:
+ # Single source - can use directly
+ source_path = source_paths[0]
+ if os.path.isdir(source_path):
+ # For directories, use relative path from parent directory
+ work_dir = os.path.dirname(source_path) or "."
+ rel_path = os.path.basename(source_path)
+ else:
+ # For files, use directory containing the file
+ work_dir = os.path.dirname(source_path) or "."
+ rel_path = os.path.basename(source_path)
+ else:
+ # Multiple sources - create temporary directory approach
+ temp_dir = tempfile.mkdtemp()
+ work_dir = temp_dir
+ rel_paths = []
+
+ try:
+ for source_path in source_paths:
+ if os.path.isfile(source_path):
+ dest_path = os.path.join(temp_dir, os.path.basename(source_path))
+ shutil.copy2(source_path, dest_path)
+ rel_paths.append(os.path.basename(source_path))
+ elif os.path.isdir(source_path):
+ dest_path = os.path.join(temp_dir, os.path.basename(source_path))
+ shutil.copytree(source_path, dest_path)
+ rel_paths.append(os.path.basename(source_path))
+ except Exception:
+ if temp_dir and os.path.exists(temp_dir):
+ shutil.rmtree(temp_dir)
+ raise
+
+ source_paths = rel_paths
- # Create temporary directory
- temp_dir = tempfile.mkdtemp()
try:
- # Step 1: Create tar file
- base_name = os.path.splitext(os.path.basename(output_path))[0]
- tar_path = os.path.join(temp_dir, f"{base_name}.tar")
+ cmd = [zip_tool, "-r"]
+ if password:
+ cmd.extend(["-P", password])
+ cmd.append(output_path)
+ cmd.extend(source_paths if len(source_paths) > 1 else [rel_path])
+
+ # Ensure output path is absolute
+ output_path_abs = os.path.abspath(output_path)
+ cmd[cmd.index(output_path)] = output_path_abs
if progress_callback:
- progress_callback(f"Creating intermediate tar file...", 50)
+ progress_callback(f"Creating ZIP archive with CLI tool...", 50)
- # Create tar file
- patoolib.create_archive(tar_path, source_paths)
+ result = _run_command_with_timeout(cmd, cwd=work_dir, timeout=2, progress_callback=progress_callback)
- if not os.path.exists(tar_path):
- raise RuntimeError("Failed to create intermediate tar file")
+ if result.returncode != 0:
+ raise RuntimeError(f"ZIP creation failed: {result.stderr}")
- # Step 2: Compress tar file
if progress_callback:
- progress_callback(f"Compressing tar file with {compression_format}...", 70)
-
- # Use patool to compress tar file
- patoolib.create_archive(output_path, [tar_path])
+ progress_callback(f"ZIP archive created successfully", 90)
- if progress_callback:
- progress_callback(f"{compression_format} archive created", 90)
-
return True
- except Exception as e:
- if progress_callback:
- progress_callback(f"Error creating {compression_format} archive: {str(e)}", -1)
- return False
finally:
- shutil.rmtree(temp_dir)
+ # Clean up temporary directory if it was created
+ if temp_dir and os.path.exists(temp_dir):
+ shutil.rmtree(temp_dir)
def _create_cab_with_cli(output_path, source_paths, progress_callback=None):
"""Create CAB archive file using gcab tool"""
- try:
- gcab_tool = _get_cli_tool("gcab")
- except FileNotFoundError:
- # Try to get gcab from system path
- gcab_tool = shutil.which("gcab")
- if not gcab_tool:
- raise RuntimeError("gcab tool not found. Please install with: brew install gcab")
+ # Use project's built-in gcab tool directly
+ gcab_tool = os.path.join(CLI_BASE_PATH, "Universal", "gcab")
+
+ if not os.path.exists(gcab_tool):
+ raise RuntimeError(f"Built-in gcab tool not found at {gcab_tool}")
# Create temporary directory for source files
with tempfile.TemporaryDirectory() as temp_dir:
@@ -744,12 +685,12 @@ def _create_cab_with_cli(output_path, source_paths, progress_callback=None):
# Update progress
if progress_callback:
progress = (i + 1) / len(source_paths) * 50 # First half of progress for copying files
- progress_callback(progress)
+ progress_callback(f"Copying files for CAB archive...", progress)
if not copied_files:
raise ValueError("No valid source files found")
- # Build gcab command
+ # Build gcab command for CAB format
cmd = [gcab_tool, "-c", "-n", output_path]
cmd.extend([str(f) for f in copied_files])
@@ -765,7 +706,7 @@ def _create_cab_with_cli(output_path, source_paths, progress_callback=None):
# Update final progress
if progress_callback:
- progress_callback(100)
+ progress_callback("CAB archive created successfully", 100)
return True
except subprocess.CalledProcessError as e:
@@ -779,21 +720,140 @@ def _create_rar_with_cli(output_path, source_paths, progress_callback=None, pass
# Fallback: Use unar/lsar
return _create_with_unar(output_path, source_paths, "rar", progress_callback)
- # Build rar command
- cmd = [rar_tool, "a", "-r"]
- if password:
- cmd.extend(["-p" + password, "-y"])
+ # Build rar command - use working directory approach
+ temp_dir = None # Initialize temp_dir to None
+ if len(source_paths) == 1:
+ # Single source - can use directly
+ source_path = source_paths[0]
+ if os.path.isdir(source_path):
+ # For directories, use relative path from parent directory
+ work_dir = os.path.dirname(source_path) or "."
+ rel_path = os.path.basename(source_path)
+ else:
+ # For files, use directory containing the file
+ work_dir = os.path.dirname(source_path) or "."
+ rel_path = os.path.basename(source_path)
else:
- cmd.append("-y")
- cmd.append(output_path)
- cmd.extend(source_paths)
-
- result = _run_command_with_timeout(cmd, timeout=2, progress_callback=progress_callback)
+ # Multiple sources - create temporary directory approach
+ temp_dir = tempfile.mkdtemp()
+ work_dir = temp_dir
+ rel_paths = []
+
+ try:
+ for source_path in source_paths:
+ if os.path.isfile(source_path):
+ dest_path = os.path.join(temp_dir, os.path.basename(source_path))
+ shutil.copy2(source_path, dest_path)
+ rel_paths.append(os.path.basename(source_path))
+ elif os.path.isdir(source_path):
+ dest_path = os.path.join(temp_dir, os.path.basename(source_path))
+ shutil.copytree(source_path, dest_path)
+ rel_paths.append(os.path.basename(source_path))
+ except Exception:
+ if temp_dir and os.path.exists(temp_dir):
+ shutil.rmtree(temp_dir)
+ raise
+
+ source_paths = rel_paths
- if result.returncode != 0:
- raise RuntimeError(f"RAR creation failed: {result.stderr}")
+ try:
+ cmd = [rar_tool, "a", "-r"]
+ if password:
+ cmd.extend(["-p" + password, "-y"])
+ else:
+ cmd.append("-y")
+ cmd.append(output_path)
+ cmd.extend(source_paths if len(source_paths) > 1 else [rel_path])
+
+ # Ensure output path is absolute
+ output_path_abs = os.path.abspath(output_path)
+ cmd[cmd.index(output_path)] = output_path_abs
+
+ result = _run_command_with_timeout(cmd, cwd=work_dir, timeout=2, progress_callback=progress_callback)
+
+ if result.returncode != 0:
+ raise RuntimeError(f"RAR creation failed: {result.stderr}")
+
+ return True
+ finally:
+ # Clean up temporary directory if it was created
+ if temp_dir and os.path.exists(temp_dir):
+ shutil.rmtree(temp_dir)
+
+def _create_tar_with_cli(output_path, source_paths, archive_format, progress_callback=None, password=None):
+ """Create tar file using system tar command"""
+ try:
+ if progress_callback:
+ progress_callback(f"Creating {archive_format} archive with tar command...", 0)
+
+ # Ensure output path is absolute
+ output_path_abs = os.path.abspath(output_path)
+
+ # Build tar command based on format
+ cmd = ["tar", "-c"]
+
+ # Add compression options based on format
+ if archive_format == "tar.gz" or archive_format == "tgz":
+ cmd.append("-z")
+ elif archive_format == "tar.bz2" or archive_format == "tbz2":
+ cmd.append("-j")
+ elif archive_format == "tar.xz" or archive_format == "txz":
+ cmd.append("-J")
+
+ # Add output file option
+ cmd.extend(["-f", output_path_abs])
+
+ # Handle password protection for zipx format
+ if archive_format == "zipx" and password:
+ # For zipx, we'll create a standard zip with password protection
+ return _create_zip_with_cli(output_path, source_paths, progress_callback, password)
+
+ # Process source paths
+ if len(source_paths) == 1:
+ source_path = source_paths[0]
+ if os.path.isdir(source_path):
+ # For directories, add directory contents
+ work_dir = os.path.dirname(source_path) or "."
+ rel_path = os.path.basename(source_path)
+ cmd.append(rel_path)
+ else:
+ # For files, use directory containing the file
+ work_dir = os.path.dirname(source_path) or "."
+ rel_path = os.path.basename(source_path)
+ cmd.append(rel_path)
+ else:
+ # Multiple sources - use parent directory as working directory
+ work_dir = os.path.commonpath([os.path.dirname(p) if os.path.isfile(p) else p for p in source_paths])
+ for source_path in source_paths:
+ if os.path.isfile(source_path):
+ cmd.append(os.path.basename(source_path))
+ elif os.path.isdir(source_path):
+ cmd.append(os.path.basename(source_path))
+
+ if progress_callback:
+ progress_callback(f"Running tar command for {archive_format}...", 50)
+
+ # Run tar command
+ result = _run_command_with_timeout(cmd, cwd=work_dir, timeout=30, progress_callback=progress_callback)
+
+ if result.returncode != 0:
+ raise RuntimeError(f"tar creation failed: {result.stderr}")
+
+ # Handle zipx format (rename tar to zipx)
+ if archive_format == "zipx":
+ temp_path = output_path_abs + ".tmp"
+ os.rename(output_path_abs, temp_path)
+ os.rename(temp_path, output_path)
+
+ if progress_callback:
+ progress_callback(f"{archive_format} archive created successfully", 100)
+
+ return True
- return True
+ except Exception as e:
+ if progress_callback:
+ progress_callback(f"Error creating {archive_format} archive: {str(e)}", -1)
+ return False
def _create_7z_with_cli(output_path, source_paths, progress_callback=None, password=None):
"""Create 7z file using 7zz CLI tool"""
@@ -804,20 +864,65 @@ def _create_7z_with_cli(output_path, source_paths, progress_callback=None, passw
_create_with_unar(output_path, source_paths, "7z", progress_callback)
return True
- cmd = [sevenz_tool, "a"]
- if password:
- cmd.extend(["-p" + password, "-y"])
+ # Build 7z command - use working directory approach
+ temp_dir = None # Initialize temp_dir to None
+ if len(source_paths) == 1:
+ # Single source - can use directly
+ source_path = source_paths[0]
+ if os.path.isdir(source_path):
+ # For directories, use relative path from parent directory
+ work_dir = os.path.dirname(source_path) or "."
+ rel_path = os.path.basename(source_path)
+ else:
+ # For files, use directory containing the file
+ work_dir = os.path.dirname(source_path) or "."
+ rel_path = os.path.basename(source_path)
else:
- cmd.append("-y")
- cmd.append(output_path)
- cmd.extend(source_paths)
-
- result = _run_command_with_timeout(cmd, timeout=2, progress_callback=progress_callback)
-
- if result.returncode != 0:
- raise RuntimeError(f"7z creation failed: {result.stderr}")
+ # Multiple sources - create temporary directory approach
+ temp_dir = tempfile.mkdtemp()
+ work_dir = temp_dir
+ rel_paths = []
+
+ try:
+ for source_path in source_paths:
+ if os.path.isfile(source_path):
+ dest_path = os.path.join(temp_dir, os.path.basename(source_path))
+ shutil.copy2(source_path, dest_path)
+ rel_paths.append(os.path.basename(source_path))
+ elif os.path.isdir(source_path):
+ dest_path = os.path.join(temp_dir, os.path.basename(source_path))
+ shutil.copytree(source_path, dest_path)
+ rel_paths.append(os.path.basename(source_path))
+ except Exception:
+ if temp_dir and os.path.exists(temp_dir):
+ shutil.rmtree(temp_dir)
+ raise
+
+ source_paths = rel_paths
- return True
+ try:
+ cmd = [sevenz_tool, "a"]
+ if password:
+ cmd.extend(["-p" + password, "-y"])
+ else:
+ cmd.append("-y")
+ cmd.append(output_path)
+ cmd.extend(source_paths if len(source_paths) > 1 else [rel_path])
+
+ # Ensure output path is absolute
+ output_path_abs = os.path.abspath(output_path)
+ cmd[cmd.index(output_path)] = output_path_abs
+
+ result = _run_command_with_timeout(cmd, cwd=work_dir, timeout=2, progress_callback=progress_callback)
+
+ if result.returncode != 0:
+ raise RuntimeError(f"7z creation failed: {result.stderr}")
+
+ return True
+ finally:
+ # Clean up temporary directory if it was created
+ if temp_dir and os.path.exists(temp_dir):
+ shutil.rmtree(temp_dir)
def _create_iso_with_system(output_path, source_paths, progress_callback=None):
"""Create ISO file using system command"""
@@ -826,12 +931,15 @@ def _create_iso_with_system(output_path, source_paths, progress_callback=None):
source_dir = source_paths[0]
+ # Ensure output path is absolute
+ output_path_abs = os.path.abspath(output_path)
+
# Try using hdiutil (macOS)
if platform.system() == "Darwin":
- cmd = ["hdiutil", "makehybrid", "-o", output_path, "-hfs", "-iso", "-joliet", source_dir]
+ cmd = ["hdiutil", "makehybrid", "-o", output_path_abs, "-hfs", "-iso", "-joliet", source_dir]
else:
# Other systems use mkisofs
- cmd = ["mkisofs", "-o", output_path, "-J", "-R", source_dir]
+ cmd = ["mkisofs", "-o", output_path_abs, "-J", "-R", source_dir]
result = _run_command_with_timeout(cmd, timeout=60, progress_callback=progress_callback)
@@ -958,11 +1066,8 @@ def extract_archive(archive_path, extract_to, progress_callback=None, password=N
# Use unar as fallback on failure
_extract_with_unar(archive_path, extract_to, progress_callback, password)
elif archive_format in ["zipx"]:
- # Use patool, fallback to unar on failure
- try:
- _extract_with_patool(archive_path, extract_to, progress_callback, password)
- except Exception:
- _extract_with_unar(archive_path, extract_to, progress_callback, password)
+ # Use unar directly for zipx format
+ _extract_with_unar(archive_path, extract_to, progress_callback, password)
elif archive_format == "cab":
# Prioritize using cabextract tool to extract CAB files
try:
@@ -1019,6 +1124,17 @@ def extract_archive(archive_path, extract_to, progress_callback=None, password=N
if progress_callback:
progress_callback(f"Archive extracted to: {extract_to}", 100)
+
+ # 为解压出的可执行文件添加执行权限
+ try:
+ _set_executable_permissions(extract_to, progress_callback)
+ if progress_callback:
+ progress_callback("Executable permissions set", 100)
+ except Exception as e:
+ # 权限设置失败不应该影响解压结果,只是记录警告
+ if progress_callback:
+ progress_callback(f"Warning: Failed to set executable permissions: {str(e)}", 90)
+
return True
except Exception as e:
@@ -1045,6 +1161,18 @@ def _extract_with_python(archive_path, extract_to, archive_format, progress_call
total_files = len(file_list)
for i, file_name in enumerate(file_list):
try:
+ # 清理绝对路径,防止解压到系统目录
+ if file_name.startswith('/') or file_name.startswith('./'):
+ # 将绝对路径转换为相对路径
+ file_name = os.path.relpath(file_name, '/')
+ if file_name == '.':
+ continue # 跳过当前目录引用
+
+ # 防止路径遍历攻击(../)
+ if '..' in file_name.split(os.sep):
+ # 跳过包含父目录引用的路径
+ continue
+
zipf.extract(file_name, extract_to)
except (RuntimeError, pyzipper.BadZipFile) as e:
error_msg = str(e).lower()
@@ -1095,29 +1223,27 @@ def _extract_with_python(archive_path, extract_to, archive_format, progress_call
total_files = len(file_list)
for i, file_name in enumerate(file_list):
try:
- # 确保文件直接解压到目标目录,不保留路径结构
+ # 清理绝对路径,防止解压到系统目录
+ if file_name.startswith('/') or file_name.startswith('./'):
+ # 将绝对路径转换为相对路径
+ file_name = os.path.relpath(file_name, '/')
+ if file_name == '.':
+ continue # 跳过当前目录引用
+
+ # 防止路径遍历攻击(../)
+ if '..' in file_name.split(os.sep):
+ # 跳过包含父目录引用的路径
+ continue
+
+ # 保留压缩包的目录结构
if file_name.endswith('/'):
- # 如果是目录,创建目录
- dir_name = os.path.basename(file_name.rstrip('/'))
- if dir_name: # 只有当目录名不为空时才创建
- os.makedirs(os.path.join(extract_to, dir_name), exist_ok=True)
+ # 如果是目录,创建完整路径
+ dir_path = os.path.join(extract_to, file_name.rstrip('/'))
+ if dir_path and dir_path != extract_to: # 确保不是空路径或根目录
+ os.makedirs(dir_path, exist_ok=True)
else:
- # 如果是文件,直接解压到目标目录
+ # 如果是文件,直接解压,保留路径结构
zipf.extract(file_name, extract_to)
- # 如果文件在子目录中,移动到目标目录
- if '/' in file_name:
- base_name = os.path.basename(file_name)
- src_path = os.path.join(extract_to, file_name)
- dst_path = os.path.join(extract_to, base_name)
- if os.path.exists(src_path) and src_path != dst_path:
- shutil.move(src_path, dst_path)
- # 尝试删除空的子目录
- parent_dir = os.path.dirname(src_path)
- try:
- if os.path.exists(parent_dir) and parent_dir != extract_to:
- os.rmdir(parent_dir)
- except:
- pass # 如果目录不为空,忽略错误
except (RuntimeError, zipfile.BadZipFile) as e:
error_msg = str(e).lower()
# 检查更多可能的密码错误关键词
@@ -1132,6 +1258,20 @@ def _extract_with_python(archive_path, extract_to, archive_format, progress_call
if progress_callback:
progress = ((i + 1) / total_files) * 100
progress_callback(f"Extracting {file_name}", progress)
+
+ # 检查解压后的目录中是否包含系统目录
+ if progress_callback:
+ try:
+ # 检查解压后的目录中是否包含系统目录
+ for root, dirs, files in os.walk(extract_to):
+ for dir_name in dirs:
+ # 检查是否是系统目录
+ if dir_name.lower() in ['bin', 'sbin', 'usr', 'etc', 'var', 'sys', 'proc', 'dev', 'boot', 'lib', 'lib64', 'opt', 'run', 'srv', 'tmp']:
+ full_path = os.path.join(root, dir_name)
+ progress_callback(f"警告: 检测到系统目录 {full_path}", 50)
+ except Exception as e:
+ # 忽略检查过程中的错误
+ pass
elif archive_format.startswith("tar"):
mode = "r"
@@ -1146,61 +1286,89 @@ def _extract_with_python(archive_path, extract_to, archive_format, progress_call
members = tarf.getmembers()
total_members = len(members)
for i, member in enumerate(members):
- # Fix TAR.GZ extraction path issue - keep only filename, remove path
+ # 保留TAR文件的目录结构,但清理绝对路径
if member.name.startswith('/') or member.name.startswith('./'):
- member.name = os.path.basename(member.name)
- elif os.path.dirname(member.name): # If it contains a path
- member.name = os.path.basename(member.name)
+ # 将绝对路径转换为相对路径,但保留目录结构
+ member.name = os.path.relpath(member.name, '/')
+ if member.name == '.':
+ member.name = os.path.basename(member.name)
tarf.extract(member, extract_to)
if progress_callback:
progress = ((i + 1) / total_members) * 100
progress_callback(f"Extracting {member.name}", progress)
+
+ # 检查解压后的目录中是否包含系统目录
+ if progress_callback:
+ try:
+ # 检查解压后的目录中是否包含系统目录
+ for root, dirs, files in os.walk(extract_to):
+ for dir_name in dirs:
+ # 检查是否是系统目录
+ if dir_name.lower() in ['bin', 'sbin', 'usr', 'etc', 'var', 'sys', 'proc', 'dev', 'boot', 'lib', 'lib64', 'opt', 'run', 'srv', 'tmp']:
+ full_path = os.path.join(root, dir_name)
+ progress_callback(f"警告: 检测到系统目录 {full_path}", 50)
+ except Exception as e:
+ # 忽略检查过程中的错误
+ pass
elif archive_format in ["bz2", "xz", "lzma"]:
# Single file compression format
import bz2
import lzma
+ import tarfile
- output_file = os.path.join(extract_to, os.path.basename(archive_path).rsplit('.', 1)[0])
+ # First, decompress the file to a temporary tar file
+ temp_tar = os.path.join(extract_to, os.path.basename(archive_path).rsplit('.', 1)[0] + '.tar')
- if archive_format == "bz2":
- with bz2.open(archive_path, 'rb') as f_in:
- with open(output_file, 'wb') as f_out:
- # Use binary mode for reading and writing
- while True:
- chunk = f_in.read(8192)
- if not chunk:
- break
- f_out.write(chunk)
- else:
- with lzma.open(archive_path, 'rb') as f_in:
- with open(output_file, 'wb') as f_out:
- # Use binary mode for reading and writing
- while True:
- chunk = f_in.read(8192)
- if not chunk:
- break
- f_out.write(chunk)
+ try:
+ if archive_format == "bz2":
+ with bz2.open(archive_path, 'rb') as f_in:
+ with open(temp_tar, 'wb') as f_out:
+ # Use binary mode for reading and writing
+ while True:
+ chunk = f_in.read(8192)
+ if not chunk:
+ break
+ f_out.write(chunk)
+ else:
+ with lzma.open(archive_path, 'rb') as f_in:
+ with open(temp_tar, 'wb') as f_out:
+ # Use binary mode for reading and writing
+ while True:
+ chunk = f_in.read(8192)
+ if not chunk:
+ break
+ f_out.write(chunk)
+
+ # Now extract the tar file
+ with tarfile.open(temp_tar, 'r') as tarf:
+ members = tarf.getmembers()
+ total_members = len(members)
+ for i, member in enumerate(members):
+ # 保留TAR文件的目录结构,但清理绝对路径
+ if member.name.startswith('/') or member.name.startswith('./'):
+ # 将绝对路径转换为相对路径,但保留目录结构
+ member.name = os.path.relpath(member.name, '/')
+ if member.name == '.':
+ member.name = os.path.basename(member.name)
+ tarf.extract(member, extract_to)
+ if progress_callback:
+ progress = ((i + 1) / total_members) * 100
+ progress_callback(f"Extracting {member.name}", progress)
+
+ # Clean up the temporary tar file
+ os.remove(temp_tar)
+
+ except Exception as e:
+ # Clean up the temporary tar file if it exists
+ if os.path.exists(temp_tar):
+ os.remove(temp_tar)
+ raise e
if progress_callback:
- progress_callback(f"Extracted {output_file}", 100)
+ progress_callback(f"Extracted {archive_format} archive", 100)
+
-def _extract_with_patool(archive_path, extract_to, progress_callback=None, password=None):
- """Extract using patool"""
- try:
- import patoolib
- except ImportError:
- raise ImportError("patool is required for this format")
-
- # patoolib doesn't directly support password, so we'll use unar for password-protected archives
- if password:
- _extract_with_unar(archive_path, extract_to, progress_callback, password)
- return
-
- patoolib.extract_archive(archive_path, outdir=extract_to)
-
- if progress_callback:
- progress_callback("Archive extracted", 100)
def _extract_rar_with_cli(archive_path, extract_to, progress_callback=None, password=None):
"""Extract RAR file using unrar CLI tool"""
@@ -1215,7 +1383,7 @@ def _extract_rar_with_cli(archive_path, extract_to, progress_callback=None, pass
if password == "":
raise RuntimeError("Empty password not allowed for encrypted RAR file")
- cmd = [unrar_tool, "x", archive_path, extract_to + "/", "-ep"] # Add -ep parameter to ignore paths
+ cmd = [unrar_tool, "x", archive_path, extract_to + "/"] # 保留目录结构
if password:
cmd.extend(["-p" + password, "-y"]) # Add password and assume yes for all queries
else:
@@ -1261,6 +1429,17 @@ def _extract_rar_with_cli(archive_path, extract_to, progress_callback=None, pass
if not extracted_files:
raise RuntimeError("Password required for encrypted RAR file")
+
+ # 检查并警告潜在的绝对路径解压问题
+ if os.path.exists(extract_to):
+ # 检查解压目录中是否包含系统目录
+ for item in os.listdir(extract_to):
+ item_path = os.path.join(extract_to, item)
+ if os.path.isdir(item_path):
+ # 检查是否是常见的系统目录
+ if item in ['var', 'etc', 'usr', 'bin', 'sbin', 'lib', 'lib64', 'sys', 'proc', 'dev']:
+ if progress_callback:
+ progress_callback(f"Warning: System directory '{item}' was extracted. This may indicate absolute paths in the archive.", -1)
def _extract_7z_with_cli(archive_path, extract_to, progress_callback=None, password=None):
"""Extract 7z file using 7zz CLI tool"""
@@ -1319,27 +1498,16 @@ def _extract_7z_with_cli(archive_path, extract_to, progress_callback=None, passw
raise RuntimeError("Password required for encrypted 7Z file")
raise RuntimeError(f"7z extraction failed: {result.stderr}")
- # 检查是否有额外的子目录,如果有,将文件移动到根目录
+ # 检查并警告潜在的绝对路径解压问题
if os.path.exists(extract_to):
- items = os.listdir(extract_to)
- # 如果只有一个子目录,且该子目录包含所有文件
- if len(items) == 1 and os.path.isdir(os.path.join(extract_to, items[0])):
- sub_dir = os.path.join(extract_to, items[0])
- # 移动子目录中的所有内容到根目录
- for item in os.listdir(sub_dir):
- src = os.path.join(sub_dir, item)
- dst = os.path.join(extract_to, item)
- if os.path.exists(dst):
- if os.path.isdir(dst):
- shutil.rmtree(dst)
- else:
- os.remove(dst)
- shutil.move(src, dst)
- # 尝试删除空子目录
- try:
- os.rmdir(sub_dir)
- except:
- pass
+ # 检查解压目录中是否包含系统目录
+ for item in os.listdir(extract_to):
+ item_path = os.path.join(extract_to, item)
+ if os.path.isdir(item_path):
+ # 检查是否是常见的系统目录
+ if item in ['var', 'etc', 'usr', 'bin', 'sbin', 'lib', 'lib64', 'sys', 'proc', 'dev']:
+ if progress_callback:
+ progress_callback(f"Warning: System directory '{item}' was extracted. This may indicate absolute paths in the archive.", -1)
return True
@@ -1393,6 +1561,17 @@ def _extract_iso_with_system(archive_path, extract_to, progress_callback=None):
if result.returncode != 0:
raise RuntimeError(f"Failed to copy files from ISO: {result.stderr}")
+ # 检查并警告潜在的绝对路径解压问题
+ if os.path.exists(extract_to):
+ # 检查解压目录中是否包含系统目录
+ for item in os.listdir(extract_to):
+ item_path = os.path.join(extract_to, item)
+ if os.path.isdir(item_path):
+ # 检查是否是常见的系统目录
+ if item in ['var', 'etc', 'usr', 'bin', 'sbin', 'lib', 'lib64', 'sys', 'proc', 'dev']:
+ if progress_callback:
+ progress_callback(f"Warning: System directory '{item}' was extracted. This may indicate absolute paths in the archive.", -1)
+
if progress_callback:
progress_callback("ISO extracted successfully", 100)
@@ -1411,6 +1590,17 @@ def _extract_iso_with_system(archive_path, extract_to, progress_callback=None):
if result.returncode != 0:
raise RuntimeError(f"7z extraction failed: {result.stderr}")
+ # 检查并警告潜在的绝对路径解压问题
+ if os.path.exists(extract_to):
+ # 检查解压目录中是否包含系统目录
+ for item in os.listdir(extract_to):
+ item_path = os.path.join(extract_to, item)
+ if os.path.isdir(item_path):
+ # 检查是否是常见的系统目录
+ if item in ['var', 'etc', 'usr', 'bin', 'sbin', 'lib', 'lib64', 'sys', 'proc', 'dev']:
+ if progress_callback:
+ progress_callback(f"Warning: System directory '{item}' was extracted. This may indicate absolute paths in the archive.", -1)
+
if progress_callback:
progress_callback("ISO extracted with 7z", 100)
@@ -1430,6 +1620,17 @@ def _extract_iso_with_system(archive_path, extract_to, progress_callback=None):
if result.returncode != 0:
raise RuntimeError(f"ISO extraction failed: {result.stderr}")
+ # 检查并警告潜在的绝对路径解压问题
+ if os.path.exists(extract_to):
+ # 检查解压目录中是否包含系统目录
+ for item in os.listdir(extract_to):
+ item_path = os.path.join(extract_to, item)
+ if os.path.isdir(item_path):
+ # 检查是否是常见的系统目录
+ if item in ['var', 'etc', 'usr', 'bin', 'sbin', 'lib', 'lib64', 'sys', 'proc', 'dev']:
+ if progress_callback:
+ progress_callback(f"Warning: System directory '{item}' was extracted. This may indicate absolute paths in the archive.", -1)
+
if progress_callback:
progress_callback("ISO extracted successfully", 100)
@@ -1457,6 +1658,17 @@ def _extract_cab_with_cabextract(archive_path, extract_to, progress_callback=Non
if result.returncode != 0:
raise RuntimeError(f"CAB extraction failed: {result.stderr}")
+
+ # 检查并警告潜在的绝对路径解压问题
+ if os.path.exists(extract_to):
+ # 检查解压目录中是否包含系统目录
+ for item in os.listdir(extract_to):
+ item_path = os.path.join(extract_to, item)
+ if os.path.isdir(item_path):
+ # 检查是否是常见的系统目录
+ if item in ['var', 'etc', 'usr', 'bin', 'sbin', 'lib', 'lib64', 'sys', 'proc', 'dev']:
+ if progress_callback:
+ progress_callback(f"Warning: System directory '{item}' was extracted. This may indicate absolute paths in the archive.", -1)
def _extract_with_unar(archive_path, extract_to, progress_callback=None, password=None):
"""Use unar as ultimate fallback"""
@@ -1491,6 +1703,19 @@ def _extract_with_unar(archive_path, extract_to, progress_callback=None, passwor
raise RuntimeError("Password required for encrypted archive")
raise RuntimeError(f"Extraction failed: {result.stderr}")
+ # 即使unar命令成功执行,也需要检查解压结果,防止创建系统目录
+ # 检查是否创建了var目录或其他系统目录
+ system_dirs = ['var', 'etc', 'usr', 'bin', 'sbin', 'lib', 'tmp', 'dev', 'proc', 'sys']
+ for root, dirs, files in os.walk(extract_to):
+ for dir_name in dirs:
+ if dir_name in system_dirs:
+ # 如果发现系统目录,需要检查是否是绝对路径导致的问题
+ full_path = os.path.join(root, dir_name)
+ rel_path = os.path.relpath(full_path, extract_to)
+ # 如果是直接在解压根目录下创建的系统目录,可能是绝对路径问题
+ if os.path.dirname(rel_path) == '' and dir_name in ['var', 'tmp']:
+ print(f"Warning: Detected potential system directory extraction: {rel_path}")
+
# 即使命令成功执行,也需要检查是否真的解压了文件
if not password:
# 检查解压目录是否为空
@@ -2199,6 +2424,155 @@ def _add_7z_with_cli(archive_path, file_to_add_path, progress_callback=None):
raise RuntimeError(f"7z add failed: {result.stderr}")
# Test function
+def batch_extract_archives(archive_paths, extract_to_base, progress_callback=None, password=None,
+ overwrite_existing=False, create_subfolders=True, error_callback=None,
+ password_callback=None, password_detector=None):
+ """
+ Extract multiple archive files in batch.
+
+ Args:
+ archive_paths (list): List of paths to archive files to extract.
+ extract_to_base (str): Base directory to extract all archives to.
+ progress_callback (function): Optional callback for overall progress updates (filename, progress).
+ password (str): Optional password for encrypted archives.
+ overwrite_existing (bool): Whether to overwrite existing files.
+ create_subfolders (bool): Whether to create subfolders for each archive.
+ error_callback (function): Optional callback for individual file error updates (filename, error).
+ password_callback (function): Optional callback to request password from user (archive_path, format, is_protected).
+ password_detector (PasswordDetector): Optional password detector instance for checking protected archives.
+
+ Returns:
+ dict: Results dictionary with 'success_count', 'error_count', 'results' list.
+ """
+ results = {
+ 'success_count': 0,
+ 'error_count': 0,
+ 'results': []
+ }
+
+ # 初始化密码检测器
+ if password_detector is None:
+ try:
+ from password_detector import PasswordDetector
+ password_detector = PasswordDetector()
+ except ImportError:
+ password_detector = None
+
+ # 缓存密码避免重复询问
+ password_cache = {}
+
+ total_archives = len(archive_paths)
+
+ for i, archive_path in enumerate(archive_paths):
+ try:
+ # Calculate overall progress
+ if progress_callback:
+ overall_progress = (i / total_archives) * 100
+ progress_callback(f"Processing {i+1}/{total_archives}: {os.path.basename(archive_path)}", overall_progress)
+
+ # Determine extraction directory for this archive
+ if create_subfolders:
+ # Create subfolder named after archive (without extension)
+ archive_name = os.path.splitext(os.path.basename(archive_path))[0]
+ archive_extract_to = os.path.join(extract_to_base, archive_name)
+ else:
+ # Extract directly to base directory
+ archive_extract_to = extract_to_base
+
+ os.makedirs(archive_extract_to, exist_ok=True)
+
+ # 检测密码保护状态
+ current_password = password
+ if password_detector and password_callback:
+ try:
+ detection_result = password_detector.is_password_protected(archive_path)
+ if detection_result['is_protected']:
+ archive_format = detection_result.get('format', 'unknown')
+
+ # 检查是否已有该格式的密码
+ if archive_format not in password_cache:
+ # 询问用户输入密码
+ if progress_callback:
+ progress_callback(f"检测到密码保护的 {archive_format.upper()} 文件: {os.path.basename(archive_path)}",
+ (i / total_archives) * 100)
+
+ requested_password = password_callback(archive_path, archive_format, True)
+ if requested_password:
+ password_cache[archive_format] = requested_password
+ current_password = requested_password
+ else:
+ # 用户取消操作
+ results['error_count'] += 1
+ error_msg = "用户取消密码输入"
+ if error_callback:
+ error_callback(archive_path, error_msg)
+ results['results'].append({
+ 'archive_path': archive_path,
+ 'extract_to': archive_extract_to,
+ 'status': 'cancelled',
+ 'error': error_msg
+ })
+ continue
+ else:
+ current_password = password_cache[archive_format]
+ elif detection_result.get('error'):
+ # 检测出错,但仍然尝试正常解压
+ if error_callback:
+ error_callback(archive_path, f"密码检测失败: {detection_result['error']}")
+ except Exception as e:
+ # 密码检测失败,继续尝试正常解压
+ if error_callback:
+ error_callback(archive_path, f"密码检测出错: {str(e)}")
+
+ # Create individual progress callback for this archive
+ def archive_progress(message, percent):
+ if progress_callback:
+ # Combine overall progress with archive-specific progress
+ archive_progress_value = (i + percent/100) / total_archives * 100
+ progress_callback(f"{os.path.basename(archive_path)}: {message}", archive_progress_value)
+
+ # Extract the archive
+ success = extract_archive(archive_path, archive_extract_to,
+ progress_callback=archive_progress, password=current_password)
+
+ if success:
+ results['success_count'] += 1
+ results['results'].append({
+ 'archive_path': archive_path,
+ 'extract_to': archive_extract_to,
+ 'status': 'success',
+ 'error': None
+ })
+ else:
+ results['error_count'] += 1
+ error_msg = f"Extraction failed"
+ if error_callback:
+ error_callback(archive_path, error_msg)
+ results['results'].append({
+ 'archive_path': archive_path,
+ 'extract_to': archive_extract_to,
+ 'status': 'error',
+ 'error': error_msg
+ })
+
+ except Exception as e:
+ results['error_count'] += 1
+ error_msg = str(e)
+ if error_callback:
+ error_callback(archive_path, error_msg)
+ results['results'].append({
+ 'archive_path': archive_path,
+ 'extract_to': archive_extract_to if 'archive_extract_to' in locals() else extract_to_base,
+ 'status': 'error',
+ 'error': error_msg
+ })
+
+ # Final progress update
+ if progress_callback:
+ progress_callback(results['success_count'], total_archives, f"Batch extraction complete: {results['success_count']}/{total_archives} successful")
+
+ return results
+
def test_archive_functions():
"""Test archive functions"""
import tempfile
@@ -2282,5 +2656,80 @@ def test_archive_functions():
print("\n=== Archive Functions Test Complete ===")
+def _set_executable_permissions(extract_to, progress_callback=None):
+ """
+ 为解压出的可执行文件添加执行权限
+
+ Args:
+ extract_to (str): 解压目标目录
+ progress_callback (function): 可选的进度回调函数
+ """
+ import stat
+ import re
+
+ # 定义可执行文件的常见扩展名
+ executable_extensions = {
+ '.sh', '.bash', '.zsh', '.fish', '.cmd', '.bat', # Shell脚本
+ '.py', '.pl', '.rb', '.php', '.js', '.ts', # 脚本语言
+ '.exe', '.app', '.bin', '.com', # 可执行文件
+ '.run', '.install', '.setup' # 安装程序
+ }
+
+ # 需要检查shebang的文件扩展名
+ script_extensions = {'.sh', '.bash', '.zsh', '.fish', '.py', '.pl', '.rb', '.php', '.js', '.ts'}
+
+ executable_files = []
+
+ def check_file_executable(file_path):
+ """检查文件是否可能是可执行文件"""
+ # 检查文件扩展名
+ _, ext = os.path.splitext(file_path.lower())
+ if ext in executable_extensions:
+ return True
+
+ # 对于没有后缀名的文件,直接返回True
+ if not ext:
+ return True
+
+ # 检查shebang行(对于脚本文件)
+ if ext in script_extensions:
+ try:
+ with open(file_path, 'rb') as f:
+ first_line = f.readline(100) # 读取前100字节
+ if first_line.startswith(b'#!'):
+ return True
+ except (IOError, UnicodeDecodeError):
+ # 如果无法读取文件,跳过
+ pass
+
+ return False
+
+ # 遍历解压目录中的所有文件
+ for root, dirs, files in os.walk(extract_to):
+ for file_name in files:
+ file_path = os.path.join(root, file_name)
+
+ # 跳过隐藏文件和系统文件
+ if file_name.startswith('.') or file_name in ['.DS_Store', 'Thumbs.db']:
+ continue
+
+ try:
+ if check_file_executable(file_path):
+ # 添加执行权限 (755: rwxr-xr-x)
+ current_stat = os.stat(file_path)
+ os.chmod(file_path, current_stat.st_mode | stat.S_IRWXU | stat.S_IRGRP | stat.S_IXGRP | stat.S_IROTH | stat.S_IXOTH)
+ executable_files.append(file_path)
+
+ if progress_callback:
+ progress_callback(f"Set executable permission: {file_name}", -1) # -1表示不更新总体进度
+
+ except (OSError, IOError) as e:
+ # 权限设置失败,记录但不影响解压
+ if progress_callback:
+ progress_callback(f"Warning: Failed to set permission for {file_name}: {str(e)}", -1)
+
+ if progress_callback and executable_files:
+ progress_callback(f"Set executable permissions for {len(executable_files)} files", -1)
+
if __name__ == "__main__":
test_archive_functions()
\ No newline at end of file
diff --git a/support/convert.py b/support/convert.py
index 437d580..713092d 100644
--- a/support/convert.py
+++ b/support/convert.py
@@ -45,27 +45,46 @@ def convert_image(input_path, output_path, output_format, min_size=16, max_size=
quality (int): Image quality for lossy formats like JPG (default: 85, range: 1-100).
progress_callback (function): Callback function to report progress.
interface_settings (dict): Interface behavior settings for controlling conversion behavior.
+
+ Returns:
+ tuple: (success, message) where success is a boolean and message is a string
"""
- if output_format not in SUPPORTED_FORMATS:
- raise ValueError(f"Unsupported output format: {output_format}. Supported formats are: {', '.join(SUPPORTED_FORMATS)}")
+ try:
+ if output_format not in SUPPORTED_FORMATS:
+ error_msg = f"Unsupported output format: {output_format}. Supported formats are: {', '.join(SUPPORTED_FORMATS)}"
+ if progress_callback:
+ progress_callback(error_msg, 0)
+ return False, error_msg
- if output_format == "icns":
- # Existing ICNS conversion logic
- _create_icns_internal(input_path, output_path, min_size, max_size, progress_callback)
- else:
- # Generic image conversion using Pillow
- try:
- img = Image.open(input_path)
-
- # For JPG, ensure the image is in RGB mode as JPG does not support alpha channel
- if output_format.lower() == "jpg":
- if progress_callback:
- progress_callback(f"Processing image for JPG conversion...", 30)
+ if not os.path.exists(input_path):
+ error_msg = f"Input file not found: {input_path}"
+ if progress_callback:
+ progress_callback(error_msg, 0)
+ return False, error_msg
+
+ if output_format == "icns":
+ # Existing ICNS conversion logic
+ _create_icns_internal(input_path, output_path, min_size, max_size, progress_callback)
+ else:
+ # Generic image conversion using Pillow
+ img = None
+ try:
+ img = Image.open(input_path)
+
+ # Create output directory if it doesn't exist
+ output_dir = os.path.dirname(output_path)
+ if output_dir and not os.path.exists(output_dir):
+ os.makedirs(output_dir, exist_ok=True)
- # Handle various image modes that need conversion to RGB
- if img.mode in ('RGBA', 'LA', 'P', 'CMYK', 'YCbCr', 'LAB', 'HSV', 'I', 'F'):
+ # For JPG, ensure the image is in RGB mode as JPG does not support alpha channel
+ if output_format.lower() == "jpg":
if progress_callback:
- progress_callback(f"Converting from {img.mode} mode to RGB...", 40)
+ progress_callback(f"Processing image for JPG conversion...", 30)
+
+ # Handle various image modes that need conversion to RGB
+ if img.mode in ('RGBA', 'LA', 'P', 'CMYK', 'YCbCr', 'LAB', 'HSV', 'I', 'F'):
+ if progress_callback:
+ progress_callback(f"Converting from {img.mode} mode to RGB...", 40)
# For modes with transparency, use white background
if img.mode in ('RGBA', 'LA', 'P'):
@@ -78,237 +97,299 @@ def convert_image(input_path, output_path, output_format, min_size=16, max_size=
else:
# For other modes, convert directly to RGB
img = img.convert('RGB')
+
+ if progress_callback:
+ progress_callback(f"Image converted to RGB mode", 50)
- if progress_callback:
- progress_callback(f"Image converted to RGB mode", 50)
-
- # Save with appropriate options for each format
- save_options = {}
-
- if output_format.lower() == "jpg":
- # Save JPG with specified quality and optimized settings
- img.save(output_path, format='JPEG', quality=quality, optimize=True, progressive=True)
- elif output_format.lower() == "webp":
- # Save WebP with quality settings
- save_options = {'quality': quality, 'method': 6}
- img.save(output_path, format='WEBP', **save_options)
- elif output_format.lower() == "tiff":
- # Save TIFF with compression
- save_options = {'compression': 'tiff_lzw'}
- img.save(output_path, format='TIFF', **save_options)
- elif output_format.lower() in ["svg", "pdf", "eps"]:
- # Vector formats require special handling
- # For now, convert raster image to these formats with basic settings
- if progress_callback:
- progress_callback(f"Converting to vector format {output_format.upper()}...", 70)
- img.save(output_path, format=output_format.upper())
- elif output_format.lower() in ["heic", "heif"]:
- # HEIC/HEIF format support
- if progress_callback:
- progress_callback(f"Converting to {output_format.upper()} format...", 70)
- # Try to save as HEIF if available, otherwise fallback
- try:
- img.save(output_path, format='HEIF', quality=quality)
- except Exception:
- # Fallback to PNG if HEIF not supported
+ # Save with appropriate options for each format
+ save_options = {}
+
+ if output_format.lower() == "jpg":
+ # Save JPG with specified quality and optimized settings
+ img.save(output_path, format='JPEG', quality=quality, optimize=True, progressive=True)
+ elif output_format.lower() == "webp":
+ # Save WebP with quality settings
+ save_options = {'quality': quality, 'method': 6}
+ img.save(output_path, format='WEBP', **save_options)
+ elif output_format.lower() == "tiff":
+ # Save TIFF with compression
+ save_options = {'compression': 'tiff_lzw'}
+ img.save(output_path, format='TIFF', **save_options)
+ elif output_format.lower() in ["svg", "pdf", "eps"]:
+ # Vector formats require special handling
+ # For now, convert raster image to these formats with basic settings
if progress_callback:
- progress_callback(f"HEIF format not available, falling back to PNG", 80)
- img.save(output_path, format='PNG')
- elif output_format.lower() in ["avif", "jxl"]:
- # Modern formats that may require additional libraries
- if progress_callback:
- progress_callback(f"Converting to {output_format.upper()} format...", 70)
- try:
- img.save(output_path, format=output_format.upper(), quality=quality)
- except Exception as format_error:
- # Fallback to WebP if modern format not supported
+ progress_callback(f"Converting to vector format {output_format.upper()}...", 70)
+ img.save(output_path, format=output_format.upper())
+ elif output_format.lower() in ["heic", "heif"]:
+ # HEIC/HEIF format support
if progress_callback:
- progress_callback(f"{output_format.upper()} format not available, falling back to WebP", 80)
- img.save(output_path, format='WEBP', quality=quality)
- elif output_format.lower() in ["dds", "exr"]:
- # Specialized formats for gaming and HDR
- if progress_callback:
- progress_callback(f"Converting to {output_format.upper()} format...", 70)
- img.save(output_path, format=output_format.upper())
- else:
- # Default handling for other formats
- img.save(output_path, format=output_format.upper())
+ progress_callback(70, 100, f"Converting to {output_format.upper()} format...")
+ # Try to save as HEIF if available, otherwise fallback
+ try:
+ img.save(output_path, format='HEIF', quality=quality)
+ except Exception:
+ # Fallback to PNG if HEIF not supported
+ if progress_callback:
+ progress_callback(f"HEIF format not available, falling back to PNG", 80)
+ img.save(output_path, format='PNG')
+ elif output_format.lower() in ["avif", "jxl"]:
+ # Modern formats that may require additional libraries
+ if progress_callback:
+ progress_callback(f"Converting to {output_format.upper()} format...", 70)
+ try:
+ img.save(output_path, format=output_format.upper(), quality=quality)
+ except Exception as format_error:
+ # Fallback to WebP if modern format not supported
+ if progress_callback:
+ progress_callback(f"{output_format.upper()} format not available, falling back to WebP", 80)
+ img.save(output_path, format='WEBP', quality=quality)
+ elif output_format.lower() in ["dds", "exr"]:
+ # Specialized formats for gaming and HDR
+ if progress_callback:
+ progress_callback(f"Converting to {output_format.upper()} format...", 70)
+ img.save(output_path, format=output_format.upper())
+ else:
+ # Default handling for other formats
+ img.save(output_path, format=output_format.upper())
+ finally:
+ # Ensure image resources are released
+ if img is not None:
+ img.close()
+ del img
if progress_callback:
- progress_callback(f"Successfully converted {input_path} to {output_path} ({output_format})", 100)
+ progress_callback(100, 100, f"Successfully converted {input_path} to {output_path} ({output_format})")
else:
print(f"Successfully converted {input_path} to {output_path} ({output_format})")
- except Exception as e:
- error_msg = f"Error converting image to {output_format.upper()}: {e}"
- if progress_callback:
- progress_callback(error_msg, 0)
- raise Exception(error_msg) from e
+
+ return True, f"Successfully converted {os.path.basename(input_path)} to {output_format.upper()}"
+ except Exception as e:
+ error_msg = f"Error converting image {os.path.basename(input_path)}: {str(e)}"
+ if progress_callback:
+ progress_callback(error_msg, 0)
+ return False, error_msg
def _create_icns_internal(png_path, icns_path, min_size=16, max_size=None, progress_callback=None):
"""
Internal function to convert a PNG image to ICNS format using iconset method.
This function contains the original logic of create_icns.
"""
- # Open the source image
- img = Image.open(png_path)
-
- # Automatically detect image size if not provided
- if max_size is None:
- max_size = min(img.width, img.height)
- if progress_callback:
- progress_callback(f"Auto-detected maximum size: {max_size}", 5)
- else:
- print(f"Auto-detected maximum size: {max_size}")
-
- if progress_callback:
- progress_callback(f"Source image size: {img.width}x{img.height}", 10)
- else:
- print(f"Source image size: {img.width}x{img.height}")
-
- # Ensure the image is square
- if img.width != img.height:
+ img = None
+ try:
+ # Open the source image
+ img = Image.open(png_path)
+
+ # Automatically detect image size if not provided
+ if max_size is None:
+ max_size = min(img.width, img.height)
+ if progress_callback:
+ progress_callback(5, 100, f"Auto-detected maximum size: {max_size}")
+ else:
+ print(f"Auto-detected maximum size: {max_size}")
+
if progress_callback:
- progress_callback("Warning: Image is not square. Cropping to square.", 15)
+ progress_callback(10, 100, f"Source image size: {img.width}x{img.height}")
else:
- print("Warning: Image is not square. Cropping to square.")
- min_dimension = min(img.width, img.height)
- left = (img.width - min_dimension) // 2
- top = (img.height - min_dimension) // 2
- right = left + min_dimension
- bottom = top + min_dimension
- img = img.crop((left, top, right, bottom))
-
- # Create a temporary directory for iconset
- with tempfile.TemporaryDirectory() as tmp_dir:
- iconset_dir = os.path.join(tmp_dir, "iconset.iconset")
- os.makedirs(iconset_dir)
+ print(f"Source image size: {img.width}x{img.height}")
- # Define standard sizes for ICNS (based on Apple's specifications)
- standard_sizes = [16, 32, 64, 128, 256, 512, 1024]
-
- # Generate icons for standard sizes within our range
- generated_sizes = []
- total_steps = len(standard_sizes) * 2 # Approximate steps
- current_step = 0
+ # Ensure the image is square
+ if img.width != img.height:
+ if progress_callback:
+ progress_callback(15, 100, "Warning: Image is not square. Cropping to square.")
+ else:
+ print("Warning: Image is not square. Cropping to square.")
+ min_dimension = min(img.width, img.height)
+ left = (img.width - min_dimension) // 2
+ top = (img.height - min_dimension) // 2
+ right = left + min_dimension
+ bottom = top + min_dimension
+ img = img.crop((left, top, right, bottom))
- for size in standard_sizes:
- if min_size <= size <= max_size:
- current_step += 1
- if progress_callback:
- progress_callback(f"Generating size: {size}x{size}", 20 + int(60 * current_step / total_steps))
- else:
- print(f"Generated size: {size}x{size}")
-
- # Create a copy of the image and resize it
- resized_img = img.copy().resize((size, size), Image.Resampling.LANCZOS)
-
- # Save as PNG in iconset directory
- filename = f"icon_{size}x{size}.png"
- resized_img.save(os.path.join(iconset_dir, filename), "PNG")
- generated_sizes.append(size)
-
- # For specific sizes, also generate retina versions
- retina_pairs = {16: 32, 32: 64, 128: 256, 256: 512, 512: 1024}
- if size in retina_pairs and retina_pairs[size] <= max_size:
+ # Create a temporary directory for iconset
+ with tempfile.TemporaryDirectory() as tmp_dir:
+ iconset_dir = os.path.join(tmp_dir, "iconset.iconset")
+ os.makedirs(iconset_dir)
+
+ # Define standard sizes for ICNS (based on Apple's specifications)
+ standard_sizes = [16, 32, 64, 128, 256, 512, 1024]
+
+ # Generate icons for standard sizes within our range
+ generated_sizes = []
+ total_steps = len(standard_sizes) * 2 # Approximate steps
+ current_step = 0
+
+ for size in standard_sizes:
+ if min_size <= size <= max_size:
current_step += 1
- retina_size = retina_pairs[size]
if progress_callback:
- progress_callback(f"Generating retina size: {retina_size}x{retina_size}", 20 + int(60 * current_step / total_steps))
+ progress_callback(20 + int(60 * current_step / total_steps), 100, f"Generating size: {size}x{size}")
else:
- print(f"Generated retina size: {retina_size}x{retina_size}")
+ print(f"Generated size: {size}x{size}")
- retina_img = img.copy().resize((retina_size, retina_size), Image.Resampling.LANCZOS)
- retina_filename = f"icon_{size}x{size}@2x.png"
- retina_img.save(os.path.join(iconset_dir, retina_filename), "PNG")
- generated_sizes.append(retina_size)
-
- # Make sure we include max_size if it's not already included
- if max_size not in generated_sizes:
- if progress_callback:
- progress_callback(f"Generating max size: {max_size}x{max_size}", 85)
- else:
- print(f"Generated max size: {max_size}x{max_size}")
+ # Create a copy of the image and resize it
+ resized_img = img.copy().resize((size, size), Image.Resampling.LANCZOS)
+
+ # Save as PNG in iconset directory
+ filename = f"icon_{size}x{size}.png"
+ resized_img.save(os.path.join(iconset_dir, filename), "PNG")
+ generated_sizes.append(size)
+
+ # For specific sizes, also generate retina versions
+ retina_pairs = {16: 32, 32: 64, 128: 256, 256: 512, 512: 1024}
+ if size in retina_pairs and retina_pairs[size] <= max_size:
+ current_step += 1
+ retina_size = retina_pairs[size]
+ if progress_callback:
+ progress_callback(20 + int(60 * current_step / total_steps), 100, f"Generating retina size: {retina_size}x{retina_size}")
+ else:
+ print(f"Generated retina size: {retina_size}x{retina_size}")
+
+ retina_img = img.copy().resize((retina_size, retina_size), Image.Resampling.LANCZOS)
+ retina_filename = f"icon_{size}x{size}@2x.png"
+ retina_img.save(os.path.join(iconset_dir, retina_filename), "PNG")
+ generated_sizes.append(retina_size)
- resized_img = img.copy().resize((max_size, max_size), Image.Resampling.LANCZOS)
- filename = f"icon_{max_size}x{max_size}.png"
- resized_img.save(os.path.join(iconset_dir, filename), "PNG")
- generated_sizes.append(max_size)
-
- # Use iconutil to create ICNS file (macOS only)
- if progress_callback:
- progress_callback("Creating ICNS file with iconutil...", 90)
- else:
- print("Creating ICNS file with iconutil...")
+ # Make sure we include max_size if it's not already included
+ if max_size not in generated_sizes:
+ if progress_callback:
+ progress_callback(85, 100, f"Generating max size: {max_size}x{max_size}")
+ else:
+ print(f"Generated max size: {max_size}x{max_size}")
+
+ resized_img = img.copy().resize((max_size, max_size), Image.Resampling.LANCZOS)
+ filename = f"icon_{max_size}x{max_size}.png"
+ resized_img.save(os.path.join(iconset_dir, filename), "PNG")
+ generated_sizes.append(max_size)
- try:
- subprocess.run(["iconutil", "-c", "icns", iconset_dir, "-o", icns_path], check=True)
+ # Use iconutil to create ICNS file (macOS only)
if progress_callback:
- progress_callback(f"Successfully converted {png_path} to {icns_path}", 100)
+ progress_callback(90, 100, "Creating ICNS file with iconutil...")
else:
- print(f"Successfully converted {png_path} to {icns_path}")
- print(f"Generated sizes: {sorted(set(generated_sizes))}")
- except subprocess.CalledProcessError as e:
- error_detail = e.stderr.decode() if e.stderr else str(e)
- error_msg = f"Error creating ICNS with iconutil: {error_detail}"
+ print("Creating ICNS file with iconutil...")
+
+ try:
+ subprocess.run(["iconutil", "-c", "icns", iconset_dir, "-o", icns_path], check=True, capture_output=True, text=True)
+ if progress_callback:
+ progress_callback(100, 100, f"Successfully converted {png_path} to {icns_path}")
+ else:
+ print(f"Successfully converted {png_path} to {icns_path}")
+ print(f"Generated sizes: {sorted(set(generated_sizes))}")
+ except subprocess.CalledProcessError as e:
+ error_detail = e.stderr if e.stderr else str(e)
+ error_msg = f"Error creating ICNS with iconutil: {error_detail}"
+ if progress_callback:
+ progress_callback(90, 100, error_msg)
+ progress_callback(90, 100, "Falling back to Pillow method...")
+ else:
+ print(error_msg)
+ print("Falling back to Pillow method...")
+ # Fallback to Pillow method if iconutil fails
+ _fallback_method_internal(iconset_dir, icns_path, progress_callback)
+ except Exception as e:
+ error_msg = f"Unexpected error during iconutil conversion: {e}"
+ if progress_callback:
+ progress_callback(90, 100, error_msg)
+ progress_callback(90, 100, "Falling back to Pillow method...")
+ else:
+ print(error_msg)
+ print("Falling back to Pillow method...")
+ _fallback_method_internal(iconset_dir, icns_path, progress_callback)
+ finally:
+ # Ensure image resources are released
+ if img is not None:
+ img.close()
+ del img
+
+def _fallback_method_internal(iconset_dir, icns_path, progress_callback=None):
+ """
+ Internal fallback method using Pillow if iconutil is not available
+ """
+ try:
+ icon_files = os.listdir(iconset_dir)
+ if not icon_files:
+ error_msg = "No icons generated, cannot create ICNS file"
if progress_callback:
- progress_callback(error_msg, 90)
- progress_callback("Falling back to Pillow method...", 90)
+ progress_callback(100, 100, error_msg)
else:
print(error_msg)
- print("Falling back to Pillow method...")
- # Fallback to Pillow method if iconutil fails
- _fallback_method_internal(iconset_dir, icns_path, progress_callback)
- except Exception as e:
- error_msg = f"Unexpected error during iconutil conversion: {e}"
+ return
+
+ # Find the largest icon file to use as main image
+ icon_paths = [os.path.join(iconset_dir, f) for f in icon_files]
+
+ # Get sizes of all icons
+ icon_sizes = []
+ for icon_path in icon_paths:
+ img = None
+ try:
+ img = Image.open(icon_path)
+ icon_sizes.append((img.size[0], icon_path))
+ finally:
+ if img is not None:
+ img.close()
+ del img
+
+ if not icon_sizes:
+ error_msg = "Could not read any icon files for fallback method"
if progress_callback:
- progress_callback(error_msg, 90)
- progress_callback("Falling back to Pillow method...", 90)
+ progress_callback(100, 100, error_msg)
else:
print(error_msg)
- print("Falling back to Pillow method...")
- _fallback_method_internal(iconset_dir, icns_path, progress_callback)
-
-def _fallback_method_internal(iconset_dir, icns_path, progress_callback=None):
- """
- Internal fallback method using Pillow if iconutil is not available
- """
- icon_files = os.listdir(iconset_dir)
- if not icon_files:
- error_msg = "No icons generated, cannot create ICNS file"
+ return
+
+ # Find the largest icon
+ largest_icon_path = max(icon_sizes, key=lambda x: x[0])[1]
+
+ # Open the largest icon as main image
+ main_img = None
+ append_images = []
+ try:
+ main_img = Image.open(largest_icon_path)
+
+ # Create list of additional images
+ for icon_path in icon_paths:
+ if icon_path != largest_icon_path:
+ img = None
+ try:
+ img = Image.open(icon_path)
+ append_images.append(img)
+ except Exception as e:
+ # Skip individual images that can't be opened
+ print(f"Warning: Could not open icon {icon_path} for fallback: {e}")
+ if img is not None:
+ img.close()
+ del img
+
+ # Save as ICNS
+ if append_images:
+ main_img.save(
+ icns_path,
+ format='ICNS',
+ append_images=append_images
+ )
+ else:
+ main_img.save(icns_path, format='ICNS')
+ finally:
+ # Ensure all image resources are released
+ if main_img is not None:
+ main_img.close()
+ del main_img
+ for img in append_images:
+ img.close()
+ del img
+
+ success_msg = f"Successfully converted using fallback method to {icns_path}"
if progress_callback:
- progress_callback(error_msg, 100)
+ progress_callback(100, 100, success_msg)
+ else:
+ print(success_msg)
+ except Exception as e:
+ error_msg = f"Error in fallback method: {e}"
+ if progress_callback:
+ progress_callback(100, 100, error_msg)
else:
print(error_msg)
- return
-
- # Find the largest icon file to use as main image
- icon_paths = [os.path.join(iconset_dir, f) for f in icon_files]
- largest_icon = max(icon_paths, key=lambda p: Image.open(p).size[0])
-
- # Open the largest icon as main image
- main_img = Image.open(largest_icon)
-
- # Create list of additional images
- append_images = []
- for icon_path in icon_paths:
- if icon_path != largest_icon:
- append_images.append(Image.open(icon_path))
-
- # Save as ICNS
- if append_images:
- main_img.save(
- icns_path,
- format='ICNS',
- append_images=append_images
- )
- else:
- main_img.save(icns_path, format='ICNS')
-
- success_msg = f"Successfully converted using fallback method to {icns_path}"
- if progress_callback:
- progress_callback(success_msg, 100)
- else:
- print(success_msg)
def main():
parser = argparse.ArgumentParser(description="Convert images to various formats (ICNS, PNG, JPG, WebP)")
diff --git a/support/notification.py b/support/notification.py
new file mode 100644
index 0000000..694fc15
--- /dev/null
+++ b/support/notification.py
@@ -0,0 +1,75 @@
+#!/usr/bin/env python3
+# -*- coding: utf-8 -*-
+"""
+System-level notification module for Converter application
+Supports macOS, Windows, and Linux platforms
+"""
+
+import platform
+import subprocess
+import sys
+
+
+def send_notification(title, message):
+ """Send system-level notification
+
+ Args:
+ title (str): Notification title
+ message (str): Notification content
+ """
+ system = platform.system()
+
+ try:
+ if system == "Darwin":
+ # macOS: Use terminal-notifier or osascript
+ try:
+ # Try terminal-notifier first (if installed)
+ subprocess.run([
+ "terminal-notifier",
+ "-title", title,
+ "-message", message,
+ "-appIcon", "Terminal"
+ ], check=True, capture_output=True)
+ except (subprocess.CalledProcessError, FileNotFoundError):
+ # Fallback to osascript (built-in)
+ script = f'display notification "{message}" with title "{title}"'
+ subprocess.run(["osascript", "-e", script], check=True, capture_output=True)
+
+ elif system == "Windows":
+ # Windows: Use win10toast if available, otherwise fallback
+ try:
+ from win10toast import ToastNotifier
+ toaster = ToastNotifier()
+ toaster.show_toast(title, message, duration=5)
+ except ImportError:
+ # Fallback to Windows PowerShell
+ script = f'''Add-Type -AssemblyName System.Windows.Forms;
+ [System.Windows.Forms.MessageBox]::Show("{message}", "{title}",
+ [System.Windows.Forms.MessageBoxButtons]::OK,
+ [System.Windows.Forms.MessageBoxIcon]::Information)'''
+ subprocess.run(["powershell", "-Command", script], check=True, capture_output=True)
+
+ elif system == "Linux":
+ # Linux: Use notify-send (most desktop environments)
+ subprocess.run([
+ "notify-send",
+ "-a", "Converter",
+ title,
+ message
+ ], check=True, capture_output=True)
+
+ else:
+ # Unsupported platform
+ print(f"[Notification] {title}: {message}")
+
+ except subprocess.CalledProcessError as e:
+ # Log error but don't crash the application
+ print(f"Failed to send notification: {e}")
+ except Exception as e:
+ # Catch all other exceptions
+ print(f"Unexpected error sending notification: {e}")
+
+
+if __name__ == "__main__":
+ # Test notification functionality
+ send_notification("Test Notification", "This is a test notification from Converter")
diff --git a/support/task_manager.py b/support/task_manager.py
new file mode 100644
index 0000000..268405e
--- /dev/null
+++ b/support/task_manager.py
@@ -0,0 +1,243 @@
+#!/usr/bin/env python3
+# -*- coding: utf-8 -*-
+"""
+Task manager module for Converter application
+Handles task queue, concurrency, and progress tracking
+"""
+
+from concurrent.futures import ThreadPoolExecutor, as_completed
+import uuid
+import time
+from PySide6.QtCore import QObject, Signal, QThread
+import threading
+
+
+class TaskStatus:
+ """Enum for task status"""
+ PENDING = "pending"
+ RUNNING = "running"
+ COMPLETED = "completed"
+ FAILED = "failed"
+ CANCELLED = "cancelled"
+
+
+class Task:
+ """Task class to represent a conversion task"""
+
+ def __init__(self, task_type, input_path, output_path, **kwargs):
+ self.task_id = str(uuid.uuid4())
+ self.task_type = task_type # "image" or "arc"
+ self.input_path = input_path
+ self.output_path = output_path
+ self.status = TaskStatus.PENDING
+ self.progress = 0
+ self.start_time = None
+ self.end_time = None
+ self.result = None
+ self.error = None
+ self.metadata = kwargs
+
+ def __str__(self):
+ return f"Task {self.task_id}: {self.task_type} - {self.status} ({self.progress}%)"
+
+
+class TaskManager(QObject):
+ """Task manager to handle task queue and concurrency"""
+
+ # Signals for progress and status updates
+ task_added = Signal(str, dict) # task_id, task_info
+ task_updated = Signal(str, dict) # task_id, task_info
+ task_completed = Signal(str, dict) # task_id, task_info
+ task_failed = Signal(str, dict, str) # task_id, task_info, error
+ progress_updated = Signal(str, int) # task_id, progress
+
+ def __init__(self, max_workers=4):
+ super().__init__()
+ self.tasks = {}
+ self.task_queue = []
+ self.executor = ThreadPoolExecutor(max_workers=max_workers)
+ self.futures = {}
+ self.is_running = False
+ self.lock = threading.Lock()
+
+ def add_task(self, task_type, input_path, output_path, **kwargs):
+ """Add a new task to the queue"""
+ task = Task(task_type, input_path, output_path, **kwargs)
+
+ with self.lock:
+ self.tasks[task.task_id] = task
+ self.task_queue.append(task)
+
+ # Emit signal
+ self.task_added.emit(task.task_id, self._get_task_info(task))
+
+ # Start processing if not already running
+ if not self.is_running:
+ self._start_processing()
+
+ return task.task_id
+
+ def _start_processing(self):
+ """Start processing tasks from the queue"""
+ if self.is_running:
+ return
+
+ self.is_running = True
+
+ def process_queue():
+ while True:
+ with self.lock:
+ if not self.task_queue:
+ self.is_running = False
+ break
+
+ # Get next task
+ task = self.task_queue.pop(0)
+ task.status = TaskStatus.RUNNING
+ task.start_time = time.time()
+
+ # Emit task updated signal
+ self.task_updated.emit(task.task_id, self._get_task_info(task))
+
+ # Submit task to executor
+ future = self.executor.submit(self._execute_task, task)
+ self.futures[future] = task.task_id
+
+ # Handle task completion
+ future.add_done_callback(self._handle_task_completion)
+
+ # Start processing in a separate thread to avoid blocking the main thread
+ threading.Thread(target=process_queue, daemon=True).start()
+
+ def _execute_task(self, task):
+ """Execute a task"""
+ try:
+ # Simulate task execution with progress updates
+ # In real implementation, this would call the actual conversion function
+ for i in range(101):
+ time.sleep(0.1) # Simulate work
+ task.progress = i
+ self.progress_updated.emit(task.task_id, i)
+
+ task.status = TaskStatus.COMPLETED
+ task.end_time = time.time()
+ task.result = "Success"
+
+ return task
+
+ except Exception as e:
+ task.status = TaskStatus.FAILED
+ task.end_time = time.time()
+ task.error = str(e)
+ raise
+
+ def _handle_task_completion(self, future):
+ """Handle task completion"""
+ task_id = self.futures.pop(future)
+
+ try:
+ task = future.result()
+ task_info = self._get_task_info(task)
+
+ if task.status == TaskStatus.COMPLETED:
+ self.task_completed.emit(task_id, task_info)
+ else:
+ self.task_failed.emit(task_id, task_info, task.error)
+
+ except Exception as e:
+ with self.lock:
+ task = self.tasks.get(task_id)
+ if task:
+ task.status = TaskStatus.FAILED
+ task.end_time = time.time()
+ task.error = str(e)
+
+ if task:
+ task_info = self._get_task_info(task)
+ self.task_failed.emit(task_id, task_info, str(e))
+
+ # Emit task updated signal
+ with self.lock:
+ if task_id in self.tasks:
+ task = self.tasks[task_id]
+ self.task_updated.emit(task_id, self._get_task_info(task))
+
+ def _get_task_info(self, task):
+ """Get task information as a dictionary"""
+ return {
+ "task_id": task.task_id,
+ "task_type": task.task_type,
+ "input_path": task.input_path,
+ "output_path": task.output_path,
+ "status": task.status,
+ "progress": task.progress,
+ "start_time": task.start_time,
+ "end_time": task.end_time,
+ "result": task.result,
+ "error": task.error,
+ "metadata": task.metadata
+ }
+
+ def get_task(self, task_id):
+ """Get a task by its ID"""
+ with self.lock:
+ return self.tasks.get(task_id)
+
+ def get_all_tasks(self):
+ """Get all tasks"""
+ with self.lock:
+ return list(self.tasks.values())
+
+ def get_active_tasks(self):
+ """Get all active tasks"""
+ with self.lock:
+ return [task for task in self.tasks.values()
+ if task.status in [TaskStatus.PENDING, TaskStatus.RUNNING]]
+
+ def get_completed_tasks(self):
+ """Get all completed tasks"""
+ with self.lock:
+ return [task for task in self.tasks.values()
+ if task.status == TaskStatus.COMPLETED]
+
+ def cancel_task(self, task_id):
+ """Cancel a task"""
+ with self.lock:
+ if task_id in self.tasks:
+ task = self.tasks[task_id]
+ if task.status in [TaskStatus.PENDING, TaskStatus.RUNNING]:
+ task.status = TaskStatus.CANCELLED
+ # Remove from queue if pending
+ if task in self.task_queue:
+ self.task_queue.remove(task)
+ return True
+ return False
+
+ def retry_task(self, task_id):
+ """Retry a failed or cancelled task"""
+ with self.lock:
+ if task_id in self.tasks:
+ old_task = self.tasks[task_id]
+ if old_task.status in [TaskStatus.FAILED, TaskStatus.CANCELLED]:
+ new_task = Task(
+ old_task.task_type,
+ old_task.input_path,
+ old_task.output_path,
+ **old_task.metadata
+ )
+ self.tasks[new_task.task_id] = new_task
+ self.task_queue.append(new_task)
+ return new_task.task_id
+ return None
+
+ def clear_completed_tasks(self):
+ """Clear all completed tasks"""
+ with self.lock:
+ completed_ids = [task.task_id for task in self.tasks.values()
+ if task.status in [TaskStatus.COMPLETED, TaskStatus.FAILED, TaskStatus.CANCELLED]]
+ for task_id in completed_ids:
+ del self.tasks[task_id]
+
+ def shutdown(self):
+ """Shutdown the task manager"""
+ self.executor.shutdown(wait=True)
diff --git a/update/README.md b/update/README.md
index ef2a947..3d450fa 100644
--- a/update/README.md
+++ b/update/README.md
@@ -41,8 +41,8 @@ from update.download_update import download_and_apply_update
# 假设从UpdateManager获取到了更新信息
update_info = {
"status": "update_available",
- "download_url": "https://github.com/user/repo/releases/tag/v2.0.0",
- "latest_version": "v2.0.0" # 注意:需要完整的tag名称
+ "download_url": "https://github.com/user/repo/releases/tag/v2.1.0A6",
+ "latest_version": "v2.1.0A6" # 注意:需要完整的tag名称
}
# 下载并应用更新
@@ -132,7 +132,7 @@ def perform_update():
#### `download_update(tag_name: str) -> dict`
下载并应用更新。
-- `tag_name`: 版本标签名称(如v2.0.0)
+- `tag_name`: 版本标签名称(如v2.1.0A6)
返回字典结构:
```python
diff --git a/update/download_update.py b/update/download_update.py
index 42c9552..78c1000 100644
--- a/update/download_update.py
+++ b/update/download_update.py
@@ -47,7 +47,7 @@ def _extract_download_url(self, tag_name: str) -> Optional[str]:
Get the actual download URL from GitHub API
Args:
- tag_name: Version tag name (e.g., v2.0.0)
+ tag_name: Version tag name (e.g., v2.1.0A6)
Returns:
str: Actual zip file download URL, returns None if extraction fails
@@ -64,13 +64,13 @@ def _extract_download_url(self, tag_name: str) -> Optional[str]:
platform_str = "intel"
# Use GitHub API to get release information
- api_url = f"https://api.github.com/repos/pyquick/Converter/releases/tags/{tag_name}"
+ api_url = f"https://api.github.com/repos/intsant/Converter/releases/tags/{tag_name}"
response = requests.get(api_url, timeout=10)
response.encoding = 'utf-8'
if response.status_code != 200:
print(f"GitHub API request failed: {api_url} (status code: {response.status_code})")
# Fallback to manually constructed URL
- download_url = f"https://github.com/pyquick/Converter/releases/download/{tag_name}/Converter_{platform_str}_darwin.zip"
+ download_url = f"https://github.com/intsant/Converter/releases/download/{tag_name}/Converter_{platform_str}_darwin.zip"
return download_url
release_data = response.json()
@@ -87,7 +87,7 @@ def _extract_download_url(self, tag_name: str) -> Optional[str]:
# If no matching file is found, fallback to manually constructed URL
print(f"No matching file found {expected_filename}, using manually constructed URL")
- download_url = f"https://github.com/pyquick/Converter/releases/download/{tag_name}/Converter_{platform_str}_darwin.zip"
+ download_url = f"https://github.com/intsant/Converter/releases/download/{tag_name}/Converter_{platform_str}_darwin.zip"
# Verify if URL is valid - use GET request as HEAD requests might be handled differently by CDN
try:
@@ -113,7 +113,7 @@ def _extract_download_url(self, tag_name: str) -> Optional[str]:
else:
platform_str = "intel"
- download_url = f"https://github.com/pyquick/Converter/releases/download/{tag_name}/Converter_{platform_str}_darwin.zip"
+ download_url = f"https://github.com/intsant/Converter/releases/download/{tag_name}/Converter_{platform_str}_darwin.zip"
return download_url
def download_update(self, tag_name: str, progress_callback=None) -> Dict[str, Any]:
@@ -121,7 +121,7 @@ def download_update(self, tag_name: str, progress_callback=None) -> Dict[str, An
Download and extract update files
Args:
- tag_name: Version tag name (e.g., v2.0.0)
+ tag_name: Version tag name (e.g., v2.1.0A6)
progress_callback: Progress callback function
Returns:
@@ -690,6 +690,9 @@ def download_and_apply_update(update_info: Dict[str, Any], progress_callback=Non
# Create target directory for update
target_directory = tempfile.mkdtemp(prefix="converter_update_")
downloader = UpdateDownloader(download_url, target_directory)
+ created_downloader = True
+ else:
+ created_downloader = False
try:
result = downloader.download_update(latest_version, progress_callback)
@@ -717,14 +720,19 @@ def download_and_apply_update(update_info: Dict[str, Any], progress_callback=Non
print(f"❌ Failed to start update process: {e}")
result["status"] = "error"
result["message"] = f"Failed to start update process: {e}"
+ # Clean up if we created the downloader and update failed
+ if created_downloader:
+ downloader.cleanup()
return result
# Add downloader object to result for cleanup by caller when appropriate
result["downloader"] = downloader
+ result["created_downloader"] = created_downloader
return result
except Exception as e:
# Clean up immediately if an exception occurs
- downloader.cleanup()
+ if created_downloader:
+ downloader.cleanup()
return {
"status": "error",
"message": f"Error occurred during download: {e}"
@@ -734,8 +742,8 @@ def download_and_apply_update(update_info: Dict[str, Any], progress_callback=Non
if __name__ == "__main__":
# Test code
test_info = {
- "download_url": "https://github.com/pyquick/converter/releases/tag/v2.0.0",
- "latest_version": "2.0.0"
+ "download_url": "https://github.com/intsant/converter/releases/tag/v2.1.0A6",
+ "latest_version": "2.1.0A6"
}
result = download_and_apply_update(test_info, "./test_update")
diff --git a/update/example_usage.py b/update/example_usage.py
index 51d985d..ffacd4f 100644
--- a/update/example_usage.py
+++ b/update/example_usage.py
@@ -20,7 +20,7 @@ def example_update_workflow():
print("=== 应用程序更新工作流程示例 ===")
# 1. 初始化更新管理器
- current_version = "2.0.0RC1" # 从配置文件或代码中获取当前版本
+ current_version = "2.1.0A6" # 从配置文件或代码中获取当前版本
manager = UpdateManager(current_version)
print(f"当前应用程序版本: {current_version}")
@@ -82,8 +82,8 @@ def quick_download_example():
test_update_info = {
"status": "update_available",
"message": "测试更新",
- "download_url": "https://github.com/pyquick/converter/releases/tag/v2.0.0",
- "latest_version": "v2.0.0" # 注意:这里需要完整的tag名称
+ "download_url": "https://github.com/intsant/converter/releases/tag/v2.1.0A6",
+ "latest_version": "v2.1.0A6" # 注意:这里需要完整的tag名称
}
print("开始测试下载功能...")
diff --git a/update/run_complete_update.py b/update/run_complete_update.py
index 17108cb..91c79e9 100755
--- a/update/run_complete_update.py
+++ b/update/run_complete_update.py
@@ -19,7 +19,7 @@ def main():
print("🔄 开始检查更新...")
# 获取当前版本
- current_version = "2.0.0RC1" # 从settings/update_settings_gui.py获取
+ current_version = "2.1.0A6" # 从settings/update_settings_gui.py获取
# 创建更新管理器
update_manager = UpdateManager(current_version)
diff --git a/update/test_update.py b/update/test_update.py
index b6e7d78..e3fb959 100644
--- a/update/test_update.py
+++ b/update/test_update.py
@@ -19,7 +19,7 @@ def main():
print("✅ 成功导入UpdateManager")
# 获取当前版本
- current_version = "2.0.0RC1"
+ current_version = "2.1.0A6"
print(f"📍 当前版本: {current_version}")
# 创建更新管理器
diff --git a/update/update_complete.command b/update/update_complete.command
index 63bd197..6328243 100755
--- a/update/update_complete.command
+++ b/update/update_complete.command
@@ -32,7 +32,7 @@ pkill -f "python.*converter" 2>/dev/null || true
# Kill Converter.app processes
echo "🛑 Terminating Converter.app processes..."
pkill -f "Converter.app" 2>/dev/null || true
-pkill -f "com.pyquick.converter" 2>/dev/null || true
+pkill -f "com.intsant.converter" 2>/dev/null || true
# Wait for processes to fully terminate
echo "⏳ Waiting for processes to terminate..."
@@ -44,7 +44,7 @@ pkill -9 -f "python.*arc_gui.py" 2>/dev/null || true
pkill -9 -f "python.*Converter.py" 2>/dev/null || true
pkill -9 -f "python.*converter" 2>/dev/null || true
pkill -9 -f "Converter.app" 2>/dev/null || true
-pkill -9 -f "com.pyquick.converter" 2>/dev/null || true
+pkill -9 -f "com.intsant.converter" 2>/dev/null || true
# Wait again
sleep 2
diff --git a/update/update_manager.py b/update/update_manager.py
index 760456d..d3a7918 100644
--- a/update/update_manager.py
+++ b/update/update_manager.py
@@ -10,7 +10,7 @@ def __init__(self, current_version: str):
self.current_version = self._parse_version(current_version)
def _parse_version(self, version_str: str) -> tuple:
- # Handles versions like 2.0.0 and pre-release like 2.0.0RC1, 2.0.0A1, 2.0.0D1, 2.0.0RC1
+ # Handles versions like 2.1.0A6 and pre-release like 2.1.0A6RC1, 2.1.0A6A1, 2.1.0A6D1, 2.1.0A6RC1
parts = version_str.split('.')
if len(parts) != 3:
raise ValueError(f"Invalid version string: {version_str}")
@@ -46,13 +46,13 @@ def _parse_version(self, version_str: str) -> tuple:
def check_for_updates(self, include_prerelease: bool, prerelease_type: Optional[str] = None) -> dict:
try:
from con import CON
- repo_owner = "pyquick"
+ repo_owner = "intsant"
repo_name = "converter"
url = f"https://api.github.com/repos/{repo_owner}/{repo_name}/releases"
headers = CON.headers.copy()
# 读取PAT设置
- settings = QSettings("pyquick", "converter")
+ settings = QSettings("intsant", "converter")
encrypted_pat = settings.value("general/github_pat", "", type=str)
# 如果有PAT,添加到headers
@@ -227,7 +227,7 @@ def get_github_pat(self) -> str:
str: 解密后的PAT,如果没有设置则返回空字符串
"""
try:
- settings = QSettings("pyquick", "converter")
+ settings = QSettings("intsant", "converter")
encrypted_pat = settings.value("general/github_pat", "", type=str)
if encrypted_pat:
import os, sys
diff --git a/update_complete.sh b/update_complete.sh
index d38f048..75a1952 100644
--- a/update_complete.sh
+++ b/update_complete.sh
@@ -39,7 +39,7 @@ try:
print('🔄 Starting to check for updates...')
# Get current version
- current_version = '2.0.0RC1'
+ current_version = '2.1.0A6'
print(f'📍 Current version: {current_version}')
# Create update manager