dockapps/wmmixer-alsa/wmmixer-alsa.c
Doug Torrance 665b8f777a wmmixer-alsa: Add to repository.
Previously, wmmixer-alsa existed as an object in the repository named
wmmixer-alsa-0.6.  However, it was not a directory and contained no files.
This patch removes this object and adds the source to wmmixer-alsa.  It was
obtained by extracting wmmixer-alsa-0.6.tar.gz from the source rpm found at
http://ftp.cc.uoc.gr/mirrors/linux/pld/pool/w/wmmixer-alsa/.
2014-08-06 13:23:10 -06:00

545 lines
13 KiB
C

#include "wmmixer-alsa.h"
int main(int argc, char **argv)
{
XGCValues gcv;
unsigned long gcm;
int exact,left,right,device_index,i;
XpmAttributes xpmattr;
XpmColorSymbol xpmcsym[4];
elementinfo *e;
scanArgs(argc, argv);
initXWin(argc, argv);
exact=left=right=device_index=-1;
init_mixer();
cure=element;
e=element;
while(e)
{
if(!strcasecmp(e->info.eid.name,"Master Volume"))
e->icon=0;
else if(!strcasecmp(e->info.eid.name,"PCM Volume"))
e->icon=1;
else if(!strcasecmp(e->info.eid.name,"MIC Volume"))
e->icon=5;
else if(!strcasecmp(e->info.eid.name,"Line Volume"))
e->icon=4;
else if(!strcasecmp(e->info.eid.name,"CD Volume"))
e->icon=3;
else if(!strcasecmp(e->info.eid.name,"Synth Volume"))
e->icon=2;
else if(!strcasecmp(e->info.eid.name,"PC Speaker Volume"))
e->icon=6;
/*
* bass = 7
* treble = 8
*/
else
e->icon=9;
e=e->next;
}
gcm=GCGraphicsExposures;
gcv.graphics_exposures=0;
gc_gc=XCreateGC(d_display, w_root, gcm, &gcv);
color[0]=mixColor(ledcolor, 0, backcolor, 100);
color[1]=mixColor(ledcolor, 100, backcolor, 0);
color[2]=mixColor(ledcolor, 60, backcolor, 40);
color[3]=mixColor(ledcolor, 25, backcolor, 75);
xpmcsym[0].name="back_color";
xpmcsym[0].value=NULL;;
xpmcsym[0].pixel=color[0];
xpmcsym[1].name="led_color_high";
xpmcsym[1].value=NULL;;
xpmcsym[1].pixel=color[1];
xpmcsym[2].name="led_color_med";
xpmcsym[2].value=NULL;;
xpmcsym[2].pixel=color[2];
xpmcsym[3].name="led_color_low";
xpmcsym[3].value=NULL;;
xpmcsym[3].pixel=color[3];
xpmattr.numsymbols=4;
xpmattr.colorsymbols=xpmcsym;
xpmattr.exactColors=0;
xpmattr.closeness=40000;
xpmattr.valuemask=XpmColorSymbols | XpmExactColors | XpmCloseness;
XpmCreatePixmapFromData(d_display, w_root, wmmixer_xpm, &pm_main, &pm_mask, &xpmattr);
XpmCreatePixmapFromData(d_display, w_root, tile_xpm, &pm_tile, NULL, &xpmattr);
XpmCreatePixmapFromData(d_display, w_root, icons_xpm, &pm_icon, NULL, &xpmattr);
pm_disp=XCreatePixmap(d_display, w_root, 64, 64, DefaultDepth(d_display, DefaultScreen(d_display)));
if(wmaker || ushape || astep)
XShapeCombineMask(d_display, w_activewin, ShapeBounding, winsize/2-32, winsize/2-32, pm_mask, ShapeSet);
else
XCopyArea(d_display, pm_tile, pm_disp, gc_gc, 0, 0, 64, 64, 0, 0);
XSetClipMask(d_display, gc_gc, pm_mask);
XCopyArea(d_display, pm_main, pm_disp, gc_gc, 0, 0, 64, 64, 0, 0);
XSetClipMask(d_display, gc_gc, None);
if(count==0)
fprintf(stderr,"%s : Sorry, no supported channels found.\n", NAME);
else
{
int done=0;
XEvent xev;
checkVol();
XSelectInput(d_display, w_activewin, ExposureMask | ButtonPressMask | ButtonReleaseMask | ButtonMotionMask);
XMapWindow(d_display, w_main);
while(!done)
{
while(XPending(d_display))
{
XNextEvent(d_display, &xev);
switch(xev.type)
{
case Expose:
repaint();
break;
case ButtonPress:
pressEvent(&xev.xbutton);
break;
case ButtonRelease:
releaseEvent(&xev.xbutton);
break;
case MotionNotify:
motionEvent(&xev.xmotion);
break;
case ClientMessage:
if(xev.xclient.data.l[0]==deleteWin)
done=1;
break;
}
}
if(btnstate & (BTNPREV | BTNNEXT))
{
rpttimer++;
if(rpttimer>=RPTINTERVAL)
{
if(btnstate & BTNNEXT)
{
cure=cure->next;
if(!cure)
cure=element;
}
else
{
cure=cure->prev;
if(!cure)
{
elementinfo *e;
e=element;
while(e->next)
e=e->next;
cure=e;
}
}
checkVol();
rpttimer=0;
}
}
else
checkVol();
XFlush(d_display);
usleep(50000);
}
}
XFreeGC(d_display, gc_gc);
XFreePixmap(d_display, pm_main);
XFreePixmap(d_display, pm_tile);
XFreePixmap(d_display, pm_disp);
XFreePixmap(d_display, pm_mask);
XFreePixmap(d_display, pm_icon);
freeXWin();
return 0;
}
void initXWin(int argc, char **argv)
{
int pos;
XWMHints wmhints;
XSizeHints shints;
winsize=astep ? ASTEPSIZE : NORMSIZE;
if((d_display=XOpenDisplay(display))==NULL)
{
fprintf(stderr,"%s : Unable to open X display '%s'.\n", NAME, XDisplayName(display));
exit(1);
}
_XA_GNUSTEP_WM_FUNC=XInternAtom(d_display, "_GNUSTEP_WM_FUNCTION", 0);
deleteWin=XInternAtom(d_display, "WM_DELETE_WINDOW", 0);
w_root=DefaultRootWindow(d_display);
shints.x=0;
shints.y=0;
shints.flags=0;
pos=(XWMGeometry(d_display, DefaultScreen(d_display), position, NULL, 0, &shints, &shints.x, &shints.y, &shints.width, &shints.height, &shints.win_gravity) & (XValue | YValue));
shints.min_width=winsize;
shints.min_height=winsize;
shints.max_width=winsize;
shints.max_height=winsize;
shints.base_width=winsize;
shints.base_height=winsize;
shints.flags=PMinSize | PMaxSize | PBaseSize;
createWin(&w_main, shints.x, shints.y);
if(wmaker || astep || pos)
shints.flags |= USPosition;
if(wmaker)
{
wmhints.initial_state=WithdrawnState;
wmhints.flags=WindowGroupHint | StateHint | IconWindowHint;
createWin(&w_icon, shints.x, shints.y);
w_activewin=w_icon;
wmhints.icon_window=w_icon;
}
else
{
wmhints.initial_state=NormalState;
wmhints.flags=WindowGroupHint | StateHint;
w_activewin=w_main;
}
wmhints.window_group=w_main;
XSetWMHints(d_display, w_main, &wmhints);
XSetWMNormalHints(d_display, w_main, &shints);
XSetCommand(d_display, w_main, argv, argc);
XStoreName(d_display, w_main, NAME);
XSetIconName(d_display, w_main, NAME);
XSetWMProtocols(d_display, w_activewin, &deleteWin, 1);
}
void freeXWin()
{
XDestroyWindow(d_display, w_main);
if(wmaker)
XDestroyWindow(d_display, w_icon);
XCloseDisplay(d_display);
}
void createWin(Window *win, int x, int y)
{
XClassHint classHint;
*win=XCreateSimpleWindow(d_display, w_root, x, y, winsize, winsize, 0, 0, 0);
classHint.res_name=NAME;
classHint.res_class=CLASS;
XSetClassHint(d_display, *win, &classHint);
}
unsigned long mixColor(char *colorname1, int prop1, char *colorname2, int prop2){
XColor color, color1, color2;
XWindowAttributes winattr;
XGetWindowAttributes(d_display, w_root, &winattr);
XParseColor(d_display, winattr.colormap, colorname1, &color1);
XParseColor(d_display, winattr.colormap, colorname2, &color2);
color.pixel=0;
color.red=(color1.red*prop1+color2.red*prop2)/(prop1+prop2);
color.green=(color1.green*prop1+color2.green*prop2)/(prop1+prop2);
color.blue=(color1.blue*prop1+color2.blue*prop2)/(prop1+prop2);
color.flags=DoRed | DoGreen | DoBlue;
XAllocColor(d_display, winattr.colormap, &color);
return color.pixel;
}
void show_params(void)
{
fprintf(stdout,"wmmixer-alsa %s\t\tby Sam Hawker\n"
"\t\t\tAdded support for ALSA by\n"
"\t\t\tMartin Dahl <dahlm@vf.telia.no>\n\n"
"Usage: wmmixer-alsa [options]\n\n"
"\t-h\t\tDisplay this help screen\n"
"\t-w\t\tUse WithdrawnState\n"
"\t-s\t\tShaped window\n"
"\t-a\t\tUse smaller window\n"
"\t-l <color>\t\tColor for led display\n"
"\t-b <color>\t\tColor for background\n"
"\t-p <position>\t\tWindow position\n"
"\t-d <display>\t\tTarget display\n",VERSION);
exit(0);
}
void scanArgs(int argc, char **argv)
{
int x;
memset(backcolor,0,32);
memset(ledcolor,0,32);
memset(display,0,32);
memset(position,0,32);
strncpy(ledcolor,LEDCOLOR,32);
strncpy(backcolor,BACKCOLOR,32);
strcpy(display,"");
strcpy(position,"");
wmaker=ushape=astep=0;
btnstate=rpttimer=0;
dragging=count=0;
for(x=1;x<argc;x++)
{
if(argv[x][0]!='-') show_params();
else switch(argv[x][1])
{
case 'w':
wmaker=1;
break;
case 's':
ushape=1;
break;
case 'a':
astep=1;
break;
case 'l':
strncpy(ledcolor,argv[x+1],32);
x++;
break;
case 'b':
strncpy(backcolor,argv[x+1],32);
x++;
break;
case 'p':
strncpy(position,argv[x+1],32);
x++;
break;
case 'd':
strncpy(display,argv[x+1],32);
x++;
break;
default:
show_params();
break;
}
}
return;
}
void checkVol(void)
{
int nl=0,nr=0;
nl=convert_range(cure->element.data.volume1.pvoices[0],cure->info.data.volume1.prange[0].min,cure->info.data.volume1.prange[0].max,0,100);
nr=convert_range(cure->element.data.volume1.pvoices[1],cure->info.data.volume1.prange[1].min,cure->info.data.volume1.prange[1].max,0,100);
if(1)
{
curleft=nl;
curright=nr;
}
else
{
if(nl!=curleft||nr!=curright)
{
if(nl!=curleft)
{
curleft=nl;
drawLeft();
}
if(nr!=curright)
{
curright=nr;
drawRight();
}
}
}
update();
repaint();
}
void setVol(int left, int right)
{
int err;
callbacks();
cure->element.data.volume1.pvoices[0]=convert_range(left,0,100,cure->info.data.volume1.prange[0].min,cure->info.data.volume1.prange[0].max);
if(cure->element.data.volume1.pvoices[0]>cure->info.data.volume1.prange[0].max)
cure->element.data.volume1.pvoices[0]=cure->info.data.volume1.prange[0].max;
if(cure->element.data.volume1.pvoices[0]<cure->info.data.volume1.prange[0].min)
cure->element.data.volume1.pvoices[0]=cure->info.data.volume1.prange[0].min;
cure->element.data.volume1.pvoices[1]=convert_range(right,0,100,cure->info.data.volume1.prange[1].min,cure->info.data.volume1.prange[1].max);
if(cure->element.data.volume1.pvoices[1]>cure->info.data.volume1.prange[1].max)
cure->element.data.volume1.pvoices[1]=cure->info.data.volume1.prange[1].max;
if((err=snd_mixer_element_write(mixer_handle,&cure->element))<0)
fprintf(stderr,"Mixer element write error: %s\n",snd_strerror(err));
return;
}
void callbacks(void)
{
int err;
snd_mixer_callbacks_t callbacks;
memset(&callbacks,0,sizeof(callbacks));
callbacks.rebuild=NULL;
callbacks.element=NULL;
callbacks.group=NULL;
if((err=snd_mixer_read(mixer_handle,&callbacks))<0)
{
fprintf(stderr,"Callbacks error: %s\n",snd_strerror(err));
return;
}
return;
}
int convert_range(int val, int omin, int omax, int nmin, int nmax)
{
int orange=omax-omin, nrange=nmax-nmin;
if(orange==0)
return 0;
return rint((((double)nrange*((double)val-(double)omin))+((double)orange/2.0))/(double)orange+(double)nmin);
}
void pressEvent(XButtonEvent *xev)
{
int x=xev->x-(winsize/2-32);
int y=xev->y-(winsize/2-32);
if(x>=5 && y>=33 && x<=16 && y<=43)
{
cure=cure->prev;
if(!cure)
{
elementinfo *e;
e=element;
while(e->next)
e=e->next;
cure=e;
}
btnstate |= BTNPREV;
rpttimer=0;
drawBtns(BTNPREV);
checkVol();
return;
}
if(x>=17 && y>=33 && x<=28 && y<=43)
{
cure=cure->next;
if(!cure)
cure=element;
btnstate|=BTNNEXT;
rpttimer=0;
drawBtns(BTNNEXT);
checkVol();
return;
}
if(x>=37 && x<=56 && y>=8 && y<=56)
{
int v=((60-y)*100)/(2*25);
dragging=1;
if(x<=50)
setVol(v,curright);
if(x>=45)
setVol(curleft,v);
checkVol();
return;
}
/* if(x>=5 && y>=47 && x<=28 && y<=57)
{
int nl,nr,flags;
mixer.DeviceSet(channel[curchannel]);
mixer.Read(&nl,&nr,&flags);
if(flags & SND_MIXER_DFLG_MUTE)
{
btnstate &= ~BTNREC;
flags &= ~SND_MIXER_DFLG_MUTE;
}
else
{
btnstate |= BTNREC;
flags |= SND_MIXER_DFLG_MUTE;
}
mixer.Write(nl,nr,flags);
checkVol();
return;
}*/
}
void releaseEvent(XButtonEvent *xev)
{
dragging=0;
btnstate &= ~(BTNPREV | BTNNEXT);
drawBtns(BTNPREV | BTNNEXT);
repaint();
}
void motionEvent(XMotionEvent *xev)
{
int x=xev->x-(winsize/2-32);
int y=xev->y-(winsize/2-32);
if(x>=37 && x<=56 && y>=8 && dragging)
{
int v=((60-y)*100)/(2*25);
if(v<0)
v=0;
if(x<=50)
setVol(v,curright);
if(x>=45)
setVol(curleft,v);
checkVol();
}
}
void repaint()
{
XEvent xev;
XCopyArea(d_display, pm_disp, w_activewin, gc_gc, 0, 0, 64, 64, winsize/2-32, winsize/2-32);
while(XCheckTypedEvent(d_display, Expose, &xev));
}
void update()
{
XCopyArea(d_display, pm_icon, pm_disp, gc_gc, cure->icon*22, 0, 22, 22, 6, 5);
drawLeft();
drawRight();
drawBtns(BTNREC);
}
void drawLeft()
{
int i;
XSetForeground(d_display, gc_gc, color[1]);
for(i=0;i<25;i++)
{
if(i==(curleft*25)/100)
XSetForeground(d_display, gc_gc, color[3]);
XFillRectangle(d_display, pm_disp, gc_gc, 37, 55-2*i, 9, 1);
}
}
void drawRight()
{
int i;
XSetForeground(d_display, gc_gc, color[1]);
for(i=0;i<25;i++)
{
if(i==(curright*25)/100)
XSetForeground(d_display, gc_gc, color[3]);
XFillRectangle(d_display, pm_disp, gc_gc, 48, 55-2*i, 9, 1);
}
}
void drawBtns(int btns)
{
if(btns & BTNPREV)
drawBtn(5, 33, 12, 11, (btnstate & BTNPREV));
if(btns & BTNNEXT)
drawBtn(17, 33, 12, 11, (btnstate & BTNNEXT));
if(btns & BTNREC)
drawBtn(5, 47, 24, 11, (btnstate & BTNREC));
}
void drawBtn(int x, int y, int w, int h, int down)
{
if(!down)
XCopyArea(d_display, pm_main, pm_disp, gc_gc, x, y, w, h, x, y);
else
{
XCopyArea(d_display, pm_main, pm_disp, gc_gc, x, y, 1, h-1, x+w-1, y+1);
XCopyArea(d_display, pm_main, pm_disp, gc_gc, x+w-1, y+1, 1, h-1, x, y);
XCopyArea(d_display, pm_main, pm_disp, gc_gc, x, y, w-1, 1, x+1, y+h-1);
XCopyArea(d_display, pm_main, pm_disp, gc_gc, x+1, y+h-1, w-1, 1, x, y);
}
}