merge hv v6, rework trace methods
[goodguy/history.git] / cinelerra-5.1 / cinelerra / bdcreate.C
index f74538037cd3fcd3c0f8f24cc362f0ff1649ee8b..6a583db6474ab41157e34974ced85d123b81aabd 100644 (file)
@@ -1,10 +1,11 @@
 #include "asset.h"
 #include "bdcreate.h"
+#include "clip.h"
 #include "edl.h"
 #include "edit.h"
 #include "edits.h"
 #include "edlsession.h"
-#include "file.inc"
+#include "file.h"
 #include "filexml.h"
 #include "keyframe.h"
 #include "labels.h"
@@ -14,6 +15,7 @@
 #include "mwindowgui.h"
 #include "plugin.h"
 #include "pluginset.h"
+#include "rescale.h"
 #include "track.h"
 #include "tracks.h"
 
 
 // BD Creation
 
+#define BD_1920x1080_2997i     0
+#define BD_1920x1080_2500i     1
+#define BD_1920x1080_2400p     2
+#define BD_1920x1080_23976p    3
+#define BD_1280x720_5994p      4
+#define BD_1280x720_5000p      5
+#define BD_1280x720_23976p     6
+#define BD_1280x720_2400p      7
+#define BD_720x576_2500i       8
+#define BD_720x480_2997i       9
+
+static struct bd_format {
+       const char *name;
+       int w, h;
+       double framerate;
+       int interlaced, wide;
+} bd_formats[] = {
+       { "1920x1080 29.97i",   1920,1080, 29.97,  1, 1 },
+       { "1920x1080 25i",      1920,1080, 25.,    1, 1 },
+       { "1920x1080 24p",      1920,1080, 24.,    0, 1 },
+       { "1920x1080 23.976p",  1920,1080, 23.976, 0, 1 },
+       { "1280x720  59.94p",   1280,720,  59.94,  0, 1 },
+       { "1280x720  50p",      1280,720,  50.,    0, 1 },
+       { "1280x720  23.976p",  1280,720,  23.976, 0, 1 },
+       { "1280x720  24p",      1280,720,  24.,    0, 1 },
+       { "720x576   25i",       720,576,  25.,    1, 0 },
+       { "720x480   29.97i",    720,480,  29.97,  1, 0 },
+};
+
 const int64_t CreateBD_Thread::BD_SIZE = 25000000000;
 const int CreateBD_Thread::BD_STREAMS = 1;
 const int CreateBD_Thread::BD_WIDTH = 1920;
 const int CreateBD_Thread::BD_HEIGHT = 1080;
-const double CreateBD_Thread::BD_ASPECT_WIDTH = 4.;
-const double CreateBD_Thread::BD_ASPECT_HEIGHT = 3.;
 const double CreateBD_Thread::BD_WIDE_ASPECT_WIDTH = 16.;
 const double CreateBD_Thread::BD_WIDE_ASPECT_HEIGHT = 9.;
+const double CreateBD_Thread::BD_ASPECT_WIDTH = 4.;
+const double CreateBD_Thread::BD_ASPECT_HEIGHT = 3.;
 const double CreateBD_Thread::BD_FRAMERATE = 24000. / 1001.;
 //const int CreateBD_Thread::BD_MAX_BITRATE = 40000000;
 const int CreateBD_Thread::BD_MAX_BITRATE = 8000000;
@@ -41,7 +72,6 @@ const int CreateBD_Thread::BD_WIDE_CHANNELS = 6;
 const double CreateBD_Thread::BD_SAMPLERATE = 48000;
 const double CreateBD_Thread::BD_KAUDIO_RATE = 224;
 
