dvd/bd scaling fixes, es/de .po file updates
[goodguy/history.git] / cinelerra-5.1 / cinelerra / dvdcreate.C
index 05dde68b9a5db2d997755def98eec644e2aa42f8..c7477269e3642f26096c0cba11410bf1ff8dc475 100644 (file)
@@ -1,4 +1,5 @@
 #include "asset.h"
+#include "clip.h"
 #include "dvdcreate.h"
 #include "edl.h"
 #include "edit.h"
 
 // DVD Creation
 
+#define HD_1920x1080_2997      0
+#define HD_1920x1080_2500      1
+#define HD_1280x720_5994p      2
+#define HD_1280x720_5000p      3
+#define HD_720x576_5000p       4
+#define HD_720x576_2500                5
+#define HD_720x480_5994p       6
+#define HD_720x480_2997                7
+
+static struct hd_format {
+       const char *name;
+       int w, h;
+       double framerate;
+} hd_formats[] = {
+       { "1920x1080 29.97",      1920,1080, 29.97  },
+       { "1920x1080 25",         1920,1080, 25     },
+       { "1280x720 59.94p",      1280,720,  59.94  },
+       { "1280x720 50p",         1280,720,  50     },
+       { "720x576 50p(PAL)",      720,576,  50     },
+       { "720x576 25 (PAL)",      720,576,  25     },
+       { "720x480 59.94p(NTSC)",  720,480,  59.94  },
+       { "720x480 29.97 (NTSC)",  720,480,  29.97  },
+};
 
 const int64_t CreateDVD_Thread::DVD_SIZE = 4700000000;
 const int CreateDVD_Thread::DVD_STREAMS = 1;
@@ -61,14 +85,25 @@ CreateDVD_Thread::CreateDVD_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_histogram = 0;
+       this->use_inverse_telecine = 0;
        this->use_wide_audio = 0;
        this->use_wide_aspect = 0;
-       this->use_label_chapters = 0;
        this->use_ffmpeg = 0;
+       this->use_resize_tracks = 0;
+       this->use_label_chapters = 0;
+
+       this->dvd_size = DVD_SIZE;
+       this->dvd_width = DVD_WIDTH;
+       this->dvd_height = DVD_HEIGHT;
+       this->dvd_aspect_width = DVD_ASPECT_WIDTH;
+       this->dvd_aspect_height = DVD_ASPECT_HEIGHT;
+       this->dvd_framerate = DVD_FRAMERATE;
+       this->dvd_samplerate = DVD_SAMPLERATE;
+       this->dvd_max_bitrate = DVD_MAX_BITRATE;
+       this->dvd_kaudio_rate = DVD_KAUDIO_RATE;
+       this->max_w = this->max_h = 0;
 }
 
 CreateDVD_Thread::~CreateDVD_Thread()
