OSDN Git Service

Change the fifo size from being statically tuned for a Mac Pro with 4 CPUs to dynamic...
[handbrake-jp/handbrake-jp-git.git] / macosx / PictureController.mm
index 30d7fae..865f1c2 100644 (file)
@@ -6,6 +6,15 @@
 
 #include "PictureController.h"
 
+@interface PictureController (Private)
+
+- (NSSize)optimalViewSizeForImageSize: (NSSize)imageSize;
+- (void)resizeSheetForViewSize: (NSSize)viewSize;
+- (void)setViewSize: (NSSize)viewSize;
+- (BOOL)viewNeedsToResizeToSize: (NSSize)newSize;
+
+@end
+
 static int GetAlignedSize( int size )
 {
     int result = 1;
@@ -18,6 +27,16 @@ static int GetAlignedSize( int size )
 
 @implementation PictureController
 
+- (id)initWithDelegate:(id)del
+{
+       if (self = [super init])
+       {
+               delegate = del;
+        [self loadMyNibFile];
+       }
+       return self;
+}
+
 - (void) SetHandle: (hb_handle_t *) handle
 {
     fHandle = handle;
@@ -32,10 +51,10 @@ static int GetAlignedSize( int size )
 
     [fWidthStepper  setValueWraps: NO];
     [fWidthStepper  setIncrement: 16];
-    [fWidthStepper  setMinValue: 16];
+    [fWidthStepper  setMinValue: 64];
     [fHeightStepper setValueWraps: NO];
     [fHeightStepper setIncrement: 16];
-    [fHeightStepper setMinValue: 16];
+    [fHeightStepper setMinValue: 64];
 
     [fCropTopStepper    setIncrement: 2];
     [fCropTopStepper    setMinValue:  0];
@@ -85,19 +104,55 @@ static int GetAlignedSize( int size )
     [fCropBottomStepper setMaxValue: title->height/2-2];
     [fCropLeftStepper   setMaxValue: title->width/2-2];
     [fCropRightStepper  setMaxValue: title->width/2-2];
-    [fDeinterlaceCheck  setState:    job->deinterlace ? NSOnState : NSOffState];
-       [fPARCheck  setState:    job->pixel_ratio ? NSOnState : NSOffState];
-       if ([fAutoCropMainWindow  intValue] == 0)
+    
+       
+       /* we use a popup to show the deinterlace settings */
+       [fDeinterlacePopUp removeAllItems];
+    [fDeinterlacePopUp addItemWithTitle: @"None"];
+    [fDeinterlacePopUp addItemWithTitle: @"Fast"];
+    [fDeinterlacePopUp addItemWithTitle: @"Slow"];
+       [fDeinterlacePopUp addItemWithTitle: @"Slower"];
+       [fDeinterlacePopUp addItemWithTitle: @"Slowest"];
+    
+       /* Set deinterlaces level according to the integer in the main window */
+       [fDeinterlacePopUp selectItemAtIndex: fPictureFilterSettings.deinterlace];
+
+       [fPARCheck setState:(job->pixel_ratio ? NSOnState : NSOffState)];
+    /* We initially set the previous state of keep ar to on */
+    keepAspectRatioPreviousState = 1;
+       if (!autoCrop)
        {
-       [fCropMatrix  selectCellAtRow: 1 column:0];
+        [fCropMatrix  selectCellAtRow: 1 column:0];
+        /* If auto, lets set the crop steppers according to current job->crop values */
+        [fCropTopStepper    setIntValue: job->crop[0]];
+        [fCropTopField      setIntValue: job->crop[0]];
+        [fCropBottomStepper setIntValue: job->crop[1]];
+        [fCropBottomField   setIntValue: job->crop[1]];
+        [fCropLeftStepper   setIntValue: job->crop[2]];
+        [fCropLeftField     setIntValue: job->crop[2]];
+        [fCropRightStepper  setIntValue: job->crop[3]];
+        [fCropRightField    setIntValue: job->crop[3]];
        }
        else
        {
-       [fCropMatrix  selectCellAtRow: 0 column:0];
+        [fCropMatrix  selectCellAtRow: 0 column:0];
        }
+       
+       
+       
+       /* we use a popup to show the denoise settings */
+       [fDenoisePopUp removeAllItems];
+    [fDenoisePopUp addItemWithTitle: @"None"];
+    [fDenoisePopUp addItemWithTitle: @"Weak"];
+       [fDenoisePopUp addItemWithTitle: @"Medium"];
+    [fDenoisePopUp addItemWithTitle: @"Strong"];
+       /* Set denoises level according to the integer in the main window */
+       [fDenoisePopUp selectItemAtIndex: fPictureFilterSettings.denoise];
+       
     MaxOutputWidth = job->width;
        MaxOutputHeight = job->height;
     fPicture = 0;
+
     [self SettingsChanged: nil];
 }
 
@@ -147,23 +202,42 @@ static int GetAlignedSize( int size )
        int arpheight = fTitle->job->pixel_aspect_height;
        int displayparwidth = titlewidth * arpwidth / arpheight;
        int displayparheight = fTitle->height-fTitle->job->crop[0]-fTitle->job->crop[1];
-       if (fTitle->job->pixel_ratio == 1)
-       {
-       
-       [fInfoField setStringValue: [NSString stringWithFormat:
-       @"Source: %dx%d, Output: %dx%d, Anamorphic: %dx%d", fTitle->width, fTitle->height,
-       titlewidth, displayparheight, displayparwidth,
-       displayparheight]];
-       
-       
-       }
-       else
-       {
-       [fInfoField setStringValue: [NSString stringWithFormat:
+
+    NSSize displaySize = NSMakeSize( (float)fTitle->width, (float)fTitle->height );
+    if( fTitle->job->pixel_ratio == 1 )
+    {
+        [fInfoField setStringValue:[NSString stringWithFormat:
+                            @"Source: %dx%d, Output: %dx%d, Anamorphic: %dx%d",
+                            fTitle->width, fTitle->height, titlewidth,
+                            displayparheight, displayparwidth, displayparheight]];
+        displaySize.width *= ((float)arpwidth) / ((float)arpheight);
+    }
+    else
+    {
+    [fInfoField setStringValue: [NSString stringWithFormat:
         @"Source: %dx%d, Output: %dx%d", fTitle->width, fTitle->height,
-        fTitle->job->width, fTitle->job->height]];     
-       }
+        fTitle->job->width, fTitle->job->height]];
+    }
 
+    NSSize viewSize = [self optimalViewSizeForImageSize:displaySize];
+    if( [self viewNeedsToResizeToSize:viewSize] )
+    {
+        [self resizeSheetForViewSize:viewSize];
+        [self setViewSize:viewSize];
+    }
+    
+    // Show the scaled text (use the height to check since the width can vary
+    // with anamorphic video).
+    if( ((int)viewSize.height) != fTitle->height )
+    {
+        float scale = viewSize.width / ((float)fTitle->width);
+        NSString *scaleString = [NSString stringWithFormat:
+            NSLocalizedString( @" (Preview scaled to %.0f%% actual size)",
+                               @"String shown when a preview is scaled" ),
+            scale * 100.0];
+        [fInfoField setStringValue:
+            [[fInfoField stringValue] stringByAppendingString:scaleString]];
+    }
 
     [fPrevButton setEnabled: ( fPicture > 0 )];
     [fNextButton setEnabled: ( fPicture < 9 )];
@@ -173,56 +247,74 @@ static int GetAlignedSize( int size )
 {
     hb_job_t * job = fTitle->job;
     
-       if ([fPARCheck state] == 1 )
+       if( [fPARCheck state] == NSOnState )
        {
-       [fWidthStepper      setIntValue: MaxOutputWidth];
-       [fWidthField        setIntValue: MaxOutputWidth];
-       
-       /* This will show correct anamorphic height values, but
-       show distorted preview picture ratio */
-       [fHeightStepper      setIntValue: fTitle->height-fTitle->job->crop[0]-fTitle->job->crop[1]];
-       [fHeightField        setIntValue: fTitle->height-fTitle->job->crop[0]-fTitle->job->crop[1]];
-       
-       /* This will show wrong anamorphic height values, but
-       show proper preview picture ratio */
-       //[fHeightStepper      setIntValue: MaxOutputHeight];
-       //[fHeightField        setIntValue: MaxOutputHeight];
-       [fRatioCheck        setState: 0];
-
-       [fWidthStepper setEnabled: NO];
-       [fWidthField setEnabled: NO];
-       [fHeightStepper setEnabled: NO];
-       [fHeightField setEnabled: NO];
-       [fRatioCheck setEnabled: NO];
-       
-       
-       }
-       else
+        [fWidthStepper      setIntValue: MaxOutputWidth];
+        [fWidthField        setIntValue: MaxOutputWidth];
+        
+        /* This will show correct anamorphic height values, but
+            show distorted preview picture ratio */
+        [fHeightStepper      setIntValue: fTitle->height-fTitle->job->crop[0]-fTitle->job->crop[1]];
+        [fHeightField        setIntValue: fTitle->height-fTitle->job->crop[0]-fTitle->job->crop[1]];
+
+        /* if the sender is the Anamorphic checkbox, record the state
+           of KeepAspect Ratio so it can be reset if Anamorphic is unchecked again */
+        if (sender == fPARCheck)
+        {
+        keepAspectRatioPreviousState = [fRatioCheck state];
+        }
+        [fRatioCheck setState:NSOffState];
+        [fRatioCheck setEnabled: NO];
+        
+        [fWidthStepper setEnabled: NO];
+        [fWidthField setEnabled: NO];
+        [fHeightStepper setEnabled: NO];
+        [fHeightField setEnabled: NO];
+        
+    }
+    else
        {
-       [fWidthStepper setEnabled: YES];
-       [fWidthField setEnabled: YES];
-       [fHeightStepper setEnabled: YES];
-       [fHeightField setEnabled: YES];
-       [fRatioCheck setEnabled: YES];
+        [fWidthStepper setEnabled: YES];
+        [fWidthField setEnabled: YES];
+        [fHeightStepper setEnabled: YES];
+        [fHeightField setEnabled: YES];
+        [fRatioCheck setEnabled: YES];
+        /* if the sender is the Anamorphic checkbox, we return the
+           keep AR checkbox to its previous state */
+        if (sender == fPARCheck)
+        {
+        [fRatioCheck setState:keepAspectRatioPreviousState];
+        }
+        
        }
        
-       
-       
     job->width       = [fWidthStepper  intValue];
     job->height      = [fHeightStepper intValue];
     job->keep_ratio  = ( [fRatioCheck state] == NSOnState );
-    job->deinterlace = ( [fDeinterlaceCheck state] == NSOnState );
+    
+       fPictureFilterSettings.deinterlace = [fDeinterlacePopUp indexOfSelectedItem];
+    /* if the gui deinterlace settings are fast through slowest, the job->deinterlace
+       value needs to be set to one, for the job as well as the previews showing deinterlacing
+       otherwise set job->deinterlace to 0 or "off" */
+    if (fPictureFilterSettings.deinterlace > 0)
+    {
+    job->deinterlace  = 1;
+    }
+    else
+    {
+    job->deinterlace  = 0;
+    }
+    fPictureFilterSettings.denoise     = [fDenoisePopUp indexOfSelectedItem];
+    fPictureFilterSettings.detelecine  = [fDetelecineCheck state];
        job->pixel_ratio = ( [fPARCheck state] == NSOnState );
 
+    autoCrop = ( [fCropMatrix selectedRow] == 0 );
+    [fCropTopStepper    setEnabled: !autoCrop];
+    [fCropBottomStepper setEnabled: !autoCrop];
+    [fCropLeftStepper   setEnabled: !autoCrop];
+    [fCropRightStepper  setEnabled: !autoCrop];
 
-
-    bool autocrop = ( [fCropMatrix selectedRow] == 0 );
-    [fCropTopStepper    setEnabled: !autocrop];
-    [fCropBottomStepper setEnabled: !autocrop];
-    [fCropLeftStepper   setEnabled: !autocrop];
-    [fCropRightStepper  setEnabled: !autocrop];
-       [fAutoCropMainWindow  setStringValue: [NSString stringWithFormat:@"%d",autocrop]];
-    if( autocrop )
+    if( autoCrop )
     {
         memcpy( job->crop, fTitle->crop, 4 * sizeof( int ) );
     }
@@ -269,7 +361,14 @@ static int GetAlignedSize( int size )
     [fCropLeftField     setIntValue: job->crop[2]];
     [fCropRightStepper  setIntValue: job->crop[3]];
     [fCropRightField    setIntValue: job->crop[3]];
-    [self Display: HB_ANIMATE_NONE];
+    /* Sanity Check Here for < 16 px preview to avoid
+       crashing hb_get_preview. In fact, just for kicks
+       lets getting previews at a min limit of 32, since
+       no human can see any meaningful detail below that */
+    if (job->width >= 64 && job->height >= 64)
+    {
+        [self Display: HB_ANIMATE_NONE];
+    }
 }
 
 - (IBAction) PreviousPicture: (id) sender
