C# Windows Forms - Drag & Drop 예제 코드
아래의 답변을 하면서,
drag&drop 관련해서 문의 드립니다.
; https://www.sysnet.pe.kr/3/0/4756
만든 예제 코드가 폐기하자니 아까워서 ^^ 그냥 살짝 설명과 함께 실어봅니다.
우선, C# Windows Forms 응용 프로그램에서 Form에 Drag & Drop을 지원하려면 다음과 같이 3가지 코드 설정만 해주시면 됩니다.
public Form1()
{
InitializeComponent();
this.AllowDrop = true;
this.DragDrop += Form1_DragDrop;
this.DragEnter += Form1_DragEnter;
}
AllowDrop을 true로 설정하면 내부적으로는 C++에서 했던 RegisterDragDrop Win32 API를 호출하면서,
RegisterDragDrop
; https://docs.microsoft.com/en-us/windows/win32/api/ole2/nf-ole2-registerdragdrop
Drag & Drop이 가능하다는 것을 시스템에 알립니다. 그다음 DragDrop, DragEnter 이벤트를 구독시켜주면 끝입니다.
private void Form1_DragEnter(object sender, DragEventArgs e)
{
if (e.Data.GetDataPresent(DataFormats.FileDrop)) e.Effect = DragDropEffects.Copy;
}
private void Form1_DragDrop(object sender, DragEventArgs e)
{
string[] files = (string[])e.Data.GetData(DataFormats.FileDrop);
foreach (string file in files)
{
this.listBox1.Items.Add(file); // 리스트 박스에 파일명 추가
}
}
이렇게만 해주고, 윈도우 탐색기에서 파일을 끌어다 폼 위에 놓으면 리스트 박스에 해당 파일명이 추가됩니다.
"
drag&drop 관련해서 문의 드립니다." 질문하신 분이 언급한 대로 이는 OLE Drag-and-drop이기 때문에 기존의 WM_DROPFILES로 했던 Win32 Event 방식과는 호환하지 않습니다. 대신 WM_DROPFILES는 Forms의 WndProc을 재정의해서 처리해야 합니다. 방법은, C/C++에서 했던 것과 동일한데 단지 약간의 P/Invoke만 추가하면 됩니다.
[DllImport("shell32.dll", CharSet = CharSet.Auto, SetLastError = true)]
private static extern int DragQueryFile(IntPtr hDrop, uint iFile, StringBuilder lpszFile, int cch);
protected override void WndProc(ref Message m)
{
if (m.Msg == 0x0233) // WM_DROPFILES == 0x0233
{
int countOfFiles = DragQueryFile(m.WParam, 0xFFFFFFFF, null, 0);
int error = Marshal.GetLastWin32Error();
if (error != 0)
{
string paramValue = m.WParam.ToString("x");
}
for (int i = 0; i <= countOfFiles; i++)
{
StringBuilder sb = new StringBuilder(1024);
DragQueryFile(m.WParam, (uint)i, sb, 1024);
if (sb != null)
{
if (sb.ToString() != "")
{
this.listBox1.Items.Add(sb.ToString());
}
}
}
}
base.WndProc(ref m);
}
그리고, UAC 환경 하에서는 권한이 낮은 응용 프로그램이 높은 응용 프로그램으로 윈도우 메시지를 보내는 것을 허용하지 않으므로 만약 여러분들이 개발한 C# Windows Forms 응용 프로그램이 높은 권한으로 실행된다면 다음의 글에 쓰인 대로,
UAC and Elevation
; https://helgeklein.com/blog/2010/03/how-to-enable-drag-and-drop-for-an-elevated-mfc-application-on-vistawindows-7/
[Tip] UAC 때문에 WM_DROPFILES 메시지를 받을수 없는 경우에 대한 처리
; http://www.tipssoft.com/bulletin/board.php?bo_table=FAQ&wr_id=1571
ChangeWindowMessageFilter Win23 API를 이용해 Drag & Drop 관련 이벤트를 등록하시면 됩니다.
ChangeWindowMessageFilter (WM_DROPFILES, MSGFLT_ADD);
ChangeWindowMessageFilter (0x0049, MSGFLT_ADD); // WM_COPYGLOBALDATA = 0x0049
그 외에 재미있는 점은, GetProp으로 RegisterDragDrop으로 등록된 IOleDropTarget을 얻을 수 있다는 것입니다.
CoInitialize(NULL);
ATOM atom = (DWORD)GlobalFindAtom(L"OleDropTargetInterface");
IUnknown *pBuffer = (IUnknown *)::GetProp(targetWindow, (LPCTSTR)(DWORD)atom);
if (pBuffer != nullptr)
{
IDropTarget *pRetVal;
if (SUCCEEDED(pBuffer->QueryInterface(IID_IDropTarget, (void **)&pRetVal)))
{
return 0;
}
DWORD dwEffect;
pRetVal->Drop(nullptr, 0, point, &dwEffect);
}
CoUninitialize();
하지만 아쉽게도 이렇게 구한 IUnknown 포인터는 마샬링되지 않은, 대상 EXE 공간 내에서만 유효한 주소 값이기 때문에 위의 코드를 외부 프로세스에서 사용할 수는 없습니다.
(
첨부한 파일은 이 글의 예제 코드를 포함합니다.)
[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]