/* GIO - GLib Input, Output and Streaming Library
*
* Copyright (C) 2017 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 .
*
* Author: Benjamin Otte
*/
#include "config.h"
#include "gdktextlistconverter-x11.h"
#include "gdkintl.h"
#include "gdkprivate-x11.h"
#define GDK_X11_TEXT_LIST_CONVERTER_CLASS(k) (G_TYPE_CHECK_CLASS_CAST((k), GDK_TYPE_X11_TEXT_LIST_CONVERTER, GdkX11TextListConverterClass))
#define GDK_IS_X11_TEXT_LIST_CONVERTER_CLASS(k) (G_TYPE_CHECK_CLASS_TYPE ((k), GDK_TYPE_X11_TEXT_LIST_CONVERTER))
#define GDK_X11_TEXT_LIST_CONVERTER_GET_CLASS(o) (G_TYPE_INSTANCE_GET_CLASS ((o), GDK_TYPE_X11_TEXT_LIST_CONVERTER, GdkX11TextListConverterClass))
typedef struct _GdkX11TextListConverterClass GdkX11TextListConverterClass;
struct _GdkX11TextListConverter
{
GObject parent_instance;
GdkDisplay *display;
const char *encoding; /* interned */
gint format;
guint encoder : 1;
};
struct _GdkX11TextListConverterClass
{
GObjectClass parent_class;
};
static GConverterResult
write_output (void *outbuf,
gsize outbuf_size,
gsize *bytes_written,
const void *data,
gssize len,
GError **error)
{
if (len < 0)
len = strlen (data) + 1;
if (outbuf_size < len)
{
g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_NO_SPACE,
_("Not enough space in destination"));
return G_CONVERTER_ERROR;
}
memcpy (outbuf, data, len);
*bytes_written = len;
return G_CONVERTER_FINISHED;
}
static GConverterResult
gdk_x11_text_list_converter_decode (GdkX11TextListConverter *conv,
const void *inbuf,
gsize inbuf_size,
void *outbuf,
gsize outbuf_size,
GConverterFlags flags,
gsize *bytes_read,
gsize *bytes_written,
GError **error)
{
gint count;
char **list;
if (!(flags & G_CONVERTER_INPUT_AT_END))
{
g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_PARTIAL_INPUT,
_("Need complete input to do conversion"));
return G_CONVERTER_ERROR;
}
count = _gdk_x11_display_text_property_to_utf8_list (conv->display,
conv->encoding,
conv->format,
inbuf,
inbuf_size,
&list);
if (count < 0)
{
/* XXX: add error handling from gdkselection-x11.c */
g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_NO_SPACE,
_("Not enough space in destination"));
return G_CONVERTER_ERROR;
}
else if (count == 0)
{
*bytes_read = inbuf_size;
return write_output (outbuf, outbuf_size, bytes_written, "", 1, error);
}
else
{
GConverterResult result;
result = write_output (outbuf, outbuf_size, bytes_written, list[0], -1, error);
g_strfreev (list);
*bytes_read = inbuf_size;
return result;
}
}
static GConverterResult
gdk_x11_text_list_converter_encode (GdkX11TextListConverter *conv,
const void *inbuf,
gsize inbuf_size,
void *outbuf,
gsize outbuf_size,
GConverterFlags flags,
gsize *bytes_read,
gsize *bytes_written,
GError **error)
{
if (!(flags & G_CONVERTER_INPUT_AT_END))
{
g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_PARTIAL_INPUT,
_("Need complete input to do conversion"));
return G_CONVERTER_ERROR;
}
if (g_str_equal (conv->encoding, "STRING") ||
g_str_equal (conv->encoding, "TEXT"))
{
GConverterResult result;
gchar *tmp, *latin1;
tmp = g_strndup (inbuf, inbuf_size);
latin1 = gdk_utf8_to_string_target (tmp);
g_free (tmp);
if (latin1)
{
result = write_output (outbuf, outbuf_size, bytes_written, latin1, -1, error);
g_free (latin1);
}
else
{
g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_INVALID_DATA,
_("Invalid byte sequence in conversion input"));
result = G_CONVERTER_ERROR;
}
return result;
}
else if (g_str_equal (conv->encoding, "COMPOUND_TEXT"))
{
GConverterResult result;
guchar *text;
GdkAtom encoding;
gint format;
gint new_length;
char *tmp;
tmp = g_strndup (inbuf, inbuf_size);
if (gdk_x11_display_utf8_to_compound_text (conv->display, tmp,
&encoding, &format, &text, &new_length))
{
if (encoding == g_intern_string (conv->encoding) &&
format == conv->format)
{
result = write_output (outbuf, outbuf_size, bytes_written, text, new_length, error);
}
else
{
g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_FAILED,
_("Invalid formats in compound text conversion."));
result = G_CONVERTER_ERROR;
}
gdk_x11_free_compound_text (text);
}
else
{
g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_INVALID_DATA,
_("Invalid byte sequence in conversion input"));
result = G_CONVERTER_ERROR;
}
g_free (tmp);
return result;
}
else
{
g_set_error (error, G_IO_ERROR, G_IO_ERROR_FAILED,
_("Unsupported encoding ā%sā"), conv->encoding);
return G_CONVERTER_ERROR;
}
return FALSE;
}
static GConverterResult
gdk_x11_text_list_converter_convert (GConverter *converter,
const void *inbuf,
gsize inbuf_size,
void *outbuf,
gsize outbuf_size,
GConverterFlags flags,
gsize *bytes_read,
gsize *bytes_written,
GError **error)
{
GdkX11TextListConverter *conv = GDK_X11_TEXT_LIST_CONVERTER (converter);
if (conv->encoder)
{
return gdk_x11_text_list_converter_encode (conv,
inbuf, inbuf_size,
outbuf, outbuf_size,
flags,
bytes_read, bytes_written,
error);
}
else
{
return gdk_x11_text_list_converter_decode (conv,
inbuf, inbuf_size,
outbuf, outbuf_size,
flags,
bytes_read, bytes_written,
error);
}
}
static void
gdk_x11_text_list_converter_reset (GConverter *converter)
{
}
static void
gdk_x11_text_list_converter_iface_init (GConverterIface *iface)
{
iface->convert = gdk_x11_text_list_converter_convert;
iface->reset = gdk_x11_text_list_converter_reset;
}
G_DEFINE_TYPE_WITH_CODE (GdkX11TextListConverter, gdk_x11_text_list_converter, G_TYPE_OBJECT,
G_IMPLEMENT_INTERFACE (G_TYPE_CONVERTER,
gdk_x11_text_list_converter_iface_init))
static void
gdk_x11_text_list_converter_finalize (GObject *object)
{
GdkX11TextListConverter *conv = GDK_X11_TEXT_LIST_CONVERTER (object);
g_object_unref (conv->display);
G_OBJECT_CLASS (gdk_x11_text_list_converter_parent_class)->finalize (object);
}
static void
gdk_x11_text_list_converter_class_init (GdkX11TextListConverterClass *klass)
{
GObjectClass *object_class = G_OBJECT_CLASS (klass);
object_class->finalize = gdk_x11_text_list_converter_finalize;
}
static void
gdk_x11_text_list_converter_init (GdkX11TextListConverter *local)
{
}
GConverter *
gdk_x11_text_list_converter_to_utf8_new (GdkDisplay *display,
const char *encoding,
int format)
{
GdkX11TextListConverter *conv;
conv = g_object_new (GDK_TYPE_X11_TEXT_LIST_CONVERTER, NULL);
conv->display = g_object_ref (display);
conv->encoding = g_intern_string (encoding);
conv->format = format;
return G_CONVERTER (conv);
}
GConverter *
gdk_x11_text_list_converter_from_utf8_new (GdkDisplay *display,
const char *encoding,
int format)
{
GdkX11TextListConverter *conv;
conv = g_object_new (GDK_TYPE_X11_TEXT_LIST_CONVERTER, NULL);
conv->display = g_object_ref (display);
conv->encoding = g_intern_string (encoding);
conv->format = format;
conv->encoder = TRUE;
return G_CONVERTER (conv);
}