Introduce dragging for all explorer nodes.

Extend drop support with variant values. A drop may be a file drop or a
value drop or both.

Rename Utils::FileDropSupport to Utils::DropSupport and add methods to
add not only files but any QVariant value to the mime data. Project
explorer adds dragged nodes (which will be needed for future ModelEditor
plugin).

Change-Id: I799542c60fdecb3e64af0d3ba47b6caa9adbcfd7
Reviewed-by: Eike Ziller <eike.ziller@theqtcompany.com>
This commit is contained in:
Jochen Becher
2015-06-04 12:35:59 +02:00
parent d311877142
commit 431b25ad27
23 changed files with 406 additions and 250 deletions

View File

@@ -35,7 +35,7 @@
#include <cplusplus/Scope.h> #include <cplusplus/Scope.h>
#include <cplusplus/Literals.h> #include <cplusplus/Literals.h>
#include <cplusplus/Symbols.h> #include <cplusplus/Symbols.h>
#include <utils/fileutils.h> #include <utils/dropsupport.h>
using namespace CPlusPlus; using namespace CPlusPlus;
@@ -258,12 +258,12 @@ Qt::DropActions OverviewModel::supportedDragActions() const
QStringList OverviewModel::mimeTypes() const QStringList OverviewModel::mimeTypes() const
{ {
return Utils::FileDropSupport::mimeTypesForFilePaths(); return Utils::DropSupport::mimeTypesForFilePaths();
} }
QMimeData *OverviewModel::mimeData(const QModelIndexList &indexes) const QMimeData *OverviewModel::mimeData(const QModelIndexList &indexes) const
{ {
auto mimeData = new Utils::FileDropMimeData; auto mimeData = new Utils::DropMimeData;
foreach (const QModelIndex &index, indexes) { foreach (const QModelIndex &index, indexes) {
const QVariant fileName = data(index, FileNameRole); const QVariant fileName = data(index, FileNameRole);
if (!fileName.canConvert<QString>()) if (!fileName.canConvert<QString>())

View File

@@ -0,0 +1,240 @@
/****************************************************************************
**
** Copyright (C) 2015 The Qt Company Ltd.
** Contact: http://www.qt.io/licensing
**
** This file is part of Qt Creator.
**
** Commercial License Usage
** Licensees holding valid commercial Qt licenses may use this file in
** accordance with the commercial license agreement provided with the
** Software or, alternatively, in accordance with the terms contained in
** a written agreement between you and The Qt Company. For licensing terms and
** conditions see http://www.qt.io/terms-conditions. For further information
** use the contact form at http://www.qt.io/contact-us.
**
** GNU Lesser General Public License Usage
** Alternatively, this file may be used under the terms of the GNU Lesser
** General Public License version 2.1 or version 3 as published by the Free
** Software Foundation and appearing in the file LICENSE.LGPLv21 and
** LICENSE.LGPLv3 included in the packaging of this file. Please review the
** following information to ensure the GNU Lesser General Public License
** requirements will be met: https://www.gnu.org/licenses/lgpl.html and
** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
**
** In addition, as a special exception, The Qt Company gives you certain additional
** rights. These rights are described in The Qt Company LGPL Exception
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
**
****************************************************************************/
#include "dropsupport.h"
#include "qtcassert.h"
#include <QUrl>
#include <QWidget>
#include <QDropEvent>
#include <QTimer>
#ifdef Q_OS_OSX
// for file drops from Finder, working around QTBUG-40449
namespace Utils {
namespace Internal {
extern QUrl filePathUrl(const QUrl &url);
} // Internal
} // Utils
#endif
namespace Utils {
static bool isFileDrop(const QMimeData *d, QList<DropSupport::FileSpec> *files = 0)
{
// internal drop
if (const DropMimeData *internalData = qobject_cast<const DropMimeData *>(d)) {
if (files)
*files = internalData->files();
return !internalData->files().isEmpty();
}
// external drop
if (files)
files->clear();
// Extract dropped files from Mime data.
if (!d->hasUrls())
return false;
const QList<QUrl> urls = d->urls();
if (urls.empty())
return false;
// Try to find local files
bool hasFiles = false;
const QList<QUrl>::const_iterator cend = urls.constEnd();
for (QList<QUrl>::const_iterator it = urls.constBegin(); it != cend; ++it) {
QUrl url = *it;
#ifdef Q_OS_OSX
// for file drops from Finder, working around QTBUG-40449
url = Internal::filePathUrl(url);
#endif
const QString fileName = url.toLocalFile();
if (!fileName.isEmpty()) {
hasFiles = true;
if (files)
files->append(DropSupport::FileSpec(fileName));
else
break; // No result list, sufficient for checking
}
}
return hasFiles;
}
DropSupport::DropSupport(QWidget *parentWidget, const DropFilterFunction &filterFunction)
: QObject(parentWidget),
m_filterFunction(filterFunction)
{
QTC_ASSERT(parentWidget, return);
parentWidget->setAcceptDrops(true);
parentWidget->installEventFilter(this);
}
QStringList DropSupport::mimeTypesForFilePaths()
{
return QStringList() << QStringLiteral("text/uri-list");
}
bool DropSupport::isFileDrop(QDropEvent *event) const
{
return Utils::isFileDrop(event->mimeData());
}
bool DropSupport::isValueDrop(QDropEvent *event) const
{
if (const DropMimeData *internalData = qobject_cast<const DropMimeData *>(event->mimeData())) {
return !internalData->values().isEmpty();
}
return false;
}
bool DropSupport::eventFilter(QObject *obj, QEvent *event)
{
Q_UNUSED(obj)
if (event->type() == QEvent::DragEnter) {
auto dee = static_cast<QDragEnterEvent *>(event);
if ((isFileDrop(dee) || isValueDrop(dee)) && (!m_filterFunction || m_filterFunction(dee, this))) {
event->accept();
} else {
event->ignore();
}
return true;
} else if (event->type() == QEvent::DragMove) {
event->accept();
return true;
} else if (event->type() == QEvent::Drop) {
bool accepted = false;
auto de = static_cast<QDropEvent *>(event);
if (!m_filterFunction || m_filterFunction(de, this)) {
const DropMimeData *fileDropMimeData = qobject_cast<const DropMimeData *>(de->mimeData());
QList<FileSpec> tempFiles;
if (Utils::isFileDrop(de->mimeData(), &tempFiles)) {
event->accept();
accepted = true;
if (fileDropMimeData && fileDropMimeData->isOverridingFileDropAction())
de->setDropAction(fileDropMimeData->overrideFileDropAction());
else
de->acceptProposedAction();
bool needToScheduleEmit = m_files.isEmpty();
m_files.append(tempFiles);
if (needToScheduleEmit) { // otherwise we already have a timer pending
// Delay the actual drop, to avoid conflict between
// actions that happen when opening files, and actions that the item views do
// after the drag operation.
// If we do not do this, e.g. dragging from Outline view crashes if the editor and
// the selected item changes
QTimer::singleShot(100, this, SLOT(emitFilesDropped()));
}
}
if (fileDropMimeData && !fileDropMimeData->values().isEmpty()) {
event->accept();
accepted = true;
bool needToScheduleEmit = m_values.isEmpty();
m_values.append(fileDropMimeData->values());
m_dropPos = de->pos();
if (needToScheduleEmit)
QTimer::singleShot(100, this, SLOT(emitValuesDropped()));
}
}
if (!accepted) {
event->ignore();
}
return true;
}
return false;
}
void DropSupport::emitFilesDropped()
{
QTC_ASSERT(!m_files.isEmpty(), return);
emit filesDropped(m_files);
m_files.clear();
}
void DropSupport::emitValuesDropped()
{
QTC_ASSERT(!m_values.isEmpty(), return);
emit valuesDropped(m_values, m_dropPos);
m_values.clear();
}
/*!
Sets the drop action to effectively use, instead of the "proposed" drop action from the
drop event. This can be useful when supporting move drags within an item view, but not
"moving" an item from the item view into a split.
*/
DropMimeData::DropMimeData()
: m_overrideDropAction(Qt::IgnoreAction),
m_isOverridingDropAction(false)
{
}
void DropMimeData::setOverrideFileDropAction(Qt::DropAction action)
{
m_isOverridingDropAction = true;
m_overrideDropAction = action;
}
Qt::DropAction DropMimeData::overrideFileDropAction() const
{
return m_overrideDropAction;
}
bool DropMimeData::isOverridingFileDropAction() const
{
return m_isOverridingDropAction;
}
void DropMimeData::addFile(const QString &filePath, int line, int column)
{
// standard mime data
QList<QUrl> currentUrls = urls();
currentUrls.append(QUrl::fromLocalFile(filePath));
setUrls(currentUrls);
// special mime data
m_files.append(DropSupport::FileSpec(filePath, line, column));
}
QList<DropSupport::FileSpec> DropMimeData::files() const
{
return m_files;
}
void DropMimeData::addValue(const QVariant &value)
{
m_values.append(value);
}
QList<QVariant> DropMimeData::values() const
{
return m_values;
}
} // namespace Utils

View File

@@ -0,0 +1,114 @@
/****************************************************************************
**
** Copyright (C) 2015 The Qt Company Ltd.
** Contact: http://www.qt.io/licensing
**
** This file is part of Qt Creator.
**
** Commercial License Usage
** Licensees holding valid commercial Qt licenses may use this file in
** accordance with the commercial license agreement provided with the
** Software or, alternatively, in accordance with the terms contained in
** a written agreement between you and The Qt Company. For licensing terms and
** conditions see http://www.qt.io/terms-conditions. For further information
** use the contact form at http://www.qt.io/contact-us.
**
** GNU Lesser General Public License Usage
** Alternatively, this file may be used under the terms of the GNU Lesser
** General Public License version 2.1 or version 3 as published by the Free
** Software Foundation and appearing in the file LICENSE.LGPLv21 and
** LICENSE.LGPLv3 included in the packaging of this file. Please review the
** following information to ensure the GNU Lesser General Public License
** requirements will be met: https://www.gnu.org/licenses/lgpl.html and
** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
**
** In addition, as a special exception, The Qt Company gives you certain additional
** rights. These rights are described in The Qt Company LGPL Exception
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
**
****************************************************************************/
#ifndef UTILS_DROPSUPPORT_H
#define UTILS_DROPSUPPORT_H
#include "utils_global.h"
#include <QObject>
#include <QMimeData>
#include <QPoint>
#include <functional>
QT_BEGIN_NAMESPACE
class QDropEvent;
class QWidget;
QT_END_NAMESPACE
namespace Utils {
class QTCREATOR_UTILS_EXPORT DropSupport : public QObject
{
Q_OBJECT
public:
struct FileSpec {
FileSpec(const QString &path, int r = -1, int c = -1) : filePath(path), line(r), column(c) {}
QString filePath;
int line;
int column;
};
// returns true if the event should be accepted
typedef std::function<bool(QDropEvent*,DropSupport*)> DropFilterFunction;
DropSupport(QWidget *parentWidget, const DropFilterFunction &filterFunction = DropFilterFunction());
static QStringList mimeTypesForFilePaths();
signals:
void filesDropped(const QList<Utils::DropSupport::FileSpec> &files);
void valuesDropped(const QList<QVariant> &values, const QPoint &dropPos);
public:
bool isFileDrop(QDropEvent *event) const;
bool isValueDrop(QDropEvent *event) const;
protected:
bool eventFilter(QObject *obj, QEvent *event);
private slots:
void emitFilesDropped();
void emitValuesDropped();
private:
DropFilterFunction m_filterFunction;
QList<FileSpec> m_files;
QList<QVariant> m_values;
QPoint m_dropPos;
};
class QTCREATOR_UTILS_EXPORT DropMimeData : public QMimeData
{
Q_OBJECT
public:
DropMimeData();
void setOverrideFileDropAction(Qt::DropAction action);
Qt::DropAction overrideFileDropAction() const;
bool isOverridingFileDropAction() const;
void addFile(const QString &filePath, int line = -1, int column = -1);
QList<DropSupport::FileSpec> files() const;
void addValue(const QVariant &value);
QList<QVariant> values() const;
private:
QList<DropSupport::FileSpec> m_files;
QList<QVariant> m_values;
Qt::DropAction m_overrideDropAction;
bool m_isOverridingDropAction;
};
} // namespace Utils
#endif // UTILS_DROPSUPPORT_H

View File

@@ -57,15 +57,6 @@ QDebug operator<<(QDebug dbg, const Utils::FileName &c)
QT_END_NAMESPACE QT_END_NAMESPACE
#ifdef Q_OS_OSX
// for file drops from Finder, working around QTBUG-40449
namespace Utils {
namespace Internal {
extern QUrl filePathUrl(const QUrl &url);
} // Internal
} // Utils
#endif
namespace Utils { namespace Utils {
/*! \class Utils::FileUtils /*! \class Utils::FileUtils
@@ -785,152 +776,6 @@ int FileNameList::removeDuplicates()
return removed; return removed;
} }
static bool isFileDrop(const QMimeData *d, QList<FileDropSupport::FileSpec> *files = 0)
{
// internal drop
if (const FileDropMimeData *internalData = qobject_cast<const FileDropMimeData *>(d)) {
if (files)
*files = internalData->files();
return true;
}
// external drop
if (files)
files->clear();
// Extract dropped files from Mime data.
if (!d->hasUrls())
return false;
const QList<QUrl> urls = d->urls();
if (urls.empty())
return false;
// Try to find local files
bool hasFiles = false;
const QList<QUrl>::const_iterator cend = urls.constEnd();
for (QList<QUrl>::const_iterator it = urls.constBegin(); it != cend; ++it) {
QUrl url = *it;
#ifdef Q_OS_OSX
// for file drops from Finder, working around QTBUG-40449
url = Internal::filePathUrl(url);
#endif
const QString fileName = url.toLocalFile();
if (!fileName.isEmpty()) {
hasFiles = true;
if (files)
files->append(FileDropSupport::FileSpec(fileName));
else
break; // No result list, sufficient for checking
}
}
return hasFiles;
}
FileDropSupport::FileDropSupport(QWidget *parentWidget, const DropFilterFunction &filterFunction)
: QObject(parentWidget),
m_filterFunction(filterFunction)
{
QTC_ASSERT(parentWidget, return);
parentWidget->setAcceptDrops(true);
parentWidget->installEventFilter(this);
}
QStringList FileDropSupport::mimeTypesForFilePaths()
{
return QStringList() << QStringLiteral("text/uri-list");
}
bool FileDropSupport::eventFilter(QObject *obj, QEvent *event)
{
Q_UNUSED(obj)
if (event->type() == QEvent::DragEnter) {
auto dee = static_cast<QDragEnterEvent *>(event);
if (isFileDrop(dee->mimeData())
&& (!m_filterFunction || m_filterFunction(dee)))
event->accept();
else
event->ignore();
return true;
} else if (event->type() == QEvent::DragMove) {
event->accept();
return true;
} else if (event->type() == QEvent::Drop) {
auto de = static_cast<QDropEvent *>(event);
QList<FileSpec> tempFiles;
if (isFileDrop(de->mimeData(), &tempFiles)
&& (!m_filterFunction || m_filterFunction(de))) {
const FileDropMimeData *fileDropMimeData = qobject_cast<const FileDropMimeData *>(de->mimeData());
event->accept();
if (fileDropMimeData && fileDropMimeData->isOverridingFileDropAction())
de->setDropAction(fileDropMimeData->overrideFileDropAction());
else
de->acceptProposedAction();
bool needToScheduleEmit = m_files.isEmpty();
m_files.append(tempFiles);
if (needToScheduleEmit) { // otherwise we already have a timer pending
// Delay the actual drop, to avoid conflict between
// actions that happen when opening files, and actions that the item views do
// after the drag operation.
// If we do not do this, e.g. dragging from Outline view crashes if the editor and
// the selected item changes
QTimer::singleShot(100, this, SLOT(emitFilesDropped()));
}
} else {
event->ignore();
}
return true;
}
return false;
}
void FileDropSupport::emitFilesDropped()
{
QTC_ASSERT(!m_files.isEmpty(), return);
emit filesDropped(m_files);
m_files.clear();
}
/*!
Sets the drop action to effectively use, instead of the "proposed" drop action from the
drop event. This can be useful when supporting move drags within an item view, but not
"moving" an item from the item view into a split.
*/
FileDropMimeData::FileDropMimeData()
: m_overrideDropAction(Qt::IgnoreAction),
m_isOverridingDropAction(false)
{
}
void FileDropMimeData::setOverrideFileDropAction(Qt::DropAction action)
{
m_isOverridingDropAction = true;
m_overrideDropAction = action;
}
Qt::DropAction FileDropMimeData::overrideFileDropAction() const
{
return m_overrideDropAction;
}
bool FileDropMimeData::isOverridingFileDropAction() const
{
return m_isOverridingDropAction;
}
void FileDropMimeData::addFile(const QString &filePath, int line, int column)
{
// standard mime data
QList<QUrl> currentUrls = urls();
currentUrls.append(QUrl::fromLocalFile(filePath));
setUrls(currentUrls);
// special mime data
m_files.append(FileDropSupport::FileSpec(filePath, line, column));
}
QList<FileDropSupport::FileSpec> FileDropMimeData::files() const
{
return m_files;
}
} // namespace Utils } // namespace Utils
QT_BEGIN_NAMESPACE QT_BEGIN_NAMESPACE

View File

@@ -36,7 +36,6 @@
#include <QCoreApplication> #include <QCoreApplication>
#include <QXmlStreamWriter> // Mac. #include <QXmlStreamWriter> // Mac.
#include <QMetaType> #include <QMetaType>
#include <QMimeData>
#include <QStringList> #include <QStringList>
#include <functional> #include <functional>
@@ -215,58 +214,6 @@ private:
bool m_autoRemove; bool m_autoRemove;
}; };
class QTCREATOR_UTILS_EXPORT FileDropSupport : public QObject
{
Q_OBJECT
public:
struct FileSpec {
FileSpec(const QString &path, int r = -1, int c = -1) : filePath(path), line(r), column(c) {}
QString filePath;
int line;
int column;
};
// returns true if the event should be accepted
typedef std::function<bool(QDropEvent*)> DropFilterFunction;
FileDropSupport(QWidget *parentWidget, const DropFilterFunction &filterFunction
= DropFilterFunction());
static QStringList mimeTypesForFilePaths();
signals:
void filesDropped(const QList<Utils::FileDropSupport::FileSpec> &files);
protected:
bool eventFilter(QObject *obj, QEvent *event);
private slots:
void emitFilesDropped();
private:
DropFilterFunction m_filterFunction;
QList<FileSpec> m_files;
};
class QTCREATOR_UTILS_EXPORT FileDropMimeData : public QMimeData
{
Q_OBJECT
public:
FileDropMimeData();
void setOverrideFileDropAction(Qt::DropAction action);
Qt::DropAction overrideFileDropAction() const;
bool isOverridingFileDropAction() const;
void addFile(const QString &filePath, int line = -1, int column = -1);
QList<FileDropSupport::FileSpec> files() const;
private:
QList<FileDropSupport::FileSpec> m_files;
Qt::DropAction m_overrideDropAction;
bool m_isOverridingDropAction;
};
} // namespace Utils } // namespace Utils
QT_BEGIN_NAMESPACE QT_BEGIN_NAMESPACE

View File

@@ -91,7 +91,8 @@ SOURCES += $$PWD/environment.cpp \
$$PWD/progressindicator.cpp \ $$PWD/progressindicator.cpp \
$$PWD/fadingindicator.cpp \ $$PWD/fadingindicator.cpp \
$$PWD/overridecursor.cpp \ $$PWD/overridecursor.cpp \
$$PWD/categorysortfiltermodel.cpp $$PWD/categorysortfiltermodel.cpp \
$$PWD/dropsupport.cpp
win32:SOURCES += $$PWD/consoleprocess_win.cpp win32:SOURCES += $$PWD/consoleprocess_win.cpp
else:SOURCES += $$PWD/consoleprocess_unix.cpp else:SOURCES += $$PWD/consoleprocess_unix.cpp
@@ -194,7 +195,8 @@ HEADERS += \
$$PWD/fadingindicator.h \ $$PWD/fadingindicator.h \
$$PWD/executeondestruction.h \ $$PWD/executeondestruction.h \
$$PWD/overridecursor.h \ $$PWD/overridecursor.h \
$$PWD/categorysortfiltermodel.h $$PWD/categorysortfiltermodel.h \
$$PWD/dropsupport.h
FORMS += $$PWD/filewizardpage.ui \ FORMS += $$PWD/filewizardpage.ui \
$$PWD/projectintropage.ui \ $$PWD/projectintropage.ui \

View File

@@ -64,6 +64,8 @@ QtcLibrary {
"detailsbutton.h", "detailsbutton.h",
"detailswidget.cpp", "detailswidget.cpp",
"detailswidget.h", "detailswidget.h",
"dropsupport.cpp",
"dropsupport.h",
"elfreader.cpp", "elfreader.cpp",
"elfreader.h", "elfreader.h",
"elidinglabel.cpp", "elidinglabel.cpp",

View File

@@ -46,6 +46,7 @@
#include <utils/qtcassert.h> #include <utils/qtcassert.h>
#include <utils/checkablemessagebox.h> #include <utils/checkablemessagebox.h>
#include <utils/theme/theme.h> #include <utils/theme/theme.h>
#include <utils/dropsupport.h>
#include <QAction> #include <QAction>
#include <QContextMenuEvent> #include <QContextMenuEvent>
@@ -420,12 +421,12 @@ Qt::DropActions BookmarkManager::supportedDragActions() const
QStringList BookmarkManager::mimeTypes() const QStringList BookmarkManager::mimeTypes() const
{ {
return FileDropSupport::mimeTypesForFilePaths(); return DropSupport::mimeTypesForFilePaths();
} }
QMimeData *BookmarkManager::mimeData(const QModelIndexList &indexes) const QMimeData *BookmarkManager::mimeData(const QModelIndexList &indexes) const
{ {
auto data = new FileDropMimeData; auto data = new DropMimeData;
foreach (const QModelIndex &index, indexes) { foreach (const QModelIndex &index, indexes) {
if (!index.isValid() || index.column() != 0 || index.row() < 0 || index.row() >= m_bookmarksList.count()) if (!index.isValid() || index.column() != 0 || index.row() < 0 || index.row() >= m_bookmarksList.count())
continue; continue;

View File

@@ -34,7 +34,7 @@
#include "classviewutils.h" #include "classviewutils.h"
#include <cplusplus/Icons.h> #include <cplusplus/Icons.h>
#include <utils/fileutils.h> #include <utils/dropsupport.h>
namespace ClassView { namespace ClassView {
namespace Internal { namespace Internal {
@@ -141,12 +141,12 @@ Qt::DropActions TreeItemModel::supportedDragActions() const
QStringList TreeItemModel::mimeTypes() const QStringList TreeItemModel::mimeTypes() const
{ {
return ::Utils::FileDropSupport::mimeTypesForFilePaths(); return ::Utils::DropSupport::mimeTypesForFilePaths();
} }
QMimeData *TreeItemModel::mimeData(const QModelIndexList &indexes) const QMimeData *TreeItemModel::mimeData(const QModelIndexList &indexes) const
{ {
auto mimeData = new ::Utils::FileDropMimeData; auto mimeData = new ::Utils::DropMimeData;
mimeData->setOverrideFileDropAction(Qt::CopyAction); mimeData->setOverrideFileDropAction(Qt::CopyAction);
foreach (const QModelIndex &index, indexes) { foreach (const QModelIndex &index, indexes) {
const QSet<SymbolLocation> locations = Utils::roleToLocations( const QSet<SymbolLocation> locations = Utils::roleToLocations(

View File

@@ -34,7 +34,7 @@
#include <coreplugin/idocument.h> #include <coreplugin/idocument.h>
#include <utils/algorithm.h> #include <utils/algorithm.h>
#include <utils/fileutils.h> #include <utils/dropsupport.h>
#include <utils/qtcassert.h> #include <utils/qtcassert.h>
#include <QAbstractItemModel> #include <QAbstractItemModel>
@@ -509,7 +509,7 @@ Qt::DropActions DocumentModelPrivate::supportedDragActions() const
QStringList DocumentModelPrivate::mimeTypes() const QStringList DocumentModelPrivate::mimeTypes() const
{ {
return Utils::FileDropSupport::mimeTypesForFilePaths(); return Utils::DropSupport::mimeTypesForFilePaths();
} }
DocumentModel::Entry *DocumentModel::entryAtRow(int row) DocumentModel::Entry *DocumentModel::entryAtRow(int row)
@@ -569,7 +569,7 @@ Qt::ItemFlags DocumentModelPrivate::flags(const QModelIndex &index) const
QMimeData *DocumentModelPrivate::mimeData(const QModelIndexList &indexes) const QMimeData *DocumentModelPrivate::mimeData(const QModelIndexList &indexes) const
{ {
auto data = new Utils::FileDropMimeData; auto data = new Utils::DropMimeData;
foreach (const QModelIndex &index, indexes) { foreach (const QModelIndex &index, indexes) {
const DocumentModel::Entry *e = DocumentModel::entryAtRow(index.row()); const DocumentModel::Entry *e = DocumentModel::entryAtRow(index.row());
if (!e || e->fileName().isEmpty()) if (!e || e->fileName().isEmpty())

View File

@@ -135,16 +135,18 @@ EditorView::EditorView(SplitterOrView *parentSplitterOrView, QWidget *parent) :
m_container->addWidget(empty); m_container->addWidget(empty);
m_widgetEditorMap.insert(empty, 0); m_widgetEditorMap.insert(empty, 0);
auto dropSupport = new FileDropSupport(this, [this](QDropEvent *event) -> bool { auto dropSupport = new DropSupport(this, [this](QDropEvent *event, DropSupport *dropSupport) -> bool {
// do not accept move events except from other editor views (i.e. their tool bars) // do not accept move events except from other editor views (i.e. their tool bars)
// otherwise e.g. item views that support moving items within themselves would // otherwise e.g. item views that support moving items within themselves would
// also "move" the item into the editor view, i.e. the item would be removed from the // also "move" the item into the editor view, i.e. the item would be removed from the
// item view // item view
if (!qobject_cast<EditorToolBar*>(event->source())) if (!qobject_cast<EditorToolBar*>(event->source()))
event->setDropAction(Qt::CopyAction); event->setDropAction(Qt::CopyAction);
if (event->type() == QDropEvent::DragEnter && !dropSupport->isFileDrop(event))
return false; // do not accept drops without files
return event->source() != m_toolBar; // do not accept drops on ourselves return event->source() != m_toolBar; // do not accept drops on ourselves
}); });
connect(dropSupport, &FileDropSupport::filesDropped, connect(dropSupport, &DropSupport::filesDropped,
this, &EditorView::openDroppedFiles); this, &EditorView::openDroppedFiles);
updateNavigatorActions(); updateNavigatorActions();
@@ -375,11 +377,11 @@ void EditorView::closeSplit()
EditorManagerPrivate::updateActions(); EditorManagerPrivate::updateActions();
} }
void EditorView::openDroppedFiles(const QList<FileDropSupport::FileSpec> &files) void EditorView::openDroppedFiles(const QList<DropSupport::FileSpec> &files)
{ {
const int count = files.size(); const int count = files.size();
for (int i = 0; i < count; ++i) { for (int i = 0; i < count; ++i) {
const FileDropSupport::FileSpec spec = files.at(i); const DropSupport::FileSpec spec = files.at(i);
EditorManagerPrivate::openEditorAt(this, spec.filePath, spec.line, spec.column, Id(), EditorManagerPrivate::openEditorAt(this, spec.filePath, spec.line, spec.column, Id(),
i < count - 1 ? EditorManager::DoNotChangeCurrentEditor i < count - 1 ? EditorManager::DoNotChangeCurrentEditor
| EditorManager::DoNotMakeVisible | EditorManager::DoNotMakeVisible

View File

@@ -33,7 +33,7 @@
#include "coreplugin/id.h" #include "coreplugin/id.h"
#include <utils/fileutils.h> #include <utils/dropsupport.h>
#include <QMap> #include <QMap>
#include <QList> #include <QList>
@@ -122,7 +122,7 @@ private slots:
void splitVertically(); void splitVertically();
void splitNewWindow(); void splitNewWindow();
void closeSplit(); void closeSplit();
void openDroppedFiles(const QList<Utils::FileDropSupport::FileSpec> &files); void openDroppedFiles(const QList<Utils::DropSupport::FileSpec> &files);
private: private:
friend class SplitterOrView; // for setParentSplitterOrView friend class SplitterOrView; // for setParentSplitterOrView

View File

@@ -456,7 +456,7 @@ bool EditorToolBar::eventFilter(QObject *obj, QEvent *event)
if (!entry) // no document if (!entry) // no document
return Utils::StyledBar::eventFilter(obj, event); return Utils::StyledBar::eventFilter(obj, event);
auto drag = new QDrag(this); auto drag = new QDrag(this);
auto data = new Utils::FileDropMimeData; auto data = new Utils::DropMimeData;
data->addFile(entry->fileName().toString()); data->addFile(entry->fileName().toString());
drag->setMimeData(data); drag->setMimeData(data);
Qt::DropAction action = drag->exec(Qt::MoveAction | Qt::CopyAction, Qt::MoveAction); Qt::DropAction action = drag->exec(Qt::MoveAction | Qt::CopyAction, Qt::MoveAction);

View File

@@ -207,10 +207,10 @@ MainWindow::MainWindow() :
statusBar()->setProperty("p_styled", true); statusBar()->setProperty("p_styled", true);
auto dropSupport = new FileDropSupport(this, [](QDropEvent *event) { auto dropSupport = new DropSupport(this, [](QDropEvent *event, DropSupport *) {
return event->source() == 0; // only accept drops from the "outside" (e.g. file manager) return event->source() == 0; // only accept drops from the "outside" (e.g. file manager)
}); });
connect(dropSupport, &FileDropSupport::filesDropped, connect(dropSupport, &DropSupport::filesDropped,
this, &MainWindow::openDroppedFiles); this, &MainWindow::openDroppedFiles);
} }
@@ -371,11 +371,11 @@ void MainWindow::closeEvent(QCloseEvent *event)
event->accept(); event->accept();
} }
void MainWindow::openDroppedFiles(const QList<FileDropSupport::FileSpec> &files) void MainWindow::openDroppedFiles(const QList<DropSupport::FileSpec> &files)
{ {
raiseWindow(); raiseWindow();
QStringList filePaths = Utils::transform(files, QStringList filePaths = Utils::transform(files,
[](const FileDropSupport::FileSpec &spec) -> QString { [](const DropSupport::FileSpec &spec) -> QString {
return spec.filePath; return spec.filePath;
}); });
openFiles(filePaths, ICore::SwitchMode); openFiles(filePaths, ICore::SwitchMode);

View File

@@ -35,7 +35,7 @@
#include "icore.h" #include "icore.h"
#include <utils/appmainwindow.h> #include <utils/appmainwindow.h>
#include <utils/fileutils.h> #include <utils/dropsupport.h>
#include <QMap> #include <QMap>
#include <QColor> #include <QColor>
@@ -135,7 +135,7 @@ private slots:
void updateFocusWidget(QWidget *old, QWidget *now); void updateFocusWidget(QWidget *old, QWidget *now);
void setSidebarVisible(bool visible); void setSidebarVisible(bool visible);
void destroyVersionDialog(); void destroyVersionDialog();
void openDroppedFiles(const QList<Utils::FileDropSupport::FileSpec> &files); void openDroppedFiles(const QList<Utils::DropSupport::FileSpec> &files);
void restoreWindowState(); void restoreWindowState();
void newItemDialogFinished(); void newItemDialogFinished();

View File

@@ -39,7 +39,7 @@
#include <texteditor/texteditor.h> #include <texteditor/texteditor.h>
#include <cplusplus/CppDocument.h> #include <cplusplus/CppDocument.h>
#include <utils/fileutils.h> #include <utils/dropsupport.h>
#include <utils/qtcassert.h> #include <utils/qtcassert.h>
#include <QSet> #include <QSet>
@@ -260,12 +260,12 @@ Qt::DropActions CppIncludeHierarchyModel::supportedDragActions() const
QStringList CppIncludeHierarchyModel::mimeTypes() const QStringList CppIncludeHierarchyModel::mimeTypes() const
{ {
return Utils::FileDropSupport::mimeTypesForFilePaths(); return Utils::DropSupport::mimeTypesForFilePaths();
} }
QMimeData *CppIncludeHierarchyModel::mimeData(const QModelIndexList &indexes) const QMimeData *CppIncludeHierarchyModel::mimeData(const QModelIndexList &indexes) const
{ {
auto data = new Utils::FileDropMimeData; auto data = new Utils::DropMimeData;
foreach (const QModelIndex &index, indexes) { foreach (const QModelIndex &index, indexes) {
const TextEditor::TextEditorWidget::Link link const TextEditor::TextEditorWidget::Link link
= index.data(LinkRole).value<TextEditor::TextEditorWidget::Link>(); = index.data(LinkRole).value<TextEditor::TextEditorWidget::Link>();

View File

@@ -40,6 +40,7 @@
#include <utils/algorithm.h> #include <utils/algorithm.h>
#include <utils/annotateditemdelegate.h> #include <utils/annotateditemdelegate.h>
#include <utils/navigationtreeview.h> #include <utils/navigationtreeview.h>
#include <utils/dropsupport.h>
#include <QApplication> #include <QApplication>
#include <QLabel> #include <QLabel>
@@ -245,12 +246,12 @@ Qt::DropActions CppTypeHierarchyModel::supportedDragActions() const
QStringList CppTypeHierarchyModel::mimeTypes() const QStringList CppTypeHierarchyModel::mimeTypes() const
{ {
return FileDropSupport::mimeTypesForFilePaths(); return DropSupport::mimeTypesForFilePaths();
} }
QMimeData *CppTypeHierarchyModel::mimeData(const QModelIndexList &indexes) const QMimeData *CppTypeHierarchyModel::mimeData(const QModelIndexList &indexes) const
{ {
auto data = new FileDropMimeData; auto data = new DropMimeData;
data->setOverrideFileDropAction(Qt::CopyAction); // do not remove the item from the model data->setOverrideFileDropAction(Qt::CopyAction); // do not remove the item from the model
foreach (const QModelIndex &index, indexes) { foreach (const QModelIndex &index, indexes) {
auto link = index.data(LinkRole).value<TextEditor::TextEditorWidget::Link>(); auto link = index.data(LinkRole).value<TextEditor::TextEditorWidget::Link>();

View File

@@ -34,10 +34,10 @@
#include <coreplugin/id.h> #include <coreplugin/id.h>
#include <QMap> #include <QMap>
#include <QVariant>
QT_BEGIN_NAMESPACE QT_BEGIN_NAMESPACE
class QByteArray; class QByteArray;
class QVariant;
class QDataStream; class QDataStream;
QT_END_NAMESPACE QT_END_NAMESPACE

View File

@@ -38,6 +38,7 @@
#include <utils/fileutils.h> #include <utils/fileutils.h>
#include <QMap> #include <QMap>
#include <QVariant>
QT_BEGIN_NAMESPACE QT_BEGIN_NAMESPACE
class QDir; class QDir;

View File

@@ -37,6 +37,7 @@
#include <coreplugin/fileiconprovider.h> #include <coreplugin/fileiconprovider.h>
#include <utils/algorithm.h> #include <utils/algorithm.h>
#include <utils/dropsupport.h>
#include <QDebug> #include <QDebug>
#include <QFileInfo> #include <QFileInfo>
@@ -313,7 +314,7 @@ Qt::ItemFlags FlatModel::flags(const QModelIndex &index) const
// We claim that everything is editable // We claim that everything is editable
// That's slightly wrong // That's slightly wrong
// We control the only view, and that one does the checks // We control the only view, and that one does the checks
Qt::ItemFlags f = Qt::ItemIsSelectable|Qt::ItemIsEnabled; Qt::ItemFlags f = Qt::ItemIsSelectable|Qt::ItemIsEnabled|Qt::ItemIsDragEnabled;
if (Node *node = nodeForIndex(index)) { if (Node *node = nodeForIndex(index)) {
if (node == m_rootNode) if (node == m_rootNode)
return 0; // no flags for session node... return 0; // no flags for session node...
@@ -321,8 +322,6 @@ Qt::ItemFlags FlatModel::flags(const QModelIndex &index) const
// either folder or file node // either folder or file node
if (node->supportedActions(node).contains(Rename)) if (node->supportedActions(node).contains(Rename))
f = f | Qt::ItemIsEditable; f = f | Qt::ItemIsEditable;
if (node->asFileNode())
f = f | Qt::ItemIsDragEnabled;
} }
} }
return f; return f;
@@ -490,16 +489,17 @@ Qt::DropActions FlatModel::supportedDragActions() const
QStringList FlatModel::mimeTypes() const QStringList FlatModel::mimeTypes() const
{ {
return Utils::FileDropSupport::mimeTypesForFilePaths(); return Utils::DropSupport::mimeTypesForFilePaths();
} }
QMimeData *FlatModel::mimeData(const QModelIndexList &indexes) const QMimeData *FlatModel::mimeData(const QModelIndexList &indexes) const
{ {
auto data = new Utils::FileDropMimeData; auto data = new Utils::DropMimeData;
foreach (const QModelIndex &index, indexes) { foreach (const QModelIndex &index, indexes) {
Node *node = nodeForIndex(index); Node *node = nodeForIndex(index);
if (node->asFileNode()) if (node->asFileNode())
data->addFile(node->path().toString()); data->addFile(node->path().toString());
data->addValue(QVariant::fromValue(node));
} }
return data; return data;
} }

View File

@@ -39,8 +39,8 @@
#include <qmljs/qmljsvalueowner.h> #include <qmljs/qmljsvalueowner.h>
#include <qmljstools/qmljsrefactoringchanges.h> #include <qmljstools/qmljsrefactoringchanges.h>
#include <utils/fileutils.h>
#include <utils/qtcassert.h> #include <utils/qtcassert.h>
#include <utils/dropsupport.h>
#include <coreplugin/icore.h> #include <coreplugin/icore.h>
#include <QDebug> #include <QDebug>
@@ -318,7 +318,7 @@ QStringList QmlOutlineModel::mimeTypes() const
{ {
QStringList types; QStringList types;
types << QLatin1String(INTERNAL_MIMETYPE); types << QLatin1String(INTERNAL_MIMETYPE);
types << Utils::FileDropSupport::mimeTypesForFilePaths(); types << Utils::DropSupport::mimeTypesForFilePaths();
return types; return types;
} }
@@ -327,7 +327,7 @@ QMimeData *QmlOutlineModel::mimeData(const QModelIndexList &indexes) const
{ {
if (indexes.count() <= 0) if (indexes.count() <= 0)
return 0; return 0;
auto data = new Utils::FileDropMimeData; auto data = new Utils::DropMimeData;
data->setOverrideFileDropAction(Qt::CopyAction); data->setOverrideFileDropAction(Qt::CopyAction);
QByteArray encoded; QByteArray encoded;
QDataStream stream(&encoded, QIODevice::WriteOnly); QDataStream stream(&encoded, QIODevice::WriteOnly);

View File

@@ -79,6 +79,7 @@
#include <coreplugin/find/basetextfind.h> #include <coreplugin/find/basetextfind.h>
#include <utils/linecolumnlabel.h> #include <utils/linecolumnlabel.h>
#include <utils/fileutils.h> #include <utils/fileutils.h>
#include <utils/dropsupport.h>
#include <utils/hostosinfo.h> #include <utils/hostosinfo.h>
#include <utils/mimetypes/mimedatabase.h> #include <utils/mimetypes/mimedatabase.h>
#include <utils/qtcassert.h> #include <utils/qtcassert.h>
@@ -7319,7 +7320,7 @@ void TextEditorLinkLabel::mouseMoveEvent(QMouseEvent *event)
if ((event->pos() - m_dragStartPosition).manhattanLength() < QApplication::startDragDistance()) if ((event->pos() - m_dragStartPosition).manhattanLength() < QApplication::startDragDistance())
return; return;
auto data = new FileDropMimeData; auto data = new DropMimeData;
data->addFile(m_link.targetFileName, m_link.targetLine, m_link.targetColumn); data->addFile(m_link.targetFileName, m_link.targetLine, m_link.targetColumn);
auto drag = new QDrag(this); auto drag = new QDrag(this);
drag->setMimeData(data); drag->setMimeData(data);

View File

@@ -37,12 +37,12 @@
#include <QObject> #include <QObject>
#include <QStringList> #include <QStringList>
#include <QVariant>
#include <functional> #include <functional>
QT_BEGIN_NAMESPACE QT_BEGIN_NAMESPACE
class QFileInfo; class QFileInfo;
class QVariant;
class QProcessEnvironment; class QProcessEnvironment;
QT_END_NAMESPACE QT_END_NAMESPACE