blob: 4b4b0bba80139a0c9e566afb93995ffd37e284ce [file] [log] [blame]
Bram Moolenaar071d4272004-06-13 20:20:40 +00001/* vi:set ts=8 sts=4 sw=4:
2 *
3 * VIM - Vi IMproved by Bram Moolenaar
4 *
5 * Do ":help uganda" in Vim to read copying and usage conditions.
6 * Do ":help credits" in Vim to see a list of people who contributed.
7 * See README.txt for an overview of the Vim source code.
8 */
9
10/*
11 * syntax.c: code for syntax highlighting
12 */
13
14#include "vim.h"
15
16/*
17 * Structure that stores information about a highlight group.
18 * The ID of a highlight group is also called group ID. It is the index in
19 * the highlight_ga array PLUS ONE.
20 */
21struct hl_group
22{
23 char_u *sg_name; /* highlight group name */
24 char_u *sg_name_u; /* uppercase of sg_name */
25/* for normal terminals */
26 int sg_term; /* "term=" highlighting attributes */
27 char_u *sg_start; /* terminal string for start highl */
28 char_u *sg_stop; /* terminal string for stop highl */
29 int sg_term_attr; /* Screen attr for term mode */
30/* for color terminals */
31 int sg_cterm; /* "cterm=" highlighting attr */
32 int sg_cterm_bold; /* bold attr was set for light color */
33 int sg_cterm_fg; /* terminal fg color number + 1 */
34 int sg_cterm_bg; /* terminal bg color number + 1 */
35 int sg_cterm_attr; /* Screen attr for color term mode */
36#ifdef FEAT_GUI
37/* for when using the GUI */
38 int sg_gui; /* "gui=" highlighting attributes */
39 guicolor_T sg_gui_fg; /* GUI foreground color handle */
40 char_u *sg_gui_fg_name;/* GUI foreground color name */
41 guicolor_T sg_gui_bg; /* GUI background color handle */
42 char_u *sg_gui_bg_name;/* GUI background color name */
Bram Moolenaare2cc9702005-03-15 22:43:58 +000043 guicolor_T sg_gui_sp; /* GUI special color handle */
44 char_u *sg_gui_sp_name;/* GUI special color name */
Bram Moolenaar071d4272004-06-13 20:20:40 +000045 GuiFont sg_font; /* GUI font handle */
46#ifdef FEAT_XFONTSET
47 GuiFontset sg_fontset; /* GUI fontset handle */
48#endif
49 char_u *sg_font_name; /* GUI font or fontset name */
50 int sg_gui_attr; /* Screen attr for GUI mode */
51#endif
52 int sg_link; /* link to this highlight group ID */
53 int sg_set; /* combination of SG_* flags */
Bram Moolenaar661b1822005-07-28 22:36:45 +000054#ifdef FEAT_EVAL
55 scid_T sg_scriptID; /* script in which the group was last set */
56#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +000057};
58
59#define SG_TERM 1 /* term has been set */
60#define SG_CTERM 2 /* cterm has been set */
61#define SG_GUI 4 /* gui has been set */
62#define SG_LINK 8 /* link has been set */
63
64static garray_T highlight_ga; /* highlight groups for 'highlight' option */
65
66#define HL_TABLE() ((struct hl_group *)((highlight_ga.ga_data)))
67
68#ifdef FEAT_CMDL_COMPL
Bram Moolenaar4f688582007-07-24 12:34:30 +000069/* Flags to indicate an additional string for highlight name completion. */
70static int include_none = 0; /* when 1 include "None" */
71static int include_default = 0; /* when 1 include "default" */
72static int include_link = 0; /* when 2 include "link" and "clear" */
Bram Moolenaar071d4272004-06-13 20:20:40 +000073#endif
74
75/*
76 * The "term", "cterm" and "gui" arguments can be any combination of the
77 * following names, separated by commas (but no spaces!).
78 */
79static char *(hl_name_table[]) =
Bram Moolenaare2cc9702005-03-15 22:43:58 +000080 {"bold", "standout", "underline", "undercurl",
81 "italic", "reverse", "inverse", "NONE"};
Bram Moolenaar071d4272004-06-13 20:20:40 +000082static int hl_attr_table[] =
Bram Moolenaare2cc9702005-03-15 22:43:58 +000083 {HL_BOLD, HL_STANDOUT, HL_UNDERLINE, HL_UNDERCURL, HL_ITALIC, HL_INVERSE, HL_INVERSE, 0};
Bram Moolenaar071d4272004-06-13 20:20:40 +000084
85static int get_attr_entry __ARGS((garray_T *table, attrentry_T *aep));
86static void syn_unadd_group __ARGS((void));
87static void set_hl_attr __ARGS((int idx));
88static void highlight_list_one __ARGS((int id));
89static int highlight_list_arg __ARGS((int id, int didh, int type, int iarg, char_u *sarg, char *name));
90static int syn_add_group __ARGS((char_u *name));
91static int syn_list_header __ARGS((int did_header, int outlen, int id));
92static int hl_has_settings __ARGS((int idx, int check_link));
93static void highlight_clear __ARGS((int idx));
94
95#ifdef FEAT_GUI
96static void gui_do_one_color __ARGS((int idx, int do_menu, int do_tooltip));
97static int set_group_colors __ARGS((char_u *name, guicolor_T *fgp, guicolor_T *bgp, int do_menu, int use_norm, int do_tooltip));
98static guicolor_T color_name2handle __ARGS((char_u *name));
99static GuiFont font_name2handle __ARGS((char_u *name));
100# ifdef FEAT_XFONTSET
101static GuiFontset fontset_name2handle __ARGS((char_u *name, int fixed_width));
102# endif
103static void hl_do_font __ARGS((int idx, char_u *arg, int do_normal, int do_menu, int do_tooltip));
104#endif
105
106/*
107 * An attribute number is the index in attr_table plus ATTR_OFF.
108 */
109#define ATTR_OFF (HL_ALL + 1)
110
111#if defined(FEAT_SYN_HL) || defined(PROTO)
112
113#define SYN_NAMELEN 50 /* maximum length of a syntax name */
114
115/* different types of offsets that are possible */
116#define SPO_MS_OFF 0 /* match start offset */
117#define SPO_ME_OFF 1 /* match end offset */
118#define SPO_HS_OFF 2 /* highl. start offset */
119#define SPO_HE_OFF 3 /* highl. end offset */
120#define SPO_RS_OFF 4 /* region start offset */
121#define SPO_RE_OFF 5 /* region end offset */
122#define SPO_LC_OFF 6 /* leading context offset */
123#define SPO_COUNT 7
124
125static char *(spo_name_tab[SPO_COUNT]) =
126 {"ms=", "me=", "hs=", "he=", "rs=", "re=", "lc="};
127
128/*
129 * The patterns that are being searched for are stored in a syn_pattern.
130 * A match item consists of one pattern.
131 * A start/end item consists of n start patterns and m end patterns.
132 * A start/skip/end item consists of n start patterns, one skip pattern and m
133 * end patterns.
134 * For the latter two, the patterns are always consecutive: start-skip-end.
135 *
136 * A character offset can be given for the matched text (_m_start and _m_end)
137 * and for the actually highlighted text (_h_start and _h_end).
138 */
139typedef struct syn_pattern
140{
141 char sp_type; /* see SPTYPE_ defines below */
142 char sp_syncing; /* this item used for syncing */
143 short sp_flags; /* see HL_ defines below */
144 struct sp_syn sp_syn; /* struct passed to in_id_list() */
145 short sp_syn_match_id; /* highlight group ID of pattern */
146 char_u *sp_pattern; /* regexp to match, pattern */
147 regprog_T *sp_prog; /* regexp to match, program */
148 int sp_ic; /* ignore-case flag for sp_prog */
149 short sp_off_flags; /* see below */
150 int sp_offsets[SPO_COUNT]; /* offsets */
151 short *sp_cont_list; /* cont. group IDs, if non-zero */
152 short *sp_next_list; /* next group IDs, if non-zero */
153 int sp_sync_idx; /* sync item index (syncing only) */
154 int sp_line_id; /* ID of last line where tried */
155 int sp_startcol; /* next match in sp_line_id line */
156} synpat_T;
157
158/* The sp_off_flags are computed like this:
159 * offset from the start of the matched text: (1 << SPO_XX_OFF)
160 * offset from the end of the matched text: (1 << (SPO_XX_OFF + SPO_COUNT))
161 * When both are present, only one is used.
162 */
163
164#define SPTYPE_MATCH 1 /* match keyword with this group ID */
165#define SPTYPE_START 2 /* match a regexp, start of item */
166#define SPTYPE_END 3 /* match a regexp, end of item */
167#define SPTYPE_SKIP 4 /* match a regexp, skip within item */
168
169#define HL_CONTAINED 0x01 /* not used on toplevel */
170#define HL_TRANSP 0x02 /* has no highlighting */
171#define HL_ONELINE 0x04 /* match within one line only */
172#define HL_HAS_EOL 0x08 /* end pattern that matches with $ */
173#define HL_SYNC_HERE 0x10 /* sync point after this item (syncing only) */
174#define HL_SYNC_THERE 0x20 /* sync point at current line (syncing only) */
175#define HL_MATCH 0x40 /* use match ID instead of item ID */
176#define HL_SKIPNL 0x80 /* nextgroup can skip newlines */
177#define HL_SKIPWHITE 0x100 /* nextgroup can skip white space */
178#define HL_SKIPEMPTY 0x200 /* nextgroup can skip empty lines */
179#define HL_KEEPEND 0x400 /* end match always kept */
180#define HL_EXCLUDENL 0x800 /* exclude NL from match */
181#define HL_DISPLAY 0x1000 /* only used for displaying, not syncing */
182#define HL_FOLD 0x2000 /* define fold */
183#define HL_EXTEND 0x4000 /* ignore a keepend */
184/* These don't fit in a short, thus can't be used for syntax items, only for
185 * si_flags and bs_flags. */
186#define HL_MATCHCONT 0x8000 /* match continued from previous line */
187#define HL_TRANS_CONT 0x10000L /* transparent item without contains arg */
188
189#define SYN_ITEMS(buf) ((synpat_T *)((buf)->b_syn_patterns.ga_data))
190
191#define NONE_IDX -2 /* value of sp_sync_idx for "NONE" */
192
193/*
194 * Flags for b_syn_sync_flags:
195 */
196#define SF_CCOMMENT 0x01 /* sync on a C-style comment */
197#define SF_MATCH 0x02 /* sync by matching a pattern */
198
199#define SYN_STATE_P(ssp) ((bufstate_T *)((ssp)->ga_data))
200
Bram Moolenaar071d4272004-06-13 20:20:40 +0000201#define MAXKEYWLEN 80 /* maximum length of a keyword */
202
203/*
204 * The attributes of the syntax item that has been recognized.
205 */
206static int current_attr = 0; /* attr of current syntax word */
207#ifdef FEAT_EVAL
208static int current_id = 0; /* ID of current char for syn_get_id() */
209static int current_trans_id = 0; /* idem, transparancy removed */
210#endif
211
Bram Moolenaar217ad922005-03-20 22:37:15 +0000212typedef struct syn_cluster_S
Bram Moolenaar071d4272004-06-13 20:20:40 +0000213{
214 char_u *scl_name; /* syntax cluster name */
215 char_u *scl_name_u; /* uppercase of scl_name */
216 short *scl_list; /* IDs in this syntax cluster */
Bram Moolenaar217ad922005-03-20 22:37:15 +0000217} syn_cluster_T;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000218
219/*
220 * Methods of combining two clusters
221 */
222#define CLUSTER_REPLACE 1 /* replace first list with second */
223#define CLUSTER_ADD 2 /* add second list to first */
224#define CLUSTER_SUBTRACT 3 /* subtract second list from first */
225
Bram Moolenaar217ad922005-03-20 22:37:15 +0000226#define SYN_CLSTR(buf) ((syn_cluster_T *)((buf)->b_syn_clusters.ga_data))
Bram Moolenaar071d4272004-06-13 20:20:40 +0000227
228/*
229 * Syntax group IDs have different types:
230 * 0 - 9999 normal syntax groups
231 * 10000 - 14999 ALLBUT indicator (current_syn_inc_tag added)
232 * 15000 - 19999 TOP indicator (current_syn_inc_tag added)
233 * 20000 - 24999 CONTAINED indicator (current_syn_inc_tag added)
234 * >= 25000 cluster IDs (subtract SYNID_CLUSTER for the cluster ID)
235 */
236#define SYNID_ALLBUT 10000 /* syntax group ID for contains=ALLBUT */
237#define SYNID_TOP 15000 /* syntax group ID for contains=TOP */
238#define SYNID_CONTAINED 20000 /* syntax group ID for contains=CONTAINED */
239#define SYNID_CLUSTER 25000 /* first syntax group ID for clusters */
240
241/*
242 * Annoying Hack(TM): ":syn include" needs this pointer to pass to
243 * expand_filename(). Most of the other syntax commands don't need it, so
244 * instead of passing it to them, we stow it here.
245 */
246static char_u **syn_cmdlinep;
247
248/*
249 * Another Annoying Hack(TM): To prevent rules from other ":syn include"'d
250 * files from from leaking into ALLBUT lists, we assign a unique ID to the
251 * rules in each ":syn include"'d file.
252 */
253static int current_syn_inc_tag = 0;
254static int running_syn_inc_tag = 0;
255
256/*
Bram Moolenaardad6b692005-01-25 22:14:34 +0000257 * In a hashtable item "hi_key" points to "keyword" in a keyentry.
258 * This avoids adding a pointer to the hashtable item.
259 * KE2HIKEY() converts a var pointer to a hashitem key pointer.
260 * HIKEY2KE() converts a hashitem key pointer to a var pointer.
261 * HI2KE() converts a hashitem pointer to a var pointer.
262 */
263static keyentry_T dumkey;
264#define KE2HIKEY(kp) ((kp)->keyword)
265#define HIKEY2KE(p) ((keyentry_T *)((p) - (dumkey.keyword - (char_u *)&dumkey)))
266#define HI2KE(hi) HIKEY2KE((hi)->hi_key)
267
268/*
Bram Moolenaar071d4272004-06-13 20:20:40 +0000269 * To reduce the time spent in keepend(), remember at which level in the state
270 * stack the first item with "keepend" is present. When "-1", there is no
271 * "keepend" on the stack.
272 */
273static int keepend_level = -1;
274
275/*
276 * For the current state we need to remember more than just the idx.
277 * When si_m_endpos.lnum is 0, the items other than si_idx are unknown.
278 * (The end positions have the column number of the next char)
279 */
280typedef struct state_item
281{
Bram Moolenaar3a36cf72007-08-21 15:29:56 +0000282 int si_idx; /* index of syntax pattern or
283 KEYWORD_IDX */
Bram Moolenaar071d4272004-06-13 20:20:40 +0000284 int si_id; /* highlight group ID for keywords */
285 int si_trans_id; /* idem, transparancy removed */
286 int si_m_lnum; /* lnum of the match */
287 int si_m_startcol; /* starting column of the match */
288 lpos_T si_m_endpos; /* just after end posn of the match */
289 lpos_T si_h_startpos; /* start position of the highlighting */
290 lpos_T si_h_endpos; /* end position of the highlighting */
291 lpos_T si_eoe_pos; /* end position of end pattern */
292 int si_end_idx; /* group ID for end pattern or zero */
293 int si_ends; /* if match ends before si_m_endpos */
294 int si_attr; /* attributes in this state */
295 long si_flags; /* HL_HAS_EOL flag in this state, and
296 * HL_SKIP* for si_next_list */
297 short *si_cont_list; /* list of contained groups */
298 short *si_next_list; /* nextgroup IDs after this item ends */
299 reg_extmatch_T *si_extmatch; /* \z(...\) matches from start
300 * pattern */
301} stateitem_T;
302
303#define KEYWORD_IDX -1 /* value of si_idx for keywords */
304#define ID_LIST_ALL (short *)-1 /* valid of si_cont_list for containing all
305 but contained groups */
306
307/*
Bram Moolenaar6ac54292005-02-02 23:07:25 +0000308 * Struct to reduce the number of arguments to get_syn_options(), it's used
309 * very often.
310 */
311typedef struct
312{
Bram Moolenaarb8017e72007-05-10 18:59:07 +0000313 int flags; /* flags for contained and transparent */
Bram Moolenaar6ac54292005-02-02 23:07:25 +0000314 int keyword; /* TRUE for ":syn keyword" */
315 int *sync_idx; /* syntax item for "grouphere" argument, NULL
316 if not allowed */
317 char has_cont_list; /* TRUE if "cont_list" can be used */
318 short *cont_list; /* group IDs for "contains" argument */
319 short *cont_in_list; /* group IDs for "containedin" argument */
320 short *next_list; /* group IDs for "nextgroup" argument */
321} syn_opt_arg_T;
322
323/*
Bram Moolenaar071d4272004-06-13 20:20:40 +0000324 * The next possible match in the current line for any pattern is remembered,
325 * to avoid having to try for a match in each column.
326 * If next_match_idx == -1, not tried (in this line) yet.
327 * If next_match_col == MAXCOL, no match found in this line.
328 * (All end positions have the column of the char after the end)
329 */
330static int next_match_col; /* column for start of next match */
331static lpos_T next_match_m_endpos; /* position for end of next match */
332static lpos_T next_match_h_startpos; /* pos. for highl. start of next match */
333static lpos_T next_match_h_endpos; /* pos. for highl. end of next match */
334static int next_match_idx; /* index of matched item */
335static long next_match_flags; /* flags for next match */
336static lpos_T next_match_eos_pos; /* end of start pattn (start region) */
337static lpos_T next_match_eoe_pos; /* pos. for end of end pattern */
338static int next_match_end_idx; /* ID of group for end pattn or zero */
339static reg_extmatch_T *next_match_extmatch = NULL;
340
341/*
342 * A state stack is an array of integers or stateitem_T, stored in a
343 * garray_T. A state stack is invalid if it's itemsize entry is zero.
344 */
345#define INVALID_STATE(ssp) ((ssp)->ga_itemsize == 0)
346#define VALID_STATE(ssp) ((ssp)->ga_itemsize != 0)
347
348/*
349 * The current state (within the line) of the recognition engine.
350 * When current_state.ga_itemsize is 0 the current state is invalid.
351 */
352static win_T *syn_win; /* current window for highlighting */
353static buf_T *syn_buf; /* current buffer for highlighting */
354static linenr_T current_lnum = 0; /* lnum of current state */
355static colnr_T current_col = 0; /* column of current state */
356static int current_state_stored = 0; /* TRUE if stored current state
357 * after setting current_finished */
358static int current_finished = 0; /* current line has been finished */
359static garray_T current_state /* current stack of state_items */
360 = {0, 0, 0, 0, NULL};
361static short *current_next_list = NULL; /* when non-zero, nextgroup list */
362static int current_next_flags = 0; /* flags for current_next_list */
363static int current_line_id = 0; /* unique number for current line */
364
365#define CUR_STATE(idx) ((stateitem_T *)(current_state.ga_data))[idx]
366
367static void syn_sync __ARGS((win_T *wp, linenr_T lnum, synstate_T *last_valid));
368static int syn_match_linecont __ARGS((linenr_T lnum));
369static void syn_start_line __ARGS((void));
370static void syn_update_ends __ARGS((int startofline));
371static void syn_stack_alloc __ARGS((void));
372static int syn_stack_cleanup __ARGS((void));
373static void syn_stack_free_entry __ARGS((buf_T *buf, synstate_T *p));
374static synstate_T *syn_stack_find_entry __ARGS((linenr_T lnum));
Bram Moolenaardbe31752008-01-13 16:40:19 +0000375static synstate_T *store_current_state __ARGS((void));
Bram Moolenaar071d4272004-06-13 20:20:40 +0000376static void load_current_state __ARGS((synstate_T *from));
377static void invalidate_current_state __ARGS((void));
378static int syn_stack_equal __ARGS((synstate_T *sp));
379static void validate_current_state __ARGS((void));
380static int syn_finish_line __ARGS((int syncing));
Bram Moolenaar56cefaf2008-01-12 15:47:10 +0000381static int syn_current_attr __ARGS((int syncing, int displaying, int *can_spell, int keep_state));
Bram Moolenaar071d4272004-06-13 20:20:40 +0000382static int did_match_already __ARGS((int idx, garray_T *gap));
383static stateitem_T *push_next_match __ARGS((stateitem_T *cur_si));
384static void check_state_ends __ARGS((void));
385static void update_si_attr __ARGS((int idx));
386static void check_keepend __ARGS((void));
387static void update_si_end __ARGS((stateitem_T *sip, int startcol, int force));
388static short *copy_id_list __ARGS((short *list));
389static int in_id_list __ARGS((stateitem_T *item, short *cont_list, struct sp_syn *ssp, int contained));
390static int push_current_state __ARGS((int idx));
391static void pop_current_state __ARGS((void));
392
393static void find_endpos __ARGS((int idx, lpos_T *startpos, lpos_T *m_endpos, lpos_T *hl_endpos, long *flagsp, lpos_T *end_endpos, int *end_idx, reg_extmatch_T *start_ext));
394static void clear_syn_state __ARGS((synstate_T *p));
395static void clear_current_state __ARGS((void));
396
397static void limit_pos __ARGS((lpos_T *pos, lpos_T *limit));
398static void limit_pos_zero __ARGS((lpos_T *pos, lpos_T *limit));
399static void syn_add_end_off __ARGS((lpos_T *result, regmmatch_T *regmatch, synpat_T *spp, int idx, int extra));
400static void syn_add_start_off __ARGS((lpos_T *result, regmmatch_T *regmatch, synpat_T *spp, int idx, int extra));
401static char_u *syn_getcurline __ARGS((void));
402static int syn_regexec __ARGS((regmmatch_T *rmp, linenr_T lnum, colnr_T col));
403static int check_keyword_id __ARGS((char_u *line, int startcol, int *endcol, long *flags, short **next_list, stateitem_T *cur_si));
404static void syn_cmd_case __ARGS((exarg_T *eap, int syncing));
Bram Moolenaarce0842a2005-07-18 21:58:11 +0000405static void syn_cmd_spell __ARGS((exarg_T *eap, int syncing));
Bram Moolenaar071d4272004-06-13 20:20:40 +0000406static void syntax_sync_clear __ARGS((void));
407static void syn_remove_pattern __ARGS((buf_T *buf, int idx));
408static void syn_clear_pattern __ARGS((buf_T *buf, int i));
409static void syn_clear_cluster __ARGS((buf_T *buf, int i));
410static void syn_cmd_clear __ARGS((exarg_T *eap, int syncing));
411static void syn_clear_one __ARGS((int id, int syncing));
412static void syn_cmd_on __ARGS((exarg_T *eap, int syncing));
413static void syn_cmd_enable __ARGS((exarg_T *eap, int syncing));
414static void syn_cmd_reset __ARGS((exarg_T *eap, int syncing));
415static void syn_cmd_manual __ARGS((exarg_T *eap, int syncing));
416static void syn_cmd_off __ARGS((exarg_T *eap, int syncing));
417static void syn_cmd_onoff __ARGS((exarg_T *eap, char *name));
418static void syn_cmd_list __ARGS((exarg_T *eap, int syncing));
419static void syn_lines_msg __ARGS((void));
420static void syn_match_msg __ARGS((void));
421static void syn_list_one __ARGS((int id, int syncing, int link_only));
422static void syn_list_cluster __ARGS((int id));
423static void put_id_list __ARGS((char_u *name, short *list, int attr));
424static void put_pattern __ARGS((char *s, int c, synpat_T *spp, int attr));
Bram Moolenaardad6b692005-01-25 22:14:34 +0000425static int syn_list_keywords __ARGS((int id, hashtab_T *ht, int did_header, int attr));
426static void syn_clear_keyword __ARGS((int id, hashtab_T *ht));
427static void clear_keywtab __ARGS((hashtab_T *ht));
Bram Moolenaar071d4272004-06-13 20:20:40 +0000428static void add_keyword __ARGS((char_u *name, int id, int flags, short *cont_in_list, short *next_list));
Bram Moolenaar071d4272004-06-13 20:20:40 +0000429static char_u *get_group_name __ARGS((char_u *arg, char_u **name_end));
Bram Moolenaar6ac54292005-02-02 23:07:25 +0000430static char_u *get_syn_options __ARGS((char_u *arg, syn_opt_arg_T *opt));
Bram Moolenaar071d4272004-06-13 20:20:40 +0000431static void syn_cmd_include __ARGS((exarg_T *eap, int syncing));
432static void syn_cmd_keyword __ARGS((exarg_T *eap, int syncing));
433static void syn_cmd_match __ARGS((exarg_T *eap, int syncing));
434static void syn_cmd_region __ARGS((exarg_T *eap, int syncing));
435#ifdef __BORLANDC__
436static int _RTLENTRYF syn_compare_stub __ARGS((const void *v1, const void *v2));
437#else
438static int syn_compare_stub __ARGS((const void *v1, const void *v2));
439#endif
440static void syn_cmd_cluster __ARGS((exarg_T *eap, int syncing));
441static int syn_scl_name2id __ARGS((char_u *name));
442static int syn_scl_namen2id __ARGS((char_u *linep, int len));
443static int syn_check_cluster __ARGS((char_u *pp, int len));
444static int syn_add_cluster __ARGS((char_u *name));
445static void init_syn_patterns __ARGS((void));
446static char_u *get_syn_pattern __ARGS((char_u *arg, synpat_T *ci));
447static void syn_cmd_sync __ARGS((exarg_T *eap, int syncing));
448static int get_id_list __ARGS((char_u **arg, int keylen, short **list));
449static void syn_combine_list __ARGS((short **clstr1, short **clstr2, int list_op));
450static void syn_incl_toplevel __ARGS((int id, int *flagsp));
451
452/*
453 * Start the syntax recognition for a line. This function is normally called
454 * from the screen updating, once for each displayed line.
455 * The buffer is remembered in syn_buf, because get_syntax_attr() doesn't get
456 * it. Careful: curbuf and curwin are likely to point to another buffer and
457 * window.
458 */
459 void
460syntax_start(wp, lnum)
461 win_T *wp;
462 linenr_T lnum;
463{
464 synstate_T *p;
465 synstate_T *last_valid = NULL;
466 synstate_T *last_min_valid = NULL;
Bram Moolenaardbe31752008-01-13 16:40:19 +0000467 synstate_T *sp, *prev = NULL;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000468 linenr_T parsed_lnum;
469 linenr_T first_stored;
470 int dist;
Bram Moolenaar3fdfa4a2004-10-07 21:02:47 +0000471 static int changedtick = 0; /* remember the last change ID */
Bram Moolenaar071d4272004-06-13 20:20:40 +0000472
Bram Moolenaar071d4272004-06-13 20:20:40 +0000473 /*
474 * After switching buffers, invalidate current_state.
Bram Moolenaar3fdfa4a2004-10-07 21:02:47 +0000475 * Also do this when a change was made, the current state may be invalid
476 * then.
Bram Moolenaar071d4272004-06-13 20:20:40 +0000477 */
Bram Moolenaar3fdfa4a2004-10-07 21:02:47 +0000478 if (syn_buf != wp->w_buffer || changedtick != syn_buf->b_changedtick)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000479 {
480 invalidate_current_state();
481 syn_buf = wp->w_buffer;
482 }
Bram Moolenaar3fdfa4a2004-10-07 21:02:47 +0000483 changedtick = syn_buf->b_changedtick;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000484 syn_win = wp;
485
486 /*
487 * Allocate syntax stack when needed.
488 */
489 syn_stack_alloc();
490 if (syn_buf->b_sst_array == NULL)
Bram Moolenaar3b56eb32005-07-11 22:40:32 +0000491 return; /* out of memory */
Bram Moolenaar071d4272004-06-13 20:20:40 +0000492 syn_buf->b_sst_lasttick = display_tick;
493
494 /*
495 * If the state of the end of the previous line is useful, store it.
496 */
497 if (VALID_STATE(&current_state)
498 && current_lnum < lnum
499 && current_lnum < syn_buf->b_ml.ml_line_count)
500 {
501 (void)syn_finish_line(FALSE);
502 if (!current_state_stored)
503 {
504 ++current_lnum;
Bram Moolenaardbe31752008-01-13 16:40:19 +0000505 (void)store_current_state();
Bram Moolenaar071d4272004-06-13 20:20:40 +0000506 }
507
508 /*
509 * If the current_lnum is now the same as "lnum", keep the current
510 * state (this happens very often!). Otherwise invalidate
511 * current_state and figure it out below.
512 */
513 if (current_lnum != lnum)
514 invalidate_current_state();
515 }
516 else
517 invalidate_current_state();
518
519 /*
520 * Try to synchronize from a saved state in b_sst_array[].
521 * Only do this if lnum is not before and not to far beyond a saved state.
522 */
523 if (INVALID_STATE(&current_state) && syn_buf->b_sst_array != NULL)
524 {
525 /* Find last valid saved state before start_lnum. */
526 for (p = syn_buf->b_sst_first; p != NULL; p = p->sst_next)
527 {
528 if (p->sst_lnum > lnum)
529 break;
530 if (p->sst_lnum <= lnum && p->sst_change_lnum == 0)
531 {
532 last_valid = p;
533 if (p->sst_lnum >= lnum - syn_buf->b_syn_sync_minlines)
534 last_min_valid = p;
535 }
536 }
537 if (last_min_valid != NULL)
538 load_current_state(last_min_valid);
539 }
540
541 /*
542 * If "lnum" is before or far beyond a line with a saved state, need to
543 * re-synchronize.
544 */
545 if (INVALID_STATE(&current_state))
546 {
547 syn_sync(wp, lnum, last_valid);
548 first_stored = current_lnum + syn_buf->b_syn_sync_minlines;
549 }
550 else
551 first_stored = current_lnum;
552
553 /*
554 * Advance from the sync point or saved state until the current line.
555 * Save some entries for syncing with later on.
556 */
Bram Moolenaar910f66f2006-04-05 20:41:53 +0000557 if (syn_buf->b_sst_len <= Rows)
558 dist = 999999;
559 else
560 dist = syn_buf->b_ml.ml_line_count / (syn_buf->b_sst_len - Rows) + 1;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000561 while (current_lnum < lnum)
562 {
563 syn_start_line();
564 (void)syn_finish_line(FALSE);
565 ++current_lnum;
566
567 /* If we parsed at least "minlines" lines or started at a valid
568 * state, the current state is considered valid. */
569 if (current_lnum >= first_stored)
570 {
571 /* Check if the saved state entry is for the current line and is
572 * equal to the current state. If so, then validate all saved
573 * states that depended on a change before the parsed line. */
574 if (prev == NULL)
Bram Moolenaardbe31752008-01-13 16:40:19 +0000575 prev = syn_stack_find_entry(current_lnum - 1);
576 if (prev == NULL)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000577 sp = syn_buf->b_sst_first;
578 else
Bram Moolenaardbe31752008-01-13 16:40:19 +0000579 sp = prev;
580 while (sp != NULL && sp->sst_lnum < current_lnum)
581 sp = sp->sst_next;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000582 if (sp != NULL
583 && sp->sst_lnum == current_lnum
584 && syn_stack_equal(sp))
585 {
586 parsed_lnum = current_lnum;
587 prev = sp;
588 while (sp != NULL && sp->sst_change_lnum <= parsed_lnum)
589 {
590 if (sp->sst_lnum <= lnum)
591 /* valid state before desired line, use this one */
592 prev = sp;
593 else if (sp->sst_change_lnum == 0)
594 /* past saved states depending on change, break here. */
595 break;
596 sp->sst_change_lnum = 0;
597 sp = sp->sst_next;
598 }
599 load_current_state(prev);
600 }
601 /* Store the state at this line when it's the first one, the line
602 * where we start parsing, or some distance from the previously
603 * saved state. But only when parsed at least 'minlines'. */
604 else if (prev == NULL
605 || current_lnum == lnum
606 || current_lnum >= prev->sst_lnum + dist)
Bram Moolenaardbe31752008-01-13 16:40:19 +0000607 prev = store_current_state();
Bram Moolenaar071d4272004-06-13 20:20:40 +0000608 }
609
610 /* This can take a long time: break when CTRL-C pressed. The current
611 * state will be wrong then. */
612 line_breakcheck();
613 if (got_int)
614 {
615 current_lnum = lnum;
616 break;
617 }
618 }
619
620 syn_start_line();
Bram Moolenaar071d4272004-06-13 20:20:40 +0000621}
622
623/*
624 * We cannot simply discard growarrays full of state_items or buf_states; we
625 * have to manually release their extmatch pointers first.
626 */
627 static void
628clear_syn_state(p)
629 synstate_T *p;
630{
631 int i;
632 garray_T *gap;
633
634 if (p->sst_stacksize > SST_FIX_STATES)
635 {
636 gap = &(p->sst_union.sst_ga);
637 for (i = 0; i < gap->ga_len; i++)
638 unref_extmatch(SYN_STATE_P(gap)[i].bs_extmatch);
639 ga_clear(gap);
640 }
641 else
642 {
643 for (i = 0; i < p->sst_stacksize; i++)
644 unref_extmatch(p->sst_union.sst_stack[i].bs_extmatch);
645 }
646}
647
648/*
649 * Cleanup the current_state stack.
650 */
651 static void
652clear_current_state()
653{
654 int i;
655 stateitem_T *sip;
656
657 sip = (stateitem_T *)(current_state.ga_data);
658 for (i = 0; i < current_state.ga_len; i++)
659 unref_extmatch(sip[i].si_extmatch);
660 ga_clear(&current_state);
661}
662
663/*
664 * Try to find a synchronisation point for line "lnum".
665 *
666 * This sets current_lnum and the current state. One of three methods is
667 * used:
668 * 1. Search backwards for the end of a C-comment.
669 * 2. Search backwards for given sync patterns.
670 * 3. Simply start on a given number of lines above "lnum".
671 */
672 static void
673syn_sync(wp, start_lnum, last_valid)
674 win_T *wp;
675 linenr_T start_lnum;
676 synstate_T *last_valid;
677{
678 buf_T *curbuf_save;
679 win_T *curwin_save;
680 pos_T cursor_save;
681 int idx;
682 linenr_T lnum;
683 linenr_T end_lnum;
684 linenr_T break_lnum;
685 int had_sync_point;
686 stateitem_T *cur_si;
687 synpat_T *spp;
688 char_u *line;
689 int found_flags = 0;
690 int found_match_idx = 0;
691 linenr_T found_current_lnum = 0;
692 int found_current_col= 0;
693 lpos_T found_m_endpos;
Bram Moolenaar81366db2005-07-24 21:16:51 +0000694 colnr_T prev_current_col;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000695
696 /*
697 * Clear any current state that might be hanging around.
698 */
699 invalidate_current_state();
700
701 /*
702 * Start at least "minlines" back. Default starting point for parsing is
703 * there.
704 * Start further back, to avoid that scrolling backwards will result in
705 * resyncing for every line. Now it resyncs only one out of N lines,
706 * where N is minlines * 1.5, or minlines * 2 if minlines is small.
707 * Watch out for overflow when minlines is MAXLNUM.
708 */
709 if (syn_buf->b_syn_sync_minlines > start_lnum)
710 start_lnum = 1;
711 else
712 {
713 if (syn_buf->b_syn_sync_minlines == 1)
714 lnum = 1;
715 else if (syn_buf->b_syn_sync_minlines < 10)
716 lnum = syn_buf->b_syn_sync_minlines * 2;
717 else
718 lnum = syn_buf->b_syn_sync_minlines * 3 / 2;
719 if (syn_buf->b_syn_sync_maxlines != 0
720 && lnum > syn_buf->b_syn_sync_maxlines)
721 lnum = syn_buf->b_syn_sync_maxlines;
722 if (lnum >= start_lnum)
723 start_lnum = 1;
724 else
725 start_lnum -= lnum;
726 }
727 current_lnum = start_lnum;
728
729 /*
730 * 1. Search backwards for the end of a C-style comment.
731 */
732 if (syn_buf->b_syn_sync_flags & SF_CCOMMENT)
733 {
734 /* Need to make syn_buf the current buffer for a moment, to be able to
735 * use find_start_comment(). */
736 curwin_save = curwin;
737 curwin = wp;
738 curbuf_save = curbuf;
739 curbuf = syn_buf;
740
741 /*
742 * Skip lines that end in a backslash.
743 */
744 for ( ; start_lnum > 1; --start_lnum)
745 {
746 line = ml_get(start_lnum - 1);
747 if (*line == NUL || *(line + STRLEN(line) - 1) != '\\')
748 break;
749 }
750 current_lnum = start_lnum;
751
752 /* set cursor to start of search */
753 cursor_save = wp->w_cursor;
754 wp->w_cursor.lnum = start_lnum;
755 wp->w_cursor.col = 0;
756
757 /*
758 * If the line is inside a comment, need to find the syntax item that
759 * defines the comment.
760 * Restrict the search for the end of a comment to b_syn_sync_maxlines.
761 */
762 if (find_start_comment((int)syn_buf->b_syn_sync_maxlines) != NULL)
763 {
764 for (idx = syn_buf->b_syn_patterns.ga_len; --idx >= 0; )
765 if (SYN_ITEMS(syn_buf)[idx].sp_syn.id == syn_buf->b_syn_sync_id
766 && SYN_ITEMS(syn_buf)[idx].sp_type == SPTYPE_START)
767 {
768 validate_current_state();
769 if (push_current_state(idx) == OK)
770 update_si_attr(current_state.ga_len - 1);
771 break;
772 }
773 }
774
775 /* restore cursor and buffer */
776 wp->w_cursor = cursor_save;
777 curwin = curwin_save;
778 curbuf = curbuf_save;
779 }
780
781 /*
782 * 2. Search backwards for given sync patterns.
783 */
784 else if (syn_buf->b_syn_sync_flags & SF_MATCH)
785 {
786 if (syn_buf->b_syn_sync_maxlines != 0
787 && start_lnum > syn_buf->b_syn_sync_maxlines)
788 break_lnum = start_lnum - syn_buf->b_syn_sync_maxlines;
789 else
790 break_lnum = 0;
791
Bram Moolenaarfd2ac762006-03-01 22:09:21 +0000792 found_m_endpos.lnum = 0;
793 found_m_endpos.col = 0;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000794 end_lnum = start_lnum;
795 lnum = start_lnum;
796 while (--lnum > break_lnum)
797 {
798 /* This can take a long time: break when CTRL-C pressed. */
799 line_breakcheck();
800 if (got_int)
801 {
802 invalidate_current_state();
803 current_lnum = start_lnum;
804 break;
805 }
806
807 /* Check if we have run into a valid saved state stack now. */
808 if (last_valid != NULL && lnum == last_valid->sst_lnum)
809 {
810 load_current_state(last_valid);
811 break;
812 }
813
814 /*
815 * Check if the previous line has the line-continuation pattern.
816 */
817 if (lnum > 1 && syn_match_linecont(lnum - 1))
818 continue;
819
820 /*
821 * Start with nothing on the state stack
822 */
823 validate_current_state();
824
825 for (current_lnum = lnum; current_lnum < end_lnum; ++current_lnum)
826 {
827 syn_start_line();
828 for (;;)
829 {
830 had_sync_point = syn_finish_line(TRUE);
831 /*
832 * When a sync point has been found, remember where, and
833 * continue to look for another one, further on in the line.
834 */
835 if (had_sync_point && current_state.ga_len)
836 {
837 cur_si = &CUR_STATE(current_state.ga_len - 1);
838 if (cur_si->si_m_endpos.lnum > start_lnum)
839 {
840 /* ignore match that goes to after where started */
841 current_lnum = end_lnum;
842 break;
843 }
Bram Moolenaar3a36cf72007-08-21 15:29:56 +0000844 if (cur_si->si_idx < 0)
845 {
846 /* Cannot happen? */
847 found_flags = 0;
848 found_match_idx = KEYWORD_IDX;
849 }
850 else
851 {
852 spp = &(SYN_ITEMS(syn_buf)[cur_si->si_idx]);
853 found_flags = spp->sp_flags;
854 found_match_idx = spp->sp_sync_idx;
855 }
Bram Moolenaar071d4272004-06-13 20:20:40 +0000856 found_current_lnum = current_lnum;
857 found_current_col = current_col;
858 found_m_endpos = cur_si->si_m_endpos;
859 /*
860 * Continue after the match (be aware of a zero-length
861 * match).
862 */
863 if (found_m_endpos.lnum > current_lnum)
864 {
865 current_lnum = found_m_endpos.lnum;
866 current_col = found_m_endpos.col;
867 if (current_lnum >= end_lnum)
868 break;
869 }
870 else if (found_m_endpos.col > current_col)
871 current_col = found_m_endpos.col;
872 else
873 ++current_col;
874
875 /* syn_current_attr() will have skipped the check for
Bram Moolenaar81366db2005-07-24 21:16:51 +0000876 * an item that ends here, need to do that now. Be
877 * careful not to go past the NUL. */
878 prev_current_col = current_col;
879 if (syn_getcurline()[current_col] != NUL)
880 ++current_col;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000881 check_state_ends();
Bram Moolenaar81366db2005-07-24 21:16:51 +0000882 current_col = prev_current_col;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000883 }
884 else
885 break;
886 }
887 }
888
889 /*
890 * If a sync point was encountered, break here.
891 */
892 if (found_flags)
893 {
894 /*
895 * Put the item that was specified by the sync point on the
896 * state stack. If there was no item specified, make the
897 * state stack empty.
898 */
899 clear_current_state();
900 if (found_match_idx >= 0
901 && push_current_state(found_match_idx) == OK)
902 update_si_attr(current_state.ga_len - 1);
903
904 /*
905 * When using "grouphere", continue from the sync point
906 * match, until the end of the line. Parsing starts at
907 * the next line.
908 * For "groupthere" the parsing starts at start_lnum.
909 */
910 if (found_flags & HL_SYNC_HERE)
911 {
912 if (current_state.ga_len)
913 {
914 cur_si = &CUR_STATE(current_state.ga_len - 1);
915 cur_si->si_h_startpos.lnum = found_current_lnum;
916 cur_si->si_h_startpos.col = found_current_col;
917 update_si_end(cur_si, (int)current_col, TRUE);
918 check_keepend();
919 }
920 current_col = found_m_endpos.col;
921 current_lnum = found_m_endpos.lnum;
922 (void)syn_finish_line(FALSE);
923 ++current_lnum;
924 }
925 else
926 current_lnum = start_lnum;
927
928 break;
929 }
930
931 end_lnum = lnum;
932 invalidate_current_state();
933 }
934
935 /* Ran into start of the file or exceeded maximum number of lines */
936 if (lnum <= break_lnum)
937 {
938 invalidate_current_state();
939 current_lnum = break_lnum + 1;
940 }
941 }
942
943 validate_current_state();
944}
945
946/*
947 * Return TRUE if the line-continuation pattern matches in line "lnum".
948 */
949 static int
950syn_match_linecont(lnum)
951 linenr_T lnum;
952{
953 regmmatch_T regmatch;
954
955 if (syn_buf->b_syn_linecont_prog != NULL)
956 {
957 regmatch.rmm_ic = syn_buf->b_syn_linecont_ic;
958 regmatch.regprog = syn_buf->b_syn_linecont_prog;
959 return syn_regexec(&regmatch, lnum, (colnr_T)0);
960 }
961 return FALSE;
962}
963
964/*
965 * Prepare the current state for the start of a line.
966 */
967 static void
968syn_start_line()
969{
970 current_finished = FALSE;
971 current_col = 0;
972
973 /*
974 * Need to update the end of a start/skip/end that continues from the
975 * previous line and regions that have "keepend".
976 */
977 if (current_state.ga_len > 0)
978 syn_update_ends(TRUE);
979
980 next_match_idx = -1;
981 ++current_line_id;
982}
983
984/*
985 * Check for items in the stack that need their end updated.
986 * When "startofline" is TRUE the last item is always updated.
987 * When "startofline" is FALSE the item with "keepend" is forcefully updated.
988 */
989 static void
990syn_update_ends(startofline)
991 int startofline;
992{
993 stateitem_T *cur_si;
994 int i;
Bram Moolenaar7bd2cd82006-10-03 15:04:36 +0000995 int seen_keepend;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000996
997 if (startofline)
998 {
999 /* Check for a match carried over from a previous line with a
1000 * contained region. The match ends as soon as the region ends. */
1001 for (i = 0; i < current_state.ga_len; ++i)
1002 {
1003 cur_si = &CUR_STATE(i);
1004 if (cur_si->si_idx >= 0
1005 && (SYN_ITEMS(syn_buf)[cur_si->si_idx]).sp_type
1006 == SPTYPE_MATCH
1007 && cur_si->si_m_endpos.lnum < current_lnum)
1008 {
1009 cur_si->si_flags |= HL_MATCHCONT;
1010 cur_si->si_m_endpos.lnum = 0;
1011 cur_si->si_m_endpos.col = 0;
1012 cur_si->si_h_endpos = cur_si->si_m_endpos;
1013 cur_si->si_ends = TRUE;
1014 }
1015 }
1016 }
1017
1018 /*
1019 * Need to update the end of a start/skip/end that continues from the
1020 * previous line. And regions that have "keepend", because they may
Bram Moolenaar7bd2cd82006-10-03 15:04:36 +00001021 * influence contained items. If we've just removed "extend"
1022 * (startofline == 0) then we should update ends of normal regions
1023 * contained inside "keepend" because "extend" could have extended
1024 * these "keepend" regions as well as contained normal regions.
Bram Moolenaar071d4272004-06-13 20:20:40 +00001025 * Then check for items ending in column 0.
1026 */
1027 i = current_state.ga_len - 1;
1028 if (keepend_level >= 0)
1029 for ( ; i > keepend_level; --i)
1030 if (CUR_STATE(i).si_flags & HL_EXTEND)
1031 break;
Bram Moolenaar7bd2cd82006-10-03 15:04:36 +00001032
1033 seen_keepend = FALSE;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001034 for ( ; i < current_state.ga_len; ++i)
1035 {
1036 cur_si = &CUR_STATE(i);
1037 if ((cur_si->si_flags & HL_KEEPEND)
Bram Moolenaar7bd2cd82006-10-03 15:04:36 +00001038 || (seen_keepend && !startofline)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001039 || (i == current_state.ga_len - 1 && startofline))
1040 {
1041 cur_si->si_h_startpos.col = 0; /* start highl. in col 0 */
1042 cur_si->si_h_startpos.lnum = current_lnum;
1043
1044 if (!(cur_si->si_flags & HL_MATCHCONT))
1045 update_si_end(cur_si, (int)current_col, !startofline);
Bram Moolenaar7bd2cd82006-10-03 15:04:36 +00001046
1047 if (!startofline && (cur_si->si_flags & HL_KEEPEND))
1048 seen_keepend = TRUE;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001049 }
1050 }
1051 check_keepend();
1052 check_state_ends();
1053}
1054
1055/****************************************
1056 * Handling of the state stack cache.
1057 */
1058
1059/*
1060 * EXPLANATION OF THE SYNTAX STATE STACK CACHE
1061 *
1062 * To speed up syntax highlighting, the state stack for the start of some
1063 * lines is cached. These entries can be used to start parsing at that point.
1064 *
1065 * The stack is kept in b_sst_array[] for each buffer. There is a list of
1066 * valid entries. b_sst_first points to the first one, then follow sst_next.
1067 * The entries are sorted on line number. The first entry is often for line 2
1068 * (line 1 always starts with an empty stack).
1069 * There is also a list for free entries. This construction is used to avoid
1070 * having to allocate and free memory blocks too often.
1071 *
1072 * When making changes to the buffer, this is logged in b_mod_*. When calling
1073 * update_screen() to update the display, it will call
1074 * syn_stack_apply_changes() for each displayed buffer to adjust the cached
1075 * entries. The entries which are inside the changed area are removed,
1076 * because they must be recomputed. Entries below the changed have their line
1077 * number adjusted for deleted/inserted lines, and have their sst_change_lnum
1078 * set to indicate that a check must be made if the changed lines would change
1079 * the cached entry.
1080 *
1081 * When later displaying lines, an entry is stored for each line. Displayed
1082 * lines are likely to be displayed again, in which case the state at the
1083 * start of the line is needed.
1084 * For not displayed lines, an entry is stored for every so many lines. These
1085 * entries will be used e.g., when scrolling backwards. The distance between
1086 * entries depends on the number of lines in the buffer. For small buffers
1087 * the distance is fixed at SST_DIST, for large buffers there is a fixed
1088 * number of entries SST_MAX_ENTRIES, and the distance is computed.
1089 */
1090
1091/*
1092 * Free b_sst_array[] for buffer "buf".
1093 * Used when syntax items changed to force resyncing everywhere.
1094 */
1095 void
1096syn_stack_free_all(buf)
1097 buf_T *buf;
1098{
1099 synstate_T *p;
1100 win_T *wp;
1101
1102 if (buf->b_sst_array != NULL)
1103 {
1104 for (p = buf->b_sst_first; p != NULL; p = p->sst_next)
1105 clear_syn_state(p);
1106 vim_free(buf->b_sst_array);
1107 buf->b_sst_array = NULL;
1108 buf->b_sst_len = 0;
1109 }
1110#ifdef FEAT_FOLDING
1111 /* When using "syntax" fold method, must update all folds. */
1112 FOR_ALL_WINDOWS(wp)
1113 {
1114 if (wp->w_buffer == buf && foldmethodIsSyntax(wp))
1115 foldUpdateAll(wp);
1116 }
1117#endif
1118}
1119
1120/*
1121 * Allocate the syntax state stack for syn_buf when needed.
1122 * If the number of entries in b_sst_array[] is much too big or a bit too
1123 * small, reallocate it.
1124 * Also used to allocate b_sst_array[] for the first time.
1125 */
1126 static void
1127syn_stack_alloc()
1128{
1129 long len;
1130 synstate_T *to, *from;
1131 synstate_T *sstp;
1132
1133 len = syn_buf->b_ml.ml_line_count / SST_DIST + Rows * 2;
1134 if (len < SST_MIN_ENTRIES)
1135 len = SST_MIN_ENTRIES;
1136 else if (len > SST_MAX_ENTRIES)
1137 len = SST_MAX_ENTRIES;
1138 if (syn_buf->b_sst_len > len * 2 || syn_buf->b_sst_len < len)
1139 {
1140 /* Allocate 50% too much, to avoid reallocating too often. */
1141 len = syn_buf->b_ml.ml_line_count;
1142 len = (len + len / 2) / SST_DIST + Rows * 2;
1143 if (len < SST_MIN_ENTRIES)
1144 len = SST_MIN_ENTRIES;
1145 else if (len > SST_MAX_ENTRIES)
1146 len = SST_MAX_ENTRIES;
1147
1148 if (syn_buf->b_sst_array != NULL)
1149 {
1150 /* When shrinking the array, cleanup the existing stack.
1151 * Make sure that all valid entries fit in the new array. */
1152 while (syn_buf->b_sst_len - syn_buf->b_sst_freecount + 2 > len
1153 && syn_stack_cleanup())
1154 ;
1155 if (len < syn_buf->b_sst_len - syn_buf->b_sst_freecount + 2)
1156 len = syn_buf->b_sst_len - syn_buf->b_sst_freecount + 2;
1157 }
1158
1159 sstp = (synstate_T *)alloc_clear((unsigned)(len * sizeof(synstate_T)));
1160 if (sstp == NULL) /* out of memory! */
1161 return;
1162
1163 to = sstp - 1;
1164 if (syn_buf->b_sst_array != NULL)
1165 {
1166 /* Move the states from the old array to the new one. */
1167 for (from = syn_buf->b_sst_first; from != NULL;
1168 from = from->sst_next)
1169 {
1170 ++to;
1171 *to = *from;
1172 to->sst_next = to + 1;
1173 }
1174 }
1175 if (to != sstp - 1)
1176 {
1177 to->sst_next = NULL;
1178 syn_buf->b_sst_first = sstp;
1179 syn_buf->b_sst_freecount = len - (int)(to - sstp) - 1;
1180 }
1181 else
1182 {
1183 syn_buf->b_sst_first = NULL;
1184 syn_buf->b_sst_freecount = len;
1185 }
1186
1187 /* Create the list of free entries. */
1188 syn_buf->b_sst_firstfree = to + 1;
1189 while (++to < sstp + len)
1190 to->sst_next = to + 1;
1191 (sstp + len - 1)->sst_next = NULL;
1192
1193 vim_free(syn_buf->b_sst_array);
1194 syn_buf->b_sst_array = sstp;
1195 syn_buf->b_sst_len = len;
1196 }
1197}
1198
1199/*
1200 * Check for changes in a buffer to affect stored syntax states. Uses the
1201 * b_mod_* fields.
1202 * Called from update_screen(), before screen is being updated, once for each
1203 * displayed buffer.
1204 */
1205 void
1206syn_stack_apply_changes(buf)
1207 buf_T *buf;
1208{
1209 synstate_T *p, *prev, *np;
1210 linenr_T n;
1211
1212 if (buf->b_sst_array == NULL) /* nothing to do */
1213 return;
1214
1215 prev = NULL;
1216 for (p = buf->b_sst_first; p != NULL; )
1217 {
Bram Moolenaar1f8a5f02005-07-01 22:41:52 +00001218 if (p->sst_lnum + buf->b_syn_sync_linebreaks > buf->b_mod_top)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001219 {
1220 n = p->sst_lnum + buf->b_mod_xlines;
1221 if (n <= buf->b_mod_bot)
1222 {
1223 /* this state is inside the changed area, remove it */
1224 np = p->sst_next;
1225 if (prev == NULL)
1226 buf->b_sst_first = np;
1227 else
1228 prev->sst_next = np;
1229 syn_stack_free_entry(buf, p);
1230 p = np;
1231 continue;
1232 }
1233 /* This state is below the changed area. Remember the line
1234 * that needs to be parsed before this entry can be made valid
1235 * again. */
1236 if (p->sst_change_lnum != 0 && p->sst_change_lnum > buf->b_mod_top)
1237 {
1238 if (p->sst_change_lnum + buf->b_mod_xlines > buf->b_mod_top)
1239 p->sst_change_lnum += buf->b_mod_xlines;
1240 else
1241 p->sst_change_lnum = buf->b_mod_top;
1242 }
1243 if (p->sst_change_lnum == 0
1244 || p->sst_change_lnum < buf->b_mod_bot)
1245 p->sst_change_lnum = buf->b_mod_bot;
1246
1247 p->sst_lnum = n;
1248 }
1249 prev = p;
1250 p = p->sst_next;
1251 }
1252}
1253
1254/*
1255 * Reduce the number of entries in the state stack for syn_buf.
1256 * Returns TRUE if at least one entry was freed.
1257 */
1258 static int
1259syn_stack_cleanup()
1260{
1261 synstate_T *p, *prev;
1262 disptick_T tick;
1263 int above;
1264 int dist;
1265 int retval = FALSE;
1266
1267 if (syn_buf->b_sst_array == NULL || syn_buf->b_sst_first == NULL)
1268 return retval;
1269
1270 /* Compute normal distance between non-displayed entries. */
Bram Moolenaar910f66f2006-04-05 20:41:53 +00001271 if (syn_buf->b_sst_len <= Rows)
1272 dist = 999999;
1273 else
1274 dist = syn_buf->b_ml.ml_line_count / (syn_buf->b_sst_len - Rows) + 1;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001275
1276 /*
1277 * Go throught the list to find the "tick" for the oldest entry that can
1278 * be removed. Set "above" when the "tick" for the oldest entry is above
1279 * "b_sst_lasttick" (the display tick wraps around).
1280 */
1281 tick = syn_buf->b_sst_lasttick;
1282 above = FALSE;
1283 prev = syn_buf->b_sst_first;
1284 for (p = prev->sst_next; p != NULL; prev = p, p = p->sst_next)
1285 {
1286 if (prev->sst_lnum + dist > p->sst_lnum)
1287 {
1288 if (p->sst_tick > syn_buf->b_sst_lasttick)
1289 {
1290 if (!above || p->sst_tick < tick)
1291 tick = p->sst_tick;
1292 above = TRUE;
1293 }
1294 else if (!above && p->sst_tick < tick)
1295 tick = p->sst_tick;
1296 }
1297 }
1298
1299 /*
1300 * Go through the list to make the entries for the oldest tick at an
1301 * interval of several lines.
1302 */
1303 prev = syn_buf->b_sst_first;
1304 for (p = prev->sst_next; p != NULL; prev = p, p = p->sst_next)
1305 {
1306 if (p->sst_tick == tick && prev->sst_lnum + dist > p->sst_lnum)
1307 {
1308 /* Move this entry from used list to free list */
1309 prev->sst_next = p->sst_next;
1310 syn_stack_free_entry(syn_buf, p);
1311 p = prev;
1312 retval = TRUE;
1313 }
1314 }
1315 return retval;
1316}
1317
1318/*
1319 * Free the allocated memory for a syn_state item.
1320 * Move the entry into the free list.
1321 */
1322 static void
1323syn_stack_free_entry(buf, p)
1324 buf_T *buf;
1325 synstate_T *p;
1326{
1327 clear_syn_state(p);
1328 p->sst_next = buf->b_sst_firstfree;
1329 buf->b_sst_firstfree = p;
1330 ++buf->b_sst_freecount;
1331}
1332
1333/*
1334 * Find an entry in the list of state stacks at or before "lnum".
1335 * Returns NULL when there is no entry or the first entry is after "lnum".
1336 */
1337 static synstate_T *
1338syn_stack_find_entry(lnum)
1339 linenr_T lnum;
1340{
1341 synstate_T *p, *prev;
1342
1343 prev = NULL;
1344 for (p = syn_buf->b_sst_first; p != NULL; prev = p, p = p->sst_next)
1345 {
1346 if (p->sst_lnum == lnum)
1347 return p;
1348 if (p->sst_lnum > lnum)
1349 break;
1350 }
1351 return prev;
1352}
1353
1354/*
1355 * Try saving the current state in b_sst_array[].
1356 * The current state must be valid for the start of the current_lnum line!
1357 */
1358 static synstate_T *
Bram Moolenaardbe31752008-01-13 16:40:19 +00001359store_current_state()
Bram Moolenaar071d4272004-06-13 20:20:40 +00001360{
1361 int i;
1362 synstate_T *p;
1363 bufstate_T *bp;
1364 stateitem_T *cur_si;
Bram Moolenaardbe31752008-01-13 16:40:19 +00001365 synstate_T *sp = syn_stack_find_entry(current_lnum);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001366
1367 /*
1368 * If the current state contains a start or end pattern that continues
1369 * from the previous line, we can't use it. Don't store it then.
1370 */
1371 for (i = current_state.ga_len - 1; i >= 0; --i)
1372 {
1373 cur_si = &CUR_STATE(i);
1374 if (cur_si->si_h_startpos.lnum >= current_lnum
1375 || cur_si->si_m_endpos.lnum >= current_lnum
1376 || cur_si->si_h_endpos.lnum >= current_lnum
1377 || (cur_si->si_end_idx
1378 && cur_si->si_eoe_pos.lnum >= current_lnum))
1379 break;
1380 }
1381 if (i >= 0)
1382 {
1383 if (sp != NULL)
1384 {
1385 /* find "sp" in the list and remove it */
1386 if (syn_buf->b_sst_first == sp)
1387 /* it's the first entry */
1388 syn_buf->b_sst_first = sp->sst_next;
1389 else
1390 {
1391 /* find the entry just before this one to adjust sst_next */
1392 for (p = syn_buf->b_sst_first; p != NULL; p = p->sst_next)
1393 if (p->sst_next == sp)
1394 break;
Bram Moolenaareb3593b2006-04-22 22:33:57 +00001395 if (p != NULL) /* just in case */
1396 p->sst_next = sp->sst_next;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001397 }
1398 syn_stack_free_entry(syn_buf, sp);
1399 sp = NULL;
1400 }
1401 }
1402 else if (sp == NULL || sp->sst_lnum != current_lnum)
1403 {
1404 /*
1405 * Add a new entry
1406 */
1407 /* If no free items, cleanup the array first. */
1408 if (syn_buf->b_sst_freecount == 0)
1409 {
1410 (void)syn_stack_cleanup();
1411 /* "sp" may have been moved to the freelist now */
1412 sp = syn_stack_find_entry(current_lnum);
1413 }
1414 /* Still no free items? Must be a strange problem... */
1415 if (syn_buf->b_sst_freecount == 0)
1416 sp = NULL;
1417 else
1418 {
1419 /* Take the first item from the free list and put it in the used
1420 * list, after *sp */
1421 p = syn_buf->b_sst_firstfree;
1422 syn_buf->b_sst_firstfree = p->sst_next;
1423 --syn_buf->b_sst_freecount;
1424 if (sp == NULL)
1425 {
1426 /* Insert in front of the list */
1427 p->sst_next = syn_buf->b_sst_first;
1428 syn_buf->b_sst_first = p;
1429 }
1430 else
1431 {
1432 /* insert in list after *sp */
1433 p->sst_next = sp->sst_next;
1434 sp->sst_next = p;
1435 }
1436 sp = p;
1437 sp->sst_stacksize = 0;
1438 sp->sst_lnum = current_lnum;
1439 }
1440 }
1441 if (sp != NULL)
1442 {
1443 /* When overwriting an existing state stack, clear it first */
1444 clear_syn_state(sp);
1445 sp->sst_stacksize = current_state.ga_len;
1446 if (current_state.ga_len > SST_FIX_STATES)
1447 {
1448 /* Need to clear it, might be something remaining from when the
1449 * length was less than SST_FIX_STATES. */
1450 ga_init2(&sp->sst_union.sst_ga, (int)sizeof(bufstate_T), 1);
1451 if (ga_grow(&sp->sst_union.sst_ga, current_state.ga_len) == FAIL)
1452 sp->sst_stacksize = 0;
1453 else
Bram Moolenaar071d4272004-06-13 20:20:40 +00001454 sp->sst_union.sst_ga.ga_len = current_state.ga_len;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001455 bp = SYN_STATE_P(&(sp->sst_union.sst_ga));
1456 }
1457 else
1458 bp = sp->sst_union.sst_stack;
1459 for (i = 0; i < sp->sst_stacksize; ++i)
1460 {
1461 bp[i].bs_idx = CUR_STATE(i).si_idx;
1462 bp[i].bs_flags = CUR_STATE(i).si_flags;
1463 bp[i].bs_extmatch = ref_extmatch(CUR_STATE(i).si_extmatch);
1464 }
1465 sp->sst_next_flags = current_next_flags;
1466 sp->sst_next_list = current_next_list;
1467 sp->sst_tick = display_tick;
1468 sp->sst_change_lnum = 0;
1469 }
1470 current_state_stored = TRUE;
1471 return sp;
1472}
1473
1474/*
1475 * Copy a state stack from "from" in b_sst_array[] to current_state;
1476 */
1477 static void
1478load_current_state(from)
1479 synstate_T *from;
1480{
1481 int i;
1482 bufstate_T *bp;
1483
1484 clear_current_state();
1485 validate_current_state();
1486 keepend_level = -1;
1487 if (from->sst_stacksize
1488 && ga_grow(&current_state, from->sst_stacksize) != FAIL)
1489 {
1490 if (from->sst_stacksize > SST_FIX_STATES)
1491 bp = SYN_STATE_P(&(from->sst_union.sst_ga));
1492 else
1493 bp = from->sst_union.sst_stack;
1494 for (i = 0; i < from->sst_stacksize; ++i)
1495 {
1496 CUR_STATE(i).si_idx = bp[i].bs_idx;
1497 CUR_STATE(i).si_flags = bp[i].bs_flags;
1498 CUR_STATE(i).si_extmatch = ref_extmatch(bp[i].bs_extmatch);
1499 if (keepend_level < 0 && (CUR_STATE(i).si_flags & HL_KEEPEND))
1500 keepend_level = i;
1501 CUR_STATE(i).si_ends = FALSE;
1502 CUR_STATE(i).si_m_lnum = 0;
1503 if (CUR_STATE(i).si_idx >= 0)
1504 CUR_STATE(i).si_next_list =
1505 (SYN_ITEMS(syn_buf)[CUR_STATE(i).si_idx]).sp_next_list;
1506 else
1507 CUR_STATE(i).si_next_list = NULL;
1508 update_si_attr(i);
1509 }
1510 current_state.ga_len = from->sst_stacksize;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001511 }
1512 current_next_list = from->sst_next_list;
1513 current_next_flags = from->sst_next_flags;
1514 current_lnum = from->sst_lnum;
1515}
1516
1517/*
1518 * Compare saved state stack "*sp" with the current state.
1519 * Return TRUE when they are equal.
1520 */
1521 static int
1522syn_stack_equal(sp)
1523 synstate_T *sp;
1524{
1525 int i, j;
1526 bufstate_T *bp;
1527 reg_extmatch_T *six, *bsx;
1528
1529 /* First a quick check if the stacks have the same size end nextlist. */
1530 if (sp->sst_stacksize == current_state.ga_len
1531 && sp->sst_next_list == current_next_list)
1532 {
1533 /* Need to compare all states on both stacks. */
1534 if (sp->sst_stacksize > SST_FIX_STATES)
1535 bp = SYN_STATE_P(&(sp->sst_union.sst_ga));
1536 else
1537 bp = sp->sst_union.sst_stack;
1538
1539 for (i = current_state.ga_len; --i >= 0; )
1540 {
1541 /* If the item has another index the state is different. */
1542 if (bp[i].bs_idx != CUR_STATE(i).si_idx)
1543 break;
1544 if (bp[i].bs_extmatch != CUR_STATE(i).si_extmatch)
1545 {
1546 /* When the extmatch pointers are different, the strings in
1547 * them can still be the same. Check if the extmatch
1548 * references are equal. */
1549 bsx = bp[i].bs_extmatch;
1550 six = CUR_STATE(i).si_extmatch;
1551 /* If one of the extmatch pointers is NULL the states are
1552 * different. */
1553 if (bsx == NULL || six == NULL)
1554 break;
1555 for (j = 0; j < NSUBEXP; ++j)
1556 {
1557 /* Check each referenced match string. They must all be
1558 * equal. */
1559 if (bsx->matches[j] != six->matches[j])
1560 {
1561 /* If the pointer is different it can still be the
1562 * same text. Compare the strings, ignore case when
1563 * the start item has the sp_ic flag set. */
1564 if (bsx->matches[j] == NULL
1565 || six->matches[j] == NULL)
1566 break;
1567 if ((SYN_ITEMS(syn_buf)[CUR_STATE(i).si_idx]).sp_ic
1568 ? MB_STRICMP(bsx->matches[j],
1569 six->matches[j]) != 0
1570 : STRCMP(bsx->matches[j], six->matches[j]) != 0)
1571 break;
1572 }
1573 }
1574 if (j != NSUBEXP)
1575 break;
1576 }
1577 }
1578 if (i < 0)
1579 return TRUE;
1580 }
1581 return FALSE;
1582}
1583
1584/*
1585 * We stop parsing syntax above line "lnum". If the stored state at or below
1586 * this line depended on a change before it, it now depends on the line below
1587 * the last parsed line.
1588 * The window looks like this:
1589 * line which changed
1590 * displayed line
1591 * displayed line
1592 * lnum -> line below window
1593 */
1594 void
1595syntax_end_parsing(lnum)
1596 linenr_T lnum;
1597{
1598 synstate_T *sp;
1599
1600 sp = syn_stack_find_entry(lnum);
1601 if (sp != NULL && sp->sst_lnum < lnum)
1602 sp = sp->sst_next;
1603
1604 if (sp != NULL && sp->sst_change_lnum != 0)
1605 sp->sst_change_lnum = lnum;
1606}
1607
1608/*
1609 * End of handling of the state stack.
1610 ****************************************/
1611
1612 static void
1613invalidate_current_state()
1614{
1615 clear_current_state();
1616 current_state.ga_itemsize = 0; /* mark current_state invalid */
1617 current_next_list = NULL;
1618 keepend_level = -1;
1619}
1620
1621 static void
1622validate_current_state()
1623{
1624 current_state.ga_itemsize = sizeof(stateitem_T);
1625 current_state.ga_growsize = 3;
1626}
1627
1628/*
1629 * Return TRUE if the syntax at start of lnum changed since last time.
1630 * This will only be called just after get_syntax_attr() for the previous
1631 * line, to check if the next line needs to be redrawn too.
1632 */
1633 int
1634syntax_check_changed(lnum)
1635 linenr_T lnum;
1636{
1637 int retval = TRUE;
1638 synstate_T *sp;
1639
Bram Moolenaar071d4272004-06-13 20:20:40 +00001640 /*
1641 * Check the state stack when:
1642 * - lnum is just below the previously syntaxed line.
1643 * - lnum is not before the lines with saved states.
1644 * - lnum is not past the lines with saved states.
1645 * - lnum is at or before the last changed line.
1646 */
1647 if (VALID_STATE(&current_state) && lnum == current_lnum + 1)
1648 {
1649 sp = syn_stack_find_entry(lnum);
1650 if (sp != NULL && sp->sst_lnum == lnum)
1651 {
1652 /*
1653 * finish the previous line (needed when not all of the line was
1654 * drawn)
1655 */
1656 (void)syn_finish_line(FALSE);
1657
1658 /*
1659 * Compare the current state with the previously saved state of
1660 * the line.
1661 */
1662 if (syn_stack_equal(sp))
1663 retval = FALSE;
1664
1665 /*
1666 * Store the current state in b_sst_array[] for later use.
1667 */
1668 ++current_lnum;
Bram Moolenaardbe31752008-01-13 16:40:19 +00001669 (void)store_current_state();
Bram Moolenaar071d4272004-06-13 20:20:40 +00001670 }
1671 }
1672
Bram Moolenaar071d4272004-06-13 20:20:40 +00001673 return retval;
1674}
1675
1676/*
1677 * Finish the current line.
1678 * This doesn't return any attributes, it only gets the state at the end of
1679 * the line. It can start anywhere in the line, as long as the current state
1680 * is valid.
1681 */
1682 static int
1683syn_finish_line(syncing)
1684 int syncing; /* called for syncing */
1685{
1686 stateitem_T *cur_si;
Bram Moolenaar81366db2005-07-24 21:16:51 +00001687 colnr_T prev_current_col;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001688
1689 if (!current_finished)
1690 {
1691 while (!current_finished)
1692 {
Bram Moolenaar56cefaf2008-01-12 15:47:10 +00001693 (void)syn_current_attr(syncing, FALSE, NULL, FALSE);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001694 /*
1695 * When syncing, and found some item, need to check the item.
1696 */
1697 if (syncing && current_state.ga_len)
1698 {
1699 /*
1700 * Check for match with sync item.
1701 */
1702 cur_si = &CUR_STATE(current_state.ga_len - 1);
1703 if (cur_si->si_idx >= 0
1704 && (SYN_ITEMS(syn_buf)[cur_si->si_idx].sp_flags
1705 & (HL_SYNC_HERE|HL_SYNC_THERE)))
1706 return TRUE;
1707
1708 /* syn_current_attr() will have skipped the check for an item
Bram Moolenaar81366db2005-07-24 21:16:51 +00001709 * that ends here, need to do that now. Be careful not to go
1710 * past the NUL. */
1711 prev_current_col = current_col;
1712 if (syn_getcurline()[current_col] != NUL)
1713 ++current_col;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001714 check_state_ends();
Bram Moolenaar81366db2005-07-24 21:16:51 +00001715 current_col = prev_current_col;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001716 }
1717 ++current_col;
1718 }
1719 }
1720 return FALSE;
1721}
1722
1723/*
1724 * Return highlight attributes for next character.
1725 * Must first call syntax_start() once for the line.
1726 * "col" is normally 0 for the first use in a line, and increments by one each
1727 * time. It's allowed to skip characters and to stop before the end of the
1728 * line. But only a "col" after a previously used column is allowed.
Bram Moolenaar217ad922005-03-20 22:37:15 +00001729 * When "can_spell" is not NULL set it to TRUE when spell-checking should be
1730 * done.
Bram Moolenaar071d4272004-06-13 20:20:40 +00001731 */
1732 int
Bram Moolenaar56cefaf2008-01-12 15:47:10 +00001733get_syntax_attr(col, can_spell, keep_state)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001734 colnr_T col;
Bram Moolenaar217ad922005-03-20 22:37:15 +00001735 int *can_spell;
Bram Moolenaar56cefaf2008-01-12 15:47:10 +00001736 int keep_state; /* keep state of char at "col" */
Bram Moolenaar071d4272004-06-13 20:20:40 +00001737{
1738 int attr = 0;
1739
Bram Moolenaar349955a2007-08-14 21:07:36 +00001740 if (can_spell != NULL)
1741 /* Default: Only do spelling when there is no @Spell cluster or when
1742 * ":syn spell toplevel" was used. */
1743 *can_spell = syn_buf->b_syn_spell == SYNSPL_DEFAULT
1744 ? (syn_buf->b_spell_cluster_id == 0)
1745 : (syn_buf->b_syn_spell == SYNSPL_TOP);
1746
Bram Moolenaar071d4272004-06-13 20:20:40 +00001747 /* check for out of memory situation */
1748 if (syn_buf->b_sst_array == NULL)
1749 return 0;
1750
Bram Moolenaarce0842a2005-07-18 21:58:11 +00001751 /* After 'synmaxcol' the attribute is always zero. */
Bram Moolenaar84fb85a2005-07-20 22:02:14 +00001752 if (syn_buf->b_p_smc > 0 && col >= (colnr_T)syn_buf->b_p_smc)
Bram Moolenaarce0842a2005-07-18 21:58:11 +00001753 {
1754 clear_current_state();
1755#ifdef FEAT_EVAL
1756 current_id = 0;
1757 current_trans_id = 0;
1758#endif
1759 return 0;
1760 }
1761
Bram Moolenaar071d4272004-06-13 20:20:40 +00001762 /* Make sure current_state is valid */
1763 if (INVALID_STATE(&current_state))
1764 validate_current_state();
1765
1766 /*
1767 * Skip from the current column to "col", get the attributes for "col".
1768 */
1769 while (current_col <= col)
1770 {
Bram Moolenaar56cefaf2008-01-12 15:47:10 +00001771 attr = syn_current_attr(FALSE, TRUE, can_spell,
1772 current_col == col ? keep_state : FALSE);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001773 ++current_col;
1774 }
1775
Bram Moolenaar071d4272004-06-13 20:20:40 +00001776 return attr;
1777}
1778
1779/*
1780 * Get syntax attributes for current_lnum, current_col.
1781 */
1782 static int
Bram Moolenaar56cefaf2008-01-12 15:47:10 +00001783syn_current_attr(syncing, displaying, can_spell, keep_state)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001784 int syncing; /* When 1: called for syncing */
1785 int displaying; /* result will be displayed */
Bram Moolenaar217ad922005-03-20 22:37:15 +00001786 int *can_spell; /* return: do spell checking */
Bram Moolenaar56cefaf2008-01-12 15:47:10 +00001787 int keep_state; /* keep syntax stack afterwards */
Bram Moolenaar071d4272004-06-13 20:20:40 +00001788{
1789 int syn_id;
1790 lpos_T endpos; /* was: char_u *endp; */
1791 lpos_T hl_startpos; /* was: int hl_startcol; */
1792 lpos_T hl_endpos;
1793 lpos_T eos_pos; /* end-of-start match (start region) */
1794 lpos_T eoe_pos; /* end-of-end pattern */
1795 int end_idx; /* group ID for end pattern */
1796 int idx;
1797 synpat_T *spp;
Bram Moolenaar217ad922005-03-20 22:37:15 +00001798 stateitem_T *cur_si, *sip = NULL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001799 int startcol;
1800 int endcol;
1801 long flags;
1802 short *next_list;
1803 int found_match; /* found usable match */
1804 static int try_next_column = FALSE; /* must try in next col */
1805 int do_keywords;
1806 regmmatch_T regmatch;
1807 lpos_T pos;
1808 int lc_col;
1809 reg_extmatch_T *cur_extmatch = NULL;
1810 char_u *line; /* current line. NOTE: becomes invalid after
1811 looking for a pattern match! */
1812
1813 /* variables for zero-width matches that have a "nextgroup" argument */
1814 int keep_next_list;
1815 int zero_width_next_list = FALSE;
1816 garray_T zero_width_next_ga;
1817
1818 /*
1819 * No character, no attributes! Past end of line?
1820 * Do try matching with an empty line (could be the start of a region).
1821 */
1822 line = syn_getcurline();
1823 if (line[current_col] == NUL && current_col != 0)
1824 {
1825 /*
1826 * If we found a match after the last column, use it.
1827 */
1828 if (next_match_idx >= 0 && next_match_col >= (int)current_col
1829 && next_match_col != MAXCOL)
1830 (void)push_next_match(NULL);
1831
1832 current_finished = TRUE;
1833 current_state_stored = FALSE;
1834 return 0;
1835 }
1836
1837 /* if the current or next character is NUL, we will finish the line now */
1838 if (line[current_col] == NUL || line[current_col + 1] == NUL)
1839 {
1840 current_finished = TRUE;
1841 current_state_stored = FALSE;
1842 }
1843
1844 /*
1845 * When in the previous column there was a match but it could not be used
1846 * (empty match or already matched in this column) need to try again in
1847 * the next column.
1848 */
1849 if (try_next_column)
1850 {
1851 next_match_idx = -1;
1852 try_next_column = FALSE;
1853 }
1854
1855 /* Only check for keywords when not syncing and there are some. */
1856 do_keywords = !syncing
Bram Moolenaardad6b692005-01-25 22:14:34 +00001857 && (syn_buf->b_keywtab.ht_used > 0
1858 || syn_buf->b_keywtab_ic.ht_used > 0);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001859
1860 /* Init the list of zero-width matches with a nextlist. This is used to
1861 * avoid matching the same item in the same position twice. */
1862 ga_init2(&zero_width_next_ga, (int)sizeof(int), 10);
1863
1864 /*
1865 * Repeat matching keywords and patterns, to find contained items at the
1866 * same column. This stops when there are no extra matches at the current
1867 * column.
1868 */
1869 do
1870 {
1871 found_match = FALSE;
1872 keep_next_list = FALSE;
1873 syn_id = 0;
1874
1875 /*
1876 * 1. Check for a current state.
1877 * Only when there is no current state, or if the current state may
1878 * contain other things, we need to check for keywords and patterns.
1879 * Always need to check for contained items if some item has the
1880 * "containedin" argument (takes extra time!).
1881 */
1882 if (current_state.ga_len)
1883 cur_si = &CUR_STATE(current_state.ga_len - 1);
1884 else
1885 cur_si = NULL;
1886
1887 if (syn_buf->b_syn_containedin || cur_si == NULL
1888 || cur_si->si_cont_list != NULL)
1889 {
1890 /*
1891 * 2. Check for keywords, if on a keyword char after a non-keyword
1892 * char. Don't do this when syncing.
1893 */
1894 if (do_keywords)
1895 {
1896 line = syn_getcurline();
1897 if (vim_iswordc_buf(line + current_col, syn_buf)
1898 && (current_col == 0
1899 || !vim_iswordc_buf(line + current_col - 1
1900#ifdef FEAT_MBYTE
1901 - (has_mbyte
1902 ? (*mb_head_off)(line, line + current_col - 1)
1903 : 0)
1904#endif
1905 , syn_buf)))
1906 {
1907 syn_id = check_keyword_id(line, (int)current_col,
1908 &endcol, &flags, &next_list, cur_si);
Bram Moolenaare2cc9702005-03-15 22:43:58 +00001909 if (syn_id != 0)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001910 {
1911 if (push_current_state(KEYWORD_IDX) == OK)
1912 {
1913 cur_si = &CUR_STATE(current_state.ga_len - 1);
1914 cur_si->si_m_startcol = current_col;
1915 cur_si->si_h_startpos.lnum = current_lnum;
1916 cur_si->si_h_startpos.col = 0; /* starts right away */
1917 cur_si->si_m_endpos.lnum = current_lnum;
1918 cur_si->si_m_endpos.col = endcol;
1919 cur_si->si_h_endpos.lnum = current_lnum;
1920 cur_si->si_h_endpos.col = endcol;
1921 cur_si->si_ends = TRUE;
1922 cur_si->si_end_idx = 0;
1923 cur_si->si_flags = flags;
1924 cur_si->si_id = syn_id;
1925 cur_si->si_trans_id = syn_id;
1926 if (flags & HL_TRANSP)
1927 {
1928 if (current_state.ga_len < 2)
1929 {
1930 cur_si->si_attr = 0;
1931 cur_si->si_trans_id = 0;
1932 }
1933 else
1934 {
1935 cur_si->si_attr = CUR_STATE(
1936 current_state.ga_len - 2).si_attr;
1937 cur_si->si_trans_id = CUR_STATE(
1938 current_state.ga_len - 2).si_trans_id;
1939 }
1940 }
1941 else
1942 cur_si->si_attr = syn_id2attr(syn_id);
1943 cur_si->si_cont_list = NULL;
1944 cur_si->si_next_list = next_list;
1945 check_keepend();
1946 }
1947 else
1948 vim_free(next_list);
1949 }
1950 }
1951 }
1952
1953 /*
Bram Moolenaare2cc9702005-03-15 22:43:58 +00001954 * 3. Check for patterns (only if no keyword found).
Bram Moolenaar071d4272004-06-13 20:20:40 +00001955 */
1956 if (syn_id == 0 && syn_buf->b_syn_patterns.ga_len)
1957 {
1958 /*
1959 * If we didn't check for a match yet, or we are past it, check
1960 * for any match with a pattern.
1961 */
1962 if (next_match_idx < 0 || next_match_col < (int)current_col)
1963 {
1964 /*
1965 * Check all relevant patterns for a match at this
1966 * position. This is complicated, because matching with a
1967 * pattern takes quite a bit of time, thus we want to
1968 * avoid doing it when it's not needed.
1969 */
1970 next_match_idx = 0; /* no match in this line yet */
1971 next_match_col = MAXCOL;
1972 for (idx = syn_buf->b_syn_patterns.ga_len; --idx >= 0; )
1973 {
1974 spp = &(SYN_ITEMS(syn_buf)[idx]);
1975 if ( spp->sp_syncing == syncing
1976 && (displaying || !(spp->sp_flags & HL_DISPLAY))
1977 && (spp->sp_type == SPTYPE_MATCH
1978 || spp->sp_type == SPTYPE_START)
1979 && (current_next_list != NULL
1980 ? in_id_list(NULL, current_next_list,
1981 &spp->sp_syn, 0)
1982 : (cur_si == NULL
1983 ? !(spp->sp_flags & HL_CONTAINED)
1984 : in_id_list(cur_si,
1985 cur_si->si_cont_list, &spp->sp_syn,
1986 spp->sp_flags & HL_CONTAINED))))
1987 {
1988 /* If we already tried matching in this line, and
1989 * there isn't a match before next_match_col, skip
1990 * this item. */
1991 if (spp->sp_line_id == current_line_id
1992 && spp->sp_startcol >= next_match_col)
1993 continue;
1994 spp->sp_line_id = current_line_id;
1995
1996 lc_col = current_col - spp->sp_offsets[SPO_LC_OFF];
1997 if (lc_col < 0)
1998 lc_col = 0;
1999
2000 regmatch.rmm_ic = spp->sp_ic;
2001 regmatch.regprog = spp->sp_prog;
2002 if (!syn_regexec(&regmatch, current_lnum,
2003 (colnr_T)lc_col))
2004 {
2005 /* no match in this line, try another one */
2006 spp->sp_startcol = MAXCOL;
2007 continue;
2008 }
2009
2010 /*
2011 * Compute the first column of the match.
2012 */
2013 syn_add_start_off(&pos, &regmatch,
2014 spp, SPO_MS_OFF, -1);
2015 if (pos.lnum > current_lnum)
2016 {
2017 /* must have used end of match in a next line,
2018 * we can't handle that */
2019 spp->sp_startcol = MAXCOL;
2020 continue;
2021 }
2022 startcol = pos.col;
2023
2024 /* remember the next column where this pattern
2025 * matches in the current line */
2026 spp->sp_startcol = startcol;
2027
2028 /*
2029 * If a previously found match starts at a lower
2030 * column number, don't use this one.
2031 */
2032 if (startcol >= next_match_col)
2033 continue;
2034
2035 /*
2036 * If we matched this pattern at this position
2037 * before, skip it. Must retry in the next
2038 * column, because it may match from there.
2039 */
2040 if (did_match_already(idx, &zero_width_next_ga))
2041 {
2042 try_next_column = TRUE;
2043 continue;
2044 }
2045
2046 endpos.lnum = regmatch.endpos[0].lnum;
2047 endpos.col = regmatch.endpos[0].col;
2048
2049 /* Compute the highlight start. */
2050 syn_add_start_off(&hl_startpos, &regmatch,
2051 spp, SPO_HS_OFF, -1);
2052
2053 /* Compute the region start. */
2054 /* Default is to use the end of the match. */
2055 syn_add_end_off(&eos_pos, &regmatch,
2056 spp, SPO_RS_OFF, 0);
2057
2058 /*
2059 * Grab the external submatches before they get
2060 * overwritten. Reference count doesn't change.
2061 */
2062 unref_extmatch(cur_extmatch);
2063 cur_extmatch = re_extmatch_out;
2064 re_extmatch_out = NULL;
2065
2066 flags = 0;
2067 eoe_pos.lnum = 0; /* avoid warning */
2068 eoe_pos.col = 0;
2069 end_idx = 0;
2070 hl_endpos.lnum = 0;
2071
2072 /*
2073 * For a "oneline" the end must be found in the
2074 * same line too. Search for it after the end of
2075 * the match with the start pattern. Set the
2076 * resulting end positions at the same time.
2077 */
2078 if (spp->sp_type == SPTYPE_START
2079 && (spp->sp_flags & HL_ONELINE))
2080 {
2081 lpos_T startpos;
2082
2083 startpos = endpos;
2084 find_endpos(idx, &startpos, &endpos, &hl_endpos,
2085 &flags, &eoe_pos, &end_idx, cur_extmatch);
2086 if (endpos.lnum == 0)
2087 continue; /* not found */
2088 }
2089
2090 /*
2091 * For a "match" the size must be > 0 after the
2092 * end offset needs has been added. Except when
2093 * syncing.
2094 */
2095 else if (spp->sp_type == SPTYPE_MATCH)
2096 {
2097 syn_add_end_off(&hl_endpos, &regmatch, spp,
2098 SPO_HE_OFF, 0);
2099 syn_add_end_off(&endpos, &regmatch, spp,
2100 SPO_ME_OFF, 0);
2101 if (endpos.lnum == current_lnum
2102 && (int)endpos.col + syncing < startcol)
2103 {
2104 /*
2105 * If an empty string is matched, may need
2106 * to try matching again at next column.
2107 */
2108 if (regmatch.startpos[0].col
2109 == regmatch.endpos[0].col)
2110 try_next_column = TRUE;
2111 continue;
2112 }
2113 }
2114
2115 /*
2116 * keep the best match so far in next_match_*
2117 */
2118 /* Highlighting must start after startpos and end
2119 * before endpos. */
2120 if (hl_startpos.lnum == current_lnum
2121 && (int)hl_startpos.col < startcol)
2122 hl_startpos.col = startcol;
2123 limit_pos_zero(&hl_endpos, &endpos);
2124
2125 next_match_idx = idx;
2126 next_match_col = startcol;
2127 next_match_m_endpos = endpos;
2128 next_match_h_endpos = hl_endpos;
2129 next_match_h_startpos = hl_startpos;
2130 next_match_flags = flags;
2131 next_match_eos_pos = eos_pos;
2132 next_match_eoe_pos = eoe_pos;
2133 next_match_end_idx = end_idx;
2134 unref_extmatch(next_match_extmatch);
2135 next_match_extmatch = cur_extmatch;
2136 cur_extmatch = NULL;
2137 }
2138 }
2139 }
2140
2141 /*
2142 * If we found a match at the current column, use it.
2143 */
2144 if (next_match_idx >= 0 && next_match_col == (int)current_col)
2145 {
2146 synpat_T *lspp;
2147
2148 /* When a zero-width item matched which has a nextgroup,
2149 * don't push the item but set nextgroup. */
2150 lspp = &(SYN_ITEMS(syn_buf)[next_match_idx]);
2151 if (next_match_m_endpos.lnum == current_lnum
2152 && next_match_m_endpos.col == current_col
2153 && lspp->sp_next_list != NULL)
2154 {
2155 current_next_list = lspp->sp_next_list;
2156 current_next_flags = lspp->sp_flags;
2157 keep_next_list = TRUE;
2158 zero_width_next_list = TRUE;
2159
2160 /* Add the index to a list, so that we can check
2161 * later that we don't match it again (and cause an
2162 * endless loop). */
2163 if (ga_grow(&zero_width_next_ga, 1) == OK)
2164 {
2165 ((int *)(zero_width_next_ga.ga_data))
2166 [zero_width_next_ga.ga_len++] = next_match_idx;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002167 }
2168 next_match_idx = -1;
2169 }
2170 else
2171 cur_si = push_next_match(cur_si);
2172 found_match = TRUE;
2173 }
2174 }
2175 }
2176
2177 /*
2178 * Handle searching for nextgroup match.
2179 */
2180 if (current_next_list != NULL && !keep_next_list)
2181 {
2182 /*
2183 * If a nextgroup was not found, continue looking for one if:
2184 * - this is an empty line and the "skipempty" option was given
2185 * - we are on white space and the "skipwhite" option was given
2186 */
2187 if (!found_match)
2188 {
2189 line = syn_getcurline();
2190 if (((current_next_flags & HL_SKIPWHITE)
2191 && vim_iswhite(line[current_col]))
2192 || ((current_next_flags & HL_SKIPEMPTY)
2193 && *line == NUL))
2194 break;
2195 }
2196
2197 /*
2198 * If a nextgroup was found: Use it, and continue looking for
2199 * contained matches.
2200 * If a nextgroup was not found: Continue looking for a normal
2201 * match.
2202 * When did set current_next_list for a zero-width item and no
2203 * match was found don't loop (would get stuck).
2204 */
2205 current_next_list = NULL;
2206 next_match_idx = -1;
2207 if (!zero_width_next_list)
2208 found_match = TRUE;
2209 }
2210
2211 } while (found_match);
2212
2213 /*
2214 * Use attributes from the current state, if within its highlighting.
2215 * If not, use attributes from the current-but-one state, etc.
2216 */
2217 current_attr = 0;
2218#ifdef FEAT_EVAL
2219 current_id = 0;
2220 current_trans_id = 0;
2221#endif
2222 if (cur_si != NULL)
2223 {
Bram Moolenaar217ad922005-03-20 22:37:15 +00002224#ifndef FEAT_EVAL
2225 int current_trans_id = 0;
2226#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00002227 for (idx = current_state.ga_len - 1; idx >= 0; --idx)
2228 {
2229 sip = &CUR_STATE(idx);
2230 if ((current_lnum > sip->si_h_startpos.lnum
2231 || (current_lnum == sip->si_h_startpos.lnum
2232 && current_col >= sip->si_h_startpos.col))
2233 && (sip->si_h_endpos.lnum == 0
2234 || current_lnum < sip->si_h_endpos.lnum
2235 || (current_lnum == sip->si_h_endpos.lnum
2236 && current_col < sip->si_h_endpos.col)))
2237 {
2238 current_attr = sip->si_attr;
2239#ifdef FEAT_EVAL
2240 current_id = sip->si_id;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002241#endif
Bram Moolenaar217ad922005-03-20 22:37:15 +00002242 current_trans_id = sip->si_trans_id;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002243 break;
2244 }
2245 }
2246
Bram Moolenaar217ad922005-03-20 22:37:15 +00002247 if (can_spell != NULL)
2248 {
2249 struct sp_syn sps;
2250
2251 /*
2252 * set "can_spell" to TRUE if spell checking is supposed to be
2253 * done in the current item.
2254 */
Bram Moolenaar217ad922005-03-20 22:37:15 +00002255 if (syn_buf->b_spell_cluster_id == 0)
Bram Moolenaar6bb68362005-03-22 23:03:44 +00002256 {
Bram Moolenaar3638c682005-06-08 22:05:14 +00002257 /* There is no @Spell cluster: Do spelling for items without
2258 * @NoSpell cluster. */
Bram Moolenaar6bb68362005-03-22 23:03:44 +00002259 if (syn_buf->b_nospell_cluster_id == 0 || current_trans_id == 0)
Bram Moolenaarce0842a2005-07-18 21:58:11 +00002260 *can_spell = (syn_buf->b_syn_spell != SYNSPL_NOTOP);
Bram Moolenaar6bb68362005-03-22 23:03:44 +00002261 else
2262 {
2263 sps.inc_tag = 0;
2264 sps.id = syn_buf->b_nospell_cluster_id;
2265 sps.cont_in_list = NULL;
2266 *can_spell = !in_id_list(sip, sip->si_cont_list, &sps, 0);
2267 }
2268 }
Bram Moolenaar217ad922005-03-20 22:37:15 +00002269 else
2270 {
Bram Moolenaar3638c682005-06-08 22:05:14 +00002271 /* The @Spell cluster is defined: Do spelling in items with
Bram Moolenaarce0842a2005-07-18 21:58:11 +00002272 * the @Spell cluster. But not when @NoSpell is also there.
2273 * At the toplevel only spell check when ":syn spell toplevel"
2274 * was used. */
Bram Moolenaar3638c682005-06-08 22:05:14 +00002275 if (current_trans_id == 0)
Bram Moolenaarce0842a2005-07-18 21:58:11 +00002276 *can_spell = (syn_buf->b_syn_spell == SYNSPL_TOP);
Bram Moolenaar3638c682005-06-08 22:05:14 +00002277 else
2278 {
2279 sps.inc_tag = 0;
2280 sps.id = syn_buf->b_spell_cluster_id;
2281 sps.cont_in_list = NULL;
2282 *can_spell = in_id_list(sip, sip->si_cont_list, &sps, 0);
2283
2284 if (syn_buf->b_nospell_cluster_id != 0)
2285 {
2286 sps.id = syn_buf->b_nospell_cluster_id;
2287 if (in_id_list(sip, sip->si_cont_list, &sps, 0))
2288 *can_spell = FALSE;
2289 }
2290 }
Bram Moolenaar217ad922005-03-20 22:37:15 +00002291 }
2292 }
2293
2294
Bram Moolenaar071d4272004-06-13 20:20:40 +00002295 /*
2296 * Check for end of current state (and the states before it) at the
2297 * next column. Don't do this for syncing, because we would miss a
2298 * single character match.
2299 * First check if the current state ends at the current column. It
2300 * may be for an empty match and a containing item might end in the
2301 * current column.
2302 */
Bram Moolenaar56cefaf2008-01-12 15:47:10 +00002303 if (!syncing && !keep_state)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002304 {
2305 check_state_ends();
Bram Moolenaar81366db2005-07-24 21:16:51 +00002306 if (current_state.ga_len > 0
2307 && syn_getcurline()[current_col] != NUL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002308 {
2309 ++current_col;
2310 check_state_ends();
2311 --current_col;
2312 }
2313 }
2314 }
Bram Moolenaar217ad922005-03-20 22:37:15 +00002315 else if (can_spell != NULL)
Bram Moolenaarce0842a2005-07-18 21:58:11 +00002316 /* Default: Only do spelling when there is no @Spell cluster or when
2317 * ":syn spell toplevel" was used. */
2318 *can_spell = syn_buf->b_syn_spell == SYNSPL_DEFAULT
2319 ? (syn_buf->b_spell_cluster_id == 0)
2320 : (syn_buf->b_syn_spell == SYNSPL_TOP);
Bram Moolenaar071d4272004-06-13 20:20:40 +00002321
2322 /* nextgroup ends at end of line, unless "skipnl" or "skipemtpy" present */
2323 if (current_next_list != NULL
2324 && syn_getcurline()[current_col + 1] == NUL
2325 && !(current_next_flags & (HL_SKIPNL | HL_SKIPEMPTY)))
2326 current_next_list = NULL;
2327
2328 if (zero_width_next_ga.ga_len > 0)
2329 ga_clear(&zero_width_next_ga);
2330
2331 /* No longer need external matches. But keep next_match_extmatch. */
2332 unref_extmatch(re_extmatch_out);
2333 re_extmatch_out = NULL;
2334 unref_extmatch(cur_extmatch);
2335
2336 return current_attr;
2337}
2338
2339
2340/*
2341 * Check if we already matched pattern "idx" at the current column.
2342 */
2343 static int
2344did_match_already(idx, gap)
2345 int idx;
2346 garray_T *gap;
2347{
2348 int i;
2349
2350 for (i = current_state.ga_len; --i >= 0; )
2351 if (CUR_STATE(i).si_m_startcol == (int)current_col
2352 && CUR_STATE(i).si_m_lnum == (int)current_lnum
2353 && CUR_STATE(i).si_idx == idx)
2354 return TRUE;
2355
2356 /* Zero-width matches with a nextgroup argument are not put on the syntax
2357 * stack, and can only be matched once anyway. */
2358 for (i = gap->ga_len; --i >= 0; )
2359 if (((int *)(gap->ga_data))[i] == idx)
2360 return TRUE;
2361
2362 return FALSE;
2363}
2364
2365/*
2366 * Push the next match onto the stack.
2367 */
2368 static stateitem_T *
2369push_next_match(cur_si)
2370 stateitem_T *cur_si;
2371{
2372 synpat_T *spp;
2373
2374 spp = &(SYN_ITEMS(syn_buf)[next_match_idx]);
2375
2376 /*
2377 * Push the item in current_state stack;
2378 */
2379 if (push_current_state(next_match_idx) == OK)
2380 {
2381 /*
2382 * If it's a start-skip-end type that crosses lines, figure out how
2383 * much it continues in this line. Otherwise just fill in the length.
2384 */
2385 cur_si = &CUR_STATE(current_state.ga_len - 1);
2386 cur_si->si_h_startpos = next_match_h_startpos;
2387 cur_si->si_m_startcol = current_col;
2388 cur_si->si_m_lnum = current_lnum;
2389 cur_si->si_flags = spp->sp_flags;
2390 cur_si->si_next_list = spp->sp_next_list;
2391 cur_si->si_extmatch = ref_extmatch(next_match_extmatch);
2392 if (spp->sp_type == SPTYPE_START && !(spp->sp_flags & HL_ONELINE))
2393 {
2394 /* Try to find the end pattern in the current line */
2395 update_si_end(cur_si, (int)(next_match_m_endpos.col), TRUE);
2396 check_keepend();
2397 }
2398 else
2399 {
2400 cur_si->si_m_endpos = next_match_m_endpos;
2401 cur_si->si_h_endpos = next_match_h_endpos;
2402 cur_si->si_ends = TRUE;
2403 cur_si->si_flags |= next_match_flags;
2404 cur_si->si_eoe_pos = next_match_eoe_pos;
2405 cur_si->si_end_idx = next_match_end_idx;
2406 }
2407 if (keepend_level < 0 && (cur_si->si_flags & HL_KEEPEND))
2408 keepend_level = current_state.ga_len - 1;
2409 check_keepend();
2410 update_si_attr(current_state.ga_len - 1);
2411
2412 /*
2413 * If the start pattern has another highlight group, push another item
2414 * on the stack for the start pattern.
2415 */
2416 if ( spp->sp_type == SPTYPE_START
2417 && spp->sp_syn_match_id != 0
2418 && push_current_state(next_match_idx) == OK)
2419 {
2420 cur_si = &CUR_STATE(current_state.ga_len - 1);
2421 cur_si->si_h_startpos = next_match_h_startpos;
2422 cur_si->si_m_startcol = current_col;
2423 cur_si->si_m_lnum = current_lnum;
2424 cur_si->si_m_endpos = next_match_eos_pos;
2425 cur_si->si_h_endpos = next_match_eos_pos;
2426 cur_si->si_ends = TRUE;
2427 cur_si->si_end_idx = 0;
2428 cur_si->si_flags = HL_MATCH;
2429 cur_si->si_next_list = NULL;
2430 check_keepend();
2431 update_si_attr(current_state.ga_len - 1);
2432 }
2433 }
2434
2435 next_match_idx = -1; /* try other match next time */
2436
2437 return cur_si;
2438}
2439
2440/*
2441 * Check for end of current state (and the states before it).
2442 */
2443 static void
2444check_state_ends()
2445{
2446 stateitem_T *cur_si;
2447 int had_extend = FALSE;
2448
2449 cur_si = &CUR_STATE(current_state.ga_len - 1);
2450 for (;;)
2451 {
2452 if (cur_si->si_ends
2453 && (cur_si->si_m_endpos.lnum < current_lnum
2454 || (cur_si->si_m_endpos.lnum == current_lnum
2455 && cur_si->si_m_endpos.col <= current_col)))
2456 {
2457 /*
2458 * If there is an end pattern group ID, highlight the end pattern
2459 * now. No need to pop the current item from the stack.
2460 * Only do this if the end pattern continues beyond the current
2461 * position.
2462 */
2463 if (cur_si->si_end_idx
2464 && (cur_si->si_eoe_pos.lnum > current_lnum
2465 || (cur_si->si_eoe_pos.lnum == current_lnum
2466 && cur_si->si_eoe_pos.col > current_col)))
2467 {
2468 cur_si->si_idx = cur_si->si_end_idx;
2469 cur_si->si_end_idx = 0;
2470 cur_si->si_m_endpos = cur_si->si_eoe_pos;
2471 cur_si->si_h_endpos = cur_si->si_eoe_pos;
2472 cur_si->si_flags |= HL_MATCH;
2473 update_si_attr(current_state.ga_len - 1);
Bram Moolenaar293ee4d2004-12-09 21:34:53 +00002474
2475 /* what matches next may be different now, clear it */
2476 next_match_idx = 0;
2477 next_match_col = MAXCOL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002478 break;
2479 }
2480 else
2481 {
2482 /* handle next_list, unless at end of line and no "skipnl" or
2483 * "skipempty" */
2484 current_next_list = cur_si->si_next_list;
2485 current_next_flags = cur_si->si_flags;
2486 if (!(current_next_flags & (HL_SKIPNL | HL_SKIPEMPTY))
2487 && syn_getcurline()[current_col] == NUL)
2488 current_next_list = NULL;
2489
2490 /* When the ended item has "extend", another item with
2491 * "keepend" now needs to check for its end. */
2492 if (cur_si->si_flags & HL_EXTEND)
2493 had_extend = TRUE;
2494
2495 pop_current_state();
2496
2497 if (current_state.ga_len == 0)
2498 break;
2499
Bram Moolenaar81993f42008-01-11 20:27:45 +00002500 if (had_extend && keepend_level >= 0)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002501 {
2502 syn_update_ends(FALSE);
2503 if (current_state.ga_len == 0)
2504 break;
2505 }
2506
2507 cur_si = &CUR_STATE(current_state.ga_len - 1);
2508
2509 /*
2510 * Only for a region the search for the end continues after
2511 * the end of the contained item. If the contained match
2512 * included the end-of-line, break here, the region continues.
2513 * Don't do this when:
2514 * - "keepend" is used for the contained item
2515 * - not at the end of the line (could be end="x$"me=e-1).
2516 * - "excludenl" is used (HL_HAS_EOL won't be set)
2517 */
2518 if (cur_si->si_idx >= 0
2519 && SYN_ITEMS(syn_buf)[cur_si->si_idx].sp_type
2520 == SPTYPE_START
2521 && !(cur_si->si_flags & (HL_MATCH | HL_KEEPEND)))
2522 {
2523 update_si_end(cur_si, (int)current_col, TRUE);
2524 check_keepend();
2525 if ((current_next_flags & HL_HAS_EOL)
2526 && keepend_level < 0
2527 && syn_getcurline()[current_col] == NUL)
2528 break;
2529 }
2530 }
2531 }
2532 else
2533 break;
2534 }
2535}
2536
2537/*
2538 * Update an entry in the current_state stack for a match or region. This
2539 * fills in si_attr, si_next_list and si_cont_list.
2540 */
2541 static void
2542update_si_attr(idx)
2543 int idx;
2544{
2545 stateitem_T *sip = &CUR_STATE(idx);
2546 synpat_T *spp;
2547
Bram Moolenaar3a36cf72007-08-21 15:29:56 +00002548 /* This should not happen... */
2549 if (sip->si_idx < 0)
2550 return;
2551
Bram Moolenaar071d4272004-06-13 20:20:40 +00002552 spp = &(SYN_ITEMS(syn_buf)[sip->si_idx]);
2553 if (sip->si_flags & HL_MATCH)
2554 sip->si_id = spp->sp_syn_match_id;
2555 else
2556 sip->si_id = spp->sp_syn.id;
2557 sip->si_attr = syn_id2attr(sip->si_id);
2558 sip->si_trans_id = sip->si_id;
2559 if (sip->si_flags & HL_MATCH)
2560 sip->si_cont_list = NULL;
2561 else
2562 sip->si_cont_list = spp->sp_cont_list;
2563
2564 /*
2565 * For transparent items, take attr from outer item.
2566 * Also take cont_list, if there is none.
2567 * Don't do this for the matchgroup of a start or end pattern.
2568 */
2569 if ((spp->sp_flags & HL_TRANSP) && !(sip->si_flags & HL_MATCH))
2570 {
2571 if (idx == 0)
2572 {
2573 sip->si_attr = 0;
2574 sip->si_trans_id = 0;
2575 if (sip->si_cont_list == NULL)
2576 sip->si_cont_list = ID_LIST_ALL;
2577 }
2578 else
2579 {
2580 sip->si_attr = CUR_STATE(idx - 1).si_attr;
2581 sip->si_trans_id = CUR_STATE(idx - 1).si_trans_id;
Bram Moolenaar293ee4d2004-12-09 21:34:53 +00002582 sip->si_h_startpos = CUR_STATE(idx - 1).si_h_startpos;
2583 sip->si_h_endpos = CUR_STATE(idx - 1).si_h_endpos;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002584 if (sip->si_cont_list == NULL)
2585 {
2586 sip->si_flags |= HL_TRANS_CONT;
2587 sip->si_cont_list = CUR_STATE(idx - 1).si_cont_list;
2588 }
2589 }
2590 }
2591}
2592
2593/*
2594 * Check the current stack for patterns with "keepend" flag.
2595 * Propagate the match-end to contained items, until a "skipend" item is found.
2596 */
2597 static void
2598check_keepend()
2599{
2600 int i;
2601 lpos_T maxpos;
Bram Moolenaar7bd2cd82006-10-03 15:04:36 +00002602 lpos_T maxpos_h;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002603 stateitem_T *sip;
2604
2605 /*
2606 * This check can consume a lot of time; only do it from the level where
2607 * there really is a keepend.
2608 */
2609 if (keepend_level < 0)
2610 return;
2611
2612 /*
2613 * Find the last index of an "extend" item. "keepend" items before that
2614 * won't do anything. If there is no "extend" item "i" will be
2615 * "keepend_level" and all "keepend" items will work normally.
2616 */
2617 for (i = current_state.ga_len - 1; i > keepend_level; --i)
2618 if (CUR_STATE(i).si_flags & HL_EXTEND)
2619 break;
2620
2621 maxpos.lnum = 0;
Bram Moolenaared39e1d2008-08-09 17:55:22 +00002622 maxpos.col = 0;
Bram Moolenaar7bd2cd82006-10-03 15:04:36 +00002623 maxpos_h.lnum = 0;
Bram Moolenaared39e1d2008-08-09 17:55:22 +00002624 maxpos_h.col = 0;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002625 for ( ; i < current_state.ga_len; ++i)
2626 {
2627 sip = &CUR_STATE(i);
2628 if (maxpos.lnum != 0)
2629 {
2630 limit_pos_zero(&sip->si_m_endpos, &maxpos);
Bram Moolenaar7bd2cd82006-10-03 15:04:36 +00002631 limit_pos_zero(&sip->si_h_endpos, &maxpos_h);
Bram Moolenaar071d4272004-06-13 20:20:40 +00002632 limit_pos_zero(&sip->si_eoe_pos, &maxpos);
2633 sip->si_ends = TRUE;
2634 }
Bram Moolenaar7bd2cd82006-10-03 15:04:36 +00002635 if (sip->si_ends && (sip->si_flags & HL_KEEPEND))
2636 {
2637 if (maxpos.lnum == 0
Bram Moolenaar071d4272004-06-13 20:20:40 +00002638 || maxpos.lnum > sip->si_m_endpos.lnum
2639 || (maxpos.lnum == sip->si_m_endpos.lnum
Bram Moolenaar7bd2cd82006-10-03 15:04:36 +00002640 && maxpos.col > sip->si_m_endpos.col))
2641 maxpos = sip->si_m_endpos;
2642 if (maxpos_h.lnum == 0
2643 || maxpos_h.lnum > sip->si_h_endpos.lnum
2644 || (maxpos_h.lnum == sip->si_h_endpos.lnum
2645 && maxpos_h.col > sip->si_h_endpos.col))
2646 maxpos_h = sip->si_h_endpos;
2647 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00002648 }
2649}
2650
2651/*
2652 * Update an entry in the current_state stack for a start-skip-end pattern.
2653 * This finds the end of the current item, if it's in the current line.
2654 *
2655 * Return the flags for the matched END.
2656 */
2657 static void
2658update_si_end(sip, startcol, force)
2659 stateitem_T *sip;
2660 int startcol; /* where to start searching for the end */
2661 int force; /* when TRUE overrule a previous end */
2662{
2663 lpos_T startpos;
2664 lpos_T endpos;
2665 lpos_T hl_endpos;
2666 lpos_T end_endpos;
2667 int end_idx;
2668
Bram Moolenaar3a36cf72007-08-21 15:29:56 +00002669 /* return quickly for a keyword */
2670 if (sip->si_idx < 0)
2671 return;
2672
Bram Moolenaar071d4272004-06-13 20:20:40 +00002673 /* Don't update when it's already done. Can be a match of an end pattern
2674 * that started in a previous line. Watch out: can also be a "keepend"
2675 * from a containing item. */
2676 if (!force && sip->si_m_endpos.lnum >= current_lnum)
2677 return;
2678
2679 /*
2680 * We need to find the end of the region. It may continue in the next
2681 * line.
2682 */
2683 end_idx = 0;
2684 startpos.lnum = current_lnum;
2685 startpos.col = startcol;
2686 find_endpos(sip->si_idx, &startpos, &endpos, &hl_endpos,
2687 &(sip->si_flags), &end_endpos, &end_idx, sip->si_extmatch);
2688
2689 if (endpos.lnum == 0)
2690 {
2691 /* No end pattern matched. */
2692 if (SYN_ITEMS(syn_buf)[sip->si_idx].sp_flags & HL_ONELINE)
2693 {
2694 /* a "oneline" never continues in the next line */
2695 sip->si_ends = TRUE;
2696 sip->si_m_endpos.lnum = current_lnum;
2697 sip->si_m_endpos.col = (colnr_T)STRLEN(syn_getcurline());
2698 }
2699 else
2700 {
2701 /* continues in the next line */
2702 sip->si_ends = FALSE;
2703 sip->si_m_endpos.lnum = 0;
2704 }
2705 sip->si_h_endpos = sip->si_m_endpos;
2706 }
2707 else
2708 {
2709 /* match within this line */
2710 sip->si_m_endpos = endpos;
2711 sip->si_h_endpos = hl_endpos;
2712 sip->si_eoe_pos = end_endpos;
2713 sip->si_ends = TRUE;
2714 sip->si_end_idx = end_idx;
2715 }
2716}
2717
2718/*
2719 * Add a new state to the current state stack.
2720 * It is cleared and the index set to "idx".
2721 * Return FAIL if it's not possible (out of memory).
2722 */
2723 static int
2724push_current_state(idx)
2725 int idx;
2726{
2727 if (ga_grow(&current_state, 1) == FAIL)
2728 return FAIL;
2729 vim_memset(&CUR_STATE(current_state.ga_len), 0, sizeof(stateitem_T));
2730 CUR_STATE(current_state.ga_len).si_idx = idx;
2731 ++current_state.ga_len;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002732 return OK;
2733}
2734
2735/*
2736 * Remove a state from the current_state stack.
2737 */
2738 static void
2739pop_current_state()
2740{
2741 if (current_state.ga_len)
2742 {
2743 unref_extmatch(CUR_STATE(current_state.ga_len - 1).si_extmatch);
2744 --current_state.ga_len;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002745 }
2746 /* after the end of a pattern, try matching a keyword or pattern */
2747 next_match_idx = -1;
2748
2749 /* if first state with "keepend" is popped, reset keepend_level */
2750 if (keepend_level >= current_state.ga_len)
2751 keepend_level = -1;
2752}
2753
2754/*
2755 * Find the end of a start/skip/end syntax region after "startpos".
2756 * Only checks one line.
2757 * Also handles a match item that continued from a previous line.
2758 * If not found, the syntax item continues in the next line. m_endpos->lnum
2759 * will be 0.
2760 * If found, the end of the region and the end of the highlighting is
2761 * computed.
2762 */
2763 static void
2764find_endpos(idx, startpos, m_endpos, hl_endpos, flagsp, end_endpos,
2765 end_idx, start_ext)
2766 int idx; /* index of the pattern */
2767 lpos_T *startpos; /* where to start looking for an END match */
2768 lpos_T *m_endpos; /* return: end of match */
2769 lpos_T *hl_endpos; /* return: end of highlighting */
2770 long *flagsp; /* return: flags of matching END */
2771 lpos_T *end_endpos; /* return: end of end pattern match */
2772 int *end_idx; /* return: group ID for end pat. match, or 0 */
2773 reg_extmatch_T *start_ext; /* submatches from the start pattern */
2774{
2775 colnr_T matchcol;
2776 synpat_T *spp, *spp_skip;
2777 int start_idx;
2778 int best_idx;
2779 regmmatch_T regmatch;
2780 regmmatch_T best_regmatch; /* startpos/endpos of best match */
2781 lpos_T pos;
2782 char_u *line;
2783 int had_match = FALSE;
2784
Bram Moolenaar3a36cf72007-08-21 15:29:56 +00002785 /* just in case we are invoked for a keyword */
2786 if (idx < 0)
2787 return;
2788
Bram Moolenaar071d4272004-06-13 20:20:40 +00002789 /*
2790 * Check for being called with a START pattern.
2791 * Can happen with a match that continues to the next line, because it
2792 * contained a region.
2793 */
2794 spp = &(SYN_ITEMS(syn_buf)[idx]);
2795 if (spp->sp_type != SPTYPE_START)
2796 {
2797 *hl_endpos = *startpos;
2798 return;
2799 }
2800
2801 /*
2802 * Find the SKIP or first END pattern after the last START pattern.
2803 */
2804 for (;;)
2805 {
2806 spp = &(SYN_ITEMS(syn_buf)[idx]);
2807 if (spp->sp_type != SPTYPE_START)
2808 break;
2809 ++idx;
2810 }
2811
2812 /*
2813 * Lookup the SKIP pattern (if present)
2814 */
2815 if (spp->sp_type == SPTYPE_SKIP)
2816 {
2817 spp_skip = spp;
2818 ++idx;
2819 }
2820 else
2821 spp_skip = NULL;
2822
2823 /* Setup external matches for syn_regexec(). */
2824 unref_extmatch(re_extmatch_in);
2825 re_extmatch_in = ref_extmatch(start_ext);
2826
2827 matchcol = startpos->col; /* start looking for a match at sstart */
2828 start_idx = idx; /* remember the first END pattern. */
2829 best_regmatch.startpos[0].col = 0; /* avoid compiler warning */
2830 for (;;)
2831 {
2832 /*
2833 * Find end pattern that matches first after "matchcol".
2834 */
2835 best_idx = -1;
2836 for (idx = start_idx; idx < syn_buf->b_syn_patterns.ga_len; ++idx)
2837 {
2838 int lc_col = matchcol;
2839
2840 spp = &(SYN_ITEMS(syn_buf)[idx]);
2841 if (spp->sp_type != SPTYPE_END) /* past last END pattern */
2842 break;
2843 lc_col -= spp->sp_offsets[SPO_LC_OFF];
2844 if (lc_col < 0)
2845 lc_col = 0;
2846
2847 regmatch.rmm_ic = spp->sp_ic;
2848 regmatch.regprog = spp->sp_prog;
2849 if (syn_regexec(&regmatch, startpos->lnum, lc_col))
2850 {
2851 if (best_idx == -1 || regmatch.startpos[0].col
2852 < best_regmatch.startpos[0].col)
2853 {
2854 best_idx = idx;
2855 best_regmatch.startpos[0] = regmatch.startpos[0];
2856 best_regmatch.endpos[0] = regmatch.endpos[0];
2857 }
2858 }
2859 }
2860
2861 /*
2862 * If all end patterns have been tried, and there is no match, the
2863 * item continues until end-of-line.
2864 */
2865 if (best_idx == -1)
2866 break;
2867
2868 /*
2869 * If the skip pattern matches before the end pattern,
2870 * continue searching after the skip pattern.
2871 */
2872 if (spp_skip != NULL)
2873 {
2874 int lc_col = matchcol - spp_skip->sp_offsets[SPO_LC_OFF];
2875
2876 if (lc_col < 0)
2877 lc_col = 0;
2878 regmatch.rmm_ic = spp_skip->sp_ic;
2879 regmatch.regprog = spp_skip->sp_prog;
2880 if (syn_regexec(&regmatch, startpos->lnum, lc_col)
2881 && regmatch.startpos[0].col
2882 <= best_regmatch.startpos[0].col)
2883 {
2884 /* Add offset to skip pattern match */
2885 syn_add_end_off(&pos, &regmatch, spp_skip, SPO_ME_OFF, 1);
2886
2887 /* If the skip pattern goes on to the next line, there is no
2888 * match with an end pattern in this line. */
2889 if (pos.lnum > startpos->lnum)
2890 break;
2891
2892 line = ml_get_buf(syn_buf, startpos->lnum, FALSE);
2893
2894 /* take care of an empty match or negative offset */
2895 if (pos.col <= matchcol)
2896 ++matchcol;
2897 else if (pos.col <= regmatch.endpos[0].col)
2898 matchcol = pos.col;
2899 else
2900 /* Be careful not to jump over the NUL at the end-of-line */
2901 for (matchcol = regmatch.endpos[0].col;
2902 line[matchcol] != NUL && matchcol < pos.col;
2903 ++matchcol)
2904 ;
2905
2906 /* if the skip pattern includes end-of-line, break here */
2907 if (line[matchcol] == NUL)
2908 break;
2909
2910 continue; /* start with first end pattern again */
2911 }
2912 }
2913
2914 /*
2915 * Match from start pattern to end pattern.
2916 * Correct for match and highlight offset of end pattern.
2917 */
2918 spp = &(SYN_ITEMS(syn_buf)[best_idx]);
2919 syn_add_end_off(m_endpos, &best_regmatch, spp, SPO_ME_OFF, 1);
2920 /* can't end before the start */
2921 if (m_endpos->lnum == startpos->lnum && m_endpos->col < startpos->col)
2922 m_endpos->col = startpos->col;
2923
2924 syn_add_end_off(end_endpos, &best_regmatch, spp, SPO_HE_OFF, 1);
2925 /* can't end before the start */
2926 if (end_endpos->lnum == startpos->lnum
2927 && end_endpos->col < startpos->col)
2928 end_endpos->col = startpos->col;
2929 /* can't end after the match */
2930 limit_pos(end_endpos, m_endpos);
2931
2932 /*
2933 * If the end group is highlighted differently, adjust the pointers.
2934 */
2935 if (spp->sp_syn_match_id != spp->sp_syn.id && spp->sp_syn_match_id != 0)
2936 {
2937 *end_idx = best_idx;
2938 if (spp->sp_off_flags & (1 << (SPO_RE_OFF + SPO_COUNT)))
2939 {
2940 hl_endpos->lnum = best_regmatch.endpos[0].lnum;
2941 hl_endpos->col = best_regmatch.endpos[0].col;
2942 }
2943 else
2944 {
2945 hl_endpos->lnum = best_regmatch.startpos[0].lnum;
2946 hl_endpos->col = best_regmatch.startpos[0].col;
2947 }
2948 hl_endpos->col += spp->sp_offsets[SPO_RE_OFF];
2949
2950 /* can't end before the start */
2951 if (hl_endpos->lnum == startpos->lnum
2952 && hl_endpos->col < startpos->col)
2953 hl_endpos->col = startpos->col;
2954 limit_pos(hl_endpos, m_endpos);
2955
2956 /* now the match ends where the highlighting ends, it is turned
2957 * into the matchgroup for the end */
2958 *m_endpos = *hl_endpos;
2959 }
2960 else
2961 {
2962 *end_idx = 0;
2963 *hl_endpos = *end_endpos;
2964 }
2965
2966 *flagsp = spp->sp_flags;
2967
2968 had_match = TRUE;
2969 break;
2970 }
2971
2972 /* no match for an END pattern in this line */
2973 if (!had_match)
2974 m_endpos->lnum = 0;
2975
2976 /* Remove external matches. */
2977 unref_extmatch(re_extmatch_in);
2978 re_extmatch_in = NULL;
2979}
2980
2981/*
2982 * Limit "pos" not to be after "limit".
2983 */
2984 static void
2985limit_pos(pos, limit)
2986 lpos_T *pos;
2987 lpos_T *limit;
2988{
2989 if (pos->lnum > limit->lnum)
2990 *pos = *limit;
2991 else if (pos->lnum == limit->lnum && pos->col > limit->col)
2992 pos->col = limit->col;
2993}
2994
2995/*
2996 * Limit "pos" not to be after "limit", unless pos->lnum is zero.
2997 */
2998 static void
2999limit_pos_zero(pos, limit)
3000 lpos_T *pos;
3001 lpos_T *limit;
3002{
3003 if (pos->lnum == 0)
3004 *pos = *limit;
3005 else
3006 limit_pos(pos, limit);
3007}
3008
3009/*
3010 * Add offset to matched text for end of match or highlight.
3011 */
3012 static void
3013syn_add_end_off(result, regmatch, spp, idx, extra)
3014 lpos_T *result; /* returned position */
3015 regmmatch_T *regmatch; /* start/end of match */
3016 synpat_T *spp; /* matched pattern */
3017 int idx; /* index of offset */
3018 int extra; /* extra chars for offset to start */
3019{
3020 int col;
Bram Moolenaar8c8de832008-06-24 22:58:06 +00003021 int off;
3022 char_u *base;
3023 char_u *p;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003024
3025 if (spp->sp_off_flags & (1 << idx))
3026 {
3027 result->lnum = regmatch->startpos[0].lnum;
Bram Moolenaar8c8de832008-06-24 22:58:06 +00003028 col = regmatch->startpos[0].col;
3029 off = spp->sp_offsets[idx] + extra;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003030 }
3031 else
3032 {
3033 result->lnum = regmatch->endpos[0].lnum;
3034 col = regmatch->endpos[0].col;
Bram Moolenaar8c8de832008-06-24 22:58:06 +00003035 off = spp->sp_offsets[idx];
Bram Moolenaar071d4272004-06-13 20:20:40 +00003036 }
Bram Moolenaar8c8de832008-06-24 22:58:06 +00003037 /* Don't go past the end of the line. Matters for "rs=e+2" when there
3038 * is a matchgroup. Watch out for match with last NL in the buffer. */
3039 if (result->lnum > syn_buf->b_ml.ml_line_count)
3040 col = 0;
3041 else if (off != 0)
Bram Moolenaara40ceaf2006-01-13 22:35:40 +00003042 {
Bram Moolenaar8c8de832008-06-24 22:58:06 +00003043 base = ml_get_buf(syn_buf, result->lnum, FALSE);
3044 p = base + col;
3045 if (off > 0)
3046 {
3047 while (off-- > 0 && *p != NUL)
3048 mb_ptr_adv(p);
3049 }
3050 else if (off < 0)
3051 {
3052 while (off++ < 0 && base < p)
3053 mb_ptr_back(base, p);
3054 }
3055 col = (int)(p - base);
Bram Moolenaara40ceaf2006-01-13 22:35:40 +00003056 }
Bram Moolenaar8c8de832008-06-24 22:58:06 +00003057 result->col = col;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003058}
3059
3060/*
3061 * Add offset to matched text for start of match or highlight.
3062 * Avoid resulting column to become negative.
3063 */
3064 static void
3065syn_add_start_off(result, regmatch, spp, idx, extra)
3066 lpos_T *result; /* returned position */
3067 regmmatch_T *regmatch; /* start/end of match */
3068 synpat_T *spp;
3069 int idx;
3070 int extra; /* extra chars for offset to end */
3071{
3072 int col;
Bram Moolenaar8c8de832008-06-24 22:58:06 +00003073 int off;
3074 char_u *base;
3075 char_u *p;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003076
3077 if (spp->sp_off_flags & (1 << (idx + SPO_COUNT)))
3078 {
3079 result->lnum = regmatch->endpos[0].lnum;
Bram Moolenaar8c8de832008-06-24 22:58:06 +00003080 col = regmatch->endpos[0].col;
3081 off = spp->sp_offsets[idx] + extra;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003082 }
3083 else
3084 {
3085 result->lnum = regmatch->startpos[0].lnum;
3086 col = regmatch->startpos[0].col;
Bram Moolenaar8c8de832008-06-24 22:58:06 +00003087 off = spp->sp_offsets[idx];
Bram Moolenaar071d4272004-06-13 20:20:40 +00003088 }
Bram Moolenaar8c8de832008-06-24 22:58:06 +00003089 if (off != 0)
3090 {
3091 base = ml_get_buf(syn_buf, result->lnum, FALSE);
3092 p = base + col;
3093 if (off > 0)
3094 {
3095 while (off-- && *p != NUL)
3096 mb_ptr_adv(p);
3097 }
3098 else if (off < 0)
3099 {
3100 while (off++ && base < p)
3101 mb_ptr_back(base, p);
3102 }
3103 col = (int)(p - base);
3104 }
3105 result->col = col;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003106}
3107
3108/*
3109 * Get current line in syntax buffer.
3110 */
3111 static char_u *
3112syn_getcurline()
3113{
3114 return ml_get_buf(syn_buf, current_lnum, FALSE);
3115}
3116
3117/*
Bram Moolenaar3b56eb32005-07-11 22:40:32 +00003118 * Call vim_regexec() to find a match with "rmp" in "syn_buf".
Bram Moolenaar071d4272004-06-13 20:20:40 +00003119 * Returns TRUE when there is a match.
3120 */
3121 static int
3122syn_regexec(rmp, lnum, col)
3123 regmmatch_T *rmp;
3124 linenr_T lnum;
3125 colnr_T col;
3126{
Bram Moolenaar3b56eb32005-07-11 22:40:32 +00003127 rmp->rmm_maxcol = syn_buf->b_p_smc;
Bram Moolenaar91a4e822008-01-19 14:59:58 +00003128 if (vim_regexec_multi(rmp, syn_win, syn_buf, lnum, col, NULL) > 0)
Bram Moolenaar071d4272004-06-13 20:20:40 +00003129 {
3130 rmp->startpos[0].lnum += lnum;
3131 rmp->endpos[0].lnum += lnum;
3132 return TRUE;
3133 }
3134 return FALSE;
3135}
3136
3137/*
3138 * Check one position in a line for a matching keyword.
3139 * The caller must check if a keyword can start at startcol.
3140 * Return it's ID if found, 0 otherwise.
3141 */
3142 static int
3143check_keyword_id(line, startcol, endcolp, flagsp, next_listp, cur_si)
3144 char_u *line;
3145 int startcol; /* position in line to check for keyword */
3146 int *endcolp; /* return: character after found keyword */
3147 long *flagsp; /* return: flags of matching keyword */
3148 short **next_listp; /* return: next_list of matching keyword */
3149 stateitem_T *cur_si; /* item at the top of the stack */
3150{
Bram Moolenaardad6b692005-01-25 22:14:34 +00003151 keyentry_T *kp;
3152 char_u *kwp;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003153 int round;
Bram Moolenaardad6b692005-01-25 22:14:34 +00003154 int kwlen;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003155 char_u keyword[MAXKEYWLEN + 1]; /* assume max. keyword len is 80 */
Bram Moolenaardad6b692005-01-25 22:14:34 +00003156 hashtab_T *ht;
3157 hashitem_T *hi;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003158
3159 /* Find first character after the keyword. First character was already
3160 * checked. */
Bram Moolenaardad6b692005-01-25 22:14:34 +00003161 kwp = line + startcol;
3162 kwlen = 0;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003163 do
3164 {
3165#ifdef FEAT_MBYTE
3166 if (has_mbyte)
Bram Moolenaar0fa313a2005-08-10 21:07:57 +00003167 kwlen += (*mb_ptr2len)(kwp + kwlen);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003168 else
3169#endif
Bram Moolenaardad6b692005-01-25 22:14:34 +00003170 ++kwlen;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003171 }
Bram Moolenaardad6b692005-01-25 22:14:34 +00003172 while (vim_iswordc_buf(kwp + kwlen, syn_buf));
Bram Moolenaar071d4272004-06-13 20:20:40 +00003173
Bram Moolenaardad6b692005-01-25 22:14:34 +00003174 if (kwlen > MAXKEYWLEN)
Bram Moolenaar071d4272004-06-13 20:20:40 +00003175 return 0;
3176
3177 /*
3178 * Must make a copy of the keyword, so we can add a NUL and make it
3179 * lowercase.
3180 */
Bram Moolenaarce0842a2005-07-18 21:58:11 +00003181 vim_strncpy(keyword, kwp, kwlen);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003182
3183 /*
3184 * Try twice:
3185 * 1. matching case
3186 * 2. ignoring case
3187 */
3188 for (round = 1; round <= 2; ++round)
3189 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00003190 ht = round == 1 ? &syn_buf->b_keywtab : &syn_buf->b_keywtab_ic;
3191 if (ht->ht_used == 0)
Bram Moolenaar071d4272004-06-13 20:20:40 +00003192 continue;
Bram Moolenaardad6b692005-01-25 22:14:34 +00003193 if (round == 2) /* ignore case */
3194 (void)str_foldcase(kwp, kwlen, keyword, MAXKEYWLEN + 1);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003195
3196 /*
Bram Moolenaardad6b692005-01-25 22:14:34 +00003197 * Find keywords that match. There can be several with different
3198 * attributes.
Bram Moolenaar071d4272004-06-13 20:20:40 +00003199 * When current_next_list is non-zero accept only that group, otherwise:
3200 * Accept a not-contained keyword at toplevel.
3201 * Accept a keyword at other levels only if it is in the contains list.
3202 */
Bram Moolenaardad6b692005-01-25 22:14:34 +00003203 hi = hash_find(ht, keyword);
3204 if (!HASHITEM_EMPTY(hi))
3205 for (kp = HI2KE(hi); kp != NULL; kp = kp->ke_next)
Bram Moolenaar071d4272004-06-13 20:20:40 +00003206 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00003207 if (current_next_list != 0
3208 ? in_id_list(NULL, current_next_list, &kp->k_syn, 0)
3209 : (cur_si == NULL
3210 ? !(kp->flags & HL_CONTAINED)
3211 : in_id_list(cur_si, cur_si->si_cont_list,
3212 &kp->k_syn, kp->flags & HL_CONTAINED)))
3213 {
3214 *endcolp = startcol + kwlen;
3215 *flagsp = kp->flags;
3216 *next_listp = kp->next_list;
3217 return kp->k_syn.id;
3218 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00003219 }
3220 }
3221 return 0;
3222}
3223
3224/*
3225 * Handle ":syntax case" command.
3226 */
3227/* ARGSUSED */
3228 static void
3229syn_cmd_case(eap, syncing)
3230 exarg_T *eap;
3231 int syncing; /* not used */
3232{
3233 char_u *arg = eap->arg;
3234 char_u *next;
3235
3236 eap->nextcmd = find_nextcmd(arg);
3237 if (eap->skip)
3238 return;
3239
3240 next = skiptowhite(arg);
3241 if (STRNICMP(arg, "match", 5) == 0 && next - arg == 5)
3242 curbuf->b_syn_ic = FALSE;
3243 else if (STRNICMP(arg, "ignore", 6) == 0 && next - arg == 6)
3244 curbuf->b_syn_ic = TRUE;
3245 else
3246 EMSG2(_("E390: Illegal argument: %s"), arg);
3247}
3248
3249/*
Bram Moolenaarce0842a2005-07-18 21:58:11 +00003250 * Handle ":syntax spell" command.
3251 */
3252/* ARGSUSED */
3253 static void
3254syn_cmd_spell(eap, syncing)
3255 exarg_T *eap;
3256 int syncing; /* not used */
3257{
3258 char_u *arg = eap->arg;
3259 char_u *next;
3260
3261 eap->nextcmd = find_nextcmd(arg);
3262 if (eap->skip)
3263 return;
3264
3265 next = skiptowhite(arg);
3266 if (STRNICMP(arg, "toplevel", 8) == 0 && next - arg == 8)
3267 curbuf->b_syn_spell = SYNSPL_TOP;
3268 else if (STRNICMP(arg, "notoplevel", 10) == 0 && next - arg == 10)
3269 curbuf->b_syn_spell = SYNSPL_NOTOP;
Bram Moolenaar86ea7642007-02-04 01:48:10 +00003270 else if (STRNICMP(arg, "default", 7) == 0 && next - arg == 7)
Bram Moolenaarce0842a2005-07-18 21:58:11 +00003271 curbuf->b_syn_spell = SYNSPL_DEFAULT;
3272 else
3273 EMSG2(_("E390: Illegal argument: %s"), arg);
3274}
3275
3276/*
Bram Moolenaar071d4272004-06-13 20:20:40 +00003277 * Clear all syntax info for one buffer.
3278 */
3279 void
3280syntax_clear(buf)
3281 buf_T *buf;
3282{
3283 int i;
3284
Bram Moolenaar5b8d8fd2005-08-16 23:01:50 +00003285 buf->b_syn_error = FALSE; /* clear previous error */
Bram Moolenaar54ee7752005-05-31 22:22:17 +00003286 buf->b_syn_ic = FALSE; /* Use case, by default */
Bram Moolenaarce0842a2005-07-18 21:58:11 +00003287 buf->b_syn_spell = SYNSPL_DEFAULT; /* default spell checking */
Bram Moolenaar54ee7752005-05-31 22:22:17 +00003288 buf->b_syn_containedin = FALSE;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003289
3290 /* free the keywords */
Bram Moolenaardad6b692005-01-25 22:14:34 +00003291 clear_keywtab(&buf->b_keywtab);
3292 clear_keywtab(&buf->b_keywtab_ic);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003293
3294 /* free the syntax patterns */
3295 for (i = buf->b_syn_patterns.ga_len; --i >= 0; )
3296 syn_clear_pattern(buf, i);
3297 ga_clear(&buf->b_syn_patterns);
3298
3299 /* free the syntax clusters */
3300 for (i = buf->b_syn_clusters.ga_len; --i >= 0; )
3301 syn_clear_cluster(buf, i);
3302 ga_clear(&buf->b_syn_clusters);
Bram Moolenaar75c50c42005-06-04 22:06:24 +00003303 buf->b_spell_cluster_id = 0;
3304 buf->b_nospell_cluster_id = 0;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003305
3306 buf->b_syn_sync_flags = 0;
3307 buf->b_syn_sync_minlines = 0;
3308 buf->b_syn_sync_maxlines = 0;
3309 buf->b_syn_sync_linebreaks = 0;
3310
3311 vim_free(buf->b_syn_linecont_prog);
3312 buf->b_syn_linecont_prog = NULL;
3313 vim_free(buf->b_syn_linecont_pat);
3314 buf->b_syn_linecont_pat = NULL;
3315#ifdef FEAT_FOLDING
3316 buf->b_syn_folditems = 0;
3317#endif
3318
3319 /* free the stored states */
3320 syn_stack_free_all(buf);
3321 invalidate_current_state();
3322}
3323
3324/*
3325 * Clear syncing info for one buffer.
3326 */
3327 static void
3328syntax_sync_clear()
3329{
3330 int i;
3331
3332 /* free the syntax patterns */
3333 for (i = curbuf->b_syn_patterns.ga_len; --i >= 0; )
3334 if (SYN_ITEMS(curbuf)[i].sp_syncing)
3335 syn_remove_pattern(curbuf, i);
3336
3337 curbuf->b_syn_sync_flags = 0;
3338 curbuf->b_syn_sync_minlines = 0;
3339 curbuf->b_syn_sync_maxlines = 0;
3340 curbuf->b_syn_sync_linebreaks = 0;
3341
3342 vim_free(curbuf->b_syn_linecont_prog);
3343 curbuf->b_syn_linecont_prog = NULL;
3344 vim_free(curbuf->b_syn_linecont_pat);
3345 curbuf->b_syn_linecont_pat = NULL;
3346
3347 syn_stack_free_all(curbuf); /* Need to recompute all syntax. */
3348}
3349
3350/*
3351 * Remove one pattern from the buffer's pattern list.
3352 */
3353 static void
3354syn_remove_pattern(buf, idx)
3355 buf_T *buf;
3356 int idx;
3357{
3358 synpat_T *spp;
3359
3360 spp = &(SYN_ITEMS(buf)[idx]);
3361#ifdef FEAT_FOLDING
3362 if (spp->sp_flags & HL_FOLD)
3363 --buf->b_syn_folditems;
3364#endif
3365 syn_clear_pattern(buf, idx);
3366 mch_memmove(spp, spp + 1,
3367 sizeof(synpat_T) * (buf->b_syn_patterns.ga_len - idx - 1));
3368 --buf->b_syn_patterns.ga_len;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003369}
3370
3371/*
3372 * Clear and free one syntax pattern. When clearing all, must be called from
3373 * last to first!
3374 */
3375 static void
3376syn_clear_pattern(buf, i)
3377 buf_T *buf;
3378 int i;
3379{
3380 vim_free(SYN_ITEMS(buf)[i].sp_pattern);
3381 vim_free(SYN_ITEMS(buf)[i].sp_prog);
3382 /* Only free sp_cont_list and sp_next_list of first start pattern */
3383 if (i == 0 || SYN_ITEMS(buf)[i - 1].sp_type != SPTYPE_START)
3384 {
3385 vim_free(SYN_ITEMS(buf)[i].sp_cont_list);
3386 vim_free(SYN_ITEMS(buf)[i].sp_next_list);
Bram Moolenaar51f78b22007-10-07 13:22:19 +00003387 vim_free(SYN_ITEMS(buf)[i].sp_syn.cont_in_list);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003388 }
3389}
3390
3391/*
3392 * Clear and free one syntax cluster.
3393 */
3394 static void
3395syn_clear_cluster(buf, i)
3396 buf_T *buf;
3397 int i;
3398{
3399 vim_free(SYN_CLSTR(buf)[i].scl_name);
3400 vim_free(SYN_CLSTR(buf)[i].scl_name_u);
3401 vim_free(SYN_CLSTR(buf)[i].scl_list);
3402}
3403
3404/*
3405 * Handle ":syntax clear" command.
3406 */
3407 static void
3408syn_cmd_clear(eap, syncing)
3409 exarg_T *eap;
3410 int syncing;
3411{
3412 char_u *arg = eap->arg;
3413 char_u *arg_end;
3414 int id;
3415
3416 eap->nextcmd = find_nextcmd(arg);
3417 if (eap->skip)
3418 return;
3419
3420 /*
3421 * We have to disable this within ":syn include @group filename",
3422 * because otherwise @group would get deleted.
3423 * Only required for Vim 5.x syntax files, 6.0 ones don't contain ":syn
3424 * clear".
3425 */
3426 if (curbuf->b_syn_topgrp != 0)
3427 return;
3428
3429 if (ends_excmd(*arg))
3430 {
3431 /*
3432 * No argument: Clear all syntax items.
3433 */
3434 if (syncing)
3435 syntax_sync_clear();
3436 else
3437 {
3438 syntax_clear(curbuf);
Bram Moolenaar2ce06f62005-01-31 19:19:04 +00003439 do_unlet((char_u *)"b:current_syntax", TRUE);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003440 }
3441 }
3442 else
3443 {
3444 /*
3445 * Clear the group IDs that are in the argument.
3446 */
3447 while (!ends_excmd(*arg))
3448 {
3449 arg_end = skiptowhite(arg);
3450 if (*arg == '@')
3451 {
3452 id = syn_scl_namen2id(arg + 1, (int)(arg_end - arg - 1));
3453 if (id == 0)
3454 {
3455 EMSG2(_("E391: No such syntax cluster: %s"), arg);
3456 break;
3457 }
3458 else
3459 {
3460 /*
3461 * We can't physically delete a cluster without changing
3462 * the IDs of other clusters, so we do the next best thing
3463 * and make it empty.
3464 */
3465 short scl_id = id - SYNID_CLUSTER;
3466
3467 vim_free(SYN_CLSTR(curbuf)[scl_id].scl_list);
3468 SYN_CLSTR(curbuf)[scl_id].scl_list = NULL;
3469 }
3470 }
3471 else
3472 {
3473 id = syn_namen2id(arg, (int)(arg_end - arg));
3474 if (id == 0)
3475 {
3476 EMSG2(_(e_nogroup), arg);
3477 break;
3478 }
3479 else
3480 syn_clear_one(id, syncing);
3481 }
3482 arg = skipwhite(arg_end);
3483 }
3484 }
Bram Moolenaar1c8f93f2006-03-12 22:10:07 +00003485 redraw_curbuf_later(SOME_VALID);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003486 syn_stack_free_all(curbuf); /* Need to recompute all syntax. */
3487}
3488
3489/*
3490 * Clear one syntax group for the current buffer.
3491 */
3492 static void
3493syn_clear_one(id, syncing)
3494 int id;
3495 int syncing;
3496{
3497 synpat_T *spp;
3498 int idx;
3499
3500 /* Clear keywords only when not ":syn sync clear group-name" */
3501 if (!syncing)
3502 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00003503 (void)syn_clear_keyword(id, &curbuf->b_keywtab);
3504 (void)syn_clear_keyword(id, &curbuf->b_keywtab_ic);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003505 }
3506
3507 /* clear the patterns for "id" */
3508 for (idx = curbuf->b_syn_patterns.ga_len; --idx >= 0; )
3509 {
3510 spp = &(SYN_ITEMS(curbuf)[idx]);
3511 if (spp->sp_syn.id != id || spp->sp_syncing != syncing)
3512 continue;
3513 syn_remove_pattern(curbuf, idx);
3514 }
3515}
3516
3517/*
3518 * Handle ":syntax on" command.
3519 */
3520/* ARGSUSED */
3521 static void
3522syn_cmd_on(eap, syncing)
3523 exarg_T *eap;
3524 int syncing; /* not used */
3525{
3526 syn_cmd_onoff(eap, "syntax");
3527}
3528
3529/*
3530 * Handle ":syntax enable" command.
3531 */
3532/* ARGSUSED */
3533 static void
3534syn_cmd_enable(eap, syncing)
3535 exarg_T *eap;
3536 int syncing; /* not used */
3537{
3538 set_internal_string_var((char_u *)"syntax_cmd", (char_u *)"enable");
3539 syn_cmd_onoff(eap, "syntax");
Bram Moolenaar2ce06f62005-01-31 19:19:04 +00003540 do_unlet((char_u *)"g:syntax_cmd", TRUE);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003541}
3542
3543/*
3544 * Handle ":syntax reset" command.
3545 */
3546/* ARGSUSED */
3547 static void
3548syn_cmd_reset(eap, syncing)
3549 exarg_T *eap;
3550 int syncing; /* not used */
3551{
3552 eap->nextcmd = check_nextcmd(eap->arg);
3553 if (!eap->skip)
3554 {
3555 set_internal_string_var((char_u *)"syntax_cmd", (char_u *)"reset");
3556 do_cmdline_cmd((char_u *)"runtime! syntax/syncolor.vim");
Bram Moolenaar2ce06f62005-01-31 19:19:04 +00003557 do_unlet((char_u *)"g:syntax_cmd", TRUE);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003558 }
3559}
3560
3561/*
3562 * Handle ":syntax manual" command.
3563 */
3564/* ARGSUSED */
3565 static void
3566syn_cmd_manual(eap, syncing)
3567 exarg_T *eap;
3568 int syncing; /* not used */
3569{
3570 syn_cmd_onoff(eap, "manual");
3571}
3572
3573/*
3574 * Handle ":syntax off" command.
3575 */
3576/* ARGSUSED */
3577 static void
3578syn_cmd_off(eap, syncing)
3579 exarg_T *eap;
3580 int syncing; /* not used */
3581{
3582 syn_cmd_onoff(eap, "nosyntax");
3583}
3584
3585 static void
3586syn_cmd_onoff(eap, name)
3587 exarg_T *eap;
3588 char *name;
3589{
3590 char_u buf[100];
3591
3592 eap->nextcmd = check_nextcmd(eap->arg);
3593 if (!eap->skip)
3594 {
3595 STRCPY(buf, "so ");
Bram Moolenaar555b2802005-05-19 21:08:39 +00003596 vim_snprintf((char *)buf + 3, sizeof(buf) - 3, SYNTAX_FNAME, name);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003597 do_cmdline_cmd(buf);
3598 }
3599}
3600
3601/*
3602 * Handle ":syntax [list]" command: list current syntax words.
3603 */
3604 static void
3605syn_cmd_list(eap, syncing)
3606 exarg_T *eap;
3607 int syncing; /* when TRUE: list syncing items */
3608{
3609 char_u *arg = eap->arg;
3610 int id;
3611 char_u *arg_end;
3612
3613 eap->nextcmd = find_nextcmd(arg);
3614 if (eap->skip)
3615 return;
3616
3617 if (!syntax_present(curbuf))
3618 {
3619 MSG(_("No Syntax items defined for this buffer"));
3620 return;
3621 }
3622
3623 if (syncing)
3624 {
3625 if (curbuf->b_syn_sync_flags & SF_CCOMMENT)
3626 {
3627 MSG_PUTS(_("syncing on C-style comments"));
3628 syn_lines_msg();
3629 syn_match_msg();
3630 return;
3631 }
3632 else if (!(curbuf->b_syn_sync_flags & SF_MATCH))
3633 {
3634 if (curbuf->b_syn_sync_minlines == 0)
3635 MSG_PUTS(_("no syncing"));
3636 else
3637 {
3638 MSG_PUTS(_("syncing starts "));
3639 msg_outnum(curbuf->b_syn_sync_minlines);
3640 MSG_PUTS(_(" lines before top line"));
3641 syn_match_msg();
3642 }
3643 return;
3644 }
3645 MSG_PUTS_TITLE(_("\n--- Syntax sync items ---"));
3646 if (curbuf->b_syn_sync_minlines > 0
3647 || curbuf->b_syn_sync_maxlines > 0
3648 || curbuf->b_syn_sync_linebreaks > 0)
3649 {
3650 MSG_PUTS(_("\nsyncing on items"));
3651 syn_lines_msg();
3652 syn_match_msg();
3653 }
3654 }
3655 else
3656 MSG_PUTS_TITLE(_("\n--- Syntax items ---"));
3657 if (ends_excmd(*arg))
3658 {
3659 /*
3660 * No argument: List all group IDs and all syntax clusters.
3661 */
3662 for (id = 1; id <= highlight_ga.ga_len && !got_int; ++id)
3663 syn_list_one(id, syncing, FALSE);
3664 for (id = 0; id < curbuf->b_syn_clusters.ga_len && !got_int; ++id)
3665 syn_list_cluster(id);
3666 }
3667 else
3668 {
3669 /*
3670 * List the group IDs and syntax clusters that are in the argument.
3671 */
3672 while (!ends_excmd(*arg) && !got_int)
3673 {
3674 arg_end = skiptowhite(arg);
3675 if (*arg == '@')
3676 {
3677 id = syn_scl_namen2id(arg + 1, (int)(arg_end - arg - 1));
3678 if (id == 0)
3679 EMSG2(_("E392: No such syntax cluster: %s"), arg);
3680 else
3681 syn_list_cluster(id - SYNID_CLUSTER);
3682 }
3683 else
3684 {
3685 id = syn_namen2id(arg, (int)(arg_end - arg));
3686 if (id == 0)
3687 EMSG2(_(e_nogroup), arg);
3688 else
3689 syn_list_one(id, syncing, TRUE);
3690 }
3691 arg = skipwhite(arg_end);
3692 }
3693 }
3694 eap->nextcmd = check_nextcmd(arg);
3695}
3696
3697 static void
3698syn_lines_msg()
3699{
3700 if (curbuf->b_syn_sync_maxlines > 0 || curbuf->b_syn_sync_minlines > 0)
3701 {
3702 MSG_PUTS("; ");
3703 if (curbuf->b_syn_sync_minlines > 0)
3704 {
3705 MSG_PUTS(_("minimal "));
3706 msg_outnum(curbuf->b_syn_sync_minlines);
3707 if (curbuf->b_syn_sync_maxlines)
3708 MSG_PUTS(", ");
3709 }
3710 if (curbuf->b_syn_sync_maxlines > 0)
3711 {
3712 MSG_PUTS(_("maximal "));
3713 msg_outnum(curbuf->b_syn_sync_maxlines);
3714 }
3715 MSG_PUTS(_(" lines before top line"));
3716 }
3717}
3718
3719 static void
3720syn_match_msg()
3721{
3722 if (curbuf->b_syn_sync_linebreaks > 0)
3723 {
3724 MSG_PUTS(_("; match "));
3725 msg_outnum(curbuf->b_syn_sync_linebreaks);
3726 MSG_PUTS(_(" line breaks"));
3727 }
3728}
3729
3730static int last_matchgroup;
3731
3732struct name_list
3733{
3734 int flag;
3735 char *name;
3736};
3737
3738static void syn_list_flags __ARGS((struct name_list *nl, int flags, int attr));
3739
3740/*
3741 * List one syntax item, for ":syntax" or "syntax list syntax_name".
3742 */
3743 static void
3744syn_list_one(id, syncing, link_only)
3745 int id;
3746 int syncing; /* when TRUE: list syncing items */
3747 int link_only; /* when TRUE; list link-only too */
3748{
3749 int attr;
3750 int idx;
3751 int did_header = FALSE;
3752 synpat_T *spp;
3753 static struct name_list namelist1[] =
3754 {
3755 {HL_DISPLAY, "display"},
3756 {HL_CONTAINED, "contained"},
3757 {HL_ONELINE, "oneline"},
3758 {HL_KEEPEND, "keepend"},
3759 {HL_EXTEND, "extend"},
3760 {HL_EXCLUDENL, "excludenl"},
3761 {HL_TRANSP, "transparent"},
3762 {HL_FOLD, "fold"},
3763 {0, NULL}
3764 };
3765 static struct name_list namelist2[] =
3766 {
3767 {HL_SKIPWHITE, "skipwhite"},
3768 {HL_SKIPNL, "skipnl"},
3769 {HL_SKIPEMPTY, "skipempty"},
3770 {0, NULL}
3771 };
3772
3773 attr = hl_attr(HLF_D); /* highlight like directories */
3774
3775 /* list the keywords for "id" */
3776 if (!syncing)
3777 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00003778 did_header = syn_list_keywords(id, &curbuf->b_keywtab, FALSE, attr);
3779 did_header = syn_list_keywords(id, &curbuf->b_keywtab_ic,
Bram Moolenaar071d4272004-06-13 20:20:40 +00003780 did_header, attr);
3781 }
3782
3783 /* list the patterns for "id" */
3784 for (idx = 0; idx < curbuf->b_syn_patterns.ga_len && !got_int; ++idx)
3785 {
3786 spp = &(SYN_ITEMS(curbuf)[idx]);
3787 if (spp->sp_syn.id != id || spp->sp_syncing != syncing)
3788 continue;
3789
3790 (void)syn_list_header(did_header, 999, id);
3791 did_header = TRUE;
3792 last_matchgroup = 0;
3793 if (spp->sp_type == SPTYPE_MATCH)
3794 {
3795 put_pattern("match", ' ', spp, attr);
3796 msg_putchar(' ');
3797 }
3798 else if (spp->sp_type == SPTYPE_START)
3799 {
3800 while (SYN_ITEMS(curbuf)[idx].sp_type == SPTYPE_START)
3801 put_pattern("start", '=', &SYN_ITEMS(curbuf)[idx++], attr);
3802 if (SYN_ITEMS(curbuf)[idx].sp_type == SPTYPE_SKIP)
3803 put_pattern("skip", '=', &SYN_ITEMS(curbuf)[idx++], attr);
3804 while (idx < curbuf->b_syn_patterns.ga_len
3805 && SYN_ITEMS(curbuf)[idx].sp_type == SPTYPE_END)
3806 put_pattern("end", '=', &SYN_ITEMS(curbuf)[idx++], attr);
3807 --idx;
3808 msg_putchar(' ');
3809 }
3810 syn_list_flags(namelist1, spp->sp_flags, attr);
3811
3812 if (spp->sp_cont_list != NULL)
3813 put_id_list((char_u *)"contains", spp->sp_cont_list, attr);
3814
3815 if (spp->sp_syn.cont_in_list != NULL)
3816 put_id_list((char_u *)"containedin",
3817 spp->sp_syn.cont_in_list, attr);
3818
3819 if (spp->sp_next_list != NULL)
3820 {
3821 put_id_list((char_u *)"nextgroup", spp->sp_next_list, attr);
3822 syn_list_flags(namelist2, spp->sp_flags, attr);
3823 }
3824 if (spp->sp_flags & (HL_SYNC_HERE|HL_SYNC_THERE))
3825 {
3826 if (spp->sp_flags & HL_SYNC_HERE)
3827 msg_puts_attr((char_u *)"grouphere", attr);
3828 else
3829 msg_puts_attr((char_u *)"groupthere", attr);
3830 msg_putchar(' ');
3831 if (spp->sp_sync_idx >= 0)
3832 msg_outtrans(HL_TABLE()[SYN_ITEMS(curbuf)
3833 [spp->sp_sync_idx].sp_syn.id - 1].sg_name);
3834 else
3835 MSG_PUTS("NONE");
3836 msg_putchar(' ');
3837 }
3838 }
3839
3840 /* list the link, if there is one */
3841 if (HL_TABLE()[id - 1].sg_link && (did_header || link_only) && !got_int)
3842 {
3843 (void)syn_list_header(did_header, 999, id);
3844 msg_puts_attr((char_u *)"links to", attr);
3845 msg_putchar(' ');
3846 msg_outtrans(HL_TABLE()[HL_TABLE()[id - 1].sg_link - 1].sg_name);
3847 }
3848}
3849
3850 static void
3851syn_list_flags(nl, flags, attr)
3852 struct name_list *nl;
3853 int flags;
3854 int attr;
3855{
3856 int i;
3857
3858 for (i = 0; nl[i].flag != 0; ++i)
3859 if (flags & nl[i].flag)
3860 {
3861 msg_puts_attr((char_u *)nl[i].name, attr);
3862 msg_putchar(' ');
3863 }
3864}
3865
3866/*
3867 * List one syntax cluster, for ":syntax" or "syntax list syntax_name".
3868 */
3869 static void
3870syn_list_cluster(id)
3871 int id;
3872{
3873 int endcol = 15;
3874
3875 /* slight hack: roughly duplicate the guts of syn_list_header() */
3876 msg_putchar('\n');
3877 msg_outtrans(SYN_CLSTR(curbuf)[id].scl_name);
3878
3879 if (msg_col >= endcol) /* output at least one space */
3880 endcol = msg_col + 1;
3881 if (Columns <= endcol) /* avoid hang for tiny window */
3882 endcol = Columns - 1;
3883
3884 msg_advance(endcol);
3885 if (SYN_CLSTR(curbuf)[id].scl_list != NULL)
3886 {
3887 put_id_list((char_u *)"cluster", SYN_CLSTR(curbuf)[id].scl_list,
3888 hl_attr(HLF_D));
3889 }
3890 else
3891 {
3892 msg_puts_attr((char_u *)"cluster", hl_attr(HLF_D));
3893 msg_puts((char_u *)"=NONE");
3894 }
3895}
3896
3897 static void
3898put_id_list(name, list, attr)
3899 char_u *name;
3900 short *list;
3901 int attr;
3902{
3903 short *p;
3904
3905 msg_puts_attr(name, attr);
3906 msg_putchar('=');
3907 for (p = list; *p; ++p)
3908 {
3909 if (*p >= SYNID_ALLBUT && *p < SYNID_TOP)
3910 {
3911 if (p[1])
3912 MSG_PUTS("ALLBUT");
3913 else
3914 MSG_PUTS("ALL");
3915 }
3916 else if (*p >= SYNID_TOP && *p < SYNID_CONTAINED)
3917 {
3918 MSG_PUTS("TOP");
3919 }
3920 else if (*p >= SYNID_CONTAINED && *p < SYNID_CLUSTER)
3921 {
3922 MSG_PUTS("CONTAINED");
3923 }
3924 else if (*p >= SYNID_CLUSTER)
3925 {
3926 short scl_id = *p - SYNID_CLUSTER;
3927
3928 msg_putchar('@');
3929 msg_outtrans(SYN_CLSTR(curbuf)[scl_id].scl_name);
3930 }
3931 else
3932 msg_outtrans(HL_TABLE()[*p - 1].sg_name);
3933 if (p[1])
3934 msg_putchar(',');
3935 }
3936 msg_putchar(' ');
3937}
3938
3939 static void
3940put_pattern(s, c, spp, attr)
3941 char *s;
3942 int c;
3943 synpat_T *spp;
3944 int attr;
3945{
3946 long n;
3947 int mask;
3948 int first;
3949 static char *sepchars = "/+=-#@\"|'^&";
3950 int i;
3951
3952 /* May have to write "matchgroup=group" */
3953 if (last_matchgroup != spp->sp_syn_match_id)
3954 {
3955 last_matchgroup = spp->sp_syn_match_id;
3956 msg_puts_attr((char_u *)"matchgroup", attr);
3957 msg_putchar('=');
3958 if (last_matchgroup == 0)
3959 msg_outtrans((char_u *)"NONE");
3960 else
3961 msg_outtrans(HL_TABLE()[last_matchgroup - 1].sg_name);
3962 msg_putchar(' ');
3963 }
3964
3965 /* output the name of the pattern and an '=' or ' ' */
3966 msg_puts_attr((char_u *)s, attr);
3967 msg_putchar(c);
3968
3969 /* output the pattern, in between a char that is not in the pattern */
3970 for (i = 0; vim_strchr(spp->sp_pattern, sepchars[i]) != NULL; )
3971 if (sepchars[++i] == NUL)
3972 {
3973 i = 0; /* no good char found, just use the first one */
3974 break;
3975 }
3976 msg_putchar(sepchars[i]);
3977 msg_outtrans(spp->sp_pattern);
3978 msg_putchar(sepchars[i]);
3979
3980 /* output any pattern options */
3981 first = TRUE;
3982 for (i = 0; i < SPO_COUNT; ++i)
3983 {
3984 mask = (1 << i);
3985 if (spp->sp_off_flags & (mask + (mask << SPO_COUNT)))
3986 {
3987 if (!first)
3988 msg_putchar(','); /* separate with commas */
3989 msg_puts((char_u *)spo_name_tab[i]);
3990 n = spp->sp_offsets[i];
3991 if (i != SPO_LC_OFF)
3992 {
3993 if (spp->sp_off_flags & mask)
3994 msg_putchar('s');
3995 else
3996 msg_putchar('e');
3997 if (n > 0)
3998 msg_putchar('+');
3999 }
4000 if (n || i == SPO_LC_OFF)
4001 msg_outnum(n);
4002 first = FALSE;
4003 }
4004 }
4005 msg_putchar(' ');
4006}
4007
4008/*
4009 * List or clear the keywords for one syntax group.
4010 * Return TRUE if the header has been printed.
4011 */
4012 static int
Bram Moolenaardad6b692005-01-25 22:14:34 +00004013syn_list_keywords(id, ht, did_header, attr)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004014 int id;
Bram Moolenaardad6b692005-01-25 22:14:34 +00004015 hashtab_T *ht;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004016 int did_header; /* header has already been printed */
4017 int attr;
4018{
Bram Moolenaar071d4272004-06-13 20:20:40 +00004019 int outlen;
Bram Moolenaardad6b692005-01-25 22:14:34 +00004020 hashitem_T *hi;
4021 keyentry_T *kp;
4022 int todo;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004023 int prev_contained = 0;
4024 short *prev_next_list = NULL;
4025 short *prev_cont_in_list = NULL;
4026 int prev_skipnl = 0;
4027 int prev_skipwhite = 0;
4028 int prev_skipempty = 0;
4029
Bram Moolenaar071d4272004-06-13 20:20:40 +00004030 /*
4031 * Unfortunately, this list of keywords is not sorted on alphabet but on
4032 * hash value...
4033 */
Bram Moolenaara93fa7e2006-04-17 22:14:47 +00004034 todo = (int)ht->ht_used;
Bram Moolenaardad6b692005-01-25 22:14:34 +00004035 for (hi = ht->ht_array; todo > 0 && !got_int; ++hi)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004036 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00004037 if (!HASHITEM_EMPTY(hi))
Bram Moolenaar071d4272004-06-13 20:20:40 +00004038 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00004039 --todo;
4040 for (kp = HI2KE(hi); kp != NULL && !got_int; kp = kp->ke_next)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004041 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00004042 if (kp->k_syn.id == id)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004043 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00004044 if (prev_contained != (kp->flags & HL_CONTAINED)
4045 || prev_skipnl != (kp->flags & HL_SKIPNL)
4046 || prev_skipwhite != (kp->flags & HL_SKIPWHITE)
4047 || prev_skipempty != (kp->flags & HL_SKIPEMPTY)
4048 || prev_cont_in_list != kp->k_syn.cont_in_list
4049 || prev_next_list != kp->next_list)
4050 outlen = 9999;
4051 else
4052 outlen = (int)STRLEN(kp->keyword);
4053 /* output "contained" and "nextgroup" on each line */
4054 if (syn_list_header(did_header, outlen, id))
Bram Moolenaar071d4272004-06-13 20:20:40 +00004055 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00004056 prev_contained = 0;
4057 prev_next_list = NULL;
4058 prev_cont_in_list = NULL;
4059 prev_skipnl = 0;
4060 prev_skipwhite = 0;
4061 prev_skipempty = 0;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004062 }
Bram Moolenaardad6b692005-01-25 22:14:34 +00004063 did_header = TRUE;
4064 if (prev_contained != (kp->flags & HL_CONTAINED))
Bram Moolenaar071d4272004-06-13 20:20:40 +00004065 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00004066 msg_puts_attr((char_u *)"contained", attr);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004067 msg_putchar(' ');
Bram Moolenaardad6b692005-01-25 22:14:34 +00004068 prev_contained = (kp->flags & HL_CONTAINED);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004069 }
Bram Moolenaardad6b692005-01-25 22:14:34 +00004070 if (kp->k_syn.cont_in_list != prev_cont_in_list)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004071 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00004072 put_id_list((char_u *)"containedin",
4073 kp->k_syn.cont_in_list, attr);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004074 msg_putchar(' ');
Bram Moolenaardad6b692005-01-25 22:14:34 +00004075 prev_cont_in_list = kp->k_syn.cont_in_list;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004076 }
Bram Moolenaardad6b692005-01-25 22:14:34 +00004077 if (kp->next_list != prev_next_list)
4078 {
4079 put_id_list((char_u *)"nextgroup", kp->next_list, attr);
4080 msg_putchar(' ');
4081 prev_next_list = kp->next_list;
4082 if (kp->flags & HL_SKIPNL)
4083 {
4084 msg_puts_attr((char_u *)"skipnl", attr);
4085 msg_putchar(' ');
4086 prev_skipnl = (kp->flags & HL_SKIPNL);
4087 }
4088 if (kp->flags & HL_SKIPWHITE)
4089 {
4090 msg_puts_attr((char_u *)"skipwhite", attr);
4091 msg_putchar(' ');
4092 prev_skipwhite = (kp->flags & HL_SKIPWHITE);
4093 }
4094 if (kp->flags & HL_SKIPEMPTY)
4095 {
4096 msg_puts_attr((char_u *)"skipempty", attr);
4097 msg_putchar(' ');
4098 prev_skipempty = (kp->flags & HL_SKIPEMPTY);
4099 }
4100 }
4101 msg_outtrans(kp->keyword);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004102 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00004103 }
4104 }
4105 }
4106
4107 return did_header;
4108}
4109
4110 static void
Bram Moolenaardad6b692005-01-25 22:14:34 +00004111syn_clear_keyword(id, ht)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004112 int id;
Bram Moolenaardad6b692005-01-25 22:14:34 +00004113 hashtab_T *ht;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004114{
Bram Moolenaardad6b692005-01-25 22:14:34 +00004115 hashitem_T *hi;
4116 keyentry_T *kp;
4117 keyentry_T *kp_prev;
4118 keyentry_T *kp_next;
4119 int todo;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004120
Bram Moolenaardad6b692005-01-25 22:14:34 +00004121 hash_lock(ht);
Bram Moolenaara93fa7e2006-04-17 22:14:47 +00004122 todo = (int)ht->ht_used;
Bram Moolenaardad6b692005-01-25 22:14:34 +00004123 for (hi = ht->ht_array; todo > 0; ++hi)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004124 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00004125 if (!HASHITEM_EMPTY(hi))
Bram Moolenaar071d4272004-06-13 20:20:40 +00004126 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00004127 --todo;
4128 kp_prev = NULL;
4129 for (kp = HI2KE(hi); kp != NULL; )
Bram Moolenaar071d4272004-06-13 20:20:40 +00004130 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00004131 if (kp->k_syn.id == id)
4132 {
4133 kp_next = kp->ke_next;
4134 if (kp_prev == NULL)
4135 {
4136 if (kp_next == NULL)
4137 hash_remove(ht, hi);
4138 else
4139 hi->hi_key = KE2HIKEY(kp_next);
4140 }
4141 else
4142 kp_prev->ke_next = kp_next;
4143 vim_free(kp->next_list);
4144 vim_free(kp->k_syn.cont_in_list);
4145 vim_free(kp);
4146 kp = kp_next;
4147 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00004148 else
Bram Moolenaardad6b692005-01-25 22:14:34 +00004149 {
4150 kp_prev = kp;
4151 kp = kp->ke_next;
4152 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00004153 }
4154 }
4155 }
Bram Moolenaardad6b692005-01-25 22:14:34 +00004156 hash_unlock(ht);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004157}
4158
4159/*
Bram Moolenaardad6b692005-01-25 22:14:34 +00004160 * Clear a whole keyword table.
Bram Moolenaar071d4272004-06-13 20:20:40 +00004161 */
4162 static void
Bram Moolenaardad6b692005-01-25 22:14:34 +00004163clear_keywtab(ht)
4164 hashtab_T *ht;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004165{
Bram Moolenaardad6b692005-01-25 22:14:34 +00004166 hashitem_T *hi;
4167 int todo;
4168 keyentry_T *kp;
4169 keyentry_T *kp_next;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004170
Bram Moolenaara93fa7e2006-04-17 22:14:47 +00004171 todo = (int)ht->ht_used;
Bram Moolenaardad6b692005-01-25 22:14:34 +00004172 for (hi = ht->ht_array; todo > 0; ++hi)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004173 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00004174 if (!HASHITEM_EMPTY(hi))
4175 {
4176 --todo;
4177 kp = HI2KE(hi);
4178 for (kp = HI2KE(hi); kp != NULL; kp = kp_next)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004179 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00004180 kp_next = kp->ke_next;
4181 vim_free(kp->next_list);
4182 vim_free(kp->k_syn.cont_in_list);
4183 vim_free(kp);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004184 }
Bram Moolenaardad6b692005-01-25 22:14:34 +00004185 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00004186 }
Bram Moolenaardad6b692005-01-25 22:14:34 +00004187 hash_clear(ht);
4188 hash_init(ht);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004189}
4190
4191/*
4192 * Add a keyword to the list of keywords.
4193 */
4194 static void
4195add_keyword(name, id, flags, cont_in_list, next_list)
4196 char_u *name; /* name of keyword */
4197 int id; /* group ID for this keyword */
4198 int flags; /* flags for this keyword */
4199 short *cont_in_list; /* containedin for this keyword */
4200 short *next_list; /* nextgroup for this keyword */
4201{
Bram Moolenaardad6b692005-01-25 22:14:34 +00004202 keyentry_T *kp;
4203 hashtab_T *ht;
4204 hashitem_T *hi;
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004205 char_u *name_ic;
Bram Moolenaardad6b692005-01-25 22:14:34 +00004206 long_u hash;
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004207 char_u name_folded[MAXKEYWLEN + 1];
Bram Moolenaar071d4272004-06-13 20:20:40 +00004208
4209 if (curbuf->b_syn_ic)
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004210 name_ic = str_foldcase(name, (int)STRLEN(name),
4211 name_folded, MAXKEYWLEN + 1);
4212 else
4213 name_ic = name;
Bram Moolenaardad6b692005-01-25 22:14:34 +00004214 kp = (keyentry_T *)alloc((int)(sizeof(keyentry_T) + STRLEN(name_ic)));
4215 if (kp == NULL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004216 return;
Bram Moolenaardad6b692005-01-25 22:14:34 +00004217 STRCPY(kp->keyword, name_ic);
Bram Moolenaardad6b692005-01-25 22:14:34 +00004218 kp->k_syn.id = id;
4219 kp->k_syn.inc_tag = current_syn_inc_tag;
4220 kp->flags = flags;
4221 kp->k_syn.cont_in_list = copy_id_list(cont_in_list);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004222 if (cont_in_list != NULL)
4223 curbuf->b_syn_containedin = TRUE;
Bram Moolenaardad6b692005-01-25 22:14:34 +00004224 kp->next_list = copy_id_list(next_list);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004225
4226 if (curbuf->b_syn_ic)
Bram Moolenaardad6b692005-01-25 22:14:34 +00004227 ht = &curbuf->b_keywtab_ic;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004228 else
Bram Moolenaardad6b692005-01-25 22:14:34 +00004229 ht = &curbuf->b_keywtab;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004230
Bram Moolenaardad6b692005-01-25 22:14:34 +00004231 hash = hash_hash(kp->keyword);
4232 hi = hash_lookup(ht, kp->keyword, hash);
4233 if (HASHITEM_EMPTY(hi))
Bram Moolenaar071d4272004-06-13 20:20:40 +00004234 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00004235 /* new keyword, add to hashtable */
4236 kp->ke_next = NULL;
4237 hash_add_item(ht, hi, kp->keyword, hash);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004238 }
Bram Moolenaardad6b692005-01-25 22:14:34 +00004239 else
Bram Moolenaar071d4272004-06-13 20:20:40 +00004240 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00004241 /* keyword already exists, prepend to list */
4242 kp->ke_next = HI2KE(hi);
4243 hi->hi_key = KE2HIKEY(kp);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004244 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00004245}
4246
4247/*
4248 * Get the start and end of the group name argument.
4249 * Return a pointer to the first argument.
4250 * Return NULL if the end of the command was found instead of further args.
4251 */
4252 static char_u *
4253get_group_name(arg, name_end)
4254 char_u *arg; /* start of the argument */
4255 char_u **name_end; /* pointer to end of the name */
4256{
4257 char_u *rest;
4258
4259 *name_end = skiptowhite(arg);
4260 rest = skipwhite(*name_end);
4261
4262 /*
4263 * Check if there are enough arguments. The first argument may be a
4264 * pattern, where '|' is allowed, so only check for NUL.
4265 */
4266 if (ends_excmd(*arg) || *rest == NUL)
4267 return NULL;
4268 return rest;
4269}
4270
4271/*
4272 * Check for syntax command option arguments.
4273 * This can be called at any place in the list of arguments, and just picks
4274 * out the arguments that are known. Can be called several times in a row to
4275 * collect all options in between other arguments.
4276 * Return a pointer to the next argument (which isn't an option).
4277 * Return NULL for any error;
4278 */
4279 static char_u *
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004280get_syn_options(arg, opt)
4281 char_u *arg; /* next argument to be checked */
4282 syn_opt_arg_T *opt; /* various things */
Bram Moolenaar071d4272004-06-13 20:20:40 +00004283{
Bram Moolenaar071d4272004-06-13 20:20:40 +00004284 char_u *gname_start, *gname;
4285 int syn_id;
4286 int len;
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004287 char *p;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004288 int i;
4289 int fidx;
4290 static struct flag
4291 {
4292 char *name;
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004293 int argtype;
4294 int flags;
4295 } flagtab[] = { {"cCoOnNtTaAiInNeEdD", 0, HL_CONTAINED},
4296 {"oOnNeElLiInNeE", 0, HL_ONELINE},
4297 {"kKeEeEpPeEnNdD", 0, HL_KEEPEND},
4298 {"eExXtTeEnNdD", 0, HL_EXTEND},
4299 {"eExXcClLuUdDeEnNlL", 0, HL_EXCLUDENL},
4300 {"tTrRaAnNsSpPaArReEnNtT", 0, HL_TRANSP},
4301 {"sSkKiIpPnNlL", 0, HL_SKIPNL},
4302 {"sSkKiIpPwWhHiItTeE", 0, HL_SKIPWHITE},
4303 {"sSkKiIpPeEmMpPtTyY", 0, HL_SKIPEMPTY},
4304 {"gGrRoOuUpPhHeErReE", 0, HL_SYNC_HERE},
4305 {"gGrRoOuUpPtThHeErReE", 0, HL_SYNC_THERE},
4306 {"dDiIsSpPlLaAyY", 0, HL_DISPLAY},
4307 {"fFoOlLdD", 0, HL_FOLD},
4308 {"cCoOnNtTaAiInNsS", 1, 0},
4309 {"cCoOnNtTaAiInNeEdDiInN", 2, 0},
4310 {"nNeExXtTgGrRoOuUpP", 3, 0},
Bram Moolenaar071d4272004-06-13 20:20:40 +00004311 };
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004312 static char *first_letters = "cCoOkKeEtTsSgGdDfFnN";
Bram Moolenaar071d4272004-06-13 20:20:40 +00004313
4314 if (arg == NULL) /* already detected error */
4315 return NULL;
4316
Bram Moolenaar071d4272004-06-13 20:20:40 +00004317 for (;;)
4318 {
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004319 /*
4320 * This is used very often when a large number of keywords is defined.
4321 * Need to skip quickly when no option name is found.
4322 * Also avoid tolower(), it's slow.
4323 */
4324 if (strchr(first_letters, *arg) == NULL)
4325 break;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004326
4327 for (fidx = sizeof(flagtab) / sizeof(struct flag); --fidx >= 0; )
4328 {
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004329 p = flagtab[fidx].name;
4330 for (i = 0, len = 0; p[i] != NUL; i += 2, ++len)
4331 if (arg[len] != p[i] && arg[len] != p[i + 1])
4332 break;
4333 if (p[i] == NUL && (vim_iswhite(arg[len])
4334 || (flagtab[fidx].argtype > 0
4335 ? arg[len] == '='
4336 : ends_excmd(arg[len]))))
Bram Moolenaar071d4272004-06-13 20:20:40 +00004337 {
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004338 if (opt->keyword
4339 && (flagtab[fidx].flags == HL_DISPLAY
4340 || flagtab[fidx].flags == HL_FOLD
4341 || flagtab[fidx].flags == HL_EXTEND))
Bram Moolenaar071d4272004-06-13 20:20:40 +00004342 /* treat "display", "fold" and "extend" as a keyword */
4343 fidx = -1;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004344 break;
4345 }
4346 }
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004347 if (fidx < 0) /* no match found */
4348 break;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004349
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004350 if (flagtab[fidx].argtype == 1)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004351 {
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004352 if (!opt->has_cont_list)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004353 {
4354 EMSG(_("E395: contains argument not accepted here"));
4355 return NULL;
4356 }
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004357 if (get_id_list(&arg, 8, &opt->cont_list) == FAIL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004358 return NULL;
4359 }
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004360 else if (flagtab[fidx].argtype == 2)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004361 {
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004362#if 0 /* cannot happen */
4363 if (opt->cont_in_list == NULL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004364 {
4365 EMSG(_("E396: containedin argument not accepted here"));
4366 return NULL;
4367 }
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004368#endif
4369 if (get_id_list(&arg, 11, &opt->cont_in_list) == FAIL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004370 return NULL;
4371 }
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004372 else if (flagtab[fidx].argtype == 3)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004373 {
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004374 if (get_id_list(&arg, 9, &opt->next_list) == FAIL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004375 return NULL;
4376 }
4377 else
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004378 {
4379 opt->flags |= flagtab[fidx].flags;
4380 arg = skipwhite(arg + len);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004381
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004382 if (flagtab[fidx].flags == HL_SYNC_HERE
4383 || flagtab[fidx].flags == HL_SYNC_THERE)
4384 {
4385 if (opt->sync_idx == NULL)
4386 {
4387 EMSG(_("E393: group[t]here not accepted here"));
4388 return NULL;
4389 }
4390 gname_start = arg;
4391 arg = skiptowhite(arg);
4392 if (gname_start == arg)
4393 return NULL;
4394 gname = vim_strnsave(gname_start, (int)(arg - gname_start));
4395 if (gname == NULL)
4396 return NULL;
4397 if (STRCMP(gname, "NONE") == 0)
4398 *opt->sync_idx = NONE_IDX;
4399 else
4400 {
4401 syn_id = syn_name2id(gname);
4402 for (i = curbuf->b_syn_patterns.ga_len; --i >= 0; )
4403 if (SYN_ITEMS(curbuf)[i].sp_syn.id == syn_id
4404 && SYN_ITEMS(curbuf)[i].sp_type == SPTYPE_START)
4405 {
4406 *opt->sync_idx = i;
4407 break;
4408 }
4409 if (i < 0)
4410 {
4411 EMSG2(_("E394: Didn't find region item for %s"), gname);
4412 vim_free(gname);
4413 return NULL;
4414 }
4415 }
4416
4417 vim_free(gname);
4418 arg = skipwhite(arg);
4419 }
4420#ifdef FEAT_FOLDING
4421 else if (flagtab[fidx].flags == HL_FOLD
4422 && foldmethodIsSyntax(curwin))
4423 /* Need to update folds later. */
4424 foldUpdateAll(curwin);
4425#endif
4426 }
4427 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00004428
4429 return arg;
4430}
4431
4432/*
4433 * Adjustments to syntax item when declared in a ":syn include"'d file.
4434 * Set the contained flag, and if the item is not already contained, add it
4435 * to the specified top-level group, if any.
4436 */
4437 static void
4438syn_incl_toplevel(id, flagsp)
4439 int id;
4440 int *flagsp;
4441{
4442 if ((*flagsp & HL_CONTAINED) || curbuf->b_syn_topgrp == 0)
4443 return;
4444 *flagsp |= HL_CONTAINED;
4445 if (curbuf->b_syn_topgrp >= SYNID_CLUSTER)
4446 {
4447 /* We have to alloc this, because syn_combine_list() will free it. */
4448 short *grp_list = (short *)alloc((unsigned)(2 * sizeof(short)));
4449 int tlg_id = curbuf->b_syn_topgrp - SYNID_CLUSTER;
4450
4451 if (grp_list != NULL)
4452 {
4453 grp_list[0] = id;
4454 grp_list[1] = 0;
4455 syn_combine_list(&SYN_CLSTR(curbuf)[tlg_id].scl_list, &grp_list,
4456 CLUSTER_ADD);
4457 }
4458 }
4459}
4460
4461/*
4462 * Handle ":syntax include [@{group-name}] filename" command.
4463 */
4464/* ARGSUSED */
4465 static void
4466syn_cmd_include(eap, syncing)
4467 exarg_T *eap;
4468 int syncing; /* not used */
4469{
4470 char_u *arg = eap->arg;
4471 int sgl_id = 1;
4472 char_u *group_name_end;
4473 char_u *rest;
4474 char_u *errormsg = NULL;
4475 int prev_toplvl_grp;
4476 int prev_syn_inc_tag;
4477 int source = FALSE;
4478
4479 eap->nextcmd = find_nextcmd(arg);
4480 if (eap->skip)
4481 return;
4482
4483 if (arg[0] == '@')
4484 {
4485 ++arg;
4486 rest = get_group_name(arg, &group_name_end);
4487 if (rest == NULL)
4488 {
4489 EMSG((char_u *)_("E397: Filename required"));
4490 return;
4491 }
4492 sgl_id = syn_check_cluster(arg, (int)(group_name_end - arg));
4493 /* separate_nextcmd() and expand_filename() depend on this */
4494 eap->arg = rest;
4495 }
4496
4497 /*
4498 * Everything that's left, up to the next command, should be the
4499 * filename to include.
4500 */
4501 eap->argt |= (XFILE | NOSPC);
4502 separate_nextcmd(eap);
4503 if (*eap->arg == '<' || *eap->arg == '$' || mch_isFullName(eap->arg))
4504 {
4505 /* For an absolute path, "$VIM/..." or "<sfile>.." we ":source" the
4506 * file. Need to expand the file name first. In other cases
4507 * ":runtime!" is used. */
4508 source = TRUE;
4509 if (expand_filename(eap, syn_cmdlinep, &errormsg) == FAIL)
4510 {
4511 if (errormsg != NULL)
4512 EMSG(errormsg);
4513 return;
4514 }
4515 }
4516
4517 /*
4518 * Save and restore the existing top-level grouplist id and ":syn
4519 * include" tag around the actual inclusion.
4520 */
4521 prev_syn_inc_tag = current_syn_inc_tag;
4522 current_syn_inc_tag = ++running_syn_inc_tag;
4523 prev_toplvl_grp = curbuf->b_syn_topgrp;
4524 curbuf->b_syn_topgrp = sgl_id;
Bram Moolenaar786b1dc2007-06-28 10:03:45 +00004525 if (source ? do_source(eap->arg, FALSE, DOSO_NONE) == FAIL
4526 : source_runtime(eap->arg, TRUE) == FAIL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004527 EMSG2(_(e_notopen), eap->arg);
4528 curbuf->b_syn_topgrp = prev_toplvl_grp;
4529 current_syn_inc_tag = prev_syn_inc_tag;
4530}
4531
4532/*
4533 * Handle ":syntax keyword {group-name} [{option}] keyword .." command.
4534 */
4535/* ARGSUSED */
4536 static void
4537syn_cmd_keyword(eap, syncing)
4538 exarg_T *eap;
4539 int syncing; /* not used */
4540{
4541 char_u *arg = eap->arg;
4542 char_u *group_name_end;
4543 int syn_id;
4544 char_u *rest;
4545 char_u *keyword_copy;
4546 char_u *p;
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004547 char_u *kw;
4548 syn_opt_arg_T syn_opt_arg;
4549 int cnt;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004550
4551 rest = get_group_name(arg, &group_name_end);
4552
4553 if (rest != NULL)
4554 {
4555 syn_id = syn_check_group(arg, (int)(group_name_end - arg));
4556
4557 /* allocate a buffer, for removing the backslashes in the keyword */
4558 keyword_copy = alloc((unsigned)STRLEN(rest) + 1);
4559 if (keyword_copy != NULL)
4560 {
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004561 syn_opt_arg.flags = 0;
4562 syn_opt_arg.keyword = TRUE;
4563 syn_opt_arg.sync_idx = NULL;
4564 syn_opt_arg.has_cont_list = FALSE;
4565 syn_opt_arg.cont_in_list = NULL;
4566 syn_opt_arg.next_list = NULL;
4567
Bram Moolenaar071d4272004-06-13 20:20:40 +00004568 /*
4569 * The options given apply to ALL keywords, so all options must be
4570 * found before keywords can be created.
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004571 * 1: collect the options and copy the keywords to keyword_copy.
Bram Moolenaar071d4272004-06-13 20:20:40 +00004572 */
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004573 cnt = 0;
4574 p = keyword_copy;
4575 for ( ; rest != NULL && !ends_excmd(*rest); rest = skipwhite(rest))
Bram Moolenaar071d4272004-06-13 20:20:40 +00004576 {
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004577 rest = get_syn_options(rest, &syn_opt_arg);
4578 if (rest == NULL || ends_excmd(*rest))
4579 break;
4580 /* Copy the keyword, removing backslashes, and add a NUL. */
4581 while (*rest != NUL && !vim_iswhite(*rest))
Bram Moolenaar071d4272004-06-13 20:20:40 +00004582 {
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004583 if (*rest == '\\' && rest[1] != NUL)
4584 ++rest;
4585 *p++ = *rest++;
4586 }
4587 *p++ = NUL;
4588 ++cnt;
4589 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00004590
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004591 if (!eap->skip)
4592 {
4593 /* Adjust flags for use of ":syn include". */
4594 syn_incl_toplevel(syn_id, &syn_opt_arg.flags);
4595
4596 /*
4597 * 2: Add an entry for each keyword.
4598 */
4599 for (kw = keyword_copy; --cnt >= 0; kw += STRLEN(kw) + 1)
4600 {
4601 for (p = vim_strchr(kw, '['); ; )
4602 {
4603 if (p != NULL)
4604 *p = NUL;
4605 add_keyword(kw, syn_id, syn_opt_arg.flags,
4606 syn_opt_arg.cont_in_list,
4607 syn_opt_arg.next_list);
Bram Moolenaar26a60b42005-02-22 08:49:11 +00004608 if (p == NULL)
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004609 break;
Bram Moolenaar26a60b42005-02-22 08:49:11 +00004610 if (p[1] == NUL)
4611 {
Bram Moolenaarfc1421e2006-04-20 22:17:20 +00004612 EMSG2(_("E789: Missing ']': %s"), kw);
Bram Moolenaar26a60b42005-02-22 08:49:11 +00004613 kw = p + 2; /* skip over the NUL */
4614 break;
4615 }
4616 if (p[1] == ']')
4617 {
4618 kw = p + 1; /* skip over the "]" */
4619 break;
4620 }
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004621#ifdef FEAT_MBYTE
4622 if (has_mbyte)
4623 {
Bram Moolenaar0fa313a2005-08-10 21:07:57 +00004624 int l = (*mb_ptr2len)(p + 1);
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004625
4626 mch_memmove(p, p + 1, l);
4627 p += l;
4628 }
4629 else
Bram Moolenaar071d4272004-06-13 20:20:40 +00004630#endif
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004631 {
4632 p[0] = p[1];
4633 ++p;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004634 }
4635 }
4636 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00004637 }
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004638
Bram Moolenaar071d4272004-06-13 20:20:40 +00004639 vim_free(keyword_copy);
Bram Moolenaar26a60b42005-02-22 08:49:11 +00004640 vim_free(syn_opt_arg.cont_in_list);
4641 vim_free(syn_opt_arg.next_list);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004642 }
4643 }
4644
4645 if (rest != NULL)
4646 eap->nextcmd = check_nextcmd(rest);
4647 else
4648 EMSG2(_(e_invarg2), arg);
4649
Bram Moolenaar1c8f93f2006-03-12 22:10:07 +00004650 redraw_curbuf_later(SOME_VALID);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004651 syn_stack_free_all(curbuf); /* Need to recompute all syntax. */
4652}
4653
4654/*
4655 * Handle ":syntax match {name} [{options}] {pattern} [{options}]".
4656 *
4657 * Also ":syntax sync match {name} [[grouphere | groupthere] {group-name}] .."
4658 */
4659 static void
4660syn_cmd_match(eap, syncing)
4661 exarg_T *eap;
4662 int syncing; /* TRUE for ":syntax sync match .. " */
4663{
4664 char_u *arg = eap->arg;
4665 char_u *group_name_end;
4666 char_u *rest;
4667 synpat_T item; /* the item found in the line */
4668 int syn_id;
4669 int idx;
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004670 syn_opt_arg_T syn_opt_arg;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004671 int sync_idx = 0;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004672
4673 /* Isolate the group name, check for validity */
4674 rest = get_group_name(arg, &group_name_end);
4675
4676 /* Get options before the pattern */
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004677 syn_opt_arg.flags = 0;
4678 syn_opt_arg.keyword = FALSE;
4679 syn_opt_arg.sync_idx = syncing ? &sync_idx : NULL;
4680 syn_opt_arg.has_cont_list = TRUE;
4681 syn_opt_arg.cont_list = NULL;
4682 syn_opt_arg.cont_in_list = NULL;
4683 syn_opt_arg.next_list = NULL;
4684 rest = get_syn_options(rest, &syn_opt_arg);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004685
4686 /* get the pattern. */
4687 init_syn_patterns();
4688 vim_memset(&item, 0, sizeof(item));
4689 rest = get_syn_pattern(rest, &item);
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004690 if (vim_regcomp_had_eol() && !(syn_opt_arg.flags & HL_EXCLUDENL))
4691 syn_opt_arg.flags |= HL_HAS_EOL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004692
4693 /* Get options after the pattern */
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004694 rest = get_syn_options(rest, &syn_opt_arg);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004695
4696 if (rest != NULL) /* all arguments are valid */
4697 {
4698 /*
4699 * Check for trailing command and illegal trailing arguments.
4700 */
4701 eap->nextcmd = check_nextcmd(rest);
4702 if (!ends_excmd(*rest) || eap->skip)
4703 rest = NULL;
4704 else if (ga_grow(&curbuf->b_syn_patterns, 1) != FAIL
4705 && (syn_id = syn_check_group(arg,
4706 (int)(group_name_end - arg))) != 0)
4707 {
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004708 syn_incl_toplevel(syn_id, &syn_opt_arg.flags);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004709 /*
4710 * Store the pattern in the syn_items list
4711 */
4712 idx = curbuf->b_syn_patterns.ga_len;
4713 SYN_ITEMS(curbuf)[idx] = item;
4714 SYN_ITEMS(curbuf)[idx].sp_syncing = syncing;
4715 SYN_ITEMS(curbuf)[idx].sp_type = SPTYPE_MATCH;
4716 SYN_ITEMS(curbuf)[idx].sp_syn.id = syn_id;
4717 SYN_ITEMS(curbuf)[idx].sp_syn.inc_tag = current_syn_inc_tag;
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004718 SYN_ITEMS(curbuf)[idx].sp_flags = syn_opt_arg.flags;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004719 SYN_ITEMS(curbuf)[idx].sp_sync_idx = sync_idx;
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004720 SYN_ITEMS(curbuf)[idx].sp_cont_list = syn_opt_arg.cont_list;
4721 SYN_ITEMS(curbuf)[idx].sp_syn.cont_in_list =
4722 syn_opt_arg.cont_in_list;
4723 if (syn_opt_arg.cont_in_list != NULL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004724 curbuf->b_syn_containedin = TRUE;
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004725 SYN_ITEMS(curbuf)[idx].sp_next_list = syn_opt_arg.next_list;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004726 ++curbuf->b_syn_patterns.ga_len;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004727
4728 /* remember that we found a match for syncing on */
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004729 if (syn_opt_arg.flags & (HL_SYNC_HERE|HL_SYNC_THERE))
Bram Moolenaar071d4272004-06-13 20:20:40 +00004730 curbuf->b_syn_sync_flags |= SF_MATCH;
4731#ifdef FEAT_FOLDING
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004732 if (syn_opt_arg.flags & HL_FOLD)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004733 ++curbuf->b_syn_folditems;
4734#endif
4735
Bram Moolenaar1c8f93f2006-03-12 22:10:07 +00004736 redraw_curbuf_later(SOME_VALID);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004737 syn_stack_free_all(curbuf); /* Need to recompute all syntax. */
4738 return; /* don't free the progs and patterns now */
4739 }
4740 }
4741
4742 /*
4743 * Something failed, free the allocated memory.
4744 */
4745 vim_free(item.sp_prog);
4746 vim_free(item.sp_pattern);
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004747 vim_free(syn_opt_arg.cont_list);
4748 vim_free(syn_opt_arg.cont_in_list);
4749 vim_free(syn_opt_arg.next_list);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004750
4751 if (rest == NULL)
4752 EMSG2(_(e_invarg2), arg);
4753}
4754
4755/*
4756 * Handle ":syntax region {group-name} [matchgroup={group-name}]
4757 * start {start} .. [skip {skip}] end {end} .. [{options}]".
4758 */
4759 static void
4760syn_cmd_region(eap, syncing)
4761 exarg_T *eap;
4762 int syncing; /* TRUE for ":syntax sync region .." */
4763{
4764 char_u *arg = eap->arg;
4765 char_u *group_name_end;
4766 char_u *rest; /* next arg, NULL on error */
4767 char_u *key_end;
4768 char_u *key = NULL;
4769 char_u *p;
4770 int item;
4771#define ITEM_START 0
4772#define ITEM_SKIP 1
4773#define ITEM_END 2
4774#define ITEM_MATCHGROUP 3
4775 struct pat_ptr
4776 {
4777 synpat_T *pp_synp; /* pointer to syn_pattern */
4778 int pp_matchgroup_id; /* matchgroup ID */
4779 struct pat_ptr *pp_next; /* pointer to next pat_ptr */
4780 } *(pat_ptrs[3]);
4781 /* patterns found in the line */
4782 struct pat_ptr *ppp;
4783 struct pat_ptr *ppp_next;
4784 int pat_count = 0; /* nr of syn_patterns found */
4785 int syn_id;
4786 int matchgroup_id = 0;
4787 int not_enough = FALSE; /* not enough arguments */
4788 int illegal = FALSE; /* illegal arguments */
4789 int success = FALSE;
4790 int idx;
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004791 syn_opt_arg_T syn_opt_arg;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004792
4793 /* Isolate the group name, check for validity */
4794 rest = get_group_name(arg, &group_name_end);
4795
4796 pat_ptrs[0] = NULL;
4797 pat_ptrs[1] = NULL;
4798 pat_ptrs[2] = NULL;
4799
4800 init_syn_patterns();
4801
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004802 syn_opt_arg.flags = 0;
4803 syn_opt_arg.keyword = FALSE;
4804 syn_opt_arg.sync_idx = NULL;
4805 syn_opt_arg.has_cont_list = TRUE;
4806 syn_opt_arg.cont_list = NULL;
4807 syn_opt_arg.cont_in_list = NULL;
4808 syn_opt_arg.next_list = NULL;
4809
Bram Moolenaar071d4272004-06-13 20:20:40 +00004810 /*
4811 * get the options, patterns and matchgroup.
4812 */
4813 while (rest != NULL && !ends_excmd(*rest))
4814 {
4815 /* Check for option arguments */
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004816 rest = get_syn_options(rest, &syn_opt_arg);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004817 if (rest == NULL || ends_excmd(*rest))
4818 break;
4819
4820 /* must be a pattern or matchgroup then */
4821 key_end = rest;
4822 while (*key_end && !vim_iswhite(*key_end) && *key_end != '=')
4823 ++key_end;
4824 vim_free(key);
4825 key = vim_strnsave_up(rest, (int)(key_end - rest));
4826 if (key == NULL) /* out of memory */
4827 {
4828 rest = NULL;
4829 break;
4830 }
4831 if (STRCMP(key, "MATCHGROUP") == 0)
4832 item = ITEM_MATCHGROUP;
4833 else if (STRCMP(key, "START") == 0)
4834 item = ITEM_START;
4835 else if (STRCMP(key, "END") == 0)
4836 item = ITEM_END;
4837 else if (STRCMP(key, "SKIP") == 0)
4838 {
4839 if (pat_ptrs[ITEM_SKIP] != NULL) /* one skip pattern allowed */
4840 {
4841 illegal = TRUE;
4842 break;
4843 }
4844 item = ITEM_SKIP;
4845 }
4846 else
4847 break;
4848 rest = skipwhite(key_end);
4849 if (*rest != '=')
4850 {
4851 rest = NULL;
4852 EMSG2(_("E398: Missing '=': %s"), arg);
4853 break;
4854 }
4855 rest = skipwhite(rest + 1);
4856 if (*rest == NUL)
4857 {
4858 not_enough = TRUE;
4859 break;
4860 }
4861
4862 if (item == ITEM_MATCHGROUP)
4863 {
4864 p = skiptowhite(rest);
4865 if ((p - rest == 4 && STRNCMP(rest, "NONE", 4) == 0) || eap->skip)
4866 matchgroup_id = 0;
4867 else
4868 {
4869 matchgroup_id = syn_check_group(rest, (int)(p - rest));
4870 if (matchgroup_id == 0)
4871 {
4872 illegal = TRUE;
4873 break;
4874 }
4875 }
4876 rest = skipwhite(p);
4877 }
4878 else
4879 {
4880 /*
4881 * Allocate room for a syn_pattern, and link it in the list of
4882 * syn_patterns for this item, at the start (because the list is
4883 * used from end to start).
4884 */
4885 ppp = (struct pat_ptr *)alloc((unsigned)sizeof(struct pat_ptr));
4886 if (ppp == NULL)
4887 {
4888 rest = NULL;
4889 break;
4890 }
4891 ppp->pp_next = pat_ptrs[item];
4892 pat_ptrs[item] = ppp;
4893 ppp->pp_synp = (synpat_T *)alloc_clear((unsigned)sizeof(synpat_T));
4894 if (ppp->pp_synp == NULL)
4895 {
4896 rest = NULL;
4897 break;
4898 }
4899
4900 /*
4901 * Get the syntax pattern and the following offset(s).
4902 */
4903 /* Enable the appropriate \z specials. */
4904 if (item == ITEM_START)
4905 reg_do_extmatch = REX_SET;
4906 else if (item == ITEM_SKIP || item == ITEM_END)
4907 reg_do_extmatch = REX_USE;
4908 rest = get_syn_pattern(rest, ppp->pp_synp);
4909 reg_do_extmatch = 0;
4910 if (item == ITEM_END && vim_regcomp_had_eol()
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004911 && !(syn_opt_arg.flags & HL_EXCLUDENL))
Bram Moolenaar071d4272004-06-13 20:20:40 +00004912 ppp->pp_synp->sp_flags |= HL_HAS_EOL;
4913 ppp->pp_matchgroup_id = matchgroup_id;
4914 ++pat_count;
4915 }
4916 }
4917 vim_free(key);
4918 if (illegal || not_enough)
4919 rest = NULL;
4920
4921 /*
4922 * Must have a "start" and "end" pattern.
4923 */
4924 if (rest != NULL && (pat_ptrs[ITEM_START] == NULL ||
4925 pat_ptrs[ITEM_END] == NULL))
4926 {
4927 not_enough = TRUE;
4928 rest = NULL;
4929 }
4930
4931 if (rest != NULL)
4932 {
4933 /*
4934 * Check for trailing garbage or command.
4935 * If OK, add the item.
4936 */
4937 eap->nextcmd = check_nextcmd(rest);
4938 if (!ends_excmd(*rest) || eap->skip)
4939 rest = NULL;
4940 else if (ga_grow(&(curbuf->b_syn_patterns), pat_count) != FAIL
4941 && (syn_id = syn_check_group(arg,
4942 (int)(group_name_end - arg))) != 0)
4943 {
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004944 syn_incl_toplevel(syn_id, &syn_opt_arg.flags);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004945 /*
4946 * Store the start/skip/end in the syn_items list
4947 */
4948 idx = curbuf->b_syn_patterns.ga_len;
4949 for (item = ITEM_START; item <= ITEM_END; ++item)
4950 {
4951 for (ppp = pat_ptrs[item]; ppp != NULL; ppp = ppp->pp_next)
4952 {
4953 SYN_ITEMS(curbuf)[idx] = *(ppp->pp_synp);
4954 SYN_ITEMS(curbuf)[idx].sp_syncing = syncing;
4955 SYN_ITEMS(curbuf)[idx].sp_type =
4956 (item == ITEM_START) ? SPTYPE_START :
4957 (item == ITEM_SKIP) ? SPTYPE_SKIP : SPTYPE_END;
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004958 SYN_ITEMS(curbuf)[idx].sp_flags |= syn_opt_arg.flags;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004959 SYN_ITEMS(curbuf)[idx].sp_syn.id = syn_id;
4960 SYN_ITEMS(curbuf)[idx].sp_syn.inc_tag = current_syn_inc_tag;
4961 SYN_ITEMS(curbuf)[idx].sp_syn_match_id =
4962 ppp->pp_matchgroup_id;
4963 if (item == ITEM_START)
4964 {
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004965 SYN_ITEMS(curbuf)[idx].sp_cont_list =
4966 syn_opt_arg.cont_list;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004967 SYN_ITEMS(curbuf)[idx].sp_syn.cont_in_list =
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004968 syn_opt_arg.cont_in_list;
4969 if (syn_opt_arg.cont_in_list != NULL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004970 curbuf->b_syn_containedin = TRUE;
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004971 SYN_ITEMS(curbuf)[idx].sp_next_list =
4972 syn_opt_arg.next_list;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004973 }
4974 ++curbuf->b_syn_patterns.ga_len;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004975 ++idx;
4976#ifdef FEAT_FOLDING
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004977 if (syn_opt_arg.flags & HL_FOLD)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004978 ++curbuf->b_syn_folditems;
4979#endif
4980 }
4981 }
4982
Bram Moolenaar1c8f93f2006-03-12 22:10:07 +00004983 redraw_curbuf_later(SOME_VALID);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004984 syn_stack_free_all(curbuf); /* Need to recompute all syntax. */
4985 success = TRUE; /* don't free the progs and patterns now */
4986 }
4987 }
4988
4989 /*
4990 * Free the allocated memory.
4991 */
4992 for (item = ITEM_START; item <= ITEM_END; ++item)
4993 for (ppp = pat_ptrs[item]; ppp != NULL; ppp = ppp_next)
4994 {
4995 if (!success)
4996 {
4997 vim_free(ppp->pp_synp->sp_prog);
4998 vim_free(ppp->pp_synp->sp_pattern);
4999 }
5000 vim_free(ppp->pp_synp);
5001 ppp_next = ppp->pp_next;
5002 vim_free(ppp);
5003 }
5004
5005 if (!success)
5006 {
Bram Moolenaar6ac54292005-02-02 23:07:25 +00005007 vim_free(syn_opt_arg.cont_list);
5008 vim_free(syn_opt_arg.cont_in_list);
5009 vim_free(syn_opt_arg.next_list);
Bram Moolenaar071d4272004-06-13 20:20:40 +00005010 if (not_enough)
5011 EMSG2(_("E399: Not enough arguments: syntax region %s"), arg);
5012 else if (illegal || rest == NULL)
5013 EMSG2(_(e_invarg2), arg);
5014 }
5015}
5016
5017/*
5018 * A simple syntax group ID comparison function suitable for use in qsort()
5019 */
5020 static int
5021#ifdef __BORLANDC__
5022_RTLENTRYF
5023#endif
5024syn_compare_stub(v1, v2)
5025 const void *v1;
5026 const void *v2;
5027{
5028 const short *s1 = v1;
5029 const short *s2 = v2;
5030
5031 return (*s1 > *s2 ? 1 : *s1 < *s2 ? -1 : 0);
5032}
5033
5034/*
5035 * Combines lists of syntax clusters.
5036 * *clstr1 and *clstr2 must both be allocated memory; they will be consumed.
5037 */
5038 static void
5039syn_combine_list(clstr1, clstr2, list_op)
5040 short **clstr1;
5041 short **clstr2;
5042 int list_op;
5043{
5044 int count1 = 0;
5045 int count2 = 0;
5046 short *g1;
5047 short *g2;
5048 short *clstr = NULL;
5049 int count;
5050 int round;
5051
5052 /*
5053 * Handle degenerate cases.
5054 */
5055 if (*clstr2 == NULL)
5056 return;
5057 if (*clstr1 == NULL || list_op == CLUSTER_REPLACE)
5058 {
5059 if (list_op == CLUSTER_REPLACE)
5060 vim_free(*clstr1);
5061 if (list_op == CLUSTER_REPLACE || list_op == CLUSTER_ADD)
5062 *clstr1 = *clstr2;
5063 else
5064 vim_free(*clstr2);
5065 return;
5066 }
5067
5068 for (g1 = *clstr1; *g1; g1++)
5069 ++count1;
5070 for (g2 = *clstr2; *g2; g2++)
5071 ++count2;
5072
5073 /*
5074 * For speed purposes, sort both lists.
5075 */
5076 qsort(*clstr1, (size_t)count1, sizeof(short), syn_compare_stub);
5077 qsort(*clstr2, (size_t)count2, sizeof(short), syn_compare_stub);
5078
5079 /*
5080 * We proceed in two passes; in round 1, we count the elements to place
5081 * in the new list, and in round 2, we allocate and populate the new
5082 * list. For speed, we use a mergesort-like method, adding the smaller
5083 * of the current elements in each list to the new list.
5084 */
5085 for (round = 1; round <= 2; round++)
5086 {
5087 g1 = *clstr1;
5088 g2 = *clstr2;
5089 count = 0;
5090
5091 /*
5092 * First, loop through the lists until one of them is empty.
5093 */
5094 while (*g1 && *g2)
5095 {
5096 /*
5097 * We always want to add from the first list.
5098 */
5099 if (*g1 < *g2)
5100 {
5101 if (round == 2)
5102 clstr[count] = *g1;
5103 count++;
5104 g1++;
5105 continue;
5106 }
5107 /*
5108 * We only want to add from the second list if we're adding the
5109 * lists.
5110 */
5111 if (list_op == CLUSTER_ADD)
5112 {
5113 if (round == 2)
5114 clstr[count] = *g2;
5115 count++;
5116 }
5117 if (*g1 == *g2)
5118 g1++;
5119 g2++;
5120 }
5121
5122 /*
5123 * Now add the leftovers from whichever list didn't get finished
5124 * first. As before, we only want to add from the second list if
5125 * we're adding the lists.
5126 */
5127 for (; *g1; g1++, count++)
5128 if (round == 2)
5129 clstr[count] = *g1;
5130 if (list_op == CLUSTER_ADD)
5131 for (; *g2; g2++, count++)
5132 if (round == 2)
5133 clstr[count] = *g2;
5134
5135 if (round == 1)
5136 {
5137 /*
5138 * If the group ended up empty, we don't need to allocate any
5139 * space for it.
5140 */
5141 if (count == 0)
5142 {
5143 clstr = NULL;
5144 break;
5145 }
5146 clstr = (short *)alloc((unsigned)((count + 1) * sizeof(short)));
5147 if (clstr == NULL)
5148 break;
5149 clstr[count] = 0;
5150 }
5151 }
5152
5153 /*
5154 * Finally, put the new list in place.
5155 */
5156 vim_free(*clstr1);
5157 vim_free(*clstr2);
5158 *clstr1 = clstr;
5159}
5160
5161/*
5162 * Lookup a syntax cluster name and return it's ID.
5163 * If it is not found, 0 is returned.
5164 */
5165 static int
5166syn_scl_name2id(name)
5167 char_u *name;
5168{
5169 int i;
5170 char_u *name_u;
5171
5172 /* Avoid using stricmp() too much, it's slow on some systems */
5173 name_u = vim_strsave_up(name);
5174 if (name_u == NULL)
5175 return 0;
5176 for (i = curbuf->b_syn_clusters.ga_len; --i >= 0; )
5177 if (SYN_CLSTR(curbuf)[i].scl_name_u != NULL
5178 && STRCMP(name_u, SYN_CLSTR(curbuf)[i].scl_name_u) == 0)
5179 break;
5180 vim_free(name_u);
5181 return (i < 0 ? 0 : i + SYNID_CLUSTER);
5182}
5183
5184/*
5185 * Like syn_scl_name2id(), but take a pointer + length argument.
5186 */
5187 static int
5188syn_scl_namen2id(linep, len)
5189 char_u *linep;
5190 int len;
5191{
5192 char_u *name;
5193 int id = 0;
5194
5195 name = vim_strnsave(linep, len);
5196 if (name != NULL)
5197 {
5198 id = syn_scl_name2id(name);
5199 vim_free(name);
5200 }
5201 return id;
5202}
5203
5204/*
5205 * Find syntax cluster name in the table and return it's ID.
5206 * The argument is a pointer to the name and the length of the name.
5207 * If it doesn't exist yet, a new entry is created.
5208 * Return 0 for failure.
5209 */
5210 static int
5211syn_check_cluster(pp, len)
5212 char_u *pp;
5213 int len;
5214{
5215 int id;
5216 char_u *name;
5217
5218 name = vim_strnsave(pp, len);
5219 if (name == NULL)
5220 return 0;
5221
5222 id = syn_scl_name2id(name);
5223 if (id == 0) /* doesn't exist yet */
5224 id = syn_add_cluster(name);
5225 else
5226 vim_free(name);
5227 return id;
5228}
5229
5230/*
5231 * Add new syntax cluster and return it's ID.
5232 * "name" must be an allocated string, it will be consumed.
5233 * Return 0 for failure.
5234 */
5235 static int
5236syn_add_cluster(name)
Bram Moolenaar217ad922005-03-20 22:37:15 +00005237 char_u *name;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005238{
Bram Moolenaar217ad922005-03-20 22:37:15 +00005239 int len;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005240
5241 /*
5242 * First call for this growarray: init growing array.
5243 */
5244 if (curbuf->b_syn_clusters.ga_data == NULL)
5245 {
Bram Moolenaar217ad922005-03-20 22:37:15 +00005246 curbuf->b_syn_clusters.ga_itemsize = sizeof(syn_cluster_T);
Bram Moolenaar071d4272004-06-13 20:20:40 +00005247 curbuf->b_syn_clusters.ga_growsize = 10;
5248 }
5249
5250 /*
5251 * Make room for at least one other cluster entry.
5252 */
5253 if (ga_grow(&curbuf->b_syn_clusters, 1) == FAIL)
5254 {
5255 vim_free(name);
5256 return 0;
5257 }
5258 len = curbuf->b_syn_clusters.ga_len;
5259
Bram Moolenaar217ad922005-03-20 22:37:15 +00005260 vim_memset(&(SYN_CLSTR(curbuf)[len]), 0, sizeof(syn_cluster_T));
Bram Moolenaar071d4272004-06-13 20:20:40 +00005261 SYN_CLSTR(curbuf)[len].scl_name = name;
5262 SYN_CLSTR(curbuf)[len].scl_name_u = vim_strsave_up(name);
5263 SYN_CLSTR(curbuf)[len].scl_list = NULL;
5264 ++curbuf->b_syn_clusters.ga_len;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005265
Bram Moolenaar217ad922005-03-20 22:37:15 +00005266 if (STRICMP(name, "Spell") == 0)
5267 curbuf->b_spell_cluster_id = len + SYNID_CLUSTER;
Bram Moolenaar6bb68362005-03-22 23:03:44 +00005268 if (STRICMP(name, "NoSpell") == 0)
5269 curbuf->b_nospell_cluster_id = len + SYNID_CLUSTER;
Bram Moolenaar217ad922005-03-20 22:37:15 +00005270
Bram Moolenaar071d4272004-06-13 20:20:40 +00005271 return len + SYNID_CLUSTER;
5272}
5273
5274/*
5275 * Handle ":syntax cluster {cluster-name} [contains={groupname},..]
5276 * [add={groupname},..] [remove={groupname},..]".
5277 */
5278/* ARGSUSED */
5279 static void
5280syn_cmd_cluster(eap, syncing)
5281 exarg_T *eap;
5282 int syncing; /* not used */
5283{
5284 char_u *arg = eap->arg;
5285 char_u *group_name_end;
5286 char_u *rest;
5287 int scl_id;
5288 short *clstr_list;
5289 int got_clstr = FALSE;
5290 int opt_len;
5291 int list_op;
5292
5293 eap->nextcmd = find_nextcmd(arg);
5294 if (eap->skip)
5295 return;
5296
5297 rest = get_group_name(arg, &group_name_end);
5298
5299 if (rest != NULL)
5300 {
5301 scl_id = syn_check_cluster(arg, (int)(group_name_end - arg))
Bram Moolenaar217ad922005-03-20 22:37:15 +00005302 - SYNID_CLUSTER;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005303
5304 for (;;)
5305 {
5306 if (STRNICMP(rest, "add", 3) == 0
5307 && (vim_iswhite(rest[3]) || rest[3] == '='))
5308 {
5309 opt_len = 3;
5310 list_op = CLUSTER_ADD;
5311 }
5312 else if (STRNICMP(rest, "remove", 6) == 0
5313 && (vim_iswhite(rest[6]) || rest[6] == '='))
5314 {
5315 opt_len = 6;
5316 list_op = CLUSTER_SUBTRACT;
5317 }
5318 else if (STRNICMP(rest, "contains", 8) == 0
5319 && (vim_iswhite(rest[8]) || rest[8] == '='))
5320 {
5321 opt_len = 8;
5322 list_op = CLUSTER_REPLACE;
5323 }
5324 else
5325 break;
5326
5327 clstr_list = NULL;
5328 if (get_id_list(&rest, opt_len, &clstr_list) == FAIL)
5329 {
5330 EMSG2(_(e_invarg2), rest);
5331 break;
5332 }
5333 syn_combine_list(&SYN_CLSTR(curbuf)[scl_id].scl_list,
5334 &clstr_list, list_op);
5335 got_clstr = TRUE;
5336 }
5337
5338 if (got_clstr)
5339 {
Bram Moolenaar1c8f93f2006-03-12 22:10:07 +00005340 redraw_curbuf_later(SOME_VALID);
Bram Moolenaar071d4272004-06-13 20:20:40 +00005341 syn_stack_free_all(curbuf); /* Need to recompute all syntax. */
5342 }
5343 }
5344
5345 if (!got_clstr)
5346 EMSG(_("E400: No cluster specified"));
5347 if (rest == NULL || !ends_excmd(*rest))
5348 EMSG2(_(e_invarg2), arg);
5349}
5350
5351/*
5352 * On first call for current buffer: Init growing array.
5353 */
5354 static void
5355init_syn_patterns()
5356{
5357 curbuf->b_syn_patterns.ga_itemsize = sizeof(synpat_T);
5358 curbuf->b_syn_patterns.ga_growsize = 10;
5359}
5360
5361/*
5362 * Get one pattern for a ":syntax match" or ":syntax region" command.
5363 * Stores the pattern and program in a synpat_T.
5364 * Returns a pointer to the next argument, or NULL in case of an error.
5365 */
5366 static char_u *
5367get_syn_pattern(arg, ci)
5368 char_u *arg;
5369 synpat_T *ci;
5370{
5371 char_u *end;
5372 int *p;
5373 int idx;
5374 char_u *cpo_save;
5375
5376 /* need at least three chars */
5377 if (arg == NULL || arg[1] == NUL || arg[2] == NUL)
5378 return NULL;
5379
5380 end = skip_regexp(arg + 1, *arg, TRUE, NULL);
5381 if (*end != *arg) /* end delimiter not found */
5382 {
5383 EMSG2(_("E401: Pattern delimiter not found: %s"), arg);
5384 return NULL;
5385 }
5386 /* store the pattern and compiled regexp program */
5387 if ((ci->sp_pattern = vim_strnsave(arg + 1, (int)(end - arg - 1))) == NULL)
5388 return NULL;
5389
5390 /* Make 'cpoptions' empty, to avoid the 'l' flag */
5391 cpo_save = p_cpo;
5392 p_cpo = (char_u *)"";
5393 ci->sp_prog = vim_regcomp(ci->sp_pattern, RE_MAGIC);
5394 p_cpo = cpo_save;
5395
5396 if (ci->sp_prog == NULL)
5397 return NULL;
5398 ci->sp_ic = curbuf->b_syn_ic;
5399
5400 /*
5401 * Check for a match, highlight or region offset.
5402 */
5403 ++end;
5404 do
5405 {
5406 for (idx = SPO_COUNT; --idx >= 0; )
5407 if (STRNCMP(end, spo_name_tab[idx], 3) == 0)
5408 break;
5409 if (idx >= 0)
5410 {
5411 p = &(ci->sp_offsets[idx]);
5412 if (idx != SPO_LC_OFF)
5413 switch (end[3])
5414 {
5415 case 's': break;
5416 case 'b': break;
5417 case 'e': idx += SPO_COUNT; break;
5418 default: idx = -1; break;
5419 }
5420 if (idx >= 0)
5421 {
5422 ci->sp_off_flags |= (1 << idx);
5423 if (idx == SPO_LC_OFF) /* lc=99 */
5424 {
5425 end += 3;
5426 *p = getdigits(&end);
5427
5428 /* "lc=" offset automatically sets "ms=" offset */
5429 if (!(ci->sp_off_flags & (1 << SPO_MS_OFF)))
5430 {
5431 ci->sp_off_flags |= (1 << SPO_MS_OFF);
5432 ci->sp_offsets[SPO_MS_OFF] = *p;
5433 }
5434 }
5435 else /* yy=x+99 */
5436 {
5437 end += 4;
5438 if (*end == '+')
5439 {
5440 ++end;
5441 *p = getdigits(&end); /* positive offset */
5442 }
5443 else if (*end == '-')
5444 {
5445 ++end;
5446 *p = -getdigits(&end); /* negative offset */
5447 }
5448 }
5449 if (*end != ',')
5450 break;
5451 ++end;
5452 }
5453 }
5454 } while (idx >= 0);
5455
5456 if (!ends_excmd(*end) && !vim_iswhite(*end))
5457 {
5458 EMSG2(_("E402: Garbage after pattern: %s"), arg);
5459 return NULL;
5460 }
5461 return skipwhite(end);
5462}
5463
5464/*
5465 * Handle ":syntax sync .." command.
5466 */
5467/* ARGSUSED */
5468 static void
5469syn_cmd_sync(eap, syncing)
5470 exarg_T *eap;
5471 int syncing; /* not used */
5472{
5473 char_u *arg_start = eap->arg;
5474 char_u *arg_end;
5475 char_u *key = NULL;
5476 char_u *next_arg;
5477 int illegal = FALSE;
5478 int finished = FALSE;
5479 long n;
5480 char_u *cpo_save;
5481
5482 if (ends_excmd(*arg_start))
5483 {
5484 syn_cmd_list(eap, TRUE);
5485 return;
5486 }
5487
5488 while (!ends_excmd(*arg_start))
5489 {
5490 arg_end = skiptowhite(arg_start);
5491 next_arg = skipwhite(arg_end);
5492 vim_free(key);
5493 key = vim_strnsave_up(arg_start, (int)(arg_end - arg_start));
5494 if (STRCMP(key, "CCOMMENT") == 0)
5495 {
5496 if (!eap->skip)
5497 curbuf->b_syn_sync_flags |= SF_CCOMMENT;
5498 if (!ends_excmd(*next_arg))
5499 {
5500 arg_end = skiptowhite(next_arg);
5501 if (!eap->skip)
5502 curbuf->b_syn_sync_id = syn_check_group(next_arg,
5503 (int)(arg_end - next_arg));
5504 next_arg = skipwhite(arg_end);
5505 }
5506 else if (!eap->skip)
5507 curbuf->b_syn_sync_id = syn_name2id((char_u *)"Comment");
5508 }
5509 else if ( STRNCMP(key, "LINES", 5) == 0
5510 || STRNCMP(key, "MINLINES", 8) == 0
5511 || STRNCMP(key, "MAXLINES", 8) == 0
5512 || STRNCMP(key, "LINEBREAKS", 10) == 0)
5513 {
5514 if (key[4] == 'S')
5515 arg_end = key + 6;
5516 else if (key[0] == 'L')
5517 arg_end = key + 11;
5518 else
5519 arg_end = key + 9;
5520 if (arg_end[-1] != '=' || !VIM_ISDIGIT(*arg_end))
5521 {
5522 illegal = TRUE;
5523 break;
5524 }
5525 n = getdigits(&arg_end);
5526 if (!eap->skip)
5527 {
5528 if (key[4] == 'B')
5529 curbuf->b_syn_sync_linebreaks = n;
5530 else if (key[1] == 'A')
5531 curbuf->b_syn_sync_maxlines = n;
5532 else
5533 curbuf->b_syn_sync_minlines = n;
5534 }
5535 }
5536 else if (STRCMP(key, "FROMSTART") == 0)
5537 {
5538 if (!eap->skip)
5539 {
5540 curbuf->b_syn_sync_minlines = MAXLNUM;
5541 curbuf->b_syn_sync_maxlines = 0;
5542 }
5543 }
5544 else if (STRCMP(key, "LINECONT") == 0)
5545 {
5546 if (curbuf->b_syn_linecont_pat != NULL)
5547 {
5548 EMSG(_("E403: syntax sync: line continuations pattern specified twice"));
5549 finished = TRUE;
5550 break;
5551 }
5552 arg_end = skip_regexp(next_arg + 1, *next_arg, TRUE, NULL);
5553 if (*arg_end != *next_arg) /* end delimiter not found */
5554 {
5555 illegal = TRUE;
5556 break;
5557 }
5558
5559 if (!eap->skip)
5560 {
5561 /* store the pattern and compiled regexp program */
5562 if ((curbuf->b_syn_linecont_pat = vim_strnsave(next_arg + 1,
5563 (int)(arg_end - next_arg - 1))) == NULL)
5564 {
5565 finished = TRUE;
5566 break;
5567 }
5568 curbuf->b_syn_linecont_ic = curbuf->b_syn_ic;
5569
5570 /* Make 'cpoptions' empty, to avoid the 'l' flag */
5571 cpo_save = p_cpo;
5572 p_cpo = (char_u *)"";
5573 curbuf->b_syn_linecont_prog =
5574 vim_regcomp(curbuf->b_syn_linecont_pat, RE_MAGIC);
5575 p_cpo = cpo_save;
5576
5577 if (curbuf->b_syn_linecont_prog == NULL)
5578 {
5579 vim_free(curbuf->b_syn_linecont_pat);
5580 curbuf->b_syn_linecont_pat = NULL;
5581 finished = TRUE;
5582 break;
5583 }
5584 }
5585 next_arg = skipwhite(arg_end + 1);
5586 }
5587 else
5588 {
5589 eap->arg = next_arg;
5590 if (STRCMP(key, "MATCH") == 0)
5591 syn_cmd_match(eap, TRUE);
5592 else if (STRCMP(key, "REGION") == 0)
5593 syn_cmd_region(eap, TRUE);
5594 else if (STRCMP(key, "CLEAR") == 0)
5595 syn_cmd_clear(eap, TRUE);
5596 else
5597 illegal = TRUE;
5598 finished = TRUE;
5599 break;
5600 }
5601 arg_start = next_arg;
5602 }
5603 vim_free(key);
5604 if (illegal)
5605 EMSG2(_("E404: Illegal arguments: %s"), arg_start);
5606 else if (!finished)
5607 {
5608 eap->nextcmd = check_nextcmd(arg_start);
Bram Moolenaar1c8f93f2006-03-12 22:10:07 +00005609 redraw_curbuf_later(SOME_VALID);
Bram Moolenaar071d4272004-06-13 20:20:40 +00005610 syn_stack_free_all(curbuf); /* Need to recompute all syntax. */
5611 }
5612}
5613
5614/*
5615 * Convert a line of highlight group names into a list of group ID numbers.
5616 * "arg" should point to the "contains" or "nextgroup" keyword.
5617 * "arg" is advanced to after the last group name.
5618 * Careful: the argument is modified (NULs added).
5619 * returns FAIL for some error, OK for success.
5620 */
5621 static int
5622get_id_list(arg, keylen, list)
5623 char_u **arg;
5624 int keylen; /* length of keyword */
5625 short **list; /* where to store the resulting list, if not
5626 NULL, the list is silently skipped! */
5627{
5628 char_u *p = NULL;
5629 char_u *end;
5630 int round;
5631 int count;
5632 int total_count = 0;
5633 short *retval = NULL;
5634 char_u *name;
5635 regmatch_T regmatch;
5636 int id;
5637 int i;
5638 int failed = FALSE;
5639
5640 /*
5641 * We parse the list twice:
5642 * round == 1: count the number of items, allocate the array.
5643 * round == 2: fill the array with the items.
5644 * In round 1 new groups may be added, causing the number of items to
5645 * grow when a regexp is used. In that case round 1 is done once again.
5646 */
5647 for (round = 1; round <= 2; ++round)
5648 {
5649 /*
5650 * skip "contains"
5651 */
5652 p = skipwhite(*arg + keylen);
5653 if (*p != '=')
5654 {
5655 EMSG2(_("E405: Missing equal sign: %s"), *arg);
5656 break;
5657 }
5658 p = skipwhite(p + 1);
5659 if (ends_excmd(*p))
5660 {
5661 EMSG2(_("E406: Empty argument: %s"), *arg);
5662 break;
5663 }
5664
5665 /*
5666 * parse the arguments after "contains"
5667 */
5668 count = 0;
5669 while (!ends_excmd(*p))
5670 {
5671 for (end = p; *end && !vim_iswhite(*end) && *end != ','; ++end)
5672 ;
5673 name = alloc((int)(end - p + 3)); /* leave room for "^$" */
5674 if (name == NULL)
5675 {
5676 failed = TRUE;
5677 break;
5678 }
Bram Moolenaarce0842a2005-07-18 21:58:11 +00005679 vim_strncpy(name + 1, p, end - p);
Bram Moolenaar071d4272004-06-13 20:20:40 +00005680 if ( STRCMP(name + 1, "ALLBUT") == 0
5681 || STRCMP(name + 1, "ALL") == 0
5682 || STRCMP(name + 1, "TOP") == 0
5683 || STRCMP(name + 1, "CONTAINED") == 0)
5684 {
5685 if (TOUPPER_ASC(**arg) != 'C')
5686 {
5687 EMSG2(_("E407: %s not allowed here"), name + 1);
5688 failed = TRUE;
5689 vim_free(name);
5690 break;
5691 }
5692 if (count != 0)
5693 {
5694 EMSG2(_("E408: %s must be first in contains list"), name + 1);
5695 failed = TRUE;
5696 vim_free(name);
5697 break;
5698 }
5699 if (name[1] == 'A')
5700 id = SYNID_ALLBUT;
5701 else if (name[1] == 'T')
5702 id = SYNID_TOP;
5703 else
5704 id = SYNID_CONTAINED;
5705 id += current_syn_inc_tag;
5706 }
5707 else if (name[1] == '@')
5708 {
5709 id = syn_check_cluster(name + 2, (int)(end - p - 1));
5710 }
5711 else
5712 {
5713 /*
5714 * Handle full group name.
5715 */
5716 if (vim_strpbrk(name + 1, (char_u *)"\\.*^$~[") == NULL)
5717 id = syn_check_group(name + 1, (int)(end - p));
5718 else
5719 {
5720 /*
5721 * Handle match of regexp with group names.
5722 */
5723 *name = '^';
5724 STRCAT(name, "$");
5725 regmatch.regprog = vim_regcomp(name, RE_MAGIC);
5726 if (regmatch.regprog == NULL)
5727 {
5728 failed = TRUE;
5729 vim_free(name);
5730 break;
5731 }
5732
5733 regmatch.rm_ic = TRUE;
5734 id = 0;
5735 for (i = highlight_ga.ga_len; --i >= 0; )
5736 {
5737 if (vim_regexec(&regmatch, HL_TABLE()[i].sg_name,
5738 (colnr_T)0))
5739 {
5740 if (round == 2)
5741 {
5742 /* Got more items than expected; can happen
5743 * when adding items that match:
5744 * "contains=a.*b,axb".
5745 * Go back to first round */
5746 if (count >= total_count)
5747 {
5748 vim_free(retval);
5749 round = 1;
5750 }
5751 else
5752 retval[count] = i + 1;
5753 }
5754 ++count;
5755 id = -1; /* remember that we found one */
5756 }
5757 }
5758 vim_free(regmatch.regprog);
5759 }
5760 }
5761 vim_free(name);
5762 if (id == 0)
5763 {
5764 EMSG2(_("E409: Unknown group name: %s"), p);
5765 failed = TRUE;
5766 break;
5767 }
5768 if (id > 0)
5769 {
5770 if (round == 2)
5771 {
5772 /* Got more items than expected, go back to first round */
5773 if (count >= total_count)
5774 {
5775 vim_free(retval);
5776 round = 1;
5777 }
5778 else
5779 retval[count] = id;
5780 }
5781 ++count;
5782 }
5783 p = skipwhite(end);
5784 if (*p != ',')
5785 break;
5786 p = skipwhite(p + 1); /* skip comma in between arguments */
5787 }
5788 if (failed)
5789 break;
5790 if (round == 1)
5791 {
5792 retval = (short *)alloc((unsigned)((count + 1) * sizeof(short)));
5793 if (retval == NULL)
5794 break;
5795 retval[count] = 0; /* zero means end of the list */
5796 total_count = count;
5797 }
5798 }
5799
5800 *arg = p;
5801 if (failed || retval == NULL)
5802 {
5803 vim_free(retval);
5804 return FAIL;
5805 }
5806
5807 if (*list == NULL)
5808 *list = retval;
5809 else
5810 vim_free(retval); /* list already found, don't overwrite it */
5811
5812 return OK;
5813}
5814
5815/*
5816 * Make a copy of an ID list.
5817 */
5818 static short *
5819copy_id_list(list)
5820 short *list;
5821{
5822 int len;
5823 int count;
5824 short *retval;
5825
5826 if (list == NULL)
5827 return NULL;
5828
5829 for (count = 0; list[count]; ++count)
5830 ;
5831 len = (count + 1) * sizeof(short);
5832 retval = (short *)alloc((unsigned)len);
5833 if (retval != NULL)
5834 mch_memmove(retval, list, (size_t)len);
5835
5836 return retval;
5837}
5838
5839/*
5840 * Check if syntax group "ssp" is in the ID list "list" of "cur_si".
5841 * "cur_si" can be NULL if not checking the "containedin" list.
5842 * Used to check if a syntax item is in the "contains" or "nextgroup" list of
5843 * the current item.
5844 * This function is called very often, keep it fast!!
5845 */
5846 static int
5847in_id_list(cur_si, list, ssp, contained)
5848 stateitem_T *cur_si; /* current item or NULL */
5849 short *list; /* id list */
5850 struct sp_syn *ssp; /* group id and ":syn include" tag of group */
5851 int contained; /* group id is contained */
5852{
5853 int retval;
5854 short *scl_list;
5855 short item;
5856 short id = ssp->id;
5857 static int depth = 0;
5858 int r;
5859
5860 /* If spp has a "containedin" list and "cur_si" is in it, return TRUE. */
Bram Moolenaar293ee4d2004-12-09 21:34:53 +00005861 if (cur_si != NULL && ssp->cont_in_list != NULL
5862 && !(cur_si->si_flags & HL_MATCH))
Bram Moolenaar071d4272004-06-13 20:20:40 +00005863 {
5864 /* Ignore transparent items without a contains argument. Double check
5865 * that we don't go back past the first one. */
5866 while ((cur_si->si_flags & HL_TRANS_CONT)
5867 && cur_si > (stateitem_T *)(current_state.ga_data))
5868 --cur_si;
5869 /* cur_si->si_idx is -1 for keywords, these never contain anything. */
5870 if (cur_si->si_idx >= 0 && in_id_list(NULL, ssp->cont_in_list,
5871 &(SYN_ITEMS(syn_buf)[cur_si->si_idx].sp_syn),
5872 SYN_ITEMS(syn_buf)[cur_si->si_idx].sp_flags & HL_CONTAINED))
5873 return TRUE;
5874 }
5875
5876 if (list == NULL)
5877 return FALSE;
5878
5879 /*
5880 * If list is ID_LIST_ALL, we are in a transparent item that isn't
5881 * inside anything. Only allow not-contained groups.
5882 */
5883 if (list == ID_LIST_ALL)
5884 return !contained;
5885
5886 /*
5887 * If the first item is "ALLBUT", return TRUE if "id" is NOT in the
5888 * contains list. We also require that "id" is at the same ":syn include"
5889 * level as the list.
5890 */
5891 item = *list;
5892 if (item >= SYNID_ALLBUT && item < SYNID_CLUSTER)
5893 {
5894 if (item < SYNID_TOP)
5895 {
5896 /* ALL or ALLBUT: accept all groups in the same file */
5897 if (item - SYNID_ALLBUT != ssp->inc_tag)
5898 return FALSE;
5899 }
5900 else if (item < SYNID_CONTAINED)
5901 {
5902 /* TOP: accept all not-contained groups in the same file */
5903 if (item - SYNID_TOP != ssp->inc_tag || contained)
5904 return FALSE;
5905 }
5906 else
5907 {
5908 /* CONTAINED: accept all contained groups in the same file */
5909 if (item - SYNID_CONTAINED != ssp->inc_tag || !contained)
5910 return FALSE;
5911 }
5912 item = *++list;
5913 retval = FALSE;
5914 }
5915 else
5916 retval = TRUE;
5917
5918 /*
5919 * Return "retval" if id is in the contains list.
5920 */
5921 while (item != 0)
5922 {
5923 if (item == id)
5924 return retval;
5925 if (item >= SYNID_CLUSTER)
5926 {
5927 scl_list = SYN_CLSTR(syn_buf)[item - SYNID_CLUSTER].scl_list;
5928 /* restrict recursiveness to 30 to avoid an endless loop for a
5929 * cluster that includes itself (indirectly) */
5930 if (scl_list != NULL && depth < 30)
5931 {
5932 ++depth;
5933 r = in_id_list(NULL, scl_list, ssp, contained);
5934 --depth;
5935 if (r)
5936 return retval;
5937 }
5938 }
5939 item = *++list;
5940 }
5941 return !retval;
5942}
5943
5944struct subcommand
5945{
5946 char *name; /* subcommand name */
5947 void (*func)__ARGS((exarg_T *, int)); /* function to call */
5948};
5949
5950static struct subcommand subcommands[] =
5951{
5952 {"case", syn_cmd_case},
5953 {"clear", syn_cmd_clear},
5954 {"cluster", syn_cmd_cluster},
5955 {"enable", syn_cmd_enable},
5956 {"include", syn_cmd_include},
5957 {"keyword", syn_cmd_keyword},
5958 {"list", syn_cmd_list},
5959 {"manual", syn_cmd_manual},
5960 {"match", syn_cmd_match},
5961 {"on", syn_cmd_on},
5962 {"off", syn_cmd_off},
5963 {"region", syn_cmd_region},
5964 {"reset", syn_cmd_reset},
Bram Moolenaarce0842a2005-07-18 21:58:11 +00005965 {"spell", syn_cmd_spell},
Bram Moolenaar071d4272004-06-13 20:20:40 +00005966 {"sync", syn_cmd_sync},
5967 {"", syn_cmd_list},
5968 {NULL, NULL}
5969};
5970
5971/*
5972 * ":syntax".
5973 * This searches the subcommands[] table for the subcommand name, and calls a
5974 * syntax_subcommand() function to do the rest.
5975 */
5976 void
5977ex_syntax(eap)
5978 exarg_T *eap;
5979{
5980 char_u *arg = eap->arg;
5981 char_u *subcmd_end;
5982 char_u *subcmd_name;
5983 int i;
5984
5985 syn_cmdlinep = eap->cmdlinep;
5986
5987 /* isolate subcommand name */
5988 for (subcmd_end = arg; ASCII_ISALPHA(*subcmd_end); ++subcmd_end)
5989 ;
5990 subcmd_name = vim_strnsave(arg, (int)(subcmd_end - arg));
5991 if (subcmd_name != NULL)
5992 {
5993 if (eap->skip) /* skip error messages for all subcommands */
5994 ++emsg_skip;
5995 for (i = 0; ; ++i)
5996 {
5997 if (subcommands[i].name == NULL)
5998 {
5999 EMSG2(_("E410: Invalid :syntax subcommand: %s"), subcmd_name);
6000 break;
6001 }
6002 if (STRCMP(subcmd_name, (char_u *)subcommands[i].name) == 0)
6003 {
6004 eap->arg = skipwhite(subcmd_end);
6005 (subcommands[i].func)(eap, FALSE);
6006 break;
6007 }
6008 }
6009 vim_free(subcmd_name);
6010 if (eap->skip)
6011 --emsg_skip;
6012 }
6013}
6014
6015 int
6016syntax_present(buf)
6017 buf_T *buf;
6018{
6019 return (buf->b_syn_patterns.ga_len != 0
6020 || buf->b_syn_clusters.ga_len != 0
Bram Moolenaar102e3a62007-08-30 17:37:40 +00006021 || buf->b_keywtab.ht_used > 0
6022 || buf->b_keywtab_ic.ht_used > 0);
Bram Moolenaar071d4272004-06-13 20:20:40 +00006023}
6024
6025#if defined(FEAT_CMDL_COMPL) || defined(PROTO)
6026
6027static enum
6028{
6029 EXP_SUBCMD, /* expand ":syn" sub-commands */
6030 EXP_CASE /* expand ":syn case" arguments */
6031} expand_what;
6032
Bram Moolenaar4f688582007-07-24 12:34:30 +00006033/*
6034 * Reset include_link, include_default, include_none to 0.
6035 * Called when we are done expanding.
6036 */
6037 void
6038reset_expand_highlight()
6039{
6040 include_link = include_default = include_none = 0;
6041}
6042
6043/*
6044 * Handle command line completion for :match and :echohl command: Add "None"
6045 * as highlight group.
6046 */
6047 void
6048set_context_in_echohl_cmd(xp, arg)
6049 expand_T *xp;
6050 char_u *arg;
6051{
6052 xp->xp_context = EXPAND_HIGHLIGHT;
6053 xp->xp_pattern = arg;
6054 include_none = 1;
6055}
Bram Moolenaar071d4272004-06-13 20:20:40 +00006056
6057/*
6058 * Handle command line completion for :syntax command.
6059 */
6060 void
6061set_context_in_syntax_cmd(xp, arg)
6062 expand_T *xp;
6063 char_u *arg;
6064{
6065 char_u *p;
6066
6067 /* Default: expand subcommands */
6068 xp->xp_context = EXPAND_SYNTAX;
6069 expand_what = EXP_SUBCMD;
6070 xp->xp_pattern = arg;
Bram Moolenaar4f688582007-07-24 12:34:30 +00006071 include_link = 0;
6072 include_default = 0;
Bram Moolenaar071d4272004-06-13 20:20:40 +00006073
6074 /* (part of) subcommand already typed */
6075 if (*arg != NUL)
6076 {
6077 p = skiptowhite(arg);
6078 if (*p != NUL) /* past first word */
6079 {
6080 xp->xp_pattern = skipwhite(p);
6081 if (*skiptowhite(xp->xp_pattern) != NUL)
6082 xp->xp_context = EXPAND_NOTHING;
6083 else if (STRNICMP(arg, "case", p - arg) == 0)
6084 expand_what = EXP_CASE;
6085 else if ( STRNICMP(arg, "keyword", p - arg) == 0
6086 || STRNICMP(arg, "region", p - arg) == 0
6087 || STRNICMP(arg, "match", p - arg) == 0
6088 || STRNICMP(arg, "list", p - arg) == 0)
6089 xp->xp_context = EXPAND_HIGHLIGHT;
6090 else
6091 xp->xp_context = EXPAND_NOTHING;
6092 }
6093 }
6094}
6095
6096static char *(case_args[]) = {"match", "ignore", NULL};
6097
6098/*
6099 * Function given to ExpandGeneric() to obtain the list syntax names for
6100 * expansion.
6101 */
6102/*ARGSUSED*/
6103 char_u *
6104get_syntax_name(xp, idx)
6105 expand_T *xp;
6106 int idx;
6107{
6108 if (expand_what == EXP_SUBCMD)
6109 return (char_u *)subcommands[idx].name;
6110 return (char_u *)case_args[idx];
6111}
6112
6113#endif /* FEAT_CMDL_COMPL */
6114
Bram Moolenaar071d4272004-06-13 20:20:40 +00006115/*
6116 * Function called for expression evaluation: get syntax ID at file position.
6117 */
6118 int
Bram Moolenaar56cefaf2008-01-12 15:47:10 +00006119syn_get_id(wp, lnum, col, trans, spellp, keep_state)
Bram Moolenaar81f1ecb2005-08-25 21:27:31 +00006120 win_T *wp;
Bram Moolenaar071d4272004-06-13 20:20:40 +00006121 long lnum;
Bram Moolenaar9d0ec2e2005-04-20 19:45:58 +00006122 colnr_T col;
Bram Moolenaar56cefaf2008-01-12 15:47:10 +00006123 int trans; /* remove transparancy */
6124 int *spellp; /* return: can do spell checking */
6125 int keep_state; /* keep state of char at "col" */
Bram Moolenaar071d4272004-06-13 20:20:40 +00006126{
6127 /* When the position is not after the current position and in the same
6128 * line of the same buffer, need to restart parsing. */
Bram Moolenaar81f1ecb2005-08-25 21:27:31 +00006129 if (wp->w_buffer != syn_buf
Bram Moolenaar071d4272004-06-13 20:20:40 +00006130 || lnum != current_lnum
Bram Moolenaar9d0ec2e2005-04-20 19:45:58 +00006131 || col < current_col)
Bram Moolenaar81f1ecb2005-08-25 21:27:31 +00006132 syntax_start(wp, lnum);
Bram Moolenaar071d4272004-06-13 20:20:40 +00006133
Bram Moolenaar56cefaf2008-01-12 15:47:10 +00006134 (void)get_syntax_attr(col, spellp, keep_state);
Bram Moolenaar071d4272004-06-13 20:20:40 +00006135
6136 return (trans ? current_trans_id : current_id);
6137}
Bram Moolenaar071d4272004-06-13 20:20:40 +00006138
Bram Moolenaar9d188ab2008-01-10 21:24:39 +00006139#if defined(FEAT_EVAL) || defined(PROTO)
6140/*
6141 * Return the syntax ID at position "i" in the current stack.
6142 * The caller must have called syn_get_id() before to fill the stack.
6143 * Returns -1 when "i" is out of range.
6144 */
6145 int
6146syn_get_stack_item(i)
6147 int i;
6148{
Bram Moolenaar56cefaf2008-01-12 15:47:10 +00006149 if (i >= current_state.ga_len)
6150 {
6151 /* Need to invalidate the state, because we didn't properly finish it
6152 * for the last character, "keep_state" was TRUE. */
6153 invalidate_current_state();
6154 current_col = MAXCOL;
Bram Moolenaar9d188ab2008-01-10 21:24:39 +00006155 return -1;
Bram Moolenaar56cefaf2008-01-12 15:47:10 +00006156 }
Bram Moolenaar9d188ab2008-01-10 21:24:39 +00006157 return CUR_STATE(i).si_id;
6158}
6159#endif
6160
Bram Moolenaar071d4272004-06-13 20:20:40 +00006161#if defined(FEAT_FOLDING) || defined(PROTO)
6162/*
6163 * Function called to get folding level for line "lnum" in window "wp".
6164 */
6165 int
6166syn_get_foldlevel(wp, lnum)
6167 win_T *wp;
6168 long lnum;
6169{
6170 int level = 0;
6171 int i;
6172
6173 /* Return quickly when there are no fold items at all. */
6174 if (wp->w_buffer->b_syn_folditems != 0)
6175 {
6176 syntax_start(wp, lnum);
6177
6178 for (i = 0; i < current_state.ga_len; ++i)
6179 if (CUR_STATE(i).si_flags & HL_FOLD)
6180 ++level;
6181 }
6182 if (level > wp->w_p_fdn)
Bram Moolenaar74c596b2006-11-01 11:44:31 +00006183 {
Bram Moolenaar071d4272004-06-13 20:20:40 +00006184 level = wp->w_p_fdn;
Bram Moolenaar74c596b2006-11-01 11:44:31 +00006185 if (level < 0)
6186 level = 0;
6187 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00006188 return level;
6189}
6190#endif
6191
6192#endif /* FEAT_SYN_HL */
6193
6194
6195/**************************************
6196 * Highlighting stuff *
6197 **************************************/
6198
6199/*
6200 * The default highlight groups. These are compiled-in for fast startup and
6201 * they still work when the runtime files can't be found.
6202 * When making changes here, also change runtime/colors/default.vim!
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006203 * The #ifdefs are needed to reduce the amount of static data. Helps to make
6204 * the 16 bit DOS (museum) version compile.
Bram Moolenaar071d4272004-06-13 20:20:40 +00006205 */
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006206#ifdef FEAT_GUI
6207# define CENT(a, b) b
6208#else
6209# define CENT(a, b) a
6210#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00006211static char *(highlight_init_both[]) =
6212 {
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006213 CENT("ErrorMsg term=standout ctermbg=DarkRed ctermfg=White",
6214 "ErrorMsg term=standout ctermbg=DarkRed ctermfg=White guibg=Red guifg=White"),
Bram Moolenaarda65f152007-05-06 13:54:35 +00006215#ifdef FEAT_SEARCH_EXTRA
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006216 CENT("IncSearch term=reverse cterm=reverse",
6217 "IncSearch term=reverse cterm=reverse gui=reverse"),
Bram Moolenaarda65f152007-05-06 13:54:35 +00006218#endif
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006219 CENT("ModeMsg term=bold cterm=bold",
6220 "ModeMsg term=bold cterm=bold gui=bold"),
6221 CENT("NonText term=bold ctermfg=Blue",
6222 "NonText term=bold ctermfg=Blue gui=bold guifg=Blue"),
6223 CENT("StatusLine term=reverse,bold cterm=reverse,bold",
6224 "StatusLine term=reverse,bold cterm=reverse,bold gui=reverse,bold"),
6225 CENT("StatusLineNC term=reverse cterm=reverse",
6226 "StatusLineNC term=reverse cterm=reverse gui=reverse"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006227#ifdef FEAT_VERTSPLIT
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006228 CENT("VertSplit term=reverse cterm=reverse",
6229 "VertSplit term=reverse cterm=reverse gui=reverse"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006230#endif
6231#ifdef FEAT_CLIPBOARD
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006232 CENT("VisualNOS term=underline,bold cterm=underline,bold",
6233 "VisualNOS term=underline,bold cterm=underline,bold gui=underline,bold"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006234#endif
6235#ifdef FEAT_DIFF
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006236 CENT("DiffText term=reverse cterm=bold ctermbg=Red",
6237 "DiffText term=reverse cterm=bold ctermbg=Red gui=bold guibg=Red"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006238#endif
6239#ifdef FEAT_INS_EXPAND
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006240 CENT("PmenuThumb cterm=reverse",
6241 "PmenuThumb cterm=reverse gui=reverse"),
6242 CENT("PmenuSbar ctermbg=Grey",
6243 "PmenuSbar ctermbg=Grey guibg=Grey"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006244#endif
6245#ifdef FEAT_WINDOWS
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006246 CENT("TabLineSel term=bold cterm=bold",
6247 "TabLineSel term=bold cterm=bold gui=bold"),
6248 CENT("TabLineFill term=reverse cterm=reverse",
6249 "TabLineFill term=reverse cterm=reverse gui=reverse"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006250#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00006251#ifdef FEAT_GUI
6252 "Cursor guibg=fg guifg=bg",
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006253 "lCursor guibg=fg guifg=bg", /* should be different, but what? */
Bram Moolenaar071d4272004-06-13 20:20:40 +00006254#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00006255 NULL
6256 };
6257
6258static char *(highlight_init_light[]) =
6259 {
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006260 CENT("Directory term=bold ctermfg=DarkBlue",
6261 "Directory term=bold ctermfg=DarkBlue guifg=Blue"),
6262 CENT("LineNr term=underline ctermfg=Brown",
6263 "LineNr term=underline ctermfg=Brown guifg=Brown"),
6264 CENT("MoreMsg term=bold ctermfg=DarkGreen",
6265 "MoreMsg term=bold ctermfg=DarkGreen gui=bold guifg=SeaGreen"),
6266 CENT("Question term=standout ctermfg=DarkGreen",
6267 "Question term=standout ctermfg=DarkGreen gui=bold guifg=SeaGreen"),
6268 CENT("Search term=reverse ctermbg=Yellow ctermfg=NONE",
6269 "Search term=reverse ctermbg=Yellow ctermfg=NONE guibg=Yellow guifg=NONE"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006270#ifdef FEAT_SPELL
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006271 CENT("SpellBad term=reverse ctermbg=LightRed",
6272 "SpellBad term=reverse ctermbg=LightRed guisp=Red gui=undercurl"),
6273 CENT("SpellCap term=reverse ctermbg=LightBlue",
6274 "SpellCap term=reverse ctermbg=LightBlue guisp=Blue gui=undercurl"),
6275 CENT("SpellRare term=reverse ctermbg=LightMagenta",
6276 "SpellRare term=reverse ctermbg=LightMagenta guisp=Magenta gui=undercurl"),
6277 CENT("SpellLocal term=underline ctermbg=Cyan",
6278 "SpellLocal term=underline ctermbg=Cyan guisp=DarkCyan gui=undercurl"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006279#endif
6280#ifdef FEAT_INS_EXPAND
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006281 CENT("Pmenu ctermbg=LightMagenta",
6282 "Pmenu ctermbg=LightMagenta guibg=LightMagenta"),
6283 CENT("PmenuSel ctermbg=LightGrey",
6284 "PmenuSel ctermbg=LightGrey guibg=Grey"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006285#endif
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006286 CENT("SpecialKey term=bold ctermfg=DarkBlue",
6287 "SpecialKey term=bold ctermfg=DarkBlue guifg=Blue"),
6288 CENT("Title term=bold ctermfg=DarkMagenta",
6289 "Title term=bold ctermfg=DarkMagenta gui=bold guifg=Magenta"),
6290 CENT("WarningMsg term=standout ctermfg=DarkRed",
6291 "WarningMsg term=standout ctermfg=DarkRed guifg=Red"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006292#ifdef FEAT_WILDMENU
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006293 CENT("WildMenu term=standout ctermbg=Yellow ctermfg=Black",
6294 "WildMenu term=standout ctermbg=Yellow ctermfg=Black guibg=Yellow guifg=Black"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006295#endif
6296#ifdef FEAT_FOLDING
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006297 CENT("Folded term=standout ctermbg=Grey ctermfg=DarkBlue",
6298 "Folded term=standout ctermbg=Grey ctermfg=DarkBlue guibg=LightGrey guifg=DarkBlue"),
6299 CENT("FoldColumn term=standout ctermbg=Grey ctermfg=DarkBlue",
6300 "FoldColumn term=standout ctermbg=Grey ctermfg=DarkBlue guibg=Grey guifg=DarkBlue"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006301#endif
6302#ifdef FEAT_SIGNS
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006303 CENT("SignColumn term=standout ctermbg=Grey ctermfg=DarkBlue",
6304 "SignColumn term=standout ctermbg=Grey ctermfg=DarkBlue guibg=Grey guifg=DarkBlue"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006305#endif
6306#ifdef FEAT_VISUAL
Bram Moolenaarc1e37902006-04-18 21:55:01 +00006307 CENT("Visual term=reverse",
6308 "Visual term=reverse guibg=LightGrey"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006309#endif
6310#ifdef FEAT_DIFF
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006311 CENT("DiffAdd term=bold ctermbg=LightBlue",
6312 "DiffAdd term=bold ctermbg=LightBlue guibg=LightBlue"),
6313 CENT("DiffChange term=bold ctermbg=LightMagenta",
6314 "DiffChange term=bold ctermbg=LightMagenta guibg=LightMagenta"),
6315 CENT("DiffDelete term=bold ctermfg=Blue ctermbg=LightCyan",
6316 "DiffDelete term=bold ctermfg=Blue ctermbg=LightCyan gui=bold guifg=Blue guibg=LightCyan"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006317#endif
6318#ifdef FEAT_WINDOWS
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006319 CENT("TabLine term=underline cterm=underline ctermfg=black ctermbg=LightGrey",
6320 "TabLine term=underline cterm=underline ctermfg=black ctermbg=LightGrey gui=underline guibg=LightGrey"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006321#endif
6322#ifdef FEAT_SYN_HL
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006323 CENT("CursorColumn term=reverse ctermbg=LightGrey",
Bram Moolenaare2f98b92006-03-29 21:18:24 +00006324 "CursorColumn term=reverse ctermbg=LightGrey guibg=Grey90"),
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006325 CENT("CursorLine term=underline cterm=underline",
Bram Moolenaare2f98b92006-03-29 21:18:24 +00006326 "CursorLine term=underline cterm=underline guibg=Grey90"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006327#endif
Bram Moolenaarb21e5842006-04-16 18:30:08 +00006328#ifdef FEAT_AUTOCMD
6329 CENT("MatchParen term=reverse ctermbg=Cyan",
6330 "MatchParen term=reverse ctermbg=Cyan guibg=Cyan"),
6331#endif
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006332#ifdef FEAT_GUI
Bram Moolenaar071d4272004-06-13 20:20:40 +00006333 "Normal gui=NONE",
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006334#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00006335 NULL
6336 };
6337
6338static char *(highlight_init_dark[]) =
6339 {
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006340 CENT("Directory term=bold ctermfg=LightCyan",
6341 "Directory term=bold ctermfg=LightCyan guifg=Cyan"),
6342 CENT("LineNr term=underline ctermfg=Yellow",
6343 "LineNr term=underline ctermfg=Yellow guifg=Yellow"),
6344 CENT("MoreMsg term=bold ctermfg=LightGreen",
6345 "MoreMsg term=bold ctermfg=LightGreen gui=bold guifg=SeaGreen"),
6346 CENT("Question term=standout ctermfg=LightGreen",
6347 "Question term=standout ctermfg=LightGreen gui=bold guifg=Green"),
6348 CENT("Search term=reverse ctermbg=Yellow ctermfg=Black",
6349 "Search term=reverse ctermbg=Yellow ctermfg=Black guibg=Yellow guifg=Black"),
6350 CENT("SpecialKey term=bold ctermfg=LightBlue",
6351 "SpecialKey term=bold ctermfg=LightBlue guifg=Cyan"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006352#ifdef FEAT_SPELL
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006353 CENT("SpellBad term=reverse ctermbg=Red",
6354 "SpellBad term=reverse ctermbg=Red guisp=Red gui=undercurl"),
6355 CENT("SpellCap term=reverse ctermbg=Blue",
6356 "SpellCap term=reverse ctermbg=Blue guisp=Blue gui=undercurl"),
6357 CENT("SpellRare term=reverse ctermbg=Magenta",
6358 "SpellRare term=reverse ctermbg=Magenta guisp=Magenta gui=undercurl"),
6359 CENT("SpellLocal term=underline ctermbg=Cyan",
6360 "SpellLocal term=underline ctermbg=Cyan guisp=Cyan gui=undercurl"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006361#endif
6362#ifdef FEAT_INS_EXPAND
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006363 CENT("Pmenu ctermbg=Magenta",
6364 "Pmenu ctermbg=Magenta guibg=Magenta"),
6365 CENT("PmenuSel ctermbg=DarkGrey",
6366 "PmenuSel ctermbg=DarkGrey guibg=DarkGrey"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006367#endif
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006368 CENT("Title term=bold ctermfg=LightMagenta",
6369 "Title term=bold ctermfg=LightMagenta gui=bold guifg=Magenta"),
6370 CENT("WarningMsg term=standout ctermfg=LightRed",
6371 "WarningMsg term=standout ctermfg=LightRed guifg=Red"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006372#ifdef FEAT_WILDMENU
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006373 CENT("WildMenu term=standout ctermbg=Yellow ctermfg=Black",
6374 "WildMenu term=standout ctermbg=Yellow ctermfg=Black guibg=Yellow guifg=Black"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006375#endif
6376#ifdef FEAT_FOLDING
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006377 CENT("Folded term=standout ctermbg=DarkGrey ctermfg=Cyan",
6378 "Folded term=standout ctermbg=DarkGrey ctermfg=Cyan guibg=DarkGrey guifg=Cyan"),
6379 CENT("FoldColumn term=standout ctermbg=DarkGrey ctermfg=Cyan",
6380 "FoldColumn term=standout ctermbg=DarkGrey ctermfg=Cyan guibg=Grey guifg=Cyan"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006381#endif
6382#ifdef FEAT_SIGNS
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006383 CENT("SignColumn term=standout ctermbg=DarkGrey ctermfg=Cyan",
6384 "SignColumn term=standout ctermbg=DarkGrey ctermfg=Cyan guibg=Grey guifg=Cyan"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006385#endif
6386#ifdef FEAT_VISUAL
Bram Moolenaarc1e37902006-04-18 21:55:01 +00006387 CENT("Visual term=reverse",
6388 "Visual term=reverse guibg=DarkGrey"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006389#endif
6390#ifdef FEAT_DIFF
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006391 CENT("DiffAdd term=bold ctermbg=DarkBlue",
6392 "DiffAdd term=bold ctermbg=DarkBlue guibg=DarkBlue"),
6393 CENT("DiffChange term=bold ctermbg=DarkMagenta",
6394 "DiffChange term=bold ctermbg=DarkMagenta guibg=DarkMagenta"),
6395 CENT("DiffDelete term=bold ctermfg=Blue ctermbg=DarkCyan",
6396 "DiffDelete term=bold ctermfg=Blue ctermbg=DarkCyan gui=bold guifg=Blue guibg=DarkCyan"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006397#endif
6398#ifdef FEAT_WINDOWS
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006399 CENT("TabLine term=underline cterm=underline ctermfg=white ctermbg=DarkGrey",
6400 "TabLine term=underline cterm=underline ctermfg=white ctermbg=DarkGrey gui=underline guibg=DarkGrey"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006401#endif
6402#ifdef FEAT_SYN_HL
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006403 CENT("CursorColumn term=reverse ctermbg=DarkGrey",
Bram Moolenaarb21e5842006-04-16 18:30:08 +00006404 "CursorColumn term=reverse ctermbg=DarkGrey guibg=Grey40"),
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006405 CENT("CursorLine term=underline cterm=underline",
Bram Moolenaarb21e5842006-04-16 18:30:08 +00006406 "CursorLine term=underline cterm=underline guibg=Grey40"),
6407#endif
6408#ifdef FEAT_AUTOCMD
6409 CENT("MatchParen term=reverse ctermbg=DarkCyan",
6410 "MatchParen term=reverse ctermbg=DarkCyan guibg=DarkCyan"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006411#endif
6412#ifdef FEAT_GUI
Bram Moolenaar071d4272004-06-13 20:20:40 +00006413 "Normal gui=NONE",
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006414#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00006415 NULL
6416 };
6417
6418 void
6419init_highlight(both, reset)
6420 int both; /* include groups where 'bg' doesn't matter */
6421 int reset; /* clear group first */
6422{
6423 int i;
6424 char **pp;
6425 static int had_both = FALSE;
6426#ifdef FEAT_EVAL
6427 char_u *p;
6428
6429 /*
6430 * Try finding the color scheme file. Used when a color file was loaded
6431 * and 'background' or 't_Co' is changed.
6432 */
6433 p = get_var_value((char_u *)"g:colors_name");
6434 if (p != NULL && load_colors(p) == OK)
6435 return;
6436#endif
6437
6438 /*
6439 * Didn't use a color file, use the compiled-in colors.
6440 */
6441 if (both)
6442 {
6443 had_both = TRUE;
6444 pp = highlight_init_both;
6445 for (i = 0; pp[i] != NULL; ++i)
6446 do_highlight((char_u *)pp[i], reset, TRUE);
6447 }
6448 else if (!had_both)
6449 /* Don't do anything before the call with both == TRUE from main().
6450 * Not everything has been setup then, and that call will overrule
6451 * everything anyway. */
6452 return;
6453
6454 if (*p_bg == 'l')
6455 pp = highlight_init_light;
6456 else
6457 pp = highlight_init_dark;
6458 for (i = 0; pp[i] != NULL; ++i)
6459 do_highlight((char_u *)pp[i], reset, TRUE);
6460
Bram Moolenaarc1e37902006-04-18 21:55:01 +00006461 /* Reverse looks ugly, but grey may not work for 8 colors. Thus let it
Bram Moolenaarf193fff2006-04-27 00:02:13 +00006462 * depend on the number of colors available.
6463 * With 8 colors brown is equal to yellow, need to use black for Search fg
Bram Moolenaarda40c852008-08-06 13:28:57 +00006464 * to avoid Statement highlighted text disappears.
6465 * Clear the attributes, needed when changing the t_Co value. */
Bram Moolenaarab194812005-09-14 21:40:12 +00006466 if (t_colors > 8)
Bram Moolenaarda40c852008-08-06 13:28:57 +00006467 do_highlight((char_u *)(*p_bg == 'l'
6468 ? "Visual cterm=NONE ctermbg=LightGrey"
6469 : "Visual cterm=NONE ctermbg=DarkGrey"), FALSE, TRUE);
Bram Moolenaarc1e37902006-04-18 21:55:01 +00006470 else
Bram Moolenaarf193fff2006-04-27 00:02:13 +00006471 {
Bram Moolenaarda40c852008-08-06 13:28:57 +00006472 do_highlight((char_u *)"Visual cterm=reverse ctermbg=NONE",
6473 FALSE, TRUE);
Bram Moolenaarf193fff2006-04-27 00:02:13 +00006474 if (*p_bg == 'l')
6475 do_highlight((char_u *)"Search ctermfg=black", FALSE, TRUE);
6476 }
Bram Moolenaarab194812005-09-14 21:40:12 +00006477
Bram Moolenaar071d4272004-06-13 20:20:40 +00006478#ifdef FEAT_SYN_HL
6479 /*
6480 * If syntax highlighting is enabled load the highlighting for it.
6481 */
6482 if (get_var_value((char_u *)"g:syntax_on") != NULL)
Bram Moolenaarc0197e22004-09-13 20:26:32 +00006483 {
6484 static int recursive = 0;
6485
6486 if (recursive >= 5)
6487 EMSG(_("E679: recursive loop loading syncolor.vim"));
6488 else
6489 {
6490 ++recursive;
Bram Moolenaar90cfdbe2005-08-12 19:59:19 +00006491 (void)source_runtime((char_u *)"syntax/syncolor.vim", TRUE);
Bram Moolenaarc0197e22004-09-13 20:26:32 +00006492 --recursive;
6493 }
6494 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00006495#endif
6496}
6497
6498/*
Bram Moolenaarcfbc5ee2004-07-02 15:38:35 +00006499 * Load color file "name".
Bram Moolenaar071d4272004-06-13 20:20:40 +00006500 * Return OK for success, FAIL for failure.
6501 */
6502 int
Bram Moolenaarcfbc5ee2004-07-02 15:38:35 +00006503load_colors(name)
6504 char_u *name;
Bram Moolenaar071d4272004-06-13 20:20:40 +00006505{
6506 char_u *buf;
6507 int retval = FAIL;
6508 static int recursive = FALSE;
6509
6510 /* When being called recursively, this is probably because setting
6511 * 'background' caused the highlighting to be reloaded. This means it is
6512 * working, thus we should return OK. */
6513 if (recursive)
6514 return OK;
6515
6516 recursive = TRUE;
Bram Moolenaarcfbc5ee2004-07-02 15:38:35 +00006517 buf = alloc((unsigned)(STRLEN(name) + 12));
Bram Moolenaar071d4272004-06-13 20:20:40 +00006518 if (buf != NULL)
6519 {
Bram Moolenaarcfbc5ee2004-07-02 15:38:35 +00006520 sprintf((char *)buf, "colors/%s.vim", name);
Bram Moolenaar90cfdbe2005-08-12 19:59:19 +00006521 retval = source_runtime(buf, FALSE);
Bram Moolenaar071d4272004-06-13 20:20:40 +00006522 vim_free(buf);
Bram Moolenaarcfbc5ee2004-07-02 15:38:35 +00006523#ifdef FEAT_AUTOCMD
6524 apply_autocmds(EVENT_COLORSCHEME, NULL, NULL, FALSE, curbuf);
6525#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00006526 }
6527 recursive = FALSE;
6528
6529 return retval;
6530}
6531
6532/*
6533 * Handle the ":highlight .." command.
6534 * When using ":hi clear" this is called recursively for each group with
6535 * "forceit" and "init" both TRUE.
6536 */
6537 void
6538do_highlight(line, forceit, init)
6539 char_u *line;
6540 int forceit;
6541 int init; /* TRUE when called for initializing */
6542{
6543 char_u *name_end;
6544 char_u *p;
6545 char_u *linep;
6546 char_u *key_start;
6547 char_u *arg_start;
6548 char_u *key = NULL, *arg = NULL;
6549 long i;
6550 int off;
6551 int len;
6552 int attr;
6553 int id;
6554 int idx;
6555 int dodefault = FALSE;
6556 int doclear = FALSE;
6557 int dolink = FALSE;
6558 int error = FALSE;
6559 int color;
6560 int is_normal_group = FALSE; /* "Normal" group */
6561#ifdef FEAT_GUI_X11
6562 int is_menu_group = FALSE; /* "Menu" group */
6563 int is_scrollbar_group = FALSE; /* "Scrollbar" group */
6564 int is_tooltip_group = FALSE; /* "Tooltip" group */
6565 int do_colors = FALSE; /* need to update colors? */
6566#else
6567# define is_menu_group 0
6568# define is_tooltip_group 0
6569#endif
6570
6571 /*
6572 * If no argument, list current highlighting.
6573 */
6574 if (ends_excmd(*line))
6575 {
6576 for (i = 1; i <= highlight_ga.ga_len && !got_int; ++i)
6577 /* TODO: only call when the group has attributes set */
6578 highlight_list_one((int)i);
6579 return;
6580 }
6581
6582 /*
6583 * Isolate the name.
6584 */
6585 name_end = skiptowhite(line);
6586 linep = skipwhite(name_end);
6587
6588 /*
6589 * Check for "default" argument.
6590 */
6591 if (STRNCMP(line, "default", name_end - line) == 0)
6592 {
6593 dodefault = TRUE;
6594 line = linep;
6595 name_end = skiptowhite(line);
6596 linep = skipwhite(name_end);
6597 }
6598
6599 /*
6600 * Check for "clear" or "link" argument.
6601 */
6602 if (STRNCMP(line, "clear", name_end - line) == 0)
6603 doclear = TRUE;
6604 if (STRNCMP(line, "link", name_end - line) == 0)
6605 dolink = TRUE;
6606
6607 /*
6608 * ":highlight {group-name}": list highlighting for one group.
6609 */
6610 if (!doclear && !dolink && ends_excmd(*linep))
6611 {
6612 id = syn_namen2id(line, (int)(name_end - line));
6613 if (id == 0)
6614 EMSG2(_("E411: highlight group not found: %s"), line);
6615 else
6616 highlight_list_one(id);
6617 return;
6618 }
6619
6620 /*
6621 * Handle ":highlight link {from} {to}" command.
6622 */
6623 if (dolink)
6624 {
6625 char_u *from_start = linep;
6626 char_u *from_end;
6627 char_u *to_start;
6628 char_u *to_end;
6629 int from_id;
6630 int to_id;
6631
6632 from_end = skiptowhite(from_start);
6633 to_start = skipwhite(from_end);
6634 to_end = skiptowhite(to_start);
6635
6636 if (ends_excmd(*from_start) || ends_excmd(*to_start))
6637 {
6638 EMSG2(_("E412: Not enough arguments: \":highlight link %s\""),
6639 from_start);
6640 return;
6641 }
6642
6643 if (!ends_excmd(*skipwhite(to_end)))
6644 {
6645 EMSG2(_("E413: Too many arguments: \":highlight link %s\""), from_start);
6646 return;
6647 }
6648
6649 from_id = syn_check_group(from_start, (int)(from_end - from_start));
6650 if (STRNCMP(to_start, "NONE", 4) == 0)
6651 to_id = 0;
6652 else
6653 to_id = syn_check_group(to_start, (int)(to_end - to_start));
6654
6655 if (from_id > 0 && (!init || HL_TABLE()[from_id - 1].sg_set == 0))
6656 {
6657 /*
6658 * Don't allow a link when there already is some highlighting
6659 * for the group, unless '!' is used
6660 */
6661 if (to_id > 0 && !forceit && !init
6662 && hl_has_settings(from_id - 1, dodefault))
6663 {
6664 if (sourcing_name == NULL && !dodefault)
6665 EMSG(_("E414: group has settings, highlight link ignored"));
6666 }
6667 else
6668 {
6669 if (!init)
6670 HL_TABLE()[from_id - 1].sg_set |= SG_LINK;
6671 HL_TABLE()[from_id - 1].sg_link = to_id;
Bram Moolenaar661b1822005-07-28 22:36:45 +00006672#ifdef FEAT_EVAL
6673 HL_TABLE()[from_id - 1].sg_scriptID = current_SID;
6674#endif
Bram Moolenaar1c8f93f2006-03-12 22:10:07 +00006675 redraw_all_later(SOME_VALID);
Bram Moolenaar071d4272004-06-13 20:20:40 +00006676 }
6677 }
6678
6679 /* Only call highlight_changed() once, after sourcing a syntax file */
6680 need_highlight_changed = TRUE;
6681
6682 return;
6683 }
6684
6685 if (doclear)
6686 {
6687 /*
6688 * ":highlight clear [group]" command.
6689 */
6690 line = linep;
6691 if (ends_excmd(*line))
6692 {
6693#ifdef FEAT_GUI
6694 /* First, we do not destroy the old values, but allocate the new
6695 * ones and update the display. THEN we destroy the old values.
6696 * If we destroy the old values first, then the old values
6697 * (such as GuiFont's or GuiFontset's) will still be displayed but
6698 * invalid because they were free'd.
6699 */
6700 if (gui.in_use)
6701 {
6702# ifdef FEAT_BEVAL_TIP
6703 gui_init_tooltip_font();
6704# endif
6705# if defined(FEAT_MENU) && (defined(FEAT_GUI_ATHENA) || defined(FEAT_GUI_MOTIF))
6706 gui_init_menu_font();
6707# endif
6708 }
6709# if defined(FEAT_GUI_MSWIN) || defined(FEAT_GUI_X11)
6710 gui_mch_def_colors();
6711# endif
6712# ifdef FEAT_GUI_X11
6713# ifdef FEAT_MENU
6714
6715 /* This only needs to be done when there is no Menu highlight
6716 * group defined by default, which IS currently the case.
6717 */
6718 gui_mch_new_menu_colors();
6719# endif
6720 if (gui.in_use)
6721 {
6722 gui_new_scrollbar_colors();
6723# ifdef FEAT_BEVAL
6724 gui_mch_new_tooltip_colors();
6725# endif
6726# ifdef FEAT_MENU
6727 gui_mch_new_menu_font();
6728# endif
6729 }
6730# endif
6731
6732 /* Ok, we're done allocating the new default graphics items.
6733 * The screen should already be refreshed at this point.
6734 * It is now Ok to clear out the old data.
6735 */
6736#endif
6737#ifdef FEAT_EVAL
Bram Moolenaar2ce06f62005-01-31 19:19:04 +00006738 do_unlet((char_u *)"colors_name", TRUE);
Bram Moolenaar071d4272004-06-13 20:20:40 +00006739#endif
6740 restore_cterm_colors();
6741
6742 /*
6743 * Clear all default highlight groups and load the defaults.
6744 */
6745 for (idx = 0; idx < highlight_ga.ga_len; ++idx)
6746 highlight_clear(idx);
6747 init_highlight(TRUE, TRUE);
6748#ifdef FEAT_GUI
6749 if (gui.in_use)
6750 highlight_gui_started();
6751#endif
6752 highlight_changed();
6753 redraw_later_clear();
6754 return;
6755 }
6756 name_end = skiptowhite(line);
6757 linep = skipwhite(name_end);
6758 }
6759
6760 /*
6761 * Find the group name in the table. If it does not exist yet, add it.
6762 */
6763 id = syn_check_group(line, (int)(name_end - line));
6764 if (id == 0) /* failed (out of memory) */
6765 return;
6766 idx = id - 1; /* index is ID minus one */
6767
6768 /* Return if "default" was used and the group already has settings. */
6769 if (dodefault && hl_has_settings(idx, TRUE))
6770 return;
6771
6772 if (STRCMP(HL_TABLE()[idx].sg_name_u, "NORMAL") == 0)
6773 is_normal_group = TRUE;
6774#ifdef FEAT_GUI_X11
6775 else if (STRCMP(HL_TABLE()[idx].sg_name_u, "MENU") == 0)
6776 is_menu_group = TRUE;
6777 else if (STRCMP(HL_TABLE()[idx].sg_name_u, "SCROLLBAR") == 0)
6778 is_scrollbar_group = TRUE;
6779 else if (STRCMP(HL_TABLE()[idx].sg_name_u, "TOOLTIP") == 0)
6780 is_tooltip_group = TRUE;
6781#endif
6782
6783 /* Clear the highlighting for ":hi clear {group}" and ":hi clear". */
6784 if (doclear || (forceit && init))
6785 {
6786 highlight_clear(idx);
6787 if (!doclear)
6788 HL_TABLE()[idx].sg_set = 0;
6789 }
6790
6791 if (!doclear)
6792 while (!ends_excmd(*linep))
6793 {
6794 key_start = linep;
6795 if (*linep == '=')
6796 {
6797 EMSG2(_("E415: unexpected equal sign: %s"), key_start);
6798 error = TRUE;
6799 break;
6800 }
6801
6802 /*
6803 * Isolate the key ("term", "ctermfg", "ctermbg", "font", "guifg" or
6804 * "guibg").
6805 */
6806 while (*linep && !vim_iswhite(*linep) && *linep != '=')
6807 ++linep;
6808 vim_free(key);
6809 key = vim_strnsave_up(key_start, (int)(linep - key_start));
6810 if (key == NULL)
6811 {
6812 error = TRUE;
6813 break;
6814 }
6815 linep = skipwhite(linep);
6816
6817 if (STRCMP(key, "NONE") == 0)
6818 {
6819 if (!init || HL_TABLE()[idx].sg_set == 0)
6820 {
6821 if (!init)
6822 HL_TABLE()[idx].sg_set |= SG_TERM+SG_CTERM+SG_GUI;
6823 highlight_clear(idx);
6824 }
6825 continue;
6826 }
6827
6828 /*
6829 * Check for the equal sign.
6830 */
6831 if (*linep != '=')
6832 {
6833 EMSG2(_("E416: missing equal sign: %s"), key_start);
6834 error = TRUE;
6835 break;
6836 }
6837 ++linep;
6838
6839 /*
6840 * Isolate the argument.
6841 */
6842 linep = skipwhite(linep);
6843 if (*linep == '\'') /* guifg='color name' */
6844 {
6845 arg_start = ++linep;
6846 linep = vim_strchr(linep, '\'');
6847 if (linep == NULL)
6848 {
6849 EMSG2(_(e_invarg2), key_start);
6850 error = TRUE;
6851 break;
6852 }
6853 }
6854 else
6855 {
6856 arg_start = linep;
6857 linep = skiptowhite(linep);
6858 }
6859 if (linep == arg_start)
6860 {
6861 EMSG2(_("E417: missing argument: %s"), key_start);
6862 error = TRUE;
6863 break;
6864 }
6865 vim_free(arg);
6866 arg = vim_strnsave(arg_start, (int)(linep - arg_start));
6867 if (arg == NULL)
6868 {
6869 error = TRUE;
6870 break;
6871 }
6872 if (*linep == '\'')
6873 ++linep;
6874
6875 /*
6876 * Store the argument.
6877 */
6878 if ( STRCMP(key, "TERM") == 0
6879 || STRCMP(key, "CTERM") == 0
6880 || STRCMP(key, "GUI") == 0)
6881 {
6882 attr = 0;
6883 off = 0;
6884 while (arg[off] != NUL)
6885 {
6886 for (i = sizeof(hl_attr_table) / sizeof(int); --i >= 0; )
6887 {
6888 len = (int)STRLEN(hl_name_table[i]);
6889 if (STRNICMP(arg + off, hl_name_table[i], len) == 0)
6890 {
6891 attr |= hl_attr_table[i];
6892 off += len;
6893 break;
6894 }
6895 }
6896 if (i < 0)
6897 {
6898 EMSG2(_("E418: Illegal value: %s"), arg);
6899 error = TRUE;
6900 break;
6901 }
6902 if (arg[off] == ',') /* another one follows */
6903 ++off;
6904 }
6905 if (error)
6906 break;
6907 if (*key == 'T')
6908 {
6909 if (!init || !(HL_TABLE()[idx].sg_set & SG_TERM))
6910 {
6911 if (!init)
6912 HL_TABLE()[idx].sg_set |= SG_TERM;
6913 HL_TABLE()[idx].sg_term = attr;
6914 }
6915 }
6916 else if (*key == 'C')
6917 {
6918 if (!init || !(HL_TABLE()[idx].sg_set & SG_CTERM))
6919 {
6920 if (!init)
6921 HL_TABLE()[idx].sg_set |= SG_CTERM;
6922 HL_TABLE()[idx].sg_cterm = attr;
6923 HL_TABLE()[idx].sg_cterm_bold = FALSE;
6924 }
6925 }
6926#ifdef FEAT_GUI
6927 else
6928 {
6929 if (!init || !(HL_TABLE()[idx].sg_set & SG_GUI))
6930 {
6931 if (!init)
6932 HL_TABLE()[idx].sg_set |= SG_GUI;
6933 HL_TABLE()[idx].sg_gui = attr;
6934 }
6935 }
6936#endif
6937 }
6938 else if (STRCMP(key, "FONT") == 0)
6939 {
6940 /* in non-GUI fonts are simply ignored */
6941#ifdef FEAT_GUI
6942 if (!gui.shell_created)
6943 {
6944 /* GUI not started yet, always accept the name. */
6945 vim_free(HL_TABLE()[idx].sg_font_name);
6946 HL_TABLE()[idx].sg_font_name = vim_strsave(arg);
6947 }
6948 else
6949 {
6950 GuiFont temp_sg_font = HL_TABLE()[idx].sg_font;
6951# ifdef FEAT_XFONTSET
6952 GuiFontset temp_sg_fontset = HL_TABLE()[idx].sg_fontset;
6953# endif
6954 /* First, save the current font/fontset.
6955 * Then try to allocate the font/fontset.
6956 * If the allocation fails, HL_TABLE()[idx].sg_font OR
6957 * sg_fontset will be set to NOFONT or NOFONTSET respectively.
6958 */
6959
6960 HL_TABLE()[idx].sg_font = NOFONT;
6961# ifdef FEAT_XFONTSET
6962 HL_TABLE()[idx].sg_fontset = NOFONTSET;
6963# endif
6964 hl_do_font(idx, arg, is_normal_group, is_menu_group,
6965 is_tooltip_group);
6966
6967# ifdef FEAT_XFONTSET
6968 if (HL_TABLE()[idx].sg_fontset != NOFONTSET)
6969 {
6970 /* New fontset was accepted. Free the old one, if there was
6971 * one.
6972 */
6973 gui_mch_free_fontset(temp_sg_fontset);
6974 vim_free(HL_TABLE()[idx].sg_font_name);
6975 HL_TABLE()[idx].sg_font_name = vim_strsave(arg);
6976 }
6977 else
6978 HL_TABLE()[idx].sg_fontset = temp_sg_fontset;
6979# endif
6980 if (HL_TABLE()[idx].sg_font != NOFONT)
6981 {
6982 /* New font was accepted. Free the old one, if there was
6983 * one.
6984 */
6985 gui_mch_free_font(temp_sg_font);
6986 vim_free(HL_TABLE()[idx].sg_font_name);
6987 HL_TABLE()[idx].sg_font_name = vim_strsave(arg);
6988 }
6989 else
6990 HL_TABLE()[idx].sg_font = temp_sg_font;
6991 }
6992#endif
6993 }
6994 else if (STRCMP(key, "CTERMFG") == 0 || STRCMP(key, "CTERMBG") == 0)
6995 {
6996 if (!init || !(HL_TABLE()[idx].sg_set & SG_CTERM))
6997 {
6998 if (!init)
6999 HL_TABLE()[idx].sg_set |= SG_CTERM;
7000
7001 /* When setting the foreground color, and previously the "bold"
7002 * flag was set for a light color, reset it now */
7003 if (key[5] == 'F' && HL_TABLE()[idx].sg_cterm_bold)
7004 {
7005 HL_TABLE()[idx].sg_cterm &= ~HL_BOLD;
7006 HL_TABLE()[idx].sg_cterm_bold = FALSE;
7007 }
7008
7009 if (VIM_ISDIGIT(*arg))
7010 color = atoi((char *)arg);
7011 else if (STRICMP(arg, "fg") == 0)
7012 {
7013 if (cterm_normal_fg_color)
7014 color = cterm_normal_fg_color - 1;
7015 else
7016 {
7017 EMSG(_("E419: FG color unknown"));
7018 error = TRUE;
7019 break;
7020 }
7021 }
7022 else if (STRICMP(arg, "bg") == 0)
7023 {
7024 if (cterm_normal_bg_color > 0)
7025 color = cterm_normal_bg_color - 1;
7026 else
7027 {
7028 EMSG(_("E420: BG color unknown"));
7029 error = TRUE;
7030 break;
7031 }
7032 }
7033 else
7034 {
7035 static char *(color_names[28]) = {
7036 "Black", "DarkBlue", "DarkGreen", "DarkCyan",
7037 "DarkRed", "DarkMagenta", "Brown", "DarkYellow",
7038 "Gray", "Grey",
7039 "LightGray", "LightGrey", "DarkGray", "DarkGrey",
7040 "Blue", "LightBlue", "Green", "LightGreen",
7041 "Cyan", "LightCyan", "Red", "LightRed", "Magenta",
7042 "LightMagenta", "Yellow", "LightYellow", "White", "NONE"};
7043 static int color_numbers_16[28] = {0, 1, 2, 3,
7044 4, 5, 6, 6,
7045 7, 7,
7046 7, 7, 8, 8,
7047 9, 9, 10, 10,
7048 11, 11, 12, 12, 13,
7049 13, 14, 14, 15, -1};
7050 /* for xterm with 88 colors... */
7051 static int color_numbers_88[28] = {0, 4, 2, 6,
7052 1, 5, 32, 72,
7053 84, 84,
7054 7, 7, 82, 82,
7055 12, 43, 10, 61,
7056 14, 63, 9, 74, 13,
7057 75, 11, 78, 15, -1};
7058 /* for xterm with 256 colors... */
7059 static int color_numbers_256[28] = {0, 4, 2, 6,
7060 1, 5, 130, 130,
7061 248, 248,
7062 7, 7, 242, 242,
7063 12, 81, 10, 121,
7064 14, 159, 9, 224, 13,
7065 225, 11, 229, 15, -1};
7066 /* for terminals with less than 16 colors... */
7067 static int color_numbers_8[28] = {0, 4, 2, 6,
7068 1, 5, 3, 3,
7069 7, 7,
7070 7, 7, 0+8, 0+8,
7071 4+8, 4+8, 2+8, 2+8,
7072 6+8, 6+8, 1+8, 1+8, 5+8,
7073 5+8, 3+8, 3+8, 7+8, -1};
7074#if defined(__QNXNTO__)
7075 static int *color_numbers_8_qansi = color_numbers_8;
7076 /* On qnx, the 8 & 16 color arrays are the same */
7077 if (STRNCMP(T_NAME, "qansi", 5) == 0)
7078 color_numbers_8_qansi = color_numbers_16;
7079#endif
7080
7081 /* reduce calls to STRICMP a bit, it can be slow */
7082 off = TOUPPER_ASC(*arg);
7083 for (i = (sizeof(color_names) / sizeof(char *)); --i >= 0; )
7084 if (off == color_names[i][0]
7085 && STRICMP(arg + 1, color_names[i] + 1) == 0)
7086 break;
7087 if (i < 0)
7088 {
7089 EMSG2(_("E421: Color name or number not recognized: %s"), key_start);
7090 error = TRUE;
7091 break;
7092 }
7093
7094 /* Use the _16 table to check if its a valid color name. */
7095 color = color_numbers_16[i];
7096 if (color >= 0)
7097 {
7098 if (t_colors == 8)
7099 {
7100 /* t_Co is 8: use the 8 colors table */
7101#if defined(__QNXNTO__)
7102 color = color_numbers_8_qansi[i];
7103#else
7104 color = color_numbers_8[i];
7105#endif
7106 if (key[5] == 'F')
7107 {
7108 /* set/reset bold attribute to get light foreground
7109 * colors (on some terminals, e.g. "linux") */
7110 if (color & 8)
7111 {
7112 HL_TABLE()[idx].sg_cterm |= HL_BOLD;
7113 HL_TABLE()[idx].sg_cterm_bold = TRUE;
7114 }
7115 else
7116 HL_TABLE()[idx].sg_cterm &= ~HL_BOLD;
7117 }
7118 color &= 7; /* truncate to 8 colors */
7119 }
7120 else if (t_colors == 16 || t_colors == 88
7121 || t_colors == 256)
7122 {
7123 /*
7124 * Guess: if the termcap entry ends in 'm', it is
7125 * probably an xterm-like terminal. Use the changed
7126 * order for colors.
7127 */
7128 if (*T_CAF != NUL)
7129 p = T_CAF;
7130 else
7131 p = T_CSF;
7132 if (*p != NUL && *(p + STRLEN(p) - 1) == 'm')
7133 switch (t_colors)
7134 {
7135 case 16:
7136 color = color_numbers_8[i];
7137 break;
7138 case 88:
7139 color = color_numbers_88[i];
7140 break;
7141 case 256:
7142 color = color_numbers_256[i];
7143 break;
7144 }
7145 }
7146 }
7147 }
7148 /* Add one to the argument, to avoid zero */
7149 if (key[5] == 'F')
7150 {
7151 HL_TABLE()[idx].sg_cterm_fg = color + 1;
7152 if (is_normal_group)
7153 {
7154 cterm_normal_fg_color = color + 1;
7155 cterm_normal_fg_bold = (HL_TABLE()[idx].sg_cterm & HL_BOLD);
7156#ifdef FEAT_GUI
7157 /* Don't do this if the GUI is used. */
7158 if (!gui.in_use && !gui.starting)
7159#endif
7160 {
7161 must_redraw = CLEAR;
7162 if (termcap_active)
7163 term_fg_color(color);
7164 }
7165 }
7166 }
7167 else
7168 {
7169 HL_TABLE()[idx].sg_cterm_bg = color + 1;
7170 if (is_normal_group)
7171 {
7172 cterm_normal_bg_color = color + 1;
7173#ifdef FEAT_GUI
7174 /* Don't mess with 'background' if the GUI is used. */
7175 if (!gui.in_use && !gui.starting)
7176#endif
7177 {
7178 must_redraw = CLEAR;
7179 if (termcap_active)
7180 term_bg_color(color);
7181 if (t_colors < 16)
7182 i = (color == 0 || color == 4);
7183 else
7184 i = (color < 7 || color == 8);
7185 /* Set the 'background' option if the value is wrong. */
7186 if (i != (*p_bg == 'd'))
7187 set_option_value((char_u *)"bg", 0L,
7188 i ? (char_u *)"dark" : (char_u *)"light", 0);
7189 }
7190 }
7191 }
7192 }
7193 }
7194 else if (STRCMP(key, "GUIFG") == 0)
7195 {
7196#ifdef FEAT_GUI /* in non-GUI guifg colors are simply ignored */
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007197 if (!init || !(HL_TABLE()[idx].sg_set & SG_GUI))
Bram Moolenaar071d4272004-06-13 20:20:40 +00007198 {
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007199 if (!init)
7200 HL_TABLE()[idx].sg_set |= SG_GUI;
7201
7202 i = color_name2handle(arg);
7203 if (i != INVALCOLOR || STRCMP(arg, "NONE") == 0 || !gui.in_use)
7204 {
7205 HL_TABLE()[idx].sg_gui_fg = i;
7206 vim_free(HL_TABLE()[idx].sg_gui_fg_name);
7207 if (STRCMP(arg, "NONE"))
7208 HL_TABLE()[idx].sg_gui_fg_name = vim_strsave(arg);
7209 else
7210 HL_TABLE()[idx].sg_gui_fg_name = NULL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007211# ifdef FEAT_GUI_X11
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007212 if (is_menu_group)
7213 gui.menu_fg_pixel = i;
7214 if (is_scrollbar_group)
7215 gui.scroll_fg_pixel = i;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007216# ifdef FEAT_BEVAL
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007217 if (is_tooltip_group)
7218 gui.tooltip_fg_pixel = i;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007219# endif
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007220 do_colors = TRUE;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007221# endif
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007222 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00007223 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00007224#endif
7225 }
7226 else if (STRCMP(key, "GUIBG") == 0)
7227 {
7228#ifdef FEAT_GUI /* in non-GUI guibg colors are simply ignored */
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007229 if (!init || !(HL_TABLE()[idx].sg_set & SG_GUI))
Bram Moolenaar071d4272004-06-13 20:20:40 +00007230 {
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007231 if (!init)
7232 HL_TABLE()[idx].sg_set |= SG_GUI;
7233
7234 i = color_name2handle(arg);
7235 if (i != INVALCOLOR || STRCMP(arg, "NONE") == 0 || !gui.in_use)
7236 {
7237 HL_TABLE()[idx].sg_gui_bg = i;
7238 vim_free(HL_TABLE()[idx].sg_gui_bg_name);
7239 if (STRCMP(arg, "NONE") != 0)
7240 HL_TABLE()[idx].sg_gui_bg_name = vim_strsave(arg);
7241 else
7242 HL_TABLE()[idx].sg_gui_bg_name = NULL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007243# ifdef FEAT_GUI_X11
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007244 if (is_menu_group)
7245 gui.menu_bg_pixel = i;
7246 if (is_scrollbar_group)
7247 gui.scroll_bg_pixel = i;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007248# ifdef FEAT_BEVAL
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007249 if (is_tooltip_group)
7250 gui.tooltip_bg_pixel = i;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007251# endif
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007252 do_colors = TRUE;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007253# endif
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007254 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00007255 }
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007256#endif
7257 }
7258 else if (STRCMP(key, "GUISP") == 0)
7259 {
7260#ifdef FEAT_GUI /* in non-GUI guisp colors are simply ignored */
7261 if (!init || !(HL_TABLE()[idx].sg_set & SG_GUI))
7262 {
7263 if (!init)
7264 HL_TABLE()[idx].sg_set |= SG_GUI;
7265
7266 i = color_name2handle(arg);
7267 if (i != INVALCOLOR || STRCMP(arg, "NONE") == 0 || !gui.in_use)
7268 {
7269 HL_TABLE()[idx].sg_gui_sp = i;
7270 vim_free(HL_TABLE()[idx].sg_gui_sp_name);
7271 if (STRCMP(arg, "NONE") != 0)
7272 HL_TABLE()[idx].sg_gui_sp_name = vim_strsave(arg);
7273 else
7274 HL_TABLE()[idx].sg_gui_sp_name = NULL;
7275 }
7276 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00007277#endif
7278 }
7279 else if (STRCMP(key, "START") == 0 || STRCMP(key, "STOP") == 0)
7280 {
7281 char_u buf[100];
7282 char_u *tname;
7283
7284 if (!init)
7285 HL_TABLE()[idx].sg_set |= SG_TERM;
7286
7287 /*
7288 * The "start" and "stop" arguments can be a literal escape
Bram Moolenaarb8017e72007-05-10 18:59:07 +00007289 * sequence, or a comma separated list of terminal codes.
Bram Moolenaar071d4272004-06-13 20:20:40 +00007290 */
7291 if (STRNCMP(arg, "t_", 2) == 0)
7292 {
7293 off = 0;
7294 buf[0] = 0;
7295 while (arg[off] != NUL)
7296 {
7297 /* Isolate one termcap name */
7298 for (len = 0; arg[off + len] &&
7299 arg[off + len] != ','; ++len)
7300 ;
7301 tname = vim_strnsave(arg + off, len);
7302 if (tname == NULL) /* out of memory */
7303 {
7304 error = TRUE;
7305 break;
7306 }
7307 /* lookup the escape sequence for the item */
7308 p = get_term_code(tname);
7309 vim_free(tname);
7310 if (p == NULL) /* ignore non-existing things */
7311 p = (char_u *)"";
7312
7313 /* Append it to the already found stuff */
7314 if ((int)(STRLEN(buf) + STRLEN(p)) >= 99)
7315 {
7316 EMSG2(_("E422: terminal code too long: %s"), arg);
7317 error = TRUE;
7318 break;
7319 }
7320 STRCAT(buf, p);
7321
7322 /* Advance to the next item */
7323 off += len;
7324 if (arg[off] == ',') /* another one follows */
7325 ++off;
7326 }
7327 }
7328 else
7329 {
7330 /*
7331 * Copy characters from arg[] to buf[], translating <> codes.
7332 */
7333 for (p = arg, off = 0; off < 100 && *p; )
7334 {
7335 len = trans_special(&p, buf + off, FALSE);
7336 if (len) /* recognized special char */
7337 off += len;
7338 else /* copy as normal char */
7339 buf[off++] = *p++;
7340 }
7341 buf[off] = NUL;
7342 }
7343 if (error)
7344 break;
7345
7346 if (STRCMP(buf, "NONE") == 0) /* resetting the value */
7347 p = NULL;
7348 else
7349 p = vim_strsave(buf);
7350 if (key[2] == 'A')
7351 {
7352 vim_free(HL_TABLE()[idx].sg_start);
7353 HL_TABLE()[idx].sg_start = p;
7354 }
7355 else
7356 {
7357 vim_free(HL_TABLE()[idx].sg_stop);
7358 HL_TABLE()[idx].sg_stop = p;
7359 }
7360 }
7361 else
7362 {
7363 EMSG2(_("E423: Illegal argument: %s"), key_start);
7364 error = TRUE;
7365 break;
7366 }
7367
7368 /*
7369 * When highlighting has been given for a group, don't link it.
7370 */
7371 if (!init || !(HL_TABLE()[idx].sg_set & SG_LINK))
7372 HL_TABLE()[idx].sg_link = 0;
7373
7374 /*
7375 * Continue with next argument.
7376 */
7377 linep = skipwhite(linep);
7378 }
7379
7380 /*
7381 * If there is an error, and it's a new entry, remove it from the table.
7382 */
7383 if (error && idx == highlight_ga.ga_len)
7384 syn_unadd_group();
7385 else
7386 {
7387 if (is_normal_group)
7388 {
7389 HL_TABLE()[idx].sg_term_attr = 0;
7390 HL_TABLE()[idx].sg_cterm_attr = 0;
7391#ifdef FEAT_GUI
7392 HL_TABLE()[idx].sg_gui_attr = 0;
7393 /*
7394 * Need to update all groups, because they might be using "bg"
7395 * and/or "fg", which have been changed now.
7396 */
7397 if (gui.in_use)
7398 highlight_gui_started();
7399#endif
7400 }
7401#ifdef FEAT_GUI_X11
7402# ifdef FEAT_MENU
7403 else if (is_menu_group)
7404 {
7405 if (gui.in_use && do_colors)
7406 gui_mch_new_menu_colors();
7407 }
7408# endif
7409 else if (is_scrollbar_group)
7410 {
7411 if (gui.in_use && do_colors)
7412 gui_new_scrollbar_colors();
7413 }
7414# ifdef FEAT_BEVAL
7415 else if (is_tooltip_group)
7416 {
7417 if (gui.in_use && do_colors)
7418 gui_mch_new_tooltip_colors();
7419 }
7420# endif
7421#endif
7422 else
7423 set_hl_attr(idx);
Bram Moolenaar661b1822005-07-28 22:36:45 +00007424#ifdef FEAT_EVAL
7425 HL_TABLE()[idx].sg_scriptID = current_SID;
7426#endif
Bram Moolenaar910f66f2006-04-05 20:41:53 +00007427 redraw_all_later(NOT_VALID);
Bram Moolenaar071d4272004-06-13 20:20:40 +00007428 }
7429 vim_free(key);
7430 vim_free(arg);
7431
7432 /* Only call highlight_changed() once, after sourcing a syntax file */
7433 need_highlight_changed = TRUE;
7434}
7435
Bram Moolenaar1ec484f2005-06-24 23:07:47 +00007436#if defined(EXITFREE) || defined(PROTO)
7437 void
7438free_highlight()
7439{
7440 int i;
7441
7442 for (i = 0; i < highlight_ga.ga_len; ++i)
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00007443 {
Bram Moolenaar1ec484f2005-06-24 23:07:47 +00007444 highlight_clear(i);
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00007445 vim_free(HL_TABLE()[i].sg_name);
7446 vim_free(HL_TABLE()[i].sg_name_u);
7447 }
Bram Moolenaar1ec484f2005-06-24 23:07:47 +00007448 ga_clear(&highlight_ga);
7449}
7450#endif
7451
Bram Moolenaar071d4272004-06-13 20:20:40 +00007452/*
7453 * Reset the cterm colors to what they were before Vim was started, if
7454 * possible. Otherwise reset them to zero.
7455 */
7456 void
7457restore_cterm_colors()
7458{
7459#if defined(MSDOS) || (defined(WIN3264) && !defined(FEAT_GUI_W32))
7460 /* Since t_me has been set, this probably means that the user
7461 * wants to use this as default colors. Need to reset default
7462 * background/foreground colors. */
7463 mch_set_normal_colors();
7464#else
7465 cterm_normal_fg_color = 0;
7466 cterm_normal_fg_bold = 0;
7467 cterm_normal_bg_color = 0;
7468#endif
7469}
7470
7471/*
7472 * Return TRUE if highlight group "idx" has any settings.
7473 * When "check_link" is TRUE also check for an existing link.
7474 */
7475 static int
7476hl_has_settings(idx, check_link)
7477 int idx;
7478 int check_link;
7479{
7480 return ( HL_TABLE()[idx].sg_term_attr != 0
7481 || HL_TABLE()[idx].sg_cterm_attr != 0
7482#ifdef FEAT_GUI
7483 || HL_TABLE()[idx].sg_gui_attr != 0
7484#endif
7485 || (check_link && (HL_TABLE()[idx].sg_set & SG_LINK)));
7486}
7487
7488/*
7489 * Clear highlighting for one group.
7490 */
7491 static void
7492highlight_clear(idx)
7493 int idx;
7494{
7495 HL_TABLE()[idx].sg_term = 0;
7496 vim_free(HL_TABLE()[idx].sg_start);
7497 HL_TABLE()[idx].sg_start = NULL;
7498 vim_free(HL_TABLE()[idx].sg_stop);
7499 HL_TABLE()[idx].sg_stop = NULL;
7500 HL_TABLE()[idx].sg_term_attr = 0;
7501 HL_TABLE()[idx].sg_cterm = 0;
7502 HL_TABLE()[idx].sg_cterm_bold = FALSE;
7503 HL_TABLE()[idx].sg_cterm_fg = 0;
7504 HL_TABLE()[idx].sg_cterm_bg = 0;
7505 HL_TABLE()[idx].sg_cterm_attr = 0;
7506#ifdef FEAT_GUI /* in non-GUI fonts are simply ignored */
7507 HL_TABLE()[idx].sg_gui = 0;
7508 HL_TABLE()[idx].sg_gui_fg = INVALCOLOR;
7509 vim_free(HL_TABLE()[idx].sg_gui_fg_name);
7510 HL_TABLE()[idx].sg_gui_fg_name = NULL;
7511 HL_TABLE()[idx].sg_gui_bg = INVALCOLOR;
7512 vim_free(HL_TABLE()[idx].sg_gui_bg_name);
7513 HL_TABLE()[idx].sg_gui_bg_name = NULL;
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007514 HL_TABLE()[idx].sg_gui_sp = INVALCOLOR;
7515 vim_free(HL_TABLE()[idx].sg_gui_sp_name);
7516 HL_TABLE()[idx].sg_gui_sp_name = NULL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007517 gui_mch_free_font(HL_TABLE()[idx].sg_font);
7518 HL_TABLE()[idx].sg_font = NOFONT;
7519# ifdef FEAT_XFONTSET
7520 gui_mch_free_fontset(HL_TABLE()[idx].sg_fontset);
7521 HL_TABLE()[idx].sg_fontset = NOFONTSET;
7522# endif
7523 vim_free(HL_TABLE()[idx].sg_font_name);
7524 HL_TABLE()[idx].sg_font_name = NULL;
7525 HL_TABLE()[idx].sg_gui_attr = 0;
7526#endif
Bram Moolenaar661b1822005-07-28 22:36:45 +00007527#ifdef FEAT_EVAL
7528 /* Clear the script ID only when there is no link, since that is not
7529 * cleared. */
7530 if (HL_TABLE()[idx].sg_link == 0)
7531 HL_TABLE()[idx].sg_scriptID = 0;
7532#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00007533}
7534
7535#if defined(FEAT_GUI) || defined(PROTO)
7536/*
7537 * Set the normal foreground and background colors according to the "Normal"
7538 * highlighighting group. For X11 also set "Menu", "Scrollbar", and
7539 * "Tooltip" colors.
7540 */
7541 void
7542set_normal_colors()
7543{
7544 if (set_group_colors((char_u *)"Normal",
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007545 &gui.norm_pixel, &gui.back_pixel,
7546 FALSE, TRUE, FALSE))
Bram Moolenaar071d4272004-06-13 20:20:40 +00007547 {
7548 gui_mch_new_colors();
7549 must_redraw = CLEAR;
7550 }
7551#ifdef FEAT_GUI_X11
7552 if (set_group_colors((char_u *)"Menu",
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007553 &gui.menu_fg_pixel, &gui.menu_bg_pixel,
7554 TRUE, FALSE, FALSE))
Bram Moolenaar071d4272004-06-13 20:20:40 +00007555 {
7556# ifdef FEAT_MENU
7557 gui_mch_new_menu_colors();
7558# endif
7559 must_redraw = CLEAR;
7560 }
7561# ifdef FEAT_BEVAL
7562 if (set_group_colors((char_u *)"Tooltip",
7563 &gui.tooltip_fg_pixel, &gui.tooltip_bg_pixel,
7564 FALSE, FALSE, TRUE))
7565 {
7566# ifdef FEAT_TOOLBAR
7567 gui_mch_new_tooltip_colors();
7568# endif
7569 must_redraw = CLEAR;
7570 }
7571#endif
7572 if (set_group_colors((char_u *)"Scrollbar",
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007573 &gui.scroll_fg_pixel, &gui.scroll_bg_pixel,
7574 FALSE, FALSE, FALSE))
Bram Moolenaar071d4272004-06-13 20:20:40 +00007575 {
7576 gui_new_scrollbar_colors();
7577 must_redraw = CLEAR;
7578 }
7579#endif
7580}
7581
7582/*
7583 * Set the colors for "Normal", "Menu", "Tooltip" or "Scrollbar".
7584 */
7585 static int
7586set_group_colors(name, fgp, bgp, do_menu, use_norm, do_tooltip)
7587 char_u *name;
7588 guicolor_T *fgp;
7589 guicolor_T *bgp;
7590 int do_menu;
7591 int use_norm;
7592 int do_tooltip;
7593{
7594 int idx;
7595
7596 idx = syn_name2id(name) - 1;
7597 if (idx >= 0)
7598 {
7599 gui_do_one_color(idx, do_menu, do_tooltip);
7600
7601 if (HL_TABLE()[idx].sg_gui_fg != INVALCOLOR)
7602 *fgp = HL_TABLE()[idx].sg_gui_fg;
7603 else if (use_norm)
7604 *fgp = gui.def_norm_pixel;
7605 if (HL_TABLE()[idx].sg_gui_bg != INVALCOLOR)
7606 *bgp = HL_TABLE()[idx].sg_gui_bg;
7607 else if (use_norm)
7608 *bgp = gui.def_back_pixel;
7609 return TRUE;
7610 }
7611 return FALSE;
7612}
7613
7614/*
7615 * Get the font of the "Normal" group.
7616 * Returns "" when it's not found or not set.
7617 */
7618 char_u *
7619hl_get_font_name()
7620{
7621 int id;
7622 char_u *s;
7623
7624 id = syn_name2id((char_u *)"Normal");
7625 if (id > 0)
7626 {
7627 s = HL_TABLE()[id - 1].sg_font_name;
7628 if (s != NULL)
7629 return s;
7630 }
7631 return (char_u *)"";
7632}
7633
7634/*
7635 * Set font for "Normal" group. Called by gui_mch_init_font() when a font has
7636 * actually chosen to be used.
7637 */
7638 void
7639hl_set_font_name(font_name)
7640 char_u *font_name;
7641{
7642 int id;
7643
7644 id = syn_name2id((char_u *)"Normal");
7645 if (id > 0)
7646 {
7647 vim_free(HL_TABLE()[id - 1].sg_font_name);
7648 HL_TABLE()[id - 1].sg_font_name = vim_strsave(font_name);
7649 }
7650}
7651
7652/*
7653 * Set background color for "Normal" group. Called by gui_set_bg_color()
7654 * when the color is known.
7655 */
7656 void
7657hl_set_bg_color_name(name)
7658 char_u *name; /* must have been allocated */
7659{
7660 int id;
7661
7662 if (name != NULL)
7663 {
7664 id = syn_name2id((char_u *)"Normal");
7665 if (id > 0)
7666 {
7667 vim_free(HL_TABLE()[id - 1].sg_gui_bg_name);
7668 HL_TABLE()[id - 1].sg_gui_bg_name = name;
7669 }
7670 }
7671}
7672
7673/*
7674 * Set foreground color for "Normal" group. Called by gui_set_fg_color()
7675 * when the color is known.
7676 */
7677 void
7678hl_set_fg_color_name(name)
7679 char_u *name; /* must have been allocated */
7680{
7681 int id;
7682
7683 if (name != NULL)
7684 {
7685 id = syn_name2id((char_u *)"Normal");
7686 if (id > 0)
7687 {
7688 vim_free(HL_TABLE()[id - 1].sg_gui_fg_name);
7689 HL_TABLE()[id - 1].sg_gui_fg_name = name;
7690 }
7691 }
7692}
7693
7694/*
7695 * Return the handle for a color name.
7696 * Returns INVALCOLOR when failed.
7697 */
7698 static guicolor_T
7699color_name2handle(name)
7700 char_u *name;
7701{
7702 if (STRCMP(name, "NONE") == 0)
7703 return INVALCOLOR;
7704
7705 if (STRICMP(name, "fg") == 0 || STRICMP(name, "foreground") == 0)
7706 return gui.norm_pixel;
7707 if (STRICMP(name, "bg") == 0 || STRICMP(name, "background") == 0)
7708 return gui.back_pixel;
7709
7710 return gui_get_color(name);
7711}
7712
7713/*
7714 * Return the handle for a font name.
7715 * Returns NOFONT when failed.
7716 */
7717 static GuiFont
7718font_name2handle(name)
7719 char_u *name;
7720{
7721 if (STRCMP(name, "NONE") == 0)
7722 return NOFONT;
7723
7724 return gui_mch_get_font(name, TRUE);
7725}
7726
7727# ifdef FEAT_XFONTSET
7728/*
7729 * Return the handle for a fontset name.
7730 * Returns NOFONTSET when failed.
7731 */
7732 static GuiFontset
7733fontset_name2handle(name, fixed_width)
7734 char_u *name;
7735 int fixed_width;
7736{
7737 if (STRCMP(name, "NONE") == 0)
7738 return NOFONTSET;
7739
7740 return gui_mch_get_fontset(name, TRUE, fixed_width);
7741}
7742# endif
7743
7744/*
7745 * Get the font or fontset for one highlight group.
7746 */
7747/*ARGSUSED*/
7748 static void
7749hl_do_font(idx, arg, do_normal, do_menu, do_tooltip)
7750 int idx;
7751 char_u *arg;
7752 int do_normal; /* set normal font */
7753 int do_menu; /* set menu font */
7754 int do_tooltip; /* set tooltip font */
7755{
7756# ifdef FEAT_XFONTSET
7757 /* If 'guifontset' is not empty, first try using the name as a
7758 * fontset. If that doesn't work, use it as a font name. */
7759 if (*p_guifontset != NUL
7760# ifdef FONTSET_ALWAYS
7761 || do_menu
7762# endif
7763# ifdef FEAT_BEVAL_TIP
7764 /* In Athena & Motif, the Tooltip highlight group is always a fontset */
7765 || do_tooltip
7766# endif
7767 )
7768 HL_TABLE()[idx].sg_fontset = fontset_name2handle(arg, 0
7769# ifdef FONTSET_ALWAYS
7770 || do_menu
7771# endif
7772# ifdef FEAT_BEVAL_TIP
7773 || do_tooltip
7774# endif
7775 );
7776 if (HL_TABLE()[idx].sg_fontset != NOFONTSET)
7777 {
7778 /* If it worked and it's the Normal group, use it as the
7779 * normal fontset. Same for the Menu group. */
7780 if (do_normal)
7781 gui_init_font(arg, TRUE);
7782# if (defined(FEAT_GUI_MOTIF) || defined(FEAT_GUI_ATHENA)) && defined(FEAT_MENU)
7783 if (do_menu)
7784 {
7785# ifdef FONTSET_ALWAYS
7786 gui.menu_fontset = HL_TABLE()[idx].sg_fontset;
7787# else
7788 /* YIKES! This is a bug waiting to crash the program */
7789 gui.menu_font = HL_TABLE()[idx].sg_fontset;
7790# endif
7791 gui_mch_new_menu_font();
7792 }
7793# ifdef FEAT_BEVAL
7794 if (do_tooltip)
7795 {
7796 /* The Athena widget set cannot currently handle switching between
7797 * displaying a single font and a fontset.
7798 * If the XtNinternational resource is set to True at widget
Bram Moolenaarb8017e72007-05-10 18:59:07 +00007799 * creation, then a fontset is always used, otherwise an
Bram Moolenaar071d4272004-06-13 20:20:40 +00007800 * XFontStruct is used.
7801 */
7802 gui.tooltip_fontset = (XFontSet)HL_TABLE()[idx].sg_fontset;
7803 gui_mch_new_tooltip_font();
7804 }
7805# endif
7806# endif
7807 }
7808 else
7809# endif
7810 {
7811 HL_TABLE()[idx].sg_font = font_name2handle(arg);
7812 /* If it worked and it's the Normal group, use it as the
7813 * normal font. Same for the Menu group. */
7814 if (HL_TABLE()[idx].sg_font != NOFONT)
7815 {
7816 if (do_normal)
7817 gui_init_font(arg, FALSE);
7818#ifndef FONTSET_ALWAYS
7819# if (defined(FEAT_GUI_MOTIF) || defined(FEAT_GUI_ATHENA)) && defined(FEAT_MENU)
7820 if (do_menu)
7821 {
7822 gui.menu_font = HL_TABLE()[idx].sg_font;
7823 gui_mch_new_menu_font();
7824 }
7825# endif
7826#endif
7827 }
7828 }
7829}
7830
7831#endif /* FEAT_GUI */
7832
7833/*
7834 * Table with the specifications for an attribute number.
7835 * Note that this table is used by ALL buffers. This is required because the
7836 * GUI can redraw at any time for any buffer.
7837 */
Bram Moolenaar6c0b44b2005-06-01 21:56:33 +00007838static garray_T term_attr_table = {0, 0, 0, 0, NULL};
Bram Moolenaar071d4272004-06-13 20:20:40 +00007839
7840#define TERM_ATTR_ENTRY(idx) ((attrentry_T *)term_attr_table.ga_data)[idx]
7841
Bram Moolenaar6c0b44b2005-06-01 21:56:33 +00007842static garray_T cterm_attr_table = {0, 0, 0, 0, NULL};
Bram Moolenaar071d4272004-06-13 20:20:40 +00007843
7844#define CTERM_ATTR_ENTRY(idx) ((attrentry_T *)cterm_attr_table.ga_data)[idx]
7845
7846#ifdef FEAT_GUI
Bram Moolenaar6c0b44b2005-06-01 21:56:33 +00007847static garray_T gui_attr_table = {0, 0, 0, 0, NULL};
Bram Moolenaar071d4272004-06-13 20:20:40 +00007848
7849#define GUI_ATTR_ENTRY(idx) ((attrentry_T *)gui_attr_table.ga_data)[idx]
7850#endif
7851
7852/*
7853 * Return the attr number for a set of colors and font.
7854 * Add a new entry to the term_attr_table, cterm_attr_table or gui_attr_table
7855 * if the combination is new.
7856 * Return 0 for error (no more room).
7857 */
7858 static int
7859get_attr_entry(table, aep)
7860 garray_T *table;
7861 attrentry_T *aep;
7862{
7863 int i;
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00007864 attrentry_T *taep;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007865 static int recursive = FALSE;
7866
7867 /*
7868 * Init the table, in case it wasn't done yet.
7869 */
7870 table->ga_itemsize = sizeof(attrentry_T);
7871 table->ga_growsize = 7;
7872
7873 /*
7874 * Try to find an entry with the same specifications.
7875 */
7876 for (i = 0; i < table->ga_len; ++i)
7877 {
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00007878 taep = &(((attrentry_T *)table->ga_data)[i]);
7879 if ( aep->ae_attr == taep->ae_attr
Bram Moolenaar071d4272004-06-13 20:20:40 +00007880 && (
7881#ifdef FEAT_GUI
7882 (table == &gui_attr_table
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00007883 && (aep->ae_u.gui.fg_color == taep->ae_u.gui.fg_color
7884 && aep->ae_u.gui.bg_color
7885 == taep->ae_u.gui.bg_color
7886 && aep->ae_u.gui.sp_color
7887 == taep->ae_u.gui.sp_color
7888 && aep->ae_u.gui.font == taep->ae_u.gui.font
Bram Moolenaar071d4272004-06-13 20:20:40 +00007889# ifdef FEAT_XFONTSET
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00007890 && aep->ae_u.gui.fontset == taep->ae_u.gui.fontset
Bram Moolenaar071d4272004-06-13 20:20:40 +00007891# endif
7892 ))
7893 ||
7894#endif
7895 (table == &term_attr_table
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00007896 && (aep->ae_u.term.start == NULL)
7897 == (taep->ae_u.term.start == NULL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00007898 && (aep->ae_u.term.start == NULL
7899 || STRCMP(aep->ae_u.term.start,
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00007900 taep->ae_u.term.start) == 0)
7901 && (aep->ae_u.term.stop == NULL)
7902 == (taep->ae_u.term.stop == NULL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00007903 && (aep->ae_u.term.stop == NULL
7904 || STRCMP(aep->ae_u.term.stop,
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00007905 taep->ae_u.term.stop) == 0))
Bram Moolenaar071d4272004-06-13 20:20:40 +00007906 || (table == &cterm_attr_table
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00007907 && aep->ae_u.cterm.fg_color
7908 == taep->ae_u.cterm.fg_color
7909 && aep->ae_u.cterm.bg_color
7910 == taep->ae_u.cterm.bg_color)
Bram Moolenaar071d4272004-06-13 20:20:40 +00007911 ))
7912
7913 return i + ATTR_OFF;
7914 }
7915
Bram Moolenaar34cdc3e2005-05-18 22:24:46 +00007916 if (table->ga_len + ATTR_OFF > MAX_TYPENR)
Bram Moolenaar071d4272004-06-13 20:20:40 +00007917 {
7918 /*
7919 * Running out of attribute entries! remove all attributes, and
7920 * compute new ones for all groups.
7921 * When called recursively, we are really out of numbers.
7922 */
7923 if (recursive)
7924 {
7925 EMSG(_("E424: Too many different highlighting attributes in use"));
7926 return 0;
7927 }
7928 recursive = TRUE;
7929
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00007930 clear_hl_tables();
7931
Bram Moolenaar071d4272004-06-13 20:20:40 +00007932 must_redraw = CLEAR;
7933
7934 for (i = 0; i < highlight_ga.ga_len; ++i)
7935 set_hl_attr(i);
7936
7937 recursive = FALSE;
7938 }
7939
7940 /*
7941 * This is a new combination of colors and font, add an entry.
7942 */
7943 if (ga_grow(table, 1) == FAIL)
7944 return 0;
7945
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00007946 taep = &(((attrentry_T *)table->ga_data)[table->ga_len]);
7947 vim_memset(taep, 0, sizeof(attrentry_T));
7948 taep->ae_attr = aep->ae_attr;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007949#ifdef FEAT_GUI
7950 if (table == &gui_attr_table)
7951 {
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00007952 taep->ae_u.gui.fg_color = aep->ae_u.gui.fg_color;
7953 taep->ae_u.gui.bg_color = aep->ae_u.gui.bg_color;
7954 taep->ae_u.gui.sp_color = aep->ae_u.gui.sp_color;
7955 taep->ae_u.gui.font = aep->ae_u.gui.font;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007956# ifdef FEAT_XFONTSET
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00007957 taep->ae_u.gui.fontset = aep->ae_u.gui.fontset;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007958# endif
7959 }
7960#endif
7961 if (table == &term_attr_table)
7962 {
7963 if (aep->ae_u.term.start == NULL)
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00007964 taep->ae_u.term.start = NULL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007965 else
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00007966 taep->ae_u.term.start = vim_strsave(aep->ae_u.term.start);
Bram Moolenaar071d4272004-06-13 20:20:40 +00007967 if (aep->ae_u.term.stop == NULL)
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00007968 taep->ae_u.term.stop = NULL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007969 else
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00007970 taep->ae_u.term.stop = vim_strsave(aep->ae_u.term.stop);
Bram Moolenaar071d4272004-06-13 20:20:40 +00007971 }
7972 else if (table == &cterm_attr_table)
7973 {
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00007974 taep->ae_u.cterm.fg_color = aep->ae_u.cterm.fg_color;
7975 taep->ae_u.cterm.bg_color = aep->ae_u.cterm.bg_color;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007976 }
7977 ++table->ga_len;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007978 return (table->ga_len - 1 + ATTR_OFF);
7979}
7980
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00007981/*
7982 * Clear all highlight tables.
7983 */
7984 void
7985clear_hl_tables()
7986{
7987 int i;
7988 attrentry_T *taep;
7989
7990#ifdef FEAT_GUI
7991 ga_clear(&gui_attr_table);
7992#endif
7993 for (i = 0; i < term_attr_table.ga_len; ++i)
7994 {
7995 taep = &(((attrentry_T *)term_attr_table.ga_data)[i]);
7996 vim_free(taep->ae_u.term.start);
7997 vim_free(taep->ae_u.term.stop);
7998 }
7999 ga_clear(&term_attr_table);
8000 ga_clear(&cterm_attr_table);
8001}
8002
Bram Moolenaar1c8f93f2006-03-12 22:10:07 +00008003#if defined(FEAT_SYN_HL) || defined(FEAT_SPELL) || defined(PROTO)
Bram Moolenaar217ad922005-03-20 22:37:15 +00008004/*
Bram Moolenaar30abd282005-06-22 22:35:10 +00008005 * Combine special attributes (e.g., for spelling) with other attributes
8006 * (e.g., for syntax highlighting).
8007 * "prim_attr" overrules "char_attr".
Bram Moolenaar217ad922005-03-20 22:37:15 +00008008 * This creates a new group when required.
8009 * Since we expect there to be few spelling mistakes we don't cache the
8010 * result.
8011 * Return the resulting attributes.
8012 */
8013 int
Bram Moolenaar30abd282005-06-22 22:35:10 +00008014hl_combine_attr(char_attr, prim_attr)
Bram Moolenaar217ad922005-03-20 22:37:15 +00008015 int char_attr;
Bram Moolenaar30abd282005-06-22 22:35:10 +00008016 int prim_attr;
Bram Moolenaar217ad922005-03-20 22:37:15 +00008017{
8018 attrentry_T *char_aep = NULL;
8019 attrentry_T *spell_aep;
8020 attrentry_T new_en;
8021
8022 if (char_attr == 0)
Bram Moolenaar30abd282005-06-22 22:35:10 +00008023 return prim_attr;
8024 if (char_attr <= HL_ALL && prim_attr <= HL_ALL)
8025 return char_attr | prim_attr;
Bram Moolenaar217ad922005-03-20 22:37:15 +00008026#ifdef FEAT_GUI
8027 if (gui.in_use)
8028 {
8029 if (char_attr > HL_ALL)
8030 char_aep = syn_gui_attr2entry(char_attr);
8031 if (char_aep != NULL)
8032 new_en = *char_aep;
8033 else
8034 {
8035 vim_memset(&new_en, 0, sizeof(new_en));
Bram Moolenaard5cdbeb2005-10-10 20:59:28 +00008036 new_en.ae_u.gui.fg_color = INVALCOLOR;
8037 new_en.ae_u.gui.bg_color = INVALCOLOR;
8038 new_en.ae_u.gui.sp_color = INVALCOLOR;
Bram Moolenaar217ad922005-03-20 22:37:15 +00008039 if (char_attr <= HL_ALL)
8040 new_en.ae_attr = char_attr;
8041 }
8042
Bram Moolenaar30abd282005-06-22 22:35:10 +00008043 if (prim_attr <= HL_ALL)
8044 new_en.ae_attr |= prim_attr;
Bram Moolenaar217ad922005-03-20 22:37:15 +00008045 else
8046 {
Bram Moolenaar30abd282005-06-22 22:35:10 +00008047 spell_aep = syn_gui_attr2entry(prim_attr);
Bram Moolenaar217ad922005-03-20 22:37:15 +00008048 if (spell_aep != NULL)
8049 {
8050 new_en.ae_attr |= spell_aep->ae_attr;
8051 if (spell_aep->ae_u.gui.fg_color != INVALCOLOR)
8052 new_en.ae_u.gui.fg_color = spell_aep->ae_u.gui.fg_color;
8053 if (spell_aep->ae_u.gui.bg_color != INVALCOLOR)
8054 new_en.ae_u.gui.bg_color = spell_aep->ae_u.gui.bg_color;
8055 if (spell_aep->ae_u.gui.sp_color != INVALCOLOR)
8056 new_en.ae_u.gui.sp_color = spell_aep->ae_u.gui.sp_color;
8057 if (spell_aep->ae_u.gui.font != NOFONT)
8058 new_en.ae_u.gui.font = spell_aep->ae_u.gui.font;
8059# ifdef FEAT_XFONTSET
8060 if (spell_aep->ae_u.gui.fontset != NOFONTSET)
8061 new_en.ae_u.gui.fontset = spell_aep->ae_u.gui.fontset;
8062# endif
8063 }
8064 }
8065 return get_attr_entry(&gui_attr_table, &new_en);
8066 }
8067#endif
8068
8069 if (t_colors > 1)
8070 {
8071 if (char_attr > HL_ALL)
8072 char_aep = syn_cterm_attr2entry(char_attr);
8073 if (char_aep != NULL)
8074 new_en = *char_aep;
8075 else
8076 {
8077 vim_memset(&new_en, 0, sizeof(new_en));
8078 if (char_attr <= HL_ALL)
8079 new_en.ae_attr = char_attr;
8080 }
8081
Bram Moolenaar30abd282005-06-22 22:35:10 +00008082 if (prim_attr <= HL_ALL)
8083 new_en.ae_attr |= prim_attr;
Bram Moolenaar217ad922005-03-20 22:37:15 +00008084 else
8085 {
Bram Moolenaar30abd282005-06-22 22:35:10 +00008086 spell_aep = syn_cterm_attr2entry(prim_attr);
Bram Moolenaar217ad922005-03-20 22:37:15 +00008087 if (spell_aep != NULL)
8088 {
8089 new_en.ae_attr |= spell_aep->ae_attr;
8090 if (spell_aep->ae_u.cterm.fg_color > 0)
8091 new_en.ae_u.cterm.fg_color = spell_aep->ae_u.cterm.fg_color;
8092 if (spell_aep->ae_u.cterm.bg_color > 0)
8093 new_en.ae_u.cterm.bg_color = spell_aep->ae_u.cterm.bg_color;
8094 }
8095 }
8096 return get_attr_entry(&cterm_attr_table, &new_en);
8097 }
8098
8099 if (char_attr > HL_ALL)
8100 char_aep = syn_term_attr2entry(char_attr);
8101 if (char_aep != NULL)
8102 new_en = *char_aep;
8103 else
8104 {
8105 vim_memset(&new_en, 0, sizeof(new_en));
8106 if (char_attr <= HL_ALL)
8107 new_en.ae_attr = char_attr;
8108 }
8109
Bram Moolenaar30abd282005-06-22 22:35:10 +00008110 if (prim_attr <= HL_ALL)
8111 new_en.ae_attr |= prim_attr;
Bram Moolenaar217ad922005-03-20 22:37:15 +00008112 else
8113 {
Bram Moolenaar3b181812005-12-17 22:10:02 +00008114 spell_aep = syn_term_attr2entry(prim_attr);
Bram Moolenaar217ad922005-03-20 22:37:15 +00008115 if (spell_aep != NULL)
8116 {
8117 new_en.ae_attr |= spell_aep->ae_attr;
8118 if (spell_aep->ae_u.term.start != NULL)
8119 {
8120 new_en.ae_u.term.start = spell_aep->ae_u.term.start;
8121 new_en.ae_u.term.stop = spell_aep->ae_u.term.stop;
8122 }
8123 }
8124 }
8125 return get_attr_entry(&term_attr_table, &new_en);
8126}
8127#endif
8128
Bram Moolenaar071d4272004-06-13 20:20:40 +00008129#ifdef FEAT_GUI
8130
8131 attrentry_T *
8132syn_gui_attr2entry(attr)
8133 int attr;
8134{
8135 attr -= ATTR_OFF;
8136 if (attr >= gui_attr_table.ga_len) /* did ":syntax clear" */
8137 return NULL;
8138 return &(GUI_ATTR_ENTRY(attr));
8139}
Bram Moolenaar071d4272004-06-13 20:20:40 +00008140#endif /* FEAT_GUI */
8141
Bram Moolenaar1c8f93f2006-03-12 22:10:07 +00008142/*
8143 * Get the highlight attributes (HL_BOLD etc.) from an attribute nr.
8144 * Only to be used when "attr" > HL_ALL.
8145 */
8146 int
8147syn_attr2attr(attr)
8148 int attr;
8149{
8150 attrentry_T *aep;
8151
8152#ifdef FEAT_GUI
8153 if (gui.in_use)
8154 aep = syn_gui_attr2entry(attr);
8155 else
8156#endif
8157 if (t_colors > 1)
8158 aep = syn_cterm_attr2entry(attr);
8159 else
8160 aep = syn_term_attr2entry(attr);
8161
8162 if (aep == NULL) /* highlighting not set */
8163 return 0;
8164 return aep->ae_attr;
8165}
8166
8167
Bram Moolenaar071d4272004-06-13 20:20:40 +00008168 attrentry_T *
8169syn_term_attr2entry(attr)
8170 int attr;
8171{
8172 attr -= ATTR_OFF;
8173 if (attr >= term_attr_table.ga_len) /* did ":syntax clear" */
8174 return NULL;
8175 return &(TERM_ATTR_ENTRY(attr));
8176}
8177
8178 attrentry_T *
8179syn_cterm_attr2entry(attr)
8180 int attr;
8181{
8182 attr -= ATTR_OFF;
8183 if (attr >= cterm_attr_table.ga_len) /* did ":syntax clear" */
8184 return NULL;
8185 return &(CTERM_ATTR_ENTRY(attr));
8186}
8187
8188#define LIST_ATTR 1
8189#define LIST_STRING 2
8190#define LIST_INT 3
8191
8192 static void
8193highlight_list_one(id)
8194 int id;
8195{
8196 struct hl_group *sgp;
8197 int didh = FALSE;
8198
8199 sgp = &HL_TABLE()[id - 1]; /* index is ID minus one */
8200
8201 didh = highlight_list_arg(id, didh, LIST_ATTR,
8202 sgp->sg_term, NULL, "term");
8203 didh = highlight_list_arg(id, didh, LIST_STRING,
8204 0, sgp->sg_start, "start");
8205 didh = highlight_list_arg(id, didh, LIST_STRING,
8206 0, sgp->sg_stop, "stop");
8207
8208 didh = highlight_list_arg(id, didh, LIST_ATTR,
8209 sgp->sg_cterm, NULL, "cterm");
8210 didh = highlight_list_arg(id, didh, LIST_INT,
8211 sgp->sg_cterm_fg, NULL, "ctermfg");
8212 didh = highlight_list_arg(id, didh, LIST_INT,
8213 sgp->sg_cterm_bg, NULL, "ctermbg");
8214
8215#ifdef FEAT_GUI
8216 didh = highlight_list_arg(id, didh, LIST_ATTR,
8217 sgp->sg_gui, NULL, "gui");
8218 didh = highlight_list_arg(id, didh, LIST_STRING,
8219 0, sgp->sg_gui_fg_name, "guifg");
8220 didh = highlight_list_arg(id, didh, LIST_STRING,
8221 0, sgp->sg_gui_bg_name, "guibg");
8222 didh = highlight_list_arg(id, didh, LIST_STRING,
Bram Moolenaar75c50c42005-06-04 22:06:24 +00008223 0, sgp->sg_gui_sp_name, "guisp");
8224 didh = highlight_list_arg(id, didh, LIST_STRING,
Bram Moolenaar071d4272004-06-13 20:20:40 +00008225 0, sgp->sg_font_name, "font");
8226#endif
8227
Bram Moolenaar661b1822005-07-28 22:36:45 +00008228 if (sgp->sg_link && !got_int)
Bram Moolenaar071d4272004-06-13 20:20:40 +00008229 {
8230 (void)syn_list_header(didh, 9999, id);
Bram Moolenaar661b1822005-07-28 22:36:45 +00008231 didh = TRUE;
Bram Moolenaar071d4272004-06-13 20:20:40 +00008232 msg_puts_attr((char_u *)"links to", hl_attr(HLF_D));
8233 msg_putchar(' ');
8234 msg_outtrans(HL_TABLE()[HL_TABLE()[id - 1].sg_link - 1].sg_name);
8235 }
Bram Moolenaar661b1822005-07-28 22:36:45 +00008236
Bram Moolenaar5e3cb7e2006-02-27 23:58:35 +00008237 if (!didh)
8238 highlight_list_arg(id, didh, LIST_STRING, 0, (char_u *)"cleared", "");
Bram Moolenaar661b1822005-07-28 22:36:45 +00008239#ifdef FEAT_EVAL
Bram Moolenaar5e3cb7e2006-02-27 23:58:35 +00008240 if (p_verbose > 0)
Bram Moolenaar661b1822005-07-28 22:36:45 +00008241 last_set_msg(sgp->sg_scriptID);
8242#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00008243}
8244
8245 static int
8246highlight_list_arg(id, didh, type, iarg, sarg, name)
8247 int id;
8248 int didh;
8249 int type;
8250 int iarg;
8251 char_u *sarg;
8252 char *name;
8253{
8254 char_u buf[100];
8255 char_u *ts;
8256 int i;
8257
Bram Moolenaar661b1822005-07-28 22:36:45 +00008258 if (got_int)
8259 return FALSE;
Bram Moolenaar071d4272004-06-13 20:20:40 +00008260 if (type == LIST_STRING ? (sarg != NULL) : (iarg != 0))
8261 {
8262 ts = buf;
8263 if (type == LIST_INT)
8264 sprintf((char *)buf, "%d", iarg - 1);
8265 else if (type == LIST_STRING)
8266 ts = sarg;
8267 else /* type == LIST_ATTR */
8268 {
8269 buf[0] = NUL;
8270 for (i = 0; hl_attr_table[i] != 0; ++i)
8271 {
8272 if (iarg & hl_attr_table[i])
8273 {
8274 if (buf[0] != NUL)
8275 STRCAT(buf, ",");
8276 STRCAT(buf, hl_name_table[i]);
8277 iarg &= ~hl_attr_table[i]; /* don't want "inverse" */
8278 }
8279 }
8280 }
8281
8282 (void)syn_list_header(didh,
8283 (int)(vim_strsize(ts) + STRLEN(name) + 1), id);
8284 didh = TRUE;
Bram Moolenaar661b1822005-07-28 22:36:45 +00008285 if (!got_int)
8286 {
Bram Moolenaar5e3cb7e2006-02-27 23:58:35 +00008287 if (*name != NUL)
8288 {
8289 MSG_PUTS_ATTR(name, hl_attr(HLF_D));
8290 MSG_PUTS_ATTR("=", hl_attr(HLF_D));
8291 }
Bram Moolenaar661b1822005-07-28 22:36:45 +00008292 msg_outtrans(ts);
8293 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00008294 }
8295 return didh;
8296}
8297
8298#if (((defined(FEAT_EVAL) || defined(FEAT_PRINTER))) && defined(FEAT_SYN_HL)) || defined(PROTO)
8299/*
8300 * Return "1" if highlight group "id" has attribute "flag".
8301 * Return NULL otherwise.
8302 */
8303 char_u *
8304highlight_has_attr(id, flag, modec)
8305 int id;
8306 int flag;
8307 int modec; /* 'g' for GUI, 'c' for cterm, 't' for term */
8308{
8309 int attr;
8310
8311 if (id <= 0 || id > highlight_ga.ga_len)
8312 return NULL;
8313
8314#ifdef FEAT_GUI
8315 if (modec == 'g')
8316 attr = HL_TABLE()[id - 1].sg_gui;
8317 else
8318#endif
8319 if (modec == 'c')
8320 attr = HL_TABLE()[id - 1].sg_cterm;
8321 else
8322 attr = HL_TABLE()[id - 1].sg_term;
8323
8324 if (attr & flag)
8325 return (char_u *)"1";
8326 return NULL;
8327}
8328#endif
8329
8330#if (defined(FEAT_SYN_HL) && defined(FEAT_EVAL)) || defined(PROTO)
8331/*
8332 * Return color name of highlight group "id".
8333 */
8334 char_u *
8335highlight_color(id, what, modec)
8336 int id;
Bram Moolenaare2cc9702005-03-15 22:43:58 +00008337 char_u *what; /* "fg", "bg", "sp", "fg#", "bg#" or "sp#" */
Bram Moolenaar071d4272004-06-13 20:20:40 +00008338 int modec; /* 'g' for GUI, 'c' for cterm, 't' for term */
8339{
8340 static char_u name[20];
8341 int n;
Bram Moolenaare2cc9702005-03-15 22:43:58 +00008342 int fg = FALSE;
8343# ifdef FEAT_GUI
8344 int sp = FALSE;
8345# endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00008346
8347 if (id <= 0 || id > highlight_ga.ga_len)
8348 return NULL;
8349
8350 if (TOLOWER_ASC(what[0]) == 'f')
8351 fg = TRUE;
Bram Moolenaare2cc9702005-03-15 22:43:58 +00008352# ifdef FEAT_GUI
8353 else if (TOLOWER_ASC(what[0]) == 's')
8354 sp = TRUE;
Bram Moolenaar071d4272004-06-13 20:20:40 +00008355 if (modec == 'g')
8356 {
8357 /* return #RRGGBB form (only possible when GUI is running) */
8358 if (gui.in_use && what[1] && what[2] == '#')
8359 {
8360 guicolor_T color;
8361 long_u rgb;
8362 static char_u buf[10];
8363
8364 if (fg)
8365 color = HL_TABLE()[id - 1].sg_gui_fg;
Bram Moolenaare2cc9702005-03-15 22:43:58 +00008366 else if (sp)
8367 color = HL_TABLE()[id - 1].sg_gui_sp;
Bram Moolenaar071d4272004-06-13 20:20:40 +00008368 else
8369 color = HL_TABLE()[id - 1].sg_gui_bg;
8370 if (color == INVALCOLOR)
8371 return NULL;
8372 rgb = gui_mch_get_rgb(color);
8373 sprintf((char *)buf, "#%02x%02x%02x",
8374 (unsigned)(rgb >> 16),
8375 (unsigned)(rgb >> 8) & 255,
8376 (unsigned)rgb & 255);
8377 return buf;
8378 }
8379 if (fg)
8380 return (HL_TABLE()[id - 1].sg_gui_fg_name);
Bram Moolenaare2cc9702005-03-15 22:43:58 +00008381 if (sp)
8382 return (HL_TABLE()[id - 1].sg_gui_sp_name);
Bram Moolenaar071d4272004-06-13 20:20:40 +00008383 return (HL_TABLE()[id - 1].sg_gui_bg_name);
8384 }
Bram Moolenaare2cc9702005-03-15 22:43:58 +00008385# endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00008386 if (modec == 'c')
8387 {
8388 if (fg)
8389 n = HL_TABLE()[id - 1].sg_cterm_fg - 1;
8390 else
8391 n = HL_TABLE()[id - 1].sg_cterm_bg - 1;
8392 sprintf((char *)name, "%d", n);
8393 return name;
8394 }
8395 /* term doesn't have color */
8396 return NULL;
8397}
8398#endif
8399
8400#if (defined(FEAT_SYN_HL) && defined(FEAT_GUI) && defined(FEAT_PRINTER)) \
8401 || defined(PROTO)
8402/*
8403 * Return color name of highlight group "id" as RGB value.
8404 */
8405 long_u
8406highlight_gui_color_rgb(id, fg)
8407 int id;
8408 int fg; /* TRUE = fg, FALSE = bg */
8409{
8410 guicolor_T color;
8411
8412 if (id <= 0 || id > highlight_ga.ga_len)
8413 return 0L;
8414
8415 if (fg)
8416 color = HL_TABLE()[id - 1].sg_gui_fg;
8417 else
8418 color = HL_TABLE()[id - 1].sg_gui_bg;
8419
8420 if (color == INVALCOLOR)
8421 return 0L;
8422
8423 return gui_mch_get_rgb(color);
8424}
8425#endif
8426
8427/*
8428 * Output the syntax list header.
8429 * Return TRUE when started a new line.
8430 */
8431 static int
8432syn_list_header(did_header, outlen, id)
8433 int did_header; /* did header already */
8434 int outlen; /* length of string that comes */
8435 int id; /* highlight group id */
8436{
8437 int endcol = 19;
8438 int newline = TRUE;
8439
8440 if (!did_header)
8441 {
8442 msg_putchar('\n');
Bram Moolenaar661b1822005-07-28 22:36:45 +00008443 if (got_int)
8444 return TRUE;
Bram Moolenaar071d4272004-06-13 20:20:40 +00008445 msg_outtrans(HL_TABLE()[id - 1].sg_name);
8446 endcol = 15;
8447 }
8448 else if (msg_col + outlen + 1 >= Columns)
Bram Moolenaar661b1822005-07-28 22:36:45 +00008449 {
Bram Moolenaar071d4272004-06-13 20:20:40 +00008450 msg_putchar('\n');
Bram Moolenaar661b1822005-07-28 22:36:45 +00008451 if (got_int)
8452 return TRUE;
8453 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00008454 else
8455 {
8456 if (msg_col >= endcol) /* wrap around is like starting a new line */
8457 newline = FALSE;
8458 }
8459
8460 if (msg_col >= endcol) /* output at least one space */
8461 endcol = msg_col + 1;
8462 if (Columns <= endcol) /* avoid hang for tiny window */
8463 endcol = Columns - 1;
8464
8465 msg_advance(endcol);
8466
8467 /* Show "xxx" with the attributes. */
8468 if (!did_header)
8469 {
8470 msg_puts_attr((char_u *)"xxx", syn_id2attr(id));
8471 msg_putchar(' ');
8472 }
8473
8474 return newline;
8475}
8476
8477/*
8478 * Set the attribute numbers for a highlight group.
8479 * Called after one of the attributes has changed.
8480 */
8481 static void
8482set_hl_attr(idx)
8483 int idx; /* index in array */
8484{
8485 attrentry_T at_en;
8486 struct hl_group *sgp = HL_TABLE() + idx;
8487
8488 /* The "Normal" group doesn't need an attribute number */
8489 if (sgp->sg_name_u != NULL && STRCMP(sgp->sg_name_u, "NORMAL") == 0)
8490 return;
8491
8492#ifdef FEAT_GUI
8493 /*
8494 * For the GUI mode: If there are other than "normal" highlighting
8495 * attributes, need to allocate an attr number.
8496 */
8497 if (sgp->sg_gui_fg == INVALCOLOR
8498 && sgp->sg_gui_bg == INVALCOLOR
Bram Moolenaare2cc9702005-03-15 22:43:58 +00008499 && sgp->sg_gui_sp == INVALCOLOR
Bram Moolenaar071d4272004-06-13 20:20:40 +00008500 && sgp->sg_font == NOFONT
8501# ifdef FEAT_XFONTSET
8502 && sgp->sg_fontset == NOFONTSET
8503# endif
8504 )
8505 {
8506 sgp->sg_gui_attr = sgp->sg_gui;
8507 }
8508 else
8509 {
8510 at_en.ae_attr = sgp->sg_gui;
8511 at_en.ae_u.gui.fg_color = sgp->sg_gui_fg;
8512 at_en.ae_u.gui.bg_color = sgp->sg_gui_bg;
Bram Moolenaare2cc9702005-03-15 22:43:58 +00008513 at_en.ae_u.gui.sp_color = sgp->sg_gui_sp;
Bram Moolenaar071d4272004-06-13 20:20:40 +00008514 at_en.ae_u.gui.font = sgp->sg_font;
8515# ifdef FEAT_XFONTSET
8516 at_en.ae_u.gui.fontset = sgp->sg_fontset;
8517# endif
8518 sgp->sg_gui_attr = get_attr_entry(&gui_attr_table, &at_en);
8519 }
8520#endif
8521 /*
8522 * For the term mode: If there are other than "normal" highlighting
8523 * attributes, need to allocate an attr number.
8524 */
8525 if (sgp->sg_start == NULL && sgp->sg_stop == NULL)
8526 sgp->sg_term_attr = sgp->sg_term;
8527 else
8528 {
8529 at_en.ae_attr = sgp->sg_term;
8530 at_en.ae_u.term.start = sgp->sg_start;
8531 at_en.ae_u.term.stop = sgp->sg_stop;
8532 sgp->sg_term_attr = get_attr_entry(&term_attr_table, &at_en);
8533 }
8534
8535 /*
8536 * For the color term mode: If there are other than "normal"
8537 * highlighting attributes, need to allocate an attr number.
8538 */
8539 if (sgp->sg_cterm_fg == 0 && sgp->sg_cterm_bg == 0)
8540 sgp->sg_cterm_attr = sgp->sg_cterm;
8541 else
8542 {
8543 at_en.ae_attr = sgp->sg_cterm;
8544 at_en.ae_u.cterm.fg_color = sgp->sg_cterm_fg;
8545 at_en.ae_u.cterm.bg_color = sgp->sg_cterm_bg;
8546 sgp->sg_cterm_attr = get_attr_entry(&cterm_attr_table, &at_en);
8547 }
8548}
8549
8550/*
8551 * Lookup a highlight group name and return it's ID.
8552 * If it is not found, 0 is returned.
8553 */
8554 int
8555syn_name2id(name)
8556 char_u *name;
8557{
8558 int i;
8559 char_u name_u[200];
8560
8561 /* Avoid using stricmp() too much, it's slow on some systems */
8562 /* Avoid alloc()/free(), these are slow too. ID names over 200 chars
8563 * don't deserve to be found! */
Bram Moolenaarce0842a2005-07-18 21:58:11 +00008564 vim_strncpy(name_u, name, 199);
Bram Moolenaar071d4272004-06-13 20:20:40 +00008565 vim_strup(name_u);
8566 for (i = highlight_ga.ga_len; --i >= 0; )
8567 if (HL_TABLE()[i].sg_name_u != NULL
8568 && STRCMP(name_u, HL_TABLE()[i].sg_name_u) == 0)
8569 break;
8570 return i + 1;
8571}
8572
8573#if defined(FEAT_EVAL) || defined(PROTO)
8574/*
8575 * Return TRUE if highlight group "name" exists.
8576 */
8577 int
8578highlight_exists(name)
8579 char_u *name;
8580{
8581 return (syn_name2id(name) > 0);
8582}
Bram Moolenaar910f66f2006-04-05 20:41:53 +00008583
8584# if defined(FEAT_SEARCH_EXTRA) || defined(PROTO)
8585/*
8586 * Return the name of highlight group "id".
8587 * When not a valid ID return an empty string.
8588 */
8589 char_u *
8590syn_id2name(id)
8591 int id;
8592{
Bram Moolenaar6ee10162007-07-26 20:58:42 +00008593 if (id <= 0 || id > highlight_ga.ga_len)
Bram Moolenaar910f66f2006-04-05 20:41:53 +00008594 return (char_u *)"";
8595 return HL_TABLE()[id - 1].sg_name;
8596}
8597# endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00008598#endif
8599
8600/*
8601 * Like syn_name2id(), but take a pointer + length argument.
8602 */
8603 int
8604syn_namen2id(linep, len)
8605 char_u *linep;
8606 int len;
8607{
8608 char_u *name;
8609 int id = 0;
8610
8611 name = vim_strnsave(linep, len);
8612 if (name != NULL)
8613 {
8614 id = syn_name2id(name);
8615 vim_free(name);
8616 }
8617 return id;
8618}
8619
8620/*
8621 * Find highlight group name in the table and return it's ID.
8622 * The argument is a pointer to the name and the length of the name.
8623 * If it doesn't exist yet, a new entry is created.
8624 * Return 0 for failure.
8625 */
8626 int
8627syn_check_group(pp, len)
8628 char_u *pp;
8629 int len;
8630{
8631 int id;
8632 char_u *name;
8633
8634 name = vim_strnsave(pp, len);
8635 if (name == NULL)
8636 return 0;
8637
8638 id = syn_name2id(name);
8639 if (id == 0) /* doesn't exist yet */
8640 id = syn_add_group(name);
8641 else
8642 vim_free(name);
8643 return id;
8644}
8645
8646/*
8647 * Add new highlight group and return it's ID.
8648 * "name" must be an allocated string, it will be consumed.
8649 * Return 0 for failure.
8650 */
8651 static int
8652syn_add_group(name)
8653 char_u *name;
8654{
8655 char_u *p;
8656
8657 /* Check that the name is ASCII letters, digits and underscore. */
8658 for (p = name; *p != NUL; ++p)
8659 {
8660 if (!vim_isprintc(*p))
8661 {
8662 EMSG(_("E669: Unprintable character in group name"));
8663 return 0;
8664 }
8665 else if (!ASCII_ISALNUM(*p) && *p != '_')
8666 {
8667 /* This is an error, but since there previously was no check only
8668 * give a warning. */
Bram Moolenaar2df6dcc2004-07-12 15:53:54 +00008669 msg_source(hl_attr(HLF_W));
Bram Moolenaar071d4272004-06-13 20:20:40 +00008670 MSG(_("W18: Invalid character in group name"));
8671 break;
8672 }
8673 }
8674
8675 /*
8676 * First call for this growarray: init growing array.
8677 */
8678 if (highlight_ga.ga_data == NULL)
8679 {
8680 highlight_ga.ga_itemsize = sizeof(struct hl_group);
8681 highlight_ga.ga_growsize = 10;
8682 }
8683
8684 /*
8685 * Make room for at least one other syntax_highlight entry.
8686 */
8687 if (ga_grow(&highlight_ga, 1) == FAIL)
8688 {
8689 vim_free(name);
8690 return 0;
8691 }
8692
8693 vim_memset(&(HL_TABLE()[highlight_ga.ga_len]), 0, sizeof(struct hl_group));
8694 HL_TABLE()[highlight_ga.ga_len].sg_name = name;
8695 HL_TABLE()[highlight_ga.ga_len].sg_name_u = vim_strsave_up(name);
8696#ifdef FEAT_GUI
8697 HL_TABLE()[highlight_ga.ga_len].sg_gui_bg = INVALCOLOR;
8698 HL_TABLE()[highlight_ga.ga_len].sg_gui_fg = INVALCOLOR;
Bram Moolenaare2cc9702005-03-15 22:43:58 +00008699 HL_TABLE()[highlight_ga.ga_len].sg_gui_sp = INVALCOLOR;
Bram Moolenaar071d4272004-06-13 20:20:40 +00008700#endif
8701 ++highlight_ga.ga_len;
Bram Moolenaar071d4272004-06-13 20:20:40 +00008702
8703 return highlight_ga.ga_len; /* ID is index plus one */
8704}
8705
8706/*
8707 * When, just after calling syn_add_group(), an error is discovered, this
8708 * function deletes the new name.
8709 */
8710 static void
8711syn_unadd_group()
8712{
8713 --highlight_ga.ga_len;
Bram Moolenaar071d4272004-06-13 20:20:40 +00008714 vim_free(HL_TABLE()[highlight_ga.ga_len].sg_name);
8715 vim_free(HL_TABLE()[highlight_ga.ga_len].sg_name_u);
8716}
8717
8718/*
8719 * Translate a group ID to highlight attributes.
8720 */
8721 int
8722syn_id2attr(hl_id)
8723 int hl_id;
8724{
8725 int attr;
8726 struct hl_group *sgp;
8727
8728 hl_id = syn_get_final_id(hl_id);
8729 sgp = &HL_TABLE()[hl_id - 1]; /* index is ID minus one */
8730
8731#ifdef FEAT_GUI
8732 /*
8733 * Only use GUI attr when the GUI is being used.
8734 */
8735 if (gui.in_use)
8736 attr = sgp->sg_gui_attr;
8737 else
8738#endif
8739 if (t_colors > 1)
8740 attr = sgp->sg_cterm_attr;
8741 else
8742 attr = sgp->sg_term_attr;
8743
8744 return attr;
8745}
8746
8747#ifdef FEAT_GUI
8748/*
8749 * Get the GUI colors and attributes for a group ID.
8750 * NOTE: the colors will be INVALCOLOR when not set, the color otherwise.
8751 */
8752 int
8753syn_id2colors(hl_id, fgp, bgp)
8754 int hl_id;
8755 guicolor_T *fgp;
8756 guicolor_T *bgp;
8757{
8758 struct hl_group *sgp;
8759
8760 hl_id = syn_get_final_id(hl_id);
8761 sgp = &HL_TABLE()[hl_id - 1]; /* index is ID minus one */
8762
8763 *fgp = sgp->sg_gui_fg;
8764 *bgp = sgp->sg_gui_bg;
8765 return sgp->sg_gui;
8766}
8767#endif
8768
8769/*
8770 * Translate a group ID to the final group ID (following links).
8771 */
8772 int
8773syn_get_final_id(hl_id)
8774 int hl_id;
8775{
8776 int count;
8777 struct hl_group *sgp;
8778
8779 if (hl_id > highlight_ga.ga_len || hl_id < 1)
8780 return 0; /* Can be called from eval!! */
8781
8782 /*
8783 * Follow links until there is no more.
8784 * Look out for loops! Break after 100 links.
8785 */
8786 for (count = 100; --count >= 0; )
8787 {
8788 sgp = &HL_TABLE()[hl_id - 1]; /* index is ID minus one */
8789 if (sgp->sg_link == 0 || sgp->sg_link > highlight_ga.ga_len)
8790 break;
8791 hl_id = sgp->sg_link;
8792 }
8793
8794 return hl_id;
8795}
8796
8797#ifdef FEAT_GUI
8798/*
8799 * Call this function just after the GUI has started.
8800 * It finds the font and color handles for the highlighting groups.
8801 */
8802 void
8803highlight_gui_started()
8804{
8805 int idx;
8806
8807 /* First get the colors from the "Normal" and "Menu" group, if set */
8808 set_normal_colors();
8809
8810 for (idx = 0; idx < highlight_ga.ga_len; ++idx)
8811 gui_do_one_color(idx, FALSE, FALSE);
8812
8813 highlight_changed();
8814}
8815
8816 static void
8817gui_do_one_color(idx, do_menu, do_tooltip)
8818 int idx;
8819 int do_menu; /* TRUE: might set the menu font */
8820 int do_tooltip; /* TRUE: might set the tooltip font */
8821{
8822 int didit = FALSE;
8823
8824 if (HL_TABLE()[idx].sg_font_name != NULL)
8825 {
8826 hl_do_font(idx, HL_TABLE()[idx].sg_font_name, FALSE, do_menu,
8827 do_tooltip);
8828 didit = TRUE;
8829 }
8830 if (HL_TABLE()[idx].sg_gui_fg_name != NULL)
8831 {
8832 HL_TABLE()[idx].sg_gui_fg =
8833 color_name2handle(HL_TABLE()[idx].sg_gui_fg_name);
8834 didit = TRUE;
8835 }
8836 if (HL_TABLE()[idx].sg_gui_bg_name != NULL)
8837 {
8838 HL_TABLE()[idx].sg_gui_bg =
8839 color_name2handle(HL_TABLE()[idx].sg_gui_bg_name);
8840 didit = TRUE;
8841 }
Bram Moolenaare2cc9702005-03-15 22:43:58 +00008842 if (HL_TABLE()[idx].sg_gui_sp_name != NULL)
8843 {
8844 HL_TABLE()[idx].sg_gui_sp =
8845 color_name2handle(HL_TABLE()[idx].sg_gui_sp_name);
8846 didit = TRUE;
8847 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00008848 if (didit) /* need to get a new attr number */
8849 set_hl_attr(idx);
8850}
8851
8852#endif
8853
8854/*
8855 * Translate the 'highlight' option into attributes in highlight_attr[] and
8856 * set up the user highlights User1..9. If FEAT_STL_OPT is in use, a set of
8857 * corresponding highlights to use on top of HLF_SNC is computed.
8858 * Called only when the 'highlight' option has been changed and upon first
8859 * screen redraw after any :highlight command.
8860 * Return FAIL when an invalid flag is found in 'highlight'. OK otherwise.
8861 */
8862 int
8863highlight_changed()
8864{
8865 int hlf;
8866 int i;
8867 char_u *p;
8868 int attr;
8869 char_u *end;
8870 int id;
8871#ifdef USER_HIGHLIGHT
8872 char_u userhl[10];
8873# ifdef FEAT_STL_OPT
8874 int id_SNC = -1;
8875 int id_S = -1;
8876 int hlcnt;
8877# endif
8878#endif
8879 static int hl_flags[HLF_COUNT] = HL_FLAGS;
8880
8881 need_highlight_changed = FALSE;
8882
8883 /*
8884 * Clear all attributes.
8885 */
8886 for (hlf = 0; hlf < (int)HLF_COUNT; ++hlf)
8887 highlight_attr[hlf] = 0;
8888
8889 /*
8890 * First set all attributes to their default value.
8891 * Then use the attributes from the 'highlight' option.
8892 */
8893 for (i = 0; i < 2; ++i)
8894 {
8895 if (i)
8896 p = p_hl;
8897 else
8898 p = get_highlight_default();
8899 if (p == NULL) /* just in case */
8900 continue;
8901
8902 while (*p)
8903 {
8904 for (hlf = 0; hlf < (int)HLF_COUNT; ++hlf)
8905 if (hl_flags[hlf] == *p)
8906 break;
8907 ++p;
8908 if (hlf == (int)HLF_COUNT || *p == NUL)
8909 return FAIL;
8910
8911 /*
8912 * Allow several hl_flags to be combined, like "bu" for
8913 * bold-underlined.
8914 */
8915 attr = 0;
8916 for ( ; *p && *p != ','; ++p) /* parse upto comma */
8917 {
8918 if (vim_iswhite(*p)) /* ignore white space */
8919 continue;
8920
8921 if (attr > HL_ALL) /* Combination with ':' is not allowed. */
8922 return FAIL;
8923
8924 switch (*p)
8925 {
8926 case 'b': attr |= HL_BOLD;
8927 break;
8928 case 'i': attr |= HL_ITALIC;
8929 break;
8930 case '-':
8931 case 'n': /* no highlighting */
8932 break;
8933 case 'r': attr |= HL_INVERSE;
8934 break;
8935 case 's': attr |= HL_STANDOUT;
8936 break;
8937 case 'u': attr |= HL_UNDERLINE;
8938 break;
Bram Moolenaare2cc9702005-03-15 22:43:58 +00008939 case 'c': attr |= HL_UNDERCURL;
8940 break;
Bram Moolenaar071d4272004-06-13 20:20:40 +00008941 case ':': ++p; /* highlight group name */
8942 if (attr || *p == NUL) /* no combinations */
8943 return FAIL;
8944 end = vim_strchr(p, ',');
8945 if (end == NULL)
8946 end = p + STRLEN(p);
8947 id = syn_check_group(p, (int)(end - p));
8948 if (id == 0)
8949 return FAIL;
8950 attr = syn_id2attr(id);
8951 p = end - 1;
8952#if defined(FEAT_STL_OPT) && defined(USER_HIGHLIGHT)
8953 if (hlf == (int)HLF_SNC)
8954 id_SNC = syn_get_final_id(id);
8955 else if (hlf == (int)HLF_S)
8956 id_S = syn_get_final_id(id);
8957#endif
8958 break;
8959 default: return FAIL;
8960 }
8961 }
8962 highlight_attr[hlf] = attr;
8963
8964 p = skip_to_option_part(p); /* skip comma and spaces */
8965 }
8966 }
8967
8968#ifdef USER_HIGHLIGHT
8969 /* Setup the user highlights
8970 *
8971 * Temporarily utilize 10 more hl entries. Have to be in there
8972 * simultaneously in case of table overflows in get_attr_entry()
8973 */
8974# ifdef FEAT_STL_OPT
8975 if (ga_grow(&highlight_ga, 10) == FAIL)
8976 return FAIL;
8977 hlcnt = highlight_ga.ga_len;
8978 if (id_S == 0)
8979 { /* Make sure id_S is always valid to simplify code below */
8980 memset(&HL_TABLE()[hlcnt + 9], 0, sizeof(struct hl_group));
8981 HL_TABLE()[hlcnt + 9].sg_term = highlight_attr[HLF_S];
8982 id_S = hlcnt + 10;
8983 }
8984# endif
8985 for (i = 0; i < 9; i++)
8986 {
8987 sprintf((char *)userhl, "User%d", i + 1);
8988 id = syn_name2id(userhl);
8989 if (id == 0)
8990 {
8991 highlight_user[i] = 0;
8992# ifdef FEAT_STL_OPT
8993 highlight_stlnc[i] = 0;
8994# endif
8995 }
8996 else
8997 {
8998# ifdef FEAT_STL_OPT
8999 struct hl_group *hlt = HL_TABLE();
9000# endif
9001
9002 highlight_user[i] = syn_id2attr(id);
9003# ifdef FEAT_STL_OPT
9004 if (id_SNC == 0)
9005 {
9006 memset(&hlt[hlcnt + i], 0, sizeof(struct hl_group));
9007 hlt[hlcnt + i].sg_term = highlight_attr[HLF_SNC];
9008 hlt[hlcnt + i].sg_cterm = highlight_attr[HLF_SNC];
9009# ifdef FEAT_GUI
9010 hlt[hlcnt + i].sg_gui = highlight_attr[HLF_SNC];
9011# endif
9012 }
9013 else
9014 mch_memmove(&hlt[hlcnt + i],
9015 &hlt[id_SNC - 1],
9016 sizeof(struct hl_group));
9017 hlt[hlcnt + i].sg_link = 0;
9018
9019 /* Apply difference between UserX and HLF_S to HLF_SNC */
9020 hlt[hlcnt + i].sg_term ^=
9021 hlt[id - 1].sg_term ^ hlt[id_S - 1].sg_term;
9022 if (hlt[id - 1].sg_start != hlt[id_S - 1].sg_start)
9023 hlt[hlcnt + i].sg_start = hlt[id - 1].sg_start;
9024 if (hlt[id - 1].sg_stop != hlt[id_S - 1].sg_stop)
9025 hlt[hlcnt + i].sg_stop = hlt[id - 1].sg_stop;
9026 hlt[hlcnt + i].sg_cterm ^=
9027 hlt[id - 1].sg_cterm ^ hlt[id_S - 1].sg_cterm;
9028 if (hlt[id - 1].sg_cterm_fg != hlt[id_S - 1].sg_cterm_fg)
9029 hlt[hlcnt + i].sg_cterm_fg = hlt[id - 1].sg_cterm_fg;
9030 if (hlt[id - 1].sg_cterm_bg != hlt[id_S - 1].sg_cterm_bg)
9031 hlt[hlcnt + i].sg_cterm_bg = hlt[id - 1].sg_cterm_bg;
9032# ifdef FEAT_GUI
9033 hlt[hlcnt + i].sg_gui ^=
9034 hlt[id - 1].sg_gui ^ hlt[id_S - 1].sg_gui;
9035 if (hlt[id - 1].sg_gui_fg != hlt[id_S - 1].sg_gui_fg)
9036 hlt[hlcnt + i].sg_gui_fg = hlt[id - 1].sg_gui_fg;
9037 if (hlt[id - 1].sg_gui_bg != hlt[id_S - 1].sg_gui_bg)
9038 hlt[hlcnt + i].sg_gui_bg = hlt[id - 1].sg_gui_bg;
Bram Moolenaare2cc9702005-03-15 22:43:58 +00009039 if (hlt[id - 1].sg_gui_sp != hlt[id_S - 1].sg_gui_sp)
9040 hlt[hlcnt + i].sg_gui_sp = hlt[id - 1].sg_gui_sp;
Bram Moolenaar071d4272004-06-13 20:20:40 +00009041 if (hlt[id - 1].sg_font != hlt[id_S - 1].sg_font)
9042 hlt[hlcnt + i].sg_font = hlt[id - 1].sg_font;
9043# ifdef FEAT_XFONTSET
9044 if (hlt[id - 1].sg_fontset != hlt[id_S - 1].sg_fontset)
9045 hlt[hlcnt + i].sg_fontset = hlt[id - 1].sg_fontset;
9046# endif
9047# endif
9048 highlight_ga.ga_len = hlcnt + i + 1;
9049 set_hl_attr(hlcnt + i); /* At long last we can apply */
9050 highlight_stlnc[i] = syn_id2attr(hlcnt + i + 1);
9051# endif
9052 }
9053 }
9054# ifdef FEAT_STL_OPT
9055 highlight_ga.ga_len = hlcnt;
9056# endif
9057
9058#endif /* USER_HIGHLIGHT */
9059
9060 return OK;
9061}
9062
Bram Moolenaar4f688582007-07-24 12:34:30 +00009063#if defined(FEAT_CMDL_COMPL) || defined(PROTO)
Bram Moolenaar071d4272004-06-13 20:20:40 +00009064
9065static void highlight_list __ARGS((void));
9066static void highlight_list_two __ARGS((int cnt, int attr));
9067
9068/*
9069 * Handle command line completion for :highlight command.
9070 */
9071 void
9072set_context_in_highlight_cmd(xp, arg)
9073 expand_T *xp;
9074 char_u *arg;
9075{
9076 char_u *p;
9077
9078 /* Default: expand group names */
9079 xp->xp_context = EXPAND_HIGHLIGHT;
9080 xp->xp_pattern = arg;
Bram Moolenaar4f688582007-07-24 12:34:30 +00009081 include_link = 2;
9082 include_default = 1;
Bram Moolenaar071d4272004-06-13 20:20:40 +00009083
9084 /* (part of) subcommand already typed */
9085 if (*arg != NUL)
9086 {
9087 p = skiptowhite(arg);
9088 if (*p != NUL) /* past "default" or group name */
9089 {
Bram Moolenaar4f688582007-07-24 12:34:30 +00009090 include_default = 0;
Bram Moolenaar071d4272004-06-13 20:20:40 +00009091 if (STRNCMP("default", arg, p - arg) == 0)
9092 {
9093 arg = skipwhite(p);
9094 xp->xp_pattern = arg;
9095 p = skiptowhite(arg);
9096 }
9097 if (*p != NUL) /* past group name */
9098 {
Bram Moolenaar4f688582007-07-24 12:34:30 +00009099 include_link = 0;
Bram Moolenaar071d4272004-06-13 20:20:40 +00009100 if (arg[1] == 'i' && arg[0] == 'N')
9101 highlight_list();
9102 if (STRNCMP("link", arg, p - arg) == 0
9103 || STRNCMP("clear", arg, p - arg) == 0)
9104 {
9105 xp->xp_pattern = skipwhite(p);
9106 p = skiptowhite(xp->xp_pattern);
9107 if (*p != NUL) /* past first group name */
9108 {
9109 xp->xp_pattern = skipwhite(p);
9110 p = skiptowhite(xp->xp_pattern);
9111 }
9112 }
9113 if (*p != NUL) /* past group name(s) */
9114 xp->xp_context = EXPAND_NOTHING;
9115 }
9116 }
9117 }
9118}
9119
9120/*
9121 * List highlighting matches in a nice way.
9122 */
9123 static void
9124highlight_list()
9125{
9126 int i;
9127
9128 for (i = 10; --i >= 0; )
9129 highlight_list_two(i, hl_attr(HLF_D));
9130 for (i = 40; --i >= 0; )
9131 highlight_list_two(99, 0);
9132}
9133
9134 static void
9135highlight_list_two(cnt, attr)
9136 int cnt;
9137 int attr;
9138{
9139 msg_puts_attr((char_u *)("N \bI \b! \b" + cnt / 11), attr);
9140 msg_clr_eos();
9141 out_flush();
9142 ui_delay(cnt == 99 ? 40L : (long)cnt * 50L, FALSE);
9143}
9144
9145#endif /* FEAT_CMDL_COMPL */
9146
9147#if defined(FEAT_CMDL_COMPL) || (defined(FEAT_SYN_HL) && defined(FEAT_EVAL)) \
9148 || defined(FEAT_SIGNS) || defined(PROTO)
9149/*
9150 * Function given to ExpandGeneric() to obtain the list of group names.
9151 * Also used for synIDattr() function.
9152 */
9153/*ARGSUSED*/
9154 char_u *
9155get_highlight_name(xp, idx)
9156 expand_T *xp;
9157 int idx;
9158{
Bram Moolenaar071d4272004-06-13 20:20:40 +00009159#ifdef FEAT_CMDL_COMPL
Bram Moolenaar4f688582007-07-24 12:34:30 +00009160 if (idx == highlight_ga.ga_len && include_none != 0)
9161 return (char_u *)"none";
9162 if (idx == highlight_ga.ga_len + include_none && include_default != 0)
Bram Moolenaar071d4272004-06-13 20:20:40 +00009163 return (char_u *)"default";
Bram Moolenaar4f688582007-07-24 12:34:30 +00009164 if (idx == highlight_ga.ga_len + include_none + include_default
9165 && include_link != 0)
9166 return (char_u *)"link";
9167 if (idx == highlight_ga.ga_len + include_none + include_default + 1
9168 && include_link != 0)
9169 return (char_u *)"clear";
9170#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00009171 if (idx < 0 || idx >= highlight_ga.ga_len)
9172 return NULL;
9173 return HL_TABLE()[idx].sg_name;
9174}
9175#endif
9176
Bram Moolenaar4f688582007-07-24 12:34:30 +00009177#if defined(FEAT_GUI) || defined(PROTO)
Bram Moolenaar071d4272004-06-13 20:20:40 +00009178/*
9179 * Free all the highlight group fonts.
9180 * Used when quitting for systems which need it.
9181 */
9182 void
9183free_highlight_fonts()
9184{
9185 int idx;
9186
9187 for (idx = 0; idx < highlight_ga.ga_len; ++idx)
9188 {
9189 gui_mch_free_font(HL_TABLE()[idx].sg_font);
9190 HL_TABLE()[idx].sg_font = NOFONT;
9191# ifdef FEAT_XFONTSET
9192 gui_mch_free_fontset(HL_TABLE()[idx].sg_fontset);
9193 HL_TABLE()[idx].sg_fontset = NOFONTSET;
9194# endif
9195 }
9196
9197 gui_mch_free_font(gui.norm_font);
9198# ifdef FEAT_XFONTSET
9199 gui_mch_free_fontset(gui.fontset);
9200# endif
9201# ifndef HAVE_GTK2
9202 gui_mch_free_font(gui.bold_font);
9203 gui_mch_free_font(gui.ital_font);
9204 gui_mch_free_font(gui.boldital_font);
9205# endif
9206}
9207#endif
9208
9209/**************************************
9210 * End of Highlighting stuff *
9211 **************************************/