我正在创建一个多显示器全屏DXGI/D3D应用程序。在准备创建它们的交换链时,我正在枚举可用的输出和适配器。
当我使用DXGI的IDXGIFactory::CreateSwapChain方法创建我的交换链时,我需要提供一个交换链描述,其中包括类型为DXGI_MODE_DESC的缓冲区描述,详细说明宽度、高度、刷新率等。如何找出当前输出的设置是什么(或者如何找出当前输出的显示模式是什么)?我不想在使用此交换链进行全屏操作时更改用户的分辨率或刷新率。
我正在创建一个多显示器全屏DXGI/D3D应用程序。在准备创建它们的交换链时,我正在枚举可用的输出和适配器。
当我使用DXGI的IDXGIFactory::CreateSwapChain方法创建我的交换链时,我需要提供一个交换链描述,其中包括类型为DXGI_MODE_DESC的缓冲区描述,详细说明宽度、高度、刷新率等。如何找出当前输出的设置是什么(或者如何找出当前输出的显示模式是什么)?我不想在使用此交换链进行全屏操作时更改用户的分辨率或刷新率。
EnumDisplaySettings
,它允许我访问当前分辨率和刷新率。结合IDXGIOutput::FindClosestMatchingMode
函数,我可以非常接近于当前的显示模式。void getClosestDisplayModeToCurrent(IDXGIOutput* output, DXGI_MODE_DESC* outCurrentDisplayMode)
{
DXGI_OUTPUT_DESC outputDesc;
output->GetDesc(&outputDesc);
HMONITOR hMonitor = outputDesc.Monitor;
MONITORINFOEX monitorInfo;
monitorInfo.cbSize = sizeof(MONITORINFOEX);
GetMonitorInfo(hMonitor, &monitorInfo);
DEVMODE devMode;
devMode.dmSize = sizeof(DEVMODE);
devMode.dmDriverExtra = 0;
EnumDisplaySettings(monitorInfo.szDevice, ENUM_CURRENT_SETTINGS, &devMode);
DXGI_MODE_DESC current;
current.Width = devMode.dmPelsWidth;
current.Height = devMode.dmPelsHeight;
bool useDefaultRefreshRate = 1 == devMode.dmDisplayFrequency || 0 == devMode.dmDisplayFrequency;
current.RefreshRate.Numerator = useDefaultRefreshRate ? 0 : devMode.dmDisplayFrequency;
current.RefreshRate.Denominator = useDefaultRefreshRate ? 0 : 1;
current.Format = DXGI_FORMAT_R8G8B8A8_UNORM;
current.ScanlineOrdering = DXGI_MODE_SCANLINE_ORDER_UNSPECIFIED;
current.Scaling = DXGI_MODE_SCALING_UNSPECIFIED;
output->FindClosestMatchingMode(¤t, outCurrentDisplayMode, NULL);
}
...但我认为这并不是正确的答案,因为我需要使用传统函数。有没有办法使用DXGI来获取精确的当前显示模式,而不是使用这种方法?
我在这里看到了一个解决方案:http://www.rastertek.com/dx11tut03.html
接下来的部分:
// Now go through all the display modes and find the one that matches the screen width and height.
// When a match is found store the numerator and denominator of the refresh rate for that monitor.
for(i=0; i<numModes; i++)
{
if(displayModeList[i].Width == (unsigned int)screenWidth)
{
if(displayModeList[i].Height == (unsigned int)screenHeight)
{
numerator = displayModeList[i].RefreshRate.Numerator;
denominator = displayModeList[i].RefreshRate.Denominator;
}
}
}
This might be what you are looking for:
// Get display mode list
std::vector<DXGI_MODE_DESC*> modeList = GetDisplayModeList(*outputItor);
for(std::vector<DXGI_MODE_DESC*>::iterator modeItor = modeList.begin(); modeItor != modeList.end(); ++modeItor)
{
// PrintDisplayModeInfo(*modeItor);
}
}
std::vector<DXGI_MODE_DESC*> GetDisplayModeList(IDXGIOutput* output)
{
UINT num = 0;
DXGI_FORMAT format = DXGI_FORMAT_R32G32B32A32_TYPELESS;
UINT flags = DXGI_ENUM_MODES_INTERLACED | DXGI_ENUM_MODES_SCALING;
// Get number of display modes
output->GetDisplayModeList(format, flags, &num, 0);
// Get display mode list
DXGI_MODE_DESC * pDescs = new DXGI_MODE_DESC[num];
output->GetDisplayModeList(format, flags, &num, pDescs);
std::vector<DXGI_MODE_DESC*> displayList;
for(int i = 0; i < num; ++i)
{
displayList.push_back(&pDescs[i]);
}
return displayList;
}
DwmGetCompositionTimingInfo
API。但是在没有启用Aero的Win7上,会失败并显示DWM_E_COMPOSITIONDISABLED
。 - Soonts