4 * Copyright (C) 2010-2013 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
23 #include "bcsignals.h"
27 #include "filesystem.h"
28 #include "formattools.h"
34 #include "maxchannels.h"
36 #include "preferences.h"
38 #include "videodevice.inc"
44 FormatTools::FormatTools(MWindow *mwindow,
45 BC_WindowBase *window,
48 this->mwindow = mwindow;
49 this->window = window;
74 prompt_audio_channels = 0;
76 prompt_video_compression = 0;
78 locked_compressor = 0;
82 FormatTools::~FormatTools()
93 if(aparams_button) delete aparams_button;
95 if(vparams_button) delete vparams_button;
97 if(aparams_thread) delete aparams_thread;
99 if(vparams_thread) delete vparams_thread;
103 void FormatTools::create_objects(
104 int &init_x, int &init_y,
105 int do_audio, int do_video, // Include support for audio, video
106 int prompt_audio, int prompt_video, // Include checkbox for audio, video
107 int prompt_audio_channels,
108 int prompt_video_compression,
109 const char *locked_compressor,
113 int horizontal_layout)
119 int margin = mwindow->theme->widget_border;
121 this->locked_compressor = locked_compressor;
122 this->recording = recording;
123 this->use_brender = brender;
124 this->do_audio = do_audio;
125 this->do_video = do_video;
126 this->prompt_audio = prompt_audio;
127 this->prompt_audio_channels = prompt_audio_channels;
128 this->prompt_video = prompt_video;
129 this->prompt_video_compression = prompt_video_compression;
130 this->file_per_label = file_per_label;
133 if(asset->format == FILE_UNKNOWN)
134 asset->format = FILE_FFMPEG;
135 //printf("FormatTools::create_objects 1\n");
140 window->add_subwindow(path_textbox = new FormatPathText(px, y, this));
141 px += path_textbox->get_w() + 5;
142 path_recent = new BC_RecentList("PATH", mwindow->defaults,
143 path_textbox, 10, px, y, xS(300), yS(100));
144 window->add_subwindow(path_recent);
145 path_recent->load_items(File::formattostr(asset->format));
146 px += path_recent->get_w();
147 window->add_subwindow(path_button = new BrowseButton(
148 mwindow->theme, window, path_textbox, px, y, asset->path,
149 _("Output to file"), _("Select a file to write to:"), 0));
153 y += path_textbox->get_h() + ys10;
162 window->add_subwindow(format_title = new BC_Title(x, y, _("File Format:")));
163 x += format_title->get_w() + margin;
164 window->add_subwindow(format_text = new BC_TextBox(x, y, xS(160), 1,
165 File::formattostr(asset->format)));
166 x += format_text->get_w() + margin;
167 //printf("FormatTools::create_objects %d %p\n", __LINE__, window);
168 window->add_subwindow(format_button = new FormatFormat(x, y, this));
169 format_button->create_objects();
170 x += format_button->get_w() + 5;
171 window->add_subwindow(ffmpeg_type = new FFMpegType(x, y, xS(70), 1, asset->fformat));
172 FFMPEG::set_asset_format(asset, mwindow->edl, asset->fformat);
173 x += ffmpeg_type->get_w();
174 window->add_subwindow(format_ffmpeg = new FormatFFMPEG(x, y, this));
175 format_ffmpeg->create_objects();
177 y += format_button->get_h() + ys10;
179 window->add_subwindow(audio_title = new BC_Title(x, y, _("Audio:"), LARGEFONT,
180 BC_WindowBase::get_resources()->audiovideo_color));
181 x += audio_title->get_w() + margin;
182 window->add_subwindow(aparams_button = new FormatAParams(mwindow, this, x, y));
183 x += aparams_button->get_w() + margin;
185 window->add_subwindow(audio_switch = new FormatAudio(x, y, this, asset->audio_data));
189 y += aparams_button->get_h() + ys10;
191 //printf("FormatTools::create_objects 6\n");
192 aparams_thread = new FormatAThread(this);
195 //printf("FormatTools::create_objects 7\n");
197 if( horizontal_layout && do_audio ) {
202 //printf("FormatTools::create_objects 8\n");
203 window->add_subwindow(video_title = new BC_Title(x, y, _("Video:"), LARGEFONT,
204 BC_WindowBase::get_resources()->audiovideo_color));
205 x += video_title->get_w() + margin;
206 if(prompt_video_compression) {
207 window->add_subwindow(vparams_button = new FormatVParams(mwindow, this, x, y));
208 x += vparams_button->get_w() + margin;
211 //printf("FormatTools::create_objects 9\n");
213 window->add_subwindow(video_switch = new FormatVideo(x, y, this, asset->video_data));
214 y += video_switch->get_h();
217 y += vparams_button->get_h();
220 //printf("FormatTools::create_objects 10\n");
222 vparams_thread = new FormatVThread(this);
225 //printf("FormatTools::create_objects 11\n");
228 if( file_per_label ) {
229 labeled_files = new FormatFilePerLabel(this, x, y, file_per_label);
230 window->add_subwindow(labeled_files);
231 y += labeled_files->get_h() + ys10;
234 //printf("FormatTools::create_objects 12\n");
240 void FormatTools::update_driver(int driver)
242 this->video_driver = driver;
244 locked_compressor = 0;
248 case VIDEO4LINUX2MPEG:
249 // Just give the user information about how the stream is going to be
250 // stored but don't change the asset.
251 // Want to be able to revert to user settings.
252 if(asset->format == FILE_MPEG) break;
253 asset->format = FILE_MPEG;
254 format_text->update(File::formattostr(asset->format));
255 asset->audio_data = 1;
256 asset->video_data = 1;
257 audio_switch->update(1);
258 video_switch->update(1);
261 case CAPTURE_IEC61883:
262 case CAPTURE_FIREWIRE:
263 case VIDEO4LINUX2JPEG:
264 case CAPTURE_JPEG_WEBCAM:
265 asset->format = FILE_FFMPEG;
266 format_text->update(File::formattostr(asset->format));
270 case CAPTURE_IEC61883:
271 case CAPTURE_FIREWIRE:
272 locked_compressor = (char*)CODEC_TAG_DVSD;
275 case VIDEO4LINUX2JPEG:
276 locked_compressor = (char*)CODEC_TAG_MJPEG;
279 case CAPTURE_JPEG_WEBCAM:
280 locked_compressor = (char*)CODEC_TAG_JPEG;
283 if( locked_compressor )
284 strcpy(asset->vcodec, locked_compressor);
286 audio_switch->update(asset->audio_data);
287 video_switch->update(asset->video_data);
291 format_text->update(File::formattostr(asset->format));
292 audio_switch->update(asset->audio_data);
293 video_switch->update(asset->video_data);
296 close_format_windows();
300 void FormatTools::update_format()
302 if( do_audio && prompt_audio && audio_switch ) {
303 audio_switch->update(asset->audio_data);
304 if( File::renders_audio(asset) )
305 audio_switch->enable();
307 audio_switch->disable();
309 if( do_video && prompt_video && video_switch ) {
310 video_switch->update(asset->video_data);
311 if( File::renders_video(asset) )
312 video_switch->enable();
314 video_switch->disable();
316 if( asset->format == FILE_FFMPEG ) {
318 format_ffmpeg->show();
322 format_ffmpeg->hide();
326 int FormatTools::handle_event()
331 Asset* FormatTools::get_asset()
336 void FormatTools::update_extension()
338 const char *extension = File::get_tag(asset->format);
339 // split multiple extensions
340 ArrayList<const char*> extensions;
341 int len = !extension ? -1 : strlen(extension);
342 const char *extension_ptr = extension;
343 for(int i = 0; i <= len; i++)
345 if(extension[i] == '/' || extension[i] == 0)
347 extensions.append(extension_ptr);
348 extension_ptr = extension + i + 1;
352 if(extensions.size())
354 char *ptr = strrchr(asset->path, '.');
357 ptr = asset->path + strlen(asset->path);
362 // test for equivalent extension
363 int need_extension = 1;
364 //int extension_len = 0;
365 for(int i = 0; i < extensions.size() && need_extension; i++)
368 extension_ptr = extensions.get(i);
371 while(*ptr1 != 0 && *extension_ptr != 0 && *extension_ptr != '/')
373 if(tolower(*ptr1) != tolower(*extension_ptr))
382 if( (!*ptr1 && (*extension_ptr && *extension_ptr != '/')) ||
383 (*ptr1 && (!*extension_ptr || *extension_ptr == '/')) )
387 //printf("FormatTools::update_extension %d %d\n", __LINE__, need_extension);
392 // change "qt" to "mov" since ffmpeg does not know qt
393 extension_ptr = asset->format != FILE_FFMPEG ? extensions.get(0) :
394 !strcmp(asset->fformat, "qt" ) ||
395 !strcmp(asset->fformat, "pro" ) ? "mov" : asset->fformat ;
396 while(*extension_ptr != 0 && *extension_ptr != '/')
397 *ptr1++ = *extension_ptr++;
401 int character1 = ptr - asset->path;
402 int character2 = strlen(asset->path);
403 // *(asset->path + character2) = 0;
406 path_textbox->update(asset->path);
407 path_textbox->set_selection(character1, character2, character2);
412 void FormatTools::update(Asset *asset, int *file_per_label)
415 this->file_per_label = file_per_label;
416 if( file_per_label ) labeled_files->update(file_per_label);
417 if( path_textbox ) path_textbox->update(asset->path);
418 format_text->update(File::formattostr(asset->format));
420 close_format_windows();
423 void FormatTools::close_format_windows()
425 // This is done in ~file
426 if( aparams_thread ) {
427 if( aparams_thread->running() )
428 aparams_thread->file->close_window();
429 aparams_thread->join();
431 if( vparams_thread ) {
432 if( vparams_thread->running() )
433 vparams_thread->file->close_window();
434 vparams_thread->join();
438 int FormatTools::get_w()
440 return asset->format != FILE_FFMPEG ? w :
441 format_ffmpeg->get_x() + format_ffmpeg->get_w();
444 void FormatTools::set_w(int w)
449 void FormatTools::reposition_window(int &init_x, int &init_y)
451 int xs10 = xS(10), xs80 = xS(80);
459 path_textbox->reposition_window(px, y);
460 px += path_textbox->get_w() + xS(5);
461 path_recent->reposition_window(px, y);
462 px += path_recent->get_w() + xS(8);
463 path_button->reposition_window(px, y);
464 y += path_textbox->get_h() + ys10;
467 format_title->reposition_window(x, y);
469 format_text->reposition_window(x, y);
470 x += format_text->get_w();
471 format_button->reposition_window(x, y);
472 x += format_button->get_w() + 5;
473 ffmpeg_type->reposition_window(x, y);
474 x += ffmpeg_type->get_w();
475 format_ffmpeg->reposition_window(x, y);
478 y += format_button->get_h() + ys10;
482 audio_title->reposition_window(x, y);
484 aparams_button->reposition_window(x, y);
485 x += aparams_button->get_w() + xs10;
486 if(prompt_audio) audio_switch->reposition_window(x, y);
489 y += aparams_button->get_h() + ys10;
495 video_title->reposition_window(x, y);
497 if(prompt_video_compression)
499 vparams_button->reposition_window(x, y);
500 x += vparams_button->get_w() + xs10;
505 video_switch->reposition_window(x, y);
506 y += video_switch->get_h();
510 y += vparams_button->get_h();
517 if( file_per_label ) {
518 labeled_files->reposition_window(x, y);
519 y += labeled_files->get_h() + ys10;
526 int FormatTools::set_audio_options()
528 // if(video_driver == CAPTURE_DVB)
533 if(!aparams_thread->running())
535 aparams_thread->start();
539 aparams_thread->file->raise_window();
544 int FormatTools::set_video_options()
546 // if(video_driver == CAPTURE_DVB)
551 if(!vparams_thread->running())
553 vparams_thread->start();
557 vparams_thread->file->raise_window();
567 FormatAParams::FormatAParams(MWindow *mwindow, FormatTools *format, int x, int y)
568 : BC_Button(x, y, mwindow->theme->get_image_set("wrench"))
570 this->format = format;
571 set_tooltip(_("Configure audio compression"));
574 FormatAParams::~FormatAParams()
578 int FormatAParams::handle_event()
580 format->set_audio_options();
581 format->handle_event();
589 FormatVParams::FormatVParams(MWindow *mwindow, FormatTools *format, int x, int y)
590 : BC_Button(x, y, mwindow->theme->get_image_set("wrench"))
592 this->format = format;
593 set_tooltip(_("Configure video compression"));
596 FormatVParams::~FormatVParams()
600 int FormatVParams::handle_event()
602 format->set_video_options();
603 format->handle_event();
611 FormatAThread::FormatAThread(FormatTools *format)
614 this->format = format;
619 FormatAThread::~FormatAThread()
621 delete file; file = 0;
625 void FormatAThread::start()
633 void FormatAThread::run()
635 file->get_options(format, 1, 0);
641 FormatVThread::FormatVThread(FormatTools *format)
644 this->format = format;
649 FormatVThread::~FormatVThread()
651 delete file; file = 0;
655 void FormatVThread::start()
662 void FormatVThread::run()
664 file->get_options(format, 0, 1);
671 FormatPathText::FormatPathText(int x, int y, FormatTools *format)
672 : BC_TextBox(x, y, format->w - x -
673 2*format->mwindow->theme->get_image_set("wrench")[0]->get_w() - xS(20), 1,
676 this->format = format;
679 FormatPathText::~FormatPathText()
682 int FormatPathText::handle_event()
684 calculate_suggestions();
685 strcpy(format->asset->path, get_text());
686 format->handle_event();
693 FormatAudio::FormatAudio(int x, int y, FormatTools *format, int default_)
697 (char*)(format->recording ? _("Record audio tracks") : _("Render audio tracks")))
699 this->format = format;
702 FormatAudio::~FormatAudio() {}
703 int FormatAudio::handle_event()
705 format->asset->audio_data = get_value();
706 format->handle_event();
711 FormatVideo::FormatVideo(int x, int y, FormatTools *format, int default_)
715 (char*)(format->recording ? _("Record video tracks") : _("Render video tracks")))
717 this->format = format;
720 FormatVideo::~FormatVideo() {}
721 int FormatVideo::handle_event()
723 format->asset->video_data = get_value();
724 format->handle_event();
731 FormatFormat::FormatFormat(int x, int y, FormatTools *format)
732 : FormatPopup(x, y, format->do_audio, format->do_video, format->use_brender)
734 this->format = format;
737 FormatFormat::~FormatFormat()
741 int FormatFormat::handle_event()
743 BC_ListBoxItem *selection = get_selection(0, 0);
745 int new_format = File::strtoformat(get_selection(0, 0)->get_text());
746 // if(new_format != format->asset->format)
748 Asset *asset = format->asset;
749 asset->format = new_format;
750 asset->audio_data = File::renders_audio(asset);
751 asset->video_data = File::renders_video(asset);
752 asset->ff_audio_options[0] = 0;
753 asset->ff_video_options[0] = 0;
754 asset->ff_format_options[0] = 0;
755 format->format_text->update(selection->get_text());
756 if( !format->use_brender )
757 format->update_extension();
758 format->close_format_windows();
759 if (format->path_recent) format->path_recent->
760 load_items(File::formattostr(format->asset->format));
761 format->update_format();
763 format->handle_event();
769 FormatFFMPEG::FormatFFMPEG(int x, int y, FormatTools *format)
772 this->format = format;
775 FormatFFMPEG::~FormatFFMPEG()
779 int FormatFFMPEG::handle_event()
781 BC_ListBoxItem *selection = get_selection(0, 0);
783 const char *text = get_selection(0, 0)->get_text();
784 format->ffmpeg_type->update(text);
785 // forces options load defaults
786 format->asset->ff_audio_options[0] = 0;
787 format->asset->ff_video_options[0] = 0;
788 format->asset->ff_format_options[0] = 0;
789 FFMPEG::set_asset_format(format->asset, format->mwindow->edl, text);
790 format->update_extension();
791 format->close_format_windows();
792 format->update_format();
793 format->handle_event();
799 FormatFilePerLabel::FormatFilePerLabel(FormatTools *format,
800 int x, int y, int *output)
801 : BC_CheckBox(x, y, *output, _("Create new file at each label"))
803 this->format = format;
804 this->output = output;
807 FormatFilePerLabel::~FormatFilePerLabel()
811 int FormatFilePerLabel::handle_event()
813 *output = get_value();
814 format->handle_event();
818 void FormatFilePerLabel::update(int *output)
820 this->output = output;
821 set_value(*output ? 1 : 0);