guiEngine.cpp 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621
  1. /*
  2. Minetest
  3. Copyright (C) 2013 sapier
  4. This program is free software; you can redistribute it and/or modify
  5. it under the terms of the GNU Lesser General Public License as published by
  6. the Free Software Foundation; either version 2.1 of the License, or
  7. (at your option) any later version.
  8. This program is distributed in the hope that it will be useful,
  9. but WITHOUT ANY WARRANTY; without even the implied warranty of
  10. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  11. GNU Lesser General Public License for more details.
  12. You should have received a copy of the GNU Lesser General Public License along
  13. with this program; if not, write to the Free Software Foundation, Inc.,
  14. 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
  15. */
  16. #include "guiEngine.h"
  17. #include <IGUIStaticText.h>
  18. #include <ICameraSceneNode.h>
  19. #include "client/renderingengine.h"
  20. #include "scripting_mainmenu.h"
  21. #include "util/numeric.h"
  22. #include "config.h"
  23. #include "version.h"
  24. #include "porting.h"
  25. #include "filesys.h"
  26. #include "settings.h"
  27. #include "guiMainMenu.h"
  28. #include "sound.h"
  29. #include "client/sound_openal.h"
  30. #include "client/clouds.h"
  31. #include "httpfetch.h"
  32. #include "log.h"
  33. #include "client/fontengine.h"
  34. #include "client/guiscalingfilter.h"
  35. #include "irrlicht_changes/static_text.h"
  36. #if ENABLE_GLES
  37. #include "client/tile.h"
  38. #endif
  39. /******************************************************************************/
  40. void TextDestGuiEngine::gotText(const StringMap &fields)
  41. {
  42. m_engine->getScriptIface()->handleMainMenuButtons(fields);
  43. }
  44. /******************************************************************************/
  45. void TextDestGuiEngine::gotText(const std::wstring &text)
  46. {
  47. m_engine->getScriptIface()->handleMainMenuEvent(wide_to_utf8(text));
  48. }
  49. /******************************************************************************/
  50. MenuTextureSource::~MenuTextureSource()
  51. {
  52. for (const std::string &texture_to_delete : m_to_delete) {
  53. const char *tname = texture_to_delete.c_str();
  54. video::ITexture *texture = m_driver->getTexture(tname);
  55. m_driver->removeTexture(texture);
  56. }
  57. }
  58. /******************************************************************************/
  59. video::ITexture *MenuTextureSource::getTexture(const std::string &name, u32 *id)
  60. {
  61. if (id)
  62. *id = 0;
  63. if (name.empty())
  64. return NULL;
  65. m_to_delete.insert(name);
  66. #if ENABLE_GLES
  67. video::ITexture *retval = m_driver->findTexture(name.c_str());
  68. if (retval)
  69. return retval;
  70. video::IImage *image = m_driver->createImageFromFile(name.c_str());
  71. if (!image)
  72. return NULL;
  73. image = Align2Npot2(image, m_driver);
  74. retval = m_driver->addTexture(name.c_str(), image);
  75. image->drop();
  76. return retval;
  77. #else
  78. return m_driver->getTexture(name.c_str());
  79. #endif
  80. }
  81. /******************************************************************************/
  82. /** MenuMusicFetcher */
  83. /******************************************************************************/
  84. void MenuMusicFetcher::fetchSounds(const std::string &name,
  85. std::set<std::string> &dst_paths,
  86. std::set<std::string> &dst_datas)
  87. {
  88. if(m_fetched.count(name))
  89. return;
  90. m_fetched.insert(name);
  91. std::string base;
  92. base = porting::path_share + DIR_DELIM + "sounds";
  93. dst_paths.insert(base + DIR_DELIM + name + ".ogg");
  94. int i;
  95. for(i=0; i<10; i++)
  96. dst_paths.insert(base + DIR_DELIM + name + "."+itos(i)+".ogg");
  97. base = porting::path_user + DIR_DELIM + "sounds";
  98. dst_paths.insert(base + DIR_DELIM + name + ".ogg");
  99. for(i=0; i<10; i++)
  100. dst_paths.insert(base + DIR_DELIM + name + "."+itos(i)+".ogg");
  101. }
  102. /******************************************************************************/
  103. /** GUIEngine */
  104. /******************************************************************************/
  105. GUIEngine::GUIEngine(JoystickController *joystick,
  106. gui::IGUIElement *parent,
  107. IMenuManager *menumgr,
  108. MainMenuData *data,
  109. bool &kill) :
  110. m_parent(parent),
  111. m_menumanager(menumgr),
  112. m_smgr(RenderingEngine::get_scene_manager()),
  113. m_data(data),
  114. m_kill(kill)
  115. {
  116. //initialize texture pointers
  117. for (image_definition &texture : m_textures) {
  118. texture.texture = NULL;
  119. }
  120. // is deleted by guiformspec!
  121. m_buttonhandler = new TextDestGuiEngine(this);
  122. //create texture source
  123. m_texture_source = new MenuTextureSource(RenderingEngine::get_video_driver());
  124. //create soundmanager
  125. MenuMusicFetcher soundfetcher;
  126. #if USE_SOUND
  127. if (g_settings->getBool("enable_sound"))
  128. m_sound_manager = createOpenALSoundManager(g_sound_manager_singleton.get(), &soundfetcher);
  129. #endif
  130. if(!m_sound_manager)
  131. m_sound_manager = &dummySoundManager;
  132. //create topleft header
  133. m_toplefttext = L"";
  134. core::rect<s32> rect(0, 0, g_fontengine->getTextWidth(m_toplefttext.c_str()),
  135. g_fontengine->getTextHeight());
  136. rect += v2s32(4, 0);
  137. m_irr_toplefttext = gui::StaticText::add(RenderingEngine::get_gui_env(),
  138. m_toplefttext, rect, false, true, 0, -1);
  139. //create formspecsource
  140. m_formspecgui = new FormspecFormSource("");
  141. /* Create menu */
  142. m_menu = new GUIFormSpecMenu(joystick,
  143. m_parent,
  144. -1,
  145. m_menumanager,
  146. NULL /* &client */,
  147. m_texture_source,
  148. m_formspecgui,
  149. m_buttonhandler,
  150. "",
  151. false);
  152. m_menu->allowClose(false);
  153. m_menu->lockSize(true,v2u32(800,600));
  154. // Initialize scripting
  155. infostream << "GUIEngine: Initializing Lua" << std::endl;
  156. m_script = new MainMenuScripting(this);
  157. try {
  158. m_script->setMainMenuData(&m_data->script_data);
  159. m_data->script_data.errormessage = "";
  160. if (!loadMainMenuScript()) {
  161. errorstream << "No future without main menu!" << std::endl;
  162. abort();
  163. }
  164. run();
  165. } catch (LuaError &e) {
  166. errorstream << "Main menu error: " << e.what() << std::endl;
  167. m_data->script_data.errormessage = e.what();
  168. }
  169. m_menu->quitMenu();
  170. m_menu->drop();
  171. m_menu = NULL;
  172. }
  173. /******************************************************************************/
  174. bool GUIEngine::loadMainMenuScript()
  175. {
  176. // Set main menu path (for core.get_mainmenu_path())
  177. m_scriptdir = g_settings->get("main_menu_path");
  178. if (m_scriptdir.empty()) {
  179. m_scriptdir = porting::path_share + DIR_DELIM + "builtin" + DIR_DELIM + "mainmenu";
  180. }
  181. // Load builtin (which will load the main menu script)
  182. std::string script = porting::path_share + DIR_DELIM "builtin" + DIR_DELIM "init.lua";
  183. try {
  184. m_script->loadScript(script);
  185. // Menu script loaded
  186. return true;
  187. } catch (const ModError &e) {
  188. errorstream << "GUIEngine: execution of menu script failed: "
  189. << e.what() << std::endl;
  190. }
  191. return false;
  192. }
  193. /******************************************************************************/
  194. void GUIEngine::run()
  195. {
  196. // Always create clouds because they may or may not be
  197. // needed based on the game selected
  198. video::IVideoDriver *driver = RenderingEngine::get_video_driver();
  199. cloudInit();
  200. unsigned int text_height = g_fontengine->getTextHeight();
  201. irr::core::dimension2d<u32> previous_screen_size(g_settings->getU16("screen_w"),
  202. g_settings->getU16("screen_h"));
  203. static const video::SColor sky_color(255, 140, 186, 250);
  204. // Reset fog color
  205. {
  206. video::SColor fog_color;
  207. video::E_FOG_TYPE fog_type = video::EFT_FOG_LINEAR;
  208. f32 fog_start = 0;
  209. f32 fog_end = 0;
  210. f32 fog_density = 0;
  211. bool fog_pixelfog = false;
  212. bool fog_rangefog = false;
  213. driver->getFog(fog_color, fog_type, fog_start, fog_end, fog_density,
  214. fog_pixelfog, fog_rangefog);
  215. driver->setFog(sky_color, fog_type, fog_start, fog_end, fog_density,
  216. fog_pixelfog, fog_rangefog);
  217. }
  218. while (RenderingEngine::run() && (!m_startgame) && (!m_kill)) {
  219. const irr::core::dimension2d<u32> &current_screen_size =
  220. RenderingEngine::get_video_driver()->getScreenSize();
  221. // Verify if window size has changed and save it if it's the case
  222. // Ensure evaluating settings->getBool after verifying screensize
  223. // First condition is cheaper
  224. if (previous_screen_size != current_screen_size &&
  225. current_screen_size != irr::core::dimension2d<u32>(0,0) &&
  226. g_settings->getBool("autosave_screensize")) {
  227. g_settings->setU16("screen_w", current_screen_size.Width);
  228. g_settings->setU16("screen_h", current_screen_size.Height);
  229. previous_screen_size = current_screen_size;
  230. }
  231. //check if we need to update the "upper left corner"-text
  232. if (text_height != g_fontengine->getTextHeight()) {
  233. updateTopLeftTextSize();
  234. text_height = g_fontengine->getTextHeight();
  235. }
  236. driver->beginScene(true, true, sky_color);
  237. if (m_clouds_enabled)
  238. {
  239. cloudPreProcess();
  240. drawOverlay(driver);
  241. }
  242. else
  243. drawBackground(driver);
  244. drawHeader(driver);
  245. drawFooter(driver);
  246. RenderingEngine::get_gui_env()->drawAll();
  247. driver->endScene();
  248. if (m_clouds_enabled)
  249. cloudPostProcess();
  250. else
  251. sleep_ms(25);
  252. m_script->step();
  253. #ifdef __ANDROID__
  254. m_menu->getAndroidUIInput();
  255. #endif
  256. }
  257. }
  258. /******************************************************************************/
  259. GUIEngine::~GUIEngine()
  260. {
  261. if (m_sound_manager != &dummySoundManager){
  262. delete m_sound_manager;
  263. m_sound_manager = NULL;
  264. }
  265. infostream<<"GUIEngine: Deinitializing scripting"<<std::endl;
  266. delete m_script;
  267. m_irr_toplefttext->setText(L"");
  268. //clean up texture pointers
  269. for (image_definition &texture : m_textures) {
  270. if (texture.texture)
  271. RenderingEngine::get_video_driver()->removeTexture(texture.texture);
  272. }
  273. delete m_texture_source;
  274. if (m_cloud.clouds)
  275. m_cloud.clouds->drop();
  276. }
  277. /******************************************************************************/
  278. void GUIEngine::cloudInit()
  279. {
  280. m_cloud.clouds = new Clouds(m_smgr, -1, rand());
  281. m_cloud.clouds->setHeight(100.0f);
  282. m_cloud.clouds->update(v3f(0, 0, 0), video::SColor(255,240,240,255));
  283. m_cloud.camera = m_smgr->addCameraSceneNode(0,
  284. v3f(0,0,0), v3f(0, 60, 100));
  285. m_cloud.camera->setFarValue(10000);
  286. m_cloud.lasttime = RenderingEngine::get_timer_time();
  287. }
  288. /******************************************************************************/
  289. void GUIEngine::cloudPreProcess()
  290. {
  291. u32 time = RenderingEngine::get_timer_time();
  292. if(time > m_cloud.lasttime)
  293. m_cloud.dtime = (time - m_cloud.lasttime) / 1000.0;
  294. else
  295. m_cloud.dtime = 0;
  296. m_cloud.lasttime = time;
  297. m_cloud.clouds->step(m_cloud.dtime*3);
  298. m_cloud.clouds->render();
  299. m_smgr->drawAll();
  300. }
  301. /******************************************************************************/
  302. void GUIEngine::cloudPostProcess()
  303. {
  304. float fps_max = g_settings->getFloat("pause_fps_max");
  305. // Time of frame without fps limit
  306. u32 busytime_u32;
  307. // not using getRealTime is necessary for wine
  308. u32 time = RenderingEngine::get_timer_time();
  309. if(time > m_cloud.lasttime)
  310. busytime_u32 = time - m_cloud.lasttime;
  311. else
  312. busytime_u32 = 0;
  313. // FPS limiter
  314. u32 frametime_min = 1000./fps_max;
  315. if (busytime_u32 < frametime_min) {
  316. u32 sleeptime = frametime_min - busytime_u32;
  317. RenderingEngine::get_raw_device()->sleep(sleeptime);
  318. }
  319. }
  320. /******************************************************************************/
  321. void GUIEngine::setFormspecPrepend(const std::string &fs)
  322. {
  323. if (m_menu) {
  324. m_menu->setFormspecPrepend(fs);
  325. }
  326. }
  327. /******************************************************************************/
  328. void GUIEngine::drawBackground(video::IVideoDriver *driver)
  329. {
  330. v2u32 screensize = driver->getScreenSize();
  331. video::ITexture* texture = m_textures[TEX_LAYER_BACKGROUND].texture;
  332. /* If no texture, draw background of solid color */
  333. if(!texture){
  334. video::SColor color(255,80,58,37);
  335. core::rect<s32> rect(0, 0, screensize.X, screensize.Y);
  336. driver->draw2DRectangle(color, rect, NULL);
  337. return;
  338. }
  339. v2u32 sourcesize = texture->getOriginalSize();
  340. if (m_textures[TEX_LAYER_BACKGROUND].tile)
  341. {
  342. v2u32 tilesize(
  343. MYMAX(sourcesize.X,m_textures[TEX_LAYER_BACKGROUND].minsize),
  344. MYMAX(sourcesize.Y,m_textures[TEX_LAYER_BACKGROUND].minsize));
  345. for (unsigned int x = 0; x < screensize.X; x += tilesize.X )
  346. {
  347. for (unsigned int y = 0; y < screensize.Y; y += tilesize.Y )
  348. {
  349. draw2DImageFilterScaled(driver, texture,
  350. core::rect<s32>(x, y, x+tilesize.X, y+tilesize.Y),
  351. core::rect<s32>(0, 0, sourcesize.X, sourcesize.Y),
  352. NULL, NULL, true);
  353. }
  354. }
  355. return;
  356. }
  357. /* Draw background texture */
  358. draw2DImageFilterScaled(driver, texture,
  359. core::rect<s32>(0, 0, screensize.X, screensize.Y),
  360. core::rect<s32>(0, 0, sourcesize.X, sourcesize.Y),
  361. NULL, NULL, true);
  362. }
  363. /******************************************************************************/
  364. void GUIEngine::drawOverlay(video::IVideoDriver *driver)
  365. {
  366. v2u32 screensize = driver->getScreenSize();
  367. video::ITexture* texture = m_textures[TEX_LAYER_OVERLAY].texture;
  368. /* If no texture, draw nothing */
  369. if(!texture)
  370. return;
  371. /* Draw background texture */
  372. v2u32 sourcesize = texture->getOriginalSize();
  373. draw2DImageFilterScaled(driver, texture,
  374. core::rect<s32>(0, 0, screensize.X, screensize.Y),
  375. core::rect<s32>(0, 0, sourcesize.X, sourcesize.Y),
  376. NULL, NULL, true);
  377. }
  378. /******************************************************************************/
  379. void GUIEngine::drawHeader(video::IVideoDriver *driver)
  380. {
  381. core::dimension2d<u32> screensize = driver->getScreenSize();
  382. video::ITexture* texture = m_textures[TEX_LAYER_HEADER].texture;
  383. /* If no texture, draw nothing */
  384. if(!texture)
  385. return;
  386. f32 mult = (((f32)screensize.Width / 2.0)) /
  387. ((f32)texture->getOriginalSize().Width);
  388. v2s32 splashsize(((f32)texture->getOriginalSize().Width) * mult,
  389. ((f32)texture->getOriginalSize().Height) * mult);
  390. // Don't draw the header if there isn't enough room
  391. s32 free_space = (((s32)screensize.Height)-320)/2;
  392. if (free_space > splashsize.Y) {
  393. core::rect<s32> splashrect(0, 0, splashsize.X, splashsize.Y);
  394. splashrect += v2s32((screensize.Width/2)-(splashsize.X/2),
  395. ((free_space/2)-splashsize.Y/2)+10);
  396. video::SColor bgcolor(255,50,50,50);
  397. draw2DImageFilterScaled(driver, texture, splashrect,
  398. core::rect<s32>(core::position2d<s32>(0,0),
  399. core::dimension2di(texture->getOriginalSize())),
  400. NULL, NULL, true);
  401. }
  402. }
  403. /******************************************************************************/
  404. void GUIEngine::drawFooter(video::IVideoDriver *driver)
  405. {
  406. core::dimension2d<u32> screensize = driver->getScreenSize();
  407. video::ITexture* texture = m_textures[TEX_LAYER_FOOTER].texture;
  408. /* If no texture, draw nothing */
  409. if(!texture)
  410. return;
  411. f32 mult = (((f32)screensize.Width)) /
  412. ((f32)texture->getOriginalSize().Width);
  413. v2s32 footersize(((f32)texture->getOriginalSize().Width) * mult,
  414. ((f32)texture->getOriginalSize().Height) * mult);
  415. // Don't draw the footer if there isn't enough room
  416. s32 free_space = (((s32)screensize.Height)-320)/2;
  417. if (free_space > footersize.Y) {
  418. core::rect<s32> rect(0,0,footersize.X,footersize.Y);
  419. rect += v2s32(screensize.Width/2,screensize.Height-footersize.Y);
  420. rect -= v2s32(footersize.X/2, 0);
  421. draw2DImageFilterScaled(driver, texture, rect,
  422. core::rect<s32>(core::position2d<s32>(0,0),
  423. core::dimension2di(texture->getOriginalSize())),
  424. NULL, NULL, true);
  425. }
  426. }
  427. /******************************************************************************/
  428. bool GUIEngine::setTexture(texture_layer layer, const std::string &texturepath,
  429. bool tile_image, unsigned int minsize)
  430. {
  431. video::IVideoDriver *driver = RenderingEngine::get_video_driver();
  432. if (m_textures[layer].texture) {
  433. driver->removeTexture(m_textures[layer].texture);
  434. m_textures[layer].texture = NULL;
  435. }
  436. if (texturepath.empty() || !fs::PathExists(texturepath)) {
  437. return false;
  438. }
  439. m_textures[layer].texture = driver->getTexture(texturepath.c_str());
  440. m_textures[layer].tile = tile_image;
  441. m_textures[layer].minsize = minsize;
  442. if (!m_textures[layer].texture) {
  443. return false;
  444. }
  445. return true;
  446. }
  447. /******************************************************************************/
  448. bool GUIEngine::downloadFile(const std::string &url, const std::string &target)
  449. {
  450. #if USE_CURL
  451. std::ofstream target_file(target.c_str(), std::ios::out | std::ios::binary);
  452. if (!target_file.good()) {
  453. return false;
  454. }
  455. HTTPFetchRequest fetch_request;
  456. HTTPFetchResult fetch_result;
  457. fetch_request.url = url;
  458. fetch_request.caller = HTTPFETCH_SYNC;
  459. fetch_request.timeout = g_settings->getS32("curl_file_download_timeout");
  460. httpfetch_sync(fetch_request, fetch_result);
  461. if (!fetch_result.succeeded) {
  462. target_file.close();
  463. fs::DeleteSingleFileOrEmptyDirectory(target);
  464. return false;
  465. }
  466. target_file << fetch_result.data;
  467. return true;
  468. #else
  469. return false;
  470. #endif
  471. }
  472. /******************************************************************************/
  473. void GUIEngine::setTopleftText(const std::string &text)
  474. {
  475. m_toplefttext = translate_string(utf8_to_wide(text));
  476. updateTopLeftTextSize();
  477. }
  478. /******************************************************************************/
  479. void GUIEngine::updateTopLeftTextSize()
  480. {
  481. core::rect<s32> rect(0, 0, g_fontengine->getTextWidth(m_toplefttext.c_str()),
  482. g_fontengine->getTextHeight());
  483. rect += v2s32(4, 0);
  484. m_irr_toplefttext->remove();
  485. m_irr_toplefttext = gui::StaticText::add(RenderingEngine::get_gui_env(),
  486. m_toplefttext, rect, false, true, 0, -1);
  487. }
  488. /******************************************************************************/
  489. s32 GUIEngine::playSound(const SimpleSoundSpec &spec, bool looped)
  490. {
  491. s32 handle = m_sound_manager->playSound(spec, looped);
  492. return handle;
  493. }
  494. /******************************************************************************/
  495. void GUIEngine::stopSound(s32 handle)
  496. {
  497. m_sound_manager->stopSound(handle);
  498. }
  499. /******************************************************************************/
  500. unsigned int GUIEngine::queueAsync(const std::string &serialized_func,
  501. const std::string &serialized_params)
  502. {
  503. return m_script->queueAsync(serialized_func, serialized_params);
  504. }