From 10ebdfa60801425280fae1d41d34b7198c84c816 Mon Sep 17 00:00:00 2001 From: MS <disinvite@users.noreply.github.com> Date: Wed, 25 Oct 2023 06:02:36 -0400 Subject: [PATCH] MxVideoPresenter::IsHit (#244) * MxVideoPresenter::IsHit * Apply clang-format * Minor logic fix --------- Co-authored-by: Christian Semmler <mail@csemmler.com> --- LEGO1/mxbitmap.h | 1 + LEGO1/mxdsaction.h | 2 + LEGO1/mxvideopresenter.cpp | 76 +++++++++++++++++++++++++++++++++++++- LEGO1/mxvideopresenter.h | 27 +++++++------- 4 files changed, 91 insertions(+), 15 deletions(-) diff --git a/LEGO1/mxbitmap.h b/LEGO1/mxbitmap.h index 28a4bfaa..e0a8dffc 100644 --- a/LEGO1/mxbitmap.h +++ b/LEGO1/mxbitmap.h @@ -58,6 +58,7 @@ public: inline BITMAPINFOHEADER* GetBmiHeader() const { return m_bmiHeader; } inline MxLong GetBmiWidth() const { return m_bmiHeader->biWidth; } + inline MxLong GetBmiStride() const { return ((m_bmiHeader->biWidth + 3) & -4); } inline MxLong GetBmiHeight() const { return m_bmiHeader->biHeight; } inline MxLong GetBmiHeightAbs() const { diff --git a/LEGO1/mxdsaction.h b/LEGO1/mxdsaction.h index 0c27235a..72d7afac 100644 --- a/LEGO1/mxdsaction.h +++ b/LEGO1/mxdsaction.h @@ -14,10 +14,12 @@ public: enum { Flag_Looping = 0x01, Flag_Bit3 = 0x04, + Flag_Bit4 = 0x08, Flag_Bit5 = 0x10, Flag_Enabled = 0x20, Flag_Parsed = 0x80, Flag_Bit9 = 0x200, + Flag_Bit10 = 0x400, }; __declspec(dllexport) MxDSAction(); diff --git a/LEGO1/mxvideopresenter.cpp b/LEGO1/mxvideopresenter.cpp index b1bef478..7378b703 100644 --- a/LEGO1/mxvideopresenter.cpp +++ b/LEGO1/mxvideopresenter.cpp @@ -118,14 +118,14 @@ MxVideoPresenter::AlphaMask::AlphaMask(const MxBitmap& p_bitmap) // TODO: would prefer not to use goto if we can figure this structure out seek_to_last_row: - bitmap_src_ptr = ((p_bitmap.GetBmiWidth() + 3) & -4) * rows_before_top + p_bitmap.GetBitmapData(); + bitmap_src_ptr = p_bitmap.GetBmiStride() * rows_before_top + p_bitmap.GetBitmapData(); } // How many bytes are there for each row of the bitmap? // (i.e. the image stride) // If this is a bottom-up DIB, we will walk it in reverse. // TODO: Same rounding trick as in MxBitmap - MxS32 row_seek = ((m_height + 3) & -4); + MxS32 row_seek = ((m_width + 3) & -4); if (p_bitmap.GetBmiHeight() < 0) row_seek = -row_seek; @@ -168,6 +168,16 @@ MxVideoPresenter::AlphaMask::~AlphaMask() delete[] m_bitmask; } +// OFFSET: LEGO1 0x100b26f0 +MxS32 MxVideoPresenter::AlphaMask::IsHit(MxU32 p_x, MxU32 p_y) +{ + if (p_x >= m_width || p_y >= m_height) + return 0; + + MxS32 pos = p_y * m_width + p_x; + return m_bitmask[pos / 8] & (1 << abs(abs(pos) & 7)) ? 1 : 0; +} + // OFFSET: LEGO1 0x100b2760 void MxVideoPresenter::Init() { @@ -228,6 +238,68 @@ void MxVideoPresenter::VTable0x64() // TODO } +// OFFSET: LEGO1 0x100b2900 +MxBool MxVideoPresenter::IsHit(MxS32 p_x, MxS32 p_y) +{ + MxDSAction* action = GetAction(); + if ((action == NULL) || (((action->GetFlags() & MxDSAction::Flag_Bit10) == 0) && !IsEnabled()) || + (!m_bitmap && !m_alpha)) + return FALSE; + + if (!m_bitmap) + return m_alpha->IsHit(p_x - GetLocationX(), p_y - GetLocationY()); + + MxLong heightAbs = m_bitmap->GetBmiHeightAbs(); + + MxLong min_x = GetLocationX(); + MxLong min_y = GetLocationY(); + + MxLong max_y = min_y + heightAbs; + MxLong max_x = min_x + m_bitmap->GetBmiWidth(); + + if (p_x < min_x || p_x >= max_x || p_y < min_y || p_y >= max_y) + return FALSE; + + MxU8* pixel; + + MxLong biCompression = m_bitmap->GetBmiHeader()->biCompression; + MxLong height = m_bitmap->GetBmiHeight(); + MxLong seek_row; + + // DECOMP: Same basic layout as AlphaMask constructor + // The idea here is to again seek to the correct place in the bitmap's + // m_data buffer. The x,y args are (most likely) screen x and y, so we + // need to shift that to coordinates local to the bitmap by removing + // the MxPresenter location x and y coordinates. + if (biCompression == BI_RGB) { + if (biCompression == BI_RGB_TOPDOWN || height < 0) { + seek_row = p_y - GetLocationY(); + } + else { + height = height > 0 ? height : -height; + seek_row = height - p_y - 1 + GetLocationY(); + } + pixel = m_bitmap->GetBmiStride() * seek_row + m_bitmap->GetBitmapData() - GetLocationX() + p_x; + } + else if (biCompression == BI_RGB_TOPDOWN) { + pixel = m_bitmap->GetBitmapData(); + } + else { + height = height > 0 ? height : -height; + height--; + pixel = m_bitmap->GetBmiStride() * height + m_bitmap->GetBitmapData(); + } + + // DECOMP: m_flags is 1 byte, so no enum here + if (m_flags & 0x10) + return (MxBool) *pixel; + + if ((GetAction()->GetFlags() & MxDSAction::Flag_Bit4) && *pixel == 0) + return FALSE; + + return TRUE; +} + // OFFSET: LEGO1 0x100b2a70 STUB void MxVideoPresenter::VTable0x6c() { diff --git a/LEGO1/mxvideopresenter.h b/LEGO1/mxvideopresenter.h index 31281cc6..14ad2d78 100644 --- a/LEGO1/mxvideopresenter.h +++ b/LEGO1/mxvideopresenter.h @@ -30,20 +30,19 @@ public: virtual void Destroy() override; // vtable+0x38 - virtual void VTable0x5c(undefined4 p_unknown1); // vtable+0x5c - virtual void VTable0x60(); // vtable+0x60 - virtual void VTable0x64(); // vtable+0x64 - virtual void VTable0x68(undefined4 p_unknown1); // vtable+0x68 - virtual void VTable0x6c(); // vtable+0x6c - virtual void VTable0x70(); // vtable+0x70 - virtual undefined VTable0x74(); // vtable+0x74 - virtual LPDIRECTDRAWSURFACE VTable0x78(); // vtable+0x78 - virtual MxBool VTable0x7c(); // vtable+0x7c - virtual MxS32 GetWidth(); // vtable+0x80 - virtual MxS32 GetHeight(); // vtable+0x84 + virtual MxBool IsHit(MxS32 p_x, MxS32 p_y) override; // vtable+0x50 + virtual void VTable0x5c(undefined4 p_unknown1); // vtable+0x5c + virtual void VTable0x60(); // vtable+0x60 + virtual void VTable0x64(); // vtable+0x64 + virtual void VTable0x68(undefined4 p_unknown1); // vtable+0x68 + virtual void VTable0x6c(); // vtable+0x6c + virtual void VTable0x70(); // vtable+0x70 + virtual undefined VTable0x74(); // vtable+0x74 + virtual LPDIRECTDRAWSURFACE VTable0x78(); // vtable+0x78 + virtual MxBool VTable0x7c(); // vtable+0x7c + virtual MxS32 GetWidth(); // vtable+0x80 + virtual MxS32 GetHeight(); // vtable+0x84 - // TODO: Not sure what this is. Seems to have size of 12 bytes - // based on 0x100b9e9a. Values are copied from the bitmap header. // SIZE 0xc struct AlphaMask { MxU8* m_bitmask; @@ -53,6 +52,8 @@ public: AlphaMask(const MxBitmap&); AlphaMask(const AlphaMask&); virtual ~AlphaMask(); + + MxS32 IsHit(MxU32 p_x, MxU32 p_y); }; MxBitmap* m_bitmap;