@@ -112,12 +147,12 @@ int CreateDVD_Thread::create_dvd_jobs(ArrayList<BatchRenderJob*> *jobs,
 
        session->video_channels = DVD_STREAMS;
        session->video_tracks = DVD_STREAMS;
-       session->frame_rate = DVD_FRAMERATE;
-       session->output_w = DVD_WIDTH;
-       session->output_h = DVD_HEIGHT;
-       session->aspect_w = use_wide_aspect ? DVD_WIDE_ASPECT_WIDTH : DVD_ASPECT_WIDTH;
-       session->aspect_h = use_wide_aspect ? DVD_WIDE_ASPECT_HEIGHT : DVD_ASPECT_HEIGHT;
-       session->sample_rate = DVD_SAMPLERATE;
+       session->frame_rate = dvd_framerate;
+       session->output_w = dvd_width;
+       session->output_h = dvd_height;
+       session->aspect_w = dvd_aspect_width;
+       session->aspect_h = dvd_aspect_height;
+       session->sample_rate = dvd_samplerate;
        session->audio_channels = session->audio_tracks =
                use_wide_audio ? DVD_WIDE_CHANNELS : DVD_CHANNELS;
 
@@ -213,11 +248,12 @@ int CreateDVD_Thread::create_dvd_jobs(ArrayList<BatchRenderJob*> *jobs,
        edl->resample(old_samplerate, new_samplerate, TRACK_AUDIO);
        edl->resample(old_framerate, new_framerate, TRACK_VIDEO);
 
-       int64_t aud_size = ((DVD_KAUDIO_RATE * total_length)/8 + 1000-1) * 1000;
-       int64_t vid_size = DVD_SIZE*0.96 - aud_size;
+       int64_t aud_size = ((dvd_kaudio_rate * total_length)/8 + 1000-1) * 1000;
+       int64_t vid_size = dvd_size*0.96 - aud_size;
        int64_t vid_bitrate = (vid_size * 8) / total_length;
        vid_bitrate /= 1000;  vid_bitrate *= 1000;
-       if( vid_bitrate > DVD_MAX_BITRATE ) vid_bitrate = DVD_MAX_BITRATE;
+       if( vid_bitrate > dvd_max_bitrate )
+               vid_bitrate = dvd_max_bitrate;
 
        char xml_filename[BCTEXTLEN];
        sprintf(xml_filename, "%s/dvd.xml", asset_dir);
@@ -253,7 +289,7 @@ int CreateDVD_Thread::create_dvd_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 = DVD_KAUDIO_RATE * 1000;
+               asset->ff_audio_bitrate = dvd_kaudio_rate * 1000;
 
                asset->video_data = 1;
                strcpy(asset->vcodec, "dvd.dvd");
@@ -294,7 +330,7 @@ int CreateDVD_Thread::create_dvd_jobs(ArrayList<BatchRenderJob*> *jobs,
                asset->signed_ = 1;
                asset->header = 0;
                asset->dither = 0;
-               asset->ac3_bitrate = DVD_KAUDIO_RATE;
+               asset->ac3_bitrate = dvd_kaudio_rate;
        }
 
        job = new BatchRenderJob(mwindow->preferences);
@@ -324,8 +360,12 @@ void CreateDVD_Thread::handle_close_event(int result)
                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>", DVD_WIDTH, DVD_HEIGHT);
+               sprintf(data,"<SCALE TYPE=%d X_FACTOR=%f Y_FACTOR=%f "
+                       "WIDTH=%d HEIGHT=%d CONSTRAIN=0>",
+                       max_w >= dvd_width || max_h >= dvd_height ? 1 : 0,
+                       max_w > 0 ? (double)dvd_width/max_w : 1,
+                       max_h > 0 ? (double)dvd_height/max_h : 1,
+                       dvd_width, dvd_height);
                keyframe.set_data(data);
                insert_video_plugin("Scale", &keyframe);
        }
@@ -373,19 +413,50 @@ BC_Window* CreateDVD_Thread::new_gui()
                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_histogram = 0;
+       use_inverse_telecine = 0;
        use_wide_audio = 0;
        use_wide_aspect = 0;
-       use_label_chapters = 0;
        use_ffmpeg = 0;
+       use_resize_tracks = 0;
+       use_label_chapters = 0;
+       use_standard = HD_720x480_2997;
+
+       dvd_size = DVD_SIZE;
+       dvd_width = DVD_WIDTH;
+       dvd_height = DVD_HEIGHT;
+       dvd_aspect_width = DVD_ASPECT_WIDTH;
+       dvd_aspect_height = DVD_ASPECT_HEIGHT;
+       dvd_framerate = DVD_FRAMERATE;
+       dvd_samplerate = DVD_SAMPLERATE;
+       dvd_max_bitrate = DVD_MAX_BITRATE;
+       dvd_kaudio_rate = DVD_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(hd_formats)/sizeof(hd_formats[0])); ++i ) {
+                       if( !EQUIV(session->frame_rate, hd_formats[i].framerate) ) continue;
+                       if( session->output_w != hd_formats[i].w ) continue;
+                       if( session->output_h != hd_formats[i].h ) continue;
+                       has_standard = i;  break;
+               }
+               if( has_standard < 0 ) {
+// or use the default standard
+                       if( !strcmp(mwindow->default_standard, "NTSC") ) has_standard = HD_720x480_2997;
+                       else if( !strcmp(mwindow->default_standard, "PAL") ) has_standard = HD_720x576_2500;
+               }
+       }
+       use_standard = has_standard >= 0 ? has_standard : HD_720x480_2997;
+
        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 CreateDVD_GUI(this, x, y, w, h);
