/*
 * Copyright (C) 12 Jun 2003 Tomas Cermak
 *
 * This file is part of wmradio program.
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write to the Free Software
 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
 */

#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdio.h>
#include <unistd.h>
#include <ctype.h>
#include <stdlib.h>
#include <string.h>
#include <time.h>
#include <X11/Xlib.h>
#include <X11/xpm.h>
#include <getopt.h>
#include "config.h"
#include "wm_envelope.h"
#include "wmradio.h"
#include "skin.h"
#include "rc.h"

Display *main_display = 0;
Window root, applet, icon, buffer;

GC NormalGC;
unsigned long gcm;
XGCValues gcv;

char radio_continue;

Atom wm_delete_window_atom, wm_protocol_atom;

Pixel GetColor(char *ColorName, Display * disp, Window win)
{
    XColor Color;
    XWindowAttributes Attributes;

    XGetWindowAttributes(disp, win, &Attributes);
    Color.pixel = 0;

    if (!XParseColor(disp, Attributes.colormap, ColorName, &Color))
        printf("wmradio: can't parse %s\n", ColorName);
    else if (!XAllocColor(disp, Attributes.colormap, &Color))
        printf("wmradio: can't allocate %s\n", ColorName);

    return Color.pixel;
}

int parse_command_line(int argc, char *argv [],RadioInfo *info)
{
    static struct option long_options[] = {
        {"dont-quit-mode",0,0,0},           /* 0 */
        {"start-muted",0,0,0},
	{"help",0,0,0},
	{"version",0,0,0},

	{"osd",0,0,0},
	{"osd-font",1,0,0},            /* 5 */
	{"osd-color",1,0,0},
	{"osd-position",1,0,0},
	{"osd-shadow-offset",1,0,0},
	{"osd-timeout",1,0,0},

	{"skin",1,0,0},

	{0,0,0,0}
    };
    int option_index = 0;
    int opt;
    int x,y,offset,timeout;

    while(1) {
        opt = getopt_long(argc,argv,"qmhvof:c:p:s:t:k:",long_options, &option_index);
	if( opt == -1 ) break;
	if( opt == 0 ) opt = option_index;
 	switch(opt) {
	case 0:
	case 'q':
	    info->dont_quit_mode = 1;
	    rc_set_variable_as_int(SECTION_CONFIG,"dont-quit-mode",1);
	    break;
	case 1:
        case 'm':
	    rc_set_variable_as_int(SECTION_CONFIG,"start-muted",1);
	    break;
	case 2:
	case 'h':
	    printf("wmradio [options]\n"
		   "  options are:\n"
		   "  -h|--help              print this help\n"
		   "  -m|--start-muted       program starts, but doesn't open radio device\n"
		   "  -q|--dont-quit-mode    program doesn't quit, just close radio device\n"
		   "  -v|--version           print version and quit\n"
		   "  -o|--osd               use osd\n"
		   "  -f|--osd-font font     osd font\n"
		   "  -c|--osd-color         font color\n"
		   "  -p|--osd-position      display position (-p 10x30 for example)\n"
		   "  -s|--osd-shadow-offset shadow offset\n"
		   "  -t|--osd-timeout       osd timeout\n"
		   "  -k|--skin              skin\n"
		   );
	    return 0;
	case 3:
	case 'v':
	    printf("This is %s %s\n", PACKAGE,VERSION);
	    return 0;
	case 4:
	case 'o':
	    rc_set_variable_as_int(SECTION_CONFIG,"osd",1);
	    break;
	case 5:
	case 'f':
	    rc_set_variable(SECTION_CONFIG,"osd-font",optarg);
	    break;
	case 6:
	case 'c':
	    rc_set_variable(SECTION_CONFIG,"osd-color",optarg);
	    break;
	case 7:
	case 'p':
	    if (sscanf(optarg,"%ix%i", &x, &y) < 2) {
		fprintf(stderr, "%s: incorrect syntax in OSD position\n", argv[0]);
	    } else {
		rc_set_variable_as_int(SECTION_CONFIG,"osd-position",x);
		/* rc_set_variable_as_int(SECTION_CONFIG,"osd-position",y); */
	    }
	    break;
	case 8:
	case 's':
	    if (sscanf(optarg,"%i", &offset) < 1) {
		fprintf(stderr, "%s: incorrect syntax in OSD shadow offset\n", argv[0]);
	    } else {
		rc_set_variable_as_int(SECTION_CONFIG,"osd-shadow-offset",offset);
	    }
	    break;
	case 9:
	case 't':
	    if (sscanf(optarg,"%i", &timeout) < 1) {
		fprintf(stderr, "%s: incorrect syntax in OSD timeout\n", argv[0]);
	    } else {
		rc_set_variable_as_int(SECTION_CONFIG,"osd-timeout",timeout);
	    }
	    break;
	case 10:
	case 'k':
	    rc_set_variable(SECTION_CONFIG,"skin",optarg);
	    break;
	}
    }
    return 1;
}

