rework histogram_bezier, init wm icon set_icon(gg), update de.po+msg/txt
[goodguy/history.git] / cinelerra-5.1 / cinelerra / bdcreate.C
index 20ecf5878e2da821200d1bc0f902d03cc9eff26b..27f728ad6dc80c1a21d20723d5eec697edf30a34 100644 (file)
@@ -1,12 +1,14 @@
 #include "asset.h"
+#include "bchash.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 "interlacemodes.h"
 #include "keyframe.h"
 #include "labels.h"
 #include "mainerror.h"
@@ -15,6 +17,8 @@
 #include "mwindowgui.h"
 #include "plugin.h"
 #include "pluginset.h"
+#include "preferences.h"
+#include "rescale.h"
 #include "track.h"
 #include "tracks.h"
 
 
 // BD Creation
 
+// selected by timezone
 #define BD_1920x1080_2997i     0
-#define BD_1920x1080_2500i     1
-#define BD_1920x1080_2400p     2
-#define BD_1920x1080_23976p    3
-#define BD_1280x720_5997p      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
+#define BD_1920x1080_25i       3
 
 static struct bd_format {
        const char *name;
        int w, h;
        double framerate;
+       int wide, interlaced;
 } bd_formats[] = {
-       { "1920x1080 29.97i",     1920,1080, 29.97  },
-       { "1920x1080 25i",        1920,1080, 25     },
-       { "1920x1080 24p",        1920,1080, 24     },
-       { "1920x1080 23.976p",    1920,1080, 23.976 },
-       { "1280x720 59.97p",      1280,720,  59.97  },
-       { "1280x720 50p",         1280,720,  50     },
-       { "1280x720 23.976p",     1280,720,  23.976 },
-       { "1280x720 24p",         1280,720,  24     },
-       { "720x576 25i (PAL)",     720,576,  25     },
-       { "720x480 29.97i (NTSC)", 720,480,  29.97  },
+// framerates are frames, not fields, per second, *=not standard
+       { "1920x1080 29.97i",   1920,1080, 29.97,  1, ILACE_MODE_TOP_FIRST },
+       { "1920x1080 29.97p*",  1920,1080, 29.97,  1, ILACE_MODE_NOTINTERLACED },
+       { "1920x1080 24p",      1920,1080, 24.,    1, ILACE_MODE_NOTINTERLACED },
+       { "1920x1080 25i",      1920,1080, 25.,    1, ILACE_MODE_TOP_FIRST },
+       { "1920x1080 23.976p",  1920,1080, 23.976, 1, ILACE_MODE_NOTINTERLACED },
+       { "1440x1080 29.97i",   1440,1080, 29.97, -1, ILACE_MODE_TOP_FIRST },
+       { "1440x1080 25i",      1440,1080, 25.,   -1, ILACE_MODE_TOP_FIRST },
+       { "1440x1080 24p",      1440,1080, 24.,   -1, ILACE_MODE_NOTINTERLACED },
+       { "1440x1080 23.976p",  1440,1080, 23.976,-1, ILACE_MODE_NOTINTERLACED },
+       { "1280x720  59.94p",   1280,720,  59.94,  1, ILACE_MODE_NOTINTERLACED },
+       { "1280x720  50p",      1280,720,  50.,    1, ILACE_MODE_NOTINTERLACED },
+       { "1280x720  24p",      1280,720,  24.,    1, ILACE_MODE_NOTINTERLACED },
+       { "1280x720  23.976p",  1280,720,  23.976, 1, ILACE_MODE_NOTINTERLACED },
+       { "720x576   25i",       720,576,  25.,    0, ILACE_MODE_BOTTOM_FIRST },
+       { "720x576   25p*",      720,576,  25.,    0, ILACE_MODE_NOTINTERLACED },
+       { "720x480   29.97i",    720,480,  29.97,  0, ILACE_MODE_BOTTOM_FIRST },
+       { "720x480   29.97p*",   720,480,  29.97,  0, ILACE_MODE_NOTINTERLACED },
 };
 
 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;
