ref: d45bc8ef8a28ff69dd1798be2c2e1fc36d307771
dir: /zuke.c/
#include <u.h>
#include <libc.h>
#include <draw.h>
#include <mouse.h>
#include <keyboard.h>
#include <thread.h>
#include <ctype.h>
#include "plist.h"
typedef struct Player Player;
enum
{
Cstart = 1,
Cstop,
Ctoggle,
Cforward,
Cbackward,
Cforwardfast,
Cbackwardfast,
Everror = 1,
Evready,
Bps = 44100*2*2, /* 44100KHz, stereo, u16 for a sample */
Seekbytes = Bps*10, /* 10 seconds */
Seekbytesfast = Bps*60, /* 1 minute */
Scrollwidth = 12,
Scrollheight = 16,
Relbufsz = Bps/5, /* 0.2 second */
};
static Meta *pl;
static int plnum;
static char *plraw;
static int plrawsize;
struct Player
{
Channel *ctl;
Channel *ev;
Channel *img;
int pcur;
};
int mainstacksize = 32768;
static int audio;
static u64int byteswritten;
static int pcur, pcurplaying;
static int scroll, scrollsz;
static Image *cola, *colb;
static Font *f;
static Image *cover;
static Channel *ev;
static Mousectl *mctl;
static Keyboardctl *kctl;
static int entering;
static int colwidth[3];
static int mincolwidth[3];
#pragma varargck type "P" int
static int
positionfmt(Fmt *f)
{
char *s, tmp[16];
u64int sec;
s = tmp;
sec = va_arg(f->args, int);
if(sec >= 3600){
s = seprint(s, tmp+sizeof(tmp), "%02lld:", sec/3600);
sec %= 3600;
}
s = seprint(s, tmp+sizeof(tmp), "%02lld:", sec/60);
sec %= 60;
seprint(s, tmp+sizeof(tmp), "%02lld", sec);
return fmtstrcpy(f, tmp);
}
static void
adjustcolumns(void)
{
int i, x, total;
if(mincolwidth[0] == 0){
mincolwidth[0] = mincolwidth[1] = mincolwidth[2] = 1;
for(i = 0; i < plnum; i++){
if((x = stringwidth(f, pl[i].artist[0])) > mincolwidth[0])
mincolwidth[0] = x;
if((x = stringwidth(f, pl[i].album)) > mincolwidth[1])
mincolwidth[1] = x;
if((x = stringwidth(f, pl[i].title)) > mincolwidth[2])
mincolwidth[2] = x;
}
}
total = mincolwidth[0] + mincolwidth[1] + mincolwidth[2];
for(i = 0; i < nelem(mincolwidth); i++)
colwidth[i] = (Dx(screen->r) - 8) * mincolwidth[i] / total;
}
static void
redraw(Image *screen, int new)
{
Image *col;
Point p, sp;
Rectangle sel, r;
int i, left, scrollcenter;
char tmp[32];
if(entering)
return;
lockdisplay(display);
if(new && getwindow(display, Refnone) < 0)
sysfatal("getwindow: %r");
else
draw(screen, screen->r, cola, nil, ZP);
scrollsz = Dy(screen->r) / f->height - 1;
adjustcolumns();
left = screen->r.min.x;
if(scrollsz < plnum){ /* add a scrollbar */
p.x = sp.x = screen->r.min.x + Scrollwidth;
p.y = screen->r.min.y;
sp.y = screen->r.max.y;
line(screen, p, sp, Endsquare, Endsquare, 0, colb, ZP);
r = screen->r;
r.max.x = r.min.x + Scrollwidth - 1;
r.min.x += 1;
if(scroll < 1)
scrollcenter = 0;
else
scrollcenter = (Dy(screen->r)-Scrollheight/2)*scroll / plnum;
r.min.y += scrollcenter + Scrollheight/4;
r.max.y = r.min.y + Scrollheight;
draw(screen, r, colb, nil, ZP);
left += Scrollwidth + 4;
}
p.x = sp.x = left + colwidth[0] + 4;
p.y = 0;
sp.y = screen->r.max.y;
line(screen, p, sp, Endsquare, Endsquare, 0, colb, ZP);
p.x = sp.x = left + colwidth[0] + 8 + colwidth[1] + 4;
p.y = 0;
sp.y = screen->r.max.y;
line(screen, p, sp, Endsquare, Endsquare, 0, colb, ZP);
sp.x = sp.y = 0;
p.x = left + 2;
p.y = screen->r.min.y + 2;
for(i = scroll; i < plnum; i++, p.y += f->height){
if(i < 0)
continue;
if(p.y > screen->r.max.y)
break;
if(pcur == i){
sel.min.x = left;
sel.min.y = p.y;
sel.max.x = screen->r.max.x;
sel.max.y = p.y + f->height;
draw(screen, sel, colb, nil, ZP);
col = cola;
}else{
col = colb;
}
sel = screen->r;
r = screen->r;
p.x = left + 2;
sel.max.x = p.x + colwidth[0];
replclipr(screen, 0, sel);
string(screen, p, col, sp, f, pl[i].artist[0]);
p.x += colwidth[0] + 8;
sel.min.x = p.x;
sel.max.x = p.x + colwidth[1];
replclipr(screen, 0, sel);
string(screen, p, col, sp, f, pl[i].album);
p.x += colwidth[1] + 8;
sel.min.x = p.x;
sel.max.x = p.x + colwidth[2];
replclipr(screen, 0, sel);
string(screen, p, col, sp, f, pl[i].title);
replclipr(screen, 0, r);
if(pcurplaying == i){
Point rightp, leftp;
leftp.y = rightp.y = p.y - 1;
leftp.x = left;
rightp.x = screen->r.max.x;
line(screen, leftp, rightp, 0, 0, 0, colb, sp);
leftp.y = rightp.y = p.y + f->height;
line(screen, leftp, rightp, 0, 0, 0, colb, sp);
}
}
if(cover != nil){
r = screen->r;
r.min.x = r.max.x - cover->r.max.x - 8;
r.min.y = r.max.y - cover->r.max.y - 8 - f->height - 4;
draw(screen, r, display->black, nil, ZP);
r.min.x += 4;
r.min.y += 4;
r.max.x -= 4;
r.max.y -= 4;
draw(screen, r, cover, nil, ZP);
}
if(pcurplaying >= 0){
snprint(tmp, sizeof(tmp), "%P/%P", (int)(byteswritten/Bps), pl[pcurplaying].duration/1000);
r = screen->r;
r.min.x = r.max.x - stringwidth(f, tmp) - 4;
r.min.y = r.max.y - f->height - 4;
draw(screen, r, display->black, nil, ZP);
r.min.x += 2;
r.min.y += 2;
string(screen, r.min, cola, sp, f, tmp);
}
flushimage(display, 1);
unlockdisplay(display);
}
static void
coverload(void *player_)
{
int p[2], pid, fd;
char *prog, *path, *s, tmp[32];
Meta *m;
Channel *ch;
Player *player;
Image *newcover;
threadsetname("cover");
player = player_;
m = &pl[player->pcur];
pid = -1;
ch = player->img;
fd = -1;
prog = nil;
if(m->imagefmt != nil && m->imagereader == 0){
if(strcmp(m->imagefmt, "image/png") == 0)
prog = "png";
else if(strcmp(m->imagefmt, "image/jpeg") == 0)
prog = "jpg";
}
if(prog == nil){
path = strdup(m->path);
if(path != nil && (s = utfrrune(path, '/')) != nil){
*s = 0;
if((s = smprint("%s/cover.jpg", path)) != nil && (fd = open(s, OREAD)) >= 0)
prog = "jpg";
free(s);
s = nil;
if(fd < 0 && (s = smprint("%s/cover.png", path)) != nil && (fd = open(s, OREAD)) >= 0)
prog = "png";
free(s);
}
free(path);
}
if(prog == nil)
goto done;
if(fd < 0){
fd = open(m->path, OREAD);
seek(fd, m->imageoffset, 0);
}
pipe(p);
if((pid = rfork(RFPROC|RFFDG|RFREND|RFNOTEG)) == 0){
dup(fd, 0); close(fd);
dup(p[1], 1); close(p[1]);
dup(open("/dev/null", OWRITE), 2);
snprint(tmp, sizeof(tmp), "%s -9t | resample -x128", prog);
execl("/bin/rc", "rc", "-c", tmp, nil);
sysfatal("execl: %r");
}
close(fd);
close(p[1]);
if(pid > 0){
newcover = readimage(display, p[0], 1);
sendp(ch, newcover);
}
close(p[0]);
done:
if(pid < 0)
sendp(ch, nil);
chanclose(ch);
chanfree(ch);
if(pid >= 0)
postnote(PNGROUP, pid, "interrupt");
threadexits(nil);
}
static Player *playernext;
static Player *playercurr;
static int
playerret(Player *player)
{
return recvul(player->ev) == Everror ? -1 : 0;
}
static void
stop(Player *player)
{
if(player == nil)
return;
if(player == playernext)
playernext = nil;
sendul(player->ctl, Cstop);
}
static void playerthread(void *player_);
static Player *
newplayer(int pcur, int loadnext)
{
Player *player;
if(playernext != nil && loadnext){
if(pcur == playernext->pcur){
player = playernext;
playernext = nil;
goto done;
}
stop(playernext);
playernext = nil;
}
player = mallocz(sizeof(*player), 1);
player->ctl = chancreate(sizeof(ulong), 0);
player->ev = chancreate(sizeof(ulong), 0);
player->pcur = pcur;
threadcreate(playerthread, player, mainstacksize);
if(playerret(player) < 0)
return nil;
done:
if(pcur < plnum-1 && playernext == nil && loadnext)
playernext = newplayer(pcur+1, 0);
return player;
}
static int
start(Player *player)
{
if(player != nil)
sendul(player->ctl, Cstart);
return -1;
}
static void
playerthread(void *player_)
{
char *buf;
Player *player;
Ioproc *io;
Image *thiscover;
ulong c;
int p[2], fd, pid, n, got, noinit, trycoverload;
u64int bytesfrom;
threadsetname("player");
player = player_;
noinit = 0;
bytesfrom = 0;
c = 0;
buf = nil;
trycoverload = 1;
io = nil;
pid = -1;
restart:
if((fd = open(pl[player->pcur].path, OREAD)) < 0){
fprint(2, "%r\n");
sendul(player->ev, Everror);
goto freeplayer;
}
pipe(p);
if((pid = rfork(RFPROC|RFFDG|RFREND|RFNOTEG)) == 0){
dup(p[0], 1); close(p[0]);
dup(fd, 0); close(fd);
close(p[1]);
dup(open("/dev/null", OWRITE), 2);
execl("/bin/play", "play", "-o", "/fd/1", nil);
sysfatal("execl: %r");
}
if(pid < 0)
sysfatal("rfork: %r");
close(fd);
close(p[0]);
byteswritten = 0;
if(!noinit){
sendul(player->ev, Evready);
buf = malloc(Relbufsz);
io = ioproc();
for(c = 0, got = 0; got < Relbufsz; got += n){
if((c = nbrecvul(player->ctl)) != 0)
break;
n = ioread(io, p[1], buf+got, Relbufsz-got);
if(n < 1)
break;
}
if(c == 0)
c = recvul(player->ctl);
if(c != Cstart)
goto freeplayer;
iowrite(io, audio, buf, got);
byteswritten = got;
bytesfrom = 0;
c = 0;
noinit = 1;
}
pcurplaying = player->pcur;
if(c != Cbackward && c != Cbackwardfast)
redraw(screen, 0);
while(1){
n = Relbufsz;
if(bytesfrom > byteswritten && n > bytesfrom-byteswritten)
n = bytesfrom-byteswritten;
n = ioread(io, p[1], buf, n);
if(n < 1)
break;
thiscover = nil;
if(player->img != nil && nbrecv(player->img, &thiscover) != 0){
freeimage(cover);
cover = thiscover;
redraw(screen, 0);
player->img = nil;
}
c = nbrecvul(player->ctl);
if(c == Cstop || c == -1)
goto stop;
if(c == Ctoggle){
c = recvul(player->ctl);
if(c == Cstop)
goto stop;
}else if(c == Cforward){
bytesfrom = byteswritten + Seekbytes;
}else if(c == Cforwardfast){
bytesfrom = byteswritten + Seekbytesfast;
}else if(c == Cbackward){ /* to seek backwards we need to restart playback */
bytesfrom = byteswritten >= Seekbytes ? byteswritten - Seekbytes : 0;
n = 0; /* not an error */
break;
}else if(c == Cbackwardfast){
bytesfrom = byteswritten >= Seekbytesfast ? byteswritten - Seekbytesfast : 0;
n = 0; /* not an error */
break;
}
c = 0;
if(bytesfrom <= byteswritten){
if(iowrite(io, audio, buf, n) != n){
fprint(2, "failed to write %d bytes: %r\n", n);
break;
}
if(trycoverload && byteswritten >= Bps){
player->img = chancreate(sizeof(Image*), 0);
proccreate(coverload, player, 4096);
trycoverload = 0;
}
}
byteswritten += n;
if(bytesfrom == byteswritten || (byteswritten/Bps > (byteswritten-n)/Bps))
redraw(screen, 0);
}
if(n == 0){ /* seeking backwards or end of the song */
close(p[1]);
if(c != Cbackward && c != Cbackwardfast){
playercurr = nil;
playercurr = newplayer((player->pcur+1) % plnum, 1);
start(playercurr);
goto stop;
}
goto restart;
}
stop:
if(player->img != nil)
freeimage(recvp(player->img));
freeplayer:
if(player == playercurr)
playercurr = nil;
if(player == playernext)
playernext = nil;
chanclose(player->ctl);
chanclose(player->ev);
chanfree(player->ctl);
chanfree(player->ev);
close(p[1]);
closeioproc(io);
free(buf);
free(player);
if(pid >= 0)
postnote(PNGROUP, pid, "interrupt");
threadexits(nil);
}
static void
toggle(Player *player)
{
if(player != nil)
sendul(player->ctl, Ctoggle);
}
static void
backward(Player *player)
{
if(player != nil)
sendul(player->ctl, Cbackward);
}
static void
forward(Player *player)
{
if(player != nil)
sendul(player->ctl, Cforward);
}
static void
backwardfast(Player *player)
{
if(player != nil)
sendul(player->ctl, Cbackwardfast);
}
static void
forwardfast(Player *player)
{
if(player != nil)
sendul(player->ctl, Cforwardfast);
}
static void
readplist(void)
{
Meta *m;
char *s, *e, *endrec;
int i, n, sz, alloc, tagsz, intval;
s = nil;
for(alloc = sz = 0;;){
alloc += 65536;
if((s = realloc(s, alloc)) == nil)
sysfatal("no memory");
for(n = 0; sz < alloc; sz += n){
n = read(0, s+sz, alloc-sz);
if(n < 0)
sysfatal("%r");
if(n == 0)
break;
}
if(n == 0)
break;
}
plraw = s;
plrawsize = sz;
plraw[plrawsize-1] = 0;
if(sz < 4 || s[0] != '#' || s[1] != ' ' || !isdigit(s[2]) || (s = memchr(plraw, '\n', sz)) == nil)
sysfatal("invalid playlist");
s++; /* at the start of the first record */
plnum = atoi(plraw+2);
pl = calloc(plnum, sizeof(Meta));
for(i = 0; i < plnum; i++, s = endrec){
if(plraw+plrawsize < s+10)
sysfatal("truncated playlist");
if(s[0] != '#' || s[1] != ' ' || !isdigit(s[2]))
sysfatal("invalid record");
if((n = strtol(s+2, &e, 10)) < 0 || n > plnum)
sysfatal("invalid track index");
if(pl[n].path != nil)
sysfatal("duplicate track index");
s[-1] = 0;
sz = strtol(e, &s, 10);
*s++ = 0; /* skip '\n' */
if(s+sz > plraw+plrawsize)
sysfatal("truncated playlist");
s[sz-1] = 0; /* '\n'→'\0' to mark the end of the record */
endrec = s+sz;
m = &pl[n];
for(;;){
if(s[0] == Pimage){
m->imageoffset = strtol(s+2, &e, 10);
m->imagesize = strtol(e+1, &s, 10);
m->imagereader = strtol(s+1, &e, 10);
m->imagefmt = e + 1;
s = strchr(e+2, '\n') + 1;
}else if(s[0] == Pchannels || s[0] == Pduration || s[0] == Psamplerate){
intval = strtol(s+2, &e, 10);
if(s[0] == Pchannels) m->channels = intval;
else if(s[0] == Pduration) m->duration = intval;
else if(s[0] == Psamplerate) m->samplerate = intval;
s = e + 1;
}else if(s[0] == Ppath){
m->path = s+2;
break; /* always the last one */
}else{
tagsz = strtol(s+1, &e, 10);
if(e+tagsz >= plraw+plrawsize)
sysfatal("truncated playlist");
e++; /* point to tag value */
e[tagsz] = 0; /* '\n'→'\0' to mark the end of the tag value */
if(s[0] == Palbum) m->album = e;
else if(s[0] == Partist && m->numartist < Maxartist) m->artist[m->numartist++] = e;
else if(s[0] == Pdate) m->date = e;
else if(s[0] == Ptitle) m->title = e;
else if(s[0] == Pdate) m->date = e;
else if(s[0] == Ptrack) m->track = e;
else sysfatal("unknown tag type %c", s[0]);
s = e + tagsz + 1;
}
s[-1] = 0;
}
}
}
static void
search(char d)
{
char *s, *snext;
static char buf[48];
static int sz;
int inc;
inc = (d == '/' || d == 'n') ? 1 : -1;
if(d == '/' || d == '?'){
entering = 1;
sz = enter(inc > 0 ? "forward:" : "backward:", buf, sizeof(buf), mctl, kctl, nil);
entering = 0;
}
if(sz < 1 || (inc > 0 && pcur >= plnum-1) || (inc < 0 && pcur < 1))
return;
s = pl[pcur + (inc > 0 ? 0 : -1)].path;
s += strlen(s) + 1;
for(; s > plraw && s < plraw+plrawsize-sz; s += inc){
if(cistrncmp(s, buf, sz) != 0)
continue;
snext = s;
for(; s != plraw && *s; s--);
if(s == plraw || (s[1] != Partist && s[1] != Palbum && s[1] != Ptitle && s[1] != Pdate)){
if(inc > 0)
s = snext;
continue;
}
for(s--; s != plraw; s--){
if(memcmp(s, "\0# ", 3) == 0 && isdigit(s[3])){
pcur = atoi(s+3);
redraw(screen, 1);
return;
}
}
break;
}
}
static void
usage(void)
{
fprint(2, "usage: zuke [-b]\n");
exits("usage");
}
void
threadmain(int argc, char **argv)
{
char tmp[256];
Point lastclick;
Rune key;
Mouse m;
Alt a[] =
{
{ nil, &m, CHANRCV },
{ nil, nil, CHANRCV },
{ nil, &key, CHANRCV },
{ nil, nil, CHANEND },
};
int fd, scrolling, oldscroll, oldpcur, oldscrolling, usingscrollbar;
USED(argc); USED(argv);
audio = open("/dev/audio", OWRITE);
if(audio < 0)
sysfatal("audio: %r");
readplist();
if(plnum < 1){
fprint(2, "empty playlist\n");
sysfatal("empty");
}
if(initdraw(nil, nil, "zuke") < 0)
sysfatal("initdraw: %r");
unlockdisplay(display);
if((mctl = initmouse(nil, screen)) == nil)
sysfatal("initmouse: %r");
if((kctl = initkeyboard(nil)) == nil)
sysfatal("initkeyboard: %r");
a[0].c = mctl->c;
a[1].c = mctl->resizec;
a[2].c = kctl->c;
f = display->defaultfont;
cola = display->white;
colb = display->black;
srand(time(0));
pcurplaying = -1;
scrolling = oldscroll = usingscrollbar = 0;
fmtinstall('P', positionfmt);
threadsetname("zuke");
snprint(tmp, sizeof(tmp), "/proc/%d/ctl", getpid());
if((fd = open(tmp, OWRITE)) >= 0){
fprint(fd, "pri 13\n");
close(fd);
}
redraw(screen, 1);
for(;;){
oldpcur = pcur;
switch(alt(a)){
case 0:
oldscrolling = scrolling;
scrolling = m.buttons & 2;
if(!oldscrolling && (m.buttons & 2) != 0){
usingscrollbar = m.xy.x < screen->r.min.x+Scrollwidth;
lastclick = m.xy;
oldscroll = scroll;
}
if(m.buttons == 0)
break;
if(scrolling){
if(scrollsz >= plnum)
break;
if(usingscrollbar)
scroll = (m.xy.y - screen->r.min.y - Scrollheight/4)*plnum / (Dy(screen->r)-Scrollheight/2);
else
scroll = oldscroll + (lastclick.y - m.xy.y) / f->height;
if(scroll > plnum-scrollsz-1)
scroll = plnum-scrollsz-1;
if(scroll < 0)
scroll = 0;
redraw(screen, 0);
}else{
pcur = scroll + (m.xy.y - screen->r.min.y)/f->height;
if(m.buttons == 4){
stop(playercurr);
playercurr = newplayer(pcur, 1);
start(playercurr);
}
}
break;
case 1: /* resize */
redraw(screen, 1);
break;
case 2:
switch(key){
case Kleft:
backward(playercurr);
break;
case Kright:
forward(playercurr);
break;
case ',':
backwardfast(playercurr);
break;
case '.':
forwardfast(playercurr);
break;
case Kup:
pcur--;
break;
case Kpgup:
pcur -= scrollsz;
break;
case Kdown:
pcur++;
break;
case Kpgdown:
pcur += scrollsz;
break;
case Kend:
pcur = plnum-1;
break;
case Khome:
pcur = 0;
break;
case 10:
stop(playercurr);
playercurr = newplayer(pcur, 1);
start(playercurr);
break;
case 'q': case Kdel:
stop(playercurr);
goto end;
case 'o':
pcur = pcurplaying;
break;
case '>':
if(playercurr == nil)
break;
pcur = pcurplaying;
if(++pcur >= plnum)
pcur = 0;
stop(playercurr);
playercurr = newplayer(pcur, 1);
start(playercurr);
break;
case '<':
if(playercurr == nil)
break;
pcur = pcurplaying;
if(--pcur < 0)
pcur = plnum-1;
stop(playercurr);
playercurr = newplayer(pcur, 1);
start(playercurr);
break;
case 's':
stop(playercurr);
playercurr = nil;
pcurplaying = -1;
redraw(screen, 1);
break;
case 'p':
toggle(playercurr);
break;
case '/': case '?': case 'n': case 'N':
search(key);
break;
}
}
if(pcur != oldpcur){
if(pcur < 0)
pcur = 0;
else if(pcur >= plnum)
pcur = plnum - 1;
if(pcur < scroll)
scroll = pcur;
else if(pcur > scroll + scrollsz)
scroll = pcur - scrollsz;
if(scroll > plnum - scrollsz)
scroll = plnum - scrollsz;
if(scroll < 0)
scroll = 0;
if(pcur != oldpcur)
redraw(screen, 0);
}
}
end:
closemouse(mctl);
closekeyboard(kctl);
threadexitsall(nil);
}