/* 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; GtkWidget *properties; } 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); void app_selection_changed_cb (GtkTreeSelection * selection, App * app); gboolean create_ui (App * app); static gboolean find_row_for_object (GtkListStore * model, GtkTreeIter * ret, GESTimelineObject * object); /* 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"); } void app_selection_changed_cb (GtkTreeSelection * selection, App * app) { int n; n = gtk_tree_selection_count_selected_rows (selection); /* some widgets should be disabled when we have an empty selection */ gtk_widget_set_sensitive (app->properties, n > 0); } /* 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 gboolean find_row_for_object (GtkListStore * model, GtkTreeIter * ret, GESTimelineObject * object) { gtk_tree_model_get_iter_first ((GtkTreeModel *) model, ret); while (gtk_list_store_iter_is_valid (model, ret)) { GESTimelineObject *obj; gtk_tree_model_get ((GtkTreeModel *) model, ret, 2, &obj, -1); if (obj == object) { g_object_unref (obj); return TRUE; } g_object_unref (obj); gtk_tree_model_iter_next ((GtkTreeModel *) model, ret); } return FALSE; } 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) { GtkTreeIter iter; GST_INFO ("layer object removed cb %p %p %p", layer, object, app); if (!find_row_for_object (GTK_LIST_STORE (app->model), &iter, object)) { g_print ("object deleted but we don't own it"); return; } gtk_list_store_remove (app->model, &iter); } /* Layout *******************************************************************/ #define GET_WIDGET(dest,name,type) {\ if (!(dest =\ type(gtk_builder_get_object(builder, name))))\ goto fail;\ } gboolean create_ui (App * app) { GtkBuilder *builder; GtkTreeView *timeline; /* construct widget tree */ builder = gtk_builder_new (); gtk_builder_add_from_file (builder, "ges-ui.glade", NULL); gtk_builder_connect_signals (builder, app); /* get a bunch of widgets from the XML tree */ GET_WIDGET (timeline, "timeline_treeview", GTK_TREE_VIEW); GET_WIDGET (app->properties, "properties", GTK_WIDGET); GET_WIDGET (app->main_window, "window", GTK_WIDGET); /* we care when the tree selection changes */ if (!(app->selection = gtk_tree_view_get_selection (timeline))) goto fail; g_signal_connect (app->selection, "changed", G_CALLBACK (app_selection_changed_cb), app); /* create the model for the treeview */ if (!(app->model = gtk_list_store_new (3, G_TYPE_STRING, G_TYPE_UINT64, G_TYPE_OBJECT))) goto fail; gtk_tree_view_set_model (timeline, GTK_TREE_MODEL (app->model)); /* register callbacks on GES objects */ 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); /* success */ g_object_unref (G_OBJECT (builder)); return TRUE; fail: g_object_unref (G_OBJECT (builder)); return FALSE; } #undef GET_WIDGET /* 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; }