336 lines
8.2 KiB
C++
336 lines
8.2 KiB
C++
/////////////////////////////////////////////////////////////////////////////
|
|
// Name: src/qt/dnd.cpp
|
|
// Author: Peter Most
|
|
// Copyright: (c) Peter Most
|
|
// Licence: wxWindows licence
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
|
|
// For compilers that support precompilation, includes "wx.h".
|
|
#include "wx/wxprec.h"
|
|
|
|
#if wxUSE_DRAG_AND_DROP
|
|
|
|
#include "wx/dnd.h"
|
|
#include "wx/scopedarray.h"
|
|
|
|
#include "wx/qt/private/converter.h"
|
|
|
|
#include <QDrag>
|
|
#include <QWidget>
|
|
#include <QMimeData>
|
|
#include <QCloseEvent>
|
|
|
|
namespace
|
|
{
|
|
wxDragResult DropActionToDragResult(Qt::DropAction action)
|
|
{
|
|
switch ( action )
|
|
{
|
|
case Qt::IgnoreAction:
|
|
return wxDragCancel;
|
|
case Qt::CopyAction:
|
|
return wxDragCopy;
|
|
case Qt::MoveAction:
|
|
return wxDragMove;
|
|
case Qt::LinkAction:
|
|
return wxDragLink;
|
|
default:
|
|
return wxDragNone;
|
|
}
|
|
}
|
|
|
|
Qt::DropAction DragResultToDropAction(wxDragResult result)
|
|
{
|
|
switch ( result )
|
|
{
|
|
case wxDragCopy:
|
|
return Qt::CopyAction;
|
|
case wxDragMove:
|
|
return Qt::MoveAction;
|
|
case wxDragLink:
|
|
return Qt::LinkAction;
|
|
default:
|
|
return Qt::IgnoreAction;
|
|
}
|
|
}
|
|
|
|
void AddDataFormat(wxDataObject* dataObject, QMimeData* mimeData, const wxDataFormat& format)
|
|
{
|
|
const size_t data_size = dataObject->GetDataSize(format);
|
|
|
|
QByteArray data(static_cast<int>(data_size), Qt::Initialization());
|
|
dataObject->GetDataHere(format, data.data());
|
|
|
|
mimeData->setData(wxQtConvertString(format.GetMimeType()), data);
|
|
}
|
|
|
|
QMimeData* CreateMimeData(wxDataObject* dataObject)
|
|
{
|
|
QMimeData* mimeData = new QMimeData();
|
|
|
|
const size_t count = dataObject->GetFormatCount();
|
|
|
|
wxScopedArray<wxDataFormat> array(dataObject->GetFormatCount());
|
|
dataObject->GetAllFormats(array.get());
|
|
|
|
for ( size_t i = 0; i < count; i++ )
|
|
{
|
|
AddDataFormat(dataObject, mimeData, array[i]);
|
|
}
|
|
|
|
return mimeData;
|
|
}
|
|
}
|
|
|
|
namespace
|
|
{
|
|
class PendingMimeDataSetter
|
|
{
|
|
public:
|
|
PendingMimeDataSetter(const QMimeData*& targetMimeData, const QMimeData* mimeData)
|
|
: m_targetMimeData(targetMimeData)
|
|
{
|
|
m_targetMimeData = mimeData;
|
|
}
|
|
|
|
~PendingMimeDataSetter()
|
|
{
|
|
m_targetMimeData = NULL;
|
|
}
|
|
|
|
private:
|
|
const QMimeData*& m_targetMimeData;
|
|
};
|
|
}
|
|
|
|
class wxDropTarget::Impl : public QObject
|
|
{
|
|
public:
|
|
explicit Impl(wxDropTarget* dropTarget)
|
|
: m_dropTarget(dropTarget),
|
|
m_widget(NULL)
|
|
{
|
|
}
|
|
|
|
~Impl()
|
|
{
|
|
Disconnect();
|
|
}
|
|
|
|
void ConnectTo(QWidget* widget)
|
|
{
|
|
Disconnect();
|
|
|
|
m_widget = widget;
|
|
|
|
if ( m_widget != NULL )
|
|
{
|
|
m_widget->setAcceptDrops(true);
|
|
m_widget->installEventFilter(this);
|
|
}
|
|
}
|
|
|
|
void Disconnect()
|
|
{
|
|
if ( m_widget != NULL )
|
|
{
|
|
m_widget->setAcceptDrops(false);
|
|
m_widget->removeEventFilter(this);
|
|
m_widget = NULL;
|
|
}
|
|
}
|
|
|
|
virtual bool eventFilter(QObject* watched, QEvent* event) wxOVERRIDE
|
|
{
|
|
if ( m_dropTarget != NULL )
|
|
{
|
|
switch ( event->type() )
|
|
{
|
|
case QEvent::Drop:
|
|
OnDrop(event);
|
|
return true;
|
|
|
|
case QEvent::DragEnter:
|
|
OnEnter(event);
|
|
return true;
|
|
|
|
case QEvent::DragMove:
|
|
OnMove(event);
|
|
return true;
|
|
|
|
case QEvent::DragLeave:
|
|
OnLeave(event);
|
|
return true;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
return QObject::eventFilter(watched, event);
|
|
}
|
|
|
|
void OnEnter(QEvent* event)
|
|
{
|
|
event->accept();
|
|
|
|
QDragEnterEvent *e = static_cast<QDragEnterEvent*>(event);
|
|
const QPoint where = e->pos();
|
|
|
|
PendingMimeDataSetter setter(m_pendingMimeData, e->mimeData());
|
|
|
|
wxDragResult result = m_dropTarget->OnEnter(where.x(), where.y(), DropActionToDragResult(e->proposedAction()));
|
|
|
|
e->setDropAction(DragResultToDropAction(result));
|
|
}
|
|
|
|
void OnLeave(QEvent* event)
|
|
{
|
|
event->accept();
|
|
m_dropTarget->OnLeave();
|
|
}
|
|
|
|
void OnMove(QEvent* event)
|
|
{
|
|
event->accept();
|
|
|
|
QDragMoveEvent *e = static_cast<QDragMoveEvent*>(event);
|
|
const QPoint where = e->pos();
|
|
|
|
PendingMimeDataSetter setter(m_pendingMimeData, e->mimeData());
|
|
|
|
wxDragResult result = m_dropTarget->OnDragOver(where.x(), where.y(), DropActionToDragResult(e->proposedAction()));
|
|
|
|
e->setDropAction(DragResultToDropAction(result));
|
|
}
|
|
|
|
void OnDrop(QEvent* event)
|
|
{
|
|
event->accept();
|
|
|
|
const QDropEvent *e = static_cast<QDropEvent*>(event);
|
|
const QPoint where = e->pos();
|
|
|
|
PendingMimeDataSetter setter(m_pendingMimeData, e->mimeData());
|
|
|
|
if ( m_dropTarget->OnDrop(where.x(), where.y()) )
|
|
{
|
|
m_dropTarget->OnData(where.x(), where.y(), DropActionToDragResult(e->dropAction()));
|
|
}
|
|
}
|
|
|
|
const QMimeData* m_pendingMimeData;
|
|
wxDropTarget* m_dropTarget;
|
|
QWidget* m_widget;
|
|
};
|
|
|
|
wxDropTarget::wxDropTarget(wxDataObject *dataObject)
|
|
: wxDropTargetBase(dataObject),
|
|
m_pImpl(new Impl(this))
|
|
{
|
|
}
|
|
|
|
wxDropTarget::~wxDropTarget()
|
|
{
|
|
delete m_pImpl;
|
|
}
|
|
|
|
bool wxDropTarget::OnDrop(wxCoord WXUNUSED(x), wxCoord WXUNUSED(y))
|
|
{
|
|
return !GetMatchingPair().GetMimeType().empty();
|
|
}
|
|
|
|
wxDragResult wxDropTarget::OnData(wxCoord WXUNUSED(x), wxCoord WXUNUSED(y), wxDragResult def)
|
|
{
|
|
GetData();
|
|
return def;
|
|
}
|
|
|
|
bool wxDropTarget::GetData()
|
|
{
|
|
const wxDataFormat droppedFormat = GetMatchingPair();
|
|
|
|
const wxString mimeType = droppedFormat.GetMimeType();
|
|
if ( mimeType.empty() )
|
|
return false;
|
|
|
|
const QByteArray data = m_pImpl->m_pendingMimeData->data(wxQtConvertString(mimeType));
|
|
|
|
return m_dataObject->SetData(droppedFormat, data.size(), data.data());
|
|
}
|
|
|
|
wxDataFormat wxDropTarget::GetMatchingPair()
|
|
{
|
|
if ( m_pImpl->m_pendingMimeData == NULL || m_dataObject == NULL )
|
|
return wxFormatInvalid;
|
|
|
|
const QStringList formats = m_pImpl->m_pendingMimeData->formats();
|
|
for ( int i = 0; i < formats.count(); ++i )
|
|
{
|
|
const wxDataFormat format(wxQtConvertString(formats[i]));
|
|
|
|
if ( m_dataObject->IsSupportedFormat(format) )
|
|
{
|
|
return format;
|
|
}
|
|
}
|
|
return wxFormatInvalid;
|
|
}
|
|
|
|
void wxDropTarget::ConnectTo(QWidget* widget)
|
|
{
|
|
m_pImpl->ConnectTo(widget);
|
|
}
|
|
|
|
void wxDropTarget::Disconnect()
|
|
{
|
|
m_pImpl->Disconnect();
|
|
}
|
|
|
|
//##############################################################################
|
|
|
|
wxDropSource::wxDropSource(wxWindow *win,
|
|
const wxIcon &WXUNUSED(copy),
|
|
const wxIcon &WXUNUSED(move),
|
|
const wxIcon &WXUNUSED(none))
|
|
: m_parentWindow(win)
|
|
{
|
|
}
|
|
|
|
wxDropSource::wxDropSource(wxDataObject& data,
|
|
wxWindow *win,
|
|
const wxIcon &WXUNUSED(copy),
|
|
const wxIcon &WXUNUSED(move),
|
|
const wxIcon &WXUNUSED(none))
|
|
: m_parentWindow(win)
|
|
{
|
|
SetData(data);
|
|
}
|
|
|
|
wxDragResult wxDropSource::DoDragDrop(int flags /*=wxDrag_CopyOnly*/)
|
|
{
|
|
wxCHECK_MSG(m_data != NULL, wxDragNone, wxT("No data in wxDropSource!"));
|
|
wxCHECK_MSG(m_parentWindow != NULL, wxDragNone, wxT("NULL parent window in wxDropSource!"));
|
|
|
|
QDrag drag(m_parentWindow->GetHandle());
|
|
drag.setMimeData(CreateMimeData(m_data));
|
|
|
|
Qt::DropActions actions = Qt::CopyAction | Qt::MoveAction;
|
|
Qt::DropAction defaultAction = Qt::CopyAction;
|
|
switch ( flags )
|
|
{
|
|
case wxDrag_CopyOnly:
|
|
actions = Qt::CopyAction;
|
|
break;
|
|
case wxDrag_DefaultMove:
|
|
defaultAction = Qt::MoveAction;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return DropActionToDragResult(drag.exec(actions, defaultAction));
|
|
}
|
|
|
|
#endif // wxUSE_DRAG_AND_DROP
|