2011-07-01 00:17:55 +04:00
|
|
|
/**
|
2012-10-09 07:02:04 +04:00
|
|
|
* FreeRDP: A Remote Desktop Protocol Implementation
|
2011-07-01 00:17:55 +04:00
|
|
|
* GDI Bitmap Functions
|
|
|
|
*
|
|
|
|
* Copyright 2010-2011 Marc-Andre Moreau <marcandre.moreau@gmail.com>
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
|
2012-08-15 01:09:01 +04:00
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
#include "config.h"
|
|
|
|
#endif
|
|
|
|
|
2011-07-01 00:17:55 +04:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdlib.h>
|
2012-08-15 01:09:01 +04:00
|
|
|
|
2011-08-31 12:35:50 +04:00
|
|
|
#include <freerdp/api.h>
|
2011-07-01 00:17:55 +04:00
|
|
|
#include <freerdp/freerdp.h>
|
2011-08-15 22:33:04 +04:00
|
|
|
#include <freerdp/gdi/gdi.h>
|
2011-10-03 04:28:20 +04:00
|
|
|
#include <freerdp/codec/color.h>
|
2011-07-01 00:17:55 +04:00
|
|
|
|
2011-08-22 21:08:01 +04:00
|
|
|
#include <freerdp/gdi/32bpp.h>
|
|
|
|
#include <freerdp/gdi/16bpp.h>
|
|
|
|
#include <freerdp/gdi/8bpp.h>
|
2011-07-01 00:17:55 +04:00
|
|
|
|
2011-08-22 21:08:01 +04:00
|
|
|
#include <freerdp/gdi/bitmap.h>
|
2011-07-01 00:17:55 +04:00
|
|
|
|
2011-10-21 07:15:18 +04:00
|
|
|
p_BitBlt BitBlt_[5] =
|
2011-07-01 00:17:55 +04:00
|
|
|
{
|
|
|
|
NULL,
|
|
|
|
BitBlt_8bpp,
|
|
|
|
BitBlt_16bpp,
|
|
|
|
NULL,
|
|
|
|
BitBlt_32bpp
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get pixel at the given coordinates.\n
|
|
|
|
* @msdn{dd144909}
|
|
|
|
* @param hdc device context
|
|
|
|
* @param nXPos pixel x position
|
|
|
|
* @param nYPos pixel y position
|
|
|
|
* @return pixel color
|
|
|
|
*/
|
|
|
|
|
2011-08-31 12:35:50 +04:00
|
|
|
INLINE GDI_COLOR gdi_GetPixel(HGDI_DC hdc, int nXPos, int nYPos)
|
2011-07-01 00:17:55 +04:00
|
|
|
{
|
|
|
|
HGDI_BITMAP hBmp = (HGDI_BITMAP) hdc->selectedObject;
|
|
|
|
GDI_COLOR* colorp = (GDI_COLOR*)&(hBmp->data[(nYPos * hBmp->width * hdc->bytesPerPixel) + nXPos * hdc->bytesPerPixel]);
|
|
|
|
return (GDI_COLOR) *colorp;
|
|
|
|
}
|
|
|
|
|
2012-10-09 11:01:37 +04:00
|
|
|
INLINE BYTE gdi_GetPixel_8bpp(HGDI_BITMAP hBmp, int X, int Y)
|
2011-07-01 00:17:55 +04:00
|
|
|
{
|
2012-10-09 11:01:37 +04:00
|
|
|
return *((BYTE*)&(hBmp->data[(Y * hBmp->width) + X]));
|
2011-07-01 00:17:55 +04:00
|
|
|
}
|
|
|
|
|
2012-10-09 11:01:37 +04:00
|
|
|
INLINE UINT16 gdi_GetPixel_16bpp(HGDI_BITMAP hBmp, int X, int Y)
|
2011-07-01 00:17:55 +04:00
|
|
|
{
|
2012-10-09 11:01:37 +04:00
|
|
|
return *((UINT16*)&(hBmp->data[(Y * hBmp->width * 2) + X * 2]));
|
2011-07-01 00:17:55 +04:00
|
|
|
}
|
|
|
|
|
2012-10-09 11:26:39 +04:00
|
|
|
INLINE UINT32 gdi_GetPixel_32bpp(HGDI_BITMAP hBmp, int X, int Y)
|
2011-07-01 00:17:55 +04:00
|
|
|
{
|
2012-10-09 11:26:39 +04:00
|
|
|
return *((UINT32*)&(hBmp->data[(Y * hBmp->width * 4) + X * 4]));
|
2011-07-01 00:17:55 +04:00
|
|
|
}
|
|
|
|
|
2012-10-09 11:01:37 +04:00
|
|
|
INLINE BYTE* gdi_GetPointer_8bpp(HGDI_BITMAP hBmp, int X, int Y)
|
2011-07-01 00:17:55 +04:00
|
|
|
{
|
2012-10-09 11:01:37 +04:00
|
|
|
return ((BYTE*)&(hBmp->data[(Y * hBmp->width) + X]));
|
2011-07-01 00:17:55 +04:00
|
|
|
}
|
|
|
|
|
2012-10-09 11:01:37 +04:00
|
|
|
INLINE UINT16* gdi_GetPointer_16bpp(HGDI_BITMAP hBmp, int X, int Y)
|
2011-07-01 00:17:55 +04:00
|
|
|
{
|
2012-10-09 11:01:37 +04:00
|
|
|
return ((UINT16*)&(hBmp->data[(Y * hBmp->width * 2) + X * 2]));
|
2011-07-01 00:17:55 +04:00
|
|
|
}
|
|
|
|
|
2012-10-09 11:26:39 +04:00
|
|
|
INLINE UINT32* gdi_GetPointer_32bpp(HGDI_BITMAP hBmp, int X, int Y)
|
2011-07-01 00:17:55 +04:00
|
|
|
{
|
2012-10-09 11:26:39 +04:00
|
|
|
return ((UINT32*)&(hBmp->data[(Y * hBmp->width * 4) + X * 4]));
|
2011-07-01 00:17:55 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set pixel at the given coordinates.\n
|
|
|
|
* @msdn{dd145078}
|
|
|
|
* @param hdc device context
|
|
|
|
* @param X pixel x position
|
|
|
|
* @param Y pixel y position
|
|
|
|
* @param crColor new pixel color
|
|
|
|
* @return
|
|
|
|
*/
|
|
|
|
|
2011-08-31 12:35:50 +04:00
|
|
|
INLINE GDI_COLOR gdi_SetPixel(HGDI_DC hdc, int X, int Y, GDI_COLOR crColor)
|
2011-07-01 00:17:55 +04:00
|
|
|
{
|
|
|
|
HGDI_BITMAP hBmp = (HGDI_BITMAP) hdc->selectedObject;
|
|
|
|
*((GDI_COLOR*)&(hBmp->data[(Y * hBmp->width * hdc->bytesPerPixel) + X * hdc->bytesPerPixel])) = crColor;
|
2015-04-17 17:21:55 +03:00
|
|
|
return crColor;
|
2011-07-01 00:17:55 +04:00
|
|
|
}
|
|
|
|
|
2012-10-09 11:01:37 +04:00
|
|
|
INLINE void gdi_SetPixel_8bpp(HGDI_BITMAP hBmp, int X, int Y, BYTE pixel)
|
2011-07-01 00:17:55 +04:00
|
|
|
{
|
2012-10-09 11:01:37 +04:00
|
|
|
*((BYTE*)&(hBmp->data[(Y * hBmp->width) + X])) = pixel;
|
2011-07-01 00:17:55 +04:00
|
|
|
}
|
|
|
|
|
2012-10-09 11:01:37 +04:00
|
|
|
INLINE void gdi_SetPixel_16bpp(HGDI_BITMAP hBmp, int X, int Y, UINT16 pixel)
|
2011-07-01 00:17:55 +04:00
|
|
|
{
|
2012-10-09 11:01:37 +04:00
|
|
|
*((UINT16*)&(hBmp->data[(Y * hBmp->width * 2) + X * 2])) = pixel;
|
2011-07-01 00:17:55 +04:00
|
|
|
}
|
|
|
|
|
2012-10-09 11:26:39 +04:00
|
|
|
INLINE void gdi_SetPixel_32bpp(HGDI_BITMAP hBmp, int X, int Y, UINT32 pixel)
|
2011-07-01 00:17:55 +04:00
|
|
|
{
|
2012-10-09 11:26:39 +04:00
|
|
|
*((UINT32*)&(hBmp->data[(Y * hBmp->width * 4) + X * 4])) = pixel;
|
2011-07-01 00:17:55 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create a new bitmap with the given width, height, color format and pixel buffer.\n
|
|
|
|
* @msdn{dd183485}
|
|
|
|
* @param nWidth width
|
|
|
|
* @param nHeight height
|
|
|
|
* @param cBitsPerPixel bits per pixel
|
|
|
|
* @param data pixel buffer
|
2015-08-12 11:49:59 +03:00
|
|
|
* @param fkt_free The function used for deallocation of the buffer, NULL for none.
|
2011-07-01 00:17:55 +04:00
|
|
|
* @return new bitmap
|
|
|
|
*/
|
|
|
|
|
2015-08-26 13:14:46 +03:00
|
|
|
HGDI_BITMAP gdi_CreateBitmap(int nWidth, int nHeight, int cBitsPerPixel, BYTE* data)
|
|
|
|
{
|
|
|
|
return gdi_CreateBitmapEx(nWidth, nHeight, cBitsPerPixel, data, _aligned_free);
|
|
|
|
}
|
|
|
|
|
|
|
|
HGDI_BITMAP gdi_CreateBitmapEx(int nWidth, int nHeight, int cBitsPerPixel, BYTE* data,
|
2015-08-12 11:49:59 +03:00
|
|
|
void (*fkt_free)(void*))
|
2011-07-01 00:17:55 +04:00
|
|
|
{
|
2015-03-16 15:55:06 +03:00
|
|
|
HGDI_BITMAP hBitmap = (HGDI_BITMAP) calloc(1, sizeof(GDI_BITMAP));
|
|
|
|
|
|
|
|
if (!hBitmap)
|
|
|
|
return NULL;
|
|
|
|
|
2011-07-01 00:17:55 +04:00
|
|
|
hBitmap->objectType = GDIOBJECT_BITMAP;
|
|
|
|
hBitmap->bitsPerPixel = cBitsPerPixel;
|
|
|
|
hBitmap->bytesPerPixel = (cBitsPerPixel + 1) / 8;
|
|
|
|
hBitmap->scanline = nWidth * hBitmap->bytesPerPixel;
|
|
|
|
hBitmap->width = nWidth;
|
|
|
|
hBitmap->height = nHeight;
|
|
|
|
hBitmap->data = data;
|
2015-08-12 11:49:59 +03:00
|
|
|
hBitmap->free = fkt_free;
|
2015-03-16 15:55:06 +03:00
|
|
|
|
2011-07-01 00:17:55 +04:00
|
|
|
return hBitmap;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create a new bitmap of the given width and height compatible with the current device context.\n
|
|
|
|
* @msdn{dd183488}
|
|
|
|
* @param hdc device context
|
|
|
|
* @param nWidth width
|
|
|
|
* @param nHeight height
|
|
|
|
* @return new bitmap
|
|
|
|
*/
|
|
|
|
|
|
|
|
HGDI_BITMAP gdi_CreateCompatibleBitmap(HGDI_DC hdc, int nWidth, int nHeight)
|
|
|
|
{
|
2015-08-12 11:49:59 +03:00
|
|
|
HGDI_BITMAP hBitmap = (HGDI_BITMAP) calloc(1, sizeof(GDI_BITMAP));
|
2014-07-08 23:07:19 +04:00
|
|
|
|
|
|
|
if (!hBitmap)
|
|
|
|
return NULL;
|
|
|
|
|
2011-07-01 00:17:55 +04:00
|
|
|
hBitmap->objectType = GDIOBJECT_BITMAP;
|
|
|
|
hBitmap->bytesPerPixel = hdc->bytesPerPixel;
|
|
|
|
hBitmap->bitsPerPixel = hdc->bitsPerPixel;
|
|
|
|
hBitmap->width = nWidth;
|
|
|
|
hBitmap->height = nHeight;
|
2014-07-08 23:07:19 +04:00
|
|
|
hBitmap->data = _aligned_malloc(nWidth * nHeight * hBitmap->bytesPerPixel, 16);
|
2015-08-12 11:49:59 +03:00
|
|
|
hBitmap->free = _aligned_free;
|
2015-04-14 11:14:23 +03:00
|
|
|
if (!hBitmap->data)
|
|
|
|
{
|
|
|
|
free(hBitmap);
|
|
|
|
return NULL;
|
|
|
|
}
|
2011-07-01 00:17:55 +04:00
|
|
|
hBitmap->scanline = nWidth * hBitmap->bytesPerPixel;
|
2015-03-16 15:55:06 +03:00
|
|
|
|
2011-07-01 00:17:55 +04:00
|
|
|
return hBitmap;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Perform a bit blit operation on the given pixel buffers.\n
|
|
|
|
* @msdn{dd183370}
|
|
|
|
* @param hdcDest destination device context
|
|
|
|
* @param nXDest destination x1
|
|
|
|
* @param nYDest destination y1
|
|
|
|
* @param nWidth width
|
|
|
|
* @param nHeight height
|
|
|
|
* @param hdcSrc source device context
|
|
|
|
* @param nXSrc source x1
|
|
|
|
* @param nYSrc source y1
|
|
|
|
* @param rop raster operation code
|
2015-04-17 17:21:55 +03:00
|
|
|
* @return 0 on failure, non-zero otherwise
|
2011-07-01 00:17:55 +04:00
|
|
|
*/
|
|
|
|
|
2015-06-26 15:32:38 +03:00
|
|
|
BOOL gdi_BitBlt(HGDI_DC hdcDest, int nXDest, int nYDest, int nWidth, int nHeight, HGDI_DC hdcSrc, int nXSrc, int nYSrc, DWORD rop)
|
2011-07-01 00:17:55 +04:00
|
|
|
{
|
2011-10-21 07:15:18 +04:00
|
|
|
p_BitBlt _BitBlt = BitBlt_[IBPP(hdcDest->bitsPerPixel)];
|
2011-07-01 00:17:55 +04:00
|
|
|
|
2015-06-26 15:32:38 +03:00
|
|
|
if (_BitBlt == NULL)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
return _BitBlt(hdcDest, nXDest, nYDest, nWidth, nHeight, hdcSrc, nXSrc, nYSrc, rop);
|
2011-07-01 00:17:55 +04:00
|
|
|
}
|