DisplayManagerWin.cpp 6.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217
  1. //
  2. // DisplayManagerWin.cpp
  3. // konvergo
  4. //
  5. // Created by Lionel CHAZALLON on 18/06/2015.
  6. //
  7. //
  8. #include <QRect>
  9. #include <math.h>
  10. #include "QsLog.h"
  11. #include "DisplayManagerWin.h"
  12. ///////////////////////////////////////////////////////////////////////////////////////////////////
  13. static DMVideoMode convertDevMode(const DEVMODEW& devmode)
  14. {
  15. DMVideoMode mode = {};
  16. mode.height = devmode.dmPelsHeight;
  17. mode.width = devmode.dmPelsWidth;
  18. mode.refreshRate = devmode.dmDisplayFrequency;
  19. mode.bitsPerPixel = devmode.dmBitsPerPel;
  20. mode.interlaced = !!(devmode.dmDisplayFlags & DM_INTERLACED);
  21. // Windows just returns integer refresh rate so let's fudge it
  22. if (mode.refreshRate == 59 ||
  23. mode.refreshRate == 29 ||
  24. mode.refreshRate == 23)
  25. mode.refreshRate = (float)(mode.refreshRate + 1) / 1.001f;
  26. return mode;
  27. }
  28. ///////////////////////////////////////////////////////////////////////////////////////////////////
  29. static bool modeEquals(const DMVideoMode& m1, const DMVideoMode& m2)
  30. {
  31. return m1.height == m2.height &&
  32. m1.width == m2.width &&
  33. fabs(m1.refreshRate - m2.refreshRate) < 1e9 &&
  34. m1.bitsPerPixel == m2.bitsPerPixel &&
  35. m1.interlaced == m2.interlaced;
  36. }
  37. ///////////////////////////////////////////////////////////////////////////////////////////////////
  38. bool DisplayManagerWin::initialize()
  39. {
  40. DISPLAY_DEVICEW displayInfo;
  41. int displayId = 0;
  42. m_displayAdapters.clear();
  43. displays.clear();
  44. while (getDisplayInfo(displayId, displayInfo))
  45. {
  46. if (displayInfo.StateFlags & (DISPLAY_DEVICE_ACTIVE | DISPLAY_DEVICE_ATTACHED))
  47. {
  48. DEVMODEW modeInfo;
  49. int modeId = 0;
  50. // add the display
  51. DMDisplayPtr display = DMDisplayPtr(new DMDisplay);
  52. display->id = displayId;
  53. display->name = QString::fromWCharArray(displayInfo.DeviceString);
  54. displays[display->id] = DMDisplayPtr(display);
  55. m_displayAdapters[display->id] = QString::fromWCharArray(displayInfo.DeviceName);
  56. while (getModeInfo(displayId, modeId, modeInfo))
  57. {
  58. // add the videomode to the display
  59. DMVideoModePtr videoMode = DMVideoModePtr(new DMVideoMode);
  60. *videoMode = convertDevMode(modeInfo);
  61. videoMode->id = modeId;
  62. display->videoModes[videoMode->id] = videoMode;
  63. modeId++;
  64. }
  65. }
  66. displayId++;
  67. }
  68. if (displays.size() == 0)
  69. {
  70. QLOG_DEBUG() << "No display found.";
  71. return false;
  72. }
  73. else
  74. return DisplayManager::initialize();
  75. }
  76. ///////////////////////////////////////////////////////////////////////////////////////////////////
  77. bool DisplayManagerWin::setDisplayMode(int display, int mode)
  78. {
  79. DEVMODEW modeInfo;
  80. if (!isValidDisplayMode(display, mode))
  81. return false;
  82. if (getModeInfo(display, mode, modeInfo))
  83. {
  84. modeInfo.dmFields = DM_PELSWIDTH | DM_PELSHEIGHT | DM_DISPLAYFREQUENCY | DM_DISPLAYFLAGS;
  85. LONG rc = ChangeDisplaySettingsExW((LPCWSTR)m_displayAdapters[display].utf16(), &modeInfo, NULL,
  86. CDS_FULLSCREEN, NULL);
  87. if (rc != DISP_CHANGE_SUCCESSFUL)
  88. {
  89. QLOG_ERROR() << "Failed to changed DisplayMode, error" << rc;
  90. return false;
  91. }
  92. else
  93. {
  94. return true;
  95. }
  96. }
  97. return false;
  98. }
  99. ///////////////////////////////////////////////////////////////////////////////////////////////////
  100. int DisplayManagerWin::getCurrentDisplayMode(int display)
  101. {
  102. if (!isValidDisplay(display))
  103. return -1;
  104. // grab current mode
  105. DEVMODEW modeInfo;
  106. ZeroMemory(&modeInfo, sizeof(modeInfo));
  107. modeInfo.dmSize = sizeof(modeInfo);
  108. // grab current mode info
  109. if (!EnumDisplaySettingsW((LPCWSTR)m_displayAdapters[display].utf16(), ENUM_CURRENT_SETTINGS,
  110. &modeInfo))
  111. {
  112. QLOG_ERROR() << "Failed to retrieve current mode";
  113. return -1;
  114. }
  115. DMVideoMode mode = convertDevMode(modeInfo);
  116. // check if current mode info matches on of our modes
  117. for (int modeId = 0; modeId < displays[display]->videoModes.size(); modeId++)
  118. {
  119. if (modeEquals(mode, *displays[display]->videoModes[modeId]))
  120. return modeId;
  121. }
  122. return -1;
  123. }
  124. ///////////////////////////////////////////////////////////////////////////////////////////////////
  125. int DisplayManagerWin::getMainDisplay()
  126. {
  127. DISPLAY_DEVICEW displayInfo;
  128. int displayId = 0;
  129. while (getDisplayInfo(displayId, displayInfo))
  130. {
  131. if (displayInfo.StateFlags & DISPLAY_DEVICE_PRIMARY_DEVICE)
  132. return displayId;
  133. displayId++;
  134. }
  135. return -1;
  136. }
  137. ///////////////////////////////////////////////////////////////////////////////////////////////////
  138. DisplayManagerWin::~DisplayManagerWin()
  139. {
  140. }
  141. ///////////////////////////////////////////////////////////////////////////////////////////////////
  142. int DisplayManagerWin::getDisplayFromPoint(int x, int y)
  143. {
  144. foreach (int displayId, displays.keys())
  145. {
  146. int currentMode = getCurrentDisplayMode(displayId);
  147. if (currentMode >= 0)
  148. {
  149. DEVMODEW modeInfo;
  150. if (getModeInfo(displayId, currentMode, modeInfo))
  151. {
  152. QRect displayRect(modeInfo.dmPosition.x, modeInfo.dmPosition.y, modeInfo.dmPelsWidth,
  153. modeInfo.dmPelsHeight);
  154. QLOG_DEBUG() << "Looking at display" << displayId << "mode" << currentMode
  155. << "at" << displayRect;
  156. if (displayRect.contains(x, y))
  157. return displayId;
  158. }
  159. }
  160. }
  161. return -1;
  162. }
  163. ///////////////////////////////////////////////////////////////////////////////////////////////////
  164. bool DisplayManagerWin::getDisplayInfo(int display, DISPLAY_DEVICEW& info)
  165. {
  166. ZeroMemory(&info, sizeof(info));
  167. info.cb = sizeof(info);
  168. return EnumDisplayDevicesW(NULL, display, &info, 0);
  169. }
  170. ///////////////////////////////////////////////////////////////////////////////////////////////////
  171. bool DisplayManagerWin::getModeInfo(int display, int mode, DEVMODEW& info)
  172. {
  173. if (m_displayAdapters.contains(display))
  174. {
  175. ZeroMemory(&info, sizeof(info));
  176. info.dmSize = sizeof(info);
  177. return EnumDisplaySettingsExW((LPCWSTR)m_displayAdapters[display].utf16(), mode, &info, 0);
  178. }
  179. return false;
  180. }