this->parent.close_object = this;
mlt_events_init( &this->parent );
+ mlt_events_register( &this->parent, "service-changed", NULL );
mlt_events_register( &this->parent, "property-changed", ( mlt_transmitter )mlt_service_property_changed );
}
/** Recursively apply attached filters
*/
-static void apply_filters( mlt_service this, mlt_frame frame, int index )
+void mlt_service_apply_filters( mlt_service this, mlt_frame frame, int index )
{
- mlt_properties properties = mlt_service_properties( this );
- mlt_service_base *base = this->local;
int i;
+ mlt_properties frame_properties = mlt_frame_properties( frame );
+ mlt_properties filter_properties = mlt_service_properties( this );
+ mlt_service_base *base = this->local;
+ mlt_position position = mlt_properties_get_position( frame_properties, "_position" );
- if ( mlt_properties_get_int( properties, "_filter_private" ) == 0 )
+ if ( index == 0 || mlt_properties_get_int( filter_properties, "_filter_private" ) == 0 )
{
// Process the frame with the attached filters
for ( i = 0; i < base->filter_count; i ++ )
{
if ( base->filters[ i ] != NULL )
{
- mlt_filter_process( base->filters[ i ], frame );
- apply_filters( mlt_filter_service( base->filters[ i ] ), frame, index );
+ mlt_properties properties = mlt_filter_properties( base->filters[ i ] );
+ mlt_position in = mlt_properties_get_position( properties, "in" );
+ mlt_position out = mlt_properties_get_position( properties, "out" );
+ if ( ( in == 0 && out == 0 ) || ( position >= in && ( position <= out || out == 0 ) ) )
+ {
+ mlt_filter_process( base->filters[ i ], frame );
+ mlt_properties_set_position( frame_properties, "_position", position - in );
+ mlt_service_apply_filters( mlt_filter_service( base->filters[ i ] ), frame, index + 1 );
+ mlt_properties_set_position( frame_properties, "_position", position );
+ }
}
}
}
{
int result = this->get_frame( this, frame, index );
if ( result == 0 )
- apply_filters( this, *frame, index );
+ mlt_service_apply_filters( this, *frame, 1 );
return result;
}
*frame = mlt_frame_init( );
return 0;
}
+static void mlt_service_filter_changed( mlt_service owner, mlt_service this )
+{
+ mlt_events_fire( mlt_service_properties( this ), "service-changed", NULL );
+}
+
/** Attach a filter.
*/
if ( base->filters != NULL )
{
+ mlt_properties props = mlt_filter_properties( filter );
mlt_properties_inc_ref( mlt_filter_properties( filter ) );
base->filters[ base->filter_count ++ ] = filter;
mlt_events_fire( properties, "service-changed", NULL );
+ mlt_events_listen( props, this, "service-changed", ( mlt_listener )mlt_service_filter_changed );
+ mlt_events_listen( props, this, "property-changed", ( mlt_listener )mlt_service_filter_changed );
}
else
{
for ( i ++ ; i < base->filter_count; i ++ )
base->filters[ i - 1 ] = base->filters[ i ];
base->filter_count --;
+ mlt_events_disconnect( mlt_filter_properties( filter ), this );
mlt_filter_close( filter );
mlt_events_fire( properties, "service-changed", NULL );
}
mlt_service_base *base = this->local;
int i = 0;
int count = base->filter_count;
+ mlt_events_block( mlt_service_properties( this ), this );
while( count -- )
mlt_service_detach( this, base->filters[ 0 ] );
free( base->filters );