Skip to content

Commit 1b463f3

Browse files
committed
2023.11
new: game ui api (@zpl-zak) new: 3D sprites + ASE loader new: orthographic cameras new: curve api new: argvadd() new: game ui demo (@zpl-zak) new: integrated lite text editor new: lua scripts can load zipped lua modules now new: ui_unsigned2/3() new: rgbatoa(), atorgba() new: obj_menu(), obj_aabb() interfaces new: window_clipboard() / window_setclipboard() new: codepoint_to_utf8() new: tools/ase2ini new: --cook-stats new: FONT_EM emoji/symbols range new: SPRITE_FLAGS new: WINDOW_BORDERLESS chg: improved cook times chg: ase2ini: 9-slices support (@zpl-zak) chg: autocomplete, autoreload and some other plugins (editor_text) chg: add userdata field in texture_t chg: added scale Y support into sprite_rect() chg: adjusted horizontal spacing (-DUI_LESSER_SPACING) chg: adjusted oversampling & pixel snaps (-DUI_SMALL_FONTS) chg: allowed dll api to add the required extensions per platform chg: append cook commands into cook recipe: avoids global lock (fwk_cook) chg: avoid potential system() call global lock in osx builds with cook on demand enabled chg: built tools/ark, tools/ase2ini (linux, osx) chg: cleaned up curve api (thanks @zpl-zak!) chg: cleaned up dll api chg: cook optimizations chg: cosmetics tweaks in color pickers (fwk_ui) chg: do not batch transparent sprite_rect()s or without scale chg: editor: added extra icons on menu bar chg: ffmpeg will use a single thread now chg: generate csv cooking stats (MAKE cook) chg: increase va() buffer to 512 KiB per thread chg: integrate fwk_editor chg: lua 5.4.4 -> 5.4.6 chg: made larger pngs, faster writing (ase2ini, 3rd_atlas) chg: made xml_string to return empty strings on invalid nodes (@zpl-zak) chg: merged files id > obj, bt > ai, buffer > pack, dll+script > extend, tween > time chg: move tools/3rd -> engine/split/ chg: multi-cursors in lite editor chg: profiler disabled only in retail builds chg: rebuilt linux+osx binaries (ase2ini, ark, cook) chg: rebuilt tools/cook (win,linux) chg: reduce stack when calculating crc32 (zip) chg: remove extra allocation when compressing (ulz) chg: simplified ui_disable() theme logic (fwk_ui) chg: smoother progress bar animations at cook time chg: ui_keyboard() displaying FN keys chg: update docs chg: update roadmap chg: updated nothings' 3rd libs (vorbis 1.19->1.22, image 2.26->2.28, image_write 1.15->1.16) brk: app_exec() not returning popen() log anymore brk: app_spawn() return codes brk: changed signature ui_color3/4() brk: changed sprite_rect() signature brk: changed ui_tileset/ui_tiled/ui_spine signatures brk: CURSOR_SHAPE enums reordered brk: renamed some KEY identifiers so they match their eval() counterparts brk: sort_64() > less_64_ptr(), sort_int() > less_int_ptr() brk: sprite api signatures brk: ui_debug() > ui_engine() fix: app_beep(), app_open_folder(), app_open_file() framerate stalls fix: app_spawn() not launching background sub-processes fix: ase2ini: heap dealloc crash fix: billboards (@zpl-zak) fix: changed default editor hz to monitor rate (@zpl-zak) fix: color3/4f() no longer clamp components in LDR range [0..255] fix: crash in font module when loading MDI font fix: cuttlefish was never intended to convert images into png files fix: do intern some previously temporary fields (fwk_reflect) fix: do not inscribe members twice (fwk_reflect) fix: do not process mouse input when mouse is hidden (3rd_nuklear_glfw3) fix: do not recurse structs indefinitely (fwk_reflect) fix: editor: rect selections could happen while fps cam was active fix: file_list("**/file") not returning existing entries fix: fixed font_face_from_mem() to use vfs api when loading shaders fix: fixed regression that would open terminal windows in retail builds (@zpl-zak) fix: fixed window_cursor()/window_has_cursor() when using CURSOR_SW_AUTO shape fix: fx_load() not returning existing postfxs in cook-on-demand builds fix: ini() not reading vfs files fix: internal base64 decoder not expanding to correct sizes fix: invalid casting in some recently introduced datatypes fix: memory stomp during cook (strsplit) fix: mutexes in obj_meta()/obj_setmeta() fix: normalize b64 de/encoders to use url safe variant fix: normalize fread/fwrite calls fix: osx compilation errors (spline, ase2ini) fix: png,jpg images were being compressed even if explicitly told cook to not doing so fix: regression: --cook-on-demand fix: regression: input boxes not working since lite was integrated fix: regression: invalid script generated when shell commands were present (thanks @zpl-zak!) fix: regression: ui_colors() not being dimmed when ui was disabled fix: silicon apple m1 fixes (@zpl-zak) fix: sprite_rect() not scaling X properly fix: string32() not decoding 4-byte utf8 codepoints fix: timer() api will retrieve epoch/freq at least once per thread now fix: unify RGB3/RGB4 macros to use RGBA colors everywhere: ddraw, palette, tty, fonts, window fix: windows overlapping menubars created with internal menu v2 api fix: wrong terminal font size lab: editor: setmouse() lab: editor: toolbar icons lab: integrate 99-nodes into editor lab: integrated rpmalloc lab: removed cook progress messages, which were using a global lock lab: script v2 api https://imgur.com/a/98J5woT https://imgur.com/a/1c2Wmst
1 parent 688fe63 commit 1b463f3

