|
|
|
@ -18,6 +18,7 @@ static GLuint fontTex;
|
|
|
|
|
static GLint uniMVP;
|
|
|
|
|
static GLint uniClipRect;
|
|
|
|
|
|
|
|
|
|
// This is the main rendering function that you have to implement and provide to ImGui (via setting up 'RenderDrawListsFn' in the ImGuiIO structuer)
|
|
|
|
|
static void ImImpl_RenderDrawLists(ImDrawList** const cmd_lists, int cmd_lists_count)
|
|
|
|
|
{
|
|
|
|
|
size_t total_vtx_count = 0;
|
|
|
|
@ -26,24 +27,7 @@ static void ImImpl_RenderDrawLists(ImDrawList** const cmd_lists, int cmd_lists_c
|
|
|
|
|
if (total_vtx_count == 0)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
int read_pos_clip_rect_buf = 0; // offset in 'clip_rect_buffer'. each PushClipRect command consume 1 of those.
|
|
|
|
|
|
|
|
|
|
ImVector<ImVec4> clip_rect_stack;
|
|
|
|
|
clip_rect_stack.push_back(ImVec4(-9999,-9999,+9999,+9999));
|
|
|
|
|
|
|
|
|
|
// Setup orthographic projection
|
|
|
|
|
const float L = 0.0f;
|
|
|
|
|
const float R = ImGui::GetIO().DisplaySize.x;
|
|
|
|
|
const float B = ImGui::GetIO().DisplaySize.y;
|
|
|
|
|
const float T = 0.0f;
|
|
|
|
|
const float mvp[4][4] =
|
|
|
|
|
{
|
|
|
|
|
{ 2.0f/(R-L), 0.0f, 0.0f, 0.0f },
|
|
|
|
|
{ 0.0f, 2.0f/(T-B), 0.0f, 0.0f },
|
|
|
|
|
{ 0.0f, 0.0f, -1.0f, 0.0f },
|
|
|
|
|
{ -(R+L)/(R-L), -(T+B)/(T-B), 0.0f, 1.0f },
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// Copy all vertices into a single contiguous GL buffer
|
|
|
|
|
glBindBuffer(GL_ARRAY_BUFFER, vbo);
|
|
|
|
|
glBindVertexArray(vao);
|
|
|
|
|
glBufferData(GL_ARRAY_BUFFER, total_vtx_count * sizeof(ImDrawVert), NULL, GL_STREAM_DRAW);
|
|
|
|
@ -54,37 +38,51 @@ static void ImImpl_RenderDrawLists(ImDrawList** const cmd_lists, int cmd_lists_c
|
|
|
|
|
for (int n = 0; n < cmd_lists_count; n++)
|
|
|
|
|
{
|
|
|
|
|
const ImDrawList* cmd_list = cmd_lists[n];
|
|
|
|
|
if (!cmd_list->vtx_buffer.empty())
|
|
|
|
|
{
|
|
|
|
|
memcpy(buffer_data, &cmd_list->vtx_buffer[0], cmd_list->vtx_buffer.size() * sizeof(ImDrawVert));
|
|
|
|
|
buffer_data += cmd_list->vtx_buffer.size() * sizeof(ImDrawVert);
|
|
|
|
|
vtx_consumed += cmd_list->vtx_buffer.size();
|
|
|
|
|
}
|
|
|
|
|
memcpy(buffer_data, &cmd_list->vtx_buffer[0], cmd_list->vtx_buffer.size() * sizeof(ImDrawVert));
|
|
|
|
|
buffer_data += cmd_list->vtx_buffer.size() * sizeof(ImDrawVert);
|
|
|
|
|
vtx_consumed += cmd_list->vtx_buffer.size();
|
|
|
|
|
}
|
|
|
|
|
glUnmapBuffer(GL_ARRAY_BUFFER);
|
|
|
|
|
|
|
|
|
|
glUseProgram(shaderProgram);
|
|
|
|
|
glUniformMatrix4fv(uniMVP, 1, GL_FALSE, &mvp[0][0]);
|
|
|
|
|
|
|
|
|
|
// Setup render state: alpha-blending enabled, no face culling, no depth testing
|
|
|
|
|
glEnable(GL_BLEND);
|
|
|
|
|
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
|
|
|
|
glDisable(GL_CULL_FACE);
|
|
|
|
|
glDisable(GL_DEPTH_TEST);
|
|
|
|
|
|
|
|
|
|
// Bind texture and enable our shader
|
|
|
|
|
glBindTexture(GL_TEXTURE_2D, fontTex);
|
|
|
|
|
glUseProgram(shaderProgram);
|
|
|
|
|
|
|
|
|
|
vtx_consumed = 0; // offset in vertex buffer. each command consume ImDrawCmd::vtx_count of those
|
|
|
|
|
bool clip_rect_dirty = true;
|
|
|
|
|
// Setup orthographic projection matrix
|
|
|
|
|
const float L = 0.0f;
|
|
|
|
|
const float R = ImGui::GetIO().DisplaySize.x;
|
|
|
|
|
const float B = ImGui::GetIO().DisplaySize.y;
|
|
|
|
|
const float T = 0.0f;
|
|
|
|
|
const float mvp[4][4] =
|
|
|
|
|
{
|
|
|
|
|
{ 2.0f/(R-L), 0.0f, 0.0f, 0.0f },
|
|
|
|
|
{ 0.0f, 2.0f/(T-B), 0.0f, 0.0f },
|
|
|
|
|
{ 0.0f, 0.0f, -1.0f, 0.0f },
|
|
|
|
|
{ -(R+L)/(R-L), -(T+B)/(T-B), 0.0f, 1.0f },
|
|
|
|
|
};
|
|
|
|
|
glUniformMatrix4fv(uniMVP, 1, GL_FALSE, &mvp[0][0]);
|
|
|
|
|
|
|
|
|
|
// Render command lists
|
|
|
|
|
vtx_consumed = 0; // offset in vertex buffer. each command consume ImDrawCmd::vtx_count of those
|
|
|
|
|
for (int n = 0; n < cmd_lists_count; n++)
|
|
|
|
|
{
|
|
|
|
|
const ImDrawList* cmd_list = cmd_lists[n];
|
|
|
|
|
if (cmd_list->commands.empty() || cmd_list->vtx_buffer.empty())
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
// Setup stack of clipping rectangles
|
|
|
|
|
bool clip_rect_dirty = true;
|
|
|
|
|
int clip_rect_buf_consumed = 0;
|
|
|
|
|
ImVector<ImVec4> clip_rect_stack;
|
|
|
|
|
clip_rect_stack.push_back(ImVec4(-9999,-9999,+9999,+9999));
|
|
|
|
|
|
|
|
|
|
// Render command list
|
|
|
|
|
const ImDrawCmd* pcmd = &cmd_list->commands.front();
|
|
|
|
|
const ImDrawCmd* pcmd_end = &cmd_list->commands.back();
|
|
|
|
|
int clip_rect_buf_consumed = 0; // offset in cmd_list->clip_rect_buffer. each PushClipRect command consume 1 of those.
|
|
|
|
|
while (pcmd <= pcmd_end)
|
|
|
|
|
{
|
|
|
|
|
const ImDrawCmd& cmd = *pcmd++;
|
|
|
|
@ -132,7 +130,6 @@ static void ImImpl_SetClipboardTextFn(const char* text, const char* text_end)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Shader sources
|
|
|
|
|
// FIXME-OPT: clip at vertex level
|
|
|
|
|
const GLchar* vertexSource =
|
|
|
|
|
"#version 150 core\n"
|
|
|
|
|
"uniform mat4 MVP;"
|
|
|
|
@ -164,6 +161,7 @@ const GLchar* fragmentSource =
|
|
|
|
|
" o_col.w *= (step(ClipRect.x,pixel_pos.x) * step(ClipRect.y,pixel_pos.y) * step(pixel_pos.x,ClipRect.z) * step(pixel_pos.y,ClipRect.w));" // Clipping: branch-less, set alpha 0.0f
|
|
|
|
|
"}";
|
|
|
|
|
|
|
|
|
|
// GLFW callbacks to get events
|
|
|
|
|
static void glfw_error_callback(int error, const char* description)
|
|
|
|
|
{
|
|
|
|
|
fputs(description, stderr);
|
|
|
|
@ -251,7 +249,6 @@ void InitImGui()
|
|
|
|
|
io.SetClipboardTextFn = ImImpl_SetClipboardTextFn;
|
|
|
|
|
io.GetClipboardTextFn = ImImpl_GetClipboardTextFn;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Setup graphics backend
|
|
|
|
|
GLint status = GL_TRUE;
|
|
|
|
|
GLenum err = GL_NO_ERROR;
|
|
|
|
@ -394,7 +391,7 @@ int main(int argc, char** argv)
|
|
|
|
|
ImGui::End();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 3) Render
|
|
|
|
|
// 3) Rendering
|
|
|
|
|
glClearColor(0.8f, 0.6f, 0.6f, 1.0f);
|
|
|
|
|
glClear(GL_COLOR_BUFFER_BIT);
|
|
|
|
|
ImGui::Render();
|
|
|
|
@ -403,6 +400,5 @@ int main(int argc, char** argv)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Shutdown();
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|