实现了一个拖放功能,可以支持将文本、图片(hbitmap)和文件从其他窗口拖放到 AutoHotkey v2 GUI 控件中。它通过 Windows 的 IDropTarget 接口实现拖放操作,使用了一些低级 API 调用和 COM 编程技巧。

 

#Requires AutoHotkey v2.0
; 来源:https://github.com/Tebayaki/AutoHotkeyScripts/blob/main/lib/RegisterDragDrop.ahk

; description Supports dragging text and hbitmap from other process.

; example
myGui := Gui()
myPic := myGui.AddPicture("w500 h500 Border")
myGui.Show()

dragDropObj := RegisterDragDrop(myPic, OnDrop)
OnDrop(guiObj, data) {
    if data.BMP {
        guiObj.Value := "*w500 *h-1 hbitmap:*" data.BMP.Handle
    }
    else if data.Text {
        guiObj.Text := data.Text
    }
    else if data.Files {
        paths := ""
        for path in data.Files
            paths .= path "`n"
        guiObj.Text := paths
    }
}


RegisterDragDrop(guiObj, callback) {
    hwnd := (guiObj is Gui || guiObj is Gui.Control) ? guiObj.Hwnd : guiObj
    dropTarget := Buffer(9 * A_PtrSize)
    dropTarget._guiObj := guiObj
    dropTarget._callback := callback
    NumPut("ptr", dropTarget.Ptr + A_PtrSize,
        "ptr", CallbackCreate(QueryInterface),
        "ptr", CallbackCreate(AddRef),
        "ptr", CallbackCreate(Release),
        "ptr", CallbackCreate(DragEnter),
        "ptr", CallbackCreate(DragOver),
        "ptr", CallbackCreate(DragLeave),
        "ptr", CallbackCreate(Drop),
        "ptr", ObjPtr(dropTarget),
        dropTarget)
    dropTarget.__Delete := Destruct
    DllCall("ole32\RegisterDragDrop", "ptr", hwnd, "ptr", dropTarget, "hresult")
    return { __Delete: (_) => DllCall("ole32\RevokeDragDrop", "ptr", hwnd, "hresult") }

    static Destruct(this) {
        loop 7
            CallbackFree(NumGet(this, A_PtrSize * A_Index, "ptr"))
    }

    static QueryInterface(this, riid, ppvObject) {
        h := NumGet(riid, 0, "int64")
        l := NumGet(riid, 8, "int64")
        if (h == 0 && l == 0x46000000000000c0) || (h == 0x112 && l == 0x46000000000000c0) {
            NumPut("ptr", this, ppvObject)
            return 0
        }
        return 0x80004002
    }

    static AddRef(this) {
        refCount := ObjAddRef(NumGet(this, 8 * A_PtrSize, "ptr"))
        ;@Debug-Output => AddRef: {refCount}
        return refCount
    }

    static Release(this) {
        refCount := ObjRelease(NumGet(this, 8 * A_PtrSize, "ptr"))
        ;@Debug-Output => Release: {refCount}
        return refCount
    }

    static DragEnter(this, pDataObj, grfKeyState, pt, pdwEffect) {
        NumPut("uint", 1, pdwEffect) ; DROPEFFECT_COPY
        return 0
    }

    static DragOver(this, grfKeyState, pt, pdwEffect) {
        NumPut("uint", 1, pdwEffect) ; DROPEFFECT_COPY
        return 0
    }

    static DragLeave(this) {
        return 0
    }

    static Drop(this, pDataObj, grfKeyState, pt, pdwEffect) {
        effect := 0
        formatEtc := Buffer(A_PtrSize == 8 ? 32 : 20, 0)
        stgMedium := Buffer(A_PtrSize == 8 ? 24 : 12, 0)
        NumPut("uint", 1, formatEtc, A_PtrSize * 2) ; dwAspect = DVASPECT_CONTENT
        NumPut("int", -1, formatEtc, A_PtrSize * 2 + 4) ; lindex = -1

        NumPut("ushort", 13, formatEtc, 0) ; cfFormat = CF_UNICODETEXT
        NumPut("uint", 1, formatEtc, A_PtrSize * 2 + 8) ; tymed = TYMED_HGLOBAL
        hr := ComCall(3, pDataObj, "ptr", formatEtc, "ptr", stgMedium, "int")
        if hr == 0 {
            hGlobal := NumGet(stgMedium, A_PtrSize, "ptr")
            pData := DllCall("GlobalLock", "ptr", hGlobal, "ptr")
            if pData {
                text := StrGet(pData)
            }
            DllCall("GlobalUnlock", "ptr", hGlobal)
            if NumGet(stgMedium, A_PtrSize * 2, "ptr") == 0 {
                DllCall("GlobalFree", "ptr", hGlobal)
            }
            effect := 1 ; DROPEFFECT_COPY
        }

        NumPut("ushort", 15, formatEtc, 0) ; cfFormat = CF_HDROP
        NumPut("uint", 1, formatEtc, A_PtrSize * 2 + 8) ; tymed = TYMED_HGLOBAL
        hr := ComCall(3, pDataObj, "ptr", formatEtc, "ptr", stgMedium, "int")
        if hr == 0 {
            if hDrop := NumGet(stgMedium, A_PtrSize, "ptr") {
                cnt := DllCall("shell32\DragQueryFileW", "ptr", hDrop, "uint", -1, "ptr", 0, "uint", 0, "uint")
                files := []
                loop cnt {
                    if cc := DllCall("shell32\DragQueryFileW", "ptr", hDrop, "uint", A_Index - 1, "ptr", 0, "uint", 0, "uint") {
                        VarSetStrCapacity(&path, cc + 1)
                        if DllCall("shell32\DragQueryFileW", "ptr", hDrop, "uint", A_Index - 1, "str", &path, "uint", cc + 1, "uint") {
                            files.Push(path)
                        }
                    }
                }
                if files.Length == 0 {
                    files := ""
                }
                if NumGet(stgMedium, A_PtrSize * 2, "ptr") == 0 {
                    DllCall("GlobalFree", "ptr", hDrop)
                }
            }
            effect := 1 ; DROPEFFECT_COPY
        }

        NumPut("ushort", 2, formatEtc, 0) ; cfFormat = CF_BITMAP
        NumPut("uint", 16, formatEtc, A_PtrSize * 2 + 8) ; tymed = TYMED_GDI
        hr := ComCall(3, pDataObj, "ptr", formatEtc, "ptr", stgMedium, "int")
        if hr == 0 {
            if hBitmap := NumGet(stgMedium, A_PtrSize, "ptr") {
                if NumGet(stgMedium, A_PtrSize * 2, "ptr") == 0 || hBitmap := DllCall("CopyImage", "ptr", hBitmap, "uint", 0, "int", 0, "int", 0, "uint", 0, "ptr") {
                    bmp := { Handle: hBitmap, __Delete: (_) => DllCall("DeleteObject", "ptr", _.Handle) }
                }
            }
            effect := 1 ; DROPEFFECT_COPY
        }

        if effect {
            dropTarget := ObjFromPtrAddRef(NumGet(this, 8 * A_PtrSize, "ptr"))
            (dropTarget._callback)(dropTarget._guiObj, { Text: text ?? "", BMP: bmp ?? "", Files: files ?? "" })
        }
        NumPut("uint", effect, pdwEffect)
        return 0
    }
}

 

声明:站内资源为整理优化好的代码上传分享与学习研究,如果是开源代码基本都会标明出处,方便大家扩展学习路径。请不要恶意搬运,破坏站长辛苦整理维护的劳动成果。本站为爱好者分享站点,所有内容不作为商业行为。如若本站上传内容侵犯了原著者的合法权益,请联系我们进行删除下架。