File tree

1,768 files changed

+2066458
-0
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

1,768 files changed

+2066458
-0
lines changed

2023.11/MAKE.bat

Lines changed: 905 additions & 0 deletions
Large diffs are not rendered by default.

2023.11/README.md

Lines changed: 430 additions & 0 deletions
Large diffs are not rendered by default.

2023.11/demos/00-loop.c

Lines changed: 9 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
#include "fwk.h" // Minimal C sample
2+
3+
int main() {
4+
window_create(75.0, 0); // 75% size, no extra flags
5+
6+
while( window_swap() && !input(KEY_ESC) ) { // game loop
7+
puts("hello FWK from C!");
8+
}
9+
}

2023.11/demos/00-script.c

Lines changed: 31 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,31 @@
1+
#include "fwk.h"
2+
3+
#define SCRIPT(...) #__VA_ARGS__
4+
5+
#if 0 // teal
6+
script_run("local tl=require(\"tl\")\ntl.loader()");
7+
script_run("addsub = require(\"s2\"); print (addsub.add(10, 20))");
8+
s2.tl:
9+
local function add(a: number, b: number): number
10+
return a + b
11+
end
12+
local s = add(1,2)
13+
print(s)
14+
#endif
15+
16+
int main() {
17+
script_init();
18+
19+
script_run(SCRIPT(
20+
window_create(75.0,0);
21+
while window_swap() do
22+
ddraw_grid(10);
23+
if ui_panel("Hello from Lua!", 0) then
24+
ui_panel_end();
25+
end;
26+
end;
27+
));
28+
29+
// script test (lua)
30+
script_run( "-- Bye.lua\nio.write(\"script test: Bye world!, from \", _VERSION, \"\\n\")" );
31+
}

2023.11/demos/01-demo2d.c

