opencv — C ++ вывод захвата окна не совпадает с размером указанного окна

У меня есть короткая программа на C ++ для захвата пикселей из этого окна эмулятора BlueStacks, а затем я буду манипулировать пикселями с помощью OpenCV, а затем отправлять ввод с помощью мыши (win32api), основываясь на некоторых решениях. Я работаю в Visual Studio Enterprise 2017.

Единственная проблема в том, что функция, кажется, захватывает меньшую область пикселей по сравнению с полным окном.

Вот пример изображения. Исходное окно находится слева, зеркальный вывод — справа.

пример

Как я могу это исправить? Я уже включил поддержку высокого разрешения DPI в настройках своего проекта, не зная, какие еще шаги предпринять, кроме добавления случайных магических чисел в функцию.

Я сделал эту программу уже на Python, но я хотел бы повторить это в C ++ для повышения производительности.

Вот видео (предупреждение: шум). И еще один.

Вот мой текущий код:

#include "stdafx.h"#include "opencv2/imgproc.hpp"#include "opencv2/highgui.hpp"#include <Windows.h>
#include <iostream>

using namespace std;
using namespace cv;

Mat hwnd2mat(HWND hwnd);

int main(int argc, char **argv)
{
printf("Hello, world!\n");
Sleep(1000);

HWND hwndDesktop;

hwndDesktop = GetForegroundWindow();
// namedWindow("output", WINDOW_NORMAL);
int key = 0;
Mat src;

while (key != 27)
{
src = hwnd2mat(hwndDesktop);
// you can do some image processing here
imshow("output", src);
key = waitKey(1); // you can change wait time
}

}

Mat hwnd2mat(HWND hwnd)
{
HDC hwindowDC, hwindowCompatibleDC;

int height, width, srcheight, srcwidth;
HBITMAP hbwindow;
Mat src;
BITMAPINFOHEADER  bi;

hwindowDC = GetDC(hwnd);
hwindowCompatibleDC = CreateCompatibleDC(hwindowDC);
SetStretchBltMode(hwindowCompatibleDC, COLORONCOLOR);

RECT windowsize;    // get the height and width of the screen
GetClientRect(hwnd, &windowsize);

srcheight = windowsize.bottom;
srcwidth = windowsize.right;
height = windowsize.bottom / 1;  //change this to whatever size you want to resize to
width = windowsize.right / 1;

src.create(height, width, CV_8UC4);

// create a bitmap
hbwindow = CreateCompatibleBitmap(hwindowDC, width, height);
bi.biSize = sizeof(BITMAPINFOHEADER);    //http://msdn.microsoft.com/en-us/library/windows/window/dd183402%28v=vs.85%29.aspx
bi.biWidth = width;
bi.biHeight = -height;  //this is the line that makes it draw upside down or not
bi.biPlanes = 1;
bi.biBitCount = 32;
bi.biCompression = BI_RGB;
bi.biSizeImage = 0;
bi.biXPelsPerMeter = 0;
bi.biYPelsPerMeter = 0;
bi.biClrUsed = 0;
bi.biClrImportant = 0;

// use the previously created device context with the bitmap
SelectObject(hwindowCompatibleDC, hbwindow);
// copy from the window device context to the bitmap device context
StretchBlt(hwindowCompatibleDC, 0, 0, width, height, hwindowDC, 0, 0, srcwidth, srcheight, SRCCOPY); //change SRCCOPY to NOTSRCCOPY for wacky colors !
GetDIBits(hwindowCompatibleDC, hbwindow, 0, height, src.data, (BITMAPINFO *)&bi, DIB_RGB_COLORS);  //copy from hwindowCompatibleDC to hbwindow

// avoid memory leak
DeleteObject(hbwindow);
DeleteDC(hwindowCompatibleDC);
ReleaseDC(hwnd, hwindowDC);

return src;
}

1

Решение

Я думаю ты хочешь GetActiveWindow() вместо GetForegroundWindow()

Увидеть этот так ответ

0

Другие решения

Растяжение связано с масштабированием DPI. Ваша собственная программа не поддерживает DPI, другая программа, похоже, поддерживает DPI. Самый простой способ сделать так, чтобы ваша программа знала о DPI, — позвонив SetProcessDPIAware(); в начале программы.

Помимо примечания, HBITMAP дескриптор не должен быть выбран в контексте устройства при вызове GetDIBits, Вы можете переписать свой код как

SetProcessDPIAware();
...

Mat hwnd2mat(HWND hwnd)
{
RECT rc;
GetClientRect(hwnd, &rc);
int width = rc.right;
int height = rc.bottom;

Mat src;
src.create(height, width, CV_8UC4);

HDC hdc = GetDC(hwnd);
HDC memdc = CreateCompatibleDC(hdc);
HBITMAP hbitmap = CreateCompatibleBitmap(hdc, width, height);
HBITMAP oldbmp = (HBITMAP)SelectObject(memdc, hbitmap);

BitBlt(memdc, 0, 0, width, height, hdc, 0, 0, SRCCOPY);
SelectObject(memdc, oldbmp);

BITMAPINFOHEADER  bi = { sizeof(BITMAPINFOHEADER), width, -height, 1, 32, BI_RGB };
GetDIBits(hdc, hbitmap, 0, height, src.data, (BITMAPINFO*)&bi, DIB_RGB_COLORS);

DeleteObject(hbitmap);
DeleteDC(memdc);
ReleaseDC(hwnd, hdc);

return src;
}

Этот код будет работать только для родных программ Win32. Для других программ, таких как Chrome, WPF, приложения Qt, это покажет пустой экран. Вам нужно будет сделать снимок экрана с окном рабочего стола.

0