@@ -461,10 +532,18 @@ int64_t CreateDVD_DiskSpace::tmp_path_space()
 
 void CreateDVD_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 = 15e9, 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;
+       }
+       m = gui->thread->use_ffmpeg ? 2 : 3;
+       int color = disk_space < media_size*m ? RED : GREEN;
        int i = 0;
        for( int64_t space=disk_space; i<5 && (space/=1000)>0; disk_space=space, ++i );
        char text[BCTEXTLEN];
@@ -627,7 +706,6 @@ CreateDVD_GUI::CreateDVD_GUI(CreateDVD_Thread *thread, int x, int y, int w, int
        asset_title = 0;
        tmp_path = 0;
        disk_space = 0;
-       needed_disk_space = 15e9;
        need_deinterlace = 0;
        need_inverse_telecine = 0;
        need_scale = 0;
@@ -664,8 +742,26 @@ void CreateDVD_GUI::create_objects()
        y += title->get_h() + pady/2;
        disk_space = new CreateDVD_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 CreateDVD_MediaSize(this, x0, y);
+       media_size->create_objects();
+       media_sizes.append(new BC_ListBoxItem("4.7GB"));
+       media_sizes.append(new BC_ListBoxItem("8.3GB"));
+       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);
+       add_subwindow(title);
+       x0 +=  title->get_w() + padx;
+       standard = new CreateDVD_Format(this, x0, y);
+       add_subwindow(standard);
+       standard->create_objects();
+       y += standard->get_h() + pady/2;
        need_deinterlace = new CreateDVD_Deinterlace(this, x, y);
        add_subwindow(need_deinterlace);
        int x1 = x + 150, x2 = x1 + 150;
@@ -726,6 +822,19 @@ int CreateDVD_GUI::close_event()
        return 1;
 }
 
+void CreateDVD_GUI::update()
+{
+       need_deinterlace->set_value(thread->use_deinterlace);
+       need_inverse_telecine->set_value(thread->use_inverse_telecine);
+       need_scale->set_value(thread->use_scale);
+       need_use_ffmpeg->set_value(thread->use_ffmpeg);
+       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);
+}
+
 int CreateDVD_Thread::
 insert_video_plugin(const char *title, KeyFrame *default_keyframe)
 {
@@ -751,25 +860,14 @@ int CreateDVD_Thread::
 resize_tracks()
 {
        Tracks *tracks = mwindow->edl->tracks;
-       int max_w = 0, max_h = 0;
+       int trk_w = max_w, trk_h = max_h;
+       if( trk_w < dvd_width ) trk_w = dvd_width;
+       if( trk_h < dvd_height ) trk_h = dvd_height;
        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;
-               }
-       }
-
-       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;
 }
