diff --git a/code/nel/include/nel/gui/group_html.h b/code/nel/include/nel/gui/group_html.h
index c489168e0..c5e708b04 100644
--- a/code/nel/include/nel/gui/group_html.h
+++ b/code/nel/include/nel/gui/group_html.h
@@ -319,6 +319,7 @@ namespace NLGUI
REFLECT_STRING("html", getHTML, setHTML)
REFLECT_STRING("home", home, setHome)
REFLECT_FLOAT("timeout", getTimeout, setTimeout)
+ REFLECT_STRING("title", getTitle, setTitle)
REFLECT_EXPORT_END
protected :
@@ -414,6 +415,8 @@ namespace NLGUI
// Set the title
void setTitle (const ucstring &title);
+ void setTitle (const std::string &title);
+ std::string getTitle() const;
// Lookup a url in local file system
bool lookupLocalFile (std::string &result, const char *url, bool isUrl);
diff --git a/code/nel/src/gui/group_html.cpp b/code/nel/src/gui/group_html.cpp
index 46aeffedc..bb2228df5 100644
--- a/code/nel/src/gui/group_html.cpp
+++ b/code/nel/src/gui/group_html.cpp
@@ -5155,6 +5155,25 @@ namespace NLGUI
}
}
+ void CGroupHTML::setTitle(const std::string &title)
+ {
+ ucstring uctitle;
+ uctitle.fromUtf8(title);
+
+ _TitleString.clear();
+ if(!_TitlePrefix.empty())
+ {
+ _TitleString = _TitlePrefix + " - ";
+ }
+ _TitleString += uctitle;
+
+ setTitle(_TitleString);
+ }
+
+ std::string CGroupHTML::getTitle() const {
+ return _TitleString.toUtf8();
+ };
+
// ***************************************************************************
bool CGroupHTML::lookupLocalFile (string &result, const char *url, bool isUrl)
@@ -5684,6 +5703,22 @@ namespace NLGUI
// create html code with image url inside and do the request again
renderHtmlString("
"+_URL+"");
}
+ else if (_TrustedDomain && type.find("text/lua") == 0)
+ {
+ setTitle(_TitleString);
+
+ _LuaScript = "\nlocal __CURRENT_WINDOW__=\""+this->_Id+"\" \n"+content;
+ CLuaManager::getInstance().executeLuaScript(_LuaScript, true);
+ _LuaScript.clear();
+
+ _Browsing = false;
+ _Connecting = false;
+
+ // disable refresh button
+ clearRefresh();
+ // disable redo into this url
+ _AskedUrl.clear();
+ }
else
{
renderHtmlString(content);
@@ -5928,7 +5963,9 @@ namespace NLGUI
return;
// push to redo, pop undo, and set current
- _BrowseRedo.push_front(_AskedUrl);
+ if (!_AskedUrl.empty())
+ _BrowseRedo.push_front(_AskedUrl);
+
_AskedUrl= _BrowseUndo.back();
_BrowseUndo.pop_back();
diff --git a/code/ryzom/client/client_default.cfg b/code/ryzom/client/client_default.cfg
index ab01dd6dd..108a17e97 100644
--- a/code/ryzom/client/client_default.cfg
+++ b/code/ryzom/client/client_default.cfg
@@ -597,6 +597,8 @@ ChannelIgnoreFilter =
"Matis", "Tryker", "Zoraï"
};
+// interval in minutes for webig notify thread to run
+WebIgNotifInterval = 10;
WebIgMainDomain = "app.ryzom.com";
WebIgTrustedDomains = {
"api.ryzom.com", "app.ryzom.com"
diff --git a/code/ryzom/client/src/client_cfg.cpp b/code/ryzom/client/src/client_cfg.cpp
index c6de70c07..b8bc72f68 100644
--- a/code/ryzom/client/src/client_cfg.cpp
+++ b/code/ryzom/client/src/client_cfg.cpp
@@ -430,6 +430,7 @@ CClientConfig::CClientConfig()
WebIgMainDomain = "atys.ryzom.com";
WebIgTrustedDomains.push_back(WebIgMainDomain);
+ WebIgNotifInterval = 10; // time in minutes
CurlMaxConnections = 2;
CurlCABundle.clear();
@@ -1087,6 +1088,7 @@ void CClientConfig::setValues()
// WEBIG //
READ_STRING_FV(WebIgMainDomain);
READ_STRINGVECTOR_FV(WebIgTrustedDomains);
+ READ_INT_FV(WebIgNotifInterval);
READ_INT_FV(CurlMaxConnections);
if (ClientCfg.CurlMaxConnections < 0)
ClientCfg.CurlMaxConnections = 2;
diff --git a/code/ryzom/client/src/client_cfg.h b/code/ryzom/client/src/client_cfg.h
index de930a63b..c9c75a152 100644
--- a/code/ryzom/client/src/client_cfg.h
+++ b/code/ryzom/client/src/client_cfg.h
@@ -312,6 +312,7 @@ struct CClientConfig
std::string WebIgMainDomain;
std::vector WebIgTrustedDomains;
+ uint WebIgNotifInterval; // value in minutes for notification thread
sint32 CurlMaxConnections;
string CurlCABundle;
diff --git a/code/ryzom/client/src/interface_v3/group_html_webig.cpp b/code/ryzom/client/src/interface_v3/group_html_webig.cpp
index 66cfecdad..ea885f17b 100644
--- a/code/ryzom/client/src/interface_v3/group_html_webig.cpp
+++ b/code/ryzom/client/src/interface_v3/group_html_webig.cpp
@@ -18,6 +18,7 @@
#include "group_html_webig.h"
#include "nel/misc/xml_auto_ptr.h"
+#include "nel/gui/lua_manager.h"
#include "../client_cfg.h"
#include "../user_entity.h"
#include "../entities.h"
@@ -149,13 +150,21 @@ size_t writeDataFromCurl(void *buffer, size_t size, size_t nmemb, void *pcl)
return size*nmemb;
}
-
-struct CWebigNotificationThread : public NLMISC::IRunnable
+class CWebigNotificationThread : public NLMISC::IRunnable
{
+private:
CURL *Curl;
+ bool _Running;
+ IThread *_Thread;
+
+public:
CWebigNotificationThread()
{
+ _Running = false;
+ _Thread = NULL;
+ curl_global_init(CURL_GLOBAL_ALL);
+
Curl = curl_easy_init();
if(!Curl) return;
curl_easy_setopt(Curl, CURLOPT_COOKIEFILE, "");
@@ -173,6 +182,12 @@ struct CWebigNotificationThread : public NLMISC::IRunnable
curl_easy_cleanup(Curl);
Curl = 0;
}
+ if (_Thread)
+ {
+ _Thread->terminate();
+ delete _Thread;
+ _Thread = NULL;
+ }
}
void get(const std::string &url)
@@ -186,61 +201,24 @@ struct CWebigNotificationThread : public NLMISC::IRunnable
curl_easy_getinfo(Curl, CURLINFO_RESPONSE_CODE, &r);
//nlwarning("result : '%s'", curlresult.c_str());
- vector notifs;
- explode(curlresult, string("|"), notifs);
-
- // Update the mail notification icon
-
- uint32 nbmail = 0;
- if(!notifs.empty() && fromString(notifs[0], nbmail))
- {
- //nlinfo("nb mail is a number %d", nbmail);
- CInterfaceManager *pIM = CInterfaceManager::getInstance();
- if(pIM)
- {
- CCDBNodeLeaf *_CheckMailNode = NLGUI::CDBManager::getInstance()->getDbProp("UI:VARIABLES:MAIL_WAITING");
- if(_CheckMailNode)
- {
- _CheckMailNode->setValue32(nbmail==0?0:1);
- CInterfaceElement *elm = CWidgetManager::getInstance()->getElementFromId("ui:interface:compass:mail:mail_nb");
- if (elm)
- {
- CViewText *vt = dynamic_cast(elm);
- vt->setText(toString("%d", nbmail));
- }
- }
- }
- }
- else
+ char *ch;
+ std::string contentType;
+ res = curl_easy_getinfo(Curl, CURLINFO_CONTENT_TYPE, &ch);
+ if (res == CURLE_OK && ch != NULL)
{
- nlwarning("this is not a number '%s'", curlresult.c_str());
+ contentType = ch;
}
- // Update the forum notification icon
-
- uint32 nbforum = 0;
- if(notifs.size() > 1 && fromString(notifs[1], nbforum))
+ // "text/lua; charset=utf8"
+ if (contentType.find("text/lua") == 0)
{
- //nlinfo("nb forum this is a number %d", nbforum);
- CInterfaceManager *pIM = CInterfaceManager::getInstance();
- if(pIM)
- {
- CCDBNodeLeaf *_CheckForumNode = NLGUI::CDBManager::getInstance()->getDbProp("UI:VARIABLES:FORUM_UPDATED");
- if(_CheckForumNode)
- {
- _CheckForumNode->setValue32(nbforum==0?0:1);
- CInterfaceElement *elm = CWidgetManager::getInstance()->getElementFromId("ui:interface:compass:forum:forum_nb");
- if (elm)
- {
- CViewText *vt = dynamic_cast(elm);
- vt->setText(toString("%d", nbforum));
- }
- }
- }
+ std::string script;
+ script = "\nlocal __WEBIG_NOTIF__= true\n" + curlresult;
+ CInterfaceManager::getInstance()->queueLuaScript(script);
}
else
{
- nlwarning("this is not a number '%s'", curlresult.c_str());
+ nlwarning("Invalid content-type '%s', expected 'text/lua'", contentType.c_str());
}
}
@@ -257,30 +235,93 @@ struct CWebigNotificationThread : public NLMISC::IRunnable
void run()
{
+ if (ClientCfg.WebIgNotifInterval == 0)
+ {
+ _Running = false;
+ nlwarning("ClientCfg.WebIgNotifInterval == 0, notification thread not running");
+ return;
+ }
+
+ std::string domain = ClientCfg.WebIgMainDomain;
+ uint32 ms = ClientCfg.WebIgNotifInterval*60*1000;
+
+ _Running = true;
// first time, we wait a small amount of time to be sure everything is initialized
- nlSleep(1*60*1000);
- while (true)
+ nlSleep(30*1000);
+ uint c = 0;
+ while (_Running)
{
- string url = "http://"+ClientCfg.WebIgMainDomain+"/index.php?app=notif&rnd="+randomString();
+ string url = "https://"+domain+"/index.php?app=notif&format=lua&rnd="+randomString();
addWebIGParams(url, true);
get(url);
- nlSleep(10*60*1000);
+
+ sleepLoop(ms);
}
}
+
+ void sleepLoop(uint ms)
+ {
+ // use smaller sleep time so stopThread() will not block too long
+ // tick == 100ms
+ uint32 ticks = ms / 100;
+ while (_Running && ticks > 0) {
+ nlSleep(100);
+ ticks--;
+ }
+ }
+
+ void startThread()
+ {
+ if (!_Thread)
+ {
+ _Thread = IThread::create(this);
+ nlassert(_Thread != NULL);
+ _Thread->start();
+ nlwarning("WebIgNotification thread started");
+ }
+ else
+ {
+ nlwarning("WebIgNotification thread already started");
+ }
+
+ }
+
+ void stopThread()
+ {
+ _Running = false;
+ if (_Thread)
+ {
+ _Thread->wait();
+ delete _Thread;
+ _Thread = NULL;
+ nlwarning("WebIgNotification thread stopped");
+ }
+ else
+ {
+ nlwarning("WebIgNotification thread already stopped");
+ }
+ }
+
+ bool isRunning() const
+ {
+ return _Running;
+ }
};
-void startWebigNotificationThread()
+static CWebigNotificationThread webigThread;
+void startWebIgNotificationThread()
{
- static bool startedWebigNotificationThread = false;
- if(!startedWebigNotificationThread)
+ if (!webigThread.isRunning())
{
- curl_global_init(CURL_GLOBAL_ALL);
- //nlinfo("startStatThread");
- CWebigNotificationThread *webigThread = new CWebigNotificationThread();
- IThread *thread = IThread::create (webigThread);
- nlassert (thread != NULL);
- thread->start ();
- startedWebigNotificationThread = true;
+ webigThread.startThread();
+ }
+}
+
+void stopWebIgNotificationThread()
+{
+ if (webigThread.isRunning())
+ {
+ webigThread.stopThread();
}
}
@@ -351,7 +392,6 @@ NLMISC_REGISTER_OBJECT(CViewBase, CGroupHTMLWebIG, std::string, "webig_html");
CGroupHTMLWebIG::CGroupHTMLWebIG(const TCtorParam ¶m)
: CGroupHTMLAuth(param)
{
- startWebigNotificationThread();
}
// ***************************************************************************
diff --git a/code/ryzom/client/src/interface_v3/group_html_webig.h b/code/ryzom/client/src/interface_v3/group_html_webig.h
index 49aac7153..62e1cfedd 100644
--- a/code/ryzom/client/src/interface_v3/group_html_webig.h
+++ b/code/ryzom/client/src/interface_v3/group_html_webig.h
@@ -20,6 +20,9 @@
#include "nel/misc/types_nl.h"
#include "nel/gui/group_html.h"
+void startWebIgNotificationThread();
+void stopWebIgNotificationThread();
+
/**
* Auth HTML group
*/
diff --git a/code/ryzom/client/src/interface_v3/interface_manager.cpp b/code/ryzom/client/src/interface_v3/interface_manager.cpp
index 158d01585..16377ac94 100644
--- a/code/ryzom/client/src/interface_v3/interface_manager.cpp
+++ b/code/ryzom/client/src/interface_v3/interface_manager.cpp
@@ -134,6 +134,8 @@ using namespace NLGUI;
#include "user_agent.h"
#include "../item_group_manager.h"
+#include "group_html_webig.h"
+
using namespace NLMISC;
namespace NLGUI
@@ -1050,6 +1052,8 @@ void CInterfaceManager::initInGame()
{
displaySystemInfo(CI18N::get("uiLogTurnedOff"));
}
+
+ startWebIgNotificationThread();
}
// ------------------------------------------------------------------------------------------------
@@ -1303,6 +1307,7 @@ void CInterfaceManager::uninitInGame0 ()
// ------------------------------------------------------------------------------------------------
void CInterfaceManager::uninitInGame1 ()
{
+ stopWebIgNotificationThread();
// release Bar Manager (HP, SAP etc... Bars)
CBarManager::getInstance()->releaseInGame();
@@ -1376,6 +1381,8 @@ void CInterfaceManager::uninitInGame1 ()
reset();
CInterfaceLink::removeAllLinks();
+ CWidgetManager::getInstance()->setPointer( NULL );
+
// Release DDX manager, before DB remove
CDDXManager::getInstance()->release();
@@ -1464,6 +1471,9 @@ void CInterfaceManager::updateFrameEvents()
H_AUTO_USE ( RZ_Client_Update_Frame_Events )
+ // lua scripts from different thread
+ flushScriptQueue();
+
flushDebugWindow();
// Handle anims done in 2 times because some AH can add or remove anims
@@ -3484,6 +3494,24 @@ void CInterfaceManager::notifyForumUpdated()
_CheckForumNode->setValue32(1);
}
+void CInterfaceManager::queueLuaScript(const std::string &script)
+{
+ CAutoMutex autoMutex(_ScriptQueueMutex);
+
+ _ScriptQueue.push(script);
+}
+
+void CInterfaceManager::flushScriptQueue()
+{
+ CAutoMutex autoMutex(_ScriptQueueMutex);
+
+ while(!_ScriptQueue.empty())
+ {
+ CLuaManager::getInstance().executeLuaScript(_ScriptQueue.front());
+ _ScriptQueue.pop();
+ }
+}
+
// ***************************************************************************
void CInterfaceManager::resetTextIndex()
diff --git a/code/ryzom/client/src/interface_v3/interface_manager.h b/code/ryzom/client/src/interface_v3/interface_manager.h
index 2aafaf0e4..4179f4bb2 100644
--- a/code/ryzom/client/src/interface_v3/interface_manager.h
+++ b/code/ryzom/client/src/interface_v3/interface_manager.h
@@ -19,8 +19,11 @@
#ifndef NL_INTERFACE_MANAGER_H
#define NL_INTERFACE_MANAGER_H
+#include
+
#include "nel/misc/types_nl.h"
#include "nel/misc/cdb_manager.h"
+#include "nel/misc/mutex.h"
#include "nel/3d/u_texture.h"
#include "nel/3d/u_text_context.h"
#include "nel/gui/interface_group.h"
@@ -427,6 +430,10 @@ public:
void notifyMailAvailable();
void notifyForumUpdated();
+ /** Queue up lua script to be run on next frame update
+ */
+ void queueLuaScript(const std::string &script);
+
/** Return true if 12-hour clock should be used
*/
static bool use12hClock();
@@ -570,6 +577,12 @@ private:
NLMISC::CCDBNodeLeaf *_CheckForumNode;
sint64 _UpdateWeatherTime;
+ // WebIG notify thread is pushing lua code here
+ std::queue _ScriptQueue;
+ NLMISC::CMutex _ScriptQueueMutex;
+
+ void flushScriptQueue();
+
// @}
/** This is the GLOBAL Action counter used to synchronize some systems (including INVENTORY) with the server.