Merge remote-tracking branch 'upstream/master'

This commit is contained in:
olebeck 2025-07-17 16:52:20 +02:00
commit b8b8fc4d19
67 changed files with 2105 additions and 1298 deletions

View File

@ -558,6 +558,7 @@ if (ISLE_BUILD_APP)
ISLE/emscripten/config.cpp ISLE/emscripten/config.cpp
ISLE/emscripten/events.cpp ISLE/emscripten/events.cpp
ISLE/emscripten/filesystem.cpp ISLE/emscripten/filesystem.cpp
ISLE/emscripten/haptic.cpp
ISLE/emscripten/messagebox.cpp ISLE/emscripten/messagebox.cpp
ISLE/emscripten/window.cpp ISLE/emscripten/window.cpp
) )

View File

@ -58,6 +58,7 @@ CMainDialog::CMainDialog(QWidget* pParent) : QDialog(pParent)
connect(m_ui->musicCheckBox, &QCheckBox::toggled, this, &CMainDialog::OnCheckboxMusic); connect(m_ui->musicCheckBox, &QCheckBox::toggled, this, &CMainDialog::OnCheckboxMusic);
connect(m_ui->sound3DCheckBox, &QCheckBox::toggled, this, &CMainDialog::OnCheckbox3DSound); connect(m_ui->sound3DCheckBox, &QCheckBox::toggled, this, &CMainDialog::OnCheckbox3DSound);
connect(m_ui->fullscreenCheckBox, &QCheckBox::toggled, this, &CMainDialog::OnCheckboxFullscreen); connect(m_ui->fullscreenCheckBox, &QCheckBox::toggled, this, &CMainDialog::OnCheckboxFullscreen);
connect(m_ui->exclusiveFullscreenCheckbox, &QCheckBox::toggled, this, &CMainDialog::OnCheckboxExclusiveFullscreen);
connect(m_ui->rumbleCheckBox, &QCheckBox::toggled, this, &CMainDialog::OnCheckboxRumble); connect(m_ui->rumbleCheckBox, &QCheckBox::toggled, this, &CMainDialog::OnCheckboxRumble);
connect(m_ui->textureCheckBox, &QCheckBox::toggled, this, &CMainDialog::OnCheckboxTexture); connect(m_ui->textureCheckBox, &QCheckBox::toggled, this, &CMainDialog::OnCheckboxTexture);
connect(m_ui->touchComboBox, &QComboBox::currentIndexChanged, this, &CMainDialog::TouchControlsChanged); connect(m_ui->touchComboBox, &QComboBox::currentIndexChanged, this, &CMainDialog::TouchControlsChanged);
@ -80,7 +81,44 @@ CMainDialog::CMainDialog(QWidget* pParent) : QDialog(pParent)
connect(m_ui->maxActorsSlider, &QSlider::valueChanged, this, &CMainDialog::MaxActorsChanged); connect(m_ui->maxActorsSlider, &QSlider::valueChanged, this, &CMainDialog::MaxActorsChanged);
connect(m_ui->maxActorsSlider, &QSlider::sliderMoved, this, &CMainDialog::MaxActorsChanged); connect(m_ui->maxActorsSlider, &QSlider::sliderMoved, this, &CMainDialog::MaxActorsChanged);
connect(m_ui->aspectRatioComboBox, &QComboBox::currentIndexChanged, this, &CMainDialog::AspectRatioChanged);
connect(m_ui->xResSpinBox, &QSpinBox::valueChanged, this, &CMainDialog::XResChanged);
connect(m_ui->yResSpinBox, &QSpinBox::valueChanged, this, &CMainDialog::YResChanged);
connect(m_ui->framerateSpinBox, &QSpinBox::valueChanged, this, &CMainDialog::FramerateChanged);
layout()->setSizeConstraint(QLayout::SetFixedSize); layout()->setSizeConstraint(QLayout::SetFixedSize);
if (currentConfigApp->m_ram_quality_limit != 0) {
m_modified = true;
const QString ramError = QString("Insufficient RAM!");
m_ui->sound3DCheckBox->setChecked(false);
m_ui->sound3DCheckBox->setEnabled(false);
m_ui->sound3DCheckBox->setToolTip(ramError);
m_ui->modelQualityHighRadioButton->setEnabled(false);
m_ui->modelQualityHighRadioButton->setToolTip(ramError);
m_ui->modelQualityLowRadioButton->setEnabled(true);
if (currentConfigApp->m_ram_quality_limit == 2) {
m_ui->modelQualityLowRadioButton->setChecked(true);
m_ui->modelQualityMediumRadioButton->setEnabled(false);
m_ui->modelQualityMediumRadioButton->setToolTip(ramError);
m_ui->maxLoDSlider->setMaximum(30);
m_ui->maxActorsSlider->setMaximum(15);
}
else {
m_ui->modelQualityMediumRadioButton->setChecked(true);
m_ui->modelQualityMediumRadioButton->setEnabled(true);
m_ui->maxLoDSlider->setMaximum(40);
m_ui->maxActorsSlider->setMaximum(30);
}
}
else {
m_ui->sound3DCheckBox->setEnabled(true);
m_ui->modelQualityLowRadioButton->setEnabled(true);
m_ui->modelQualityMediumRadioButton->setEnabled(true);
m_ui->modelQualityHighRadioButton->setEnabled(true);
m_ui->maxLoDSlider->setMaximum(60);
m_ui->maxActorsSlider->setMaximum(40);
}
} }
CMainDialog::~CMainDialog() CMainDialog::~CMainDialog()
@ -98,7 +136,7 @@ bool CMainDialog::OnInitDialog()
int device_i = 0; int device_i = 0;
int selected = 0; int selected = 0;
char device_name[256]; char device_name[256];
const list<MxDriver>& driver_list = enumerator->GetDriverList(); const list<MxDriver>& driver_list = enumerator->m_ddInfo;
for (list<MxDriver>::const_iterator it_driver = driver_list.begin(); it_driver != driver_list.end(); it_driver++) { for (list<MxDriver>::const_iterator it_driver = driver_list.begin(); it_driver != driver_list.end(); it_driver++) {
const MxDriver& driver = *it_driver; const MxDriver& driver = *it_driver;
for (list<Direct3DDeviceInfo>::const_iterator it_device = driver.m_devices.begin(); for (list<Direct3DDeviceInfo>::const_iterator it_device = driver.m_devices.begin();
@ -125,6 +163,7 @@ bool CMainDialog::OnInitDialog()
m_ui->LoDNum->setNum((int) currentConfigApp->m_max_lod * 10); m_ui->LoDNum->setNum((int) currentConfigApp->m_max_lod * 10);
m_ui->maxActorsSlider->setValue(currentConfigApp->m_max_actors); m_ui->maxActorsSlider->setValue(currentConfigApp->m_max_actors);
m_ui->maxActorsNum->setNum(currentConfigApp->m_max_actors); m_ui->maxActorsNum->setNum(currentConfigApp->m_max_actors);
UpdateInterface(); UpdateInterface();
return true; return true;
} }
@ -221,6 +260,8 @@ void CMainDialog::UpdateInterface()
} }
m_ui->musicCheckBox->setChecked(currentConfigApp->m_music); m_ui->musicCheckBox->setChecked(currentConfigApp->m_music);
m_ui->fullscreenCheckBox->setChecked(currentConfigApp->m_full_screen); m_ui->fullscreenCheckBox->setChecked(currentConfigApp->m_full_screen);
m_ui->exclusiveFullscreenCheckbox->setEnabled(currentConfigApp->m_full_screen);
m_ui->exclusiveFullscreenCheckbox->setChecked(currentConfigApp->m_exclusive_full_screen);
m_ui->rumbleCheckBox->setChecked(currentConfigApp->m_haptic); m_ui->rumbleCheckBox->setChecked(currentConfigApp->m_haptic);
m_ui->touchComboBox->setCurrentIndex(currentConfigApp->m_touch_scheme); m_ui->touchComboBox->setCurrentIndex(currentConfigApp->m_touch_scheme);
m_ui->transitionTypeComboBox->setCurrentIndex(currentConfigApp->m_transition_type); m_ui->transitionTypeComboBox->setCurrentIndex(currentConfigApp->m_transition_type);
@ -231,6 +272,11 @@ void CMainDialog::UpdateInterface()
m_ui->texturePath->setEnabled(currentConfigApp->m_texture_load); m_ui->texturePath->setEnabled(currentConfigApp->m_texture_load);
m_ui->texturePathOpen->setEnabled(currentConfigApp->m_texture_load); m_ui->texturePathOpen->setEnabled(currentConfigApp->m_texture_load);
m_ui->aspectRatioComboBox->setCurrentIndex(currentConfigApp->m_aspect_ratio);
m_ui->xResSpinBox->setValue(currentConfigApp->m_x_res);
m_ui->yResSpinBox->setValue(currentConfigApp->m_y_res);
m_ui->framerateSpinBox->setValue(static_cast<int>(std::round(1000.0f / currentConfigApp->m_frame_delta)));
} }
// FUNCTION: CONFIG 0x004045e0 // FUNCTION: CONFIG 0x004045e0
@ -301,6 +347,14 @@ void CMainDialog::OnCheckboxMusic(bool checked)
void CMainDialog::OnCheckboxFullscreen(bool checked) void CMainDialog::OnCheckboxFullscreen(bool checked)
{ {
currentConfigApp->m_full_screen = checked; currentConfigApp->m_full_screen = checked;
m_ui->exclusiveFullscreenCheckbox->setEnabled(checked);
m_modified = true;
UpdateInterface();
}
void CMainDialog::OnCheckboxExclusiveFullscreen(bool checked)
{
currentConfigApp->m_exclusive_full_screen = checked;
m_modified = true; m_modified = true;
UpdateInterface(); UpdateInterface();
} }
@ -444,3 +498,59 @@ void CMainDialog::TexturePathEdited()
} }
UpdateInterface(); UpdateInterface();
} }
void CMainDialog::AspectRatioChanged(int index)
{
currentConfigApp->m_aspect_ratio = index;
EnsureAspectRatio();
m_modified = true;
UpdateInterface();
}
void CMainDialog::XResChanged(int i)
{
currentConfigApp->m_x_res = i;
m_modified = true;
UpdateInterface();
}
void CMainDialog::YResChanged(int i)
{
currentConfigApp->m_y_res = i;
EnsureAspectRatio();
m_modified = true;
UpdateInterface();
}
void CMainDialog::EnsureAspectRatio()
{
if (currentConfigApp->m_aspect_ratio != 3) {
m_ui->xResSpinBox->setReadOnly(true);
switch (currentConfigApp->m_aspect_ratio) {
case 0: {
float standardAspect = 4.0f / 3.0f;
currentConfigApp->m_x_res = static_cast<int>(std::round((currentConfigApp->m_y_res) * standardAspect));
break;
}
case 1: {
float wideAspect = 16.0f / 9.0f;
currentConfigApp->m_x_res = static_cast<int>(std::round((currentConfigApp->m_y_res) * wideAspect));
break;
}
case 2: {
currentConfigApp->m_x_res = currentConfigApp->m_y_res;
break;
}
}
}
else {
m_ui->xResSpinBox->setReadOnly(false);
}
}
void CMainDialog::FramerateChanged(int i)
{
currentConfigApp->m_frame_delta = (1000.0f / static_cast<float>(i));
m_modified = true;
UpdateInterface();
}

View File

@ -42,6 +42,7 @@ private slots:
void OnRadiobuttonTextureHighQuality(bool checked); void OnRadiobuttonTextureHighQuality(bool checked);
void OnCheckboxMusic(bool checked); void OnCheckboxMusic(bool checked);
void OnCheckboxFullscreen(bool checked); void OnCheckboxFullscreen(bool checked);
void OnCheckboxExclusiveFullscreen(bool checked);
void OnCheckboxRumble(bool checked); void OnCheckboxRumble(bool checked);
void OnCheckboxTexture(bool checked); void OnCheckboxTexture(bool checked);
void TouchControlsChanged(int index); void TouchControlsChanged(int index);
@ -57,6 +58,11 @@ private slots:
void MaxActorsChanged(int value); void MaxActorsChanged(int value);
void SelectTexturePathDialog(); void SelectTexturePathDialog();
void TexturePathEdited(); void TexturePathEdited();
void XResChanged(int i);
void YResChanged(int i);
void AspectRatioChanged(int index);
void EnsureAspectRatio();
void FramerateChanged(int i);
}; };
// SYNTHETIC: CONFIG 0x00403de0 // SYNTHETIC: CONFIG 0x00403de0

View File

@ -67,9 +67,14 @@ bool CConfigApp::InitInstance()
return FALSE; return FALSE;
} }
SDL_DestroyWindow(window); SDL_DestroyWindow(window);
m_aspect_ratio = 0;
m_x_res = 640;
m_y_res = 480;
m_frame_delta = 10.0f;
m_driver = NULL; m_driver = NULL;
m_device = NULL; m_device = NULL;
m_full_screen = TRUE; m_full_screen = TRUE;
m_exclusive_full_screen = FALSE;
m_transition_type = 3; // 3: Mosaic m_transition_type = 3; // 3: Mosaic
m_wide_view_angle = TRUE; m_wide_view_angle = TRUE;
m_use_joystick = TRUE; m_use_joystick = TRUE;
@ -84,6 +89,7 @@ bool CConfigApp::InitInstance()
m_texture_path = "/textures/"; m_texture_path = "/textures/";
int totalRamMiB = SDL_GetSystemRAM(); int totalRamMiB = SDL_GetSystemRAM();
if (totalRamMiB < 12) { if (totalRamMiB < 12) {
m_ram_quality_limit = 2;
m_3d_sound = FALSE; m_3d_sound = FALSE;
m_model_quality = 0; m_model_quality = 0;
m_texture_quality = 1; m_texture_quality = 1;
@ -91,6 +97,7 @@ bool CConfigApp::InitInstance()
m_max_actors = 5; m_max_actors = 5;
} }
else if (totalRamMiB < 20) { else if (totalRamMiB < 20) {
m_ram_quality_limit = 1;
m_3d_sound = FALSE; m_3d_sound = FALSE;
m_model_quality = 1; m_model_quality = 1;
m_texture_quality = 1; m_texture_quality = 1;
@ -98,6 +105,7 @@ bool CConfigApp::InitInstance()
m_max_actors = 10; m_max_actors = 10;
} }
else { else {
m_ram_quality_limit = 0;
m_model_quality = 2; m_model_quality = 2;
m_3d_sound = TRUE; m_3d_sound = TRUE;
m_texture_quality = 1; m_texture_quality = 1;
@ -126,7 +134,7 @@ D3DCOLORMODEL CConfigApp::GetHardwareDeviceColorModel() const
// FUNCTION: CONFIG 0x00403410 // FUNCTION: CONFIG 0x00403410
bool CConfigApp::IsPrimaryDriver() const bool CConfigApp::IsPrimaryDriver() const
{ {
return m_driver == &m_device_enumerator->GetDriverList().front(); return m_driver == &m_device_enumerator->m_ddInfo.front();
} }
// FUNCTION: CONFIG 0x00403430 // FUNCTION: CONFIG 0x00403430
@ -158,6 +166,7 @@ bool CConfigApp::ReadRegisterSettings()
m_display_bit_depth = iniparser_getint(dict, "isle:Display Bit Depth", -1); m_display_bit_depth = iniparser_getint(dict, "isle:Display Bit Depth", -1);
m_flip_surfaces = iniparser_getboolean(dict, "isle:Flip Surfaces", m_flip_surfaces); m_flip_surfaces = iniparser_getboolean(dict, "isle:Flip Surfaces", m_flip_surfaces);
m_full_screen = iniparser_getboolean(dict, "isle:Full Screen", m_full_screen); m_full_screen = iniparser_getboolean(dict, "isle:Full Screen", m_full_screen);
m_exclusive_full_screen = iniparser_getboolean(dict, "isle:Exclusive Full Screen", m_exclusive_full_screen);
m_transition_type = iniparser_getint(dict, "isle:Transition Type", m_transition_type); m_transition_type = iniparser_getint(dict, "isle:Transition Type", m_transition_type);
m_touch_scheme = iniparser_getint(dict, "isle:Touch Scheme", m_touch_scheme); m_touch_scheme = iniparser_getint(dict, "isle:Touch Scheme", m_touch_scheme);
m_3d_video_ram = iniparser_getboolean(dict, "isle:Back Buffers in Video RAM", m_3d_video_ram); m_3d_video_ram = iniparser_getboolean(dict, "isle:Back Buffers in Video RAM", m_3d_video_ram);
@ -174,6 +183,10 @@ bool CConfigApp::ReadRegisterSettings()
m_max_actors = iniparser_getint(dict, "isle:Max Allowed Extras", m_max_actors); m_max_actors = iniparser_getint(dict, "isle:Max Allowed Extras", m_max_actors);
m_texture_load = iniparser_getboolean(dict, "extensions:texture loader", m_texture_load); m_texture_load = iniparser_getboolean(dict, "extensions:texture loader", m_texture_load);
m_texture_path = iniparser_getstring(dict, "texture loader:texture path", m_texture_path.c_str()); m_texture_path = iniparser_getstring(dict, "texture loader:texture path", m_texture_path.c_str());
m_aspect_ratio = iniparser_getint(dict, "isle:Aspect Ratio", m_aspect_ratio);
m_x_res = iniparser_getint(dict, "isle:Horizontal Resolution", m_x_res);
m_y_res = iniparser_getint(dict, "isle:Vertical Resolution", m_y_res);
m_frame_delta = iniparser_getdouble(dict, "isle:Frame Delta", m_frame_delta);
iniparser_freedict(dict); iniparser_freedict(dict);
return true; return true;
} }
@ -230,7 +243,7 @@ bool CConfigApp::ValidateSettings()
is_modified = TRUE; is_modified = TRUE;
} }
if (m_max_lod < 0.0f || m_max_lod > 5.0f) { if (m_max_lod < 0.0f || m_max_lod > 6.0f) {
m_max_lod = 3.5f; m_max_lod = 3.5f;
is_modified = TRUE; is_modified = TRUE;
} }
@ -326,6 +339,7 @@ void CConfigApp::WriteRegisterSettings() const
SetIniInt(dict, "isle:Display Bit Depth", m_display_bit_depth); SetIniInt(dict, "isle:Display Bit Depth", m_display_bit_depth);
SetIniBool(dict, "isle:Flip Surfaces", m_flip_surfaces); SetIniBool(dict, "isle:Flip Surfaces", m_flip_surfaces);
SetIniBool(dict, "isle:Full Screen", m_full_screen); SetIniBool(dict, "isle:Full Screen", m_full_screen);
SetIniBool(dict, "isle:Exclusive Full Screen", m_exclusive_full_screen);
SetIniBool(dict, "isle:Wide View Angle", m_wide_view_angle); SetIniBool(dict, "isle:Wide View Angle", m_wide_view_angle);
SetIniInt(dict, "isle:Transition Type", m_transition_type); SetIniInt(dict, "isle:Transition Type", m_transition_type);
@ -350,6 +364,11 @@ void CConfigApp::WriteRegisterSettings() const
iniparser_set(dict, "isle:Max LOD", std::to_string(m_max_lod).c_str()); iniparser_set(dict, "isle:Max LOD", std::to_string(m_max_lod).c_str());
SetIniInt(dict, "isle:Max Allowed Extras", m_max_actors); SetIniInt(dict, "isle:Max Allowed Extras", m_max_actors);
SetIniInt(dict, "isle:Aspect Ratio", m_aspect_ratio);
SetIniInt(dict, "isle:Horizontal Resolution", m_x_res);
SetIniInt(dict, "isle:Vertical Resolution", m_y_res);
iniparser_set(dict, "isle:Frame Delta", std::to_string(m_frame_delta).c_str());
#undef SetIniBool #undef SetIniBool
#undef SetIniInt #undef SetIniInt

View File

@ -59,12 +59,17 @@ class CConfigApp {
// DECLARE_MESSAGE_MAP() // DECLARE_MESSAGE_MAP()
public: public:
int m_aspect_ratio;
int m_x_res;
int m_y_res;
float m_frame_delta;
LegoDeviceEnumerate* m_device_enumerator; LegoDeviceEnumerate* m_device_enumerator;
MxDriver* m_driver; MxDriver* m_driver;
Direct3DDeviceInfo* m_device; Direct3DDeviceInfo* m_device;
int m_display_bit_depth; int m_display_bit_depth;
bool m_flip_surfaces; bool m_flip_surfaces;
bool m_full_screen; bool m_full_screen;
bool m_exclusive_full_screen;
int m_transition_type; int m_transition_type;
bool m_3d_video_ram; bool m_3d_video_ram;
bool m_wide_view_angle; bool m_wide_view_angle;
@ -85,6 +90,7 @@ class CConfigApp {
float m_max_lod; float m_max_lod;
int m_max_actors; int m_max_actors;
int m_touch_scheme; int m_touch_scheme;
int m_ram_quality_limit;
}; };
extern CConfigApp g_theApp; extern CConfigApp g_theApp;

File diff suppressed because it is too large Load Diff

View File

@ -0,0 +1,55 @@
#include "haptic.h"
#include "compat.h"
#include "lego/sources/misc/legoutil.h"
#include "legoinputmanager.h"
#include "misc.h"
#include <emscripten/html5.h>
void Emscripten_HandleRumbleEvent(float p_lowFrequencyRumble, float p_highFrequencyRumble, MxU32 p_milliseconds)
{
std::visit(
overloaded{
[](LegoInputManager::SDL_KeyboardID_v p_id) {},
[](LegoInputManager::SDL_MouseID_v p_id) {},
[p_lowFrequencyRumble, p_highFrequencyRumble, p_milliseconds](LegoInputManager::SDL_JoystickID_v p_id) {
const char* name = SDL_GetJoystickNameForID((SDL_JoystickID) p_id);
if (name) {
MAIN_THREAD_EM_ASM(
{
const name = UTF8ToString($0);
const gamepads = navigator.getGamepads();
for (const gamepad of gamepads) {
if (gamepad && gamepad.connected && gamepad.id == name && gamepad.vibrationActuator) {
gamepad.vibrationActuator.playEffect("dual-rumble", {
startDelay : 0,
weakMagnitude : $1,
strongMagnitude : $2,
duration : $3,
});
break;
}
}
},
name,
SDL_clamp(p_lowFrequencyRumble, 0, 1),
SDL_clamp(p_highFrequencyRumble, 0, 1),
p_milliseconds
);
}
},
[p_milliseconds](LegoInputManager::SDL_TouchID_v p_id) {
MAIN_THREAD_EM_ASM(
{
if (navigator.vibrate) {
navigator.vibrate($0);
}
},
p_milliseconds
);
}
},
InputManager()->GetLastInputMethod()
);
}

8
ISLE/emscripten/haptic.h Normal file
View File

@ -0,0 +1,8 @@
#ifndef EMSCRIPTEN_HAPTIC_H
#define EMSCRIPTEN_HAPTIC_H
#include "mxtypes.h"
void Emscripten_HandleRumbleEvent(float p_lowFrequencyRumble, float p_highFrequencyRumble, MxU32 p_milliseconds);
#endif // EMSCRIPTEN_HAPTIC_H

View File

