ref: 1ae61e6b64ff05a50525e05e6782ad49be1e37a4
dir: /sys/src/cmd/fontsel.c/
#include <u.h>
#include <libc.h>
#include <thread.h>
#include <draw.h>
#include <bio.h>
#include <keyboard.h>
#include <mouse.h>
#define MIN(a,b) ((a)<=(b)?(a):(b))
#define MAX(a,b) ((a)>=(b)?(a):(b))
typedef struct Fontdir Fontdir;
struct Fontdir {
char *name;
char *prefix;
char **fonts;
int nfonts;
int isttf;
union {
int ifont;
int sz;
};
};
enum
{
Ckey,
Cmouse,
Cresize,
Numchan,
Ttfdefsz = 16,
};
static char *textdef[] = {
"Cwm fjord bank glyphs vext quiz!",
"Gud hjälpe Zorns mö qvickt få byxa?",
"Разъярённый чтец эгоистично бьёт пятью",
"жердями шустрого фехтовальщика.",
"",
"static int",
"dosomestuff(Stuff *s, char *t, int n)",
"{",
" if(s == nil && (t == nil || n < 1))",
" return 0;",
" fprint(2, \"# %c\\n\", t[0]);",
" return dostuff(s) || dot(t, n);",
"}",
nil
};
static char **text = textdef;
static char *prefixes[] = {
"/lib/font/bit",
"/lib/font/ttf",
};
static Font *f;
static Fontdir *dirs, *cdir;
static int ndirs, idir;
static char lasterr[256];
int mainstacksize = 32768;
static void
redraw(void)
{
Point p;
int i, w, maxw;
char t[256];
lockdisplay(display);
draw(screen, screen->r, display->white, nil, ZP);
p = screen->r.min;
if(f == nil){
p.x += Dx(screen->r)/2 - stringwidth(font, lasterr)/2;
p.y += Dy(screen->r)/2 - font->height/2;
string(screen, p, display->black, ZP, font, lasterr);
}else{
maxw = 0;
for(i = 0; text[i] != nil; i++){
if((w = stringwidth(f, text[i])) > maxw)
maxw = w;
}
p.x += Dx(screen->r)/2 - maxw/2;
p.y += Dy(screen->r)/2 - i*f->height/2;
for(i = 0; text[i] != nil; i++){
string(screen, p, display->black, ZP, f, text[i]);
p.y += f->height;
}
}
if(cdir->isttf)
snprint(t, sizeof(t), "/n/ttf/%s.%d/font", cdir->name, cdir->sz);
else
snprint(t, sizeof(t), "%s/%s", cdir->name, cdir->fonts[cdir->ifont]);
p = screen->r.max;
p.x -= Dx(screen->r)/2 + stringwidth(font, t)/2;
p.y -= font->height;
string(screen, p, display->black, ZP, font, t);
flushimage(display, 1);
unlockdisplay(display);
}
static int
fcmp(void *a, void *b)
{
return strcmp(*(char**)a, *(char**)b);
}
static int
fontdir(char *t, int f, Fontdir *fdir)
{
Dir *d;
int doff, k;
long i, n;
k = strlen(t);
if(k > 4 && (cistrcmp(&t[k-4], ".ttf") == 0 || cistrcmp(&t[k-4], ".otf") == 0)){
fdir->nfonts = 1;
fdir->sz = Ttfdefsz;
fdir->isttf = 1;
return 0;
}
if((n = dirreadall(f, &d)) < 1)
return -1;
doff = strlen(t);
t[doff++] = '/';
for(i = 0; i < n; i++){
if((k = strlen(d[i].name)) < 5 || strcmp(&d[i].name[k-5], ".font") != 0)
continue;
if((fdir->fonts = realloc(fdir->fonts, sizeof(*fdir->fonts)*(fdir->nfonts+1))) == nil)
sysfatal("no memory");
d[i].name[k-5] = 0;
strcpy(t+doff, d[i].name);
fdir->fonts[fdir->nfonts++] = strdup(t+doff);
}
free(d);
if(fdir->nfonts > 0)
qsort(fdir->fonts, fdir->nfonts, sizeof(*fdir->fonts), fcmp);
return 0;
}
static int
dcmp(void *a_, void *b_)
{
Fontdir *a, *b;
a = (Fontdir*)a_;
b = (Fontdir*)b_;
return strcmp(a->name, b->name);
}
static void
findfonts(char *prefix)
{
Dir *d, *din;
int f, fin, doff;
long i, n;
char t[1024];
doff = sprint(t, "%s", prefix);
t[doff++] = '/';
t[doff] = 0;
if((f = open(t, OREAD)) < 0){
fprint(2, "font dir: %r\n");
return;
}
if((n = dirreadall(f, &d)) < 1){
fprint(2, "%s: no fonts\n", t);
close(f);
return;
}
close(f);
for(i = 0; i < n; i++){
strcpy(t+doff, d[i].name);
if((fin = open(t, OREAD)) < 0)
continue;
if((din = dirfstat(fin)) != nil){
if((dirs = realloc(dirs, sizeof(Fontdir)*(ndirs+1))) == nil)
sysfatal("no memory");
memset(&dirs[ndirs], 0, sizeof(Fontdir));
dirs[ndirs].prefix = prefix;
if(fontdir(t, fin, &dirs[ndirs]) == 0 && dirs[ndirs].nfonts > 0)
dirs[ndirs++].name = strdup(d[i].name);
free(din);
}
close(fin);
}
free(d);
}
static void
newfont(void)
{
char t[512];
lockdisplay(display);
if(f != nil)
freefont(f);
if(cdir->isttf)
snprint(t, sizeof(t), "/n/ttf/%s.%d/font", cdir->name, cdir->sz);
else
snprint(t, sizeof(t), "%s/%s/%s.font", cdir->prefix, cdir->name, cdir->fonts[cdir->ifont]);
if((f = openfont(display, t)) == nil)
snprint(lasterr, sizeof(lasterr), "%r");
unlockdisplay(display);
}
static char *
dirgen(int i)
{
return i < ndirs ? dirs[i].name : nil;
}
static char *
fontgen(int i)
{
return i < cdir->nfonts ? cdir->fonts[i] : nil;
}
static void
usage(void)
{
print("usage: %s [FILE]\n", argv0);
threadexitsall("usage");
}
static void
loadtext(int f)
{
Biobuf b;
int i;
if(f < 0 || Binit(&b, f, OREAD) != 0)
sysfatal("loadtext: %r");
text = nil;
for(i = 0; i < 256; i++){
if((text = realloc(text, (i+1)*sizeof(char*))) == nil)
sysfatal("memory");
if((text[i] = Brdstr(&b, '\n', 1)) == nil)
break;
}
close(f);
}
void
threadmain(int argc, char **argv)
{
Mousectl *mctl;
Keyboardctl *kctl;
Rune r;
Mouse m;
Menu menu;
Alt a[Numchan+1] = {
[Ckey] = {nil, &r, CHANRCV},
[Cmouse] = {nil, &m, CHANRCV },
[Cresize] = {nil, nil, CHANRCV},
{nil, nil, CHANEND},
};
int n;
ARGBEGIN{
default:
usage();
break;
}ARGEND;
if(argc > 1)
usage();
else if(argc == 1)
loadtext(strcmp(argv[0], "-") == 0 ? 0 : open(argv[0], OREAD));
for(n = 0; n < nelem(prefixes); n++)
findfonts(prefixes[n]);
if(ndirs < 1)
sysfatal("no fonts");
qsort(dirs, ndirs, sizeof(*dirs), dcmp);
if(initdraw(nil, nil, "fontsel") < 0)
sysfatal("initdraw: %r");
if((kctl = initkeyboard(nil)) == nil)
sysfatal("initkeyboard: %r");
a[Ckey].c = kctl->c;
if((mctl = initmouse(nil, screen)) == nil)
sysfatal("initmouse: %r");
a[Cmouse].c = mctl->c;
a[Cresize].c = mctl->resizec;
display->locking = 1;
unlockdisplay(display);
memset(&menu, 0, sizeof(menu));
cdir = &dirs[0];
newfont();
redraw();
for(;;){
switch(alt(a)){
case -1:
goto end;
case Ckey:
switch (r) {
case Kdel:
case 'q':
goto end;
case '-':
cdir->ifont = MAX(cdir->isttf ? 6 : 0, cdir->ifont-1);
newfont();
redraw();
break;
case '+':
cdir->ifont = MIN(cdir->isttf ? 256 : cdir->nfonts-1, cdir->ifont+1);
newfont();
redraw();
break;
}
break;
case Cmouse:
if(m.buttons == 2){
menu.gen = dirgen;
menu.lasthit = idir;
if((n = menuhit(2, mctl, &menu, nil)) >= 0){
idir = n;
cdir = &dirs[idir];
newfont();
redraw();
}
}else if(m.buttons == 4 && cdir->isttf == 0){
menu.gen = fontgen;
menu.lasthit = cdir->ifont;
if((n = menuhit(3, mctl, &menu, nil)) >= 0){
cdir->ifont = n;
newfont();
redraw();
}
}
break;
case Cresize:
getwindow(display, Refnone);
redraw();
break;
}
}
end:
if(f != nil){
if(cdir->isttf)
print("/n/ttf/%s.%d/font\n", cdir->name, cdir->sz);
else
print("%s/%s/%s.font\n", cdir->prefix, cdir->name, cdir->fonts[cdir->ifont]);
}
threadexitsall(nil);
}