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
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; |
|
}
|
|
|