You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
libfm-qt-packaging/src/folderview.cpp

1243 lines
48 KiB

/*
* 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*/) {
}