0

我正在尝试使用带有 user32.dll 的系统调用来获取剪贴板的内容。我希望它是来自打印屏幕的图像数据。

现在我有这个:

if opened := openClipboard(0); !opened {
    fmt.Println("Failed to open Clipboard")
}

handle := getClipboardData(CF_BITMAP)

// get buffer

img, _, err := Decode(buffer)

我需要使用句柄将数据放入可读缓冲区。

我从 github 上的 AllenDang/w32 和 atotto/clipboard 获得了一些灵感。根据 atotto 的实现,以下内容适用于文本:

text := syscall.UTF16ToString((*[1 << 20]uint16)(unsafe.Pointer(handle))[:])

但是我怎样才能得到一个包含我可以解码的图像数据的缓冲区呢?

[更新]

按照@kostix 提供的解决方案,我编写了一个半工作示例:

image.RegisterFormat("bmp", "bmp", bmp.Decode, bmp.DecodeConfig)

if opened := w32.OpenClipboard(0); opened == false {
    fmt.Println("Error: Failed to open Clipboard")
}

//fmt.Printf("Format: %d\n", w32.EnumClipboardFormats(w32.CF_BITMAP))
handle := w32.GetClipboardData(w32.CF_DIB)
size := globalSize(w32.HGLOBAL(handle))
if handle != 0 {
    pData := w32.GlobalLock(w32.HGLOBAL(handle))
    if pData != nil {
        data := (*[1 << 25]byte)(pData)[:size]
        // The data is either in DIB format and missing the BITMAPFILEHEADER
        // or there are other issues since it can't be decoded at this point
        buffer := bytes.NewBuffer(data)
        img, _, err := image.Decode(buffer)
        if err != nil {
            fmt.Printf("Failed decoding: %s", err)
            os.Exit(1)
        }

        fmt.Println(img.At(0, 0).RGBA())
    }

    w32.GlobalUnlock(w32.HGLOBAL(pData))
}
w32.CloseClipboard()

AllenDang/w32 包含您需要的大部分内容,但有时您需要自己实现一些东西,例如 globalSize():

var (
    modkernel32    = syscall.NewLazyDLL("kernel32.dll")
    procGlobalSize = modkernel32.NewProc("GlobalSize")
)

func globalSize(hMem w32.HGLOBAL) uint {
    ret, _, _ := procGlobalSize.Call(uintptr(hMem))

    if ret == 0 {
        panic("GlobalSize failed")
    }

    return uint(ret)
}

也许有人会想出一个解决方案来获取 BMP 数据。与此同时,我将采取不同的路线。

4

1 回答 1

1

@JimB 是正确的:user32!GetClipboardData()返回 a ,并且那里HGLOBAL的注释示例建议使用to a) 全局锁定该句柄,并且 b) 生成指向它所引用的内存的正确指针。kernel32!GlobalLock()

完成后,您将需要kernel32!GlobalUnlock()手柄。

至于将从 Win32 API 函数获得的指针转换为 Go 可读的东西,通常的技巧是将指针转换为非常大的切片。引用“ Go wiki article on cgo”的“Turning C arrays into Go slices”

要创建由 C 数组支持的 Go 切片(不复制原始数据),需要在运行时获取此长度并使用类型转换为指向一个非常大的数组的指针,然后将其切片为您想要的长度(如果您使用 Go 1.2 > 或更高版本,还记得设置上限),例如(有关可运行示例,请参见http://play.golang.org/p/XuC0xqtAIC ):

import "C"
import "unsafe"
...
var theCArray *C.YourType = C.getTheArray()
length := C.getTheArrayLength()
slice := (*[1 << 30]C.YourType)(unsafe.Pointer(theCArray))[:length:length]

重要的是要记住,Go 垃圾收集器不会与这些数据交互,并且如果它从 C 端释放出来,任何使用切片的 Go 代码的行为都是不确定的。

在您的情况下,它会更简单:

h := GlobalLock()
defer GlobalUnlock(h)
length := somehowGetLengthOfImageInTheClipboard()
slice := (*[1 << 30]byte)(unsafe.Pointer((uintptr(h)))[:length:length]

然后你需要实际读取位图。

这取决于可用于从剪贴板导出的与设备无关的位图 (DIB) 的格式。

请参阅作为开始。

通常,BITMAPINFOHEADERetc 的定义很容易在 MSDN 站点上在线获得。

于 2016-12-20T06:11:06.910 回答