ADBFileExplorer
ADBFileExplorer copied to clipboard
modified main.py file to enable select feature
`# ADB File Explorer
Copyright (C) 2
file_selection_action = QAction('Select Multiple Files', self) file_selection_action.triggered.connect(self.select_multiple_files) self.file_menu.addAction(file_selection_action) 022 Azat Aldeshov import sys from typing import Any
from PyQt5 import QtCore, QtGui
from PyQt5.QtCore import Qt, QPoint, QModelIndex, QAbstractListModel, QVariant, QRect, QSize, QEvent, QObject
from PyQt5.QtGui import QPixmap, QColor, QPalette, QMovie, QKeySequence
from PyQt5.QtWidgets import QMenu, QAction, QMessageBox, QFileDialog, QStyle, QWidget, QStyledItemDelegate,
QStyleOptionViewItem, QApplication, QListView, QVBoxLayout, QLabel, QSizePolicy, QHBoxLayout, QTextEdit,
QMainWindow
from app.core.configurations import Resources from app.core.main import Adb from app.core.managers import Global from app.data.models import FileType, MessageData, MessageType from app.data.repositories import FileRepository from app.gui.explorer.toolbar import ParentButton, UploadTools, PathBar from app.helpers.tools import AsyncRepositoryWorker, ProgressCallbackHelper, read_string_from_file
class FileHeaderWidget(QWidget): def init(self, parent=None): super(FileHeaderWidget, self).init(parent) self.setLayout(QHBoxLayout(self)) policy = QSizePolicy(QSizePolicy.Ignored, QSizePolicy.Preferred)
self.file = QLabel('File', self)
self.file.setContentsMargins(45, 0, 0, 0)
policy.setHorizontalStretch(39)
self.file.setSizePolicy(policy)
self.layout().addWidget(self.file)
self.permissions = QLabel('Permissions', self)
self.permissions.setAlignment(Qt.AlignCenter)
policy.setHorizontalStretch(18)
self.permissions.setSizePolicy(policy)
self.layout().addWidget(self.permissions)
self.size = QLabel('Size', self)
self.size.setAlignment(Qt.AlignCenter)
policy.setHorizontalStretch(21)
self.size.setSizePolicy(policy)
self.layout().addWidget(self.size)
self.date = QLabel('Date', self)
self.date.setAlignment(Qt.AlignCenter)
policy.setHorizontalStretch(22)
self.date.setSizePolicy(policy)
self.layout().addWidget(self.date)
self.setStyleSheet("QWidget { background-color: #E5E5E5; font-weight: 500; border: 1px solid #C0C0C0 }")
class FileExplorerToolbar(QWidget): def init(self, parent=None): super(FileExplorerToolbar, self).init(parent) self.setLayout(QHBoxLayout(self)) policy = QSizePolicy(QSizePolicy.Ignored, QSizePolicy.Preferred) policy.setHorizontalStretch(1)
self.upload_tools = UploadTools(self)
self.upload_tools.setSizePolicy(policy)
self.layout().addWidget(self.upload_tools)
self.parent_button = ParentButton(self)
self.parent_button.setSizePolicy(policy)
self.layout().addWidget(self.parent_button)
self.path_bar = PathBar(self)
policy.setHorizontalStretch(8)
self.path_bar.setSizePolicy(policy)
self.layout().addWidget(self.path_bar)
class FileItemDelegate(QStyledItemDelegate): def sizeHint(self, option: 'QStyleOptionViewItem', index: QtCore.QModelIndex) -> QtCore.QSize: result = super(FileItemDelegate, self).sizeHint(option, index) result.setHeight(40) return result
def setEditorData(self, editor: QWidget, index: QtCore.QModelIndex):
editor.setText(index.model().data(index, Qt.EditRole))
def updateEditorGeometry(self, editor: QWidget, option: 'QStyleOptionViewItem', index: QtCore.QModelIndex):
editor.setGeometry(
option.rect.left() + 48, option.rect.top(), int(option.rect.width() / 2.5) - 55, option.rect.height()
)
def setModelData(self, editor: QWidget, model: QtCore.QAbstractItemModel, index: QtCore.QModelIndex):
model.setData(index, editor.text(), Qt.EditRole)
@staticmethod
def paint_line(painter: QtGui.QPainter, color: QColor, x, y, w, h):
painter.setPen(color)
painter.drawLine(x, y, w, h)
@staticmethod
def paint_text(painter: QtGui.QPainter, text: str, color: QColor, options, x, y, w, h):
painter.setPen(color)
painter.drawText(QRect(x, y, w, h), options, text)
def paint(self, painter: QtGui.QPainter, option: 'QStyleOptionViewItem', index: QtCore.QModelIndex):
if not index.data():
return super(FileItemDelegate, self).paint(painter, option, index)
self.initStyleOption(option, index)
style = option.widget.style() if option.widget else QApplication.style()
style.drawControl(QStyle.CE_ItemViewItem, option, painter, option.widget)
line_color = QColor("#CCCCCC")
text_color = option.palette.color(QPalette.Normal, QPalette.Text)
top = option.rect.top()
bottom = option.rect.height()
first_start = option.rect.left() + 50
second_start = option.rect.left() + int(option.rect.width() / 2.5)
third_start = option.rect.left() + int(option.rect.width() / 1.75)
fourth_start = option.rect.left() + int(option.rect.width() / 1.25)
end = option.rect.width() + option.rect.left()
self.paint_text(
painter, index.data().name, text_color, option.displayAlignment,
first_start, top, second_start - first_start - 4, bottom
)
self.paint_line(painter, line_color, second_start - 2, top, second_start - 1, bottom)
self.paint_text(
painter, index.data().permissions, text_color, Qt.AlignCenter | option.displayAlignment,
second_start, top, third_start - second_start - 4, bottom
)
self.paint_line(painter, line_color, third_start - 2, top, third_start - 1, bottom)
self.paint_text(
painter, index.data().size, text_color, Qt.AlignCenter | option.displayAlignment,
third_start, top, fourth_start - third_start - 4, bottom
)
self.paint_line(painter, line_color, fourth_start - 2, top, fourth_start - 1, bottom)
self.paint_text(
painter, index.data().date, text_color, Qt.AlignCenter | option.displayAlignment,
fourth_start, top, end - fourth_start, bottom
)
class FileListModel(QAbstractListModel): def init(self, parent=None): super().init(parent) self.items = []
def clear(self):
self.beginResetModel()
self.items.clear()
self.endResetModel()
def populate(self, files: list):
self.beginResetModel()
self.items.clear()
self.items = files
self.endResetModel()
def rowCount(self, parent: QModelIndex = ...) -> int:
return len(self.items)
def icon_path(self, index: QModelIndex = ...):
file_type = self.items[index.row()].type
if file_type == FileType.DIRECTORY:
return Resources.icon_folder
elif file_type == FileType.FILE:
return Resources.icon_file
elif file_type == FileType.LINK:
link_type = self.items[index.row()].link_type
if link_type == FileType.DIRECTORY:
return Resources.icon_link_folder
elif link_type == FileType.FILE:
return Resources.icon_link_file
return Resources.icon_link_file_unknown
return Resources.icon_file_unknown
def flags(self, index: QModelIndex) -> Qt.ItemFlags:
if not index.isValid():
return Qt.NoItemFlags
return Qt.ItemIsEditable | Qt.ItemIsEnabled | Qt.ItemIsSelectable
def setData(self, index: QModelIndex, value: Any, role: int = ...) -> bool:
if role == Qt.EditRole and value:
data, error = FileRepository.rename(self.items[index.row()], value)
if error:
Global().communicate.notification.emit(
MessageData(
timeout=10000,
title="Rename",
body="<span style='color: red; font-weight: 600'> %s </span>" % error,
)
)
Global.communicate.files__refresh.emit()
return super(FileListModel, self).setData(index, value, role)
def data(self, index: QModelIndex, role: int = ...) -> Any:
if not index.isValid():
return QVariant()
if role == Qt.DisplayRole:
return self.items[index.row()]
elif role == Qt.EditRole:
return self.items[index.row()].name
elif role == Qt.DecorationRole:
return QPixmap(self.icon_path(index)).scaled(32, 32, Qt.KeepAspectRatio)
return QVariant()
class FileExplorerWidget(QWidget): FILES_WORKER_ID = 300 DOWNLOAD_WORKER_ID = 399
def __init__(self, parent=None):
super(FileExplorerWidget, self).__init__(parent)
self.main_layout = QVBoxLayout(self)
self.toolbar = FileExplorerToolbar(self)
self.main_layout.addWidget(self.toolbar)
self.header = FileHeaderWidget(self)
self.main_layout.addWidget(self.header)
self.list = QListView(self)
self.model = FileListModel(self.list)
self.list.setSpacing(1)
self.list.setModel(self.model)
self.list.installEventFilter(self)
self.list.doubleClicked.connect(self.open)
self.list.setItemDelegate(FileItemDelegate(self.list))
self.list.setContextMenuPolicy(Qt.CustomContextMenu)
self.list.customContextMenuRequested.connect(self.context_menu)
self.list.setStyleSheet(read_string_from_file(Resources.style_file_list))
self.list.setSelectionMode(QListView.SelectionMode.ExtendedSelection)
self.layout().addWidget(self.list)
self.loading = QLabel(self)
self.loading.setAlignment(Qt.AlignCenter)
self.loading_movie = QMovie(Resources.anim_loading, parent=self.loading)
self.loading_movie.setScaledSize(QSize(48, 48))
self.loading.setMovie(self.loading_movie)
self.main_layout.addWidget(self.loading)
self.empty_label = QLabel("Folder is empty", self)
self.empty_label.setAlignment(Qt.AlignCenter)
self.empty_label.setStyleSheet("color: #969696; border: 1px solid #969696")
self.layout().addWidget(self.empty_label)
self.main_layout.setStretch(self.layout().count() - 1, 1)
self.main_layout.setStretch(self.layout().count() - 2, 1)
self.text_view_window = None
self.setLayout(self.main_layout)
Global().communicate.files__refresh.connect(self.update)
@property
def file(self):
if self.list and self.list.currentIndex():
return self.model.items[self.list.currentIndex().row()]
@property
def files(self):
if self.list and len(self.list.selectedIndexes()) > 0:
return map(lambda index: self.model.items[index.row()], self.list.selectedIndexes())
def update(self):
super(FileExplorerWidget, self).update()
worker = AsyncRepositoryWorker(
name="Files",
worker_id=self.FILES_WORKER_ID,
repository_method=FileRepository.files,
response_callback=self._async_response,
arguments=()
)
if Adb.worker().work(worker):
# First Setup loading view
self.model.clear()
self.list.setHidden(True)
self.loading.setHidden(False)
self.empty_label.setHidden(True)
self.loading_movie.start()
# Then start async worker
worker.start()
Global().communicate.path_toolbar__refresh.emit()
def close(self) -> bool:
Global().communicate.files__refresh.disconnect()
return super(FileExplorerWidget, self).close()
def _async_response(self, files: list, error: str):
self.loading_movie.stop()
self.loading.setHidden(True)
if error:
print(error, file=sys.stderr)
if not files:
Global().communicate.notification.emit(
MessageData(
title='Files',
timeout=15000,
body="<span style='color: red; font-weight: 600'> %s </span>" % error
)
)
if not files:
self.empty_label.setHidden(False)
else:
self.list.setHidden(False)
self.model.populate(files)
self.list.setFocus()
def eventFilter(self, obj: 'QObject', event: 'QEvent') -> bool:
if obj == self.list and \
event.type() == QEvent.KeyPress and \
event.matches(QKeySequence.InsertParagraphSeparator) and \
not self.list.isPersistentEditorOpen(self.list.currentIndex()):
self.open(self.list.currentIndex())
return super(FileExplorerWidget, self).eventFilter(obj, event)
def open(self, index: QModelIndex = ...):
if Adb.manager().open(self.model.items[index.row()]):
Global().communicate.files__refresh.emit()
def context_menu(self, pos: QPoint):
menu = QMenu()
menu.addSection("Actions")
action_copy = QAction('Copy to...', self)
action_copy.setDisabled(True)
menu.addAction(action_copy)
action_move = QAction('Move to...', self)
action_move.setDisabled(True)
menu.addAction(action_move)
action_rename = QAction('Rename', self)
action_rename.triggered.connect(self.rename)
menu.addAction(action_rename)
action_open_file = QAction('Open', self)
action_open_file.triggered.connect(self.open_file)
menu.addAction(action_open_file)
action_delete = QAction('Delete', self)
action_delete.triggered.connect(self.delete)
menu.addAction(action_delete)
action_download = QAction('Download', self)
action_download.triggered.connect(self.download_files)
menu.addAction(action_download)
action_download_to = QAction('Download to...', self)
action_download_to.triggered.connect(self.download_to)
menu.addAction(action_download_to)
menu.addSeparator()
action_properties = QAction('Properties', self)
action_properties.triggered.connect(self.file_properties)
menu.addAction(action_properties)
menu.exec(self.mapToGlobal(pos))
@staticmethod
def default_response(data, error):
if error:
Global().communicate.notification.emit(
MessageData(
title='Download error',
timeout=15000,
body="<span style='color: red; font-weight: 600'> %s </span>" % error
)
)
if data:
Global().communicate.notification.emit(
MessageData(
title='Downloaded',
timeout=15000,
body=data
)
)
def rename(self):
self.list.edit(self.list.currentIndex())
def open_file(self):
# QDesktopServices.openUrl(QUrl.fromLocalFile("downloaded_path")) open via external app
if not self.file.isdir:
data, error = FileRepository.open_file(self.file)
if error:
Global().communicate.notification.emit(
MessageData(
title='File',
timeout=15000,
body="<span style='color: red; font-weight: 600'> %s </span>" % error
)
)
else:
self.text_view_window = TextView(self.file.name, data)
self.text_view_window.show()
def delete(self):
file_names = ', '.join(map(lambda f: f.name, self.files))
reply = QMessageBox.critical(
self,
'Delete',
"Do you want to delete '%s'? It cannot be undone!" % file_names,
QMessageBox.Yes | QMessageBox.No, QMessageBox.No
)
if reply == QMessageBox.Yes:
for file in self.files:
data, error = FileRepository.delete(file)
if data:
Global().communicate.notification.emit(
MessageData(
timeout=10000,
title="Delete",
body=data,
)
)
if error:
Global().communicate.notification.emit(
MessageData(
timeout=10000,
title="Delete",
body="<span style='color: red; font-weight: 600'> %s </span>" % error,
)
)
Global.communicate.files__refresh.emit()
def download_to(self):
dir_name = QFileDialog.getExistingDirectory(self, 'Download to', '~')
if dir_name:
self.download_files(dir_name)
def download_files(self, destination: str = None):
for file in self.files:
helper = ProgressCallbackHelper()
worker = AsyncRepositoryWorker(
worker_id=self.DOWNLOAD_WORKER_ID,
name="Download",
repository_method=FileRepository.download,
response_callback=self.default_response,
arguments=(
helper.progress_callback.emit, file.path, destination
)
)
if Adb.worker().work(worker):
Global().communicate.notification.emit(
MessageData(
title="Downloading to",
message_type=MessageType.LOADING_MESSAGE,
message_catcher=worker.set_loading_widget
)
)
helper.setup(worker, worker.update_loading_widget)
worker.start()
def file_properties(self):
file, error = FileRepository.file(self.file.path)
file = file if file else self.file
if error:
Global().communicate.notification.emit(
MessageData(
timeout=10000,
title="Opening folder",
body="<span style='color: red; font-weight: 600'> %s </span>" % error,
)
)
info = "<br/><u><b>%s</b></u><br/>" % str(file)
info += "<pre>Name: %s</pre>" % file.name or '-'
info += "<pre>Owner: %s</pre>" % file.owner or '-'
info += "<pre>Group: %s</pre>" % file.group or '-'
info += "<pre>Size: %s</pre>" % file.raw_size or '-'
info += "<pre>Permissions: %s</pre>" % file.permissions or '-'
info += "<pre>Date: %s</pre>" % file.raw_date or '-'
info += "<pre>Type: %s</pre>" % file.type or '-'
if file.type == FileType.LINK:
info += "<pre>Links to: %s</pre>" % file.link or '-'
properties = QMessageBox(self)
properties.setStyleSheet("background-color: #DDDDDD")
properties.setIconPixmap(
QPixmap(self.model.icon_path(self.list.currentIndex())).scaled(128, 128, Qt.KeepAspectRatio)
)
properties.setWindowTitle('Properties')
properties.setInformativeText(info)
properties.exec_()
class TextView(QMainWindow): def init(self, filename, data): QMainWindow.init(self)
self.setMinimumSize(QSize(500, 300))
self.setWindowTitle(filename)
self.text_edit = QTextEdit(self)
self.setCentralWidget(self.text_edit)
self.text_edit.insertPlainText(data)
self.text_edit.move(10, 10)
def select_multiple_files(self): file_names, _ = QFileDialog.getOpenFileNames(self, 'Select Files') if file_names: self.handle_selected_files(file_names)
def handle_selected_files(self, file_names: list): # Implement your logic here # For now, let's just print the selected files print("Selected files:", file_names) ` modified_files.zip
enable multi-file selection, we'll need to implement the following steps:
- Add a Button or Menu Action: This will serve as the trigger for the user to initiate the multi-file selection.
- Open a Multi-File Selection Dialog: When the user clicks the button or menu action, a file selection dialog will open, allowing the user to select multiple files.
- Handle Selected Files: Once the user selects files and confirms, we'll need to process or handle these files based on the desired functionality (e.g., copying, moving, etc.).
Given the content of files.py
, I'll provide code snippets to achieve the above steps:
1. Add a Button or Menu Action:
You can add a button to the toolbar or a menu action. For simplicity, let's add a menu action:
file_selection_action = QAction('Select Multiple Files', self)
file_selection_action.triggered.connect(self.select_multiple_files)
self.file_menu.addAction(file_selection_action)
2. Open a Multi-File Selection Dialog:
We'll create a method called select_multiple_files
:
def select_multiple_files(self):
file_names, _ = QFileDialog.getOpenFileNames(self, 'Select Files')
if file_names:
self.handle_selected_files(file_names)
Here, QFileDialog.getOpenFileNames
returns a tuple where the first element is a list of selected file paths.
3. Handle Selected Files:
Finally, implement the handle_selected_files
method to process the selected files:
def handle_selected_files(self, file_names: list):
# Implement your logic here
# For now, let's just print the selected files
print("Selected files:", file_names)
Integrating these snippets into files.py
will enable multi-file selection in the GUI.
please check if this working or not