--- trunk/xvidcore/vfw/src/config.c 2005/07/11 09:25:39 1625 +++ trunk/xvidcore/vfw/src/config.c 2005/10/22 22:32:44 1650 @@ -104,7 +104,7 @@ /* MPEG-4 PROFILES/LEVELS ============================================================== */ /* ===================================================================================== */ -#define DXN_PROFILES +#define EXTRA_PROFILES /* default vbv_occupancy is (64/170)*vbv_buffer_size */ @@ -130,29 +130,38 @@ { "ARTS @ L4", 0x94, 352, 288, 30, 16, 792, 396, 11880, 100, 80*16368, 16384, 2000000, 0, -1, PROFILE_ARTS }, #endif - { "AS @ L0", 0xf0, 176, 144, 30, 1, 297, 99, 2970, 100, 10*16368, 2048, 128000, 0, -1, PROFILE_AS }, - { "AS @ L1", 0xf1, 176, 144, 30, 4, 297, 99, 2970, 100, 10*16368, 2048, 128000, 0, -1, PROFILE_AS }, - { "AS @ L2", 0xf2, 352, 288, 15, 4, 1188, 396, 5940, 100, 40*16368, 4096, 384000, 0, -1, PROFILE_AS }, - { "AS @ L3", 0xf3, 352, 288, 30, 4, 1188, 396, 11880, 100, 40*16368, 4096, 768000, 0, -1, PROFILE_AS }, + { "Advanced Simple @ L0", 0xf0, 176, 144, 30, 1, 297, 99, 2970, 100, 10*16368, 2048, 128000, 0, -1, PROFILE_AS }, + { "Advanced Simple @ L1", 0xf1, 176, 144, 30, 4, 297, 99, 2970, 100, 10*16368, 2048, 128000, 0, -1, PROFILE_AS }, + { "Advanced Simple @ L2", 0xf2, 352, 288, 15, 4, 1188, 396, 5940, 100, 40*16368, 4096, 384000, 0, -1, PROFILE_AS }, + { "Advanced Simple @ L3", 0xf3, 352, 288, 30, 4, 1188, 396, 11880, 100, 40*16368, 4096, 768000, 0, -1, PROFILE_AS }, /* ISMA Profile 1, (ASP) @ L3b (CIF, 1.5 Mb/s) CIF(352x288), 30fps, 1.5Mbps max ??? */ - { "AS @ L4", 0xf4, 352, 576, 30, 4, 2376, 792, 23760, 50, 80*16368, 8192, 3000000, 0, -1, PROFILE_AS }, - { "AS @ L5", 0xf5, 720, 576, 30, 4, 4860, 1620, 48600, 25, 112*16368, 16384, 8000000, 0, -1, PROFILE_AS }, + { "Advanced Simple @ L4", 0xf4, 352, 576, 30, 4, 2376, 792, 23760, 50, 80*16368, 8192, 3000000, 0, -1, PROFILE_AS }, + { "Advanced Simple @ L5", 0xf5, 720, 576, 30, 4, 4860, 1620, 48600, 25, 112*16368, 16384, 8000000, 0, -1, PROFILE_AS }, -#ifdef DXN_PROFILES -// information provided by DivXNetworks, USA. -// "DivX Certified Profile Compatibility v1.1", February 2005 - { "DXN Handheld", 0x00, 176, 144, 15, 1, 198, 99, 1485, 100, 32*8192, -1, 537600, 800000, 0, PROFILE_ADAPTQUANT|PROFILE_DXN }, - { "DXN Portable NTSC",0x00, 352, 240, 30, 1, 990, 330, 36000, 100, 384*8192, -1, 4854000, 8000000, 1, PROFILE_4MV|PROFILE_ADAPTQUANT|PROFILE_BVOP|PROFILE_DXN }, - { "DXN Portable PAL", 0x00, 352, 288, 25, 1, 1188, 396, 36000, 100, 384*8192, -1, 4854000, 8000000, 1, PROFILE_4MV|PROFILE_ADAPTQUANT|PROFILE_BVOP|PROFILE_DXN }, - { "DXN HT NTSC", 0x00, 720, 480, 30, 1, 4050, 1350, 40500, 100, 384*8192, -1, 4854000, 8000000, 1, PROFILE_4MV|PROFILE_ADAPTQUANT|PROFILE_BVOP|PROFILE_INTERLACE|PROFILE_DXN }, - { "DXN HT PAL", 0x00, 720, 576, 25, 1, 4860, 1620, 40500, 100, 384*8192, -1, 4854000, 8000000, 1, PROFILE_4MV|PROFILE_ADAPTQUANT|PROFILE_BVOP|PROFILE_INTERLACE|PROFILE_DXN }, - { "DXN HDTV", 0x00, 1280, 720, 30, 1,10800, 3600, 108000, 100, 768*8192, -1, 9708400, 16000000, 2, PROFILE_4MV|PROFILE_ADAPTQUANT|PROFILE_BVOP|PROFILE_INTERLACE|PROFILE_DXN }, +#ifdef EXTRA_PROFILES + { "Handheld", 0x00, 176, 144, 15, 1, 198, 99, 1485, 100, 32*8192, -1, 537600, 800000, 0, PROFILE_ADAPTQUANT|PROFILE_EXTRA }, + { "Portable NTSC", 0x00, 352, 240, 30, 1, 990, 330, 36000, 100, 384*8192, -1, 4854000, 8000000, 1, PROFILE_4MV|PROFILE_ADAPTQUANT|PROFILE_BVOP|PROFILE_EXTRA }, + { "Portable PAL", 0x00, 352, 288, 25, 1, 1188, 396, 36000, 100, 384*8192, -1, 4854000, 8000000, 1, PROFILE_4MV|PROFILE_ADAPTQUANT|PROFILE_BVOP|PROFILE_EXTRA }, + { "Home Theatre NTSC", 0x00, 720, 480, 30, 1, 4050, 1350, 40500, 100, 384*8192, -1, 4854000, 8000000, 1, PROFILE_4MV|PROFILE_ADAPTQUANT|PROFILE_BVOP|PROFILE_INTERLACE|PROFILE_EXTRA }, + { "Home Theatre PAL", 0x00, 720, 576, 25, 1, 4860, 1620, 40500, 100, 384*8192, -1, 4854000, 8000000, 1, PROFILE_4MV|PROFILE_ADAPTQUANT|PROFILE_BVOP|PROFILE_INTERLACE|PROFILE_EXTRA }, + { "Cinema Plus NTSC", 0x00, 720, 480, 30, 1, 4050, 1350, 40500, 100, 384*8192, -1, 4854000, 8000000, 3, PROFILE_4MV|PROFILE_ADAPTQUANT|PROFILE_BVOP|PROFILE_MPEGQUANT|PROFILE_INTERLACE|PROFILE_QPEL|PROFILE_EXTRA }, + { "Cinema Plus PAL", 0x00, 720, 576, 25, 1, 4860, 1620, 40500, 100, 384*8192, -1, 4854000, 8000000, 3, PROFILE_4MV|PROFILE_ADAPTQUANT|PROFILE_BVOP|PROFILE_MPEGQUANT|PROFILE_INTERLACE|PROFILE_QPEL|PROFILE_EXTRA }, + { "HDTV", 0x00, 1280, 720, 30, 1,10800, 3600, 108000, 100, 768*8192, -1, 9708400, 16000000, 2, PROFILE_4MV|PROFILE_ADAPTQUANT|PROFILE_BVOP|PROFILE_INTERLACE|PROFILE_EXTRA }, #endif - { "(unrestricted)", 0x00, 0, 0, 0, 0, 0, 0, 0, 100, 0*16368, -1, 0, 0, -1, 0xffffffff & ~PROFILE_DXN }, + { "(unrestricted)", 0x00, 0, 0, 0, 0, 0, 0, 0, 100, 0*16368, -1, 0, 0, -1, 0xffffffff & ~PROFILE_EXTRA }, }; +const quality_t quality_table[] = +{ + /* name | m vhq bf cme tbo kfi fdr | iquant pquant bquant trellis */ + { "Real-time", 1, 0, 0, 0, 0, 300, 0, 1, 31, 1, 31, 1, 31, 0 }, + { QUALITY_GENERAL_STRING, 6, 1, 0, 1, 0, 300, 0, 1, 31, 1, 31, 1, 31, 1 }, +}; + +const int quality_table_num = sizeof(quality_table)/sizeof(quality_t); + typedef struct { char * name; float value; @@ -261,22 +270,22 @@ {"audio_size", ®.audio_size, 0}, /* motion */ - {"motion_search", ®.motion_search, 6}, - {"vhq_mode", ®.vhq_mode, 1}, - {"vhq_bframe", ®.vhq_bframe, 0}, - {"chromame", ®.chromame, 1}, - {"turbo", ®.turbo, 0}, - {"max_key_interval", ®.max_key_interval, 300}, - {"frame_drop_ratio", ®.frame_drop_ratio, 0}, + {"motion_search", ®.quality_user.motion_search, 6}, + {"vhq_mode", ®.quality_user.vhq_mode, 1}, + {"vhq_bframe", ®.quality_user.vhq_bframe, 0}, + {"chromame", ®.quality_user.chromame, 1}, + {"turbo", ®.quality_user.turbo, 0}, + {"max_key_interval", ®.quality_user.max_key_interval, 300}, + {"frame_drop_ratio", ®.quality_user.frame_drop_ratio, 0}, /* quant */ - {"min_iquant", ®.min_iquant, 1}, - {"max_iquant", ®.max_iquant, 31}, - {"min_pquant", ®.min_pquant, 1}, - {"max_pquant", ®.max_pquant, 31}, - {"min_bquant", ®.min_bquant, 1}, - {"max_bquant", ®.max_bquant, 31}, - {"trellis_quant", ®.trellis_quant, 1}, + {"min_iquant", ®.quality_user.min_iquant, 1}, + {"max_iquant", ®.quality_user.max_iquant, 31}, + {"min_pquant", ®.quality_user.min_pquant, 1}, + {"max_pquant", ®.quality_user.max_pquant, 31}, + {"min_bquant", ®.quality_user.min_bquant, 1}, + {"max_bquant", ®.quality_user.max_bquant, 31}, + {"trellis_quant", ®.quality_user.trellis_quant, 1}, /* debug */ {"fourcc_used", ®.fourcc_used, 0}, @@ -296,6 +305,7 @@ static const REG_STR reg_strs[] = { {"profile", reg.profile_name, "(unrestricted)"}, + {"quality", reg.quality_name, QUALITY_GENERAL_STRING}, {"stats", reg.stats, CONFIG_2PASS_FILE}, }; @@ -383,6 +393,7 @@ } } + /* find profile table index */ reg.profile = 0; for (i=0; imotion_search, 0); - SendDlgItemMessage(hDlg, IDC_VHQ, CB_SETCURSEL, config->vhq_mode, 0); - CheckDlg(hDlg, IDC_VHQ_BFRAME, config->vhq_bframe); - CheckDlg(hDlg, IDC_CHROMAME, config->chromame); - CheckDlg(hDlg, IDC_TURBO, config->turbo); - SetDlgItemInt(hDlg, IDC_FRAMEDROP, config->frame_drop_ratio, FALSE); - SetDlgItemInt(hDlg, IDC_MAXKEY, config->max_key_interval, FALSE); + { + const int userdef = (config->quality==quality_table_num); + const quality_t* quality_preset = userdef ? &config->quality_user : &quality_table[config->quality]; + + SendDlgItemMessage(hDlg, IDC_MOTION, CB_SETCURSEL, quality_preset->motion_search, 0); + SendDlgItemMessage(hDlg, IDC_VHQ, CB_SETCURSEL, quality_preset->vhq_mode, 0); + CheckDlg(hDlg, IDC_VHQ_BFRAME, quality_preset->vhq_bframe); + CheckDlg(hDlg, IDC_CHROMAME, quality_preset->chromame); + CheckDlg(hDlg, IDC_TURBO, quality_preset->turbo); + SetDlgItemInt(hDlg, IDC_FRAMEDROP, quality_preset->frame_drop_ratio, FALSE); + SetDlgItemInt(hDlg, IDC_MAXKEY, quality_preset->max_key_interval, FALSE); + + EnableDlgWindow(hDlg, IDC_MOTION, userdef); + EnableDlgWindow(hDlg, IDC_VHQ, userdef); + EnableDlgWindow(hDlg, IDC_VHQ_BFRAME, userdef); + EnableDlgWindow(hDlg, IDC_CHROMAME, userdef); + EnableDlgWindow(hDlg, IDC_TURBO, userdef); + EnableDlgWindow(hDlg, IDC_FRAMEDROP, userdef); + EnableDlgWindow(hDlg, IDC_MAXKEY, userdef); break; + } case IDD_QUANT : - SetDlgItemInt(hDlg, IDC_MINIQUANT, config->min_iquant, FALSE); - SetDlgItemInt(hDlg, IDC_MAXIQUANT, config->max_iquant, FALSE); - SetDlgItemInt(hDlg, IDC_MINPQUANT, config->min_pquant, FALSE); - SetDlgItemInt(hDlg, IDC_MAXPQUANT, config->max_pquant, FALSE); - SetDlgItemInt(hDlg, IDC_MINBQUANT, config->min_bquant, FALSE); - SetDlgItemInt(hDlg, IDC_MAXBQUANT, config->max_bquant, FALSE); - CheckDlg(hDlg, IDC_TRELLISQUANT, config->trellis_quant); + { + const int userdef = (config->quality==quality_table_num); + const quality_t* quality_preset = userdef ? &config->quality_user : &quality_table[config->quality]; + + SetDlgItemInt(hDlg, IDC_MINIQUANT, quality_preset->min_iquant, FALSE); + SetDlgItemInt(hDlg, IDC_MAXIQUANT, quality_preset->max_iquant, FALSE); + SetDlgItemInt(hDlg, IDC_MINPQUANT, quality_preset->min_pquant, FALSE); + SetDlgItemInt(hDlg, IDC_MAXPQUANT, quality_preset->max_pquant, FALSE); + SetDlgItemInt(hDlg, IDC_MINBQUANT, quality_preset->min_bquant, FALSE); + SetDlgItemInt(hDlg, IDC_MAXBQUANT, quality_preset->max_bquant, FALSE); + CheckDlg(hDlg, IDC_TRELLISQUANT, quality_preset->trellis_quant); + + EnableDlgWindow(hDlg, IDC_MINIQUANT, userdef); + EnableDlgWindow(hDlg, IDC_MAXIQUANT, userdef); + EnableDlgWindow(hDlg, IDC_MINPQUANT, userdef); + EnableDlgWindow(hDlg, IDC_MAXPQUANT, userdef); + EnableDlgWindow(hDlg, IDC_MINBQUANT, userdef); + EnableDlgWindow(hDlg, IDC_MAXBQUANT, userdef); + EnableDlgWindow(hDlg, IDC_TRELLISQUANT, userdef); break; + } - case IDD_DEBUG : + case IDD_COMMON : CheckDlg(hDlg, IDC_CPU_MMX, (config->cpu & XVID_CPU_MMX)); CheckDlg(hDlg, IDC_CPU_MMXEXT, (config->cpu & XVID_CPU_MMXEXT)); CheckDlg(hDlg, IDC_CPU_SSE, (config->cpu & XVID_CPU_SSE)); @@ -1213,11 +1260,12 @@ CheckRadioButton(hDlg, IDC_CPU_AUTO, IDC_CPU_FORCE, config->cpu & XVID_CPU_FORCE ? IDC_CPU_FORCE : IDC_CPU_AUTO ); + set_dlgitem_hex(hDlg, IDC_DEBUG, config->debug); + break; + case IDD_ENC: SetDlgItemInt(hDlg, IDC_NUMTHREADS, config->num_threads, FALSE); - SendDlgItemMessage(hDlg, IDC_FOURCC, CB_SETCURSEL, config->fourcc_used, 0); - set_dlgitem_hex(hDlg, IDC_DEBUG, config->debug); CheckDlg(hDlg, IDC_VOPDEBUG, config->vop_debug); CheckDlg(hDlg, IDC_DISPLAY_STATUS, config->display_status); break; @@ -1365,36 +1413,40 @@ break; case IDD_MOTION : - config->motion_search = SendDlgItemMessage(hDlg, IDC_MOTION, CB_GETCURSEL, 0, 0); - config->vhq_mode = SendDlgItemMessage(hDlg, IDC_VHQ, CB_GETCURSEL, 0, 0); - config->vhq_bframe = IsDlgButtonChecked(hDlg, IDC_VHQ_BFRAME); - config->chromame = IsDlgChecked(hDlg, IDC_CHROMAME); - config->turbo = IsDlgChecked(hDlg, IDC_TURBO); + if (config->quality==quality_table_num) { + config->quality_user.motion_search = SendDlgItemMessage(hDlg, IDC_MOTION, CB_GETCURSEL, 0, 0); + config->quality_user.vhq_mode = SendDlgItemMessage(hDlg, IDC_VHQ, CB_GETCURSEL, 0, 0); + config->quality_user.vhq_bframe = IsDlgButtonChecked(hDlg, IDC_VHQ_BFRAME); + config->quality_user.chromame = IsDlgChecked(hDlg, IDC_CHROMAME); + config->quality_user.turbo = IsDlgChecked(hDlg, IDC_TURBO); - config->frame_drop_ratio = config_get_uint(hDlg, IDC_FRAMEDROP, config->frame_drop_ratio); + config->quality_user.frame_drop_ratio = config_get_uint(hDlg, IDC_FRAMEDROP, config->quality_user.frame_drop_ratio); - config->max_key_interval = config_get_uint(hDlg, IDC_MAXKEY, config->max_key_interval); + config->quality_user.max_key_interval = config_get_uint(hDlg, IDC_MAXKEY, config->quality_user.max_key_interval); + } break; case IDD_QUANT : - config->min_iquant = config_get_uint(hDlg, IDC_MINIQUANT, config->min_iquant); - config->max_iquant = config_get_uint(hDlg, IDC_MAXIQUANT, config->max_iquant); - config->min_pquant = config_get_uint(hDlg, IDC_MINPQUANT, config->min_pquant); - config->max_pquant = config_get_uint(hDlg, IDC_MAXPQUANT, config->max_pquant); - config->min_bquant = config_get_uint(hDlg, IDC_MINBQUANT, config->min_bquant); - config->max_bquant = config_get_uint(hDlg, IDC_MAXBQUANT, config->max_bquant); - - CONSTRAINVAL(config->min_iquant, 1, 31); - CONSTRAINVAL(config->max_iquant, config->min_iquant, 31); - CONSTRAINVAL(config->min_pquant, 1, 31); - CONSTRAINVAL(config->max_pquant, config->min_pquant, 31); - CONSTRAINVAL(config->min_bquant, 1, 31); - CONSTRAINVAL(config->max_bquant, config->min_bquant, 31); + if (config->quality==quality_table_num) { + config->quality_user.min_iquant = config_get_uint(hDlg, IDC_MINIQUANT, config->quality_user.min_iquant); + config->quality_user.max_iquant = config_get_uint(hDlg, IDC_MAXIQUANT, config->quality_user.max_iquant); + config->quality_user.min_pquant = config_get_uint(hDlg, IDC_MINPQUANT, config->quality_user.min_pquant); + config->quality_user.max_pquant = config_get_uint(hDlg, IDC_MAXPQUANT, config->quality_user.max_pquant); + config->quality_user.min_bquant = config_get_uint(hDlg, IDC_MINBQUANT, config->quality_user.min_bquant); + config->quality_user.max_bquant = config_get_uint(hDlg, IDC_MAXBQUANT, config->quality_user.max_bquant); + + CONSTRAINVAL(config->quality_user.min_iquant, 1, 31); + CONSTRAINVAL(config->quality_user.max_iquant, config->quality_user.min_iquant, 31); + CONSTRAINVAL(config->quality_user.min_pquant, 1, 31); + CONSTRAINVAL(config->quality_user.max_pquant, config->quality_user.min_pquant, 31); + CONSTRAINVAL(config->quality_user.min_bquant, 1, 31); + CONSTRAINVAL(config->quality_user.max_bquant, config->quality_user.min_bquant, 31); - config->trellis_quant = IsDlgChecked(hDlg, IDC_TRELLISQUANT); + config->quality_user.trellis_quant = IsDlgChecked(hDlg, IDC_TRELLISQUANT); + } break; - case IDD_DEBUG : + case IDD_COMMON : config->cpu = 0; config->cpu |= IsDlgChecked(hDlg, IDC_CPU_MMX) ? XVID_CPU_MMX : 0; config->cpu |= IsDlgChecked(hDlg, IDC_CPU_MMXEXT) ? XVID_CPU_MMXEXT : 0; @@ -1403,11 +1455,12 @@ config->cpu |= IsDlgChecked(hDlg, IDC_CPU_3DNOW) ? XVID_CPU_3DNOW : 0; config->cpu |= IsDlgChecked(hDlg, IDC_CPU_3DNOWEXT) ? XVID_CPU_3DNOWEXT : 0; config->cpu |= IsDlgChecked(hDlg, IDC_CPU_FORCE) ? XVID_CPU_FORCE : 0; + config->debug = get_dlgitem_hex(hDlg, IDC_DEBUG, config->debug); + break; + case IDD_ENC : config->num_threads = config_get_uint(hDlg, IDC_NUMTHREADS, config->num_threads); - config->fourcc_used = SendDlgItemMessage(hDlg, IDC_FOURCC, CB_GETCURSEL, 0, 0); - config->debug = get_dlgitem_hex(hDlg, IDC_DEBUG, config->debug); config->vop_debug = IsDlgChecked(hDlg, IDC_VOPDEBUG); config->display_status = IsDlgChecked(hDlg, IDC_DISPLAY_STATUS); break; @@ -1813,6 +1866,7 @@ SendDlgItemMessage(hDlg, IDC_PROFILE, CB_SETCURSEL, config->profile, 0); SendDlgItemMessage(hDlg, IDC_MODE, CB_SETCURSEL, config->mode, 0); + SendDlgItemMessage(hDlg, IDC_QUALITY, CB_SETCURSEL, config->quality, 0); g_use_bitrate = config->use_2pass_bitrate; @@ -1833,6 +1887,7 @@ { config->profile = SendDlgItemMessage(hDlg, IDC_PROFILE, CB_GETCURSEL, 0, 0); config->mode = SendDlgItemMessage(hDlg, IDC_MODE, CB_GETCURSEL, 0, 0); + config->quality = SendDlgItemMessage(hDlg, IDC_QUALITY, CB_GETCURSEL, 0, 0); if (g_use_bitrate) { config->bitrate = config_get_uint(hDlg, IDC_BITRATE, config->bitrate); @@ -1852,8 +1907,8 @@ static const int pass2_dlgs[] = { IDD_RC_2PASS2 }; static const int bitrate_dlgs[] = { IDD_BITRATE }; static const int zone_dlgs[] = { IDD_ZONE }; -static const int decoder_dlgs[] = { IDD_DEC }; -static const int adv_dlgs[] = { IDD_MOTION, IDD_QUANT, IDD_DEBUG}; +static const int quality_dlgs[] = { IDD_MOTION, IDD_QUANT }; +static const int other_dlgs[] = { IDD_ENC, IDD_DEC, IDD_COMMON }; BOOL CALLBACK main_proc(HWND hDlg, UINT uMsg, WPARAM wParam, LPARAM lParam) @@ -1877,6 +1932,10 @@ SendDlgItemMessage(hDlg, IDC_MODE, CB_ADDSTRING, 0, (LPARAM)"Null test speed"); #endif + for (i=0; i= 0x0300) SendMessage(g_hTooltip, TTM_SETMAXTIPWIDTH, 0, 400); +#endif EnumChildWindows(hDlg, enum_tooltips, 0); } @@ -1894,7 +1955,12 @@ { DWORD ext_style = ListView_GetExtendedListViewStyle(GetDlgItem(hDlg,IDC_ZONES)); - ext_style |= LVS_EX_FULLROWSELECT | LVS_EX_FLATSB ; +#if (_WIN32_IE >= 0x0300) + ext_style |= LVS_EX_FULLROWSELECT; +#endif +#if( _WIN32_IE >= 0x0400 ) + ext_style |= LVS_EX_FLATSB ; +#endif ListView_SetExtendedListViewStyle(GetDlgItem(hDlg,IDC_ZONES), ext_style); } @@ -1988,9 +2054,9 @@ main_upload(hDlg, config); break; - case IDC_DECODER : + case IDC_OTHER : main_download(hDlg, config); - adv_dialog(hDlg, config, decoder_dlgs, sizeof(decoder_dlgs)/sizeof(int)); + adv_dialog(hDlg, config, other_dlgs, sizeof(other_dlgs)/sizeof(int)); main_mode(hDlg, config); break; @@ -2064,9 +2130,23 @@ } break; - case IDC_ADVANCED : + case IDC_QUALITY_ADV : main_download(hDlg, config); - adv_dialog(hDlg, config, adv_dlgs, sizeof(adv_dlgs)/sizeof(int)); + + if (config->quality < quality_table_num) { + int result = MessageBox(hDlg, + "The built-in quality presets are read-only. Would you like to copy the values\n" + "of the selected preset into the \"" QUALITY_USER_STRING "\" preset for editing?", + "Question", MB_YESNOCANCEL|MB_DEFBUTTON2|MB_ICONQUESTION); + + if (result==0 || result==IDCANCEL) break; + if (result==IDYES) { + memcpy(&config->quality_user, &quality_table[config->quality], sizeof(quality_t)); + config->quality = quality_table_num; + } + } + adv_dialog(hDlg, config, quality_dlgs, sizeof(quality_dlgs)/sizeof(int)); + SendDlgItemMessage(hDlg, IDC_QUALITY, CB_SETCURSEL, config->quality, 0); break; case IDC_DEFAULTS :