/* GTK - The GIMP Toolkit * Copyright (C) 2017 Benjamin Otte * * 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 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 . */ /** * SECTION:gdkcontentformats * @Title: Content Formats * @Short_description: Advertising and negotiating of content * exchange formats * @See_also: #GdkDragContext, #GdkClipboard, #GdkContentProvider * * This section describes the #GdkContentFormats structure that is used to * advertise and negotiate the format of content passed between different * widgets, windows or applications using for example the clipboard or * drag'n'drop. * * GDK supports content in 2 forms: #GType and mime type. * Using #GTypes is meant only for in-process content transfers. Mime types * are meant to be used for data passing both in-process and out-of-process. * The details of how data is passed is described in the documentation of * the actual implementations. * * A #GdkContentFormats describes a set of possible formats content can be * exchanged in. It is assumed that this set is ordered. #GTypes are more * important than mime types. Order between different #Gtypes or mime types * is the order they were added in, most important first. Functions that * care about order, such as gdk_content_formats_union() will describe in * their documentation how they interpret that order, though in general the * order of the first argument is considered the primary order of the result, * followed by the order of further arguments. * * For debugging purposes, the function gdk_content_formats_to_string() exists. * It will print a comma-seperated formats of formats from most important to least * important. * * #GdkContentFormats is an immutable struct. After creation, you cannot change * the types it represents. Instead, new #GdkContentFormats have to be created. * The #GdkContentFormatsBuilder structure is meant to help in this endeavor. */ /** * GdkContentFormats: * * A #GdkContentFormats struct is a reference counted struct * and should be treated as opaque. */ #include "config.h" #include "gdkcontentformats.h" #include "gdkcontentformatsprivate.h" #include struct _GdkContentFormats { /*< private >*/ guint ref_count; const char **mime_types; /* interned */ gsize n_mime_types; GType *gtypes; gsize n_gtypes; }; G_DEFINE_BOXED_TYPE (GdkContentFormats, gdk_content_formats, gdk_content_formats_ref, gdk_content_formats_unref) /** * gdk_intern_mime_type: * @string: (transfer none): string of a potential mime type * * Canonicalizes the given mime type and interns the result. * * If @string is not a valid mime type, %NULL is returned instead. * See RFC 2048 for the syntax if mime types. * * Returns: An interned string for the canonicalized mime type * or %NULL if the string wasn't a valid mime type **/ const char * gdk_intern_mime_type (const char *string) { char *tmp; g_return_val_if_fail (string != NULL, NULL); if (!strchr (string, '/')) return NULL; tmp = g_ascii_strdown (string, -1); string = g_intern_string (tmp); g_free (tmp); return string; } static GdkContentFormats * gdk_content_formats_new_take (GType * gtypes, gsize n_gtypes, const char **mime_types, gsize n_mime_types) { GdkContentFormats *result = g_slice_new0 (GdkContentFormats); result->ref_count = 1; result->gtypes = gtypes; result->n_gtypes = n_gtypes; result->mime_types = mime_types; result->n_mime_types = n_mime_types; return result; } /** * gdk_content_formats_new: * @mime_types: (array length=n_mime_types) (allow-none): Pointer to an * array of mime types * @n_mime_types: number of entries in @mime_types. * * Creates a new #GdkContentFormats from an array of mime types. * * The mime types must be valid and different from each other or the * behavior of the return value is undefined. If you cannot guarantee * this, use #GdkContentFormatsBuilder instead. * * Returns: (transfer full): the new #GdkContentFormats. **/ GdkContentFormats * gdk_content_formats_new (const char **mime_types, guint n_mime_types) { guint i; const char **mime_types_copy; if (n_mime_types == 0) return gdk_content_formats_new_take (NULL, 0, NULL, 0); mime_types_copy = g_new (const char *, n_mime_types + 1); for (i = 0; i < n_mime_types; i++) mime_types_copy[i] = g_intern_string (mime_types[i]); mime_types_copy[n_mime_types] = NULL; return gdk_content_formats_new_take (NULL, 0, mime_types_copy, n_mime_types); } /** * gdk_content_formats_new_for_gtype: * @type: a $GType * * Creates a new #GdkContentFormats for a given #GType. * * Returns: a new #GdkContentFormats **/ GdkContentFormats * gdk_content_formats_new_for_gtype (GType type) { GType *data; g_return_val_if_fail (type != G_TYPE_INVALID, NULL); data = g_new (GType, 2); data[0] = type; data[1] = G_TYPE_INVALID; return gdk_content_formats_new_take (data, 1, NULL, 0); } /** * gdk_content_formats_ref: * @formats: a #GdkContentFormats * * Increases the reference count of a #GdkContentFormats by one. * * Returns: the passed in #GdkContentFormats. **/ GdkContentFormats * gdk_content_formats_ref (GdkContentFormats *formats) { g_return_val_if_fail (formats != NULL, NULL); formats->ref_count++; return formats; } /** * gdk_content_formats_unref: * @formats: a #GdkContentFormats * * Decreases the reference count of a #GdkContentFormats by one. * If the resulting reference count is zero, frees the formats. **/ void gdk_content_formats_unref (GdkContentFormats *formats) { g_return_if_fail (formats != NULL); g_return_if_fail (formats->ref_count > 0); formats->ref_count--; if (formats->ref_count > 0) return; g_free (formats->gtypes); g_free (formats->mime_types); g_slice_free (GdkContentFormats, formats); } /** * gdk_content_formats_print: * @formats: a #GdkContentFormats * @string: a #GString to print into * * Prints the given @formats into a string for human consumption. * This is meant for debugging and logging. * * The form of the representation may change at any time and is * not guaranteed to stay identical. **/ void gdk_content_formats_print (GdkContentFormats *formats, GString *string) { gsize i; g_return_if_fail (formats != NULL); g_return_if_fail (string != NULL); g_string_append (string, "{ "); for (i = 0; i < formats->n_gtypes; i++) { if (i > 0) g_string_append (string, ", "); g_string_append (string, g_type_name (formats->gtypes[i])); } for (i = 0; i < formats->n_mime_types; i++) { if (i > 0 || formats->n_gtypes > 0) g_string_append (string, ", "); g_string_append (string, formats->mime_types[i]); } g_string_append (string, " }"); } /** * gdk_content_formats_to_string: * @formats: a #GdkContentFormats * * Prints the given @formats into a human-readable string. * This is a small wrapper around gdk_content_formats_print() to help * when debugging. * * Returns: (transfer full): a new string **/ char * gdk_content_formats_to_string (GdkContentFormats *formats) { GString *string; g_return_val_if_fail (formats != NULL, NULL); string = g_string_new (NULL); gdk_content_formats_print (formats, string); return g_string_free (string, FALSE); } /** * gdk_content_formats_union: * @first: (transfer full): the #GdkContentFormats to merge into * @second: (transfer none): the #GdkContentFormats to merge from * * Append all missing types from @second to @first, in the order * they had in @second. * * Returns: a new #GdkContentFormats */ GdkContentFormats * gdk_content_formats_union (GdkContentFormats *first, const GdkContentFormats *second) { GdkContentFormatsBuilder *builder; g_return_val_if_fail (first != NULL, NULL); g_return_val_if_fail (second != NULL, NULL); builder = gdk_content_formats_builder_new (); gdk_content_formats_builder_add_formats (builder, first); gdk_content_formats_unref (first); gdk_content_formats_builder_add_formats (builder, second); return gdk_content_formats_builder_free_to_formats (builder); } static gboolean gdk_content_formats_contain_interned_mime_type (const GdkContentFormats *formats, const char *mime_type) { gsize i; for (i = 0; i < formats->n_mime_types; i++) { if (mime_type == formats->mime_types[i]) return TRUE; } return FALSE; } /** * gdk_content_formats_match: * @first: the primary #GdkContentFormats to intersect * @second: the #GdkContentFormats to intersect with * * Checks if @first and @second have any matching formats. * * Returns: %TRUE if a matching format was found. */ gboolean gdk_content_formats_match (const GdkContentFormats *first, const GdkContentFormats *second) { g_return_val_if_fail (first != NULL, FALSE); g_return_val_if_fail (second != NULL, FALSE); return gdk_content_formats_match_gtype (first, second) != G_TYPE_INVALID || gdk_content_formats_match_mime_type (first, second) != NULL; } /** * gdk_content_formats_match_gtype: * @first: the primary #GdkContentFormats to intersect * @second: the #GdkContentFormats to intersect with * * Finds the first #GType from @first that is also contained * in @second. If no matching #GType is found, %G_TYPE_INVALID * is returned. * * Returns: The first common #GType or %G_TYPE_INVALID if none. **/ GType gdk_content_formats_match_gtype (const GdkContentFormats *first, const GdkContentFormats *second) { gsize i; g_return_val_if_fail (first != NULL, FALSE); g_return_val_if_fail (second != NULL, FALSE); for (i = 0; i < first->n_gtypes; i++) { if (gdk_content_formats_contain_gtype (second, first->gtypes[i])) return first->gtypes[i]; } return G_TYPE_INVALID; } /** * gdk_content_formats_match_mime_type: * @first: the primary #GdkContentFormats to intersect * @second: the #GdkContentFormats to intersect with * * Finds the first mime type from @first that is also contained * in @second. If no matching mime type is found, %NULL is * returned. * * Returns: The first common mime type or %NULL if none. **/ const char * gdk_content_formats_match_mime_type (const GdkContentFormats *first, const GdkContentFormats *second) { gsize i; g_return_val_if_fail (first != NULL, FALSE); g_return_val_if_fail (second != NULL, FALSE); for (i = 0; i < first->n_mime_types; i++) { if (gdk_content_formats_contain_interned_mime_type (second, first->mime_types[i])) return first->mime_types[i]; } return NULL; } /** * gdk_content_formats_contain_gtype: * @formats: a #GdkContentFormats * @type: the #GType to search for * * Checks if a given #GType is part of the given @formats. * * Returns: %TRUE if the #GType was found **/ gboolean gdk_content_formats_contain_gtype (const GdkContentFormats *formats, GType type) { g_return_val_if_fail (formats != NULL, FALSE); gsize i; for (i = 0; i < formats->n_gtypes; i++) { if (type == formats->gtypes[i]) return TRUE; } return FALSE; } /** * gdk_content_formats_contain_mime_type: * @formats: a #GdkContentFormats * @mime_type: the mime type to search for * * Checks if a given mime type is part of the given @formats. * * Returns: %TRUE if the mime_type was found **/ gboolean gdk_content_formats_contain_mime_type (const GdkContentFormats *formats, const char *mime_type) { g_return_val_if_fail (formats != NULL, FALSE); g_return_val_if_fail (mime_type != NULL, FALSE); return gdk_content_formats_contain_interned_mime_type (formats, g_intern_string (mime_type)); } /** * gdk_content_formats_get_gtypes: * @formats: a #GdkContentFormats * @n_gtypes: (out) (allow-none): optional pointer to take the * number of #GTypes contained in the return value * * Gets the #GTypes included in @formats. Note that @formats may not * contain any #GTypes, in particular when they are empty. In that * case %NULL will be returned. * * Returns: (transfer none) (nullable): %G_TYPE_INVALID-terminated array of * types included in @formats or %NULL if none. **/ const GType * gdk_content_formats_get_gtypes (GdkContentFormats *formats, gsize *n_gtypes) { g_return_val_if_fail (formats != NULL, NULL); if (n_gtypes) *n_gtypes = formats->n_gtypes; return formats->gtypes; } /** * gdk_content_formats_get_mime_types: * @formats: a #GdkContentFormats * @n_mime_types: (out) (allow-none): optional pointer to take the * number of mime types contained in the return value * * Gets the mime types included in @formats. Note that @formats may not * contain any mime types, in particular when they are empty. In that * case %NULL will be returned. * * Returns: (transfer none) (nullable): %NULL-terminated array of * interned strings of mime types included in @formats or %NULL * if none. **/ const char * const * gdk_content_formats_get_mime_types (GdkContentFormats *formats, gsize *n_mime_types) { g_return_val_if_fail (formats != NULL, NULL); if (n_mime_types) *n_mime_types = formats->n_mime_types; return formats->mime_types; } /** * GdkContentFormatsBuilder: * * A #GdkContentFormatsBuilder struct is an opaque struct. It is meant to * not be kept around and only be used to create new #GdkContentFormats * objects. */ struct _GdkContentFormatsBuilder { int ref_count; /* (element-type GType) */ GSList *gtypes; gsize n_gtypes; /* (element-type utf8) (interned) */ GSList *mime_types; gsize n_mime_types; }; G_DEFINE_BOXED_TYPE (GdkContentFormatsBuilder, gdk_content_formats_builder, gdk_content_formats_builder_ref, gdk_content_formats_builder_unref) /** * gdk_content_formats_builder_new: * * Create a new #GdkContentFormatsBuilder object. The resulting builder * would create an empty #GdkContentFormats. Use addition functions to add * types to it. * * Returns: a new #GdkContentFormatsBuilder **/ GdkContentFormatsBuilder * gdk_content_formats_builder_new (void) { GdkContentFormatsBuilder *builder; builder = g_slice_new0 (GdkContentFormatsBuilder); builder->ref_count = 1; return builder; } /** * gdk_content_formats_builder_ref: * @builder: a #GdkContentFormatsBuilder * * Acquires a reference on the given @builder. * * This function is intended primarily for bindings. #GdkContentFormatsBuilder objects * should not be kept around. * * Returns: (transfer none): the given #GdkContentFormatsBuilder with * its reference count increased */ GdkContentFormatsBuilder * gdk_content_formats_builder_ref (GdkContentFormatsBuilder *builder) { g_return_val_if_fail (builder != NULL, NULL); g_return_val_if_fail (builder->ref_count > 0, NULL); builder->ref_count += 1; return builder; } static void gdk_content_formats_builder_clear (GdkContentFormatsBuilder *builder) { g_clear_pointer (&builder->gtypes, g_slist_free); g_clear_pointer (&builder->mime_types, g_slist_free); } /** * gdk_content_formats_builder_unref: * @builder: a #GdkContentFormatsBuilder * * Releases a reference on the given @builder. */ void gdk_content_formats_builder_unref (GdkContentFormatsBuilder *builder) { g_return_if_fail (builder != NULL); g_return_if_fail (builder->ref_count > 0); builder->ref_count -= 1; if (builder->ref_count > 0) return; gdk_content_formats_builder_clear (builder); g_slice_free (GdkContentFormatsBuilder, builder); } /** * gdk_content_formats_builder_free_to_formats: (skip) * @builder: a #GdkContentFormatsBuilder * * Creates a new #GdkContentFormats from the current state of the * given @builder, and frees the @builder instance. * * Returns: (transfer full): the newly created #GdkContentFormats * with all the formats added to @builder */ GdkContentFormats * gdk_content_formats_builder_free_to_formats (GdkContentFormatsBuilder *builder) { GdkContentFormats *res; g_return_val_if_fail (builder != NULL, NULL); res = gdk_content_formats_builder_to_formats (builder); gdk_content_formats_builder_unref (builder); return res; } /** * gdk_content_formats_builder_to_formats: * @builder: a #GdkContentFormatsBuilder * * Creates a new #GdkContentFormats from the given @builder. * * The given #GdkContentFormatsBuilder is reset once this function returns; * you cannot call this function multiple times on the same @builder instance. * * This function is intended primarily for bindings. C code should use * gdk_content_formats_builder_free_to_formats(). * * Returns: (transfer full): the newly created #GdkContentFormats * with all the formats added to @builder */ GdkContentFormats * gdk_content_formats_builder_to_formats (GdkContentFormatsBuilder *builder) { GdkContentFormats *result; GType *gtypes; const char **mime_types; GSList *l; gsize i; g_return_val_if_fail (builder != NULL, NULL); gtypes = g_new (GType, builder->n_gtypes + 1); i = builder->n_gtypes; gtypes[i--] = G_TYPE_INVALID; /* add backwards because most important type is last in the list */ for (l = builder->gtypes; l; l = l->next) gtypes[i--] = GPOINTER_TO_SIZE (l->data); mime_types = g_new (const char *, builder->n_mime_types + 1); i = builder->n_mime_types; mime_types[i--] = NULL; /* add backwards because most important type is last in the list */ for (l = builder->mime_types; l; l = l->next) mime_types[i--] = l->data; result = gdk_content_formats_new_take (gtypes, builder->n_gtypes, mime_types, builder->n_mime_types); gdk_content_formats_builder_clear (builder); return result; } /** * gdk_content_formats_builder_add_formats: * @builder: a #GdkContentFormatsBuilder * @formats: the formats to add * * Appends all formats from @formats to @builder, skipping those that * already exist. **/ void gdk_content_formats_builder_add_formats (GdkContentFormatsBuilder *builder, const GdkContentFormats *formats) { gsize i; g_return_if_fail (builder != NULL); g_return_if_fail (formats != NULL); for (i = 0; i < formats->n_gtypes; i++) gdk_content_formats_builder_add_gtype (builder, formats->gtypes[i]); for (i = 0; i < formats->n_mime_types; i++) gdk_content_formats_builder_add_mime_type (builder, formats->mime_types[i]); } /** * gdk_content_formats_builder_add_gtype: * @builder: a #GdkContentFormatsBuilder * @type: a #GType * * Appends @gtype to @builder if it has not already been added. **/ void gdk_content_formats_builder_add_gtype (GdkContentFormatsBuilder *builder, GType type) { g_return_if_fail (builder != NULL); g_return_if_fail (type != G_TYPE_INVALID); if (g_slist_find (builder->gtypes, GSIZE_TO_POINTER (type))) return; builder->gtypes = g_slist_prepend (builder->gtypes, GSIZE_TO_POINTER (type)); builder->n_gtypes++; } /** * gdk_content_formats_builder_add_mime_type: * @builder: a #GdkContentFormatsBuilder * @mime_type: a mime type * * Appends @mime_type to @builder if it has not already been added. **/ void gdk_content_formats_builder_add_mime_type (GdkContentFormatsBuilder *builder, const char *mime_type) { g_return_if_fail (builder != NULL); g_return_if_fail (mime_type != NULL); mime_type = g_intern_string (mime_type); if (g_slist_find (builder->mime_types, mime_type)) return; builder->mime_types = g_slist_prepend (builder->mime_types, (gpointer) mime_type); builder->n_mime_types++; } /* G_DEFINE_BOXED wants this */ typedef gpointer GdkFileList; static gpointer gdk_file_list_copy (gpointer list) { return g_slist_copy_deep (list, (GCopyFunc) g_object_ref, NULL); } static void gdk_file_list_free (gpointer list) { g_slist_free_full (list, g_object_unref); } G_DEFINE_BOXED_TYPE (GdkFileList, gdk_file_list, gdk_file_list_copy, gdk_file_list_free)