openspades/Sources/Gui/MainScreenHelper.cpp

366 lines
9.3 KiB
C++
Raw Normal View History

2013-11-23 22:13:04 +09:00
/*
Copyright (c) 2013 yvt
2013-11-24 00:24:46 +09:00
Portion of the code is based on Serverbrowser.cpp.
2013-11-23 22:13:04 +09:00
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-11-24 00:24:46 +09:00
#include <OpenSpades.h>
2013-11-23 22:13:04 +09:00
#include "MainScreenHelper.h"
2013-11-24 00:24:46 +09:00
#include "Serverbrowser.h"
#include <Core/AutoLocker.h>
#include <Core/Thread.h>
#include <curl/curl.h>
#include <json/json.h>
2013-11-24 02:27:08 +09:00
#include "MainScreen.h"
2013-12-02 22:04:56 +09:00
#include <Core/Settings.h>
#include <algorithm>
2013-12-17 01:28:56 +01:00
#include <cctype>
2013-12-02 21:50:12 +09:00
#include "MainWindow.h" // for credits
2013-12-02 22:04:56 +09:00
SPADES_SETTING(cl_serverListUrl, "http://services.buildandshoot.com/serverlist.json");
2013-11-23 22:13:04 +09:00
namespace spades {
namespace gui {
2013-11-24 00:24:46 +09:00
// 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:
2013-11-23 22:13:04 +09:00
2013-11-24 00:24:46 +09:00
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 );
2013-12-02 22:04:56 +09:00
curl_easy_setopt( cHandle, CURLOPT_URL, cl_serverListUrl.CString() );
2013-11-24 00:24:46 +09:00
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();
2013-11-23 22:13:04 +09:00
}
MainScreenHelper::~MainScreenHelper() {
2013-11-24 00:24:46 +09:00
SPADES_MARK_FUNCTION();
if(query) {
query->MarkForAutoDeletion();
}
delete result;
delete newResult;
2013-11-23 22:13:04 +09:00
}
void MainScreenHelper::MainScreenDestroyed() {
2013-11-24 00:24:46 +09:00
SPADES_MARK_FUNCTION();
2013-11-23 22:13:04 +09:00
mainScreen = NULL;
}
2013-11-24 00:24:46 +09:00
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();
}
2013-12-02 21:50:12 +09:00
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, "&copy;", "(C)");
return TextifyHTML(html);
}
2013-11-24 02:27:08 +09:00
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) {
2013-11-24 00:24:46 +09:00
if(result == NULL){
return NULL;
}
2013-11-24 02:27:08 +09:00
std::vector<MainScreenServerItem *> lst = result->list;
2013-11-24 00:24:46 +09:00
if(lst.empty())
return NULL;
2013-11-24 02:27:08 +09:00
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 ) );
}
2013-11-24 00:24:46 +09:00
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;
}
2013-11-24 02:27:08 +09:00
std::string MainScreenHelper::ConnectServer() {
if(mainScreen == NULL){
return "mainScreen == NULL";
}
return mainScreen->Connect();
}
2013-11-24 00:24:46 +09:00
std::string MainScreenHelper::GetServerListQueryMessage() {
if(result == NULL)
return "";
return result->message;
}
2013-11-24 02:27:08 +09:00
std::string MainScreenHelper::GetPendingErrorMessage() {
std::string s = errorMessage;
errorMessage.clear();
return s;
}
2013-11-24 00:24:46 +09:00
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();
}
2013-11-23 22:13:04 +09:00
}
}