| import sys |
| import os |
| import json |
| import threading |
| import shutil |
| from pathlib import Path |
| from datetime import datetime |
|
|
| from PyQt6.QtWidgets import ( |
| QApplication, QMainWindow, QWidget, QVBoxLayout, QHBoxLayout, |
| QPushButton, QLabel, QFileDialog, QComboBox, QProgressBar, |
| QTextEdit, QTabWidget, QGroupBox, QGridLayout, QMessageBox, |
| QListWidget, QListWidgetItem, QCheckBox, QSpinBox, QLineEdit, |
| QSplitter, QFrame, QToolButton, QSizePolicy |
| ) |
| from PyQt6.QtCore import Qt, QThread, pyqtSignal, QSize, QPropertyAnimation, QEasingCurve, QRect |
| from PyQt6.QtGui import QFont, QIcon, QDragEnterEvent, QDropEvent, QPalette, QColor, QAction, QBrush, QPen, QPainter, \ |
| QPixmap |
|
|
| |
| import warnings |
|
|
| warnings.filterwarnings("ignore", message="Couldn't find ffmpeg or avconv") |
|
|
| |
| ffmpeg_path = None |
| for path in [ |
| shutil.which('ffmpeg'), |
| shutil.which('ffmpeg.exe'), |
| r"C:\ffmpeg\bin\ffmpeg.exe", |
| r"C:\Program Files\ffmpeg\bin\ffmpeg.exe" |
| ]: |
| if path and os.path.exists(path): |
| ffmpeg_path = path |
| os.environ["FFMPEG_BINARY"] = ffmpeg_path |
| break |
|
|
| |
| from converters.document_converter import DocumentConverter |
| from converters.image_converter import ImageConverter |
| from converters.video_converter import VideoConverter |
| from converters.audio_converter import AudioConverter |
| from ai.ai_assistant import AIAssistant |
|
|
|
|
| class ConversionThread(QThread): |
| progress_update = pyqtSignal(int) |
| status_update = pyqtSignal(str) |
| conversion_complete = pyqtSignal(bool, str) |
|
|
| def __init__(self, converter, input_path, output_path, options=None): |
| super().__init__() |
| self.converter = converter |
| self.input_path = input_path |
| self.output_path = output_path |
| self.options = options or {} |
|
|
| def run(self): |
| try: |
| self.status_update.emit(f"Converting: {os.path.basename(self.input_path)}") |
| success = self.converter.convert( |
| self.input_path, |
| self.output_path, |
| self.options, |
| self.progress_update |
| ) |
|
|
| if success: |
| self.conversion_complete.emit(True, f"β
Successfully converted: {os.path.basename(self.input_path)}") |
| else: |
| self.conversion_complete.emit(False, f"β Failed to convert: {os.path.basename(self.input_path)}") |
| except Exception as e: |
| self.conversion_complete.emit(False, f"β Error: {str(e)}") |
|
|
|
|
| class FileConverterApp(QMainWindow): |
| def __init__(self): |
| super().__init__() |
| self.setWindowTitle("AI File Converter Pro") |
| self.setGeometry(100, 100, 1400, 900) |
|
|
| |
| self.setup_logo() |
| |
|
|
| |
| self.sidebar_visible = True |
|
|
| |
| self.setStyleSheet(""" |
| QMainWindow { |
| background-color: #1e1e2e; |
| } |
| QGroupBox { |
| font-weight: bold; |
| border: 2px solid #313244; |
| border-radius: 8px; |
| margin-top: 10px; |
| padding-top: 10px; |
| color: #cdd6f4; |
| font-size: 13px; |
| } |
| QGroupBox::title { |
| subcontrol-origin: margin; |
| left: 10px; |
| padding: 0 5px 0 5px; |
| color: #89b4fa; |
| } |
| QPushButton { |
| background-color: #89b4fa; |
| border: none; |
| padding: 8px; |
| border-radius: 6px; |
| font-weight: bold; |
| color: #1e1e2e; |
| } |
| QPushButton:hover { |
| background-color: #b4befe; |
| } |
| QPushButton:pressed { |
| background-color: #6c7086; |
| } |
| QToolButton { |
| background-color: #313244; |
| border: 1px solid #45475a; |
| border-radius: 6px; |
| padding: 8px; |
| color: #cdd6f4; |
| } |
| QToolButton:hover { |
| background-color: #45475a; |
| } |
| QListWidget, QTextEdit, QLineEdit, QComboBox { |
| background-color: #313244; |
| border: 1px solid #45475a; |
| border-radius: 6px; |
| padding: 5px; |
| color: #cdd6f4; |
| } |
| QProgressBar { |
| border: 2px solid #45475a; |
| border-radius: 6px; |
| text-align: center; |
| color: #cdd6f4; |
| } |
| QProgressBar::chunk { |
| background-color: #89b4fa; |
| border-radius: 4px; |
| } |
| QTabWidget::pane { |
| background-color: #1e1e2e; |
| border: 1px solid #313244; |
| border-radius: 6px; |
| } |
| QTabBar::tab { |
| background-color: #313244; |
| padding: 8px; |
| margin: 2px; |
| border-radius: 4px; |
| color: #cdd6f4; |
| } |
| QTabBar::tab:selected { |
| background-color: #89b4fa; |
| color: #1e1e2e; |
| } |
| QLabel { |
| color: #cdd6f4; |
| } |
| QCheckBox { |
| color: #cdd6f4; |
| } |
| QSplitter::handle { |
| background-color: #313244; |
| margin: 2px; |
| } |
| QSplitter::handle:hover { |
| background-color: #89b4fa; |
| } |
| """) |
|
|
| |
| self.doc_converter = DocumentConverter() |
| self.img_converter = ImageConverter() |
| self.video_converter = VideoConverter() |
| self.audio_converter = AudioConverter() |
| self.ai_assistant = AIAssistant() |
|
|
| |
| self.conversion_queue = [] |
| self.current_conversion = None |
| self.active_threads = [] |
| self.converted_count = 0 |
| self.failed_count = 0 |
|
|
| self.setup_ui() |
| self.setup_drag_drop() |
| self.setup_menu_bar() |
|
|
| def setup_logo(self): |
| """Setup window icon and UI logo""" |
| |
| logo_paths = [ |
| Path(__file__).parent / "logo.png", |
| Path(__file__).parent / "logo.ico", |
| Path(__file__).parent / "assets" / "logo.png", |
| Path(__file__).parent / "assets" / "logo.ico", |
| |
| Path(sys.executable).parent / "logo.png", |
| Path(sys.executable).parent / "logo.ico", |
| Path(sys.executable).parent / "assets" / "logo.png", |
| ] |
|
|
| icon_loaded = False |
| for logo_path in logo_paths: |
| if logo_path.exists(): |
| |
| self.setWindowIcon(QIcon(str(logo_path))) |
| icon_loaded = True |
| print(f"β
Logo loaded from: {logo_path}") |
| break |
|
|
| if not icon_loaded: |
| |
| self.setWindowIcon(self.create_default_icon()) |
| print("β οΈ Using default generated icon. Place 'logo.png' in the application directory for custom logo.") |
|
|
| def create_default_icon(self): |
| """Create a default icon programmatically""" |
| pixmap = QPixmap(256, 256) |
| pixmap.fill(Qt.GlobalColor.transparent) |
|
|
| painter = QPainter(pixmap) |
| painter.setRenderHint(QPainter.RenderHint.Antialiasing) |
|
|
| |
| painter.setBrush(QBrush(QColor(137, 180, 250))) |
| painter.setPen(Qt.PenStyle.NoPen) |
| painter.drawEllipse(20, 20, 216, 216) |
|
|
| |
| painter.setBrush(QBrush(QColor(30, 30, 46))) |
| painter.drawEllipse(40, 40, 176, 176) |
|
|
| |
| painter.setPen(QPen(QColor(137, 180, 250), 2)) |
| font = QFont("Arial", 80, QFont.Weight.Bold) |
| painter.setFont(font) |
| painter.drawText(QRect(40, 70, 176, 100), Qt.AlignmentFlag.AlignCenter, "AI") |
|
|
| |
| painter.setPen(QPen(QColor(166, 227, 161), 5)) |
| painter.setBrush(Qt.BrushStyle.NoBrush) |
|
|
| |
| painter.drawLine(180, 180, 210, 195) |
| painter.drawLine(210, 195, 180, 210) |
|
|
| |
| painter.drawLine(76, 180, 46, 195) |
| painter.drawLine(46, 195, 76, 210) |
|
|
| painter.end() |
|
|
| return QIcon(pixmap) |
|
|
| def create_ui_logo(self, size=48): |
| """Create a logo widget for the UI""" |
| logo_label = QLabel() |
|
|
| |
| logo_path = Path(__file__).parent / "logo.png" |
| if not logo_path.exists() and hasattr(sys, 'frozen'): |
| |
| logo_path = Path(sys.executable).parent / "logo.png" |
|
|
| if logo_path.exists(): |
| pixmap = QPixmap(str(logo_path)) |
| scaled_pixmap = pixmap.scaled(size, size, |
| Qt.AspectRatioMode.KeepAspectRatio, |
| Qt.TransformationMode.SmoothTransformation) |
| logo_label.setPixmap(scaled_pixmap) |
| else: |
| |
| pixmap = QPixmap(size, size) |
| pixmap.fill(Qt.GlobalColor.transparent) |
|
|
| painter = QPainter(pixmap) |
| painter.setRenderHint(QPainter.RenderHint.Antialiasing) |
|
|
| |
| painter.setBrush(QBrush(QColor(137, 180, 250))) |
| painter.setPen(Qt.PenStyle.NoPen) |
| painter.drawEllipse(4, 4, size - 8, size - 8) |
|
|
| |
| painter.setBrush(QBrush(QColor(30, 30, 46))) |
| painter.drawEllipse(8, 8, size - 16, size - 16) |
|
|
| |
| painter.setPen(QPen(QColor(137, 180, 250), 1)) |
| font = QFont("Arial", size // 3, QFont.Weight.Bold) |
| painter.setFont(font) |
| painter.drawText(QRect(8, 8, size - 16, size - 16), |
| Qt.AlignmentFlag.AlignCenter, "AI") |
|
|
| painter.end() |
| logo_label.setPixmap(pixmap) |
|
|
| logo_label.setFixedSize(size, size) |
| logo_label.setToolTip("AI File Converter Pro") |
| return logo_label |
|
|
| def setup_ui(self): |
| central_widget = QWidget() |
| self.setCentralWidget(central_widget) |
| main_layout = QHBoxLayout(central_widget) |
| main_layout.setContentsMargins(0, 0, 0, 0) |
| main_layout.setSpacing(0) |
|
|
| |
| self.main_splitter = QSplitter(Qt.Orientation.Horizontal) |
|
|
| |
| self.left_panel = self.create_left_panel_with_logo() |
| self.main_splitter.addWidget(self.left_panel) |
|
|
| |
| self.sidebar_widget = self.create_sidebar() |
| self.main_splitter.addWidget(self.sidebar_widget) |
|
|
| |
| self.main_splitter.setSizes([980, 420]) |
|
|
| main_layout.addWidget(self.main_splitter) |
|
|
| def setup_menu_bar(self): |
| """Setup menu bar with toggle option""" |
| menubar = self.menuBar() |
| menubar.setStyleSheet(""" |
| QMenuBar { |
| background-color: #1e1e2e; |
| color: #cdd6f4; |
| border-bottom: 1px solid #313244; |
| } |
| QMenuBar::item:selected { |
| background-color: #313244; |
| } |
| QMenu { |
| background-color: #1e1e2e; |
| color: #cdd6f4; |
| border: 1px solid #313244; |
| } |
| QMenu::item:selected { |
| background-color: #89b4fa; |
| color: #1e1e2e; |
| } |
| """) |
|
|
| |
| file_menu = menubar.addMenu("π File") |
|
|
| exit_action = QAction("πͺ Exit", self) |
| exit_action.setShortcut("Ctrl+Q") |
| exit_action.triggered.connect(self.close) |
| file_menu.addAction(exit_action) |
|
|
| |
| view_menu = menubar.addMenu("ποΈ View") |
|
|
| toggle_action = QAction("π€ Toggle AI Assistant", self) |
| toggle_action.setShortcut("Ctrl+Shift+A") |
| toggle_action.triggered.connect(self.toggle_sidebar) |
| view_menu.addAction(toggle_action) |
|
|
| view_menu.addSeparator() |
|
|
| reset_view_action = QAction("π Reset Layout", self) |
| reset_view_action.setShortcut("Ctrl+Shift+R") |
| reset_view_action.triggered.connect(self.reset_layout) |
| view_menu.addAction(reset_view_action) |
|
|
| |
| help_menu = menubar.addMenu("β Help") |
|
|
| about_action = QAction("π About", self) |
| about_action.setShortcut("F1") |
| about_action.triggered.connect(self.show_about) |
| help_menu.addAction(about_action) |
|
|
| def show_about(self): |
| """Show about dialog with logo""" |
| about_text = """<div style='text-align: center;'> |
| <h2 style='color: #89b4fa;'>π― AI File Converter Pro</h2> |
| <p><b>Version 2.0</b></p> |
| <p>Advanced file conversion tool with AI assistance</p> |
| <br> |
| <p><b>Features:</b></p> |
| <p>β’ Convert documents, images, videos, and audio</p> |
| <p>β’ AI-powered format recommendations</p> |
| <p>β’ Natural language commands</p> |
| <p>β’ Batch processing support</p> |
| <p>β’ Quality optimization</p> |
| <br> |
| <p>Β© 2024 AI File Converter Pro</p> |
| <p>All rights reserved</p> |
| </div>""" |
|
|
| about_dialog = QMessageBox(self) |
| about_dialog.setWindowTitle("About AI File Converter Pro") |
| about_dialog.setText(about_text) |
|
|
| |
| logo_path = Path(__file__).parent / "logo.png" |
| if not logo_path.exists() and hasattr(sys, 'frozen'): |
| logo_path = Path(sys.executable).parent / "logo.png" |
|
|
| if logo_path.exists(): |
| logo_pixmap = QPixmap(str(logo_path)).scaled(64, 64, |
| Qt.AspectRatioMode.KeepAspectRatio, |
| Qt.TransformationMode.SmoothTransformation) |
| about_dialog.setIconPixmap(logo_pixmap) |
|
|
| about_dialog.exec() |
|
|
| def create_left_panel_with_logo(self): |
| """Create the main left panel with logo in top-left corner""" |
| left_panel = QWidget() |
| left_layout = QVBoxLayout(left_panel) |
| left_layout.setSpacing(10) |
| left_layout.setContentsMargins(10, 10, 10, 10) |
|
|
| |
| header_widget = QWidget() |
| header_layout = QHBoxLayout(header_widget) |
| header_layout.setContentsMargins(0, 0, 0, 10) |
|
|
| |
| logo_widget = self.create_ui_logo(48) |
| header_layout.addWidget(logo_widget) |
|
|
| |
| title_label = QLabel("AI File Converter Pro") |
| title_label.setStyleSheet(""" |
| font-size: 20px; |
| font-weight: bold; |
| color: #89b4fa; |
| padding: 5px; |
| """) |
| header_layout.addWidget(title_label) |
|
|
| header_layout.addStretch() |
|
|
| |
| version_label = QLabel("v2.0") |
| version_label.setStyleSheet("color: #6c7086; font-size: 12px;") |
| header_layout.addWidget(version_label) |
|
|
| left_layout.addWidget(header_widget) |
| |
|
|
| |
| file_group = QGroupBox("π Files to Convert") |
| file_layout = QVBoxLayout() |
|
|
| self.file_list = QListWidget() |
| self.file_list.setSelectionMode(QListWidget.SelectionMode.ExtendedSelection) |
| self.file_list.setMinimumHeight(200) |
| file_layout.addWidget(self.file_list) |
|
|
| btn_layout = QHBoxLayout() |
| add_btn = QPushButton("β Add Files") |
| add_btn.clicked.connect(self.add_files) |
| add_folder_btn = QPushButton("π Add Folder") |
| add_folder_btn.clicked.connect(self.add_folder) |
| remove_btn = QPushButton("β Remove Selected") |
| remove_btn.clicked.connect(self.remove_selected) |
| clear_btn = QPushButton("ποΈ Clear All") |
| clear_btn.clicked.connect(self.clear_files) |
|
|
| btn_layout.addWidget(add_btn) |
| btn_layout.addWidget(add_folder_btn) |
| btn_layout.addWidget(remove_btn) |
| btn_layout.addWidget(clear_btn) |
| file_layout.addLayout(btn_layout) |
|
|
| file_group.setLayout(file_layout) |
| left_layout.addWidget(file_group) |
|
|
| |
| converter_group = QGroupBox("βοΈ Converter Settings") |
| converter_layout = QGridLayout() |
| converter_layout.setSpacing(10) |
|
|
| converter_layout.addWidget(QLabel("File Type:"), 0, 0) |
| self.file_type_combo = QComboBox() |
| self.file_type_combo.addItems(["Document", "Image", "Video", "Audio"]) |
| self.file_type_combo.currentTextChanged.connect(self.on_file_type_changed) |
| converter_layout.addWidget(self.file_type_combo, 0, 1) |
|
|
| converter_layout.addWidget(QLabel("Output Format:"), 1, 0) |
| self.output_format_combo = QComboBox() |
| converter_layout.addWidget(self.output_format_combo, 1, 1) |
|
|
| converter_layout.addWidget(QLabel("Quality:"), 2, 0) |
| self.quality_combo = QComboBox() |
| self.quality_combo.addItems(["High", "Medium", "Low"]) |
| converter_layout.addWidget(self.quality_combo, 2, 1) |
|
|
| converter_layout.addWidget(QLabel("Output Folder:"), 3, 0) |
| self.output_folder = QLineEdit() |
| self.output_folder.setText(str(Path.home() / "Downloads" / "Converted")) |
| converter_layout.addWidget(self.output_folder, 3, 1) |
|
|
| browse_btn = QPushButton("π Browse") |
| browse_btn.clicked.connect(self.browse_output_folder) |
| converter_layout.addWidget(browse_btn, 3, 2) |
|
|
| |
| Path(self.output_folder.text()).mkdir(parents=True, exist_ok=True) |
|
|
| self.ai_checkbox = QCheckBox("π€ Enable AI Enhancement") |
| self.ai_checkbox.setChecked(False) |
| converter_layout.addWidget(self.ai_checkbox, 4, 0, 1, 2) |
|
|
| converter_group.setLayout(converter_layout) |
| left_layout.addWidget(converter_group) |
|
|
| |
| progress_group = QGroupBox("π Conversion Progress") |
| progress_layout = QVBoxLayout() |
|
|
| self.progress_bar = QProgressBar() |
| progress_layout.addWidget(self.progress_bar) |
|
|
| self.status_label = QLabel("β
Ready to convert") |
| self.status_label.setAlignment(Qt.AlignmentFlag.AlignCenter) |
| progress_layout.addWidget(self.status_label) |
|
|
| self.stats_label = QLabel("π Converted: 0 | β Failed: 0") |
| self.stats_label.setAlignment(Qt.AlignmentFlag.AlignCenter) |
| progress_layout.addWidget(self.stats_label) |
|
|
| progress_group.setLayout(progress_layout) |
| left_layout.addWidget(progress_group) |
|
|
| |
| self.convert_btn = QPushButton("π Start Conversion") |
| self.convert_btn.setMinimumHeight(50) |
| self.convert_btn.setStyleSheet(""" |
| QPushButton { |
| background-color: #a6e3a1; |
| font-size: 16px; |
| font-weight: bold; |
| } |
| QPushButton:hover { |
| background-color: #94e2d5; |
| } |
| """) |
| self.convert_btn.clicked.connect(self.start_conversion) |
| left_layout.addWidget(self.convert_btn) |
|
|
| return left_panel |
|
|
| def create_sidebar(self): |
| """Create the collapsible AI sidebar""" |
| sidebar_container = QWidget() |
| sidebar_layout = QVBoxLayout(sidebar_container) |
| sidebar_layout.setContentsMargins(0, 0, 0, 0) |
| sidebar_layout.setSpacing(0) |
|
|
| |
| header = QWidget() |
| header.setStyleSheet(""" |
| QWidget { |
| background-color: #313244; |
| border-bottom: 1px solid #45475a; |
| } |
| """) |
| header_layout = QHBoxLayout(header) |
| header_layout.setContentsMargins(10, 10, 10, 10) |
|
|
| title_label = QLabel("π€ AI Assistant") |
| title_label.setStyleSheet("font-size: 14px; font-weight: bold; color: #89b4fa;") |
| header_layout.addWidget(title_label) |
|
|
| header_layout.addStretch() |
|
|
| |
| self.close_sidebar_btn = QToolButton() |
| self.close_sidebar_btn.setText("β") |
| self.close_sidebar_btn.setToolTip("Close Sidebar (Ctrl+Shift+A)") |
| self.close_sidebar_btn.setStyleSheet(""" |
| QToolButton { |
| background-color: #45475a; |
| border: none; |
| font-size: 14px; |
| padding: 5px; |
| } |
| QToolButton:hover { |
| background-color: #6c7086; |
| } |
| """) |
| self.close_sidebar_btn.clicked.connect(self.toggle_sidebar) |
| header_layout.addWidget(self.close_sidebar_btn) |
|
|
| sidebar_layout.addWidget(header) |
|
|
| |
| right_panel = QTabWidget() |
| right_panel.setStyleSheet(""" |
| QTabWidget::pane { |
| border: none; |
| } |
| """) |
|
|
| |
| ai_widget = QWidget() |
| ai_layout = QVBoxLayout(ai_widget) |
| ai_layout.setSpacing(10) |
|
|
| |
| suggestions_group = QGroupBox("π‘ Quick Commands") |
| suggestions_layout = QGridLayout() |
| suggestions_group.setMaximumHeight(220) |
|
|
| suggestions = [ |
| ("πΌοΈ To PNG", "convert this to png"), |
| ("π To PDF", "convert to pdf"), |
| ("π΅ To MP3", "make it mp3"), |
| ("π¬ Web format", "best format for web"), |
| ("π Analyze", "analyze this file"), |
| ("πΎ Compress", "how to compress video"), |
| ("π§ Email size", "best format for email"), |
| ("π¨οΈ Print", "best format for printing"), |
| ] |
|
|
| row, col = 0, 0 |
| for text, cmd in suggestions: |
| btn = QPushButton(text) |
| btn.setMaximumHeight(30) |
| btn.clicked.connect(lambda checked, c=cmd: self.ai_input.setText(c)) |
| suggestions_layout.addWidget(btn, row, col) |
| col += 1 |
| if col >= 2: |
| col = 0 |
| row += 1 |
|
|
| suggestions_group.setLayout(suggestions_layout) |
| ai_layout.addWidget(suggestions_group) |
|
|
| |
| self.ai_input = QTextEdit() |
| self.ai_input.setPlaceholderText("""π¬ Ask me anything about file conversion... |
| |
| Examples: |
| β’ "Convert this to PNG" |
| β’ "What's the best format for web images?" |
| β’ "How to compress video for email?" |
| β’ "Analyze this file" |
| β’ "Make it smaller without losing quality" """) |
| self.ai_input.setMaximumHeight(100) |
| ai_layout.addWidget(self.ai_input) |
|
|
| |
| ai_buttons_layout = QHBoxLayout() |
| ai_send_btn = QPushButton("π Ask AI") |
| ai_send_btn.clicked.connect(self.ask_ai_assistant) |
| ai_clear_btn = QPushButton("ποΈ Clear") |
| ai_clear_btn.clicked.connect(lambda: self.ai_input.clear()) |
| ai_buttons_layout.addWidget(ai_send_btn) |
| ai_buttons_layout.addWidget(ai_clear_btn) |
| ai_layout.addLayout(ai_buttons_layout) |
|
|
| |
| self.ai_response = QTextEdit() |
| self.ai_response.setReadOnly(True) |
| self.ai_response.setPlaceholderText("AI response will appear here...") |
| self.ai_response.setMinimumHeight(200) |
| ai_layout.addWidget(self.ai_response) |
|
|
| right_panel.addTab(ai_widget, "π¬ Assistant") |
|
|
| |
| log_widget = QWidget() |
| log_layout = QVBoxLayout(log_widget) |
| self.log_text = QTextEdit() |
| self.log_text.setReadOnly(True) |
| log_layout.addWidget(self.log_text) |
|
|
| log_btn_layout = QHBoxLayout() |
| clear_log_btn = QPushButton("ποΈ Clear Log") |
| clear_log_btn.clicked.connect(lambda: self.log_text.clear()) |
| log_btn_layout.addStretch() |
| log_btn_layout.addWidget(clear_log_btn) |
| log_layout.addLayout(log_btn_layout) |
|
|
| right_panel.addTab(log_widget, "π Log") |
|
|
| sidebar_layout.addWidget(right_panel) |
|
|
| return sidebar_container |
|
|
| def toggle_sidebar(self): |
| """Toggle the AI sidebar visibility""" |
| self.sidebar_visible = not self.sidebar_visible |
|
|
| if self.sidebar_visible: |
| self.sidebar_widget.setVisible(True) |
| self.close_sidebar_btn.setText("β") |
| self.main_splitter.setSizes([980, 420]) |
| else: |
| self.sidebar_widget.setVisible(False) |
| self.close_sidebar_btn.setText("βΆ") |
| self.main_splitter.setSizes([1400, 0]) |
|
|
| |
| self.close_sidebar_btn.setToolTip( |
| "Open Sidebar (Ctrl+Shift+A)" if not self.sidebar_visible else "Close Sidebar (Ctrl+Shift+A)") |
|
|
| def reset_layout(self): |
| """Reset layout to default""" |
| self.main_splitter.setSizes([980, 420]) |
| if not self.sidebar_visible: |
| self.toggle_sidebar() |
| self.log_message("π Layout reset to default") |
|
|
| def setup_drag_drop(self): |
| self.setAcceptDrops(True) |
|
|
| def dragEnterEvent(self, event: QDragEnterEvent): |
| if event.mimeData().hasUrls(): |
| event.acceptProposedAction() |
|
|
| def dropEvent(self, event: QDropEvent): |
| for url in event.mimeData().urls(): |
| file_path = url.toLocalFile() |
| if os.path.isfile(file_path): |
| self.add_file_to_list(file_path) |
|
|
| def add_files(self): |
| files, _ = QFileDialog.getOpenFileNames( |
| self, "Select Files to Convert", "", |
| "All Files (*.*)" |
| ) |
| for file in files: |
| self.add_file_to_list(file) |
|
|
| def add_folder(self): |
| folder = QFileDialog.getExistingDirectory(self, "Select Folder") |
| if folder: |
| for root, dirs, files in os.walk(folder): |
| for file in files: |
| self.add_file_to_list(os.path.join(root, file)) |
|
|
| def remove_selected(self): |
| for item in self.file_list.selectedItems(): |
| self.file_list.takeItem(self.file_list.row(item)) |
|
|
| def add_file_to_list(self, file_path): |
| |
| for i in range(self.file_list.count()): |
| if self.file_list.item(i).data(Qt.ItemDataRole.UserRole) == file_path: |
| return |
| item = QListWidgetItem(f"π {os.path.basename(file_path)}") |
| item.setToolTip(file_path) |
| item.setData(Qt.ItemDataRole.UserRole, file_path) |
| self.file_list.addItem(item) |
|
|
| def clear_files(self): |
| self.file_list.clear() |
|
|
| def on_file_type_changed(self, file_type): |
| self.output_format_combo.clear() |
|
|
| formats = { |
| "Document": ["pdf", "docx", "txt", "html", "md"], |
| "Image": ["png", "jpg", "webp", "bmp", "tiff"], |
| "Video": ["mp4", "avi", "mkv", "mov", "webm", "gif"], |
| "Audio": ["mp3", "wav", "ogg", "m4a", "flac"] |
| } |
| self.output_format_combo.addItems(formats.get(file_type, [])) |
|
|
| def browse_output_folder(self): |
| folder = QFileDialog.getExistingDirectory(self, "Select Output Folder") |
| if folder: |
| self.output_folder.setText(folder) |
| Path(folder).mkdir(parents=True, exist_ok=True) |
|
|
| def get_converter(self): |
| file_type = self.file_type_combo.currentText() |
| converters = { |
| "Document": self.doc_converter, |
| "Image": self.img_converter, |
| "Video": self.video_converter, |
| "Audio": self.audio_converter |
| } |
| return converters.get(file_type) |
|
|
| def start_conversion(self): |
| if self.file_list.count() == 0: |
| QMessageBox.warning(self, "Warning", "Please add files to convert") |
| return |
|
|
| |
| self.converted_count = 0 |
| self.failed_count = 0 |
| self.update_stats() |
|
|
| |
| self.conversion_queue = [] |
| selected_items = self.file_list.selectedItems() |
| if selected_items: |
| for item in selected_items: |
| self.conversion_queue.append(item.data(Qt.ItemDataRole.UserRole)) |
| else: |
| for i in range(self.file_list.count()): |
| self.conversion_queue.append(self.file_list.item(i).data(Qt.ItemDataRole.UserRole)) |
|
|
| self.convert_btn.setEnabled(False) |
| self.process_next_conversion() |
|
|
| def process_next_conversion(self): |
| if not self.conversion_queue: |
| self.log_message("π All conversions completed!") |
| self.status_label.setText("β
All conversions completed!") |
| self.progress_bar.setValue(100) |
| self.convert_btn.setEnabled(True) |
| QMessageBox.information(self, "Success", |
| f"Conversion complete!\nβ
Success: {self.converted_count}\nβ Failed: {self.failed_count}") |
| return |
|
|
| input_path = self.conversion_queue.pop(0) |
| self.current_conversion = input_path |
|
|
| |
| output_format = self.output_format_combo.currentText() |
| output_folder = self.output_folder.text() |
| base_name = Path(input_path).stem |
| output_path = Path(output_folder) / f"{base_name}.{output_format}" |
|
|
| |
| Path(output_folder).mkdir(parents=True, exist_ok=True) |
|
|
| |
| quality_map = { |
| "High": {"video": "18", "audio": "320k", "image": 95}, |
| "Medium": {"video": "23", "audio": "192k", "image": 85}, |
| "Low": {"video": "28", "audio": "128k", "image": 70} |
| } |
| quality = self.quality_combo.currentText() |
|
|
| |
| options = { |
| "ai_enhancement": self.ai_checkbox.isChecked(), |
| "quality": quality, |
| "quality_settings": quality_map[quality] |
| } |
|
|
| |
| converter = self.get_converter() |
| if converter: |
| self.conversion_thread = ConversionThread( |
| converter, input_path, str(output_path), options |
| ) |
| self.conversion_thread.progress_update.connect(self.update_progress) |
| self.conversion_thread.status_update.connect(self.update_status) |
| self.conversion_thread.conversion_complete.connect(self.on_conversion_complete) |
| self.conversion_thread.start() |
| self.active_threads.append(self.conversion_thread) |
| else: |
| self.log_message(f"β Invalid converter for: {input_path}") |
| self.process_next_conversion() |
|
|
| def update_progress(self, value): |
| self.progress_bar.setValue(value) |
|
|
| def update_status(self, status): |
| self.status_label.setText(status) |
| self.log_message(status) |
|
|
| def on_conversion_complete(self, success, message): |
| self.log_message(message) |
| if success: |
| self.converted_count += 1 |
| else: |
| self.failed_count += 1 |
| self.update_stats() |
| self.process_next_conversion() |
|
|
| def update_stats(self): |
| self.stats_label.setText(f"π Converted: {self.converted_count} | β Failed: {self.failed_count}") |
|
|
| def log_message(self, message): |
| timestamp = datetime.now().strftime("%H:%M:%S") |
| self.log_text.append(f"[{timestamp}] {message}") |
|
|
| def ask_ai_assistant(self): |
| """Process AI query and potentially auto-configure settings""" |
| question = self.ai_input.toPlainText() |
| if not question.strip(): |
| return |
|
|
| self.ai_response.setPlainText("π€ AI Assistant is thinking...") |
|
|
| |
| response = self.ai_assistant.ask(question, self.get_conversion_context()) |
|
|
| |
| suggested_format = self.extract_format_from_response(response) |
| if suggested_format: |
| |
| index = self.output_format_combo.findText(suggested_format.lower()) |
| if index >= 0: |
| self.output_format_combo.setCurrentIndex(index) |
| response += f"\n\nβ
**Auto-configured**: Output format changed to {suggested_format.upper()}" |
|
|
| |
| suggested_type = self.extract_file_type_from_response(response) |
| if suggested_type: |
| index = self.file_type_combo.findText(suggested_type.title()) |
| if index >= 0: |
| self.file_type_combo.setCurrentIndex(index) |
| response += f"\n\nβ
**Auto-configured**: File type changed to {suggested_type.title()}" |
|
|
| self.ai_response.setPlainText(response) |
|
|
| def extract_format_from_response(self, response: str) -> str: |
| """Extract suggested format from AI response""" |
| formats = ['pdf', 'docx', 'txt', 'html', 'md', 'png', 'jpg', 'jpeg', |
| 'webp', 'bmp', 'tiff', 'mp4', 'avi', 'mkv', 'mov', 'webm', |
| 'gif', 'mp3', 'wav', 'ogg', 'm4a', 'flac'] |
|
|
| for fmt in formats: |
| if f" {fmt.upper()}" in response.upper() or f"to {fmt}" in response.lower(): |
| return fmt |
| return "" |
|
|
| def extract_file_type_from_response(self, response: str) -> str: |
| """Extract suggested file type from AI response""" |
| types = ['document', 'image', 'video', 'audio'] |
| for t in types: |
| if t in response.lower(): |
| return t |
| return "" |
|
|
| def get_conversion_context(self): |
| return { |
| "file_type": self.file_type_combo.currentText(), |
| "output_format": self.output_format_combo.currentText(), |
| "file_count": self.file_list.count(), |
| "ai_enhancement": self.ai_checkbox.isChecked(), |
| "quality": self.quality_combo.currentText() |
| } |
|
|
|
|
| if __name__ == "__main__": |
| app = QApplication(sys.argv) |
| window = FileConverterApp() |
| window.show() |
| sys.exit(app.exec()) |