/**
 * GMyth Library
 *
 * @file gmyth/mmyth_tvplayer.c
 * 
 * @brief <p> This component provides playback of the remote A/V using
 * GStreamer.
 * 
 * Copyright (C) 2006 INdT - Instituto Nokia de Tecnologia.
 * @author Hallyson Luiz de Morais Melo <hallyson.melo@indt.org.br>
 *
 *//*
 * 
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU Lesser 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 Lesser 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 "mmyth_tvplayer.h"
 
#include <gdk/gdkx.h>

#include "gmyth_context.h"
#include "gmyth_remote_util.h"

typedef struct _GstPlayerWindowStateChange
{
    GstElement *play;
    GstState old_state, new_state;
    MMythTVPlayer *tvplayer;
} GstPlayerWindowStateChange;

typedef struct _GstPlayerWindowTagFound
{
    GstElement *play;
    GstTagList *taglist;
    MMythTVPlayer *tvplayer;
} GstPlayerWindowTagFound;

/*
static gboolean idle_state (gpointer data);
*/
static gboolean bus_call (GstBus * bus, GstMessage * msg, gpointer data);

static void mmyth_tvplayer_class_init          (MMythTVPlayerClass *klass);
static void mmyth_tvplayer_init                (MMythTVPlayer *object);

static void mmyth_tvplayer_dispose  (GObject *object);
static void mmyth_tvplayer_finalize (GObject *object);

G_DEFINE_TYPE(MMythTVPlayer, mmyth_tvplayer, G_TYPE_OBJECT)

static gboolean mmyth_tvplayer_create_pipeline (MMythTVPlayer* tvplayer);
static void     new_pad_cb (GstElement *element, 
                            GstPad *pad, gpointer data);

static gboolean expose_cb (GtkWidget * widget, 
                           GdkEventExpose * event, 
                           gpointer user_data);

static void
mmyth_tvplayer_class_init (MMythTVPlayerClass *klass)
{
	GObjectClass *gobject_class;

    gobject_class = (GObjectClass *) klass;
	
    gobject_class->dispose  = mmyth_tvplayer_dispose;
    gobject_class->finalize = mmyth_tvplayer_finalize;	
}

static void
new_pad_cb (GstElement *element, GstPad *pad, gpointer data)
{
	MMythTVPlayer *tvplayer = MMYTH_TVPLAYER (data);
	GstPadLinkReturn ret;
	char *s;
	
	s = gst_caps_to_string (pad->caps);

	if ( s[0] == 'a') {
		ret = gst_pad_link (pad, gst_element_get_pad (tvplayer->audioqueue1, "sink"));
	} else {
		ret = gst_pad_link (pad, gst_element_get_pad (tvplayer->videoqueue1, "sink")); 
	}
	
	g_free(s);
}

static gboolean
expose_cb (GtkWidget * widget, GdkEventExpose * event, gpointer user_data)
{
	MMythTVPlayer *tvplayer = MMYTH_TVPLAYER (user_data);

	if (tvplayer && tvplayer->videow) {
		gst_x_overlay_set_xwindow_id (GST_X_OVERLAY (tvplayer->gst_videosink),
				GDK_WINDOW_XWINDOW (widget->window));
		return TRUE;
	}

	g_warning ("MMythTVPlayer expose called before setting video window\n");

	return FALSE;
}

static void
mmyth_tvplayer_init (MMythTVPlayer *tvplayer)
{
	tvplayer->gst_pipeline = NULL;
	tvplayer->gst_source = NULL;
	tvplayer->gst_videodec = NULL;
	tvplayer->gst_videosink = NULL;
	tvplayer->gst_videocolortrs = NULL;
	tvplayer->videoqueue1 = NULL;
	tvplayer->videoqueue2 = NULL;
	tvplayer->audioqueue1 = NULL;
	tvplayer->audioqueue2 = NULL;   

	/* GTKWidget for rendering the video */
	tvplayer->videow = NULL;
	tvplayer->expose_handler = 0;

	tvplayer->backend_hostname = NULL;
	tvplayer->backend_port = 0;
	tvplayer->local_hostname = NULL;

	tvplayer->recorder = NULL;
	tvplayer->tvchain = NULL;
	tvplayer->proginfo = NULL;
}

