2015-05-08 16:02:36 +00:00
|
|
|
/*
|
2023-06-23 11:04:32 +00:00
|
|
|
* PROJECT: PAINT for ReactOS
|
|
|
|
* LICENSE: LGPL-2.0-or-later (https://spdx.org/licenses/LGPL-2.0-or-later)
|
|
|
|
* PURPOSE: Some DIB related functions
|
2023-06-27 18:22:21 +00:00
|
|
|
* COPYRIGHT: Copyright 2015 Benedikt Freisen <b.freisen@gmx.net>
|
2015-05-08 16:02:36 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include "precomp.h"
|
|
|
|
|
2023-06-17 12:15:35 +00:00
|
|
|
INT g_fileSize = 0;
|
2023-03-28 13:31:26 +00:00
|
|
|
float g_xDpi = 96;
|
|
|
|
float g_yDpi = 96;
|
2023-06-17 12:15:35 +00:00
|
|
|
SYSTEMTIME g_fileTime;
|
2023-03-28 13:31:26 +00:00
|
|
|
|
2023-08-20 07:46:18 +00:00
|
|
|
#define WIDTHBYTES(i) (((i) + 31) / 32 * 4)
|
|
|
|
|
|
|
|
struct BITMAPINFOEX : BITMAPINFO
|
|
|
|
{
|
|
|
|
RGBQUAD bmiColorsExtra[256 - 1];
|
|
|
|
};
|
|
|
|
|
2015-05-08 16:02:36 +00:00
|
|
|
/* FUNCTIONS ********************************************************/
|
|
|
|
|
2023-06-04 02:24:00 +00:00
|
|
|
// Convert DPI (dots per inch) into PPCM (pixels per centimeter)
|
|
|
|
float PpcmFromDpi(float dpi)
|
2023-03-28 13:31:26 +00:00
|
|
|
{
|
2023-06-04 02:24:00 +00:00
|
|
|
// 1 DPI is 0.0254 meter. 1 centimeter is 1/100 meter.
|
|
|
|
return dpi / (0.0254f * 100.0f);
|
2023-03-28 13:31:26 +00:00
|
|
|
}
|
|
|
|
|
2015-05-08 16:02:36 +00:00
|
|
|
HBITMAP
|
|
|
|
CreateDIBWithProperties(int width, int height)
|
|
|
|
{
|
|
|
|
BITMAPINFO bmi;
|
|
|
|
ZeroMemory(&bmi, sizeof(BITMAPINFO));
|
|
|
|
bmi.bmiHeader.biSize = sizeof(BITMAPINFOHEADER);
|
|
|
|
bmi.bmiHeader.biWidth = width;
|
|
|
|
bmi.bmiHeader.biHeight = height;
|
|
|
|
bmi.bmiHeader.biPlanes = 1;
|
|
|
|
bmi.bmiHeader.biBitCount = 24;
|
|
|
|
return CreateDIBSection(NULL, &bmi, DIB_RGB_COLORS, NULL, NULL, 0);
|
|
|
|
}
|
|
|
|
|
2023-06-19 00:56:02 +00:00
|
|
|
HBITMAP
|
|
|
|
CreateMonoBitmap(int width, int height, BOOL bWhite)
|
|
|
|
{
|
|
|
|
HBITMAP hbm = CreateBitmap(width, height, 1, 1, NULL);
|
|
|
|
if (hbm == NULL)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
if (bWhite)
|
|
|
|
{
|
|
|
|
HDC hdc = CreateCompatibleDC(NULL);
|
|
|
|
HGDIOBJ hbmOld = SelectObject(hdc, hbm);
|
|
|
|
RECT rc = { 0, 0, width, height };
|
|
|
|
FillRect(hdc, &rc, (HBRUSH)GetStockObject(WHITE_BRUSH));
|
|
|
|
SelectObject(hdc, hbmOld);
|
|
|
|
DeleteDC(hdc);
|
|
|
|
}
|
|
|
|
|
|
|
|
return hbm;
|
|
|
|
}
|
|
|
|
|
2020-04-28 22:44:18 +00:00
|
|
|
HBITMAP
|
|
|
|
CreateColorDIB(int width, int height, COLORREF rgb)
|
|
|
|
{
|
|
|
|
HBITMAP ret = CreateDIBWithProperties(width, height);
|
|
|
|
if (!ret)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
if (rgb)
|
|
|
|
{
|
|
|
|
HDC hdc = CreateCompatibleDC(NULL);
|
|
|
|
HGDIOBJ hbmOld = SelectObject(hdc, ret);
|
|
|
|
RECT rc;
|
|
|
|
SetRect(&rc, 0, 0, width, height);
|
|
|
|
HBRUSH hbr = CreateSolidBrush(rgb);
|
|
|
|
FillRect(hdc, &rc, hbr);
|
|
|
|
DeleteObject(hbr);
|
|
|
|
SelectObject(hdc, hbmOld);
|
|
|
|
DeleteDC(hdc);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2023-06-19 00:56:02 +00:00
|
|
|
HBITMAP CopyMonoImage(HBITMAP hbm, INT cx, INT cy)
|
|
|
|
{
|
|
|
|
BITMAP bm;
|
2023-11-04 10:25:45 +00:00
|
|
|
if (!::GetObjectW(hbm, sizeof(bm), &bm))
|
2023-06-19 00:56:02 +00:00
|
|
|
return NULL;
|
|
|
|
|
|
|
|
if (cx == 0 || cy == 0)
|
|
|
|
{
|
|
|
|
cx = bm.bmWidth;
|
|
|
|
cy = bm.bmHeight;
|
|
|
|
}
|
|
|
|
|
2023-11-04 10:25:45 +00:00
|
|
|
HBITMAP hbmNew = ::CreateBitmap(cx, cy, 1, 1, NULL);
|
2023-06-19 00:56:02 +00:00
|
|
|
if (!hbmNew)
|
|
|
|
return NULL;
|
|
|
|
|
2023-11-04 10:25:45 +00:00
|
|
|
HDC hdc1 = ::CreateCompatibleDC(NULL);
|
|
|
|
HDC hdc2 = ::CreateCompatibleDC(NULL);
|
|
|
|
HGDIOBJ hbm1Old = ::SelectObject(hdc1, hbm);
|
|
|
|
HGDIOBJ hbm2Old = ::SelectObject(hdc2, hbmNew);
|
|
|
|
::StretchBlt(hdc2, 0, 0, cx, cy, hdc1, 0, 0, bm.bmWidth, bm.bmHeight, SRCCOPY);
|
|
|
|
::SelectObject(hdc1, hbm1Old);
|
|
|
|
::SelectObject(hdc2, hbm2Old);
|
|
|
|
::DeleteDC(hdc1);
|
|
|
|
::DeleteDC(hdc2);
|
2023-06-19 00:56:02 +00:00
|
|
|
return hbmNew;
|
|
|
|
}
|
|
|
|
|
2023-06-15 23:05:23 +00:00
|
|
|
HBITMAP CachedBufferDIB(HBITMAP hbm, int minimalWidth, int minimalHeight)
|
|
|
|
{
|
|
|
|
if (minimalWidth <= 0)
|
|
|
|
minimalWidth = 1;
|
|
|
|
if (minimalHeight <= 0)
|
|
|
|
minimalHeight = 1;
|
|
|
|
|
|
|
|
BITMAP bm;
|
2023-11-04 10:25:45 +00:00
|
|
|
if (!GetObjectW(hbm, sizeof(bm), &bm))
|
2023-06-15 23:05:23 +00:00
|
|
|
hbm = NULL;
|
|
|
|
|
|
|
|
if (hbm && minimalWidth <= bm.bmWidth && minimalHeight <= bm.bmHeight)
|
|
|
|
return hbm;
|
|
|
|
|
|
|
|
if (hbm)
|
|
|
|
DeleteObject(hbm);
|
|
|
|
|
|
|
|
return CreateDIBWithProperties((minimalWidth * 3) / 2, (minimalHeight * 3) / 2);
|
|
|
|
}
|
|
|
|
|
2015-05-08 16:02:36 +00:00
|
|
|
int
|
|
|
|
GetDIBWidth(HBITMAP hBitmap)
|
|
|
|
{
|
|
|
|
BITMAP bm;
|
2023-11-04 10:25:45 +00:00
|
|
|
::GetObjectW(hBitmap, sizeof(BITMAP), &bm);
|
2015-05-08 16:02:36 +00:00
|
|
|
return bm.bmWidth;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
GetDIBHeight(HBITMAP hBitmap)
|
|
|
|
{
|
|
|
|
BITMAP bm;
|
2023-11-04 10:25:45 +00:00
|
|
|
::GetObjectW(hBitmap, sizeof(BITMAP), &bm);
|
2015-05-08 16:02:36 +00:00
|
|
|
return bm.bmHeight;
|
|
|
|
}
|
|
|
|
|
2023-08-09 14:59:34 +00:00
|
|
|
BOOL SaveDIBToFile(HBITMAP hBitmap, LPCWSTR FileName, BOOL fIsMainFile, REFGUID guidFileType)
|
2015-05-08 16:02:36 +00:00
|
|
|
{
|
2023-09-10 13:28:28 +00:00
|
|
|
CWaitCursor waitCursor;
|
|
|
|
|
2023-03-28 13:31:26 +00:00
|
|
|
CImageDx img;
|
2016-09-26 19:53:42 +00:00
|
|
|
img.Attach(hBitmap);
|
2023-08-09 14:59:34 +00:00
|
|
|
HRESULT hr = img.SaveDx(FileName, guidFileType, g_xDpi, g_yDpi);
|
2016-09-26 19:53:42 +00:00
|
|
|
img.Detach();
|
2015-05-08 16:02:36 +00:00
|
|
|
|
2023-07-05 03:06:22 +00:00
|
|
|
if (FAILED(hr))
|
2020-04-28 23:39:47 +00:00
|
|
|
{
|
2023-07-05 03:06:22 +00:00
|
|
|
ShowError(IDS_SAVEERROR, FileName);
|
2020-04-28 22:44:18 +00:00
|
|
|
return FALSE;
|
2020-04-28 23:39:47 +00:00
|
|
|
}
|
2015-05-08 16:02:36 +00:00
|
|
|
|
2023-07-05 03:06:22 +00:00
|
|
|
if (!fIsMainFile)
|
|
|
|
return TRUE;
|
2015-05-08 16:02:36 +00:00
|
|
|
|
2023-07-05 03:06:22 +00:00
|
|
|
WIN32_FIND_DATAW find;
|
|
|
|
HANDLE hFind = ::FindFirstFileW(FileName, &find);
|
|
|
|
if (hFind == INVALID_HANDLE_VALUE)
|
|
|
|
{
|
|
|
|
ShowError(IDS_SAVEERROR, FileName);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
::FindClose(hFind);
|
2020-04-28 22:44:18 +00:00
|
|
|
|
2023-07-05 03:06:22 +00:00
|
|
|
SetFileInfo(FileName, &find, TRUE);
|
2023-06-17 12:15:35 +00:00
|
|
|
g_imageSaved = TRUE;
|
2020-04-28 22:44:18 +00:00
|
|
|
return TRUE;
|
2015-05-08 16:02:36 +00:00
|
|
|
}
|
|
|
|
|
2023-07-05 03:06:22 +00:00
|
|
|
void SetFileInfo(LPCWSTR name, LPWIN32_FIND_DATAW pFound, BOOL isAFile)
|
2015-05-08 16:02:36 +00:00
|
|
|
{
|
2023-07-05 03:06:22 +00:00
|
|
|
// update file time and size
|
|
|
|
if (pFound)
|
2020-04-28 22:44:18 +00:00
|
|
|
{
|
2023-07-05 03:06:22 +00:00
|
|
|
FILETIME ft;
|
|
|
|
::FileTimeToLocalFileTime(&pFound->ftLastWriteTime, &ft);
|
|
|
|
::FileTimeToSystemTime(&ft, &g_fileTime);
|
2023-03-28 13:31:26 +00:00
|
|
|
|
2023-07-05 03:06:22 +00:00
|
|
|
g_fileSize = pFound->nFileSizeLow;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2023-06-17 12:15:35 +00:00
|
|
|
ZeroMemory(&g_fileTime, sizeof(g_fileTime));
|
2023-07-05 03:06:22 +00:00
|
|
|
g_fileSize = 0;
|
2020-04-28 22:44:18 +00:00
|
|
|
}
|
|
|
|
|
2023-06-17 12:15:35 +00:00
|
|
|
// update g_szFileName
|
2020-04-28 22:44:18 +00:00
|
|
|
if (name && name[0])
|
2023-07-05 03:06:22 +00:00
|
|
|
{
|
|
|
|
CStringW strName = name;
|
|
|
|
::GetFullPathNameW(strName, _countof(g_szFileName), g_szFileName, NULL);
|
|
|
|
// The following code won't work correctly when (name == g_szFileName):
|
|
|
|
// ::GetFullPathNameW(name, _countof(g_szFileName), g_szFileName, NULL);
|
|
|
|
}
|
2020-04-28 22:44:18 +00:00
|
|
|
else
|
2023-07-05 03:06:22 +00:00
|
|
|
{
|
|
|
|
::LoadStringW(g_hinstExe, IDS_DEFAULTFILENAME, g_szFileName, _countof(g_szFileName));
|
|
|
|
}
|
2020-04-28 22:44:18 +00:00
|
|
|
|
|
|
|
// set title
|
2023-11-04 10:25:45 +00:00
|
|
|
CStringW strTitle;
|
|
|
|
strTitle.Format(IDS_WINDOWTITLE, PathFindFileNameW(g_szFileName));
|
2020-04-28 22:44:18 +00:00
|
|
|
mainWindow.SetWindowText(strTitle);
|
|
|
|
|
|
|
|
// update file info and recent
|
2023-07-05 03:06:22 +00:00
|
|
|
g_isAFile = isAFile;
|
2023-06-17 12:15:35 +00:00
|
|
|
if (g_isAFile)
|
|
|
|
registrySettings.SetMostRecentFile(g_szFileName);
|
2020-04-28 22:44:18 +00:00
|
|
|
|
2023-06-17 12:15:35 +00:00
|
|
|
g_imageSaved = TRUE;
|
2023-07-05 03:06:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
HBITMAP InitializeImage(LPCWSTR name, LPWIN32_FIND_DATAW pFound, BOOL isFile)
|
|
|
|
{
|
|
|
|
COLORREF white = RGB(255, 255, 255);
|
|
|
|
HBITMAP hBitmap = CreateColorDIB(registrySettings.BMPWidth, registrySettings.BMPHeight, white);
|
|
|
|
if (hBitmap == NULL)
|
2023-10-24 17:35:49 +00:00
|
|
|
{
|
|
|
|
ShowOutOfMemory();
|
2023-07-05 03:06:22 +00:00
|
|
|
return NULL;
|
2023-10-24 17:35:49 +00:00
|
|
|
}
|
2023-07-05 03:06:22 +00:00
|
|
|
|
|
|
|
HDC hScreenDC = ::GetDC(NULL);
|
2023-10-12 10:43:29 +00:00
|
|
|
g_xDpi = (float)::GetDeviceCaps(hScreenDC, LOGPIXELSX);
|
|
|
|
g_yDpi = (float)::GetDeviceCaps(hScreenDC, LOGPIXELSY);
|
2023-07-05 03:06:22 +00:00
|
|
|
::ReleaseDC(NULL, hScreenDC);
|
|
|
|
|
|
|
|
return SetBitmapAndInfo(hBitmap, name, pFound, isFile);
|
|
|
|
}
|
2020-04-28 22:44:18 +00:00
|
|
|
|
2023-07-05 03:06:22 +00:00
|
|
|
HBITMAP SetBitmapAndInfo(HBITMAP hBitmap, LPCWSTR name, LPWIN32_FIND_DATAW pFound, BOOL isFile)
|
|
|
|
{
|
|
|
|
// update image
|
2023-10-16 22:25:50 +00:00
|
|
|
canvasWindow.updateScrollPos();
|
2023-07-05 03:06:22 +00:00
|
|
|
imageModel.PushImageForUndo(hBitmap);
|
|
|
|
imageModel.ClearHistory();
|
|
|
|
|
|
|
|
SetFileInfo(name, pFound, isFile);
|
|
|
|
g_imageSaved = TRUE;
|
2020-04-28 22:44:18 +00:00
|
|
|
return hBitmap;
|
|
|
|
}
|
2015-05-08 16:02:36 +00:00
|
|
|
|
2023-07-05 03:06:22 +00:00
|
|
|
HBITMAP DoLoadImageFile(HWND hwnd, LPCWSTR name, BOOL fIsMainFile)
|
2020-04-28 22:44:18 +00:00
|
|
|
{
|
2023-09-10 13:28:28 +00:00
|
|
|
CWaitCursor waitCursor;
|
|
|
|
|
2020-04-28 22:44:18 +00:00
|
|
|
// find the file
|
2023-11-04 10:25:45 +00:00
|
|
|
WIN32_FIND_DATAW find;
|
2023-07-05 03:06:22 +00:00
|
|
|
HANDLE hFind = ::FindFirstFileW(name, &find);
|
|
|
|
if (hFind == INVALID_HANDLE_VALUE) // does not exist
|
2015-05-08 16:02:36 +00:00
|
|
|
{
|
2023-07-05 03:06:22 +00:00
|
|
|
ShowError(IDS_LOADERRORTEXT, name);
|
2020-04-28 22:44:18 +00:00
|
|
|
return NULL;
|
2015-05-08 16:02:36 +00:00
|
|
|
}
|
2023-07-05 03:06:22 +00:00
|
|
|
::FindClose(hFind);
|
2015-05-08 16:02:36 +00:00
|
|
|
|
2020-04-28 22:44:18 +00:00
|
|
|
// is file empty?
|
2023-07-05 03:06:22 +00:00
|
|
|
if (find.nFileSizeLow == 0 && find.nFileSizeHigh == 0)
|
2015-05-08 16:02:36 +00:00
|
|
|
{
|
2020-04-28 22:44:18 +00:00
|
|
|
if (fIsMainFile)
|
2023-07-05 03:06:22 +00:00
|
|
|
return InitializeImage(name, &find, TRUE);
|
2015-05-08 16:02:36 +00:00
|
|
|
}
|
|
|
|
|
2020-04-28 22:44:18 +00:00
|
|
|
// load the image
|
2023-03-28 13:31:26 +00:00
|
|
|
CImageDx img;
|
2023-08-08 23:17:07 +00:00
|
|
|
float xDpi = 0, yDpi = 0;
|
2023-07-05 03:06:22 +00:00
|
|
|
HRESULT hr = img.LoadDx(name, &xDpi, &yDpi);
|
2023-10-05 11:01:41 +00:00
|
|
|
if (FAILED(hr) && fIsMainFile)
|
|
|
|
{
|
|
|
|
imageModel.ClearHistory();
|
|
|
|
hr = img.LoadDx(name, &xDpi, &yDpi);
|
|
|
|
}
|
2023-07-05 03:06:22 +00:00
|
|
|
if (FAILED(hr))
|
|
|
|
{
|
2023-10-05 11:01:41 +00:00
|
|
|
ATLTRACE("hr: 0x%08lX\n", hr);
|
2023-07-05 03:06:22 +00:00
|
|
|
ShowError(IDS_LOADERRORTEXT, name);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
HBITMAP hBitmap = img.Detach();
|
|
|
|
if (!fIsMainFile)
|
|
|
|
return hBitmap;
|
2023-03-28 13:31:26 +00:00
|
|
|
|
2023-08-08 23:17:07 +00:00
|
|
|
if (xDpi <= 0 || yDpi <= 0)
|
|
|
|
{
|
|
|
|
HDC hDC = ::GetDC(NULL);
|
2023-10-12 10:43:29 +00:00
|
|
|
xDpi = (float)::GetDeviceCaps(hDC, LOGPIXELSX);
|
|
|
|
yDpi = (float)::GetDeviceCaps(hDC, LOGPIXELSY);
|
2023-08-08 23:17:07 +00:00
|
|
|
::ReleaseDC(NULL, hDC);
|
|
|
|
}
|
|
|
|
|
2023-07-05 03:06:22 +00:00
|
|
|
g_xDpi = xDpi;
|
|
|
|
g_yDpi = yDpi;
|
2023-03-28 13:31:26 +00:00
|
|
|
|
2023-07-05 03:06:22 +00:00
|
|
|
SetBitmapAndInfo(hBitmap, name, &find, TRUE);
|
2020-04-28 22:44:18 +00:00
|
|
|
return hBitmap;
|
2015-05-08 16:02:36 +00:00
|
|
|
}
|
2022-02-14 03:08:34 +00:00
|
|
|
|
2023-04-01 13:01:04 +00:00
|
|
|
HBITMAP Rotate90DegreeBlt(HDC hDC1, INT cx, INT cy, BOOL bRight, BOOL bMono)
|
2022-02-14 03:08:34 +00:00
|
|
|
{
|
2023-04-01 13:01:04 +00:00
|
|
|
HBITMAP hbm2;
|
|
|
|
if (bMono)
|
|
|
|
hbm2 = ::CreateBitmap(cy, cx, 1, 1, NULL);
|
|
|
|
else
|
|
|
|
hbm2 = CreateDIBWithProperties(cy, cx);
|
2022-02-14 03:08:34 +00:00
|
|
|
if (!hbm2)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
HDC hDC2 = CreateCompatibleDC(NULL);
|
|
|
|
HGDIOBJ hbm2Old = SelectObject(hDC2, hbm2);
|
|
|
|
if (bRight)
|
|
|
|
{
|
|
|
|
for (INT y = 0; y < cy; ++y)
|
|
|
|
{
|
|
|
|
for (INT x = 0; x < cx; ++x)
|
|
|
|
{
|
|
|
|
COLORREF rgb = GetPixel(hDC1, x, y);
|
|
|
|
SetPixelV(hDC2, cy - (y + 1), x, rgb);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
for (INT y = 0; y < cy; ++y)
|
|
|
|
{
|
|
|
|
for (INT x = 0; x < cx; ++x)
|
|
|
|
{
|
|
|
|
COLORREF rgb = GetPixel(hDC1, x, y);
|
|
|
|
SetPixelV(hDC2, y, cx - (x + 1), rgb);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
SelectObject(hDC2, hbm2Old);
|
|
|
|
DeleteDC(hDC2);
|
|
|
|
return hbm2;
|
|
|
|
}
|
2022-02-14 07:18:18 +00:00
|
|
|
|
2023-06-19 00:56:02 +00:00
|
|
|
HBITMAP SkewDIB(HDC hDC1, HBITMAP hbm, INT nDegree, BOOL bVertical, BOOL bMono)
|
2022-02-14 07:18:18 +00:00
|
|
|
{
|
2023-09-10 13:28:28 +00:00
|
|
|
CWaitCursor waitCursor;
|
|
|
|
|
2022-02-14 07:18:18 +00:00
|
|
|
if (nDegree == 0)
|
|
|
|
return CopyDIBImage(hbm);
|
|
|
|
|
|
|
|
const double eTan = tan(abs(nDegree) * M_PI / 180);
|
|
|
|
|
|
|
|
BITMAP bm;
|
2023-11-04 10:25:45 +00:00
|
|
|
::GetObjectW(hbm, sizeof(bm), &bm);
|
2022-02-14 07:18:18 +00:00
|
|
|
INT cx = bm.bmWidth, cy = bm.bmHeight, dx = 0, dy = 0;
|
|
|
|
if (bVertical)
|
|
|
|
dy = INT(cx * eTan);
|
|
|
|
else
|
|
|
|
dx = INT(cy * eTan);
|
|
|
|
|
|
|
|
if (dx == 0 && dy == 0)
|
|
|
|
return CopyDIBImage(hbm);
|
|
|
|
|
2023-06-19 00:56:02 +00:00
|
|
|
HBITMAP hbmNew;
|
|
|
|
if (bMono)
|
|
|
|
hbmNew = CreateMonoBitmap(cx + dx, cy + dy, FALSE);
|
|
|
|
else
|
|
|
|
hbmNew = CreateColorDIB(cx + dx, cy + dy, RGB(255, 255, 255));
|
2022-02-14 07:18:18 +00:00
|
|
|
if (!hbmNew)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
HDC hDC2 = CreateCompatibleDC(NULL);
|
|
|
|
HGDIOBJ hbm2Old = SelectObject(hDC2, hbmNew);
|
|
|
|
if (bVertical)
|
|
|
|
{
|
|
|
|
for (INT x = 0; x < cx; ++x)
|
|
|
|
{
|
|
|
|
INT delta = INT(x * eTan);
|
|
|
|
if (nDegree > 0)
|
2023-11-04 10:25:45 +00:00
|
|
|
::BitBlt(hDC2, x, (dy - delta), 1, cy, hDC1, x, 0, SRCCOPY);
|
2022-02-14 07:18:18 +00:00
|
|
|
else
|
2023-11-04 10:25:45 +00:00
|
|
|
::BitBlt(hDC2, x, delta, 1, cy, hDC1, x, 0, SRCCOPY);
|
2022-02-14 07:18:18 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
for (INT y = 0; y < cy; ++y)
|
|
|
|
{
|
|
|
|
INT delta = INT(y * eTan);
|
|
|
|
if (nDegree > 0)
|
2023-11-04 10:25:45 +00:00
|
|
|
::BitBlt(hDC2, (dx - delta), y, cx, 1, hDC1, 0, y, SRCCOPY);
|
2022-02-14 07:18:18 +00:00
|
|
|
else
|
2023-11-04 10:25:45 +00:00
|
|
|
::BitBlt(hDC2, delta, y, cx, 1, hDC1, 0, y, SRCCOPY);
|
2022-02-14 07:18:18 +00:00
|
|
|
}
|
|
|
|
}
|
2023-06-19 00:56:02 +00:00
|
|
|
|
2022-02-14 07:18:18 +00:00
|
|
|
SelectObject(hDC2, hbm2Old);
|
|
|
|
DeleteDC(hDC2);
|
|
|
|
return hbmNew;
|
|
|
|
}
|
2023-06-18 10:48:20 +00:00
|
|
|
|
2023-11-18 05:25:19 +00:00
|
|
|
HBITMAP getSubImage(HBITMAP hbmWhole, const RECT& rcPartial)
|
|
|
|
{
|
|
|
|
CRect rc = rcPartial;
|
|
|
|
HBITMAP hbmPart = CreateDIBWithProperties(rc.Width(), rc.Height());
|
|
|
|
if (!hbmPart)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
HDC hDC1 = ::CreateCompatibleDC(NULL);
|
|
|
|
HDC hDC2 = ::CreateCompatibleDC(NULL);
|
|
|
|
HGDIOBJ hbm1Old = ::SelectObject(hDC1, hbmWhole);
|
|
|
|
HGDIOBJ hbm2Old = ::SelectObject(hDC2, hbmPart);
|
|
|
|
::BitBlt(hDC2, 0, 0, rc.Width(), rc.Height(), hDC1, rc.left, rc.top, SRCCOPY);
|
|
|
|
::SelectObject(hDC1, hbm1Old);
|
|
|
|
::SelectObject(hDC2, hbm2Old);
|
|
|
|
::DeleteDC(hDC1);
|
|
|
|
::DeleteDC(hDC2);
|
|
|
|
return hbmPart;
|
|
|
|
}
|
|
|
|
|
|
|
|
void putSubImage(HBITMAP hbmWhole, const RECT& rcPartial, HBITMAP hbmPart)
|
|
|
|
{
|
|
|
|
CRect rc = rcPartial;
|
|
|
|
HDC hDC1 = ::CreateCompatibleDC(NULL);
|
|
|
|
HDC hDC2 = ::CreateCompatibleDC(NULL);
|
|
|
|
HGDIOBJ hbm1Old = ::SelectObject(hDC1, hbmWhole);
|
|
|
|
HGDIOBJ hbm2Old = ::SelectObject(hDC2, hbmPart);
|
|
|
|
::BitBlt(hDC1, rc.left, rc.top, rc.Width(), rc.Height(), hDC2, 0, 0, SRCCOPY);
|
|
|
|
::SelectObject(hDC1, hbm1Old);
|
|
|
|
::SelectObject(hDC2, hbm2Old);
|
|
|
|
::DeleteDC(hDC1);
|
|
|
|
::DeleteDC(hDC2);
|
|
|
|
}
|
|
|
|
|
2023-06-18 10:48:20 +00:00
|
|
|
struct BITMAPINFODX : BITMAPINFO
|
|
|
|
{
|
|
|
|
RGBQUAD bmiColorsAdditional[256 - 1];
|
|
|
|
};
|
|
|
|
|
|
|
|
HGLOBAL BitmapToClipboardDIB(HBITMAP hBitmap)
|
|
|
|
{
|
2023-09-10 13:28:28 +00:00
|
|
|
CWaitCursor waitCursor;
|
|
|
|
|
2023-06-18 10:48:20 +00:00
|
|
|
BITMAP bm;
|
2023-11-04 10:25:45 +00:00
|
|
|
if (!GetObjectW(hBitmap, sizeof(BITMAP), &bm))
|
2023-06-18 10:48:20 +00:00
|
|
|
return NULL;
|
|
|
|
|
|
|
|
BITMAPINFODX bmi;
|
|
|
|
ZeroMemory(&bmi, sizeof(bmi));
|
|
|
|
bmi.bmiHeader.biSize = sizeof(BITMAPINFOHEADER);
|
|
|
|
bmi.bmiHeader.biWidth = bm.bmWidth;
|
|
|
|
bmi.bmiHeader.biHeight = bm.bmHeight;
|
|
|
|
bmi.bmiHeader.biPlanes = 1;
|
|
|
|
bmi.bmiHeader.biBitCount = bm.bmBitsPixel;
|
|
|
|
bmi.bmiHeader.biCompression = BI_RGB;
|
|
|
|
bmi.bmiHeader.biSizeImage = bm.bmWidthBytes * bm.bmHeight;
|
|
|
|
|
|
|
|
INT cColors;
|
|
|
|
if (bm.bmBitsPixel < 16)
|
|
|
|
cColors = 1 << bm.bmBitsPixel;
|
|
|
|
else
|
|
|
|
cColors = 0;
|
|
|
|
|
|
|
|
HDC hDC = CreateCompatibleDC(NULL);
|
|
|
|
|
|
|
|
if (cColors)
|
|
|
|
{
|
|
|
|
HGDIOBJ hbmOld = SelectObject(hDC, hBitmap);
|
|
|
|
cColors = GetDIBColorTable(hDC, 0, cColors, bmi.bmiColors);
|
|
|
|
SelectObject(hDC, hbmOld);
|
|
|
|
}
|
|
|
|
|
|
|
|
DWORD cbColors = cColors * sizeof(RGBQUAD);
|
|
|
|
DWORD dwSize = sizeof(BITMAPINFOHEADER) + cbColors + bmi.bmiHeader.biSizeImage;
|
|
|
|
HGLOBAL hGlobal = GlobalAlloc(GHND | GMEM_SHARE, dwSize);
|
|
|
|
if (hGlobal)
|
|
|
|
{
|
|
|
|
LPBYTE pb = (LPBYTE)GlobalLock(hGlobal);
|
|
|
|
if (pb)
|
|
|
|
{
|
|
|
|
CopyMemory(pb, &bmi, sizeof(BITMAPINFOHEADER));
|
|
|
|
pb += sizeof(BITMAPINFOHEADER);
|
|
|
|
|
|
|
|
CopyMemory(pb, bmi.bmiColors, cbColors);
|
|
|
|
pb += cbColors;
|
|
|
|
|
|
|
|
GetDIBits(hDC, hBitmap, 0, bm.bmHeight, pb, &bmi, DIB_RGB_COLORS);
|
|
|
|
|
|
|
|
GlobalUnlock(hGlobal);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
GlobalFree(hGlobal);
|
|
|
|
hGlobal = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
DeleteDC(hDC);
|
|
|
|
|
|
|
|
return hGlobal;
|
|
|
|
}
|
|
|
|
|
|
|
|
HBITMAP BitmapFromClipboardDIB(HGLOBAL hGlobal)
|
|
|
|
{
|
2023-09-10 13:28:28 +00:00
|
|
|
CWaitCursor waitCursor;
|
|
|
|
|
2023-06-18 10:48:20 +00:00
|
|
|
LPBYTE pb = (LPBYTE)GlobalLock(hGlobal);
|
|
|
|
if (!pb)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
LPBITMAPINFO pbmi = (LPBITMAPINFO)pb;
|
|
|
|
pb += pbmi->bmiHeader.biSize;
|
|
|
|
|
|
|
|
INT cColors = 0, cbColors = 0;
|
|
|
|
if (pbmi->bmiHeader.biSize == sizeof(BITMAPCOREHEADER))
|
|
|
|
{
|
|
|
|
LPBITMAPCOREINFO pbmci = (LPBITMAPCOREINFO)pbmi;
|
|
|
|
WORD BitCount = pbmci->bmciHeader.bcBitCount;
|
|
|
|
if (BitCount < 16)
|
|
|
|
{
|
|
|
|
cColors = (1 << BitCount);
|
|
|
|
cbColors = cColors * sizeof(RGBTRIPLE);
|
|
|
|
pb += cbColors;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (pbmi->bmiHeader.biSize >= sizeof(BITMAPINFOHEADER))
|
|
|
|
{
|
|
|
|
WORD BitCount = pbmi->bmiHeader.biBitCount;
|
|
|
|
if (BitCount < 16)
|
|
|
|
{
|
|
|
|
cColors = (1 << BitCount);
|
|
|
|
cbColors = cColors * sizeof(RGBQUAD);
|
|
|
|
pb += cbColors;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
HDC hDC = CreateCompatibleDC(NULL);
|
|
|
|
HBITMAP hBitmap = CreateDIBSection(hDC, pbmi, DIB_RGB_COLORS, NULL, NULL, 0);
|
|
|
|
if (hBitmap)
|
|
|
|
{
|
|
|
|
SetDIBits(hDC, hBitmap, 0, labs(pbmi->bmiHeader.biHeight), pb, pbmi, DIB_RGB_COLORS);
|
|
|
|
}
|
|
|
|
DeleteDC(hDC);
|
|
|
|
|
|
|
|
GlobalUnlock(hGlobal);
|
|
|
|
|
|
|
|
return hBitmap;
|
|
|
|
}
|
|
|
|
|
|
|
|
HBITMAP BitmapFromHEMF(HENHMETAFILE hEMF)
|
|
|
|
{
|
2023-09-10 13:28:28 +00:00
|
|
|
CWaitCursor waitCursor;
|
|
|
|
|
2023-06-18 10:48:20 +00:00
|
|
|
ENHMETAHEADER header;
|
|
|
|
if (!GetEnhMetaFileHeader(hEMF, sizeof(header), &header))
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
CRect rc = *(LPRECT)&header.rclBounds;
|
|
|
|
INT cx = rc.Width(), cy = rc.Height();
|
|
|
|
HBITMAP hbm = CreateColorDIB(cx, cy, RGB(255, 255, 255));
|
|
|
|
if (!hbm)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
HDC hDC = CreateCompatibleDC(NULL);
|
|
|
|
HGDIOBJ hbmOld = SelectObject(hDC, hbm);
|
|
|
|
PlayEnhMetaFile(hDC, hEMF, &rc);
|
|
|
|
SelectObject(hDC, hbmOld);
|
|
|
|
DeleteDC(hDC);
|
|
|
|
|
|
|
|
return hbm;
|
|
|
|
}
|
2023-08-20 07:46:18 +00:00
|
|
|
|
|
|
|
BOOL IsBitmapBlackAndWhite(HBITMAP hbm)
|
|
|
|
{
|
2023-09-10 13:28:28 +00:00
|
|
|
CWaitCursor waitCursor;
|
|
|
|
|
2023-08-20 07:46:18 +00:00
|
|
|
BITMAP bm;
|
|
|
|
if (!::GetObjectW(hbm, sizeof(bm), &bm))
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
if (bm.bmBitsPixel == 1)
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
BITMAPINFOEX bmi;
|
|
|
|
ZeroMemory(&bmi, sizeof(bmi));
|
|
|
|
bmi.bmiHeader.biSize = sizeof(bmi.bmiHeader);
|
|
|
|
bmi.bmiHeader.biWidth = bm.bmWidth;
|
|
|
|
bmi.bmiHeader.biHeight = bm.bmHeight;
|
|
|
|
bmi.bmiHeader.biPlanes = 1;
|
|
|
|
bmi.bmiHeader.biBitCount = 24;
|
|
|
|
|
|
|
|
DWORD widthbytes = WIDTHBYTES(24 * bm.bmWidth);
|
|
|
|
DWORD cbBits = widthbytes * bm.bmHeight;
|
|
|
|
LPBYTE pbBits = new BYTE[cbBits];
|
|
|
|
|
|
|
|
HDC hdc = ::CreateCompatibleDC(NULL);
|
|
|
|
::GetDIBits(hdc, hbm, 0, bm.bmHeight, pbBits, &bmi, DIB_RGB_COLORS);
|
|
|
|
::DeleteDC(hdc);
|
|
|
|
|
|
|
|
BOOL bBlackAndWhite = TRUE;
|
|
|
|
for (LONG y = 0; y < bm.bmHeight; ++y)
|
|
|
|
{
|
|
|
|
LPBYTE pbLine = &pbBits[widthbytes * y];
|
|
|
|
for (LONG x = 0; x < bm.bmWidth; ++x)
|
|
|
|
{
|
|
|
|
BYTE Blue = *pbLine++;
|
|
|
|
BYTE Green = *pbLine++;
|
|
|
|
BYTE Red = *pbLine++;
|
|
|
|
COLORREF rgbColor = RGB(Red, Green, Blue);
|
|
|
|
if (rgbColor != RGB(0, 0, 0) && rgbColor != RGB(255, 255, 255))
|
|
|
|
{
|
|
|
|
bBlackAndWhite = FALSE;
|
|
|
|
goto Finish;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Finish:
|
|
|
|
delete[] pbBits;
|
|
|
|
|
|
|
|
return bBlackAndWhite;
|
|
|
|
}
|
|
|
|
|
|
|
|
HBITMAP ConvertToBlackAndWhite(HBITMAP hbm)
|
|
|
|
{
|
2023-09-10 13:28:28 +00:00
|
|
|
CWaitCursor waitCursor;
|
|
|
|
|
2023-08-20 07:46:18 +00:00
|
|
|
BITMAP bm;
|
2023-11-04 10:25:45 +00:00
|
|
|
if (!::GetObjectW(hbm, sizeof(bm), &bm))
|
2023-08-20 07:46:18 +00:00
|
|
|
return NULL;
|
|
|
|
|
|
|
|
BITMAPINFOEX bmi;
|
|
|
|
ZeroMemory(&bmi, sizeof(bmi));
|
|
|
|
bmi.bmiHeader.biSize = sizeof(BITMAPINFOHEADER);
|
|
|
|
bmi.bmiHeader.biWidth = bm.bmWidth;
|
|
|
|
bmi.bmiHeader.biHeight = bm.bmHeight;
|
|
|
|
bmi.bmiHeader.biPlanes = 1;
|
|
|
|
bmi.bmiHeader.biBitCount = 1;
|
|
|
|
bmi.bmiColors[1].rgbBlue = 255;
|
|
|
|
bmi.bmiColors[1].rgbGreen = 255;
|
|
|
|
bmi.bmiColors[1].rgbRed = 255;
|
|
|
|
HDC hdc = ::CreateCompatibleDC(NULL);
|
|
|
|
LPVOID pvMonoBits;
|
|
|
|
HBITMAP hMonoBitmap = ::CreateDIBSection(hdc, &bmi, DIB_RGB_COLORS, &pvMonoBits, NULL, 0);
|
|
|
|
if (!hMonoBitmap)
|
|
|
|
{
|
|
|
|
::DeleteDC(hdc);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
HBITMAP hNewBitmap = CreateDIBWithProperties(bm.bmWidth, bm.bmHeight);
|
|
|
|
if (hNewBitmap)
|
|
|
|
{
|
|
|
|
::GetDIBits(hdc, hbm, 0, bm.bmHeight, pvMonoBits, &bmi, DIB_RGB_COLORS);
|
|
|
|
::SetDIBits(hdc, hNewBitmap, 0, bm.bmHeight, pvMonoBits, &bmi, DIB_RGB_COLORS);
|
|
|
|
}
|
|
|
|
::DeleteObject(hMonoBitmap);
|
|
|
|
::DeleteDC(hdc);
|
|
|
|
|
|
|
|
return hNewBitmap;
|
|
|
|
}
|