使用過QQ的人都知道,只要把檔案拖拽到訊息框中就可以傳送檔案了。那麼這種功能是如何?的呢?其實很簡單,只需要響應一個WM_DROPFILES訊息就可以了。
在基於對話方塊的程式中,預設是沒有這個訊息的,按下Ctrl+W,彈出類嚮導對話方塊,選擇Class Info標籤,在Message fileter下拉式清單中選擇Window,然後再點擊Message Maps標籤,就出現WM_DROPFILES訊息了,添加該訊息的響應函數:
void CTestDlg::OnDropFiles(HDROP hDropInfo)
{
// TODO: Add your message handler code here and/or call default
CDialog::OnDropFiles(hDropInfo);
}
另外,要讓對話方塊能夠接受檔案拖拽,還需要設定對話方塊屬性。在對話方塊上點擊右鍵,選擇Properties->Extended Styles,點選Accept files選項即可。
要獲得當前拖拽的檔案的完整檔案名稱(含路徑),只需要一個函數:
UINT DragQueryFile(
HDROP hDrop,
UINT iFile,
LPTSTR lpszFile,
UINT cch
);
參數解釋:
hDrop: HDROP標識符,即響應函數中的hDropInfo參數
iFile: 待查詢的檔案索引號,從0開始。可以同時拖拽多個檔案,因此就需要一個索引號來進行區分。如果該參數為0xFFFFFFFF,則該函數返回拖拽的檔案的個數
lpszFile: 用於存放檔案名稱的緩衝區首地址
cch: 緩衝區長度
傳回值:檔案名稱長度
另外,查詢完成後需要釋放系統分配記憶體,使用下面這個函數:
VOID DragFinish(
HDROP hDrop
);
下面是一個完整的程式碼範例,將檔案拖拽到對話方塊上後會彈出訊息框顯示完整檔案名稱:
void CTestDlg::OnDropFiles(HDROP hDropInfo)
{
// TODO: Add your message handler code here and/or call default
UINT count;
char filePath[200];
count = DragQueryFile(hDropInfo, 0xFFFFFFFF, NULL, 0);
if(count)
{
for(UINT i=0; i<count; i++)
{
int pathLen = DragQueryFile(hDropInfo, i, filePath, sizeof(filePath));
AfxMessageBox(filePath);
}
}
DragFinish(hDropInfo);
CDialog::OnDropFiles(hDropInfo);
}
同理,如果只有把檔案拖拽到特定的控制項中時才有響應,只需要把該控制項的Accept files樣式勾選上即可。
==============================================================================
// ReclogCheckDlg.h : 標頭檔
//
#pragma once
// CReclogCheckDlg 對話方塊
class CReclogCheckDlg : public CDialog
{
// 構造
public:
CReclogCheckDlg(CWnd* pParent = NULL); // 標準建構函式
// 對話方塊資料
enum { IDD = IDD_RECLOGCHECK_DIALOG };
protected:
virtual void DoDataExchange(CDataExchange* pDX); // DDX/DDV 支援
// 實現
protected:
HICON m_hIcon;
// 產生的訊息映射函數
virtual BOOL OnInitDialog();
afx_msg void OnSysCommand(UINT nID, LPARAM lParam);
afx_msg void OnPaint();
afx_msg HCURSOR OnQueryDragIcon();
DECLARE_MESSAGE_MAP()
public:
afx_msg void OnDropFiles(HDROP hDropInfo);
};
==============================================================================
// ReclogCheckDlg.cpp : 實現檔案
//
#include "stdafx.h"
#include "ReclogCheck.h"
#include "ReclogCheckDlg.h"
#include "./reclogcheckdlg.h"
#ifdef _DEBUG
#define new DEBUG_NEW
#endif
// 用於應用程式“關於”功能表項目的 CAboutDlg 對話方塊
class CAboutDlg : public CDialog
{
public:
CAboutDlg();
// 對話方塊資料
enum { IDD = IDD_ABOUTBOX };
protected:
virtual void DoDataExchange(CDataExchange* pDX); // DDX/DDV 支援
// 實現
protected:
DECLARE_MESSAGE_MAP()
};
CAboutDlg::CAboutDlg() : CDialog(CAboutDlg::IDD)
{
}
void CAboutDlg::DoDataExchange(CDataExchange* pDX)
{
CDialog::DoDataExchange(pDX);
}
BEGIN_MESSAGE_MAP(CAboutDlg, CDialog)
END_MESSAGE_MAP()
// CReclogCheckDlg 對話方塊
CReclogCheckDlg::CReclogCheckDlg(CWnd* pParent /*=NULL*/)
: CDialog(CReclogCheckDlg::IDD, pParent)
{
m_hIcon = AfxGetApp()->LoadIcon(IDR_MAINFRAME);
}
void CReclogCheckDlg::DoDataExchange(CDataExchange* pDX)
{
CDialog::DoDataExchange(pDX);
}
BEGIN_MESSAGE_MAP(CReclogCheckDlg, CDialog)
ON_WM_SYSCOMMAND()
ON_WM_PAINT()
ON_WM_QUERYDRAGICON()
//}}AFX_MSG_MAP
ON_WM_DROPFILES()
END_MESSAGE_MAP()
// CReclogCheckDlg 訊息處理常式
BOOL CReclogCheckDlg::OnInitDialog()
{
CDialog::OnInitDialog();
// 將/“關於.../”功能表項目添加到系統功能表中。
// IDM_ABOUTBOX 必須在系統命令範圍內。
ASSERT((IDM_ABOUTBOX & 0xFFF0) == IDM_ABOUTBOX);
ASSERT(IDM_ABOUTBOX < 0xF000);
CMenu* pSysMenu = GetSystemMenu(FALSE);
if (pSysMenu != NULL)
{
CString strAboutMenu;
strAboutMenu.LoadString(IDS_ABOUTBOX);
if (!strAboutMenu.IsEmpty())
{
pSysMenu->AppendMenu(MF_SEPARATOR);
pSysMenu->AppendMenu(MF_STRING, IDM_ABOUTBOX, strAboutMenu);
}
}
// 設定此對話方塊的表徵圖。當應用程式主視窗不是對話方塊時,架構將自動
// 執行此操作
SetIcon(m_hIcon, TRUE); // 設定大表徵圖
SetIcon(m_hIcon, FALSE); // 設定小表徵圖
// TODO: 在此添加額外的初始化代碼
return TRUE; // 除非設定了控制項的焦點,否則返回 TRUE
}
void CReclogCheckDlg::OnSysCommand(UINT nID, LPARAM lParam)
{
if ((nID & 0xFFF0) == IDM_ABOUTBOX)
{
CAboutDlg dlgAbout;
dlgAbout.DoModal();
}
else
{
CDialog::OnSysCommand(nID, lParam);
}
}
// 如果向對話方塊添加最小化按鈕,則需要下面的代碼
// 來繪製該表徵圖。對於使用文檔/視圖模型的 MFC 應用程式,
// 這將由架構自動完成。
void CReclogCheckDlg::OnPaint()
{
if (IsIconic())
{
CPaintDC dc(this); // 用於繪製的裝置上下文
SendMessage(WM_ICONERASEBKGND, reinterpret_cast<WPARAM>(dc.GetSafeHdc()), 0);
// 使表徵圖在工作矩形中置中
int cxIcon = GetSystemMetrics(SM_CXICON);
int cyIcon = GetSystemMetrics(SM_CYICON);
CRect rect;
GetClientRect(&rect);
int x = (rect.Width() - cxIcon + 1) / 2;
int y = (rect.Height() - cyIcon + 1) / 2;
// 繪製表徵圖
dc.DrawIcon(x, y, m_hIcon);
}
else
{
CDialog::OnPaint();
}
}
//當使用者拖動已最小化的視窗時系統調用此函數取得游標顯示。
HCURSOR CReclogCheckDlg::OnQueryDragIcon()
{
return static_cast<HCURSOR>(m_hIcon);
}
void CReclogCheckDlg::OnDropFiles(HDROP hDropInfo)
{
// TODO: 在此添加訊息處理常式代碼和/或調用預設值
UINT count;
char filePath[200];
count = DragQueryFile(hDropInfo, 0xFFFFFFFF, NULL, 0);
if(count)
{
for(UINT i=0; i<count; i++)
{
int pathLen = DragQueryFile(hDropInfo, i, filePath, sizeof(filePath));
AfxMessageBox(filePath);
}
}
DragFinish(hDropInfo);
CDialog::OnDropFiles(hDropInfo);
}
==============================================================================