364 lines
9.2 KiB
C++
364 lines
9.2 KiB
C++
/*
|
|
Copyright (c) 2013 yvt
|
|
Portion of the code is based on Serverbrowser.cpp.
|
|
|
|
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/>.
|
|
|
|
*/
|
|
#include <OpenSpades.h>
|
|
#include "MainScreenHelper.h"
|
|
#include "Serverbrowser.h"
|
|
#include <Core/AutoLocker.h>
|
|
#include <Core/Thread.h>
|
|
#include <curl/curl.h>
|
|
#include <json/json.h>
|
|
#include "MainScreen.h"
|
|
|
|
#include "MainWindow.h" // for credits
|
|
|
|
#define SERVICE_URL "http://services.buildandshoot.com/serverlist.json"
|
|
|
|
namespace spades {
|
|
namespace gui {
|
|
|
|
// FIXME: mostly duplicated code with Serverbrowser.cpp
|
|
class MainScreenHelper::ServerListQuery: public Thread {
|
|
Mutex infoLock;
|
|
Handle<MainScreenHelper> owner;
|
|
std::string buffer;
|
|
|
|
static size_t curlWriteCallback( void *ptr, size_t size, size_t nmemb, ServerListQuery* query ) {
|
|
size_t dataSize = size * nmemb;
|
|
return query->writeCallback( ptr, dataSize );
|
|
}
|
|
|
|
size_t writeCallback( void *ptr, size_t size ) {
|
|
|
|
buffer.append(reinterpret_cast<char*>(ptr), size);
|
|
return size;
|
|
}
|
|
|
|
void ReturnResult(MainScreenServerList *list){
|
|
AutoLocker lock(&(owner->newResultArrayLock));
|
|
delete owner->newResult;
|
|
owner->newResult = list;
|
|
owner = NULL; // release owner
|
|
}
|
|
|
|
void ProcessResponse() {
|
|
Json::Reader reader;
|
|
Json::Value root;
|
|
MainScreenServerList *resp = new MainScreenServerList();
|
|
try{
|
|
if( reader.parse( buffer, root, false ) ) {
|
|
for( Json::Value::iterator it = root.begin(); it != root.end(); ++it ) {
|
|
Json::Value &obj = *it;
|
|
Serveritem* srv = Serveritem::create( obj );
|
|
if( srv ) {
|
|
resp->list.push_back(new MainScreenServerItem(srv));
|
|
delete srv;
|
|
}
|
|
}
|
|
}
|
|
ReturnResult(resp);
|
|
}catch(...){
|
|
delete resp;
|
|
throw;
|
|
}
|
|
}
|
|
|
|
public:
|
|
|
|
ServerListQuery(MainScreenHelper *owner):
|
|
owner(owner){
|
|
|
|
}
|
|
|
|
virtual void Run() {
|
|
try{
|
|
CURL* cHandle = curl_easy_init();
|
|
if( cHandle ) {
|
|
try{
|
|
curl_easy_setopt( cHandle, CURLOPT_USERAGENT, OpenSpades_VER_STR );
|
|
curl_easy_setopt( cHandle, CURLOPT_URL, SERVICE_URL );
|
|
curl_easy_setopt( cHandle, CURLOPT_WRITEFUNCTION, &ServerListQuery::curlWriteCallback );
|
|
curl_easy_setopt( cHandle, CURLOPT_WRITEDATA, this );
|
|
if( 0 == curl_easy_perform( cHandle ) ) {
|
|
|
|
ProcessResponse();
|
|
|
|
} else {
|
|
SPRaise("HTTP request error.");
|
|
}
|
|
curl_easy_cleanup( cHandle );
|
|
}catch(...){
|
|
curl_easy_cleanup( cHandle );
|
|
throw;
|
|
}
|
|
}else{
|
|
SPRaise("Failed to create cURL object.");
|
|
}
|
|
}catch(std::exception& ex) {
|
|
MainScreenServerList *lst = new MainScreenServerList();
|
|
lst->message = ex.what();
|
|
ReturnResult(lst);
|
|
}catch(...) {
|
|
MainScreenServerList *lst = new MainScreenServerList();
|
|
lst->message = "Unknown error.";
|
|
ReturnResult(lst);
|
|
}
|
|
}
|
|
};
|
|
|
|
MainScreenHelper::MainScreenHelper(MainScreen *scr):
|
|
mainScreen(scr),
|
|
result(NULL),
|
|
newResult(NULL),
|
|
query(NULL){
|
|
SPADES_MARK_FUNCTION();
|
|
}
|
|
|
|
MainScreenHelper::~MainScreenHelper() {
|
|
SPADES_MARK_FUNCTION();
|
|
if(query) {
|
|
query->MarkForAutoDeletion();
|
|
}
|
|
delete result;
|
|
delete newResult;
|
|
}
|
|
|
|
void MainScreenHelper::MainScreenDestroyed() {
|
|
SPADES_MARK_FUNCTION();
|
|
mainScreen = NULL;
|
|
}
|
|
|
|
bool MainScreenHelper::PollServerListState() {
|
|
SPADES_MARK_FUNCTION();
|
|
AutoLocker lock(&newResultArrayLock);
|
|
if(newResult) {
|
|
delete result;
|
|
result = newResult;
|
|
newResult = NULL;
|
|
query->MarkForAutoDeletion();
|
|
query = NULL;
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void MainScreenHelper::StartQuery() {
|
|
if(query) {
|
|
// ongoing
|
|
return;
|
|
}
|
|
|
|
query = new ServerListQuery(this);
|
|
query->Start();
|
|
}
|
|
|
|
static std::string TextifyHTML(const std::string& html) {
|
|
std::string out;
|
|
int blankLineLength = 0;
|
|
bool inTag = false;
|
|
bool inTagName = false;
|
|
bool hasLetter = false;
|
|
std::string tagName;
|
|
for(size_t i = 0; i < html.size(); i++){
|
|
if(html[i] == '<') {
|
|
inTag = true;
|
|
inTagName = true;
|
|
tagName.clear();
|
|
}
|
|
if(inTag) {
|
|
if(inTagName) {
|
|
if(html[i] == '/')
|
|
i++;
|
|
if(html[i] < 'a' || html[i] > 'z') {
|
|
inTagName = false;
|
|
if(html[i] == '>') {
|
|
inTag = false;
|
|
}
|
|
if(tagName == "br" ||
|
|
tagName == "h3" ||
|
|
tagName == "h2") {
|
|
if(blankLineLength < 1) {
|
|
out += "\n";
|
|
hasLetter = false;
|
|
blankLineLength++;
|
|
}
|
|
}
|
|
}else{
|
|
tagName += html[i];
|
|
}
|
|
}else if(html[i] == '>') {
|
|
inTag = false;
|
|
}
|
|
continue;
|
|
}
|
|
|
|
bool isLetter = !isspace(html[i]);
|
|
if(hasLetter == false && isLetter == false)
|
|
continue;
|
|
hasLetter = true;
|
|
blankLineLength = 0;
|
|
out += html[i];
|
|
}
|
|
return out;
|
|
}
|
|
|
|
std::string MainScreenHelper::GetCredits() {
|
|
std::string html = std::string(reinterpret_cast<const char *>(aboutText), sizeof(aboutText));
|
|
html = Replace(html, "${PACKAGE_STRING}", PACKAGE_STRING);
|
|
html = Replace(html, "©", "(C)");
|
|
|
|
return TextifyHTML(html);
|
|
}
|
|
|
|
struct serverSorter {
|
|
int type;
|
|
bool order;
|
|
serverSorter( int type_, bool order_ ) : type(type_), order(order_) {;}
|
|
bool operator() ( MainScreenServerItem* x, MainScreenServerItem* y ) const
|
|
{
|
|
if( order ) {
|
|
MainScreenServerItem* t = x;
|
|
x = y;
|
|
y = t;
|
|
}
|
|
switch( type ) {
|
|
default:
|
|
case 0: return asInt( x->GetPing(), y->GetPing() );
|
|
case 1: return asInt( x->GetNumPlayers(), y->GetNumPlayers() );
|
|
case 2: return asString( x->GetName(), y->GetName() );
|
|
case 3: return asString( x->GetMapName(), y->GetMapName() );
|
|
case 4: return asString( x->GetGameMode(), y->GetGameMode() );
|
|
case 5: return asString( x->GetProtocol(), y->GetProtocol() );
|
|
case 6: return asString( x->GetCountry(), y->GetCountry() );
|
|
}
|
|
}
|
|
bool asInt( int x, int y ) const
|
|
{
|
|
if( x == y ) return false;
|
|
return (x<y);
|
|
}
|
|
bool asString( const std::string& x, const std::string& y ) const
|
|
{
|
|
std::string::size_type t = 0;
|
|
for( t = 0; t < x.length() && t < y.length(); ++ t ) {
|
|
int xx = std::tolower(x[t]);
|
|
int yy = std::tolower(y[t]);
|
|
if( xx != yy ) {
|
|
return xx < yy;
|
|
}
|
|
}
|
|
if( x.length() == y.length() ) {
|
|
return false;
|
|
}
|
|
return x.length() < y.length();
|
|
}
|
|
static bool ciCharLess( char c1, char c2 )
|
|
{
|
|
return (std::tolower( static_cast<char>( c1 ) ) < std::tolower( static_cast<char>( c1 ) ));
|
|
}
|
|
};
|
|
|
|
|
|
CScriptArray *MainScreenHelper::GetServerList(std::string sortKey, bool descending) {
|
|
if(result == NULL){
|
|
return NULL;
|
|
}
|
|
|
|
std::vector<MainScreenServerItem *> lst = result->list;
|
|
if(lst.empty())
|
|
return NULL;
|
|
|
|
int sortKeyId = -1;
|
|
if(sortKey == "Ping") {
|
|
sortKeyId = 0;
|
|
}else if(sortKey == "NumPlayers") {
|
|
sortKeyId = 1;
|
|
}else if(sortKey == "Name") {
|
|
sortKeyId = 2;
|
|
}else if(sortKey == "MapName") {
|
|
sortKeyId = 3;
|
|
}else if(sortKey == "GameMode") {
|
|
sortKeyId = 4;
|
|
}else if(sortKey == "Protocol") {
|
|
sortKeyId = 5;
|
|
}else if(sortKey == "Country") {
|
|
sortKeyId = 6;
|
|
}
|
|
if(sortKeyId != -1){
|
|
std::sort( lst.begin(), lst.end(), serverSorter( sortKeyId, descending ) );
|
|
}
|
|
|
|
asIScriptEngine *eng = ScriptManager::GetInstance()->GetEngine();
|
|
asIObjectType* t = eng->GetObjectTypeById(eng->GetTypeIdByDecl("array<spades::MainScreenServerItem@>"));
|
|
SPAssert(t != NULL);
|
|
CScriptArray *arr = new CScriptArray(lst.size(), t);
|
|
for(size_t i = 0; i < lst.size(); i++){
|
|
arr->SetValue((asUINT)i, &(lst[i]));
|
|
}
|
|
return arr;
|
|
}
|
|
|
|
std::string MainScreenHelper::ConnectServer() {
|
|
if(mainScreen == NULL){
|
|
return "mainScreen == NULL";
|
|
}
|
|
return mainScreen->Connect();
|
|
}
|
|
|
|
std::string MainScreenHelper::GetServerListQueryMessage() {
|
|
if(result == NULL)
|
|
return "";
|
|
return result->message;
|
|
}
|
|
|
|
std::string MainScreenHelper::GetPendingErrorMessage() {
|
|
std::string s = errorMessage;
|
|
errorMessage.clear();
|
|
return s;
|
|
}
|
|
|
|
MainScreenServerList::~MainScreenServerList() {
|
|
for(size_t i = 0; i < list.size(); i++)
|
|
list[i]->Release();
|
|
}
|
|
|
|
MainScreenServerItem::MainScreenServerItem(Serveritem *item) {
|
|
SPADES_MARK_FUNCTION();
|
|
name = item->Name();
|
|
address = item->Ip();
|
|
mapName = item->Map();
|
|
gameMode = item->GameMode();
|
|
country = item->Country();
|
|
protocol = item->Version();
|
|
ping = item->Ping();
|
|
numPlayers = item->Players();
|
|
maxPlayers = item->MaxPlayers();
|
|
}
|
|
|
|
MainScreenServerItem::~MainScreenServerItem() {
|
|
SPADES_MARK_FUNCTION();
|
|
|
|
}
|
|
|
|
|
|
}
|
|
}
|
|
|