文章目录
- tiny tool - get_file_path_name_by_drop_file
- 概述
- 工程效果
- 收获的知识点
- vs2022工程, 必须自己设置对话框可以接受文件的风格
- vs2022建立的工程, 默认是unicode编码, 设置剪贴板数据时, 必须要设置为unicode的格式, 否则剪切板中只有第一个字符
- 工程主要实现
- END
 
tiny tool - get_file_path_name_by_drop_file
概述
用EPLAN做黑盒时, 需要插入元件图片. 觉得有点麻烦.
 用win10资源管理器自带的文件属性, 需要自己将路径和文件名拼在一起.
 自己做一个小工具, 将文件拖入程序, 然后自动得到文件全路径, 然后点击拷贝按钮, 拷贝进剪切板.
 在EPLAN插入图片时, 就可以用剪切板中的文件全路径了.
工程效果

收获的知识点
vs2022工程, 必须自己设置对话框可以接受文件的风格
BOOL CgetfilepathnamebydropfileDlg::OnInitDialog()
{
...
	this->DragAcceptFiles(TRUE); // 资源文件UI中, 已经没有接受文件这个选项, 只能用API来设置
	return TRUE;  // 除非将焦点设置到控件,否则返回 TRUE
}
vs2022建立的工程, 默认是unicode编码, 设置剪贴板数据时, 必须要设置为unicode的格式, 否则剪切板中只有第一个字符
BOOL CgetfilepathnamebydropfileDlg::EditCopy(VOID)
{
...
	::SetClipboardData(CF_TEXT | CF_UNICODETEXT, hglbCopy); // 如果不带 CF_UNICODETEXT 选项, 剪切板中只有第一个字符(即使是都是英文的字符串)
	// Close the clipboard. 
	::CloseClipboard();
	return TRUE;
}
开始用的代码片段都是MS官方的, 还不好使, 给自己整的愣住了.
 自己捣鼓了一会, 才怀疑到剪切版数据格式的设置问题上.
工程主要实现
// get_file_path_name_by_drop_fileDlg.cpp: 实现文件
//
#include "pch.h"
#include "framework.h"
#include "get_file_path_name_by_drop_file.h"
#include "get_file_path_name_by_drop_fileDlg.h"
#include "afxdialogex.h"
#ifdef _DEBUG
#define new DEBUG_NEW
#endif
// 用于应用程序“关于”菜单项的 CAboutDlg 对话框
class CAboutDlg : public CDialog
{
public:
	CAboutDlg();
// 对话框数据
#ifdef AFX_DESIGN_TIME
	enum { IDD = IDD_ABOUTBOX };
#endif
	protected:
	virtual void DoDataExchange(CDataExchange* pDX);    // DDX/DDV 支持
// 实现
protected:
	DECLARE_MESSAGE_MAP()
};
CAboutDlg::CAboutDlg() : CDialog(IDD_ABOUTBOX)
{
}
void CAboutDlg::DoDataExchange(CDataExchange* pDX)
{
	CDialog::DoDataExchange(pDX);
}
BEGIN_MESSAGE_MAP(CAboutDlg, CDialog)
END_MESSAGE_MAP()
// CgetfilepathnamebydropfileDlg 对话框
CgetfilepathnamebydropfileDlg::CgetfilepathnamebydropfileDlg(CWnd* pParent /*=nullptr*/)
	: CDialog(IDD_GET_FILE_PATH_NAME_BY_DROP_FILE_DIALOG, pParent)
{
	m_hIcon = AfxGetApp()->LoadIcon(IDR_MAINFRAME);
}
void CgetfilepathnamebydropfileDlg::DoDataExchange(CDataExchange* pDX)
{
	CDialog::DoDataExchange(pDX);
	DDX_Control(pDX, IDC_EDIT_FILE_PATH_NAME, m_ctrlFilePathName);
}
BEGIN_MESSAGE_MAP(CgetfilepathnamebydropfileDlg, CDialog)
	ON_WM_SYSCOMMAND()
	ON_WM_PAINT()
	ON_WM_QUERYDRAGICON()
	ON_BN_CLICKED(IDC_BTN_COPY, &CgetfilepathnamebydropfileDlg::OnBnClickedBtnCopy)
	ON_WM_DROPFILES()
