videosource: Always add a deinterlace at the beining of videosrcbin
[platform/upstream/gst-editing-services.git] / ges / ges-video-source.c
1 /* GStreamer Editing Services
2  * Copyright (C) 2009 Edward Hervey <edward.hervey@collabora.co.uk>
3  *               2009 Nokia Corporation
4  *
5  * This library is free software; you can redistribute it and/or
6  * modify it under the terms of the GNU Library General Public
7  * License as published by the Free Software Foundation; either
8  * version 2 of the License, or (at your option) any later version.
9  *
10  * This library is distributed in the hope that it will be useful,
11  * but WITHOUT ANY WARRANTY; without even the implied warranty of
12  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
13  * Library General Public License for more details.
14  *
15  * You should have received a copy of the GNU Library General Public
16  * License along with this library; if not, write to the
17  * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
18  * Boston, MA 02110-1301, USA.
19  */
20
21 /**
22  * SECTION:ges-video-source
23  * @short_description: Base Class for video sources
24  */
25
26 #include <gst/pbutils/missing-plugins.h>
27
28 #include "ges-internal.h"
29 #include "ges/ges-meta-container.h"
30 #include "ges-track-element.h"
31 #include "ges-video-source.h"
32 #include "ges-layer.h"
33 #include "gstframepositionner.h"
34
35 G_DEFINE_ABSTRACT_TYPE (GESVideoSource, ges_video_source, GES_TYPE_SOURCE);
36
37 struct _GESVideoSourcePrivate
38 {
39   GstFramePositionner *positionner;
40   GstElement *capsfilter;
41   GESLayer *layer;
42 };
43
44 /* TrackElement VMethods */
45 static void
46 layer_priority_changed_cb (GESLayer * layer, GParamSpec * arg G_GNUC_UNUSED,
47     GESVideoSource * self)
48 {
49   g_object_set (self->priv->positionner, "zorder",
50       10000 - ges_layer_get_priority (layer), NULL);
51 }
52
53 static void
54 layer_changed_cb (GESClip * clip, GParamSpec * arg G_GNUC_UNUSED,
55     GESVideoSource * self)
56 {
57   GESVideoSourcePrivate *priv = self->priv;
58
59   if (priv->layer) {
60     g_signal_handlers_disconnect_by_func (priv->layer,
61         layer_priority_changed_cb, self);
62   }
63
64   priv->layer = ges_clip_get_layer (clip);
65   if (priv->layer == NULL)
66     return;
67
68   /* We do not need any ref ourself as our parent owns one and we are connected
69    * to it */
70   g_object_unref (priv->layer);
71   /* 10000 is the max value of zorder on videomixerpad, hardcoded */
72   g_signal_connect (self->priv->layer, "notify::priority",
73       G_CALLBACK (layer_priority_changed_cb), self);
74
75   g_object_set (self->priv->positionner, "zorder",
76       10000 - ges_layer_get_priority (self->priv->layer), NULL);
77 }
78
79 static void
80 post_missing_element_message (GstElement * element, const gchar * name)
81 {
82   GstMessage *msg;
83
84   msg = gst_missing_element_message_new (element, name);
85   gst_element_post_message (element, msg);
86 }
87
88 static GstElement *
89 ges_video_source_create_element (GESTrackElement * trksrc)
90 {
91   GstElement *topbin;
92   GstElement *sub_element;
93   GESVideoSourceClass *source_class = GES_VIDEO_SOURCE_GET_CLASS (trksrc);
94   GESVideoSource *self;
95   GstElement *positionner, *videoscale, *videorate, *capsfilter, *videoconvert,
96       *deinterlace;
97   const gchar *props[] = { "alpha", "posx", "posy", "width", "height", NULL };
98   GESTimelineElement *parent;
99
100   if (!source_class->create_source)
101     return NULL;
102
103   sub_element = source_class->create_source (trksrc);
104
105   self = (GESVideoSource *) trksrc;
106
107   /* That positionner will add metadata to buffers according to its
108      properties, acting like a proxy for our smart-mixer dynamic pads. */
109   positionner = gst_element_factory_make ("framepositionner", "frame_tagger");
110
111   videoscale =
112       gst_element_factory_make ("videoscale", "track-element-videoscale");
113   videoconvert =
114       gst_element_factory_make ("videoconvert", "track-element-videoconvert");
115   videorate = gst_element_factory_make ("videorate", "track-element-videorate");
116   deinterlace = gst_element_factory_make ("deinterlace", "deinterlace");
117   if (deinterlace == NULL) {
118     deinterlace = gst_element_factory_make ("avdeinterlace", "deinterlace");
119   }
120   capsfilter =
121       gst_element_factory_make ("capsfilter", "track-element-capsfilter");
122
123   ges_frame_positionner_set_source_and_filter (GST_FRAME_POSITIONNER
124       (positionner), trksrc, capsfilter);
125
126   ges_track_element_add_children_props (trksrc, positionner, NULL, NULL, props);
127
128   if (deinterlace == NULL) {
129     post_missing_element_message (sub_element, "deinterlace");
130
131     GST_ELEMENT_WARNING (sub_element, CORE, MISSING_PLUGIN,
132         ("Missing element '%s' - check your GStreamer installation.",
133             "deinterlace"), ("deinterlacing won't work"));
134     topbin =
135         ges_source_create_topbin ("videosrcbin", sub_element, videoconvert,
136         positionner, videoscale, videorate, capsfilter, NULL);
137   } else {
138     topbin =
139         ges_source_create_topbin ("videosrcbin", sub_element, videoconvert,
140         deinterlace, positionner, videoscale, videorate, capsfilter, NULL);
141   }
142
143   parent = ges_timeline_element_get_parent (GES_TIMELINE_ELEMENT (trksrc));
144   if (parent) {
145     self->priv->positionner = GST_FRAME_POSITIONNER (positionner);
146     g_signal_connect (parent, "notify::layer",
147         (GCallback) layer_changed_cb, trksrc);
148     layer_changed_cb (GES_CLIP (parent), NULL, self);
149     gst_object_unref (parent);
150   } else {
151     GST_ERROR ("No parent timeline element, SHOULD NOT HAPPEN");
152   }
153
154   self->priv->capsfilter = capsfilter;
155
156   return topbin;
157 }
158
159 static gboolean
160 _set_parent (GESTimelineElement * self, GESTimelineElement * parent)
161 {
162   GESVideoSourcePrivate *priv = GES_VIDEO_SOURCE (self)->priv;
163
164   if (self->parent) {
165     if (priv->layer) {
166       g_signal_handlers_disconnect_by_func (priv->layer,
167           layer_priority_changed_cb, self);
168       priv->layer = NULL;
169     }
170
171     g_signal_handlers_disconnect_by_func (self->parent, layer_changed_cb, self);
172   }
173
174   if (parent && priv->positionner)
175     layer_changed_cb (GES_CLIP (parent), NULL, GES_VIDEO_SOURCE (self));
176
177
178   return TRUE;
179 }
180
181 static void
182 ges_video_source_class_init (GESVideoSourceClass * klass)
183 {
184   GESTrackElementClass *track_class = GES_TRACK_ELEMENT_CLASS (klass);
185   GESTimelineElementClass *element_class = GES_TIMELINE_ELEMENT_CLASS (klass);
186   GESVideoSourceClass *video_source_class = GES_VIDEO_SOURCE_CLASS (klass);
187
188   g_type_class_add_private (klass, sizeof (GESVideoSourcePrivate));
189
190   element_class->set_parent = _set_parent;
191   track_class->gnlobject_factorytype = "gnlsource";
192   track_class->create_element = ges_video_source_create_element;
193   video_source_class->create_source = NULL;
194 }
195
196 static void
197 ges_video_source_init (GESVideoSource * self)
198 {
199   self->priv = G_TYPE_INSTANCE_GET_PRIVATE (self,
200       GES_TYPE_VIDEO_SOURCE, GESVideoSourcePrivate);
201   self->priv->positionner = NULL;
202   self->priv->capsfilter = NULL;
203 }