Blame gst-libs/gst/gl/cocoa/gstglwindow_cocoa.m

Packit 971217
/*
Packit 971217
 * GStreamer
Packit 971217
 * Copyright (C) 2008 Julien Isorce <julien.isorce@gmail.com>
Packit 971217
 * Copyright (C) 2014 Sebastian Dröge <sebastian@centricular.com>
Packit 971217
 *
Packit 971217
 * This library is free software; you can redistribute it and/or
Packit 971217
 * modify it un der the terms of the GNU Library General Public
Packit 971217
 * License as published by the Free Software Foundation; either
Packit 971217
 * version 2 of the License, or (at your option) any later version.
Packit 971217
 *
Packit 971217
 * This library is distributed in the hope that it will be useful,
Packit 971217
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 971217
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 971217
 * Library General Public License for more details.
Packit 971217
 *
Packit 971217
 * You should have received a copy of the GNU Library General Public
Packit 971217
 * License along with this library; if not, write to the
Packit 971217
 * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
Packit 971217
 * Boston, MA 02110-1301, USA.
Packit 971217
 */
Packit 971217
Packit 971217
#ifdef HAVE_CONFIG_H
Packit 971217
#include "config.h"
Packit 971217
#endif
Packit 971217
Packit 971217
#include <Cocoa/Cocoa.h>
Packit 971217
#include <QuartzCore/QuartzCore.h>
Packit 971217
Packit 971217
#include "gstgl_cocoa_private.h"
Packit 971217
Packit 971217
#if MAC_OS_X_VERSION_MAX_ALLOWED < 101200
Packit 971217
#define NSWindowStyleMaskTitled              NSTitledWindowMask
Packit 971217
#define NSWindowStyleMaskClosable            NSClosableWindowMask
Packit 971217
#define NSWindowStyleMaskResizable           NSResizableWindowMask
Packit 971217
#define NSWindowStyleMaskMiniaturizable      NSMiniaturizableWindowMask
Packit 971217
#endif
Packit 971217
Packit 971217
/* =============================================================*/
Packit 971217
/*                                                              */
Packit 971217
/*               GstGLNSWindow declaration                      */
Packit 971217
/*                                                              */
Packit 971217
/* =============================================================*/
Packit 971217
Packit 971217
@interface GstGLNSWindow: NSWindow {
Packit 971217
  BOOL m_isClosed;
Packit 971217
  GstGLWindowCocoa *window_cocoa;
Packit 971217
}
Packit 971217
- (id)initWithContentRect:(NSRect)contentRect
Packit 971217
    styleMask: (unsigned int) styleMask
Packit 971217
    backing: (NSBackingStoreType) bufferingType
Packit 971217
    defer: (BOOL) flag screen: (NSScreen *) aScreen
Packit 971217
    gstWin: (GstGLWindowCocoa *) window;
Packit 971217
- (void) setClosed;
Packit 971217
- (BOOL) isClosed;
Packit 971217
- (BOOL) canBecomeMainWindow;
Packit 971217
- (BOOL) canBecomeKeyWindow;
Packit 971217
@end
Packit 971217
Packit 971217
/* =============================================================*/
Packit 971217
/*                                                              */
Packit 971217
/*                      GstGLWindow                             */
Packit 971217
/*                                                              */
Packit 971217
/* =============================================================*/
Packit 971217
Packit 971217
#define GST_GL_WINDOW_COCOA_GET_PRIVATE(o)  \
Packit 971217
  (G_TYPE_INSTANCE_GET_PRIVATE((o), GST_TYPE_GL_WINDOW_COCOA, GstGLWindowCocoaPrivate))
Packit 971217
Packit 971217
#define GST_CAT_DEFAULT gst_gl_window_cocoa_debug
Packit 971217
GST_DEBUG_CATEGORY_STATIC (GST_CAT_DEFAULT);
Packit 971217
Packit 971217
#define DEBUG_INIT \
Packit 971217
  GST_DEBUG_CATEGORY_GET (GST_CAT_DEFAULT, "glwindow");
Packit 971217
#define gst_gl_window_cocoa_parent_class parent_class
Packit 971217
G_DEFINE_TYPE_WITH_CODE (GstGLWindowCocoa, gst_gl_window_cocoa, GST_TYPE_GL_WINDOW, DEBUG_INIT);
Packit 971217
static void gst_gl_window_cocoa_finalize (GObject * object);
Packit 971217
Packit 971217
static gboolean gst_gl_window_cocoa_open (GstGLWindow *window, GError **err);
Packit 971217
static void gst_gl_window_cocoa_close (GstGLWindow *window);
Packit 971217
static guintptr gst_gl_window_cocoa_get_window_handle (GstGLWindow * window);
Packit 971217
static void gst_gl_window_cocoa_set_window_handle (GstGLWindow * window,
Packit 971217
    guintptr handle);