+const int CreateBD_Thread::BD_MAX_BITRATE = 10000000;
 const int CreateBD_Thread::BD_CHANNELS = 2;
 const int CreateBD_Thread::BD_WIDE_CHANNELS = 6;
 const double CreateBD_Thread::BD_SAMPLERATE = 48000;
-const double CreateBD_Thread::BD_KAUDIO_RATE = 224;
-
+const double CreateBD_Thread::BD_KAUDIO_RATE = 192;
+const int CreateBD_Thread::BD_INTERLACE_MODE = ILACE_MODE_NOTINTERLACED;
 
 CreateBD_MenuItem::CreateBD_MenuItem(MWindow *mwindow)
  : BC_MenuItem(_("BD Render..."), _("Ctrl-d"), 'd')
 {
-       set_ctrl(1); 
+       set_ctrl(1);
        this->mwindow = mwindow;
 }
 
@@ -91,19 +97,18 @@ CreateBD_Thread::CreateBD_Thread(MWindow *mwindow)
        this->mwindow = mwindow;
        this->gui = 0;
        this->use_deinterlace = 0;
-       this->use_scale = 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_ASPECT_WIDTH;
-       this->bd_aspect_height = BD_ASPECT_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;
@@ -116,8 +121,53 @@ CreateBD_Thread::~CreateBD_Thread()
        close_window();
 }
 