END_MESSAGE_MAP()
// CgetfilepathnamebydropfileDlg 消息处理程序
BOOL CgetfilepathnamebydropfileDlg::OnInitDialog()
{
	CDialog::OnInitDialog();
	// 将“关于...”菜单项添加到系统菜单中。
	// IDM_ABOUTBOX 必须在系统命令范围内。
	ASSERT((IDM_ABOUTBOX & 0xFFF0) == IDM_ABOUTBOX);
	ASSERT(IDM_ABOUTBOX < 0xF000);
	CMenu* pSysMenu = GetSystemMenu(FALSE);
	if (pSysMenu != nullptr)
	{
		BOOL bNameValid;
		CString strAboutMenu;
		bNameValid = strAboutMenu.LoadString(IDS_ABOUTBOX);
		ASSERT(bNameValid);
		if (!strAboutMenu.IsEmpty())
		{
			pSysMenu->AppendMenu(MF_SEPARATOR);
			pSysMenu->AppendMenu(MF_STRING, IDM_ABOUTBOX, strAboutMenu);
		}
	}
	// 设置此对话框的图标。  当应用程序主窗口不是对话框时,框架将自动
	//  执行此操作
	SetIcon(m_hIcon, TRUE);			// 设置大图标
	SetIcon(m_hIcon, FALSE);		// 设置小图标
	// TODO: 在此添加额外的初始化代码
	this->DragAcceptFiles(TRUE); // 资源文件UI中, 已经没有接受文件这个选项, 只能用API来设置
	return TRUE;  // 除非将焦点设置到控件,否则返回 TRUE
}
void CgetfilepathnamebydropfileDlg::OnSysCommand(UINT nID, LPARAM lParam)
{
	if ((nID & 0xFFF0) == IDM_ABOUTBOX)
	{
		CAboutDlg dlgAbout;
		dlgAbout.DoModal();
	}
	else
	{
		CDialog::OnSysCommand(nID, lParam);
	}
}
// 如果向对话框添加最小化按钮,则需要下面的代码
//  来绘制该图标。  对于使用文档/视图模型的 MFC 应用程序,
//  这将由框架自动完成。
void CgetfilepathnamebydropfileDlg::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 CgetfilepathnamebydropfileDlg::OnQueryDragIcon()
{
	return static_cast<HCURSOR>(m_hIcon);
}
BOOL CgetfilepathnamebydropfileDlg::EditCopy(VOID)
{
	LPTSTR  lptstrCopy;
	HGLOBAL hglbCopy;
	int cch = 0;
	// Open the clipboard, and empty it. 
	if (!::OpenClipboard(NULL))
		return FALSE;
	::EmptyClipboard();
	CString str;
	m_ctrlFilePathName.GetWindowTextW(str);
	cch = str.GetLength();
	hglbCopy = ::GlobalAlloc(GMEM_MOVEABLE,
		(cch + 1) * sizeof(TCHAR));
	if (hglbCopy == NULL)
	{
		::CloseClipboard();
		return FALSE;
	}
	// Lock the handle and copy the text to the buffer. 
	lptstrCopy = (LPTSTR)::GlobalLock(hglbCopy);
	memcpy(lptstrCopy, str,
		cch * sizeof(TCHAR));
	lptstrCopy[cch] = (TCHAR)'\0';    // null character 
	::GlobalUnlock(hglbCopy);
	// Place the handle on the clipboard. 
	::SetClipboardData(CF_TEXT | CF_UNICODETEXT, hglbCopy); // 如果不带 CF_UNICODETEXT 选项, 剪切板中只有第一个字符(即使是都是英文的字符串)
	// Close the clipboard. 
	::CloseClipboard();
	return TRUE;
}
void CgetfilepathnamebydropfileDlg::OnBnClickedBtnCopy()
{
	// TODO: 在此添加控件通知处理程序代码
	EditCopy();
}
void CgetfilepathnamebydropfileDlg::OnDropFiles(HDROP hDropInfo)
{
	// TODO: 在此添加消息处理程序代码和/或调用默认值
	TCHAR szTmp[MAX_PATH * 2];
	UINT nRc = 0;
	nRc = ::DragQueryFile(hDropInfo, 0, szTmp, sizeof(szTmp) / sizeof(szTmp[0]));
	if (nRc > 0) {
		m_ctrlFilePathName.SetWindowTextW(szTmp);
	}
	::DragFinish(hDropInfo);
	CDialog::OnDropFiles(hDropInfo);
}


















![[RocketMQ] Broker接收消息入口源码 (九)](https://img-blog.csdnimg.cn/cd572fe87f2f44fd8c0c082719ee6999.png)
