Repository for OpenCV's extra modules
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

248 lines
7.6 KiB

// This file is part of OpenCV project.
// It is subject to the license terms in the LICENSE file found in the top-level directory
// of this distribution and at http://opencv.org/license.html.
// Copyright Amir Hassan (kallaballa) <amir@viel-zu.org>
#include <opencv2/v4d/v4d.hpp>
//adapted from https://gitlab.com/wikibooks-opengl/modern-tutorials/-/blob/master/tut05_cube/cube.cpp
using namespace cv::v4d;
class CubeDemoPlan : public Plan {
public:
using Plan::Plan;
/* Demo Parameters */
int glowKernelSize_ = 0;
/* OpenGL constants */
constexpr static GLuint TRIANGLES_ = 12;
constexpr static GLuint VERTICES_INDEX_ = 0;
constexpr static GLuint COLOR_INDEX_ = 1;
//Cube vertices, colors and indices
constexpr static float VERTICES[24] = {
// Front face
0.5, 0.5, 0.5, -0.5, 0.5, 0.5, -0.5, -0.5, 0.5, 0.5, -0.5, 0.5,
// Back face
0.5, 0.5, -0.5, -0.5, 0.5, -0.5, -0.5, -0.5, -0.5, 0.5, -0.5, -0.5
};
constexpr static float VERTEX_COLORS_[24] = {
1.0, 0.4, 0.6, 1.0, 0.9, 0.2, 0.7, 0.3, 0.8, 0.5, 0.3, 1.0,
0.2, 0.6, 1.0, 0.6, 1.0, 0.4, 0.6, 0.8, 0.8, 0.4, 0.8, 0.8
};
constexpr static unsigned short TRIANGLE_INDICES_[36] = {
// Front
0, 1, 2, 2, 3, 0,
// Right
0, 3, 7, 7, 4, 0,
// Bottom
2, 6, 7, 7, 3, 2,
// Left
1, 5, 6, 6, 2, 1,
// Back
4, 7, 6, 6, 5, 4,
// Top
5, 1, 0, 0, 4, 5
};
private:
struct Cache {
cv::UMat down_;
cv::UMat up_;
cv::UMat blur_;
cv::UMat dst16_;
} cache_;
GLuint vao_ = 0;
GLuint shaderProgram_ = 0;
GLuint uniformTransform_= 0;
//Simple transform & pass-through shaders
static GLuint load_shader() {
//Shader versions "330" and "300 es" are very similar.
//If you are careful you can write the same code for both versions.
#if !defined(OPENCV_V4D_USE_ES3)
const string shaderVersion = "330";
#else
const string shaderVersion = "300 es";
#endif
const string vert =
" #version " + shaderVersion
+ R"(
precision lowp float;
layout(location = 0) in vec3 pos;
layout(location = 1) in vec3 vertex_color;
uniform mat4 transform;
out vec3 color;
void main() {
gl_Position = transform * vec4(pos, 1.0);
color = vertex_color;
}
)";
const string frag =
" #version " + shaderVersion
+ R"(
precision lowp float;
in vec3 color;
out vec4 frag_color;
void main() {
frag_color = vec4(color, 1.0);
}
)";
//Initialize the shaders and returns the program
unsigned int handles[3];
cv::v4d::initShader(handles, vert.c_str(), frag.c_str(), "fragColor");
return handles[0];
}
//Initializes objects, buffers, shaders and uniforms
static void init_scene(const cv::Size& sz, GLuint& vao, GLuint& shaderProgram, GLuint& uniformTransform) {
glEnable (GL_DEPTH_TEST);
glGenVertexArrays(1, &vao);
glBindVertexArray(vao);
unsigned int triangles_ebo;
glGenBuffers(1, &triangles_ebo);
glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, triangles_ebo);
glBufferData(GL_ELEMENT_ARRAY_BUFFER, sizeof TRIANGLE_INDICES_, TRIANGLE_INDICES_,
GL_STATIC_DRAW);
unsigned int verticies_vbo;
glGenBuffers(1, &verticies_vbo);
glBindBuffer(GL_ARRAY_BUFFER, verticies_vbo);
glBufferData(GL_ARRAY_BUFFER, sizeof VERTICES, VERTICES, GL_STATIC_DRAW);
glVertexAttribPointer(VERTICES_INDEX_, 3, GL_FLOAT, GL_FALSE, 0, NULL);
glEnableVertexAttribArray(VERTICES_INDEX_);
unsigned int colors_vbo;
glGenBuffers(1, &colors_vbo);
glBindBuffer(GL_ARRAY_BUFFER, colors_vbo);
glBufferData(GL_ARRAY_BUFFER, sizeof VERTEX_COLORS_, VERTEX_COLORS_, GL_STATIC_DRAW);
glVertexAttribPointer(COLOR_INDEX_, 3, GL_FLOAT, GL_FALSE, 0, NULL);
glEnableVertexAttribArray(COLOR_INDEX_);
glBindVertexArray(0);
glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, 0);
glBindBuffer(GL_ARRAY_BUFFER, 0);
shaderProgram = load_shader();
uniformTransform = glGetUniformLocation(shaderProgram, "transform");
glViewport(0,0, sz.width, sz.height);
}
//Renders a rotating rainbow-colored cube on a blueish background
static void render_scene(GLuint &vao, GLuint &shaderProgram,
GLuint &uniformTransform) {
//Clear the background
glClearColor(0.2, 0.24, 0.4, 1);
glClear(GL_COLOR_BUFFER_BIT);
//Use the prepared shader program
glUseProgram(shaderProgram);
//Scale and rotate the cube depending on the current time.
float angle = fmod(
double(cv::getTickCount()) / double(cv::getTickFrequency()),
2 * M_PI);
float scale = 0.25;
cv::Matx44f scaleMat(scale, 0.0, 0.0, 0.0, 0.0, scale, 0.0, 0.0, 0.0, 0.0,
scale, 0.0, 0.0, 0.0, 0.0, 1.0);
cv::Matx44f rotXMat(1.0, 0.0, 0.0, 0.0, 0.0, cos(angle), -sin(angle), 0.0,
0.0, sin(angle), cos(angle), 0.0, 0.0, 0.0, 0.0, 1.0);
cv::Matx44f rotYMat(cos(angle), 0.0, sin(angle), 0.0, 0.0, 1.0, 0.0, 0.0,
-sin(angle), 0.0, cos(angle), 0.0, 0.0, 0.0, 0.0, 1.0);
cv::Matx44f rotZMat(cos(angle), -sin(angle), 0.0, 0.0, sin(angle),
cos(angle), 0.0, 0.0, 0.0, 0.0, 1.0, 0.0, 0.0, 0.0, 0.0, 1.0);
//calculate the transform
cv::Matx44f transform = scaleMat * rotXMat * rotYMat * rotZMat;
//set the corresponding uniform
glUniformMatrix4fv(uniformTransform, 1, GL_FALSE, transform.val);
//Bind the prepared vertex array object
glBindVertexArray(vao);
//Draw
glDrawElements(GL_TRIANGLES, TRIANGLES_ * 3, GL_UNSIGNED_SHORT, NULL);
}
//applies a glow effect to an image
static void glow_effect(const cv::UMat& src, cv::UMat& dst, const int ksize, Cache& cache) {
cv::bitwise_not(src, dst);
//Resize for some extra performance
cv::resize(dst, cache.down_, cv::Size(), 0.5, 0.5);
//Cheap blur
cv::boxFilter(cache.down_, cache.blur_, -1, cv::Size(ksize, ksize), cv::Point(-1, -1), true,
cv::BORDER_REPLICATE);
//Back to original size
cv::resize(cache.blur_, cache.up_, src.size());
//Multiply the src image with a blurred version of itself
cv::multiply(dst, cache.up_, cache.dst16_, 1, CV_16U);
//Normalize and convert back to CV_8U
cv::divide(cache.dst16_, cv::Scalar::all(255.0), dst, 1, CV_8U);
cv::bitwise_not(dst, dst);
}
public:
void setup(cv::Ptr<V4D> window) override {
int diag = hypot(double(size().width), double(size().height));
glowKernelSize_ = std::max(int(diag / 138 % 2 == 0 ? diag / 138 + 1 : diag / 138), 1);
window->gl([](const cv::Size& sz, GLuint& v, GLuint& sp, GLuint& ut){
init_scene(sz, v, sp, ut);
}, size(), vao_, shaderProgram_, uniformTransform_);
}
void infer(cv::Ptr<V4D> window) override {
window->gl([](){
//Clear the background
glClearColor(0.2f, 0.24f, 0.4f, 1.0f);
glClear(GL_COLOR_BUFFER_BIT);
});
//Render using multiple OpenGL contexts
window->gl([](GLuint& v, GLuint& sp, GLuint& ut){
render_scene(v, sp, ut);
}, vao_, shaderProgram_, uniformTransform_);
//Aquire the frame buffer for use by OpenCV
window->fb([](cv::UMat& framebuffer, const cv::Rect& viewport, int glowKernelSize, Cache& cache) {
cv::UMat roi = framebuffer(viewport);
glow_effect(roi, roi, glowKernelSize, cache);
}, viewport(), glowKernelSize_, cache_);
window->write();
}
};
int main() {
cv::Ptr<CubeDemoPlan> plan = new CubeDemoPlan(cv::Size(1280, 720));
cv::Ptr<V4D> window = V4D::make(plan->size(), "Cube Demo", ALL);
//Creates a writer sink (which might be hardware accelerated)
auto sink = makeWriterSink(window, "cube-demo.mkv", 60, plan->size());
window->setSink(sink);
window->run(plan);
return 0;
}