2010-08-09 23:28:56 +10:00
|
|
|
#include <assert.h>
|
|
|
|
|
2015-06-20 15:44:05 +10:00
|
|
|
struct VideoGDI : Video {
|
|
|
|
~VideoGDI() { term(); }
|
2010-08-09 23:28:56 +10:00
|
|
|
|
2015-06-12 23:14:38 +10:00
|
|
|
uint32_t* buffer = nullptr;
|
|
|
|
HBITMAP bitmap = nullptr;
|
|
|
|
HDC bitmapdc = nullptr;
|
2010-08-09 23:28:56 +10:00
|
|
|
BITMAPINFO bmi;
|
|
|
|
|
|
|
|
struct {
|
2015-06-12 23:14:38 +10:00
|
|
|
HWND handle = nullptr;
|
2010-08-09 23:28:56 +10:00
|
|
|
|
2015-06-20 15:44:05 +10:00
|
|
|
unsigned width = 0;
|
|
|
|
unsigned height = 0;
|
2010-08-09 23:28:56 +10:00
|
|
|
} settings;
|
|
|
|
|
2015-06-12 23:14:38 +10:00
|
|
|
auto cap(const string& name) -> bool {
|
2010-08-09 23:28:56 +10:00
|
|
|
if(name == Video::Handle) return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-06-12 23:14:38 +10:00
|
|
|
auto get(const string& name) -> any {
|
2010-08-09 23:28:56 +10:00
|
|
|
if(name == Video::Handle) return (uintptr_t)settings.handle;
|
2015-06-12 23:14:38 +10:00
|
|
|
return {};
|
2010-08-09 23:28:56 +10:00
|
|
|
}
|
|
|
|
|
2015-06-12 23:14:38 +10:00
|
|
|
auto set(const string& name, const any& value) -> bool {
|
|
|
|
if(name == Video::Handle && value.is<uintptr_t>()) {
|
|
|
|
settings.handle = (HWND)value.get<uintptr_t>();
|
2010-08-09 23:28:56 +10:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-06-12 23:14:38 +10:00
|
|
|
auto lock(uint32_t*& data, unsigned& pitch, unsigned width, unsigned height) -> bool {
|
2010-08-09 23:28:56 +10:00
|
|
|
settings.width = width;
|
|
|
|
settings.height = height;
|
|
|
|
|
|
|
|
pitch = 1024 * 4;
|
|
|
|
return data = buffer;
|
|
|
|
}
|
|
|
|
|
2015-06-12 23:14:38 +10:00
|
|
|
auto unlock() -> void {}
|
2010-08-09 23:28:56 +10:00
|
|
|
|
2015-06-12 23:14:38 +10:00
|
|
|
auto clear() -> void {}
|
2010-08-09 23:28:56 +10:00
|
|
|
|
2015-06-12 23:14:38 +10:00
|
|
|
auto refresh() -> void {
|
2010-08-09 23:28:56 +10:00
|
|
|
RECT rc;
|
|
|
|
GetClientRect(settings.handle, &rc);
|
|
|
|
|
|
|
|
SetDIBits(bitmapdc, bitmap, 0, settings.height, (void*)buffer, &bmi, DIB_RGB_COLORS);
|
|
|
|
HDC hdc = GetDC(settings.handle);
|
|
|
|
StretchBlt(hdc, rc.left, rc.top, rc.right, rc.bottom, bitmapdc, 0, 1024 - settings.height, settings.width, settings.height, SRCCOPY);
|
|
|
|
ReleaseDC(settings.handle, hdc);
|
|
|
|
}
|
|
|
|
|
2015-06-12 23:14:38 +10:00
|
|
|
auto init() -> bool {
|
2015-06-20 15:44:05 +10:00
|
|
|
buffer = (uint32_t*)memory::allocate(1024 * 1024 * sizeof(uint32_t));
|
|
|
|
|
2010-08-09 23:28:56 +10:00
|
|
|
HDC hdc = GetDC(settings.handle);
|
|
|
|
bitmapdc = CreateCompatibleDC(hdc);
|
|
|
|
assert(bitmapdc);
|
|
|
|
bitmap = CreateCompatibleBitmap(hdc, 1024, 1024);
|
|
|
|
assert(bitmap);
|
|
|
|
SelectObject(bitmapdc, bitmap);
|
|
|
|
ReleaseDC(settings.handle, hdc);
|
|
|
|
|
|
|
|
memset(&bmi, 0, sizeof(BITMAPINFO));
|
|
|
|
bmi.bmiHeader.biSize = sizeof(BITMAPINFOHEADER);
|
|
|
|
bmi.bmiHeader.biWidth = 1024;
|
|
|
|
bmi.bmiHeader.biHeight = -1024;
|
|
|
|
bmi.bmiHeader.biPlanes = 1;
|
|
|
|
bmi.bmiHeader.biBitCount = 32; //biBitCount of 15 is invalid, biBitCount of 16 is really RGB555
|
|
|
|
bmi.bmiHeader.biCompression = BI_RGB;
|
|
|
|
bmi.bmiHeader.biSizeImage = 1024 * 1024 * sizeof(uint32_t);
|
|
|
|
|
|
|
|
settings.width = 256;
|
|
|
|
settings.height = 256;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-06-12 23:14:38 +10:00
|
|
|
auto term() -> void {
|
2010-08-09 23:28:56 +10:00
|
|
|
DeleteObject(bitmap);
|
|
|
|
DeleteDC(bitmapdc);
|
2015-06-20 15:44:05 +10:00
|
|
|
if(buffer) { memory::free(buffer); buffer = nullptr; }
|
2010-08-09 23:28:56 +10:00
|
|
|
}
|
|
|
|
};
|