Packit 971217
static void gst_gl_window_cocoa_draw (GstGLWindow * window);
Packit 971217
static void gst_gl_window_cocoa_set_preferred_size (GstGLWindow * window,
Packit 971217
    gint width, gint height);
Packit 971217
static void gst_gl_window_cocoa_show (GstGLWindow * window);
Packit 971217
static void gst_gl_window_cocoa_queue_resize (GstGLWindow * window);
Packit 971217
static void gst_gl_window_cocoa_send_message_async (GstGLWindow * window,
Packit 971217
    GstGLWindowCB callback, gpointer data, GDestroyNotify destroy);
Packit 971217
static gboolean gst_gl_window_cocoa_set_render_rectangle (GstGLWindow * window,
Packit 971217
    gint x, gint y, gint width, gint height);
Packit 971217
Packit 971217
struct _GstGLWindowCocoaPrivate
Packit 971217
{
Packit 971217
  gpointer internal_win_id;
Packit 971217
  gpointer external_view;
Packit 971217
  gboolean visible;
Packit 971217
  gint preferred_width;
Packit 971217
  gint preferred_height;
Packit 971217
Packit 971217
  /* atomic set when the internal NSView has been created */
Packit 971217
  int view_ready;
Packit 971217
Packit 971217
  gpointer gl_queue;
Packit 971217
};
Packit 971217
Packit 971217
static void
Packit 971217
gst_gl_window_cocoa_class_init (GstGLWindowCocoaClass * klass)
Packit 971217
{
Packit 971217
  GstGLWindowClass *window_class = (GstGLWindowClass *) klass;
Packit 971217
  GObjectClass *gobject_class = (GObjectClass *) klass;
Packit 971217
Packit 971217
  g_type_class_add_private (klass, sizeof (GstGLWindowCocoaPrivate));
Packit 971217
Packit 971217
  window_class->open = GST_DEBUG_FUNCPTR (gst_gl_window_cocoa_open);
Packit 971217
  window_class->close = GST_DEBUG_FUNCPTR (gst_gl_window_cocoa_close);
Packit 971217
  window_class->get_window_handle =
Packit 971217
      GST_DEBUG_FUNCPTR (gst_gl_window_cocoa_get_window_handle);
Packit 971217
  window_class->set_window_handle =
Packit 971217
      GST_DEBUG_FUNCPTR (gst_gl_window_cocoa_set_window_handle);
Packit 971217
  window_class->draw = GST_DEBUG_FUNCPTR (gst_gl_window_cocoa_draw);
Packit 971217
  window_class->set_preferred_size =
Packit 971217
      GST_DEBUG_FUNCPTR (gst_gl_window_cocoa_set_preferred_size);
Packit 971217
  window_class->show = GST_DEBUG_FUNCPTR (gst_gl_window_cocoa_show);
Packit 971217
  window_class->queue_resize = GST_DEBUG_FUNCPTR (gst_gl_window_cocoa_queue_resize);
Packit 971217
  window_class->send_message_async =
Packit 971217
      GST_DEBUG_FUNCPTR (gst_gl_window_cocoa_send_message_async);
Packit 971217
  window_class->set_render_rectangle =
Packit 971217
      GST_DEBUG_FUNCPTR (gst_gl_window_cocoa_set_render_rectangle);
Packit 971217
Packit 971217
  gobject_class->finalize = gst_gl_window_cocoa_finalize;
Packit 971217
}
Packit 971217
Packit 971217
static void
Packit 971217
gst_gl_window_cocoa_init (GstGLWindowCocoa * window)
Packit 971217
{
Packit 971217
  window->priv = GST_GL_WINDOW_COCOA_GET_PRIVATE (window);
Packit 971217
Packit 971217
  window->priv->preferred_width = 320;
Packit 971217
  window->priv->preferred_height = 240;
Packit 971217
#if OS_OBJECT_USE_OBJC
Packit 971217
  window->priv->gl_queue = (__bridge_retained gpointer)
Packit 971217
      (dispatch_queue_create ("org.freedesktop.gstreamer.glwindow", NULL));
Packit 971217
#else
Packit 971217
  window->priv->gl_queue = (gpointer)
Packit 971217
      (dispatch_queue_create ("org.freedesktop.gstreamer.glwindow", NULL));
Packit 971217
#endif
Packit 971217
}
Packit 971217
Packit 971217
static void
Packit 971217
gst_gl_window_cocoa_finalize (GObject * object)
Packit 971217
{
Packit 971217
  GstGLWindowCocoa *window = GST_GL_WINDOW_COCOA (object);
Packit 971217
Packit 971217
#if OS_OBJECT_USE_OBJC
Packit 971217
  /* Let ARC clean up our queue */
Packit 971217
  dispatch_queue_t queue = (__bridge_transfer dispatch_queue_t) window->priv->gl_queue;
Packit 971217
#else
Packit 971217
  dispatch_release (window->priv->gl_queue);
Packit 971217
#endif
Packit 971217
Packit 971217
  window->priv->gl_queue = NULL;
Packit 971217
  G_OBJECT_CLASS (parent_class)->finalize (object);
Packit 971217
}
Packit 971217
Packit 971217
GstGLWindowCocoa *
Packit 971217
gst_gl_window_cocoa_new (GstGLDisplay * display)
Packit 971217
{
Packit 971217
  GstGLWindowCocoa *window;
Packit 971217
Packit 971217
  if ((gst_gl_display_get_handle_type (display) & GST_GL_DISPLAY_TYPE_COCOA) == 0)
Packit 971217
    /* we require an cocoa display to create CGL windows */
Packit 971217
    return NULL;
Packit 971217
Packit 971217
  window = g_object_new (GST_TYPE_GL_WINDOW_COCOA, NULL);
Packit 971217
  gst_object_ref_sink (window);
Packit 971217
Packit 971217
  return window;
Packit 971217
}
Packit 971217
Packit 971217
/* Must be called from the main thread */
Packit 971217
gboolean
Packit 971217
gst_gl_window_cocoa_create_window (GstGLWindowCocoa *window_cocoa)
Packit 971217
{
Packit 971217
  GstGLWindowCocoaPrivate *priv = window_cocoa->priv;
Packit 971217
  GstGLWindow *window = GST_GL_WINDOW (window_cocoa);
Packit 971217
  GstGLNSWindow *internal_win_id;
Packit 971217
  NSRect mainRect = [[NSScreen mainScreen] visibleFrame];
Packit 971217
  gint h = priv->preferred_height;
Packit 971217
  gint y = mainRect.size.height > h ? (mainRect.size.height - h) * 0.5 : 0;
Packit 971217
  NSRect rect = NSMakeRect (0, y, priv->preferred_width, priv->preferred_height);
Packit 971217
  NSRect windowRect = NSMakeRect (0, y, priv->preferred_width, priv->preferred_height);
Packit 971217
  GstGLContext *context = gst_gl_window_get_context (window);
Packit 971217
  GstGLContextCocoa *context_cocoa;
Packit 971217
  GstGLCAOpenGLLayer *layer;
Packit 971217
  GstGLNSView *glView;
Packit 971217
Packit 971217
  if (!context)
Packit 971217
    return FALSE;
Packit 971217
Packit 971217
  context_cocoa = GST_GL_CONTEXT_COCOA (context);
Packit 971217
  layer = [[GstGLCAOpenGLLayer alloc] initWithGstGLContext:context];
Packit 971217
  layer.autoresizingMask = kCALayerWidthSizable | kCALayerHeightSizable;
Packit 971217
  layer.needsDisplayOnBoundsChange = YES;
Packit 971217
  glView = [[GstGLNSView alloc] initWithFrameLayer:window_cocoa rect:windowRect layer:layer];
Packit 971217
Packit 971217
  gst_object_unref (context);
Packit 971217
Packit 971217
  internal_win_id = [[GstGLNSWindow alloc] initWithContentRect:rect styleMask:
Packit 971217
      (NSWindowStyleMaskTitled | NSWindowStyleMaskClosable |
Packit 971217
      NSWindowStyleMaskResizable | NSWindowStyleMaskMiniaturizable)
Packit 971217
      backing: NSBackingStoreBuffered defer: NO screen: nil gstWin: window_cocoa];
Packit 971217
Packit 971217
  priv->internal_win_id = (__bridge_retained gpointer)internal_win_id;
Packit 971217
Packit 971217
  GST_DEBUG ("NSWindow id: %"G_GUINTPTR_FORMAT, (guintptr) priv->internal_win_id);
Packit 971217
Packit 971217
  [internal_win_id setContentView:glView];
Packit 971217
Packit 971217
  g_atomic_int_set (&window_cocoa->priv->view_ready, 1);
Packit 971217
Packit 971217
  /* Set the window handle for real now that the NSWindow has been created. */
Packit 971217
  if (priv->external_view)
Packit 971217
    gst_gl_window_cocoa_set_window_handle (window,
Packit 971217
        (guintptr) priv->external_view);
Packit 971217
Packit 971217
  return TRUE;
Packit 971217
}
Packit 971217
Packit 971217
static gboolean
Packit 971217
gst_gl_window_cocoa_open (GstGLWindow *window, GError **err)
Packit 971217
{
Packit 971217
  GstGLWindowCocoa *window_cocoa;
Packit 971217
Packit 971217
  window_cocoa = GST_GL_WINDOW_COCOA (window);
Packit 971217
Packit 971217
  return TRUE;
Packit 971217
}
Packit 971217
Packit 971217
static void
Packit 971217
gst_gl_window_cocoa_close (GstGLWindow *window)
Packit 971217
{
Packit 971217
  GstGLWindowCocoa *window_cocoa = GST_GL_WINDOW_COCOA (window);
Packit 971217
  GstGLNSWindow *internal_win_id = (__bridge GstGLNSWindow *)window_cocoa->priv->internal_win_id;
Packit 971217
Packit 971217
  [[internal_win_id contentView] removeFromSuperview];
Packit 971217
  CFBridgingRelease(window_cocoa->priv->internal_win_id);
Packit 971217
  window_cocoa->priv->internal_win_id = NULL;
Packit 971217
}
Packit 971217
Packit 971217
static guintptr
Packit 971217
gst_gl_window_cocoa_get_window_handle (GstGLWindow *window)
Packit 971217
{
Packit 971217
  return (guintptr) GST_GL_WINDOW_COCOA (window)->priv->internal_win_id;
Packit 971217
}
Packit 971217
Packit 971217
static void
Packit 971217
gst_gl_window_cocoa_set_window_handle (GstGLWindow * window, guintptr handle)
Packit 971217
{
Packit 971217
  GstGLWindowCocoa *window_cocoa;
Packit 971217
  GstGLWindowCocoaPrivate *priv;
Packit 971217
Packit 971217
  window_cocoa = GST_GL_WINDOW_COCOA (window);
Packit 971217
  priv = window_cocoa->priv;
Packit 971217
Packit 971217
  if (priv->internal_win_id) {
Packit 971217
    if (handle) {
Packit 971217
      priv->external_view = (gpointer)handle;
Packit 971217
      priv->visible = TRUE;
Packit 971217
    } else {
Packit 971217
      /* bring back our internal window */
Packit 971217
      priv->external_view = 0;
Packit 971217
      priv->visible = FALSE;
Packit 971217
    }
Packit 971217
Packit 971217
Packit 971217
    dispatch_async (dispatch_get_main_queue (), ^{
Packit 971217
      GstGLNSWindow *internal_win_id =
Packit 971217
          (__bridge GstGLNSWindow *)window_cocoa->priv->internal_win_id;
Packit 971217
      NSView *external_view =
Packit 971217
          (__bridge NSView *)window_cocoa->priv->external_view;
Packit 971217
Packit 971217
      NSView *view = [internal_win_id contentView];
Packit 971217
      [internal_win_id orderOut:internal_win_id];
Packit 971217
Packit 971217
      [external_view addSubview: view];
Packit 971217
Packit 971217
      [external_view setAutoresizesSubviews: YES];
Packit 971217
      [view setFrame: [external_view bounds]];
Packit 971217
      [view setAutoresizingMask: NSViewWidthSizable|NSViewHeightSizable];
Packit 971217
    });
Packit 971217
  } else {
Packit 971217
    /* no internal window yet so delay it to the next drawing */
Packit 971217
    priv->external_view = (gpointer)handle;
Packit 971217
    priv->visible = FALSE;
Packit 971217
  }
Packit 971217
}
Packit 971217
Packit 971217
static void
Packit 971217
_show_window (gpointer data)
Packit 971217
{
Packit 971217
  GstGLWindowCocoa *window_cocoa = GST_GL_WINDOW_COCOA (data);
Packit 971217
  GstGLWindowCocoaPrivate *priv = window_cocoa->priv;
Packit 971217
  GstGLNSWindow *internal_win_id = (__bridge GstGLNSWindow *)priv->internal_win_id;
Packit 971217
Packit 971217
  GST_DEBUG_OBJECT (window_cocoa, "make the window available\n");
Packit 971217
  [internal_win_id makeMainWindow];
Packit 971217
  [internal_win_id orderFrontRegardless];
Packit 971217
  [internal_win_id setViewsNeedDisplay:YES];
Packit 971217
Packit 971217
  priv->visible = TRUE;
Packit 971217
}
Packit 971217
Packit 971217
static void
Packit 971217
gst_gl_window_cocoa_show (GstGLWindow * window)
Packit 971217
{
Packit 971217
  GstGLWindowCocoa *window_cocoa = GST_GL_WINDOW_COCOA (window);
Packit 971217
  GstGLWindowCocoaPrivate *priv = window_cocoa->priv;
Packit 971217
Packit 971217
  if (!priv->visible) {
Packit 971217
    /* useful when set_window_handle is called before
Packit 971217
     * the internal NSWindow */
Packit 971217
    if (priv->external_view) {
Packit 971217
      gst_gl_window_cocoa_set_window_handle (window, (guintptr) priv->external_view);
Packit 971217
      priv->visible = TRUE;
Packit 971217
      return;
Packit 971217
    }
Packit 971217
Packit 971217
    if (!priv->external_view && !priv->visible)
Packit 971217
      _invoke_on_main ((GstGLWindowCB) _show_window, gst_object_ref (window),
Packit 971217
          (GDestroyNotify) gst_object_unref);
Packit 971217
  }
Packit 971217
}
Packit 971217
Packit 971217
static void
Packit 971217
gst_gl_window_cocoa_queue_resize (GstGLWindow * window)
Packit 971217
{
Packit 971217
  GstGLWindowCocoa *window_cocoa = GST_GL_WINDOW_COCOA (window);
Packit 971217
  GstGLNSView *view;
Packit 971217
  GstGLWindowCocoaPrivate *priv = window_cocoa->priv;
Packit 971217
  GstGLNSWindow *internal_win_id = (__bridge GstGLNSWindow *)priv->internal_win_id;
Packit 971217
Packit 971217
  if (!g_atomic_int_get (&window_cocoa->priv->view_ready))
Packit 971217
    return;
Packit 971217
Packit 971217
  view = (GstGLNSView *)[internal_win_id contentView];
Packit 971217
Packit 971217
  [view->layer queueResize];
Packit 971217
}
Packit 971217
Packit 971217
static void
Packit 971217
gst_gl_window_cocoa_draw (GstGLWindow * window)
Packit 971217
{
Packit 971217
  GstGLWindowCocoa *window_cocoa = GST_GL_WINDOW_COCOA (window);
Packit 971217
  GstGLNSView *view;
Packit 971217
  GstGLWindowCocoaPrivate *priv = window_cocoa->priv;
Packit 971217
  GstGLNSWindow *internal_win_id = (__bridge GstGLNSWindow *)priv->internal_win_id;
Packit 971217
Packit 971217
  /* As the view is created asynchronously in the main thread we cannot know
Packit 971217
   * exactly when it will be ready to draw to */
Packit 971217
  if (!g_atomic_int_get (&window_cocoa->priv->view_ready))
Packit 971217
    return;
Packit 971217
Packit 971217
  view = (GstGLNSView *)[internal_win_id contentView];
Packit 971217
Packit 971217
  /* this redraws the GstGLCAOpenGLLayer which calls
Packit 971217
   * gst_gl_window_cocoa_draw_thread(). Use an explicit CATransaction since we
Packit 971217
   * don't know how often the main runloop is running.
Packit 971217
   */
Packit 971217
  [CATransaction begin];
Packit 971217
  [view setNeedsDisplay:YES];
Packit 971217
  [CATransaction commit];
Packit 971217
}
Packit 971217
Packit 971217
static void
Packit 971217
gst_gl_window_cocoa_set_preferred_size (GstGLWindow * window, gint width,
Packit 971217
    gint height)
