view Plugins/Input/wma/wma.c @ 138:afcdbd5aef24 trunk

[svn] Rename bmp-wma.c to wma.c.
author nenolod
date Thu, 10 Nov 2005 14:57:56 -0800
parents Plugins/Input/wma/bmp-wma.c@b8d4c1faa6d7
children 15c2210499ea
line wrap: on
line source

/*  
 *  Audacious, a cross-platform multimedia player.
 *  Copyright (C) 2005 Audacious Team
 *
 *  Based on BMP-WMA:
 *  Copyright (C) 2004 Roman Bogorodskiy <bogorodskiy@inbox.ru>
 *
 *  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.
 */

#ifdef HAVE_MALLOC_H
#include <malloc.h>
#endif
#include <pthread.h>

#include <glib.h>
#include <glib/gprintf.h>

#include <audacious/plugin.h>
#include <audacious/configfile.h>
#include <audacious/util.h>
#include <audacious/titlestring.h>
#include <audacious/vfs.h>

#ifdef HAVE_CONFIG_H
#	include "config.h"
#endif

#include "avcodec.h"
#include "avformat.h"
#include "iir.h"

#define ST_BUFF 1024

static GtkWidget *about_dialog;
static GtkWidget *dialog;

static gboolean wma_decode = 0;
static gboolean wma_pause = 0;
static gboolean wma_eq_on = 0;
static int wma_seekpos = -1;
static int wma_st_buff, wma_idx;
static pthread_t wma_decode_thread;
static pthread_mutex_t  wma_mutex = PTHREAD_MUTEX_INITIALIZER;
static AVCodecContext *c = NULL;
static AVFormatContext *ic = NULL;
static uint8_t *wma_outbuf, *wma_s_outbuf;

char description[64];
static void wma_about(void);
static void wma_init(void);
static int wma_is_our_file(char *filename);
static void wma_play_file(char *filename);
static void wma_stop(void);
static void wma_seek(int time);
static void wma_do_pause(short p);
static int wma_get_time(void);
static void wma_get_song_info(char *filename, char **title, int *length);
static void wma_file_info_box(char *filename); 
static void wma_set_eq(int q_on, float q_preamp, float *q_bands);
static char *wsong_title;
static int wsong_time;

InputPlugin *get_iplugin_info(void);

InputPlugin wma_ip =
{
	NULL,           	/* Filled in by xmms */
	NULL,           	/* Filled in by xmms */
    	description,    	/* The description that is shown in the preferences box */
    	wma_init,           	/* Called when the plugin is loaded */
    	wma_about,          	/* Show the about box */
    	NULL,  	    		/* Show the configure box */
    	wma_is_our_file,    	/* Return 1 if the plugin can handle the file */
    	NULL,           	/* Scan dir */
    	wma_play_file,      	/* Play file */
    	wma_stop,           	/* Stop */
    	wma_do_pause,       	/* Pause */
    	wma_seek,           	/* Seek */
    	wma_set_eq,         	/* Set the equalizer, most plugins won't be able to do this */
    	wma_get_time,       	/* Get the time, usually returns the output plugins output time */
    	NULL,           	/* Get volume */
    	NULL,           	/* Set volume */
    	NULL,           	/* OBSOLETE! */
    	NULL,           	/* OBSOLETE! */
    	NULL,           	/* Send data to the visualization plugins */
    	NULL,           	/* Fill in the stuff that is shown in the player window */
    	NULL,           	/* Show some text in the song title box. */
    	wma_get_song_info,  	/* Function to grab the title string */
    	wma_file_info_box,  	/* Bring up an info window for the filename passed in */
    	NULL            	/* Handle to the current output plugin. */
};

InputPlugin *get_iplugin_info(void)
{
	wma_ip.description = g_strdup_printf("WMA Audio Plugin");
	return &wma_ip;
}

static void wma_about(void) 
{
	static GtkWidget *aboutbox;
	gchar *text;
    
	if (aboutbox)
 		return;
	
	text = g_strdup_printf("WMA Plugin %s\n\n"
        		"Created by Roman Bogorodskiy <bogorodskiy@inbox.ru>\n"
		       	"Based on xmms-wma written by Mokrushin I.V. aka McMCC <mcmcc@mail.ru>\n"
			"See AUTHORS for details\n",
                        VERSION);

	aboutbox = xmms_show_message("About the WMA decoder",
                             	text,
                             	"OK", FALSE, NULL, NULL);
             
	g_free(text);
	gtk_signal_connect(GTK_OBJECT(aboutbox), "destroy",
                           GTK_SIGNAL_FUNC(gtk_widget_destroyed), &aboutbox);
}