static void
mmyth_tvplayer_dispose (GObject *object)
{

	G_OBJECT_CLASS (mmyth_tvplayer_parent_class)->dispose (object);
}

static void
mmyth_tvplayer_finalize (GObject *object)
{
	g_signal_handlers_destroy (object);

	MMythTVPlayer *tvplayer = MMYTH_TVPLAYER (object);

	g_debug ("[%s] Finalizing tvplayer", __FUNCTION__);
	
	if (tvplayer->videow != NULL) {
		if (g_signal_handler_is_connected (tvplayer->videow, 
						tvplayer->expose_handler)) {
			g_signal_handler_disconnect (tvplayer->videow,
				tvplayer->expose_handler);
		}
		g_object_unref (tvplayer->videow);
	}
	
	if ( tvplayer->recorder != NULL )
		g_object_unref (tvplayer->recorder);
	if ( tvplayer->tvchain != NULL )
		g_object_unref (tvplayer->tvchain);
	if ( tvplayer->proginfo != NULL )
		g_object_unref (tvplayer->proginfo);	

	// Release Gstreamer elements
	if ( tvplayer->gst_pipeline != NULL )
		g_object_unref (tvplayer->gst_pipeline);	
	if ( tvplayer->gst_source != NULL )
		g_object_unref (tvplayer->gst_source);	
	if ( tvplayer->gst_videodec != NULL )
		g_object_unref (tvplayer->gst_videodec);	
	if ( tvplayer->gst_videocolortrs != NULL )
		g_object_unref (tvplayer->gst_videocolortrs);
	if ( tvplayer->gst_videosink != NULL )
		g_object_unref (tvplayer->gst_videosink);	
	if ( tvplayer->videoqueue1 != NULL )
		g_object_unref (tvplayer->videoqueue1);
	if ( tvplayer->videoqueue2 != NULL )
		g_object_unref (tvplayer->videoqueue2);
	if ( tvplayer->audioqueue1 != NULL )
		g_object_unref (tvplayer->audioqueue1);	
	if ( tvplayer->audioqueue2 != NULL )
		g_object_unref (tvplayer->audioqueue2);	

	G_OBJECT_CLASS (mmyth_tvplayer_parent_class)->finalize (object);
}

/** Creates a new instance of MMythTVPlayer. 
 * 
 * @return a new instance of MMythTVPlayer.
 */
MMythTVPlayer *
mmyth_tvplayer_new ()
{
    MMythTVPlayer *tvplayer = 
        MMYTH_TVPLAYER (g_object_new(MMYTH_TVPLAYER_TYPE, NULL));
    
    return tvplayer;
}

/** Initializes the tv player.
 *
 * @param tvplayer the object instance.
 * @return gboolean TRUE if the pipeline was created 
 * successfully, FALSE otherwise.
 */
gboolean
mmyth_tvplayer_initialize (MMythTVPlayer *tvplayer)
{
	
    if (!mmyth_tvplayer_create_pipeline (tvplayer)) {
    	g_warning ("[%s] Error while creating pipeline. TV Player not initialized", __FUNCTION__);
		return FALSE;
	} else {
		g_debug ("[%s] GStreamer pipeline created", __FUNCTION__);	
    }

	return TRUE;
}

/** Creates the GStreamer pipeline used by the player.
 *
 * @param tvplayer the object instance.
 * @return gboolean TRUE if the pipeline was created 
 * successfully, FALSE otherwise.
 */