@ -54,6 +54,7 @@
#include "emscripten/config.h" #include "emscripten/config.h"
#include "emscripten/events.h" #include "emscripten/events.h"
#include "emscripten/filesystem.h" #include "emscripten/filesystem.h"
#include "emscripten/haptic.h"
#include "emscripten/messagebox.h" #include "emscripten/messagebox.h"
#include "emscripten/window.h" #include "emscripten/window.h"
#endif #endif
@ -145,8 +146,6 @@ IsleApp::IsleApp()
m_drawCursor = FALSE; m_drawCursor = FALSE;
m_use3dSound = TRUE; m_use3dSound = TRUE;
m_useMusic = TRUE; m_useMusic = TRUE;
m_useJoystick = TRUE;
m_joystickIndex = 0;
m_wideViewAngle = TRUE; m_wideViewAngle = TRUE;
m_islandQuality = 2; m_islandQuality = 2;
m_islandTexture = 1; m_islandTexture = 1;
@ -184,6 +183,10 @@ IsleApp::IsleApp()
m_cursorSensitivity = 4; m_cursorSensitivity = 4;
m_touchScheme = LegoInputManager::e_gamepad; m_touchScheme = LegoInputManager::e_gamepad;
m_haptic = TRUE; m_haptic = TRUE;
m_xRes = 640;
m_yRes = 480;
m_frameRate = 100.0f;
m_exclusiveFullScreen = FALSE;
} }
// FUNCTION: ISLE 0x4011a0 // FUNCTION: ISLE 0x4011a0
@ -321,7 +324,7 @@ SDL_AppResult SDL_AppInit(void** appstate, int argc, char** argv)
SDL_SetHint(SDL_HINT_MOUSE_TOUCH_EVENTS, "0"); SDL_SetHint(SDL_HINT_MOUSE_TOUCH_EVENTS, "0");
SDL_SetHint(SDL_HINT_TOUCH_MOUSE_EVENTS, "0"); SDL_SetHint(SDL_HINT_TOUCH_MOUSE_EVENTS, "0");
if (!SDL_Init(SDL_INIT_VIDEO | SDL_INIT_AUDIO | SDL_INIT_GAMEPAD)) { if (!SDL_Init(SDL_INIT_VIDEO | SDL_INIT_AUDIO | SDL_INIT_GAMEPAD | SDL_INIT_HAPTIC)) {
char buffer[256]; char buffer[256];
SDL_snprintf( SDL_snprintf(
buffer, buffer,
@ -340,7 +343,8 @@ SDL_AppResult SDL_AppInit(void** appstate, int argc, char** argv)
// Create global app instance // Create global app instance
g_isle = new IsleApp(); g_isle = new IsleApp();
if (g_isle->ParseArguments(argc, argv) != SUCCESS) { switch (g_isle->ParseArguments(argc, argv)) {
case SDL_APP_FAILURE:
Any_ShowSimpleMessageBox( Any_ShowSimpleMessageBox(
SDL_MESSAGEBOX_ERROR, SDL_MESSAGEBOX_ERROR,
"LEGO® Island Error", "LEGO® Island Error",
@ -348,6 +352,10 @@ SDL_AppResult SDL_AppInit(void** appstate, int argc, char** argv)
window window
); );
return SDL_APP_FAILURE; return SDL_APP_FAILURE;
case SDL_APP_SUCCESS:
return SDL_APP_SUCCESS;
case SDL_APP_CONTINUE:
break;
} }
// Create window // Create window
@ -364,6 +372,23 @@ SDL_AppResult SDL_AppInit(void** appstate, int argc, char** argv)
// Get reference to window // Get reference to window
*appstate = g_isle->GetWindowHandle(); *appstate = g_isle->GetWindowHandle();
// Currently, SDL doesn't send SDL_EVENT_MOUSE_ADDED at startup (unlike for gamepads)
// This will probably be fixed in the future: https://github.com/libsdl-org/SDL/issues/12815
{
int count;
SDL_MouseID* mice = SDL_GetMice(&count);
if (mice) {
for (int i = 0; i < count; i++) {
if (InputManager()) {
InputManager()->AddMouse(mice[i]);
}
}
SDL_free(mice);
}
}
#ifdef __EMSCRIPTEN__ #ifdef __EMSCRIPTEN__
SDL_AddEventWatch( SDL_AddEventWatch(
[](void* userdata, SDL_Event* event) -> bool { [](void* userdata, SDL_Event* event) -> bool {
@ -447,6 +472,10 @@ SDL_AppResult SDL_AppEvent(void* appstate, SDL_Event* event)
return SDL_APP_CONTINUE; return SDL_APP_CONTINUE;
} }
if (InputManager()) {
InputManager()->UpdateLastInputMethod(event);
}
switch (event->type) { switch (event->type) {
case SDL_EVENT_WINDOW_PIXEL_SIZE_CHANGED: case SDL_EVENT_WINDOW_PIXEL_SIZE_CHANGED:
case SDL_EVENT_MOUSE_MOTION: case SDL_EVENT_MOUSE_MOTION:
@ -506,13 +535,36 @@ SDL_AppResult SDL_AppEvent(void* appstate, SDL_Event* event)
} }
break; break;
} }
case SDL_EVENT_GAMEPAD_ADDED: case SDL_EVENT_KEYBOARD_ADDED:
case SDL_EVENT_GAMEPAD_REMOVED: {
if (InputManager()) { if (InputManager()) {
InputManager()->GetJoystick(); InputManager()->AddKeyboard(event->kdevice.which);
}
break;
case SDL_EVENT_KEYBOARD_REMOVED:
if (InputManager()) {
InputManager()->RemoveKeyboard(event->kdevice.which);
}
break;
case SDL_EVENT_MOUSE_ADDED:
if (InputManager()) {
InputManager()->AddMouse(event->mdevice.which);
}
break;
case SDL_EVENT_MOUSE_REMOVED:
if (InputManager()) {
InputManager()->RemoveMouse(event->mdevice.which);
}
break;
case SDL_EVENT_GAMEPAD_ADDED:
if (InputManager()) {
InputManager()->AddJoystick(event->jdevice.which);
}
break;
case SDL_EVENT_GAMEPAD_REMOVED:
if (InputManager()) {
InputManager()->RemoveJoystick(event->jdevice.which);
} }
break; break;
}
case SDL_EVENT_GAMEPAD_BUTTON_DOWN: { case SDL_EVENT_GAMEPAD_BUTTON_DOWN: {
switch (event->gbutton.button) { switch (event->gbutton.button) {
case SDL_GAMEPAD_BUTTON_DPAD_UP: case SDL_GAMEPAD_BUTTON_DPAD_UP:
@ -781,8 +833,11 @@ SDL_AppResult SDL_AppEvent(void* appstate, SDL_Event* event)
} }
} }
else if (event->user.type == g_legoSdlEvents.m_hitActor && g_isle->GetHaptic()) { else if (event->user.type == g_legoSdlEvents.m_hitActor && g_isle->GetHaptic()) {
if (InputManager()) { if (!InputManager()->HandleRumbleEvent(0.5f, 0.5f, 0.5f, 700)) {
InputManager()->HandleRumbleEvent(); // Platform-specific handling
#ifdef __EMSCRIPTEN__
Emscripten_HandleRumbleEvent(0.5f, 0.5f, 700);
#endif
} }
} }
@ -861,6 +916,15 @@ MxResult IsleApp::SetupWindow()
#endif #endif
window = SDL_CreateWindowWithProperties(props); window = SDL_CreateWindowWithProperties(props);
if (m_exclusiveFullScreen && m_fullScreen) {
SDL_DisplayMode closestMode;
SDL_DisplayID displayID = SDL_GetDisplayForWindow(window);
if (SDL_GetClosestFullscreenDisplayMode(displayID, m_xRes, m_yRes, m_frameRate, true, &closestMode)) {
SDL_SetWindowFullscreenMode(window, &closestMode);
}
}
#ifdef MINIWIN #ifdef MINIWIN
m_windowHandle = reinterpret_cast<HWND>(window); m_windowHandle = reinterpret_cast<HWND>(window);
#else #else
@ -927,10 +991,6 @@ MxResult IsleApp::SetupWindow()
MxTransitionManager::configureMxTransitionManager(m_transitionType); MxTransitionManager::configureMxTransitionManager(m_transitionType);
RealtimeView::SetUserMaxLOD(m_maxLod); RealtimeView::SetUserMaxLOD(m_maxLod);
if (LegoOmni::GetInstance()) { if (LegoOmni::GetInstance()) {
if (LegoOmni::GetInstance()->GetInputManager()) {
LegoOmni::GetInstance()->GetInputManager()->SetUseJoystick(m_useJoystick);
LegoOmni::GetInstance()->GetInputManager()->SetJoystickIndex(m_joystickIndex);
}
if (LegoOmni::GetInstance()->GetVideoManager()) { if (LegoOmni::GetInstance()->GetVideoManager()) {
LegoOmni::GetInstance()->GetVideoManager()->SetCursorBitmap(m_cursorCurrentBitmap); LegoOmni::GetInstance()->GetVideoManager()->SetCursorBitmap(m_cursorCurrentBitmap);
} }
@ -1017,13 +1077,12 @@ bool IsleApp::LoadConfig()
iniparser_set(dict, "isle:Flip Surfaces", m_flipSurfaces ? "true" : "false"); iniparser_set(dict, "isle:Flip Surfaces", m_flipSurfaces ? "true" : "false");
iniparser_set(dict, "isle:Full Screen", m_fullScreen ? "true" : "false"); iniparser_set(dict, "isle:Full Screen", m_fullScreen ? "true" : "false");
iniparser_set(dict, "isle:Exclusive Full Screen", m_exclusiveFullScreen ? "true" : "false");
iniparser_set(dict, "isle:Wide View Angle", m_wideViewAngle ? "true" : "false"); iniparser_set(dict, "isle:Wide View Angle", m_wideViewAngle ? "true" : "false");
iniparser_set(dict, "isle:3DSound", m_use3dSound ? "true" : "false"); iniparser_set(dict, "isle:3DSound", m_use3dSound ? "true" : "false");
iniparser_set(dict, "isle:Music", m_useMusic ? "true" : "false"); iniparser_set(dict, "isle:Music", m_useMusic ? "true" : "false");
iniparser_set(dict, "isle:UseJoystick", m_useJoystick ? "true" : "false");
iniparser_set(dict, "isle:JoystickIndex", SDL_itoa(m_joystickIndex, buf, 10));
SDL_snprintf(buf, sizeof(buf), "%f", m_cursorSensitivity); SDL_snprintf(buf, sizeof(buf), "%f", m_cursorSensitivity);
iniparser_set(dict, "isle:Cursor Sensitivity", buf); iniparser_set(dict, "isle:Cursor Sensitivity", buf);
@ -1037,6 +1096,9 @@ bool IsleApp::LoadConfig()
iniparser_set(dict, "isle:Transition Type", SDL_itoa(m_transitionType, buf, 10)); iniparser_set(dict, "isle:Transition Type", SDL_itoa(m_transitionType, buf, 10));
iniparser_set(dict, "isle:Touch Scheme", SDL_itoa(m_touchScheme, buf, 10)); iniparser_set(dict, "isle:Touch Scheme", SDL_itoa(m_touchScheme, buf, 10));
iniparser_set(dict, "isle:Haptic", m_haptic ? "true" : "false"); iniparser_set(dict, "isle:Haptic", m_haptic ? "true" : "false");
iniparser_set(dict, "isle:Horizontal Resolution", SDL_itoa(m_xRes, buf, 10));
iniparser_set(dict, "isle:Vertical Resolution", SDL_itoa(m_yRes, buf, 10));
iniparser_set(dict, "isle:Frame Delta", SDL_itoa(m_frameDelta, buf, 10));
#ifdef EXTENSIONS #ifdef EXTENSIONS
iniparser_set(dict, "extensions", NULL); iniparser_set(dict, "extensions", NULL);
@ -1082,11 +1144,10 @@ bool IsleApp::LoadConfig()
m_flipSurfaces = iniparser_getboolean(dict, "isle:Flip Surfaces", m_flipSurfaces); m_flipSurfaces = iniparser_getboolean(dict, "isle:Flip Surfaces", m_flipSurfaces);
m_fullScreen = iniparser_getboolean(dict, "isle:Full Screen", m_fullScreen); m_fullScreen = iniparser_getboolean(dict, "isle:Full Screen", m_fullScreen);
m_exclusiveFullScreen = iniparser_getboolean(dict, "isle:Exclusive Full Screen", m_exclusiveFullScreen);
m_wideViewAngle = iniparser_getboolean(dict, "isle:Wide View Angle", m_wideViewAngle); m_wideViewAngle = iniparser_getboolean(dict, "isle:Wide View Angle", m_wideViewAngle);
m_use3dSound = iniparser_getboolean(dict, "isle:3DSound", m_use3dSound); m_use3dSound = iniparser_getboolean(dict, "isle:3DSound", m_use3dSound);
m_useMusic = iniparser_getboolean(dict, "isle:Music", m_useMusic); m_useMusic = iniparser_getboolean(dict, "isle:Music", m_useMusic);
m_useJoystick = iniparser_getboolean(dict, "isle:UseJoystick", m_useJoystick);
m_joystickIndex = iniparser_getint(dict, "isle:JoystickIndex", m_joystickIndex);
m_cursorSensitivity = iniparser_getdouble(dict, "isle:Cursor Sensitivity", m_cursorSensitivity); m_cursorSensitivity = iniparser_getdouble(dict, "isle:Cursor Sensitivity", m_cursorSensitivity);
MxS32 backBuffersInVRAM = iniparser_getboolean(dict, "isle:Back Buffers in Video RAM", -1); MxS32 backBuffersInVRAM = iniparser_getboolean(dict, "isle:Back Buffers in Video RAM", -1);
@ -1112,6 +1173,13 @@ bool IsleApp::LoadConfig()
(MxTransitionManager::TransitionType) iniparser_getint(dict, "isle:Transition Type", m_transitionType); (MxTransitionManager::TransitionType) iniparser_getint(dict, "isle:Transition Type", m_transitionType);
m_touchScheme = (LegoInputManager::TouchScheme) iniparser_getint(dict, "isle:Touch Scheme", m_touchScheme); m_touchScheme = (LegoInputManager::TouchScheme) iniparser_getint(dict, "isle:Touch Scheme", m_touchScheme);
m_haptic = iniparser_getboolean(dict, "isle:Haptic", m_haptic); m_haptic = iniparser_getboolean(dict, "isle:Haptic", m_haptic);
m_xRes = iniparser_getint(dict, "isle:Horizontal Resolution", m_xRes);
m_yRes = iniparser_getint(dict, "isle:Vertical Resolution", m_yRes);
if (!m_fullScreen) {
m_videoParam.GetRect() = MxRect32(0, 0, (m_xRes - 1), (m_yRes - 1));
}
m_frameRate = (1000.0f / iniparser_getdouble(dict, "isle:Frame Delta", m_frameDelta));
m_frameDelta = static_cast<int>(std::round(iniparser_getdouble(dict, "isle:Frame Delta", m_frameDelta)));
const char* deviceId = iniparser_getstring(dict, "isle:3D Device ID", NULL); const char* deviceId = iniparser_getstring(dict, "isle:3D Device ID", NULL);
if (deviceId != NULL) { if (deviceId != NULL) {
@ -1290,7 +1358,7 @@ void IsleApp::SetupCursor(Cursor p_cursor)
} }
} }
MxResult IsleApp::ParseArguments(int argc, char** argv) SDL_AppResult IsleApp::ParseArguments(int argc, char** argv)
{ {
for (int i = 1, consumed; i < argc; i += consumed) { for (int i = 1, consumed; i < argc; i += consumed) {
consumed = -1; consumed = -1;
@ -1307,13 +1375,29 @@ MxResult IsleApp::ParseArguments(int argc, char** argv)
#endif #endif
consumed = 1; consumed = 1;
} }
else if (strcmp(argv[i], "--help") == 0) {
DisplayArgumentHelp();
return SDL_APP_SUCCESS;
}
if (consumed <= 0) { if (consumed <= 0) {
SDL_Log("Invalid argument(s): %s", argv[i]); SDL_Log("Invalid argument(s): %s", argv[i]);
return FAILURE; DisplayArgumentHelp();
return SDL_APP_FAILURE;
} }
} }
return SUCCESS; return SDL_APP_CONTINUE;
}
void IsleApp::DisplayArgumentHelp()
{
SDL_Log("Usage: isle [options]");
SDL_Log("Options:");
SDL_Log(" --ini <path> Set custom path to .ini config");
#ifdef ISLE_DEBUG
SDL_Log(" --debug Launch in debug mode");
#endif
SDL_Log(" --help Show this help message");
} }
MxResult IsleApp::VerifyFilesystem() MxResult IsleApp::VerifyFilesystem()

View File

@ -61,7 +61,7 @@ class IsleApp {
void SetGameStarted(MxS32 p_gameStarted) { m_gameStarted = p_gameStarted; } void SetGameStarted(MxS32 p_gameStarted) { m_gameStarted = p_gameStarted; }
void SetDrawCursor(MxS32 p_drawCursor) { m_drawCursor = p_drawCursor; } void SetDrawCursor(MxS32 p_drawCursor) { m_drawCursor = p_drawCursor; }
MxResult ParseArguments(int argc, char** argv); SDL_AppResult ParseArguments(int argc, char** argv);
MxResult VerifyFilesystem(); MxResult VerifyFilesystem();
void DetectGameVersion(); void DetectGameVersion();
void MoveVirtualMouseViaJoystick(); void MoveVirtualMouseViaJoystick();
@ -72,7 +72,7 @@ class IsleApp {
char* m_cdPath; // 0x04 char* m_cdPath; // 0x04
char* m_deviceId; // 0x08 char* m_deviceId; // 0x08
char* m_savePath; // 0x0c char* m_savePath; // 0x0c
MxS32 m_fullScreen; // 0x10 MxBool m_fullScreen; // 0x10
MxS32 m_flipSurfaces; // 0x14 MxS32 m_flipSurfaces; // 0x14
MxS32 m_backBuffersInVram; // 0x18 MxS32 m_backBuffersInVram; // 0x18
MxS32 m_using8bit; // 0x1c MxS32 m_using8bit; // 0x1c
@ -80,8 +80,6 @@ class IsleApp {
MxS32 m_hasLightSupport; // 0x24 MxS32 m_hasLightSupport; // 0x24
MxS32 m_use3dSound; // 0x28 MxS32 m_use3dSound; // 0x28
MxS32 m_useMusic; // 0x2c MxS32 m_useMusic; // 0x2c
MxS32 m_useJoystick; // 0x30
MxS32 m_joystickIndex; // 0x34
MxS32 m_wideViewAngle; // 0x38 MxS32 m_wideViewAngle; // 0x38
MxS32 m_islandQuality; // 0x3c MxS32 m_islandQuality; // 0x3c
MxS32 m_islandTexture; // 0x40 MxS32 m_islandTexture; // 0x40
@ -101,6 +99,7 @@ class IsleApp {
const CursorBitmap* m_cursorCurrentBitmap; const CursorBitmap* m_cursorCurrentBitmap;
char* m_mediaPath; char* m_mediaPath;
MxFloat m_cursorSensitivity; MxFloat m_cursorSensitivity;
void DisplayArgumentHelp();
char* m_iniPath; char* m_iniPath;
MxFloat m_maxLod; MxFloat m_maxLod;
@ -108,6 +107,10 @@ class IsleApp {
MxTransitionManager::TransitionType m_transitionType; MxTransitionManager::TransitionType m_transitionType;
LegoInputManager::TouchScheme m_touchScheme; LegoInputManager::TouchScheme m_touchScheme;
MxBool m_haptic; MxBool m_haptic;
MxS32 m_xRes;
MxS32 m_yRes;
MxFloat m_frameRate;
MxBool m_exclusiveFullScreen;
}; };
extern IsleApp* g_isle; extern IsleApp* g_isle;

View File

@ -20,12 +20,12 @@ class MxQuaternionTransformer;
struct Act3ListElement { struct Act3ListElement {
MxU32 m_objectId; // 0x00 MxU32 m_objectId; // 0x00
undefined4 m_unk0x04; // 0x04 undefined4 m_unk0x04; // 0x04
undefined m_unk0x08; // 0x08 MxBool m_hasStarted; // 0x08
Act3ListElement() {} Act3ListElement() {}
Act3ListElement(MxU32 p_objectId, undefined4 p_unk0x04, undefined p_unk0x08) Act3ListElement(MxU32 p_objectId, undefined4 p_unk0x04, MxBool p_hasStarted)
: m_objectId(p_objectId), m_unk0x04(p_unk0x04), m_unk0x08(p_unk0x08) : m_objectId(p_objectId), m_unk0x04(p_unk0x04), m_hasStarted(p_hasStarted)
{ {
} }
@ -36,12 +36,18 @@ struct Act3ListElement {
// SIZE 0x10 // SIZE 0x10
class Act3List : private list<Act3ListElement> { class Act3List : private list<Act3ListElement> {
public: public:
enum InsertMode {
e_replaceAction = 1,
e_queueAction = 2,
e_onlyIfEmpty = 3
};
Act3List() { m_unk0x0c = 0; } Act3List() { m_unk0x0c = 0; }
void Insert(MxS32 p_objectId, MxS32 p_option); void Insert(MxS32 p_objectId, InsertMode p_option);
void FUN_10071fa0(); void DeleteActionWrapper();
void Clear(); void Clear();
void FUN_100720d0(MxU32 p_objectId); void RemoveByObjectIdOrFirst(MxU32 p_objectId);
private: private:
undefined4 m_unk0x0c; // 0x0c undefined4 m_unk0x0c; // 0x0c

View File

@ -34,6 +34,21 @@ class Act1State : public LegoState {
c_floor3 c_floor3
}; };
enum {
e_none = 0,
e_initial = 1,
e_elevator = 2,
e_pizza = 3,
e_helicopter = 4,
e_transitionToJetski = 5,
e_transitionToRacecar = 6,
e_transitionToTowtrack = 7,
e_towtrack = 8,
e_transitionToAmbulance = 9,
e_ambulance = 10,
e_jukebox = 11,
};
Act1State(); Act1State();
// FUNCTION: LEGO1 0x100338a0 // FUNCTION: LEGO1 0x100338a0
@ -58,11 +73,11 @@ class Act1State : public LegoState {
void RemoveActors(); void RemoveActors();
void PlaceActors(); void PlaceActors();
MxU32 GetUnknown18() { return m_unk0x018; } MxU32 GetState() { return m_state; }
ElevatorFloor GetElevatorFloor() { return (ElevatorFloor) m_elevFloor; } ElevatorFloor GetElevatorFloor() { return (ElevatorFloor) m_elevFloor; }
MxU8 GetUnknown21() { return m_unk0x021; } MxU8 GetUnknown21() { return m_unk0x021; }
void SetUnknown18(MxU32 p_unk0x18) { m_unk0x018 = p_unk0x18; } void SetState(MxU32 p_state) { m_state = p_state; }
void SetElevatorFloor(ElevatorFloor p_elevFloor) { m_elevFloor = p_elevFloor; } void SetElevatorFloor(ElevatorFloor p_elevFloor) { m_elevFloor = p_elevFloor; }
void SetUnknown21(MxU8 p_unk0x21) { m_unk0x021 = p_unk0x21; } void SetUnknown21(MxU8 p_unk0x21) { m_unk0x021 = p_unk0x21; }
@ -73,7 +88,7 @@ class Act1State : public LegoState {
Playlist m_cptClickDialogue; // 0x008 Playlist m_cptClickDialogue; // 0x008
IsleScript::Script m_currentCptClickDialogue; // 0x014 IsleScript::Script m_currentCptClickDialogue; // 0x014
MxU32 m_unk0x018; // 0x018 MxU32 m_state; // 0x018
MxS16 m_elevFloor; // 0x01c MxS16 m_elevFloor; // 0x01c
MxBool m_unk0x01e; // 0x01e MxBool m_unk0x01e; // 0x01e
MxBool m_unk0x01f; // 0x01f MxBool m_unk0x01f; // 0x01f

View File

@ -92,7 +92,11 @@ class LegoAnimPresenter : public MxVideoPresenter {
const char* GetActionObjectName(); const char* GetActionObjectName();
void SetCurrentWorld(LegoWorld* p_currentWorld) { m_currentWorld = p_currentWorld; } void SetCurrentWorld(LegoWorld* p_currentWorld) { m_currentWorld = p_currentWorld; }
// FUNCTION: BETA10 0x1005aad0
void SetUnknown0x0cTo1() { m_unk0x9c = 1; } void SetUnknown0x0cTo1() { m_unk0x9c = 1; }
// FUNCTION: BETA10 0x1005ab00
void SetUnknown0xa0(Matrix4* p_unk0xa0) { m_unk0xa0 = p_unk0xa0; } void SetUnknown0xa0(Matrix4* p_unk0xa0) { m_unk0xa0 = p_unk0xa0; }
LegoAnim* GetAnimation() { return m_anim; } LegoAnim* GetAnimation() { return m_anim; }

View File

@ -8,6 +8,7 @@
#include "mxpresenter.h" #include "mxpresenter.h"
#include "mxqueue.h" #include "mxqueue.h"
#include <SDL3/SDL_haptic.h>
#include <SDL3/SDL_joystick.h> #include <SDL3/SDL_joystick.h>
#include <SDL3/SDL_keyboard.h> #include <SDL3/SDL_keyboard.h>
#include <SDL3/SDL_keycode.h> #include <SDL3/SDL_keycode.h>
@ -19,6 +20,7 @@
#endif #endif
#include <map> #include <map>
#include <variant>
class LegoCameraController; class LegoCameraController;
class LegoControlManager; class LegoControlManager;
@ -92,6 +94,7 @@ class LegoInputManager : public MxPresenter {
}; };
enum TouchScheme { enum TouchScheme {
e_none = -1,
e_mouse = 0, e_mouse = 0,
e_arrowKeys, e_arrowKeys,
e_gamepad, e_gamepad,
@ -129,9 +132,8 @@ class LegoInputManager : public MxPresenter {
void SetUnknown88(MxBool p_unk0x88) { m_unk0x88 = p_unk0x88; } void SetUnknown88(MxBool p_unk0x88) { m_unk0x88 = p_unk0x88; }
void SetUnknown335(MxBool p_unk0x335) { m_unk0x335 = p_unk0x335; } void SetUnknown335(MxBool p_unk0x335) { m_unk0x335 = p_unk0x335; }
void SetUnknown336(MxBool p_unk0x336) { m_unk0x336 = p_unk0x336; } void SetUnknown336(MxBool p_unk0x336) { m_unk0x336 = p_unk0x336; }
void SetUseJoystick(MxBool p_useJoystick) { m_useJoystick = p_useJoystick; }
void SetJoystickIndex(MxS32 p_joystickIndex) { m_joystickIndex = p_joystickIndex; }
// FUNCTION: BETA10 0x1002e290
void DisableInputProcessing() void DisableInputProcessing()
{ {
m_unk0x88 = TRUE; m_unk0x88 = TRUE;
@ -152,13 +154,31 @@ class LegoInputManager : public MxPresenter {
void GetKeyboardState(); void GetKeyboardState();
MxResult GetNavigationKeyStates(MxU32& p_keyFlags); MxResult GetNavigationKeyStates(MxU32& p_keyFlags);
MxResult GetNavigationTouchStates(MxU32& p_keyFlags); MxResult GetNavigationTouchStates(MxU32& p_keyFlags);
LEGO1_EXPORT void AddKeyboard(SDL_KeyboardID p_keyboardID);
LEGO1_EXPORT void RemoveKeyboard(SDL_KeyboardID p_keyboardID);
LEGO1_EXPORT void AddMouse(SDL_MouseID p_mouseID);
LEGO1_EXPORT void RemoveMouse(SDL_MouseID p_mouseID);
LEGO1_EXPORT void AddJoystick(SDL_JoystickID p_joystickID);
LEGO1_EXPORT void RemoveJoystick(SDL_JoystickID p_joystickID);
LEGO1_EXPORT MxBool HandleTouchEvent(SDL_Event* p_event, TouchScheme p_touchScheme); LEGO1_EXPORT MxBool HandleTouchEvent(SDL_Event* p_event, TouchScheme p_touchScheme);
LEGO1_EXPORT MxBool HandleRumbleEvent(); LEGO1_EXPORT MxBool
HandleRumbleEvent(float p_strength, float p_lowFrequencyRumble, float p_highFrequencyRumble, MxU32 p_milliseconds);
LEGO1_EXPORT void UpdateLastInputMethod(SDL_Event* p_event);
const auto& GetLastInputMethod() { return m_lastInputMethod; }
// clang-format off
enum class SDL_KeyboardID_v : SDL_KeyboardID {};
enum class SDL_MouseID_v : SDL_MouseID {};
enum class SDL_JoystickID_v : SDL_JoystickID {};
enum class SDL_TouchID_v : SDL_TouchID {};
// clang-format on
// SYNTHETIC: LEGO1 0x1005b8d0 // SYNTHETIC: LEGO1 0x1005b8d0
// LegoInputManager::`scalar deleting destructor' // LegoInputManager::`scalar deleting destructor'
private: private:
void InitializeHaptics();
MxCriticalSection m_criticalSection; // 0x58 MxCriticalSection m_criticalSection; // 0x58
LegoNotifyList* m_keyboardNotifyList; // 0x5c LegoNotifyList* m_keyboardNotifyList; // 0x5c
LegoCameraController* m_camera; // 0x60 LegoCameraController* m_camera; // 0x60
@ -175,16 +195,18 @@ class LegoInputManager : public MxPresenter {
MxBool m_unk0x88; // 0x88 MxBool m_unk0x88; // 0x88
const bool* m_keyboardState; const bool* m_keyboardState;
MxBool m_unk0x195; // 0x195 MxBool m_unk0x195; // 0x195
SDL_JoystickID* m_joyids; MxBool m_unk0x335; // 0x335
SDL_Gamepad* m_joystick; MxBool m_unk0x336; // 0x336
MxS32 m_joystickIndex; // 0x19c
MxBool m_useJoystick; // 0x334
MxBool m_unk0x335; // 0x335
MxBool m_unk0x336; // 0x336
std::map<SDL_FingerID, SDL_FPoint> m_touchOrigins; TouchScheme m_touchScheme = e_none;
SDL_Point m_touchVirtualThumb = {0, 0};
SDL_FPoint m_touchVirtualThumbOrigin;
std::map<SDL_FingerID, MxU32> m_touchFlags; std::map<SDL_FingerID, MxU32> m_touchFlags;
std::map<SDL_FingerID, std::pair<MxU32, SDL_FPoint>> m_touchLastMotion; std::map<SDL_KeyboardID, std::pair<void*, void*>> m_keyboards;
std::map<SDL_MouseID, std::pair<void*, SDL_Haptic*>> m_mice;
std::map<SDL_JoystickID, std::pair<SDL_Gamepad*, SDL_Haptic*>> m_joysticks;
std::map<SDL_HapticID, SDL_Haptic*> m_otherHaptics;
std::variant<SDL_KeyboardID_v, SDL_MouseID_v, SDL_JoystickID_v, SDL_TouchID_v> m_lastInputMethod;
}; };
// TEMPLATE: LEGO1 0x10028850 // TEMPLATE: LEGO1 0x10028850

View File

@ -134,7 +134,7 @@ class LegoOmni : public MxOmni {
LegoROI* FindROI(const char* p_name); LegoROI* FindROI(const char* p_name);
void AddWorld(LegoWorld* p_world); void AddWorld(LegoWorld* p_world);
void DeleteWorld(LegoWorld* p_world); void DeleteWorld(LegoWorld* p_world);
void FUN_1005b4f0(MxBool p_disable, MxU16 p_flags); void Disable(MxBool p_disable, MxU16 p_flags);
LEGO1_EXPORT void CreateBackgroundAudio(); LEGO1_EXPORT void CreateBackgroundAudio();
LEGO1_EXPORT void RemoveWorld(const MxAtomId& p_atom, MxLong p_objectId); LEGO1_EXPORT void RemoveWorld(const MxAtomId& p_atom, MxLong p_objectId);
MxResult RegisterWorlds(); MxResult RegisterWorlds();

View File

@ -70,7 +70,9 @@ class LegoVideoManager : public MxVideoManager {
MxBool GetRender3D() { return m_render3d; } MxBool GetRender3D() { return m_render3d; }
double GetElapsedSeconds() { return m_elapsedSeconds; } double GetElapsedSeconds() { return m_elapsedSeconds; }
// FUNCTION: BETA10 0x1002e290
void SetRender3D(MxBool p_render3d) { m_render3d = p_render3d; } void SetRender3D(MxBool p_render3d) { m_render3d = p_render3d; }
void SetUnk0x554(MxBool p_unk0x554) { m_unk0x554 = p_unk0x554; } void SetUnk0x554(MxBool p_unk0x554) { m_unk0x554 = p_unk0x554; }
private: private:

View File

@ -49,7 +49,7 @@ LegoPlantManager* PlantManager();
LegoBuildingManager* BuildingManager(); LegoBuildingManager* BuildingManager();
LegoTextureContainer* TextureContainer(); LegoTextureContainer* TextureContainer();
ViewLODListManager* GetViewLODListManager(); ViewLODListManager* GetViewLODListManager();
void FUN_10015820(MxBool p_disable, MxU16 p_flags); void Disable(MxBool p_disable, MxU16 p_flags);
LegoROI* FindROI(const char* p_name); LegoROI* FindROI(const char* p_name);
void SetROIVisible(const char* p_name, MxBool p_visible); void SetROIVisible(const char* p_name, MxBool p_visible);
void SetUserActor(LegoPathActor* p_userActor); void SetUserActor(LegoPathActor* p_userActor);

View File

@ -12,6 +12,13 @@ class MxEndActionNotificationParam;
// SIZE 0x28 // SIZE 0x28
class TowTrackMissionState : public LegoState { class TowTrackMissionState : public LegoState {
public: public:
enum {
e_none = 0,
e_started = 1,
e_hookedUp = 2,
e_hookingUp = 3,
};
TowTrackMissionState(); TowTrackMissionState();
// FUNCTION: LEGO1 0x1004dde0 // FUNCTION: LEGO1 0x1004dde0
@ -126,19 +133,19 @@ class TowTrackMissionState : public LegoState {
// SYNTHETIC: LEGO1 0x1004e060 // SYNTHETIC: LEGO1 0x1004e060
// TowTrackMissionState::`scalar deleting destructor' // TowTrackMissionState::`scalar deleting destructor'
undefined4 m_unk0x08; // 0x08 MxU32 m_state; // 0x08
MxLong m_startTime; // 0x0c MxLong m_startTime; // 0x0c
MxBool m_unk0x10; // 0x10 MxBool m_takingTooLong; // 0x10
MxS16 m_peScore; // 0x12 MxS16 m_peScore; // 0x12
MxS16 m_maScore; // 0x14 MxS16 m_maScore; // 0x14
MxS16 m_paScore; // 0x16 MxS16 m_paScore; // 0x16
MxS16 m_niScore; // 0x18 MxS16 m_niScore; // 0x18
MxS16 m_laScore; // 0x1a MxS16 m_laScore; // 0x1a
MxS16 m_peHighScore; // 0x1c MxS16 m_peHighScore; // 0x1c
MxS16 m_maHighScore; // 0x1e MxS16 m_maHighScore; // 0x1e
MxS16 m_paHighScore; // 0x20 MxS16 m_paHighScore; // 0x20
MxS16 m_niHighScore; // 0x22 MxS16 m_niHighScore; // 0x22
MxS16 m_laHighScore; // 0x24 MxS16 m_laHighScore; // 0x24
}; };
// VTABLE: LEGO1 0x100d7ee0 // VTABLE: LEGO1 0x100d7ee0
@ -174,10 +181,10 @@ class TowTrack : public IslePathActor {
virtual MxLong HandleEndAction(MxEndActionNotificationParam& p_param); // vtable+0xf0 virtual MxLong HandleEndAction(MxEndActionNotificationParam& p_param); // vtable+0xf0
void CreateState(); void CreateState();
void FUN_1004dab0(); void Init();
void ActivateSceneActions(); void ActivateSceneActions();
void StopActions(); void StopActions();
void FUN_1004dbe0(); void Reset();
// SYNTHETIC: LEGO1 0x1004c950 // SYNTHETIC: LEGO1 0x1004c950
// TowTrack::`scalar deleting destructor' // TowTrack::`scalar deleting destructor'
@ -192,8 +199,8 @@ class TowTrack : public IslePathActor {
TowTrackMissionState* m_state; // 0x164 TowTrackMissionState* m_state; // 0x164
MxS16 m_unk0x168; // 0x168 MxS16 m_unk0x168; // 0x168
MxS16 m_actorId; // 0x16a MxS16 m_actorId; // 0x16a
MxS16 m_unk0x16c; // 0x16c MxS16 m_treeBlockageTriggered; // 0x16c
MxS16 m_unk0x16e; // 0x16e MxS16 m_speedComplaintTriggered; // 0x16e
IsleScript::Script m_lastAction; // 0x170 IsleScript::Script m_lastAction; // 0x170
IsleScript::Script m_lastAnimation; // 0x174 IsleScript::Script m_lastAnimation; // 0x174
MxFloat m_fuel; // 0x178 MxFloat m_fuel; // 0x178

View File

@ -367,7 +367,7 @@ MxLong Ambulance::HandlePathStruct(LegoPathStructNotificationParam& p_param)
// FUNCTION: BETA10 0x10023506 // FUNCTION: BETA10 0x10023506
MxLong Ambulance::HandleClick() MxLong Ambulance::HandleClick()
{ {
if (((Act1State*) GameState()->GetState("Act1State"))->m_unk0x018 != 10) { if (((Act1State*) GameState()->GetState("Act1State"))->m_state != Act1State::e_ambulance) {
return 1; return 1;
} }
@ -375,7 +375,7 @@ MxLong Ambulance::HandleClick()
return 1; return 1;
} }
FUN_10015820(TRUE, 0); Disable(TRUE, 0);
((Isle*) CurrentWorld())->SetDestLocation(LegoGameState::e_ambulance); ((Isle*) CurrentWorld())->SetDestLocation(LegoGameState::e_ambulance);
TransitionManager()->StartTransition(MxTransitionManager::e_mosaic, 50, FALSE, FALSE); TransitionManager()->StartTransition(MxTransitionManager::e_mosaic, 50, FALSE, FALSE);
@ -578,7 +578,7 @@ void Ambulance::Reset()
{ {
StopAction(m_lastAction); StopAction(m_lastAction);
BackgroundAudioManager()->RaiseVolume(); BackgroundAudioManager()->RaiseVolume();
((Act1State*) GameState()->GetState("Act1State"))->m_unk0x018 = 0; ((Act1State*) GameState()->GetState("Act1State"))->m_state = Act1State::e_none;
m_state->m_state = AmbulanceMissionState::e_ready; m_state->m_state = AmbulanceMissionState::e_ready;
m_atBeachTask = 0; m_atBeachTask = 0;
m_atPoliceTask = 0; m_atPoliceTask = 0;

View File

@ -54,7 +54,7 @@ MxLong Bike::HandleClick()
{ {
if (CanExit()) { if (CanExit()) {
Act1State* state = (Act1State*) GameState()->GetState("Act1State"); Act1State* state = (Act1State*) GameState()->GetState("Act1State");
FUN_10015820(TRUE, 0); Disable(TRUE, 0);
((Isle*) CurrentWorld())->SetDestLocation(LegoGameState::Area::e_bike); ((Isle*) CurrentWorld())->SetDestLocation(LegoGameState::Area::e_bike);
TransitionManager()->StartTransition(MxTransitionManager::e_mosaic, 50, FALSE, TRUE); TransitionManager()->StartTransition(MxTransitionManager::e_mosaic, 50, FALSE, TRUE);

View File

@ -53,7 +53,7 @@ MxLong InfoCenterEntity::HandleClick(LegoEventNotificationParam& p_param)
isle->SetDestLocation(LegoGameState::Area::e_infomain); isle->SetDestLocation(LegoGameState::Area::e_infomain);
Act1State* act1state = (Act1State*) GameState()->GetState("Act1State"); Act1State* act1state = (Act1State*) GameState()->GetState("Act1State");
act1state->SetUnknown18(0); act1state->SetState(Act1State::e_none);
break; break;
} }
case LegoGameState::Act::e_act2: { case LegoGameState::Act::e_act2: {
@ -83,8 +83,8 @@ MxLong GasStationEntity::HandleClick(LegoEventNotificationParam& p_param)
if (CanExit()) { if (CanExit()) {
Act1State* state = (Act1State*) GameState()->GetState("Act1State"); Act1State* state = (Act1State*) GameState()->GetState("Act1State");
if (state->GetUnknown18() != 8) { if (state->GetState() != Act1State::e_towtrack) {
state->SetUnknown18(0); state->SetState(Act1State::e_none);
if (UserActor()->GetActorId() != GameState()->GetActorId()) { if (UserActor()->GetActorId() != GameState()->GetActorId()) {
((IslePathActor*) UserActor())->Exit(); ((IslePathActor*) UserActor())->Exit();
@ -107,8 +107,8 @@ MxLong HospitalEntity::HandleClick(LegoEventNotificationParam& p_param)
if (CanExit()) { if (CanExit()) {
Act1State* act1State = (Act1State*) GameState()->GetState("Act1State"); Act1State* act1State = (Act1State*) GameState()->GetState("Act1State");
if (act1State->GetUnknown18() != 10) { if (act1State->GetState() != Act1State::e_ambulance) {
act1State->SetUnknown18(0); act1State->SetState(Act1State::e_none);
if (UserActor()->GetActorId() != GameState()->GetActorId()) { if (UserActor()->GetActorId() != GameState()->GetActorId()) {
((IslePathActor*) UserActor())->Exit(); ((IslePathActor*) UserActor())->Exit();
@ -131,8 +131,8 @@ MxLong PoliceEntity::HandleClick(LegoEventNotificationParam& p_param)
if (CanExit()) { if (CanExit()) {
Act1State* state = (Act1State*) GameState()->GetState("Act1State"); Act1State* state = (Act1State*) GameState()->GetState("Act1State");
if (state->GetUnknown18() != 10) { if (state->GetState() != Act1State::e_ambulance) {
state->SetUnknown18(0); state->SetState(Act1State::e_none);
if (UserActor()->GetActorId() != GameState()->GetActorId()) { if (UserActor()->GetActorId() != GameState()->GetActorId()) {
((IslePathActor*) UserActor())->Exit(); ((IslePathActor*) UserActor())->Exit();
@ -154,7 +154,7 @@ MxLong BeachHouseEntity::HandleClick(LegoEventNotificationParam& p_param)
{ {
if (CanExit()) { if (CanExit()) {
Act1State* state = (Act1State*) GameState()->GetState("Act1State"); Act1State* state = (Act1State*) GameState()->GetState("Act1State");
state->SetUnknown18(0); state->SetState(Act1State::e_none);
if (UserActor()->GetActorId() != GameState()->GetActorId()) { if (UserActor()->GetActorId() != GameState()->GetActorId()) {
((IslePathActor*) UserActor())->Exit(); ((IslePathActor*) UserActor())->Exit();
@ -175,7 +175,7 @@ MxLong RaceStandsEntity::HandleClick(LegoEventNotificationParam& p_param)
{ {
if (CanExit()) { if (CanExit()) {
Act1State* state = (Act1State*) GameState()->GetState("Act1State"); Act1State* state = (Act1State*) GameState()->GetState("Act1State");
state->SetUnknown18(0); state->SetState(Act1State::e_none);
if (UserActor()->GetActorId() != GameState()->GetActorId()) { if (UserActor()->GetActorId() != GameState()->GetActorId()) {
((IslePathActor*) UserActor())->Exit(); ((IslePathActor*) UserActor())->Exit();

View File

@ -45,7 +45,7 @@ MxLong BumpBouy::Notify(MxParam& p_param)
Act1State* isleState = (Act1State*) GameState()->GetState("Act1State"); Act1State* isleState = (Act1State*) GameState()->GetState("Act1State");
assert(isleState); assert(isleState);
isleState->m_unk0x018 = 5; isleState->m_state = Act1State::e_transitionToJetski;
Isle* isle = (Isle*) FindWorld(*g_isleScript, IsleScript::c__Isle); Isle* isle = (Isle*) FindWorld(*g_isleScript, IsleScript::c__Isle);
assert(isle); assert(isle);

View File

@ -95,7 +95,7 @@ MxLong DuneBuggy::HandleClick()
return 1; return 1;
} }
FUN_10015820(TRUE, 0); Disable(TRUE, 0);
((Isle*) CurrentWorld())->SetDestLocation(LegoGameState::Area::e_dunecar); ((Isle*) CurrentWorld())->SetDestLocation(LegoGameState::Area::e_dunecar);
TransitionManager()->StartTransition(MxTransitionManager::e_mosaic, 50, FALSE, TRUE); TransitionManager()->StartTransition(MxTransitionManager::e_mosaic, 50, FALSE, TRUE);

View File

@ -148,7 +148,7 @@ MxLong Helicopter::HandleClick()
IslePathActor::c_spawnBit1 | IslePathActor::c_playMusic | IslePathActor::c_spawnBit3 IslePathActor::c_spawnBit1 | IslePathActor::c_playMusic | IslePathActor::c_spawnBit3
); );
((Isle*) CurrentWorld())->SetDestLocation(LegoGameState::e_copter); ((Isle*) CurrentWorld())->SetDestLocation(LegoGameState::e_copter);
FUN_10015820(TRUE, 0); Disable(TRUE, 0);
TransitionManager()->StartTransition(MxTransitionManager::e_mosaic, 50, FALSE, TRUE); TransitionManager()->StartTransition(MxTransitionManager::e_mosaic, 50, FALSE, TRUE);
SetActorState(c_disabled); SetActorState(c_disabled);
PlayMusic(JukeboxScript::c_Jail_Music); PlayMusic(JukeboxScript::c_Jail_Music);
@ -214,7 +214,7 @@ MxLong Helicopter::HandleControl(LegoControlManagerNotificationParam& p_param)
Act1State* act1State = (Act1State*) GameState()->GetState("Act1State"); Act1State* act1State = (Act1State*) GameState()->GetState("Act1State");
assert(act1State); assert(act1State);
if (m_state->m_unk0x08 == 0) { if (m_state->m_unk0x08 == 0) {
act1State->m_unk0x018 = 4; act1State->m_state = Act1State::e_helicopter;
m_state->m_unk0x08 = 1; m_state->m_unk0x08 = 1;
m_world->RemoveActor(this); m_world->RemoveActor(this);
InvokeAction(Extra::ActionType::e_start, script, IsleScript::c_HelicopterTakeOff_Anim, NULL); InvokeAction(Extra::ActionType::e_start, script, IsleScript::c_HelicopterTakeOff_Anim, NULL);
@ -318,7 +318,7 @@ MxLong Helicopter::HandleEndAnim(LegoEndAnimNotificationParam& p_param)
if (GameState()->GetCurrentAct() == LegoGameState::e_act1) { if (GameState()->GetCurrentAct() == LegoGameState::e_act1) {
Act1State* act1State = (Act1State*) GameState()->GetState("Act1State"); Act1State* act1State = (Act1State*) GameState()->GetState("Act1State");
assert(act1State); assert(act1State);
act1State->m_unk0x018 = 4; act1State->m_state = Act1State::e_helicopter;
SpawnPlayer( SpawnPlayer(
LegoGameState::e_unk42, LegoGameState::e_unk42,
TRUE, TRUE,
@ -359,7 +359,7 @@ MxLong Helicopter::HandleEndAnim(LegoEndAnimNotificationParam& p_param)
if (GameState()->GetCurrentAct() == LegoGameState::e_act1) { if (GameState()->GetCurrentAct() == LegoGameState::e_act1) {
Act1State* act1State = (Act1State*) GameState()->GetState("Act1State"); Act1State* act1State = (Act1State*) GameState()->GetState("Act1State");
assert(act1State); assert(act1State);
act1State->m_unk0x018 = 0; act1State->m_state = Act1State::e_none;
SpawnPlayer( SpawnPlayer(
LegoGameState::e_unk41, LegoGameState::e_unk41,
TRUE, TRUE,

View File

@ -87,7 +87,7 @@ MxLong Jetski::HandleClick()
return 1; return 1;
} }
FUN_10015820(TRUE, 0); Disable(TRUE, 0);
((Isle*) CurrentWorld())->SetDestLocation(LegoGameState::Area::e_jetski); ((Isle*) CurrentWorld())->SetDestLocation(LegoGameState::Area::e_jetski);
TransitionManager()->StartTransition(MxTransitionManager::e_mosaic, 50, FALSE, TRUE); TransitionManager()->StartTransition(MxTransitionManager::e_mosaic, 50, FALSE, TRUE);
@ -164,7 +164,7 @@ void Jetski::ActivateSceneActions()
PlayMusic(JukeboxScript::c_JetskiRace_Music); PlayMusic(JukeboxScript::c_JetskiRace_Music);
Act1State* act1state = (Act1State*) GameState()->GetState("Act1State"); Act1State* act1state = (Act1State*) GameState()->GetState("Act1State");
if (!act1state->m_unk0x018) { if (!act1state->m_state) {
if (act1state->m_unk0x022) { if (act1state->m_unk0x022) {
PlayCamAnim(this, FALSE, 68, TRUE); PlayCamAnim(this, FALSE, 68, TRUE);
} }

View File

@ -90,7 +90,7 @@ MxLong Motocycle::HandleClick()
return 1; return 1;
} }
FUN_10015820(TRUE, 0); Disable(TRUE, 0);
((Isle*) CurrentWorld())->SetDestLocation(LegoGameState::Area::e_motocycle); ((Isle*) CurrentWorld())->SetDestLocation(LegoGameState::Area::e_motocycle);
TransitionManager()->StartTransition(MxTransitionManager::e_mosaic, 50, FALSE, TRUE); TransitionManager()->StartTransition(MxTransitionManager::e_mosaic, 50, FALSE, TRUE);

View File

@ -182,7 +182,7 @@ void Pizza::FUN_10038220(IsleScript::Script p_objectId)
AnimationManager()->FUN_10064740(NULL); AnimationManager()->FUN_10064740(NULL);
m_mission = m_state->GetMission(GameState()->GetActorId()); m_mission = m_state->GetMission(GameState()->GetActorId());
m_state->m_unk0x0c = 1; m_state->m_unk0x0c = 1;
m_act1state->m_unk0x018 = 3; m_act1state->m_state = Act1State::e_pizza;
m_mission->m_startTime = INT_MIN; m_mission->m_startTime = INT_MIN;
g_isleFlags &= ~Isle::c_playMusic; g_isleFlags &= ~Isle::c_playMusic;
AnimationManager()->EnableCamAnims(FALSE); AnimationManager()->EnableCamAnims(FALSE);
@ -200,7 +200,7 @@ void Pizza::FUN_100382b0()
InvokeAction(Extra::e_stop, *g_isleScript, m_speechAction, NULL); InvokeAction(Extra::e_stop, *g_isleScript, m_speechAction, NULL);
} }
m_act1state->m_unk0x018 = 0; m_act1state->m_state = Act1State::e_none;
m_state->m_unk0x0c = 0; m_state->m_unk0x0c = 0;
UserActor()->SetActorState(LegoPathActor::c_initial); UserActor()->SetActorState(LegoPathActor::c_initial);
g_isleFlags |= Isle::c_playMusic; g_isleFlags |= Isle::c_playMusic;
@ -245,7 +245,7 @@ MxLong Pizza::HandleClick()
} }
if (m_state->m_unk0x0c == 2) { if (m_state->m_unk0x0c == 2) {
m_act1state->m_unk0x018 = 3; m_act1state->m_state = Act1State::e_pizza;
if (m_skateBoard == NULL) { if (m_skateBoard == NULL) {
m_skateBoard = (SkateBoard*) m_world->Find(m_atomId, IsleScript::c_SkateBoard_Actor); m_skateBoard = (SkateBoard*) m_world->Find(m_atomId, IsleScript::c_SkateBoard_Actor);
@ -558,7 +558,7 @@ MxLong Pizza::HandleEndAction(MxEndActionNotificationParam& p_param)
break; break;
case 8: case 8:
if (m_state->GetPlayedAction() == objectId) { if (m_state->GetPlayedAction() == objectId) {
m_act1state->m_unk0x018 = 0; m_act1state->m_state = Act1State::e_none;
m_state->m_unk0x0c = 0; m_state->m_unk0x0c = 0;
GameState()->m_currentArea = LegoGameState::e_isle; GameState()->m_currentArea = LegoGameState::e_isle;
TickleManager()->UnregisterClient(this); TickleManager()->UnregisterClient(this);

View File

@ -56,7 +56,7 @@ MxResult SkateBoard::Create(MxDSAction& p_dsAction)
// FUNCTION: LEGO1 0x10010050 // FUNCTION: LEGO1 0x10010050
void SkateBoard::Exit() void SkateBoard::Exit()
{ {
if (m_act1state->m_unk0x018 == 3) { if (m_act1state->m_state == Act1State::e_pizza) {
Pizza* pizza = (Pizza*) CurrentWorld()->Find(*g_isleScript, IsleScript::c_Pizza_Actor); Pizza* pizza = (Pizza*) CurrentWorld()->Find(*g_isleScript, IsleScript::c_Pizza_Actor);
pizza->StopActions(); pizza->StopActions();
pizza->FUN_100382b0(); pizza->FUN_100382b0();
@ -75,11 +75,11 @@ MxLong SkateBoard::HandleClick()
{ {
Act1State* state = (Act1State*) GameState()->GetState("Act1State"); Act1State* state = (Act1State*) GameState()->GetState("Act1State");
if (!CanExit() && state->m_unk0x018 != 3) { if (!CanExit() && state->m_state != Act1State::e_pizza) {
return 1; return 1;
} }
FUN_10015820(TRUE, 0); Disable(TRUE, 0);
((Isle*) CurrentWorld())->SetDestLocation(LegoGameState::Area::e_skateboard); ((Isle*) CurrentWorld())->SetDestLocation(LegoGameState::Area::e_skateboard);
TransitionManager()->StartTransition(MxTransitionManager::e_mosaic, 50, FALSE, TRUE); TransitionManager()->StartTransition(MxTransitionManager::e_mosaic, 50, FALSE, TRUE);
@ -148,7 +148,7 @@ MxLong SkateBoard::HandleNotification0()
// FUNCTION: LEGO1 0x10010510 // FUNCTION: LEGO1 0x10010510
void SkateBoard::ActivateSceneActions() void SkateBoard::ActivateSceneActions()
{ {
if (m_act1state->m_unk0x018 != 3) { if (m_act1state->m_state != Act1State::e_pizza) {
PlayMusic(JukeboxScript::c_BeachBlvd_Music); PlayMusic(JukeboxScript::c_BeachBlvd_Music);
if (!m_act1state->m_unk0x022) { if (!m_act1state->m_unk0x022) {

View File

@ -34,9 +34,9 @@ TowTrack::TowTrack()
m_unk0x168 = 0; m_unk0x168 = 0;
m_actorId = -1; m_actorId = -1;
m_state = NULL; m_state = NULL;
m_unk0x16c = 0; m_treeBlockageTriggered = 0;
m_lastAction = IsleScript::c_noneIsle; m_lastAction = IsleScript::c_noneIsle;
m_unk0x16e = 0; m_speedComplaintTriggered = 0;
m_lastAnimation = IsleScript::c_noneIsle; m_lastAnimation = IsleScript::c_noneIsle;
m_maxLinearVel = 40.0; m_maxLinearVel = 40.0;
m_fuel = 1.0; m_fuel = 1.0;
@ -64,7 +64,7 @@ MxResult TowTrack::Create(MxDSAction& p_dsAction)
m_state = (TowTrackMissionState*) GameState()->GetState("TowTrackMissionState"); m_state = (TowTrackMissionState*) GameState()->GetState("TowTrackMissionState");
if (!m_state) { if (!m_state) {
m_state = new TowTrackMissionState(); m_state = new TowTrackMissionState();
m_state->m_unk0x08 = 0; m_state->m_state = TowTrackMissionState::e_none;
GameState()->RegisterState(m_state); GameState()->RegisterState(m_state);
} }
} }
@ -98,9 +98,10 @@ void TowTrack::Animate(float p_time)
sprintf(buf, "%g", m_fuel); sprintf(buf, "%g", m_fuel);
VariableTable()->SetVariable(g_varTOWFUEL, buf); VariableTable()->SetVariable(g_varTOWFUEL, buf);
if (p_time - m_state->m_startTime > 100000.0f && m_state->m_unk0x08 == 1 && !m_state->m_unk0x10) { if (p_time - m_state->m_startTime > 100000.0f && m_state->m_state == TowTrackMissionState::e_started &&
!m_state->m_takingTooLong) {
PlayAction(IsleScript::c_Avo909In_PlayWav); PlayAction(IsleScript::c_Avo909In_PlayWav);
m_state->m_unk0x10 = TRUE; m_state->m_takingTooLong = TRUE;
} }
} }
} }
@ -190,7 +191,7 @@ MxLong TowTrack::HandleEndAction(MxEndActionNotificationParam& p_param)
} }
} }
else if (objectId == IsleScript::c_wrt074sl_RunAnim || objectId == IsleScript::c_wrt075rh_RunAnim || objectId == IsleScript::c_wrt076df_RunAnim || objectId == IsleScript::c_wrt078ni_RunAnim) { else if (objectId == IsleScript::c_wrt074sl_RunAnim || objectId == IsleScript::c_wrt075rh_RunAnim || objectId == IsleScript::c_wrt076df_RunAnim || objectId == IsleScript::c_wrt078ni_RunAnim) {
m_state->m_unk0x08 = 2; m_state->m_state = TowTrackMissionState::e_hookedUp;
CurrentWorld()->PlaceActor(UserActor()); CurrentWorld()->PlaceActor(UserActor());
HandleClick(); HandleClick();
} }
@ -274,7 +275,7 @@ MxLong TowTrack::HandleEndAction(MxEndActionNotificationParam& p_param)
m_state->UpdateScore(LegoState::e_yellow, m_actorId); m_state->UpdateScore(LegoState::e_yellow, m_actorId);
} }
else if (objectId == IsleScript::c_wgs098nu_RunAnim || objectId == IsleScript::c_wgs099nu_RunAnim || objectId == IsleScript::c_wgs100nu_RunAnim || objectId == IsleScript::c_wgs101nu_RunAnim || objectId == IsleScript::c_wgs102nu_RunAnim || objectId == IsleScript::c_wgs085nu_RunAnim || objectId == IsleScript::c_wgs086nu_RunAnim || objectId == IsleScript::c_wgs087nu_RunAnim || objectId == IsleScript::c_wgs088nu_RunAnim || objectId == IsleScript::c_wgs089nu_RunAnim || objectId == IsleScript::c_wgs091nu_RunAnim || objectId == IsleScript::c_wgs092nu_RunAnim || objectId == IsleScript::c_wgs093nu_RunAnim || objectId == IsleScript::c_wgs094nu_RunAnim || objectId == IsleScript::c_wgs095nu_RunAnim) { else if (objectId == IsleScript::c_wgs098nu_RunAnim || objectId == IsleScript::c_wgs099nu_RunAnim || objectId == IsleScript::c_wgs100nu_RunAnim || objectId == IsleScript::c_wgs101nu_RunAnim || objectId == IsleScript::c_wgs102nu_RunAnim || objectId == IsleScript::c_wgs085nu_RunAnim || objectId == IsleScript::c_wgs086nu_RunAnim || objectId == IsleScript::c_wgs087nu_RunAnim || objectId == IsleScript::c_wgs088nu_RunAnim || objectId == IsleScript::c_wgs089nu_RunAnim || objectId == IsleScript::c_wgs091nu_RunAnim || objectId == IsleScript::c_wgs092nu_RunAnim || objectId == IsleScript::c_wgs093nu_RunAnim || objectId == IsleScript::c_wgs094nu_RunAnim || objectId == IsleScript::c_wgs095nu_RunAnim) {
((Act1State*) GameState()->GetState("Act1State"))->m_unk0x018 = 0; ((Act1State*) GameState()->GetState("Act1State"))->m_state = Act1State::e_none;
AnimationManager()->FUN_1005f6d0(TRUE); AnimationManager()->FUN_1005f6d0(TRUE);
g_isleFlags |= Isle::c_playMusic; g_isleFlags |= Isle::c_playMusic;
AnimationManager()->EnableCamAnims(TRUE); AnimationManager()->EnableCamAnims(TRUE);
@ -299,10 +300,10 @@ MxLong TowTrack::HandlePathStruct(LegoPathStructNotificationParam& p_param)
return 0; return 0;
} }
if (m_state->m_unk0x08 == 2 && if (m_state->m_state == TowTrackMissionState::e_hookedUp &&
((p_param.GetTrigger() == LegoPathStruct::c_camAnim && (p_param.GetData() == 9 || p_param.GetData() == 8)) || ((p_param.GetTrigger() == LegoPathStruct::c_camAnim && (p_param.GetData() == 9 || p_param.GetData() == 8)) ||
(p_param.GetTrigger() == LegoPathStruct::c_w && p_param.GetData() == 0x169))) { (p_param.GetTrigger() == LegoPathStruct::c_w && p_param.GetData() == 0x169))) {
m_state->m_unk0x08 = 0; m_state->m_state = TowTrackMissionState::e_none;
MxLong time = Timer()->GetTime() - m_state->m_startTime; MxLong time = Timer()->GetTime() - m_state->m_startTime;
Leave(); Leave();
@ -317,8 +318,8 @@ MxLong TowTrack::HandlePathStruct(LegoPathStructNotificationParam& p_param)
PlayFinalAnimation(IsleScript::c_wgs097nu_RunAnim); PlayFinalAnimation(IsleScript::c_wgs097nu_RunAnim);
} }
} }
else if (m_state->m_unk0x08 == 1 && p_param.GetTrigger() == LegoPathStruct::c_camAnim && p_param.GetData() == 0x37) { else if (m_state->m_state == TowTrackMissionState::e_started && p_param.GetTrigger() == LegoPathStruct::c_camAnim && p_param.GetData() == 0x37) {
m_state->m_unk0x08 = 3; m_state->m_state = TowTrackMissionState::e_hookingUp;
StopActions(); StopActions();
if (m_lastAction != IsleScript::c_noneIsle) { if (m_lastAction != IsleScript::c_noneIsle) {
@ -328,20 +329,20 @@ MxLong TowTrack::HandlePathStruct(LegoPathStructNotificationParam& p_param)
Leave(); Leave();
PlayFinalAnimation(IsleScript::c_wrt060bm_RunAnim); PlayFinalAnimation(IsleScript::c_wrt060bm_RunAnim);
} }
else if (p_param.GetTrigger() == LegoPathStruct::c_w && m_state->m_unk0x08 == 1) { else if (p_param.GetTrigger() == LegoPathStruct::c_w && m_state->m_state == TowTrackMissionState::e_started) {
if (p_param.GetData() == 0x15f) { if (p_param.GetData() == 0x15f) {
if (m_unk0x16c == 0) { if (m_treeBlockageTriggered == 0) {
m_unk0x16c = 1; m_treeBlockageTriggered = 1;
InvokeAction(Extra::e_start, *g_isleScript, IsleScript::c_wns050p1_RunAnim, NULL); InvokeAction(Extra::e_start, *g_isleScript, IsleScript::c_wns050p1_RunAnim, NULL);
} }
} }
else if (p_param.GetData() == 0x160) { else if (p_param.GetData() == 0x160) {
if (m_unk0x16e == 0) { if (m_speedComplaintTriggered == 0) {
m_unk0x16e = 1; m_speedComplaintTriggered = 1;
InvokeAction(Extra::e_start, *g_isleScript, IsleScript::c_wns046mg_RunAnim, NULL); InvokeAction(Extra::e_start, *g_isleScript, IsleScript::c_wns046mg_RunAnim, NULL);
} }
if (!m_state->m_unk0x10 && m_lastAction == IsleScript::c_noneIsle) { if (!m_state->m_takingTooLong && m_lastAction == IsleScript::c_noneIsle) {
if (m_actorId < LegoActor::c_pepper || m_actorId > LegoActor::c_laura) { if (m_actorId < LegoActor::c_pepper || m_actorId > LegoActor::c_laura) {
m_actorId = LegoActor::c_laura; m_actorId = LegoActor::c_laura;
} }
@ -405,15 +406,15 @@ MxLong TowTrack::HandlePathStruct(LegoPathStructNotificationParam& p_param)
// FUNCTION: LEGO1 0x1004d690 // FUNCTION: LEGO1 0x1004d690
MxLong TowTrack::HandleClick() MxLong TowTrack::HandleClick()
{ {
if (((Act1State*) GameState()->GetState("Act1State"))->m_unk0x018 != 8) { if (((Act1State*) GameState()->GetState("Act1State"))->m_state != Act1State::e_towtrack) {
return 1; return 1;
} }
if (m_state->m_unk0x08 == 3) { if (m_state->m_state == TowTrackMissionState::e_hookingUp) {
return 1; return 1;
} }
FUN_10015820(TRUE, 0); Disable(TRUE, 0);
((Isle*) CurrentWorld())->SetDestLocation(LegoGameState::e_towtrack); ((Isle*) CurrentWorld())->SetDestLocation(LegoGameState::e_towtrack);
TransitionManager()->StartTransition(MxTransitionManager::e_mosaic, 50, FALSE, FALSE); TransitionManager()->StartTransition(MxTransitionManager::e_mosaic, 50, FALSE, FALSE);
@ -428,11 +429,11 @@ MxLong TowTrack::HandleClick()
InvokeAction(Extra::e_start, *g_isleScript, IsleScript::c_TowTrackDashboard, NULL); InvokeAction(Extra::e_start, *g_isleScript, IsleScript::c_TowTrackDashboard, NULL);
ControlManager()->Register(this); ControlManager()->Register(this);
if (m_state->m_unk0x08 == 0) { if (m_state->m_state == TowTrackMissionState::e_none) {
return 1; return 1;
} }
if (m_state->m_unk0x08 == 2) { if (m_state->m_state == TowTrackMissionState::e_hookedUp) {
SpawnPlayer(LegoGameState::e_unk52, TRUE, 0); SpawnPlayer(LegoGameState::e_unk52, TRUE, 0);
FindROI("rcred")->SetVisibility(FALSE); FindROI("rcred")->SetVisibility(FALSE);
} }
@ -441,7 +442,7 @@ MxLong TowTrack::HandleClick()
m_lastAction = IsleScript::c_noneIsle; m_lastAction = IsleScript::c_noneIsle;
m_lastAnimation = IsleScript::c_noneIsle; m_lastAnimation = IsleScript::c_noneIsle;
m_state->m_startTime = Timer()->GetTime(); m_state->m_startTime = Timer()->GetTime();
m_state->m_unk0x10 = FALSE; m_state->m_takingTooLong = FALSE;
InvokeAction(Extra::e_start, *g_isleScript, IsleScript::c_wns057rd_RunAnim, NULL); InvokeAction(Extra::e_start, *g_isleScript, IsleScript::c_wns057rd_RunAnim, NULL);
InvokeAction(Extra::e_start, *g_isleScript, IsleScript::c_wns048p1_RunAnim, NULL); InvokeAction(Extra::e_start, *g_isleScript, IsleScript::c_wns048p1_RunAnim, NULL);
InvokeAction(Extra::e_start, *g_isleScript, IsleScript::c_wns049p1_RunAnim, NULL); InvokeAction(Extra::e_start, *g_isleScript, IsleScript::c_wns049p1_RunAnim, NULL);
@ -459,7 +460,7 @@ void TowTrack::Exit()
{ {
GameState()->m_currentArea = LegoGameState::e_garageExterior; GameState()->m_currentArea = LegoGameState::e_garageExterior;
StopActions(); StopActions();
FUN_1004dbe0(); Reset();
Leave(); Leave();
} }
@ -509,9 +510,9 @@ MxLong TowTrack::HandleControl(LegoControlManagerNotificationParam& p_param)
} }
// FUNCTION: LEGO1 0x1004dab0 // FUNCTION: LEGO1 0x1004dab0
void TowTrack::FUN_1004dab0() void TowTrack::Init()
{ {
m_state->m_unk0x08 = 1; m_state->m_state = TowTrackMissionState::e_started;
HandleClick(); HandleClick();
} }
@ -520,8 +521,8 @@ void TowTrack::ActivateSceneActions()
{ {
PlayMusic(JukeboxScript::c_JBMusic2); PlayMusic(JukeboxScript::c_JBMusic2);
if (m_state->m_unk0x08 != 0) { if (m_state->m_state != TowTrackMissionState::e_none) {
if (m_state->m_unk0x08 == 2) { if (m_state->m_state == TowTrackMissionState::e_hookedUp) {
PlayAction(IsleScript::c_wrt082na_PlayWav); PlayAction(IsleScript::c_wrt082na_PlayWav);
} }
else { else {
@ -545,22 +546,22 @@ void TowTrack::StopActions()
} }
// FUNCTION: LEGO1 0x1004dbe0 // FUNCTION: LEGO1 0x1004dbe0
void TowTrack::FUN_1004dbe0() void TowTrack::Reset()
{ {
if (m_lastAction != -1) { if (m_lastAction != -1) {
InvokeAction(Extra::e_stop, *g_isleScript, m_lastAction, NULL); InvokeAction(Extra::e_stop, *g_isleScript, m_lastAction, NULL);
} }
((Act1State*) GameState()->GetState("Act1State"))->m_unk0x018 = 0; ((Act1State*) GameState()->GetState("Act1State"))->m_state = Act1State::e_none;
m_state->m_unk0x08 = 0; m_state->m_state = TowTrackMissionState::e_none;
g_isleFlags |= Isle::c_playMusic; g_isleFlags |= Isle::c_playMusic;
AnimationManager()->EnableCamAnims(TRUE); AnimationManager()->EnableCamAnims(TRUE);
AnimationManager()->FUN_1005f6d0(TRUE); AnimationManager()->FUN_1005f6d0(TRUE);
m_state->m_startTime = INT_MIN; m_state->m_startTime = INT_MIN;
m_state->m_unk0x10 = FALSE; m_state->m_takingTooLong = FALSE;
m_state = NULL; m_state = NULL;
m_unk0x16c = 0; m_treeBlockageTriggered = 0;
m_unk0x16e = 0; m_speedComplaintTriggered = 0;
} }
// FUNCTION: LEGO1 0x1004dc80 // FUNCTION: LEGO1 0x1004dc80
@ -595,9 +596,9 @@ void TowTrack::PlayAction(IsleScript::Script p_objectId)
// FUNCTION: LEGO1 0x1004dd30 // FUNCTION: LEGO1 0x1004dd30
TowTrackMissionState::TowTrackMissionState() TowTrackMissionState::TowTrackMissionState()
{ {
m_unk0x08 = 0; m_state = TowTrackMissionState::e_none;
m_startTime = 0; m_startTime = 0;
m_unk0x10 = FALSE; m_takingTooLong = FALSE;
m_peScore = 0; m_peScore = 0;
m_maScore = 0; m_maScore = 0;
m_paScore = 0; m_paScore = 0;

View File

@ -1256,7 +1256,7 @@ void LegoCarBuild::FUN_10024ef0()
m_buildState->m_animationState = LegoVehicleBuildState::e_cutscene; m_buildState->m_animationState = LegoVehicleBuildState::e_cutscene;
FUN_10025720(FUN_10025d70()); FUN_10025720(FUN_10025d70());
m_buildState->m_unk0x4c += 1; m_buildState->m_unk0x4c += 1;
FUN_10015820(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen); Disable(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen);
} }
// FUNCTION: LEGO1 0x10024f30 // FUNCTION: LEGO1 0x10024f30

View File

@ -866,7 +866,7 @@ void LegoGameState::SwitchArea(Area p_area)
m_previousArea = m_currentArea; m_previousArea = m_currentArea;
m_currentArea = p_area; m_currentArea = p_area;
FUN_10015820(TRUE, LegoOmni::c_disableInput | LegoOmni::c_disable3d); Disable(TRUE, LegoOmni::c_disableInput | LegoOmni::c_disable3d);
BackgroundAudioManager()->Stop(); BackgroundAudioManager()->Stop();
AnimationManager()->Suspend(); AnimationManager()->Suspend();
VideoManager()->SetUnk0x554(FALSE); VideoManager()->SetUnk0x554(FALSE);
@ -967,7 +967,7 @@ void LegoGameState::SwitchArea(Area p_area)
Act1State* state = (Act1State*) GameState()->GetState("Act1State"); Act1State* state = (Act1State*) GameState()->GetState("Act1State");
LoadIsle(); LoadIsle();
if (state->GetUnknown18() == 7) { if (state->GetState() == Act1State::e_transitionToTowtrack) {
VideoManager()->Get3DManager()->SetFrustrum(90, 0.1f, 250.0f); VideoManager()->Get3DManager()->SetFrustrum(90, 0.1f, 250.0f);
} }
else { else {

View File

@ -591,9 +591,10 @@ MxBool CanExit()
GameState()->m_currentArea != LegoGameState::e_polidoor) { GameState()->m_currentArea != LegoGameState::e_polidoor) {
if (UserActor() == NULL || !UserActor()->IsA("TowTrack")) { if (UserActor() == NULL || !UserActor()->IsA("TowTrack")) {
if (UserActor() == NULL || !UserActor()->IsA("Ambulance")) { if (UserActor() == NULL || !UserActor()->IsA("Ambulance")) {
MxU32 unk0x18 = act1State->GetUnknown18(); MxU32 mission = act1State->GetState();
if (unk0x18 != 10 && unk0x18 != 8 && unk0x18 != 3) { if (mission != Act1State::e_ambulance && mission != Act1State::e_towtrack &&
mission != Act1State::e_pizza) {
return TRUE; return TRUE;
} }
} }

View File

@ -140,10 +140,10 @@ ViewLODListManager* GetViewLODListManager()
// FUNCTION: LEGO1 0x10015820 // FUNCTION: LEGO1 0x10015820
// FUNCTION: BETA10 0x100e4c92 // FUNCTION: BETA10 0x100e4c92
void FUN_10015820(MxBool p_disable, MxU16 p_flags) void Disable(MxBool p_disable, MxU16 p_flags)
{ {
assert(LegoOmni::GetInstance()); assert(LegoOmni::GetInstance());
LegoOmni::GetInstance()->FUN_1005b4f0(p_disable, p_flags); LegoOmni::GetInstance()->Disable(p_disable, p_flags);
} }
// FUNCTION: LEGO1 0x10015840 // FUNCTION: LEGO1 0x10015840

View File

@ -63,7 +63,7 @@ LegoWorldPresenter::~LegoWorldPresenter()
} }
if (result == FALSE) { if (result == FALSE) {
FUN_10015820(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen); Disable(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen);
} }
if (m_entity) { if (m_entity) {

View File

@ -11,6 +11,8 @@
#include "mxdebug.h" #include "mxdebug.h"
#include "roi/legoroi.h" #include "roi/legoroi.h"
#include <SDL3/SDL_log.h>
DECOMP_SIZE_ASSERT(LegoInputManager, 0x338) DECOMP_SIZE_ASSERT(LegoInputManager, 0x338)
DECOMP_SIZE_ASSERT(LegoNotifyList, 0x18) DECOMP_SIZE_ASSERT(LegoNotifyList, 0x18)
DECOMP_SIZE_ASSERT(LegoNotifyListCursor, 0x10) DECOMP_SIZE_ASSERT(LegoNotifyListCursor, 0x10)
@ -40,10 +42,6 @@ LegoInputManager::LegoInputManager()
m_unk0x81 = FALSE; m_unk0x81 = FALSE;
m_unk0x88 = FALSE; m_unk0x88 = FALSE;
m_unk0x195 = 0; m_unk0x195 = 0;
m_joyids = NULL;
m_joystickIndex = -1;
m_joystick = NULL;
m_useJoystick = FALSE;
m_unk0x335 = FALSE; m_unk0x335 = FALSE;
m_unk0x336 = FALSE; m_unk0x336 = FALSE;
m_unk0x74 = 0x19; m_unk0x74 = 0x19;
@ -71,8 +69,6 @@ MxResult LegoInputManager::Create(HWND p_hwnd)
m_eventQueue = new LegoEventQueue; m_eventQueue = new LegoEventQueue;
} }
GetJoystick();
if (!m_keyboardNotifyList || !m_eventQueue) { if (!m_keyboardNotifyList || !m_eventQueue) {
Destroy(); Destroy();
result = FAILURE; result = FAILURE;
@ -98,7 +94,23 @@ void LegoInputManager::Destroy()
delete m_controlManager; delete m_controlManager;
} }
SDL_free(m_joyids); for (const auto& [id, joystick] : m_joysticks) {
if (joystick.second) {
SDL_CloseHaptic(joystick.second);
}
SDL_CloseGamepad(joystick.first);
}
for (const auto& [id, mouse] : m_mice) {
if (mouse.second) {
SDL_CloseHaptic(mouse.second);
}
}
for (const auto& [id, haptic] : m_otherHaptics) {
SDL_CloseHaptic(haptic);
}
} }
// FUNCTION: LEGO1 0x1005c0f0 // FUNCTION: LEGO1 0x1005c0f0
@ -145,74 +157,39 @@ MxResult LegoInputManager::GetNavigationKeyStates(MxU32& p_keyFlags)
return SUCCESS; return SUCCESS;
} }
// FUNCTION: LEGO1 0x1005c240
MxResult LegoInputManager::GetJoystick()
{
if (m_joystick != NULL && SDL_GamepadConnected(m_joystick) == TRUE) {
return SUCCESS;
}
MxS32 numJoysticks = 0;
if (m_joyids != NULL) {
SDL_free(m_joyids);
m_joyids = NULL;
}
m_joyids = SDL_GetGamepads(&numJoysticks);
if (m_useJoystick != FALSE && numJoysticks != 0) {
MxS32 joyid = m_joystickIndex;
if (joyid >= 0) {
m_joystick = SDL_OpenGamepad(m_joyids[joyid]);
if (m_joystick != NULL) {
return SUCCESS;
}
}
for (joyid = 0; joyid < numJoysticks; joyid++) {
m_joystick = SDL_OpenGamepad(m_joyids[joyid]);
if (m_joystick != NULL) {
return SUCCESS;
}
}
}
return FAILURE;
}
// FUNCTION: LEGO1 0x1005c320 // FUNCTION: LEGO1 0x1005c320
MxResult LegoInputManager::GetJoystickState(MxU32* p_joystickX, MxU32* p_joystickY, MxU32* p_povPosition) MxResult LegoInputManager::GetJoystickState(MxU32* p_joystickX, MxU32* p_joystickY, MxU32* p_povPosition)
{ {
if (m_useJoystick != FALSE) { if (!std::holds_alternative<SDL_JoystickID_v>(m_lastInputMethod) &&
if (GetJoystick() == -1) { !(std::holds_alternative<SDL_TouchID_v>(m_lastInputMethod) && m_touchScheme == e_gamepad)) {
if (m_joystick != NULL) { return FAILURE;
// GetJoystick() failed but handle to joystick is still open, close it }
SDL_CloseGamepad(m_joystick);
m_joystick = NULL;
}
return FAILURE; MxS16 xPos, yPos = 0;
} for (const auto& [id, joystick] : m_joysticks) {
xPos = SDL_GetGamepadAxis(joystick.first, SDL_GAMEPAD_AXIS_LEFTX);
MxS16 xPos = SDL_GetGamepadAxis(m_joystick, SDL_GAMEPAD_AXIS_LEFTX); yPos = SDL_GetGamepadAxis(joystick.first, SDL_GAMEPAD_AXIS_LEFTY);
MxS16 yPos = SDL_GetGamepadAxis(m_joystick, SDL_GAMEPAD_AXIS_LEFTY);
if (xPos > -8000 && xPos < 8000) { if (xPos > -8000 && xPos < 8000) {
// Ignore small axis values
xPos = 0; xPos = 0;
} }
if (yPos > -8000 && yPos < 8000) { if (yPos > -8000 && yPos < 8000) {
// Ignore small axis values
yPos = 0; yPos = 0;
} }
// normalize values acquired from joystick axes if (xPos || yPos) {
*p_joystickX = ((xPos + 32768) * 100) / 65535; break;
*p_joystickY = ((yPos + 32768) * 100) / 65535; }
*p_povPosition = -1;
return SUCCESS;
} }
return FAILURE; if (!xPos && !yPos) {
xPos = m_touchVirtualThumb.x;
yPos = m_touchVirtualThumb.y;
}
*p_joystickX = ((xPos + 32768) * 100) / 65535;
*p_joystickY = ((yPos + 32768) * 100) / 65535;
*p_povPosition = -1;
return SUCCESS;
} }
// FUNCTION: LEGO1 0x1005c470 // FUNCTION: LEGO1 0x1005c470
@ -528,27 +505,105 @@ void LegoInputManager::EnableInputProcessing()
MxResult LegoInputManager::GetNavigationTouchStates(MxU32& p_keyStates) MxResult LegoInputManager::GetNavigationTouchStates(MxU32& p_keyStates)
{ {
for (auto& [fingerID, touchFlags] : m_touchFlags) { if (m_touchScheme == e_arrowKeys) {
p_keyStates |= touchFlags; for (auto& [fingerID, touchFlags] : m_touchFlags) {
p_keyStates |= touchFlags;
// We need to clear these as they are not meant to be persistent in e_gamepad mode.
if (m_touchOrigins.count(fingerID) && m_touchLastMotion.count(fingerID)) {
const MxU32 inactivityThreshold = 3;
if (m_touchLastMotion[fingerID].first++ > inactivityThreshold) {
touchFlags &= ~c_left;
touchFlags &= ~c_right;
m_touchOrigins[fingerID].x = m_touchLastMotion[fingerID].second.x;
}
} }
} }
return SUCCESS; return SUCCESS;
} }
void LegoInputManager::AddKeyboard(SDL_KeyboardID p_keyboardID)
{
if (m_keyboards.count(p_keyboardID)) {
return;
}
m_keyboards[p_keyboardID] = {nullptr, nullptr};
}
void LegoInputManager::RemoveKeyboard(SDL_KeyboardID p_keyboardID)
{
if (!m_keyboards.count(p_keyboardID)) {
return;
}
m_keyboards.erase(p_keyboardID);
}
void LegoInputManager::AddMouse(SDL_MouseID p_mouseID)
{
if (m_mice.count(p_mouseID)) {
return;
}
// Currently no way to get an individual haptic device for a mouse.
SDL_Haptic* haptic = SDL_OpenHapticFromMouse();
if (haptic) {
if (!SDL_InitHapticRumble(haptic)) {
SDL_CloseHaptic(haptic);
haptic = nullptr;
}
}
m_mice[p_mouseID] = {nullptr, haptic};
}
void LegoInputManager::RemoveMouse(SDL_MouseID p_mouseID)
{
if (!m_mice.count(p_mouseID)) {
return;
}
if (m_mice[p_mouseID].second) {
SDL_CloseHaptic(m_mice[p_mouseID].second);
}
m_mice.erase(p_mouseID);
}
void LegoInputManager::AddJoystick(SDL_JoystickID p_joystickID)
{
if (m_joysticks.count(p_joystickID)) {
return;
}
SDL_Gamepad* joystick = SDL_OpenGamepad(p_joystickID);
if (joystick) {
SDL_Haptic* haptic = SDL_OpenHapticFromJoystick(SDL_GetGamepadJoystick(joystick));
if (haptic) {
if (!SDL_InitHapticRumble(haptic)) {
SDL_CloseHaptic(haptic);
haptic = nullptr;
}
}
m_joysticks[p_joystickID] = {joystick, haptic};
}
else {
SDL_Log("Failed to open gamepad: %s", SDL_GetError());
}
}
void LegoInputManager::RemoveJoystick(SDL_JoystickID p_joystickID)
{
if (!m_joysticks.count(p_joystickID)) {
return;
}
if (m_joysticks[p_joystickID].second) {
SDL_CloseHaptic(m_joysticks[p_joystickID].second);
}
SDL_CloseGamepad(m_joysticks[p_joystickID].first);
m_joysticks.erase(p_joystickID);
}
MxBool LegoInputManager::HandleTouchEvent(SDL_Event* p_event, TouchScheme p_touchScheme) MxBool LegoInputManager::HandleTouchEvent(SDL_Event* p_event, TouchScheme p_touchScheme)
{ {
const SDL_TouchFingerEvent& event = p_event->tfinger; const SDL_TouchFingerEvent& event = p_event->tfinger;
m_touchScheme = p_touchScheme;
switch (p_touchScheme) { switch (p_touchScheme) {
case e_mouse: case e_mouse:
@ -580,64 +635,167 @@ MxBool LegoInputManager::HandleTouchEvent(SDL_Event* p_event, TouchScheme p_touc
break; break;
} }
break; break;
case e_gamepad: case e_gamepad: {
static SDL_FingerID g_finger = (SDL_FingerID) 0;
switch (p_event->type) { switch (p_event->type) {
case SDL_EVENT_FINGER_DOWN: case SDL_EVENT_FINGER_DOWN:
m_touchOrigins[event.fingerID] = {event.x, event.y}; if (!g_finger) {
g_finger = event.fingerID;
m_touchVirtualThumb = {0, 0};
m_touchVirtualThumbOrigin = {event.x, event.y};
}
break; break;
case SDL_EVENT_FINGER_UP: case SDL_EVENT_FINGER_UP:
m_touchOrigins.erase(event.fingerID); if (event.fingerID == g_finger) {
m_touchFlags.erase(event.fingerID); g_finger = 0;
m_touchVirtualThumb = {0, 0};
m_touchVirtualThumbOrigin = {0, 0};
}
break; break;
case SDL_EVENT_FINGER_MOTION: case SDL_EVENT_FINGER_MOTION:
if (m_touchOrigins.count(event.fingerID)) { if (event.fingerID == g_finger) {
const float activationThreshold = 0.03f; const float thumbstickRadius = 0.25f;
m_touchFlags[event.fingerID] &= ~c_down; const float deltaX =
m_touchFlags[event.fingerID] &= ~c_up; SDL_clamp(event.x - m_touchVirtualThumbOrigin.x, -thumbstickRadius, thumbstickRadius);
const float deltaY =
SDL_clamp(event.y - m_touchVirtualThumbOrigin.y, -thumbstickRadius, thumbstickRadius);
const float deltaY = event.y - m_touchOrigins[event.fingerID].y; m_touchVirtualThumb = {
if (SDL_fabsf(deltaY) > activationThreshold) { (int) (deltaX / thumbstickRadius * 32767.0f),
if (deltaY > 0) { (int) (deltaY / thumbstickRadius * 32767.0f),
m_touchFlags[event.fingerID] |= c_down; };
}
else if (deltaY < 0) {
m_touchFlags[event.fingerID] |= c_up;
}
}
const float deltaX = event.x - m_touchOrigins[event.fingerID].x;
if (SDL_fabsf(deltaX) > activationThreshold && event.dx) {
if (event.dx > 0) {
m_touchFlags[event.fingerID] |= c_right;
m_touchFlags[event.fingerID] &= ~c_left;
}
else if (event.dx < 0) {
m_touchFlags[event.fingerID] |= c_left;
m_touchFlags[event.fingerID] &= ~c_right;
}
m_touchLastMotion[event.fingerID] = {0, {event.x, event.y}};
}
} }
break; break;
} }
break; break;
} }
}
return TRUE; return TRUE;
} }
MxBool LegoInputManager::HandleRumbleEvent() MxBool LegoInputManager::HandleRumbleEvent(
float p_strength,
float p_lowFrequencyRumble,
float p_highFrequencyRumble,
MxU32 p_milliseconds
)
{ {
if (m_joystick != NULL && SDL_GamepadConnected(m_joystick) == TRUE) { static bool g_hapticsInitialized = false;
const Uint16 frequency = 65535 / 2;
const Uint32 durationMs = 700; if (!g_hapticsInitialized) {
SDL_RumbleGamepad(m_joystick, frequency, frequency, durationMs); InitializeHaptics();
} g_hapticsInitialized = true;
else {
return FALSE;
} }
// Add support for SDL Haptic API SDL_Haptic* haptic = nullptr;
return TRUE; std::visit(
overloaded{
[](SDL_KeyboardID_v p_id) {},
[&haptic, this](SDL_MouseID_v p_id) {
if (m_mice.count((SDL_MouseID) p_id)) {
haptic = m_mice[(SDL_MouseID) p_id].second;
}
},
[&haptic, this](SDL_JoystickID_v p_id) {
if (m_joysticks.count((SDL_JoystickID) p_id)) {
haptic = m_joysticks[(SDL_JoystickID) p_id].second;
}
},
[&haptic, this](SDL_TouchID_v p_id) {
// We can't currently correlate Touch devices with Haptic devices
if (!m_otherHaptics.empty()) {
haptic = m_otherHaptics.begin()->second;
}
}
},
m_lastInputMethod
);
if (haptic) {
return SDL_PlayHapticRumble(haptic, p_strength, p_milliseconds);
}
// A joystick isn't necessarily a haptic device; try basic rumble instead
if (const SDL_JoystickID_v* joystick = std::get_if<SDL_JoystickID_v>(&m_lastInputMethod)) {
if (m_joysticks.count((SDL_JoystickID) *joystick)) {
return SDL_RumbleGamepad(
m_joysticks[(SDL_JoystickID) *joystick].first,
SDL_clamp(p_lowFrequencyRumble, 0, 1) * USHRT_MAX,
SDL_clamp(p_highFrequencyRumble, 0, 1) * USHRT_MAX,
p_milliseconds
);
}
}
return FALSE;
}
void LegoInputManager::InitializeHaptics()
{
// We don't get added/removed events for haptic devices that are not joysticks or mice,
// so we initialize "otherHaptics" once at this point.
std::vector<SDL_HapticID> existingHaptics;
for (const auto& [id, mouse] : m_mice) {
if (mouse.second) {
existingHaptics.push_back(SDL_GetHapticID(mouse.second));
}
}
for (const auto& [id, joystick] : m_joysticks) {
if (joystick.second) {
existingHaptics.push_back(SDL_GetHapticID(joystick.second));
}
}
int count;
SDL_HapticID* haptics = SDL_GetHaptics(&count);
if (haptics) {
for (int i = 0; i < count; i++) {
if (std::find(existingHaptics.begin(), existingHaptics.end(), haptics[i]) == existingHaptics.end()) {
SDL_Haptic* haptic = SDL_OpenHaptic(haptics[i]);
if (haptic) {
if (SDL_InitHapticRumble(haptic)) {
m_otherHaptics[haptics[i]] = haptic;
}
else {
SDL_CloseHaptic(haptic);
}
}
}
}
SDL_free(haptics);
}
}
void LegoInputManager::UpdateLastInputMethod(SDL_Event* p_event)
{
switch (p_event->type) {
case SDL_EVENT_KEY_DOWN:
case SDL_EVENT_KEY_UP:
m_lastInputMethod = SDL_KeyboardID_v{p_event->key.which};
break;
case SDL_EVENT_MOUSE_BUTTON_DOWN:
case SDL_EVENT_MOUSE_BUTTON_UP:
m_lastInputMethod = SDL_MouseID_v{p_event->button.which};
break;
case SDL_EVENT_MOUSE_MOTION:
m_lastInputMethod = SDL_MouseID_v{p_event->motion.which};
break;
case SDL_EVENT_GAMEPAD_BUTTON_DOWN:
case SDL_EVENT_GAMEPAD_BUTTON_UP:
m_lastInputMethod = SDL_JoystickID_v{p_event->gbutton.which};
break;
case SDL_EVENT_GAMEPAD_AXIS_MOTION:
m_lastInputMethod = SDL_JoystickID_v{p_event->gaxis.which};
break;
case SDL_EVENT_FINGER_MOTION:
case SDL_EVENT_FINGER_DOWN:
case SDL_EVENT_FINGER_UP:
m_lastInputMethod = SDL_TouchID_v{p_event->tfinger.touchID};
break;
}
} }

View File

@ -543,26 +543,36 @@ LegoOmni::World LegoOmni::GetWorldId(const char* p_key)
} }
// FUNCTION: LEGO1 0x1005b4f0 // FUNCTION: LEGO1 0x1005b4f0
void LegoOmni::FUN_1005b4f0(MxBool p_disable, MxU16 p_flags) // FUNCTION: BETA10 0x1008eeec
void LegoOmni::Disable(MxBool p_disable, MxU16 p_flags)
{ {
if (p_disable) { #ifdef BETA10
if (p_flags & c_disableInput) { if (this->m_paused != p_disable) {
m_inputManager->DisableInputProcessing(); // This is probably a different variable, but this code was mostly added for structural matching
} m_paused = p_disable;
#endif
if (p_flags & c_disable3d) { if (p_disable) {
((LegoVideoManager*) m_videoManager)->SetRender3D(FALSE); if (p_flags & c_disableInput) {
} m_inputManager->DisableInputProcessing();
}
if (p_flags & c_clearScreen) { if (p_flags & c_disable3d) {
m_videoManager->GetDisplaySurface()->ClearScreen(); ((LegoVideoManager*) m_videoManager)->SetRender3D(FALSE);
}
if (p_flags & c_clearScreen) {
m_videoManager->GetDisplaySurface()->ClearScreen();
}
} }
else {
m_inputManager->EnableInputProcessing();
((LegoVideoManager*) m_videoManager)->SetRender3D(TRUE);
((LegoVideoManager*) m_videoManager)->UpdateView(0, 0, 0, 0);
}
#ifdef BETA10
} }
else { #endif
m_inputManager->EnableInputProcessing();
((LegoVideoManager*) m_videoManager)->SetRender3D(TRUE);
((LegoVideoManager*) m_videoManager)->UpdateView(0, 0, 0, 0);
}
} }
// FUNCTION: LEGO1 0x1005b560 // FUNCTION: LEGO1 0x1005b560

View File

@ -97,7 +97,7 @@ MxResult CarRace::Create(MxDSAction& p_dsAction)
m_raceState = raceState; m_raceState = raceState;
m_act1State->m_unk0x018 = 6; m_act1State->m_state = Act1State::e_transitionToRacecar;
m_unk0x144 = -1; m_unk0x144 = -1;
m_unk0x148 = -1; m_unk0x148 = -1;
m_unk0x14c = -1; m_unk0x14c = -1;
@ -126,7 +126,7 @@ void CarRace::ReadyWorld()
BackgroundAudioManager()->PlayMusic(action, 5, MxPresenter::e_repeating); BackgroundAudioManager()->PlayMusic(action, 5, MxPresenter::e_repeating);
AnimationManager()->Resume(); AnimationManager()->Resume();
FUN_10015820(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen); Disable(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen);
m_unk0x144 = g_unk0x100d5d10[SDL_rand(8)]; m_unk0x144 = g_unk0x100d5d10[SDL_rand(8)];
@ -181,7 +181,7 @@ MxLong CarRace::HandlePathStruct(LegoPathStructNotificationParam& p_param)
{ {
MxLong result = 0; MxLong result = 0;
if (p_param.GetTrigger() == 68) { if (p_param.GetTrigger() == LegoPathStruct::c_d) {
MxEntity* sender = (MxEntity*) p_param.GetSender(); MxEntity* sender = (MxEntity*) p_param.GetSender();
MxS32 paramData = p_param.GetData(); MxS32 paramData = p_param.GetData();
@ -341,7 +341,7 @@ MxLong CarRace::HandleControl(LegoControlManagerNotificationParam& p_param)
switch (p_param.m_clickedObjectId) { switch (p_param.m_clickedObjectId) {
case 3: case 3:
InvokeAction(Extra::e_stop, *g_carraceScript, CarraceScript::c_irtx08ra_PlayWav, NULL); InvokeAction(Extra::e_stop, *g_carraceScript, CarraceScript::c_irtx08ra_PlayWav, NULL);
m_act1State->m_unk0x018 = 0; m_act1State->m_state = Act1State::e_none;
VariableTable()->SetVariable(g_raceState, ""); VariableTable()->SetVariable(g_raceState, "");
VariableTable()->SetVariable(g_strHIT_WALL_SOUND, ""); VariableTable()->SetVariable(g_strHIT_WALL_SOUND, "");
NavController()->SetDeadZone(NavController()->GetDefaultDeadZone()); NavController()->SetDeadZone(NavController()->GetDefaultDeadZone());
@ -353,7 +353,7 @@ MxLong CarRace::HandleControl(LegoControlManagerNotificationParam& p_param)
break; break;
case 98: case 98:
InvokeAction(Extra::e_stop, *g_carraceScript, CarraceScript::c_irtx08ra_PlayWav, NULL); InvokeAction(Extra::e_stop, *g_carraceScript, CarraceScript::c_irtx08ra_PlayWav, NULL);
m_act1State->m_unk0x018 = 0; m_act1State->m_state = Act1State::e_none;
VariableTable()->SetVariable(g_raceState, ""); VariableTable()->SetVariable(g_raceState, "");
VariableTable()->SetVariable(g_strHIT_WALL_SOUND, ""); VariableTable()->SetVariable(g_strHIT_WALL_SOUND, "");
NavController()->SetDeadZone(NavController()->GetDefaultDeadZone()); NavController()->SetDeadZone(NavController()->GetDefaultDeadZone());
@ -415,7 +415,7 @@ MxBool CarRace::Escape()
AnimationManager()->FUN_10061010(FALSE); AnimationManager()->FUN_10061010(FALSE);
DeleteObjects(&m_atomId, 500, 999); DeleteObjects(&m_atomId, 500, 999);
m_act1State->m_unk0x018 = 0; m_act1State->m_state = Act1State::e_none;
VariableTable()->SetVariable(g_strHIT_WALL_SOUND, ""); VariableTable()->SetVariable(g_strHIT_WALL_SOUND, "");
VariableTable()->SetVariable(g_raceState, ""); VariableTable()->SetVariable(g_raceState, "");

View File

@ -97,7 +97,7 @@ void JetskiRace::ReadyWorld()
m_unk0x12c = (MxStillPresenter*) Find("MxPresenter", "JetskiLocator3"); m_unk0x12c = (MxStillPresenter*) Find("MxPresenter", "JetskiLocator3");
m_unk0x12c->SetPosition(m_unk0x130.GetLeft(), m_unk0x130.GetTop()); m_unk0x12c->SetPosition(m_unk0x130.GetLeft(), m_unk0x130.GetTop());
FUN_10015820(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen); Disable(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen);
VariableTable()->SetVariable("DISTANCE", "0.036"); VariableTable()->SetVariable("DISTANCE", "0.036");
@ -129,7 +129,7 @@ MxLong JetskiRace::HandleControl(LegoControlManagerNotificationParam& p_param)
if (p_param.m_unk0x28 == 1) { if (p_param.m_unk0x28 == 1) {
switch (p_param.m_clickedObjectId) { switch (p_param.m_clickedObjectId) {
case JetraceScript::c_JetskiArms_Ctl: case JetraceScript::c_JetskiArms_Ctl:
m_act1State->m_unk0x018 = 0; m_act1State->m_state = Act1State::e_none;
VariableTable()->SetVariable(g_raceState, ""); VariableTable()->SetVariable(g_raceState, "");
VariableTable()->SetVariable(g_strHIT_WALL_SOUND, ""); VariableTable()->SetVariable(g_strHIT_WALL_SOUND, "");
LegoRaceCar::InitYouCantStopSound(); LegoRaceCar::InitYouCantStopSound();
@ -137,7 +137,7 @@ MxLong JetskiRace::HandleControl(LegoControlManagerNotificationParam& p_param)
TransitionManager()->StartTransition(MxTransitionManager::e_mosaic, 50, FALSE, FALSE); TransitionManager()->StartTransition(MxTransitionManager::e_mosaic, 50, FALSE, FALSE);
break; break;
case JetraceScript::c_JetskiInfo_Ctl: case JetraceScript::c_JetskiInfo_Ctl:
m_act1State->m_unk0x018 = 0; m_act1State->m_state = Act1State::e_none;
VariableTable()->SetVariable(g_raceState, ""); VariableTable()->SetVariable(g_raceState, "");
VariableTable()->SetVariable(g_strHIT_WALL_SOUND, ""); VariableTable()->SetVariable(g_strHIT_WALL_SOUND, "");
LegoRaceCar::InitYouCantStopSound(); LegoRaceCar::InitYouCantStopSound();
@ -160,7 +160,7 @@ MxLong JetskiRace::HandlePathStruct(LegoPathStructNotificationParam& p_param)
MxLong result = 0; MxLong result = 0;
MxEntity* sender = (MxEntity*) p_param.GetSender(); MxEntity* sender = (MxEntity*) p_param.GetSender();
if (p_param.GetTrigger() == 68) { if (p_param.GetTrigger() == LegoPathStruct::c_d) {
MxS32 paramData = p_param.GetData(); MxS32 paramData = p_param.GetData();
switch (sender->GetEntityId()) { switch (sender->GetEntityId()) {
@ -290,7 +290,7 @@ MxBool JetskiRace::Escape()
{ {
AnimationManager()->FUN_10061010(FALSE); AnimationManager()->FUN_10061010(FALSE);
DeleteObjects(&m_atomId, 500, 999); DeleteObjects(&m_atomId, 500, 999);
m_act1State->m_unk0x018 = 0; m_act1State->m_state = Act1State::e_none;
VariableTable()->SetVariable(g_raceState, ""); VariableTable()->SetVariable(g_raceState, "");
VariableTable()->SetVariable(g_strHIT_WALL_SOUND, ""); VariableTable()->SetVariable(g_strHIT_WALL_SOUND, "");
m_destLocation = LegoGameState::e_infomain; m_destLocation = LegoGameState::e_infomain;

View File

@ -706,11 +706,14 @@ MxResult LegoAnimPresenter::FUN_1006b140(LegoROI* p_roi)
if (p_roi == NULL) { if (p_roi == NULL) {
return FAILURE; return FAILURE;
} }
#ifdef BETA10
MxMatrix unused_matrix;
#endif
Matrix4* mn = new MxMatrix(); Matrix4* mn = new MxMatrix();
assert(mn); assert(mn);
MxMatrix local58; MxMatrix inverse;
const Matrix4& local2world = p_roi->GetLocal2World(); const Matrix4& local2world = p_roi->GetLocal2World();
MxMatrix* local5c; MxMatrix* local5c;
MxU32 i; MxU32 i;
@ -721,7 +724,7 @@ MxResult LegoAnimPresenter::FUN_1006b140(LegoROI* p_roi)
for (i = 1; i <= m_roiMapSize; i++) { for (i = 1; i <= m_roiMapSize; i++) {
if (m_roiMap[i] == p_roi) { if (m_roiMap[i] == p_roi) {
if (local5c[i].BETA_1005a590(local58) != SUCCESS) { if (local5c[i].Invert(inverse) != SUCCESS) {
goto done; goto done;
} }
@ -730,7 +733,7 @@ MxResult LegoAnimPresenter::FUN_1006b140(LegoROI* p_roi)
} }
{ {
mn->Product(local58, local2world); mn->Product(inverse, local2world);
SetUnknown0xa0(mn); SetUnknown0xa0(mn);
delete[] local5c; delete[] local5c;
SetUnknown0x0cTo1(); SetUnknown0x0cTo1();

View File

@ -114,7 +114,7 @@ void LegoHideAnimPresenter::FUN_1006db60(LegoTreeNode* p_node, LegoTime p_time)
} }
if (m_boundaryMap != NULL) { if (m_boundaryMap != NULL) {
LegoPathBoundary* boundary = m_boundaryMap[data->GetUnknown0x22()]; LegoPathBoundary* boundary = m_boundaryMap[data->GetBoundaryIndex()];
if (boundary != NULL) { if (boundary != NULL) {
newB = data->GetVisibility(p_time); newB = data->GetVisibility(p_time);
@ -163,7 +163,7 @@ void LegoHideAnimPresenter::FUN_1006e3f0(LegoHideAnimStructMap& p_map, LegoTreeN
FUN_1006e470(p_map, data, name, boundary); FUN_1006e470(p_map, data, name, boundary);
} }
else { else {
data->SetUnknown0x22(0); data->SetBoundaryIndex(0);
} }
} }
@ -190,7 +190,7 @@ void LegoHideAnimPresenter::FUN_1006e470(
animStruct.m_index = p_map.size() + 1; animStruct.m_index = p_map.size() + 1;
animStruct.m_boundary = p_boundary; animStruct.m_boundary = p_boundary;
p_data->SetUnknown0x22(animStruct.m_index); p_data->SetBoundaryIndex(animStruct.m_index);
char* name = new char[strlen(p_name) + 1]; char* name = new char[strlen(p_name) + 1];
strcpy(name, p_name); strcpy(name, p_name);
@ -198,7 +198,7 @@ void LegoHideAnimPresenter::FUN_1006e470(
p_map[name] = animStruct; p_map[name] = animStruct;
} }
else { else {
p_data->SetUnknown0x22((*it).second.m_index); p_data->SetBoundaryIndex((*it).second.m_index);
} }
} }

View File

@ -587,13 +587,13 @@ void LegoVideoManager::EnableFullScreenMovie(MxBool p_enable, MxBool p_scale)
// FUNCTION: LEGO1 0x1007c440 // FUNCTION: LEGO1 0x1007c440
void LegoVideoManager::SetSkyColor(float p_red, float p_green, float p_blue) void LegoVideoManager::SetSkyColor(float p_red, float p_green, float p_blue)
{ {
PALETTEENTRY colorStrucure; PALETTEENTRY colorStructure;
colorStrucure.peRed = (p_red * 255.0f); colorStructure.peRed = (p_red * 255.0f);
colorStrucure.peGreen = (p_green * 255.0f); colorStructure.peGreen = (p_green * 255.0f);
colorStrucure.peBlue = (p_blue * 255.0f); colorStructure.peBlue = (p_blue * 255.0f);
colorStrucure.peFlags = D3DPAL_RESERVED | PC_NOCOLLAPSE; colorStructure.peFlags = D3DPAL_RESERVED | PC_NOCOLLAPSE;
m_videoParam.GetPalette()->SetSkyColor(&colorStrucure); m_videoParam.GetPalette()->SetSkyColor(&colorStructure);
m_videoParam.GetPalette()->SetOverrideSkyColor(TRUE); m_videoParam.GetPalette()->SetOverrideSkyColor(TRUE);
m_3dManager->GetLego3DView()->GetView()->SetBackgroundColor(p_red, p_green, p_blue); m_3dManager->GetLego3DView()->GetView()->SetBackgroundColor(p_red, p_green, p_blue);
} }

View File

@ -110,16 +110,16 @@ Act3Script::Script g_unk0x100d95e8[] =
{Act3Script::c_tlp053in_RunAnim, Act3Script::c_tlp064la_RunAnim, Act3Script::c_tlp068in_RunAnim}; {Act3Script::c_tlp053in_RunAnim, Act3Script::c_tlp064la_RunAnim, Act3Script::c_tlp068in_RunAnim};
// FUNCTION: LEGO1 0x10071d40 // FUNCTION: LEGO1 0x10071d40
void Act3List::Insert(MxS32 p_objectId, MxS32 p_option) void Act3List::Insert(MxS32 p_objectId, InsertMode p_option)
{ {
if (m_unk0x0c) { if (m_unk0x0c) {
return; return;
} }
switch (p_option) { switch (p_option) {
case 1: case InsertMode::e_replaceAction:
if (!empty()) { if (!empty()) {
FUN_10071fa0(); DeleteActionWrapper();
push_back(Act3ListElement(p_objectId, p_option, FALSE)); push_back(Act3ListElement(p_objectId, p_option, FALSE));
} }
else { else {
@ -127,7 +127,7 @@ void Act3List::Insert(MxS32 p_objectId, MxS32 p_option)
push_back(Act3ListElement(p_objectId, p_option, TRUE)); push_back(Act3ListElement(p_objectId, p_option, TRUE));
} }
break; break;
case 2: case InsertMode::e_queueAction:
if (empty()) { if (empty()) {
push_back(Act3ListElement(p_objectId, p_option, TRUE)); push_back(Act3ListElement(p_objectId, p_option, TRUE));
InvokeAction(Extra::e_start, *g_act3Script, p_objectId, NULL); InvokeAction(Extra::e_start, *g_act3Script, p_objectId, NULL);
@ -136,7 +136,7 @@ void Act3List::Insert(MxS32 p_objectId, MxS32 p_option)
push_back(Act3ListElement(p_objectId, p_option, FALSE)); push_back(Act3ListElement(p_objectId, p_option, FALSE));
} }
break; break;
case 3: case InsertMode::e_onlyIfEmpty:
if (empty()) { if (empty()) {
push_back(Act3ListElement(p_objectId, p_option, TRUE)); push_back(Act3ListElement(p_objectId, p_option, TRUE));
InvokeAction(Extra::e_start, *g_act3Script, p_objectId, NULL); InvokeAction(Extra::e_start, *g_act3Script, p_objectId, NULL);
@ -146,7 +146,7 @@ void Act3List::Insert(MxS32 p_objectId, MxS32 p_option)
} }
// FUNCTION: LEGO1 0x10071fa0 // FUNCTION: LEGO1 0x10071fa0
void Act3List::FUN_10071fa0() void Act3List::DeleteActionWrapper()
{ {
DeleteAction(); DeleteAction();
} }
@ -162,7 +162,7 @@ void Act3List::Clear()
} }
for (Act3List::iterator it = begin(); it != end();) { for (Act3List::iterator it = begin(); it != end();) {
if ((*it).m_unk0x08) { if ((*it).m_hasStarted) {
MxDSAction ds; MxDSAction ds;
ds.SetAtomId(*g_act3Script); ds.SetAtomId(*g_act3Script);
ds.SetObjectId((*it).m_objectId); ds.SetObjectId((*it).m_objectId);
@ -173,50 +173,64 @@ void Act3List::Clear()
} }
} }
// Removes the element with the given objectId from the list, or the first if `p_objectId` is zero.
// FUNCTION: LEGO1 0x100720d0 // FUNCTION: LEGO1 0x100720d0
void Act3List::FUN_100720d0(MxU32 p_objectId) void Act3List::RemoveByObjectIdOrFirst(MxU32 p_objectId)
{ {
if (m_unk0x0c == 0) { if (m_unk0x0c) {
MxU32 removed = FALSE; return;
}
if (!empty()) { MxU32 removed = FALSE;
if (p_objectId != 0) { Act3List::iterator it;
for (Act3List::iterator it = begin(); it != end(); it++) { // This iterator appears to be unnecessary - maybe left in by accident, or it was used for assertions.
if ((*it).m_unk0x08 && (*it).m_objectId == p_objectId) { // Removing it decreases the match percentage.
erase(it); Act3List::iterator unusedIterator;
removed = TRUE;
break; if (empty()) {
} return;
} }
}
else { if (!p_objectId) {
pop_front(); pop_front();
removed = TRUE;
}
else {
for (it = begin(); it != end(); it++) {
// Removing this variable decreases the match, but replacing `*it` by `unused` below also does.
Act3ListElement& unused = *it;
if ((*it).m_hasStarted && (*it).m_objectId == p_objectId) {
erase(it);
removed = TRUE; removed = TRUE;
break;
} }
}
}
if (removed && size() > 0) { if (removed && size() > 0) {
// TODO: Match it = begin();
Act3List::iterator it = begin(); unusedIterator = it;
Act3ListElement& item = *(it++); Act3ListElement& firstItem = front();
it++;
for (; it != end(); it++) { while (it != end()) {
if ((*it).m_unk0x04 == 1) { if ((*it).m_unk0x04 == 1) {
for (Act3List::iterator it2 = begin(); it2 != it;) { for (Act3List::iterator it2 = begin(); it2 != it; erase(it2++)) {
if ((*it2).m_unk0x08) { if ((*it2).m_hasStarted) {
FUN_10071fa0(); DeleteActionWrapper();
return; return;
}
it2 = erase(it2);
}
} }
} }
if (!item.m_unk0x08) {
item.m_unk0x08 = TRUE;
InvokeAction(Extra::e_start, *g_act3Script, item.m_objectId, NULL);
}
} }
it++;
unusedIterator++;
}
if (!firstItem.m_hasStarted) {
firstItem.m_hasStarted = TRUE;
InvokeAction(Extra::e_start, *g_act3Script, firstItem.m_objectId, NULL);
} }
} }
} }
@ -455,14 +469,14 @@ void Act3::TriggerHitSound(undefined4 p_param1)
m_bricksterDonutSound = 0; m_bricksterDonutSound = 0;
} }
m_unk0x4220.Insert(g_bricksterDonutSounds[m_bricksterDonutSound++], 1); m_unk0x4220.Insert(g_bricksterDonutSounds[m_bricksterDonutSound++], Act3List::e_replaceAction);
return; return;
} }
default: default:
return; return;
} }
m_unk0x4220.Insert(objectId, 3); m_unk0x4220.Insert(objectId, Act3List::e_onlyIfEmpty);
} }
// FUNCTION: LEGO1 0x10072c30 // FUNCTION: LEGO1 0x10072c30
@ -553,7 +567,7 @@ MxLong Act3::Notify(MxParam& p_param)
if (param.GetAction() != NULL && param.GetAction()->GetAtomId() == *g_act3Script) { if (param.GetAction() != NULL && param.GetAction()->GetAtomId() == *g_act3Script) {
if (param.GetAction()->GetObjectId() == Act3Script::c_HelicopterDashboard) { if (param.GetAction()->GetObjectId() == Act3Script::c_HelicopterDashboard) {
MxDSAction action; MxDSAction action;
FUN_10015820(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen); Disable(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen);
SetAppCursor(e_cursorArrow); SetAppCursor(e_cursorArrow);
VideoManager()->Get3DManager()->SetFrustrum(45.0f, 0.1f, 125.0f); VideoManager()->Get3DManager()->SetFrustrum(45.0f, 0.1f, 125.0f);
@ -613,7 +627,7 @@ MxLong Act3::Notify(MxParam& p_param)
} while (length < (MxS32) sizeOfArray(m_helicopterDots)); } while (length < (MxS32) sizeOfArray(m_helicopterDots));
} }
else { else {
m_unk0x4220.FUN_100720d0(param.GetAction()->GetObjectId()); m_unk0x4220.RemoveByObjectIdOrFirst(param.GetAction()->GetObjectId());
} }
} }
break; break;
@ -633,9 +647,9 @@ MxLong Act3::Notify(MxParam& p_param)
case c_notificationEndAnim: case c_notificationEndAnim:
if (m_state->m_unk0x08 == 1) { if (m_state->m_unk0x08 == 1) {
assert(m_copter && m_brickster && m_cop1 && m_cop2); assert(m_copter && m_brickster && m_cop1 && m_cop2);
m_unk0x4220.FUN_100720d0(0); m_unk0x4220.RemoveByObjectIdOrFirst(0);
m_state->m_unk0x08 = 0; m_state->m_unk0x08 = 0;
FUN_10015820(TRUE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen); Disable(TRUE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen);
m_copter->HandleClick(); m_copter->HandleClick();
m_copter->m_state->m_unk0x08 = 1; m_copter->m_state->m_unk0x08 = 1;
m_copter->HandleEndAnim((LegoEndAnimNotificationParam&) param); m_copter->HandleEndAnim((LegoEndAnimNotificationParam&) param);
@ -686,7 +700,7 @@ MxResult Act3::Tickle()
if (m_unk0x426c != (Act3Script::Script) 0) { if (m_unk0x426c != (Act3Script::Script) 0) {
if (AnimationManager()->FUN_10064ee0(m_unk0x426c)) { if (AnimationManager()->FUN_10064ee0(m_unk0x426c)) {
FUN_10015820(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen); Disable(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen);
TickleManager()->UnregisterClient(this); TickleManager()->UnregisterClient(this);
m_unk0x426c = (Act3Script::Script) 0; m_unk0x426c = (Act3Script::Script) 0;
} }
@ -881,7 +895,7 @@ void Act3::Enable(MxBool p_enable)
GameState()->StopArea(LegoGameState::e_infomain); GameState()->StopArea(LegoGameState::e_infomain);
} }
FUN_10015820(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen); Disable(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen);
PlayMusic(JukeboxScript::c_Act3Music); PlayMusic(JukeboxScript::c_Act3Music);
GameState()->m_isDirty = TRUE; GameState()->m_isDirty = TRUE;

View File

@ -79,7 +79,7 @@ void ElevatorBottom::ReadyWorld()
{ {
LegoWorld::ReadyWorld(); LegoWorld::ReadyWorld();
PlayMusic(JukeboxScript::c_InformationCenter_Music); PlayMusic(JukeboxScript::c_InformationCenter_Music);
FUN_10015820(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen); Disable(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen);
} }
// FUNCTION: LEGO1 0x100181d0 // FUNCTION: LEGO1 0x100181d0

View File

@ -267,7 +267,7 @@ void GasStation::ReadyWorld()
break; break;
} }
FUN_10015820(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen); Disable(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen);
} }
// FUNCTION: LEGO1 0x10005590 // FUNCTION: LEGO1 0x10005590
@ -330,7 +330,7 @@ MxLong GasStation::HandleEndAction(MxEndActionNotificationParam& p_param)
break; break;
case GasStationState::e_afterAcceptingQuest: case GasStationState::e_afterAcceptingQuest:
m_state->m_state = GasStationState::e_beforeExitingForQuest; m_state->m_state = GasStationState::e_beforeExitingForQuest;
((Act1State*) GameState()->GetState("Act1State"))->m_unk0x018 = 7; ((Act1State*) GameState()->GetState("Act1State"))->m_state = Act1State::e_transitionToTowtrack;
m_destLocation = LegoGameState::e_garageExited; m_destLocation = LegoGameState::e_garageExited;
m_radio.Stop(); m_radio.Stop();
BackgroundAudioManager()->Stop(); BackgroundAudioManager()->Stop();

View File

@ -216,7 +216,7 @@ void Hospital::ReadyWorld()
m_setWithCurrentAction = 1; m_setWithCurrentAction = 1;
} }
FUN_10015820(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen); Disable(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen);
} }
// FUNCTION: LEGO1 0x10074dd0 // FUNCTION: LEGO1 0x10074dd0
@ -367,7 +367,7 @@ MxLong Hospital::HandleEndAction(MxEndActionNotificationParam& p_param)
case HospitalState::e_afterAcceptingQuest: case HospitalState::e_afterAcceptingQuest:
m_hospitalState->m_state = HospitalState::e_beforeEnteringAmbulance; m_hospitalState->m_state = HospitalState::e_beforeEnteringAmbulance;
act1State = (Act1State*) GameState()->GetState("Act1State"); act1State = (Act1State*) GameState()->GetState("Act1State");
act1State->SetUnknown18(9); act1State->SetState(Act1State::e_transitionToAmbulance);
case HospitalState::e_exitToFront: case HospitalState::e_exitToFront:
if (m_exited == FALSE) { if (m_exited == FALSE) {
m_exited = TRUE; m_exited = TRUE;
@ -421,7 +421,7 @@ MxLong Hospital::HandleButtonDown(LegoControlManagerNotificationParam& p_param)
Act1State* act1State = (Act1State*) GameState()->GetState("Act1State"); Act1State* act1State = (Act1State*) GameState()->GetState("Act1State");
assert(act1State); assert(act1State);
act1State->m_unk0x018 = 9; act1State->m_state = Act1State::e_transitionToAmbulance;
m_destLocation = LegoGameState::e_hospitalExited; m_destLocation = LegoGameState::e_hospitalExited;
DeleteObjects( DeleteObjects(

View File

@ -472,7 +472,7 @@ void Infocenter::ReadyWorld()
PlayAction(InfomainScript::c_iicx18in_RunAnim); PlayAction(InfomainScript::c_iicx18in_RunAnim);
PlayMusic(JukeboxScript::c_InformationCenter_Music); PlayMusic(JukeboxScript::c_InformationCenter_Music);
FUN_10015820(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen); Disable(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen);
return; return;
case 5: case 5:
default: { default: {
@ -485,7 +485,7 @@ void Infocenter::ReadyWorld()
m_bigInfoBlinkTimer = 1; m_bigInfoBlinkTimer = 1;
} }
FUN_10015820(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen); Disable(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen);
if (!m_infocenterState->HasRegistered()) { if (!m_infocenterState->HasRegistered()) {
m_bookAnimationTimer = 1; m_bookAnimationTimer = 1;
@ -496,7 +496,7 @@ void Infocenter::ReadyWorld()
case 8: case 8:
PlayMusic(JukeboxScript::c_InformationCenter_Music); PlayMusic(JukeboxScript::c_InformationCenter_Music);
PlayAction(InfomainScript::c_iic043in_RunAnim); PlayAction(InfomainScript::c_iic043in_RunAnim);
FUN_10015820(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen); Disable(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen);
return; return;
case 0xf: case 0xf:
m_infocenterState->m_unk0x74 = 2; m_infocenterState->m_unk0x74 = 2;
@ -506,7 +506,7 @@ void Infocenter::ReadyWorld()
PlayAction(InfomainScript::c_iicx17in_RunAnim); PlayAction(InfomainScript::c_iicx17in_RunAnim);
PlayMusic(JukeboxScript::c_InformationCenter_Music); PlayMusic(JukeboxScript::c_InformationCenter_Music);
FUN_10015820(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen); Disable(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen);
return; return;
} }
break; break;
@ -515,7 +515,7 @@ void Infocenter::ReadyWorld()
PlayMusic(JukeboxScript::c_InformationCenter_Music); PlayMusic(JukeboxScript::c_InformationCenter_Music);
bgRed->Enable(TRUE); bgRed->Enable(TRUE);
PlayAction(InfomainScript::c_iic043in_RunAnim); PlayAction(InfomainScript::c_iic043in_RunAnim);
FUN_10015820(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen); Disable(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen);
return; return;
} }
@ -545,7 +545,7 @@ void Infocenter::ReadyWorld()
PlayAction(script); PlayAction(script);
InputManager()->DisableInputProcessing(); InputManager()->DisableInputProcessing();
FUN_10015820(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen); Disable(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen);
return; return;
} }
@ -560,7 +560,7 @@ void Infocenter::ReadyWorld()
PlayMusic(JukeboxScript::c_InformationCenter_Music); PlayMusic(JukeboxScript::c_InformationCenter_Music);
bgRed->Enable(TRUE); bgRed->Enable(TRUE);
PlayAction(InfomainScript::c_iic043in_RunAnim); PlayAction(InfomainScript::c_iic043in_RunAnim);
FUN_10015820(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen); Disable(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen);
return; return;
} }
@ -597,7 +597,7 @@ void Infocenter::ReadyWorld()
PlayAction(script); PlayAction(script);
InputManager()->DisableInputProcessing(); InputManager()->DisableInputProcessing();
FUN_10015820(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen); Disable(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen);
return; return;
} }
@ -610,7 +610,7 @@ void Infocenter::ReadyWorld()
} }
m_infocenterState->m_unk0x74 = 11; m_infocenterState->m_unk0x74 = 11;
FUN_10015820(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen); Disable(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen);
} }
// FUNCTION: LEGO1 0x1006f9a0 // FUNCTION: LEGO1 0x1006f9a0
@ -1291,7 +1291,7 @@ void Infocenter::StopCutscene()
VideoManager()->EnableFullScreenMovie(FALSE); VideoManager()->EnableFullScreenMovie(FALSE);
InputManager()->SetUnknown335(FALSE); InputManager()->SetUnknown335(FALSE);
SetAppCursor(e_cursorArrow); SetAppCursor(e_cursorArrow);
FUN_10015820(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen); Disable(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen);
} }
// FUNCTION: LEGO1 0x10070d00 // FUNCTION: LEGO1 0x10070d00

View File

@ -88,7 +88,7 @@ void InfocenterDoor::ReadyWorld()
{ {
LegoWorld::ReadyWorld(); LegoWorld::ReadyWorld();
PlayMusic(JukeboxScript::c_InformationCenter_Music); PlayMusic(JukeboxScript::c_InformationCenter_Music);
FUN_10015820(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen); Disable(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen);
} }
// FUNCTION: LEGO1 0x10037a90 // FUNCTION: LEGO1 0x10037a90

View File

@ -142,11 +142,11 @@ MxLong Isle::Notify(MxParam& p_param)
break; break;
case c_notificationButtonUp: case c_notificationButtonUp:
case c_notificationButtonDown: case c_notificationButtonDown:
switch (m_act1state->m_unk0x018) { switch (m_act1state->m_state) {
case 3: case Act1State::e_pizza:
result = m_pizza->Notify(p_param); result = m_pizza->Notify(p_param);
break; break;
case 10: case Act1State::e_ambulance:
result = m_ambulance->Notify(p_param); result = m_ambulance->Notify(p_param);
break; break;
} }
@ -155,14 +155,14 @@ MxLong Isle::Notify(MxParam& p_param)
result = HandleControl((LegoControlManagerNotificationParam&) p_param); result = HandleControl((LegoControlManagerNotificationParam&) p_param);
break; break;
case c_notificationEndAnim: case c_notificationEndAnim:
switch (m_act1state->m_unk0x018) { switch (m_act1state->m_state) {
case 4: case Act1State::e_helicopter:
result = UserActor()->Notify(p_param); result = UserActor()->Notify(p_param);
break; break;
case 8: case Act1State::e_towtrack:
result = m_towtrack->Notify(p_param); result = m_towtrack->Notify(p_param);
break; break;
case 10: case Act1State::e_ambulance:
result = m_ambulance->Notify(p_param); result = m_ambulance->Notify(p_param);
break; break;
} }
@ -187,18 +187,18 @@ MxLong Isle::HandleEndAction(MxEndActionNotificationParam& p_param)
{ {
MxLong result; MxLong result;
switch (m_act1state->m_unk0x018) { switch (m_act1state->m_state) {
case 2: case Act1State::e_elevator:
HandleElevatorEndAction(); HandleElevatorEndAction();
result = 1; result = 1;
break; break;
case 3: case Act1State::e_pizza:
result = m_pizza->Notify(p_param); result = m_pizza->Notify(p_param);
break; break;
case 8: case Act1State::e_towtrack:
result = m_towtrack->Notify(p_param); result = m_towtrack->Notify(p_param);
break; break;
case 10: case Act1State::e_ambulance:
result = m_ambulance->Notify(p_param); result = m_ambulance->Notify(p_param);
break; break;
default: default:
@ -241,12 +241,12 @@ void Isle::HandleElevatorEndAction()
case Act1State::c_floor1: case Act1State::c_floor1:
m_destLocation = LegoGameState::e_infomain; m_destLocation = LegoGameState::e_infomain;
TransitionManager()->StartTransition(MxTransitionManager::e_mosaic, 50, FALSE, FALSE); TransitionManager()->StartTransition(MxTransitionManager::e_mosaic, 50, FALSE, FALSE);
m_act1state->m_unk0x018 = 0; m_act1state->m_state = Act1State::e_none;
break; break;
case Act1State::c_floor2: case Act1State::c_floor2:
if (m_act1state->m_unk0x01e) { if (m_act1state->m_unk0x01e) {
m_act1state->m_unk0x01e = FALSE; m_act1state->m_unk0x01e = FALSE;
m_act1state->m_unk0x018 = 0; m_act1state->m_state = Act1State::e_none;
InputManager()->EnableInputProcessing(); InputManager()->EnableInputProcessing();
} }
else { else {
@ -258,7 +258,7 @@ void Isle::HandleElevatorEndAction()
case Act1State::c_floor3: case Act1State::c_floor3:
m_destLocation = LegoGameState::e_elevopen; m_destLocation = LegoGameState::e_elevopen;
TransitionManager()->StartTransition(MxTransitionManager::e_mosaic, 50, FALSE, FALSE); TransitionManager()->StartTransition(MxTransitionManager::e_mosaic, 50, FALSE, FALSE);
m_act1state->m_unk0x018 = 0; m_act1state->m_state = Act1State::e_none;
break; break;
} }
} }
@ -270,14 +270,14 @@ void Isle::ReadyWorld()
if (m_act1state->GetUnknown21()) { if (m_act1state->GetUnknown21()) {
GameState()->SwitchArea(LegoGameState::e_infomain); GameState()->SwitchArea(LegoGameState::e_infomain);
m_act1state->SetUnknown18(0); m_act1state->SetState(Act1State::e_none);
m_act1state->SetUnknown21(0); m_act1state->SetUnknown21(0);
} }
else if (GameState()->GetLoadedAct() != LegoGameState::e_act1) { else if (GameState()->GetLoadedAct() != LegoGameState::e_act1) {
EnableAnimations(TRUE); EnableAnimations(TRUE);
FUN_10032620(); FUN_10032620();
m_act1state->PlaceActors(); m_act1state->PlaceActors();
FUN_10015820(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen); Disable(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen);
} }
} }
@ -289,7 +289,7 @@ MxLong Isle::HandleControl(LegoControlManagerNotificationParam& p_param)
switch (p_param.m_clickedObjectId) { switch (p_param.m_clickedObjectId) {
case IsleScript::c_ElevRide_Info_Ctl: case IsleScript::c_ElevRide_Info_Ctl:
m_act1state->m_unk0x018 = 2; m_act1state->m_state = Act1State::e_elevator;
switch (m_act1state->m_elevFloor) { switch (m_act1state->m_elevFloor) {
case Act1State::c_floor1: case Act1State::c_floor1:
@ -309,7 +309,7 @@ MxLong Isle::HandleControl(LegoControlManagerNotificationParam& p_param)
m_act1state->m_elevFloor = Act1State::c_floor1; m_act1state->m_elevFloor = Act1State::c_floor1;
break; break;
case IsleScript::c_ElevRide_Two_Ctl: case IsleScript::c_ElevRide_Two_Ctl:
m_act1state->m_unk0x018 = 2; m_act1state->m_state = Act1State::e_elevator;
switch (m_act1state->m_elevFloor) { switch (m_act1state->m_elevFloor) {
case Act1State::c_floor1: case Act1State::c_floor1:
@ -329,7 +329,7 @@ MxLong Isle::HandleControl(LegoControlManagerNotificationParam& p_param)
m_act1state->m_elevFloor = Act1State::c_floor2; m_act1state->m_elevFloor = Act1State::c_floor2;
break; break;
case IsleScript::c_ElevRide_Three_Ctl: case IsleScript::c_ElevRide_Three_Ctl:
m_act1state->m_unk0x018 = 2; m_act1state->m_state = Act1State::e_elevator;
switch (m_act1state->m_elevFloor) { switch (m_act1state->m_elevFloor) {
case Act1State::c_floor1: case Act1State::c_floor1:
@ -478,14 +478,14 @@ MxLong Isle::HandlePathStruct(LegoPathStructNotificationParam& p_param)
} }
} }
switch (m_act1state->m_unk0x018) { switch (m_act1state->m_state) {
case 3: case Act1State::e_pizza:
result = m_pizza->Notify(p_param); result = m_pizza->Notify(p_param);
break; break;
case 8: case Act1State::e_towtrack:
result = m_towtrack->Notify(p_param); result = m_towtrack->Notify(p_param);
break; break;
case 10: case Act1State::e_ambulance:
result = m_ambulance->Notify(p_param); result = m_ambulance->Notify(p_param);
break; break;
} }
@ -502,7 +502,7 @@ MxLong Isle::HandlePathStruct(LegoPathStructNotificationParam& p_param)
result = 1; result = 1;
break; break;
case 0x131: case 0x131:
if (m_act1state->m_unk0x018 != 10) { if (m_act1state->m_state != Act1State::e_ambulance) {
AnimationManager()->FUN_10064740(NULL); AnimationManager()->FUN_10064740(NULL);
} }
result = 1; result = 1;
@ -558,7 +558,7 @@ void Isle::Enable(MxBool p_enable)
EnableAnimations(TRUE); EnableAnimations(TRUE);
if (m_act1state->m_unk0x018 == 0) { if (m_act1state->m_state == Act1State::e_none) {
MxS32 locations[] = {1, 2, 3, 4, 5, 6, 7, 8, 9, 10}; MxS32 locations[] = {1, 2, 3, 4, 5, 6, 7, 8, 9, 10};
for (MxU32 i = 0; i < 5; i++) { for (MxU32 i = 0; i < 5; i++) {
@ -595,7 +595,7 @@ void Isle::Enable(MxBool p_enable)
break; break;
case LegoGameState::e_jetrace2: case LegoGameState::e_jetrace2:
if (((JetskiRaceState*) GameState()->GetState("JetskiRaceState"))->m_unk0x28 == 2) { if (((JetskiRaceState*) GameState()->GetState("JetskiRaceState"))->m_unk0x28 == 2) {
m_act1state->m_unk0x018 = 5; m_act1state->m_state = Act1State::e_transitionToJetski;
} }
PlaceActor(UserActor()); PlaceActor(UserActor());
@ -704,10 +704,10 @@ void Isle::Enable(MxBool p_enable)
break; break;
} }
switch (m_act1state->m_unk0x018) { switch (m_act1state->m_state) {
case 0: case Act1State::e_none:
case 1: case Act1State::e_initial:
m_act1state->m_unk0x018 = 0; m_act1state->m_state = Act1State::e_none;
if (GameState()->m_currentArea == LegoGameState::e_pizzeriaExterior) { if (GameState()->m_currentArea == LegoGameState::e_pizzeriaExterior) {
AnimationManager()->FUN_10064740(NULL); AnimationManager()->FUN_10064740(NULL);
@ -728,7 +728,7 @@ void Isle::Enable(MxBool p_enable)
} }
} }
break; break;
case 5: { case Act1State::e_transitionToJetski: {
((IslePathActor*) UserActor()) ((IslePathActor*) UserActor())
->SpawnPlayer( ->SpawnPlayer(
LegoGameState::e_jetrace2, LegoGameState::e_jetrace2,
@ -756,12 +756,12 @@ void Isle::Enable(MxBool p_enable)
->FUN_10060dc0(script, NULL, TRUE, LegoAnimationManager::e_unk1, NULL, FALSE, FALSE, TRUE, FALSE); ->FUN_10060dc0(script, NULL, TRUE, LegoAnimationManager::e_unk1, NULL, FALSE, FALSE, TRUE, FALSE);
} }
m_act1state->m_unk0x018 = 0; m_act1state->m_state = Act1State::e_none;
EnableAnimations(FALSE); EnableAnimations(FALSE);
AnimationManager()->FUN_10064670(NULL); AnimationManager()->FUN_10064670(NULL);
break; break;
} }
case 6: { case Act1State::e_transitionToRacecar: {
GameState()->m_currentArea = LegoGameState::e_carraceExterior; GameState()->m_currentArea = LegoGameState::e_carraceExterior;
((IslePathActor*) UserActor()) ((IslePathActor*) UserActor())
->SpawnPlayer( ->SpawnPlayer(
@ -790,21 +790,21 @@ void Isle::Enable(MxBool p_enable)
->FUN_10060dc0(script, NULL, TRUE, LegoAnimationManager::e_unk1, NULL, FALSE, FALSE, TRUE, FALSE); ->FUN_10060dc0(script, NULL, TRUE, LegoAnimationManager::e_unk1, NULL, FALSE, FALSE, TRUE, FALSE);
} }
m_act1state->m_unk0x018 = 0; m_act1state->m_state = Act1State::e_none;
EnableAnimations(TRUE); EnableAnimations(TRUE);
break; break;
} }
case 7: case Act1State::e_transitionToTowtrack:
m_act1state->m_unk0x018 = 8; m_act1state->m_state = Act1State::e_towtrack;
AnimationManager()->FUN_1005f6d0(FALSE); AnimationManager()->FUN_1005f6d0(FALSE);
AnimationManager()->EnableCamAnims(FALSE); AnimationManager()->EnableCamAnims(FALSE);
g_isleFlags &= ~c_playMusic; g_isleFlags &= ~c_playMusic;
m_towtrack->FUN_1004dab0(); m_towtrack->Init();
break; break;
case 9: case Act1State::e_transitionToAmbulance:
m_act1state->m_unk0x018 = 10; m_act1state->m_state = Act1State::e_ambulance;
AnimationManager()->FUN_1005f6d0(FALSE); AnimationManager()->FUN_1005f6d0(FALSE);
AnimationManager()->EnableCamAnims(FALSE); AnimationManager()->EnableCamAnims(FALSE);
@ -813,7 +813,7 @@ void Isle::Enable(MxBool p_enable)
m_ambulance->Init(); m_ambulance->Init();
break; break;
case 11: case 11:
m_act1state->m_unk0x018 = 0; m_act1state->m_state = Act1State::e_none;
((IslePathActor*) UserActor()) ((IslePathActor*) UserActor())
->SpawnPlayer( ->SpawnPlayer(
LegoGameState::e_jukeboxExterior, LegoGameState::e_jukeboxExterior,
@ -828,18 +828,18 @@ void Isle::Enable(MxBool p_enable)
SetAppCursor(e_cursorArrow); SetAppCursor(e_cursorArrow);
if (m_act1state->m_unk0x018 != 8 && if (m_act1state->m_state != Act1State::e_towtrack &&
(m_act1state->m_unk0x018 != 0 || GameState()->m_currentArea != LegoGameState::e_elevride) && (m_act1state->m_state != Act1State::e_none || GameState()->m_currentArea != LegoGameState::e_elevride) &&
(m_act1state->m_unk0x018 != 0 || GameState()->m_currentArea != LegoGameState::e_polidoor) && (m_act1state->m_state != Act1State::e_none || GameState()->m_currentArea != LegoGameState::e_polidoor) &&
(m_act1state->m_unk0x018 != 0 || GameState()->m_currentArea != LegoGameState::e_garadoor) && (m_act1state->m_state != Act1State::e_none || GameState()->m_currentArea != LegoGameState::e_garadoor) &&
(m_act1state->m_unk0x018 != 0 || GameState()->m_currentArea != LegoGameState::e_bike) && (m_act1state->m_state != Act1State::e_none || GameState()->m_currentArea != LegoGameState::e_bike) &&
(m_act1state->m_unk0x018 != 0 || GameState()->m_currentArea != LegoGameState::e_dunecar) && (m_act1state->m_state != Act1State::e_none || GameState()->m_currentArea != LegoGameState::e_dunecar) &&
(m_act1state->m_unk0x018 != 0 || GameState()->m_currentArea != LegoGameState::e_motocycle) && (m_act1state->m_state != Act1State::e_none || GameState()->m_currentArea != LegoGameState::e_motocycle) &&
(m_act1state->m_unk0x018 != 0 || GameState()->m_currentArea != LegoGameState::e_copter) && (m_act1state->m_state != Act1State::e_none || GameState()->m_currentArea != LegoGameState::e_copter) &&
(m_act1state->m_unk0x018 != 0 || GameState()->m_currentArea != LegoGameState::e_jetski) && (m_act1state->m_state != Act1State::e_none || GameState()->m_currentArea != LegoGameState::e_jetski) &&
(m_act1state->m_unk0x018 != 0 || GameState()->m_currentArea != LegoGameState::e_skateboard) && (m_act1state->m_state != Act1State::e_none || GameState()->m_currentArea != LegoGameState::e_skateboard) &&
(m_act1state->m_unk0x018 != 0 || GameState()->m_currentArea != LegoGameState::e_jetrace2)) { (m_act1state->m_state != Act1State::e_none || GameState()->m_currentArea != LegoGameState::e_jetrace2)) {
FUN_10015820(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen); Disable(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen);
} }
SetROIVisible("stretch", FALSE); SetROIVisible("stretch", FALSE);
@ -897,7 +897,7 @@ MxLong Isle::HandleTransitionEnd()
DeleteObjects(&m_atomId, IsleScript::c_Avo900Ps_PlayWav, IsleScript::c_Avo907Ps_PlayWav); DeleteObjects(&m_atomId, IsleScript::c_Avo900Ps_PlayWav, IsleScript::c_Avo907Ps_PlayWav);
if (m_destLocation != LegoGameState::e_skateboard) { if (m_destLocation != LegoGameState::e_skateboard) {
m_act1state->m_unk0x018 = 0; m_act1state->m_state = Act1State::e_none;
} }
switch (m_destLocation) { switch (m_destLocation) {
@ -961,7 +961,7 @@ MxLong Isle::HandleTransitionEnd()
m_destLocation = LegoGameState::e_undefined; m_destLocation = LegoGameState::e_undefined;
VariableTable()->SetVariable("VISIBILITY", "Show Gas"); VariableTable()->SetVariable("VISIBILITY", "Show Gas");
AnimationManager()->Resume(); AnimationManager()->Resume();
FUN_10015820(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen); Disable(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen);
SetAppCursor(e_cursorArrow); SetAppCursor(e_cursorArrow);
SetIsWorldActive(TRUE); SetIsWorldActive(TRUE);
break; break;
@ -971,7 +971,7 @@ MxLong Isle::HandleTransitionEnd()
m_destLocation = LegoGameState::e_undefined; m_destLocation = LegoGameState::e_undefined;
VariableTable()->SetVariable("VISIBILITY", "Show Policsta"); VariableTable()->SetVariable("VISIBILITY", "Show Policsta");
AnimationManager()->Resume(); AnimationManager()->Resume();
FUN_10015820(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen); Disable(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen);
SetAppCursor(e_cursorArrow); SetAppCursor(e_cursorArrow);
SetIsWorldActive(TRUE); SetIsWorldActive(TRUE);
break; break;
@ -1023,7 +1023,7 @@ MxLong Isle::HandleTransitionEnd()
break; break;
case LegoGameState::e_ambulance: case LegoGameState::e_ambulance:
m_act1state->m_unk0x01f = TRUE; m_act1state->m_unk0x01f = TRUE;
m_act1state->m_unk0x018 = 10; m_act1state->m_state = Act1State::e_ambulance;
FUN_10032d30(IsleScript::c_AmbulanceFuelMeter, JukeboxScript::c_MusicTheme1, NULL, TRUE); FUN_10032d30(IsleScript::c_AmbulanceFuelMeter, JukeboxScript::c_MusicTheme1, NULL, TRUE);
if (!m_act1state->m_unk0x01f) { if (!m_act1state->m_unk0x01f) {
@ -1032,7 +1032,7 @@ MxLong Isle::HandleTransitionEnd()
break; break;
case LegoGameState::e_towtrack: case LegoGameState::e_towtrack:
m_act1state->m_unk0x01f = TRUE; m_act1state->m_unk0x01f = TRUE;
m_act1state->m_unk0x018 = 8; m_act1state->m_state = Act1State::e_towtrack;
FUN_10032d30(IsleScript::c_TowFuelMeter, JukeboxScript::c_MusicTheme1, NULL, TRUE); FUN_10032d30(IsleScript::c_TowFuelMeter, JukeboxScript::c_MusicTheme1, NULL, TRUE);
if (!m_act1state->m_unk0x01f) { if (!m_act1state->m_unk0x01f) {
@ -1082,7 +1082,7 @@ void Isle::FUN_10032d30(
VariableTable()->SetVariable(g_varCAMERALOCATION, p_cameraLocation); VariableTable()->SetVariable(g_varCAMERALOCATION, p_cameraLocation);
} }
FUN_10015820(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen); Disable(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen);
SetAppCursor(e_cursorArrow); SetAppCursor(e_cursorArrow);
m_destLocation = LegoGameState::e_undefined; m_destLocation = LegoGameState::e_undefined;
m_act1state->m_unk0x01f = FALSE; m_act1state->m_unk0x01f = FALSE;
@ -1167,7 +1167,7 @@ void Isle::CreateState()
m_act1state = (Act1State*) GameState()->GetState("Act1State"); m_act1state = (Act1State*) GameState()->GetState("Act1State");
if (!m_act1state) { if (!m_act1state) {
m_act1state = (Act1State*) GameState()->CreateState("Act1State"); m_act1state = (Act1State*) GameState()->CreateState("Act1State");
m_act1state->m_unk0x018 = 0; m_act1state->m_state = Act1State::e_none;
} }
m_radio.CreateState(); m_radio.CreateState();
@ -1193,20 +1193,20 @@ MxBool Isle::Escape()
m_radio.Stop(); m_radio.Stop();
BackgroundAudioManager()->Stop(); BackgroundAudioManager()->Stop();
switch (m_act1state->m_unk0x018) { switch (m_act1state->m_state) {
case 3: case Act1State::e_pizza:
if (UserActor() != NULL) { if (UserActor() != NULL) {
m_pizza->StopActions(); m_pizza->StopActions();
m_pizza->FUN_100382b0(); m_pizza->FUN_100382b0();
} }
break; break;
case 8: case Act1State::e_towtrack:
if (UserActor() != NULL && !UserActor()->IsA("TowTrack")) { if (UserActor() != NULL && !UserActor()->IsA("TowTrack")) {
m_towtrack->StopActions(); m_towtrack->StopActions();
m_towtrack->FUN_1004dbe0(); m_towtrack->Reset();
} }
break; break;
case 10: case Act1State::e_ambulance:
if (UserActor() != NULL && !UserActor()->IsA("Ambulance")) { if (UserActor() != NULL && !UserActor()->IsA("Ambulance")) {
m_ambulance->StopActions(); m_ambulance->StopActions();
m_ambulance->Reset(); m_ambulance->Reset();
@ -1239,7 +1239,7 @@ MxBool Isle::Escape()
VariableTable()->SetVariable("VISIBILITY", "Show Gas"); VariableTable()->SetVariable("VISIBILITY", "Show Gas");
} }
m_act1state->m_unk0x018 = 0; m_act1state->m_state = Act1State::e_none;
m_destLocation = LegoGameState::e_infomain; m_destLocation = LegoGameState::e_infomain;
return TRUE; return TRUE;
} }
@ -1247,21 +1247,21 @@ MxBool Isle::Escape()
// FUNCTION: LEGO1 0x10033350 // FUNCTION: LEGO1 0x10033350
void Isle::FUN_10033350() void Isle::FUN_10033350()
{ {
if (m_act1state->m_unk0x018 == 10) { if (m_act1state->m_state == Act1State::e_ambulance) {
if (UserActor() != NULL && !UserActor()->IsA("Ambulance")) { if (UserActor() != NULL && !UserActor()->IsA("Ambulance")) {
m_ambulance->StopActions(); m_ambulance->StopActions();
m_ambulance->Reset(); m_ambulance->Reset();
} }
} }
if (m_act1state->m_unk0x018 == 8) { if (m_act1state->m_state == Act1State::e_towtrack) {
if (UserActor() != NULL && !UserActor()->IsA("TowTrack")) { if (UserActor() != NULL && !UserActor()->IsA("TowTrack")) {
m_towtrack->StopActions(); m_towtrack->StopActions();
m_towtrack->FUN_1004dbe0(); m_towtrack->Reset();
} }
} }
if (m_act1state->m_unk0x018 == 3) { if (m_act1state->m_state == Act1State::e_pizza) {
if (UserActor() != NULL) { if (UserActor() != NULL) {
m_pizza->StopActions(); m_pizza->StopActions();
m_pizza->FUN_100382b0(); m_pizza->FUN_100382b0();
@ -1293,7 +1293,7 @@ void Isle::FUN_10033350()
Act1State::Act1State() Act1State::Act1State()
{ {
m_elevFloor = Act1State::c_floor1; m_elevFloor = Act1State::c_floor1;
m_unk0x018 = 1; m_state = Act1State::e_initial;
m_unk0x01e = FALSE; m_unk0x01e = FALSE;
m_cptClickDialogue = Playlist((MxU32*) g_cptClickDialogue, sizeOfArray(g_cptClickDialogue), Playlist::e_loop); m_cptClickDialogue = Playlist((MxU32*) g_cptClickDialogue, sizeOfArray(g_cptClickDialogue), Playlist::e_loop);
m_unk0x01f = FALSE; m_unk0x01f = FALSE;

View File

@ -211,7 +211,7 @@ MxBool JukeBox::HandleControl(LegoControlManagerNotificationParam& p_param)
break; break;
case JukeboxwScript::c_Note_Ctl: case JukeboxwScript::c_Note_Ctl:
Act1State* act1State = (Act1State*) GameState()->GetState("Act1State"); Act1State* act1State = (Act1State*) GameState()->GetState("Act1State");
act1State->m_unk0x018 = 11; act1State->m_state = Act1State::e_jukebox;
m_destLocation = LegoGameState::Area::e_jukeboxExterior; m_destLocation = LegoGameState::Area::e_jukeboxExterior;
TransitionManager()->StartTransition(MxTransitionManager::e_mosaic, 50, 0, FALSE); TransitionManager()->StartTransition(MxTransitionManager::e_mosaic, 50, 0, FALSE);
break; break;
@ -247,7 +247,7 @@ MxResult JukeBox::Tickle()
if (m_unk0x100 == 1) { if (m_unk0x100 == 1) {
m_unk0x100 = 0; m_unk0x100 = 0;
FUN_10015820(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen); Disable(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen);
} }
return SUCCESS; return SUCCESS;

View File

@ -192,7 +192,7 @@ MxResult LegoAct2::Tickle()
case 2: case 2:
if (g_unk0x100f4474) { if (g_unk0x100f4474) {
if (AnimationManager()->FUN_10064ee0(g_unk0x100f4474)) { if (AnimationManager()->FUN_10064ee0(g_unk0x100f4474)) {
FUN_10015820(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen); Disable(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen);
g_unk0x100f4474 = (Act2mainScript::Script) 0; g_unk0x100f4474 = (Act2mainScript::Script) 0;
} }
} }
@ -200,7 +200,7 @@ MxResult LegoAct2::Tickle()
m_unk0x10d0 += 50; m_unk0x10d0 += 50;
break; break;
case 3: case 3:
FUN_10015820(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen); Disable(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen);
m_unk0x10d0 = 0; m_unk0x10d0 = 0;
m_unk0x10c4 = 4; m_unk0x10c4 = 4;
FUN_10052560(Act2mainScript::c_tja009ni_RunAnim, TRUE, TRUE, NULL, NULL, NULL); FUN_10052560(Act2mainScript::c_tja009ni_RunAnim, TRUE, TRUE, NULL, NULL, NULL);
@ -540,7 +540,7 @@ void LegoAct2::Enable(MxBool p_enable)
GameState()->StopArea(LegoGameState::e_infomain); GameState()->StopArea(LegoGameState::e_infomain);
} }
FUN_10015820(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen); Disable(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen);
if (m_unk0x10c4 != 6 && m_unk0x10c4 != 12) { if (m_unk0x10c4 != 6 && m_unk0x10c4 != 12) {
PlayMusic(m_music); PlayMusic(m_music);

View File

@ -95,7 +95,7 @@ void Police::ReadyWorld()
{ {
LegoWorld::ReadyWorld(); LegoWorld::ReadyWorld();
PlayMusic(JukeboxScript::c_PoliceStation_Music); PlayMusic(JukeboxScript::c_PoliceStation_Music);
FUN_10015820(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen); Disable(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen);
} }
// FUNCTION: LEGO1 0x1005e550 // FUNCTION: LEGO1 0x1005e550

View File

@ -159,7 +159,7 @@ void Score::ReadyWorld()
PlayMusic(JukeboxScript::c_InformationCenter_Music); PlayMusic(JukeboxScript::c_InformationCenter_Music);
} }
FUN_10015820(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen); Disable(FALSE, LegoOmni::c_disableInput | LegoOmni::c_disable3d | LegoOmni::c_clearScreen);
} }
// FUNCTION: LEGO1 0x100016d0 // FUNCTION: LEGO1 0x100016d0

View File

@ -524,7 +524,7 @@ LegoAnimNodeData::LegoAnimNodeData()
m_translationKeys = NULL; m_translationKeys = NULL;
m_roiIndex = 0; m_roiIndex = 0;
m_rotationKeys = NULL; m_rotationKeys = NULL;
m_unk0x22 = 0; m_boundaryIndex = 0;
m_scaleKeys = NULL; m_scaleKeys = NULL;
m_morphKeys = NULL; m_morphKeys = NULL;
m_translationIndex = 0; m_translationIndex = 0;

View File

@ -190,7 +190,7 @@ class LegoAnimNodeData : public LegoTreeNodeData {
LegoU16 GetROIIndex() { return m_roiIndex; } LegoU16 GetROIIndex() { return m_roiIndex; }
// FUNCTION: BETA10 0x1005d5c0 // FUNCTION: BETA10 0x1005d5c0
LegoU16 GetUnknown0x22() { return m_unk0x22; } LegoU16 GetBoundaryIndex() { return m_boundaryIndex; }
// FUNCTION: BETA10 0x10073b80 // FUNCTION: BETA10 0x10073b80
LegoRotationKey* GetRotationKey(MxS32 index) { return &m_rotationKeys[index]; } LegoRotationKey* GetRotationKey(MxS32 index) { return &m_rotationKeys[index]; }
@ -217,7 +217,7 @@ class LegoAnimNodeData : public LegoTreeNodeData {
void SetROIIndex(LegoU16 p_roiIndex) { m_roiIndex = p_roiIndex; } void SetROIIndex(LegoU16 p_roiIndex) { m_roiIndex = p_roiIndex; }
// FUNCTION: BETA10 0x1005f2e0 // FUNCTION: BETA10 0x1005f2e0
void SetUnknown0x22(LegoU16 p_unk0x22) { m_unk0x22 = p_unk0x22; } void SetBoundaryIndex(LegoU16 p_boundaryIndex) { m_boundaryIndex = p_boundaryIndex; }
LegoResult CreateLocalTransform(LegoTime p_time, Matrix4& p_matrix) LegoResult CreateLocalTransform(LegoTime p_time, Matrix4& p_matrix)
{ {
@ -280,7 +280,7 @@ class LegoAnimNodeData : public LegoTreeNodeData {
LegoScaleKey* m_scaleKeys; // 0x18 LegoScaleKey* m_scaleKeys; // 0x18
LegoMorphKey* m_morphKeys; // 0x1c LegoMorphKey* m_morphKeys; // 0x1c
LegoU16 m_roiIndex; // 0x20 LegoU16 m_roiIndex; // 0x20
LegoU16 m_unk0x22; // 0x22 LegoU16 m_boundaryIndex; // 0x22
LegoU32 m_translationIndex; // 0x24 LegoU32 m_translationIndex; // 0x24
LegoU32 m_rotationIndex; // 0x28 LegoU32 m_rotationIndex; // 0x28
LegoU32 m_scaleIndex; // 0x2c LegoU32 m_scaleIndex; // 0x2c

View File

@ -59,4 +59,12 @@ inline T RToD(T p_r)
return p_r * 180.0F / 3.1416F; return p_r * 180.0F / 3.1416F;
} }
template <class... Ts>
struct overloaded : Ts... {
using Ts::operator()...;
};
template <class... Ts>
overloaded(Ts...) -> overloaded<Ts...>;
#endif // __LEGOUTIL_H #endif // __LEGOUTIL_H

View File

@ -56,7 +56,7 @@ int LegoDeviceEnumerate::ProcessDeviceBytes(int p_deviceNum, GUID& p_guid)
GUID4 deviceGuid; GUID4 deviceGuid;
memcpy(&deviceGuid, &p_guid, sizeof(GUID4)); memcpy(&deviceGuid, &p_guid, sizeof(GUID4));
for (list<MxDriver>::iterator it = m_list.begin(); it != m_list.end(); it++, i++) { for (list<MxDriver>::iterator it = m_ddInfo.begin(); it != m_ddInfo.end(); it++, i++) {
if (p_deviceNum >= 0 && p_deviceNum < i) { if (p_deviceNum >= 0 && p_deviceNum < i) {
return -1; return -1;
} }
@ -91,7 +91,7 @@ int LegoDeviceEnumerate::GetDevice(int p_deviceNum, MxDriver*& p_driver, Direct3
int i = 0; int i = 0;
for (list<MxDriver>::iterator it = m_list.begin(); it != m_list.end(); it++) { for (list<MxDriver>::iterator it = m_ddInfo.begin(); it != m_ddInfo.end(); it++) {
p_driver = &*it; p_driver = &*it;
for (list<Direct3DDeviceInfo>::iterator it2 = p_driver->m_devices.begin(); it2 != p_driver->m_devices.end(); for (list<Direct3DDeviceInfo>::iterator it2 = p_driver->m_devices.begin(); it2 != p_driver->m_devices.end();
@ -115,7 +115,7 @@ int LegoDeviceEnumerate::FormatDeviceName(char* p_buffer, const MxDriver* p_ddIn
int number = 0; int number = 0;
assert(p_ddInfo && p_d3dInfo); assert(p_ddInfo && p_d3dInfo);
for (list<MxDriver>::const_iterator it = m_list.begin(); it != m_list.end(); it++, number++) { for (list<MxDriver>::const_iterator it = m_ddInfo.begin(); it != m_ddInfo.end(); it++, number++) {
if (&(*it) == p_ddInfo) { if (&(*it) == p_ddInfo) {
GUID4 guid; GUID4 guid;
memcpy(&guid, p_d3dInfo->m_guid, sizeof(GUID4)); memcpy(&guid, p_d3dInfo->m_guid, sizeof(GUID4));
@ -138,7 +138,7 @@ int LegoDeviceEnumerate::BETA_1011cc65(int p_idx, char* p_buffer)
int i = 0; int i = 0;
int j = 0; int j = 0;
for (list<MxDriver>::iterator it = m_list.begin(); it != m_list.end(); it++, i++) { for (list<MxDriver>::iterator it = m_ddInfo.begin(); it != m_ddInfo.end(); it++, i++) {
MxDriver& driver = *it; MxDriver& driver = *it;
for (list<Direct3DDeviceInfo>::iterator it2 = driver.m_devices.begin(); it2 != driver.m_devices.end(); it2++) { for (list<Direct3DDeviceInfo>::iterator it2 = driver.m_devices.begin(); it2 != driver.m_devices.end(); it2++) {
@ -165,14 +165,14 @@ int LegoDeviceEnumerate::GetBestDevice()
return -1; return -1;
} }
if (m_list.size() == 0) { if (m_ddInfo.size() == 0) {
return -1; return -1;
} }
int j = 0; int j = 0;
int k = -1; int k = -1;
for (list<MxDriver>::iterator it = m_list.begin(); it != m_list.end(); it++) { for (list<MxDriver>::iterator it = m_ddInfo.begin(); it != m_ddInfo.end(); it++) {
MxDriver& driver = *it; MxDriver& driver = *it;
for (list<Direct3DDeviceInfo>::iterator it2 = driver.m_devices.begin(); it2 != driver.m_devices.end(); it2++) { for (list<Direct3DDeviceInfo>::iterator it2 = driver.m_devices.begin(); it2 != driver.m_devices.end(); it2++) {
@ -199,7 +199,7 @@ int LegoDeviceEnumerate::FUN_1009d210()
return -1; return -1;
} }
for (list<MxDriver>::iterator it = m_list.begin(); it != m_list.end();) { for (list<MxDriver>::iterator it = m_ddInfo.begin(); it != m_ddInfo.end();) {
MxDriver& driver = *it; MxDriver& driver = *it;
for (list<Direct3DDeviceInfo>::iterator it2 = driver.m_devices.begin(); it2 != driver.m_devices.end();) { for (list<Direct3DDeviceInfo>::iterator it2 = driver.m_devices.begin(); it2 != driver.m_devices.end();) {
@ -212,14 +212,14 @@ int LegoDeviceEnumerate::FUN_1009d210()
} }
if (!driver.m_devices.size()) { if (!driver.m_devices.size()) {
m_list.erase(it++); m_ddInfo.erase(it++);
} }
else { else {
it++; it++;
} }
} }
if (!m_list.size()) { if (!m_ddInfo.size()) {
return -1; return -1;
} }
@ -231,7 +231,7 @@ int LegoDeviceEnumerate::FUN_1009d210()
// FUNCTION: BETA10 0x1011d235 // FUNCTION: BETA10 0x1011d235
unsigned char LegoDeviceEnumerate::FUN_1009d3d0(Direct3DDeviceInfo& p_device) unsigned char LegoDeviceEnumerate::FUN_1009d3d0(Direct3DDeviceInfo& p_device)
{ {
if (m_list.size() <= 0) { if (m_ddInfo.size() <= 0) {
return FALSE; return FALSE;
} }
@ -246,7 +246,7 @@ unsigned char LegoDeviceEnumerate::FUN_1009d3d0(Direct3DDeviceInfo& p_device)
} }
} }
MxDriver& front = m_list.front(); MxDriver& front = m_ddInfo.front();
for (list<Direct3DDeviceInfo>::iterator it = front.m_devices.begin(); it != front.m_devices.end(); it++) { for (list<Direct3DDeviceInfo>::iterator it = front.m_devices.begin(); it != front.m_devices.end(); it++) {
if ((&*it) == &p_device) { if ((&*it) == &p_device) {
return TRUE; return TRUE;

View File

@ -234,7 +234,7 @@ BOOL MxDirect3D::SetDevice(MxDeviceEnumerate& p_deviceEnumerate, MxDriver* p_dri
assert(d); assert(d);
int i = 0; int i = 0;
for (list<MxDriver>::iterator it = p_deviceEnumerate.m_list.begin(); it != p_deviceEnumerate.m_list.end(); for (list<MxDriver>::iterator it = p_deviceEnumerate.m_ddInfo.begin(); it != p_deviceEnumerate.m_ddInfo.end();
it++, i++) { it++, i++) {
MxDriver& driver = *it; MxDriver& driver = *it;

View File

@ -30,13 +30,12 @@ MxAssignedDevice::~MxAssignedDevice()
} }
// FUNCTION: BETA10 0x1011d7f0 // FUNCTION: BETA10 0x1011d7f0
MxDriver::MxDriver(LPGUID p_guid) MxDriver::MxDriver()
{ {
m_guid = NULL; m_guid = NULL;
m_driverDesc = NULL; m_driverDesc = NULL;
m_driverName = NULL; m_driverName = NULL;
memset(&m_ddCaps, 0, sizeof(m_ddCaps)); memset(&m_ddCaps, 0, sizeof(m_ddCaps));
// TODO: ret vs ret 4
} }
// FUNCTION: CONFIG 0x00401180 // FUNCTION: CONFIG 0x00401180
@ -99,6 +98,12 @@ void MxDriver::Init(LPGUID p_guid, LPCSTR p_driverDesc, LPCSTR p_driverName)
} }
} }
// FUNCTION: BETA10 0x1011dba4
Direct3DDeviceInfo::Direct3DDeviceInfo()
{
memset(this, 0, sizeof(*this));
}
// FUNCTION: CONFIG 0x401420 // FUNCTION: CONFIG 0x401420
// FUNCTION: LEGO1 0x1009bd20 // FUNCTION: LEGO1 0x1009bd20
// FUNCTION: BETA10 0x1011dbd0 // FUNCTION: BETA10 0x1011dbd0
@ -198,7 +203,7 @@ MxDeviceEnumerate::~MxDeviceEnumerate()
BOOL MxDeviceEnumerate::EnumDirectDrawCallback(LPGUID p_guid, LPSTR p_driverDesc, LPSTR p_driverName) BOOL MxDeviceEnumerate::EnumDirectDrawCallback(LPGUID p_guid, LPSTR p_driverDesc, LPSTR p_driverName)
{ {
MxDriver driver(p_guid, p_driverDesc, p_driverName); MxDriver driver(p_guid, p_driverDesc, p_driverName);
m_list.push_back(driver); m_ddInfo.push_back(driver);
// Must be zeroed because held resources are copied by pointer only // Must be zeroed because held resources are copied by pointer only
// and should not be freed at the end of this function // and should not be freed at the end of this function
@ -209,46 +214,47 @@ BOOL MxDeviceEnumerate::EnumDirectDrawCallback(LPGUID p_guid, LPSTR p_driverDesc
LPDIRECT3D2 lpDirect3d2 = NULL; LPDIRECT3D2 lpDirect3d2 = NULL;
LPDIRECTDRAW lpDD = NULL; LPDIRECTDRAW lpDD = NULL;
MxDriver& newDevice = m_list.back(); MxDriver& newDevice = m_ddInfo.back();
HRESULT result = DirectDrawCreate(newDevice.m_guid, &lpDD, NULL); HRESULT result = DirectDrawCreate(newDevice.m_guid, &lpDD, NULL);
if (result != DD_OK) { if (result != DD_OK) {
BuildErrorString("DirectDraw Create failed: %s\n", EnumerateErrorToString(result)); BuildErrorString("DirectDraw Create failed: %s\n", EnumerateErrorToString(result));
} goto done;
else {
result = lpDD->SetCooperativeLevel(m_hWnd, DDSCL_NORMAL);
if (result != DD_OK) {
BuildErrorString("SetCooperativeLevel failed: %s\n", EnumerateErrorToString(result));
}
else {
newDevice.m_ddCaps.dwSize = sizeof(newDevice.m_ddCaps);
result = lpDD->GetCaps(&newDevice.m_ddCaps, NULL);
if (result != DD_OK) {
BuildErrorString("GetCaps failed: %s\n", EnumerateErrorToString(result));
}
else {
result = lpDD->QueryInterface(IID_IDirect3D2, (LPVOID*) &lpDirect3d2);
if (result != DD_OK) {
BuildErrorString("D3D creation failed: %s\n", EnumerateErrorToString(result));
}
else {
result = lpDirect3d2->EnumDevices(DevicesEnumerateCallback, this);
if (result != DD_OK) {
BuildErrorString("D3D enum devices failed: %s\n", EnumerateErrorToString(result));
}
else {
if (!newDevice.m_devices.size()) {
m_list.pop_back();
}
}
}
}
}
} }
result = lpDD->SetCooperativeLevel(m_hWnd, DDSCL_NORMAL);
if (result != DD_OK) {
BuildErrorString("SetCooperativeLevel failed: %s\n", EnumerateErrorToString(result));
goto done;
}
newDevice.m_ddCaps.dwSize = sizeof(newDevice.m_ddCaps);
result = lpDD->GetCaps(&newDevice.m_ddCaps, NULL);
if (result != DD_OK) {
BuildErrorString("GetCaps failed: %s\n", EnumerateErrorToString(result));
goto done;
}
result = lpDD->QueryInterface(IID_IDirect3D2, (LPVOID*) &lpDirect3d2);
if (result != DD_OK) {
BuildErrorString("D3D creation failed: %s\n", EnumerateErrorToString(result));
goto done;
}
result = lpDirect3d2->EnumDevices(DevicesEnumerateCallback, this);
if (result != DD_OK) {
BuildErrorString("D3D enum devices failed: %s\n", EnumerateErrorToString(result));
goto done;
}
if (!newDevice.m_devices.size()) {
m_ddInfo.pop_back();
}
done:
if (lpDirect3d2) { if (lpDirect3d2) {
lpDirect3d2->Release(); lpDirect3d2->Release();
} }
@ -281,15 +287,14 @@ HRESULT CALLBACK MxDeviceEnumerate::DevicesEnumerateCallback(
LPSTR p_deviceName, LPSTR p_deviceName,
LPD3DDEVICEDESC p_HWDesc, LPD3DDEVICEDESC p_HWDesc,
LPD3DDEVICEDESC p_HELDesc, LPD3DDEVICEDESC p_HELDesc,
LPVOID p_context LPVOID p_d
) )
{ {
if (p_context == NULL) { if (p_d == NULL) {
assert(0); assert(0);
} }
return ((MxDeviceEnumerate*) p_context) return ((MxDeviceEnumerate*) p_d)->EnumDevicesCallback(p_guid, p_deviceDesc, p_deviceName, p_HWDesc, p_HELDesc);
->EnumDevicesCallback(p_guid, p_deviceDesc, p_deviceName, p_HWDesc, p_HELDesc);
} }
// FUNCTION: CONFIG 0x00401cd0 // FUNCTION: CONFIG 0x00401cd0
@ -304,7 +309,7 @@ HRESULT MxDeviceEnumerate::EnumDevicesCallback(
) )
{ {
Direct3DDeviceInfo device(p_guid, p_deviceDesc, p_deviceName, p_HWDesc, p_HELDesc); Direct3DDeviceInfo device(p_guid, p_deviceDesc, p_deviceName, p_HWDesc, p_HELDesc);
m_list.back().m_devices.push_back(device); m_ddInfo.back().m_devices.push_back(device);
memset(&device, 0, sizeof(device)); memset(&device, 0, sizeof(device));
return DDENUMRET_OK; return DDENUMRET_OK;
} }
@ -334,13 +339,10 @@ int MxDeviceEnumerate::DoEnumerate(HWND hWnd)
// FUNCTION: LEGO1 0x1009c710 // FUNCTION: LEGO1 0x1009c710
// FUNCTION: BETA10 0x1011e476 // FUNCTION: BETA10 0x1011e476
BOOL CALLBACK BOOL CALLBACK
MxDeviceEnumerate::DirectDrawEnumerateCallback(LPGUID p_guid, LPSTR p_driverDesc, LPSTR p_driverName, LPVOID p_context) MxDeviceEnumerate::DirectDrawEnumerateCallback(LPGUID p_guid, LPSTR p_driverDesc, LPSTR p_driverName, LPVOID p_d)
{ {
if (p_context == NULL) { assert(p_d);
assert(0); return ((MxDeviceEnumerate*) p_d)->EnumDirectDrawCallback(p_guid, p_driverDesc, p_driverName);
}
return ((MxDeviceEnumerate*) p_context)->EnumDirectDrawCallback(p_guid, p_driverDesc, p_driverName);
} }
// FUNCTION: CONFIG 0x00401e30 // FUNCTION: CONFIG 0x00401e30
@ -351,79 +353,123 @@ const char* MxDeviceEnumerate::EnumerateErrorToString(HRESULT p_error)
switch (p_error) { switch (p_error) {
case DD_OK: case DD_OK:
return "No error."; return "No error.";
case DDERR_GENERIC:
return "Generic failure.";
case DDERR_UNSUPPORTED:
return "Action not supported.";
case DDERR_INVALIDPARAMS:
return "One or more of the parameters passed to the function are incorrect.";
case DDERR_OUTOFMEMORY:
return "DirectDraw does not have enough memory to perform the operation.";
case DDERR_CANNOTATTACHSURFACE:
return "This surface can not be attached to the requested surface.";
case DDERR_ALREADYINITIALIZED: case DDERR_ALREADYINITIALIZED:
return "This object is already initialized."; return "This object is already initialized.";
case DDERR_CURRENTLYNOTAVAIL: case DDERR_BLTFASTCANTCLIP:
return "Support is currently not available."; return "Return if a clipper object is attached to the source surface passed into a BltFast call.";
case DDERR_CANNOTATTACHSURFACE:
return "This surface can not be attached to the requested surface.";
case DDERR_CANNOTDETACHSURFACE: case DDERR_CANNOTDETACHSURFACE:
return "This surface can not be detached from the requested surface."; return "This surface can not be detached from the requested surface.";
case DDERR_HEIGHTALIGN: case DDERR_CANTCREATEDC:
return "Height of rectangle provided is not a multiple of reqd alignment."; return "Windows can not create any more DCs.";
case DDERR_CANTDUPLICATE:
return "Can't duplicate primary & 3D surfaces, or surfaces that are implicitly created.";
case DDERR_CLIPPERISUSINGHWND:
return "An attempt was made to set a cliplist for a clipper object that is already monitoring an hwnd.";
case DDERR_COLORKEYNOTSET:
return "No src color key specified for this operation.";
case DDERR_CURRENTLYNOTAVAIL:
return "Support is currently not available.";
case DDERR_DIRECTDRAWALREADYCREATED:
return "A DirectDraw object representing this driver has already been created for this process.";
case DDERR_EXCEPTION: case DDERR_EXCEPTION:
return "An exception was encountered while performing the requested operation."; return "An exception was encountered while performing the requested operation.";
case DDERR_INVALIDCAPS: case DDERR_EXCLUSIVEMODEALREADYSET:
return "One or more of the caps bits passed to the callback are incorrect."; return "An attempt was made to set the cooperative level when it was already set to exclusive.";
case DDERR_GENERIC:
return "Generic failure.";
case DDERR_HEIGHTALIGN:
return "Height of rectangle provided is not a multiple of reqd alignment.";
case DDERR_HWNDALREADYSET:
return "The CooperativeLevel HWND has already been set. It can not be reset while the process has surfaces or "
"palettes created.";
case DDERR_HWNDSUBCLASSED:
return "HWND used by DirectDraw CooperativeLevel has been subclassed, this prevents DirectDraw from restoring "
"state.";
case DDERR_IMPLICITLYCREATED:
return "This surface can not be restored because it is an implicitly created surface.";
case DDERR_INCOMPATIBLEPRIMARY: case DDERR_INCOMPATIBLEPRIMARY:
return "Unable to match primary surface creation request with existing primary surface."; return "Unable to match primary surface creation request with existing primary surface.";
case DDERR_INVALIDMODE: case DDERR_INVALIDCAPS:
return "DirectDraw does not support the requested mode."; return "One or more of the caps bits passed to the callback are incorrect.";
case DDERR_INVALIDCLIPLIST: case DDERR_INVALIDCLIPLIST:
return "DirectDraw does not support the provided cliplist."; return "DirectDraw does not support the provided cliplist.";
case DDERR_INVALIDPIXELFORMAT: case DDERR_INVALIDDIRECTDRAWGUID:
return "The pixel format was invalid as specified."; return "The GUID passed to DirectDrawCreate is not a valid DirectDraw driver identifier.";
case DDERR_INVALIDMODE:
return "DirectDraw does not support the requested mode.";
case DDERR_INVALIDOBJECT: case DDERR_INVALIDOBJECT:
return "DirectDraw received a pointer that was an invalid DIRECTDRAW object."; return "DirectDraw received a pointer that was an invalid DIRECTDRAW object.";
case DDERR_LOCKEDSURFACES: case DDERR_INVALIDPARAMS:
return "Operation could not be carried out because one or more surfaces are locked."; return "One or more of the parameters passed to the function are incorrect.";
case DDERR_INVALIDPIXELFORMAT:
return "The pixel format was invalid as specified.";
case DDERR_INVALIDPOSITION:
return "Returned when the position of the overlay on the destination is no longer legal for that "
"destination.";
case DDERR_INVALIDRECT: case DDERR_INVALIDRECT:
return "Rectangle provided was invalid."; return "Rectangle provided was invalid.";
case DDERR_LOCKEDSURFACES:
return "Operation could not be carried out because one or more surfaces are locked.";
case DDERR_NO3D:
return "There is no 3D present.";
case DDERR_NOALPHAHW: case DDERR_NOALPHAHW:
return "Operation could not be carried out because there is no alpha accleration hardware present or " return "Operation could not be carried out because there is no alpha accleration hardware present or "
"available."; "available.";
case DDERR_NO3D: case DDERR_NOBLTHW:
return "There is no 3D present."; return "No blitter hardware present.";
case DDERR_NOCOLORCONVHW:
return "Operation could not be carried out because there is no color conversion hardware present or available.";
case DDERR_NOCLIPLIST: case DDERR_NOCLIPLIST:
return "No cliplist available."; return "No cliplist available.";
case DDERR_NOCLIPPERATTACHED:
return "No clipper object attached to surface object.";
case DDERR_NOCOLORCONVHW:
return "Operation could not be carried out because there is no color conversion hardware present or "
"available.";
case DDERR_NOCOLORKEY: case DDERR_NOCOLORKEY:
return "Surface doesn't currently have a color key"; return "Surface doesn't currently have a color key";
case DDERR_NOCOLORKEYHW:
return "Operation could not be carried out because there is no hardware support of the destination color "
"key.";
case DDERR_NOCOOPERATIVELEVELSET: case DDERR_NOCOOPERATIVELEVELSET:
return "Create function called without DirectDraw object method SetCooperativeLevel being called."; return "Create function called without DirectDraw object method SetCooperativeLevel being called.";
case DDERR_NODC:
return "No DC was ever created for this surface.";
case DDERR_NODDROPSHW:
return "No DirectDraw ROP hardware.";
case DDERR_NODIRECTDRAWHW:
return "A hardware-only DirectDraw object creation was attempted but the driver did not support any "
"hardware.";
case DDERR_NOEMULATION:
return "Software emulation not available.";
case DDERR_NOEXCLUSIVEMODE: case DDERR_NOEXCLUSIVEMODE:
return "Operation requires the application to have exclusive mode but the application does not have exclusive " return "Operation requires the application to have exclusive mode but the application does not have exclusive "
"mode."; "mode.";
case DDERR_NOCOLORKEYHW:
return "Operation could not be carried out because there is no hardware support of the destination color key.";
case DDERR_NOGDI:
return "There is no GDI present.";
case DDERR_NOFLIPHW: case DDERR_NOFLIPHW:
return "Flipping visible surfaces is not supported."; return "Flipping visible surfaces is not supported.";
case DDERR_NOTFOUND: case DDERR_NOGDI:
return "Requested item was not found."; return "There is no GDI present.";
case DDERR_NOHWND:
return "Clipper notification requires an HWND or no HWND has previously been set as the CooperativeLevel "
"HWND.";
case DDERR_NOMIRRORHW: case DDERR_NOMIRRORHW:
return "Operation could not be carried out because there is no hardware present or available."; return "Operation could not be carried out because there is no hardware present or available.";
case DDERR_NOOVERLAYDEST:
return "Returned when GetOverlayPosition is called on an overlay that UpdateOverlay has never been called on "
"to establish a destination.";
case DDERR_NOOVERLAYHW:
return "Operation could not be carried out because there is no overlay hardware present or available.";
case DDERR_NOPALETTEATTACHED:
return "No palette object attached to this surface.";
case DDERR_NOPALETTEHW:
return "No hardware support for 16 or 256 color palettes.";
case DDERR_NORASTEROPHW: case DDERR_NORASTEROPHW:
return "Operation could not be carried out because there is no appropriate raster op hardware present or " return "Operation could not be carried out because there is no appropriate raster op hardware present or "
"available."; "available.";
case DDERR_NOOVERLAYHW:
return "Operation could not be carried out because there is no overlay hardware present or available.";
case DDERR_NOSTRETCHHW:
return "Operation could not be carried out because there is no hardware support for stretching.";
case DDERR_NOROTATIONHW: case DDERR_NOROTATIONHW:
return "Operation could not be carried out because there is no rotation hardware present or available."; return "Operation could not be carried out because there is no rotation hardware present or available.";
case DDERR_NOTEXTUREHW: case DDERR_NOSTRETCHHW:
return "Operation could not be carried out because there is no texture mapping hardware present or available."; return "Operation could not be carried out because there is no hardware support for stretching.";
case DDERR_NOT4BITCOLOR: case DDERR_NOT4BITCOLOR:
return "DirectDrawSurface is not in 4 bit color palette and the requested operation requires 4 bit color " return "DirectDrawSurface is not in 4 bit color palette and the requested operation requires 4 bit color "
"palette."; "palette.";
@ -432,118 +478,80 @@ const char* MxDeviceEnumerate::EnumerateErrorToString(HRESULT p_error)
"index palette."; "index palette.";
case DDERR_NOT8BITCOLOR: case DDERR_NOT8BITCOLOR:
return "DirectDrawSurface is not in 8 bit color mode and the requested operation requires 8 bit color."; return "DirectDrawSurface is not in 8 bit color mode and the requested operation requires 8 bit color.";
case DDERR_NOZBUFFERHW: case DDERR_NOTAOVERLAYSURFACE:
return "Operation could not be carried out because there is no hardware support for zbuffer blitting."; return "Returned when an overlay member is called for a non-overlay surface.";
case DDERR_NOTEXTUREHW:
return "Operation could not be carried out because there is no texture mapping hardware present or "
"available.";
case DDERR_NOTFLIPPABLE:
return "An attempt has been made to flip a surface that is not flippable.";
case DDERR_NOTFOUND:
return "Requested item was not found.";
case DDERR_NOTLOCKED:
return "Surface was not locked. An attempt to unlock a surface that was not locked at all, or by this "
"process, has been attempted.";
case DDERR_NOTPALETTIZED:
return "The surface being used is not a palette-based surface.";
case DDERR_NOVSYNCHW: case DDERR_NOVSYNCHW:
return "Operation could not be carried out because there is no hardware support for vertical blank " return "Operation could not be carried out because there is no hardware support for vertical blank "
"synchronized operations."; "synchronized operations.";
case DDERR_OUTOFCAPS: case DDERR_NOZBUFFERHW:
return "The hardware needed for the requested operation has already been allocated."; return "Operation could not be carried out because there is no hardware support for zbuffer blitting.";
case DDERR_NOZOVERLAYHW: case DDERR_NOZOVERLAYHW:
return "Overlay surfaces could not be z layered based on their BltOrder because the hardware does not support " return "Overlay surfaces could not be z layered based on their BltOrder because the hardware does not support "
"z layering of overlays."; "z layering of overlays.";
case DDERR_COLORKEYNOTSET: case DDERR_OUTOFCAPS:
return "No src color key specified for this operation."; return "The hardware needed for the requested operation has already been allocated.";
case DDERR_OUTOFMEMORY:
return "DirectDraw does not have enough memory to perform the operation.";
case DDERR_OUTOFVIDEOMEMORY: case DDERR_OUTOFVIDEOMEMORY:
return "DirectDraw does not have enough memory to perform the operation."; return "DirectDraw does not have enough memory to perform the operation.";
case DDERR_OVERLAYCANTCLIP: case DDERR_OVERLAYCANTCLIP:
return "The hardware does not support clipped overlays."; return "The hardware does not support clipped overlays.";
case DDERR_OVERLAYCOLORKEYONLYONEACTIVE: case DDERR_OVERLAYCOLORKEYONLYONEACTIVE:
return "Can only have ony color key active at one time for overlays."; return "Can only have ony color key active at one time for overlays.";
case DDERR_OVERLAYNOTVISIBLE:
return "Returned when GetOverlayPosition is called on a hidden overlay.";
case DDERR_PALETTEBUSY: case DDERR_PALETTEBUSY:
return "Access to this palette is being refused because the palette is already locked by another thread."; return "Access to this palette is being refused because the palette is already locked by another thread.";
case DDERR_SURFACEALREADYDEPENDENT: case DDERR_PRIMARYSURFACEALREADYEXISTS:
return "This surface is already a dependency of the surface it is being made a dependency of."; return "This process already has created a primary surface.";
case DDERR_REGIONTOOSMALL:
return "Region passed to Clipper::GetClipList is too small.";
case DDERR_SURFACEALREADYATTACHED: case DDERR_SURFACEALREADYATTACHED:
return "This surface is already attached to the surface it is being attached to."; return "This surface is already attached to the surface it is being attached to.";
case DDERR_SURFACEISOBSCURED: case DDERR_SURFACEALREADYDEPENDENT:
return "Access to surface refused because the surface is obscured."; return "This surface is already a dependency of the surface it is being made a dependency of.";
case DDERR_SURFACEBUSY: case DDERR_SURFACEBUSY:
return "Access to this surface is being refused because the surface is already locked by another thread."; return "Access to this surface is being refused because the surface is already locked by another thread.";
case DDERR_SURFACENOTATTACHED: case DDERR_SURFACEISOBSCURED:
return "The requested surface is not attached."; return "Access to surface refused because the surface is obscured.";
case DDERR_SURFACELOST: case DDERR_SURFACELOST:
return "Access to this surface is being refused because the surface memory is gone. The DirectDrawSurface " return "Access to this surface is being refused because the surface memory is gone. The DirectDrawSurface "
"object representing this surface should have Restore called on it."; "object representing this surface should have Restore called on it.";
case DDERR_TOOBIGSIZE: case DDERR_SURFACENOTATTACHED:
return "Size requested by DirectDraw is too large, but the individual height and width are OK."; return "The requested surface is not attached.";
case DDERR_TOOBIGHEIGHT: case DDERR_TOOBIGHEIGHT:
return "Height requested by DirectDraw is too large."; return "Height requested by DirectDraw is too large.";
case DDERR_UNSUPPORTEDFORMAT: case DDERR_TOOBIGSIZE:
return "FOURCC format requested is unsupported by DirectDraw."; return "Size requested by DirectDraw is too large, but the individual height and width are OK.";
case DDERR_TOOBIGWIDTH: case DDERR_TOOBIGWIDTH:
return "Width requested by DirectDraw is too large."; return "Width requested by DirectDraw is too large.";
case DDERR_VERTICALBLANKINPROGRESS: case DDERR_UNSUPPORTED:
return "Vertical blank is in progress."; return "Action not supported.";
case DDERR_UNSUPPORTEDFORMAT:
return "FOURCC format requested is unsupported by DirectDraw.";
case DDERR_UNSUPPORTEDMASK: case DDERR_UNSUPPORTEDMASK:
return "Bitmask in the pixel format requested is unsupported by DirectDraw."; return "Bitmask in the pixel format requested is unsupported by DirectDraw.";
case DDERR_XALIGN: case DDERR_VERTICALBLANKINPROGRESS:
return "Rectangle provided was not horizontally aligned on required boundary."; return "Vertical blank is in progress.";
case DDERR_WASSTILLDRAWING: case DDERR_WASSTILLDRAWING:
return "Informs DirectDraw that the previous Blt which is transfering information to or from this Surface is " return "Informs DirectDraw that the previous Blt which is transfering information to or from this Surface is "
"incomplete."; "incomplete.";
case DDERR_INVALIDDIRECTDRAWGUID:
return "The GUID passed to DirectDrawCreate is not a valid DirectDraw driver identifier.";
case DDERR_DIRECTDRAWALREADYCREATED:
return "A DirectDraw object representing this driver has already been created for this process.";
case DDERR_NODIRECTDRAWHW:
return "A hardware-only DirectDraw object creation was attempted but the driver did not support any hardware.";
case DDERR_PRIMARYSURFACEALREADYEXISTS:
return "This process already has created a primary surface.";
case DDERR_NOEMULATION:
return "Software emulation not available.";
case DDERR_REGIONTOOSMALL:
return "Region passed to Clipper::GetClipList is too small.";
case DDERR_CLIPPERISUSINGHWND:
return "An attempt was made to set a cliplist for a clipper object that is already monitoring an hwnd.";
case DDERR_NOCLIPPERATTACHED:
return "No clipper object attached to surface object.";
case DDERR_NOHWND:
return "Clipper notification requires an HWND or no HWND has previously been set as the CooperativeLevel HWND.";
case DDERR_HWNDSUBCLASSED:
return "HWND used by DirectDraw CooperativeLevel has been subclassed, this prevents DirectDraw from restoring "
"state.";
case DDERR_HWNDALREADYSET:
return "The CooperativeLevel HWND has already been set. It can not be reset while the process has surfaces or "
"palettes created.";
case DDERR_NOPALETTEATTACHED:
return "No palette object attached to this surface.";
case DDERR_NOPALETTEHW:
return "No hardware support for 16 or 256 color palettes.";
case DDERR_BLTFASTCANTCLIP:
return "Return if a clipper object is attached to the source surface passed into a BltFast call.";
case DDERR_NOBLTHW:
return "No blitter hardware present.";
case DDERR_NODDROPSHW:
return "No DirectDraw ROP hardware.";
case DDERR_OVERLAYNOTVISIBLE:
return "Returned when GetOverlayPosition is called on a hidden overlay.";
case DDERR_NOOVERLAYDEST:
return "Returned when GetOverlayPosition is called on an overlay that UpdateOverlay has never been called on "
"to establish a destination.";
case DDERR_INVALIDPOSITION:
return "Returned when the position of the overlay on the destination is no longer legal for that destination.";
case DDERR_NOTAOVERLAYSURFACE:
return "Returned when an overlay member is called for a non-overlay surface.";
case DDERR_EXCLUSIVEMODEALREADYSET:
return "An attempt was made to set the cooperative level when it was already set to exclusive.";
case DDERR_NOTFLIPPABLE:
return "An attempt has been made to flip a surface that is not flippable.";
case DDERR_CANTDUPLICATE:
return "Can't duplicate primary & 3D surfaces, or surfaces that are implicitly created.";
case DDERR_NOTLOCKED:
return "Surface was not locked. An attempt to unlock a surface that was not locked at all, or by this "
"process, has been attempted.";
case DDERR_CANTCREATEDC:
return "Windows can not create any more DCs.";
case DDERR_NODC:
return "No DC was ever created for this surface.";
case DDERR_WRONGMODE: case DDERR_WRONGMODE:
return "This surface can not be restored because it was created in a different mode."; return "This surface can not be restored because it was created in a different mode.";
case DDERR_IMPLICITLYCREATED: case DDERR_XALIGN:
return "This surface can not be restored because it is an implicitly created surface."; return "Rectangle provided was not horizontally aligned on required boundary.";
case DDERR_NOTPALETTIZED:
return "The surface being used is not a palette-based surface.";
default: default:
return "Unrecognized error value."; return "Unrecognized error value.";
} }

View File

@ -66,7 +66,7 @@ class MxAssignedDevice {
// SIZE 0x1a4 // SIZE 0x1a4
struct Direct3DDeviceInfo { struct Direct3DDeviceInfo {
Direct3DDeviceInfo() {} Direct3DDeviceInfo();
~Direct3DDeviceInfo(); ~Direct3DDeviceInfo();
Direct3DDeviceInfo( Direct3DDeviceInfo(
LPGUID p_guid, LPGUID p_guid,
@ -115,9 +115,8 @@ struct MxDisplayMode {
// SIZE 0x190 // SIZE 0x190
struct MxDriver { struct MxDriver {
MxDriver() {} MxDriver();
~MxDriver(); ~MxDriver();
MxDriver(LPGUID p_guid);
MxDriver(LPGUID p_guid, LPCSTR p_driverDesc, LPCSTR p_driverName); MxDriver(LPGUID p_guid, LPCSTR p_driverDesc, LPCSTR p_driverName);
void Init(LPGUID p_guid, LPCSTR p_driverDesc, LPCSTR p_driverName); void Init(LPGUID p_guid, LPCSTR p_driverDesc, LPCSTR p_driverName);
@ -206,20 +205,19 @@ class MxDeviceEnumerate {
); );
const char* EnumerateErrorToString(HRESULT p_error); const char* EnumerateErrorToString(HRESULT p_error);
static void BuildErrorString(const char*, ...); static void BuildErrorString(const char*, ...);
static BOOL CALLBACK static BOOL CALLBACK DirectDrawEnumerateCallback(LPGUID p_guid, LPSTR p_driverDesc, LPSTR p_driverName, LPVOID p_d);
DirectDrawEnumerateCallback(LPGUID p_guid, LPSTR p_driverDesc, LPSTR p_driverName, LPVOID p_context);
static HRESULT CALLBACK DevicesEnumerateCallback( static HRESULT CALLBACK DevicesEnumerateCallback(
LPGUID p_guid, LPGUID p_guid,
LPSTR p_deviceDesc, LPSTR p_deviceDesc,
LPSTR p_deviceName, LPSTR p_deviceName,
LPD3DDEVICEDESC p_HWDesc, LPD3DDEVICEDESC p_HWDesc,
LPD3DDEVICEDESC p_HELDesc, LPD3DDEVICEDESC p_HELDesc,
LPVOID p_context LPVOID p_d
); );
friend class MxDirect3D; friend class MxDirect3D;
friend class CConfigApp;
const list<MxDriver>& GetDriverList() const { return m_list; } friend class CMainDialog;
// SIZE 0x10 // SIZE 0x10
struct GUID4 { struct GUID4 {
@ -240,7 +238,7 @@ class MxDeviceEnumerate {
unsigned char IsInitialized() const { return m_initialized; } unsigned char IsInitialized() const { return m_initialized; }
protected: protected:
list<MxDriver> m_list; // 0x04 list<MxDriver> m_ddInfo; // 0x04
unsigned char m_initialized; // 0x10 unsigned char m_initialized; // 0x10
HWND m_hWnd; HWND m_hWnd;
}; };

View File

@ -45,7 +45,10 @@ class MxVideoManager : public MxMediaManager {
MxVideoParam& GetVideoParam() { return this->m_videoParam; } MxVideoParam& GetVideoParam() { return this->m_videoParam; }
LPDIRECTDRAW GetDirectDraw() { return this->m_pDirectDraw; } LPDIRECTDRAW GetDirectDraw() { return this->m_pDirectDraw; }
// FUNCTION: BETA10 0x1002e290
MxDisplaySurface* GetDisplaySurface() { return this->m_displaySurface; } MxDisplaySurface* GetDisplaySurface() { return this->m_displaySurface; }
MxRegion* GetRegion() { return this->m_region; } MxRegion* GetRegion() { return this->m_region; }
// SYNTHETIC: LEGO1 0x100be280 // SYNTHETIC: LEGO1 0x100be280

View File

@ -46,16 +46,18 @@ void MxDisplaySurface::Init()
} }
// FUNCTION: LEGO1 0x100ba640 // FUNCTION: LEGO1 0x100ba640
// FUNCTION: BETA10 0x1013f506
void MxDisplaySurface::ClearScreen() void MxDisplaySurface::ClearScreen()
{ {
MxS32 i;
MxS32 backBuffers; MxS32 backBuffers;
DDSURFACEDESC desc; DDSURFACEDESC desc;
if (!m_videoParam.Flags().GetFlipSurfaces()) { if (m_videoParam.Flags().GetFlipSurfaces()) {
backBuffers = 2; backBuffers = m_videoParam.GetBackBuffers() + 1;
} }
else { else {
backBuffers = m_videoParam.GetBackBuffers() + 1; backBuffers = 2;
} }
MxS32 width = m_videoParam.GetRect().GetWidth(); MxS32 width = m_videoParam.GetRect().GetWidth();
@ -1196,6 +1198,8 @@ LPDIRECTDRAWSURFACE MxDisplaySurface::CreateCursorSurface(const CursorBitmap* p_
} }
MxS32 bytesPerPixel = ddsd.ddpfPixelFormat.dwRGBBitCount / 8; MxS32 bytesPerPixel = ddsd.ddpfPixelFormat.dwRGBBitCount / 8;
MxBool isAlphaAvailable = ((ddsd.ddpfPixelFormat.dwFlags & DDPF_ALPHAPIXELS) == DDPF_ALPHAPIXELS) &&
(ddsd.ddpfPixelFormat.dwRGBAlphaBitMask != 0);
ddsd.dwWidth = p_cursorBitmap->width; ddsd.dwWidth = p_cursorBitmap->width;
ddsd.dwHeight = p_cursorBitmap->height; ddsd.dwHeight = p_cursorBitmap->height;
@ -1258,7 +1262,12 @@ LPDIRECTDRAWSURFACE MxDisplaySurface::CreateCursorSurface(const CursorBitmap* p_
MxS32 pixel; MxS32 pixel;
if (!isOpaque) { if (!isOpaque) {
pixel = RGB8888_CREATE(0xff, 0, 0xff, 0); // Transparent pixel if (isAlphaAvailable) {
pixel = RGB8888_CREATE(0, 0, 0, 0);
}
else {
pixel = RGB8888_CREATE(0xff, 0, 0xff, 0);
} // Transparent pixel
} }
else { else {
pixel = isBlack ? RGB8888_CREATE(0, 0, 0, 0xff) : RGB8888_CREATE(0xff, 0xff, 0xff, 0xff); pixel = isBlack ? RGB8888_CREATE(0, 0, 0, 0xff) : RGB8888_CREATE(0xff, 0xff, 0xff, 0xff);
@ -1288,10 +1297,12 @@ LPDIRECTDRAWSURFACE MxDisplaySurface::CreateCursorSurface(const CursorBitmap* p_
break; break;
} }
default: { default: {
DDCOLORKEY colorkey; if (!isAlphaAvailable) {
colorkey.dwColorSpaceHighValue = RGB8888_CREATE(0xff, 0, 0xff, 0); DDCOLORKEY colorkey;
colorkey.dwColorSpaceLowValue = RGB8888_CREATE(0xff, 0, 0xff, 0); colorkey.dwColorSpaceHighValue = RGB8888_CREATE(0xff, 0, 0xff, 0);
newSurface->SetColorKey(DDCKEY_SRCBLT, &colorkey); colorkey.dwColorSpaceLowValue = RGB8888_CREATE(0xff, 0, 0xff, 0);
newSurface->SetColorKey(DDCKEY_SRCBLT, &colorkey);
}
break; break;
} }
} }

View File

@ -47,7 +47,7 @@ class Matrix4 {
inline void RotateX(const float& p_angle); inline void RotateX(const float& p_angle);
inline void RotateY(const float& p_angle); inline void RotateY(const float& p_angle);
inline void RotateZ(const float& p_angle); inline void RotateZ(const float& p_angle);
inline int BETA_1005a590(Matrix4& p_mat); inline int Invert(Matrix4& p_mat);
inline void Swap(int p_d1, int p_d2); inline void Swap(int p_d1, int p_d2);
// FUNCTION: BETA10 0x1001c670 // FUNCTION: BETA10 0x1001c670

View File

@ -288,75 +288,83 @@ void Matrix4::RotateZ(const float& p_angle)
} }
// FUNCTION: BETA10 0x1005a590 // FUNCTION: BETA10 0x1005a590
int Matrix4::BETA_1005a590(Matrix4& p_mat) int Matrix4::Invert(Matrix4& p_mat)
{ {
float local5c[4][4]; // Inlined at LEGO1 0x1006b2d3
Matrix4 localc(local5c);
localc = *this; float copyData[4][4];
Matrix4 copy(copyData);
copy = *this;
p_mat.SetIdentity(); p_mat.SetIdentity();
for (int i = 0; i < 4; i++) { for (int i = 0; i < 4; i++) {
int local1c = i; int pivotColumn = i;
int local10; int column;
for (local10 = i + 1; local10 < 4; local10++) { for (column = i + 1; column < 4; column++) {
if (fabs(localc[local1c][i]) < fabs(localc[local10][i])) { if (fabs(copy[pivotColumn][i]) < fabs(copy[column][i])) {
local1c = local10; pivotColumn = column;
} }
} }
if (local1c != i) { if (pivotColumn != i) {
localc.Swap(local1c, i); copy.Swap(pivotColumn, i);
p_mat.Swap(local1c, i); p_mat.Swap(pivotColumn, i);
} }
if (localc[i][i] < 0.001f && localc[i][i] > -0.001f) { if (copy[i][i] < 0.001f && copy[i][i] > -0.001f) {
// FAILURE from mxtypes.h
return -1; return -1;
} }
float local60 = localc[i][i]; float pivotValue = copy[i][i];
int local18; int k;
for (local18 = 0; local18 < 4; local18++) { for (k = 0; k < 4; k++) {
p_mat[i][local18] /= local60; p_mat[i][k] /= pivotValue;
} }
for (local18 = 0; local18 < 4; local18++) { for (k = 0; k < 4; k++) {
localc[i][local18] /= local60; copy[i][k] /= pivotValue;
} }
for (local10 = 0; local10 < 4; local10++) { for (column = 0; column < 4; column++) {
if (i != local10) { if (i != column) {
float afStack70[4]; float tempColumn[4];
for (local18 = 0; local18 < 4; local18++) { for (k = 0; k < 4; k++) {
afStack70[local18] = p_mat[i][local18] * localc[local10][i]; tempColumn[k] = p_mat[i][k] * copy[column][i];
} }
for (local18 = 0; local18 < 4; local18++) { for (k = 0; k < 4; k++) {
p_mat[local10][local18] -= afStack70[local18]; p_mat[column][k] -= tempColumn[k];
} }
for (local18 = 0; local18 < 4; local18++) { for (k = 0; k < 4; k++) {
afStack70[local18] = localc[i][local18] * localc[local10][i]; tempColumn[k] = copy[i][k] * copy[column][i];
} }
for (local18 = 0; local18 < 4; local18++) { for (k = 0; k < 4; k++) {
localc[local10][local18] -= afStack70[local18]; copy[column][k] -= tempColumn[k];
} }
} }
} }
} }
// SUCCESS from mxtypes.h
return 0; return 0;
} }
// FUNCTION: LEGO1 0x1006b500 // FUNCTION: LEGO1 0x1006b500
// FUNCTION: BETA10 0x1005aa20
void Matrix4::Swap(int p_d1, int p_d2) void Matrix4::Swap(int p_d1, int p_d2)
{ {
for (int i = 0; i < 4; i++) { // This function is affected by entropy even in debug builds
float e = m_data[p_d1][i]; int i;
float e;
for (i = 0; i < 4; i++) {
e = m_data[p_d1][i];
m_data[p_d1][i] = m_data[p_d2][i]; m_data[p_d1][i] = m_data[p_d2][i];
m_data[p_d2][i] = e; m_data[p_d2][i] = e;
} }

View File

@ -235,7 +235,7 @@ inline void ViewManager::ManageVisibilityAndDetailRecursively(ViewROI* p_from, i
if (p_from->GetWorldBoundingSphere().Radius() > 0.001F) { if (p_from->GetWorldBoundingSphere().Radius() > 0.001F) {
float projectedSize = ProjectedSize(p_from->GetWorldBoundingSphere()); float projectedSize = ProjectedSize(p_from->GetWorldBoundingSphere());
if (projectedSize < seconds_allowed * g_viewDistance) { if (RealtimeView::GetUserMaxLOD() <= 5.0f && projectedSize < seconds_allowed * g_viewDistance) {
if (p_from->GetLodLevel() != ViewROI::c_lodLevelInvisible) { if (p_from->GetLodLevel() != ViewROI::c_lodLevelInvisible) {
ManageVisibilityAndDetailRecursively(p_from, ViewROI::c_lodLevelInvisible); ManageVisibilityAndDetailRecursively(p_from, ViewROI::c_lodLevelInvisible);
} }
@ -361,7 +361,7 @@ inline int ViewManager::CalculateLODLevel(float p_maximumScale, float p_initialS
assert(from); assert(from);
if (GetFirstLODIndex(from) != 0) { if (GetFirstLODIndex(from) != 0) {
if (p_maximumScale < g_minLODThreshold) { if (RealtimeView::GetUserMaxLOD() <= 5.0f && p_maximumScale < g_minLODThreshold) {
return 0; return 0;
} }
else { else {

View File

@ -175,6 +175,7 @@ static Uint32 UploadTextureData(SDL_Surface* src, bool useNPOT, bool isUI, float
Uint32 OpenGL1Renderer::GetTextureId(IDirect3DRMTexture* iTexture, bool isUI, float scaleX, float scaleY) Uint32 OpenGL1Renderer::GetTextureId(IDirect3DRMTexture* iTexture, bool isUI, float scaleX, float scaleY)
{ {
SDL_GL_MakeCurrent(DDWindow, m_context);
auto texture = static_cast<Direct3DRMTextureImpl*>(iTexture); auto texture = static_cast<Direct3DRMTextureImpl*>(iTexture);
auto surface = static_cast<DirectDrawSurfaceImpl*>(texture->m_surface); auto surface = static_cast<DirectDrawSurfaceImpl*>(texture->m_surface);
@ -314,6 +315,7 @@ Uint32 OpenGL1Renderer::GetMeshId(IDirect3DRMMesh* mesh, const MeshGroup* meshGr
HRESULT OpenGL1Renderer::BeginFrame() HRESULT OpenGL1Renderer::BeginFrame()
{ {
SDL_GL_MakeCurrent(DDWindow, m_context);
GL11_BeginFrame((Matrix4x4*) &m_projection[0][0]); GL11_BeginFrame((Matrix4x4*) &m_projection[0][0]);
int lightIdx = 0; int lightIdx = 0;
@ -361,6 +363,7 @@ HRESULT OpenGL1Renderer::FinalizeFrame()
void OpenGL1Renderer::Resize(int width, int height, const ViewportTransform& viewportTransform) void OpenGL1Renderer::Resize(int width, int height, const ViewportTransform& viewportTransform)
{ {
SDL_GL_MakeCurrent(DDWindow, m_context);
m_width = width; m_width = width;
m_height = height; m_height = height;
m_viewportTransform = viewportTransform; m_viewportTransform = viewportTransform;
@ -371,12 +374,14 @@ void OpenGL1Renderer::Resize(int width, int height, const ViewportTransform& vie
void OpenGL1Renderer::Clear(float r, float g, float b) void OpenGL1Renderer::Clear(float r, float g, float b)
{ {
SDL_GL_MakeCurrent(DDWindow, m_context);
m_dirty = true; m_dirty = true;
GL11_Clear(r, g, b); GL11_Clear(r, g, b);
} }
void OpenGL1Renderer::Flip() void OpenGL1Renderer::Flip()
{ {
SDL_GL_MakeCurrent(DDWindow, m_context);
if (m_dirty) { if (m_dirty) {
SDL_GL_SwapWindow(DDWindow); SDL_GL_SwapWindow(DDWindow);
m_dirty = false; m_dirty = false;
@ -385,6 +390,7 @@ void OpenGL1Renderer::Flip()
void OpenGL1Renderer::Draw2DImage(Uint32 textureId, const SDL_Rect& srcRect, const SDL_Rect& dstRect, FColor color) void OpenGL1Renderer::Draw2DImage(Uint32 textureId, const SDL_Rect& srcRect, const SDL_Rect& dstRect, FColor color)
{ {
SDL_GL_MakeCurrent(DDWindow, m_context);
m_dirty = true; m_dirty = true;
float left = -m_viewportTransform.offsetX / m_viewportTransform.scale; float left = -m_viewportTransform.offsetX / m_viewportTransform.scale;

View File

@ -390,6 +390,7 @@ void OpenGLES2Renderer::AddTextureDestroyCallback(Uint32 id, IDirect3DRMTexture*
Uint32 OpenGLES2Renderer::GetTextureId(IDirect3DRMTexture* iTexture, bool isUI, float scaleX, float scaleY) Uint32 OpenGLES2Renderer::GetTextureId(IDirect3DRMTexture* iTexture, bool isUI, float scaleX, float scaleY)
{ {
SDL_GL_MakeCurrent(DDWindow, m_context);
auto texture = static_cast<Direct3DRMTextureImpl*>(iTexture); auto texture = static_cast<Direct3DRMTextureImpl*>(iTexture);
auto surface = static_cast<DirectDrawSurfaceImpl*>(texture->m_surface); auto surface = static_cast<DirectDrawSurfaceImpl*>(texture->m_surface);
@ -484,6 +485,7 @@ Uint32 OpenGLES2Renderer::GetMeshId(IDirect3DRMMesh* mesh, const MeshGroup* mesh
HRESULT OpenGLES2Renderer::BeginFrame() HRESULT OpenGLES2Renderer::BeginFrame()
{ {
SDL_GL_MakeCurrent(DDWindow, m_context);
m_dirty = true; m_dirty = true;
glBindFramebuffer(GL_FRAMEBUFFER, m_fbo); glBindFramebuffer(GL_FRAMEBUFFER, m_fbo);
@ -599,6 +601,7 @@ HRESULT OpenGLES2Renderer::FinalizeFrame()
void OpenGLES2Renderer::Resize(int width, int height, const ViewportTransform& viewportTransform) void OpenGLES2Renderer::Resize(int width, int height, const ViewportTransform& viewportTransform)
{ {
SDL_GL_MakeCurrent(DDWindow, m_context);
m_width = width; m_width = width;
m_height = height; m_height = height;
m_viewportTransform = viewportTransform; m_viewportTransform = viewportTransform;
@ -639,6 +642,7 @@ void OpenGLES2Renderer::Resize(int width, int height, const ViewportTransform& v
void OpenGLES2Renderer::Clear(float r, float g, float b) void OpenGLES2Renderer::Clear(float r, float g, float b)
{ {
SDL_GL_MakeCurrent(DDWindow, m_context);
m_dirty = true; m_dirty = true;
glBindFramebuffer(GL_FRAMEBUFFER, m_fbo); glBindFramebuffer(GL_FRAMEBUFFER, m_fbo);
@ -651,6 +655,7 @@ void OpenGLES2Renderer::Clear(float r, float g, float b)
void OpenGLES2Renderer::Flip() void OpenGLES2Renderer::Flip()
{ {
SDL_GL_MakeCurrent(DDWindow, m_context);
if (!m_dirty) { if (!m_dirty) {
return; return;
} }
@ -714,6 +719,7 @@ void OpenGLES2Renderer::Flip()
void OpenGLES2Renderer::Draw2DImage(Uint32 textureId, const SDL_Rect& srcRect, const SDL_Rect& dstRect, FColor color) void OpenGLES2Renderer::Draw2DImage(Uint32 textureId, const SDL_Rect& srcRect, const SDL_Rect& dstRect, FColor color)
{ {
SDL_GL_MakeCurrent(DDWindow, m_context);
m_dirty = true; m_dirty = true;
glBindFramebuffer(GL_FRAMEBUFFER, m_fbo); glBindFramebuffer(GL_FRAMEBUFFER, m_fbo);

View File

@ -74,3 +74,7 @@ cksize: "Re-defined Windows name"
fccType: "Re-defined Windows name" fccType: "Re-defined Windows name"
dwDataOffset: "Re-defined Windows name" dwDataOffset: "Re-defined Windows name"
fccType: "Re-defined Windows name" fccType: "Re-defined Windows name"
SDL_KeyboardID_v: "SDL-based name"
SDL_MouseID_v: "SDL-based name"
SDL_JoystickID_v: "SDL-based name"
SDL_TouchID_v: "SDL-based name"