You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
2286 lines
73 KiB
2286 lines
73 KiB
/*==LICENSE==* |
|
|
|
CyanWorlds.com Engine - MMOG client, server and tools |
|
Copyright (C) 2011 Cyan Worlds, Inc. |
|
|
|
This program is free software: you can redistribute it and/or modify |
|
it under the terms of the GNU General Public License as published by |
|
the Free Software Foundation, either version 3 of the License, or |
|
(at your option) any later version. |
|
|
|
This program is distributed in the hope that it will be useful, |
|
but WITHOUT ANY WARRANTY; without even the implied warranty of |
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
|
GNU General Public License for more details. |
|
|
|
You should have received a copy of the GNU General Public License |
|
along with this program. If not, see <http://www.gnu.org/licenses/>. |
|
|
|
Additional permissions under GNU GPL version 3 section 7 |
|
|
|
If you modify this Program, or any covered work, by linking or |
|
combining it with any of RAD Game Tools Bink SDK, Autodesk 3ds Max SDK, |
|
NVIDIA PhysX SDK, Microsoft DirectX SDK, OpenSSL library, Independent |
|
JPEG Group JPEG library, Microsoft Windows Media SDK, or Apple QuickTime SDK |
|
(or a modified version of those libraries), |
|
containing parts covered by the terms of the Bink SDK EULA, 3ds Max EULA, |
|
PhysX SDK EULA, DirectX SDK EULA, OpenSSL and SSLeay licenses, IJG |
|
JPEG Library README, Windows Media SDK EULA, or QuickTime SDK EULA, the |
|
licensors of this Program grant you additional |
|
permission to convey the resulting work. Corresponding Source for a |
|
non-source form of such a combination shall include the source code for |
|
the parts of OpenSSL and IJG JPEG Library used as well as that of the covered |
|
work. |
|
|
|
You can contact Cyan Worlds, Inc. by email legal@cyan.com |
|
or by snail mail at: |
|
Cyan Worlds, Inc. |
|
14617 N Newport Hwy |
|
Mead, WA 99021 |
|
|
|
*==LICENSE==*/ |
|
#pragma warning(disable: 4284) |
|
#include "HeadSpin.h" |
|
#include "hsWindows.h" |
|
#include "plClient.h" |
|
#include "hsStream.h" |
|
#include "plResMgr/plResManager.h" |
|
#include "plResMgr/plKeyFinder.h" |
|
#include "pnKeyedObject/plKey.h" |
|
#include "pnKeyedObject/plFixedKey.h" |
|
#include "pnMessage/plRefMsg.h" |
|
#include "pnSceneObject/plSceneObject.h" |
|
#include "pnSceneObject/plCoordinateInterface.h" |
|
#include "plScene/plSceneNode.h" |
|
#include "pnMessage/plTimeMsg.h" |
|
#include "pnMessage/plClientMsg.h" |
|
#include "pfCamera/plVirtualCamNeu.h" |
|
#include "hsTimer.h" |
|
#include "plPipeline/hsG3DDeviceSelector.h" |
|
#include "plFile/plEncryptedStream.h" |
|
#include "plInputCore/plInputManager.h" |
|
#include "plInputCore/plInputInterfaceMgr.h" |
|
#include "plInputCore/plInputDevice.h" |
|
#include "plPhysX/plSimulationMgr.h" |
|
#include "plNetClient/plNetClientMgr.h" |
|
#include "plAvatar/plAvatarMgr.h" |
|
#include "plScene/plRelevanceMgr.h" |
|
#include "pnTimer/plTimerCallbackManager.h" |
|
#include "pfAudio/plListener.h" |
|
#include "pnMessage/plCmdIfaceModMsg.h" |
|
#include "plMessage/plRoomLoadNotifyMsg.h" |
|
#include "pnMessage/plPlayerPageMsg.h" |
|
#include "pnMessage/plCameraMsg.h" |
|
#include "plMessage/plTransitionMsg.h" |
|
#include "plMessage/plLinkToAgeMsg.h" |
|
#include "plMessage/plNetCommMsgs.h" |
|
#include "plMessage/plAgeLoadedMsg.h" |
|
#include "plMessage/plResPatcherMsg.h" |
|
|
|
#include "pfConsoleCore/pfConsoleEngine.h" |
|
#include "pfConsole/pfConsole.h" |
|
#include "pfConsole/pfConsoleDirSrc.h" |
|
#include "plScene/plPageTreeMgr.h" |
|
#include "plScene/plVisMgr.h" |
|
|
|
#include "pfKI/pfKI.h" |
|
|
|
#include "plAudio/plAudioSystem.h" |
|
#include "plAudio/plAudioCaps.h" |
|
|
|
#include "plStatGather/plProfileManagerFull.h" |
|
|
|
#include "plPipeline.h" |
|
#include "plPipeDebugFlags.h" |
|
#include "plPipeline/plPipelineCreate.h" |
|
#include "plPipeline/plTransitionMgr.h" |
|
#include "plPipeline/plCaptureRender.h" |
|
#include "plPipeline/plDynamicEnvMap.h" |
|
#include "plNetClient/plLinkEffectsMgr.h" |
|
#include "plAvatar/plAvatarClothing.h" |
|
#include "plAvatar/plArmatureMod.h" |
|
#include "pnMessage/plProxyDrawMsg.h" |
|
|
|
#include "plScene/plRenderRequest.h" |
|
#include "plDrawable/plAccessGeometry.h" |
|
#include "plPipeResReq.h" |
|
#include "plDrawable/plVisLOSMgr.h" |
|
|
|
#include "plGImage/plBitmap.h" |
|
|
|
#include "plStatusLog/plStatusLog.h" |
|
#include "plProgressMgr/plProgressMgr.h" |
|
#include "plPipeline/plDTProgressMgr.h" |
|
#include "pfMoviePlayer/plMoviePlayer.h" |
|
#include "plMessage/plMovieMsg.h" |
|
|
|
#include "plSDL/plSDL.h" |
|
|
|
#include "pnDispatch/plDispatch.h" |
|
#include "pnDispatch/plDispatchLogBase.h" |
|
#include "pfGameGUIMgr/pfGameGUIMgr.h" |
|
#include "pfPython/cyMisc.h" |
|
#include "plMessage/plInputEventMsg.h" |
|
#include "plMessage/plRenderRequestMsg.h" |
|
#include "pnMessage/plEventCallbackMsg.h" |
|
#include "plModifier/plSimpleModifier.h" |
|
#include "plAudible.h" |
|
#include "plMessage/plAnimCmdMsg.h" |
|
#include "pnMessage/plSoundMsg.h" |
|
#include "pnMessage/plAudioSysMsg.h" |
|
#include "plMessage/plRenderMsg.h" |
|
#include "plAgeLoader/plResPatcher.h" |
|
#include "pfPython/cyPythonInterface.h" |
|
#include "plUnifiedTime/plClientUnifiedTime.h" |
|
#include "pfAnimation/plAnimDebugList.h" |
|
#include "pfGameGUIMgr/pfGUICtrlGenerator.h" |
|
|
|
#include "plGImage/plFontCache.h" |
|
|
|
#include "pfJournalBook/pfJournalBook.h" |
|
|
|
#include "plAnimation/plAGAnimInstance.h" |
|
#include "plAgeLoader/plAgeLoader.h" |
|
|
|
#include "plQuality.h" |
|
#include "plGLight/plShadowCaster.h" |
|
|
|
#include "plNetClient/plNetLinkingMgr.h" |
|
#include "plNetCommon/plNetCommonConstants.h" |
|
#include "plNetGameLib/plNetGameLib.h" |
|
|
|
#include "pfLocalizationMgr/pfLocalizationMgr.h" |
|
#include "pfPatcher/plManifests.h" |
|
|
|
#include "plTweak.h" |
|
|
|
#define MSG_LOADING_BAR |
|
|
|
// static hsVector3 gAbsDown(0,0,-1.f); |
|
|
|
static plDispatchBase* gDisp = nil; |
|
static plTimerCallbackManager* gTimerMgr = nil; |
|
static plAudioSystem* gAudio = nil; |
|
|
|
#ifdef HS_BUILD_FOR_WIN32 |
|
extern ITaskbarList3* gTaskbarList; |
|
#endif |
|
|
|
bool plClient::fDelayMS = false; |
|
|
|
plClient* plClient::fInstance=nil; |
|
|
|
static hsTArray<HMODULE> fLoadedDLLs; |
|
|
|
plClient::plClient() |
|
: fPipeline(nil), |
|
fDone(false), |
|
fQuitIntro(false), |
|
fWindowHndl(nil), |
|
fInputManager(nil), |
|
fConsole(nil), |
|
fCurrentNode(nil), |
|
fNewCamera(nil), |
|
fpAuxInitDir(nil), |
|
fTransitionMgr(nil), |
|
fLinkEffectsMgr(nil), |
|
fProgressBar(nil), |
|
fGameGUIMgr(nil), |
|
fWindowActive(false), |
|
fAnimDebugList(nil), |
|
fClampCap(-1), |
|
fQuality(0), |
|
fPageMgr(nil), |
|
fFontCache(nil), |
|
fHoldLoadRequests(false), |
|
fNumLoadingRooms(0), |
|
fNumPostLoadMsgs(0), |
|
fPostLoadMsgInc(0.f) |
|
{ |
|
#ifndef PLASMA_EXTERNAL_RELEASE |
|
bPythonDebugConnected = false; |
|
#endif |
|
|
|
hsStatusMessage("Constructing client\n"); |
|
plClient::SetInstance(this); |
|
// gNextRoom[0] = '\0'; |
|
|
|
// Setup the timer. These can be overriden with console commands. |
|
hsTimer::SetRealTime(true); |
|
#ifdef HS_DEBUGGING |
|
// hsTimer::SetRealTime(false); |
|
hsTimer::SetTimeClamp(0.1f); |
|
#else // HS_DEBUGGING |
|
// hsTimer::SetRealTime(true); |
|
hsTimer::SetTimeClamp(0); |
|
#endif // HS_DEBUGGING |
|
|
|
IDetectAudioVideoSettings(); // need to do this before the console is created |
|
|
|
/// allow console commands to start working early |
|
// Create the console engine |
|
fConsoleEngine = new pfConsoleEngine(); |
|
|
|
// create network mgr before console runs |
|
plNetClientMgr::SetInstance(new plNetClientMgr); |
|
plAgeLoader::SetInstance(new plAgeLoader); |
|
|
|
// Use it to parse the init directory |
|
plFileName initFolder = plFileSystem::GetInitPath(); |
|
pfConsoleDirSrc dirSrc(fConsoleEngine, initFolder, "*.ini"); |
|
|
|
#ifndef PLASMA_EXTERNAL_RELEASE |
|
// internal builds also parse the local init folder |
|
dirSrc.ParseDirectory("init", "*.ini"); |
|
#endif |
|
|
|
/// End of console stuff |
|
} |
|
|
|
plClient::~plClient() |
|
{ |
|
hsStatusMessage("Destructing client\n"); |
|
|
|
plClient::SetInstance( nil ); |
|
|
|
delete fPageMgr; |
|
delete [] fpAuxInitDir; |
|
} |
|
|
|
#include "plGImage/plAVIWriter.h" |
|
#include "pfCharacter/pfMarkerMgr.h" |
|
|
|
template<typename T> |
|
static void IUnRegisterAs(T*& ko, plFixedKeyId id) |
|
{ |
|
if (ko) { |
|
ko->UnRegisterAs(id); |
|
ko = nullptr; |
|
} |
|
} |
|
|
|
bool plClient::Shutdown() |
|
{ |
|
plSynchEnabler ps(false); // disable dirty state tracking during shutdown |
|
delete fProgressBar; |
|
|
|
// Just in case, clear this out (trying to fix a crash bug where this is still active at shutdown) |
|
plDispatch::SetMsgRecieveCallback(nil); |
|
|
|
// Let the resmanager know we're going to be shutting down. |
|
hsgResMgr::ResMgr()->BeginShutdown(); |
|
|
|
// Must kill off all movies before shutting down audio. |
|
IKillMovies(); |
|
|
|
plgAudioSys::Activate(false); |
|
|
|
// Get any proxies to commit suicide. |
|
plProxyDrawMsg* nuke = new plProxyDrawMsg(plProxyDrawMsg::kAllTypes |
|
| plProxyDrawMsg::kDestroy); |
|
plgDispatch::MsgSend(nuke); |
|
|
|
if (plAVIWriter::IsInitialized()) |
|
plAVIWriter::Instance().Shutdown(); |
|
|
|
hsStatusMessage( "Shutting down client...\n" ); |
|
|
|
// First, before anybody else goes away, write out our key mappings |
|
if( plInputInterfaceMgr::GetInstance() ) |
|
plInputInterfaceMgr::GetInstance()->WriteKeyMap(); |
|
|
|
// tell Python that its ok to shutdown |
|
PythonInterface::WeAreInShutdown(); |
|
|
|
// Shutdown the journalBook API |
|
pfJournalBook::SingletonShutdown(); |
|
|
|
/// Take down the KI |
|
pfGameGUIMgr *mgr = pfGameGUIMgr::GetInstance(); |
|
if( mgr ) |
|
mgr->UnloadDialog( "KIBlackBar" ); // unload the blackbar which will bootstrap in the rest of the KI dialogs |
|
|
|
// Take down our GUI control generator |
|
pfGUICtrlGenerator::Instance().Shutdown(); |
|
|
|
if (plNetClientMgr* nc = plNetClientMgr::GetInstance()) |
|
nc->Shutdown(); |
|
if (plAgeLoader* al = plAgeLoader::GetInstance()) |
|
al->Shutdown(); |
|
|
|
IUnRegisterAs(fInputManager, kInput_KEY); |
|
IUnRegisterAs(fGameGUIMgr, kGameGUIMgr_KEY); |
|
|
|
for (int i = 0; i < fRooms.Count(); i++) |
|
{ |
|
plSceneNode *sn = fRooms[i].fNode; |
|
GetKey()->Release(sn->GetKey()); |
|
} |
|
fRooms.Reset(); |
|
fRoomsLoading.clear(); |
|
|
|
// Shutdown plNetClientMgr |
|
|
|
plAccessGeometry::DeInit(); |
|
|
|
delete fPipeline; |
|
fPipeline = nil; |
|
|
|
if (plSimulationMgr::GetInstance()) |
|
plSimulationMgr::Shutdown(); |
|
plAvatarMgr::ShutDown(); |
|
plRelevanceMgr::DeInit(); |
|
|
|
if (fPageMgr) |
|
fPageMgr->Reset(); |
|
|
|
IUnRegisterAs(fTransitionMgr, kTransitionMgr_KEY); |
|
|
|
delete fConsoleEngine; |
|
fConsoleEngine = nil; |
|
|
|
IUnRegisterAs(fLinkEffectsMgr, kLinkEffectsMgr_KEY); |
|
|
|
plClothingMgr::DeInit(); |
|
|
|
IUnRegisterAs(fFontCache, kFontCache_KEY); |
|
|
|
pfMarkerMgr::Shutdown(); |
|
|
|
delete fAnimDebugList; |
|
|
|
IUnRegisterAs(fConsole, kConsoleObject_KEY); |
|
|
|
PythonInterface::finiPython(); |
|
|
|
IUnRegisterAs(fNewCamera, kVirtualCamera1_KEY); |
|
|
|
// mark the listener for death. |
|
// there's no need to keep this around... |
|
plUoid lu(kListenerMod_KEY); |
|
plKey pLKey = hsgResMgr::ResMgr()->FindKey(lu); |
|
if (pLKey) |
|
{ |
|
plListener* pLMod = plListener::ConvertNoRef(pLKey->GetObjectPtr()); |
|
if (pLMod) |
|
pLMod->UnRegisterAs(kListenerMod_KEY); |
|
} |
|
|
|
plgAudioSys::Shutdown(); |
|
|
|
if (pfLocalizationMgr::InstanceValid()) |
|
pfLocalizationMgr::Shutdown(); |
|
|
|
ShutdownDLLs(); |
|
|
|
plVisLOSMgr::DeInit(); |
|
|
|
delete fPageMgr; |
|
fPageMgr = nil; |
|
plGlobalVisMgr::DeInit(); |
|
|
|
#ifdef TRACK_AG_ALLOCS |
|
DumpAGAllocs(); |
|
#endif // TRACK_AG_ALLOCS |
|
|
|
// This will destruct the client. Do it last. |
|
UnRegisterAs(kClient_KEY); |
|
|
|
return false; |
|
} |
|
|
|
void plClient::InitAuxInits() |
|
{ |
|
// Use another init directory specified in Command line Arg -i |
|
if (fpAuxInitDir) |
|
pfConsoleDirSrc dirSrc( fConsoleEngine, fpAuxInitDir, "*.ini" ); |
|
} |
|
|
|
void plClient::InitInputs() |
|
{ |
|
hsStatusMessage("InitInputs client\n"); |
|
fInputManager = new plInputManager( fWindowHndl ); |
|
fInputManager->CreateInterfaceMod(fPipeline); |
|
fInputManager->RegisterAs( kInput_KEY ); |
|
plgDispatch::Dispatch()->RegisterForExactType(plIMouseXEventMsg::Index(), fInputManager->GetKey()); |
|
plgDispatch::Dispatch()->RegisterForExactType(plIMouseYEventMsg::Index(), fInputManager->GetKey()); |
|
plgDispatch::Dispatch()->RegisterForExactType(plIMouseBEventMsg::Index(), fInputManager->GetKey()); |
|
plgDispatch::Dispatch()->RegisterForExactType(plEvalMsg::Index(), fInputManager->GetKey()); |
|
plInputDevice* pKeyboard = new plKeyboardDevice(); |
|
fInputManager->AddInputDevice(pKeyboard); |
|
|
|
plInputDevice* pMouse = new plMouseDevice(); |
|
fInputManager->AddInputDevice(pMouse); |
|
|
|
if( fWindowActive ) |
|
fInputManager->Activate( true ); |
|
} |
|
|
|
void plClient::ISetGraphicsDefaults() |
|
{ |
|
// couldn't find display mode set defaults write to ini file |
|
plFileName graphicsIniFile = plFileName::Join(plFileSystem::GetInitPath(), "graphics.ini"); |
|
IWriteDefaultGraphicsSettings(graphicsIniFile); |
|
plPipeline::fInitialPipeParams.Windowed = plPipeline::fDefaultPipeParams.Windowed; |
|
plPipeline::fInitialPipeParams.AntiAliasingAmount = plPipeline::fDefaultPipeParams.AntiAliasingAmount; |
|
plPipeline::fInitialPipeParams.AnisotropicLevel = plPipeline::fDefaultPipeParams.AnisotropicLevel; |
|
plPipeline::fInitialPipeParams.TextureQuality = plPipeline::fDefaultPipeParams.TextureQuality; |
|
plPipeline::fInitialPipeParams.VSync = plPipeline::fDefaultPipeParams.VSync; |
|
plShadowCaster::EnableShadowCast(plPipeline::fDefaultPipeParams.Shadows ? true : false); |
|
plQuality::SetQuality(plPipeline::fDefaultPipeParams.VideoQuality); |
|
if( (fClampCap >= 0) && (fClampCap < plQuality::GetCapability()) ) |
|
plQuality::SetCapability(fClampCap); |
|
plDynamicCamMap::SetEnabled(plPipeline::fDefaultPipeParams.PlanarReflections ? true : false); |
|
} |
|
|
|
bool plClient::InitPipeline() |
|
{ |
|
hsStatusMessage("InitPipeline client\n"); |
|
HWND hWnd = fWindowHndl; |
|
|
|
hsG3DDeviceModeRecord dmr; |
|
hsG3DDeviceSelector devSel; |
|
devSel.Enumerate(hWnd); |
|
devSel.RemoveUnusableDevModes(true); |
|
|
|
if (!devSel.GetDefault(&dmr)) |
|
{ |
|
hsMessageBox("No suitable rendering devices found.","Plasma", hsMessageBoxNormal, hsMessageBoxIconError); |
|
return true; |
|
} |
|
|
|
hsG3DDeviceRecord *rec = (hsG3DDeviceRecord *)dmr.GetDevice(); |
|
int res = -1; |
|
|
|
if(!plPipeline::fInitialPipeParams.Windowed) |
|
{ |
|
// find our resolution if we're not in windowed mode |
|
for ( int i = 0; i < rec->GetModes().GetCount(); i++ ) |
|
{ |
|
hsG3DDeviceMode *mode = rec->GetMode(i); |
|
if ((mode->GetWidth() == plPipeline::fInitialPipeParams.Width) && |
|
(mode->GetHeight() == plPipeline::fInitialPipeParams.Height) && |
|
(mode->GetColorDepth() == plPipeline::fInitialPipeParams.ColorDepth)) |
|
{ |
|
res = i; |
|
break; |
|
} |
|
} |
|
if(res != -1) |
|
{ |
|
// found it set it as the current mode. |
|
dmr = hsG3DDeviceModeRecord(*rec, *rec->GetMode(res)); |
|
} |
|
else |
|
{ |
|
ISetGraphicsDefaults(); |
|
} |
|
} |
|
|
|
if(plPipeline::fInitialPipeParams.TextureQuality == -1) |
|
{ |
|
plPipeline::fInitialPipeParams.TextureQuality = dmr.GetDevice()->GetCap(hsG3DDeviceSelector::kCapsPixelShader) ? 2 : 1; |
|
} |
|
else |
|
{ |
|
// clamp value to range |
|
if(plPipeline::fInitialPipeParams.TextureQuality > 2) plPipeline::fInitialPipeParams.TextureQuality = 2; |
|
if(plPipeline::fInitialPipeParams.TextureQuality < 0) plPipeline::fInitialPipeParams.TextureQuality = 0; |
|
plBitmap::SetGlobalLevelChopCount(2 - plPipeline::fInitialPipeParams.TextureQuality); |
|
} |
|
|
|
plPipeline *pipe = plPipelineCreate::CreatePipeline( hWnd, &dmr ); |
|
if( pipe->GetErrorString() != nil ) |
|
{ |
|
ISetGraphicsDefaults(); |
|
#ifdef PLASMA_EXTERNAL_RELEASE |
|
hsMessageBox("There was an error initializing the video card.\nSetting defaults.", "Error", hsMessageBoxNormal); |
|
#else |
|
hsMessageBox( pipe->GetErrorString(), "Error creating pipeline", hsMessageBoxNormal ); |
|
#endif |
|
delete pipe; |
|
devSel.GetDefault(&dmr); |
|
pipe = plPipelineCreate::CreatePipeline( hWnd, &dmr ); |
|
if(pipe->GetErrorString() != nil) |
|
{ |
|
// not much else we can do |
|
return true; |
|
} |
|
} |
|
fPipeline = pipe; |
|
|
|
hsVector3 up; |
|
hsPoint3 from, at; |
|
from.Set(0, 0, 10.f); |
|
at.Set(0, 20.f, 10.f); |
|
up.Set(0,0,-1.f); |
|
hsMatrix44 cam; |
|
cam.MakeCamera(&from,&at,&up); |
|
|
|
float yon = 500.0f; |
|
|
|
pipe->SetFOV( 60.f, int32_t( 60.f * pipe->Height() / pipe->Width() ) ); |
|
pipe->SetDepth( 0.3f, yon ); |
|
|
|
hsMatrix44 id; |
|
id.Reset(); |
|
|
|
pipe->SetWorldToCamera( cam, id ); |
|
pipe->RefreshMatrices(); |
|
|
|
// Do this so we're still black before we show progress bars, but the correct color coming out of 'em |
|
fClearColor.Set( 0.f, 0.f, 0.f, 1.f ); |
|
pipe->SetClear(&fClearColor); |
|
pipe->ClearRenderTarget(); |
|
|
|
plAccessGeometry::Init(pipe); |
|
|
|
if( fPipeline ) |
|
fPipeline->LoadResources(); |
|
|
|
return false; |
|
} |
|
|
|
//============================================================================ |
|
void plClient::SetClearColor( hsColorRGBA &color ) |
|
{ |
|
fClearColor = color; |
|
if( fPipeline != nil ) |
|
{ |
|
fPipeline->SetClear(&fClearColor, nil); |
|
} |
|
} |
|
|
|
//============================================================================ |
|
void plClient::IDispatchMsgReceiveCallback() |
|
{ |
|
if (fInstance->fProgressBar) |
|
fInstance->fProgressBar->Increment(1); |
|
|
|
static char buf[30]; |
|
sprintf(buf, "Msg %d", fInstance->fNumPostLoadMsgs); |
|
fInstance->IIncProgress(fInstance->fPostLoadMsgInc, buf); |
|
|
|
fInstance->fNumPostLoadMsgs++; |
|
} |
|
|
|
|
|
//============================================================================ |
|
bool plClient::MsgReceive(plMessage* msg) |
|
{ |
|
if (plGenRefMsg * genRefMsg = plGenRefMsg::ConvertNoRef(msg)) { |
|
// do nothing, we just use the client's key to ref vault image nodes. |
|
return true; |
|
} |
|
|
|
plClientRefMsg* pRefMsg = plClientRefMsg::ConvertNoRef(msg); |
|
if (pRefMsg) |
|
{ |
|
switch(pRefMsg->fType) |
|
{ |
|
case plClientRefMsg::kLoadRoom : |
|
#ifndef PLASMA_EXTERNAL_RELEASE |
|
plStatusLog::AddLineS( "pageouts.log", ".. ClientRefMsg received for room %s", pRefMsg->GetRef() != nil ? pRefMsg->GetRef()->GetKey()->GetUoid().GetObjectName().c_str() : "nilref" ); |
|
#endif |
|
|
|
// was it that the room was loaded? |
|
if (hsCheckBits(pRefMsg->GetContext(), plRefMsg::kOnCreate)) |
|
IRoomLoaded(plSceneNode::Convert(pRefMsg->GetRef()), false); |
|
// or was it that the room was unloaded? |
|
else if (hsCheckBits(pRefMsg->GetContext(), plRefMsg::kOnDestroy)) |
|
IRoomUnloaded(plSceneNode::Convert(pRefMsg->GetRef())); |
|
#ifndef PLASMA_EXTERNAL_RELEASE |
|
else |
|
plStatusLog::AddLineS("pageouts.log", ".. refMsg is UNHANDLED"); |
|
#endif |
|
break; |
|
|
|
case plClientRefMsg::kLoadRoomHold: |
|
if (hsCheckBits(pRefMsg->GetContext(), plRefMsg::kOnCreate)) |
|
IRoomLoaded(plSceneNode::Convert(pRefMsg->GetRef()), true); |
|
break; |
|
|
|
// |
|
// Manually add room. |
|
// Add to pageMgr, but don't load the entire room. |
|
// |
|
case plClientRefMsg::kManualRoom: |
|
{ |
|
if (pRefMsg->GetContext() & plRefMsg::kOnCreate || |
|
pRefMsg->GetContext() & plRefMsg::kOnRequest) |
|
{ |
|
bool found=false; |
|
plSceneNode *pNode = plSceneNode::ConvertNoRef(pRefMsg->GetRef()); |
|
int i; |
|
for (i = 0; i < fRooms.Count(); i++) |
|
{ |
|
if (fRooms[i].fNode->GetKey() == pRefMsg->GetSender()) |
|
{ |
|
found=true; |
|
break; |
|
} |
|
} |
|
if (!found) |
|
{ |
|
if (pNode) |
|
{ |
|
fRooms.Append( plRoomRec( pNode, 0 ) ); |
|
fPageMgr->AddNode(pNode); |
|
} |
|
} |
|
} |
|
else |
|
{ |
|
plSceneNode* node = plSceneNode::ConvertNoRef(pRefMsg->GetRef()); |
|
if(node) |
|
{ |
|
int i; |
|
for (i = 0; i < fRooms.Count(); i++) |
|
{ |
|
if (fRooms[i].fNode->GetKey() == node->GetKey()) |
|
{ |
|
fRooms.Remove(i); |
|
break; |
|
} |
|
} |
|
fPageMgr->RemoveNode(node); |
|
} |
|
} |
|
} |
|
break; |
|
|
|
} |
|
} |
|
|
|
plClientMsg* pMsg = plClientMsg::ConvertNoRef(msg); |
|
if (pMsg) |
|
{ |
|
switch(pMsg->GetClientMsgFlag()) |
|
{ |
|
case plClientMsg::kQuit: |
|
SetDone(true); |
|
break; |
|
|
|
case plClientMsg::kLoadRoom: |
|
case plClientMsg::kLoadRoomHold: |
|
{ |
|
IQueueRoomLoad(pMsg->GetRoomLocs(), (pMsg->GetClientMsgFlag() == plClientMsg::kLoadRoomHold)); |
|
if (!fHoldLoadRequests) |
|
ILoadNextRoom(); |
|
} |
|
break; |
|
|
|
case plClientMsg::kUnloadRoom: |
|
IUnloadRooms(pMsg->GetRoomLocs()); |
|
break; |
|
|
|
case plClientMsg::kLoadNextRoom: |
|
ILoadNextRoom(); |
|
break; |
|
|
|
// Load optimizations: messages to pre-load and un-load all the keys in a given age |
|
case plClientMsg::kLoadAgeKeys: |
|
{ |
|
plResManager *mgr = (plResManager *)hsgResMgr::ResMgr(); |
|
mgr->LoadAgeKeys( pMsg->GetAgeName() ); |
|
} |
|
break; |
|
|
|
case plClientMsg::kReleaseAgeKeys: |
|
{ |
|
plResManager *mgr = (plResManager *)hsgResMgr::ResMgr(); |
|
mgr->DropAgeKeys( pMsg->GetAgeName() ); |
|
} |
|
break; |
|
|
|
case plClientMsg::kDisableRenderScene: |
|
{ |
|
plClient::GetInstance()->SetFlag( plClient::kFlagDBGDisableRender, true ); |
|
} |
|
break; |
|
case plClientMsg::kEnableRenderScene: |
|
{ |
|
plClient::GetInstance()->SetFlag( plClient::kFlagDBGDisableRender, false ); |
|
} |
|
break; |
|
|
|
case plClientMsg::kResetGraphicsDevice: |
|
{ |
|
ResetDisplayDevice(pMsg->fGraphicsSettings.fWidth, pMsg->fGraphicsSettings.fHeight, pMsg->fGraphicsSettings.fColorDepth, pMsg->fGraphicsSettings.fWindowed, pMsg->fGraphicsSettings.fNumAASamples, pMsg->fGraphicsSettings.fMaxAnisoSamples, pMsg->fGraphicsSettings.fVSync); |
|
} |
|
break; |
|
|
|
case plClientMsg::kSetGraphicsDefaults: |
|
{ |
|
ISetGraphicsDefaults(); |
|
ResetDisplayDevice(plPipeline::fDefaultPipeParams.Width, plPipeline::fDefaultPipeParams.Height, plPipeline::fDefaultPipeParams.ColorDepth, plPipeline::fDefaultPipeParams.Windowed, |
|
plPipeline::fDefaultPipeParams.AntiAliasingAmount, plPipeline::fDefaultPipeParams.AnisotropicLevel, plPipeline::fDefaultPipeParams.VSync); |
|
} |
|
break; |
|
|
|
case plClientMsg::kFlashWindow: |
|
{ |
|
FlashWindow(); |
|
} |
|
break; |
|
} |
|
return true; |
|
} |
|
plRenderRequestMsg* rendReq = plRenderRequestMsg::ConvertNoRef(msg); |
|
if( rendReq ) |
|
{ |
|
IAddRenderRequest(rendReq->Request()); |
|
return true; |
|
} |
|
plEventCallbackMsg* callback = plEventCallbackMsg::ConvertNoRef(msg); |
|
if( callback ) |
|
{ |
|
plString str = plFormat("Callback event from {}\n", callback->GetSender() |
|
? callback->GetSender()->GetName() |
|
: "Unknown"); |
|
hsStatusMessage(str.c_str()); |
|
static int gotten = 0; |
|
if( ++gotten > 5 ) |
|
{ |
|
plSimpleModifier* simpMod = plSimpleModifier::ConvertNoRef(callback->GetSender()->ObjectIsLoaded()); |
|
plAudible* aud = plAudible::ConvertNoRef(callback->GetSender()->ObjectIsLoaded()); |
|
if( simpMod ) |
|
{ |
|
plAnimCmdMsg* cmd = new plAnimCmdMsg; |
|
cmd->AddReceiver(simpMod->GetKey()); |
|
cmd->SetCmd(plAnimCmdMsg::kRemoveCallbacks); |
|
cmd->AddCallback(callback); |
|
plgDispatch::MsgSend(cmd); |
|
hsRefCnt_SafeUnRef(callback); |
|
} |
|
else if( aud ) |
|
{ |
|
plSoundMsg* cmd = new plSoundMsg; |
|
cmd->AddReceiver(aud->GetKey()); |
|
cmd->SetCmd(plSoundMsg::kRemoveCallbacks); |
|
cmd->AddCallback(callback); |
|
plgDispatch::MsgSend(cmd); |
|
hsRefCnt_SafeUnRef(callback); |
|
} |
|
hsStatusMessage("Removed\n"); |
|
gotten = 0; |
|
} |
|
return true; |
|
} |
|
plMovieMsg* mov = plMovieMsg::ConvertNoRef(msg); |
|
if( mov ) |
|
{ |
|
return IHandleMovieMsg(mov); |
|
} |
|
|
|
plLinkEffectsTriggerMsg* linkFX = plLinkEffectsTriggerMsg::ConvertNoRef(msg); |
|
if (linkFX) |
|
{ |
|
if (!linkFX->IsLeavingAge()) |
|
{ |
|
#ifdef MSG_LOADING_BAR |
|
// Temporary stat gathering stuff |
|
#if 0//ndef PLASMA_EXTERNAL_RELEASE |
|
hsUNIXStream s; |
|
s.Open("Messages.txt", "at"); |
|
static bool firstLog = true; |
|
if (firstLog) |
|
{ |
|
firstLog = false; |
|
s.WriteString("------------------------------------\n"); |
|
} |
|
char buf[256]; |
|
sprintf(buf, "%s %d\n", plAgeLoader::GetInstance()->GetCurrAgeFilename(), fNumPostLoadMsgs); |
|
s.WriteString(buf); |
|
s.Close(); |
|
#endif |
|
#endif |
|
} |
|
return true; |
|
} |
|
|
|
//============================================================================ |
|
// plResPatcherMsg |
|
//============================================================================ |
|
if (plResPatcherMsg * resMsg = plResPatcherMsg::ConvertNoRef(msg)) { |
|
IHandlePatcherMsg(resMsg); |
|
return true; |
|
} |
|
|
|
//============================================================================ |
|
// plNetCommAuthMsg |
|
//============================================================================ |
|
if (plNetCommAuthMsg* authMsg = plNetCommAuthMsg::ConvertNoRef(msg)) { |
|
plgDispatch::Dispatch()->UnRegisterForExactType(plNetCommAuthMsg::Index(), GetKey()); |
|
if (IS_NET_SUCCESS(authMsg->result)) { |
|
SetFlag(kFlagInitialAuthComplete); |
|
IPatchGlobalAgeFiles(); |
|
} |
|
} |
|
|
|
return hsKeyedObject::MsgReceive(msg); |
|
} |
|
|
|
//============================================================================ |
|
bool plClient::IHandleMovieMsg(plMovieMsg* mov) |
|
{ |
|
if (mov->GetFileName().IsEmpty()) |
|
return true; |
|
|
|
size_t i = fMovies.size(); |
|
if (!(mov->GetCmd() & plMovieMsg::kMake)) |
|
{ |
|
for (i = 0; i < fMovies.size(); i++) |
|
{ |
|
if (mov->GetFileName().CompareI(fMovies[i]->GetFileName().AsString()) == 0) |
|
break; |
|
} |
|
} |
|
if (i == fMovies.size()) |
|
{ |
|
fMovies.push_back(new plMoviePlayer()); |
|
fMovies[i]->SetFileName(mov->GetFileName()); |
|
} |
|
|
|
if (mov->GetCmd() & plMovieMsg::kAddCallbacks) |
|
{ |
|
int j; |
|
for (j = 0; j < mov->GetNumCallbacks(); j++) |
|
fMovies[i]->AddCallback(mov->GetCallback(j)); |
|
} |
|
if (mov->GetCmd() & plMovieMsg::kMove) |
|
fMovies[i]->SetPosition(mov->GetCenter()); |
|
if (mov->GetCmd() & plMovieMsg::kScale) |
|
fMovies[i]->SetScale(mov->GetScale()); |
|
if (mov->GetCmd() & plMovieMsg::kColorAndOpacity) |
|
fMovies[i]->SetColor(mov->GetColor()); |
|
if (mov->GetCmd() & plMovieMsg::kColor) |
|
{ |
|
hsColorRGBA c = fMovies[i]->GetColor(); |
|
c.Set(mov->GetColor().r, mov->GetColor().g, mov->GetColor().b, c.a); |
|
fMovies[i]->SetColor(c); |
|
} |
|
if (mov->GetCmd() & plMovieMsg::kOpacity) |
|
{ |
|
hsColorRGBA c = fMovies[i]->GetColor(); |
|
c.a = mov->GetColor().a; |
|
fMovies[i]->SetColor(c); |
|
} |
|
if (mov->GetCmd() & plMovieMsg::kFadeIn) |
|
{ |
|
fMovies[i]->SetFadeFromColor(mov->GetFadeInColor()); |
|
fMovies[i]->SetFadeFromTime(mov->GetFadeInSecs()); |
|
} |
|
if (mov->GetCmd() & plMovieMsg::kFadeOut) |
|
{ |
|
fMovies[i]->SetFadeToColor(mov->GetFadeOutColor()); |
|
fMovies[i]->SetFadeToTime(mov->GetFadeOutSecs()); |
|
} |
|
if (mov->GetCmd() & plMovieMsg::kVolume) |
|
fMovies[i]->SetVolume(mov->GetVolume()); |
|
|
|
if (mov->GetCmd() & plMovieMsg::kStart) |
|
fMovies[i]->Start(); |
|
if (mov->GetCmd() & plMovieMsg::kPause) |
|
fMovies[i]->Pause(true); |
|
if (mov->GetCmd() & plMovieMsg::kResume) |
|
fMovies[i]->Pause(false); |
|
if (mov->GetCmd() & plMovieMsg::kStop) |
|
fMovies[i]->Stop(); |
|
|
|
// If a movie has lost its filename, it means something went horribly wrong |
|
// with playing it and it has shutdown. Or we just stopped it. Either way, |
|
// we need to clear it out of our list. |
|
if (!fMovies[i]->GetFileName().IsValid()) |
|
{ |
|
delete fMovies[i]; |
|
fMovies[i] = fMovies.back(); |
|
fMovies.pop_back(); |
|
} |
|
return true; |
|
} |
|
|
|
int plClient::IFindRoomByLoc(const plLocation& loc) |
|
{ |
|
for (int i = 0; i < fRooms.Count(); i++) |
|
{ |
|
if (fRooms[i].fNode->GetKey()->GetUoid().GetLocation() == loc) |
|
return i; |
|
} |
|
|
|
return -1; |
|
} |
|
|
|
bool plClient::IIsRoomLoading(const plLocation& loc) |
|
{ |
|
for (int i = 0; i < fRoomsLoading.size(); i++) |
|
{ |
|
if (fRoomsLoading[i] == loc) |
|
return true; |
|
} |
|
return false; |
|
} |
|
|
|
void plClient::SetHoldLoadRequests(bool hold) |
|
{ |
|
fHoldLoadRequests = hold; |
|
if (!fHoldLoadRequests) |
|
ILoadNextRoom(); |
|
} |
|
|
|
#include "plResMgr/plPageInfo.h" |
|
|
|
void plClient::IQueueRoomLoad(const std::vector<plLocation>& locs, bool hold) |
|
{ |
|
bool allSameAge = true; |
|
plString lastAgeName; |
|
|
|
uint32_t numRooms = 0; |
|
for (int i = 0; i < locs.size(); i++) |
|
{ |
|
const plLocation& loc = locs[i]; |
|
|
|
const plPageInfo* info = plKeyFinder::Instance().GetLocationInfo(loc); |
|
bool alreadyLoaded = (IFindRoomByLoc(loc) != -1); |
|
bool isLoading = IIsRoomLoading(loc); |
|
if (!info || alreadyLoaded || isLoading) |
|
{ |
|
#ifdef HS_DEBUGGING |
|
if (!info) |
|
hsStatusMessageF("Ignoring LoadRoom request for location 0x%x because we can't find the location", loc.GetSequenceNumber()); |
|
else if (alreadyLoaded) |
|
hsStatusMessageF("Ignoring LoadRoom request for %s-%s, since room is already loaded", info->GetAge().c_str(), info->GetPage().c_str()); |
|
else if (isLoading) |
|
hsStatusMessageF("Ignoring LoadRoom request for %s-%s, since room is currently loading", info->GetAge().c_str(), info->GetPage().c_str()); |
|
#endif |
|
|
|
continue; |
|
} |
|
|
|
fLoadRooms.push_back(new LoadRequest(loc, hold)); |
|
|
|
if (lastAgeName.IsNull() || info->GetAge() == lastAgeName) |
|
lastAgeName = info->GetAge(); |
|
else |
|
allSameAge = false; |
|
|
|
// hsStatusMessageF("+++ Loading room %s-%s", info.GetAge(), info.GetPage()); |
|
numRooms++; |
|
} |
|
|
|
if (numRooms == 0) |
|
return; |
|
|
|
fNumLoadingRooms += numRooms; |
|
} |
|
|
|
void plClient::ILoadNextRoom() |
|
{ |
|
LoadRequest* req = nil; |
|
|
|
while (!fLoadRooms.empty()) |
|
{ |
|
req = fLoadRooms.front(); |
|
fLoadRooms.pop_front(); |
|
|
|
bool alreadyLoaded = (IFindRoomByLoc(req->loc) != -1); |
|
bool isLoading = IIsRoomLoading(req->loc); |
|
if (alreadyLoaded || isLoading) |
|
{ |
|
delete req; |
|
req = nil; |
|
fNumLoadingRooms--; |
|
} |
|
else |
|
break; |
|
} |
|
|
|
if (req) |
|
{ |
|
plClientRefMsg* pRefMsg = new plClientRefMsg(GetKey(), |
|
plRefMsg::kOnCreate, -1, |
|
req->hold ? plClientRefMsg::kLoadRoomHold : plClientRefMsg::kLoadRoom); |
|
|
|
fRoomsLoading.push_back(req->loc); // flag the location as currently loading |
|
|
|
// PageInPage is not guaranteed to finish synchronously, just FYI |
|
plResManager *mgr = (plResManager *)hsgResMgr::ResMgr(); |
|
mgr->PageInRoom(req->loc, plSceneNode::Index(), pRefMsg); |
|
|
|
delete req; |
|
|
|
plClientMsg* nextRoom = new plClientMsg(plClientMsg::kLoadNextRoom); |
|
nextRoom->Send(GetKey()); |
|
} |
|
} |
|
|
|
void plClient::IUnloadRooms(const std::vector<plLocation>& locs) |
|
{ |
|
for (int i = 0; i < locs.size(); i++) |
|
{ |
|
const plLocation& loc = locs[i]; |
|
|
|
if (!loc.IsValid()) |
|
continue; |
|
|
|
plKey nodeKey = nil; |
|
|
|
// First, look in our room list. It *should* be there, which allows us to avoid a |
|
// potential nasty reload-find in the resMgr. |
|
int roomIdx = IFindRoomByLoc(loc); |
|
if (roomIdx != -1) |
|
nodeKey = fRooms[roomIdx].fNode->GetKey(); |
|
|
|
if (nodeKey == nil) |
|
{ |
|
nodeKey = plKeyFinder::Instance().FindSceneNodeKey(loc); |
|
} |
|
|
|
if (nodeKey != nil) |
|
{ |
|
plSceneNode* node = plSceneNode::ConvertNoRef(nodeKey->ObjectIsLoaded()); |
|
if (node) |
|
{ |
|
#ifndef PLASMA_EXTERNAL_RELEASE |
|
plStatusLog::AddLineS("pageouts.log", "SceneNode for %s loaded; Removing node", |
|
node->GetKey()->GetUoid().GetObjectName().c_str()); |
|
#endif |
|
fPageMgr->RemoveNode(node); |
|
} |
|
else |
|
{ |
|
#ifndef PLASMA_EXTERNAL_RELEASE |
|
plStatusLog::AddLineS("pageouts.log", "SceneNode for %s NOT loaded", |
|
nodeKey->GetUoid().GetObjectName().c_str()); |
|
#endif |
|
} |
|
GetKey()->Release(nodeKey); // release notify interest in scene node |
|
|
|
uint32_t recFlags = 0; |
|
if (roomIdx != -1) |
|
{ |
|
recFlags = fRooms[roomIdx].fFlags; |
|
fRooms.Remove(roomIdx); |
|
} |
|
|
|
if (node == fCurrentNode) |
|
fCurrentNode = nil; |
|
|
|
#ifndef PLASMA_EXTERNAL_RELEASE |
|
plStatusLog::AddLineS("pageouts.log", "Telling netClientMgr about paging out %s", |
|
nodeKey->GetUoid().GetObjectName().c_str()); |
|
#endif |
|
|
|
if (plNetClientMgr::GetInstance() != nil) |
|
{ |
|
// Don't care really about the message that just came in, we care whether it was really held or not |
|
if (!hsCheckBits(recFlags, plRoomRec::kHeld)) |
|
plAgeLoader::GetInstance()->StartPagingOutRoom(&nodeKey, 1); |
|
// Tell NetClientManager not to expect any pageout info on this guy, since he was held |
|
else |
|
plAgeLoader::GetInstance()->IgnorePagingOutRoom(&nodeKey, 1); |
|
} |
|
} |
|
else |
|
{ |
|
#ifndef PLASMA_EXTERNAL_RELEASE |
|
// plStatusLog::AddLineS("pageouts.log", "++ Can't find node key for paging out room %s, loc 0x%x", |
|
// pMsg->GetRoomName() != nil ? pMsg->GetRoomName() : "", |
|
// loc.GetSequenceNumber()); |
|
#endif |
|
} |
|
} |
|
} |
|
|
|
void plClient::IRoomLoaded(plSceneNode* node, bool hold) |
|
{ |
|
fCurrentNode = node; |
|
// make sure we don't already have this room in the list: |
|
bool bAppend = true; |
|
for (int i = 0; i < fRooms.Count(); i++) |
|
{ |
|
if (fRooms[i].fNode == fCurrentNode) |
|
{ |
|
bAppend = false; |
|
break; |
|
} |
|
} |
|
if (bAppend) |
|
{ |
|
if (hold) |
|
{ |
|
fRooms.Append(plRoomRec(fCurrentNode, plRoomRec::kHeld)); |
|
} |
|
else |
|
{ |
|
fRooms.Append(plRoomRec(fCurrentNode, 0)); |
|
fPageMgr->AddNode(fCurrentNode); |
|
} |
|
} |
|
|
|
fNumLoadingRooms--; |
|
|
|
// Shut down the progress bar if that was the last room |
|
if (fProgressBar != nil && fNumLoadingRooms <= 0) |
|
{ |
|
#ifdef MSG_LOADING_BAR |
|
if (!hold) |
|
{ |
|
struct AgeMsgCount { const char* AgeName; int NumMsgs; }; |
|
static AgeMsgCount ageMsgCount[] = |
|
{ |
|
{ "BahroCave", 2600 }, |
|
{ "BaronCityOffice", 670 }, |
|
{ "city", 269000 }, |
|
{ "Cleft", 11000 }, |
|
{ "Garden", 19700 }, |
|
{ "Garrison", 28800 }, |
|
{ "Gira", 3300 }, |
|
{ "Kadish", 19700 }, |
|
{ "Neighborhood", 19900 }, |
|
{ "Nexus", 1400 }, |
|
{ "Personal", 20300 }, |
|
{ "Teledahn", 48000 } |
|
}; |
|
|
|
char name[256]; |
|
strcpy(name, &fProgressBar->GetTitle().c_str()[strlen("Loading ")]); |
|
name[strlen(name)-3] = '\0'; |
|
|
|
// Get the precalculated value for how many messages will be |
|
// sent out before the screen actually fades in |
|
int numMsgs = 0; |
|
for (int i = 0; i < sizeof(ageMsgCount)/sizeof(AgeMsgCount); i++) |
|
{ |
|
if (strcmp(ageMsgCount[i].AgeName, name) == 0) |
|
{ |
|
numMsgs = ageMsgCount[i].NumMsgs; |
|
break; |
|
} |
|
} |
|
|
|
fNumPostLoadMsgs = 0; |
|
|
|
// The last 10% of the age loading bar is for messages, so adjust |
|
// our progress bar increment to fill the bar fully when all |
|
// messages have been sent |
|
float max = fProgressBar->GetMax(); |
|
float amtLeft = max - (max * 0.9f); |
|
fPostLoadMsgInc = (numMsgs != 0) ? amtLeft / numMsgs : 0; |
|
|
|
#ifndef PLASMA_EXTERNAL_RELEASE |
|
if (plDispatchLogBase::IsLogging()) |
|
plDispatchLogBase::GetInstance()->LogStatusBarChange(fProgressBar->GetTitle().c_str(), "displaying messages"); |
|
#endif // PLASMA_EXTERNAL_RELEASE |
|
#endif |
|
} |
|
} |
|
|
|
hsRefCnt_SafeUnRef(fCurrentNode); |
|
plKey pRmKey = fCurrentNode->GetKey(); |
|
plAgeLoader::GetInstance()->FinishedPagingInRoom(&pRmKey, 1); |
|
// *** this used to call "ActivateNode" (in physics) which wasn't implemented. |
|
// *** we should make this "turn on" physics for the selected node |
|
// *** depending on what guarantees we can make about the load state -- anything useful? |
|
|
|
// now tell all those who are interested that a room was loaded |
|
if (!hold) |
|
{ |
|
plRoomLoadNotifyMsg* loadmsg = new plRoomLoadNotifyMsg; |
|
loadmsg->SetRoom(pRmKey); |
|
loadmsg->SetWhatHappen(plRoomLoadNotifyMsg::kLoaded); |
|
plgDispatch::MsgSend(loadmsg); |
|
} |
|
else |
|
hsStatusMessageF("Done loading hold room %s, t=%f\n", pRmKey->GetName().c_str(), hsTimer::GetSeconds()); |
|
|
|
plLocation loc = pRmKey->GetUoid().GetLocation(); |
|
for (int i = 0; i < fRoomsLoading.size(); i++) |
|
{ |
|
if (fRoomsLoading[i] == loc) |
|
{ |
|
fRoomsLoading.erase(fRoomsLoading.begin() + i); |
|
break; |
|
} |
|
} |
|
|
|
if (!fNumLoadingRooms) |
|
IStopProgress(); |
|
} |
|
|
|
//============================================================================ |
|
void plClient::IRoomUnloaded(plSceneNode* node) |
|
{ |
|
#ifndef PLASMA_EXTERNAL_RELEASE |
|
plStatusLog::AddLineS("pageouts.log", ".. refMsg is onDestroy"); |
|
#endif |
|
|
|
fCurrentNode = node; |
|
hsRefCnt_SafeUnRef(fCurrentNode); |
|
plKey pRmKey = fCurrentNode->GetKey(); |
|
if (plAgeLoader::GetInstance()) |
|
plAgeLoader::GetInstance()->FinishedPagingOutRoom(&pRmKey, 1); |
|
|
|
// tell all those who are interested that a room was unloaded |
|
plRoomLoadNotifyMsg* loadmsg = new plRoomLoadNotifyMsg; |
|
loadmsg->SetRoom(pRmKey); |
|
loadmsg->SetWhatHappen(plRoomLoadNotifyMsg::kUnloaded); |
|
plgDispatch::MsgSend(loadmsg); |
|
} |
|
|
|
void plClient::IReadKeyedObjCallback(plKey key) |
|
{ |
|
fInstance->IIncProgress(1, key->GetName().c_str()); |
|
} |
|
|
|
//============================================================================ |
|
void plClient::IProgressMgrCallbackProc(plOperationProgress * progress) |
|
{ |
|
if(!fInstance) |
|
return; |
|
|
|
// Increments the taskbar progress [Windows 7+] |
|
#ifdef HS_BUILD_FOR_WIN32 |
|
if (gTaskbarList && fInstance->GetWindowHandle()) |
|
{ |
|
static TBPFLAG lastState = TBPF_NOPROGRESS; |
|
TBPFLAG myState; |
|
|
|
// So, calling making these kernel calls is kind of SLOW. So, let's |
|
// hide that behind a userland check--this helps linking go faster! |
|
if (progress->IsAborting()) |
|
myState = TBPF_ERROR; |
|
else if (progress->IsLastUpdate()) |
|
myState = TBPF_NOPROGRESS; |
|
else if (progress->GetMax() == 0.f) |
|
myState = TBPF_INDETERMINATE; |
|
else |
|
myState = TBPF_NORMAL; |
|
|
|
if (myState == TBPF_NORMAL) |
|
// This sets us to TBPF_NORMAL |
|
gTaskbarList->SetProgressValue(fInstance->GetWindowHandle(), (ULONGLONG)progress->GetProgress(), (ULONGLONG)progress->GetMax()); |
|
else if (myState != lastState) |
|
gTaskbarList->SetProgressState(fInstance->GetWindowHandle(), myState); |
|
lastState = myState; |
|
} |
|
#endif |
|
|
|
fInstance->fMessagePumpProc(); |
|
|
|
// HACK HACK HACK HACK! |
|
// Yes, this is the ORIGINAL, EVIL famerate limit from plClient::IDraw (except I bumped it to 60fps) |
|
// As it so happens, this callback is happening in the main resource loading thread |
|
// Without this NASTY ASS HACK, we draw after loading every KO, which starves the loader. |
|
// At some point, a better solution should be found... Like running the loader in a separate thread. |
|
static float lastDrawTime; |
|
static const float kMaxFrameRate = 1.f/60.f; |
|
float currTime = (float) hsTimer::GetSeconds(); |
|
if ((currTime - lastDrawTime) > kMaxFrameRate) |
|
{ |
|
fInstance->IDraw(); |
|
lastDrawTime = currTime; |
|
} |
|
} |
|
|
|
//============================================================================ |
|
void plClient::IIncProgress (float byHowMuch, const char * text) |
|
{ |
|
if (fProgressBar) { |
|
#ifndef PLASMA_EXTERNAL_RELEASE |
|
fProgressBar->SetStatusText( text ); |
|
#endif |
|
fProgressBar->Increment( byHowMuch ); |
|
} |
|
} |
|
|
|
//============================================================================ |
|
void plClient::IStartProgress( const char *title, float len ) |
|
{ |
|
if (fProgressBar) |
|
{ |
|
fProgressBar->SetLength(fProgressBar->GetMax()+len); |
|
} |
|
else |
|
{ |
|
fProgressBar = plProgressMgr::GetInstance()->RegisterOperation(len, title, plProgressMgr::kNone, false, true); |
|
#ifndef PLASMA_EXTERNAL_RELEASE |
|
if (plDispatchLogBase::IsLogging()) |
|
plDispatchLogBase::GetInstance()->LogStatusBarChange(fProgressBar->GetTitle().c_str(), "starting"); |
|
#endif // PLASMA_EXTERNAL_RELEASE |
|
|
|
((plResManager*)hsgResMgr::ResMgr())->SetProgressBarProc(IReadKeyedObjCallback); |
|
plDispatch::SetMsgRecieveCallback(IDispatchMsgReceiveCallback); |
|
|
|
fLastProgressUpdate = 0.f; |
|
} |
|
// Workaround for NVidia driver bug, showing up as BCO not there first time. |
|
// See Mantis bug 0014590. |
|
if( fPipeline ) |
|
fPipeline->LoadResources(); |
|
} |
|
|
|
|
|
//============================================================================ |
|
void plClient::IStopProgress( void ) |
|
{ |
|
if (fProgressBar) |
|
{ |
|
#ifndef PLASMA_EXTERNAL_RELEASE |
|
if (plDispatchLogBase::IsLogging()) |
|
plDispatchLogBase::GetInstance()->LogStatusBarChange(fProgressBar->GetTitle().c_str(), "done"); |
|
#endif // PLASMA_EXTERNAL_RELEASE |
|
|
|
plDispatch::SetMsgRecieveCallback(nil); |
|
((plResManager*)hsgResMgr::ResMgr())->SetProgressBarProc(IReadKeyedObjCallback); |
|
delete fProgressBar; |
|
fProgressBar = nil; |
|
|
|
plPipeResReq::Request(); |
|
|
|
fFlags.SetBit(kFlagGlobalDataLoaded); |
|
if (fFlags.IsBitSet(kFlagAsyncInitComplete)) |
|
ICompleteInit(); |
|
} |
|
} |
|
|
|
/***************************************************************************** |
|
* |
|
* |
|
* |
|
***/ |
|
|
|
extern bool gDataServerLocal; |
|
|
|
#include "plQuality.h" |
|
#include "plLoadMask.h" |
|
|
|
//============================================================================ |
|
bool plClient::StartInit() |
|
{ |
|
hsStatusMessage("Init client\n"); |
|
fFlags.SetBit( kFlagIniting ); |
|
|
|
pfLocalizationMgr::Initialize("dat"); |
|
|
|
plQuality::SetQuality(fQuality); |
|
if( (GetClampCap() >= 0) && (GetClampCap() < plQuality::GetCapability()) ) |
|
plQuality::SetCapability(GetClampCap()); |
|
|
|
/// 2.16.2001 mcn - Moved console engine init to constructor, |
|
/// so we could use console commands even before the pipeline init |
|
|
|
plDTProgressMgr::DeclareThyself(); |
|
|
|
// Set our callback for the progress manager so everybody else can use it |
|
fLastProgressUpdate = 0.f; |
|
plProgressMgr::GetInstance()->SetCallbackProc( IProgressMgrCallbackProc ); |
|
|
|
// Check the registry, which deletes data files that are either corrupt or |
|
// have old version numbers. If the file still exists on the file server |
|
// then it will be patched on-the-fly as needed (unless you're running with |
|
// local data of course). |
|
((plResManager *)hsgResMgr::ResMgr())->VerifyPages(); |
|
|
|
plgAudioSys::Init(); |
|
gAudio = plgAudioSys::Sys(); |
|
|
|
RegisterAs( kClient_KEY ); |
|
|
|
InitDLLs(); |
|
|
|
plGlobalVisMgr::Init(); |
|
fPageMgr = new plPageTreeMgr; |
|
|
|
plVisLOSMgr::Init(fPipeline, fPageMgr); |
|
|
|
// init globals |
|
plAvatarMgr::GetInstance(); |
|
plRelevanceMgr::Init(); |
|
|
|
gDisp = plgDispatch::Dispatch(); |
|
gTimerMgr = plgTimerCallbackMgr::Mgr(); |
|
|
|
// |
|
// initialize input system |
|
// |
|
InitInputs(); |
|
|
|
/// Init the console object |
|
/// Note: this can be done last because the console engine was inited first, and |
|
/// everything in code that works with the console does so through the console engine |
|
|
|
fConsole = new pfConsole(); |
|
pfConsole::SetPipeline( fPipeline ); |
|
fConsole->RegisterAs( kConsoleObject_KEY ); // fixedKey from plFixedKey.h |
|
fConsole->Init( fConsoleEngine ); |
|
|
|
/// Init the font cache |
|
fFontCache = new plFontCache(); |
|
|
|
/// Init the transition manager |
|
fTransitionMgr = new plTransitionMgr(); |
|
fTransitionMgr->RegisterAs( kTransitionMgr_KEY ); // fixedKey from plFixedKey.h |
|
fTransitionMgr->Init(); |
|
|
|
// Init the Age Linking effects manager |
|
fLinkEffectsMgr = new plLinkEffectsMgr(); |
|
fLinkEffectsMgr->RegisterAs( kLinkEffectsMgr_KEY ); // fixedKey from plFixedKey.h |
|
fLinkEffectsMgr->Init(); |
|
|
|
/// Init the in-game GUI manager |
|
fGameGUIMgr = new pfGameGUIMgr(); |
|
fGameGUIMgr->RegisterAs( kGameGUIMgr_KEY ); |
|
fGameGUIMgr->Init(); |
|
|
|
plgAudioSys::Activate(true); |
|
|
|
// |
|
// Init Net before loading things |
|
// |
|
plgDispatch::Dispatch()->RegisterForExactType(plNetCommAuthMsg::Index(), GetKey()); |
|
plNetClientMgr::GetInstance()->RegisterAs(kNetClientMgr_KEY); |
|
plAgeLoader::GetInstance()->Init(); |
|
|
|
plCmdIfaceModMsg* pModMsg2 = new plCmdIfaceModMsg; |
|
pModMsg2->SetBCastFlag(plMessage::kBCastByExactType); |
|
pModMsg2->SetSender(fConsole->GetKey()); |
|
pModMsg2->SetCmd(plCmdIfaceModMsg::kAdd); |
|
plgDispatch::MsgSend(pModMsg2); |
|
|
|
// create new virtual camera |
|
fNewCamera = new plVirtualCam1; |
|
fNewCamera->RegisterAs( kVirtualCamera1_KEY ); |
|
fNewCamera->Init(); |
|
fNewCamera->SetPipeline( GetPipeline() ); |
|
|
|
plVirtualCam1::Refresh(); |
|
pfGameGUIMgr::GetInstance()->SetAspectRatio( (float)fPipeline->Width() / (float)fPipeline->Height() ); |
|
plMouseDevice::Instance()->SetDisplayResolution((float)fPipeline->Width(), (float)fPipeline->Height()); |
|
plInputManager::SetRecenterMouse(false); |
|
|
|
plgDispatch::Dispatch()->RegisterForExactType(plMovieMsg::Index(), GetKey()); |
|
|
|
// create the listener for the audio system: |
|
plListener* pLMod = new plListener; |
|
pLMod->RegisterAs(kListenerMod_KEY ); |
|
|
|
plgDispatch::Dispatch()->RegisterForExactType(plEvalMsg::Index(), pLMod->GetKey()); |
|
plgDispatch::Dispatch()->RegisterForExactType(plAudioSysMsg::Index(), pLMod->GetKey()); |
|
|
|
plSynchedObject::PushSynchDisabled(false); // enable dirty tracking |
|
return true; |
|
} |
|
|
|
//============================================================================ |
|
bool plClient::BeginGame() |
|
{ |
|
plNetClientMgr::GetInstance()->Init(); |
|
IPlayIntroMovie("avi/CyanWorlds.webm", 0.f, 0.f, 0.f, 1.f, 1.f, 0.75); |
|
SetFlag(kFlagIntroComplete); |
|
if (GetDone()) return false; |
|
IPatchGlobalAgeFiles(); |
|
return true; |
|
} |
|
|
|
//============================================================================ |
|
void plClient::IPatchGlobalAgeFiles( void ) |
|
{ |
|
if (HasFlag(kFlagIntroComplete) && HasFlag(kFlagInitialAuthComplete)) { |
|
plgDispatch::Dispatch()->RegisterForExactType(plResPatcherMsg::Index(), GetKey()); |
|
|
|
plResPatcher* patcher = plResPatcher::GetInstance(); |
|
patcher->Update(plManifest::EssentialGameManifests()); |
|
} |
|
} |
|
|
|
void plClient::InitDLLs() |
|
{ |
|
hsStatusMessage("Init dlls client\n"); |
|
typedef void (*PInitGlobalsFunc) (hsResMgr *, plFactory *, plTimerCallbackManager *, plTimerShare*, |
|
plNetClientApp*); |
|
|
|
std::vector<plFileName> dlls = plFileSystem::ListDir("ModDLL", "*.dll"); |
|
for (auto iter = dlls.begin(); iter != dlls.end(); ++iter) |
|
{ |
|
HMODULE hMod = LoadLibraryW(iter->AsString().ToWchar()); |
|
if (hMod) |
|
{ |
|
PInitGlobalsFunc initGlobals = (PInitGlobalsFunc)GetProcAddress(hMod, "InitGlobals"); |
|
(*initGlobals)(hsgResMgr::ResMgr(), plFactory::GetTheFactory(), plgTimerCallbackMgr::Mgr(), |
|
hsTimer::GetTheTimer(), plNetClientApp::GetInstance()); |
|
fLoadedDLLs.Append(hMod); |
|
} |
|
} |
|
} |
|
|
|
void plClient::ShutdownDLLs() |
|
{ |
|
int j; |
|
for( j = 0; j < fLoadedDLLs.GetCount(); j++ ) |
|
{ |
|
BOOL ret = FreeLibrary(fLoadedDLLs[j]); |
|
if( !ret ) |
|
hsStatusMessage("Failed to free lib\n"); |
|
} |
|
fLoadedDLLs.Reset(); |
|
} |
|
|
|
bool plClient::MainLoop() |
|
{ |
|
#if defined(HAVE_CYPYTHONIDE) && !defined(PLASMA_EXTERNAL_RELEASE) |
|
if (PythonInterface::UsePythonDebugger()) |
|
{ |
|
PythonInterface::PythonDebugger()->Update(); |
|
|
|
if (PythonInterface::PythonDebugger()->IsConnected()) |
|
{ |
|
bPythonDebugConnected = true; |
|
if (PythonInterface::DebuggerRequestedExit() && PythonInterface::PythonDebugger()->ExitOnStop()) |
|
SetDone(true); // debugger requested that we stop running, so exit nicely |
|
} |
|
else |
|
bPythonDebugConnected = false; |
|
} |
|
#endif |
|
|
|
#ifdef PLASMA_EXTERNAL_RELEASE |
|
if (DebugIsDebuggerPresent()) |
|
{ |
|
NetCliAuthLogClientDebuggerConnect(); |
|
SetDone(true); |
|
} |
|
#endif |
|
|
|
if(plClient::fDelayMS) |
|
Sleep(5); |
|
|
|
// Reset our stats |
|
plProfileManager::Instance().BeginFrame(); |
|
|
|
if (IUpdate()) |
|
return true; |
|
|
|
if (IDraw()) |
|
return true; |
|
|
|
plProfileManagerFull::Instance().EndFrame(); |
|
plProfileManager::Instance().EndFrame(); |
|
|
|
// Draw the stats |
|
plProfileManagerFull::Instance().Update(); |
|
|
|
return false; |
|
} |
|
|
|
#include "plProfile.h" |
|
|
|
plProfile_Extern(DrawTime); |
|
plProfile_Extern(UpdateTime); |
|
plProfile_CreateTimer("ResMgr", "Update", ResMgr); |
|
plProfile_CreateTimer("DispatchQueue", "Update", DispatchQueue); |
|
plProfile_CreateTimer("RenderSetup", "Update", RenderMsg); |
|
plProfile_CreateTimer("Simulation", "Update", Simulation); |
|
plProfile_CreateTimer("NetTime", "Update", UpdateNetTime); |
|
plProfile_Extern(TimeMsg); |
|
plProfile_Extern(EvalMsg); |
|
plProfile_Extern(TransformMsg); |
|
plProfile_Extern(CameraMsg); |
|
plProfile_Extern(AnimatingPhysicals); |
|
plProfile_Extern(StoppedAnimPhysicals); |
|
|
|
plProfile_CreateTimer("BeginRender", "Render", BeginRender); |
|
plProfile_CreateTimer("ClearRender", "Render", ClearRender); |
|
plProfile_CreateTimer("PreRender", "Render", PreRender); |
|
plProfile_CreateTimer("MainRender", "Render", MainRender); |
|
plProfile_CreateTimer("PostRender", "Render", PostRender); |
|
plProfile_CreateTimer("Movies", "Render", Movies); |
|
plProfile_CreateTimer("Console", "Render", Console); |
|
plProfile_CreateTimer("StatusLog", "Render", StatusLog); |
|
plProfile_CreateTimer("ProgressMgr", "Render", ProgressMgr); |
|
plProfile_CreateTimer("ScreenElem", "Render", ScreenElem); |
|
plProfile_CreateTimer("EndRender", "Render", EndRender); |
|
|
|
|
|
bool plClient::IUpdate() |
|
{ |
|
plProfile_BeginTiming(UpdateTime); |
|
|
|
// reset timer on first frame if realtime and not clamping, to avoid initial large delta |
|
if (hsTimer::GetSysSeconds()==0 && hsTimer::IsRealTime() && hsTimer::GetTimeClamp()==0) |
|
hsTimer::SetRealTime(true); |
|
|
|
plProfile_BeginTiming(DispatchQueue); |
|
plgDispatch::Dispatch()->MsgQueueProcess(); |
|
plProfile_EndTiming(DispatchQueue); |
|
|
|
const char *inputUpdate = "Update"; |
|
if (fInputManager) // Is this used anymore? Seems to always be nil. |
|
fInputManager->Update(); |
|
|
|
hsTimer::IncSysSeconds(); |
|
plClientUnifiedTime::SetSysTime(); // keep a unified time, based on sysSeconds |
|
// Time may have been clamped in IncSysSeconds, depending on hsTimer's current mode. |
|
|
|
double currTime = hsTimer::GetSysSeconds(); |
|
float delSecs = hsTimer::GetDelSysSeconds(); |
|
|
|
// do not change this ordering |
|
|
|
plProfile_BeginTiming(UpdateNetTime); |
|
plNetClientMgr::GetInstance()->Update(currTime); |
|
plProfile_EndTiming(UpdateNetTime); |
|
|
|
// update python |
|
//plCaptureRender::Update(fPipeline); |
|
plCaptureRender::Update(); |
|
cyMisc::Update(currTime); |
|
|
|
// This TimeMsg doesn't really do much, except somehow it flushes the dispatch |
|
// after the NetClientMgr updates, delivering any SelfDestruct messages in the |
|
// queue. This is important to prevent objects that are about to go away from |
|
// starting trouble during their update. So to get rid of this message, some |
|
// other way of flushing the dispatch after NegClientMgr's update is needed. mf |
|
plProfile_BeginTiming(TimeMsg); |
|
plTimeMsg* msg = new plTimeMsg(nil, nil, nil, nil); |
|
plgDispatch::MsgSend(msg); |
|
plProfile_EndTiming(TimeMsg); |
|
|
|
plProfile_BeginTiming(EvalMsg); |
|
plEvalMsg* eval = new plEvalMsg(nil, nil, nil, nil); |
|
plgDispatch::MsgSend(eval); |
|
plProfile_EndTiming(EvalMsg); |
|
|
|
char *xFormLap1 = "Main"; |
|
plProfile_BeginLap(TransformMsg, xFormLap1); |
|
plTransformMsg* xform = new plTransformMsg(nil, nil, nil, nil); |
|
plgDispatch::MsgSend(xform); |
|
plProfile_EndLap(TransformMsg, xFormLap1); |
|
|
|
plCoordinateInterface::SetTransformPhase(plCoordinateInterface::kTransformPhaseDelayed); |
|
|
|
if (fAnimDebugList) |
|
fAnimDebugList->ShowReport(); |
|
|
|
plProfile_BeginTiming(Simulation); |
|
plSimulationMgr::GetInstance()->Advance(delSecs); |
|
plProfile_EndTiming(Simulation); |
|
|
|
// At this point, we just register for a plDelayedTransformMsg when dirtied. |
|
if (!plCoordinateInterface::GetDelayedTransformsEnabled()) |
|
{ |
|
char *xFormLap2 = "Simulation"; |
|
plProfile_BeginLap(TransformMsg, xFormLap2); |
|
xform = new plTransformMsg(nil, nil, nil, nil); |
|
plgDispatch::MsgSend(xform); |
|
plProfile_EndLap(TransformMsg, xFormLap2); |
|
} |
|
else |
|
{ |
|
char *xFormLap3 = "Delayed"; |
|
plProfile_BeginLap(TransformMsg, xFormLap3); |
|
xform = new plDelayedTransformMsg(nil, nil, nil, nil); |
|
plgDispatch::MsgSend(xform); |
|
plProfile_EndLap(TransformMsg, xFormLap3); |
|
} |
|
|
|
plCoordinateInterface::SetTransformPhase(plCoordinateInterface::kTransformPhaseNormal); |
|
|
|
plProfile_BeginTiming(CameraMsg); |
|
plCameraMsg* cameras = new plCameraMsg; |
|
cameras->SetCmd(plCameraMsg::kUpdateCameras); |
|
cameras->SetBCastFlag(plMessage::kBCastByExactType); |
|
plgDispatch::MsgSend(cameras); |
|
plProfile_EndTiming(CameraMsg); |
|
|
|
return false; |
|
} |
|
|
|
bool plClient::IDrawProgress() { |
|
// Reset our stats |
|
plProfileManager::Instance().BeginFrame(); |
|
|
|
plProfile_BeginTiming(DrawTime); |
|
if( fPipeline->BeginRender() ) |
|
{ |
|
return IFlushRenderRequests(); |
|
} |
|
// Override the clear color to black. |
|
fPipeline->ClearRenderTarget(&hsColorRGBA().Set(0.f, 0.f, 0.f, 1.f)); |
|
|
|
#ifndef PLASMA_EXTERNAL_RELEASE |
|
fConsole->Draw( fPipeline ); |
|
#endif |
|
|
|
plStatusLogMgr::GetInstance().Draw(); |
|
plProgressMgr::GetInstance()->Draw( fPipeline ); |
|
fPipeline->RenderScreenElements(); |
|
fPipeline->EndRender(); |
|
plProfile_EndTiming(DrawTime); |
|
|
|
plProfileManager::Instance().EndFrame(); |
|
|
|
return false; |
|
} |
|
|
|
bool plClient::IDraw() |
|
{ |
|
// If we're shutting down, don't attempt to draw. Doing so |
|
// tends to cause a device reload each frame. |
|
if (fDone) |
|
return true; |
|
|
|
if (plProgressMgr::GetInstance()->IsActive()) |
|
return IDrawProgress(); |
|
|
|
plProfile_Extern(VisEval); |
|
plProfile_BeginTiming(VisEval); |
|
plGlobalVisMgr::Instance()->Eval(fPipeline->GetViewPositionWorld()); |
|
plProfile_EndTiming(VisEval); |
|
|
|
plProfile_BeginTiming(RenderMsg); |
|
plRenderMsg* rendMsg = new plRenderMsg(fPipeline); |
|
plgDispatch::MsgSend(rendMsg); |
|
plProfile_EndTiming(RenderMsg); |
|
|
|
plPreResourceMsg* preMsg = new plPreResourceMsg(fPipeline); |
|
plgDispatch::MsgSend(preMsg); |
|
|
|
// This might not be the ideal place for this, but it |
|
// needs to be AFTER the plRenderMsg is sent, and |
|
// BEFORE BeginRender. (plRenderMsg causes construction of |
|
// Dynamic objects (e.g. RT's), BeginRender uses them (e.g. shadows). |
|
if( plPipeResReq::Check() || fPipeline->CheckResources() ) |
|
{ |
|
fPipeline->LoadResources(); |
|
} |
|
|
|
plProfile_EndTiming(UpdateTime); |
|
|
|
plProfile_BeginTiming(DrawTime); |
|
|
|
plProfile_BeginTiming(BeginRender); |
|
if( fPipeline->BeginRender() ) |
|
{ |
|
plProfile_EndTiming(BeginRender); |
|
return IFlushRenderRequests(); |
|
} |
|
plProfile_EndTiming(BeginRender); |
|
|
|
plProfile_BeginTiming(ClearRender); |
|
fPipeline->ClearRenderTarget(); |
|
plProfile_EndTiming(ClearRender); |
|
|
|
plProfile_BeginTiming(PreRender); |
|
if( !fFlags.IsBitSet( kFlagDBGDisableRRequests ) ) |
|
IProcessPreRenderRequests(); |
|
plProfile_EndTiming(PreRender); |
|
|
|
plProfile_BeginTiming(MainRender); |
|
if( !fFlags.IsBitSet( kFlagDBGDisableRender ) ) |
|
fPageMgr->Render(fPipeline); |
|
plProfile_EndTiming(MainRender); |
|
|
|
plProfile_BeginTiming(PostRender); |
|
if( !fFlags.IsBitSet( kFlagDBGDisableRRequests ) ) |
|
IProcessPostRenderRequests(); |
|
plProfile_EndTiming(PostRender); |
|
|
|
plProfile_BeginTiming(Movies); |
|
IServiceMovies(); |
|
plProfile_EndTiming(Movies); |
|
|
|
#ifndef PLASMA_EXTERNAL_RELEASE |
|
plProfile_BeginTiming(Console); |
|
fConsole->Draw( fPipeline ); |
|
plProfile_EndTiming(Console); |
|
#endif |
|
|
|
plProfile_BeginTiming(StatusLog); |
|
plStatusLogMgr::GetInstance().Draw(); |
|
plProfile_EndTiming(StatusLog); |
|
|
|
plProfile_BeginTiming(ProgressMgr); |
|
plProgressMgr::GetInstance()->Draw( fPipeline ); |
|
plProfile_EndTiming(ProgressMgr); |
|
|
|
fLastProgressUpdate = hsTimer::GetSeconds(); |
|
|
|
plProfile_BeginTiming(ScreenElem); |
|
fPipeline->RenderScreenElements(); |
|
plProfile_EndTiming(ScreenElem); |
|
|
|
plProfile_BeginTiming(EndRender); |
|
fPipeline->EndRender(); |
|
plProfile_EndTiming(EndRender); |
|
|
|
plProfile_EndTiming(DrawTime); |
|
|
|
return false; |
|
} |
|
|
|
void plClient::IServiceMovies() |
|
{ |
|
for (size_t i = 0; i < fMovies.size(); i++) |
|
{ |
|
if (!fMovies[i]->NextFrame()) |
|
{ |
|
delete fMovies[i]; |
|
fMovies[i] = fMovies.back(); |
|
fMovies.pop_back(); |
|
i--; |
|
} |
|
} |
|
} |
|
|
|
void plClient::IKillMovies() |
|
{ |
|
for (size_t i = 0; i < fMovies.size(); i++) |
|
delete fMovies[i]; |
|
fMovies.clear(); |
|
} |
|
|
|
bool plClient::IPlayIntroMovie(const char* movieName, float endDelay, float posX, float posY, float scaleX, float scaleY, float volume /* = 1.0 */) |
|
{ |
|
SetQuitIntro(false); |
|
plMoviePlayer player; |
|
player.SetPosition(posX, posY); |
|
player.SetScale(scaleX, scaleY); |
|
player.SetFileName(movieName); |
|
player.SetFadeToTime(endDelay); |
|
player.SetFadeToColor(hsColorRGBA().Set(0, 0, 0, 1.f)); |
|
player.SetVolume(volume); |
|
bool firstTry = true; // flag to make sure that we don't quit before we even start |
|
|
|
if (player.Start()) |
|
{ |
|
while (true) |
|
{ |
|
if (fInstance) |
|
fInstance->fMessagePumpProc(); |
|
|
|
if (GetDone()) |
|
return true; |
|
if (firstTry) |
|
{ |
|
firstTry = false; |
|
SetQuitIntro(false); |
|
} |
|
else |
|
{ |
|
if (GetQuitIntro()) |
|
return true; |
|
} |
|
|
|
bool done = false; |
|
if (!fPipeline->BeginRender()) |
|
{ |
|
fPipeline->ClearRenderTarget(); |
|
done = !player.NextFrame(); |
|
|
|
fPipeline->RenderScreenElements(); |
|
fPipeline->EndRender(); |
|
} |
|
|
|
if (done) |
|
return true; |
|
} |
|
return true; |
|
} |
|
return false; |
|
} |
|
|
|
bool plClient::IFlushRenderRequests() |
|
{ |
|
// For those requesting ack's, we could go through and send them |
|
// mail telling them their request was ill-timed. But hopefully, |
|
// the lack of an acknowledgement will serve as notice. |
|
int i; |
|
for( i = 0; i < fPreRenderRequests.GetCount(); i++ ) |
|
hsRefCnt_SafeUnRef(fPreRenderRequests[i]); |
|
fPreRenderRequests.Reset(); |
|
|
|
for( i = 0; i < fPostRenderRequests.GetCount(); i++ ) |
|
hsRefCnt_SafeUnRef(fPostRenderRequests[i]); |
|
fPostRenderRequests.Reset(); |
|
|
|
return false; |
|
} |
|
|
|
void plClient::IProcessRenderRequests(hsTArray<plRenderRequest*>& reqs) |
|
{ |
|
int i; |
|
for( i = 0; i < reqs.GetCount(); i++ ) |
|
{ |
|
reqs[i]->Render(fPipeline, fPageMgr); |
|
hsRefCnt_SafeUnRef(reqs[i]); |
|
} |
|
reqs.SetCount(0); |
|
} |
|
|
|
void plClient::IProcessPreRenderRequests() |
|
{ |
|
IProcessRenderRequests(fPreRenderRequests); |
|
} |
|
|
|
void plClient::IProcessPostRenderRequests() |
|
{ |
|
IProcessRenderRequests(fPostRenderRequests); |
|
} |
|
|
|
void plClient::IAddRenderRequest(plRenderRequest* req) |
|
{ |
|
if( req->GetPriority() < 0 ) |
|
{ |
|
int i; |
|
for( i = 0; i < fPreRenderRequests.GetCount(); i++ ) |
|
{ |
|
if( req->GetPriority() < fPreRenderRequests[i]->GetPriority() ) |
|
break; |
|
} |
|
fPreRenderRequests.Insert(i, req); |
|
hsRefCnt_SafeRef(req); |
|
} |
|
else |
|
{ |
|
int i; |
|
for( i = 0; i < fPostRenderRequests.GetCount(); i++ ) |
|
{ |
|
if( req->GetPriority() < fPostRenderRequests[i]->GetPriority() ) |
|
break; |
|
} |
|
fPostRenderRequests.Insert(i, req); |
|
hsRefCnt_SafeRef(req); |
|
} |
|
} |
|
|
|
void plClient::ResetDisplayDevice(int Width, int Height, int ColorDepth, bool Windowed, int NumAASamples, int MaxAnisotropicSamples, bool VSync) |
|
{ |
|
if(!fPipeline) return; |
|
|
|
WindowActivate(false); |
|
|
|
fPipeline->ResetDisplayDevice(Width, Height, ColorDepth, Windowed, NumAASamples, MaxAnisotropicSamples, VSync); |
|
|
|
ResizeDisplayDevice(Width, Height, Windowed); |
|
|
|
WindowActivate(true); |
|
} |
|
|
|
void plClient::ResizeDisplayDevice(int Width, int Height, bool Windowed) |
|
{ |
|
|
|
if (plMouseDevice::Instance()) |
|
plMouseDevice::Instance()->SetDisplayResolution((float)Width, (float)Height); |
|
|
|
float aspectratio = (float)Width / (float)Height; |
|
if (pfGameGUIMgr::GetInstance()) |
|
pfGameGUIMgr::GetInstance()->SetAspectRatio( aspectratio ); |
|
|
|
|
|
uint32_t winStyle, winExStyle; |
|
if( Windowed ) |
|
{ |
|
// WS_VISIBLE appears necessary to avoid leaving behind framebuffer junk when going from windowed to a smaller window |
|
winStyle = WS_OVERLAPPED | WS_CAPTION | WS_SYSMENU | WS_MINIMIZEBOX | WS_VISIBLE; |
|
winExStyle = WS_EX_APPWINDOW | WS_EX_WINDOWEDGE; |
|
} else { |
|
winStyle = WS_POPUP; |
|
winExStyle = WS_EX_APPWINDOW; |
|
} |
|
SetWindowLong(fWindowHndl, GWL_STYLE, winStyle); |
|
SetWindowLong(fWindowHndl, GWL_EXSTYLE, winExStyle); |
|
|
|
|
|
uint32_t flags = SWP_NOCOPYBITS | SWP_SHOWWINDOW | SWP_FRAMECHANGED; |
|
uint32_t OutsideWidth, OutsideHeight; |
|
HWND insertAfter; |
|
if( Windowed ) |
|
{ |
|
RECT winRect = { 0, 0, Width, Height }; |
|
AdjustWindowRectEx(&winRect, winStyle, false, winExStyle); |
|
OutsideWidth = winRect.right - winRect.left; |
|
OutsideHeight = winRect.bottom - winRect.top; |
|
insertAfter = HWND_NOTOPMOST; |
|
} else { |
|
OutsideWidth = Width; |
|
OutsideHeight = Height; |
|
insertAfter = HWND_TOP; |
|
} |
|
SetWindowPos( fWindowHndl, insertAfter, 0, 0, OutsideWidth, OutsideHeight, flags ); |
|
} |
|
|
|
void WriteBool(hsStream *stream, char *name, bool on ) |
|
{ |
|
char command[256]; |
|
sprintf(command, "%s %s\r\n", name, on ? "true" : "false"); |
|
stream->WriteString(command); |
|
} |
|
|
|
void WriteInt(hsStream *stream, char *name, int val ) |
|
{ |
|
char command[256]; |
|
sprintf(command, "%s %d\r\n", name, val); |
|
stream->WriteString(command); |
|
} |
|
|
|
void WriteString(hsStream *stream, const char *name, const char *val) |
|
{ |
|
char command[256]; |
|
sprintf(command, "%s %s\r\n", name, val); |
|
stream->WriteString(command); |
|
} |
|
|
|
// Detect audio/video settings and save them to their respective ini file, if ini files don't exist |
|
void plClient::IDetectAudioVideoSettings() |
|
{ |
|
// Setup default pipeline settings |
|
bool devmode = true; |
|
hsG3DDeviceModeRecord dmr; |
|
hsG3DDeviceSelector devSel; |
|
devSel.Enumerate(fWindowHndl); |
|
devSel.RemoveUnusableDevModes(true); |
|
|
|
if (!devSel.GetDefault(&dmr)) |
|
devmode = false; |
|
hsG3DDeviceRecord *rec = (hsG3DDeviceRecord *)dmr.GetDevice(); |
|
const hsG3DDeviceMode *mode = dmr.GetMode(); |
|
|
|
bool pixelshaders = rec->GetCap(hsG3DDeviceSelector::kCapsPixelShader); |
|
|
|
plPipeline::fDefaultPipeParams.ColorDepth = hsG3DDeviceSelector::kDefaultDepth; |
|
#if defined(HS_DEBUGGING) || defined(DEBUG) |
|
plPipeline::fDefaultPipeParams.Windowed = true; |
|
#else |
|
plPipeline::fDefaultPipeParams.Windowed = false; |
|
#endif |
|
|
|
// Use current desktop resolution for fullscreen mode |
|
if(!plPipeline::fDefaultPipeParams.Windowed) |
|
{ |
|
plPipeline::fDefaultPipeParams.Width = GetSystemMetrics(SM_CXSCREEN); |
|
plPipeline::fDefaultPipeParams.Height = GetSystemMetrics(SM_CYSCREEN); |
|
} |
|
else |
|
{ |
|
plPipeline::fDefaultPipeParams.Width = hsG3DDeviceSelector::kDefaultWidth; |
|
plPipeline::fDefaultPipeParams.Height = hsG3DDeviceSelector::kDefaultHeight; |
|
} |
|
|
|
plPipeline::fDefaultPipeParams.Shadows = 1; |
|
|
|
// enable planar reflections if pixelshaders are available |
|
plPipeline::fDefaultPipeParams.PlanarReflections = 1; |
|
|
|
// enable 2x antialiasing and anisotropic to 2 samples if pixelshader version is greater that 2.0 |
|
plPipeline::fDefaultPipeParams.AntiAliasingAmount = rec->GetMaxAnisotropicSamples() ? 2 : 0; |
|
plPipeline::fDefaultPipeParams.AnisotropicLevel = mode->GetNumFSAATypes() ? 2 : 0; |
|
|
|
plPipeline::fDefaultPipeParams.TextureQuality = pixelshaders ? 2 : 1; |
|
plPipeline::fDefaultPipeParams.VideoQuality = pixelshaders ? 2 : 1; |
|
|
|
plPipeline::fDefaultPipeParams.VSync = false; |
|
|
|
int val = 0; |
|
hsStream *stream = nil; |
|
hsUNIXStream s; |
|
plFileName audioIniFile = plFileName::Join(plFileSystem::GetInitPath(), "audio.ini"); |
|
plFileName graphicsIniFile = plFileName::Join(plFileSystem::GetInitPath(), "graphics.ini"); |
|
|
|
#ifndef PLASMA_EXTERNAL_RELEASE |
|
// internal builds can use the local dir |
|
if (plFileInfo("init/audio.ini").Exists()) |
|
audioIniFile = "init/audio.ini"; |
|
if (plFileInfo("init/graphics.ini").Exists()) |
|
graphicsIniFile = "init/graphics.ini"; |
|
#endif |
|
|
|
//check to see if audio.ini exists |
|
if (s.Open(audioIniFile)) |
|
{ |
|
s.Close(); |
|
} |
|
else |
|
{ |
|
stream = plEncryptedStream::OpenEncryptedFileWrite(audioIniFile); |
|
|
|
plAudioCaps caps = plAudioCapsDetector::Detect(false, true); |
|
|
|
char deviceName[256]; |
|
sprintf(deviceName, "\"%s\"", DEFAULT_AUDIO_DEVICE_NAME); |
|
|
|
WriteBool(stream, "Audio.Initialize", caps.IsAvailable()); |
|
WriteBool(stream, "Audio.UseEAX", false); |
|
WriteInt(stream, "Audio.SetPriorityCutoff", 6); |
|
WriteInt(stream, "Audio.MuteAll", false); |
|
WriteInt(stream, "Audio.SetChannelVolume SoundFX", 1); |
|
WriteInt(stream, "Audio.SetChannelVolume BgndMusic", 1); |
|
WriteInt(stream, "Audio.SetChannelVolume Ambience", 1); |
|
WriteInt(stream, "Audio.SetChannelVolume NPCVoice", 1); |
|
WriteInt(stream, "Audio.EnableVoiceRecording", 1); |
|
WriteString(stream, "Audio.SetDeviceName", deviceName ); |
|
stream->Close(); |
|
delete stream; |
|
stream = nil; |
|
} |
|
|
|
// check to see if graphics.ini exists |
|
if (s.Open(graphicsIniFile)) |
|
{ |
|
s.Close(); |
|
} |
|
else |
|
{ |
|
IWriteDefaultGraphicsSettings(graphicsIniFile); |
|
} |
|
} |
|
|
|
void plClient::IWriteDefaultGraphicsSettings(const plFileName& destFile) |
|
{ |
|
hsStream *stream = plEncryptedStream::OpenEncryptedFileWrite(destFile); |
|
|
|
WriteInt(stream, "Graphics.Width", plPipeline::fDefaultPipeParams.Width); |
|
WriteInt(stream, "Graphics.Height", plPipeline::fDefaultPipeParams.Height); |
|
WriteInt(stream, "Graphics.ColorDepth", plPipeline::fDefaultPipeParams.ColorDepth); |
|
WriteBool(stream, "Graphics.Windowed", plPipeline::fDefaultPipeParams.Windowed); |
|
WriteInt(stream, "Graphics.AntiAliasAmount", plPipeline::fDefaultPipeParams.AntiAliasingAmount); |
|
WriteInt(stream, "Graphics.AnisotropicLevel", plPipeline::fDefaultPipeParams.AnisotropicLevel ); |
|
WriteInt(stream, "Graphics.TextureQuality", plPipeline::fDefaultPipeParams.TextureQuality); |
|
WriteInt(stream, "Quality.Level", plPipeline::fDefaultPipeParams.VideoQuality); |
|
WriteInt(stream, "Graphics.Shadow.Enable", plPipeline::fDefaultPipeParams.Shadows); |
|
WriteInt(stream, "Graphics.EnablePlanarReflections", plPipeline::fDefaultPipeParams.PlanarReflections); |
|
WriteBool(stream, "Graphics.EnableVSync", plPipeline::fDefaultPipeParams.VSync); |
|
stream->Close(); |
|
delete stream; |
|
stream = nil; |
|
} |
|
|
|
|
|
void plClient::WindowActivate(bool active) |
|
{ |
|
if (GetDone()) |
|
return; |
|
|
|
if( !fWindowActive != !active ) |
|
{ |
|
if( fInputManager != nil ) |
|
fInputManager->Activate( active ); |
|
|
|
plArmatureMod::WindowActivate( active ); |
|
} |
|
fWindowActive = active; |
|
} |
|
|
|
void plClient::FlashWindow() |
|
{ |
|
#ifdef HS_BUILD_FOR_WIN32 |
|
FLASHWINFO info; |
|
info.cbSize = sizeof(info); |
|
info.dwFlags = FLASHW_TIMERNOFG | FLASHW_ALL; |
|
info.hwnd = fWindowHndl; |
|
info.uCount = -1; |
|
FlashWindowEx(&info); |
|
#endif |
|
} |
|
|
|
//============================================================================ |
|
void plClient::IOnAsyncInitComplete () { |
|
// Init State Desc Language (files should now be downloaded and in place) |
|
plSDLMgr::GetInstance()->SetNetApp(plNetClientMgr::GetInstance()); |
|
plSDLMgr::GetInstance()->Init( plSDL::kDisallowTimeStamping ); |
|
|
|
PythonInterface::initPython(); |
|
// set the pipeline for the python cyMisc module so that it can do a screen capture |
|
cyMisc::SetPipeline( fPipeline ); |
|
|
|
// Load our custom fonts from our current dat directory |
|
fFontCache->LoadCustomFonts("dat"); |
|
|
|
// We'd like to do a SetHoldLoadRequests here, but the GUI stuff doesn't draw right |
|
// if you try to delay the loading for it. To work around that, we allocate a |
|
// global loading bar in advance and set it to a big enough range that when the GUI's |
|
// are done loading about the right amount of it is filled. |
|
fNumLoadingRooms++; |
|
IStartProgress("Loading Global...", 0); |
|
|
|
/// Init the KI |
|
pfGameGUIMgr *mgr = pfGameGUIMgr::GetInstance(); |
|
mgr->LoadDialog( "KIBlackBar" ); // load the blackbar which will bootstrap in the rest of the KI dialogs |
|
|
|
// Init the journal book API |
|
pfJournalBook::SingletonInit(); |
|
|
|
SetHoldLoadRequests(true); |
|
fProgressBar->SetLength(fProgressBar->GetProgress()); |
|
|
|
plClothingMgr::Init(); |
|
// Load in any clothing data |
|
((plResManager*)hsgResMgr::ResMgr())->PageInAge("GlobalClothing"); |
|
|
|
pfMarkerMgr::Instance(); |
|
|
|
fAnimDebugList = new plAnimDebugList(); |
|
|
|
/// Now parse final init files (*.fni). These are files just like ini files, only to be run |
|
/// after all hell has broken loose in the client. |
|
plFileName initFolder = plFileSystem::GetInitPath(); |
|
pfConsoleDirSrc dirSrc(fConsoleEngine, initFolder, "net*.fni"); // connect to net first |
|
#ifndef PLASMA_EXTERNAL_RELEASE |
|
// internal builds also parse the local init folder |
|
dirSrc.ParseDirectory("init", "net*.fni"); |
|
#endif |
|
|
|
dirSrc.ParseDirectory(initFolder, "*.fni"); |
|
#ifndef PLASMA_EXTERNAL_RELEASE |
|
// internal builds also parse the local init folder |
|
dirSrc.ParseDirectory("init", "*.fni"); |
|
#endif |
|
|
|
// run fni in the Aux Init dir |
|
if (fpAuxInitDir) |
|
{ |
|
dirSrc.ParseDirectory(fpAuxInitDir, "net*.fni"); // connect to net first |
|
dirSrc.ParseDirectory(fpAuxInitDir, "*.fni"); |
|
} |
|
|
|
fNumLoadingRooms--; |
|
|
|
((plResManager*)hsgResMgr::ResMgr())->PageInAge("GlobalAnimations"); |
|
SetHoldLoadRequests(false); |
|
|
|
// Tell the transition manager to start faded out. This is so we don't |
|
// get a frame or two of non-faded drawing before we do our initial fade in |
|
(void)(new plTransitionMsg( plTransitionMsg::kFadeOut, 0.0f, true ))->Send(); |
|
|
|
fFlags.SetBit(kFlagAsyncInitComplete); |
|
if (fFlags.IsBitSet(kFlagGlobalDataLoaded)) |
|
ICompleteInit(); |
|
} |
|
|
|
//============================================================================ |
|
void plClient::ICompleteInit () { |
|
// Reset clear color on the pipeline |
|
// fPipeline->ClearRenderTarget( &fClearColor, &depth ); |
|
|
|
plSimulationMgr::GetInstance()->Resume(); // start the sim at the last possible minute |
|
|
|
fFlags.SetBit( kFlagIniting, false ); |
|
hsStatusMessage("Client init complete."); |
|
|
|
// Tell everyone we're ready to rock. |
|
plClientMsg* clientMsg = new plClientMsg(plClientMsg::kInitComplete); |
|
clientMsg->SetBCastFlag(plMessage::kBCastByType); |
|
clientMsg->Send(); |
|
} |
|
|
|
//============================================================================ |
|
void plClient::IHandlePatcherMsg (plResPatcherMsg * msg) { |
|
plgDispatch::Dispatch()->UnRegisterForExactType(plResPatcherMsg::Index(), GetKey()); |
|
|
|
if (!msg->Success()) { |
|
plNetClientApp::GetInstance()->QueueDisableNet(true, msg->GetError().c_str()); |
|
return; |
|
} |
|
|
|
IOnAsyncInitComplete(); |
|
}
|
|
|