static gboolean
mmyth_tvplayer_create_pipeline (MMythTVPlayer* tvplayer)
{
    GstElement *pipeline;
    GstElement *source, *parser;
    GstElement *videodec, *videosink;
    GstElement *videocolortrs;
#ifndef MAEMO_PLATFORM    
    GstElement *audiodec, *audioconv;
#endif
    GstElement *audiosink;
    GstElement *videoqueue1, *videoqueue2, *audioqueue1, *audioqueue2;

    g_debug ("MMythTVPlayer: Setting the Gstreamer pipeline\n");
	
    pipeline = gst_pipeline_new ("video-player");
    source = gst_element_factory_make ("mythtvsrc", "myth-source");
    parser = gst_element_factory_make ("nuvdemux", "nuv-demux");

    /* Gstreamer Video elements */
    videoqueue1 = gst_element_factory_make ("queue", "video-queue1");
    videodec = gst_element_factory_make ("divxdec", "video-decoder");
    videoqueue2 = gst_element_factory_make ("queue", "video-queue2");
    videocolortrs = gst_element_factory_make ("ffmpegcolorspace", "image-color-transforms");

#ifdef MAEMO_PLATFORM
    videosink = gst_element_factory_make ("sdlvideosink", "image-output");
#else
    videosink = gst_element_factory_make ("xvimagesink", "image-output");
#endif
    
    /* Gstreamer Audio elements */
    audioqueue1 = gst_element_factory_make ("queue", "audio-queue1");    
    audioqueue2 = gst_element_factory_make ("queue", "audio-queue2");
#ifdef MAEMO_PLATFORM    
    audiosink = gst_element_factory_make ("dspmp3sink", "audio-output");
#else    
    audiodec = gst_element_factory_make ("mad", "audio-decoder");
    audioconv = gst_element_factory_make ("audioconvert", "audio-converter");
    audiosink = gst_element_factory_make ("alsasink", "audio-output");
#endif    
    
    if (!(pipeline && source && parser && videodec && videosink) ||
    	!(videoqueue1 && videoqueue2 && audioqueue1 && audioqueue2 && audiosink)) {
        /* FIXME: hanlde the error correctly */
        /* video_alignment is not being created (below) 
           and is causing problems to the ui */

	    tvplayer->gst_pipeline = NULL;
	    tvplayer->gst_videodec = NULL;
	    tvplayer->gst_videosink = NULL;
	    tvplayer->gst_videocolortrs = NULL;
           
        g_warning ("GstElement creation error!\n");
        return FALSE;
    }

#ifndef MAEMO_PLATFORM    
    if (!(audiodec && audioconv)) {
        g_warning ("GstElement for audio stream creation error!");
        return FALSE;
    }
#endif    
    
    tvplayer->gst_pipeline = pipeline;
    tvplayer->gst_source = source;
    tvplayer->gst_videodec = videodec;
    tvplayer->gst_videosink = videosink;
    tvplayer->gst_videocolortrs = videocolortrs;
    g_object_ref (tvplayer->gst_pipeline);
    g_object_ref (tvplayer->gst_source);
    g_object_ref (tvplayer->gst_videodec);
    g_object_ref (tvplayer->gst_videosink);
    g_object_ref (tvplayer->gst_videocolortrs);

    tvplayer->videoqueue1 = videoqueue1;
    tvplayer->videoqueue2 = videoqueue2;
    tvplayer->audioqueue1 = audioqueue1;
    tvplayer->audioqueue2 = audioqueue2;
    g_object_ref (tvplayer->videoqueue1);
    g_object_ref (tvplayer->videoqueue2);
    g_object_ref (tvplayer->audioqueue1);
    g_object_ref (tvplayer->audioqueue2);
  	
    g_object_set (G_OBJECT (videosink), "sync", TRUE, NULL);
    g_object_set (G_OBJECT (audiosink), "sync", FALSE, NULL);

    gst_bus_add_watch (gst_pipeline_get_bus (GST_PIPELINE (tvplayer->gst_pipeline)),
                       bus_call, tvplayer);

    gst_bin_add_many (GST_BIN (pipeline), source, parser, videoqueue1,
    			videodec, videoqueue2, videocolortrs, videosink, audioqueue1, 
			audiodec, audioconv, audioqueue2, audiosink, NULL);

    {
//        GstCaps *rtpcaps = gst_caps_new_simple ("application/x-rtp", NULL);
//        gst_element_link_filtered(source, parser, rtpcaps);
    }
    
    gst_element_link (source, parser);
    gst_element_link_many (videoqueue1, videodec, videoqueue2, videocolortrs, videosink, NULL);
    gst_element_link_many (audioqueue1, audiodec, audioconv, audioqueue2, audiosink, NULL);
    
    g_signal_connect (parser, "pad-added", G_CALLBACK (new_pad_cb), tvplayer);
    
    return TRUE;
}

