aboutsummaryrefslogtreecommitdiffhomepage
diff options
context:
space:
mode:
authornyamatongwe <devnull@localhost>2003-08-17 05:24:13 +0000
committernyamatongwe <devnull@localhost>2003-08-17 05:24:13 +0000
commitf704830636e29648e5ce767bc190680b3d7c9dc6 (patch)
tree95c38b2937a1a231febcfd28158391c53f0f6556
parent62255bcb12756d04ae09d361f98c41ee9c19b289 (diff)
downloadscintilla-mirror-f704830636e29648e5ce767bc190680b3d7c9dc6.tar.gz
Added UTF-8 support for clipboard, primary selection and drag and drop.
Disabled on Win32.
-rw-r--r--gtk/ScintillaGTK.cxx238
1 files changed, 161 insertions, 77 deletions
diff --git a/gtk/ScintillaGTK.cxx b/gtk/ScintillaGTK.cxx
index 50f743f1d..913ee4f30 100644
--- a/gtk/ScintillaGTK.cxx
+++ b/gtk/ScintillaGTK.cxx
@@ -54,7 +54,9 @@
#if GTK_MAJOR_VERSION < 2
#define INTERNATIONAL_INPUT
-#else
+#endif
+
+#if !PLAT_GTK_WIN32
#include <iconv.h>
#endif
@@ -87,7 +89,10 @@ class ScintillaGTK : public ScintillaBase {
GtkWidgetClass *parentClass;
- static GdkAtom clipboard_atom;
+ GdkAtom atomClipboard;
+ GdkAtom atomSought;
+ GdkAtom atomUTF8;
+ GdkAtom atomString;
#if PLAT_GTK_WIN32
CLIPFORMAT cfColumnSelect;
@@ -145,7 +150,7 @@ private:
virtual void AddToPopUp(const char *label, int cmd = 0, bool enabled = true);
bool OwnPrimarySelection();
virtual void ClaimSelection();
- char *GetGtkSelectionText(const GtkSelectionData *selectionData, unsigned int *len, bool *isRectangular);
+ void GetGtkSelectionText(const GtkSelectionData *selectionData, SelectionText &selText);
void ReceivedSelection(GtkSelectionData *selection_data);
void ReceivedDrop(GtkSelectionData *selection_data);
void GetSelection(GtkSelectionData *selection_data, guint info, SelectionText *selected);
@@ -225,12 +230,11 @@ enum {
static gint scintilla_signals[LAST_SIGNAL] = { 0 };
static GtkWidgetClass* parent_class = NULL;
-GdkAtom ScintillaGTK::clipboard_atom = GDK_NONE;
-
enum {
TARGET_STRING,
TARGET_TEXT,
- TARGET_COMPOUND_TEXT
+ TARGET_COMPOUND_TEXT,
+ TARGET_UTF8_STRING
};
static GtkWidget *PWidget(Window &w) {
@@ -256,6 +260,11 @@ ScintillaGTK::ScintillaGTK(_ScintillaObject *sci_) :
sci = sci_;
wMain = GTK_WIDGET(sci);
+ atomClipboard = GDK_NONE;
+ atomSought = GDK_NONE;
+ atomUTF8 = GDK_NONE;
+ atomString = GDK_NONE;
+
#if PLAT_GTK_WIN32
// There does not seem to be a real standard for indicating that the clipboard
// contains a rectangular selection, so copy Developer Studio.
@@ -565,9 +574,10 @@ void ScintillaGTK::Initialise() {
gtk_widget_grab_focus(PWidget(wMain));
static const GtkTargetEntry targets[] = {
+ { "UTF8_STRING", 0, TARGET_UTF8_STRING },
{ "STRING", 0, TARGET_STRING },
- { "TEXT", 0, TARGET_TEXT },
- { "COMPOUND_TEXT", 0, TARGET_COMPOUND_TEXT },
+// { "TEXT", 0, TARGET_TEXT },
+// { "COMPOUND_TEXT", 0, TARGET_COMPOUND_TEXT },
{ "text/uri-list", 0, 0 },
};
static const gint n_targets = sizeof(targets) / sizeof(targets[0]);
@@ -575,10 +585,11 @@ void ScintillaGTK::Initialise() {
gtk_selection_add_targets(GTK_WIDGET(PWidget(wMain)), GDK_SELECTION_PRIMARY,
targets, n_targets);
- if (!clipboard_atom)
- clipboard_atom = gdk_atom_intern("CLIPBOARD", FALSE);
+ atomClipboard = gdk_atom_intern("CLIPBOARD", FALSE);
+ atomUTF8 = gdk_atom_intern("UTF8_STRING", FALSE);
+ atomString = gdk_atom_intern("STRING", FALSE);
- gtk_selection_add_targets(GTK_WIDGET(PWidget(wMain)), clipboard_atom,
+ gtk_selection_add_targets(GTK_WIDGET(PWidget(wMain)), atomClipboard,
targets, n_targets);
gtk_drag_dest_set(GTK_WIDGET(PWidget(wMain)),
@@ -860,7 +871,6 @@ void ScintillaGTK::NotifyURIDropped(const char *list) {
const char *CharacterSetID(int characterSet);
#if GTK_MAJOR_VERSION >= 2
-
#define IS_ACC(x) \
((x) >= 65103 && (x) <= 65111)
#define IS_CHAR(x) \
@@ -972,7 +982,7 @@ int ScintillaGTK::KeyDefault(int key, int modifiers) {
void ScintillaGTK::CopyToClipboard(const SelectionText &selectedText) {
copyText.Copy(selectedText.s, selectedText.len);
gtk_selection_owner_set(GTK_WIDGET(PWidget(wMain)),
- clipboard_atom,
+ atomClipboard,
GDK_CURRENT_TIME);
}
@@ -980,7 +990,7 @@ void ScintillaGTK::Copy() {
if (currentPos != anchor) {
CopySelectionRange(&copyText);
gtk_selection_owner_set(GTK_WIDGET(PWidget(wMain)),
- clipboard_atom,
+ atomClipboard,
GDK_CURRENT_TIME);
#if PLAT_GTK_WIN32
if (selType == selRectangle) {
@@ -993,9 +1003,9 @@ void ScintillaGTK::Copy() {
}
void ScintillaGTK::Paste() {
+ atomSought = atomUTF8;
gtk_selection_convert(GTK_WIDGET(PWidget(wMain)),
- clipboard_atom,
- gdk_atom_intern("STRING", FALSE), GDK_CURRENT_TIME);
+ atomClipboard, atomSought, GDK_CURRENT_TIME);
}
void ScintillaGTK::CreateCallTipWindow(PRectangle rc) {
@@ -1066,74 +1076,124 @@ void ScintillaGTK::ClaimSelection() {
}
}
-char *ScintillaGTK::GetGtkSelectionText(const GtkSelectionData *selectionData,
- unsigned int* len, bool* isRectangular) {
- char *dest;
- unsigned int i;
- const char *sptr;
- char *dptr;
+#if !PLAT_GTK_WIN32
+static char *ConvertText(size_t *lenResult, char *s, size_t len, const char *charSetDest, const char *charSetSource) {
+ *lenResult = 0;
+ char *destForm = 0;
+ iconv_t iconvh = iconv_open(charSetDest, charSetSource);
+ if (iconvh != ((iconv_t)(-1))) {
+ destForm = new char[len*3+1];
+ char *pin = s;
+ size_t inLeft = len;
+ char *pout = destForm;
+ size_t outLeft = len*3+1;
+ size_t conversions = iconv(iconvh, &pin, &inLeft, &pout, &outLeft);
+ if (conversions == ((size_t)(-1))) {
+fprintf(stderr, "iconv failed for %s\n", static_cast<char *>(s));
+ delete []destForm;
+ destForm = 0;
+ } else {
+//fprintf(stderr, "iconv OK %s %d\n", destForm, pout - destForm);
+ *pout = '\0';
+ *lenResult = pout - destForm;
+ }
+ iconv_close(iconvh);
+ } else {
+//fprintf(stderr, "Can not iconv %s %s\n", charSetDest, charSetSource);
+ }
+ if (!destForm) {
+ destForm = new char[1];
+ destForm[0] = '\0';
+ *lenResult = 0;
+ }
+ return destForm;
+}
+#endif
+
+// Detect rectangular text, convert line ends to current mode, convert from or to UTF-8
+void ScintillaGTK::GetGtkSelectionText(const GtkSelectionData *selectionData, SelectionText &selText) {
+ char *data = reinterpret_cast<char *>(selectionData->data);
+ size_t len = selectionData->length;
+ GdkAtom selectionType = selectionData->type;
// Return empty string if selection is not a string
- if (selectionData->type != GDK_TARGET_STRING) {
- dest = new char[1];
- strcpy(dest, "");
- *isRectangular = false;
- *len = 0;
- return dest;
+ if ((selectionType != GDK_TARGET_STRING) && (selectionType != atomUTF8)) {
+ char *empty = new char[1];
+ empty[0] = '\0';
+ selText.Set(empty,0);
+ return;
}
+ // Check for "\n\0" ending to string indicating that selection is rectangular
+ bool isRectangular = ((len > 2) && (data[len - 1] == 0 && data[len - 2] == '\n'));
+
// Need to convert to correct newline form for this file: win32gtk *always* returns
// only \n line delimiter from clipboard, and linux/unix gtk may also not send the
// form that matches the document (this is probably not effectively standardized by X)
- dest = new char[(2 * static_cast<unsigned int>(selectionData->length)) + 1];
- sptr = reinterpret_cast<const char *>(selectionData->data);
- dptr = dest;
- for (i = 0; i < static_cast<unsigned int>(selectionData->length) && *sptr != '\0'; i++) {
+ char *dest = new char[2 * len + 1];
+ const char *sptr = data;
+ char *dptr = dest;
+ for (size_t i = 0; (i < len) && (*sptr != '\0'); i++) {
if (*sptr == '\n' || *sptr == '\r') {
if (pdoc->eolMode == SC_EOL_CR) {
*dptr++ = '\r';
- }
- else if (pdoc->eolMode == SC_EOL_LF) {
+ } else if (pdoc->eolMode == SC_EOL_LF) {
*dptr++ = '\n';
- }
- else { // pdoc->eolMode == SC_EOL_CRLF
+ } else { // pdoc->eolMode == SC_EOL_CRLF
*dptr++ = '\r';
*dptr++ = '\n';
}
- if (*sptr == '\r' && i+1 < static_cast<unsigned int>(selectionData->length) && *(sptr+1) == '\n') {
+ if ((*sptr == '\r') && (i+1 < len) && (*(sptr+1) == '\n')) {
i++;
sptr++;
}
sptr++;
- }
- else {
+ } else {
*dptr++ = *sptr++;
}
}
*dptr++ = '\0';
-
- *len = (dptr - dest) - 1;
-
- // Check for "\n\0" ending to string indicating that selection is rectangular
-#if PLAT_GTK_WIN32
- *isRectangular = ::IsClipboardFormatAvailable(cfColumnSelect) != 0;
-#else
- *isRectangular = ((selectionData->length > 2) &&
- (selectionData->data[selectionData->length - 1] == 0 &&
- selectionData->data[selectionData->length - 2] == '\n'));
+ len = (dptr - dest) - 1;
+ selText.Set(dest, len, isRectangular);
+
+#if !PLAT_GTK_WIN32
+ // Possible character set conversion
+ const char *charSetBuffer =
+ CharacterSetID(vs.styles[STYLE_DEFAULT].characterSet);
+ if (*charSetBuffer) {
+ if (IsUnicodeMode()) {
+ if (selectionType == GDK_TARGET_STRING) {
+ // Convert to UTF-8
+//fprintf(stderr, "Convert to UTF-8 from %s\n", charSetBuffer);
+ dest = ConvertText(&len, dest, len, "UTF8", charSetBuffer);
+ selText.Set(dest, len, isRectangular);
+ }
+ } else {
+ if (selectionType == atomUTF8) {
+//fprintf(stderr, "Convert to locale %s\n", charSetBuffer);
+ // Convert to locale
+ dest = ConvertText(&len, dest, len, charSetBuffer, "UTF8");
+ selText.Set(dest, len, isRectangular);
+ }
+ }
+ }
#endif
- return dest;
}
void ScintillaGTK::ReceivedSelection(GtkSelectionData *selection_data) {
- if (selection_data->type == GDK_TARGET_STRING) {
- //Platform::DebugPrintf("Received String Selection %x %d\n", selection_data->selection, selection_data->length);
- if (((selection_data->selection == clipboard_atom) ||
- (selection_data->selection == GDK_SELECTION_PRIMARY)) &&
- (selection_data->length > 0)) {
- unsigned int len;
- bool isRectangular;
- char *ptr = GetGtkSelectionText(selection_data, &len, &isRectangular);
+ if ((selection_data->selection == atomClipboard) ||
+ (selection_data->selection == GDK_SELECTION_PRIMARY)) {
+ if ((atomSought == atomUTF8) && (selection_data->length <= 0)) {
+ atomSought = atomString;
+ gtk_selection_convert(GTK_WIDGET(PWidget(wMain)),
+ atomClipboard, atomSought, GDK_CURRENT_TIME);
+ } else if ((selection_data->length > 0) &&
+ ((selection_data->type == GDK_TARGET_STRING) || (selection_data->type == atomUTF8))) {
+ SelectionText selText;
+ GetGtkSelectionText(selection_data, selText);
+#if PLAT_GTK_WIN32
+ selText.rectangular = ::IsClipboardFormatAvailable(cfColumnSelect) != 0;
+#endif
pdoc->BeginUndoAction();
int selStart = SelectionStart();
@@ -1141,28 +1201,27 @@ void ScintillaGTK::ReceivedSelection(GtkSelectionData *selection_data) {
ClearSelection();
}
- if (isRectangular) {
- PasteRectangular(selStart, ptr, len);
+ if (selText.rectangular) {
+ PasteRectangular(selStart, selText.s, selText.len);
} else {
- pdoc->InsertString(currentPos, ptr, len);
- SetEmptySelection(currentPos + len);
+ pdoc->InsertString(currentPos, selText.s, selText.len);
+ SetEmptySelection(currentPos + selText.len);
}
pdoc->EndUndoAction();
- delete []ptr;
}
}
+// else fprintf(stderr, "Target non string %d %d\n", (int)(selection_data->type),
+// (int)(atomUTF8));
Redraw();
}
void ScintillaGTK::ReceivedDrop(GtkSelectionData *selection_data) {
dragWasDropped = true;
- if (selection_data->type == GDK_TARGET_STRING) {
+ if ((selection_data->type == GDK_TARGET_STRING) || (selection_data->type == atomUTF8)) {
if (selection_data->length > 0) {
- unsigned int len;
- bool isRectangular;
- char *ptr = GetGtkSelectionText(selection_data, &len, &isRectangular);
- DropAt(posDrop, ptr, false, isRectangular);
- delete []ptr;
+ SelectionText selText;
+ GetGtkSelectionText(selection_data, selText);
+ DropAt(posDrop, selText.s, false, selText.rectangular);
}
} else {
char *ptr = reinterpret_cast<char *>(selection_data->data);
@@ -1192,11 +1251,9 @@ void ScintillaGTK::GetSelection(GtkSelectionData *selection_data, guint info, Se
if (pdoc->eolMode == SC_EOL_CR && *sptr == '\r') {
*dptr++ = '\n';
sptr++;
- }
- else if (pdoc->eolMode != SC_EOL_CR && *sptr == '\r') {
+ } else if (pdoc->eolMode != SC_EOL_CR && *sptr == '\r') {
sptr++;
- }
- else {
+ } else {
*dptr++ = *sptr++;
}
}
@@ -1204,8 +1261,31 @@ void ScintillaGTK::GetSelection(GtkSelectionData *selection_data, guint info, Se
selBuffer = tmpstr;
#endif
- if (info == TARGET_STRING) {
- int len = strlen(selBuffer);
+ char *tmputf = 0;
+ if ((info == TARGET_UTF8_STRING) || (info == TARGET_STRING)) {
+ size_t len = strlen(selBuffer);
+#if !PLAT_GTK_WIN32
+ // Possible character set conversion
+ const char *charSetBuffer =
+ CharacterSetID(vs.styles[STYLE_DEFAULT].characterSet);
+ if (info == TARGET_UTF8_STRING) {
+ //fprintf(stderr, "Copy to clipboard as UTF-8\n");
+ if (!IsUnicodeMode()) {
+ // Convert to UTF-8
+ //fprintf(stderr, "Convert to UTF-8 from %s\n", charSetBuffer);
+ tmputf = ConvertText(&len, selBuffer, len, "UTF8", charSetBuffer);
+ selBuffer = tmputf;
+ }
+ } else if (info == TARGET_STRING) {
+ if (IsUnicodeMode()) {
+ //fprintf(stderr, "Convert to locale %s\n", charSetBuffer);
+ // Convert to locale
+ tmputf = ConvertText(&len, selBuffer, len, charSetBuffer, "UTF8");
+ selBuffer = tmputf;
+ }
+ }
+#endif
+
// Here is a somewhat evil kludge.
// As I can not work out how to store data on the clipboard in multiple formats
// and need some way to mark the clipping as being stream or rectangular,
@@ -1217,7 +1297,9 @@ void ScintillaGTK::GetSelection(GtkSelectionData *selection_data, guint info, Se
if (text->rectangular)
len++;
#endif
- gtk_selection_data_set(selection_data, GDK_SELECTION_TYPE_STRING,
+ gtk_selection_data_set(selection_data,
+ (info == TARGET_STRING) ?
+ GDK_SELECTION_TYPE_STRING : atomUTF8,
8, reinterpret_cast<unsigned char *>(selBuffer),
len);
} else if ((info == TARGET_TEXT) || (info == TARGET_COMPOUND_TEXT)) {
@@ -1232,6 +1314,7 @@ void ScintillaGTK::GetSelection(GtkSelectionData *selection_data, guint info, Se
gdk_free_compound_text(text);
}
+ delete []tmputf;
#if PLAT_GTK_WIN32
delete []tmpstr;
#endif
@@ -1356,8 +1439,9 @@ gint ScintillaGTK::PressThis(GdkEventButton *event) {
CopySelectionRange(&primary);
SetSelection(pos, pos);
+ atomSought = atomUTF8;
gtk_selection_convert(GTK_WIDGET(PWidget(wMain)), GDK_SELECTION_PRIMARY,
- gdk_atom_intern("STRING", FALSE), event->time);
+ atomSought, event->time);
} else if (event->button == 3) {
if (displayPopupMenu) {
// PopUp menu