feat: 添加ElaWidgetTool库

This commit is contained in:
sleepwithoutbz
2025-09-19 22:40:52 +08:00
parent 5f93e8caf6
commit 4eef5c7fd5
407 changed files with 36325 additions and 7 deletions

View File

@@ -0,0 +1,326 @@
#include "ElaDxgi.h"
#ifdef Q_OS_WIN
#include <QDateTime>
#include <QDebug>
ElaDxgi::ElaDxgi(QObject* parent)
: QObject(parent)
{
_pIsInitSuccess = false;
_pIsGrabActive = false;
_pGrabFrameRate = 120;
_pTimeoutMsValue = 50;
_pIsGrabStoped = true;
_pIsGrabCenter = false;
}
ElaDxgi::~ElaDxgi()
{
releaseInterface();
}
bool ElaDxgi::initialize(int dxID, int outputID)
{
_pIsInitSuccess = false;
_pDxDeviceID = dxID;
_pOutputDeviceID = outputID;
releaseInterface();
ID3D11Device* d3dDevice = nullptr;
ID3D11DeviceContext* d3dContext = nullptr;
D3D_FEATURE_LEVEL feat = D3D_FEATURE_LEVEL_11_0;
HRESULT hr = D3D11CreateDevice(nullptr, D3D_DRIVER_TYPE_HARDWARE, nullptr, 0, nullptr, 0,
D3D11_SDK_VERSION, &d3dDevice, &feat, &d3dContext);
if (FAILED(hr))
{
_pLastError = "Failed to D3D11CreateDevice ErrorCode = " + QString::number(uint(hr), 16);
qDebug() << _pLastError;
if (d3dDevice)
{
d3dDevice->Release();
}
if (d3dContext)
{
d3dContext->Release();
}
return false;
}
IDXGIFactory* dxgiFactory = nullptr;
CreateDXGIFactory(__uuidof(dxgiFactory), reinterpret_cast<void**>(&dxgiFactory));
IDXGIAdapter* dxgiAdapter = nullptr;
QVector<IDXGIAdapter*> dxgiAdapterVector;
for (uint i = 0; dxgiFactory->EnumAdapters(i, &dxgiAdapter) != DXGI_ERROR_NOT_FOUND; ++i)
{
dxgiAdapterVector.append(dxgiAdapter);
}
dxgiFactory->Release();
dxgiAdapter = nullptr;
_pDxDeviceList.clear();
for (int i = 0; i < dxgiAdapterVector.count(); i++)
{
IDXGIAdapter* adapt = dxgiAdapterVector.at(i);
DXGI_ADAPTER_DESC desc;
adapt->GetDesc(&desc);
_pDxDeviceList.append(QString::fromWCharArray(desc.Description));
}
if (dxID >= 0 && dxID < dxgiAdapterVector.count())
{
dxgiAdapter = dxgiAdapterVector.at(dxID);
}
if (!dxgiAdapter)
{
_pLastError = "Failed to found gpu";
qDebug() << _pLastError;
d3dDevice->Release();
d3dContext->Release();
for (auto adapter: dxgiAdapterVector)
{
adapter->Release();
}
return false;
}
IDXGIOutput* dxgiOutput = nullptr;
QVector<IDXGIOutput*> dxgiOutputVector;
for (uint i = 0; dxgiAdapter->EnumOutputs(i, &dxgiOutput) != DXGI_ERROR_NOT_FOUND; ++i)
{
dxgiOutputVector.append(dxgiOutput);
}
for (auto adapter: dxgiAdapterVector)
{
adapter->Release();
}
dxgiOutput = nullptr;
_pOutputDeviceList.clear();
for (int i = 0; i < dxgiOutputVector.count(); i++)
{
const auto& output = dxgiOutputVector.at(i);
DXGI_OUTPUT_DESC desc;
output->GetDesc(&desc);
_pOutputDeviceList.append(QString::fromWCharArray(desc.DeviceName));
}
if (outputID >= 0 && outputID < dxgiOutputVector.count())
{
dxgiOutput = dxgiOutputVector.at(outputID);
}
if (!dxgiOutput)
{
_pLastError = "Failed to found screen!";
qDebug() << _pLastError;
d3dDevice->Release();
d3dContext->Release();
for (auto output: dxgiOutputVector)
{
output->Release();
}
return false;
}
IDXGIOutput6* dxgiOutput6 = nullptr;
hr = dxgiOutput->QueryInterface(__uuidof(IDXGIOutput6), reinterpret_cast<void**>(&dxgiOutput6));
for (auto output: dxgiOutputVector)
{
output->Release();
}
if (FAILED(hr))
{
_pLastError = "Failed to QueryInterface IDXGIOutput6 ErrorCode = " + QString::number(uint(hr), 16);
qDebug() << _pLastError;
d3dDevice->Release();
d3dContext->Release();
dxgiOutput6->Release();
return false;
}
UINT supportedFormatsCount = 1; // 支持的格式数
DXGI_FORMAT supportedFormats[1] = {DXGI_FORMAT_B8G8R8A8_UNORM};
hr = dxgiOutput6->DuplicateOutput1(d3dDevice, 0, supportedFormatsCount, supportedFormats, &_duplication);
dxgiOutput6->Release();
if (FAILED(hr))
{
_pLastError = "Failed to DuplicateOutput ErrorCode = " + QString::number(uint(hr), 16);
qDebug() << _pLastError;
d3dDevice->Release();
d3dContext->Release();
if (_duplication)
{
_duplication->Release();
_duplication = nullptr;
}
return false;
}
_device = d3dDevice;
_context = d3dContext;
_pIsInitSuccess = true;
return true;
}
QImage ElaDxgi::getGrabImage() const
{
QImage grabImage(_imageBits, _descWidth, _descHeight, QImage::Format_ARGB32);
if (_pIsGrabCenter)
{
return grabImage.copy(QRect((_descWidth - _pGrabArea.width()) / 2, (_descHeight - _pGrabArea.height()) / 2, _pGrabArea.width(), _pGrabArea.height()));
}
else
{
return grabImage.copy(_pGrabArea);
}
}
void ElaDxgi::onGrabScreen()
{
if (!_duplication || !_device || !_context)
{
setIsGrabStoped(true);
return;
}
_pIsGrabStoped = false;
while (_pIsGrabActive)
{
IDXGIResource* desktopRes = nullptr;
DXGI_OUTDUPL_FRAME_INFO frameInfo;
while (true)
{
if (!_pIsGrabActive)
{
setIsGrabStoped(true);
return;
}
_grabTimer.start();
_duplication->ReleaseFrame();
HRESULT hr = _duplication->AcquireNextFrame(_pTimeoutMsValue, &frameInfo, &desktopRes);
if (FAILED(hr))
{
if (hr != DXGI_ERROR_WAIT_TIMEOUT)
{
if (desktopRes)
{
desktopRes->Release();
desktopRes = nullptr;
}
while (true)
{
if (!_duplication || !_device || !_context)
{
setIsGrabStoped(true);
return;
}
if (initialize(_pDxDeviceID, _pOutputDeviceID))
{
break;
}
}
}
continue;
}
if (frameInfo.LastPresentTime.QuadPart)
{
break;
}
}
D3D11_TEXTURE2D_DESC desc;
ID3D11Texture2D* textrueRes = nullptr;
HRESULT hr = desktopRes->QueryInterface(__uuidof(ID3D11Texture2D), reinterpret_cast<void**>(&textrueRes));
desktopRes->Release();
if (FAILED(hr))
{
qDebug() << "Failed to ID3D11Texture2D result =" << QString::number(uint(hr), 16);
continue;
}
textrueRes->GetDesc(&desc);
D3D11_TEXTURE2D_DESC texDesc;
ZeroMemory(&texDesc, sizeof(texDesc));
texDesc.Width = desc.Width;
texDesc.Height = desc.Height;
texDesc.MipLevels = 1;
texDesc.ArraySize = 1;
texDesc.SampleDesc.Count = 1;
texDesc.SampleDesc.Quality = 0;
texDesc.Usage = D3D11_USAGE_STAGING;
texDesc.Format = desc.Format;
texDesc.BindFlags = 0;
texDesc.CPUAccessFlags = D3D11_CPU_ACCESS_READ;
texDesc.MiscFlags = 0;
_device->CreateTexture2D(&texDesc, nullptr, &_texture);
_context->CopyResource(_texture, textrueRes);
IDXGISurface1* surface = nullptr;
hr = _texture->QueryInterface(__uuidof(IDXGISurface1), reinterpret_cast<void**>(&surface));
if (FAILED(hr))
{
qDebug() << "Failed to QueryInterface IDXGISurface1 ErrorCode ="
<< QString::number(uint(hr), 16);
continue;
}
DXGI_MAPPED_RECT map;
surface->Map(&map, DXGI_MAP_READ);
_imageBits = static_cast<uchar*>(map.pBits);
_descWidth = desc.Width;
_descHeight = desc.Height;
surface->Unmap();
surface->Release();
_texture->Release();
QImage grabImage(_imageBits, _descWidth, _descHeight, QImage::Format_ARGB32);
if (_pIsGrabCenter)
{
Q_EMIT grabScreenOver(std::move(grabImage.copy(QRect((_descWidth - _pGrabArea.width()) / 2, (_descHeight - _pGrabArea.height()) / 2, _pGrabArea.width(), _pGrabArea.height()))));
}
else
{
Q_EMIT grabScreenOver(std::move(grabImage.copy(_pGrabArea)));
}
if (_lastGrabTime == 0)
{
_lastGrabTime = _grabTimer.elapsed(); // 毫秒
}
else
{
_lastGrabTime = (_grabTimer.elapsed() + _lastGrabTime) / 2;
}
_cpuSleepTime = (1000 - _lastGrabTime * _pGrabFrameRate) / _pGrabFrameRate;
cpuSleep(_cpuSleepTime * 1000);
// qDebug() << _cpuSleepTime << _lastGrabTime;
}
setIsGrabStoped(true);
}
void ElaDxgi::releaseInterface()
{
if (_duplication)
{
_duplication->Release();
_duplication = nullptr;
}
if (_device)
{
_device->Release();
_device = nullptr;
}
if (_context)
{
_context->Release();
_context = nullptr;
}
}
void ElaDxgi::cpuSleep(qint64 usec)
{
if (usec <= 0)
{
return;
}
LARGE_INTEGER cpuFerq;
LARGE_INTEGER startTime;
LARGE_INTEGER endTime;
QueryPerformanceFrequency(&cpuFerq);
QueryPerformanceCounter(&startTime);
while (true)
{
QueryPerformanceCounter(&endTime);
if (((endTime.QuadPart - startTime.QuadPart) * 1000000) / cpuFerq.QuadPart > usec)
{
break;
}
}
}
#endif