/** Configures the backend and the tv player 
 * for playing the recorded content A/V.
 *
 * FIXME: Change filename to program info or other structure about the recorded
 *
 * @param tvplayer the object instance.
 * @param filename the file uri of the recorded content to be played.
 * @return TRUE if successfull, FALSE if any error happens.
 */
gboolean
mmyth_tvplayer_record_setup (MMythTVPlayer *tvplayer, gchar *filename)
{
	GMythSettings *msettings = gmyth_context_get_settings();
	
	// FIXME: we should receive the uri instead of filename
	GString *hostname = gmyth_settings_get_backend_hostname (msettings);
	int port = gmyth_settings_get_backend_port(msettings);
	
	GString *fullpath = g_string_new ("myth://");
	g_string_append_printf (fullpath, "%s:%d/%s", hostname->str, port, filename);
	
	tvplayer->is_livetv = FALSE;

	g_debug ("[%s] Setting record uri to gstreamer pipeline to %s", __FUNCTION__, fullpath->str);
		
    g_object_set (G_OBJECT (tvplayer->gst_source), "location",
          fullpath->str, NULL);
          
    return TRUE;
}

/** Configures the backend and the tv player 
 * for playing the live tv.
 *
 * @param tvplayer the object instance.
 * @return TRUE if successfull, FALSE if any error happens.
 */
gboolean
mmyth_tvplayer_livetv_setup (MMythTVPlayer *tvplayer)
{
	GMythSettings *msettings = gmyth_context_get_settings ();
	gboolean res = TRUE;

	res = gmyth_context_check_connection();
	if (!res) {
		g_warning ("[%s] LiveTV can not connect to backend", __FUNCTION__);	
		res = FALSE;
		goto error;
	}

	tvplayer->backend_hostname = gmyth_settings_get_backend_hostname(msettings);
	tvplayer->backend_port = gmyth_settings_get_backend_port (msettings);

	tvplayer->local_hostname  = g_string_new("");    
	gmyth_context_get_local_hostname (tvplayer->local_hostname);

	if ( tvplayer->local_hostname == NULL ) {
		res = FALSE;
		goto error;
	}

	// Gets the remote encoder num
	tvplayer->recorder = remote_request_next_free_recorder (-1);

	if ( tvplayer->recorder == NULL ) {
		g_warning ("[%s] None remote encoder available", __FUNCTION__);
		res = FALSE;
		goto error;
	}

	// Creates livetv chain handler
	tvplayer->tvchain = GMYTH_TVCHAIN ( g_object_new(GMYTH_TVCHAIN_TYPE, NULL) );
	gmyth_tvchain_initialize ( tvplayer->tvchain, tvplayer->local_hostname );

	if ( tvplayer->tvchain == NULL || tvplayer->tvchain->tvchain_id == NULL ) {
		res = FALSE;
		goto error;
	}

	// Init remote encoder. Opens its control socket.
	res = gmyth_recorder_setup(tvplayer->recorder);
	if ( !res ) {
		g_warning ("[%s] Fail while setting remote encoder\n", __FUNCTION__);
		res = FALSE;
		goto error;
	}
	// Spawn live tv. Uses the socket to send mythprotocol data to start livetv in the backend (remotelly)
	res = gmyth_recorder_spawntv ( tvplayer->recorder,
			gmyth_tvchain_get_id(tvplayer->tvchain) );
	if (!res) {
		g_warning ("[%s] Fail while spawn tv\n", __FUNCTION__);
		res = FALSE;
		goto error;
	}

	// Reload all TV chain from Mysql database.
	gmyth_tvchain_reload_all (tvplayer->tvchain);

	if ( tvplayer->tvchain == NULL ) {
		res = FALSE;
		goto error;
	}

	// Get program info from database using chanid and starttime
	tvplayer->proginfo = gmyth_tvchain_get_program_at (tvplayer->tvchain, -1);
	if ( tvplayer->proginfo == NULL ) {
		g_warning ("[%s] LiveTV not successfully started.\n", __FUNCTION__ );
		res = FALSE;
		goto error;
	} else {
		g_debug ("[%s] MythLiveTV: All requests to backend to start TV were OK.\n", __FUNCTION__ );
	}

	return res;

error:
	if ( tvplayer->backend_hostname != NULL ) {
		g_string_free( tvplayer->backend_hostname, TRUE );
		res = FALSE;
	}

	if ( tvplayer->local_hostname != NULL ) {
		g_string_free( tvplayer->local_hostname, TRUE );
		res = FALSE;
	}

	if ( tvplayer->recorder != NULL ) {
		g_object_unref (tvplayer->recorder);
		tvplayer->recorder = NULL;
	}

	if ( tvplayer->tvchain != NULL ) {
		g_object_unref (tvplayer->tvchain);
		tvplayer->tvchain = NULL;
	}

	if ( tvplayer->proginfo != NULL ) {
		g_object_unref (tvplayer->proginfo);
		tvplayer->proginfo = NULL;
	}

	return res;

}