static void wma_init(void)
{
    	
	avcodec_init();
    	avcodec_register_all();
    	av_register_all();
    	init_iir();
}

static int wma_is_our_file(char *filename)
{
	gchar *ext;
    	ext = strrchr(filename, '.');
    
	if (ext)
   		if (!strncasecmp(ext, ".wma", 4))
            		return TRUE;
    
	return FALSE;
}

static void wma_do_pause(short p)
{
	
	wma_pause = p;
    	wma_ip.output->pause(wma_pause);
}

static void wma_seek(int time) 
{
	wma_seekpos = time;
    	if (wma_pause)
		wma_ip.output->pause(0);
	
    	while (wma_decode && wma_seekpos!=-1) 
		xmms_usleep(10000);
	
    	if (wma_pause)
		wma_ip.output->pause(1);
}

static int wma_get_time(void)
{
	wma_ip.output->buffer_free();
    	if (wma_decode) 
		return wma_ip.output->output_time();
    
	return -1;
}

static void wma_set_eq(int q_on, float q_preamp, float *q_bands)
{
	int chn;
    	int index;
    	float value;

    	wma_eq_on = q_on;
    
	if (wma_eq_on) {
		q_preamp = q_preamp/1.6;
        	for (chn = 0; chn < c->channels; chn++)
            		preamp[chn] = 1.0 + 0.0932471 * q_preamp + 0.00279033 * q_preamp * q_preamp;
        
		for (index = 0; index < 10; index++) {
            		value = q_bands[index]/1.2;
            	
			for (chn = 0; chn < c->channels; chn++)
                		gain[index][chn] = 0.03 * value + 0.000999999 * value * value;
        	}
    	}
}

static gchar *extname(const char *filename)
{
	gchar *ext = strrchr(filename, '.');
    	
	if (ext != NULL)
        	++ext;
    	return ext;
}

static char* w_getstr(char* str)
{
	/* TODO
	 * Seems, this function was stolen from mpg123 xmms plugin,
	 * we need to get rid of it since I feel it is not useful)
	 */
	
	if (str && strlen(str) > 0)
        	return str;
    
    	return NULL;
}

static gchar *get_song_title(AVFormatContext *in, gchar * filename)
{
	gchar *ret = NULL;
    	TitleInput *input;

    	XMMS_NEW_TITLEINPUT(input);
    
    	if ((in->title[0] != '\0') || (in->author[0] != '\0') || (in->album[0] != '\0') ||
       		(in->comment[0] != '\0') || (in->genre[0] != '\0') || (in->year != 0) || (in->track != 0))
    	{	
		input->performer = w_getstr(in->author);
		input->album_name = w_getstr(in->album);
		input->track_name = w_getstr(in->title);
		input->year = in->year;
		input->track_number = in->track;
		input->genre = w_getstr(in->genre);
		input->comment = w_getstr(in->comment);
    	}
    
	input->file_name = (gchar *)g_basename(filename);
    	input->file_path = filename;
    	input->file_ext = extname(filename);
    	ret = xmms_get_titlestring(xmms_get_gentitle_format(), input);
    	
	g_free(input);

    	if (!ret) {
        	ret = g_strdup(g_basename(filename));
            
		if (extname(ret) != NULL)
                	*(extname(ret) - 1) = '\0';
    	}
    
	return ret;
}

static guint get_song_time(AVFormatContext *in)
{
	
	if (in->duration)
		return in->duration/1000;
    	else
		return 0;
}

static void wma_get_song_info(char *filename, char **title_real, int *len_real)
{
	AVFormatContext *in = NULL;
    
    	(*len_real) = -1;
    	(*title_real) = NULL;

    	if (av_open_input_file(&in, filename, NULL, 0, NULL) < 0)
		return;
	
    	av_find_stream_info(in);
    	(*len_real) = get_song_time(in);
    	(*title_real) = get_song_title(in, filename);
    	av_close_input_file(in);
}

