2017-07-10 21:02:08 +00:00
|
|
|
// Copyright (C) 2016-2017 Sebastian Dröge <sebastian@centricular.com>
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
|
|
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
|
|
|
// option. This file may not be copied, modified, or distributed
|
|
|
|
// except according to those terms.
|
|
|
|
|
2018-09-28 09:00:08 +00:00
|
|
|
use caps_features::*;
|
2018-04-01 08:30:03 +00:00
|
|
|
use miniobject::*;
|
2017-07-10 21:02:08 +00:00
|
|
|
use std::fmt;
|
2018-09-28 09:00:08 +00:00
|
|
|
use std::ptr;
|
2017-07-13 11:57:02 +00:00
|
|
|
use std::str;
|
2017-07-10 21:02:08 +00:00
|
|
|
use structure::*;
|
|
|
|
|
2017-07-24 23:03:18 +00:00
|
|
|
use CapsIntersectMode;
|
|
|
|
|
2018-04-01 08:30:03 +00:00
|
|
|
use glib;
|
2019-10-04 06:11:30 +00:00
|
|
|
use glib::translate::{from_glib, from_glib_full, FromGlibPtrFull, ToGlib, ToGlibPtr};
|
2017-11-15 17:18:58 +00:00
|
|
|
use glib::value::ToSendValue;
|
2019-03-19 07:58:20 +00:00
|
|
|
use gst_sys;
|
|
|
|
|
|
|
|
gst_define_mini_object_wrapper!(
|
|
|
|
Caps,
|
|
|
|
CapsRef,
|
|
|
|
gst_sys::GstCaps,
|
|
|
|
[Debug, PartialEq, Eq,],
|
|
|
|
|| gst_sys::gst_caps_get_type()
|
|
|
|
);
|
2017-07-10 21:02:08 +00:00
|
|
|
|
2018-09-28 15:11:46 +00:00
|
|
|
impl Caps {
|
2017-11-11 12:14:49 +00:00
|
|
|
pub fn builder(name: &str) -> Builder {
|
|
|
|
assert_initialized_main_thread!();
|
|
|
|
Builder::new(name)
|
|
|
|
}
|
|
|
|
|
2017-07-10 21:31:53 +00:00
|
|
|
pub fn new_empty() -> Self {
|
|
|
|
assert_initialized_main_thread!();
|
2019-03-19 07:58:20 +00:00
|
|
|
unsafe { from_glib_full(gst_sys::gst_caps_new_empty()) }
|
2017-07-10 21:02:08 +00:00
|
|
|
}
|
|
|
|
|
2017-07-10 21:31:53 +00:00
|
|
|
pub fn new_any() -> Self {
|
|
|
|
assert_initialized_main_thread!();
|
2019-03-19 07:58:20 +00:00
|
|
|
unsafe { from_glib_full(gst_sys::gst_caps_new_any()) }
|
2017-07-10 21:02:08 +00:00
|
|
|
}
|
|
|
|
|
2019-06-06 06:09:34 +00:00
|
|
|
pub fn new_simple(name: &str, values: &[(&str, &dyn ToSendValue)]) -> Self {
|
2017-07-10 21:31:53 +00:00
|
|
|
assert_initialized_main_thread!();
|
2017-10-11 12:32:03 +00:00
|
|
|
let mut caps = Caps::new_empty();
|
2017-07-10 21:02:08 +00:00
|
|
|
|
2017-07-10 21:31:53 +00:00
|
|
|
let structure = Structure::new(name, values);
|
|
|
|
caps.get_mut().unwrap().append_structure(structure);
|
2017-07-10 21:02:08 +00:00
|
|
|
|
|
|
|
caps
|
|
|
|
}
|
|
|
|
|
2017-07-24 23:03:18 +00:00
|
|
|
pub fn fixate(caps: Self) -> Self {
|
2017-08-30 11:39:09 +00:00
|
|
|
skip_assert_initialized!();
|
2019-03-19 07:58:20 +00:00
|
|
|
unsafe { from_glib_full(gst_sys::gst_caps_fixate(caps.into_ptr())) }
|
2017-07-24 23:03:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn merge(caps: Self, other: Self) -> Self {
|
2017-08-30 11:39:09 +00:00
|
|
|
skip_assert_initialized!();
|
2019-03-19 07:58:20 +00:00
|
|
|
unsafe { from_glib_full(gst_sys::gst_caps_merge(caps.into_ptr(), other.into_ptr())) }
|
2017-07-24 23:03:18 +00:00
|
|
|
}
|
|
|
|
|
2018-09-28 09:00:08 +00:00
|
|
|
pub fn merge_structure(caps: Self, structure: Structure) -> Self {
|
2017-08-30 11:39:09 +00:00
|
|
|
skip_assert_initialized!();
|
2017-07-24 23:03:18 +00:00
|
|
|
unsafe {
|
2019-03-19 07:58:20 +00:00
|
|
|
from_glib_full(gst_sys::gst_caps_merge_structure(
|
2017-07-24 23:03:18 +00:00
|
|
|
caps.into_ptr(),
|
2018-09-28 09:00:08 +00:00
|
|
|
structure.into_ptr(),
|
|
|
|
))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn merge_structure_full(
|
|
|
|
caps: Self,
|
|
|
|
structure: Structure,
|
|
|
|
features: Option<CapsFeatures>,
|
|
|
|
) -> Self {
|
|
|
|
skip_assert_initialized!();
|
|
|
|
unsafe {
|
2019-03-19 07:58:20 +00:00
|
|
|
from_glib_full(gst_sys::gst_caps_merge_structure_full(
|
2018-09-28 09:00:08 +00:00
|
|
|
caps.into_ptr(),
|
|
|
|
structure.into_ptr(),
|
|
|
|
features.map(|f| f.into_ptr()).unwrap_or(ptr::null_mut()),
|
2017-07-24 23:03:18 +00:00
|
|
|
))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn normalize(caps: Self) -> Self {
|
2017-08-30 11:39:09 +00:00
|
|
|
skip_assert_initialized!();
|
2019-03-19 07:58:20 +00:00
|
|
|
unsafe { from_glib_full(gst_sys::gst_caps_normalize(caps.into_ptr())) }
|
2017-07-24 23:03:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn simplify(caps: Self) -> Self {
|
2017-08-30 11:39:09 +00:00
|
|
|
skip_assert_initialized!();
|
2019-03-19 07:58:20 +00:00
|
|
|
unsafe { from_glib_full(gst_sys::gst_caps_simplify(caps.into_ptr())) }
|
2017-07-24 23:03:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn truncate(caps: Self) -> Self {
|
2017-08-30 11:39:09 +00:00
|
|
|
skip_assert_initialized!();
|
2019-03-19 07:58:20 +00:00
|
|
|
unsafe { from_glib_full(gst_sys::gst_caps_truncate(caps.into_ptr())) }
|
2017-07-24 23:03:18 +00:00
|
|
|
}
|
2017-07-10 21:31:53 +00:00
|
|
|
}
|
2017-07-10 21:02:08 +00:00
|
|
|
|
2017-07-13 11:57:02 +00:00
|
|
|
impl str::FromStr for Caps {
|
2019-12-17 19:00:42 +00:00
|
|
|
type Err = glib::BoolError;
|
2017-07-13 11:57:02 +00:00
|
|
|
|
2019-12-17 19:00:42 +00:00
|
|
|
fn from_str(s: &str) -> Result<Self, glib::BoolError> {
|
2019-10-04 06:11:30 +00:00
|
|
|
assert_initialized_main_thread!();
|
|
|
|
unsafe {
|
2019-12-17 19:00:42 +00:00
|
|
|
Option::<_>::from_glib_full(gst_sys::gst_caps_from_string(s.to_glib_none().0))
|
|
|
|
.ok_or_else(|| glib_bool_error!("Failed to parse caps from string"))
|
2019-10-04 06:11:30 +00:00
|
|
|
}
|
2017-07-13 11:57:02 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-28 16:14:35 +00:00
|
|
|
impl fmt::Display for Caps {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
self.0.fmt(f)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-10 21:31:53 +00:00
|
|
|
impl CapsRef {
|
2019-06-06 06:09:34 +00:00
|
|
|
pub fn set_simple(&mut self, values: &[(&str, &dyn ToSendValue)]) {
|
2017-08-02 16:40:31 +00:00
|
|
|
for &(name, value) in values {
|
2017-07-27 22:36:44 +00:00
|
|
|
let value = value.to_value();
|
|
|
|
|
2017-07-10 21:02:08 +00:00
|
|
|
unsafe {
|
2019-03-19 07:58:20 +00:00
|
|
|
gst_sys::gst_caps_set_value(
|
2017-07-10 21:33:24 +00:00
|
|
|
self.as_mut_ptr(),
|
|
|
|
name.to_glib_none().0,
|
|
|
|
value.to_glib_none().0,
|
|
|
|
);
|
2017-07-10 21:02:08 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn get_structure(&self, idx: u32) -> Option<&StructureRef> {
|
2018-07-06 08:36:00 +00:00
|
|
|
if idx >= self.get_size() {
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
|
2017-07-10 21:02:08 +00:00
|
|
|
unsafe {
|
2019-03-19 07:58:20 +00:00
|
|
|
let structure = gst_sys::gst_caps_get_structure(self.as_ptr(), idx);
|
2017-07-10 21:02:08 +00:00
|
|
|
if structure.is_null() {
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
|
2018-09-28 09:00:08 +00:00
|
|
|
Some(StructureRef::from_glib_borrow(structure))
|
2017-07-10 21:02:08 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-02 16:40:31 +00:00
|
|
|
pub fn get_mut_structure(&mut self, idx: u32) -> Option<&mut StructureRef> {
|
2018-07-06 08:36:00 +00:00
|
|
|
if idx >= self.get_size() {
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
|
2017-07-10 21:02:08 +00:00
|
|
|
unsafe {
|
2019-03-19 07:58:20 +00:00
|
|
|
let structure = gst_sys::gst_caps_get_structure(self.as_ptr(), idx);
|
2017-07-10 21:02:08 +00:00
|
|
|
if structure.is_null() {
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
|
2018-09-28 09:00:08 +00:00
|
|
|
Some(StructureRef::from_glib_borrow_mut(structure))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn get_features(&self, idx: u32) -> Option<&CapsFeaturesRef> {
|
|
|
|
if idx >= self.get_size() {
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsafe {
|
2019-03-19 07:58:20 +00:00
|
|
|
let features = gst_sys::gst_caps_get_features(self.as_ptr(), idx);
|
2018-09-28 09:00:08 +00:00
|
|
|
Some(CapsFeaturesRef::from_glib_borrow(features))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn get_mut_features(&mut self, idx: u32) -> Option<&mut CapsFeaturesRef> {
|
|
|
|
if idx >= self.get_size() {
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsafe {
|
2019-03-19 07:58:20 +00:00
|
|
|
let features = gst_sys::gst_caps_get_features(self.as_ptr(), idx);
|
2018-09-28 09:00:08 +00:00
|
|
|
Some(CapsFeaturesRef::from_glib_borrow_mut(features))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn set_features(&mut self, idx: u32, features: Option<CapsFeatures>) {
|
|
|
|
assert!(idx < self.get_size());
|
|
|
|
|
|
|
|
unsafe {
|
2019-03-19 07:58:20 +00:00
|
|
|
gst_sys::gst_caps_set_features(
|
2018-09-28 09:00:08 +00:00
|
|
|
self.as_mut_ptr(),
|
|
|
|
idx,
|
|
|
|
features.map(|f| f.into_ptr()).unwrap_or(ptr::null_mut()),
|
|
|
|
)
|
2017-07-10 21:02:08 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-23 16:53:10 +00:00
|
|
|
#[cfg(any(feature = "v1_16", feature = "dox"))]
|
|
|
|
pub fn set_features_simple(&mut self, features: Option<CapsFeatures>) {
|
|
|
|
unsafe {
|
|
|
|
gst_sys::gst_caps_set_features_simple(
|
|
|
|
self.as_mut_ptr(),
|
|
|
|
features.map(|f| f.into_ptr()).unwrap_or(ptr::null_mut()),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-10 21:02:08 +00:00
|
|
|
pub fn get_size(&self) -> u32 {
|
2019-03-19 07:58:20 +00:00
|
|
|
unsafe { gst_sys::gst_caps_get_size(self.as_ptr()) }
|
2017-07-10 21:02:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn iter(&self) -> Iter {
|
|
|
|
Iter::new(self)
|
|
|
|
}
|
|
|
|
|
2017-07-24 22:33:14 +00:00
|
|
|
pub fn iter_mut(&mut self) -> IterMut {
|
|
|
|
IterMut::new(self)
|
|
|
|
}
|
2017-07-10 21:02:08 +00:00
|
|
|
|
2018-09-28 09:00:08 +00:00
|
|
|
pub fn iter_with_features(&self) -> IterFeatures {
|
|
|
|
IterFeatures::new(self)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn iter_with_features_mut(&mut self) -> IterFeaturesMut {
|
|
|
|
IterFeaturesMut::new(self)
|
|
|
|
}
|
|
|
|
|
2017-07-24 23:03:18 +00:00
|
|
|
pub fn append_structure(&mut self, structure: Structure) {
|
2019-03-19 07:58:20 +00:00
|
|
|
unsafe { gst_sys::gst_caps_append_structure(self.as_mut_ptr(), structure.into_ptr()) }
|
2017-07-24 23:03:18 +00:00
|
|
|
}
|
|
|
|
|
2018-09-28 09:00:08 +00:00
|
|
|
pub fn append_structure_full(&mut self, structure: Structure, features: Option<CapsFeatures>) {
|
|
|
|
unsafe {
|
2019-03-19 07:58:20 +00:00
|
|
|
gst_sys::gst_caps_append_structure_full(
|
2018-09-28 09:00:08 +00:00
|
|
|
self.as_mut_ptr(),
|
|
|
|
structure.into_ptr(),
|
|
|
|
features.map(|f| f.into_ptr()).unwrap_or(ptr::null_mut()),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-24 23:03:18 +00:00
|
|
|
pub fn remove_structure(&mut self, idx: u32) {
|
2019-03-19 07:58:20 +00:00
|
|
|
unsafe { gst_sys::gst_caps_remove_structure(self.as_mut_ptr(), idx) }
|
2017-07-24 23:03:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn append(&mut self, other: Caps) {
|
2019-03-19 07:58:20 +00:00
|
|
|
unsafe { gst_sys::gst_caps_append(self.as_mut_ptr(), other.into_ptr()) }
|
2017-07-24 23:03:18 +00:00
|
|
|
}
|
|
|
|
|
2018-07-06 15:58:54 +00:00
|
|
|
pub fn can_intersect(&self, other: &Self) -> bool {
|
2019-03-19 07:58:20 +00:00
|
|
|
unsafe {
|
|
|
|
from_glib(gst_sys::gst_caps_can_intersect(
|
|
|
|
self.as_ptr(),
|
|
|
|
other.as_ptr(),
|
|
|
|
))
|
|
|
|
}
|
2017-07-24 23:03:18 +00:00
|
|
|
}
|
|
|
|
|
2018-07-06 15:58:54 +00:00
|
|
|
pub fn intersect(&self, other: &Self) -> Caps {
|
2017-07-24 23:03:18 +00:00
|
|
|
unsafe {
|
2019-03-19 07:58:20 +00:00
|
|
|
from_glib_full(gst_sys::gst_caps_intersect(
|
2017-07-24 23:03:18 +00:00
|
|
|
self.as_mut_ptr(),
|
|
|
|
other.as_mut_ptr(),
|
|
|
|
))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-06 15:58:54 +00:00
|
|
|
pub fn intersect_with_mode(&self, other: &Self, mode: CapsIntersectMode) -> Caps {
|
2017-07-24 23:03:18 +00:00
|
|
|
unsafe {
|
2019-03-19 07:58:20 +00:00
|
|
|
from_glib_full(gst_sys::gst_caps_intersect_full(
|
2017-07-24 23:03:18 +00:00
|
|
|
self.as_mut_ptr(),
|
|
|
|
other.as_mut_ptr(),
|
|
|
|
mode.to_glib(),
|
|
|
|
))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-06 15:58:54 +00:00
|
|
|
pub fn is_always_compatible(&self, other: &Self) -> bool {
|
2017-07-24 23:03:18 +00:00
|
|
|
unsafe {
|
2019-03-19 07:58:20 +00:00
|
|
|
from_glib(gst_sys::gst_caps_is_always_compatible(
|
2017-07-24 23:03:18 +00:00
|
|
|
self.as_ptr(),
|
|
|
|
other.as_ptr(),
|
|
|
|
))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn is_any(&self) -> bool {
|
2019-03-19 07:58:20 +00:00
|
|
|
unsafe { from_glib(gst_sys::gst_caps_is_any(self.as_ptr())) }
|
2017-07-24 23:03:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn is_empty(&self) -> bool {
|
2019-03-19 07:58:20 +00:00
|
|
|
unsafe { from_glib(gst_sys::gst_caps_is_empty(self.as_ptr())) }
|
2017-07-24 23:03:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn is_fixed(&self) -> bool {
|
2019-03-19 07:58:20 +00:00
|
|
|
unsafe { from_glib(gst_sys::gst_caps_is_fixed(self.as_ptr())) }
|
2017-07-24 23:03:18 +00:00
|
|
|
}
|
|
|
|
|
2018-07-06 15:58:54 +00:00
|
|
|
pub fn is_equal_fixed(&self, other: &Self) -> bool {
|
2019-03-19 07:58:20 +00:00
|
|
|
unsafe {
|
|
|
|
from_glib(gst_sys::gst_caps_is_equal_fixed(
|
|
|
|
self.as_ptr(),
|
|
|
|
other.as_ptr(),
|
|
|
|
))
|
|
|
|
}
|
2017-07-24 23:03:18 +00:00
|
|
|
}
|
|
|
|
|
2018-07-06 15:58:54 +00:00
|
|
|
pub fn is_strictly_equal(&self, other: &Self) -> bool {
|
2017-07-24 23:03:18 +00:00
|
|
|
unsafe {
|
2019-03-19 07:58:20 +00:00
|
|
|
from_glib(gst_sys::gst_caps_is_strictly_equal(
|
2017-07-24 23:03:18 +00:00
|
|
|
self.as_ptr(),
|
|
|
|
other.as_ptr(),
|
|
|
|
))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-06 15:58:54 +00:00
|
|
|
pub fn is_subset(&self, superset: &Self) -> bool {
|
2019-03-19 07:58:20 +00:00
|
|
|
unsafe {
|
|
|
|
from_glib(gst_sys::gst_caps_is_subset(
|
|
|
|
self.as_ptr(),
|
|
|
|
superset.as_ptr(),
|
|
|
|
))
|
|
|
|
}
|
2017-07-24 23:03:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn is_subset_structure(&self, structure: &StructureRef) -> bool {
|
|
|
|
unsafe {
|
2019-03-19 07:58:20 +00:00
|
|
|
from_glib(gst_sys::gst_caps_is_subset_structure(
|
2017-07-24 23:03:18 +00:00
|
|
|
self.as_ptr(),
|
|
|
|
structure.as_ptr(),
|
|
|
|
))
|
|
|
|
}
|
|
|
|
}
|
2018-07-06 15:56:55 +00:00
|
|
|
|
2018-09-28 09:00:08 +00:00
|
|
|
pub fn is_subset_structure_full(
|
|
|
|
&self,
|
|
|
|
structure: &StructureRef,
|
|
|
|
features: Option<&CapsFeaturesRef>,
|
|
|
|
) -> bool {
|
|
|
|
unsafe {
|
2019-03-19 07:58:20 +00:00
|
|
|
from_glib(gst_sys::gst_caps_is_subset_structure_full(
|
2018-09-28 09:00:08 +00:00
|
|
|
self.as_ptr(),
|
|
|
|
structure.as_ptr(),
|
|
|
|
features.map(|f| f.as_ptr()).unwrap_or(ptr::null()),
|
|
|
|
))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-06 15:56:55 +00:00
|
|
|
pub fn subtract(&self, other: &Self) -> Caps {
|
|
|
|
skip_assert_initialized!();
|
2018-07-27 10:36:40 +00:00
|
|
|
unsafe {
|
2019-03-19 07:58:20 +00:00
|
|
|
from_glib_full(gst_sys::gst_caps_subtract(
|
2018-07-27 10:36:40 +00:00
|
|
|
self.as_mut_ptr(),
|
|
|
|
other.as_mut_ptr(),
|
|
|
|
))
|
|
|
|
}
|
2018-07-06 15:56:55 +00:00
|
|
|
}
|
2017-07-10 21:02:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
macro_rules! define_iter(
|
2018-09-28 09:00:08 +00:00
|
|
|
($name:ident, $typ:ty, $styp:ty, $get_item:expr) => {
|
2019-01-22 15:43:29 +00:00
|
|
|
#[derive(Debug)]
|
2017-08-30 11:39:09 +00:00
|
|
|
pub struct $name<'a> {
|
|
|
|
caps: $typ,
|
|
|
|
idx: u32,
|
|
|
|
n_structures: u32,
|
|
|
|
}
|
2017-07-10 21:02:08 +00:00
|
|
|
|
2017-08-30 11:39:09 +00:00
|
|
|
impl<'a> $name<'a> {
|
|
|
|
fn new(caps: $typ) -> $name<'a> {
|
|
|
|
skip_assert_initialized!();
|
|
|
|
let n_structures = caps.get_size();
|
2017-07-10 21:02:08 +00:00
|
|
|
|
2017-08-30 11:39:09 +00:00
|
|
|
$name {
|
|
|
|
caps: caps,
|
|
|
|
idx: 0,
|
|
|
|
n_structures: n_structures,
|
|
|
|
}
|
2017-07-10 21:02:08 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-30 11:39:09 +00:00
|
|
|
impl<'a> Iterator for $name<'a> {
|
|
|
|
type Item = $styp;
|
2017-07-10 21:02:08 +00:00
|
|
|
|
2017-08-30 11:39:09 +00:00
|
|
|
fn next(&mut self) -> Option<Self::Item> {
|
|
|
|
if self.idx >= self.n_structures {
|
2017-07-24 22:33:14 +00:00
|
|
|
return None;
|
|
|
|
}
|
|
|
|
|
2017-08-30 11:39:09 +00:00
|
|
|
unsafe {
|
2018-10-11 08:30:12 +00:00
|
|
|
let item = $get_item(self.caps, self.idx)?;
|
2017-08-30 11:39:09 +00:00
|
|
|
self.idx += 1;
|
2018-10-11 08:30:12 +00:00
|
|
|
Some(item)
|
2017-08-30 11:39:09 +00:00
|
|
|
}
|
2017-07-10 21:02:08 +00:00
|
|
|
}
|
|
|
|
|
2017-08-30 11:39:09 +00:00
|
|
|
fn size_hint(&self) -> (usize, Option<usize>) {
|
|
|
|
if self.idx == self.n_structures {
|
|
|
|
return (0, Some(0));
|
|
|
|
}
|
2017-07-10 21:02:08 +00:00
|
|
|
|
2017-08-30 11:39:09 +00:00
|
|
|
let remaining = (self.n_structures - self.idx) as usize;
|
2017-07-10 21:02:08 +00:00
|
|
|
|
2017-08-30 11:39:09 +00:00
|
|
|
(remaining, Some(remaining))
|
2017-07-10 21:02:08 +00:00
|
|
|
}
|
2017-08-30 11:39:09 +00:00
|
|
|
}
|
2017-07-10 21:02:08 +00:00
|
|
|
|
2017-08-30 11:39:09 +00:00
|
|
|
impl<'a> DoubleEndedIterator for $name<'a> {
|
|
|
|
fn next_back(&mut self) -> Option<Self::Item> {
|
|
|
|
if self.idx == self.n_structures {
|
2017-07-24 22:33:14 +00:00
|
|
|
return None;
|
|
|
|
}
|
|
|
|
|
2017-08-30 11:39:09 +00:00
|
|
|
self.n_structures -= 1;
|
|
|
|
|
|
|
|
unsafe {
|
2018-09-28 09:00:08 +00:00
|
|
|
$get_item(self.caps, self.n_structures)
|
2017-08-30 11:39:09 +00:00
|
|
|
}
|
2017-07-10 21:02:08 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-30 11:39:09 +00:00
|
|
|
impl<'a> ExactSizeIterator for $name<'a> {}
|
|
|
|
}
|
2017-07-10 21:02:08 +00:00
|
|
|
);
|
|
|
|
|
2018-09-28 09:00:08 +00:00
|
|
|
define_iter!(
|
|
|
|
Iter,
|
|
|
|
&'a CapsRef,
|
|
|
|
&'a StructureRef,
|
|
|
|
|caps: &CapsRef, idx| {
|
2019-03-19 07:58:20 +00:00
|
|
|
let ptr = gst_sys::gst_caps_get_structure(caps.as_ptr(), idx);
|
2018-09-28 09:00:08 +00:00
|
|
|
if ptr.is_null() {
|
|
|
|
None
|
|
|
|
} else {
|
|
|
|
Some(StructureRef::from_glib_borrow(
|
2019-03-19 07:58:20 +00:00
|
|
|
ptr as *const gst_sys::GstStructure,
|
2018-09-28 09:00:08 +00:00
|
|
|
))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
define_iter!(
|
|
|
|
IterMut,
|
|
|
|
&'a mut CapsRef,
|
|
|
|
&'a mut StructureRef,
|
|
|
|
|caps: &CapsRef, idx| {
|
2019-03-19 07:58:20 +00:00
|
|
|
let ptr = gst_sys::gst_caps_get_structure(caps.as_ptr(), idx);
|
2018-09-28 09:00:08 +00:00
|
|
|
if ptr.is_null() {
|
|
|
|
None
|
|
|
|
} else {
|
|
|
|
Some(StructureRef::from_glib_borrow_mut(
|
2019-03-19 07:58:20 +00:00
|
|
|
ptr as *mut gst_sys::GstStructure,
|
2018-09-28 09:00:08 +00:00
|
|
|
))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
define_iter!(
|
|
|
|
IterFeatures,
|
|
|
|
&'a CapsRef,
|
|
|
|
(&'a StructureRef, &'a CapsFeaturesRef),
|
|
|
|
|caps: &CapsRef, idx| {
|
2019-03-19 07:58:20 +00:00
|
|
|
let ptr1 = gst_sys::gst_caps_get_structure(caps.as_ptr(), idx);
|
|
|
|
let ptr2 = gst_sys::gst_caps_get_features(caps.as_ptr(), idx);
|
2018-09-28 09:00:08 +00:00
|
|
|
if ptr1.is_null() || ptr2.is_null() {
|
|
|
|
None
|
|
|
|
} else {
|
|
|
|
Some((
|
2019-03-19 07:58:20 +00:00
|
|
|
StructureRef::from_glib_borrow(ptr1 as *const gst_sys::GstStructure),
|
|
|
|
CapsFeaturesRef::from_glib_borrow(ptr2 as *const gst_sys::GstCapsFeatures),
|
2018-09-28 09:00:08 +00:00
|
|
|
))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
define_iter!(
|
|
|
|
IterFeaturesMut,
|
|
|
|
&'a mut CapsRef,
|
|
|
|
(&'a mut StructureRef, &'a mut CapsFeaturesRef),
|
|
|
|
|caps: &CapsRef, idx| {
|
2019-03-19 07:58:20 +00:00
|
|
|
let ptr1 = gst_sys::gst_caps_get_structure(caps.as_ptr(), idx);
|
|
|
|
let ptr2 = gst_sys::gst_caps_get_features(caps.as_ptr(), idx);
|
2018-09-28 09:00:08 +00:00
|
|
|
if ptr1.is_null() || ptr2.is_null() {
|
|
|
|
None
|
|
|
|
} else {
|
|
|
|
Some((
|
2019-03-19 07:58:20 +00:00
|
|
|
StructureRef::from_glib_borrow_mut(ptr1 as *mut gst_sys::GstStructure),
|
|
|
|
CapsFeaturesRef::from_glib_borrow_mut(ptr2 as *mut gst_sys::GstCapsFeatures),
|
2018-09-28 09:00:08 +00:00
|
|
|
))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
2017-07-10 21:02:08 +00:00
|
|
|
|
|
|
|
impl fmt::Debug for CapsRef {
|
2017-12-01 09:21:20 +00:00
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
2017-12-01 17:02:53 +00:00
|
|
|
f.debug_tuple("Caps").field(&self.to_string()).finish()
|
2017-12-01 09:21:20 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl fmt::Display for CapsRef {
|
2017-07-10 21:02:08 +00:00
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
2019-10-04 06:11:30 +00:00
|
|
|
let s =
|
|
|
|
unsafe { glib::GString::from_glib_full(gst_sys::gst_caps_to_string(self.as_ptr())) };
|
|
|
|
f.write_str(&s)
|
2017-07-10 21:02:08 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl PartialEq for CapsRef {
|
|
|
|
fn eq(&self, other: &CapsRef) -> bool {
|
2019-03-19 07:58:20 +00:00
|
|
|
unsafe { from_glib(gst_sys::gst_caps_is_equal(self.as_ptr(), other.as_ptr())) }
|
2017-07-10 21:02:08 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Eq for CapsRef {}
|
|
|
|
|
2019-01-22 15:43:29 +00:00
|
|
|
#[derive(Debug)]
|
2018-09-28 09:00:08 +00:00
|
|
|
pub struct Builder<'a> {
|
2017-11-11 12:14:49 +00:00
|
|
|
s: ::Structure,
|
2018-09-28 09:00:08 +00:00
|
|
|
features: Option<&'a [&'a str]>,
|
|
|
|
any_features: bool,
|
2017-11-11 12:14:49 +00:00
|
|
|
}
|
|
|
|
|
2018-09-28 09:00:08 +00:00
|
|
|
impl<'a> Builder<'a> {
|
|
|
|
fn new<'b>(name: &'b str) -> Builder<'a> {
|
2017-11-11 12:14:49 +00:00
|
|
|
Builder {
|
|
|
|
s: ::Structure::new_empty(name),
|
2018-09-28 09:00:08 +00:00
|
|
|
features: None,
|
|
|
|
any_features: false,
|
2017-11-11 12:14:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-28 09:00:08 +00:00
|
|
|
pub fn field<'b, V: ToSendValue>(mut self, name: &'b str, value: &'b V) -> Self {
|
2017-11-11 12:14:49 +00:00
|
|
|
self.s.set(name, value);
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
2018-09-28 09:00:08 +00:00
|
|
|
pub fn features(mut self, features: &'a [&'a str]) -> Self {
|
|
|
|
self.features = Some(features);
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn any_features(mut self) -> Self {
|
|
|
|
self.any_features = true;
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
2017-11-11 12:14:49 +00:00
|
|
|
pub fn build(self) -> Caps {
|
|
|
|
let mut caps = Caps::new_empty();
|
2018-09-28 09:00:08 +00:00
|
|
|
let features = if self.any_features {
|
|
|
|
Some(CapsFeatures::new_any())
|
|
|
|
} else {
|
|
|
|
self.features.map(|f| CapsFeatures::new(f))
|
|
|
|
};
|
|
|
|
|
|
|
|
caps.get_mut()
|
|
|
|
.unwrap()
|
|
|
|
.append_structure_full(self.s, features);
|
2017-11-11 12:14:49 +00:00
|
|
|
caps
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-10 21:02:08 +00:00
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
use super::*;
|
2017-07-28 15:56:27 +00:00
|
|
|
use Array;
|
2018-04-01 08:30:03 +00:00
|
|
|
use Fraction;
|
2017-07-10 21:02:08 +00:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_simple() {
|
2017-07-12 07:32:39 +00:00
|
|
|
::init().unwrap();
|
2017-07-10 21:02:08 +00:00
|
|
|
|
2018-09-28 09:00:08 +00:00
|
|
|
let mut caps = Caps::new_simple(
|
2017-07-10 21:02:08 +00:00
|
|
|
"foo/bar",
|
|
|
|
&[
|
2017-07-27 22:36:44 +00:00
|
|
|
("int", &12),
|
|
|
|
("bool", &true),
|
|
|
|
("string", &"bla"),
|
|
|
|
("fraction", &Fraction::new(1, 2)),
|
2017-07-28 15:56:27 +00:00
|
|
|
("array", &Array::new(&[&1, &2])),
|
2017-07-10 21:02:08 +00:00
|
|
|
],
|
|
|
|
);
|
|
|
|
assert_eq!(
|
|
|
|
caps.to_string(),
|
2017-07-28 15:56:27 +00:00
|
|
|
"foo/bar, int=(int)12, bool=(boolean)true, string=(string)bla, fraction=(fraction)1/2, array=(int)< 1, 2 >"
|
|
|
|
);
|
2017-07-10 21:02:08 +00:00
|
|
|
|
2018-09-28 09:00:08 +00:00
|
|
|
{
|
|
|
|
let s = caps.get_structure(0).unwrap();
|
|
|
|
assert_eq!(
|
|
|
|
s,
|
|
|
|
Structure::new(
|
|
|
|
"foo/bar",
|
|
|
|
&[
|
|
|
|
("int", &12),
|
|
|
|
("bool", &true),
|
|
|
|
("string", &"bla"),
|
|
|
|
("fraction", &Fraction::new(1, 2)),
|
|
|
|
("array", &Array::new(&[&1, &2])),
|
|
|
|
],
|
2018-10-08 12:02:23 +00:00
|
|
|
)
|
|
|
|
.as_ref()
|
2018-09-28 09:00:08 +00:00
|
|
|
);
|
|
|
|
}
|
2018-10-28 13:47:02 +00:00
|
|
|
assert!(caps
|
|
|
|
.get_features(0)
|
|
|
|
.unwrap()
|
|
|
|
.is_equal(::CAPS_FEATURES_MEMORY_SYSTEM_MEMORY.as_ref()));
|
2018-09-28 09:00:08 +00:00
|
|
|
|
|
|
|
{
|
|
|
|
let caps = caps.get_mut().unwrap();
|
|
|
|
caps.set_features(0, Some(CapsFeatures::new(&["foo:bla"])));
|
|
|
|
}
|
2018-10-28 13:47:02 +00:00
|
|
|
assert!(caps
|
|
|
|
.get_features(0)
|
|
|
|
.unwrap()
|
|
|
|
.is_equal(CapsFeatures::new(&["foo:bla"]).as_ref()));
|
2017-07-10 21:02:08 +00:00
|
|
|
}
|
2017-11-11 12:14:49 +00:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_builder() {
|
|
|
|
::init().unwrap();
|
|
|
|
|
|
|
|
let caps = Caps::builder("foo/bar")
|
2017-11-26 21:58:02 +00:00
|
|
|
.field("int", &12)
|
|
|
|
.field("bool", &true)
|
|
|
|
.field("string", &"bla")
|
|
|
|
.field("fraction", &Fraction::new(1, 2))
|
|
|
|
.field("array", &Array::new(&[&1, &2]))
|
2017-11-11 12:14:49 +00:00
|
|
|
.build();
|
|
|
|
assert_eq!(
|
|
|
|
caps.to_string(),
|
|
|
|
"foo/bar, int=(int)12, bool=(boolean)true, string=(string)bla, fraction=(fraction)1/2, array=(int)< 1, 2 >"
|
|
|
|
);
|
2018-09-28 09:00:08 +00:00
|
|
|
|
|
|
|
let caps = Caps::builder("foo/bar")
|
|
|
|
.field("int", &12)
|
|
|
|
.any_features()
|
|
|
|
.build();
|
|
|
|
assert_eq!(caps.to_string(), "foo/bar(ANY), int=(int)12");
|
|
|
|
|
|
|
|
let caps = Caps::builder("foo/bar")
|
|
|
|
.field("int", &12)
|
|
|
|
.features(&["foo:bla", "foo:baz"])
|
|
|
|
.build();
|
|
|
|
assert_eq!(caps.to_string(), "foo/bar(foo:bla, foo:baz), int=(int)12");
|
2017-11-11 12:14:49 +00:00
|
|
|
}
|
2019-06-18 11:03:19 +00:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_display() {
|
|
|
|
::init().unwrap();
|
|
|
|
|
|
|
|
let caps = Caps::new_simple("foo/bar", &[]);
|
|
|
|
format!("{}", caps);
|
|
|
|
}
|
2017-07-10 21:02:08 +00:00
|
|
|
}
|