4 * Copyright (C) 1997-2011 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 "bcdisplayinfo.h"
24 #include "bcsignals.h"
26 #include "condition.h"
30 #include "edlsession.h"
32 #include "filesystem.h"
34 #include "indexable.h"
36 #include "localsession.h"
40 #include "pluginclient.h"
41 #include "pluginserver.h"
42 #include "preferences.h"
44 #include "transportque.inc"
55 PluginClientThread::PluginClientThread(PluginClient *client)
58 this->client = client;
60 init_complete = new Condition(0, "PluginClientThread::init_complete");
63 PluginClientThread::~PluginClientThread()
66 //printf("PluginClientThread::~PluginClientThread %p %d\n", this, __LINE__);
67 delete window; window = 0;
68 //printf("PluginClientThread::~PluginClientThread %p %d\n", this, __LINE__);
72 void PluginClientThread::run()
76 if(client->window_x < 0) client->window_x = info.get_abs_cursor_x();
77 if(client->window_y < 0) client->window_y = info.get_abs_cursor_y();
79 window = (PluginClientWindow*)client->new_window();
82 window->lock_window("PluginClientThread::run");
83 window->create_objects();
84 window->unlock_window();
86 /* Only set it here so tracking doesn't update it until everything is created. */
87 client->thread = this;
88 init_complete->unlock();
90 result = window->run_window();
91 window->lock_window("PluginClientThread::run");
92 //printf("PluginClientThread::run %p %d\n", this, __LINE__);
93 window->hide_window(1);
94 window->unlock_window();
95 window->done_event(result);
96 // Can't save defaults in the destructor because it's not called immediately
98 /* if(client->defaults) */ client->save_defaults_xml();
99 /* This is needed when the GUI is closed from itself */
100 if(result) client->client_side_close();
105 client->thread = this;
106 init_complete->unlock();
110 BC_WindowBase* PluginClientThread::get_window()
115 PluginClient* PluginClientThread::get_client()
125 PluginClientFrame::PluginClientFrame(int data_size,
129 this->data_size = data_size;
131 this->period_n = period_n;
132 this->period_d = period_d;
135 PluginClientFrame::~PluginClientFrame()
144 PluginClientWindow::PluginClientWindow(PluginClient *client,
145 int w, int h, int min_w, int min_h, int allow_resize)
146 : BC_Window(client->gui_string,
147 client->window_x /* - w / 2 */,
148 client->window_y /* - h / 2 */,
149 (int)(w*get_resources()->font_scale+0.5), (int)(h*get_resources()->font_scale+0.5),
150 (int)(min_w*get_resources()->font_scale+0.5), (int)(min_h*get_resources()->font_scale+0.5),
153 this->client = client;
156 PluginClientWindow::PluginClientWindow(const char *title,
157 int x, int y, int w, int h, int min_w, int min_h, int allow_resize)
158 : BC_Window(title, x, y,
159 (int)(w*get_resources()->font_scale+0.5), (int)(h*get_resources()->font_scale+0.5),
160 (int)(min_w*get_resources()->font_scale+0.5), (int)(min_h*get_resources()->font_scale+0.5),
166 PluginClientWindow::~PluginClientWindow()
171 int PluginClientWindow::translation_event()
175 client->window_x = get_x();
176 client->window_y = get_y();
182 int PluginClientWindow::close_event()
184 /* Set result to 1 to indicate a client side close */
193 PluginClient::PluginClient(PluginServer *server)
196 this->server = server;
197 smp = server->preferences->project_smp;
199 update_timer = new Timer;
200 // Virtual functions don't work here.
203 PluginClient::~PluginClient()
211 // Virtual functions don't work here.
212 if(defaults) delete defaults;
213 frame_buffer.remove_all_objects();
217 int PluginClient::reset()
226 realtime_priority = 0;
228 total_in_buffers = 0;
229 total_out_buffers = 0;
233 direction = PLAY_FORWARD;
240 void PluginClient::hide_gui()
242 if(thread && thread->window)
244 //printf("PluginClient::delete_thread %d\n", __LINE__);
245 /* This is needed when the GUI is closed from elsewhere than itself */
246 /* Since we now use autodelete, this is all that has to be done, thread will take care of itself ... */
247 /* Thread join will wait if this was not called from the thread itself or go on if it was */
248 thread->window->lock_window("PluginClient::hide_gui");
249 thread->window->set_done(0);
250 //printf("PluginClient::hide_gui %d thread->window=%p\n", __LINE__, thread->window);
251 thread->window->unlock_window();
255 // For realtime plugins initialize buffers
256 int PluginClient::plugin_init_realtime(int realtime_priority,
257 int total_in_buffers,
261 // Get parameters for all
262 master_gui_on = get_gui_status();
266 // get parameters depending on video or audio
267 init_realtime_parameters();
269 this->realtime_priority = realtime_priority;
270 this->total_in_buffers = this->total_out_buffers = total_in_buffers;
271 this->out_buffer_size = this->in_buffer_size = buffer_size;
275 int PluginClient::plugin_start_loop(int64_t start,
280 //printf("PluginClient::plugin_start_loop %d %ld %ld %ld %d\n",
281 // __LINE__, start, end, buffer_size, total_buffers);
282 this->source_start = start;
283 this->total_len = end - start;
286 this->in_buffer_size = this->out_buffer_size = buffer_size;
287 this->total_in_buffers = this->total_out_buffers = total_buffers;
292 int PluginClient::plugin_process_loop()
294 return process_loop();
297 int PluginClient::plugin_stop_loop()
302 MainProgressBar* PluginClient::start_progress(char *string, int64_t length)
304 return server->start_progress(string, length);
308 // Non realtime parameters
309 int PluginClient::plugin_get_parameters()
311 int result = get_parameters();
312 if(defaults) save_defaults();
316 // ========================= main loop
318 int PluginClient::is_multichannel() { return 0; }
319 int PluginClient::is_synthesis() { return 0; }
320 int PluginClient::is_realtime() { return 0; }
321 int PluginClient::is_fileio() { return 0; }
322 int PluginClient::delete_buffer_ptrs() { return 0; }
323 const char* PluginClient::plugin_title() { return _("Untitled"); }
325 Theme* PluginClient::new_theme() { return 0; }
327 int PluginClient::load_configuration()
332 Theme* PluginClient::get_theme()
334 return server->get_theme();
337 int PluginClient::show_gui()
339 load_configuration();
340 thread = new PluginClientThread(this);
342 thread->init_complete->lock("PluginClient::show_gui");
343 // Must wait before sending any hide_gui
344 if( !thread->window ) return 1;
345 thread->window->init_wait();
349 void PluginClient::raise_window()
351 if(thread && thread->window)
353 thread->window->lock_window("PluginClient::raise_window");
354 thread->window->raise_window();
355 thread->window->flush();
356 thread->window->unlock_window();
360 int PluginClient::set_string()
364 thread->window->lock_window("PluginClient::set_string");
365 thread->window->put_title(gui_string);
366 thread->window->unlock_window();
375 void PluginClient::begin_process_buffer()
377 // Delete all unused GUI frames
378 frame_buffer.remove_all_objects();
382 void PluginClient::end_process_buffer()
384 if(frame_buffer.size())
392 void PluginClient::plugin_update_gui()
397 // Delete unused GUI frames
398 while(frame_buffer.size() > MAX_FRAME_BUFFER)
399 frame_buffer.remove_object_number(0);
403 void PluginClient::update_gui()
407 int PluginClient::get_gui_update_frames()
409 if(frame_buffer.size())
411 PluginClientFrame *frame = frame_buffer.get(0);
412 int total_frames = update_timer->get_difference() *
416 if(total_frames) update_timer->subtract(total_frames *
421 // printf("PluginClient::get_gui_update_frames %d %ld %d %d %d\n",
423 // update_timer->get_difference(),
424 // frame->period_n * 1000 / frame->period_d,
426 // frame_buffer.size());
429 for(int i = 0; i < frame_buffer.size(); i++)
430 if(frame_buffer.get(i)->force) total_frames++;
431 total_frames = MIN(frame_buffer.size(), total_frames);
442 PluginClientFrame* PluginClient::get_gui_frame()
444 if(frame_buffer.size())
446 PluginClientFrame *frame = frame_buffer.get(0);
447 frame_buffer.remove_number(0);
456 void PluginClient::add_gui_frame(PluginClientFrame *frame)
458 frame_buffer.append(frame);
461 void PluginClient::send_render_gui()
463 server->send_render_gui(&frame_buffer);
466 void PluginClient::send_render_gui(void *data)
468 server->send_render_gui(data);
471 void PluginClient::send_render_gui(void *data, int size)
473 server->send_render_gui(data, size);
476 void PluginClient::plugin_render_gui(void *data, int size)
478 render_gui(data, size);
482 void PluginClient::plugin_render_gui(void *data)
487 void PluginClient::render_gui(void *data)
491 thread->get_window()->lock_window("PluginClient::render_gui");
493 // Set all previous frames to draw immediately
494 for(int i = 0; i < frame_buffer.size(); i++)
495 frame_buffer.get(i)->force = 1;
497 ArrayList<PluginClientFrame*> *src =
498 (ArrayList<PluginClientFrame*>*)data;
500 // Shift GUI data to GUI client
503 this->frame_buffer.append(src->get(0));
504 src->remove_number(0);
507 // Start the timer for the current buffer
508 update_timer->update();
509 thread->get_window()->unlock_window();
513 void PluginClient::render_gui(void *data, int size)
515 printf("PluginClient::render_gui %d\n", __LINE__);
525 int PluginClient::is_audio() { return 0; }
526 int PluginClient::is_video() { return 0; }
527 int PluginClient::is_theme() { return 0; }
528 int PluginClient::uses_gui() { return 1; }
529 int PluginClient::is_transition() { return 0; }
530 int PluginClient::load_defaults()
532 // printf("PluginClient::load_defaults undefined in %s.\n", plugin_title());
536 int PluginClient::save_defaults()
539 // printf("PluginClient::save_defaults undefined in %s.\n", plugin_title());
543 void PluginClient::load_defaults_xml()
545 char path[BCTEXTLEN];
546 server->get_defaults_path(path);
548 fs.complete_path(path);
550 //printf("PluginClient::load_defaults_xml %d %s\n", __LINE__, path);
552 KeyFrame temp_keyframe;
553 FILE *fp = fopen(path, "r");
555 struct stat st; int fd = fileno(fp);
556 int64_t sz = !fstat(fd, &st) ? st.st_size : BCTEXTLEN;
557 char *data = temp_keyframe.get_data(sz+1);
558 int data_size = fread(data, 1, sz, fp);
559 if( data_size < 0 ) data_size = 0;
560 if( data_size > 0 ) {
562 temp_keyframe.xbuf->oseek(data_size);
563 // Get window extents
565 for( int state=0; i<(data_size-8) && state>=0; ++i ) {
566 if( !data[i] || data[i] == '<' ) break;
567 if( !isdigit(data[i]) ) continue;
569 window_x = atoi(data + i);
573 window_y = atoi(data + i);
576 while( i<data_size && isdigit(data[i]) ) ++i;
578 temp_keyframe.xbuf->iseek(i);
579 read_data(&temp_keyframe);
585 //printf("PluginClient::load_defaults_xml %d %s\n", __LINE__, path);
588 void PluginClient::save_defaults_xml()
590 char path[BCTEXTLEN];
591 server->get_defaults_path(path);
593 fs.complete_path(path);
596 KeyFrame temp_keyframe;
597 save_data(&temp_keyframe);
599 const char *data = temp_keyframe.get_data();
600 int len = strlen(data);
601 FILE *fp = fopen(path, "w");
604 fprintf(fp, "%d\n%d\n", window_x, window_y);
605 if( len > 0 && !fwrite(data, len, 1, fp) ) {
606 fprintf(stderr, "PluginClient::save_defaults_xml %d \"%s\" %d bytes: %s\n",
607 __LINE__, path, len, strerror(errno));
615 int PluginClient::is_defaults()
617 return using_defaults;
620 BC_Hash* PluginClient::get_defaults()
624 PluginClientThread* PluginClient::get_thread()
629 BC_WindowBase* PluginClient::new_window()
631 printf("PluginClient::new_window undefined in %s.\n", plugin_title());
634 int PluginClient::get_parameters() { return 0; }
635 int PluginClient::get_samplerate() { return get_project_samplerate(); }
636 double PluginClient::get_framerate() { return get_project_framerate(); }
637 int PluginClient::init_realtime_parameters() { return 0; }
638 int PluginClient::delete_nonrealtime_parameters() { return 0; }
639 int PluginClient::start_loop() { return 0; };
640 int PluginClient::process_loop() { return 0; };
641 int PluginClient::stop_loop() { return 0; };
643 void PluginClient::set_interactive()
648 int64_t PluginClient::get_in_buffers(int64_t recommended_size)
650 return recommended_size;
653 int64_t PluginClient::get_out_buffers(int64_t recommended_size)
655 return recommended_size;
658 int PluginClient::get_gui_status()
660 return server->get_gui_status();
663 // close event from client side
664 void PluginClient::client_side_close()
666 // Last command executed
667 server->client_side_close();
670 int PluginClient::stop_gui_client()
672 if(!client_gui_on) return 0;
677 int PluginClient::get_project_samplerate()
679 return server->get_project_samplerate();
682 double PluginClient::get_project_framerate()
684 return server->get_project_framerate();
687 const char *PluginClient::get_source_path()
689 int64_t source_position = server->plugin->startproject;
690 Edit *edit = server->plugin->track->edits->editof(source_position,PLAY_FORWARD,0);
691 Indexable *indexable = edit ? edit->get_source() : 0;
692 return indexable ? indexable->path : 0;
696 void PluginClient::update_display_title()
698 server->generate_display_title(gui_string);
702 char* PluginClient::get_gui_string()
708 char* PluginClient::get_path()
713 char* PluginClient::get_plugin_dir()
715 return server->preferences->plugin_dir;
718 int PluginClient::set_string_client(char *string)
720 strcpy(gui_string, string);
726 int PluginClient::get_interpolation_type()
728 return server->get_interpolation_type();
732 float PluginClient::get_red()
734 EDL *edl = server->mwindow ? server->mwindow->edl : server->edl;
735 return !edl ? 0 : edl->local_session->use_max ?
736 edl->local_session->red_max :
737 edl->local_session->red;
740 float PluginClient::get_green()
742 EDL *edl = server->mwindow ? server->mwindow->edl : server->edl;
743 return !edl ? 0 : edl->local_session->use_max ?
744 edl->local_session->green_max :
745 edl->local_session->green;
748 float PluginClient::get_blue()
750 EDL *edl = server->mwindow ? server->mwindow->edl : server->edl;
751 return !edl ? 0 : edl->local_session->use_max ?
752 edl->local_session->blue_max :
753 edl->local_session->blue;
757 int64_t PluginClient::get_source_position()
759 return source_position;
762 int64_t PluginClient::get_source_start()
767 int64_t PluginClient::get_total_len()
772 int PluginClient::get_direction()
778 int64_t PluginClient::local_to_edl(int64_t position)
783 int64_t PluginClient::edl_to_local(int64_t position)
788 int PluginClient::get_use_opengl()
790 return server->get_use_opengl();
793 int PluginClient::get_total_buffers()
795 return total_in_buffers;
798 int PluginClient::get_buffer_size()
800 return in_buffer_size;
803 int PluginClient::get_project_smp()
805 //printf("PluginClient::get_project_smp %d %d\n", __LINE__, smp);
809 const char* PluginClient::get_defaultdir()
811 return File::get_plugin_path();
815 int PluginClient::send_hide_gui()
817 // Stop the GUI server and delete GUI messages
822 int PluginClient::send_configure_change()
825 server->mwindow->undo->update_undo_before(_("tweek"), this);
826 #ifdef USE_KEYFRAME_SPANNING
828 save_data(&keyframe);
829 server->apply_keyframe(&keyframe);
831 KeyFrame* keyframe = server->get_keyframe();
832 // Call save routine in plugin
836 server->mwindow->undo->update_undo_after(_("tweek"), LOAD_AUTOMATION);
837 server->sync_parameters();
842 KeyFrame* PluginClient::get_prev_keyframe(int64_t position, int is_local)
844 if(is_local) position = local_to_edl(position);
845 return server->get_prev_keyframe(position);
848 KeyFrame* PluginClient::get_next_keyframe(int64_t position, int is_local)
850 if(is_local) position = local_to_edl(position);
851 return server->get_next_keyframe(position);
854 void PluginClient::get_camera(float *x, float *y, float *z, int64_t position)
856 server->get_camera(x, y, z, position, direction);
859 void PluginClient::get_projector(float *x, float *y, float *z, int64_t position)
861 server->get_projector(x, y, z, position, direction);
865 EDLSession* PluginClient::get_edlsession()
868 return server->edl->session;
872 int PluginClient::gui_open()
874 return server->gui_open();