DisplayComponent.cpp 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397
  1. #include "QsLog.h"
  2. #include "DisplayComponent.h"
  3. #include "DisplayManager.h"
  4. #include "settings/SettingsComponent.h"
  5. #include <QGuiApplication>
  6. #include <QWindow>
  7. #ifdef Q_OS_MAC
  8. #include "osx/DisplayManagerOSX.h"
  9. #elif defined(TARGET_RPI)
  10. #include "rpi/DisplayManagerRPI.h"
  11. #elif defined(USE_X11XRANDR)
  12. #include "x11/DisplayManagerX11.h"
  13. #elif defined(Q_OS_WIN)
  14. #include "win/DisplayManagerWin.h"
  15. #endif
  16. #include "dummy/DisplayManagerDummy.h"
  17. #include "input/InputComponent.h"
  18. ///////////////////////////////////////////////////////////////////////////////////////////////////
  19. DisplayComponent::DisplayComponent(QObject* parent) : ComponentBase(parent), m_initTimer(this)
  20. {
  21. m_displayManager = NULL;
  22. m_lastVideoMode = -1;
  23. m_lastDisplay = -1;
  24. m_applicationWindow = NULL;
  25. }
  26. ///////////////////////////////////////////////////////////////////////////////////////////////////
  27. DisplayComponent::~DisplayComponent()
  28. {
  29. }
  30. ///////////////////////////////////////////////////////////////////////////////////////////////////
  31. bool DisplayComponent::initializeDisplayManager()
  32. {
  33. m_initTimer.setSingleShot(false);
  34. bool res = false;
  35. if (m_displayManager)
  36. res = m_displayManager->initialize();
  37. emit refreshRateChanged();
  38. return res;
  39. }
  40. ///////////////////////////////////////////////////////////////////////////////////////////////////
  41. bool DisplayComponent::componentInitialize()
  42. {
  43. #if 0
  44. m_displayManager = new DisplayManagerDummy(this);
  45. #elif defined(Q_OS_MAC)
  46. m_displayManager = new DisplayManagerOSX(this);
  47. #elif defined(TARGET_RPI)
  48. m_displayManager = new DisplayManagerRPI(this);
  49. #elif defined(USE_X11XRANDR)
  50. m_displayManager = new DisplayManagerX11(this);
  51. #elif defined(Q_OS_WIN)
  52. m_displayManager = new DisplayManagerWin(this);
  53. #endif
  54. if (initializeDisplayManager())
  55. {
  56. QGuiApplication* app = (QGuiApplication*)QGuiApplication::instance();
  57. connect(app, SIGNAL(screenAdded(QScreen*)), this, SLOT(monitorChange()));
  58. connect(app, SIGNAL(screenRemoved(QScreen*)), this, SLOT(monitorChange()));
  59. foreach (QScreen *screen, app->screens())
  60. {
  61. connect(screen, SIGNAL(refreshRateChanged(qreal)), this, SLOT(monitorChange()));
  62. connect(screen, SIGNAL(geometryChanged(QRect)), this, SLOT(monitorChange()));
  63. }
  64. #ifdef TARGET_RPI
  65. // The firmware doesn't always make the best decision. Hope we do better.
  66. QLOG_INFO() << "Trying to switch to best display mode.";
  67. switchToBestOverallVideoMode(0);
  68. #endif
  69. return true;
  70. }
  71. return false;
  72. }
  73. //////////////////////////////////////////////////////////////////////////////////////////////////
  74. void DisplayComponent::monitorChange()
  75. {
  76. QLOG_INFO() << "Monitor change detected.";
  77. if (!m_initTimer.isSingleShot())
  78. {
  79. m_initTimer.setSingleShot(true);
  80. m_initTimer.singleShot(1000, this, SLOT(initializeDisplayManager()));
  81. }
  82. }
  83. //////////////////////////////////////////////////////////////////////////////////////////////////
  84. bool DisplayComponent::switchToBestVideoMode(float frameRate)
  85. {
  86. if (!m_displayManager)
  87. return false;
  88. int currentDisplay = getApplicationDisplay();
  89. if (currentDisplay < 0)
  90. {
  91. QLOG_INFO() << "Not switching rate - current display not found.";
  92. return false;
  93. }
  94. int currentMode = m_displayManager->getCurrentDisplayMode(currentDisplay);
  95. if (m_lastVideoMode < 0)
  96. {
  97. m_lastVideoMode = currentMode;
  98. m_lastDisplay = currentDisplay;
  99. }
  100. QLOG_DEBUG() << "Current display:" << currentDisplay << "mode:" << currentMode;
  101. DMMatchMediaInfo matchInfo(frameRate, false);
  102. int bestmode = m_displayManager->findBestMatch(currentDisplay, matchInfo);
  103. if (bestmode >= 0)
  104. {
  105. if (bestmode != currentMode)
  106. {
  107. QLOG_DEBUG()
  108. << "Best video matching mode is "
  109. << m_displayManager->displays[currentDisplay]->videoModes[bestmode]->getPrettyName()
  110. << "on display" << currentDisplay;
  111. return m_displayManager->setDisplayMode(currentDisplay, bestmode);
  112. }
  113. QLOG_INFO() << "No better video mode than the currently active one found.";
  114. }
  115. else
  116. {
  117. QLOG_DEBUG() << "No video mode found as better match.";
  118. }
  119. return false;
  120. }
  121. //////////////////////////////////////////////////////////////////////////////////////////////////
  122. bool DisplayComponent::switchToBestOverallVideoMode(int display)
  123. {
  124. if (!m_displayManager || !m_displayManager->isValidDisplay(display))
  125. return false;
  126. if (!SettingsComponent::Get().value(SETTINGS_SECTION_MAIN, "hdmi_poweron").toBool())
  127. {
  128. QLOG_INFO() << "Switching to best mode disabled.";
  129. return false;
  130. }
  131. int bestmode = m_displayManager->findBestMode(display);
  132. if (bestmode < 0)
  133. return false;
  134. QLOG_INFO() << "We think mode" << bestmode << "is the best mode.";
  135. if (bestmode == m_displayManager->getCurrentDisplayMode(display))
  136. {
  137. QLOG_INFO() << "This mode is the currently active one. Not switching.";
  138. return false;
  139. }
  140. if (!m_displayManager->setDisplayMode(display, bestmode))
  141. {
  142. QLOG_INFO() << "Switching mode failed.";
  143. return false;
  144. }
  145. QLOG_INFO() << "Switching mode successful.";
  146. return true;
  147. }
  148. //////////////////////////////////////////////////////////////////////////////////////////////////
  149. double DisplayComponent::currentRefreshRate()
  150. {
  151. if (!m_displayManager)
  152. return 0;
  153. int currentDisplay = getApplicationDisplay();
  154. if (currentDisplay < 0)
  155. return 0;
  156. int mode = m_displayManager->getCurrentDisplayMode(currentDisplay);
  157. if (mode < 0)
  158. return 0;
  159. return m_displayManager->displays[currentDisplay]->videoModes[mode]->refreshRate;
  160. }
  161. //////////////////////////////////////////////////////////////////////////////////////////////////
  162. bool DisplayComponent::restorePreviousVideoMode()
  163. {
  164. if (!m_displayManager)
  165. return false;
  166. if (!m_displayManager->isValidDisplayMode(m_lastDisplay, m_lastVideoMode))
  167. return false;
  168. bool ret = true;
  169. if (m_displayManager->getCurrentDisplayMode(m_lastDisplay) != m_lastVideoMode)
  170. {
  171. QLOG_DEBUG()
  172. << "Restoring VideoMode to"
  173. << m_displayManager->displays[m_lastDisplay]->videoModes[m_lastVideoMode]->getPrettyName()
  174. << "on display" << m_lastDisplay;
  175. ret = m_displayManager->setDisplayMode(m_lastDisplay, m_lastVideoMode);
  176. }
  177. m_lastVideoMode = -1;
  178. m_lastDisplay = -1;
  179. return ret;
  180. }
  181. //////////////////////////////////////////////////////////////////////////////////////////////////
  182. int DisplayComponent::getApplicationDisplay(bool silent)
  183. {
  184. QWindow* activeWindow = m_applicationWindow;
  185. int display = -1;
  186. if (activeWindow && m_displayManager)
  187. {
  188. if (!silent)
  189. {
  190. QLOG_DEBUG() << "Looking for a display at:" << activeWindow->geometry()
  191. << "(center:" << activeWindow->geometry().center() << ")";
  192. }
  193. display = m_displayManager->getDisplayFromPoint(activeWindow->geometry().center());
  194. }
  195. if (!silent)
  196. {
  197. QLOG_DEBUG() << "Display index:" << display;
  198. }
  199. return display;
  200. }
  201. /////////////////////////////////////////////////////////////////////////////////////////
  202. QString DisplayComponent::displayName(int display)
  203. {
  204. if (display < 0)
  205. return "(not found)";
  206. QString id = QString("#%0 ").arg(display);
  207. if (m_displayManager->isValidDisplay(display))
  208. return id + m_displayManager->displays[display]->name;
  209. else
  210. return id + "(not valid)";
  211. }
  212. /////////////////////////////////////////////////////////////////////////////////////////
  213. QString DisplayComponent::modePretty(int display, int mode)
  214. {
  215. if (mode < 0)
  216. return "(not found)";
  217. QString id = QString("#%0 ").arg(mode);
  218. if (m_displayManager->isValidDisplayMode(display, mode))
  219. return id + m_displayManager->displays[display]->videoModes[mode]->getPrettyName();
  220. else
  221. return id + "(not valid)";
  222. }
  223. /////////////////////////////////////////////////////////////////////////////////////////
  224. QString DisplayComponent::debugInformation()
  225. {
  226. QString debugInfo;
  227. QTextStream stream(&debugInfo);
  228. stream << "Display" << endl;
  229. if (!m_displayManager)
  230. {
  231. stream << " (no DisplayManager initialized)" << endl;
  232. }
  233. else
  234. {
  235. int display = getApplicationDisplay(true);
  236. int mode = display < 0 ? -1 : m_displayManager->getCurrentDisplayMode(display);
  237. stream << " Current screen: " << displayName(display) << endl;
  238. if (display >= 0)
  239. stream << " Current mode: " << modePretty(display, mode) << endl;
  240. if (m_displayManager->isValidDisplayMode(m_lastDisplay, m_lastVideoMode))
  241. {
  242. stream << " Switch back on screen: " << displayName(m_lastDisplay) << endl;
  243. stream << " Switch back to mode: " << modePretty(m_lastDisplay, m_lastVideoMode) << endl;
  244. }
  245. }
  246. stream << endl;
  247. stream << flush;
  248. return debugInfo;
  249. }
  250. ///////////////////////////////////////////////////////////////////////////////////////////////////
  251. static bool modeEqualsFuzzy(const DMVideoMode& m1, const DMVideoMode& m2, float tolerance)
  252. {
  253. return m1.height == m2.height &&
  254. m1.width == m2.width &&
  255. fabs(m1.refreshRate - m2.refreshRate) < tolerance &&
  256. m1.bitsPerPixel == m2.bitsPerPixel &&
  257. m1.interlaced == m2.interlaced;
  258. }
  259. /////////////////////////////////////////////////////////////////////////////////////////
  260. void DisplayComponent::switchCommand(QString command)
  261. {
  262. if (!m_displayManager)
  263. return;
  264. int currentDisplay = getApplicationDisplay();
  265. if (currentDisplay < 0)
  266. return;
  267. int id = m_displayManager->getCurrentDisplayMode(currentDisplay);
  268. if (id < 0)
  269. return;
  270. DMVideoMode current_mode = *m_displayManager->displays[currentDisplay]->videoModes[id];
  271. DMVideoMode mode = current_mode;
  272. foreach (QString a, command.split(" "))
  273. {
  274. a = a.trimmed();
  275. if (a == "p")
  276. {
  277. mode.interlaced = false;
  278. }
  279. else if (a == "i")
  280. {
  281. mode.interlaced = true;
  282. }
  283. else if (a.endsWith("hz"))
  284. {
  285. a = a.mid(0, a.size() - 2);
  286. bool ok;
  287. float rate = a.toFloat(&ok);
  288. if (ok)
  289. mode.refreshRate = rate;
  290. }
  291. else if (a.indexOf("x") >= 0)
  292. {
  293. QStringList sub = a.split("x");
  294. if (sub.size() != 2)
  295. continue;
  296. bool ok;
  297. int w = sub[0].toInt(&ok);
  298. if (!ok)
  299. continue;
  300. int h = sub[1].toInt(&ok);
  301. if (!ok)
  302. continue;
  303. mode.width = w;
  304. mode.height = h;
  305. }
  306. }
  307. QLOG_INFO() << "Current mode:" << current_mode.getPrettyName();
  308. QLOG_INFO() << "Mode requested by command:" << mode.getPrettyName();
  309. foreach (auto cur, m_displayManager->displays[currentDisplay]->videoModes)
  310. {
  311. // Require matching one digit before the decimal point.
  312. // This doesn't work if there are several modes that would match. To do
  313. // this without requiring the user to give the refresh rate with full
  314. // precision, we'd have to find a closest match instead.
  315. if (modeEqualsFuzzy(*cur, mode, 0.1))
  316. {
  317. QLOG_INFO() << "Found mode to switch to:" << cur->getPrettyName();
  318. if (m_displayManager->setDisplayMode(currentDisplay, cur->id))
  319. {
  320. m_lastDisplay = m_lastVideoMode = -1;
  321. }
  322. else
  323. {
  324. QLOG_INFO() << "Switching failed.";
  325. }
  326. return;
  327. }
  328. }
  329. QLOG_INFO() << "Requested mode not found.";
  330. }
  331. /////////////////////////////////////////////////////////////////////////////////////////
  332. void DisplayComponent::componentPostInitialize()
  333. {
  334. InputComponent::Get().registerHostCommand("switch", this, "switchCommand");
  335. if (m_displayManager)
  336. InputComponent::Get().registerHostCommand("recreateRpiUI", m_displayManager, "resetRendering");
  337. }