f53df7da64
Code submissions have continually suffered from formatting inconsistencies that constantly have to be addressed. Using clang-format simplifies this by making code formatting more consistent, and allows automation of the code formatting so that maintainers can focus more on the code itself instead of code formatting.
524 lines
12 KiB
C
524 lines
12 KiB
C
/******************************************************************************
|
|
Copyright (C) 2014 by Nibbles
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation, either version 2 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
******************************************************************************/
|
|
|
|
#include <obs-module.h>
|
|
#include <util/platform.h>
|
|
#include <ft2build.h>
|
|
#include FT_FREETYPE_H
|
|
#include <sys/stat.h>
|
|
#include "text-freetype2.h"
|
|
#include "obs-convenience.h"
|
|
|
|
float offsets[16] = {-2.0f, 0.0f, 0.0f, -2.0f, 2.0f, 0.0f, 2.0f, 0.0f,
|
|
0.0f, 2.0f, 0.0f, 2.0f, -2.0f, 0.0f, -2.0f, 0.0f};
|
|
|
|
extern uint32_t texbuf_w, texbuf_h;
|
|
|
|
void draw_outlines(struct ft2_source *srcdata)
|
|
{
|
|
// Horrible (hopefully temporary) solution for outlines.
|
|
uint32_t *tmp;
|
|
|
|
struct gs_vb_data *vdata = gs_vertexbuffer_get_data(srcdata->vbuf);
|
|
|
|
if (!srcdata->text)
|
|
return;
|
|
|
|
tmp = vdata->colors;
|
|
vdata->colors = srcdata->colorbuf;
|
|
|
|
gs_matrix_push();
|
|
for (int32_t i = 0; i < 8; i++) {
|
|
gs_matrix_translate3f(offsets[i * 2], offsets[(i * 2) + 1],
|
|
0.0f);
|
|
draw_uv_vbuffer(srcdata->vbuf, srcdata->tex,
|
|
srcdata->draw_effect,
|
|
(uint32_t)wcslen(srcdata->text) * 6);
|
|
}
|
|
gs_matrix_identity();
|
|
gs_matrix_pop();
|
|
|
|
vdata->colors = tmp;
|
|
}
|
|
|
|
void draw_drop_shadow(struct ft2_source *srcdata)
|
|
{
|
|
// Horrible (hopefully temporary) solution for drop shadow.
|
|
uint32_t *tmp;
|
|
|
|
struct gs_vb_data *vdata = gs_vertexbuffer_get_data(srcdata->vbuf);
|
|
|
|
if (!srcdata->text)
|
|
return;
|
|
|
|
tmp = vdata->colors;
|
|
vdata->colors = srcdata->colorbuf;
|
|
|
|
gs_matrix_push();
|
|
gs_matrix_translate3f(4.0f, 4.0f, 0.0f);
|
|
draw_uv_vbuffer(srcdata->vbuf, srcdata->tex, srcdata->draw_effect,
|
|
(uint32_t)wcslen(srcdata->text) * 6);
|
|
gs_matrix_identity();
|
|
gs_matrix_pop();
|
|
|
|
vdata->colors = tmp;
|
|
}
|
|
|
|
void set_up_vertex_buffer(struct ft2_source *srcdata)
|
|
{
|
|
FT_UInt glyph_index = 0;
|
|
uint32_t x = 0, space_pos = 0, word_width = 0;
|
|
size_t len;
|
|
|
|
if (!srcdata->text)
|
|
return;
|
|
|
|
if (srcdata->custom_width >= 100)
|
|
srcdata->cx = srcdata->custom_width;
|
|
else
|
|
srcdata->cx = get_ft2_text_width(srcdata->text, srcdata);
|
|
srcdata->cy = srcdata->max_h;
|
|
|
|
obs_enter_graphics();
|
|
if (srcdata->vbuf != NULL) {
|
|
gs_vertbuffer_t *tmpvbuf = srcdata->vbuf;
|
|
srcdata->vbuf = NULL;
|
|
gs_vertexbuffer_destroy(tmpvbuf);
|
|
}
|
|
|
|
if (*srcdata->text == 0) {
|
|
obs_leave_graphics();
|
|
return;
|
|
}
|
|
|
|
srcdata->vbuf =
|
|
create_uv_vbuffer((uint32_t)wcslen(srcdata->text) * 6, true);
|
|
|
|
if (srcdata->custom_width <= 100)
|
|
goto skip_word_wrap;
|
|
if (!srcdata->word_wrap)
|
|
goto skip_word_wrap;
|
|
|
|
len = wcslen(srcdata->text);
|
|
|
|
for (uint32_t i = 0; i <= len; i++) {
|
|
if (i == wcslen(srcdata->text))
|
|
goto eos_check;
|
|
|
|
if (srcdata->text[i] != L' ' && srcdata->text[i] != L'\n')
|
|
goto next_char;
|
|
|
|
eos_check:;
|
|
if (x + word_width > srcdata->custom_width) {
|
|
if (space_pos != 0)
|
|
srcdata->text[space_pos] = L'\n';
|
|
x = 0;
|
|
}
|
|
if (i == wcslen(srcdata->text))
|
|
goto eos_skip;
|
|
|
|
x += word_width;
|
|
word_width = 0;
|
|
if (srcdata->text[i] == L'\n')
|
|
x = 0;
|
|
if (srcdata->text[i] == L' ')
|
|
space_pos = i;
|
|
next_char:;
|
|
glyph_index =
|
|
FT_Get_Char_Index(srcdata->font_face, srcdata->text[i]);
|
|
word_width += src_glyph->xadv;
|
|
eos_skip:;
|
|
}
|
|
|
|
skip_word_wrap:;
|
|
fill_vertex_buffer(srcdata);
|
|
obs_leave_graphics();
|
|
}
|
|
|
|
void fill_vertex_buffer(struct ft2_source *srcdata)
|
|
{
|
|
struct gs_vb_data *vdata = gs_vertexbuffer_get_data(srcdata->vbuf);
|
|
if (vdata == NULL || !srcdata->text)
|
|
return;
|
|
|
|
struct vec2 *tvarray = (struct vec2 *)vdata->tvarray[0].array;
|
|
uint32_t *col = (uint32_t *)vdata->colors;
|
|
|
|
FT_UInt glyph_index = 0;
|
|
|
|
uint32_t dx = 0, dy = srcdata->max_h, max_y = dy;
|
|
uint32_t cur_glyph = 0;
|
|
size_t len = wcslen(srcdata->text);
|
|
|
|
if (srcdata->colorbuf != NULL) {
|
|
bfree(srcdata->colorbuf);
|
|
srcdata->colorbuf = NULL;
|
|
}
|
|
srcdata->colorbuf =
|
|
bzalloc(sizeof(uint32_t) * wcslen(srcdata->text) * 6);
|
|
for (size_t i = 0; i < len * 6; i++) {
|
|
srcdata->colorbuf[i] = 0xFF000000;
|
|
}
|
|
|
|
for (size_t i = 0; i < len; i++) {
|
|
add_linebreak:;
|
|
if (srcdata->text[i] != L'\n')
|
|
goto draw_glyph;
|
|
dx = 0;
|
|
i++;
|
|
dy += srcdata->max_h + 4;
|
|
if (i == wcslen(srcdata->text))
|
|
goto skip_glyph;
|
|
if (srcdata->text[i] == L'\n')
|
|
goto add_linebreak;
|
|
draw_glyph:;
|
|
// Skip filthy dual byte Windows line breaks
|
|
if (srcdata->text[i] == L'\r')
|
|
goto skip_glyph;
|
|
|
|
glyph_index =
|
|
FT_Get_Char_Index(srcdata->font_face, srcdata->text[i]);
|
|
if (src_glyph == NULL)
|
|
goto skip_glyph;
|
|
|
|
if (srcdata->custom_width < 100)
|
|
goto skip_custom_width;
|
|
|
|
if (dx + src_glyph->xadv > srcdata->custom_width) {
|
|
dx = 0;
|
|
dy += srcdata->max_h + 4;
|
|
}
|
|
|
|
skip_custom_width:;
|
|
|
|
set_v3_rect(vdata->points + (cur_glyph * 6),
|
|
(float)dx + (float)src_glyph->xoff,
|
|
(float)dy - (float)src_glyph->yoff,
|
|
(float)src_glyph->w, (float)src_glyph->h);
|
|
set_v2_uv(tvarray + (cur_glyph * 6), src_glyph->u, src_glyph->v,
|
|
src_glyph->u2, src_glyph->v2);
|
|
set_rect_colors2(col + (cur_glyph * 6), srcdata->color[0],
|
|
srcdata->color[1]);
|
|
dx += src_glyph->xadv;
|
|
if (dy - (float)src_glyph->yoff + src_glyph->h > max_y)
|
|
max_y = dy - src_glyph->yoff + src_glyph->h;
|
|
cur_glyph++;
|
|
skip_glyph:;
|
|
}
|
|
|
|
srcdata->cy = max_y;
|
|
}
|
|
|
|
void cache_standard_glyphs(struct ft2_source *srcdata)
|
|
{
|
|
for (uint32_t i = 0; i < num_cache_slots; i++) {
|
|
if (srcdata->cacheglyphs[i] != NULL) {
|
|
bfree(srcdata->cacheglyphs[i]);
|
|
srcdata->cacheglyphs[i] = NULL;
|
|
}
|
|
}
|
|
|
|
srcdata->texbuf_x = 0;
|
|
srcdata->texbuf_y = 0;
|
|
|
|
cache_glyphs(srcdata, L"abcdefghijklmnopqrstuvwxyz"
|
|
L"ABCDEFGHIJKLMNOPQRSTUVWXYZ1234567890"
|
|
L"!@#$%^&*()-_=+,<.>/?\\|[]{}`~ \'\"\0");
|
|
}
|
|
|
|
#define glyph_pos x + (y * slot->bitmap.pitch)
|
|
#define buf_pos (dx + x) + ((dy + y) * texbuf_w)
|
|
|
|
void cache_glyphs(struct ft2_source *srcdata, wchar_t *cache_glyphs)
|
|
{
|
|
FT_GlyphSlot slot;
|
|
FT_UInt glyph_index = 0;
|
|
|
|
if (!srcdata->font_face || !cache_glyphs)
|
|
return;
|
|
|
|
slot = srcdata->font_face->glyph;
|
|
|
|
uint32_t dx = srcdata->texbuf_x, dy = srcdata->texbuf_y;
|
|
|
|
int32_t cached_glyphs = 0;
|
|
size_t len = wcslen(cache_glyphs);
|
|
|
|
for (size_t i = 0; i < len; i++) {
|
|
glyph_index =
|
|
FT_Get_Char_Index(srcdata->font_face, cache_glyphs[i]);
|
|
|
|
if (src_glyph != NULL)
|
|
goto skip_glyph;
|
|
|
|
FT_Load_Glyph(srcdata->font_face, glyph_index, FT_LOAD_DEFAULT);
|
|
FT_Render_Glyph(slot, FT_RENDER_MODE_NORMAL);
|
|
|
|
uint32_t g_w = slot->bitmap.width;
|
|
uint32_t g_h = slot->bitmap.rows;
|
|
|
|
if (srcdata->max_h < g_h)
|
|
srcdata->max_h = g_h;
|
|
|
|
if (dx + g_w >= texbuf_w) {
|
|
dx = 0;
|
|
dy += srcdata->max_h + 1;
|
|
}
|
|
|
|
if (dy + g_h >= texbuf_h) {
|
|
blog(LOG_WARNING,
|
|
"Out of space trying to render glyphs");
|
|
break;
|
|
}
|
|
|
|
src_glyph = bzalloc(sizeof(struct glyph_info));
|
|
src_glyph->u = (float)dx / (float)texbuf_w;
|
|
src_glyph->u2 = (float)(dx + g_w) / (float)texbuf_w;
|
|
src_glyph->v = (float)dy / (float)texbuf_h;
|
|
src_glyph->v2 = (float)(dy + g_h) / (float)texbuf_h;
|
|
src_glyph->w = g_w;
|
|
src_glyph->h = g_h;
|
|
src_glyph->yoff = slot->bitmap_top;
|
|
src_glyph->xoff = slot->bitmap_left;
|
|
src_glyph->xadv = slot->advance.x >> 6;
|
|
|
|
for (uint32_t y = 0; y < g_h; y++) {
|
|
for (uint32_t x = 0; x < g_w; x++)
|
|
srcdata->texbuf[buf_pos] =
|
|
slot->bitmap.buffer[glyph_pos];
|
|
}
|
|
|
|
dx += (g_w + 1);
|
|
if (dx >= texbuf_w) {
|
|
dx = 0;
|
|
dy += srcdata->max_h;
|
|
}
|
|
|
|
cached_glyphs++;
|
|
skip_glyph:;
|
|
}
|
|
|
|
srcdata->texbuf_x = dx;
|
|
srcdata->texbuf_y = dy;
|
|
|
|
if (cached_glyphs > 0) {
|
|
|
|
obs_enter_graphics();
|
|
|
|
if (srcdata->tex != NULL) {
|
|
gs_texture_t *tmp_texture = NULL;
|
|
tmp_texture = srcdata->tex;
|
|
srcdata->tex = NULL;
|
|
gs_texture_destroy(tmp_texture);
|
|
}
|
|
|
|
srcdata->tex = gs_texture_create(
|
|
texbuf_w, texbuf_h, GS_A8, 1,
|
|
(const uint8_t **)&srcdata->texbuf, 0);
|
|
|
|
obs_leave_graphics();
|
|
}
|
|
}
|
|
|
|
time_t get_modified_timestamp(char *filename)
|
|
{
|
|
struct stat stats;
|
|
|
|
// stat is apparently terrifying and horrible, but we only call it once
|
|
// every second at most.
|
|
if (os_stat(filename, &stats) != 0)
|
|
return -1;
|
|
|
|
return stats.st_mtime;
|
|
}
|
|
|
|
static void remove_cr(wchar_t *source)
|
|
{
|
|
int j = 0;
|
|
for (int i = 0; source[i] != '\0'; ++i) {
|
|
if (source[i] != L'\r') {
|
|
source[j++] = source[i];
|
|
}
|
|
}
|
|
source[j] = '\0';
|
|
}
|
|
|
|
void load_text_from_file(struct ft2_source *srcdata, const char *filename)
|
|
{
|
|
FILE *tmp_file = NULL;
|
|
uint32_t filesize = 0;
|
|
char *tmp_read = NULL;
|
|
uint16_t header = 0;
|
|
size_t bytes_read;
|
|
|
|
tmp_file = os_fopen(filename, "rb");
|
|
if (tmp_file == NULL) {
|
|
if (!srcdata->file_load_failed) {
|
|
blog(LOG_WARNING, "Failed to open file %s", filename);
|
|
srcdata->file_load_failed = true;
|
|
}
|
|
return;
|
|
}
|
|
fseek(tmp_file, 0, SEEK_END);
|
|
filesize = (uint32_t)ftell(tmp_file);
|
|
fseek(tmp_file, 0, SEEK_SET);
|
|
bytes_read = fread(&header, 2, 1, tmp_file);
|
|
|
|
if (bytes_read == 2 && header == 0xFEFF) {
|
|
// File is already in UTF-16 format
|
|
if (srcdata->text != NULL) {
|
|
bfree(srcdata->text);
|
|
srcdata->text = NULL;
|
|
}
|
|
srcdata->text = bzalloc(filesize);
|
|
bytes_read = fread(srcdata->text, filesize - 2, 1, tmp_file);
|
|
|
|
bfree(tmp_read);
|
|
fclose(tmp_file);
|
|
|
|
return;
|
|
}
|
|
|
|
fseek(tmp_file, 0, SEEK_SET);
|
|
|
|
tmp_read = bzalloc(filesize + 1);
|
|
bytes_read = fread(tmp_read, filesize, 1, tmp_file);
|
|
fclose(tmp_file);
|
|
|
|
if (srcdata->text != NULL) {
|
|
bfree(srcdata->text);
|
|
srcdata->text = NULL;
|
|
}
|
|
srcdata->text = bzalloc((strlen(tmp_read) + 1) * sizeof(wchar_t));
|
|
os_utf8_to_wcs(tmp_read, strlen(tmp_read), srcdata->text,
|
|
(strlen(tmp_read) + 1));
|
|
|
|
remove_cr(srcdata->text);
|
|
bfree(tmp_read);
|
|
}
|
|
|
|
void read_from_end(struct ft2_source *srcdata, const char *filename)
|
|
{
|
|
FILE *tmp_file = NULL;
|
|
uint32_t filesize = 0, cur_pos = 0, log_lines = 0;
|
|
char *tmp_read = NULL;
|
|
uint16_t value = 0, line_breaks = 0;
|
|
size_t bytes_read;
|
|
char bvalue;
|
|
|
|
bool utf16 = false;
|
|
|
|
tmp_file = fopen(filename, "rb");
|
|
if (tmp_file == NULL) {
|
|
if (!srcdata->file_load_failed) {
|
|
blog(LOG_WARNING, "Failed to open file %s", filename);
|
|
srcdata->file_load_failed = true;
|
|
}
|
|
return;
|
|
}
|
|
bytes_read = fread(&value, 2, 1, tmp_file);
|
|
|
|
if (bytes_read == 2 && value == 0xFEFF)
|
|
utf16 = true;
|
|
|
|
fseek(tmp_file, 0, SEEK_END);
|
|
filesize = (uint32_t)ftell(tmp_file);
|
|
cur_pos = filesize;
|
|
log_lines = srcdata->log_lines;
|
|
|
|
while (line_breaks <= log_lines && cur_pos != 0) {
|
|
if (!utf16)
|
|
cur_pos--;
|
|
else
|
|
cur_pos -= 2;
|
|
fseek(tmp_file, cur_pos, SEEK_SET);
|
|
|
|
if (!utf16) {
|
|
bytes_read = fread(&bvalue, 1, 1, tmp_file);
|
|
if (bytes_read == 1 && bvalue == '\n')
|
|
line_breaks++;
|
|
} else {
|
|
bytes_read = fread(&value, 2, 1, tmp_file);
|
|
if (bytes_read == 2 && value == L'\n')
|
|
line_breaks++;
|
|
}
|
|
}
|
|
|
|
if (cur_pos != 0)
|
|
cur_pos += (utf16) ? 2 : 1;
|
|
|
|
fseek(tmp_file, cur_pos, SEEK_SET);
|
|
|
|
if (utf16) {
|
|
if (srcdata->text != NULL) {
|
|
bfree(srcdata->text);
|
|
srcdata->text = NULL;
|
|
}
|
|
srcdata->text = bzalloc(filesize - cur_pos);
|
|
bytes_read =
|
|
fread(srcdata->text, (filesize - cur_pos), 1, tmp_file);
|
|
|
|
remove_cr(srcdata->text);
|
|
bfree(tmp_read);
|
|
fclose(tmp_file);
|
|
|
|
return;
|
|
}
|
|
|
|
tmp_read = bzalloc((filesize - cur_pos) + 1);
|
|
bytes_read = fread(tmp_read, filesize - cur_pos, 1, tmp_file);
|
|
fclose(tmp_file);
|
|
|
|
if (srcdata->text != NULL) {
|
|
bfree(srcdata->text);
|
|
srcdata->text = NULL;
|
|
}
|
|
srcdata->text = bzalloc((strlen(tmp_read) + 1) * sizeof(wchar_t));
|
|
os_utf8_to_wcs(tmp_read, strlen(tmp_read), srcdata->text,
|
|
(strlen(tmp_read) + 1));
|
|
|
|
remove_cr(srcdata->text);
|
|
bfree(tmp_read);
|
|
}
|
|
|
|
uint32_t get_ft2_text_width(wchar_t *text, struct ft2_source *srcdata)
|
|
{
|
|
FT_GlyphSlot slot = srcdata->font_face->glyph;
|
|
FT_UInt glyph_index = 0;
|
|
uint32_t w = 0, max_w = 0;
|
|
size_t len;
|
|
|
|
if (!text)
|
|
return 0;
|
|
|
|
len = wcslen(text);
|
|
for (size_t i = 0; i < len; i++) {
|
|
glyph_index = FT_Get_Char_Index(srcdata->font_face, text[i]);
|
|
FT_Load_Glyph(srcdata->font_face, glyph_index, FT_LOAD_DEFAULT);
|
|
|
|
if (text[i] == L'\n')
|
|
w = 0;
|
|
else {
|
|
w += slot->advance.x >> 6;
|
|
if (w > max_w)
|
|
max_w = w;
|
|
}
|
|
}
|
|
|
|
return max_w;
|
|
}
|