mirror of
https://github.com/MidnightCommander/mc
synced 2024-12-22 04:22:34 +03:00
Ticket #2788 (aspell support)
Add aspell support for internal editor. The aspell library is dinamycally loaded. Signed-off-by: Ilia Maslakov <il.smind@gmail.com>
This commit is contained in:
parent
54b2b2fe67
commit
1d4ca9608a
@ -8,8 +8,10 @@ m4_include([m4.include/fstypename.m4])
|
|||||||
m4_include([m4.include/fsusage.m4])
|
m4_include([m4.include/fsusage.m4])
|
||||||
m4_include([m4.include/mountlist.m4])
|
m4_include([m4.include/mountlist.m4])
|
||||||
m4_include([m4.include/mc-get-fs-info.m4])
|
m4_include([m4.include/mc-get-fs-info.m4])
|
||||||
|
m4_include([m4.include/mc-with-x.m4])
|
||||||
m4_include([m4.include/mc-use-termcap.m4])
|
m4_include([m4.include/mc-use-termcap.m4])
|
||||||
m4_include([m4.include/mc-with-screen.m4])
|
m4_include([m4.include/mc-with-screen.m4])
|
||||||
|
m4_include([m4.include/mc-with-edit.m4])
|
||||||
m4_include([m4.include/mc-background.m4])
|
m4_include([m4.include/mc-background.m4])
|
||||||
m4_include([m4.include/ac-glib.m4])
|
m4_include([m4.include/ac-glib.m4])
|
||||||
m4_include([m4.include/mc-vfs.m4])
|
m4_include([m4.include/mc-vfs.m4])
|
||||||
|
28
configure.ac
28
configure.ac
@ -51,6 +51,7 @@ DX_INIT_DOXYGEN(mc,doxygen.cfg,devel)
|
|||||||
dnl PKG_CHECK_MODULES([CHECK], [check >= 0.9.4])
|
dnl PKG_CHECK_MODULES([CHECK], [check >= 0.9.4])
|
||||||
|
|
||||||
AC_CHECK_GLIB
|
AC_CHECK_GLIB
|
||||||
|
AC_G_MODULE_SUPPORTED
|
||||||
|
|
||||||
AC_ARG_ENABLE([mclib],
|
AC_ARG_ENABLE([mclib],
|
||||||
[AS_HELP_STRING([--enable-mclib], [Compile shared library libmc.so @<:@no@:>@])],
|
[AS_HELP_STRING([--enable-mclib], [Compile shared library libmc.so @<:@no@:>@])],
|
||||||
@ -210,14 +211,9 @@ AC_FUNC_STRCOLL
|
|||||||
|
|
||||||
mc_AC_GET_FS_INFO
|
mc_AC_GET_FS_INFO
|
||||||
|
|
||||||
dnl
|
|
||||||
dnl X11 support.
|
dnl X11 support.
|
||||||
dnl Used to read keyboard modifiers when running under X11.
|
dnl Used to read keyboard modifiers when running under X11.
|
||||||
AC_PATH_XTRA
|
MC_WITH_X
|
||||||
|
|
||||||
dnl
|
|
||||||
dnl Check if the gmodule functionality supported on this system.
|
|
||||||
AC_G_MODULE_SUPPORTED
|
|
||||||
|
|
||||||
dnl
|
dnl
|
||||||
dnl Sequent wants getprocessstats
|
dnl Sequent wants getprocessstats
|
||||||
@ -360,24 +356,9 @@ fi
|
|||||||
AC_MSG_RESULT([$result])
|
AC_MSG_RESULT([$result])
|
||||||
subshell="$result"
|
subshell="$result"
|
||||||
|
|
||||||
|
|
||||||
MC_WITH_SCREEN
|
MC_WITH_SCREEN
|
||||||
|
|
||||||
|
MC_WITH_EDIT
|
||||||
dnl
|
|
||||||
dnl Internal editor support.
|
|
||||||
dnl
|
|
||||||
AC_ARG_WITH(edit,
|
|
||||||
[ --with-edit Enable internal editor [[yes]]])
|
|
||||||
|
|
||||||
if test x$with_edit != xno; then
|
|
||||||
AC_DEFINE(USE_INTERNAL_EDIT, 1, [Define to enable internal editor])
|
|
||||||
use_edit=yes
|
|
||||||
edit_msg="yes"
|
|
||||||
AC_MSG_NOTICE([using internal editor])
|
|
||||||
else
|
|
||||||
edit_msg="no"
|
|
||||||
fi
|
|
||||||
|
|
||||||
dnl
|
dnl
|
||||||
dnl Diff viewer support.
|
dnl Diff viewer support.
|
||||||
@ -495,7 +476,8 @@ AC_SUBST(MAN_DATE)
|
|||||||
AM_CONDITIONAL(USE_NLS, [test x"$USE_NLS" = xyes])
|
AM_CONDITIONAL(USE_NLS, [test x"$USE_NLS" = xyes])
|
||||||
AM_CONDITIONAL(USE_MAINTAINER_MODE, [test x"$USE_MAINTAINER_MODE" = xyes])
|
AM_CONDITIONAL(USE_MAINTAINER_MODE, [test x"$USE_MAINTAINER_MODE" = xyes])
|
||||||
AM_CONDITIONAL(USE_SCREEN_SLANG, [test x"$with_screen" = xslang])
|
AM_CONDITIONAL(USE_SCREEN_SLANG, [test x"$with_screen" = xslang])
|
||||||
AM_CONDITIONAL(USE_EDIT, [test -n "$use_edit"])
|
AM_CONDITIONAL(USE_EDIT, [test x"$use_edit" = xyes ])
|
||||||
|
AM_CONDITIONAL(USE_ASPELL, [test x"$enable_aspell" = xyes ])
|
||||||
AM_CONDITIONAL(USE_DIFF, [test -n "$use_diff"])
|
AM_CONDITIONAL(USE_DIFF, [test -n "$use_diff"])
|
||||||
AM_CONDITIONAL(CHARSET, [test -n "$have_charset"])
|
AM_CONDITIONAL(CHARSET, [test -n "$have_charset"])
|
||||||
AM_CONDITIONAL(CONS_SAVER, [test -n "$cons_saver"])
|
AM_CONDITIONAL(CONS_SAVER, [test -n "$cons_saver"])
|
||||||
|
@ -21,6 +21,7 @@ Build requirements for GNU Midnight Commander
|
|||||||
- gettext
|
- gettext
|
||||||
- cvs
|
- cvs
|
||||||
- libssh2 >= 1.2.5 is required only for sftp vfs (1.2.7 if you need ssh-agent support)
|
- libssh2 >= 1.2.5 is required only for sftp vfs (1.2.7 if you need ssh-agent support)
|
||||||
|
- libaspell to support spell checking in the internal editor
|
||||||
|
|
||||||
|
|
||||||
Installation instructions for GNU Midnight Commander
|
Installation instructions for GNU Midnight Commander
|
||||||
@ -89,6 +90,10 @@ incomplete, use `configure --help' to get the full list):
|
|||||||
built-in file editor. The built-in editor is compiled in by
|
built-in file editor. The built-in editor is compiled in by
|
||||||
default.
|
default.
|
||||||
|
|
||||||
|
`--enable-aspell'
|
||||||
|
This option adds spell check support in the internal editor using
|
||||||
|
libaspell. Disabled by default.
|
||||||
|
|
||||||
`--without-gpm-mouse'
|
`--without-gpm-mouse'
|
||||||
Use this flag to disable gpm mouse support (e.g. if you want to
|
Use this flag to disable gpm mouse support (e.g. if you want to
|
||||||
use mouse only on X terminals).
|
use mouse only on X terminals).
|
||||||
|
@ -6,7 +6,7 @@
|
|||||||
|
|
||||||
Written by:
|
Written by:
|
||||||
Vitja Makarov, 2005
|
Vitja Makarov, 2005
|
||||||
Ilia Maslakov <il.smind@gmail.com>, 2009
|
Ilia Maslakov <il.smind@gmail.com>, 2009, 2012
|
||||||
Andrew Borodin <aborodin@vmail.ru>, 2009, 2010, 2011, 2012
|
Andrew Borodin <aborodin@vmail.ru>, 2009, 2010, 2011, 2012
|
||||||
|
|
||||||
This file is part of the Midnight Commander.
|
This file is part of the Midnight Commander.
|
||||||
@ -281,6 +281,11 @@ static name_keymap_t command_names[] = {
|
|||||||
{"MacroStartStopRecord", CK_MacroStartStopRecord},
|
{"MacroStartStopRecord", CK_MacroStartStopRecord},
|
||||||
{"MacroDelete", CK_MacroDelete},
|
{"MacroDelete", CK_MacroDelete},
|
||||||
{"RepeatStartStopRecord", CK_RepeatStartStopRecord},
|
{"RepeatStartStopRecord", CK_RepeatStartStopRecord},
|
||||||
|
#ifdef HAVE_ASPELL
|
||||||
|
{"SpellCheck", CK_SpellCheck},
|
||||||
|
{"SpellCheckCurrentWord", CK_SpellCheckCurrentWord},
|
||||||
|
{"SpellCheckSelectLang", CK_SpellCheckSelectLang},
|
||||||
|
#endif /* HAVE_ASPELL */
|
||||||
{"BookmarkFlush", CK_BookmarkFlush},
|
{"BookmarkFlush", CK_BookmarkFlush},
|
||||||
{"BookmarkNext", CK_BookmarkNext},
|
{"BookmarkNext", CK_BookmarkNext},
|
||||||
{"BookmarkPrev", CK_BookmarkPrev},
|
{"BookmarkPrev", CK_BookmarkPrev},
|
||||||
|
@ -288,6 +288,9 @@ enum
|
|||||||
CK_WindowNext,
|
CK_WindowNext,
|
||||||
CK_WindowPrev,
|
CK_WindowPrev,
|
||||||
/* misc commands */
|
/* misc commands */
|
||||||
|
CK_SpellCheck,
|
||||||
|
CK_SpellCheckCurrentWord,
|
||||||
|
CK_SpellCheckSelectLang,
|
||||||
CK_InsertOverwrite,
|
CK_InsertOverwrite,
|
||||||
CK_ParagraphFormat,
|
CK_ParagraphFormat,
|
||||||
CK_MatchBracket,
|
CK_MatchBracket,
|
||||||
|
@ -6,41 +6,32 @@ dnl
|
|||||||
AC_DEFUN([AC_G_MODULE_SUPPORTED], [
|
AC_DEFUN([AC_G_MODULE_SUPPORTED], [
|
||||||
|
|
||||||
g_module_supported=""
|
g_module_supported=""
|
||||||
if test x"$no_x" = xyes; then
|
|
||||||
textmode_x11_support="no"
|
found_gmodule=no
|
||||||
|
PKG_CHECK_MODULES(GMODULE, [gmodule-no-export-2.0 >= 2.8], [found_gmodule=yes], [:])
|
||||||
|
if test x"$found_gmodule" = xyes; then
|
||||||
|
g_module_supported="gmodule-no-export-2.0"
|
||||||
else
|
else
|
||||||
found_gmodule=no
|
dnl try fallback to the generic gmodule
|
||||||
PKG_CHECK_MODULES(GMODULE, [gmodule-no-export-2.0 >= 2.8], [found_gmodule=yes], [:])
|
PKG_CHECK_MODULES(GMODULE, [gmodule-2.0 >= 2.8], [found_gmodule=yes], [:])
|
||||||
if test x"$found_gmodule" = xyes; then
|
if test x"$found_gmodule" = xyes; then
|
||||||
g_module_supported="gmodule-no-export-2.0"
|
g_module_supported="gmodule-2.0"
|
||||||
else
|
|
||||||
dnl try fallback to the generic gmodule
|
|
||||||
PKG_CHECK_MODULES(GMODULE, [gmodule-2.0 >= 2.8], [found_gmodule=yes], [:])
|
|
||||||
if test x"$found_gmodule" = xyes; then
|
|
||||||
g_module_supported="gmodule-2.0"
|
|
||||||
fi
|
|
||||||
fi
|
fi
|
||||||
|
|
||||||
CPPFLAGS="$CPPFLAGS $X_CFLAGS"
|
|
||||||
case x"$g_module_supported" in
|
|
||||||
xgmodule-no-export-2.0|xgmodule-2.0)
|
|
||||||
if test x`$PKG_CONFIG --variable=gmodule_supported "$g_module_supported"` = xtrue; then
|
|
||||||
AC_DEFINE([HAVE_GMODULE], [1], [Defined if gmodule functionality is supported])
|
|
||||||
else
|
|
||||||
g_module_supported=""
|
|
||||||
fi
|
|
||||||
;;
|
|
||||||
*)
|
|
||||||
MCLIBS="$X_LIBS $X_PRE_LIBS -lX11 $X_EXTRA_LIBS"
|
|
||||||
g_module_supported=""
|
|
||||||
;;
|
|
||||||
esac
|
|
||||||
|
|
||||||
AC_DEFINE([HAVE_TEXTMODE_X11_SUPPORT], [1],
|
|
||||||
[Define to enable getting events from X Window System])
|
|
||||||
textmode_x11_support="yes"
|
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
case x"$g_module_supported" in
|
||||||
|
xgmodule-no-export-2.0|xgmodule-2.0)
|
||||||
|
if test x`$PKG_CONFIG --variable=gmodule_supported "$g_module_supported"` = xtrue; then
|
||||||
|
AC_DEFINE([HAVE_GMODULE], [1], [Defined if gmodule functionality is supported])
|
||||||
|
else
|
||||||
|
g_module_supported=""
|
||||||
|
fi
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
g_module_supported=""
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
|
||||||
AM_CONDITIONAL([HAVE_GMODULE], [test x"$g_module_supported" != x])
|
AM_CONDITIONAL([HAVE_GMODULE], [test x"$g_module_supported" != x])
|
||||||
|
|
||||||
dnl
|
dnl
|
||||||
|
46
m4.include/mc-with-edit.m4
Normal file
46
m4.include/mc-with-edit.m4
Normal file
@ -0,0 +1,46 @@
|
|||||||
|
dnl
|
||||||
|
dnl Internal editor support.
|
||||||
|
dnl
|
||||||
|
AC_DEFUN([MC_WITH_EDIT], [
|
||||||
|
|
||||||
|
AC_ARG_WITH([edit], AS_HELP_STRING([--with-edit], [Enable internal editor @<:@yes@:>@]))
|
||||||
|
|
||||||
|
if test x$with_edit != xno; then
|
||||||
|
AC_DEFINE(USE_INTERNAL_EDIT, 1, [Define to enable internal editor])
|
||||||
|
use_edit=yes
|
||||||
|
AC_MSG_NOTICE([using internal editor])
|
||||||
|
else
|
||||||
|
use_edit=no
|
||||||
|
edit_msg="no"
|
||||||
|
fi
|
||||||
|
|
||||||
|
dnl ASpell support.
|
||||||
|
AC_ARG_ENABLE([aspell],
|
||||||
|
AS_HELP_STRING([--enable-aspell], [Enable aspell support for internal editor @<:@no@:>@]),
|
||||||
|
[
|
||||||
|
if test "x$enableval" = xno; then
|
||||||
|
enable_aspell=no
|
||||||
|
else
|
||||||
|
enable_aspell=yes
|
||||||
|
fi
|
||||||
|
],
|
||||||
|
[enable_aspell=no]
|
||||||
|
)
|
||||||
|
|
||||||
|
if test x$with_edit != xno -a x$enable_aspell != xno; then
|
||||||
|
AC_CHECK_HEADERS([aspell.h], [], [
|
||||||
|
AC_ERROR([Could not find aspell development headers])
|
||||||
|
], [])
|
||||||
|
|
||||||
|
if test x"$g_module_supported" != x; then
|
||||||
|
AC_DEFINE(HAVE_ASPELL, 1, [Define to enable aspell support])
|
||||||
|
edit_msg="yes with aspell support"
|
||||||
|
AC_MSG_NOTICE([using aspell for internal editor])
|
||||||
|
else
|
||||||
|
enable_aspell=no
|
||||||
|
AC_MSG_NOTICE([aspell support is disabled because gmodule support is not available])
|
||||||
|
fi
|
||||||
|
else
|
||||||
|
edit_msg="yes"
|
||||||
|
fi
|
||||||
|
])
|
22
m4.include/mc-with-x.m4
Normal file
22
m4.include/mc-with-x.m4
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
|
||||||
|
dnl X11 support.
|
||||||
|
dnl Used to read keyboard modifiers when running under X11.
|
||||||
|
AC_DEFUN([MC_WITH_X], [
|
||||||
|
|
||||||
|
AC_PATH_XTRA
|
||||||
|
|
||||||
|
if test x"$no_x" = xyes; then
|
||||||
|
textmode_x11_support="no"
|
||||||
|
else
|
||||||
|
AC_DEFINE([HAVE_TEXTMODE_X11_SUPPORT], [1],
|
||||||
|
[Define to enable getting events from X Window System])
|
||||||
|
textmode_x11_support="yes"
|
||||||
|
|
||||||
|
CPPFLAGS="$CPPFLAGS $X_CFLAGS"
|
||||||
|
|
||||||
|
if test x"$g_module_supported" = x; then
|
||||||
|
MCLIBS="$MCLIBS $X_LIBS $X_PRE_LIBS -lX11 $X_EXTRA_LIBS"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
])
|
@ -337,6 +337,9 @@ RepeatStartStopRecord =
|
|||||||
SelectCodepage = alt-e
|
SelectCodepage = alt-e
|
||||||
Options =
|
Options =
|
||||||
OptionsSaveMode =
|
OptionsSaveMode =
|
||||||
|
SpellCheck =
|
||||||
|
SpellCheckCurrentWord = ctrl-p
|
||||||
|
SpellCheckSelectLang =
|
||||||
LearnKeys =
|
LearnKeys =
|
||||||
WindowMove =
|
WindowMove =
|
||||||
WindowResize =
|
WindowResize =
|
||||||
|
@ -337,6 +337,9 @@ RepeatStartStopRecord =
|
|||||||
SelectCodepage = alt-e
|
SelectCodepage = alt-e
|
||||||
Options =
|
Options =
|
||||||
OptionsSaveMode =
|
OptionsSaveMode =
|
||||||
|
SpellCheck =
|
||||||
|
SpellCheckCurrentWord =
|
||||||
|
SpellCheckSelectLang =
|
||||||
LearnKeys =
|
LearnKeys =
|
||||||
WindowMove =
|
WindowMove =
|
||||||
WindowResize =
|
WindowResize =
|
||||||
|
@ -3,7 +3,7 @@ EXTRA_DIST =
|
|||||||
if USE_EDIT
|
if USE_EDIT
|
||||||
noinst_LTLIBRARIES = libedit.la
|
noinst_LTLIBRARIES = libedit.la
|
||||||
else
|
else
|
||||||
noinst_LTLIBRARIES =
|
noinst_LTLIBRARIES =
|
||||||
endif
|
endif
|
||||||
|
|
||||||
libedit_la_SOURCES = \
|
libedit_la_SOURCES = \
|
||||||
@ -13,5 +13,12 @@ libedit_la_SOURCES = \
|
|||||||
syntax.c wordproc.c \
|
syntax.c wordproc.c \
|
||||||
choosesyntax.c etags.c etags.h editcmd_dialogs.c editcmd_dialogs.h
|
choosesyntax.c etags.c etags.h editcmd_dialogs.c editcmd_dialogs.h
|
||||||
|
|
||||||
libedit_la_CFLAGS = $(GLIB_CFLAGS) -I$(top_srcdir) $(PCRE_CFLAGS)
|
if USE_ASPELL
|
||||||
|
if HAVE_GMODULE
|
||||||
|
libedit_la_SOURCES += \
|
||||||
|
spell.c spell.h \
|
||||||
|
spell_dialogs.c spell_dialogs.h
|
||||||
|
endif
|
||||||
|
endif
|
||||||
|
|
||||||
|
libedit_la_CFLAGS = $(GLIB_CFLAGS) -I$(top_srcdir) $(PCRE_CFLAGS)
|
||||||
|
@ -212,6 +212,7 @@ int edit_block_delete_cmd (WEdit * edit);
|
|||||||
void edit_delete_line (WEdit * edit);
|
void edit_delete_line (WEdit * edit);
|
||||||
|
|
||||||
int edit_delete (WEdit * edit, const int byte_delete);
|
int edit_delete (WEdit * edit, const int byte_delete);
|
||||||
|
int edit_backspace (WEdit * edit, const int byte_delete);
|
||||||
void edit_insert (WEdit * edit, int c);
|
void edit_insert (WEdit * edit, int c);
|
||||||
void edit_cursor_move (WEdit * edit, long increment);
|
void edit_cursor_move (WEdit * edit, long increment);
|
||||||
void edit_push_undo_action (WEdit * edit, long c, ...);
|
void edit_push_undo_action (WEdit * edit, long c, ...);
|
||||||
@ -242,12 +243,21 @@ mc_search_cbret_t edit_search_cmd_callback (const void *user_data, gsize char_of
|
|||||||
int *current_char);
|
int *current_char);
|
||||||
void edit_complete_word_cmd (WEdit * edit);
|
void edit_complete_word_cmd (WEdit * edit);
|
||||||
void edit_get_match_keyword_cmd (WEdit * edit);
|
void edit_get_match_keyword_cmd (WEdit * edit);
|
||||||
|
|
||||||
|
#ifdef HAVE_ASPELL
|
||||||
|
int edit_suggest_current_word (WEdit * edit);
|
||||||
|
void edit_spellcheck_file (WEdit * edit);
|
||||||
|
void edit_set_spell_lang (void);
|
||||||
|
#endif
|
||||||
|
|
||||||
int edit_save_block (WEdit * edit, const char *filename, long start, long finish);
|
int edit_save_block (WEdit * edit, const char *filename, long start, long finish);
|
||||||
int edit_save_block_cmd (WEdit * edit);
|
int edit_save_block_cmd (WEdit * edit);
|
||||||
gboolean edit_insert_file_cmd (WEdit * edit);
|
gboolean edit_insert_file_cmd (WEdit * edit);
|
||||||
void edit_insert_over (WEdit * edit);
|
void edit_insert_over (WEdit * edit);
|
||||||
int edit_insert_column_of_text_from_file (WEdit * edit, int file,
|
int edit_insert_column_of_text_from_file (WEdit * edit, int file,
|
||||||
long *start_pos, long *end_pos, int *col1, int *col2);
|
long *start_pos, long *end_pos, int *col1, int *col2);
|
||||||
|
|
||||||
|
char *edit_get_word_from_pos (WEdit * edit, long start_pos, long *start, gsize *len, gsize *cut);
|
||||||
long edit_insert_file (WEdit * edit, const vfs_path_t * filename_vpath);
|
long edit_insert_file (WEdit * edit, const vfs_path_t * filename_vpath);
|
||||||
gboolean edit_load_back_cmd (WEdit * edit);
|
gboolean edit_load_back_cmd (WEdit * edit);
|
||||||
gboolean edit_load_forward_cmd (WEdit * edit);
|
gboolean edit_load_forward_cmd (WEdit * edit);
|
||||||
@ -307,6 +317,7 @@ void book_mark_serialize (WEdit * edit, int color);
|
|||||||
void book_mark_restore (WEdit * edit, int color);
|
void book_mark_restore (WEdit * edit, int color);
|
||||||
|
|
||||||
int line_is_blank (WEdit * edit, long line);
|
int line_is_blank (WEdit * edit, long line);
|
||||||
|
gboolean is_break_char (char c);
|
||||||
int edit_indent_width (WEdit * edit, long p);
|
int edit_indent_width (WEdit * edit, long p);
|
||||||
void edit_insert_indent (WEdit * edit, int indent);
|
void edit_insert_indent (WEdit * edit, int indent);
|
||||||
void edit_options_dialog (Dlg_head * h);
|
void edit_options_dialog (Dlg_head * h);
|
||||||
|
@ -70,6 +70,9 @@
|
|||||||
|
|
||||||
#include "edit-impl.h"
|
#include "edit-impl.h"
|
||||||
#include "editwidget.h"
|
#include "editwidget.h"
|
||||||
|
#ifdef HAVE_ASPELL
|
||||||
|
#include "spell.h"
|
||||||
|
#endif
|
||||||
|
|
||||||
/*** global variables ****************************************************************************/
|
/*** global variables ****************************************************************************/
|
||||||
|
|
||||||
@ -717,72 +720,6 @@ edit_get_prev_utf (WEdit * edit, long byte_index, int *char_width)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* --------------------------------------------------------------------------------------------- */
|
|
||||||
|
|
||||||
static int
|
|
||||||
edit_backspace (WEdit * edit, const int byte_delete)
|
|
||||||
{
|
|
||||||
int p = 0;
|
|
||||||
int cw = 1;
|
|
||||||
int i;
|
|
||||||
|
|
||||||
if (!edit->curs1)
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
cw = 1;
|
|
||||||
|
|
||||||
if (edit->mark2 != edit->mark1)
|
|
||||||
edit_push_markers (edit);
|
|
||||||
|
|
||||||
if (edit->utf8 && byte_delete == 0)
|
|
||||||
{
|
|
||||||
edit_get_prev_utf (edit, edit->curs1, &cw);
|
|
||||||
if (cw < 1)
|
|
||||||
cw = 1;
|
|
||||||
}
|
|
||||||
for (i = 1; i <= cw; i++)
|
|
||||||
{
|
|
||||||
if (edit->mark1 >= edit->curs1)
|
|
||||||
{
|
|
||||||
edit->mark1--;
|
|
||||||
edit->end_mark_curs--;
|
|
||||||
}
|
|
||||||
if (edit->mark2 >= edit->curs1)
|
|
||||||
edit->mark2--;
|
|
||||||
if (edit->last_get_rule >= edit->curs1)
|
|
||||||
edit->last_get_rule--;
|
|
||||||
|
|
||||||
p = *(edit->buffers1[(edit->curs1 - 1) >> S_EDIT_BUF_SIZE] +
|
|
||||||
((edit->curs1 - 1) & M_EDIT_BUF_SIZE));
|
|
||||||
if (!((edit->curs1 - 1) & M_EDIT_BUF_SIZE))
|
|
||||||
{
|
|
||||||
g_free (edit->buffers1[edit->curs1 >> S_EDIT_BUF_SIZE]);
|
|
||||||
edit->buffers1[edit->curs1 >> S_EDIT_BUF_SIZE] = NULL;
|
|
||||||
}
|
|
||||||
edit->last_byte--;
|
|
||||||
edit->curs1--;
|
|
||||||
edit_push_undo_action (edit, p);
|
|
||||||
}
|
|
||||||
edit_modification (edit);
|
|
||||||
if (p == '\n')
|
|
||||||
{
|
|
||||||
if (edit->book_mark)
|
|
||||||
book_mark_dec (edit, edit->curs_line);
|
|
||||||
edit->curs_line--;
|
|
||||||
edit->total_lines--;
|
|
||||||
edit->force |= REDRAW_AFTER_CURSOR;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (edit->curs1 < edit->start_display)
|
|
||||||
{
|
|
||||||
edit->start_display--;
|
|
||||||
if (p == '\n')
|
|
||||||
edit->start_line--;
|
|
||||||
}
|
|
||||||
|
|
||||||
return p;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* --------------------------------------------------------------------------------------------- */
|
/* --------------------------------------------------------------------------------------------- */
|
||||||
/* high level cursor movement commands */
|
/* high level cursor movement commands */
|
||||||
/* --------------------------------------------------------------------------------------------- */
|
/* --------------------------------------------------------------------------------------------- */
|
||||||
@ -2053,8 +1990,55 @@ edit_write_stream (WEdit * edit, FILE * f)
|
|||||||
return edit->last_byte;
|
return edit->last_byte;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* --------------------------------------------------------------------------------------------- */
|
||||||
|
|
||||||
|
gboolean
|
||||||
|
is_break_char (char c)
|
||||||
|
{
|
||||||
|
return (isspace (c) || strchr ("{}[]()<>=|/\\!?~-+`'\",.;:#$%^&*", c));
|
||||||
|
}
|
||||||
|
|
||||||
|
/* --------------------------------------------------------------------------------------------- */
|
||||||
|
|
||||||
|
char *
|
||||||
|
edit_get_word_from_pos (WEdit *edit, long start_pos, long *start, gsize *len, gsize *cut)
|
||||||
|
{
|
||||||
|
long word_start;
|
||||||
|
long cut_len = 0;
|
||||||
|
GString *match_expr;
|
||||||
|
unsigned char *bufpos;
|
||||||
|
int c1, c2;
|
||||||
|
|
||||||
|
for (word_start = start_pos; word_start != 0; word_start--, cut_len++)
|
||||||
|
{
|
||||||
|
c1 = edit_get_byte (edit, word_start);
|
||||||
|
c2 = edit_get_byte (edit, word_start - 1);
|
||||||
|
|
||||||
|
if (is_break_char (c1) != is_break_char (c2) || c1 == '\n' || c2 == '\n')
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
bufpos = &edit->buffers1[word_start >> S_EDIT_BUF_SIZE][word_start & M_EDIT_BUF_SIZE];
|
||||||
|
match_expr = g_string_sized_new (16);
|
||||||
|
|
||||||
|
do
|
||||||
|
{
|
||||||
|
c1 = edit_get_byte (edit, word_start + match_expr->len);
|
||||||
|
c2 = edit_get_byte (edit, word_start + match_expr->len + 1);
|
||||||
|
g_string_append_c (match_expr, c1);
|
||||||
|
}
|
||||||
|
while (!(is_break_char (c1) != is_break_char (c2) || c1 == '\n' || c2 == '\n'));
|
||||||
|
|
||||||
|
*len = match_expr->len;
|
||||||
|
*start = word_start;
|
||||||
|
*cut = cut_len;
|
||||||
|
|
||||||
|
return g_string_free (match_expr, FALSE);
|
||||||
|
}
|
||||||
|
|
||||||
/* --------------------------------------------------------------------------------------------- */
|
/* --------------------------------------------------------------------------------------------- */
|
||||||
/** inserts a file at the cursor, returns count of inserted bytes on success */
|
/** inserts a file at the cursor, returns count of inserted bytes on success */
|
||||||
|
|
||||||
long
|
long
|
||||||
edit_insert_file (WEdit * edit, const vfs_path_t * filename_vpath)
|
edit_insert_file (WEdit * edit, const vfs_path_t * filename_vpath)
|
||||||
{
|
{
|
||||||
@ -2274,6 +2258,7 @@ edit_init (WEdit * edit, int y, int x, int lines, int cols, const vfs_path_t * f
|
|||||||
}
|
}
|
||||||
|
|
||||||
edit_load_macro_cmd (edit);
|
edit_load_macro_cmd (edit);
|
||||||
|
|
||||||
return edit;
|
return edit;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2783,6 +2768,72 @@ edit_delete (WEdit * edit, const int byte_delete)
|
|||||||
return p;
|
return p;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* --------------------------------------------------------------------------------------------- */
|
||||||
|
|
||||||
|
int
|
||||||
|
edit_backspace (WEdit * edit, const int byte_delete)
|
||||||
|
{
|
||||||
|
int p = 0;
|
||||||
|
int cw = 1;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
if (edit->curs1 == 0)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
cw = 1;
|
||||||
|
|
||||||
|
if (edit->mark2 != edit->mark1)
|
||||||
|
edit_push_markers (edit);
|
||||||
|
|
||||||
|
if (edit->utf8 && byte_delete == 0)
|
||||||
|
{
|
||||||
|
edit_get_prev_utf (edit, edit->curs1, &cw);
|
||||||
|
if (cw < 1)
|
||||||
|
cw = 1;
|
||||||
|
}
|
||||||
|
for (i = 1; i <= cw; i++)
|
||||||
|
{
|
||||||
|
if (edit->mark1 >= edit->curs1)
|
||||||
|
{
|
||||||
|
edit->mark1--;
|
||||||
|
edit->end_mark_curs--;
|
||||||
|
}
|
||||||
|
if (edit->mark2 >= edit->curs1)
|
||||||
|
edit->mark2--;
|
||||||
|
if (edit->last_get_rule >= edit->curs1)
|
||||||
|
edit->last_get_rule--;
|
||||||
|
|
||||||
|
p = *(edit->buffers1[(edit->curs1 - 1) >> S_EDIT_BUF_SIZE] +
|
||||||
|
((edit->curs1 - 1) & M_EDIT_BUF_SIZE));
|
||||||
|
if (!((edit->curs1 - 1) & M_EDIT_BUF_SIZE))
|
||||||
|
{
|
||||||
|
g_free (edit->buffers1[edit->curs1 >> S_EDIT_BUF_SIZE]);
|
||||||
|
edit->buffers1[edit->curs1 >> S_EDIT_BUF_SIZE] = NULL;
|
||||||
|
}
|
||||||
|
edit->last_byte--;
|
||||||
|
edit->curs1--;
|
||||||
|
edit_push_undo_action (edit, p);
|
||||||
|
}
|
||||||
|
edit_modification (edit);
|
||||||
|
if (p == '\n')
|
||||||
|
{
|
||||||
|
if (edit->book_mark)
|
||||||
|
book_mark_dec (edit, edit->curs_line);
|
||||||
|
edit->curs_line--;
|
||||||
|
edit->total_lines--;
|
||||||
|
edit->force |= REDRAW_AFTER_CURSOR;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (edit->curs1 < edit->start_display)
|
||||||
|
{
|
||||||
|
edit->start_display--;
|
||||||
|
if (p == '\n')
|
||||||
|
edit->start_line--;
|
||||||
|
}
|
||||||
|
|
||||||
|
return p;
|
||||||
|
}
|
||||||
|
|
||||||
/* --------------------------------------------------------------------------------------------- */
|
/* --------------------------------------------------------------------------------------------- */
|
||||||
/** moves the cursor right or left: increment positive or negative respectively */
|
/** moves the cursor right or left: increment positive or negative respectively */
|
||||||
|
|
||||||
@ -4075,6 +4126,17 @@ edit_execute_cmd (WEdit * edit, unsigned long command, int char_for_insertion)
|
|||||||
case CK_Find:
|
case CK_Find:
|
||||||
edit_get_match_keyword_cmd (edit);
|
edit_get_match_keyword_cmd (edit);
|
||||||
break;
|
break;
|
||||||
|
#ifdef HAVE_ASPELL
|
||||||
|
case CK_SpellCheckCurrentWord:
|
||||||
|
edit_suggest_current_word (edit);
|
||||||
|
break;
|
||||||
|
case CK_SpellCheck:
|
||||||
|
edit_spellcheck_file (edit);
|
||||||
|
break;
|
||||||
|
case CK_SpellCheckSelectLang:
|
||||||
|
edit_set_spell_lang ();
|
||||||
|
break;
|
||||||
|
#endif
|
||||||
case CK_Date:
|
case CK_Date:
|
||||||
{
|
{
|
||||||
char s[BUF_MEDIUM];
|
char s[BUF_MEDIUM];
|
||||||
|
@ -7,7 +7,7 @@
|
|||||||
* \author Paul Sheer
|
* \author Paul Sheer
|
||||||
* \date 1996, 1997
|
* \date 1996, 1997
|
||||||
* \author Andrew Borodin
|
* \author Andrew Borodin
|
||||||
* \date 2009
|
* \date 2009, 2012
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#ifndef MC__EDIT_H
|
#ifndef MC__EDIT_H
|
||||||
@ -15,6 +15,7 @@
|
|||||||
|
|
||||||
#include "lib/global.h" /* PATH_SEP_STR */
|
#include "lib/global.h" /* PATH_SEP_STR */
|
||||||
#include "lib/fileloc.h"
|
#include "lib/fileloc.h"
|
||||||
|
#include "lib/vfs/vfs.h" /* vfs_path_t */
|
||||||
|
|
||||||
/*** typedefs(not structures) and defined constants **********************************************/
|
/*** typedefs(not structures) and defined constants **********************************************/
|
||||||
|
|
||||||
|
@ -8,6 +8,7 @@
|
|||||||
Written by:
|
Written by:
|
||||||
Paul Sheer, 1996, 1997
|
Paul Sheer, 1996, 1997
|
||||||
Andrew Borodin <aborodin@vmail.ru> 2012
|
Andrew Borodin <aborodin@vmail.ru> 2012
|
||||||
|
Ilia Maslakov <il.smind@gmail.com> 2012
|
||||||
|
|
||||||
This file is part of the Midnight Commander.
|
This file is part of the Midnight Commander.
|
||||||
|
|
||||||
@ -78,6 +79,10 @@
|
|||||||
#include "edit-impl.h"
|
#include "edit-impl.h"
|
||||||
#include "editwidget.h"
|
#include "editwidget.h"
|
||||||
#include "editcmd_dialogs.h"
|
#include "editcmd_dialogs.h"
|
||||||
|
#ifdef HAVE_ASPELL
|
||||||
|
#include "spell.h"
|
||||||
|
#include "spell_dialogs.h"
|
||||||
|
#endif
|
||||||
#include "etags.h"
|
#include "etags.h"
|
||||||
|
|
||||||
/*** global variables ****************************************************************************/
|
/*** global variables ****************************************************************************/
|
||||||
@ -1082,14 +1087,6 @@ pipe_mail (WEdit * edit, char *to, char *subject, char *cc)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* --------------------------------------------------------------------------------------------- */
|
|
||||||
|
|
||||||
static gboolean
|
|
||||||
is_break_char (char c)
|
|
||||||
{
|
|
||||||
return (isspace (c) || strchr ("{}[]()<>=|/\\!?~-+`'\",.;:#$%^&*", c));
|
|
||||||
}
|
|
||||||
|
|
||||||
/* --------------------------------------------------------------------------------------------- */
|
/* --------------------------------------------------------------------------------------------- */
|
||||||
/** find first character of current word */
|
/** find first character of current word */
|
||||||
|
|
||||||
@ -3575,3 +3572,136 @@ edit_get_match_keyword_cmd (WEdit * edit)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* --------------------------------------------------------------------------------------------- */
|
/* --------------------------------------------------------------------------------------------- */
|
||||||
|
|
||||||
|
#ifdef HAVE_ASPELL
|
||||||
|
int
|
||||||
|
edit_suggest_current_word (WEdit * edit)
|
||||||
|
{
|
||||||
|
gsize cut_len = 0;
|
||||||
|
gsize word_len = 0;
|
||||||
|
long word_start = 0;
|
||||||
|
int retval = B_SKIP_WORD;
|
||||||
|
char *match_word;
|
||||||
|
|
||||||
|
/* search start of word to spell check */
|
||||||
|
match_word = edit_get_word_from_pos (edit, edit->curs1, &word_start, &word_len, &cut_len);
|
||||||
|
|
||||||
|
#ifdef HAVE_CHARSET
|
||||||
|
if (mc_global.source_codepage >= 0 && (mc_global.source_codepage != mc_global.display_codepage))
|
||||||
|
{
|
||||||
|
GString *tmp_word;
|
||||||
|
|
||||||
|
tmp_word = str_convert_to_display (match_word);
|
||||||
|
g_free (match_word);
|
||||||
|
match_word = g_string_free (tmp_word, FALSE);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
if (!aspell_check (match_word, (int) word_len))
|
||||||
|
{
|
||||||
|
GArray *suggest;
|
||||||
|
unsigned int res;
|
||||||
|
|
||||||
|
suggest = g_array_new (TRUE, FALSE, sizeof (char *));
|
||||||
|
|
||||||
|
res = aspell_suggest (suggest, match_word, (int) word_len);
|
||||||
|
if (res != 0)
|
||||||
|
{
|
||||||
|
char *new_word = NULL;
|
||||||
|
|
||||||
|
edit->found_start = word_start;
|
||||||
|
edit->found_len = word_len;
|
||||||
|
edit->force |= REDRAW_PAGE;
|
||||||
|
edit_scroll_screen_over_cursor (edit);
|
||||||
|
edit_render_keypress (edit);
|
||||||
|
|
||||||
|
retval = spell_dialog_spell_suggest_show (edit, match_word, &new_word, suggest);
|
||||||
|
edit_cursor_move (edit, word_len - cut_len);
|
||||||
|
|
||||||
|
if (retval == B_ENTER && new_word != NULL)
|
||||||
|
{
|
||||||
|
guint i;
|
||||||
|
char *cp_word;
|
||||||
|
|
||||||
|
#ifdef HAVE_CHARSET
|
||||||
|
if (mc_global.source_codepage >= 0 &&
|
||||||
|
(mc_global.source_codepage != mc_global.display_codepage))
|
||||||
|
{
|
||||||
|
GString *tmp_word;
|
||||||
|
|
||||||
|
tmp_word = str_convert_to_input (new_word);
|
||||||
|
g_free (new_word);
|
||||||
|
new_word = g_string_free (tmp_word, FALSE);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
cp_word = new_word;
|
||||||
|
for (i = 0; i < word_len; i++)
|
||||||
|
edit_backspace (edit, 1);
|
||||||
|
for (; *new_word; new_word++)
|
||||||
|
edit_insert (edit, *new_word);
|
||||||
|
g_free (cp_word);
|
||||||
|
}
|
||||||
|
else if (retval == B_ADD_WORD && match_word != NULL)
|
||||||
|
aspell_add_to_dict (match_word, (int) word_len);
|
||||||
|
}
|
||||||
|
|
||||||
|
g_array_free (suggest, TRUE);
|
||||||
|
edit->found_start = 0;
|
||||||
|
edit->found_len = 0;
|
||||||
|
}
|
||||||
|
g_free (match_word);
|
||||||
|
return retval;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* --------------------------------------------------------------------------------------------- */
|
||||||
|
|
||||||
|
void
|
||||||
|
edit_spellcheck_file (WEdit * edit)
|
||||||
|
{
|
||||||
|
if (edit->curs_line > 0)
|
||||||
|
{
|
||||||
|
edit_cursor_move (edit, -edit->curs1);
|
||||||
|
edit_move_to_prev_col (edit, 0);
|
||||||
|
edit_update_curs_row (edit);
|
||||||
|
}
|
||||||
|
|
||||||
|
do
|
||||||
|
{
|
||||||
|
int c1, c2;
|
||||||
|
|
||||||
|
c2 = edit_get_byte (edit, edit->curs1);
|
||||||
|
|
||||||
|
do
|
||||||
|
{
|
||||||
|
if (edit->curs1 >= edit->last_byte)
|
||||||
|
return;
|
||||||
|
|
||||||
|
c1 = c2;
|
||||||
|
edit_cursor_move (edit, 1);
|
||||||
|
c2 = edit_get_byte (edit, edit->curs1);
|
||||||
|
}
|
||||||
|
while (is_break_char (c1) || is_break_char (c2));
|
||||||
|
}
|
||||||
|
while (edit_suggest_current_word (edit) != B_CANCEL);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* --------------------------------------------------------------------------------------------- */
|
||||||
|
|
||||||
|
void
|
||||||
|
edit_set_spell_lang (void)
|
||||||
|
{
|
||||||
|
GArray *lang_list;
|
||||||
|
|
||||||
|
lang_list = g_array_new (TRUE, FALSE, sizeof (char *));
|
||||||
|
if (aspell_get_lang_list (lang_list) != 0)
|
||||||
|
{
|
||||||
|
char *lang;
|
||||||
|
|
||||||
|
lang = spell_dialog_lang_list_show (lang_list);
|
||||||
|
if (lang != NULL)
|
||||||
|
(void) aspell_set_lang (lang);
|
||||||
|
}
|
||||||
|
aspell_array_clean (lang_list);
|
||||||
|
}
|
||||||
|
#endif /* HAVE_ASPELL */
|
||||||
|
|
||||||
|
/* --------------------------------------------------------------------------------------------- */
|
||||||
|
@ -43,6 +43,10 @@
|
|||||||
#include "src/editor/etags.h"
|
#include "src/editor/etags.h"
|
||||||
#include "src/editor/editcmd_dialogs.h"
|
#include "src/editor/editcmd_dialogs.h"
|
||||||
|
|
||||||
|
#ifdef HAVE_ASPELL
|
||||||
|
#include "src/editor/spell.h"
|
||||||
|
#endif
|
||||||
|
|
||||||
/*** global variables ****************************************************************************/
|
/*** global variables ****************************************************************************/
|
||||||
|
|
||||||
edit_search_options_t edit_search_options = {
|
edit_search_options_t edit_search_options = {
|
||||||
|
@ -175,10 +175,18 @@ create_command_menu (void)
|
|||||||
g_list_prepend (entries,
|
g_list_prepend (entries,
|
||||||
menu_entry_create (_("Record/Repeat &actions"), CK_RepeatStartStopRecord));
|
menu_entry_create (_("Record/Repeat &actions"), CK_RepeatStartStopRecord));
|
||||||
entries = g_list_prepend (entries, menu_separator_create ());
|
entries = g_list_prepend (entries, menu_separator_create ());
|
||||||
|
#ifdef HAVE_ASPELL
|
||||||
entries =
|
entries =
|
||||||
g_list_prepend (entries, menu_entry_create (_("'ispell' s&pell check"), CK_PipeBlock (1)));
|
g_list_prepend (entries, menu_entry_create (_("S&pell check"), CK_SpellCheck));
|
||||||
|
entries =
|
||||||
|
g_list_prepend (entries, menu_entry_create (_("C&heck word"), CK_SpellCheckCurrentWord));
|
||||||
|
entries =
|
||||||
|
g_list_prepend (entries, menu_entry_create (_("Change spelling &language"), CK_SpellCheckSelectLang));
|
||||||
|
entries = g_list_prepend (entries, menu_separator_create ());
|
||||||
|
#endif /* HAVE_ASPELL */
|
||||||
entries = g_list_prepend (entries, menu_entry_create (_("&Mail..."), CK_Mail));
|
entries = g_list_prepend (entries, menu_entry_create (_("&Mail..."), CK_Mail));
|
||||||
|
|
||||||
|
|
||||||
return g_list_reverse (entries);
|
return g_list_reverse (entries);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -63,6 +63,9 @@
|
|||||||
|
|
||||||
#include "edit-impl.h"
|
#include "edit-impl.h"
|
||||||
#include "editwidget.h"
|
#include "editwidget.h"
|
||||||
|
#ifdef HAVE_ASPELL
|
||||||
|
#include "spell.h"
|
||||||
|
#endif
|
||||||
|
|
||||||
/*** global variables ****************************************************************************/
|
/*** global variables ****************************************************************************/
|
||||||
|
|
||||||
@ -97,6 +100,10 @@ edit_dlg_init (void)
|
|||||||
{
|
{
|
||||||
edit_window_state_char = mc_skin_get ("editor", "window-state-char", "*");
|
edit_window_state_char = mc_skin_get ("editor", "window-state-char", "*");
|
||||||
edit_window_close_char = mc_skin_get ("editor", "window-close-char", "X");
|
edit_window_close_char = mc_skin_get ("editor", "window-close-char", "X");
|
||||||
|
|
||||||
|
#ifdef HAVE_ASPELL
|
||||||
|
aspell_init ();
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
edit_dlg_init_refcounter++;
|
edit_dlg_init_refcounter++;
|
||||||
@ -117,6 +124,10 @@ edit_dlg_deinit (void)
|
|||||||
{
|
{
|
||||||
g_free (edit_window_state_char);
|
g_free (edit_window_state_char);
|
||||||
g_free (edit_window_close_char);
|
g_free (edit_window_close_char);
|
||||||
|
|
||||||
|
#ifdef HAVE_ASPELL
|
||||||
|
aspell_clean ();
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
558
src/editor/spell.c
Normal file
558
src/editor/spell.c
Normal file
@ -0,0 +1,558 @@
|
|||||||
|
/*
|
||||||
|
Editor spell checker
|
||||||
|
|
||||||
|
Copyright (C) 2012
|
||||||
|
The Free Software Foundation, Inc.
|
||||||
|
|
||||||
|
Written by:
|
||||||
|
Ilia Maslakov <il.smind@gmail.com>, 2012
|
||||||
|
|
||||||
|
This file is part of the Midnight Commander.
|
||||||
|
|
||||||
|
The Midnight Commander is free software: you can redistribute it
|
||||||
|
and/or modify it under the terms of the GNU General Public License as
|
||||||
|
published by the Free Software Foundation, either version 3 of the License,
|
||||||
|
or (at your option) any later version.
|
||||||
|
|
||||||
|
The Midnight Commander is distributed in the hope that it will be useful,
|
||||||
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
GNU General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU General Public License
|
||||||
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <config.h>
|
||||||
|
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <gmodule.h>
|
||||||
|
#include <aspell.h>
|
||||||
|
|
||||||
|
#ifdef HAVE_CHARSET
|
||||||
|
#include "lib/charsets.h"
|
||||||
|
#endif
|
||||||
|
#include "lib/strutil.h"
|
||||||
|
|
||||||
|
#include "edit-impl.h"
|
||||||
|
#include "spell.h"
|
||||||
|
|
||||||
|
/*** global variables ****************************************************************************/
|
||||||
|
|
||||||
|
/*** file scope macro definitions ****************************************************************/
|
||||||
|
|
||||||
|
/*** file scope type declarations ****************************************************************/
|
||||||
|
|
||||||
|
typedef struct aspell_struct
|
||||||
|
{
|
||||||
|
AspellConfig *config;
|
||||||
|
AspellSpeller *speller;
|
||||||
|
} spell_t;
|
||||||
|
|
||||||
|
/*** file scope variables ************************************************************************/
|
||||||
|
|
||||||
|
static GModule *spell_module = NULL;
|
||||||
|
static spell_t *global_speller = NULL;
|
||||||
|
|
||||||
|
static struct AspellConfig *(*mc_new_aspell_config) (void);
|
||||||
|
static int (*mc_aspell_config_replace) (struct AspellConfig * ths, const char *key,
|
||||||
|
const char *value);
|
||||||
|
static struct AspellCanHaveError *(*mc_new_aspell_speller) (struct AspellConfig * config);
|
||||||
|
static unsigned int (*mc_aspell_error_number) (const struct AspellCanHaveError * ths);
|
||||||
|
static const char *(*mc_aspell_speller_error_message) (const struct AspellSpeller * ths);
|
||||||
|
const struct AspellError *(*mc_aspell_speller_error) (const struct AspellSpeller * ths);
|
||||||
|
|
||||||
|
static struct AspellSpeller *(*mc_to_aspell_speller) (struct AspellCanHaveError * obj);
|
||||||
|
static int (*mc_aspell_speller_check) (struct AspellSpeller * ths, const char *word, int word_size);
|
||||||
|
static const struct AspellWordList *(*mc_aspell_speller_suggest) (struct AspellSpeller * ths,
|
||||||
|
const char *word, int word_size);
|
||||||
|
static struct AspellStringEnumeration *(*mc_aspell_word_list_elements) (const struct AspellWordList
|
||||||
|
* ths);
|
||||||
|
static const char *(*mc_aspell_config_retrieve) (struct AspellConfig * ths, const char *key);
|
||||||
|
static void (*mc_delete_aspell_speller) (struct AspellSpeller * ths);
|
||||||
|
/*
|
||||||
|
static void (*mc_delete_aspell_config) (struct AspellConfig * ths);
|
||||||
|
*/
|
||||||
|
static void (*mc_delete_aspell_can_have_error) (struct AspellCanHaveError * ths);
|
||||||
|
static const char *(*mc_aspell_error_message) (const struct AspellCanHaveError * ths);
|
||||||
|
static void (*mc_delete_aspell_string_enumeration) (struct AspellStringEnumeration * ths);
|
||||||
|
static struct AspellDictInfoEnumeration *(*mc_aspell_dict_info_list_elements)
|
||||||
|
(const struct AspellDictInfoList * ths);
|
||||||
|
static struct AspellDictInfoList *(*mc_get_aspell_dict_info_list) (struct AspellConfig * config);
|
||||||
|
static const struct AspellDictInfo *(*mc_aspell_dict_info_enumeration_next)
|
||||||
|
(struct AspellDictInfoEnumeration * ths);
|
||||||
|
static const char *(*mc_aspell_string_enumeration_next) (struct AspellStringEnumeration * ths);
|
||||||
|
static void (*mc_delete_aspell_dict_info_enumeration) (struct AspellDictInfoEnumeration * ths);
|
||||||
|
static unsigned int (*mc_aspell_word_list_size) (const struct AspellWordList * ths);
|
||||||
|
static const struct AspellError *(*mc_aspell_error) (const struct AspellCanHaveError * ths);
|
||||||
|
static int (*mc_aspell_speller_add_to_personal) (struct AspellSpeller * ths, const char * word,
|
||||||
|
int word_size);
|
||||||
|
static int (*mc_aspell_speller_save_all_word_lists) (struct AspellSpeller * ths);
|
||||||
|
|
||||||
|
static struct
|
||||||
|
{
|
||||||
|
const char *code;
|
||||||
|
const char *name;
|
||||||
|
} spell_codes_map[] =
|
||||||
|
{
|
||||||
|
/* *INDENT-OFF* */
|
||||||
|
{"br", "Breton"},
|
||||||
|
{"cs", "Czech"},
|
||||||
|
{"cy", "Welsh"},
|
||||||
|
{"da", "Danish"},
|
||||||
|
{"de", "German"},
|
||||||
|
{"el", "Greek"},
|
||||||
|
{"en", "English"},
|
||||||
|
{"en_GB", "British English"},
|
||||||
|
{"en_CA", "Canadian English"},
|
||||||
|
{"en_US", "American English"},
|
||||||
|
{"eo", "Esperanto"},
|
||||||
|
{"es", "Spanish"},
|
||||||
|
{"fo", "Faroese"},
|
||||||
|
{"fr", "French"},
|
||||||
|
{"it", "Italian"},
|
||||||
|
{"nl", "Dutch"},
|
||||||
|
{"no", "Norwegian"},
|
||||||
|
{"pl", "Polish"},
|
||||||
|
{"pt", "Portuguese"},
|
||||||
|
{"ro", "Romanian"},
|
||||||
|
{"ru", "Russian"},
|
||||||
|
{"sk", "Slovak"},
|
||||||
|
{"sv", "Swedish"},
|
||||||
|
{"uk", "Ukrainian"},
|
||||||
|
{NULL, NULL}
|
||||||
|
/* *INDENT-ON* */
|
||||||
|
};
|
||||||
|
|
||||||
|
/*** file scope functions ************************************************************************/
|
||||||
|
/* --------------------------------------------------------------------------------------------- */
|
||||||
|
/**
|
||||||
|
* Found the language name by language code. For example: en_US -> American English.
|
||||||
|
*
|
||||||
|
* @param code Short name of the language (ru, en, pl, uk, etc...)
|
||||||
|
* @returns the language name
|
||||||
|
*/
|
||||||
|
|
||||||
|
static const char *
|
||||||
|
spell_decode_lang (const char *code)
|
||||||
|
{
|
||||||
|
size_t i;
|
||||||
|
|
||||||
|
for (i = 0; spell_codes_map[i].code != NULL; i++)
|
||||||
|
{
|
||||||
|
if (strcmp (spell_codes_map[i].code, code) == 0)
|
||||||
|
return spell_codes_map[i].name;
|
||||||
|
}
|
||||||
|
|
||||||
|
return code;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* --------------------------------------------------------------------------------------------- */
|
||||||
|
/**
|
||||||
|
* Checks if aspell library and symbols are available.
|
||||||
|
*
|
||||||
|
* @returns FALSE or error
|
||||||
|
*/
|
||||||
|
|
||||||
|
static gboolean
|
||||||
|
spell_available (void)
|
||||||
|
{
|
||||||
|
gchar *spell_module_fname;
|
||||||
|
gboolean ret = FALSE;
|
||||||
|
|
||||||
|
if (spell_module != NULL)
|
||||||
|
return TRUE;
|
||||||
|
|
||||||
|
spell_module_fname = g_module_build_path (NULL, "libaspell");
|
||||||
|
spell_module = g_module_open (spell_module_fname, G_MODULE_BIND_LAZY);
|
||||||
|
|
||||||
|
g_free (spell_module_fname);
|
||||||
|
|
||||||
|
if (spell_module == NULL)
|
||||||
|
return FALSE;
|
||||||
|
|
||||||
|
if (!g_module_symbol (spell_module, "new_aspell_config", (void *) &mc_new_aspell_config))
|
||||||
|
goto error_ret;
|
||||||
|
|
||||||
|
if (!g_module_symbol (spell_module, "aspell_dict_info_list_elements",
|
||||||
|
(void *) &mc_aspell_dict_info_list_elements))
|
||||||
|
goto error_ret;
|
||||||
|
|
||||||
|
if (!g_module_symbol (spell_module, "aspell_dict_info_enumeration_next",
|
||||||
|
(void *) &mc_aspell_dict_info_enumeration_next))
|
||||||
|
goto error_ret;
|
||||||
|
|
||||||
|
if (!g_module_symbol (spell_module, "new_aspell_speller", (void *) &mc_new_aspell_speller))
|
||||||
|
goto error_ret;
|
||||||
|
|
||||||
|
if (!g_module_symbol (spell_module, "aspell_error_number", (void *) &mc_aspell_error_number))
|
||||||
|
goto error_ret;
|
||||||
|
|
||||||
|
if (!g_module_symbol (spell_module, "aspell_speller_error_message",
|
||||||
|
(void *) &mc_aspell_speller_error_message))
|
||||||
|
goto error_ret;
|
||||||
|
|
||||||
|
if (!g_module_symbol (spell_module, "aspell_speller_error",
|
||||||
|
(void *) &mc_aspell_speller_error))
|
||||||
|
goto error_ret;
|
||||||
|
|
||||||
|
if (!g_module_symbol (spell_module, "aspell_error", (void *) &mc_aspell_error))
|
||||||
|
goto error_ret;
|
||||||
|
|
||||||
|
if (!g_module_symbol (spell_module, "to_aspell_speller", (void *) &mc_to_aspell_speller))
|
||||||
|
goto error_ret;
|
||||||
|
|
||||||
|
if (!g_module_symbol (spell_module, "aspell_speller_check", (void *) &mc_aspell_speller_check))
|
||||||
|
goto error_ret;
|
||||||
|
|
||||||
|
if (!g_module_symbol
|
||||||
|
(spell_module, "aspell_speller_suggest", (void *) &mc_aspell_speller_suggest))
|
||||||
|
goto error_ret;
|
||||||
|
|
||||||
|
if (!g_module_symbol
|
||||||
|
(spell_module, "aspell_word_list_elements", (void *) &mc_aspell_word_list_elements))
|
||||||
|
goto error_ret;
|
||||||
|
|
||||||
|
if (!g_module_symbol (spell_module, "aspell_string_enumeration_next",
|
||||||
|
(void *) &mc_aspell_string_enumeration_next))
|
||||||
|
goto error_ret;
|
||||||
|
|
||||||
|
if (!g_module_symbol
|
||||||
|
(spell_module, "aspell_config_replace", (void *) &mc_aspell_config_replace))
|
||||||
|
goto error_ret;
|
||||||
|
|
||||||
|
if (!g_module_symbol (spell_module, "aspell_error_message", (void *) &mc_aspell_error_message))
|
||||||
|
goto error_ret;
|
||||||
|
|
||||||
|
if (!g_module_symbol
|
||||||
|
(spell_module, "delete_aspell_speller", (void *) &mc_delete_aspell_speller))
|
||||||
|
goto error_ret;
|
||||||
|
|
||||||
|
if (!g_module_symbol (spell_module, "delete_aspell_config", (void *) &mc_delete_aspell_speller))
|
||||||
|
goto error_ret;
|
||||||
|
|
||||||
|
if (!g_module_symbol (spell_module, "delete_aspell_string_enumeration",
|
||||||
|
(void *) &mc_delete_aspell_string_enumeration))
|
||||||
|
goto error_ret;
|
||||||
|
|
||||||
|
if (!g_module_symbol (spell_module, "get_aspell_dict_info_list",
|
||||||
|
(void *) &mc_get_aspell_dict_info_list))
|
||||||
|
goto error_ret;
|
||||||
|
|
||||||
|
if (!g_module_symbol (spell_module, "delete_aspell_can_have_error",
|
||||||
|
(void *) &mc_delete_aspell_can_have_error))
|
||||||
|
goto error_ret;
|
||||||
|
|
||||||
|
if (!g_module_symbol (spell_module, "delete_aspell_dict_info_enumeration",
|
||||||
|
(void *) &mc_delete_aspell_dict_info_enumeration))
|
||||||
|
goto error_ret;
|
||||||
|
|
||||||
|
if (!g_module_symbol
|
||||||
|
(spell_module, "aspell_config_retrieve", (void *) &mc_aspell_config_retrieve))
|
||||||
|
goto error_ret;
|
||||||
|
|
||||||
|
if (!g_module_symbol
|
||||||
|
(spell_module, "aspell_word_list_size", (void *) &mc_aspell_word_list_size))
|
||||||
|
goto error_ret;
|
||||||
|
|
||||||
|
if (!g_module_symbol (spell_module, "aspell_speller_add_to_personal",
|
||||||
|
(void *) &mc_aspell_speller_add_to_personal))
|
||||||
|
goto error_ret;
|
||||||
|
|
||||||
|
if (!g_module_symbol (spell_module, "aspell_speller_save_all_word_lists",
|
||||||
|
(void *) &mc_aspell_speller_save_all_word_lists))
|
||||||
|
goto error_ret;
|
||||||
|
|
||||||
|
ret = TRUE;
|
||||||
|
|
||||||
|
error_ret:
|
||||||
|
if (!ret)
|
||||||
|
{
|
||||||
|
g_module_close (spell_module);
|
||||||
|
spell_module = NULL;
|
||||||
|
}
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* --------------------------------------------------------------------------------------------- */
|
||||||
|
/*** public functions ****************************************************************************/
|
||||||
|
/* --------------------------------------------------------------------------------------------- */
|
||||||
|
/**
|
||||||
|
* Initialization of Aspell support.
|
||||||
|
*/
|
||||||
|
|
||||||
|
void
|
||||||
|
aspell_init (void)
|
||||||
|
{
|
||||||
|
AspellCanHaveError *error = NULL;
|
||||||
|
|
||||||
|
if (global_speller != NULL)
|
||||||
|
return;
|
||||||
|
|
||||||
|
global_speller = g_try_malloc (sizeof (spell_t));
|
||||||
|
if (global_speller == NULL)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (!spell_available ())
|
||||||
|
{
|
||||||
|
g_free (global_speller);
|
||||||
|
global_speller = NULL;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
global_speller->config = mc_new_aspell_config ();
|
||||||
|
global_speller->speller = NULL;
|
||||||
|
|
||||||
|
error = mc_new_aspell_speller (global_speller->config);
|
||||||
|
|
||||||
|
if (mc_aspell_error_number (error) == 0)
|
||||||
|
global_speller->speller = mc_to_aspell_speller (error);
|
||||||
|
else
|
||||||
|
{
|
||||||
|
edit_error_dialog (_("Error"), mc_aspell_error_message (error));
|
||||||
|
mc_delete_aspell_can_have_error (error);
|
||||||
|
g_free (global_speller);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* --------------------------------------------------------------------------------------------- */
|
||||||
|
/**
|
||||||
|
* Deinitialization of Aspell support.
|
||||||
|
*/
|
||||||
|
|
||||||
|
void
|
||||||
|
aspell_clean (void)
|
||||||
|
{
|
||||||
|
if (global_speller == NULL)
|
||||||
|
return;
|
||||||
|
|
||||||
|
mc_delete_aspell_speller (global_speller->speller);
|
||||||
|
g_free (global_speller);
|
||||||
|
global_speller = NULL;
|
||||||
|
|
||||||
|
g_module_close (spell_module);
|
||||||
|
spell_module = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* --------------------------------------------------------------------------------------------- */
|
||||||
|
/**
|
||||||
|
* Get array of available languages.
|
||||||
|
*
|
||||||
|
* @param lang_list Array of languages. Must be cleared before use
|
||||||
|
* @returns language list length
|
||||||
|
*/
|
||||||
|
|
||||||
|
unsigned int
|
||||||
|
aspell_get_lang_list (GArray * lang_list)
|
||||||
|
{
|
||||||
|
AspellDictInfoList *dlist;
|
||||||
|
AspellDictInfoEnumeration *elem;
|
||||||
|
const AspellDictInfo *entry;
|
||||||
|
unsigned int i = 0;
|
||||||
|
|
||||||
|
if (spell_module == NULL)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
/* the returned pointer should _not_ need to be deleted */
|
||||||
|
dlist = mc_get_aspell_dict_info_list (global_speller->config);
|
||||||
|
elem = mc_aspell_dict_info_list_elements (dlist);
|
||||||
|
|
||||||
|
while ((entry = mc_aspell_dict_info_enumeration_next (elem)) != NULL)
|
||||||
|
{
|
||||||
|
if (entry->name != NULL)
|
||||||
|
{
|
||||||
|
char *tmp;
|
||||||
|
|
||||||
|
tmp = g_strdup (entry->name);
|
||||||
|
g_array_append_val (lang_list, tmp);
|
||||||
|
i++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
mc_delete_aspell_dict_info_enumeration (elem);
|
||||||
|
|
||||||
|
return i;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* --------------------------------------------------------------------------------------------- */
|
||||||
|
/**
|
||||||
|
* Clear the array of languages.
|
||||||
|
*
|
||||||
|
* @param array Array of languages
|
||||||
|
*/
|
||||||
|
|
||||||
|
void
|
||||||
|
aspell_array_clean (GArray * array)
|
||||||
|
{
|
||||||
|
if (array != NULL)
|
||||||
|
{
|
||||||
|
guint i = 0;
|
||||||
|
|
||||||
|
for (i = 0; i < array->len; ++i)
|
||||||
|
{
|
||||||
|
char *tmp;
|
||||||
|
|
||||||
|
tmp = g_array_index (array, char *, i);
|
||||||
|
g_free (tmp);
|
||||||
|
}
|
||||||
|
g_array_free (array, TRUE);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* --------------------------------------------------------------------------------------------- */
|
||||||
|
/**
|
||||||
|
* Get the current language name.
|
||||||
|
*
|
||||||
|
* @returns Language name
|
||||||
|
*/
|
||||||
|
|
||||||
|
const char *
|
||||||
|
aspell_get_lang (void)
|
||||||
|
{
|
||||||
|
const char *code;
|
||||||
|
|
||||||
|
code = mc_aspell_config_retrieve (global_speller->config, "lang");
|
||||||
|
return spell_decode_lang (code);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* --------------------------------------------------------------------------------------------- */
|
||||||
|
/**
|
||||||
|
* Set the language.
|
||||||
|
*
|
||||||
|
* @param Language name
|
||||||
|
* @returns FALSE or error
|
||||||
|
*/
|
||||||
|
|
||||||
|
gboolean
|
||||||
|
aspell_set_lang (const char *lang)
|
||||||
|
{
|
||||||
|
if (lang != NULL)
|
||||||
|
{
|
||||||
|
int res;
|
||||||
|
AspellCanHaveError *error;
|
||||||
|
const char *spell_codeset;
|
||||||
|
|
||||||
|
#ifdef HAVE_CHARSET
|
||||||
|
if (mc_global.source_codepage > 0)
|
||||||
|
spell_codeset = get_codepage_id (mc_global.source_codepage);
|
||||||
|
else
|
||||||
|
#endif
|
||||||
|
spell_codeset = str_detect_termencoding ();
|
||||||
|
|
||||||
|
res = mc_aspell_config_replace (global_speller->config, "lang", lang);
|
||||||
|
res = mc_aspell_config_replace (global_speller->config, "encoding", spell_codeset);
|
||||||
|
|
||||||
|
/* the returned pointer should _not_ need to be deleted */
|
||||||
|
if (global_speller->speller != NULL)
|
||||||
|
mc_delete_aspell_speller (global_speller->speller);
|
||||||
|
|
||||||
|
global_speller->speller = NULL;
|
||||||
|
|
||||||
|
error = mc_new_aspell_speller (global_speller->config);
|
||||||
|
if (mc_aspell_error (error) != 0)
|
||||||
|
{
|
||||||
|
mc_delete_aspell_can_have_error (error);
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
global_speller->speller = mc_to_aspell_speller (error);
|
||||||
|
}
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* --------------------------------------------------------------------------------------------- */
|
||||||
|
/**
|
||||||
|
* Check word.
|
||||||
|
*
|
||||||
|
* @param word Word for spell check
|
||||||
|
* @param word_size Word size (in bytes)
|
||||||
|
* @returns FALSE if word is not in the dictionary
|
||||||
|
*/
|
||||||
|
|
||||||
|
gboolean
|
||||||
|
aspell_check (const char *word, const int word_size)
|
||||||
|
{
|
||||||
|
int res = 0;
|
||||||
|
|
||||||
|
if (word != NULL && global_speller != NULL && global_speller->speller != NULL)
|
||||||
|
res = mc_aspell_speller_check (global_speller->speller, word, word_size);
|
||||||
|
|
||||||
|
return (res == 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* --------------------------------------------------------------------------------------------- */
|
||||||
|
/**
|
||||||
|
* Examine dictionaries and suggest possible words that may repalce the incorrect word.
|
||||||
|
*
|
||||||
|
* @param suggest array of words to iterate through
|
||||||
|
* @param word Word for spell check
|
||||||
|
* @param word_size Word size (in bytes)
|
||||||
|
* @returns count of suggests for the word
|
||||||
|
*/
|
||||||
|
|
||||||
|
unsigned int
|
||||||
|
aspell_suggest (GArray * suggest, const char *word, const int word_size)
|
||||||
|
{
|
||||||
|
unsigned int size = 0;
|
||||||
|
|
||||||
|
if (word != NULL && global_speller != NULL && global_speller->speller != NULL)
|
||||||
|
{
|
||||||
|
const AspellWordList *wordlist;
|
||||||
|
|
||||||
|
wordlist = mc_aspell_speller_suggest (global_speller->speller, word, word_size);
|
||||||
|
if (wordlist != NULL)
|
||||||
|
{
|
||||||
|
AspellStringEnumeration *elements = NULL;
|
||||||
|
unsigned int i;
|
||||||
|
|
||||||
|
elements = mc_aspell_word_list_elements (wordlist);
|
||||||
|
size = mc_aspell_word_list_size (wordlist);
|
||||||
|
|
||||||
|
for (i = 0; i < size; i++)
|
||||||
|
{
|
||||||
|
const char *cur_sugg_word;
|
||||||
|
|
||||||
|
cur_sugg_word = g_strdup (mc_aspell_string_enumeration_next (elements));
|
||||||
|
if (cur_sugg_word != NULL)
|
||||||
|
g_array_append_val (suggest, cur_sugg_word);
|
||||||
|
}
|
||||||
|
|
||||||
|
mc_delete_aspell_string_enumeration (elements);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return size;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* --------------------------------------------------------------------------------------------- */
|
||||||
|
/*
|
||||||
|
* Add word to personal dictionary.
|
||||||
|
*
|
||||||
|
* @param word Word for spell check
|
||||||
|
* @param word_size Word size (in bytes)
|
||||||
|
* @returns FALSE or error
|
||||||
|
*/
|
||||||
|
gboolean
|
||||||
|
aspell_add_to_dict (const char *word, int word_size)
|
||||||
|
{
|
||||||
|
mc_aspell_speller_add_to_personal (global_speller->speller, word, word_size);
|
||||||
|
|
||||||
|
if (mc_aspell_speller_error (global_speller->speller) != 0)
|
||||||
|
{
|
||||||
|
edit_error_dialog (_("Error"), mc_aspell_speller_error_message (global_speller->speller));
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
mc_aspell_speller_save_all_word_lists (global_speller->speller);
|
||||||
|
|
||||||
|
if (mc_aspell_speller_error (global_speller->speller) != 0)
|
||||||
|
{
|
||||||
|
edit_error_dialog (_("Error"), mc_aspell_speller_error_message (global_speller->speller));
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* --------------------------------------------------------------------------------------------- */
|
28
src/editor/spell.h
Normal file
28
src/editor/spell.h
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
#ifndef MC__EDIT_ASPELL_H
|
||||||
|
#define MC__EDIT_ASPELL_H
|
||||||
|
|
||||||
|
#include "lib/global.h" /* include <glib.h> */
|
||||||
|
|
||||||
|
/*** typedefs(not structures) and defined constants **********************************************/
|
||||||
|
|
||||||
|
/*** enums ***************************************************************************************/
|
||||||
|
|
||||||
|
/*** structures declarations (and typedefs of structures)*****************************************/
|
||||||
|
|
||||||
|
/*** global variables defined in .c file *********************************************************/
|
||||||
|
|
||||||
|
/*** declarations of public functions ************************************************************/
|
||||||
|
|
||||||
|
void aspell_init (void);
|
||||||
|
void aspell_clean (void);
|
||||||
|
gboolean aspell_check (const char *word, const int word_size);
|
||||||
|
unsigned int aspell_suggest (GArray *suggest, const char *word, const int word_size);
|
||||||
|
void aspell_array_clean (GArray *array);
|
||||||
|
unsigned int aspell_get_lang_list (GArray *lang_list);
|
||||||
|
const char *aspell_get_lang (void);
|
||||||
|
gboolean aspell_set_lang (const char *lang);
|
||||||
|
gboolean aspell_add_to_dict (const char *word, const int word_size);
|
||||||
|
|
||||||
|
/*** inline functions ****************************************************************************/
|
||||||
|
|
||||||
|
#endif /* MC__EDIT_ASPELL_H */
|
182
src/editor/spell_dialogs.c
Normal file
182
src/editor/spell_dialogs.c
Normal file
@ -0,0 +1,182 @@
|
|||||||
|
/*
|
||||||
|
Editor spell checker dialogs
|
||||||
|
|
||||||
|
Copyright (C) 2012
|
||||||
|
The Free Software Foundation, Inc.
|
||||||
|
|
||||||
|
Written by:
|
||||||
|
Ilia Maslakov <il.smind@gmail.com>, 2012
|
||||||
|
|
||||||
|
This file is part of the Midnight Commander.
|
||||||
|
|
||||||
|
The Midnight Commander is free software: you can redistribute it
|
||||||
|
and/or modify it under the terms of the GNU General Public License as
|
||||||
|
published by the Free Software Foundation, either version 3 of the License,
|
||||||
|
or (at your option) any later version.
|
||||||
|
|
||||||
|
The Midnight Commander is distributed in the hope that it will be useful,
|
||||||
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
GNU General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU General Public License
|
||||||
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <config.h>
|
||||||
|
|
||||||
|
#include "lib/global.h"
|
||||||
|
#include "lib/strutil.h" /* str_term_width1 */
|
||||||
|
#include "lib/widget.h"
|
||||||
|
#include "lib/tty/tty.h" /* COLS, LINES */
|
||||||
|
|
||||||
|
#include "editwidget.h"
|
||||||
|
|
||||||
|
#include "spell.h"
|
||||||
|
#include "spell_dialogs.h"
|
||||||
|
|
||||||
|
/*** global variables ****************************************************************************/
|
||||||
|
|
||||||
|
/*** file scope macro definitions ****************************************************************/
|
||||||
|
|
||||||
|
/*** file scope type declarations ****************************************************************/
|
||||||
|
|
||||||
|
/*** file scope variables ************************************************************************/
|
||||||
|
|
||||||
|
/*** file scope functions ************************************************************************/
|
||||||
|
|
||||||
|
/* --------------------------------------------------------------------------------------------- */
|
||||||
|
/*** public functions ****************************************************************************/
|
||||||
|
/* --------------------------------------------------------------------------------------------- */
|
||||||
|
/**
|
||||||
|
* Show suggests for the current word.
|
||||||
|
*
|
||||||
|
* @param edit Editor object
|
||||||
|
* @param word Word for spell check
|
||||||
|
* @param new_word Word to replace the incorrect word
|
||||||
|
* @param suggest Array of suggests for current word
|
||||||
|
* @returns code of pressed button
|
||||||
|
*/
|
||||||
|
|
||||||
|
int
|
||||||
|
spell_dialog_spell_suggest_show (WEdit * edit, const char *word, char **new_word, GArray *suggest)
|
||||||
|
{
|
||||||
|
|
||||||
|
int sug_dlg_h = 14; /* dialog height */
|
||||||
|
int sug_dlg_w = 29; /* dialog width */
|
||||||
|
int xpos, ypos;
|
||||||
|
char *lang_label;
|
||||||
|
char *word_label;
|
||||||
|
unsigned int i;
|
||||||
|
int res;
|
||||||
|
char *curr = NULL;
|
||||||
|
Dlg_head *sug_dlg;
|
||||||
|
WListbox *sug_list;
|
||||||
|
int max_btn_len = 0;
|
||||||
|
int add_len;
|
||||||
|
int replace_len;
|
||||||
|
int skip_len;
|
||||||
|
int cancel_len;
|
||||||
|
WButton *add_btn;
|
||||||
|
WButton *replace_btn;
|
||||||
|
WButton *skip_btn;
|
||||||
|
WButton *cancel_button;
|
||||||
|
int word_label_len;
|
||||||
|
|
||||||
|
/* calculate the dialog metrics */
|
||||||
|
xpos = (COLS - sug_dlg_w) / 2;
|
||||||
|
ypos = (LINES - sug_dlg_h) * 2 / 3;
|
||||||
|
|
||||||
|
/* Sometimes menu can hide replaced text. I don't like it */
|
||||||
|
if ((edit->curs_row >= ypos - 1) && (edit->curs_row <= ypos + sug_dlg_h - 1))
|
||||||
|
ypos -= sug_dlg_h;
|
||||||
|
|
||||||
|
add_btn = button_new (5, 28, B_ADD_WORD, NORMAL_BUTTON, _("&Add word"), 0);
|
||||||
|
add_len = button_get_len (add_btn);
|
||||||
|
replace_btn = button_new (7, 28, B_ENTER, NORMAL_BUTTON, _("&Replace"), 0);
|
||||||
|
replace_len = button_get_len (replace_btn);
|
||||||
|
skip_btn = button_new (9, 28, B_SKIP_WORD, NORMAL_BUTTON, _("&Skip"), 0);
|
||||||
|
skip_len = button_get_len (skip_btn);
|
||||||
|
cancel_button = button_new (11, 28, B_CANCEL, NORMAL_BUTTON, _("&Cancel"), 0);
|
||||||
|
cancel_len = button_get_len (cancel_button);
|
||||||
|
|
||||||
|
max_btn_len = max (replace_len, skip_len);
|
||||||
|
max_btn_len = max (max_btn_len, cancel_len);
|
||||||
|
|
||||||
|
lang_label = g_strdup_printf ("%s: %s", _("Language"), aspell_get_lang ());
|
||||||
|
word_label = g_strdup_printf ("%s: %s", _("Misspelled"), word);
|
||||||
|
word_label_len = str_term_width1 (word_label) + 5;
|
||||||
|
|
||||||
|
sug_dlg_w += max_btn_len;
|
||||||
|
sug_dlg_w = max (sug_dlg_w, word_label_len) + 1;
|
||||||
|
sug_dlg = create_dlg (TRUE, ypos, xpos, sug_dlg_h, sug_dlg_w,
|
||||||
|
dialog_colors, NULL, NULL, "[ASpell]", _("Check word"), DLG_COMPACT);
|
||||||
|
|
||||||
|
sug_list = listbox_new (5, 2, sug_dlg_h - 7, 24, FALSE, NULL);
|
||||||
|
for (i = 0; i < suggest->len; i++)
|
||||||
|
listbox_add_item (sug_list, LISTBOX_APPEND_AT_END, 0, g_array_index (suggest, char *, i),
|
||||||
|
NULL);
|
||||||
|
add_widget (sug_dlg, sug_list);
|
||||||
|
|
||||||
|
add_widget (sug_dlg, add_btn);
|
||||||
|
add_widget (sug_dlg, replace_btn);
|
||||||
|
add_widget (sug_dlg, skip_btn);
|
||||||
|
add_widget (sug_dlg, cancel_button);
|
||||||
|
|
||||||
|
add_widget (sug_dlg, label_new (1, 2, lang_label));
|
||||||
|
add_widget (sug_dlg, label_new (3, 2, word_label));
|
||||||
|
add_widget (sug_dlg, groupbox_new (4, 2, sug_dlg_h - 5, 25, _("Suggest")));
|
||||||
|
|
||||||
|
res = run_dlg (sug_dlg);
|
||||||
|
if (res == B_ENTER)
|
||||||
|
{
|
||||||
|
char *tmp = NULL;
|
||||||
|
listbox_get_current (sug_list, &curr, NULL);
|
||||||
|
|
||||||
|
if (curr != NULL)
|
||||||
|
tmp = g_strdup (curr);
|
||||||
|
*new_word = tmp;
|
||||||
|
}
|
||||||
|
|
||||||
|
destroy_dlg (sug_dlg);
|
||||||
|
g_free (lang_label);
|
||||||
|
g_free (word_label);
|
||||||
|
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* --------------------------------------------------------------------------------------------- */
|
||||||
|
/**
|
||||||
|
* Show dialog to select language for spell check.
|
||||||
|
*
|
||||||
|
* @param languages Array of available languages
|
||||||
|
* @returns name of choosed language
|
||||||
|
*/
|
||||||
|
|
||||||
|
char *
|
||||||
|
spell_dialog_lang_list_show (GArray *languages)
|
||||||
|
{
|
||||||
|
|
||||||
|
int lang_dlg_h = 12; /* dialog height */
|
||||||
|
int lang_dlg_w = 30; /* dialog width */
|
||||||
|
char *selected_lang = NULL;
|
||||||
|
unsigned int i;
|
||||||
|
int res;
|
||||||
|
Listbox *lang_list;
|
||||||
|
|
||||||
|
/* Create listbox */
|
||||||
|
lang_list = create_listbox_window_centered (-1, -1, lang_dlg_h, lang_dlg_w,
|
||||||
|
_("Select language"), "[ASpell]");
|
||||||
|
|
||||||
|
for (i = 0; i < languages->len; i++)
|
||||||
|
LISTBOX_APPEND_TEXT (lang_list, 0, g_array_index (languages, char *, i), NULL);
|
||||||
|
|
||||||
|
res = run_listbox (lang_list);
|
||||||
|
if (res >= 0)
|
||||||
|
selected_lang = g_strdup (g_array_index (languages, char *, (unsigned int) res));
|
||||||
|
|
||||||
|
return selected_lang;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/* --------------------------------------------------------------------------------------------- */
|
25
src/editor/spell_dialogs.h
Normal file
25
src/editor/spell_dialogs.h
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
#ifndef MC__EDIT_ASPELL_DIALOGS_H
|
||||||
|
#define MC__EDIT_ASPELL_DIALOGS_H
|
||||||
|
|
||||||
|
#include "lib/global.h" /* include <glib.h> */
|
||||||
|
|
||||||
|
/*** typedefs(not structures) and defined constants **********************************************/
|
||||||
|
|
||||||
|
#define B_SKIP_WORD (B_USER+3)
|
||||||
|
#define B_ADD_WORD (B_USER+4)
|
||||||
|
|
||||||
|
/*** enums ***************************************************************************************/
|
||||||
|
|
||||||
|
/*** structures declarations (and typedefs of structures)*****************************************/
|
||||||
|
|
||||||
|
/*** global variables defined in .c file *********************************************************/
|
||||||
|
|
||||||
|
/*** declarations of public functions ************************************************************/
|
||||||
|
|
||||||
|
int spell_dialog_spell_suggest_show (WEdit * edit, const char *word, char **new_word,
|
||||||
|
GArray *suggest);
|
||||||
|
char *spell_dialog_lang_list_show (GArray *languages);
|
||||||
|
|
||||||
|
/*** inline functions ****************************************************************************/
|
||||||
|
|
||||||
|
#endif /* MC__EDIT_ASPELL_DIALOGS_H */
|
@ -416,6 +416,9 @@ static const global_keymap_ini_t default_editor_keymap[] = {
|
|||||||
{"Sort", "alt-t"},
|
{"Sort", "alt-t"},
|
||||||
{"Mail", "alt-m"},
|
{"Mail", "alt-m"},
|
||||||
{"ExternalCommand", "alt-u"},
|
{"ExternalCommand", "alt-u"},
|
||||||
|
#ifdef HAVE_ASPELL
|
||||||
|
{"SpellCheckCurrentWord", "ctrl-p"},
|
||||||
|
#endif
|
||||||
{"ExtendedKeyMap", "ctrl-x"},
|
{"ExtendedKeyMap", "ctrl-x"},
|
||||||
{NULL, NULL}
|
{NULL, NULL}
|
||||||
};
|
};
|
||||||
|
Loading…
Reference in New Issue
Block a user