Windows中Drag&Drop初探(三)

至此,我们成功地为CListCtrlEx添加了文件拖入操作的支持。一个完整的拖放操作,还包括拖出动作,所以必须要为该类再添加拖出操作,即,将列表中的某一项或者多项拖出成为一个文件。这就需要用到另一个类:COleDataSource。具体步骤如下:

CListCtrlEx中加入一个COleDataSource的实例,并映射列表框的LVN_BEGINDRAG消息处理函数,在此我们添加拖出操作的代码。

实现拖出非常简单,只需要依次调用COleDataSource的三个函数即可:Empty用于清空原先对象中缓存的数据,CacheGlobalData用来缓存数据以进行拖放操作,最后调用DoDragDrop启动本次拖放操作。

但在调用之前,必须要做一些准备工作。主要的任务就是创建一个DROPFILES结构体,并拷贝要拖放的文件名到结构体后的内存中。DROPFILES结构体定义了CF_HDROP剪贴板格式,紧跟它后面的是一系列被拖放文件的路径名。它的定义如下:

typedef struct _DROPFILES

{

    DWORD     pFiles;  //文件名起始地址

    POINT      pt;     //鼠标放下的位置,坐标由fNC成员指定

    BOOL        fNC;    //TRUE表示适用屏幕坐标系,否则使用客户坐标系

    BOOL        fWide;  //文件名字符串是否使用宽字符

} DROPFILES, FAR* LPDROPFILES;

拖放之前的准备动作的代码如下:

uBufferSize = sizeof(DROPFILES) + uBufferSize + 1;

      hMemData = GlobalAlloc(GPTR,uBufferSize);

      ASSERT(hMemData != NULL);

      

      lpDropFiles = (LPDROPFILES)GlobalLock(hMemData); //锁定之,并设置相关成员

      ASSERT(lpDropFiles != NULL);

      lpDropFiles->pFiles = sizeof(DROPFILES);

#ifdef _UNICODE

       lpDropFiles->fWide = TRUE;

#else

       lpDropFiles->fWide = FALSE;

#endif

 

       //把选中的所有文件名依次复制到DROPFILES结构体后面(全局内存中)

       pItemPos = strSelectedList.GetHeadPosition();

       pszStart = (char*)((LPBYTE)lpDropFiles + sizeof(DROPFILES));

       while(pItemPos != NULL)

       {

              lstrcpy(pszStart, (LPCTSTR)strSelectedList.GetNext(pItemPos));

              pszStart = strchr(pszStart,'/0') + 1; //下次的起始位置是上一次结尾+1

       }

准备完毕之后就可以进行拖放了,拖放动作有DoDragDrop函数触发,其原型如下:

DROPEFFECT DoDragDrop(

DWORD dwEffects = DROPEFFECT_COPY|DROPEFFECT_MOVE|DROPEFFECT_LINK, LPCRECT lpRectStartDrag = NULL,

COleDropSource* pDropSource = NULL

);

这里,dwEffects指定了允许施加于本COleDataSource实例之上的动作集:剪切、复制或无动作。

    lpRectStartDrag指示拖放操作真正开始的矩形,如果鼠标没有移出该矩形,则拖放操作视作放弃处理。如果本成员设为NULL,则该起始矩形将为一个像素大小。

    pDropSource表明拖放所使用的COleDataSource对象。

而该函数的返回值,则表明本次拖放操作所实际产生的效果,至于具体产生何种效果,则由系统决定。譬如在拖放时按住Shift键,将产生剪切效果;按住Ctrl键,将产生复制效果,等等。

拖放的代码如下:

       m_oleDataSource.Empty();

       m_oleDataSource.CacheGlobalData(CF_HDROP, hMemData);

       DropResult = m_oleDataSource.DoDragDrop(DROPEFFECT_MOVE|DROPEFFECT_COPY);

最后一点要注意的是,在Windows NT 4.0以上的系统中,即使实际产生的是DROPEFFECT_MOVE动作,DoDragDrop函数也只返回DROPEFFECT_NONE。产生这个问题的原因在于,Windows NT 4.0Shell会直接移动文件本身来对移动操作进行优化。返回值DROPEFFECT_MOVE最初的含义,就是通知执行拖放操作的应用程序去删除原位置上的文件。但是因为Shell已经替应用程序完成了这个(删除)动作,所以,函数返回DROPEFFECT_NONE。要想知道文件是否真的被移动了也很简单,只要在函数返回之后检查一下原位置上的文件是否存在就可以了。

Windows 9x系列的操作系统也会对移动进行同样的优化动作,但是它不会返回DROPEFFECT_NONE来代替DROPEFFECT_MOVE。详细的解释参见MS知识库Q182219

PySide6 是一个用于构建跨平台桌面应用程序的 Python 框架,它是 Qt for Python 的官方实现。PySide6 提供了一套丰富的 GUI 组件,以及一套完整的拖放(Drag & Drop)功能,允许用户在程序内部或从外部将数据拖放到应用程序的部件(Widgets)上。 要使用 PySide6 实现拖放功能,通常需要使用几个相关的类和方法,主要包括 `QDrag`、`QDropEvent` 和 `QSpeciesEvent`。以下是一个简单的使用示例: 1. 为部件启用拖放功能,你需要重写部件的 `dragEnterEvent`、`dragMoveEvent`、`dropEvent` 等事件处理方法。 2. 创建一个 `QDrag` 对象,通常在 `mousePressEvent` 或 `mouseMoveEvent` 创建。 3. 使用 `QDrag` 对象调用 `exec_()` 方法,它会显示一个拖放光标,并等待用户完成拖放操作。 4. 在 `dropEvent` 方法处理拖放的数据。你需要调用 `QDropEvent` 提供的方法来获取数据。 示例代码可能如下: ```python from PySide6.QtWidgets import QApplication, QWidget, QLCDNumber, QVBoxLayout, QDrag, QMimeData from PySide6.QtCore import Qt class DragDropWidget(QWidget): def __init__(self): super().__init__() self.initUI() def initUI(self): lcd = QLCDNumber(self) layout = QVBoxLayout(self) layout.addWidget(lcd) self.setLayout(layout) self.setWindowTitle('PySide6 Drag & Drop Example') self.setAcceptDrops(True) # 开启拖放功能 def dragEnterEvent(self, event): if event.mimeData().hasUrls(): event.accept() else: event.ignore() def dropEvent(self, event): if event.mimeData().hasUrls(): event.accept() for url in event.mimeData().urls(): print(url.toString()) # 处理拖放的数据 else: event.ignore() if __name__ == '__main__': import sys app = QApplication(sys.argv) ex = DragDropWidget() ex.show() sys.exit(app.exec_()) ``` 在上述示例,我们创建了一个简单的窗口,其包含一个 `QLCDNumber` 组件,并为该窗口启用了拖放功能。当用户拖放文件到该窗口时,程序会打印出文件的路径。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值