首先,澄清一下,我不询问如何将文件拖放到exe的图标上。我想知道如何处理拖放到已经运行的 win32控制台应用程序。我也不询问如何通过Windows消息泵处理基于WinMain的应用程序内部的拖放操作。我想在一个程序中使用入口点 int main()执行此操作,该程序没有WndProc(尚未)或任何东西。
那就是说,我想知道我的目标是否可以实现(并希望它是)。
我有一个在控制台窗口中运行的服务器应用程序。由于代码库很大并且有许多奇怪的耦合,它是一个“仅输出”控制台,适用于所有密集目的。尽管如此,我仍然可以处理按键等操作,因为我有更新循环滴答声。我希望能够将完整命令(使用自定义语法)的文件拖放到我正在运行的应用程序中并让它处理它们。
这可能吗?我在想,我可能会得到一个指向控制台HWND的指针(希望这是一个东西?),然后可能将该窗口子类化为使用自定义WndProc来监听WM_DROPFILES消息。
我从未真正尝试在 int main()程序中设置windows消息的处理,而不是 WinMain 程序,但我希望它能以某种方式可能的。
任何帮助将不胜感激! 奇怪的解决方案很好。
答案 0 :(得分:6)
AFAIK,控制台窗口默认不支持拖放。您始终可以使用自己的消息循环创建自己的单独弹出窗口,以便用户可以将项目拖动到其中。
要在控制台窗口本身上使用拖放功能,请尝试使用GetConsoleWindow()
获取控制台HWND,然后执行以下任一操作:
使用SetWindowLong/Ptr()
或SetWindowSubClass()
对HWND进行子类化,然后使用DragAcceptFiles()
注册HWND以开始接收WM_DROPFILES
条消息。请务必再次致电DragAcceptFiles()
以停止接收消息,然后在退出应用之前取消挂钩子类。
实现IDropTarget
接口,然后使用RegisterDragDrop()
注册HWND以开始接收通知。请务必在退出应用之前致电RevokeDragDrop()
。
WM_DROPFILES
更容易编码,但IDropTarget
处理虚拟物品和物理文件时更灵活。
答案 1 :(得分:1)
#include <vector>
#include <string>
#include <iostream>
#include <conio.h>
int main()
{
std::cout << "Please drop files and press [Enter] when done ...\n";
std::vector< std::string > files;
for( int ch = _getch(); ch != '\r'; ch = _getch() ) {
std::string file_name;
if( ch == '\"' ) { // path containing spaces. read til next '"' ...
while( ( ch = _getch() ) != '\"' )
file_name += ch;
} else { // path not containing spaces. read as long as chars are coming rapidly.
file_name += ch;
while( _kbhit() )
file_name += _getch();
}
files.push_back( file_name );
}
std::cout << "You dropped these files:\n";
for( auto & i : files )
std::cout << i << '\n';
}