VideoPlayer.cpp 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621
  1. #include "VideoPlayer.h"
  2. #include "DecodeVedio.h"
  3. #include <QPainter>
  4. #include <QResizeEvent>
  5. #include <QEventLoop>
  6. #include <QVBoxLayout>
  7. #include "spdlog/spdlog.h"
  8. #include "FmtLog/fmtlog.h"
  9. VideoPlayer::VideoPlayer(QWidget *parent) : QWidget(parent)
  10. {
  11. // /* 初始化解码线程 */
  12. // m_threadDecode = new QThread(this);
  13. // m_decodeVedio = new DecodeVedio(m_threadDecode);
  14. m_previewImage = 2;
  15. m_fps = 0;
  16. m_semRefresh = new QSemaphore(0);
  17. m_timerRefreshUI.setSingleShot(false);
  18. /* 设置精度毫秒级 */
  19. m_timerRefreshUI.setTimerType(Qt::PreciseTimer);
  20. connect(&m_timerRefreshUI, &QTimer::timeout, this, &VideoPlayer::do_refreshUI);
  21. connect(this, &VideoPlayer::signal_refreshImage, this, &VideoPlayer::do_refreshSamImage);
  22. SPDLOG_TRACE("播放器线程ID:{}", QThread::currentThreadId());
  23. QStringList listDecoder;
  24. DecodeVedio::findHWDecoder(listDecoder);
  25. if(listDecoder.isEmpty())
  26. {
  27. SPDLOG_WARN("没有找到硬件解码器");
  28. }else {
  29. SPDLOG_DEBUG("支持的硬件解码器:");
  30. for(auto it : listDecoder)
  31. {
  32. SPDLOG_DEBUG("{}", it.toStdString());
  33. }
  34. }
  35. }
  36. VideoPlayer::~VideoPlayer()
  37. {
  38. if(m_timerRefreshUI.isActive())
  39. {
  40. m_timerRefreshUI.stop();
  41. }
  42. delete m_decodeVedio;
  43. if(m_image)
  44. {
  45. delete m_image;
  46. }
  47. SPDLOG_DEBUG("视频播放器已关闭");
  48. }
  49. /**
  50. * @brief 设置播放视频,启动定时器,定时器间隔决定播放的速度
  51. * 视频的宽和高使用QImage进行缩放
  52. * 视频大小在直接设置这个类的resize即可,有最小大小限制
  53. *
  54. * @param fileName
  55. */
  56. void VideoPlayer::openPlayVedio(const QString& fileName)
  57. {
  58. if(m_isOpenFile)
  59. {
  60. m_isOpenFile = false;
  61. stop();
  62. }
  63. if(m_decodeVedio == nullptr)
  64. {
  65. /* 初始化解码线程 */
  66. m_threadDecode = new QThread(this);
  67. m_decodeVedio = new DecodeVedio(m_threadDecode);
  68. connect(m_decodeVedio, &DecodeVedio::signal_playCompleted, this, &VideoPlayer::do_playCompleted);
  69. }
  70. if(m_decodeVedio->isDecoding())
  71. {
  72. m_decodeVedio->stopDecodeVedio();
  73. }
  74. if(fileName.isEmpty())
  75. {
  76. SPDLOG_WARN("文件名为空");
  77. return;
  78. }
  79. m_fileName = fileName;
  80. m_isOpenFile = true;
  81. m_isLocalFile = isLocalFile(fileName);
  82. m_decodeVedio->openVedio(fileName);
  83. /* 获取原始视频信息 */
  84. m_srcWidth = m_decodeVedio->getSrcVideoSize().width();
  85. m_srcHeight = m_decodeVedio->getSrcVideoSize().height();
  86. m_fps = m_decodeVedio->getFPS();
  87. m_duration = m_decodeVedio->getDuration();
  88. auto totalFarame = m_decodeVedio->getTotalFrame();
  89. SPDLOG_INFO("视频编码格式:{}", m_decodeVedio->getDecoderName().toStdString());
  90. int hh = m_duration / 3600000;
  91. int mm = (m_duration % 3600000) / 60000;
  92. int ss = (m_duration % 60000) / 1000;
  93. int ms = m_duration % 1000;
  94. SPDLOG_INFO("视频分辨率:{}x{} 帧率:{} 总帧数:{}", m_srcWidth, m_srcHeight, m_fps, totalFarame);
  95. SPDLOG_INFO("时长:{}h:{}m:{}.{}s 总时长:{}ms", hh, mm, ss, ms, m_duration);
  96. /* 设置视频宽和高的最小大小 */
  97. this->setMinimumSize(160,90);
  98. /* 开启定时器刷新 */
  99. if(m_fps <= 0)
  100. {
  101. m_fps = 25;
  102. }
  103. /* 开启解码,手动刷新第一帧 */
  104. m_decodeVedio->startDecodeVedio();
  105. m_semRefresh->release(2);
  106. emit signal_refreshImage();
  107. this->show();
  108. SPDLOG_DEBUG("打开视频成功:{}", fileName.toStdString());
  109. }
  110. /* 播放视频 */
  111. bool VideoPlayer::play()
  112. {
  113. if(!m_isOpenFile)
  114. {
  115. SPDLOG_ERROR("未打开视频文件!");
  116. return false;
  117. }
  118. if(m_playStatus)
  119. {
  120. return false;
  121. }
  122. /* 设置刷新时间 */
  123. m_timerRefreshUI.setSingleShot(false);
  124. m_interval = qRound64(1000.0 / m_fps);
  125. SPDLOG_DEBUG("刷新UI的定时间隔:{}",m_interval);
  126. m_timerRefreshUI.start(m_interval);
  127. m_playStatus = true;
  128. return true;
  129. }
  130. /* 暂停播放 */
  131. void VideoPlayer::pause()
  132. {
  133. if(!m_isOpenFile)
  134. {
  135. SPDLOG_ERROR("未打开视频文件!");
  136. return;
  137. }
  138. if(!m_isLocalFile)
  139. {
  140. SPDLOG_ERROR("不是本地视频文件,无法暂停!");
  141. return;
  142. }
  143. if(!m_playStatus)
  144. {
  145. return;
  146. }
  147. m_timerRefreshUI.stop();
  148. m_playStatus = false;
  149. }
  150. /* 停止播放,停止后停止解码,将时间等复位到开始时间 */
  151. void VideoPlayer::stop()
  152. {
  153. if(!m_isOpenFile)
  154. {
  155. SPDLOG_ERROR("未打开视频文件!");
  156. return;
  157. }
  158. SPDLOG_DEBUG("...停止播放...");
  159. m_fileName = QString();
  160. if(m_timerRefreshUI.isActive())
  161. {
  162. m_timerRefreshUI.stop();
  163. }
  164. // SPDLOG_DEBUG("...停止解码...");
  165. /* 删除解码器 */
  166. delete m_decodeVedio;
  167. m_decodeVedio = nullptr;
  168. delete m_threadDecode;
  169. m_threadDecode = nullptr;
  170. m_playStatus = false;
  171. m_isOpenFile = false;
  172. /* 绘制黑帧 */
  173. SPDLOG_DEBUG("绘制黑帧");
  174. m_image = new QImage(m_nowWidth, m_nowHeight, QImage::Format_RGB32);
  175. m_image->fill(Qt::black);
  176. update();
  177. }
  178. /* 后退,单位ms */
  179. void VideoPlayer::backward(qint64 ms)
  180. {
  181. if(!m_isOpenFile)
  182. {
  183. SPDLOG_ERROR("未打开视频文件!");
  184. return;
  185. }
  186. if(!m_isLocalFile)
  187. {
  188. SPDLOG_ERROR("不是本地视频文件,无法后退!");
  189. return;
  190. }
  191. /* 获取当前位置 */
  192. qint64 pos = m_decodeVedio->getCurrentPos();
  193. pos = pos - ms;
  194. if(pos < 0)
  195. {
  196. pos = 0;
  197. }
  198. setCurrentPos(pos);
  199. }
  200. /* 前进,单位ms */
  201. void VideoPlayer::forward(qint64 ms)
  202. {
  203. if(!m_isOpenFile)
  204. {
  205. SPDLOG_ERROR("未打开视频文件!");
  206. return;
  207. }
  208. if(!m_isLocalFile)
  209. {
  210. SPDLOG_ERROR("不是本地视频文件,无法前进!");
  211. return;
  212. }
  213. /* 获取当前位置 */
  214. qint64 pos = m_decodeVedio->getCurrentPos();
  215. SPDLOG_DEBUG("pos:{} ms:{}", pos, ms);
  216. pos = pos + ms;
  217. setCurrentPos(pos);
  218. }
  219. /* 获取视频时长 */
  220. qint64 VideoPlayer::getDuration()
  221. {
  222. if(!m_isOpenFile)
  223. {
  224. SPDLOG_ERROR("未打开视频文件!");
  225. return -1;
  226. }
  227. auto duration = m_decodeVedio->getDuration();
  228. if(duration <= 0)
  229. {
  230. return 0;
  231. }
  232. return duration;
  233. }
  234. /* 获取当前播放位置 */
  235. qint64 VideoPlayer::getCurrentPos()
  236. {
  237. if(!m_isOpenFile)
  238. {
  239. SPDLOG_ERROR("未打开视频文件!");
  240. return -1;
  241. }
  242. auto pos = m_decodeVedio->getCurrentPos();
  243. if(pos < 0)
  244. {
  245. return 0;
  246. }
  247. return pos;
  248. }
  249. /* 设置当前播放位置,单位ms */
  250. void VideoPlayer::setCurrentPos(qint64 pos)
  251. {
  252. if(!m_isOpenFile)
  253. {
  254. SPDLOG_ERROR("未打开视频文件!");
  255. return;
  256. }
  257. if(!m_isLocalFile)
  258. {
  259. SPDLOG_ERROR("不是本地视频文件,无法设置播放位置!");
  260. return;
  261. }
  262. if(pos < 0)
  263. {
  264. pos = 0;
  265. }
  266. /* 先停止播放 */
  267. bool temp = m_playStatus;
  268. if(m_playStatus)
  269. {
  270. m_timerRefreshUI.stop();
  271. m_playStatus = false;
  272. }
  273. m_decodeVedio->setCurrentPos(pos);
  274. /* 继续播放 */
  275. if(temp)
  276. {
  277. // SPDLOG_INFO("..........开启定时器..........");
  278. m_timerRefreshUI.start(m_interval);
  279. m_playStatus = true;
  280. }else
  281. {
  282. /* 刷新2张照片 */
  283. m_semRefresh->release(m_previewImage);
  284. emit signal_refreshImage();
  285. }
  286. }
  287. /* 设置播放视频大小 */
  288. void VideoPlayer::setPlayWidgetSize(int width,int height)
  289. {
  290. /* 对宽和高就行缩放,保持比例,同时将其居中放置
  291. * 先计算出比例,和16/9相对比
  292. * 大于16/9,以高为最大极限,计算出宽度和x坐标
  293. * 小于16/9,以宽为最大极限,计算出高度和y坐标 */
  294. double srcRatio = m_srcWidth*1.0 / m_srcHeight;
  295. double ratio = width*1.0 / height;
  296. long w1 = 0, h1 = 0;
  297. int srcX = this->pos().rx(), srcY = this->pos().ry();
  298. int x1 = srcX, y1 = srcY;
  299. if(ratio > srcRatio)
  300. {
  301. w1 = height * srcRatio;
  302. x1 = (width - w1) / 2;
  303. h1 = height;
  304. y1 = srcY;
  305. }
  306. else if(ratio < srcRatio)
  307. {
  308. h1 = width / srcRatio;
  309. y1 = (height - h1) / 2;
  310. w1 = width;
  311. x1 = srcX;
  312. }else {
  313. w1 = width;
  314. h1 = height;
  315. x1 = srcX;
  316. y1 = srcY;
  317. }
  318. this->move(x1, y1);
  319. m_nowWidth = w1;
  320. m_nowHeight = h1;
  321. this->resize(w1, h1);
  322. // SPDLOG_DEBUG("设置窗口位置:{}x{}, 大小:{}x{}, 传入大小:{}x{}", x1, y1, w1, h1, width, height);
  323. SPDLOG_DEBUG("现在位置和大小:{}x{}, {}x{}", this->pos().rx(), this->pos().ry(), this->width(), this->height());
  324. }
  325. /**
  326. * @brief 设置播放窗口,这用于独占一个传入的widget,这里会自动添加一个布局,外面窗口变化,这里也跟随着变化
  327. *
  328. * @param widget
  329. * @param flag
  330. * @arg true:独占widget,并设置一个layout,会随着传入的widget大小变化
  331. * @arg false:不独占
  332. */
  333. void VideoPlayer::setPlayWidget(QWidget* widget, bool flag)
  334. {
  335. if(widget == nullptr)
  336. {
  337. SPDLOG_WARN("传入的widget为空");
  338. return;
  339. }
  340. if(flag)
  341. {
  342. /* 设置布局 */
  343. QVBoxLayout* layout = new QVBoxLayout(widget);
  344. layout->addWidget(this);
  345. layout->setMargin(0);
  346. layout->setSpacing(0);
  347. widget->setLayout(layout);
  348. }else
  349. {
  350. this->setParent(widget);
  351. /* 设置窗口大小 */
  352. setPlayWidgetSize(widget->width(), widget->height());
  353. }
  354. }
  355. /**
  356. * @brief 设置预览图片数目,在暂停时跳转,可能会有花屏或者黑帧,可以设置跳转图片个数跳过黑帧
  357. * 默认是2帧
  358. *
  359. * @param num
  360. */
  361. void VideoPlayer::setPreviewImage(int num)
  362. {
  363. m_previewImage = num;
  364. }
  365. /**
  366. * @brief 设置帧率,有些视频无法获取到帧率,就会使用默认的25fps,如果需要,可以通过这个函数设置
  367. * 注意:这个函数需要在打开视频文件之后设置,打开一次视频文件会覆盖这个参数
  368. *
  369. * @param fps
  370. */
  371. void VideoPlayer::setFPS(int fps)
  372. {
  373. m_fps = fps;
  374. if(m_decodeVedio != nullptr)
  375. {
  376. m_decodeVedio->setFPS(fps);
  377. }
  378. if(m_timerRefreshUI.isActive())
  379. {
  380. m_timerRefreshUI.stop();
  381. m_interval = qRound64(1000.0 / m_fps);
  382. m_timerRefreshUI.start(m_interval);
  383. }
  384. }
  385. /* 设置播放回调函数 */
  386. // void VideoPlayer::setPlayCallBack(std::function<Play_CallBack> playCallBack,void* context)
  387. // {
  388. // m_funcPlayCB = playCallBack;
  389. // m_context = context;
  390. // }
  391. void VideoPlayer::paintEvent(QPaintEvent *event)
  392. {
  393. if(m_image != nullptr)
  394. {
  395. // SPDLOG_TRACE("开始绘制画面...");
  396. /* 对图像进行缩放 */
  397. if(m_srcWidth != m_nowWidth || m_srcHeight != m_nowHeight)
  398. {
  399. *m_image = m_image->scaled(m_nowWidth, m_nowHeight, Qt::KeepAspectRatio, Qt::SmoothTransformation);
  400. }
  401. QPainter painter(this);
  402. painter.drawImage(0, 0, *m_image);
  403. }
  404. }
  405. void VideoPlayer::resizeEvent(QResizeEvent *event)
  406. {
  407. SPDLOG_TRACE("窗口大小改变...");
  408. m_nowWidth = event->size().width();
  409. m_nowHeight = event->size().height();
  410. QWidget::resizeEvent(event);
  411. }
  412. /* 刷新一张图片,直到有图片为止 */
  413. void VideoPlayer::refreshOneUIUntilHave()
  414. {
  415. if(m_decodeVedio != nullptr)
  416. {
  417. // SPDLOG_DEBUG("取出一帧图片...");
  418. /* 删除上一帧图片 */
  419. if(m_image != nullptr)
  420. {
  421. delete m_image;
  422. m_image = nullptr;
  423. }
  424. /* 如果没有图片,这个函数会阻塞 */
  425. m_image = m_decodeVedio->getOneImageUntilHave();
  426. if(m_image)
  427. {
  428. if(m_image->isNull())
  429. {
  430. SPDLOG_WARN("取出的图片为空...");
  431. return;
  432. }
  433. // SPDLOG_DEBUG("绘制画面...");
  434. update();
  435. }
  436. }
  437. }
  438. /* 双击事件函数 */
  439. void VideoPlayer::mouseDoubleClickEvent(QMouseEvent *event)
  440. {
  441. if(event->button() == Qt::LeftButton)
  442. {
  443. // SPDLOG_DEBUG("双击事件...");
  444. // if(m_funcPlayCB != nullptr)
  445. // {
  446. // m_funcPlayCB(this, 5, nullptr, 0, m_context);
  447. // }else {
  448. // SPDLOG_INFO("没有设置回调函数");
  449. // }
  450. }
  451. }
  452. /* 取出画面,刷新UI */
  453. void VideoPlayer::do_refreshUI()
  454. {
  455. if(m_decodeVedio != nullptr)
  456. {
  457. // SPDLOG_DEBUG("取出一帧图片...");
  458. /* 删除上一帧图片 */
  459. if(m_image != nullptr)
  460. {
  461. delete m_image;
  462. m_image = nullptr;
  463. }
  464. m_image = m_decodeVedio->getOneImage();
  465. if(m_image)
  466. {
  467. if(m_image->isNull())
  468. {
  469. SPDLOG_WARN("取出的图片为空...");
  470. return;
  471. }
  472. // SPDLOG_DEBUG("绘制画面...");
  473. update();
  474. }
  475. // m_decodeVedio->wakeUpCondQueueNoEmpty();
  476. }
  477. }
  478. /* 通过信号刷新第一张图片 */
  479. void VideoPlayer::do_refreshSamImage()
  480. {
  481. if(!m_isOpenFile)
  482. {
  483. return;
  484. }
  485. while(m_semRefresh->tryAcquire(1))
  486. {
  487. /* 取出第一张 */
  488. if(m_decodeVedio != nullptr)
  489. {
  490. // SPDLOG_DEBUG("取出一帧图片...");
  491. /* 删除上一帧图片 */
  492. if(m_image != nullptr)
  493. {
  494. delete m_image;
  495. m_image = nullptr;
  496. }
  497. /* 等待图片,最多等待50ms */
  498. m_image = m_decodeVedio->getOneImageUntilHave(100);
  499. if(m_image)
  500. {
  501. if(m_image->isNull())
  502. {
  503. SPDLOG_WARN("取出的图片为空...");
  504. return;
  505. }
  506. SPDLOG_DEBUG("绘制预览画面。");
  507. update();
  508. }
  509. }
  510. }
  511. }
  512. /* 播放完成 */
  513. void VideoPlayer::do_playCompleted()
  514. {
  515. SPDLOG_INFO("视频播放完成。");
  516. m_timerRefreshUI.stop();
  517. /* 手动刷新剩余的环形队列中的图片 */
  518. while(true)
  519. {
  520. if(m_decodeVedio != nullptr)
  521. {
  522. QImage* image = nullptr;
  523. image = m_decodeVedio->getOneImage();
  524. if(image == nullptr)
  525. {
  526. break;
  527. }
  528. /* 删除上一帧图片 */
  529. if(m_image != nullptr)
  530. {
  531. delete m_image;
  532. m_image = nullptr;
  533. }
  534. m_image = image;
  535. if(m_image->isNull())
  536. {
  537. SPDLOG_WARN("取出的图片为空...");
  538. return;
  539. }
  540. // SPDLOG_DEBUG("绘制画面...");
  541. update();
  542. }
  543. }
  544. m_playStatus = false;
  545. // if(m_funcPlayCB != nullptr)
  546. // {
  547. // /* 播放完成的回调函数 */
  548. // m_funcPlayCB(this, 2, nullptr, 0, m_context);
  549. // }
  550. }
  551. /* 判断是否是本地文件 */
  552. bool VideoPlayer::isLocalFile(const QString& fileName)
  553. {
  554. if(fileName.isEmpty())
  555. {
  556. return false;
  557. }
  558. if(fileName.startsWith("http://") || fileName.startsWith("rtsp://")
  559. || fileName.startsWith("rtmp://") || fileName.startsWith("https://"))
  560. {
  561. return false;
  562. }
  563. return true;
  564. }