| /* GStreamer |
| * Copyright (C) <2015> Stefan Sauer <ensonic@users.sf.net> |
| * |
| * 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 <math.h> |
| |
| #include <glib.h> |
| #include <gtk/gtk.h> |
| |
| #include <gst/gst.h> |
| #include <gst/controller/gstinterpolationcontrolsource.h> |
| |
| 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; |
| } |