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.