Packit 971217
{
Packit 971217
  GstGLWindowCocoa *window_cocoa = GST_GL_WINDOW_COCOA (window);
Packit 971217
Packit 971217
  window_cocoa->priv->preferred_width = width;
Packit 971217
  window_cocoa->priv->preferred_height = height;
Packit 971217
}
Packit 971217
Packit 971217
static void
Packit 971217
gst_gl_cocoa_draw_cb (GstGLWindowCocoa *window_cocoa)
Packit 971217
{
Packit 971217
  GstGLWindowCocoaPrivate *priv = window_cocoa->priv;
Packit 971217
  GstGLNSWindow *internal_win_id = (__bridge GstGLNSWindow *)priv->internal_win_id;
Packit 971217
Packit 971217
  if (internal_win_id && ![internal_win_id isClosed]) {
Packit 971217
   GstGLWindow *window = GST_GL_WINDOW (window_cocoa);
Packit 971217
Packit 971217
    /* draw opengl scene in the back buffer */
Packit 971217
    if (window->draw)
Packit 971217
      window->draw (window->draw_data);
Packit 971217
  }
Packit 971217
}
Packit 971217
Packit 971217
static void
Packit 971217
gst_gl_cocoa_resize_cb (GstGLNSView * view, guint width, guint height)
Packit 971217
{
Packit 971217
  GstGLWindowCocoa *window_cocoa = view->window_cocoa;
Packit 971217
  GstGLWindow *window = GST_GL_WINDOW (window_cocoa);
Packit 971217
  GstGLContext *context = gst_gl_window_get_context (window);
Packit 971217
  GstGLWindowCocoaPrivate *priv = window_cocoa->priv;
Packit 971217
  GstGLNSWindow *internal_win_id = (__bridge GstGLNSWindow *)priv->internal_win_id;
Packit 971217
Packit 971217
  if (internal_win_id && ![internal_win_id isClosed]) {
Packit 971217
    const GstGLFuncs *gl;
Packit 971217
    NSRect bounds = [view bounds];
Packit 971217
    NSRect visibleRect = [view visibleRect];
Packit 971217
    gint viewport_dim[4];
Packit 971217
Packit 971217
    gl = context->gl_vtable;
Packit 971217
Packit 971217
#if MAC_OS_X_VERSION_MAX_ALLOWED >= 1070
Packit 971217
    bounds = [view convertRectToBacking:bounds];
Packit 971217
    visibleRect = [view convertRectToBacking:visibleRect];
Packit 971217
#endif
Packit 971217
Packit 971217
    GST_DEBUG_OBJECT (window, "Window resized: bounds %lf %lf %lf %lf "
Packit 971217
                      "visibleRect %lf %lf %lf %lf",
Packit 971217
                      bounds.origin.x, bounds.origin.y,
Packit 971217
                      bounds.size.width, bounds.size.height,
Packit 971217
                      visibleRect.origin.x, visibleRect.origin.y,
Packit 971217
                      visibleRect.size.width, visibleRect.size.height);
Packit 971217
Packit 971217
    gst_gl_window_resize (window, width, height);
Packit 971217
    gl->GetIntegerv (GL_VIEWPORT, viewport_dim);
Packit 971217
Packit 971217
    gl->Viewport (viewport_dim[0] - visibleRect.origin.x,
Packit 971217
                  viewport_dim[1] - visibleRect.origin.y,
Packit 971217
                  viewport_dim[2], viewport_dim[3]);
Packit 971217
  }
Packit 971217
Packit 971217
  gst_object_unref (context);
Packit 971217
}
Packit 971217
Packit 971217
static void
Packit 971217
gst_gl_window_cocoa_send_message_async (GstGLWindow * window,
Packit 971217
    GstGLWindowCB callback, gpointer data, GDestroyNotify destroy)
