/* Pure Gtk+-3 communication module. -*- coding: utf-8 -*- Copyright (C) 1989, 1993-1994, 2005-2006, 2008-2020 Free Software Foundation, Inc. This file is part of GNU Emacs. GNU Emacs is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. GNU Emacs 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 General Public License for more details. You should have received a copy of the GNU General Public License along with GNU Emacs. If not, see . */ /* This should be the first include, as it may set up #defines affecting interpretation of even the system includes. */ #include #include "pgtkterm.h" static void im_context_commit_cb (GtkIMContext * imc, gchar * str, gpointer user_data) { struct pgtk_display_info *dpyinfo = user_data; struct frame *f = dpyinfo->im.focused_frame; if (dpyinfo->im.context == NULL) return; if (f == NULL) return; pgtk_enqueue_string (f, str); } static gboolean im_context_retrieve_surrounding_cb (GtkIMContext * imc, gpointer user_data) { gtk_im_context_set_surrounding (imc, "", -1, 0); return TRUE; } static gboolean im_context_delete_surrounding_cb (GtkIMContext * imc, int offset, int n_chars, gpointer user_data) { return TRUE; } static Lisp_Object make_color_string (PangoAttrColor * pac) { char buf[256]; sprintf (buf, "#%02x%02x%02x", pac->color.red >> 8, pac->color.green >> 8, pac->color.blue >> 8); return build_string (buf); } static void im_context_preedit_changed_cb (GtkIMContext * imc, gpointer user_data) { struct pgtk_display_info *dpyinfo = user_data; struct frame *f = dpyinfo->im.focused_frame; char *str; PangoAttrList *attrs; int pos; if (dpyinfo->im.context == NULL) return; if (f == NULL) return; gtk_im_context_get_preedit_string (imc, &str, &attrs, &pos); /* * ( * (TEXT (ul . COLOR) (bg . COLOR) (fg . COLOR)) * ... * ) */ Lisp_Object list = Qnil; PangoAttrIterator *iter; iter = pango_attr_list_get_iterator (attrs); do { int st, ed; int has_underline = 0; Lisp_Object part = Qnil; pango_attr_iterator_range (iter, &st, &ed); if (ed > strlen (str)) ed = strlen (str); if (st >= ed) continue; Lisp_Object text = make_string (str + st, ed - st); part = Fcons (text, part); PangoAttrInt *ul = (PangoAttrInt *) pango_attr_iterator_get (iter, PANGO_ATTR_UNDERLINE); if (ul != NULL) { if (ul->value != PANGO_UNDERLINE_NONE) has_underline = 1; } PangoAttrColor *pac; if (has_underline) { pac = (PangoAttrColor *) pango_attr_iterator_get (iter, PANGO_ATTR_UNDERLINE_COLOR); if (pac != NULL) part = Fcons (Fcons (Qul, make_color_string (pac)), part); else part = Fcons (Fcons (Qul, Qt), part); } pac = (PangoAttrColor *) pango_attr_iterator_get (iter, PANGO_ATTR_FOREGROUND); if (pac != NULL) part = Fcons (Fcons (Qfg, make_color_string (pac)), part); pac = (PangoAttrColor *) pango_attr_iterator_get (iter, PANGO_ATTR_BACKGROUND); if (pac != NULL) part = Fcons (Fcons (Qbg, make_color_string (pac)), part); part = Fnreverse (part); list = Fcons (part, list); } while (pango_attr_iterator_next (iter)); list = Fnreverse (list); pgtk_enqueue_preedit (f, list); g_free (str); pango_attr_list_unref (attrs); } static void im_context_preedit_end_cb (GtkIMContext * imc, gpointer user_data) { struct pgtk_display_info *dpyinfo = user_data; struct frame *f = dpyinfo->im.focused_frame; if (dpyinfo->im.context == NULL) return; if (f == NULL) return; pgtk_enqueue_preedit (f, Qnil); } static void im_context_preedit_start_cb (GtkIMContext * imc, gpointer user_data) { } void pgtk_im_focus_in (struct frame *f) { struct pgtk_display_info *dpyinfo = FRAME_DISPLAY_INFO (f); if (dpyinfo->im.context != NULL) { gtk_im_context_reset (dpyinfo->im.context); gtk_im_context_set_client_window (dpyinfo->im.context, gtk_widget_get_window (FRAME_GTK_WIDGET (f))); gtk_im_context_focus_in (dpyinfo->im.context); } dpyinfo->im.focused_frame = f; } void pgtk_im_focus_out (struct frame *f) { struct pgtk_display_info *dpyinfo = FRAME_DISPLAY_INFO (f); if (dpyinfo->im.focused_frame == f) { if (dpyinfo->im.context != NULL) { gtk_im_context_reset (dpyinfo->im.context); gtk_im_context_focus_out (dpyinfo->im.context); gtk_im_context_set_client_window (dpyinfo->im.context, NULL); } dpyinfo->im.focused_frame = NULL; } } bool pgtk_im_filter_keypress (struct frame *f, GdkEventKey * ev) { struct pgtk_display_info *dpyinfo = FRAME_DISPLAY_INFO (f); if (dpyinfo->im.context != NULL) { if (gtk_im_context_filter_keypress (dpyinfo->im.context, ev)) return true; } return false; } void pgtk_im_set_cursor_location (struct frame *f, int x, int y, int width, int height) { struct pgtk_display_info *dpyinfo = FRAME_DISPLAY_INFO (f); if (dpyinfo->im.context != NULL && dpyinfo->im.focused_frame == f) { GdkRectangle area = { x, y, width, height }; gtk_im_context_set_cursor_location (dpyinfo->im.context, &area); } } static void pgtk_im_use_context (struct pgtk_display_info *dpyinfo, bool use_p) { if (!use_p) { if (dpyinfo->im.context != NULL) { gtk_im_context_reset (dpyinfo->im.context); gtk_im_context_focus_out (dpyinfo->im.context); gtk_im_context_set_client_window (dpyinfo->im.context, NULL); g_object_unref (dpyinfo->im.context); dpyinfo->im.context = NULL; } } else { if (dpyinfo->im.context == NULL) { dpyinfo->im.context = gtk_im_multicontext_new (); g_signal_connect (dpyinfo->im.context, "commit", G_CALLBACK (im_context_commit_cb), dpyinfo); g_signal_connect (dpyinfo->im.context, "retrieve-surrounding", G_CALLBACK (im_context_retrieve_surrounding_cb), dpyinfo); g_signal_connect (dpyinfo->im.context, "delete-surrounding", G_CALLBACK (im_context_delete_surrounding_cb), dpyinfo); g_signal_connect (dpyinfo->im.context, "preedit-changed", G_CALLBACK (im_context_preedit_changed_cb), dpyinfo); g_signal_connect (dpyinfo->im.context, "preedit-end", G_CALLBACK (im_context_preedit_end_cb), dpyinfo); g_signal_connect (dpyinfo->im.context, "preedit-start", G_CALLBACK (im_context_preedit_start_cb), dpyinfo); gtk_im_context_set_use_preedit (dpyinfo->im.context, TRUE); if (dpyinfo->im.focused_frame) pgtk_im_focus_in (dpyinfo->im.focused_frame); } } } void pgtk_im_init (struct pgtk_display_info *dpyinfo) { dpyinfo->im.context = NULL; pgtk_im_use_context (dpyinfo, !NILP (Vpgtk_use_im_context_on_new_connection)); } void pgtk_im_finish (struct pgtk_display_info *dpyinfo) { if (dpyinfo->im.context != NULL) g_object_unref (dpyinfo->im.context); dpyinfo->im.context = NULL; } DEFUN ("pgtk-use-im-context", Fpgtk_use_im_context, Spgtk_use_im_context, 1, 2, 0, doc: /* Set whether to use GtkIMContext. */) (Lisp_Object use_p, Lisp_Object terminal) { struct pgtk_display_info *dpyinfo = check_pgtk_display_info (terminal); pgtk_im_use_context (dpyinfo, !NILP (use_p)); return Qnil; } void syms_of_pgtkim (void) { defsubr (&Spgtk_use_im_context); DEFSYM (Qpgtk_refresh_preedit, "pgtk-refresh-preedit"); DEFSYM (Qul, "ul"); DEFSYM (Qfg, "fg"); DEFSYM (Qbg, "bg"); DEFVAR_LISP ("pgtk-use-im-context-on-new-connection", Vpgtk_use_im_context_on_new_connection, doc: /* Whether to use GtkIMContext on a new connection. If you want to change it after connection, use the `pgtk-use-im-context' function. */ ); Vpgtk_use_im_context_on_new_connection = Qt; }