#include "mxwavepresenter.h" #include "decomp.h" #include "define.h" #include "legoomni.h" #include "mxautolocker.h" #include "mxdssound.h" #include "mxomni.h" #include "mxsoundmanager.h" DECOMP_SIZE_ASSERT(MxWavePresenter, 0x6c); DECOMP_SIZE_ASSERT(MxWavePresenter::WaveFormat, 0x1c); // FUNCTION: LEGO1 0x1000d640 MxWavePresenter::~MxWavePresenter() { Destroy(TRUE); } // FUNCTION: LEGO1 0x1000d6a0 void MxWavePresenter::Destroy() { Destroy(FALSE); } // FUNCTION: LEGO1 0x1000d6b0 MxBool MxWavePresenter::IsPaused() { return m_paused; } // FUNCTION: LEGO1 0x100b1ad0 void MxWavePresenter::Init() { m_waveFormat = NULL; m_dsBuffer = NULL; m_chunkLength = 0; m_lockSize = 0; m_writtenChunks = 0; m_started = FALSE; m_unk0x66 = FALSE; m_paused = FALSE; } // FUNCTION: LEGO1 0x100b1af0 MxResult MxWavePresenter::AddToManager() { MxResult result = MxSoundPresenter::AddToManager(); Init(); return result; } // FUNCTION: LEGO1 0x100b1b10 void MxWavePresenter::Destroy(MxBool p_fromDestructor) { if (m_dsBuffer) { m_dsBuffer->Stop(); m_dsBuffer->Release(); } if (m_waveFormat) delete[] ((MxU8*) m_waveFormat); Init(); if (!p_fromDestructor) MxSoundPresenter::Destroy(FALSE); } // FUNCTION: LEGO1 0x100b1b60 MxS8 MxWavePresenter::GetPlayedChunks() { DWORD dwCurrentPlayCursor, dwCurrentWriteCursor; MxS8 playedChunks = -1; if (m_dsBuffer->GetCurrentPosition(&dwCurrentPlayCursor, &dwCurrentWriteCursor) == DS_OK) playedChunks = dwCurrentPlayCursor / m_chunkLength; return playedChunks; } // FUNCTION: LEGO1 0x100b1ba0 MxBool MxWavePresenter::FUN_100b1ba0() { return !m_started || GetPlayedChunks() != m_writtenChunks; } // FUNCTION: LEGO1 0x100b1bd0 void MxWavePresenter::WriteToSoundBuffer(void* p_audioPtr, MxU32 p_length) { DWORD dwStatus; LPVOID pvAudioPtr1; DWORD dwOffset; LPVOID pvAudioPtr2; DWORD dwAudioBytes1; DWORD dwAudioBytes2; dwOffset = m_chunkLength * m_writtenChunks; m_dsBuffer->GetStatus(&dwStatus); if (dwStatus == DSBSTATUS_BUFFERLOST) { m_dsBuffer->Restore(); m_dsBuffer->GetStatus(&dwStatus); } if (dwStatus != DSBSTATUS_BUFFERLOST) { if (m_action->GetFlags() & MxDSAction::Flag_Looping) { m_writtenChunks++; m_lockSize = p_length; } else { m_writtenChunks = 1 - m_writtenChunks; m_lockSize = m_chunkLength; } if (m_dsBuffer->Lock(dwOffset, m_lockSize, &pvAudioPtr1, &dwAudioBytes1, &pvAudioPtr2, &dwAudioBytes2, 0) == DS_OK) { memcpy(pvAudioPtr1, p_audioPtr, p_length); if (m_lockSize > p_length && !(m_action->GetFlags() & MxDSAction::Flag_Looping)) { memset((MxU8*) pvAudioPtr1 + p_length, m_silenceData, m_lockSize - p_length); } m_dsBuffer->Unlock(pvAudioPtr1, m_lockSize, pvAudioPtr2, 0); } } } // FUNCTION: LEGO1 0x100b1cf0 void MxWavePresenter::ReadyTickle() { MxStreamChunk* chunk = NextChunk(); if (chunk) { m_waveFormat = (WaveFormat*) new MxU8[chunk->GetLength()]; memcpy(m_waveFormat, chunk->GetData(), chunk->GetLength()); m_subscriber->FUN_100b8390(chunk); ParseExtra(); m_previousTickleStates |= 1 << (unsigned char) m_currentTickleState; m_currentTickleState = TickleState_Starting; } } // FUNCTION: LEGO1 0x100b1d50 void MxWavePresenter::StartingTickle() { MxStreamChunk* chunk = NextChunk(); if (chunk && m_action->GetElapsedTime() >= chunk->GetTime()) { MxU32 length = chunk->GetLength(); WAVEFORMATEX waveFormatEx; m_chunkLength = length; memset(&waveFormatEx, 0, sizeof(waveFormatEx)); waveFormatEx.wFormatTag = m_waveFormat->m_waveFormatEx.wFormatTag; waveFormatEx.nChannels = m_waveFormat->m_waveFormatEx.nChannels; waveFormatEx.nSamplesPerSec = m_waveFormat->m_waveFormatEx.nSamplesPerSec; waveFormatEx.nAvgBytesPerSec = m_waveFormat->m_waveFormatEx.nAvgBytesPerSec; waveFormatEx.nBlockAlign = m_waveFormat->m_waveFormatEx.nBlockAlign; waveFormatEx.wBitsPerSample = m_waveFormat->m_waveFormatEx.wBitsPerSample; if (waveFormatEx.wBitsPerSample == 8) m_silenceData = 0x7F; if (waveFormatEx.wBitsPerSample == 16) m_silenceData = 0; DSBUFFERDESC desc; memset(&desc, 0, sizeof(desc)); desc.dwSize = sizeof(desc); if (m_unk0x66) desc.dwFlags = DSBCAPS_CTRLFREQUENCY | DSBCAPS_CTRL3D | DSBCAPS_CTRLVOLUME; else desc.dwFlags = DSBCAPS_CTRLFREQUENCY | DSBCAPS_CTRLPAN | DSBCAPS_CTRLVOLUME; if (m_action->GetFlags() & MxDSAction::Flag_Looping) desc.dwBufferBytes = m_waveFormat->m_waveFormatEx.nAvgBytesPerSec * (m_action->GetDuration() / m_action->GetLoopCount()) / 1000; else desc.dwBufferBytes = 2 * length; desc.lpwfxFormat = &waveFormatEx; if (MSoundManager()->GetDirectSound()->CreateSoundBuffer(&desc, &m_dsBuffer, NULL) != DS_OK) { EndAction(); } else { SetVolume(((MxDSSound*) m_action)->GetVolume()); m_previousTickleStates |= 1 << (unsigned char) m_currentTickleState; m_currentTickleState = TickleState_Streaming; } } } // FUNCTION: LEGO1 0x100b1ea0 void MxWavePresenter::StreamingTickle() { if (!m_currentChunk) { if (!(m_action->GetFlags() & MxDSAction::Flag_Looping)) { MxStreamChunk* chunk = FUN_100b5650(); if (chunk && chunk->GetFlags() & MxDSChunk::Flag_End && !(chunk->GetFlags() & MxDSChunk::Flag_Bit16)) { chunk->SetFlags(chunk->GetFlags() | MxDSChunk::Flag_Bit16); m_currentChunk = new MxStreamChunk; MxU8* data = new MxU8[m_chunkLength]; memset(data, m_silenceData, m_chunkLength); m_currentChunk->SetLength(m_chunkLength); m_currentChunk->SetData(data); m_currentChunk->SetTime(chunk->GetTime() + 1000); m_currentChunk->SetFlags(MxDSChunk::Flag_Bit1); } } MxMediaPresenter::StreamingTickle(); } } // FUNCTION: LEGO1 0x100b20c0 void MxWavePresenter::DoneTickle() { if (m_dsBuffer) { DWORD dwCurrentPlayCursor, dwCurrentWriteCursor; m_dsBuffer->GetCurrentPosition(&dwCurrentPlayCursor, &dwCurrentWriteCursor); MxS8 playedChunks = dwCurrentPlayCursor / m_chunkLength; if (m_action->GetFlags() & MxDSAction::Flag_Bit7 || m_action->GetFlags() & MxDSAction::Flag_Looping || m_writtenChunks != playedChunks || m_lockSize + (m_chunkLength * playedChunks) <= dwCurrentPlayCursor) MxMediaPresenter::DoneTickle(); } else MxMediaPresenter::DoneTickle(); } // FUNCTION: LEGO1 0x100b2130 void MxWavePresenter::AppendChunk(MxStreamChunk* p_chunk) { WriteToSoundBuffer(p_chunk->GetData(), p_chunk->GetLength()); if (IsEnabled()) m_subscriber->FUN_100b8390(p_chunk); } // FUNCTION: LEGO1 0x100b2160 MxResult MxWavePresenter::PutData() { MxAutoLocker lock(&m_criticalSection); if (IsEnabled()) { switch (m_currentTickleState) { case TickleState_Streaming: if (m_currentChunk && FUN_100b1ba0()) { WriteToSoundBuffer(m_currentChunk->GetData(), m_currentChunk->GetLength()); m_subscriber->FUN_100b8390(m_currentChunk); m_currentChunk = NULL; } if (!m_started) { m_dsBuffer->SetCurrentPosition(0); if (m_dsBuffer->Play(0, 0, DSBPLAY_LOOPING) == DS_OK) m_started = TRUE; } break; case TickleState_Repeating: if (m_started) break; m_dsBuffer->SetCurrentPosition(0); if (m_dsBuffer->Play(0, 0, m_action->GetLoopCount() > 1) == DS_OK) m_started = TRUE; } } return SUCCESS; } // FUNCTION: LEGO1 0x100b2280 void MxWavePresenter::EndAction() { if (m_action) { MxAutoLocker lock(&m_criticalSection); MxMediaPresenter::EndAction(); if (m_dsBuffer) m_dsBuffer->Stop(); } } // FUNCTION: LEGO1 0x100b2300 void MxWavePresenter::SetVolume(MxS32 p_volume) { m_criticalSection.Enter(); m_volume = p_volume; if (m_dsBuffer != NULL) { MxS32 volume = p_volume * MxOmni::GetInstance()->GetSoundManager()->GetVolume() / 100; MxS32 otherVolume = MxOmni::GetInstance()->GetSoundManager()->FUN_100aecf0(volume); m_dsBuffer->SetVolume(otherVolume); } m_criticalSection.Leave(); } // FUNCTION: LEGO1 0x100b2360 void MxWavePresenter::Enable(MxBool p_enable) { if (IsEnabled() != p_enable) { MxSoundPresenter::Enable(p_enable); if (p_enable) { m_writtenChunks = 0; m_started = FALSE; } else if (m_dsBuffer) m_dsBuffer->Stop(); } } // FUNCTION: LEGO1 0x100b23a0 void MxWavePresenter::ParseExtra() { char extraCopy[512]; MxSoundPresenter::ParseExtra(); *((MxU16*) &extraCopy[0]) = m_action->GetExtraLength(); char* extraData = m_action->GetExtraData(); if (*((MxU16*) &extraCopy[0])) { MxU16 len = *((MxU16*) &extraCopy[0]); memcpy(extraCopy, extraData, len); extraCopy[len] = '\0'; char soundValue[512]; if (KeyValueStringParse(soundValue, g_strSOUND, extraCopy)) { if (!strcmpi(soundValue, "FALSE")) Enable(FALSE); } } } // FUNCTION: LEGO1 0x100b2440 void MxWavePresenter::Pause() { if (!m_paused && m_started) { if (m_dsBuffer) m_dsBuffer->Stop(); m_paused = TRUE; } } // FUNCTION: LEGO1 0x100b2470 void MxWavePresenter::Resume() { if (m_paused) { if (m_dsBuffer && m_started) { switch (m_currentTickleState) { case TickleState_Streaming: m_dsBuffer->Play(0, 0, DSBPLAY_LOOPING); break; case TickleState_Repeating: m_dsBuffer->Play(0, 0, m_action->GetLoopCount() > 1); break; case TickleState_Done: m_dsBuffer->Play(0, 0, 0); } } m_paused = FALSE; } }