blob: d05a5dfdc83205ab61d4868c2cc98869a5898b5d [file] [log] [blame]
// ==========================================================
// Copy / paste routines
//
// - Floris van den Berg (flvdberg@wxs.nl)
// - Alexander Dymerets (sashad@te.net.ua)
// - Hervé Drolon (drolon@infonie.fr)
// - Manfred Tausch (manfred.tausch@t-online.de)
// - Riley McNiff (rmcniff@marexgroup.com)
// - Carsten Klein (cklein05@users.sourceforge.net)
//
// This file is part of FreeImage 3
//
// COVERED CODE IS PROVIDED UNDER THIS LICENSE ON AN "AS IS" BASIS, WITHOUT WARRANTY
// OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, WITHOUT LIMITATION, WARRANTIES
// THAT THE COVERED CODE IS FREE OF DEFECTS, MERCHANTABLE, FIT FOR A PARTICULAR PURPOSE
// OR NON-INFRINGING. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE COVERED
// CODE IS WITH YOU. SHOULD ANY COVERED CODE PROVE DEFECTIVE IN ANY RESPECT, YOU (NOT
// THE INITIAL DEVELOPER OR ANY OTHER CONTRIBUTOR) ASSUME THE COST OF ANY NECESSARY
// SERVICING, REPAIR OR CORRECTION. THIS DISCLAIMER OF WARRANTY CONSTITUTES AN ESSENTIAL
// PART OF THIS LICENSE. NO USE OF ANY COVERED CODE IS AUTHORIZED HEREUNDER EXCEPT UNDER
// THIS DISCLAIMER.
//
// Use at your own risk!
// ==========================================================
#include "FreeImage.h"
#include "Utilities.h"
// ----------------------------------------------------------
// Helpers
// ----------------------------------------------------------
/////////////////////////////////////////////////////////////
// Alpha blending / combine functions
// ----------------------------------------------------------
/// 1-bit
static BOOL Combine1(FIBITMAP *dst_dib, FIBITMAP *src_dib, unsigned x, unsigned y, unsigned alpha);
/// 4-bit
static BOOL Combine4(FIBITMAP *dst_dib, FIBITMAP *src_dib, unsigned x, unsigned y, unsigned alpha);
/// 8-bit
static BOOL Combine8(FIBITMAP *dst_dib, FIBITMAP *src_dib, unsigned x, unsigned y, unsigned alpha);
/// 16-bit 555
static BOOL Combine16_555(FIBITMAP *dst_dib, FIBITMAP *src_dib, unsigned x, unsigned y, unsigned alpha);
/// 16-bit 565
static BOOL Combine16_565(FIBITMAP *dst_dib, FIBITMAP *src_dib, unsigned x, unsigned y, unsigned alpha);
/// 24-bit
static BOOL Combine24(FIBITMAP *dst_dib, FIBITMAP *src_dib, unsigned x, unsigned y, unsigned alpha);
/// 32- bit
static BOOL Combine32(FIBITMAP *dst_dib, FIBITMAP *src_dib, unsigned x, unsigned y, unsigned alpha);
// ----------------------------------------------------------
// ----------------------------------------------------------
// 1-bit
// ----------------------------------------------------------
static BOOL
Combine1(FIBITMAP *dst_dib, FIBITMAP *src_dib, unsigned x, unsigned y, unsigned alpha) {
BOOL value;
// check the bit depth of src and dst images
if((FreeImage_GetBPP(dst_dib) != 1) || (FreeImage_GetBPP(src_dib) != 1)) {
return FALSE;
}
// check the size of src image
if((x + FreeImage_GetWidth(src_dib) > FreeImage_GetWidth(dst_dib)) || (y + FreeImage_GetHeight(src_dib) > FreeImage_GetHeight(dst_dib))) {
return FALSE;
}
BYTE *dst_bits = FreeImage_GetBits(dst_dib) + ((FreeImage_GetHeight(dst_dib) - FreeImage_GetHeight(src_dib) - y) * FreeImage_GetPitch(dst_dib));
BYTE *src_bits = FreeImage_GetBits(src_dib);
// combine images
for(unsigned rows = 0; rows < FreeImage_GetHeight(src_dib); rows++) {
for(unsigned cols = 0; cols < FreeImage_GetWidth(src_dib); cols++) {
// get bit at (rows, cols) in src image
value = (src_bits[cols >> 3] & (0x80 >> (cols & 0x07))) != 0;
// set bit at (rows, x+cols) in dst image
value ? dst_bits[(x + cols) >> 3] |= (0x80 >> ((x + cols) & 0x7)) : dst_bits[(x + cols) >> 3] &= (0xFF7F >> ((x + cols) & 0x7));
}
dst_bits += FreeImage_GetPitch(dst_dib);
src_bits += FreeImage_GetPitch(src_dib);
}
return TRUE;
}
// ----------------------------------------------------------
// 4-bit
// ----------------------------------------------------------
static BOOL
Combine4(FIBITMAP *dst_dib, FIBITMAP *src_dib, unsigned x, unsigned y, unsigned alpha) {
int swapTable[16];
BOOL bOddStart, bOddEnd;
// check the bit depth of src and dst images
if((FreeImage_GetBPP(dst_dib) != 4) || (FreeImage_GetBPP(src_dib) != 4)) {
return FALSE;
}
// check the size of src image
if((x + FreeImage_GetWidth(src_dib) > FreeImage_GetWidth(dst_dib)) || (y + FreeImage_GetHeight(src_dib) > FreeImage_GetHeight(dst_dib))) {
return FALSE;
}
// get src and dst palettes
RGBQUAD *src_pal = FreeImage_GetPalette(src_dib);
RGBQUAD *dst_pal = FreeImage_GetPalette(dst_dib);
if (src_pal == NULL || dst_pal == NULL) {
return FALSE;
}
// build a swap table for the closest color match from the source palette to the destination palette
for (int i = 0; i < 16; i++) {
WORD min_diff = (WORD)-1;
for (int j = 0; j < 16; j++) {
// calculates the color difference using a Manhattan distance
WORD abs_diff = (WORD)(
abs(src_pal[i].rgbBlue - dst_pal[j].rgbBlue)
+ abs(src_pal[i].rgbGreen - dst_pal[j].rgbGreen)
+ abs(src_pal[i].rgbRed - dst_pal[j].rgbRed)
);
if (abs_diff < min_diff) {
swapTable[i] = j;
min_diff = abs_diff;
if (abs_diff == 0) {
break;
}
}
}
}
BYTE *dst_bits = FreeImage_GetBits(dst_dib) + ((FreeImage_GetHeight(dst_dib) - FreeImage_GetHeight(src_dib) - y) * FreeImage_GetPitch(dst_dib)) + (x >> 1);
BYTE *src_bits = FreeImage_GetBits(src_dib);
// combine images
// allocate space for our temporary row
unsigned src_line = FreeImage_GetLine(src_dib);
unsigned src_width = FreeImage_GetWidth(src_dib);
unsigned src_height = FreeImage_GetHeight(src_dib);
BYTE *buffer = (BYTE *)malloc(src_line * sizeof(BYTE));
if (buffer == NULL) {
return FALSE;
}
bOddStart = (x & 0x01) ? TRUE : FALSE;
if ((bOddStart && !(src_width & 0x01)) || (!bOddStart && (src_width & 0x01))) {
bOddEnd = TRUE;
}
else {
bOddEnd = FALSE;
}
for(unsigned rows = 0; rows < src_height; rows++) {
memcpy(buffer, src_bits, src_line);
// change the values in the temp row to be those from the swap table
for (unsigned cols = 0; cols < src_line; cols++) {
buffer[cols] = (BYTE)((swapTable[HINIBBLE(buffer[cols]) >> 4] << 4) + swapTable[LOWNIBBLE(buffer[cols])]);
}
if (bOddStart) {
buffer[0] = HINIBBLE(dst_bits[0]) + LOWNIBBLE(buffer[0]);
}
if (bOddEnd) {
buffer[src_line - 1] = HINIBBLE(buffer[src_line - 1]) + LOWNIBBLE(dst_bits[src_line - 1]);
}
memcpy(dst_bits, buffer, src_line);
dst_bits += FreeImage_GetPitch(dst_dib);
src_bits += FreeImage_GetPitch(src_dib);
}
free(buffer);
return TRUE;
}
// ----------------------------------------------------------
// 8-bit
// ----------------------------------------------------------
static BOOL
Combine8(FIBITMAP *dst_dib, FIBITMAP *src_dib, unsigned x, unsigned y, unsigned alpha) {
// check the bit depth of src and dst images
if((FreeImage_GetBPP(dst_dib) != 8) || (FreeImage_GetBPP(src_dib) != 8)) {
return FALSE;
}
// check the size of src image
if((x + FreeImage_GetWidth(src_dib) > FreeImage_GetWidth(dst_dib)) || (y + FreeImage_GetHeight(src_dib) > FreeImage_GetHeight(dst_dib))) {
return FALSE;
}
BYTE *dst_bits = FreeImage_GetBits(dst_dib) + ((FreeImage_GetHeight(dst_dib) - FreeImage_GetHeight(src_dib) - y) * FreeImage_GetPitch(dst_dib)) + (x);
BYTE *src_bits = FreeImage_GetBits(src_dib);
if(alpha > 255) {
// combine images
for(unsigned rows = 0; rows < FreeImage_GetHeight(src_dib); rows++) {
memcpy(dst_bits, src_bits, FreeImage_GetLine(src_dib));
dst_bits += FreeImage_GetPitch(dst_dib);
src_bits += FreeImage_GetPitch(src_dib);
}
} else {
// alpha blend images
for(unsigned rows = 0; rows < FreeImage_GetHeight(src_dib); rows++) {
for (unsigned cols = 0; cols < FreeImage_GetLine(src_dib); cols++) {
dst_bits[cols] = (BYTE)(((src_bits[cols] - dst_bits[cols]) * alpha + (dst_bits[cols] << 8)) >> 8);
}
dst_bits += FreeImage_GetPitch(dst_dib);
src_bits += FreeImage_GetPitch(src_dib);
}
}
return TRUE;
}
// ----------------------------------------------------------
// 16-bit
// ----------------------------------------------------------
static BOOL
Combine16_555(FIBITMAP *dst_dib, FIBITMAP *src_dib, unsigned x, unsigned y, unsigned alpha) {
// check the bit depth of src and dst images
if((FreeImage_GetBPP(dst_dib) != 16) || (FreeImage_GetBPP(src_dib) != 16)) {
return FALSE;
}
// check the size of src image
if((x + FreeImage_GetWidth(src_dib) > FreeImage_GetWidth(dst_dib)) || (y + FreeImage_GetHeight(src_dib) > FreeImage_GetHeight(dst_dib))) {
return FALSE;
}
BYTE *dst_bits = FreeImage_GetBits(dst_dib) + ((FreeImage_GetHeight(dst_dib) - FreeImage_GetHeight(src_dib) - y) * FreeImage_GetPitch(dst_dib)) + (x * 2);
BYTE *src_bits = FreeImage_GetBits(src_dib);
if (alpha > 255) {
for(unsigned rows = 0; rows < FreeImage_GetHeight(src_dib); rows++) {
memcpy(dst_bits, src_bits, FreeImage_GetLine(src_dib));
dst_bits += FreeImage_GetPitch(dst_dib);
src_bits += FreeImage_GetPitch(src_dib);
}
} else {
for(unsigned rows = 0; rows < FreeImage_GetHeight(src_dib); rows++) {
for(unsigned cols = 0; cols < FreeImage_GetLine(src_dib); cols += 2) {
RGBTRIPLE color_s;
RGBTRIPLE color_t;
WORD *tmp1 = (WORD *)&dst_bits[cols];
WORD *tmp2 = (WORD *)&src_bits[cols];
// convert 16-bit colors to 24-bit
color_s.rgbtRed = (BYTE)(((*tmp1 & FI16_555_RED_MASK) >> FI16_555_RED_SHIFT) << 3);
color_s.rgbtGreen = (BYTE)(((*tmp1 & FI16_555_GREEN_MASK) >> FI16_555_GREEN_SHIFT) << 3);
color_s.rgbtBlue = (BYTE)(((*tmp1 & FI16_555_BLUE_MASK) >> FI16_555_BLUE_SHIFT) << 3);
color_t.rgbtRed = (BYTE)(((*tmp2 & FI16_555_RED_MASK) >> FI16_555_RED_SHIFT) << 3);
color_t.rgbtGreen = (BYTE)(((*tmp2 & FI16_555_GREEN_MASK) >> FI16_555_GREEN_SHIFT) << 3);
color_t.rgbtBlue = (BYTE)(((*tmp2 & FI16_555_BLUE_MASK) >> FI16_555_BLUE_SHIFT) << 3);
// alpha blend
color_s.rgbtRed = (BYTE)(((color_t.rgbtRed - color_s.rgbtRed) * alpha + (color_s.rgbtRed << 8)) >> 8);
color_s.rgbtGreen = (BYTE)(((color_t.rgbtGreen - color_s.rgbtGreen) * alpha + (color_s.rgbtGreen << 8)) >> 8);
color_s.rgbtBlue = (BYTE)(((color_t.rgbtBlue - color_s.rgbtBlue) * alpha + (color_s.rgbtBlue << 8)) >> 8);
// convert 24-bit color back to 16-bit
*tmp1 = RGB555(color_s.rgbtRed, color_s.rgbtGreen, color_s.rgbtBlue);
}
dst_bits += FreeImage_GetPitch(dst_dib);
src_bits += FreeImage_GetPitch(src_dib);
}
}
return TRUE;
}
static BOOL
Combine16_565(FIBITMAP *dst_dib, FIBITMAP *src_dib, unsigned x, unsigned y, unsigned alpha) {
// check the bit depth of src and dst images
if((FreeImage_GetBPP(dst_dib) != 16) || (FreeImage_GetBPP(src_dib) != 16)) {
return FALSE;
}
// check the size of src image
if((x + FreeImage_GetWidth(src_dib) > FreeImage_GetWidth(dst_dib)) || (y + FreeImage_GetHeight(src_dib) > FreeImage_GetHeight(dst_dib))) {
return FALSE;
}
BYTE *dst_bits = FreeImage_GetBits(dst_dib) + ((FreeImage_GetHeight(dst_dib) - FreeImage_GetHeight(src_dib) - y) * FreeImage_GetPitch(dst_dib)) + (x * 2);
BYTE *src_bits = FreeImage_GetBits(src_dib);
if (alpha > 255) {
for(unsigned rows = 0; rows < FreeImage_GetHeight(src_dib); rows++) {
memcpy(dst_bits, src_bits, FreeImage_GetLine(src_dib));
dst_bits += FreeImage_GetPitch(dst_dib);
src_bits += FreeImage_GetPitch(src_dib);
}
} else {
for(unsigned rows = 0; rows < FreeImage_GetHeight(src_dib); rows++) {
for(unsigned cols = 0; cols < FreeImage_GetLine(src_dib); cols += 2) {
RGBTRIPLE color_s;
RGBTRIPLE color_t;
WORD *tmp1 = (WORD *)&dst_bits[cols];
WORD *tmp2 = (WORD *)&src_bits[cols];
// convert 16-bit colors to 24-bit
color_s.rgbtRed = (BYTE)(((*tmp1 & FI16_565_RED_MASK) >> FI16_565_RED_SHIFT) << 3);
color_s.rgbtGreen = (BYTE)(((*tmp1 & FI16_565_GREEN_MASK) >> FI16_565_GREEN_SHIFT) << 2);
color_s.rgbtBlue = (BYTE)(((*tmp1 & FI16_565_BLUE_MASK) >> FI16_565_BLUE_SHIFT) << 3);
color_t.rgbtRed = (BYTE)(((*tmp2 & FI16_565_RED_MASK) >> FI16_565_RED_SHIFT) << 3);
color_t.rgbtGreen = (BYTE)(((*tmp2 & FI16_565_GREEN_MASK) >> FI16_565_GREEN_SHIFT) << 2);
color_t.rgbtBlue = (BYTE)(((*tmp2 & FI16_565_BLUE_MASK) >> FI16_565_BLUE_SHIFT) << 3);
// alpha blend
color_s.rgbtRed = (BYTE)(((color_t.rgbtRed - color_s.rgbtRed) * alpha + (color_s.rgbtRed << 8)) >> 8);
color_s.rgbtGreen = (BYTE)(((color_t.rgbtGreen - color_s.rgbtGreen) * alpha + (color_s.rgbtGreen << 8)) >> 8);
color_s.rgbtBlue = (BYTE)(((color_t.rgbtBlue - color_s.rgbtBlue) * alpha + (color_s.rgbtBlue << 8)) >> 8);
// convert 24-bit color back to 16-bit
*tmp1 = RGB565(color_s.rgbtRed, color_s.rgbtGreen, color_s.rgbtBlue);
}
dst_bits += FreeImage_GetPitch(dst_dib);
src_bits += FreeImage_GetPitch(src_dib);
}
}
return TRUE;
}
// ----------------------------------------------------------
// 24-bit
// ----------------------------------------------------------
static BOOL
Combine24(FIBITMAP *dst_dib, FIBITMAP *src_dib, unsigned x, unsigned y, unsigned alpha) {
// check the bit depth of src and dst images
if((FreeImage_GetBPP(dst_dib) != 24) || (FreeImage_GetBPP(src_dib) != 24)) {
return FALSE;
}
// check the size of src image
if((x + FreeImage_GetWidth(src_dib) > FreeImage_GetWidth(dst_dib)) || (y + FreeImage_GetHeight(src_dib) > FreeImage_GetHeight(dst_dib))) {
return FALSE;
}
BYTE *dst_bits = FreeImage_GetBits(dst_dib) + ((FreeImage_GetHeight(dst_dib) - FreeImage_GetHeight(src_dib) - y) * FreeImage_GetPitch(dst_dib)) + (x * 3);
BYTE *src_bits = FreeImage_GetBits(src_dib);
if(alpha > 255) {
// combine images
for(unsigned rows = 0; rows < FreeImage_GetHeight(src_dib); rows++) {
memcpy(dst_bits, src_bits, FreeImage_GetLine(src_dib));
dst_bits += FreeImage_GetPitch(dst_dib);
src_bits += FreeImage_GetPitch(src_dib);
}
} else {
// alpha blend images
for(unsigned rows = 0; rows < FreeImage_GetHeight(src_dib); rows++) {
for (unsigned cols = 0; cols < FreeImage_GetLine(src_dib); cols++) {
dst_bits[cols] = (BYTE)(((src_bits[cols] - dst_bits[cols]) * alpha + (dst_bits[cols] << 8)) >> 8);
}
dst_bits += FreeImage_GetPitch(dst_dib);
src_bits += FreeImage_GetPitch(src_dib);
}
}
return TRUE;
}
// ----------------------------------------------------------
// 32-bit
// ----------------------------------------------------------
static BOOL
Combine32(FIBITMAP *dst_dib, FIBITMAP *src_dib, unsigned x, unsigned y, unsigned alpha) {
// check the bit depth of src and dst images
if((FreeImage_GetBPP(dst_dib) != 32) || (FreeImage_GetBPP(src_dib) != 32)) {
return FALSE;
}
// check the size of src image
if((x + FreeImage_GetWidth(src_dib) > FreeImage_GetWidth(dst_dib)) || (y + FreeImage_GetHeight(src_dib) > FreeImage_GetHeight(dst_dib))) {
return FALSE;
}
BYTE *dst_bits = FreeImage_GetBits(dst_dib) + ((FreeImage_GetHeight(dst_dib) - FreeImage_GetHeight(src_dib) - y) * FreeImage_GetPitch(dst_dib)) + (x * 4);
BYTE *src_bits = FreeImage_GetBits(src_dib);
if (alpha > 255) {
// combine images
for(unsigned rows = 0; rows < FreeImage_GetHeight(src_dib); rows++) {
memcpy(dst_bits, src_bits, FreeImage_GetLine(src_dib));
dst_bits += FreeImage_GetPitch(dst_dib);
src_bits += FreeImage_GetPitch(src_dib);
}
} else {
// alpha blend images
for(unsigned rows = 0; rows < FreeImage_GetHeight(src_dib); rows++) {
for(unsigned cols = 0; cols < FreeImage_GetLine(src_dib); cols++) {
dst_bits[cols] = (BYTE)(((src_bits[cols] - dst_bits[cols]) * alpha + (dst_bits[cols] << 8)) >> 8);
}
dst_bits += FreeImage_GetPitch(dst_dib);
src_bits += FreeImage_GetPitch(src_dib);
}
}
return TRUE;
}
// ----------------------------------------------------------
// Any type other than FIBITMAP
// ----------------------------------------------------------
static BOOL
CombineSameType(FIBITMAP *dst_dib, FIBITMAP *src_dib, unsigned x, unsigned y) {
// check the bit depth of src and dst images
if(FreeImage_GetImageType(dst_dib) != FreeImage_GetImageType(src_dib)) {
return FALSE;
}
unsigned src_width = FreeImage_GetWidth(src_dib);
unsigned src_height = FreeImage_GetHeight(src_dib);
unsigned src_pitch = FreeImage_GetPitch(src_dib);
unsigned src_line = FreeImage_GetLine(src_dib);
unsigned dst_width = FreeImage_GetWidth(dst_dib);
unsigned dst_height = FreeImage_GetHeight(dst_dib);
unsigned dst_pitch = FreeImage_GetPitch(dst_dib);
// check the size of src image
if((x + src_width > dst_width) || (y + src_height > dst_height)) {
return FALSE;
}
BYTE *dst_bits = FreeImage_GetBits(dst_dib) + ((dst_height - src_height - y) * dst_pitch) + (x * (src_line / src_width));
BYTE *src_bits = FreeImage_GetBits(src_dib);
// combine images
for(unsigned rows = 0; rows < src_height; rows++) {
memcpy(dst_bits, src_bits, src_line);
dst_bits += dst_pitch;
src_bits += src_pitch;
}
return TRUE;
}
// ----------------------------------------------------------
// FreeImage interface
// ----------------------------------------------------------
/**
Copy a sub part of the current image and returns it as a FIBITMAP*.
Works with any bitmap type.
@param left Specifies the left position of the cropped rectangle.
@param top Specifies the top position of the cropped rectangle.
@param right Specifies the right position of the cropped rectangle.
@param bottom Specifies the bottom position of the cropped rectangle.
@return Returns the subimage if successful, NULL otherwise.
*/
FIBITMAP * DLL_CALLCONV
FreeImage_Copy(FIBITMAP *src, int left, int top, int right, int bottom) {
if(!FreeImage_HasPixels(src))
return NULL;
// normalize the rectangle
if(right < left) {
INPLACESWAP(left, right);
}
if(bottom < top) {
INPLACESWAP(top, bottom);
}
// check the size of the sub image
int src_width = FreeImage_GetWidth(src);
int src_height = FreeImage_GetHeight(src);
if((left < 0) || (right > src_width) || (top < 0) || (bottom > src_height)) {
return NULL;
}
// allocate the sub image
unsigned bpp = FreeImage_GetBPP(src);
int dst_width = (right - left);
int dst_height = (bottom - top);
FIBITMAP *dst =
FreeImage_AllocateT(FreeImage_GetImageType(src),
dst_width,
dst_height,
bpp,
FreeImage_GetRedMask(src), FreeImage_GetGreenMask(src), FreeImage_GetBlueMask(src));
if(NULL == dst) return NULL;
// get the dimensions
int dst_line = FreeImage_GetLine(dst);
int dst_pitch = FreeImage_GetPitch(dst);
int src_pitch = FreeImage_GetPitch(src);
// get the pointers to the bits and such
BYTE *src_bits = FreeImage_GetScanLine(src, src_height - top - dst_height);
switch(bpp) {
case 1:
// point to x = 0
break;
case 4:
// point to x = 0
break;
default:
{
// calculate the number of bytes per pixel
unsigned bytespp = FreeImage_GetLine(src) / FreeImage_GetWidth(src);
// point to x = left
src_bits += left * bytespp;
}
break;
}
// point to x = 0
BYTE *dst_bits = FreeImage_GetBits(dst);
// copy the palette
memcpy(FreeImage_GetPalette(dst), FreeImage_GetPalette(src), FreeImage_GetColorsUsed(src) * sizeof(RGBQUAD));
// copy the bits
if(bpp == 1) {
BOOL value;
unsigned y_src, y_dst;
for(int y = 0; y < dst_height; y++) {
y_src = y * src_pitch;
y_dst = y * dst_pitch;
for(int x = 0; x < dst_width; x++) {
// get bit at (y, x) in src image
value = (src_bits[y_src + ((left+x) >> 3)] & (0x80 >> ((left+x) & 0x07))) != 0;
// set bit at (y, x) in dst image
value ? dst_bits[y_dst + (x >> 3)] |= (0x80 >> (x & 0x7)) : dst_bits[y_dst + (x >> 3)] &= (0xff7f >> (x & 0x7));
}
}
}
else if(bpp == 4) {
BYTE shift, value;
unsigned y_src, y_dst;
for(int y = 0; y < dst_height; y++) {
y_src = y * src_pitch;
y_dst = y * dst_pitch;
for(int x = 0; x < dst_width; x++) {
// get nibble at (y, x) in src image
shift = (BYTE)((1 - (left+x) % 2) << 2);
value = (src_bits[y_src + ((left+x) >> 1)] & (0x0F << shift)) >> shift;
// set nibble at (y, x) in dst image
shift = (BYTE)((1 - x % 2) << 2);
dst_bits[y_dst + (x >> 1)] &= ~(0x0F << shift);
dst_bits[y_dst + (x >> 1)] |= ((value & 0x0F) << shift);
}
}
}
else if(bpp >= 8) {
for(int y = 0; y < dst_height; y++) {
memcpy(dst_bits + (y * dst_pitch), src_bits + (y * src_pitch), dst_line);
}
}
// copy metadata from src to dst
FreeImage_CloneMetadata(dst, src);
// copy transparency table
FreeImage_SetTransparencyTable(dst, FreeImage_GetTransparencyTable(src), FreeImage_GetTransparencyCount(src));
// copy background color
RGBQUAD bkcolor;
if( FreeImage_GetBackgroundColor(src, &bkcolor) ) {
FreeImage_SetBackgroundColor(dst, &bkcolor);
}
// clone resolution
FreeImage_SetDotsPerMeterX(dst, FreeImage_GetDotsPerMeterX(src));
FreeImage_SetDotsPerMeterY(dst, FreeImage_GetDotsPerMeterY(src));
// clone ICC profile
FIICCPROFILE *src_profile = FreeImage_GetICCProfile(src);
FIICCPROFILE *dst_profile = FreeImage_CreateICCProfile(dst, src_profile->data, src_profile->size);
dst_profile->flags = src_profile->flags;
return dst;
}
/**
Alpha blend or combine a sub part image with the current image.
The bit depth of dst bitmap must be greater than or equal to the bit depth of src.
Upper promotion of src is done internally. Supported bit depth equals to 1, 4, 8, 16, 24 or 32.
@param src Source subimage
@param left Specifies the left position of the sub image.
@param top Specifies the top position of the sub image.
@param alpha Alpha blend factor. The source and destination images are alpha blended if
alpha = 0..255. If alpha > 255, then the source image is combined to the destination image.
@return Returns TRUE if successful, FALSE otherwise.
*/
BOOL DLL_CALLCONV
FreeImage_Paste(FIBITMAP *dst, FIBITMAP *src, int left, int top, int alpha) {
BOOL bResult = FALSE;
if(!FreeImage_HasPixels(src) || !FreeImage_HasPixels(dst)) return FALSE;
// check the size of src image
if((left < 0) || (top < 0)) {
return FALSE;
}
if((left + FreeImage_GetWidth(src) > FreeImage_GetWidth(dst)) || (top + FreeImage_GetHeight(src) > FreeImage_GetHeight(dst))) {
return FALSE;
}
// check data type
const FREE_IMAGE_TYPE image_type = FreeImage_GetImageType(dst);
if(image_type != FreeImage_GetImageType(src)) {
// no conversion between data type is done
return FALSE;
}
if(image_type == FIT_BITMAP) {
FIBITMAP *clone = NULL;
// check the bit depth of src and dst images
unsigned bpp_src = FreeImage_GetBPP(src);
unsigned bpp_dst = FreeImage_GetBPP(dst);
BOOL isRGB565 = FALSE;
if ((FreeImage_GetRedMask(dst) == FI16_565_RED_MASK) && (FreeImage_GetGreenMask(dst) == FI16_565_GREEN_MASK) && (FreeImage_GetBlueMask(dst) == FI16_565_BLUE_MASK)) {
isRGB565 = TRUE;
} else {
// includes case where all the masks are 0
isRGB565 = FALSE;
}
// perform promotion if needed
if(bpp_dst == bpp_src) {
clone = src;
} else if(bpp_dst > bpp_src) {
// perform promotion
switch(bpp_dst) {
case 4:
clone = FreeImage_ConvertTo4Bits(src);
break;
case 8:
clone = FreeImage_ConvertTo8Bits(src);
break;
case 16:
if (isRGB565) {
clone = FreeImage_ConvertTo16Bits565(src);
} else {
// includes case where all the masks are 0
clone = FreeImage_ConvertTo16Bits555(src);
}
break;
case 24:
clone = FreeImage_ConvertTo24Bits(src);
break;
case 32:
clone = FreeImage_ConvertTo32Bits(src);
break;
default:
return FALSE;
}
} else {
return FALSE;
}
if(!clone) return FALSE;
// paste src to dst
switch(FreeImage_GetBPP(dst)) {
case 1:
bResult = Combine1(dst, clone, (unsigned)left, (unsigned)top, (unsigned)alpha);
break;
case 4:
bResult = Combine4(dst, clone, (unsigned)left, (unsigned)top, (unsigned)alpha);
break;
case 8:
bResult = Combine8(dst, clone, (unsigned)left, (unsigned)top, (unsigned)alpha);
break;
case 16:
if (isRGB565) {
bResult = Combine16_565(dst, clone, (unsigned)left, (unsigned)top, (unsigned)alpha);
} else {
// includes case where all the masks are 0
bResult = Combine16_555(dst, clone, (unsigned)left, (unsigned)top, (unsigned)alpha);
}
break;
case 24:
bResult = Combine24(dst, clone, (unsigned)left, (unsigned)top, (unsigned)alpha);
break;
case 32:
bResult = Combine32(dst, clone, (unsigned)left, (unsigned)top, (unsigned)alpha);
break;
}
if(clone != src)
FreeImage_Unload(clone);
}
else { // any type other than FITBITMAP
bResult = CombineSameType(dst, src, (unsigned)left, (unsigned)top);
}
return bResult;
}
// ----------------------------------------------------------
/** @brief Creates a dynamic read/write view into a FreeImage bitmap.
A dynamic view is a FreeImage bitmap with its own width and height, that,
however, shares its bits with another FreeImage bitmap. Typically, views
are used to define one or more rectangular sub-images of an existing
bitmap. All FreeImage operations, like saving, displaying and all the
toolkit functions, when applied to the view, only affect the view's
rectangular area.
Although the view's backing image's bits not need to be copied around,
which makes the view much faster than similar solutions using
FreeImage_Copy, a view uses some private memory that needs to be freed by
calling FreeImage_Unload on the view's handle to prevent memory leaks.
Only the backing image's pixels are shared by the view. For all other image
data, notably for the resolution, background color, color palette,
transparency table and for the ICC profile, the view gets a private copy
of the data. By default, the backing image's metadata is NOT copied to
the view.
As with all FreeImage functions that take a rectangle region, top and left
positions are included, whereas right and bottom positions are excluded
from the rectangle area.
Since the memory block shared by the backing image and the view must start
at a byte boundary, the value of parameter left must be a multiple of 8
for 1-bit images and a multiple of 2 for 4-bit images.
@param dib The FreeImage bitmap on which to create the view.
@param left The left position of the view's area.
@param top The top position of the view's area.
@param right The right position of the view's area.
@param bottom The bottom position of the view's area.
@return Returns a handle to the newly created view or NULL if the view
was not created.
*/
FIBITMAP * DLL_CALLCONV
FreeImage_CreateView(FIBITMAP *dib, unsigned left, unsigned top, unsigned right, unsigned bottom) {
if (!FreeImage_HasPixels(dib)) {
return NULL;
}
// normalize the rectangle
if (right < left) {
INPLACESWAP(left, right);
}
if (bottom < top) {
INPLACESWAP(top, bottom);
}
// check the size of the sub image
unsigned width = FreeImage_GetWidth(dib);
unsigned height = FreeImage_GetHeight(dib);
if (left < 0 || right > width || top < 0 || bottom > height) {
return NULL;
}
unsigned bpp = FreeImage_GetBPP(dib);
BYTE *bits = FreeImage_GetScanLine(dib, height - bottom);
switch (bpp) {
case 1:
if (left % 8 != 0) {
// view can only start at a byte boundary
return NULL;
}
bits += (left / 8);
break;
case 4:
if (left % 2 != 0) {
// view can only start at a byte boundary
return NULL;
}
bits += (left / 2);
break;
default:
bits += left * (bpp / 8);
break;
}
FIBITMAP *dst = FreeImage_AllocateHeaderForBits(bits, FreeImage_GetPitch(dib), FreeImage_GetImageType(dib),
right - left, bottom - top,
bpp,
FreeImage_GetRedMask(dib), FreeImage_GetGreenMask(dib), FreeImage_GetBlueMask(dib));
if (dst == NULL) {
return NULL;
}
// copy some basic image properties needed for displaying and saving
// resolution
FreeImage_SetDotsPerMeterX(dst, FreeImage_GetDotsPerMeterX(dib));
FreeImage_SetDotsPerMeterY(dst, FreeImage_GetDotsPerMeterY(dib));
// background color
RGBQUAD bkcolor;
if (FreeImage_GetBackgroundColor(dib, &bkcolor)) {
FreeImage_SetBackgroundColor(dst, &bkcolor);
}
// palette
memcpy(FreeImage_GetPalette(dst), FreeImage_GetPalette(dib), FreeImage_GetColorsUsed(dib) * sizeof(RGBQUAD));
// transparency table
FreeImage_SetTransparencyTable(dst, FreeImage_GetTransparencyTable(dib), FreeImage_GetTransparencyCount(dib));
// ICC profile
FIICCPROFILE *src_profile = FreeImage_GetICCProfile(dib);
FIICCPROFILE *dst_profile = FreeImage_CreateICCProfile(dst, src_profile->data, src_profile->size);
dst_profile->flags = src_profile->flags;
return dst;
}