-int CreateBD_Thread::create_bd_jobs(ArrayList<BatchRenderJob*> *jobs,
-       const char *tmp_path, const char *asset_title)
+int CreateBD_Thread::get_udfs_mount(char *udfs, char *mopts, char *mntpt)
+{
+       int ret = 0;
+// default: mount -t udf -o loop $1/bd.udfs $1/udfs
+       strcpy(udfs,"$1/bd.udfs");
+       strcpy(mopts,"-t udf -o loop $1/bd.udfs ");
+       strcpy(mntpt,"$1/udfs");
+       const char *home = getenv("HOME");
+       if( !home ) return ret;
+       FILE *fp = fopen("/etc/fstab","r");
+       if( !fp ) return ret;
+       int len = strlen(home);
+       char line[BCTEXTLEN], typ[BCTEXTLEN], file[BCTEXTLEN];
+       char mpnt[BCTEXTLEN], opts[BCTEXTLEN];
+       while( fgets(line,sizeof(line),fp) ) {
+// search "/etc/fstab" for: $HOME/img_file $HOME/bluray udf noauto,loop,rw,user
+               if( line[0] == '#' || line[0] == ';' ) continue;
+               if( sscanf(line,"%s %s %s %s ", &file[0], &mpnt[0], &typ[0], &opts[0]) != 4 )
+                       continue;
+               if( strcmp("udf", typ) ) continue;
+               if( strncmp(home, file, len) || file[len] != '/' ) continue;
+               if( strncmp(home, mpnt, len) ) continue;
+               if( strcmp(&mpnt[len], "/bluray") ) continue;
+               int loop = 0, user = 0, rw = 0, noauto = 0;
+               char *op = opts, *ep = op + sizeof(opts)-1;
+               while( op < ep && *op ) {
+                       char opt[BCTEXTLEN], *cp = opt;
+                       while( op < ep && *op && (*cp=*op++)!=',' ) ++cp;
+                       *cp = 0;
+                       if( !strcmp("loop", opt) ) loop = 1;
+                       else if( !strcmp("user", opt) ) user = 1;
+                       else if( !strcmp("noauto", opt) ) noauto = 1;
+                       else if( !strcmp("rw", opt) ) rw = 1;
+               }
+               if( loop && user && rw && noauto ) {
+                       strcpy(udfs, file);
+                       strcpy(mopts, "");
+                       strcpy(mntpt, mpnt);
+                       ret = 1;
+                       break;
+               }
+       }
+       fclose(fp);
+       return ret;
+}
+
+int CreateBD_Thread::create_bd_jobs(ArrayList<BatchRenderJob*> *jobs, const char *asset_dir)
 {
        EDL *edl = mwindow->edl;
        if( !edl || !edl->session ) {
@@ -136,9 +186,6 @@ int CreateBD_Thread::create_bd_jobs(ArrayList<BatchRenderJob*> *jobs,
                return 1;
        }
 
-       char asset_dir[BCTEXTLEN];
-       sprintf(asset_dir, "%s/%s", tmp_path, asset_title);
-
        if( mkdir(asset_dir, 0777) ) {
                char err[BCTEXTLEN], msg[BCTEXTLEN];
                strerror_r(errno, err, sizeof(err));
@@ -160,6 +207,7 @@ int CreateBD_Thread::create_bd_jobs(ArrayList<BatchRenderJob*> *jobs,
        session->sample_rate = bd_samplerate;
        session->audio_channels = session->audio_tracks =
                use_wide_audio ? BD_WIDE_CHANNELS : BD_CHANNELS;
+       session->interlace_mode = bd_interlace_mode;
 
        char script_filename[BCTEXTLEN];
        sprintf(script_filename, "%s/bd.sh", asset_dir);
@@ -172,43 +220,39 @@ int CreateBD_Thread::create_bd_jobs(ArrayList<BatchRenderJob*> *jobs,
                MainError::show_error(msg);
                return 1;
        }
-       char exe_path[BCTEXTLEN];
-       get_exe_path(exe_path);
+       char udfs[BCTEXTLEN], mopts[BCTEXTLEN], mntpt[BCTEXTLEN];
+       int is_usr_mnt = get_udfs_mount(udfs, mopts, mntpt);
+       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,"sz=`du -cb $1/bd.m2ts* | tail -1 | sed -e 's/[ \t].*//'`\n");
        fprintf(fp,"blks=$((sz/2048 + 4096))\n");
-       fprintf(fp,"mkudffs $1/bd.udfs $blks\n");
-       fprintf(fp,"mount -o loop $1/bd.udfs $1/udfs\n");
-       fprintf(fp,"bdwrite $1/udfs $1/bd.m2ts\n");
-       fprintf(fp,"umount $1/udfs\n");
+       fprintf(fp,"rm -f %s\n", udfs);
+       fprintf(fp,"mkudffs -b 2048 %s $blks\n", udfs);
+       fprintf(fp,"mount %s%s\n", mopts, mntpt);
+       fprintf(fp,"bdwrite %s $1/bd.m2ts*\n",mntpt);
+       fprintf(fp,"umount %s\n",mntpt);
+       if( is_usr_mnt )
+               fprintf(fp,"mv -f %s $1/bd.udfs\n", udfs);
        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,"kill $$\n");
        fprintf(fp,"\n");
        fclose(fp);
 
-       if( use_wide_audio ) {
-               session->audio_channels = session->audio_tracks = BD_WIDE_CHANNELS;
-               session->achannel_positions[0] = 90;
-               session->achannel_positions[1] = 150;
-               session->achannel_positions[2] = 30;
-               session->achannel_positions[3] = 210;
-               session->achannel_positions[4] = 330;
-               session->achannel_positions[5] = 270;
-               if( edl->tracks->recordable_audio_tracks() == BD_WIDE_CHANNELS )
-                       mwindow->remap_audio(MWindow::AUDIO_1_TO_1);
-       }
-       else {
-               session->audio_channels = session->audio_tracks = BD_CHANNELS;
-               session->achannel_positions[0] = 180;
-               session->achannel_positions[1] = 0;
-               if( edl->tracks->recordable_audio_tracks() == BD_WIDE_CHANNELS )
-                       mwindow->remap_audio(MWindow::AUDIO_5_1_TO_2);
-       }
+       session->audio_channels = session->audio_tracks =
+               !use_wide_audio ? BD_CHANNELS : BD_WIDE_CHANNELS;
+       for( int i=0; i<MAX_CHANNELS; ++i )
+               session->achannel_positions[i] = default_audio_channel_position(i, session->audio_channels);
+       int audio_mapping = edl->tracks->recordable_audio_tracks() == BD_WIDE_CHANNELS &&
+               !use_wide_audio ? MWindow::AUDIO_5_1_TO_2 : MWindow::AUDIO_1_TO_1;
+       mwindow->remap_audio(audio_mapping);
 
        double new_samplerate = session->sample_rate;
        double new_framerate = session->frame_rate;
+       edl->retrack();
        edl->rechannel();
        edl->resample(old_samplerate, new_samplerate, TRACK_AUDIO);
        edl->resample(old_framerate, new_framerate, TRACK_VIDEO);
@@ -222,7 +266,7 @@ int CreateBD_Thread::create_bd_jobs(ArrayList<BatchRenderJob*> *jobs,
        char xml_filename[BCTEXTLEN];
        sprintf(xml_filename, "%s/bd.xml", asset_dir);
        FileXML xml_file;
-       edl->save_xml(&xml_file, xml_filename, 0, 0);
+       edl->save_xml(&xml_file, xml_filename);
        xml_file.terminate_string();
        if( xml_file.write_to_file(xml_filename) ) {
                char msg[BCTEXTLEN];
@@ -231,7 +275,7 @@ int CreateBD_Thread::create_bd_jobs(ArrayList<BatchRenderJob*> *jobs,
                return 1;
        }
 
-       BatchRenderJob *job = new BatchRenderJob(mwindow->preferences);
+       BatchRenderJob *job = new BatchRenderJob(mwindow->preferences, use_label_chapters);
        jobs->append(job);
        strcpy(&job->edl_path[0], xml_filename);
        Asset *asset = job->asset;
@@ -241,6 +285,7 @@ int CreateBD_Thread::create_bd_jobs(ArrayList<BatchRenderJob*> *jobs,
        asset->width = session->output_w;
        asset->height = session->output_h;
        asset->aspect_ratio = session->aspect_w / session->aspect_h;
+       asset->interlace_mode = session->interlace_mode;
 
        char option_path[BCTEXTLEN];
        sprintf(&asset->path[0],"%s/bd.m2ts", asset_dir);
@@ -249,6 +294,7 @@ int CreateBD_Thread::create_bd_jobs(ArrayList<BatchRenderJob*> *jobs,
 
        asset->audio_data = 1;
        strcpy(asset->acodec, "bluray.m2ts");
+       //mwindow->defaults->get("DEFAULT_BLURAY_ACODEC", asset->acodec);
        FFMPEG::set_option_path(option_path, "audio/%s", asset->acodec);
        FFMPEG::load_options(option_path, asset->ff_audio_options,
                         sizeof(asset->ff_audio_options));
@@ -256,13 +302,24 @@ int CreateBD_Thread::create_bd_jobs(ArrayList<BatchRenderJob*> *jobs,
 
        asset->video_data = 1;
        strcpy(asset->vcodec, "bluray.m2ts");
+       //mwindow->defaults->get("DEFAULT_BLURAY_VCODEC", asset->vcodec);
        FFMPEG::set_option_path(option_path, "video/%s", asset->vcodec);
        FFMPEG::load_options(option_path, asset->ff_video_options,
                 sizeof(asset->ff_video_options));
+       const char *opts = 0;
+       switch( asset->interlace_mode ) {
+       case ILACE_MODE_TOP_FIRST:    opts = ":tff\n";  break;
+       case ILACE_MODE_BOTTOM_FIRST: opts = ":bff\n";  break;
+       }
+       if( opts ) {
+               int len = strlen(asset->ff_video_options);
+               char *cp = asset->ff_video_options + len-1;
+               strncpy(cp, opts, sizeof(asset->ff_video_options)-len);
+       }
        asset->ff_video_bitrate = vid_bitrate;
-       asset->ff_video_quality = 0;
+       asset->ff_video_quality = -1;
 
-       job = new BatchRenderJob(mwindow->preferences);
+       job = new BatchRenderJob(mwindow->preferences, 0, 0);
        jobs->append(job);
        job->edl_path[0] = '@';
        strcpy(&job->edl_path[1], script_filename);
@@ -274,6 +331,7 @@ int CreateBD_Thread::create_bd_jobs(ArrayList<BatchRenderJob*> *jobs,
 void CreateBD_Thread::handle_close_event(int result)
 {
        if( result ) return;
+       mwindow->defaults->update("WORK_DIRECTORY", tmp_path);
        mwindow->batch_render->load_defaults(mwindow->defaults);
        mwindow->undo->update_undo_before();
        KeyFrame keyframe;  char data[BCTEXTLEN];
@@ -288,15 +346,38 @@ 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=%d X_FACTOR=%f Y_FACTOR=%f "
-                       "WIDTH=%d HEIGHT=%d CONSTRAIN=0>",
-                       max_w >= bd_width || max_h >= bd_height ? 1 : 0,
-                       max_w > 0 ? (double)bd_width/max_w : 1,
-                       max_h > 0 ? (double)bd_height/max_h : 1,
-                       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);
+                       }
+                       vtrk->optimize();
+               }
        }
        if( use_resize_tracks )
                resize_tracks();
@@ -322,44 +403,48 @@ void CreateBD_Thread::handle_close_event(int result)
                keyframe.set_data(data);
                insert_video_plugin("Histogram", &keyframe);
        }
-       mwindow->batch_render->reset();
-       create_bd_jobs(&mwindow->batch_render->jobs, tmp_path, asset_title);
-       mwindow->save_backup();
+
+       char asset_dir[BCTEXTLEN], jobs_path[BCTEXTLEN];
+       snprintf(asset_dir, sizeof(asset_dir), "%s/%s", tmp_path, asset_title);
+       snprintf(jobs_path, sizeof(jobs_path), "%s/bd.jobs", asset_dir);
+       mwindow->batch_render->reset(jobs_path);
+       int ret = create_bd_jobs(&mwindow->batch_render->jobs, asset_dir);
        mwindow->undo->update_undo_after(_("create bd"), LOAD_ALL);
        mwindow->resync_guis();
-       mwindow->batch_render->handle_close_event(0);
+       if( ret ) return;
+       mwindow->batch_render->save_jobs();
        mwindow->batch_render->start();
 }
 
 BC_Window* CreateBD_Thread::new_gui()
 {
-       memset(tmp_path,0,sizeof(tmp_path));
-       strcpy(tmp_path,"/tmp");
+       strcpy(tmp_path, "/tmp");
+       mwindow->defaults->get("WORK_DIRECTORY", tmp_path);
        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_scale = 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;
-
+       use_standard = !strcmp(mwindow->default_std(),"NTSC") ?
+                BD_1920x1080_2997i : BD_1920x1080_25i;
        bd_size = BD_SIZE;
        bd_width = BD_WIDTH;
        bd_height = BD_HEIGHT;
-       bd_aspect_width = BD_ASPECT_WIDTH;
-       bd_aspect_height = BD_ASPECT_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;
+       bd_interlace_mode = BD_INTERLACE_MODE;
        max_w = 0; max_h = 0;
 
        int has_standard = -1;
@@ -373,7 +458,8 @@ BC_Window* CreateBD_Thread::new_gui()
                        has_standard = i;  break;
                }
        }
-       use_standard = has_standard >= 0 ? has_standard : BD_1920x1080_2400p;
+       if( has_standard >= 0 )
+               use_standard = has_standard;
 
        option_presets();
        int scr_x = mwindow->gui->get_screen_x(0, -1);
@@ -384,6 +470,12 @@ BC_Window* CreateBD_Thread::new_gui()
 
        gui = new CreateBD_GUI(this, x, y, w, h);
        gui->create_objects();
+
+       if( getuid() != 0 ) {
+               mwindow->show_warning(
+                       &mwindow->preferences->bd_warn_root,
+                       _("Must be root to mount UDFS images\n"));
+       }
        return gui;
 }
 
@@ -446,7 +538,7 @@ CreateBD_DiskSpace::~CreateBD_DiskSpace()
 
 int64_t CreateBD_DiskSpace::tmp_path_space()
 {
-       const char *path = gui->tmp_path->get_text();
+       const char *path = gui->thread->tmp_path;
        if( access(path,R_OK+W_OK) ) return 0;
        struct statfs sfs;
        if( statfs(path, &sfs) ) return 0;
@@ -487,13 +579,15 @@ CreateBD_TmpPath::~CreateBD_TmpPath()
 
 int CreateBD_TmpPath::handle_event()
 {
+       get_text();
        gui->disk_space->update();
        return 1;
 }
 
 
 CreateBD_AssetTitle::CreateBD_AssetTitle(CreateBD_GUI *gui, int x, int y, int w)
- : BC_TextBox(x, y, w, 1, 0, gui->thread->asset_title, 1, MEDIUMFONT)
+ : BC_TextBox(x, y, w, 1, -(int)sizeof(gui->thread->asset_title),
+               gui->thread->asset_title, 1, MEDIUMFONT)
 {
        this->gui = gui;
 }
@@ -502,6 +596,11 @@ CreateBD_AssetTitle::~CreateBD_AssetTitle()
 {
 }
 
+int CreateBD_AssetTitle::handle_event()
+{
+       get_text();
+       return 1;
+}
 
 CreateBD_Deinterlace::CreateBD_Deinterlace(CreateBD_GUI *gui, int x, int y)
  : BC_CheckBox(x, y, &gui->thread->use_deinterlace, _("Deinterlace"))
@@ -543,17 +642,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"))
 {
@@ -595,17 +683,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)
@@ -616,14 +693,16 @@ 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;
+       standard = 0;
+       scale = 0;
        need_deinterlace = 0;
        need_inverse_telecine = 0;
-       need_scale = 0;
        need_resize_tracks = 0;
        need_histogram = 0;
+       non_standard = 0;
        need_wide_audio = 0;
-       need_wide_aspect = 0;
        need_label_chapters = 0;
        ok = 0;
        cancel = 0;
@@ -645,51 +724,59 @@ 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);
-       x0 +=  title->get_w() + padx;
-       media_size = new CreateBD_MediaSize(this, x0, y);
+       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();
-       x0 = x;
        y += disk_space->get_h() + pady/2;
-       title = new BC_Title(x0, y, _("Format:"), MEDIUMFONT, YELLOW);
+       title = new BC_Title(x, y, _("Format:"), MEDIUMFONT, YELLOW);
        add_subwindow(title);
-       x0 +=  title->get_w() + padx;
-       standard = new CreateBD_Format(this, x0, y);
+       standard = new CreateBD_Format(this, title->get_w() + padx, y);
        add_subwindow(standard);
        standard->create_objects();
+       standard->set_text(bd_formats[thread->use_standard].name);
+       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;
+       non_standard = new BC_Title(x, y+5, "", MEDIUMFONT, RED);
+       add_subwindow(non_standard);
+       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();
@@ -711,7 +798,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,
@@ -733,13 +821,12 @@ int CreateBD_GUI::close_event()
 
 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_scale->set_value(thread->use_scale);
        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_wide_aspect->set_value(thread->use_wide_aspect);
 //     need_label_chapters->set_value(thread->use_label_chapters);
 }
 
