3 #include "automation.h"
8 #include "edlsession.h"
12 #include "localsession.h"
15 #include "mainsession.h"
16 #include "pluginserver.h"
17 #include "pluginset.h"
19 #include "trackcanvas.h"
21 #include "trackscroll.h"
22 #include "transition.h"
23 #include "transportque.h"
27 int Tracks::clear(double start, double end, int clear_plugins)
31 for(current_track = first;
33 current_track = current_track->next)
35 if(current_track->record)
37 current_track->clear(start,
49 void Tracks::clear_automation(double selectionstart, double selectionend)
53 for(current_track = first; current_track; current_track = current_track->next)
55 if(current_track->record)
57 current_track->clear_automation(selectionstart,
65 int Tracks::clear_default_keyframe()
67 for(Track *current = first; current; current = NEXT)
70 current->clear_automation(0, 0, 0, 1);
75 int Tracks::clear_handle(double start,
77 double &longest_distance,
84 for(current_track = first; current_track; current_track = current_track->next)
86 if(current_track->record)
88 current_track->clear_handle(start,
93 if(distance > longest_distance) longest_distance = distance;
100 int Tracks::copy_automation(double selectionstart,
106 // called by MWindow::copy_automation for copying automation alone
107 Track* current_track;
109 file->tag.set_title("AUTO_CLIPBOARD");
110 file->tag.set_property("LENGTH", selectionend - selectionstart);
111 file->tag.set_property("FRAMERATE", edl->session->frame_rate);
112 file->tag.set_property("SAMPLERATE", edl->session->sample_rate);
114 file->append_newline();
115 file->append_newline();
117 for(current_track = first;
119 current_track = current_track->next)
121 if(current_track->record)
123 current_track->copy_automation(selectionstart,
131 file->tag.set_title("/AUTO_CLIPBOARD");
133 file->append_newline();
134 file->terminate_string();
138 int Tracks::copy_default_keyframe(FileXML *file)
140 copy_automation(0, 0, file, 1, 0);
144 int Tracks::delete_tracks()
146 int total_deleted = 0;
152 for (Track* current = first;
158 delete_track(current);
164 return total_deleted;
167 void Tracks::move_edits(ArrayList<Edit*> *edits,
170 int edit_labels, // Ignored
171 int edit_plugins, // Ignored
174 //printf("Tracks::move_edits 1\n");
175 for(Track *dest_track = track; dest_track; dest_track = dest_track->next)
177 if(dest_track->record)
179 // Need a local copy of the source edit since the original source edit may
180 // change in the editing operation.
181 Edit *source_edit = 0;
182 Track *source_track = 0;
186 if(dest_track->data_type == TRACK_AUDIO)
188 int current_aedit = 0;
190 while(current_aedit < edits->total &&
191 edits->values[current_aedit]->track->data_type != TRACK_AUDIO)
194 if(current_aedit < edits->total)
196 source_edit = edits->values[current_aedit];
197 source_track = source_edit->track;
198 edits->remove_number(current_aedit);
202 if(dest_track->data_type == TRACK_VIDEO)
204 int current_vedit = 0;
205 while(current_vedit < edits->total &&
206 edits->values[current_vedit]->track->data_type != TRACK_VIDEO)
209 if(current_vedit < edits->total)
211 source_edit = edits->values[current_vedit];
212 source_track = source_edit->track;
213 edits->remove_number(current_vedit);
217 //printf("Tracks::move_edits 2 %s %s %d\n", source_track->title, dest_track->title, source_edit->length);
220 int64_t position_i = source_track->to_units(position, 0);
221 // Source edit changes
222 int64_t source_length = source_edit->length;
223 int64_t source_startproject = source_edit->startproject;
227 // This works like this: CUT edit, INSERT edit at final position, keyframes also follow
228 // FIXME: there should be a GUI way to tell whenever user also wants to move autos or not
231 AutoConf temp_autoconf;
233 temp_autoconf.set_all();
235 source_track->automation->copy(source_edit->startproject,
236 source_edit->startproject + source_edit->length,
240 temp.terminate_string();
242 // Insert new keyframes
243 //printf("Tracks::move_edits 2 %d %p\n", result->startproject, result->asset);
244 source_track->automation->clear(source_edit->startproject,
245 source_edit->startproject + source_edit->length,
248 int64_t position_a = position_i;
249 if (dest_track == source_track)
251 if (position_a > source_edit->startproject)
252 position_a -= source_length;
255 dest_track->automation->paste_silence(position_a,
256 position_a + source_length);
257 while(!temp.read_tag())
258 dest_track->automation->paste(position_a,
266 Edit *dest_edit = dest_track->edits->shift(position_i,
268 Edit *result = dest_track->edits->insert_before(dest_edit,
269 new Edit(edl, dest_track));
270 result->copy_from(source_edit);
271 result->startproject = position_i;
272 result->length = source_length;
275 source_track->edits->clear(source_edit->startproject,
276 source_edit->startproject + source_length);
279 //this is outline for future thinking how it is supposed to be done trough C&P mechanisms
281 source_track->cut(source_edit->startproject,
282 source_edit->startproject + source_edit->length,
285 temp.terminate_string();
287 dest_track->paste_silence(position_a,
288 position_a + source_length,
290 while(!temp.read_tag())
291 dest_track->paste(position_a, // MISSING PIECE OF FUNCTIONALITY
302 // ONLY edit is moved, all other edits stay where they are
304 // Copy edit to temp, delete the edit, insert the edit
305 Edit *temp_edit = new Edit(edl, dest_track);
306 temp_edit->copy_from(source_edit);
307 // we call the edits directly since we do not want to move keyframes or anything else
308 source_track->edits->clear(source_startproject,
309 source_startproject + source_length);
310 source_track->edits->paste_silence(source_startproject,
311 source_startproject + source_length);
313 dest_track->edits->clear(position_i,
314 position_i + source_length);
315 Edit *dest_edit = dest_track->edits->shift(position_i, source_length);
316 Edit *result = dest_track->edits->insert_before(dest_edit,
317 new Edit(edl, dest_track));
318 result->copy_from(temp_edit);
319 result->startproject = position_i;
320 result->length = source_length;
326 source_track->optimize();
327 dest_track->optimize();
333 void Tracks::move_effect(Plugin *plugin,
334 PluginSet *dest_plugin_set,
336 int64_t dest_position)
338 Track *source_track = plugin->track;
341 // Insert on an existing plugin set
342 if(!dest_track && dest_plugin_set)
344 Track *dest_track = dest_plugin_set->track;
347 // Assume this operation never splits a plugin
348 // Shift destination plugins back
349 dest_plugin_set->shift(dest_position, plugin->length);
353 for(current = (Plugin*)dest_plugin_set->first; current; current = (Plugin*)NEXT)
354 if(current->startproject >= dest_position) break;
356 result = (Plugin*)dest_plugin_set->insert_before(current,
357 new Plugin(edl, dest_plugin_set, ""));
360 // Create a new plugin set
364 if(edl->local_session->get_selectionend() >
365 edl->local_session->get_selectionstart())
367 start = edl->local_session->get_selectionstart();
368 length = edl->local_session->get_selectionend() -
372 if(dest_track->get_length() > 0)
375 length = dest_track->get_length();
380 length = dest_track->from_units(plugin->length);
384 result = dest_track->insert_effect("",
385 &plugin->shared_location,
390 plugin->plugin_type);
395 result->copy_from(plugin);
396 result->shift(dest_position - plugin->startproject);
398 // Clear new plugin from old set
399 plugin->plugin_set->clear(plugin->startproject, plugin->startproject + plugin->length);
402 source_track->optimize();
407 int Tracks::concatenate_tracks(int edit_plugins)
409 Track *output_track, *first_output_track, *input_track;
410 int i, data_type = TRACK_AUDIO;
414 IntAuto *play_keyframe = 0;
417 for(i = 0; i < 2; i++)
419 // Get first output track
420 for(output_track = first;
422 output_track = output_track->next)
423 if(output_track->data_type == data_type &&
424 output_track->record) break;
426 first_output_track = output_track;
428 // Get first input track
429 for(input_track = first;
431 input_track = input_track->next)
433 if(input_track->data_type == data_type &&
435 !input_track->record) break;
439 if(output_track && input_track)
441 // Transfer input track to end of output track one at a time
444 output_start = output_track->get_length();
445 output_track->insert_track(input_track,
450 // Get next source and destination
451 for(input_track = input_track->next;
453 input_track = input_track->next)
456 if(input_track->data_type == data_type &&
457 !input_track->record &&
458 input_track->play) break;
461 for(output_track = output_track->next;
463 output_track = output_track->next)
465 if(output_track->data_type == data_type &&
466 output_track->record) break;
471 output_track = first_output_track;
477 if(data_type == TRACK_AUDIO) data_type = TRACK_VIDEO;
483 int Tracks::delete_all_tracks()
485 while(last) delete last;
490 void Tracks::change_modules(int old_location, int new_location, int do_swap)
492 for(Track* current = first ; current; current = current->next)
494 current->change_modules(old_location, new_location, do_swap);
498 void Tracks::change_plugins(SharedLocation &old_location, SharedLocation &new_location, int do_swap)
500 for(Track* current = first ; current; current = current->next)
502 current->change_plugins(old_location, new_location, do_swap);
508 // =========================================== EDL editing
511 int Tracks::copy(double start,
518 if(start == end && !all) return 1;
526 if(current->record || all)
528 current->copy(start, end, file,output_path);
537 int Tracks::move_track_up(Track *track)
539 Track *next_track = track->previous;
540 if(!next_track) next_track = last;
542 change_modules(number_of(track), number_of(next_track), 1);
544 // printf("Tracks::move_track_up 1 %p %p\n", track, next_track);
546 // for(Track *current = first; current && count < 5; current = NEXT, count++)
547 // printf("Tracks::move_track_up %p %p %p\n", current->previous, current, current->next);
548 // printf("Tracks::move_track_up 2\n");
550 swap(track, next_track);
553 // for(Track *current = first; current && count < 5; current = NEXT, count++)
554 // printf("Tracks::move_track_up %p %p %p\n", current->previous, current, current->next);
555 // printf("Tracks::move_track_up 3\n");
560 int Tracks::move_track_down(Track *track)
562 Track *next_track = track->next;
563 if(!next_track) next_track = first;
565 change_modules(number_of(track), number_of(next_track), 1);
566 swap(track, next_track);
571 int Tracks::move_tracks_up()
573 Track *track, *next_track;
580 next_track = track->next;
586 change_modules(number_of(track->previous), number_of(track), 1);
588 swap(track->previous, track);
597 int Tracks::move_tracks_down()
599 Track *track, *previous_track;
604 track = previous_track)
606 previous_track = track->previous;
612 change_modules(number_of(track), number_of(track->next), 1);
614 swap(track, track->next);
625 void Tracks::paste_audio_transition(PluginServer *server)
627 for(Track *current = first; current; current = NEXT)
629 if(current->data_type == TRACK_AUDIO &&
632 int64_t position = current->to_units(
633 edl->local_session->get_selectionstart(), 0);
634 Edit *current_edit = current->edits->editof(position,
639 paste_transition(server, current_edit);
645 void Tracks::paste_automation(double selectionstart,
649 Track* current_atrack = 0;
650 Track* current_vtrack = 0;
653 double frame_rate = edl->session->frame_rate;
654 int64_t sample_rate = edl->session->sample_rate;
655 char string[BCTEXTLEN];
660 result = file->read_tag();
662 !file->tag.title_is("AUTO_CLIPBOARD"));
666 length = file->tag.get_property("LENGTH", 0);
667 frame_rate = file->tag.get_property("FRAMERATE", frame_rate);
668 sample_rate = file->tag.get_property("SAMPLERATE", sample_rate);
673 result = file->read_tag();
677 if(file->tag.title_is("/AUTO_CLIPBOARD"))
682 if(file->tag.title_is("TRACK"))
684 file->tag.get_property("TYPE", string);
686 if(!strcmp(string, "AUDIO"))
688 // Get next audio track
690 current_atrack = first;
692 current_atrack = current_atrack->next;
694 while(current_atrack &&
695 (current_atrack->data_type != TRACK_AUDIO ||
696 !current_atrack->record))
697 current_atrack = current_atrack->next;
702 current_atrack->paste_automation(selectionstart,
712 // Get next video track
714 current_vtrack = first;
716 current_vtrack = current_vtrack->next;
718 while(current_vtrack &&
719 (current_vtrack->data_type != TRACK_VIDEO ||
720 !current_vtrack->record))
721 current_vtrack = current_vtrack->next;
726 //printf("Tracks::paste_automation 1 %s %d\n", current_vtrack->title, current_vtrack->record);
727 current_vtrack->paste_automation(selectionstart,
741 int Tracks::paste_default_keyframe(FileXML *file)
743 paste_automation(0, file, 1);
747 void Tracks::paste_transition(PluginServer *server, Edit *dest_edit)
749 dest_edit->insert_transition(server->title);
752 void Tracks::paste_video_transition(PluginServer *server, int first_track)
754 for(Track *current = first; current; current = NEXT)
756 if(current->data_type == TRACK_VIDEO &&
759 int64_t position = current->to_units(
760 edl->local_session->get_selectionstart(), 0);
761 Edit *current_edit = current->edits->editof(position,
766 paste_transition(server, current_edit);
768 if(first_track) break;
774 int Tracks::paste_silence(double start, double end, int edit_plugins)
776 Track* current_track;
778 for(current_track = first;
780 current_track = current_track->next)
782 if(current_track->record)
784 current_track->paste_silence(start, end, edit_plugins);
792 int Tracks::select_auto(int cursor_x, int cursor_y)
795 for(Track* current = first; current && !result; current = NEXT) { result = current->select_auto(&auto_conf, cursor_x, cursor_y); }
799 int Tracks::move_auto(int cursor_x, int cursor_y, int shift_down)
803 for(Track* current = first; current && !result; current = NEXT)
805 result = current->move_auto(&auto_conf, cursor_x, cursor_y, shift_down);
810 int Tracks::modify_edithandles(double &oldposition,
819 for(current = first; current; current = NEXT)
823 current->modify_edithandles(oldposition,
834 int Tracks::modify_pluginhandles(double &oldposition,
843 for(current = first; current; current = NEXT)
847 current->modify_pluginhandles(oldposition,
860 int Tracks::purge_asset(Asset *asset)
862 Track *current_track;
865 for(current_track = first; current_track; current_track = current_track->next)
867 result += current_track->purge_asset(asset);
872 int Tracks::asset_used(Asset *asset)
874 Track *current_track;
877 for(current_track = first; current_track; current_track = current_track->next)
879 result += current_track->asset_used(asset);
884 int Tracks::scale_time(float rate_scale, int ignore_record, int scale_edits, int scale_autos, int64_t start, int64_t end)
886 Track *current_track;
888 for(current_track = first;
890 current_track = current_track->next)
892 if((current_track->record || ignore_record) &&
893 current_track->data_type == TRACK_VIDEO)
895 current_track->scale_time(rate_scale, scale_edits, scale_autos, start, end);