2013-08-29 11:45:22 +09:00
|
|
|
/*
|
|
|
|
Copyright (c) 2013 yvt
|
|
|
|
|
|
|
|
This file is part of OpenSpades.
|
|
|
|
|
|
|
|
OpenSpades 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.
|
|
|
|
|
|
|
|
OpenSpades 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 OpenSpades. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
*/
|
2013-08-18 16:18:06 +09:00
|
|
|
|
2013-08-29 00:21:27 +02:00
|
|
|
#include <OpenSpades.h>
|
2013-11-22 22:50:14 +09:00
|
|
|
#include <Imports/SDL.h>
|
|
|
|
#include "Main.h"
|
2013-08-18 16:18:06 +09:00
|
|
|
#include "MainWindow.h"
|
2013-11-22 22:50:14 +09:00
|
|
|
#include "MainScreen.h"
|
2013-09-13 23:07:49 +02:00
|
|
|
#include <Core/FileManager.h>
|
|
|
|
#include <Core/DirectoryFileSystem.h>
|
|
|
|
#include <Core/Debug.h>
|
|
|
|
#include <Core/Settings.h>
|
|
|
|
#include <Core/ConcurrentDispatch.h>
|
|
|
|
#include <Core/ZipFileSystem.h>
|
2013-10-09 23:18:13 +02:00
|
|
|
#include <Core/ServerAddress.h>
|
2013-09-12 02:10:44 +09:00
|
|
|
#include "ErrorDialog.h"
|
2013-11-22 22:50:14 +09:00
|
|
|
#include "Runner.h"
|
|
|
|
#include <Client/Client.h>
|
2013-08-18 16:18:06 +09:00
|
|
|
|
2013-09-13 23:07:49 +02:00
|
|
|
#include <Core/VoxelModel.h>
|
|
|
|
#include <Draw/GLOptimizedVoxelModel.h>
|
2013-08-18 16:18:06 +09:00
|
|
|
|
2013-09-14 02:05:05 +02:00
|
|
|
#include <ScriptBindings/ScriptManager.h>
|
2013-09-12 02:10:44 +09:00
|
|
|
|
2013-09-13 23:07:36 +02:00
|
|
|
#include <algorithm> //std::sort
|
|
|
|
|
2013-08-18 16:18:06 +09:00
|
|
|
#ifdef WIN32
|
|
|
|
#include <windows.h>
|
|
|
|
#include <shlobj.h>
|
2013-10-09 23:18:13 +02:00
|
|
|
#define strncasecmp(x,y,z) _strnicmp(x,y,z)
|
|
|
|
#define strcasecmp(x,y) _stricmp(x,y)
|
2013-11-01 22:08:03 +01:00
|
|
|
|
2013-11-22 22:50:14 +09:00
|
|
|
SPADES_SETTING(cg_smp, "0");
|
|
|
|
|
2013-11-01 22:08:03 +01:00
|
|
|
//lm: without doing it this way, we will get a low-res icon or an ugly resampled icon in our window.
|
|
|
|
//we cannot use the fltk function on the console window, because it's not an Fl_Window...
|
|
|
|
void setIcon( HWND hWnd )
|
|
|
|
{
|
|
|
|
HINSTANCE hInstance = GetModuleHandle(NULL);
|
|
|
|
HICON hIcon = (HICON)LoadImageA( hInstance, "AppIcon", IMAGE_ICON, GetSystemMetrics(SM_CXICON), GetSystemMetrics(SM_CYICON), 0 );
|
|
|
|
if( hIcon ) {
|
|
|
|
SendMessage( hWnd, WM_SETICON, ICON_BIG, (LPARAM)hIcon );
|
|
|
|
}
|
|
|
|
hIcon = (HICON)LoadImageA( hInstance, "AppIcon", IMAGE_ICON, GetSystemMetrics(SM_CXSMICON), GetSystemMetrics(SM_CYSMICON), 0 );
|
|
|
|
if( hIcon ) {
|
|
|
|
SendMessage( hWnd, WM_SETICON, ICON_SMALL, (LPARAM)hIcon );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-18 16:18:06 +09:00
|
|
|
#endif
|
|
|
|
|
2013-11-01 22:08:03 +01:00
|
|
|
//fltk
|
|
|
|
void setWindowIcon( Fl_Window* window )
|
|
|
|
{
|
|
|
|
#ifdef _WIN32
|
|
|
|
window->icon( (char *)LoadIconA( GetModuleHandle(NULL), "AppIcon" ) );
|
|
|
|
#else
|
|
|
|
//check for mac / linux icon with fltk?
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2013-08-18 16:18:06 +09:00
|
|
|
#ifdef __APPLE__
|
|
|
|
#include <xmmintrin.h>
|
|
|
|
#endif
|
|
|
|
|
2013-08-26 01:27:44 +09:00
|
|
|
#include <FL/fl_ask.H>
|
|
|
|
|
2013-10-09 23:18:13 +02:00
|
|
|
SPADES_SETTING(cg_lastQuickConnectHost, "");
|
|
|
|
SPADES_SETTING(cg_protocolVersion, "");
|
|
|
|
int cg_autoConnect = 0;
|
|
|
|
|
|
|
|
int argsHandler(int argc, char **argv, int &i)
|
|
|
|
{
|
|
|
|
if( char* a = argv[i] ) {
|
|
|
|
if( !strncasecmp( a, "aos://", 6 ) ) {
|
|
|
|
cg_lastQuickConnectHost = a;
|
|
|
|
cg_autoConnect = 1;
|
|
|
|
return ++i;
|
|
|
|
}
|
|
|
|
//lm: we attempt to detect protocol version, allowing with or without a prefix 'v='
|
|
|
|
//we set proto, but without url we will not auto-connect
|
|
|
|
if( a[0] == 'v' && a[1] == '=' ) { a += 2; }
|
|
|
|
if( !strcasecmp( a, "75" ) || !strcasecmp( a, "075" ) || !strcasecmp( a, "0.75" ) ) {
|
|
|
|
cg_protocolVersion = 3;
|
|
|
|
return ++i;
|
|
|
|
}
|
|
|
|
if( !strcasecmp( a, "76" ) || !strcasecmp( a, "076" ) || !strcasecmp( a, "0.76" ) ) {
|
|
|
|
cg_protocolVersion = 4;
|
|
|
|
return ++i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-11-22 22:50:14 +09:00
|
|
|
namespace spades {
|
|
|
|
void StartClient(const spades::ServerAddress& addr, const std::string& playerName){
|
|
|
|
class ConcreteRunner: public spades::gui::Runner {
|
|
|
|
spades::ServerAddress addr;
|
|
|
|
std::string playerName;
|
|
|
|
protected:
|
|
|
|
virtual spades::gui::View *CreateView(spades::client::IRenderer *renderer, spades::client::IAudioDevice *audio) {
|
|
|
|
return new spades::client::Client(renderer, audio, addr, playerName);
|
|
|
|
}
|
|
|
|
public:
|
|
|
|
ConcreteRunner(const spades::ServerAddress& addr,
|
|
|
|
const std::string& playerName):
|
|
|
|
addr(addr), playerName(playerName){ }
|
|
|
|
};
|
|
|
|
ConcreteRunner runner(addr, playerName);
|
|
|
|
runner.RunProtected();
|
|
|
|
}
|
|
|
|
void StartMainScreen(){
|
|
|
|
class ConcreteRunner: public spades::gui::Runner {
|
|
|
|
protected:
|
|
|
|
virtual spades::gui::View *CreateView(spades::client::IRenderer *renderer, spades::client::IAudioDevice *audio) {
|
|
|
|
return new spades::gui::MainScreen(renderer, audio);
|
|
|
|
}
|
|
|
|
public:
|
|
|
|
};
|
|
|
|
ConcreteRunner runner;
|
|
|
|
runner.RunProtected();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-18 16:18:06 +09:00
|
|
|
int main(int argc, char ** argv)
|
|
|
|
{
|
|
|
|
|
|
|
|
// Enable FPE
|
|
|
|
#if 0
|
|
|
|
#ifdef __APPLE__
|
|
|
|
short fpflags = 0x1332; // Default FP flags, change this however you want.
|
|
|
|
__asm__("fnclex\n\tfldcw %0\n\t": "=m"(fpflags));
|
|
|
|
|
|
|
|
_MM_SET_EXCEPTION_MASK(_MM_GET_EXCEPTION_MASK() & ~_MM_MASK_INVALID);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#endif
|
2013-09-12 02:10:44 +09:00
|
|
|
try{
|
|
|
|
|
|
|
|
Fl::scheme("gtk+");
|
2013-08-26 01:27:44 +09:00
|
|
|
|
2013-09-12 02:10:44 +09:00
|
|
|
spades::reflection::Backtrace::StartBacktrace();
|
|
|
|
|
|
|
|
SPADES_MARK_FUNCTION();
|
|
|
|
spades::DispatchQueue::GetThreadQueue()->MarkSDLVideoThread();
|
|
|
|
|
|
|
|
SPLog("Package: " PACKAGE_STRING);
|
|
|
|
|
|
|
|
// default resource directories
|
|
|
|
#ifdef WIN32
|
|
|
|
static char buf[4096];
|
|
|
|
GetModuleFileName(NULL, buf, 4096);
|
|
|
|
std::string appdir = buf;
|
|
|
|
appdir = appdir.substr(0, appdir.find_last_of('\\')+1);
|
2013-08-26 01:27:44 +09:00
|
|
|
|
2013-11-01 22:08:03 +01:00
|
|
|
spades::FileManager::AddFileSystem(new spades::DirectoryFileSystem(appdir + "Resources", false));
|
2013-09-12 02:10:44 +09:00
|
|
|
|
2013-11-01 22:08:03 +01:00
|
|
|
if(SUCCEEDED(SHGetFolderPath(NULL, CSIDL_APPDATA, NULL, 0, buf))){
|
2013-09-12 02:10:44 +09:00
|
|
|
std::string datadir = buf;
|
|
|
|
datadir += "\\OpenSpades\\Resources";
|
2013-11-01 22:08:03 +01:00
|
|
|
spades::FileManager::AddFileSystem(new spades::DirectoryFileSystem(datadir, true));
|
|
|
|
}
|
|
|
|
//fltk has a console window on windows (can disable while building, maybe use a builtin console for a later release?)
|
|
|
|
HWND hCon = GetConsoleWindow();
|
|
|
|
if( NULL != hCon ) {
|
|
|
|
setIcon( hCon );
|
2013-09-12 02:10:44 +09:00
|
|
|
}
|
2013-11-01 22:08:03 +01:00
|
|
|
|
2013-08-18 16:18:06 +09:00
|
|
|
#elif defined(__APPLE__)
|
2013-09-12 02:10:44 +09:00
|
|
|
std::string home = getenv("HOME");
|
|
|
|
spades::FileManager::AddFileSystem
|
|
|
|
(new spades::DirectoryFileSystem("./Resources", false));
|
|
|
|
|
|
|
|
spades::FileManager::AddFileSystem
|
|
|
|
(new spades::DirectoryFileSystem(home+"/Library/Application Support/OpenSpades/Resources", true));
|
2013-08-18 16:18:06 +09:00
|
|
|
#else
|
2013-09-12 02:10:44 +09:00
|
|
|
std::string home = getenv("HOME");
|
|
|
|
spades::FileManager::AddFileSystem
|
|
|
|
(new spades::DirectoryFileSystem("./Resources", false));
|
|
|
|
|
|
|
|
spades::FileManager::AddFileSystem
|
|
|
|
(new spades::DirectoryFileSystem(home+"/.openspades/Resources", true));
|
2013-08-18 16:18:06 +09:00
|
|
|
#endif
|
2013-09-12 02:10:44 +09:00
|
|
|
|
|
|
|
try{
|
|
|
|
spades::StartLog();
|
|
|
|
}catch(const std::exception& ex){
|
|
|
|
fl_alert("Failed to start recording log because of the following error:\n%s\n\n"
|
|
|
|
"OpenSpades will continue to run, but any critical events are not logged.", ex.what());
|
|
|
|
}
|
|
|
|
SPLog("Log Started.");
|
|
|
|
|
2013-10-08 22:06:36 +02:00
|
|
|
#if defined(RESDIR_DEFINED) && !NDEBUG
|
2013-11-01 22:08:03 +01:00
|
|
|
spades::FileManager::AddFileSystem(new spades::DirectoryFileSystem(RESDIR, false));
|
2013-08-18 18:49:41 +09:00
|
|
|
#endif
|
2013-09-12 02:10:44 +09:00
|
|
|
|
|
|
|
/*
|
|
|
|
spades::FileManager::AddFileSystem
|
|
|
|
(new spades::DirectoryFileSystem("/Users/tcpp/Programs/MacPrograms2/OpenSpades/Resources", false));
|
|
|
|
*/
|
|
|
|
// add all zip files
|
|
|
|
{
|
|
|
|
std::vector<spades::IFileSystem*> fss;
|
|
|
|
|
|
|
|
std::vector<std::string> files = spades::FileManager::EnumFiles("");
|
|
|
|
|
|
|
|
struct Comparator {
|
|
|
|
static int GetPakId(const std::string& str) {
|
|
|
|
if(str.size() >= 4 && str[0] == 'p' &&
|
|
|
|
str[1] == 'a' && str[2] == 'k' &&
|
|
|
|
(str[3] >= '0' && str[3] <= '9')){
|
|
|
|
return atoi(str.c_str() + 3);
|
|
|
|
}else{
|
|
|
|
return 32767;
|
|
|
|
}
|
2013-09-11 19:46:08 +09:00
|
|
|
}
|
2013-09-12 02:10:44 +09:00
|
|
|
static bool Compare(const std::string& a,
|
|
|
|
const std::string& b) {
|
|
|
|
int pa = GetPakId(a);
|
|
|
|
int pb = GetPakId(b);
|
|
|
|
if(pa == pb){
|
|
|
|
return a < b;
|
|
|
|
}else{
|
|
|
|
return pa < pb;
|
|
|
|
}
|
2013-09-11 19:46:08 +09:00
|
|
|
}
|
2013-09-12 02:10:44 +09:00
|
|
|
};
|
2013-08-18 16:18:06 +09:00
|
|
|
|
2013-09-12 02:10:44 +09:00
|
|
|
std::sort(files.begin(), files.end(), Comparator::Compare);
|
2013-08-18 16:18:06 +09:00
|
|
|
|
2013-09-12 02:10:44 +09:00
|
|
|
for(size_t i = 0; i < files.size(); i++){
|
|
|
|
std::string name = files[i];
|
|
|
|
|
|
|
|
// check extension
|
|
|
|
if(name.size() < 4 ||
|
|
|
|
name.rfind(".pak") != name.size() - 4){
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(spades::FileManager::FileExists(name.c_str())) {
|
|
|
|
spades::IStream *stream = spades::FileManager::OpenForReading(name.c_str());
|
|
|
|
spades::ZipFileSystem *fs = new spades::ZipFileSystem(stream);
|
|
|
|
SPLog("Pak Registered: %s\n", name.c_str());
|
|
|
|
fss.push_back(fs);
|
|
|
|
}
|
|
|
|
}
|
2013-11-03 15:01:13 +09:00
|
|
|
for(size_t i = fss.size(); i > 0; i--){
|
|
|
|
spades::FileManager::AddFileSystem(fss[i - 1]);
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
|
|
|
}
|
2013-09-12 02:10:44 +09:00
|
|
|
|
|
|
|
SPLog("Initializing script engine");
|
|
|
|
spades::ScriptManager::GetInstance();
|
|
|
|
|
2013-11-22 22:50:14 +09:00
|
|
|
SPLog("Initializing window system");
|
2013-10-09 23:18:13 +02:00
|
|
|
int dum = 0;
|
|
|
|
Fl::args( argc, argv, dum, argsHandler );
|
|
|
|
|
2013-10-10 20:16:29 +02:00
|
|
|
MainWindow* win = NULL;
|
2013-10-09 23:18:13 +02:00
|
|
|
if( !cg_autoConnect ) {
|
2013-11-22 22:50:14 +09:00
|
|
|
if(!Fl::event_shift()) {
|
|
|
|
// TODO: always show main window for first run
|
|
|
|
|
|
|
|
SPLog("Starting main screen");
|
|
|
|
spades::StartMainScreen();
|
|
|
|
}else{
|
|
|
|
SPLog("Initializing main window");
|
|
|
|
win = new MainWindow();
|
|
|
|
win->Init();
|
|
|
|
setWindowIcon( win );
|
|
|
|
win->show(argc, argv);
|
|
|
|
win->CheckGLCapability();
|
|
|
|
|
|
|
|
SPLog("Entering FLTK main loop");
|
|
|
|
Fl::run();
|
|
|
|
SPLog("Leaving FLTK main loop");
|
|
|
|
}
|
2013-10-09 23:18:13 +02:00
|
|
|
} else {
|
|
|
|
spades::ServerAddress host(cg_lastQuickConnectHost.CString(), (int)cg_protocolVersion == 3 ? spades::ProtocolVersion::v075 : spades::ProtocolVersion::v076 );
|
|
|
|
MainWindow::StartGame( host );
|
|
|
|
}
|
2013-09-12 02:10:44 +09:00
|
|
|
|
|
|
|
spades::Settings::GetInstance()->Flush();
|
|
|
|
|
2013-10-10 20:16:29 +02:00
|
|
|
if( win ) {
|
|
|
|
delete win;
|
|
|
|
}
|
|
|
|
|
2013-09-12 02:10:44 +09:00
|
|
|
}catch(const std::exception& ex) {
|
|
|
|
|
|
|
|
ErrorDialog dlg;
|
2013-11-01 22:08:03 +01:00
|
|
|
setWindowIcon( &dlg );
|
2013-09-12 02:10:44 +09:00
|
|
|
dlg.set_modal();
|
|
|
|
dlg.result = 0;
|
|
|
|
|
|
|
|
// TODO: free this buffer (just leaking)
|
|
|
|
Fl_Text_Buffer *buf = new Fl_Text_Buffer;
|
|
|
|
buf->append(ex.what());
|
|
|
|
dlg.infoView->wrap_mode(Fl_Text_Display::WRAP_AT_BOUNDS, 0);
|
|
|
|
dlg.infoView->buffer(buf);
|
|
|
|
dlg.helpView->value("See SystemMessages.log for more details.");
|
|
|
|
dlg.show();
|
|
|
|
while(dlg.visible()){
|
|
|
|
Fl::wait();
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
2013-09-12 02:10:44 +09:00
|
|
|
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|