mirror of
https://github.com/kovidgoyal/kitty.git
synced 2026-07-03 11:12:30 +08:00
1935 lines
91 KiB
C
1935 lines
91 KiB
C
/*
|
|
* shaders.c
|
|
* Copyright (C) 2017 Kovid Goyal <kovid at kovidgoyal.net>
|
|
*
|
|
* Distributed under terms of the GPL3 license.
|
|
*/
|
|
|
|
#include "fonts.h"
|
|
#include "gl.h"
|
|
#include "cleanup.h"
|
|
#include "colors.h"
|
|
#include <stddef.h>
|
|
#include <string.h>
|
|
#include "text-cache.h"
|
|
#include "window_logo.h"
|
|
#include "srgb_gamma.h"
|
|
#include "uniforms_generated.h"
|
|
#include "state.h"
|
|
|
|
enum {
|
|
CELL_PROGRAM, CELL_FG_PROGRAM, CELL_BG_PROGRAM, CELL_PROGRAM_SENTINEL,
|
|
BORDERS_PROGRAM,
|
|
GRAPHICS_PROGRAM, GRAPHICS_PREMULT_PROGRAM, GRAPHICS_ALPHA_MASK_PROGRAM,
|
|
BGIMAGE_PROGRAM,
|
|
TINT_PROGRAM,
|
|
TRAIL_PROGRAM,
|
|
BLIT_PROGRAM,
|
|
SCREENSHOT_PROGRAM,
|
|
ROUNDED_RECT_PROGRAM,
|
|
NUM_PROGRAMS
|
|
};
|
|
enum { SPRITE_MAP_UNIT, GRAPHICS_UNIT, SPRITE_DECORATIONS_MAP_UNIT };
|
|
|
|
typedef struct UIRenderData {
|
|
unsigned screen_width, screen_height, cell_width, cell_height, screen_left, screen_top, full_framebuffer_width, full_framebuffer_height;
|
|
Window *window; Screen *screen; OSWindow *os_window;
|
|
GraphicsRenderData grd;
|
|
WindowLogoRenderData *window_logo;
|
|
float bg_alpha, inactive_text_alpha;
|
|
bool has_background_image;
|
|
color_type background_color; // RGB only
|
|
} UIRenderData;
|
|
|
|
static inline float
|
|
row_offset_for_screen(const Screen *screen) {
|
|
if (!pixel_scroll_enabled(screen) || !screen->cell_size.height) return 0.f;
|
|
return -1.f + (float)(screen->pixel_scroll_offset_y / (double)screen->cell_size.height);
|
|
}
|
|
|
|
static inline float
|
|
scroll_offset_lines_for_screen(const Screen *screen) {
|
|
if (!pixel_scroll_enabled(screen) || !screen->cell_size.height) return 0.f;
|
|
return (float)(screen->pixel_scroll_offset_y / (double)screen->cell_size.height);
|
|
}
|
|
|
|
// Sprites {{{
|
|
typedef struct {
|
|
int xnum, ynum, x, y, z, last_num_of_layers, last_ynum;
|
|
GLuint texture_id;
|
|
GLint max_texture_size, max_array_texture_layers;
|
|
struct decorations_map {
|
|
GLuint texture_id;
|
|
unsigned width, height;
|
|
size_t count;
|
|
} decorations_map;
|
|
} SpriteMap;
|
|
|
|
static const SpriteMap NEW_SPRITE_MAP = { .xnum = 1, .ynum = 1, .last_num_of_layers = 1, .last_ynum = -1 };
|
|
static GLint max_texture_size = 0, max_array_texture_layers = 0;
|
|
|
|
static GLfloat
|
|
srgb_color(uint8_t color) {
|
|
return srgb_lut[color];
|
|
}
|
|
|
|
static void
|
|
color_vec3(GLint location, color_type color) {
|
|
glUniform3f(location, srgb_lut[(color >> 16) & 0xFF], srgb_lut[(color >> 8) & 0xFF], srgb_lut[color & 0xFF]);
|
|
}
|
|
|
|
static void
|
|
color_vec4_premult(GLint location, color_type color, GLfloat alpha) {
|
|
glUniform4f(location, srgb_lut[(color >> 16) & 0xFF]*alpha, srgb_lut[(color >> 8) & 0xFF]*alpha, srgb_lut[color & 0xFF]*alpha, alpha);
|
|
}
|
|
|
|
static void
|
|
color_vec4(GLint location, color_type color, GLfloat alpha) {
|
|
glUniform4f(location, srgb_lut[(color >> 16) & 0xFF], srgb_lut[(color >> 8) & 0xFF], srgb_lut[color & 0xFF], alpha);
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
clear_current_framebuffer(void) { glClearColor(0, 0, 0, 0); glClear(GL_COLOR_BUFFER_BIT); }
|
|
|
|
SPRITE_MAP_HANDLE
|
|
alloc_sprite_map(void) {
|
|
if (!max_texture_size) {
|
|
glGetIntegerv(GL_MAX_TEXTURE_SIZE, &(max_texture_size));
|
|
glGetIntegerv(GL_MAX_ARRAY_TEXTURE_LAYERS, &(max_array_texture_layers));
|
|
#ifdef __APPLE__
|
|
// Since on Apple we could have multiple GPUs, with different capabilities,
|
|
// upper bound the values according to the data from https://developer.apple.com/graphicsimaging/opengl/capabilities/
|
|
max_texture_size = MIN(8192, max_texture_size);
|
|
max_array_texture_layers = MIN(512, max_array_texture_layers);
|
|
#endif
|
|
sprite_tracker_set_limits(max_texture_size, max_array_texture_layers);
|
|
}
|
|
SpriteMap *ans = calloc(1, sizeof(SpriteMap));
|
|
if (!ans) fatal("Out of memory allocating a sprite map");
|
|
*ans = NEW_SPRITE_MAP;
|
|
ans->max_texture_size = max_texture_size;
|
|
ans->max_array_texture_layers = max_array_texture_layers;
|
|
return (SPRITE_MAP_HANDLE)ans;
|
|
}
|
|
|
|
void
|
|
free_sprite_data(FONTS_DATA_HANDLE fg) {
|
|
SpriteMap *sprite_map = (SpriteMap*)fg->sprite_map;
|
|
if (sprite_map) {
|
|
if (sprite_map->texture_id) free_texture(&sprite_map->texture_id);
|
|
if (sprite_map->decorations_map.texture_id) free_texture(&sprite_map->decorations_map.texture_id);
|
|
free(sprite_map);
|
|
fg->sprite_map = NULL;
|
|
}
|
|
}
|
|
|
|
|
|
static void
|
|
copy_32bit_texture(GLuint old_texture, GLuint new_texture, GLenum texture_type) {
|
|
// requires new texture to be at least as big as old texture. Assumes textures are 32bits per pixel
|
|
GLint width, height, layers;
|
|
glBindTexture(texture_type, old_texture);
|
|
glGetTexLevelParameteriv(texture_type, 0, GL_TEXTURE_WIDTH, &width);
|
|
glGetTexLevelParameteriv(texture_type, 0, GL_TEXTURE_HEIGHT, &height);
|
|
glGetTexLevelParameteriv(texture_type, 0, GL_TEXTURE_DEPTH, &layers);
|
|
if (GLAD_GL_ARB_copy_image) { glCopyImageSubData(old_texture, texture_type, 0, 0, 0, 0, new_texture, texture_type, 0, 0, 0, 0, width, height, layers); return; }
|
|
|
|
static bool copy_image_warned = false;
|
|
// ARB_copy_image not available, do a slow roundtrip copy
|
|
if (!copy_image_warned) {
|
|
copy_image_warned = true;
|
|
log_error("WARNING: Your system's OpenGL implementation does not have glCopyImageSubData, falling back to a slower implementation");
|
|
}
|
|
|
|
GLint internal_format;
|
|
glGetTexLevelParameteriv(texture_type, 0, GL_TEXTURE_INTERNAL_FORMAT, &internal_format);
|
|
GLenum format, type;
|
|
switch(internal_format) {
|
|
case GL_R8UI: case GL_R8I: case GL_R16UI: case GL_R16I: case GL_R32UI: case GL_R32I: case GL_RG8UI: case GL_RG8I:
|
|
case GL_RG16UI: case GL_RG16I: case GL_RG32UI: case GL_RG32I: case GL_RGB8UI: case GL_RGB8I: case GL_RGB16UI:
|
|
case GL_RGB16I: case GL_RGB32UI: case GL_RGB32I: case GL_RGBA8UI: case GL_RGBA8I: case GL_RGBA16UI: case GL_RGBA16I:
|
|
case GL_RGBA32UI: case GL_RGBA32I:
|
|
format = GL_RED_INTEGER;
|
|
type = GL_UNSIGNED_INT;
|
|
break;
|
|
default:
|
|
format = GL_RGBA;
|
|
type = GL_UNSIGNED_INT_8_8_8_8;
|
|
break;
|
|
}
|
|
glPixelStorei(GL_UNPACK_ALIGNMENT, 4);
|
|
RAII_ALLOC(uint8_t, pixels, malloc((size_t)width * height * layers * 4u));
|
|
if (!pixels) fatal("Out of memory");
|
|
glGetTexImage(texture_type, 0, format, type, pixels);
|
|
glBindTexture(texture_type, new_texture);
|
|
glPixelStorei(GL_PACK_ALIGNMENT, 4);
|
|
if (texture_type == GL_TEXTURE_2D_ARRAY) glTexSubImage3D(GL_TEXTURE_2D_ARRAY, 0, 0, 0, 0, width, height, layers, format, type, pixels);
|
|
else glTexSubImage2D(GL_TEXTURE_2D, 0, 0, 0, width, height, format, type, pixels);
|
|
}
|
|
|
|
static GLuint
|
|
setup_new_sprites_texture(GLenum texture_type) {
|
|
GLuint tex;
|
|
glGenTextures(1, &tex);
|
|
glBindTexture(texture_type, tex);
|
|
// We use GL_NEAREST otherwise glyphs that touch the edge of the cell
|
|
// often show a border between cells
|
|
glTexParameteri(texture_type, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
|
|
glTexParameteri(texture_type, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
|
|
glTexParameteri(texture_type, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE);
|
|
glTexParameteri(texture_type, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE);
|
|
return tex;
|
|
}
|
|
|
|
static void
|
|
realloc_sprite_decorations_texture_if_needed(FONTS_DATA_HANDLE fg) {
|
|
#define dm (sm->decorations_map)
|
|
SpriteMap *sm = (SpriteMap*)fg->sprite_map;
|
|
size_t current_capacity = (size_t)dm.width * dm.height;
|
|
if (dm.count < current_capacity && dm.texture_id) return;
|
|
GLint new_capacity = dm.count + 256;
|
|
GLint width = new_capacity, height = 1;
|
|
if (new_capacity > sm->max_texture_size) {
|
|
width = sm->max_texture_size;
|
|
height = 1 + new_capacity / width;
|
|
}
|
|
if (height > sm->max_texture_size) fatal("Max texture size too small for sprite decorations map, maybe switch to using a GL_TEXTURE_2D_ARRAY");
|
|
const GLenum texture_type = GL_TEXTURE_2D;
|
|
GLuint tex = setup_new_sprites_texture(texture_type);
|
|
glTexImage2D(texture_type, 0, GL_R32UI, width, height, 0, GL_RED_INTEGER, GL_UNSIGNED_INT, NULL);
|
|
if (dm.texture_id) { // copy data from old texture
|
|
copy_32bit_texture(dm.texture_id, tex, texture_type);
|
|
free_texture(&dm.texture_id);
|
|
}
|
|
glBindTexture(texture_type, 0);
|
|
dm.texture_id = tex; dm.width = width; dm.height = height;
|
|
#undef dm
|
|
}
|
|
|
|
static void
|
|
realloc_sprite_texture(FONTS_DATA_HANDLE fg) {
|
|
unsigned int xnum, ynum, z, znum, width, height;
|
|
sprite_tracker_current_layout(fg, &xnum, &ynum, &z);
|
|
znum = z + 1;
|
|
SpriteMap *sprite_map = (SpriteMap*)fg->sprite_map;
|
|
width = xnum * fg->fcm.cell_width; height = ynum * (fg->fcm.cell_height + 1);
|
|
const GLenum texture_type = GL_TEXTURE_2D_ARRAY;
|
|
GLuint tex = setup_new_sprites_texture(texture_type);
|
|
glTexStorage3D(texture_type, 1, GL_SRGB8_ALPHA8, width, height, znum);
|
|
if (sprite_map->texture_id) { // copy old texture data into new texture
|
|
copy_32bit_texture(sprite_map->texture_id, tex, texture_type);
|
|
free_texture(&sprite_map->texture_id);
|
|
}
|
|
glBindTexture(texture_type, 0);
|
|
sprite_map->last_num_of_layers = znum;
|
|
sprite_map->last_ynum = ynum;
|
|
sprite_map->texture_id = tex;
|
|
}
|
|
|
|
static void
|
|
ensure_sprite_map(FONTS_DATA_HANDLE fg) {
|
|
SpriteMap *sprite_map = (SpriteMap*)fg->sprite_map;
|
|
if (!sprite_map->texture_id) realloc_sprite_texture(fg);
|
|
if (!sprite_map->decorations_map.texture_id) realloc_sprite_decorations_texture_if_needed(fg);
|
|
// We have to rebind since we don't know if the texture was ever bound
|
|
// in the context of the current OSWindow
|
|
glActiveTexture(GL_TEXTURE0 + SPRITE_DECORATIONS_MAP_UNIT);
|
|
glBindTexture(GL_TEXTURE_2D, sprite_map->decorations_map.texture_id);
|
|
glActiveTexture(GL_TEXTURE0 + SPRITE_MAP_UNIT);
|
|
glBindTexture(GL_TEXTURE_2D_ARRAY, sprite_map->texture_id);
|
|
}
|
|
|
|
void
|
|
send_sprite_to_gpu(FONTS_DATA_HANDLE fg, sprite_index idx, pixel *buf, sprite_index decoration_idx) {
|
|
SpriteMap *sprite_map = (SpriteMap*)fg->sprite_map;
|
|
unsigned int xnum, ynum, znum, x, y, z;
|
|
#define dm (sprite_map->decorations_map)
|
|
if (idx >= dm.count) dm.count = idx + 1;
|
|
realloc_sprite_decorations_texture_if_needed(fg);
|
|
div_t d = div(idx, dm.width);
|
|
x = d.rem; y = d.quot;
|
|
glActiveTexture(GL_TEXTURE0 + SPRITE_DECORATIONS_MAP_UNIT);
|
|
glBindTexture(GL_TEXTURE_2D, dm.texture_id);
|
|
glPixelStorei(GL_UNPACK_ALIGNMENT, 4);
|
|
glTexSubImage2D(GL_TEXTURE_2D, 0, x, y, 1, 1, GL_RED_INTEGER, GL_UNSIGNED_INT, &decoration_idx);
|
|
#undef dm
|
|
sprite_tracker_current_layout(fg, &xnum, &ynum, &znum);
|
|
if ((int)znum >= sprite_map->last_num_of_layers || (znum == 0 && (int)ynum > sprite_map->last_ynum)) {
|
|
realloc_sprite_texture(fg);
|
|
sprite_tracker_current_layout(fg, &xnum, &ynum, &znum);
|
|
}
|
|
glActiveTexture(GL_TEXTURE0 + SPRITE_MAP_UNIT);
|
|
glBindTexture(GL_TEXTURE_2D_ARRAY, sprite_map->texture_id);
|
|
glPixelStorei(GL_UNPACK_ALIGNMENT, 4);
|
|
sprite_index_to_pos(idx, xnum, ynum, &x, &y, &z);
|
|
x *= fg->fcm.cell_width; y *= (fg->fcm.cell_height + 1);
|
|
glTexSubImage3D(GL_TEXTURE_2D_ARRAY, 0, x, y, z, fg->fcm.cell_width, fg->fcm.cell_height + 1, 1, GL_RGBA, GL_UNSIGNED_INT_8_8_8_8, buf);
|
|
}
|
|
|
|
void
|
|
send_image_to_gpu(GLuint *tex_id, const void* data, GLsizei width, GLsizei height, bool is_opaque, bool is_4byte_aligned, bool linear, RepeatStrategy repeat) {
|
|
if (!(*tex_id)) { glGenTextures(1, tex_id); }
|
|
glBindTexture(GL_TEXTURE_2D, *tex_id);
|
|
glPixelStorei(GL_UNPACK_ALIGNMENT, is_4byte_aligned ? 4 : 1);
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, linear ? GL_LINEAR : GL_NEAREST);
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, linear ? GL_LINEAR : GL_NEAREST);
|
|
RepeatStrategy r;
|
|
switch (repeat) {
|
|
case REPEAT_MIRROR:
|
|
r = GL_MIRRORED_REPEAT; break;
|
|
case REPEAT_CLAMP: {
|
|
static const GLfloat border_color[4] = {0};
|
|
glTexParameterfv(GL_TEXTURE_2D, GL_TEXTURE_BORDER_COLOR, border_color);
|
|
r = GL_CLAMP_TO_BORDER;
|
|
break;
|
|
}
|
|
default:
|
|
r = GL_REPEAT;
|
|
}
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, r);
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, r);
|
|
glTexImage2D(GL_TEXTURE_2D, 0, GL_SRGB_ALPHA, width, height, 0, is_opaque ? GL_RGB : GL_RGBA, GL_UNSIGNED_BYTE, data);
|
|
}
|
|
|
|
// }}}
|
|
|
|
// Rounded rect {{{
|
|
typedef struct {
|
|
Rounded_rectUniforms uniforms;
|
|
} RoundedRectProgramLayout;
|
|
static RoundedRectProgramLayout rounded_rect_program_layout;
|
|
|
|
static double
|
|
thickness_as_float(const OSWindow *os_window, unsigned level) {
|
|
level = MIN(level, arraysz(OPT(box_drawing_scale)));
|
|
double pts = OPT(box_drawing_scale)[level];
|
|
double dpi = (os_window->fonts_data->logical_dpi_x + os_window->fonts_data->logical_dpi_y) / 2.0;
|
|
return pts * dpi / 72.0;
|
|
}
|
|
|
|
|
|
static void
|
|
draw_rounded_rect(
|
|
const OSWindow *os_window, Viewport rect, unsigned framebuffer_height,
|
|
unsigned thickness_level, unsigned corner_radius_px,
|
|
color_type srgb_color, color_type srgb_background, float bg_alpha
|
|
) {
|
|
float thickness = (float)thickness_as_float(os_window, thickness_level);
|
|
bind_program(ROUNDED_RECT_PROGRAM);
|
|
color_vec4(rounded_rect_program_layout.uniforms.color, srgb_color, 1.f);
|
|
color_vec4(rounded_rect_program_layout.uniforms.background_color, srgb_background, bg_alpha);
|
|
// y co-ord has to be changed to co-ord system with origin at bottom left
|
|
float y = (float)framebuffer_height - (float)(rect.top + rect.height);
|
|
glUniform4f(rounded_rect_program_layout.uniforms.rect, rect.left, y, rect.width, rect.height);
|
|
glUniform2f(rounded_rect_program_layout.uniforms.params, thickness, corner_radius_px);
|
|
save_viewport_using_top_left_origin(rect.left, rect.top, rect.width, rect.height, framebuffer_height);
|
|
draw_quad(true, 0);
|
|
restore_viewport();
|
|
}
|
|
// }}}
|
|
|
|
// Cell {{{
|
|
|
|
enum { CELL_RENDER_DATA_BINDING_POINT = 0, COLOR_TABLE_BINDING_POINT = 1 };
|
|
|
|
typedef struct {
|
|
UniformBlock render_data;
|
|
UniformBlock color_table;
|
|
GLint color_table_stride;
|
|
CellUniforms uniforms;
|
|
} CellProgramLayout;
|
|
static CellProgramLayout cell_program_layouts[NUM_PROGRAMS];
|
|
|
|
typedef struct {
|
|
GraphicsUniforms uniforms;
|
|
} GraphicsProgramLayout;
|
|
static GraphicsProgramLayout graphics_program_layouts[NUM_PROGRAMS];
|
|
|
|
typedef struct {
|
|
BgimageUniforms uniforms;
|
|
} BGImageProgramLayout;
|
|
static BGImageProgramLayout bgimage_program_layout;
|
|
|
|
typedef struct {
|
|
TintUniforms uniforms;
|
|
} TintProgramLayout;
|
|
static TintProgramLayout tint_program_layout;
|
|
|
|
typedef struct {
|
|
TrailUniforms uniforms;
|
|
} TrailProgramLayout;
|
|
static TrailProgramLayout trail_program_layout;
|
|
|
|
typedef struct {
|
|
BlitUniforms uniforms;
|
|
} BlitProgramLayout;
|
|
static BlitProgramLayout blit_program_layout;
|
|
|
|
typedef struct {
|
|
ScreenshotUniforms uniforms;
|
|
} ScreenshotProgramLayout;
|
|
static ScreenshotProgramLayout screenshot_program_layout;
|
|
|
|
static void
|
|
init_cell_program(void) {
|
|
for (int i = CELL_PROGRAM; i < CELL_PROGRAM_SENTINEL; i++) {
|
|
cell_program_layouts[i].render_data.index = block_index(i, "CellRenderData");
|
|
cell_program_layouts[i].render_data.size = block_size(i, cell_program_layouts[i].render_data.index);
|
|
cell_program_layouts[i].color_table.index = block_index(i, "ColorTable");
|
|
cell_program_layouts[i].color_table.size = block_size(i, cell_program_layouts[i].color_table.index);
|
|
cell_program_layouts[i].color_table_stride = get_uniform_information(i, "color_table[0]", GL_UNIFORM_ARRAY_STRIDE);
|
|
glUniformBlockBinding(program_id(i), cell_program_layouts[i].render_data.index, CELL_RENDER_DATA_BINDING_POINT);
|
|
glUniformBlockBinding(program_id(i), cell_program_layouts[i].color_table.index, COLOR_TABLE_BINDING_POINT);
|
|
get_uniform_locations_cell(i, &cell_program_layouts[i].uniforms);
|
|
bind_program(i);
|
|
glUniform1fv(cell_program_layouts[i].uniforms.gamma_lut, arraysz(srgb_lut), srgb_lut);
|
|
}
|
|
|
|
// Sanity check to ensure the attribute location binding worked
|
|
#define C(p, name, expected) { int aloc = attrib_location(p, #name); if (aloc != expected && aloc != -1) fatal("The attribute location for %s is %d != %d in program: %d", #name, aloc, expected, p); }
|
|
for (int p = CELL_PROGRAM; p < CELL_PROGRAM_SENTINEL; p++) {
|
|
C(p, colors, 0); C(p, sprite_idx, 1); C(p, is_selected, 2); C(p, decorations_sprite_map, 3);
|
|
}
|
|
#undef C
|
|
for (int i = GRAPHICS_PROGRAM; i <= GRAPHICS_ALPHA_MASK_PROGRAM; i++) {
|
|
get_uniform_locations_graphics(i, &graphics_program_layouts[i].uniforms);
|
|
}
|
|
get_uniform_locations_bgimage(BGIMAGE_PROGRAM, &bgimage_program_layout.uniforms);
|
|
get_uniform_locations_tint(TINT_PROGRAM, &tint_program_layout.uniforms);
|
|
get_uniform_locations_trail(TRAIL_PROGRAM, &trail_program_layout.uniforms);
|
|
get_uniform_locations_blit(BLIT_PROGRAM, &blit_program_layout.uniforms);
|
|
get_uniform_locations_screenshot(SCREENSHOT_PROGRAM, &screenshot_program_layout.uniforms);
|
|
get_uniform_locations_rounded_rect(ROUNDED_RECT_PROGRAM, &rounded_rect_program_layout.uniforms);
|
|
}
|
|
|
|
#define CELL_BUFFERS enum { cell_data_buffer, selection_buffer, uniform_buffer, color_table_buffer };
|
|
|
|
ssize_t
|
|
create_cell_vao(void) {
|
|
ssize_t vao_idx = create_vao();
|
|
#define A(name, size, dtype, offset, stride) \
|
|
add_attribute_to_vao(CELL_PROGRAM, vao_idx, #name, \
|
|
/*size=*/size, /*dtype=*/dtype, /*stride=*/stride, /*offset=*/offset, /*divisor=*/1);
|
|
#define A1(name, size, dtype, offset) A(name, size, dtype, (void*)(offsetof(GPUCell, offset)), sizeof(GPUCell))
|
|
|
|
add_buffer_to_vao(vao_idx, GL_ARRAY_BUFFER);
|
|
A1(sprite_idx, 2, GL_UNSIGNED_INT, sprite_idx);
|
|
A1(colors, 3, GL_UNSIGNED_INT, fg);
|
|
|
|
add_buffer_to_vao(vao_idx, GL_ARRAY_BUFFER);
|
|
A(is_selected, 1, GL_UNSIGNED_BYTE, NULL, 0);
|
|
|
|
size_t bufnum = add_buffer_to_vao(vao_idx, GL_UNIFORM_BUFFER);
|
|
alloc_vao_buffer(vao_idx, cell_program_layouts[CELL_PROGRAM].render_data.size, bufnum, GL_STREAM_DRAW);
|
|
|
|
size_t ctbufnum = add_buffer_to_vao(vao_idx, GL_UNIFORM_BUFFER);
|
|
alloc_vao_buffer(vao_idx, cell_program_layouts[CELL_PROGRAM].color_table.size, ctbufnum, GL_STATIC_DRAW);
|
|
|
|
return vao_idx;
|
|
#undef A
|
|
#undef A1
|
|
}
|
|
|
|
ssize_t
|
|
create_graphics_vao(void) {
|
|
ssize_t vao_idx = create_vao();
|
|
add_buffer_to_vao(vao_idx, GL_ARRAY_BUFFER);
|
|
add_attribute_to_vao(GRAPHICS_PROGRAM, vao_idx, "src", 4, GL_FLOAT, 0, NULL, 0);
|
|
return vao_idx;
|
|
}
|
|
|
|
#define IS_SPECIAL_COLOR(name) (screen->color_profile->overridden.name.type == COLOR_IS_SPECIAL || (screen->color_profile->overridden.name.type == COLOR_NOT_SET && screen->color_profile->configured.name.type == COLOR_IS_SPECIAL))
|
|
|
|
static void
|
|
pick_cursor_color(color_type cell_fg, color_type cell_bg, color_type *cursor_fg, color_type *cursor_bg, color_type default_fg, color_type default_bg) {
|
|
ARGB32 fg, bg, dfg, dbg;
|
|
fg.rgb = cell_fg; bg.rgb = cell_bg;
|
|
*cursor_fg = cell_bg; *cursor_bg = cell_fg;
|
|
double cell_contrast = rgb_contrast(fg, bg);
|
|
if (cell_contrast < 2.5) {
|
|
dfg.rgb = default_fg; dbg.rgb = default_bg;
|
|
if (rgb_contrast(dfg, dbg) > cell_contrast) {
|
|
*cursor_fg = default_bg; *cursor_bg = default_fg;
|
|
}
|
|
}
|
|
}
|
|
|
|
static bool
|
|
has_bgimage(OSWindow *w) {
|
|
return background_image_for_os_window(w) != NULL;
|
|
}
|
|
|
|
static color_type
|
|
cell_update_uniform_block(ssize_t vao_idx, Screen *screen, int uniform_buffer, int color_table_buf, const CursorRenderInfo *cursor, OSWindow *os_window, float inactive_text_alpha, float bg_alpha) {
|
|
struct GPUCellRenderData {
|
|
GLfloat use_cell_bg_for_selection_fg, use_cell_fg_for_selection_color, use_cell_for_selection_bg;
|
|
|
|
GLuint default_fg, highlight_fg, highlight_bg, main_cursor_fg, main_cursor_bg, url_color, url_style, inverted, extra_cursor_fg, extra_cursor_bg;
|
|
|
|
GLuint columns, lines, sprites_xnum, sprites_ynum, cursor_shape, cell_width, cell_height;
|
|
GLuint cursor_x1, cursor_x2, cursor_y1, cursor_y2;
|
|
GLfloat cursor_opacity, inactive_text_alpha, dim_opacity, blink_opacity;
|
|
|
|
GLuint bg_colors0, bg_colors1, bg_colors2, bg_colors3, bg_colors4, bg_colors5, bg_colors6, bg_colors7;
|
|
GLfloat bg_opacities0, bg_opacities1, bg_opacities2, bg_opacities3, bg_opacities4, bg_opacities5, bg_opacities6, bg_opacities7;
|
|
};
|
|
// Send the uniform data
|
|
ColorProfile *cp = screen->paused_rendering.expires_at ? &screen->paused_rendering.color_profile : screen->color_profile;
|
|
if (cp->dirty || screen->reload_all_gpu_data) {
|
|
GLuint *ct_buf = (GLuint*)map_vao_buffer_for_write_only(vao_idx, color_table_buf, 0, cell_program_layouts[CELL_PROGRAM].color_table.size);
|
|
copy_color_table_to_buffer(cp, ct_buf, 0, cell_program_layouts[CELL_PROGRAM].color_table_stride / sizeof(GLuint));
|
|
unmap_vao_buffer(vao_idx, color_table_buf);
|
|
}
|
|
struct GPUCellRenderData *rd = (struct GPUCellRenderData*)map_vao_buffer_for_write_only(vao_idx, uniform_buffer, 0, cell_program_layouts[CELL_PROGRAM].render_data.size);
|
|
#define COLOR(name) colorprofile_to_color(cp, cp->overridden.name, cp->configured.name).rgb
|
|
rd->default_fg = COLOR(default_fg);
|
|
rd->highlight_fg = COLOR(highlight_fg); rd->highlight_bg = COLOR(highlight_bg);
|
|
rd->extra_cursor_fg = screen->extra_cursors.color.text.val;
|
|
rd->extra_cursor_bg = screen->extra_cursors.color.cursor.val;
|
|
rd->bg_colors0 = COLOR(default_bg);
|
|
rd->bg_opacities0 = bg_alpha;
|
|
#define SETBG(which) { \
|
|
colorprofile_to_transparent_color(cp, which - 1, &rd->bg_colors##which, &rd->bg_opacities##which); }
|
|
SETBG(1); SETBG(2); SETBG(3); SETBG(4); SETBG(5); SETBG(6); SETBG(7);
|
|
#undef SETBG
|
|
// selection
|
|
if (IS_SPECIAL_COLOR(highlight_fg)) {
|
|
if (IS_SPECIAL_COLOR(highlight_bg)) {
|
|
rd->use_cell_bg_for_selection_fg = 1.f; rd->use_cell_fg_for_selection_color = 0.f;
|
|
} else {
|
|
rd->use_cell_bg_for_selection_fg = 0.f; rd->use_cell_fg_for_selection_color = 1.f;
|
|
}
|
|
} else {
|
|
rd->use_cell_bg_for_selection_fg = 0.f; rd->use_cell_fg_for_selection_color = 0.f;
|
|
}
|
|
rd->use_cell_for_selection_bg = IS_SPECIAL_COLOR(highlight_bg) ? 1. : 0.;
|
|
// Cursor position
|
|
Line *line_for_cursor = NULL;
|
|
rd->cursor_opacity = MAX(0, MIN(cursor->cursor_opacity, 1));
|
|
rd->blink_opacity = MAX(0, MIN(cursor->text_blink_opacity, 1));
|
|
if (rd->cursor_opacity != 0 && cursor->is_visible) {
|
|
rd->cursor_x1 = cursor->x, rd->cursor_y1 = cursor->y;
|
|
rd->cursor_x2 = cursor->x, rd->cursor_y2 = cursor->y;
|
|
if (pixel_scroll_enabled(screen)) {
|
|
rd->cursor_y1 += 1;
|
|
rd->cursor_y2 += 1;
|
|
}
|
|
CursorShape cs = (cursor->is_focused || OPT(cursor_shape_unfocused) == NO_CURSOR_SHAPE) ? cursor->shape : OPT(cursor_shape_unfocused);
|
|
rd->cursor_shape = cs;
|
|
color_type cell_fg = rd->default_fg, cell_bg = rd->bg_colors0;
|
|
index_type cell_color_x = cursor->x;
|
|
bool reversed = false;
|
|
if (cursor->x < screen->columns && cursor->y < screen->lines) {
|
|
if (screen->paused_rendering.expires_at) {
|
|
linebuf_init_line(screen->paused_rendering.linebuf, cursor->y); line_for_cursor = screen->paused_rendering.linebuf->line;
|
|
} else {
|
|
linebuf_init_line(screen->linebuf, cursor->y); line_for_cursor = screen->linebuf->line;
|
|
}
|
|
}
|
|
if (line_for_cursor) {
|
|
colors_for_cell(line_for_cursor, cp, &cell_color_x, &cell_fg, &cell_bg, &reversed);
|
|
const CPUCell *cursor_cell;
|
|
const bool large_cursor = ((cursor_cell = &line_for_cursor->cpu_cells[cursor->x])->is_multicell) && cursor_cell->x == 0 && cursor_cell->y == 0;
|
|
if (large_cursor) {
|
|
switch(cs) {
|
|
case CURSOR_BEAM:
|
|
rd->cursor_y2 += cursor_cell->scale - 1; break;
|
|
case CURSOR_UNDERLINE:
|
|
rd->cursor_y1 += cursor_cell->scale - 1;
|
|
rd->cursor_y2 = rd->cursor_y1;
|
|
rd->cursor_x2 += mcd_x_limit(cursor_cell) - 1;
|
|
break;
|
|
case CURSOR_BLOCK:
|
|
rd->cursor_y2 += cursor_cell->scale - 1;
|
|
rd->cursor_x2 += mcd_x_limit(cursor_cell) - 1;
|
|
break;
|
|
case CURSOR_HOLLOW: case NUM_OF_CURSOR_SHAPES: case NO_CURSOR_SHAPE: break;
|
|
};
|
|
}
|
|
}
|
|
// If you change the following algorithm remember to change it in the cell shader for extra cursors too
|
|
if (IS_SPECIAL_COLOR(cursor_color)) {
|
|
if (line_for_cursor) pick_cursor_color(cell_fg, cell_bg, &rd->main_cursor_fg, &rd->main_cursor_bg, rd->default_fg, rd->bg_colors0);
|
|
else { rd->main_cursor_fg = rd->bg_colors0; rd->main_cursor_bg = rd->default_fg; }
|
|
if (cell_bg == cell_fg) {
|
|
rd->main_cursor_fg = rd->bg_colors0; rd->main_cursor_bg = rd->default_fg;
|
|
} else { rd->main_cursor_fg = cell_bg; rd->main_cursor_bg = cell_fg; }
|
|
} else {
|
|
rd->main_cursor_bg = COLOR(cursor_color);
|
|
if (IS_SPECIAL_COLOR(cursor_text_color)) rd->main_cursor_fg = cell_bg;
|
|
else rd->main_cursor_fg = COLOR(cursor_text_color);
|
|
}
|
|
// store last rendered cursor color for trail rendering
|
|
screen->last_rendered.cursor_bg = rd->main_cursor_bg;
|
|
} else {
|
|
rd->cursor_shape = 0;
|
|
rd->cursor_x1 = screen->columns + 1; rd->cursor_x2 = screen->columns;
|
|
rd->cursor_y1 = screen->lines + 1; rd->cursor_y2 = screen->lines;
|
|
}
|
|
|
|
rd->columns = screen->columns; rd->lines = screen->lines;
|
|
|
|
unsigned int x, y, z;
|
|
sprite_tracker_current_layout(os_window->fonts_data, &x, &y, &z);
|
|
rd->sprites_xnum = x; rd->sprites_ynum = y;
|
|
rd->inverted = screen_invert_colors(screen) ? 1 : 0;
|
|
rd->cell_width = os_window->fonts_data->fcm.cell_width;
|
|
rd->cell_height = os_window->fonts_data->fcm.cell_height;
|
|
rd->inactive_text_alpha = inactive_text_alpha;
|
|
rd->dim_opacity = OPT(dim_opacity);
|
|
|
|
#undef COLOR
|
|
rd->url_color = OPT(url_color); rd->url_style = OPT(url_style);
|
|
color_type default_bg = rd->bg_colors0;
|
|
unmap_vao_buffer(vao_idx, uniform_buffer); rd = NULL;
|
|
return default_bg;
|
|
}
|
|
|
|
static bool
|
|
cell_prepare_to_render(ssize_t vao_idx, Screen *screen, FONTS_DATA_HANDLE fonts_data) {
|
|
size_t sz;
|
|
CELL_BUFFERS;
|
|
void *address;
|
|
bool changed = false;
|
|
|
|
ensure_sprite_map(fonts_data);
|
|
const Cursor *cursor = screen->paused_rendering.expires_at ? &screen->paused_rendering.cursor : screen->cursor;
|
|
|
|
bool cursor_pos_changed = cursor->x != screen->last_rendered.cursor.x \
|
|
|| cursor->y != screen->last_rendered.cursor.y;
|
|
bool disable_ligatures = screen->disable_ligatures == DISABLE_LIGATURES_CURSOR;
|
|
bool screen_resized = screen->last_rendered.columns != screen->columns || screen->last_rendered.lines != screen->lines;
|
|
|
|
#define update_cell_data { \
|
|
const unsigned int render_lines = render_lines_for_screen(screen); \
|
|
sz = sizeof(GPUCell) * render_lines * screen->columns; \
|
|
address = alloc_and_map_vao_buffer(vao_idx, sz, cell_data_buffer, true); \
|
|
screen_update_cell_data(screen, address, fonts_data, disable_ligatures && cursor_pos_changed); \
|
|
unmap_vao_buffer(vao_idx, cell_data_buffer); address = NULL; \
|
|
changed = true; \
|
|
}
|
|
|
|
if (screen->paused_rendering.expires_at) {
|
|
if (!screen->paused_rendering.cell_data_updated) update_cell_data;
|
|
} else if (screen->reload_all_gpu_data || screen->scroll_changed || screen->is_dirty || screen_resized || (disable_ligatures && cursor_pos_changed)) update_cell_data;
|
|
|
|
#define update_selection_data { \
|
|
const unsigned int render_lines = render_lines_for_screen(screen); \
|
|
sz = (size_t)render_lines * screen->columns; \
|
|
address = alloc_and_map_vao_buffer(vao_idx, sz, selection_buffer, true); \
|
|
screen_apply_selection(screen, address, sz); \
|
|
unmap_vao_buffer(vao_idx, selection_buffer); address = NULL; \
|
|
changed = true; \
|
|
}
|
|
|
|
#define update_graphics_data(grman) \
|
|
grman_update_layers(grman, screen->scrolled_by, scroll_offset_lines_for_screen(screen), -1.f, 1.f, 2.f/screen->columns, 2.f/screen->lines, screen->columns, screen->lines, screen->cell_size)
|
|
|
|
if (screen->paused_rendering.expires_at) {
|
|
if (!screen->paused_rendering.cell_data_updated) {
|
|
update_selection_data; update_graphics_data(screen->paused_rendering.grman);
|
|
}
|
|
screen->paused_rendering.cell_data_updated = true;
|
|
screen->last_rendered.scrolled_by = screen->paused_rendering.scrolled_by;
|
|
} else {
|
|
if (screen->reload_all_gpu_data || screen_resized || screen_is_selection_dirty(screen)) update_selection_data;
|
|
if (update_graphics_data(screen->grman)) changed = true;
|
|
screen->last_rendered.scrolled_by = screen->scrolled_by;
|
|
}
|
|
#undef update_selection_data
|
|
#undef update_cell_data
|
|
screen->last_rendered.columns = screen->columns;
|
|
screen->last_rendered.lines = screen->lines;
|
|
screen->last_rendered.cursor = screen->cursor_render_info;
|
|
|
|
return changed;
|
|
}
|
|
|
|
static void
|
|
draw_graphics(int program, ImageRenderData *data, GLuint start, GLuint count, float extra_alpha) {
|
|
bind_program(program);
|
|
if (program != GRAPHICS_ALPHA_MASK_PROGRAM) glUniform1f(graphics_program_layouts[program].uniforms.extra_alpha, extra_alpha);
|
|
glActiveTexture(GL_TEXTURE0 + GRAPHICS_UNIT);
|
|
GraphicsUniforms *u = &graphics_program_layouts[program].uniforms;
|
|
for (GLuint i=0; i < count;) {
|
|
ImageRenderData *group = data + start + i;
|
|
glBindTexture(GL_TEXTURE_2D, group->texture_id);
|
|
if (group->group_count == 0) { i++; continue; }
|
|
for (GLuint k=0; k < group->group_count; k++, i++) {
|
|
ImageRenderData *rd = data + start + i;
|
|
glUniform4f(u->src_rect, rd->src_rect.left, rd->src_rect.top, rd->src_rect.right, rd->src_rect.bottom);
|
|
glUniform4f(u->dest_rect, rd->dest_rect.left, rd->dest_rect.top, rd->dest_rect.right, rd->dest_rect.bottom);
|
|
draw_quad(true, 0);
|
|
}
|
|
}
|
|
}
|
|
|
|
static ImageRenderData*
|
|
load_alpha_mask_texture(size_t width, size_t height, uint8_t *canvas) {
|
|
static ImageRenderData data = {.group_count=1};
|
|
if (!data.texture_id) { glGenTextures(1, &data.texture_id); }
|
|
glBindTexture(GL_TEXTURE_2D, data.texture_id);
|
|
glPixelStorei(GL_UNPACK_ALIGNMENT, 1);
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE);
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE);
|
|
glTexImage2D(GL_TEXTURE_2D, 0, GL_RED, width, height, 0, GL_RED, GL_UNSIGNED_BYTE, canvas);
|
|
return &data;
|
|
}
|
|
|
|
static void
|
|
setup_texture_as_render_target(unsigned width, unsigned height, GLuint *texture_id, GLuint *framebuffer_id) {
|
|
glGenTextures(1, texture_id); glGenFramebuffers(1, framebuffer_id);
|
|
glBindTexture(GL_TEXTURE_2D, *texture_id);
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE);
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE);
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
|
|
// We use GL_RGBA16 to avoid incorrect colors due to quantization loss when
|
|
// blending, see https://github.com/kovidgoyal/kitty/issues/8953
|
|
static struct { bool ok; int fmt; } status = { false, GL_RGBA16};
|
|
glTexImage2D(GL_TEXTURE_2D, 0, status.fmt, width, height, 0, GL_RGBA, GL_UNSIGNED_BYTE, NULL);
|
|
bind_framebuffer_for_output(*framebuffer_id);
|
|
glFramebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D, *texture_id, 0);
|
|
if (!status.ok) {
|
|
if (check_framebuffer_status() == NULL) {
|
|
status.ok = true;
|
|
} else {
|
|
if (status.fmt == GL_RGBA16) {
|
|
// Driver does not support 16 bit FBO so let it choose the
|
|
// format. It will probably end up choosing 8 bit but
|
|
// inaccurate colors are better than completely broken rendering.
|
|
// See https://github.com/kovidgoyal/kitty/issues/9068
|
|
status.fmt = GL_RGBA;
|
|
free_framebuffer(framebuffer_id);
|
|
free_texture(texture_id);
|
|
setup_texture_as_render_target(width, height, texture_id, framebuffer_id);
|
|
log_error("WARNING: Your GPU driver does not support 16bit textures as framebuffer targets, some colors may be slightly inaccurate.");
|
|
} else {
|
|
fatal("Your GPU driver does not support indirect rendering to a GL_RGBA texture via a framebuffer");
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
static void
|
|
set_cell_uniforms(bool force) {
|
|
static bool constants_set = false;
|
|
if (!constants_set || force) {
|
|
float text_contrast = 1.0f + OPT(text_contrast) * 0.01f;
|
|
float text_gamma_adjustment = OPT(text_gamma_adjustment) < 0.01f ? 1.0f : 1.0f / OPT(text_gamma_adjustment);
|
|
for (int i = GRAPHICS_PROGRAM; i <= GRAPHICS_ALPHA_MASK_PROGRAM; i++) {
|
|
bind_program(i); glUniform1i(graphics_program_layouts[i].uniforms.image, GRAPHICS_UNIT);
|
|
}
|
|
for (int i = CELL_PROGRAM; i < CELL_PROGRAM_SENTINEL; i++) {
|
|
bind_program(i); const CellUniforms *cu = &cell_program_layouts[i].uniforms;
|
|
glUniform1i(cu->sprites, SPRITE_MAP_UNIT);
|
|
glUniform1i(cu->sprite_decorations_map, SPRITE_DECORATIONS_MAP_UNIT);
|
|
glUniform1f(cu->text_contrast, text_contrast);
|
|
glUniform1f(cu->text_gamma_adjustment, text_gamma_adjustment);
|
|
}
|
|
bind_program(BLIT_PROGRAM); glUniform1i(blit_program_layout.uniforms.image, GRAPHICS_UNIT);
|
|
bind_program(SCREENSHOT_PROGRAM); glUniform1i(screenshot_program_layout.uniforms.image, GRAPHICS_UNIT);
|
|
constants_set = true;
|
|
}
|
|
}
|
|
|
|
|
|
// UI Layer {{{
|
|
static Animation *default_visual_bell_animation = NULL;
|
|
|
|
static float
|
|
get_visual_bell_intensity(Screen *screen) {
|
|
if (screen->start_visual_bell_at > 0) {
|
|
if (!default_visual_bell_animation) {
|
|
default_visual_bell_animation = alloc_animation();
|
|
if (!default_visual_bell_animation) fatal("Out of memory");
|
|
add_cubic_bezier_animation(default_visual_bell_animation, 0, 1, EASE_IN_OUT);
|
|
add_cubic_bezier_animation(default_visual_bell_animation, 1, 0, EASE_IN_OUT);
|
|
}
|
|
const monotonic_t progress = monotonic() - screen->start_visual_bell_at;
|
|
const monotonic_t duration = OPT(visual_bell_duration) / 2;
|
|
if (progress <= duration) {
|
|
Animation *a = animation_is_valid(OPT(animation.visual_bell)) ? OPT(animation.visual_bell) : default_visual_bell_animation;
|
|
return (float)apply_easing_curve(a, progress / (double)duration, duration);
|
|
}
|
|
screen->start_visual_bell_at = 0;
|
|
}
|
|
return 0.0f;
|
|
}
|
|
|
|
static void
|
|
draw_visual_bell_flash(GLfloat intensity, const color_type flash) {
|
|
bind_program(TINT_PROGRAM);
|
|
GLfloat attenuation = 0.4f;
|
|
#define C(shift) srgb_color((flash >> shift) & 0xFF)
|
|
const GLfloat r = C(16), g = C(8), b = C(0);
|
|
const GLfloat max_channel = r > g ? (r > b ? r : b) : (g > b ? g : b);
|
|
#undef C
|
|
#define C(x) (x * intensity * attenuation)
|
|
if (max_channel > 0.45) attenuation = 0.6f; // light color
|
|
glUniform4f(tint_program_layout.uniforms.tint_color, C(r), C(g), C(b), C(1));
|
|
#undef C
|
|
glUniform4f(tint_program_layout.uniforms.edges, -1, 1, 1, -1);
|
|
draw_quad(true, 0);
|
|
}
|
|
|
|
static color_type
|
|
get_flash_color(const Screen *screen) {
|
|
#define COLOR(name, fallback) colorprofile_to_color_with_fallback(screen->color_profile, screen->color_profile->overridden.name, screen->color_profile->configured.name, screen->color_profile->overridden.fallback, screen->color_profile->configured.fallback)
|
|
return !IS_SPECIAL_COLOR(highlight_bg) ? COLOR(visual_bell_color, highlight_bg) : COLOR(visual_bell_color, default_fg);
|
|
#undef COLOR
|
|
}
|
|
|
|
static void
|
|
draw_visual_bell(const UIRenderData *ui) {
|
|
if (!ui->screen->start_visual_bell_at) return;
|
|
Screen *screen = ui->screen;
|
|
float intensity = get_visual_bell_intensity(screen);
|
|
if (intensity <= 0) return;
|
|
draw_visual_bell_flash(intensity, get_flash_color(screen));
|
|
}
|
|
|
|
static void
|
|
draw_drag_preview_overlay(const UIRenderData *ui) {
|
|
const Screen *screen = ui->screen;
|
|
if (!screen->start_drag_overlay_at || !screen->drag_overlay_quadrant) return;
|
|
const monotonic_t elapsed = monotonic() - screen->start_drag_overlay_at;
|
|
const monotonic_t fade_ms = ms_to_monotonic_t(150ll);
|
|
float intensity = elapsed >= fade_ms ? 1.0f : (float)elapsed / (float)fade_ms;
|
|
GLfloat left = -1.f, top = 1.f, right = 1.f, bottom = -1.f;
|
|
switch (screen->drag_overlay_quadrant) {
|
|
case 1: right = 0.f; break; // left half
|
|
case 2: left = 0.f; break; // right half
|
|
case 3: bottom = 0.f; break; // top half
|
|
case 4: top = 0.f; break; // bottom half
|
|
case 5: break; // full window + title bar highlight (title bar hover)
|
|
case 6: break; // full window, no title bar highlight (body hover)
|
|
default: return;
|
|
}
|
|
bind_program(TINT_PROGRAM);
|
|
float a = intensity * 0.25f;
|
|
color_type hint = get_flash_color(screen);
|
|
#define C(shift) (srgb_color((hint >> shift) & 0xFF) * a)
|
|
glUniform4f(tint_program_layout.uniforms.tint_color, C(16), C(8), C(0), a);
|
|
#undef C
|
|
glUniform4f(tint_program_layout.uniforms.edges, left, top, right, bottom);
|
|
draw_quad(true, 0);
|
|
}
|
|
|
|
static bool
|
|
has_scrollbar(Window *w, Screen *screen) {
|
|
if (screen->linebuf != screen->main_linebuf || !screen->historybuf->count) return false;
|
|
switch (OPT(scrollbar)) {
|
|
case SCROLLBAR_NEVER: return false;
|
|
case SCROLLBAR_ALWAYS: return true;
|
|
case SCROLLBAR_ON_SCROLLED: return screen->scrolled_by > 0;
|
|
case SCROLLBAR_ON_HOVERED: return w->scrollbar.is_hovering;
|
|
case SCROLLBAR_ON_SCROLL_AND_HOVER: return screen->scrolled_by > 0 && w->scrollbar.is_hovering;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
static unsigned
|
|
render_a_bar(const UIRenderData *ui, WindowBarData *bar, PyObject *title, bool along_bottom) {
|
|
unsigned border_width = (unsigned)ceil(thickness_as_float(ui->os_window, 1));
|
|
unsigned bar_height = ui->cell_height + 2;
|
|
unsigned bar_width = ui->screen_width - 2 * border_width;
|
|
if (!bar->buf || bar->width != bar_width || bar->height != bar_height) {
|
|
free(bar->buf);
|
|
bar->buf = malloc((size_t)4 * bar_width * bar_height);
|
|
if (!bar->buf) return 0;
|
|
bar->height = bar_height;
|
|
bar->width = bar_width;
|
|
bar->needs_render = true;
|
|
}
|
|
#define RGBCOL(which, fallback) ( 0xff000000 | colorprofile_to_color_with_fallback(ui->screen->color_profile, ui->screen->color_profile->overridden.which, ui->screen->color_profile->configured.which, ui->screen->color_profile->overridden.fallback, ui->screen->color_profile->configured.fallback))
|
|
color_type fg = RGBCOL(default_fg, default_fg), bg = RGBCOL(default_bg, default_bg);
|
|
#undef RGBCOL
|
|
if (bar->last_drawn_title_object_id != title || bar->needs_render) {
|
|
static char titlebuf[2048] = {0};
|
|
if (!title) return 0;
|
|
snprintf(titlebuf, arraysz(titlebuf), " %s", PyUnicode_AsUTF8(title));
|
|
if (!draw_window_title(ui->os_window->fonts_data->font_sz_in_pts, ui->os_window->fonts_data->logical_dpi_y, titlebuf, fg, bg, bar->buf, bar_width, bar_height, NULL)) return 0;
|
|
Py_CLEAR(bar->last_drawn_title_object_id);
|
|
bar->last_drawn_title_object_id = Py_NewRef(title);
|
|
}
|
|
static ImageRenderData data = {.group_count=1};
|
|
gpu_data_for_image(&data, -1, 1, 1, -1);
|
|
glGenTextures(1, &data.texture_id);
|
|
glBindTexture(GL_TEXTURE_2D, data.texture_id);
|
|
glPixelStorei(GL_UNPACK_ALIGNMENT, 1);
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE);
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE);
|
|
glTexImage2D(GL_TEXTURE_2D, 0, GL_SRGB_ALPHA, bar_width, bar_height, 0, GL_RGBA, GL_UNSIGNED_BYTE, bar->buf);
|
|
bind_program(GRAPHICS_PROGRAM);
|
|
Viewport border_rect = {
|
|
.height=bar_height + 2 * border_width, .left=ui->screen_left, .width=ui->screen_width, .top=ui->screen_top};
|
|
if (along_bottom) border_rect.top += ui->screen_height - border_rect.height;
|
|
const unsigned sh = ui->full_framebuffer_height;
|
|
// first blank the area to be drawn to background
|
|
enable_scissor_using_top_left_origin(border_rect, sh);
|
|
blank_canvas(ui->bg_alpha, bg, false);
|
|
disable_scissor();
|
|
// then draw the rendered text
|
|
save_viewport_using_top_left_origin(
|
|
border_rect.left + border_width, border_rect.top + border_width, bar_width, bar_height, sh);
|
|
draw_graphics(GRAPHICS_PROGRAM, &data, 0, 1, 1.f);
|
|
restore_viewport();
|
|
free_texture(&data.texture_id);
|
|
// finally draw border with transparent bg
|
|
draw_rounded_rect(ui->os_window, border_rect, sh, 1, ui->cell_width, fg, bg, 0.f);
|
|
return border_rect.height;
|
|
}
|
|
|
|
static bool
|
|
show_hyperlink_targets_with_modifiers(int mods) {
|
|
switch (OPT(show_hyperlink_targets)) {
|
|
case SHOW_HYPERLINK_TARGETS_ALWAYS: return true;
|
|
case SHOW_HYPERLINK_TARGETS_CTRL: return (mods & GLFW_MOD_CONTROL) != 0;
|
|
case SHOW_HYPERLINK_TARGETS_SHIFT: return (mods & GLFW_MOD_SHIFT) != 0;
|
|
case SHOW_HYPERLINK_TARGETS_SUPER: return (mods & GLFW_MOD_SUPER) != 0;
|
|
case SHOW_HYPERLINK_TARGETS_ALT: return (mods & GLFW_MOD_ALT) != 0;
|
|
default: return false;
|
|
}
|
|
}
|
|
|
|
static bool
|
|
has_hyperlink_target(OSWindow *os_window, Window *w, Screen *screen) {
|
|
return show_hyperlink_targets_with_modifiers(global_state.mods_at_last_key_or_button_event) &&
|
|
screen->current_hyperlink_under_mouse.id &&
|
|
w && !is_mouse_hidden(os_window) &&
|
|
global_state.mouse_hover_in_window == w->id;
|
|
}
|
|
|
|
static void
|
|
draw_hyperlink_target(const UIRenderData *ui) {
|
|
if (!has_hyperlink_target(ui->os_window, ui->window, ui->screen)) return;
|
|
Screen *screen = ui->screen;
|
|
Window *window = ui->window;
|
|
const bool along_bottom = screen->current_hyperlink_under_mouse.y < 3;
|
|
WindowBarData *bd = &window->url_target_bar_data;
|
|
hyperlink_id_type hid = screen->current_hyperlink_under_mouse.id;
|
|
if (bd->hyperlink_id_for_title_object != hid) {
|
|
bd->hyperlink_id_for_title_object = hid;
|
|
Py_CLEAR(bd->last_drawn_title_object_id);
|
|
const char *url = get_hyperlink_for_id(screen->hyperlink_pool, bd->hyperlink_id_for_title_object, true);
|
|
if (url == NULL) url = "";
|
|
bd->last_drawn_title_object_id = PyObject_CallMethod(global_state.boss, "sanitize_url_for_display_to_user", "s", url);
|
|
if (bd->last_drawn_title_object_id == NULL) { PyErr_Print(); return; }
|
|
bd->needs_render = true;
|
|
}
|
|
if (bd->last_drawn_title_object_id == NULL) return;
|
|
PyObject *ref = Py_NewRef(bd->last_drawn_title_object_id); // render_a_bar clears bd->last_drawn_title_object_id
|
|
render_a_bar(ui, &window->title_bar_data, bd->last_drawn_title_object_id, along_bottom);
|
|
Py_DECREF(ref);
|
|
}
|
|
|
|
static bool
|
|
has_window_number(Window *w, Screen *screen) {
|
|
return w != NULL && screen->display_window_char != 0;
|
|
}
|
|
|
|
static void
|
|
draw_window_number(const UIRenderData *ui) {
|
|
if (!has_window_number(ui->window, ui->screen)) return;
|
|
unsigned title_bar_height = 0, requested_height = ui->screen_height;
|
|
if (ui->window->title && PyUnicode_Check(ui->window->title) && (requested_height > (ui->cell_height + 1) * 2)) {
|
|
title_bar_height = render_a_bar(ui, &ui->window->title_bar_data, ui->window->title, false);
|
|
}
|
|
unsigned height_for_letter = ui->screen_height - title_bar_height - ui->cell_height;
|
|
unsigned width_for_letter = ui->screen_width - ui->cell_width;
|
|
requested_height = MIN(12 * ui->cell_height, MIN(height_for_letter, width_for_letter));
|
|
if (requested_height < 4) return;
|
|
#define lr ui->screen->last_rendered_window_char
|
|
if (!lr.canvas || lr.ch != ui->screen->display_window_char || lr.requested_height != requested_height) {
|
|
free(lr.canvas); lr.canvas = NULL;
|
|
lr.requested_height = requested_height; lr.height_px = requested_height; lr.ch = 0;
|
|
lr.canvas = draw_single_ascii_char(ui->screen->display_window_char, &lr.width_px, &lr.height_px);
|
|
if (lr.height_px < 4 || lr.width_px < 4 || !lr.canvas) return;
|
|
lr.ch = ui->screen->display_window_char;
|
|
}
|
|
unsigned letter_x = 0, letter_y = title_bar_height;
|
|
if (lr.width_px < ui->screen_width) letter_x = (ui->screen_width - lr.width_px) / 2;
|
|
if (lr.height_px + title_bar_height < ui->screen_height) letter_y += (ui->screen_height - lr.height_px - title_bar_height) / 2;
|
|
bind_program(GRAPHICS_ALPHA_MASK_PROGRAM);
|
|
ImageRenderData *ird = load_alpha_mask_texture(lr.width_px, lr.height_px, lr.canvas);
|
|
gpu_data_for_image(ird, -1, 1, 1, -1);
|
|
glUniform1i(graphics_program_layouts[GRAPHICS_ALPHA_MASK_PROGRAM].uniforms.image, GRAPHICS_UNIT);
|
|
color_type digit_color = colorprofile_to_color_with_fallback(ui->screen->color_profile, ui->screen->color_profile->overridden.highlight_bg, ui->screen->color_profile->configured.highlight_bg, ui->screen->color_profile->overridden.default_fg, ui->screen->color_profile->configured.default_fg);
|
|
color_vec3(graphics_program_layouts[GRAPHICS_ALPHA_MASK_PROGRAM].uniforms.amask_fg, digit_color);
|
|
glUniform4f(graphics_program_layouts[GRAPHICS_ALPHA_MASK_PROGRAM].uniforms.amask_bg_premult, 0.f, 0.f, 0.f, 0.f);
|
|
save_viewport_using_top_left_origin(
|
|
ui->screen_left + letter_x, ui->screen_top + letter_y, lr.width_px, lr.height_px, ui->full_framebuffer_height);
|
|
draw_graphics(GRAPHICS_ALPHA_MASK_PROGRAM, ird, 0, 1, 1.f);
|
|
restore_viewport();
|
|
#undef lr
|
|
}
|
|
|
|
// Helper function to extract and apply opacity to color components
|
|
static void
|
|
set_color_uniform_with_opacity(color_type color, float opacity) {
|
|
float r = srgb_color((color >> 16) & 0xFF) * opacity;
|
|
float g = srgb_color((color >> 8) & 0xFF) * opacity;
|
|
float b = srgb_color(color & 0xFF) * opacity;
|
|
glUniform4f(tint_program_layout.uniforms.tint_color, r, g, b, opacity);
|
|
}
|
|
|
|
static color_type
|
|
scrollbar_color(Screen *screen, unsigned val) {
|
|
switch (val & 0xff) {
|
|
#define C(which) colorprofile_to_color(screen->color_profile, screen->color_profile->overridden.which, screen->color_profile->configured.which).rgb
|
|
case 0: return C(default_fg);
|
|
case 1: return C(highlight_bg);
|
|
#undef C
|
|
default: return val >> 8;
|
|
}
|
|
}
|
|
|
|
static void
|
|
draw_scrollbar(const UIRenderData *ui) {
|
|
Screen *screen = ui->screen;
|
|
Window *window = ui->window;
|
|
if (!window || !screen || !has_scrollbar(window, screen)) return;
|
|
|
|
color_type bar_color = scrollbar_color(screen, OPT(scrollbar_handle_color)), track_color = scrollbar_color(screen, OPT(scrollbar_track_color));
|
|
double cell_frac = (double)screen->pixel_scroll_offset_y / screen->cell_size.height;
|
|
if (!OPT(pixel_scroll)) cell_frac = 0;
|
|
float bar_frac = (float)(screen->scrolled_by + cell_frac) / MAX(1u, (float)screen->historybuf->count);
|
|
float opacity = OPT(scrollbar_handle_opacity);
|
|
float track_opacity = window->scrollbar.is_hovering ? OPT(scrollbar_track_hover_opacity) : OPT(scrollbar_track_opacity);
|
|
GLsizei scrollbar_width_px = (GLsizei)(OPT(scrollbar_width) * ui->cell_width);
|
|
if (window->scrollbar.is_hovering) scrollbar_width_px = (GLsizei)(OPT(scrollbar_hover_width) * ui->cell_width);
|
|
GLsizei scrollbar_gap_px = (GLsizei)(OPT(scrollbar_gap) * ui->cell_width);
|
|
unsigned scrollbar_radius = (unsigned)(OPT(scrollbar_radius) * ui->cell_width);
|
|
|
|
// Calculate window boundaries including padding
|
|
GLsizei window_right_edge = ui->screen_left + ui->screen_width + window->render_data.geometry.spaces.right;
|
|
GLsizei window_top_edge = ui->screen_top - window->render_data.geometry.spaces.top;
|
|
GLsizei window_height = ui->screen_height + window->render_data.geometry.spaces.top + window->render_data.geometry.spaces.bottom;
|
|
|
|
// Position scrollbar on right side with gap
|
|
GLsizei scrollbar_left = window_right_edge - scrollbar_width_px - scrollbar_gap_px;
|
|
GLsizei scrollbar_top = window_top_edge + scrollbar_gap_px;
|
|
GLsizei scrollbar_height = window_height - 2 * scrollbar_gap_px;
|
|
|
|
// Calculate thumb size and position
|
|
float visible_fraction = (float)screen->lines / (float)(screen->lines + screen->historybuf->count);
|
|
float min_thumb_height_fraction = (OPT(scrollbar_min_handle_height) * ui->cell_height) / (float)window_height;
|
|
float thumb_height_fraction = MAX(min_thumb_height_fraction, visible_fraction);
|
|
|
|
// Convert to OpenGL coordinates (range -1.0 to 1.0, total span = 2.0)
|
|
const float GL_COORD_SPAN = 2.0f;
|
|
float thumb_height_gl = thumb_height_fraction * GL_COORD_SPAN;
|
|
float available_space = GL_COORD_SPAN - thumb_height_gl;
|
|
float thumb_bottom_gl = -1.0f + available_space * bar_frac;
|
|
float thumb_top_gl = thumb_bottom_gl + thumb_height_gl;
|
|
|
|
// Store thumb position for mouse interaction (normalized window coordinates)
|
|
float scrollbar_top_in_window = (float)(window_top_edge + scrollbar_gap_px) / (float)ui->full_framebuffer_height;
|
|
float scrollbar_height_in_window = (float)(window_height - 2 * scrollbar_gap_px) / (float)ui->full_framebuffer_height;
|
|
float thumb_top_fraction = (1.0f - thumb_top_gl) / 2.0f;
|
|
float thumb_bottom_fraction = (1.0f - thumb_bottom_gl) / 2.0f;
|
|
window->scrollbar.thumb_top = scrollbar_top_in_window + thumb_top_fraction * scrollbar_height_in_window;
|
|
window->scrollbar.thumb_bottom = scrollbar_top_in_window + thumb_bottom_fraction * scrollbar_height_in_window;
|
|
|
|
// Set viewport for scrollbar area
|
|
save_viewport_using_top_left_origin(
|
|
scrollbar_left, scrollbar_top, scrollbar_width_px, scrollbar_height,
|
|
ui->full_framebuffer_height
|
|
);
|
|
|
|
// Draw scrollbar track (background)
|
|
if (track_opacity > 0) {
|
|
bind_program(TINT_PROGRAM);
|
|
set_color_uniform_with_opacity(track_color, track_opacity);
|
|
glUniform4f(tint_program_layout.uniforms.edges, -1.f, 1.f, 1.f, -1.f);
|
|
draw_quad(true, 0);
|
|
}
|
|
|
|
// Draw scrollbar thumb (handle)
|
|
if (scrollbar_radius > 0) {
|
|
// Rounded thumb - use separate viewport and rounded rect program
|
|
GLsizei thumb_height_px = (GLsizei)(thumb_height_fraction * scrollbar_height);
|
|
GLsizei thumb_top_px = scrollbar_top + (GLsizei)(thumb_top_fraction * scrollbar_height);
|
|
|
|
restore_viewport();
|
|
|
|
bind_program(ROUNDED_RECT_PROGRAM);
|
|
color_vec4(rounded_rect_program_layout.uniforms.color, bar_color, opacity);
|
|
color_vec4(rounded_rect_program_layout.uniforms.background_color, 0, 0.0f);
|
|
|
|
float y = (float)ui->full_framebuffer_height - (float)(thumb_top_px + thumb_height_px);
|
|
glUniform4f(rounded_rect_program_layout.uniforms.rect,
|
|
(float)scrollbar_left, y,
|
|
(float)scrollbar_width_px, (float)thumb_height_px);
|
|
|
|
float thickness = (float)MAX(scrollbar_width_px, thumb_height_px);
|
|
glUniform2f(rounded_rect_program_layout.uniforms.params, thickness, (float)scrollbar_radius);
|
|
|
|
save_viewport_using_top_left_origin(scrollbar_left, thumb_top_px,
|
|
scrollbar_width_px, thumb_height_px,
|
|
ui->full_framebuffer_height);
|
|
draw_quad(true, 0);
|
|
restore_viewport();
|
|
} else {
|
|
set_color_uniform_with_opacity(bar_color, opacity);
|
|
glUniform4f(tint_program_layout.uniforms.edges, -1.f, thumb_top_gl, 1.f, thumb_bottom_gl);
|
|
draw_quad(true, 0);
|
|
restore_viewport();
|
|
}
|
|
}
|
|
|
|
static bool
|
|
has_progress_bar(Screen *screen) {
|
|
if (OPT(progress_bar) == PROGRESS_BAR_HIDDEN) return false;
|
|
return screen && screen->progress_state != PROGRESS_STATE_UNSET;
|
|
}
|
|
|
|
static void
|
|
draw_progress_handle(const UIRenderData *ui, color_type bar_color, float opacity, unsigned bar_radius,
|
|
GLsizei track_left, GLsizei track_top, GLsizei track_width, GLsizei track_height,
|
|
float handle_start, float handle_size, bool is_horizontal) {
|
|
// handle_start and handle_size are fractions of the track length (0..1)
|
|
// For horizontal: handle moves left-to-right; For vertical: handle moves top-to-bottom
|
|
// Use lroundf to avoid sub-pixel jitter at the leading edge
|
|
GLsizei handle_left, handle_top;
|
|
GLsizei handle_w, handle_h;
|
|
if (is_horizontal) {
|
|
GLsizei handle_start_px = (GLsizei)lroundf(handle_start * track_width);
|
|
GLsizei handle_end_px = (GLsizei)lroundf((handle_start + handle_size) * track_width);
|
|
GLsizei handle_w_px = handle_end_px - handle_start_px;
|
|
if (handle_w_px < 1) handle_w_px = 1;
|
|
handle_left = track_left + handle_start_px;
|
|
handle_top = track_top;
|
|
handle_w = handle_w_px;
|
|
handle_h = track_height;
|
|
} else {
|
|
GLsizei handle_start_px = (GLsizei)lroundf(handle_start * track_height);
|
|
GLsizei handle_end_px = (GLsizei)lroundf((handle_start + handle_size) * track_height);
|
|
GLsizei handle_h_px = handle_end_px - handle_start_px;
|
|
if (handle_h_px < 1) handle_h_px = 1;
|
|
handle_left = track_left;
|
|
handle_top = track_top + handle_start_px;
|
|
handle_w = track_width;
|
|
handle_h = handle_h_px;
|
|
}
|
|
|
|
if (bar_radius > 0) {
|
|
bind_program(ROUNDED_RECT_PROGRAM);
|
|
color_vec4(rounded_rect_program_layout.uniforms.color, bar_color, opacity);
|
|
color_vec4(rounded_rect_program_layout.uniforms.background_color, 0, 0.0f);
|
|
|
|
float y = (float)ui->full_framebuffer_height - (float)(handle_top + handle_h);
|
|
glUniform4f(rounded_rect_program_layout.uniforms.rect,
|
|
(float)handle_left, y,
|
|
(float)handle_w, (float)handle_h);
|
|
|
|
float thickness = (float)(is_horizontal ? handle_h : handle_w);
|
|
glUniform2f(rounded_rect_program_layout.uniforms.params, thickness, (float)bar_radius);
|
|
|
|
save_viewport_using_top_left_origin(handle_left, handle_top, handle_w, handle_h, ui->full_framebuffer_height);
|
|
draw_quad(true, 0);
|
|
restore_viewport();
|
|
} else {
|
|
// Use GL coordinates within the track viewport, snapped to pixel boundaries
|
|
GLsizei track_len = is_horizontal ? track_width : track_height;
|
|
float start_snapped = (float)lroundf(handle_start * track_len) / (float)track_len;
|
|
float end_snapped = (float)lroundf((handle_start + handle_size) * track_len) / (float)track_len;
|
|
float start_gl = -1.0f + 2.0f * start_snapped;
|
|
float end_gl = -1.0f + 2.0f * end_snapped;
|
|
bind_program(TINT_PROGRAM);
|
|
set_color_uniform_with_opacity(bar_color, opacity);
|
|
if (is_horizontal) {
|
|
// edges: left, top, right, bottom
|
|
save_viewport_using_top_left_origin(track_left, track_top, track_width, track_height, ui->full_framebuffer_height);
|
|
glUniform4f(tint_program_layout.uniforms.edges, start_gl, 1.f, end_gl, -1.f);
|
|
} else {
|
|
save_viewport_using_top_left_origin(track_left, track_top, track_width, track_height, ui->full_framebuffer_height);
|
|
// For vertical: start_gl is top in GL coords (inverted y), so bottom_gl = -end_gl, top_gl = -start_gl
|
|
glUniform4f(tint_program_layout.uniforms.edges, -1.f, -start_gl, 1.f, -end_gl);
|
|
}
|
|
draw_quad(true, 0);
|
|
restore_viewport();
|
|
}
|
|
}
|
|
|
|
static void
|
|
draw_progress_bar(const UIRenderData *ui) {
|
|
Screen *screen = ui->screen;
|
|
Window *window = ui->window;
|
|
if (!window || !has_progress_bar(screen)) return;
|
|
|
|
const ProgressBarPosition pos = OPT(progress_bar);
|
|
const bool is_horizontal = (pos == PROGRESS_BAR_TOP || pos == PROGRESS_BAR_BOTTOM);
|
|
color_type bar_color = scrollbar_color(screen, OPT(scrollbar_handle_color));
|
|
color_type track_color = scrollbar_color(screen, OPT(scrollbar_track_color));
|
|
float opacity = OPT(scrollbar_handle_opacity);
|
|
float track_opacity = OPT(scrollbar_track_hover_opacity);
|
|
GLsizei bar_thickness_px = (GLsizei)(OPT(scrollbar_width) * ui->cell_width);
|
|
GLsizei gap_px = (GLsizei)(OPT(scrollbar_gap) * ui->cell_width);
|
|
unsigned bar_radius = (unsigned)(OPT(scrollbar_radius) * ui->cell_width);
|
|
if (bar_thickness_px < 1) return;
|
|
|
|
// Calculate window boundaries including padding
|
|
GLsizei window_left_edge = ui->screen_left - (GLsizei)window->render_data.geometry.spaces.left;
|
|
GLsizei window_top_edge = ui->screen_top - (GLsizei)window->render_data.geometry.spaces.top;
|
|
GLsizei window_width = ui->screen_width + (GLsizei)(window->render_data.geometry.spaces.left + window->render_data.geometry.spaces.right);
|
|
GLsizei window_height = ui->screen_height + (GLsizei)(window->render_data.geometry.spaces.top + window->render_data.geometry.spaces.bottom);
|
|
|
|
// Position the track depending on the chosen edge
|
|
GLsizei track_left, track_top, track_width, track_height;
|
|
switch (pos) {
|
|
case PROGRESS_BAR_BOTTOM:
|
|
track_left = window_left_edge + gap_px;
|
|
track_width = window_width - 2 * gap_px;
|
|
track_height = bar_thickness_px;
|
|
track_top = window_top_edge + window_height - bar_thickness_px - gap_px;
|
|
break;
|
|
case PROGRESS_BAR_TOP:
|
|
track_left = window_left_edge + gap_px;
|
|
track_width = window_width - 2 * gap_px;
|
|
track_height = bar_thickness_px;
|
|
track_top = window_top_edge + gap_px;
|
|
break;
|
|
case PROGRESS_BAR_LEFT:
|
|
track_left = window_left_edge + gap_px;
|
|
track_width = bar_thickness_px;
|
|
track_top = window_top_edge + gap_px;
|
|
track_height = window_height - 2 * gap_px;
|
|
break;
|
|
case PROGRESS_BAR_RIGHT:
|
|
track_left = window_left_edge + window_width - bar_thickness_px - gap_px;
|
|
track_width = bar_thickness_px;
|
|
track_top = window_top_edge + gap_px;
|
|
track_height = window_height - 2 * gap_px;
|
|
break;
|
|
default:
|
|
return;
|
|
}
|
|
if (track_width <= 0 || track_height <= 0) return;
|
|
|
|
// Calculate fill fraction and indeterminate animation
|
|
float fill_fraction = 0.0f;
|
|
bool is_indeterminate = false;
|
|
switch (screen->progress_state) {
|
|
case PROGRESS_STATE_SET:
|
|
case PROGRESS_STATE_PAUSED:
|
|
fill_fraction = screen->progress_percent / 100.0f;
|
|
break;
|
|
case PROGRESS_STATE_ERROR:
|
|
fill_fraction = 1.0f;
|
|
break;
|
|
case PROGRESS_STATE_INDETERMINATE:
|
|
is_indeterminate = true;
|
|
break;
|
|
default:
|
|
return;
|
|
}
|
|
|
|
// Draw track (background)
|
|
save_viewport_using_top_left_origin(track_left, track_top, track_width, track_height, ui->full_framebuffer_height);
|
|
if (track_opacity > 0) {
|
|
bind_program(TINT_PROGRAM);
|
|
set_color_uniform_with_opacity(track_color, track_opacity);
|
|
glUniform4f(tint_program_layout.uniforms.edges, -1.f, 1.f, 1.f, -1.f);
|
|
draw_quad(true, 0);
|
|
}
|
|
restore_viewport();
|
|
|
|
// Draw fill or indeterminate handle
|
|
// For vertical bars, progress grows from bottom to top, so invert the fraction
|
|
if (is_indeterminate) {
|
|
// Animate a handle bouncing back and forth like a scrollbar thumb
|
|
const float handle_size = 0.15f; // 15% of track length
|
|
const monotonic_t cycle = s_double_to_monotonic_t(2.0); // 2 second full cycle (forth and back)
|
|
monotonic_t elapsed = screen->progress_indeterminate_anim_at > 0 ? monotonic() - screen->progress_indeterminate_anim_at : 0;
|
|
double t = (double)(elapsed % cycle) / (double)cycle; // 0..1 over one cycle
|
|
// Triangle wave: goes 0→1→0 over one cycle
|
|
float pos_frac = (float)(t < 0.5 ? t * 2.0 : 2.0 - t * 2.0);
|
|
if (!is_horizontal) pos_frac = 1.0f - pos_frac; // vertical: bounce bottom-to-top first
|
|
float handle_start = pos_frac * (1.0f - handle_size);
|
|
if (opacity > 0.0f) {
|
|
draw_progress_handle(ui, bar_color, opacity, bar_radius,
|
|
track_left, track_top, track_width, track_height,
|
|
handle_start, handle_size, is_horizontal);
|
|
}
|
|
} else if (fill_fraction > 0.0f && opacity > 0.0f) {
|
|
float handle_start = is_horizontal ? 0.0f : 1.0f - fill_fraction;
|
|
draw_progress_handle(ui, bar_color, opacity, bar_radius,
|
|
track_left, track_top, track_width, track_height,
|
|
handle_start, fill_fraction, is_horizontal);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
draw_window_logo(const UIRenderData *ui) {
|
|
struct { unsigned width, height; int left, top; } w;
|
|
WindowLogoRenderData *wl = ui->window_logo;
|
|
w.height = wl->instance->height; w.width = wl->instance->width;
|
|
if (OPT(window_logo_scale.width) > 0 || OPT(window_logo_scale.height) > 0) {
|
|
unsigned scaled_wl_width = ui->screen_width, scaled_wl_height = ui->screen_height;
|
|
|
|
// [sx] Scales logo to sx % of the viewports shortest dimension, preserving aspect ratio
|
|
if (OPT(window_logo_scale.height) < 0) {
|
|
if (ui->screen_height < ui->screen_width) {
|
|
scaled_wl_height = (int)(ui->screen_height * OPT(window_logo_scale.width) / 100);
|
|
scaled_wl_width = wl->instance->width * scaled_wl_height / wl->instance->height;
|
|
} else {
|
|
scaled_wl_width = (int)(ui->screen_width * OPT(window_logo_scale.width) / 100);
|
|
scaled_wl_height = wl->instance->height * scaled_wl_width / wl->instance->width;
|
|
}
|
|
}
|
|
// [0 sy] Scales logo's y dimension to sy % of viewporty keeping original x dimension
|
|
else if (OPT(window_logo_scale.width) == 0.0) {
|
|
scaled_wl_height = (int)(scaled_wl_height * OPT(window_logo_scale.height) / 100);
|
|
scaled_wl_width = wl->instance->width;
|
|
}
|
|
// [sx 0] Scales logo's x dimension to sx % of viewportx keeping original y dimension
|
|
else if (OPT(window_logo_scale.height) == 0.0) {
|
|
scaled_wl_width = (int)(scaled_wl_width * OPT(window_logo_scale.width) / 100);
|
|
scaled_wl_height = wl->instance->height;
|
|
}
|
|
// [sx sy] Scales logo's x and y dimension to sx and sy % of viewportx and viewporty respectively
|
|
else {
|
|
scaled_wl_height = (int)(scaled_wl_height * OPT(window_logo_scale.height) / 100);
|
|
scaled_wl_width = (int)(scaled_wl_width * OPT(window_logo_scale.width) / 100);
|
|
}
|
|
w.width = scaled_wl_width; w.height = scaled_wl_height;
|
|
}
|
|
w.left = (int)(ui->screen_width * wl->position.canvas_x - w.width * wl->position.image_x);
|
|
w.top = (int)(ui->screen_height * wl->position.canvas_y - w.height * wl->position.image_y);
|
|
float left = gl_pos_x(w.left, ui->screen_width), top = gl_pos_y(w.top, ui->screen_height);
|
|
ImageRenderData d = {.texture_id = wl->instance->texture_id};
|
|
gpu_data_for_image(&d, left, top, left + gl_size(w.width, ui->screen_width), top - gl_size(w.height, ui->screen_height));
|
|
draw_graphics(GRAPHICS_PROGRAM, &d, 0, 1, ui->inactive_text_alpha * OPT(window_logo_alpha));
|
|
}
|
|
|
|
bool
|
|
screen_needs_rendering_in_layers(OSWindow *os_window, Window *w, Screen *screen) {
|
|
const bool has_ui = w && ((screen->start_visual_bell_at | screen->start_drag_overlay_at) || has_scrollbar(w, screen) || has_progress_bar(screen) || has_hyperlink_target(os_window, w, screen) || has_window_number(w, screen) || w->window_logo.id);
|
|
GraphicsManager *grman = screen->paused_rendering.expires_at && screen->paused_rendering.grman ? screen->paused_rendering.grman : screen->grman;
|
|
return has_ui || grman_has_images(grman);
|
|
}
|
|
|
|
bool
|
|
current_framebuffer_is_ok(void) {
|
|
return check_framebuffer_status() == NULL;
|
|
}
|
|
|
|
// }}}
|
|
|
|
enum { DRAW_NEITHER_BG = 0, DRAW_DEFAULT_BG = 1, DRAW_NON_DEFAULT_BG = 2, DRAW_BOTH_BG = 3};
|
|
|
|
static void
|
|
call_cell_program(int program, const UIRenderData *ui, ssize_t vao_idx, bool for_final_output, unsigned draw_bg_bitfield) {
|
|
bind_program(program);
|
|
CELL_BUFFERS;
|
|
bind_vao_uniform_buffer(vao_idx, uniform_buffer, CELL_RENDER_DATA_BINDING_POINT);
|
|
bind_vao_uniform_buffer(vao_idx, color_table_buffer, COLOR_TABLE_BINDING_POINT);
|
|
glUniform1ui(cell_program_layouts[program].uniforms.draw_bg_bitfield, draw_bg_bitfield);
|
|
glUniform1f(cell_program_layouts[program].uniforms.row_offset, row_offset_for_screen(ui->screen));
|
|
if (for_final_output) glEnable(GL_FRAMEBUFFER_SRGB);
|
|
draw_quad(!for_final_output, render_lines_for_screen(ui->screen) * ui->screen->columns);
|
|
if (for_final_output) glDisable(GL_FRAMEBUFFER_SRGB);
|
|
}
|
|
|
|
static void
|
|
draw_cells_without_layers(const UIRenderData *ui, ssize_t vao_idx) {
|
|
call_cell_program(CELL_PROGRAM, ui, vao_idx, true, DRAW_BOTH_BG);
|
|
}
|
|
|
|
static void
|
|
draw_tint(const UIRenderData *ui) {
|
|
bind_program(TINT_PROGRAM);
|
|
color_vec4_premult(tint_program_layout.uniforms.tint_color, ui->background_color, OPT(background_tint));
|
|
glUniform4f(tint_program_layout.uniforms.edges, -1, 1, 1, -1);
|
|
draw_quad(true, 0);
|
|
}
|
|
|
|
static void
|
|
draw_cells_with_layers(const UIRenderData *ui, ssize_t vao_idx) {
|
|
if (ui->has_background_image && OPT(background_tint) > 0) draw_tint(ui);
|
|
const bool has_content_between_background_and_foreground = ui->window_logo != NULL || ui->grd.num_of_below_refs > 0 || ui->grd.num_of_negative_refs > 0;
|
|
if (has_content_between_background_and_foreground) {
|
|
if (!ui->has_background_image) call_cell_program(CELL_BG_PROGRAM, ui, vao_idx, false, DRAW_DEFAULT_BG);
|
|
if (ui->window_logo != NULL) draw_window_logo(ui);
|
|
if (ui->grd.num_of_below_refs > 0) draw_graphics(
|
|
GRAPHICS_PROGRAM, ui->grd.images, 0, ui->grd.num_of_below_refs, ui->inactive_text_alpha);
|
|
call_cell_program(CELL_BG_PROGRAM, ui, vao_idx, false, DRAW_NON_DEFAULT_BG);
|
|
if (ui->grd.num_of_negative_refs) draw_graphics(
|
|
GRAPHICS_PROGRAM, ui->grd.images, ui->grd.num_of_below_refs, ui->grd.num_of_negative_refs,
|
|
ui->inactive_text_alpha);
|
|
call_cell_program(CELL_FG_PROGRAM, ui, vao_idx, false, DRAW_NEITHER_BG);
|
|
} else call_cell_program(CELL_PROGRAM, ui, vao_idx, false, ui->has_background_image ? DRAW_NON_DEFAULT_BG : DRAW_BOTH_BG);
|
|
|
|
if (ui->grd.num_of_positive_refs > 0) draw_graphics(
|
|
GRAPHICS_PROGRAM, ui->grd.images, ui->grd.num_of_below_refs + ui->grd.num_of_negative_refs,
|
|
ui->grd.num_of_positive_refs, ui->inactive_text_alpha);
|
|
|
|
draw_visual_bell(ui);
|
|
draw_drag_preview_overlay(ui);
|
|
draw_progress_bar(ui);
|
|
draw_scrollbar(ui);
|
|
draw_hyperlink_target(ui);
|
|
draw_window_number(ui);
|
|
}
|
|
|
|
void
|
|
blank_canvas(float background_opacity, color_type color_in_srgb, bool for_final_output) {
|
|
if (for_final_output) {
|
|
// we need to write pre-multiplied sRGB color to framebuffer
|
|
#define C(shift) ((((color_in_srgb >> shift) & 0xFF) / 255.f) * background_opacity)
|
|
glClearColor(C(16), C(8), C(0), background_opacity);
|
|
#undef C
|
|
} else {
|
|
// we need to write pre-multiplied linear color to framebuffer
|
|
#define C(shift) srgb_color((color_in_srgb >> shift) & 0xFF) * background_opacity
|
|
glClearColor(C(16), C(8), C(0), background_opacity);
|
|
#undef C
|
|
}
|
|
glClear(GL_COLOR_BUFFER_BIT);
|
|
}
|
|
|
|
bool
|
|
send_cell_data_to_gpu(ssize_t vao_idx, Screen *screen, OSWindow *os_window) {
|
|
bool changed = false;
|
|
if (os_window->fonts_data) {
|
|
if (cell_prepare_to_render(vao_idx, screen, os_window->fonts_data)) changed = true;
|
|
}
|
|
return changed;
|
|
}
|
|
|
|
void
|
|
draw_cells(const WindowRenderData *srd, OSWindow *os_window, bool is_active_window, bool is_tab_bar, bool is_single_window, Window *window) {
|
|
Screen *screen = srd->screen;
|
|
CELL_BUFFERS;
|
|
bind_vertex_array(srd->vao_idx);
|
|
// When inactive_text_alpha is negative, its absolute value is used as the
|
|
// opacity and fading is based only on whether the current window is active.
|
|
const float inactive_text_alpha = fabsf(OPT(inactive_text_alpha));
|
|
const bool use_active_window_only = OPT(inactive_text_alpha) < 0.f;
|
|
float current_inactive_text_alpha = use_active_window_only ?
|
|
(is_tab_bar || is_active_window ? 1.0f : inactive_text_alpha) :
|
|
(is_tab_bar || (!is_single_window && is_active_window) || (is_single_window && screen->cursor_render_info.is_focused) ? 1.0f : inactive_text_alpha);
|
|
float bg_alpha = effective_os_window_alpha(os_window);
|
|
|
|
color_type default_bg = cell_update_uniform_block(
|
|
srd->vao_idx, screen, uniform_buffer, color_table_buffer, &screen->cursor_render_info, os_window, current_inactive_text_alpha, bg_alpha);
|
|
set_cell_uniforms(screen->reload_all_gpu_data);
|
|
WindowLogoRenderData *wl;
|
|
if (window && (wl = &window->window_logo) && wl->id && (wl->instance = find_window_logo(global_state.all_window_logos, wl->id)) && wl->instance->load_from_disk_ok) {
|
|
if (!window->window_logo.instance->texture_id) {
|
|
set_on_gpu_state(window->window_logo.instance, true);
|
|
}
|
|
} else wl = NULL;
|
|
GraphicsManager *grman = screen->paused_rendering.expires_at && screen->paused_rendering.grman ? screen->paused_rendering.grman : screen->grman;
|
|
UIRenderData ui = {
|
|
.screen_width = srd->geometry.right - srd->geometry.left,
|
|
.screen_height = srd->geometry.bottom - srd->geometry.top,
|
|
.cell_width = os_window->fonts_data->fcm.cell_width,
|
|
.cell_height = os_window->fonts_data->fcm.cell_height,
|
|
.screen_left = srd->geometry.left, .screen_top = srd->geometry.top,
|
|
.full_framebuffer_width = os_window->viewport_width, .full_framebuffer_height = os_window->viewport_height,
|
|
.window = window, .screen = screen, .os_window = os_window, .grd = grman_render_data(grman), .window_logo = wl,
|
|
.inactive_text_alpha = current_inactive_text_alpha, .has_background_image = has_bgimage(os_window),
|
|
.background_color = default_bg, .bg_alpha=effective_os_window_alpha(os_window),
|
|
};
|
|
screen->reload_all_gpu_data = false;
|
|
save_viewport_using_top_left_origin(
|
|
ui.screen_left, ui.screen_top, ui.screen_width, ui.screen_height, ui.full_framebuffer_height);
|
|
if (ui.os_window->needs_layers) draw_cells_with_layers(&ui, srd->vao_idx);
|
|
else draw_cells_without_layers(&ui, srd->vao_idx);
|
|
restore_viewport();
|
|
}
|
|
// }}}
|
|
|
|
// Borders {{{
|
|
|
|
typedef struct BorderProgramLayout {
|
|
BorderUniforms uniforms;
|
|
} BorderProgramLayout;
|
|
static BorderProgramLayout border_program_layout;
|
|
|
|
static void
|
|
init_borders_program(void) {
|
|
get_uniform_locations_border(BORDERS_PROGRAM, &border_program_layout.uniforms);
|
|
bind_program(BORDERS_PROGRAM);
|
|
glUniform1fv(border_program_layout.uniforms.gamma_lut, 256, srgb_lut);
|
|
}
|
|
|
|
ssize_t
|
|
create_border_vao(void) {
|
|
ssize_t vao_idx = create_vao();
|
|
|
|
add_buffer_to_vao(vao_idx, GL_ARRAY_BUFFER);
|
|
add_attribute_to_vao(BORDERS_PROGRAM, vao_idx, "rect",
|
|
/*size=*/4, /*dtype=*/GL_FLOAT, /*stride=*/sizeof(BorderRect), /*offset=*/(void*)offsetof(BorderRect, left), /*divisor=*/1);
|
|
add_attribute_to_vao(BORDERS_PROGRAM, vao_idx, "rect_color",
|
|
/*size=*/1, /*dtype=*/GL_UNSIGNED_INT, /*stride=*/sizeof(BorderRect), /*offset=*/(void*)(offsetof(BorderRect, color)), /*divisor=*/1);
|
|
|
|
return vao_idx;
|
|
}
|
|
|
|
void
|
|
draw_borders(ssize_t vao_idx, unsigned int num_border_rects, BorderRect *rect_buf, bool rect_data_is_dirty, color_type active_window_bg, unsigned int num_visible_windows, bool all_windows_have_same_bg, OSWindow *w) {
|
|
float background_opacity = effective_os_window_alpha(w);
|
|
if (!num_border_rects) return;
|
|
bind_program(BORDERS_PROGRAM); bind_vertex_array(vao_idx);
|
|
const bool has_background_image = has_bgimage(w);
|
|
if (has_background_image) background_opacity = OPT(background_tint) * OPT(background_tint_gaps);
|
|
if (rect_data_is_dirty) {
|
|
const size_t sz = sizeof(BorderRect) * num_border_rects;
|
|
void *borders_buf_address = alloc_and_map_vao_buffer(vao_idx, sz, 0, false);
|
|
if (borders_buf_address) memcpy(borders_buf_address, rect_buf, sz);
|
|
unmap_vao_buffer(vao_idx, 0);
|
|
}
|
|
color_type default_bg = (num_visible_windows > 1 && !all_windows_have_same_bg) ? OPT(background) : active_window_bg;
|
|
GLuint colors[9] = {
|
|
default_bg, OPT(active_border_color), OPT(inactive_border_color), 0,
|
|
OPT(bell_border_color), OPT(tab_bar_background), OPT(tab_bar_margin_color),
|
|
w->tab_bar_edge_color.left, w->tab_bar_edge_color.right
|
|
};
|
|
glUniform1uiv(border_program_layout.uniforms.colors, arraysz(colors), colors);
|
|
glUniform1f(border_program_layout.uniforms.background_opacity, background_opacity);
|
|
if (!w->needs_layers) glEnable(GL_FRAMEBUFFER_SRGB);
|
|
draw_quad(has_background_image, num_border_rects);
|
|
if (!w->needs_layers) glDisable(GL_FRAMEBUFFER_SRGB);
|
|
unbind_program(); unbind_vertex_array();
|
|
}
|
|
|
|
// }}}
|
|
|
|
// Cursor Trail {{{
|
|
void
|
|
draw_cursor_trail(CursorTrail *trail, Window *active_window) {
|
|
bind_program(TRAIL_PROGRAM);
|
|
|
|
glUniform4fv(trail_program_layout.uniforms.x_coords, 1, trail->corner_x);
|
|
glUniform4fv(trail_program_layout.uniforms.y_coords, 1, trail->corner_y);
|
|
|
|
glUniform2fv(trail_program_layout.uniforms.cursor_edge_x, 1, trail->cursor_edge_x);
|
|
glUniform2fv(trail_program_layout.uniforms.cursor_edge_y, 1, trail->cursor_edge_y);
|
|
|
|
color_type trail_color = OPT(cursor_trail_color);
|
|
if (trail_color == 0) { // 0 means "none" was specified
|
|
trail_color = active_window ? active_window->render_data.screen->last_rendered.cursor_bg : OPT(foreground);
|
|
}
|
|
color_vec3(trail_program_layout.uniforms.trail_color, trail_color);
|
|
|
|
glUniform1f(trail_program_layout.uniforms.trail_opacity, trail->opacity);
|
|
|
|
draw_quad(true, 0);
|
|
unbind_program();
|
|
}
|
|
|
|
// }}}
|
|
|
|
// OSWindow {{{
|
|
static void
|
|
draw_bg_image(OSWindow *os_window, Tab *tab) {
|
|
BackgroundImage *bg = background_image_for_os_window(os_window);
|
|
if (!bg) return;
|
|
BackgroundImageLayout layout = os_window->background_image.has_layout ? os_window->background_image.layout : OPT(background_image_layout);
|
|
BackgroundImageRenderSettings s = {
|
|
.os_window.width = os_window->viewport_width, .os_window.height = os_window->viewport_height,
|
|
.instance_id = bg->id, .layout=layout,
|
|
.linear=OPT(background_image_linear), .bgcolor=OPT(background), .opacity=effective_os_window_alpha(os_window),
|
|
};
|
|
GLfloat iwidth = bg->width, iheight = bg->height;
|
|
GLfloat vwidth = s.os_window.width, vheight = s.os_window.height;
|
|
if (CENTER_SCALED == layout) {
|
|
GLfloat ifrac = iwidth / iheight;
|
|
if (ifrac > (vwidth / vheight)) {
|
|
iheight = vheight;
|
|
iwidth = iheight * ifrac;
|
|
} else {
|
|
iwidth = vwidth;
|
|
iheight = iwidth / ifrac;
|
|
}
|
|
}
|
|
GLfloat tiled = 0.f;;
|
|
GLfloat left = -1.0, top = 1.0, right = 1.0, bottom = -1.0;
|
|
switch (layout) {
|
|
case TILING: case MIRRORED: case CLAMPED:
|
|
tiled = 1.f; break;
|
|
case SCALED:
|
|
break;
|
|
case CENTER_CLAMPED:
|
|
case CENTER_SCALED: {
|
|
GLfloat wfrac = (vwidth - iwidth) / vwidth;
|
|
GLfloat hfrac = (vheight - iheight) / vheight;
|
|
left += wfrac;
|
|
right -= wfrac;
|
|
top -= hfrac;
|
|
bottom += hfrac;
|
|
} break;
|
|
}
|
|
bind_program(BGIMAGE_PROGRAM);
|
|
// altough we dont use this VO we need to ensure *some* VAO is bound at this point.
|
|
bind_vertex_array(tab->border_rects.vao_idx);
|
|
glUniform4f(bgimage_program_layout.uniforms.sizes, vwidth, vheight, iwidth, iheight);
|
|
glUniform1f(bgimage_program_layout.uniforms.tiled, tiled);
|
|
glUniform4f(bgimage_program_layout.uniforms.positions, left, top, right, bottom);
|
|
glUniform1i(bgimage_program_layout.uniforms.image, GRAPHICS_UNIT);
|
|
color_vec4(bgimage_program_layout.uniforms.background, s.bgcolor, s.opacity);
|
|
glActiveTexture(GL_TEXTURE0 + GRAPHICS_UNIT);
|
|
glBindTexture(GL_TEXTURE_2D, bg->texture_id);
|
|
draw_quad(false, 0);
|
|
unbind_program();
|
|
}
|
|
|
|
static void
|
|
gpu_data_for_centered_image(ImageRenderData *ans, unsigned int screen_width_px, unsigned int screen_height_px, unsigned int width, unsigned int height) {
|
|
float width_frac = 2 * MIN(1, width / (float)screen_width_px), height_frac = 2 * MIN(1, height / (float)screen_height_px);
|
|
float hmargin = (2 - width_frac) / 2;
|
|
float vmargin = (2 - height_frac) / 2;
|
|
gpu_data_for_image(ans, -1 + hmargin, 1 - vmargin, -1 + hmargin + width_frac, 1 - vmargin - height_frac);
|
|
}
|
|
|
|
static void
|
|
draw_centered_alpha_mask(size_t screen_width, size_t screen_height, size_t width, size_t height, uint8_t *canvas, float background_opacity) {
|
|
ImageRenderData *data = load_alpha_mask_texture(width, height, canvas);
|
|
gpu_data_for_centered_image(data, screen_width, screen_height, width, height);
|
|
bind_program(GRAPHICS_ALPHA_MASK_PROGRAM);
|
|
glUniform1i(graphics_program_layouts[GRAPHICS_ALPHA_MASK_PROGRAM].uniforms.image, GRAPHICS_UNIT);
|
|
color_vec3(graphics_program_layouts[GRAPHICS_ALPHA_MASK_PROGRAM].uniforms.amask_fg, OPT(foreground));
|
|
color_vec4_premult(graphics_program_layouts[GRAPHICS_ALPHA_MASK_PROGRAM].uniforms.amask_bg_premult, OPT(background), background_opacity);
|
|
draw_graphics(GRAPHICS_ALPHA_MASK_PROGRAM, data, 0, 1, 1.0);
|
|
}
|
|
|
|
static void
|
|
draw_resizing_text(OSWindow *w) {
|
|
if (monotonic() - w->created_at > ms_to_monotonic_t(1000) && w->live_resize.num_of_resize_events > 1) {
|
|
char text[32] = {0};
|
|
unsigned int width = w->live_resize.width, height = w->live_resize.height;
|
|
snprintf(text, sizeof(text), "%u by %u cells", width / w->fonts_data->fcm.cell_width, height / w->fonts_data->fcm.cell_height);
|
|
StringCanvas rendered = render_simple_text(w->fonts_data, text);
|
|
if (rendered.canvas) {
|
|
draw_centered_alpha_mask(width, height, rendered.width, rendered.height, rendered.canvas, MAX(0.2f, MIN(OPT(background_opacity), 0.8f)));
|
|
free(rendered.canvas);
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
blank_os_window(OSWindow *osw) {
|
|
color_type color = OPT(background);
|
|
if (osw->num_tabs > 0) {
|
|
Tab *t = osw->tabs + osw->active_tab;
|
|
if (t->num_windows == 1) {
|
|
Window *w = t->windows + t->active_window;
|
|
Screen *s = w->render_data.screen;
|
|
if (s) {
|
|
color = colorprofile_to_color(s->color_profile, s->color_profile->overridden.default_bg, s->color_profile->configured.default_bg).rgb;
|
|
}
|
|
}
|
|
}
|
|
blank_canvas(effective_os_window_alpha(osw), color, true);
|
|
}
|
|
|
|
static void
|
|
start_os_window_rendering(OSWindow *os_window, Tab *tab) {
|
|
// note that during live resize rendering is done in layers
|
|
if (os_window->live_resize.in_progress) blank_os_window(os_window);
|
|
if (os_window->needs_layers) {
|
|
// Ensure the global shared texture is large enough for this window
|
|
if (global_state.layers_render_texture.width < os_window->viewport_width ||
|
|
global_state.layers_render_texture.height < os_window->viewport_height) {
|
|
if (global_state.layers_render_texture.texture_id) free_texture(&global_state.layers_render_texture.texture_id);
|
|
if (global_state.layers_render_texture.framebuffer_id) free_framebuffer(&global_state.layers_render_texture.framebuffer_id);
|
|
unsigned new_w = (unsigned)MAX(global_state.layers_render_texture.width, os_window->viewport_width);
|
|
unsigned new_h = (unsigned)MAX(global_state.layers_render_texture.height, os_window->viewport_height);
|
|
setup_texture_as_render_target(new_w, new_h, &global_state.layers_render_texture.texture_id, &global_state.layers_render_texture.framebuffer_id);
|
|
global_state.layers_render_texture.width = (int)new_w;
|
|
global_state.layers_render_texture.height = (int)new_h;
|
|
global_state.layers_render_texture.texture_generation++;
|
|
}
|
|
// Create per-window framebuffer if needed and attach the global texture to it
|
|
if (!os_window->indirect_output.framebuffer_id) glGenFramebuffers(1, &os_window->indirect_output.framebuffer_id);
|
|
if (os_window->indirect_output.attached_texture_generation != global_state.layers_render_texture.texture_generation) {
|
|
bind_framebuffer_for_output(os_window->indirect_output.framebuffer_id);
|
|
glFramebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D, global_state.layers_render_texture.texture_id, 0);
|
|
os_window->indirect_output.attached_texture_generation = global_state.layers_render_texture.texture_generation;
|
|
}
|
|
set_framebuffer_to_use_for_output(os_window->indirect_output.framebuffer_id);
|
|
bind_framebuffer_for_output(0);
|
|
save_viewport_using_bottom_left_origin(0, 0, os_window->viewport_width, os_window->viewport_height);
|
|
clear_current_framebuffer();
|
|
draw_bg_image(os_window, tab);
|
|
}
|
|
}
|
|
|
|
static void
|
|
stop_os_window_rendering(OSWindow *os_window, Tab *tab, Window *active_window) {
|
|
if (OPT(cursor_trail) && tab->cursor_trail.needs_render) draw_cursor_trail(&tab->cursor_trail, active_window);
|
|
if (os_window->needs_layers) {
|
|
set_framebuffer_to_use_for_output(0);
|
|
bind_framebuffer_for_output(0);
|
|
bind_program(BLIT_PROGRAM);
|
|
glActiveTexture(GL_TEXTURE0 + GRAPHICS_UNIT);
|
|
glBindTexture(GL_TEXTURE_2D, global_state.layers_render_texture.texture_id);
|
|
float sx = global_state.layers_render_texture.width > 0 ? (float)os_window->viewport_width / (float)global_state.layers_render_texture.width : 1.f;
|
|
float sy = global_state.layers_render_texture.height > 0 ? (float)os_window->viewport_height / (float)global_state.layers_render_texture.height : 1.f;
|
|
glUniform4f(blit_program_layout.uniforms.src_rect, 0, sy, sx, 0);
|
|
glUniform4f(blit_program_layout.uniforms.dest_rect, -1, 1, 1, -1);
|
|
restore_viewport();
|
|
if (os_window->live_resize.in_progress) save_viewport_using_top_left_origin(
|
|
0, 0, os_window->viewport_width, os_window->viewport_height, os_window->live_resize.height);
|
|
draw_quad(false, 0);
|
|
if (os_window->live_resize.in_progress) {
|
|
restore_viewport();
|
|
draw_resizing_text(os_window);
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
setup_os_window_for_rendering(OSWindow *os_window, Tab *tab, Window *active_window, bool start) {
|
|
if (start) start_os_window_rendering(os_window, tab);
|
|
else stop_os_window_rendering(os_window, tab, active_window);
|
|
}
|
|
|
|
// Take a screenshot of the OS Window, must be called immediately after
|
|
// the OSWindow is rendered into the back buffer and before the buffers
|
|
// are swapped. If thumb_w or thumb_h are zero the are set to the corresponding
|
|
// dimension of the source region (viewport or central region without tab bar).
|
|
// Takes a screenshot of a rectangular region of the OSWindow's framebuffer.
|
|
// The region parameter specifies which part of the framebuffer to capture.
|
|
// Scaling is performed on the GPU using the SCREENSHOT_PROGRAM shader for better performance.
|
|
// The shader properly handles sRGB color space conversion and downscaling.
|
|
// Setting the thumbnail dimensions to zero disables scaling.
|
|
void
|
|
take_screenshot_of_rectangular_region(OSWindow *os_window, Region region, unsigned char *dst_buf, unsigned *thumb_w, unsigned *thumb_h) {
|
|
unsigned vw = os_window->viewport_width;
|
|
unsigned vh = os_window->viewport_height;
|
|
|
|
// Calculate the source region dimensions
|
|
unsigned src_height = region.bottom - region.top;
|
|
unsigned src_width = region.right - region.left;
|
|
|
|
if (!*thumb_w) *thumb_w = src_width;
|
|
if (!*thumb_h) *thumb_h = src_height;
|
|
*thumb_w = MIN(src_width, *thumb_w);
|
|
*thumb_h = MIN(src_height, *thumb_h);
|
|
|
|
// Create a texture to hold the current framebuffer content
|
|
GLuint src_texture = 0;
|
|
glGenTextures(1, &src_texture);
|
|
glBindTexture(GL_TEXTURE_2D, src_texture);
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE);
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE);
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
|
|
|
|
// Copy the current framebuffer to the texture
|
|
glCopyTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, 0, 0, vw, vh, 0);
|
|
|
|
// Create a temporary framebuffer for GPU-based scaling
|
|
GLuint temp_texture = 0, temp_framebuffer = 0;
|
|
setup_texture_as_render_target(*thumb_w, *thumb_h, &temp_texture, &temp_framebuffer);
|
|
|
|
// Save current state
|
|
GLint current_framebuffer;
|
|
glGetIntegerv(GL_DRAW_FRAMEBUFFER_BINDING, ¤t_framebuffer);
|
|
|
|
// Bind our temporary framebuffer for rendering
|
|
bind_framebuffer_for_output(temp_framebuffer);
|
|
save_viewport_using_bottom_left_origin(0, 0, *thumb_w, *thumb_h);
|
|
|
|
// Use the screenshot program to render the scaled framebuffer with proper color space handling
|
|
bind_program(SCREENSHOT_PROGRAM);
|
|
|
|
// Set source rectangle (normalized coordinates: 0 to 1)
|
|
// Note: OpenGL texture origin is bottom-left, but Region uses top-left origin
|
|
// Convert from screen coordinates (top-left origin) to OpenGL texture coordinates (bottom-left origin)
|
|
float src_left_norm = (float)region.left / (float)vw;
|
|
float src_right_norm = (float)region.right / (float)vw;
|
|
float src_bottom_norm = (float)(vh - region.bottom) / (float)vh;
|
|
float src_top_norm = (float)(vh - region.top) / (float)vh;
|
|
glUniform4f(screenshot_program_layout.uniforms.src_rect, src_left_norm, src_top_norm, src_right_norm, src_bottom_norm);
|
|
|
|
// Set destination rectangle (NDC coordinates: -1 to 1)
|
|
glUniform4f(screenshot_program_layout.uniforms.dest_rect, -1.0f, -1.0f, 1.0f, 1.0f);
|
|
|
|
// Set the source texture size for proper downscaling
|
|
glUniform2f(screenshot_program_layout.uniforms.src_size, (float)vw, (float)vh);
|
|
|
|
// Bind the source texture
|
|
glActiveTexture(GL_TEXTURE0 + GRAPHICS_UNIT);
|
|
glBindTexture(GL_TEXTURE_2D, src_texture);
|
|
|
|
// Draw the scaled quad
|
|
draw_quad(false, 0);
|
|
|
|
// Read the scaled result
|
|
glPixelStorei(GL_PACK_ALIGNMENT, 1);
|
|
glReadPixels(0, 0, *thumb_w, *thumb_h, GL_RGBA, GL_UNSIGNED_BYTE, dst_buf);
|
|
glPixelStorei(GL_PACK_ALIGNMENT, 4);
|
|
|
|
// Restore previous state
|
|
bind_framebuffer_for_output(current_framebuffer);
|
|
restore_viewport();
|
|
|
|
// Clean up temporary resources
|
|
free_texture(&src_texture);
|
|
free_texture(&temp_texture);
|
|
free_framebuffer(&temp_framebuffer);
|
|
}
|
|
// }}}
|
|
|
|
// Python API {{{
|
|
|
|
static PyObject*
|
|
pygpu_driver_version_string(PyObject *self UNUSED, PyObject *args UNUSED) {
|
|
return PyUnicode_FromString(global_state.gl_version ? gl_version_string() : "");
|
|
}
|
|
|
|
static bool
|
|
attach_shaders(PyObject *sources, GLuint program_id, GLenum shader_type) {
|
|
RAII_ALLOC(const GLchar*, c_sources, calloc(PyTuple_GET_SIZE(sources), sizeof(GLchar*)));
|
|
for (Py_ssize_t i = 0; i < PyTuple_GET_SIZE(sources); i++) {
|
|
PyObject *temp = PyTuple_GET_ITEM(sources, i);
|
|
if (!PyUnicode_Check(temp)) { PyErr_SetString(PyExc_TypeError, "shaders must be strings"); return false; }
|
|
c_sources[i] = PyUnicode_AsUTF8(temp);
|
|
}
|
|
GLuint shader_id = compile_shaders(shader_type, PyTuple_GET_SIZE(sources), c_sources);
|
|
if (shader_id == 0) return false;
|
|
glAttachShader(program_id, shader_id);
|
|
glDeleteShader(shader_id);
|
|
return true;
|
|
}
|
|
|
|
static PyObject*
|
|
compile_program(PyObject UNUSED *self, PyObject *args) {
|
|
PyObject *vertex_shaders, *fragment_shaders;
|
|
int which, allow_recompile = 0;
|
|
if (!PyArg_ParseTuple(args, "iO!O!|p", &which, &PyTuple_Type, &vertex_shaders, &PyTuple_Type, &fragment_shaders, &allow_recompile)) return NULL;
|
|
if (which < 0 || which >= NUM_PROGRAMS) { PyErr_Format(PyExc_ValueError, "Unknown program: %d", which); return NULL; }
|
|
Program *program = program_ptr(which);
|
|
if (program->id != 0) {
|
|
if (allow_recompile) { glDeleteProgram(program->id); program->id = 0; }
|
|
else { PyErr_SetString(PyExc_ValueError, "program already compiled"); return NULL; }
|
|
}
|
|
#define fail_compile() { glDeleteProgram(program->id); return NULL; }
|
|
program->id = glCreateProgram();
|
|
if (!attach_shaders(vertex_shaders, program->id, GL_VERTEX_SHADER)) fail_compile();
|
|
if (!attach_shaders(fragment_shaders, program->id, GL_FRAGMENT_SHADER)) fail_compile();
|
|
glLinkProgram(program->id);
|
|
GLint ret = GL_FALSE;
|
|
glGetProgramiv(program->id, GL_LINK_STATUS, &ret);
|
|
if (ret != GL_TRUE) {
|
|
GLsizei len;
|
|
static char glbuf[4096];
|
|
glGetProgramInfoLog(program->id, sizeof(glbuf), &len, glbuf);
|
|
PyErr_Format(PyExc_ValueError, "Failed to link GLSL shaders:\n%s", glbuf);
|
|
fail_compile();
|
|
}
|
|
#undef fail_compile
|
|
init_uniforms(which);
|
|
return Py_BuildValue("I", program->id);
|
|
}
|
|
|
|
#define PYWRAP0(name) static PyObject* py##name(PYNOARG)
|
|
#define PYWRAP1(name) static PyObject* py##name(PyObject UNUSED *self, PyObject *args)
|
|
#define PA(fmt, ...) if(!PyArg_ParseTuple(args, fmt, __VA_ARGS__)) return NULL;
|
|
#define ONE_INT(name) PYWRAP1(name) { name(PyLong_AsSsize_t(args)); Py_RETURN_NONE; }
|
|
#define TWO_INT(name) PYWRAP1(name) { int a, b; PA("ii", &a, &b); name(a, b); Py_RETURN_NONE; }
|
|
#define NO_ARG(name) PYWRAP0(name) { name(); Py_RETURN_NONE; }
|
|
#define NO_ARG_INT(name) PYWRAP0(name) { return PyLong_FromSsize_t(name()); }
|
|
|
|
ONE_INT(bind_program)
|
|
NO_ARG(unbind_program)
|
|
|
|
PYWRAP0(create_vao) {
|
|
int ans = create_vao();
|
|
if (ans < 0) return NULL;
|
|
return Py_BuildValue("i", ans);
|
|
}
|
|
|
|
ONE_INT(bind_vertex_array)
|
|
NO_ARG(unbind_vertex_array)
|
|
TWO_INT(unmap_vao_buffer)
|
|
|
|
NO_ARG(init_borders_program)
|
|
|
|
NO_ARG(init_cell_program)
|
|
|
|
static PyObject*
|
|
sprite_map_set_limits(PyObject UNUSED *self, PyObject *args) {
|
|
unsigned int w, h;
|
|
if(!PyArg_ParseTuple(args, "II", &w, &h)) return NULL;
|
|
sprite_tracker_set_limits(w, h);
|
|
max_texture_size = w; max_array_texture_layers = h;
|
|
Py_RETURN_NONE;
|
|
}
|
|
|
|
#define M(name, arg_type) {#name, (PyCFunction)name, arg_type, NULL}
|
|
#define MW(name, arg_type) {#name, (PyCFunction)py##name, arg_type, NULL}
|
|
static PyMethodDef module_methods[] = {
|
|
M(compile_program, METH_VARARGS),
|
|
M(sprite_map_set_limits, METH_VARARGS),
|
|
MW(create_vao, METH_NOARGS),
|
|
MW(gpu_driver_version_string, METH_NOARGS),
|
|
MW(bind_vertex_array, METH_O),
|
|
MW(unbind_vertex_array, METH_NOARGS),
|
|
MW(unmap_vao_buffer, METH_VARARGS),
|
|
MW(bind_program, METH_O),
|
|
MW(unbind_program, METH_NOARGS),
|
|
MW(init_borders_program, METH_NOARGS),
|
|
MW(init_cell_program, METH_NOARGS),
|
|
|
|
{NULL, NULL, 0, NULL} /* Sentinel */
|
|
};
|
|
|
|
static void
|
|
finalize(void) {
|
|
default_visual_bell_animation = free_animation(default_visual_bell_animation);
|
|
}
|
|
|
|
bool
|
|
init_shaders(PyObject *module) {
|
|
#define C(x) if (PyModule_AddIntConstant(module, #x, x) != 0) { PyErr_NoMemory(); return false; }
|
|
C(CELL_PROGRAM); C(CELL_FG_PROGRAM); C(CELL_BG_PROGRAM); C(BORDERS_PROGRAM);
|
|
C(GRAPHICS_PROGRAM); C(GRAPHICS_PREMULT_PROGRAM); C(GRAPHICS_ALPHA_MASK_PROGRAM);
|
|
C(BGIMAGE_PROGRAM); C(TINT_PROGRAM); C(TRAIL_PROGRAM); C(BLIT_PROGRAM); C(SCREENSHOT_PROGRAM); C(ROUNDED_RECT_PROGRAM);
|
|
C(GLSL_VERSION);
|
|
C(GL_VERSION);
|
|
C(GL_VENDOR);
|
|
C(GL_SHADING_LANGUAGE_VERSION);
|
|
C(GL_RENDERER);
|
|
C(GL_TRIANGLE_FAN); C(GL_TRIANGLE_STRIP); C(GL_TRIANGLES); C(GL_LINE_LOOP);
|
|
C(GL_COLOR_BUFFER_BIT);
|
|
C(GL_VERTEX_SHADER);
|
|
C(GL_FRAGMENT_SHADER);
|
|
C(GL_TRUE);
|
|
C(GL_FALSE);
|
|
C(GL_COMPILE_STATUS);
|
|
C(GL_LINK_STATUS);
|
|
C(GL_MAX_ARRAY_TEXTURE_LAYERS); C(GL_TEXTURE_BINDING_BUFFER); C(GL_MAX_TEXTURE_BUFFER_SIZE);
|
|
C(GL_MAX_TEXTURE_SIZE);
|
|
C(GL_TEXTURE_2D_ARRAY);
|
|
C(GL_LINEAR); C(GL_CLAMP_TO_EDGE); C(GL_NEAREST);
|
|
C(GL_TEXTURE_MIN_FILTER); C(GL_TEXTURE_MAG_FILTER);
|
|
C(GL_TEXTURE_WRAP_S); C(GL_TEXTURE_WRAP_T);
|
|
C(GL_UNPACK_ALIGNMENT);
|
|
C(GL_R8); C(GL_RED); C(GL_UNSIGNED_BYTE); C(GL_UNSIGNED_SHORT); C(GL_R32UI); C(GL_RGB32UI); C(GL_RGBA);
|
|
C(GL_TEXTURE_BUFFER); C(GL_STATIC_DRAW); C(GL_STREAM_DRAW); C(GL_DYNAMIC_DRAW);
|
|
C(GL_SRC_ALPHA); C(GL_ONE_MINUS_SRC_ALPHA);
|
|
C(GL_WRITE_ONLY); C(GL_READ_ONLY); C(GL_READ_WRITE);
|
|
C(GL_BLEND); C(GL_FLOAT); C(GL_UNSIGNED_INT); C(GL_ARRAY_BUFFER); C(GL_UNIFORM_BUFFER);
|
|
|
|
#undef C
|
|
if (PyModule_AddFunctions(module, module_methods) != 0) return false;
|
|
register_at_exit_cleanup_func(SHADERS_CLEANUP_FUNC, finalize);
|
|
return true;
|
|
}
|
|
// }}}
|