603 lines
17 KiB
C
603 lines
17 KiB
C
/*
|
|
* Copyright © 2021 Red Hat, Inc.
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* This library 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 GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library. If not, see <http://www.gnu.org/licenses/>.
|
|
*
|
|
* SPDX-License-Identifier: LGPL-2.1-or-later
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include <glib/gi18n.h>
|
|
|
|
#include "gdkdragprivate.h"
|
|
#include "gdkmacospasteboard-private.h"
|
|
#include "gdkmacosutils-private.h"
|
|
|
|
enum {
|
|
TYPE_STRING,
|
|
TYPE_PBOARD,
|
|
TYPE_URL,
|
|
TYPE_FILE_URL,
|
|
TYPE_COLOR,
|
|
TYPE_TIFF,
|
|
TYPE_PNG,
|
|
TYPE_LAST
|
|
};
|
|
|
|
#define PTYPE(k) (get_pasteboard_type(TYPE_##k))
|
|
|
|
static NSPasteboardType pasteboard_types[TYPE_LAST];
|
|
|
|
static NSPasteboardType
|
|
get_pasteboard_type (int type)
|
|
{
|
|
static gsize initialized = FALSE;
|
|
|
|
g_assert (type >= 0);
|
|
g_assert (type < TYPE_LAST);
|
|
|
|
if (g_once_init_enter (&initialized))
|
|
{
|
|
pasteboard_types[TYPE_PNG] = NSPasteboardTypePNG;
|
|
pasteboard_types[TYPE_STRING] = NSPasteboardTypeString;
|
|
pasteboard_types[TYPE_TIFF] = NSPasteboardTypeTIFF;
|
|
pasteboard_types[TYPE_COLOR] = NSPasteboardTypeColor;
|
|
|
|
G_GNUC_BEGIN_IGNORE_DEPRECATIONS
|
|
pasteboard_types[TYPE_PBOARD] = NSStringPboardType;
|
|
G_GNUC_END_IGNORE_DEPRECATIONS
|
|
|
|
#ifdef AVAILABLE_MAC_OS_X_VERSION_10_13_AND_LATER
|
|
pasteboard_types[TYPE_URL] = NSPasteboardTypeURL;
|
|
pasteboard_types[TYPE_FILE_URL] = NSPasteboardTypeFileURL;
|
|
#else
|
|
pasteboard_types[TYPE_URL] = [[NSString alloc] initWithUTF8String:"public.url"];
|
|
pasteboard_types[TYPE_FILE_URL] = [[NSString alloc] initWithUTF8String:"public.file-url"];
|
|
#endif
|
|
|
|
g_once_init_leave (&initialized, TRUE);
|
|
}
|
|
|
|
return pasteboard_types[type];
|
|
}
|
|
|
|
const char *
|
|
_gdk_macos_pasteboard_from_ns_type (NSPasteboardType type)
|
|
{
|
|
G_GNUC_BEGIN_IGNORE_DEPRECATIONS;
|
|
|
|
if ([type isEqualToString:PTYPE(STRING)] ||
|
|
[type isEqualToString:PTYPE(PBOARD)])
|
|
return g_intern_string ("text/plain;charset=utf-8");
|
|
else if ([type isEqualToString:PTYPE(URL)] ||
|
|
[type isEqualToString:PTYPE(FILE_URL)])
|
|
return g_intern_string ("text/uri-list");
|
|
else if ([type isEqualToString:PTYPE(COLOR)])
|
|
return g_intern_string ("application/x-color");
|
|
else if ([type isEqualToString:PTYPE(TIFF)])
|
|
return g_intern_string ("image/tiff");
|
|
else if ([type isEqualToString:PTYPE(PNG)])
|
|
return g_intern_string ("image/png");
|
|
|
|
G_GNUC_END_IGNORE_DEPRECATIONS;
|
|
|
|
return NULL;
|
|
}
|
|
|
|
NSPasteboardType
|
|
_gdk_macos_pasteboard_to_ns_type (const char *mime_type,
|
|
NSPasteboardType *alternate)
|
|
{
|
|
if (alternate)
|
|
*alternate = NULL;
|
|
|
|
if (g_strcmp0 (mime_type, "text/plain;charset=utf-8") == 0)
|
|
{
|
|
return PTYPE(STRING);
|
|
}
|
|
else if (g_strcmp0 (mime_type, "text/uri-list") == 0)
|
|
{
|
|
if (alternate)
|
|
*alternate = PTYPE(URL);
|
|
return PTYPE(FILE_URL);
|
|
}
|
|
else if (g_strcmp0 (mime_type, "application/x-color") == 0)
|
|
{
|
|
return PTYPE(COLOR);
|
|
}
|
|
else if (g_strcmp0 (mime_type, "image/tiff") == 0)
|
|
{
|
|
return PTYPE(TIFF);
|
|
}
|
|
else if (g_strcmp0 (mime_type, "image/png") == 0)
|
|
{
|
|
return PTYPE(PNG);
|
|
}
|
|
|
|
return nil;
|
|
}
|
|
|
|
static void
|
|
populate_content_formats (GdkContentFormatsBuilder *builder,
|
|
NSPasteboardType type)
|
|
{
|
|
const char *mime_type;
|
|
|
|
g_assert (builder != NULL);
|
|
g_assert (type != NULL);
|
|
|
|
if ((mime_type = _gdk_macos_pasteboard_from_ns_type (type)))
|
|
gdk_content_formats_builder_add_mime_type (builder, mime_type);
|
|
}
|
|
|
|
static GdkContentFormats *
|
|
load_offer_formats (NSPasteboard *pasteboard)
|
|
{
|
|
GDK_BEGIN_MACOS_ALLOC_POOL;
|
|
|
|
GdkContentFormatsBuilder *builder;
|
|
GdkContentFormats *formats;
|
|
|
|
builder = gdk_content_formats_builder_new ();
|
|
for (NSPasteboardType type in [pasteboard types])
|
|
populate_content_formats (builder, type);
|
|
formats = gdk_content_formats_builder_free_to_formats (builder);
|
|
|
|
GDK_END_MACOS_ALLOC_POOL;
|
|
|
|
return g_steal_pointer (&formats);
|
|
}
|
|
|
|
GdkContentFormats *
|
|
_gdk_macos_pasteboard_load_formats (NSPasteboard *pasteboard)
|
|
{
|
|
return load_offer_formats (pasteboard);
|
|
}
|
|
|
|
static GInputStream *
|
|
create_stream_from_nsdata (NSData *data)
|
|
{
|
|
const guint8 *bytes = [data bytes];
|
|
gsize len = [data length];
|
|
|
|
return g_memory_input_stream_new_from_data (g_memdup2 (bytes, len), len, g_free);
|
|
}
|
|
|
|
void
|
|
_gdk_macos_pasteboard_read_async (GObject *object,
|
|
NSPasteboard *pasteboard,
|
|
GdkContentFormats *formats,
|
|
int io_priority,
|
|
GCancellable *cancellable,
|
|
GAsyncReadyCallback callback,
|
|
gpointer user_data)
|
|
{
|
|
GDK_BEGIN_MACOS_ALLOC_POOL;
|
|
|
|
GdkContentFormats *offer_formats = NULL;
|
|
const char *mime_type;
|
|
GInputStream *stream = NULL;
|
|
GTask *task = NULL;
|
|
|
|
g_assert (G_IS_OBJECT (object));
|
|
g_assert (pasteboard != NULL);
|
|
g_assert (formats != NULL);
|
|
|
|
task = g_task_new (object, cancellable, callback, user_data);
|
|
g_task_set_source_tag (task, _gdk_macos_pasteboard_read_async);
|
|
g_task_set_priority (task, io_priority);
|
|
|
|
offer_formats = load_offer_formats (pasteboard);
|
|
mime_type = gdk_content_formats_match_mime_type (formats, offer_formats);
|
|
|
|
if (mime_type == NULL)
|
|
{
|
|
g_task_return_new_error (task,
|
|
G_IO_ERROR,
|
|
G_IO_ERROR_NOT_SUPPORTED,
|
|
"%s",
|
|
_("No compatible transfer format found"));
|
|
goto cleanup;
|
|
}
|
|
|
|
if (strcmp (mime_type, "text/plain;charset=utf-8") == 0)
|
|
{
|
|
NSString *nsstr = [pasteboard stringForType:NSPasteboardTypeString];
|
|
|
|
if (nsstr != NULL)
|
|
{
|
|
const char *str = [nsstr UTF8String];
|
|
stream = g_memory_input_stream_new_from_data (g_strdup (str),
|
|
strlen (str) + 1,
|
|
g_free);
|
|
}
|
|
}
|
|
else if (strcmp (mime_type, "text/uri-list") == 0)
|
|
{
|
|
G_GNUC_BEGIN_IGNORE_DEPRECATIONS;
|
|
|
|
if ([[pasteboard types] containsObject:PTYPE(FILE_URL)])
|
|
{
|
|
GString *str = g_string_new (NULL);
|
|
NSArray *files = [pasteboard propertyListForType:NSFilenamesPboardType];
|
|
gsize n_files = [files count];
|
|
char *data;
|
|
guint len;
|
|
|
|
for (gsize i = 0; i < n_files; ++i)
|
|
{
|
|
NSString* uriString = [files objectAtIndex:i];
|
|
uriString = [@"file://" stringByAppendingString:uriString];
|
|
uriString = [uriString stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding];
|
|
|
|
g_string_append_printf (str,
|
|
"%s\r\n",
|
|
[uriString cStringUsingEncoding:NSUTF8StringEncoding]);
|
|
}
|
|
|
|
len = str->len;
|
|
data = g_string_free (str, FALSE);
|
|
stream = g_memory_input_stream_new_from_data (data, len, g_free);
|
|
}
|
|
|
|
G_GNUC_END_IGNORE_DEPRECATIONS;
|
|
}
|
|
else if (strcmp (mime_type, "application/x-color") == 0)
|
|
{
|
|
NSColorSpace *colorspace;
|
|
NSColor *nscolor;
|
|
guint16 color[4];
|
|
|
|
colorspace = [NSColorSpace genericRGBColorSpace];
|
|
nscolor = [[NSColor colorFromPasteboard:pasteboard]
|
|
colorUsingColorSpace:colorspace];
|
|
|
|
color[0] = 0xffff * [nscolor redComponent];
|
|
color[1] = 0xffff * [nscolor greenComponent];
|
|
color[2] = 0xffff * [nscolor blueComponent];
|
|
color[3] = 0xffff * [nscolor alphaComponent];
|
|
|
|
stream = g_memory_input_stream_new_from_data (g_memdup2 (&color, sizeof color),
|
|
sizeof color,
|
|
g_free);
|
|
}
|
|
else if (strcmp (mime_type, "image/tiff") == 0)
|
|
{
|
|
NSData *data = [pasteboard dataForType:PTYPE(TIFF)];
|
|
stream = create_stream_from_nsdata (data);
|
|
}
|
|
else if (strcmp (mime_type, "image/png") == 0)
|
|
{
|
|
NSData *data = [pasteboard dataForType:PTYPE(PNG)];
|
|
stream = create_stream_from_nsdata (data);
|
|
}
|
|
|
|
if (stream != NULL)
|
|
{
|
|
g_task_set_task_data (task, g_strdup (mime_type), g_free);
|
|
g_task_return_pointer (task, g_steal_pointer (&stream), g_object_unref);
|
|
}
|
|
else
|
|
{
|
|
g_task_return_new_error (task,
|
|
G_IO_ERROR,
|
|
G_IO_ERROR_FAILED,
|
|
_("Failed to decode contents with mime-type of '%s'"),
|
|
mime_type);
|
|
}
|
|
|
|
cleanup:
|
|
g_clear_object (&task);
|
|
g_clear_pointer (&offer_formats, gdk_content_formats_unref);
|
|
|
|
GDK_END_MACOS_ALLOC_POOL;
|
|
}
|
|
|
|
GInputStream *
|
|
_gdk_macos_pasteboard_read_finish (GObject *object,
|
|
GAsyncResult *result,
|
|
const char **out_mime_type,
|
|
GError **error)
|
|
{
|
|
GTask *task = (GTask *)result;
|
|
|
|
g_assert (G_IS_OBJECT (object));
|
|
g_assert (G_IS_TASK (task));
|
|
|
|
if (out_mime_type != NULL)
|
|
*out_mime_type = g_strdup (g_task_get_task_data (task));
|
|
|
|
return g_task_propagate_pointer (task, error);
|
|
}
|
|
|
|
void
|
|
_gdk_macos_pasteboard_register_drag_types (NSWindow *window)
|
|
{
|
|
[window registerForDraggedTypes:[NSArray arrayWithObjects:PTYPE(STRING),
|
|
PTYPE(PBOARD),
|
|
PTYPE(URL),
|
|
PTYPE(FILE_URL),
|
|
PTYPE(COLOR),
|
|
PTYPE(TIFF),
|
|
PTYPE(PNG),
|
|
nil]];
|
|
}
|
|
|
|
@implementation GdkMacosPasteboardItemDataProvider
|
|
|
|
-(id)initForClipboard:(GdkClipboard*)clipboard withContentProvider:(GdkContentProvider*)contentProvider
|
|
{
|
|
[super init];
|
|
g_set_object (&self->_clipboard, clipboard);
|
|
g_set_object (&self->_contentProvider, contentProvider);
|
|
return self;
|
|
}
|
|
|
|
-(id)initForDrag:(GdkDrag*)drag withContentProvider:(GdkContentProvider*)contentProvider
|
|
{
|
|
[super init];
|
|
g_set_object (&self->_drag, drag);
|
|
g_set_object (&self->_contentProvider, contentProvider);
|
|
return self;
|
|
}
|
|
|
|
-(void)dealloc
|
|
{
|
|
g_clear_object (&self->_contentProvider);
|
|
g_clear_object (&self->_clipboard);
|
|
g_clear_object (&self->_drag);
|
|
[super dealloc];
|
|
}
|
|
|
|
-(NSArray<NSPasteboardType> *)types
|
|
{
|
|
NSMutableArray *ret = [[NSMutableArray alloc] init];
|
|
GdkContentFormats *serializable;
|
|
const char * const *mime_types;
|
|
gsize n_mime_types;
|
|
|
|
serializable = gdk_content_provider_ref_storable_formats (self->_contentProvider);
|
|
serializable = gdk_content_formats_union_serialize_mime_types (serializable);
|
|
mime_types = gdk_content_formats_get_mime_types (serializable, &n_mime_types);
|
|
|
|
for (gsize i = 0; i < n_mime_types; i++)
|
|
{
|
|
const char *mime_type = mime_types[i];
|
|
NSPasteboardType type;
|
|
NSPasteboardType alternate = nil;
|
|
|
|
if ((type = _gdk_macos_pasteboard_to_ns_type (mime_type, &alternate)))
|
|
{
|
|
[ret addObject:type];
|
|
if (alternate)
|
|
[ret addObject:alternate];
|
|
}
|
|
}
|
|
|
|
gdk_content_formats_unref (serializable);
|
|
|
|
/* Default to an url type (think gobject://internal)
|
|
* to support internal, GType-based DnD.
|
|
*/
|
|
if (n_mime_types == 0)
|
|
{
|
|
GdkContentFormats *formats;
|
|
gsize n_gtypes;
|
|
|
|
formats = gdk_content_provider_ref_formats (self->_contentProvider);
|
|
gdk_content_formats_get_gtypes (formats, &n_gtypes);
|
|
|
|
if (n_gtypes)
|
|
[ret addObject:NSPasteboardTypeURL];
|
|
|
|
gdk_content_formats_unref (formats);
|
|
}
|
|
|
|
return g_steal_pointer (&ret);
|
|
}
|
|
|
|
typedef struct
|
|
{
|
|
GMemoryOutputStream *stream;
|
|
NSPasteboardItem *item;
|
|
NSPasteboardType type;
|
|
GMainContext *main_context;
|
|
guint done : 1;
|
|
} WriteRequest;
|
|
|
|
static void
|
|
write_request_free (WriteRequest *wr)
|
|
{
|
|
g_clear_pointer (&wr->main_context, g_main_context_unref);
|
|
g_clear_object (&wr->stream);
|
|
[wr->item release];
|
|
g_free (wr);
|
|
}
|
|
|
|
static void
|
|
on_data_ready_cb (GObject *object,
|
|
GAsyncResult *result,
|
|
gpointer user_data)
|
|
{
|
|
GDK_BEGIN_MACOS_ALLOC_POOL;
|
|
|
|
WriteRequest *wr = user_data;
|
|
GError *error = NULL;
|
|
NSData *data = nil;
|
|
gboolean ret;
|
|
|
|
g_assert (G_IS_OBJECT (object));
|
|
g_assert (GDK_IS_CLIPBOARD (object) || GDK_IS_DRAG (object));
|
|
g_assert (G_IS_ASYNC_RESULT (result));
|
|
g_assert (wr != NULL);
|
|
g_assert (G_IS_MEMORY_OUTPUT_STREAM (wr->stream));
|
|
g_assert ([wr->item isKindOfClass:[NSPasteboardItem class]]);
|
|
|
|
if (GDK_IS_CLIPBOARD (object))
|
|
ret = gdk_clipboard_write_finish (GDK_CLIPBOARD (object), result, &error);
|
|
else if (GDK_IS_DRAG (object))
|
|
ret = gdk_drag_write_finish (GDK_DRAG (object), result, &error);
|
|
else
|
|
g_return_if_reached ();
|
|
|
|
if (ret)
|
|
{
|
|
gsize size;
|
|
gpointer bytes;
|
|
|
|
g_output_stream_close (G_OUTPUT_STREAM (wr->stream), NULL, NULL);
|
|
|
|
size = g_memory_output_stream_get_data_size (wr->stream);
|
|
bytes = g_memory_output_stream_steal_data (wr->stream);
|
|
data = [[NSData alloc] initWithBytesNoCopy:bytes
|
|
length:size
|
|
deallocator:^(void *alloc, NSUInteger length) { g_free (alloc); }];
|
|
}
|
|
else
|
|
{
|
|
g_warning ("Failed to serialize pasteboard contents: %s",
|
|
error->message);
|
|
g_clear_error (&error);
|
|
}
|
|
|
|
[wr->item setData:data forType:wr->type];
|
|
|
|
wr->done = TRUE;
|
|
|
|
GDK_END_MACOS_ALLOC_POOL;
|
|
}
|
|
|
|
-(void)pasteboard:(NSPasteboard *)pasteboard item:(NSPasteboardItem *)item provideDataForType:(NSPasteboardType)type
|
|
{
|
|
const char *mime_type = _gdk_macos_pasteboard_from_ns_type (type);
|
|
GMainContext *main_context = g_main_context_default ();
|
|
WriteRequest *wr;
|
|
|
|
if (self->_contentProvider == NULL || mime_type == NULL)
|
|
{
|
|
[item setData:[NSData data] forType:type];
|
|
return;
|
|
}
|
|
|
|
wr = g_new0 (WriteRequest, 1);
|
|
wr->item = [item retain];
|
|
wr->stream = G_MEMORY_OUTPUT_STREAM (g_memory_output_stream_new_resizable ());
|
|
wr->type = type;
|
|
wr->main_context = g_main_context_ref (main_context);
|
|
wr->done = FALSE;
|
|
|
|
if (GDK_IS_CLIPBOARD (self->_clipboard))
|
|
gdk_clipboard_write_async (self->_clipboard,
|
|
mime_type,
|
|
G_OUTPUT_STREAM (wr->stream),
|
|
G_PRIORITY_DEFAULT,
|
|
NULL,
|
|
on_data_ready_cb,
|
|
wr);
|
|
else if (GDK_IS_DRAG (self->_drag))
|
|
gdk_drag_write_async (self->_drag,
|
|
mime_type,
|
|
G_OUTPUT_STREAM (wr->stream),
|
|
G_PRIORITY_DEFAULT,
|
|
NULL,
|
|
on_data_ready_cb,
|
|
wr);
|
|
else
|
|
g_return_if_reached ();
|
|
|
|
/* We're forced to provide data synchronously via this API
|
|
* so we must block on the main loop. Using another main loop
|
|
* than the default tends to get us locked up here, so that is
|
|
* what we'll do for now.
|
|
*/
|
|
while (!wr->done)
|
|
g_main_context_iteration (wr->main_context, TRUE);
|
|
|
|
write_request_free (wr);
|
|
}
|
|
|
|
-(void)pasteboardFinishedWithDataProvider:(NSPasteboard *)pasteboard
|
|
{
|
|
g_clear_object (&self->_clipboard);
|
|
g_clear_object (&self->_drag);
|
|
g_clear_object (&self->_contentProvider);
|
|
}
|
|
|
|
@end
|
|
|
|
@implementation GdkMacosPasteboardItem
|
|
|
|
-(id)initForClipboard:(GdkClipboard*)clipboard withContentProvider:(GdkContentProvider*)contentProvider
|
|
{
|
|
GdkMacosPasteboardItemDataProvider *dataProvider;
|
|
|
|
dataProvider = [[GdkMacosPasteboardItemDataProvider alloc] initForClipboard:clipboard withContentProvider:contentProvider];
|
|
|
|
[super init];
|
|
g_set_object (&self->_clipboard, clipboard);
|
|
g_set_object (&self->_contentProvider, contentProvider);
|
|
[self setDataProvider:dataProvider forTypes:[dataProvider types]];
|
|
|
|
[dataProvider release];
|
|
|
|
return self;
|
|
}
|
|
|
|
-(id)initForDrag:(GdkDrag*)drag withContentProvider:(GdkContentProvider*)contentProvider
|
|
{
|
|
GdkMacosPasteboardItemDataProvider *dataProvider;
|
|
|
|
dataProvider = [[GdkMacosPasteboardItemDataProvider alloc] initForDrag:drag withContentProvider:contentProvider];
|
|
|
|
[super init];
|
|
g_set_object (&self->_drag, drag);
|
|
g_set_object (&self->_contentProvider, contentProvider);
|
|
[self setDataProvider:dataProvider forTypes:[dataProvider types]];
|
|
|
|
[dataProvider release];
|
|
|
|
return self;
|
|
}
|
|
|
|
-(void)dealloc
|
|
{
|
|
g_clear_object (&self->_contentProvider);
|
|
g_clear_object (&self->_clipboard);
|
|
g_clear_object (&self->_drag);
|
|
[super dealloc];
|
|
}
|
|
|
|
-(NSRect)draggingFrame
|
|
{
|
|
return self->_draggingFrame;
|
|
}
|
|
|
|
-(void)setDraggingFrame:(NSRect)draggingFrame;
|
|
{
|
|
self->_draggingFrame = draggingFrame;
|
|
}
|
|
|
|
-(id)item
|
|
{
|
|
return self;
|
|
}
|
|
|
|
-(NSArray* (^) (void))imageComponentsProvider
|
|
{
|
|
return nil;
|
|
}
|
|
|
|
@end
|