6975 lines
276 KiB
Diff
6975 lines
276 KiB
Diff
diff -Nru ibus-table-1.9.18.orig/Makefile.am ibus-table-1.9.18/Makefile.am
|
||
--- ibus-table-1.9.18.orig/Makefile.am 2020-07-22 11:52:11.640532230 +0200
|
||
+++ ibus-table-1.9.18/Makefile.am 2020-07-22 14:43:51.905260956 +0200
|
||
@@ -30,6 +30,7 @@
|
||
data \
|
||
po \
|
||
setup \
|
||
+ tests \
|
||
$(NULL)
|
||
|
||
ACLOCAL_AMFLAGS = -I m4
|
||
diff -Nru ibus-table-1.9.18.orig/Makefile.in ibus-table-1.9.18/Makefile.in
|
||
--- ibus-table-1.9.18.orig/Makefile.in 2017-08-02 11:32:47.000000000 +0200
|
||
+++ ibus-table-1.9.18/Makefile.in 2020-07-22 16:15:15.492860836 +0200
|
||
@@ -1,7 +1,7 @@
|
||
-# Makefile.in generated by automake 1.15 from Makefile.am.
|
||
+# Makefile.in generated by automake 1.16.1 from Makefile.am.
|
||
# @configure_input@
|
||
|
||
-# Copyright (C) 1994-2014 Free Software Foundation, Inc.
|
||
+# Copyright (C) 1994-2018 Free Software Foundation, Inc.
|
||
|
||
# This Makefile.in is free software; the Free Software Foundation
|
||
# gives unlimited permission to copy and/or distribute it,
|
||
@@ -168,7 +168,7 @@
|
||
$(RECURSIVE_CLEAN_TARGETS) \
|
||
$(am__extra_recursive_targets)
|
||
AM_RECURSIVE_TARGETS = $(am__recursive_targets:-recursive=) TAGS CTAGS \
|
||
- cscope distdir dist dist-all distcheck
|
||
+ cscope distdir distdir-am dist dist-all distcheck
|
||
am__tagged_files = $(HEADERS) $(SOURCES) $(TAGS_FILES) $(LISP)
|
||
# Read a list of newline-separated strings from the standard input,
|
||
# and print each of them once, without duplicates. Input order is
|
||
@@ -193,7 +193,7 @@
|
||
am__DIST_COMMON = $(srcdir)/Makefile.in $(srcdir)/ibus-table.pc.in \
|
||
$(srcdir)/ibus-table.spec.in ABOUT-NLS AUTHORS COPYING \
|
||
ChangeLog INSTALL NEWS README compile config.guess \
|
||
- config.rpath config.sub install-sh missing
|
||
+ config.rpath config.sub install-sh missing py-compile
|
||
DISTFILES = $(DIST_COMMON) $(DIST_SOURCES) $(TEXINFOS) $(EXTRA_DIST)
|
||
distdir = $(PACKAGE)-$(VERSION)
|
||
top_distdir = $(distdir)
|
||
@@ -392,6 +392,7 @@
|
||
data \
|
||
po \
|
||
setup \
|
||
+ tests \
|
||
$(NULL)
|
||
|
||
ACLOCAL_AMFLAGS = -I m4
|
||
@@ -457,8 +458,8 @@
|
||
echo ' $(SHELL) ./config.status'; \
|
||
$(SHELL) ./config.status;; \
|
||
*) \
|
||
- echo ' cd $(top_builddir) && $(SHELL) ./config.status $@ $(am__depfiles_maybe)'; \
|
||
- cd $(top_builddir) && $(SHELL) ./config.status $@ $(am__depfiles_maybe);; \
|
||
+ echo ' cd $(top_builddir) && $(SHELL) ./config.status $@ $(am__maybe_remake_depfiles)'; \
|
||
+ cd $(top_builddir) && $(SHELL) ./config.status $@ $(am__maybe_remake_depfiles);; \
|
||
esac;
|
||
|
||
$(top_builddir)/config.status: $(top_srcdir)/configure $(CONFIG_STATUS_DEPENDENCIES)
|
||
@@ -622,7 +623,10 @@
|
||
-rm -f TAGS ID GTAGS GRTAGS GSYMS GPATH tags
|
||
-rm -f cscope.out cscope.in.out cscope.po.out cscope.files
|
||
|
||
-distdir: $(DISTFILES)
|
||
+distdir: $(BUILT_SOURCES)
|
||
+ $(MAKE) $(AM_MAKEFLAGS) distdir-am
|
||
+
|
||
+distdir-am: $(DISTFILES)
|
||
$(am__remove_distdir)
|
||
test -d "$(distdir)" || mkdir "$(distdir)"
|
||
@srcdirstrip=`echo "$(srcdir)" | sed 's/[].[^$$\\*]/\\\\&/g'`; \
|
||
diff -Nru ibus-table-1.9.18.orig/configure.ac ibus-table-1.9.18/configure.ac
|
||
diff -Nru ibus-table-1.9.18.orig/tests/Makefile.in ibus-table-1.9.18/tests/Makefile.in
|
||
--- ibus-table-1.9.18.orig/tests/Makefile.in 1970-01-01 01:00:00.000000000 +0100
|
||
+++ ibus-table-1.9.18/tests/Makefile.in 2020-07-22 16:28:37.394963243 +0200
|
||
@@ -0,0 +1,853 @@
|
||
+# Makefile.in generated by automake 1.16.1 from Makefile.am.
|
||
+# @configure_input@
|
||
+
|
||
+# Copyright (C) 1994-2018 Free Software Foundation, Inc.
|
||
+
|
||
+# This Makefile.in is free software; the Free Software Foundation
|
||
+# gives unlimited permission to copy and/or distribute it,
|
||
+# with or without modifications, as long as this notice is preserved.
|
||
+
|
||
+# This program is distributed in the hope that it will be useful,
|
||
+# but WITHOUT ANY WARRANTY, to the extent permitted by law; without
|
||
+# even the implied warranty of MERCHANTABILITY or FITNESS FOR A
|
||
+# PARTICULAR PURPOSE.
|
||
+
|
||
+@SET_MAKE@
|
||
+
|
||
+# vim:set noet ts=4
|
||
+#
|
||
+# ibus-table - The Tables engine for IBus
|
||
+#
|
||
+# Copyright (c) 2018 Mike FABIAN <mfabian@redhat.com>
|
||
+#
|
||
+# 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.1 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
||
+#
|
||
+VPATH = @srcdir@
|
||
+am__is_gnu_make = { \
|
||
+ if test -z '$(MAKELEVEL)'; then \
|
||
+ false; \
|
||
+ elif test -n '$(MAKE_HOST)'; then \
|
||
+ true; \
|
||
+ elif test -n '$(MAKE_VERSION)' && test -n '$(CURDIR)'; then \
|
||
+ true; \
|
||
+ else \
|
||
+ false; \
|
||
+ fi; \
|
||
+}
|
||
+am__make_running_with_option = \
|
||
+ case $${target_option-} in \
|
||
+ ?) ;; \
|
||
+ *) echo "am__make_running_with_option: internal error: invalid" \
|
||
+ "target option '$${target_option-}' specified" >&2; \
|
||
+ exit 1;; \
|
||
+ esac; \
|
||
+ has_opt=no; \
|
||
+ sane_makeflags=$$MAKEFLAGS; \
|
||
+ if $(am__is_gnu_make); then \
|
||
+ sane_makeflags=$$MFLAGS; \
|
||
+ else \
|
||
+ case $$MAKEFLAGS in \
|
||
+ *\\[\ \ ]*) \
|
||
+ bs=\\; \
|
||
+ sane_makeflags=`printf '%s\n' "$$MAKEFLAGS" \
|
||
+ | sed "s/$$bs$$bs[$$bs $$bs ]*//g"`;; \
|
||
+ esac; \
|
||
+ fi; \
|
||
+ skip_next=no; \
|
||
+ strip_trailopt () \
|
||
+ { \
|
||
+ flg=`printf '%s\n' "$$flg" | sed "s/$$1.*$$//"`; \
|
||
+ }; \
|
||
+ for flg in $$sane_makeflags; do \
|
||
+ test $$skip_next = yes && { skip_next=no; continue; }; \
|
||
+ case $$flg in \
|
||
+ *=*|--*) continue;; \
|
||
+ -*I) strip_trailopt 'I'; skip_next=yes;; \
|
||
+ -*I?*) strip_trailopt 'I';; \
|
||
+ -*O) strip_trailopt 'O'; skip_next=yes;; \
|
||
+ -*O?*) strip_trailopt 'O';; \
|
||
+ -*l) strip_trailopt 'l'; skip_next=yes;; \
|
||
+ -*l?*) strip_trailopt 'l';; \
|
||
+ -[dEDm]) skip_next=yes;; \
|
||
+ -[JT]) skip_next=yes;; \
|
||
+ esac; \
|
||
+ case $$flg in \
|
||
+ *$$target_option*) has_opt=yes; break;; \
|
||
+ esac; \
|
||
+ done; \
|
||
+ test $$has_opt = yes
|
||
+am__make_dryrun = (target_option=n; $(am__make_running_with_option))
|
||
+am__make_keepgoing = (target_option=k; $(am__make_running_with_option))
|
||
+pkgdatadir = $(datadir)/@PACKAGE@
|
||
+pkgincludedir = $(includedir)/@PACKAGE@
|
||
+pkglibdir = $(libdir)/@PACKAGE@
|
||
+pkglibexecdir = $(libexecdir)/@PACKAGE@
|
||
+am__cd = CDPATH="$${ZSH_VERSION+.}$(PATH_SEPARATOR)" && cd
|
||
+install_sh_DATA = $(install_sh) -c -m 644
|
||
+install_sh_PROGRAM = $(install_sh) -c
|
||
+install_sh_SCRIPT = $(install_sh) -c
|
||
+INSTALL_HEADER = $(INSTALL_DATA)
|
||
+transform = $(program_transform_name)
|
||
+NORMAL_INSTALL = :
|
||
+PRE_INSTALL = :
|
||
+POST_INSTALL = :
|
||
+NORMAL_UNINSTALL = :
|
||
+PRE_UNINSTALL = :
|
||
+POST_UNINSTALL = :
|
||
+build_triplet = @build@
|
||
+host_triplet = @host@
|
||
+subdir = tests
|
||
+ACLOCAL_M4 = $(top_srcdir)/aclocal.m4
|
||
+am__aclocal_m4_deps = $(top_srcdir)/m4/gettext.m4 \
|
||
+ $(top_srcdir)/m4/iconv.m4 $(top_srcdir)/m4/lib-ld.m4 \
|
||
+ $(top_srcdir)/m4/lib-link.m4 $(top_srcdir)/m4/lib-prefix.m4 \
|
||
+ $(top_srcdir)/m4/nls.m4 $(top_srcdir)/m4/po.m4 \
|
||
+ $(top_srcdir)/m4/progtest.m4 $(top_srcdir)/configure.ac
|
||
+am__configure_deps = $(am__aclocal_m4_deps) $(CONFIGURE_DEPENDENCIES) \
|
||
+ $(ACLOCAL_M4)
|
||
+DIST_COMMON = $(srcdir)/Makefile.am $(am__DIST_COMMON)
|
||
+mkinstalldirs = $(install_sh) -d
|
||
+CONFIG_CLEAN_FILES =
|
||
+CONFIG_CLEAN_VPATH_FILES =
|
||
+AM_V_P = $(am__v_P_@AM_V@)
|
||
+am__v_P_ = $(am__v_P_@AM_DEFAULT_V@)
|
||
+am__v_P_0 = false
|
||
+am__v_P_1 = :
|
||
+AM_V_GEN = $(am__v_GEN_@AM_V@)
|
||
+am__v_GEN_ = $(am__v_GEN_@AM_DEFAULT_V@)
|
||
+am__v_GEN_0 = @echo " GEN " $@;
|
||
+am__v_GEN_1 =
|
||
+AM_V_at = $(am__v_at_@AM_V@)
|
||
+am__v_at_ = $(am__v_at_@AM_DEFAULT_V@)
|
||
+am__v_at_0 = @
|
||
+am__v_at_1 =
|
||
+SOURCES =
|
||
+DIST_SOURCES =
|
||
+am__can_run_installinfo = \
|
||
+ case $$AM_UPDATE_INFO_DIR in \
|
||
+ n|no|NO) false;; \
|
||
+ *) (install-info --version) >/dev/null 2>&1;; \
|
||
+ esac
|
||
+am__tagged_files = $(HEADERS) $(SOURCES) $(TAGS_FILES) $(LISP)
|
||
+am__tty_colors_dummy = \
|
||
+ mgn= red= grn= lgn= blu= brg= std=; \
|
||
+ am__color_tests=no
|
||
+am__tty_colors = { \
|
||
+ $(am__tty_colors_dummy); \
|
||
+ if test "X$(AM_COLOR_TESTS)" = Xno; then \
|
||
+ am__color_tests=no; \
|
||
+ elif test "X$(AM_COLOR_TESTS)" = Xalways; then \
|
||
+ am__color_tests=yes; \
|
||
+ elif test "X$$TERM" != Xdumb && { test -t 1; } 2>/dev/null; then \
|
||
+ am__color_tests=yes; \
|
||
+ fi; \
|
||
+ if test $$am__color_tests = yes; then \
|
||
+ red='[0;31m'; \
|
||
+ grn='[0;32m'; \
|
||
+ lgn='[1;32m'; \
|
||
+ blu='[1;34m'; \
|
||
+ mgn='[0;35m'; \
|
||
+ brg='[1m'; \
|
||
+ std='[m'; \
|
||
+ fi; \
|
||
+}
|
||
+am__vpath_adj_setup = srcdirstrip=`echo "$(srcdir)" | sed 's|.|.|g'`;
|
||
+am__vpath_adj = case $$p in \
|
||
+ $(srcdir)/*) f=`echo "$$p" | sed "s|^$$srcdirstrip/||"`;; \
|
||
+ *) f=$$p;; \
|
||
+ esac;
|
||
+am__strip_dir = f=`echo $$p | sed -e 's|^.*/||'`;
|
||
+am__install_max = 40
|
||
+am__nobase_strip_setup = \
|
||
+ srcdirstrip=`echo "$(srcdir)" | sed 's/[].[^$$\\*|]/\\\\&/g'`
|
||
+am__nobase_strip = \
|
||
+ for p in $$list; do echo "$$p"; done | sed -e "s|$$srcdirstrip/||"
|
||
+am__nobase_list = $(am__nobase_strip_setup); \
|
||
+ for p in $$list; do echo "$$p $$p"; done | \
|
||
+ sed "s| $$srcdirstrip/| |;"' / .*\//!s/ .*/ ./; s,\( .*\)/[^/]*$$,\1,' | \
|
||
+ $(AWK) 'BEGIN { files["."] = "" } { files[$$2] = files[$$2] " " $$1; \
|
||
+ if (++n[$$2] == $(am__install_max)) \
|
||
+ { print $$2, files[$$2]; n[$$2] = 0; files[$$2] = "" } } \
|
||
+ END { for (dir in files) print dir, files[dir] }'
|
||
+am__base_list = \
|
||
+ sed '$$!N;$$!N;$$!N;$$!N;$$!N;$$!N;$$!N;s/\n/ /g' | \
|
||
+ sed '$$!N;$$!N;$$!N;$$!N;s/\n/ /g'
|
||
+am__uninstall_files_from_dir = { \
|
||
+ test -z "$$files" \
|
||
+ || { test ! -d "$$dir" && test ! -f "$$dir" && test ! -r "$$dir"; } \
|
||
+ || { echo " ( cd '$$dir' && rm -f" $$files ")"; \
|
||
+ $(am__cd) "$$dir" && rm -f $$files; }; \
|
||
+ }
|
||
+am__recheck_rx = ^[ ]*:recheck:[ ]*
|
||
+am__global_test_result_rx = ^[ ]*:global-test-result:[ ]*
|
||
+am__copy_in_global_log_rx = ^[ ]*:copy-in-global-log:[ ]*
|
||
+# A command that, given a newline-separated list of test names on the
|
||
+# standard input, print the name of the tests that are to be re-run
|
||
+# upon "make recheck".
|
||
+am__list_recheck_tests = $(AWK) '{ \
|
||
+ recheck = 1; \
|
||
+ while ((rc = (getline line < ($$0 ".trs"))) != 0) \
|
||
+ { \
|
||
+ if (rc < 0) \
|
||
+ { \
|
||
+ if ((getline line2 < ($$0 ".log")) < 0) \
|
||
+ recheck = 0; \
|
||
+ break; \
|
||
+ } \
|
||
+ else if (line ~ /$(am__recheck_rx)[nN][Oo]/) \
|
||
+ { \
|
||
+ recheck = 0; \
|
||
+ break; \
|
||
+ } \
|
||
+ else if (line ~ /$(am__recheck_rx)[yY][eE][sS]/) \
|
||
+ { \
|
||
+ break; \
|
||
+ } \
|
||
+ }; \
|
||
+ if (recheck) \
|
||
+ print $$0; \
|
||
+ close ($$0 ".trs"); \
|
||
+ close ($$0 ".log"); \
|
||
+}'
|
||
+# A command that, given a newline-separated list of test names on the
|
||
+# standard input, create the global log from their .trs and .log files.
|
||
+am__create_global_log = $(AWK) ' \
|
||
+function fatal(msg) \
|
||
+{ \
|
||
+ print "fatal: making $@: " msg | "cat >&2"; \
|
||
+ exit 1; \
|
||
+} \
|
||
+function rst_section(header) \
|
||
+{ \
|
||
+ print header; \
|
||
+ len = length(header); \
|
||
+ for (i = 1; i <= len; i = i + 1) \
|
||
+ printf "="; \
|
||
+ printf "\n\n"; \
|
||
+} \
|
||
+{ \
|
||
+ copy_in_global_log = 1; \
|
||
+ global_test_result = "RUN"; \
|
||
+ while ((rc = (getline line < ($$0 ".trs"))) != 0) \
|
||
+ { \
|
||
+ if (rc < 0) \
|
||
+ fatal("failed to read from " $$0 ".trs"); \
|
||
+ if (line ~ /$(am__global_test_result_rx)/) \
|
||
+ { \
|
||
+ sub("$(am__global_test_result_rx)", "", line); \
|
||
+ sub("[ ]*$$", "", line); \
|
||
+ global_test_result = line; \
|
||
+ } \
|
||
+ else if (line ~ /$(am__copy_in_global_log_rx)[nN][oO]/) \
|
||
+ copy_in_global_log = 0; \
|
||
+ }; \
|
||
+ if (copy_in_global_log) \
|
||
+ { \
|
||
+ rst_section(global_test_result ": " $$0); \
|
||
+ while ((rc = (getline line < ($$0 ".log"))) != 0) \
|
||
+ { \
|
||
+ if (rc < 0) \
|
||
+ fatal("failed to read from " $$0 ".log"); \
|
||
+ print line; \
|
||
+ }; \
|
||
+ printf "\n"; \
|
||
+ }; \
|
||
+ close ($$0 ".trs"); \
|
||
+ close ($$0 ".log"); \
|
||
+}'
|
||
+# Restructured Text title.
|
||
+am__rst_title = { sed 's/.*/ & /;h;s/./=/g;p;x;s/ *$$//;p;g' && echo; }
|
||
+# Solaris 10 'make', and several other traditional 'make' implementations,
|
||
+# pass "-e" to $(SHELL), and POSIX 2008 even requires this. Work around it
|
||
+# by disabling -e (using the XSI extension "set +e") if it's set.
|
||
+am__sh_e_setup = case $$- in *e*) set +e;; esac
|
||
+# Default flags passed to test drivers.
|
||
+am__common_driver_flags = \
|
||
+ --color-tests "$$am__color_tests" \
|
||
+ --enable-hard-errors "$$am__enable_hard_errors" \
|
||
+ --expect-failure "$$am__expect_failure"
|
||
+# To be inserted before the command running the test. Creates the
|
||
+# directory for the log if needed. Stores in $dir the directory
|
||
+# containing $f, in $tst the test, in $log the log. Executes the
|
||
+# developer- defined test setup AM_TESTS_ENVIRONMENT (if any), and
|
||
+# passes TESTS_ENVIRONMENT. Set up options for the wrapper that
|
||
+# will run the test scripts (or their associated LOG_COMPILER, if
|
||
+# thy have one).
|
||
+am__check_pre = \
|
||
+$(am__sh_e_setup); \
|
||
+$(am__vpath_adj_setup) $(am__vpath_adj) \
|
||
+$(am__tty_colors); \
|
||
+srcdir=$(srcdir); export srcdir; \
|
||
+case "$@" in \
|
||
+ */*) am__odir=`echo "./$@" | sed 's|/[^/]*$$||'`;; \
|
||
+ *) am__odir=.;; \
|
||
+esac; \
|
||
+test "x$$am__odir" = x"." || test -d "$$am__odir" \
|
||
+ || $(MKDIR_P) "$$am__odir" || exit $$?; \
|
||
+if test -f "./$$f"; then dir=./; \
|
||
+elif test -f "$$f"; then dir=; \
|
||
+else dir="$(srcdir)/"; fi; \
|
||
+tst=$$dir$$f; log='$@'; \
|
||
+if test -n '$(DISABLE_HARD_ERRORS)'; then \
|
||
+ am__enable_hard_errors=no; \
|
||
+else \
|
||
+ am__enable_hard_errors=yes; \
|
||
+fi; \
|
||
+case " $(XFAIL_TESTS) " in \
|
||
+ *[\ \ ]$$f[\ \ ]* | *[\ \ ]$$dir$$f[\ \ ]*) \
|
||
+ am__expect_failure=yes;; \
|
||
+ *) \
|
||
+ am__expect_failure=no;; \
|
||
+esac; \
|
||
+$(AM_TESTS_ENVIRONMENT) $(TESTS_ENVIRONMENT)
|
||
+# A shell command to get the names of the tests scripts with any registered
|
||
+# extension removed (i.e., equivalently, the names of the test logs, with
|
||
+# the '.log' extension removed). The result is saved in the shell variable
|
||
+# '$bases'. This honors runtime overriding of TESTS and TEST_LOGS. Sadly,
|
||
+# we cannot use something simpler, involving e.g., "$(TEST_LOGS:.log=)",
|
||
+# since that might cause problem with VPATH rewrites for suffix-less tests.
|
||
+# See also 'test-harness-vpath-rewrite.sh' and 'test-trs-basic.sh'.
|
||
+am__set_TESTS_bases = \
|
||
+ bases='$(TEST_LOGS)'; \
|
||
+ bases=`for i in $$bases; do echo $$i; done | sed 's/\.log$$//'`; \
|
||
+ bases=`echo $$bases`
|
||
+RECHECK_LOGS = $(TEST_LOGS)
|
||
+AM_RECURSIVE_TARGETS = check recheck
|
||
+TEST_SUITE_LOG = test-suite.log
|
||
+TEST_EXTENSIONS = @EXEEXT@ .test
|
||
+LOG_DRIVER = $(SHELL) $(top_srcdir)/test-driver
|
||
+LOG_COMPILE = $(LOG_COMPILER) $(AM_LOG_FLAGS) $(LOG_FLAGS)
|
||
+am__set_b = \
|
||
+ case '$@' in \
|
||
+ */*) \
|
||
+ case '$*' in \
|
||
+ */*) b='$*';; \
|
||
+ *) b=`echo '$@' | sed 's/\.log$$//'`; \
|
||
+ esac;; \
|
||
+ *) \
|
||
+ b='$*';; \
|
||
+ esac
|
||
+am__test_logs1 = $(TESTS:=.log)
|
||
+am__test_logs2 = $(am__test_logs1:@EXEEXT@.log=.log)
|
||
+TEST_LOGS = $(am__test_logs2:.test.log=.log)
|
||
+TEST_LOG_DRIVER = $(SHELL) $(top_srcdir)/test-driver
|
||
+TEST_LOG_COMPILE = $(TEST_LOG_COMPILER) $(AM_TEST_LOG_FLAGS) \
|
||
+ $(TEST_LOG_FLAGS)
|
||
+am__DIST_COMMON = $(srcdir)/Makefile.in $(top_srcdir)/test-driver
|
||
+DISTFILES = $(DIST_COMMON) $(DIST_SOURCES) $(TEXINFOS) $(EXTRA_DIST)
|
||
+ACLOCAL = @ACLOCAL@
|
||
+AMTAR = @AMTAR@
|
||
+AM_DEFAULT_VERBOSITY = @AM_DEFAULT_VERBOSITY@
|
||
+AUTOCONF = @AUTOCONF@
|
||
+AUTOHEADER = @AUTOHEADER@
|
||
+AUTOMAKE = @AUTOMAKE@
|
||
+AWK = @AWK@
|
||
+CC = @CC@
|
||
+CCDEPMODE = @CCDEPMODE@
|
||
+CFLAGS = @CFLAGS@
|
||
+CPPFLAGS = @CPPFLAGS@
|
||
+CYGPATH_W = @CYGPATH_W@
|
||
+DEFS = @DEFS@
|
||
+DEPDIR = @DEPDIR@
|
||
+ECHO_C = @ECHO_C@
|
||
+ECHO_N = @ECHO_N@
|
||
+ECHO_T = @ECHO_T@
|
||
+EXEEXT = @EXEEXT@
|
||
+GETTEXT_PACKAGE = @GETTEXT_PACKAGE@
|
||
+GMSGFMT = @GMSGFMT@
|
||
+GMSGFMT_015 = @GMSGFMT_015@
|
||
+IBUS_CFLAGS = @IBUS_CFLAGS@
|
||
+IBUS_LIBS = @IBUS_LIBS@
|
||
+INSTALL = @INSTALL@
|
||
+INSTALL_DATA = @INSTALL_DATA@
|
||
+INSTALL_PROGRAM = @INSTALL_PROGRAM@
|
||
+INSTALL_SCRIPT = @INSTALL_SCRIPT@
|
||
+INSTALL_STRIP_PROGRAM = @INSTALL_STRIP_PROGRAM@
|
||
+INTLLIBS = @INTLLIBS@
|
||
+INTL_MACOSX_LIBS = @INTL_MACOSX_LIBS@
|
||
+LDFLAGS = @LDFLAGS@
|
||
+LIBICONV = @LIBICONV@
|
||
+LIBINTL = @LIBINTL@
|
||
+LIBOBJS = @LIBOBJS@
|
||
+LIBS = @LIBS@
|
||
+LTLIBICONV = @LTLIBICONV@
|
||
+LTLIBINTL = @LTLIBINTL@
|
||
+LTLIBOBJS = @LTLIBOBJS@
|
||
+MAINT = @MAINT@
|
||
+MAKEINFO = @MAKEINFO@
|
||
+MKDIR_P = @MKDIR_P@
|
||
+MSGFMT = @MSGFMT@
|
||
+MSGFMT_015 = @MSGFMT_015@
|
||
+MSGMERGE = @MSGMERGE@
|
||
+OBJEXT = @OBJEXT@
|
||
+PACKAGE = @PACKAGE@
|
||
+PACKAGE_BUGREPORT = @PACKAGE_BUGREPORT@
|
||
+PACKAGE_NAME = @PACKAGE_NAME@
|
||
+PACKAGE_STRING = @PACKAGE_STRING@
|
||
+PACKAGE_TARNAME = @PACKAGE_TARNAME@
|
||
+PACKAGE_URL = @PACKAGE_URL@
|
||
+PACKAGE_VERSION = @PACKAGE_VERSION@
|
||
+PATH_SEPARATOR = @PATH_SEPARATOR@
|
||
+PKG_CONFIG = @PKG_CONFIG@
|
||
+PKG_CONFIG_LIBDIR = @PKG_CONFIG_LIBDIR@
|
||
+PKG_CONFIG_PATH = @PKG_CONFIG_PATH@
|
||
+POSUB = @POSUB@
|
||
+PYTHON = @PYTHON@
|
||
+PYTHON_EXEC_PREFIX = @PYTHON_EXEC_PREFIX@
|
||
+PYTHON_PLATFORM = @PYTHON_PLATFORM@
|
||
+PYTHON_PREFIX = @PYTHON_PREFIX@
|
||
+PYTHON_VERSION = @PYTHON_VERSION@
|
||
+SET_MAKE = @SET_MAKE@
|
||
+SHELL = @SHELL@
|
||
+STRIP = @STRIP@
|
||
+USE_NLS = @USE_NLS@
|
||
+VERSION = @VERSION@
|
||
+XGETTEXT = @XGETTEXT@
|
||
+XGETTEXT_015 = @XGETTEXT_015@
|
||
+abs_builddir = @abs_builddir@
|
||
+abs_srcdir = @abs_srcdir@
|
||
+abs_top_builddir = @abs_top_builddir@
|
||
+abs_top_srcdir = @abs_top_srcdir@
|
||
+ac_ct_CC = @ac_ct_CC@
|
||
+am__include = @am__include@
|
||
+am__leading_dot = @am__leading_dot@
|
||
+am__quote = @am__quote@
|
||
+am__tar = @am__tar@
|
||
+am__untar = @am__untar@
|
||
+bindir = @bindir@
|
||
+build = @build@
|
||
+build_alias = @build_alias@
|
||
+build_cpu = @build_cpu@
|
||
+build_os = @build_os@
|
||
+build_vendor = @build_vendor@
|
||
+builddir = @builddir@
|
||
+datadir = @datadir@
|
||
+datarootdir = @datarootdir@
|
||
+docdir = @docdir@
|
||
+dvidir = @dvidir@
|
||
+exec_prefix = @exec_prefix@
|
||
+host = @host@
|
||
+host_alias = @host_alias@
|
||
+host_cpu = @host_cpu@
|
||
+host_os = @host_os@
|
||
+host_vendor = @host_vendor@
|
||
+htmldir = @htmldir@
|
||
+includedir = @includedir@
|
||
+infodir = @infodir@
|
||
+install_sh = @install_sh@
|
||
+libdir = @libdir@
|
||
+libexecdir = @libexecdir@
|
||
+localedir = @localedir@
|
||
+localstatedir = @localstatedir@
|
||
+mandir = @mandir@
|
||
+mkdir_p = @mkdir_p@
|
||
+oldincludedir = @oldincludedir@
|
||
+pdfdir = @pdfdir@
|
||
+pkgpyexecdir = @pkgpyexecdir@
|
||
+pkgpythondir = @pkgpythondir@
|
||
+prefix = @prefix@
|
||
+program_transform_name = @program_transform_name@
|
||
+psdir = @psdir@
|
||
+pyexecdir = @pyexecdir@
|
||
+pythondir = @pythondir@
|
||
+sbindir = @sbindir@
|
||
+sharedstatedir = @sharedstatedir@
|
||
+srcdir = @srcdir@
|
||
+sysconfdir = @sysconfdir@
|
||
+target_alias = @target_alias@
|
||
+top_build_prefix = @top_build_prefix@
|
||
+top_builddir = @top_builddir@
|
||
+top_srcdir = @top_srcdir@
|
||
+TESTS = run_tests
|
||
+EXTRA_DIST = \
|
||
+ run_tests.in \
|
||
+ test_it.py \
|
||
+ __init__.py \
|
||
+ $(NULL)
|
||
+
|
||
+CLEANFILES = \
|
||
+ run_tests \
|
||
+ $(NULL)
|
||
+
|
||
+MAINTAINERCLEANFILES = \
|
||
+ Makefile.in \
|
||
+ $(NULL)
|
||
+
|
||
+all: all-am
|
||
+
|
||
+.SUFFIXES:
|
||
+.SUFFIXES: .log .test .test$(EXEEXT) .trs
|
||
+$(srcdir)/Makefile.in: @MAINTAINER_MODE_TRUE@ $(srcdir)/Makefile.am $(am__configure_deps)
|
||
+ @for dep in $?; do \
|
||
+ case '$(am__configure_deps)' in \
|
||
+ *$$dep*) \
|
||
+ ( cd $(top_builddir) && $(MAKE) $(AM_MAKEFLAGS) am--refresh ) \
|
||
+ && { if test -f $@; then exit 0; else break; fi; }; \
|
||
+ exit 1;; \
|
||
+ esac; \
|
||
+ done; \
|
||
+ echo ' cd $(top_srcdir) && $(AUTOMAKE) --gnu tests/Makefile'; \
|
||
+ $(am__cd) $(top_srcdir) && \
|
||
+ $(AUTOMAKE) --gnu tests/Makefile
|
||
+Makefile: $(srcdir)/Makefile.in $(top_builddir)/config.status
|
||
+ @case '$?' in \
|
||
+ *config.status*) \
|
||
+ cd $(top_builddir) && $(MAKE) $(AM_MAKEFLAGS) am--refresh;; \
|
||
+ *) \
|
||
+ echo ' cd $(top_builddir) && $(SHELL) ./config.status $(subdir)/$@ $(am__maybe_remake_depfiles)'; \
|
||
+ cd $(top_builddir) && $(SHELL) ./config.status $(subdir)/$@ $(am__maybe_remake_depfiles);; \
|
||
+ esac;
|
||
+
|
||
+$(top_builddir)/config.status: $(top_srcdir)/configure $(CONFIG_STATUS_DEPENDENCIES)
|
||
+ cd $(top_builddir) && $(MAKE) $(AM_MAKEFLAGS) am--refresh
|
||
+
|
||
+$(top_srcdir)/configure: @MAINTAINER_MODE_TRUE@ $(am__configure_deps)
|
||
+ cd $(top_builddir) && $(MAKE) $(AM_MAKEFLAGS) am--refresh
|
||
+$(ACLOCAL_M4): @MAINTAINER_MODE_TRUE@ $(am__aclocal_m4_deps)
|
||
+ cd $(top_builddir) && $(MAKE) $(AM_MAKEFLAGS) am--refresh
|
||
+$(am__aclocal_m4_deps):
|
||
+tags TAGS:
|
||
+
|
||
+ctags CTAGS:
|
||
+
|
||
+cscope cscopelist:
|
||
+
|
||
+
|
||
+# Recover from deleted '.trs' file; this should ensure that
|
||
+# "rm -f foo.log; make foo.trs" re-run 'foo.test', and re-create
|
||
+# both 'foo.log' and 'foo.trs'. Break the recipe in two subshells
|
||
+# to avoid problems with "make -n".
|
||
+.log.trs:
|
||
+ rm -f $< $@
|
||
+ $(MAKE) $(AM_MAKEFLAGS) $<
|
||
+
|
||
+# Leading 'am--fnord' is there to ensure the list of targets does not
|
||
+# expand to empty, as could happen e.g. with make check TESTS=''.
|
||
+am--fnord $(TEST_LOGS) $(TEST_LOGS:.log=.trs): $(am__force_recheck)
|
||
+am--force-recheck:
|
||
+ @:
|
||
+
|
||
+$(TEST_SUITE_LOG): $(TEST_LOGS)
|
||
+ @$(am__set_TESTS_bases); \
|
||
+ am__f_ok () { test -f "$$1" && test -r "$$1"; }; \
|
||
+ redo_bases=`for i in $$bases; do \
|
||
+ am__f_ok $$i.trs && am__f_ok $$i.log || echo $$i; \
|
||
+ done`; \
|
||
+ if test -n "$$redo_bases"; then \
|
||
+ redo_logs=`for i in $$redo_bases; do echo $$i.log; done`; \
|
||
+ redo_results=`for i in $$redo_bases; do echo $$i.trs; done`; \
|
||
+ if $(am__make_dryrun); then :; else \
|
||
+ rm -f $$redo_logs && rm -f $$redo_results || exit 1; \
|
||
+ fi; \
|
||
+ fi; \
|
||
+ if test -n "$$am__remaking_logs"; then \
|
||
+ echo "fatal: making $(TEST_SUITE_LOG): possible infinite" \
|
||
+ "recursion detected" >&2; \
|
||
+ elif test -n "$$redo_logs"; then \
|
||
+ am__remaking_logs=yes $(MAKE) $(AM_MAKEFLAGS) $$redo_logs; \
|
||
+ fi; \
|
||
+ if $(am__make_dryrun); then :; else \
|
||
+ st=0; \
|
||
+ errmsg="fatal: making $(TEST_SUITE_LOG): failed to create"; \
|
||
+ for i in $$redo_bases; do \
|
||
+ test -f $$i.trs && test -r $$i.trs \
|
||
+ || { echo "$$errmsg $$i.trs" >&2; st=1; }; \
|
||
+ test -f $$i.log && test -r $$i.log \
|
||
+ || { echo "$$errmsg $$i.log" >&2; st=1; }; \
|
||
+ done; \
|
||
+ test $$st -eq 0 || exit 1; \
|
||
+ fi
|
||
+ @$(am__sh_e_setup); $(am__tty_colors); $(am__set_TESTS_bases); \
|
||
+ ws='[ ]'; \
|
||
+ results=`for b in $$bases; do echo $$b.trs; done`; \
|
||
+ test -n "$$results" || results=/dev/null; \
|
||
+ all=` grep "^$$ws*:test-result:" $$results | wc -l`; \
|
||
+ pass=` grep "^$$ws*:test-result:$$ws*PASS" $$results | wc -l`; \
|
||
+ fail=` grep "^$$ws*:test-result:$$ws*FAIL" $$results | wc -l`; \
|
||
+ skip=` grep "^$$ws*:test-result:$$ws*SKIP" $$results | wc -l`; \
|
||
+ xfail=`grep "^$$ws*:test-result:$$ws*XFAIL" $$results | wc -l`; \
|
||
+ xpass=`grep "^$$ws*:test-result:$$ws*XPASS" $$results | wc -l`; \
|
||
+ error=`grep "^$$ws*:test-result:$$ws*ERROR" $$results | wc -l`; \
|
||
+ if test `expr $$fail + $$xpass + $$error` -eq 0; then \
|
||
+ success=true; \
|
||
+ else \
|
||
+ success=false; \
|
||
+ fi; \
|
||
+ br='==================='; br=$$br$$br$$br$$br; \
|
||
+ result_count () \
|
||
+ { \
|
||
+ if test x"$$1" = x"--maybe-color"; then \
|
||
+ maybe_colorize=yes; \
|
||
+ elif test x"$$1" = x"--no-color"; then \
|
||
+ maybe_colorize=no; \
|
||
+ else \
|
||
+ echo "$@: invalid 'result_count' usage" >&2; exit 4; \
|
||
+ fi; \
|
||
+ shift; \
|
||
+ desc=$$1 count=$$2; \
|
||
+ if test $$maybe_colorize = yes && test $$count -gt 0; then \
|
||
+ color_start=$$3 color_end=$$std; \
|
||
+ else \
|
||
+ color_start= color_end=; \
|
||
+ fi; \
|
||
+ echo "$${color_start}# $$desc $$count$${color_end}"; \
|
||
+ }; \
|
||
+ create_testsuite_report () \
|
||
+ { \
|
||
+ result_count $$1 "TOTAL:" $$all "$$brg"; \
|
||
+ result_count $$1 "PASS: " $$pass "$$grn"; \
|
||
+ result_count $$1 "SKIP: " $$skip "$$blu"; \
|
||
+ result_count $$1 "XFAIL:" $$xfail "$$lgn"; \
|
||
+ result_count $$1 "FAIL: " $$fail "$$red"; \
|
||
+ result_count $$1 "XPASS:" $$xpass "$$red"; \
|
||
+ result_count $$1 "ERROR:" $$error "$$mgn"; \
|
||
+ }; \
|
||
+ { \
|
||
+ echo "$(PACKAGE_STRING): $(subdir)/$(TEST_SUITE_LOG)" | \
|
||
+ $(am__rst_title); \
|
||
+ create_testsuite_report --no-color; \
|
||
+ echo; \
|
||
+ echo ".. contents:: :depth: 2"; \
|
||
+ echo; \
|
||
+ for b in $$bases; do echo $$b; done \
|
||
+ | $(am__create_global_log); \
|
||
+ } >$(TEST_SUITE_LOG).tmp || exit 1; \
|
||
+ mv $(TEST_SUITE_LOG).tmp $(TEST_SUITE_LOG); \
|
||
+ if $$success; then \
|
||
+ col="$$grn"; \
|
||
+ else \
|
||
+ col="$$red"; \
|
||
+ test x"$$VERBOSE" = x || cat $(TEST_SUITE_LOG); \
|
||
+ fi; \
|
||
+ echo "$${col}$$br$${std}"; \
|
||
+ echo "$${col}Testsuite summary for $(PACKAGE_STRING)$${std}"; \
|
||
+ echo "$${col}$$br$${std}"; \
|
||
+ create_testsuite_report --maybe-color; \
|
||
+ echo "$$col$$br$$std"; \
|
||
+ if $$success; then :; else \
|
||
+ echo "$${col}See $(subdir)/$(TEST_SUITE_LOG)$${std}"; \
|
||
+ if test -n "$(PACKAGE_BUGREPORT)"; then \
|
||
+ echo "$${col}Please report to $(PACKAGE_BUGREPORT)$${std}"; \
|
||
+ fi; \
|
||
+ echo "$$col$$br$$std"; \
|
||
+ fi; \
|
||
+ $$success || exit 1
|
||
+
|
||
+check-TESTS:
|
||
+ @list='$(RECHECK_LOGS)'; test -z "$$list" || rm -f $$list
|
||
+ @list='$(RECHECK_LOGS:.log=.trs)'; test -z "$$list" || rm -f $$list
|
||
+ @test -z "$(TEST_SUITE_LOG)" || rm -f $(TEST_SUITE_LOG)
|
||
+ @set +e; $(am__set_TESTS_bases); \
|
||
+ log_list=`for i in $$bases; do echo $$i.log; done`; \
|
||
+ trs_list=`for i in $$bases; do echo $$i.trs; done`; \
|
||
+ log_list=`echo $$log_list`; trs_list=`echo $$trs_list`; \
|
||
+ $(MAKE) $(AM_MAKEFLAGS) $(TEST_SUITE_LOG) TEST_LOGS="$$log_list"; \
|
||
+ exit $$?;
|
||
+recheck: all
|
||
+ @test -z "$(TEST_SUITE_LOG)" || rm -f $(TEST_SUITE_LOG)
|
||
+ @set +e; $(am__set_TESTS_bases); \
|
||
+ bases=`for i in $$bases; do echo $$i; done \
|
||
+ | $(am__list_recheck_tests)` || exit 1; \
|
||
+ log_list=`for i in $$bases; do echo $$i.log; done`; \
|
||
+ log_list=`echo $$log_list`; \
|
||
+ $(MAKE) $(AM_MAKEFLAGS) $(TEST_SUITE_LOG) \
|
||
+ am__force_recheck=am--force-recheck \
|
||
+ TEST_LOGS="$$log_list"; \
|
||
+ exit $$?
|
||
+run_tests.log: run_tests
|
||
+ @p='run_tests'; \
|
||
+ b='run_tests'; \
|
||
+ $(am__check_pre) $(LOG_DRIVER) --test-name "$$f" \
|
||
+ --log-file $$b.log --trs-file $$b.trs \
|
||
+ $(am__common_driver_flags) $(AM_LOG_DRIVER_FLAGS) $(LOG_DRIVER_FLAGS) -- $(LOG_COMPILE) \
|
||
+ "$$tst" $(AM_TESTS_FD_REDIRECT)
|
||
+.test.log:
|
||
+ @p='$<'; \
|
||
+ $(am__set_b); \
|
||
+ $(am__check_pre) $(TEST_LOG_DRIVER) --test-name "$$f" \
|
||
+ --log-file $$b.log --trs-file $$b.trs \
|
||
+ $(am__common_driver_flags) $(AM_TEST_LOG_DRIVER_FLAGS) $(TEST_LOG_DRIVER_FLAGS) -- $(TEST_LOG_COMPILE) \
|
||
+ "$$tst" $(AM_TESTS_FD_REDIRECT)
|
||
+@am__EXEEXT_TRUE@.test$(EXEEXT).log:
|
||
+@am__EXEEXT_TRUE@ @p='$<'; \
|
||
+@am__EXEEXT_TRUE@ $(am__set_b); \
|
||
+@am__EXEEXT_TRUE@ $(am__check_pre) $(TEST_LOG_DRIVER) --test-name "$$f" \
|
||
+@am__EXEEXT_TRUE@ --log-file $$b.log --trs-file $$b.trs \
|
||
+@am__EXEEXT_TRUE@ $(am__common_driver_flags) $(AM_TEST_LOG_DRIVER_FLAGS) $(TEST_LOG_DRIVER_FLAGS) -- $(TEST_LOG_COMPILE) \
|
||
+@am__EXEEXT_TRUE@ "$$tst" $(AM_TESTS_FD_REDIRECT)
|
||
+
|
||
+distdir: $(BUILT_SOURCES)
|
||
+ $(MAKE) $(AM_MAKEFLAGS) distdir-am
|
||
+
|
||
+distdir-am: $(DISTFILES)
|
||
+ @srcdirstrip=`echo "$(srcdir)" | sed 's/[].[^$$\\*]/\\\\&/g'`; \
|
||
+ topsrcdirstrip=`echo "$(top_srcdir)" | sed 's/[].[^$$\\*]/\\\\&/g'`; \
|
||
+ list='$(DISTFILES)'; \
|
||
+ dist_files=`for file in $$list; do echo $$file; done | \
|
||
+ sed -e "s|^$$srcdirstrip/||;t" \
|
||
+ -e "s|^$$topsrcdirstrip/|$(top_builddir)/|;t"`; \
|
||
+ case $$dist_files in \
|
||
+ */*) $(MKDIR_P) `echo "$$dist_files" | \
|
||
+ sed '/\//!d;s|^|$(distdir)/|;s,/[^/]*$$,,' | \
|
||
+ sort -u` ;; \
|
||
+ esac; \
|
||
+ for file in $$dist_files; do \
|
||
+ if test -f $$file || test -d $$file; then d=.; else d=$(srcdir); fi; \
|
||
+ if test -d $$d/$$file; then \
|
||
+ dir=`echo "/$$file" | sed -e 's,/[^/]*$$,,'`; \
|
||
+ if test -d "$(distdir)/$$file"; then \
|
||
+ find "$(distdir)/$$file" -type d ! -perm -700 -exec chmod u+rwx {} \;; \
|
||
+ fi; \
|
||
+ if test -d $(srcdir)/$$file && test $$d != $(srcdir); then \
|
||
+ cp -fpR $(srcdir)/$$file "$(distdir)$$dir" || exit 1; \
|
||
+ find "$(distdir)/$$file" -type d ! -perm -700 -exec chmod u+rwx {} \;; \
|
||
+ fi; \
|
||
+ cp -fpR $$d/$$file "$(distdir)$$dir" || exit 1; \
|
||
+ else \
|
||
+ test -f "$(distdir)/$$file" \
|
||
+ || cp -p $$d/$$file "$(distdir)/$$file" \
|
||
+ || exit 1; \
|
||
+ fi; \
|
||
+ done
|
||
+check-am: all-am
|
||
+ $(MAKE) $(AM_MAKEFLAGS) check-TESTS
|
||
+check: check-am
|
||
+all-am: Makefile
|
||
+installdirs:
|
||
+install: install-am
|
||
+install-exec: install-exec-am
|
||
+install-data: install-data-am
|
||
+uninstall: uninstall-am
|
||
+
|
||
+install-am: all-am
|
||
+ @$(MAKE) $(AM_MAKEFLAGS) install-exec-am install-data-am
|
||
+
|
||
+installcheck: installcheck-am
|
||
+install-strip:
|
||
+ if test -z '$(STRIP)'; then \
|
||
+ $(MAKE) $(AM_MAKEFLAGS) INSTALL_PROGRAM="$(INSTALL_STRIP_PROGRAM)" \
|
||
+ install_sh_PROGRAM="$(INSTALL_STRIP_PROGRAM)" INSTALL_STRIP_FLAG=-s \
|
||
+ install; \
|
||
+ else \
|
||
+ $(MAKE) $(AM_MAKEFLAGS) INSTALL_PROGRAM="$(INSTALL_STRIP_PROGRAM)" \
|
||
+ install_sh_PROGRAM="$(INSTALL_STRIP_PROGRAM)" INSTALL_STRIP_FLAG=-s \
|
||
+ "INSTALL_PROGRAM_ENV=STRIPPROG='$(STRIP)'" install; \
|
||
+ fi
|
||
+mostlyclean-generic:
|
||
+ -test -z "$(TEST_LOGS)" || rm -f $(TEST_LOGS)
|
||
+ -test -z "$(TEST_LOGS:.log=.trs)" || rm -f $(TEST_LOGS:.log=.trs)
|
||
+ -test -z "$(TEST_SUITE_LOG)" || rm -f $(TEST_SUITE_LOG)
|
||
+
|
||
+clean-generic:
|
||
+ -test -z "$(CLEANFILES)" || rm -f $(CLEANFILES)
|
||
+
|
||
+distclean-generic:
|
||
+ -test -z "$(CONFIG_CLEAN_FILES)" || rm -f $(CONFIG_CLEAN_FILES)
|
||
+ -test . = "$(srcdir)" || test -z "$(CONFIG_CLEAN_VPATH_FILES)" || rm -f $(CONFIG_CLEAN_VPATH_FILES)
|
||
+
|
||
+maintainer-clean-generic:
|
||
+ @echo "This command is intended for maintainers to use"
|
||
+ @echo "it deletes files that may require special tools to rebuild."
|
||
+ -test -z "$(MAINTAINERCLEANFILES)" || rm -f $(MAINTAINERCLEANFILES)
|
||
+clean: clean-am
|
||
+
|
||
+clean-am: clean-generic mostlyclean-am
|
||
+
|
||
+distclean: distclean-am
|
||
+ -rm -f Makefile
|
||
+distclean-am: clean-am distclean-generic
|
||
+
|
||
+dvi: dvi-am
|
||
+
|
||
+dvi-am:
|
||
+
|
||
+html: html-am
|
||
+
|
||
+html-am:
|
||
+
|
||
+info: info-am
|
||
+
|
||
+info-am:
|
||
+
|
||
+install-data-am:
|
||
+
|
||
+install-dvi: install-dvi-am
|
||
+
|
||
+install-dvi-am:
|
||
+
|
||
+install-exec-am:
|
||
+
|
||
+install-html: install-html-am
|
||
+
|
||
+install-html-am:
|
||
+
|
||
+install-info: install-info-am
|
||
+
|
||
+install-info-am:
|
||
+
|
||
+install-man:
|
||
+
|
||
+install-pdf: install-pdf-am
|
||
+
|
||
+install-pdf-am:
|
||
+
|
||
+install-ps: install-ps-am
|
||
+
|
||
+install-ps-am:
|
||
+
|
||
+installcheck-am:
|
||
+
|
||
+maintainer-clean: maintainer-clean-am
|
||
+ -rm -f Makefile
|
||
+maintainer-clean-am: distclean-am maintainer-clean-generic
|
||
+
|
||
+mostlyclean: mostlyclean-am
|
||
+
|
||
+mostlyclean-am: mostlyclean-generic
|
||
+
|
||
+pdf: pdf-am
|
||
+
|
||
+pdf-am:
|
||
+
|
||
+ps: ps-am
|
||
+
|
||
+ps-am:
|
||
+
|
||
+uninstall-am:
|
||
+
|
||
+.MAKE: check-am install-am install-strip
|
||
+
|
||
+.PHONY: all all-am check check-TESTS check-am clean clean-generic \
|
||
+ cscopelist-am ctags-am distclean distclean-generic distdir dvi \
|
||
+ dvi-am html html-am info info-am install install-am \
|
||
+ install-data install-data-am install-dvi install-dvi-am \
|
||
+ install-exec install-exec-am install-html install-html-am \
|
||
+ install-info install-info-am install-man install-pdf \
|
||
+ install-pdf-am install-ps install-ps-am install-strip \
|
||
+ installcheck installcheck-am installdirs maintainer-clean \
|
||
+ maintainer-clean-generic mostlyclean mostlyclean-generic pdf \
|
||
+ pdf-am ps ps-am recheck tags-am uninstall uninstall-am
|
||
+
|
||
+.PRECIOUS: Makefile
|
||
+
|
||
+
|
||
+run_tests: run_tests.in
|
||
+ sed -e 's&@PYTHON_BIN@&$(PYTHON)&g' \
|
||
+ -e 's&@SRCDIR@&$(srcdir)&g' $< > $@
|
||
+ chmod +x $@
|
||
+
|
||
+# Tell versions [3.59,3.63) of GNU make to not export all variables.
|
||
+# Otherwise a system limit (for SysV at least) may be exceeded.
|
||
+.NOEXPORT:
|
||
--- ibus-table-1.9.18.orig/configure.ac 2020-07-22 11:52:11.639532241 +0200
|
||
+++ ibus-table-1.9.18/configure.ac 2020-07-22 14:43:51.905260956 +0200
|
||
@@ -68,6 +68,7 @@
|
||
setup/Makefile
|
||
setup/ibus-setup-table
|
||
setup/version.py
|
||
+ tests/Makefile
|
||
ibus-table.spec
|
||
ibus-table.pc]
|
||
)
|
||
diff -Nru ibus-table-1.9.18.orig/engine/Makefile.am ibus-table-1.9.18/engine/Makefile.am
|
||
--- ibus-table-1.9.18.orig/engine/Makefile.am 2020-07-22 11:52:11.650532123 +0200
|
||
+++ ibus-table-1.9.18/engine/Makefile.am 2020-07-22 14:43:51.906260946 +0200
|
||
@@ -32,6 +32,7 @@
|
||
table.py \
|
||
tabcreatedb.py \
|
||
tabsqlitedb.py \
|
||
+ it_util.py \
|
||
$(NULL)
|
||
engine_table_DATA = \
|
||
$(NULL)
|
||
diff -Nru ibus-table-1.9.18.orig/engine/it_util.py ibus-table-1.9.18/engine/it_util.py
|
||
--- ibus-table-1.9.18.orig/engine/it_util.py 1970-01-01 01:00:00.000000000 +0100
|
||
+++ ibus-table-1.9.18/engine/it_util.py 2020-07-22 14:43:51.906260946 +0200
|
||
@@ -0,0 +1,63 @@
|
||
+# -*- coding: utf-8 -*-
|
||
+# vim:et sts=4 sw=4
|
||
+#
|
||
+# ibus-table - The Tables engine for IBus
|
||
+#
|
||
+# Copyright (c) 2008-2009 Yu Yuwei <acevery@gmail.com>
|
||
+# Copyright (c) 2009-2014 Caius "kaio" CHANCE <me@kaio.net>
|
||
+# Copyright (c) 2012-2015 Mike FABIAN <mfabian@redhat.com>
|
||
+#
|
||
+# 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.1 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
||
+#
|
||
+'''
|
||
+Utility functions used in ibus-table
|
||
+'''
|
||
+
|
||
+import sys
|
||
+import re
|
||
+import string
|
||
+
|
||
+def config_section_normalize(section):
|
||
+ '''Replaces “_:” with “-” in the dconf section and converts to lower case
|
||
+
|
||
+ :param section: The name of the dconf section
|
||
+ :type section: string
|
||
+ :rtype: string
|
||
+
|
||
+ To make the comparison of the dconf sections work correctly.
|
||
+
|
||
+ I avoid using .lower() here because it is locale dependent, when
|
||
+ using .lower() this would not achieve the desired effect of
|
||
+ comparing the dconf sections case insentively in some locales, it
|
||
+ would fail for example if Turkish locale (tr_TR.UTF-8) is set.
|
||
+
|
||
+ Examples:
|
||
+
|
||
+ >>> config_section_normalize('Foo_bAr:Baz')
|
||
+ 'foo-bar-baz'
|
||
+ '''
|
||
+ return re.sub(r'[_:]', r'-', section).translate(
|
||
+ bytes.maketrans(
|
||
+ bytes(string.ascii_uppercase.encode('ascii')),
|
||
+ bytes(string.ascii_lowercase.encode('ascii'))))
|
||
+
|
||
+
|
||
+if __name__ == "__main__":
|
||
+ import doctest
|
||
+ (FAILED, ATTEMPTED) = doctest.testmod()
|
||
+ if FAILED:
|
||
+ sys.exit(1)
|
||
+ else:
|
||
+ sys.exit(0)
|
||
diff -Nru ibus-table-1.9.18.orig/engine/table.py ibus-table-1.9.18/engine/table.py
|
||
--- ibus-table-1.9.18.orig/engine/table.py 2020-07-22 11:52:11.650532123 +0200
|
||
+++ ibus-table-1.9.18/engine/table.py 2020-07-22 14:43:51.907260935 +0200
|
||
@@ -37,6 +37,7 @@
|
||
import re
|
||
from gi.repository import GObject
|
||
import time
|
||
+import it_util
|
||
|
||
debug_level = int(0)
|
||
|
||
@@ -215,8 +216,10 @@
|
||
max_key_length, database):
|
||
self.db = database
|
||
self._config = config
|
||
- engine_name = os.path.basename(self.db.filename).replace('.db', '')
|
||
- self._config_section = "engine/Table/%s" % engine_name.replace(' ','_')
|
||
+ engine_name = os.path.basename(
|
||
+ self.db.filename).replace('.db', '').replace(' ','_')
|
||
+ self._config_section = it_util.config_section_normalize(
|
||
+ "engine/Table/%s" % engine_name)
|
||
self._max_key_length = int(max_key_length)
|
||
self._max_key_length_pinyin = 7
|
||
self._valid_input_chars = valid_input_chars
|
||
@@ -320,7 +323,7 @@
|
||
self._config_section,
|
||
"ChineseMode"))
|
||
if self._chinese_mode == None:
|
||
- self._chinese_mode = self.get_chinese_mode()
|
||
+ self._chinese_mode = self.get_default_chinese_mode()
|
||
elif debug_level > 1:
|
||
sys.stderr.write(
|
||
"Chinese mode found in user config, mode=%s\n"
|
||
@@ -342,7 +345,7 @@
|
||
def get_new_lookup_table(
|
||
self, page_size=10,
|
||
select_keys=[49, 50, 51, 52, 53, 54, 55, 56, 57, 48],
|
||
- orientation=True):
|
||
+ orientation=IBus.Orientation.VERTICAL):
|
||
'''
|
||
[49, 50, 51, 52, 53, 54, 55, 56, 57, 48] are the key codes
|
||
for the characters ['1', '2', '3', '4', '5', '6', '7', '8', '0']
|
||
@@ -351,7 +354,7 @@
|
||
page_size = 1
|
||
if page_size > len(select_keys):
|
||
page_size = len(select_keys)
|
||
- lookup_table = IBus.LookupTable.new(
|
||
+ lookup_table = IBus.LookupTable(
|
||
page_size=page_size,
|
||
cursor_pos=0,
|
||
cursor_visible=True,
|
||
@@ -371,7 +374,7 @@
|
||
"""
|
||
return self._select_keys
|
||
|
||
- def get_chinese_mode (self):
|
||
+ def get_default_chinese_mode (self):
|
||
'''
|
||
Use db value or LC_CTYPE in your box to determine the _chinese_mode
|
||
'''
|
||
@@ -380,7 +383,7 @@
|
||
if __db_chinese_mode >= 0:
|
||
if debug_level > 1:
|
||
sys.stderr.write(
|
||
- "get_chinese_mode(): "
|
||
+ "get_default_chinese_mode(): "
|
||
+ "default Chinese mode found in database, mode=%s\n"
|
||
%__db_chinese_mode)
|
||
return __db_chinese_mode
|
||
@@ -390,19 +393,19 @@
|
||
__lc = os.environ['LC_ALL'].split('.')[0].lower()
|
||
if debug_level > 1:
|
||
sys.stderr.write(
|
||
- "get_chinese_mode(): __lc=%s found in LC_ALL\n"
|
||
+ "get_default_chinese_mode(): __lc=%s found in LC_ALL\n"
|
||
% __lc)
|
||
elif 'LC_CTYPE' in os.environ:
|
||
__lc = os.environ['LC_CTYPE'].split('.')[0].lower()
|
||
if debug_level > 1:
|
||
sys.stderr.write(
|
||
- "get_chinese_mode(): __lc=%s found in LC_CTYPE\n"
|
||
+ "get_default_chinese_mode(): __lc=%s found in LC_CTYPE\n"
|
||
% __lc)
|
||
else:
|
||
__lc = os.environ['LANG'].split('.')[0].lower()
|
||
if debug_level > 1:
|
||
sys.stderr.write(
|
||
- "get_chinese_mode(): __lc=%s found in LANG\n"
|
||
+ "get_default_chinese_mode(): __lc=%s found in LANG\n"
|
||
% __lc)
|
||
|
||
if '_cn' in __lc or '_sg' in __lc:
|
||
@@ -419,14 +422,14 @@
|
||
# variant:
|
||
if debug_level > 1:
|
||
sys.stderr.write(
|
||
- "get_chinese_mode(): last fallback, "
|
||
+ "get_default_chinese_mode(): last fallback, "
|
||
+ "database is Chinese but we don’t know "
|
||
+ "which variant.\n")
|
||
return 4 # show all Chinese characters
|
||
else:
|
||
if debug_level > 1:
|
||
sys.stderr.write(
|
||
- "get_chinese_mode(): last fallback, "
|
||
+ "get_default_chinese_mode(): last fallback, "
|
||
+ "database is not Chinese, returning -1.\n")
|
||
return -1
|
||
except:
|
||
@@ -1202,7 +1205,7 @@
|
||
class tabengine (IBus.Engine):
|
||
'''The IM Engine for Tables'''
|
||
|
||
- def __init__(self, bus, obj_path, db ):
|
||
+ def __init__(self, bus, obj_path, db, unit_test=False):
|
||
super(tabengine, self).__init__(connection=bus.get_connection(),
|
||
object_path=obj_path)
|
||
global debug_level
|
||
@@ -1210,6 +1213,7 @@
|
||
debug_level = int(os.getenv('IBUS_TABLE_DEBUG_LEVEL'))
|
||
except (TypeError, ValueError):
|
||
debug_level = int(0)
|
||
+ self._unit_test = unit_test
|
||
self._input_purpose = 0
|
||
self._has_input_purpose = False
|
||
if hasattr(IBus, 'InputPurpose'):
|
||
@@ -1224,12 +1228,16 @@
|
||
os.path.sep, 'icons', os.path.sep)
|
||
# name for config section
|
||
self._engine_name = os.path.basename(
|
||
- self.db.filename).replace('.db', '')
|
||
- self._config_section = (
|
||
- "engine/Table/%s" % self._engine_name.replace(' ','_'))
|
||
+ self.db.filename).replace('.db', '').replace(' ','_')
|
||
+ self._config_section = it_util.config_section_normalize(
|
||
+ "engine/Table/%s" % self._engine_name)
|
||
+ if debug_level > 1:
|
||
+ sys.stderr.write(
|
||
+ 'tabengine.__init__() self._config_section = %s\n'
|
||
+ % self._config_section)
|
||
|
||
# config module
|
||
- self._config = self._bus.get_config ()
|
||
+ self._config = self._bus.get_config()
|
||
self._config.connect ("value-changed", self.config_value_changed_cb)
|
||
|
||
# self._ime_py: Indicates whether this table supports pinyin mode
|
||
@@ -1699,7 +1707,7 @@
|
||
self._save_user_count = 0
|
||
super(tabengine, self).destroy()
|
||
|
||
- def set_input_mode(self, mode=0):
|
||
+ def set_input_mode(self, mode=1):
|
||
if mode == self._input_mode:
|
||
return
|
||
self._input_mode = mode
|
||
@@ -1722,6 +1730,9 @@
|
||
self._full_width_punct[self._input_mode])
|
||
self.reset()
|
||
|
||
+ def get_input_mode(self):
|
||
+ return self._input_mode
|
||
+
|
||
def set_pinyin_mode(self, mode=False):
|
||
if mode == self._editor._py_mode:
|
||
return
|
||
@@ -1741,76 +1752,302 @@
|
||
self._input_mode)
|
||
self._update_ui()
|
||
|
||
- def set_onechar_mode(self, mode=False):
|
||
+ def set_onechar_mode(self, mode=False, update_dconf=True):
|
||
if mode == self._editor._onechar:
|
||
return
|
||
self._editor._onechar = mode
|
||
self._init_or_update_property_menu(
|
||
self.onechar_mode_menu, mode)
|
||
- self._config.set_value(
|
||
- self._config_section,
|
||
- "OneChar",
|
||
- GLib.Variant.new_boolean(mode))
|
||
+ self.db.reset_phrases_cache()
|
||
+ if update_dconf:
|
||
+ self._config.set_value(
|
||
+ self._config_section,
|
||
+ "OneChar",
|
||
+ GLib.Variant.new_boolean(mode))
|
||
+
|
||
+ def get_onechar_mode(self):
|
||
+ return self._editor._onechar
|
||
|
||
- def set_autocommit_mode(self, mode=False):
|
||
+ def set_autocommit_mode(self, mode=False, update_dconf=True):
|
||
if mode == self._auto_commit:
|
||
return
|
||
self._auto_commit = mode
|
||
self._init_or_update_property_menu(
|
||
self.autocommit_mode_menu, mode)
|
||
- self._config.set_value(
|
||
- self._config_section,
|
||
- "AutoCommit",
|
||
- GLib.Variant.new_boolean(mode))
|
||
+ if update_dconf:
|
||
+ self._config.set_value(
|
||
+ self._config_section,
|
||
+ "AutoCommit",
|
||
+ GLib.Variant.new_boolean(mode))
|
||
|
||
- def set_letter_width(self, mode=False, input_mode=0):
|
||
- if mode == self._full_width_letter[input_mode]:
|
||
+ def get_autocommit_mode(self):
|
||
+ return self._auto_commit
|
||
+
|
||
+ def set_autoselect_mode(self, mode=False, update_dconf=True):
|
||
+ if mode == self._auto_select:
|
||
return
|
||
- self._full_width_letter[input_mode] = mode
|
||
- self._editor._full_width_letter[input_mode] = mode
|
||
- if input_mode == self._input_mode:
|
||
- self._init_or_update_property_menu(
|
||
- self.letter_width_menu, mode)
|
||
- if input_mode:
|
||
+ self._auto_select = mode
|
||
+ self._editor._auto_select = mode
|
||
+ if update_dconf:
|
||
self._config.set_value(
|
||
self._config_section,
|
||
- "TabDefFullWidthLetter",
|
||
+ "AutoSelect",
|
||
GLib.Variant.new_boolean(mode))
|
||
- else:
|
||
+
|
||
+ def get_autoselect_mode(self):
|
||
+ return self._auto_select
|
||
+
|
||
+ def set_autowildcard_mode(self, mode=False, update_dconf=True):
|
||
+ if mode == self._auto_wildcard:
|
||
+ return
|
||
+ self._auto_wildcard = mode
|
||
+ self._editor._auto_wildcard = mode
|
||
+ self.db.reset_phrases_cache()
|
||
+ if update_dconf:
|
||
self._config.set_value(
|
||
self._config_section,
|
||
- "EnDefFullWidthLetter",
|
||
+ "AutoWildcard",
|
||
GLib.Variant.new_boolean(mode))
|
||
|
||
- def set_punctuation_width(self, mode=False, input_mode=0):
|
||
- if mode == self._full_width_punct[input_mode]:
|
||
+ def get_autowildcard_mode(self):
|
||
+ return self._auto_wildcard
|
||
+
|
||
+ def set_single_wildcard_char(self, char=u'', update_dconf=True):
|
||
+ if char == self._single_wildcard_char:
|
||
return
|
||
- self._full_width_punct[input_mode] = mode
|
||
- self._editor._full_width_punct[input_mode] = mode
|
||
- if input_mode == self._input_mode:
|
||
- self._init_or_update_property_menu(
|
||
- self.punctuation_width_menu, mode)
|
||
- if input_mode:
|
||
+ self._single_wildcard_char = char
|
||
+ self._editor._single_wildcard_char = char
|
||
+ self.db.reset_phrases_cache()
|
||
+ if update_dconf:
|
||
+ self._config.set_value(
|
||
+ self._config_section,
|
||
+ "singlewildcardchar",
|
||
+ GLib.Variant.new_string(char))
|
||
+
|
||
+ def get_single_wildcard_char(self):
|
||
+ return self._single_wildcard_char
|
||
+
|
||
+ def set_multi_wildcard_char(self, char=u'', update_dconf=True):
|
||
+ if char == self._multi_wildcard_char:
|
||
+ return
|
||
+ self._multi_wildcard_char = char
|
||
+ self._editor._multi_wildcard_char = char
|
||
+ self.db.reset_phrases_cache()
|
||
+ if update_dconf:
|
||
+ self._config.set_value(
|
||
+ self._config_section,
|
||
+ "multiwildcardchar",
|
||
+ GLib.Variant.new_string(char))
|
||
+
|
||
+ def get_multi_wildcard_char(self):
|
||
+ return self._multi_wildcard_char
|
||
+
|
||
+ def set_space_key_behavior_mode(self, mode=False, update_dconf=True):
|
||
+ '''Sets the behaviour of the space key
|
||
+
|
||
+ :param mode: How the space key should behave
|
||
+ :type mode: Boolean
|
||
+ True: space is used as a page down key
|
||
+ and not as a commit key.
|
||
+ False: space is used as a commit key
|
||
+ and not used as a page down key
|
||
+ :param update_dconf: Whether to write the change to dconf.
|
||
+ Set this to False if this method is
|
||
+ called because the dconf key changed
|
||
+ to avoid endless loops when the dconf
|
||
+ key is changed twice in a short time.
|
||
+ :type update_dconf: boolean
|
||
+ '''
|
||
+ if debug_level > 1:
|
||
+ sys.stderr.write(
|
||
+ "set_space_key_behavior_mode(%s)\n"
|
||
+ %mode)
|
||
+ if mode == True:
|
||
+ # space is used as a page down key and not as a commit key:
|
||
+ if IBus.KEY_space not in self._page_down_keys:
|
||
+ self._page_down_keys.append(IBus.KEY_space)
|
||
+ if IBus.KEY_space in self._commit_keys:
|
||
+ self._commit_keys.remove(IBus.KEY_space)
|
||
+ if mode == False:
|
||
+ # space is used as a commit key and not used as a page down key:
|
||
+ if IBus.KEY_space in self._page_down_keys:
|
||
+ self._page_down_keys.remove(IBus.KEY_space)
|
||
+ if IBus.KEY_space not in self._commit_keys:
|
||
+ self._commit_keys.append(IBus.KEY_space)
|
||
+ if debug_level > 1:
|
||
+ sys.stderr.write(
|
||
+ 'set_space_key_behavior_mode(): self._page_down_keys=%s\n'
|
||
+ % repr(self._page_down_keys))
|
||
+ sys.stderr.write(
|
||
+ 'set_space_key_behavior_mode(): self._commit_keys=%s\n'
|
||
+ % repr(self._commit_keys))
|
||
+ if update_dconf:
|
||
self._config.set_value(
|
||
self._config_section,
|
||
- "TabDefFullWidthPunct",
|
||
+ "spacekeybehavior",
|
||
GLib.Variant.new_boolean(mode))
|
||
- else:
|
||
+
|
||
+ def get_space_key_behavior_mode(self):
|
||
+ mode = False
|
||
+ if IBus.KEY_space in self._page_down_keys:
|
||
+ mode = True
|
||
+ if IBus.KEY_space in self._commit_keys:
|
||
+ # commit key behaviour overrides the page down behaviour
|
||
+ mode = False
|
||
+ return mode
|
||
+
|
||
+ def set_always_show_lookup(self, mode=False, update_dconf=True):
|
||
+ if mode == self._always_show_lookup:
|
||
+ return
|
||
+ self._always_show_lookup = mode
|
||
+ if update_dconf:
|
||
self._config.set_value(
|
||
self._config_section,
|
||
- "EnDefFullWidthPunct",
|
||
+ "AlwaysShowLookup",
|
||
GLib.Variant.new_boolean(mode))
|
||
|
||
- def set_chinese_mode(self, mode=0):
|
||
+ def get_always_show_lookup(self):
|
||
+ return self._always_show_lookup
|
||
+
|
||
+ def set_lookup_table_orientation(self, orientation, update_dconf=True):
|
||
+ '''Sets the orientation of the lookup table
|
||
+
|
||
+ :param orientation: The orientation of the lookup table
|
||
+ :type mode: integer >= 0 and <= 2
|
||
+ IBUS_ORIENTATION_HORIZONTAL = 0,
|
||
+ IBUS_ORIENTATION_VERTICAL = 1,
|
||
+ IBUS_ORIENTATION_SYSTEM = 2.
|
||
+ :param update_dconf: Whether to write the change to dconf.
|
||
+ Set this to False if this method is
|
||
+ called because the dconf key changed
|
||
+ to avoid endless loops when the dconf
|
||
+ key is changed twice in a short time.
|
||
+ :type update_dconf: boolean
|
||
+ '''
|
||
+ if debug_level > 1:
|
||
+ sys.stderr.write(
|
||
+ "set_lookup_table_orientation(%s)\n"
|
||
+ %orientation)
|
||
+ if orientation == self._editor._orientation:
|
||
+ return
|
||
+ if orientation >= 0 and orientation <= 2:
|
||
+ self._editor._orientation = orientation
|
||
+ self._editor._lookup_table.set_orientation(orientation)
|
||
+ if update_dconf:
|
||
+ self._config.set_value(
|
||
+ self._config_section,
|
||
+ 'lookuptableorientation',
|
||
+ GLib.Variant.new_int32(orientation))
|
||
+
|
||
+ def get_lookup_table_orientation(self):
|
||
+ '''Returns the current orientation of the lookup table
|
||
+
|
||
+ :rtype: integer
|
||
+ '''
|
||
+ return self._editor._orientation
|
||
+
|
||
+ def set_page_size(self, page_size, update_dconf=True):
|
||
+ '''Sets the page size of the lookup table
|
||
+
|
||
+ :param orientation: The orientation of the lookup table
|
||
+ :type mode: integer >= 1 and <= number of select keys
|
||
+ :param update_dconf: Whether to write the change to dconf.
|
||
+ Set this to False if this method is
|
||
+ called because the dconf key changed
|
||
+ to avoid endless loops when the dconf
|
||
+ key is changed twice in a short time.
|
||
+ :type update_dconf: boolean
|
||
+ '''
|
||
+ if debug_level > 1:
|
||
+ sys.stderr.write(
|
||
+ "set_page_size(%s)\n"
|
||
+ %page_size)
|
||
+ if page_size == self._editor._page_size:
|
||
+ return
|
||
+ if page_size > len(self._editor._select_keys):
|
||
+ page_size = len(self._editor._select_keys)
|
||
+ if page_size < 1:
|
||
+ page_size = 1
|
||
+ self._editor._page_size = page_size
|
||
+ self._editor._lookup_table = self._editor.get_new_lookup_table(
|
||
+ page_size = self._editor._page_size,
|
||
+ select_keys = self._editor._select_keys,
|
||
+ orientation = self._editor._orientation)
|
||
+ self.reset()
|
||
+ if update_dconf:
|
||
+ self._config.set_value(
|
||
+ self._config_section,
|
||
+ 'lookuptablepagesize',
|
||
+ GLib.Variant.new_int32(page_size))
|
||
+
|
||
+ def get_page_size(self):
|
||
+ '''Returns the current page size of the lookup table
|
||
+
|
||
+ :rtype: integer
|
||
+ '''
|
||
+ return self._editor._page_size
|
||
+
|
||
+ def set_letter_width(self, mode=False, input_mode=0, update_dconf=True):
|
||
+ if mode == self._full_width_letter[input_mode]:
|
||
+ return
|
||
+ self._full_width_letter[input_mode] = mode
|
||
+ self._editor._full_width_letter[input_mode] = mode
|
||
+ if input_mode == self._input_mode:
|
||
+ self._init_or_update_property_menu(
|
||
+ self.letter_width_menu, mode)
|
||
+ if update_dconf:
|
||
+ if input_mode:
|
||
+ self._config.set_value(
|
||
+ self._config_section,
|
||
+ "TabDefFullWidthLetter",
|
||
+ GLib.Variant.new_boolean(mode))
|
||
+ else:
|
||
+ self._config.set_value(
|
||
+ self._config_section,
|
||
+ "EnDefFullWidthLetter",
|
||
+ GLib.Variant.new_boolean(mode))
|
||
+
|
||
+ def get_letter_width(self):
|
||
+ return self._full_width_letter
|
||
+
|
||
+ def set_punctuation_width(self, mode=False, input_mode=0, update_dconf=True):
|
||
+ if mode == self._full_width_punct[input_mode]:
|
||
+ return
|
||
+ self._full_width_punct[input_mode] = mode
|
||
+ self._editor._full_width_punct[input_mode] = mode
|
||
+ if input_mode == self._input_mode:
|
||
+ self._init_or_update_property_menu(
|
||
+ self.punctuation_width_menu, mode)
|
||
+ if update_dconf:
|
||
+ if input_mode:
|
||
+ self._config.set_value(
|
||
+ self._config_section,
|
||
+ "TabDefFullWidthPunct",
|
||
+ GLib.Variant.new_boolean(mode))
|
||
+ else:
|
||
+ self._config.set_value(
|
||
+ self._config_section,
|
||
+ "EnDefFullWidthPunct",
|
||
+ GLib.Variant.new_boolean(mode))
|
||
+
|
||
+ def get_punctuation_width(self):
|
||
+ return self._full_width_punct
|
||
+
|
||
+ def set_chinese_mode(self, mode=0, update_dconf=True):
|
||
if mode == self._editor._chinese_mode:
|
||
return
|
||
self._editor._chinese_mode = mode
|
||
+ self.db.reset_phrases_cache()
|
||
self._init_or_update_property_menu(
|
||
self.chinese_mode_menu, mode)
|
||
- self._config.set_value(
|
||
- self._config_section,
|
||
- "ChineseMode",
|
||
- GLib.Variant.new_int32(mode))
|
||
+ if update_dconf:
|
||
+ self._config.set_value(
|
||
+ self._config_section,
|
||
+ "ChineseMode",
|
||
+ GLib.Variant.new_int32(mode))
|
||
+
|
||
+ def get_chinese_mode(self):
|
||
+ return self._editor._chinese_mode
|
||
|
||
def _init_or_update_property_menu(self, menu, current_mode=0):
|
||
key = menu['key']
|
||
@@ -2233,6 +2470,44 @@
|
||
return True
|
||
return False
|
||
|
||
+ def _return_false(self, keyval, keycode, state):
|
||
+ '''A replacement for “return False” in do_process_key_event()
|
||
+
|
||
+ do_process_key_event should return “True” if a key event has
|
||
+ been handled completely. It should return “False” if the key
|
||
+ event should be passed to the application.
|
||
+
|
||
+ But just doing “return False” doesn’t work well when trying to
|
||
+ do the unit tests. The MockEngine class in the unit tests
|
||
+ cannot get that return value. Therefore, it cannot do the
|
||
+ necessary updates to the self._mock_committed_text etc. which
|
||
+ prevents proper testing of the effects of such keys passed to
|
||
+ the application. Instead of “return False”, one can also use
|
||
+ self.forward_key_event(keyval, keycode, keystate) to pass the
|
||
+ key to the application. And this works fine with the unit
|
||
+ tests because a forward_key_event function is implemented in
|
||
+ MockEngine as well which then gets the key and can test its
|
||
+ effects.
|
||
+
|
||
+ Unfortunately, “forward_key_event()” does not work in Qt5
|
||
+ applications because the ibus module in Qt5 does not implement
|
||
+ “forward_key_event()”. Therefore, always using
|
||
+ “forward_key_event()” instead of “return False” in
|
||
+ “do_process_key_event()” would break ibus-typing-booster
|
||
+ completely for all Qt5 applictions.
|
||
+
|
||
+ To work around this problem and make unit testing possible
|
||
+ without breaking Qt5 applications, we use this helper function
|
||
+ which uses “forward_key_event()” when unit testing and “return
|
||
+ False” during normal usage.
|
||
+
|
||
+ '''
|
||
+ if self._unit_test:
|
||
+ self.forward_key_event(keyval, keycode, state)
|
||
+ return True
|
||
+ else:
|
||
+ return False
|
||
+
|
||
def do_process_key_event(self, keyval, keycode, state):
|
||
'''Process Key Events
|
||
Key Events include Key Press and Key Release,
|
||
@@ -2243,9 +2518,13 @@
|
||
if (self._has_input_purpose
|
||
and self._input_purpose
|
||
in [IBus.InputPurpose.PASSWORD, IBus.InputPurpose.PIN]):
|
||
- return False
|
||
+ return self._return_false(keyval, keycode, state)
|
||
|
||
key = KeyEvent(keyval, keycode, state)
|
||
+ if debug_level > 1:
|
||
+ sys.stderr.write(
|
||
+ "process_key_event() "
|
||
+ "KeyEvent object: %s" % key)
|
||
|
||
result = self._process_key_event (key)
|
||
self._prev_key = key
|
||
@@ -2308,13 +2587,13 @@
|
||
def _english_mode_process_key_event(self, key):
|
||
# Ignore key release events
|
||
if key.state & IBus.ModifierType.RELEASE_MASK:
|
||
- return False
|
||
+ return self._return_false(key.val, key.code, key.state)
|
||
if key.val >= 128:
|
||
- return False
|
||
+ return self._return_false(key.val, key.code, key.state)
|
||
# we ignore all hotkeys here
|
||
if (key.state
|
||
& (IBus.ModifierType.CONTROL_MASK|IBus.ModifierType.MOD1_MASK)):
|
||
- return False
|
||
+ return self._return_false(key.val, key.code, key.state)
|
||
keychar = IBus.keyval_to_unicode(key.val)
|
||
if type(keychar) != type(u''):
|
||
keychar = keychar.decode('UTF-8')
|
||
@@ -2323,7 +2602,7 @@
|
||
else:
|
||
trans_char = self.cond_letter_translate(keychar)
|
||
if trans_char == keychar:
|
||
- return False
|
||
+ return self._return_false(key.val, key.code, key.state)
|
||
self.commit_string(trans_char)
|
||
return True
|
||
|
||
@@ -2387,7 +2666,7 @@
|
||
# (Must be below all self._match_hotkey() callse
|
||
# because these match on a release event).
|
||
if key.state & IBus.ModifierType.RELEASE_MASK:
|
||
- return False
|
||
+ return self._return_false(key.val, key.code, key.state)
|
||
|
||
keychar = IBus.keyval_to_unicode(key.val)
|
||
if type(keychar) != type(u''):
|
||
@@ -2419,7 +2698,7 @@
|
||
trans_char = self.cond_letter_translate(keychar)
|
||
if trans_char == keychar:
|
||
self._prev_char = trans_char
|
||
- return False
|
||
+ return self._return_false(key.val, key.code, key.state)
|
||
else:
|
||
self.commit_string(trans_char)
|
||
return True
|
||
@@ -2441,12 +2720,12 @@
|
||
# input but it ends up here. If it is leading input
|
||
# (i.e. the preëdit is empty) we should always pass
|
||
# IBus.KEY_KP_Enter to the application:
|
||
- return False
|
||
+ return self._return_false(key.val, key.code, key.state)
|
||
if self._auto_select:
|
||
self._editor.commit_to_preedit()
|
||
commit_string = self._editor.get_preedit_string_complete()
|
||
self.commit_string(commit_string)
|
||
- return False
|
||
+ return self._return_false(key.val, key.code, key.state)
|
||
else:
|
||
commit_string = self._editor.get_preedit_tabkeys_complete()
|
||
self.commit_string(commit_string)
|
||
@@ -2474,18 +2753,18 @@
|
||
# to “шшш”.
|
||
self._editor.commit_to_preedit()
|
||
self.commit_string(self._editor.get_preedit_string_complete())
|
||
- return False
|
||
+ return self._return_false(key.val, key.code, key.state)
|
||
|
||
if key.val in (IBus.KEY_Down, IBus.KEY_KP_Down) :
|
||
if not self._editor.get_preedit_string_complete():
|
||
- return False
|
||
+ return self._return_false(key.val, key.code, key.state)
|
||
res = self._editor.cursor_down()
|
||
self._update_ui()
|
||
return res
|
||
|
||
if key.val in (IBus.KEY_Up, IBus.KEY_KP_Up):
|
||
if not self._editor.get_preedit_string_complete():
|
||
- return False
|
||
+ return self._return_false(key.val, key.code, key.state)
|
||
res = self._editor.cursor_up()
|
||
self._update_ui()
|
||
return res
|
||
@@ -2493,7 +2772,7 @@
|
||
if (key.val in (IBus.KEY_Left, IBus.KEY_KP_Left)
|
||
and key.state & IBus.ModifierType.CONTROL_MASK):
|
||
if not self._editor.get_preedit_string_complete():
|
||
- return False
|
||
+ return self._return_false(key.val, key.code, key.state)
|
||
self._editor.control_arrow_left()
|
||
self._update_ui()
|
||
return True
|
||
@@ -2501,21 +2780,21 @@
|
||
if (key.val in (IBus.KEY_Right, IBus.KEY_KP_Right)
|
||
and key.state & IBus.ModifierType.CONTROL_MASK):
|
||
if not self._editor.get_preedit_string_complete():
|
||
- return False
|
||
+ return self._return_false(key.val, key.code, key.state)
|
||
self._editor.control_arrow_right()
|
||
self._update_ui()
|
||
return True
|
||
|
||
if key.val in (IBus.KEY_Left, IBus.KEY_KP_Left):
|
||
if not self._editor.get_preedit_string_complete():
|
||
- return False
|
||
+ return self._return_false(key.val, key.code, key.state)
|
||
self._editor.arrow_left()
|
||
self._update_ui()
|
||
return True
|
||
|
||
if key.val in (IBus.KEY_Right, IBus.KEY_KP_Right):
|
||
if not self._editor.get_preedit_string_complete():
|
||
- return False
|
||
+ return self._return_false(key.val, key.code, key.state)
|
||
self._editor.arrow_right()
|
||
self._update_ui()
|
||
return True
|
||
@@ -2523,14 +2802,14 @@
|
||
if (key.val == IBus.KEY_BackSpace
|
||
and key.state & IBus.ModifierType.CONTROL_MASK):
|
||
if not self._editor.get_preedit_string_complete():
|
||
- return False
|
||
+ return self._return_false(key.val, key.code, key.state)
|
||
self._editor.remove_preedit_before_cursor()
|
||
self._update_ui()
|
||
return True
|
||
|
||
if key.val == IBus.KEY_BackSpace:
|
||
if not self._editor.get_preedit_string_complete():
|
||
- return False
|
||
+ return self._return_false(key.val, key.code, key.state)
|
||
self._editor.remove_char()
|
||
self._update_ui()
|
||
return True
|
||
@@ -2538,14 +2817,14 @@
|
||
if (key.val == IBus.KEY_Delete
|
||
and key.state & IBus.ModifierType.CONTROL_MASK):
|
||
if not self._editor.get_preedit_string_complete():
|
||
- return False
|
||
+ return self._return_false(key.val, key.code, key.state)
|
||
self._editor.remove_preedit_after_cursor()
|
||
self._update_ui()
|
||
return True
|
||
|
||
if key.val == IBus.KEY_Delete:
|
||
if not self._editor.get_preedit_string_complete():
|
||
- return False
|
||
+ return self._return_false(key.val, key.code, key.state)
|
||
self._editor.delete()
|
||
self._update_ui()
|
||
return True
|
||
@@ -2567,10 +2846,10 @@
|
||
# now we ignore all other hotkeys
|
||
if (key.state
|
||
& (IBus.ModifierType.CONTROL_MASK|IBus.ModifierType.MOD1_MASK)):
|
||
- return False
|
||
+ return self._return_false(key.val, key.code, key.state)
|
||
|
||
if key.state & IBus.ModifierType.MOD1_MASK:
|
||
- return False
|
||
+ return self._return_false(key.val, key.code, key.state)
|
||
|
||
# Section to handle valid input characters:
|
||
#
|
||
@@ -2731,7 +3010,7 @@
|
||
#
|
||
# returned no result. So whatever this was, we cannot handle it,
|
||
# just pass it through to the application by returning “False”.
|
||
- return False
|
||
+ return self._return_false(key.val, key.code, key.state)
|
||
|
||
def do_focus_in (self):
|
||
if debug_level > 1:
|
||
@@ -2802,92 +3081,47 @@
|
||
self.set_input_mode(value)
|
||
return
|
||
if name == u'autoselect':
|
||
- self._editor._auto_select = value
|
||
- self._auto_select = value
|
||
+ self.set_autoselect_mode(value, update_dconf=False)
|
||
return
|
||
if name == u'autocommit':
|
||
- self.set_autocommit_mode(value)
|
||
+ self.set_autocommit_mode(value, update_dconf=False)
|
||
return
|
||
if name == u'chinesemode':
|
||
- self.set_chinese_mode(value)
|
||
- self.db.reset_phrases_cache()
|
||
+ self.set_chinese_mode(value, update_dconf=False)
|
||
return
|
||
if name == u'endeffullwidthletter':
|
||
- self.set_letter_width(value, input_mode=0)
|
||
+ self.set_letter_width(value, input_mode=0, update_dconf=False)
|
||
return
|
||
if name == u'endeffullwidthpunct':
|
||
- self.set_punctuation_width(value, input_mode=0)
|
||
+ self.set_punctuation_width(value, input_mode=0, update_dconf=False)
|
||
return
|
||
if name == u'lookuptableorientation':
|
||
- self._editor._orientation = value
|
||
- self._editor._lookup_table.set_orientation(value)
|
||
+ self.set_lookup_table_orientation(value, update_dconf=False)
|
||
return
|
||
if name == u'lookuptablepagesize':
|
||
- if value > len(self._editor._select_keys):
|
||
- value = len(self._editor._select_keys)
|
||
- self._config.set_value(
|
||
- self._config_section,
|
||
- 'lookuptablepagesize',
|
||
- GLib.Variant.new_int32(value))
|
||
- if value < 1:
|
||
- value = 1
|
||
- self._config.set_value(
|
||
- self._config_section,
|
||
- 'lookuptablepagesize',
|
||
- GLib.Variant.new_int32(value))
|
||
- self._editor._page_size = value
|
||
- self._editor._lookup_table = self._editor.get_new_lookup_table(
|
||
- page_size = self._editor._page_size,
|
||
- select_keys = self._editor._select_keys,
|
||
- orientation = self._editor._orientation)
|
||
- self.reset()
|
||
- return
|
||
- if name == u'lookuptableselectkeys':
|
||
- self._editor.set_select_keys(value)
|
||
+ self.set_page_size(value, update_dconf=False)
|
||
return
|
||
if name == u'onechar':
|
||
- self.set_onechar_mode(value)
|
||
- self.db.reset_phrases_cache()
|
||
+ self.set_onechar_mode(value, update_dconf=False)
|
||
return
|
||
if name == u'tabdeffullwidthletter':
|
||
- self.set_letter_width(value, input_mode=1)
|
||
+ self.set_letter_width(value, input_mode=1, update_dconf=False)
|
||
return
|
||
if name == u'tabdeffullwidthpunct':
|
||
- self.set_punctuation_width(value, input_mode=1)
|
||
+ self.set_punctuation_width(value, input_mode=1, update_dconf=False)
|
||
return
|
||
if name == u'alwaysshowlookup':
|
||
- self._always_show_lookup = value
|
||
+ self.set_always_show_lookup(value, update_dconf=False)
|
||
return
|
||
if name == u'spacekeybehavior':
|
||
- if value == True:
|
||
- # space is used as a page down key and not as a commit key:
|
||
- if IBus.KEY_space not in self._page_down_keys:
|
||
- self._page_down_keys.append(IBus.KEY_space)
|
||
- if IBus.KEY_space in self._commit_keys:
|
||
- self._commit_keys.remove(IBus.KEY_space)
|
||
- if value == False:
|
||
- # space is used as a commit key and not used as a page down key:
|
||
- if IBus.KEY_space in self._page_down_keys:
|
||
- self._page_down_keys.remove(IBus.KEY_space)
|
||
- if IBus.KEY_space not in self._commit_keys:
|
||
- self._commit_keys.append(IBus.KEY_space)
|
||
- if debug_level > 1:
|
||
- sys.stderr.write(
|
||
- "self._page_down_keys=%s\n"
|
||
- % repr(self._page_down_keys))
|
||
+ self.set_space_key_behavior_mode(value, update_dconf=False)
|
||
return
|
||
if name == u'singlewildcardchar':
|
||
- self._single_wildcard_char = value
|
||
- self._editor._single_wildcard_char = value
|
||
- self.db.reset_phrases_cache()
|
||
+ self.set_single_wildcard_char(value, update_dconf=False)
|
||
return
|
||
if name == u'multiwildcardchar':
|
||
- self._multi_wildcard_char = value
|
||
- self._editor._multi_wildcard_char = value
|
||
- self.db.reset_phrases_cache()
|
||
+ self.set_multi_wildcard_char(value, update_dconf=False)
|
||
return
|
||
if name == u'autowildcard':
|
||
- self._auto_wildcard = value
|
||
- self._editor._auto_wildcard = value
|
||
- self.db.reset_phrases_cache()
|
||
+ self.set_autowildcard_mode(value, update_dconf=False)
|
||
return
|
||
diff -Nru ibus-table-1.9.18.orig/engine/table.py.orig ibus-table-1.9.18/engine/table.py.orig
|
||
--- ibus-table-1.9.18.orig/engine/table.py.orig 1970-01-01 01:00:00.000000000 +0100
|
||
+++ ibus-table-1.9.18/engine/table.py.orig 2020-07-22 14:43:13.607657038 +0200
|
||
@@ -0,0 +1,2890 @@
|
||
+# -*- coding: utf-8 -*-
|
||
+# vim:et sts=4 sw=4
|
||
+#
|
||
+# ibus-table - The Tables engine for IBus
|
||
+#
|
||
+# Copyright (c) 2008-2009 Yu Yuwei <acevery@gmail.com>
|
||
+# Copyright (c) 2009-2014 Caius "kaio" CHANCE <me@kaio.net>
|
||
+# Copyright (c) 2012-2015 Mike FABIAN <mfabian@redhat.com>
|
||
+#
|
||
+# 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.1 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
||
+#
|
||
+
|
||
+__all__ = (
|
||
+ "tabengine",
|
||
+)
|
||
+
|
||
+import sys
|
||
+import os
|
||
+import string
|
||
+from gi import require_version
|
||
+require_version('IBus', '1.0')
|
||
+from gi.repository import IBus
|
||
+from gi.repository import GLib
|
||
+#import tabsqlitedb
|
||
+import re
|
||
+from gi.repository import GObject
|
||
+import time
|
||
+
|
||
+debug_level = int(0)
|
||
+
|
||
+from gettext import dgettext
|
||
+_ = lambda a : dgettext ("ibus-table", a)
|
||
+N_ = lambda a : a
|
||
+
|
||
+
|
||
+def ascii_ispunct(character):
|
||
+ '''
|
||
+ Use our own function instead of ascii.ispunct()
|
||
+ from “from curses import ascii” because the behaviour
|
||
+ of the latter is kind of weird. In Python 3.3.2 it does
|
||
+ for example:
|
||
+
|
||
+ >>> from curses import ascii
|
||
+ >>> ascii.ispunct('.')
|
||
+ True
|
||
+ >>> ascii.ispunct(u'.')
|
||
+ True
|
||
+ >>> ascii.ispunct('a')
|
||
+ False
|
||
+ >>> ascii.ispunct(u'a')
|
||
+ False
|
||
+ >>>
|
||
+ >>> ascii.ispunct(u'あ')
|
||
+ True
|
||
+ >>> ascii.ispunct('あ')
|
||
+ True
|
||
+ >>>
|
||
+
|
||
+ あ isn’t punctuation. ascii.ispunct() only really works
|
||
+ in the ascii range, it returns weird results when used
|
||
+ over the whole unicode range. Maybe we should better use
|
||
+ unicodedata.category(), which works fine to figure out
|
||
+ what is punctuation for all of unicode. But at the moment
|
||
+ I am only porting from Python2 to Python3 and just want to
|
||
+ preserve the original behaviour for the moment.
|
||
+ '''
|
||
+ if character in '''!"#$%&'()*+,-./:;<=>?@[\]^_`{|}~''':
|
||
+ return True
|
||
+ else:
|
||
+ return False
|
||
+
|
||
+def variant_to_value(variant):
|
||
+ if type(variant) != GLib.Variant:
|
||
+ return variant
|
||
+ type_string = variant.get_type_string()
|
||
+ if type_string == 's':
|
||
+ return variant.get_string()
|
||
+ elif type_string == 'i':
|
||
+ return variant.get_int32()
|
||
+ elif type_string == 'b':
|
||
+ return variant.get_boolean()
|
||
+ elif type_string == 'as':
|
||
+ # In the latest pygobject3 3.3.4 or later, g_variant_dup_strv
|
||
+ # returns the allocated strv but in the previous release,
|
||
+ # it returned the tuple of (strv, length)
|
||
+ if type(GLib.Variant.new_strv([]).dup_strv()) == tuple:
|
||
+ return variant.dup_strv()[0]
|
||
+ else:
|
||
+ return variant.dup_strv()
|
||
+ else:
|
||
+ print('error: unknown variant type: %s' %type_string)
|
||
+ return variant
|
||
+
|
||
+def argb(a, r, g, b):
|
||
+ return (((a & 0xff)<<24)
|
||
+ + ((r & 0xff) << 16)
|
||
+ + ((g & 0xff) << 8)
|
||
+ + (b & 0xff))
|
||
+
|
||
+def rgb(r, g, b):
|
||
+ return argb(255, r, g, b)
|
||
+
|
||
+__half_full_table = [
|
||
+ (0x0020, 0x3000, 1),
|
||
+ (0x0021, 0xFF01, 0x5E),
|
||
+ (0x00A2, 0xFFE0, 2),
|
||
+ (0x00A5, 0xFFE5, 1),
|
||
+ (0x00A6, 0xFFE4, 1),
|
||
+ (0x00AC, 0xFFE2, 1),
|
||
+ (0x00AF, 0xFFE3, 1),
|
||
+ (0x20A9, 0xFFE6, 1),
|
||
+ (0xFF61, 0x3002, 1),
|
||
+ (0xFF62, 0x300C, 2),
|
||
+ (0xFF64, 0x3001, 1),
|
||
+ (0xFF65, 0x30FB, 1),
|
||
+ (0xFF66, 0x30F2, 1),
|
||
+ (0xFF67, 0x30A1, 1),
|
||
+ (0xFF68, 0x30A3, 1),
|
||
+ (0xFF69, 0x30A5, 1),
|
||
+ (0xFF6A, 0x30A7, 1),
|
||
+ (0xFF6B, 0x30A9, 1),
|
||
+ (0xFF6C, 0x30E3, 1),
|
||
+ (0xFF6D, 0x30E5, 1),
|
||
+ (0xFF6E, 0x30E7, 1),
|
||
+ (0xFF6F, 0x30C3, 1),
|
||
+ (0xFF70, 0x30FC, 1),
|
||
+ (0xFF71, 0x30A2, 1),
|
||
+ (0xFF72, 0x30A4, 1),
|
||
+ (0xFF73, 0x30A6, 1),
|
||
+ (0xFF74, 0x30A8, 1),
|
||
+ (0xFF75, 0x30AA, 2),
|
||
+ (0xFF77, 0x30AD, 1),
|
||
+ (0xFF78, 0x30AF, 1),
|
||
+ (0xFF79, 0x30B1, 1),
|
||
+ (0xFF7A, 0x30B3, 1),
|
||
+ (0xFF7B, 0x30B5, 1),
|
||
+ (0xFF7C, 0x30B7, 1),
|
||
+ (0xFF7D, 0x30B9, 1),
|
||
+ (0xFF7E, 0x30BB, 1),
|
||
+ (0xFF7F, 0x30BD, 1),
|
||
+ (0xFF80, 0x30BF, 1),
|
||
+ (0xFF81, 0x30C1, 1),
|
||
+ (0xFF82, 0x30C4, 1),
|
||
+ (0xFF83, 0x30C6, 1),
|
||
+ (0xFF84, 0x30C8, 1),
|
||
+ (0xFF85, 0x30CA, 6),
|
||
+ (0xFF8B, 0x30D2, 1),
|
||
+ (0xFF8C, 0x30D5, 1),
|
||
+ (0xFF8D, 0x30D8, 1),
|
||
+ (0xFF8E, 0x30DB, 1),
|
||
+ (0xFF8F, 0x30DE, 5),
|
||
+ (0xFF94, 0x30E4, 1),
|
||
+ (0xFF95, 0x30E6, 1),
|
||
+ (0xFF96, 0x30E8, 6),
|
||
+ (0xFF9C, 0x30EF, 1),
|
||
+ (0xFF9D, 0x30F3, 1),
|
||
+ (0xFFA0, 0x3164, 1),
|
||
+ (0xFFA1, 0x3131, 30),
|
||
+ (0xFFC2, 0x314F, 6),
|
||
+ (0xFFCA, 0x3155, 6),
|
||
+ (0xFFD2, 0x315B, 9),
|
||
+ (0xFFE9, 0x2190, 4),
|
||
+ (0xFFED, 0x25A0, 1),
|
||
+ (0xFFEE, 0x25CB, 1)]
|
||
+
|
||
+def unichar_half_to_full(c):
|
||
+ code = ord(c)
|
||
+ for half, full, size in __half_full_table:
|
||
+ if code >= half and code < half + size:
|
||
+ if sys.version_info >= (3, 0, 0):
|
||
+ return chr(full + code - half)
|
||
+ else:
|
||
+ return unichr(full + code - half)
|
||
+ return c
|
||
+
|
||
+def unichar_full_to_half(c):
|
||
+ code = ord(c)
|
||
+ for half, full, size in __half_full_table:
|
||
+ if code >= full and code < full + size:
|
||
+ if sys.version_info >= (3, 0, 0):
|
||
+ return chr(half + code - full)
|
||
+ else:
|
||
+ return unichr(half + code - full)
|
||
+ return c
|
||
+
|
||
+SAVE_USER_COUNT_MAX = 16
|
||
+SAVE_USER_TIMEOUT = 30 # in seconds
|
||
+
|
||
+class KeyEvent:
|
||
+ def __init__(self, keyval, keycode, state):
|
||
+ self.val = keyval
|
||
+ self.code = keycode
|
||
+ self.state = state
|
||
+ def __str__(self):
|
||
+ return "%s 0x%08x" % (IBus.keyval_name(self.val), self.state)
|
||
+
|
||
+
|
||
+class editor(object):
|
||
+ '''Hold user inputs chars and preedit string'''
|
||
+ def __init__ (self, config, valid_input_chars, pinyin_valid_input_chars,
|
||
+ single_wildcard_char, multi_wildcard_char,
|
||
+ auto_wildcard, full_width_letter, full_width_punct,
|
||
+ max_key_length, database):
|
||
+ self.db = database
|
||
+ self._config = config
|
||
+ engine_name = os.path.basename(self.db.filename).replace('.db', '')
|
||
+ self._config_section = "engine/Table/%s" % engine_name.replace(' ','_')
|
||
+ self._max_key_length = int(max_key_length)
|
||
+ self._max_key_length_pinyin = 7
|
||
+ self._valid_input_chars = valid_input_chars
|
||
+ self._pinyin_valid_input_chars = pinyin_valid_input_chars
|
||
+ self._single_wildcard_char = single_wildcard_char
|
||
+ self._multi_wildcard_char = multi_wildcard_char
|
||
+ self._auto_wildcard = auto_wildcard
|
||
+ self._full_width_letter = full_width_letter
|
||
+ self._full_width_punct = full_width_punct
|
||
+ #
|
||
+ # The values below will be reset in
|
||
+ # self.clear_input_not_committed_to_preedit()
|
||
+ self._chars_valid = u'' # valid user input in table mode
|
||
+ self._chars_invalid = u'' # invalid user input in table mode
|
||
+ self._chars_valid_update_candidates_last = u''
|
||
+ self._chars_invalid_update_candidates_last = u''
|
||
+ # self._candidates holds the “best” candidates matching the user input
|
||
+ # [(tabkeys, phrase, freq, user_freq), ...]
|
||
+ self._candidates = []
|
||
+ self._candidates_previous = []
|
||
+
|
||
+ # self._u_chars: holds the user input of the phrases which
|
||
+ # have been automatically committed to preedit (but not yet
|
||
+ # “really” committed).
|
||
+ self._u_chars = []
|
||
+ # self._strings: holds the phrases which have been
|
||
+ # automatically committed to preedit (but not yet “really”
|
||
+ # committed).
|
||
+ #
|
||
+ # self._u_chars and self._strings should always have the same
|
||
+ # length, if I understand it correctly.
|
||
+ #
|
||
+ # Example when using the wubi-jidian86 table:
|
||
+ #
|
||
+ # self._u_chars = ['gaaa', 'gggg', 'ihty']
|
||
+ # self._strings = ['形式', '王', '小']
|
||
+ #
|
||
+ # I.e. after typing 'gaaa', '形式' is in the preedit and
|
||
+ # both self._u_chars and self._strings are empty. When typing
|
||
+ # another 'g', the maximum key length of the wubi table (which is 4)
|
||
+ # is exceeded and '形式' is automatically committed to the preedit
|
||
+ # (but not yet “really” committed, i.e. not yet committed into
|
||
+ # the application). The key 'gaaa' and the matching phrase '形式'
|
||
+ # are stored in self._u_chars and self._strings respectively
|
||
+ # and 'gaaa' is removed from self._chars_valid. Now self._chars_valid
|
||
+ # contains only the 'g' which starts a new search for candidates ...
|
||
+ # When removing the 'g' with backspace, the 'gaaa' is moved
|
||
+ # back from self._u_chars into self._chars_valid again and
|
||
+ # the same candidate list is shown as before the last 'g' had
|
||
+ # been entered.
|
||
+ self._strings = []
|
||
+ # self._cursor_precommit: The cursor
|
||
+ # position inthe array of strings which have already been
|
||
+ # committed to preëdit but not yet “really” committed.
|
||
+ self._cursor_precommit = 0
|
||
+
|
||
+ self._prompt_characters = eval(
|
||
+ self.db.ime_properties.get('char_prompts'))
|
||
+
|
||
+ select_keys_csv = variant_to_value(self._config.get_value(
|
||
+ self._config_section,
|
||
+ "LookupTableSelectKeys"))
|
||
+ if select_keys_csv == None:
|
||
+ select_keys_csv = self.db.get_select_keys()
|
||
+ if select_keys_csv == None:
|
||
+ select_keys_csv = '1,2,3,4,5,6,7,8,9'
|
||
+ self._select_keys = [
|
||
+ IBus.keyval_from_name(y)
|
||
+ for y in [x.strip() for x in select_keys_csv.split(",")]]
|
||
+ self._page_size = variant_to_value(self._config.get_value(
|
||
+ self._config_section,
|
||
+ "lookuptablepagesize"))
|
||
+ if self._page_size == None or self._page_size > len(self._select_keys):
|
||
+ self._page_size = len(self._select_keys)
|
||
+ self._orientation = variant_to_value(self._config.get_value(
|
||
+ self._config_section,
|
||
+ "LookupTableOrientation"))
|
||
+ if self._orientation == None:
|
||
+ self._orientation = self.db.get_orientation()
|
||
+ self._lookup_table = self.get_new_lookup_table(
|
||
+ page_size = self._page_size,
|
||
+ select_keys = self._select_keys,
|
||
+ orientation = self._orientation)
|
||
+ # self._py_mode: whether in pinyin mode
|
||
+ self._py_mode = False
|
||
+ # self._onechar: whether we only select single character
|
||
+ self._onechar = variant_to_value(self._config.get_value(
|
||
+ self._config_section,
|
||
+ "OneChar"))
|
||
+ if self._onechar == None:
|
||
+ self._onechar = False
|
||
+ # self._chinese_mode: the candidate filter mode,
|
||
+ # 0 means to show simplified Chinese only
|
||
+ # 1 means to show traditional Chinese only
|
||
+ # 2 means to show all characters but show simplified Chinese first
|
||
+ # 3 means to show all characters but show traditional Chinese first
|
||
+ # 4 means to show all characters
|
||
+ # we use LC_CTYPE or LANG to determine which one to use if
|
||
+ # no default comes from the config.
|
||
+ self._chinese_mode = variant_to_value(self._config.get_value(
|
||
+ self._config_section,
|
||
+ "ChineseMode"))
|
||
+ if self._chinese_mode == None:
|
||
+ self._chinese_mode = self.get_chinese_mode()
|
||
+ elif debug_level > 1:
|
||
+ sys.stderr.write(
|
||
+ "Chinese mode found in user config, mode=%s\n"
|
||
+ % self._chinese_mode)
|
||
+
|
||
+ # If auto select is true, then the first candidate phrase will
|
||
+ # be selected automatically during typing. Auto select is true
|
||
+ # by default for the stroke5 table for example.
|
||
+ self._auto_select = variant_to_value(self._config.get_value(
|
||
+ self._config_section,
|
||
+ "AutoSelect"))
|
||
+ if self._auto_select == None:
|
||
+ if self.db.ime_properties.get('auto_select') != None:
|
||
+ self._auto_select = self.db.ime_properties.get(
|
||
+ 'auto_select').lower() == u'true'
|
||
+ else:
|
||
+ self._auto_select = False
|
||
+
|
||
+ def get_new_lookup_table(
|
||
+ self, page_size=10,
|
||
+ select_keys=[49, 50, 51, 52, 53, 54, 55, 56, 57, 48],
|
||
+ orientation=True):
|
||
+ '''
|
||
+ [49, 50, 51, 52, 53, 54, 55, 56, 57, 48] are the key codes
|
||
+ for the characters ['1', '2', '3', '4', '5', '6', '7', '8', '0']
|
||
+ '''
|
||
+ if page_size < 1:
|
||
+ page_size = 1
|
||
+ if page_size > len(select_keys):
|
||
+ page_size = len(select_keys)
|
||
+ lookup_table = IBus.LookupTable.new(
|
||
+ page_size=page_size,
|
||
+ cursor_pos=0,
|
||
+ cursor_visible=True,
|
||
+ round=True)
|
||
+ for keycode in select_keys:
|
||
+ lookup_table.append_label(
|
||
+ IBus.Text.new_from_string("%s." %IBus.keyval_name(keycode)))
|
||
+ lookup_table.set_orientation(orientation)
|
||
+ return lookup_table
|
||
+
|
||
+ def get_select_keys(self):
|
||
+ """
|
||
+ Returns the list of key codes for the select keys.
|
||
+ For example, if the select keys are ["1", "2", ...] the
|
||
+ key codes are [49, 50, ...]. If the select keys are
|
||
+ ["F1", "F2", ...] the key codes are [65470, 65471, ...]
|
||
+ """
|
||
+ return self._select_keys
|
||
+
|
||
+ def get_chinese_mode (self):
|
||
+ '''
|
||
+ Use db value or LC_CTYPE in your box to determine the _chinese_mode
|
||
+ '''
|
||
+ # use db value, if applicable
|
||
+ __db_chinese_mode = self.db.get_chinese_mode()
|
||
+ if __db_chinese_mode >= 0:
|
||
+ if debug_level > 1:
|
||
+ sys.stderr.write(
|
||
+ "get_chinese_mode(): "
|
||
+ + "default Chinese mode found in database, mode=%s\n"
|
||
+ %__db_chinese_mode)
|
||
+ return __db_chinese_mode
|
||
+ # otherwise
|
||
+ try:
|
||
+ if 'LC_ALL' in os.environ:
|
||
+ __lc = os.environ['LC_ALL'].split('.')[0].lower()
|
||
+ if debug_level > 1:
|
||
+ sys.stderr.write(
|
||
+ "get_chinese_mode(): __lc=%s found in LC_ALL\n"
|
||
+ % __lc)
|
||
+ elif 'LC_CTYPE' in os.environ:
|
||
+ __lc = os.environ['LC_CTYPE'].split('.')[0].lower()
|
||
+ if debug_level > 1:
|
||
+ sys.stderr.write(
|
||
+ "get_chinese_mode(): __lc=%s found in LC_CTYPE\n"
|
||
+ % __lc)
|
||
+ else:
|
||
+ __lc = os.environ['LANG'].split('.')[0].lower()
|
||
+ if debug_level > 1:
|
||
+ sys.stderr.write(
|
||
+ "get_chinese_mode(): __lc=%s found in LANG\n"
|
||
+ % __lc)
|
||
+
|
||
+ if '_cn' in __lc or '_sg' in __lc:
|
||
+ # CN and SG should prefer traditional Chinese by default
|
||
+ return 2 # show simplified Chinese first
|
||
+ elif '_hk' in __lc or '_tw' in __lc or '_mo' in __lc:
|
||
+ # HK, TW, and MO should prefer traditional Chinese by default
|
||
+ return 3 # show traditional Chinese first
|
||
+ else:
|
||
+ if self.db._is_chinese:
|
||
+ # This table is used for Chinese, but we don’t
|
||
+ # know for which variant. Therefore, better show
|
||
+ # all Chinese characters and don’t prefer any
|
||
+ # variant:
|
||
+ if debug_level > 1:
|
||
+ sys.stderr.write(
|
||
+ "get_chinese_mode(): last fallback, "
|
||
+ + "database is Chinese but we don’t know "
|
||
+ + "which variant.\n")
|
||
+ return 4 # show all Chinese characters
|
||
+ else:
|
||
+ if debug_level > 1:
|
||
+ sys.stderr.write(
|
||
+ "get_chinese_mode(): last fallback, "
|
||
+ + "database is not Chinese, returning -1.\n")
|
||
+ return -1
|
||
+ except:
|
||
+ import traceback
|
||
+ traceback.print_exc()
|
||
+ return -1
|
||
+
|
||
+ def clear_all_input_and_preedit(self):
|
||
+ '''
|
||
+ Clear all input, whether committed to preëdit or not.
|
||
+ '''
|
||
+ if debug_level > 1:
|
||
+ sys.stderr.write("clear_all_input_and_preedit()\n")
|
||
+ self.clear_input_not_committed_to_preedit()
|
||
+ self._u_chars = []
|
||
+ self._strings = []
|
||
+ self._cursor_precommit = 0
|
||
+ self.update_candidates()
|
||
+
|
||
+ def is_empty(self):
|
||
+ return u'' == self._chars_valid + self._chars_invalid
|
||
+
|
||
+ def clear_input_not_committed_to_preedit(self):
|
||
+ '''
|
||
+ Clear the input which has not yet been committed to preëdit.
|
||
+ '''
|
||
+ if debug_level > 1:
|
||
+ sys.stderr.write("clear_input_not_committed_to_preedit()\n")
|
||
+ self._chars_valid = u''
|
||
+ self._chars_invalid = u''
|
||
+ self._chars_valid_update_candidates_last = u''
|
||
+ self._chars_invalid_update_candidates_last = u''
|
||
+ self._lookup_table.clear()
|
||
+ self._lookup_table.set_cursor_visible(True)
|
||
+ self._candidates = []
|
||
+ self._candidates_previous = []
|
||
+
|
||
+ def add_input(self, c):
|
||
+ '''
|
||
+ Add input character and update candidates.
|
||
+
|
||
+ Returns “True” if candidates were found, “False” if not.
|
||
+ '''
|
||
+ if (self._chars_invalid
|
||
+ or (not self._py_mode
|
||
+ and (c not in
|
||
+ self._valid_input_chars
|
||
+ + self._single_wildcard_char
|
||
+ + self._multi_wildcard_char))
|
||
+ or (self._py_mode
|
||
+ and (c not in
|
||
+ self._pinyin_valid_input_chars
|
||
+ + self._single_wildcard_char
|
||
+ + self._multi_wildcard_char))):
|
||
+ self._chars_invalid += c
|
||
+ else:
|
||
+ self._chars_valid += c
|
||
+ res = self.update_candidates()
|
||
+ return res
|
||
+
|
||
+ def pop_input(self):
|
||
+ '''remove and display last input char held'''
|
||
+ _c = ''
|
||
+ if self._chars_invalid:
|
||
+ _c = self._chars_invalid[-1]
|
||
+ self._chars_invalid = self._chars_invalid[:-1]
|
||
+ elif self._chars_valid:
|
||
+ _c = self._chars_valid[-1]
|
||
+ self._chars_valid = self._chars_valid[:-1]
|
||
+ if (not self._chars_valid) and self._u_chars:
|
||
+ self._chars_valid = self._u_chars.pop(
|
||
+ self._cursor_precommit - 1)
|
||
+ self._strings.pop(self._cursor_precommit - 1)
|
||
+ self._cursor_precommit -= 1
|
||
+ self.update_candidates ()
|
||
+ return _c
|
||
+
|
||
+ def get_input_chars (self):
|
||
+ '''get characters held, valid and invalid'''
|
||
+ return self._chars_valid + self._chars_invalid
|
||
+
|
||
+ def split_strings_committed_to_preedit(self, index, index_in_phrase):
|
||
+ head = self._strings[index][:index_in_phrase]
|
||
+ tail = self._strings[index][index_in_phrase:]
|
||
+ self._u_chars.pop(index)
|
||
+ self._strings.pop(index)
|
||
+ self._u_chars.insert(index, self.db.parse_phrase(head))
|
||
+ self._strings.insert(index, head)
|
||
+ self._u_chars.insert(index+1, self.db.parse_phrase(tail))
|
||
+ self._strings.insert(index+1, tail)
|
||
+
|
||
+ def remove_preedit_before_cursor(self):
|
||
+ '''Remove preëdit left of cursor'''
|
||
+ if self._chars_invalid:
|
||
+ return
|
||
+ if self.get_input_chars():
|
||
+ self.commit_to_preedit()
|
||
+ if not self._strings:
|
||
+ return
|
||
+ if self._cursor_precommit <= 0:
|
||
+ return
|
||
+ self._u_chars = self._u_chars[self._cursor_precommit:]
|
||
+ self._strings = self._strings[self._cursor_precommit:]
|
||
+ self._cursor_precommit = 0
|
||
+
|
||
+ def remove_preedit_after_cursor(self):
|
||
+ '''Remove preëdit right of cursor'''
|
||
+ if self._chars_invalid:
|
||
+ return
|
||
+ if self.get_input_chars():
|
||
+ self.commit_to_preedit()
|
||
+ if not self._strings:
|
||
+ return
|
||
+ if self._cursor_precommit >= len(self._strings):
|
||
+ return
|
||
+ self._u_chars = self._u_chars[:self._cursor_precommit]
|
||
+ self._strings = self._strings[:self._cursor_precommit]
|
||
+ self._cursor_precommit = len(self._strings)
|
||
+
|
||
+ def remove_preedit_character_before_cursor(self):
|
||
+ '''Remove character before cursor in strings comitted to preëdit'''
|
||
+ if self._chars_invalid:
|
||
+ return
|
||
+ if self.get_input_chars():
|
||
+ self.commit_to_preedit()
|
||
+ if not self._strings:
|
||
+ return
|
||
+ if self._cursor_precommit < 1:
|
||
+ return
|
||
+ self._cursor_precommit -= 1
|
||
+ self._chars_valid = self._u_chars.pop(self._cursor_precommit)
|
||
+ self._strings.pop(self._cursor_precommit)
|
||
+ self.update_candidates()
|
||
+
|
||
+ def remove_preedit_character_after_cursor (self):
|
||
+ '''Remove character after cursor in strings committed to preëdit'''
|
||
+ if self._chars_invalid:
|
||
+ return
|
||
+ if self.get_input_chars():
|
||
+ self.commit_to_preedit()
|
||
+ if not self._strings:
|
||
+ return
|
||
+ if self._cursor_precommit > len(self._strings) - 1:
|
||
+ return
|
||
+ self._u_chars.pop(self._cursor_precommit)
|
||
+ self._strings.pop(self._cursor_precommit)
|
||
+
|
||
+ def get_preedit_tabkeys_parts(self):
|
||
+ '''Returns the tabkeys which were used to type the parts
|
||
+ of the preëdit string.
|
||
+
|
||
+ Such as “(left_of_current_edit, current_edit, right_of_current_edit)”
|
||
+
|
||
+ “left_of_current_edit” and “right_of_current_edit” are
|
||
+ strings of tabkeys which have been typed to get the phrases
|
||
+ which have already been committed to preëdit, but not
|
||
+ “really” committed yet. “current_edit” is the string of
|
||
+ tabkeys of the part of the preëdit string which is not
|
||
+ committed at all.
|
||
+
|
||
+ For example, the return value could look like:
|
||
+
|
||
+ (('gggg', 'aahw'), 'adwu', ('ijgl', 'jbus'))
|
||
+
|
||
+ See also get_preedit_string_parts() which might return something
|
||
+ like
|
||
+
|
||
+ (('王', '工具'), '其', ('漫画', '最新'))
|
||
+
|
||
+ when the wubi-jidian86 table is used.
|
||
+ '''
|
||
+ left_of_current_edit = ()
|
||
+ current_edit = u''
|
||
+ right_of_current_edit = ()
|
||
+ if self.get_input_chars():
|
||
+ current_edit = self.get_input_chars()
|
||
+ if self._u_chars:
|
||
+ left_of_current_edit = tuple(
|
||
+ self._u_chars[:self._cursor_precommit])
|
||
+ right_of_current_edit = tuple(
|
||
+ self._u_chars[self._cursor_precommit:])
|
||
+ return (left_of_current_edit, current_edit, right_of_current_edit)
|
||
+
|
||
+ def get_preedit_tabkeys_complete(self):
|
||
+ '''Returns the tabkeys which belong to the parts of the preëdit
|
||
+ string as a single string
|
||
+ '''
|
||
+ (left_tabkeys,
|
||
+ current_tabkeys,
|
||
+ right_tabkeys) = self.get_preedit_tabkeys_parts()
|
||
+ return (u''.join(left_tabkeys)
|
||
+ + current_tabkeys
|
||
+ + u''.join(right_tabkeys))
|
||
+
|
||
+ def get_preedit_string_parts(self):
|
||
+ '''Returns the phrases which are parts of the preëdit string.
|
||
+
|
||
+ Such as “(left_of_current_edit, current_edit, right_of_current_edit)”
|
||
+
|
||
+ “left_of_current_edit” and “right_of_current_edit” are
|
||
+ tuples of strings which have already been committed to preëdit, but not
|
||
+ “really” committed yet. “current_edit” is the phrase in the part of the
|
||
+ preëdit string which is not yet committed at all.
|
||
+
|
||
+ For example, the return value could look like:
|
||
+
|
||
+ (('王', '工具'), '其', ('漫画', '最新'))
|
||
+
|
||
+ See also get_preedit_tabkeys_parts() which might return something
|
||
+ like
|
||
+
|
||
+ (('gggg', 'aahw'), 'adwu', ('ijgl', 'jbus'))
|
||
+
|
||
+ when the wubi-jidian86 table is used.
|
||
+ '''
|
||
+ left_of_current_edit = ()
|
||
+ current_edit = u''
|
||
+ right_of_current_edit = ()
|
||
+ if self._candidates:
|
||
+ current_edit = self._candidates[
|
||
+ int(self._lookup_table.get_cursor_pos())][1]
|
||
+ elif self.get_input_chars():
|
||
+ current_edit = self.get_input_chars()
|
||
+ if self._strings:
|
||
+ left_of_current_edit = tuple(
|
||
+ self._strings[:self._cursor_precommit])
|
||
+ right_of_current_edit = tuple(
|
||
+ self._strings[self._cursor_precommit:])
|
||
+ return (left_of_current_edit, current_edit, right_of_current_edit)
|
||
+
|
||
+ def get_preedit_string_complete(self):
|
||
+ '''Returns the phrases which are parts of the preëdit string as a
|
||
+ single string
|
||
+
|
||
+ '''
|
||
+ (left_strings,
|
||
+ current_string,
|
||
+ right_strings) = self.get_preedit_string_parts()
|
||
+ return u''.join(left_strings) + current_string + u''.join(right_strings)
|
||
+
|
||
+ def get_caret (self):
|
||
+ '''Get caret position in preëdit string'''
|
||
+ caret = 0
|
||
+ if self._cursor_precommit and self._strings:
|
||
+ for x in self._strings[:self._cursor_precommit]:
|
||
+ caret += len(x)
|
||
+ if self._candidates:
|
||
+ caret += len(
|
||
+ self._candidates[int(self._lookup_table.get_cursor_pos())][1])
|
||
+ else:
|
||
+ caret += len(self.get_input_chars())
|
||
+ return caret
|
||
+
|
||
+ def arrow_left(self):
|
||
+ '''Move cursor left in the preëdit string.'''
|
||
+ if self._chars_invalid:
|
||
+ return
|
||
+ if self.get_input_chars():
|
||
+ self.commit_to_preedit()
|
||
+ if not self._strings:
|
||
+ return
|
||
+ if self._cursor_precommit <= 0:
|
||
+ return
|
||
+ if len(self._strings[self._cursor_precommit-1]) <= 1:
|
||
+ self._cursor_precommit -= 1
|
||
+ else:
|
||
+ self.split_strings_committed_to_preedit(
|
||
+ self._cursor_precommit-1, -1)
|
||
+ self.update_candidates()
|
||
+
|
||
+ def arrow_right(self):
|
||
+ '''Move cursor right in the preëdit string.'''
|
||
+ if self._chars_invalid:
|
||
+ return
|
||
+ if self.get_input_chars():
|
||
+ self.commit_to_preedit()
|
||
+ if not self._strings:
|
||
+ return
|
||
+ if self._cursor_precommit >= len(self._strings):
|
||
+ return
|
||
+ self._cursor_precommit += 1
|
||
+ if len(self._strings[self._cursor_precommit-1]) > 1:
|
||
+ self.split_strings_committed_to_preedit(self._cursor_precommit-1, 1)
|
||
+ self.update_candidates()
|
||
+
|
||
+ def control_arrow_left(self):
|
||
+ '''Move cursor to the beginning of the preëdit string.'''
|
||
+ if self._chars_invalid:
|
||
+ return
|
||
+ if self.get_input_chars():
|
||
+ self.commit_to_preedit()
|
||
+ if not self._strings:
|
||
+ return
|
||
+ self._cursor_precommit = 0
|
||
+ self.update_candidates ()
|
||
+
|
||
+ def control_arrow_right(self):
|
||
+ '''Move cursor to the end of the preëdit string'''
|
||
+ if self._chars_invalid:
|
||
+ return
|
||
+ if self.get_input_chars():
|
||
+ self.commit_to_preedit()
|
||
+ if not self._strings:
|
||
+ return
|
||
+ self._cursor_precommit = len(self._strings)
|
||
+ self.update_candidates ()
|
||
+
|
||
+ def append_candidate_to_lookup_table(
|
||
+ self, tabkeys=u'', phrase=u'', freq=0, user_freq=0):
|
||
+ '''append candidate to lookup_table'''
|
||
+ if debug_level > 1:
|
||
+ sys.stderr.write(
|
||
+ "append_candidate() "
|
||
+ + "tabkeys=%(t)s phrase=%(p)s freq=%(f)s user_freq=%(u)s\n"
|
||
+ % {'t': tabkeys, 'p': phrase, 'f': freq, 'u': user_freq})
|
||
+ if not tabkeys or not phrase:
|
||
+ return
|
||
+ regexp = self._chars_valid
|
||
+ if self._multi_wildcard_char:
|
||
+ regexp = regexp.replace(
|
||
+ self._multi_wildcard_char, '_multi_wildchard_char_')
|
||
+ if self._single_wildcard_char:
|
||
+ regexp = regexp.replace(
|
||
+ self._single_wildcard_char, '_single_wildchard_char_')
|
||
+ regexp = re.escape(regexp)
|
||
+ regexp = regexp.replace('_multi_wildchard_char_', '.*')
|
||
+ regexp = regexp.replace('_single_wildchard_char_', '.?')
|
||
+ match = re.match(r'^'+regexp, tabkeys)
|
||
+ if match:
|
||
+ remaining_tabkeys = tabkeys[match.end():]
|
||
+ else:
|
||
+ # This should never happen! For the candidates
|
||
+ # added to the lookup table here, a match has
|
||
+ # been found for self._chars_valid in the database.
|
||
+ # In that case, the above regular expression should
|
||
+ # match as well.
|
||
+ remaining_tabkeys = tabkeys
|
||
+ if debug_level > 1:
|
||
+ sys.stderr.write(
|
||
+ "append_candidate() "
|
||
+ + "remaining_tabkeys=%(remaining_tabkeys)s "
|
||
+ + "self._chars_valid=%(chars_valid)s phrase=%(phrase)s\n"
|
||
+ % {'remaining_tabkeys': remaining_tabkeys,
|
||
+ 'chars_valid': self._chars_valid,
|
||
+ 'phrase': phrase})
|
||
+ table_code = u''
|
||
+ if self.db._is_chinese and self._py_mode:
|
||
+ # restore tune symbol
|
||
+ remaining_tabkeys = remaining_tabkeys.replace(
|
||
+ '!','↑1').replace(
|
||
+ '@','↑2').replace(
|
||
+ '#','↑3').replace(
|
||
+ '$','↑4').replace(
|
||
+ '%','↑5')
|
||
+ # If in pinyin mode, phrase can only be one character.
|
||
+ # When using pinyin mode for a table like Wubi or Cangjie,
|
||
+ # the reason is probably because one does not know the
|
||
+ # Wubi or Cangjie code. So get that code from the table
|
||
+ # and display it as well to help the user learn that code.
|
||
+ # The Wubi tables contain several codes for the same
|
||
+ # character, therefore self.db.find_zi_code(phrase) may
|
||
+ # return a list. The last code in that list is the full
|
||
+ # table code for that characters, other entries in that
|
||
+ # list are shorter substrings of the full table code which
|
||
+ # are not interesting to display. Therefore, we use only
|
||
+ # the last element of the list of table codes.
|
||
+ possible_table_codes = self.db.find_zi_code(phrase)
|
||
+ if possible_table_codes:
|
||
+ table_code = possible_table_codes[-1]
|
||
+ table_code_new = u''
|
||
+ for char in table_code:
|
||
+ if char in self._prompt_characters:
|
||
+ table_code_new += self._prompt_characters[char]
|
||
+ else:
|
||
+ table_code_new += char
|
||
+ table_code = table_code_new
|
||
+ if not self._py_mode:
|
||
+ remaining_tabkeys_new = u''
|
||
+ for char in remaining_tabkeys:
|
||
+ if char in self._prompt_characters:
|
||
+ remaining_tabkeys_new += self._prompt_characters[char]
|
||
+ else:
|
||
+ remaining_tabkeys_new += char
|
||
+ remaining_tabkeys = remaining_tabkeys_new
|
||
+ candidate_text = phrase + u' ' + remaining_tabkeys
|
||
+ if table_code:
|
||
+ candidate_text = candidate_text + u' ' + table_code
|
||
+ attrs = IBus.AttrList ()
|
||
+ attrs.append(IBus.attr_foreground_new(
|
||
+ rgb(0x19,0x73,0xa2), 0, len(candidate_text)))
|
||
+ if not self._py_mode and freq < 0:
|
||
+ # this is a user defined phrase:
|
||
+ attrs.append(
|
||
+ IBus.attr_foreground_new(rgb(0x77,0x00,0xc3), 0, len(phrase)))
|
||
+ elif not self._py_mode and user_freq > 0:
|
||
+ # this is a system phrase which has already been used by the user:
|
||
+ attrs.append(IBus.attr_foreground_new(
|
||
+ rgb(0x00,0x00,0x00), 0, len(phrase)))
|
||
+ else:
|
||
+ # this is a system phrase that has not been used yet:
|
||
+ attrs.append(IBus.attr_foreground_new(
|
||
+ rgb(0x00,0x00,0x00), 0, len(phrase)))
|
||
+ if debug_level > 0:
|
||
+ debug_text = u' ' + str(freq) + u' ' + str(user_freq)
|
||
+ candidate_text += debug_text
|
||
+ attrs.append(IBus.attr_foreground_new(
|
||
+ rgb(0x00,0xff,0x00),
|
||
+ len(candidate_text) - len(debug_text),
|
||
+ len(candidate_text)))
|
||
+ text = IBus.Text.new_from_string(candidate_text)
|
||
+ i = 0
|
||
+ while attrs.get(i) != None:
|
||
+ attr = attrs.get(i)
|
||
+ text.append_attribute(attr.get_attr_type(),
|
||
+ attr.get_value(),
|
||
+ attr.get_start_index(),
|
||
+ attr.get_end_index())
|
||
+ i += 1
|
||
+ self._lookup_table.append_candidate (text)
|
||
+ self._lookup_table.set_cursor_visible(True)
|
||
+
|
||
+ def update_candidates (self):
|
||
+ '''
|
||
+ Searches for candidates and updates the lookuptable.
|
||
+
|
||
+ Returns “True” if candidates were found and “False” if not.
|
||
+ '''
|
||
+ if debug_level > 1:
|
||
+ sys.stderr.write(
|
||
+ "update_candidates() "
|
||
+ + "self._chars_valid=%(chars_valid)s "
|
||
+ + "self._chars_invalid=%(chars_invalid)s "
|
||
+ + "self._chars_valid_update_candidates_last=%(chars_last)s "
|
||
+ + "self._candidates=%(candidates)s "
|
||
+ + "self.db.startchars=%(start)s "
|
||
+ + "self._strings=%(strings)s\n"
|
||
+ % {'chars_valid': self._chars_valid,
|
||
+ 'chars_invalid': self._chars_invalid,
|
||
+ 'chars_last': self._chars_valid_update_candidates_last,
|
||
+ 'candidates': self._candidates,
|
||
+ 'start': self.db.startchars,
|
||
+ 'strings': self._strings})
|
||
+ if (self._chars_valid == self._chars_valid_update_candidates_last
|
||
+ and
|
||
+ self._chars_invalid == self._chars_invalid_update_candidates_last):
|
||
+ # The input did not change since we came here last, do
|
||
+ # nothing and leave candidates and lookup table unchanged:
|
||
+ if self._candidates:
|
||
+ return True
|
||
+ else:
|
||
+ return False
|
||
+ self._chars_valid_update_candidates_last = self._chars_valid
|
||
+ self._chars_invalid_update_candidates_last = self._chars_invalid
|
||
+ self._lookup_table.clear()
|
||
+ self._lookup_table.set_cursor_visible(True)
|
||
+ if self._chars_invalid or not self._chars_valid:
|
||
+ self._candidates = []
|
||
+ self._candidates_previous = self._candidates
|
||
+ return False
|
||
+ if self._py_mode and self.db._is_chinese:
|
||
+ self._candidates = self.db.select_chinese_characters_by_pinyin(
|
||
+ tabkeys=self._chars_valid,
|
||
+ chinese_mode=self._chinese_mode,
|
||
+ single_wildcard_char=self._single_wildcard_char,
|
||
+ multi_wildcard_char=self._multi_wildcard_char)
|
||
+ else:
|
||
+ self._candidates = self.db.select_words(
|
||
+ tabkeys=self._chars_valid,
|
||
+ onechar=self._onechar,
|
||
+ chinese_mode=self._chinese_mode,
|
||
+ single_wildcard_char=self._single_wildcard_char,
|
||
+ multi_wildcard_char=self._multi_wildcard_char,
|
||
+ auto_wildcard=self._auto_wildcard)
|
||
+ # If only a wildcard character has been typed, insert a
|
||
+ # special candidate at the first position for the wildcard
|
||
+ # character itself. For example, if “?” is used as a
|
||
+ # wildcard character and this is the only character typed, add
|
||
+ # a candidate ('?', '?', 0, 1000000000) in halfwidth mode or a
|
||
+ # candidate ('?', '?', 0, 1000000000) in fullwidth mode.
|
||
+ # This is needed to make it possible to input the wildcard
|
||
+ # characters themselves, if “?” acted only as a wildcard
|
||
+ # it would be impossible to input a fullwidth question mark.
|
||
+ if (self._chars_valid
|
||
+ in [self._single_wildcard_char, self._multi_wildcard_char]):
|
||
+ wildcard_key = self._chars_valid
|
||
+ wildcard_phrase = self._chars_valid
|
||
+ if ascii_ispunct(wildcard_key):
|
||
+ if self._full_width_punct[1]:
|
||
+ wildcard_phrase = unichar_half_to_full(wildcard_phrase)
|
||
+ else:
|
||
+ wildcard_phrase = unichar_full_to_half(wildcard_phrase)
|
||
+ else:
|
||
+ if self._full_width_letter[1]:
|
||
+ wildcard_phrase = unichar_half_to_full(wildcard_phrase)
|
||
+ else:
|
||
+ wildcard_phrase = unichar_full_to_half(wildcard_phrase)
|
||
+ self._candidates.insert(
|
||
+ 0, (wildcard_key, wildcard_phrase, 0, 1000000000))
|
||
+ if self._candidates:
|
||
+ self.fill_lookup_table()
|
||
+ self._candidates_previous = self._candidates
|
||
+ return True
|
||
+ # There are only valid and no invalid input characters but no
|
||
+ # matching candidates could be found from the databases. The
|
||
+ # last of self._chars_valid must have caused this. That
|
||
+ # character is valid in the sense that it is listed in
|
||
+ # self._valid_input_chars, it is only invalid in the sense
|
||
+ # that after adding this character, no candidates could be
|
||
+ # found anymore. Add this character to self._chars_invalid
|
||
+ # and remove it from self._chars_valid.
|
||
+ self._chars_invalid += self._chars_valid[-1]
|
||
+ self._chars_valid = self._chars_valid[:-1]
|
||
+ self._chars_valid_update_candidates_last = self._chars_valid
|
||
+ self._chars_invalid_update_candidates_last = self._chars_invalid
|
||
+ return False
|
||
+
|
||
+ def commit_to_preedit(self):
|
||
+ '''Add selected phrase in lookup table to preëdit string'''
|
||
+ if not self._chars_valid:
|
||
+ return False
|
||
+ if self._candidates:
|
||
+ self._u_chars.insert(self._cursor_precommit,
|
||
+ self._candidates[self.get_cursor_pos()][0])
|
||
+ self._strings.insert(self._cursor_precommit,
|
||
+ self._candidates[self.get_cursor_pos()][1])
|
||
+ self._cursor_precommit += 1
|
||
+ self.clear_input_not_committed_to_preedit()
|
||
+ self.update_candidates()
|
||
+ return True
|
||
+
|
||
+ def commit_to_preedit_current_page(self, index):
|
||
+ '''
|
||
+ Commits the candidate at position “index” in the current
|
||
+ page of the lookup table to the preëdit. Does not yet “really”
|
||
+ commit the candidate, only to the preëdit.
|
||
+ '''
|
||
+ cursor_pos = self._lookup_table.get_cursor_pos()
|
||
+ cursor_in_page = self._lookup_table.get_cursor_in_page()
|
||
+ current_page_start = cursor_pos - cursor_in_page
|
||
+ real_index = current_page_start + index
|
||
+ if real_index >= len(self._candidates):
|
||
+ # the index given is out of range we do not commit anything
|
||
+ return False
|
||
+ self._lookup_table.set_cursor_pos(real_index)
|
||
+ return self.commit_to_preedit()
|
||
+
|
||
+ def get_aux_strings (self):
|
||
+ '''Get aux strings'''
|
||
+ input_chars = self.get_input_chars ()
|
||
+ if input_chars:
|
||
+ aux_string = input_chars
|
||
+ if debug_level > 0 and self._u_chars:
|
||
+ (tabkeys_left,
|
||
+ tabkeys_current,
|
||
+ tabkeys_right) = self.get_preedit_tabkeys_parts()
|
||
+ (strings_left,
|
||
+ string_current,
|
||
+ strings_right) = self.get_preedit_string_parts()
|
||
+ aux_string = u''
|
||
+ for i in range(0, len(strings_left)):
|
||
+ aux_string += (
|
||
+ u'('
|
||
+ + tabkeys_left[i] + u' '+ strings_left[i]
|
||
+ + u') ')
|
||
+ aux_string += input_chars
|
||
+ for i in range(0, len(strings_right)):
|
||
+ aux_string += (
|
||
+ u' ('
|
||
+ + tabkeys_right[i]+u' '+strings_right[i]
|
||
+ + u')')
|
||
+ if self._py_mode:
|
||
+ aux_string = aux_string.replace(
|
||
+ '!','1').replace(
|
||
+ '@','2').replace(
|
||
+ '#','3').replace(
|
||
+ '$','4').replace(
|
||
+ '%','5')
|
||
+ else:
|
||
+ aux_string_new = u''
|
||
+ for char in aux_string:
|
||
+ if char in self._prompt_characters:
|
||
+ aux_string_new += self._prompt_characters[char]
|
||
+ else:
|
||
+ aux_string_new += char
|
||
+ aux_string = aux_string_new
|
||
+ return aux_string
|
||
+
|
||
+ # There are no input strings at the moment. But there could
|
||
+ # be stuff committed to the preëdit. If there is something
|
||
+ # committed to the preëdit, show some information in the
|
||
+ # auxiliary text.
|
||
+ #
|
||
+ # For the character at the position of the cursor in the
|
||
+ # preëdit, show a list of possible input key sequences which
|
||
+ # could be used to type that character at the left side of the
|
||
+ # auxiliary text.
|
||
+ #
|
||
+ # If the preëdit is longer than one character, show the input
|
||
+ # key sequence which will be defined for the complete current
|
||
+ # contents of the preëdit, if the preëdit is committed.
|
||
+ aux_string = u''
|
||
+ if self._strings:
|
||
+ if self._cursor_precommit >= len(self._strings):
|
||
+ char = self._strings[-1][0]
|
||
+ else:
|
||
+ char = self._strings[self._cursor_precommit][0]
|
||
+ aux_string = u' '.join(self.db.find_zi_code(char))
|
||
+ cstr = u''.join(self._strings)
|
||
+ if self.db.user_can_define_phrase:
|
||
+ if len(cstr) > 1:
|
||
+ aux_string += (u'\t#: ' + self.db.parse_phrase(cstr))
|
||
+ aux_string_new = u''
|
||
+ for char in aux_string:
|
||
+ if char in self._prompt_characters:
|
||
+ aux_string_new += self._prompt_characters[char]
|
||
+ else:
|
||
+ aux_string_new += char
|
||
+ return aux_string_new
|
||
+
|
||
+ def fill_lookup_table(self):
|
||
+ '''Fill more entries to self._lookup_table if needed.
|
||
+
|
||
+ If the cursor in _lookup_table moved beyond current length,
|
||
+ add more entries from _candidiate[0] to _lookup_table.'''
|
||
+
|
||
+ looklen = self._lookup_table.get_number_of_candidates()
|
||
+ psize = self._lookup_table.get_page_size()
|
||
+ if (self._lookup_table.get_cursor_pos() + psize >= looklen and
|
||
+ looklen < len(self._candidates)):
|
||
+ endpos = looklen + psize
|
||
+ batch = self._candidates[looklen:endpos]
|
||
+ for x in batch:
|
||
+ self.append_candidate_to_lookup_table(
|
||
+ tabkeys=x[0], phrase=x[1], freq=x[2], user_freq=x[3])
|
||
+
|
||
+ def cursor_down(self):
|
||
+ '''Process Arrow Down Key Event
|
||
+ Move Lookup Table cursor down'''
|
||
+ self.fill_lookup_table()
|
||
+
|
||
+ res = self._lookup_table.cursor_down()
|
||
+ self.update_candidates ()
|
||
+ if not res and self._candidates:
|
||
+ return True
|
||
+ return res
|
||
+
|
||
+ def cursor_up(self):
|
||
+ '''Process Arrow Up Key Event
|
||
+ Move Lookup Table cursor up'''
|
||
+ res = self._lookup_table.cursor_up()
|
||
+ self.update_candidates ()
|
||
+ if not res and self._candidates:
|
||
+ return True
|
||
+ return res
|
||
+
|
||
+ def page_down(self):
|
||
+ '''Process Page Down Key Event
|
||
+ Move Lookup Table page down'''
|
||
+ self.fill_lookup_table()
|
||
+ res = self._lookup_table.page_down()
|
||
+ self.update_candidates ()
|
||
+ if not res and self._candidates:
|
||
+ return True
|
||
+ return res
|
||
+
|
||
+ def page_up(self):
|
||
+ '''Process Page Up Key Event
|
||
+ move Lookup Table page up'''
|
||
+ res = self._lookup_table.page_up()
|
||
+ self.update_candidates ()
|
||
+ if not res and self._candidates:
|
||
+ return True
|
||
+ return res
|
||
+
|
||
+ def select_key(self, keycode):
|
||
+ '''
|
||
+ Commit a candidate which was selected by typing a selection key
|
||
+ from the lookup table to the preedit. Does not yet “really”
|
||
+ commit the candidate, only to the preedit.
|
||
+ '''
|
||
+ if keycode not in self._select_keys:
|
||
+ return False
|
||
+ return self.commit_to_preedit_current_page(
|
||
+ self._select_keys.index(keycode))
|
||
+
|
||
+ def remove_candidate_from_user_database(self, keycode):
|
||
+ '''Remove a candidate displayed in the lookup table from the user
|
||
+ database.
|
||
+
|
||
+ The candidate indicated by the selection key with the key code
|
||
+ “keycode” is removed, if possible. If it is not in the user
|
||
+ database at all, nothing happens.
|
||
+
|
||
+ If this is a candidate which is also in the system database,
|
||
+ removing it from the user database only means that its user
|
||
+ frequency data is reset. It might still appear in subsequent
|
||
+ matches but with much lower priority.
|
||
+
|
||
+ If this is a candidate which is user defined and not in the system
|
||
+ database, it will not match at all anymore after removing it.
|
||
+
|
||
+ '''
|
||
+ if keycode not in self._select_keys:
|
||
+ return False
|
||
+ index = self._select_keys.index(keycode)
|
||
+ cursor_pos = self._lookup_table.get_cursor_pos()
|
||
+ cursor_in_page = self._lookup_table.get_cursor_in_page()
|
||
+ current_page_start = cursor_pos - cursor_in_page
|
||
+ real_index = current_page_start + index
|
||
+ if len(self._candidates) > real_index: # this index is valid
|
||
+ candidate = self._candidates[real_index]
|
||
+ self.db.remove_phrase(
|
||
+ tabkeys=candidate[0], phrase=candidate[1], commit=True)
|
||
+ # call update_candidates() to get a new SQL query. The
|
||
+ # input has not really changed, therefore we must clear
|
||
+ # the remembered list of characters to
|
||
+ # force update_candidates() to really do something and not
|
||
+ # return immediately:
|
||
+ self._chars_valid_update_candidates_last = u''
|
||
+ self._chars_invalid_update_candidates_last = u''
|
||
+ self.update_candidates()
|
||
+ return True
|
||
+ else:
|
||
+ return False
|
||
+
|
||
+ def get_cursor_pos (self):
|
||
+ '''get lookup table cursor position'''
|
||
+ return self._lookup_table.get_cursor_pos()
|
||
+
|
||
+ def get_lookup_table (self):
|
||
+ '''Get lookup table'''
|
||
+ return self._lookup_table
|
||
+
|
||
+ def remove_char(self):
|
||
+ '''Process remove_char Key Event'''
|
||
+ if debug_level > 1:
|
||
+ sys.stderr.write("remove_char()\n")
|
||
+ if self.get_input_chars():
|
||
+ self.pop_input ()
|
||
+ return
|
||
+ self.remove_preedit_character_before_cursor()
|
||
+
|
||
+ def delete(self):
|
||
+ '''Process delete Key Event'''
|
||
+ if self.get_input_chars():
|
||
+ return
|
||
+ self.remove_preedit_character_after_cursor()
|
||
+
|
||
+ def cycle_next_cand(self):
|
||
+ '''Cycle cursor to next candidate in the page.'''
|
||
+ total = len(self._candidates)
|
||
+
|
||
+ if total > 0:
|
||
+ page_size = self._lookup_table.get_page_size()
|
||
+ pos = self._lookup_table.get_cursor_pos()
|
||
+ page = int(pos/page_size)
|
||
+ pos += 1
|
||
+ if pos >= (page+1)*page_size or pos >= total:
|
||
+ pos = page*page_size
|
||
+ res = self._lookup_table.set_cursor_pos(pos)
|
||
+ return True
|
||
+ else:
|
||
+ return False
|
||
+
|
||
+ def one_candidate (self):
|
||
+ '''Return true if there is only one candidate'''
|
||
+ return len(self._candidates) == 1
|
||
+
|
||
+
|
||
+########################
|
||
+### Engine Class #####
|
||
+####################
|
||
+class tabengine (IBus.Engine):
|
||
+ '''The IM Engine for Tables'''
|
||
+
|
||
+ def __init__(self, bus, obj_path, db ):
|
||
+ super(tabengine, self).__init__(connection=bus.get_connection(),
|
||
+ object_path=obj_path)
|
||
+ global debug_level
|
||
+ try:
|
||
+ debug_level = int(os.getenv('IBUS_TABLE_DEBUG_LEVEL'))
|
||
+ except (TypeError, ValueError):
|
||
+ debug_level = int(0)
|
||
+ self._input_purpose = 0
|
||
+ self._has_input_purpose = False
|
||
+ if hasattr(IBus, 'InputPurpose'):
|
||
+ self._has_input_purpose = True
|
||
+ self._bus = bus
|
||
+ # this is the backend sql db we need for our IME
|
||
+ # we receive this db from IMEngineFactory
|
||
+ #self.db = tabsqlitedb.tabsqlitedb( name = dbname )
|
||
+ self.db = db
|
||
+ self._setup_pid = 0
|
||
+ self._icon_dir = '%s%s%s%s' % (os.getenv('IBUS_TABLE_LOCATION'),
|
||
+ os.path.sep, 'icons', os.path.sep)
|
||
+ # name for config section
|
||
+ self._engine_name = os.path.basename(
|
||
+ self.db.filename).replace('.db', '')
|
||
+ self._config_section = (
|
||
+ "engine/Table/%s" % self._engine_name.replace(' ','_'))
|
||
+
|
||
+ # config module
|
||
+ self._config = self._bus.get_config ()
|
||
+ self._config.connect ("value-changed", self.config_value_changed_cb)
|
||
+
|
||
+ # self._ime_py: Indicates whether this table supports pinyin mode
|
||
+ self._ime_py = self.db.ime_properties.get('pinyin_mode')
|
||
+ if self._ime_py:
|
||
+ if self._ime_py.lower() == u'true':
|
||
+ self._ime_py = True
|
||
+ else:
|
||
+ self._ime_py = False
|
||
+ else:
|
||
+ print('We could not find "pinyin_mode" entry in database, '
|
||
+ + 'is it an outdated database?')
|
||
+ self._ime_py = False
|
||
+
|
||
+ self._symbol = self.db.ime_properties.get('symbol')
|
||
+ if self._symbol == None or self._symbol == u'':
|
||
+ self._symbol = self.db.ime_properties.get('status_prompt')
|
||
+ if self._symbol == None:
|
||
+ self._symbol = u''
|
||
+ # some Chinese tables have “STATUS_PROMPT = CN” replace it
|
||
+ # with the shorter and nicer “中”:
|
||
+ if self._symbol == u'CN':
|
||
+ self._symbol = u'中'
|
||
+ # workaround for the translit and translit-ua tables which
|
||
+ # have 2 character symbols. '☑' + self._symbol then is
|
||
+ # 3 characters and currently gnome-shell ignores symbols longer
|
||
+ # than 3 characters:
|
||
+ if self._symbol == u'Ya':
|
||
+ self._symbol = u'Я'
|
||
+ if self._symbol == u'Yi':
|
||
+ self._symbol = u'Ї'
|
||
+ # now we check and update the valid input characters
|
||
+ self._valid_input_chars = self.db.ime_properties.get(
|
||
+ 'valid_input_chars')
|
||
+ self._pinyin_valid_input_chars = u'abcdefghijklmnopqrstuvwxyz!@#$%'
|
||
+
|
||
+ self._single_wildcard_char = variant_to_value(self._config.get_value(
|
||
+ self._config_section,
|
||
+ "singlewildcardchar"))
|
||
+ if self._single_wildcard_char == None:
|
||
+ self._single_wildcard_char = self.db.ime_properties.get(
|
||
+ 'single_wildcard_char')
|
||
+ if self._single_wildcard_char == None:
|
||
+ self._single_wildcard_char = u''
|
||
+ if len(self._single_wildcard_char) > 1:
|
||
+ self._single_wildcard_char = self._single_wildcard_char[0]
|
||
+
|
||
+ self._multi_wildcard_char = variant_to_value(self._config.get_value(
|
||
+ self._config_section,
|
||
+ "multiwildcardchar"))
|
||
+ if self._multi_wildcard_char == None:
|
||
+ self._multi_wildcard_char = self.db.ime_properties.get(
|
||
+ 'multi_wildcard_char')
|
||
+ if self._multi_wildcard_char == None:
|
||
+ self._multi_wildcard_char = u''
|
||
+ if len(self._multi_wildcard_char) > 1:
|
||
+ self._multi_wildcard_char = self._multi_wildcard_char[0]
|
||
+
|
||
+ self._auto_wildcard = variant_to_value(self._config.get_value(
|
||
+ self._config_section,
|
||
+ "autowildcard"))
|
||
+ if self._auto_wildcard == None:
|
||
+ self._auto_wildcard = self.db.ime_properties.get('auto_wildcard')
|
||
+ if self._auto_wildcard and self._auto_wildcard.lower() == u'false':
|
||
+ self._auto_wildcard = False
|
||
+ else:
|
||
+ self._auto_wildcard = True
|
||
+
|
||
+ self._max_key_length = int(self.db.ime_properties.get('max_key_length'))
|
||
+ self._max_key_length_pinyin = 7
|
||
+
|
||
+ self._page_up_keys = [
|
||
+ IBus.KEY_Page_Up,
|
||
+ IBus.KEY_KP_Page_Up,
|
||
+ IBus.KEY_minus
|
||
+ ]
|
||
+ self._page_down_keys = [
|
||
+ IBus.KEY_Page_Down,
|
||
+ IBus.KEY_KP_Page_Down,
|
||
+ IBus.KEY_equal
|
||
+ ]
|
||
+ # If page up or page down keys are defined in the database,
|
||
+ # use the values from the database instead of the above
|
||
+ # hardcoded defaults:
|
||
+ page_up_keys_csv = self.db.ime_properties.get('page_up_keys')
|
||
+ page_down_keys_csv = self.db.ime_properties.get('page_down_keys')
|
||
+ if page_up_keys_csv:
|
||
+ self._page_up_keys = [
|
||
+ IBus.keyval_from_name(x)
|
||
+ for x in page_up_keys_csv.split(',')]
|
||
+ if page_down_keys_csv:
|
||
+ self._page_down_keys = [
|
||
+ IBus.keyval_from_name(x)
|
||
+ for x in page_down_keys_csv.split(',')]
|
||
+ # Remove keys from the page up/down keys if they are needed
|
||
+ # for input (for example, '=' or '-' could well be needed for
|
||
+ # input. Input is more important):
|
||
+ for character in (
|
||
+ self._valid_input_chars
|
||
+ + self._single_wildcard_char
|
||
+ + self._multi_wildcard_char):
|
||
+ keyval = IBus.unicode_to_keyval(character)
|
||
+ if keyval in self._page_up_keys:
|
||
+ self._page_up_keys.remove(keyval)
|
||
+ if keyval in self._page_down_keys:
|
||
+ self._page_down_keys.remove(keyval)
|
||
+ self._commit_keys = [IBus.KEY_space]
|
||
+ # If commit keys are are defined in the database, use the
|
||
+ # value from the database instead of the above hardcoded
|
||
+ # default:
|
||
+ commit_keys_csv = self.db.ime_properties.get('commit_keys')
|
||
+ if commit_keys_csv:
|
||
+ self._commit_keys = [
|
||
+ IBus.keyval_from_name(x)
|
||
+ for x in commit_keys_csv.split(',')]
|
||
+ # If commit keys conflict with page up/down keys, remove them
|
||
+ # from the page up/down keys (They cannot really be used for
|
||
+ # both at the same time. Theoretically, keys from the page
|
||
+ # up/down keys could still be used to commit when the number
|
||
+ # of candidates is 0 because then there is nothing to
|
||
+ # page. But that would be only confusing):
|
||
+ for keyval in self._commit_keys:
|
||
+ if keyval in self._page_up_keys:
|
||
+ self._page_up_keys.remove(keyval)
|
||
+ if keyval in self._page_down_keys:
|
||
+ self._page_down_keys.remove(keyval)
|
||
+ # Finally, check the user setting, i.e. the config value
|
||
+ # “spacekeybehavior” and let the user have the last word
|
||
+ # how to use the space key:
|
||
+ spacekeybehavior = variant_to_value(self._config.get_value(
|
||
+ self._config_section,
|
||
+ "spacekeybehavior"))
|
||
+ if spacekeybehavior == True:
|
||
+ # space is used as a page down key and not as a commit key:
|
||
+ if IBus.KEY_space not in self._page_down_keys:
|
||
+ self._page_down_keys.append(IBus.KEY_space)
|
||
+ if IBus.KEY_space in self._commit_keys:
|
||
+ self._commit_keys.remove(IBus.KEY_space)
|
||
+ if spacekeybehavior == False:
|
||
+ # space is used as a commit key and not used as a page down key:
|
||
+ if IBus.KEY_space in self._page_down_keys:
|
||
+ self._page_down_keys.remove(IBus.KEY_space)
|
||
+ if IBus.KEY_space not in self._commit_keys:
|
||
+ self._commit_keys.append(IBus.KEY_space)
|
||
+ if debug_level > 1:
|
||
+ sys.stderr.write(
|
||
+ "self._page_down_keys=%s\n" %repr(self._page_down_keys))
|
||
+ sys.stderr.write(
|
||
+ "self._commit_keys=%s\n" %repr(self._commit_keys))
|
||
+
|
||
+ # 0 = Direct input, i.e. table input OFF (aka “English input mode”),
|
||
+ # most characters are just passed through to the application
|
||
+ # (but some fullwidth ↔ halfwidth conversion may be done even
|
||
+ # in this mode, depending on the settings)
|
||
+ # 1 = Table input ON (aka “Table input mode”, “Chinese mode”)
|
||
+ self._input_mode = variant_to_value(self._config.get_value(
|
||
+ self._config_section,
|
||
+ "inputmode"))
|
||
+ if self._input_mode == None:
|
||
+ self._input_mode = 1
|
||
+
|
||
+ # self._prev_key: hold the key event last time.
|
||
+ self._prev_key = None
|
||
+ self._prev_char = None
|
||
+ self._double_quotation_state = False
|
||
+ self._single_quotation_state = False
|
||
+
|
||
+ self._full_width_letter = [
|
||
+ variant_to_value(self._config.get_value(
|
||
+ self._config_section,
|
||
+ "EnDefFullWidthLetter")),
|
||
+ variant_to_value(self._config.get_value(
|
||
+ self._config_section,
|
||
+ "TabDefFullWidthLetter"))
|
||
+ ]
|
||
+ if self._full_width_letter[0] == None:
|
||
+ self._full_width_letter[0] = False
|
||
+ if self._full_width_letter[1] == None:
|
||
+ self._full_width_letter[1] = self.db.ime_properties.get(
|
||
+ 'def_full_width_letter').lower() == u'true'
|
||
+ self._full_width_punct = [
|
||
+ variant_to_value(self._config.get_value(
|
||
+ self._config_section,
|
||
+ "EnDefFullWidthPunct")),
|
||
+ variant_to_value(self._config.get_value(
|
||
+ self._config_section,
|
||
+ "TabDefFullWidthPunct"))
|
||
+ ]
|
||
+ if self._full_width_punct[0] == None:
|
||
+ self._full_width_punct[0] = False
|
||
+ if self._full_width_punct[1] == None:
|
||
+ self._full_width_punct[1] = self.db.ime_properties.get(
|
||
+ 'def_full_width_punct').lower() == u'true'
|
||
+
|
||
+ self._auto_commit = variant_to_value(self._config.get_value(
|
||
+ self._config_section,
|
||
+ "AutoCommit"))
|
||
+ if self._auto_commit == None:
|
||
+ self._auto_commit = self.db.ime_properties.get(
|
||
+ 'auto_commit').lower() == u'true'
|
||
+
|
||
+ # If auto select is true, then the first candidate phrase will
|
||
+ # be selected automatically during typing. Auto select is true
|
||
+ # by default for the stroke5 table for example.
|
||
+ self._auto_select = variant_to_value(self._config.get_value(
|
||
+ self._config_section,
|
||
+ "AutoSelect"))
|
||
+ if self._auto_select == None:
|
||
+ if self.db.ime_properties.get('auto_select') != None:
|
||
+ self._auto_select = self.db.ime_properties.get(
|
||
+ 'auto_select').lower() == u'true'
|
||
+ else:
|
||
+ self._auto_select = False
|
||
+
|
||
+ self._always_show_lookup = variant_to_value(self._config.get_value(
|
||
+ self._config_section,
|
||
+ "AlwaysShowLookup"))
|
||
+ if self._always_show_lookup == None:
|
||
+ if self.db.ime_properties.get('always_show_lookup') != None:
|
||
+ self._always_show_lookup = self.db.ime_properties.get(
|
||
+ 'always_show_lookup').lower() == u'true'
|
||
+ else:
|
||
+ self._always_show_lookup = True
|
||
+
|
||
+ self._editor = editor(self._config,
|
||
+ self._valid_input_chars,
|
||
+ self._pinyin_valid_input_chars,
|
||
+ self._single_wildcard_char,
|
||
+ self._multi_wildcard_char,
|
||
+ self._auto_wildcard,
|
||
+ self._full_width_letter,
|
||
+ self._full_width_punct,
|
||
+ self._max_key_length,
|
||
+ self.db)
|
||
+
|
||
+ self.chinese_mode_properties = {
|
||
+ 'ChineseMode.Simplified': {
|
||
+ # show simplified Chinese only
|
||
+ 'number': 0,
|
||
+ 'symbol': '簡',
|
||
+ 'icon': 'sc-mode.svg',
|
||
+ 'label': _('Simplified Chinese'),
|
||
+ 'tooltip':
|
||
+ _('Switch to “Simplified Chinese only”.')},
|
||
+ 'ChineseMode.Traditional': {
|
||
+ # show traditional Chinese only
|
||
+ 'number': 1,
|
||
+ 'symbol': '繁',
|
||
+ 'icon': 'tc-mode.svg',
|
||
+ 'label': _('Traditional Chinese'),
|
||
+ 'tooltip':
|
||
+ _('Switch to “Traditional Chinese only”.')},
|
||
+ 'ChineseMode.SimplifiedFirst': {
|
||
+ # show all but simplified first
|
||
+ 'number': 2,
|
||
+ 'symbol': '簡/大',
|
||
+ 'icon': 'scb-mode.svg',
|
||
+ 'label': _('Simplified Chinese first'),
|
||
+ 'tooltip':
|
||
+ _('Switch to “Simplified Chinese before traditional”.')},
|
||
+ 'ChineseMode.TraditionalFirst': {
|
||
+ # show all but traditional first
|
||
+ 'number': 3,
|
||
+ 'symbol': '繁/大',
|
||
+ 'icon': 'tcb-mode.svg',
|
||
+ 'label': _('Traditional Chinese first'),
|
||
+ 'tooltip':
|
||
+ _('Switch to “Traditional Chinese before simplified”.')},
|
||
+ 'ChineseMode.All': {
|
||
+ # show all Chinese characters, no particular order
|
||
+ 'number': 4,
|
||
+ 'symbol': '大',
|
||
+ 'icon': 'cb-mode.svg',
|
||
+ 'label': _('All Chinese characters'),
|
||
+ 'tooltip': _('Switch to “All Chinese characters”.')}
|
||
+ }
|
||
+ self.chinese_mode_menu = {
|
||
+ 'key': 'ChineseMode',
|
||
+ 'label': _('Chinese mode'),
|
||
+ 'tooltip': _('Switch Chinese mode'),
|
||
+ 'shortcut_hint': '(Ctrl-;)',
|
||
+ 'sub_properties': self.chinese_mode_properties
|
||
+ }
|
||
+ if self.db._is_chinese:
|
||
+ self.input_mode_properties = {
|
||
+ 'InputMode.Direct': {
|
||
+ 'number': 0,
|
||
+ 'symbol': '英',
|
||
+ 'icon': 'english.svg',
|
||
+ 'label': _('English'),
|
||
+ 'tooltip': _('Switch to English input')},
|
||
+ 'InputMode.Table': {
|
||
+ 'number': 1,
|
||
+ 'symbol': '中',
|
||
+ 'symbol_table': '中',
|
||
+ 'symbol_pinyin': '拼音',
|
||
+ 'icon': 'chinese.svg',
|
||
+ 'label': _('Chinese'),
|
||
+ 'tooltip': _('Switch to Chinese input')}
|
||
+ }
|
||
+ else:
|
||
+ self.input_mode_properties = {
|
||
+ 'InputMode.Direct': {
|
||
+ 'number': 0,
|
||
+ 'symbol': '☐' + self._symbol,
|
||
+ 'icon': 'english.svg',
|
||
+ 'label': _('Direct'),
|
||
+ 'tooltip': _('Switch to direct input')},
|
||
+ 'InputMode.Table': {
|
||
+ 'number': 1,
|
||
+ 'symbol': '☑' + self._symbol,
|
||
+ 'icon': 'ibus-table.svg',
|
||
+ 'label': _('Table'),
|
||
+ 'tooltip': _('Switch to table input')}
|
||
+ }
|
||
+ # The symbol of the property “InputMode” is displayed
|
||
+ # in the input method indicator of the Gnome3 panel.
|
||
+ # This depends on the property name “InputMode” and
|
||
+ # is case sensitive!
|
||
+ self.input_mode_menu = {
|
||
+ 'key': 'InputMode',
|
||
+ 'label': _('Input mode'),
|
||
+ 'tooltip': _('Switch Input mode'),
|
||
+ 'shortcut_hint': '(Left Shift)',
|
||
+ 'sub_properties': self.input_mode_properties
|
||
+ }
|
||
+ self.letter_width_properties = {
|
||
+ 'LetterWidth.Half': {
|
||
+ 'number': 0,
|
||
+ 'symbol': '◑',
|
||
+ 'icon': 'half-letter.svg',
|
||
+ 'label': _('Half'),
|
||
+ 'tooltip': _('Switch to halfwidth letters')},
|
||
+ 'LetterWidth.Full': {
|
||
+ 'number': 1,
|
||
+ 'symbol': '●',
|
||
+ 'icon': 'full-letter.svg',
|
||
+ 'label': _('Full'),
|
||
+ 'tooltip': _('Switch to fullwidth letters')}
|
||
+ }
|
||
+ self.letter_width_menu = {
|
||
+ 'key': 'LetterWidth',
|
||
+ 'label': _('Letter width'),
|
||
+ 'tooltip': _('Switch letter width'),
|
||
+ 'shortcut_hint': '(Shift-Space)',
|
||
+ 'sub_properties': self.letter_width_properties
|
||
+ }
|
||
+ self.punctuation_width_properties = {
|
||
+ 'PunctuationWidth.Half': {
|
||
+ 'number': 0,
|
||
+ 'symbol': ',.',
|
||
+ 'icon': 'half-punct.svg',
|
||
+ 'label': _('Half'),
|
||
+ 'tooltip': _('Switch to halfwidth punctuation')},
|
||
+ 'PunctuationWidth.Full': {
|
||
+ 'number': 1,
|
||
+ 'symbol': '、。',
|
||
+ 'icon': 'full-punct.svg',
|
||
+ 'label': _('Full'),
|
||
+ 'tooltip': _('Switch to fullwidth punctuation')}
|
||
+ }
|
||
+ self.punctuation_width_menu = {
|
||
+ 'key': 'PunctuationWidth',
|
||
+ 'label': _('Punctuation width'),
|
||
+ 'tooltip': _('Switch punctuation width'),
|
||
+ 'shortcut_hint': '(Ctrl-.)',
|
||
+ 'sub_properties': self.punctuation_width_properties
|
||
+ }
|
||
+ self.pinyin_mode_properties = {
|
||
+ 'PinyinMode.Table': {
|
||
+ 'number': 0,
|
||
+ 'symbol': '☐ 拼音',
|
||
+ 'icon': 'tab-mode.svg',
|
||
+ 'label': _('Table'),
|
||
+ 'tooltip': _('Switch to table mode')},
|
||
+ 'PinyinMode.Pinyin': {
|
||
+ 'number': 1,
|
||
+ 'symbol': '☑ 拼音',
|
||
+ 'icon': 'py-mode.svg',
|
||
+ 'label': _('Pinyin'),
|
||
+ 'tooltip': _('Switch to pinyin mode')}
|
||
+ }
|
||
+ self.pinyin_mode_menu = {
|
||
+ 'key': 'PinyinMode',
|
||
+ 'label': _('Pinyin mode'),
|
||
+ 'tooltip': _('Switch pinyin mode'),
|
||
+ 'shortcut_hint': '(Right Shift)',
|
||
+ 'sub_properties': self.pinyin_mode_properties
|
||
+ }
|
||
+ self.onechar_mode_properties = {
|
||
+ 'OneCharMode.Phrase': {
|
||
+ 'number': 0,
|
||
+ 'symbol': '☐ 1',
|
||
+ 'icon': 'phrase.svg',
|
||
+ 'label': _('Multiple character match'),
|
||
+ 'tooltip': _('Switch to matching multiple characters at once')},
|
||
+ 'OneCharMode.OneChar': {
|
||
+ 'number': 1,
|
||
+ 'symbol': '☑ 1',
|
||
+ 'icon': 'onechar.svg',
|
||
+ 'label': _('Single character match'),
|
||
+ 'tooltip': _('Switch to matching only single characters')}
|
||
+ }
|
||
+ self.onechar_mode_menu = {
|
||
+ 'key': 'OneCharMode',
|
||
+ 'label': _('Onechar mode'),
|
||
+ 'tooltip': _('Switch onechar mode'),
|
||
+ 'shortcut_hint': '(Ctrl-,)',
|
||
+ 'sub_properties': self.onechar_mode_properties
|
||
+ }
|
||
+ self.autocommit_mode_properties = {
|
||
+ 'AutoCommitMode.Direct': {
|
||
+ 'number': 0,
|
||
+ 'symbol': '☐ ↑',
|
||
+ 'icon': 'ncommit.svg',
|
||
+ 'label': _('Normal'),
|
||
+ 'tooltip':
|
||
+ _('Switch to normal commit mode '
|
||
+ + '(automatic commits go into the preedit '
|
||
+ + 'instead of into the application. '
|
||
+ + 'This enables automatic definitions of new shortcuts)')},
|
||
+ 'AutoCommitMode.Normal': {
|
||
+ 'number': 1,
|
||
+ 'symbol': '☑ ↑',
|
||
+ 'icon': 'acommit.svg',
|
||
+ 'label': _('Direct'),
|
||
+ 'tooltip':
|
||
+ _('Switch to direct commit mode '
|
||
+ + '(automatic commits go directly into the application)')}
|
||
+ }
|
||
+ self.autocommit_mode_menu = {
|
||
+ 'key': 'AutoCommitMode',
|
||
+ 'label': _('Auto commit mode'),
|
||
+ 'tooltip': _('Switch autocommit mode'),
|
||
+ 'shortcut_hint': '(Ctrl-/)',
|
||
+ 'sub_properties': self.autocommit_mode_properties
|
||
+ }
|
||
+ self._prop_dict = {}
|
||
+ self._init_properties()
|
||
+
|
||
+ self._on = False
|
||
+ self._save_user_count = 0
|
||
+ self._save_user_start = time.time()
|
||
+
|
||
+ self._save_user_count_max = SAVE_USER_COUNT_MAX
|
||
+ self._save_user_timeout = SAVE_USER_TIMEOUT
|
||
+ self.reset()
|
||
+
|
||
+ self.sync_timeout_id = GObject.timeout_add_seconds(1,
|
||
+ self._sync_user_db)
|
||
+
|
||
+ def reset(self):
|
||
+ self._editor.clear_all_input_and_preedit()
|
||
+ self._double_quotation_state = False
|
||
+ self._single_quotation_state = False
|
||
+ self._prev_key = None
|
||
+ self._update_ui()
|
||
+
|
||
+ def do_destroy(self):
|
||
+ if self.sync_timeout_id > 0:
|
||
+ GObject.source_remove(self.sync_timeout_id)
|
||
+ self.sync_timeout_id = 0
|
||
+ self.reset ()
|
||
+ self.do_focus_out ()
|
||
+ if self._save_user_count > 0:
|
||
+ self.db.sync_usrdb()
|
||
+ self._save_user_count = 0
|
||
+ super(tabengine, self).destroy()
|
||
+
|
||
+ def set_input_mode(self, mode=0):
|
||
+ if mode == self._input_mode:
|
||
+ return
|
||
+ self._input_mode = mode
|
||
+ # Not saved to config on purpose. In the setup tool one
|
||
+ # can select whether “Table input” or “Direct input” should
|
||
+ # be the default when the input method starts. But when
|
||
+ # changing this input mode using the property menu,
|
||
+ # the change is not remembered.
|
||
+ self._init_or_update_property_menu(
|
||
+ self.input_mode_menu,
|
||
+ self._input_mode)
|
||
+ # Letter width and punctuation width depend on the input mode.
|
||
+ # Therefore, the properties for letter width and punctuation
|
||
+ # width need to be updated here:
|
||
+ self._init_or_update_property_menu(
|
||
+ self.letter_width_menu,
|
||
+ self._full_width_letter[self._input_mode])
|
||
+ self._init_or_update_property_menu(
|
||
+ self.punctuation_width_menu,
|
||
+ self._full_width_punct[self._input_mode])
|
||
+ self.reset()
|
||
+
|
||
+ def set_pinyin_mode(self, mode=False):
|
||
+ if mode == self._editor._py_mode:
|
||
+ return
|
||
+ # The pinyin mode is never saved to config on purpose
|
||
+ self._editor.commit_to_preedit()
|
||
+ self._editor._py_mode = mode
|
||
+ self._init_or_update_property_menu(
|
||
+ self.pinyin_mode_menu, mode)
|
||
+ if mode:
|
||
+ self.input_mode_properties['InputMode.Table']['symbol'] = (
|
||
+ self.input_mode_properties['InputMode.Table']['symbol_pinyin'])
|
||
+ else:
|
||
+ self.input_mode_properties['InputMode.Table']['symbol'] = (
|
||
+ self.input_mode_properties['InputMode.Table']['symbol_table'])
|
||
+ self._init_or_update_property_menu(
|
||
+ self.input_mode_menu,
|
||
+ self._input_mode)
|
||
+ self._update_ui()
|
||
+
|
||
+ def set_onechar_mode(self, mode=False):
|
||
+ if mode == self._editor._onechar:
|
||
+ return
|
||
+ self._editor._onechar = mode
|
||
+ self._init_or_update_property_menu(
|
||
+ self.onechar_mode_menu, mode)
|
||
+ self._config.set_value(
|
||
+ self._config_section,
|
||
+ "OneChar",
|
||
+ GLib.Variant.new_boolean(mode))
|
||
+
|
||
+ def set_autocommit_mode(self, mode=False):
|
||
+ if mode == self._auto_commit:
|
||
+ return
|
||
+ self._auto_commit = mode
|
||
+ self._init_or_update_property_menu(
|
||
+ self.autocommit_mode_menu, mode)
|
||
+ self._config.set_value(
|
||
+ self._config_section,
|
||
+ "AutoCommit",
|
||
+ GLib.Variant.new_boolean(mode))
|
||
+
|
||
+ def set_letter_width(self, mode=False, input_mode=0):
|
||
+ if mode == self._full_width_letter[input_mode]:
|
||
+ return
|
||
+ self._full_width_letter[input_mode] = mode
|
||
+ self._editor._full_width_letter[input_mode] = mode
|
||
+ if input_mode == self._input_mode:
|
||
+ self._init_or_update_property_menu(
|
||
+ self.letter_width_menu, mode)
|
||
+ if input_mode:
|
||
+ self._config.set_value(
|
||
+ self._config_section,
|
||
+ "TabDefFullWidthLetter",
|
||
+ GLib.Variant.new_boolean(mode))
|
||
+ else:
|
||
+ self._config.set_value(
|
||
+ self._config_section,
|
||
+ "EnDefFullWidthLetter",
|
||
+ GLib.Variant.new_boolean(mode))
|
||
+
|
||
+ def set_punctuation_width(self, mode=False, input_mode=0):
|
||
+ if mode == self._full_width_punct[input_mode]:
|
||
+ return
|
||
+ self._full_width_punct[input_mode] = mode
|
||
+ self._editor._full_width_punct[input_mode] = mode
|
||
+ if input_mode == self._input_mode:
|
||
+ self._init_or_update_property_menu(
|
||
+ self.punctuation_width_menu, mode)
|
||
+ if input_mode:
|
||
+ self._config.set_value(
|
||
+ self._config_section,
|
||
+ "TabDefFullWidthPunct",
|
||
+ GLib.Variant.new_boolean(mode))
|
||
+ else:
|
||
+ self._config.set_value(
|
||
+ self._config_section,
|
||
+ "EnDefFullWidthPunct",
|
||
+ GLib.Variant.new_boolean(mode))
|
||
+
|
||
+ def set_chinese_mode(self, mode=0):
|
||
+ if mode == self._editor._chinese_mode:
|
||
+ return
|
||
+ self._editor._chinese_mode = mode
|
||
+ self._init_or_update_property_menu(
|
||
+ self.chinese_mode_menu, mode)
|
||
+ self._config.set_value(
|
||
+ self._config_section,
|
||
+ "ChineseMode",
|
||
+ GLib.Variant.new_int32(mode))
|
||
+
|
||
+ def _init_or_update_property_menu(self, menu, current_mode=0):
|
||
+ key = menu['key']
|
||
+ if key in self._prop_dict:
|
||
+ update_prop = True
|
||
+ else:
|
||
+ update_prop = False
|
||
+ sub_properties = menu['sub_properties']
|
||
+ for prop in sub_properties:
|
||
+ if sub_properties[prop]['number'] == int(current_mode):
|
||
+ symbol = sub_properties[prop]['symbol']
|
||
+ icon = sub_properties[prop]['icon']
|
||
+ label = '%(label)s (%(symbol)s) %(shortcut_hint)s' % {
|
||
+ 'label': menu['label'],
|
||
+ 'symbol': symbol,
|
||
+ 'shortcut_hint': menu['shortcut_hint']}
|
||
+ tooltip = '%(tooltip)s\n%(shortcut_hint)s' % {
|
||
+ 'tooltip': menu['tooltip'],
|
||
+ 'shortcut_hint': menu['shortcut_hint']}
|
||
+ self._prop_dict[key] = IBus.Property(
|
||
+ key=key,
|
||
+ prop_type=IBus.PropType.MENU,
|
||
+ label=IBus.Text.new_from_string(label),
|
||
+ symbol=IBus.Text.new_from_string(symbol),
|
||
+ icon=os.path.join(self._icon_dir, icon),
|
||
+ tooltip=IBus.Text.new_from_string(tooltip),
|
||
+ sensitive=True,
|
||
+ visible=True,
|
||
+ state=IBus.PropState.UNCHECKED,
|
||
+ sub_props=None)
|
||
+ self._prop_dict[key].set_sub_props(
|
||
+ self._init_sub_properties(
|
||
+ sub_properties, current_mode=current_mode))
|
||
+ if update_prop:
|
||
+ self.properties.update_property(self._prop_dict[key])
|
||
+ self.update_property(self._prop_dict[key])
|
||
+ else:
|
||
+ self.properties.append(self._prop_dict[key])
|
||
+
|
||
+ def _init_sub_properties(self, modes, current_mode=0):
|
||
+ sub_props = IBus.PropList()
|
||
+ for mode in sorted(modes, key=lambda x: (modes[x]['number'])):
|
||
+ sub_props.append(IBus.Property(
|
||
+ key=mode,
|
||
+ prop_type=IBus.PropType.RADIO,
|
||
+ label=IBus.Text.new_from_string(modes[mode]['label']),
|
||
+ icon=os.path.join(modes[mode]['icon']),
|
||
+ tooltip=IBus.Text.new_from_string(modes[mode]['tooltip']),
|
||
+ sensitive=True,
|
||
+ visible=True,
|
||
+ state=IBus.PropState.UNCHECKED,
|
||
+ sub_props=None))
|
||
+ i = 0
|
||
+ while sub_props.get(i) != None:
|
||
+ prop = sub_props.get(i)
|
||
+ key = prop.get_key()
|
||
+ self._prop_dict[key] = prop
|
||
+ if modes[key]['number'] == int(current_mode):
|
||
+ prop.set_state(IBus.PropState.CHECKED)
|
||
+ else:
|
||
+ prop.set_state(IBus.PropState.UNCHECKED)
|
||
+ self.update_property(prop) # important!
|
||
+ i += 1
|
||
+ return sub_props
|
||
+
|
||
+ def _init_properties(self):
|
||
+ self._prop_dict = {}
|
||
+ self.properties = IBus.PropList()
|
||
+
|
||
+ self._init_or_update_property_menu(
|
||
+ self.input_mode_menu,
|
||
+ self._input_mode)
|
||
+
|
||
+ if self.db._is_chinese and self._editor._chinese_mode != -1:
|
||
+ self._init_or_update_property_menu(
|
||
+ self.chinese_mode_menu,
|
||
+ self._editor._chinese_mode)
|
||
+
|
||
+ if self.db._is_cjk:
|
||
+ self._init_or_update_property_menu(
|
||
+ self.letter_width_menu,
|
||
+ self._full_width_letter[self._input_mode])
|
||
+ self._init_or_update_property_menu(
|
||
+ self.punctuation_width_menu,
|
||
+ self._full_width_punct[self._input_mode])
|
||
+
|
||
+ if self._ime_py:
|
||
+ self._init_or_update_property_menu(
|
||
+ self.pinyin_mode_menu,
|
||
+ self._editor._py_mode)
|
||
+
|
||
+ if self.db._is_cjk:
|
||
+ self._init_or_update_property_menu(
|
||
+ self.onechar_mode_menu,
|
||
+ self._editor._onechar)
|
||
+
|
||
+ if self.db.user_can_define_phrase and self.db.rules:
|
||
+ self._init_or_update_property_menu(
|
||
+ self.autocommit_mode_menu,
|
||
+ self._auto_commit)
|
||
+
|
||
+ self._setup_property = IBus.Property(
|
||
+ key = u'setup',
|
||
+ label = IBus.Text.new_from_string(_('Setup')),
|
||
+ icon = 'gtk-preferences',
|
||
+ tooltip = IBus.Text.new_from_string(_('Configure ibus-table “%(engine-name)s”') %{
|
||
+ 'engine-name': self._engine_name}),
|
||
+ sensitive = True,
|
||
+ visible = True)
|
||
+ self.properties.append(self._setup_property)
|
||
+
|
||
+ self.register_properties(self.properties)
|
||
+
|
||
+ def do_property_activate(
|
||
+ self, property, prop_state = IBus.PropState.UNCHECKED):
|
||
+ '''
|
||
+ Handle clicks on properties
|
||
+ '''
|
||
+ if debug_level > 1:
|
||
+ sys.stderr.write(
|
||
+ "do_property_activate() property=%(p)s prop_state=%(ps)s\n"
|
||
+ % {'p': property, 'ps': prop_state})
|
||
+ if property == "setup":
|
||
+ self._start_setup()
|
||
+ return
|
||
+ if prop_state != IBus.PropState.CHECKED:
|
||
+ # If the mouse just hovered over a menu button and
|
||
+ # no sub-menu entry was clicked, there is nothing to do:
|
||
+ return
|
||
+ if property.startswith(self.input_mode_menu['key']+'.'):
|
||
+ self.set_input_mode(
|
||
+ self.input_mode_properties[property]['number'])
|
||
+ return
|
||
+ if (property.startswith(self.pinyin_mode_menu['key']+'.')
|
||
+ and self._ime_py):
|
||
+ self.set_pinyin_mode(
|
||
+ bool(self.pinyin_mode_properties[property]['number']))
|
||
+ return
|
||
+ if (property.startswith(self.onechar_mode_menu['key']+'.')
|
||
+ and self.db._is_cjk):
|
||
+ self.set_onechar_mode(
|
||
+ bool(self.onechar_mode_properties[property]['number']))
|
||
+ return
|
||
+ if (property.startswith(self.autocommit_mode_menu['key']+'.')
|
||
+ and self.db.user_can_define_phrase and self.db.rules):
|
||
+ self.set_autocommit_mode(
|
||
+ bool(self.autocommit_mode_properties[property]['number']))
|
||
+ return
|
||
+ if (property.startswith(self.letter_width_menu['key']+'.')
|
||
+ and self.db._is_cjk):
|
||
+ self.set_letter_width(
|
||
+ bool(self.letter_width_properties[property]['number']),
|
||
+ input_mode=self._input_mode)
|
||
+ return
|
||
+ if (property.startswith(self.punctuation_width_menu['key']+'.')
|
||
+ and self.db._is_cjk):
|
||
+ self.set_punctuation_width(
|
||
+ bool(self.punctuation_width_properties[property]['number']),
|
||
+ input_mode=self._input_mode)
|
||
+ return
|
||
+ if (property.startswith(self.chinese_mode_menu['key']+'.')
|
||
+ and self.db._is_chinese
|
||
+ and self._editor._chinese_mode != -1):
|
||
+ self.set_chinese_mode(
|
||
+ self.chinese_mode_properties[property]['number'])
|
||
+ return
|
||
+
|
||
+ def _start_setup(self):
|
||
+ if self._setup_pid != 0:
|
||
+ pid, state = os.waitpid(self._setup_pid, os.P_NOWAIT)
|
||
+ if pid != self._setup_pid:
|
||
+ # If the last setup tool started from here is still
|
||
+ # running the pid returned by the above os.waitpid()
|
||
+ # is 0. In that case just return, don’t start a
|
||
+ # second setup tool.
|
||
+ return
|
||
+ self._setup_pid = 0
|
||
+ setup_cmd = os.path.join(
|
||
+ os.getenv('IBUS_TABLE_LIB_LOCATION'),
|
||
+ 'ibus-setup-table')
|
||
+ self._setup_pid = os.spawnl(
|
||
+ os.P_NOWAIT,
|
||
+ setup_cmd,
|
||
+ 'ibus-setup-table',
|
||
+ '--engine-name table:%s' %self._engine_name)
|
||
+
|
||
+ def _update_preedit(self):
|
||
+ '''Update Preedit String in UI'''
|
||
+ preedit_string_parts = self._editor.get_preedit_string_parts()
|
||
+ left_of_current_edit = u''.join(preedit_string_parts[0])
|
||
+ current_edit = preedit_string_parts[1]
|
||
+ right_of_current_edit = u''.join(preedit_string_parts[2])
|
||
+ if not self._editor._py_mode:
|
||
+ current_edit_new = u''
|
||
+ for char in current_edit:
|
||
+ if char in self._editor._prompt_characters:
|
||
+ current_edit_new += self._editor._prompt_characters[char]
|
||
+ else:
|
||
+ current_edit_new += char
|
||
+ current_edit = current_edit_new
|
||
+ preedit_string_complete = (
|
||
+ left_of_current_edit + current_edit + right_of_current_edit)
|
||
+ if not preedit_string_complete:
|
||
+ super(tabengine, self).update_preedit_text(
|
||
+ IBus.Text.new_from_string(u''), 0, False)
|
||
+ return
|
||
+ color_left = rgb(0xf9, 0x0f, 0x0f) # bright red
|
||
+ color_right = rgb(0x1e, 0xdc, 0x1a) # light green
|
||
+ color_invalid = rgb(0xff, 0x00, 0xff) # magenta
|
||
+ attrs = IBus.AttrList()
|
||
+ attrs.append(
|
||
+ IBus.attr_foreground_new(
|
||
+ color_left,
|
||
+ 0,
|
||
+ len(left_of_current_edit)))
|
||
+ attrs.append(
|
||
+ IBus.attr_foreground_new(
|
||
+ color_right,
|
||
+ len(left_of_current_edit) + len(current_edit),
|
||
+ len(preedit_string_complete)))
|
||
+ if self._editor._chars_invalid:
|
||
+ attrs.append(
|
||
+ IBus.attr_foreground_new(
|
||
+ color_invalid,
|
||
+ len(left_of_current_edit) + len(current_edit)
|
||
+ - len(self._editor._chars_invalid),
|
||
+ len(left_of_current_edit) + len(current_edit)
|
||
+ ))
|
||
+ attrs.append(
|
||
+ IBus.attr_underline_new(
|
||
+ IBus.AttrUnderline.SINGLE,
|
||
+ 0,
|
||
+ len(preedit_string_complete)))
|
||
+ text = IBus.Text.new_from_string(preedit_string_complete)
|
||
+ i = 0
|
||
+ while attrs.get(i) != None:
|
||
+ attr = attrs.get(i)
|
||
+ text.append_attribute(attr.get_attr_type(),
|
||
+ attr.get_value(),
|
||
+ attr.get_start_index(),
|
||
+ attr.get_end_index())
|
||
+ i += 1
|
||
+ super(tabengine, self).update_preedit_text(
|
||
+ text, self._editor.get_caret(), True)
|
||
+
|
||
+ def _update_aux (self):
|
||
+ '''Update Aux String in UI'''
|
||
+ aux_string = self._editor.get_aux_strings()
|
||
+ if len(self._editor._candidates) > 0:
|
||
+ aux_string += u' (%d / %d)' % (
|
||
+ self._editor._lookup_table.get_cursor_pos() +1,
|
||
+ self._editor._lookup_table.get_number_of_candidates())
|
||
+ if aux_string:
|
||
+ attrs = IBus.AttrList()
|
||
+ attrs.append(IBus.attr_foreground_new(
|
||
+ rgb(0x95,0x15,0xb5),0, len(aux_string)))
|
||
+ text = IBus.Text.new_from_string(aux_string)
|
||
+ i = 0
|
||
+ while attrs.get(i) != None:
|
||
+ attr = attrs.get(i)
|
||
+ text.append_attribute(attr.get_attr_type(),
|
||
+ attr.get_value(),
|
||
+ attr.get_start_index(),
|
||
+ attr.get_end_index())
|
||
+ i += 1
|
||
+ visible = True
|
||
+ if not aux_string or not self._always_show_lookup:
|
||
+ visible = False
|
||
+ super(tabengine, self).update_auxiliary_text(text, visible)
|
||
+ else:
|
||
+ self.hide_auxiliary_text()
|
||
+
|
||
+ def _update_lookup_table (self):
|
||
+ '''Update Lookup Table in UI'''
|
||
+ if len(self._editor._candidates) == 0:
|
||
+ # Also make sure to hide lookup table if there are
|
||
+ # no candidates to display. On f17, this makes no
|
||
+ # difference but gnome-shell in f18 will display
|
||
+ # an empty suggestion popup if the number of candidates
|
||
+ # is zero!
|
||
+ self.hide_lookup_table()
|
||
+ return
|
||
+ if self._editor.is_empty ():
|
||
+ self.hide_lookup_table()
|
||
+ return
|
||
+ if not self._always_show_lookup:
|
||
+ self.hide_lookup_table()
|
||
+ return
|
||
+ self.update_lookup_table(self._editor.get_lookup_table(), True)
|
||
+
|
||
+ def _update_ui (self):
|
||
+ '''Update User Interface'''
|
||
+ self._update_lookup_table ()
|
||
+ self._update_preedit ()
|
||
+ self._update_aux ()
|
||
+
|
||
+ def _check_phrase (self, tabkeys=u'', phrase=u''):
|
||
+ """Check the given phrase and update save user db info"""
|
||
+ if not tabkeys or not phrase:
|
||
+ return
|
||
+ self.db.check_phrase(tabkeys=tabkeys, phrase=phrase)
|
||
+
|
||
+ if self._save_user_count <= 0:
|
||
+ self._save_user_start = time.time()
|
||
+ self._save_user_count += 1
|
||
+
|
||
+ def _sync_user_db(self):
|
||
+ """Save user db to disk"""
|
||
+ if self._save_user_count >= 0:
|
||
+ now = time.time()
|
||
+ time_delta = now - self._save_user_start
|
||
+ if (self._save_user_count > self._save_user_count_max or
|
||
+ time_delta >= self._save_user_timeout):
|
||
+ self.db.sync_usrdb()
|
||
+ self._save_user_count = 0
|
||
+ self._save_user_start = now
|
||
+ return True
|
||
+
|
||
+ def commit_string (self, phrase, tabkeys=u''):
|
||
+ if debug_level > 1:
|
||
+ sys.stderr.write("commit_string() phrase=%(p)s\n"
|
||
+ %{'p': phrase})
|
||
+ self._editor.clear_all_input_and_preedit()
|
||
+ self._update_ui()
|
||
+ super(tabengine, self).commit_text(IBus.Text.new_from_string(phrase))
|
||
+ if len(phrase) > 0:
|
||
+ self._prev_char = phrase[-1]
|
||
+ else:
|
||
+ self._prev_char = None
|
||
+ self._check_phrase(tabkeys=tabkeys, phrase=phrase)
|
||
+
|
||
+ def commit_everything_unless_invalid(self):
|
||
+ '''
|
||
+ Commits the current input to the preëdit and then
|
||
+ commits the preëdit to the application unless there are
|
||
+ invalid input characters.
|
||
+
|
||
+ Returns “True” if something was committed, “False” if not.
|
||
+ '''
|
||
+ if debug_level > 1:
|
||
+ sys.stderr.write("commit_everything_unless_invalid()\n")
|
||
+ if self._editor._chars_invalid:
|
||
+ return False
|
||
+ if not self._editor.is_empty():
|
||
+ self._editor.commit_to_preedit()
|
||
+ self.commit_string(self._editor.get_preedit_string_complete(),
|
||
+ tabkeys=self._editor.get_preedit_tabkeys_complete())
|
||
+ return True
|
||
+
|
||
+ def _convert_to_full_width(self, c):
|
||
+ '''Convert half width character to full width'''
|
||
+
|
||
+ # This function handles punctuation that does not comply to the
|
||
+ # Unicode conversion formula in unichar_half_to_full(c).
|
||
+ # For ".", "\"", "'"; there are even variations under specific
|
||
+ # cases. This function should be more abstracted by extracting
|
||
+ # that to another handling function later on.
|
||
+ special_punct_dict = {u"<": u"《", # 《 U+300A LEFT DOUBLE ANGLE BRACKET
|
||
+ u">": u"》", # 》 U+300B RIGHT DOUBLE ANGLE BRACKET
|
||
+ u"[": u"「", # 「 U+300C LEFT CORNER BRACKET
|
||
+ u"]": u"」", # 」U+300D RIGHT CORNER BRACKET
|
||
+ u"{": u"『", # 『 U+300E LEFT WHITE CORNER BRACKET
|
||
+ u"}": u"』", # 』U+300F RIGHT WHITE CORNER BRACKET
|
||
+ u"\\": u"、", # 、 U+3001 IDEOGRAPHIC COMMA
|
||
+ u"^": u"……", # … U+2026 HORIZONTAL ELLIPSIS
|
||
+ u"_": u"——", # — U+2014 EM DASH
|
||
+ u"$": u"¥" # ¥ U+FFE5 FULLWIDTH YEN SIGN
|
||
+ }
|
||
+
|
||
+ # special puncts w/o further conditions
|
||
+ if c in special_punct_dict.keys():
|
||
+ if c in [u"\\", u"^", u"_", u"$"]:
|
||
+ return special_punct_dict[c]
|
||
+ elif self._input_mode:
|
||
+ return special_punct_dict[c]
|
||
+
|
||
+ # special puncts w/ further conditions
|
||
+ if c == u".":
|
||
+ if (self._prev_char
|
||
+ and self._prev_char.isdigit()
|
||
+ and self._prev_key
|
||
+ and chr(self._prev_key.val) == self._prev_char):
|
||
+ return u"."
|
||
+ else:
|
||
+ return u"。" # 。U+3002 IDEOGRAPHIC FULL STOP
|
||
+ elif c == u"\"":
|
||
+ self._double_quotation_state = not self._double_quotation_state
|
||
+ if self._double_quotation_state:
|
||
+ return u"“" # “ U+201C LEFT DOUBLE QUOTATION MARK
|
||
+ else:
|
||
+ return u"”" # ” U+201D RIGHT DOUBLE QUOTATION MARK
|
||
+ elif c == u"'":
|
||
+ self._single_quotation_state = not self._single_quotation_state
|
||
+ if self._single_quotation_state:
|
||
+ return u"‘" # ‘ U+2018 LEFT SINGLE QUOTATION MARK
|
||
+ else:
|
||
+ return u"’" # ’ U+2019 RIGHT SINGLE QUOTATION MARK
|
||
+
|
||
+ return unichar_half_to_full(c)
|
||
+
|
||
+ def _match_hotkey (self, key, keyval, state):
|
||
+
|
||
+ # Match only when keys are released
|
||
+ state = state | IBus.ModifierType.RELEASE_MASK
|
||
+ if key.val == keyval and (key.state & state) == state:
|
||
+ # If it is a key release event, the previous key
|
||
+ # must have been the same key pressed down.
|
||
+ if (self._prev_key
|
||
+ and key.val == self._prev_key.val):
|
||
+ return True
|
||
+
|
||
+ return False
|
||
+
|
||
+ def do_candidate_clicked(self, index, button, state):
|
||
+ if self._editor.commit_to_preedit_current_page(index):
|
||
+ # commits to preëdit
|
||
+ self.commit_string(
|
||
+ self._editor.get_preedit_string_complete(),
|
||
+ tabkeys=self._editor.get_preedit_tabkeys_complete())
|
||
+ return True
|
||
+ return False
|
||
+
|
||
+ def do_process_key_event(self, keyval, keycode, state):
|
||
+ '''Process Key Events
|
||
+ Key Events include Key Press and Key Release,
|
||
+ modifier means Key Pressed
|
||
+ '''
|
||
+ if debug_level > 1:
|
||
+ sys.stderr.write("do_process_key_event()\n")
|
||
+ if (self._has_input_purpose
|
||
+ and self._input_purpose
|
||
+ in [IBus.InputPurpose.PASSWORD, IBus.InputPurpose.PIN]):
|
||
+ return False
|
||
+
|
||
+ key = KeyEvent(keyval, keycode, state)
|
||
+
|
||
+ result = self._process_key_event (key)
|
||
+ self._prev_key = key
|
||
+ return result
|
||
+
|
||
+ def _process_key_event (self, key):
|
||
+ '''Internal method to process key event'''
|
||
+ # Match mode switch hotkey
|
||
+ if (self._editor.is_empty()
|
||
+ and (self._match_hotkey(
|
||
+ key, IBus.KEY_Shift_L,
|
||
+ IBus.ModifierType.SHIFT_MASK))):
|
||
+ self.set_input_mode(int(not self._input_mode))
|
||
+ return True
|
||
+
|
||
+ # Match fullwidth/halfwidth letter mode switch hotkey
|
||
+ if self.db._is_cjk:
|
||
+ if (key.val == IBus.KEY_space
|
||
+ and key.state & IBus.ModifierType.SHIFT_MASK
|
||
+ and not key.state & IBus.ModifierType.RELEASE_MASK):
|
||
+ # Ignore when Shift+Space was pressed, the key release
|
||
+ # event will toggle the fullwidth/halfwidth letter mode, we
|
||
+ # don’t want to insert an extra space on the key press
|
||
+ # event.
|
||
+ return True
|
||
+ if (self._match_hotkey(
|
||
+ key, IBus.KEY_space,
|
||
+ IBus.ModifierType.SHIFT_MASK)):
|
||
+ self.set_letter_width(
|
||
+ not self._full_width_letter[self._input_mode],
|
||
+ input_mode = self._input_mode)
|
||
+ return True
|
||
+
|
||
+ # Match full half punct mode switch hotkey
|
||
+ if (self._match_hotkey(
|
||
+ key, IBus.KEY_period,
|
||
+ IBus.ModifierType.CONTROL_MASK) and self.db._is_cjk):
|
||
+ self.set_punctuation_width(
|
||
+ not self._full_width_punct[self._input_mode],
|
||
+ input_mode = self._input_mode)
|
||
+ return True
|
||
+
|
||
+ if self._input_mode:
|
||
+ return self._table_mode_process_key_event (key)
|
||
+ else:
|
||
+ return self._english_mode_process_key_event (key)
|
||
+
|
||
+ def cond_letter_translate(self, char):
|
||
+ if self._full_width_letter[self._input_mode] and self.db._is_cjk:
|
||
+ return self._convert_to_full_width(char)
|
||
+ else:
|
||
+ return char
|
||
+
|
||
+ def cond_punct_translate(self, char):
|
||
+ if self._full_width_punct[self._input_mode] and self.db._is_cjk:
|
||
+ return self._convert_to_full_width(char)
|
||
+ else:
|
||
+ return char
|
||
+
|
||
+ def _english_mode_process_key_event(self, key):
|
||
+ # Ignore key release events
|
||
+ if key.state & IBus.ModifierType.RELEASE_MASK:
|
||
+ return False
|
||
+ if key.val >= 128:
|
||
+ return False
|
||
+ # we ignore all hotkeys here
|
||
+ if (key.state
|
||
+ & (IBus.ModifierType.CONTROL_MASK|IBus.ModifierType.MOD1_MASK)):
|
||
+ return False
|
||
+ keychar = IBus.keyval_to_unicode(key.val)
|
||
+ if type(keychar) != type(u''):
|
||
+ keychar = keychar.decode('UTF-8')
|
||
+ if ascii_ispunct(keychar):
|
||
+ trans_char = self.cond_punct_translate(keychar)
|
||
+ else:
|
||
+ trans_char = self.cond_letter_translate(keychar)
|
||
+ if trans_char == keychar:
|
||
+ return False
|
||
+ self.commit_string(trans_char)
|
||
+ return True
|
||
+
|
||
+ def _table_mode_process_key_event(self, key):
|
||
+ if debug_level > 0:
|
||
+ sys.stderr.write('_table_mode_process_key_event() ')
|
||
+ sys.stderr.write('repr(key)=%(key)s\n' %{'key': key})
|
||
+ # Change pinyin mode
|
||
+ # (change only if the editor is empty. When the editor
|
||
+ # is not empty, the right shift key should commit to preëdit
|
||
+ # and not change the pinyin mode).
|
||
+ if (self._ime_py
|
||
+ and self._editor.is_empty()
|
||
+ and self._match_hotkey(
|
||
+ key, IBus.KEY_Shift_R,
|
||
+ IBus.ModifierType.SHIFT_MASK)):
|
||
+ self.set_pinyin_mode(not self._editor._py_mode)
|
||
+ return True
|
||
+ # process commit to preedit
|
||
+ if (self._match_hotkey(
|
||
+ key, IBus.KEY_Shift_R,
|
||
+ IBus.ModifierType.SHIFT_MASK)
|
||
+ or self._match_hotkey(
|
||
+ key, IBus.KEY_Shift_L,
|
||
+ IBus.ModifierType.SHIFT_MASK)):
|
||
+ res = self._editor.commit_to_preedit()
|
||
+ self._update_ui()
|
||
+ return res
|
||
+
|
||
+ # Left ALT key to cycle candidates in the current page.
|
||
+ if (self._match_hotkey(
|
||
+ key, IBus.KEY_Alt_L,
|
||
+ IBus.ModifierType.MOD1_MASK)):
|
||
+ res = self._editor.cycle_next_cand()
|
||
+ self._update_ui()
|
||
+ return res
|
||
+
|
||
+ # Match single char mode switch hotkey
|
||
+ if (self._match_hotkey(
|
||
+ key, IBus.KEY_comma,
|
||
+ IBus.ModifierType.CONTROL_MASK) and self.db._is_cjk):
|
||
+ self.set_onechar_mode(not self._editor._onechar)
|
||
+ return True
|
||
+
|
||
+ # Match direct commit mode switch hotkey
|
||
+ if (self._match_hotkey(
|
||
+ key, IBus.KEY_slash,
|
||
+ IBus.ModifierType.CONTROL_MASK)
|
||
+ and self.db.user_can_define_phrase and self.db.rules):
|
||
+ self.set_autocommit_mode(not self._auto_commit)
|
||
+ return True
|
||
+
|
||
+ # Match Chinese mode shift
|
||
+ if (self._match_hotkey(
|
||
+ key, IBus.KEY_semicolon,
|
||
+ IBus.ModifierType.CONTROL_MASK) and self.db._is_chinese):
|
||
+ self.set_chinese_mode((self._editor._chinese_mode+1) % 5)
|
||
+ return True
|
||
+
|
||
+ # Ignore key release events
|
||
+ # (Must be below all self._match_hotkey() callse
|
||
+ # because these match on a release event).
|
||
+ if key.state & IBus.ModifierType.RELEASE_MASK:
|
||
+ return False
|
||
+
|
||
+ keychar = IBus.keyval_to_unicode(key.val)
|
||
+ if type(keychar) != type(u''):
|
||
+ keychar = keychar.decode('UTF-8')
|
||
+
|
||
+ # Section to handle leading invalid input:
|
||
+ #
|
||
+ # This is the first character typed, if it is invalid
|
||
+ # input, handle it immediately here, if it is valid, continue.
|
||
+ if (self._editor.is_empty()
|
||
+ and not self._editor.get_preedit_string_complete()):
|
||
+ if ((keychar not in (
|
||
+ self._valid_input_chars
|
||
+ + self._single_wildcard_char
|
||
+ + self._multi_wildcard_char)
|
||
+ or (self.db.startchars and keychar not in self.db.startchars))
|
||
+ and (not key.state &
|
||
+ (IBus.ModifierType.MOD1_MASK |
|
||
+ IBus.ModifierType.CONTROL_MASK))):
|
||
+ if debug_level > 0:
|
||
+ sys.stderr.write(
|
||
+ '_table_mode_process_key_event() '
|
||
+ + 'leading invalid input: '
|
||
+ + 'repr(keychar)=%(keychar)s\n'
|
||
+ % {'keychar': keychar})
|
||
+ if ascii_ispunct(keychar):
|
||
+ trans_char = self.cond_punct_translate(keychar)
|
||
+ else:
|
||
+ trans_char = self.cond_letter_translate(keychar)
|
||
+ if trans_char == keychar:
|
||
+ self._prev_char = trans_char
|
||
+ return False
|
||
+ else:
|
||
+ self.commit_string(trans_char)
|
||
+ return True
|
||
+
|
||
+ if key.val == IBus.KEY_Escape:
|
||
+ self.reset()
|
||
+ self._update_ui()
|
||
+ return True
|
||
+
|
||
+ if key.val in (IBus.KEY_Return, IBus.KEY_KP_Enter):
|
||
+ if (self._editor.is_empty()
|
||
+ and not self._editor.get_preedit_string_complete()):
|
||
+ # When IBus.KEY_Return is typed,
|
||
+ # IBus.keyval_to_unicode(key.val) returns a non-empty
|
||
+ # string. But when IBus.KEY_KP_Enter is typed it
|
||
+ # returns an empty string. Therefore, when typing
|
||
+ # IBus.KEY_KP_Enter as leading input, the key is not
|
||
+ # handled by the section to handle leading invalid
|
||
+ # input but it ends up here. If it is leading input
|
||
+ # (i.e. the preëdit is empty) we should always pass
|
||
+ # IBus.KEY_KP_Enter to the application:
|
||
+ return False
|
||
+ if self._auto_select:
|
||
+ self._editor.commit_to_preedit()
|
||
+ commit_string = self._editor.get_preedit_string_complete()
|
||
+ self.commit_string(commit_string)
|
||
+ return False
|
||
+ else:
|
||
+ commit_string = self._editor.get_preedit_tabkeys_complete()
|
||
+ self.commit_string(commit_string)
|
||
+ return True
|
||
+
|
||
+ if key.val in (IBus.KEY_Tab, IBus.KEY_KP_Tab) and self._auto_select:
|
||
+ # Used for example for the Russian transliteration method
|
||
+ # “translit”, which uses “auto select”. If for example
|
||
+ # a file with the name “шшш” exists and one types in
|
||
+ # a bash shell:
|
||
+ #
|
||
+ # “ls sh”
|
||
+ #
|
||
+ # the “sh” is converted to “ш” and one sees
|
||
+ #
|
||
+ # “ls ш”
|
||
+ #
|
||
+ # in the shell where the “ш” is still in preëdit
|
||
+ # because “shh” would be converted to “щ”, i.e. there
|
||
+ # is more than one candidate and the input method is still
|
||
+ # waiting whether one more “h” will be typed or not. But
|
||
+ # if the next character typed is a Tab, the preëdit is
|
||
+ # committed here and “False” is returned to pass the Tab
|
||
+ # character through to the bash to complete the file name
|
||
+ # to “шшш”.
|
||
+ self._editor.commit_to_preedit()
|
||
+ self.commit_string(self._editor.get_preedit_string_complete())
|
||
+ return False
|
||
+
|
||
+ if key.val in (IBus.KEY_Down, IBus.KEY_KP_Down) :
|
||
+ if not self._editor.get_preedit_string_complete():
|
||
+ return False
|
||
+ res = self._editor.cursor_down()
|
||
+ self._update_ui()
|
||
+ return res
|
||
+
|
||
+ if key.val in (IBus.KEY_Up, IBus.KEY_KP_Up):
|
||
+ if not self._editor.get_preedit_string_complete():
|
||
+ return False
|
||
+ res = self._editor.cursor_up()
|
||
+ self._update_ui()
|
||
+ return res
|
||
+
|
||
+ if (key.val in (IBus.KEY_Left, IBus.KEY_KP_Left)
|
||
+ and key.state & IBus.ModifierType.CONTROL_MASK):
|
||
+ if not self._editor.get_preedit_string_complete():
|
||
+ return False
|
||
+ self._editor.control_arrow_left()
|
||
+ self._update_ui()
|
||
+ return True
|
||
+
|
||
+ if (key.val in (IBus.KEY_Right, IBus.KEY_KP_Right)
|
||
+ and key.state & IBus.ModifierType.CONTROL_MASK):
|
||
+ if not self._editor.get_preedit_string_complete():
|
||
+ return False
|
||
+ self._editor.control_arrow_right()
|
||
+ self._update_ui()
|
||
+ return True
|
||
+
|
||
+ if key.val in (IBus.KEY_Left, IBus.KEY_KP_Left):
|
||
+ if not self._editor.get_preedit_string_complete():
|
||
+ return False
|
||
+ self._editor.arrow_left()
|
||
+ self._update_ui()
|
||
+ return True
|
||
+
|
||
+ if key.val in (IBus.KEY_Right, IBus.KEY_KP_Right):
|
||
+ if not self._editor.get_preedit_string_complete():
|
||
+ return False
|
||
+ self._editor.arrow_right()
|
||
+ self._update_ui()
|
||
+ return True
|
||
+
|
||
+ if (key.val == IBus.KEY_BackSpace
|
||
+ and key.state & IBus.ModifierType.CONTROL_MASK):
|
||
+ if not self._editor.get_preedit_string_complete():
|
||
+ return False
|
||
+ self._editor.remove_preedit_before_cursor()
|
||
+ self._update_ui()
|
||
+ return True
|
||
+
|
||
+ if key.val == IBus.KEY_BackSpace:
|
||
+ if not self._editor.get_preedit_string_complete():
|
||
+ return False
|
||
+ self._editor.remove_char()
|
||
+ self._update_ui()
|
||
+ return True
|
||
+
|
||
+ if (key.val == IBus.KEY_Delete
|
||
+ and key.state & IBus.ModifierType.CONTROL_MASK):
|
||
+ if not self._editor.get_preedit_string_complete():
|
||
+ return False
|
||
+ self._editor.remove_preedit_after_cursor()
|
||
+ self._update_ui()
|
||
+ return True
|
||
+
|
||
+ if key.val == IBus.KEY_Delete:
|
||
+ if not self._editor.get_preedit_string_complete():
|
||
+ return False
|
||
+ self._editor.delete()
|
||
+ self._update_ui()
|
||
+ return True
|
||
+
|
||
+ if (key.val in self._editor.get_select_keys()
|
||
+ and self._editor._candidates
|
||
+ and key.state & IBus.ModifierType.CONTROL_MASK):
|
||
+ res = self._editor.select_key(key.val)
|
||
+ self._update_ui()
|
||
+ return res
|
||
+
|
||
+ if (key.val in self._editor.get_select_keys()
|
||
+ and self._editor._candidates
|
||
+ and key.state & IBus.ModifierType.MOD1_MASK):
|
||
+ res = self._editor.remove_candidate_from_user_database(key.val)
|
||
+ self._update_ui()
|
||
+ return res
|
||
+
|
||
+ # now we ignore all other hotkeys
|
||
+ if (key.state
|
||
+ & (IBus.ModifierType.CONTROL_MASK|IBus.ModifierType.MOD1_MASK)):
|
||
+ return False
|
||
+
|
||
+ if key.state & IBus.ModifierType.MOD1_MASK:
|
||
+ return False
|
||
+
|
||
+ # Section to handle valid input characters:
|
||
+ #
|
||
+ # All keys which could possibly conflict with the valid input
|
||
+ # characters should be checked below this section. These are
|
||
+ # SELECT_KEYS, PAGE_UP_KEYS, PAGE_DOWN_KEYS, and COMMIT_KEYS.
|
||
+ #
|
||
+ # For example, consider a table has
|
||
+ #
|
||
+ # SELECT_KEYS = 1,2,3,4,5,6,7,8,9,0
|
||
+ #
|
||
+ # and
|
||
+ #
|
||
+ # VALID_INPUT_CHARS = 0123456789abcdef
|
||
+ #
|
||
+ # (Currently the cns11643 table has this, for example)
|
||
+ #
|
||
+ # Then the digit “1” could be interpreted either as an input
|
||
+ # character or as a select key but of course not both. If the
|
||
+ # meaning as a select key or page down key were preferred,
|
||
+ # this would make some input impossible which probably makes
|
||
+ # the whole input method useless. If the meaning as an input
|
||
+ # character is preferred, this makes selection using that key
|
||
+ # impossible. Making selection by key impossible is not nice
|
||
+ # either, but it is not a complete show stopper as there are
|
||
+ # still other possibilities to select, for example using the
|
||
+ # arrow-up/arrow-down keys or click with the mouse.
|
||
+ #
|
||
+ # Of course one should maybe consider fixing the conflict
|
||
+ # between the keys by using different SELECT_KEYS and/or
|
||
+ # PAGE_UP_KEYS/PAGE_DOWN_KEYS in that table ...
|
||
+ if (keychar
|
||
+ and (keychar in (self._valid_input_chars
|
||
+ + self._single_wildcard_char
|
||
+ + self._multi_wildcard_char)
|
||
+ or (self._editor._py_mode
|
||
+ and keychar in (self._pinyin_valid_input_chars
|
||
+ + self._single_wildcard_char
|
||
+ + self._multi_wildcard_char)))):
|
||
+ if debug_level > 0:
|
||
+ sys.stderr.write(
|
||
+ '_table_mode_process_key_event() valid input: '
|
||
+ + 'repr(keychar)=%(keychar)s\n'
|
||
+ % {'keychar': keychar})
|
||
+ if self._editor._py_mode:
|
||
+ if ((len(self._editor._chars_valid)
|
||
+ == self._max_key_length_pinyin)
|
||
+ or (len(self._editor._chars_valid) > 1
|
||
+ and self._editor._chars_valid[-1] in '!@#$%')):
|
||
+ if self._auto_commit:
|
||
+ self.commit_everything_unless_invalid()
|
||
+ else:
|
||
+ self._editor.commit_to_preedit()
|
||
+ else:
|
||
+ if ((len(self._editor._chars_valid)
|
||
+ == self._max_key_length)
|
||
+ or (len(self._editor._chars_valid)
|
||
+ in self.db.possible_tabkeys_lengths)):
|
||
+ if self._auto_commit:
|
||
+ self.commit_everything_unless_invalid()
|
||
+ else:
|
||
+ self._editor.commit_to_preedit()
|
||
+ res = self._editor.add_input(keychar)
|
||
+ if not res:
|
||
+ if self._auto_select and self._editor._candidates_previous:
|
||
+ # Used for example for the Russian transliteration method
|
||
+ # “translit”, which uses “auto select”.
|
||
+ # The “translit” table contains:
|
||
+ #
|
||
+ # sh ш
|
||
+ # shh щ
|
||
+ #
|
||
+ # so typing “sh” matches “ш” and “щ”. The
|
||
+ # candidate with the shortest key sequence comes
|
||
+ # first in the lookup table, therefore “sh ш”
|
||
+ # is shown in the preëdit (The other candidate,
|
||
+ # “shh щ” comes second in the lookup table and
|
||
+ # could be selected using arrow-down. But
|
||
+ # “translit” hides the lookup table by default).
|
||
+ #
|
||
+ # Now, when after typing “sh” one types “s”,
|
||
+ # the key “shs” has no match, so add_input('s')
|
||
+ # returns “False” and we end up here. We pop the
|
||
+ # last character “s” which caused the match to
|
||
+ # fail, commit first of the previous candidates,
|
||
+ # i.e. “sh ш” and feed the “s” into the
|
||
+ # key event handler again.
|
||
+ self._editor.pop_input()
|
||
+ self.commit_everything_unless_invalid()
|
||
+ return self._table_mode_process_key_event(key)
|
||
+ self.commit_everything_unless_invalid()
|
||
+ self._update_ui()
|
||
+ return True
|
||
+ else:
|
||
+ if (self._auto_commit and self._editor.one_candidate()
|
||
+ and
|
||
+ (self._editor._chars_valid
|
||
+ == self._editor._candidates[0][0])):
|
||
+ self.commit_everything_unless_invalid()
|
||
+ self._update_ui()
|
||
+ return True
|
||
+
|
||
+ if key.val in self._commit_keys:
|
||
+ if self.commit_everything_unless_invalid():
|
||
+ if self._editor._auto_select:
|
||
+ self.commit_string(u' ')
|
||
+ return True
|
||
+
|
||
+ if key.val in self._page_down_keys and self._editor._candidates:
|
||
+ res = self._editor.page_down()
|
||
+ self._update_ui()
|
||
+ return res
|
||
+
|
||
+ if key.val in self._page_up_keys and self._editor._candidates:
|
||
+ res = self._editor.page_up()
|
||
+ self._update_ui()
|
||
+ return res
|
||
+
|
||
+ if (key.val in self._editor.get_select_keys()
|
||
+ and self._editor._candidates):
|
||
+ if self._editor.select_key(key.val): # commits to preëdit
|
||
+ self.commit_string(
|
||
+ self._editor.get_preedit_string_complete(),
|
||
+ tabkeys=self._editor.get_preedit_tabkeys_complete())
|
||
+ return True
|
||
+
|
||
+ # Section to handle trailing invalid input:
|
||
+ #
|
||
+ # If the key has still not been handled when this point is
|
||
+ # reached, it cannot be a valid input character. Neither can
|
||
+ # it be a select key nor a page-up/page-down key. Adding this
|
||
+ # key to the tabkeys and search for matching candidates in the
|
||
+ # table would thus be pointless.
|
||
+ #
|
||
+ # So we commit all pending input immediately and then commit
|
||
+ # this invalid input character as well, possibly converted to
|
||
+ # fullwidth or halfwidth.
|
||
+ if keychar:
|
||
+ if debug_level > 0:
|
||
+ sys.stderr.write(
|
||
+ '_table_mode_process_key_event() trailing invalid input: '
|
||
+ + 'repr(keychar)=%(keychar)s\n'
|
||
+ % {'keychar': keychar})
|
||
+ if not self._editor._candidates:
|
||
+ self.commit_string(self._editor.get_preedit_tabkeys_complete())
|
||
+ else:
|
||
+ self._editor.commit_to_preedit()
|
||
+ self.commit_string(self._editor.get_preedit_string_complete())
|
||
+ if ascii_ispunct(keychar):
|
||
+ self.commit_string(self.cond_punct_translate(keychar))
|
||
+ else:
|
||
+ self.commit_string(self.cond_letter_translate(keychar))
|
||
+ return True
|
||
+
|
||
+ # What kind of key was this??
|
||
+ #
|
||
+ # keychar = IBus.keyval_to_unicode(key.val)
|
||
+ #
|
||
+ # returned no result. So whatever this was, we cannot handle it,
|
||
+ # just pass it through to the application by returning “False”.
|
||
+ return False
|
||
+
|
||
+ def do_focus_in (self):
|
||
+ if debug_level > 1:
|
||
+ sys.stderr.write("do_focus_in()")
|
||
+ if self._on:
|
||
+ self.register_properties(self.properties)
|
||
+ self._init_or_update_property_menu(
|
||
+ self.input_mode_menu,
|
||
+ self._input_mode)
|
||
+ self._update_ui ()
|
||
+
|
||
+ def do_focus_out (self):
|
||
+ if self._has_input_purpose:
|
||
+ self._input_purpose = 0
|
||
+ self._editor.clear_all_input_and_preedit()
|
||
+
|
||
+ def do_set_content_type(self, purpose, hints):
|
||
+ if self._has_input_purpose:
|
||
+ self._input_purpose = purpose
|
||
+
|
||
+ def do_enable (self):
|
||
+ self._on = True
|
||
+ self.do_focus_in()
|
||
+
|
||
+ def do_disable (self):
|
||
+ self._on = False
|
||
+
|
||
+ def do_page_up (self):
|
||
+ if self._editor.page_up ():
|
||
+ self._update_ui ()
|
||
+ return True
|
||
+ return False
|
||
+
|
||
+ def do_page_down (self):
|
||
+ if self._editor.page_down ():
|
||
+ self._update_ui ()
|
||
+ return True
|
||
+ return False
|
||
+
|
||
+ def config_section_normalize(self, section):
|
||
+ # This function replaces _: with - in the dconf
|
||
+ # section and converts to lower case to make
|
||
+ # the comparison of the dconf sections work correctly.
|
||
+ # I avoid using .lower() here because it is locale dependent,
|
||
+ # when using .lower() this would not achieve the desired
|
||
+ # effect of comparing the dconf sections case insentively
|
||
+ # in some locales, it would fail for example if Turkish
|
||
+ # locale (tr_TR.UTF-8) is set.
|
||
+ if sys.version_info >= (3, 0, 0): # Python3
|
||
+ return re.sub(r'[_:]', r'-', section).translate(
|
||
+ ''.maketrans(
|
||
+ string.ascii_uppercase,
|
||
+ string.ascii_lowercase))
|
||
+ else: # Python2
|
||
+ return re.sub(r'[_:]', r'-', section).translate(
|
||
+ string.maketrans(
|
||
+ string.ascii_uppercase,
|
||
+ string.ascii_lowercase).decode('ISO-8859-1'))
|
||
+
|
||
+ def config_value_changed_cb(self, config, section, name, value):
|
||
+ if (self.config_section_normalize(self._config_section)
|
||
+ != self.config_section_normalize(section)):
|
||
+ return
|
||
+ value = variant_to_value(value)
|
||
+ print('config value %(n)s for engine %(en)s changed to %(value)s'
|
||
+ % {'n': name, 'en': self._engine_name, 'value': value})
|
||
+ if name == u'inputmode':
|
||
+ self.set_input_mode(value)
|
||
+ return
|
||
+ if name == u'autoselect':
|
||
+ self._editor._auto_select = value
|
||
+ self._auto_select = value
|
||
+ return
|
||
+ if name == u'autocommit':
|
||
+ self.set_autocommit_mode(value)
|
||
+ return
|
||
+ if name == u'chinesemode':
|
||
+ self.set_chinese_mode(value)
|
||
+ self.db.reset_phrases_cache()
|
||
+ return
|
||
+ if name == u'endeffullwidthletter':
|
||
+ self.set_letter_width(value, input_mode=0)
|
||
+ return
|
||
+ if name == u'endeffullwidthpunct':
|
||
+ self.set_punctuation_width(value, input_mode=0)
|
||
+ return
|
||
+ if name == u'lookuptableorientation':
|
||
+ self._editor._orientation = value
|
||
+ self._editor._lookup_table.set_orientation(value)
|
||
+ return
|
||
+ if name == u'lookuptablepagesize':
|
||
+ if value > len(self._editor._select_keys):
|
||
+ value = len(self._editor._select_keys)
|
||
+ self._config.set_value(
|
||
+ self._config_section,
|
||
+ 'lookuptablepagesize',
|
||
+ GLib.Variant.new_int32(value))
|
||
+ if value < 1:
|
||
+ value = 1
|
||
+ self._config.set_value(
|
||
+ self._config_section,
|
||
+ 'lookuptablepagesize',
|
||
+ GLib.Variant.new_int32(value))
|
||
+ self._editor._page_size = value
|
||
+ self._editor._lookup_table = self._editor.get_new_lookup_table(
|
||
+ page_size = self._editor._page_size,
|
||
+ select_keys = self._editor._select_keys,
|
||
+ orientation = self._editor._orientation)
|
||
+ self.reset()
|
||
+ return
|
||
+ if name == u'onechar':
|
||
+ self.set_onechar_mode(value)
|
||
+ self.db.reset_phrases_cache()
|
||
+ return
|
||
+ if name == u'tabdeffullwidthletter':
|
||
+ self.set_letter_width(value, input_mode=1)
|
||
+ return
|
||
+ if name == u'tabdeffullwidthpunct':
|
||
+ self.set_punctuation_width(value, input_mode=1)
|
||
+ return
|
||
+ if name == u'alwaysshowlookup':
|
||
+ self._always_show_lookup = value
|
||
+ return
|
||
+ if name == u'spacekeybehavior':
|
||
+ if value == True:
|
||
+ # space is used as a page down key and not as a commit key:
|
||
+ if IBus.KEY_space not in self._page_down_keys:
|
||
+ self._page_down_keys.append(IBus.KEY_space)
|
||
+ if IBus.KEY_space in self._commit_keys:
|
||
+ self._commit_keys.remove(IBus.KEY_space)
|
||
+ if value == False:
|
||
+ # space is used as a commit key and not used as a page down key:
|
||
+ if IBus.KEY_space in self._page_down_keys:
|
||
+ self._page_down_keys.remove(IBus.KEY_space)
|
||
+ if IBus.KEY_space not in self._commit_keys:
|
||
+ self._commit_keys.append(IBus.KEY_space)
|
||
+ if debug_level > 1:
|
||
+ sys.stderr.write(
|
||
+ "self._page_down_keys=%s\n"
|
||
+ % repr(self._page_down_keys))
|
||
+ return
|
||
+ if name == u'singlewildcardchar':
|
||
+ self._single_wildcard_char = value
|
||
+ self._editor._single_wildcard_char = value
|
||
+ self.db.reset_phrases_cache()
|
||
+ return
|
||
+ if name == u'multiwildcardchar':
|
||
+ self._multi_wildcard_char = value
|
||
+ self._editor._multi_wildcard_char = value
|
||
+ self.db.reset_phrases_cache()
|
||
+ return
|
||
+ if name == u'autowildcard':
|
||
+ self._auto_wildcard = value
|
||
+ self._editor._auto_wildcard = value
|
||
+ self.db.reset_phrases_cache()
|
||
+ return
|
||
diff -Nru ibus-table-1.9.18.orig/engine/tabsqlitedb.py ibus-table-1.9.18/engine/tabsqlitedb.py
|
||
--- ibus-table-1.9.18.orig/engine/tabsqlitedb.py 2020-07-22 11:52:11.651532112 +0200
|
||
+++ ibus-table-1.9.18/engine/tabsqlitedb.py 2020-07-22 14:43:51.907260935 +0200
|
||
@@ -1047,6 +1047,8 @@
|
||
traceback.print_exc ()
|
||
|
||
def init_user_db(self, db_file):
|
||
+ if db_file == ':memory:':
|
||
+ return
|
||
if not path.exists(db_file):
|
||
db = sqlite3.connect(db_file)
|
||
# 20000 pages should be enough to cache the whole database
|
||
diff -Nru ibus-table-1.9.18.orig/engine/tabsqlitedb.py.orig ibus-table-1.9.18/engine/tabsqlitedb.py.orig
|
||
--- ibus-table-1.9.18.orig/engine/tabsqlitedb.py.orig 1970-01-01 01:00:00.000000000 +0100
|
||
+++ ibus-table-1.9.18/engine/tabsqlitedb.py.orig 2020-07-22 11:52:11.651532112 +0200
|
||
@@ -0,0 +1,1433 @@
|
||
+# -*- coding: utf-8 -*-
|
||
+# vim:et sts=4 sw=4
|
||
+#
|
||
+# ibus-table - The Tables engine for IBus
|
||
+#
|
||
+# Copyright (c) 2008-2009 Yu Yuwei <acevery@gmail.com>
|
||
+# Copyright (c) 2009-2014 Caius "kaio" CHANCE <me@kaio.net>
|
||
+# Copyright (c) 2012-2015 Mike FABIAN <mfabian@redhat.com>
|
||
+#
|
||
+# 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.1 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
||
+#
|
||
+
|
||
+import sys
|
||
+if sys.version_info < (3, 0, 0):
|
||
+ reload (sys)
|
||
+ sys.setdefaultencoding('utf-8')
|
||
+import os
|
||
+import os.path as path
|
||
+import shutil
|
||
+import sqlite3
|
||
+import uuid
|
||
+import time
|
||
+import re
|
||
+import chinese_variants
|
||
+
|
||
+debug_level = int(0)
|
||
+
|
||
+database_version = '1.00'
|
||
+
|
||
+patt_r = re.compile(r'c([ea])(\d):(.*)')
|
||
+patt_p = re.compile(r'p(-{0,1}\d)(-{0,1}\d)')
|
||
+
|
||
+chinese_nocheck_chars = u"“”‘’《》〈〉〔〕「」『』【】〖〗()[]{}"\
|
||
+ u".。,、;:?!…—·ˉˇ¨々~‖∶"'`|"\
|
||
+ u"⒈⒉⒊⒋⒌⒍⒎⒏⒐⒑⒒⒓⒔⒕⒖⒗⒘⒙⒚⒛"\
|
||
+ u"АБВГДЕЖЗИЙКЛМНОПРСТУФХЦЧШЩЪЫЬЭЮЯЁ"\
|
||
+ u"ⅠⅡⅢⅣⅤⅥⅦⅧⅨⅩⅪⅫ"\
|
||
+ u"⒈⒉⒊⒋⒌⒍⒎⒏⒐⒑⒒⒓⒔⒕⒖⒗⒘⒙⒚⒛"\
|
||
+ u"㎎㎏㎜㎝㎞㎡㏄㏎㏑㏒㏕"\
|
||
+ u"ΑΒΓΔΕΖΗΘΙΚΛΜΝΞΟΠΡΣΤΥΦΧΨΩ"\
|
||
+ u"⑴⑵⑶⑷⑸⑹⑺⑻⑼⑽⑾⑿⒀⒁⒂⒃⒄⒅⒆⒇"\
|
||
+ u"€$¢£¥"\
|
||
+ u"¤→↑←↓↖↗↘↙"\
|
||
+ u"ァアィイゥウェエォオカガキギクグケゲコゴサザシジ"\
|
||
+ u"スズセゼソゾタダチヂッツヅテデトドナニヌネノハバパ"\
|
||
+ u"ヒビピフブプヘベペホボポマミムメモャヤュユョヨラ"\
|
||
+ u"リルレロヮワヰヱヲンヴヵヶーヽヾ"\
|
||
+ u"ぁあぃいぅうぇえぉおかがきぎぱくぐけげこごさざしじ"\
|
||
+ u"すずせぜそぞただちぢっつづてでとどなにぬねのはば"\
|
||
+ u"ひびぴふぶぷへべぺほぼぽまみむめもゃやゅゆょよらり"\
|
||
+ u"るれろゎわゐゑをん゛゜ゝゞ"\
|
||
+ u"勹灬冫艹屮辶刂匚阝廾丨虍彐卩钅冂冖宀疒肀丿攵凵犭"\
|
||
+ u"亻彡饣礻扌氵纟亠囗忄讠衤廴尢夂丶"\
|
||
+ u"āáǎàōóǒòêēéěèīíǐìǖǘǚǜüūúǔù"\
|
||
+ u"+-<=>±×÷∈∏∑∕√∝∞∟∠∣∥∧∨∩∪∫∮"\
|
||
+ u"∴∵∶∷∽≈≌≒≠≡≤≥≦≧≮≯⊕⊙⊥⊿℃°‰"\
|
||
+ u"♂♀§№☆★○●◎◇◆□■△▲※〓#&@\^_ ̄"\
|
||
+ u"абвгдежзийклмнопрстуфхцчшщъыьэюяё"\
|
||
+ u"ⅰⅱⅲⅳⅴⅵⅶⅷⅸⅹβγδεζηαικλμνξοπρστυφθψω"\
|
||
+ u"①②③④⑤⑥⑦⑧⑨⑩①②③④⑤⑥⑦⑧⑨⑩"\
|
||
+ u"㈠㈡㈢㈣㈤㈥㈦㈧㈨㈩㈠㈡㈢㈣㈤㈥㈦㈧㈨㈩"\
|
||
+ u"ㄅㄆㄇㄈㄉㄊㄋㄌㄍㄎㄏㄐㄑㄒㄓㄔㄕㄖㄗㄘㄙㄧㄨㄩ"\
|
||
+ u"ㄚㄛㄜㄝㄞㄟㄠㄡㄢㄣㄤㄥㄦ"
|
||
+
|
||
+class ImeProperties:
|
||
+ def __init__(self, db=None, default_properties={}):
|
||
+ '''
|
||
+ “db” is the handle of the sqlite3 database file obtained by
|
||
+ sqlite3.connect().
|
||
+ '''
|
||
+ if not db:
|
||
+ return None
|
||
+ self.ime_property_cache = default_properties
|
||
+ sqlstr = 'SELECT attr, val FROM main.ime;'
|
||
+ try:
|
||
+ results = db.execute(sqlstr).fetchall()
|
||
+ except:
|
||
+ import traceback
|
||
+ traceback.print_exc()
|
||
+ for result in results:
|
||
+ self.ime_property_cache[result[0]] = result[1]
|
||
+
|
||
+ def get(self, key):
|
||
+ if key in self.ime_property_cache:
|
||
+ return self.ime_property_cache[key]
|
||
+ else:
|
||
+ return None
|
||
+
|
||
+class tabsqlitedb:
|
||
+ '''Phrase database for tables
|
||
+
|
||
+ The phrases table in the database has columns with the names:
|
||
+
|
||
+ “id”, “tabkeys”, “phrase”, “freq”, “user_freq”
|
||
+
|
||
+ There are 2 databases, sysdb, userdb.
|
||
+
|
||
+ sysdb: System database for the input method, for example something
|
||
+ like /usr/share/ibus-table/tables/wubi-jidian86.db
|
||
+ “user_freq” is always 0 in a system database. “freq”
|
||
+ is some number in a system database indicating a frequency
|
||
+ of use of that phrase relative to the other phrases in that
|
||
+ database.
|
||
+
|
||
+ user_db: Database on disk where the phrases used or defined by the
|
||
+ user are stored. “user_freq” is a counter which counts how
|
||
+ many times that combination of “tabkeys” and “phrase” has
|
||
+ been used. “freq” is equal to 0 for all combinations of
|
||
+ “tabkeys” and “phrase” where an entry for that phrase is
|
||
+ already in the system database which starts with the same
|
||
+ “tabkeys”.
|
||
+ For combinations of “tabkeys” and “phrase” which do not exist
|
||
+ at all in the system database, “freq” is equal to -1 to
|
||
+ indidated that this is a user defined phrase.
|
||
+ '''
|
||
+ def __init__(
|
||
+ self, filename = None, user_db = None, create_database = False):
|
||
+ global debug_level
|
||
+ try:
|
||
+ debug_level = int(os.getenv('IBUS_TABLE_DEBUG_LEVEL'))
|
||
+ except (TypeError, ValueError):
|
||
+ debug_level = int(0)
|
||
+ self.old_phrases = []
|
||
+ self.filename = filename
|
||
+ self._user_db = user_db
|
||
+ self.reset_phrases_cache()
|
||
+
|
||
+ if create_database or os.path.isfile(self.filename):
|
||
+ self.db = sqlite3.connect(self.filename)
|
||
+ else:
|
||
+ print('Cannot open database file %s' %self.filename)
|
||
+ try:
|
||
+ self.db.execute('PRAGMA encoding = "UTF-8";')
|
||
+ self.db.execute('PRAGMA case_sensitive_like = true;')
|
||
+ self.db.execute('PRAGMA page_size = 4096;')
|
||
+ # 20000 pages should be enough to cache the whole database
|
||
+ self.db.execute('PRAGMA cache_size = 20000;')
|
||
+ self.db.execute('PRAGMA temp_store = MEMORY;')
|
||
+ self.db.execute('PRAGMA journal_size_limit = 1000000;')
|
||
+ self.db.execute('PRAGMA synchronous = NORMAL;')
|
||
+ except:
|
||
+ import traceback
|
||
+ traceback.print_exc()
|
||
+ print('Error while initializing database.')
|
||
+ # create IME property table
|
||
+ self.db.executescript(
|
||
+ 'CREATE TABLE IF NOT EXISTS main.ime (attr TEXT, val TEXT);')
|
||
+ # Initalize missing attributes in the ime table with some
|
||
+ # default values, they should be updated using the attributes
|
||
+ # found in the source when creating a system database with
|
||
+ # tabcreatedb.py
|
||
+ self._default_ime_attributes = {
|
||
+ 'name':'',
|
||
+ 'name.zh_cn':'',
|
||
+ 'name.zh_hk':'',
|
||
+ 'name.zh_tw':'',
|
||
+ 'author':'somebody',
|
||
+ 'uuid':'%s' % uuid.uuid4(),
|
||
+ 'serial_number':'%s' % time.strftime('%Y%m%d'),
|
||
+ 'icon':'ibus-table.svg',
|
||
+ 'license':'LGPL',
|
||
+ 'languages':'',
|
||
+ 'language_filter':'',
|
||
+ 'valid_input_chars':'abcdefghijklmnopqrstuvwxyz',
|
||
+ 'max_key_length':'4',
|
||
+ 'commit_keys':'space',
|
||
+ # 'forward_keys':'Return',
|
||
+ 'select_keys':'1,2,3,4,5,6,7,8,9,0',
|
||
+ 'page_up_keys':'Page_Up,minus',
|
||
+ 'page_down_keys':'Page_Down,equal',
|
||
+ 'status_prompt':'',
|
||
+ 'def_full_width_punct':'true',
|
||
+ 'def_full_width_letter':'false',
|
||
+ 'user_can_define_phrase':'false',
|
||
+ 'pinyin_mode':'false',
|
||
+ 'dynamic_adjust':'false',
|
||
+ 'auto_select':'false',
|
||
+ 'auto_commit':'false',
|
||
+ # 'no_check_chars':u'',
|
||
+ 'description':'A IME under IBus Table',
|
||
+ 'layout':'us',
|
||
+ 'symbol':'',
|
||
+ 'rules':'',
|
||
+ 'least_commit_length':'0',
|
||
+ 'start_chars':'',
|
||
+ 'orientation':'true',
|
||
+ 'always_show_lookup':'true',
|
||
+ 'char_prompts':'{}'
|
||
+ # we use this entry for those IME, which don't
|
||
+ # have rules to build up phrase, but still need
|
||
+ # auto commit to preedit
|
||
+ }
|
||
+ if create_database:
|
||
+ select_sqlstr = '''
|
||
+ SELECT val FROM main.ime WHERE attr = :attr;'''
|
||
+ insert_sqlstr = '''
|
||
+ INSERT INTO main.ime (attr, val) VALUES (:attr, :val);'''
|
||
+ for attr in self._default_ime_attributes:
|
||
+ sqlargs = {
|
||
+ 'attr': attr,
|
||
+ 'val': self._default_ime_attributes[attr]
|
||
+ }
|
||
+ if not self.db.execute(select_sqlstr, sqlargs).fetchall():
|
||
+ self.db.execute(insert_sqlstr, sqlargs)
|
||
+ self.ime_properties = ImeProperties(
|
||
+ db=self.db,
|
||
+ default_properties=self._default_ime_attributes)
|
||
+ # shared variables in this class:
|
||
+ self._mlen = int(self.ime_properties.get("max_key_length"))
|
||
+ self._is_chinese = self.is_chinese()
|
||
+ self._is_cjk = self.is_cjk()
|
||
+ self.user_can_define_phrase = self.ime_properties.get(
|
||
+ 'user_can_define_phrase')
|
||
+ if self.user_can_define_phrase:
|
||
+ if self.user_can_define_phrase.lower() == u'true' :
|
||
+ self.user_can_define_phrase = True
|
||
+ else:
|
||
+ self.user_can_define_phrase = False
|
||
+ else:
|
||
+ print(
|
||
+ 'Could not find "user_can_define_phrase" entry from database, '
|
||
+ + 'is it an outdated database?')
|
||
+ self.user_can_define_phrase = False
|
||
+
|
||
+ self.dynamic_adjust = self.ime_properties.get('dynamic_adjust')
|
||
+ if self.dynamic_adjust:
|
||
+ if self.dynamic_adjust.lower() == u'true' :
|
||
+ self.dynamic_adjust = True
|
||
+ else:
|
||
+ self.dynamic_adjust = False
|
||
+ else:
|
||
+ print(
|
||
+ 'Could not find "dynamic_adjust" entry from database, '
|
||
+ + 'is it an outdated database?')
|
||
+ self.dynamic_adjust = False
|
||
+
|
||
+ self.rules = self.get_rules ()
|
||
+ self.possible_tabkeys_lengths = self.get_possible_tabkeys_lengths()
|
||
+ self.startchars = self.get_start_chars ()
|
||
+
|
||
+ if not user_db or create_database:
|
||
+ # No user database requested or we are
|
||
+ # just creating the system database and
|
||
+ # we do not need a user database for that
|
||
+ return
|
||
+
|
||
+ if user_db != ":memory:":
|
||
+ # Do not move this import to the beginning of this script!
|
||
+ # If for example the home directory is not writeable,
|
||
+ # ibus_table_location.py would fail because it cannot
|
||
+ # create some directories.
|
||
+ #
|
||
+ # But for tabcreatedb.py, no such directories are needed,
|
||
+ # tabcreatedb.py should not fail just because
|
||
+ # ibus_table_location.py cannot create some directories.
|
||
+ #
|
||
+ # “HOME=/foobar ibus-table-createdb” should not fail if
|
||
+ # “/foobar” is not writeable.
|
||
+ import ibus_table_location
|
||
+ tables_path = path.join(ibus_table_location.data_home(), "tables")
|
||
+ if not path.isdir(tables_path):
|
||
+ old_tables_path = os.path.join(
|
||
+ os.getenv('HOME'), '.ibus/tables')
|
||
+ if path.isdir(old_tables_path):
|
||
+ if os.access(os.path.join(
|
||
+ old_tables_path, 'debug.log'), os.F_OK):
|
||
+ os.unlink(os.path.join(old_tables_path, 'debug.log'))
|
||
+ if os.access(os.path.join(
|
||
+ old_tables_path, 'setup-debug.log'), os.F_OK):
|
||
+ os.unlink(os.path.join(
|
||
+ old_tables_path, 'setup-debug.log'))
|
||
+ shutil.copytree(old_tables_path, tables_path)
|
||
+ shutil.rmtree(old_tables_path)
|
||
+ os.symlink(tables_path, old_tables_path)
|
||
+ else:
|
||
+ os.makedirs(tables_path)
|
||
+ user_db = path.join(tables_path, user_db)
|
||
+ if not path.exists(user_db):
|
||
+ sys.stderr.write(
|
||
+ 'The user database %(udb)s does not exist yet.\n'
|
||
+ % {'udb': user_db})
|
||
+ else:
|
||
+ try:
|
||
+ desc = self.get_database_desc(user_db)
|
||
+ phrase_table_column_names = [
|
||
+ 'id', 'tabkeys', 'phrase','freq','user_freq']
|
||
+ if (desc == None
|
||
+ or desc["version"] != database_version
|
||
+ or (self.get_number_of_columns_of_phrase_table(user_db)
|
||
+ != len(phrase_table_column_names))):
|
||
+ sys.stderr.write(
|
||
+ 'The user database %s seems to be incompatible.\n'
|
||
+ % user_db)
|
||
+ if desc == None:
|
||
+ sys.stderr.write(
|
||
+ 'There is no version information in '
|
||
+ + 'the database.\n')
|
||
+ self.old_phrases = self.extract_user_phrases(
|
||
+ user_db, old_database_version = '0.0')
|
||
+ elif desc["version"] != database_version:
|
||
+ sys.stderr.write(
|
||
+ 'The version of the database does not match '
|
||
+ + '(too old or too new?).\n'
|
||
+ 'ibus-table wants version=%s\n'
|
||
+ % database_version
|
||
+ + 'But the database actually has version=%s\n'
|
||
+ % desc['version'])
|
||
+ self.old_phrases = self.extract_user_phrases(
|
||
+ user_db, old_database_version = desc['version'])
|
||
+ elif (self.get_number_of_columns_of_phrase_table(
|
||
+ user_db)
|
||
+ != len(phrase_table_column_names)):
|
||
+ sys.stderr.write(
|
||
+ 'The number of columns of the database '
|
||
+ + 'does not match.\n'
|
||
+ + 'ibus-table expects %s columns.\n'
|
||
+ % len(phrase_table_column_names)
|
||
+ + 'But the database actually has %s columns.\n'
|
||
+ % self.get_number_of_columns_of_phrase_table(
|
||
+ user_db)
|
||
+ + 'But the versions of the databases are '
|
||
+ + 'identical.\n'
|
||
+ + 'This should never happen!\n')
|
||
+ self.old_phrases = None
|
||
+ from time import strftime
|
||
+ timestamp = strftime('-%Y-%m-%d_%H:%M:%S')
|
||
+ sys.stderr.write(
|
||
+ 'Renaming the incompatible database to "%s".\n'
|
||
+ % user_db+timestamp)
|
||
+ if os.path.exists(user_db):
|
||
+ os.rename(user_db, user_db+timestamp)
|
||
+ if os.path.exists(user_db+'-shm'):
|
||
+ os.rename(user_db+'-shm', user_db+'-shm'+timestamp)
|
||
+ if os.path.exists(user_db+'-wal'):
|
||
+ os.rename(user_db+'-wal', user_db+'-wal'+timestamp)
|
||
+ sys.stderr.write(
|
||
+ 'Creating a new, empty database "s".\n'
|
||
+ % user_db)
|
||
+ self.init_user_db(user_db)
|
||
+ sys.stderr.write(
|
||
+ 'If user phrases were successfully recovered from '
|
||
+ + 'the old,\n'
|
||
+ + 'incompatible database, they will be used to '
|
||
+ + 'initialize the new database.\n')
|
||
+ else:
|
||
+ sys.stderr.write(
|
||
+ 'Compatible database %s found.\n' % user_db)
|
||
+ except:
|
||
+ import traceback
|
||
+ traceback.print_exc()
|
||
+
|
||
+ # open user phrase database
|
||
+ try:
|
||
+ sys.stderr.write(
|
||
+ 'Connect to the database %(name)s.\n' %{'name': user_db})
|
||
+ self.db.executescript('''
|
||
+ ATTACH DATABASE "%s" AS user_db;
|
||
+ PRAGMA user_db.encoding = "UTF-8";
|
||
+ PRAGMA user_db.case_sensitive_like = true;
|
||
+ PRAGMA user_db.page_size = 4096;
|
||
+ PRAGMA user_db.cache_size = 20000;
|
||
+ PRAGMA user_db.temp_store = MEMORY;
|
||
+ PRAGMA user_db.journal_mode = WAL;
|
||
+ PRAGMA user_db.journal_size_limit = 1000000;
|
||
+ PRAGMA user_db.synchronous = NORMAL;
|
||
+ ''' % user_db)
|
||
+ except:
|
||
+ sys.stderr.write('Could not open the database %s.\n' % user_db)
|
||
+ from time import strftime
|
||
+ timestamp = strftime('-%Y-%m-%d_%H:%M:%S')
|
||
+ sys.stderr.write('Renaming the incompatible database to "%s".\n'
|
||
+ % user_db+timestamp)
|
||
+ if os.path.exists(user_db):
|
||
+ os.rename(user_db, user_db+timestamp)
|
||
+ if os.path.exists(user_db+'-shm'):
|
||
+ os.rename(user_db+'-shm', user_db+'-shm'+timestamp)
|
||
+ if os.path.exists(user_db+'-wal'):
|
||
+ os.rename(user_db+'-wal', user_db+'-wal'+timestamp)
|
||
+ sys.stderr.write('Creating a new, empty database "%s".\n'
|
||
+ % user_db)
|
||
+ self.init_user_db(user_db)
|
||
+ self.db.executescript('''
|
||
+ ATTACH DATABASE "%s" AS user_db;
|
||
+ PRAGMA user_db.encoding = "UTF-8";
|
||
+ PRAGMA user_db.case_sensitive_like = true;
|
||
+ PRAGMA user_db.page_size = 4096;
|
||
+ PRAGMA user_db.cache_size = 20000;
|
||
+ PRAGMA user_db.temp_store = MEMORY;
|
||
+ PRAGMA user_db.journal_mode = WAL;
|
||
+ PRAGMA user_db.journal_size_limit = 1000000;
|
||
+ PRAGMA user_db.synchronous = NORMAL;
|
||
+ ''' % user_db)
|
||
+ self.create_tables("user_db")
|
||
+ if self.old_phrases:
|
||
+ sqlargs = []
|
||
+ for x in self.old_phrases:
|
||
+ sqlargs.append(
|
||
+ {'tabkeys': x[0],
|
||
+ 'phrase': x[1],
|
||
+ 'freq': x[2],
|
||
+ 'user_freq': x[3]})
|
||
+ sqlstr = '''
|
||
+ INSERT INTO user_db.phrases (tabkeys, phrase, freq, user_freq)
|
||
+ VALUES (:tabkeys, :phrase, :freq, :user_freq)
|
||
+ '''
|
||
+ try:
|
||
+ self.db.executemany(sqlstr, sqlargs)
|
||
+ except:
|
||
+ import traceback
|
||
+ traceback.print_exec()
|
||
+ self.db.commit ()
|
||
+ self.db.execute('PRAGMA wal_checkpoint;')
|
||
+
|
||
+ # try create all tables in user database
|
||
+ self.create_indexes ("user_db", commit=False)
|
||
+ self.generate_userdb_desc ()
|
||
+
|
||
+ def update_phrase(
|
||
+ self, tabkeys=u'', phrase=u'',
|
||
+ user_freq=0, database='user_db', commit=True):
|
||
+ '''update phrase freqs'''
|
||
+ if debug_level > 1:
|
||
+ sys.stderr.write(
|
||
+ 'update_phrase() tabkeys=%(t)s phrase=%(p)s '
|
||
+ % {'t': tabkeys, 'p': phrase}
|
||
+ + 'user_freq=%(u)s database=%(d)s\n'
|
||
+ % {'u': user_freq, 'd': database})
|
||
+ if not tabkeys or not phrase:
|
||
+ return
|
||
+ sqlstr = '''
|
||
+ UPDATE %s.phrases SET user_freq = :user_freq
|
||
+ WHERE tabkeys = :tabkeys AND phrase = :phrase
|
||
+ ;''' % database
|
||
+ sqlargs = {'user_freq': user_freq, 'tabkeys': tabkeys, 'phrase': phrase}
|
||
+ try:
|
||
+ self.db.execute(sqlstr, sqlargs)
|
||
+ if commit:
|
||
+ self.db.commit()
|
||
+ self.invalidate_phrases_cache(tabkeys)
|
||
+ except:
|
||
+ import traceback
|
||
+ traceback.print_exc()
|
||
+
|
||
+ def sync_usrdb (self):
|
||
+ '''
|
||
+ Trigger a checkpoint operation.
|
||
+ '''
|
||
+ if self._user_db is None:
|
||
+ return
|
||
+ self.db.commit()
|
||
+ self.db.execute('PRAGMA wal_checkpoint;')
|
||
+
|
||
+ def reset_phrases_cache (self):
|
||
+ self._phrases_cache = {}
|
||
+
|
||
+ def invalidate_phrases_cache (self, tabkeys=u''):
|
||
+ for i in range(1, self._mlen + 1):
|
||
+ if self._phrases_cache.get(tabkeys[0:i]):
|
||
+ self._phrases_cache.pop(tabkeys[0:i])
|
||
+
|
||
+ def is_chinese (self):
|
||
+ __lang = self.ime_properties.get('languages')
|
||
+ if __lang:
|
||
+ __langs = __lang.split(',')
|
||
+ for _l in __langs:
|
||
+ if _l.lower().find('zh') != -1:
|
||
+ return True
|
||
+ return False
|
||
+
|
||
+ def is_cjk(self):
|
||
+ languages = self.ime_properties.get('languages')
|
||
+ if languages:
|
||
+ languages = languages.split(',')
|
||
+ for language in languages:
|
||
+ for lang in ['zh', 'ja', 'ko']:
|
||
+ if language.strip().startswith(lang):
|
||
+ return True
|
||
+ return False
|
||
+
|
||
+ def get_chinese_mode (self):
|
||
+ try:
|
||
+ __dict = {'cm0':0, 'cm1':1, 'cm2':2, 'cm3':3, 'cm4':4}
|
||
+ __filt = self.ime_properties.get('language_filter')
|
||
+ return __dict[__filt]
|
||
+ except:
|
||
+ return -1
|
||
+
|
||
+ def get_select_keys (self):
|
||
+ ret = self.ime_properties.get("select_keys")
|
||
+ if ret:
|
||
+ return ret
|
||
+ return "1,2,3,4,5,6,7,8,9,0"
|
||
+
|
||
+ def get_orientation (self):
|
||
+ try:
|
||
+ return int(self.ime_properties.get('orientation'))
|
||
+ except:
|
||
+ return 1
|
||
+
|
||
+ def create_tables (self, database):
|
||
+ '''Create tables that contain all phrase'''
|
||
+ if database == 'main':
|
||
+ sqlstr = '''
|
||
+ CREATE TABLE IF NOT EXISTS %s.goucima
|
||
+ (zi TEXT PRIMARY KEY, goucima TEXT);
|
||
+ ''' % database
|
||
+ self.db.execute (sqlstr)
|
||
+ sqlstr = '''
|
||
+ CREATE TABLE IF NOT EXISTS %s.pinyin
|
||
+ (pinyin TEXT, zi TEXT, freq INTEGER);
|
||
+ ''' % database
|
||
+ self.db.execute(sqlstr)
|
||
+
|
||
+ sqlstr = '''
|
||
+ CREATE TABLE IF NOT EXISTS %s.phrases
|
||
+ (id INTEGER PRIMARY KEY, tabkeys TEXT, phrase TEXT,
|
||
+ freq INTEGER, user_freq INTEGER);
|
||
+ ''' % database
|
||
+ self.db.execute (sqlstr)
|
||
+ self.db.commit()
|
||
+
|
||
+ def update_ime (self, attrs):
|
||
+ '''Update or insert attributes in ime table, attrs is a iterable object
|
||
+ Like [(attr,val), (attr,val), ...]
|
||
+
|
||
+ This is called only by tabcreatedb.py.
|
||
+ '''
|
||
+ select_sqlstr = 'SELECT val from main.ime WHERE attr = :attr'
|
||
+ update_sqlstr = 'UPDATE main.ime SET val = :val WHERE attr = :attr;'
|
||
+ insert_sqlstr = 'INSERT INTO main.ime (attr, val) VALUES (:attr, :val);'
|
||
+ for attr, val in attrs:
|
||
+ sqlargs = {'attr': attr, 'val': val}
|
||
+ if self.db.execute(select_sqlstr, sqlargs).fetchall():
|
||
+ self.db.execute(update_sqlstr, sqlargs)
|
||
+ else:
|
||
+ self.db.execute(insert_sqlstr, sqlargs)
|
||
+ self.db.commit()
|
||
+ # update ime properties cache:
|
||
+ self.ime_properties = ImeProperties(
|
||
+ db=self.db,
|
||
+ default_properties=self._default_ime_attributes)
|
||
+ # The self variables used by tabcreatedb.py need to be updated now:
|
||
+ self._mlen = int(self.ime_properties.get('max_key_length'))
|
||
+ self._is_chinese = self.is_chinese()
|
||
+ self.user_can_define_phrase = self.ime_properties.get(
|
||
+ 'user_can_define_phrase')
|
||
+ if self.user_can_define_phrase:
|
||
+ if self.user_can_define_phrase.lower() == u'true' :
|
||
+ self.user_can_define_phrase = True
|
||
+ else:
|
||
+ self.user_can_define_phrase = False
|
||
+ else:
|
||
+ print(
|
||
+ 'Could not find "user_can_define_phrase" entry from database, '
|
||
+ + 'is it a outdated database?')
|
||
+ self.user_can_define_phrase = False
|
||
+ self.rules = self.get_rules()
|
||
+
|
||
+ def get_rules (self):
|
||
+ '''Get phrase construct rules'''
|
||
+ rules = {}
|
||
+ if self.user_can_define_phrase:
|
||
+ try:
|
||
+ _rules = self.ime_properties.get('rules')
|
||
+ if _rules:
|
||
+ _rules = _rules.strip().split(';')
|
||
+ for rule in _rules:
|
||
+ res = patt_r.match (rule)
|
||
+ if res:
|
||
+ cms = []
|
||
+ if res.group(1) == 'a':
|
||
+ rules['above'] = int(res.group(2))
|
||
+ _cms = res.group(3).split('+')
|
||
+ if len(_cms) > self._mlen:
|
||
+ print('rule: "%s" over max key length' %rule)
|
||
+ break
|
||
+ for _cm in _cms:
|
||
+ cm_res = patt_p.match(_cm)
|
||
+ cms.append((int(cm_res.group(1)),
|
||
+ int(cm_res.group(2))))
|
||
+ rules[int(res.group(2))]=cms
|
||
+ else:
|
||
+ print('not a legal rule: "%s"' %rule)
|
||
+ except Exception:
|
||
+ import traceback
|
||
+ traceback.print_exc ()
|
||
+ return rules
|
||
+ else:
|
||
+ return ""
|
||
+
|
||
+ def get_possible_tabkeys_lengths(self):
|
||
+ '''Return a list of the possible lengths for tabkeys in this table.
|
||
+
|
||
+ Example:
|
||
+
|
||
+ If the table source has rules like:
|
||
+
|
||
+ RULES = ce2:p11+p12+p21+p22;ce3:p11+p21+p22+p31;ca4:p11+p21+p31+p41
|
||
+
|
||
+ self._rules will be set to
|
||
+
|
||
+ self._rules={2: [(1, 1), (1, 2), (2, 1), (2, 2)], 3: [(1, 1), (1, 2), (2, 1), (3, 1)], 4: [(1, 1), (2, 1), (3, 1), (-1, 1)], 'above': 4}
|
||
+
|
||
+ and then this function returns “[4, 4, 4]”
|
||
+
|
||
+ Or, if the table source has no RULES but LEAST_COMMIT_LENGTH=2
|
||
+ and MAX_KEY_LENGTH = 4, then it returns “[2, 3, 4]”
|
||
+
|
||
+ I cannot find any tables which use LEAST_COMMIT_LENGTH though.
|
||
+ '''
|
||
+ if self.rules:
|
||
+ max_len = self.rules["above"]
|
||
+ return [len(self.rules[x]) for x in range(2, max_len+1)][:]
|
||
+ else:
|
||
+ try:
|
||
+ least_commit_len = int(
|
||
+ self.ime_properties.get('least_commit_length'))
|
||
+ except:
|
||
+ least_commit_len = 0
|
||
+ if least_commit_len > 0:
|
||
+ return list(range(least_commit_len, self._mlen + 1))
|
||
+ else:
|
||
+ return []
|
||
+
|
||
+ def get_start_chars (self):
|
||
+ '''return possible start chars of IME'''
|
||
+ return self.ime_properties.get('start_chars')
|
||
+
|
||
+ def get_no_check_chars (self):
|
||
+ '''Get the characters which engine should not change freq'''
|
||
+ _chars = self.ime_properties.get('no_check_chars')
|
||
+ if type(_chars) != type(u''):
|
||
+ _chars = _chars.decode('utf-8')
|
||
+ return _chars
|
||
+
|
||
+ def add_phrases (self, phrases, database = 'main'):
|
||
+ '''Add many phrases to database fast. Used by tabcreatedb.py when
|
||
+ creating the system database from scratch.
|
||
+
|
||
+ “phrases” is a iterable object which looks like:
|
||
+
|
||
+ [(tabkeys, phrase, freq ,user_freq), (tabkeys, phrase, freq, user_freq), ...]
|
||
+
|
||
+ This function does not check whether phrases are already
|
||
+ there. As this function is only used while creating the
|
||
+ system database, it is not really necessary to check whether
|
||
+ phrases are already there because the database is initially
|
||
+ empty anyway. And the caller should take care that the
|
||
+ “phrases” argument does not contain duplicates.
|
||
+
|
||
+ '''
|
||
+ if debug_level > 1:
|
||
+ sys.stderr.write("add_phrases() len(phrases)=%s\n"
|
||
+ %len(phrases))
|
||
+ insert_sqlstr = '''
|
||
+ INSERT INTO %(database)s.phrases
|
||
+ (tabkeys, phrase, freq, user_freq)
|
||
+ VALUES (:tabkeys, :phrase, :freq, :user_freq);
|
||
+ ''' % {'database': database}
|
||
+ insert_sqlargs = []
|
||
+ for (tabkeys, phrase, freq, user_freq) in phrases:
|
||
+ insert_sqlargs.append({
|
||
+ 'tabkeys': tabkeys,
|
||
+ 'phrase': phrase,
|
||
+ 'freq': freq,
|
||
+ 'user_freq': user_freq})
|
||
+ self.invalidate_phrases_cache(tabkeys)
|
||
+ self.db.executemany(insert_sqlstr, insert_sqlargs)
|
||
+ self.db.commit()
|
||
+ self.db.execute('PRAGMA wal_checkpoint;')
|
||
+
|
||
+ def add_phrase(
|
||
+ self, tabkeys=u'', phrase=u'', freq=0, user_freq=0,
|
||
+ database='main',commit=True):
|
||
+ '''Add phrase to database, phrase is a object of
|
||
+ (tabkeys, phrase, freq ,user_freq)
|
||
+ '''
|
||
+ if debug_level > 1:
|
||
+ sys.stderr.write(
|
||
+ 'add_phrase tabkeys=%(t)s phrase=%(p)s '
|
||
+ % {'t': tabkeys, 'p': phrase}
|
||
+ + 'freq=%(f)s user_freq=%(u)s\n'
|
||
+ % {'f': freq, 'u': user_freq})
|
||
+ if not tabkeys or not phrase:
|
||
+ return
|
||
+ select_sqlstr = '''
|
||
+ SELECT * FROM %(database)s.phrases
|
||
+ WHERE tabkeys = :tabkeys AND phrase = :phrase;
|
||
+ ''' % {'database': database}
|
||
+ select_sqlargs = {'tabkeys': tabkeys, 'phrase': phrase}
|
||
+ results = self.db.execute(select_sqlstr, select_sqlargs).fetchall()
|
||
+ if results:
|
||
+ # there is already such a phrase, i.e. add_phrase was called
|
||
+ # in error, do nothing to avoid duplicate entries.
|
||
+ if debug_level > 1:
|
||
+ sys.stderr.write(
|
||
+ 'add_phrase() '
|
||
+ + 'select_sqlstr=%(sql)s select_sqlargs=%(arg)s '
|
||
+ % {'sql': select_sqlstr, 'arg': select_sqlargs}
|
||
+ + 'already there!: results=%(r)s \n'
|
||
+ % {'r': results})
|
||
+ return
|
||
+
|
||
+ insert_sqlstr = '''
|
||
+ INSERT INTO %(database)s.phrases
|
||
+ (tabkeys, phrase, freq, user_freq)
|
||
+ VALUES (:tabkeys, :phrase, :freq, :user_freq);
|
||
+ ''' % {'database': database}
|
||
+ insert_sqlargs = {
|
||
+ 'tabkeys': tabkeys,
|
||
+ 'phrase': phrase,
|
||
+ 'freq': freq,
|
||
+ 'user_freq': user_freq}
|
||
+ if debug_level > 1:
|
||
+ sys.stderr.write(
|
||
+ 'add_phrase() insert_sqlstr=%(sql)s insert_sqlargs=%(arg)s\n'
|
||
+ % {'sql': insert_sqlstr, 'arg': insert_sqlargs})
|
||
+ try:
|
||
+ self.db.execute (insert_sqlstr, insert_sqlargs)
|
||
+ if commit:
|
||
+ self.db.commit()
|
||
+ self.invalidate_phrases_cache(tabkeys)
|
||
+ except:
|
||
+ import traceback
|
||
+ traceback.print_exc()
|
||
+
|
||
+ def add_goucima (self, goucimas):
|
||
+ '''Add goucima into database, goucimas is iterable object
|
||
+ Like goucimas = [(zi,goucima), (zi,goucima), ...]
|
||
+ '''
|
||
+ sqlstr = '''
|
||
+ INSERT INTO main.goucima (zi, goucima) VALUES (:zi, :goucima);
|
||
+ '''
|
||
+ sqlargs = []
|
||
+ for zi, goucima in goucimas:
|
||
+ sqlargs.append({'zi': zi, 'goucima': goucima})
|
||
+ try:
|
||
+ self.db.commit()
|
||
+ self.db.executemany(sqlstr, sqlargs)
|
||
+ self.db.commit()
|
||
+ self.db.execute('PRAGMA wal_checkpoint;')
|
||
+ except:
|
||
+ import traceback
|
||
+ traceback.print_exc()
|
||
+
|
||
+ def add_pinyin (self, pinyins, database = 'main'):
|
||
+ '''Add pinyin to database, pinyins is a iterable object
|
||
+ Like: [(zi,pinyin, freq), (zi, pinyin, freq), ...]
|
||
+ '''
|
||
+ sqlstr = '''
|
||
+ INSERT INTO %s.pinyin (pinyin, zi, freq) VALUES (:pinyin, :zi, :freq);
|
||
+ ''' % database
|
||
+ count = 0
|
||
+ for pinyin, zi, freq in pinyins:
|
||
+ count += 1
|
||
+ pinyin = pinyin.replace(
|
||
+ '1','!').replace(
|
||
+ '2','@').replace(
|
||
+ '3','#').replace(
|
||
+ '4','$').replace(
|
||
+ '5','%')
|
||
+ try:
|
||
+ self.db.execute(
|
||
+ sqlstr, {'pinyin': pinyin, 'zi': zi, 'freq': freq})
|
||
+ except Exception:
|
||
+ sys.stderr.write(
|
||
+ 'Error when inserting into pinyin table. '
|
||
+ + 'count=%(c)s pinyin=%(p)s zi=%(z)s freq=%(f)s\n'
|
||
+ % {'c': count, 'p': pinyin, 'z': zi, 'f': freq})
|
||
+ import traceback
|
||
+ traceback.print_exc()
|
||
+ self.db.commit()
|
||
+
|
||
+ def optimize_database (self, database='main'):
|
||
+ sqlstr = '''
|
||
+ CREATE TABLE tmp AS SELECT * FROM %(database)s.phrases;
|
||
+ DELETE FROM %(database)s.phrases;
|
||
+ INSERT INTO %(database)s.phrases SELECT * FROM tmp ORDER BY
|
||
+ tabkeys ASC, phrase ASC, user_freq DESC, freq DESC, id ASC;
|
||
+ DROP TABLE tmp;
|
||
+ CREATE TABLE tmp AS SELECT * FROM %(database)s.goucima;
|
||
+ DELETE FROM %(database)s.goucima;
|
||
+ INSERT INTO %(database)s.goucima SELECT * FROM tmp ORDER BY zi, goucima;
|
||
+ DROP TABLE tmp;
|
||
+ CREATE TABLE tmp AS SELECT * FROM %(database)s.pinyin;
|
||
+ DELETE FROM %(database)s.pinyin;
|
||
+ INSERT INTO %(database)s.pinyin SELECT * FROM tmp ORDER BY pinyin ASC, freq DESC;
|
||
+ DROP TABLE tmp;
|
||
+ ''' % {'database':database}
|
||
+ self.db.executescript (sqlstr)
|
||
+ self.db.executescript ("VACUUM;")
|
||
+ self.db.commit()
|
||
+
|
||
+ def drop_indexes(self, database):
|
||
+ '''Drop the indexes in the database to reduce its size
|
||
+
|
||
+ We do not use any indexes at the moment, therefore this
|
||
+ function does nothing.
|
||
+ '''
|
||
+ if debug_level > 1:
|
||
+ sys.stderr.write("drop_indexes()\n")
|
||
+ return
|
||
+
|
||
+ def create_indexes(self, database, commit=True):
|
||
+ '''Create indexes for the database.
|
||
+
|
||
+ We do not use any indexes at the moment, therefore
|
||
+ this function does nothing. We used indexes before,
|
||
+ but benchmarking showed that none of them was really
|
||
+ speeding anything up, therefore we deleted all of them
|
||
+ to get much smaller databases (about half the size).
|
||
+
|
||
+ If some index turns out to be very useful in future, it could
|
||
+ be created here (and dropped in “drop_indexes()”).
|
||
+ '''
|
||
+ if debug_level > 1:
|
||
+ sys.stderr.write("create_indexes()\n")
|
||
+ return
|
||
+
|
||
+ def big5_code(self, phrase):
|
||
+ try:
|
||
+ big5 = phrase.encode('Big5')
|
||
+ except:
|
||
+ big5 = b'\xff\xff' # higher than any Big5 code
|
||
+ return big5
|
||
+
|
||
+ def best_candidates(
|
||
+ self, typed_tabkeys=u'', candidates=[], chinese_mode=-1):
|
||
+ '''
|
||
+ “candidates” is an array containing something like:
|
||
+ [(tabkeys, phrase, freq, user_freq), ...]
|
||
+
|
||
+ “typed_tabkeys” is key sequence the user really typed, which
|
||
+ maybe only the beginning part of the “tabkeys” in a matched
|
||
+ candidate.
|
||
+ '''
|
||
+ maximum_number_of_candidates = 100
|
||
+ engine_name = os.path.basename(self.filename).replace('.db', '')
|
||
+ if engine_name in [
|
||
+ 'cangjie3', 'cangjie5', 'cangjie-big',
|
||
+ 'quick-classic', 'quick3', 'quick5']:
|
||
+ code_point_function = self.big5_code
|
||
+ else:
|
||
+ code_point_function = lambda x: (1)
|
||
+ if chinese_mode in (2, 3) and self._is_chinese:
|
||
+ if chinese_mode == 2:
|
||
+ bitmask = (1 << 0) # used in simplified Chinese
|
||
+ else:
|
||
+ bitmask = (1 << 1) # used in traditional Chinese
|
||
+ return sorted(candidates,
|
||
+ key=lambda x: (
|
||
+ - int(
|
||
+ typed_tabkeys == x[0]
|
||
+ ), # exact matches first!
|
||
+ -1*x[3], # user_freq descending
|
||
+ # Prefer characters used in the
|
||
+ # desired Chinese variant:
|
||
+ -(bitmask
|
||
+ & chinese_variants.detect_chinese_category(
|
||
+ x[1])),
|
||
+ -1*x[2], # freq descending
|
||
+ len(x[0]), # len(tabkeys) ascending
|
||
+ x[0], # tabkeys alphabetical
|
||
+ code_point_function(x[1][0]),
|
||
+ # Unicode codepoint of first character of phrase:
|
||
+ ord(x[1][0])
|
||
+ ))[:maximum_number_of_candidates]
|
||
+ return sorted(candidates,
|
||
+ key=lambda x: (
|
||
+ - int(
|
||
+ typed_tabkeys == x[0]
|
||
+ ), # exact matches first!
|
||
+ -1*x[3], # user_freq descending
|
||
+ -1*x[2], # freq descending
|
||
+ len(x[0]), # len(tabkeys) ascending
|
||
+ x[0], # tabkeys alphabetical
|
||
+ code_point_function(x[1][0]),
|
||
+ # Unicode codepoint of first character of phrase:
|
||
+ ord(x[1][0])
|
||
+ ))[:maximum_number_of_candidates]
|
||
+
|
||
+ def select_words(
|
||
+ self, tabkeys=u'', onechar=False, chinese_mode=-1,
|
||
+ single_wildcard_char=u'', multi_wildcard_char=u'',
|
||
+ auto_wildcard=False):
|
||
+ '''
|
||
+ Get matching phrases for tabkeys from the database.
|
||
+ '''
|
||
+ if not tabkeys:
|
||
+ return []
|
||
+ # query phrases cache first
|
||
+ best = self._phrases_cache.get(tabkeys)
|
||
+ if best:
|
||
+ return best
|
||
+ one_char_condition = ''
|
||
+ if onechar:
|
||
+ # for some users really like to select only single characters
|
||
+ one_char_condition = ' AND length(phrase)=1 '
|
||
+
|
||
+ if self.user_can_define_phrase or self.dynamic_adjust:
|
||
+ sqlstr = '''
|
||
+ SELECT tabkeys, phrase, freq, user_freq FROM
|
||
+ (
|
||
+ SELECT tabkeys, phrase, freq, user_freq FROM main.phrases
|
||
+ WHERE tabkeys LIKE :tabkeys ESCAPE :escapechar %(one_char_condition)s
|
||
+ UNION ALL
|
||
+ SELECT tabkeys, phrase, freq, user_freq FROM user_db.phrases
|
||
+ WHERE tabkeys LIKE :tabkeys ESCAPE :escapechar %(one_char_condition)s
|
||
+ )
|
||
+ ''' % {'one_char_condition': one_char_condition}
|
||
+ else:
|
||
+ sqlstr = '''
|
||
+ SELECT tabkeys, phrase, freq, user_freq FROM main.phrases
|
||
+ WHERE tabkeys LIKE :tabkeys ESCAPE :escapechar %(one_char_condition)s
|
||
+ ''' % {'one_char_condition': one_char_condition}
|
||
+ escapechar = '☺'
|
||
+ for c in '!@#':
|
||
+ if c not in [single_wildcard_char, multi_wildcard_char]:
|
||
+ escapechar = c
|
||
+ tabkeys_for_like = tabkeys
|
||
+ tabkeys_for_like = tabkeys_for_like.replace(
|
||
+ escapechar, escapechar+escapechar)
|
||
+ if '%' not in [single_wildcard_char, multi_wildcard_char]:
|
||
+ tabkeys_for_like = tabkeys_for_like.replace('%', escapechar+'%')
|
||
+ if '_' not in [single_wildcard_char, multi_wildcard_char]:
|
||
+ tabkeys_for_like = tabkeys_for_like.replace('_', escapechar+'_')
|
||
+ if single_wildcard_char:
|
||
+ tabkeys_for_like = tabkeys_for_like.replace(
|
||
+ single_wildcard_char, '_')
|
||
+ if multi_wildcard_char:
|
||
+ tabkeys_for_like = tabkeys_for_like.replace(
|
||
+ multi_wildcard_char, '%%')
|
||
+ if auto_wildcard:
|
||
+ tabkeys_for_like += '%%'
|
||
+ sqlargs = {'tabkeys': tabkeys_for_like, 'escapechar': escapechar}
|
||
+ unfiltered_results = self.db.execute(sqlstr, sqlargs).fetchall()
|
||
+ bitmask = None
|
||
+ if chinese_mode == 0:
|
||
+ bitmask = (1 << 0) # simplified only
|
||
+ elif chinese_mode == 1:
|
||
+ bitmask = (1 << 1) # traditional only
|
||
+ if not bitmask:
|
||
+ results = unfiltered_results
|
||
+ else:
|
||
+ results = []
|
||
+ for result in unfiltered_results:
|
||
+ if (bitmask
|
||
+ & chinese_variants.detect_chinese_category(result[1])):
|
||
+ results.append(result)
|
||
+ # merge matches from the system database and from the user
|
||
+ # database to avoid duplicates in the candidate list for
|
||
+ # example, if we have the result ('aaaa', '工', 551000000, 0)
|
||
+ # from the system database and ('aaaa', '工', 0, 5) from the
|
||
+ # user database, these should be merged into one match
|
||
+ # ('aaaa', '工', 551000000, 5).
|
||
+ phrase_frequencies = {}
|
||
+ for result in results:
|
||
+ key = (result[0], result[1])
|
||
+ if key not in phrase_frequencies:
|
||
+ phrase_frequencies[key] = result
|
||
+ else:
|
||
+ phrase_frequencies.update([(
|
||
+ key,
|
||
+ key +
|
||
+ (
|
||
+ max(result[2], phrase_frequencies[key][2]),
|
||
+ max(result[3], phrase_frequencies[key][3]))
|
||
+ )])
|
||
+ best = self.best_candidates(
|
||
+ typed_tabkeys=tabkeys,
|
||
+ candidates=phrase_frequencies.values(),
|
||
+ chinese_mode=chinese_mode)
|
||
+ if debug_level > 1:
|
||
+ sys.stderr.write("select_words() best=%s\n" %repr(best))
|
||
+ self._phrases_cache[tabkeys] = best
|
||
+ return best
|
||
+
|
||
+ def select_chinese_characters_by_pinyin(
|
||
+ self, tabkeys=u'', chinese_mode=-1, single_wildcard_char=u'',
|
||
+ multi_wildcard_char=u''):
|
||
+ '''
|
||
+ Get Chinese characters matching the pinyin given by tabkeys
|
||
+ from the database.
|
||
+ '''
|
||
+ if not tabkeys:
|
||
+ return []
|
||
+ sqlstr = '''
|
||
+ SELECT pinyin, zi, freq FROM main.pinyin WHERE pinyin LIKE :tabkeys
|
||
+ ORDER BY freq DESC, pinyin ASC
|
||
+ ;'''
|
||
+ tabkeys_for_like = tabkeys
|
||
+ if single_wildcard_char:
|
||
+ tabkeys_for_like = tabkeys_for_like.replace(
|
||
+ single_wildcard_char, '_')
|
||
+ if multi_wildcard_char:
|
||
+ tabkeys_for_like = tabkeys_for_like.replace(
|
||
+ multi_wildcard_char, '%%')
|
||
+ tabkeys_for_like += '%%'
|
||
+ sqlargs = {'tabkeys': tabkeys_for_like}
|
||
+ results = self.db.execute(sqlstr, sqlargs).fetchall()
|
||
+ # now convert the results into a list of candidates in the format
|
||
+ # which was returned before I simplified the pinyin database table.
|
||
+ bitmask = None
|
||
+ if chinese_mode == 0:
|
||
+ bitmask = (1 << 0) # simplified only
|
||
+ elif chinese_mode == 1:
|
||
+ bitmask = (1 << 1) # traditional only
|
||
+ phrase_frequencies = []
|
||
+ for (pinyin, zi, freq) in results:
|
||
+ if not bitmask:
|
||
+ phrase_frequencies.append(tuple([pinyin, zi, freq, 0]))
|
||
+ else:
|
||
+ if bitmask & chinese_variants.detect_chinese_category(zi):
|
||
+ phrase_frequencies.append(tuple([pinyin, zi, freq, 0]))
|
||
+ return self.best_candidates(
|
||
+ typed_tabkeys=tabkeys,
|
||
+ candidates=phrase_frequencies,
|
||
+ chinese_mode=chinese_mode)
|
||
+
|
||
+ def generate_userdb_desc (self):
|
||
+ try:
|
||
+ sqlstring = (
|
||
+ 'CREATE TABLE IF NOT EXISTS user_db.desc '
|
||
+ + '(name PRIMARY KEY, value);')
|
||
+ self.db.executescript (sqlstring)
|
||
+ sqlstring = 'INSERT OR IGNORE INTO user_db.desc VALUES (?, ?);'
|
||
+ self.db.execute (sqlstring, ('version', database_version))
|
||
+ sqlstring = (
|
||
+ 'INSERT OR IGNORE INTO user_db.desc '
|
||
+ + 'VALUES (?, DATETIME("now", "localtime"));')
|
||
+ self.db.execute (sqlstring, ("create-time", ))
|
||
+ self.db.commit ()
|
||
+ except:
|
||
+ import traceback
|
||
+ traceback.print_exc ()
|
||
+
|
||
+ def init_user_db(self, db_file):
|
||
+ if not path.exists(db_file):
|
||
+ db = sqlite3.connect(db_file)
|
||
+ # 20000 pages should be enough to cache the whole database
|
||
+ db.executescript('''
|
||
+ PRAGMA encoding = "UTF-8";
|
||
+ PRAGMA case_sensitive_like = true;
|
||
+ PRAGMA page_size = 4096;
|
||
+ PRAGMA cache_size = 20000;
|
||
+ PRAGMA temp_store = MEMORY;
|
||
+ PRAGMA journal_mode = WAL;
|
||
+ PRAGMA journal_size_limit = 1000000;
|
||
+ PRAGMA synchronous = NORMAL;
|
||
+ ''')
|
||
+ db.commit()
|
||
+
|
||
+ def get_database_desc (self, db_file):
|
||
+ if not path.exists (db_file):
|
||
+ return None
|
||
+ try:
|
||
+ db = sqlite3.connect (db_file)
|
||
+ desc = {}
|
||
+ for row in db.execute ("SELECT * FROM desc;").fetchall():
|
||
+ desc [row[0]] = row[1]
|
||
+ db.close()
|
||
+ return desc
|
||
+ except:
|
||
+ return None
|
||
+
|
||
+ def get_number_of_columns_of_phrase_table(self, db_file):
|
||
+ '''
|
||
+ Get the number of columns in the 'phrases' table in
|
||
+ the database in db_file.
|
||
+
|
||
+ Determines the number of columns by parsing this:
|
||
+
|
||
+ sqlite> select sql from sqlite_master where name='phrases';
|
||
+ CREATE TABLE phrases
|
||
+ (id INTEGER PRIMARY KEY, tabkeys TEXT, phrase TEXT,
|
||
+ freq INTEGER, user_freq INTEGER)
|
||
+ sqlite>
|
||
+
|
||
+ This result could be on a single line, as above, or on multiple
|
||
+ lines.
|
||
+ '''
|
||
+ if not path.exists (db_file):
|
||
+ return 0
|
||
+ try:
|
||
+ db = sqlite3.connect (db_file)
|
||
+ tp_res = db.execute(
|
||
+ "select sql from sqlite_master where name='phrases';"
|
||
+ ).fetchall()
|
||
+ # Remove possible line breaks from the string where we
|
||
+ # want to match:
|
||
+ string = ' '.join(tp_res[0][0].splitlines())
|
||
+ res = re.match(r'.*\((.*)\)', string)
|
||
+ if res:
|
||
+ tp = res.group(1).split(',')
|
||
+ return len(tp)
|
||
+ else:
|
||
+ return 0
|
||
+ except:
|
||
+ return 0
|
||
+
|
||
+ def get_goucima (self, zi):
|
||
+ '''Get goucima of given character'''
|
||
+ if not zi:
|
||
+ return u''
|
||
+ sqlstr = 'SELECT goucima FROM main.goucima WHERE zi = :zi;'
|
||
+ results = self.db.execute(sqlstr, {'zi': zi}).fetchall()
|
||
+ if results:
|
||
+ goucima = results[0][0]
|
||
+ else:
|
||
+ goucima = u''
|
||
+ if debug_level > 1:
|
||
+ sys.stderr.write("get_goucima() goucima=%s\n" %goucima)
|
||
+ return goucima
|
||
+
|
||
+ def parse_phrase (self, phrase):
|
||
+ '''Parse phrase to get its table code
|
||
+
|
||
+ Example:
|
||
+
|
||
+ Let’s assume we use wubi-jidian86. The rules in the source of
|
||
+ that table are:
|
||
+
|
||
+ RULES = ce2:p11+p12+p21+p22;ce3:p11+p21+p31+p32;ca4:p11+p21+p31+p-11
|
||
+
|
||
+ “ce2” is a rule for phrases of length 2, “ce3” is a rule
|
||
+ for phrases of length 3, “ca4” is a rule for phrases of
|
||
+ length 4 *and* for all phrases with a length greater then
|
||
+ 4. “pnm” in such a rule means to use the n-th character of
|
||
+ the phrase and take the m-th character of the table code of
|
||
+ that character. I.e. “p-11” is the first character of the
|
||
+ table code of the last character in the phrase.
|
||
+
|
||
+ Let’s assume the phrase is “天下大事”. The goucima (構詞碼
|
||
+ = “word formation keys”) for these 4 characters are:
|
||
+
|
||
+ character goucima
|
||
+ 天 gdi
|
||
+ 下 ghi
|
||
+ 大 dddd
|
||
+ 事 gkvh
|
||
+
|
||
+ (If no special goucima are defined by the user, the longest
|
||
+ encoding for a single character in a table is the goucima for
|
||
+ that character).
|
||
+
|
||
+ The length of the phrase “天下大事” is 4 characters,
|
||
+ therefore the rule ca4:p11+p21+p31+p-11 applies, i.e. the
|
||
+ table code for “天下大事” is calculated by using the first,
|
||
+ second, third and last character of the phrase and taking the
|
||
+ first character of the goucima for each of these. Therefore,
|
||
+ the table code for “天下大事” is “ggdg”.
|
||
+
|
||
+ '''
|
||
+ if debug_level > 1:
|
||
+ sys.stderr.write(
|
||
+ 'parse_phrase() phrase=%(p)s rules%(r)s\n'
|
||
+ % {'p': phrase, 'r': self.rules})
|
||
+ if type(phrase) != type(u''):
|
||
+ phrase = phrase.decode('UTF-8')
|
||
+ # Shouldn’t this function try first whether the system database
|
||
+ # already has an entry for this phrase and if yes return it
|
||
+ # instead of constructing a new entry according to the rules?
|
||
+ # And construct a new entry only when no entry already exists
|
||
+ # in the system database??
|
||
+ if len(phrase) == 0:
|
||
+ return u''
|
||
+ if len(phrase) == 1:
|
||
+ return self.get_goucima(phrase)
|
||
+ if not self.rules:
|
||
+ return u''
|
||
+ if len(phrase) in self.rules:
|
||
+ rule = self.rules[len(phrase)]
|
||
+ elif len(phrase) > self.rules['above']:
|
||
+ rule = self.rules[self.rules['above']]
|
||
+ else:
|
||
+ sys.stderr.write(
|
||
+ 'No rule for this phrase length. phrase=%(p)s rules=%(r)s\n'
|
||
+ %{'p': phrase, 'r': self.rules})
|
||
+ return u''
|
||
+ if len(rule) > self._mlen:
|
||
+ sys.stderr.write(
|
||
+ 'Rule exceeds maximum key length. rule=%(r)s self._mlen=%(m)s\n'
|
||
+ %{'r': rule, 'm': self._mlen})
|
||
+ return u''
|
||
+ tabkeys = u''
|
||
+ for (zi, ma) in rule:
|
||
+ if zi > 0:
|
||
+ zi -= 1
|
||
+ if ma > 0:
|
||
+ ma -= 1
|
||
+ tabkey = self.get_goucima(phrase[zi])[ma]
|
||
+ if not tabkey:
|
||
+ return u''
|
||
+ tabkeys += tabkey
|
||
+ if debug_level > 1:
|
||
+ sys.stderr.write("parse_phrase() tabkeys=%s\n" %tabkeys)
|
||
+ return tabkeys
|
||
+
|
||
+ def is_in_system_database(self, tabkeys=u'', phrase=u''):
|
||
+ '''
|
||
+ Checks whether “phrase” can be matched in the system database
|
||
+ with a key sequence *starting* with “tabkeys”.
|
||
+ '''
|
||
+ if debug_level > 1:
|
||
+ sys.stderr.write(
|
||
+ 'is_in_system_database() tabkeys=%(t)s phrase=%(p)s\n'
|
||
+ % {'t': tabkeys, 'p': phrase})
|
||
+ if not tabkeys or not phrase:
|
||
+ return False
|
||
+ sqlstr = '''
|
||
+ SELECT * FROM main.phrases
|
||
+ WHERE tabkeys LIKE :tabkeys AND phrase = :phrase;
|
||
+ '''
|
||
+ sqlargs = {'tabkeys': tabkeys+'%%', 'phrase': phrase}
|
||
+ results = self.db.execute(sqlstr, sqlargs).fetchall()
|
||
+ if debug_level > 1:
|
||
+ sys.stderr.write(
|
||
+ 'is_in_system_database() tabkeys=%(t)s phrase=%(p)s '
|
||
+ % {'t': tabkeys, 'p': phrase}
|
||
+ + 'results=%(r)s\n'
|
||
+ % {'r': results})
|
||
+ if results:
|
||
+ return True
|
||
+ else:
|
||
+ return False
|
||
+
|
||
+ def user_frequency(self, tabkeys=u'', phrase=u''):
|
||
+ if debug_level > 1:
|
||
+ sys.stderr.write(
|
||
+ 'user_frequency() tabkeys=%(t)s phrase=%(p)s\n'
|
||
+ % {'t': tabkeys, 'p': phrase})
|
||
+ if not tabkeys or not phrase:
|
||
+ return 0
|
||
+ sqlstr = '''
|
||
+ SELECT sum(user_freq) FROM user_db.phrases
|
||
+ WHERE tabkeys = :tabkeys AND phrase = :phrase GROUP BY tabkeys, phrase;
|
||
+ '''
|
||
+ sqlargs = {'tabkeys': tabkeys, 'phrase': phrase}
|
||
+ result = self.db.execute(sqlstr, sqlargs).fetchall()
|
||
+ if debug_level > 1:
|
||
+ sys.stderr.write("user_frequency() result=%s\n" %result)
|
||
+ if result:
|
||
+ return result[0][0]
|
||
+ else:
|
||
+ return 0
|
||
+
|
||
+ def check_phrase(self, tabkeys=u'', phrase=u''):
|
||
+ '''Adjust user_freq in user database if necessary.
|
||
+
|
||
+ Also, if the phrase is not in the system database, and it is a
|
||
+ Chinese table, and defining user phrases is allowed, add it as
|
||
+ a user defined phrase to the user database if it is not yet
|
||
+ there.
|
||
+ '''
|
||
+ if debug_level > 1:
|
||
+ sys.stderr.write(
|
||
+ 'check_phrase_internal() tabkey=%(t)s phrase=%(p)s\n'
|
||
+ % {'t': tabkeys, 'p': phrase})
|
||
+ if type(phrase) != type(u''):
|
||
+ phrase = phrase.decode('utf8')
|
||
+ if type(tabkeys) != type(u''):
|
||
+ tabkeys = tabkeys.decode('utf8')
|
||
+ if not tabkeys or not phrase:
|
||
+ return
|
||
+ if self._is_chinese and phrase in chinese_nocheck_chars:
|
||
+ return
|
||
+ if not self.dynamic_adjust:
|
||
+ if not self.user_can_define_phrase or not self.is_chinese:
|
||
+ return
|
||
+ tabkeys = self.parse_phrase(phrase)
|
||
+ if not tabkeys:
|
||
+ # no tabkeys could be constructed from the rules in the table
|
||
+ return
|
||
+ if self.is_in_system_database(tabkeys=tabkeys, phrase=phrase):
|
||
+ # if it is in the system database, it does not need to
|
||
+ # be defined
|
||
+ return
|
||
+ if self.user_frequency(tabkeys=tabkeys, phrase=phrase) > 0:
|
||
+ # if it is in the user database, it has been defined before
|
||
+ return
|
||
+ # add this user defined phrase to the user database:
|
||
+ self.add_phrase(
|
||
+ tabkeys=tabkeys, phrase=phrase, freq=-1, user_freq=1,
|
||
+ database='user_db')
|
||
+ else:
|
||
+ if self.is_in_system_database(tabkeys=tabkeys, phrase=phrase):
|
||
+ user_freq = self.user_frequency(tabkeys=tabkeys, phrase=phrase)
|
||
+ if user_freq > 0:
|
||
+ self.update_phrase(
|
||
+ tabkeys=tabkeys, phrase=phrase, user_freq=user_freq+1)
|
||
+ else:
|
||
+ self.add_phrase(
|
||
+ tabkeys=tabkeys, phrase=phrase, freq=0, user_freq=1,
|
||
+ database='user_db')
|
||
+ else:
|
||
+ if not self.user_can_define_phrase or not self.is_chinese:
|
||
+ return
|
||
+ tabkeys = self.parse_phrase(phrase)
|
||
+ if not tabkeys:
|
||
+ # no tabkeys could be constructed from the rules
|
||
+ # in the table
|
||
+ return
|
||
+ user_freq = self.user_frequency(tabkeys=tabkeys, phrase=phrase)
|
||
+ if user_freq > 0:
|
||
+ self.update_phrase(
|
||
+ tabkeys=tabkeys, phrase=phrase, user_freq=user_freq+1)
|
||
+ else:
|
||
+ self.add_phrase(
|
||
+ tabkeys=tabkeys, phrase=phrase, freq=-1, user_freq=1,
|
||
+ database='user_db')
|
||
+
|
||
+ def find_zi_code (self, phrase):
|
||
+ '''
|
||
+ Return the list of possible tabkeys for a phrase.
|
||
+
|
||
+ For example, if “phrase” is “你” and the table is wubi-jidian.86.txt,
|
||
+ the result will be ['wq', 'wqi', 'wqiy'] because that table
|
||
+ contains the following 3 lines matching that phrase exactly:
|
||
+
|
||
+ wq 你 597727619
|
||
+ wqi 你 1490000000
|
||
+ wqiy 你 1490000000
|
||
+ '''
|
||
+ if type(phrase) != type(u''):
|
||
+ phrase = phrase.decode('utf8')
|
||
+ sqlstr = '''
|
||
+ SELECT tabkeys FROM main.phrases WHERE phrase = :phrase
|
||
+ ORDER by length(tabkeys) ASC;
|
||
+ '''
|
||
+ sqlargs = {'phrase': phrase}
|
||
+ results = self.db.execute(sqlstr, sqlargs).fetchall()
|
||
+ list_of_possible_tabkeys = [x[0] for x in results]
|
||
+ return list_of_possible_tabkeys
|
||
+
|
||
+ def remove_phrase (
|
||
+ self, tabkeys=u'', phrase=u'', database='user_db', commit=True):
|
||
+ '''Remove phrase from database
|
||
+ '''
|
||
+ if not phrase:
|
||
+ return
|
||
+ if tabkeys:
|
||
+ delete_sqlstr = '''
|
||
+ DELETE FROM %(database)s.phrases
|
||
+ WHERE tabkeys = :tabkeys AND phrase = :phrase;
|
||
+ ''' % {'database': database}
|
||
+ else:
|
||
+ delete_sqlstr = '''
|
||
+ DELETE FROM %(database)s.phrases
|
||
+ WHERE phrase = :phrase;
|
||
+ ''' % {'database': database}
|
||
+ delete_sqlargs = {'tabkeys': tabkeys, 'phrase': phrase}
|
||
+ self.db.execute(delete_sqlstr, delete_sqlargs)
|
||
+ if commit:
|
||
+ self.db.commit()
|
||
+ self.invalidate_phrases_cache(tabkeys)
|
||
+
|
||
+ def extract_user_phrases(
|
||
+ self, database_file='', old_database_version='0.0'):
|
||
+ '''extract user phrases from database'''
|
||
+ sys.stderr.write(
|
||
+ 'Trying to recover the phrases from the old, '
|
||
+ + 'incompatible database.\n')
|
||
+ try:
|
||
+ db = sqlite3.connect(database_file)
|
||
+ db.execute('PRAGMA wal_checkpoint;')
|
||
+ if old_database_version >= '1.00':
|
||
+ phrases = db.execute(
|
||
+ '''
|
||
+ SELECT tabkeys, phrase, freq, sum(user_freq) FROM phrases
|
||
+ GROUP BY tabkeys, phrase, freq;
|
||
+ '''
|
||
+ ).fetchall()
|
||
+ db.close()
|
||
+ phrases = sorted(
|
||
+ phrases, key=lambda x: (x[0], x[1], x[2], x[3]))
|
||
+ sys.stderr.write(
|
||
+ 'Recovered phrases from the old database: phrases=%s\n'
|
||
+ % repr(phrases))
|
||
+ return phrases[:]
|
||
+ else:
|
||
+ # database is very old, it may still use many columns
|
||
+ # of type INTEGER for the tabkeys. Therefore, ignore
|
||
+ # the tabkeys in the database and try to get them
|
||
+ # from the system database instead.
|
||
+ phrases = []
|
||
+ results = db.execute(
|
||
+ 'SELECT phrase, sum(user_freq) '
|
||
+ + 'FROM phrases GROUP BY phrase;'
|
||
+ ).fetchall()
|
||
+ for result in results:
|
||
+ sqlstr = '''
|
||
+ SELECT tabkeys FROM main.phrases WHERE phrase = :phrase
|
||
+ ORDER BY length(tabkeys) DESC;
|
||
+ '''
|
||
+ sqlargs = {'phrase': result[0]}
|
||
+ tabkeys_results = self.db.execute(
|
||
+ sqlstr, sqlargs).fetchall()
|
||
+ if tabkeys_results:
|
||
+ phrases.append(
|
||
+ (tabkeys_results[0][0], result[0], 0, result[1]))
|
||
+ else:
|
||
+ # No tabkeys for that phrase could not be
|
||
+ # found in the system database. Try to get
|
||
+ # tabkeys by calling self.parse_phrase(), that
|
||
+ # might return something if the table has
|
||
+ # rules to construct user defined phrases:
|
||
+ tabkeys = self.parse_phrase(result[0])
|
||
+ if tabkeys:
|
||
+ # for user defined phrases, the “freq” column is -1:
|
||
+ phrases.append((tabkeys, result[0], -1, result[1]))
|
||
+ db.close()
|
||
+ phrases = sorted(
|
||
+ phrases, key=lambda x: (x[0], x[1], x[2], x[3]))
|
||
+ sys.stderr.write(
|
||
+ 'Recovered phrases from the very old database: phrases=%s\n'
|
||
+ % repr(phrases))
|
||
+ return phrases[:]
|
||
+ except:
|
||
+ import traceback
|
||
+ traceback.print_exc()
|
||
+ return []
|
||
diff -Nru ibus-table-1.9.18.orig/setup/main.py ibus-table-1.9.18/setup/main.py
|
||
--- ibus-table-1.9.18.orig/setup/main.py 2020-07-22 11:52:11.654532080 +0200
|
||
+++ ibus-table-1.9.18/setup/main.py 2020-07-22 14:43:51.908260925 +0200
|
||
@@ -62,7 +62,7 @@
|
||
"endeffullwidthletter": False,
|
||
"endeffullwidthpunct": False,
|
||
"alwaysshowlookup": True,
|
||
- "lookuptableorientation": True,
|
||
+ "lookuptableorientation": 1, # 0 = horizontal, 1 = vertical, 2 = system
|
||
"lookuptablepagesize": 6,
|
||
"onechar": False,
|
||
"autoselect": False,
|
||
@@ -224,12 +224,8 @@
|
||
and type(auto_commit) == type(u'')
|
||
and auto_commit.lower() in [u'true', u'false']):
|
||
OPTION_DEFAULTS['autocommit'] = auto_commit.lower() == u'true'
|
||
- orientation = self.tabsqlitedb.ime_properties.get('orientation')
|
||
- if (orientation
|
||
- and type(orientation) == type(u'')
|
||
- and orientation.lower() in [u'true', u'false']):
|
||
- OPTION_DEFAULTS['lookuptableorientation'] = (
|
||
- orientation.lower() == u'true')
|
||
+ orientation = self.tabsqlitedb.get_orientation()
|
||
+ OPTION_DEFAULTS['lookuptableorientation'] = orientation
|
||
# if space is a page down key, set the option
|
||
# “spacekeybehavior” to “True”:
|
||
page_down_keys_csv = self.tabsqlitedb.ime_properties.get(
|
||
@@ -258,14 +254,16 @@
|
||
single_wildcard_char = self.tabsqlitedb.ime_properties.get(
|
||
'single_wildcard_char')
|
||
if (single_wildcard_char
|
||
- and type(single_wildcard_char) == type(u'')
|
||
- and len(single_wildcard_char) == 1):
|
||
+ and type(single_wildcard_char) == type(u'')):
|
||
+ if len(single_wildcard_char) > 1:
|
||
+ single_wildcard_char = single_wildcard_char[0]
|
||
OPTION_DEFAULTS['singlewildcardchar'] = single_wildcard_char
|
||
multi_wildcard_char = self.tabsqlitedb.ime_properties.get(
|
||
'multi_wildcard_char')
|
||
if (multi_wildcard_char
|
||
- and type(multi_wildcard_char) == type(u'')
|
||
- and len(multi_wildcard_char) == 1):
|
||
+ and type(multi_wildcard_char) == type(u'')):
|
||
+ if len(multi_wildcard_char) > 1:
|
||
+ multi_wildcard_char = multi_wildcard_char[0]
|
||
OPTION_DEFAULTS['multiwildcardchar'] = multi_wildcard_char
|
||
|
||
def __restore_defaults(self):
|
||
diff -Nru ibus-table-1.9.18.orig/tests/.gitignore ibus-table-1.9.18/tests/.gitignore
|
||
--- ibus-table-1.9.18.orig/tests/.gitignore 1970-01-01 01:00:00.000000000 +0100
|
||
+++ ibus-table-1.9.18/tests/.gitignore 2020-07-22 14:43:51.908260925 +0200
|
||
@@ -0,0 +1,5 @@
|
||
+run_tests
|
||
+run_tests.log
|
||
+run_tests.trs
|
||
+test-suite.log
|
||
+__pycache__/
|
||
diff -Nru ibus-table-1.9.18.orig/tests/Makefile.am ibus-table-1.9.18/tests/Makefile.am
|
||
--- ibus-table-1.9.18.orig/tests/Makefile.am 1970-01-01 01:00:00.000000000 +0100
|
||
+++ ibus-table-1.9.18/tests/Makefile.am 2020-07-22 14:43:51.908260925 +0200
|
||
@@ -0,0 +1,41 @@
|
||
+# vim:set noet ts=4
|
||
+#
|
||
+# ibus-table - The Tables engine for IBus
|
||
+#
|
||
+# Copyright (c) 2018 Mike FABIAN <mfabian@redhat.com>
|
||
+#
|
||
+# 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.1 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
||
+#
|
||
+
|
||
+TESTS = run_tests
|
||
+
|
||
+run_tests: run_tests.in
|
||
+ sed -e 's&@PYTHON_BIN@&$(PYTHON)&g' \
|
||
+ -e 's&@SRCDIR@&$(srcdir)&g' $< > $@
|
||
+ chmod +x $@
|
||
+
|
||
+EXTRA_DIST = \
|
||
+ run_tests.in \
|
||
+ test_it.py \
|
||
+ __init__.py \
|
||
+ $(NULL)
|
||
+
|
||
+CLEANFILES = \
|
||
+ run_tests \
|
||
+ $(NULL)
|
||
+
|
||
+MAINTAINERCLEANFILES = \
|
||
+ Makefile.in \
|
||
+ $(NULL)
|
||
diff -Nru ibus-table-1.9.18.orig/tests/run_tests.in ibus-table-1.9.18/tests/run_tests.in
|
||
--- ibus-table-1.9.18.orig/tests/run_tests.in 1970-01-01 01:00:00.000000000 +0100
|
||
+++ ibus-table-1.9.18/tests/run_tests.in 2020-07-22 14:43:51.908260925 +0200
|
||
@@ -0,0 +1,254 @@
|
||
+#!/usr/bin/python3
|
||
+
|
||
+# ibus-table - The Tables engine for IBus
|
||
+#
|
||
+# Copyright (c) 2018 Mike FABIAN <mfabian@redhat.com>
|
||
+#
|
||
+# 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.1 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
||
+#
|
||
+
|
||
+import os
|
||
+import sys
|
||
+import unittest
|
||
+
|
||
+from gi import require_version
|
||
+require_version('IBus', '1.0')
|
||
+from gi.repository import IBus
|
||
+
|
||
+# -- Define some mock classes for the tests ----------------------------------
|
||
+class MockEngine:
|
||
+ def __init__(self, engine_name = '', connection = None, object_path = ''):
|
||
+ self.mock_auxiliary_text = ''
|
||
+ self.mock_preedit_text = ''
|
||
+ self.mock_preedit_text_cursor_pos = 0
|
||
+ self.mock_preedit_text_visible = True
|
||
+ self.mock_committed_text = ''
|
||
+ self.mock_committed_text_cursor_pos = 0
|
||
+ self.client_capabilities = (
|
||
+ IBus.Capabilite.PREEDIT_TEXT
|
||
+ | IBus.Capabilite.AUXILIARY_TEXT
|
||
+ | IBus.Capabilite.LOOKUP_TABLE
|
||
+ | IBus.Capabilite.FOCUS
|
||
+ | IBus.Capabilite.PROPERTY)
|
||
+ # There are lots of weird problems with surrounding text
|
||
+ # which makes this hard to test. Therefore this mock
|
||
+ # engine does not try to support surrounding text, i.e.
|
||
+ # we omit “| IBus.Capabilite.SURROUNDING_TEXT” here.
|
||
+
|
||
+ def update_auxiliary_text(self, text, visible):
|
||
+ self.mock_auxiliary_text = text.text
|
||
+
|
||
+ def hide_auxiliary_text(self):
|
||
+ pass
|
||
+
|
||
+ def commit_text(self, text):
|
||
+ self.mock_committed_text = (
|
||
+ self.mock_committed_text[
|
||
+ :self.mock_committed_text_cursor_pos]
|
||
+ + text.text
|
||
+ + self.mock_committed_text[
|
||
+ self.mock_committed_text_cursor_pos:])
|
||
+ self.mock_committed_text_cursor_pos += len(text.text)
|
||
+
|
||
+ def forward_key_event(self, val, code, state):
|
||
+ if (val == IBus.KEY_Left
|
||
+ and self.mock_committed_text_cursor_pos > 0):
|
||
+ self.mock_committed_text_cursor_pos -= 1
|
||
+ return
|
||
+ unicode = IBus.keyval_to_unicode(val)
|
||
+ if unicode:
|
||
+ self.mock_committed_text = (
|
||
+ self.mock_committed_text[
|
||
+ :self.mock_committed_text_cursor_pos]
|
||
+ + unicode
|
||
+ + self.mock_committed_text[
|
||
+ self.mock_committed_text_cursor_pos:])
|
||
+ self.mock_committed_text_cursor_pos += len(unicode)
|
||
+
|
||
+ def update_lookup_table(self, table, visible):
|
||
+ pass
|
||
+
|
||
+ def update_preedit_text(self, text, cursor_pos, visible):
|
||
+ self.mock_preedit_text = text.get_text()
|
||
+ self.mock_preedit_text_cursor_pos = cursor_pos
|
||
+ self.mock_preedit_text_visible = visible
|
||
+
|
||
+ def register_properties(self, property_list):
|
||
+ pass
|
||
+
|
||
+ def update_property(self, property):
|
||
+ pass
|
||
+
|
||
+ def hide_lookup_table(self):
|
||
+ pass
|
||
+
|
||
+class MockLookupTable:
|
||
+ def __init__(self, page_size = 9, cursor_pos = 0, cursor_visible = False, round = True):
|
||
+ self.clear()
|
||
+ self.mock_page_size = page_size
|
||
+ self.mock_cursor_pos = cursor_pos
|
||
+ self.mock_cursor_visible = cursor_visible
|
||
+ self.cursor_visible = cursor_visible
|
||
+ self.mock_round = round
|
||
+ self.mock_candidates = []
|
||
+ self.mock_labels = []
|
||
+ self.mock_page_number = 0
|
||
+
|
||
+ def clear(self):
|
||
+ self.mock_candidates = []
|
||
+ self.mock_cursor_pos = 0
|
||
+
|
||
+ def set_page_size(self, size):
|
||
+ self.mock_page_size = size
|
||
+
|
||
+ def get_page_size(self):
|
||
+ return self.mock_page_size
|
||
+
|
||
+ def set_round(self, round):
|
||
+ self.mock_round = round
|
||
+
|
||
+ def set_cursor_pos(self, pos):
|
||
+ self.mock_cursor_pos = pos
|
||
+
|
||
+ def get_cursor_pos(self):
|
||
+ return self.mock_cursor_pos
|
||
+
|
||
+ def get_cursor_in_page(self):
|
||
+ return (self.mock_cursor_pos
|
||
+ - self.mock_page_size * self.mock_page_number)
|
||
+
|
||
+ def set_cursor_visible(self, visible):
|
||
+ self.mock_cursor_visible = visible
|
||
+ self.cursor_visible = visible
|
||
+
|
||
+ def cursor_down(self):
|
||
+ if len(self.mock_candidates):
|
||
+ self.mock_cursor_pos += 1
|
||
+ self.mock_cursor_pos %= len(self.mock_candidates)
|
||
+
|
||
+ def cursor_up(self):
|
||
+ if len(self.mock_candidates):
|
||
+ if self.mock_cursor_pos > 0:
|
||
+ self.mock_cursor_pos -= 1
|
||
+ else:
|
||
+ self.mock_cursor_pos = len(self.mock_candidates) - 1
|
||
+
|
||
+ def page_down(self):
|
||
+ if len(self.mock_candidates):
|
||
+ self.mock_page_number += 1
|
||
+ self.mock_cursor_pos += self.mock_page_size
|
||
+
|
||
+ def page_up(self):
|
||
+ if len(self.mock_candidates):
|
||
+ if self.mock_page_number > 0:
|
||
+ self.mock_page_number -= 1
|
||
+ self.mock_cursor_pos -= self.mock_page_size
|
||
+
|
||
+ def set_orientation(self, orientation):
|
||
+ self.mock_orientation = orientation
|
||
+
|
||
+ def get_number_of_candidates(self):
|
||
+ return len(self.mock_candidates)
|
||
+
|
||
+ def append_candidate(self, candidate):
|
||
+ self.mock_candidates.append(candidate.get_text())
|
||
+
|
||
+ def get_candidate(self, index):
|
||
+ return self.mock_candidates[index]
|
||
+
|
||
+ def get_number_of_candidates(self):
|
||
+ return len(self.mock_candidates)
|
||
+
|
||
+ def append_label(self, label):
|
||
+ self.mock_labels.append(label.get_text())
|
||
+
|
||
+class MockPropList:
|
||
+ def __init__(self, *args, **kwargs):
|
||
+ self._mock_proplist = []
|
||
+
|
||
+ def append(self, property):
|
||
+ self._mock_proplist.append(property)
|
||
+
|
||
+ def get(self, index):
|
||
+ if index >= 0 and index < len(self._mock_proplist):
|
||
+ return self._mock_proplist[index]
|
||
+ else:
|
||
+ return None
|
||
+
|
||
+ def update_property(self, property):
|
||
+ pass
|
||
+
|
||
+class MockProperty:
|
||
+ def __init__(self,
|
||
+ key='',
|
||
+ prop_type=IBus.PropType.RADIO,
|
||
+ label=IBus.Text.new_from_string(''),
|
||
+ symbol=IBus.Text.new_from_string(''),
|
||
+ icon='',
|
||
+ tooltip=IBus.Text.new_from_string(''),
|
||
+ sensitive=True,
|
||
+ visible=True,
|
||
+ state=IBus.PropState.UNCHECKED,
|
||
+ sub_props=None):
|
||
+ self.mock_property_key = key
|
||
+ self.mock_property_prop_type = prop_type
|
||
+ self.mock_property_label = label.get_text()
|
||
+ self.mock_property_symbol = symbol.get_text()
|
||
+ self.mock_property_icon = icon
|
||
+ self.mock_property_tooltip = tooltip.get_text()
|
||
+ self.mock_property_sensitive = sensitive
|
||
+ self.mock_property_visible = visible
|
||
+ self.mock_property_state = state
|
||
+ self.mock_property_sub_props = sub_props
|
||
+
|
||
+ def set_label(self, ibus_text):
|
||
+ self.mock_property_label = ibus_text.get_text()
|
||
+
|
||
+ def set_symbol(self, ibus_text):
|
||
+ self.mock_property_symbol = ibus_text.get_text()
|
||
+
|
||
+ def set_tooltip(self, ibus_text):
|
||
+ self.mock_property_tooltip = ibus_text.get_text()
|
||
+
|
||
+ def set_sensitive(self, sensitive):
|
||
+ self.mock_property_sensitive = sensitive
|
||
+
|
||
+ def set_visible(self, visible):
|
||
+ self.mock_property_visible = visible
|
||
+
|
||
+ def set_state(self, state):
|
||
+ self.mock_property_state = state
|
||
+
|
||
+ def set_sub_props(self, proplist):
|
||
+ self.mock_property_sub_props = proplist
|
||
+
|
||
+ def get_key(self):
|
||
+ return self.mock_property_key
|
||
+
|
||
+# -- Monkey patch the environment with the mock classes ----------------------
|
||
+sys.modules["gi.repository.IBus"].Engine = MockEngine
|
||
+sys.modules["gi.repository.IBus"].LookupTable = MockLookupTable
|
||
+sys.modules["gi.repository.IBus"].Property = MockProperty
|
||
+sys.modules["gi.repository.IBus"].PropList = MockPropList
|
||
+
|
||
+# -- Load and run our unit tests ---------------------------------------------
|
||
+os.environ['IBUS_TABLE_DEBUG_LEVEL'] = '255'
|
||
+loader = unittest.TestLoader()
|
||
+suite = loader.discover(".")
|
||
+runner = unittest.TextTestRunner(stream = sys.stderr, verbosity = 255)
|
||
+result = runner.run(suite)
|
||
+
|
||
+if result.failures or result.errors:
|
||
+ sys.exit(1)
|
||
diff -Nru ibus-table-1.9.18.orig/tests/test_it.py ibus-table-1.9.18/tests/test_it.py
|
||
--- ibus-table-1.9.18.orig/tests/test_it.py 1970-01-01 01:00:00.000000000 +0100
|
||
+++ ibus-table-1.9.18/tests/test_it.py 2020-07-22 14:43:51.908260925 +0200
|
||
@@ -0,0 +1,403 @@
|
||
+# -*- coding: utf-8 -*-
|
||
+# vim:et sts=4 sw=4
|
||
+#
|
||
+# ibus-table - The Tables engine for IBus
|
||
+#
|
||
+# Copyright (c) 2018 Mike FABIAN <mfabian@redhat.com>
|
||
+#
|
||
+# 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.1 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
||
+#
|
||
+
|
||
+'''
|
||
+This file implements the test cases for the unit tests of ibus-table
|
||
+'''
|
||
+
|
||
+import sys
|
||
+import os
|
||
+import unicodedata
|
||
+import unittest
|
||
+import subprocess
|
||
+
|
||
+from gi import require_version
|
||
+require_version('IBus', '1.0')
|
||
+from gi.repository import IBus
|
||
+
|
||
+sys.path.insert(0, "../engine")
|
||
+from table import *
|
||
+import tabsqlitedb
|
||
+import it_util
|
||
+#sys.path.pop(0)
|
||
+
|
||
+ENGINE = None
|
||
+TABSQLITEDB = None
|
||
+ORIG_INPUT_MODE = None
|
||
+ORIG_CHINESE_MODE = None
|
||
+ORIG_LETTER_WIDTH = None
|
||
+ORIG_PUNCTUATION_WIDTH = None
|
||
+ORIG_ALWAYS_SHOW_LOOKUP = None
|
||
+ORIG_LOOKUP_TABLE_ORIENTATION = None
|
||
+ORIG_PAGE_SIZE = None
|
||
+ORIG_ONECHAR_MODE = None
|
||
+ORIG_AUTOSELECT_MODE = None
|
||
+ORIG_AUTOCOMMIT_MODE = None
|
||
+ORIG_SPACE_KEY_BEHAVIOR_MODE = None
|
||
+ORIG_AUTOWILDCARD_MODE = None
|
||
+ORIG_SINGLE_WILDCARD_CHAR = None
|
||
+ORIG_MULTI_WILDCARD_CHAR = None
|
||
+
|
||
+def backup_original_settings():
|
||
+ global ENGINE
|
||
+ global ORIG_INPUT_MODE
|
||
+ global ORIG_CHINESE_MODE
|
||
+ global ORIG_LETTER_WIDTH
|
||
+ global ORIG_PUNCTUATION_WIDTH
|
||
+ global ORIG_ALWAYS_SHOW_LOOKUP
|
||
+ global ORIG_LOOKUP_TABLE_ORIENTATION
|
||
+ global ORIG_PAGE_SIZE
|
||
+ global ORIG_ONECHAR_MODE
|
||
+ global ORIG_AUTOSELECT_MODE
|
||
+ global ORIG_AUTOCOMMIT_MODE
|
||
+ global ORIG_SPACE_KEY_BEHAVIOR_MODE
|
||
+ global ORIG_AUTOWILDCARD_MODE
|
||
+ global ORIG_SINGLE_WILDCARD_CHAR
|
||
+ global ORIG_MULTI_WILDCARD_CHAR
|
||
+ ORIG_INPUT_MODE = ENGINE.get_input_mode()
|
||
+ ORIG_CHINESE_MODE = ENGINE.get_chinese_mode()
|
||
+ ORIG_LETTER_WIDTH = ENGINE.get_letter_width()
|
||
+ ORIG_PUNCTUATION_WIDTH = ENGINE.get_punctuation_width()
|
||
+ ORIG_ALWAYS_SHOW_LOOKUP = ENGINE.get_always_show_lookup()
|
||
+ ORIG_LOOKUP_TABLE_ORIENTATION = ENGINE.get_lookup_table_orientation()
|
||
+ ORIG_PAGE_SIZE = ENGINE.get_page_size()
|
||
+ ORIG_ONECHAR_MODE = ENGINE.get_onechar_mode()
|
||
+ ORIG_AUTOSELECT_MODE = ENGINE.get_autoselect_mode()
|
||
+ ORIG_AUTOCOMMIT_MODE = ENGINE.get_autocommit_mode()
|
||
+ ORIG_SPACE_KEY_BEHAVIOR_MODE = ENGINE.get_space_key_behavior_mode()
|
||
+ ORIG_AUTOWILDCARD_MODE = ENGINE.get_autowildcard_mode()
|
||
+ ORIG_SINGLE_WILDCARD_CHAR = ENGINE.get_single_wildcard_char()
|
||
+ ORIG_MULTI_WILDCARD_CHAR = ENGINE.get_multi_wildcard_char()
|
||
+
|
||
+def restore_original_settings():
|
||
+ global ENGINE
|
||
+ global ORIG_INPUT_MODE
|
||
+ global ORIG_CHINESE_MODE
|
||
+ global ORIG_LETTER_WIDTH
|
||
+ global ORIG_PUNCTUATION_WIDTH
|
||
+ global ORIG_ALWAYS_SHOW_LOOKUP
|
||
+ global ORIG_LOOKUP_TABLE_ORIENTATION
|
||
+ global ORIG_PAGE_SIZE
|
||
+ global ORIG_ONECHAR_MODE
|
||
+ global ORIG_AUTOSELECT_MODE
|
||
+ global ORIG_AUTOCOMMIT_MODE
|
||
+ global ORIG_SPACE_KEY_BEHAVIOR_MODE
|
||
+ global ORIG_AUTOWILDCARD_MODE
|
||
+ global ORIG_SINGLE_WILDCARD_CHAR
|
||
+ global ORIG_MULTI_WILDCARD_CHAR
|
||
+ ENGINE.set_input_mode(ORIG_INPUT_MODE)
|
||
+ ENGINE.set_chinese_mode(ORIG_CHINESE_MODE)
|
||
+ ENGINE.set_letter_width(ORIG_LETTER_WIDTH[0], input_mode=0)
|
||
+ ENGINE.set_letter_width(ORIG_LETTER_WIDTH[1], input_mode=1)
|
||
+ ENGINE.set_punctuation_width(ORIG_PUNCTUATION_WIDTH[0], input_mode=0)
|
||
+ ENGINE.set_punctuation_width(ORIG_PUNCTUATION_WIDTH[1], input_mode=1)
|
||
+ ENGINE.set_always_show_lookup(ORIG_ALWAYS_SHOW_LOOKUP)
|
||
+ ENGINE.set_lookup_table_orientation(ORIG_LOOKUP_TABLE_ORIENTATION)
|
||
+ ENGINE.set_page_size(ORIG_PAGE_SIZE)
|
||
+ ENGINE.set_onechar_mode(ORIG_ONECHAR_MODE)
|
||
+ ENGINE.set_autoselect_mode(ORIG_AUTOSELECT_MODE)
|
||
+ ENGINE.set_autocommit_mode(ORIG_AUTOCOMMIT_MODE)
|
||
+ ENGINE.set_space_key_behavior_mode(ORIG_SPACE_KEY_BEHAVIOR_MODE)
|
||
+ ENGINE.set_autowildcard_mode(ORIG_AUTOWILDCARD_MODE)
|
||
+ ENGINE.set_single_wildcard_char(ORIG_SINGLE_WILDCARD_CHAR)
|
||
+ ENGINE.set_multi_wildcard_char(ORIG_MULTI_WILDCARD_CHAR)
|
||
+
|
||
+def set_default_settings():
|
||
+ global ENGINE
|
||
+ global TABSQLITEDB
|
||
+ ENGINE.set_input_mode(mode=1)
|
||
+ chinese_mode = 0
|
||
+ language_filter = TABSQLITEDB.ime_properties.get('language_filter')
|
||
+ if language_filter in ['cm0', 'cm1', 'cm2', 'cm3', 'cm4']:
|
||
+ chinese_mode = int(language_filter[-1])
|
||
+ ENGINE.set_chinese_mode(mode=chinese_mode)
|
||
+
|
||
+ letter_width_mode = False
|
||
+ def_full_width_letter = TABSQLITEDB.ime_properties.get(
|
||
+ 'def_full_width_letter')
|
||
+ if def_full_width_letter:
|
||
+ letter_width_mode = (def_full_width_letter.lower() == u'true')
|
||
+ ENGINE.set_letter_width(mode=letter_width_mode, input_mode=0)
|
||
+ ENGINE.set_letter_width(mode=letter_width_mode, input_mode=1)
|
||
+
|
||
+ punctuation_width_mode = False
|
||
+ def_full_width_punct = TABSQLITEDB.ime_properties.get(
|
||
+ 'def_full_width_punct')
|
||
+ if def_full_width_punct:
|
||
+ punctuation_width_mode = (def_full_width_punct.lower() == u'true')
|
||
+ ENGINE.set_punctuation_width(mode=punctuation_width_mode, input_mode=0)
|
||
+ ENGINE.set_punctuation_width(mode=punctuation_width_mode, input_mode=1)
|
||
+
|
||
+ always_show_lookup_mode = True
|
||
+ always_show_lookup = TABSQLITEDB.ime_properties.get(
|
||
+ 'always_show_lookup')
|
||
+ if always_show_lookup:
|
||
+ always_show_lookup_mode = (always_show_lookup.lower() == u'true')
|
||
+ ENGINE.set_always_show_lookup(always_show_lookup_mode)
|
||
+
|
||
+ orientation = TABSQLITEDB.get_orientation()
|
||
+ ENGINE.set_lookup_table_orientation(orientation)
|
||
+
|
||
+ page_size = 6
|
||
+ select_keys_csv = TABSQLITEDB.ime_properties.get('select_keys')
|
||
+ # select_keys_csv is something like: "1,2,3,4,5,6,7,8,9,0"
|
||
+ if select_keys_csv:
|
||
+ ENGINE.set_page_size(len(select_keys_csv.split(",")))
|
||
+
|
||
+ onechar = False
|
||
+ ENGINE.set_onechar_mode(onechar)
|
||
+
|
||
+ auto_select_mode = False
|
||
+ auto_select = TABSQLITEDB.ime_properties.get('auto_select')
|
||
+ if auto_select:
|
||
+ auto_select_mode = (auto_select.lower() == u'true')
|
||
+ ENGINE.set_autoselect_mode(auto_select_mode)
|
||
+
|
||
+ auto_commit_mode = False
|
||
+ auto_commit = TABSQLITEDB.ime_properties.get('auto_commit')
|
||
+ if auto_commit:
|
||
+ auto_commit_mode = (auto_commit.lower() == u'true')
|
||
+ ENGINE.set_autocommit_mode(auto_commit_mode)
|
||
+
|
||
+ space_key_behavior_mode = False
|
||
+ # if space is a page down key, set the option
|
||
+ # “spacekeybehavior” to “True”:
|
||
+ page_down_keys_csv = TABSQLITEDB.ime_properties.get(
|
||
+ 'page_down_keys')
|
||
+ if page_down_keys_csv:
|
||
+ page_down_keys = [
|
||
+ IBus.keyval_from_name(x)
|
||
+ for x in page_down_keys_csv.split(',')]
|
||
+ if IBus.KEY_space in page_down_keys:
|
||
+ space_key_behavior_mode = True
|
||
+ # if space is a commit key, set the option
|
||
+ # “spacekeybehavior” to “False” (overrides if space is
|
||
+ # also a page down key):
|
||
+ commit_keys_csv = TABSQLITEDB.ime_properties.get('commit_keys')
|
||
+ if commit_keys_csv:
|
||
+ commit_keys = [
|
||
+ IBus.keyval_from_name(x)
|
||
+ for x in commit_keys_csv.split(',')]
|
||
+ if IBus.KEY_space in commit_keys:
|
||
+ space_key_behavior_mode = False
|
||
+ ENGINE.set_space_key_behavior_mode(space_key_behavior_mode)
|
||
+
|
||
+ auto_wildcard_mode = True
|
||
+ auto_wildcard = TABSQLITEDB.ime_properties.get('auto_wildcard')
|
||
+ if auto_wildcard:
|
||
+ auto_wildcard_mode = (auto_wildcard.lower() == u'true')
|
||
+ ENGINE.set_autowildcard_mode(auto_wildcard_mode)
|
||
+
|
||
+ single_wildcard_char = TABSQLITEDB.ime_properties.get(
|
||
+ 'single_wildcard_char')
|
||
+ if not single_wildcard_char:
|
||
+ single_wildcard_char = u''
|
||
+ if len(single_wildcard_char) > 1:
|
||
+ single_wildcard_char = single_wildcard_char[0]
|
||
+ ENGINE.set_single_wildcard_char(single_wildcard_char)
|
||
+
|
||
+ multi_wildcard_char = TABSQLITEDB.ime_properties.get(
|
||
+ 'multi_wildcard_char')
|
||
+ if not multi_wildcard_char:
|
||
+ multi_wildcard_char = u''
|
||
+ if len(multi_wildcard_char) > 1:
|
||
+ multi_wildcard_char = multi_wildcard_char[0]
|
||
+ ENGINE.set_multi_wildcard_char(multi_wildcard_char)
|
||
+
|
||
+def set_up(engine_name):
|
||
+ global TABSQLITEDB
|
||
+ global ENGINE
|
||
+ bus = IBus.Bus()
|
||
+ db_dir = '/usr/share/ibus-table/tables'
|
||
+ db_file = os.path.join(db_dir, engine_name + '.db')
|
||
+ TABSQLITEDB = tabsqlitedb.tabsqlitedb(
|
||
+ filename=db_file, user_db=':memory:')
|
||
+ ENGINE = tabengine(
|
||
+ bus,
|
||
+ '/com/redhat/IBus/engines/table/%s/engine/0' %engine_name,
|
||
+ TABSQLITEDB,
|
||
+ unit_test = True)
|
||
+ backup_original_settings()
|
||
+ set_default_settings()
|
||
+
|
||
+def tear_down():
|
||
+ restore_original_settings()
|
||
+
|
||
+class Wubi_Jidian86TestCase(unittest.TestCase):
|
||
+ def setUp(self):
|
||
+ set_up('wubi-jidian86')
|
||
+
|
||
+ def tearDown(self):
|
||
+ tear_down()
|
||
+
|
||
+ def test_dummy(self):
|
||
+ self.assertEqual(True, True)
|
||
+
|
||
+ def test_single_char_commit_with_space(self):
|
||
+ ENGINE.do_process_key_event(IBus.KEY_a, 0, 0)
|
||
+ ENGINE.do_process_key_event(IBus.KEY_space, 0, 0)
|
||
+ self.assertEqual(ENGINE.mock_committed_text, '工')
|
||
+
|
||
+ def test_commit_to_preedit_and_switching_to_pinyin_and_defining_a_phrase(self):
|
||
+ ENGINE.do_process_key_event(IBus.KEY_a, 0, 0)
|
||
+ # commit to preëdit needs a press and release of either
|
||
+ # the left or the right shift key:
|
||
+ ENGINE.do_process_key_event(
|
||
+ IBus.KEY_Shift_L, 0,
|
||
+ IBus.ModifierType.SHIFT_MASK)
|
||
+ ENGINE.do_process_key_event(
|
||
+ IBus.KEY_Shift_L, 0,
|
||
+ IBus.ModifierType.SHIFT_MASK | IBus.ModifierType.RELEASE_MASK)
|
||
+ self.assertEqual(ENGINE.mock_preedit_text, '工')
|
||
+ ENGINE.do_process_key_event(IBus.KEY_b, 0, 0)
|
||
+ ENGINE.do_process_key_event(
|
||
+ IBus.KEY_Shift_R, 0,
|
||
+ IBus.ModifierType.SHIFT_MASK)
|
||
+ ENGINE.do_process_key_event(
|
||
+ IBus.KEY_Shift_R, 0,
|
||
+ IBus.ModifierType.SHIFT_MASK | IBus.ModifierType.RELEASE_MASK)
|
||
+ self.assertEqual(ENGINE.mock_preedit_text, '工了')
|
||
+ ENGINE.do_process_key_event(IBus.KEY_c, 0, 0)
|
||
+ ENGINE.do_process_key_event(
|
||
+ IBus.KEY_Shift_R, 0,
|
||
+ IBus.ModifierType.SHIFT_MASK)
|
||
+ ENGINE.do_process_key_event(
|
||
+ IBus.KEY_Shift_R, 0,
|
||
+ IBus.ModifierType.SHIFT_MASK | IBus.ModifierType.RELEASE_MASK)
|
||
+ self.assertEqual(ENGINE.mock_preedit_text, '工了以')
|
||
+ ENGINE.do_process_key_event(IBus.KEY_d, 0, 0)
|
||
+ ENGINE.do_process_key_event(
|
||
+ IBus.KEY_Shift_L, 0,
|
||
+ IBus.ModifierType.SHIFT_MASK)
|
||
+ ENGINE.do_process_key_event(
|
||
+ IBus.KEY_Shift_L, 0,
|
||
+ IBus.ModifierType.SHIFT_MASK | IBus.ModifierType.RELEASE_MASK)
|
||
+ self.assertEqual(ENGINE.mock_preedit_text, '工了以在')
|
||
+ # Move left two characters in the preëdit:
|
||
+ ENGINE.do_process_key_event(IBus.KEY_Left, 0, 0)
|
||
+ ENGINE.do_process_key_event(IBus.KEY_Left, 0, 0)
|
||
+ # Switch to pinyin mode by pressing and releasing the right
|
||
+ # shift key:
|
||
+ ENGINE.do_process_key_event(
|
||
+ IBus.KEY_Shift_R, 0,
|
||
+ IBus.ModifierType.SHIFT_MASK)
|
||
+ ENGINE.do_process_key_event(
|
||
+ IBus.KEY_Shift_R, 0,
|
||
+ IBus.ModifierType.SHIFT_MASK | IBus.ModifierType.RELEASE_MASK)
|
||
+ ENGINE.do_process_key_event(IBus.KEY_n, 0, 0)
|
||
+ ENGINE.do_process_key_event(IBus.KEY_i, 0, 0)
|
||
+ ENGINE.do_process_key_event(IBus.KEY_numbersign, 0, 0)
|
||
+ ENGINE.do_process_key_event(
|
||
+ IBus.KEY_Shift_L, 0,
|
||
+ IBus.ModifierType.SHIFT_MASK)
|
||
+ ENGINE.do_process_key_event(
|
||
+ IBus.KEY_Shift_L, 0,
|
||
+ IBus.ModifierType.SHIFT_MASK | IBus.ModifierType.RELEASE_MASK)
|
||
+ self.assertEqual(ENGINE.mock_preedit_text, '工了你以在')
|
||
+ ENGINE.do_process_key_event(IBus.KEY_h, 0, 0)
|
||
+ ENGINE.do_process_key_event(IBus.KEY_a, 0, 0)
|
||
+ ENGINE.do_process_key_event(IBus.KEY_o, 0, 0)
|
||
+ ENGINE.do_process_key_event(IBus.KEY_numbersign, 0, 0)
|
||
+ ENGINE.do_process_key_event(
|
||
+ IBus.KEY_Shift_L, 0,
|
||
+ IBus.ModifierType.SHIFT_MASK)
|
||
+ ENGINE.do_process_key_event(
|
||
+ IBus.KEY_Shift_L, 0,
|
||
+ IBus.ModifierType.SHIFT_MASK | IBus.ModifierType.RELEASE_MASK)
|
||
+ self.assertEqual(ENGINE.mock_preedit_text, '工了你好以在')
|
||
+ # Move right two characters in the preëdit (triggers a commit to preëdit):
|
||
+ ENGINE.do_process_key_event(IBus.KEY_Right, 0, 0)
|
||
+ ENGINE.do_process_key_event(IBus.KEY_Right, 0, 0)
|
||
+ self.assertEqual(ENGINE.mock_auxiliary_text, 'd dhf dhfd\t#: abwd')
|
||
+ # commit the preëdit:
|
||
+ ENGINE.do_process_key_event(IBus.KEY_space, 0, 0)
|
||
+ self.assertEqual(ENGINE.mock_committed_text, '工了你好以在')
|
||
+ # Switch out of pinyin mode:
|
||
+ ENGINE.do_process_key_event(
|
||
+ IBus.KEY_Shift_R, 0,
|
||
+ IBus.ModifierType.SHIFT_MASK)
|
||
+ ENGINE.do_process_key_event(
|
||
+ IBus.KEY_Shift_R, 0,
|
||
+ IBus.ModifierType.SHIFT_MASK | IBus.ModifierType.RELEASE_MASK)
|
||
+ # “abwd” shown on the right of the auxiliary text above shows the
|
||
+ # newly defined shortcut for this phrase. Let’s try to type
|
||
+ # the same phrase again using the new shortcut:
|
||
+ ENGINE.do_process_key_event(IBus.KEY_a, 0, 0)
|
||
+ self.assertEqual(ENGINE.mock_preedit_text, '工')
|
||
+ ENGINE.do_process_key_event(IBus.KEY_b, 0, 0)
|
||
+ self.assertEqual(ENGINE.mock_preedit_text, '节')
|
||
+ ENGINE.do_process_key_event(IBus.KEY_w, 0, 0)
|
||
+ self.assertEqual(ENGINE.mock_preedit_text, '工了你好以在')
|
||
+ ENGINE.do_process_key_event(IBus.KEY_d, 0, 0)
|
||
+ self.assertEqual(ENGINE.mock_preedit_text, '工了你好以在')
|
||
+ # commit the preëdit:
|
||
+ ENGINE.do_process_key_event(IBus.KEY_space, 0, 0)
|
||
+ self.assertEqual(ENGINE.mock_committed_text, '工了你好以在工了你好以在')
|
||
+
|
||
+class Stroke5TestCase(unittest.TestCase):
|
||
+ def setUp(self):
|
||
+ set_up('stroke5')
|
||
+
|
||
+ def tearDown(self):
|
||
+ tear_down()
|
||
+
|
||
+ def test_dummy(self):
|
||
+ self.assertEqual(True, True)
|
||
+
|
||
+ def test_single_char_commit_with_space(self):
|
||
+ ENGINE.do_process_key_event(IBus.KEY_comma, 0, 0)
|
||
+ ENGINE.do_process_key_event(IBus.KEY_slash, 0, 0)
|
||
+ ENGINE.do_process_key_event(IBus.KEY_n, 0, 0)
|
||
+ ENGINE.do_process_key_event(IBus.KEY_m, 0, 0)
|
||
+ ENGINE.do_process_key_event(IBus.KEY_m, 0, 0)
|
||
+ ENGINE.do_process_key_event(IBus.KEY_space, 0, 0)
|
||
+ self.assertEqual(ENGINE.mock_committed_text, '的')
|
||
+
|
||
+class TranslitTestCase(unittest.TestCase):
|
||
+ def setUp(self):
|
||
+ set_up('translit')
|
||
+
|
||
+ def tearDown(self):
|
||
+ tear_down()
|
||
+
|
||
+ def test_dummy(self):
|
||
+ self.assertEqual(True, True)
|
||
+
|
||
+ def test_sh_multiple_match(self):
|
||
+ ENGINE.do_process_key_event(IBus.KEY_s, 0, 0)
|
||
+ self.assertEqual(ENGINE.mock_preedit_text, 'с')
|
||
+ ENGINE.do_process_key_event(IBus.KEY_h, 0, 0)
|
||
+ self.assertEqual(ENGINE.mock_preedit_text, 'ш')
|
||
+ ENGINE.do_process_key_event(IBus.KEY_s, 0, 0)
|
||
+ self.assertEqual(ENGINE.mock_committed_text, 'ш')
|
||
+ self.assertEqual(ENGINE.mock_preedit_text, 'с')
|
||
+ ENGINE.do_process_key_event(IBus.KEY_h, 0, 0)
|
||
+ self.assertEqual(ENGINE.mock_committed_text, 'ш')
|
||
+ self.assertEqual(ENGINE.mock_preedit_text, 'ш')
|
||
+ ENGINE.do_process_key_event(IBus.KEY_h, 0, 0)
|
||
+ self.assertEqual(ENGINE.mock_committed_text, 'шщ')
|
||
+ self.assertEqual(ENGINE.mock_preedit_text, '')
|
||
+ ENGINE.do_process_key_event(IBus.KEY_s, 0, 0)
|
||
+ self.assertEqual(ENGINE.mock_committed_text, 'шщ')
|
||
+ self.assertEqual(ENGINE.mock_preedit_text, 'с')
|
||
+ ENGINE.do_process_key_event(IBus.KEY_space, 0, 0)
|
||
+ self.assertEqual(ENGINE.mock_committed_text, 'шщс ')
|