在項目中,我們有要錄製屏幕的需求,錄製屏幕有多種方法,可以基於windows api的BitBlt 、ffmpeg的AVDevice、directx 和QT,本文講解基於QT5的抓屏。QT相對於其他幾種抓屏方式優勢是跨平臺做的比較好,接口簡單,劣勢相對於directx抓屏消耗大一些,但directx只能是windows。經過測試directx抓屏速度最快,其他接口速度基本一致。
在我的電腦上qt 1080P屏幕抓屏消耗60~80毫秒,720p消耗30~50,也就是說如果抓屏1080我一秒最多錄製10幀。這個性能要算好,不然會造成阻塞。
先獲取主屏幕然後抓屏存到QPixmap
//獲取主屏幕
QScreen *scr = QGuiApplication::primaryScreen();
//抓屏
QPixmap pix= scr->grabWindow(QApplication::desktop()->winId());
將QPixmap轉爲可以直接訪問rgb數據的QImage
就兩行代碼非常簡單,抓屏後數據存放在pix中,將QPixmap轉換爲QImage就可以取到rgb數據了。
QImage img= pix.toImage();
uchar *date = img.bits();
QImage格式分析
取到截屏數據後,我們就要進行編碼壓縮,編碼可以使用ffmpeg,需要確定取到的數據格式,可以通過img.format成員獲取,我這邊獲取的是Format_RGB32。另外要注意QT中的圖像數據不一定每個像素連續存儲,每一行會以4對齊,最簡單解決方案,就是要保證你圖像的寬度是4的倍數。如果無法保證,可以通過 img.bytesPerLine() 獲取每一行的字節數。
如果想要將桌面繪製到QWidget中測試,重載paintEvent函數
void QTScreen::paintEvent(QPaintEvent *event)
{
QScreen *scr = QGuiApplication::primaryScreen();
//抓屏
QPixmap pix= scr->grabWindow(QApplication::desktop()->winId());
QImage img= pix.toImage();
QPainter p;
p.begin(this);
p.drawImage(QPoint(0, 0), qi);
p.end();
}
更多的資料也可以關注我51CTO上的視頻課程
夏老師的課堂 http://edu.51cto.com/lecturer/12016059.html
C++跨平臺開發和音視頻技術實戰主題套餐
C++實戰FFmpeg音視頻編碼實戰屏幕錄像機視頻課程-基於QT5和ffmpeg sdk
http://edu.51cto.com/course/10359.html