|
|
|
/*
|
|
|
|
* Copyright (C) 2012 - 2015 Hong Jen Yee (PCMan) <pcman.tw@gmail.com>
|
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library 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
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
* License along with this library; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
#include "folderview.h"
|
|
|
|
#include "foldermodel.h"
|
|
|
|
#include <QHeaderView>
|
|
|
|
#include <QVBoxLayout>
|
|
|
|
#include <QContextMenuEvent>
|
|
|
|
#include "proxyfoldermodel.h"
|
|
|
|
#include "folderitemdelegate.h"
|
|
|
|
#include "dndactionmenu.h"
|
|
|
|
#include "filemenu.h"
|
|
|
|
#include "foldermenu.h"
|
|
|
|
#include "filelauncher.h"
|
|
|
|
#include "utilities.h"
|
|
|
|
#include <QTimer>
|
|
|
|
#include <QDate>
|
|
|
|
#include <QDebug>
|
|
|
|
#include <QClipboard>
|
|
|
|
#include <QMimeData>
|
|
|
|
#include <QHoverEvent>
|
|
|
|
#include <QApplication>
|
|
|
|
#include <QScrollBar>
|
|
|
|
#include <QMetaType>
|
|
|
|
#include <QMessageBox>
|
|
|
|
#include <QLineEdit>
|
|
|
|
#include <QTextEdit>
|
|
|
|
#include <QX11Info> // for XDS support
|
|
|
|
#include <xcb/xcb.h> // for XDS support
|
|
|
|
#include "xdndworkaround.h" // for XDS support
|
|
|
|
#include "path.h"
|
|
|
|
#include "folderview_p.h"
|
|
|
|
#include "utilities.h"
|
|
|
|
|
|
|
|
Q_DECLARE_OPAQUE_POINTER(FmFileInfo*)
|
|
|
|
|
|
|
|
using namespace Fm;
|
|
|
|
|
|
|
|
FolderViewListView::FolderViewListView(QWidget* parent):
|
|
|
|
QListView(parent),
|
|
|
|
activationAllowed_(true) {
|
|
|
|
connect(this, &QListView::activated, this, &FolderViewListView::activation);
|
|
|
|
// inline renaming
|
|
|
|
setEditTriggers(QAbstractItemView::NoEditTriggers);
|
|
|
|
}
|
|
|
|
|
|
|
|
FolderViewListView::~FolderViewListView() {
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderViewListView::startDrag(Qt::DropActions supportedActions) {
|
|
|
|
if(movement() != Static) {
|
|
|
|
QListView::startDrag(supportedActions);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
QAbstractItemView::startDrag(supportedActions);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderViewListView::mousePressEvent(QMouseEvent* event) {
|
|
|
|
QListView::mousePressEvent(event);
|
|
|
|
static_cast<FolderView*>(parent())->childMousePressEvent(event);
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderViewListView::mouseMoveEvent(QMouseEvent* event) {
|
|
|
|
// NOTE: Filter the BACK & FORWARD buttons to not Drag & Drop with them.
|
|
|
|
// (by default Qt views drag with any button)
|
|
|
|
if (event->buttons() == Qt::NoButton || event->buttons() & ~(Qt::BackButton | Qt::ForwardButton))
|
|
|
|
QListView::mouseMoveEvent(event);
|
|
|
|
}
|
|
|
|
|
|
|
|
QModelIndex FolderViewListView::indexAt(const QPoint& point) const {
|
|
|
|
QModelIndex index = QListView::indexAt(point);
|
|
|
|
// NOTE: QListView has a severe design flaw here. It does hit-testing based on the
|
|
|
|
// total bound rect of the item. The width of an item is determined by max(icon_width, text_width).
|
|
|
|
// So if the text label is much wider than the icon, when you click outside the icon but
|
|
|
|
// the point is still within the outer bound rect, the item is still selected.
|
|
|
|
// This results in very poor usability. Let's do precise hit-testing here.
|
|
|
|
// An item is hit only when the point is in the icon or text label.
|
|
|
|
// If the point is in the bound rectangle but outside the icon or text, it should not be selected.
|
|
|
|
if(viewMode() == QListView::IconMode && index.isValid()) {
|
|
|
|
QRect visRect = visualRect(index); // visible area on the screen
|
|
|
|
FolderItemDelegate* delegate = static_cast<FolderItemDelegate*>(itemDelegateForColumn(FolderModel::ColumnFileName));
|
|
|
|
QSize margins = delegate->getMargins();
|
|
|
|
QSize _iconSize = iconSize();
|
|
|
|
if(point.y() < visRect.top() + margins.height()) { // above icon
|
|
|
|
return QModelIndex();
|
|
|
|
}
|
|
|
|
else if(point.y() < visRect.top() + margins.height() + _iconSize.height()) { // on the icon area
|
|
|
|
int iconXMargin = (visRect.width() - _iconSize.width()) / 2;
|
|
|
|
if(point.x() < (visRect.left() + iconXMargin) || point.x() > (visRect.right() + 1 - iconXMargin)) {
|
|
|
|
// to the left or right of the icon
|
|
|
|
return QModelIndex();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
QSize _textSize = delegate->iconViewTextSize(index);
|
|
|
|
int textHMargin = (visRect.width() - _textSize.width()) / 2;
|
|
|
|
if(point.y() > visRect.top() + margins.height() + _iconSize.height() + _textSize.height() // below text
|
|
|
|
// on the text area but to the left or right of the text
|
|
|
|
|| point.x() < visRect.left() + textHMargin || point.x() > visRect.right() + 1 - textHMargin) {
|
|
|
|
return QModelIndex();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// qDebug() << "visualRect: " << visRect << "point:" << point;
|
|
|
|
}
|
|
|
|
return index;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// NOTE:
|
|
|
|
// QListView has a problem which I consider a bug or a design flaw.
|
|
|
|
// When you set movement property to Static, theoratically the icons
|
|
|
|
// should not be movable. However, if you turned on icon mode,
|
|
|
|
// the icons becomes freely movable despite the value of movement is Static.
|
|
|
|
// To overcome this bug, we override all drag handling methods, and
|
|
|
|
// call QAbstractItemView directly, bypassing QListView.
|
|
|
|
// In this way, we can workaround the buggy behavior.
|
|
|
|
// The drag handlers of QListView basically does the same things
|
|
|
|
// as its parent QAbstractItemView, but it also stores the currently
|
|
|
|
// dragged item and paint them in the view as needed.
|
|
|
|
// TODO: I really should file a bug report to Qt developers.
|
|
|
|
|
|
|
|
void FolderViewListView::dragEnterEvent(QDragEnterEvent* event) {
|
|
|
|
if(movement() != Static) {
|
|
|
|
QListView::dragEnterEvent(event);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
QAbstractItemView::dragEnterEvent(event);
|
|
|
|
}
|
|
|
|
qDebug("dragEnterEvent");
|
|
|
|
//static_cast<FolderView*>(parent())->childDragEnterEvent(event);
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderViewListView::dragLeaveEvent(QDragLeaveEvent* e) {
|
|
|
|
if(movement() != Static) {
|
|
|
|
QListView::dragLeaveEvent(e);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
QAbstractItemView::dragLeaveEvent(e);
|
|
|
|
}
|
|
|
|
static_cast<FolderView*>(parent())->childDragLeaveEvent(e);
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderViewListView::dragMoveEvent(QDragMoveEvent* e) {
|
|
|
|
if(movement() != Static) {
|
|
|
|
QListView::dragMoveEvent(e);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
QAbstractItemView::dragMoveEvent(e);
|
|
|
|
}
|
|
|
|
static_cast<FolderView*>(parent())->childDragMoveEvent(e);
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderViewListView::dropEvent(QDropEvent* e) {
|
|
|
|
|
|
|
|
static_cast<FolderView*>(parent())->childDropEvent(e);
|
|
|
|
|
|
|
|
if(movement() != Static) {
|
|
|
|
QListView::dropEvent(e);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
QAbstractItemView::dropEvent(e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderViewListView::mouseReleaseEvent(QMouseEvent* event) {
|
|
|
|
bool activationWasAllowed = activationAllowed_;
|
|
|
|
if((!style()->styleHint(QStyle::SH_ItemView_ActivateItemOnSingleClick, nullptr, this)) || (event->button() != Qt::LeftButton)) {
|
|
|
|
activationAllowed_ = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
QListView::mouseReleaseEvent(event);
|
|
|
|
|
|
|
|
activationAllowed_ = activationWasAllowed;
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderViewListView::mouseDoubleClickEvent(QMouseEvent* event) {
|
|
|
|
bool activationWasAllowed = activationAllowed_;
|
|
|
|
if((style()->styleHint(QStyle::SH_ItemView_ActivateItemOnSingleClick, nullptr, this)) || (event->button() != Qt::LeftButton)) {
|
|
|
|
activationAllowed_ = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
QListView::mouseDoubleClickEvent(event);
|
|
|
|
|
|
|
|
activationAllowed_ = activationWasAllowed;
|
|
|
|
}
|
|
|
|
|
|
|
|
QModelIndex FolderViewListView::moveCursor(CursorAction cursorAction, Qt::KeyboardModifiers modifiers) {
|
|
|
|
QAbstractItemModel* model_ = model();
|
|
|
|
|
|
|
|
if(model_ && currentIndex().isValid()) {
|
|
|
|
FolderView::ViewMode viewMode = static_cast<FolderView*>(parent())->viewMode();
|
|
|
|
if((viewMode == FolderView::IconMode) || (viewMode == FolderView::ThumbnailMode)) {
|
|
|
|
int next = (layoutDirection() == Qt::RightToLeft) ? - 1 : 1;
|
|
|
|
|
|
|
|
if(cursorAction == QAbstractItemView::MoveRight) {
|
|
|
|
return model_->index(currentIndex().row() + next, 0);
|
|
|
|
}
|
|
|
|
else if(cursorAction == QAbstractItemView::MoveLeft) {
|
|
|
|
return model_->index(currentIndex().row() - next, 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return QListView::moveCursor(cursorAction, modifiers);
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderViewListView::activation(const QModelIndex& index) {
|
|
|
|
if(activationAllowed_) {
|
|
|
|
Q_EMIT activatedFiltered(index);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
|
|
|
FolderViewTreeView::FolderViewTreeView(QWidget* parent):
|
|
|
|
QTreeView(parent),
|
|
|
|
doingLayout_(false),
|
|
|
|
layoutTimer_(nullptr),
|
|
|
|
activationAllowed_(true) {
|
|
|
|
|
|
|
|
header()->setStretchLastSection(true);
|
|
|
|
setIndentation(0);
|
|
|
|
/* the default true value may cause a crash on entering a folder
|
|
|
|
by double clicking because of the viewport update done by
|
|
|
|
QTreeView::mouseDoubleClickEvent() (a Qt bug?) */
|
|
|
|
setExpandsOnDoubleClick(false);
|
|
|
|
|
|
|
|
connect(this, &QTreeView::activated, this, &FolderViewTreeView::activation);
|
|
|
|
// don't open editor on double clicking
|
|
|
|
setEditTriggers(QAbstractItemView::NoEditTriggers);
|
|
|
|
}
|
|
|
|
|
|
|
|
FolderViewTreeView::~FolderViewTreeView() {
|
|
|
|
if(layoutTimer_) {
|
|
|
|
delete layoutTimer_;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderViewTreeView::setModel(QAbstractItemModel* model) {
|
|
|
|
QTreeView::setModel(model);
|
|
|
|
layoutColumns();
|
|
|
|
if(ProxyFolderModel* proxyModel = qobject_cast<ProxyFolderModel*>(model)) {
|
|
|
|
connect(proxyModel, &ProxyFolderModel::sortFilterChanged, this, &FolderViewTreeView::onSortFilterChanged,
|
|
|
|
Qt::UniqueConnection);
|
|
|
|
onSortFilterChanged();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderViewTreeView::mousePressEvent(QMouseEvent* event) {
|
|
|
|
QTreeView::mousePressEvent(event);
|
|
|
|
static_cast<FolderView*>(parent())->childMousePressEvent(event);
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderViewTreeView::mouseMoveEvent(QMouseEvent* event) {
|
|
|
|
// NOTE: Filter the BACK & FORWARD buttons to not Drag & Drop with them.
|
|
|
|
// (by default Qt views drag with any button)
|
|
|
|
if (event->buttons() == Qt::NoButton || event->buttons() & ~(Qt::BackButton | Qt::ForwardButton))
|
|
|
|
QTreeView::mouseMoveEvent(event);
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderViewTreeView::dragEnterEvent(QDragEnterEvent* event) {
|
|
|
|
QTreeView::dragEnterEvent(event);
|
|
|
|
//static_cast<FolderView*>(parent())->childDragEnterEvent(event);
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderViewTreeView::dragLeaveEvent(QDragLeaveEvent* e) {
|
|
|
|
QTreeView::dragLeaveEvent(e);
|
|
|
|
static_cast<FolderView*>(parent())->childDragLeaveEvent(e);
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderViewTreeView::dragMoveEvent(QDragMoveEvent* e) {
|
|
|
|
QTreeView::dragMoveEvent(e);
|
|
|
|
static_cast<FolderView*>(parent())->childDragMoveEvent(e);
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderViewTreeView::dropEvent(QDropEvent* e) {
|
|
|
|
static_cast<FolderView*>(parent())->childDropEvent(e);
|
|
|
|
QTreeView::dropEvent(e);
|
|
|
|
}
|
|
|
|
|
|
|
|
// the default list mode of QListView handles column widths
|
|
|
|
// very badly (worse than gtk+) and it's not very flexible.
|
|
|
|
// so, let's handle column widths outselves.
|
|
|
|
void FolderViewTreeView::layoutColumns() {
|
|
|
|
// qDebug("layoutColumns");
|
|
|
|
if(!model()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
doingLayout_ = true;
|
|
|
|
QHeaderView* headerView = header();
|
|
|
|
// the width that's available for showing the columns.
|
|
|
|
int availWidth = viewport()->contentsRect().width();
|
|
|
|
|
|
|
|
// get the width that every column want
|
|
|
|
int numCols = headerView->count();
|
|
|
|
if(numCols > 0) {
|
|
|
|
int desiredWidth = 0;
|
|
|
|
int* widths = new int[numCols]; // array to store the widths every column needs
|
|
|
|
QStyleOptionHeader opt;
|
|
|
|
opt.initFrom(headerView);
|
|
|
|
opt.fontMetrics = QFontMetrics(font());
|
|
|
|
if (headerView->isSortIndicatorShown()) {
|
|
|
|
opt.sortIndicator = QStyleOptionHeader::SortDown;
|
|
|
|
}
|
|
|
|
QAbstractItemModel* model_ = model();
|
|
|
|
int column;
|
|
|
|
for(column = 0; column < numCols; ++column) {
|
|
|
|
int columnId = headerView->logicalIndex(column);
|
|
|
|
// get the size that the column needs
|
|
|
|
if(model_) {
|
|
|
|
QVariant data = model_->headerData(columnId, Qt::Horizontal, Qt::DisplayRole);
|
|
|
|
if(data.isValid()) {
|
|
|
|
opt.text = data.isValid() ? data.toString() : QString();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
opt.section = columnId;
|
|
|
|
widths[column] = qMax(sizeHintForColumn(columnId),
|
|
|
|
style()->sizeFromContents(QStyle::CT_HeaderSection, &opt, QSize(), headerView).width());
|
|
|
|
// compute the total width needed
|
|
|
|
desiredWidth += widths[column];
|
|
|
|
}
|
|
|
|
|
|
|
|
int filenameColumn = headerView->visualIndex(FolderModel::ColumnFileName);
|
|
|
|
// if the total witdh we want exceeds the available space
|
|
|
|
if(desiredWidth > availWidth) {
|
|
|
|
// Compute the width available for the filename column
|
|
|
|
int filenameAvailWidth = availWidth - desiredWidth + widths[filenameColumn];
|
|
|
|
|
|
|
|
// Compute the minimum acceptable width for the filename column
|
|
|
|
int filenameMinWidth = qMin(200, sizeHintForColumn(filenameColumn));
|
|
|
|
|
|
|
|
if(filenameAvailWidth > filenameMinWidth) {
|
|
|
|
// Shrink the filename column to the available width
|
|
|
|
widths[filenameColumn] = filenameAvailWidth;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// Set the filename column to its minimum width
|
|
|
|
widths[filenameColumn] = filenameMinWidth;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// Fill the extra available space with the filename column
|
|
|
|
widths[filenameColumn] += availWidth - desiredWidth;
|
|
|
|
}
|
|
|
|
|
|
|
|
// really do the resizing for every column
|
|
|
|
for(int column = 0; column < numCols; ++column) {
|
|
|
|
headerView->resizeSection(headerView->logicalIndex(column), widths[column]);
|
|
|
|
}
|
|
|
|
delete []widths;
|
|
|
|
}
|
|
|
|
doingLayout_ = false;
|
|
|
|
|
|
|
|
if(layoutTimer_) {
|
|
|
|
delete layoutTimer_;
|
|
|
|
layoutTimer_ = nullptr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderViewTreeView::resizeEvent(QResizeEvent* event) {
|
|
|
|
QAbstractItemView::resizeEvent(event);
|
|
|
|
// prevent endless recursion.
|
|
|
|
// When manually resizing columns, at the point where a horizontal scroll
|
|
|
|
// bar has to be inserted or removed, the vertical size changes, a resize
|
|
|
|
// event occurs and the column headers are flickering badly if the column
|
|
|
|
// layout is modified at this point. Therefore only layout the columns if
|
|
|
|
// the horizontal size changes.
|
|
|
|
if(!doingLayout_ && event->size().width() != event->oldSize().width()) {
|
|
|
|
layoutColumns(); // layoutColumns() also triggers resizeEvent
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderViewTreeView::rowsInserted(const QModelIndex& parent, int start, int end) {
|
|
|
|
QTreeView::rowsInserted(parent, start, end);
|
|
|
|
queueLayoutColumns();
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderViewTreeView::rowsAboutToBeRemoved(const QModelIndex& parent, int start, int end) {
|
|
|
|
QTreeView::rowsAboutToBeRemoved(parent, start, end);
|
|
|
|
queueLayoutColumns();
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderViewTreeView::dataChanged(const QModelIndex& topLeft, const QModelIndex& bottomRight, const QVector<int>& roles /*= QVector<int>{}*/) {
|
|
|
|
QTreeView::dataChanged(topLeft, bottomRight, roles);
|
|
|
|
// FIXME: this will be very inefficient
|
|
|
|
// queueLayoutColumns();
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderViewTreeView::reset() {
|
|
|
|
// Sometimes when the content of the model is radically changed, Qt does reset()
|
|
|
|
// on the model rather than doing large amount of insertion and deletion.
|
|
|
|
// This is for performance reason so in this case rowsInserted() and rowsAboutToBeRemoved()
|
|
|
|
// might not be called. Hence we also have to re-layout the columns when the model is reset.
|
|
|
|
// This fixes bug #190
|
|
|
|
// https://github.com/lxde/pcmanfm-qt/issues/190
|
|
|
|
QTreeView::reset();
|
|
|
|
queueLayoutColumns();
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderViewTreeView::queueLayoutColumns() {
|
|
|
|
// qDebug("queueLayoutColumns");
|
|
|
|
if(!layoutTimer_) {
|
|
|
|
layoutTimer_ = new QTimer();
|
|
|
|
layoutTimer_->setSingleShot(true);
|
|
|
|
layoutTimer_->setInterval(0);
|
|
|
|
connect(layoutTimer_, &QTimer::timeout, this, &FolderViewTreeView::layoutColumns);
|
|
|
|
}
|
|
|
|
layoutTimer_->start();
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderViewTreeView::mouseReleaseEvent(QMouseEvent* event) {
|
|
|
|
bool activationWasAllowed = activationAllowed_;
|
|
|
|
if((!style()->styleHint(QStyle::SH_ItemView_ActivateItemOnSingleClick, nullptr, this)) || (event->button() != Qt::LeftButton)) {
|
|
|
|
activationAllowed_ = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
QTreeView::mouseReleaseEvent(event);
|
|
|
|
|
|
|
|
activationAllowed_ = activationWasAllowed;
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderViewTreeView::mouseDoubleClickEvent(QMouseEvent* event) {
|
|
|
|
bool activationWasAllowed = activationAllowed_;
|
|
|
|
if((style()->styleHint(QStyle::SH_ItemView_ActivateItemOnSingleClick, nullptr, this)) || (event->button() != Qt::LeftButton)) {
|
|
|
|
activationAllowed_ = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
QTreeView::mouseDoubleClickEvent(event);
|
|
|
|
|
|
|
|
activationAllowed_ = activationWasAllowed;
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderViewTreeView::activation(const QModelIndex& index) {
|
|
|
|
if(activationAllowed_) {
|
|
|
|
Q_EMIT activatedFiltered(index);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderViewTreeView::onSortFilterChanged() {
|
|
|
|
if(QSortFilterProxyModel* proxyModel = qobject_cast<QSortFilterProxyModel*>(model())) {
|
|
|
|
header()->setSortIndicatorShown(true);
|
|
|
|
header()->setSortIndicator(proxyModel->sortColumn(), proxyModel->sortOrder());
|
|
|
|
if(!isSortingEnabled()) {
|
|
|
|
setSortingEnabled(true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
|
|
|
FolderView::FolderView(FolderView::ViewMode _mode, QWidget *parent):
|
|
|
|
QWidget(parent),
|
|
|
|
view(nullptr),
|
|
|
|
model_(nullptr),
|
|
|
|
mode((ViewMode)0),
|
|
|
|
fileLauncher_(nullptr),
|
|
|
|
autoSelectionDelay_(600),
|
|
|
|
autoSelectionTimer_(nullptr),
|
|
|
|
selChangedTimer_(nullptr),
|
|
|
|
itemDelegateMargins_(QSize(3, 3)) {
|
|
|
|
|
|
|
|
iconSize_[IconMode - FirstViewMode] = QSize(48, 48);
|
|
|
|
iconSize_[CompactMode - FirstViewMode] = QSize(24, 24);
|
|
|
|
iconSize_[ThumbnailMode - FirstViewMode] = QSize(128, 128);
|
|
|
|
iconSize_[DetailedListMode - FirstViewMode] = QSize(24, 24);
|
|
|
|
|
|
|
|
QVBoxLayout* layout = new QVBoxLayout();
|
|
|
|
layout->setMargin(0);
|
|
|
|
setLayout(layout);
|
|
|
|
|
|
|
|
setViewMode(_mode);
|
|
|
|
setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding);
|
|
|
|
|
|
|
|
connect(this, &FolderView::clicked, this, &FolderView::onFileClicked);
|
|
|
|
connect(QApplication::clipboard(), &QClipboard::dataChanged, this, &FolderView::onClipboardDataChange);
|
|
|
|
}
|
|
|
|
|
|
|
|
FolderView::~FolderView() {
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderView::onItemActivated(QModelIndex index) {
|
|
|
|
if(index.isValid() && index.model()) {
|
|
|
|
QVariant data = index.model()->data(index, FolderModel::FileInfoRole);
|
|
|
|
auto info = data.value<std::shared_ptr<const Fm::FileInfo>>();
|
|
|
|
if(info) {
|
|
|
|
if(!(QApplication::keyboardModifiers() & (Qt::ShiftModifier | Qt::ControlModifier | Qt::AltModifier | Qt::MetaModifier))) {
|
|
|
|
Q_EMIT clicked(ActivatedClick, info);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderView::onSelChangedTimeout() {
|
|
|
|
selChangedTimer_->deleteLater();
|
|
|
|
selChangedTimer_ = nullptr;
|
|
|
|
// qDebug()<<"selected:" << nSel;
|
|
|
|
Q_EMIT selChanged();
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderView::onSelectionChanged(const QItemSelection& /*selected*/, const QItemSelection& /*deselected*/) {
|
|
|
|
// It's possible that the selected items change too often and this slot gets called for thousands of times.
|
|
|
|
// For example, when you select thousands of files and delete them, we will get one selectionChanged() event
|
|
|
|
// for every deleted file. So, we use a timer to delay the handling to avoid too frequent updates of the UI.
|
|
|
|
if(!selChangedTimer_) {
|
|
|
|
selChangedTimer_ = new QTimer(this);
|
|
|
|
selChangedTimer_->setSingleShot(true);
|
|
|
|
connect(selChangedTimer_, &QTimer::timeout, this, &FolderView::onSelChangedTimeout);
|
|
|
|
selChangedTimer_->start(200);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderView::onClosingEditor(QWidget* editor, QAbstractItemDelegate::EndEditHint hint) {
|
|
|
|
if (hint != QAbstractItemDelegate::NoHint) {
|
|
|
|
// we set the hint to NoHint in FolderItemDelegate::eventFilter()
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
QString newName;
|
|
|
|
if (qobject_cast<QTextEdit*>(editor)) { // icon and thumbnail view
|
|
|
|
newName = qobject_cast<QTextEdit*>(editor)->toPlainText();
|
|
|
|
}
|
|
|
|
else if (qobject_cast<QLineEdit*>(editor)) { // compact view
|
|
|
|
newName = qobject_cast<QLineEdit*>(editor)->text();
|
|
|
|
}
|
|
|
|
if (newName.isEmpty()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
// the editor will be deleted by QAbstractItemDelegate::destroyEditor() when no longer needed
|
|
|
|
|
|
|
|
QModelIndex index = view->selectionModel()->currentIndex();
|
|
|
|
if(index.isValid() && index.model()) {
|
|
|
|
QVariant data = index.model()->data(index, FolderModel::FileInfoRole);
|
|
|
|
auto info = data.value<std::shared_ptr<const Fm::FileInfo>>();
|
|
|
|
if (info) {
|
|
|
|
auto oldName = QString::fromStdString(info->name());
|
|
|
|
if(newName == oldName) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
QWidget* parent = window();
|
|
|
|
if (window() == this) { // supposedly desktop, in case it uses this
|
|
|
|
parent = nullptr;
|
|
|
|
}
|
|
|
|
changeFileName(info->path(), newName, parent);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderView::setViewMode(ViewMode _mode) {
|
|
|
|
if(_mode == mode) { // if it's the same more, ignore
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
// FIXME: retain old selection
|
|
|
|
|
|
|
|
// since only detailed list mode uses QTreeView, and others
|
|
|
|
// all use QListView, it's wise to preserve QListView when possible.
|
|
|
|
bool recreateView = false;
|
|
|
|
if(view && (mode == DetailedListMode || _mode == DetailedListMode)) {
|
|
|
|
delete view; // FIXME: no virtual dtor?
|
|
|
|
view = nullptr;
|
|
|
|
recreateView = true;
|
|
|
|
}
|
|
|
|
mode = _mode;
|
|
|
|
QSize iconSize = iconSize_[mode - FirstViewMode];
|
|
|
|
|
|
|
|
FolderItemDelegate* delegate = nullptr;
|
|
|
|
if(mode == DetailedListMode) {
|
|
|
|
FolderViewTreeView* treeView = new FolderViewTreeView(this);
|
|
|
|
connect(treeView, &FolderViewTreeView::activatedFiltered, this, &FolderView::onItemActivated);
|
|
|
|
setFocusProxy(treeView);
|
|
|
|
|
|
|
|
view = treeView;
|
|
|
|
treeView->setItemsExpandable(false);
|
|
|
|
treeView->setRootIsDecorated(false);
|
|
|
|
treeView->setAllColumnsShowFocus(false);
|
|
|
|
|
|
|
|
// set our own custom delegate
|
|
|
|
delegate = new FolderItemDelegate(treeView);
|
|
|
|
treeView->setItemDelegateForColumn(FolderModel::ColumnFileName, delegate);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
FolderViewListView* listView;
|
|
|
|
if(view) {
|
|
|
|
listView = static_cast<FolderViewListView*>(view);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
listView = new FolderViewListView(this);
|
|
|
|
connect(listView, &FolderViewListView::activatedFiltered, this, &FolderView::onItemActivated);
|
|
|
|
view = listView;
|
|
|
|
}
|
|
|
|
setFocusProxy(listView);
|
|
|
|
|
|
|
|
// set our own custom delegate
|
|
|
|
delegate = new FolderItemDelegate(listView);
|
|
|
|
listView->setItemDelegateForColumn(FolderModel::ColumnFileName, delegate);
|
|
|
|
// FIXME: should we expose the delegate?
|
|
|
|
listView->setMovement(QListView::Static);
|
|
|
|
/* If listView is already visible, setMovement() will lay out items again with delay
|
|
|
|
(see Qt, QListView::setMovement(), d->doDelayedItemsLayout()) and thus drop events
|
|
|
|
will remain disabled for the viewport. So, we should re-enable drop events here. */
|
|
|
|
if(listView->viewport()->isVisible()) {
|
|
|
|
listView->viewport()->setAcceptDrops(true);
|
|
|
|
}
|
|
|
|
listView->setResizeMode(QListView::Adjust);
|
|
|
|
listView->setWrapping(true);
|
|
|
|
switch(mode) {
|
|
|
|
case IconMode: {
|
|
|
|
listView->setViewMode(QListView::IconMode);
|
|
|
|
listView->setWordWrap(true);
|
|
|
|
listView->setFlow(QListView::LeftToRight);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case CompactMode: {
|
|
|
|
listView->setViewMode(QListView::ListMode);
|
|
|
|
listView->setWordWrap(false);
|
|
|
|
listView->setFlow(QListView::QListView::TopToBottom);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case ThumbnailMode: {
|
|
|
|
listView->setViewMode(QListView::IconMode);
|
|
|
|
listView->setWordWrap(true);
|
|
|
|
listView->setFlow(QListView::LeftToRight);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
;
|
|
|
|
}
|
|
|
|
updateGridSize();
|
|
|
|
}
|
|
|
|
if(view) {
|
|
|
|
// we have to install the event filter on the viewport instead of the view itself.
|
|
|
|
view->viewport()->installEventFilter(this);
|
|
|
|
// we want the QEvent::HoverMove event for single click + auto-selection support
|
|
|
|
view->viewport()->setAttribute(Qt::WA_Hover, true);
|
|
|
|
view->setContextMenuPolicy(Qt::NoContextMenu); // defer the context menu handling to parent widgets
|
|
|
|
view->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding);
|
|
|
|
view->setIconSize(iconSize);
|
|
|
|
|
|
|
|
view->setSelectionMode(QAbstractItemView::ExtendedSelection);
|
|
|
|
layout()->addWidget(view);
|
|
|
|
|
|
|
|
// enable dnd
|
|
|
|
view->setDragEnabled(true);
|
|
|
|
view->setAcceptDrops(true);
|
|
|
|
view->setDragDropMode(QAbstractItemView::DragDrop);
|
|
|
|
view->setDropIndicatorShown(true);
|
|
|
|
|
|
|
|
// inline renaming
|
|
|
|
if(delegate) {
|
|
|
|
connect(delegate, &QAbstractItemDelegate::closeEditor, this, &FolderView::onClosingEditor);
|
|
|
|
}
|
|
|
|
|
|
|
|
if(model_) {
|
|
|
|
// FIXME: preserve selections
|
|
|
|
model_->setThumbnailSize(iconSize.width());
|
|
|
|
view->setModel(model_);
|
|
|
|
if(recreateView) {
|
|
|
|
connect(view->selectionModel(), &QItemSelectionModel::selectionChanged, this, &FolderView::onSelectionChanged);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// set proper grid size for the QListView based on current view mode, icon size, and font size.
|
|
|
|
void FolderView::updateGridSize() {
|
|
|
|
if(mode == DetailedListMode || !view) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
FolderViewListView* listView = static_cast<FolderViewListView*>(view);
|
|
|
|
QSize icon = iconSize(mode); // size of the icon
|
|
|
|
QFontMetrics fm = fontMetrics(); // size of current font
|
|
|
|
QSize grid; // the final grid size
|
|
|
|
switch(mode) {
|
|
|
|
case IconMode:
|
|
|
|
case ThumbnailMode: {
|
|
|
|
// NOTE by PCMan about finding the optimal text label size:
|
|
|
|
// The average filename length on my root filesystem is roughly 18-20 chars.
|
|
|
|
// So, a reasonable size for the text label is about 10 chars each line since string of this length
|
|
|
|
// can be shown in two lines. If you consider word wrap, then the result is around 10 chars per word.
|
|
|
|
// In average, 10 char per line should be enough to display a "word" in the filename without breaking.
|
|
|
|
// The values can be estimated with this command:
|
|
|
|
// > find / | xargs basename -a | sed -e s'/[_-]/ /g' | wc -mcw
|
|
|
|
// However, this average only applies to English. For some Asian characters, such as Chinese chars,
|
|
|
|
// each char actually takes doubled space. To be safe, we use 13 chars per line x average char width
|
|
|
|
// to get a nearly optimal width for the text label. As most of the filenames have less than 40 chars
|
|
|
|
// 13 chars x 3 lines should be enough to show the full filenames for most files.
|
|
|
|
int textWidth = fm.averageCharWidth() * 13;
|
|
|
|
int textHeight = fm.lineSpacing() * 3;
|
|
|
|
grid.setWidth(qMax(icon.width(), textWidth) + 4); // a margin of 2 px for selection rects
|
|
|
|
grid.setHeight(icon.height() + textHeight + 4); // a margin of 2 px for selection rects
|
|
|
|
// grow to include margins
|
|
|
|
grid += 2*itemDelegateMargins_;
|
|
|
|
// let horizontal and vertical spacings be set only by itemDelegateMargins_
|
|
|
|
listView->setSpacing(0);
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
// FIXME: set proper item size
|
|
|
|
listView->setSpacing(2);
|
|
|
|
; // do not use grid size
|
|
|
|
}
|
|
|
|
|
|
|
|
FolderItemDelegate* delegate = static_cast<FolderItemDelegate*>(listView->itemDelegateForColumn(FolderModel::ColumnFileName));
|
|
|
|
delegate->setItemSize(grid);
|
|
|
|
delegate->setIconSize(icon);
|
|
|
|
delegate->setMargins(itemDelegateMargins_);
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderView::setIconSize(ViewMode mode, QSize size) {
|
|
|
|
Q_ASSERT(mode >= FirstViewMode && mode <= LastViewMode);
|
|
|
|
iconSize_[mode - FirstViewMode] = size;
|
|
|
|
if(viewMode() == mode) {
|
|
|
|
view->setIconSize(size);
|
|
|
|
if(model_) {
|
|
|
|
model_->setThumbnailSize(size.width());
|
|
|
|
}
|
|
|
|
updateGridSize();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
QSize FolderView::iconSize(ViewMode mode) const {
|
|
|
|
Q_ASSERT(mode >= FirstViewMode && mode <= LastViewMode);
|
|
|
|
return iconSize_[mode - FirstViewMode];
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderView::setMargins(QSize size) {
|
|
|
|
if(itemDelegateMargins_ != size.expandedTo(QSize(0, 0))) {
|
|
|
|
itemDelegateMargins_ = size.expandedTo(QSize(0, 0));
|
|
|
|
updateGridSize();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
FolderView::ViewMode FolderView::viewMode() const {
|
|
|
|
return mode;
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderView::setAutoSelectionDelay(int delay) {
|
|
|
|
autoSelectionDelay_ = delay;
|
|
|
|
}
|
|
|
|
|
|
|
|
QAbstractItemView* FolderView::childView() const {
|
|
|
|
return view;
|
|
|
|
}
|
|
|
|
|
|
|
|
ProxyFolderModel* FolderView::model() const {
|
|
|
|
return model_;
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderView::setModel(ProxyFolderModel* model) {
|
|
|
|
if(view) {
|
|
|
|
view->setModel(model);
|
|
|
|
QSize iconSize = iconSize_[mode - FirstViewMode];
|
|
|
|
model->setThumbnailSize(iconSize.width());
|
|
|
|
if(view->selectionModel()) {
|
|
|
|
connect(view->selectionModel(), &QItemSelectionModel::selectionChanged, this, &FolderView::onSelectionChanged);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if(model_) {
|
|
|
|
delete model_;
|
|
|
|
}
|
|
|
|
model_ = model;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool FolderView::event(QEvent* event) {
|
|
|
|
switch(event->type()) {
|
|
|
|
case QEvent::StyleChange:
|
|
|
|
break;
|
|
|
|
case QEvent::FontChange:
|
|
|
|
updateGridSize();
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return QWidget::event(event);
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderView::contextMenuEvent(QContextMenuEvent* event) {
|
|
|
|
QWidget::contextMenuEvent(event);
|
|
|
|
QPoint pos = event->pos();
|
|
|
|
QPoint view_pos = view->mapFromParent(pos);
|
|
|
|
QPoint viewport_pos = view->viewport()->mapFromParent(view_pos);
|
|
|
|
emitClickedAt(ContextMenuClick, viewport_pos);
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderView::childMousePressEvent(QMouseEvent* event) {
|
|
|
|
// called from mousePressEvent() of child view
|
|
|
|
Qt::MouseButton button = event->button();
|
|
|
|
if(button == Qt::MiddleButton) {
|
|
|
|
emitClickedAt(MiddleClick, event->pos());
|
|
|
|
}
|
|
|
|
else if(button == Qt::BackButton) {
|
|
|
|
Q_EMIT clickedBack();
|
|
|
|
}
|
|
|
|
else if(button == Qt::ForwardButton) {
|
|
|
|
Q_EMIT clickedForward();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderView::emitClickedAt(ClickType type, const QPoint& pos) {
|
|
|
|
// indexAt() needs a point in "viewport" coordinates.
|
|
|
|
QModelIndex index = view->indexAt(pos);
|
|
|
|
if(index.isValid()) {
|
|
|
|
QVariant data = index.data(FolderModel::FileInfoRole);
|
|
|
|
auto info = data.value<std::shared_ptr<const Fm::FileInfo>>();
|
|
|
|
Q_EMIT clicked(type, info);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// FIXME: should we show popup menu for the selected files instead
|
|
|
|
// if there are selected files?
|
|
|
|
if(type == ContextMenuClick) {
|
|
|
|
// clear current selection if clicked outside selected files
|
|
|
|
view->clearSelection();
|
|
|
|
Q_EMIT clicked(type, nullptr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
QModelIndexList FolderView::selectedRows(int column) const {
|
|
|
|
QItemSelectionModel* selModel = selectionModel();
|
|
|
|
if(selModel) {
|
|
|
|
return selModel->selectedRows(column);
|
|
|
|
}
|
|
|
|
return QModelIndexList();
|
|
|
|
}
|
|
|
|
|
|
|
|
// This returns all selected "cells", which means all cells of the same row are returned.
|
|
|
|
QModelIndexList FolderView::selectedIndexes() const {
|
|
|
|
QItemSelectionModel* selModel = selectionModel();
|
|
|
|
if(selModel) {
|
|
|
|
return selModel->selectedIndexes();
|
|
|
|
}
|
|
|
|
return QModelIndexList();
|
|
|
|
}
|
|
|
|
|
|
|
|
QItemSelectionModel* FolderView::selectionModel() const {
|
|
|
|
return view ? view->selectionModel() : nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
Fm::FilePathList FolderView::selectedFilePaths() const {
|
|
|
|
if(model_) {
|
|
|
|
QModelIndexList selIndexes = mode == DetailedListMode ? selectedRows() : selectedIndexes();
|
|
|
|
if(!selIndexes.isEmpty()) {
|
|
|
|
Fm::FilePathList paths;
|
|
|
|
QModelIndexList::const_iterator it;
|
|
|
|
for(it = selIndexes.constBegin(); it != selIndexes.constEnd(); ++it) {
|
|
|
|
auto file = model_->fileInfoFromIndex(*it);
|
|
|
|
paths.push_back(file->path());
|
|
|
|
}
|
|
|
|
return paths;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return Fm::FilePathList();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool FolderView::hasSelection() const {
|
|
|
|
QItemSelectionModel* selModel = selectionModel();
|
|
|
|
return selModel ? selModel->hasSelection() : false;
|
|
|
|
}
|
|
|
|
|
|
|
|
QModelIndex FolderView::indexFromFolderPath(const Fm::FilePath& folderPath) const {
|
|
|
|
if(!model_ || !folderPath.isValid()) {
|
|
|
|
return QModelIndex();
|
|
|
|
}
|
|
|
|
QModelIndex index;
|
|
|
|
int count = model_->rowCount();
|
|
|
|
for(int row = 0; row < count; ++row) {
|
|
|
|
index = model_->index(row, 0);
|
|
|
|
auto info = model_->fileInfoFromIndex(index);
|
|
|
|
if(info && info->isDir() && folderPath == info->path()) {
|
|
|
|
return index;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return QModelIndex();
|
|
|
|
}
|
|
|
|
|
|
|
|
Fm::FileInfoList FolderView::selectedFiles() const {
|
|
|
|
if(model_) {
|
|
|
|
QModelIndexList selIndexes = mode == DetailedListMode ? selectedRows() : selectedIndexes();
|
|
|
|
if(!selIndexes.isEmpty()) {
|
|
|
|
Fm::FileInfoList files;
|
|
|
|
QModelIndexList::const_iterator it;
|
|
|
|
for(it = selIndexes.constBegin(); it != selIndexes.constEnd(); ++it) {
|
|
|
|
auto file = model_->fileInfoFromIndex(*it);
|
|
|
|
files.push_back(file);
|
|
|
|
}
|
|
|
|
return files;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return Fm::FileInfoList();
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderView::selectAll() {
|
|
|
|
if(mode == DetailedListMode) {
|
|
|
|
view->selectAll();
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// NOTE: By default QListView::selectAll() selects all columns in the model.
|
|
|
|
// However, QListView only show the first column. Normal selection by mouse
|
|
|
|
// can only select the first column of every row. I consider this discripancy yet
|
|
|
|
// another design flaw of Qt. To make them consistent, we do it ourselves by only
|
|
|
|
// selecting the first column of every row and do not select all columns as Qt does.
|
|
|
|
// I'll report a Qt bug for this later.
|
|
|
|
if(model_) {
|
|
|
|
const QItemSelection sel{model_->index(0, 0), model_->index(model_->rowCount() - 1, 0)};
|
|
|
|
selectionModel()->select(sel, QItemSelectionModel::Select);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderView::invertSelection() {
|
|
|
|
if(model_) {
|
|
|
|
QItemSelectionModel* selModel = view->selectionModel();
|
|
|
|
int rows = model_->rowCount();
|
|
|
|
QItemSelectionModel::SelectionFlags flags = QItemSelectionModel::Toggle;
|
|
|
|
if(mode == DetailedListMode) {
|
|
|
|
flags |= QItemSelectionModel::Rows;
|
|
|
|
}
|
|
|
|
for(int row = 0; row < rows; ++row) {
|
|
|
|
QModelIndex index = model_->index(row, 0);
|
|
|
|
selModel->select(index, flags);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderView::childDragEnterEvent(QDragEnterEvent* event) {
|
|
|
|
qDebug("drag enter");
|
|
|
|
if(event->mimeData()->hasFormat("text/uri-list")) {
|
|
|
|
event->accept();
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
event->ignore();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderView::childDragLeaveEvent(QDragLeaveEvent* e) {
|
|
|
|
qDebug("drag leave");
|
|
|
|
e->accept();
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderView::childDragMoveEvent(QDragMoveEvent* /*e*/) {
|
|
|
|
qDebug("drag move");
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderView::childDropEvent(QDropEvent* e) {
|
|
|
|
// qDebug("drop");
|
|
|
|
// Try to support XDS
|
|
|
|
// NOTE: in theory, it's not possible to implement XDS with pure Qt.
|
|
|
|
// We achieved this with some dirty XCB/XDND workarounds.
|
|
|
|
// Please refer to XdndWorkaround::clientMessage() in xdndworkaround.cpp for details.
|
|
|
|
if(QX11Info::isPlatformX11() && e->mimeData()->hasFormat("XdndDirectSave0")) {
|
|
|
|
e->setDropAction(Qt::CopyAction);
|
|
|
|
const QWidget* targetWidget = childView()->viewport();
|
|
|
|
// these are dynamic QObject property set by our XDND workarounds in xdndworkaround.cpp.
|
|
|
|
xcb_window_t dndSource = xcb_window_t(targetWidget->property("xdnd::lastDragSource").toUInt());
|
|
|
|
//xcb_timestamp_t dropTimestamp = (xcb_timestamp_t)targetWidget->property("xdnd::lastDropTime").toUInt();
|
|
|
|
// qDebug() << "XDS: source window" << dndSource << dropTimestamp;
|
|
|
|
if(dndSource != 0) {
|
|
|
|
xcb_atom_t XdndDirectSaveAtom = XdndWorkaround::internAtom("XdndDirectSave0", 15);
|
|
|
|
xcb_atom_t textAtom = XdndWorkaround::internAtom("text/plain", 10);
|
|
|
|
|
|
|
|
// 1. get the filename from XdndDirectSave property of the source window
|
|
|
|
QByteArray basename = XdndWorkaround::windowProperty(dndSource, XdndDirectSaveAtom, textAtom, 1024);
|
|
|
|
|
|
|
|
// 2. construct the fill URI for the file, and update the source window property.
|
|
|
|
Fm::FilePath filePath;
|
|
|
|
if(model_) {
|
|
|
|
QModelIndex index = view->indexAt(e->pos());
|
|
|
|
auto info = model_->fileInfoFromIndex(index);
|
|
|
|
if(info && info->isDir()) {
|
|
|
|
filePath = info->path().child(basename);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if(!filePath.isValid()) {
|
|
|
|
filePath = path().child(basename);
|
|
|
|
}
|
|
|
|
QByteArray fileUri = filePath.uri().get();
|
|
|
|
XdndWorkaround::setWindowProperty(dndSource, XdndDirectSaveAtom, textAtom, (void*)fileUri.constData(), fileUri.length());
|
|
|
|
|
|
|
|
// 3. send to XDS selection data request with type "XdndDirectSave" to the source window and
|
|
|
|
// receive result from the source window. (S: success, E: error, or F: failure)
|
|
|
|
QByteArray result = e->mimeData()->data("XdndDirectSave0");
|
|
|
|
// NOTE: there seems to be some bugs in file-roller so it always replies with "E" even if the
|
|
|
|
// file extraction is finished successfully. Anyways, we ignore any error at the moment.
|
|
|
|
}
|
|
|
|
e->accept(); // yeah! we've done with XDS so stop Qt from further event propagation.
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(e->keyboardModifiers() == Qt::NoModifier) {
|
|
|
|
// if no key modifiers are used, popup a menu
|
|
|
|
// to ask the user for the action he/she wants to perform.
|
|
|
|
Qt::DropAction action = DndActionMenu::askUser(e->possibleActions(), QCursor::pos());
|
|
|
|
e->setDropAction(action);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool FolderView::eventFilter(QObject* watched, QEvent* event) {
|
|
|
|
// NOTE: Instead of simply filtering the drag and drop events of the child view in
|
|
|
|
// the event filter, we overrided each event handler virtual methods in
|
|
|
|
// both QListView and QTreeView and added some childXXXEvent() callbacks.
|
|
|
|
// We did this because of a design flaw of Qt.
|
|
|
|
// All QAbstractScrollArea derived widgets, including QAbstractItemView
|
|
|
|
// contains an internal child widget, which is called a viewport.
|
|
|
|
// The events actually comes from the child viewport, not the parent view itself.
|
|
|
|
// Qt redirects the events of viewport to the viewportEvent() method of
|
|
|
|
// QAbstractScrollArea and let the parent widget handle the events.
|
|
|
|
// Qt implemented this using a event filter installed on the child viewport widget.
|
|
|
|
// That means, when we try to install an event filter on the viewport,
|
|
|
|
// there is already a filter installed by Qt which will be called before ours.
|
|
|
|
// So we can never intercept the event handling of QAbstractItemView by using a filter.
|
|
|
|
// That's why we override respective virtual methods for different events.
|
|
|
|
if(view && watched == view->viewport()) {
|
|
|
|
switch(event->type()) {
|
|
|
|
case QEvent::HoverMove:
|
|
|
|
// activate items on single click
|
|
|
|
if(style()->styleHint(QStyle::SH_ItemView_ActivateItemOnSingleClick)) {
|
|
|
|
QHoverEvent* hoverEvent = static_cast<QHoverEvent*>(event);
|
|
|
|
QModelIndex index = view->indexAt(hoverEvent->pos()); // find out the hovered item
|
|
|
|
if(index.isValid()) { // change the cursor to a hand when hovering on an item
|
|
|
|
setCursor(Qt::PointingHandCursor);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
setCursor(Qt::ArrowCursor);
|
|
|
|
}
|
|
|
|
// turn on auto-selection for hovered item when single click mode is used.
|
|
|
|
if(autoSelectionDelay_ > 0 && model_) {
|
|
|
|
if(!autoSelectionTimer_) {
|
|
|
|
autoSelectionTimer_ = new QTimer(this);
|
|
|
|
connect(autoSelectionTimer_, &QTimer::timeout, this, &FolderView::onAutoSelectionTimeout);
|
|
|
|
lastAutoSelectionIndex_ = QModelIndex();
|
|
|
|
}
|
|
|
|
autoSelectionTimer_->start(autoSelectionDelay_);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case QEvent::HoverLeave:
|
|
|
|
if(style()->styleHint(QStyle::SH_ItemView_ActivateItemOnSingleClick)) {
|
|
|
|
setCursor(Qt::ArrowCursor);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case QEvent::Wheel:
|
|
|
|
// don't let the view scroll during an inline renaming
|
|
|
|
if (view) {
|
|
|
|
FolderItemDelegate* delegate = nullptr;
|
|
|
|
if(mode == DetailedListMode) {
|
|
|
|
FolderViewTreeView* treeView = static_cast<FolderViewTreeView*>(view);
|
|
|
|
delegate = static_cast<FolderItemDelegate*>(treeView->itemDelegateForColumn(FolderModel::ColumnFileName));
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
FolderViewListView* listView = static_cast<FolderViewListView*>(view);
|
|
|
|
delegate = static_cast<FolderItemDelegate*>(listView->itemDelegateForColumn(FolderModel::ColumnFileName));
|
|
|
|
}
|
|
|
|
if (delegate && delegate->hasEditor()) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// This is to fix #85: Scrolling doesn't work in compact view
|
|
|
|
// Actually, I think it's the bug of Qt, not ours.
|
|
|
|
// When in compact mode, only the horizontal scroll bar is used and the vertical one is hidden.
|
|
|
|
// So, when a user scroll his mouse wheel, it's reasonable to scroll the horizontal scollbar.
|
|
|
|
// Qt does not implement such a simple feature, unfortunately.
|
|
|
|
// We do it by forwarding the scroll event in the viewport to the horizontal scrollbar.
|
|
|
|
// FIXME: if someday Qt supports this, we have to disable the workaround.
|
|
|
|
if(mode == CompactMode) {
|
|
|
|
QScrollBar* scroll = view->horizontalScrollBar();
|
|
|
|
if(scroll) {
|
|
|
|
QApplication::sendEvent(scroll, event);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return QObject::eventFilter(watched, event);
|
|
|
|
}
|
|
|
|
|
|
|
|
// this slot handles auto-selection of items.
|
|
|
|
void FolderView::onAutoSelectionTimeout() {
|
|
|
|
if(QApplication::mouseButtons() != Qt::NoButton) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
Qt::KeyboardModifiers mods = QApplication::keyboardModifiers();
|
|
|
|
QPoint pos = view->viewport()->mapFromGlobal(QCursor::pos()); // convert to viewport coordinates
|
|
|
|
QModelIndex index = view->indexAt(pos); // find out the hovered item
|
|
|
|
QItemSelectionModel::SelectionFlags flags = (mode == DetailedListMode ? QItemSelectionModel::Rows : QItemSelectionModel::NoUpdate);
|
|
|
|
QItemSelectionModel* selModel = view->selectionModel();
|
|
|
|
|
|
|
|
if(mods & Qt::ControlModifier) { // Ctrl key is pressed
|
|
|
|
if(selModel->isSelected(index) && index != lastAutoSelectionIndex_) {
|
|
|
|
// unselect a previously selected item
|
|
|
|
selModel->select(index, flags | QItemSelectionModel::Deselect);
|
|
|
|
lastAutoSelectionIndex_ = QModelIndex();
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// select an unselected item
|
|
|
|
selModel->select(index, flags | QItemSelectionModel::Select);
|
|
|
|
lastAutoSelectionIndex_ = index;
|
|
|
|
}
|
|
|
|
selModel->setCurrentIndex(index, QItemSelectionModel::NoUpdate); // move the cursor
|
|
|
|
}
|
|
|
|
else if(mods & Qt::ShiftModifier) { // Shift key is pressed
|
|
|
|
// select all items between current index and the hovered index.
|
|
|
|
QModelIndex current = selModel->currentIndex();
|
|
|
|
if(selModel->hasSelection() && current.isValid()) {
|
|
|
|
selModel->clear(); // clear old selection
|
|
|
|
selModel->setCurrentIndex(current, QItemSelectionModel::NoUpdate);
|
|
|
|
int begin = current.row();
|
|
|
|
int end = index.row();
|
|
|
|
if(begin > end) {
|
|
|
|
qSwap(begin, end);
|
|
|
|
}
|
|
|
|
for(int row = begin; row <= end; ++row) {
|
|
|
|
QModelIndex sel = model_->index(row, 0);
|
|
|
|
selModel->select(sel, flags | QItemSelectionModel::Select);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else { // no items are selected, select the hovered item.
|
|
|
|
if(index.isValid()) {
|
|
|
|
selModel->select(index, flags | QItemSelectionModel::SelectCurrent);
|
|
|
|
selModel->setCurrentIndex(index, QItemSelectionModel::NoUpdate);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
lastAutoSelectionIndex_ = index;
|
|
|
|
}
|
|
|
|
else if(mods == Qt::NoModifier) { // no modifier keys are pressed.
|
|
|
|
if(index.isValid()) {
|
|
|
|
// select the hovered item
|
|
|
|
view->clearSelection();
|
|
|
|
selModel->select(index, flags | QItemSelectionModel::SelectCurrent);
|
|
|
|
selModel->setCurrentIndex(index, QItemSelectionModel::NoUpdate);
|
|
|
|
}
|
|
|
|
lastAutoSelectionIndex_ = index;
|
|
|
|
}
|
|
|
|
|
|
|
|
autoSelectionTimer_->deleteLater();
|
|
|
|
autoSelectionTimer_ = nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderView::onFileClicked(int type, const std::shared_ptr<const Fm::FileInfo> &fileInfo) {
|
|
|
|
if(type == ActivatedClick) {
|
|
|
|
if(fileLauncher_) {
|
|
|
|
Fm::FileInfoList files;
|
|
|
|
files.push_back(fileInfo);
|
|
|
|
fileLauncher_->launchFiles(nullptr, std::move(files));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if(type == ContextMenuClick) {
|
|
|
|
Fm::FilePath folderPath;
|
|
|
|
bool isWritableDir(true);
|
|
|
|
auto files = selectedFiles();
|
|
|
|
if(!files.empty()) {
|
|
|
|
auto& first = files.front();
|
|
|
|
if(files.size() == 1 && first->isDir()) {
|
|
|
|
folderPath = first->path();
|
|
|
|
isWritableDir = first->isWritable();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if(!folderPath.isValid()) {
|
|
|
|
folderPath = path();
|
|
|
|
if(auto info = folderInfo()) {
|
|
|
|
isWritableDir = info->isWritable();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
QMenu* menu = nullptr;
|
|
|
|
if(fileInfo) {
|
|
|
|
// show context menu
|
|
|
|
auto files = selectedFiles();
|
|
|
|
if(!files.empty()) {
|
|
|
|
QModelIndexList selIndexes = mode == DetailedListMode ? selectedRows() : selectedIndexes();
|
|
|
|
Fm::FileMenu* fileMenu = (view && selIndexes.size() == 1)
|
|
|
|
? new Fm::FileMenu(files, fileInfo, folderPath, isWritableDir, QString(), view)
|
|
|
|
: new Fm::FileMenu(files, fileInfo, folderPath, isWritableDir);
|
|
|
|
fileMenu->setFileLauncher(fileLauncher_);
|
|
|
|
prepareFileMenu(fileMenu);
|
|
|
|
menu = fileMenu;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (folderInfo()) {
|
|
|
|
Fm::FolderMenu* folderMenu = new Fm::FolderMenu(this);
|
|
|
|
prepareFolderMenu(folderMenu);
|
|
|
|
menu = folderMenu;
|
|
|
|
}
|
|
|
|
if(menu) {
|
|
|
|
menu->exec(QCursor::pos());
|
|
|
|
delete menu;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderView::onClipboardDataChange() {
|
|
|
|
if(model_) {
|
|
|
|
const QClipboard* clipboard = QApplication::clipboard();
|
|
|
|
const QMimeData* data = clipboard->mimeData();
|
|
|
|
Fm::FilePathList paths;
|
|
|
|
bool isCutSelection;
|
|
|
|
std::tie(paths, isCutSelection) = Fm::parseClipboardData(*data);
|
|
|
|
if(!folder()->path().hasUriScheme("search") // skip for search results
|
|
|
|
&& isCutSelection
|
|
|
|
&& Fm::isCurrentPidClipboardData(*data)) { // set cut files only with this app
|
|
|
|
auto cutDirPath = paths.size() > 0 ? paths[0].parent(): FilePath();
|
|
|
|
if(folder()->path() == cutDirPath) {
|
|
|
|
model_->setCutFiles(selectionModel()->selection());
|
|
|
|
}
|
|
|
|
else if(folder()->hadCutFilesUnset() || folder()->hasCutFiles()) {
|
|
|
|
model_->setCutFiles(QItemSelection());
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
folder()->setCutFiles(std::make_shared<HashSet>()); // clean Folder::cutFilesHashSet_
|
|
|
|
if(folder()->hadCutFilesUnset()) {
|
|
|
|
model_->setCutFiles(QItemSelection()); // update indexes if there were cut files here
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderView::prepareFileMenu(FileMenu* /*menu*/) {
|
|
|
|
}
|
|
|
|
|
|
|
|
void FolderView::prepareFolderMenu(FolderMenu* /*menu*/) {
|
|
|
|
}
|