-
 CreateBD_MenuItem::CreateBD_MenuItem(MWindow *mwindow)
  : BC_MenuItem(_("BD Render..."), _("Ctrl-d"), 'd')
 {
@@ -62,13 +92,23 @@ CreateBD_Thread::CreateBD_Thread(MWindow *mwindow)
        this->mwindow = mwindow;
        this->gui = 0;
        this->use_deinterlace = 0;
-       this->use_inverse_telecine = 0;
-       this->use_scale = 0;
-       this->use_resize_tracks = 0;
+       this->use_scale = Rescale::none;
        this->use_histogram = 0;
+       this->use_inverse_telecine = 0;
        this->use_wide_audio = 0;
-       this->use_wide_aspect = 0;
+       this->use_resize_tracks = 0;
        this->use_label_chapters = 0;
+
+       this->bd_size = BD_SIZE;
+       this->bd_width = BD_WIDTH;
+       this->bd_height = BD_HEIGHT;
+       this->bd_aspect_width = BD_WIDE_ASPECT_WIDTH;
+       this->bd_aspect_height = BD_WIDE_ASPECT_HEIGHT;
+       this->bd_framerate = BD_FRAMERATE;
+       this->bd_samplerate = BD_SAMPLERATE;
+       this->bd_max_bitrate = BD_MAX_BITRATE;
+       this->bd_kaudio_rate = BD_KAUDIO_RATE;
+       this->max_w = this->max_h = 0;
 }
 
 CreateBD_Thread::~CreateBD_Thread()