@@ -785,10 +872,9 @@ option_presets()
 {
 // reset only probed options
        use_deinterlace = 0;
-       use_scale = 0;
+       use_scale = Rescale::none;
        use_resize_tracks = 0;
        use_wide_audio = 0;
-       use_wide_aspect = 0;
        use_label_chapters = 0;
 
        if( !mwindow->edl ) return 1;
@@ -796,6 +882,13 @@ option_presets()
        bd_width = bd_formats[use_standard].w;
        bd_height = bd_formats[use_standard].h;
        bd_framerate = bd_formats[use_standard].framerate;
+       int wide = bd_formats[use_standard].wide;
+       bd_aspect_width  = wide < 0 ? 1. :
+               wide > 0 ? BD_WIDE_ASPECT_WIDTH  : BD_ASPECT_WIDTH;
+       bd_aspect_height = wide < 0 ? 1. :
+               wide > 0 ? BD_WIDE_ASPECT_HEIGHT : BD_ASPECT_HEIGHT;
+       bd_interlace_mode = bd_formats[use_standard].interlaced;
+       double bd_aspect = bd_aspect_width / bd_aspect_height;
 
        Tracks *tracks = mwindow->edl->tracks;
        max_w = 0;  max_h = 0;
@@ -810,19 +903,24 @@ 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( wide >= 0 && !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;
-                                               plugin;
-                                               plugin = (Plugin*)plugin->next) {
-                                       if( !strcmp(plugin->title, _("Deinterlace")) )
+                               for( Plugin *plugin = (Plugin*)trk->plugin_set[i]->first;
+                                               plugin; plugin=(Plugin*)plugin->next ) {
+                                       if( !strcmp(plugin->title, "Deinterlace") )
                                                has_deinterlace = 1;
-                                       if( !strcmp(plugin->title, _("Auto Scale")) ||
-                                           !strcmp(plugin->title, _("Scale")) )
+                                       if( !strcmp(plugin->title, "Auto Scale") ||
+                                           !strcmp(plugin->title, "Scale Ratio") ||
+                                           !strcmp(plugin->title, "Scale") )
                                                has_scale = 1;
                                }
                        }
@@ -830,8 +928,8 @@ option_presets()
                }
        }
        if( has_scale )
