]> git.sesse.net Git - vlc/blobdiff - modules/gui/macosx/VideoView.m
macosx/CAS: implement basic error checking for the stream-out settings
[vlc] / modules / gui / macosx / VideoView.m
index 350d4c1df0a48c5010b305f0437cced422f59414..cd88942d5f47d1a5c05ae25589062607147898f6 100644 (file)
@@ -14,7 +14,7 @@
  * it under the terms of the GNU General Public License as published by
  * the Free Software Foundation; either version 2 of the License, or
  * (at your option) any later version.
- * 
+ *
  * This program is distributed in the hope that it will be useful,
  * but WITHOUT ANY WARRANTY; without even the implied warranty of
  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 #import "MainWindow.h"
 
 #import <vlc_common.h>
-#import <vlc_vout_window.h>
-#import <vlc_vout_display.h>
 #import <vlc_keys.h>
-#import <vlc_mouse.h>
+
+
 /*****************************************************************************
- * DeviceCallback: Callback triggered when the video-device variable is changed
+ * VLCVoutView implementation
  *****************************************************************************/
-int DeviceCallback( vlc_object_t *p_this, const char *psz_variable,
-                     vlc_value_t old_val, vlc_value_t new_val, void *param )
+@implementation VLCVoutView
+
+#pragma mark -
+#pragma mark drag & drop support
+
+- (void)dealloc
 {
-    vlc_value_t val;
-    vout_thread_t *p_vout = (vout_thread_t *)p_this;
+    if (p_vout)
+        vlc_object_release(p_vout);
+
+    [self unregisterDraggedTypes];
+    [super dealloc];
+}
 
-    msg_Dbg( p_vout, "set %"PRId64, new_val.i_int );
-    var_Create( p_vout->p_libvlc, "video-device", VLC_VAR_INTEGER );
-    var_Set( p_vout->p_libvlc, "video-device", new_val );
+-(id)initWithFrame:(NSRect)frameRect
+{
+    if (self = [super initWithFrame:frameRect]) {
+        [self registerForDraggedTypes:[NSArray arrayWithObject: NSFilenamesPboardType]];
+    }
 
-    val.b_bool = true;
-    var_Set( p_vout, "intf-change", val );
-    return VLC_SUCCESS;
+    i_lastScrollWheelDirection = 0;
+    f_cumulated_magnification = 0.0;
+
+    return self;
 }
 
-/*****************************************************************************
- * VLCOpenGLVideoView interface excerpt
- * full implementation in modules/video_output/macosx.m:95
- *****************************************************************************/
-@interface VLCOpenGLVideoView : NSOpenGLView
+- (NSDragOperation)draggingEntered:(id <NSDraggingInfo>)sender
 {
+    if ((NSDragOperationGeneric & [sender draggingSourceOperationMask]) == NSDragOperationGeneric)
+        return NSDragOperationGeneric;
+    return NSDragOperationNone;
 }
-- (vout_display_t *)voutDisplay;
-@end
 
-/*****************************************************************************
- * VLCVoutView implementation
- *****************************************************************************/
-@implementation VLCVoutView
-- (void)closeVout
+- (BOOL)prepareForDragOperation:(id <NSDraggingInfo>)sender
 {
-    vout_thread_t * p_vout = getVout();
-    if( !p_vout )
-    {
-        var_DelCallback( p_vout, "video-device", DeviceCallback, NULL );
-        vlc_object_release( p_vout );
-    }
+    return YES;
 }
 
-- (void)scrollWheel:(NSEvent *)theEvent
+- (BOOL)performDragOperation:(id <NSDraggingInfo>)sender
 {
-    VLCControls * o_controls = (VLCControls *)[[NSApp delegate] controls];
-    [o_controls scrollWheel: theEvent];
+    BOOL b_returned;
+    b_returned = [[VLCCoreInteraction sharedInstance] performDragOperation: sender];
+
+    [self setNeedsDisplay:YES];
+    return b_returned;
 }
 
