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.
294 lines
13 KiB
294 lines
13 KiB
Description: Fix DND and drop indicator on desktop
|
|
(1) When items are dragged and dropped on any desktop position other than
|
|
that of a directory, they will stick to it (and to its following positions),
|
|
moving next sticky items if needed. When desktop items are dropped on
|
|
themselves or on a directory, the DND menu will be shown.
|
|
.
|
|
(2) The drop rectangle (when dragging desktop items on desktop folders) is
|
|
also fixed by drawing it explicitly.
|
|
Author: Tsu Jan <tsujan2000@gmail.com>
|
|
Origin: upstream
|
|
Bug: https://github.com/lxqt/pcmanfm-qt/issues/706, https://github.com/lxqt/pcmanfm-qt/issues/722
|
|
Applied-Upstream: commit:748a105
|
|
Last-Update: 2018-07-14
|
|
--- a/pcmanfm/desktopwindow.cpp
|
|
+++ b/pcmanfm/desktopwindow.cpp
|
|
@@ -91,6 +91,7 @@ DesktopWindow::DesktopWindow(int screenN
|
|
listView_->setMovement(QListView::Snap);
|
|
listView_->setResizeMode(QListView::Adjust);
|
|
listView_->setFlow(QListView::TopToBottom);
|
|
+ listView_->setDropIndicatorShown(false); // we draw the drop indicator ourself
|
|
|
|
// This is to workaround Qt bug 54384 which affects Qt >= 5.6
|
|
// https://bugreports.qt.io/browse/QTBUG-54384
|
|
@@ -126,7 +127,6 @@ DesktopWindow::DesktopWindow(int screenN
|
|
connect(proxyModel_, &Fm::ProxyFolderModel::layoutChanged, this, &DesktopWindow::onLayoutChanged);
|
|
connect(proxyModel_, &Fm::ProxyFolderModel::sortFilterChanged, this, &DesktopWindow::onModelSortFilterChanged);
|
|
connect(proxyModel_, &Fm::ProxyFolderModel::dataChanged, this, &DesktopWindow::onDataChanged);
|
|
- connect(listView_, &QListView::indexesMoved, this, &DesktopWindow::onIndexesMoved);
|
|
}
|
|
|
|
// remove frame
|
|
@@ -710,41 +710,6 @@ void DesktopWindow::onDataChanged(const
|
|
}
|
|
}
|
|
|
|
-void DesktopWindow::onIndexesMoved(const QModelIndexList& indexes) {
|
|
- auto delegate = static_cast<Fm::FolderItemDelegate*>(listView_->itemDelegateForColumn(0));
|
|
- auto itemSize = delegate->itemSize();
|
|
- // remember the custom position for the items
|
|
- for(const QModelIndex& index : indexes) {
|
|
- // Under some circumstances, Qt might emit indexMoved for
|
|
- // every single cells in the same row. (when QAbstractItemView::SelectItems is set)
|
|
- // So indexes list may contain several indixes for the same row.
|
|
- // Since we only care about rows, not individual cells,
|
|
- // let's handle column 0 of every row here.
|
|
- if(index.column() == 0) {
|
|
- auto file = proxyModel_->fileInfoFromIndex(index);
|
|
- QRect itemRect = listView_->rectForIndex(index);
|
|
- QPoint tl = itemRect.topLeft();
|
|
- QRect workArea = qApp->desktop()->availableGeometry(screenNum_);
|
|
- workArea.adjust(12, 12, -12, -12);
|
|
-
|
|
- // check if the position is occupied by another item
|
|
- auto existingItem = std::find_if(customItemPos_.cbegin(), customItemPos_.cend(), [tl](const std::pair<std::string, QPoint>& elem){
|
|
- return elem.second == tl;
|
|
- });
|
|
-
|
|
- if(existingItem == customItemPos_.cend() // don't put items on each other
|
|
- && tl.x() >= workArea.x() && tl.y() >= workArea.y()
|
|
- && tl.x() + itemSize.width() <= workArea.right() + 1 // for historical reasons (-> Qt doc)
|
|
- && tl.y() + itemSize.height() <= workArea.bottom() + 1) { // as above
|
|
- customItemPos_[file->name()] = tl;
|
|
- // qDebug() << "indexMoved:" << name << index << itemRect;
|
|
- }
|
|
- }
|
|
- }
|
|
- saveItemPositions();
|
|
- queueRelayout();
|
|
-}
|
|
-
|
|
void DesktopWindow::onFolderStartLoading() { // desktop may be reloaded
|
|
if(model_) {
|
|
disconnect(model_, &Fm::FolderModel::filesAdded, this, &DesktopWindow::onFilesAdded);
|
|
@@ -1247,6 +1212,11 @@ bool DesktopWindow::eventFilter(QObject*
|
|
}
|
|
}
|
|
break;
|
|
+ case QEvent::Paint:
|
|
+ // NOTE: The drop indicator isn't drawn/updated automatically, perhaps,
|
|
+ // because we paint desktop ourself. So, we draw it here.
|
|
+ paintDropIndicator();
|
|
+ break;
|
|
default:
|
|
break;
|
|
}
|
|
@@ -1254,6 +1224,34 @@ bool DesktopWindow::eventFilter(QObject*
|
|
return Fm::FolderView::eventFilter(watched, event);
|
|
}
|
|
|
|
+void DesktopWindow::childDragMoveEvent(QDragMoveEvent* e) {
|
|
+ // see DesktopWindow::eventFilter for an explanation
|
|
+ QRect oldDropRect = dropRect_;
|
|
+ dropRect_ = QRect();
|
|
+ QModelIndex index = listView_->indexAt(e->pos());
|
|
+ if(index.isValid() && index.model()) {
|
|
+ QVariant data = index.model()->data(index, Fm::FolderModel::Role::FileInfoRole);
|
|
+ auto info = data.value<std::shared_ptr<const Fm::FileInfo>>();
|
|
+ if(info && info->isDir()) {
|
|
+ dropRect_ = listView_->rectForIndex(index);
|
|
+ }
|
|
+ }
|
|
+ if(oldDropRect != dropRect_){
|
|
+ listView_->viewport()->update();
|
|
+ }
|
|
+}
|
|
+
|
|
+void DesktopWindow::paintDropIndicator()
|
|
+{
|
|
+ if(!dropRect_.isNull()) {
|
|
+ QPainter painter(listView_->viewport());
|
|
+ QStyleOption opt;
|
|
+ opt.init(listView_->viewport());
|
|
+ opt.rect = dropRect_;
|
|
+ style()->drawPrimitive(QStyle::PE_IndicatorItemViewItemDrop, &opt, &painter, listView_);
|
|
+ }
|
|
+}
|
|
+
|
|
void DesktopWindow::childDropEvent(QDropEvent* e) {
|
|
const QMimeData* mimeData = e->mimeData();
|
|
bool moveItem = false;
|
|
@@ -1264,45 +1262,59 @@ void DesktopWindow::childDropEvent(QDrop
|
|
QModelIndex dropIndex = listView_->indexAt(e->pos());
|
|
if(dropIndex.isValid()) { // drop on an item
|
|
QModelIndexList selected = selectedIndexes(); // the dragged items
|
|
- if(selected.contains(dropIndex)) { // drop on self, ignore
|
|
- moveItem = true;
|
|
+ if(!selected.contains(dropIndex)) { // not a drop on self
|
|
+ if(auto file = proxyModel_->fileInfoFromIndex(dropIndex)) {
|
|
+ if(!file->isDir()) { // drop on a non-directory file
|
|
+ moveItem = true;
|
|
+ }
|
|
+ }
|
|
}
|
|
}
|
|
- else { // drop on a blank area
|
|
+ else { // drop on a blank area (maybe, between other items)
|
|
moveItem = true;
|
|
}
|
|
}
|
|
}
|
|
if(moveItem) {
|
|
e->accept();
|
|
- }
|
|
- else {
|
|
+ // move selected items to the drop position, putting them successively
|
|
auto delegate = static_cast<Fm::FolderItemDelegate*>(listView_->itemDelegateForColumn(0));
|
|
auto grid = delegate->itemSize();
|
|
+ QRect workArea = qApp->desktop()->availableGeometry(screenNum_);
|
|
+ workArea.adjust(12, 12, -12, -12);
|
|
+ QPoint pos = mapFromGlobal(e->pos());
|
|
+ const QModelIndexList indexes = selectedIndexes();
|
|
+ for(const QModelIndex& indx : indexes) {
|
|
+ if(auto file = proxyModel_->fileInfoFromIndex(indx)) {
|
|
+ stickToPosition(QString::fromStdString(file->name()), pos, workArea, grid);
|
|
+ }
|
|
+ }
|
|
+ saveItemPositions();
|
|
+ queueRelayout();
|
|
+ }
|
|
+ else {
|
|
Fm::FolderView::childDropEvent(e);
|
|
+ // remove the drop indicator
|
|
+ dropRect_ = QRect();
|
|
+ listView_->viewport()->update();
|
|
// position dropped items successively, starting with the drop rectangle
|
|
if(mimeData->hasUrls()
|
|
&& (e->dropAction() == Qt::CopyAction
|
|
|| e->dropAction() == Qt::MoveAction
|
|
|| e->dropAction() == Qt::LinkAction)) {
|
|
- QList<QUrl> urlList = mimeData->urls();
|
|
- for(int i = 0; i < urlList.count(); ++i) {
|
|
- std::string name = urlList.at(i).fileName().toUtf8().constData();
|
|
- if(!name.empty()) { // respect the positions of existing files
|
|
- QString desktopDir = XdgDir::readDesktopDir() + QString(QLatin1String("/"));
|
|
- if(!QFile::exists(desktopDir + QString::fromStdString(name))) {
|
|
- QRect workArea = qApp->desktop()->availableGeometry(screenNum_);
|
|
- workArea.adjust(12, 12, -12, -12);
|
|
- QPoint pos = mapFromGlobal(e->pos());
|
|
- alignToGrid(pos, workArea.topLeft(), grid, listView_->spacing());
|
|
- if(i > 0)
|
|
- pos.setY(pos.y() + grid.height() + listView_->spacing());
|
|
- if(pos.y() + grid.height() > workArea.bottom() + 1) {
|
|
- pos.setX(pos.x() + grid.width() + listView_->spacing());
|
|
- pos.setY(workArea.top());
|
|
- }
|
|
- customItemPos_[name] = pos;
|
|
- }
|
|
+ auto delegate = static_cast<Fm::FolderItemDelegate*>(listView_->itemDelegateForColumn(0));
|
|
+ auto grid = delegate->itemSize();
|
|
+ QRect workArea = qApp->desktop()->availableGeometry(screenNum_);
|
|
+ workArea.adjust(12, 12, -12, -12);
|
|
+ const QString desktopDir = XdgDir::readDesktopDir() + QString(QLatin1String("/"));
|
|
+ QPoint pos = mapFromGlobal(e->pos());
|
|
+ const QList<QUrl> urlList = mimeData->urls();
|
|
+ for(const QUrl& url : urlList) {
|
|
+ QString name = url.fileName();
|
|
+ if(!name.isEmpty()
|
|
+ // don't stick to the position if there is an overwrite prompt
|
|
+ && !QFile::exists(desktopDir + name)) {
|
|
+ stickToPosition(name, pos, workArea, grid);
|
|
}
|
|
}
|
|
saveItemPositions();
|
|
@@ -1310,10 +1322,39 @@ void DesktopWindow::childDropEvent(QDrop
|
|
}
|
|
}
|
|
|
|
+// This function recursively repositions next sticky items if needed.
|
|
+void DesktopWindow::stickToPosition(const QString& file, QPoint& pos, const QRect& workArea, const QSize& grid) {
|
|
+ // normalize the position
|
|
+ alignToGrid(pos, workArea.topLeft(), grid, listView_->spacing());
|
|
+ if(pos.y() + grid.height() > workArea.bottom() + 1) {
|
|
+ pos.setX(pos.x() + grid.width() + listView_->spacing());
|
|
+ pos.setY(workArea.top());
|
|
+ }
|
|
+ // find if there is a sticky item at this position
|
|
+ QString otherFile;
|
|
+ auto oldItem = std::find_if(customItemPos_.cbegin(),
|
|
+ customItemPos_.cend(),
|
|
+ [pos](const std::pair<std::string, QPoint>& elem) {
|
|
+ return elem.second == pos;
|
|
+ });
|
|
+ if(oldItem != customItemPos_.cend()) {
|
|
+ otherFile = QString::fromStdString(oldItem->first);
|
|
+ }
|
|
+ // stick to the position
|
|
+ customItemPos_[file.toStdString()] = pos;
|
|
+ // find the next position
|
|
+ pos.setY(pos.y() + grid.height() + listView_->spacing());
|
|
+ // if there was another sticky item at the same position, move it to the next position
|
|
+ if(!otherFile.isEmpty() && otherFile != file) {
|
|
+ QPoint nextPos = pos;
|
|
+ stickToPosition(otherFile, nextPos, workArea, grid);
|
|
+ }
|
|
+}
|
|
+
|
|
void DesktopWindow::alignToGrid(QPoint& pos, const QPoint& topLeft, const QSize& grid, const int spacing) {
|
|
qreal w = qAbs((qreal)pos.x() - (qreal)topLeft.x())
|
|
/ (qreal)(grid.width() + spacing);
|
|
- qreal h = qAbs(pos.y() - (qreal)topLeft.y())
|
|
+ qreal h = qAbs((qreal)pos.y() - (qreal)topLeft.y())
|
|
/ (qreal)(grid.height() + spacing);
|
|
pos.setX(topLeft.x() + qRound(w) * (grid.width() + spacing));
|
|
pos.setY(topLeft.y() + qRound(h) * (grid.height() + spacing));
|
|
@@ -1324,7 +1365,7 @@ void DesktopWindow::closeEvent(QCloseEve
|
|
event->ignore();
|
|
}
|
|
|
|
-void DesktopWindow::paintEvent(QPaintEvent *event) {
|
|
+void DesktopWindow::paintEvent(QPaintEvent* event) {
|
|
paintBackground(event);
|
|
QWidget::paintEvent(event);
|
|
}
|
|
--- a/pcmanfm/desktopwindow.h
|
|
+++ b/pcmanfm/desktopwindow.h
|
|
@@ -98,9 +98,10 @@ protected:
|
|
virtual bool event(QEvent* event) override;
|
|
virtual bool eventFilter(QObject* watched, QEvent* event) override;
|
|
|
|
+ virtual void childDragMoveEvent(QDragMoveEvent* e) override;
|
|
virtual void childDropEvent(QDropEvent* e) override;
|
|
virtual void closeEvent(QCloseEvent* event) override;
|
|
- virtual void paintEvent(QPaintEvent *event) override;
|
|
+ virtual void paintEvent(QPaintEvent* event) override;
|
|
|
|
protected Q_SLOTS:
|
|
void onOpenDirRequested(const Fm::FilePath& path, int target);
|
|
@@ -112,7 +113,6 @@ protected Q_SLOTS:
|
|
void onRowsInserted(const QModelIndex& parent, int start, int end);
|
|
void onLayoutChanged();
|
|
void onModelSortFilterChanged();
|
|
- void onIndexesMoved(const QModelIndexList& indexes);
|
|
void onDataChanged(const QModelIndex& topLeft, const QModelIndex& bottomRight);
|
|
void onFolderStartLoading();
|
|
void onFolderFinishLoading();
|
|
@@ -136,6 +136,8 @@ private:
|
|
void removeBottomGap();
|
|
void addDesktopActions(QMenu* menu);
|
|
void paintBackground(QPaintEvent* event);
|
|
+ void paintDropIndicator();
|
|
+ void stickToPosition(const QString& file, QPoint& pos, const QRect& workArea, const QSize& grid);
|
|
static void alignToGrid(QPoint& pos, const QPoint& topLeft, const QSize& grid, const int spacing);
|
|
|
|
private:
|
|
@@ -164,6 +166,8 @@ private:
|
|
QHash<QModelIndex, QString> displayNames_; // only for desktop entries and shortcuts
|
|
QTimer* relayoutTimer_;
|
|
QTimer* selectionTimer_;
|
|
+
|
|
+ QRect dropRect_;
|
|
};
|
|
|
|
}
|