summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--src/ex_docmd.c13
-rw-r--r--src/normal.c21
-rw-r--r--src/spell.c590
3 files changed, 486 insertions, 138 deletions
diff --git a/src/ex_docmd.c b/src/ex_docmd.c
index 503be3fec..09e12bcc0 100644
--- a/src/ex_docmd.c
+++ b/src/ex_docmd.c
@@ -222,8 +222,7 @@ static void ex_popup __ARGS((exarg_T *eap));
#endif
#ifndef FEAT_SYN_HL
# define ex_syntax ex_ni
-#endif
-#if !defined(FEAT_SYN_HL) || !defined(FEAT_MBYTE)
+# define ex_spell ex_ni
# define ex_mkspell ex_ni
#endif
#ifndef FEAT_MZSCHEME
@@ -9906,7 +9905,7 @@ ex_viminfo(eap)
#if defined(FEAT_GUI_DIALOG) || defined(FEAT_CON_DIALOG) || defined(PROTO)
/*
* Make a dialog message in "buff[IOSIZE]".
- * "format" must contain "%.*s".
+ * "format" must contain "%s".
*/
void
dialog_msg(buff, format, fname)
@@ -9914,15 +9913,9 @@ dialog_msg(buff, format, fname)
char *format;
char_u *fname;
{
- int len;
-
if (fname == NULL)
fname = (char_u *)_("Untitled");
- len = (int)STRLEN(format) + (int)STRLEN(fname);
- if (len >= IOSIZE)
- sprintf((char *)buff, format, (int)(IOSIZE - STRLEN(format)), fname);
- else
- sprintf((char *)buff, format, (int)STRLEN(fname), fname);
+ vim_snprintf((char *)buff, IOSIZE, format, fname);
}
#endif
diff --git a/src/normal.c b/src/normal.c
index c3c762757..98ff6965e 100644
--- a/src/normal.c
+++ b/src/normal.c
@@ -4665,6 +4665,27 @@ dozet:
#endif /* FEAT_FOLDING */
+#ifdef FEAT_SYN_HL
+ case 'g': /* "zg": add good word to word list */
+ case 'w': /* "zw": add wrong word to word list */
+ {
+ char_u *ptr = NULL;
+ int len;
+
+ if (checkclearop(cap->oap))
+ break;
+# ifdef FEAT_VISUAL
+ if (VIsual_active && get_visual_text(cap, &ptr, &len)
+ == FAIL)
+ return;
+# endif
+ if (ptr == NULL && (len = find_ident_under_cursor(&ptr,
+ FIND_IDENT)) == 0)
+ return;
+ spell_add_word(ptr, len, nchar == 'w');
+ }
+#endif
+
default: clearopbeep(cap->oap);
}
diff --git a/src/spell.c b/src/spell.c
index 734c7768e..4063895be 100644
--- a/src/spell.c
+++ b/src/spell.c
@@ -151,6 +151,8 @@ struct slang_S
{
slang_T *sl_next; /* next language */
char_u *sl_name; /* language name "en", "en.rare", "nl", etc. */
+ char_u *sl_fname; /* name of .spl file */
+ int sl_add; /* TRUE if it's an addition. */
char_u *sl_fbyts; /* case-folded word bytes */
int *sl_fidxs; /* case-folded word indexes */
char_u *sl_kbyts; /* keep-case word bytes */
@@ -248,17 +250,19 @@ static int set_spell_finish __ARGS((spelltab_T *new_st));
static slang_T *slang_alloc __ARGS((char_u *lang));
static void slang_free __ARGS((slang_T *lp));
+static void slang_clear __ARGS((slang_T *lp));
static void find_word __ARGS((matchinf_T *mip, int keepcap));
static void spell_load_lang __ARGS((char_u *lang));
-static void spell_load_file __ARGS((char_u *fname, void *cookie));
+static char_u *spell_enc __ARGS((void));
+static void spell_load_cb __ARGS((char_u *fname, void *cookie));
+static void spell_load_file __ARGS((char_u *fname, char_u *lang, slang_T *old_lp));
static int read_tree __ARGS((FILE *fd, char_u *byts, int *idxs, int maxidx, int startidx));
static int find_region __ARGS((char_u *rp, char_u *region));
static int captype __ARGS((char_u *word, char_u *end));
+static void spell_reload_one __ARGS((char_u *fname));
static int set_spell_charflags __ARGS((char_u *flags, int cnt, char_u *upp));
-#ifdef FEAT_MBYTE
static int set_spell_chartab __ARGS((char_u *fol, char_u *low, char_u *upp));
static void write_spell_chartab __ARGS((FILE *fd));
-#endif
static int spell_isupper __ARGS((int c));
static int spell_casefold __ARGS((char_u *p, int len, char_u *buf, int buflen));
@@ -633,7 +637,7 @@ spell_move_to(dir, allwords)
int col;
int can_spell;
- if (!curwin->w_p_spell || *curwin->w_buffer->b_p_spl == NUL)
+ if (!curwin->w_p_spell || *curbuf->b_p_spl == NUL)
{
EMSG(_("E756: Spell checking not enabled"));
return FAIL;
@@ -746,44 +750,59 @@ spell_move_to(dir, allwords)
/*
* Load word list(s) for "lang" from Vim spell file(s).
- * "lang" must be the language without the region: "en".
+ * "lang" must be the language without the region: e.g., "en".
*/
static void
spell_load_lang(lang)
char_u *lang;
{
- char_u fname_enc[80];
- char_u *p;
+ char_u fname_enc[85];
int r;
char_u langcp[MAXWLEN + 1];
- /* Copy the language name to pass it to spell_load_file() as a cookie.
+ /* Copy the language name to pass it to spell_load_cb() as a cookie.
* It's truncated when an error is detected. */
STRCPY(langcp, lang);
- /* Find all spell files for "lang" in 'runtimepath' and load them.
- * Use 'encoding', except that we use "latin1" for "latin9". */
-#ifdef FEAT_MBYTE
- if (STRLEN(p_enc) < 60 && STRCMP(p_enc, "iso-8859-15") != 0)
- p = p_enc;
- else
-#endif
- p = (char_u *)"latin1";
- vim_snprintf((char *)fname_enc, sizeof(fname_enc),
- "spell/%s.%s.spl", lang, p);
- r = do_in_runtimepath(fname_enc, TRUE, spell_load_file, &langcp);
+ /*
+ * Find the first spell file for "lang" in 'runtimepath' and load it.
+ */
+ vim_snprintf((char *)fname_enc, sizeof(fname_enc) - 5,
+ "spell/%s.%s.spl", lang, spell_enc());
+ r = do_in_runtimepath(fname_enc, FALSE, spell_load_cb, &langcp);
if (r == FAIL && *langcp != NUL)
{
/* Try loading the ASCII version. */
- vim_snprintf((char *)fname_enc, sizeof(fname_enc),
+ vim_snprintf((char *)fname_enc, sizeof(fname_enc) - 5,
"spell/%s.ascii.spl", lang);
- r = do_in_runtimepath(fname_enc, TRUE, spell_load_file, &langcp);
+ r = do_in_runtimepath(fname_enc, FALSE, spell_load_cb, &langcp);
}
if (r == FAIL)
smsg((char_u *)_("Warning: Cannot find word list \"%s\""),
fname_enc + 6);
+ else if (*langcp != NUL)
+ {
+ /* Load all the additions. */
+ STRCPY(fname_enc + STRLEN(fname_enc) - 3, "add.spl");
+ do_in_runtimepath(fname_enc, TRUE, spell_load_cb, &langcp);
+ }
+}
+
+/*
+ * Return the encoding used for spell checking: Use 'encoding', except that we
+ * use "latin1" for "latin9". And limit to 60 characters (just in case).
+ */
+ static char_u *
+spell_enc()
+{
+
+#ifdef FEAT_MBYTE
+ if (STRLEN(p_enc) < 60 && STRCMP(p_enc, "iso-8859-15") != 0)
+ return p_enc;
+#endif
+ return (char_u *)"latin1";
}
/*
@@ -813,13 +832,29 @@ slang_free(lp)
slang_T *lp;
{
vim_free(lp->sl_name);
+ vim_free(lp->sl_fname);
+ slang_clear(lp);
+ vim_free(lp);
+}
+
+/*
+ * Clear an slang_T so that the file can be reloaded.
+ */
+ static void
+slang_clear(lp)
+ slang_T *lp;
+{
vim_free(lp->sl_fbyts);
+ lp->sl_fbyts = NULL;
vim_free(lp->sl_kbyts);
+ lp->sl_kbyts = NULL;
vim_free(lp->sl_fidxs);
+ lp->sl_fidxs = NULL;
vim_free(lp->sl_kidxs);
+ lp->sl_kidxs = NULL;
ga_clear(&lp->sl_rep);
vim_free(lp->sl_try);
- vim_free(lp);
+ lp->sl_try = NULL;
}
/*
@@ -827,11 +862,28 @@ slang_free(lp)
* Invoked through do_in_runtimepath().
*/
static void
-spell_load_file(fname, cookie)
+spell_load_cb(fname, cookie)
char_u *fname;
void *cookie; /* points to the language name */
{
- char_u *lang = cookie;
+ spell_load_file(fname, (char_u *)cookie, NULL);
+}
+
+/*
+ * Load one spell file and store the info into a slang_T.
+ *
+ * This is invoked in two ways:
+ * - From spell_load_cb() to load a spell file for the first time. "lang" is
+ * the language name, "old_lp" is NULL. Will allocate an slang_T.
+ * - To reload a spell file that was changed. "lang" is NULL and "old_lp"
+ * points to the existing slang_T.
+ */
+ static void
+spell_load_file(fname, lang, old_lp)
+ char_u *fname;
+ char_u *lang;
+ slang_T *old_lp;
+{
FILE *fd;
char_u buf[MAXWLEN + 1];
char_u *p;
@@ -844,16 +896,35 @@ spell_load_file(fname, cookie)
char_u *fol;
slang_T *lp = NULL;
- fd = fopen((char *)fname, "r");
+ fd = mch_fopen((char *)fname, "r");
if (fd == NULL)
{
EMSG2(_(e_notopen), fname);
goto endFAIL;
}
+ if (p_verbose > 2)
+ {
+ verbose_enter();
+ smsg((char_u *)_("Reading spell file \"%s\""), fname);
+ verbose_leave();
+ }
- lp = slang_alloc(lang);
- if (lp == NULL)
- goto endFAIL;
+ if (old_lp == NULL)
+ {
+ lp = slang_alloc(lang);
+ if (lp == NULL)
+ goto endFAIL;
+
+ /* Remember the file name, used to reload the file when it's updated. */
+ lp->sl_fname = vim_strsave(fname);
+ if (lp->sl_fname == NULL)
+ goto endFAIL;
+
+ /* Check for .add.spl. */
+ lp->sl_add = strstr((char *)gettail(fname), ".add.") != NULL;
+ }
+ else
+ lp = old_lp;
/* Set sourcing_name, so that error messages mention the file name. */
sourcing_name = fname;
@@ -978,15 +1049,20 @@ formerr:
}
}
- lp->sl_next = first_lang;
- first_lang = lp;
+ /* For a new file link it in the list of spell files. */
+ if (old_lp == NULL)
+ {
+ lp->sl_next = first_lang;
+ first_lang = lp;
+ }
goto endOK;
endFAIL:
- /* truncating the name signals the error to spell_load_lang() */
- *lang = NUL;
- if (lp != NULL)
+ if (lang != NULL)
+ /* truncating the name signals the error to spell_load_lang() */
+ *lang = NUL;
+ if (lp != NULL && old_lp == NULL)
slang_free(lp);
endOK:
@@ -1143,7 +1219,7 @@ did_set_spelllang(buf)
for (lp = first_lang; lp != NULL; lp = lp->sl_next)
if (STRNICMP(lp->sl_name, lang, 2) == 0)
{
- if (region == NULL)
+ if (region == NULL || lp->sl_add)
region_mask = REGION_ALL;
else
{
@@ -1237,10 +1313,11 @@ captype(word, end)
if (p >= end)
return 0; /* only non-word characters, illegal word */
#ifdef FEAT_MBYTE
- c = mb_ptr2char_adv(&p);
-#else
- c = *p++;
+ if (has_mbyte)
+ c = mb_ptr2char_adv(&p);
+ else
#endif
+ c = *p++;
firstcap = allcap = spell_isupper(c);
/*
@@ -1307,11 +1384,27 @@ spell_reload()
}
# endif
+/*
+ * Reload the spell file "fname" if it's loaded.
+ */
+ static void
+spell_reload_one(fname)
+ char_u *fname;
+{
+ slang_T *lp;
+
+ for (lp = first_lang; lp != NULL; lp = lp->sl_next)
+ if (fullpathcmp(fname, lp->sl_fname, FALSE) == FPC_SAME)
+ {
+ slang_clear(lp);
+ spell_load_file(fname, NULL, lp);
+ redraw_all_later(NOT_VALID);
+ }
+}
+
-#if defined(FEAT_MBYTE) || defined(PROTO)
/*
* Functions for ":mkspell".
- * Only possible with the multi-byte feature.
*/
#define MAXLINELEN 500 /* Maximum length in bytes of a line in a .aff
@@ -1323,8 +1416,8 @@ typedef struct afffile_S
{
char_u *af_enc; /* "SET", normalized, alloc'ed string or NULL */
char_u *af_try; /* "TRY" line in "af_enc" encoding */
- int af_rar; /* ID for rare word */
- int af_huh; /* ID for keep-case word */
+ int af_rar; /* RAR ID for rare word */
+ int af_kep; /* KEP ID for keep-case word */
hashtab_T af_pref; /* hashtable for prefixes, affheader_T */
hashtab_T af_suff; /* hashtable for suffixes, affheader_T */
garray_T af_rep; /* list of repentry_T entries from REP lines */
@@ -1395,9 +1488,11 @@ typedef struct spellinfo_S
wordnode_T *si_keeproot; /* tree with keep-case words */
sblock_T *si_blocks; /* memory blocks used */
int si_ascii; /* handling only ASCII words */
+ int si_add; /* addition file */
int si_region; /* region mask */
vimconv_T si_conv; /* for conversion to 'encoding' */
int si_memtot; /* runtime memory used */
+ int si_verbose; /* verbose messages */
} spellinfo_T;
static afffile_T *spell_read_aff __ARGS((char_u *fname, spellinfo_T *spin));
@@ -1412,11 +1507,13 @@ static void free_blocks __ARGS((sblock_T *bl));
static wordnode_T *wordtree_alloc __ARGS((sblock_T **blp));
static int store_word __ARGS((char_u *word, spellinfo_T *spin, int flags));
static int tree_add_word __ARGS((char_u *word, wordnode_T *tree, int flags, int region, sblock_T **blp));
-static void wordtree_compress __ARGS((wordnode_T *root));
+static void wordtree_compress __ARGS((wordnode_T *root, spellinfo_T *spin));
static int node_compress __ARGS((wordnode_T *node, hashtab_T *ht, int *tot));
static int node_equal __ARGS((wordnode_T *n1, wordnode_T *n2));
static void write_vim_spell __ARGS((char_u *fname, spellinfo_T *spin, int regcount, char_u *regchars));
static int put_tree __ARGS((FILE *fd, wordnode_T *node, int index, int regionmask));
+static void mkspell __ARGS((int fcount, char_u **fnames, int ascii, int overwrite, int verbose));
+static void init_spellfile __ARGS((void));
/*
* Read an affix ".aff" file.
@@ -1447,15 +1544,22 @@ spell_read_aff(fname, spin)
/*
* Open the file.
*/
- fd = fopen((char *)fname, "r");
+ fd = mch_fopen((char *)fname, "r");
if (fd == NULL)
{
EMSG2(_(e_notopen), fname);
return NULL;
}
- smsg((char_u *)_("Reading affix file %s..."), fname);
- out_flush();
+ if (spin->si_verbose || p_verbose > 2)
+ {
+ if (!spin->si_verbose)
+ verbose_enter();
+ smsg((char_u *)_("Reading affix file %s..."), fname);
+ out_flush();
+ if (!spin->si_verbose)
+ verbose_leave();
+ }
/*
* Allocate and init the afffile_T structure.
@@ -1481,6 +1585,7 @@ spell_read_aff(fname, spin)
/* Convert from "SET" to 'encoding' when needed. */
vim_free(pc);
+#ifdef FEAT_MBYTE
if (spin->si_conv.vc_type != CONV_NONE)
{
pc = string_convert(&spin->si_conv, rline, NULL);
@@ -1493,6 +1598,7 @@ spell_read_aff(fname, spin)
line = pc;
}
else
+#endif
{
pc = NULL;
line = rline;
@@ -1523,6 +1629,7 @@ spell_read_aff(fname, spin)
if (STRCMP(items[0], "SET") == 0 && itemcnt == 2
&& aff->af_enc == NULL)
{
+#ifdef FEAT_MBYTE
/* Setup for conversion from "ENC" to 'encoding'. */
aff->af_enc = enc_canonize(items[1]);
if (aff->af_enc != NULL && !spin->si_ascii
@@ -1530,6 +1637,9 @@ spell_read_aff(fname, spin)
p_enc) == FAIL)
smsg((char_u *)_("Conversion in %s not supported: from %s to %s"),
fname, aff->af_enc, p_enc);
+#else
+ smsg((char_u *)_("Conversion in %s not supported"), fname);
+#endif
}
else if (STRCMP(items[0], "NOSPLITSUGS") == 0 && itemcnt == 1)
{
@@ -1547,10 +1657,10 @@ spell_read_aff(fname, spin)
if (items[1][1] != NUL)
smsg((char_u *)_(e_affname), fname, lnum, items[1]);
}
- else if (STRCMP(items[0], "HUH") == 0 && itemcnt == 2
- && aff->af_huh == 0)
+ else if (STRCMP(items[0], "KEP") == 0 && itemcnt == 2
+ && aff->af_kep == 0)
{
- aff->af_huh = items[1][0];
+ aff->af_kep = items[1][0];
if (items[1][1] != NUL)
smsg((char_u *)_(e_affname), fname, lnum, items[1]);
}
@@ -1782,7 +1892,7 @@ spell_read_dic(fname, spin, affile)
/*
* Open the file.
*/
- fd = fopen((char *)fname, "r");
+ fd = mch_fopen((char *)fname, "r");
if (fd == NULL)
{
EMSG2(_(e_notopen), fname);
@@ -1792,8 +1902,15 @@ spell_read_dic(fname, spin, affile)
/* The hashtable is only used to detect duplicated words. */
hash_init(&ht);
- smsg((char_u *)_("Reading dictionary file %s..."), fname);
- out_flush();
+ if (spin->si_verbose || p_verbose > 2)
+ {
+ if (!spin->si_verbose)
+ verbose_enter();
+ smsg((char_u *)_("Reading dictionary file %s..."), fname);
+ out_flush();
+ if (!spin->si_verbose)
+ verbose_leave();
+ }
/* Read and ignore the first line: word count. */
(void)vim_fgets(line, MAXLINELEN, fd);
@@ -1820,7 +1937,7 @@ spell_read_dic(fname, spin, affile)
line[l] = NUL;
/* This takes time, print a message now and then. */
- if ((lnum & 0x3ff) == 0)
+ if (spin->si_verbose && (lnum & 0x3ff) == 0)
{
vim_snprintf((char *)message, sizeof(message),
_("line %6d - %s"), lnum, line);
@@ -1844,6 +1961,7 @@ spell_read_dic(fname, spin, affile)
continue;
}
+#ifdef FEAT_MBYTE
/* Convert from "SET" to 'encoding' when needed. */
if (spin->si_conv.vc_type != CONV_NONE)
{
@@ -1857,6 +1975,7 @@ spell_read_dic(fname, spin, affile)
w = pc;
}
else
+#endif
{
pc = NULL;
w = line;
@@ -1883,8 +2002,8 @@ spell_read_dic(fname, spin, affile)
{
/* Check for affix name that stands for keep-case word and stands
* for rare word (if defined). */
- if (affile->af_huh != NUL
- && vim_strchr(afflist, affile->af_huh) != NULL)
+ if (affile->af_kep != NUL
+ && vim_strchr(afflist, affile->af_kep) != NULL)
flags |= WF_KEEPCAP;
if (affile->af_rar != NUL
&& vim_strchr(afflist, affile->af_rar) != NULL)
@@ -1983,9 +2102,17 @@ store_aff_word(word, spin, afflist, ht, xht, comb, flags)
STRCPY(newword, ae->ae_add);
p = word;
if (ae->ae_chop != NULL)
+ {
/* Skip chop string. */
- for (i = mb_charlen(ae->ae_chop); i > 0; --i)
+#ifdef FEAT_MBYTE
+ if (has_mbyte)
+ i = mb_charlen(ae->ae_chop);
+ else
+#endif
+ i = STRLEN(ae->ae_chop);
+ for ( ; i > 0; --i)
mb_ptr_adv(p);
+ }
STRCAT(newword, p);
}
else
@@ -1996,7 +2123,13 @@ store_aff_word(word, spin, afflist, ht, xht, comb, flags)
{
/* Remove chop string. */
p = newword + STRLEN(newword);
- for (i = mb_charlen(ae->ae_chop); i > 0; --i)
+#ifdef FEAT_MBYTE
+ if (has_mbyte)
+ i = mb_charlen(ae->ae_chop);
+ else
+#endif
+ i = STRLEN(ae->ae_chop);
+ for ( ; i > 0; --i)
mb_ptr_back(newword, p);
*p = NUL;
}
@@ -2040,21 +2173,27 @@ spell_read_wordfile(fname, spin)
int retval = OK;
int did_word = FALSE;
int non_ascii = 0;
- char_u *enc;
int flags;
/*
* Open the file.
*/
- fd = fopen((char *)fname, "r");
+ fd = mch_fopen((char *)fname, "r");
if (fd == NULL)
{
EMSG2(_(e_notopen), fname);
return FAIL;
}
- smsg((char_u *)_("Reading word file %s..."), fname);
- out_flush();
+ if (spin->si_verbose || p_verbose > 2)
+ {
+ if (!spin->si_verbose)
+ verbose_enter();
+ smsg((char_u *)_("Reading word file %s..."), fname);
+ out_flush();
+ if (!spin->si_verbose)
+ verbose_leave();
+ }
/*
* Read all the lines in the file one by one.
@@ -2078,6 +2217,7 @@ spell_read_wordfile(fname, spin)
/* Convert from "=encoding={encoding}" to 'encoding' when needed. */
vim_free(pc);
+#ifdef FEAT_MBYTE
if (spin->si_conv.vc_type != CONV_NONE)
{
pc = string_convert(&spin->si_conv, rline, NULL);
@@ -2090,6 +2230,7 @@ spell_read_wordfile(fname, spin)
line = pc;
}
else
+#endif
{
pc = NULL;
line = rline;
@@ -2110,6 +2251,9 @@ spell_read_wordfile(fname, spin)
fname, lnum, line);
else
{
+#ifdef FEAT_MBYTE
+ char_u *enc;
+
/* Setup for conversion to 'encoding'. */
enc = enc_canonize(line + 10);
if (enc != NULL && !spin->si_ascii
@@ -2118,6 +2262,9 @@ spell_read_wordfile(fname, spin)
smsg((char_u *)_("Conversion in %s not supported: from %s to %s"),
fname, line + 10, p_enc);
vim_free(enc);
+#else
+ smsg((char_u *)_("Conversion in %s not supported"), fname);
+#endif
}
continue;
}
@@ -2169,9 +2316,15 @@ spell_read_wordfile(fname, spin)
vim_free(pc);
fclose(fd);
- if (spin->si_ascii && non_ascii > 0)
+ if (spin->si_ascii && non_ascii > 0 && (spin->si_verbose || p_verbose > 2))
+ {
+ if (p_verbose > 2)
+ verbose_enter();
smsg((char_u *)_("Ignored %d words with non-ASCII characters"),
non_ascii);
+ if (p_verbose > 2)
+ verbose_leave();
+ }
return retval;
}
@@ -2343,8 +2496,9 @@ tree_add_word(word, root, flags, region, blp)
* Compress a tree: find tails that are identical and can be shared.
*/
static void
-wordtree_compress(root)
+wordtree_compress(root, spin)
wordnode_T *root;
+ spellinfo_T *spin;
{
hashtab_T ht;
int n;
@@ -2354,8 +2508,15 @@ wordtree_compress(root)
{
hash_init(&ht);
n = node_compress(root, &ht, &tot);
- smsg((char_u *)_("Compressed %d of %d nodes; %d%% remaining"),
+ if (spin->si_verbose || p_verbose > 2)
+ {
+ if (!spin->si_verbose)
+ verbose_enter();
+ smsg((char_u *)_("Compressed %d of %d nodes; %d%% remaining"),
n, tot, (tot - n) * 100 / tot);
+ if (p_verbose > 2)
+ verbose_leave();
+ }
hash_clear(&ht);
}
}
@@ -2516,7 +2677,7 @@ write_vim_spell(fname, spin, regcount, regchars)
wordnode_T *tree;
int nodecount;
- fd = fopen((char *)fname, "w");
+ fd = mch_fopen((char *)fname, "w");
if (fd == NULL)
{
EMSG2(_(e_notopen), fname);
@@ -2690,7 +2851,8 @@ put_tree(fd, node, index, regionmask)
/*
- * ":mkspell outfile infile ..."
+ * ":mkspell [-ascii] outfile infile ..."
+ * ":mkspell [-ascii] addfile"
*/
void
ex_mkspell(eap)
@@ -2698,84 +2860,141 @@ ex_mkspell(eap)
{
int fcount;
char_u **fnames;
+ char_u *arg = eap->arg;
+ int ascii = FALSE;
+
+ if (STRNCMP(arg, "-ascii", 6) == 0)
+ {
+ ascii = TRUE;
+ arg = skipwhite(arg + 6);
+ }
+
+ /* Expand all the remaining arguments (e.g., $VIMRUNTIME). */
+ if (get_arglist_exp(arg, &fcount, &fnames) == OK)
+ {
+ mkspell(fcount, fnames, ascii, eap->forceit, TRUE);
+ FreeWild(fcount, fnames);
+ }
+}
+
+/*
+ * Create a Vim spell file from one or more word lists.
+ * "fnames[0]" is the output file name.
+ * "fnames[fcount - 1]" is the last input file name.
+ * Exception: when "fnames[0]" ends in ".add" it's used as the input file name
+ * and ".spl" is appended to make the output file name.
+ */
+ static void
+mkspell(fcount, fnames, ascii, overwrite, verbose)
+ int fcount;
+ char_u **fnames;
+ int ascii; /* -ascii argument given */
+ int overwrite; /* overwrite existing output file */
+ int verbose; /* give progress messages */
+{
char_u fname[MAXPATHL];
char_u wfname[MAXPATHL];
+ char_u **innames;
+ int incount;
afffile_T *(afile[8]);
int i;
int len;
char_u region_name[16];
struct stat st;
- char_u *arg = eap->arg;
int error = FALSE;
spellinfo_T spin;
vim_memset(&spin, 0, sizeof(spin));
+ spin.si_verbose = verbose;
+ spin.si_ascii = ascii;
- if (STRNCMP(arg, "-ascii", 6) == 0)
+ /* default: fnames[0] is output file, following are input files */
+ innames = &fnames[1];
+ incount = fcount - 1;
+
+ if (fcount >= 1)
{
- spin.si_ascii = TRUE;
- arg = skipwhite(arg + 6);
+ len = STRLEN(fnames[0]);
+ if (fcount == 1 && len > 4 && STRCMP(fnames[0] + len - 4, ".add") == 0)
+ {
+ /* For ":mkspell path/en.latin1.add" output file is
+ * "path/en.latin1.add.spl". */
+ innames = &fnames[0];
+ incount = 1;
+ vim_snprintf((char *)wfname, sizeof(wfname), "%s.spl", fnames[0]);
+ }
+ else if (len > 4 && STRCMP(fnames[0] + len - 4, ".spl") == 0)
+ {
+ /* Name ends in ".spl", use as the file name. */
+ STRNCPY(wfname, fnames[0], sizeof(wfname));
+ wfname[sizeof(wfname) - 1] = NUL;
+ }
+ else
+ /* Name should be language, make the file name from it. */
+ vim_snprintf((char *)wfname, sizeof(wfname), "%s.%s.spl", fnames[0],
+ spin.si_ascii ? (char_u *)"ascii" : spell_enc());
+
+ /* Check for .ascii.spl. */
+ if (strstr((char *)gettail(wfname), ".ascii.") != NULL)
+ spin.si_ascii = TRUE;
+
+ /* Check for .add.spl. */
+ if (strstr((char *)gettail(wfname), ".add.") != NULL)
+ spin.si_add = TRUE;
}
- /* Expand all the remaining arguments (e.g., $VIMRUNTIME). */
- if (get_arglist_exp(arg, &fcount, &fnames) == FAIL)
- return;
- if (fcount < 2)
+ if (incount <= 0)
EMSG(_(e_invarg)); /* need at least output and input names */
- else if (fcount > 9)
+ else if (incount > 8)
EMSG(_("E754: Only up to 8 regions supported"));
else
{
/* Check for overwriting before doing things that may take a lot of
* time. */
- vim_snprintf((char *)wfname, sizeof(wfname), "%s.%s.spl", fnames[0],
- spin.si_ascii ? (char_u *)"ascii" : p_enc);
- if (!eap->forceit && mch_stat((char *)wfname, &st) >= 0)
+ if (!overwrite && mch_stat((char *)wfname, &st) >= 0)
{
EMSG(_(e_exists));
- goto theend;
+ return;
}
- if (mch_isdir(fnames[0]))
+ if (mch_isdir(wfname))
{
- EMSG2(_(e_isadir2), fnames[0]);
- goto theend;
+ EMSG2(_(e_isadir2), wfname);
+ return;
}
/*
* Init the aff and dic pointers.
* Get the region names if there are more than 2 arguments.
*/
- for (i = 1; i < fcount; ++i)
+ for (i = 0; i < incount; ++i)
{
- afile[i - 1] = NULL;
+ afile[i] = NULL;
if (fcount > 2)
{
- len = STRLEN(fnames[i]);
- if (STRLEN(gettail(fnames[i])) < 5 || fnames[i][len - 3] != '_')
- {
- EMSG2(_("E755: Invalid region in %s"), fnames[i]);
- goto theend;
- }
- else
+ len = STRLEN(innames[i]);
+ if (STRLEN(gettail(innames[i])) < 5
+ || innames[i][len - 3] != '_')
{
- region_name[(i - 1) * 2] = TOLOWER_ASC(fnames[i][len - 2]);
- region_name[(i - 1) * 2 + 1] =
- TOLOWER_ASC(fnames[i][len - 1]);
+ EMSG2(_("E755: Invalid region in %s"), innames[i]);
+ return;
}
+ region_name[i * 2] = TOLOWER_ASC(innames[i][len - 2]);
+ region_name[i * 2 + 1] = TOLOWER_ASC(innames[i][len - 1]);
}
}
- /* Clear the char type tables, don't want to use any of the currently
- * used spell properties. */
- init_spell_chartab();
+ if (!spin.si_add)
+ /* Clear the char type tables, don't want to use any of the
+ * currently used spell properties. */
+ init_spell_chartab();
spin.si_foldroot = wordtree_alloc(&spin.si_blocks);
spin.si_keeproot = wordtree_alloc(&spin.si_blocks);
if (spin.si_foldroot == NULL || spin.si_keeproot == NULL)
{
error = TRUE;
- goto theend;
+ return;
}
/*
@@ -2783,25 +3002,25 @@ ex_mkspell(eap)
* Text is converted to 'encoding'.
* Words are stored in the case-folded and keep-case trees.
*/
- for (i = 1; i < fcount && !error; ++i)
+ for (i = 0; i < incount && !error; ++i)
{
spin.si_conv.vc_type = CONV_NONE;
- spin.si_region = 1 << (i - 1);
+ spin.si_region = 1 << i;
- vim_snprintf((char *)fname, sizeof(fname), "%s.aff", fnames[i]);
+ vim_snprintf((char *)fname, sizeof(fname), "%s.aff", innames[i]);
if (mch_stat((char *)fname, &st) >= 0)
{
/* Read the .aff file. Will init "spin->si_conv" based on the
* "SET" line. */
- afile[i - 1] = spell_read_aff(fname, &spin);
- if (afile[i - 1] == NULL)
+ afile[i] = spell_read_aff(fname, &spin);
+ if (afile[i] == NULL)
error = TRUE;
else
{
/* Read the .dic file and store the words in the trees. */
vim_snprintf((char *)fname, sizeof(fname), "%s.dic",
- fnames[i]);
- if (spell_read_dic(fname, &spin, afile[i - 1]) == FAIL)
+ innames[i]);
+ if (spell_read_dic(fname, &spin, afile[i]) == FAIL)
error = TRUE;
}
}
@@ -2809,12 +3028,14 @@ ex_mkspell(eap)
{
/* No .aff file, try reading the file as a word list. Store
* the words in the trees. */
- if (spell_read_wordfile(fnames[i], &spin) == FAIL)
+ if (spell_read_wordfile(innames[i], &spin) == FAIL)
error = TRUE;
}
+#ifdef FEAT_MBYTE
/* Free any conversion stuff. */
convert_setup(&spin.si_conv, NULL, NULL);
+#endif
}
if (!error)
@@ -2828,10 +3049,17 @@ ex_mkspell(eap)
/*
* Combine tails in the tree.
*/
- MSG(_("Compressing word tree..."));
- out_flush();
- wordtree_compress(spin.si_foldroot);
- wordtree_compress(spin.si_keeproot);
+ if (verbose || p_verbose > 2)
+ {
+ if (!verbose)
+ verbose_enter();
+ MSG(_("Compressing word tree..."));
+ out_flush();
+ if (!verbose)
+ verbose_leave();
+ }
+ wordtree_compress(spin.si_foldroot, &spin);
+ wordtree_compress(spin.si_keeproot, &spin);
}
if (!error)
@@ -2839,33 +3067,138 @@ ex_mkspell(eap)
/*
* Write the info in the spell file.
*/
- smsg((char_u *)_("Writing spell file %s..."), wfname);
- out_flush();
- write_vim_spell(wfname, &spin, fcount - 1, region_name);
- MSG(_("Done!"));
+ if (verbose || p_verbose > 2)
+ {
+ if (!verbose)
+ verbose_enter();
+ smsg((char_u *)_("Writing spell file %s..."), wfname);
+ out_flush();
+ if (!verbose)
+ verbose_leave();
+ }
+
+ write_vim_spell(wfname, &spin, incount, region_name);
- smsg((char_u *)_("Estimated runtime memory use: %d bytes"),
+ if (verbose || p_verbose > 2)
+ {
+ if (!verbose)
+ verbose_enter();
+ MSG(_("Done!"));
+ smsg((char_u *)_("Estimated runtime memory use: %d bytes"),
spin.si_memtot);
- out_flush();
+ out_flush();
+ if (!verbose)
+ verbose_leave();
+ }
- /* May need to reload spell dictionaries */
- spell_reload();
+ /* If the file is loaded need to reload it. */
+ spell_reload_one(wfname);
}
/* Free the allocated memory. */
free_blocks(spin.si_blocks);
/* Free the .aff file structures. */
- for (i = 1; i < fcount; ++i)
- if (afile[i - 1] != NULL)
- spell_free_aff(afile[i - 1]);
+ for (i = 0; i < incount; ++i)
+ if (afile[i] != NULL)
+ spell_free_aff(afile[i]);
}
+}
-theend:
- FreeWild(fcount, fnames);
+
+/*
+ * ":spellgood {word}"
+ * ":spellwrong {word}"
+ */
+ void
+ex_spell(eap)
+ exarg_T *eap;
+{
+ spell_add_word(eap->arg, STRLEN(eap->arg), eap->cmdidx == CMD_spellwrong);
+}
+
+/*
+ * Add "word[len]" to 'spellfile' as a good or bad word.
+ */
+ void
+spell_add_word(word, len, bad)
+ char_u *word;
+ int len;
+ int bad;
+{
+ FILE *fd;
+ buf_T *buf;
+
+ if (*curbuf->b_p_spf == NUL)
+ init_spellfile();
+ if (*curbuf->b_p_spf == NUL)
+ EMSG(_("E999: 'spellfile' is not set"));
+ else
+ {
+ /* Check that the user isn't editing the .add file somewhere. */
+ buf = buflist_findname_exp(curbuf->b_p_spf);
+ if (buf != NULL && buf->b_ml.ml_mfp == NULL)
+ buf = NULL;
+ if (buf != NULL && bufIsChanged(buf))
+ EMSG(_(e_bufloaded));
+ else
+ {
+ fd = mch_fopen((char *)curbuf->b_p_spf, "a");
+ if (fd == NULL)
+ EMSG2(_(e_notopen), curbuf->b_p_spf);
+ else
+ {
+ if (bad)
+ fprintf(fd, "/!%.*s\n", len, word);
+ else
+ fprintf(fd, "%.*s\n", len, word);
+ fclose(fd);
+
+ /* Update the .add.spl file. */
+ mkspell(1, &curbuf->b_p_spf, FALSE, TRUE, FALSE);
+
+ /* If the .add file is edited somewhere, reload it. */
+ if (buf != NULL)
+ buf_reload(buf);
+ }
+ }
+ }
}
-#endif /* FEAT_MBYTE */
+/*
+ * Initialize 'spellfile' for the current buffer.
+ */
+ static void
+init_spellfile()
+{
+ char_u buf[MAXPATHL];
+ int l;
+ slang_T *sl;
+ char_u *rtp;
+
+ if (*curbuf->b_p_spl != NUL && curbuf->b_langp.ga_len > 0)
+ {
+ /* Loop over all entries in 'runtimepath'. */
+ rtp = p_rtp;
+ while (*rtp != NUL)
+ {
+ /* Copy the path from 'runtimepath' to buf[]. */
+ copy_option_part(&rtp, buf, MAXPATHL, ",");
+ if (filewritable(buf) == 2)
+ {
+ sl = LANGP_ENTRY(curbuf->b_langp, 0)->lp_slang;
+ l = STRLEN(buf);
+ vim_snprintf((char *)buf + l, MAXPATHL - l,
+ "/spell/%.2s.%s.add",
+ sl->sl_name,
+ strstr((char *)gettail(sl->sl_fname), ".ascii.") != NULL
+ ? (char_u *)"ascii" : spell_enc());
+ set_option_value((char_u *)"spellfile", 0L, buf, OPT_LOCAL);
+ break;
+ }
+ }
+ }
+}
/*
@@ -2937,7 +3270,6 @@ init_spell_chartab()
}
}
-#if defined(FEAT_MBYTE) || defined(PROTO)
static char *e_affform = N_("E761: Format error in affix file FOL, LOW or UPP");
static char *e_affrange = N_("E762: Character in FOL, LOW or UPP is out of range");
@@ -3016,7 +3348,6 @@ set_spell_chartab(fol, low, upp)
return set_spell_finish(&new_st);
}
-#endif
/*
* Set the spell character tables from strings in the .spl file.
@@ -3082,7 +3413,6 @@ set_spell_finish(new_st)
return OK;
}
-#if defined(FEAT_MBYTE) || defined(PROTO)
/*
* Write the current tables into the .spl file.
* This makes sure the same characters are recognized as word characters when
@@ -3107,13 +3437,17 @@ write_spell_chartab(fd)
flags |= SPELL_ISUPPER;
fputc(flags, fd); /* <charflags> */
- len += mb_char2bytes(spelltab.st_fold[i], charbuf + len);
+#ifdef FEAT_MBYTE
+ if (has_mbyte)
+ len += mb_char2bytes(spelltab.st_fold[i], charbuf + len);
+ else
+#endif
+ charbuf[len++] = spelltab.st_fold[i];
}
put_bytes(fd, (long_u)len, 2); /* <fcharlen> */
fwrite(charbuf, (size_t)len, (size_t)1, fd); /* <fchars> */
}
-#endif
/*
* Return TRUE if "c" is an upper-case character for spelling.