camerabin2: Add camera-src property

Adds a property to select the camera source element to be used.
Changing only happens on the next NULL->READY transition
This commit is contained in:
Thiago Santos 2010-12-13 16:36:19 -03:00
parent adc3cdc6aa
commit 59c48d3443
2 changed files with 113 additions and 50 deletions

View file

@ -63,6 +63,7 @@ enum
PROP_0, PROP_0,
PROP_MODE, PROP_MODE,
PROP_LOCATION, PROP_LOCATION,
PROP_CAMERA_SRC,
PROP_IMAGE_CAPTURE_SUPPORTED_CAPS, PROP_IMAGE_CAPTURE_SUPPORTED_CAPS,
PROP_VIDEO_CAPTURE_SUPPORTED_CAPS PROP_VIDEO_CAPTURE_SUPPORTED_CAPS
}; };
@ -196,7 +197,6 @@ gst_camera_bin_dispose (GObject * object)
{ {
GstCameraBin *camerabin = GST_CAMERA_BIN_CAST (object); GstCameraBin *camerabin = GST_CAMERA_BIN_CAST (object);
g_free (camerabin->image_location); g_free (camerabin->image_location);
g_free (camerabin->video_location); g_free (camerabin->video_location);
@ -205,6 +205,8 @@ gst_camera_bin_dispose (GObject * object)
camerabin->src_capture_notify_id); camerabin->src_capture_notify_id);
if (camerabin->src) if (camerabin->src)
gst_object_unref (camerabin->src); gst_object_unref (camerabin->src);
if (camerabin->user_src)
gst_object_unref (camerabin->user_src);
if (camerabin->viewfinderbin) if (camerabin->viewfinderbin)
gst_object_unref (camerabin->viewfinderbin); gst_object_unref (camerabin->viewfinderbin);
@ -285,6 +287,11 @@ gst_camera_bin_class_init (GstCameraBinClass * klass)
"Default for images is img_%d and vid_%d for videos", "Default for images is img_%d and vid_%d for videos",
DEFAULT_IMG_LOCATION, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS)); DEFAULT_IMG_LOCATION, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
g_object_class_install_property (object_class, PROP_CAMERA_SRC,
g_param_spec_object ("camera-src", "Camera source",
"The camera source element to be used",
GST_TYPE_ELEMENT, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
g_object_class_install_property (object_class, g_object_class_install_property (object_class,
PROP_IMAGE_CAPTURE_SUPPORTED_CAPS, PROP_IMAGE_CAPTURE_SUPPORTED_CAPS,
g_param_spec_boxed ("image-capture-supported-caps", g_param_spec_boxed ("image-capture-supported-caps",
@ -352,63 +359,98 @@ gst_camera_bin_init (GstCameraBin * camerabin)
static gboolean static gboolean
gst_camera_bin_create_elements (GstCameraBin * camera) gst_camera_bin_create_elements (GstCameraBin * camera)
{ {
if (camera->elements_created) gboolean new_src = FALSE;
return TRUE;
camera->src = gst_element_factory_make ("v4l2camerasrc", "camerasrc"); if (!camera->elements_created) {
camera->videobin = gst_element_factory_make ("videorecordingbin", "videobin");
camera->imagebin = gst_element_factory_make ("imagecapturebin", "imagebin");
camera->videobin_queue = gst_element_factory_make ("queue", "videobin-queue"); camera->videobin =
camera->imagebin_queue = gst_element_factory_make ("queue", "imagebin-queue"); gst_element_factory_make ("videorecordingbin", "videobin");
camera->viewfinderbin_queue = gst_element_factory_make ("queue", camera->imagebin = gst_element_factory_make ("imagecapturebin", "imagebin");
"viewfinderbin-queue");
camera->videobin_capsfilter = gst_element_factory_make ("capsfilter", camera->videobin_queue =
"videobin-capsfilter"); gst_element_factory_make ("queue", "videobin-queue");
camera->imagebin_capsfilter = gst_element_factory_make ("capsfilter", camera->imagebin_queue =
"imagebin-capsfilter"); gst_element_factory_make ("queue", "imagebin-queue");
camera->viewfinderbin_capsfilter = gst_element_factory_make ("capsfilter", camera->viewfinderbin_queue =
"viewfinderbin-capsfilter"); gst_element_factory_make ("queue", "viewfinderbin-queue");
gst_bin_add_many (GST_BIN_CAST (camera), gst_object_ref (camera->src), camera->videobin_capsfilter = gst_element_factory_make ("capsfilter",
gst_object_ref (camera->videobin), gst_object_ref (camera->imagebin), "videobin-capsfilter");
gst_object_ref (camera->videobin_queue), camera->imagebin_capsfilter = gst_element_factory_make ("capsfilter",
gst_object_ref (camera->imagebin_queue), "imagebin-capsfilter");
gst_object_ref (camera->viewfinderbin_queue), camera->viewfinderbin_capsfilter = gst_element_factory_make ("capsfilter",
gst_object_ref (camera->videobin_capsfilter), "viewfinderbin-capsfilter");
gst_object_ref (camera->imagebin_capsfilter),
gst_object_ref (camera->viewfinderbin_capsfilter), NULL);
/* Linking can be optimized TODO */ gst_bin_add_many (GST_BIN_CAST (camera),
gst_element_link_many (camera->videobin_queue, camera->videobin_capsfilter, gst_object_ref (camera->videobin), gst_object_ref (camera->imagebin),
camera->videobin, NULL); gst_object_ref (camera->videobin_queue),
gst_element_link_many (camera->imagebin_queue, camera->imagebin_capsfilter, gst_object_ref (camera->imagebin_queue),
camera->imagebin, NULL); gst_object_ref (camera->viewfinderbin_queue),
gst_element_link_many (camera->viewfinderbin_queue, gst_object_ref (camera->videobin_capsfilter),
camera->viewfinderbin_capsfilter, camera->viewfinderbin, NULL); gst_object_ref (camera->imagebin_capsfilter),
gst_element_link_pads (camera->src, "vfsrc", camera->viewfinderbin_queue, gst_object_ref (camera->viewfinderbin_capsfilter), NULL);
"sink");
gst_element_link_pads (camera->src, "imgsrc", camera->imagebin_queue, "sink");
gst_element_link_pads (camera->src, "vidsrc", camera->videobin_queue, "sink");
/* /* Linking can be optimized TODO */
* Video can't get into playing as its internal filesink will open gst_element_link_many (camera->videobin_queue, camera->videobin_capsfilter,
* a file for writing and leave it empty if unused. camera->videobin, NULL);
* gst_element_link_many (camera->imagebin_queue, camera->imagebin_capsfilter,
* Its state is managed using the current mode and the source's camera->imagebin, NULL);
* ready-for-capture notify callback. When we are at video mode and gst_element_link_many (camera->viewfinderbin_queue,
* the source's ready-for-capture goes to FALSE it means it is camera->viewfinderbin_capsfilter, camera->viewfinderbin, NULL);
* starting recording, so we should prepare the video bin. /*
*/ * Video can't get into playing as its internal filesink will open
gst_element_set_locked_state (camera->videobin, TRUE); * a file for writing and leave it empty if unused.
camera->src_capture_notify_id = g_signal_connect (G_OBJECT (camera->src), *
"notify::ready-for-capture", * Its state is managed using the current mode and the source's
G_CALLBACK (gst_camera_bin_src_notify_readyforcapture), camera); * ready-for-capture notify callback. When we are at video mode and
* the source's ready-for-capture goes to FALSE it means it is
* starting recording, so we should prepare the video bin.
*/
gst_element_set_locked_state (camera->videobin, TRUE);
g_object_set (camera->videobin, "location", camera->video_location, NULL);
g_object_set (camera->imagebin, "location", camera->image_location, NULL);
}
/* check if we need to replace the camera src */
if (camera->src) {
if (camera->user_src && camera->user_src != camera->src) {
if (camera->src_capture_notify_id)
g_signal_handler_disconnect (camera->src,
camera->src_capture_notify_id);
gst_bin_remove (GST_BIN_CAST (camera), camera->src);
gst_object_unref (camera->src);
camera->src = NULL;
}
}
if (!camera->src) {
if (camera->user_src) {
camera->src = gst_object_ref (camera->user_src);
} else {
camera->src = gst_element_factory_make ("v4l2camerasrc", "camerasrc");
}
new_src = TRUE;
}
g_assert (camera->src != NULL);
g_object_set (camera->src, "mode", camera->mode, NULL); g_object_set (camera->src, "mode", camera->mode, NULL);
g_object_set (camera->videobin, "location", camera->video_location, NULL); if (new_src) {
g_object_set (camera->imagebin, "location", camera->image_location, NULL); gst_bin_add (GST_BIN_CAST (camera), gst_object_ref (camera->src));
camera->src_capture_notify_id = g_signal_connect (G_OBJECT (camera->src),
"notify::ready-for-capture",
G_CALLBACK (gst_camera_bin_src_notify_readyforcapture), camera);
gst_element_link_pads (camera->src, "vfsrc", camera->viewfinderbin_queue,
"sink");
gst_element_link_pads (camera->src, "imgsrc", camera->imagebin_queue,
"sink");
gst_element_link_pads (camera->src, "vidsrc", camera->videobin_queue,
"sink");
}
camera->elements_created = TRUE; camera->elements_created = TRUE;
return TRUE; return TRUE;
@ -459,6 +501,20 @@ gst_camera_bin_set_location (GstCameraBin * camera, const gchar * location)
} }
} }
static void
gst_camera_bin_set_camera_src (GstCameraBin * camera, GstElement * src)
{
GST_DEBUG_OBJECT (GST_OBJECT (camera),
"Setting camera source %" GST_PTR_FORMAT, src);
if (camera->user_src)
g_object_unref (camera->user_src);
if (src)
g_object_ref (src);
camera->user_src = src;
}
static void static void
gst_camera_bin_set_property (GObject * object, guint prop_id, gst_camera_bin_set_property (GObject * object, guint prop_id,
const GValue * value, GParamSpec * pspec) const GValue * value, GParamSpec * pspec)
@ -472,6 +528,9 @@ gst_camera_bin_set_property (GObject * object, guint prop_id,
case PROP_LOCATION: case PROP_LOCATION:
gst_camera_bin_set_location (camera, g_value_get_string (value)); gst_camera_bin_set_location (camera, g_value_get_string (value));
break; break;
case PROP_CAMERA_SRC:
gst_camera_bin_set_camera_src (camera, g_value_get_object (value));
break;
default: default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
break; break;
@ -495,6 +554,9 @@ gst_camera_bin_get_property (GObject * object, guint prop_id,
g_value_set_string (value, camera->image_location); g_value_set_string (value, camera->image_location);
} }
break; break;
case PROP_CAMERA_SRC:
g_value_set_object (value, camera->src);
break;
case PROP_VIDEO_CAPTURE_SUPPORTED_CAPS: case PROP_VIDEO_CAPTURE_SUPPORTED_CAPS:
case PROP_IMAGE_CAPTURE_SUPPORTED_CAPS:{ case PROP_IMAGE_CAPTURE_SUPPORTED_CAPS:{
GstPad *pad; GstPad *pad;

View file

@ -38,6 +38,7 @@ struct _GstCameraBin
GstPipeline pipeline; GstPipeline pipeline;
GstElement *src; GstElement *src;
GstElement *user_src;
gulong src_capture_notify_id; gulong src_capture_notify_id;
GstElement *videobin; GstElement *videobin;