@@ -112,13 +152,12 @@ int CreateBD_Thread::create_bd_jobs(ArrayList<BatchRenderJob*> *jobs,
 
        session->video_channels = BD_STREAMS;
        session->video_tracks = BD_STREAMS;
-// use session framerate
-//     session->frame_rate = BD_FRAMERATE;
-       session->output_w = BD_WIDTH;
-       session->output_h = BD_HEIGHT;
-       session->aspect_w = use_wide_aspect ? BD_WIDE_ASPECT_WIDTH : BD_ASPECT_WIDTH;
-       session->aspect_h = use_wide_aspect ? BD_WIDE_ASPECT_HEIGHT : BD_ASPECT_HEIGHT;
-       session->sample_rate = BD_SAMPLERATE;
+       session->frame_rate = bd_framerate;
+       session->output_w = bd_width;
+       session->output_h = bd_height;
+       session->aspect_w = bd_aspect_width;
+       session->aspect_h = bd_aspect_height;
+       session->sample_rate = bd_samplerate;
        session->audio_channels = session->audio_tracks =
                use_wide_audio ? BD_WIDE_CHANNELS : BD_CHANNELS;
 
@@ -133,10 +172,9 @@ int CreateBD_Thread::create_bd_jobs(ArrayList<BatchRenderJob*> *jobs,
                MainError::show_error(msg);
                return 1;
        }
-       char exe_path[BCTEXTLEN];
-       get_exe_path(exe_path);
+       const char *exec_path = File::get_cinlib_path();
        fprintf(fp,"#!/bin/bash -ex\n");
-       fprintf(fp,"PATH=$PATH:%s\n",exe_path);
+       fprintf(fp,"PATH=$PATH:%s\n",exec_path);
        fprintf(fp,"mkdir -p $1/udfs\n");
        fprintf(fp,"sz=`du -sb $1/bd.m2ts | sed -e 's/[ \t].*//'`\n");
        fprintf(fp,"blks=$((sz/2048 + 4096))\n");
@@ -145,7 +183,8 @@ int CreateBD_Thread::create_bd_jobs(ArrayList<BatchRenderJob*> *jobs,
        fprintf(fp,"bdwrite $1/udfs $1/bd.m2ts\n");
        fprintf(fp,"umount $1/udfs\n");
        fprintf(fp,"echo To burn bluray, load writable media and run:\n");
-       fprintf(fp,"echo growisofs -dvd-compat -Z /dev/bd=$1/bd.udfs\n");
+       fprintf(fp,"echo for WORM: growisofs -dvd-compat -Z /dev/bd=$1/bd.udfs\n");
+       fprintf(fp,"echo for RW:   dd if=$1/bd.udfs of=/dev/bd bs=2048000\n");
        fprintf(fp,"\n");
        fclose(fp);
 
@@ -174,11 +213,11 @@ int CreateBD_Thread::create_bd_jobs(ArrayList<BatchRenderJob*> *jobs,
        edl->resample(old_samplerate, new_samplerate, TRACK_AUDIO);
        edl->resample(old_framerate, new_framerate, TRACK_VIDEO);
 
-       int64_t aud_size = ((BD_KAUDIO_RATE * total_length)/8 + 1000-1) * 1000;
-       int64_t vid_size = BD_SIZE*0.96 - aud_size;
+       int64_t aud_size = ((bd_kaudio_rate * total_length)/8 + 1000-1) * 1000;
+       int64_t vid_size = bd_size*0.96 - aud_size;
        int64_t vid_bitrate = (vid_size * 8) / total_length;
        vid_bitrate /= 1000;  vid_bitrate *= 1000;
-       if( vid_bitrate > BD_MAX_BITRATE ) vid_bitrate = BD_MAX_BITRATE;
+       if( vid_bitrate > bd_max_bitrate ) vid_bitrate = bd_max_bitrate;
 
        char xml_filename[BCTEXTLEN];
        sprintf(xml_filename, "%s/bd.xml", asset_dir);
@@ -213,7 +252,7 @@ int CreateBD_Thread::create_bd_jobs(ArrayList<BatchRenderJob*> *jobs,
        FFMPEG::set_option_path(option_path, "audio/%s", asset->acodec);
        FFMPEG::load_options(option_path, asset->ff_audio_options,
                         sizeof(asset->ff_audio_options));
-       asset->ff_audio_bitrate = BD_KAUDIO_RATE * 1000;
+       asset->ff_audio_bitrate = bd_kaudio_rate * 1000;
 
        asset->video_data = 1;
        strcpy(asset->vcodec, "bluray.m2ts");
@@ -249,11 +288,37 @@ void CreateBD_Thread::handle_close_event(int result)
                keyframe.set_data(data);
                insert_video_plugin("Inverse Telecine", &keyframe);
        }
-       if( use_scale ) {
-               sprintf(data,"<SCALE TYPE=1 X_FACTOR=1 Y_FACTOR=1 "
-                       " WIDTH=%d HEIGHT=%d CONSTRAIN=0>", BD_WIDTH, BD_HEIGHT);
-               keyframe.set_data(data);
-               insert_video_plugin("Scale", &keyframe);
+       if( use_scale != Rescale::none ) {
+               double bd_aspect = bd_aspect_height > 0 ? bd_aspect_width / bd_aspect_height : 1;
+
+               Tracks *tracks = mwindow->edl->tracks;
+               for( Track *vtrk=tracks->first; vtrk; vtrk=vtrk->next ) {
+                       if( vtrk->data_type != TRACK_VIDEO ) continue;
+                       if( !vtrk->record ) continue;
+                       vtrk->expand_view = 1;
+                       PluginSet *plugin_set = new PluginSet(mwindow->edl, vtrk);
+                       vtrk->plugin_set.append(plugin_set);
+                       Edits *edits = vtrk->edits;
+                       for( Edit *edit=edits->first; edit; edit=edit->next ) {
+                               Indexable *indexable = edit->get_source();
+                               if( !indexable ) continue;
+                               Rescale in(indexable);
+                               Rescale out(bd_width, bd_height, bd_aspect);
+                               float src_w, src_h, dst_w, dst_h;
+                               in.rescale(out,use_scale, src_w,src_h, dst_w,dst_h);
+                               sprintf(data,"<SCALERATIO TYPE=%d"
+                                       " IN_W=%d IN_H=%d IN_ASPECT_RATIO=%f"
+                                       " OUT_W=%d OUT_H=%d OUT_ASPECT_RATIO=%f"
+                                       " SRC_X=%f SRC_Y=%f SRC_W=%f SRC_H=%f"
+                                       " DST_X=%f DST_Y=%f DST_W=%f DST_H=%f>", use_scale,
+                                       in.w, in.h, in.aspect, out.w, out.h, out.aspect,
+                                       0., 0., src_w, src_h, 0., 0., dst_w, dst_h);
+                               keyframe.set_data(data);
+                               plugin_set->insert_plugin(_("Scale Ratio"),
+                                       edit->startproject, edit->length,
+                                       PLUGIN_STANDALONE, 0, &keyframe, 0);
+                       }
+               }
        }
        if( use_resize_tracks )
                resize_tracks();
@@ -279,7 +344,7 @@ void CreateBD_Thread::handle_close_event(int result)
                keyframe.set_data(data);
                insert_video_plugin("Histogram", &keyframe);
        }
-       mwindow->batch_render->reset();
+       mwindow->batch_render->reset(1);
        create_bd_jobs(&mwindow->batch_render->jobs, tmp_path, asset_title);
        mwindow->save_backup();
        mwindow->undo->update_undo_after(_("create bd"), LOAD_ALL);
@@ -293,24 +358,50 @@ BC_Window* CreateBD_Thread::new_gui()
        memset(tmp_path,0,sizeof(tmp_path));
        strcpy(tmp_path,"/tmp");
        memset(asset_title,0,sizeof(asset_title));
-       time_t dt;      time(&dt);
-       struct tm dtm;  localtime_r(&dt, &dtm);
+       time_t dt;      time(&dt);
+       struct tm dtm;  localtime_r(&dt, &dtm);
        sprintf(asset_title, "bd_%02d%02d%02d-%02d%02d%02d",
                dtm.tm_year+1900, dtm.tm_mon+1, dtm.tm_mday,
                dtm.tm_hour, dtm.tm_min, dtm.tm_sec);
        use_deinterlace = 0;
-       use_inverse_telecine = 0;
-       use_scale = 0;
-       use_resize_tracks = 0;
+       use_scale = Rescale::none;
        use_histogram = 0;
+       use_inverse_telecine = 0;
        use_wide_audio = 0;
-       use_wide_aspect = 0;
+       use_resize_tracks = 0;
        use_label_chapters = 0;
+       use_standard = BD_1920x1080_2997i;
+
+       bd_size = BD_SIZE;
+       bd_width = BD_WIDTH;
+       bd_height = BD_HEIGHT;
+       bd_aspect_width = BD_WIDE_ASPECT_WIDTH;
+       bd_aspect_height = BD_WIDE_ASPECT_HEIGHT;
+       bd_framerate = BD_FRAMERATE;
+       bd_samplerate = BD_SAMPLERATE;
+       bd_max_bitrate = BD_MAX_BITRATE;
+       bd_kaudio_rate = BD_KAUDIO_RATE;
+       max_w = 0; max_h = 0;
+
+       int has_standard = -1;
+       if( mwindow->edl ) {
+               EDLSession *session = mwindow->edl->session;
+// match the session to any known standard
+               for( int i=0; i<(int)(sizeof(bd_formats)/sizeof(bd_formats[0])); ++i ) {
+                       if( !EQUIV(session->frame_rate, bd_formats[i].framerate) ) continue;
+                       if( session->output_w != bd_formats[i].w ) continue;
+                       if( session->output_h != bd_formats[i].h ) continue;
+                       has_standard = i;  break;
+               }
+       }
+       if( has_standard >= 0 )
+               use_standard = has_standard;
+
        option_presets();
        int scr_x = mwindow->gui->get_screen_x(0, -1);
        int scr_w = mwindow->gui->get_screen_w(0, -1);
        int scr_h = mwindow->gui->get_screen_h(0, -1);
-       int w = 500, h = 250;
+       int w = 500, h = 280;
        int x = scr_x + scr_w/2 - w/2, y = scr_h/2 - h/2;
 
        gui = new CreateBD_GUI(this, x, y, w, h);
@@ -386,10 +477,17 @@ int64_t CreateBD_DiskSpace::tmp_path_space()
 
 void CreateBD_DiskSpace::update()
 {
-//     gui->disk_space->set_color(get_bg_color());
-       int64_t disk_space = tmp_path_space();
-       int color = disk_space<gui->needed_disk_space ? RED : GREEN;
        static const char *suffix[] = { "", "KB", "MB", "GB", "TB", "PB" };
+       int64_t disk_space = tmp_path_space();
+       double media_size = 100e9, msz = 0, m = 1;
+       char sfx[BCSTRLEN];
+       if( sscanf(gui->media_size->get_text(), "%lf%s", &msz, sfx) == 2 ) {
+               int i = sizeof(suffix)/sizeof(suffix[0]);
+               while( --i >= 0 && strcmp(sfx, suffix[i]) );
+               while( --i >= 0 ) m *= 1000;
+               media_size = msz * m;
+       }
+       int color = disk_space < media_size*2 ? RED : GREEN;
        int i = 0;
        for( int64_t space=disk_space; i<5 && (space/=1000)>0; disk_space=space, ++i );
        char text[BCTEXTLEN];
@@ -467,17 +565,6 @@ int CreateBD_InverseTelecine::handle_event()
 }
 
 
-CreateBD_Scale::CreateBD_Scale(CreateBD_GUI *gui, int x, int y)
- : BC_CheckBox(x, y, &gui->thread->use_scale, _("Scale"))
-{
-       this->gui = gui;
-}
-
-CreateBD_Scale::~CreateBD_Scale()
-{
-}
-
-
 CreateBD_ResizeTracks::CreateBD_ResizeTracks(CreateBD_GUI *gui, int x, int y)
  : BC_CheckBox(x, y, &gui->thread->use_resize_tracks, _("Resize Tracks"))
 {
@@ -519,17 +606,6 @@ CreateBD_WideAudio::~CreateBD_WideAudio()
 {
 }
 
-CreateBD_WideAspect::CreateBD_WideAspect(CreateBD_GUI *gui, int x, int y)
- : BC_CheckBox(x, y, &gui->thread->use_wide_aspect, _("Aspect 16x9"))
-{
-       this->gui = gui;
-}
-
-CreateBD_WideAspect::~CreateBD_WideAspect()
-{
-}
-
-
 
 CreateBD_GUI::CreateBD_GUI(CreateBD_Thread *thread, int x, int y, int w, int h)
  : BC_Window(_(PROGRAM_NAME ": Create BD"), x, y, w, h, 50, 50, 1, 0, 1)
@@ -540,15 +616,15 @@ CreateBD_GUI::CreateBD_GUI(CreateBD_Thread *thread, int x, int y, int w, int h)
        cancel_x = cancel_y = cancel_w = cancel_h = 0;
        asset_title = 0;
        tmp_path = 0;
+       btmp_path = 0;
        disk_space = 0;
-       needed_disk_space = 100e9;
+       standard = 0;
+       scale = 0;
        need_deinterlace = 0;
        need_inverse_telecine = 0;
-       need_scale = 0;
        need_resize_tracks = 0;
        need_histogram = 0;
        need_wide_audio = 0;
-       need_wide_aspect = 0;
        need_label_chapters = 0;
        ok = 0;
        cancel = 0;
@@ -570,33 +646,56 @@ void CreateBD_GUI::create_objects()
        asset_title = new CreateBD_AssetTitle(this, at_x, at_y, get_w()-at_x-10);
        add_subwindow(asset_title);
        y += title->get_h() + pady/2;
-       title = new BC_Title(x, y, _("tmp path:"), MEDIUMFONT, YELLOW);
+       title = new BC_Title(x, y, _("Work path:"), MEDIUMFONT, YELLOW);
        add_subwindow(title);
        tmp_x = x + title->get_w();  tmp_y = y;
-       tmp_path = new CreateBD_TmpPath(this, tmp_x, tmp_y,  get_w()-tmp_x-10);
+       tmp_path = new CreateBD_TmpPath(this, tmp_x, tmp_y,  get_w()-tmp_x-35);
        add_subwindow(tmp_path);
+       btmp_path = new BrowseButton(thread->mwindow->theme, this, tmp_path,
+               tmp_x+tmp_path->get_w(), tmp_y, "/tmp",
+               _("Work path"), _("Select a Work directory:"), 1);
+       add_subwindow(btmp_path);
        y += title->get_h() + pady/2;
        disk_space = new CreateBD_DiskSpace(this, x, y);
        add_subwindow(disk_space);
+       int x0 = get_w() - 170;
+       title = new BC_Title(x0, y, _("Media:"), MEDIUMFONT, YELLOW);
+       add_subwindow(title);
+       int x1 =  x0+title->get_w()+padx;
+       media_size = new CreateBD_MediaSize(this, x1, y);
+       media_size->create_objects();
+       media_sizes.append(new BC_ListBoxItem("25GB"));
+       media_sizes.append(new BC_ListBoxItem("50GB"));
+       media_size->update_list(&media_sizes);
+       media_size->update(media_sizes[0]->get_text());
        disk_space->update();
        y += disk_space->get_h() + pady/2;
+       title = new BC_Title(x, y, _("Format:"), MEDIUMFONT, YELLOW);
+       add_subwindow(title);
+       standard = new CreateBD_Format(this, title->get_w() + padx, y);
+       add_subwindow(standard);
+       standard->create_objects();
+       x0 -= 30;
+       title = new BC_Title(x0, y, _("Scale:"), MEDIUMFONT, YELLOW);
+       add_subwindow(title);
+       x1 = x0+title->get_w()+padx;
+       scale = new CreateBD_Scale(this, x1, y);
+       add_subwindow(scale);
+       scale->create_objects();
+       y += standard->get_h() + pady/2;
        need_deinterlace = new CreateBD_Deinterlace(this, x, y);
        add_subwindow(need_deinterlace);
-       int x1 = x + 150, x2 = x1 + 150;
+       x1 = x + 170; //, x2 = x1 + 150;
        need_inverse_telecine = new CreateBD_InverseTelecine(this, x1, y);
        add_subwindow(need_inverse_telecine);
        y += need_deinterlace->get_h() + pady/2;
-       need_scale = new CreateBD_Scale(this, x, y);
-       add_subwindow(need_scale);
+       need_histogram = new CreateBD_Histogram(this, x, y);
+       add_subwindow(need_histogram);
        need_wide_audio = new CreateBD_WideAudio(this, x1, y);
        add_subwindow(need_wide_audio);
-       need_resize_tracks = new CreateBD_ResizeTracks(this, x2, y);
+       y += need_histogram->get_h() + pady/2;
+       need_resize_tracks = new CreateBD_ResizeTracks(this, x1, y);
        add_subwindow(need_resize_tracks);
-       y += need_scale->get_h() + pady/2;
-       need_histogram = new CreateBD_Histogram(this, x, y);
-       add_subwindow(need_histogram);
-       need_wide_aspect = new CreateBD_WideAspect(this, x1, y);
-       add_subwindow(need_wide_aspect);
 //     need_label_chapters = new CreateBD_LabelChapters(this, x2, y);
 //     add_subwindow(need_label_chapters);
        ok_w = BC_OKButton::calculate_w();
@@ -618,7 +717,8 @@ void CreateBD_GUI::create_objects()
 int CreateBD_GUI::resize_event(int w, int h)
 {
        asset_title->reposition_window(at_x, at_y, get_w()-at_x-10);
-       tmp_path->reposition_window(tmp_x, tmp_y,  get_w()-tmp_x-10);
+       tmp_path->reposition_window(tmp_x, tmp_y,  get_w()-tmp_x-35);
+       btmp_path->reposition_window(tmp_x+tmp_path->get_w(), tmp_y);
        ok_y = h - ok_h - 10;
        ok->reposition_window(ok_x, ok_y);
        cancel_x = w - cancel_w - 10,
@@ -638,6 +738,17 @@ int CreateBD_GUI::close_event()
        return 1;
 }
 
+void CreateBD_GUI::update()
+{
+       scale->set_value(thread->use_scale);
+       need_deinterlace->set_value(thread->use_deinterlace);
+       need_inverse_telecine->set_value(thread->use_inverse_telecine);
+       need_resize_tracks->set_value(thread->use_resize_tracks);
+       need_histogram->set_value(thread->use_histogram);
+       need_wide_audio->set_value(thread->use_wide_audio);
+//     need_label_chapters->set_value(thread->use_label_chapters);
+}
+
 int CreateBD_Thread::
 insert_video_plugin(const char *title, KeyFrame *default_keyframe)
 {
@@ -654,7 +765,6 @@ insert_video_plugin(const char *title, KeyFrame *default_keyframe)
                                edit->startproject, edit->length,
                                PLUGIN_STANDALONE, 0, default_keyframe, 0);
                }
-               vtrk->optimize();
        }
        return 0;
 }
@@ -663,25 +773,14 @@ int CreateBD_Thread::
 resize_tracks()
 {
        Tracks *tracks = mwindow->edl->tracks;
-       int max_w = 0, max_h = 0;
-       for( Track *vtrk=tracks->first; vtrk; vtrk=vtrk->next ) {
-               if( vtrk->data_type != TRACK_VIDEO ) continue;
-               if( !vtrk->record ) continue;
-               Edits *edits = vtrk->edits;
-               for( Edit *edit=edits->first; edit; edit=edit->next ) {
-                       Indexable *indexable = edit->get_source();
-                       int w = indexable->get_w();
-                       if( w > max_w ) max_w = w;
-                       int h = indexable->get_h();
-                       if( h > max_h ) max_h = h;
-               }
-       }
-
+       int trk_w = max_w, trk_h = max_h;
+       if( trk_w < bd_width ) trk_w = bd_width;
+       if( trk_h < bd_height ) trk_h = bd_height;
        for( Track *vtrk=tracks->first; vtrk; vtrk=vtrk->next ) {
                if( vtrk->data_type != TRACK_VIDEO ) continue;
                if( !vtrk->record ) continue;
-               vtrk->track_w = max_w;
-               vtrk->track_h = max_h;
+               vtrk->track_w = trk_w;
+               vtrk->track_h = trk_h;
        }
        return 0;
 }
@@ -689,9 +788,26 @@ resize_tracks()
 int CreateBD_Thread::
 option_presets()
 {
+// reset only probed options
+       use_deinterlace = 0;
+       use_scale = Rescale::none;
+       use_resize_tracks = 0;
+       use_wide_audio = 0;
+       use_label_chapters = 0;
+
        if( !mwindow->edl ) return 1;
+
+       bd_width = bd_formats[use_standard].w;
+       bd_height = bd_formats[use_standard].h;
+       bd_framerate = bd_formats[use_standard].framerate;
+       bd_aspect_width = bd_formats[use_standard].wide ?
+               BD_WIDE_ASPECT_WIDTH : BD_ASPECT_WIDTH;
+       bd_aspect_height = bd_formats[use_standard].wide ?
+               BD_WIDE_ASPECT_HEIGHT : BD_ASPECT_HEIGHT;
+       double bd_aspect = bd_aspect_width / bd_aspect_height;
+
        Tracks *tracks = mwindow->edl->tracks;
-       int max_w = 0, max_h = 0;
+       max_w = 0;  max_h = 0;
        int has_deinterlace = 0, has_scale = 0;
        for( Track *trk=tracks->first; trk; trk=trk->next ) {
                if( !trk->record ) continue;
@@ -703,10 +819,14 @@ option_presets()
                                Indexable *indexable = edit->get_source();
                                int w = indexable->get_w();
                                if( w > max_w ) max_w = w;
-                               if( w != BD_WIDTH ) use_scale = 1;
+                               if( w != bd_width ) use_scale = Rescale::scaled;
                                int h = indexable->get_h();
                                if( h > max_h ) max_h = h;
-                               if( h != BD_HEIGHT ) use_scale = 1;
+                               if( h != bd_height ) use_scale = Rescale::scaled;
+                               float aw, ah;
+                               MWindow::create_aspect_ratio(aw, ah, w, h);
+                               double aspect = ah > 0 ? aw / ah : 1;
+                               if( !EQUIV(aspect, bd_aspect) ) use_scale = Rescale::scaled;
                        }
                        for( int i=0; i<trk->plugin_set.size(); ++i ) {
                                for(Plugin *plugin = (Plugin*)trk->plugin_set[i]->first;
@@ -715,6 +835,7 @@ option_presets()
                                        if( !strcmp(plugin->title, _("Deinterlace")) )
                                                has_deinterlace = 1;
                                        if( !strcmp(plugin->title, _("Auto Scale")) ||
+                                           !strcmp(plugin->title, _("Scale Ratio")) ||
                                            !strcmp(plugin->title, _("Scale")) )
                                                has_scale = 1;
                                }
@@ -723,10 +844,10 @@ option_presets()
                }
        }
        if( has_scale )
-               use_scale = 0;
-       if( use_scale ) {
-               if( max_w != BD_WIDTH ) use_resize_tracks = 1;
-               if( max_h != BD_HEIGHT ) use_resize_tracks = 1;
+               use_scale = Rescale::none;
+       if( use_scale != Rescale::none ) {
+               if( max_w != bd_width ) use_resize_tracks = 1;
+               if( max_h != bd_height ) use_resize_tracks = 1;
        }
        for( Track *trk=tracks->first; trk && !use_resize_tracks; trk=trk->next ) {
                if( !trk->record ) continue;
@@ -737,15 +858,122 @@ option_presets()
                        break;
                }
        }
-       if( !has_deinterlace && max_h > 2*BD_HEIGHT ) use_deinterlace = 1;
+       if( !has_deinterlace && max_h > 2*bd_height ) use_deinterlace = 1;
        // Labels *labels = mwindow->edl->labels;
        // use_label_chapters = labels && labels->first ? 1 : 0;
-       float w, h;
-       MWindow::create_aspect_ratio(w, h, max_w, max_h);
-       if( w == BD_WIDE_ASPECT_WIDTH && h == BD_WIDE_ASPECT_HEIGHT )
-               use_wide_aspect = 1;
+
        if( tracks->recordable_audio_tracks() == BD_WIDE_CHANNELS )
                use_wide_audio = 1;
+
        return 0;
 }
 