static void wma_playbuff(int out_size)
{
	FifoBuffer f;
    	int sst_buff;
    
    	fifo_init(&f, out_size*2);
    	fifo_write(&f, wma_outbuf, out_size, &f.wptr);
    
	while (!fifo_read(&f, wma_s_outbuf, wma_st_buff, &f.rptr) && wma_decode) {
        	if (wma_eq_on)
            		sst_buff = iir((gpointer)&wma_s_outbuf, wma_st_buff);
        	else
	    		sst_buff = wma_st_buff;
		
		if (wma_pause)
	    		memset(wma_s_outbuf, 0, sst_buff);	
    	
		while (wma_ip.output->buffer_free() < wma_st_buff)
    			xmms_usleep(20000);
		
		if (wma_seekpos == -1)
			wma_ip.add_vis_pcm(wma_ip.output->written_time(), FMT_S16_NE,
            			    	c->channels, sst_buff, (short *)wma_s_outbuf);
	
		wma_ip.output->write_audio((short *)wma_s_outbuf, sst_buff);
		memset(wma_s_outbuf, 0, sst_buff);
    	}
    
	fifo_free(&f);
    	
	return;
}

static void *wma_play_loop(void *arg)
{
	uint8_t *inbuf_ptr;
    	int out_size, size, len;
    	AVPacket pkt;
    
    	pthread_mutex_lock(&wma_mutex);
    	
	while (wma_decode) {
		if (wma_seekpos != -1) {
	    		av_seek_frame(ic, wma_idx, wma_seekpos * 1000000LL);
	    		wma_ip.output->flush(wma_seekpos * 1000);
	    		wma_seekpos = -1;
		}

        	if (av_read_frame(ic, &pkt) < 0)
        		break;

        	size = pkt.size;
        	inbuf_ptr = pkt.data;
	
        	if (size == 0)
            		break;
	
        	while (size > 0) {
            		len = avcodec_decode_audio(c, (short *)wma_outbuf, 
					&out_size, inbuf_ptr, size);
	    	
			if (len < 0)
        			break;
	    
            		if (out_size <= 0)
                		continue;

	    		wma_playbuff(out_size);

            		size -= len;
            		inbuf_ptr += len;
            
			if (pkt.data)
                		av_free_packet(&pkt);
        	}
    	}
	
    	while (wma_decode && wma_ip.output->buffer_playing()) 
		xmms_usleep(30000);
    
	wma_decode = 0;
    
	/* XXX
	 * that's all odd 
	 */
	
	if (wma_s_outbuf)
        	g_free(wma_s_outbuf);
    
	if (wma_outbuf)
		g_free(wma_outbuf);
    
	if (pkt.data)
    		av_free_packet(&pkt);
    	if (c)
		avcodec_close(c);
    	if (ic)
        	av_close_input_file(ic);
    
	pthread_mutex_unlock(&wma_mutex);
    	pthread_exit(NULL);
}

static void wma_play_file(char *filename) 
{
	AVCodec *codec;
    
    	if (av_open_input_file(&ic, filename, NULL, 0, NULL) < 0)
		return;
    
    	for (wma_idx = 0; wma_idx < ic->nb_streams; wma_idx++) {
        	c = &ic->streams[wma_idx]->codec;
        	
		if(c->codec_type == CODEC_TYPE_AUDIO)
            	break;
 	}

    	av_find_stream_info(ic);

    	codec = avcodec_find_decoder(c->codec_id);

    	if (!codec)
		return;
	
    	if (avcodec_open(c, codec) < 0)
		return;
	    	    
    	wsong_title = get_song_title(ic, filename);
    	wsong_time = get_song_time(ic);

    	if (wma_ip.output->open_audio(FMT_S16_NE, c->sample_rate, c->channels) <= 0)
		return;

    	wma_st_buff  = ST_BUFF;
	
    	wma_ip.set_info(wsong_title, wsong_time, c->bit_rate, c->sample_rate, c->channels);

    	wma_s_outbuf = g_malloc0(wma_st_buff);
    	wma_outbuf = g_malloc0(AVCODEC_MAX_AUDIO_FRAME_SIZE);
    	wma_seekpos = -1;
    	wma_decode = 1;
    	pthread_create(&wma_decode_thread, NULL, wma_play_loop, NULL);
}

static void wma_stop(void) 
{
	wma_decode = 0;
    	
	if (wma_pause)
		wma_do_pause(0);
    	
	pthread_join(wma_decode_thread, NULL);
    	wma_ip.output->close_audio();
}	

