histogram rework to add sum_frames, update suv/cakewalk autorange icons, add new...
[goodguy/cinelerra.git] / cinelerra-5.1 / cinelerra / convert.C
1 /*
2  * CINELERRA
3  * Copyright (C) 2015 Adam Williams <broadcast at earthling dot net>
4  *
5  * This program is free software; you can redistribute it and/or modify
6  * it under the terms of the GNU General Public License as published by
7  * the Free Software Foundation; either version 2 of the License, or
8  * (at your option) any later version.
9  *
10  * This program is distributed in the hope that it will be useful,
11  * but WITHOUT ANY WARRANTY; without even the implied warranty of
12  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13  * GNU General Public License for more details.
14  *
15  * You should have received a copy of the GNU General Public License
16  * along with this program; if not, write to the Free Software
17  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
18  *
19  */
20
21 #include "assets.h"
22 #include "audiodevice.h"
23 #include "autos.h"
24 #include "bchash.h"
25 #include "bcpot.h"
26 #include "bctimer.h"
27 #include "cache.h"
28 #include "convert.h"
29 #include "edl.h"
30 #include "edlsession.h"
31 #include "file.h"
32 #include "filesystem.h"
33 #include "formattools.h"
34 #include "indexfile.h"
35 #include "language.h"
36 #include "localsession.h"
37 #include "mainerror.h"
38 #include "mainindexes.h"
39 #include "mainprogress.h"
40 #include "mainundo.h"
41 #include "mutex.h"
42 #include "mwindow.h"
43 #include "mwindowgui.h"
44 #include "packagedispatcher.h"
45 #include "packagerenderer.h"
46 #include "panautos.h"
47 #include "panauto.h"
48 #include "preferences.h"
49 #include "render.h"
50 #include "theme.h"
51 #include "tracks.h"
52
53
54 #define WIDTH xS(400)
55 #define HEIGHT yS(360)
56 #define MAX_SCALE 16
57
58 ConvertRender::ConvertRender(MWindow *mwindow)
59  : Thread(1, 0, 0)
60 {
61         this->mwindow = mwindow;
62         suffix = 0;
63         format_asset = 0;
64         progress = 0;
65         progress_timer = new Timer;
66         convert_progress = 0;
67         counter_lock = new Mutex("ConvertDialog::counter_lock");
68         total_rendered = 0;
69         remove_originals = 0;
70         failed = 0;  canceled = 0;
71         result = 0;
72         beep = 0;
73         to_proxy = 0;
74         renderer = 0;
75 }
76
77 ConvertRender::~ConvertRender()
78 {
79         if( running() ) {
80                 canceled = 1;
81                 if( renderer )
82                         renderer->set_result(1);
83                 cancel();
84                 join();
85         }
86         delete renderer;
87         delete [] suffix;
88         delete progress;
89         delete counter_lock;
90         delete progress_timer;
91         delete convert_progress;
92         if( format_asset )
93                 format_asset->remove_user();
94         reset();
95 }
96
97 void ConvertRender::reset()
98 {
99         for( int i=0,n=orig_idxbls.size(); i<n; ++i )
100                 orig_idxbls[i]->remove_user();
101         orig_idxbls.remove_all();
102         for( int i=0,n=orig_copies.size(); i<n; ++i )
103                 orig_copies[i]->remove_user();
104         orig_copies.remove_all();
105         for( int i=0,n=needed_idxbls.size(); i<n; ++i )
106                 needed_idxbls[i]->remove_user();
107         needed_idxbls.remove_all();
108         for( int i=0,n=needed_copies.size(); i<n; ++i )
109                 needed_copies[i]->remove_user();
110         needed_copies.remove_all();
111 }
112
113 void ConvertRender::to_convert_path(char *new_path, Indexable *idxbl)
114 {
115         if( to_proxy ) {
116                 char *bp = idxbl->path, *cp = strrchr(bp, '/');
117                 if( cp ) bp = cp+1;
118                 char filename[BCTEXTLEN], proxy_path[BCTEXTLEN];
119                 strcpy(filename, bp);
120                 File::getenv_path(proxy_path, mwindow->preferences->nested_proxy_path);
121                 sprintf(new_path, "%s/%s", proxy_path, filename);
122         }
123         else
124                 strcpy(new_path, idxbl->path);
125         int n = strlen(suffix);
126         char *ep = new_path + strlen(new_path);
127         char *sfx = strrchr(new_path, '.');
128         if( sfx ) {
129 // insert suffix, path.sfx => path+suffix-sfx.ext
130                 char *bp = ep, *cp = (ep += n);
131                 while( --bp > sfx ) *--cp = *bp;
132                 *--cp = '-';
133         }
134         else {
135 // insert suffix, path => path+suffix.ext
136                 sfx = ep;  ep += n;
137         }
138         for( const char *cp=suffix; --n>=0; ++cp ) *sfx++ = *cp;
139         *ep++ = '.';
140         const char *ext = format_asset->format == FILE_FFMPEG ?
141                         format_asset->fformat :
142                         File::get_tag(format_asset->format);
143         while( *ext ) *ep++ = *ext++;
144         *ep = 0;
145 }
146
147 int ConvertRender::from_convert_path(char *path, Indexable *idxbl)
148 {
149         strcpy(path, idxbl->path);
150         char *ep = path + strlen(path);
151         const char *ext = format_asset->format == FILE_FFMPEG ?
152                         format_asset->fformat :
153                         File::get_tag(format_asset->format);
154         const char *rp = ext + strlen(ext);
155         do {
156                 --rp;  --ep;
157         } while( rp>=ext && ep>=path && *ep == *rp );
158         if( rp >= ext || ep < path && *--ep != '.' ) return 1;   // didnt find .ext
159         int n = strlen(suffix), len = 0;
160         char *lp = ep - n;  // <suffix-chars>+.ext
161         for( ; --lp>=path ; ++len ) {
162                 if( strncmp(lp, suffix, n) ) continue;
163                 if( !len || lp[n] == '-' ) break;
164         }
165         if( lp < path ) return 1;  // didnt find suffix
166         if( *(rp=lp+n) == '-' ) {
167 // remove suffix, path+suffix-sfx.ext >= path.sfx
168                 *lp++ = '.';  ++rp;
169                 while( rp < ep ) *lp++ = *rp++;
170         }
171 // remove suffix, path+suffix.ext => path
172         *lp = 0;
173         return 0;
174 }
175
176 double ConvertRender::get_video_length(Indexable *idxbl)
177 {
178         int64_t video_frames = idxbl->get_video_frames();
179         double frame_rate = idxbl->get_frame_rate();
180         if( video_frames < 0 && mwindow->edl->session->si_useduration )
181                 video_frames = mwindow->edl->session->si_duration * frame_rate;
182         if( video_frames < 0 ) video_frames = 1;
183         return !video_frames ? 0 : video_frames / frame_rate;
184 }
185
186 double ConvertRender::get_audio_length(Indexable *idxbl)
187 {
188         int64_t audio_samples = idxbl->get_audio_samples();
189         return !audio_samples ? 0 :
190                 (double)audio_samples / idxbl->get_sample_rate();
191 }
192
193 double ConvertRender::get_length(Indexable *idxbl)
194 {
195         return bmax(get_video_length(idxbl), get_audio_length(idxbl));
196 }
197
198 int ConvertRender::match_format(Asset *asset)
199 {
200 // close enough
201         return format_asset->audio_data == asset->audio_data &&
202                 format_asset->video_data == asset->video_data ? 1 : 0;
203 }
204
205 EDL *ConvertRender::convert_edl(EDL *edl, Indexable *idxbl)
206 {
207         Asset *copy_asset = edl->assets->get_asset(idxbl->path);
208         if( !copy_asset ) return 0;
209         if( !copy_asset->layers && !copy_asset->channels ) return 0;
210 // make a clip from 1st video track and audio tracks
211         EDL *copy_edl = new EDL(edl);
212         copy_edl->create_objects();
213         copy_edl->set_path(copy_asset->path);
214         char path[BCTEXTLEN];
215         FileSystem fs;  fs.extract_name(path, copy_asset->path);
216         strcpy(copy_edl->local_session->clip_title, path);
217         strcpy(copy_edl->local_session->clip_notes, _("Transcode clip"));
218
219         double video_length = get_video_length(idxbl);
220         double audio_length = get_audio_length(idxbl);
221         copy_edl->session->video_tracks =
222                  video_length > 0 ? 1 : 0;
223         copy_edl->session->audio_tracks =
224                  audio_length > 0 ? copy_asset->channels : 0;
225
226         copy_edl->create_default_tracks();
227         Track *current = copy_edl->session->video_tracks ?
228                 copy_edl->tracks->first : 0;
229         for( int vtrack=0; current; current=NEXT ) {
230                 if( current->data_type != TRACK_VIDEO ) continue;
231                 current->insert_asset(copy_asset, 0, video_length, 0, vtrack);
232                 break;
233         }
234         current = copy_edl->session->audio_tracks ?
235                 copy_edl->tracks->first : 0;
236         for( int atrack=0; current; current=NEXT ) {
237                 if( current->data_type != TRACK_AUDIO ) continue;
238                 current->insert_asset(copy_asset, 0, audio_length, 0, atrack);
239                 Autos *pan_autos = current->automation->autos[AUTOMATION_PAN];
240                 PanAuto *pan_auto = (PanAuto*)pan_autos->get_auto_for_editing(-1);
241                 for( int i=0; i < MAXCHANNELS; ++i ) pan_auto->values[i] = 0;
242                 pan_auto->values[atrack++] = 1;
243                 if( atrack >= MAXCHANNELS ) atrack = 0;
244         }
245         copy_edl->folder_no = AW_MEDIA_FOLDER;
246         return copy_edl;
247 }
248
249 int ConvertRender::add_original(EDL *edl, Indexable *idxbl)
250 {
251         char new_path[BCTEXTLEN];
252 // if idxbl already a convert
253         if( !from_convert_path(new_path, idxbl) ) return 0;
254 // don't convert if not readable
255         if( idxbl->is_asset && access(idxbl->path, R_OK) ) return 0;
256 // found readable unconverted asset
257         to_convert_path(new_path, idxbl);
258 // add to orig_idxbls & orig_copies if it isn't already there.
259         int got_it = 0;
260         for( int i = 0; !got_it && i<orig_copies.size(); ++i )
261                 got_it = !strcmp(orig_copies[i]->path, new_path);
262         if( got_it ) return 0;
263         idxbl->add_user();
264         orig_idxbls.append(idxbl);
265         int needed = 1;
266         Asset *convert = edl->assets->get_asset(new_path);
267         if( !convert ) {
268                 convert = new Asset(new_path);
269                 FileSystem fs;
270                 if( fs.get_size(new_path) > 0 ) {
271 // copy already on disk
272                         int64_t orig_mtime = fs.get_date(idxbl->path);
273                         int64_t convert_mtime = fs.get_date(new_path);
274 // render needed if it is too old
275                         if( orig_mtime < convert_mtime ) {
276                                 File file;
277                                 int ret = file.open_file(mwindow->preferences, convert, 1, 0);
278 // render not needed if can use copy
279                                 if( ret == FILE_OK ) {
280                                         if( match_format(file.asset) ) {
281                                                 mwindow->mainindexes->add_indexable(convert);
282                                                 mwindow->mainindexes->start_build();
283                                                 needed = 0;
284                                         }
285                                         else
286                                                 needed = -1;
287                                 }
288                         }
289                 }
290         }
291         else if( match_format(convert) ) {
292 // dont render if copy already an assets
293                 convert->add_user();
294                 needed = 0;
295         }
296         else
297                 needed = -1;
298         if( needed < 0 ) {
299                 eprintf(_("transcode target file exists but is incorrect format:\n%s\n"
300                           "remove file from disk before transcode to new format.\n"), new_path);
301                 return -1;
302         }
303         orig_copies.append(convert);
304         if( needed ) {
305                 convert->copy_format(format_asset, 0);
306 // new compression parameters
307                 convert->video_data = format_asset->video_data;
308                 if( convert->video_data ) {
309                         convert->layers = 1;
310                         convert->width = idxbl->get_w();
311                         if( convert->width & 1 ) ++convert->width;
312                         convert->actual_width = convert->width;
313                         convert->height = idxbl->get_h();
314                         if( convert->height & 1 ) ++convert->height;
315                         convert->actual_height = convert->height;
316                         convert->frame_rate = mwindow->edl->session->frame_rate;
317                 }
318                 convert->audio_data = format_asset->audio_data;
319                 if( convert->audio_data ) {
320                         convert->sample_rate = mwindow->edl->session->sample_rate;
321                 }
322                 convert->folder_no = AW_MEDIA_FOLDER;
323                 add_needed(idxbl, convert);
324         }
325         return 1;
326 }
327
328 void ConvertRender::add_needed(Indexable *idxbl, Asset *convert)
329 {
330         needed_idxbls.append(idxbl);
331         idxbl->add_user();
332         needed_copies.append(convert);
333         convert->add_user();
334 }
335
336
337 int ConvertRender::is_canceled()
338 {
339         return progress->is_cancelled();
340 }
341
342 int ConvertRender::find_convertable_assets(EDL *edl)
343 {
344         reset();
345         Asset *orig_asset = edl->assets->first;
346         int count = 0;
347         for( ; orig_asset; orig_asset=orig_asset->next ) {
348                 int ret = add_original(edl, orig_asset);
349                 if( ret < 0 ) return -1;
350                 if( ret ) ++count;
351         }
352         return count;
353 }
354
355 void ConvertRender::set_format(Asset *asset, const char *suffix, int to_proxy)
356 {
357         delete [] this->suffix;
358         this->suffix = cstrdup(suffix);
359         if( !format_asset )
360                 format_asset = new Asset();
361         format_asset->copy_from(asset, 0);
362         this->to_proxy = to_proxy;
363 }
364
365 void ConvertRender::start_convert(float beep, int remove_originals)
366 {
367         this->beep = beep;
368         this->remove_originals = remove_originals;
369         start();
370 }
371
372 void ConvertRender::run()
373 {
374         mwindow->stop_brender();
375         result = 0;
376         total_rendered = 0;
377         failed = 0;  canceled = 0;
378
379         progress_timer->update();
380         start_progress();
381
382         for( int i=0; !failed && !canceled && i<needed_copies.size(); ++i )
383                 create_copy(i);
384
385         canceled = progress->is_cancelled();
386 printf(_("convert: failed=%d canceled=%d\n"), failed, canceled);
387         double elapsed_time = progress_timer->get_scaled_difference(1);
388
389         char elapsed[BCSTRLEN], text[BCSTRLEN];
390         Units::totext(elapsed, elapsed_time, TIME_HMS2);
391         printf(_("TranscodeRender::run: done in %s\n"), elapsed);
392         if( canceled )
393                 strcpy(text, _("transcode cancelled"));
394         else if( failed )
395                 strcpy(text, _("transcode failed"));
396         else
397                 sprintf(text, _("transcode %d files, render time %s"),
398                         needed_copies.size(), elapsed);
399 // stop progress bar
400         stop_progress(text);
401
402         if( !failed ) {
403                 mwindow->finish_convert(remove_originals);
404         }
405         else if( !canceled ) {
406                 eprintf(_("Error making transcode."));
407         }
408
409         if( !canceled && beep > 0 ) {
410                 if( failed ) {
411                         mwindow->beep(4000., 0.5, beep);
412                         usleep(250000);
413                         mwindow->beep(1000., 0.5, beep);
414                         usleep(250000);
415                         mwindow->beep(4000., 0.5, beep);
416                 }
417                 else
418                         mwindow->beep(2000., 2.0, beep);
419         }
420         mwindow->restart_brender();
421 }
422
423 void ConvertRender::start_progress()
424 {
425         double total_len= 0;
426         for( int i = 0; i < needed_idxbls.size(); i++ ) {
427                 Indexable *orig_idxbl = needed_idxbls[i];
428                 double length = get_length(orig_idxbl);
429                 total_len += length;
430         }
431         int64_t total_samples = total_len * format_asset->sample_rate;
432         mwindow->gui->lock_window("Render::start_progress");
433         progress = mwindow->mainprogress->
434                 start_progress(_("Transcode files..."), total_samples);
435         mwindow->gui->unlock_window();
436         convert_progress = new ConvertProgress(mwindow, this);
437         convert_progress->start();
438 }
439
440 void ConvertRender::stop_progress(const char *msg)
441 {
442         delete convert_progress;  convert_progress = 0;
443         mwindow->gui->lock_window("ConvertRender::stop_progress");
444         progress->update(0);
445         mwindow->mainprogress->end_progress(progress);
446         progress = 0;
447         mwindow->gui->show_message(msg);
448         mwindow->gui->update_default_message();
449         mwindow->gui->unlock_window();
450 }
451
452
453 ConvertPackageRenderer::ConvertPackageRenderer(ConvertRender *render)
454  : PackageRenderer()
455 {
456         this->render = render;
457 }
458
459 ConvertPackageRenderer::~ConvertPackageRenderer()
460 {
461 }
462
463 int ConvertPackageRenderer::get_master()
464 {
465         return 1;
466 }
467
468 int ConvertPackageRenderer::get_result()
469 {
470         return render->result;
471 }
472
473 void ConvertPackageRenderer::set_result(int value)
474 {
475         if( value )
476                 render->result = value;
477 }
478
479 void ConvertPackageRenderer::set_progress(int64_t value)
480 {
481         render->counter_lock->lock("ConvertPackageRenderer::set_progress");
482 // Increase total rendered for all nodes
483         render->total_rendered += value;
484         render->counter_lock->unlock();
485 }
486
487 int ConvertPackageRenderer::progress_cancelled()
488 {
489         return render->progress && render->progress->is_cancelled();
490 }
491
492 ConvertProgress::ConvertProgress(MWindow *mwindow, ConvertRender *render)
493  : Thread(1, 0, 0)
494 {
495         this->mwindow = mwindow;
496         this->render = render;
497         last_value = 0;
498 }
499
500 ConvertProgress::~ConvertProgress()
501 {
502         cancel();
503         join();
504 }
505
506 void ConvertProgress::run()
507 {
508         Thread::disable_cancel();
509         for( ;; ) {
510                 if( render->total_rendered != last_value ) {
511                         render->progress->update(render->total_rendered);
512                         last_value = render->total_rendered;
513                 }
514
515                 Thread::enable_cancel();
516                 sleep(1);
517                 Thread::disable_cancel();
518         }
519 }
520
521 void ConvertRender::create_copy(int i)
522 {
523         Indexable *orig_idxbl = needed_idxbls[i];
524         Asset *needed_copy = needed_copies[i];
525         EDL *edl = convert_edl(mwindow->edl, orig_idxbl);
526         double length = get_length(orig_idxbl);
527         renderer = new ConvertPackageRenderer(this);
528         renderer->initialize(mwindow, edl, mwindow->preferences, needed_copy);
529         PackageDispatcher dispatcher;
530         dispatcher.create_packages(mwindow, edl, mwindow->preferences,
531                 SINGLE_PASS, needed_copy, 0, length, 0);
532         RenderPackage *package = dispatcher.get_package(0);
533         if( !renderer->render_package(package) ) {
534                 Asset *asset = mwindow->edl->assets->update(needed_copy);
535                 mwindow->mainindexes->add_indexable(asset);
536                 mwindow->mainindexes->start_build();
537         }
538         else
539                 failed = 1;
540         delete renderer;  renderer = 0;
541         edl->remove_user();
542 }
543
544 ConvertWindow::ConvertWindow(MWindow *mwindow, ConvertDialog *dialog, int x, int y)
545  : BC_Window(_(PROGRAM_NAME ": Transcode settings"), x, y, WIDTH, HEIGHT,
546                 -1, -1, 0, 0, 1)
547 {
548         this->mwindow = mwindow;
549         this->dialog = dialog;
550         format_tools = 0;
551 }
552
553 ConvertWindow::~ConvertWindow()
554 {
555         lock_window("ConvertWindow::~ConvertWindow");
556         delete format_tools;
557         unlock_window();
558 }
559
560
561 void ConvertWindow::create_objects()
562 {
563         lock_window("ConvertWindow::create_objects");
564         int margin = mwindow->theme->widget_border;
565         int lmargin = margin + xS(10);
566
567         int x = lmargin;
568         int y = margin + yS(10);
569
570         BC_Title *text;
571         add_subwindow(text = new BC_Title(x, y,
572                 _("Render untagged assets and replace in project")));
573         y += text->get_h() + margin + yS(10);
574         int y1 = y;
575         y += BC_Title::calculate_h(this, _("Tag suffix:")) + margin + yS(10);
576
577         x = lmargin;
578         format_tools = new ConvertFormatTools(mwindow, this, dialog->asset);
579         format_tools->create_objects(x, y, 1, 1, 1, 1, 0, 1, 0, 1, // skip the path
580                 0, 0);
581
582         x = lmargin;
583         add_subwindow(text = new BC_Title(x, y1, _("Tag suffix:")));
584         x = format_tools->format_text->get_x();
585         add_subwindow(suffix_text = new ConvertSuffixText(this, dialog, x, y1));
586         x = lmargin;
587         y += margin + yS(10);
588
589         add_subwindow(remove_originals = new ConvertRemoveOriginals(this, x, y));
590         x = lmargin;
591         y += remove_originals->get_h() + margin;
592         add_subwindow(to_proxy_path = new ConvertToProxyPath(this, x, y));
593         y += to_proxy_path->get_h() + margin;
594
595         add_subwindow(beep_on_done = new ConvertBeepOnDone(this, x, y));
596         x += beep_on_done->get_w() + margin + xS(10);
597         add_subwindow(new BC_Title(x, y+yS(10), _("Beep on done volume")));
598 //      y += beep_on_done->get_h() + margin;
599
600         add_subwindow(new BC_OKButton(this));
601         add_subwindow(new BC_CancelButton(this));
602         show_window(1);
603         unlock_window();
604 }
605
606 ConvertSuffixText::ConvertSuffixText(ConvertWindow *gui,
607                 ConvertDialog *dialog, int x, int y)
608  : BC_TextBox(x, y, 160, 1, dialog->suffix)
609 {
610         this->gui = gui;
611         this->dialog = dialog;
612 }
613
614 ConvertSuffixText::~ConvertSuffixText()
615 {
616 }
617
618 int ConvertSuffixText::handle_event()
619 {
620         strcpy(dialog->suffix, get_text());
621         return 1;
622 }
623
624 ConvertFormatTools::ConvertFormatTools(MWindow *mwindow, ConvertWindow *gui, Asset *asset)
625  : FormatTools(mwindow, gui, asset)
626 {
627         this->gui = gui;
628 }
629
630 void ConvertFormatTools::update_format()
631 {
632         asset->save_defaults(mwindow->defaults, "CONVERT_", 1, 1, 0, 0, 0);
633         FormatTools::update_format();
634 }
635
636
637 ConvertMenuItem::ConvertMenuItem(MWindow *mwindow)
638  : BC_MenuItem(_("Transcode..."),  _("Alt-e"), 'e')
639 {
640         this->mwindow = mwindow;
641         set_alt();
642         dialog = 0;
643 }
644 ConvertMenuItem::~ConvertMenuItem()
645 {
646         delete dialog;
647 }
648
649 void ConvertMenuItem::create_objects()
650 {
651         dialog = new ConvertDialog(mwindow);
652 }
653
654 int ConvertMenuItem::handle_event()
655 {
656         mwindow->gui->unlock_window();
657         dialog->start();
658         mwindow->gui->lock_window("ConvertMenuItem::handle_event");
659         return 1;
660 }
661
662
663 ConvertDialog::ConvertDialog(MWindow *mwindow)
664 {
665         this->mwindow = mwindow;
666         gui = 0;
667         asset = new Asset;
668         strcpy(suffix, ".transcode");
669 // quicker than some, not as good as others
670         asset->format = FILE_FFMPEG;
671         strcpy(asset->fformat, "mp4");
672         strcpy(asset->vcodec, "h264.mp4");
673         asset->ff_video_bitrate = 2560000;
674         asset->video_data = 1;
675         strcpy(asset->acodec, "h264.mp4");
676         asset->ff_audio_bitrate = 256000;
677         asset->audio_data = 1;
678         remove_originals = 1;
679         beep = 0;
680         to_proxy = 0;
681 }
682
683 ConvertDialog::~ConvertDialog()
684 {
685         close_window();
686         asset->remove_user();
687 }
688
689 BC_Window* ConvertDialog::new_gui()
690 {
691         asset->format = FILE_FFMPEG;
692         asset->frame_rate = mwindow->edl->session->frame_rate;
693         asset->sample_rate = mwindow->edl->session->sample_rate;
694         asset->load_defaults(mwindow->defaults, "CONVERT_", 1, 1, 0, 1, 0);
695         remove_originals = mwindow->defaults->get("CONVERT_REMOVE_ORIGINALS", remove_originals);
696         beep = mwindow->defaults->get("CONVERT_BEEP", beep);
697         to_proxy = mwindow->defaults->get("CONVERT_TO_PROXY", to_proxy);
698         mwindow->defaults->get("CONVERT_SUFFIX", suffix);
699         mwindow->gui->lock_window("ConvertDialog::new_gui");
700         int cx, cy;
701         mwindow->gui->get_abs_cursor(cx, cy);
702         gui = new ConvertWindow(mwindow, this, cx - WIDTH/2, cy - HEIGHT/2);
703         gui->create_objects();
704         mwindow->gui->unlock_window();
705         return gui;
706 }
707
708 void ConvertDialog::handle_close_event(int result)
709 {
710         if( result ) return;
711         mwindow->defaults->update("CONVERT_SUFFIX", suffix);
712         mwindow->defaults->update("CONVERT_REMOVE_ORIGINALS", remove_originals);
713         mwindow->defaults->update("CONVERT_BEEP", beep);
714         mwindow->defaults->update("CONVERT_TO_PROXY", to_proxy);
715         asset->save_defaults(mwindow->defaults, "CONVERT_", 1, 1, 0, 1, 0);
716         mwindow->start_convert(asset, suffix, beep, to_proxy, remove_originals);
717 }
718
719
720 ConvertRemoveOriginals::ConvertRemoveOriginals(ConvertWindow *gui, int x, int y)
721  : BC_CheckBox(x, y, gui->dialog->remove_originals, _("Remove originals from project"))
722 {
723         this->gui = gui;
724 }
725
726 ConvertRemoveOriginals::~ConvertRemoveOriginals()
727 {
728 }
729
730 int ConvertRemoveOriginals::handle_event()
731 {
732         gui->dialog->remove_originals = get_value();
733         return 1;
734 }
735
736 ConvertToProxyPath::ConvertToProxyPath(ConvertWindow *gui, int x, int y)
737  : BC_CheckBox(x, y, gui->dialog->to_proxy, _("Into Nested Proxy directory"))
738 {
739         this->gui = gui;
740 }
741
742 ConvertToProxyPath::~ConvertToProxyPath()
743 {
744 }
745
746 int ConvertToProxyPath::handle_event()
747 {
748         gui->dialog->to_proxy = get_value();
749         return 1;
750 }
751
752 ConvertBeepOnDone::ConvertBeepOnDone(ConvertWindow *gui, int x, int y)
753  : BC_FPot(x, y, gui->dialog->beep*100.f, 0.f, 100.f)
754 {
755         this->gui = gui;
756 }
757
758 int ConvertBeepOnDone::handle_event()
759 {
760         gui->dialog->beep = get_value()/100.f;
761         return 1;
762 }
763