+
+CreateBD_FormatItem::CreateBD_FormatItem(CreateBD_Format *popup,
+               int standard, const char *name)
+ : BC_MenuItem(name)
+{
+       this->popup = popup;
+       this->standard = standard;
+}
+
+CreateBD_FormatItem::~CreateBD_FormatItem()
+{
+}
+
+int CreateBD_FormatItem::handle_event()
+{
+       popup->set_text(get_text());
+       popup->gui->thread->use_standard = standard;
+       return popup->handle_event();
+}
+
+
+CreateBD_Format::CreateBD_Format(CreateBD_GUI *gui, int x, int y)
+ : BC_PopupMenu(x, y, 180, bd_formats[gui->thread->use_standard].name, 1)
+{
+       this->gui = gui;
+}
+
+CreateBD_Format::~CreateBD_Format()
+{
+}
+
+void CreateBD_Format::create_objects()
+{
+       for( int i=0; i<(int)(sizeof(bd_formats)/sizeof(bd_formats[0])); ++i ) {
+               add_item(new CreateBD_FormatItem(this, i, bd_formats[i].name));
+       }
+       set_value(gui->thread->use_standard);
+}
+
+int CreateBD_Format::handle_event()
+{
+       gui->thread->option_presets();
+       gui->update();
+       return 1;
+}
+
+
+CreateBD_ScaleItem::CreateBD_ScaleItem(CreateBD_Scale *popup,
+               int scale, const char *text)
+ : BC_MenuItem(text)
+{
+       this->popup = popup;
+       this->scale = scale;
+}
+
+CreateBD_ScaleItem::~CreateBD_ScaleItem()
+{
+}
+
+int CreateBD_ScaleItem::handle_event()
+{
+       popup->gui->thread->use_scale = scale;
+       popup->set_value(scale);
+       return popup->handle_event();
+}
+
+
+CreateBD_Scale::CreateBD_Scale(CreateBD_GUI *gui, int x, int y)
+ : BC_PopupMenu(x, y, 100, "", 1)
+{
+       this->gui = gui;
+}
+
+CreateBD_Scale::~CreateBD_Scale()
+{
+}
+
+void CreateBD_Scale::create_objects()
+{
+
+       for( int i=0; i<(int)Rescale::n_scale_types; ++i ) {
+               add_item(new CreateBD_ScaleItem(this, i, Rescale::scale_types[i]));
+       }
+       set_value(gui->thread->use_scale);
+}
+
+int CreateBD_Scale::handle_event()
+{
+       gui->update();
+       return 1;
+}
+
+
+CreateBD_MediaSize::CreateBD_MediaSize(CreateBD_GUI *gui, int x, int y)
+ : BC_PopupTextBox(gui, 0, 0, x, y, 70,50)
+{
+       this->gui = gui;
+}
+
+CreateBD_MediaSize::~CreateBD_MediaSize()
+{
+}
+
+int CreateBD_MediaSize::handle_event()
+{
+       gui->disk_space->update();
+       return 1;
+}
+