/* vim:set et ts=4 sts=4: * * ibus-libpinyin - Intelligent Pinyin engine based on libpinyin for IBus * * Copyright (c) 2008-2010 Peng Huang * * This program 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 2, or (at your option) * any later version. * * This program 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 this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ #include "PYPConfig.h" #include #include "PYBus.h" #include "PYLibPinyin.h" namespace PY { const gchar * const CONFIG_CORRECT_PINYIN = "CorrectPinyin"; const gchar * const CONFIG_FUZZY_PINYIN = "FuzzyPinyin"; const gchar * const CONFIG_ORIENTATION = "LookupTableOrientation"; const gchar * const CONFIG_PAGE_SIZE = "LookupTablePageSize"; const gchar * const CONFIG_SHIFT_SELECT_CANDIDATE = "ShiftSelectCandidate"; const gchar * const CONFIG_MINUS_EQUAL_PAGE = "MinusEqualPage"; const gchar * const CONFIG_COMMA_PERIOD_PAGE = "CommaPeriodPage"; const gchar * const CONFIG_AUTO_COMMIT = "AutoCommit"; const gchar * const CONFIG_DOUBLE_PINYIN = "DoublePinyin"; const gchar * const CONFIG_DOUBLE_PINYIN_SCHEMA = "DoublePinyinSchema"; const gchar * const CONFIG_DOUBLE_PINYIN_SHOW_RAW = "DoublePinyinShowRaw"; const gchar * const CONFIG_INIT_CHINESE = "InitChinese"; const gchar * const CONFIG_INIT_FULL = "InitFull"; const gchar * const CONFIG_INIT_FULL_PUNCT = "InitFullPunct"; const gchar * const CONFIG_INIT_SIMP_CHINESE = "InitSimplifiedChinese"; const gchar * const CONFIG_SPECIAL_PHRASES = "SpecialPhrases"; const gchar * const CONFIG_BOPOMOFO_KEYBOARD_MAPPING = "BopomofoKeyboardMapping"; const gchar * const CONFIG_SELECT_KEYS = "SelectKeys"; const gchar * const CONFIG_GUIDE_KEY = "GuideKey"; const gchar * const CONFIG_AUXILIARY_SELECT_KEY_F = "AuxiliarySelectKey_F"; const gchar * const CONFIG_AUXILIARY_SELECT_KEY_KP = "AuxiliarySelectKey_KP"; const gchar * const CONFIG_ENTER_KEY = "EnterKey"; const guint PINYIN_DEFAULT_OPTION = PINYIN_INCOMPLETE | CHEWING_INCOMPLETE| PINYIN_CORRECT_ALL| 0; std::unique_ptr LibPinyinPinyinConfig::m_instance; std::unique_ptr LibPinyinBopomofoConfig::m_instance; LibPinyinConfig::LibPinyinConfig (Bus & bus, const std::string & name) : Config (bus, name) { initDefaultValues (); g_signal_connect (get (), "value-changed", G_CALLBACK (valueChangedCallback), this); } LibPinyinConfig::~LibPinyinConfig (void) { } void LibPinyinConfig::initDefaultValues (void) { m_option = PINYIN_DEFAULT_OPTION; m_option_mask = PINYIN_INCOMPLETE | CHEWING_INCOMPLETE | PINYIN_CORRECT_ALL; m_orientation = IBUS_ORIENTATION_HORIZONTAL; m_page_size = 5; m_shift_select_candidate = FALSE; m_minus_equal_page = TRUE; m_comma_period_page = TRUE; m_auto_commit = FALSE; m_double_pinyin = FALSE; m_double_pinyin_schema = 0; m_double_pinyin_show_raw = FALSE; m_init_chinese = TRUE; m_init_full = FALSE; m_init_full_punct = TRUE; m_init_simp_chinese = TRUE; m_special_phrases = TRUE; } static const struct { const gchar * const name; guint option; } options [] = { { "IncompletePinyin", PINYIN_INCOMPLETE|CHEWING_INCOMPLETE}, /* fuzzy pinyin */ { "FuzzyPinyin_C_CH", PINYIN_AMB_C_CH }, { "FuzzyPinyin_Z_ZH", PINYIN_AMB_Z_ZH }, { "FuzzyPinyin_S_SH", PINYIN_AMB_S_SH }, { "FuzzyPinyin_L_N", PINYIN_AMB_L_N }, { "FuzzyPinyin_F_H", PINYIN_AMB_F_H }, { "FuzzyPinyin_L_R", PINYIN_AMB_L_R }, { "FuzzyPinyin_G_K", PINYIN_AMB_G_K }, { "FuzzyPinyin_AN_ANG", PINYIN_AMB_AN_ANG }, { "FuzzyPinyin_EN_ENG", PINYIN_AMB_EN_ENG }, { "FuzzyPinyin_IN_ING", PINYIN_AMB_IN_ING }, }; void LibPinyinConfig::readDefaultValues (void) { #if defined(HAVE_IBUS_CONFIG_GET_VALUES) /* read all values together */ initDefaultValues (); GVariant *values = ibus_config_get_values (get (), m_section.c_str ()); g_return_if_fail (values != NULL); GVariantIter iter; gchar *name; GVariant *value; g_variant_iter_init (&iter, values); while (g_variant_iter_next (&iter, "{sv}", &name, &value)) { valueChanged (m_section, name, value); g_free (name); g_variant_unref (value); } g_variant_unref (values); #else /* others */ m_orientation = read (CONFIG_ORIENTATION, 0); if (m_orientation != IBUS_ORIENTATION_VERTICAL && m_orientation != IBUS_ORIENTATION_HORIZONTAL) { m_orientation = IBUS_ORIENTATION_HORIZONTAL; g_warn_if_reached (); } m_page_size = read (CONFIG_PAGE_SIZE, 5); if (m_page_size > 10) { m_page_size = 5; g_warn_if_reached (); } /* fuzzy pinyin */ if (read (CONFIG_FUZZY_PINYIN, false)) m_option_mask |= PINYIN_AMB_ALL; else m_option_mask &= ~PINYIN_AMB_ALL; /* read values */ for (guint i = 0; i < G_N_ELEMENTS (options); i++) { if (read (options[i].name, (options[i].option & PINYIN_DEFAULT_OPTION) != 0)) { m_option |= options[i].option; } else { m_option &= ~options[i].option; } } #endif } static inline bool normalizeGVariant (GVariant *value, bool defval) { if (value == NULL || g_variant_classify (value) != G_VARIANT_CLASS_BOOLEAN) return defval; return g_variant_get_boolean (value); } static inline gint normalizeGVariant (GVariant *value, gint defval) { if (value == NULL || g_variant_classify (value) != G_VARIANT_CLASS_INT32) return defval; return g_variant_get_int32 (value); } static inline std::string normalizeGVariant (GVariant *value, const std::string &defval) { if (value == NULL || g_variant_classify (value) != G_VARIANT_CLASS_STRING) return defval; return g_variant_get_string (value, NULL); } gboolean LibPinyinConfig::valueChanged (const std::string §ion, const std::string &name, GVariant *value) { if (m_section != section) return FALSE; /* lookup table page size */ if (CONFIG_ORIENTATION == name) { m_orientation = normalizeGVariant (value, IBUS_ORIENTATION_HORIZONTAL); if (m_orientation != IBUS_ORIENTATION_VERTICAL && m_orientation != IBUS_ORIENTATION_HORIZONTAL) { m_orientation = IBUS_ORIENTATION_HORIZONTAL; g_warn_if_reached (); } } else if (CONFIG_PAGE_SIZE == name) { m_page_size = normalizeGVariant (value, 5); if (m_page_size > 10) { m_page_size = 5; g_warn_if_reached (); } } /* fuzzy pinyin */ else if (CONFIG_FUZZY_PINYIN == name) { if (normalizeGVariant (value, false)) m_option_mask |= PINYIN_AMB_ALL; else m_option_mask &= ~PINYIN_AMB_ALL; } else { for (guint i = 0; i < G_N_ELEMENTS (options); i++) { if (G_LIKELY (options[i].name != name)) continue; if (normalizeGVariant (value, (options[i].option & PINYIN_DEFAULT_OPTION) != 0)) m_option |= options[i].option; else m_option &= ~options[i].option; return TRUE; } return FALSE; } return TRUE; } void LibPinyinConfig::valueChangedCallback (IBusConfig *config, const gchar *section, const gchar *name, GVariant *value, LibPinyinConfig *self) { self->valueChanged (section, name, value); if (self->m_section != section) return; #ifdef IBUS_BUILD_LIBPINYIN if (self->m_section == "engine/Pinyin") LibPinyinBackEnd::instance ().setPinyinOptions (self); if (self->m_section == "engine/Bopomofo") LibPinyinBackEnd::instance ().setChewingOptions (self); #endif } static const struct { const gchar * const name; guint option; } pinyin_options [] = { /* correct */ { "CorrectPinyin_GN_NG", PINYIN_CORRECT_GN_NG }, { "CorrectPinyin_GN_NG", PINYIN_CORRECT_GN_NG }, { "CorrectPinyin_MG_NG", PINYIN_CORRECT_MG_NG }, { "CorrectPinyin_IOU_IU", PINYIN_CORRECT_IOU_IU }, { "CorrectPinyin_UEI_UI", PINYIN_CORRECT_UEI_UI }, { "CorrectPinyin_UEN_UN", PINYIN_CORRECT_UEN_UN }, { "CorrectPinyin_UE_VE", PINYIN_CORRECT_UE_VE }, { "CorrectPinyin_V_U", PINYIN_CORRECT_V_U }, { "CorrectPinyin_VE_UE", PINYIN_CORRECT_V_U }, { "CorrectPinyin_ON_ONG", PINYIN_CORRECT_ON_ONG }, }; LibPinyinPinyinConfig::LibPinyinPinyinConfig (Bus & bus) : LibPinyinConfig (bus, "Pinyin") { } void LibPinyinPinyinConfig::init (Bus & bus) { if (m_instance.get () == NULL) { m_instance.reset (new LibPinyinPinyinConfig (bus)); m_instance->readDefaultValues (); } } void LibPinyinPinyinConfig::readDefaultValues (void) { LibPinyinConfig::readDefaultValues (); #if !defined(HAVE_IBUS_CONFIG_GET_VALUES) /* double pinyin */ m_double_pinyin = read (CONFIG_DOUBLE_PINYIN, false); m_double_pinyin_schema = read (CONFIG_DOUBLE_PINYIN_SCHEMA, 0); if (m_double_pinyin_schema > DOUBLE_PINYIN_LAST) { m_double_pinyin_schema = 0; g_warn_if_reached (); } m_double_pinyin_show_raw = read (CONFIG_DOUBLE_PINYIN_SHOW_RAW, false); /* init states */ m_init_chinese = read (CONFIG_INIT_CHINESE, true); m_init_full = read (CONFIG_INIT_FULL, false); m_init_full_punct = read (CONFIG_INIT_FULL_PUNCT, true); m_init_simp_chinese = read (CONFIG_INIT_SIMP_CHINESE, true); m_special_phrases = read (CONFIG_SPECIAL_PHRASES, true); /* other */ m_shift_select_candidate = read (CONFIG_SHIFT_SELECT_CANDIDATE, false); m_minus_equal_page = read (CONFIG_MINUS_EQUAL_PAGE, true); m_comma_period_page = read (CONFIG_COMMA_PERIOD_PAGE, true); m_auto_commit = read (CONFIG_AUTO_COMMIT, false); /* correct pinyin */ if (read (CONFIG_CORRECT_PINYIN, true)) m_option_mask |= PINYIN_CORRECT_ALL; else m_option_mask &= ~PINYIN_CORRECT_ALL; /* read values */ for (guint i = 0; i < G_N_ELEMENTS (pinyin_options); i++) { if (read (pinyin_options[i].name, (pinyin_options[i].option & PINYIN_DEFAULT_OPTION) != 0)) m_option |= pinyin_options[i].option; else m_option &= ~pinyin_options[i].option; } #endif } gboolean LibPinyinPinyinConfig::valueChanged (const std::string §ion, const std::string &name, GVariant *value) { if (m_section != section) return FALSE; if (LibPinyinConfig::valueChanged (section, name, value)) return TRUE; /* double pinyin */ if (CONFIG_DOUBLE_PINYIN == name) m_double_pinyin = normalizeGVariant (value, false); else if (CONFIG_DOUBLE_PINYIN_SCHEMA == name) { m_double_pinyin_schema = normalizeGVariant (value, 0); #if 0 if (m_double_pinyin_schema > DOUBLE_PINYIN_LAST) { m_double_pinyin_schema = 0; g_warn_if_reached (); } #endif } else if (CONFIG_DOUBLE_PINYIN_SHOW_RAW == name) m_double_pinyin_show_raw = normalizeGVariant (value, false); /* init states */ else if (CONFIG_INIT_CHINESE == name) m_init_chinese = normalizeGVariant (value, true); else if (CONFIG_INIT_FULL == name) m_init_full = normalizeGVariant (value, true); else if (CONFIG_INIT_FULL_PUNCT == name) m_init_full_punct = normalizeGVariant (value, true); else if (CONFIG_INIT_SIMP_CHINESE == name) m_init_simp_chinese = normalizeGVariant (value, true); else if (CONFIG_SPECIAL_PHRASES == name) m_special_phrases = normalizeGVariant (value, true); /* others */ else if (CONFIG_SHIFT_SELECT_CANDIDATE == name) m_shift_select_candidate = normalizeGVariant (value, false); else if (CONFIG_MINUS_EQUAL_PAGE == name) m_minus_equal_page = normalizeGVariant (value, true); else if (CONFIG_COMMA_PERIOD_PAGE == name) m_comma_period_page = normalizeGVariant (value, true); else if (CONFIG_AUTO_COMMIT == name) m_auto_commit = normalizeGVariant (value, false); /* correct pinyin */ else if (CONFIG_CORRECT_PINYIN == name) { if (normalizeGVariant (value, true)) m_option_mask |= PINYIN_CORRECT_ALL; else m_option_mask &= ~PINYIN_CORRECT_ALL; } else { for (guint i = 0; i < G_N_ELEMENTS (pinyin_options); i++) { if (G_LIKELY (pinyin_options[i].name != name)) continue; if (normalizeGVariant (value, (pinyin_options[i].option & PINYIN_DEFAULT_OPTION) != 0)) m_option |= pinyin_options[i].option; else m_option &= ~pinyin_options[i].option; return TRUE; } return FALSE; } return TRUE; } LibPinyinBopomofoConfig::LibPinyinBopomofoConfig (Bus & bus) : LibPinyinConfig (bus, "Bopomofo") { } void LibPinyinBopomofoConfig::init (Bus & bus) { if (m_instance.get () == NULL) { m_instance.reset (new LibPinyinBopomofoConfig (bus)); m_instance->readDefaultValues (); } } void LibPinyinBopomofoConfig::readDefaultValues (void) { LibPinyinConfig::readDefaultValues (); #if !defined(HAVE_IBUS_CONFIG_GET_VALUES) /* init states */ m_init_chinese = read (CONFIG_INIT_CHINESE, true); m_init_full = read (CONFIG_INIT_FULL, false); m_init_full_punct = read (CONFIG_INIT_FULL_PUNCT, true); m_init_simp_chinese = read (CONFIG_INIT_SIMP_CHINESE, false); m_special_phrases = read (CONFIG_SPECIAL_PHRASES, false); m_bopomofo_keyboard_mapping = read (CONFIG_BOPOMOFO_KEYBOARD_MAPPING, 0); m_select_keys = read (CONFIG_SELECT_KEYS, 0); if (m_select_keys >= 9) m_select_keys = 0; m_guide_key = read (CONFIG_GUIDE_KEY, true); m_auxiliary_select_key_f = read (CONFIG_AUXILIARY_SELECT_KEY_F, true); m_auxiliary_select_key_kp = read (CONFIG_AUXILIARY_SELECT_KEY_KP, true); m_enter_key = read (CONFIG_ENTER_KEY, true); #endif } gboolean LibPinyinBopomofoConfig::valueChanged (const std::string §ion, const std::string &name, GVariant *value) { if (m_section != section) return FALSE; if (LibPinyinConfig::valueChanged (section, name, value)) return TRUE; /* init states */ if (CONFIG_INIT_CHINESE == name) m_init_chinese = normalizeGVariant (value, true); else if (CONFIG_INIT_FULL == name) m_init_full = normalizeGVariant (value, true); else if (CONFIG_INIT_FULL_PUNCT == name) m_init_full_punct = normalizeGVariant (value, true); else if (CONFIG_INIT_SIMP_CHINESE == name) m_init_simp_chinese = normalizeGVariant (value, false); else if (CONFIG_SPECIAL_PHRASES == name) m_special_phrases = normalizeGVariant (value, false); else if (CONFIG_BOPOMOFO_KEYBOARD_MAPPING == name) m_bopomofo_keyboard_mapping = normalizeGVariant (value, 0); else if (CONFIG_SELECT_KEYS == name) { m_select_keys = normalizeGVariant (value, 0); if (m_select_keys >= 9) m_select_keys = 0; } else if (CONFIG_GUIDE_KEY == name) m_guide_key = normalizeGVariant (value, true); else if (CONFIG_AUXILIARY_SELECT_KEY_F == name) m_auxiliary_select_key_f = normalizeGVariant (value, true); else if (CONFIG_AUXILIARY_SELECT_KEY_KP == name) m_auxiliary_select_key_kp = normalizeGVariant (value, true); else if (CONFIG_ENTER_KEY == name) m_enter_key = normalizeGVariant (value, true); else return FALSE; return TRUE; } };