/* GStreamer Editing Services * Copyright (C) 2010 Brandon Lewis * 2010 Nokia Corporation * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Library General Public * License as published by the Free Software Foundation; either * version 2 of the License, or (at your option) any later version. * * This library 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 * Library General Public License for more details. * * You should have received a copy of the GNU Library General Public * License along with this library; if not, write to the * Free Software Foundation, Inc., 59 Temple Place - Suite 330, * Boston, MA 02111-1307, USA. */ #include #include #include typedef struct App { GESTimeline *timeline; GESTimelinePipeline *pipeline; GESTimelineLayer *layer; GtkWidget *main_window; GtkListStore *model; GtkTreeSelection *selection; } App; App *app_new (void); void app_dispose (App * app); void app_add_file (App * app, gchar * filename); GList *app_get_selected_objects (App * app); void app_delete_objects (App * app, GList * objects); void window_destroy_cb (GtkObject * window, App * app); void quit_item_activate_cb (GtkMenuItem * item, App * app); void delete_item_activate_cb (GtkMenuItem * item, App * app); void add_file_item_activate_cb (GtkMenuItem * item, App * app); gboolean create_ui (App * app); /* signal handlers **********************************************************/ void window_destroy_cb (GtkObject * window, App * app) { gtk_main_quit (); } void quit_item_activate_cb (GtkMenuItem * item, App * app) { gtk_main_quit (); } void delete_item_activate_cb (GtkMenuItem * item, App * app) { /* get a gslist of selected track objects */ GList *objects = NULL; objects = app_get_selected_objects (app); app_delete_objects (app, objects); } void add_file_item_activate_cb (GtkMenuItem * item, App * app) { GST_DEBUG ("add file signal handler"); /* TODO: solicit this information from the user */ app_add_file (app, (gchar *) "/home/brandon/media/small-mvi_0008.avi"); } /* application methods ******************************************************/ static void selection_foreach (GtkTreeModel * model, GtkTreePath * path, GtkTreeIter * iter, gpointer user); static void selection_foreach (GtkTreeModel * model, GtkTreePath * path, GtkTreeIter * iter, gpointer user) { GList **ret = user; GESTimelineObject *obj; gtk_tree_model_get (model, iter, 2, &obj, -1); *ret = g_list_append (*ret, obj); return; } GList * app_get_selected_objects (App * app) { GList *objects = NULL; g_print ("selection: %p\n", app->selection); gtk_tree_selection_selected_foreach (GTK_TREE_SELECTION (app->selection), selection_foreach, &objects); return objects; } void app_delete_objects (App * app, GList * objects) { GList *cur; for (cur = objects; cur; cur = cur->next) { ges_timeline_layer_remove_object (app->layer, GES_TIMELINE_OBJECT (cur->data)); g_object_unref ((GObject *) cur->data); cur->data = NULL; } g_list_free (objects); } void app_add_file (App * app, gchar * filename) { gchar *uri; GESTimelineObject *obj; GST_DEBUG ("adding file %s", filename); if (!g_file_test (filename, G_FILE_TEST_EXISTS)) { /* TODO: error notification in UI */ return; } uri = g_strdup_printf ("file://%s", filename); obj = GES_TIMELINE_OBJECT (ges_timeline_filesource_new (uri)); g_free (uri); ges_timeline_layer_add_object (app->layer, obj); } App * app_new (void) { App *ret; ret = g_new0 (App, 1); if (!ret) return NULL; if (!(ret->timeline = ges_timeline_new_audio_video ())) goto fail; if (!(ret->pipeline = ges_timeline_pipeline_new ())) goto fail; if (!ges_timeline_pipeline_add_timeline (ret->pipeline, ret->timeline)) goto fail; if (!(ret->layer = (GESTimelineLayer *) ges_simple_timeline_layer_new ())) goto fail; if (!(ges_timeline_add_layer (ret->timeline, ret->layer))) goto fail; if (!(create_ui (ret))) goto fail; return ret; fail: app_dispose (ret); return NULL; } void app_dispose (App * app) { if (app) { if (app->pipeline) gst_object_unref (app->pipeline); g_free (app); } } /* Backend callbacks ********************************************************/ static gchar * desc_for_object (GESTimelineObject * object) { gchar *uri; /* there is only one type of object at the moment */ /* return the uri */ g_object_get (object, "uri", &uri, NULL); return uri; } static void layer_object_added_cb (GESTimelineLayer * layer, GESTimelineObject * object, App * app) { GtkTreeIter iter; gchar *description; guint64 duration = 0; GST_INFO ("layer object added cb %p %p %p", layer, object, app); description = desc_for_object (object); g_object_get (object, "duration", &duration, NULL); gtk_list_store_append (app->model, &iter); gtk_list_store_set (app->model, &iter, 0, description, 1, duration, 2, object, -1); } static void layer_object_removed_cb (GESTimelineLayer * layer, GESTimelineObject * object, App * app) { GST_INFO ("layer object removed cb %p %p %p", layer, object, app); g_print ("layer object removed"); } /* Layout *******************************************************************/ gboolean create_ui (App * app) { GtkBuilder *builder; GtkTreeView *timeline; builder = gtk_builder_new (); gtk_builder_add_from_file (builder, "ges-ui.glade", NULL); timeline = GTK_TREE_VIEW (gtk_builder_get_object (builder, "timeline_treeview")); app->main_window = GTK_WIDGET (gtk_builder_get_object (builder, "window")); app->model = gtk_list_store_new (3, G_TYPE_STRING, G_TYPE_UINT64, G_TYPE_OBJECT); if (!(app->main_window && app->model && timeline)) { g_print ("Could not retrieve all widgets from the XML"); goto fail; } app->selection = gtk_tree_view_get_selection (timeline); if (!app->selection) goto fail; gtk_tree_view_set_model (timeline, GTK_TREE_MODEL (app->model)); gtk_builder_connect_signals (builder, app); g_object_unref (G_OBJECT (builder)); gtk_widget_show (app->main_window); g_signal_connect (app->layer, "object-added", G_CALLBACK (layer_object_added_cb), app); g_signal_connect (app->layer, "object-removed", G_CALLBACK (layer_object_removed_cb), app); return TRUE; fail: g_object_unref (G_OBJECT (builder)); return FALSE; } /* main *********************************************************************/ int main (int argc, char *argv[]) { App *app; /* intialize GStreamer and GES */ if (!g_thread_supported ()) g_thread_init (NULL); gst_init (&argc, &argv); ges_init (); /* initialize UI */ gtk_init (&argc, &argv); if ((app = app_new ())) { gtk_main (); app_dispose (app); } return 0; }