mirror of
https://github.com/ptitSeb/Serious-Engine
synced 2024-11-23 19:00:27 +01:00
1436 lines
45 KiB
C++
1436 lines
45 KiB
C++
/* Copyright (c) 2002-2012 Croteam Ltd. All rights reserved. */
|
|
|
|
#include "StdH.h"
|
|
#include <io.h>
|
|
#include <fcntl.h>
|
|
#include <sys/stat.h>
|
|
#include <process.h>
|
|
#include <Engine/CurrentVersion.h>
|
|
#include <GameMP/Game.h>
|
|
#define DECL_DLL
|
|
#include <EntitiesMP/Global.h>
|
|
#include "resource.h"
|
|
#include "SplashScreen.h"
|
|
#include "MainWindow.h"
|
|
#include "GlSettings.h"
|
|
#include "LevelInfo.h"
|
|
#include "LCDDrawing.h"
|
|
#include "CmdLine.h"
|
|
#include "Credits.h"
|
|
|
|
|
|
extern CGame *_pGame = NULL;
|
|
|
|
// application state variables
|
|
extern BOOL _bRunning = TRUE;
|
|
extern BOOL _bQuitScreen = TRUE;
|
|
extern BOOL bMenuActive = FALSE;
|
|
extern BOOL bMenuRendering = FALSE;
|
|
|
|
extern BOOL _bDefiningKey;
|
|
static BOOL _bReconsiderInput = FALSE;
|
|
extern PIX _pixDesktopWidth = 0; // desktop width when started (for some tests)
|
|
|
|
static INDEX sam_iMaxFPSActive = 500;
|
|
static INDEX sam_iMaxFPSInactive = 10;
|
|
static INDEX sam_bPauseOnMinimize = TRUE; // auto-pause when window has been minimized
|
|
extern INDEX sam_bWideScreen = FALSE;
|
|
extern FLOAT sam_fPlayerOffset = 0.0f;
|
|
|
|
// display mode settings
|
|
extern INDEX sam_bFullScreenActive = FALSE;
|
|
extern INDEX sam_iScreenSizeI = 1024; // current size of the window
|
|
extern INDEX sam_iScreenSizeJ = 768; // current size of the window
|
|
extern INDEX sam_iDisplayDepth = 0; // 0==default, 1==16bit, 2==32bit
|
|
extern INDEX sam_iDisplayAdapter = 0;
|
|
extern INDEX sam_iGfxAPI = 0; // 0==OpenGL
|
|
extern INDEX sam_bFirstStarted = FALSE;
|
|
extern FLOAT sam_tmDisplayModeReport = 5.0f;
|
|
extern INDEX sam_bShowAllLevels = FALSE;
|
|
extern INDEX sam_bMentalActivated = FALSE;
|
|
|
|
// network settings
|
|
extern CTString sam_strNetworkSettings = "";
|
|
// command line
|
|
extern CTString sam_strCommandLine = "";
|
|
|
|
// 0...app started for the first time
|
|
// 1...all ok
|
|
// 2...automatic fallback
|
|
static INDEX _iDisplayModeChangeFlag = 0;
|
|
static TIME _tmDisplayModeChanged = 100.0f; // when display mode was last changed
|
|
|
|
// rendering preferences for automatic settings
|
|
extern INDEX sam_iVideoSetup = 1; // 0==speed, 1==normal, 2==quality, 3==custom
|
|
// automatic adjustment of audio quality
|
|
extern BOOL sam_bAutoAdjustAudio = TRUE;
|
|
|
|
extern INDEX sam_bAutoPlayDemos = TRUE;
|
|
static INDEX _bInAutoPlayLoop = TRUE;
|
|
|
|
// menu calling
|
|
extern INDEX sam_bMenuSave = FALSE;
|
|
extern INDEX sam_bMenuLoad = FALSE;
|
|
extern INDEX sam_bMenuControls = FALSE;
|
|
extern INDEX sam_bMenuHiScore = FALSE;
|
|
extern INDEX sam_bToggleConsole = FALSE;
|
|
extern INDEX sam_iStartCredits = FALSE;
|
|
|
|
// for mod re-loading
|
|
extern CTFileName _fnmModToLoad = CTString("");
|
|
extern CTString _strModServerJoin = CTString("");
|
|
extern CTString _strURLToVisit = CTString("");
|
|
|
|
|
|
// state variables fo addon execution
|
|
// 0 - nothing
|
|
// 1 - start (invoke console)
|
|
// 2 - console invoked, waiting for one redraw
|
|
extern INDEX _iAddonExecState = 0;
|
|
extern CTFileName _fnmAddonToExec = CTString("");
|
|
|
|
// logo textures
|
|
static CTextureObject _toLogoCT;
|
|
static CTextureObject _toLogoODI;
|
|
static CTextureObject _toLogoEAX;
|
|
extern CTextureObject *_ptoLogoCT = NULL;
|
|
extern CTextureObject *_ptoLogoODI = NULL;
|
|
extern CTextureObject *_ptoLogoEAX = NULL;
|
|
|
|
extern CTString sam_strVersion = "1.10";
|
|
extern CTString sam_strModName = TRANS("- O P E N S O U R C E -");
|
|
#if _SE_DEMO
|
|
extern CTString sam_strFirstLevel = "Levels\\KarnakDemo.wld";
|
|
#else
|
|
extern CTString sam_strFirstLevel = "Levels\\LevelsMP\\1_0_InTheLastEpisode.wld.wld";
|
|
#endif
|
|
extern CTString sam_strIntroLevel = "Levels\\LevelsMP\\Intro.wld";
|
|
extern CTString sam_strGameName = "serioussamse";
|
|
|
|
extern CTString sam_strTechTestLevel = "Levels\\LevelsMP\\TechTest.wld";
|
|
extern CTString sam_strTrainingLevel = "Levels\\KarnakDemo.wld";
|
|
|
|
ENGINE_API extern INDEX snd_iFormat;
|
|
|
|
|
|
// main window canvas
|
|
CDrawPort *pdp;
|
|
CDrawPort *pdpNormal;
|
|
CDrawPort *pdpWideScreen;
|
|
CViewPort *pvpViewPort;
|
|
HINSTANCE _hInstance;
|
|
|
|
|
|
static void PlayDemo(void* pArgs)
|
|
{
|
|
CTString strDemoFilename = *NEXTARGUMENT(CTString*);
|
|
_gmMenuGameMode = GM_DEMO;
|
|
CTFileName fnDemo = "demos\\" + strDemoFilename + ".dem";
|
|
extern BOOL LSLoadDemo(const CTFileName &fnm);
|
|
LSLoadDemo(fnDemo);
|
|
}
|
|
|
|
static void ApplyRenderingPreferences(void)
|
|
{
|
|
ApplyGLSettings(TRUE);
|
|
}
|
|
|
|
extern void ApplyVideoMode(void)
|
|
{
|
|
StartNewMode( (GfxAPIType)sam_iGfxAPI, sam_iDisplayAdapter, sam_iScreenSizeI, sam_iScreenSizeJ,
|
|
(enum DisplayDepth)sam_iDisplayDepth, sam_bFullScreenActive);
|
|
}
|
|
|
|
static void BenchMark(void)
|
|
{
|
|
_pGfx->Benchmark(pvpViewPort, pdp);
|
|
}
|
|
|
|
|
|
static void QuitGame(void)
|
|
{
|
|
_bRunning = FALSE;
|
|
_bQuitScreen = FALSE;
|
|
}
|
|
|
|
// check if another app is already running
|
|
static HANDLE _hLock = NULL;
|
|
static CTFileName _fnmLock;
|
|
static void DirectoryLockOn(void)
|
|
{
|
|
// create lock filename
|
|
_fnmLock = _fnmApplicationPath+"SeriousSam.loc";
|
|
// try to open lock file
|
|
_hLock = CreateFileA(
|
|
_fnmLock,
|
|
GENERIC_WRITE,
|
|
0/*no sharing*/,
|
|
NULL, // pointer to security attributes
|
|
CREATE_ALWAYS,
|
|
FILE_ATTRIBUTE_NORMAL|FILE_FLAG_DELETE_ON_CLOSE, // file attributes
|
|
NULL);
|
|
// if failed
|
|
if (_hLock==NULL || GetLastError()!=0) {
|
|
// report warning
|
|
CPrintF(TRANS("WARNING: SeriousSam didn't shut down properly last time!\n"));
|
|
}
|
|
}
|
|
static void DirectoryLockOff(void)
|
|
{
|
|
// if lock is open
|
|
if (_hLock!=NULL) {
|
|
// close it
|
|
CloseHandle(_hLock);
|
|
}
|
|
}
|
|
|
|
void End(void);
|
|
|
|
|
|
// automaticaly manage input enable/disable toggling
|
|
static BOOL _bInputEnabled = FALSE;
|
|
void UpdateInputEnabledState(void)
|
|
{
|
|
// do nothing if window is invalid
|
|
if( _hwndMain==NULL) return;
|
|
|
|
// input should be enabled if application is active
|
|
// and no menu is active and no console is active
|
|
BOOL bShouldBeEnabled = (!IsIconic(_hwndMain) && !bMenuActive && _pGame->gm_csConsoleState==CS_OFF
|
|
&& (_pGame->gm_csComputerState==CS_OFF || _pGame->gm_csComputerState==CS_ONINBACKGROUND))
|
|
|| _bDefiningKey;
|
|
|
|
// if should be turned off
|
|
if( (!bShouldBeEnabled && _bInputEnabled) || _bReconsiderInput) {
|
|
// disable it and remember new state
|
|
_pInput->DisableInput();
|
|
_bInputEnabled = FALSE;
|
|
}
|
|
// if should be turned on
|
|
if( bShouldBeEnabled && !_bInputEnabled) {
|
|
// enable it and remember new state
|
|
_pInput->EnableInput(_hwndMain);
|
|
_bInputEnabled = TRUE;
|
|
}
|
|
_bReconsiderInput = FALSE;
|
|
}
|
|
|
|
|
|
// automaticaly manage pause toggling
|
|
void UpdatePauseState(void)
|
|
{
|
|
BOOL bShouldPause = (_gmRunningGameMode==GM_SINGLE_PLAYER) && (bMenuActive ||
|
|
_pGame->gm_csConsoleState ==CS_ON || _pGame->gm_csConsoleState ==CS_TURNINGON || _pGame->gm_csConsoleState ==CS_TURNINGOFF ||
|
|
_pGame->gm_csComputerState==CS_ON || _pGame->gm_csComputerState==CS_TURNINGON || _pGame->gm_csComputerState==CS_TURNINGOFF);
|
|
_pNetwork->SetLocalPause(bShouldPause);
|
|
}
|
|
|
|
|
|
// limit current frame rate if neeeded
|
|
void LimitFrameRate(void)
|
|
{
|
|
// measure passed time for each loop
|
|
static CTimerValue tvLast(-1.0f);
|
|
CTimerValue tvNow = _pTimer->GetHighPrecisionTimer();
|
|
TIME tmCurrentDelta = (tvNow-tvLast).GetSeconds();
|
|
|
|
// limit maximum frame rate
|
|
sam_iMaxFPSActive = ClampDn( (INDEX)sam_iMaxFPSActive, 1L);
|
|
sam_iMaxFPSInactive = ClampDn( (INDEX)sam_iMaxFPSInactive, 1L);
|
|
INDEX iMaxFPS = sam_iMaxFPSActive;
|
|
if( IsIconic(_hwndMain)) iMaxFPS = sam_iMaxFPSInactive;
|
|
if(_pGame->gm_CurrentSplitScreenCfg==CGame::SSC_DEDICATED) {
|
|
iMaxFPS = ClampDn(iMaxFPS, 60L); // never go very slow if dedicated server
|
|
}
|
|
TIME tmWantedDelta = 1.0f / iMaxFPS;
|
|
if( tmCurrentDelta<tmWantedDelta) Sleep( (tmWantedDelta-tmCurrentDelta)*1000.0f);
|
|
|
|
// remember new time
|
|
tvLast = _pTimer->GetHighPrecisionTimer();
|
|
}
|
|
|
|
// load first demo
|
|
void StartNextDemo(void)
|
|
{
|
|
if (!sam_bAutoPlayDemos || !_bInAutoPlayLoop) {
|
|
_bInAutoPlayLoop = FALSE;
|
|
return;
|
|
}
|
|
|
|
// skip if no demos
|
|
if(_lhAutoDemos.IsEmpty()) {
|
|
_bInAutoPlayLoop = FALSE;
|
|
return;
|
|
}
|
|
|
|
// get first demo level and cycle the list
|
|
CLevelInfo *pli = LIST_HEAD(_lhAutoDemos, CLevelInfo, li_lnNode);
|
|
pli->li_lnNode.Remove();
|
|
_lhAutoDemos.AddTail(pli->li_lnNode);
|
|
|
|
// if intro
|
|
if (pli->li_fnLevel==sam_strIntroLevel) {
|
|
// start intro
|
|
_gmRunningGameMode = GM_NONE;
|
|
_pGame->gm_aiStartLocalPlayers[0] = 0;
|
|
_pGame->gm_aiStartLocalPlayers[1] = -1;
|
|
_pGame->gm_aiStartLocalPlayers[2] = -1;
|
|
_pGame->gm_aiStartLocalPlayers[3] = -1;
|
|
_pGame->gm_strNetworkProvider = "Local";
|
|
_pGame->gm_StartSplitScreenCfg = CGame::SSC_PLAY1;
|
|
|
|
_pShell->SetINDEX("gam_iStartDifficulty", CSessionProperties::GD_NORMAL);
|
|
_pShell->SetINDEX("gam_iStartMode", CSessionProperties::GM_FLYOVER);
|
|
|
|
CUniversalSessionProperties sp;
|
|
_pGame->SetSinglePlayerSession(sp);
|
|
|
|
_pGame->gm_bFirstLoading = TRUE;
|
|
|
|
if (_pGame->NewGame( sam_strIntroLevel, sam_strIntroLevel, sp)) {
|
|
_gmRunningGameMode = GM_INTRO;
|
|
}
|
|
// if not intro
|
|
} else {
|
|
// start the demo
|
|
_pGame->gm_StartSplitScreenCfg = CGame::SSC_OBSERVER;
|
|
_pGame->gm_aiStartLocalPlayers[0] = -1;
|
|
_pGame->gm_aiStartLocalPlayers[1] = -1;
|
|
_pGame->gm_aiStartLocalPlayers[2] = -1;
|
|
_pGame->gm_aiStartLocalPlayers[3] = -1;
|
|
// play the demo
|
|
_pGame->gm_strNetworkProvider = "Local";
|
|
_gmRunningGameMode = GM_NONE;
|
|
if( _pGame->StartDemoPlay( pli->li_fnLevel)) {
|
|
_gmRunningGameMode = GM_DEMO;
|
|
CON_DiscardLastLineTimes();
|
|
}
|
|
}
|
|
|
|
if (_gmRunningGameMode==GM_NONE) {
|
|
_bInAutoPlayLoop = FALSE;
|
|
}
|
|
}
|
|
|
|
BOOL _bCDPathFound = FALSE;
|
|
|
|
BOOL FileExistsOnHD(const CTString &strFile)
|
|
{
|
|
FILE *f = fopen(_fnmApplicationPath+strFile, "rb");
|
|
if (f!=NULL) {
|
|
fclose(f);
|
|
return TRUE;
|
|
} else {
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
void TrimString(char *str)
|
|
{
|
|
int i = strlen(str);
|
|
if (str[i-1]=='\n' || str[i-1]=='\r') {
|
|
str[i-1]=0;
|
|
}
|
|
}
|
|
|
|
// run web browser and view an url
|
|
void RunBrowser(const char *strUrl)
|
|
{
|
|
int iResult = (int)ShellExecuteA( _hwndMain, "OPEN", strUrl, NULL, NULL, SW_SHOWMAXIMIZED);
|
|
if (iResult<32) {
|
|
// should report error?
|
|
NOTHING;
|
|
}
|
|
}
|
|
|
|
void LoadAndForceTexture(CTextureObject &to, CTextureObject *&pto, const CTFileName &fnm)
|
|
{
|
|
try {
|
|
to.SetData_t(fnm);
|
|
CTextureData *ptd = (CTextureData*)to.GetData();
|
|
ptd->Force( TEX_CONSTANT);
|
|
ptd = ptd->td_ptdBaseTexture;
|
|
if( ptd!=NULL) ptd->Force( TEX_CONSTANT);
|
|
pto = &to;
|
|
} catch( char *pchrError) {
|
|
(void*)pchrError;
|
|
pto = NULL;
|
|
}
|
|
}
|
|
|
|
void InitializeGame(void)
|
|
{
|
|
try {
|
|
#ifndef NDEBUG
|
|
#define GAMEDLL (_fnmApplicationExe.FileDir()+"Game"+_strModExt+"D.dll")
|
|
#else
|
|
#define GAMEDLL (_fnmApplicationExe.FileDir()+"Game"+_strModExt+".dll")
|
|
#endif
|
|
CTFileName fnmExpanded;
|
|
ExpandFilePath(EFP_READ, CTString(GAMEDLL), fnmExpanded);
|
|
|
|
CPrintF(TRANS("Loading game library '%s'...\n"), (const char *)fnmExpanded);
|
|
HMODULE hGame = LoadLibraryA(fnmExpanded);
|
|
if (hGame==NULL) {
|
|
ThrowF_t("%s", GetWindowsError(GetLastError()));
|
|
}
|
|
CGame* (*GAME_Create)(void) = (CGame* (*)(void))GetProcAddress(hGame, "GAME_Create");
|
|
if (GAME_Create==NULL) {
|
|
ThrowF_t("%s", GetWindowsError(GetLastError()));
|
|
}
|
|
_pGame = GAME_Create();
|
|
|
|
} catch (char *strError) {
|
|
FatalError("%s", strError);
|
|
}
|
|
// init game - this will load persistent symbols
|
|
_pGame->Initialize(CTString("Data\\SeriousSam.gms"));
|
|
}
|
|
|
|
BOOL Init( HINSTANCE hInstance, int nCmdShow, CTString strCmdLine)
|
|
{
|
|
_hInstance = hInstance;
|
|
ShowSplashScreen(hInstance);
|
|
|
|
// remember desktop width
|
|
_pixDesktopWidth = ::GetSystemMetrics(SM_CXSCREEN);
|
|
|
|
// prepare main window
|
|
MainWindow_Init();
|
|
OpenMainWindowInvisible();
|
|
|
|
// parse command line before initializing engine
|
|
ParseCommandLine(strCmdLine);
|
|
|
|
// initialize engine
|
|
SE_InitEngine(sam_strGameName);
|
|
|
|
SE_LoadDefaultFonts();
|
|
// now print the output of command line parsing
|
|
CPrintF("%s", cmd_strOutput);
|
|
|
|
// lock the directory
|
|
DirectoryLockOn();
|
|
|
|
// load all translation tables
|
|
InitTranslation();
|
|
try {
|
|
AddTranslationTablesDir_t(CTString("Data\\Translations\\"), CTString("*.txt"));
|
|
FinishTranslationTable();
|
|
} catch (char *strError) {
|
|
FatalError("%s", strError);
|
|
}
|
|
|
|
// always disable all warnings when in serious sam
|
|
_pShell->Execute( "con_bNoWarnings=1;");
|
|
|
|
// declare shell symbols
|
|
_pShell->DeclareSymbol("user void PlayDemo(CTString);", &PlayDemo);
|
|
_pShell->DeclareSymbol("persistent INDEX sam_bFullScreen;", &sam_bFullScreenActive);
|
|
_pShell->DeclareSymbol("persistent INDEX sam_iScreenSizeI;", &sam_iScreenSizeI);
|
|
_pShell->DeclareSymbol("persistent INDEX sam_iScreenSizeJ;", &sam_iScreenSizeJ);
|
|
_pShell->DeclareSymbol("persistent INDEX sam_iDisplayDepth;", &sam_iDisplayDepth);
|
|
_pShell->DeclareSymbol("persistent INDEX sam_iDisplayAdapter;", &sam_iDisplayAdapter);
|
|
_pShell->DeclareSymbol("persistent INDEX sam_iGfxAPI;", &sam_iGfxAPI);
|
|
_pShell->DeclareSymbol("persistent INDEX sam_bFirstStarted;", &sam_bFirstStarted);
|
|
_pShell->DeclareSymbol("persistent INDEX sam_bAutoAdjustAudio;", &sam_bAutoAdjustAudio);
|
|
_pShell->DeclareSymbol("persistent user INDEX sam_bWideScreen;", &sam_bWideScreen);
|
|
_pShell->DeclareSymbol("persistent user FLOAT sam_fPlayerOffset;", &sam_fPlayerOffset);
|
|
_pShell->DeclareSymbol("persistent user INDEX sam_bAutoPlayDemos;", &sam_bAutoPlayDemos);
|
|
_pShell->DeclareSymbol("persistent user INDEX sam_iMaxFPSActive;", &sam_iMaxFPSActive);
|
|
_pShell->DeclareSymbol("persistent user INDEX sam_iMaxFPSInactive;", &sam_iMaxFPSInactive);
|
|
_pShell->DeclareSymbol("persistent user INDEX sam_bPauseOnMinimize;", &sam_bPauseOnMinimize);
|
|
_pShell->DeclareSymbol("persistent user FLOAT sam_tmDisplayModeReport;", &sam_tmDisplayModeReport);
|
|
_pShell->DeclareSymbol("persistent user CTString sam_strNetworkSettings;", &sam_strNetworkSettings);
|
|
_pShell->DeclareSymbol("persistent user CTString sam_strIntroLevel;", &sam_strIntroLevel);
|
|
_pShell->DeclareSymbol("persistent user CTString sam_strGameName;", &sam_strGameName);
|
|
_pShell->DeclareSymbol("user CTString sam_strVersion;", &sam_strVersion);
|
|
_pShell->DeclareSymbol("user CTString sam_strFirstLevel;", &sam_strFirstLevel);
|
|
_pShell->DeclareSymbol("user CTString sam_strModName;", &sam_strModName);
|
|
_pShell->DeclareSymbol("persistent INDEX sam_bShowAllLevels;", &sam_bShowAllLevels);
|
|
_pShell->DeclareSymbol("persistent INDEX sam_bMentalActivated;", &sam_bMentalActivated);
|
|
|
|
_pShell->DeclareSymbol("user CTString sam_strTechTestLevel;", &sam_strTechTestLevel);
|
|
_pShell->DeclareSymbol("user CTString sam_strTrainingLevel;", &sam_strTrainingLevel);
|
|
|
|
_pShell->DeclareSymbol("user void Quit(void);", &QuitGame);
|
|
|
|
_pShell->DeclareSymbol("persistent user INDEX sam_iVideoSetup;", &sam_iVideoSetup);
|
|
_pShell->DeclareSymbol("user void ApplyRenderingPreferences(void);", &ApplyRenderingPreferences);
|
|
_pShell->DeclareSymbol("user void ApplyVideoMode(void);", &ApplyVideoMode);
|
|
_pShell->DeclareSymbol("user void Benchmark(void);", &BenchMark);
|
|
|
|
_pShell->DeclareSymbol("user INDEX sam_bMenuSave;", &sam_bMenuSave);
|
|
_pShell->DeclareSymbol("user INDEX sam_bMenuLoad;", &sam_bMenuLoad);
|
|
_pShell->DeclareSymbol("user INDEX sam_bMenuControls;", &sam_bMenuControls);
|
|
_pShell->DeclareSymbol("user INDEX sam_bMenuHiScore;", &sam_bMenuHiScore);
|
|
_pShell->DeclareSymbol("user INDEX sam_bToggleConsole;",&sam_bToggleConsole);
|
|
_pShell->DeclareSymbol("INDEX sam_iStartCredits;", &sam_iStartCredits);
|
|
|
|
InitializeGame();
|
|
_pNetwork->md_strGameID = sam_strGameName;
|
|
|
|
LCDInit();
|
|
|
|
if( sam_bFirstStarted) {
|
|
InfoMessage("%s", TRANS(
|
|
"SeriousSam is starting for the first time.\n"
|
|
"If you experience any problems, please consult\n"
|
|
"ReadMe file for troubleshooting information."));
|
|
}
|
|
|
|
// initialize sound library
|
|
snd_iFormat = Clamp( snd_iFormat, (INDEX)CSoundLibrary::SF_NONE, (INDEX)CSoundLibrary::SF_44100_16);
|
|
_pSound->SetFormat( (enum CSoundLibrary::SoundFormat)snd_iFormat);
|
|
|
|
if (sam_bAutoAdjustAudio) {
|
|
_pShell->Execute("include \"Scripts\\Addons\\SFX-AutoAdjust.ini\"");
|
|
}
|
|
|
|
// execute script given on command line
|
|
if (cmd_strScript!="") {
|
|
CPrintF("Command line script: '%s'\n", cmd_strScript);
|
|
CTString strCmd;
|
|
strCmd.PrintF("include \"%s\"", cmd_strScript);
|
|
_pShell->Execute(strCmd);
|
|
}
|
|
|
|
// load logo textures
|
|
LoadAndForceTexture(_toLogoCT, _ptoLogoCT, CTFILENAME("Textures\\Logo\\LogoCT.tex"));
|
|
LoadAndForceTexture(_toLogoODI, _ptoLogoODI, CTFILENAME("Textures\\Logo\\GodGamesLogo.tex"));
|
|
LoadAndForceTexture(_toLogoEAX, _ptoLogoEAX, CTFILENAME("Textures\\Logo\\LogoEAX.tex"));
|
|
|
|
// !! NOTE !! Re-enable these to allow mod support.
|
|
//LoadStringVar(CTString("Data\\Var\\Sam_Version.var"), sam_strVersion);
|
|
//LoadStringVar(CTString("Data\\Var\\ModName.var"), sam_strModName);
|
|
CPrintF(TRANS("Serious Sam version: %s\n"), sam_strVersion);
|
|
CPrintF(TRANS("Active mod: %s\n"), sam_strModName);
|
|
InitializeMenus();
|
|
|
|
// if there is a mod
|
|
if (_fnmMod!="") {
|
|
// execute the mod startup script
|
|
_pShell->Execute(CTString("include \"Scripts\\Mod_startup.ini\";"));
|
|
}
|
|
|
|
// init gl settings module
|
|
InitGLSettings();
|
|
|
|
// init level-info subsystem
|
|
LoadLevelsList();
|
|
LoadDemosList();
|
|
|
|
// apply application mode
|
|
StartNewMode( (GfxAPIType)sam_iGfxAPI, sam_iDisplayAdapter, sam_iScreenSizeI, sam_iScreenSizeJ,
|
|
(enum DisplayDepth)sam_iDisplayDepth, sam_bFullScreenActive);
|
|
|
|
// set default mode reporting
|
|
if( sam_bFirstStarted) {
|
|
_iDisplayModeChangeFlag = 0;
|
|
sam_bFirstStarted = FALSE;
|
|
}
|
|
|
|
HideSplashScreen();
|
|
|
|
if (cmd_strPassword!="") {
|
|
_pShell->SetString("net_strConnectPassword", cmd_strPassword);
|
|
}
|
|
|
|
#if TECHTESTONLY
|
|
cmd_strWorld = CTString("Levels\\TechTestElsa.wld");
|
|
#endif
|
|
|
|
// if connecting to server from command line
|
|
if (cmd_strServer!="") {
|
|
CTString strPort = "";
|
|
if (cmd_iPort>0) {
|
|
_pShell->SetINDEX("net_iPort", cmd_iPort);
|
|
strPort.PrintF(":%d", cmd_iPort);
|
|
}
|
|
CPrintF(TRANS("Command line connection: '%s%s'\n"), cmd_strServer, strPort);
|
|
// go to join menu
|
|
_pGame->gam_strJoinAddress = cmd_strServer;
|
|
if (cmd_bQuickJoin) {
|
|
extern void JoinNetworkGame(void);
|
|
JoinNetworkGame();
|
|
} else {
|
|
StartMenus("join");
|
|
}
|
|
// if starting world from command line
|
|
} else if (cmd_strWorld!="") {
|
|
CPrintF(TRANS("Command line world: '%s'\n"), cmd_strWorld);
|
|
// try to start the game with that level
|
|
try {
|
|
if (cmd_iGoToMarker>=0) {
|
|
CPrintF(TRANS("Command line marker: %d\n"), cmd_iGoToMarker);
|
|
CTString strCommand;
|
|
strCommand.PrintF("cht_iGoToMarker = %d;", cmd_iGoToMarker);
|
|
_pShell->Execute(strCommand);
|
|
}
|
|
_pGame->gam_strCustomLevel = cmd_strWorld;
|
|
if (cmd_bServer) {
|
|
extern void StartNetworkGame(void);
|
|
StartNetworkGame();
|
|
} else {
|
|
extern void StartSinglePlayerGame(void);
|
|
StartSinglePlayerGame();
|
|
}
|
|
} catch (char *strError) {
|
|
CPrintF(TRANS("Cannot start '%s': '%s'\n"), cmd_strWorld, strError);
|
|
}
|
|
// if no relevant starting at command line
|
|
} else {
|
|
StartNextDemo();
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
void End(void)
|
|
{
|
|
_pGame->DisableLoadingHook();
|
|
// cleanup level-info subsystem
|
|
ClearLevelsList();
|
|
ClearDemosList();
|
|
|
|
// destroy the main window and its canvas
|
|
if (pvpViewPort!=NULL) {
|
|
_pGfx->DestroyWindowCanvas( pvpViewPort);
|
|
pvpViewPort = NULL;
|
|
pdpNormal = NULL;
|
|
}
|
|
CloseMainWindow();
|
|
MainWindow_End();
|
|
DestroyMenus();
|
|
_pGame->End();
|
|
LCDEnd();
|
|
// unlock the directory
|
|
DirectoryLockOff();
|
|
SE_EndEngine();
|
|
}
|
|
|
|
|
|
// print display mode info if needed
|
|
void PrintDisplayModeInfo(void)
|
|
{
|
|
// skip if timed out
|
|
if( _pTimer->GetRealTimeTick() > (_tmDisplayModeChanged+sam_tmDisplayModeReport)) return;
|
|
|
|
// cache some general vars
|
|
SLONG slDPWidth = pdp->GetWidth();
|
|
SLONG slDPHeight = pdp->GetHeight();
|
|
if( pdp->IsDualHead()) slDPWidth/=2;
|
|
|
|
CDisplayMode dm;
|
|
dm.dm_pixSizeI = slDPWidth;
|
|
dm.dm_pixSizeJ = slDPHeight;
|
|
// determine proper text scale for statistics display
|
|
FLOAT fTextScale = (FLOAT)slDPWidth/640.0f;
|
|
|
|
// get resolution
|
|
CTString strRes;
|
|
extern CTString _strPreferencesDescription;
|
|
strRes.PrintF( "%dx%dx%s", slDPWidth, slDPHeight, _pGfx->gl_dmCurrentDisplayMode.DepthString());
|
|
if( dm.IsDualHead()) strRes += TRANS(" DualMonitor");
|
|
if( dm.IsWideScreen()) strRes += TRANS(" WideScreen");
|
|
if( _pGfx->gl_eCurrentAPI==GAT_OGL) strRes += " (OpenGL)";
|
|
#ifdef SE1_D3D
|
|
else if( _pGfx->gl_eCurrentAPI==GAT_D3D) strRes += " (Direct3D)";
|
|
#endif // SE1_D3D
|
|
|
|
CTString strDescr;
|
|
strDescr.PrintF("\n%s (%s)\n", _strPreferencesDescription, RenderingPreferencesDescription(sam_iVideoSetup));
|
|
strRes+=strDescr;
|
|
// tell if application is started for the first time, or failed to set mode
|
|
if( _iDisplayModeChangeFlag==0) {
|
|
strRes += TRANS("Display mode set by default!");
|
|
} else if( _iDisplayModeChangeFlag==2) {
|
|
strRes += TRANS("Last mode set failed!");
|
|
}
|
|
|
|
// print it all
|
|
pdp->SetFont( _pfdDisplayFont);
|
|
pdp->SetTextScaling( fTextScale);
|
|
pdp->SetTextAspect( 1.0f);
|
|
pdp->PutText( strRes, slDPWidth*0.05f, slDPHeight*0.85f, LCDGetColor(C_GREEN|255, "display mode"));
|
|
}
|
|
|
|
// do the main game loop and render screen
|
|
void DoGame(void)
|
|
{
|
|
// set flag if not in game
|
|
if( !_pGame->gm_bGameOn) _gmRunningGameMode = GM_NONE;
|
|
|
|
if( _gmRunningGameMode==GM_DEMO && _pNetwork->IsDemoPlayFinished()
|
|
||_gmRunningGameMode==GM_INTRO && _pNetwork->IsGameFinished()) {
|
|
_pGame->StopGame();
|
|
_gmRunningGameMode = GM_NONE;
|
|
|
|
// load next demo
|
|
StartNextDemo();
|
|
if (!_bInAutoPlayLoop) {
|
|
// start menu
|
|
StartMenus();
|
|
}
|
|
}
|
|
|
|
// do the main game loop
|
|
if( _gmRunningGameMode != GM_NONE) {
|
|
_pGame->GameMainLoop();
|
|
// if game is not started
|
|
} else {
|
|
// just handle broadcast messages
|
|
_pNetwork->GameInactive();
|
|
}
|
|
|
|
if (sam_iStartCredits>0) {
|
|
Credits_On(sam_iStartCredits);
|
|
sam_iStartCredits = 0;
|
|
}
|
|
if (sam_iStartCredits<0) {
|
|
Credits_Off();
|
|
sam_iStartCredits = 0;
|
|
}
|
|
if( _gmRunningGameMode==GM_NONE) {
|
|
Credits_Off();
|
|
sam_iStartCredits = 0;
|
|
}
|
|
|
|
// redraw the view
|
|
if( !IsIconic(_hwndMain) && pdp!=NULL && pdp->Lock())
|
|
{
|
|
if( _gmRunningGameMode!=GM_NONE && !bMenuActive ) {
|
|
// handle pretouching of textures and shadowmaps
|
|
pdp->Unlock();
|
|
_pGame->GameRedrawView( pdp, (_pGame->gm_csConsoleState!=CS_OFF || bMenuActive)?0:GRV_SHOWEXTRAS);
|
|
pdp->Lock();
|
|
_pGame->ComputerRender(pdp);
|
|
pdp->Unlock();
|
|
CDrawPort dpScroller(pdp, TRUE);
|
|
dpScroller.Lock();
|
|
if (Credits_Render(&dpScroller)==0) {
|
|
Credits_Off();
|
|
}
|
|
dpScroller.Unlock();
|
|
pdp->Lock();
|
|
} else {
|
|
pdp->Fill( LCDGetColor(C_dGREEN|CT_OPAQUE, "bcg fill"));
|
|
}
|
|
|
|
// do menu
|
|
if( bMenuRendering) {
|
|
// clear z-buffer
|
|
pdp->FillZBuffer( ZBUF_BACK);
|
|
// remember if we should render menus next tick
|
|
bMenuRendering = DoMenu(pdp);
|
|
}
|
|
|
|
// print display mode info if needed
|
|
PrintDisplayModeInfo();
|
|
|
|
// render console
|
|
_pGame->ConsoleRender(pdp);
|
|
|
|
// done with all
|
|
pdp->Unlock();
|
|
|
|
// clear upper and lower parts of screen if in wide screen mode
|
|
if( pdp==pdpWideScreen && pdpNormal->Lock()) {
|
|
const PIX pixWidth = pdpWideScreen->GetWidth();
|
|
const PIX pixHeight = (pdpNormal->GetHeight() - pdpWideScreen->GetHeight()) /2;
|
|
const PIX pixJOfs = pixHeight + pdpWideScreen->GetHeight()-1;
|
|
pdpNormal->Fill( 0, 0, pixWidth, pixHeight, C_BLACK|CT_OPAQUE);
|
|
pdpNormal->Fill( 0, pixJOfs, pixWidth, pixHeight, C_BLACK|CT_OPAQUE);
|
|
pdpNormal->Unlock();
|
|
}
|
|
// show
|
|
pvpViewPort->SwapBuffers();
|
|
}
|
|
}
|
|
|
|
|
|
void TeleportPlayer(int iPosition)
|
|
{
|
|
CTString strCommand;
|
|
strCommand.PrintF( "cht_iGoToMarker = %d;", iPosition);
|
|
_pShell->Execute(strCommand);
|
|
}
|
|
|
|
|
|
CTextureObject _toStarField;
|
|
static FLOAT _fLastVolume = 1.0f;
|
|
void RenderStarfield(CDrawPort *pdp, FLOAT fStrength)
|
|
{
|
|
CTextureData *ptd = (CTextureData *)_toStarField.GetData();
|
|
// skip if no texture
|
|
if(ptd==NULL) return;
|
|
|
|
PIX pixSizeI = pdp->GetWidth();
|
|
PIX pixSizeJ = pdp->GetHeight();
|
|
FLOAT fStretch = pixSizeI/640.0f;
|
|
fStretch*=FLOAT(ptd->GetPixWidth())/ptd->GetWidth();
|
|
|
|
PIXaabbox2D boxScreen(PIX2D(0,0), PIX2D(pixSizeI, pixSizeJ));
|
|
MEXaabbox2D boxTexture(MEX2D(0, 0), MEX2D(pixSizeI/fStretch, pixSizeJ/fStretch));
|
|
pdp->PutTexture(&_toStarField, boxScreen, boxTexture, LerpColor(C_BLACK, C_WHITE, fStrength)|CT_OPAQUE);
|
|
}
|
|
|
|
|
|
FLOAT RenderQuitScreen(CDrawPort *pdp, CViewPort *pvp)
|
|
{
|
|
CDrawPort dpQuit(pdp, TRUE);
|
|
CDrawPort dpWide;
|
|
dpQuit.MakeWideScreen(&dpWide);
|
|
// redraw the view
|
|
if (!dpWide.Lock()) {
|
|
return 0;
|
|
}
|
|
|
|
dpWide.Fill(C_BLACK|CT_OPAQUE);
|
|
RenderStarfield(&dpWide, _fLastVolume);
|
|
|
|
FLOAT fVolume = Credits_Render(&dpWide);
|
|
_fLastVolume = fVolume;
|
|
|
|
dpWide.Unlock();
|
|
pvp->SwapBuffers();
|
|
|
|
return fVolume;
|
|
}
|
|
void QuitScreenLoop(void)
|
|
{
|
|
|
|
Credits_On(3);
|
|
CSoundObject soMusic;
|
|
try {
|
|
_toStarField.SetData_t(CTFILENAME("Textures\\Background\\Night01\\Stars01.tex"));
|
|
soMusic.Play_t(CTFILENAME("Music\\Credits.mp3"), SOF_NONGAME|SOF_MUSIC|SOF_LOOP);
|
|
} catch (char *strError) {
|
|
CPrintF("%s\n", strError);
|
|
}
|
|
// while it is still running
|
|
FOREVER {
|
|
FLOAT fVolume = RenderQuitScreen(pdp, pvpViewPort);
|
|
if (fVolume<=0) {
|
|
return;
|
|
}
|
|
// assure we can listen to non-3d sounds
|
|
soMusic.SetVolume(fVolume, fVolume);
|
|
_pSound->UpdateSounds();
|
|
// while there are any messages in the message queue
|
|
MSG msg;
|
|
while(PeekMessage(&msg, NULL, 0, 0, PM_REMOVE)) {
|
|
// if it is not a keyboard or mouse message
|
|
if(msg.message==WM_LBUTTONDOWN||
|
|
msg.message==WM_RBUTTONDOWN||
|
|
msg.message==WM_KEYDOWN) {
|
|
return;
|
|
}
|
|
}
|
|
//Sleep(5);
|
|
}
|
|
}
|
|
|
|
|
|
int SubMain( HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow)
|
|
{
|
|
(void)hPrevInstance;
|
|
|
|
if( !Init( hInstance, nCmdShow, lpCmdLine )) return FALSE;
|
|
|
|
// initialy, application is running and active, console and menu are off
|
|
_bRunning = TRUE;
|
|
_bQuitScreen = TRUE;
|
|
_pGame->gm_csConsoleState = CS_OFF;
|
|
_pGame->gm_csComputerState = CS_OFF;
|
|
// bMenuActive = FALSE;
|
|
// bMenuRendering = FALSE;
|
|
// while it is still running
|
|
while( _bRunning && _fnmModToLoad=="")
|
|
{
|
|
// while there are any messages in the message queue
|
|
MSG msg;
|
|
while( PeekMessage( &msg, NULL, 0, 0, PM_REMOVE)) {
|
|
// if it is not a mouse message
|
|
if( !(msg.message>=WM_MOUSEFIRST && msg.message<=WM_MOUSELAST) ) {
|
|
// if not system key messages
|
|
if( !(msg.message==WM_KEYDOWN && msg.wParam==VK_F10
|
|
||msg.message==WM_SYSKEYDOWN)) {
|
|
// dispatch it
|
|
TranslateMessage(&msg);
|
|
DispatchMessage(&msg);
|
|
}
|
|
}
|
|
|
|
// system commands (also send by the application itself)
|
|
if( msg.message==WM_SYSCOMMAND)
|
|
{
|
|
switch( msg.wParam & ~0x0F) {
|
|
// if should minimize
|
|
case SC_MINIMIZE:
|
|
if( _bWindowChanging) break;
|
|
_bWindowChanging = TRUE;
|
|
_bReconsiderInput = TRUE;
|
|
// if allowed, not already paused and only in single player game mode
|
|
if( sam_bPauseOnMinimize && !_pNetwork->IsPaused() && _gmRunningGameMode==GM_SINGLE_PLAYER) {
|
|
// pause game
|
|
_pNetwork->TogglePause();
|
|
}
|
|
// if in full screen
|
|
if( sam_bFullScreenActive) {
|
|
// reset display mode and minimize window
|
|
_pGfx->ResetDisplayMode();
|
|
ShowWindow(_hwndMain, SW_MINIMIZE);
|
|
// if not in full screen
|
|
} else {
|
|
// just minimize the window
|
|
ShowWindow(_hwndMain, SW_MINIMIZE);
|
|
}
|
|
break;
|
|
// if should restore
|
|
case SC_RESTORE:
|
|
if( _bWindowChanging) break;
|
|
_bWindowChanging = TRUE;
|
|
_bReconsiderInput = TRUE;
|
|
// if in full screen
|
|
if( sam_bFullScreenActive) {
|
|
ShowWindow(_hwndMain, SW_SHOWNORMAL);
|
|
// set the display mode once again
|
|
StartNewMode( (GfxAPIType)sam_iGfxAPI, sam_iDisplayAdapter, sam_iScreenSizeI, sam_iScreenSizeJ,
|
|
(enum DisplayDepth)sam_iDisplayDepth, sam_bFullScreenActive);
|
|
// if not in full screen
|
|
} else {
|
|
// restore window
|
|
ShowWindow(_hwndMain, SW_SHOWNORMAL);
|
|
}
|
|
break;
|
|
// if should maximize
|
|
case SC_MAXIMIZE:
|
|
if( _bWindowChanging) break;
|
|
_bWindowChanging = TRUE;
|
|
_bReconsiderInput = TRUE;
|
|
// go to full screen
|
|
StartNewMode( (GfxAPIType)sam_iGfxAPI, sam_iDisplayAdapter, sam_iScreenSizeI, sam_iScreenSizeJ,
|
|
(enum DisplayDepth)sam_iDisplayDepth, TRUE);
|
|
ShowWindow( _hwndMain, SW_SHOWNORMAL);
|
|
break;
|
|
}
|
|
}
|
|
|
|
// toggle full-screen on alt-enter
|
|
if( msg.message==WM_SYSKEYDOWN && msg.wParam==VK_RETURN && !IsIconic(_hwndMain)) {
|
|
StartNewMode( (GfxAPIType)sam_iGfxAPI, sam_iDisplayAdapter, sam_iScreenSizeI, sam_iScreenSizeJ,
|
|
(enum DisplayDepth)sam_iDisplayDepth, !sam_bFullScreenActive);
|
|
}
|
|
|
|
// if application should stop
|
|
if( msg.message==WM_QUIT || msg.message==WM_CLOSE) {
|
|
// stop running
|
|
_bRunning = FALSE;
|
|
_bQuitScreen = FALSE;
|
|
}
|
|
|
|
// if application is deactivated or minimized
|
|
if( (msg.message==WM_ACTIVATE && (LOWORD(msg.wParam)==WA_INACTIVE || HIWORD(msg.wParam)))
|
|
|| msg.message==WM_CANCELMODE
|
|
|| msg.message==WM_KILLFOCUS
|
|
|| (msg.message==WM_ACTIVATEAPP && !msg.wParam)) {
|
|
// if application is running and in full screen mode
|
|
if( !_bWindowChanging && _bRunning) {
|
|
// minimize if in full screen
|
|
if( sam_bFullScreenActive) PostMessage(NULL, WM_SYSCOMMAND, SC_MINIMIZE, 0);
|
|
// just disable input if not in full screen
|
|
else _pInput->DisableInput();
|
|
}
|
|
}
|
|
// if application is activated or minimized
|
|
else if( (msg.message==WM_ACTIVATE && (LOWORD(msg.wParam)==WA_ACTIVE || LOWORD(msg.wParam)==WA_CLICKACTIVE))
|
|
|| msg.message==WM_SETFOCUS
|
|
|| (msg.message==WM_ACTIVATEAPP && msg.wParam)) {
|
|
// enable input back again if needed
|
|
_bReconsiderInput = TRUE;
|
|
}
|
|
|
|
if (msg.message==WM_KEYDOWN && msg.wParam==VK_ESCAPE &&
|
|
(_gmRunningGameMode==GM_DEMO || _gmRunningGameMode==GM_INTRO)) {
|
|
_pGame->StopGame();
|
|
_gmRunningGameMode=GM_NONE;
|
|
}
|
|
|
|
if (_pGame->gm_csConsoleState==CS_TALK && msg.message==WM_KEYDOWN && msg.wParam==VK_ESCAPE) {
|
|
_pGame->gm_csConsoleState = CS_OFF;
|
|
msg.message=WM_NULL;
|
|
}
|
|
|
|
BOOL bMenuForced = (_gmRunningGameMode==GM_NONE &&
|
|
(_pGame->gm_csConsoleState==CS_OFF || _pGame->gm_csConsoleState==CS_TURNINGOFF));
|
|
BOOL bMenuToggle = (msg.message==WM_KEYDOWN && msg.wParam==VK_ESCAPE
|
|
&& (_pGame->gm_csComputerState==CS_OFF || _pGame->gm_csComputerState==CS_ONINBACKGROUND));
|
|
if( !bMenuActive) {
|
|
if( bMenuForced || bMenuToggle) {
|
|
// if console is active
|
|
if( _pGame->gm_csConsoleState==CS_ON || _pGame->gm_csConsoleState==CS_TURNINGON) {
|
|
// deactivate it
|
|
_pGame->gm_csConsoleState = CS_TURNINGOFF;
|
|
_iAddonExecState = 0;
|
|
}
|
|
// delete key down message so menu would not exit because of it
|
|
msg.message=WM_NULL;
|
|
// start menu
|
|
StartMenus();
|
|
}
|
|
} else {
|
|
if (bMenuForced && bMenuToggle && pgmCurrentMenu->gm_pgmParentMenu == NULL) {
|
|
// delete key down message so menu would not exit because of it
|
|
msg.message=WM_NULL;
|
|
}
|
|
}
|
|
|
|
// if neither menu nor console is running
|
|
if (!bMenuActive && (_pGame->gm_csConsoleState==CS_OFF || _pGame->gm_csConsoleState==CS_TURNINGOFF)) {
|
|
// if current menu is not root
|
|
if (!IsMenusInRoot()) {
|
|
// start current menu
|
|
StartMenus();
|
|
}
|
|
}
|
|
|
|
if (sam_bMenuSave) {
|
|
sam_bMenuSave = FALSE;
|
|
StartMenus("save");
|
|
}
|
|
if (sam_bMenuLoad) {
|
|
sam_bMenuLoad = FALSE;
|
|
StartMenus("load");
|
|
}
|
|
if (sam_bMenuControls) {
|
|
sam_bMenuControls = FALSE;
|
|
StartMenus("controls");
|
|
}
|
|
if (sam_bMenuHiScore) {
|
|
sam_bMenuHiScore = FALSE;
|
|
StartMenus("hiscore");
|
|
}
|
|
|
|
// interpret console key presses
|
|
if (_iAddonExecState==0) {
|
|
if (msg.message==WM_KEYDOWN) {
|
|
_pGame->ConsoleKeyDown(msg);
|
|
if (_pGame->gm_csConsoleState!=CS_ON) {
|
|
_pGame->ComputerKeyDown(msg);
|
|
}
|
|
} else if (msg.message==WM_KEYUP) {
|
|
// special handler for talk (not to invoke return key bind)
|
|
if( msg.wParam==VK_RETURN && _pGame->gm_csConsoleState==CS_TALK) _pGame->gm_csConsoleState = CS_OFF;
|
|
} else if (msg.message==WM_CHAR) {
|
|
_pGame->ConsoleChar(msg);
|
|
}
|
|
if (msg.message==WM_LBUTTONDOWN
|
|
||msg.message==WM_RBUTTONDOWN
|
|
||msg.message==WM_LBUTTONDBLCLK
|
|
||msg.message==WM_RBUTTONDBLCLK
|
|
||msg.message==WM_LBUTTONUP
|
|
||msg.message==WM_RBUTTONUP) {
|
|
if (_pGame->gm_csConsoleState!=CS_ON) {
|
|
_pGame->ComputerKeyDown(msg);
|
|
}
|
|
}
|
|
}
|
|
// if menu is active and no input on
|
|
if( bMenuActive && !_pInput->IsInputEnabled()) {
|
|
// pass keyboard/mouse messages to menu
|
|
if(msg.message==WM_KEYDOWN) {
|
|
MenuOnKeyDown( msg.wParam);
|
|
} else if (msg.message==WM_LBUTTONDOWN || msg.message==WM_LBUTTONDBLCLK) {
|
|
MenuOnKeyDown(VK_LBUTTON);
|
|
} else if (msg.message==WM_RBUTTONDOWN || msg.message==WM_RBUTTONDBLCLK) {
|
|
MenuOnKeyDown(VK_RBUTTON);
|
|
} else if (msg.message==WM_MOUSEMOVE) {
|
|
MenuOnMouseMove(LOWORD(msg.lParam), HIWORD(msg.lParam));
|
|
#ifndef WM_MOUSEWHEEL
|
|
#define WM_MOUSEWHEEL 0x020A
|
|
#endif
|
|
} else if (msg.message==WM_MOUSEWHEEL) {
|
|
SWORD swDir = SWORD(UWORD(HIWORD(msg.wParam)));
|
|
if (swDir>0) {
|
|
MenuOnKeyDown(11);
|
|
} else if (swDir<0) {
|
|
MenuOnKeyDown(10);
|
|
}
|
|
} else if (msg.message==WM_CHAR) {
|
|
MenuOnChar(msg);
|
|
}
|
|
}
|
|
|
|
// if toggling console
|
|
BOOL bConsoleKey = sam_bToggleConsole || msg.message==WM_KEYDOWN &&
|
|
(MapVirtualKey(msg.wParam, 0)==41 // scan code for '~'
|
|
|| msg.wParam==VK_F1 || (msg.wParam==VK_ESCAPE && _iAddonExecState==3));
|
|
if(bConsoleKey && !_bDefiningKey)
|
|
{
|
|
sam_bToggleConsole = FALSE;
|
|
if( _iAddonExecState==3) _iAddonExecState = 0;
|
|
// if it is up, or pulling up
|
|
if( _pGame->gm_csConsoleState==CS_OFF || _pGame->gm_csConsoleState==CS_TURNINGOFF) {
|
|
// start it moving down and disable menu
|
|
_pGame->gm_csConsoleState = CS_TURNINGON;
|
|
// stop all IFeel effects
|
|
IFeel_StopEffect(NULL);
|
|
if( bMenuActive) {
|
|
StopMenus(FALSE);
|
|
}
|
|
// if it is down, or dropping down
|
|
} else if( _pGame->gm_csConsoleState==CS_ON || _pGame->gm_csConsoleState==CS_TURNINGON) {
|
|
// start it moving up
|
|
_pGame->gm_csConsoleState = CS_TURNINGOFF;
|
|
}
|
|
}
|
|
|
|
if (_pShell->GetINDEX("con_bTalk") && _pGame->gm_csConsoleState==CS_OFF) {
|
|
_pShell->SetINDEX("con_bTalk", FALSE);
|
|
_pGame->gm_csConsoleState = CS_TALK;
|
|
}
|
|
|
|
// if pause pressed
|
|
if (msg.message==WM_KEYDOWN && msg.wParam==VK_PAUSE) {
|
|
// toggle pause
|
|
_pNetwork->TogglePause();
|
|
}
|
|
|
|
// if command sent from external application
|
|
if (msg.message==WM_COMMAND) {
|
|
// if teleport player
|
|
if (msg.wParam==1001) {
|
|
// teleport player
|
|
TeleportPlayer(msg.lParam);
|
|
// restore
|
|
PostMessage(NULL, WM_SYSCOMMAND, SC_RESTORE, 0);
|
|
}
|
|
}
|
|
|
|
// if demo is playing
|
|
if (_gmRunningGameMode==GM_DEMO ||
|
|
_gmRunningGameMode==GM_INTRO ) {
|
|
// check if escape is pressed
|
|
BOOL bEscape = (msg.message==WM_KEYDOWN && msg.wParam==VK_ESCAPE);
|
|
// check if console-invoke key is pressed
|
|
BOOL bTilde = (msg.message==WM_KEYDOWN &&
|
|
(msg.wParam==VK_F1 || MapVirtualKey(msg.wParam, 0)==41));// scan code for '~'
|
|
|
|
// check if any key is pressed
|
|
BOOL bAnyKey = (
|
|
(msg.message==WM_KEYDOWN && (msg.wParam==VK_SPACE || msg.wParam==VK_RETURN))||
|
|
msg.message==WM_LBUTTONDOWN||msg.message==WM_RBUTTONDOWN);
|
|
|
|
// if escape is pressed
|
|
if (bEscape) {
|
|
// stop demo
|
|
_pGame->StopGame();
|
|
_bInAutoPlayLoop = FALSE;
|
|
_gmRunningGameMode = GM_NONE;
|
|
// if any other key is pressed except console invoking
|
|
} else if (bAnyKey && !bTilde) {
|
|
// if not in menu or in console
|
|
if (!bMenuActive && !bMenuRendering && _pGame->gm_csConsoleState==CS_OFF) {
|
|
// skip to next demo
|
|
_pGame->StopGame();
|
|
_gmRunningGameMode = GM_NONE;
|
|
StartNextDemo();
|
|
}
|
|
}
|
|
}
|
|
|
|
} // loop while there are messages
|
|
|
|
// when all messages are removed, window has surely changed
|
|
_bWindowChanging = FALSE;
|
|
|
|
// get real cursor position
|
|
if( _pGame->gm_csComputerState!=CS_OFF && _pGame->gm_csComputerState!=CS_ONINBACKGROUND) {
|
|
POINT pt;
|
|
::GetCursorPos(&pt);
|
|
::ScreenToClient(_hwndMain, &pt);
|
|
_pGame->ComputerMouseMove(pt.x, pt.y);
|
|
}
|
|
|
|
// if addon is to be executed
|
|
if (_iAddonExecState==1) {
|
|
// print header and start console
|
|
CPrintF(TRANS("---- Executing addon: '%s'\n"), (const char*)_fnmAddonToExec);
|
|
sam_bToggleConsole = TRUE;
|
|
_iAddonExecState = 2;
|
|
// if addon is ready for execution
|
|
} else if (_iAddonExecState==2 && _pGame->gm_csConsoleState == CS_ON) {
|
|
// execute it
|
|
CTString strCmd;
|
|
strCmd.PrintF("include \"%s\"", (const char*)_fnmAddonToExec);
|
|
_pShell->Execute(strCmd);
|
|
CPrintF(TRANS("Addon done, press Escape to close console\n"));
|
|
_iAddonExecState = 3;
|
|
}
|
|
|
|
// automaticaly manage input enable/disable toggling
|
|
UpdateInputEnabledState();
|
|
// automaticaly manage pause toggling
|
|
UpdatePauseState();
|
|
// notify game whether menu is active
|
|
_pGame->gm_bMenuOn = bMenuActive;
|
|
|
|
// do the main game loop and render screen
|
|
DoGame();
|
|
|
|
// limit current frame rate if neeeded
|
|
LimitFrameRate();
|
|
|
|
} // end of main application loop
|
|
|
|
_pInput->DisableInput();
|
|
_pGame->StopGame();
|
|
|
|
// invoke quit screen if needed
|
|
if( _bQuitScreen && _fnmModToLoad=="") QuitScreenLoop();
|
|
|
|
End();
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
void CheckModReload(void)
|
|
{
|
|
if (_fnmModToLoad!="") {
|
|
CTString strCommand = _fnmApplicationExe.FileDir()+"SeriousSam.exe";
|
|
//+mod "+_fnmModToLoad.FileName()+"\"";
|
|
CTString strMod = _fnmModToLoad.FileName();
|
|
const char *argv[7];
|
|
argv[0] = strCommand;
|
|
argv[1] = "+game";
|
|
argv[2] = strMod;
|
|
argv[3] = NULL;
|
|
if (_strModServerJoin!="") {
|
|
argv[3] = "+connect";
|
|
argv[4] = _strModServerJoin;
|
|
argv[5] = "+quickjoin";
|
|
argv[6] = NULL;
|
|
}
|
|
_execv(strCommand, argv);
|
|
}
|
|
}
|
|
|
|
void CheckTeaser(void)
|
|
{
|
|
CTFileName fnmTeaser = _fnmApplicationExe.FileDir()+CTString("AfterSam.exe");
|
|
if (fopen(fnmTeaser, "r")!=NULL) {
|
|
Sleep(500);
|
|
_execl(fnmTeaser, "\""+fnmTeaser+"\"", NULL);
|
|
}
|
|
}
|
|
|
|
void CheckBrowser(void)
|
|
{
|
|
if (_strURLToVisit!="") {
|
|
RunBrowser(_strURLToVisit);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
int PASCAL WinMain( HINSTANCE hInstance, HINSTANCE hPrevInstance,
|
|
LPSTR lpCmdLine, int nCmdShow)
|
|
{
|
|
int iResult;
|
|
CTSTREAM_BEGIN {
|
|
iResult = SubMain(hInstance, hPrevInstance, lpCmdLine, nCmdShow);
|
|
} CTSTREAM_END;
|
|
|
|
CheckModReload();
|
|
|
|
CheckTeaser();
|
|
|
|
CheckBrowser();
|
|
|
|
return iResult;
|
|
}
|
|
|
|
|
|
// try to start a new display mode
|
|
BOOL TryToSetDisplayMode( enum GfxAPIType eGfxAPI, INDEX iAdapter, PIX pixSizeI, PIX pixSizeJ,
|
|
enum DisplayDepth eColorDepth, BOOL bFullScreenMode)
|
|
{
|
|
CDisplayMode dmTmp;
|
|
dmTmp.dm_ddDepth = eColorDepth;
|
|
CPrintF( TRANS(" Starting display mode: %dx%dx%s (%s)\n"),
|
|
pixSizeI, pixSizeJ, dmTmp.DepthString(),
|
|
bFullScreenMode ? TRANS("fullscreen") : TRANS("window"));
|
|
|
|
// mark to start ignoring window size/position messages until settled down
|
|
_bWindowChanging = TRUE;
|
|
|
|
// destroy canvas if existing
|
|
_pGame->DisableLoadingHook();
|
|
if( pvpViewPort!=NULL) {
|
|
_pGfx->DestroyWindowCanvas( pvpViewPort);
|
|
pvpViewPort = NULL;
|
|
pdpNormal = NULL;
|
|
}
|
|
|
|
// close the application window
|
|
CloseMainWindow();
|
|
|
|
// try to set new display mode
|
|
BOOL bSuccess;
|
|
if( bFullScreenMode) {
|
|
#ifdef SE1_D3D
|
|
if( eGfxAPI==GAT_D3D) OpenMainWindowFullScreen( pixSizeI, pixSizeJ);
|
|
#endif // SE1_D3D
|
|
bSuccess = _pGfx->SetDisplayMode( eGfxAPI, iAdapter, pixSizeI, pixSizeJ, eColorDepth);
|
|
if( bSuccess && eGfxAPI==GAT_OGL) OpenMainWindowFullScreen( pixSizeI, pixSizeJ);
|
|
} else {
|
|
#ifdef SE1_D3D
|
|
if( eGfxAPI==GAT_D3D) OpenMainWindowNormal( pixSizeI, pixSizeJ);
|
|
#endif // SE1_D3D
|
|
bSuccess = _pGfx->ResetDisplayMode( eGfxAPI);
|
|
if( bSuccess && eGfxAPI==GAT_OGL) OpenMainWindowNormal( pixSizeI, pixSizeJ);
|
|
#ifdef SE1_D3D
|
|
if( bSuccess && eGfxAPI==GAT_D3D) ResetMainWindowNormal();
|
|
#endif // SE1_D3D
|
|
}
|
|
|
|
// if new mode was set
|
|
if( bSuccess) {
|
|
// create canvas
|
|
ASSERT( pvpViewPort==NULL);
|
|
ASSERT( pdpNormal==NULL);
|
|
_pGfx->CreateWindowCanvas( _hwndMain, &pvpViewPort, &pdpNormal);
|
|
|
|
// erase context of both buffers (for the sake of wide-screen)
|
|
pdp = pdpNormal;
|
|
if( pdp!=NULL && pdp->Lock()) {
|
|
pdp->Fill(C_BLACK|CT_OPAQUE);
|
|
pdp->Unlock();
|
|
pvpViewPort->SwapBuffers();
|
|
pdp->Lock();
|
|
pdp->Fill(C_BLACK|CT_OPAQUE);
|
|
pdp->Unlock();
|
|
pvpViewPort->SwapBuffers();
|
|
}
|
|
|
|
// lets try some wide screen screaming :)
|
|
const PIX pixYBegAdj = pdp->GetHeight() * 21/24;
|
|
const PIX pixYEndAdj = pdp->GetHeight() * 3/24;
|
|
const PIX pixXEnd = pdp->GetWidth();
|
|
pdpWideScreen = new CDrawPort( pdp, PIXaabbox2D( PIX2D(0,pixYBegAdj), PIX2D(pixXEnd, pixYEndAdj)));
|
|
pdpWideScreen->dp_fWideAdjustment = 9.0f / 12.0f;
|
|
if( sam_bWideScreen) pdp = pdpWideScreen;
|
|
|
|
// initial screen fill and swap, just to get context running
|
|
BOOL bSuccess = FALSE;
|
|
if( pdp!=NULL && pdp->Lock()) {
|
|
pdp->Fill( LCDGetColor( C_dGREEN|CT_OPAQUE, "bcg fill"));
|
|
pdp->Unlock();
|
|
pvpViewPort->SwapBuffers();
|
|
bSuccess = TRUE;
|
|
}
|
|
_pGame->EnableLoadingHook(pdp);
|
|
|
|
// if the mode is not working, or is not accelerated
|
|
if( !bSuccess || !_pGfx->IsCurrentModeAccelerated())
|
|
{ // report error
|
|
CPrintF( TRANS("This mode does not support hardware acceleration.\n"));
|
|
// destroy canvas if existing
|
|
if( pvpViewPort!=NULL) {
|
|
_pGame->DisableLoadingHook();
|
|
_pGfx->DestroyWindowCanvas( pvpViewPort);
|
|
pvpViewPort = NULL;
|
|
pdpNormal = NULL;
|
|
}
|
|
// close the application window
|
|
CloseMainWindow();
|
|
// report failure
|
|
return FALSE;
|
|
}
|
|
|
|
// remember new settings
|
|
sam_bFullScreenActive = bFullScreenMode;
|
|
sam_iScreenSizeI = pixSizeI;
|
|
sam_iScreenSizeJ = pixSizeJ;
|
|
sam_iDisplayDepth = eColorDepth;
|
|
sam_iDisplayAdapter = iAdapter;
|
|
sam_iGfxAPI = eGfxAPI;
|
|
|
|
// report success
|
|
return TRUE;
|
|
}
|
|
|
|
// if couldn't set new mode
|
|
else {
|
|
// close the application window
|
|
CloseMainWindow();
|
|
// report failure
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
|
|
// list of possible display modes for recovery
|
|
const INDEX aDefaultModes[][3] =
|
|
{ // color, API, adapter
|
|
{ DD_DEFAULT, GAT_OGL, 0},
|
|
{ DD_16BIT, GAT_OGL, 0},
|
|
{ DD_16BIT, GAT_OGL, 1}, // 3dfx Voodoo2
|
|
#ifdef SE1_D3D
|
|
{ DD_DEFAULT, GAT_D3D, 0},
|
|
{ DD_16BIT, GAT_D3D, 0},
|
|
{ DD_16BIT, GAT_D3D, 1},
|
|
#endif // SE1_D3D
|
|
};
|
|
const INDEX ctDefaultModes = ARRAYCOUNT(aDefaultModes);
|
|
|
|
// start new display mode
|
|
void StartNewMode( enum GfxAPIType eGfxAPI, INDEX iAdapter, PIX pixSizeI, PIX pixSizeJ,
|
|
enum DisplayDepth eColorDepth, BOOL bFullScreenMode)
|
|
{
|
|
CPrintF( TRANS("\n* START NEW DISPLAY MODE ...\n"));
|
|
|
|
// try to set the mode
|
|
BOOL bSuccess = TryToSetDisplayMode( eGfxAPI, iAdapter, pixSizeI, pixSizeJ, eColorDepth, bFullScreenMode);
|
|
|
|
// if failed
|
|
if( !bSuccess)
|
|
{
|
|
// report failure and reset to default resolution
|
|
_iDisplayModeChangeFlag = 2; // failure
|
|
CPrintF( TRANS("Requested display mode could not be set!\n"));
|
|
pixSizeI = 640;
|
|
pixSizeJ = 480;
|
|
bFullScreenMode = TRUE;
|
|
// try to revert to one of recovery modes
|
|
for( INDEX iMode=0; iMode<ctDefaultModes; iMode++) {
|
|
eColorDepth = (DisplayDepth)aDefaultModes[iMode][0];
|
|
eGfxAPI = (GfxAPIType) aDefaultModes[iMode][1];
|
|
iAdapter = aDefaultModes[iMode][2];
|
|
CPrintF(TRANS("\nTrying recovery mode %d...\n"), iMode);
|
|
bSuccess = TryToSetDisplayMode( eGfxAPI, iAdapter, pixSizeI, pixSizeJ, eColorDepth, bFullScreenMode);
|
|
if( bSuccess) break;
|
|
}
|
|
// if all failed
|
|
if( !bSuccess) {
|
|
FatalError(TRANS(
|
|
"Cannot set display mode!\n"
|
|
"Serious Sam was unable to find display mode with hardware acceleration.\n"
|
|
"Make sure you install proper drivers for your video card as recommended\n"
|
|
"in documentation and set your desktop to 16 bit (65536 colors).\n"
|
|
"Please see ReadMe file for troubleshooting information.\n"));
|
|
}
|
|
|
|
// if succeeded
|
|
} else {
|
|
_iDisplayModeChangeFlag = 1; // all ok
|
|
}
|
|
|
|
// apply 3D-acc settings
|
|
ApplyGLSettings(FALSE);
|
|
|
|
// remember time of mode setting
|
|
_tmDisplayModeChanged = _pTimer->GetRealTimeTick();
|
|
}
|