Packit 971217
{
Packit 971217
  GstGLWindowCocoa *window_cocoa = (GstGLWindowCocoa *) window;
Packit 971217
  GstGLContext *context = gst_gl_window_get_context (window);
Packit 971217
  GThread *thread = gst_gl_context_get_thread (context);
Packit 971217
  GstGLWindowCocoaPrivate *priv = window_cocoa->priv;
Packit 971217
#if OS_OBJECT_USE_OBJC
Packit 971217
  dispatch_queue_t gl_queue = (__bridge dispatch_queue_t)priv->gl_queue;
Packit 971217
#else
Packit 971217
  dispatch_queue_t gl_queue = (dispatch_queue_t)priv->gl_queue;
Packit 971217
#endif
Packit 971217
Packit 971217
  if (thread == g_thread_self()) {
Packit 971217
    /* this case happens for nested calls happening from inside the GCD queue */
Packit 971217
    callback (data);
Packit 971217
    if (destroy)
Packit 971217
      destroy (data);
Packit 971217
    gst_object_unref (context);
Packit 971217
  } else {
Packit 971217
    dispatch_async (gl_queue, ^{
Packit 971217
      gst_gl_context_activate (context, TRUE);
Packit 971217
      gst_object_unref (context);
Packit 971217
      callback (data);
Packit 971217
      if (destroy)
Packit 971217
        destroy (data);
Packit 971217
    });
Packit 971217
  }
Packit 971217
  if (thread)
Packit 971217
    g_thread_unref (thread);
Packit 971217
}
Packit 971217
Packit 971217
struct SetRenderRectangle
Packit 971217
{
Packit 971217
 GstGLWindowCocoa *window_cocoa;
Packit 971217
 GstVideoRectangle rect;
Packit 971217
};
Packit 971217
Packit 971217
static void
Packit 971217
_free_set_render_rectangle (struct SetRenderRectangle *render)
Packit 971217
{
Packit 971217
  if (render) {
Packit 971217
    if (render->window_cocoa) {
Packit 971217
      gst_object_unref (render->window_cocoa);
Packit 971217
    }
Packit 971217
    g_free (render);
Packit 971217
  }
Packit 971217
}
Packit 971217
Packit 971217
static void
Packit 971217
_set_render_rectangle (gpointer data)
Packit 971217
{
Packit 971217
 struct SetRenderRectangle *render = data;
Packit 971217
 NSView *view;
Packit 971217
 GstGLWindowCocoaPrivate *priv = render->window_cocoa->priv;
Packit 971217
 GstGLNSWindow *internal_win_id = (__bridge GstGLNSWindow *)priv->internal_win_id;
Packit 971217
Packit 971217
 GST_LOG_OBJECT (render->window_cocoa, "setting render rectangle %i,%i+%ix%i",
Packit 971217
                 render->rect.x, render->rect.y, render->rect.w, render->rect.h);
Packit 971217
 if (!g_atomic_int_get (&render->window_cocoa->priv->view_ready)) {
Packit 971217
   return;
Packit 971217
 }
Packit 971217
Packit 971217
 view = [internal_win_id contentView];
Packit 971217
 NSRect newMainViewFrame = NSMakeRect(render->rect.x,
Packit 971217
                                      render->rect.y,
Packit 971217
                                      render->rect.w,
Packit 971217
                                      render->rect.h);
Packit 971217
Packit 971217
 [view.superview setFrame:newMainViewFrame];
Packit 971217
 [view setFrame: view.superview.bounds];
Packit 971217
Packit 971217
 [CATransaction begin];
Packit 971217
 [view setNeedsDisplay:YES];
Packit 971217
 [CATransaction commit];
Packit 971217
}
Packit 971217
Packit 971217
static gboolean
Packit 971217
gst_gl_window_cocoa_set_render_rectangle (GstGLWindow * window, gint x, gint y, gint width, gint height)
Packit 971217
{
Packit 971217
 GstGLWindowCocoa *window_cocoa = (GstGLWindowCocoa *) window;
Packit 971217
 struct SetRenderRectangle *render;
Packit 971217
Packit 971217
 render = g_new0 (struct SetRenderRectangle, 1);
Packit 971217
 render->window_cocoa = gst_object_ref (window_cocoa);
Packit 971217
 render->rect.x = x;
Packit 971217
 render->rect.y = y;
Packit 971217
 render->rect.w = width;
Packit 971217
 render->rect.h = height;
Packit 971217
Packit 971217
 _invoke_on_main ((GstGLWindowCB) _set_render_rectangle, render,
Packit 971217
                  (GDestroyNotify) _free_set_render_rectangle);
Packit 971217
Packit 971217
 return TRUE;
Packit 971217
}
Packit 971217
Packit 971217
/* =============================================================*/
Packit 971217
/*                                                              */
Packit 971217
/*                    GstGLNSWindow implementation              */
Packit 971217
/*                                                              */
Packit 971217
/* =============================================================*/
Packit 971217
Packit 971217
/* Must be called from the main thread */
Packit 971217
@implementation GstGLNSWindow
Packit 971217
Packit 971217
- (id) initWithContentRect: (NSRect) contentRect
Packit 971217
        styleMask: (unsigned int) styleMask
