Blame modules/input/gtkimcontextthai.c

Packit 98cdb6
/* GTK - The GIMP Toolkit
Packit 98cdb6
 *
Packit 98cdb6
 * This library is free software; you can redistribute it and/or
Packit 98cdb6
 * modify it under the terms of the GNU Lesser General Public
Packit 98cdb6
 * License as published by the Free Software Foundation; either
Packit 98cdb6
 * version 2 of the License, or (at your option) any later version.
Packit 98cdb6
 *
Packit 98cdb6
 * This library is distributed in the hope that it will be useful,
Packit 98cdb6
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 98cdb6
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 98cdb6
 * Lesser General Public License for more details.
Packit 98cdb6
 *
Packit 98cdb6
 * You should have received a copy of the GNU Lesser General Public
Packit 98cdb6
 * License along with this library; if not, write to the
Packit 98cdb6
 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
Packit 98cdb6
 * Boston, MA 02111-1307, USA.
Packit 98cdb6
 *
Packit 98cdb6
 * Author:  Theppitak Karoonboonyanan <thep@linux.thai.net>
Packit 98cdb6
 *
Packit 98cdb6
 */
Packit 98cdb6
Packit 98cdb6
#include <string.h>
Packit 98cdb6
Packit 98cdb6
#include <gdk/gdkkeysyms.h>
Packit 98cdb6
#include "gtkimcontextthai.h"
Packit 98cdb6
#include "thai-charprop.h"
Packit 98cdb6
Packit 98cdb6
static void     gtk_im_context_thai_class_init          (GtkIMContextThaiClass *class);
Packit 98cdb6
static void     gtk_im_context_thai_init                (GtkIMContextThai      *im_context_thai);
Packit 98cdb6
static gboolean gtk_im_context_thai_filter_keypress     (GtkIMContext          *context,
Packit 98cdb6
						         GdkEventKey           *key);
Packit 98cdb6
Packit 98cdb6
#ifndef GTK_IM_CONTEXT_THAI_NO_FALLBACK
Packit 98cdb6
static void     forget_previous_chars (GtkIMContextThai *context_thai);
Packit 98cdb6
static void     remember_previous_char (GtkIMContextThai *context_thai,
Packit 98cdb6
                                        gunichar new_char);