static void wma_file_info_box (char *filename) 
{
	GtkWidget *dialog_vbox1;
  	GtkWidget *vbox1;
  	GtkWidget *hbox1;
  	GtkWidget *label_name;
  	GtkWidget *entry_filename;
  	GtkWidget *hbox2;
  	GtkWidget *frame_wma_info;
  	GtkWidget *alignment1;
  	GtkWidget *table1;
  	GtkWidget *label_album;
  	GtkWidget *label_year;
  	GtkWidget *label_track;
  	GtkWidget *label_genre;
  	GtkWidget *label_comments;
  	/*GtkWidget *label_copyright;*/
  	GtkWidget *label_wma_version;
  	GtkWidget *label_bitrate;
  	GtkWidget *label_rate;
 	GtkWidget *label_chans;
  	GtkWidget *label_play_time;
  	GtkWidget *label_filesize;
  	GtkWidget *label_wma_vers_val;
  	GtkWidget *label_bitrate_val;
  	GtkWidget *label_rate_val;
  	GtkWidget *label_chans_val;
  	GtkWidget *label_playtime_val;
  	GtkWidget *label_filesize_val;
  	GtkWidget *label4;
  	GtkWidget *frame_tags;
  	GtkWidget *alignment2;
  	GtkWidget *table2;
  	GtkWidget *label_artist;
  	GtkWidget *label_title;
  	GtkWidget *entry_artist;
  	GtkWidget *entry_album;
  	GtkWidget *entry_year;
  	GtkWidget *entry_title;
  	GtkWidget *entry_track;
  	GtkWidget *entry_genre;
  	GtkWidget *entry_comments;
  /*	GtkWidget *entry_copyright;*/
  	GtkWidget *label5;
  	GtkWidget *dialog_action_area1;
  	GtkWidget *okbutton;
	
	AVFormatContext *in = NULL;
    	AVCodecContext *s = NULL;
    	AVCodec *codec;
	gint tns, thh, tmm, tss;
	gint i;
	gchar *title,
	      *channels,
	      *bitrate,
	      *playtime,
	      *samplerate,
	      *filesize;
	VFSFile *f;

	if (dialog) {
		(void)g_printf("Info dialog is already opened!\n");
		return;
	}

	if (av_open_input_file(&in, filename, NULL, 0, NULL) < 0) 
		return;

    	for(i = 0; i < in->nb_streams; i++) {
        	s = &in->streams[i]->codec;
        	if(s->codec_type == CODEC_TYPE_AUDIO)
            		break;
    	}
	
    	av_find_stream_info(in);
    	codec = avcodec_find_decoder(s->codec_id);

	/* window title */
  	title = g_strdup_printf("File Info - %s", g_basename(filename));
  	
	/* channels */
    	if (s->channels == 1)
		channels = g_strdup("MONO");
	else
		channels = g_strdup("STEREO");

	/* bitrate */
	bitrate = g_strdup_printf("%d Kb/s", (s->bit_rate / 1000));
	
	/* playtime */
	if (in->duration != 0) {
        	tns = in->duration/1000000LL;
        	thh = tns/3600;
        	tmm = (tns%3600)/60;
        	tss = (tns%60);
		playtime = g_strdup_printf("%02d:%02d:%02d", thh, tmm, tss);
    	} else 
		playtime = g_strdup("N/A");

	/* samplerate */
	samplerate = g_strdup_printf("%d Hz", s->sample_rate);

	/* filesize */
	f = vfs_fopen(filename, "rb");

	if (f == NULL)
		return;

	vfs_fseek(f, 0, SEEK_END);
	filesize = g_strdup_printf("%lu Bytes", vfs_ftell(f));
	vfs_fclose(f);
	
  	dialog = gtk_dialog_new();
  	
	gtk_signal_connect(GTK_OBJECT(dialog), "destroy",
			GTK_SIGNAL_FUNC(gtk_widget_destroyed), &dialog);
	
	gtk_window_set_title(GTK_WINDOW(dialog), title);
  	gtk_window_set_type_hint(GTK_WINDOW(dialog), GDK_WINDOW_TYPE_HINT_DIALOG);
	gtk_window_set_policy(GTK_WINDOW(dialog), FALSE, FALSE, FALSE);
	
  	dialog_vbox1 = GTK_DIALOG(dialog)->vbox;
  	gtk_widget_show(dialog_vbox1);

  	vbox1 = gtk_vbox_new(FALSE, 0);
  	gtk_widget_show(vbox1);
  	gtk_box_pack_start(GTK_BOX(dialog_vbox1), vbox1, TRUE, TRUE, 0);

  	hbox1 = gtk_hbox_new (FALSE, 0);
  	gtk_widget_show (hbox1);
  	gtk_box_pack_start(GTK_BOX(vbox1), hbox1, FALSE, FALSE, 0);

  	label_name = gtk_label_new("<b>Name:</b>");
  	gtk_widget_show(label_name);
  	gtk_box_pack_start(GTK_BOX (hbox1), label_name, FALSE, FALSE, 0);
  	gtk_misc_set_alignment(GTK_MISC (label_name), 0.48, 0.51);
  	gtk_misc_set_padding(GTK_MISC (label_name), 10, 10);
	gtk_label_set_use_markup(GTK_LABEL(label_name), TRUE);
  
  	entry_filename = gtk_entry_new();
  	gtk_widget_show(entry_filename);
  	gtk_box_pack_start(GTK_BOX(hbox1), entry_filename, TRUE, TRUE, 4);
  	gtk_editable_set_editable(GTK_EDITABLE(entry_filename), FALSE);
	gtk_entry_set_text(GTK_ENTRY(entry_filename), filename);
  
  	hbox2 = gtk_hbox_new(FALSE, 0);
  	gtk_widget_show(hbox2);
  	gtk_box_pack_start(GTK_BOX(vbox1), hbox2, TRUE, TRUE, 0);

	frame_wma_info = gtk_frame_new(NULL);
  	gtk_widget_show(frame_wma_info);
  	gtk_box_pack_start(GTK_BOX(hbox2), frame_wma_info, TRUE, TRUE, 0);
  	gtk_frame_set_shadow_type(GTK_FRAME (frame_wma_info), GTK_SHADOW_ETCHED_IN);
	gtk_container_set_border_width (GTK_CONTAINER(frame_wma_info), 10);
	
  	alignment1 = gtk_alignment_new(0.5, 0.5, 1, 1);
  	gtk_widget_show(alignment1);
  	gtk_container_add(GTK_CONTAINER(frame_wma_info), alignment1);
  	gtk_alignment_set_padding(GTK_ALIGNMENT(alignment1), 0, 0, 0, 0);
  	gtk_container_set_border_width(GTK_CONTAINER(alignment1), 2);

  	table1 = gtk_table_new(6, 2, FALSE);
  	gtk_widget_show(table1);
  	gtk_container_add(GTK_CONTAINER(alignment1), table1);
  	gtk_container_set_border_width(GTK_CONTAINER(table1), 6);
  	gtk_table_set_row_spacings(GTK_TABLE(table1), 3);
  	gtk_table_set_col_spacings(GTK_TABLE(table1), 8);

  	/* WMA Version label */
	label_wma_version = gtk_label_new("<b>WMA Version:</b>");
  	gtk_widget_show(label_wma_version);
  	gtk_table_attach(GTK_TABLE(table1), label_wma_version, 0, 1, 0, 1,
                    	(GtkAttachOptions) (GTK_FILL),
                    	(GtkAttachOptions) (0), 0, 0);
  	gtk_misc_set_alignment(GTK_MISC(label_wma_version), 0, 0.5);
	gtk_label_set_use_markup(GTK_LABEL(label_wma_version), TRUE);

	/* Bitrate */
  	label_bitrate = gtk_label_new("<b>Bitrate:</b>");
  	gtk_widget_show(label_bitrate);
  	gtk_table_attach(GTK_TABLE(table1), label_bitrate, 0, 1, 1, 2,
                    	(GtkAttachOptions) (GTK_FILL),
                    	(GtkAttachOptions) (0), 0, 0);
  	gtk_misc_set_alignment(GTK_MISC(label_bitrate), 0, 0.5);
  	gtk_label_set_use_markup(GTK_LABEL(label_bitrate), TRUE);

	/* Samplerate */
  	label_rate = gtk_label_new("<b>Samplerate:</b>");
  	gtk_widget_show(label_rate);
  	gtk_table_attach(GTK_TABLE(table1), label_rate, 0, 1, 2, 3,
                    	(GtkAttachOptions) (GTK_FILL),
                    	(GtkAttachOptions) (0), 0, 0);
  	gtk_misc_set_alignment(GTK_MISC(label_rate), 0, 0.5);
  	gtk_label_set_use_markup(GTK_LABEL(label_rate), TRUE);

	/* Channels */
  	label_chans = gtk_label_new("<b>Channels:</b>");
  	gtk_widget_show(label_chans);
  	gtk_table_attach(GTK_TABLE (table1), label_chans, 0, 1, 3, 4,
                    	(GtkAttachOptions) (GTK_FILL),
                    	(GtkAttachOptions) (0), 0, 0);
  	gtk_misc_set_alignment(GTK_MISC(label_chans), 0, 0.5);
  	gtk_label_set_use_markup(GTK_LABEL(label_chans), TRUE);

	/* Play time */
  	label_play_time = gtk_label_new("<b>Play time:</b>");
  	gtk_widget_show(label_play_time);
  	gtk_table_attach(GTK_TABLE (table1), label_play_time, 0, 1, 4, 5,
                    	(GtkAttachOptions) (GTK_FILL),
                    	(GtkAttachOptions) (0), 0, 0);
  	gtk_misc_set_alignment(GTK_MISC(label_play_time), 0, 0.5);
  	gtk_label_set_use_markup(GTK_LABEL(label_play_time), TRUE);

	/* Filesize */
  	label_filesize = gtk_label_new("<b>Filesize:</b>");
  	gtk_widget_show(label_filesize);
  	gtk_table_attach(GTK_TABLE(table1), label_filesize, 0, 1, 5, 6,
                    	(GtkAttachOptions) (GTK_FILL),
                    	(GtkAttachOptions) (0), 0, 0);
  	gtk_misc_set_alignment(GTK_MISC(label_filesize), 0, 0.5);
  	gtk_label_set_use_markup(GTK_LABEL(label_filesize), TRUE);

  	
	label_wma_vers_val = gtk_label_new(codec->name);
  	gtk_widget_show(label_wma_vers_val);
  	gtk_table_attach(GTK_TABLE(table1), label_wma_vers_val, 1, 2, 0, 1,
                    	(GtkAttachOptions)(GTK_FILL),
                    	(GtkAttachOptions) (0), 0, 0);
  	gtk_misc_set_alignment(GTK_MISC(label_wma_vers_val), 0, 0.5);

  	label_bitrate_val = gtk_label_new(bitrate);
  	gtk_widget_show(label_bitrate_val);
  	gtk_table_attach(GTK_TABLE(table1), label_bitrate_val, 1, 2, 1, 2,
                    (GtkAttachOptions)(GTK_FILL),
                    (GtkAttachOptions) (0), 0, 0);
  	gtk_misc_set_alignment(GTK_MISC(label_bitrate_val), 0, 0.5);

  	label_rate_val = gtk_label_new(samplerate);
  	gtk_widget_show(label_rate_val);
  	gtk_table_attach(GTK_TABLE(table1), label_rate_val, 1, 2, 2, 3,
                    (GtkAttachOptions)(GTK_FILL),
                    (GtkAttachOptions) (0), 0, 0);
  	gtk_misc_set_alignment(GTK_MISC(label_rate_val), 0, 0.5);

  	label_chans_val = gtk_label_new(channels);
  	gtk_widget_show(label_chans_val);
  	gtk_table_attach(GTK_TABLE(table1), label_chans_val, 1, 2, 3, 4,
                    	(GtkAttachOptions)(GTK_FILL),
                    	(GtkAttachOptions) (0), 0, 0);
  	gtk_misc_set_alignment(GTK_MISC (label_chans_val), 0, 0.5);

  	label_playtime_val = gtk_label_new(playtime);
  	gtk_widget_show(label_playtime_val);
  	gtk_table_attach(GTK_TABLE(table1), label_playtime_val, 1, 2, 4, 5,
                    (GtkAttachOptions)(GTK_FILL),
                    (GtkAttachOptions) (0), 0, 0);
  	gtk_misc_set_alignment(GTK_MISC(label_playtime_val), 0, 0.5);

  	label_filesize_val = gtk_label_new(filesize);
  	gtk_widget_show(label_filesize_val);
  	gtk_table_attach(GTK_TABLE (table1), label_filesize_val, 1, 2, 5, 6,
                    	(GtkAttachOptions)(GTK_FILL),
                    	(GtkAttachOptions) (0), 0, 0);
  	gtk_misc_set_alignment(GTK_MISC(label_filesize_val), 0, 0.5);

  	label4 = gtk_label_new ("WMA Info");
  	gtk_widget_show(label4);
  	gtk_frame_set_label_widget(GTK_FRAME(frame_wma_info), label4);

  	frame_tags = gtk_frame_new (NULL);
  	gtk_widget_show (frame_tags);
  	gtk_box_pack_start (GTK_BOX (hbox2), frame_tags, TRUE, TRUE, 0);
  	gtk_frame_set_shadow_type (GTK_FRAME (frame_tags), GTK_SHADOW_ETCHED_IN);
	gtk_container_set_border_width (GTK_CONTAINER (frame_tags), 10);

  
  	alignment2 = gtk_alignment_new (0.5, 0.5, 1, 1);
  	gtk_widget_show (alignment2);
 	gtk_container_add (GTK_CONTAINER (frame_tags), alignment2);
  	gtk_alignment_set_padding (GTK_ALIGNMENT (alignment2), 0, 0, 12, 0);
	gtk_container_set_border_width (GTK_CONTAINER (alignment2), 2);

  
  	table2 = gtk_table_new(8, 2, FALSE);
  	gtk_widget_show(table2);
  	gtk_container_add(GTK_CONTAINER(alignment2), table2);
  	gtk_container_set_border_width(GTK_CONTAINER(table2), 6);
  	gtk_table_set_row_spacings(GTK_TABLE(table2), 3);
  	gtk_table_set_col_spacings(GTK_TABLE(table2), 8);

  	/* Artist */
  	label_artist = gtk_label_new("<b>Artist:</b>");
  	gtk_widget_show(label_artist);
  	gtk_table_attach(GTK_TABLE (table2), label_artist, 0, 1, 0, 1,
                    (GtkAttachOptions) (GTK_FILL),
                    (GtkAttachOptions) (0), 0, 0);
  	gtk_misc_set_alignment(GTK_MISC(label_artist), 0, 0.5);
	gtk_label_set_use_markup(GTK_LABEL(label_artist), TRUE);

	/* Title */
  	label_title = gtk_label_new("<b>Title:</b>");
  	gtk_widget_show(label_title);
  	gtk_table_attach(GTK_TABLE (table2), label_title, 0, 1, 1, 2,
                    (GtkAttachOptions) (GTK_FILL),
                    (GtkAttachOptions) (0), 0, 0);
  	gtk_misc_set_alignment(GTK_MISC(label_title), 0, 0.5);
	gtk_label_set_use_markup(GTK_LABEL(label_title), TRUE);

	/* Album */
	label_album = gtk_label_new("<b>Album:</b>");
	gtk_widget_show(label_album);
	gtk_table_attach(GTK_TABLE (table2), label_album, 0, 1, 2, 3,
			(GtkAttachOptions) (GTK_FILL),
			(GtkAttachOptions) (0), 0, 0);
	gtk_misc_set_alignment(GTK_MISC(label_album), 0, 0.5);
	gtk_label_set_use_markup(GTK_LABEL(label_album), TRUE);
			
	/* Comments */
	label_comments = gtk_label_new("<b>Comments:</b>");
	gtk_widget_show(label_comments);
	gtk_table_attach(GTK_TABLE(table2), label_comments, 0, 1, 3, 4,
			(GtkAttachOptions) (GTK_FILL),
			(GtkAttachOptions) (0), 0, 0);
        gtk_misc_set_alignment(GTK_MISC(label_comments), 0, 0.5);
	gtk_label_set_use_markup(GTK_LABEL(label_comments), TRUE);
	
	/* Year */
	label_year = gtk_label_new("<b>Year:</b>");
	gtk_widget_show(label_year);
	gtk_table_attach(GTK_TABLE (table2), label_year, 0, 1, 4, 5,
			(GtkAttachOptions) (GTK_FILL),
			(GtkAttachOptions) (0), 0, 0);
        gtk_misc_set_alignment(GTK_MISC(label_year), 0, 0.5);
	gtk_label_set_use_markup(GTK_LABEL(label_year), TRUE);

	/* Track */
	label_track = gtk_label_new("<b>Track:</b>");
	gtk_widget_show(label_track);
	gtk_table_attach(GTK_TABLE (table2), label_track, 0, 1, 5, 6,
			(GtkAttachOptions) (GTK_FILL),
			(GtkAttachOptions) (0), 0, 0);
        gtk_misc_set_alignment(GTK_MISC(label_track), 0, 0.5);
	gtk_label_set_use_markup(GTK_LABEL(label_track), TRUE);

	/* Genre */
	label_genre = gtk_label_new("<b>Genre:</b>");
	gtk_widget_show(label_genre);
	gtk_table_attach(GTK_TABLE (table2), label_genre, 0, 1, 6, 7,
			(GtkAttachOptions) (GTK_FILL),
			(GtkAttachOptions) (0), 0, 0);
        gtk_misc_set_alignment(GTK_MISC (label_genre), 0, 0.5);
	gtk_label_set_use_markup(GTK_LABEL(label_genre), TRUE);
		
  
  	entry_artist = gtk_entry_new();
  	gtk_widget_show (entry_artist);
  	gtk_table_attach (GTK_TABLE (table2), entry_artist, 1, 2, 0, 1,
                    (GtkAttachOptions) (GTK_EXPAND | GTK_FILL),
                    (GtkAttachOptions) (0), 0, 0);
 	gtk_editable_set_editable (GTK_EDITABLE (entry_artist), FALSE);
	gtk_entry_set_text(GTK_ENTRY(entry_artist), in->author);
  
  	entry_title = gtk_entry_new();
  	gtk_widget_show(entry_title);
  	gtk_table_attach (GTK_TABLE (table2), entry_title, 1, 2, 1, 2,
                    (GtkAttachOptions) (GTK_EXPAND | GTK_FILL),
                    (GtkAttachOptions) (0), 0, 0);
  	gtk_editable_set_editable(GTK_EDITABLE (entry_title), FALSE);
	gtk_entry_set_text(GTK_ENTRY(entry_title), in->title);

	entry_album = gtk_entry_new();
	gtk_widget_show(entry_album);
	gtk_table_attach(GTK_TABLE (table2), entry_album, 1, 2, 2, 3,
			(GtkAttachOptions) (GTK_EXPAND | GTK_FILL),
			(GtkAttachOptions) (0), 0, 0);
        gtk_editable_set_editable(GTK_EDITABLE (entry_album), FALSE);
	gtk_entry_set_text(GTK_ENTRY(entry_album), in->album);

	entry_comments = gtk_entry_new();
	gtk_widget_show(entry_comments);
	gtk_table_attach(GTK_TABLE (table2), entry_comments, 1, 2, 3, 4,
			(GtkAttachOptions) (GTK_EXPAND | GTK_FILL),
			(GtkAttachOptions) (0), 0, 0);
        gtk_editable_set_editable(GTK_EDITABLE (entry_comments), FALSE);
	gtk_entry_set_text(GTK_ENTRY(entry_comments), in->comment);

	entry_year = gtk_entry_new();
	gtk_widget_show(entry_year);
	gtk_table_attach(GTK_TABLE (table2), entry_year, 1, 2, 4, 5,
			(GtkAttachOptions) (GTK_EXPAND | GTK_FILL),
			(GtkAttachOptions) (0), 0, 0);
        gtk_editable_set_editable(GTK_EDITABLE (entry_year), FALSE);
	gtk_entry_set_text(GTK_ENTRY(entry_year), g_strdup_printf("%d", in->year));

	entry_track = gtk_entry_new();
	gtk_widget_show(entry_track);
	gtk_table_attach(GTK_TABLE (table2), entry_track, 1, 2, 5, 6,
			(GtkAttachOptions) (GTK_EXPAND | GTK_FILL),
	                (GtkAttachOptions) (0), 0, 0);
        gtk_editable_set_editable(GTK_EDITABLE (entry_track), FALSE);
	gtk_entry_set_text(GTK_ENTRY(entry_track), g_strdup_printf("%d", in->track));

	entry_genre = gtk_entry_new();
	gtk_widget_show(entry_genre);
	gtk_table_attach(GTK_TABLE (table2), entry_genre, 1, 2, 6, 7,
			(GtkAttachOptions) (GTK_EXPAND | GTK_FILL),
			(GtkAttachOptions) (0), 0, 0);
        gtk_editable_set_editable(GTK_EDITABLE (entry_genre), FALSE);
	gtk_entry_set_text(GTK_ENTRY(entry_genre), in->genre);
		
		
  	label5 = gtk_label_new("Tags");
  	gtk_widget_show(label5);
  	gtk_frame_set_label_widget(GTK_FRAME(frame_tags), label5);
  

  	dialog_action_area1 = GTK_DIALOG(dialog)->action_area;
  	gtk_widget_show(dialog_action_area1);
  	gtk_button_box_set_layout(GTK_BUTTON_BOX(dialog_action_area1), GTK_BUTTONBOX_END);

  	okbutton = gtk_button_new_from_stock("gtk-ok");
  	gtk_widget_show(okbutton);
  	gtk_dialog_add_action_widget(GTK_DIALOG(dialog), okbutton, GTK_RESPONSE_OK);
  	GTK_WIDGET_SET_FLAGS(okbutton, GTK_CAN_DEFAULT);

       	gtk_signal_connect_object(GTK_OBJECT(okbutton), "clicked",
			GTK_SIGNAL_FUNC(gtk_widget_destroy), GTK_OBJECT(dialog));
  
	gtk_widget_show(dialog);

      	g_free(title);
	g_free(channels);
	g_free(bitrate);
	g_free(playtime);
	g_free(samplerate);
	g_free(filesize);	
}