// Take a look at the license at the top of the repository in the LICENSE file. use glib::object::{Cast, ObjectExt}; use glib::signal::SignalHandlerId; use glib::translate::*; use glib::IsA; use crate::ClockTime; use crate::ObjectFlags; pub trait GstObjectExtManual: 'static { fn connect_deep_notify( &self, name: Option<&str>, f: F, ) -> SignalHandlerId; fn set_object_flags(&self, flags: ObjectFlags); fn unset_object_flags(&self, flags: ObjectFlags); fn get_object_flags(&self) -> ObjectFlags; fn get_g_value_array( &self, property_name: &str, timestamp: ClockTime, interval: ClockTime, values: &mut [glib::Value], ) -> Result<(), glib::error::BoolError>; } impl> GstObjectExtManual for O { fn connect_deep_notify< F: Fn(&Self, &crate::Object, &glib::ParamSpec) + Send + Sync + 'static, >( &self, name: Option<&str>, f: F, ) -> SignalHandlerId { let signal_name = if let Some(name) = name { format!("deep-notify::{}", name) } else { "deep-notify".into() }; let obj: Borrowed = unsafe { from_glib_borrow(self.as_ptr() as *mut glib::gobject_ffi::GObject) }; obj.connect(signal_name.as_str(), false, move |values| { // It would be nice to display the actual signal name in the panic messages below, // but that would require to copy `signal_name` so as to move it into the closure // which seems too much for the messages of development errors let obj: O = unsafe { values[0] .get::() .unwrap_or_else(|err| { panic!("Object signal \"deep-notify\": values[0]: {}", err) }) .expect("Object signal \"deep-notify\": values[0] not defined") .unsafe_cast() }; let prop_obj: crate::Object = values[1] .get() .unwrap_or_else(|err| panic!("Object signal \"deep-notify\": values[1]: {}", err)) .expect("Object signal \"deep-notify\": values[1] not defined"); let pspec = unsafe { let pspec = glib::gobject_ffi::g_value_get_param(values[2].to_glib_none().0); from_glib_none(pspec) }; f(&obj, &prop_obj, &pspec); None }) .unwrap() } fn set_object_flags(&self, flags: ObjectFlags) { unsafe { let ptr: *mut ffi::GstObject = self.as_ptr() as *mut _; let _guard = crate::utils::MutexGuard::lock(&(*ptr).lock); (*ptr).flags |= flags.to_glib(); } } fn unset_object_flags(&self, flags: ObjectFlags) { unsafe { let ptr: *mut ffi::GstObject = self.as_ptr() as *mut _; let _guard = crate::utils::MutexGuard::lock(&(*ptr).lock); (*ptr).flags &= !flags.to_glib(); } } fn get_object_flags(&self) -> ObjectFlags { unsafe { let ptr: *mut ffi::GstObject = self.as_ptr() as *mut _; let _guard = crate::utils::MutexGuard::lock(&(*ptr).lock); from_glib((*ptr).flags) } } fn get_g_value_array( &self, property_name: &str, timestamp: ClockTime, interval: ClockTime, values: &mut [glib::Value], ) -> Result<(), glib::error::BoolError> { let n_values = values.len() as u32; unsafe { glib::result_from_gboolean!( ffi::gst_object_get_g_value_array( self.as_ref().to_glib_none().0, property_name.to_glib_none().0, timestamp.to_glib(), interval.to_glib(), n_values, values.as_mut_ptr() as *mut glib::gobject_ffi::GValue, ), "Failed to get value array" ) } } } #[cfg(test)] mod tests { use super::*; use crate::prelude::*; use std::sync::{Arc, Mutex}; #[test] fn test_deep_notify() { crate::init().unwrap(); let bin = crate::Bin::new(None); let identity = crate::ElementFactory::make("identity", Some("id")).unwrap(); bin.add(&identity).unwrap(); let notify = Arc::new(Mutex::new(None)); let notify_clone = notify.clone(); bin.connect_deep_notify(None, move |_, id, prop| { *notify_clone.lock().unwrap() = Some((id.clone(), prop.get_name())); }); identity.set_property("silent", &false).unwrap(); assert_eq!( *notify.lock().unwrap(), Some((identity.upcast::(), "silent")) ); } }