2019-09-10 17:23:16 +08:00
|
|
|
#include <opencv2/imgproc.hpp>
|
2019-10-18 02:04:03 +08:00
|
|
|
#include <opencv2/gapi/render/render.hpp> // Kernel API's
|
2019-09-10 17:23:16 +08:00
|
|
|
|
|
|
|
#include "api/render_ocv.hpp"
|
|
|
|
|
|
|
|
namespace cv
|
|
|
|
{
|
|
|
|
namespace gapi
|
|
|
|
{
|
|
|
|
namespace wip
|
|
|
|
{
|
|
|
|
namespace draw
|
|
|
|
{
|
|
|
|
|
2019-10-30 18:33:39 +08:00
|
|
|
// FIXME Support `decim` mosaic parameter
|
|
|
|
inline void mosaic(cv::Mat& mat, const cv::Rect &rect, int cellSz)
|
2019-09-10 17:23:16 +08:00
|
|
|
{
|
|
|
|
cv::Mat msc_roi = mat(rect);
|
|
|
|
int crop_x = msc_roi.cols - msc_roi.cols % cellSz;
|
|
|
|
int crop_y = msc_roi.rows - msc_roi.rows % cellSz;
|
|
|
|
|
2019-10-30 18:33:39 +08:00
|
|
|
for(int i = 0; i < crop_y; i += cellSz ) {
|
2019-09-10 17:23:16 +08:00
|
|
|
for(int j = 0; j < crop_x; j += cellSz) {
|
|
|
|
auto cell_roi = msc_roi(cv::Rect(j, i, cellSz, cellSz));
|
|
|
|
cell_roi = cv::mean(cell_roi);
|
|
|
|
}
|
2019-10-30 18:33:39 +08:00
|
|
|
}
|
2019-09-10 17:23:16 +08:00
|
|
|
};
|
|
|
|
|
2019-10-30 18:33:39 +08:00
|
|
|
inline void image(cv::Mat& mat,
|
|
|
|
const cv::Point& org,
|
|
|
|
const cv::Mat& img,
|
|
|
|
const cv::Mat& alpha)
|
2019-09-10 17:23:16 +08:00
|
|
|
{
|
2019-10-30 18:33:39 +08:00
|
|
|
auto roi = mat(cv::Rect(org, img.size()));
|
2019-09-10 17:23:16 +08:00
|
|
|
cv::Mat img32f_w;
|
|
|
|
cv::merge(std::vector<cv::Mat>(3, alpha), img32f_w);
|
|
|
|
|
|
|
|
cv::Mat roi32f_w(roi.size(), CV_32FC3, cv::Scalar::all(1.0));
|
|
|
|
roi32f_w -= img32f_w;
|
|
|
|
|
|
|
|
cv::Mat img32f, roi32f;
|
2019-10-30 18:33:39 +08:00
|
|
|
|
2019-09-10 17:23:16 +08:00
|
|
|
img.convertTo(img32f, CV_32F, 1.0/255);
|
|
|
|
roi.convertTo(roi32f, CV_32F, 1.0/255);
|
|
|
|
|
|
|
|
cv::multiply(img32f, img32f_w, img32f);
|
|
|
|
cv::multiply(roi32f, roi32f_w, roi32f);
|
|
|
|
roi32f += img32f;
|
|
|
|
|
|
|
|
roi32f.convertTo(roi, CV_8U, 255.0);
|
|
|
|
};
|
|
|
|
|
2019-10-30 18:33:39 +08:00
|
|
|
inline void poly(cv::Mat& mat,
|
|
|
|
const cv::gapi::wip::draw::Poly& pp)
|
2019-09-10 17:23:16 +08:00
|
|
|
{
|
2019-10-30 18:33:39 +08:00
|
|
|
std::vector<std::vector<cv::Point>> points{pp.points};
|
|
|
|
cv::fillPoly(mat, points, pp.color, pp.lt, pp.shift);
|
2019-09-10 17:23:16 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
struct BGR2YUVConverter
|
|
|
|
{
|
|
|
|
cv::Scalar cvtColor(const cv::Scalar& bgr) const
|
|
|
|
{
|
|
|
|
double y = bgr[2] * 0.299000 + bgr[1] * 0.587000 + bgr[0] * 0.114000;
|
|
|
|
double u = bgr[2] * -0.168736 + bgr[1] * -0.331264 + bgr[0] * 0.500000 + 128;
|
|
|
|
double v = bgr[2] * 0.500000 + bgr[1] * -0.418688 + bgr[0] * -0.081312 + 128;
|
|
|
|
|
|
|
|
return {y, u, v};
|
|
|
|
}
|
|
|
|
|
|
|
|
void cvtImg(const cv::Mat& in, cv::Mat& out) { cv::cvtColor(in, out, cv::COLOR_BGR2YUV); };
|
|
|
|
};
|
|
|
|
|
|
|
|
struct EmptyConverter
|
|
|
|
{
|
|
|
|
cv::Scalar cvtColor(const cv::Scalar& bgr) const { return bgr; };
|
|
|
|
void cvtImg(const cv::Mat& in, cv::Mat& out) const { out = in; };
|
|
|
|
};
|
|
|
|
|
|
|
|
// FIXME util::visitor ?
|
|
|
|
template <typename ColorConverter>
|
|
|
|
void drawPrimitivesOCV(cv::Mat &in, const Prims &prims)
|
|
|
|
{
|
|
|
|
ColorConverter converter;
|
|
|
|
for (const auto &p : prims)
|
|
|
|
{
|
|
|
|
switch (p.index())
|
|
|
|
{
|
|
|
|
case Prim::index_of<Rect>():
|
|
|
|
{
|
2019-10-30 18:33:39 +08:00
|
|
|
const auto& rp = cv::util::get<Rect>(p);
|
|
|
|
const auto color = converter.cvtColor(rp.color);
|
|
|
|
cv::rectangle(in, rp.rect, color , rp.thick);
|
2019-09-10 17:23:16 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case Prim::index_of<Text>():
|
|
|
|
{
|
2019-10-30 18:33:39 +08:00
|
|
|
const auto& tp = cv::util::get<Text>(p);
|
|
|
|
const auto color = converter.cvtColor(tp.color);
|
|
|
|
cv::putText(in, tp.text, tp.org, tp.ff, tp.fs, color, tp.thick, tp.lt, tp.bottom_left_origin);
|
2019-09-10 17:23:16 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case Prim::index_of<Circle>():
|
|
|
|
{
|
2019-10-30 18:33:39 +08:00
|
|
|
const auto& cp = cv::util::get<Circle>(p);
|
|
|
|
const auto color = converter.cvtColor(cp.color);
|
|
|
|
cv::circle(in, cp.center, cp.radius, color, cp.thick);
|
2019-09-10 17:23:16 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case Prim::index_of<Line>():
|
|
|
|
{
|
2019-10-30 18:33:39 +08:00
|
|
|
const auto& lp = cv::util::get<Line>(p);
|
|
|
|
const auto color = converter.cvtColor(lp.color);
|
|
|
|
cv::line(in, lp.pt1, lp.pt2, color, lp.thick);
|
2019-09-10 17:23:16 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case Prim::index_of<Mosaic>():
|
|
|
|
{
|
2019-10-30 18:33:39 +08:00
|
|
|
const auto& mp = cv::util::get<Mosaic>(p);
|
|
|
|
GAPI_Assert(mp.decim == 0 && "Only decim = 0 supported now");
|
|
|
|
mosaic(in, mp.mos, mp.cellSz);
|
2019-09-10 17:23:16 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case Prim::index_of<Image>():
|
|
|
|
{
|
2019-10-30 18:33:39 +08:00
|
|
|
const auto& ip = cv::util::get<Image>(p);
|
2019-09-10 17:23:16 +08:00
|
|
|
|
|
|
|
cv::Mat img;
|
2019-10-30 18:33:39 +08:00
|
|
|
converter.cvtImg(ip.img, img);
|
2019-09-10 17:23:16 +08:00
|
|
|
|
2019-10-30 18:33:39 +08:00
|
|
|
image(in, ip.org, img, ip.alpha);
|
2019-09-10 17:23:16 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case Prim::index_of<Poly>():
|
|
|
|
{
|
2019-10-30 18:33:39 +08:00
|
|
|
auto pp = cv::util::get<Poly>(p);
|
|
|
|
pp.color = converter.cvtColor(pp.color);
|
|
|
|
poly(in, pp);
|
2019-09-10 17:23:16 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
default: cv::util::throw_error(std::logic_error("Unsupported draw operation"));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void drawPrimitivesOCVBGR(cv::Mat &in, const Prims &prims)
|
|
|
|
{
|
|
|
|
drawPrimitivesOCV<EmptyConverter>(in, prims);
|
|
|
|
}
|
|
|
|
|
|
|
|
void drawPrimitivesOCVYUV(cv::Mat &in, const Prims &prims)
|
|
|
|
{
|
|
|
|
drawPrimitivesOCV<BGR2YUVConverter>(in, prims);
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace draw
|
|
|
|
} // namespace wip
|
|
|
|
} // namespace gapi
|
|
|
|
} // namespace cv
|