Lines changed: 155 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,155 @@
1+
// 2d demo: window, audio, camera, font, tiled, render, fx, spritesheet, input, ui. @todo: spine
2+
// - rlyeh, public domain.
3+
//
4+
// Compile with:
5+
// `make demos\01-demo2d.c` (windows)
6+
// `sh MAKE.bat demos/01-demo2d.c` (linux, osx)
7+
8+
#include "fwk.h"
9+
10+
void demo_kids(vec3 offs) {
11+
// init
12+
static texture_t kids; do_once kids = texture( "spriteSheetExample.png", TEXTURE_LINEAR );
13+
static vec3 pos[2] = {{490,362},{442,362}}, vel[2] = {0};
14+
static int row[2] = {0,3}, frame[2] = {0};
15+
static int inputs[2][4] = {{KEY_W,KEY_A,KEY_S,KEY_D},{KEY_UP,KEY_LEFT,KEY_DOWN,KEY_RIGHT}};
16+
17+
// move
18+
for( int i = 0; i < countof(pos); ++i ) {
19+
vel[i].x = input(inputs[i][3]) - input(inputs[i][1]);
20+
vel[i].y = input(inputs[i][2]) - input(inputs[i][0]);
21+
pos[i].x = fmod(pos[i].x + vel[i].x, window_width() + 128);
22+
pos[i].y = fmod(pos[i].y + vel[i].y, window_height() + 128);
23+
frame[i] += vel[i].x || vel[i].y;
24+
}
25+
26+
// render
27+
for( int i = 0; i < countof(pos); ++i ) {
28+
int col = frame[i] / 10, num_frame = row[i] * 4 + col % 4; // 4x4 tilesheet
29+
float position[3] = {pos[i].x,pos[i].y,pos[i].y}, offset[2]={0,0}, scale[2]={0.5,0.5};
30+
float spritesheet[3]={num_frame,4,4};
31+
sprite_sheet(kids,
32+
spritesheet, // num_frame in a 4x4 spritesheet
33+
position, 0, // position(x,y,depth:sort-by-Y), angle
34+
offset, scale, // offset(x,y), scale(x,y)
35+
WHITE, 0 // tint_color, no flags
36+
);
37+
}
38+
}
39+
40+
void demo_hud() {
41+
// draw pixel-art hud, 16x16 ui element, scaled and positioned in resolution-independant way
42+
static texture_t inputs; do_once inputs = texture( "prompts_tilemap_34x24_16x16x1.png", TEXTURE_LINEAR );
43+
float spritesheet[3] = {17,34,24}, offset[2] = {0, - 2*absf(sin(window_time()*5))}; // sprite cell and animation
44+
float scale[2] = {3, 3}, tile_w = 16 * scale[0], tile_h = 16 * scale[1]; // scaling
45+
float position[3] = {window_width() - tile_w, window_height() - tile_h, window_height() }; // position in screen-coordinates (x,y,z-index)
46+
sprite_sheet(inputs, spritesheet, position, 0/*deg*/, offset, scale, WHITE, SPRITE_RESOLUTION_INDEPENDANT);
47+
}
48+
49+
int main() {
50+
// window (80% sized, MSAA x4 flag)
51+
window_create(80.0, WINDOW_MSAA4);
52+
window_title(__FILE__);
53+
54+
// tiled map
55+
tiled_t tmx = tiled(vfs_read("castle.tmx"));
56+
// tmx.parallax = true;
57+
58+
// spine model
59+
//spine_t *spn = spine("goblins.json", "goblins.atlas", 0);
60+
61+
// camera 2d
62+
camera_t cam = camera();
63+
cam.position = vec3(window_width()/2, window_height()/2, 3); // at(CX,CY) zoom(x3)
64+
camera_enable(&cam);
65+
66+
// audio (both clips & streams)
67+
audio_t clip1 = audio_clip( "coin.wav" );
68+
audio_t clip2 = audio_clip( "pew.sfxr" );
69+
audio_t stream1 = audio_stream( "larry.mid" );
70+
audio_t stream2 = audio_stream( "waterworld-map.fur" );
71+
audio_t BGM = stream1;
72+
audio_play(BGM, 0);
73+
74+
// font config: faces (6 max) and colors (10 max)
75+
#define FONT_CJK FONT_FACE3
76+
#define FONT_YELLOW FONT_COLOR2
77+
#define FONT_LIME FONT_COLOR3
78+
font_face(FONT_CJK, "mplus-1p-medium.ttf", 48.f, FONT_JP|FONT_2048); // CJK|FONT_2048|FONT_OVERSAMPLE_Y);
79+
font_color(FONT_YELLOW, RGB4(255,255,0,255));
80+
font_color(FONT_LIME, RGB4(128,255,0,255));
81+
82+
// fx: load all post fx files in all subdirs. enable a few filters by default
83+
fx_load("fx**.fs");
84+
fx_enable(fx_find("fxCRT2.fs"), 1);
85+
fx_enable(fx_find("fxGrain.fs"), 1);
86+
fx_enable(fx_find("fxContrast.fs"), 1);
87+
fx_enable(fx_find("fxVignette.fs"), 1);
88+
89+
// demo loop
90+
while (window_swap() && !input_down(KEY_ESC)) {
91+
92+
// handle input
93+
if( input_down(KEY_F5) ) window_reload();
94+
if( input_down(KEY_F11) ) window_fullscreen( !window_has_fullscreen() );
95+
96+
// camera panning (x,y) & zooming (z)
97+
if( !ui_hover() && !ui_active() ) {
98+
if( input(MOUSE_L) ) cam.position.x += input_diff(MOUSE_X);
99+
if( input(MOUSE_L) ) cam.position.y += input_diff(MOUSE_Y);
100+
cam.position.z += input_diff(MOUSE_W) * 0.1;
101+
}
102+
103+
// apply post-fxs from here
104+
fx_begin();
105+
106+
profile("Rendering") {
107+
vec3 center = add3(cam.position,vec3(-window_width()/1,-window_height()/2,0));
108+
// render tiled map
109+
tiled_render(tmx, center);
110+
//
111+
demo_kids(vec3(0,0,1));
112+
demo_hud();
113+
// render spine model
114+
// spine_animate(spn, !window_has_pause() * window_delta());
115+
// spine_render(spn, vec3(cam.position.x, cam.position.y, 1), true );
116+
// sprite_flush();
117+
}
118+
119+
// subtitle sample
120+
font_print(
121+
FONT_BOTTOM FONT_CENTER
122+
FONT_CJK FONT_H1
123+
FONT_YELLOW "私はガラスを食べられます。" FONT_LIME "それは私を傷つけません。\n"
124+
);
125+
126+
// post-fxs end here
127+
fx_end();
128+
129+
// ui
130+
if( ui_panel("Audio", 0)) {
131+
if( ui_label2_toolbar("BGM: Track #1", ICON_MD_VOLUME_UP)) audio_stop(BGM), audio_play(BGM = stream1, AUDIO_SINGLE_INSTANCE);
132+
if( ui_label2_toolbar("BGM: Track #2", ICON_MD_VOLUME_UP)) audio_stop(BGM), audio_play(BGM = stream2, AUDIO_SINGLE_INSTANCE);
133+
if( ui_label2_toolbar("SFX: Coin", ICON_MD_VOLUME_UP)) audio_play(clip1, 0);
134+
if( ui_label2_toolbar("SFX: Pew", ICON_MD_VOLUME_UP)) audio_play(clip2, 0);
135+
ui_panel_end();
136+
}
137+
if( ui_panel("Tiled", 0)) {
138+
ui_float("Zoom in", &cam.position.z);
139+
ui_tiled(&tmx);
140+
ui_panel_end();
141+
}
142+
/*if( ui_panel("Spine", 0)) {
143+
ui_spine(spn);
144+
ui_panel_end();
145+
}*/
146+
}
147+
}
148+
149+
// this demo supersedes following old sources:
150+
// https://github.com/r-lyeh/FWK/blob/45e34d7890b2b8fe1f4994f4b76e496280d83cb6/demos/00-audio.c
151+
// https://github.com/r-lyeh/FWK/blob/45e34d7890b2b8fe1f4994f4b76e496280d83cb6/demos/00-font.c
152+
// https://github.com/r-lyeh/FWK/blob/45e34d7890b2b8fe1f4994f4b76e496280d83cb6/demos/00-spine.c
153+
// https://github.com/r-lyeh/FWK/blob/45e34d7890b2b8fe1f4994f4b76e496280d83cb6/demos/00-sprite.c
154+
// https://github.com/r-lyeh/FWK/blob/45e34d7890b2b8fe1f4994f4b76e496280d83cb6/demos/00-tiled.c
155+
// https://github.com/r-lyeh/FWK/blob/45e34d7890b2b8fe1f4994f4b76e496280d83cb6/demos/00-tilemap.c

