Corrections to filter attachment and in/out point handling
[melted] / src / modules / inigo / producer_inigo.c
1 /*
2 * producer_inigo.c -- simple inigo test case
3 * Copyright (C) 2003-2004 Ushodaya Enterprises Limited
4 * Author: Charles Yates <charles.yates@pandora.be>
5 *
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; either version 2 of the License, or
9 * (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, write to the Free Software Foundation,
18 * Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
19 */
20
21 #include "producer_inigo.h"
22
23 #include <stdio.h>
24 #include <stdlib.h>
25 #include <string.h>
26
27 #include <framework/mlt.h>
28
29 mlt_producer producer_inigo_file_init( char *file )
30 {
31 FILE *input = fopen( file, "r" );
32 char **args = calloc( sizeof( char * ), 1000 );
33 int count = 0;
34 char temp[ 2048 ];
35
36 if ( input != NULL )
37 {
38 while( fgets( temp, 2048, input ) )
39 {
40 temp[ strlen( temp ) - 1 ] = '\0';
41 if ( strcmp( temp, "" ) )
42 args[ count ++ ] = strdup( temp );
43 }
44 }
45
46 mlt_producer result = producer_inigo_init( args );
47
48 if ( result != NULL )
49 {
50 mlt_properties properties = mlt_producer_properties( result );
51 mlt_properties_set( properties, "resource", file );
52 }
53
54 while( count -- )
55 free( args[ count ] );
56 free( args );
57
58 return result;
59 }
60
61 static void track_service( mlt_field field, void *service, mlt_destructor destructor )
62 {
63 mlt_properties properties = mlt_field_properties( field );
64 int registered = mlt_properties_get_int( properties, "registered" );
65 char *key = mlt_properties_get( properties, "registered" );
66 mlt_properties_set_data( properties, key, service, 0, destructor, NULL );
67 mlt_properties_set_int( properties, "registered", ++ registered );
68 }
69
70 static mlt_producer create_producer( mlt_field field, char *file )
71 {
72 mlt_producer result = mlt_factory_producer( "fezzik", file );
73
74 if ( result != NULL )
75 track_service( field, result, ( mlt_destructor )mlt_producer_close );
76
77 return result;
78 }
79
80 static mlt_filter create_attach( mlt_field field, char *id, int track )
81 {
82 char *arg = strchr( id, ':' );
83 if ( arg != NULL )
84 *arg ++ = '\0';
85 mlt_filter filter = mlt_factory_filter( id, arg );
86 if ( filter != NULL )
87 track_service( field, filter, ( mlt_destructor )mlt_filter_close );
88 return filter;
89 }
90
91 static mlt_filter create_filter( mlt_field field, char *id, int track )
92 {
93 char *arg = strchr( id, ':' );
94 if ( arg != NULL )
95 *arg ++ = '\0';
96 mlt_filter filter = mlt_factory_filter( id, arg );
97 if ( filter != NULL )
98 {
99 mlt_field_plant_filter( field, filter, track );
100 track_service( field, filter, ( mlt_destructor )mlt_filter_close );
101 }
102 return filter;
103 }
104
105 static mlt_transition create_transition( mlt_field field, char *id, int track )
106 {
107 char *arg = strchr( id, ':' );
108 if ( arg != NULL )
109 *arg ++ = '\0';
110 mlt_transition transition = mlt_factory_transition( id, arg );
111 if ( transition != NULL )
112 {
113 mlt_field_plant_transition( field, transition, track, track + 1 );
114 track_service( field, transition, ( mlt_destructor )mlt_transition_close );
115 }
116 return transition;
117 }
118
119 mlt_producer producer_inigo_init( char **argv )
120 {
121 int i;
122 int track = 0;
123 mlt_producer producer = NULL;
124 mlt_tractor mix = NULL;
125 mlt_playlist playlist = mlt_playlist_init( );
126 mlt_properties group = mlt_properties_new( );
127 mlt_properties properties = group;
128 mlt_tractor tractor = mlt_tractor_new( );
129 mlt_field field = mlt_tractor_field( tractor );
130 mlt_properties field_properties = mlt_field_properties( field );
131 mlt_multitrack multitrack = mlt_tractor_multitrack( tractor );
132 char *title = NULL;
133
134 // We need to track the number of registered filters
135 mlt_properties_set_int( field_properties, "registered", 0 );
136
137 // Parse the arguments
138 for ( i = 0; argv[ i ] != NULL; i ++ )
139 {
140 if ( !strcmp( argv[ i ], "-group" ) )
141 {
142 if ( mlt_properties_count( group ) != 0 )
143 {
144 mlt_properties_close( group );
145 group = mlt_properties_new( );
146 }
147 if ( group != NULL )
148 properties = group;
149 }
150 else if ( !strcmp( argv[ i ], "-attach" ) || !strcmp( argv[ i ], "-chain" ) )
151 {
152 int type = !strcmp( argv[ i ], "-attach" ) ? 0 : 1;
153 mlt_filter filter = create_attach( field, argv[ ++ i ], track );
154 if ( producer != NULL && !mlt_producer_is_cut( producer ) )
155 {
156 mlt_playlist_clip_info info;
157 mlt_playlist_append( playlist, producer );
158 mlt_playlist_get_clip_info( playlist, &info, mlt_playlist_count( playlist ) - 1 );
159 producer = info.cut;
160 }
161
162 if ( filter != NULL && mlt_playlist_count( playlist ) > 0 )
163 {
164 if ( type == 0 )
165 mlt_service_attach( ( mlt_service )properties, filter );
166 else
167 mlt_service_attach( ( mlt_service )producer, filter );
168
169 properties = mlt_filter_properties( filter );
170 mlt_properties_inherit( properties, group );
171 }
172 }
173 else if ( !strcmp( argv[ i ], "-repeat" ) )
174 {
175 int repeat = atoi( argv[ ++ i ] );
176 if ( producer != NULL && !mlt_producer_is_cut( producer ) )
177 mlt_playlist_append( playlist, producer );
178 producer = NULL;
179 if ( mlt_playlist_count( playlist ) > 0 )
180 {
181 mlt_playlist_clip_info info;
182 mlt_playlist_repeat_clip( playlist, mlt_playlist_count( playlist ) - 1, repeat );
183 mlt_playlist_get_clip_info( playlist, &info, mlt_playlist_count( playlist ) - 1 );
184 producer = info.cut;
185 properties = mlt_producer_properties( producer );
186 }
187 }
188 else if ( !strcmp( argv[ i ], "-split" ) )
189 {
190 int split = atoi( argv[ ++ i ] );
191 if ( producer != NULL && !mlt_producer_is_cut( producer ) )
192 mlt_playlist_append( playlist, producer );
193 producer = NULL;
194 if ( mlt_playlist_count( playlist ) > 0 )
195 {
196 mlt_playlist_clip_info info;
197 mlt_playlist_get_clip_info( playlist, &info, mlt_playlist_count( playlist ) - 1 );
198 split = split < 0 ? info.frame_out + split : split;
199 mlt_playlist_split( playlist, mlt_playlist_count( playlist ) - 1, split );
200 mlt_playlist_get_clip_info( playlist, &info, mlt_playlist_count( playlist ) - 1 );
201 producer = info.cut;
202 properties = mlt_producer_properties( producer );
203 }
204 }
205 else if ( !strcmp( argv[ i ], "-swap" ) )
206 {
207 if ( producer != NULL && !mlt_producer_is_cut( producer ) )
208 mlt_playlist_append( playlist, producer );
209 producer = NULL;
210 if ( mlt_playlist_count( playlist ) >= 2 )
211 {
212 mlt_playlist_clip_info info;
213 mlt_playlist_move( playlist, mlt_playlist_count( playlist ) - 2, mlt_playlist_count( playlist ) - 1 );
214 mlt_playlist_get_clip_info( playlist, &info, mlt_playlist_count( playlist ) - 1 );
215 producer = info.cut;
216 properties = mlt_producer_properties( producer );
217 }
218 }
219 else if ( !strcmp( argv[ i ], "-join" ) )
220 {
221 int clips = atoi( argv[ ++ i ] );
222 if ( producer != NULL && !mlt_producer_is_cut( producer ) )
223 mlt_playlist_append( playlist, producer );
224 producer = NULL;
225 if ( mlt_playlist_count( playlist ) > 0 )
226 {
227 mlt_playlist_clip_info info;
228 mlt_playlist_join( playlist, mlt_playlist_count( playlist ) - clips - 1, clips, 0 );
229 mlt_playlist_get_clip_info( playlist, &info, mlt_playlist_count( playlist ) - 1 );
230 producer = info.cut;
231 properties = mlt_producer_properties( producer );
232 }
233 }
234 else if ( !strcmp( argv[ i ], "-remove" ) )
235 {
236 if ( producer != NULL && !mlt_producer_is_cut( producer ) )
237 mlt_playlist_append( playlist, producer );
238 producer = NULL;
239 if ( mlt_playlist_count( playlist ) > 0 )
240 {
241 mlt_playlist_clip_info info;
242 mlt_playlist_remove( playlist, mlt_playlist_count( playlist ) - 1 );
243 mlt_playlist_get_clip_info( playlist, &info, mlt_playlist_count( playlist ) - 1 );
244 producer = info.cut;
245 properties = mlt_producer_properties( producer );
246 }
247 }
248 else if ( !strcmp( argv[ i ], "-mix" ) )
249 {
250 int length = atoi( argv[ ++ i ] );
251 if ( producer != NULL && !mlt_producer_is_cut( producer ) )
252 mlt_playlist_append( playlist, producer );
253 producer = NULL;
254 if ( mlt_playlist_count( playlist ) >= 2 )
255 {
256 if ( mlt_playlist_mix( playlist, mlt_playlist_count( playlist ) - 2, length, NULL ) == 0 )
257 {
258 mlt_playlist_clip_info info;
259 mlt_playlist_get_clip_info( playlist, &info, mlt_playlist_count( playlist ) - 1 );
260 if ( mlt_properties_get_data( ( mlt_properties )info.producer, "mlt_mix", NULL ) == NULL )
261 mlt_playlist_get_clip_info( playlist, &info, mlt_playlist_count( playlist ) - 2 );
262 mix = ( mlt_tractor )mlt_properties_get_data( ( mlt_properties )info.producer, "mlt_mix", NULL );
263 properties = NULL;
264 }
265 else
266 {
267 fprintf( stderr, "Mix failed?\n" );
268 }
269 }
270 else
271 {
272 fprintf( stderr, "Invalid position for a mix...\n" );
273 }
274 }
275 else if ( !strcmp( argv[ i ], "-mixer" ) )
276 {
277 if ( mix != NULL )
278 {
279 char *id = strdup( argv[ ++ i ] );
280 char *arg = strchr( id, ':' );
281 mlt_field field = mlt_tractor_field( mix );
282 mlt_transition transition = NULL;
283 if ( arg != NULL )
284 *arg ++ = '\0';
285 transition = mlt_factory_transition( id, arg );
286 if ( transition != NULL )
287 {
288 properties = mlt_transition_properties( transition );
289 mlt_properties_inherit( properties, group );
290 mlt_field_plant_transition( field, transition, 0, 1 );
291 mlt_properties_set_position( properties, "in", 0 );
292 mlt_properties_set_position( properties, "out", mlt_producer_get_out( ( mlt_producer )mix ) );
293 mlt_transition_close( transition );
294 }
295 free( id );
296 }
297 else
298 {
299 fprintf( stderr, "Invalid mixer...\n" );
300 }
301 }
302 else if ( !strcmp( argv[ i ], "-filter" ) )
303 {
304 mlt_filter filter = create_filter( field, argv[ ++ i ], track );
305 if ( filter != NULL )
306 {
307 properties = mlt_filter_properties( filter );
308 mlt_properties_inherit( properties, group );
309 }
310 }
311 else if ( !strcmp( argv[ i ], "-transition" ) )
312 {
313 mlt_transition transition = create_transition( field, argv[ ++ i ], track - 1 );
314 if ( transition != NULL )
315 {
316 properties = mlt_transition_properties( transition );
317 mlt_properties_inherit( properties, group );
318 }
319 }
320 else if ( !strcmp( argv[ i ], "-blank" ) )
321 {
322 if ( producer != NULL && !mlt_producer_is_cut( producer ) )
323 mlt_playlist_append( playlist, producer );
324 producer = NULL;
325 mlt_playlist_blank( playlist, atof( argv[ ++ i ] ) );
326 }
327 else if ( !strcmp( argv[ i ], "-track" ) ||
328 !strcmp( argv[ i ], "-hide-track" ) ||
329 !strcmp( argv[ i ], "-hide-video" ) ||
330 !strcmp( argv[ i ], "-hide-audio" ) )
331 {
332 if ( producer != NULL && !mlt_producer_is_cut( producer ) )
333 mlt_playlist_append( playlist, producer );
334 producer = NULL;
335 mlt_multitrack_connect( multitrack, mlt_playlist_producer( playlist ), track ++ );
336 track_service( field, playlist, ( mlt_destructor )mlt_playlist_close );
337 playlist = mlt_playlist_init( );
338 if ( playlist != NULL )
339 {
340 properties = mlt_playlist_properties( playlist );
341 if ( !strcmp( argv[ i ], "-hide-track" ) )
342 mlt_properties_set_int( properties, "hide", 3 );
343 else if ( !strcmp( argv[ i ], "-hide-video" ) )
344 mlt_properties_set_int( properties, "hide", 1 );
345 else if ( !strcmp( argv[ i ], "-hide-audio" ) )
346 mlt_properties_set_int( properties, "hide", 2 );
347 }
348 }
349 else if ( strchr( argv[ i ], '=' ) )
350 {
351 mlt_properties_parse( properties, argv[ i ] );
352 }
353 else if ( argv[ i ][ 0 ] != '-' )
354 {
355 if ( producer != NULL && !mlt_producer_is_cut( producer ) )
356 mlt_playlist_append( playlist, producer );
357 if ( title == NULL )
358 title = argv[ i ];
359 producer = create_producer( field, argv[ i ] );
360 if ( producer != NULL )
361 {
362 properties = mlt_producer_properties( producer );
363 mlt_properties_inherit( properties, group );
364 }
365 }
366 else
367 {
368 if ( !strcmp( argv[ i ], "-serialise" ) )
369 i += 2;
370 else if ( !strcmp( argv[ i ], "-consumer" ) )
371 i += 2;
372
373 while ( argv[ i ] != NULL && strchr( argv[ i ], '=' ) )
374 i ++;
375
376 i --;
377 }
378 }
379
380 // Connect last producer to playlist
381 if ( producer != NULL && !mlt_producer_is_cut( producer ) )
382 mlt_playlist_append( playlist, producer );
383
384 // Track the last playlist too
385 track_service( field, playlist, ( mlt_destructor )mlt_playlist_close );
386
387 // We must have a playlist to connect
388 if ( mlt_playlist_count( playlist ) > 0 )
389 mlt_multitrack_connect( multitrack, mlt_playlist_producer( playlist ), track );
390
391 mlt_producer prod = mlt_tractor_producer( tractor );
392 mlt_properties props = mlt_tractor_properties( tractor );
393 mlt_properties_set_data( props, "group", group, 0, ( mlt_destructor )mlt_properties_close, NULL );
394 mlt_properties_set_position( props, "length", mlt_producer_get_out( mlt_multitrack_producer( multitrack ) ) + 1 );
395 mlt_producer_set_in_and_out( prod, 0, mlt_producer_get_out( mlt_multitrack_producer( multitrack ) ) );
396 mlt_properties_set_double( props, "fps", mlt_producer_get_fps( mlt_multitrack_producer( multitrack ) ) );
397 if ( title != NULL )
398 mlt_properties_set( props, "title", strchr( title, '/' ) ? strrchr( title, '/' ) + 1 : title );
399
400 return prod;
401 }