summaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
Diffstat (limited to 'src')
-rw-r--r--src/gui/updatewindow.cpp207
-rw-r--r--src/gui/updatewindow.h19
-rw-r--r--src/main.cpp166
3 files changed, 202 insertions, 190 deletions
diff --git a/src/gui/updatewindow.cpp b/src/gui/updatewindow.cpp
index b037103b..7ad9b175 100644
--- a/src/gui/updatewindow.cpp
+++ b/src/gui/updatewindow.cpp
@@ -22,50 +22,215 @@
*/
#include "updatewindow.h"
+#include "gui.h"
+#include "../main.h"
+#include "../log.h"
+#include <curl/curl.h>
#include <sstream>
+#include <iostream>
+#include <cstdio>
+#include "SDL_thread.h"
+
+UpdaterWindow *updaterWindow;
+float progress = 0.0f;
+SDL_Thread *thread;
+std::string updateHost = "themanaworld.org/files";
+bool downloadComplete = false;
UpdaterWindow::UpdaterWindow()
- : Window("Updating")
+ : Window("Updating...")
{
- setContentSize(320, 96);
-
- vbox = new VBox();
- label = new gcn::Label("Default text");
- progressBar = new ProgressBar(0.0, 0, 0, 0, 0, 0, 191, 63);
+ int h = 100;
+ int w = 320;
+ setContentSize(w, h);
- vbox->setPosition(4, 0);
- vbox->setSize(getWidth() - 12, getHeight() - 24);
+ label = new gcn::Label("Connecting...");
+ label->setPosition(5,5);
+ progressBar = new ProgressBar(0.0, 5, 25, w - 10, 40, 37, 70, 23);
+ cancelButton = new Button("Cancel");
+ cancelButton->setPosition(5, h - 5 - cancelButton->getHeight());
+ cancelButton->setEventId("cancel");
+ cancelButton->addActionListener(this);
+ playButton = new Button("Play");
+ playButton->setPosition(cancelButton->getX() + cancelButton->getWidth() + 5,
+ h - 5 - playButton->getHeight());
+ playButton->setEventId("play");
+ playButton->setEnabled(false);
+ playButton->addActionListener(this);
- vbox->add(label);
- vbox->add(progressBar);
-
- add(vbox);
+ add(label);
+ add(progressBar);
+ add(cancelButton);
+ add(playButton);
+
+ cancelButton->requestFocus();
+ setLocationRelativeTo(getParent());
}
UpdaterWindow::~UpdaterWindow()
{
delete label;
delete progressBar;
- delete vbox;
+ delete cancelButton;
+ delete playButton;
}
-void UpdaterWindow::setProgress(double p)
+void UpdaterWindow::setProgress(float p)
{
- progress = p;
+ progressBar->setProgress(p);
}
void UpdaterWindow::setLabel(const std::string &str)
{
- labelText = str;
+ label->setCaption(str);
+ label->adjustSize();
+}
+
+void UpdaterWindow::enable()
+{
+ playButton->setEnabled(true);
}
void UpdaterWindow::draw(gcn::Graphics *graphics)
{
- std::stringstream ss;
- ss << labelText << " (" << progress * 100 << "%)";
+ Window::draw(graphics);
+}
- label->setCaption(ss.str());
- progressBar->setProgress(progress);
+void UpdaterWindow::action(const std::string& eventId)
+{
+ if (eventId == "cancel") {
+ state = EXIT;
+ }
+ else if (eventId == "play") {
+ state = LOGIN;
+ }
+}
- Window::draw(graphics);
+int updateProgress(void *ptr,
+ double t, /* dltotal */
+ double d, /* dlnow */
+ double ultotal,
+ double ulnow)
+{
+ std::stringstream labelString;
+ progress = d/t;
+ labelString << (char *)ptr << " (" << (int)(progress*100) << "%)";
+ updaterWindow->setLabel(labelString.str());
+ updaterWindow->setProgress(progress);
+ if(state!=UPDATE) {
+ // If the action was canceled return an error code to stop the thread
+ return -1;
+ }
+ return 0;
+}
+
+int downloadThread(void *ptr) {
+ CURL *curl;
+ CURLcode res;
+ FILE *outfile;
+ std::string fileName((char *)ptr);
+ std::string url(updateHost);
+ url += "/" + fileName;
+ logger->log("Downloading: %s", url.c_str());
+
+ curl = curl_easy_init();
+ if(curl)
+ {
+ downloadComplete = false;
+ progress = 0.0f;
+ // TODO: download in the proper folder (data?)
+ outfile = fopen(fileName.c_str(), "wb");
+ curl_easy_setopt(curl, CURLOPT_URL, url.c_str());
+ curl_easy_setopt(curl, CURLOPT_WRITEDATA, outfile);
+ curl_easy_setopt(curl, CURLOPT_NOPROGRESS, FALSE);
+ curl_easy_setopt(curl, CURLOPT_PROGRESSFUNCTION, updateProgress);
+ curl_easy_setopt(curl, CURLOPT_PROGRESSDATA, ptr);
+
+ res = curl_easy_perform(curl);
+
+ fclose(outfile);
+ curl_easy_cleanup(curl);
+ downloadComplete = true;
+ }
+}
+
+int download(std::string url) {
+ thread = SDL_CreateThread(downloadThread, (void *)url.c_str());
+ if ( thread == NULL ) {
+ logger->log("Unable to create thread");
+ return 0;
+ }
+}
+
+void updateData() {
+ updaterWindow = new UpdaterWindow();
+ state = UPDATE;
+
+ std::string updateHost =
+ config.getValue("updatehost", "themanaworld.org/files");
+ // Try to download the updates list
+ download("resources.txt");
+ std::ifstream in;
+
+ while (state == UPDATE)
+ {
+ // Handle SDL events
+ SDL_Event event;
+ while (SDL_PollEvent(&event)) {
+ switch (event.type) {
+ case SDL_QUIT:
+ state = EXIT;
+ break;
+
+ case SDL_KEYDOWN:
+ if (event.key.keysym.sym == SDLK_ESCAPE)
+ {
+ state = EXIT;
+ }
+ break;
+ }
+
+ guiInput->pushInput(event);
+ }
+
+ // If not alredy downloading another file
+ if (downloadComplete) {
+ // Try to open resources.txt
+ if (!in.is_open())
+ {
+ in.open("resources.txt");
+ if (!in.is_open())
+ {
+ logger->error("Unable to open resources.txt");
+ }
+ // TODO: check for error 404
+ }
+ else {
+ if (!in.eof())
+ {
+ // Download each update
+ std::string line;
+ getline(in, line);
+ download(line);
+ }
+ else {
+ // All updates downloaded
+ updaterWindow->enable();
+ updaterWindow->setLabel("Completed");
+ }
+ }
+ }
+
+ gui->logic();
+
+ login_wallpaper->draw(screen, 0, 0);
+ gui->draw();
+ guiGraphics->updateScreen();
+ }
+
+ in.close();
+
+ SDL_WaitThread(thread, NULL);
+
+ delete updaterWindow;
}
diff --git a/src/gui/updatewindow.h b/src/gui/updatewindow.h
index b602c053..89e13610 100644
--- a/src/gui/updatewindow.h
+++ b/src/gui/updatewindow.h
@@ -27,20 +27,21 @@
#include "window.h"
#include "vbox.h"
#include "progressbar.h"
+#include "button.h"
/**
* Update progress window GUI
*
* \ingroup GUI
*/
-class UpdaterWindow : public Window
+class UpdaterWindow : public Window, public gcn::ActionListener
{
protected:
std::string labelText; /**< Text for caption label */
- double progress; /**< Progress */
- VBox *vbox;
gcn::Label *label; /**< Progress bar caption */
+ Button *cancelButton; /**< Button to stop the update process */
+ Button *playButton; /**< Button to start playing */
ProgressBar *progressBar; /**< Update progress bar */
public:
@@ -57,14 +58,24 @@ class UpdaterWindow : public Window
/**
* Set's progress bar status
*/
- void setProgress(double);
+ void setProgress(float p);
/**
* Set's label above progress
*/
void setLabel(const std::string &);
+
+ /**
+ * Enables play button
+ */
+ void enable();
+
+ void action(const std::string& eventId);
void draw(gcn::Graphics *);
+
+ int updateState;
};
+void updateData();
#endif
diff --git a/src/main.cpp b/src/main.cpp
index 5fb543fc..3c0d13a9 100644
--- a/src/main.cpp
+++ b/src/main.cpp
@@ -87,8 +87,6 @@ Configuration config; /**< Xml file configuration reader */
Logger *logger; /**< Log object */
ItemManager *itemDb; /**< Item database object */
-UpdaterWindow *updaterWindow; /**< Update window */
-
/**
* Allows the next frame to be drawn (part of framerate limiting)
*/
@@ -341,91 +339,6 @@ void exit_engine()
delete logger;
}
-/** Update progress callback */
-int
-progressCallback(void *clientp, double dltotal, double dlnow,
- double utotal, double ulnow)
-{
- // update progress bar..
- updaterWindow->setProgress(dlnow / dltotal);
-
- // draw
- gui->logic();
- gui->draw();
- graphics->updateScreen();
- return CURLE_OK;
-}
-
-/** Get filename from URL */
-const char *urlFilename(const char *url)
-{
- for (int i = strlen(url); i > 0; i--)
- {
- if (url[i] == '/') {
- return &url[i+1];
- }
- }
- return NULL;
-}
-
-/** Download file from location */
-int download(const char *location)
-{
- if (location == NULL) {
- return false;
- }
- // find local file location
- const char *name = urlFilename(location);
- if (!name) {
- return true;
- }
-
- std::string dest = homeDir;
- dest += "/";
- dest += name;
-
- FILE *fp = NULL;
- fp = fopen(dest.c_str(), "w");
- if (!fp) {
- return false;
- }
-
- std::cout << "Downloading '" << location << "'" << std::endl;
-
- // init curl
- curl_global_init(CURL_GLOBAL_WIN32);
- // download file
- CURL *curl = curl_easy_init();
- int ret = 0;
- if (curl)
- {
- curl_easy_setopt(curl, CURLOPT_URL, "curl.haxx.se");
- std::cout << "Debug: setopt" << std::endl;
- /*curl_easy_setopt(curl, CURLOPT_NOPROGRESS, false);
- curl_easy_setopt(curl, CURLOPT_PROGRESSFUNCTION, progressCallback);
- curl_easy_setopt(curl, CURLOPT_PROGRESSDATA, NULL);
- curl_easy_setopt(curl, CURLOPT_WRITEDATA, fp);*/
-
- //ret = curl_easy_perform(curl);
- std::cout << "Debug: perform" << std::endl;
- curl_easy_cleanup(curl);
- std::cout << "Debug: cleanup" << std::endl;
- }
- else {
- ret = 1;
- }
-
- // cleanup curl
- curl_global_cleanup();
- fclose(fp);
-
- if (ret != 0)
- std::cout << "Failed!" << std::endl;
- std::cout << " (" << ret << ")" << std::endl;
-
- return (ret == CURLE_OK) ? true : false;
-}
-
/** Check to see if a file exists */
int exists(const std::string &file)
{
@@ -441,82 +354,6 @@ int exists(const std::string &file)
}
}
-/** Update the game data */
-void update()
-{
- updaterWindow = new UpdaterWindow();
-
- std::string host =
- config.getValue("updatehost", "http://themanaworld.org/");
-
- // Add / to host if it's not there yet.
- if (host.at(host.length() - 1) != '/')
- {
- host += "/";
- }
-
- std::string fullLocation = host;
- fullLocation += "resources.txt";
-
- std::string fullName = homeDir;
- //fullName += "/";
- fullName += "resources.txt";
-
- updaterWindow->setLabel(fullLocation);
-
- // Get resources file
- if (!download(fullLocation.c_str()))
- {
- std::cout << "Error downloading resources.txt" << std::endl;
- delete updaterWindow;
- return;
- }
-
- std::cout << "Opening " << fullName << std::endl;
-
- std::ifstream in(fullName.c_str());
- if (!in.is_open())
- {
- std::cout << "Error opening" << std::endl;
- delete updaterWindow;
- return;
- }
-
- std::string line;
-
- while (!in.eof())
- {
- getline(in, line);
-
- // check for XML tag (if it is XML tag it is error)
- if (line[0] == '<') {
- std::cout << "Error: resources.txt download error (404)" << std::endl;
- break;
- }
-
- fullName = homeDir;
- fullName += "/";
- fullName += line;
-
- fullLocation = host;
- fullLocation += line;
-
- updaterWindow->setLabel(fullLocation);
-
- if (!exists(fullName)) {
- if (!download(fullLocation.c_str())) {
- std::cout << "Failed to download " << line << std::endl;
- }
- }
- else {
- PHYSFS_addToSearchPath(fullName.c_str(), 1);
- }
- }
- in.close();
- guiTop->remove(updaterWindow);
- delete updaterWindow;
-}
-
/** Main */
int main(int argc, char *argv[])
{
@@ -588,8 +425,7 @@ int main(int argc, char *argv[])
graphics->updateScreen();
break;
case UPDATE:
- update();
- state = LOGIN;
+ updateData();
break;
default:
state = EXIT;