Clear unknowns in MxControlPresenter (#1606)

This commit is contained in:
Fabian Neundorf
2025-07-04 21:53:50 +02:00
committed by GitHub
parent df3d144ed9
commit 5544640b22
5 changed files with 70 additions and 63 deletions

View File

@@ -44,21 +44,28 @@ public:
void EndAction() override; // vtable+0x40 void EndAction() override; // vtable+0x40
MxBool HasTickleStatePassed(TickleState p_tickleState) override; // vtable+0x48 MxBool HasTickleStatePassed(TickleState p_tickleState) override; // vtable+0x48
void Enable(MxBool p_enable) override; // vtable+0x54 void Enable(MxBool p_enable) override; // vtable+0x54
virtual void VTable0x6c(MxS16 p_unk0x4e); // vtable+0x6c virtual void UpdateEnabledChild(MxS16 p_enabledChild); // vtable+0x6c
MxBool FUN_10044480(LegoControlManagerNotificationParam* p_param, MxPresenter* p_presenter); MxBool Notify(LegoControlManagerNotificationParam* p_param, MxPresenter* p_presenter);
MxBool FUN_10044270(MxS32 p_x, MxS32 p_y, MxPresenter* p_presenter); MxBool CheckButtonDown(MxS32 p_x, MxS32 p_y, MxPresenter* p_presenter);
MxS16 GetUnknown0x4e() { return m_unk0x4e; } MxS16 GetEnabledChild() { return m_enabledChild; }
private: private:
MxS16 m_unk0x4c; // 0x4c enum {
MxS16 m_unk0x4e; // 0x4e e_none,
MxBool m_unk0x50; // 0x50 e_toggle,
MxS16 m_unk0x52; // 0x52 e_grid,
MxS16 m_unk0x54; // 0x54 e_map,
MxS16 m_unk0x56; // 0x56 };
MxS16* m_states; // 0x58
MxS16 m_style; // 0x4c
MxS16 m_enabledChild; // 0x4e
MxBool m_unk0x50; // 0x50
MxS16 m_columnsOrRows; // 0x52
MxS16 m_rowsOrColumns; // 0x54
MxS16 m_stateOrCellIndex; // 0x56
MxS16* m_states; // 0x58
}; };
// SYNTHETIC: LEGO1 0x100440f0 // SYNTHETIC: LEGO1 0x100440f0

View File

@@ -131,7 +131,7 @@ void Radio::Stop()
MxControlPresenter* presenter = (MxControlPresenter*) world->Find(world->GetAtomId(), IsleScript::c_Radio_Ctl); MxControlPresenter* presenter = (MxControlPresenter*) world->Find(world->GetAtomId(), IsleScript::c_Radio_Ctl);
if (presenter) { if (presenter) {
presenter->VTable0x6c(0); presenter->UpdateEnabledChild(0);
} }
BackgroundAudioManager()->Stop(); BackgroundAudioManager()->Stop();

View File

