OSDN Git Service

MacGui: Remove Sparkle's "Send anonymouns system information" preference as the decis...
[handbrake-jp/handbrake-jp-git.git] / macosx / HBAdvancedController.m
1 /* HBAdvancedController
2
3     This file is part of the HandBrake source code.
4     Homepage: <http://handbrake.fr/>.
5     It may be used under the terms of the GNU General Public License. */
6     
7 #import "HBAdvancedController.h"
8
9 @implementation HBAdvancedController
10
11 - (id)init
12 {
13     [super init];
14     [self loadMyNibFile];
15     
16     return self;
17 }
18
19 - (void) setView: (NSBox *) box
20 {
21     fOptionsBox = box;
22     [fOptionsBox setContentView:fX264optView];
23 }
24
25 - (BOOL) loadMyNibFile
26 {
27     if(![NSBundle loadNibNamed:@"AdvancedView" owner:self])
28     {
29         NSLog(@"Warning! Could not load myNib file.\n");
30         return NO;
31     }
32     
33     return YES;
34 }
35
36 - (NSString *) optionsString
37 {
38     return [fDisplayX264Options stringValue];
39 }
40
41 - (void) setOptions: (NSString *)string
42 {
43     [fDisplayX264Options setStringValue:string];
44     [self X264AdvancedOptionsSet:nil];
45 }
46
47 - (void) setHidden: (BOOL) hide
48 {
49     if(hide)
50     {
51         [fOptionsBox setContentView:fEmptyView];
52         [fX264optViewTitleLabel setStringValue: @"Only Used With The x264 (H.264) Codec"];
53     }
54     else
55     {
56         [fOptionsBox setContentView:fX264optView];
57         [fX264optViewTitleLabel setStringValue: @""];
58     }
59     return;
60 }
61
62  - (void) enableUI: (bool) b
63 {
64     unsigned i;
65     NSControl * controls[] =
66       { fX264optViewTitleLabel,fDisplayX264Options,fDisplayX264OptionsLabel,fX264optBframesLabel,
67         fX264optBframesPopUp,fX264optRefLabel,fX264optRefPopUp,fX264optNfpskipLabel,fX264optNfpskipSwitch,
68         fX264optNodctdcmtLabel,fX264optNodctdcmtSwitch,fX264optSubmeLabel,fX264optSubmePopUp,
69         fX264optTrellisLabel,fX264optTrellisPopUp,fX264optMixedRefsLabel,fX264optMixedRefsSwitch,
70         fX264optMotionEstLabel,fX264optMotionEstPopUp,fX264optMERangeLabel,fX264optMERangePopUp,
71         fX264optWeightBLabel,fX264optWeightBSwitch, fX264optBPyramidLabel,fX264optBPyramidSwitch,
72         fX264optDirectPredLabel,fX264optDirectPredPopUp,fX264optDeblockLabel,fX264optAnalyseLabel,
73         fX264optAnalysePopUp,fX264opt8x8dctLabel,fX264opt8x8dctSwitch,fX264optCabacLabel,fX264optCabacSwitch,
74         fX264optAlphaDeblockPopUp,fX264optBetaDeblockPopUp, fX264optPsyRDSlider, fX264optPsyRDLabel, fX264optPsyTrellisSlider, fX264optPsyTrellisLabel, fX264optBAdaptPopUp, fX264optBAdaptLabel };
75
76     for( i = 0; i < sizeof( controls ) / sizeof( NSControl * ); i++ )
77     {
78         if( [[controls[i] className] isEqualToString: @"NSTextField"] )
79         {
80             NSTextField * tf = (NSTextField *) controls[i];
81             if( ![tf isBezeled] )
82             {
83                 [tf setTextColor: b ? [NSColor controlTextColor] :
84                     [NSColor disabledControlTextColor]];
85                 continue;
86             }
87         }
88         [controls[i] setEnabled: b];
89
90     }
91     
92     [fX264optView setWantsLayer:YES];
93 }
94
95 - (void)dealloc
96 {
97     [super dealloc];
98 }
99
100 /**
101  * Populates the option widgets
102  */
103 - (IBAction) X264AdvancedOptionsSet: (id) sender
104 {
105     /*Set opt widget values here*/
106     
107     NSString * toolTip = @"";
108     
109     /*B-Frames fX264optBframesPopUp*/
110     int i;
111     [fX264optBframesPopUp removeAllItems];
112     [fX264optBframesPopUp addItemWithTitle:@"Default (0)"];
113     for (i=0; i<17;i++)
114     {
115         [fX264optBframesPopUp addItemWithTitle:[NSString stringWithFormat:@"%d",i]];
116     }
117     toolTip =
118         @"Sane values are 1-6. B-Frames are smaller than other frames, so they let you pack in more quality at the same bitrate. Use more of them with animated material: 9-16.";
119     [fX264optBframesPopUp setToolTip: toolTip];
120     [fX264optBframesLabel setToolTip: toolTip];
121     
122     /*Reference Frames fX264optRefPopUp*/
123     [fX264optRefPopUp removeAllItems];
124     [fX264optRefPopUp addItemWithTitle:@"Default (1)"];
125     for (i=0; i<17;i++)
126     {
127         [fX264optRefPopUp addItemWithTitle:[NSString stringWithFormat:@"%d",i]];
128     }
129     toolTip =
130         @"Sane values are 1-6. The more you add, the higher the quality — but the slower the encode. Be careful...too many and QuickTime struggle to play the video back.";
131     [fX264optRefPopUp setToolTip: toolTip];
132     [fX264optRefLabel setToolTip: toolTip];
133     
134     /*No Fast P-Skip fX264optNfpskipSwitch BOOLEAN*/
135     [fX264optNfpskipSwitch setState:0];
136     toolTip =
137         @"This can help with blocking on solid colors like blue skies, but it also slows down the encode.";
138     [fX264optNfpskipSwitch setToolTip: toolTip];
139     [fX264optNfpskipLabel setToolTip: toolTip];
140     
141     /*No Dict Decimate fX264optNodctdcmtSwitch BOOLEAN*/
142     [fX264optNodctdcmtSwitch setState:0];    
143     toolTip =
144         @"To save space, x264 will \"zero out\" blocks when it thinks they won't be perceptible by the viewer. This negligibly reduces quality, but in rare cases it can mess up and produce visible artifacts. This situation can be alleviated by telling x264 not to decimate DCT blocks.\n\nIt increases quality but also bitrate/file size, so if you use it when you've specified a target bitrate you will end up with a worse picture than without it. However, when used with constant quality encoding, or if you boost the average bitrate to compensate, you might get a better result.";
145     [fX264optNodctdcmtSwitch setToolTip: toolTip];
146     [fX264optNodctdcmtLabel setToolTip: toolTip];
147     
148     /*Sub Me fX264optSubmePopUp*/
149     [fX264optSubmePopUp removeAllItems];
150     [fX264optSubmePopUp addItemWithTitle:@"Default (6)"];
151     for (i=0; i<10;i++)
152     {
153         [fX264optSubmePopUp addItemWithTitle:[NSString stringWithFormat:@"%d",i]];
154     }
155     toolTip =
156         @"This setting is finer-grained than the motion estimation settings above. Instead of dealing with whole pixels, it deals with 4 fractional pixels, or quarter pixels (qpel). Higher levels increase quality by further refining the motion prediction for these quarter pixels, but take longer to encode.\n\nThe default, 6, turns on a feature called rate distortion optimization, including psychovisual enhancements. 7 enables that rate distortion for B-frames. 8 refines those decisions for I and P frames, and 9 adds on refinement for B-frames as well.";
157     [fX264optSubmePopUp setToolTip: toolTip];
158     [fX264optSubmeLabel setToolTip: toolTip];
159     
160     /*Trellis fX264optTrellisPopUp*/
161     [fX264optTrellisPopUp removeAllItems];
162     [fX264optTrellisPopUp addItemWithTitle:@"Default (0)"];
163     for (i=0; i<3;i++)
164     {
165         [fX264optTrellisPopUp addItemWithTitle:[NSString stringWithFormat:@"%d",i]];
166     }
167     [fX264optTrellisPopUp setWantsLayer:YES];
168     toolTip =
169         @"Trellis fine-tunes how bitrate is doled out, so it can reduce file size/bitrate or increase quality. A value of 1 means it only fine-tunes the final encode of a block of pixels, while 2 means it is considered during earlier phases of the decision-making process as well.";
170     [fX264optTrellisPopUp setToolTip: toolTip];
171     [fX264optTrellisLabel setToolTip: toolTip];
172     
173     /*Mixed-references fX264optMixedRefsSwitch BOOLEAN*/
174     [fX264optMixedRefsSwitch setState:0];
175     [fX264optMixedRefsSwitch setWantsLayer:YES];
176     toolTip =
177         @"With this on, different references can be used for different parts of each 16x16 pixel macroblock, increasing quality.";
178     [fX264optMixedRefsSwitch setToolTip: toolTip];
179     [fX264optMixedRefsLabel setToolTip: toolTip];
180     
181     /*Motion Estimation fX264optMotionEstPopUp*/
182     [fX264optMotionEstPopUp removeAllItems];
183     [fX264optMotionEstPopUp addItemWithTitle:@"Default (Hexagon)"];
184     [fX264optMotionEstPopUp addItemWithTitle:@"Diamond"];
185     [fX264optMotionEstPopUp addItemWithTitle:@"Hexagon"];
186     [fX264optMotionEstPopUp addItemWithTitle:@"Uneven Multi-Hexagon"];
187     [fX264optMotionEstPopUp addItemWithTitle:@"Exhaustive"];
188     [fX264optMotionEstPopUp addItemWithTitle:@"Transformed Exhaustive"];
189     toolTip =
190         @"Controls the motion estimation method. Motion estimation is how the encoder decides how each block of pixels in a frame has moved, compared to most similar blocks in the other frames it references. There are many ways of finding the most similar blocks, with varying speeds and accuracy.\n\nAt the most basic setting, dia, x264 will only consider a diamond-shaped region around each pixel.\n\nThe default setting, hex, is similar to dia but uses a hexagon shape.\n\nUneven multi-hexagon, umh, searches a number of different patterns across a wider area and thus is slower than hex and dia but further increases compression efficiency and quality.\n\nesa, an exhaustive search of a square around each pixel (whose size is controlled by the me-range parameter), is much slower and offers only minimal quality gains.\n\ntesa, transformed exhaustive search, which performs just as thorough a search, is slower still but offers further slight improvements to quality.";
191     [fX264optMotionEstPopUp setToolTip: toolTip];
192     [fX264optMotionEstLabel setToolTip: toolTip];
193     
194     /*Motion Estimation range fX264optMERangePopUp*/
195     [fX264optMERangePopUp removeAllItems];
196     [fX264optMERangePopUp addItemWithTitle:@"Default (16)"];
197     for (i=4; i<65;i++)
198     {
199         [fX264optMERangePopUp addItemWithTitle:[NSString stringWithFormat:@"%d",i]];
200     }
201     toolTip =
202         @"This range is the radius, in pixels, x264 should use for motion estimation searches. It only has an effect when you use Uneven Multi-Hexagonal, Exhaustive, or Transformed Exhaustive searching. 24, 32, and 64 are good values, with each being progressively smaller for progressively less improvement to picture quality.";
203     [fX264optMERangePopUp setToolTip: toolTip];
204     [fX264optMERangeLabel setToolTip: toolTip];
205     
206     /*Weighted B-Frame Prediction fX264optWeightBSwitch BOOLEAN*/
207     [fX264optWeightBSwitch setState:0];
208     [fX264optWeightBSwitch setWantsLayer:YES];
209     toolTip =
210         @"Sometimes x264 will base a B-frame's motion compensation on frames both before and after. With weighted B-frames, the amount of influence each frame has is related to its distance from the frame being encoded, instead of both having equal influence.";
211     [fX264optWeightBSwitch setToolTip: toolTip];
212     [fX264optWeightBLabel setToolTip: toolTip];
213     
214     /*B-frame Pyramids fX264optBPyramidSwitch BOOLEAN*/
215     [fX264optBPyramidSwitch setState:0];
216     [fX264optBPyramidSwitch setWantsLayer:YES];
217     toolTip =
218         @"B-frame pyramids are a High Profile feature. Pyramidal B-frames mean that B-frames don't just reference surrounding reference frames — instead, it also treats a previous B-frame as a reference, improving quality/lowering bitrate at the expense of complexity. Logically, to reference an earlier B-frame, you must tell x264 to use at least 2 B-frames.";
219     [fX264optBPyramidSwitch setToolTip: toolTip];
220     [fX264optBPyramidLabel setToolTip: toolTip];
221     
222     /*Direct B-Frame Prediction Mode fX264optDirectPredPopUp*/
223     [fX264optDirectPredPopUp removeAllItems];
224     [fX264optDirectPredPopUp addItemWithTitle:@"Default (Spatial)"];
225     [fX264optDirectPredPopUp addItemWithTitle:@"None"];
226     [fX264optDirectPredPopUp addItemWithTitle:@"Spatial"];
227     [fX264optDirectPredPopUp addItemWithTitle:@"Temporal"];
228     [fX264optDirectPredPopUp addItemWithTitle:@"Automatic"];
229     [fX264optDirectPredPopUp setWantsLayer:YES];
230     toolTip =
231         @"Direct prediction tells x264 what method to use when guessing motion for certain parts of a B-frame. It can either look at other parts of the current frame (spatial) or compare against the following P-frameframe (temporal). You're best off setting this to automatic, so x264 decides which method is best on its own. Don't select none assuming it will be faster; instead it will take longer and look worse. If you're going to choose between spatial and temporal, spatial is usually better.";
232     [fX264optDirectPredPopUp setToolTip: toolTip];
233     [fX264optDirectPredLabel setToolTip: toolTip];
234     
235     /* Adaptive B-Frames Mode fX264optBAdaptPopUp */
236     [fX264optBAdaptPopUp removeAllItems];
237     [fX264optBAdaptPopUp addItemWithTitle:@"Default (Fast)"];
238     [fX264optBAdaptPopUp addItemWithTitle:@"Off"];
239     [fX264optBAdaptPopUp addItemWithTitle:@"Fast"];
240     [fX264optBAdaptPopUp addItemWithTitle:@"Optimal"];
241     [fX264optBAdaptPopUp setWantsLayer:YES];
242     toolTip =
243         @"When adaptive B-Frames are disabled, the number of B-Frames you specify is the constant length of every B-Frame sequence. When one of the adaptive modes is enabled, the number of B-Frames is treated as a maximum, with the length of each sequence varying, but never exceeding the max.\n\nFast mode takes the same amount of time no matter how many B-frames you specify. However, it doesn't always make the best decisions on how many B-Frames to use in a sequence.\n\nOptimal mode gets slower as the maximum number of B-Frames increases, but does a much better job at deciding sequence length, which can mean smaller file sizes and better quality.";
244     [fX264optBAdaptPopUp setToolTip: toolTip];
245     [fX264optBAdaptLabel setToolTip: toolTip];
246     
247     /*Alpha Deblock*/
248     [fX264optAlphaDeblockPopUp removeAllItems];
249     [fX264optAlphaDeblockPopUp addItemWithTitle:@"Default (0)"];
250     for (i=-6; i<7;i++)
251     {
252         [fX264optAlphaDeblockPopUp addItemWithTitle:[NSString stringWithFormat:@"%d",i]];
253     }
254     toolTip =
255         @"x264 includes an in-loop deblocking filter. What this means is that blocky compression artifacts are smoothed away when you play back the video. It has two settings: strength and threshold, just like a simple filter in Photoshop.\n\nStrength controls the amount of deblocking applied to the whole frame. If you drop down below 0, you reduce the amount of blurring. Go too negative, and you'll get an effect somewhat like oversharpening an image. Go into positive values, and the image may become too soft.\n\nThreshold controls how sensitive the filter is to whether something in a block is detail that needs to be preserved: lower numbers blur details less.\n\nThe default deblocking values are 0 and 0. This does not mean zero deblocking. It means x264 will apply the regular deblocking strength and thresholds the codec authors have selected as working the best in most cases.\n\nWhile many, many people stick with the default deblocking values of 0,0, and you should never change the deblocking without disabling adaptive quantization, other people disagree. Some prefer a slightly less blurred image for live action material, and use values like -2,-1 or -2,-2. Others will raise it to 1,1 or even 3,3 for animation. While the values for each setting extend from -6 to 6, the consensus is that going below -3 or above 3 is worthless.";
256     [fX264optAlphaDeblockPopUp setToolTip: toolTip];
257     [fX264optDeblockLabel setToolTip: toolTip];
258
259     /*Beta Deblock*/
260     [fX264optBetaDeblockPopUp removeAllItems];
261     [fX264optBetaDeblockPopUp addItemWithTitle:@"Default (0)"];
262     for (i=-6; i<7;i++)
263     {
264         [fX264optBetaDeblockPopUp addItemWithTitle:[NSString stringWithFormat:@"%d",i]];
265     }
266     [fX264optBetaDeblockPopUp setToolTip: toolTip];
267     [fX264optDeblockLabel setToolTip: toolTip];
268
269     /* Analysis fX264optAnalysePopUp */
270     [fX264optAnalysePopUp removeAllItems];
271     [fX264optAnalysePopUp addItemWithTitle:@"Default (some)"]; /* 0=default */
272     [fX264optAnalysePopUp addItemWithTitle:[NSString stringWithFormat:@"None"]]; /* 1=none */
273     [fX264optAnalysePopUp addItemWithTitle:[NSString stringWithFormat:@"All"]]; /* 2=all */
274     toolTip =
275         @"Analysis controls how finely x264 divides up a frame to capture detail. Full macroblocks are 16x16 pixels, but x264 can go down all the way to 4x4 blocks if it judges it necessary. By default it only breaks up key frames that much. To give x264 the freedom to make the best decisions for all frames, use \"all\" analysis. If you want to create a high profile H.264 video (which is less compatible with the world at large than main profile), also check the \"8x8 DCT blocks\" box to add yet another block size for analysis.";
276     [fX264optAnalysePopUp setToolTip: toolTip];
277     [fX264optAnalyseLabel setToolTip: toolTip];
278
279     /* 8x8 DCT fX264op8x8dctSwitch */
280     [fX264opt8x8dctSwitch setState:0];
281     [fX264opt8x8dctSwitch setWantsLayer:YES];
282     toolTip =
283         @"Checking this box lets x264 break key frames down into 8x8 blocks of pixels for analysis. This is a high profile feature of H.264, which makes it less compatible. It should slightly decrease bitrate or improve quality. Turn it on whenever possible.";
284     [fX264opt8x8dctSwitch setToolTip: toolTip];
285     [fX264opt8x8dctLabel setToolTip: toolTip];
286
287     /* CABAC fX264opCabacSwitch */
288     [fX264optCabacSwitch setState:1];
289     toolTip =
290         @"CABAC, or context adaptive binary arithmetic coding, is used by x264 to reduce the bitrate needed for a given quality by 15\%. This makes it very cool and very useful, and it should be left on whenever possible. However, it is incompatible with the iPod, and makes the AppleTV struggle. So turn it off for those.\n\nCABAC is a kind of entropy coding, which means that it compresses data by making shorthand symbols to represent long streams of data. The \"entropy\" part means that the symbols it uses the most often are the smallest. When you disable CABAC, another entropy coding scheme gets enabled, called CAVLC (context adaptive variable-length coding). CAVLC is a lot less efficient, which is why it needs 15\% more bitrate to achieve the same quality as CABAC.";
291     [fX264optCabacSwitch setToolTip: toolTip];
292     [fX264optCabacLabel setToolTip: toolTip];
293     
294     /* PsyRDO fX264optPsyRDSlider */
295     [fX264optPsyRDSlider setMinValue:0.0];
296     [fX264optPsyRDSlider setMaxValue:1.0];
297     [fX264optPsyRDSlider setTickMarkPosition:NSTickMarkBelow];
298     [fX264optPsyRDSlider setNumberOfTickMarks:10];
299     [fX264optPsyRDSlider setAllowsTickMarkValuesOnly:YES];
300     [fX264optPsyRDSlider setFloatValue:1.0];
301     toolTip =
302         @"Psychovisual Rate Distortion Optimization sure is a mouthful, isn't it? Basically, it means x264 tries to retain detail, for better quality to the human eye, as opposed to trying to maximize quality the way a computer understands it, through signal-to-noise ratios that have trouble telling apart fine detail and noise.";
303     [fX264optPsyRDSlider setToolTip: toolTip];
304     [fX264optPsyRDLabel setToolTip: toolTip];
305
306     /* PsyTrellis fX264optPsyRDSlider */
307     [fX264optPsyTrellisSlider setMinValue:0.0];
308     [fX264optPsyTrellisSlider setMaxValue:1.0];
309     [fX264optPsyTrellisSlider setTickMarkPosition:NSTickMarkBelow];
310     [fX264optPsyTrellisSlider setNumberOfTickMarks:10];
311     [fX264optPsyTrellisSlider setAllowsTickMarkValuesOnly:YES];
312     [fX264optPsyTrellisSlider setFloatValue:0.0];
313     toolTip =
314         @"Psychovisual Trellis tries to retain more sharpness and detail, but can cause artifacting. It is considered experimental, which is why it's off by default. Good values are 0.1 to 0.2.";
315     [fX264optPsyTrellisSlider setToolTip: toolTip];
316     [fX264optPsyTrellisLabel setToolTip: toolTip];
317
318     /* Standardize the option string */
319     [self X264AdvancedOptionsStandardizeOptString:nil];
320
321     /* Set Current GUI Settings based on newly standardized string */
322     [self X264AdvancedOptionsSetCurrentSettings:sender];
323
324     /* Fade out options that don't apply */
325     [self X264AdvancedOptionsAnimate: sender];
326 }
327
328 /**
329  * Cleans the option string to use a standard format of option=value
330  */
331 - (IBAction) X264AdvancedOptionsStandardizeOptString: (id) sender
332 {
333     /* Set widgets depending on the opt string in field */
334     NSString * thisOpt; // The separated option such as "bframes=3"
335     NSString * optName = @""; // The option name such as "bframes"
336     NSString * optValue = @"";// The option value such as "3"
337     NSString * changedOptString = @"";
338     NSArray *currentOptsArray;
339     
340     /*First, we get an opt string to process */
341     NSString *currentOptString = [fDisplayX264Options stringValue];
342     
343     /* Verify there is an opt string to process by making sure an
344        option is getting its value set. If so, start to process it. */
345     NSRange currentOptRange = [currentOptString rangeOfString:@"="];
346     if (currentOptRange.location != NSNotFound)
347     {
348         /*Put individual options into an array based on the ":" separator for processing, result is "<opt>=<value>"*/
349         currentOptsArray = [currentOptString componentsSeparatedByString:@":"];
350         
351         /*iterate through the array and get <opts> and <values*/
352         int loopcounter;
353         int currentOptsArrayCount = [currentOptsArray count];
354         for (loopcounter = 0; loopcounter < currentOptsArrayCount; loopcounter++)
355         {
356             thisOpt = [currentOptsArray objectAtIndex:loopcounter];
357             
358             NSRange splitOptRange = [thisOpt rangeOfString:@"="];
359             if (splitOptRange.location != NSNotFound)
360             {
361                 optName = [thisOpt substringToIndex:splitOptRange.location];
362                 optValue = [thisOpt substringFromIndex:splitOptRange.location + 1];
363                 
364                 /* Standardize the names here depending on whats in the string */
365                 optName = [self X264AdvancedOptionsStandardizeOptNames:optName];
366                 thisOpt = [NSString stringWithFormat:@"%@=%@",optName,optValue];    
367             }
368             else // No value given so we use a default of "1"
369             {
370                 optName = thisOpt;
371
372                 /* Standardize the names here depending on whats in the string */
373                 optName = [self X264AdvancedOptionsStandardizeOptNames:optName];
374                 thisOpt = [NSString stringWithFormat:@"%@=%d",optName,1];
375             }
376             
377             /* Construct New String for opts here.*/
378             if ([thisOpt isEqualToString:@""])
379             {
380                 /* Blank option, just add it to the string. (Why?) */
381                 changedOptString = [NSString stringWithFormat:@"%@%@",changedOptString,thisOpt];
382             }
383             else
384             {
385                 if ([changedOptString isEqualToString:@""])
386                 {
387                     /* Blank string, output the current option. */
388                     changedOptString = [NSString stringWithFormat:@"%@",thisOpt];
389                 }
390                 else
391                 {
392                     /* Option exists and string exists, so append the option
393                        to the string with a semi-colon inbetween them.       */
394                     changedOptString = [NSString stringWithFormat:@"%@:%@",changedOptString,thisOpt];
395                 }
396             }
397         }
398     }
399     
400     /* Change the option string to reflect the new standardized option string */
401     [fDisplayX264Options setStringValue:[NSString stringWithFormat:changedOptString]];
402 }
403
404 /**
405  * Cleans the option string to use a standard set of option names, by conflating synonyms.
406  */
407 - (NSString *) X264AdvancedOptionsStandardizeOptNames:(NSString *) cleanOptNameString
408 {
409     /* Reference Frames */
410     if ([cleanOptNameString isEqualToString:@"ref"] || [cleanOptNameString isEqualToString:@"frameref"])
411     {
412         cleanOptNameString = @"ref";
413     }
414     
415     /*No Fast PSkip nofast_pskip*/
416     if ([cleanOptNameString isEqualToString:@"no-fast-pskip"] || [cleanOptNameString isEqualToString:@"no_fast_pskip"] || [cleanOptNameString isEqualToString:@"nofast_pskip"])
417     {
418         cleanOptNameString = @"no-fast-pskip";
419     }
420     
421     /*No Dict Decimate*/
422     if ([cleanOptNameString isEqualToString:@"no-dct-decimate"] || [cleanOptNameString isEqualToString:@"no_dct_decimate"] || [cleanOptNameString isEqualToString:@"nodct_decimate"])
423     {
424         cleanOptNameString = @"no-dct-decimate";
425     }
426     
427     /*Subme*/
428     if ([cleanOptNameString isEqualToString:@"subme"])
429     {
430         cleanOptNameString = @"subq";
431     }
432     
433     /*ME Range*/
434     if ([cleanOptNameString isEqualToString:@"me-range"] || [cleanOptNameString isEqualToString:@"me_range"])
435         cleanOptNameString = @"merange";
436     
437     /*WeightB*/
438     if ([cleanOptNameString isEqualToString:@"weight-b"] || [cleanOptNameString isEqualToString:@"weight_b"])
439     {
440         cleanOptNameString = @"weightb";
441     }
442     
443     /*B Pyramid*/
444     if ([cleanOptNameString isEqualToString:@"b_pyramid"])
445     {
446         cleanOptNameString = @"b-pyramid";
447     }
448     
449     /*Direct Prediction*/
450     if ([cleanOptNameString isEqualToString:@"direct-pred"] || [cleanOptNameString isEqualToString:@"direct_pred"])
451     {
452         cleanOptNameString = @"direct";
453     }
454     
455     /*Deblocking*/
456     if ([cleanOptNameString isEqualToString:@"filter"])
457     {
458         cleanOptNameString = @"deblock";
459     }
460     
461     /*Analysis*/
462     if ([cleanOptNameString isEqualToString:@"partitions"])
463     {
464         cleanOptNameString = @"analyse";
465     }
466     
467     return cleanOptNameString;    
468 }
469
470 /**
471  * Fades options in and out depending on whether they're available..
472  */
473 - (IBAction) X264AdvancedOptionsAnimate: (id) sender
474 {
475     /* Lots of situations to cover.
476        - B-frames (when 0 turn of b-frame specific stuff, when < 2 disable b-pyramid)
477        - CABAC (when 0 turn off trellis)
478        - analysis (if none, turn off 8x8dct)
479        - refs (under 2, disable mixed-refs)
480        - subme (if under 6, turn off psy-rd and psy-trel)
481        - trellis (if 0, turn off psy-trel)
482     */
483     
484     if( sender == fX264optBframesPopUp || sender == nil || sender == fDisplayX264Options )
485     {
486         if ( [fX264optBframesPopUp indexOfSelectedItem ] < 2)
487         {
488             /* If the b-frame widget is at 0 or 1, the user has chosen
489                not to use b-frames at all. So disable the options
490                that can only be used when b-frames are enabled.        */
491             
492             if( [fX264optWeightBSwitch isHidden] == false)
493             {
494                 [[fX264optWeightBSwitch animator] setHidden:YES];
495                 [[fX264optWeightBLabel animator] setHidden:YES];
496                 if ( [fX264optWeightBSwitch state] == 1 )
497                     [fX264optWeightBSwitch performClick:self];
498             }
499
500             if( [fX264optBPyramidSwitch isHidden] == false )
501             {
502                 [[fX264optBPyramidSwitch animator] setHidden:YES];
503                 [[fX264optBPyramidLabel animator] setHidden:YES];
504                 if ( [fX264optBPyramidSwitch state] == 1 )
505                     [fX264optBPyramidSwitch performClick:self];
506             }
507
508             if( [fX264optDirectPredPopUp isHidden] == false )
509             {
510                 [[fX264optDirectPredPopUp animator] setHidden:YES];
511                 [[fX264optDirectPredLabel animator] setHidden:YES];
512                 if ( [fX264optDirectPredPopUp indexOfSelectedItem] > 0 )
513                 {
514                     [fX264optDirectPredPopUp selectItemAtIndex: 0];
515                     [[fX264optDirectPredPopUp cell] performClick:self];
516                 }
517             }
518
519             if( [fX264optBAdaptPopUp isHidden] == false )
520             {
521                 [[fX264optBAdaptPopUp animator] setHidden:YES];
522                 [[fX264optBAdaptLabel animator] setHidden:YES];
523                 if ( [fX264optBAdaptPopUp indexOfSelectedItem] > 0 )
524                 {
525                     [fX264optBAdaptPopUp selectItemAtIndex: 0];
526                     [[fX264optBAdaptPopUp cell] performClick:self];
527                 }
528             }
529         }
530         else if ( [fX264optBframesPopUp indexOfSelectedItem ] == 2)
531         {
532             /* Only 1 b-frame? Disable b-pyramid. */
533             if( [fX264optBPyramidSwitch isHidden] == false )
534             {
535                 [[fX264optBPyramidSwitch animator] setHidden:YES];
536                 [[fX264optBPyramidLabel animator] setHidden:YES];
537                 if ( [fX264optBPyramidSwitch state] == 1 )
538                     [fX264optBPyramidSwitch performClick:self];
539             }
540
541             if( [fX264optWeightBSwitch isHidden] == true )
542             {
543                 [[fX264optWeightBSwitch animator] setHidden:NO];
544                 [[fX264optWeightBLabel animator] setHidden:NO];
545             }
546             
547             if( [fX264optDirectPredPopUp isHidden] == true )
548             {
549                 [[fX264optDirectPredPopUp animator] setHidden:NO];
550                 [[fX264optDirectPredLabel animator] setHidden:NO];
551             }
552             
553             if( [fX264optBAdaptPopUp isHidden] == true )
554             {
555                 [[fX264optBAdaptPopUp animator] setHidden:NO];
556                 [[fX264optBAdaptLabel animator] setHidden:NO];
557             }
558         }
559         else
560         {
561             if( [fX264optBPyramidSwitch isHidden] == true )
562             {
563                 [[fX264optBPyramidSwitch animator] setHidden:NO];
564                 [[fX264optBPyramidLabel animator] setHidden:NO];
565             }
566
567             if( [fX264optWeightBSwitch isHidden] == true )
568             {
569                 [[fX264optWeightBSwitch animator] setHidden:NO];
570                 [[fX264optWeightBLabel animator] setHidden:NO];
571             }
572             
573             if( [fX264optDirectPredPopUp isHidden] == true )
574             {
575                 [[fX264optDirectPredPopUp animator] setHidden:NO];
576                 [[fX264optDirectPredLabel animator] setHidden:NO];
577             }
578             
579             if( [fX264optBAdaptPopUp isHidden] == true )
580             {
581                 [[fX264optBAdaptPopUp animator] setHidden:NO];
582                 [[fX264optBAdaptLabel animator] setHidden:NO];
583             }
584         }
585     }
586     
587     if( sender == fX264optCabacSwitch || sender == nil || sender == fDisplayX264Options )
588     {
589         if ( [fX264optCabacSwitch state] == false)
590         {
591             if( [fX264optTrellisPopUp isHidden] == false )
592             {
593                 /* Without CABAC entropy coding, trellis doesn't run. */
594                 [[fX264optTrellisPopUp animator] setHidden:YES];
595                 [[fX264optTrellisLabel animator] setHidden:YES];
596                 [fX264optTrellisPopUp selectItemAtIndex:0];
597                 [[fX264optTrellisPopUp cell] performClick:self];
598             }
599         }
600         else if( [fX264optTrellisPopUp isHidden] == true)
601         {
602             [[fX264optTrellisPopUp animator] setHidden:NO];
603             [[fX264optTrellisLabel animator] setHidden:NO];
604         }
605     }
606     
607     if( sender == fX264optAnalysePopUp || sender == nil || sender == fDisplayX264Options )
608     {
609         if ( [fX264optAnalysePopUp indexOfSelectedItem] == 1)
610         {
611             /* No analysis? Disable 8x8dct */
612             if( [fX264opt8x8dctSwitch isHidden] == false )
613             {
614                 [[fX264opt8x8dctSwitch animator] setHidden:YES];
615                 [[fX264opt8x8dctLabel animator] setHidden:YES];
616                 if ( [fX264opt8x8dctSwitch state] == 1 )
617                     [fX264opt8x8dctSwitch performClick:self];
618             }
619         }
620         else
621         {
622             if( [fX264opt8x8dctSwitch isHidden] == true )
623             {
624                 [[fX264opt8x8dctSwitch animator] setHidden:NO];
625                 [[fX264opt8x8dctLabel animator] setHidden:NO];
626             }
627         }
628     }
629     
630     if( sender == fX264optRefPopUp || sender == nil || sender == fDisplayX264Options )
631     {
632         if ( [fX264optRefPopUp indexOfSelectedItem] < 3)
633         {
634             if( [fX264optMixedRefsSwitch isHidden] == false )
635             {
636                 /* Only do mixed-refs when there are at least 2 refs to mix. */
637                 [[fX264optMixedRefsSwitch animator] setHidden:YES];
638                 [[fX264optMixedRefsLabel animator] setHidden:YES];
639                 if( [fX264optMixedRefsSwitch state] == 1 )
640                     [fX264optMixedRefsSwitch performClick:self];
641             }
642         }
643         else
644         {
645             if( [fX264optMixedRefsSwitch isHidden] == true )
646             {
647                 [[fX264optMixedRefsSwitch animator] setHidden:NO];
648                 [[fX264optMixedRefsLabel animator] setHidden:NO];
649             }
650         }
651     }
652     
653     if( sender == fX264optMotionEstPopUp || sender == nil || sender == fDisplayX264Options )
654     {
655         if ( [fX264optMotionEstPopUp indexOfSelectedItem] < 3 )
656         {
657             /* ME-range can only be above 16 if me >= umh
658               and changing it to < 16 is idiotic so hide it . */
659             if( [fX264optMERangePopUp isHidden] == false )
660             {
661                 [[fX264optMERangePopUp animator] setHidden:YES];
662                 [[fX264optMERangeLabel animator] setHidden:YES];
663                 if ( [fX264optMERangePopUp indexOfSelectedItem] > 0 )
664                 {
665                     [fX264optMERangePopUp selectItemAtIndex:0];
666                     [[fX264optMERangePopUp cell] performClick:self];
667                 }
668             }
669         }
670         else
671         {
672             if( [fX264optMERangePopUp isHidden] == true )
673             {
674                 [[fX264optMERangePopUp animator] setHidden:NO];
675                 [[fX264optMERangeLabel animator] setHidden:NO];
676             }
677         }
678     }
679     
680     if( sender == fX264optSubmePopUp || sender == nil || sender == fDisplayX264Options )
681     {
682         if( [fX264optSubmePopUp indexOfSelectedItem] != 0 && [fX264optSubmePopUp indexOfSelectedItem] < 7 )
683         {
684             /* No Psy-RDO or Psy=trel if subme < 6. */
685             if( [fX264optPsyRDSlider isHidden] == false )
686             {
687                 [[fX264optPsyRDSlider animator] setHidden:YES];
688                 [[fX264optPsyRDLabel animator] setHidden:YES];
689                 [[fX264optPsyRDSlider animator] setFloatValue:1];
690                 if ( [fX264optPsyRDSlider floatValue] < 1.0 )
691                 {
692                     [fX264optPsyRDSlider setFloatValue:1.0];
693                     [[fX264optPsyRDSlider cell] performClick:self];            
694                 }
695             }
696
697             if( [fX264optPsyTrellisSlider isHidden] == false)
698             {
699                 [[fX264optPsyTrellisSlider animator] setHidden:YES];
700                 [[fX264optPsyTrellisLabel animator] setHidden:YES];
701                 [[fX264optPsyTrellisSlider animator] setFloatValue:0];
702                 if ( [fX264optPsyTrellisSlider floatValue] > 0.0 )
703                 {
704                     [fX264optPsyTrellisSlider setFloatValue:0.0];
705                     [[fX264optPsyTrellisSlider cell] performClick:self];
706                 }
707             }
708         }
709         else
710         {
711             if( [fX264optPsyRDSlider isHidden] == true )
712             {
713                 [[fX264optPsyRDSlider animator] setHidden:NO];
714                 [[fX264optPsyRDLabel animator] setHidden:NO];
715             }
716
717             if( [fX264optTrellisPopUp indexOfSelectedItem] >= 2 && [fX264optCabacSwitch state] == true && [fX264optPsyTrellisSlider isHidden] == true )
718             {
719                 [[fX264optPsyTrellisSlider animator] setHidden:NO];
720                 [[fX264optPsyTrellisLabel animator] setHidden:NO];
721             }
722         }
723     }
724     
725     if( sender == fX264optTrellisPopUp || sender == nil || sender == fDisplayX264Options )
726     {
727         if( [fX264optTrellisPopUp indexOfSelectedItem] < 2 )
728         {
729             if( [fX264optPsyTrellisSlider isHidden] == false )
730             {
731                 /* No Psy-trellis without trellis. */
732                 [[fX264optPsyTrellisSlider animator] setHidden:YES];
733                 [[fX264optPsyTrellisLabel animator] setHidden:YES];
734                 [[fX264optPsyTrellisSlider animator] setFloatValue:0.0];
735                 [[fX264optPsyTrellisSlider cell] performClick:self];
736             }
737         }
738         else
739         {
740             if( ( [fX264optSubmePopUp indexOfSelectedItem] == 0 || [fX264optSubmePopUp indexOfSelectedItem] >= 7 ) && [fX264optCabacSwitch state] == true  && [fX264optPsyTrellisSlider isHidden] == true )
741             {
742                 [[fX264optPsyTrellisSlider animator] setHidden:NO];
743                 [[fX264optPsyTrellisLabel animator] setHidden:NO];
744             }
745         }
746     }
747 }
748
749 /**
750  * Resets the GUI widgets to the contents of the option string.
751  */
752 - (IBAction) X264AdvancedOptionsSetCurrentSettings: (id) sender
753 {
754     /* Set widgets depending on the opt string in field */
755     NSString * thisOpt; // The separated option such as "bframes=3"
756     NSString * optName = @""; // The option name such as "bframes"
757     NSString * optValue = @"";// The option value such as "3"
758     NSArray *currentOptsArray;
759     
760     /*First, we get an opt string to process */
761     NSString *currentOptString = [fDisplayX264Options stringValue];
762     
763     /* Verify there is an opt string to process by making sure an
764        option is getting its value set. If so, start to process it. */
765     NSRange currentOptRange = [currentOptString rangeOfString:@"="];
766     if (currentOptRange.location != NSNotFound)
767     {
768         /*Put individual options into an array based on the ":" separator for processing, result is "<opt>=<value>"*/
769         currentOptsArray = [currentOptString componentsSeparatedByString:@":"];
770         
771         /*iterate through the array and get <opts> and <values*/
772         int loopcounter;
773         int currentOptsArrayCount = [currentOptsArray count];
774         for (loopcounter = 0; loopcounter < currentOptsArrayCount; loopcounter++)
775         {
776             thisOpt = [currentOptsArray objectAtIndex:loopcounter];
777             
778             /* Verify the option sets a value */
779             NSRange splitOptRange = [thisOpt rangeOfString:@"="];            
780             if (splitOptRange.location != NSNotFound)
781             {
782                 /* Split thisOpt into an optName setting an optValue. */
783                 optName = [thisOpt substringToIndex:splitOptRange.location];
784                 optValue = [thisOpt substringFromIndex:splitOptRange.location + 1];
785                 
786                 /*Run through the available widgets for x264 opts and set them, as you add widgets, 
787                     they need to be added here. This should be moved to its own method probably*/
788                 
789                 /*bframes NSPopUpButton*/
790                 if ([optName isEqualToString:@"bframes"])
791                 {
792                     [fX264optBframesPopUp selectItemAtIndex:[optValue intValue]+1];
793                 }
794                 /*ref NSPopUpButton*/
795                 if ([optName isEqualToString:@"ref"])
796                 {
797                     [fX264optRefPopUp selectItemAtIndex:[optValue intValue]+1];
798                 }
799                 /*No Fast PSkip NSButton*/
800                 if ([optName isEqualToString:@"no-fast-pskip"])
801                 {
802                     [fX264optNfpskipSwitch setState:[optValue intValue]];
803                 }
804                 /*No Dict Decimate NSButton*/
805                 if ([optName isEqualToString:@"no-dct-decimate"])
806                 {
807                     [fX264optNodctdcmtSwitch setState:[optValue intValue]];
808                 }
809                 /*Sub Me NSPopUpButton*/
810                 if ([optName isEqualToString:@"subq"])
811                 {
812                     [fX264optSubmePopUp selectItemAtIndex:[optValue intValue]+1];
813                 }
814                 /*Trellis NSPopUpButton*/
815                 if ([optName isEqualToString:@"trellis"])
816                 {
817                     [fX264optTrellisPopUp selectItemAtIndex:[optValue intValue]+1];
818                 }
819                 /*Mixed Refs NSButton*/
820                 if ([optName isEqualToString:@"mixed-refs"])
821                 {
822                     [fX264optMixedRefsSwitch setState:[optValue intValue]];
823                 }
824                 /*Motion Estimation NSPopUpButton*/
825                 if ([optName isEqualToString:@"me"])
826                 {
827                     if ([optValue isEqualToString:@"dia"])
828                         [fX264optMotionEstPopUp selectItemAtIndex:1];
829                     else if ([optValue isEqualToString:@"hex"])
830                         [fX264optMotionEstPopUp selectItemAtIndex:2];
831                     else if ([optValue isEqualToString:@"umh"])
832                         [fX264optMotionEstPopUp selectItemAtIndex:3];
833                     else if ([optValue isEqualToString:@"esa"])
834                         [fX264optMotionEstPopUp selectItemAtIndex:4];
835                     else if ([optValue isEqualToString:@"tesa"])
836                         [fX264optMotionEstPopUp selectItemAtIndex:5];
837                 }
838                 /*ME Range NSPopUpButton*/
839                 if ([optName isEqualToString:@"merange"])
840                 {
841                     [fX264optMERangePopUp selectItemAtIndex:[optValue intValue]-3];
842                 }
843                 /* Adaptive B-Frames NSPopUpButton*/
844                 if ([optName isEqualToString:@"b-adapt"])
845                 {
846                     [fX264optBAdaptPopUp selectItemAtIndex:[optValue intValue]+1];
847                 }
848                 /*Weighted B-Frames NSButton*/
849                 if ([optName isEqualToString:@"weightb"])
850                 {
851                     [fX264optWeightBSwitch setState:[optValue intValue]];
852                 }
853                 /*B Pyramid NSPButton*/
854                 if ([optName isEqualToString:@"b-pyramid"])
855                 {
856                     [fX264optBPyramidSwitch setState:[optValue intValue]];
857                 }
858                 /*Direct B-frame Prediction NSPopUpButton*/
859                 if ([optName isEqualToString:@"direct"])
860                 {
861                     if ([optValue isEqualToString:@"none"])
862                         [fX264optDirectPredPopUp selectItemAtIndex:1];
863                     else if ([optValue isEqualToString:@"spatial"])
864                         [fX264optDirectPredPopUp selectItemAtIndex:2];
865                     else if ([optValue isEqualToString:@"temporal"])
866                         [fX264optDirectPredPopUp selectItemAtIndex:3];
867                     else if ([optValue isEqualToString:@"auto"])
868                         [fX264optDirectPredPopUp selectItemAtIndex:4];                        
869                 }
870                 /*Deblocking NSPopUpButtons*/
871                 if ([optName isEqualToString:@"deblock"])
872                 {
873                     NSString * alphaDeblock = @"";
874                     NSString * betaDeblock = @"";
875                     
876                     NSRange splitDeblock = [optValue rangeOfString:@","];
877                     alphaDeblock = [optValue substringToIndex:splitDeblock.location];
878                     betaDeblock = [optValue substringFromIndex:splitDeblock.location + 1];
879                     
880                     if ([alphaDeblock isEqualToString:@"0"] && [betaDeblock isEqualToString:@"0"])
881                     {
882                         /* When both filters are at 0, default */
883                         [fX264optAlphaDeblockPopUp selectItemAtIndex:0];                        
884                         [fX264optBetaDeblockPopUp selectItemAtIndex:0];                               
885                     }
886                     else
887                     {
888                         if (![alphaDeblock isEqualToString:@"0"])
889                         {
890                             /* Alpha isn't 0, so set it. The offset of 7 is
891                                because filters start at -6 instead of at 0. */
892                             [fX264optAlphaDeblockPopUp selectItemAtIndex:[alphaDeblock intValue]+7];
893                         }
894                         else
895                         {
896                             /* Set alpha filter to 0, which is 7 up
897                                because filters start at -6, not 0. */
898                             [fX264optAlphaDeblockPopUp selectItemAtIndex:7];                        
899                         }
900                         
901                         if (![betaDeblock isEqualToString:@"0"])
902                         {
903                             /* Beta isn't 0, so set it. */
904                             [fX264optBetaDeblockPopUp selectItemAtIndex:[betaDeblock intValue]+7];
905                         }
906                         else
907                         {
908                             /* Set beta filter to 0. */
909                             [fX264optBetaDeblockPopUp selectItemAtIndex:7];                        
910                         }
911                     }
912                 }
913                 /* Analysis NSPopUpButton */
914                 if ([optName isEqualToString:@"analyse"])
915                 {
916                     if ([optValue isEqualToString:@"p8x8,b8x8,i8x8,i4x4"])
917                     {
918                         /* Default ("some") */
919                         [fX264optAnalysePopUp selectItemAtIndex:0];
920                     }
921                     if ([optValue isEqualToString:@"none"])
922                     {
923                         [fX264optAnalysePopUp selectItemAtIndex:1];
924                     }
925                     if ([optValue isEqualToString:@"all"])
926                     {
927                         [fX264optAnalysePopUp selectItemAtIndex:2];
928                     }
929                 }
930                 /* 8x8 DCT NSButton */
931                 if ([optName isEqualToString:@"8x8dct"])
932                 {
933                     [fX264opt8x8dctSwitch setState:[optValue intValue]];
934                 }
935                 /* CABAC NSButton */
936                 if ([optName isEqualToString:@"cabac"])
937                 {
938                     [fX264optCabacSwitch setState:[optValue intValue]];
939                 }
940                 /* Psy-RD and Psy-Trellis NSSliders */
941                 if ([optName isEqualToString:@"psy-rd"])
942                 {
943                     NSString * rdOpt = @"";
944                     NSString * trellisOpt = @"";
945                     
946                     NSRange splitRD = [optValue rangeOfString:@","];
947                     rdOpt = [optValue substringToIndex:splitRD.location];
948                     trellisOpt = [optValue substringFromIndex:splitRD.location + 1];
949                     
950                     [fX264optPsyRDSlider setFloatValue:[rdOpt floatValue]];
951                     [fX264optPsyTrellisSlider setFloatValue:[trellisOpt floatValue]];
952                 }                                                              
953             }
954         }
955     }
956 }
957
958 - (NSString *) X264AdvancedOptionsOptIDToString: (id) widget
959 {
960     /*Determine which outlet is being used and set optName to process accordingly */
961     NSString * optNameToChange = @""; // The option name such as "bframes"
962     
963     if (widget == fX264optBframesPopUp)
964     {
965         optNameToChange = @"bframes";
966     }
967     if (widget == fX264optRefPopUp)
968     {
969         optNameToChange = @"ref";
970     }
971     if (widget == fX264optNfpskipSwitch)
972     {
973         optNameToChange = @"no-fast-pskip";
974     }
975     if (widget == fX264optNodctdcmtSwitch)
976     {
977         optNameToChange = @"no-dct-decimate";
978     }
979     if (widget == fX264optSubmePopUp)
980     {
981         optNameToChange = @"subq";
982     }
983     if (widget == fX264optTrellisPopUp)
984     {
985         optNameToChange = @"trellis";
986     }
987     if (widget == fX264optMixedRefsSwitch)
988     {
989         optNameToChange = @"mixed-refs";
990     }
991     if (widget == fX264optMotionEstPopUp)
992     {
993         optNameToChange = @"me";
994     }
995     if (widget == fX264optMERangePopUp)
996     {
997         optNameToChange = @"merange";
998     }
999     if (widget == fX264optBAdaptPopUp)
1000     {
1001         optNameToChange = @"b-adapt";
1002     }
1003     if (widget == fX264optWeightBSwitch)
1004     {
1005         optNameToChange = @"weightb";
1006     }
1007     if (widget == fX264optBPyramidSwitch)
1008     {
1009         optNameToChange = @"b-pyramid";
1010     }
1011     if (widget == fX264optDirectPredPopUp)
1012     {
1013         optNameToChange = @"direct";
1014     }
1015     if (widget == fX264optAlphaDeblockPopUp)
1016     {
1017         optNameToChange = @"deblock";
1018     }
1019     if (widget == fX264optBetaDeblockPopUp)
1020     {
1021         optNameToChange = @"deblock";
1022     }        
1023     if (widget == fX264optAnalysePopUp)
1024     {
1025         optNameToChange = @"analyse";
1026     }
1027     if (widget == fX264opt8x8dctSwitch)
1028     {
1029         optNameToChange = @"8x8dct";
1030     }
1031     if (widget == fX264optCabacSwitch)
1032     {
1033         optNameToChange = @"cabac";
1034     }
1035     if( widget == fX264optPsyRDSlider)
1036     {
1037         optNameToChange = @"psy-rd";
1038     }
1039     if( widget == fX264optPsyTrellisSlider)
1040     {
1041         optNameToChange = @"psy-rd";
1042     }
1043     
1044     return optNameToChange;
1045 }
1046
1047 - (NSString *) X264AdvancedOptionsWidgetToString: (NSString *) optName withID: (id) sender
1048 {
1049     NSString * thisOpt = @""; // The option=value string the method will return
1050     
1051     if ([optName isEqualToString:@"deblock"])
1052     {
1053         if ((([fX264optAlphaDeblockPopUp indexOfSelectedItem] == 0) || ([fX264optAlphaDeblockPopUp indexOfSelectedItem] == 7)) && (([fX264optBetaDeblockPopUp indexOfSelectedItem] == 0) || ([fX264optBetaDeblockPopUp indexOfSelectedItem] == 7)))
1054         {
1055             /* When both deblock widgets are 0 or default or a mix of the two,
1056                use a blank string, since deblocking defaults to 0,0.           */
1057             thisOpt = @"";                                
1058         }
1059         else
1060         {
1061             /* Otherwise the format is deblock=a,b, where a and b both have an array
1062                offset of 7 because deblocking values start at -6 instead of at zero. */
1063             thisOpt = [NSString stringWithFormat:@"%@=%d,%d",optName, ([fX264optAlphaDeblockPopUp indexOfSelectedItem] != 0) ? [fX264optAlphaDeblockPopUp indexOfSelectedItem]-7 : 0,([fX264optBetaDeblockPopUp indexOfSelectedItem] != 0) ? [fX264optBetaDeblockPopUp indexOfSelectedItem]-7 : 0];
1064         }
1065     }
1066     
1067     else if ([optName isEqualToString:@"psy-rd"])
1068     {
1069         if( [fX264optPsyRDSlider floatValue] == 1.0 && [fX264optPsyTrellisSlider floatValue] == 0.0 ) 
1070         {
1071             /* When  PsyRD is 1 and PsyTrel is 0 they're default values and can be ignored. */
1072             thisOpt = @"";                                
1073         }
1074         else
1075         {
1076             /* Otherwise the format is deblock=a,b, where a and b both have an array
1077                offset of 7 because deblocking values start at -6 instead of at zero. */
1078             thisOpt = [NSString stringWithFormat:@"%@=%0.1f,%0.1f", optName, [fX264optPsyRDSlider floatValue], [fX264optPsyTrellisSlider floatValue] ];
1079         }
1080     }
1081     
1082     else if /*Boolean Switches*/ ([optName isEqualToString:@"mixed-refs"] || [optName isEqualToString:@"weightb"] ||  [optName isEqualToString:@"b-pyramid"] || [optName isEqualToString:@"no-fast-pskip"] || [optName isEqualToString:@"no-dct-decimate"] || [optName isEqualToString:@"8x8dct"] )
1083     {
1084         /* Here is where we take care of the boolean options that work overtly:
1085            no-dct-decimate being on means no-dct-decimate=1, etc. Some options
1086            require the inverse, but those will be handled a couple lines down. */
1087         if ([sender state] == 0)
1088         {
1089             /* When these options are false, don't include them. They all default
1090                to being set off, so they don't need to be mentioned at all.       */
1091             thisOpt = @"";
1092         }
1093         else
1094         {
1095             /* Otherwise, include them as optioname=1 */
1096             thisOpt = [NSString stringWithFormat:@"%@=%d",optName,1];
1097         }
1098     }
1099     
1100     else if ([optName isEqualToString:@"cabac"])
1101     {
1102         /* CABAC is odd, in that it defaults to being on. That means
1103            it only needs to be included in the string when turned off. */
1104         if ([sender state] == 1)
1105         {
1106             /* It's true so don't include it. */
1107             thisOpt = @"";
1108         }
1109         else
1110         {
1111             /* Otherwise, include cabac=0 in the string to enable CAVLC. */
1112             thisOpt = [NSString stringWithFormat:@"%@=%d",optName,0];
1113         }
1114     }
1115                                             
1116     else if (([sender indexOfSelectedItem] == 0) && (sender != fX264optAlphaDeblockPopUp) && (sender != fX264optBetaDeblockPopUp) ) // means that "unspecified" is chosen, lets then remove it from the string
1117     {
1118         /* When a widget is at index 0, it's default. Default means don't add to the string.
1119            The exception for deblocking is because for those, *both* need to at index 0
1120            for it to default, so it's handled separately, above this section.                */
1121         thisOpt = @"";
1122     }
1123     
1124     else if ([optName isEqualToString:@"me"])
1125     {
1126         /* Motion estimation uses string values, so this switch
1127            pairs the widget index with the right value string.  */
1128         switch ([sender indexOfSelectedItem])
1129         {   
1130             case 1:
1131                 thisOpt = [NSString stringWithFormat:@"%@=%@",optName,@"dia"];
1132                 break;
1133                 
1134             case 2:
1135                 thisOpt = [NSString stringWithFormat:@"%@=%@",optName,@"hex"];
1136                 break;
1137                 
1138             case 3:
1139                 thisOpt = [NSString stringWithFormat:@"%@=%@",optName,@"umh"];
1140                 break;
1141                 
1142             case 4:
1143                 thisOpt = [NSString stringWithFormat:@"%@=%@",optName,@"esa"];
1144                 break;
1145             
1146             case 5:
1147                 thisOpt = [NSString stringWithFormat:@"%@=%@",optName,@"tesa"];
1148             
1149             default:
1150                 break;
1151         }
1152     }
1153     
1154     else if ([optName isEqualToString:@"direct"])
1155     {
1156         /* Direct prediction uses string values, so this switch
1157            pairs the right string value with the right widget index. */
1158         switch ([sender indexOfSelectedItem])
1159         {   
1160             case 1:
1161                 thisOpt = [NSString stringWithFormat:@"%@=%@",optName,@"none"];
1162                 break;
1163                 
1164             case 2:
1165                 thisOpt = [NSString stringWithFormat:@"%@=%@",optName,@"spatial"];
1166                 break;
1167                 
1168             case 3:
1169                 thisOpt = [NSString stringWithFormat:@"%@=%@",optName,@"temporal"];
1170                 break;
1171                 
1172             case 4:
1173                 thisOpt = [NSString stringWithFormat:@"%@=%@",optName,@"auto"];
1174                 break;
1175                 
1176             default:
1177                 break;
1178         }
1179     }
1180     
1181     else if ([optName isEqualToString:@"analyse"])
1182     {
1183         /* Analysis uses string values as well. */
1184         switch ([sender indexOfSelectedItem])
1185         {   
1186             case 1:
1187                 thisOpt = [NSString stringWithFormat:@"%@=%@",optName,@"none"];
1188                 break;
1189                 
1190             case 2:
1191                 thisOpt = [NSString stringWithFormat:@"%@=%@",optName,@"all"];
1192                 break;
1193                 
1194             default:
1195                 break;
1196         }
1197     }
1198     
1199     else if ([optName isEqualToString:@"merange"])
1200     {
1201         /* Motion estimation range uses an odd array offset because in addition
1202            to starting with index 0 as default, index 1 starts at 4 instead of 1,
1203            because merange can't go below 4. So it has to be handled separately.  */
1204         thisOpt = [NSString stringWithFormat:@"%@=%d",optName,[sender indexOfSelectedItem]+3];
1205     }
1206     
1207     else if ([optName isEqualToString:@"b-adapt"])
1208     {
1209         /* B-adapt starts at index 0 with default then goes 0, 1, 2)*/
1210         thisOpt = [NSString stringWithFormat:@"%@=%d", optName, [sender indexOfSelectedItem]-1];
1211     }
1212     
1213     else // we have a valid value to change, so change it
1214     {
1215         if ( [sender indexOfSelectedItem] != 0 )
1216         /* Here's our general case, that catches things like ref frames and b-frames.
1217            Basically, any options that are PopUp menus with index 0 as default and
1218            index 1 as 1, with numerical values, are all handled right here. All of
1219            the above stuff is for the exceptions to the general case.              */
1220             thisOpt = [NSString stringWithFormat:@"%@=%d",optName,[sender indexOfSelectedItem]-1];
1221     }
1222     
1223     return thisOpt;
1224 }
1225
1226 - (BOOL) X264AdvancedOptionsIsOpt: (NSString *) optNameToChange inString: (NSString *) currentOptString
1227 {
1228     /* If the option is in the string but not the beginning of it,
1229        it will be in the form of ":optName=value" so we really want
1230        to be looking for ":optNameToChange=" rather than "optNameToChange". */
1231     NSString *checkOptNameToChange = [NSString stringWithFormat:@":%@=",optNameToChange];
1232     
1233     /* Now we store the part of the string up through the option name in currentOptRange. */
1234     NSRange currentOptRange = [currentOptString rangeOfString:checkOptNameToChange];
1235
1236     /*  We need to know if the option is at the beginning of the string.
1237         If it is at the start, it won't be preceded by a colon.
1238         To figure this out, we'll use the rangeOfString method. First,
1239         store what the option name would be if if it was at the beginning,
1240         in checkOptNameToChangeBeginning. Then, find its range in the string.
1241         If the range is 0, it's the first option listed in the string.       */        
1242     NSString *checkOptNameToChangeBeginning = [NSString stringWithFormat:@"%@=",optNameToChange];
1243     NSRange currentOptRangeBeginning = [currentOptString rangeOfString:checkOptNameToChangeBeginning];
1244
1245     if (currentOptRange.location != NSNotFound || currentOptRangeBeginning.location == 0)
1246         return true;
1247     else
1248         return false;
1249
1250
1251 /**
1252  * Resets the option string to mirror the GUI widgets.
1253  */
1254 - (IBAction) X264AdvancedOptionsChanged: (id) sender
1255 {
1256     /* Look up the equivalent string option name of the calling widget. */
1257     NSString * optNameToChange = [self X264AdvancedOptionsOptIDToString: sender];
1258     
1259     NSString * thisOpt = @"";  // The separated option such as "bframes=3"
1260     NSString * optName = @"";  // The option name such as "bframes"
1261     NSString * optValue = @""; // The option value such as "3"
1262     NSArray *currentOptsArray;
1263     
1264     /* Get the current opt string being displayed. */
1265     NSString *currentOptString = [fDisplayX264Options stringValue];
1266     
1267     /* There are going to be a few possibilities.
1268        - The option might start off the string.
1269        - The option might be in the middle of the string.
1270        - The option might not be in the string at all yet.
1271        - The string itself might not yet exist.             */
1272     
1273     if( [self X264AdvancedOptionsIsOpt: optNameToChange inString: currentOptString] )
1274     {
1275         /* If the option is in the string wth a semicolon, or starts the string, it's time to edit.
1276            This means parsing the whole string into an array of options and values. From there,
1277            iterate through the options, and when you reach the one that's been changed, edit it.   */
1278         
1279         /* Create new empty opt string*/
1280         NSString *changedOptString = @"";
1281         
1282         /* Put individual options into an array based on the ":"
1283            separator for processing, result is "<opt>=<value>"   */
1284         currentOptsArray = [currentOptString componentsSeparatedByString:@":"];
1285         
1286         /* Iterate through the array and get <opts> and <values*/
1287         int loopcounter;
1288         int currentOptsArrayCount = [currentOptsArray count];
1289         for (loopcounter = 0; loopcounter < currentOptsArrayCount; loopcounter++)
1290         {
1291             thisOpt = [currentOptsArray objectAtIndex:loopcounter];
1292             NSRange splitOptRange = [thisOpt rangeOfString:@"="];
1293             
1294             if (splitOptRange.location != NSNotFound)
1295             {
1296                 /* First off, it's time to handle option strings that
1297                    already have at least one option=value pair in them. */
1298                    
1299                 optName = [thisOpt substringToIndex:splitOptRange.location];
1300                 optValue = [thisOpt substringFromIndex:splitOptRange.location + 1];
1301
1302                 /*If the optNameToChange is found, appropriately change the value or delete it if
1303                     "Unspecified" is set.*/
1304                 if ([optName isEqualToString:optNameToChange])
1305                 {
1306                     thisOpt = [self X264AdvancedOptionsWidgetToString: optName withID: sender];
1307                 }
1308             }
1309             
1310             /* Construct New String for opts here */
1311             if ([thisOpt isEqualToString:@""])
1312             {
1313                 /* Blank option, so just add it to the string. (Why?) */
1314                 changedOptString = [NSString stringWithFormat:@"%@%@",changedOptString,thisOpt];
1315             }
1316             else
1317             {
1318                 if ([changedOptString isEqualToString:@""])
1319                 {
1320                     /* No existing string, make the string this option. */
1321                     changedOptString = [NSString stringWithFormat:@"%@",thisOpt];
1322                 }
1323                 else
1324                 {
1325                     /* Existing string, existing option. Append the
1326                        option to the string, preceding it with a colon. */
1327                     changedOptString = [NSString stringWithFormat:@"%@:%@",changedOptString,thisOpt];
1328                 }
1329             }
1330         }
1331         
1332         /* Change the dislayed option string to reflect the new modified settings */
1333         [fDisplayX264Options setStringValue:[NSString stringWithFormat:changedOptString]];    
1334     }
1335     else // if none exists, add it to the string
1336     {
1337         /* This is where options that aren't already in the string are handled. */
1338         if ([[fDisplayX264Options stringValue] isEqualToString: @""])
1339         {
1340             
1341             [fDisplayX264Options setStringValue:
1342                 [self X264AdvancedOptionsWidgetToString: optNameToChange withID: sender]];
1343         }
1344         else
1345         {
1346             /* The string isn't empty, and the option isn't already in it, so
1347                it will need to be appended to the current string with a colon,
1348                as long as the string to be appended isn't just blank (default). */
1349             if( [[self X264AdvancedOptionsWidgetToString: optNameToChange withID: sender] isEqualToString: @""] == false )
1350             {
1351                 [fDisplayX264Options setStringValue:
1352                     [NSString stringWithFormat:@"%@:%@",
1353                         currentOptString,
1354                         [self X264AdvancedOptionsWidgetToString: optNameToChange withID: sender] ]];                
1355             }
1356         }
1357     }
1358     
1359     /* We now need to reset the opt widgets since we changed some stuff */        
1360     [self X264AdvancedOptionsSet:sender];        
1361 }
1362
1363 @end