Packit 971217
    backing: (NSBackingStoreType) bufferingType
Packit 971217
    defer: (BOOL) flag screen: (NSScreen *) aScreen
Packit 971217
    gstWin: (GstGLWindowCocoa *) cocoa {
Packit 971217
Packit 971217
  m_isClosed = NO;
Packit 971217
  window_cocoa = cocoa;
Packit 971217
  GstGLWindowCocoaPrivate *priv = window_cocoa->priv;
Packit 971217
  GstGLNSWindow *internal_win_id = (__bridge GstGLNSWindow *)priv->internal_win_id;
Packit 971217
Packit 971217
  self = [super initWithContentRect: contentRect
Packit 971217
        styleMask: styleMask backing: bufferingType
Packit 971217
        defer: flag screen:aScreen];
Packit 971217
Packit 971217
  [self setReleasedWhenClosed:NO];
Packit 971217
Packit 971217
  GST_DEBUG ("initializing GstGLNSWindow\n");
Packit 971217
Packit 971217
  [self setTitle:@"OpenGL renderer"];
Packit 971217
Packit 971217
  [self setBackgroundColor:[NSColor blackColor]];
Packit 971217
Packit 971217
  [self orderOut:internal_win_id];
Packit 971217
Packit 971217
  return self;
Packit 971217
}
Packit 971217
Packit 971217
- (void) setClosed {
Packit 971217
  m_isClosed = YES;
Packit 971217
}
Packit 971217
Packit 971217
- (BOOL) isClosed {
Packit 971217
  return m_isClosed;
Packit 971217
}
Packit 971217
Packit 971217
- (BOOL) canBecomeMainWindow {
Packit 971217
  return YES;
Packit 971217
}
Packit 971217
Packit 971217
- (BOOL) canBecomeKeyWindow {
Packit 971217
  return YES;
Packit 971217
}
Packit 971217
Packit 971217
static void
Packit 971217
close_window_cb (gpointer data)
Packit 971217
{
Packit 971217
  GstGLWindowCocoa *window_cocoa = data;
Packit 971217
  GstGLWindow *window;
Packit 971217
Packit 971217
  window = GST_GL_WINDOW (window_cocoa);
Packit 971217
Packit 971217
  if (window->close) {
Packit 971217
    window->close (window->close_data);
Packit 971217
  }
Packit 971217
}
Packit 971217
Packit 971217
/* Called in the main thread which is never the gl thread */
Packit 971217
- (BOOL) windowShouldClose:(id)sender {
Packit 971217
Packit 971217
  GstGLWindowCocoaPrivate *priv = window_cocoa->priv;
Packit 971217
  GstGLNSWindow *internal_win_id = (__bridge GstGLNSWindow *)priv->internal_win_id;
Packit 971217
  GST_DEBUG ("user clicked the close button\n");
Packit 971217
  [internal_win_id setClosed];
Packit 971217
  gst_gl_window_send_message_async (GST_GL_WINDOW (window_cocoa),
Packit 971217
      (GstGLWindowCB) close_window_cb, gst_object_ref (window_cocoa),
Packit 971217
      (GDestroyNotify) gst_object_unref);
Packit 971217
  return YES;
Packit 971217
}
Packit 971217
Packit 971217
@end
Packit 971217
Packit 971217
/* =============================================================*/
Packit 971217
/*                                                              */
Packit 971217
/*                GstGLNSView implementation              */
Packit 971217
/*                                                              */
Packit 971217
/* =============================================================*/
Packit 971217
Packit 971217
@implementation GstGLNSView
Packit 971217
Packit 971217
/* Must be called from the application main thread */
Packit 971217
- (id)initWithFrameLayer:(GstGLWindowCocoa *)window rect:(NSRect)contentRect layer:(CALayer *)layerContent {
Packit 971217
Packit 971217
  self = [super initWithFrame: contentRect];
Packit 971217
Packit 971217
  window_cocoa = window;
Packit 971217
Packit 971217
  /* The order of the next two calls matters.  This creates a layer-hosted
Packit 971217
   * NSView.  Calling setWantsLayer before setLayer will create a
Packit 971217
   * layer-backed NSView.  See the apple developer documentation on the
Packit 971217
   * difference.
Packit 971217
   */
Packit 971217
  [self setLayer:layerContent];
Packit 971217
  [self setWantsLayer:YES];
Packit 971217
  self->layer = (GstGLCAOpenGLLayer *)layerContent;
Packit 971217
  [self->layer setDrawCallback:(GstGLWindowCB)gst_gl_cocoa_draw_cb
Packit 971217
      data:window notify:NULL];
Packit 971217
  [self->layer setResizeCallback:(GstGLWindowResizeCB)gst_gl_cocoa_resize_cb
Packit 971217
      data:(__bridge_retained gpointer)self notify:(GDestroyNotify)CFRelease];
Packit 971217
Packit 971217
  [self setLayerContentsRedrawPolicy:NSViewLayerContentsRedrawOnSetNeedsDisplay];
Packit 971217
Packit 971217
  [self setWantsBestResolutionOpenGLSurface:YES];
Packit 971217
Packit 971217
  return self;
Packit 971217
}
Packit 971217
Packit 971217
- (void) dealloc {
Packit 971217
  self->layer = nil;
Packit 971217
}
Packit 971217
Packit 971217
- (void)renewGState {
Packit 971217
  /* Don't update the screen until we redraw, this
Packit 971217
   * prevents flickering during scrolling, clipping,
Packit 971217
   * resizing, etc
Packit 971217
   */
Packit 971217
  [[self window] disableScreenUpdatesUntilFlush];
Packit 971217
Packit 971217
  [super renewGState];
Packit 971217
}
Packit 971217
Packit 971217
- (BOOL) isOpaque {
Packit 971217
    return YES;
Packit 971217
}
Packit 971217
Packit 971217
- (BOOL) isFlipped {
Packit 971217
    return NO;
Packit 971217
}
Packit 971217
Packit 971217
@end
Packit 971217
Packit 971217
void
Packit 971217
_invoke_on_main (GstGLWindowCB func, gpointer data, GDestroyNotify notify)
Packit 971217
{
Packit 971217
  if ([NSThread isMainThread]) {
Packit 971217
    func (data);
Packit 971217
    if (notify)
Packit 971217
      notify (data);
Packit 971217
  } else {
Packit 971217
    dispatch_async (dispatch_get_main_queue (), ^{
Packit 971217
      func (data);
Packit 971217
      if (notify)
Packit 971217
        notify (data);
Packit 971217
    });
Packit 971217
  }
Packit 971217
}