@@ -777,9 +875,22 @@ resize_tracks()
 int CreateDVD_Thread::
 option_presets()
 {
+// reset only probed options
+       use_deinterlace = 0;
+       use_scale = 0;
+       use_resize_tracks = 0;
+       use_wide_audio = 0;
+       use_wide_aspect = 0;
+       use_label_chapters = 0;
+
        if( !mwindow->edl ) return 1;
+
+       dvd_width = hd_formats[use_standard].w;
+       dvd_height = hd_formats[use_standard].h;
+       dvd_framerate = hd_formats[use_standard].framerate;
+
        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;
@@ -791,10 +902,10 @@ option_presets()
                                Indexable *indexable = edit->get_source();
                                int w = indexable->get_w();
                                if( w > max_w ) max_w = w;
-                               if( w != DVD_WIDTH ) use_scale = 1;
+                               if( w != dvd_width ) use_scale = 1;
                                int h = indexable->get_h();
                                if( h > max_h ) max_h = h;
-                               if( h != DVD_HEIGHT ) use_scale = 1;
+                               if( h != dvd_height ) use_scale = 1;
                        }
                        for( int i=0; i<trk->plugin_set.size(); ++i ) {
                                for(Plugin *plugin = (Plugin*)trk->plugin_set[i]->first;
@@ -813,8 +924,8 @@ option_presets()
        if( has_scale )
                use_scale = 0;
        if( use_scale ) {
-               if( max_w != DVD_WIDTH ) use_resize_tracks = 1;
-               if( max_h != DVD_HEIGHT ) use_resize_tracks = 1;
+               if( max_w != dvd_width ) use_resize_tracks = 1;
+               if( max_h != dvd_height ) use_resize_tracks = 1;
        }
        for( Track *trk=tracks->first; trk && !use_resize_tracks; trk=trk->next ) {
                if( !trk->record ) continue;
@@ -825,15 +936,82 @@ option_presets()
                        break;
                }
        }
-       if( !has_deinterlace && max_h > 2*DVD_HEIGHT ) use_deinterlace = 1;
+       if( !has_deinterlace && max_h > 2*dvd_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 == DVD_WIDE_ASPECT_WIDTH && h == DVD_WIDE_ASPECT_HEIGHT )
+
+       float aw, ah;
+       MWindow::create_aspect_ratio(aw, ah, max_w, max_h);
+       if( aw == DVD_WIDE_ASPECT_WIDTH && ah == DVD_WIDE_ASPECT_HEIGHT )
                use_wide_aspect = 1;
+       dvd_aspect_width = use_wide_aspect ? DVD_WIDE_ASPECT_WIDTH : DVD_ASPECT_WIDTH;
+       dvd_aspect_height = use_wide_aspect ? DVD_WIDE_ASPECT_HEIGHT : DVD_ASPECT_HEIGHT;
+
        if( tracks->recordable_audio_tracks() == DVD_WIDE_CHANNELS )
                use_wide_audio = 1;
+
        return 0;
 }
 
+
+
+CreateDVD_FormatItem::CreateDVD_FormatItem(CreateDVD_Format *popup,
+               int standard, const char *name)
+ : BC_MenuItem(name)
+{
+       this->popup = popup;
+       this->standard = standard;
+}
+
+CreateDVD_FormatItem::~CreateDVD_FormatItem()
+{
+}
+
+int CreateDVD_FormatItem::handle_event()
+{
+       popup->set_text(get_text());
+       popup->gui->thread->use_standard = standard;
+       return popup->handle_event();
+}
+
+
+CreateDVD_Format::CreateDVD_Format(CreateDVD_GUI *gui, int x, int y)
+ : BC_PopupMenu(x, y, 180, hd_formats[gui->thread->use_standard].name, 1)
+{
+       this->gui = gui;
+}
+
+CreateDVD_Format::~CreateDVD_Format()
+{
+}
+
+void CreateDVD_Format::create_objects()
+{
+       for( int i=0; i<(int)(sizeof(hd_formats)/sizeof(hd_formats[0])); ++i ) {
+               add_item(new CreateDVD_FormatItem(this, i, hd_formats[i].name));
+       }
+}
+
+int CreateDVD_Format::handle_event()
+{
+       gui->thread->option_presets();
+       gui->update();
+       return 1;
+}
+
+CreateDVD_MediaSize::CreateDVD_MediaSize(CreateDVD_GUI *gui, int x, int y)
+ : BC_PopupTextBox(gui, 0, 0, x, y, 70,50)
+{
+       this->gui = gui;
+}
+
+CreateDVD_MediaSize::~CreateDVD_MediaSize()
+{
+}
+
+int CreateDVD_MediaSize::handle_event()
+{
+       gui->disk_space->update();
+       return 1;
+}
+