isle/LEGO1/mxbitmap.cpp
2023-07-28 22:15:26 -04:00

224 lines
5.4 KiB
C++

#include "mxbitmap.h"
// The way that the BITMAPFILEHEADER structure ensures the file type is by ensuring it is "BM", which is literally just 0x424d.
// Sources: https://learn.microsoft.com/en-us/windows/win32/api/wingdi/ns-wingdi-bitmapfileheader, DirectX Complete (1998)
// GLOBAL: LEGO1 0x10102184
WORD g_bitmapSignature = 0x424d;
// OFFSET: LEGO1 0x100bc980
MxBitmap::MxBitmap()
{
this->m_info = NULL;
this->m_bmiHeader = NULL;
this->m_paletteData = NULL;
this->m_data = NULL;
this->m_bmiColorsProvided = FALSE;
this->m_palette = NULL;
}
// OFFSET: LEGO1 0x100bca10
MxBitmap::~MxBitmap()
{
if (this->m_info != NULL)
delete m_info;
if (this->m_data != NULL)
delete m_data;
if (this->m_palette != NULL)
delete m_palette;
}
// OFFSET: LEGO1 0x100bcc40 STUB
int MxBitmap::vtable14(int)
{
return 0;
}
// OFFSET: LEGO1 0x100bcba0
int MxBitmap::vtable18(BITMAPINFOHEADER *p_bmiHeader)
{
BITMAPINFO* bmi;
LPVOID *data;
MxResult result = FAILURE;
int height;
int width;
bmi = new BITMAPINFO;
width = p_bmiHeader->biWidth;
height = p_bmiHeader->biHeight;
this->m_info = bmi;
if (bmi != NULL) {
data = (LPVOID*) malloc((width + 3U & 0xfffffffc) * height);
this->m_data = (LPVOID*) data;
if(data != NULL) {
bmi = this->m_info;
for (width = 0x10a; width != 0; width--) {
(bmi->bmiHeader).biSize = p_bmiHeader->biSize;
p_bmiHeader = (BITMAPINFOHEADER*)&p_bmiHeader->biWidth;
bmi = (BITMAPINFO*)&(bmi->bmiHeader).biWidth;
}
result = SUCCESS;
this->m_bmiHeader = &this->m_info->bmiHeader;
this->m_paletteData = this->m_info->bmiColors;
}
}
if (result != SUCCESS) {
if (this->m_info != NULL) {
delete this->m_info;
this->m_info = NULL;
}
if (this->m_data != NULL) {
delete this->m_data;
this->m_data = NULL;
}
}
return result;
}
// OFFSET: LEGO1 0x100bcaa0 STUB
int MxBitmap::vtable1c(int p_width, int p_height, MxPalette *p_palette, int)
{
return 0;
}
// OFFSET: LEGO1 0x100bcd60
MxResult MxBitmap::LoadFile(HANDLE p_handle)
{
void* lpBuffer;
BITMAPINFO *infoHdr;
MxS32 height;
MxBool operation_ret;
MxResult result = FAILURE;
DWORD bytesRead;
BITMAPFILEHEADER hdr;
operation_ret = ReadFile(p_handle, &hdr, 14, &bytesRead, NULL);
if ((operation_ret != 0) && (hdr.bfType == g_bitmapSignature)) {
infoHdr = (BITMAPINFO*) malloc(1064);
this->m_info = infoHdr;
if(infoHdr != NULL) {
operation_ret = ReadFile(p_handle, infoHdr, 1064, &bytesRead, NULL);
if ((operation_ret != 0) && ((this->m_info->bmiHeader).biBitCount == 8)) {
lpBuffer = (void*) malloc(hdr.bfSize - 1078);
this->m_data = (LPVOID*) lpBuffer;
if (this->m_data != NULL) {
operation_ret = ReadFile(p_handle, lpBuffer, hdr.bfSize - 1078, &bytesRead, NULL);
if(operation_ret != 0) {
this->m_bmiHeader = &this->m_info->bmiHeader;
this->m_paletteData = this->m_info->bmiColors;
if((this->m_info->bmiHeader).biSizeImage == 0) {
height = (this->m_info->bmiHeader).biHeight;
if (height < 1) {
height *= -1;
}
(this->m_info->bmiHeader).biSizeImage = ((this->m_info->bmiHeader).biWidth + 3U & 0xfffffffc) * height;
}
result = SUCCESS;
}
}
}
}
}
if (result != SUCCESS) {
if (this->m_info != NULL) {
delete this->m_info;
this->m_info = NULL;
}
if (this->m_data != NULL) {
delete this->m_data;
this->m_data = NULL;
}
}
return result;
}
// OFFSET: LEGO1 0x100bcd10
MxLong MxBitmap::Read(const char *p_filename)
{
MxResult result = FAILURE;
HANDLE handle = CreateFileA(
p_filename,
GENERIC_READ,
FILE_SHARE_READ,
NULL,
OPEN_EXISTING,
FILE_ATTRIBUTE_NORMAL,
NULL
);
if (handle != INVALID_HANDLE_VALUE && !LoadFile(handle))
result = SUCCESS;
if (handle)
CloseHandle(handle);
return result;
}
// OFFSET: LEGO1 0x1004e0d0
int MxBitmap::vtable28(int)
{
return -1;
}
// OFFSET: LEGO1 0x100bce70 STUB
void MxBitmap::vtable2c(int, int, int, int, int, int, int)
{
}
// OFFSET: LEGO1 0x100bd020 STUB
void MxBitmap::vtable30(int, int, int, int, int, int, int)
{
}
// OFFSET: LEGO1 0x100bd1c0
MxPalette *MxBitmap::CreatePalette()
{
// FIXME: doesn't match
MxPalette *pal;
MxPalette *ppal;
MxBool success;
pal = NULL;
success = FALSE;
if(this->m_bmiColorsProvided == FALSE) {
ppal = new MxPalette(this->m_paletteData);
if (ppal != NULL) {
pal = ppal;
}
} else {
if(this->m_bmiColorsProvided != TRUE) {
if(!success && pal != NULL) {
delete pal;
pal = NULL;
}
}
pal = this->m_palette->Clone();
}
if(pal != NULL) {
success = TRUE;
}
return pal;
}
// OFFSET: LEGO1 0x100bd280 STUB
void MxBitmap::vtable38(void*)
{
}
// OFFSET: LEGO1 0x100bd2d0 STUB
int MxBitmap::vtable3c(MxBool)
{
return 0;
}
// OFFSET: LEGO1 0x100bd3e0
MxResult MxBitmap::CopyColorData(HDC p_hdc, int p_xSrc, int p_ySrc, int p_xDest, int p_yDest, int p_destWidth, int p_destHeight)
{
// Compression fix?
if ((this->m_bmiHeader->biCompression != 16) && (0 < this->m_bmiHeader->biHeight)) {
p_ySrc = (this->m_bmiHeader->biHeight - p_destHeight) - p_ySrc;
}
return StretchDIBits(p_hdc, p_xDest, p_yDest, p_destWidth, p_destHeight, p_xSrc, p_ySrc, p_destWidth, p_destHeight, this->m_data, this->m_info, this->m_bmiColorsProvided, SRCCOPY);
}