/* Copyright (c) 2002-2012 Croteam Ltd. This program is free software; you can redistribute it and/or modify it under the terms of version 2 of the GNU General Public License as published by the Free Software Foundation 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, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. */ #include "Engine/StdH.h" #include #include #include #include #include #include #include #include INDEX con_bNoWarnings = 0; // global handle for application window in full-screen mode extern HWND _hwndMain; extern BOOL _bFullScreen; /* * Report error and terminate program. */ static BOOL _bInFatalError = FALSE; void FatalError(const char *strFormat, ...) { // disable recursion if (_bInFatalError) return; _bInFatalError = TRUE; // reset default windows display mode first // (this is a low overhead and shouldn't allocate memory) CDS_ResetMode(); #ifdef PLATFORM_WIN32 // hide fullscreen window if any if( _bFullScreen) { // must do minimize first - don't know why :( ShowWindow( _hwndMain, SW_MINIMIZE); ShowWindow( _hwndMain, SW_HIDE); } #else SDL_DestroyWindow((SDL_Window *) _hwndMain); // just get rid of it, it's all burning down anyhow. _hwndMain = NULL; #endif // format the message in buffer va_list arg; va_start(arg, strFormat); CTString strBuffer; strBuffer.VPrintF(strFormat, arg); va_end(arg); if (_pConsole!=NULL) { // print the buffer to the console CPutString(TRANS("FatalError:\n")); CPutString(strBuffer); // make sure the console log was written safely _pConsole->CloseLog(); } #ifdef PLATFORM_WIN32 // create message box with just OK button MessageBoxA(NULL, strBuffer, TRANS("Fatal Error"), MB_OK|MB_ICONHAND|MB_SETFOREGROUND|MB_TASKMODAL); extern void EnableWindowsKeys(void); EnableWindowsKeys(); #else SDL_ShowSimpleMessageBox(SDL_MESSAGEBOX_ERROR, TRANSV("Fatal Error"), strBuffer, NULL); SDL_Quit(); #endif _bInFatalError = FALSE; // exit program exit(EXIT_FAILURE); } /* * Report warning without terminating program (stops program until user responds). */ void WarningMessage(const char *strFormat, ...) { // format the message in buffer va_list arg; va_start(arg, strFormat); CTString strBuffer; strBuffer.VPrintF(strFormat, arg); va_end(arg); // print it to console CPrintF("%s\n", (const char *) strBuffer); // if warnings are enabled if( !con_bNoWarnings) { // create message box #ifdef PLATFORM_WIN32 MessageBoxA(NULL, (const char *) strBuffer, TRANS("Warning"), MB_OK|MB_ICONEXCLAMATION|MB_SETFOREGROUND|MB_TASKMODAL); #else SDL_ShowSimpleMessageBox(SDL_MESSAGEBOX_WARNING, TRANSV("Warning"), strBuffer, (SDL_Window *) _hwndMain); #endif } } void InfoMessage(const char *strFormat, ...) { // format the message in buffer va_list arg; va_start(arg, strFormat); CTString strBuffer; strBuffer.VPrintF(strFormat, arg); va_end(arg); // print it to console CPrintF("%s\n", (const char *) strBuffer); // create message box #ifdef PLATFORM_WIN32 MessageBoxA(NULL, (const char *) strBuffer, TRANS("Information"), MB_OK|MB_ICONINFORMATION|MB_SETFOREGROUND|MB_TASKMODAL); #else SDL_ShowSimpleMessageBox(SDL_MESSAGEBOX_INFORMATION, TRANSV("Information"), strBuffer, (SDL_Window *) _hwndMain); #endif } /* Ask user for yes/no answer(stops program until user responds). */ BOOL YesNoMessage(const char *strFormat, ...) { // format the message in buffer va_list arg; va_start(arg, strFormat); CTString strBuffer; strBuffer.VPrintF(strFormat, arg); va_end(arg); // print it to console CPrintF("%s\n", (const char *) strBuffer); // create message box #ifdef PLATFORM_WIN32 return MessageBoxA(NULL, strBuffer, TRANS("Question"), MB_YESNO|MB_ICONQUESTION|MB_SETFOREGROUND|MB_TASKMODAL)==IDYES; #else const SDL_MessageBoxButtonData ynbuttons[] = { { SDL_MESSAGEBOX_BUTTON_ESCAPEKEY_DEFAULT, 0, TRANSV("No") }, { SDL_MESSAGEBOX_BUTTON_RETURNKEY_DEFAULT, 1, TRANSV("Yes") }, }; const SDL_MessageBoxData ynbox = { SDL_MESSAGEBOX_INFORMATION, // !!! FIXME: no SDL_MESSAGEBOX_QUESTION (SDL_Window *) _hwndMain, TRANSV("Question"), strBuffer, SDL_arraysize(ynbuttons), ynbuttons, NULL }; int button = 0; return ((SDL_ShowMessageBox(&ynbox, &button) != -1) && (button == 1)); #endif } /* * Throw an exception of formatted string. */ void ThrowF_t(const char *strFormat, ...) // throws char * { const SLONG slBufferSize = 256; //char strBuffer[slBufferSize+1]; // Can't throw from the stack like this! static char *strBuffer = NULL; // !!! FIXME: This could be dangerous if you call this in a catch handler... delete[] strBuffer; strBuffer = new char[slBufferSize+1]; // format the message in buffer va_list arg; va_start(arg, strFormat); // variable arguments start after this argument _vsnprintf(strBuffer, slBufferSize, strFormat, arg); va_end(arg); throw strBuffer; } /* * Get the name string for error code. */ const char *ErrorName(const struct ErrorTable *pet, SLONG ulErrCode) { for (INDEX i=0; iet_Count; i++) { if (pet->et_Errors[i].ec_Code == ulErrCode) { return pet->et_Errors[i].ec_Name; } } return TRANS("CROTEAM_UNKNOWN"); } /* * Get the description string for error code. */ const char *ErrorDescription(const struct ErrorTable *pet, SLONG ulErrCode) { for (INDEX i=0; iet_Count; i++) { if (pet->et_Errors[i].ec_Code == ulErrCode) { return pet->et_Errors[i].ec_Description; } } return TRANS("Unknown error"); } /* * Get the description string for windows error code. */ extern const CTString GetWindowsError(DWORD dwWindowsErrorCode) { #ifdef PLATFORM_WIN32 // buffer to recieve error description LPVOID lpMsgBuf; // call function that will prepare text abount given windows error code FormatMessage( FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM, NULL, dwWindowsErrorCode, MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), (LPTSTR) &lpMsgBuf, 0, NULL ); // create result CTString from prepared message CTString strResultMessage = (char *)lpMsgBuf; // Free the buffer. LocalFree( lpMsgBuf ); return strResultMessage; #else return CTString("This isn't Windows, so calling this function is probably a portability bug."); #endif } // must be in separate function to disable stupid optimizer extern void Breakpoint(void) { #ifdef PLATFORM_WIN32 __asm int 0x03; #else SDL_TriggerBreakpoint(); #endif }