2023.11/demos/01-easing.c

Lines changed: 58 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,58 @@
1+
#include "fwk.h"
2+
3+
struct {
4+
float (*ease)(float);
5+
const char *name;
6+
} easings[] = {
7+
{ease_nop, "ease_nop"},
8+
{ease_linear, "ease_linear"},
9+
{ease_out_sine, "ease_out_sine"},
10+
{ease_out_quad, "ease_out_quad"},
11+
{ease_out_cubic, "ease_out_cubic"},
12+
{ease_out_quart, "ease_out_quart"},
13+
{ease_out_quint, "ease_out_quint"},
14+
{ease_out_expo, "ease_out_expo"},
15+
{ease_out_circ, "ease_out_circ"},
16+
{ease_out_back, "ease_out_back"},
17+
{ease_out_elastic, "ease_out_elastic"},
18+
{ease_out_bounce, "ease_out_bounce"},
19+
{ease_in_sine, "ease_in_sine"},
20+
{ease_in_quad, "ease_in_quad"},
21+
{ease_in_cubic, "ease_in_cubic"},
22+
{ease_in_quart, "ease_in_quart"},
23+
{ease_in_quint, "ease_in_quint"},
24+
{ease_in_expo, "ease_in_expo"},
25+
{ease_in_circ, "ease_in_circ"},
26+
{ease_in_back, "ease_in_back"},
27+
{ease_in_elastic, "ease_in_elastic"},
28+
{ease_in_bounce, "ease_in_bounce"},
29+
{ease_inout_sine, "ease_inout_sine"},
30+
{ease_inout_quad, "ease_inout_quad"},
31+
{ease_inout_cubic, "ease_inout_cubic"},
32+
{ease_inout_quart, "ease_inout_quart"},
33+
{ease_inout_quint, "ease_inout_quint"},
34+
{ease_inout_expo, "ease_inout_expo"},
35+
{ease_inout_circ, "ease_inout_circ"},
36+
{ease_inout_back, "ease_inout_back"},
37+
{ease_inout_elastic, "ease_inout_elastic"},
38+
{ease_inout_bounce, "ease_inout_bounce"},
39+
{ease_inout_perlin, "ease_inout_perlin"},
40+
};
41+
42+
int main() {
43+
window_create(0.75, WINDOW_SQUARE);
44+
while(window_swap()) {
45+
static double timer = 0; timer = fmod(timer+window_delta(), 2); // loops every 2s
46+
47+
static int open = 1;
48+
if( ui_window("ease", &open) ) {
49+
float linear_delta = timer / 2.f; // delta is [0..1]
50+
for( int i = 0; i < countof(easings); ++i) {
51+
float nonlinear_delta = easings[i].ease(linear_delta);
52+
// visualize
53+
ui_slider( easings[i].name, &nonlinear_delta );
54+
}
55+
ui_window_end();
56+
}
57+
}
58+
}

