KonvergoWindow.cpp 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596
  1. #include "KonvergoWindow.h"
  2. #include <QTimer>
  3. #include <QJsonObject>
  4. #include <QScreen>
  5. #include <QQuickItem>
  6. #include <QGuiApplication>
  7. #include <QMessageBox>
  8. #include <QPushButton>
  9. #include "core/Version.h"
  10. #include "system/UpdaterComponent.h"
  11. #include "input/InputKeyboard.h"
  12. #include "settings/SettingsComponent.h"
  13. #include "settings/SettingsSection.h"
  14. #include "system/SystemComponent.h"
  15. #include "player/PlayerComponent.h"
  16. #include "player/PlayerQuickItem.h"
  17. #include "display/DisplayComponent.h"
  18. #include "QsLog.h"
  19. #include "utils/Utils.h"
  20. #include "Globals.h"
  21. #include "EventFilter.h"
  22. ///////////////////////////////////////////////////////////////////////////////////////////////////
  23. KonvergoWindow::KonvergoWindow(QWindow* parent) :
  24. QQuickWindow(parent),
  25. m_debugLayer(false),
  26. m_lastScale(1.0),
  27. m_ignoreFullscreenSettingsChange(0),
  28. m_showedUpdateDialog(false),
  29. m_osxPresentationOptions(0)
  30. {
  31. // NSWindowCollectionBehaviorFullScreenPrimary is only set on OSX if Qt::WindowFullscreenButtonHint is set on the window.
  32. setFlags(flags() | Qt::WindowFullscreenButtonHint);
  33. m_infoTimer = new QTimer(this);
  34. m_infoTimer->setInterval(1000);
  35. m_webDesktopMode = (SettingsComponent::Get().value(SETTINGS_SECTION_MAIN, "webMode").toString() == "desktop");
  36. installEventFilter(new EventFilter(this));
  37. connect(m_infoTimer, &QTimer::timeout, this, &KonvergoWindow::updateDebugInfo);
  38. InputComponent::Get().registerHostCommand("close", this, "close");
  39. InputComponent::Get().registerHostCommand("toggleDebug", this, "toggleDebug");
  40. InputComponent::Get().registerHostCommand("reload", this, "reloadWeb");
  41. InputComponent::Get().registerHostCommand("fullscreen", this, "toggleFullscreen");
  42. InputComponent::Get().registerHostCommand("minimize", this, "minimizeWindow");
  43. InputComponent::Get().registerHostCommand("fullscreenCurrentMode", this, "toggleFullscreenNoSwitch");
  44. #ifdef TARGET_RPI
  45. // On RPI, we use dispmanx layering - the video is on a layer below Konvergo,
  46. // and during playback the Konvergo window is partially transparent. The OSD
  47. // will be visible on top of the video as part of the Konvergo window.
  48. setColor(QColor("transparent"));
  49. #else
  50. setColor(QColor("#111111"));
  51. #endif
  52. QRect loadedGeo = loadGeometry();
  53. notifyScale(loadedGeo.size());
  54. connect(SettingsComponent::Get().getSection(SETTINGS_SECTION_MAIN), &SettingsSection::valuesUpdated,
  55. this, &KonvergoWindow::updateMainSectionSettings);
  56. connect(this, &KonvergoWindow::visibilityChanged,
  57. this, &KonvergoWindow::onVisibilityChanged);
  58. connect(this, &KonvergoWindow::enableVideoWindowSignal,
  59. this, &KonvergoWindow::enableVideoWindow, Qt::QueuedConnection);
  60. connect(&PlayerComponent::Get(), &PlayerComponent::windowVisible,
  61. this, &KonvergoWindow::playerWindowVisible);
  62. // this is using old syntax because ... reasons. QQuickCloseEvent is not public class
  63. connect(this, SIGNAL(closing(QQuickCloseEvent*)), this, SLOT(closingWindow()));
  64. connect(qApp, &QCoreApplication::aboutToQuit, this, &KonvergoWindow::closingWindow);
  65. connect(&UpdaterComponent::Get(), &UpdaterComponent::downloadComplete,
  66. this, &KonvergoWindow::showUpdateDialog);
  67. #ifdef Q_OS_MAC
  68. m_osxPresentationOptions = 0;
  69. #endif
  70. #ifdef KONVERGO_OPENELEC
  71. setVisibility(QWindow::FullScreen);
  72. #else
  73. updateWindowState(false);
  74. #endif
  75. emit enableVideoWindowSignal();
  76. }
  77. /////////////////////////////////////////////////////////////////////////////////////////
  78. void KonvergoWindow::showUpdateDialog()
  79. {
  80. if (m_webDesktopMode && !m_showedUpdateDialog)
  81. {
  82. QVariantHash updateInfo = UpdaterComponent::Get().updateInfo();
  83. QString currentVersion = Version::GetCanonicalVersionString().split("-")[0];
  84. QString newVersion = updateInfo["version"].toString().split("-")[0];
  85. QMessageBox* message = new QMessageBox(nullptr);
  86. message->setIcon(QMessageBox::Information);
  87. message->setWindowModality(Qt::ApplicationModal);
  88. message->setWindowTitle("Update found!");
  89. message->setText("An update to Plex Media Player was found");
  90. auto infoText = QString("You are currently running version %0\nDo you wish to install version %1 now?")
  91. .arg(currentVersion)
  92. .arg(newVersion);
  93. message->setInformativeText(infoText);
  94. auto details = QString("ChangeLog for version %0\n\nNew:\n%1\n\nFixed:\n%2")
  95. .arg(newVersion)
  96. .arg(updateInfo["new"].toString())
  97. .arg(updateInfo["fixed"].toString());
  98. message->setDetailedText(details);
  99. auto updateNow = message->addButton("Install Now", QMessageBox::AcceptRole);
  100. auto updateLater = message->addButton("Install on Next Restart", QMessageBox::RejectRole);
  101. message->setDefaultButton(updateNow);
  102. m_showedUpdateDialog = true;
  103. connect(message, &QMessageBox::buttonClicked, [=](QAbstractButton* button)
  104. {
  105. if (button == updateNow)
  106. UpdaterComponent::Get().doUpdate();
  107. else if (button == updateLater)
  108. message->close();
  109. message->deleteLater();
  110. });
  111. message->show();
  112. }
  113. }
  114. /////////////////////////////////////////////////////////////////////////////////////////
  115. void KonvergoWindow::closingWindow()
  116. {
  117. if (!SettingsComponent::Get().value(SETTINGS_SECTION_MAIN, "fullscreen").toBool())
  118. saveGeometry();
  119. qApp->quit();
  120. }
  121. ///////////////////////////////////////////////////////////////////////////////////////////////////
  122. KonvergoWindow::~KonvergoWindow()
  123. {
  124. DisplayComponent::Get().setApplicationWindow(nullptr);
  125. }
  126. ///////////////////////////////////////////////////////////////////////////////////////////////////
  127. bool KonvergoWindow::fitsInScreens(const QRect& rc)
  128. {
  129. for(QScreen *screen : QGuiApplication::screens())
  130. {
  131. if (screen->virtualGeometry().contains(rc))
  132. return true;
  133. }
  134. return false;
  135. }
  136. ///////////////////////////////////////////////////////////////////////////////////////////////////
  137. void KonvergoWindow::saveGeometry()
  138. {
  139. QLOG_DEBUG() << "Window state when saving geometry:" << visibility();
  140. if (visibility() != QWindow::Windowed)
  141. {
  142. QLOG_DEBUG() << "Not saving geometry.";
  143. return;
  144. }
  145. QRect rc = geometry();
  146. // lets make sure we are not saving something craycray
  147. if (rc.size().width() < windowMinSize().width() || rc.size().height() < windowMinSize().height())
  148. return;
  149. if (!fitsInScreens(rc))
  150. return;
  151. QLOG_DEBUG() << "Saving window geometry:" << rc;
  152. QVariantMap map = {{"x", rc.x()}, {"y", rc.y()},
  153. {"width", rc.width()}, {"height", rc.height()}};
  154. SettingsComponent::Get().setValue(SETTINGS_SECTION_STATE, "geometry", map);
  155. SettingsComponent::Get().setValue(SETTINGS_SECTION_STATE, "lastUsedScreen", screen()->name());
  156. }
  157. ///////////////////////////////////////////////////////////////////////////////////////////////////
  158. QRect KonvergoWindow::loadGeometry()
  159. {
  160. QRect rc = loadGeometryRect();
  161. QScreen* myScreen = loadLastScreen();
  162. if (!myScreen)
  163. myScreen = screen();
  164. QRect nsize = rc;
  165. if (SettingsComponent::Get().value(SETTINGS_SECTION_MAIN, "fullscreen").toBool())
  166. {
  167. QLOG_DEBUG() << "Load FullScreen geo...";
  168. // On OSX we need to set the geometry to the size we want when we
  169. // return from fullscreen otherwise when we exit fullscreen it
  170. // will stay small or big. On Windows we need to set it to max
  171. // resolution for the screen (i.e. fullscreen) otherwise it will
  172. // just scale the webcontent to the minimum size we have defined
  173. //
  174. #ifndef Q_OS_MAC
  175. nsize = myScreen->geometry();
  176. #endif
  177. setGeometry(nsize);
  178. setScreen(myScreen);
  179. }
  180. else
  181. {
  182. setGeometry(nsize);
  183. saveGeometry();
  184. }
  185. return nsize;
  186. }
  187. ///////////////////////////////////////////////////////////////////////////////////////////////////
  188. QRect KonvergoWindow::loadGeometryRect()
  189. {
  190. // if we dont have anything, default to 720p in the middle of the screen
  191. QRect defaultRect = QRect((screen()->geometry().width() - WEBUI_SIZE.width()) / 2,
  192. (screen()->geometry().height() - WEBUI_SIZE.height()) / 2,
  193. WEBUI_SIZE.width(), WEBUI_SIZE.height());
  194. QVariantMap map = SettingsComponent::Get().value(SETTINGS_SECTION_STATE, "geometry").toMap();
  195. if (map.isEmpty())
  196. return defaultRect;
  197. QRect rc(map["x"].toInt(), map["y"].toInt(), map["width"].toInt(), map["height"].toInt());
  198. QLOG_DEBUG() << "Restoring geo:" << rc;
  199. if (!rc.isValid() || rc.isEmpty())
  200. {
  201. QLOG_DEBUG() << "Geo bad, going for defaults";
  202. return defaultRect;
  203. }
  204. QSize minsz = windowMinSize();
  205. // Clamp to min size if we have really small values in there
  206. if (rc.size().width() < minsz.width())
  207. rc.setWidth(minsz.width());
  208. if (rc.size().height() < minsz.height())
  209. rc.setHeight(minsz.height());
  210. // also make sure we are not putting windows outside the screen somewhere
  211. if (!fitsInScreens(rc))
  212. {
  213. QLOG_DEBUG() << "Could not fit stored geo into current screens";
  214. return defaultRect;
  215. }
  216. return rc;
  217. }
  218. ///////////////////////////////////////////////////////////////////////////////////////////////////
  219. void KonvergoWindow::enableVideoWindow()
  220. {
  221. PlayerComponent::Get().setWindow(this);
  222. DisplayComponent::Get().setApplicationWindow(this);
  223. }
  224. ///////////////////////////////////////////////////////////////////////////////////////////////////
  225. void KonvergoWindow::setFullScreen(bool enable)
  226. {
  227. QLOG_DEBUG() << "setting fullscreen = " << enable;
  228. SettingsComponent::Get().setValue(SETTINGS_SECTION_MAIN, "fullscreen", enable);
  229. }
  230. ///////////////////////////////////////////////////////////////////////////////////////////////////
  231. void KonvergoWindow::setAlwaysOnTop(bool enable)
  232. {
  233. QLOG_DEBUG() << "setting always on top = " << enable;
  234. // Update the settings value.
  235. SettingsComponent::Get().setValue(SETTINGS_SECTION_MAIN, "alwaysOnTop", enable);
  236. }
  237. ///////////////////////////////////////////////////////////////////////////////////////////////////
  238. void KonvergoWindow::playerWindowVisible(bool visible)
  239. {
  240. // adjust webengineview transparecy depending on player visibility
  241. QQuickItem *web = findChild<QQuickItem *>("web");
  242. if (web)
  243. web->setProperty("backgroundColor", visible ? "transparent" : "#111111");
  244. }
  245. ///////////////////////////////////////////////////////////////////////////////////////////////////
  246. void KonvergoWindow::updateMainSectionSettings(const QVariantMap& values)
  247. {
  248. // update mouse visibility if needed
  249. if (values.find("disablemouse") != values.end())
  250. {
  251. SystemComponent::Get().setCursorVisibility(!SettingsComponent::Get().value(SETTINGS_SECTION_MAIN, "disablemouse").toBool());
  252. }
  253. if (values.contains("alwaysOnTop"))
  254. updateWindowState();
  255. if (values.contains("fullscreen") && !m_ignoreFullscreenSettingsChange)
  256. {
  257. InputComponent::Get().cancelAutoRepeat();
  258. updateWindowState();
  259. }
  260. if (values.contains("webMode"))
  261. {
  262. InputComponent::Get().cancelAutoRepeat();
  263. bool oldDesktopMode = m_webDesktopMode;
  264. bool newDesktopMode = (SettingsComponent::Get().value(SETTINGS_SECTION_MAIN, "webMode").toString() == "desktop");
  265. if (SettingsComponent::Get().value(SETTINGS_SECTION_MAIN, "layout").toString() != "auto")
  266. {
  267. m_webDesktopMode = newDesktopMode;
  268. emit webDesktopModeChanged();
  269. emit webUrlChanged();
  270. updateWindowState();
  271. notifyScale(size());
  272. }
  273. else
  274. {
  275. bool fullscreen = SettingsComponent::Get().value(SETTINGS_SECTION_MAIN, "fullscreen").toBool();
  276. if (oldDesktopMode && !newDesktopMode)
  277. fullscreen = true;
  278. else if (!oldDesktopMode && newDesktopMode)
  279. fullscreen = false;
  280. PlayerComponent::Get().stop();
  281. SettingsComponent::Get().setValue(SETTINGS_SECTION_MAIN, "fullscreen", fullscreen);
  282. QTimer::singleShot(0, [=]
  283. {
  284. m_webDesktopMode = newDesktopMode;
  285. emit webDesktopModeChanged();
  286. emit webUrlChanged();
  287. SystemComponent::Get().setCursorVisibility(true);
  288. updateWindowState();
  289. notifyScale(size());
  290. });
  291. }
  292. }
  293. if (values.contains("startupurl"))
  294. emit webUrlChanged();
  295. }
  296. ///////////////////////////////////////////////////////////////////////////////////////////////////
  297. void KonvergoWindow::updateWindowState(bool saveGeo)
  298. {
  299. if (SettingsComponent::Get().value(SETTINGS_SECTION_MAIN, "fullscreen").toBool() || SystemComponent::Get().isOpenELEC())
  300. {
  301. // if we were go from windowed to fullscreen
  302. // we want to store our current windowed position
  303. if (!isFullScreen() && saveGeo)
  304. saveGeometry();
  305. setVisibility(QWindow::FullScreen);
  306. }
  307. else
  308. {
  309. setVisibility(QWindow::Windowed);
  310. loadGeometry();
  311. Qt::WindowFlags forceOnTopFlags = Qt::WindowStaysOnTopHint;
  312. #ifdef Q_WS_X11
  313. forceOnTopFlags = forceOnTopFlags | Qt::X11BypassWindowManagerHint;
  314. #endif
  315. if (SettingsComponent::Get().value(SETTINGS_SECTION_MAIN, "alwaysOnTop").toBool())
  316. setFlags(flags() | forceOnTopFlags);
  317. else
  318. setFlags(flags() &~ forceOnTopFlags);
  319. }
  320. }
  321. ///////////////////////////////////////////////////////////////////////////////////////////////////
  322. class ScopedDecrementer
  323. {
  324. Q_DISABLE_COPY(ScopedDecrementer)
  325. int* m_value;
  326. public:
  327. ScopedDecrementer(int* value) : m_value(value) {}
  328. ~ScopedDecrementer() { (*m_value)--; }
  329. };
  330. ///////////////////////////////////////////////////////////////////////////////////////////////////
  331. void KonvergoWindow::onVisibilityChanged(QWindow::Visibility visibility)
  332. {
  333. QLOG_DEBUG() << (visibility == QWindow::FullScreen ? "FullScreen" : "Windowed") << "visibility set to " << visibility;
  334. if (visibility == QWindow::FullScreen || visibility == QWindow::Windowed)
  335. {
  336. m_ignoreFullscreenSettingsChange++;
  337. ScopedDecrementer decrement(&m_ignoreFullscreenSettingsChange);
  338. bool fs = visibility == QWindow::FullScreen;
  339. SettingsComponent::Get().setValue(SETTINGS_SECTION_MAIN, "fullscreen", fs);
  340. }
  341. if (visibility == QWindow::Windowed)
  342. {
  343. loadGeometry();
  344. #ifdef Q_OS_MAC
  345. QTimer::singleShot(1 * 1000, [&] { OSXUtils::SetPresentationOptions(m_osxPresentationOptions); });
  346. #endif
  347. }
  348. else if (visibility == QWindow::FullScreen)
  349. {
  350. #ifdef Q_OS_MAC
  351. QTimer::singleShot(1 * 1000, [&] {
  352. OSXUtils::SetPresentationOptions(m_osxPresentationOptions | OSXUtils::GetPresentationOptionsForFullscreen(!m_webDesktopMode));
  353. });
  354. #endif
  355. }
  356. if (visibility == QWindow::Minimized)
  357. InputComponent::Get().cancelAutoRepeat();
  358. notifyScale(size());
  359. }
  360. /////////////////////////////////////////////////////////////////////////////////////////
  361. void KonvergoWindow::focusOutEvent(QFocusEvent * ev)
  362. {
  363. #ifdef Q_OS_WIN32
  364. // Do this to workaround DWM compositor bugs with fullscreened OpenGL applications.
  365. // The compositor will not properly redraw anything when focusing other windows.
  366. if (visibility() == QWindow::FullScreen && SettingsComponent::Get().value(SETTINGS_SECTION_MAIN, "minimizeOnDefocus").toBool())
  367. {
  368. QLOG_DEBUG() << "minimizing window";
  369. showMinimized();
  370. }
  371. #endif
  372. }
  373. /////////////////////////////////////////////////////////////////////////////////////////
  374. void KonvergoWindow::RegisterClass()
  375. {
  376. qmlRegisterType<KonvergoWindow>("Konvergo", 1, 0, "KonvergoWindow");
  377. }
  378. /////////////////////////////////////////////////////////////////////////////////////////
  379. void KonvergoWindow::onScreenCountChanged(int newCount)
  380. {
  381. updateWindowState(false);
  382. }
  383. /////////////////////////////////////////////////////////////////////////////////////////
  384. void KonvergoWindow::updateDebugInfo()
  385. {
  386. if (m_systemDebugInfo.size() == 0)
  387. m_systemDebugInfo = SystemComponent::Get().debugInformation();
  388. m_debugInfo = m_systemDebugInfo;
  389. m_debugInfo += DisplayComponent::Get().debugInformation();
  390. PlayerQuickItem* video = findChild<PlayerQuickItem*>("video");
  391. if (video)
  392. m_debugInfo += video->debugInfo();
  393. m_videoInfo = PlayerComponent::Get().videoInformation();
  394. emit debugInfoChanged();
  395. }
  396. /////////////////////////////////////////////////////////////////////////////////////////
  397. void KonvergoWindow::toggleDebug()
  398. {
  399. if (property("showDebugLayer").toBool())
  400. {
  401. m_infoTimer->stop();
  402. setProperty("showDebugLayer", false);
  403. }
  404. else
  405. {
  406. m_infoTimer->start();
  407. updateDebugInfo();
  408. setProperty("showDebugLayer", true);
  409. }
  410. }
  411. /////////////////////////////////////////////////////////////////////////////////////////
  412. void KonvergoWindow::notifyScale(const QSize& size)
  413. {
  414. qreal scale = CalculateScale(size);
  415. if (scale != m_lastScale)
  416. {
  417. QLOG_DEBUG() << "windowScale updated to:" << scale << "webscale:" << CalculateWebScale(size, devicePixelRatio());
  418. m_lastScale = scale;
  419. emit SystemComponent::Get().scaleChanged(CalculateWebScale(size, devicePixelRatio()));
  420. }
  421. emit webScaleChanged();
  422. }
  423. /////////////////////////////////////////////////////////////////////////////////////////
  424. void KonvergoWindow::resizeEvent(QResizeEvent* event)
  425. {
  426. QLOG_DEBUG() << "resize event:" << event->size();
  427. // This next block was added at some point to workaround a problem with
  428. // resizing on windows. Unfortunately it broke the desktop client behavior
  429. // and when retried on Windows 10 with Qt5.7 the original bug seems to be
  430. // gone. I'll keep this code around until such a time that we dont get any
  431. // complaints about it.
  432. //
  433. #if 0
  434. // This next block should never really be needed in a prefect world...
  435. // Unfortunatly this is an imperfect world and on windows sometimes what
  436. // would happen on startup is that we got a resize event that would make
  437. // the window much smaller than fullscreen.
  438. //
  439. if (isFullScreen())
  440. {
  441. QSize fsSize = screen()->size();
  442. if (event->size().width() < fsSize.width() || event->size().height() < fsSize.height())
  443. {
  444. QLOG_DEBUG() << "Ignoring resize event when in fullscreen...";
  445. return;
  446. }
  447. }
  448. #endif
  449. notifyScale(event->size());
  450. QQuickWindow::resizeEvent(event);
  451. }
  452. /////////////////////////////////////////////////////////////////////////////////////////
  453. #define ROUND(x) (qRound(x * 1000) / 1000.0)
  454. /////////////////////////////////////////////////////////////////////////////////////////
  455. qreal KonvergoWindow::CalculateScale(const QSize& size)
  456. {
  457. qreal horizontalScale = (qreal)size.width() / (qreal)WEBUI_SIZE.width();
  458. qreal verticalScale = (qreal)size.height() / (qreal)WEBUI_SIZE.height();
  459. return ROUND(qMin(horizontalScale, verticalScale));
  460. }
  461. /////////////////////////////////////////////////////////////////////////////////////////
  462. qreal KonvergoWindow::CalculateWebScale(const QSize& size, qreal devicePixelRatio)
  463. {
  464. qreal horizontalScale = (qreal)size.width() / (qreal)WEBUI_SIZE.width();
  465. qreal verticalScale = (qreal)size.height() / (qreal)WEBUI_SIZE.height();
  466. qreal minScale = qMin(horizontalScale, qMin(verticalScale, (qreal)(WEBUI_MAX_HEIGHT / devicePixelRatio) / (qreal)WEBUI_SIZE.height()));
  467. qreal minWinScale = 240.0 / (qreal)WEBUI_SIZE.height();
  468. return ROUND(qMax(minWinScale, minScale));
  469. }
  470. /////////////////////////////////////////////////////////////////////////////////////////
  471. QScreen* KonvergoWindow::loadLastScreen()
  472. {
  473. QString screenName = SettingsComponent::Get().value(SETTINGS_SECTION_STATE, "lastUsedScreen").toString();
  474. if (screenName.isEmpty())
  475. return nullptr;
  476. for (QScreen* scr : QGuiApplication::screens())
  477. {
  478. if (scr->name() == screenName)
  479. return scr;
  480. }
  481. QLOG_DEBUG() << "Tried to find screen:" << screenName << "but it was not present";
  482. return nullptr;
  483. }
  484. /////////////////////////////////////////////////////////////////////////////////////////
  485. QString KonvergoWindow::webUrl()
  486. {
  487. auto url = SettingsComponent::Get().getWebClientUrl(m_webDesktopMode);
  488. if (m_webDesktopMode)
  489. return url;
  490. return url + QString("?initialScale=%0").arg(webScale());
  491. }