@@ -1650,8 +1650,8 @@ void LegoCarBuild::FUN_10025db0(const char* p_param1, undefined4 p_param2)
} }
} }
else { else {
if (m_unk0x33c->GetUnknown0x4e() != sVar3) { if (m_unk0x33c->GetEnabledChild() != sVar3) {
m_unk0x33c->VTable0x6c(sVar3); m_unk0x33c->UpdateEnabledChild(sVar3);
} }
g_unk0x100f11cc = -1; g_unk0x100f11cc = -1;
@@ -1664,7 +1664,7 @@ void LegoCarBuild::FUN_10025e40()
{ {
SetPresentersEnabled(m_presentersEnabled); SetPresentersEnabled(m_presentersEnabled);
if (m_unk0x33c && m_Yellow_Ctl != m_unk0x33c) { if (m_unk0x33c && m_Yellow_Ctl != m_unk0x33c) {
m_unk0x33c->VTable0x6c(0); m_unk0x33c->UpdateEnabledChild(0);
} }
} }

View File

@@ -16,12 +16,12 @@ DECOMP_SIZE_ASSERT(MxControlPresenter, 0x5c)
// FUNCTION: LEGO1 0x10043f50 // FUNCTION: LEGO1 0x10043f50
MxControlPresenter::MxControlPresenter() MxControlPresenter::MxControlPresenter()
{ {
m_unk0x4c = 0; m_style = e_none;
m_unk0x4e = -1; m_enabledChild = -1;
m_unk0x50 = FALSE; m_unk0x50 = FALSE;
m_unk0x52 = 0; m_columnsOrRows = 0;
m_states = NULL; m_states = NULL;
m_unk0x54 = 0; m_rowsOrColumns = 0;
} }
// FUNCTION: LEGO1 0x10044110 // FUNCTION: LEGO1 0x10044110
@@ -35,7 +35,7 @@ MxControlPresenter::~MxControlPresenter()
// FUNCTION: LEGO1 0x10044180 // FUNCTION: LEGO1 0x10044180
MxResult MxControlPresenter::AddToManager() MxResult MxControlPresenter::AddToManager()
{ {
m_unk0x4e = 0; m_enabledChild = 0;
return SUCCESS; return SUCCESS;
} }
@@ -49,11 +49,11 @@ MxResult MxControlPresenter::StartAction(MxStreamController* p_controller, MxDSA
MxS16 i = 0; MxS16 i = 0;
for (MxCompositePresenterList::iterator it = m_list.begin(); it != m_list.end(); it++) { for (MxCompositePresenterList::iterator it = m_list.begin(); it != m_list.end(); it++) {
(*it)->Enable((m_unk0x4c != 3 || m_unk0x4e) && IsEnabled() ? m_unk0x4e == i : FALSE); (*it)->Enable((m_style != e_map || m_enabledChild) && IsEnabled() ? m_enabledChild == i : FALSE);
i++; i++;
} }
if (m_unk0x4c == 3) { if (m_style == e_map) {
MxDSAction* action = (*m_list.begin())->GetAction(); MxDSAction* action = (*m_list.begin())->GetAction();
action->SetFlags(action->GetFlags() | MxDSAction::c_bit11); action->SetFlags(action->GetFlags() | MxDSAction::c_bit11);
} }
@@ -74,12 +74,12 @@ void MxControlPresenter::EndAction()
// FUNCTION: LEGO1 0x10044270 // FUNCTION: LEGO1 0x10044270
// FUNCTION: BETA10 0x100eae68 // FUNCTION: BETA10 0x100eae68
MxBool MxControlPresenter::FUN_10044270(MxS32 p_x, MxS32 p_y, MxPresenter* p_presenter) MxBool MxControlPresenter::CheckButtonDown(MxS32 p_x, MxS32 p_y, MxPresenter* p_presenter)
{ {
assert(p_presenter); assert(p_presenter);
MxVideoPresenter* presenter = (MxVideoPresenter*) p_presenter; MxVideoPresenter* presenter = (MxVideoPresenter*) p_presenter;
if (m_unk0x4c == 3) { if (m_style == e_map) {
MxStillPresenter* map = (MxStillPresenter*) m_list.front(); MxStillPresenter* map = (MxStillPresenter*) m_list.front();
assert(map && map->IsA("MxStillPresenter")); assert(map && map->IsA("MxStillPresenter"));
@@ -94,23 +94,23 @@ MxBool MxControlPresenter::FUN_10044270(MxS32 p_x, MxS32 p_y, MxPresenter* p_pre
? NULL ? NULL
: map->GetBitmap()->GetStart(p_x - rect.GetLeft(), p_y - rect.GetTop()); : map->GetBitmap()->GetStart(p_x - rect.GetLeft(), p_y - rect.GetTop());
m_unk0x56 = 0; m_stateOrCellIndex = 0;
if (m_states) { if (m_states) {
for (MxS16 i = 1; i <= *m_states; i++) { for (MxS16 i = 1; i <= *m_states; i++) {
// TODO: Can we match without the cast here? // TODO: Can we match without the cast here?
if (m_states[i] == (MxS16) *start) { if (m_states[i] == (MxS16) *start) {
m_unk0x56 = i; m_stateOrCellIndex = i;
break; break;
} }
} }
} }
else { else {
if (*start != 0) { if (*start != 0) {
m_unk0x56 = 1; m_stateOrCellIndex = 1;
} }
} }
if (m_unk0x56) { if (m_stateOrCellIndex) {
return TRUE; return TRUE;
} }
} }
@@ -119,21 +119,21 @@ MxBool MxControlPresenter::FUN_10044270(MxS32 p_x, MxS32 p_y, MxPresenter* p_pre
} }
else { else {
if (ContainsPresenter(m_list, presenter)) { if (ContainsPresenter(m_list, presenter)) {
if (m_unk0x4c == 2) { if (m_style == e_grid) {
MxS32 width = presenter->GetWidth(); MxS32 width = presenter->GetWidth();
MxS32 height = presenter->GetHeight(); MxS32 height = presenter->GetHeight();
if (m_unk0x52 == 2 && m_unk0x54 == 2) { if (m_columnsOrRows == 2 && m_rowsOrColumns == 2) {
if (p_x < presenter->GetX() + width / 2) { if (p_x < presenter->GetX() + width / 2) {
m_unk0x56 = (p_y >= presenter->GetY() + height / 2) ? 3 : 1; m_stateOrCellIndex = (p_y >= presenter->GetY() + height / 2) ? 3 : 1;
} }
else { else {
m_unk0x56 = (p_y >= presenter->GetY() + height / 2) ? 4 : 2; m_stateOrCellIndex = (p_y >= presenter->GetY() + height / 2) ? 4 : 2;
} }
} }
} }
else { else {
m_unk0x56 = -1; m_stateOrCellIndex = -1;
} }
return TRUE; return TRUE;
@@ -144,27 +144,27 @@ MxBool MxControlPresenter::FUN_10044270(MxS32 p_x, MxS32 p_y, MxPresenter* p_pre
} }
// FUNCTION: LEGO1 0x10044480 // FUNCTION: LEGO1 0x10044480
MxBool MxControlPresenter::FUN_10044480(LegoControlManagerNotificationParam* p_param, MxPresenter* p_presenter) MxBool MxControlPresenter::Notify(LegoControlManagerNotificationParam* p_param, MxPresenter* p_presenter)
{ {
if (IsEnabled()) { if (IsEnabled()) {
switch (p_param->GetNotification()) { switch (p_param->GetNotification()) {
case c_notificationButtonUp: case c_notificationButtonUp:
if (m_unk0x4c == 0 || m_unk0x4c == 2 || m_unk0x4c == 3) { if (m_style == e_none || m_style == e_grid || m_style == e_map) {
p_param->SetClickedObjectId(m_action->GetObjectId()); p_param->SetClickedObjectId(m_action->GetObjectId());
p_param->SetClickedAtom(m_action->GetAtomId().GetInternal()); p_param->SetClickedAtom(m_action->GetAtomId().GetInternal());
VTable0x6c(0); UpdateEnabledChild(0);
p_param->SetNotification(c_notificationControl); p_param->SetNotification(c_notificationControl);
p_param->SetUnknown0x28(m_unk0x4e); p_param->SetUnknown0x28(m_enabledChild);
return TRUE; return TRUE;
} }
break; break;
case c_notificationButtonDown: case c_notificationButtonDown:
if (FUN_10044270(p_param->GetX(), p_param->GetY(), p_presenter)) { if (CheckButtonDown(p_param->GetX(), p_param->GetY(), p_presenter)) {
p_param->SetClickedObjectId(m_action->GetObjectId()); p_param->SetClickedObjectId(m_action->GetObjectId());
p_param->SetClickedAtom(m_action->GetAtomId().GetInternal()); p_param->SetClickedAtom(m_action->GetAtomId().GetInternal());
VTable0x6c(m_unk0x56); UpdateEnabledChild(m_stateOrCellIndex);
p_param->SetNotification(c_notificationControl); p_param->SetNotification(c_notificationControl);
p_param->SetUnknown0x28(m_unk0x4e); p_param->SetUnknown0x28(m_enabledChild);
return TRUE; return TRUE;
} }
break; break;
@@ -175,25 +175,25 @@ MxBool MxControlPresenter::FUN_10044480(LegoControlManagerNotificationParam* p_p
} }
// FUNCTION: LEGO1 0x10044540 // FUNCTION: LEGO1 0x10044540
void MxControlPresenter::VTable0x6c(MxS16 p_unk0x4e) void MxControlPresenter::UpdateEnabledChild(MxS16 p_enabledChild)
{ {
if (p_unk0x4e == -1) { if (p_enabledChild == -1) {
if ((MxS16) ((MxDSMultiAction*) m_action)->GetActionList()->GetNumElements() - m_unk0x4e == 1) { if ((MxS16) ((MxDSMultiAction*) m_action)->GetActionList()->GetNumElements() - m_enabledChild == 1) {
m_unk0x4e = 0; m_enabledChild = 0;
} }
else { else {
m_unk0x4e++; m_enabledChild++;
} }
} }
else { else {
m_unk0x4e = p_unk0x4e; m_enabledChild = p_enabledChild;
} }
m_action->SetTimeStarted(Timer()->GetTime()); m_action->SetTimeStarted(Timer()->GetTime());
MxS16 i = 0; MxS16 i = 0;
for (MxCompositePresenterList::iterator it = m_list.begin(); it != m_list.end(); it++) { for (MxCompositePresenterList::iterator it = m_list.begin(); it != m_list.end(); it++) {
(*it)->Enable(((m_unk0x4c == 3 && m_unk0x4e == 0) || !IsEnabled()) ? FALSE : m_unk0x4e == i); (*it)->Enable(((m_style == e_map && m_enabledChild == 0) || !IsEnabled()) ? FALSE : m_enabledChild == i);
i++; i++;
} }
} }
@@ -224,20 +224,20 @@ void MxControlPresenter::ParseExtra()
char* token = strtok(output, g_parseExtraTokens); char* token = strtok(output, g_parseExtraTokens);
if (!strcmpi(token, g_strTOGGLE)) { if (!strcmpi(token, g_strTOGGLE)) {
m_unk0x4c = 1; m_style = e_toggle;
} }
else if (!strcmpi(token, g_strGRID)) { else if (!strcmpi(token, g_strGRID)) {
m_unk0x4c = 2; m_style = e_grid;
token = strtok(NULL, g_parseExtraTokens); token = strtok(NULL, g_parseExtraTokens);
assert(token); assert(token);
m_unk0x52 = atoi(token); m_columnsOrRows = atoi(token);
token = strtok(NULL, g_parseExtraTokens); token = strtok(NULL, g_parseExtraTokens);
assert(token); assert(token);
m_unk0x54 = atoi(token); m_rowsOrColumns = atoi(token);
} }
else if (!strcmpi(token, g_strMAP)) { else if (!strcmpi(token, g_strMAP)) {
m_unk0x4c = 3; m_style = e_map;
token = strtok(NULL, g_parseExtraTokens); token = strtok(NULL, g_parseExtraTokens);
if (token) { if (token) {
@@ -254,7 +254,7 @@ void MxControlPresenter::ParseExtra()
} }
} }
else { else {
m_unk0x4c = 0; m_style = e_none;
} }
} }
@@ -274,8 +274,8 @@ void MxControlPresenter::Enable(MxBool p_enable)
MxS16 i = 0; MxS16 i = 0;
for (MxCompositePresenterList::iterator it = m_list.begin(); it != m_list.end(); it++) { for (MxCompositePresenterList::iterator it = m_list.begin(); it != m_list.end(); it++) {
if (i == m_unk0x4e) { if (i == m_enabledChild) {
(*it)->Enable((m_unk0x4c != 3 || i != 0) ? p_enable : 0); (*it)->Enable((m_style != e_map || i != 0) ? p_enable : 0);
break; break;
} }
@@ -283,7 +283,7 @@ void MxControlPresenter::Enable(MxBool p_enable)
} }
if (!p_enable) { if (!p_enable) {
m_unk0x4e = 0; m_enabledChild = 0;
} }
} }
} }
@@ -294,10 +294,10 @@ MxBool MxControlPresenter::HasTickleStatePassed(TickleState p_tickleState)
MxCompositePresenterList::const_iterator it = m_list.begin(); MxCompositePresenterList::const_iterator it = m_list.begin();
#ifdef COMPAT_MODE #ifdef COMPAT_MODE
advance(it, m_unk0x4e); advance(it, m_enabledChild);
#else #else
// Uses forward iterator logic instead of bidrectional for some reason. // Uses forward iterator logic instead of bidrectional for some reason.
_Advance(it, m_unk0x4e, forward_iterator_tag()); _Advance(it, m_enabledChild, forward_iterator_tag());
#endif #endif
return (*it)->HasTickleStatePassed(p_tickleState); return (*it)->HasTickleStatePassed(p_tickleState);

View File

@@ -131,9 +131,9 @@ void LegoControlManager::FUN_100293c0(MxU32 p_objectId, const char* p_atom, MxS1
MxDSAction* action = control->GetAction(); MxDSAction* action = control->GetAction();
if (action->GetObjectId() == p_objectId && action->GetAtomId().GetInternal() == p_atom) { if (action->GetObjectId() == p_objectId && action->GetAtomId().GetInternal() == p_atom) {
((MxControlPresenter*) control)->VTable0x6c(p_unk0x4e); ((MxControlPresenter*) control)->UpdateEnabledChild(p_unk0x4e);
if (((MxControlPresenter*) control)->GetUnknown0x4e() == 0) { if (((MxControlPresenter*) control)->GetEnabledChild() == 0) {
g_unk0x100f31b0 = -1; g_unk0x100f31b0 = -1;
g_unk0x100f31b4 = NULL; g_unk0x100f31b4 = NULL;
break; break;
@@ -154,7 +154,7 @@ MxControlPresenter* LegoControlManager::FUN_100294e0(MxS32 p_x, MxS32 p_y)
if (presenter) { if (presenter) {
while (cursor.Next(control)) { while (cursor.Next(control)) {
if (((MxControlPresenter*) control)->FUN_10044270(p_x, p_y, presenter)) { if (((MxControlPresenter*) control)->CheckButtonDown(p_x, p_y, presenter)) {
return (MxControlPresenter*) control; return (MxControlPresenter*) control;
} }
} }
@@ -185,7 +185,7 @@ MxBool LegoControlManager::FUN_10029630()
MxPresenter* presenter; MxPresenter* presenter;
while (cursor.Next(presenter)) { while (cursor.Next(presenter)) {
if (((MxControlPresenter*) presenter)->FUN_10044480(&m_event, m_unk0x14)) { if (((MxControlPresenter*) presenter)->Notify(&m_event, m_unk0x14)) {
g_unk0x100f31b0 = m_event.m_clickedObjectId; g_unk0x100f31b0 = m_event.m_clickedObjectId;
g_unk0x100f31b4 = m_event.GetClickedAtom(); g_unk0x100f31b4 = m_event.GetClickedAtom();
FUN_100292e0(); FUN_100292e0();
@@ -206,7 +206,7 @@ MxBool LegoControlManager::FUN_10029750()
while (cursor.Next(presenter)) { while (cursor.Next(presenter)) {
if (presenter->GetAction() && presenter->GetAction()->GetObjectId() == g_unk0x100f31b0 && if (presenter->GetAction() && presenter->GetAction()->GetObjectId() == g_unk0x100f31b0 &&
presenter->GetAction()->GetAtomId().GetInternal() == g_unk0x100f31b4) { presenter->GetAction()->GetAtomId().GetInternal() == g_unk0x100f31b4) {
if (((MxControlPresenter*) presenter)->FUN_10044480(&m_event, m_unk0x14)) { if (((MxControlPresenter*) presenter)->Notify(&m_event, m_unk0x14)) {
FUN_100292e0(); FUN_100292e0();
} }