+- (void)concludeDragOperation:(id <NSDraggingInfo>)sender
+{
+    [self setNeedsDisplay:YES];
+}
+
+#pragma mark -
+#pragma mark vout actions
+
 - (void)keyDown:(NSEvent *)o_event
 {
     unichar key = 0;
@@ -99,197 +108,184 @@ int DeviceCallback( vlc_object_t *p_this, const char *psz_variable,
 
     i_pressed_modifiers = [o_event modifierFlags];
 
-    if( i_pressed_modifiers & NSShiftKeyMask )
+    if (i_pressed_modifiers & NSShiftKeyMask)
         val.i_int |= KEY_MODIFIER_SHIFT;
-    if( i_pressed_modifiers & NSControlKeyMask )
+    if (i_pressed_modifiers & NSControlKeyMask)
         val.i_int |= KEY_MODIFIER_CTRL;
-    if( i_pressed_modifiers & NSAlternateKeyMask )
+    if (i_pressed_modifiers & NSAlternateKeyMask)
         val.i_int |= KEY_MODIFIER_ALT;
-    if( i_pressed_modifiers & NSCommandKeyMask )
+    if (i_pressed_modifiers & NSCommandKeyMask)
         val.i_int |= KEY_MODIFIER_COMMAND;
 
-    key = [[[o_event charactersIgnoringModifiers] lowercaseString] characterAtIndex: 0];
-
-    if( key )
-    {
-        vout_thread_t * p_vout = getVout();
-        /* Escape should always get you out of fullscreen */
-        if( key == (unichar) 0x1b )
-        {
-            playlist_t * p_playlist = pl_Get( VLCIntf );
-             if( var_GetBool( p_playlist, "fullscreen") )
-                 [[VLCCoreInteraction sharedInstance] toggleFullscreen];
-        }
-        /* handle Lion's default key combo for fullscreen-toggle in addition to our own hotkeys */
-        else if( key == 'f' && i_pressed_modifiers & NSControlKeyMask && i_pressed_modifiers & NSCommandKeyMask )
-            [[VLCCoreInteraction sharedInstance] toggleFullscreen];
-        else if ( p_vout )
-        {
-            if( key == ' ' )
-            {
-                [[VLCCoreInteraction sharedInstance] play];
-            }
-            else
-            {
-                val.i_int |= (int)CocoaKeyToVLC( key );
-                var_Set( p_vout->p_libvlc, "key-pressed", val );
-            }
-            vlc_object_release( p_vout );
-        }
-        else
-            msg_Dbg( VLCIntf, "could not send keyevent to VLC core" );
-    }
-    else
-        [super keyDown: o_event];
-}
+    NSString * characters = [o_event charactersIgnoringModifiers];
+    if ([characters length] > 0) {
+        key = [[characters lowercaseString] characterAtIndex: 0];
 
-- (void)mouseDown:(NSEvent *)o_event
-{
-    vout_thread_t * p_vout = getVout();
-    vlc_value_t val;
-    if( p_vout )
-    {
-        if( ( [o_event type] == NSLeftMouseDown ) &&
-          ( ! ( [o_event modifierFlags] &  NSControlKeyMask ) ) )
-        {
-            if( [o_event clickCount] <= 1 )
-            {
-                /* single clicking */
-                vout_display_SendEventMousePressed( [[[self subviews] objectAtIndex:0] voutDisplay], MOUSE_BUTTON_LEFT );
+        if (key) {
+            /* Escape should always get you out of fullscreen */
+            if (key == (unichar) 0x1b) {
+                playlist_t * p_playlist = pl_Get(VLCIntf);
+                 if (var_GetBool(p_playlist, "fullscreen"))
+                     [[VLCCoreInteraction sharedInstance] toggleFullscreen];
             }
-            else
-            {
-                /* multiple clicking */
+            /* handle Lion's default key combo for fullscreen-toggle in addition to our own hotkeys */
+            else if (key == 'f' && i_pressed_modifiers & NSControlKeyMask && i_pressed_modifiers & NSCommandKeyMask)
                 [[VLCCoreInteraction sharedInstance] toggleFullscreen];
+            else if (p_vout) {
+                if (key == ' ')
+                    [[VLCCoreInteraction sharedInstance] play];
+                else {
+                    val.i_int |= (int)CocoaKeyToVLC(key);
+                    var_Set(p_vout->p_libvlc, "key-pressed", val);
+                }
             }
+            else
+                msg_Dbg(VLCIntf, "could not send keyevent to VLC core");
+
+            return;
         }
-        else if( ( [o_event type] == NSRightMouseDown ) ||
-               ( ( [o_event type] == NSLeftMouseDown ) &&
-                 ( [o_event modifierFlags] &  NSControlKeyMask ) ) )
-        {
-            msg_Dbg( p_vout, "received NSRightMouseDown (generic method) or Ctrl clic" );
-            [NSMenu popUpContextMenu: [[VLCMainMenu sharedInstance] voutMenu] withEvent: o_event forView: self];
-        }
-        vlc_object_release( p_vout );
     }
+    [super keyDown: o_event];
+}
 
-    [super mouseDown: o_event];
+- (BOOL)performKeyEquivalent:(NSEvent *)o_event
+{
+    return [[VLCMainWindow sharedInstance] performKeyEquivalent: o_event];
 }
 
-- (void)otherMouseDown:(NSEvent *)o_event
+- (void)mouseDown:(NSEvent *)o_event
 {
-    if( [o_event type] == NSOtherMouseDown )
-    {
-        vout_thread_t * p_vout = getVout();
-        vlc_value_t val;
-
-        if (p_vout)
-        {
-            vout_display_SendEventMousePressed( [[[self subviews] objectAtIndex:0] voutDisplay], MOUSE_BUTTON_CENTER );
-        }
-        vlc_object_release( p_vout );
-    }
+    if (([o_event type] == NSLeftMouseDown) && (! ([o_event modifierFlags] &  NSControlKeyMask))) {
+        if ([o_event clickCount] > 1)
+            [[VLCCoreInteraction sharedInstance] toggleFullscreen];
+    } else if (([o_event type] == NSRightMouseDown) ||
+               (([o_event type] == NSLeftMouseDown) &&
+               ([o_event modifierFlags] &  NSControlKeyMask)))
+        [NSMenu popUpContextMenu: [[VLCMainMenu sharedInstance] voutMenu] withEvent: o_event forView: self];
 
     [super mouseDown: o_event];
 }
 
 - (void)rightMouseDown:(NSEvent *)o_event
 {
-    if( [o_event type] == NSRightMouseDown )
-    {
-        vout_thread_t * p_vout = getVout();
-        if (p_vout)
-            [NSMenu popUpContextMenu: [[VLCMainMenu sharedInstance] voutMenu] withEvent: o_event forView: self];
-        vlc_object_release( p_vout );
-    }
+    if ([o_event type] == NSRightMouseDown)
+        [NSMenu popUpContextMenu: [[VLCMainMenu sharedInstance] voutMenu] withEvent: o_event forView: self];
 
     [super mouseDown: o_event];
 }
 
-- (void)mouseUp:(NSEvent *)o_event
+- (void)rightMouseUp:(NSEvent *)o_event
 {
-    if( [o_event type] == NSLeftMouseUp )
-    {
-        vout_thread_t * p_vout = getVout();
-        if (p_vout)
-        {
-            vout_display_SendEventMouseReleased( [[[self subviews] objectAtIndex:0] voutDisplay], MOUSE_BUTTON_LEFT );
-            vlc_object_release( p_vout );
-        }
-    }
+    if ([o_event type] == NSRightMouseUp)
+        [NSMenu popUpContextMenu: [[VLCMainMenu sharedInstance] voutMenu] withEvent: o_event forView: self];
 
     [super mouseUp: o_event];
 }
 
-- (void)otherMouseUp:(NSEvent *)o_event
+- (void)mouseMoved:(NSEvent *)o_event
 {
-    if( [o_event type] == NSOtherMouseUp )
-    {
-        vout_thread_t * p_vout = getVout();
-        if (p_vout)
-        {
-            vout_display_SendEventMouseReleased( [[[self subviews] objectAtIndex:0] voutDisplay], MOUSE_BUTTON_CENTER );
-            vlc_object_release( p_vout );
-        }
-    }
+    NSPoint ml = [self convertPoint: [o_event locationInWindow] fromView: nil];
+    if ([self mouse: ml inRect: [self bounds]])
+        [[VLCMain sharedInstance] showFullscreenController];
 
-    [super mouseUp: o_event];
+    [super mouseMoved: o_event];
 }
 
-- (void)rightMouseUp:(NSEvent *)o_event
+- (void)resetScrollWheelDirection
 {
-    if( [o_event type] == NSRightMouseUp )
-    {
-        vout_thread_t * p_vout = getVout();
-        if (p_vout)
-        {
-            [NSMenu popUpContextMenu: [[VLCMainMenu sharedInstance] voutMenu] withEvent: o_event forView: self];
-            vlc_object_release( p_vout );
-        }
-    }
-
-    [super mouseUp: o_event];
+    /* release the scroll direction 0.8 secs after the last event */
+    if (([NSDate timeIntervalSinceReferenceDate] - t_lastScrollEvent) >= 0.80)
+        i_lastScrollWheelDirection = 0;
 }
 
-- (void)mouseDragged:(NSEvent *)o_event
+- (void)scrollWheel:(NSEvent *)theEvent
 {
-    [self mouseMoved: o_event];
+    intf_thread_t * p_intf = VLCIntf;
+    CGFloat f_deltaX = [theEvent deltaX];
+    CGFloat f_deltaY = [theEvent deltaY];
+
+    if (!OSX_SNOW_LEOPARD && [theEvent isDirectionInvertedFromDevice]) {
+        f_deltaX = -f_deltaX;
+        f_deltaY = -f_deltaY;
+    }
+
+    CGFloat f_yabsvalue = f_deltaY > 0.0f ? f_deltaY : -f_deltaY;
+    CGFloat f_xabsvalue = f_deltaX > 0.0f ? f_deltaX : -f_deltaX;
+
+    int i_yvlckey, i_xvlckey = 0;
+    if (f_deltaY < 0.0f)
+        i_yvlckey = KEY_MOUSEWHEELDOWN;
+    else
+        i_yvlckey = KEY_MOUSEWHEELUP;
+
+    if (f_deltaX < 0.0f)
+        i_xvlckey = KEY_MOUSEWHEELRIGHT;
+    else
+        i_xvlckey = KEY_MOUSEWHEELLEFT;
+
+    /* in the following, we're forwarding either a x or a y event */
+    /* Multiple key events are send depending on the intensity of the event */
+    /* the opposite direction is being blocked for 0.8 secs */
+    if (f_yabsvalue > 0.05) {
+        if (i_lastScrollWheelDirection < 0) // last was a X
+            return;
+
+        i_lastScrollWheelDirection = 1; // Y
+        for (NSUInteger i = 0; i < (int)(f_yabsvalue/4.+1.); i++)
+            var_SetInteger(p_intf->p_libvlc, "key-pressed", i_yvlckey);
+
+        t_lastScrollEvent = [NSDate timeIntervalSinceReferenceDate];
+        [self performSelector:@selector(resetScrollWheelDirection)
+                   withObject: NULL
+                   afterDelay:1.00];
+        return;
+    }
+    if (f_xabsvalue > 0.05) {
+        if (i_lastScrollWheelDirection > 0) // last was a Y
+            return;
+
+        i_lastScrollWheelDirection = -1; // X
+        for (NSUInteger i = 0; i < (int)(f_xabsvalue/6.+1.); i++)
+            var_SetInteger(p_intf->p_libvlc, "key-pressed", i_xvlckey);
+
+        t_lastScrollEvent = [NSDate timeIntervalSinceReferenceDate];
+        [self performSelector:@selector(resetScrollWheelDirection)
+                   withObject: NULL
+                   afterDelay:1.00];
+    }
 }
 
-- (void)otherMouseDragged:(NSEvent *)o_event
+#pragma mark -
+#pragma mark Handling of vout related actions
+
+- (void)setVoutThread:(vout_thread_t *)p_vout_thread
 {
-    [self mouseMoved: o_event];
+    assert(p_vout == NULL);
+    p_vout = p_vout_thread;
+    vlc_object_hold(p_vout);
 }
 
-- (void)rightMouseDragged:(NSEvent *)o_event
+- (vout_thread_t *)voutThread
 {
-    [self mouseMoved: o_event];
+    if (p_vout) {
+        vlc_object_hold(p_vout);
+        return p_vout;
+    }
+
+    return NULL;
 }
 
-- (void)mouseMoved:(NSEvent *)o_event
+- (void)releaseVoutThread
 {
-    vout_thread_t * p_vout = getVout();
-    if (p_vout)
-    {
-        NSPoint ml;
-        NSRect s_rect;
-        BOOL b_inside;
-
-        s_rect = [self bounds];
-        ml = [self convertPoint: [o_event locationInWindow] fromView: nil];
-        b_inside = [self mouse: ml inRect: s_rect];
-
-        if( b_inside )
-        {
-            vout_display_SendEventMouseMoved( [[[self subviews] objectAtIndex:0] voutDisplay], ((int)ml.x), ((int)s_rect.size.height - ((int)ml.y)) );
-            [[VLCMain sharedInstance] showFullscreenController];
-        }
-        vlc_object_release( p_vout );
+    if (p_vout) {
+        vlc_object_release(p_vout);
+        p_vout = NULL;
     }
-
-    [super mouseMoved: o_event];
 }
 
+#pragma mark -
+#pragma mark Basic view behaviour and touch events handling
+
 - (BOOL)mouseDownCanMoveWindow
 {
     return YES;
@@ -310,4 +306,30 @@ int DeviceCallback( vlc_object_t *p_this, const char *psz_variable,
     /* while we need to be the first responder most of the time, we need to give up that status when toggling the playlist */
     return YES;
 }
+
+-(void)didAddSubview:(NSView *)subview
+{
+    [[self window] makeFirstResponder: subview];
+}
+
+- (void)magnifyWithEvent:(NSEvent *)event
+{
+    f_cumulated_magnification += [event magnification];
+
+    // This is the result of [NSEvent standardMagnificationThreshold].
+    // Unfortunately, this is a private API, currently.
+    CGFloat f_threshold = 0.3;
+    BOOL b_fullscreen = [[VLCMainWindow sharedInstance] fullscreen];
+
+    if ((f_cumulated_magnification > f_threshold && !b_fullscreen) || (f_cumulated_magnification < -f_threshold && b_fullscreen)) {
+        f_cumulated_magnification = 0.0;
+        [[VLCCoreInteraction sharedInstance] toggleFullscreen];
+    }
+}
+
+- (void)beginGestureWithEvent:(NSEvent *)event
+{
+    f_cumulated_magnification = 0.0;
+}
+
 @end