-               use_scale = 0;
-       if( use_scale ) {
+               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;
        }
@@ -847,12 +945,6 @@ option_presets()
        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 aw, ah;
-       MWindow::create_aspect_ratio(aw, ah, max_w, max_h);
-       if( aw == BD_WIDE_ASPECT_WIDTH && ah == BD_WIDE_ASPECT_HEIGHT )
-               use_wide_aspect = 1;
-       bd_aspect_width = use_wide_aspect ? BD_WIDE_ASPECT_WIDTH : BD_ASPECT_WIDTH;
-       bd_aspect_height = use_wide_aspect ? BD_WIDE_ASPECT_HEIGHT : BD_ASPECT_HEIGHT;
 
        if( tracks->recordable_audio_tracks() == BD_WIDE_CHANNELS )
                use_wide_audio = 1;
@@ -875,8 +967,20 @@ CreateBD_FormatItem::~CreateBD_FormatItem()
 
 int CreateBD_FormatItem::handle_event()
 {
-       popup->set_text(get_text());
+       const char *text = get_text();
+       int standard = this->standard;
+       if( standard < 0 ) {
+               BC_SubMenu *submenu = get_submenu();
+               CreateBD_FormatItem *sub_item0 =
+                       (CreateBD_FormatItem *)submenu->get_item(0);
+               standard = sub_item0->standard;
+               text = sub_item0->get_text();
+       }
        popup->gui->thread->use_standard = standard;
+       popup->set_text(text);
+       int n = strlen(text)-1;
+       int not_standard = n >= 0 && text[n] == '*' ? 1 : 0;
+       popup->gui->non_standard->update(not_standard ? _("* non-standard format") : "", 0);
        return popup->handle_event();
 }
 
@@ -893,8 +997,18 @@ CreateBD_Format::~CreateBD_Format()
 
 void CreateBD_Format::create_objects()
 {
+       BC_SubMenu *submenu = 0;
+       CreateBD_FormatItem *item = 0;
+       int ww = 0, hh = 0;
        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));
+               if( ww != bd_formats[i].w || hh != bd_formats[i].h ) {
+                       ww = bd_formats[i].w;  hh = bd_formats[i].h;
+                       char string[BCSTRLEN];
+                       sprintf(string, "%dx%d", ww,hh);
+                       add_item(item = new CreateBD_FormatItem(this, -1, string));
+                       item->add_submenu(submenu = new BC_SubMenu());
+               }
+               submenu->add_submenuitem(new CreateBD_FormatItem(this, i, bd_formats[i].name));
        }
 }
 
@@ -905,6 +1019,53 @@ int CreateBD_Format::handle_event()
        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)
 {