Packit 98cdb6
#endif /* !GTK_IM_CONTEXT_THAI_NO_FALLBACK */
Packit 98cdb6
Packit 98cdb6
static GObjectClass *parent_class;
Packit 98cdb6
Packit 98cdb6
GType gtk_type_im_context_thai = 0;
Packit 98cdb6
Packit 98cdb6
void
Packit 98cdb6
gtk_im_context_thai_register_type (GTypeModule *type_module)
Packit 98cdb6
{
Packit 98cdb6
  const GTypeInfo im_context_thai_info =
Packit 98cdb6
  {
Packit 98cdb6
    sizeof (GtkIMContextThaiClass),
Packit 98cdb6
    (GBaseInitFunc) NULL,
Packit 98cdb6
    (GBaseFinalizeFunc) NULL,
Packit 98cdb6
    (GClassInitFunc) gtk_im_context_thai_class_init,
Packit 98cdb6
    NULL,           /* class_finalize */    
Packit 98cdb6
    NULL,           /* class_data */
Packit 98cdb6
    sizeof (GtkIMContextThai),
Packit 98cdb6
    0,
Packit 98cdb6
    (GInstanceInitFunc) gtk_im_context_thai_init,
Packit 98cdb6
  };
Packit 98cdb6
Packit 98cdb6
  gtk_type_im_context_thai = 
Packit 98cdb6
    g_type_module_register_type (type_module,
Packit 98cdb6
                                 GTK_TYPE_IM_CONTEXT,
Packit 98cdb6
                                 "GtkIMContextThai",
Packit 98cdb6
                                 &im_context_thai_info, 0);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gtk_im_context_thai_class_init (GtkIMContextThaiClass *class)
Packit 98cdb6
{
Packit 98cdb6
  GtkIMContextClass *im_context_class = GTK_IM_CONTEXT_CLASS (class);
Packit 98cdb6
Packit 98cdb6
  parent_class = g_type_class_peek_parent (class);
Packit 98cdb6
Packit 98cdb6
  im_context_class->filter_keypress = gtk_im_context_thai_filter_keypress;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
gtk_im_context_thai_init (GtkIMContextThai *context_thai)
Packit 98cdb6
{
Packit 98cdb6
#ifndef GTK_IM_CONTEXT_THAI_NO_FALLBACK
Packit 98cdb6
  forget_previous_chars (context_thai);
Packit 98cdb6
#endif /* !GTK_IM_CONTEXT_THAI_NO_FALLBACK */
Packit 98cdb6
  context_thai->isc_mode = ISC_BASICCHECK;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
GtkIMContext *
Packit 98cdb6
gtk_im_context_thai_new (void)
Packit 98cdb6
{
Packit 98cdb6
  GtkIMContextThai *result;
Packit 98cdb6
Packit 98cdb6
  result = GTK_IM_CONTEXT_THAI (g_object_new (GTK_TYPE_IM_CONTEXT_THAI, NULL));
Packit 98cdb6
Packit 98cdb6
  return GTK_IM_CONTEXT (result);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
GtkIMContextThaiISCMode
Packit 98cdb6
gtk_im_context_thai_get_isc_mode (GtkIMContextThai *context_thai)
Packit 98cdb6
{
Packit 98cdb6
  return context_thai->isc_mode;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
GtkIMContextThaiISCMode
Packit 98cdb6
gtk_im_context_thai_set_isc_mode (GtkIMContextThai *context_thai,
Packit 98cdb6
                                  GtkIMContextThaiISCMode mode)
Packit 98cdb6
{
Packit 98cdb6
  GtkIMContextThaiISCMode prev_mode = context_thai->isc_mode;
Packit 98cdb6
  context_thai->isc_mode = mode;
Packit 98cdb6
  return prev_mode;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static gboolean
Packit 98cdb6
is_context_lost_key(guint keyval)
Packit 98cdb6
{
Packit 98cdb6
  return ((keyval & 0xFF00) == 0xFF00) &&
Packit 98cdb6
         (keyval == GDK_BackSpace ||
Packit 98cdb6
          keyval == GDK_Tab ||
Packit 98cdb6
          keyval == GDK_Linefeed ||
Packit 98cdb6
          keyval == GDK_Clear ||
Packit 98cdb6
          keyval == GDK_Return ||
Packit 98cdb6
          keyval == GDK_Pause ||
Packit 98cdb6
          keyval == GDK_Scroll_Lock ||
Packit 98cdb6
          keyval == GDK_Sys_Req ||
Packit 98cdb6
          keyval == GDK_Escape ||
Packit 98cdb6
          keyval == GDK_Delete ||
Packit 98cdb6
          (GDK_Home <= keyval && keyval <= GDK_Begin) || /* IsCursorkey */
Packit 98cdb6
          (GDK_KP_Space <= keyval && keyval <= GDK_KP_Delete) || /* IsKeypadKey, non-chars only */
Packit 98cdb6
          (GDK_Select <= keyval && keyval <= GDK_Break) || /* IsMiscFunctionKey */
Packit 98cdb6
          (GDK_F1 <= keyval && keyval <= GDK_F35)); /* IsFunctionKey */
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static gboolean
Packit 98cdb6
is_context_intact_key(guint keyval)
Packit 98cdb6
{
Packit 98cdb6
  return (((keyval & 0xFF00) == 0xFF00) &&
Packit 98cdb6
           ((GDK_Shift_L <= keyval && keyval <= GDK_Hyper_R) || /* IsModifierKey */
Packit 98cdb6
            (keyval == GDK_Mode_switch) ||
Packit 98cdb6
            (keyval == GDK_Num_Lock))) ||
Packit 98cdb6
         (((keyval & 0xFE00) == 0xFE00) &&
Packit 98cdb6
          (GDK_ISO_Lock <= keyval && keyval <= GDK_ISO_Last_Group_Lock));
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static gboolean
Packit 98cdb6
thai_is_accept (gunichar new_char, gunichar prev_char, gint isc_mode)
Packit 98cdb6
{
Packit 98cdb6
  switch (isc_mode)
Packit 98cdb6
    {
Packit 98cdb6
    case ISC_PASSTHROUGH:
Packit 98cdb6
      return TRUE;
Packit 98cdb6
Packit 98cdb6
    case ISC_BASICCHECK:
Packit 98cdb6
      return TAC_compose_input (prev_char, new_char) != 'R';
Packit 98cdb6
Packit 98cdb6
    case ISC_STRICT:
Packit 98cdb6
      {
Packit 98cdb6
        int op = TAC_compose_input (prev_char, new_char);
Packit 98cdb6
        return op != 'R' && op != 'S';
Packit 98cdb6
      }
Packit 98cdb6
Packit 98cdb6
    default:
Packit 98cdb6
      return FALSE;
Packit 98cdb6
    }
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
#define thai_is_composible(n,p)  (TAC_compose_input ((p), (n)) == 'C')
Packit 98cdb6
Packit 98cdb6
#ifndef GTK_IM_CONTEXT_THAI_NO_FALLBACK
Packit 98cdb6
static void
Packit 98cdb6
forget_previous_chars (GtkIMContextThai *context_thai)
Packit 98cdb6
{
Packit 98cdb6
  memset (context_thai->char_buff, 0, sizeof (context_thai->char_buff));
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static void
Packit 98cdb6
remember_previous_char (GtkIMContextThai *context_thai, gunichar new_char)
Packit 98cdb6
{
Packit 98cdb6
  memmove (context_thai->char_buff + 1, context_thai->char_buff,
Packit 98cdb6
           (GTK_IM_CONTEXT_THAI_BUFF_SIZE - 1) * sizeof (context_thai->char_buff[0]));
Packit 98cdb6
  context_thai->char_buff[0] = new_char;
Packit 98cdb6
}
Packit 98cdb6
#endif /* !GTK_IM_CONTEXT_THAI_NO_FALLBACK */
Packit 98cdb6
Packit 98cdb6
static gunichar
Packit 98cdb6
get_previous_char (GtkIMContextThai *context_thai, gint offset)
Packit 98cdb6
{
Packit 98cdb6
  gchar *surrounding;
Packit 98cdb6
  gint  cursor_index;
Packit 98cdb6
Packit 98cdb6
  if (gtk_im_context_get_surrounding ((GtkIMContext *)context_thai,
Packit 98cdb6
                                      &surrounding, &cursor_index))
Packit 98cdb6
    {
Packit 98cdb6
      gunichar prev_char;
Packit 98cdb6
      gchar *p, *q;
Packit 98cdb6
Packit 98cdb6
      prev_char = 0;
Packit 98cdb6
      p = surrounding + cursor_index;
Packit 98cdb6
      for (q = p; offset < 0 && q > surrounding; ++offset)
Packit 98cdb6
        q = g_utf8_prev_char (q);
Packit 98cdb6
      if (offset == 0)
Packit 98cdb6
        {
Packit 98cdb6
          prev_char = g_utf8_get_char_validated (q, p - q);
Packit 98cdb6
          if (prev_char < 0)
Packit 98cdb6
            prev_char = 0;
Packit 98cdb6
        }
Packit 98cdb6
      g_free (surrounding);
Packit 98cdb6
      return prev_char;
Packit 98cdb6
    }
Packit 98cdb6
#ifndef GTK_IM_CONTEXT_THAI_NO_FALLBACK
Packit 98cdb6
  else
Packit 98cdb6
    {
Packit 98cdb6
      offset = -offset - 1;
Packit 98cdb6
      if (0 <= offset && offset < GTK_IM_CONTEXT_THAI_BUFF_SIZE)
Packit 98cdb6
        return context_thai->char_buff[offset];
Packit 98cdb6
    }
Packit 98cdb6
#endif /* !GTK_IM_CONTEXT_THAI_NO_FALLBACK */
Packit 98cdb6
Packit 98cdb6
    return 0;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static gboolean
Packit 98cdb6
gtk_im_context_thai_commit_chars (GtkIMContextThai *context_thai,
Packit 98cdb6
                                  gunichar *s, gsize len)
Packit 98cdb6
{
Packit 98cdb6
  gchar *utf8;
Packit 98cdb6
Packit 98cdb6
  utf8 = g_ucs4_to_utf8 (s, len, NULL, NULL, NULL);
Packit 98cdb6
  if (!utf8)
Packit 98cdb6
    return FALSE;
Packit 98cdb6
Packit 98cdb6
  g_signal_emit_by_name (context_thai, "commit", utf8);
Packit 98cdb6
Packit 98cdb6
  g_free (utf8);
Packit 98cdb6
  return TRUE;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static gboolean
Packit 98cdb6
accept_input (GtkIMContextThai *context_thai, gunichar new_char)
Packit 98cdb6
{
Packit 98cdb6
#ifndef GTK_IM_CONTEXT_THAI_NO_FALLBACK
Packit 98cdb6
  remember_previous_char (context_thai, new_char);
Packit 98cdb6
#endif /* !GTK_IM_CONTEXT_THAI_NO_FALLBACK */
Packit 98cdb6
Packit 98cdb6
  return gtk_im_context_thai_commit_chars (context_thai, &new_char, 1);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static gboolean
Packit 98cdb6
reorder_input (GtkIMContextThai *context_thai,
Packit 98cdb6
               gunichar prev_char, gunichar new_char)
Packit 98cdb6
{
Packit 98cdb6
  gunichar buf[2];
Packit 98cdb6
Packit 98cdb6
  if (!gtk_im_context_delete_surrounding (GTK_IM_CONTEXT (context_thai), -1, 1))
Packit 98cdb6
    return FALSE;
Packit 98cdb6
Packit 98cdb6
#ifndef GTK_IM_CONTEXT_THAI_NO_FALLBACK
Packit 98cdb6
  forget_previous_chars (context_thai);
Packit 98cdb6
  remember_previous_char (context_thai, new_char);
Packit 98cdb6
  remember_previous_char (context_thai, prev_char);
Packit 98cdb6
#endif /* !GTK_IM_CONTEXT_THAI_NO_FALLBACK */
Packit 98cdb6
Packit 98cdb6
  buf[0] = new_char;
Packit 98cdb6
  buf[1] = prev_char;
Packit 98cdb6
  return gtk_im_context_thai_commit_chars (context_thai, buf, 2);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static gboolean
Packit 98cdb6
replace_input (GtkIMContextThai *context_thai, gunichar new_char)
Packit 98cdb6
{
Packit 98cdb6
  if (!gtk_im_context_delete_surrounding (GTK_IM_CONTEXT (context_thai), -1, 1))
Packit 98cdb6
    return FALSE;
Packit 98cdb6
Packit 98cdb6
#ifndef GTK_IM_CONTEXT_THAI_NO_FALLBACK
Packit 98cdb6
  forget_previous_chars (context_thai);
Packit 98cdb6
  remember_previous_char (context_thai, new_char);
Packit 98cdb6
#endif /* !GTK_IM_CONTEXT_THAI_NO_FALLBACK */
Packit 98cdb6
Packit 98cdb6
  return gtk_im_context_thai_commit_chars (context_thai, &new_char, 1);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static gboolean
Packit 98cdb6
gtk_im_context_thai_filter_keypress (GtkIMContext *context,
Packit 98cdb6
                                     GdkEventKey  *event)
Packit 98cdb6
{
Packit 98cdb6
  GtkIMContextThai *context_thai = GTK_IM_CONTEXT_THAI (context);
Packit 98cdb6
  gunichar prev_char, new_char;
Packit 98cdb6
  gboolean is_reject;
Packit 98cdb6
  GtkIMContextThaiISCMode isc_mode;
Packit 98cdb6
Packit 98cdb6
  if (event->type != GDK_KEY_PRESS)
Packit 98cdb6
    return FALSE;
Packit 98cdb6
Packit 98cdb6
  if (event->state & (GDK_MODIFIER_MASK
Packit 98cdb6
                      & ~(GDK_SHIFT_MASK | GDK_LOCK_MASK | GDK_MOD2_MASK)) ||
Packit 98cdb6
      is_context_lost_key (event->keyval))
Packit 98cdb6
    {
Packit 98cdb6
#ifndef GTK_IM_CONTEXT_THAI_NO_FALLBACK
Packit 98cdb6
      forget_previous_chars (context_thai);
Packit 98cdb6
#endif /* !GTK_IM_CONTEXT_THAI_NO_FALLBACK */
Packit 98cdb6
      return FALSE;
Packit 98cdb6
    }
Packit 98cdb6
  if (event->keyval == 0 || is_context_intact_key (event->keyval))
Packit 98cdb6
    {
Packit 98cdb6
      return FALSE;
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  prev_char = get_previous_char (context_thai, -1);
Packit 98cdb6
  if (!prev_char)
Packit 98cdb6
    prev_char = ' ';
Packit 98cdb6
  new_char = gdk_keyval_to_unicode (event->keyval);
Packit 98cdb6
  is_reject = TRUE;
Packit 98cdb6
  isc_mode = gtk_im_context_thai_get_isc_mode (context_thai);
Packit 98cdb6
  if (thai_is_accept (new_char, prev_char, isc_mode))
Packit 98cdb6
    {
Packit 98cdb6
      accept_input (context_thai, new_char);
Packit 98cdb6
      is_reject = FALSE;
Packit 98cdb6
    }
Packit 98cdb6
  else
Packit 98cdb6
    {
Packit 98cdb6
      gunichar context_char;
Packit 98cdb6
Packit 98cdb6
      /* rejected, trying to correct */
Packit 98cdb6
      context_char = get_previous_char (context_thai, -2);
Packit 98cdb6
      if (context_char)
Packit 98cdb6
        {
Packit 98cdb6
          if (thai_is_composible (new_char, context_char))
Packit 98cdb6
            {
Packit 98cdb6
              if (thai_is_composible (prev_char, new_char))
Packit 98cdb6
                is_reject = !reorder_input (context_thai, prev_char, new_char);
Packit 98cdb6
              else if (thai_is_composible (prev_char, context_char))
Packit 98cdb6
                is_reject = !replace_input (context_thai, new_char);
Packit 98cdb6
              else if ((TAC_char_class (prev_char) == FV1
Packit 98cdb6
                        || TAC_char_class (prev_char) == AM)
Packit 98cdb6
                       && TAC_char_class (new_char) == TONE)
Packit 98cdb6
                is_reject = !reorder_input (context_thai, prev_char, new_char);
Packit 98cdb6
            }
Packit 98cdb6
	  else if (thai_is_accept (new_char, context_char, isc_mode))
Packit 98cdb6
            is_reject = !replace_input (context_thai, new_char);
Packit 98cdb6
        }
Packit 98cdb6
    }
Packit 98cdb6
  if (is_reject)
Packit 98cdb6
    {
Packit 98cdb6
      /* reject character */
Packit 98cdb6
      gdk_beep ();
Packit 98cdb6
    }
Packit 98cdb6
  return TRUE;
Packit 98cdb6
}
Packit 98cdb6