1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
|
#include "video-widget.hpp"
#include <opencv2/imgproc.hpp>
void cv_video_widget::update_image(const cv::Mat& frame)
{
if (fresh())
return;
auto [ W, H ] = preview_size();
if (W < 1 || H < 1 || frame.rows < 1 || frame.cols < 1)
return;
cv::Mat const* __restrict scaled = nullptr;
if (frame3.cols != W || frame3.rows != H)
{
frame3 = cv::Mat(H, W, frame.type());
frame2 = cv::Mat(H, W, CV_8UC4);
if (!frame2.isContinuous() || !frame3.isContinuous())
std::abort();
}
if (frame.cols != W || frame.rows != H)
{
cv::resize(frame, frame3, { W, H }, 0, 0, cv::INTER_NEAREST);
scaled = &frame3;
}
else if (!frame.isContinuous())
{
frame.copyTo(frame3);
scaled = &frame3;
}
else
scaled = &frame;
int color_cvt = 0;
constexpr int nchannels = 4;
switch (scaled->channels())
{
case 1:
color_cvt = cv::COLOR_GRAY2BGRA;
break;
case 3:
color_cvt = cv::COLOR_BGR2BGRA;
break;
case nchannels:
break;
default:
unreachable();
break;
}
cv::Mat const* color;
if (color_cvt != cv::COLOR_COLORCVT_MAX)
{
cv::cvtColor(*scaled, frame2, color_cvt);
color = &frame2;
}
else
color = scaled;
int width = color->cols, height = color->rows;
unsigned stride = color->step.p[0];
set_image(color->data, width, height, stride, QImage::Format_ARGB32);
set_fresh(true);
}
|