[cvs] / xvidcore / vfw / src / config.c Repository:
ViewVC logotype

Diff of /xvidcore/vfw/src/config.c

Parent Directory Parent Directory | Revision Log Revision Log | View Patch Patch

revision 1.1, Sat Feb 22 08:24:01 2003 UTC revision 1.1.2.40, Sat Mar 6 11:39:11 2004 UTC
# Line 0  Line 1 
1    /**************************************************************************
2     *
3     *      XVID VFW FRONTEND
4     *      config
5     *
6     *      This program is free software; you can redistribute it and/or modify
7     *      it under the terms of the GNU General Public License as published by
8     *      the Free Software Foundation; either version 2 of the License, or
9     *      (at your option) any later version.
10     *
11     *      This program is distributed in the hope that it will be useful,
12     *      but WITHOUT ANY WARRANTY; without even the implied warranty of
13     *      MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14     *      GNU General Public License for more details.
15     *
16     *      You should have received a copy of the GNU General Public License
17     *      along with this program; if not, write to the Free Software
18     *      Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
19     *
20     *************************************************************************/
21    
22    /**************************************************************************
23     *
24     *      History:
25     *
26     *      15.06.2002      added bframes options
27     *      21.04.2002      fixed custom matrix support, tried to get dll size down
28     *      17.04.2002      re-enabled lumi masking in 1st pass
29     *      15.04.2002      updated cbr support
30     *      07.04.2002      min keyframe interval checkbox
31     *                              2-pass max bitrate and overflow customization
32     *      04.04.2002      interlacing support
33     *                              hinted ME support
34     *      24.03.2002      daniel smith <danielsmith@astroboymail.com>
35     *                              added Foxer's new CBR engine
36     *                              - cbr_buffer is being used as reaction delay (quick hack)
37     *      23.03.2002      daniel smith <danielsmith@astroboymail.com>
38     *                              added load defaults button
39     *                              merged foxer's alternative 2-pass code (2-pass alt tab)
40     *                              added proper tooltips
41     *                              moved registry data into reg_ints/reg_strs arrays
42     *                              added DEBUGERR output on errors instead of returning
43     *      16.03.2002      daniel smith <danielsmith@astroboymail.com>
44     *                              rewrote/restructured most of file
45     *                              added tooltips (kind of - dirty message hook method)
46     *                              split tabs into a main dialog / advanced prop sheet
47     *                              advanced controls are now enabled/disabled by mode
48     *                              added modulated quantization, DX50 fourcc
49     *      11.03.2002  Min Chen <chenm001@163.com>
50     *                        now get Core Version use xvid_init()
51     *      05.03.2002  Min Chen <chenm001@163.com>
52     *                              Add Core version display to about box
53     *      01.12.2001      inital version; (c)2001 peter ross <pross@xvid.org>
54     *
55     *************************************************************************/
56    
57    
58    #include <windows.h>
59    #include <commctrl.h>
60    #include <stdio.h>  /* sprintf */
61    #include <xvid.h>       /* XviD API */
62    
63    #include "debug.h"
64    #include "config.h"
65    #include "resource.h"
66    
67    
68    #define CONSTRAINVAL(X,Y,Z) if((X)<(Y)) X=Y; if((X)>(Z)) X=Z;
69    #define IsDlgChecked(hwnd,idc)  (IsDlgButtonChecked(hwnd,idc) == BST_CHECKED)
70    #define CheckDlg(hwnd,idc,value) CheckDlgButton(hwnd,idc, value?BST_CHECKED:BST_UNCHECKED)
71    #define EnableDlgWindow(hwnd,idc,state) EnableWindow(GetDlgItem(hwnd,idc),state)
72    
73    static void zones_update(HWND hDlg, CONFIG * config);
74    
75    HINSTANCE g_hInst;
76    HWND g_hTooltip;
77    
78    static int g_use_bitrate = 1;
79    
80    
81    int pp_dy, pp_duv, pp_dr, pp_fe; /* decoder options */
82    
83    /* enumerates child windows, assigns tooltips */
84    BOOL CALLBACK enum_tooltips(HWND hWnd, LPARAM lParam)
85    {
86            char help[500];
87    
88            if (LoadString(g_hInst, GetDlgCtrlID(hWnd), help, 500))
89            {
90                    TOOLINFO ti;
91                    ti.cbSize = sizeof(TOOLINFO);
92                    ti.uFlags = TTF_SUBCLASS | TTF_IDISHWND;
93                    ti.hwnd = GetParent(hWnd);
94                    ti.uId  = (LPARAM)hWnd;
95                    ti.lpszText = help;
96                    SendMessage(g_hTooltip, TTM_ADDTOOL, 0, (LPARAM)&ti);
97            }
98    
99            return TRUE;
100    }
101    
102    
103    /* ===================================================================================== */
104    /* MPEG-4 PROFILES/LEVELS ============================================================== */
105    /* ===================================================================================== */
106    
107    
108    
109    /* default vbv_occupancy is (64/170)*vbv_buffer_size */
110    
111    const profile_t profiles[] =
112    {
113    /*      name                             p@l,   w       h  fps  obj  Tvmv  vmv  vcv   ac%        vbv      pkt   kbps  flags */
114            { "Simple @ L0",           0x08,  176, 144, 15,  1,  198,   99,   1485, 100,  10*16368,  2048,   64, 0 },
115            /* simple@l0: max f_code=1, intra_dc_vlc_threshold=0 */
116            /* if ac preidition is used, adaptive quantization must not be used */
117            /* <=qcif must be used */
118            { "Simple @ L1",           0x01,  176, 144, 15,  4,  198,   99,   1485, 100,  10*16368,  2048,   64, PROFILE_ADAPTQUANT },
119            { "Simple @ L2",           0x02,  352, 288, 15,  4,  792,  396,   5940, 100,  40*16368,  4096,  128, PROFILE_ADAPTQUANT },
120            { "Simple @ L3",           0x03,  352, 288, 15,  4,  792,  396,  11880, 100,  40*16368,  8192,  384, PROFILE_ADAPTQUANT },
121    
122            { "ARTS @ L1",           0x91,  176, 144, 15,  4,  198,   99,   1485, 100,  10*16368,  8192,   64, PROFILE_ARTS },
123            { "ARTS @ L2",           0x92,  352, 288, 15,  4,  792,  396,   5940, 100,  40*16368, 16384,  128, PROFILE_ARTS },
124            { "ARTS @ L3",           0x93,  352, 288, 30,  4,  792,  396,  11880, 100,  40*16368, 16384,  384, PROFILE_ARTS },
125            { "ARTS @ L4",           0x94,  352, 288, 30, 16,  792,  396,  11880, 100,  80*16368, 16384, 2000, PROFILE_ARTS },
126    
127            { "AS @ L0",               0xf0,  176, 144, 30,  1,  297,   99,   2970, 100,  10*16368,  2048,  128, PROFILE_AS },
128            { "AS @ L1",               0xf1,  176, 144, 30,  4,  297,   99,   2970, 100,  10*16368,  2048,  128, PROFILE_AS },
129            { "AS @ L2",               0xf2,  352, 288, 15,  4, 1188,  396,   5940, 100,  40*16368,  4096,  384, PROFILE_AS },
130            { "AS @ L3",               0xf3,  352, 288, 30,  4, 1188,  396,  11880, 100,  40*16368,  4096,  768, PROFILE_AS },
131     /*  ISMA Profile 1, (ASP) @ L3b (CIF, 1.5 Mb/s) CIF(352x288), 30fps, 1.5Mbps max ??? */
132            { "AS @ L4",               0xf4,  352, 576, 30,  4, 2376,  792,  23760,  50,  80*16368,  8192, 3000, PROFILE_AS },
133            { "AS @ L5",               0xf5,  720, 576, 30,  4, 4860, 1620,  48600,  25, 112*16368, 16384, 8000, PROFILE_AS },
134    
135    #ifdef DXN_PROFILES
136            { "DXN Handheld",          0x00,  176, 144, 15, -1,  198,   99,   1485, 100,  16*16368, -1,  128, PROFILE_ADAPTQUANT },
137            { "DXN Portable NTSC", 0x00,  352, 240, 30, -1,  990,  330,   9900, 100,  64*16368,     -1,  768, PROFILE_ADAPTQUANT|PROFILE_BVOP },
138            { "DXN Portable PAL",  0x00,  352, 288, 25, -1, 1188,  396,   9900, 100,  64*16368,     -1,  768, PROFILE_ADAPTQUANT|PROFILE_BVOP },
139            { "DXN HT NTSC",           0x00,  720, 480, 30, -1, 4050, 1350,  40500, 100, 192*16368, -1, 4000, PROFILE_ADAPTQUANT|PROFILE_BVOP|PROFILE_INTERLACE },
140            { "DXN HT PAL",         0x00,  720, 576, 25, -1, 4860, 1620,  40500, 100, 192*16368,    -1, 4000, PROFILE_ADAPTQUANT|PROFILE_BVOP|PROFILE_INTERLACE },
141            { "DXN HDTV",             0x00, 1280, 720, 30, -1,10800, 3600, 108000, 100, 384*16368,  -1, 8000, PROFILE_ADAPTQUANT|PROFILE_BVOP|PROFILE_INTERLACE },
142    #endif
143    
144            { "(unrestricted)",     0x00,   0,   0,  0,  0, 0,      0,        0, 100,   0*16368,     0,     0, 0xffffffff },
145    };
146    
147    
148    typedef struct {
149            char * name;
150            float value;
151    } named_float_t;
152    
153    static const named_float_t video_fps_list[] = {
154            {  "15.0",                      15.0F   },
155            {  "23.976 (FILM)",     23.976F },
156            {  "25.0 (PAL)",        25.0F   },
157            {  "29.97 (NTSC)",      29.970F },
158            {  "30.0",                      30.0F   }
159    };
160    
161    
162    typedef struct {
163            char * name;
164            int avi_interval;               /* audio overhead intervals (milliseconds) */
165            float mkv_multiplier;   /* mkv multiplier */
166    } named_int_t;
167    
168    
169    #define NO_AUDIO        5
170    static const named_int_t audio_type_list[] = {
171            {       "MP3-CBR",              1000,   48000/1152/6                                    },
172            {       "MP3-VBR",                24,   48000/1152/6                                    },
173            {       "OGG",     /*?*/1000,   48000*(0.7F/1024 + 0.3F/180)    },
174            {       "AC3",                    64,   48000/1536/6                                    },
175            {       "DTS",                    21,   /*?*/48000/1152/6                               },
176            {       "(None)",                  0,   0                                                               },
177    };
178    
179    
180    /* ===================================================================================== */
181    /* REGISTRY ============================================================================ */
182    /* ===================================================================================== */
183    
184    /* registry info structs */
185    CONFIG reg;
186    
187    static const REG_INT reg_ints[] = {
188            {"mode",                                        &reg.mode,                                              RC_MODE_1PASS},
189            {"bitrate",                                     &reg.bitrate,                                   700},
190            {"desired_size",                        &reg.desired_size,                              570000},
191            {"use_2pass_bitrate",           &reg.use_2pass_bitrate,                 0},
192            {"desired_quant",                       &reg.desired_quant,                             DEFAULT_QUANT}, /* 100-base float */
193    
194            /* profile */
195            {"quant_type",                          &reg.quant_type,                                0},
196            {"lum_masking",                         &reg.lum_masking,                               0},
197            {"interlacing",                         &reg.interlacing,                               0},
198            {"qpel",                                        &reg.qpel,                                              0},
199            {"gmc",                                         &reg.gmc,                                               0},
200            {"reduced_resolution",          &reg.reduced_resolution,                0},
201            {"use_bvop",                            &reg.use_bvop,                                  1},
202            {"max_bframes",                         &reg.max_bframes,                               2},
203            {"bquant_ratio",                        &reg.bquant_ratio,                              150},   /* 100-base float */
204            {"bquant_offset",                       &reg.bquant_offset,                             100},   /* 100-base float */
205            {"packed",                                      &reg.packed,                                    1},
206            {"closed_gov",                          &reg.closed_gov,                                1},
207    
208            /* aspect ratio */
209            {"ar_mode",                                     &reg.ar_mode,                                   0},
210            {"aspect_ratio",                        &reg.display_aspect_ratio,              0},
211            {"par_x",                                       &reg.par_x,                                             1},
212            {"par_y",                                       &reg.par_y,                                             1},
213            {"ar_x",                                        &reg.ar_x,                                              4},
214            {"ar_y",                                        &reg.ar_y,                                              3},
215    
216            /* zones */
217            {"num_zones",                           &reg.num_zones,                                 1},
218    
219            /* single pass */
220            {"rc_reaction_delay_factor",&reg.rc_reaction_delay_factor,      16},
221            {"rc_averaging_period",         &reg.rc_averaging_period,               100},
222            {"rc_buffer",                           &reg.rc_buffer,                                 100},
223    
224            /* 2pass1 */
225            {"discard1pass",                        &reg.discard1pass,                              1},
226            {"full1pass",                           &reg.full1pass,                                 0},
227    
228            /* 2pass2 */
229            {"keyframe_boost",                      &reg.keyframe_boost,                    10},
230            {"kfreduction",                         &reg.kfreduction,                               20},
231            {"kfthreshold",                         &reg.kfthreshold,                               1},
232            {"curve_compression_high",      &reg.curve_compression_high,    0},
233            {"curve_compression_low",       &reg.curve_compression_low,             0},
234            {"overflow_control_strength", &reg.overflow_control_strength, 5},
235            {"twopass_max_overflow_improvement", &reg.twopass_max_overflow_improvement, 5},
236            {"twopass_max_overflow_degradation", &reg.twopass_max_overflow_degradation, 5},
237    
238            /* bitrate calculator */
239            {"container_type",                      &reg.container_type,                    1},
240            {"target_size",                         &reg.target_size,                               650 * 1024},
241            {"subtitle_size",                       &reg.subtitle_size,                             0},
242            {"hours",                               &reg.hours,                                             1},
243            {"minutes",                             &reg.minutes,                                   30},
244            {"seconds",                             &reg.seconds,                                   0},
245            {"fps",                                 &reg.fps,                                               2},
246            {"audio_mode",                          &reg.audio_mode,                                0},
247            {"audio_type",                          &reg.audio_type,                                0},
248            {"audio_rate",                          &reg.audio_rate,                                128},
249            {"audio_size",                          &reg.audio_size,                                0},
250    
251            /* motion */
252            {"motion_search",                       &reg.motion_search,                             6},
253            {"vhq_mode",                            &reg.vhq_mode,                                  1},
254            {"chromame",                            &reg.chromame,                                  1},
255            {"cartoon_mode",                        &reg.cartoon_mode,                              0},
256            {"turbo",                                       &reg.turbo,                                             0},
257            {"max_key_interval",            &reg.max_key_interval,                  300},
258            {"frame_drop_ratio",            &reg.frame_drop_ratio,                  0},
259    
260            /* quant */
261            {"min_iquant",                          &reg.min_iquant,                                1},
262            {"max_iquant",                          &reg.max_iquant,                                31},
263            {"min_pquant",                          &reg.min_pquant,                                1},
264            {"max_pquant",                          &reg.max_pquant,                                31},
265            {"min_bquant",                          &reg.min_bquant,                                1},
266            {"max_bquant",                          &reg.max_bquant,                                31},
267            {"trellis_quant",                       &reg.trellis_quant,                             0},
268    
269            /* debug */
270            {"fourcc_used",                         &reg.fourcc_used,                               0},
271            {"debug",                                       &reg.debug,                                             0x0},
272            {"vop_debug",                           &reg.vop_debug,                                 0},
273            {"display_status",                      &reg.display_status,                    1},
274    
275            /* decoder, shared with dshow */
276            {"Deblock_Y",                           &pp_dy,                                                 0},
277            {"Deblock_UV",                          &pp_duv,                                                0},
278            {"Dering",                                      &pp_dr,                                                 0},
279            {"FilmEffect",                          &pp_fe,                                                 0},
280    
281    };
282    
283    static const REG_STR reg_strs[] = {
284            {"profile",                                     reg.profile_name,                               "AS @ L5"},
285            {"stats",                                       reg.stats,                                              CONFIG_2PASS_FILE},
286    };
287    
288    
289    zone_t stmp;
290    static const REG_INT reg_zone[] = {
291            {"zone%i_frame",                        &stmp.frame,                                    0},
292            {"zone%i_mode",                         &stmp.mode,                                             RC_ZONE_WEIGHT},
293            {"zone%i_weight",                       &stmp.weight,                                   100},     /* 100-base float */
294            {"zone%i_quant",                        &stmp.quant,                                    500},     /* 100-base float */
295            {"zone%i_type",                         &stmp.type,                                             XVID_TYPE_AUTO},
296            {"zone%i_greyscale",            &stmp.greyscale,                                0},
297            {"zone%i_chroma_opt",           &stmp.chroma_opt,                               0},
298            {"zone%i_bvop_threshold",   &stmp.bvop_threshold,                       0},
299    };
300    
301    static const BYTE default_qmatrix_intra[] = {
302            8, 17,18,19,21,23,25,27,
303            17,18,19,21,23,25,27,28,
304            20,21,22,23,24,26,28,30,
305            21,22,23,24,26,28,30,32,
306            22,23,24,26,28,30,32,35,
307            23,24,26,28,30,32,35,38,
308            25,26,28,30,32,35,38,41,
309            27,28,30,32,35,38,41,45
310    };
311    
312    static const BYTE default_qmatrix_inter[] = {
313            16,17,18,19,20,21,22,23,
314            17,18,19,20,21,22,23,24,
315            18,19,20,21,22,23,24,25,
316            19,20,21,22,23,24,26,27,
317            20,21,22,23,25,26,27,28,
318            21,22,23,24,26,27,28,30,
319            22,23,24,26,27,28,30,31,
320            23,24,25,27,28,30,31,33
321    };
322    
323    
324    
325    #define REG_GET_B(X, Y, Z) size=sizeof((Z));if(RegQueryValueEx(hKey, X, 0, 0, Y, &size) != ERROR_SUCCESS) {memcpy(Y, Z, sizeof((Z)));}
326    
327    #define XVID_DLL_NAME "xvidcore.dll"
328    
329    void config_reg_get(CONFIG * config)
330    {
331            char tmp[32];
332            HKEY hKey;
333            DWORD size;
334            int i,j;
335            xvid_gbl_info_t info;
336            HINSTANCE m_hdll;
337    
338            memset(&info, 0, sizeof(info));
339            info.version = XVID_VERSION;
340    
341            m_hdll = LoadLibrary(XVID_DLL_NAME);
342            if (m_hdll != NULL) {
343    
344                    ((int (__cdecl *)(void *, int, void *, void *))GetProcAddress(m_hdll, "xvid_global"))
345                            (0, XVID_GBL_INFO, &info, NULL);
346    
347                    FreeLibrary(m_hdll);
348            }
349    
350            reg.cpu = info.cpu_flags;
351            reg.num_threads = info.num_threads;
352    
353            RegOpenKeyEx(XVID_REG_KEY, XVID_REG_PARENT "\\" XVID_REG_CHILD, 0, KEY_READ, &hKey);
354    
355            /* read integer values */
356            for (i=0 ; i<sizeof(reg_ints)/sizeof(REG_INT); i++) {
357                    size = sizeof(int);
358                    if (RegQueryValueEx(hKey, reg_ints[i].reg_value, 0, 0, (LPBYTE)reg_ints[i].config_int, &size) != ERROR_SUCCESS) {
359                            *reg_ints[i].config_int = reg_ints[i].def;
360                    }
361            }
362    
363            /* read string values */
364            for (i=0 ; i<sizeof(reg_strs)/sizeof(REG_STR); i++) {
365                    size = MAX_PATH;
366                    if (RegQueryValueEx(hKey, reg_strs[i].reg_value, 0, 0, (LPBYTE)reg_strs[i].config_str, &size) != ERROR_SUCCESS) {
367                            memcpy(reg_strs[i].config_str, reg_strs[i].def, MAX_PATH);
368                    }
369            }
370    
371            reg.profile = 0;
372            for (i=0; i<sizeof(profiles)/sizeof(profile_t); i++) {
373                    if (lstrcmpi(profiles[i].name, reg.profile_name) == 0) {
374                            reg.profile = i;
375                    }
376            }
377    
378            memcpy(config, &reg, sizeof(CONFIG));
379    
380    
381            /* read quant matrices */
382            REG_GET_B("qmatrix_intra", config->qmatrix_intra, default_qmatrix_intra);
383            REG_GET_B("qmatrix_inter", config->qmatrix_inter, default_qmatrix_inter);
384    
385    
386            /* read zones */
387            if (config->num_zones>MAX_ZONES) {
388                    config->num_zones=MAX_ZONES;
389            }else if (config->num_zones<=0) {
390                    config->num_zones = 1;
391            }
392    
393            for (i=0; i<config->num_zones; i++) {
394                    for (j=0; j<sizeof(reg_zone)/sizeof(REG_INT); j++)  {
395                            size = sizeof(int);
396    
397                            wsprintf(tmp, reg_zone[j].reg_value, i);
398                            if (RegQueryValueEx(hKey, tmp, 0, 0, (LPBYTE)reg_zone[j].config_int, &size) != ERROR_SUCCESS)
399                                    *reg_zone[j].config_int = reg_zone[j].def;
400                    }
401    
402                    memcpy(&config->zones[i], &stmp, sizeof(zone_t));
403            }
404    
405            RegCloseKey(hKey);
406    }
407    
408    
409    /* put config settings in registry */
410    
411    #define REG_SET_B(X, Y) RegSetValueEx(hKey, X, 0, REG_BINARY, Y, sizeof((Y)))
412    
413    void config_reg_set(CONFIG * config)
414    {
415            char tmp[64];
416            HKEY hKey;
417            DWORD dispo;
418            int i,j;
419    
420            if (RegCreateKeyEx(
421                            XVID_REG_KEY,
422                            XVID_REG_PARENT "\\" XVID_REG_CHILD,
423                            0,
424                            XVID_REG_CLASS,
425                            REG_OPTION_NON_VOLATILE,
426                            KEY_WRITE,
427                            0,
428                            &hKey,
429                            &dispo) != ERROR_SUCCESS)
430            {
431                    DPRINTF("Couldn't create XVID_REG_SUBKEY - GetLastError=%i", GetLastError());
432                    return;
433            }
434    
435            memcpy(&reg, config, sizeof(CONFIG));
436    
437            /* set integer values */
438            for (i=0 ; i<sizeof(reg_ints)/sizeof(REG_INT); i++) {
439                    RegSetValueEx(hKey, reg_ints[i].reg_value, 0, REG_DWORD, (LPBYTE)reg_ints[i].config_int, sizeof(int));
440            }
441    
442            /* set string values */
443            strcpy(reg.profile_name, profiles[reg.profile].name);
444            for (i=0 ; i<sizeof(reg_strs)/sizeof(REG_STR); i++) {
445                    RegSetValueEx(hKey, reg_strs[i].reg_value, 0, REG_SZ, reg_strs[i].config_str, lstrlen(reg_strs[i].config_str)+1);
446            }
447    
448            /* set quant matrices */
449            REG_SET_B("qmatrix_intra", config->qmatrix_intra);
450            REG_SET_B("qmatrix_inter", config->qmatrix_inter);
451    
452            /* set seections */
453            for (i=0; i<config->num_zones; i++) {
454                    memcpy(&stmp, &config->zones[i], sizeof(zone_t));
455                    for (j=0; j<sizeof(reg_zone)/sizeof(REG_INT); j++)  {
456                            wsprintf(tmp, reg_zone[j].reg_value, i);
457                            RegSetValueEx(hKey, tmp, 0, REG_DWORD, (LPBYTE)reg_zone[j].config_int, sizeof(int));
458                    }
459            }
460    
461            RegCloseKey(hKey);
462    }
463    
464    
465    /* clear XviD registry key, load defaults */
466    
467    static void config_reg_default(CONFIG * config)
468    {
469            HKEY hKey;
470    
471            if (RegOpenKeyEx(XVID_REG_KEY, XVID_REG_PARENT, 0, KEY_ALL_ACCESS, &hKey)) {
472                    DPRINTF("Couldn't open registry key for deletion - GetLastError=%i", GetLastError());
473                    return;
474            }
475    
476            if (RegDeleteKey(hKey, XVID_REG_CHILD)) {
477                    DPRINTF("Couldn't delete registry key - GetLastError=%i", GetLastError());
478                    return;
479            }
480    
481            RegCloseKey(hKey);
482            config_reg_get(config);
483            config_reg_set(config);
484    }
485    
486    
487    /* leaves current config value if dialog item is empty */
488    static int config_get_int(HWND hDlg, INT item, int config)
489    {
490            BOOL success = FALSE;
491            int tmp = GetDlgItemInt(hDlg, item, &success, TRUE);
492            return (success) ? tmp : config;
493    }
494    
495    
496    static int config_get_uint(HWND hDlg, UINT item, int config)
497    {
498            BOOL success = FALSE;
499            int tmp = GetDlgItemInt(hDlg, item, &success, FALSE);
500            return (success) ? tmp : config;
501    }
502    
503    /* get uint from combobox
504       GetDlgItemInt doesnt work properly for cb list items */
505    #define UINT_BUF_SZ     20
506    static int config_get_cbuint(HWND hDlg, UINT item, int def)
507    {
508            int sel = SendMessage(GetDlgItem(hDlg, item), CB_GETCURSEL, 0, 0);
509            char buf[UINT_BUF_SZ];
510    
511            if (sel<0) {
512                    return config_get_uint(hDlg, item, def);
513            }
514    
515            if (SendMessage(GetDlgItem(hDlg, item), CB_GETLBTEXT, sel, (LPARAM)buf) == CB_ERR) {
516                    return def;
517            }
518    
519            return atoi(buf);
520    }
521    
522    
523    /* we use "100 base" floats */
524    
525    #define FLOAT_BUF_SZ    20
526    static int get_dlgitem_float(HWND hDlg, UINT item, int def)
527    {
528            char buf[FLOAT_BUF_SZ];
529    
530            if (GetDlgItemText(hDlg, item, buf, FLOAT_BUF_SZ) == 0)
531                    return def;
532    
533            return (int)(atof(buf)*100);
534    }
535    
536    static void set_dlgitem_float(HWND hDlg, UINT item, int value)
537    {
538            char buf[FLOAT_BUF_SZ];
539            sprintf(buf, "%.2f", (float)value/100);
540            SetDlgItemText(hDlg, item, buf);
541    }
542    
543    
544    #define HEX_BUF_SZ  16
545    static unsigned int get_dlgitem_hex(HWND hDlg, UINT item, unsigned int def)
546    {
547            char buf[HEX_BUF_SZ];
548            unsigned int value;
549    
550            if (GetDlgItemText(hDlg, item, buf, HEX_BUF_SZ) == 0)
551                    return def;
552    
553            if (sscanf(buf,"0x%x", &value)==1 || sscanf(buf,"%x", &value)==1) {
554                    return value;
555            }
556    
557            return def;
558    }
559    
560    static void set_dlgitem_hex(HWND hDlg, UINT item, int value)
561    {
562            char buf[HEX_BUF_SZ];
563            wsprintf(buf, "0x%x", value);
564            SetDlgItemText(hDlg, item, buf);
565    }
566    
567    /* ===================================================================================== */
568    /* QUANT MATRIX DIALOG ================================================================= */
569    /* ===================================================================================== */
570    
571    static void quant_upload(HWND hDlg, CONFIG* config)
572    {
573            int i;
574    
575            for (i=0 ; i<64; i++) {
576                    SetDlgItemInt(hDlg, IDC_QINTRA00 + i, config->qmatrix_intra[i], FALSE);
577                    SetDlgItemInt(hDlg, IDC_QINTER00 + i, config->qmatrix_inter[i], FALSE);
578            }
579    }
580    
581    
582    static void quant_download(HWND hDlg, CONFIG* config)
583    {
584            int i;
585    
586            for (i=0; i<64; i++) {
587                    int temp;
588    
589                    temp = config_get_uint(hDlg, i + IDC_QINTRA00, config->qmatrix_intra[i]);
590                    CONSTRAINVAL(temp, 1, 255);
591                    config->qmatrix_intra[i] = temp;
592    
593                    temp = config_get_uint(hDlg, i + IDC_QINTER00, config->qmatrix_inter[i]);
594                    CONSTRAINVAL(temp, 1, 255);
595                    config->qmatrix_inter[i] = temp;
596            }
597    }
598    
599    
600    static void quant_loadsave(HWND hDlg, CONFIG * config, int save)
601    {
602            char file[MAX_PATH];
603            OPENFILENAME ofn;
604            HANDLE hFile;
605            DWORD read=128, wrote=0;
606            BYTE quant_data[128];
607    
608            strcpy(file, "\\matrix");
609            memset(&ofn, 0, sizeof(OPENFILENAME));
610            ofn.lStructSize = sizeof(OPENFILENAME);
611    
612            ofn.hwndOwner = hDlg;
613            ofn.lpstrFilter = "All files (*.*)\0*.*\0\0";
614            ofn.lpstrFile = file;
615            ofn.nMaxFile = MAX_PATH;
616            ofn.Flags = OFN_PATHMUSTEXIST;
617    
618            if (save) {
619                    ofn.Flags |= OFN_OVERWRITEPROMPT;
620                    if (GetSaveFileName(&ofn)) {
621                            hFile = CreateFile(file, GENERIC_WRITE, 0, 0, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, 0);
622    
623                            quant_download(hDlg, config);
624                            memcpy(quant_data, config->qmatrix_intra, 64);
625                            memcpy(quant_data+64, config->qmatrix_inter, 64);
626    
627                            if (hFile == INVALID_HANDLE_VALUE) {
628                                    DPRINTF("Couldn't save quant matrix");
629                            }else{
630                                    if (!WriteFile(hFile, quant_data, 128, &wrote, 0)) {
631                                            DPRINTF("Couldnt write quant matrix");
632                                    }
633                            }
634                            CloseHandle(hFile);
635                    }
636            }else{
637                    ofn.Flags |= OFN_FILEMUSTEXIST;
638                    if (GetOpenFileName(&ofn)) {
639                            hFile = CreateFile(file, GENERIC_READ, 0, 0, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0);
640    
641                            if (hFile == INVALID_HANDLE_VALUE) {
642                                    DPRINTF("Couldn't load quant matrix");
643                            } else {
644                                    if (!ReadFile(hFile, quant_data, 128, &read, 0)) {
645                                            DPRINTF("Couldnt read quant matrix");
646                                    }else{
647                                            memcpy(config->qmatrix_intra, quant_data, 64);
648                                            memcpy(config->qmatrix_inter, quant_data+64, 64);
649                                            quant_upload(hDlg, config);
650                                    }
651                            }
652                            CloseHandle(hFile);
653                    }
654            }
655    }
656    
657    /* quantization matrix dialog proc */
658    
659    static BOOL CALLBACK quantmatrix_proc(HWND hDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
660    {
661            CONFIG* config = (CONFIG*)GetWindowLong(hDlg, GWL_USERDATA);
662    
663            switch (uMsg)
664            {
665            case WM_INITDIALOG :
666                    SetWindowLong(hDlg, GWL_USERDATA, lParam);
667                    config = (CONFIG*)lParam;
668                    quant_upload(hDlg, config);
669    
670                    if (g_hTooltip)
671                    {
672                            EnumChildWindows(hDlg, enum_tooltips, 0);
673                    }
674                    break;
675    
676            case WM_COMMAND :
677    
678                    if (HIWORD(wParam) == BN_CLICKED) {
679                            switch(LOWORD(wParam)) {
680                            case IDOK :
681                                    quant_download(hDlg, config);
682                                    EndDialog(hDlg, IDOK);
683                                    break;
684    
685                            case IDCANCEL :
686                                    EndDialog(hDlg, IDCANCEL);
687                                    break;
688    
689                            case IDC_SAVE :
690                                    quant_loadsave(hDlg, config, 1);
691                                    break;
692    
693                            case IDC_LOAD :
694                                    quant_loadsave(hDlg, config, 0);
695                                    break;
696    
697                            default :
698                                    return FALSE;
699                            }
700                            break;
701                    }
702                    return FALSE;
703    
704            default :
705                    return FALSE;
706            }
707    
708            return TRUE;
709    }
710    
711    
712    /* ===================================================================================== */
713    /* ADVANCED DIALOG PAGES ================================================================ */
714    /* ===================================================================================== */
715    
716    /* initialise pages */
717    static void adv_init(HWND hDlg, int idd, CONFIG * config)
718    {
719            unsigned int i;
720    
721            switch(idd) {
722            case IDD_PROFILE :
723                    for (i=0; i<sizeof(profiles)/sizeof(profile_t); i++)
724                            SendDlgItemMessage(hDlg, IDC_PROFILE_PROFILE, CB_ADDSTRING, 0, (LPARAM)profiles[i].name);
725                    SendDlgItemMessage(hDlg, IDC_QUANTTYPE, CB_ADDSTRING, 0, (LPARAM)"H.263");
726                    SendDlgItemMessage(hDlg, IDC_QUANTTYPE, CB_ADDSTRING, 0, (LPARAM)"MPEG");
727                    SendDlgItemMessage(hDlg, IDC_QUANTTYPE, CB_ADDSTRING, 0, (LPARAM)"MPEG-Custom");
728                    break;
729    
730            case IDD_AR:
731                    SendDlgItemMessage(hDlg, IDC_ASPECT_RATIO, CB_ADDSTRING, 0, (LPARAM)"Square (default)");
732                    SendDlgItemMessage(hDlg, IDC_ASPECT_RATIO, CB_ADDSTRING, 0, (LPARAM)"4:3 PAL");
733                    SendDlgItemMessage(hDlg, IDC_ASPECT_RATIO, CB_ADDSTRING, 0, (LPARAM)"4:3 NTSC");
734                    SendDlgItemMessage(hDlg, IDC_ASPECT_RATIO, CB_ADDSTRING, 0, (LPARAM)"16:9 PAL");
735                    SendDlgItemMessage(hDlg, IDC_ASPECT_RATIO, CB_ADDSTRING, 0, (LPARAM)"16:9 NTSC");
736                    SendDlgItemMessage(hDlg, IDC_ASPECT_RATIO, CB_ADDSTRING, 0, (LPARAM)"Custom...");
737                    break;
738    
739            case IDD_LEVEL :
740                    for (i=0; i<sizeof(profiles)/sizeof(profile_t); i++)
741                            SendDlgItemMessage(hDlg, IDC_LEVEL_PROFILE, CB_ADDSTRING, 0, (LPARAM)profiles[i].name);
742                    break;
743    
744            case IDD_BITRATE :
745                    SendDlgItemMessage(hDlg, IDC_BITRATE_CFORMAT, CB_ADDSTRING, 0, (LPARAM)"AVI-Legacy");
746                    SendDlgItemMessage(hDlg, IDC_BITRATE_CFORMAT, CB_ADDSTRING, 0, (LPARAM)"AVI-OpenDML");
747                    SendDlgItemMessage(hDlg, IDC_BITRATE_CFORMAT, CB_ADDSTRING, 0, (LPARAM)"Matroska");
748                    SendDlgItemMessage(hDlg, IDC_BITRATE_CFORMAT, CB_ADDSTRING, 0, (LPARAM)"OGM");
749                    SendDlgItemMessage(hDlg, IDC_BITRATE_CFORMAT, CB_ADDSTRING, 0, (LPARAM)"(None)");
750    
751                    SendDlgItemMessage(hDlg, IDC_BITRATE_TSIZE, CB_ADDSTRING, 0, (LPARAM)"665600");
752                    SendDlgItemMessage(hDlg, IDC_BITRATE_TSIZE, CB_ADDSTRING, 0, (LPARAM)"716800");
753                    SendDlgItemMessage(hDlg, IDC_BITRATE_TSIZE, CB_ADDSTRING, 0, (LPARAM)"1331200");
754                    SendDlgItemMessage(hDlg, IDC_BITRATE_TSIZE, CB_ADDSTRING, 0, (LPARAM)"1433600");
755    
756                    for (i=0; i<sizeof(video_fps_list)/sizeof(named_float_t); i++)
757                            SendDlgItemMessage(hDlg, IDC_BITRATE_FPS, CB_ADDSTRING, 0, (LPARAM)video_fps_list[i].name);
758    
759                    for (i=0; i<sizeof(audio_type_list)/sizeof(named_int_t); i++)
760                            SendDlgItemMessage(hDlg, IDC_BITRATE_AFORMAT, CB_ADDSTRING, 0, (LPARAM)audio_type_list[i].name);
761    
762                    SendDlgItemMessage(hDlg, IDC_BITRATE_ARATE, CB_ADDSTRING, 0, (LPARAM)"32");
763                    SendDlgItemMessage(hDlg, IDC_BITRATE_ARATE, CB_ADDSTRING, 0, (LPARAM)"56");
764                    SendDlgItemMessage(hDlg, IDC_BITRATE_ARATE, CB_ADDSTRING, 0, (LPARAM)"64");
765                    SendDlgItemMessage(hDlg, IDC_BITRATE_ARATE, CB_ADDSTRING, 0, (LPARAM)"96");
766                    SendDlgItemMessage(hDlg, IDC_BITRATE_ARATE, CB_ADDSTRING, 0, (LPARAM)"112");
767                    SendDlgItemMessage(hDlg, IDC_BITRATE_ARATE, CB_ADDSTRING, 0, (LPARAM)"128");
768                    SendDlgItemMessage(hDlg, IDC_BITRATE_ARATE, CB_ADDSTRING, 0, (LPARAM)"160");
769                    SendDlgItemMessage(hDlg, IDC_BITRATE_ARATE, CB_ADDSTRING, 0, (LPARAM)"192");
770                    SendDlgItemMessage(hDlg, IDC_BITRATE_ARATE, CB_ADDSTRING, 0, (LPARAM)"224");
771                    SendDlgItemMessage(hDlg, IDC_BITRATE_ARATE, CB_ADDSTRING, 0, (LPARAM)"256");
772                    SendDlgItemMessage(hDlg, IDC_BITRATE_ARATE, CB_ADDSTRING, 0, (LPARAM)"384");
773                    SendDlgItemMessage(hDlg, IDC_BITRATE_ARATE, CB_ADDSTRING, 0, (LPARAM)"448");
774                    SendDlgItemMessage(hDlg, IDC_BITRATE_ARATE, CB_ADDSTRING, 0, (LPARAM)"512");
775                    break;
776    
777            case IDD_ZONE :
778                    EnableDlgWindow(hDlg, IDC_ZONE_FETCH, config->ci_valid);
779                    break;
780    
781            case IDD_MOTION :
782                    SendDlgItemMessage(hDlg, IDC_MOTION, CB_ADDSTRING, 0, (LPARAM)"0 - None");
783                    SendDlgItemMessage(hDlg, IDC_MOTION, CB_ADDSTRING, 0, (LPARAM)"1 - Very Low");
784                    SendDlgItemMessage(hDlg, IDC_MOTION, CB_ADDSTRING, 0, (LPARAM)"2 - Low");
785                    SendDlgItemMessage(hDlg, IDC_MOTION, CB_ADDSTRING, 0, (LPARAM)"3 - Medium");
786                    SendDlgItemMessage(hDlg, IDC_MOTION, CB_ADDSTRING, 0, (LPARAM)"4 - High");
787                    SendDlgItemMessage(hDlg, IDC_MOTION, CB_ADDSTRING, 0, (LPARAM)"5 - Very High");
788                    SendDlgItemMessage(hDlg, IDC_MOTION, CB_ADDSTRING, 0, (LPARAM)"6 - Ultra High");
789    
790                    SendDlgItemMessage(hDlg, IDC_VHQ, CB_ADDSTRING, 0, (LPARAM)"0 - Off");
791                    SendDlgItemMessage(hDlg, IDC_VHQ, CB_ADDSTRING, 0, (LPARAM)"1 - Mode Decision");
792                    SendDlgItemMessage(hDlg, IDC_VHQ, CB_ADDSTRING, 0, (LPARAM)"2 - Limited Search");
793                    SendDlgItemMessage(hDlg, IDC_VHQ, CB_ADDSTRING, 0, (LPARAM)"3 - Medium Search");
794                    SendDlgItemMessage(hDlg, IDC_VHQ, CB_ADDSTRING, 0, (LPARAM)"4 - Wide Search");
795                    break;
796    
797            case IDD_DEBUG :
798                    /* force threads disabled */
799                    EnableWindow(GetDlgItem(hDlg, IDC_NUMTHREADS_STATIC), FALSE);
800                    EnableWindow(GetDlgItem(hDlg, IDC_NUMTHREADS), FALSE);
801    
802                    SendDlgItemMessage(hDlg, IDC_FOURCC, CB_ADDSTRING, 0, (LPARAM)"XVID");
803                    SendDlgItemMessage(hDlg, IDC_FOURCC, CB_ADDSTRING, 0, (LPARAM)"DIVX");
804                    SendDlgItemMessage(hDlg, IDC_FOURCC, CB_ADDSTRING, 0, (LPARAM)"DX50");
805                    break;
806            }
807    }
808    
809    
810    /* enable/disable controls based on encoder-mode or user selection */
811    
812    static void adv_mode(HWND hDlg, int idd, CONFIG * config)
813    {
814            int profile;
815            int weight_en, quant_en;
816            int cpu_force;
817            int custom_quant, bvops;
818            int ar_mode, ar_par;
819    
820            switch(idd) {
821            case IDD_PROFILE :
822                    profile = SendDlgItemMessage(hDlg, IDC_PROFILE_PROFILE, CB_GETCURSEL, 0, 0);
823                    EnableDlgWindow(hDlg, IDC_BVOP, profiles[profile].flags&PROFILE_BVOP);
824    
825                    EnableDlgWindow(hDlg, IDC_QUANTTYPE_S, profiles[profile].flags&PROFILE_MPEGQUANT);
826                    EnableDlgWindow(hDlg, IDC_QUANTTYPE_S, profiles[profile].flags&PROFILE_MPEGQUANT);
827                    EnableDlgWindow(hDlg, IDC_QUANTTYPE, profiles[profile].flags&PROFILE_MPEGQUANT);
828                    custom_quant = (profiles[profile].flags&PROFILE_MPEGQUANT) && SendDlgItemMessage(hDlg, IDC_QUANTTYPE, CB_GETCURSEL, 0, 0)==QUANT_MODE_CUSTOM;
829                    EnableDlgWindow(hDlg, IDC_QUANTMATRIX, custom_quant);
830                    EnableDlgWindow(hDlg, IDC_LUMMASK, profiles[profile].flags&PROFILE_ADAPTQUANT);
831                    EnableDlgWindow(hDlg, IDC_INTERLACING, profiles[profile].flags&PROFILE_INTERLACE);
832                    EnableDlgWindow(hDlg, IDC_QPEL, profiles[profile].flags&PROFILE_QPEL);
833                    EnableDlgWindow(hDlg, IDC_GMC, profiles[profile].flags&PROFILE_GMC);
834                    EnableDlgWindow(hDlg, IDC_REDUCED, profiles[profile].flags&PROFILE_REDUCED);
835    
836                    bvops = (profiles[profile].flags&PROFILE_BVOP) && IsDlgChecked(hDlg, IDC_BVOP);
837                    EnableDlgWindow(hDlg, IDC_MAXBFRAMES,      bvops);
838                    EnableDlgWindow(hDlg, IDC_BQUANTRATIO,    bvops);
839                    EnableDlgWindow(hDlg, IDC_BQUANTOFFSET,  bvops);
840                    EnableDlgWindow(hDlg, IDC_MAXBFRAMES_S,  bvops);
841                    EnableDlgWindow(hDlg, IDC_BQUANTRATIO_S,        bvops);
842                    EnableDlgWindow(hDlg, IDC_BQUANTOFFSET_S,   bvops);
843                    EnableDlgWindow(hDlg, IDC_PACKED,                  bvops);
844                    EnableDlgWindow(hDlg, IDC_CLOSEDGOV,            bvops);
845                    break;
846    
847            case IDD_AR:
848                    ar_mode = IsDlgChecked(hDlg, IDC_PAR);
849                    EnableDlgWindow(hDlg, IDC_ASPECT_RATIO, ar_mode);
850    
851                    ar_par = SendDlgItemMessage(hDlg, IDC_ASPECT_RATIO, CB_GETCURSEL, 0, 0);
852                    if (ar_par == 5) { /* custom par */
853                            SetDlgItemInt(hDlg, IDC_PARY, config->par_y, FALSE);
854                            SetDlgItemInt(hDlg, IDC_PARX, config->par_x, FALSE);
855    
856                            EnableDlgWindow(hDlg, IDC_PARX, ar_mode);
857                            EnableDlgWindow(hDlg, IDC_PARY, ar_mode);
858                    } else {
859                            SetDlgItemInt(hDlg, IDC_PARX, PARS[ar_par][0], FALSE);
860                            SetDlgItemInt(hDlg, IDC_PARY, PARS[ar_par][1], FALSE);
861                            EnableDlgWindow(hDlg, IDC_PARX, FALSE);
862                            EnableDlgWindow(hDlg, IDC_PARY, FALSE);
863                    }
864    
865                    ar_mode = IsDlgChecked(hDlg, IDC_AR);
866    
867                    config->ar_x = config_get_uint(hDlg, IDC_ARX, config->ar_x);
868                    config->ar_y = config_get_uint(hDlg, IDC_ARY, config->ar_y);
869    
870                    EnableDlgWindow(hDlg, IDC_ARX, ar_mode);
871                    EnableDlgWindow(hDlg, IDC_ARY, ar_mode);
872                    break;
873    
874            case IDD_LEVEL :
875                    profile = SendDlgItemMessage(hDlg, IDC_LEVEL_PROFILE, CB_GETCURSEL, 0, 0);
876                    SetDlgItemInt(hDlg, IDC_LEVEL_WIDTH, profiles[profile].width, FALSE);
877                    SetDlgItemInt(hDlg, IDC_LEVEL_HEIGHT, profiles[profile].height, FALSE);
878                    SetDlgItemInt(hDlg, IDC_LEVEL_FPS, profiles[profile].fps, FALSE);
879                    SetDlgItemInt(hDlg, IDC_LEVEL_VMV, profiles[profile].max_vmv_buffer_sz, FALSE);
880                    SetDlgItemInt(hDlg, IDC_LEVEL_VCV, profiles[profile].vcv_decoder_rate, FALSE);
881                    SetDlgItemInt(hDlg, IDC_LEVEL_VBV, profiles[profile].max_vbv_size, FALSE);
882                    SetDlgItemInt(hDlg, IDC_LEVEL_BITRATE, profiles[profile].max_bitrate, FALSE);
883                    break;
884    
885            case IDD_BITRATE :
886                    {
887                            int ctype = SendDlgItemMessage(hDlg, IDC_BITRATE_CFORMAT, CB_GETCURSEL, 0, 0);
888                            int target_size = config_get_cbuint(hDlg, IDC_BITRATE_TSIZE, 0);
889                            int subtitle_size = config_get_uint(hDlg, IDC_BITRATE_SSIZE, 0);
890                            int fps = SendDlgItemMessage(hDlg, IDC_BITRATE_FPS, CB_GETCURSEL, 0, 0);
891    
892                            int duration =
893                                    3600 * config_get_uint(hDlg, IDC_BITRATE_HOURS, 0) +
894                                    60 * config_get_uint(hDlg, IDC_BITRATE_MINUTES, 0) +
895                                    config_get_uint(hDlg, IDC_BITRATE_SECONDS, 0);
896    
897                            int audio_type = SendDlgItemMessage(hDlg, IDC_BITRATE_AFORMAT, CB_GETCURSEL, 0, 0);
898                            int audio_mode = IsDlgChecked(hDlg, IDC_BITRATE_AMODE_SIZE);
899                            int audio_rate = config_get_cbuint(hDlg, IDC_BITRATE_ARATE, 0);
900                            int audio_size = config_get_uint(hDlg, IDC_BITRATE_ASIZE, 0);
901    
902                            int frames;
903                            int overhead;
904                            int vsize;
905    
906                            if (duration == 0)
907                                    break;
908    
909                            if (fps < 0 || fps >= sizeof(video_fps_list)/sizeof(named_float_t)) {
910                                    fps = 0;
911                            }
912                            if (audio_type < 0 || audio_type >= sizeof(audio_type_list)/sizeof(named_int_t)) {
913                                    audio_type = 0;
914                            }
915    
916                            EnableDlgWindow(hDlg, IDC_BITRATE_AMODE_RATE, audio_type!=NO_AUDIO);
917                            EnableDlgWindow(hDlg, IDC_BITRATE_AMODE_SIZE, audio_type!=NO_AUDIO);
918                            EnableDlgWindow(hDlg, IDC_BITRATE_ARATE, audio_type!=NO_AUDIO && !audio_mode);
919                            EnableDlgWindow(hDlg, IDC_BITRATE_ASIZE, audio_type!=NO_AUDIO && audio_mode);
920                            EnableDlgWindow(hDlg, IDC_BITRATE_ASELECT, audio_type!=NO_AUDIO && audio_mode);
921    
922                            /* step 1: calculate number of frames */
923                            frames = (int)(duration * video_fps_list[fps].value);
924    
925                            /* step 2: calculate audio_size (kbytes)*/
926                            if (audio_type!=NO_AUDIO) {
927                                    if (audio_mode==0) {
928                                            audio_size = (duration * audio_rate) / 8;
929                                    }
930                            }else{
931                                    audio_size = 0;
932                            }
933    
934                            /* step 3: calculate container overhead */
935    
936                            switch(ctype) {
937                            case 0 :        /* AVI */
938                            case 1 :        /* AVI-OpenDML */
939    
940                                    overhead = frames;
941    
942                                    if (audio_type!=NO_AUDIO) {
943                                            overhead += (duration * 1000) / audio_type_list[audio_type].avi_interval;
944                                    }
945    
946                                    overhead *= (ctype==0) ? 24 : 16;
947    
948                                    overhead /= 1024;
949                                    break;
950    
951                            case 2 :        /* Matroska: gknot formula */
952    
953                                    /* common overhead */
954                                    overhead = 40 + 12 + 8+ 16*duration + 200 + 100*1/*one audio stream*/ + 11*duration;
955    
956                                    /* video overhead */
957                                    overhead += frames*8 + (int)(frames * 4 * 0.94);
958    
959                                    /* cue tables and menu seek entries (300k default) */
960                                    overhead += 300 * 1024;
961    
962                                    /* audio */
963                                    overhead += (int)(duration * audio_type_list[audio_type].mkv_multiplier);
964    
965                                    overhead /= 1024;
966                                    break;
967    
968                            case 3 :        /* OGM: inaccurate model */
969                                    overhead = (int)(0.0039F * (target_size - subtitle_size));
970                                    break;
971    
972                            default :       /* (none) */
973                                    overhead = 0;
974                                    break;
975                            }
976    
977                            SetDlgItemInt(hDlg, IDC_BITRATE_COVERHEAD, overhead, TRUE);
978    
979                            /* final video bitstream size */
980                            vsize = target_size - subtitle_size - audio_size - overhead;
981                            if (vsize > 0) {
982                                    SetDlgItemInt(hDlg, IDC_BITRATE_VSIZE, vsize, TRUE);
983                                    /* convert from kbytes to kbits-per-second */
984                                    SetDlgItemInt(hDlg, IDC_BITRATE_VRATE, (vsize * 8 * 128) / (duration * 125), TRUE);
985                            }else{
986                                    SetDlgItemText(hDlg, IDC_BITRATE_VSIZE, "Overflow");
987                                    SetDlgItemText(hDlg, IDC_BITRATE_VRATE, "Overflow");
988                            }
989    
990                    }
991                    break;
992    
993            case IDD_ZONE :
994                    weight_en = IsDlgChecked(hDlg, IDC_ZONE_MODE_WEIGHT);
995                    quant_en =   IsDlgChecked(hDlg, IDC_ZONE_MODE_QUANT);
996                    EnableDlgWindow(hDlg, IDC_ZONE_WEIGHT, weight_en);
997                    EnableDlgWindow(hDlg, IDC_ZONE_QUANT, quant_en);
998                    EnableDlgWindow(hDlg, IDC_ZONE_SLIDER, weight_en|quant_en);
999    
1000                    if (weight_en) {
1001                            SendDlgItemMessage(hDlg, IDC_ZONE_SLIDER, TBM_SETRANGE, TRUE, MAKELONG(001,200));
1002                            SendDlgItemMessage(hDlg, IDC_ZONE_SLIDER, TBM_SETPOS, TRUE, get_dlgitem_float(hDlg, IDC_ZONE_WEIGHT, 100));
1003                            SetDlgItemText(hDlg, IDC_ZONE_MIN, "0.01");
1004                            SetDlgItemText(hDlg, IDC_ZONE_MAX, "2.00");
1005                    }else if (quant_en) {
1006                            SendDlgItemMessage(hDlg, IDC_ZONE_SLIDER, TBM_SETRANGE, TRUE, MAKELONG(100,3100));
1007                            SendDlgItemMessage(hDlg, IDC_ZONE_SLIDER, TBM_SETPOS, TRUE, get_dlgitem_float(hDlg, IDC_ZONE_QUANT, 100));
1008                            SetDlgItemText(hDlg, IDC_ZONE_MIN, "1");
1009                            SetDlgItemText(hDlg, IDC_ZONE_MAX, "31");
1010                    }
1011    
1012                    bvops = (profiles[config->profile].flags&PROFILE_BVOP) && config->use_bvop;
1013                    EnableDlgWindow(hDlg, IDC_ZONE_BVOPTHRESHOLD_S, bvops);
1014                    EnableDlgWindow(hDlg, IDC_ZONE_BVOPTHRESHOLD, bvops);
1015                    break;
1016    
1017            case IDD_DEBUG :
1018                    cpu_force                       = IsDlgChecked(hDlg, IDC_CPU_FORCE);
1019                    EnableDlgWindow(hDlg, IDC_CPU_MMX,              cpu_force);
1020                    EnableDlgWindow(hDlg, IDC_CPU_MMXEXT,   cpu_force);
1021                    EnableDlgWindow(hDlg, IDC_CPU_SSE,              cpu_force);
1022                    EnableDlgWindow(hDlg, IDC_CPU_SSE2,             cpu_force);
1023                    EnableDlgWindow(hDlg, IDC_CPU_3DNOW,    cpu_force);
1024                    EnableDlgWindow(hDlg, IDC_CPU_3DNOWEXT, cpu_force);
1025                    break;
1026            }
1027    }
1028    
1029    
1030    /* upload config data into dialog */
1031    static void adv_upload(HWND hDlg, int idd, CONFIG * config)
1032    {
1033            switch (idd)
1034            {
1035            case IDD_PROFILE :
1036                    SendDlgItemMessage(hDlg, IDC_PROFILE_PROFILE, CB_SETCURSEL, config->profile, 0);
1037    
1038                    SendDlgItemMessage(hDlg, IDC_QUANTTYPE, CB_SETCURSEL, config->quant_type, 0);
1039                    CheckDlg(hDlg, IDC_LUMMASK, config->lum_masking);
1040                    CheckDlg(hDlg, IDC_INTERLACING, config->interlacing);
1041                    CheckDlg(hDlg, IDC_QPEL, config->qpel);
1042                    CheckDlg(hDlg, IDC_GMC, config->gmc);
1043                    CheckDlg(hDlg, IDC_REDUCED, config->reduced_resolution);
1044                    CheckDlg(hDlg, IDC_BVOP, config->use_bvop);
1045    
1046                    SetDlgItemInt(hDlg, IDC_MAXBFRAMES, config->max_bframes, FALSE);
1047                    set_dlgitem_float(hDlg, IDC_BQUANTRATIO, config->bquant_ratio);
1048                    set_dlgitem_float(hDlg, IDC_BQUANTOFFSET, config->bquant_offset);
1049                    CheckDlg(hDlg, IDC_PACKED, config->packed);
1050                    CheckDlg(hDlg, IDC_CLOSEDGOV, config->closed_gov);
1051    
1052                    break;
1053            case IDD_AR:
1054                    CheckRadioButton(hDlg, IDC_AR, IDC_PAR, config->ar_mode == 0 ? IDC_PAR : IDC_AR);
1055                    SendDlgItemMessage(hDlg, IDC_ASPECT_RATIO, CB_SETCURSEL, (config->display_aspect_ratio), 0);
1056                    SetDlgItemInt(hDlg, IDC_ARX, config->ar_x, FALSE);
1057                    SetDlgItemInt(hDlg, IDC_ARY, config->ar_y, FALSE);
1058                    break;
1059    
1060            case IDD_LEVEL :
1061                    SendDlgItemMessage(hDlg, IDC_LEVEL_PROFILE, CB_SETCURSEL, config->profile, 0);
1062                    break;
1063    
1064            case IDD_RC_CBR :
1065                    SetDlgItemInt(hDlg, IDC_CBR_REACTIONDELAY, config->rc_reaction_delay_factor, FALSE);
1066                    SetDlgItemInt(hDlg, IDC_CBR_AVERAGINGPERIOD, config->rc_averaging_period, FALSE);
1067                    SetDlgItemInt(hDlg, IDC_CBR_BUFFER, config->rc_buffer, FALSE);
1068                    break;
1069    
1070            case IDD_RC_2PASS1 :
1071                    SetDlgItemText(hDlg, IDC_STATS, config->stats);
1072                    CheckDlg(hDlg, IDC_DISCARD1PASS, config->discard1pass);
1073                    CheckDlg(hDlg, IDC_FULL1PASS, config->full1pass);
1074                    break;
1075    
1076            case IDD_RC_2PASS2 :
1077                    SetDlgItemText(hDlg, IDC_STATS, config->stats);
1078                    SetDlgItemInt(hDlg, IDC_KFBOOST, config->keyframe_boost, FALSE);
1079                    SetDlgItemInt(hDlg, IDC_KFREDUCTION, config->kfreduction, FALSE);
1080    
1081                    SetDlgItemInt(hDlg, IDC_OVERFLOW_CONTROL_STRENGTH, config->overflow_control_strength, FALSE);
1082                    SetDlgItemInt(hDlg, IDC_OVERIMP, config->twopass_max_overflow_improvement, FALSE);
1083                    SetDlgItemInt(hDlg, IDC_OVERDEG, config->twopass_max_overflow_degradation, FALSE);
1084    
1085                    SetDlgItemInt(hDlg, IDC_CURVECOMPH, config->curve_compression_high, FALSE);
1086                    SetDlgItemInt(hDlg, IDC_CURVECOMPL, config->curve_compression_low, FALSE);
1087                    SetDlgItemInt(hDlg, IDC_MINKEY, config->kfthreshold, FALSE);
1088                    break;
1089    
1090            case IDD_BITRATE :
1091                    SendDlgItemMessage(hDlg, IDC_BITRATE_CFORMAT, CB_SETCURSEL, config->container_type, 0);
1092                    SetDlgItemInt(hDlg, IDC_BITRATE_TSIZE, config->target_size, FALSE);
1093                    SetDlgItemInt(hDlg, IDC_BITRATE_SSIZE, config->subtitle_size, FALSE);
1094    
1095                    SetDlgItemInt(hDlg, IDC_BITRATE_HOURS, config->hours, FALSE);
1096                    SetDlgItemInt(hDlg, IDC_BITRATE_MINUTES, config->minutes, FALSE);
1097                    SetDlgItemInt(hDlg, IDC_BITRATE_SECONDS, config->seconds, FALSE);
1098                    SendDlgItemMessage(hDlg, IDC_BITRATE_FPS, CB_SETCURSEL, config->fps, 0);
1099    
1100                    SendDlgItemMessage(hDlg, IDC_BITRATE_AFORMAT, CB_SETCURSEL, config->audio_type, 0);
1101                    CheckRadioButton(hDlg, IDC_BITRATE_AMODE_RATE, IDC_BITRATE_AMODE_SIZE, config->audio_mode == 0 ? IDC_BITRATE_AMODE_RATE : IDC_BITRATE_AMODE_SIZE);
1102                    SetDlgItemInt(hDlg, IDC_BITRATE_ARATE, config->audio_rate, FALSE);
1103                    SetDlgItemInt(hDlg, IDC_BITRATE_ASIZE, config->audio_size, FALSE);
1104                    break;
1105    
1106            case IDD_ZONE :
1107                    SetDlgItemInt(hDlg, IDC_ZONE_FRAME, config->zones[config->cur_zone].frame, FALSE);
1108    
1109                    CheckDlgButton(hDlg, IDC_ZONE_MODE_WEIGHT,   config->zones[config->cur_zone].mode == RC_ZONE_WEIGHT);
1110                    CheckDlgButton(hDlg, IDC_ZONE_MODE_QUANT,                config->zones[config->cur_zone].mode == RC_ZONE_QUANT);
1111    
1112                    set_dlgitem_float(hDlg, IDC_ZONE_WEIGHT, config->zones[config->cur_zone].weight);
1113                    set_dlgitem_float(hDlg, IDC_ZONE_QUANT, config->zones[config->cur_zone].quant);
1114    
1115                    CheckDlgButton(hDlg, IDC_ZONE_FORCEIVOP, config->zones[config->cur_zone].type==XVID_TYPE_IVOP);
1116                    CheckDlgButton(hDlg, IDC_ZONE_GREYSCALE, config->zones[config->cur_zone].greyscale);
1117                    CheckDlgButton(hDlg, IDC_ZONE_CHROMAOPT, config->zones[config->cur_zone].chroma_opt);
1118    
1119                    SetDlgItemInt(hDlg, IDC_ZONE_BVOPTHRESHOLD, config->zones[config->cur_zone].bvop_threshold, TRUE);
1120                    break;
1121    
1122            case IDD_MOTION :
1123                    SendDlgItemMessage(hDlg, IDC_MOTION, CB_SETCURSEL, config->motion_search, 0);
1124                    SendDlgItemMessage(hDlg, IDC_VHQ, CB_SETCURSEL, config->vhq_mode, 0);
1125                    CheckDlg(hDlg, IDC_CHROMAME, config->chromame);
1126                    CheckDlg(hDlg, IDC_CARTOON, config->cartoon_mode);
1127                    CheckDlg(hDlg, IDC_TURBO, config->turbo);
1128                    SetDlgItemInt(hDlg, IDC_FRAMEDROP, config->frame_drop_ratio, FALSE);
1129                    SetDlgItemInt(hDlg, IDC_MAXKEY, config->max_key_interval, FALSE);
1130                    break;
1131    
1132            case IDD_QUANT :
1133                    SetDlgItemInt(hDlg, IDC_MINIQUANT, config->min_iquant, FALSE);
1134                    SetDlgItemInt(hDlg, IDC_MAXIQUANT, config->max_iquant, FALSE);
1135                    SetDlgItemInt(hDlg, IDC_MINPQUANT, config->min_pquant, FALSE);
1136                    SetDlgItemInt(hDlg, IDC_MAXPQUANT, config->max_pquant, FALSE);
1137                    SetDlgItemInt(hDlg, IDC_MINBQUANT, config->min_bquant, FALSE);
1138                    SetDlgItemInt(hDlg, IDC_MAXBQUANT, config->max_bquant, FALSE);
1139                    CheckDlg(hDlg, IDC_TRELLISQUANT, config->trellis_quant);
1140                    break;
1141    
1142            case IDD_DEBUG :
1143                    CheckDlg(hDlg, IDC_CPU_MMX, (config->cpu & XVID_CPU_MMX));
1144                    CheckDlg(hDlg, IDC_CPU_MMXEXT, (config->cpu & XVID_CPU_MMXEXT));
1145                    CheckDlg(hDlg, IDC_CPU_SSE, (config->cpu & XVID_CPU_SSE));
1146                    CheckDlg(hDlg, IDC_CPU_SSE2, (config->cpu & XVID_CPU_SSE2));
1147                    CheckDlg(hDlg, IDC_CPU_3DNOW, (config->cpu & XVID_CPU_3DNOW));
1148                    CheckDlg(hDlg, IDC_CPU_3DNOWEXT, (config->cpu & XVID_CPU_3DNOWEXT));
1149    
1150                    CheckRadioButton(hDlg, IDC_CPU_AUTO, IDC_CPU_FORCE,
1151                            config->cpu & XVID_CPU_FORCE ? IDC_CPU_FORCE : IDC_CPU_AUTO );
1152    
1153                    SetDlgItemInt(hDlg, IDC_NUMTHREADS, config->num_threads, FALSE);
1154    
1155                    SendDlgItemMessage(hDlg, IDC_FOURCC, CB_SETCURSEL, config->fourcc_used, 0);
1156                    set_dlgitem_hex(hDlg, IDC_DEBUG, config->debug);
1157                    CheckDlg(hDlg, IDC_VOPDEBUG, config->vop_debug);
1158                    CheckDlg(hDlg, IDC_DISPLAY_STATUS, config->display_status);
1159                    break;
1160    
1161            case IDD_DEC :
1162                    CheckDlg(hDlg, IDC_DEC_DY,      pp_dy);
1163                    CheckDlg(hDlg, IDC_DEC_DUV,     pp_duv);
1164                    CheckDlg(hDlg, IDC_DEC_DR,      pp_dr);
1165                    CheckDlg(hDlg, IDC_DEC_FE,      pp_fe);
1166                    break;
1167            }
1168    }
1169    
1170    
1171    /* download config data from dialog */
1172    
1173    static void adv_download(HWND hDlg, int idd, CONFIG * config)
1174    {
1175            switch (idd)
1176            {
1177            case IDD_PROFILE :
1178                    config->profile = SendDlgItemMessage(hDlg, IDC_PROFILE_PROFILE, CB_GETCURSEL, 0, 0);
1179    
1180                    config->quant_type = SendDlgItemMessage(hDlg, IDC_QUANTTYPE, CB_GETCURSEL, 0, 0);
1181                    config->lum_masking = IsDlgChecked(hDlg, IDC_LUMMASK);
1182                    config->interlacing = IsDlgChecked(hDlg, IDC_INTERLACING);
1183                    config->qpel = IsDlgChecked(hDlg, IDC_QPEL);
1184                    config->gmc = IsDlgChecked(hDlg, IDC_GMC);
1185                    config->reduced_resolution = IsDlgChecked(hDlg, IDC_REDUCED);
1186    
1187                    config->use_bvop = IsDlgChecked(hDlg, IDC_BVOP);
1188                    config->max_bframes = config_get_uint(hDlg, IDC_MAXBFRAMES, config->max_bframes);
1189                    config->bquant_ratio = get_dlgitem_float(hDlg, IDC_BQUANTRATIO, config->bquant_ratio);
1190                    config->bquant_offset = get_dlgitem_float(hDlg, IDC_BQUANTOFFSET, config->bquant_offset);
1191                    config->packed = IsDlgChecked(hDlg, IDC_PACKED);
1192                    config->closed_gov = IsDlgChecked(hDlg, IDC_CLOSEDGOV);
1193                    break;
1194    
1195            case IDD_AR:
1196                    config->ar_mode = IsDlgChecked(hDlg, IDC_PAR) ? 0:1;
1197                    config->ar_x = config_get_uint(hDlg, IDC_ARX, config->ar_x);
1198                    config->ar_y = config_get_uint(hDlg, IDC_ARY, config->ar_y);
1199                    config->display_aspect_ratio = SendDlgItemMessage(hDlg, IDC_ASPECT_RATIO, CB_GETCURSEL, 0, 0);
1200                    if (config->display_aspect_ratio == 5) {
1201                            config->par_x = config_get_uint(hDlg, IDC_PARX, config->par_x);
1202                            config->par_y = config_get_uint(hDlg, IDC_PARY, config->par_y);
1203                    }
1204                    break;
1205    
1206            case IDD_LEVEL :
1207                    config->profile = SendDlgItemMessage(hDlg, IDC_LEVEL_PROFILE, CB_GETCURSEL, 0, 0);
1208                    break;
1209    
1210            case IDD_RC_CBR :
1211                    config->rc_reaction_delay_factor = config_get_uint(hDlg, IDC_CBR_REACTIONDELAY, config->rc_reaction_delay_factor);
1212                    config->rc_averaging_period = config_get_uint(hDlg, IDC_CBR_AVERAGINGPERIOD, config->rc_averaging_period);
1213                    config->rc_buffer = config_get_uint(hDlg, IDC_CBR_BUFFER, config->rc_buffer);
1214                    break;
1215    
1216            case IDD_RC_2PASS1 :
1217                    if (GetDlgItemText(hDlg, IDC_STATS, config->stats, MAX_PATH) == 0)
1218                            lstrcpy(config->stats, CONFIG_2PASS_FILE);
1219                    config->discard1pass = IsDlgChecked(hDlg, IDC_DISCARD1PASS);
1220                    config->full1pass = IsDlgChecked(hDlg, IDC_FULL1PASS);
1221                    break;
1222    
1223            case IDD_RC_2PASS2 :
1224                    if (GetDlgItemText(hDlg, IDC_STATS, config->stats, MAX_PATH) == 0)
1225                            lstrcpy(config->stats, CONFIG_2PASS_FILE);
1226    
1227                    config->keyframe_boost = GetDlgItemInt(hDlg, IDC_KFBOOST, NULL, FALSE);
1228                    config->kfreduction = GetDlgItemInt(hDlg, IDC_KFREDUCTION, NULL, FALSE);
1229                    CONSTRAINVAL(config->keyframe_boost, 0, 1000);
1230    
1231                    config->overflow_control_strength = GetDlgItemInt(hDlg, IDC_OVERFLOW_CONTROL_STRENGTH, NULL, FALSE);
1232                    config->twopass_max_overflow_improvement = config_get_uint(hDlg, IDC_OVERIMP, config->twopass_max_overflow_improvement);
1233                    config->twopass_max_overflow_degradation = config_get_uint(hDlg, IDC_OVERDEG, config->twopass_max_overflow_degradation);
1234                    CONSTRAINVAL(config->twopass_max_overflow_improvement, 1, 80);
1235                    CONSTRAINVAL(config->twopass_max_overflow_degradation, 1, 80);
1236                    CONSTRAINVAL(config->overflow_control_strength, 0, 100);
1237    
1238                    config->curve_compression_high = GetDlgItemInt(hDlg, IDC_CURVECOMPH, NULL, FALSE);
1239                    config->curve_compression_low = GetDlgItemInt(hDlg, IDC_CURVECOMPL, NULL, FALSE);
1240                    CONSTRAINVAL(config->curve_compression_high, 0, 100);
1241                    CONSTRAINVAL(config->curve_compression_low, 0, 100);
1242    
1243                    config->kfthreshold = config_get_uint(hDlg, IDC_MINKEY, config->kfthreshold);
1244    
1245                    break;
1246    
1247            case IDD_BITRATE :
1248                    config->container_type = SendDlgItemMessage(hDlg, IDC_BITRATE_CFORMAT, CB_GETCURSEL, 0, 0);
1249                    config->target_size = config_get_uint(hDlg, IDC_BITRATE_TSIZE, config->target_size);
1250                    config->subtitle_size = config_get_uint(hDlg, IDC_BITRATE_SSIZE, config->subtitle_size);
1251    
1252                    config->hours = config_get_uint(hDlg, IDC_BITRATE_HOURS, config->hours);
1253                    config->minutes = config_get_uint(hDlg, IDC_BITRATE_MINUTES, config->minutes);
1254                    config->seconds = config_get_uint(hDlg, IDC_BITRATE_SECONDS, config->seconds);
1255                    config->fps = SendDlgItemMessage(hDlg, IDC_BITRATE_FPS, CB_GETCURSEL, 0, 0);
1256    
1257                    config->audio_type = SendDlgItemMessage(hDlg, IDC_BITRATE_AFORMAT, CB_GETCURSEL, 0, 0);
1258                    config->audio_mode = IsDlgChecked(hDlg, IDC_BITRATE_AMODE_SIZE) ? 1 : 0 ;
1259                    config->audio_rate = config_get_uint(hDlg, IDC_BITRATE_ARATE, config->audio_rate);
1260                    config->audio_size = config_get_uint(hDlg, IDC_BITRATE_ASIZE, config->audio_size);
1261    
1262                    /* the main window uses "AVI bitrate/filesize" not "video bitrate/filesize",
1263                       so we have to compensate by frames * 24 bytes */
1264                    {
1265                            int frame_compensate = 24 * (int)(
1266                                    (3600*config->hours +
1267                                       60*config->minutes +
1268                                          config->seconds) * video_fps_list[config->fps].value) / 1024;
1269    
1270                            int bitrate_compensate = (int)(24 * video_fps_list[config->fps].value) / 125;
1271    
1272                            config->desired_size =
1273                                                    config_get_uint(hDlg, IDC_BITRATE_VSIZE, config->desired_size) + frame_compensate;
1274    
1275                            config->bitrate =
1276                                                    config_get_uint(hDlg, IDC_BITRATE_VRATE, config->bitrate) + bitrate_compensate;
1277                    }
1278                    break;
1279    
1280            case IDD_ZONE :
1281                    config->zones[config->cur_zone].frame = config_get_uint(hDlg, IDC_ZONE_FRAME, config->zones[config->cur_zone].frame);
1282    
1283                    if (IsDlgChecked(hDlg, IDC_ZONE_MODE_WEIGHT)) {
1284                            config->zones[config->cur_zone].mode = RC_ZONE_WEIGHT;
1285                    }else if (IsDlgChecked(hDlg, IDC_ZONE_MODE_QUANT)) {
1286                            config->zones[config->cur_zone].mode = RC_ZONE_QUANT;
1287                    }
1288    
1289                    config->zones[config->cur_zone].weight = get_dlgitem_float(hDlg, IDC_ZONE_WEIGHT, config->zones[config->cur_zone].weight);
1290                    config->zones[config->cur_zone].quant =  get_dlgitem_float(hDlg, IDC_ZONE_QUANT, config->zones[config->cur_zone].quant);
1291    
1292                    config->zones[config->cur_zone].type = IsDlgButtonChecked(hDlg, IDC_ZONE_FORCEIVOP)?XVID_TYPE_IVOP:XVID_TYPE_AUTO;
1293                    config->zones[config->cur_zone].greyscale = IsDlgButtonChecked(hDlg, IDC_ZONE_GREYSCALE);
1294                    config->zones[config->cur_zone].chroma_opt = IsDlgButtonChecked(hDlg, IDC_ZONE_CHROMAOPT);
1295    
1296                    config->zones[config->cur_zone].bvop_threshold = config_get_int(hDlg, IDC_ZONE_BVOPTHRESHOLD, config->zones[config->cur_zone].bvop_threshold);
1297                    break;
1298    
1299            case IDD_MOTION :
1300                    config->motion_search = SendDlgItemMessage(hDlg, IDC_MOTION, CB_GETCURSEL, 0, 0);
1301                    config->vhq_mode = SendDlgItemMessage(hDlg, IDC_VHQ, CB_GETCURSEL, 0, 0);
1302                    config->chromame = IsDlgChecked(hDlg, IDC_CHROMAME);
1303                    config->cartoon_mode = IsDlgChecked(hDlg, IDC_CARTOON);
1304                    config->turbo = IsDlgChecked(hDlg, IDC_TURBO);
1305    
1306                    config->frame_drop_ratio = config_get_uint(hDlg, IDC_FRAMEDROP, config->frame_drop_ratio);
1307    
1308                    config->max_key_interval = config_get_uint(hDlg, IDC_MAXKEY, config->max_key_interval);
1309                    break;
1310    
1311            case IDD_QUANT :
1312                    config->min_iquant = config_get_uint(hDlg, IDC_MINIQUANT, config->min_iquant);
1313                    config->max_iquant = config_get_uint(hDlg, IDC_MAXIQUANT, config->max_iquant);
1314                    config->min_pquant = config_get_uint(hDlg, IDC_MINPQUANT, config->min_pquant);
1315                    config->max_pquant = config_get_uint(hDlg, IDC_MAXPQUANT, config->max_pquant);
1316                    config->min_bquant = config_get_uint(hDlg, IDC_MINBQUANT, config->min_bquant);
1317                    config->max_bquant = config_get_uint(hDlg, IDC_MAXBQUANT, config->max_bquant);
1318    
1319                    CONSTRAINVAL(config->min_iquant, 1, 31);
1320                    CONSTRAINVAL(config->max_iquant, config->min_iquant, 31);
1321                    CONSTRAINVAL(config->min_pquant, 1, 31);
1322                    CONSTRAINVAL(config->max_pquant, config->min_pquant, 31);
1323                    CONSTRAINVAL(config->min_bquant, 1, 31);
1324                    CONSTRAINVAL(config->max_bquant, config->min_bquant, 31);
1325    
1326                    config->trellis_quant = IsDlgChecked(hDlg, IDC_TRELLISQUANT);
1327                    break;
1328    
1329            case IDD_DEBUG :
1330                    config->cpu = 0;
1331                    config->cpu |= IsDlgChecked(hDlg, IDC_CPU_MMX)    ? XVID_CPU_MMX : 0;
1332                    config->cpu |= IsDlgChecked(hDlg, IDC_CPU_MMXEXT)   ? XVID_CPU_MMXEXT : 0;
1333                    config->cpu |= IsDlgChecked(hDlg, IDC_CPU_SSE)    ? XVID_CPU_SSE : 0;
1334                    config->cpu |= IsDlgChecked(hDlg, IDC_CPU_SSE2)  ? XVID_CPU_SSE2 : 0;
1335                    config->cpu |= IsDlgChecked(hDlg, IDC_CPU_3DNOW)        ? XVID_CPU_3DNOW : 0;
1336                    config->cpu |= IsDlgChecked(hDlg, IDC_CPU_3DNOWEXT) ? XVID_CPU_3DNOWEXT : 0;
1337                    config->cpu |= IsDlgChecked(hDlg, IDC_CPU_FORCE)        ? XVID_CPU_FORCE : 0;
1338    
1339                    config->num_threads = config_get_uint(hDlg, IDC_NUMTHREADS, config->num_threads);
1340    
1341                    config->fourcc_used = SendDlgItemMessage(hDlg, IDC_FOURCC, CB_GETCURSEL, 0, 0);
1342                    config->debug = get_dlgitem_hex(hDlg, IDC_DEBUG, config->debug);
1343                    config->vop_debug = IsDlgChecked(hDlg, IDC_VOPDEBUG);
1344                    config->display_status = IsDlgChecked(hDlg, IDC_DISPLAY_STATUS);
1345                    break;
1346    
1347            case IDD_DEC :
1348                    pp_dy = IsDlgChecked(hDlg, IDC_DEC_DY);
1349                    pp_duv = IsDlgChecked(hDlg, IDC_DEC_DUV);
1350                    pp_dr = IsDlgChecked(hDlg, IDC_DEC_DR);
1351                    pp_fe = IsDlgChecked(hDlg, IDC_DEC_FE);
1352                    break;
1353            }
1354    }
1355    
1356    
1357    
1358    /* advanced dialog proc */
1359    
1360    static BOOL CALLBACK adv_proc(HWND hDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
1361    {
1362            PROPSHEETINFO *psi;
1363    
1364            psi = (PROPSHEETINFO*)GetWindowLong(hDlg, GWL_USERDATA);
1365    
1366            switch (uMsg)
1367            {
1368            case WM_INITDIALOG :
1369                    psi = (PROPSHEETINFO*) ((LPPROPSHEETPAGE)lParam)->lParam;
1370                    SetWindowLong(hDlg, GWL_USERDATA, (LPARAM)psi);
1371    
1372                    if (g_hTooltip)
1373                            EnumChildWindows(hDlg, enum_tooltips, 0);
1374    
1375                    adv_init(hDlg, psi->idd, psi->config);
1376                    break;
1377    
1378            case WM_COMMAND :
1379                    if (HIWORD(wParam) == BN_CLICKED)
1380                    {
1381                            switch (LOWORD(wParam))
1382                            {
1383                            case IDC_BVOP :
1384                            case IDC_ZONE_MODE_WEIGHT :
1385                            case IDC_ZONE_MODE_QUANT :
1386                            case IDC_ZONE_BVOPTHRESHOLD_ENABLE :
1387                            case IDC_CPU_AUTO :
1388                            case IDC_CPU_FORCE :
1389                            case IDC_AR :
1390                            case IDC_PAR :
1391                            case IDC_BITRATE_AMODE_RATE :
1392                            case IDC_BITRATE_AMODE_SIZE :
1393                                    adv_mode(hDlg, psi->idd, psi->config);
1394                                    break;
1395    
1396                            case IDC_BITRATE_SSELECT :
1397                            case IDC_BITRATE_ASELECT :
1398                                    {
1399                                    OPENFILENAME ofn;
1400                                    char filename[MAX_PATH] = "";
1401    
1402                                    memset(&ofn, 0, sizeof(OPENFILENAME));
1403                                    ofn.lStructSize = sizeof(OPENFILENAME);
1404    
1405                                    ofn.hwndOwner = hDlg;
1406                                    if (LOWORD(wParam)==IDC_BITRATE_SSELECT) {
1407                                            ofn.lpstrFilter = "Subtitle files (*.sub, *.ssa)\0*.sub;*.ssa\0All files (*.*)\0*.*\0\0";
1408                                    }else{
1409                                            ofn.lpstrFilter = "Audio files (*.mp3, *.ac3)\0*.mp3; *.ac3\0All files (*.*)\0*.*\0\0";
1410                                    }
1411    
1412                                    ofn.lpstrFile = filename;
1413                                    ofn.nMaxFile = MAX_PATH;
1414                                    ofn.Flags = OFN_PATHMUSTEXIST | OFN_FILEMUSTEXIST;
1415    
1416                                    if (GetOpenFileName(&ofn)) {
1417                                            HANDLE hFile;
1418                                            DWORD filesize;
1419    
1420                                            if ((hFile = CreateFile(filename, GENERIC_READ, 0, 0, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0)) == INVALID_HANDLE_VALUE ||
1421                                                    (filesize = GetFileSize(hFile, NULL)) == INVALID_FILE_SIZE) {
1422                                                    MessageBox(hDlg, "Could not get file size", "Error", 0);
1423                                            }else{
1424                                                    SetDlgItemInt(hDlg,
1425                                                                    LOWORD(wParam)==IDC_BITRATE_SSELECT? IDC_BITRATE_SSIZE : IDC_BITRATE_ASIZE,
1426                                                                    filesize / 1024, FALSE);
1427                                                    CloseHandle(hFile);
1428                                            }
1429                                    }
1430                                    }
1431                                    break;
1432    
1433                            case IDC_QUANTMATRIX :
1434                                    DialogBoxParam(g_hInst, MAKEINTRESOURCE(IDD_QUANTMATRIX), hDlg, quantmatrix_proc, (LPARAM)psi->config);
1435                                    break;
1436    
1437                            case IDC_STATS_BROWSE :
1438                            {
1439                                    OPENFILENAME ofn;
1440                                    char tmp[MAX_PATH];
1441    
1442                                    GetDlgItemText(hDlg, IDC_STATS, tmp, MAX_PATH);
1443    
1444                                    memset(&ofn, 0, sizeof(OPENFILENAME));
1445                                    ofn.lStructSize = sizeof(OPENFILENAME);
1446    
1447                                    ofn.hwndOwner = hDlg;
1448                                    ofn.lpstrFilter = "bitrate curve (*.pass)\0*.pass\0All files (*.*)\0*.*\0\0";
1449                                    ofn.lpstrFile = tmp;
1450                                    ofn.nMaxFile = MAX_PATH;
1451                                    ofn.Flags = OFN_PATHMUSTEXIST;
1452    
1453                                    if (psi->idd == IDD_RC_2PASS1) {
1454                                            ofn.Flags |= OFN_OVERWRITEPROMPT;
1455                                    }else{
1456                                            ofn.Flags |= OFN_FILEMUSTEXIST;
1457                                    }
1458    
1459                                    if ((psi->idd==IDD_RC_2PASS1 && GetSaveFileName(&ofn)) ||
1460                                            (psi->idd==IDD_RC_2PASS2 && GetOpenFileName(&ofn))) {
1461                                            SetDlgItemText(hDlg, IDC_STATS, tmp);
1462                                    }
1463                                    }
1464                                    break;
1465    
1466                            case IDC_ZONE_FETCH :
1467                                    SetDlgItemInt(hDlg, IDC_ZONE_FRAME, psi->config->ci.ciActiveFrame, FALSE);
1468                                    break;
1469    
1470                            case IDC_AR_DEFAULT:
1471                                    CheckRadioButton(hDlg, IDC_AR, IDC_PAR, IDC_PAR);
1472                                    SendDlgItemMessage(hDlg, IDC_ASPECT_RATIO, CB_SETCURSEL, 0, 0);
1473                                    adv_mode(hDlg, psi->idd, psi->config);
1474                                    break;
1475                            case IDC_AR_4_3:
1476                                    SetDlgItemInt(hDlg, IDC_ARX, 4, FALSE);
1477                                    SetDlgItemInt(hDlg, IDC_ARY, 3, FALSE);
1478                                    CheckRadioButton(hDlg, IDC_AR, IDC_PAR, IDC_AR);
1479                                    adv_mode(hDlg, psi->idd, psi->config);
1480                                    break;
1481                            case IDC_AR_16_9:
1482                                    SetDlgItemInt(hDlg, IDC_ARX, 16, FALSE);
1483                                    SetDlgItemInt(hDlg, IDC_ARY, 9, FALSE);
1484                                    CheckRadioButton(hDlg, IDC_AR, IDC_PAR, IDC_AR);
1485                                    adv_mode(hDlg, psi->idd, psi->config);
1486                                    break;
1487                            case IDC_AR_235_100:
1488                                    SetDlgItemInt(hDlg, IDC_ARX, 235, FALSE);
1489                                    SetDlgItemInt(hDlg, IDC_ARY, 100, FALSE);
1490                                    CheckRadioButton(hDlg, IDC_AR, IDC_PAR, IDC_AR);
1491                                    adv_mode(hDlg, psi->idd, psi->config);
1492                                    break;
1493    
1494                            default :
1495                                    return TRUE;
1496                            }
1497                    }else if ((HIWORD(wParam) == CBN_EDITCHANGE || HIWORD(wParam)==CBN_SELCHANGE) &&
1498                            (LOWORD(wParam)==IDC_BITRATE_TSIZE ||
1499                             LOWORD(wParam)==IDC_BITRATE_ARATE )) {
1500    
1501                            adv_mode(hDlg, psi->idd, psi->config);
1502    
1503                    }else if (HIWORD(wParam) == LBN_SELCHANGE &&
1504                            (LOWORD(wParam) == IDC_PROFILE_PROFILE ||
1505                             LOWORD(wParam) == IDC_LEVEL_PROFILE ||
1506                             LOWORD(wParam) == IDC_QUANTTYPE ||
1507                             LOWORD(wParam) == IDC_ASPECT_RATIO ||
1508                             LOWORD(wParam) == IDC_BITRATE_CFORMAT ||
1509                             LOWORD(wParam) == IDC_BITRATE_AFORMAT ||
1510                             LOWORD(wParam) == IDC_BITRATE_FPS)) {
1511                            adv_mode(hDlg, psi->idd, psi->config);
1512                    }else if (HIWORD(wParam) == EN_UPDATE && (LOWORD(wParam)==IDC_ZONE_WEIGHT || LOWORD(wParam)==IDC_ZONE_QUANT)) {
1513    
1514                            SendDlgItemMessage(hDlg, IDC_ZONE_SLIDER, TBM_SETPOS, TRUE,
1515                                            get_dlgitem_float(hDlg, LOWORD(wParam), 100));
1516    
1517                    } else if (HIWORD(wParam) == EN_UPDATE && (LOWORD(wParam)==IDC_PARX || LOWORD(wParam)==IDC_PARY)) {
1518    
1519                            if (5 == SendDlgItemMessage(hDlg, IDC_ASPECT_RATIO, CB_GETCURSEL, 0, 0)) {
1520                                    if(LOWORD(wParam)==IDC_PARX)
1521                                            psi->config->par_x = config_get_uint(hDlg, LOWORD(wParam), psi->config->par_x);
1522                                    else
1523                                            psi->config->par_y = config_get_uint(hDlg, LOWORD(wParam), psi->config->par_y);
1524                            }
1525                    } else if (HIWORD(wParam) == EN_UPDATE &&
1526                            (LOWORD(wParam)==IDC_BITRATE_SSIZE ||
1527                             LOWORD(wParam)==IDC_BITRATE_HOURS ||
1528                             LOWORD(wParam)==IDC_BITRATE_MINUTES ||
1529                             LOWORD(wParam)==IDC_BITRATE_SECONDS ||
1530                             LOWORD(wParam)==IDC_BITRATE_ASIZE)) {
1531                            adv_mode(hDlg, psi->idd, psi->config);
1532                    } else
1533                            return 0;
1534                    break;
1535    
1536            case WM_HSCROLL :
1537                    if((HWND)lParam == GetDlgItem(hDlg, IDC_ZONE_SLIDER)) {
1538                            int idc = IsDlgChecked(hDlg, IDC_ZONE_MODE_WEIGHT) ? IDC_ZONE_WEIGHT : IDC_ZONE_QUANT;
1539                            set_dlgitem_float(hDlg, idc, SendMessage((HWND)lParam, TBM_GETPOS, 0, 0) );
1540                            break;
1541                    }
1542                    return 0;
1543    
1544    
1545            case WM_NOTIFY :
1546                    switch (((NMHDR *)lParam)->code)
1547                    {
1548                    case PSN_SETACTIVE :
1549                            DPRINTF("PSN_SET");
1550                            adv_upload(hDlg, psi->idd, psi->config);
1551                            adv_mode(hDlg, psi->idd, psi->config);
1552                            SetWindowLong(hDlg, DWL_MSGRESULT, FALSE);
1553                            break;
1554    
1555                    case PSN_KILLACTIVE :
1556                            DPRINTF("PSN_KILL");
1557                            adv_download(hDlg, psi->idd, psi->config);
1558                            SetWindowLong(hDlg, DWL_MSGRESULT, FALSE);
1559                            break;
1560    
1561                    case PSN_APPLY :
1562                            DPRINTF("PSN_APPLY");
1563                            psi->config->save = TRUE;
1564                            SetWindowLong(hDlg, DWL_MSGRESULT, FALSE);
1565                            break;
1566                    }
1567                    break;
1568    
1569            default :
1570                    return 0;
1571            }
1572    
1573            return 1;
1574    }
1575    
1576    
1577    
1578    
1579    /* load advanced options property sheet
1580      returns true, if the user accepted the changes
1581      or fasle if changes were canceled.
1582    
1583      */
1584    
1585    #ifndef PSH_NOCONTEXTHELP
1586    #define PSH_NOCONTEXTHELP 0x02000000
1587    #endif
1588    
1589    static BOOL adv_dialog(HWND hParent, CONFIG * config, const int * dlgs, int size)
1590    {
1591            PROPSHEETINFO psi[6];
1592            PROPSHEETPAGE psp[6];
1593            PROPSHEETHEADER psh;
1594            CONFIG temp;
1595            int i;
1596    
1597            config->save = FALSE;
1598            memcpy(&temp, config, sizeof(CONFIG));
1599    
1600            for (i=0; i<size; i++)
1601            {
1602                    psp[i].dwSize = sizeof(PROPSHEETPAGE);
1603                    psp[i].dwFlags = 0;
1604                    psp[i].hInstance = g_hInst;
1605                    psp[i].pfnDlgProc = adv_proc;
1606                    psp[i].lParam = (LPARAM)&psi[i];
1607                    psp[i].pfnCallback = NULL;
1608                    psp[i].pszTemplate = MAKEINTRESOURCE(dlgs[i]);
1609    
1610                    psi[i].idd = dlgs[i];
1611                    psi[i].config = &temp;
1612            }
1613    
1614            psh.dwSize = sizeof(PROPSHEETHEADER);
1615            psh.dwFlags = PSH_PROPSHEETPAGE | PSH_NOAPPLYNOW | PSH_NOCONTEXTHELP;
1616            psh.hwndParent = hParent;
1617            psh.hInstance = g_hInst;
1618            psh.pszCaption = (LPSTR) "XviD Configuration";
1619            psh.nPages = size;
1620            psh.nStartPage = 0;
1621            psh.ppsp = (LPCPROPSHEETPAGE)&psp;
1622            psh.pfnCallback = NULL;
1623            PropertySheet(&psh);
1624    
1625            if (temp.save)
1626                    memcpy(config, &temp, sizeof(CONFIG));
1627    
1628            return temp.save;
1629    }
1630    
1631    /* ===================================================================================== */
1632    /* MAIN DIALOG ========================================================================= */
1633    /* ===================================================================================== */
1634    
1635    
1636    static void main_insert_zone(HWND hDlg, zone_t * s, int i, BOOL insert)
1637    {
1638            char tmp[32];
1639    
1640            wsprintf(tmp,"%i",s->frame);
1641    
1642            if (insert) {
1643                    LVITEM lvi;
1644    
1645                    lvi.mask = LVIF_TEXT | LVIF_IMAGE | LVIF_PARAM | LVIF_STATE;
1646                    lvi.state = 0;
1647                    lvi.stateMask = 0;
1648                    lvi.iImage = 0;
1649                    lvi.pszText = tmp;
1650                    lvi.cchTextMax = strlen(tmp);
1651                    lvi.iItem = i;
1652                    lvi.iSubItem = 0;
1653                    ListView_InsertItem(hDlg, &lvi);
1654            }else{
1655                    ListView_SetItemText(hDlg, i, 0, tmp);
1656            }
1657    
1658            if (s->mode == RC_ZONE_WEIGHT) {
1659                    sprintf(tmp,"W %.2f",(float)s->weight/100);
1660            }else if (s->mode == RC_ZONE_QUANT) {
1661                    sprintf(tmp,"Q %.2f",(float)s->quant/100);
1662            }else {
1663                    strcpy(tmp,"EXT");
1664            }
1665            ListView_SetItemText(hDlg, i, 1, tmp);
1666    
1667            tmp[0] = '\0';
1668            if (s->type==XVID_TYPE_IVOP)
1669                    strcat(tmp, "K ");
1670    
1671            if (s->greyscale)
1672                    strcat(tmp, "G ");
1673    
1674            if (s->chroma_opt)
1675                    strcat(tmp, "C ");
1676    
1677            ListView_SetItemText(hDlg, i, 2, tmp);
1678    }
1679    
1680    
1681    static void main_mode(HWND hDlg, CONFIG * config)
1682    {
1683            const int profile = SendDlgItemMessage(hDlg, IDC_PROFILE, CB_GETCURSEL, 0, 0);
1684            const int rc_mode = SendDlgItemMessage(hDlg, IDC_MODE, CB_GETCURSEL, 0, 0);
1685            /* enable target rate/size control only for 1pass and 2pass  modes*/
1686            const int target_en = rc_mode==RC_MODE_1PASS || rc_mode==RC_MODE_2PASS2;
1687            const int target_en_slider = rc_mode==RC_MODE_1PASS ||
1688                    (rc_mode==RC_MODE_2PASS2 && config->use_2pass_bitrate);
1689    
1690            char buf[16];
1691            int max;
1692    
1693            g_use_bitrate = config->use_2pass_bitrate;
1694    
1695            if (g_use_bitrate) {
1696                    SetDlgItemText(hDlg, IDC_BITRATE_S, "Target bitrate (kbps):");
1697    
1698                    wsprintf(buf, "%i kbps", DEFAULT_MIN_KBPS);
1699                    SetDlgItemText(hDlg, IDC_BITRATE_MIN, buf);
1700    
1701                    max = profiles[profile].max_bitrate;
1702                    if (max == 0) max = DEFAULT_MAX_KBPS;
1703                    wsprintf(buf, "%i kbps", max);
1704                    SetDlgItemText(hDlg, IDC_BITRATE_MAX, buf);
1705    
1706                    SendDlgItemMessage(hDlg, IDC_SLIDER, TBM_SETRANGE, TRUE, MAKELONG(DEFAULT_MIN_KBPS, max));
1707                    SendDlgItemMessage(hDlg, IDC_SLIDER, TBM_SETPOS, TRUE,
1708                                                    config_get_uint(hDlg, IDC_BITRATE, DEFAULT_MIN_KBPS) );
1709    
1710            } else if (rc_mode==RC_MODE_2PASS2) {
1711                    SetDlgItemText(hDlg, IDC_BITRATE_S, "Target size (kbytes):");
1712            } else if (rc_mode==RC_MODE_1PASS) {
1713                    SetDlgItemText(hDlg, IDC_BITRATE_S, "Target quantizer:");
1714                    SendDlgItemMessage(hDlg, IDC_SLIDER, TBM_SETRANGE, TRUE, MAKELONG(100, 3100));
1715                    SendDlgItemMessage(hDlg, IDC_SLIDER, TBM_SETPOS, TRUE,
1716                                                            get_dlgitem_float(hDlg, IDC_BITRATE, DEFAULT_QUANT ));
1717                    SetDlgItemText(hDlg, IDC_BITRATE_MIN, "1 (maximum quality)");
1718                    SetDlgItemText(hDlg, IDC_BITRATE_MAX, "(smallest file) 31");
1719    
1720            }
1721    
1722            EnableDlgWindow(hDlg, IDC_BITRATE_S, target_en);
1723            EnableDlgWindow(hDlg, IDC_BITRATE, target_en);
1724            EnableDlgWindow(hDlg, IDC_BITRATE_ADV, target_en);
1725    
1726            EnableDlgWindow(hDlg, IDC_BITRATE_MIN, target_en_slider);
1727            EnableDlgWindow(hDlg, IDC_BITRATE_MAX, target_en_slider);
1728            EnableDlgWindow(hDlg, IDC_SLIDER, target_en_slider);
1729    }
1730    
1731    
1732    static void main_upload(HWND hDlg, CONFIG * config)
1733    {
1734    
1735            SendDlgItemMessage(hDlg, IDC_PROFILE, CB_SETCURSEL, config->profile, 0);
1736            SendDlgItemMessage(hDlg, IDC_MODE, CB_SETCURSEL, config->mode, 0);
1737    
1738            g_use_bitrate = config->use_2pass_bitrate;
1739    
1740            if (g_use_bitrate) {
1741                    SetDlgItemInt(hDlg, IDC_BITRATE, config->bitrate, FALSE);
1742            } else if (config->mode == RC_MODE_2PASS2) {
1743                    SetDlgItemInt(hDlg, IDC_BITRATE, config->desired_size, FALSE);
1744            } else if (config->mode == RC_MODE_1PASS) {
1745                    set_dlgitem_float(hDlg, IDC_BITRATE, config->desired_quant);
1746            }
1747    
1748            zones_update(hDlg, config);
1749    }
1750    
1751    
1752    /* downloads data from main dialog */
1753    static void main_download(HWND hDlg, CONFIG * config)
1754    {
1755            config->profile = SendDlgItemMessage(hDlg, IDC_PROFILE, CB_GETCURSEL, 0, 0);
1756            config->mode = SendDlgItemMessage(hDlg, IDC_MODE, CB_GETCURSEL, 0, 0);
1757    
1758            if (g_use_bitrate) {
1759                    config->bitrate = config_get_uint(hDlg, IDC_BITRATE, config->bitrate);
1760            } else if (config->mode == RC_MODE_2PASS2) {
1761                    config->desired_size = config_get_uint(hDlg, IDC_BITRATE, config->desired_size);
1762            } else if (config->mode == RC_MODE_1PASS) {
1763                    config->desired_quant = get_dlgitem_float(hDlg, IDC_BITRATE, config->desired_quant);
1764            }
1765    }
1766    
1767    
1768    /* main dialog proc */
1769    
1770    static const int profile_dlgs[] = { IDD_PROFILE, IDD_LEVEL, IDD_AR };
1771    static const int single_dlgs[] = { IDD_RC_CBR };
1772    static const int pass1_dlgs[] = { IDD_RC_2PASS1 };
1773    static const int pass2_dlgs[] = { IDD_RC_2PASS2 };
1774    static const int bitrate_dlgs[] = { IDD_BITRATE };
1775    static const int zone_dlgs[] = { IDD_ZONE };
1776    static const int decoder_dlgs[] = { IDD_DEC };
1777    static const int adv_dlgs[] = { IDD_MOTION, IDD_QUANT, IDD_DEBUG};
1778    
1779    
1780    BOOL CALLBACK main_proc(HWND hDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
1781    {
1782            CONFIG* config = (CONFIG*)GetWindowLong(hDlg, GWL_USERDATA);
1783            unsigned int i;
1784    
1785            switch (uMsg)
1786            {
1787            case WM_INITDIALOG :
1788                    SetWindowLong(hDlg, GWL_USERDATA, lParam);
1789                    config = (CONFIG*)lParam;
1790    
1791                    for (i=0; i<sizeof(profiles)/sizeof(profile_t); i++)
1792                            SendDlgItemMessage(hDlg, IDC_PROFILE, CB_ADDSTRING, 0, (LPARAM)profiles[i].name);
1793    
1794                    SendDlgItemMessage(hDlg, IDC_MODE, CB_ADDSTRING, 0, (LPARAM)"Single pass");
1795                    SendDlgItemMessage(hDlg, IDC_MODE, CB_ADDSTRING, 0, (LPARAM)"Twopass - 1st pass");
1796                    SendDlgItemMessage(hDlg, IDC_MODE, CB_ADDSTRING, 0, (LPARAM)"Twopass - 2nd pass");
1797    #ifdef _DEBUG
1798                    SendDlgItemMessage(hDlg, IDC_MODE, CB_ADDSTRING, 0, (LPARAM)"Null test speed");
1799    #endif
1800    
1801                    InitCommonControls();
1802    
1803                    if ((g_hTooltip = CreateWindow(TOOLTIPS_CLASS, NULL, TTS_ALWAYSTIP,
1804                                    CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT,
1805                                    NULL, NULL, g_hInst, NULL)))
1806                    {
1807                            SetWindowPos(g_hTooltip, HWND_TOPMOST, 0, 0, 0, 0, SWP_NOMOVE|SWP_NOSIZE|SWP_NOACTIVATE);
1808                            SendMessage(g_hTooltip, TTM_SETDELAYTIME, TTDT_AUTOMATIC, MAKELONG(1500, 0));
1809                            SendMessage(g_hTooltip, TTM_SETMAXTIPWIDTH, 0, 400);
1810    
1811                            EnumChildWindows(hDlg, enum_tooltips, 0);
1812                    }
1813    
1814                    SetClassLong(GetDlgItem(hDlg, IDC_BITRATE_S), GCL_HCURSOR, (LONG)LoadCursor(NULL, IDC_HAND));
1815    
1816                    {
1817                            DWORD ext_style = ListView_GetExtendedListViewStyle(GetDlgItem(hDlg,IDC_ZONES));
1818                            ext_style |= LVS_EX_FULLROWSELECT | LVS_EX_FLATSB ;
1819                            ListView_SetExtendedListViewStyle(GetDlgItem(hDlg,IDC_ZONES), ext_style);
1820                    }
1821    
1822                    {
1823                            typedef struct {
1824                                    char * name;
1825                                    int value;
1826                            } char_int_t;
1827    
1828                            const static char_int_t columns[] = {
1829                                    {"Frame #",      64},
1830                                    {"Weight/Quant",  82},
1831                                    {"Modifiers",   120}};
1832    
1833                            LVCOLUMN lvc;
1834                            int i;
1835    
1836                            /* Initialize the LVCOLUMN structure.  */
1837                            lvc.mask = LVCF_FMT | LVCF_WIDTH | LVCF_TEXT | LVCF_SUBITEM;
1838                            lvc.fmt = LVCFMT_LEFT;
1839    
1840                            /* Add the columns.  */
1841                            for (i=0; i<sizeof(columns)/sizeof(char_int_t); i++) {
1842                                    lvc.pszText = (char*)columns[i].name;
1843                                    lvc.cchTextMax = strlen(columns[i].name);
1844                                    lvc.iSubItem = i;
1845                                    lvc.cx = columns[i].value;  /* column width, pixels */
1846                                    ListView_InsertColumn(GetDlgItem(hDlg,IDC_ZONES), i, &lvc);
1847                            }
1848                    }
1849    
1850                    /* XXX: main_mode needs RC_MODE_xxx, main_upload needs g_use_bitrate set correctly... */
1851                    main_upload(hDlg, config);
1852                    main_mode(hDlg, config);
1853                    main_upload(hDlg, config);
1854                    break;
1855    
1856            case WM_NOTIFY :
1857                    {
1858                            NMHDR * n = (NMHDR*)lParam;
1859    
1860                            if (n->code == NM_DBLCLK) {
1861                                     NMLISTVIEW * nmlv = (NMLISTVIEW*) lParam;
1862                                     config->cur_zone = nmlv->iItem;
1863    
1864                                     main_download(hDlg, config);
1865                                     if (config->cur_zone >= 0 && adv_dialog(hDlg, config, zone_dlgs, sizeof(zone_dlgs)/sizeof(int))) {
1866                                             zones_update(hDlg, config);
1867                                     }
1868                                     break;
1869                            }
1870    
1871                    break;
1872                    }
1873    
1874            case WM_COMMAND :
1875                    if (HIWORD(wParam) == BN_CLICKED) {
1876    
1877                            switch(LOWORD(wParam)) {
1878                            case IDC_PROFILE_ADV :
1879                                    main_download(hDlg, config);
1880                                    adv_dialog(hDlg, config, profile_dlgs, sizeof(profile_dlgs)/sizeof(int));
1881    
1882                                    SendDlgItemMessage(hDlg, IDC_PROFILE, CB_SETCURSEL, config->profile, 0);
1883                                    main_mode(hDlg, config);
1884                                    break;
1885    
1886                            case IDC_MODE_ADV :
1887                                    main_download(hDlg, config);
1888                                    if (config->mode == RC_MODE_1PASS) {
1889                                            adv_dialog(hDlg, config, single_dlgs, sizeof(single_dlgs)/sizeof(int));
1890                                    }else if (config->mode == RC_MODE_2PASS1) {
1891                                            adv_dialog(hDlg, config, pass1_dlgs, sizeof(pass1_dlgs)/sizeof(int));
1892                                    }else if (config->mode == RC_MODE_2PASS2) {
1893                                            adv_dialog(hDlg, config, pass2_dlgs, sizeof(pass2_dlgs)/sizeof(int));
1894                                    }
1895                                    break;
1896    
1897                            case IDC_BITRATE_S :
1898                                    /* alternate between bitrate/desired_length metrics */
1899                                    main_download(hDlg, config);
1900                                    config->use_2pass_bitrate = !config->use_2pass_bitrate;
1901                                    main_mode(hDlg, config);
1902                                    main_upload(hDlg, config);
1903                                    break;
1904    
1905                            case IDC_BITRATE_ADV :
1906                                    main_download(hDlg, config);
1907                                    adv_dialog(hDlg, config, bitrate_dlgs, sizeof(bitrate_dlgs)/sizeof(int));
1908                                    main_mode(hDlg, config);
1909                                    main_upload(hDlg, config);
1910                                    break;
1911    
1912                            case IDC_DECODER :
1913                                    main_download(hDlg, config);
1914                                    adv_dialog(hDlg, config, decoder_dlgs, sizeof(decoder_dlgs)/sizeof(int));
1915                                    main_mode(hDlg, config);
1916                                    break;
1917    
1918                            case IDC_ADD :
1919                            {
1920                                    int i, sel, new_frame;
1921    
1922                                    if (config->num_zones >= MAX_ZONES) {
1923                                            MessageBox(hDlg, "Exceeded maximum number of zones.\nIncrease config.h:MAX_ZONES and rebuild.", "Warning", 0);
1924                                            break;
1925                                    }
1926    
1927                                    sel = ListView_GetNextItem(GetDlgItem(hDlg, IDC_ZONES), -1, LVNI_SELECTED);
1928    
1929                                    if (sel<0) {
1930                                            if (config->ci_valid && config->ci.ciActiveFrame>0) {
1931                                                    for(sel=0; sel<config->num_zones-1 && config->zones[sel].frame<config->ci.ciActiveFrame; sel++) ;
1932                                                    sel--;
1933                                                    new_frame = config->ci.ciActiveFrame;
1934                                            }else{
1935                                                    sel = config->num_zones-1;
1936                                                    new_frame = sel<0 ? 0 : config->zones[sel].frame + 1;
1937                                            }
1938                                    }else{
1939                                            new_frame = config->zones[sel].frame + 1;
1940                                    }
1941    
1942                                    for(i=config->num_zones-1; i>sel; i--) {
1943                                            config->zones[i+1] = config->zones[i];
1944                                    }
1945                                    config->num_zones++;
1946                                    config->zones[sel+1].frame = new_frame;
1947                                    config->zones[sel+1].mode = RC_ZONE_WEIGHT;
1948                                    config->zones[sel+1].weight = 100;
1949                                    config->zones[sel+1].quant = 500;
1950                                    config->zones[sel+1].type = XVID_TYPE_AUTO;
1951                                    config->zones[sel+1].greyscale = 0;
1952                                    config->zones[sel+1].chroma_opt = 0;
1953                                    config->zones[sel+1].bvop_threshold = 0;
1954    
1955                                    ListView_SetItemState(GetDlgItem(hDlg, IDC_ZONES), sel, 0x00000000, LVIS_SELECTED);
1956                                    zones_update(hDlg, config);
1957                                    ListView_SetItemState(GetDlgItem(hDlg, IDC_ZONES), sel+1, 0xffffffff, LVIS_SELECTED);
1958                                    break;
1959                            }
1960    
1961                            case IDC_REMOVE :
1962                            {
1963                                    int i, sel;
1964                                    sel = ListView_GetNextItem(GetDlgItem(hDlg, IDC_ZONES), -1, LVNI_SELECTED);
1965    
1966                                    if (sel == -1 || config->num_zones < 1) {
1967                                            /*MessageBox(hDlg, "Nothing selected", "Warning", 0);*/
1968                                            break;
1969                                    }
1970    
1971                                    for (i=sel; i<config->num_zones-1; i++)
1972                                            config->zones[i] = config->zones[i+1];
1973    
1974                                    config->num_zones--;
1975    
1976                                    zones_update(hDlg, config);
1977                                    break;
1978                            }
1979    
1980                            case IDC_EDIT :
1981                                    main_download(hDlg, config);
1982                                    config->cur_zone = ListView_GetNextItem(GetDlgItem(hDlg, IDC_ZONES), -1, LVNI_SELECTED);
1983                                    if (config->cur_zone != -1 && adv_dialog(hDlg, config, zone_dlgs, sizeof(zone_dlgs)/sizeof(int))) {
1984                                            zones_update(hDlg, config);
1985                                    }
1986                                    break;
1987    
1988                            case IDC_ADVANCED :
1989                                    main_download(hDlg, config);
1990                                    adv_dialog(hDlg, config, adv_dlgs, sizeof(adv_dlgs)/sizeof(int));
1991                                    break;
1992    
1993                            case IDC_DEFAULTS :
1994                                    config_reg_default(config);
1995                                    main_mode(hDlg, config);
1996                                    main_upload(hDlg, config);
1997                                    break;
1998    
1999                            case IDOK :
2000                                    main_download(hDlg, config);
2001                                    config->save = TRUE;
2002                                    EndDialog(hDlg, IDOK);
2003                                    break;
2004    
2005                            case IDCANCEL :
2006                                    config->save = FALSE;
2007                                    EndDialog(hDlg, IDCANCEL);
2008                                    break;
2009                            }
2010                    } else if (HIWORD(wParam) == LBN_SELCHANGE &&
2011                            (LOWORD(wParam)==IDC_PROFILE || LOWORD(wParam)==IDC_MODE)) {
2012    
2013                            config->mode = SendDlgItemMessage(hDlg, IDC_MODE, CB_GETCURSEL, 0, 0);
2014                            config->profile = SendDlgItemMessage(hDlg, IDC_PROFILE, CB_GETCURSEL, 0, 0);
2015    
2016                            if (!g_use_bitrate) {
2017                                    if (config->mode == RC_MODE_1PASS)
2018                                            set_dlgitem_float(hDlg, IDC_BITRATE, config->desired_quant);
2019                                    else if (config->mode == RC_MODE_2PASS2)
2020                                            SetDlgItemInt(hDlg, IDC_BITRATE, config->desired_size, FALSE);
2021                            }
2022    
2023                            main_mode(hDlg, config);
2024                            main_upload(hDlg, config);
2025    
2026                    }else if (HIWORD(wParam)==EN_UPDATE && LOWORD(wParam)==IDC_BITRATE) {
2027    
2028                            if (g_use_bitrate) {
2029                                    SendDlgItemMessage(hDlg, IDC_SLIDER, TBM_SETPOS, TRUE,
2030                                                    config_get_uint(hDlg, IDC_BITRATE, DEFAULT_MIN_KBPS) );
2031                            } else if (config->mode == RC_MODE_1PASS) {
2032                                    SendDlgItemMessage(hDlg, IDC_SLIDER, TBM_SETPOS, TRUE,
2033                                                    get_dlgitem_float(hDlg, IDC_BITRATE, DEFAULT_QUANT) );
2034                            }
2035                            main_download(hDlg, config);
2036    
2037                    }else {
2038                            return 0;
2039                    }
2040                    break;
2041    
2042            case WM_HSCROLL :
2043                    if((HWND)lParam == GetDlgItem(hDlg, IDC_SLIDER)) {
2044                            if (g_use_bitrate)
2045                                    SetDlgItemInt(hDlg, IDC_BITRATE, SendMessage((HWND)lParam, TBM_GETPOS, 0, 0), FALSE);
2046                            else
2047                                    set_dlgitem_float(hDlg, IDC_BITRATE, SendMessage((HWND)lParam, TBM_GETPOS, 0, 0));
2048    
2049                            main_download(hDlg, config);
2050                            break;
2051                    }
2052                    return 0;
2053    
2054            default :
2055                    return 0;
2056            }
2057    
2058            return 1;
2059    }
2060    
2061    
2062    
2063    /* ===================================================================================== */
2064    /* ABOUT DIALOG ======================================================================== */
2065    /* ===================================================================================== */
2066    
2067    BOOL CALLBACK about_proc(HWND hDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
2068    {
2069            switch (uMsg)
2070            {
2071            case WM_INITDIALOG :
2072                    {
2073                            xvid_gbl_info_t info;
2074                            char core[100];
2075                            HFONT hFont;
2076                            LOGFONT lfData;
2077                            HINSTANCE m_hdll;
2078    
2079                            SetDlgItemText(hDlg, IDC_BUILD, XVID_BUILD);
2080                            SetDlgItemText(hDlg, IDC_SPECIAL_BUILD, XVID_SPECIAL_BUILD);
2081    
2082                            memset(&info, 0, sizeof(info));
2083                            info.version = XVID_VERSION;
2084    
2085                            m_hdll = LoadLibrary(XVID_DLL_NAME);
2086                            if (m_hdll != NULL) {
2087    
2088                                    ((int (__cdecl *)(void *, int, void *, void *))GetProcAddress(m_hdll, "xvid_global"))
2089                                            (0, XVID_GBL_INFO, &info, NULL);
2090    
2091                                    wsprintf(core, "xvidcore.dll version %d.%d.%d (\"%s\")",
2092                                            XVID_VERSION_MAJOR(info.actual_version),
2093                                            XVID_VERSION_MINOR(info.actual_version),
2094                                            XVID_VERSION_PATCH(info.actual_version),
2095                                            info.build);
2096    
2097                                    FreeLibrary(m_hdll);
2098                            } else {
2099                                    wsprintf(core, "xvidcore.dll not found!");
2100                            }
2101    
2102                            SetDlgItemText(hDlg, IDC_CORE, core);
2103    
2104                            hFont = (HFONT)SendDlgItemMessage(hDlg, IDC_WEBSITE, WM_GETFONT, 0, 0L);
2105    
2106                            if (GetObject(hFont, sizeof(LOGFONT), &lfData)) {
2107                                    lfData.lfUnderline = 1;
2108    
2109                                    hFont = CreateFontIndirect(&lfData);
2110                                    if (hFont) {
2111                                            SendDlgItemMessage(hDlg, IDC_WEBSITE, WM_SETFONT, (WPARAM)hFont, 1L);
2112                                    }
2113                            }
2114    
2115                            SetClassLong(GetDlgItem(hDlg, IDC_WEBSITE), GCL_HCURSOR, (LONG)LoadCursor(NULL, IDC_HAND));
2116                            SetDlgItemText(hDlg, IDC_WEBSITE, XVID_WEBSITE);
2117                    }
2118                    break;
2119            case WM_CTLCOLORSTATIC :
2120                    if ((HWND)lParam == GetDlgItem(hDlg, IDC_WEBSITE))
2121                    {
2122                            SetBkMode((HDC)wParam, TRANSPARENT) ;
2123                            SetTextColor((HDC)wParam, RGB(0x00,0x00,0xc0));
2124                            return (BOOL)GetStockObject(NULL_BRUSH);
2125                    }
2126                    return 0;
2127    
2128            case WM_COMMAND :
2129                    if (LOWORD(wParam) == IDC_WEBSITE && HIWORD(wParam) == STN_CLICKED)
2130                    {
2131                            ShellExecute(hDlg, "open", XVID_WEBSITE, NULL, NULL, SW_SHOWNORMAL);
2132                    }
2133                    else if (LOWORD(wParam) == IDOK || LOWORD(wParam) == IDCANCEL)
2134                    {
2135                            EndDialog(hDlg, LOWORD(wParam));
2136                    }
2137                    break;
2138    
2139            default :
2140                    return 0;
2141            }
2142    
2143            return 1;
2144    }
2145    
2146    
2147    void
2148    sort_zones(zone_t * zones, int zone_num, int * sel)
2149    {
2150            int i, j;
2151            zone_t tmp;
2152            for (i = 0; i < zone_num; i++) {
2153                    int cur = i;
2154                    int min_f = zones[i].frame;
2155                    for (j = i + 1; j < zone_num; j++) {
2156                            if (zones[j].frame < min_f) {
2157                                    min_f = zones[j].frame;
2158                                    cur = j;
2159                            }
2160                    }
2161                    if (cur != i) {
2162                            tmp = zones[i];
2163                            zones[i] = zones[cur];
2164                            zones[cur] = tmp;
2165                            if (i == *sel) *sel = cur;
2166                            else if (cur == *sel) *sel = i;
2167                    }
2168            }
2169    }
2170    
2171    
2172    static void
2173    zones_update(HWND hDlg, CONFIG * config)
2174    {
2175            int i, sel;
2176    
2177            sel = ListView_GetNextItem(GetDlgItem(hDlg, IDC_ZONES), -1, LVNI_SELECTED);
2178    
2179            sort_zones(config->zones, config->num_zones, &sel);
2180    
2181            ListView_DeleteAllItems(GetDlgItem(hDlg,IDC_ZONES));
2182    
2183            for (i = 0; i < config->num_zones; i++)
2184                    main_insert_zone(GetDlgItem(hDlg,IDC_ZONES), &config->zones[i], i, TRUE);
2185    
2186            if (sel == -1 && config->num_zones > 0) sel = 0;
2187            if (sel >= config->num_zones) sel = config->num_zones-1;
2188    
2189            config->cur_zone = sel;
2190            ListView_SetItemState(GetDlgItem(hDlg, IDC_ZONES), sel, 0xffffffff, LVIS_SELECTED);
2191    }

Legend:
Removed from v.1.1  
changed lines
  Added in v.1.1.2.40

No admin address has been configured
ViewVC Help
Powered by ViewVC 1.0.4