2023.11/demos/01-font.c

Lines changed: 114 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,114 @@
1+
#include "fwk.h"
2+
3+
int main() {
4+
window_create(75.0, WINDOW_MSAA8);
5+
6+
// style: our aliases
7+
#define FONT_REGULAR FONT_FACE1
8+
#define FONT_ITALIC FONT_FACE2
9+
#define FONT_BOLD FONT_FACE3
10+
#define FONT_JAPANESE FONT_FACE4
11+
#define FONT_MONOSPACE FONT_FACE5
12+
13+
#define FONT_GRAY FONT_COLOR2
14+
#define FONT_ORANGE FONT_COLOR3
15+
#define FONT_LIME FONT_COLOR4
16+
#define FONT_GREEN FONT_COLOR5
17+
#define FONT_CYAN FONT_COLOR6
18+
19+
#define FONT_LARGEST FONT_H1
20+
#define FONT_LARGE FONT_H2
21+
#define FONT_MEDIUM FONT_H3
22+
#define FONT_NORMAL FONT_H4
23+
#define FONT_SMALL FONT_H5
24+
#define FONT_TINY FONT_H6
25+
26+
// style: atlas size, unicode ranges and font faces (up to 6 faces)
27+
font_face(FONT_REGULAR, "B612""-Regular.ttf", 48.f, FONT_EU|FONT_AR|FONT_RU|FONT_2048);
28+
font_face(FONT_ITALIC, "B612""-Italic.ttf", 48.f, FONT_EU|FONT_AR|FONT_RU|FONT_2048);
29+
font_face(FONT_BOLD, "B612""-Bold.ttf", 48.f, FONT_EU|FONT_AR|FONT_RU|FONT_2048);
30+
font_face(FONT_JAPANESE, "mplus-1p-medium.ttf", 48.f, FONT_JP|FONT_2048); // CJK|FONT_2048|FONT_OVERSAMPLE_Y);
31+
font_face(FONT_MONOSPACE, "B612Mono""-Regular.ttf", 24.f, FONT_EU|FONT_512);
32+
33+
// style: colors (up to 10 colors)
34+
font_color(FONT_GRAY, RGB4(100,100,100,255));
35+
font_color(FONT_ORANGE, RGB4(255,192,0,255));
36+
font_color(FONT_LIME, RGB4(192,255,0,255));
37+
font_color(FONT_GREEN, RGB4(0,255,192,255));
38+
font_color(FONT_CYAN, RGB4(0,192,255,255));
39+
40+
// prepare color highlighting for following code snippet
41+
const char *source =
42+
FONT_MONOSPACE FONT_LARGEST
43+
"int main(int argc, char **argv) {\n"
44+
" for( int i = 0; i < 10; ++i)\n"
45+
" puts(\"hello world\");\n"
46+
" return 0;\n"
47+
"}\n";
48+
const void *colors = font_colorize(source, "void,int,char", "if,else,for,do,while,return,switch,case,break,default,");
49+
50+
// read input file with strings to display
51+
array(char*) lines = 0;
52+
for each_substring( vfs_read("pangrams.txt"), "\r\n", it ) array_push(lines, STRDUP(it));
53+
54+
// demo loop
55+
while( window_swap() && !input(KEY_ESC) ) {
56+
ddraw_grid(0);
57+
58+
// initial spacing
59+
font_goto(0, 50);
60+
61+
// print a code snippet with syntax highlighting
62+
font_highlight(source, colors);
63+
64+
// print a few strings with markup codes
65+
font_print(
66+
FONT_REGULAR
67+
FONT_LARGEST FONT_GRAY "The quick "
68+
FONT_LARGE FONT_LIME "brown "
69+
FONT_MEDIUM FONT_GRAY "fox "
70+
FONT_NORMAL "jumps over "
71+
FONT_SMALL "the lazy "
72+
FONT_TINY "dog.\n");
73+
74+
font_print(
75+
FONT_REGULAR FONT_LARGE FONT_CYAN
76+
"Now is the time for all " FONT_ITALIC "good men " FONT_REGULAR "to come to the aid of " FONT_BOLD "the party.\n");
77+
78+
font_print(
79+
FONT_ITALIC FONT_LARGE FONT_GREEN
80+
"Ég get etið gler án þess að meiða mig!\n");
81+
82+
font_print(
83+
FONT_BOLD FONT_LARGE FONT_ORANGE
84+
"Эх, чужак! Общий съём цен шляп (юфть)—вдрызг!.\n");
85+
86+
font_print(
87+
FONT_JAPANESE
88+
"私はガラスを食べられます。それは私を傷つけません。\n");
89+
90+
font_print( "This text ");
91+
font_print( "should display concatenated, ");
92+
font_print( "as there are no linefeeds.\n" );
93+
94+
// i18n: pangrams.txt file, line browser
95+
static int counter = 0;
96+
counter += input_down(KEY_RIGHT)-input_down(KEY_LEFT);
97+
counter += counter < 0 ? array_count(lines) : 0;
98+
font_print( va("<< %s >>\n", lines[counter % array_count(lines)]) );
99+
100+
// this does not work yet. you cant chain alignments yet...
101+
//font_print(FONT_TOP "Top" FONT_MIDDLE "Middle" FONT_BASELINE "Baseline" FONT_BOTTOM "Bottom\n");
102+
//font_print(FONT_LEFT "Left" FONT_CENTER "Center" FONT_RIGHT "Right\n");
103+
104+
// ... alignment must be the first tag in a string for now. this is a temporary hack.
105+
font_print(FONT_LEFT "left");
106+
font_print(FONT_CENTER "center");
107+
font_print(FONT_RIGHT "right");
108+
109+
font_print(FONT_TOP FONT_CENTER "top\n");
110+
font_print(FONT_MIDDLE FONT_RIGHT "middle\n");
111+
font_print(FONT_BASELINE FONT_RIGHT "baseline\n");
112+
font_print(FONT_BOTTOM FONT_CENTER "bottom\n");
113+
}
114+
}

0 commit comments

Comments
 (0)