@@ -294,8 +393,188 @@ static int GetAlignedSize( int size )
 
 - (IBAction) ClosePanel: (id) sender
 {
+    if ([delegate respondsToSelector:@selector(pictureSettingsDidChange)])
+        [delegate pictureSettingsDidChange];
+        
+    [NSApp endSheet: fPicturePanel];
+    [fPicturePanel orderOut: self];
+}
 
-       [NSApp stopModal];
+- (BOOL) autoCrop
+{
+    return autoCrop;
+}
+- (void) setAutoCrop: (BOOL) setting
+{
+    autoCrop = setting;
 }
 
+- (int) detelecine
+{
+    return fPictureFilterSettings.detelecine;
+}
+
+- (void) setDetelecine: (int) setting
+{
+    fPictureFilterSettings.detelecine = setting;
+}
+
+- (int) deinterlace
+{
+    return fPictureFilterSettings.deinterlace;
+}
+
+- (void) setDeinterlace: (int) setting {
+    fPictureFilterSettings.deinterlace = setting;
+}
+
+- (int) denoise
+{
+    return fPictureFilterSettings.denoise;
+}
+
+- (void) setDenoise: (int) setting
+{
+    fPictureFilterSettings.denoise = setting;
+}
+
+- (void)showPanelInWindow: (NSWindow *)fWindow forTitle: (hb_title_t *)title
+{
+    [self SetTitle:title];
+    
+    [NSApp beginSheet:fPicturePanel
+       modalForWindow:fWindow
+        modalDelegate:nil
+       didEndSelector:nil
+          contextInfo:NULL];
+}
+
+- (BOOL) loadMyNibFile
+{
+    if(![NSBundle loadNibNamed:@"PictureSettings" owner:self])
+    {
+        NSLog(@"Warning! Could not load myNib file.\n");
+        return NO;
+    }
+    
+    return YES;
+}
+
+@end
+
+@implementation PictureController (Private)
+
+//
+// -[PictureController(Private) optimalViewSizeForImageSize:]
+//
+// Given the size of the preview image to be shown, returns the best possible
+// size for the OpenGL view.
+//
+- (NSSize)optimalViewSizeForImageSize: (NSSize)imageSize
+{
+    // The min size is 320x240
+    float minWidth = 320.0;
+    float minHeight = 240.0;
+    
+    // The max size of the view is when the sheet is taking up 85% of the screen.
+    NSSize screenSize = [[NSScreen mainScreen] frame].size;
+    NSSize sheetSize = [fPicturePanel frame].size;
+    NSSize viewAreaSize = [fPictureGLViewArea frame].size;
+    float paddingX = sheetSize.width - viewAreaSize.width;
+    float paddingY = sheetSize.height - viewAreaSize.height;
+    float maxWidth = (0.85 * screenSize.width) - paddingX;
+    float maxHeight = (0.85 * screenSize.height) - paddingY;
+    
+    NSSize resultSize = imageSize;
+    
+    // Its better to have a view that's too small than a view that's too big, so
+    // apply the maximum constraints last.
+    if( resultSize.width < minWidth )
+    {
+        resultSize.height *= (minWidth / resultSize.width);
+        resultSize.width = minWidth;
+    }
+    if( resultSize.height < minHeight )
+    {
+        resultSize.width *= (minHeight / resultSize.height);
+        resultSize.height = minHeight;
+    }
+    if( resultSize.width > maxWidth )
+    {
+        resultSize.height *= (maxWidth / resultSize.width);
+        resultSize.width = maxWidth;
+    }
+    if( resultSize.height > maxHeight )
+    {
+        resultSize.width *= (maxHeight / resultSize.height);
+        resultSize.height = maxHeight;
+    }
+    
+    return resultSize;
+}
+
+//
+// -[PictureController(Private) resizePanelForViewSize:animate:]
+//
+// Resizes the entire sheet to accomodate an OpenGL view of a particular size.
+//
+- (void)resizeSheetForViewSize: (NSSize)viewSize
+{
+    // Figure out the deltas for the new frame area
+    NSSize currentSize = [fPictureGLViewArea frame].size;
+    float deltaX = viewSize.width - currentSize.width;
+    float deltaY = viewSize.height - currentSize.height;
+    
+    // Now resize the whole panel by those same deltas, but don't exceed the min
+    NSRect frame = [fPicturePanel frame];
+    NSSize maxSize = [fPicturePanel maxSize];
+    NSSize minSize = [fPicturePanel minSize];
+    frame.size.width += deltaX;
+    frame.size.height += deltaY;
+    if( frame.size.width < minSize.width )
+    {
+        frame.size.width = minSize.width;
+    }
+    if( frame.size.height < minSize.height )
+    {
+        frame.size.height = minSize.height;
+    }
+
+    // But now the sheet is off-center, so also shift the origin to center it and
+    // keep the top aligned.
+    frame.origin.x -= (deltaX / 2.0);
+    frame.origin.y -= deltaY;
+
+    [fPicturePanel setFrame:frame display:YES animate:YES];
+}
+
+//
+// -[PictureController(Private) setViewSize:]
+//
+// Changes the OpenGL view's size and centers it vertially inside of its area.
+// Assumes resizeSheetForViewSize: has already been called.
+//
+- (void)setViewSize: (NSSize)viewSize
+{
+    [fPictureGLView setFrameSize:viewSize];
+    
+    // center it vertically
+    NSPoint origin = [fPictureGLViewArea frame].origin;
+    origin.y += ([fPictureGLViewArea frame].size.height -
+                 [fPictureGLView frame].size.height) / 2.0;
+    [fPictureGLView setFrameOrigin:origin];
+}
+
+//
+// -[PictureController(Private) viewNeedsToResizeToSize:]
+//
+// Returns YES if the view will need to resize to match the given size.
+//
+- (BOOL)viewNeedsToResizeToSize: (NSSize)newSize
+{
+    NSSize viewSize = [fPictureGLView frame].size;
+    return (newSize.width != viewSize.width || newSize.height != viewSize.height);
+}
+
+
 @end