mirror of
https://git.sr.ht/~calebccff/pbsplash
synced 2026-01-12 20:48:40 -09:00
lots of fixes, add animations!
This commit is contained in:
parent
5f6f3726bb
commit
ec1bfbd2fd
8 changed files with 140 additions and 73 deletions
3
.gitignore
vendored
3
.gitignore
vendored
|
|
@ -1,3 +1,4 @@
|
||||||
builddir/
|
builddir/
|
||||||
build/
|
build/
|
||||||
|
*.json
|
||||||
|
*.jsonc
|
||||||
|
|
|
||||||
4
.vscode/settings.json
vendored
4
.vscode/settings.json
vendored
|
|
@ -10,6 +10,8 @@
|
||||||
"vector": "c",
|
"vector": "c",
|
||||||
"string_view": "c",
|
"string_view": "c",
|
||||||
"initializer_list": "c",
|
"initializer_list": "c",
|
||||||
"valarray": "c"
|
"valarray": "c",
|
||||||
|
"random": "c",
|
||||||
|
"nanosvgrast.h": "c"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
@ -71,6 +71,8 @@ extern "C" {
|
||||||
nsvgDelete(image);
|
nsvgDelete(image);
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#define NSVG_MAX_UNICODE_LEN 32
|
||||||
|
|
||||||
enum NSVGpaintType {
|
enum NSVGpaintType {
|
||||||
NSVG_PAINT_NONE = 0,
|
NSVG_PAINT_NONE = 0,
|
||||||
NSVG_PAINT_COLOR = 1,
|
NSVG_PAINT_COLOR = 1,
|
||||||
|
|
@ -151,7 +153,7 @@ typedef struct NSVGshape
|
||||||
char fillRule; // Fill rule, see NSVGfillRule.
|
char fillRule; // Fill rule, see NSVGfillRule.
|
||||||
unsigned char flags; // Logical or of NSVG_FLAGS_* flags
|
unsigned char flags; // Logical or of NSVG_FLAGS_* flags
|
||||||
float bounds[4]; // Tight bounding box of the shape [minx,miny,maxx,maxy].
|
float bounds[4]; // Tight bounding box of the shape [minx,miny,maxx,maxy].
|
||||||
const char *unicode; // Unicode character code.
|
char unicode[NSVG_MAX_UNICODE_LEN]; // Unicode character code.
|
||||||
int horizAdvX; // Horizontal distance to advance after rendering glyph.
|
int horizAdvX; // Horizontal distance to advance after rendering glyph.
|
||||||
NSVGpath* paths; // Linked list of paths in the image.
|
NSVGpath* paths; // Linked list of paths in the image.
|
||||||
struct NSVGshape* next; // Pointer to next shape, or NULL if last element.
|
struct NSVGshape* next; // Pointer to next shape, or NULL if last element.
|
||||||
|
|
@ -457,7 +459,7 @@ typedef struct NSVGparser
|
||||||
float dpi;
|
float dpi;
|
||||||
char pathFlag;
|
char pathFlag;
|
||||||
char defsFlag;
|
char defsFlag;
|
||||||
char *unicodeFlag;
|
char unicodeFlag[NSVG_MAX_UNICODE_LEN];
|
||||||
const char *horizAdvFlag;
|
const char *horizAdvFlag;
|
||||||
} NSVGparser;
|
} NSVGparser;
|
||||||
|
|
||||||
|
|
@ -977,7 +979,7 @@ static void nsvg__addShape(NSVGparser* p)
|
||||||
shape->opacity = attr->opacity;
|
shape->opacity = attr->opacity;
|
||||||
|
|
||||||
if (p->unicodeFlag) {
|
if (p->unicodeFlag) {
|
||||||
shape->unicode = p->unicodeFlag;
|
strcat(shape->unicode, p->unicodeFlag);
|
||||||
if (p->horizAdvFlag) {
|
if (p->horizAdvFlag) {
|
||||||
shape->horizAdvX = strtol(p->horizAdvFlag, &end, 10);
|
shape->horizAdvX = strtol(p->horizAdvFlag, &end, 10);
|
||||||
if (end == p->horizAdvFlag)
|
if (end == p->horizAdvFlag)
|
||||||
|
|
@ -986,7 +988,7 @@ static void nsvg__addShape(NSVGparser* p)
|
||||||
if (shape->horizAdvX == 0) {
|
if (shape->horizAdvX == 0) {
|
||||||
shape->horizAdvX = p->image->defaultHorizAdv;
|
shape->horizAdvX = p->image->defaultHorizAdv;
|
||||||
}
|
}
|
||||||
p->unicodeFlag = NULL;
|
p->unicodeFlag[0] = '\0';
|
||||||
p->horizAdvFlag = NULL;
|
p->horizAdvFlag = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -2254,8 +2256,8 @@ static void nsvg__parsePath(NSVGparser* p, const char** attr)
|
||||||
for (i = 0; attr[i]; i += 2) {
|
for (i = 0; attr[i]; i += 2) {
|
||||||
if (strcmp(attr[i], "d") == 0) {
|
if (strcmp(attr[i], "d") == 0) {
|
||||||
s = attr[i + 1];
|
s = attr[i + 1];
|
||||||
} else if (strcmp(attr[i], "unicode") == 0) {
|
} else if (strcmp(attr[i], "unicode") == 0
|
||||||
p->unicodeFlag = malloc(strlen(attr[i+1]));
|
&& strlen(attr[i+1]) < NSVG_MAX_UNICODE_LEN) {
|
||||||
strcpy(p->unicodeFlag, attr[i+1]);
|
strcpy(p->unicodeFlag, attr[i+1]);
|
||||||
} else if (strcmp(attr[i], "horiz-adv-x") == 0) {
|
} else if (strcmp(attr[i], "horiz-adv-x") == 0) {
|
||||||
p->horizAdvFlag = attr[i+1];
|
p->horizAdvFlag = attr[i+1];
|
||||||
|
|
@ -3032,7 +3034,7 @@ NSVGimage* nsvgParseFromFile(const char* filename, const char* units, float dpi)
|
||||||
error:
|
error:
|
||||||
if (fp) fclose(fp);
|
if (fp) fclose(fp);
|
||||||
if (data) free(data);
|
if (data) free(data);
|
||||||
if (image) nsvgDelete(image);
|
nsvgDelete(image);
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
15
include/pbsplash.h
Normal file
15
include/pbsplash.h
Normal file
|
|
@ -0,0 +1,15 @@
|
||||||
|
#ifndef __pbsplash_h__
|
||||||
|
#define __pbsplash_h__
|
||||||
|
|
||||||
|
struct col {
|
||||||
|
union {
|
||||||
|
unsigned int rgba;
|
||||||
|
struct {
|
||||||
|
unsigned char r, g, b, a;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
void animate_frame(int frame, int w, int y_off);
|
||||||
|
|
||||||
|
#endif
|
||||||
27
src/animate.c
Normal file
27
src/animate.c
Normal file
|
|
@ -0,0 +1,27 @@
|
||||||
|
#include <math.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <tfblib/tfblib.h>
|
||||||
|
#include <tfblib/tfb_colors.h>
|
||||||
|
#include "pbsplash.h"
|
||||||
|
|
||||||
|
struct col color = {.r = 255, .g = 255, .b = 255, .a = 255};
|
||||||
|
|
||||||
|
// FIXME: calculate constants based on display size/resolution
|
||||||
|
|
||||||
|
#define n_circles 5
|
||||||
|
#define amplitude 40
|
||||||
|
#define rad 12
|
||||||
|
|
||||||
|
void animate_frame(int frame, int w, int y_off)
|
||||||
|
{
|
||||||
|
unsigned int t_col = tfb_make_color(color.r, color.g, color.b);
|
||||||
|
for (unsigned int i = 0; i < n_circles; i++)
|
||||||
|
{
|
||||||
|
int c_dist = w * 0.05;
|
||||||
|
int x = i * c_dist + w / 2 - c_dist * n_circles / 2.f;
|
||||||
|
double s = sin(frame / 30.0 * 3.1415 + i * 0.5);
|
||||||
|
int y = y_off + s * amplitude;
|
||||||
|
tfb_fill_rect(x - rad- 1, y_off - amplitude - rad, rad* 2 + 2, 400 +rad* 2, tfb_black);
|
||||||
|
tfb_fill_circle(x, y, rad, t_col);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -1,5 +1,6 @@
|
||||||
src = [
|
src = [
|
||||||
'pbsplash.c',
|
'pbsplash.c',
|
||||||
|
'animate.c',
|
||||||
]
|
]
|
||||||
|
|
||||||
executable('pbsplash', src,
|
executable('pbsplash', src,
|
||||||
|
|
|
||||||
133
src/pbsplash.c
133
src/pbsplash.c
|
|
@ -1,6 +1,7 @@
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <signal.h>
|
#include <signal.h>
|
||||||
|
#include <time.h>
|
||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
#include <sys/stat.h>
|
#include <sys/stat.h>
|
||||||
#include <fcntl.h>
|
#include <fcntl.h>
|
||||||
|
|
@ -15,6 +16,8 @@
|
||||||
#define NANOSVGRAST_IMPLEMENTATION
|
#define NANOSVGRAST_IMPLEMENTATION
|
||||||
#include "nanosvgrast.h"
|
#include "nanosvgrast.h"
|
||||||
|
|
||||||
|
#include "pbsplash.h"
|
||||||
|
|
||||||
#define MSG_MAX_LEN 4096
|
#define MSG_MAX_LEN 4096
|
||||||
#define DEFAULT_FONT_PATH "/usr/share/pbsplash/OpenSans-Regular.svg"
|
#define DEFAULT_FONT_PATH "/usr/share/pbsplash/OpenSans-Regular.svg"
|
||||||
#define LOGO_SIZE_MAX_MM 90
|
#define LOGO_SIZE_MAX_MM 90
|
||||||
|
|
@ -22,11 +25,12 @@
|
||||||
#define PT_TO_MM 0.38f
|
#define PT_TO_MM 0.38f
|
||||||
#define TTY_PATH_LEN 11
|
#define TTY_PATH_LEN 11
|
||||||
|
|
||||||
#define DEBUGRENDER 1
|
#define DEBUGRENDER 0
|
||||||
|
|
||||||
volatile sig_atomic_t terminate = 0;
|
volatile sig_atomic_t terminate = 0;
|
||||||
|
|
||||||
bool debug = true;
|
bool debug = true;
|
||||||
|
struct col background_color = {.r = 0, .g = 0, .b = 0, .a = 255};
|
||||||
|
|
||||||
#define LOG(fmt, ...) do { if (debug) fprintf(stdout, fmt, ##__VA_ARGS__); } while (0)
|
#define LOG(fmt, ...) do { if (debug) fprintf(stdout, fmt, ##__VA_ARGS__); } while (0)
|
||||||
|
|
||||||
|
|
@ -51,18 +55,10 @@ void term(int signum)
|
||||||
terminate = 1;
|
terminate = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void blit_buf(unsigned char* buf, int x, int y, int w, int h, bool vflip, bool redraw)
|
||||||
// Blit an SVG to the framebuffer using tfblib. The image is
|
|
||||||
// scaled based on the target width and height.
|
|
||||||
static void draw_svg(NSVGimage *image, int x, int y, int w, int h)
|
|
||||||
{
|
{
|
||||||
float sz = (int)((float)w / (float)image->width * 100.f) / 100.f;
|
unsigned int bg_col = tfb_make_color(background_color.r, background_color.g, background_color.b);
|
||||||
LOG("%f\n", sz);
|
unsigned int col;
|
||||||
//sz = 1.5;
|
|
||||||
NSVGrasterizer *rast = nsvgCreateRasterizer();
|
|
||||||
unsigned char *img = malloc(w * h * 4);
|
|
||||||
nsvgRasterize(rast, image, 0, 0, sz, img, w, h, w * 4);
|
|
||||||
|
|
||||||
for (size_t i = 0; i < w; i++)
|
for (size_t i = 0; i < w; i++)
|
||||||
{
|
{
|
||||||
for (size_t j = 0; j < h; j++)
|
for (size_t j = 0; j < h; j++)
|
||||||
|
|
@ -73,15 +69,59 @@ static void draw_svg(NSVGimage *image, int x, int y, int w, int h)
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
unsigned int col = tfb_make_color(img[(j * w + i) * 4 + 0], img[(j * w + i) * 4 + 1], img[(j * w + i) * 4 + 2]);
|
struct col rgba = *(struct col *)(buf + (j * w + i) * 4);
|
||||||
|
if (!redraw && rgba.a == 0)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (redraw && rgba.a == 0) {
|
||||||
|
col = bg_col;
|
||||||
|
} else {
|
||||||
|
rgba.r = (rgba.r * rgba.a + background_color.r * (255 - rgba.a)) >> 8;
|
||||||
|
rgba.g = (rgba.g * rgba.a + background_color.g * (255 - rgba.a)) >> 8;
|
||||||
|
rgba.b = (rgba.b * rgba.a + background_color.b * (255 - rgba.a)) >> 8;
|
||||||
|
col = tfb_make_color(rgba.r, rgba.g, rgba.b);
|
||||||
|
}
|
||||||
|
if (vflip)
|
||||||
|
tfb_draw_pixel(x + i, y + h - j, col);
|
||||||
|
else
|
||||||
tfb_draw_pixel(x + i, y + j, col);
|
tfb_draw_pixel(x + i, y + j, col);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
free(img);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
static void draw_svg(NSVGimage *image, int x, int y, int w, int h)
|
||||||
|
{
|
||||||
|
float sz = (int)((float)w / (float)image->width * 100.f) / 100.f;
|
||||||
|
LOG("draw_svg: %dx%d, %dx%d, %f\n", x, y, w, h, sz);
|
||||||
|
//sz = 1.5;
|
||||||
|
NSVGrasterizer *rast = nsvgCreateRasterizer();
|
||||||
|
unsigned char *img = malloc(w * h * 4);
|
||||||
|
nsvgRasterize(rast, image, 0, 0, sz, img, w, h, w * 4);
|
||||||
|
|
||||||
|
blit_buf(img, x, y, w, h, false, false);
|
||||||
|
|
||||||
|
free(img);
|
||||||
|
nsvgDeleteRasterizer(rast);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void draw_text(NSVGimage *font, char *text, int x, int y, int width, int height, float scale, unsigned int tfb_col)
|
||||||
|
{
|
||||||
|
LOG("text '%s': fontsz=%f, x=%d, y=%d, dimensions: %d x %d\n", text,
|
||||||
|
scale, x, y, width, height);
|
||||||
|
NSVGshape **shapes = nsvgGetTextShapes(font, text, strlen(text));
|
||||||
|
unsigned char *img = malloc(width * height * 4);
|
||||||
|
NSVGrasterizer *rast = nsvgCreateRasterizer();
|
||||||
|
|
||||||
|
nsvgRasterizeText(rast, font, 0, 0, scale, img, width, height, width * 4, text);
|
||||||
|
|
||||||
|
blit_buf(img, x, y, width, height, true, false);
|
||||||
|
|
||||||
|
free(img);
|
||||||
|
free(shapes);
|
||||||
|
nsvgDeleteRasterizer(rast);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
* Get the dimensions of a string in pixels.
|
* Get the dimensions of a string in pixels.
|
||||||
* based on the font size and the font SVG file.
|
* based on the font size and the font SVG file.
|
||||||
*/
|
*/
|
||||||
|
|
@ -105,37 +145,8 @@ static void getTextDimensions(NSVGimage *font, char *text, float scale, int *wid
|
||||||
} else {
|
} else {
|
||||||
*width += font->defaultHorizAdv * scale;
|
*width += font->defaultHorizAdv * scale;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static void draw_text(NSVGimage *font, char *text, int x, int y, int width, int height, float scale, unsigned int tfb_col)
|
|
||||||
{
|
|
||||||
LOG("text '%s': fontsz=%f, x=%d, y=%d, dimensions: %d x %d\n", text,
|
|
||||||
scale, x, y, width, height);
|
|
||||||
NSVGshape **shapes = nsvgGetTextShapes(font, text, strlen(text));
|
|
||||||
unsigned char *img = malloc(width * height * 4);
|
|
||||||
NSVGrasterizer *rast = nsvgCreateRasterizer();
|
|
||||||
|
|
||||||
nsvgRasterizeText(rast, font, 0, 0, scale, img, width, height, width * 4, text);
|
|
||||||
|
|
||||||
for (size_t i = 0; i < width; i++)
|
|
||||||
{
|
|
||||||
for (size_t j = 0; j < height; j++)
|
|
||||||
{
|
|
||||||
#if DEBUGRENDER == 1
|
|
||||||
if (i == 0 || i == width - 1 || j == 0 || j == height-1) {
|
|
||||||
tfb_draw_pixel(x + i, y + height - j, tfb_red);
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
unsigned int col = tfb_make_color(img[(j * width + i) * 4 + 0], img[(j * width + i) * 4 + 1], img[(j * width + i) * 4 + 2]);
|
|
||||||
if (col != tfb_black)
|
|
||||||
tfb_draw_pixel(x + i, y + height - j, tfb_col);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
free(img);
|
|
||||||
free(shapes);
|
free(shapes);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -145,7 +156,7 @@ int main(int argc, char **argv)
|
||||||
char *message = NULL;
|
char *message = NULL;
|
||||||
char *splash_image = NULL;
|
char *splash_image = NULL;
|
||||||
char *font_path = DEFAULT_FONT_PATH;
|
char *font_path = DEFAULT_FONT_PATH;
|
||||||
char active_tty[TTY_PATH_LEN + 5];
|
char active_tty[TTY_PATH_LEN + 1];
|
||||||
NSVGimage *image;
|
NSVGimage *image;
|
||||||
NSVGimage *font;
|
NSVGimage *font;
|
||||||
struct sigaction action;
|
struct sigaction action;
|
||||||
|
|
@ -209,22 +220,19 @@ int main(int argc, char **argv)
|
||||||
|
|
||||||
{
|
{
|
||||||
FILE *fp = fopen("/sys/devices/virtual/tty/tty0/active", "r");
|
FILE *fp = fopen("/sys/devices/virtual/tty/tty0/active", "r");
|
||||||
char ch;
|
int len = strlen(active_tty);
|
||||||
|
char *ptr = active_tty + len;
|
||||||
if(fp != NULL)
|
if(fp != NULL)
|
||||||
{
|
{
|
||||||
while((ch = getc(fp)) != EOF && strlen(active_tty) < TTY_PATH_LEN)
|
fgets(ptr, TTY_PATH_LEN - len, fp);
|
||||||
{
|
*(ptr + strlen(ptr) - 1) = '\0';
|
||||||
if (ch == '\n')
|
|
||||||
break;
|
|
||||||
strcat(active_tty, &ch);
|
|
||||||
}
|
|
||||||
fclose(fp);
|
fclose(fp);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
LOG("active tty: '%s'\n", active_tty);
|
LOG("active tty: '%s'\n", active_tty);
|
||||||
|
|
||||||
if ((rc = tfb_acquire_fb(0, "/dev/fb0", active_tty)) != TFB_SUCCESS)
|
if ((rc = tfb_acquire_fb(/*TFB_FL_USE_DOUBLE_BUFFER*/ 0, "/dev/fb0", active_tty)) != TFB_SUCCESS)
|
||||||
{
|
{
|
||||||
fprintf(stderr, "tfb_acquire_fb() failed with error code: %d\n", rc);
|
fprintf(stderr, "tfb_acquire_fb() failed with error code: %d\n", rc);
|
||||||
rc = 1;
|
rc = 1;
|
||||||
|
|
@ -260,7 +268,7 @@ int main(int argc, char **argv)
|
||||||
|
|
||||||
LOG("%dx%d @ %dx%dmm, dpi=%ld, logo_size_px=%f\n", w, h, w_mm, h_mm, dpi, logo_size_px);
|
LOG("%dx%d @ %dx%dmm, dpi=%ld, logo_size_px=%f\n", w, h, w_mm, h_mm, dpi, logo_size_px);
|
||||||
|
|
||||||
image = nsvgParseFromFile(splash_image, "px", logo_size_px);
|
image = nsvgParseFromFile(splash_image, "", logo_size_px);
|
||||||
if (!image)
|
if (!image)
|
||||||
{
|
{
|
||||||
fprintf(stderr, "failed to load SVG image\n");
|
fprintf(stderr, "failed to load SVG image\n");
|
||||||
|
|
@ -277,7 +285,7 @@ int main(int argc, char **argv)
|
||||||
x -= image_w * 0.5f;
|
x -= image_w * 0.5f;
|
||||||
y -= image_h * 0.5f;
|
y -= image_h * 0.5f;
|
||||||
|
|
||||||
tfb_clear_screen(tfb_black);
|
tfb_clear_screen(tfb_make_color(background_color.r, background_color.g, background_color.b));
|
||||||
|
|
||||||
draw_svg(image, x, y, image_w, image_h);
|
draw_svg(image, x, y, image_w, image_h);
|
||||||
|
|
||||||
|
|
@ -306,11 +314,22 @@ int main(int argc, char **argv)
|
||||||
|
|
||||||
tfb_flush_window();
|
tfb_flush_window();
|
||||||
tfb_flush_fb();
|
tfb_flush_fb();
|
||||||
|
#define ANIM_HEIGHT 600
|
||||||
|
//unsigned char* animation_buf = malloc(w * ANIM_HEIGHT * 4);
|
||||||
|
int frame = 0;
|
||||||
while (!terminate)
|
while (!terminate)
|
||||||
{
|
{
|
||||||
sleep(1);
|
//printf("FRAME: %d\n", frame);
|
||||||
|
animate_frame(frame++, w, y + image_h + 500);
|
||||||
|
//blit_buf(animation_buf, 0, y + image_h + 200, w, ANIM_HEIGHT, false, true);
|
||||||
|
//memset(animation_buf, 0, w * ANIM_HEIGHT * 4);
|
||||||
|
//tfb_flush_window();
|
||||||
|
tfb_flush_fb();
|
||||||
|
//usleep(1666);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//free(animation_buf);
|
||||||
|
|
||||||
out:
|
out:
|
||||||
nsvgDelete(font);
|
nsvgDelete(font);
|
||||||
nsvgDelete(image);
|
nsvgDelete(image);
|
||||||
|
|
|
||||||
Loading…
Reference in a new issue