4 * Copyright (C) 2008 Adam Williams <broadcast at earthling dot net>
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.
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.
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
18 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
22 #include "bcsignals.h"
24 #include "edlsession.h"
27 #include "keyframes.h"
28 #include "localsession.h"
32 #include "pluginpopup.h"
33 #include "pluginset.h"
34 #include "pluginserver.h"
37 #include "virtualnode.h"
40 Plugin::Plugin(EDL *edl, Track *track, const char *title)
45 strcpy(this->title, title);
46 plugin_type = PLUGIN_NONE;
52 keyframes = new KeyFrames(edl, this);
53 keyframes->create_objects();
57 Plugin::Plugin(EDL *edl, PluginSet *plugin_set, const char *title)
58 : Edit(edl, plugin_set)
60 this->track = plugin_set->track;
61 this->plugin_set = plugin_set;
62 strcpy(this->title, title);
63 plugin_type = PLUGIN_NONE;
69 keyframes = new KeyFrames(edl, this);
70 keyframes->create_objects();
75 while(keyframes->last) delete keyframes->last;
79 Edit& Plugin::operator=(Edit& edit)
85 Plugin& Plugin::operator=(Plugin& edit)
91 int Plugin::operator==(Plugin& that)
93 return identical(&that);
96 int Plugin::operator==(Edit& that)
98 return identical((Plugin*)&that);
101 int Plugin::silence()
103 return plugin_type == PLUGIN_NONE ? 1 : 0;
106 void Plugin::clear_keyframes(int64_t start, int64_t end)
108 keyframes->clear(start, end, 0);
112 void Plugin::init(const char *title,
113 int64_t unit_position, int64_t unit_length, int plugin_type,
114 SharedLocation *shared_location, KeyFrame *default_keyframe)
116 if( title ) strcpy(this->title, title);
117 if( shared_location ) this->shared_location = *shared_location;
118 this->plugin_type = plugin_type;
119 if( default_keyframe )
120 *this->keyframes->default_auto = *default_keyframe;
121 this->keyframes->default_auto->position = unit_position;
122 this->startproject = unit_position;
123 this->length = unit_length;
126 void Plugin::copy_base(Edit *edit)
128 Plugin *plugin = (Plugin*)edit;
130 this->startsource = edit->startsource;
131 this->startproject = edit->startproject;
132 this->length = edit->length;
133 this->orig_id = edit->orig_id;
135 this->plugin_type = plugin->plugin_type;
136 this->in = plugin->in;
137 this->out = plugin->out;
138 this->show = plugin->show;
139 this->on = plugin->on;
140 // dont copy gui_id, it will be a duplicate ref
141 // Should reconfigure this based on where the first track is now.
142 this->shared_location = plugin->shared_location;
143 strcpy(this->title, plugin->title);
146 void Plugin::copy_from(Edit *edit)
149 copy_keyframes((Plugin*)edit);
152 void Plugin::copy_keyframes(Plugin *plugin)
155 keyframes->copy_from(plugin->keyframes);
158 void Plugin::copy_keyframes(int64_t start,
164 // Only 1 default is copied from where the start position is
165 int64_t endproject = startproject + length;
168 start < endproject &&
169 start >= startproject))
170 keyframes->copy(start, end, file, default_only, active_only);
173 void Plugin::synchronize_params(Edit *edit)
175 Plugin *plugin = (Plugin*)edit;
176 this->in = plugin->in;
177 this->out = plugin->out;
178 this->show = plugin->show;
179 this->gui_id = plugin->gui_id;
180 this->on = plugin->on;
181 strcpy(this->title, plugin->title);
182 copy_keyframes(plugin);
185 void Plugin::shift_keyframes(int64_t position)
187 for(KeyFrame *keyframe = (KeyFrame*)keyframes->first;
189 keyframe = (KeyFrame*)keyframe->next)
191 keyframe->position += position;
196 void Plugin::equivalent_output(Edit *edit, int64_t *result)
198 Plugin *plugin = (Plugin*)edit;
199 // End of plugin changed
200 if(startproject + length != plugin->startproject + plugin->length)
202 if(*result < 0 || startproject + length < *result)
203 *result = startproject + length;
206 // Start of plugin changed
207 if( startproject != plugin->startproject || plugin_type != plugin->plugin_type ||
208 on != plugin->on || !(shared_location == plugin->shared_location) ||
209 strcmp(title, plugin->title) ) {
210 if( *result < 0 || startproject < *result )
211 *result = startproject;
215 keyframes->equivalent_output(plugin->keyframes, startproject, result);
218 const char* Plugin::type_to_text(int type)
221 case PLUGIN_STANDALONE: return _("standalone");
222 case PLUGIN_SHAREDPLUGIN: return _("shared plugin");
223 case PLUGIN_SHAREDMODULE: return _("shared module");
228 int Plugin::is_synthesis(int64_t position, int direction, int depth)
231 printf("Plugin::is_synthesis %d: depth range limit, type=%s, title=%s\n",
232 __LINE__, type_to_text(plugin_type), title);
235 switch( plugin_type ) {
236 case PLUGIN_STANDALONE: {
238 printf("Plugin::is_synthesis track not defined\n");
241 PluginServer *plugin_server = MWindow::scan_plugindb(title, track->data_type);
242 return plugin_server->get_synthesis(); }
244 // Dereference real plugin and descend another level
245 case PLUGIN_SHAREDPLUGIN: {
246 int real_module_number = shared_location.module;
247 int real_plugin_number = shared_location.plugin;
248 Track *track = edl->tracks->number(real_module_number);
249 // Get shared plugin from master track
250 Plugin *plugin = track->get_current_plugin(position,
251 real_plugin_number, direction, 0, 0);
254 return plugin->is_synthesis(position, direction, depth+1);
257 // Dereference the real track and descend
258 case PLUGIN_SHAREDMODULE: {
259 int real_module_number = shared_location.module;
260 Track *track = edl->tracks->number(real_module_number);
261 return track->is_synthesis(position, direction, depth+1); }
268 int Plugin::identical(Plugin *that)
271 if(plugin_type != that->plugin_type) return 0;
273 // Test title or location
276 case PLUGIN_STANDALONE:
277 if(strcmp(title, that->title)) return 0;
279 case PLUGIN_SHAREDPLUGIN:
280 if(shared_location.module != that->shared_location.module ||
281 shared_location.plugin != that->shared_location.plugin) return 0;
283 case PLUGIN_SHAREDMODULE:
284 if(shared_location.module != that->shared_location.module) return 0;
288 // Test remaining fields
289 return (this->on == that->on &&
290 ((KeyFrame*)keyframes->default_auto)->identical(
291 ((KeyFrame*)that->keyframes->default_auto)));
294 int Plugin::keyframe_exists(KeyFrame *ptr)
296 for(KeyFrame *current = (KeyFrame*)keyframes->first;
298 current = (KeyFrame*)NEXT)
300 if(current == ptr) return 1;
306 void Plugin::change_plugin(char *title,
307 SharedLocation *shared_location,
310 strcpy(this->title, title);
311 this->shared_location = *shared_location;
312 this->plugin_type = plugin_type;
317 KeyFrame* Plugin::get_prev_keyframe(int64_t position,
320 return keyframes->get_prev_keyframe(position, direction);
323 KeyFrame* Plugin::get_next_keyframe(int64_t position,
328 // This doesn't work for playback because edl->selectionstart doesn't
329 // change during playback at the same rate as PluginClient::source_position.
332 //printf("Plugin::get_next_keyframe position < 0\n");
333 position = track->to_units(edl->local_session->get_selectionstart(1), 0);
336 // Get keyframe after current position
337 for(current = (KeyFrame*)keyframes->first;
339 current = (KeyFrame*)NEXT)
341 if(direction == PLAY_FORWARD && current->position > position) break;
343 if(direction == PLAY_REVERSE && current->position >= position) break;
346 // Nothing after current position
347 if(!current && keyframes->last)
349 current = (KeyFrame*)keyframes->last;
355 current = (KeyFrame*)keyframes->default_auto;
361 KeyFrame* Plugin::get_keyframe()
363 return keyframes->get_keyframe();
366 void Plugin::copy(int64_t start, int64_t end, FileXML *file)
368 int64_t endproject = startproject + length;
370 if((startproject >= start && startproject <= end) || // startproject in range
371 (endproject <= end && endproject >= start) || // endproject in range
372 (startproject <= start && endproject >= end)) // range in project
375 int64_t startproject_in_selection = startproject; // start of edit in selection in project
376 int64_t startsource_in_selection = startsource; // start of source in selection in source
377 //int64_t endsource_in_selection = startsource + length; // end of source in selection
378 int64_t length_in_selection = length; // length of edit in selection
380 if(startproject < start)
381 { // start is after start of edit in project
382 int64_t length_difference = start - startproject;
384 startsource_in_selection += length_difference;
385 startproject_in_selection += length_difference;
386 length_in_selection -= length_difference;
389 // end is before end of edit in project
392 length_in_selection = end - startproject_in_selection;
395 // Plugins don't store silence
396 file->tag.set_title("PLUGIN");
397 // file->tag.set_property("STARTPROJECT", startproject_in_selection - start);
398 file->tag.set_property("LENGTH", length_in_selection);
399 file->tag.set_property("TYPE", plugin_type);
400 file->tag.set_property("TITLE", title);
402 file->append_newline();
405 if(plugin_type == PLUGIN_SHAREDPLUGIN ||
406 plugin_type == PLUGIN_SHAREDMODULE)
408 shared_location.save(file);
415 file->tag.set_title("IN");
417 file->tag.set_title("/IN");
422 file->tag.set_title("OUT");
424 file->tag.set_title("/OUT");
429 file->tag.set_title("SHOW");
431 file->tag.set_title("/SHOW");
436 file->tag.set_title("ON");
438 file->tag.set_title("/ON");
441 file->append_newline();
444 keyframes->copy(start, end, file, 0, 0);
446 file->tag.set_title("/PLUGIN");
448 file->append_newline();
452 void Plugin::load(FileXML *file)
455 int first_keyframe = 1;
458 // Currently show is ignored when loading
461 while(keyframes->last) delete keyframes->last;
464 result = file->read_tag();
466 //printf("Plugin::load 1 %s\n", file->tag.get_title());
469 if(file->tag.title_is("/PLUGIN"))
474 if(file->tag.title_is("SHARED_LOCATION"))
476 shared_location.load(file);
479 if(file->tag.title_is("IN"))
484 if(file->tag.title_is("OUT"))
489 if(file->tag.title_is("SHOW"))
494 if(file->tag.title_is("ON"))
499 if(file->tag.title_is("KEYFRAME"))
504 keyframes->default_auto->load(file);
508 // Override default keyframe
510 KeyFrame *keyframe = (KeyFrame*)keyframes->append(new KeyFrame(edl, keyframes));
511 keyframe->position = file->tag.get_property("POSITION", (int64_t)0);
512 keyframe->load(file);
519 void Plugin::get_shared_location(SharedLocation *result)
521 if(plugin_type == PLUGIN_STANDALONE && plugin_set)
523 result->module = edl->tracks->number_of(track);
524 result->plugin = track->plugin_set.number_of(plugin_set);
528 *result = this->shared_location;
532 Track* Plugin::get_shared_track()
534 return edl->tracks->get_item_number(shared_location.module);
538 void Plugin::calculate_title(char *string, int use_nudge)
540 switch( plugin_type ) {
541 case PLUGIN_STANDALONE:
543 strcpy(string, _(title));
545 case PLUGIN_SHAREDPLUGIN:
546 case PLUGIN_SHAREDMODULE:
547 shared_location.calculate_title(string, edl,
548 startproject, 0, plugin_type, use_nudge);
553 void Plugin::fix_plugin_title(char *title)
555 MWindow::fix_plugin_title(title);
559 void Plugin::paste(FileXML *file)
561 length = file->tag.get_property("LENGTH", (int64_t)0);
564 void Plugin::resample(double old_rate, double new_rate)
566 // Resample keyframes in here
567 keyframes->resample(old_rate, new_rate);
570 void Plugin::shift(int64_t difference)
572 Edit::shift(difference);
574 if(edl->session->autos_follow_edits)
575 shift_keyframes(difference);
578 void Plugin::dump(FILE *fp)
580 fprintf(fp," PLUGIN: type=%d, id %d, orig_id %d, title=\"%s\" on=%d track=%d plugin=%d gui_id=%d\n",
581 plugin_type, id, orig_id, title, on, shared_location.module, shared_location.plugin, gui_id);
582 fprintf(fp," startproject %jd length %jd\n", startproject, length);