void video_mainloop(void)
{
    XEvent xe;
    RadioEvent re;

    radio_continue = 1;
    while (radio_continue) {
        /* X Events */
        while (XPending(main_display)) {
            XNextEvent(main_display, &xe);
            switch (xe.type) {
            case Expose:
		re.type = REVENT_EXPOSE;
		wmradio_handle_event(&re);
                break;
	    case ClientMessage:
	        if(xe.xclient.message_type == wm_protocol_atom) {
		    Atom a = (xe.xclient.data.l)[0];
		    if( a == wm_delete_window_atom ) {
			re.type = REVENT_QUIT;
			printf("quit\n");
			wmradio_handle_event(&re);
		    }
		}
		break;
            case DestroyNotify:
		re.type = REVENT_QUIT;
		wmradio_handle_event(&re);
		printf("quit\n");
                return;
            case ButtonPress:
 	        if(xe.xbutton.button < 4) {
		    re.type = REVENT_BUTTON_PRESS;
		    if(xe.xbutton.button == 4) re.type = REVENT_SCROLL_UP;
		    if(xe.xbutton.button == 5) re.type = REVENT_SCROLL_DOWN;
		    re.x = xe.xbutton.x;
		    re.y = xe.xbutton.y;
		    re.button = xe.xbutton.button;
		    re.control = xe.xbutton.state & ControlMask ?
		        CONTROL_STATE_PRESSED : CONTROL_STATE_NOT_PRESSED;
		    re.shift = xe.xbutton.state & ShiftMask ?
		        CONTROL_STATE_PRESSED : CONTROL_STATE_NOT_PRESSED;
		    wmradio_handle_event(&re);
		}
                break;
            case ButtonRelease:
		re.type = REVENT_BUTTON_RELEASE;
		re.x = xe.xbutton.x;
		re.y = xe.xbutton.y;
		re.button = xe.xbutton.button;
		re.control = xe.xbutton.state & ControlMask ?
		    CONTROL_STATE_PRESSED : CONTROL_STATE_NOT_PRESSED;
		re.shift = xe.xbutton.state & ShiftMask ?
		    CONTROL_STATE_PRESSED : CONTROL_STATE_NOT_PRESSED;
 	        if(re.button == 4) re.type = REVENT_SCROLL_UP;
 	        if(re.button == 5) re.type = REVENT_SCROLL_DOWN;
		wmradio_handle_event(&re);
                break;
	    }
        } /* XPending */
	usleep(100000);
	re.type = REVENT_TIMER;
	wmradio_handle_event(&re);
    }
    XCloseDisplay(main_display);
}

void video_close(void)
{
    radio_continue = 0;
}

void video_draw(float freq,int stereo)
{
    skin_to_window(main_display,buffer, NormalGC,freq,stereo);
    XCopyArea(main_display,buffer, icon, NormalGC,0,0,skin_width(),skin_height(),0,0);
    XCopyArea(main_display,buffer, applet, NormalGC,0,0,skin_width(),skin_height(),0,0);
}

int main(int argc, char *argv [])
{
    Pixel foreground,background;
    XWMHints WmHints;
    Status status;
    XClassHint classhint;
    XTextProperty title;
    int screen;
    char * appletname = "WmRadio";

    wmradio_init_radio_info();
    rc_read_config();
    parse_command_line(argc,argv,wmradio_radio_info());

    main_display = XOpenDisplay(NULL);
    if (!main_display) {
        printf("wmradio: can't open display %s.\n",XDisplayName(NULL));
        return 0;
    }
    screen = DefaultScreen(main_display);
    root = RootWindow(main_display,screen);

    background = GetColor("black", main_display, root);
    foreground = GetColor("white", main_display, root);
    create_skin(rc_get_variable(SECTION_CONFIG,"skin","default.skin"),main_display,root);
    wmradio_init();
    applet = XCreateSimpleWindow(main_display,
				 root,
				 0,0,skin_width(),skin_height(),
				 0,
				 foreground,background);
    icon   = XCreateSimpleWindow(main_display,
				 root,
				 0,0,skin_width(),skin_height(),
				 0,
				 foreground,background);
    WmHints.flags = StateHint | IconWindowHint;
    WmHints.initial_state = WithdrawnState;
    WmHints.icon_window = icon;
    WmHints.window_group = applet;
    WmHints.flags |= WindowGroupHint;

    XSetWMHints(main_display,
		applet,
		&WmHints);

    buffer = XCreatePixmap(main_display,
			   root,
			   skin_width(),skin_height(),
			   DefaultDepth(main_display,screen)/*16  color_depth */);

    status = XStringListToTextProperty(&appletname, 1, &title);
    XSetWMName(main_display, applet, &title);
    XSetWMName(main_display, icon, &title);
    classhint.res_name = "wmradio" ;
    classhint.res_class = "WMRADIO";
    XSetClassHint(main_display, applet, &classhint);
    XStoreName(main_display, applet, "WmRadio");
    XSetIconName(main_display, applet, "WmRadio");

    wm_delete_window_atom = XInternAtom(main_display, "WM_DELETE_WINDOW", 0);
    wm_protocol_atom = XInternAtom(main_display, "WM_PROTOCOLS", 0);
    XSetWMProtocols(main_display, applet, &wm_delete_window_atom, 1);

    status = XMapWindow(main_display, applet);
    gcm = GCForeground | GCBackground | GCGraphicsExposures;
    gcv.foreground = foreground;
    gcv.background = background;
    gcv.graphics_exposures = 0;
    NormalGC = XCreateGC(main_display, root, gcm, &gcv);
    XSelectInput(main_display, applet,
		 ButtonPressMask | ExposureMask |
		 ButtonReleaseMask | PointerMotionMask |
		 StructureNotifyMask);
    XSelectInput(main_display, icon,
		 ButtonPressMask | ExposureMask |
		 ButtonReleaseMask | PointerMotionMask |
		 StructureNotifyMask);
    video_mainloop();
    wmradio_done();
    return 0;
}