657 lines
14 KiB
C
657 lines
14 KiB
C
/****************************** Module Header ******************************\
|
|
* Module Name: cldib.c
|
|
|
|
* Copyright (c) 1985 - 1999, Microsoft Corporation
|
|
|
|
|
|
* 09-26-95 ChrisWil Ported dib-scale code.
|
|
*/
|
|
|
|
#include "precomp.h"
|
|
#pragma hdrstop
|
|
|
|
/*
|
|
* Constants.
|
|
*/
|
|
#define FX1 65536 // 1.0 in fixed point
|
|
|
|
|
|
/*
|
|
* Local Macros.
|
|
*/
|
|
#define BPPTOINDEX(bpp) ((UINT)(bpp) >> 3)
|
|
|
|
#define RGBQ(r,g,b) RGB(b, g, r)
|
|
#define RGBQR(rgb) GetBValue(rgb)
|
|
#define RGBQG(rgb) GetGValue(rgb)
|
|
#define RGBQB(rgb) GetRValue(rgb)
|
|
|
|
#define Pel4(p,x) (BYTE)(((x) & 1) ? (((LPBYTE)(p))[(x)/2] & 15) : (((LPBYTE)(p))[(x)/2] >> 4))
|
|
#define Pel8(p,x) (BYTE)(((LPBYTE)(p))[(x)])
|
|
#define Pel16(p,x) (((WORD UNALIGNED *)(p))[(x)])
|
|
#define Pel24(p,x) (*(DWORD UNALIGNED *)((LPBYTE)(p) + (x) * 3))
|
|
|
|
/*
|
|
* Function Typedefs.
|
|
*/
|
|
typedef VOID (*SCALEPROC)(LPDWORD, LPBYTE, long, int, int, int, int, LPBYTE, long, int, int);
|
|
typedef VOID (*INITPROC)(LPBITMAPINFOHEADER);
|
|
|
|
/*
|
|
* Local Routines.
|
|
*/
|
|
BOOL ScaleDIB(LPBITMAPINFOHEADER, LPVOID, LPBITMAPINFOHEADER, LPVOID);
|
|
VOID InitDst8(LPBITMAPINFOHEADER);
|
|
VOID Scale48(LPDWORD, LPBYTE, long, int, int, int, int, LPBYTE, long, int, int);
|
|
VOID Scale88(LPDWORD, LPBYTE, long, int, int, int, int, LPBYTE, long, int, int);
|
|
VOID Scale424(LPDWORD, LPBYTE, long, int, int, int, int, LPBYTE, long, int, int);
|
|
VOID Scale824(LPDWORD, LPBYTE, long, int, int, int, int, LPBYTE, long, int, int);
|
|
VOID Scale2424(LPDWORD, LPBYTE, long, int, int, int, int, LPBYTE, long, int, int);
|
|
BYTE Map8(COLORREF);
|
|
COLORREF MixRGBI(LPDWORD, BYTE, BYTE,BYTE, BYTE, int, int);
|
|
COLORREF MixRGB(DWORD, DWORD, DWORD, DWORD, int, int);
|
|
|
|
/*
|
|
* Globals needed for color-mapping of resources.
|
|
*/
|
|
SCALEPROC ScaleProc[4][4] = {
|
|
NULL, Scale48, NULL, Scale424,
|
|
NULL, Scale88, NULL, Scale824,
|
|
NULL, NULL , NULL, NULL,
|
|
NULL, NULL , NULL, Scale2424
|
|
};
|
|
|
|
INITPROC InitDst[] = {
|
|
NULL, InitDst8, NULL, NULL
|
|
};
|
|
|
|
BYTE rmap[256], gmap[256], bmap[256];
|
|
|
|
|
|
/*
|
|
* SmartStretchDIBits
|
|
|
|
* calls GDI StretchDIBits, unless the stretch is 2:1 and the source
|
|
* is 4bpp, then is does it itself in a nice way.
|
|
|
|
* this optimization should just be put into GDI.
|
|
|
|
* can we change the passed bits? I assume we cant, could avoid a alloc
|
|
* if so.
|
|
|
|
*/
|
|
|
|
int SmartStretchDIBits(
|
|
HDC hdc,
|
|
int xD,
|
|
int yD,
|
|
int dxD,
|
|
int dyD,
|
|
int xS,
|
|
int yS,
|
|
int dxS,
|
|
int dyS,
|
|
LPVOID lpBits,
|
|
LPBITMAPINFO lpbi,
|
|
UINT wUsage,
|
|
DWORD rop)
|
|
{
|
|
LPBYTE lpBitsNew;
|
|
UINT bpp;
|
|
RGBQUAD rgb;
|
|
LPBITMAPINFOHEADER lpbiNew = NULL;
|
|
|
|
int i;
|
|
|
|
/*
|
|
* thunk to USER32
|
|
*/
|
|
UserAssert(rop == SRCCOPY);
|
|
UserAssert(wUsage == DIB_RGB_COLORS);
|
|
UserAssert(xS == 0 && yS == 0);
|
|
|
|
if ((GetDIBColorTable(hdc, 0, 1, &rgb) != 1) &&
|
|
(dxD != dxS || dyD != dyS) && // 1:1 stretch just call GDI
|
|
(dxD >= dxS/2) && (dyD >= dyS/2) && // less than 1:2 shrink call GDI
|
|
(lpbi->bmiHeader.biCompression == 0) && // must be un-compressed
|
|
(lpbi->bmiHeader.biBitCount == 4 || // input must be 4,8,24
|
|
lpbi->bmiHeader.biBitCount == 8 ||
|
|
lpbi->bmiHeader.biBitCount == 24)) {
|
|
|
|
bpp = GetDeviceCaps(hdc, BITSPIXEL) * GetDeviceCaps(hdc, PLANES);
|
|
|
|
bpp = (bpp > 8 ? 24 : 8);
|
|
|
|
lpbiNew = (LPBITMAPINFOHEADER)UserLocalAlloc(0,
|
|
sizeof(BITMAPINFOHEADER) + (256 * sizeof(RGBQUAD)) +
|
|
(UINT)((((dxD * bpp) / 8) + 3) & ~3) * (UINT)dyD);
|
|
|
|
if (lpbiNew) {
|
|
|
|
*lpbiNew = lpbi->bmiHeader;
|
|
lpbiNew->biWidth = dxD;
|
|
lpbiNew->biHeight = dyD;
|
|
lpbiNew->biBitCount = (WORD)bpp;
|
|
lpBitsNew = (LPBYTE)lpbiNew +
|
|
sizeof(BITMAPINFOHEADER) +
|
|
(256 * sizeof(RGBQUAD));
|
|
|
|
if (ScaleDIB((LPBITMAPINFOHEADER)lpbi,
|
|
(LPVOID)lpBits,
|
|
lpbiNew,
|
|
lpBitsNew)) {
|
|
|
|
lpbi = (LPBITMAPINFO)lpbiNew;
|
|
lpBits = lpBitsNew;
|
|
dxS = dxD;
|
|
dyS = dyD;
|
|
}
|
|
}
|
|
}
|
|
|
|
i = StretchDIBits(hdc,
|
|
xD,
|
|
yD,
|
|
dxD,
|
|
dyD,
|
|
xS,
|
|
yS,
|
|
dxS,
|
|
dyS,
|
|
lpBits,
|
|
lpbi,
|
|
wUsage,
|
|
rop);
|
|
|
|
if (lpbiNew)
|
|
UserLocalFree(lpbiNew);
|
|
|
|
return i;
|
|
}
|
|
|
|
/*
|
|
* ScaleDIB
|
|
|
|
* Parameters
|
|
* ----------
|
|
* lpbiSrc - BITMAPINFO of source.
|
|
* lpSrc - Input bits to crunch.
|
|
* lpbiDst - BITMAPINFO of destination.
|
|
* lpDst - Output bits to crunch.
|
|
|
|
|
|
*/
|
|
|
|
BOOL ScaleDIB(
|
|
LPBITMAPINFOHEADER lpbiSrc,
|
|
LPVOID lpSrc,
|
|
LPBITMAPINFOHEADER lpbiDst,
|
|
LPVOID lpDst)
|
|
{
|
|
LPBYTE pbSrc;
|
|
LPBYTE pbDst;
|
|
LPDWORD pal;
|
|
int dxSrc;
|
|
int dySrc;
|
|
int dxDst;
|
|
int dyDst;
|
|
int iSrc;
|
|
int iDst;
|
|
int x0;
|
|
int y0;
|
|
int sdx;
|
|
int sdy;
|
|
LONG WidthBytesSrc;
|
|
LONG WidthBytesDst;
|
|
|
|
if ((lpbiSrc->biCompression != BI_RGB) ||
|
|
(lpbiDst->biCompression != BI_RGB)) {
|
|
return FALSE;
|
|
}
|
|
|
|
iSrc = BPPTOINDEX(lpbiSrc->biBitCount);
|
|
iDst = BPPTOINDEX(lpbiDst->biBitCount);
|
|
|
|
if (ScaleProc[iSrc][iDst] == NULL)
|
|
return FALSE;
|
|
|
|
dxSrc = (int)lpbiSrc->biWidth;
|
|
dySrc = (int)lpbiSrc->biHeight;
|
|
|
|
dxDst = (int)lpbiDst->biWidth;
|
|
dyDst = (int)lpbiDst->biHeight;
|
|
|
|
WidthBytesDst = BitmapWidth(lpbiDst->biWidth, lpbiDst->biBitCount);
|
|
WidthBytesSrc = BitmapWidth(lpbiSrc->biWidth, lpbiSrc->biBitCount);
|
|
|
|
lpbiDst->biSizeImage = (WidthBytesDst * dyDst);
|
|
|
|
pbSrc = (LPBYTE)lpSrc;
|
|
pbDst = (LPBYTE)lpDst;
|
|
|
|
pal = (LPDWORD)(lpbiSrc + 1);
|
|
|
|
if (InitDst[iDst])
|
|
InitDst[iDst](lpbiDst);
|
|
|
|
if ((dxSrc == (dxDst * 2)) && (dySrc == (dyDst * 2))) {
|
|
|
|
sdx = FX1 * 2;
|
|
sdy = FX1 * 2;
|
|
y0 = FX1 / 2;
|
|
x0 = FX1 / 2;
|
|
|
|
} else {
|
|
|
|
sdx = (dxSrc - 1) * FX1 / (dxDst - 1);
|
|
sdy = (dySrc - 1) * FX1 / (dyDst - 1);
|
|
y0 = 0;
|
|
x0 = 0;
|
|
}
|
|
|
|
/*
|
|
* Make sure we don't croak on a bad bitmap since this can hose winlogon
|
|
* if it's the desktop wallpaper.
|
|
*/
|
|
try {
|
|
ScaleProc[iSrc][iDst](pal,
|
|
pbSrc,
|
|
WidthBytesSrc,
|
|
x0,
|
|
y0,
|
|
sdx,
|
|
sdy,
|
|
pbDst,
|
|
WidthBytesDst,
|
|
dxDst,
|
|
dyDst);
|
|
} except (W32ExceptionHandler(FALSE, RIP_WARNING)) {
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/*
|
|
* Scale48
|
|
|
|
|
|
*/
|
|
|
|
VOID Scale48(
|
|
LPDWORD pal,
|
|
LPBYTE pbSrc,
|
|
LONG WidthBytesSrc,
|
|
int x0,
|
|
int y0,
|
|
int sdx,
|
|
int sdy,
|
|
LPBYTE pbDst,
|
|
LONG WidthBytesDst,
|
|
int dxDst,
|
|
int dyDst)
|
|
{
|
|
BYTE b0;
|
|
BYTE b1;
|
|
BYTE b2;
|
|
BYTE b3;
|
|
LPBYTE pb;
|
|
int x;
|
|
int y;
|
|
UINT sx;
|
|
UINT sy;
|
|
|
|
for (sy=y0, y=0; y < dyDst; y++, sy+=sdy) {
|
|
|
|
pb = pbSrc + (WidthBytesSrc * (sy / FX1));
|
|
|
|
for (sx=x0, x=0; x < dxDst; x++, sx+=sdx) {
|
|
|
|
b0 = Pel4(pb, (sx / FX1));
|
|
b1 = Pel4(pb, (sx / FX1) + 1);
|
|
b2 = Pel4(pb + WidthBytesSrc, (sx / FX1));
|
|
b3 = Pel4(pb + WidthBytesSrc, (sx / FX1) + 1);
|
|
|
|
pbDst[x] = Map8(MixRGBI(pal, b0, b1, b2, b3, sx % FX1, sy % FX1));
|
|
}
|
|
|
|
pbDst += WidthBytesDst;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Scale88
|
|
|
|
|
|
*/
|
|
|
|
VOID Scale88(
|
|
LPDWORD pal,
|
|
LPBYTE pbSrc,
|
|
LONG WidthBytesSrc,
|
|
int x0,
|
|
int y0,
|
|
int sdx,
|
|
int sdy,
|
|
LPBYTE pbDst,
|
|
LONG WidthBytesDst,
|
|
int dxDst,
|
|
int dyDst)
|
|
{
|
|
BYTE b0;
|
|
BYTE b1;
|
|
BYTE b2;
|
|
BYTE b3;
|
|
LPBYTE pb;
|
|
int x;
|
|
int y;
|
|
UINT sx;
|
|
UINT sy;
|
|
|
|
for (sy=y0, y=0; y < dyDst; y++, sy+=sdy) {
|
|
|
|
pb = pbSrc + (WidthBytesSrc * (sy / FX1));
|
|
|
|
for (sx=x0, x=0; x < dxDst; x++, sx+=sdx) {
|
|
|
|
b0 = Pel8(pb, (sx / FX1));
|
|
b1 = Pel8(pb, (sx / FX1) + 1);
|
|
b2 = Pel8(pb + WidthBytesSrc, (sx / FX1));
|
|
b3 = Pel8(pb + WidthBytesSrc, (sx / FX1) + 1);
|
|
|
|
pbDst[x] = Map8(MixRGBI(pal, b0, b1, b2, b3, sx % FX1, sy % FX1));
|
|
}
|
|
|
|
pbDst += WidthBytesDst;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Scale424
|
|
|
|
|
|
*/
|
|
|
|
VOID Scale424(
|
|
LPDWORD pal,
|
|
LPBYTE pbSrc,
|
|
LONG WidthBytesSrc,
|
|
int x0,
|
|
int y0,
|
|
int sdx,
|
|
int sdy,
|
|
LPBYTE pbDst,
|
|
LONG WidthBytesDst,
|
|
int dxDst,
|
|
int dyDst)
|
|
{
|
|
BYTE b0;
|
|
BYTE b1;
|
|
BYTE b2;
|
|
BYTE b3;
|
|
LPBYTE pb;
|
|
int x;
|
|
int y;
|
|
UINT sx;
|
|
UINT sy;
|
|
COLORREF rgb;
|
|
|
|
for (sy=y0, y=0; y < dyDst; y++, sy+=sdy) {
|
|
|
|
pb = pbSrc + (WidthBytesSrc * (sy / FX1));
|
|
|
|
for (sx=x0, x=0; x < dxDst; x++, sx+=sdx) {
|
|
|
|
b0 = Pel4(pb, (sx / FX1));
|
|
b1 = Pel4(pb, (sx / FX1) + 1);
|
|
b2 = Pel4(pb + WidthBytesSrc, (sx / FX1));
|
|
b3 = Pel4(pb + WidthBytesSrc, (sx / FX1) + 1);
|
|
|
|
rgb = MixRGBI(pal, b0, b1, b2, b3, sx % FX1, sy % FX1);
|
|
|
|
*pbDst++ = GetBValue(rgb);
|
|
*pbDst++ = GetGValue(rgb);
|
|
*pbDst++ = GetRValue(rgb);
|
|
}
|
|
|
|
pbDst += (WidthBytesDst - (dxDst * 3));
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Scale824
|
|
|
|
|
|
*/
|
|
|
|
VOID Scale824(
|
|
LPDWORD pal,
|
|
LPBYTE pbSrc,
|
|
LONG WidthBytesSrc,
|
|
int x0,
|
|
int y0,
|
|
int sdx,
|
|
int sdy,
|
|
LPBYTE pbDst,
|
|
LONG WidthBytesDst,
|
|
int dxDst,
|
|
int dyDst)
|
|
{
|
|
BYTE b0;
|
|
BYTE b1;
|
|
BYTE b2;
|
|
BYTE b3;
|
|
LPBYTE pb;
|
|
int x;
|
|
int y;
|
|
UINT sx;
|
|
UINT sy;
|
|
COLORREF rgb;
|
|
|
|
for (sy=y0, y=0; y < dyDst; y++, sy+=sdy) {
|
|
|
|
pb = pbSrc + (WidthBytesSrc * (sy / FX1));
|
|
|
|
for (sx=x0, x=0; x < dxDst; x++, sx+=sdx) {
|
|
|
|
b0 = Pel8(pb, (sx / FX1));
|
|
b1 = Pel8(pb, (sx / FX1) + 1);
|
|
b2 = Pel8(pb + WidthBytesSrc, (sx / FX1));
|
|
b3 = Pel8(pb + WidthBytesSrc, (sx / FX1) + 1);
|
|
|
|
rgb = MixRGBI(pal, b0, b1, b2, b3, sx % FX1, sy % FX1);
|
|
|
|
*pbDst++ = GetBValue(rgb);
|
|
*pbDst++ = GetGValue(rgb);
|
|
*pbDst++ = GetRValue(rgb);
|
|
}
|
|
|
|
pbDst += (WidthBytesDst - (dxDst * 3));
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Scale2424
|
|
|
|
|
|
*/
|
|
|
|
VOID Scale2424(
|
|
LPDWORD pal,
|
|
LPBYTE pbSrc,
|
|
LONG WidthBytesSrc,
|
|
int x0,
|
|
int y0,
|
|
int sdx,
|
|
int sdy,
|
|
LPBYTE pbDst,
|
|
LONG WidthBytesDst,
|
|
int dxDst,
|
|
int dyDst)
|
|
{
|
|
DWORD bgr0;
|
|
DWORD bgr1;
|
|
DWORD bgr2;
|
|
DWORD bgr3;
|
|
LPBYTE pb;
|
|
int x;
|
|
int y;
|
|
UINT sx;
|
|
UINT sy;
|
|
COLORREF rgb;
|
|
|
|
UNREFERENCED_PARAMETER(pal);
|
|
|
|
for (sy=y0, y=0; y < dyDst; y++, sy+=sdy) {
|
|
|
|
pb = pbSrc + (WidthBytesSrc * (sy / FX1));
|
|
|
|
for (sx=x0, x=0; x < dxDst; x++, sx+=sdx) {
|
|
|
|
bgr0 = Pel24(pb, (sx / FX1));
|
|
bgr1 = Pel24(pb, (sx / FX1) + 1);
|
|
bgr2 = Pel24(pb + WidthBytesSrc, (sx / FX1));
|
|
bgr3 = Pel24(pb + WidthBytesSrc, (sx / FX1) + 1);
|
|
|
|
rgb = MixRGB(bgr0, bgr1, bgr2, bgr3, sx % FX1, sy % FX1);
|
|
|
|
*pbDst++ = GetBValue(rgb);
|
|
*pbDst++ = GetGValue(rgb);
|
|
*pbDst++ = GetRValue(rgb);
|
|
}
|
|
|
|
pbDst += (WidthBytesDst - (dxDst * 3));
|
|
}
|
|
}
|
|
|
|
/*
|
|
* MixRGB
|
|
|
|
* r0 x r1
|
|
* y *
|
|
* r2 r3
|
|
|
|
* Note: inputs are RGBQUADs, output is a COLORREF.
|
|
|
|
*/
|
|
|
|
COLORREF MixRGB(
|
|
DWORD r0,
|
|
DWORD r1,
|
|
DWORD r2,
|
|
DWORD r3,
|
|
int x,
|
|
int y)
|
|
{
|
|
int r;
|
|
int g;
|
|
int b;
|
|
|
|
if (x == 0 && y == 0) {
|
|
|
|
r = RGBQR(r0);
|
|
g = RGBQG(r0);
|
|
b = RGBQB(r0);
|
|
|
|
} else if (x == 0) {
|
|
|
|
r = ((FX1-y) * RGBQR(r0) + y * RGBQR(r2))/FX1;
|
|
g = ((FX1-y) * RGBQG(r0) + y * RGBQG(r2))/FX1;
|
|
b = ((FX1-y) * RGBQB(r0) + y * RGBQB(r2))/FX1;
|
|
}
|
|
else if (y == 0)
|
|
{
|
|
r = ((FX1-x) * RGBQR(r0) + x * RGBQR(r1))/FX1;
|
|
g = ((FX1-x) * RGBQG(r0) + x * RGBQG(r1))/FX1;
|
|
b = ((FX1-x) * RGBQB(r0) + x * RGBQB(r1))/FX1;
|
|
}
|
|
else if (x == FX1/2 && y == FX1/2)
|
|
{
|
|
r = (RGBQR(r0) + RGBQR(r1) + RGBQR(r2) + RGBQR(r3))/4;
|
|
g = (RGBQG(r0) + RGBQG(r1) + RGBQG(r2) + RGBQG(r3))/4;
|
|
b = (RGBQB(r0) + RGBQB(r1) + RGBQB(r2) + RGBQB(r3))/4;
|
|
}
|
|
else
|
|
{
|
|
r =((ULONG)RGBQR(r0) * (FX1-x) / FX1 * (FX1-y) + (ULONG)RGBQR(r1) * x / FX1 * (FX1-y) +
|
|
(ULONG)RGBQR(r2) * (FX1-x) / FX1 * y + (ULONG)RGBQR(r3) * x / FX1 * y )/FX1;
|
|
|
|
g =((ULONG)RGBQG(r0) * (FX1-x) / FX1 * (FX1-y) + (ULONG)RGBQG(r1) * x / FX1 * (FX1-y) +
|
|
(ULONG)RGBQG(r2) * (FX1-x) / FX1 * y + (ULONG)RGBQG(r3) * x / FX1 * y )/FX1;
|
|
|
|
b =((ULONG)RGBQB(r0) * (FX1-x) / FX1 * (FX1-y) + (ULONG)RGBQB(r1) * x / FX1 * (FX1-y) +
|
|
(ULONG)RGBQB(r2) * (FX1-x) / FX1 * y + (ULONG)RGBQB(r3) * x / FX1 * y )/FX1;
|
|
}
|
|
|
|
return RGB(r,g,b);
|
|
}
|
|
|
|
/*
|
|
* MixRGBI
|
|
|
|
|
|
*/
|
|
|
|
_inline COLORREF MixRGBI(
|
|
LPDWORD pal,
|
|
BYTE b0,
|
|
BYTE b1,
|
|
BYTE b2,
|
|
BYTE b3,
|
|
int x,
|
|
int y)
|
|
{
|
|
if (((b0 == b1) && (b1 == b2) && (b2 == b3)) || ((x == 0) && (y == 0)))
|
|
return RGBX(pal[b0]);
|
|
else
|
|
return MixRGB(pal[b0], pal[b1], pal[b2], pal[b3], x, y);
|
|
}
|
|
|
|
/*
|
|
* InitDst8
|
|
|
|
|
|
*/
|
|
|
|
_inline VOID InitDst8(
|
|
LPBITMAPINFOHEADER lpbi)
|
|
{
|
|
int r;
|
|
int g;
|
|
int b;
|
|
LPDWORD pdw;
|
|
|
|
pdw = (LPDWORD)(lpbi + 1);
|
|
lpbi->biClrUsed = 6*6*6;
|
|
|
|
for (r=0; r < 6; r++) {
|
|
|
|
for (g=0; g < 6; g++) {
|
|
|
|
for (b=0; b < 6; b++) {
|
|
*pdw++ = RGBQ((r * 255) / 5, (g * 255) / 5, (b * 255) / 5);
|
|
}
|
|
}
|
|
}
|
|
|
|
for (b=0; b < 256; b++) {
|
|
bmap[b] = (b * 5 + 128) / 255;
|
|
gmap[b] = 6 * bmap[b];
|
|
rmap[b] = 36 * bmap[b];
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Map8
|
|
|
|
|
|
*/
|
|
|
|
_inline BYTE Map8(
|
|
COLORREF rgb)
|
|
{
|
|
return rmap[GetRValue(rgb)] +
|
|
gmap[GetGValue(rgb)] +
|
|
bmap[GetBValue(rgb)];
|
|
}
|