/** Sets the GTK video widget for the tv player. 
 *
 * @param tvplayer the object instance.
 * @param videow the GTK video window.
 * @return TRUE if successfull, FALSE if any error happens.
 */
gboolean
mmyth_tvplayer_set_widget (MMythTVPlayer *tvplayer, GtkWidget *videow)
{
	tvplayer->videow = videow;
	g_object_ref (videow);
	
	g_debug ("[%s] Setting widget for tv player render", __FUNCTION__);
	
    tvplayer->expose_handler = g_signal_connect (tvplayer->videow, "expose-event", 
                                                 G_CALLBACK (expose_cb), tvplayer);

    //g_signal_connect(miptv_ui->videow, "size_request", G_CALLBACK(cb_preferred_video_size), miptv_ui);

    return TRUE;
}

static gboolean
bus_call (GstBus * bus, GstMessage * msg, gpointer data)
{
    //MMythTVPlayer *tvplayer = MMYTH_TVPLAYER ( data );
    //GMainLoop *loop = tvplayer->loop;

    switch (GST_MESSAGE_TYPE (msg)) {
        case GST_MESSAGE_EOS:
			printf ("End of stream\n");
            //g_idle_add ((GSourceFunc) idle_eos, data);
            gst_element_set_state ( GST_ELEMENT (GST_MESSAGE_SRC (msg)), GST_STATE_NULL );
	    gst_element_set_locked_state ( GST_ELEMENT (GST_MESSAGE_SRC (msg)), TRUE );
            break;
        case GST_MESSAGE_ERROR:
        {
                gchar *debug;
                GError *err;

                gst_message_parse_error (msg, &err, &debug);
                g_free (debug);

                printf ("Error: %s\n", err->message);
                g_error_free (err);

                //g_main_loop_quit (loop);
        }
            break;
        default:
            printf (gst_message_type_get_name (GST_MESSAGE_TYPE (msg)));
            printf ("\n");
            break;
    }

    return TRUE;
}


#if 0
static gboolean
idle_state (gpointer data)
{
    GstPlayerWindowStateChange *st = data;

    if (st->old_state == GST_STATE_PLAYING) {
        if (st->miptv_ui->idle_id != 0) {
            g_source_remove (st->miptv_ui->idle_id);
            st->miptv_ui->idle_id = 0;
        }
    }
    else if (st->new_state == GST_STATE_PLAYING) {
        if (st->miptv_ui->idle_id != 0)
            g_source_remove (st->miptv_ui->idle_id);

        st->miptv_ui->idle_id = g_idle_add (cb_iterate, st->miptv_ui);
    }

    /* new movie loaded? */
    if (st->old_state == GST_STATE_READY && st->new_state > GST_STATE_READY) {

        /* gboolean have_video = FALSE; */

        gtk_widget_show (st->miptv_ui->videow);

        gtk_window_resize (GTK_WINDOW (st->miptv_ui->main_window), 1, 1);

    }

    /* discarded movie? */
    if (st->old_state > GST_STATE_READY && st->new_state == GST_STATE_READY) {

        if (st->miptv_ui->tagcache) {
            gst_tag_list_free (st->miptv_ui->tagcache);
            st->miptv_ui->tagcache = NULL;
        }
    }

    gst_object_unref (GST_OBJECT (st->play));
    //g_object_unref (G_OBJECT (st->win));
    g_free (st);

    /* once */
    return FALSE;
}

