forked from AuroraMiddleware/gtk
1b7cff4755
2002-01-10 Tor Lillqvist <tml@iki.fi> Apply the same fixes and improvements as to the gtk-1-3-win32-production branch: Bug fixes and cleanup of selection and DND functionality. Still doesn't work as well as the win32-production branch, though, but getting closer. After this, need to add Archaeopteryx Software's OLE2 DND support. * gdk/win32/gdkselection-win32.c (gdk_selection_owner_set, gdk_selection_send_notify, generate_selection_notify): Don't use SendMessage() to generate events for the same app, instead use gdk_event_put(). * gdk/win32/gdkprivate-win32.h * gdk/win32/gdkglobals-win32.c * gdk/win32/gdkmain-win32.c * gdk/win32/gdkevents-win32.c: Thus, remove declaration, definition, initialisation and handling of gdk_selection_notify_msg, gdk_selection_request_msg and gdk_selection_clear_msg. * gdk/win32/gdkselection-win32.c (gdk_text_property_to_text_list, gdk_free_text_list, gdk_string_to_compound_text, gdk_free_compound_text): Implement trivially, witrh a text_list always having a single element, and a compound text always consisting of just a single (UTF-8!) string. Let's see how well this works. * gdk/win32/gdkselection-win32.c (gdk_selection_convert): Fix non-ASCII paste from the clipboard: Try getting the same formats from the Windows clipboard that gdk_property_change() puts there: CF_UNICODETEXT, UTF8_STRING or CF_TEXT+CF_LOCALE. * gdk/win32/gdkproperty-win32.c (gdk_property_change): When storing text on the clipboard, handle non-ASCII text correctly. The logic is as follows: If we have only ASCII characters, use CF_TEXT. Else, if we are on NT, use CF_UNICODETEXT. Else (we are on Win9x), if all the characters are present in the code page of some installed locale, use CF_TEXT and also set CF_LOCALE to that locale. Else (still on Win9x) store as RTF. We use a very simple RTF string, just the text, no fonts or other crap, with the non-ASCII characters as Unicode \uN keywords. Additionally, also store the UTF-8 string as such, under the format "UTF8_STRING", so that GDK can also paste from the Clipboard what it has copied there. (Thus no need to implement any RTF parser.) (find_common_locale): New function, implements the search for a locale for case 3 above. * gdk/win32/gdkglobals-win32.c: New global variables compound_text, text_uri_list, utf8_string, cf_rtf and cf_utf8_string. * gdk/win32/gdkim-win32.c (_gdk_ucs2_to_utf8): New function, converts from a wchar_t string to UTF-8. (_gdk_utf8_to_ucs2): Rename from _gdk_win32_nmbstowchar_ts. (_gdk_utf8_to_wcs): Rename from gdk_nmbstowchar_ts. * gdk/win32/gdkevents-win32.c (build_keypress_event): Use _gdk_ucs2_to_utf8(). * gdk/win32/gdkselection-win32.c: Remove some unnecessary logging. * gdk/win32/gdkdnd-win32.c: Plug memory leaks, the gdk_drag_context_ref() was called unnecessarily in a couple of places, meaning drag contexts were never freed. The same memory leaks seem to be present in gdk/linux-fb/gdkselection-fb.c, BTW. (gdk_drop_reply): For WIN32_DROPFILES drops, free the temporarily stored file list. * gdk/win32/gdkselection-win32.c: Clarify the use of the sel_prop_table. Now it is used only for storing the GDK_SELECTION "properties". The file names dropped with WM_DROPFILES -style DND is stored temporarily (between the drop and the target picking them up) in a separate place. Have a separate hash table to map selection atoms to owner windows. This used to be quite mixed up. (_gdk_dropfiles_store): New function, to store the dropped file list for the drop target to possibly fetch, and clear it afterwards, from gdk_drop_reply(). (gdk_selection_owner_get): Much simplified now.
378 lines
6.7 KiB
C
378 lines
6.7 KiB
C
/* GDK - The GIMP Drawing Kit
|
|
* Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald
|
|
*
|
|
* 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, write to the
|
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
|
* Boston, MA 02111-1307, USA.
|
|
*/
|
|
|
|
/*
|
|
* Modified by the GTK+ Team and others 1997-2000. See the AUTHORS
|
|
* file for a list of people on the GTK+ Team. See the ChangeLog
|
|
* files for a list of changes. These files are distributed with
|
|
* GTK+ at ftp://ftp.gtk.org/pub/gtk/.
|
|
*/
|
|
|
|
#include <config.h>
|
|
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <locale.h>
|
|
|
|
#include "gdkpixmap.h"
|
|
#include "gdkinternals.h"
|
|
#include "gdki18n.h"
|
|
#include "gdkwin32.h"
|
|
|
|
/*
|
|
*--------------------------------------------------------------
|
|
* gdk_set_locale
|
|
*
|
|
* Arguments:
|
|
*
|
|
* Results:
|
|
*
|
|
* Side effects:
|
|
*
|
|
*--------------------------------------------------------------
|
|
*/
|
|
|
|
gchar*
|
|
gdk_set_locale (void)
|
|
{
|
|
if (!setlocale (LC_ALL, ""))
|
|
g_warning ("locale not supported by C library");
|
|
|
|
return g_win32_getlocale ();
|
|
}
|
|
|
|
/*
|
|
* gdk_wcstombs
|
|
*
|
|
* Returns a multi-byte string converted from the specified array
|
|
* of wide characters. The string is newly allocated. The array of
|
|
* wide characters must be null-terminated. If the conversion is
|
|
* failed, it returns NULL.
|
|
*
|
|
* On Win32, we always use UTF-8.
|
|
*/
|
|
gchar *
|
|
gdk_wcstombs (const GdkWChar *src)
|
|
{
|
|
gint len;
|
|
const GdkWChar *wcp;
|
|
guchar *mbstr, *bp;
|
|
|
|
wcp = src;
|
|
len = 0;
|
|
while (*wcp)
|
|
{
|
|
const GdkWChar c = *wcp++;
|
|
|
|
if (c < 0x80)
|
|
len += 1;
|
|
else if (c < 0x800)
|
|
len += 2;
|
|
else if (c < 0x10000)
|
|
len += 3;
|
|
else if (c < 0x200000)
|
|
len += 4;
|
|
else if (c < 0x4000000)
|
|
len += 5;
|
|
else
|
|
len += 6;
|
|
}
|
|
|
|
mbstr = g_malloc (len + 1);
|
|
|
|
wcp = src;
|
|
bp = mbstr;
|
|
while (*wcp)
|
|
{
|
|
int first;
|
|
GdkWChar c = *wcp++;
|
|
|
|
if (c < 0x80)
|
|
{
|
|
first = 0;
|
|
len = 1;
|
|
}
|
|
else if (c < 0x800)
|
|
{
|
|
first = 0xc0;
|
|
len = 2;
|
|
}
|
|
else if (c < 0x10000)
|
|
{
|
|
first = 0xe0;
|
|
len = 3;
|
|
}
|
|
else if (c < 0x200000)
|
|
{
|
|
first = 0xf0;
|
|
len = 4;
|
|
}
|
|
else if (c < 0x4000000)
|
|
{
|
|
first = 0xf8;
|
|
len = 5;
|
|
}
|
|
else
|
|
{
|
|
first = 0xfc;
|
|
len = 6;
|
|
}
|
|
|
|
/* Woo-hoo! */
|
|
switch (len)
|
|
{
|
|
case 6: bp[5] = (c & 0x3f) | 0x80; c >>= 6; /* Fall through */
|
|
case 5: bp[4] = (c & 0x3f) | 0x80; c >>= 6; /* Fall through */
|
|
case 4: bp[3] = (c & 0x3f) | 0x80; c >>= 6; /* Fall through */
|
|
case 3: bp[2] = (c & 0x3f) | 0x80; c >>= 6; /* Fall through */
|
|
case 2: bp[1] = (c & 0x3f) | 0x80; c >>= 6; /* Fall through */
|
|
case 1: bp[0] = c | first;
|
|
}
|
|
|
|
bp += len;
|
|
}
|
|
*bp = 0;
|
|
return mbstr;
|
|
}
|
|
|
|
/* A vesion that converts from wchar_t strings to UTF-8 */
|
|
|
|
gchar *
|
|
_gdk_ucs2_to_utf8 (const wchar_t *src,
|
|
gint src_len)
|
|
{
|
|
gint len;
|
|
const wchar_t *wcp;
|
|
guchar *mbstr, *bp;
|
|
|
|
wcp = src;
|
|
len = 0;
|
|
while (wcp < src + src_len)
|
|
{
|
|
const wchar_t c = *wcp++;
|
|
|
|
if (c < 0x80)
|
|
len += 1;
|
|
else if (c < 0x800)
|
|
len += 2;
|
|
else
|
|
len += 3;
|
|
}
|
|
|
|
mbstr = g_malloc (len + 1);
|
|
|
|
wcp = src;
|
|
bp = mbstr;
|
|
while (wcp < src + src_len)
|
|
{
|
|
int first;
|
|
wchar_t c = *wcp++;
|
|
|
|
if (c < 0x80)
|
|
{
|
|
first = 0;
|
|
len = 1;
|
|
}
|
|
else if (c < 0x800)
|
|
{
|
|
first = 0xc0;
|
|
len = 2;
|
|
}
|
|
else
|
|
{
|
|
first = 0xe0;
|
|
len = 3;
|
|
}
|
|
|
|
/* Woo-hoo! */
|
|
switch (len)
|
|
{
|
|
case 3: bp[2] = (c & 0x3f) | 0x80; c >>= 6; /* Fall through */
|
|
case 2: bp[1] = (c & 0x3f) | 0x80; c >>= 6; /* Fall through */
|
|
case 1: bp[0] = c | first;
|
|
}
|
|
|
|
bp += len;
|
|
}
|
|
*bp = 0;
|
|
|
|
return mbstr;
|
|
}
|
|
|
|
/* Convert from UTF-8 to GdkWChar */
|
|
|
|
gint
|
|
_gdk_utf8_to_wcs (GdkWChar *dest,
|
|
const gchar *src,
|
|
gint src_len,
|
|
gint dest_max)
|
|
{
|
|
guchar *cp, *end;
|
|
gint n;
|
|
|
|
cp = (guchar *) src;
|
|
end = cp + src_len;
|
|
n = 0;
|
|
while (cp != end && dest != dest + dest_max)
|
|
{
|
|
gint i, mask = 0, len;
|
|
guchar c = *cp;
|
|
|
|
if (c < 0x80)
|
|
{
|
|
len = 1;
|
|
mask = 0x7f;
|
|
}
|
|
else if ((c & 0xe0) == 0xc0)
|
|
{
|
|
len = 2;
|
|
mask = 0x1f;
|
|
}
|
|
else if ((c & 0xf0) == 0xe0)
|
|
{
|
|
len = 3;
|
|
mask = 0x0f;
|
|
}
|
|
else if ((c & 0xf8) == 0xf0)
|
|
{
|
|
len = 4;
|
|
mask = 0x07;
|
|
}
|
|
else if ((c & 0xfc) == 0xf8)
|
|
{
|
|
len = 5;
|
|
mask = 0x03;
|
|
}
|
|
else if ((c & 0xfc) == 0xfc)
|
|
{
|
|
len = 6;
|
|
mask = 0x01;
|
|
}
|
|
else
|
|
return -1;
|
|
|
|
if (cp + len > end)
|
|
return -1;
|
|
|
|
*dest = (cp[0] & mask);
|
|
for (i = 1; i < len; i++)
|
|
{
|
|
if ((cp[i] & 0xc0) != 0x80)
|
|
return -1;
|
|
*dest <<= 6;
|
|
*dest |= (cp[i] & 0x3f);
|
|
}
|
|
if (*dest == -1)
|
|
return -1;
|
|
|
|
cp += len;
|
|
dest++;
|
|
n++;
|
|
}
|
|
if (cp != end)
|
|
return -1;
|
|
|
|
return n;
|
|
}
|
|
|
|
/*
|
|
* gdk_mbstowcs
|
|
*
|
|
* Converts the specified string into GDK wide characters, and,
|
|
* returns the number of wide characters written. The string 'src'
|
|
* must be null-terminated. If the conversion is failed, it returns
|
|
* -1.
|
|
*
|
|
* On Win32, the string is assumed to be in UTF-8. Also note that
|
|
* GdkWChar is 32 bits, while wchar_t, and the wide characters the
|
|
* Windows API uses, are 16 bits!
|
|
*/
|
|
|
|
gint
|
|
gdk_mbstowcs (GdkWChar *dest,
|
|
const gchar *src,
|
|
gint dest_max)
|
|
{
|
|
return _gdk_utf8_to_wcs (dest, src, strlen (src), dest_max);
|
|
}
|
|
|
|
|
|
/* A version that converts to a wchar_t string */
|
|
|
|
gint
|
|
_gdk_utf8_to_ucs2 (wchar_t *dest,
|
|
const gchar *src,
|
|
gint src_len,
|
|
gint dest_max)
|
|
{
|
|
wchar_t *wcp;
|
|
guchar *cp, *end;
|
|
gint n;
|
|
|
|
wcp = dest;
|
|
cp = (guchar *) src;
|
|
end = cp + src_len;
|
|
n = 0;
|
|
while (cp != end && wcp != dest + dest_max)
|
|
{
|
|
gint i, mask = 0, len;
|
|
guchar c = *cp;
|
|
|
|
if (c < 0x80)
|
|
{
|
|
len = 1;
|
|
mask = 0x7f;
|
|
}
|
|
else if ((c & 0xe0) == 0xc0)
|
|
{
|
|
len = 2;
|
|
mask = 0x1f;
|
|
}
|
|
else if ((c & 0xf0) == 0xe0)
|
|
{
|
|
len = 3;
|
|
mask = 0x0f;
|
|
}
|
|
else /* Other lengths are not possible with 16-bit wchar_t! */
|
|
return -1;
|
|
|
|
if (cp + len > end)
|
|
return -1;
|
|
|
|
*wcp = (cp[0] & mask);
|
|
for (i = 1; i < len; i++)
|
|
{
|
|
if ((cp[i] & 0xc0) != 0x80)
|
|
return -1;
|
|
*wcp <<= 6;
|
|
*wcp |= (cp[i] & 0x3f);
|
|
}
|
|
if (*wcp == 0xFFFF)
|
|
return -1;
|
|
|
|
cp += len;
|
|
wcp++;
|
|
n++;
|
|
}
|
|
if (cp != end)
|
|
return -1;
|
|
|
|
return n;
|
|
}
|