/* GStreamer * Copyright (C) <2015> Stefan Sauer * * controller-graph: explore interpolation types * * 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., 51 Franklin St, Fifth Floor, * Boston, MA 02110-1301, USA. */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include #include #include #include #include GtkWidget *graph; GstControlSource *cs = NULL; gdouble yval[] = { 0.0, 0.2, 0.8, 0.1, 0.1, 1.0 }; static gboolean on_graph_draw (GtkWidget * widget, cairo_t * cr, gpointer user_data) { GtkStyleContext *style = gtk_widget_get_style_context (widget); GtkAllocation alloc; gint x, y, w, h; gdouble *data; guint64 ts, ts_step; gint i; GstTimedValueControlSource *tvcs = (GstTimedValueControlSource *) cs; gtk_widget_get_allocation (widget, &alloc); w = alloc.width; h = alloc.height; gtk_render_background (style, cr, 0, 0, w, h); // add some border: x = 5; y = 5; w -= (x + x); h -= (y + y); // build graph ts = G_GUINT64_CONSTANT (0); ts_step = w / (G_N_ELEMENTS (yval) - 1); gst_timed_value_control_source_unset_all (tvcs); for (i = 0; i < G_N_ELEMENTS (yval); i++) { gst_timed_value_control_source_set (tvcs, ts, yval[i]); ts += ts_step; } data = g_new (gdouble, w); gst_control_source_get_value_array (cs, 0, 1, w, data); // draw background cairo_set_source_rgb (cr, 0.5, 0.5, 0.5); cairo_rectangle (cr, x, y, w, h); cairo_stroke_preserve (cr); cairo_set_source_rgb (cr, 1.0, 1.0, 1.0); cairo_fill (cr); // plot graph cairo_set_source_rgb (cr, 0.0, 0.0, 0.0); cairo_set_line_width (cr, 1.0); cairo_move_to (cr, x, y + data[0] * h); for (i = 1; i < w; i++) { cairo_line_to (cr, x + i, y + CLAMP (data[i], 0.0, 1.0) * h); } cairo_stroke (cr); // plot control points ts = G_GUINT64_CONSTANT (0); for (i = 0; i < G_N_ELEMENTS (yval); i++) { cairo_set_source_rgb (cr, 0.0, 0.0, 0.0); cairo_arc (cr, x + ts, y + yval[i] * h, 3.0, 0.0, 2 * M_PI); cairo_stroke_preserve (cr); cairo_set_source_rgb (cr, 1.0, 1.0, 1.0); cairo_fill (cr); ts += ts_step; } g_free (data); return TRUE; } static void on_mode_changed (GtkComboBox * combo, gpointer user_data) { g_object_set (cs, "mode", gtk_combo_box_get_active (combo), NULL); gtk_widget_queue_draw (graph); } static void on_yval_changed (GtkSpinButton * spin, gpointer user_data) { guint ix = GPOINTER_TO_UINT (user_data); yval[ix] = gtk_spin_button_get_value (spin); gtk_widget_queue_draw (graph); } int main (int argc, char **argv) { GtkWidget *window; GtkWidget *layout, *label, *combo, *box, *spin; GEnumClass *enum_class; GEnumValue *enum_value; gint i; gst_init (&argc, &argv); gtk_init (&argc, &argv); cs = gst_interpolation_control_source_new (); g_object_set (cs, "mode", GST_INTERPOLATION_MODE_LINEAR, NULL); window = gtk_window_new (GTK_WINDOW_TOPLEVEL); g_signal_connect (G_OBJECT (window), "delete-event", gtk_main_quit, NULL); gtk_window_set_default_size (GTK_WINDOW (window), 320, 240); gtk_window_set_title (GTK_WINDOW (window), "GstInterpolationControlSource demo"); layout = gtk_grid_new (); graph = gtk_drawing_area_new (); gtk_widget_add_events (graph, GDK_POINTER_MOTION_MASK); g_signal_connect (graph, "draw", G_CALLBACK (on_graph_draw), NULL); g_object_set (graph, "hexpand", TRUE, "vexpand", TRUE, "margin-bottom", 3, NULL); gtk_grid_attach (GTK_GRID (layout), graph, 0, 0, 2, 1); // add controls to move the yvals box = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 3); g_object_set (box, "homogeneous", TRUE, "margin-bottom", 3, NULL); for (i = 0; i < G_N_ELEMENTS (yval); i++) { spin = gtk_spin_button_new_with_range (0.0, 1.0, 0.05); gtk_spin_button_set_value (GTK_SPIN_BUTTON (spin), yval[i]); g_signal_connect (spin, "changed", G_CALLBACK (on_yval_changed), GUINT_TO_POINTER (i)); gtk_container_add (GTK_CONTAINER (box), spin); } gtk_grid_attach (GTK_GRID (layout), box, 0, 1, 2, 1); // combo for interpolation modes label = gtk_label_new ("interpolation mode"); gtk_grid_attach (GTK_GRID (layout), label, 0, 2, 1, 1); combo = gtk_combo_box_text_new (); enum_class = g_type_class_ref (GST_TYPE_INTERPOLATION_MODE); for (i = enum_class->minimum; i <= enum_class->maximum; i++) { if ((enum_value = g_enum_get_value (enum_class, i))) { gtk_combo_box_text_append_text (GTK_COMBO_BOX_TEXT (combo), enum_value->value_nick); } } gtk_combo_box_set_active (GTK_COMBO_BOX (combo), GST_INTERPOLATION_MODE_LINEAR); g_signal_connect (combo, "changed", G_CALLBACK (on_mode_changed), NULL); g_object_set (combo, "hexpand", TRUE, "margin-left", 3, NULL); gtk_grid_attach (GTK_GRID (layout), combo, 1, 2, 1, 1); gtk_container_set_border_width (GTK_CONTAINER (window), 6); gtk_container_add (GTK_CONTAINER (window), layout); gtk_widget_show_all (window); gtk_main (); return 0; }