#endif

/** Stops playing the current A/V.
 *
 * FIXME: How to proceed differently between livetv 
 * and recorded content?
 *
 * @param tvplayer the object instance.
 * @return void 
 */
void
mmyth_tvplayer_stop_playing (MMythTVPlayer *tvplayer) 
{
    g_debug ("[%s] Setting gstreamer pipeline state to NULL", __FUNCTION__);

    gst_element_set_state (tvplayer->gst_pipeline, GST_STATE_NULL);
    
    if (tvplayer->is_livetv) {
	    if (!gmyth_recorder_stop_livetv (tvplayer->recorder)) {
	    	g_warning ("[%s] Error while stoping remote encoder", __FUNCTION__);	
	    }
    }
}

/** Queries if the tvplayer is active playing A/V content.
 *
 * @param tvplayer the object instance.
 * @return TRUE if the tvplayer is active, FALSE otherwise.
 */
gboolean
mmyth_tvplayer_is_playing (MMythTVPlayer *tvplayer)
{
	return (GST_STATE (tvplayer->gst_pipeline) == GST_STATE_PLAYING);
}

/** Static function that sets the tvplayer state to PLAYING.
 *
 * @param tvplayer the object instance.
 * @return TRUE if the tvplayer is active, FALSE otherwise.
 */
static gboolean
idle_play (gpointer data)
{
    MMythTVPlayer *tvplayer = MMYTH_TVPLAYER (data);

	g_debug ("MMythTVPlayer: Setting pipeline state to PLAYING\n");

    gst_element_set_state (tvplayer->gst_pipeline, GST_STATE_PLAYING);

    return FALSE;
}

/** Start playing A/V with the tvplayer attributes.
 *
 * @param tvplayer the object instance.
 */
void
mmyth_tvplayer_start_playing (MMythTVPlayer *tvplayer)
{

	// FIXME: Move this to livetv_setup??
	if (tvplayer->is_livetv) {

	#if 0
		if (!tvplayer || !(tvplayer->proginfo) || !(tvplayer->local_hostname)
				|| !(tvplayer->gst_source)) {
			g_warning ("GMythtvPlayer not ready to start playing\n");		
		}

		if (!(tvplayer->proginfo->pathname)) {
			g_warning ("[%s] Playback url is null, could not play the myth content", __FUNCTION__);
			return;
		}

		g_debug ("MMythTVPlayer: Start playing %s", tvplayer->proginfo->pathname->str);
#endif
		g_object_set (G_OBJECT (tvplayer->gst_source), "mythtv-live",
				TRUE, NULL);
#if 0
		if ( tvplayer->tvchain != NULL ) {
			GString *str_chainid = gmyth_tvchain_get_id(tvplayer->tvchain);
			g_print( "[%s]\tCHAIN ID: %s\n", __FUNCTION__, str_chainid->str );

			g_object_set (G_OBJECT (tvplayer->gst_source), "mythtv-live-chainid", 
					g_strdup( str_chainid->str ), NULL);      
			if ( str_chainid!=NULL)
				g_string_free( str_chainid, FALSE );
		}

		if ( tvplayer->recorder != NULL )	
			g_object_set (G_OBJECT (tvplayer->gst_source), "mythtv-live-id",
					tvplayer->recorder->recorder_num, NULL );
		g_debug ("[%s] Setting location to %s", __FUNCTION__, 
				tvplayer->proginfo->pathname->str);

		/* Sets the gstreamer properties acording to the service access address */
		g_object_set (G_OBJECT (tvplayer->gst_source), "location",
				tvplayer->proginfo->pathname->str, NULL);              
#endif
	}

	g_object_set (G_OBJECT (tvplayer->gst_source), "mythtv-version",
              MYTHTV_VERSION_DEFAULT, NULL);

    g_object_set (G_OBJECT (tvplayer->gst_source), "mythtv-debug",
              TRUE, NULL);

    g_idle_add (idle_play, tvplayer);

}