657 lines
14 KiB
C
Raw Normal View History

2001-01-01 00:00:00 +01:00
/****************************** 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)];
}