blob: 8d976ea4056872080d2c260d88cc5bd59d5cc9cb [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() */
Bram Moolenaarf5b63862009-12-16 17:13:44 +0000209static int current_trans_id = 0; /* idem, transparency removed */
Bram Moolenaar071d4272004-06-13 20:20:40 +0000210#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 */
Bram Moolenaarf5b63862009-12-16 17:13:44 +0000285 int si_trans_id; /* idem, transparency removed */
Bram Moolenaar071d4272004-06-13 20:20:40 +0000286 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 /*
Bram Moolenaarf5b63862009-12-16 17:13:44 +00001277 * Go through the list to find the "tick" for the oldest entry that can
Bram Moolenaar071d4272004-06-13 20:20:40 +00001278 * 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
Bram Moolenaarf5b63862009-12-16 17:13:44 +00002322 /* nextgroup ends at end of line, unless "skipnl" or "skipempty" present */
Bram Moolenaar071d4272004-06-13 20:20:40 +00002323 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 Moolenaar72b73c12010-02-24 17:22:20 +01003089 if (result->lnum > syn_buf->b_ml.ml_line_count)
3090 {
3091 /* a "\n" at the end of the pattern may take us below the last line */
3092 result->lnum = syn_buf->b_ml.ml_line_count;
Bram Moolenaar8b9c05f2010-03-02 17:54:33 +01003093 col = (int)STRLEN(ml_get_buf(syn_buf, result->lnum, FALSE));
Bram Moolenaar72b73c12010-02-24 17:22:20 +01003094 }
Bram Moolenaar8c8de832008-06-24 22:58:06 +00003095 if (off != 0)
3096 {
3097 base = ml_get_buf(syn_buf, result->lnum, FALSE);
3098 p = base + col;
3099 if (off > 0)
3100 {
3101 while (off-- && *p != NUL)
3102 mb_ptr_adv(p);
3103 }
3104 else if (off < 0)
3105 {
3106 while (off++ && base < p)
3107 mb_ptr_back(base, p);
3108 }
3109 col = (int)(p - base);
3110 }
3111 result->col = col;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003112}
3113
3114/*
3115 * Get current line in syntax buffer.
3116 */
3117 static char_u *
3118syn_getcurline()
3119{
3120 return ml_get_buf(syn_buf, current_lnum, FALSE);
3121}
3122
3123/*
Bram Moolenaar3b56eb32005-07-11 22:40:32 +00003124 * Call vim_regexec() to find a match with "rmp" in "syn_buf".
Bram Moolenaar071d4272004-06-13 20:20:40 +00003125 * Returns TRUE when there is a match.
3126 */
3127 static int
3128syn_regexec(rmp, lnum, col)
3129 regmmatch_T *rmp;
3130 linenr_T lnum;
3131 colnr_T col;
3132{
Bram Moolenaar3b56eb32005-07-11 22:40:32 +00003133 rmp->rmm_maxcol = syn_buf->b_p_smc;
Bram Moolenaar91a4e822008-01-19 14:59:58 +00003134 if (vim_regexec_multi(rmp, syn_win, syn_buf, lnum, col, NULL) > 0)
Bram Moolenaar071d4272004-06-13 20:20:40 +00003135 {
3136 rmp->startpos[0].lnum += lnum;
3137 rmp->endpos[0].lnum += lnum;
3138 return TRUE;
3139 }
3140 return FALSE;
3141}
3142
3143/*
3144 * Check one position in a line for a matching keyword.
3145 * The caller must check if a keyword can start at startcol.
3146 * Return it's ID if found, 0 otherwise.
3147 */
3148 static int
3149check_keyword_id(line, startcol, endcolp, flagsp, next_listp, cur_si)
3150 char_u *line;
3151 int startcol; /* position in line to check for keyword */
3152 int *endcolp; /* return: character after found keyword */
3153 long *flagsp; /* return: flags of matching keyword */
3154 short **next_listp; /* return: next_list of matching keyword */
3155 stateitem_T *cur_si; /* item at the top of the stack */
3156{
Bram Moolenaardad6b692005-01-25 22:14:34 +00003157 keyentry_T *kp;
3158 char_u *kwp;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003159 int round;
Bram Moolenaardad6b692005-01-25 22:14:34 +00003160 int kwlen;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003161 char_u keyword[MAXKEYWLEN + 1]; /* assume max. keyword len is 80 */
Bram Moolenaardad6b692005-01-25 22:14:34 +00003162 hashtab_T *ht;
3163 hashitem_T *hi;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003164
3165 /* Find first character after the keyword. First character was already
3166 * checked. */
Bram Moolenaardad6b692005-01-25 22:14:34 +00003167 kwp = line + startcol;
3168 kwlen = 0;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003169 do
3170 {
3171#ifdef FEAT_MBYTE
3172 if (has_mbyte)
Bram Moolenaar0fa313a2005-08-10 21:07:57 +00003173 kwlen += (*mb_ptr2len)(kwp + kwlen);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003174 else
3175#endif
Bram Moolenaardad6b692005-01-25 22:14:34 +00003176 ++kwlen;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003177 }
Bram Moolenaardad6b692005-01-25 22:14:34 +00003178 while (vim_iswordc_buf(kwp + kwlen, syn_buf));
Bram Moolenaar071d4272004-06-13 20:20:40 +00003179
Bram Moolenaardad6b692005-01-25 22:14:34 +00003180 if (kwlen > MAXKEYWLEN)
Bram Moolenaar071d4272004-06-13 20:20:40 +00003181 return 0;
3182
3183 /*
3184 * Must make a copy of the keyword, so we can add a NUL and make it
3185 * lowercase.
3186 */
Bram Moolenaarce0842a2005-07-18 21:58:11 +00003187 vim_strncpy(keyword, kwp, kwlen);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003188
3189 /*
3190 * Try twice:
3191 * 1. matching case
3192 * 2. ignoring case
3193 */
3194 for (round = 1; round <= 2; ++round)
3195 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00003196 ht = round == 1 ? &syn_buf->b_keywtab : &syn_buf->b_keywtab_ic;
3197 if (ht->ht_used == 0)
Bram Moolenaar071d4272004-06-13 20:20:40 +00003198 continue;
Bram Moolenaardad6b692005-01-25 22:14:34 +00003199 if (round == 2) /* ignore case */
3200 (void)str_foldcase(kwp, kwlen, keyword, MAXKEYWLEN + 1);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003201
3202 /*
Bram Moolenaardad6b692005-01-25 22:14:34 +00003203 * Find keywords that match. There can be several with different
3204 * attributes.
Bram Moolenaar071d4272004-06-13 20:20:40 +00003205 * When current_next_list is non-zero accept only that group, otherwise:
3206 * Accept a not-contained keyword at toplevel.
3207 * Accept a keyword at other levels only if it is in the contains list.
3208 */
Bram Moolenaardad6b692005-01-25 22:14:34 +00003209 hi = hash_find(ht, keyword);
3210 if (!HASHITEM_EMPTY(hi))
3211 for (kp = HI2KE(hi); kp != NULL; kp = kp->ke_next)
Bram Moolenaar071d4272004-06-13 20:20:40 +00003212 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00003213 if (current_next_list != 0
3214 ? in_id_list(NULL, current_next_list, &kp->k_syn, 0)
3215 : (cur_si == NULL
3216 ? !(kp->flags & HL_CONTAINED)
3217 : in_id_list(cur_si, cur_si->si_cont_list,
3218 &kp->k_syn, kp->flags & HL_CONTAINED)))
3219 {
3220 *endcolp = startcol + kwlen;
3221 *flagsp = kp->flags;
3222 *next_listp = kp->next_list;
3223 return kp->k_syn.id;
3224 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00003225 }
3226 }
3227 return 0;
3228}
3229
3230/*
3231 * Handle ":syntax case" command.
3232 */
Bram Moolenaar071d4272004-06-13 20:20:40 +00003233 static void
3234syn_cmd_case(eap, syncing)
3235 exarg_T *eap;
Bram Moolenaar2c4278f2009-05-17 11:33:22 +00003236 int syncing UNUSED;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003237{
3238 char_u *arg = eap->arg;
3239 char_u *next;
3240
3241 eap->nextcmd = find_nextcmd(arg);
3242 if (eap->skip)
3243 return;
3244
3245 next = skiptowhite(arg);
3246 if (STRNICMP(arg, "match", 5) == 0 && next - arg == 5)
3247 curbuf->b_syn_ic = FALSE;
3248 else if (STRNICMP(arg, "ignore", 6) == 0 && next - arg == 6)
3249 curbuf->b_syn_ic = TRUE;
3250 else
3251 EMSG2(_("E390: Illegal argument: %s"), arg);
3252}
3253
3254/*
Bram Moolenaarce0842a2005-07-18 21:58:11 +00003255 * Handle ":syntax spell" command.
3256 */
Bram Moolenaarce0842a2005-07-18 21:58:11 +00003257 static void
3258syn_cmd_spell(eap, syncing)
3259 exarg_T *eap;
Bram Moolenaar2c4278f2009-05-17 11:33:22 +00003260 int syncing UNUSED;
Bram Moolenaarce0842a2005-07-18 21:58:11 +00003261{
3262 char_u *arg = eap->arg;
3263 char_u *next;
3264
3265 eap->nextcmd = find_nextcmd(arg);
3266 if (eap->skip)
3267 return;
3268
3269 next = skiptowhite(arg);
3270 if (STRNICMP(arg, "toplevel", 8) == 0 && next - arg == 8)
3271 curbuf->b_syn_spell = SYNSPL_TOP;
3272 else if (STRNICMP(arg, "notoplevel", 10) == 0 && next - arg == 10)
3273 curbuf->b_syn_spell = SYNSPL_NOTOP;
Bram Moolenaar86ea7642007-02-04 01:48:10 +00003274 else if (STRNICMP(arg, "default", 7) == 0 && next - arg == 7)
Bram Moolenaarce0842a2005-07-18 21:58:11 +00003275 curbuf->b_syn_spell = SYNSPL_DEFAULT;
3276 else
3277 EMSG2(_("E390: Illegal argument: %s"), arg);
3278}
3279
3280/*
Bram Moolenaar071d4272004-06-13 20:20:40 +00003281 * Clear all syntax info for one buffer.
3282 */
3283 void
3284syntax_clear(buf)
3285 buf_T *buf;
3286{
3287 int i;
3288
Bram Moolenaar5b8d8fd2005-08-16 23:01:50 +00003289 buf->b_syn_error = FALSE; /* clear previous error */
Bram Moolenaar54ee7752005-05-31 22:22:17 +00003290 buf->b_syn_ic = FALSE; /* Use case, by default */
Bram Moolenaarce0842a2005-07-18 21:58:11 +00003291 buf->b_syn_spell = SYNSPL_DEFAULT; /* default spell checking */
Bram Moolenaar54ee7752005-05-31 22:22:17 +00003292 buf->b_syn_containedin = FALSE;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003293
3294 /* free the keywords */
Bram Moolenaardad6b692005-01-25 22:14:34 +00003295 clear_keywtab(&buf->b_keywtab);
3296 clear_keywtab(&buf->b_keywtab_ic);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003297
3298 /* free the syntax patterns */
3299 for (i = buf->b_syn_patterns.ga_len; --i >= 0; )
3300 syn_clear_pattern(buf, i);
3301 ga_clear(&buf->b_syn_patterns);
3302
3303 /* free the syntax clusters */
3304 for (i = buf->b_syn_clusters.ga_len; --i >= 0; )
3305 syn_clear_cluster(buf, i);
3306 ga_clear(&buf->b_syn_clusters);
Bram Moolenaar75c50c42005-06-04 22:06:24 +00003307 buf->b_spell_cluster_id = 0;
3308 buf->b_nospell_cluster_id = 0;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003309
3310 buf->b_syn_sync_flags = 0;
3311 buf->b_syn_sync_minlines = 0;
3312 buf->b_syn_sync_maxlines = 0;
3313 buf->b_syn_sync_linebreaks = 0;
3314
3315 vim_free(buf->b_syn_linecont_prog);
3316 buf->b_syn_linecont_prog = NULL;
3317 vim_free(buf->b_syn_linecont_pat);
3318 buf->b_syn_linecont_pat = NULL;
3319#ifdef FEAT_FOLDING
3320 buf->b_syn_folditems = 0;
3321#endif
3322
3323 /* free the stored states */
3324 syn_stack_free_all(buf);
3325 invalidate_current_state();
3326}
3327
3328/*
3329 * Clear syncing info for one buffer.
3330 */
3331 static void
3332syntax_sync_clear()
3333{
3334 int i;
3335
3336 /* free the syntax patterns */
3337 for (i = curbuf->b_syn_patterns.ga_len; --i >= 0; )
3338 if (SYN_ITEMS(curbuf)[i].sp_syncing)
3339 syn_remove_pattern(curbuf, i);
3340
3341 curbuf->b_syn_sync_flags = 0;
3342 curbuf->b_syn_sync_minlines = 0;
3343 curbuf->b_syn_sync_maxlines = 0;
3344 curbuf->b_syn_sync_linebreaks = 0;
3345
3346 vim_free(curbuf->b_syn_linecont_prog);
3347 curbuf->b_syn_linecont_prog = NULL;
3348 vim_free(curbuf->b_syn_linecont_pat);
3349 curbuf->b_syn_linecont_pat = NULL;
3350
3351 syn_stack_free_all(curbuf); /* Need to recompute all syntax. */
3352}
3353
3354/*
3355 * Remove one pattern from the buffer's pattern list.
3356 */
3357 static void
3358syn_remove_pattern(buf, idx)
3359 buf_T *buf;
3360 int idx;
3361{
3362 synpat_T *spp;
3363
3364 spp = &(SYN_ITEMS(buf)[idx]);
3365#ifdef FEAT_FOLDING
3366 if (spp->sp_flags & HL_FOLD)
3367 --buf->b_syn_folditems;
3368#endif
3369 syn_clear_pattern(buf, idx);
3370 mch_memmove(spp, spp + 1,
3371 sizeof(synpat_T) * (buf->b_syn_patterns.ga_len - idx - 1));
3372 --buf->b_syn_patterns.ga_len;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003373}
3374
3375/*
3376 * Clear and free one syntax pattern. When clearing all, must be called from
3377 * last to first!
3378 */
3379 static void
3380syn_clear_pattern(buf, i)
3381 buf_T *buf;
3382 int i;
3383{
3384 vim_free(SYN_ITEMS(buf)[i].sp_pattern);
3385 vim_free(SYN_ITEMS(buf)[i].sp_prog);
3386 /* Only free sp_cont_list and sp_next_list of first start pattern */
3387 if (i == 0 || SYN_ITEMS(buf)[i - 1].sp_type != SPTYPE_START)
3388 {
3389 vim_free(SYN_ITEMS(buf)[i].sp_cont_list);
3390 vim_free(SYN_ITEMS(buf)[i].sp_next_list);
Bram Moolenaar51f78b22007-10-07 13:22:19 +00003391 vim_free(SYN_ITEMS(buf)[i].sp_syn.cont_in_list);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003392 }
3393}
3394
3395/*
3396 * Clear and free one syntax cluster.
3397 */
3398 static void
3399syn_clear_cluster(buf, i)
3400 buf_T *buf;
3401 int i;
3402{
3403 vim_free(SYN_CLSTR(buf)[i].scl_name);
3404 vim_free(SYN_CLSTR(buf)[i].scl_name_u);
3405 vim_free(SYN_CLSTR(buf)[i].scl_list);
3406}
3407
3408/*
3409 * Handle ":syntax clear" command.
3410 */
3411 static void
3412syn_cmd_clear(eap, syncing)
3413 exarg_T *eap;
3414 int syncing;
3415{
3416 char_u *arg = eap->arg;
3417 char_u *arg_end;
3418 int id;
3419
3420 eap->nextcmd = find_nextcmd(arg);
3421 if (eap->skip)
3422 return;
3423
3424 /*
3425 * We have to disable this within ":syn include @group filename",
3426 * because otherwise @group would get deleted.
3427 * Only required for Vim 5.x syntax files, 6.0 ones don't contain ":syn
3428 * clear".
3429 */
3430 if (curbuf->b_syn_topgrp != 0)
3431 return;
3432
3433 if (ends_excmd(*arg))
3434 {
3435 /*
3436 * No argument: Clear all syntax items.
3437 */
3438 if (syncing)
3439 syntax_sync_clear();
3440 else
3441 {
3442 syntax_clear(curbuf);
Bram Moolenaar2ce06f62005-01-31 19:19:04 +00003443 do_unlet((char_u *)"b:current_syntax", TRUE);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003444 }
3445 }
3446 else
3447 {
3448 /*
3449 * Clear the group IDs that are in the argument.
3450 */
3451 while (!ends_excmd(*arg))
3452 {
3453 arg_end = skiptowhite(arg);
3454 if (*arg == '@')
3455 {
3456 id = syn_scl_namen2id(arg + 1, (int)(arg_end - arg - 1));
3457 if (id == 0)
3458 {
3459 EMSG2(_("E391: No such syntax cluster: %s"), arg);
3460 break;
3461 }
3462 else
3463 {
3464 /*
3465 * We can't physically delete a cluster without changing
3466 * the IDs of other clusters, so we do the next best thing
3467 * and make it empty.
3468 */
3469 short scl_id = id - SYNID_CLUSTER;
3470
3471 vim_free(SYN_CLSTR(curbuf)[scl_id].scl_list);
3472 SYN_CLSTR(curbuf)[scl_id].scl_list = NULL;
3473 }
3474 }
3475 else
3476 {
3477 id = syn_namen2id(arg, (int)(arg_end - arg));
3478 if (id == 0)
3479 {
3480 EMSG2(_(e_nogroup), arg);
3481 break;
3482 }
3483 else
3484 syn_clear_one(id, syncing);
3485 }
3486 arg = skipwhite(arg_end);
3487 }
3488 }
Bram Moolenaar1c8f93f2006-03-12 22:10:07 +00003489 redraw_curbuf_later(SOME_VALID);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003490 syn_stack_free_all(curbuf); /* Need to recompute all syntax. */
3491}
3492
3493/*
3494 * Clear one syntax group for the current buffer.
3495 */
3496 static void
3497syn_clear_one(id, syncing)
3498 int id;
3499 int syncing;
3500{
3501 synpat_T *spp;
3502 int idx;
3503
3504 /* Clear keywords only when not ":syn sync clear group-name" */
3505 if (!syncing)
3506 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00003507 (void)syn_clear_keyword(id, &curbuf->b_keywtab);
3508 (void)syn_clear_keyword(id, &curbuf->b_keywtab_ic);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003509 }
3510
3511 /* clear the patterns for "id" */
3512 for (idx = curbuf->b_syn_patterns.ga_len; --idx >= 0; )
3513 {
3514 spp = &(SYN_ITEMS(curbuf)[idx]);
3515 if (spp->sp_syn.id != id || spp->sp_syncing != syncing)
3516 continue;
3517 syn_remove_pattern(curbuf, idx);
3518 }
3519}
3520
3521/*
3522 * Handle ":syntax on" command.
3523 */
Bram Moolenaar071d4272004-06-13 20:20:40 +00003524 static void
3525syn_cmd_on(eap, syncing)
3526 exarg_T *eap;
Bram Moolenaar2c4278f2009-05-17 11:33:22 +00003527 int syncing UNUSED;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003528{
3529 syn_cmd_onoff(eap, "syntax");
3530}
3531
3532/*
3533 * Handle ":syntax enable" command.
3534 */
Bram Moolenaar071d4272004-06-13 20:20:40 +00003535 static void
3536syn_cmd_enable(eap, syncing)
3537 exarg_T *eap;
Bram Moolenaar2c4278f2009-05-17 11:33:22 +00003538 int syncing UNUSED;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003539{
3540 set_internal_string_var((char_u *)"syntax_cmd", (char_u *)"enable");
3541 syn_cmd_onoff(eap, "syntax");
Bram Moolenaar2ce06f62005-01-31 19:19:04 +00003542 do_unlet((char_u *)"g:syntax_cmd", TRUE);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003543}
3544
3545/*
3546 * Handle ":syntax reset" command.
3547 */
Bram Moolenaar071d4272004-06-13 20:20:40 +00003548 static void
3549syn_cmd_reset(eap, syncing)
3550 exarg_T *eap;
Bram Moolenaar2c4278f2009-05-17 11:33:22 +00003551 int syncing UNUSED;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003552{
3553 eap->nextcmd = check_nextcmd(eap->arg);
3554 if (!eap->skip)
3555 {
3556 set_internal_string_var((char_u *)"syntax_cmd", (char_u *)"reset");
3557 do_cmdline_cmd((char_u *)"runtime! syntax/syncolor.vim");
Bram Moolenaar2ce06f62005-01-31 19:19:04 +00003558 do_unlet((char_u *)"g:syntax_cmd", TRUE);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003559 }
3560}
3561
3562/*
3563 * Handle ":syntax manual" command.
3564 */
Bram Moolenaar071d4272004-06-13 20:20:40 +00003565 static void
3566syn_cmd_manual(eap, syncing)
3567 exarg_T *eap;
Bram Moolenaar2c4278f2009-05-17 11:33:22 +00003568 int syncing UNUSED;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003569{
3570 syn_cmd_onoff(eap, "manual");
3571}
3572
3573/*
3574 * Handle ":syntax off" command.
3575 */
Bram Moolenaar071d4272004-06-13 20:20:40 +00003576 static void
3577syn_cmd_off(eap, syncing)
3578 exarg_T *eap;
Bram Moolenaar2c4278f2009-05-17 11:33:22 +00003579 int syncing UNUSED;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003580{
3581 syn_cmd_onoff(eap, "nosyntax");
3582}
3583
3584 static void
3585syn_cmd_onoff(eap, name)
3586 exarg_T *eap;
3587 char *name;
3588{
3589 char_u buf[100];
3590
3591 eap->nextcmd = check_nextcmd(eap->arg);
3592 if (!eap->skip)
3593 {
3594 STRCPY(buf, "so ");
Bram Moolenaar555b2802005-05-19 21:08:39 +00003595 vim_snprintf((char *)buf + 3, sizeof(buf) - 3, SYNTAX_FNAME, name);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003596 do_cmdline_cmd(buf);
3597 }
3598}
3599
3600/*
3601 * Handle ":syntax [list]" command: list current syntax words.
3602 */
3603 static void
3604syn_cmd_list(eap, syncing)
3605 exarg_T *eap;
3606 int syncing; /* when TRUE: list syncing items */
3607{
3608 char_u *arg = eap->arg;
3609 int id;
3610 char_u *arg_end;
3611
3612 eap->nextcmd = find_nextcmd(arg);
3613 if (eap->skip)
3614 return;
3615
3616 if (!syntax_present(curbuf))
3617 {
3618 MSG(_("No Syntax items defined for this buffer"));
3619 return;
3620 }
3621
3622 if (syncing)
3623 {
3624 if (curbuf->b_syn_sync_flags & SF_CCOMMENT)
3625 {
3626 MSG_PUTS(_("syncing on C-style comments"));
3627 syn_lines_msg();
3628 syn_match_msg();
3629 return;
3630 }
3631 else if (!(curbuf->b_syn_sync_flags & SF_MATCH))
3632 {
3633 if (curbuf->b_syn_sync_minlines == 0)
3634 MSG_PUTS(_("no syncing"));
3635 else
3636 {
3637 MSG_PUTS(_("syncing starts "));
3638 msg_outnum(curbuf->b_syn_sync_minlines);
3639 MSG_PUTS(_(" lines before top line"));
3640 syn_match_msg();
3641 }
3642 return;
3643 }
3644 MSG_PUTS_TITLE(_("\n--- Syntax sync items ---"));
3645 if (curbuf->b_syn_sync_minlines > 0
3646 || curbuf->b_syn_sync_maxlines > 0
3647 || curbuf->b_syn_sync_linebreaks > 0)
3648 {
3649 MSG_PUTS(_("\nsyncing on items"));
3650 syn_lines_msg();
3651 syn_match_msg();
3652 }
3653 }
3654 else
3655 MSG_PUTS_TITLE(_("\n--- Syntax items ---"));
3656 if (ends_excmd(*arg))
3657 {
3658 /*
3659 * No argument: List all group IDs and all syntax clusters.
3660 */
3661 for (id = 1; id <= highlight_ga.ga_len && !got_int; ++id)
3662 syn_list_one(id, syncing, FALSE);
3663 for (id = 0; id < curbuf->b_syn_clusters.ga_len && !got_int; ++id)
3664 syn_list_cluster(id);
3665 }
3666 else
3667 {
3668 /*
3669 * List the group IDs and syntax clusters that are in the argument.
3670 */
3671 while (!ends_excmd(*arg) && !got_int)
3672 {
3673 arg_end = skiptowhite(arg);
3674 if (*arg == '@')
3675 {
3676 id = syn_scl_namen2id(arg + 1, (int)(arg_end - arg - 1));
3677 if (id == 0)
3678 EMSG2(_("E392: No such syntax cluster: %s"), arg);
3679 else
3680 syn_list_cluster(id - SYNID_CLUSTER);
3681 }
3682 else
3683 {
3684 id = syn_namen2id(arg, (int)(arg_end - arg));
3685 if (id == 0)
3686 EMSG2(_(e_nogroup), arg);
3687 else
3688 syn_list_one(id, syncing, TRUE);
3689 }
3690 arg = skipwhite(arg_end);
3691 }
3692 }
3693 eap->nextcmd = check_nextcmd(arg);
3694}
3695
3696 static void
3697syn_lines_msg()
3698{
3699 if (curbuf->b_syn_sync_maxlines > 0 || curbuf->b_syn_sync_minlines > 0)
3700 {
3701 MSG_PUTS("; ");
3702 if (curbuf->b_syn_sync_minlines > 0)
3703 {
3704 MSG_PUTS(_("minimal "));
3705 msg_outnum(curbuf->b_syn_sync_minlines);
3706 if (curbuf->b_syn_sync_maxlines)
3707 MSG_PUTS(", ");
3708 }
3709 if (curbuf->b_syn_sync_maxlines > 0)
3710 {
3711 MSG_PUTS(_("maximal "));
3712 msg_outnum(curbuf->b_syn_sync_maxlines);
3713 }
3714 MSG_PUTS(_(" lines before top line"));
3715 }
3716}
3717
3718 static void
3719syn_match_msg()
3720{
3721 if (curbuf->b_syn_sync_linebreaks > 0)
3722 {
3723 MSG_PUTS(_("; match "));
3724 msg_outnum(curbuf->b_syn_sync_linebreaks);
3725 MSG_PUTS(_(" line breaks"));
3726 }
3727}
3728
3729static int last_matchgroup;
3730
3731struct name_list
3732{
3733 int flag;
3734 char *name;
3735};
3736
3737static void syn_list_flags __ARGS((struct name_list *nl, int flags, int attr));
3738
3739/*
3740 * List one syntax item, for ":syntax" or "syntax list syntax_name".
3741 */
3742 static void
3743syn_list_one(id, syncing, link_only)
3744 int id;
3745 int syncing; /* when TRUE: list syncing items */
3746 int link_only; /* when TRUE; list link-only too */
3747{
3748 int attr;
3749 int idx;
3750 int did_header = FALSE;
3751 synpat_T *spp;
3752 static struct name_list namelist1[] =
3753 {
3754 {HL_DISPLAY, "display"},
3755 {HL_CONTAINED, "contained"},
3756 {HL_ONELINE, "oneline"},
3757 {HL_KEEPEND, "keepend"},
3758 {HL_EXTEND, "extend"},
3759 {HL_EXCLUDENL, "excludenl"},
3760 {HL_TRANSP, "transparent"},
3761 {HL_FOLD, "fold"},
3762 {0, NULL}
3763 };
3764 static struct name_list namelist2[] =
3765 {
3766 {HL_SKIPWHITE, "skipwhite"},
3767 {HL_SKIPNL, "skipnl"},
3768 {HL_SKIPEMPTY, "skipempty"},
3769 {0, NULL}
3770 };
3771
3772 attr = hl_attr(HLF_D); /* highlight like directories */
3773
3774 /* list the keywords for "id" */
3775 if (!syncing)
3776 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00003777 did_header = syn_list_keywords(id, &curbuf->b_keywtab, FALSE, attr);
3778 did_header = syn_list_keywords(id, &curbuf->b_keywtab_ic,
Bram Moolenaar071d4272004-06-13 20:20:40 +00003779 did_header, attr);
3780 }
3781
3782 /* list the patterns for "id" */
3783 for (idx = 0; idx < curbuf->b_syn_patterns.ga_len && !got_int; ++idx)
3784 {
3785 spp = &(SYN_ITEMS(curbuf)[idx]);
3786 if (spp->sp_syn.id != id || spp->sp_syncing != syncing)
3787 continue;
3788
3789 (void)syn_list_header(did_header, 999, id);
3790 did_header = TRUE;
3791 last_matchgroup = 0;
3792 if (spp->sp_type == SPTYPE_MATCH)
3793 {
3794 put_pattern("match", ' ', spp, attr);
3795 msg_putchar(' ');
3796 }
3797 else if (spp->sp_type == SPTYPE_START)
3798 {
3799 while (SYN_ITEMS(curbuf)[idx].sp_type == SPTYPE_START)
3800 put_pattern("start", '=', &SYN_ITEMS(curbuf)[idx++], attr);
3801 if (SYN_ITEMS(curbuf)[idx].sp_type == SPTYPE_SKIP)
3802 put_pattern("skip", '=', &SYN_ITEMS(curbuf)[idx++], attr);
3803 while (idx < curbuf->b_syn_patterns.ga_len
3804 && SYN_ITEMS(curbuf)[idx].sp_type == SPTYPE_END)
3805 put_pattern("end", '=', &SYN_ITEMS(curbuf)[idx++], attr);
3806 --idx;
3807 msg_putchar(' ');
3808 }
3809 syn_list_flags(namelist1, spp->sp_flags, attr);
3810
3811 if (spp->sp_cont_list != NULL)
3812 put_id_list((char_u *)"contains", spp->sp_cont_list, attr);
3813
3814 if (spp->sp_syn.cont_in_list != NULL)
3815 put_id_list((char_u *)"containedin",
3816 spp->sp_syn.cont_in_list, attr);
3817
3818 if (spp->sp_next_list != NULL)
3819 {
3820 put_id_list((char_u *)"nextgroup", spp->sp_next_list, attr);
3821 syn_list_flags(namelist2, spp->sp_flags, attr);
3822 }
3823 if (spp->sp_flags & (HL_SYNC_HERE|HL_SYNC_THERE))
3824 {
3825 if (spp->sp_flags & HL_SYNC_HERE)
3826 msg_puts_attr((char_u *)"grouphere", attr);
3827 else
3828 msg_puts_attr((char_u *)"groupthere", attr);
3829 msg_putchar(' ');
3830 if (spp->sp_sync_idx >= 0)
3831 msg_outtrans(HL_TABLE()[SYN_ITEMS(curbuf)
3832 [spp->sp_sync_idx].sp_syn.id - 1].sg_name);
3833 else
3834 MSG_PUTS("NONE");
3835 msg_putchar(' ');
3836 }
3837 }
3838
3839 /* list the link, if there is one */
3840 if (HL_TABLE()[id - 1].sg_link && (did_header || link_only) && !got_int)
3841 {
3842 (void)syn_list_header(did_header, 999, id);
3843 msg_puts_attr((char_u *)"links to", attr);
3844 msg_putchar(' ');
3845 msg_outtrans(HL_TABLE()[HL_TABLE()[id - 1].sg_link - 1].sg_name);
3846 }
3847}
3848
3849 static void
3850syn_list_flags(nl, flags, attr)
3851 struct name_list *nl;
3852 int flags;
3853 int attr;
3854{
3855 int i;
3856
3857 for (i = 0; nl[i].flag != 0; ++i)
3858 if (flags & nl[i].flag)
3859 {
3860 msg_puts_attr((char_u *)nl[i].name, attr);
3861 msg_putchar(' ');
3862 }
3863}
3864
3865/*
3866 * List one syntax cluster, for ":syntax" or "syntax list syntax_name".
3867 */
3868 static void
3869syn_list_cluster(id)
3870 int id;
3871{
3872 int endcol = 15;
3873
3874 /* slight hack: roughly duplicate the guts of syn_list_header() */
3875 msg_putchar('\n');
3876 msg_outtrans(SYN_CLSTR(curbuf)[id].scl_name);
3877
3878 if (msg_col >= endcol) /* output at least one space */
3879 endcol = msg_col + 1;
3880 if (Columns <= endcol) /* avoid hang for tiny window */
3881 endcol = Columns - 1;
3882
3883 msg_advance(endcol);
3884 if (SYN_CLSTR(curbuf)[id].scl_list != NULL)
3885 {
3886 put_id_list((char_u *)"cluster", SYN_CLSTR(curbuf)[id].scl_list,
3887 hl_attr(HLF_D));
3888 }
3889 else
3890 {
3891 msg_puts_attr((char_u *)"cluster", hl_attr(HLF_D));
3892 msg_puts((char_u *)"=NONE");
3893 }
3894}
3895
3896 static void
3897put_id_list(name, list, attr)
3898 char_u *name;
3899 short *list;
3900 int attr;
3901{
3902 short *p;
3903
3904 msg_puts_attr(name, attr);
3905 msg_putchar('=');
3906 for (p = list; *p; ++p)
3907 {
3908 if (*p >= SYNID_ALLBUT && *p < SYNID_TOP)
3909 {
3910 if (p[1])
3911 MSG_PUTS("ALLBUT");
3912 else
3913 MSG_PUTS("ALL");
3914 }
3915 else if (*p >= SYNID_TOP && *p < SYNID_CONTAINED)
3916 {
3917 MSG_PUTS("TOP");
3918 }
3919 else if (*p >= SYNID_CONTAINED && *p < SYNID_CLUSTER)
3920 {
3921 MSG_PUTS("CONTAINED");
3922 }
3923 else if (*p >= SYNID_CLUSTER)
3924 {
3925 short scl_id = *p - SYNID_CLUSTER;
3926
3927 msg_putchar('@');
3928 msg_outtrans(SYN_CLSTR(curbuf)[scl_id].scl_name);
3929 }
3930 else
3931 msg_outtrans(HL_TABLE()[*p - 1].sg_name);
3932 if (p[1])
3933 msg_putchar(',');
3934 }
3935 msg_putchar(' ');
3936}
3937
3938 static void
3939put_pattern(s, c, spp, attr)
3940 char *s;
3941 int c;
3942 synpat_T *spp;
3943 int attr;
3944{
3945 long n;
3946 int mask;
3947 int first;
3948 static char *sepchars = "/+=-#@\"|'^&";
3949 int i;
3950
3951 /* May have to write "matchgroup=group" */
3952 if (last_matchgroup != spp->sp_syn_match_id)
3953 {
3954 last_matchgroup = spp->sp_syn_match_id;
3955 msg_puts_attr((char_u *)"matchgroup", attr);
3956 msg_putchar('=');
3957 if (last_matchgroup == 0)
3958 msg_outtrans((char_u *)"NONE");
3959 else
3960 msg_outtrans(HL_TABLE()[last_matchgroup - 1].sg_name);
3961 msg_putchar(' ');
3962 }
3963
3964 /* output the name of the pattern and an '=' or ' ' */
3965 msg_puts_attr((char_u *)s, attr);
3966 msg_putchar(c);
3967
3968 /* output the pattern, in between a char that is not in the pattern */
3969 for (i = 0; vim_strchr(spp->sp_pattern, sepchars[i]) != NULL; )
3970 if (sepchars[++i] == NUL)
3971 {
3972 i = 0; /* no good char found, just use the first one */
3973 break;
3974 }
3975 msg_putchar(sepchars[i]);
3976 msg_outtrans(spp->sp_pattern);
3977 msg_putchar(sepchars[i]);
3978
3979 /* output any pattern options */
3980 first = TRUE;
3981 for (i = 0; i < SPO_COUNT; ++i)
3982 {
3983 mask = (1 << i);
3984 if (spp->sp_off_flags & (mask + (mask << SPO_COUNT)))
3985 {
3986 if (!first)
3987 msg_putchar(','); /* separate with commas */
3988 msg_puts((char_u *)spo_name_tab[i]);
3989 n = spp->sp_offsets[i];
3990 if (i != SPO_LC_OFF)
3991 {
3992 if (spp->sp_off_flags & mask)
3993 msg_putchar('s');
3994 else
3995 msg_putchar('e');
3996 if (n > 0)
3997 msg_putchar('+');
3998 }
3999 if (n || i == SPO_LC_OFF)
4000 msg_outnum(n);
4001 first = FALSE;
4002 }
4003 }
4004 msg_putchar(' ');
4005}
4006
4007/*
4008 * List or clear the keywords for one syntax group.
4009 * Return TRUE if the header has been printed.
4010 */
4011 static int
Bram Moolenaardad6b692005-01-25 22:14:34 +00004012syn_list_keywords(id, ht, did_header, attr)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004013 int id;
Bram Moolenaardad6b692005-01-25 22:14:34 +00004014 hashtab_T *ht;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004015 int did_header; /* header has already been printed */
4016 int attr;
4017{
Bram Moolenaar071d4272004-06-13 20:20:40 +00004018 int outlen;
Bram Moolenaardad6b692005-01-25 22:14:34 +00004019 hashitem_T *hi;
4020 keyentry_T *kp;
4021 int todo;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004022 int prev_contained = 0;
4023 short *prev_next_list = NULL;
4024 short *prev_cont_in_list = NULL;
4025 int prev_skipnl = 0;
4026 int prev_skipwhite = 0;
4027 int prev_skipempty = 0;
4028
Bram Moolenaar071d4272004-06-13 20:20:40 +00004029 /*
4030 * Unfortunately, this list of keywords is not sorted on alphabet but on
4031 * hash value...
4032 */
Bram Moolenaara93fa7e2006-04-17 22:14:47 +00004033 todo = (int)ht->ht_used;
Bram Moolenaardad6b692005-01-25 22:14:34 +00004034 for (hi = ht->ht_array; todo > 0 && !got_int; ++hi)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004035 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00004036 if (!HASHITEM_EMPTY(hi))
Bram Moolenaar071d4272004-06-13 20:20:40 +00004037 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00004038 --todo;
4039 for (kp = HI2KE(hi); kp != NULL && !got_int; kp = kp->ke_next)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004040 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00004041 if (kp->k_syn.id == id)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004042 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00004043 if (prev_contained != (kp->flags & HL_CONTAINED)
4044 || prev_skipnl != (kp->flags & HL_SKIPNL)
4045 || prev_skipwhite != (kp->flags & HL_SKIPWHITE)
4046 || prev_skipempty != (kp->flags & HL_SKIPEMPTY)
4047 || prev_cont_in_list != kp->k_syn.cont_in_list
4048 || prev_next_list != kp->next_list)
4049 outlen = 9999;
4050 else
4051 outlen = (int)STRLEN(kp->keyword);
4052 /* output "contained" and "nextgroup" on each line */
4053 if (syn_list_header(did_header, outlen, id))
Bram Moolenaar071d4272004-06-13 20:20:40 +00004054 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00004055 prev_contained = 0;
4056 prev_next_list = NULL;
4057 prev_cont_in_list = NULL;
4058 prev_skipnl = 0;
4059 prev_skipwhite = 0;
4060 prev_skipempty = 0;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004061 }
Bram Moolenaardad6b692005-01-25 22:14:34 +00004062 did_header = TRUE;
4063 if (prev_contained != (kp->flags & HL_CONTAINED))
Bram Moolenaar071d4272004-06-13 20:20:40 +00004064 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00004065 msg_puts_attr((char_u *)"contained", attr);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004066 msg_putchar(' ');
Bram Moolenaardad6b692005-01-25 22:14:34 +00004067 prev_contained = (kp->flags & HL_CONTAINED);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004068 }
Bram Moolenaardad6b692005-01-25 22:14:34 +00004069 if (kp->k_syn.cont_in_list != prev_cont_in_list)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004070 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00004071 put_id_list((char_u *)"containedin",
4072 kp->k_syn.cont_in_list, attr);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004073 msg_putchar(' ');
Bram Moolenaardad6b692005-01-25 22:14:34 +00004074 prev_cont_in_list = kp->k_syn.cont_in_list;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004075 }
Bram Moolenaardad6b692005-01-25 22:14:34 +00004076 if (kp->next_list != prev_next_list)
4077 {
4078 put_id_list((char_u *)"nextgroup", kp->next_list, attr);
4079 msg_putchar(' ');
4080 prev_next_list = kp->next_list;
4081 if (kp->flags & HL_SKIPNL)
4082 {
4083 msg_puts_attr((char_u *)"skipnl", attr);
4084 msg_putchar(' ');
4085 prev_skipnl = (kp->flags & HL_SKIPNL);
4086 }
4087 if (kp->flags & HL_SKIPWHITE)
4088 {
4089 msg_puts_attr((char_u *)"skipwhite", attr);
4090 msg_putchar(' ');
4091 prev_skipwhite = (kp->flags & HL_SKIPWHITE);
4092 }
4093 if (kp->flags & HL_SKIPEMPTY)
4094 {
4095 msg_puts_attr((char_u *)"skipempty", attr);
4096 msg_putchar(' ');
4097 prev_skipempty = (kp->flags & HL_SKIPEMPTY);
4098 }
4099 }
4100 msg_outtrans(kp->keyword);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004101 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00004102 }
4103 }
4104 }
4105
4106 return did_header;
4107}
4108
4109 static void
Bram Moolenaardad6b692005-01-25 22:14:34 +00004110syn_clear_keyword(id, ht)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004111 int id;
Bram Moolenaardad6b692005-01-25 22:14:34 +00004112 hashtab_T *ht;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004113{
Bram Moolenaardad6b692005-01-25 22:14:34 +00004114 hashitem_T *hi;
4115 keyentry_T *kp;
4116 keyentry_T *kp_prev;
4117 keyentry_T *kp_next;
4118 int todo;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004119
Bram Moolenaardad6b692005-01-25 22:14:34 +00004120 hash_lock(ht);
Bram Moolenaara93fa7e2006-04-17 22:14:47 +00004121 todo = (int)ht->ht_used;
Bram Moolenaardad6b692005-01-25 22:14:34 +00004122 for (hi = ht->ht_array; todo > 0; ++hi)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004123 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00004124 if (!HASHITEM_EMPTY(hi))
Bram Moolenaar071d4272004-06-13 20:20:40 +00004125 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00004126 --todo;
4127 kp_prev = NULL;
4128 for (kp = HI2KE(hi); kp != NULL; )
Bram Moolenaar071d4272004-06-13 20:20:40 +00004129 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00004130 if (kp->k_syn.id == id)
4131 {
4132 kp_next = kp->ke_next;
4133 if (kp_prev == NULL)
4134 {
4135 if (kp_next == NULL)
4136 hash_remove(ht, hi);
4137 else
4138 hi->hi_key = KE2HIKEY(kp_next);
4139 }
4140 else
4141 kp_prev->ke_next = kp_next;
4142 vim_free(kp->next_list);
4143 vim_free(kp->k_syn.cont_in_list);
4144 vim_free(kp);
4145 kp = kp_next;
4146 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00004147 else
Bram Moolenaardad6b692005-01-25 22:14:34 +00004148 {
4149 kp_prev = kp;
4150 kp = kp->ke_next;
4151 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00004152 }
4153 }
4154 }
Bram Moolenaardad6b692005-01-25 22:14:34 +00004155 hash_unlock(ht);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004156}
4157
4158/*
Bram Moolenaardad6b692005-01-25 22:14:34 +00004159 * Clear a whole keyword table.
Bram Moolenaar071d4272004-06-13 20:20:40 +00004160 */
4161 static void
Bram Moolenaardad6b692005-01-25 22:14:34 +00004162clear_keywtab(ht)
4163 hashtab_T *ht;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004164{
Bram Moolenaardad6b692005-01-25 22:14:34 +00004165 hashitem_T *hi;
4166 int todo;
4167 keyentry_T *kp;
4168 keyentry_T *kp_next;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004169
Bram Moolenaara93fa7e2006-04-17 22:14:47 +00004170 todo = (int)ht->ht_used;
Bram Moolenaardad6b692005-01-25 22:14:34 +00004171 for (hi = ht->ht_array; todo > 0; ++hi)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004172 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00004173 if (!HASHITEM_EMPTY(hi))
4174 {
4175 --todo;
Bram Moolenaardad6b692005-01-25 22:14:34 +00004176 for (kp = HI2KE(hi); kp != NULL; kp = kp_next)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004177 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00004178 kp_next = kp->ke_next;
4179 vim_free(kp->next_list);
4180 vim_free(kp->k_syn.cont_in_list);
4181 vim_free(kp);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004182 }
Bram Moolenaardad6b692005-01-25 22:14:34 +00004183 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00004184 }
Bram Moolenaardad6b692005-01-25 22:14:34 +00004185 hash_clear(ht);
4186 hash_init(ht);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004187}
4188
4189/*
4190 * Add a keyword to the list of keywords.
4191 */
4192 static void
4193add_keyword(name, id, flags, cont_in_list, next_list)
4194 char_u *name; /* name of keyword */
4195 int id; /* group ID for this keyword */
4196 int flags; /* flags for this keyword */
4197 short *cont_in_list; /* containedin for this keyword */
4198 short *next_list; /* nextgroup for this keyword */
4199{
Bram Moolenaardad6b692005-01-25 22:14:34 +00004200 keyentry_T *kp;
4201 hashtab_T *ht;
4202 hashitem_T *hi;
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004203 char_u *name_ic;
Bram Moolenaardad6b692005-01-25 22:14:34 +00004204 long_u hash;
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004205 char_u name_folded[MAXKEYWLEN + 1];
Bram Moolenaar071d4272004-06-13 20:20:40 +00004206
4207 if (curbuf->b_syn_ic)
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004208 name_ic = str_foldcase(name, (int)STRLEN(name),
4209 name_folded, MAXKEYWLEN + 1);
4210 else
4211 name_ic = name;
Bram Moolenaardad6b692005-01-25 22:14:34 +00004212 kp = (keyentry_T *)alloc((int)(sizeof(keyentry_T) + STRLEN(name_ic)));
4213 if (kp == NULL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004214 return;
Bram Moolenaardad6b692005-01-25 22:14:34 +00004215 STRCPY(kp->keyword, name_ic);
Bram Moolenaardad6b692005-01-25 22:14:34 +00004216 kp->k_syn.id = id;
4217 kp->k_syn.inc_tag = current_syn_inc_tag;
4218 kp->flags = flags;
4219 kp->k_syn.cont_in_list = copy_id_list(cont_in_list);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004220 if (cont_in_list != NULL)
4221 curbuf->b_syn_containedin = TRUE;
Bram Moolenaardad6b692005-01-25 22:14:34 +00004222 kp->next_list = copy_id_list(next_list);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004223
4224 if (curbuf->b_syn_ic)
Bram Moolenaardad6b692005-01-25 22:14:34 +00004225 ht = &curbuf->b_keywtab_ic;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004226 else
Bram Moolenaardad6b692005-01-25 22:14:34 +00004227 ht = &curbuf->b_keywtab;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004228
Bram Moolenaardad6b692005-01-25 22:14:34 +00004229 hash = hash_hash(kp->keyword);
4230 hi = hash_lookup(ht, kp->keyword, hash);
4231 if (HASHITEM_EMPTY(hi))
Bram Moolenaar071d4272004-06-13 20:20:40 +00004232 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00004233 /* new keyword, add to hashtable */
4234 kp->ke_next = NULL;
4235 hash_add_item(ht, hi, kp->keyword, hash);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004236 }
Bram Moolenaardad6b692005-01-25 22:14:34 +00004237 else
Bram Moolenaar071d4272004-06-13 20:20:40 +00004238 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00004239 /* keyword already exists, prepend to list */
4240 kp->ke_next = HI2KE(hi);
4241 hi->hi_key = KE2HIKEY(kp);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004242 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00004243}
4244
4245/*
4246 * Get the start and end of the group name argument.
4247 * Return a pointer to the first argument.
4248 * Return NULL if the end of the command was found instead of further args.
4249 */
4250 static char_u *
4251get_group_name(arg, name_end)
4252 char_u *arg; /* start of the argument */
4253 char_u **name_end; /* pointer to end of the name */
4254{
4255 char_u *rest;
4256
4257 *name_end = skiptowhite(arg);
4258 rest = skipwhite(*name_end);
4259
4260 /*
4261 * Check if there are enough arguments. The first argument may be a
4262 * pattern, where '|' is allowed, so only check for NUL.
4263 */
4264 if (ends_excmd(*arg) || *rest == NUL)
4265 return NULL;
4266 return rest;
4267}
4268
4269/*
4270 * Check for syntax command option arguments.
4271 * This can be called at any place in the list of arguments, and just picks
4272 * out the arguments that are known. Can be called several times in a row to
4273 * collect all options in between other arguments.
4274 * Return a pointer to the next argument (which isn't an option).
4275 * Return NULL for any error;
4276 */
4277 static char_u *
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004278get_syn_options(arg, opt)
4279 char_u *arg; /* next argument to be checked */
4280 syn_opt_arg_T *opt; /* various things */
Bram Moolenaar071d4272004-06-13 20:20:40 +00004281{
Bram Moolenaar071d4272004-06-13 20:20:40 +00004282 char_u *gname_start, *gname;
4283 int syn_id;
4284 int len;
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004285 char *p;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004286 int i;
4287 int fidx;
4288 static struct flag
4289 {
4290 char *name;
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004291 int argtype;
4292 int flags;
4293 } flagtab[] = { {"cCoOnNtTaAiInNeEdD", 0, HL_CONTAINED},
4294 {"oOnNeElLiInNeE", 0, HL_ONELINE},
4295 {"kKeEeEpPeEnNdD", 0, HL_KEEPEND},
4296 {"eExXtTeEnNdD", 0, HL_EXTEND},
4297 {"eExXcClLuUdDeEnNlL", 0, HL_EXCLUDENL},
4298 {"tTrRaAnNsSpPaArReEnNtT", 0, HL_TRANSP},
4299 {"sSkKiIpPnNlL", 0, HL_SKIPNL},
4300 {"sSkKiIpPwWhHiItTeE", 0, HL_SKIPWHITE},
4301 {"sSkKiIpPeEmMpPtTyY", 0, HL_SKIPEMPTY},
4302 {"gGrRoOuUpPhHeErReE", 0, HL_SYNC_HERE},
4303 {"gGrRoOuUpPtThHeErReE", 0, HL_SYNC_THERE},
4304 {"dDiIsSpPlLaAyY", 0, HL_DISPLAY},
4305 {"fFoOlLdD", 0, HL_FOLD},
4306 {"cCoOnNtTaAiInNsS", 1, 0},
4307 {"cCoOnNtTaAiInNeEdDiInN", 2, 0},
4308 {"nNeExXtTgGrRoOuUpP", 3, 0},
Bram Moolenaar071d4272004-06-13 20:20:40 +00004309 };
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004310 static char *first_letters = "cCoOkKeEtTsSgGdDfFnN";
Bram Moolenaar071d4272004-06-13 20:20:40 +00004311
4312 if (arg == NULL) /* already detected error */
4313 return NULL;
4314
Bram Moolenaar071d4272004-06-13 20:20:40 +00004315 for (;;)
4316 {
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004317 /*
4318 * This is used very often when a large number of keywords is defined.
4319 * Need to skip quickly when no option name is found.
4320 * Also avoid tolower(), it's slow.
4321 */
4322 if (strchr(first_letters, *arg) == NULL)
4323 break;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004324
4325 for (fidx = sizeof(flagtab) / sizeof(struct flag); --fidx >= 0; )
4326 {
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004327 p = flagtab[fidx].name;
4328 for (i = 0, len = 0; p[i] != NUL; i += 2, ++len)
4329 if (arg[len] != p[i] && arg[len] != p[i + 1])
4330 break;
4331 if (p[i] == NUL && (vim_iswhite(arg[len])
4332 || (flagtab[fidx].argtype > 0
4333 ? arg[len] == '='
4334 : ends_excmd(arg[len]))))
Bram Moolenaar071d4272004-06-13 20:20:40 +00004335 {
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004336 if (opt->keyword
4337 && (flagtab[fidx].flags == HL_DISPLAY
4338 || flagtab[fidx].flags == HL_FOLD
4339 || flagtab[fidx].flags == HL_EXTEND))
Bram Moolenaar071d4272004-06-13 20:20:40 +00004340 /* treat "display", "fold" and "extend" as a keyword */
4341 fidx = -1;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004342 break;
4343 }
4344 }
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004345 if (fidx < 0) /* no match found */
4346 break;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004347
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004348 if (flagtab[fidx].argtype == 1)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004349 {
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004350 if (!opt->has_cont_list)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004351 {
4352 EMSG(_("E395: contains argument not accepted here"));
4353 return NULL;
4354 }
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004355 if (get_id_list(&arg, 8, &opt->cont_list) == FAIL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004356 return NULL;
4357 }
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004358 else if (flagtab[fidx].argtype == 2)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004359 {
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004360#if 0 /* cannot happen */
4361 if (opt->cont_in_list == NULL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004362 {
4363 EMSG(_("E396: containedin argument not accepted here"));
4364 return NULL;
4365 }
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004366#endif
4367 if (get_id_list(&arg, 11, &opt->cont_in_list) == FAIL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004368 return NULL;
4369 }
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004370 else if (flagtab[fidx].argtype == 3)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004371 {
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004372 if (get_id_list(&arg, 9, &opt->next_list) == FAIL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004373 return NULL;
4374 }
4375 else
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004376 {
4377 opt->flags |= flagtab[fidx].flags;
4378 arg = skipwhite(arg + len);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004379
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004380 if (flagtab[fidx].flags == HL_SYNC_HERE
4381 || flagtab[fidx].flags == HL_SYNC_THERE)
4382 {
4383 if (opt->sync_idx == NULL)
4384 {
4385 EMSG(_("E393: group[t]here not accepted here"));
4386 return NULL;
4387 }
4388 gname_start = arg;
4389 arg = skiptowhite(arg);
4390 if (gname_start == arg)
4391 return NULL;
4392 gname = vim_strnsave(gname_start, (int)(arg - gname_start));
4393 if (gname == NULL)
4394 return NULL;
4395 if (STRCMP(gname, "NONE") == 0)
4396 *opt->sync_idx = NONE_IDX;
4397 else
4398 {
4399 syn_id = syn_name2id(gname);
4400 for (i = curbuf->b_syn_patterns.ga_len; --i >= 0; )
4401 if (SYN_ITEMS(curbuf)[i].sp_syn.id == syn_id
4402 && SYN_ITEMS(curbuf)[i].sp_type == SPTYPE_START)
4403 {
4404 *opt->sync_idx = i;
4405 break;
4406 }
4407 if (i < 0)
4408 {
4409 EMSG2(_("E394: Didn't find region item for %s"), gname);
4410 vim_free(gname);
4411 return NULL;
4412 }
4413 }
4414
4415 vim_free(gname);
4416 arg = skipwhite(arg);
4417 }
4418#ifdef FEAT_FOLDING
4419 else if (flagtab[fidx].flags == HL_FOLD
4420 && foldmethodIsSyntax(curwin))
4421 /* Need to update folds later. */
4422 foldUpdateAll(curwin);
4423#endif
4424 }
4425 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00004426
4427 return arg;
4428}
4429
4430/*
4431 * Adjustments to syntax item when declared in a ":syn include"'d file.
4432 * Set the contained flag, and if the item is not already contained, add it
4433 * to the specified top-level group, if any.
4434 */
4435 static void
4436syn_incl_toplevel(id, flagsp)
4437 int id;
4438 int *flagsp;
4439{
4440 if ((*flagsp & HL_CONTAINED) || curbuf->b_syn_topgrp == 0)
4441 return;
4442 *flagsp |= HL_CONTAINED;
4443 if (curbuf->b_syn_topgrp >= SYNID_CLUSTER)
4444 {
4445 /* We have to alloc this, because syn_combine_list() will free it. */
4446 short *grp_list = (short *)alloc((unsigned)(2 * sizeof(short)));
4447 int tlg_id = curbuf->b_syn_topgrp - SYNID_CLUSTER;
4448
4449 if (grp_list != NULL)
4450 {
4451 grp_list[0] = id;
4452 grp_list[1] = 0;
4453 syn_combine_list(&SYN_CLSTR(curbuf)[tlg_id].scl_list, &grp_list,
4454 CLUSTER_ADD);
4455 }
4456 }
4457}
4458
4459/*
4460 * Handle ":syntax include [@{group-name}] filename" command.
4461 */
Bram Moolenaar071d4272004-06-13 20:20:40 +00004462 static void
4463syn_cmd_include(eap, syncing)
4464 exarg_T *eap;
Bram Moolenaar2c4278f2009-05-17 11:33:22 +00004465 int syncing UNUSED;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004466{
4467 char_u *arg = eap->arg;
4468 int sgl_id = 1;
4469 char_u *group_name_end;
4470 char_u *rest;
4471 char_u *errormsg = NULL;
4472 int prev_toplvl_grp;
4473 int prev_syn_inc_tag;
4474 int source = FALSE;
4475
4476 eap->nextcmd = find_nextcmd(arg);
4477 if (eap->skip)
4478 return;
4479
4480 if (arg[0] == '@')
4481 {
4482 ++arg;
4483 rest = get_group_name(arg, &group_name_end);
4484 if (rest == NULL)
4485 {
4486 EMSG((char_u *)_("E397: Filename required"));
4487 return;
4488 }
4489 sgl_id = syn_check_cluster(arg, (int)(group_name_end - arg));
4490 /* separate_nextcmd() and expand_filename() depend on this */
4491 eap->arg = rest;
4492 }
4493
4494 /*
4495 * Everything that's left, up to the next command, should be the
4496 * filename to include.
4497 */
4498 eap->argt |= (XFILE | NOSPC);
4499 separate_nextcmd(eap);
4500 if (*eap->arg == '<' || *eap->arg == '$' || mch_isFullName(eap->arg))
4501 {
4502 /* For an absolute path, "$VIM/..." or "<sfile>.." we ":source" the
4503 * file. Need to expand the file name first. In other cases
4504 * ":runtime!" is used. */
4505 source = TRUE;
4506 if (expand_filename(eap, syn_cmdlinep, &errormsg) == FAIL)
4507 {
4508 if (errormsg != NULL)
4509 EMSG(errormsg);
4510 return;
4511 }
4512 }
4513
4514 /*
4515 * Save and restore the existing top-level grouplist id and ":syn
4516 * include" tag around the actual inclusion.
4517 */
4518 prev_syn_inc_tag = current_syn_inc_tag;
4519 current_syn_inc_tag = ++running_syn_inc_tag;
4520 prev_toplvl_grp = curbuf->b_syn_topgrp;
4521 curbuf->b_syn_topgrp = sgl_id;
Bram Moolenaar786b1dc2007-06-28 10:03:45 +00004522 if (source ? do_source(eap->arg, FALSE, DOSO_NONE) == FAIL
4523 : source_runtime(eap->arg, TRUE) == FAIL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004524 EMSG2(_(e_notopen), eap->arg);
4525 curbuf->b_syn_topgrp = prev_toplvl_grp;
4526 current_syn_inc_tag = prev_syn_inc_tag;
4527}
4528
4529/*
4530 * Handle ":syntax keyword {group-name} [{option}] keyword .." command.
4531 */
Bram Moolenaar071d4272004-06-13 20:20:40 +00004532 static void
4533syn_cmd_keyword(eap, syncing)
4534 exarg_T *eap;
Bram Moolenaar2c4278f2009-05-17 11:33:22 +00004535 int syncing UNUSED;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004536{
4537 char_u *arg = eap->arg;
4538 char_u *group_name_end;
4539 int syn_id;
4540 char_u *rest;
4541 char_u *keyword_copy;
4542 char_u *p;
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004543 char_u *kw;
4544 syn_opt_arg_T syn_opt_arg;
4545 int cnt;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004546
4547 rest = get_group_name(arg, &group_name_end);
4548
4549 if (rest != NULL)
4550 {
4551 syn_id = syn_check_group(arg, (int)(group_name_end - arg));
4552
4553 /* allocate a buffer, for removing the backslashes in the keyword */
4554 keyword_copy = alloc((unsigned)STRLEN(rest) + 1);
4555 if (keyword_copy != NULL)
4556 {
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004557 syn_opt_arg.flags = 0;
4558 syn_opt_arg.keyword = TRUE;
4559 syn_opt_arg.sync_idx = NULL;
4560 syn_opt_arg.has_cont_list = FALSE;
4561 syn_opt_arg.cont_in_list = NULL;
4562 syn_opt_arg.next_list = NULL;
4563
Bram Moolenaar071d4272004-06-13 20:20:40 +00004564 /*
4565 * The options given apply to ALL keywords, so all options must be
4566 * found before keywords can be created.
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004567 * 1: collect the options and copy the keywords to keyword_copy.
Bram Moolenaar071d4272004-06-13 20:20:40 +00004568 */
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004569 cnt = 0;
4570 p = keyword_copy;
4571 for ( ; rest != NULL && !ends_excmd(*rest); rest = skipwhite(rest))
Bram Moolenaar071d4272004-06-13 20:20:40 +00004572 {
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004573 rest = get_syn_options(rest, &syn_opt_arg);
4574 if (rest == NULL || ends_excmd(*rest))
4575 break;
4576 /* Copy the keyword, removing backslashes, and add a NUL. */
4577 while (*rest != NUL && !vim_iswhite(*rest))
Bram Moolenaar071d4272004-06-13 20:20:40 +00004578 {
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004579 if (*rest == '\\' && rest[1] != NUL)
4580 ++rest;
4581 *p++ = *rest++;
4582 }
4583 *p++ = NUL;
4584 ++cnt;
4585 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00004586
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004587 if (!eap->skip)
4588 {
4589 /* Adjust flags for use of ":syn include". */
4590 syn_incl_toplevel(syn_id, &syn_opt_arg.flags);
4591
4592 /*
4593 * 2: Add an entry for each keyword.
4594 */
4595 for (kw = keyword_copy; --cnt >= 0; kw += STRLEN(kw) + 1)
4596 {
4597 for (p = vim_strchr(kw, '['); ; )
4598 {
4599 if (p != NULL)
4600 *p = NUL;
4601 add_keyword(kw, syn_id, syn_opt_arg.flags,
4602 syn_opt_arg.cont_in_list,
4603 syn_opt_arg.next_list);
Bram Moolenaar26a60b42005-02-22 08:49:11 +00004604 if (p == NULL)
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004605 break;
Bram Moolenaar26a60b42005-02-22 08:49:11 +00004606 if (p[1] == NUL)
4607 {
Bram Moolenaarfc1421e2006-04-20 22:17:20 +00004608 EMSG2(_("E789: Missing ']': %s"), kw);
Bram Moolenaar26a60b42005-02-22 08:49:11 +00004609 kw = p + 2; /* skip over the NUL */
4610 break;
4611 }
4612 if (p[1] == ']')
4613 {
4614 kw = p + 1; /* skip over the "]" */
4615 break;
4616 }
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004617#ifdef FEAT_MBYTE
4618 if (has_mbyte)
4619 {
Bram Moolenaar0fa313a2005-08-10 21:07:57 +00004620 int l = (*mb_ptr2len)(p + 1);
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004621
4622 mch_memmove(p, p + 1, l);
4623 p += l;
4624 }
4625 else
Bram Moolenaar071d4272004-06-13 20:20:40 +00004626#endif
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004627 {
4628 p[0] = p[1];
4629 ++p;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004630 }
4631 }
4632 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00004633 }
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004634
Bram Moolenaar071d4272004-06-13 20:20:40 +00004635 vim_free(keyword_copy);
Bram Moolenaar26a60b42005-02-22 08:49:11 +00004636 vim_free(syn_opt_arg.cont_in_list);
4637 vim_free(syn_opt_arg.next_list);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004638 }
4639 }
4640
4641 if (rest != NULL)
4642 eap->nextcmd = check_nextcmd(rest);
4643 else
4644 EMSG2(_(e_invarg2), arg);
4645
Bram Moolenaar1c8f93f2006-03-12 22:10:07 +00004646 redraw_curbuf_later(SOME_VALID);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004647 syn_stack_free_all(curbuf); /* Need to recompute all syntax. */
4648}
4649
4650/*
4651 * Handle ":syntax match {name} [{options}] {pattern} [{options}]".
4652 *
4653 * Also ":syntax sync match {name} [[grouphere | groupthere] {group-name}] .."
4654 */
4655 static void
4656syn_cmd_match(eap, syncing)
4657 exarg_T *eap;
4658 int syncing; /* TRUE for ":syntax sync match .. " */
4659{
4660 char_u *arg = eap->arg;
4661 char_u *group_name_end;
4662 char_u *rest;
4663 synpat_T item; /* the item found in the line */
4664 int syn_id;
4665 int idx;
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004666 syn_opt_arg_T syn_opt_arg;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004667 int sync_idx = 0;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004668
4669 /* Isolate the group name, check for validity */
4670 rest = get_group_name(arg, &group_name_end);
4671
4672 /* Get options before the pattern */
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004673 syn_opt_arg.flags = 0;
4674 syn_opt_arg.keyword = FALSE;
4675 syn_opt_arg.sync_idx = syncing ? &sync_idx : NULL;
4676 syn_opt_arg.has_cont_list = TRUE;
4677 syn_opt_arg.cont_list = NULL;
4678 syn_opt_arg.cont_in_list = NULL;
4679 syn_opt_arg.next_list = NULL;
4680 rest = get_syn_options(rest, &syn_opt_arg);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004681
4682 /* get the pattern. */
4683 init_syn_patterns();
4684 vim_memset(&item, 0, sizeof(item));
4685 rest = get_syn_pattern(rest, &item);
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004686 if (vim_regcomp_had_eol() && !(syn_opt_arg.flags & HL_EXCLUDENL))
4687 syn_opt_arg.flags |= HL_HAS_EOL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004688
4689 /* Get options after the pattern */
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004690 rest = get_syn_options(rest, &syn_opt_arg);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004691
4692 if (rest != NULL) /* all arguments are valid */
4693 {
4694 /*
4695 * Check for trailing command and illegal trailing arguments.
4696 */
4697 eap->nextcmd = check_nextcmd(rest);
4698 if (!ends_excmd(*rest) || eap->skip)
4699 rest = NULL;
4700 else if (ga_grow(&curbuf->b_syn_patterns, 1) != FAIL
4701 && (syn_id = syn_check_group(arg,
4702 (int)(group_name_end - arg))) != 0)
4703 {
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004704 syn_incl_toplevel(syn_id, &syn_opt_arg.flags);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004705 /*
4706 * Store the pattern in the syn_items list
4707 */
4708 idx = curbuf->b_syn_patterns.ga_len;
4709 SYN_ITEMS(curbuf)[idx] = item;
4710 SYN_ITEMS(curbuf)[idx].sp_syncing = syncing;
4711 SYN_ITEMS(curbuf)[idx].sp_type = SPTYPE_MATCH;
4712 SYN_ITEMS(curbuf)[idx].sp_syn.id = syn_id;
4713 SYN_ITEMS(curbuf)[idx].sp_syn.inc_tag = current_syn_inc_tag;
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004714 SYN_ITEMS(curbuf)[idx].sp_flags = syn_opt_arg.flags;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004715 SYN_ITEMS(curbuf)[idx].sp_sync_idx = sync_idx;
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004716 SYN_ITEMS(curbuf)[idx].sp_cont_list = syn_opt_arg.cont_list;
4717 SYN_ITEMS(curbuf)[idx].sp_syn.cont_in_list =
4718 syn_opt_arg.cont_in_list;
4719 if (syn_opt_arg.cont_in_list != NULL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004720 curbuf->b_syn_containedin = TRUE;
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004721 SYN_ITEMS(curbuf)[idx].sp_next_list = syn_opt_arg.next_list;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004722 ++curbuf->b_syn_patterns.ga_len;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004723
4724 /* remember that we found a match for syncing on */
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004725 if (syn_opt_arg.flags & (HL_SYNC_HERE|HL_SYNC_THERE))
Bram Moolenaar071d4272004-06-13 20:20:40 +00004726 curbuf->b_syn_sync_flags |= SF_MATCH;
4727#ifdef FEAT_FOLDING
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004728 if (syn_opt_arg.flags & HL_FOLD)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004729 ++curbuf->b_syn_folditems;
4730#endif
4731
Bram Moolenaar1c8f93f2006-03-12 22:10:07 +00004732 redraw_curbuf_later(SOME_VALID);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004733 syn_stack_free_all(curbuf); /* Need to recompute all syntax. */
4734 return; /* don't free the progs and patterns now */
4735 }
4736 }
4737
4738 /*
4739 * Something failed, free the allocated memory.
4740 */
4741 vim_free(item.sp_prog);
4742 vim_free(item.sp_pattern);
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004743 vim_free(syn_opt_arg.cont_list);
4744 vim_free(syn_opt_arg.cont_in_list);
4745 vim_free(syn_opt_arg.next_list);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004746
4747 if (rest == NULL)
4748 EMSG2(_(e_invarg2), arg);
4749}
4750
4751/*
4752 * Handle ":syntax region {group-name} [matchgroup={group-name}]
4753 * start {start} .. [skip {skip}] end {end} .. [{options}]".
4754 */
4755 static void
4756syn_cmd_region(eap, syncing)
4757 exarg_T *eap;
4758 int syncing; /* TRUE for ":syntax sync region .." */
4759{
4760 char_u *arg = eap->arg;
4761 char_u *group_name_end;
4762 char_u *rest; /* next arg, NULL on error */
4763 char_u *key_end;
4764 char_u *key = NULL;
4765 char_u *p;
4766 int item;
4767#define ITEM_START 0
4768#define ITEM_SKIP 1
4769#define ITEM_END 2
4770#define ITEM_MATCHGROUP 3
4771 struct pat_ptr
4772 {
4773 synpat_T *pp_synp; /* pointer to syn_pattern */
4774 int pp_matchgroup_id; /* matchgroup ID */
4775 struct pat_ptr *pp_next; /* pointer to next pat_ptr */
4776 } *(pat_ptrs[3]);
4777 /* patterns found in the line */
4778 struct pat_ptr *ppp;
4779 struct pat_ptr *ppp_next;
4780 int pat_count = 0; /* nr of syn_patterns found */
4781 int syn_id;
4782 int matchgroup_id = 0;
4783 int not_enough = FALSE; /* not enough arguments */
4784 int illegal = FALSE; /* illegal arguments */
4785 int success = FALSE;
4786 int idx;
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004787 syn_opt_arg_T syn_opt_arg;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004788
4789 /* Isolate the group name, check for validity */
4790 rest = get_group_name(arg, &group_name_end);
4791
4792 pat_ptrs[0] = NULL;
4793 pat_ptrs[1] = NULL;
4794 pat_ptrs[2] = NULL;
4795
4796 init_syn_patterns();
4797
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004798 syn_opt_arg.flags = 0;
4799 syn_opt_arg.keyword = FALSE;
4800 syn_opt_arg.sync_idx = NULL;
4801 syn_opt_arg.has_cont_list = TRUE;
4802 syn_opt_arg.cont_list = NULL;
4803 syn_opt_arg.cont_in_list = NULL;
4804 syn_opt_arg.next_list = NULL;
4805
Bram Moolenaar071d4272004-06-13 20:20:40 +00004806 /*
4807 * get the options, patterns and matchgroup.
4808 */
4809 while (rest != NULL && !ends_excmd(*rest))
4810 {
4811 /* Check for option arguments */
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004812 rest = get_syn_options(rest, &syn_opt_arg);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004813 if (rest == NULL || ends_excmd(*rest))
4814 break;
4815
4816 /* must be a pattern or matchgroup then */
4817 key_end = rest;
4818 while (*key_end && !vim_iswhite(*key_end) && *key_end != '=')
4819 ++key_end;
4820 vim_free(key);
4821 key = vim_strnsave_up(rest, (int)(key_end - rest));
4822 if (key == NULL) /* out of memory */
4823 {
4824 rest = NULL;
4825 break;
4826 }
4827 if (STRCMP(key, "MATCHGROUP") == 0)
4828 item = ITEM_MATCHGROUP;
4829 else if (STRCMP(key, "START") == 0)
4830 item = ITEM_START;
4831 else if (STRCMP(key, "END") == 0)
4832 item = ITEM_END;
4833 else if (STRCMP(key, "SKIP") == 0)
4834 {
4835 if (pat_ptrs[ITEM_SKIP] != NULL) /* one skip pattern allowed */
4836 {
4837 illegal = TRUE;
4838 break;
4839 }
4840 item = ITEM_SKIP;
4841 }
4842 else
4843 break;
4844 rest = skipwhite(key_end);
4845 if (*rest != '=')
4846 {
4847 rest = NULL;
4848 EMSG2(_("E398: Missing '=': %s"), arg);
4849 break;
4850 }
4851 rest = skipwhite(rest + 1);
4852 if (*rest == NUL)
4853 {
4854 not_enough = TRUE;
4855 break;
4856 }
4857
4858 if (item == ITEM_MATCHGROUP)
4859 {
4860 p = skiptowhite(rest);
4861 if ((p - rest == 4 && STRNCMP(rest, "NONE", 4) == 0) || eap->skip)
4862 matchgroup_id = 0;
4863 else
4864 {
4865 matchgroup_id = syn_check_group(rest, (int)(p - rest));
4866 if (matchgroup_id == 0)
4867 {
4868 illegal = TRUE;
4869 break;
4870 }
4871 }
4872 rest = skipwhite(p);
4873 }
4874 else
4875 {
4876 /*
4877 * Allocate room for a syn_pattern, and link it in the list of
4878 * syn_patterns for this item, at the start (because the list is
4879 * used from end to start).
4880 */
4881 ppp = (struct pat_ptr *)alloc((unsigned)sizeof(struct pat_ptr));
4882 if (ppp == NULL)
4883 {
4884 rest = NULL;
4885 break;
4886 }
4887 ppp->pp_next = pat_ptrs[item];
4888 pat_ptrs[item] = ppp;
4889 ppp->pp_synp = (synpat_T *)alloc_clear((unsigned)sizeof(synpat_T));
4890 if (ppp->pp_synp == NULL)
4891 {
4892 rest = NULL;
4893 break;
4894 }
4895
4896 /*
4897 * Get the syntax pattern and the following offset(s).
4898 */
4899 /* Enable the appropriate \z specials. */
4900 if (item == ITEM_START)
4901 reg_do_extmatch = REX_SET;
4902 else if (item == ITEM_SKIP || item == ITEM_END)
4903 reg_do_extmatch = REX_USE;
4904 rest = get_syn_pattern(rest, ppp->pp_synp);
4905 reg_do_extmatch = 0;
4906 if (item == ITEM_END && vim_regcomp_had_eol()
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004907 && !(syn_opt_arg.flags & HL_EXCLUDENL))
Bram Moolenaar071d4272004-06-13 20:20:40 +00004908 ppp->pp_synp->sp_flags |= HL_HAS_EOL;
4909 ppp->pp_matchgroup_id = matchgroup_id;
4910 ++pat_count;
4911 }
4912 }
4913 vim_free(key);
4914 if (illegal || not_enough)
4915 rest = NULL;
4916
4917 /*
4918 * Must have a "start" and "end" pattern.
4919 */
4920 if (rest != NULL && (pat_ptrs[ITEM_START] == NULL ||
4921 pat_ptrs[ITEM_END] == NULL))
4922 {
4923 not_enough = TRUE;
4924 rest = NULL;
4925 }
4926
4927 if (rest != NULL)
4928 {
4929 /*
4930 * Check for trailing garbage or command.
4931 * If OK, add the item.
4932 */
4933 eap->nextcmd = check_nextcmd(rest);
4934 if (!ends_excmd(*rest) || eap->skip)
4935 rest = NULL;
4936 else if (ga_grow(&(curbuf->b_syn_patterns), pat_count) != FAIL
4937 && (syn_id = syn_check_group(arg,
4938 (int)(group_name_end - arg))) != 0)
4939 {
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004940 syn_incl_toplevel(syn_id, &syn_opt_arg.flags);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004941 /*
4942 * Store the start/skip/end in the syn_items list
4943 */
4944 idx = curbuf->b_syn_patterns.ga_len;
4945 for (item = ITEM_START; item <= ITEM_END; ++item)
4946 {
4947 for (ppp = pat_ptrs[item]; ppp != NULL; ppp = ppp->pp_next)
4948 {
4949 SYN_ITEMS(curbuf)[idx] = *(ppp->pp_synp);
4950 SYN_ITEMS(curbuf)[idx].sp_syncing = syncing;
4951 SYN_ITEMS(curbuf)[idx].sp_type =
4952 (item == ITEM_START) ? SPTYPE_START :
4953 (item == ITEM_SKIP) ? SPTYPE_SKIP : SPTYPE_END;
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004954 SYN_ITEMS(curbuf)[idx].sp_flags |= syn_opt_arg.flags;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004955 SYN_ITEMS(curbuf)[idx].sp_syn.id = syn_id;
4956 SYN_ITEMS(curbuf)[idx].sp_syn.inc_tag = current_syn_inc_tag;
4957 SYN_ITEMS(curbuf)[idx].sp_syn_match_id =
4958 ppp->pp_matchgroup_id;
4959 if (item == ITEM_START)
4960 {
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004961 SYN_ITEMS(curbuf)[idx].sp_cont_list =
4962 syn_opt_arg.cont_list;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004963 SYN_ITEMS(curbuf)[idx].sp_syn.cont_in_list =
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004964 syn_opt_arg.cont_in_list;
4965 if (syn_opt_arg.cont_in_list != NULL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004966 curbuf->b_syn_containedin = TRUE;
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004967 SYN_ITEMS(curbuf)[idx].sp_next_list =
4968 syn_opt_arg.next_list;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004969 }
4970 ++curbuf->b_syn_patterns.ga_len;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004971 ++idx;
4972#ifdef FEAT_FOLDING
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004973 if (syn_opt_arg.flags & HL_FOLD)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004974 ++curbuf->b_syn_folditems;
4975#endif
4976 }
4977 }
4978
Bram Moolenaar1c8f93f2006-03-12 22:10:07 +00004979 redraw_curbuf_later(SOME_VALID);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004980 syn_stack_free_all(curbuf); /* Need to recompute all syntax. */
4981 success = TRUE; /* don't free the progs and patterns now */
4982 }
4983 }
4984
4985 /*
4986 * Free the allocated memory.
4987 */
4988 for (item = ITEM_START; item <= ITEM_END; ++item)
4989 for (ppp = pat_ptrs[item]; ppp != NULL; ppp = ppp_next)
4990 {
4991 if (!success)
4992 {
4993 vim_free(ppp->pp_synp->sp_prog);
4994 vim_free(ppp->pp_synp->sp_pattern);
4995 }
4996 vim_free(ppp->pp_synp);
4997 ppp_next = ppp->pp_next;
4998 vim_free(ppp);
4999 }
5000
5001 if (!success)
5002 {
Bram Moolenaar6ac54292005-02-02 23:07:25 +00005003 vim_free(syn_opt_arg.cont_list);
5004 vim_free(syn_opt_arg.cont_in_list);
5005 vim_free(syn_opt_arg.next_list);
Bram Moolenaar071d4272004-06-13 20:20:40 +00005006 if (not_enough)
5007 EMSG2(_("E399: Not enough arguments: syntax region %s"), arg);
5008 else if (illegal || rest == NULL)
5009 EMSG2(_(e_invarg2), arg);
5010 }
5011}
5012
5013/*
5014 * A simple syntax group ID comparison function suitable for use in qsort()
5015 */
5016 static int
5017#ifdef __BORLANDC__
5018_RTLENTRYF
5019#endif
5020syn_compare_stub(v1, v2)
5021 const void *v1;
5022 const void *v2;
5023{
5024 const short *s1 = v1;
5025 const short *s2 = v2;
5026
5027 return (*s1 > *s2 ? 1 : *s1 < *s2 ? -1 : 0);
5028}
5029
5030/*
5031 * Combines lists of syntax clusters.
5032 * *clstr1 and *clstr2 must both be allocated memory; they will be consumed.
5033 */
5034 static void
5035syn_combine_list(clstr1, clstr2, list_op)
5036 short **clstr1;
5037 short **clstr2;
5038 int list_op;
5039{
5040 int count1 = 0;
5041 int count2 = 0;
5042 short *g1;
5043 short *g2;
5044 short *clstr = NULL;
5045 int count;
5046 int round;
5047
5048 /*
5049 * Handle degenerate cases.
5050 */
5051 if (*clstr2 == NULL)
5052 return;
5053 if (*clstr1 == NULL || list_op == CLUSTER_REPLACE)
5054 {
5055 if (list_op == CLUSTER_REPLACE)
5056 vim_free(*clstr1);
5057 if (list_op == CLUSTER_REPLACE || list_op == CLUSTER_ADD)
5058 *clstr1 = *clstr2;
5059 else
5060 vim_free(*clstr2);
5061 return;
5062 }
5063
5064 for (g1 = *clstr1; *g1; g1++)
5065 ++count1;
5066 for (g2 = *clstr2; *g2; g2++)
5067 ++count2;
5068
5069 /*
5070 * For speed purposes, sort both lists.
5071 */
5072 qsort(*clstr1, (size_t)count1, sizeof(short), syn_compare_stub);
5073 qsort(*clstr2, (size_t)count2, sizeof(short), syn_compare_stub);
5074
5075 /*
5076 * We proceed in two passes; in round 1, we count the elements to place
5077 * in the new list, and in round 2, we allocate and populate the new
5078 * list. For speed, we use a mergesort-like method, adding the smaller
5079 * of the current elements in each list to the new list.
5080 */
5081 for (round = 1; round <= 2; round++)
5082 {
5083 g1 = *clstr1;
5084 g2 = *clstr2;
5085 count = 0;
5086
5087 /*
5088 * First, loop through the lists until one of them is empty.
5089 */
5090 while (*g1 && *g2)
5091 {
5092 /*
5093 * We always want to add from the first list.
5094 */
5095 if (*g1 < *g2)
5096 {
5097 if (round == 2)
5098 clstr[count] = *g1;
5099 count++;
5100 g1++;
5101 continue;
5102 }
5103 /*
5104 * We only want to add from the second list if we're adding the
5105 * lists.
5106 */
5107 if (list_op == CLUSTER_ADD)
5108 {
5109 if (round == 2)
5110 clstr[count] = *g2;
5111 count++;
5112 }
5113 if (*g1 == *g2)
5114 g1++;
5115 g2++;
5116 }
5117
5118 /*
5119 * Now add the leftovers from whichever list didn't get finished
5120 * first. As before, we only want to add from the second list if
5121 * we're adding the lists.
5122 */
5123 for (; *g1; g1++, count++)
5124 if (round == 2)
5125 clstr[count] = *g1;
5126 if (list_op == CLUSTER_ADD)
5127 for (; *g2; g2++, count++)
5128 if (round == 2)
5129 clstr[count] = *g2;
5130
5131 if (round == 1)
5132 {
5133 /*
5134 * If the group ended up empty, we don't need to allocate any
5135 * space for it.
5136 */
5137 if (count == 0)
5138 {
5139 clstr = NULL;
5140 break;
5141 }
5142 clstr = (short *)alloc((unsigned)((count + 1) * sizeof(short)));
5143 if (clstr == NULL)
5144 break;
5145 clstr[count] = 0;
5146 }
5147 }
5148
5149 /*
5150 * Finally, put the new list in place.
5151 */
5152 vim_free(*clstr1);
5153 vim_free(*clstr2);
5154 *clstr1 = clstr;
5155}
5156
5157/*
5158 * Lookup a syntax cluster name and return it's ID.
5159 * If it is not found, 0 is returned.
5160 */
5161 static int
5162syn_scl_name2id(name)
5163 char_u *name;
5164{
5165 int i;
5166 char_u *name_u;
5167
5168 /* Avoid using stricmp() too much, it's slow on some systems */
5169 name_u = vim_strsave_up(name);
5170 if (name_u == NULL)
5171 return 0;
5172 for (i = curbuf->b_syn_clusters.ga_len; --i >= 0; )
5173 if (SYN_CLSTR(curbuf)[i].scl_name_u != NULL
5174 && STRCMP(name_u, SYN_CLSTR(curbuf)[i].scl_name_u) == 0)
5175 break;
5176 vim_free(name_u);
5177 return (i < 0 ? 0 : i + SYNID_CLUSTER);
5178}
5179
5180/*
5181 * Like syn_scl_name2id(), but take a pointer + length argument.
5182 */
5183 static int
5184syn_scl_namen2id(linep, len)
5185 char_u *linep;
5186 int len;
5187{
5188 char_u *name;
5189 int id = 0;
5190
5191 name = vim_strnsave(linep, len);
5192 if (name != NULL)
5193 {
5194 id = syn_scl_name2id(name);
5195 vim_free(name);
5196 }
5197 return id;
5198}
5199
5200/*
5201 * Find syntax cluster name in the table and return it's ID.
5202 * The argument is a pointer to the name and the length of the name.
5203 * If it doesn't exist yet, a new entry is created.
5204 * Return 0 for failure.
5205 */
5206 static int
5207syn_check_cluster(pp, len)
5208 char_u *pp;
5209 int len;
5210{
5211 int id;
5212 char_u *name;
5213
5214 name = vim_strnsave(pp, len);
5215 if (name == NULL)
5216 return 0;
5217
5218 id = syn_scl_name2id(name);
5219 if (id == 0) /* doesn't exist yet */
5220 id = syn_add_cluster(name);
5221 else
5222 vim_free(name);
5223 return id;
5224}
5225
5226/*
5227 * Add new syntax cluster and return it's ID.
5228 * "name" must be an allocated string, it will be consumed.
5229 * Return 0 for failure.
5230 */
5231 static int
5232syn_add_cluster(name)
Bram Moolenaar217ad922005-03-20 22:37:15 +00005233 char_u *name;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005234{
Bram Moolenaar217ad922005-03-20 22:37:15 +00005235 int len;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005236
5237 /*
5238 * First call for this growarray: init growing array.
5239 */
5240 if (curbuf->b_syn_clusters.ga_data == NULL)
5241 {
Bram Moolenaar217ad922005-03-20 22:37:15 +00005242 curbuf->b_syn_clusters.ga_itemsize = sizeof(syn_cluster_T);
Bram Moolenaar071d4272004-06-13 20:20:40 +00005243 curbuf->b_syn_clusters.ga_growsize = 10;
5244 }
5245
5246 /*
5247 * Make room for at least one other cluster entry.
5248 */
5249 if (ga_grow(&curbuf->b_syn_clusters, 1) == FAIL)
5250 {
5251 vim_free(name);
5252 return 0;
5253 }
5254 len = curbuf->b_syn_clusters.ga_len;
5255
Bram Moolenaar217ad922005-03-20 22:37:15 +00005256 vim_memset(&(SYN_CLSTR(curbuf)[len]), 0, sizeof(syn_cluster_T));
Bram Moolenaar071d4272004-06-13 20:20:40 +00005257 SYN_CLSTR(curbuf)[len].scl_name = name;
5258 SYN_CLSTR(curbuf)[len].scl_name_u = vim_strsave_up(name);
5259 SYN_CLSTR(curbuf)[len].scl_list = NULL;
5260 ++curbuf->b_syn_clusters.ga_len;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005261
Bram Moolenaar217ad922005-03-20 22:37:15 +00005262 if (STRICMP(name, "Spell") == 0)
5263 curbuf->b_spell_cluster_id = len + SYNID_CLUSTER;
Bram Moolenaar6bb68362005-03-22 23:03:44 +00005264 if (STRICMP(name, "NoSpell") == 0)
5265 curbuf->b_nospell_cluster_id = len + SYNID_CLUSTER;
Bram Moolenaar217ad922005-03-20 22:37:15 +00005266
Bram Moolenaar071d4272004-06-13 20:20:40 +00005267 return len + SYNID_CLUSTER;
5268}
5269
5270/*
5271 * Handle ":syntax cluster {cluster-name} [contains={groupname},..]
5272 * [add={groupname},..] [remove={groupname},..]".
5273 */
Bram Moolenaar071d4272004-06-13 20:20:40 +00005274 static void
5275syn_cmd_cluster(eap, syncing)
5276 exarg_T *eap;
Bram Moolenaar2c4278f2009-05-17 11:33:22 +00005277 int syncing UNUSED;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005278{
5279 char_u *arg = eap->arg;
5280 char_u *group_name_end;
5281 char_u *rest;
5282 int scl_id;
5283 short *clstr_list;
5284 int got_clstr = FALSE;
5285 int opt_len;
5286 int list_op;
5287
5288 eap->nextcmd = find_nextcmd(arg);
5289 if (eap->skip)
5290 return;
5291
5292 rest = get_group_name(arg, &group_name_end);
5293
5294 if (rest != NULL)
5295 {
5296 scl_id = syn_check_cluster(arg, (int)(group_name_end - arg))
Bram Moolenaar217ad922005-03-20 22:37:15 +00005297 - SYNID_CLUSTER;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005298
5299 for (;;)
5300 {
5301 if (STRNICMP(rest, "add", 3) == 0
5302 && (vim_iswhite(rest[3]) || rest[3] == '='))
5303 {
5304 opt_len = 3;
5305 list_op = CLUSTER_ADD;
5306 }
5307 else if (STRNICMP(rest, "remove", 6) == 0
5308 && (vim_iswhite(rest[6]) || rest[6] == '='))
5309 {
5310 opt_len = 6;
5311 list_op = CLUSTER_SUBTRACT;
5312 }
5313 else if (STRNICMP(rest, "contains", 8) == 0
5314 && (vim_iswhite(rest[8]) || rest[8] == '='))
5315 {
5316 opt_len = 8;
5317 list_op = CLUSTER_REPLACE;
5318 }
5319 else
5320 break;
5321
5322 clstr_list = NULL;
5323 if (get_id_list(&rest, opt_len, &clstr_list) == FAIL)
5324 {
5325 EMSG2(_(e_invarg2), rest);
5326 break;
5327 }
5328 syn_combine_list(&SYN_CLSTR(curbuf)[scl_id].scl_list,
5329 &clstr_list, list_op);
5330 got_clstr = TRUE;
5331 }
5332
5333 if (got_clstr)
5334 {
Bram Moolenaar1c8f93f2006-03-12 22:10:07 +00005335 redraw_curbuf_later(SOME_VALID);
Bram Moolenaar071d4272004-06-13 20:20:40 +00005336 syn_stack_free_all(curbuf); /* Need to recompute all syntax. */
5337 }
5338 }
5339
5340 if (!got_clstr)
5341 EMSG(_("E400: No cluster specified"));
5342 if (rest == NULL || !ends_excmd(*rest))
5343 EMSG2(_(e_invarg2), arg);
5344}
5345
5346/*
5347 * On first call for current buffer: Init growing array.
5348 */
5349 static void
5350init_syn_patterns()
5351{
5352 curbuf->b_syn_patterns.ga_itemsize = sizeof(synpat_T);
5353 curbuf->b_syn_patterns.ga_growsize = 10;
5354}
5355
5356/*
5357 * Get one pattern for a ":syntax match" or ":syntax region" command.
5358 * Stores the pattern and program in a synpat_T.
5359 * Returns a pointer to the next argument, or NULL in case of an error.
5360 */
5361 static char_u *
5362get_syn_pattern(arg, ci)
5363 char_u *arg;
5364 synpat_T *ci;
5365{
5366 char_u *end;
5367 int *p;
5368 int idx;
5369 char_u *cpo_save;
5370
5371 /* need at least three chars */
5372 if (arg == NULL || arg[1] == NUL || arg[2] == NUL)
5373 return NULL;
5374
5375 end = skip_regexp(arg + 1, *arg, TRUE, NULL);
5376 if (*end != *arg) /* end delimiter not found */
5377 {
5378 EMSG2(_("E401: Pattern delimiter not found: %s"), arg);
5379 return NULL;
5380 }
5381 /* store the pattern and compiled regexp program */
5382 if ((ci->sp_pattern = vim_strnsave(arg + 1, (int)(end - arg - 1))) == NULL)
5383 return NULL;
5384
5385 /* Make 'cpoptions' empty, to avoid the 'l' flag */
5386 cpo_save = p_cpo;
5387 p_cpo = (char_u *)"";
5388 ci->sp_prog = vim_regcomp(ci->sp_pattern, RE_MAGIC);
5389 p_cpo = cpo_save;
5390
5391 if (ci->sp_prog == NULL)
5392 return NULL;
5393 ci->sp_ic = curbuf->b_syn_ic;
5394
5395 /*
5396 * Check for a match, highlight or region offset.
5397 */
5398 ++end;
5399 do
5400 {
5401 for (idx = SPO_COUNT; --idx >= 0; )
5402 if (STRNCMP(end, spo_name_tab[idx], 3) == 0)
5403 break;
5404 if (idx >= 0)
5405 {
5406 p = &(ci->sp_offsets[idx]);
5407 if (idx != SPO_LC_OFF)
5408 switch (end[3])
5409 {
5410 case 's': break;
5411 case 'b': break;
5412 case 'e': idx += SPO_COUNT; break;
5413 default: idx = -1; break;
5414 }
5415 if (idx >= 0)
5416 {
5417 ci->sp_off_flags |= (1 << idx);
5418 if (idx == SPO_LC_OFF) /* lc=99 */
5419 {
5420 end += 3;
5421 *p = getdigits(&end);
5422
5423 /* "lc=" offset automatically sets "ms=" offset */
5424 if (!(ci->sp_off_flags & (1 << SPO_MS_OFF)))
5425 {
5426 ci->sp_off_flags |= (1 << SPO_MS_OFF);
5427 ci->sp_offsets[SPO_MS_OFF] = *p;
5428 }
5429 }
5430 else /* yy=x+99 */
5431 {
5432 end += 4;
5433 if (*end == '+')
5434 {
5435 ++end;
5436 *p = getdigits(&end); /* positive offset */
5437 }
5438 else if (*end == '-')
5439 {
5440 ++end;
5441 *p = -getdigits(&end); /* negative offset */
5442 }
5443 }
5444 if (*end != ',')
5445 break;
5446 ++end;
5447 }
5448 }
5449 } while (idx >= 0);
5450
5451 if (!ends_excmd(*end) && !vim_iswhite(*end))
5452 {
5453 EMSG2(_("E402: Garbage after pattern: %s"), arg);
5454 return NULL;
5455 }
5456 return skipwhite(end);
5457}
5458
5459/*
5460 * Handle ":syntax sync .." command.
5461 */
Bram Moolenaar071d4272004-06-13 20:20:40 +00005462 static void
5463syn_cmd_sync(eap, syncing)
5464 exarg_T *eap;
Bram Moolenaar2c4278f2009-05-17 11:33:22 +00005465 int syncing UNUSED;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005466{
5467 char_u *arg_start = eap->arg;
5468 char_u *arg_end;
5469 char_u *key = NULL;
5470 char_u *next_arg;
5471 int illegal = FALSE;
5472 int finished = FALSE;
5473 long n;
5474 char_u *cpo_save;
5475
5476 if (ends_excmd(*arg_start))
5477 {
5478 syn_cmd_list(eap, TRUE);
5479 return;
5480 }
5481
5482 while (!ends_excmd(*arg_start))
5483 {
5484 arg_end = skiptowhite(arg_start);
5485 next_arg = skipwhite(arg_end);
5486 vim_free(key);
5487 key = vim_strnsave_up(arg_start, (int)(arg_end - arg_start));
5488 if (STRCMP(key, "CCOMMENT") == 0)
5489 {
5490 if (!eap->skip)
5491 curbuf->b_syn_sync_flags |= SF_CCOMMENT;
5492 if (!ends_excmd(*next_arg))
5493 {
5494 arg_end = skiptowhite(next_arg);
5495 if (!eap->skip)
5496 curbuf->b_syn_sync_id = syn_check_group(next_arg,
5497 (int)(arg_end - next_arg));
5498 next_arg = skipwhite(arg_end);
5499 }
5500 else if (!eap->skip)
5501 curbuf->b_syn_sync_id = syn_name2id((char_u *)"Comment");
5502 }
5503 else if ( STRNCMP(key, "LINES", 5) == 0
5504 || STRNCMP(key, "MINLINES", 8) == 0
5505 || STRNCMP(key, "MAXLINES", 8) == 0
5506 || STRNCMP(key, "LINEBREAKS", 10) == 0)
5507 {
5508 if (key[4] == 'S')
5509 arg_end = key + 6;
5510 else if (key[0] == 'L')
5511 arg_end = key + 11;
5512 else
5513 arg_end = key + 9;
5514 if (arg_end[-1] != '=' || !VIM_ISDIGIT(*arg_end))
5515 {
5516 illegal = TRUE;
5517 break;
5518 }
5519 n = getdigits(&arg_end);
5520 if (!eap->skip)
5521 {
5522 if (key[4] == 'B')
5523 curbuf->b_syn_sync_linebreaks = n;
5524 else if (key[1] == 'A')
5525 curbuf->b_syn_sync_maxlines = n;
5526 else
5527 curbuf->b_syn_sync_minlines = n;
5528 }
5529 }
5530 else if (STRCMP(key, "FROMSTART") == 0)
5531 {
5532 if (!eap->skip)
5533 {
5534 curbuf->b_syn_sync_minlines = MAXLNUM;
5535 curbuf->b_syn_sync_maxlines = 0;
5536 }
5537 }
5538 else if (STRCMP(key, "LINECONT") == 0)
5539 {
5540 if (curbuf->b_syn_linecont_pat != NULL)
5541 {
5542 EMSG(_("E403: syntax sync: line continuations pattern specified twice"));
5543 finished = TRUE;
5544 break;
5545 }
5546 arg_end = skip_regexp(next_arg + 1, *next_arg, TRUE, NULL);
5547 if (*arg_end != *next_arg) /* end delimiter not found */
5548 {
5549 illegal = TRUE;
5550 break;
5551 }
5552
5553 if (!eap->skip)
5554 {
5555 /* store the pattern and compiled regexp program */
5556 if ((curbuf->b_syn_linecont_pat = vim_strnsave(next_arg + 1,
5557 (int)(arg_end - next_arg - 1))) == NULL)
5558 {
5559 finished = TRUE;
5560 break;
5561 }
5562 curbuf->b_syn_linecont_ic = curbuf->b_syn_ic;
5563
5564 /* Make 'cpoptions' empty, to avoid the 'l' flag */
5565 cpo_save = p_cpo;
5566 p_cpo = (char_u *)"";
5567 curbuf->b_syn_linecont_prog =
5568 vim_regcomp(curbuf->b_syn_linecont_pat, RE_MAGIC);
5569 p_cpo = cpo_save;
5570
5571 if (curbuf->b_syn_linecont_prog == NULL)
5572 {
5573 vim_free(curbuf->b_syn_linecont_pat);
5574 curbuf->b_syn_linecont_pat = NULL;
5575 finished = TRUE;
5576 break;
5577 }
5578 }
5579 next_arg = skipwhite(arg_end + 1);
5580 }
5581 else
5582 {
5583 eap->arg = next_arg;
5584 if (STRCMP(key, "MATCH") == 0)
5585 syn_cmd_match(eap, TRUE);
5586 else if (STRCMP(key, "REGION") == 0)
5587 syn_cmd_region(eap, TRUE);
5588 else if (STRCMP(key, "CLEAR") == 0)
5589 syn_cmd_clear(eap, TRUE);
5590 else
5591 illegal = TRUE;
5592 finished = TRUE;
5593 break;
5594 }
5595 arg_start = next_arg;
5596 }
5597 vim_free(key);
5598 if (illegal)
5599 EMSG2(_("E404: Illegal arguments: %s"), arg_start);
5600 else if (!finished)
5601 {
5602 eap->nextcmd = check_nextcmd(arg_start);
Bram Moolenaar1c8f93f2006-03-12 22:10:07 +00005603 redraw_curbuf_later(SOME_VALID);
Bram Moolenaar071d4272004-06-13 20:20:40 +00005604 syn_stack_free_all(curbuf); /* Need to recompute all syntax. */
5605 }
5606}
5607
5608/*
5609 * Convert a line of highlight group names into a list of group ID numbers.
5610 * "arg" should point to the "contains" or "nextgroup" keyword.
5611 * "arg" is advanced to after the last group name.
5612 * Careful: the argument is modified (NULs added).
5613 * returns FAIL for some error, OK for success.
5614 */
5615 static int
5616get_id_list(arg, keylen, list)
5617 char_u **arg;
5618 int keylen; /* length of keyword */
5619 short **list; /* where to store the resulting list, if not
5620 NULL, the list is silently skipped! */
5621{
5622 char_u *p = NULL;
5623 char_u *end;
5624 int round;
5625 int count;
5626 int total_count = 0;
5627 short *retval = NULL;
5628 char_u *name;
5629 regmatch_T regmatch;
5630 int id;
5631 int i;
5632 int failed = FALSE;
5633
5634 /*
5635 * We parse the list twice:
5636 * round == 1: count the number of items, allocate the array.
5637 * round == 2: fill the array with the items.
5638 * In round 1 new groups may be added, causing the number of items to
5639 * grow when a regexp is used. In that case round 1 is done once again.
5640 */
5641 for (round = 1; round <= 2; ++round)
5642 {
5643 /*
5644 * skip "contains"
5645 */
5646 p = skipwhite(*arg + keylen);
5647 if (*p != '=')
5648 {
5649 EMSG2(_("E405: Missing equal sign: %s"), *arg);
5650 break;
5651 }
5652 p = skipwhite(p + 1);
5653 if (ends_excmd(*p))
5654 {
5655 EMSG2(_("E406: Empty argument: %s"), *arg);
5656 break;
5657 }
5658
5659 /*
5660 * parse the arguments after "contains"
5661 */
5662 count = 0;
5663 while (!ends_excmd(*p))
5664 {
5665 for (end = p; *end && !vim_iswhite(*end) && *end != ','; ++end)
5666 ;
5667 name = alloc((int)(end - p + 3)); /* leave room for "^$" */
5668 if (name == NULL)
5669 {
5670 failed = TRUE;
5671 break;
5672 }
Bram Moolenaarce0842a2005-07-18 21:58:11 +00005673 vim_strncpy(name + 1, p, end - p);
Bram Moolenaar071d4272004-06-13 20:20:40 +00005674 if ( STRCMP(name + 1, "ALLBUT") == 0
5675 || STRCMP(name + 1, "ALL") == 0
5676 || STRCMP(name + 1, "TOP") == 0
5677 || STRCMP(name + 1, "CONTAINED") == 0)
5678 {
5679 if (TOUPPER_ASC(**arg) != 'C')
5680 {
5681 EMSG2(_("E407: %s not allowed here"), name + 1);
5682 failed = TRUE;
5683 vim_free(name);
5684 break;
5685 }
5686 if (count != 0)
5687 {
5688 EMSG2(_("E408: %s must be first in contains list"), name + 1);
5689 failed = TRUE;
5690 vim_free(name);
5691 break;
5692 }
5693 if (name[1] == 'A')
5694 id = SYNID_ALLBUT;
5695 else if (name[1] == 'T')
5696 id = SYNID_TOP;
5697 else
5698 id = SYNID_CONTAINED;
5699 id += current_syn_inc_tag;
5700 }
5701 else if (name[1] == '@')
5702 {
5703 id = syn_check_cluster(name + 2, (int)(end - p - 1));
5704 }
5705 else
5706 {
5707 /*
5708 * Handle full group name.
5709 */
5710 if (vim_strpbrk(name + 1, (char_u *)"\\.*^$~[") == NULL)
5711 id = syn_check_group(name + 1, (int)(end - p));
5712 else
5713 {
5714 /*
5715 * Handle match of regexp with group names.
5716 */
5717 *name = '^';
5718 STRCAT(name, "$");
5719 regmatch.regprog = vim_regcomp(name, RE_MAGIC);
5720 if (regmatch.regprog == NULL)
5721 {
5722 failed = TRUE;
5723 vim_free(name);
5724 break;
5725 }
5726
5727 regmatch.rm_ic = TRUE;
5728 id = 0;
5729 for (i = highlight_ga.ga_len; --i >= 0; )
5730 {
5731 if (vim_regexec(&regmatch, HL_TABLE()[i].sg_name,
5732 (colnr_T)0))
5733 {
5734 if (round == 2)
5735 {
5736 /* Got more items than expected; can happen
5737 * when adding items that match:
5738 * "contains=a.*b,axb".
5739 * Go back to first round */
5740 if (count >= total_count)
5741 {
5742 vim_free(retval);
5743 round = 1;
5744 }
5745 else
5746 retval[count] = i + 1;
5747 }
5748 ++count;
5749 id = -1; /* remember that we found one */
5750 }
5751 }
5752 vim_free(regmatch.regprog);
5753 }
5754 }
5755 vim_free(name);
5756 if (id == 0)
5757 {
5758 EMSG2(_("E409: Unknown group name: %s"), p);
5759 failed = TRUE;
5760 break;
5761 }
5762 if (id > 0)
5763 {
5764 if (round == 2)
5765 {
5766 /* Got more items than expected, go back to first round */
5767 if (count >= total_count)
5768 {
5769 vim_free(retval);
5770 round = 1;
5771 }
5772 else
5773 retval[count] = id;
5774 }
5775 ++count;
5776 }
5777 p = skipwhite(end);
5778 if (*p != ',')
5779 break;
5780 p = skipwhite(p + 1); /* skip comma in between arguments */
5781 }
5782 if (failed)
5783 break;
5784 if (round == 1)
5785 {
5786 retval = (short *)alloc((unsigned)((count + 1) * sizeof(short)));
5787 if (retval == NULL)
5788 break;
5789 retval[count] = 0; /* zero means end of the list */
5790 total_count = count;
5791 }
5792 }
5793
5794 *arg = p;
5795 if (failed || retval == NULL)
5796 {
5797 vim_free(retval);
5798 return FAIL;
5799 }
5800
5801 if (*list == NULL)
5802 *list = retval;
5803 else
5804 vim_free(retval); /* list already found, don't overwrite it */
5805
5806 return OK;
5807}
5808
5809/*
5810 * Make a copy of an ID list.
5811 */
5812 static short *
5813copy_id_list(list)
5814 short *list;
5815{
5816 int len;
5817 int count;
5818 short *retval;
5819
5820 if (list == NULL)
5821 return NULL;
5822
5823 for (count = 0; list[count]; ++count)
5824 ;
5825 len = (count + 1) * sizeof(short);
5826 retval = (short *)alloc((unsigned)len);
5827 if (retval != NULL)
5828 mch_memmove(retval, list, (size_t)len);
5829
5830 return retval;
5831}
5832
5833/*
5834 * Check if syntax group "ssp" is in the ID list "list" of "cur_si".
5835 * "cur_si" can be NULL if not checking the "containedin" list.
5836 * Used to check if a syntax item is in the "contains" or "nextgroup" list of
5837 * the current item.
5838 * This function is called very often, keep it fast!!
5839 */
5840 static int
5841in_id_list(cur_si, list, ssp, contained)
5842 stateitem_T *cur_si; /* current item or NULL */
5843 short *list; /* id list */
5844 struct sp_syn *ssp; /* group id and ":syn include" tag of group */
5845 int contained; /* group id is contained */
5846{
5847 int retval;
5848 short *scl_list;
5849 short item;
5850 short id = ssp->id;
5851 static int depth = 0;
5852 int r;
5853
5854 /* If spp has a "containedin" list and "cur_si" is in it, return TRUE. */
Bram Moolenaar293ee4d2004-12-09 21:34:53 +00005855 if (cur_si != NULL && ssp->cont_in_list != NULL
5856 && !(cur_si->si_flags & HL_MATCH))
Bram Moolenaar071d4272004-06-13 20:20:40 +00005857 {
5858 /* Ignore transparent items without a contains argument. Double check
5859 * that we don't go back past the first one. */
5860 while ((cur_si->si_flags & HL_TRANS_CONT)
5861 && cur_si > (stateitem_T *)(current_state.ga_data))
5862 --cur_si;
5863 /* cur_si->si_idx is -1 for keywords, these never contain anything. */
5864 if (cur_si->si_idx >= 0 && in_id_list(NULL, ssp->cont_in_list,
5865 &(SYN_ITEMS(syn_buf)[cur_si->si_idx].sp_syn),
5866 SYN_ITEMS(syn_buf)[cur_si->si_idx].sp_flags & HL_CONTAINED))
5867 return TRUE;
5868 }
5869
5870 if (list == NULL)
5871 return FALSE;
5872
5873 /*
5874 * If list is ID_LIST_ALL, we are in a transparent item that isn't
5875 * inside anything. Only allow not-contained groups.
5876 */
5877 if (list == ID_LIST_ALL)
5878 return !contained;
5879
5880 /*
5881 * If the first item is "ALLBUT", return TRUE if "id" is NOT in the
5882 * contains list. We also require that "id" is at the same ":syn include"
5883 * level as the list.
5884 */
5885 item = *list;
5886 if (item >= SYNID_ALLBUT && item < SYNID_CLUSTER)
5887 {
5888 if (item < SYNID_TOP)
5889 {
5890 /* ALL or ALLBUT: accept all groups in the same file */
5891 if (item - SYNID_ALLBUT != ssp->inc_tag)
5892 return FALSE;
5893 }
5894 else if (item < SYNID_CONTAINED)
5895 {
5896 /* TOP: accept all not-contained groups in the same file */
5897 if (item - SYNID_TOP != ssp->inc_tag || contained)
5898 return FALSE;
5899 }
5900 else
5901 {
5902 /* CONTAINED: accept all contained groups in the same file */
5903 if (item - SYNID_CONTAINED != ssp->inc_tag || !contained)
5904 return FALSE;
5905 }
5906 item = *++list;
5907 retval = FALSE;
5908 }
5909 else
5910 retval = TRUE;
5911
5912 /*
5913 * Return "retval" if id is in the contains list.
5914 */
5915 while (item != 0)
5916 {
5917 if (item == id)
5918 return retval;
5919 if (item >= SYNID_CLUSTER)
5920 {
5921 scl_list = SYN_CLSTR(syn_buf)[item - SYNID_CLUSTER].scl_list;
5922 /* restrict recursiveness to 30 to avoid an endless loop for a
5923 * cluster that includes itself (indirectly) */
5924 if (scl_list != NULL && depth < 30)
5925 {
5926 ++depth;
5927 r = in_id_list(NULL, scl_list, ssp, contained);
5928 --depth;
5929 if (r)
5930 return retval;
5931 }
5932 }
5933 item = *++list;
5934 }
5935 return !retval;
5936}
5937
5938struct subcommand
5939{
5940 char *name; /* subcommand name */
5941 void (*func)__ARGS((exarg_T *, int)); /* function to call */
5942};
5943
5944static struct subcommand subcommands[] =
5945{
5946 {"case", syn_cmd_case},
5947 {"clear", syn_cmd_clear},
5948 {"cluster", syn_cmd_cluster},
5949 {"enable", syn_cmd_enable},
5950 {"include", syn_cmd_include},
5951 {"keyword", syn_cmd_keyword},
5952 {"list", syn_cmd_list},
5953 {"manual", syn_cmd_manual},
5954 {"match", syn_cmd_match},
5955 {"on", syn_cmd_on},
5956 {"off", syn_cmd_off},
5957 {"region", syn_cmd_region},
5958 {"reset", syn_cmd_reset},
Bram Moolenaarce0842a2005-07-18 21:58:11 +00005959 {"spell", syn_cmd_spell},
Bram Moolenaar071d4272004-06-13 20:20:40 +00005960 {"sync", syn_cmd_sync},
5961 {"", syn_cmd_list},
5962 {NULL, NULL}
5963};
5964
5965/*
5966 * ":syntax".
5967 * This searches the subcommands[] table for the subcommand name, and calls a
5968 * syntax_subcommand() function to do the rest.
5969 */
5970 void
5971ex_syntax(eap)
5972 exarg_T *eap;
5973{
5974 char_u *arg = eap->arg;
5975 char_u *subcmd_end;
5976 char_u *subcmd_name;
5977 int i;
5978
5979 syn_cmdlinep = eap->cmdlinep;
5980
5981 /* isolate subcommand name */
5982 for (subcmd_end = arg; ASCII_ISALPHA(*subcmd_end); ++subcmd_end)
5983 ;
5984 subcmd_name = vim_strnsave(arg, (int)(subcmd_end - arg));
5985 if (subcmd_name != NULL)
5986 {
5987 if (eap->skip) /* skip error messages for all subcommands */
5988 ++emsg_skip;
5989 for (i = 0; ; ++i)
5990 {
5991 if (subcommands[i].name == NULL)
5992 {
5993 EMSG2(_("E410: Invalid :syntax subcommand: %s"), subcmd_name);
5994 break;
5995 }
5996 if (STRCMP(subcmd_name, (char_u *)subcommands[i].name) == 0)
5997 {
5998 eap->arg = skipwhite(subcmd_end);
5999 (subcommands[i].func)(eap, FALSE);
6000 break;
6001 }
6002 }
6003 vim_free(subcmd_name);
6004 if (eap->skip)
6005 --emsg_skip;
6006 }
6007}
6008
6009 int
6010syntax_present(buf)
6011 buf_T *buf;
6012{
6013 return (buf->b_syn_patterns.ga_len != 0
6014 || buf->b_syn_clusters.ga_len != 0
Bram Moolenaar102e3a62007-08-30 17:37:40 +00006015 || buf->b_keywtab.ht_used > 0
6016 || buf->b_keywtab_ic.ht_used > 0);
Bram Moolenaar071d4272004-06-13 20:20:40 +00006017}
6018
6019#if defined(FEAT_CMDL_COMPL) || defined(PROTO)
6020
6021static enum
6022{
6023 EXP_SUBCMD, /* expand ":syn" sub-commands */
6024 EXP_CASE /* expand ":syn case" arguments */
6025} expand_what;
6026
Bram Moolenaar4f688582007-07-24 12:34:30 +00006027/*
6028 * Reset include_link, include_default, include_none to 0.
6029 * Called when we are done expanding.
6030 */
6031 void
6032reset_expand_highlight()
6033{
6034 include_link = include_default = include_none = 0;
6035}
6036
6037/*
6038 * Handle command line completion for :match and :echohl command: Add "None"
6039 * as highlight group.
6040 */
6041 void
6042set_context_in_echohl_cmd(xp, arg)
6043 expand_T *xp;
6044 char_u *arg;
6045{
6046 xp->xp_context = EXPAND_HIGHLIGHT;
6047 xp->xp_pattern = arg;
6048 include_none = 1;
6049}
Bram Moolenaar071d4272004-06-13 20:20:40 +00006050
6051/*
6052 * Handle command line completion for :syntax command.
6053 */
6054 void
6055set_context_in_syntax_cmd(xp, arg)
6056 expand_T *xp;
6057 char_u *arg;
6058{
6059 char_u *p;
6060
6061 /* Default: expand subcommands */
6062 xp->xp_context = EXPAND_SYNTAX;
6063 expand_what = EXP_SUBCMD;
6064 xp->xp_pattern = arg;
Bram Moolenaar4f688582007-07-24 12:34:30 +00006065 include_link = 0;
6066 include_default = 0;
Bram Moolenaar071d4272004-06-13 20:20:40 +00006067
6068 /* (part of) subcommand already typed */
6069 if (*arg != NUL)
6070 {
6071 p = skiptowhite(arg);
6072 if (*p != NUL) /* past first word */
6073 {
6074 xp->xp_pattern = skipwhite(p);
6075 if (*skiptowhite(xp->xp_pattern) != NUL)
6076 xp->xp_context = EXPAND_NOTHING;
6077 else if (STRNICMP(arg, "case", p - arg) == 0)
6078 expand_what = EXP_CASE;
6079 else if ( STRNICMP(arg, "keyword", p - arg) == 0
6080 || STRNICMP(arg, "region", p - arg) == 0
6081 || STRNICMP(arg, "match", p - arg) == 0
6082 || STRNICMP(arg, "list", p - arg) == 0)
6083 xp->xp_context = EXPAND_HIGHLIGHT;
6084 else
6085 xp->xp_context = EXPAND_NOTHING;
6086 }
6087 }
6088}
6089
6090static char *(case_args[]) = {"match", "ignore", NULL};
6091
6092/*
6093 * Function given to ExpandGeneric() to obtain the list syntax names for
6094 * expansion.
6095 */
Bram Moolenaar071d4272004-06-13 20:20:40 +00006096 char_u *
6097get_syntax_name(xp, idx)
Bram Moolenaar2c4278f2009-05-17 11:33:22 +00006098 expand_T *xp UNUSED;
Bram Moolenaar071d4272004-06-13 20:20:40 +00006099 int idx;
6100{
6101 if (expand_what == EXP_SUBCMD)
6102 return (char_u *)subcommands[idx].name;
6103 return (char_u *)case_args[idx];
6104}
6105
6106#endif /* FEAT_CMDL_COMPL */
6107
Bram Moolenaar071d4272004-06-13 20:20:40 +00006108/*
6109 * Function called for expression evaluation: get syntax ID at file position.
6110 */
6111 int
Bram Moolenaar56cefaf2008-01-12 15:47:10 +00006112syn_get_id(wp, lnum, col, trans, spellp, keep_state)
Bram Moolenaar81f1ecb2005-08-25 21:27:31 +00006113 win_T *wp;
Bram Moolenaar071d4272004-06-13 20:20:40 +00006114 long lnum;
Bram Moolenaar9d0ec2e2005-04-20 19:45:58 +00006115 colnr_T col;
Bram Moolenaarf5b63862009-12-16 17:13:44 +00006116 int trans; /* remove transparency */
Bram Moolenaar56cefaf2008-01-12 15:47:10 +00006117 int *spellp; /* return: can do spell checking */
6118 int keep_state; /* keep state of char at "col" */
Bram Moolenaar071d4272004-06-13 20:20:40 +00006119{
6120 /* When the position is not after the current position and in the same
6121 * line of the same buffer, need to restart parsing. */
Bram Moolenaar81f1ecb2005-08-25 21:27:31 +00006122 if (wp->w_buffer != syn_buf
Bram Moolenaar071d4272004-06-13 20:20:40 +00006123 || lnum != current_lnum
Bram Moolenaar9d0ec2e2005-04-20 19:45:58 +00006124 || col < current_col)
Bram Moolenaar81f1ecb2005-08-25 21:27:31 +00006125 syntax_start(wp, lnum);
Bram Moolenaar071d4272004-06-13 20:20:40 +00006126
Bram Moolenaar56cefaf2008-01-12 15:47:10 +00006127 (void)get_syntax_attr(col, spellp, keep_state);
Bram Moolenaar071d4272004-06-13 20:20:40 +00006128
6129 return (trans ? current_trans_id : current_id);
6130}
Bram Moolenaar071d4272004-06-13 20:20:40 +00006131
Bram Moolenaar9d188ab2008-01-10 21:24:39 +00006132#if defined(FEAT_EVAL) || defined(PROTO)
6133/*
6134 * Return the syntax ID at position "i" in the current stack.
6135 * The caller must have called syn_get_id() before to fill the stack.
6136 * Returns -1 when "i" is out of range.
6137 */
6138 int
6139syn_get_stack_item(i)
6140 int i;
6141{
Bram Moolenaar56cefaf2008-01-12 15:47:10 +00006142 if (i >= current_state.ga_len)
6143 {
6144 /* Need to invalidate the state, because we didn't properly finish it
6145 * for the last character, "keep_state" was TRUE. */
6146 invalidate_current_state();
6147 current_col = MAXCOL;
Bram Moolenaar9d188ab2008-01-10 21:24:39 +00006148 return -1;
Bram Moolenaar56cefaf2008-01-12 15:47:10 +00006149 }
Bram Moolenaar9d188ab2008-01-10 21:24:39 +00006150 return CUR_STATE(i).si_id;
6151}
6152#endif
6153
Bram Moolenaar071d4272004-06-13 20:20:40 +00006154#if defined(FEAT_FOLDING) || defined(PROTO)
6155/*
6156 * Function called to get folding level for line "lnum" in window "wp".
6157 */
6158 int
6159syn_get_foldlevel(wp, lnum)
6160 win_T *wp;
6161 long lnum;
6162{
6163 int level = 0;
6164 int i;
6165
6166 /* Return quickly when there are no fold items at all. */
6167 if (wp->w_buffer->b_syn_folditems != 0)
6168 {
6169 syntax_start(wp, lnum);
6170
6171 for (i = 0; i < current_state.ga_len; ++i)
6172 if (CUR_STATE(i).si_flags & HL_FOLD)
6173 ++level;
6174 }
6175 if (level > wp->w_p_fdn)
Bram Moolenaar74c596b2006-11-01 11:44:31 +00006176 {
Bram Moolenaar071d4272004-06-13 20:20:40 +00006177 level = wp->w_p_fdn;
Bram Moolenaar74c596b2006-11-01 11:44:31 +00006178 if (level < 0)
6179 level = 0;
6180 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00006181 return level;
6182}
6183#endif
6184
6185#endif /* FEAT_SYN_HL */
6186
6187
6188/**************************************
6189 * Highlighting stuff *
6190 **************************************/
6191
6192/*
6193 * The default highlight groups. These are compiled-in for fast startup and
6194 * they still work when the runtime files can't be found.
6195 * When making changes here, also change runtime/colors/default.vim!
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006196 * The #ifdefs are needed to reduce the amount of static data. Helps to make
6197 * the 16 bit DOS (museum) version compile.
Bram Moolenaar071d4272004-06-13 20:20:40 +00006198 */
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006199#ifdef FEAT_GUI
6200# define CENT(a, b) b
6201#else
6202# define CENT(a, b) a
6203#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00006204static char *(highlight_init_both[]) =
6205 {
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006206 CENT("ErrorMsg term=standout ctermbg=DarkRed ctermfg=White",
6207 "ErrorMsg term=standout ctermbg=DarkRed ctermfg=White guibg=Red guifg=White"),
Bram Moolenaarda65f152007-05-06 13:54:35 +00006208#ifdef FEAT_SEARCH_EXTRA
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006209 CENT("IncSearch term=reverse cterm=reverse",
6210 "IncSearch term=reverse cterm=reverse gui=reverse"),
Bram Moolenaarda65f152007-05-06 13:54:35 +00006211#endif
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006212 CENT("ModeMsg term=bold cterm=bold",
6213 "ModeMsg term=bold cterm=bold gui=bold"),
6214 CENT("NonText term=bold ctermfg=Blue",
6215 "NonText term=bold ctermfg=Blue gui=bold guifg=Blue"),
6216 CENT("StatusLine term=reverse,bold cterm=reverse,bold",
6217 "StatusLine term=reverse,bold cterm=reverse,bold gui=reverse,bold"),
6218 CENT("StatusLineNC term=reverse cterm=reverse",
6219 "StatusLineNC term=reverse cterm=reverse gui=reverse"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006220#ifdef FEAT_VERTSPLIT
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006221 CENT("VertSplit term=reverse cterm=reverse",
6222 "VertSplit term=reverse cterm=reverse gui=reverse"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006223#endif
6224#ifdef FEAT_CLIPBOARD
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006225 CENT("VisualNOS term=underline,bold cterm=underline,bold",
6226 "VisualNOS term=underline,bold cterm=underline,bold gui=underline,bold"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006227#endif
6228#ifdef FEAT_DIFF
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006229 CENT("DiffText term=reverse cterm=bold ctermbg=Red",
6230 "DiffText term=reverse cterm=bold ctermbg=Red gui=bold guibg=Red"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006231#endif
6232#ifdef FEAT_INS_EXPAND
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006233 CENT("PmenuThumb cterm=reverse",
6234 "PmenuThumb cterm=reverse gui=reverse"),
6235 CENT("PmenuSbar ctermbg=Grey",
6236 "PmenuSbar ctermbg=Grey guibg=Grey"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006237#endif
6238#ifdef FEAT_WINDOWS
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006239 CENT("TabLineSel term=bold cterm=bold",
6240 "TabLineSel term=bold cterm=bold gui=bold"),
6241 CENT("TabLineFill term=reverse cterm=reverse",
6242 "TabLineFill term=reverse cterm=reverse gui=reverse"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006243#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00006244#ifdef FEAT_GUI
6245 "Cursor guibg=fg guifg=bg",
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006246 "lCursor guibg=fg guifg=bg", /* should be different, but what? */
Bram Moolenaar071d4272004-06-13 20:20:40 +00006247#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00006248 NULL
6249 };
6250
6251static char *(highlight_init_light[]) =
6252 {
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006253 CENT("Directory term=bold ctermfg=DarkBlue",
6254 "Directory term=bold ctermfg=DarkBlue guifg=Blue"),
6255 CENT("LineNr term=underline ctermfg=Brown",
6256 "LineNr term=underline ctermfg=Brown guifg=Brown"),
6257 CENT("MoreMsg term=bold ctermfg=DarkGreen",
6258 "MoreMsg term=bold ctermfg=DarkGreen gui=bold guifg=SeaGreen"),
6259 CENT("Question term=standout ctermfg=DarkGreen",
6260 "Question term=standout ctermfg=DarkGreen gui=bold guifg=SeaGreen"),
6261 CENT("Search term=reverse ctermbg=Yellow ctermfg=NONE",
6262 "Search term=reverse ctermbg=Yellow ctermfg=NONE guibg=Yellow guifg=NONE"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006263#ifdef FEAT_SPELL
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006264 CENT("SpellBad term=reverse ctermbg=LightRed",
6265 "SpellBad term=reverse ctermbg=LightRed guisp=Red gui=undercurl"),
6266 CENT("SpellCap term=reverse ctermbg=LightBlue",
6267 "SpellCap term=reverse ctermbg=LightBlue guisp=Blue gui=undercurl"),
6268 CENT("SpellRare term=reverse ctermbg=LightMagenta",
6269 "SpellRare term=reverse ctermbg=LightMagenta guisp=Magenta gui=undercurl"),
6270 CENT("SpellLocal term=underline ctermbg=Cyan",
6271 "SpellLocal term=underline ctermbg=Cyan guisp=DarkCyan gui=undercurl"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006272#endif
6273#ifdef FEAT_INS_EXPAND
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006274 CENT("Pmenu ctermbg=LightMagenta",
6275 "Pmenu ctermbg=LightMagenta guibg=LightMagenta"),
6276 CENT("PmenuSel ctermbg=LightGrey",
6277 "PmenuSel ctermbg=LightGrey guibg=Grey"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006278#endif
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006279 CENT("SpecialKey term=bold ctermfg=DarkBlue",
6280 "SpecialKey term=bold ctermfg=DarkBlue guifg=Blue"),
6281 CENT("Title term=bold ctermfg=DarkMagenta",
6282 "Title term=bold ctermfg=DarkMagenta gui=bold guifg=Magenta"),
6283 CENT("WarningMsg term=standout ctermfg=DarkRed",
6284 "WarningMsg term=standout ctermfg=DarkRed guifg=Red"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006285#ifdef FEAT_WILDMENU
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006286 CENT("WildMenu term=standout ctermbg=Yellow ctermfg=Black",
6287 "WildMenu term=standout ctermbg=Yellow ctermfg=Black guibg=Yellow guifg=Black"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006288#endif
6289#ifdef FEAT_FOLDING
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006290 CENT("Folded term=standout ctermbg=Grey ctermfg=DarkBlue",
6291 "Folded term=standout ctermbg=Grey ctermfg=DarkBlue guibg=LightGrey guifg=DarkBlue"),
6292 CENT("FoldColumn term=standout ctermbg=Grey ctermfg=DarkBlue",
6293 "FoldColumn term=standout ctermbg=Grey ctermfg=DarkBlue guibg=Grey guifg=DarkBlue"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006294#endif
6295#ifdef FEAT_SIGNS
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006296 CENT("SignColumn term=standout ctermbg=Grey ctermfg=DarkBlue",
6297 "SignColumn term=standout ctermbg=Grey ctermfg=DarkBlue guibg=Grey guifg=DarkBlue"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006298#endif
6299#ifdef FEAT_VISUAL
Bram Moolenaarc1e37902006-04-18 21:55:01 +00006300 CENT("Visual term=reverse",
6301 "Visual term=reverse guibg=LightGrey"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006302#endif
6303#ifdef FEAT_DIFF
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006304 CENT("DiffAdd term=bold ctermbg=LightBlue",
6305 "DiffAdd term=bold ctermbg=LightBlue guibg=LightBlue"),
6306 CENT("DiffChange term=bold ctermbg=LightMagenta",
6307 "DiffChange term=bold ctermbg=LightMagenta guibg=LightMagenta"),
6308 CENT("DiffDelete term=bold ctermfg=Blue ctermbg=LightCyan",
6309 "DiffDelete term=bold ctermfg=Blue ctermbg=LightCyan gui=bold guifg=Blue guibg=LightCyan"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006310#endif
6311#ifdef FEAT_WINDOWS
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006312 CENT("TabLine term=underline cterm=underline ctermfg=black ctermbg=LightGrey",
6313 "TabLine term=underline cterm=underline ctermfg=black ctermbg=LightGrey gui=underline guibg=LightGrey"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006314#endif
6315#ifdef FEAT_SYN_HL
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006316 CENT("CursorColumn term=reverse ctermbg=LightGrey",
Bram Moolenaare2f98b92006-03-29 21:18:24 +00006317 "CursorColumn term=reverse ctermbg=LightGrey guibg=Grey90"),
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006318 CENT("CursorLine term=underline cterm=underline",
Bram Moolenaare2f98b92006-03-29 21:18:24 +00006319 "CursorLine term=underline cterm=underline guibg=Grey90"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006320#endif
Bram Moolenaarb21e5842006-04-16 18:30:08 +00006321#ifdef FEAT_AUTOCMD
6322 CENT("MatchParen term=reverse ctermbg=Cyan",
6323 "MatchParen term=reverse ctermbg=Cyan guibg=Cyan"),
6324#endif
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006325#ifdef FEAT_GUI
Bram Moolenaar071d4272004-06-13 20:20:40 +00006326 "Normal gui=NONE",
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006327#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00006328 NULL
6329 };
6330
6331static char *(highlight_init_dark[]) =
6332 {
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006333 CENT("Directory term=bold ctermfg=LightCyan",
6334 "Directory term=bold ctermfg=LightCyan guifg=Cyan"),
6335 CENT("LineNr term=underline ctermfg=Yellow",
6336 "LineNr term=underline ctermfg=Yellow guifg=Yellow"),
6337 CENT("MoreMsg term=bold ctermfg=LightGreen",
6338 "MoreMsg term=bold ctermfg=LightGreen gui=bold guifg=SeaGreen"),
6339 CENT("Question term=standout ctermfg=LightGreen",
6340 "Question term=standout ctermfg=LightGreen gui=bold guifg=Green"),
6341 CENT("Search term=reverse ctermbg=Yellow ctermfg=Black",
6342 "Search term=reverse ctermbg=Yellow ctermfg=Black guibg=Yellow guifg=Black"),
6343 CENT("SpecialKey term=bold ctermfg=LightBlue",
6344 "SpecialKey term=bold ctermfg=LightBlue guifg=Cyan"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006345#ifdef FEAT_SPELL
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006346 CENT("SpellBad term=reverse ctermbg=Red",
6347 "SpellBad term=reverse ctermbg=Red guisp=Red gui=undercurl"),
6348 CENT("SpellCap term=reverse ctermbg=Blue",
6349 "SpellCap term=reverse ctermbg=Blue guisp=Blue gui=undercurl"),
6350 CENT("SpellRare term=reverse ctermbg=Magenta",
6351 "SpellRare term=reverse ctermbg=Magenta guisp=Magenta gui=undercurl"),
6352 CENT("SpellLocal term=underline ctermbg=Cyan",
6353 "SpellLocal term=underline ctermbg=Cyan guisp=Cyan gui=undercurl"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006354#endif
6355#ifdef FEAT_INS_EXPAND
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006356 CENT("Pmenu ctermbg=Magenta",
6357 "Pmenu ctermbg=Magenta guibg=Magenta"),
6358 CENT("PmenuSel ctermbg=DarkGrey",
6359 "PmenuSel ctermbg=DarkGrey guibg=DarkGrey"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006360#endif
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006361 CENT("Title term=bold ctermfg=LightMagenta",
6362 "Title term=bold ctermfg=LightMagenta gui=bold guifg=Magenta"),
6363 CENT("WarningMsg term=standout ctermfg=LightRed",
6364 "WarningMsg term=standout ctermfg=LightRed guifg=Red"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006365#ifdef FEAT_WILDMENU
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006366 CENT("WildMenu term=standout ctermbg=Yellow ctermfg=Black",
6367 "WildMenu term=standout ctermbg=Yellow ctermfg=Black guibg=Yellow guifg=Black"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006368#endif
6369#ifdef FEAT_FOLDING
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006370 CENT("Folded term=standout ctermbg=DarkGrey ctermfg=Cyan",
6371 "Folded term=standout ctermbg=DarkGrey ctermfg=Cyan guibg=DarkGrey guifg=Cyan"),
6372 CENT("FoldColumn term=standout ctermbg=DarkGrey ctermfg=Cyan",
6373 "FoldColumn term=standout ctermbg=DarkGrey ctermfg=Cyan guibg=Grey guifg=Cyan"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006374#endif
6375#ifdef FEAT_SIGNS
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006376 CENT("SignColumn term=standout ctermbg=DarkGrey ctermfg=Cyan",
6377 "SignColumn term=standout ctermbg=DarkGrey ctermfg=Cyan guibg=Grey guifg=Cyan"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006378#endif
6379#ifdef FEAT_VISUAL
Bram Moolenaarc1e37902006-04-18 21:55:01 +00006380 CENT("Visual term=reverse",
6381 "Visual term=reverse guibg=DarkGrey"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006382#endif
6383#ifdef FEAT_DIFF
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006384 CENT("DiffAdd term=bold ctermbg=DarkBlue",
6385 "DiffAdd term=bold ctermbg=DarkBlue guibg=DarkBlue"),
6386 CENT("DiffChange term=bold ctermbg=DarkMagenta",
6387 "DiffChange term=bold ctermbg=DarkMagenta guibg=DarkMagenta"),
6388 CENT("DiffDelete term=bold ctermfg=Blue ctermbg=DarkCyan",
6389 "DiffDelete term=bold ctermfg=Blue ctermbg=DarkCyan gui=bold guifg=Blue guibg=DarkCyan"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006390#endif
6391#ifdef FEAT_WINDOWS
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006392 CENT("TabLine term=underline cterm=underline ctermfg=white ctermbg=DarkGrey",
6393 "TabLine term=underline cterm=underline ctermfg=white ctermbg=DarkGrey gui=underline guibg=DarkGrey"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006394#endif
6395#ifdef FEAT_SYN_HL
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006396 CENT("CursorColumn term=reverse ctermbg=DarkGrey",
Bram Moolenaarb21e5842006-04-16 18:30:08 +00006397 "CursorColumn term=reverse ctermbg=DarkGrey guibg=Grey40"),
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006398 CENT("CursorLine term=underline cterm=underline",
Bram Moolenaarb21e5842006-04-16 18:30:08 +00006399 "CursorLine term=underline cterm=underline guibg=Grey40"),
6400#endif
6401#ifdef FEAT_AUTOCMD
6402 CENT("MatchParen term=reverse ctermbg=DarkCyan",
6403 "MatchParen term=reverse ctermbg=DarkCyan guibg=DarkCyan"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006404#endif
6405#ifdef FEAT_GUI
Bram Moolenaar071d4272004-06-13 20:20:40 +00006406 "Normal gui=NONE",
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006407#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00006408 NULL
6409 };
6410
6411 void
6412init_highlight(both, reset)
6413 int both; /* include groups where 'bg' doesn't matter */
6414 int reset; /* clear group first */
6415{
6416 int i;
6417 char **pp;
6418 static int had_both = FALSE;
6419#ifdef FEAT_EVAL
6420 char_u *p;
6421
6422 /*
6423 * Try finding the color scheme file. Used when a color file was loaded
6424 * and 'background' or 't_Co' is changed.
6425 */
6426 p = get_var_value((char_u *)"g:colors_name");
6427 if (p != NULL && load_colors(p) == OK)
6428 return;
6429#endif
6430
6431 /*
6432 * Didn't use a color file, use the compiled-in colors.
6433 */
6434 if (both)
6435 {
6436 had_both = TRUE;
6437 pp = highlight_init_both;
6438 for (i = 0; pp[i] != NULL; ++i)
6439 do_highlight((char_u *)pp[i], reset, TRUE);
6440 }
6441 else if (!had_both)
6442 /* Don't do anything before the call with both == TRUE from main().
6443 * Not everything has been setup then, and that call will overrule
6444 * everything anyway. */
6445 return;
6446
6447 if (*p_bg == 'l')
6448 pp = highlight_init_light;
6449 else
6450 pp = highlight_init_dark;
6451 for (i = 0; pp[i] != NULL; ++i)
6452 do_highlight((char_u *)pp[i], reset, TRUE);
6453
Bram Moolenaarc1e37902006-04-18 21:55:01 +00006454 /* Reverse looks ugly, but grey may not work for 8 colors. Thus let it
Bram Moolenaarf193fff2006-04-27 00:02:13 +00006455 * depend on the number of colors available.
6456 * With 8 colors brown is equal to yellow, need to use black for Search fg
Bram Moolenaarda40c852008-08-06 13:28:57 +00006457 * to avoid Statement highlighted text disappears.
6458 * Clear the attributes, needed when changing the t_Co value. */
Bram Moolenaarab194812005-09-14 21:40:12 +00006459 if (t_colors > 8)
Bram Moolenaarda40c852008-08-06 13:28:57 +00006460 do_highlight((char_u *)(*p_bg == 'l'
6461 ? "Visual cterm=NONE ctermbg=LightGrey"
6462 : "Visual cterm=NONE ctermbg=DarkGrey"), FALSE, TRUE);
Bram Moolenaarc1e37902006-04-18 21:55:01 +00006463 else
Bram Moolenaarf193fff2006-04-27 00:02:13 +00006464 {
Bram Moolenaarda40c852008-08-06 13:28:57 +00006465 do_highlight((char_u *)"Visual cterm=reverse ctermbg=NONE",
6466 FALSE, TRUE);
Bram Moolenaarf193fff2006-04-27 00:02:13 +00006467 if (*p_bg == 'l')
6468 do_highlight((char_u *)"Search ctermfg=black", FALSE, TRUE);
6469 }
Bram Moolenaarab194812005-09-14 21:40:12 +00006470
Bram Moolenaar071d4272004-06-13 20:20:40 +00006471#ifdef FEAT_SYN_HL
6472 /*
6473 * If syntax highlighting is enabled load the highlighting for it.
6474 */
6475 if (get_var_value((char_u *)"g:syntax_on") != NULL)
Bram Moolenaarc0197e22004-09-13 20:26:32 +00006476 {
6477 static int recursive = 0;
6478
6479 if (recursive >= 5)
6480 EMSG(_("E679: recursive loop loading syncolor.vim"));
6481 else
6482 {
6483 ++recursive;
Bram Moolenaar90cfdbe2005-08-12 19:59:19 +00006484 (void)source_runtime((char_u *)"syntax/syncolor.vim", TRUE);
Bram Moolenaarc0197e22004-09-13 20:26:32 +00006485 --recursive;
6486 }
6487 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00006488#endif
6489}
6490
6491/*
Bram Moolenaarcfbc5ee2004-07-02 15:38:35 +00006492 * Load color file "name".
Bram Moolenaar071d4272004-06-13 20:20:40 +00006493 * Return OK for success, FAIL for failure.
6494 */
6495 int
Bram Moolenaarcfbc5ee2004-07-02 15:38:35 +00006496load_colors(name)
6497 char_u *name;
Bram Moolenaar071d4272004-06-13 20:20:40 +00006498{
6499 char_u *buf;
6500 int retval = FAIL;
6501 static int recursive = FALSE;
6502
6503 /* When being called recursively, this is probably because setting
6504 * 'background' caused the highlighting to be reloaded. This means it is
6505 * working, thus we should return OK. */
6506 if (recursive)
6507 return OK;
6508
6509 recursive = TRUE;
Bram Moolenaarcfbc5ee2004-07-02 15:38:35 +00006510 buf = alloc((unsigned)(STRLEN(name) + 12));
Bram Moolenaar071d4272004-06-13 20:20:40 +00006511 if (buf != NULL)
6512 {
Bram Moolenaarcfbc5ee2004-07-02 15:38:35 +00006513 sprintf((char *)buf, "colors/%s.vim", name);
Bram Moolenaar90cfdbe2005-08-12 19:59:19 +00006514 retval = source_runtime(buf, FALSE);
Bram Moolenaar071d4272004-06-13 20:20:40 +00006515 vim_free(buf);
Bram Moolenaarcfbc5ee2004-07-02 15:38:35 +00006516#ifdef FEAT_AUTOCMD
6517 apply_autocmds(EVENT_COLORSCHEME, NULL, NULL, FALSE, curbuf);
6518#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00006519 }
6520 recursive = FALSE;
6521
6522 return retval;
6523}
6524
6525/*
6526 * Handle the ":highlight .." command.
6527 * When using ":hi clear" this is called recursively for each group with
6528 * "forceit" and "init" both TRUE.
6529 */
6530 void
6531do_highlight(line, forceit, init)
6532 char_u *line;
6533 int forceit;
6534 int init; /* TRUE when called for initializing */
6535{
6536 char_u *name_end;
6537 char_u *p;
6538 char_u *linep;
6539 char_u *key_start;
6540 char_u *arg_start;
6541 char_u *key = NULL, *arg = NULL;
6542 long i;
6543 int off;
6544 int len;
6545 int attr;
6546 int id;
6547 int idx;
6548 int dodefault = FALSE;
6549 int doclear = FALSE;
6550 int dolink = FALSE;
6551 int error = FALSE;
6552 int color;
6553 int is_normal_group = FALSE; /* "Normal" group */
6554#ifdef FEAT_GUI_X11
6555 int is_menu_group = FALSE; /* "Menu" group */
6556 int is_scrollbar_group = FALSE; /* "Scrollbar" group */
6557 int is_tooltip_group = FALSE; /* "Tooltip" group */
6558 int do_colors = FALSE; /* need to update colors? */
6559#else
6560# define is_menu_group 0
6561# define is_tooltip_group 0
6562#endif
6563
6564 /*
6565 * If no argument, list current highlighting.
6566 */
6567 if (ends_excmd(*line))
6568 {
6569 for (i = 1; i <= highlight_ga.ga_len && !got_int; ++i)
6570 /* TODO: only call when the group has attributes set */
6571 highlight_list_one((int)i);
6572 return;
6573 }
6574
6575 /*
6576 * Isolate the name.
6577 */
6578 name_end = skiptowhite(line);
6579 linep = skipwhite(name_end);
6580
6581 /*
6582 * Check for "default" argument.
6583 */
6584 if (STRNCMP(line, "default", name_end - line) == 0)
6585 {
6586 dodefault = TRUE;
6587 line = linep;
6588 name_end = skiptowhite(line);
6589 linep = skipwhite(name_end);
6590 }
6591
6592 /*
6593 * Check for "clear" or "link" argument.
6594 */
6595 if (STRNCMP(line, "clear", name_end - line) == 0)
6596 doclear = TRUE;
6597 if (STRNCMP(line, "link", name_end - line) == 0)
6598 dolink = TRUE;
6599
6600 /*
6601 * ":highlight {group-name}": list highlighting for one group.
6602 */
6603 if (!doclear && !dolink && ends_excmd(*linep))
6604 {
6605 id = syn_namen2id(line, (int)(name_end - line));
6606 if (id == 0)
6607 EMSG2(_("E411: highlight group not found: %s"), line);
6608 else
6609 highlight_list_one(id);
6610 return;
6611 }
6612
6613 /*
6614 * Handle ":highlight link {from} {to}" command.
6615 */
6616 if (dolink)
6617 {
6618 char_u *from_start = linep;
6619 char_u *from_end;
6620 char_u *to_start;
6621 char_u *to_end;
6622 int from_id;
6623 int to_id;
6624
6625 from_end = skiptowhite(from_start);
6626 to_start = skipwhite(from_end);
6627 to_end = skiptowhite(to_start);
6628
6629 if (ends_excmd(*from_start) || ends_excmd(*to_start))
6630 {
6631 EMSG2(_("E412: Not enough arguments: \":highlight link %s\""),
6632 from_start);
6633 return;
6634 }
6635
6636 if (!ends_excmd(*skipwhite(to_end)))
6637 {
6638 EMSG2(_("E413: Too many arguments: \":highlight link %s\""), from_start);
6639 return;
6640 }
6641
6642 from_id = syn_check_group(from_start, (int)(from_end - from_start));
6643 if (STRNCMP(to_start, "NONE", 4) == 0)
6644 to_id = 0;
6645 else
6646 to_id = syn_check_group(to_start, (int)(to_end - to_start));
6647
6648 if (from_id > 0 && (!init || HL_TABLE()[from_id - 1].sg_set == 0))
6649 {
6650 /*
6651 * Don't allow a link when there already is some highlighting
6652 * for the group, unless '!' is used
6653 */
6654 if (to_id > 0 && !forceit && !init
6655 && hl_has_settings(from_id - 1, dodefault))
6656 {
6657 if (sourcing_name == NULL && !dodefault)
6658 EMSG(_("E414: group has settings, highlight link ignored"));
6659 }
6660 else
6661 {
6662 if (!init)
6663 HL_TABLE()[from_id - 1].sg_set |= SG_LINK;
6664 HL_TABLE()[from_id - 1].sg_link = to_id;
Bram Moolenaar661b1822005-07-28 22:36:45 +00006665#ifdef FEAT_EVAL
6666 HL_TABLE()[from_id - 1].sg_scriptID = current_SID;
6667#endif
Bram Moolenaar1c8f93f2006-03-12 22:10:07 +00006668 redraw_all_later(SOME_VALID);
Bram Moolenaar071d4272004-06-13 20:20:40 +00006669 }
6670 }
6671
6672 /* Only call highlight_changed() once, after sourcing a syntax file */
6673 need_highlight_changed = TRUE;
6674
6675 return;
6676 }
6677
6678 if (doclear)
6679 {
6680 /*
6681 * ":highlight clear [group]" command.
6682 */
6683 line = linep;
6684 if (ends_excmd(*line))
6685 {
6686#ifdef FEAT_GUI
6687 /* First, we do not destroy the old values, but allocate the new
6688 * ones and update the display. THEN we destroy the old values.
6689 * If we destroy the old values first, then the old values
6690 * (such as GuiFont's or GuiFontset's) will still be displayed but
6691 * invalid because they were free'd.
6692 */
6693 if (gui.in_use)
6694 {
6695# ifdef FEAT_BEVAL_TIP
6696 gui_init_tooltip_font();
6697# endif
6698# if defined(FEAT_MENU) && (defined(FEAT_GUI_ATHENA) || defined(FEAT_GUI_MOTIF))
6699 gui_init_menu_font();
6700# endif
6701 }
6702# if defined(FEAT_GUI_MSWIN) || defined(FEAT_GUI_X11)
6703 gui_mch_def_colors();
6704# endif
6705# ifdef FEAT_GUI_X11
6706# ifdef FEAT_MENU
6707
6708 /* This only needs to be done when there is no Menu highlight
6709 * group defined by default, which IS currently the case.
6710 */
6711 gui_mch_new_menu_colors();
6712# endif
6713 if (gui.in_use)
6714 {
6715 gui_new_scrollbar_colors();
6716# ifdef FEAT_BEVAL
6717 gui_mch_new_tooltip_colors();
6718# endif
6719# ifdef FEAT_MENU
6720 gui_mch_new_menu_font();
6721# endif
6722 }
6723# endif
6724
6725 /* Ok, we're done allocating the new default graphics items.
6726 * The screen should already be refreshed at this point.
6727 * It is now Ok to clear out the old data.
6728 */
6729#endif
6730#ifdef FEAT_EVAL
Bram Moolenaar2ce06f62005-01-31 19:19:04 +00006731 do_unlet((char_u *)"colors_name", TRUE);
Bram Moolenaar071d4272004-06-13 20:20:40 +00006732#endif
6733 restore_cterm_colors();
6734
6735 /*
6736 * Clear all default highlight groups and load the defaults.
6737 */
6738 for (idx = 0; idx < highlight_ga.ga_len; ++idx)
6739 highlight_clear(idx);
6740 init_highlight(TRUE, TRUE);
6741#ifdef FEAT_GUI
6742 if (gui.in_use)
6743 highlight_gui_started();
6744#endif
6745 highlight_changed();
6746 redraw_later_clear();
6747 return;
6748 }
6749 name_end = skiptowhite(line);
6750 linep = skipwhite(name_end);
6751 }
6752
6753 /*
6754 * Find the group name in the table. If it does not exist yet, add it.
6755 */
6756 id = syn_check_group(line, (int)(name_end - line));
6757 if (id == 0) /* failed (out of memory) */
6758 return;
6759 idx = id - 1; /* index is ID minus one */
6760
6761 /* Return if "default" was used and the group already has settings. */
6762 if (dodefault && hl_has_settings(idx, TRUE))
6763 return;
6764
6765 if (STRCMP(HL_TABLE()[idx].sg_name_u, "NORMAL") == 0)
6766 is_normal_group = TRUE;
6767#ifdef FEAT_GUI_X11
6768 else if (STRCMP(HL_TABLE()[idx].sg_name_u, "MENU") == 0)
6769 is_menu_group = TRUE;
6770 else if (STRCMP(HL_TABLE()[idx].sg_name_u, "SCROLLBAR") == 0)
6771 is_scrollbar_group = TRUE;
6772 else if (STRCMP(HL_TABLE()[idx].sg_name_u, "TOOLTIP") == 0)
6773 is_tooltip_group = TRUE;
6774#endif
6775
6776 /* Clear the highlighting for ":hi clear {group}" and ":hi clear". */
6777 if (doclear || (forceit && init))
6778 {
6779 highlight_clear(idx);
6780 if (!doclear)
6781 HL_TABLE()[idx].sg_set = 0;
6782 }
6783
6784 if (!doclear)
6785 while (!ends_excmd(*linep))
6786 {
6787 key_start = linep;
6788 if (*linep == '=')
6789 {
6790 EMSG2(_("E415: unexpected equal sign: %s"), key_start);
6791 error = TRUE;
6792 break;
6793 }
6794
6795 /*
6796 * Isolate the key ("term", "ctermfg", "ctermbg", "font", "guifg" or
6797 * "guibg").
6798 */
6799 while (*linep && !vim_iswhite(*linep) && *linep != '=')
6800 ++linep;
6801 vim_free(key);
6802 key = vim_strnsave_up(key_start, (int)(linep - key_start));
6803 if (key == NULL)
6804 {
6805 error = TRUE;
6806 break;
6807 }
6808 linep = skipwhite(linep);
6809
6810 if (STRCMP(key, "NONE") == 0)
6811 {
6812 if (!init || HL_TABLE()[idx].sg_set == 0)
6813 {
6814 if (!init)
6815 HL_TABLE()[idx].sg_set |= SG_TERM+SG_CTERM+SG_GUI;
6816 highlight_clear(idx);
6817 }
6818 continue;
6819 }
6820
6821 /*
6822 * Check for the equal sign.
6823 */
6824 if (*linep != '=')
6825 {
6826 EMSG2(_("E416: missing equal sign: %s"), key_start);
6827 error = TRUE;
6828 break;
6829 }
6830 ++linep;
6831
6832 /*
6833 * Isolate the argument.
6834 */
6835 linep = skipwhite(linep);
6836 if (*linep == '\'') /* guifg='color name' */
6837 {
6838 arg_start = ++linep;
6839 linep = vim_strchr(linep, '\'');
6840 if (linep == NULL)
6841 {
6842 EMSG2(_(e_invarg2), key_start);
6843 error = TRUE;
6844 break;
6845 }
6846 }
6847 else
6848 {
6849 arg_start = linep;
6850 linep = skiptowhite(linep);
6851 }
6852 if (linep == arg_start)
6853 {
6854 EMSG2(_("E417: missing argument: %s"), key_start);
6855 error = TRUE;
6856 break;
6857 }
6858 vim_free(arg);
6859 arg = vim_strnsave(arg_start, (int)(linep - arg_start));
6860 if (arg == NULL)
6861 {
6862 error = TRUE;
6863 break;
6864 }
6865 if (*linep == '\'')
6866 ++linep;
6867
6868 /*
6869 * Store the argument.
6870 */
6871 if ( STRCMP(key, "TERM") == 0
6872 || STRCMP(key, "CTERM") == 0
6873 || STRCMP(key, "GUI") == 0)
6874 {
6875 attr = 0;
6876 off = 0;
6877 while (arg[off] != NUL)
6878 {
6879 for (i = sizeof(hl_attr_table) / sizeof(int); --i >= 0; )
6880 {
6881 len = (int)STRLEN(hl_name_table[i]);
6882 if (STRNICMP(arg + off, hl_name_table[i], len) == 0)
6883 {
6884 attr |= hl_attr_table[i];
6885 off += len;
6886 break;
6887 }
6888 }
6889 if (i < 0)
6890 {
6891 EMSG2(_("E418: Illegal value: %s"), arg);
6892 error = TRUE;
6893 break;
6894 }
6895 if (arg[off] == ',') /* another one follows */
6896 ++off;
6897 }
6898 if (error)
6899 break;
6900 if (*key == 'T')
6901 {
6902 if (!init || !(HL_TABLE()[idx].sg_set & SG_TERM))
6903 {
6904 if (!init)
6905 HL_TABLE()[idx].sg_set |= SG_TERM;
6906 HL_TABLE()[idx].sg_term = attr;
6907 }
6908 }
6909 else if (*key == 'C')
6910 {
6911 if (!init || !(HL_TABLE()[idx].sg_set & SG_CTERM))
6912 {
6913 if (!init)
6914 HL_TABLE()[idx].sg_set |= SG_CTERM;
6915 HL_TABLE()[idx].sg_cterm = attr;
6916 HL_TABLE()[idx].sg_cterm_bold = FALSE;
6917 }
6918 }
6919#ifdef FEAT_GUI
6920 else
6921 {
6922 if (!init || !(HL_TABLE()[idx].sg_set & SG_GUI))
6923 {
6924 if (!init)
6925 HL_TABLE()[idx].sg_set |= SG_GUI;
6926 HL_TABLE()[idx].sg_gui = attr;
6927 }
6928 }
6929#endif
6930 }
6931 else if (STRCMP(key, "FONT") == 0)
6932 {
6933 /* in non-GUI fonts are simply ignored */
6934#ifdef FEAT_GUI
6935 if (!gui.shell_created)
6936 {
6937 /* GUI not started yet, always accept the name. */
6938 vim_free(HL_TABLE()[idx].sg_font_name);
6939 HL_TABLE()[idx].sg_font_name = vim_strsave(arg);
6940 }
6941 else
6942 {
6943 GuiFont temp_sg_font = HL_TABLE()[idx].sg_font;
6944# ifdef FEAT_XFONTSET
6945 GuiFontset temp_sg_fontset = HL_TABLE()[idx].sg_fontset;
6946# endif
6947 /* First, save the current font/fontset.
6948 * Then try to allocate the font/fontset.
6949 * If the allocation fails, HL_TABLE()[idx].sg_font OR
6950 * sg_fontset will be set to NOFONT or NOFONTSET respectively.
6951 */
6952
6953 HL_TABLE()[idx].sg_font = NOFONT;
6954# ifdef FEAT_XFONTSET
6955 HL_TABLE()[idx].sg_fontset = NOFONTSET;
6956# endif
6957 hl_do_font(idx, arg, is_normal_group, is_menu_group,
6958 is_tooltip_group);
6959
6960# ifdef FEAT_XFONTSET
6961 if (HL_TABLE()[idx].sg_fontset != NOFONTSET)
6962 {
6963 /* New fontset was accepted. Free the old one, if there was
6964 * one.
6965 */
6966 gui_mch_free_fontset(temp_sg_fontset);
6967 vim_free(HL_TABLE()[idx].sg_font_name);
6968 HL_TABLE()[idx].sg_font_name = vim_strsave(arg);
6969 }
6970 else
6971 HL_TABLE()[idx].sg_fontset = temp_sg_fontset;
6972# endif
6973 if (HL_TABLE()[idx].sg_font != NOFONT)
6974 {
6975 /* New font was accepted. Free the old one, if there was
6976 * one.
6977 */
6978 gui_mch_free_font(temp_sg_font);
6979 vim_free(HL_TABLE()[idx].sg_font_name);
6980 HL_TABLE()[idx].sg_font_name = vim_strsave(arg);
6981 }
6982 else
6983 HL_TABLE()[idx].sg_font = temp_sg_font;
6984 }
6985#endif
6986 }
6987 else if (STRCMP(key, "CTERMFG") == 0 || STRCMP(key, "CTERMBG") == 0)
6988 {
6989 if (!init || !(HL_TABLE()[idx].sg_set & SG_CTERM))
6990 {
6991 if (!init)
6992 HL_TABLE()[idx].sg_set |= SG_CTERM;
6993
6994 /* When setting the foreground color, and previously the "bold"
6995 * flag was set for a light color, reset it now */
6996 if (key[5] == 'F' && HL_TABLE()[idx].sg_cterm_bold)
6997 {
6998 HL_TABLE()[idx].sg_cterm &= ~HL_BOLD;
6999 HL_TABLE()[idx].sg_cterm_bold = FALSE;
7000 }
7001
7002 if (VIM_ISDIGIT(*arg))
7003 color = atoi((char *)arg);
7004 else if (STRICMP(arg, "fg") == 0)
7005 {
7006 if (cterm_normal_fg_color)
7007 color = cterm_normal_fg_color - 1;
7008 else
7009 {
7010 EMSG(_("E419: FG color unknown"));
7011 error = TRUE;
7012 break;
7013 }
7014 }
7015 else if (STRICMP(arg, "bg") == 0)
7016 {
7017 if (cterm_normal_bg_color > 0)
7018 color = cterm_normal_bg_color - 1;
7019 else
7020 {
7021 EMSG(_("E420: BG color unknown"));
7022 error = TRUE;
7023 break;
7024 }
7025 }
7026 else
7027 {
7028 static char *(color_names[28]) = {
7029 "Black", "DarkBlue", "DarkGreen", "DarkCyan",
7030 "DarkRed", "DarkMagenta", "Brown", "DarkYellow",
7031 "Gray", "Grey",
7032 "LightGray", "LightGrey", "DarkGray", "DarkGrey",
7033 "Blue", "LightBlue", "Green", "LightGreen",
7034 "Cyan", "LightCyan", "Red", "LightRed", "Magenta",
7035 "LightMagenta", "Yellow", "LightYellow", "White", "NONE"};
7036 static int color_numbers_16[28] = {0, 1, 2, 3,
7037 4, 5, 6, 6,
7038 7, 7,
7039 7, 7, 8, 8,
7040 9, 9, 10, 10,
7041 11, 11, 12, 12, 13,
7042 13, 14, 14, 15, -1};
7043 /* for xterm with 88 colors... */
7044 static int color_numbers_88[28] = {0, 4, 2, 6,
7045 1, 5, 32, 72,
7046 84, 84,
7047 7, 7, 82, 82,
7048 12, 43, 10, 61,
7049 14, 63, 9, 74, 13,
7050 75, 11, 78, 15, -1};
7051 /* for xterm with 256 colors... */
7052 static int color_numbers_256[28] = {0, 4, 2, 6,
7053 1, 5, 130, 130,
7054 248, 248,
7055 7, 7, 242, 242,
7056 12, 81, 10, 121,
7057 14, 159, 9, 224, 13,
7058 225, 11, 229, 15, -1};
7059 /* for terminals with less than 16 colors... */
7060 static int color_numbers_8[28] = {0, 4, 2, 6,
7061 1, 5, 3, 3,
7062 7, 7,
7063 7, 7, 0+8, 0+8,
7064 4+8, 4+8, 2+8, 2+8,
7065 6+8, 6+8, 1+8, 1+8, 5+8,
7066 5+8, 3+8, 3+8, 7+8, -1};
7067#if defined(__QNXNTO__)
7068 static int *color_numbers_8_qansi = color_numbers_8;
7069 /* On qnx, the 8 & 16 color arrays are the same */
7070 if (STRNCMP(T_NAME, "qansi", 5) == 0)
7071 color_numbers_8_qansi = color_numbers_16;
7072#endif
7073
7074 /* reduce calls to STRICMP a bit, it can be slow */
7075 off = TOUPPER_ASC(*arg);
7076 for (i = (sizeof(color_names) / sizeof(char *)); --i >= 0; )
7077 if (off == color_names[i][0]
7078 && STRICMP(arg + 1, color_names[i] + 1) == 0)
7079 break;
7080 if (i < 0)
7081 {
7082 EMSG2(_("E421: Color name or number not recognized: %s"), key_start);
7083 error = TRUE;
7084 break;
7085 }
7086
7087 /* Use the _16 table to check if its a valid color name. */
7088 color = color_numbers_16[i];
7089 if (color >= 0)
7090 {
7091 if (t_colors == 8)
7092 {
7093 /* t_Co is 8: use the 8 colors table */
7094#if defined(__QNXNTO__)
7095 color = color_numbers_8_qansi[i];
7096#else
7097 color = color_numbers_8[i];
7098#endif
7099 if (key[5] == 'F')
7100 {
7101 /* set/reset bold attribute to get light foreground
7102 * colors (on some terminals, e.g. "linux") */
7103 if (color & 8)
7104 {
7105 HL_TABLE()[idx].sg_cterm |= HL_BOLD;
7106 HL_TABLE()[idx].sg_cterm_bold = TRUE;
7107 }
7108 else
7109 HL_TABLE()[idx].sg_cterm &= ~HL_BOLD;
7110 }
7111 color &= 7; /* truncate to 8 colors */
7112 }
7113 else if (t_colors == 16 || t_colors == 88
7114 || t_colors == 256)
7115 {
7116 /*
7117 * Guess: if the termcap entry ends in 'm', it is
7118 * probably an xterm-like terminal. Use the changed
7119 * order for colors.
7120 */
7121 if (*T_CAF != NUL)
7122 p = T_CAF;
7123 else
7124 p = T_CSF;
7125 if (*p != NUL && *(p + STRLEN(p) - 1) == 'm')
7126 switch (t_colors)
7127 {
7128 case 16:
7129 color = color_numbers_8[i];
7130 break;
7131 case 88:
7132 color = color_numbers_88[i];
7133 break;
7134 case 256:
7135 color = color_numbers_256[i];
7136 break;
7137 }
7138 }
7139 }
7140 }
7141 /* Add one to the argument, to avoid zero */
7142 if (key[5] == 'F')
7143 {
7144 HL_TABLE()[idx].sg_cterm_fg = color + 1;
7145 if (is_normal_group)
7146 {
7147 cterm_normal_fg_color = color + 1;
7148 cterm_normal_fg_bold = (HL_TABLE()[idx].sg_cterm & HL_BOLD);
7149#ifdef FEAT_GUI
7150 /* Don't do this if the GUI is used. */
7151 if (!gui.in_use && !gui.starting)
7152#endif
7153 {
7154 must_redraw = CLEAR;
7155 if (termcap_active)
7156 term_fg_color(color);
7157 }
7158 }
7159 }
7160 else
7161 {
7162 HL_TABLE()[idx].sg_cterm_bg = color + 1;
7163 if (is_normal_group)
7164 {
7165 cterm_normal_bg_color = color + 1;
7166#ifdef FEAT_GUI
7167 /* Don't mess with 'background' if the GUI is used. */
7168 if (!gui.in_use && !gui.starting)
7169#endif
7170 {
7171 must_redraw = CLEAR;
7172 if (termcap_active)
7173 term_bg_color(color);
7174 if (t_colors < 16)
7175 i = (color == 0 || color == 4);
7176 else
7177 i = (color < 7 || color == 8);
7178 /* Set the 'background' option if the value is wrong. */
7179 if (i != (*p_bg == 'd'))
7180 set_option_value((char_u *)"bg", 0L,
7181 i ? (char_u *)"dark" : (char_u *)"light", 0);
7182 }
7183 }
7184 }
7185 }
7186 }
7187 else if (STRCMP(key, "GUIFG") == 0)
7188 {
7189#ifdef FEAT_GUI /* in non-GUI guifg colors are simply ignored */
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007190 if (!init || !(HL_TABLE()[idx].sg_set & SG_GUI))
Bram Moolenaar071d4272004-06-13 20:20:40 +00007191 {
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007192 if (!init)
7193 HL_TABLE()[idx].sg_set |= SG_GUI;
7194
7195 i = color_name2handle(arg);
7196 if (i != INVALCOLOR || STRCMP(arg, "NONE") == 0 || !gui.in_use)
7197 {
7198 HL_TABLE()[idx].sg_gui_fg = i;
7199 vim_free(HL_TABLE()[idx].sg_gui_fg_name);
7200 if (STRCMP(arg, "NONE"))
7201 HL_TABLE()[idx].sg_gui_fg_name = vim_strsave(arg);
7202 else
7203 HL_TABLE()[idx].sg_gui_fg_name = NULL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007204# ifdef FEAT_GUI_X11
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007205 if (is_menu_group)
7206 gui.menu_fg_pixel = i;
7207 if (is_scrollbar_group)
7208 gui.scroll_fg_pixel = i;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007209# ifdef FEAT_BEVAL
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007210 if (is_tooltip_group)
7211 gui.tooltip_fg_pixel = i;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007212# endif
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007213 do_colors = TRUE;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007214# endif
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007215 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00007216 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00007217#endif
7218 }
7219 else if (STRCMP(key, "GUIBG") == 0)
7220 {
7221#ifdef FEAT_GUI /* in non-GUI guibg colors are simply ignored */
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007222 if (!init || !(HL_TABLE()[idx].sg_set & SG_GUI))
Bram Moolenaar071d4272004-06-13 20:20:40 +00007223 {
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007224 if (!init)
7225 HL_TABLE()[idx].sg_set |= SG_GUI;
7226
7227 i = color_name2handle(arg);
7228 if (i != INVALCOLOR || STRCMP(arg, "NONE") == 0 || !gui.in_use)
7229 {
7230 HL_TABLE()[idx].sg_gui_bg = i;
7231 vim_free(HL_TABLE()[idx].sg_gui_bg_name);
7232 if (STRCMP(arg, "NONE") != 0)
7233 HL_TABLE()[idx].sg_gui_bg_name = vim_strsave(arg);
7234 else
7235 HL_TABLE()[idx].sg_gui_bg_name = NULL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007236# ifdef FEAT_GUI_X11
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007237 if (is_menu_group)
7238 gui.menu_bg_pixel = i;
7239 if (is_scrollbar_group)
7240 gui.scroll_bg_pixel = i;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007241# ifdef FEAT_BEVAL
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007242 if (is_tooltip_group)
7243 gui.tooltip_bg_pixel = i;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007244# endif
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007245 do_colors = TRUE;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007246# endif
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007247 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00007248 }
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007249#endif
7250 }
7251 else if (STRCMP(key, "GUISP") == 0)
7252 {
7253#ifdef FEAT_GUI /* in non-GUI guisp colors are simply ignored */
7254 if (!init || !(HL_TABLE()[idx].sg_set & SG_GUI))
7255 {
7256 if (!init)
7257 HL_TABLE()[idx].sg_set |= SG_GUI;
7258
7259 i = color_name2handle(arg);
7260 if (i != INVALCOLOR || STRCMP(arg, "NONE") == 0 || !gui.in_use)
7261 {
7262 HL_TABLE()[idx].sg_gui_sp = i;
7263 vim_free(HL_TABLE()[idx].sg_gui_sp_name);
7264 if (STRCMP(arg, "NONE") != 0)
7265 HL_TABLE()[idx].sg_gui_sp_name = vim_strsave(arg);
7266 else
7267 HL_TABLE()[idx].sg_gui_sp_name = NULL;
7268 }
7269 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00007270#endif
7271 }
7272 else if (STRCMP(key, "START") == 0 || STRCMP(key, "STOP") == 0)
7273 {
7274 char_u buf[100];
7275 char_u *tname;
7276
7277 if (!init)
7278 HL_TABLE()[idx].sg_set |= SG_TERM;
7279
7280 /*
7281 * The "start" and "stop" arguments can be a literal escape
Bram Moolenaarb8017e72007-05-10 18:59:07 +00007282 * sequence, or a comma separated list of terminal codes.
Bram Moolenaar071d4272004-06-13 20:20:40 +00007283 */
7284 if (STRNCMP(arg, "t_", 2) == 0)
7285 {
7286 off = 0;
7287 buf[0] = 0;
7288 while (arg[off] != NUL)
7289 {
7290 /* Isolate one termcap name */
7291 for (len = 0; arg[off + len] &&
7292 arg[off + len] != ','; ++len)
7293 ;
7294 tname = vim_strnsave(arg + off, len);
7295 if (tname == NULL) /* out of memory */
7296 {
7297 error = TRUE;
7298 break;
7299 }
7300 /* lookup the escape sequence for the item */
7301 p = get_term_code(tname);
7302 vim_free(tname);
7303 if (p == NULL) /* ignore non-existing things */
7304 p = (char_u *)"";
7305
7306 /* Append it to the already found stuff */
7307 if ((int)(STRLEN(buf) + STRLEN(p)) >= 99)
7308 {
7309 EMSG2(_("E422: terminal code too long: %s"), arg);
7310 error = TRUE;
7311 break;
7312 }
7313 STRCAT(buf, p);
7314
7315 /* Advance to the next item */
7316 off += len;
7317 if (arg[off] == ',') /* another one follows */
7318 ++off;
7319 }
7320 }
7321 else
7322 {
7323 /*
7324 * Copy characters from arg[] to buf[], translating <> codes.
7325 */
7326 for (p = arg, off = 0; off < 100 && *p; )
7327 {
7328 len = trans_special(&p, buf + off, FALSE);
7329 if (len) /* recognized special char */
7330 off += len;
7331 else /* copy as normal char */
7332 buf[off++] = *p++;
7333 }
7334 buf[off] = NUL;
7335 }
7336 if (error)
7337 break;
7338
7339 if (STRCMP(buf, "NONE") == 0) /* resetting the value */
7340 p = NULL;
7341 else
7342 p = vim_strsave(buf);
7343 if (key[2] == 'A')
7344 {
7345 vim_free(HL_TABLE()[idx].sg_start);
7346 HL_TABLE()[idx].sg_start = p;
7347 }
7348 else
7349 {
7350 vim_free(HL_TABLE()[idx].sg_stop);
7351 HL_TABLE()[idx].sg_stop = p;
7352 }
7353 }
7354 else
7355 {
7356 EMSG2(_("E423: Illegal argument: %s"), key_start);
7357 error = TRUE;
7358 break;
7359 }
7360
7361 /*
7362 * When highlighting has been given for a group, don't link it.
7363 */
7364 if (!init || !(HL_TABLE()[idx].sg_set & SG_LINK))
7365 HL_TABLE()[idx].sg_link = 0;
7366
7367 /*
7368 * Continue with next argument.
7369 */
7370 linep = skipwhite(linep);
7371 }
7372
7373 /*
7374 * If there is an error, and it's a new entry, remove it from the table.
7375 */
7376 if (error && idx == highlight_ga.ga_len)
7377 syn_unadd_group();
7378 else
7379 {
7380 if (is_normal_group)
7381 {
7382 HL_TABLE()[idx].sg_term_attr = 0;
7383 HL_TABLE()[idx].sg_cterm_attr = 0;
7384#ifdef FEAT_GUI
7385 HL_TABLE()[idx].sg_gui_attr = 0;
7386 /*
7387 * Need to update all groups, because they might be using "bg"
7388 * and/or "fg", which have been changed now.
7389 */
7390 if (gui.in_use)
7391 highlight_gui_started();
7392#endif
7393 }
7394#ifdef FEAT_GUI_X11
7395# ifdef FEAT_MENU
7396 else if (is_menu_group)
7397 {
7398 if (gui.in_use && do_colors)
7399 gui_mch_new_menu_colors();
7400 }
7401# endif
7402 else if (is_scrollbar_group)
7403 {
7404 if (gui.in_use && do_colors)
7405 gui_new_scrollbar_colors();
7406 }
7407# ifdef FEAT_BEVAL
7408 else if (is_tooltip_group)
7409 {
7410 if (gui.in_use && do_colors)
7411 gui_mch_new_tooltip_colors();
7412 }
7413# endif
7414#endif
7415 else
7416 set_hl_attr(idx);
Bram Moolenaar661b1822005-07-28 22:36:45 +00007417#ifdef FEAT_EVAL
7418 HL_TABLE()[idx].sg_scriptID = current_SID;
7419#endif
Bram Moolenaar910f66f2006-04-05 20:41:53 +00007420 redraw_all_later(NOT_VALID);
Bram Moolenaar071d4272004-06-13 20:20:40 +00007421 }
7422 vim_free(key);
7423 vim_free(arg);
7424
7425 /* Only call highlight_changed() once, after sourcing a syntax file */
7426 need_highlight_changed = TRUE;
7427}
7428
Bram Moolenaar1ec484f2005-06-24 23:07:47 +00007429#if defined(EXITFREE) || defined(PROTO)
7430 void
7431free_highlight()
7432{
7433 int i;
7434
7435 for (i = 0; i < highlight_ga.ga_len; ++i)
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00007436 {
Bram Moolenaar1ec484f2005-06-24 23:07:47 +00007437 highlight_clear(i);
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00007438 vim_free(HL_TABLE()[i].sg_name);
7439 vim_free(HL_TABLE()[i].sg_name_u);
7440 }
Bram Moolenaar1ec484f2005-06-24 23:07:47 +00007441 ga_clear(&highlight_ga);
7442}
7443#endif
7444
Bram Moolenaar071d4272004-06-13 20:20:40 +00007445/*
7446 * Reset the cterm colors to what they were before Vim was started, if
7447 * possible. Otherwise reset them to zero.
7448 */
7449 void
7450restore_cterm_colors()
7451{
7452#if defined(MSDOS) || (defined(WIN3264) && !defined(FEAT_GUI_W32))
7453 /* Since t_me has been set, this probably means that the user
7454 * wants to use this as default colors. Need to reset default
7455 * background/foreground colors. */
7456 mch_set_normal_colors();
7457#else
7458 cterm_normal_fg_color = 0;
7459 cterm_normal_fg_bold = 0;
7460 cterm_normal_bg_color = 0;
7461#endif
7462}
7463
7464/*
7465 * Return TRUE if highlight group "idx" has any settings.
7466 * When "check_link" is TRUE also check for an existing link.
7467 */
7468 static int
7469hl_has_settings(idx, check_link)
7470 int idx;
7471 int check_link;
7472{
7473 return ( HL_TABLE()[idx].sg_term_attr != 0
7474 || HL_TABLE()[idx].sg_cterm_attr != 0
7475#ifdef FEAT_GUI
7476 || HL_TABLE()[idx].sg_gui_attr != 0
7477#endif
7478 || (check_link && (HL_TABLE()[idx].sg_set & SG_LINK)));
7479}
7480
7481/*
7482 * Clear highlighting for one group.
7483 */
7484 static void
7485highlight_clear(idx)
7486 int idx;
7487{
7488 HL_TABLE()[idx].sg_term = 0;
7489 vim_free(HL_TABLE()[idx].sg_start);
7490 HL_TABLE()[idx].sg_start = NULL;
7491 vim_free(HL_TABLE()[idx].sg_stop);
7492 HL_TABLE()[idx].sg_stop = NULL;
7493 HL_TABLE()[idx].sg_term_attr = 0;
7494 HL_TABLE()[idx].sg_cterm = 0;
7495 HL_TABLE()[idx].sg_cterm_bold = FALSE;
7496 HL_TABLE()[idx].sg_cterm_fg = 0;
7497 HL_TABLE()[idx].sg_cterm_bg = 0;
7498 HL_TABLE()[idx].sg_cterm_attr = 0;
7499#ifdef FEAT_GUI /* in non-GUI fonts are simply ignored */
7500 HL_TABLE()[idx].sg_gui = 0;
7501 HL_TABLE()[idx].sg_gui_fg = INVALCOLOR;
7502 vim_free(HL_TABLE()[idx].sg_gui_fg_name);
7503 HL_TABLE()[idx].sg_gui_fg_name = NULL;
7504 HL_TABLE()[idx].sg_gui_bg = INVALCOLOR;
7505 vim_free(HL_TABLE()[idx].sg_gui_bg_name);
7506 HL_TABLE()[idx].sg_gui_bg_name = NULL;
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007507 HL_TABLE()[idx].sg_gui_sp = INVALCOLOR;
7508 vim_free(HL_TABLE()[idx].sg_gui_sp_name);
7509 HL_TABLE()[idx].sg_gui_sp_name = NULL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007510 gui_mch_free_font(HL_TABLE()[idx].sg_font);
7511 HL_TABLE()[idx].sg_font = NOFONT;
7512# ifdef FEAT_XFONTSET
7513 gui_mch_free_fontset(HL_TABLE()[idx].sg_fontset);
7514 HL_TABLE()[idx].sg_fontset = NOFONTSET;
7515# endif
7516 vim_free(HL_TABLE()[idx].sg_font_name);
7517 HL_TABLE()[idx].sg_font_name = NULL;
7518 HL_TABLE()[idx].sg_gui_attr = 0;
7519#endif
Bram Moolenaar661b1822005-07-28 22:36:45 +00007520#ifdef FEAT_EVAL
7521 /* Clear the script ID only when there is no link, since that is not
7522 * cleared. */
7523 if (HL_TABLE()[idx].sg_link == 0)
7524 HL_TABLE()[idx].sg_scriptID = 0;
7525#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00007526}
7527
7528#if defined(FEAT_GUI) || defined(PROTO)
7529/*
7530 * Set the normal foreground and background colors according to the "Normal"
Bram Moolenaarf5b63862009-12-16 17:13:44 +00007531 * highlighting group. For X11 also set "Menu", "Scrollbar", and
Bram Moolenaar071d4272004-06-13 20:20:40 +00007532 * "Tooltip" colors.
7533 */
7534 void
7535set_normal_colors()
7536{
7537 if (set_group_colors((char_u *)"Normal",
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007538 &gui.norm_pixel, &gui.back_pixel,
7539 FALSE, TRUE, FALSE))
Bram Moolenaar071d4272004-06-13 20:20:40 +00007540 {
7541 gui_mch_new_colors();
7542 must_redraw = CLEAR;
7543 }
7544#ifdef FEAT_GUI_X11
7545 if (set_group_colors((char_u *)"Menu",
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007546 &gui.menu_fg_pixel, &gui.menu_bg_pixel,
7547 TRUE, FALSE, FALSE))
Bram Moolenaar071d4272004-06-13 20:20:40 +00007548 {
7549# ifdef FEAT_MENU
7550 gui_mch_new_menu_colors();
7551# endif
7552 must_redraw = CLEAR;
7553 }
7554# ifdef FEAT_BEVAL
7555 if (set_group_colors((char_u *)"Tooltip",
7556 &gui.tooltip_fg_pixel, &gui.tooltip_bg_pixel,
7557 FALSE, FALSE, TRUE))
7558 {
7559# ifdef FEAT_TOOLBAR
7560 gui_mch_new_tooltip_colors();
7561# endif
7562 must_redraw = CLEAR;
7563 }
7564#endif
7565 if (set_group_colors((char_u *)"Scrollbar",
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007566 &gui.scroll_fg_pixel, &gui.scroll_bg_pixel,
7567 FALSE, FALSE, FALSE))
Bram Moolenaar071d4272004-06-13 20:20:40 +00007568 {
7569 gui_new_scrollbar_colors();
7570 must_redraw = CLEAR;
7571 }
7572#endif
7573}
7574
7575/*
7576 * Set the colors for "Normal", "Menu", "Tooltip" or "Scrollbar".
7577 */
7578 static int
7579set_group_colors(name, fgp, bgp, do_menu, use_norm, do_tooltip)
7580 char_u *name;
7581 guicolor_T *fgp;
7582 guicolor_T *bgp;
7583 int do_menu;
7584 int use_norm;
7585 int do_tooltip;
7586{
7587 int idx;
7588
7589 idx = syn_name2id(name) - 1;
7590 if (idx >= 0)
7591 {
7592 gui_do_one_color(idx, do_menu, do_tooltip);
7593
7594 if (HL_TABLE()[idx].sg_gui_fg != INVALCOLOR)
7595 *fgp = HL_TABLE()[idx].sg_gui_fg;
7596 else if (use_norm)
7597 *fgp = gui.def_norm_pixel;
7598 if (HL_TABLE()[idx].sg_gui_bg != INVALCOLOR)
7599 *bgp = HL_TABLE()[idx].sg_gui_bg;
7600 else if (use_norm)
7601 *bgp = gui.def_back_pixel;
7602 return TRUE;
7603 }
7604 return FALSE;
7605}
7606
7607/*
7608 * Get the font of the "Normal" group.
7609 * Returns "" when it's not found or not set.
7610 */
7611 char_u *
7612hl_get_font_name()
7613{
7614 int id;
7615 char_u *s;
7616
7617 id = syn_name2id((char_u *)"Normal");
7618 if (id > 0)
7619 {
7620 s = HL_TABLE()[id - 1].sg_font_name;
7621 if (s != NULL)
7622 return s;
7623 }
7624 return (char_u *)"";
7625}
7626
7627/*
7628 * Set font for "Normal" group. Called by gui_mch_init_font() when a font has
7629 * actually chosen to be used.
7630 */
7631 void
7632hl_set_font_name(font_name)
7633 char_u *font_name;
7634{
7635 int id;
7636
7637 id = syn_name2id((char_u *)"Normal");
7638 if (id > 0)
7639 {
7640 vim_free(HL_TABLE()[id - 1].sg_font_name);
7641 HL_TABLE()[id - 1].sg_font_name = vim_strsave(font_name);
7642 }
7643}
7644
7645/*
7646 * Set background color for "Normal" group. Called by gui_set_bg_color()
7647 * when the color is known.
7648 */
7649 void
7650hl_set_bg_color_name(name)
7651 char_u *name; /* must have been allocated */
7652{
7653 int id;
7654
7655 if (name != NULL)
7656 {
7657 id = syn_name2id((char_u *)"Normal");
7658 if (id > 0)
7659 {
7660 vim_free(HL_TABLE()[id - 1].sg_gui_bg_name);
7661 HL_TABLE()[id - 1].sg_gui_bg_name = name;
7662 }
7663 }
7664}
7665
7666/*
7667 * Set foreground color for "Normal" group. Called by gui_set_fg_color()
7668 * when the color is known.
7669 */
7670 void
7671hl_set_fg_color_name(name)
7672 char_u *name; /* must have been allocated */
7673{
7674 int id;
7675
7676 if (name != NULL)
7677 {
7678 id = syn_name2id((char_u *)"Normal");
7679 if (id > 0)
7680 {
7681 vim_free(HL_TABLE()[id - 1].sg_gui_fg_name);
7682 HL_TABLE()[id - 1].sg_gui_fg_name = name;
7683 }
7684 }
7685}
7686
7687/*
7688 * Return the handle for a color name.
7689 * Returns INVALCOLOR when failed.
7690 */
7691 static guicolor_T
7692color_name2handle(name)
7693 char_u *name;
7694{
7695 if (STRCMP(name, "NONE") == 0)
7696 return INVALCOLOR;
7697
7698 if (STRICMP(name, "fg") == 0 || STRICMP(name, "foreground") == 0)
7699 return gui.norm_pixel;
7700 if (STRICMP(name, "bg") == 0 || STRICMP(name, "background") == 0)
7701 return gui.back_pixel;
7702
7703 return gui_get_color(name);
7704}
7705
7706/*
7707 * Return the handle for a font name.
7708 * Returns NOFONT when failed.
7709 */
7710 static GuiFont
7711font_name2handle(name)
7712 char_u *name;
7713{
7714 if (STRCMP(name, "NONE") == 0)
7715 return NOFONT;
7716
7717 return gui_mch_get_font(name, TRUE);
7718}
7719
7720# ifdef FEAT_XFONTSET
7721/*
7722 * Return the handle for a fontset name.
7723 * Returns NOFONTSET when failed.
7724 */
7725 static GuiFontset
7726fontset_name2handle(name, fixed_width)
7727 char_u *name;
7728 int fixed_width;
7729{
7730 if (STRCMP(name, "NONE") == 0)
7731 return NOFONTSET;
7732
7733 return gui_mch_get_fontset(name, TRUE, fixed_width);
7734}
7735# endif
7736
7737/*
7738 * Get the font or fontset for one highlight group.
7739 */
Bram Moolenaar071d4272004-06-13 20:20:40 +00007740 static void
7741hl_do_font(idx, arg, do_normal, do_menu, do_tooltip)
7742 int idx;
7743 char_u *arg;
Bram Moolenaar2c4278f2009-05-17 11:33:22 +00007744 int do_normal; /* set normal font */
7745 int do_menu UNUSED; /* set menu font */
7746 int do_tooltip UNUSED; /* set tooltip font */
Bram Moolenaar071d4272004-06-13 20:20:40 +00007747{
7748# ifdef FEAT_XFONTSET
7749 /* If 'guifontset' is not empty, first try using the name as a
7750 * fontset. If that doesn't work, use it as a font name. */
7751 if (*p_guifontset != NUL
7752# ifdef FONTSET_ALWAYS
7753 || do_menu
7754# endif
7755# ifdef FEAT_BEVAL_TIP
7756 /* In Athena & Motif, the Tooltip highlight group is always a fontset */
7757 || do_tooltip
7758# endif
7759 )
7760 HL_TABLE()[idx].sg_fontset = fontset_name2handle(arg, 0
7761# ifdef FONTSET_ALWAYS
7762 || do_menu
7763# endif
7764# ifdef FEAT_BEVAL_TIP
7765 || do_tooltip
7766# endif
7767 );
7768 if (HL_TABLE()[idx].sg_fontset != NOFONTSET)
7769 {
7770 /* If it worked and it's the Normal group, use it as the
7771 * normal fontset. Same for the Menu group. */
7772 if (do_normal)
7773 gui_init_font(arg, TRUE);
7774# if (defined(FEAT_GUI_MOTIF) || defined(FEAT_GUI_ATHENA)) && defined(FEAT_MENU)
7775 if (do_menu)
7776 {
7777# ifdef FONTSET_ALWAYS
7778 gui.menu_fontset = HL_TABLE()[idx].sg_fontset;
7779# else
7780 /* YIKES! This is a bug waiting to crash the program */
7781 gui.menu_font = HL_TABLE()[idx].sg_fontset;
7782# endif
7783 gui_mch_new_menu_font();
7784 }
7785# ifdef FEAT_BEVAL
7786 if (do_tooltip)
7787 {
7788 /* The Athena widget set cannot currently handle switching between
7789 * displaying a single font and a fontset.
7790 * If the XtNinternational resource is set to True at widget
Bram Moolenaarb8017e72007-05-10 18:59:07 +00007791 * creation, then a fontset is always used, otherwise an
Bram Moolenaar071d4272004-06-13 20:20:40 +00007792 * XFontStruct is used.
7793 */
7794 gui.tooltip_fontset = (XFontSet)HL_TABLE()[idx].sg_fontset;
7795 gui_mch_new_tooltip_font();
7796 }
7797# endif
7798# endif
7799 }
7800 else
7801# endif
7802 {
7803 HL_TABLE()[idx].sg_font = font_name2handle(arg);
7804 /* If it worked and it's the Normal group, use it as the
7805 * normal font. Same for the Menu group. */
7806 if (HL_TABLE()[idx].sg_font != NOFONT)
7807 {
7808 if (do_normal)
7809 gui_init_font(arg, FALSE);
7810#ifndef FONTSET_ALWAYS
7811# if (defined(FEAT_GUI_MOTIF) || defined(FEAT_GUI_ATHENA)) && defined(FEAT_MENU)
7812 if (do_menu)
7813 {
7814 gui.menu_font = HL_TABLE()[idx].sg_font;
7815 gui_mch_new_menu_font();
7816 }
7817# endif
7818#endif
7819 }
7820 }
7821}
7822
7823#endif /* FEAT_GUI */
7824
7825/*
7826 * Table with the specifications for an attribute number.
7827 * Note that this table is used by ALL buffers. This is required because the
7828 * GUI can redraw at any time for any buffer.
7829 */
Bram Moolenaar6c0b44b2005-06-01 21:56:33 +00007830static garray_T term_attr_table = {0, 0, 0, 0, NULL};
Bram Moolenaar071d4272004-06-13 20:20:40 +00007831
7832#define TERM_ATTR_ENTRY(idx) ((attrentry_T *)term_attr_table.ga_data)[idx]
7833
Bram Moolenaar6c0b44b2005-06-01 21:56:33 +00007834static garray_T cterm_attr_table = {0, 0, 0, 0, NULL};
Bram Moolenaar071d4272004-06-13 20:20:40 +00007835
7836#define CTERM_ATTR_ENTRY(idx) ((attrentry_T *)cterm_attr_table.ga_data)[idx]
7837
7838#ifdef FEAT_GUI
Bram Moolenaar6c0b44b2005-06-01 21:56:33 +00007839static garray_T gui_attr_table = {0, 0, 0, 0, NULL};
Bram Moolenaar071d4272004-06-13 20:20:40 +00007840
7841#define GUI_ATTR_ENTRY(idx) ((attrentry_T *)gui_attr_table.ga_data)[idx]
7842#endif
7843
7844/*
7845 * Return the attr number for a set of colors and font.
7846 * Add a new entry to the term_attr_table, cterm_attr_table or gui_attr_table
7847 * if the combination is new.
7848 * Return 0 for error (no more room).
7849 */
7850 static int
7851get_attr_entry(table, aep)
7852 garray_T *table;
7853 attrentry_T *aep;
7854{
7855 int i;
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00007856 attrentry_T *taep;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007857 static int recursive = FALSE;
7858
7859 /*
7860 * Init the table, in case it wasn't done yet.
7861 */
7862 table->ga_itemsize = sizeof(attrentry_T);
7863 table->ga_growsize = 7;
7864
7865 /*
7866 * Try to find an entry with the same specifications.
7867 */
7868 for (i = 0; i < table->ga_len; ++i)
7869 {
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00007870 taep = &(((attrentry_T *)table->ga_data)[i]);
7871 if ( aep->ae_attr == taep->ae_attr
Bram Moolenaar071d4272004-06-13 20:20:40 +00007872 && (
7873#ifdef FEAT_GUI
7874 (table == &gui_attr_table
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00007875 && (aep->ae_u.gui.fg_color == taep->ae_u.gui.fg_color
7876 && aep->ae_u.gui.bg_color
7877 == taep->ae_u.gui.bg_color
7878 && aep->ae_u.gui.sp_color
7879 == taep->ae_u.gui.sp_color
7880 && aep->ae_u.gui.font == taep->ae_u.gui.font
Bram Moolenaar071d4272004-06-13 20:20:40 +00007881# ifdef FEAT_XFONTSET
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00007882 && aep->ae_u.gui.fontset == taep->ae_u.gui.fontset
Bram Moolenaar071d4272004-06-13 20:20:40 +00007883# endif
7884 ))
7885 ||
7886#endif
7887 (table == &term_attr_table
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00007888 && (aep->ae_u.term.start == NULL)
7889 == (taep->ae_u.term.start == NULL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00007890 && (aep->ae_u.term.start == NULL
7891 || STRCMP(aep->ae_u.term.start,
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00007892 taep->ae_u.term.start) == 0)
7893 && (aep->ae_u.term.stop == NULL)
7894 == (taep->ae_u.term.stop == NULL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00007895 && (aep->ae_u.term.stop == NULL
7896 || STRCMP(aep->ae_u.term.stop,
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00007897 taep->ae_u.term.stop) == 0))
Bram Moolenaar071d4272004-06-13 20:20:40 +00007898 || (table == &cterm_attr_table
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00007899 && aep->ae_u.cterm.fg_color
7900 == taep->ae_u.cterm.fg_color
7901 && aep->ae_u.cterm.bg_color
7902 == taep->ae_u.cterm.bg_color)
Bram Moolenaar071d4272004-06-13 20:20:40 +00007903 ))
7904
7905 return i + ATTR_OFF;
7906 }
7907
Bram Moolenaar34cdc3e2005-05-18 22:24:46 +00007908 if (table->ga_len + ATTR_OFF > MAX_TYPENR)
Bram Moolenaar071d4272004-06-13 20:20:40 +00007909 {
7910 /*
7911 * Running out of attribute entries! remove all attributes, and
7912 * compute new ones for all groups.
7913 * When called recursively, we are really out of numbers.
7914 */
7915 if (recursive)
7916 {
7917 EMSG(_("E424: Too many different highlighting attributes in use"));
7918 return 0;
7919 }
7920 recursive = TRUE;
7921
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00007922 clear_hl_tables();
7923
Bram Moolenaar071d4272004-06-13 20:20:40 +00007924 must_redraw = CLEAR;
7925
7926 for (i = 0; i < highlight_ga.ga_len; ++i)
7927 set_hl_attr(i);
7928
7929 recursive = FALSE;
7930 }
7931
7932 /*
7933 * This is a new combination of colors and font, add an entry.
7934 */
7935 if (ga_grow(table, 1) == FAIL)
7936 return 0;
7937
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00007938 taep = &(((attrentry_T *)table->ga_data)[table->ga_len]);
7939 vim_memset(taep, 0, sizeof(attrentry_T));
7940 taep->ae_attr = aep->ae_attr;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007941#ifdef FEAT_GUI
7942 if (table == &gui_attr_table)
7943 {
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00007944 taep->ae_u.gui.fg_color = aep->ae_u.gui.fg_color;
7945 taep->ae_u.gui.bg_color = aep->ae_u.gui.bg_color;
7946 taep->ae_u.gui.sp_color = aep->ae_u.gui.sp_color;
7947 taep->ae_u.gui.font = aep->ae_u.gui.font;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007948# ifdef FEAT_XFONTSET
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00007949 taep->ae_u.gui.fontset = aep->ae_u.gui.fontset;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007950# endif
7951 }
7952#endif
7953 if (table == &term_attr_table)
7954 {
7955 if (aep->ae_u.term.start == NULL)
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00007956 taep->ae_u.term.start = NULL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007957 else
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00007958 taep->ae_u.term.start = vim_strsave(aep->ae_u.term.start);
Bram Moolenaar071d4272004-06-13 20:20:40 +00007959 if (aep->ae_u.term.stop == NULL)
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00007960 taep->ae_u.term.stop = NULL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007961 else
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00007962 taep->ae_u.term.stop = vim_strsave(aep->ae_u.term.stop);
Bram Moolenaar071d4272004-06-13 20:20:40 +00007963 }
7964 else if (table == &cterm_attr_table)
7965 {
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00007966 taep->ae_u.cterm.fg_color = aep->ae_u.cterm.fg_color;
7967 taep->ae_u.cterm.bg_color = aep->ae_u.cterm.bg_color;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007968 }
7969 ++table->ga_len;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007970 return (table->ga_len - 1 + ATTR_OFF);
7971}
7972
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00007973/*
7974 * Clear all highlight tables.
7975 */
7976 void
7977clear_hl_tables()
7978{
7979 int i;
7980 attrentry_T *taep;
7981
7982#ifdef FEAT_GUI
7983 ga_clear(&gui_attr_table);
7984#endif
7985 for (i = 0; i < term_attr_table.ga_len; ++i)
7986 {
7987 taep = &(((attrentry_T *)term_attr_table.ga_data)[i]);
7988 vim_free(taep->ae_u.term.start);
7989 vim_free(taep->ae_u.term.stop);
7990 }
7991 ga_clear(&term_attr_table);
7992 ga_clear(&cterm_attr_table);
7993}
7994
Bram Moolenaar1c8f93f2006-03-12 22:10:07 +00007995#if defined(FEAT_SYN_HL) || defined(FEAT_SPELL) || defined(PROTO)
Bram Moolenaar217ad922005-03-20 22:37:15 +00007996/*
Bram Moolenaar30abd282005-06-22 22:35:10 +00007997 * Combine special attributes (e.g., for spelling) with other attributes
7998 * (e.g., for syntax highlighting).
7999 * "prim_attr" overrules "char_attr".
Bram Moolenaar217ad922005-03-20 22:37:15 +00008000 * This creates a new group when required.
8001 * Since we expect there to be few spelling mistakes we don't cache the
8002 * result.
8003 * Return the resulting attributes.
8004 */
8005 int
Bram Moolenaar30abd282005-06-22 22:35:10 +00008006hl_combine_attr(char_attr, prim_attr)
Bram Moolenaar217ad922005-03-20 22:37:15 +00008007 int char_attr;
Bram Moolenaar30abd282005-06-22 22:35:10 +00008008 int prim_attr;
Bram Moolenaar217ad922005-03-20 22:37:15 +00008009{
8010 attrentry_T *char_aep = NULL;
8011 attrentry_T *spell_aep;
8012 attrentry_T new_en;
8013
8014 if (char_attr == 0)
Bram Moolenaar30abd282005-06-22 22:35:10 +00008015 return prim_attr;
8016 if (char_attr <= HL_ALL && prim_attr <= HL_ALL)
8017 return char_attr | prim_attr;
Bram Moolenaar217ad922005-03-20 22:37:15 +00008018#ifdef FEAT_GUI
8019 if (gui.in_use)
8020 {
8021 if (char_attr > HL_ALL)
8022 char_aep = syn_gui_attr2entry(char_attr);
8023 if (char_aep != NULL)
8024 new_en = *char_aep;
8025 else
8026 {
8027 vim_memset(&new_en, 0, sizeof(new_en));
Bram Moolenaard5cdbeb2005-10-10 20:59:28 +00008028 new_en.ae_u.gui.fg_color = INVALCOLOR;
8029 new_en.ae_u.gui.bg_color = INVALCOLOR;
8030 new_en.ae_u.gui.sp_color = INVALCOLOR;
Bram Moolenaar217ad922005-03-20 22:37:15 +00008031 if (char_attr <= HL_ALL)
8032 new_en.ae_attr = char_attr;
8033 }
8034
Bram Moolenaar30abd282005-06-22 22:35:10 +00008035 if (prim_attr <= HL_ALL)
8036 new_en.ae_attr |= prim_attr;
Bram Moolenaar217ad922005-03-20 22:37:15 +00008037 else
8038 {
Bram Moolenaar30abd282005-06-22 22:35:10 +00008039 spell_aep = syn_gui_attr2entry(prim_attr);
Bram Moolenaar217ad922005-03-20 22:37:15 +00008040 if (spell_aep != NULL)
8041 {
8042 new_en.ae_attr |= spell_aep->ae_attr;
8043 if (spell_aep->ae_u.gui.fg_color != INVALCOLOR)
8044 new_en.ae_u.gui.fg_color = spell_aep->ae_u.gui.fg_color;
8045 if (spell_aep->ae_u.gui.bg_color != INVALCOLOR)
8046 new_en.ae_u.gui.bg_color = spell_aep->ae_u.gui.bg_color;
8047 if (spell_aep->ae_u.gui.sp_color != INVALCOLOR)
8048 new_en.ae_u.gui.sp_color = spell_aep->ae_u.gui.sp_color;
8049 if (spell_aep->ae_u.gui.font != NOFONT)
8050 new_en.ae_u.gui.font = spell_aep->ae_u.gui.font;
8051# ifdef FEAT_XFONTSET
8052 if (spell_aep->ae_u.gui.fontset != NOFONTSET)
8053 new_en.ae_u.gui.fontset = spell_aep->ae_u.gui.fontset;
8054# endif
8055 }
8056 }
8057 return get_attr_entry(&gui_attr_table, &new_en);
8058 }
8059#endif
8060
8061 if (t_colors > 1)
8062 {
8063 if (char_attr > HL_ALL)
8064 char_aep = syn_cterm_attr2entry(char_attr);
8065 if (char_aep != NULL)
8066 new_en = *char_aep;
8067 else
8068 {
8069 vim_memset(&new_en, 0, sizeof(new_en));
8070 if (char_attr <= HL_ALL)
8071 new_en.ae_attr = char_attr;
8072 }
8073
Bram Moolenaar30abd282005-06-22 22:35:10 +00008074 if (prim_attr <= HL_ALL)
8075 new_en.ae_attr |= prim_attr;
Bram Moolenaar217ad922005-03-20 22:37:15 +00008076 else
8077 {
Bram Moolenaar30abd282005-06-22 22:35:10 +00008078 spell_aep = syn_cterm_attr2entry(prim_attr);
Bram Moolenaar217ad922005-03-20 22:37:15 +00008079 if (spell_aep != NULL)
8080 {
8081 new_en.ae_attr |= spell_aep->ae_attr;
8082 if (spell_aep->ae_u.cterm.fg_color > 0)
8083 new_en.ae_u.cterm.fg_color = spell_aep->ae_u.cterm.fg_color;
8084 if (spell_aep->ae_u.cterm.bg_color > 0)
8085 new_en.ae_u.cterm.bg_color = spell_aep->ae_u.cterm.bg_color;
8086 }
8087 }
8088 return get_attr_entry(&cterm_attr_table, &new_en);
8089 }
8090
8091 if (char_attr > HL_ALL)
8092 char_aep = syn_term_attr2entry(char_attr);
8093 if (char_aep != NULL)
8094 new_en = *char_aep;
8095 else
8096 {
8097 vim_memset(&new_en, 0, sizeof(new_en));
8098 if (char_attr <= HL_ALL)
8099 new_en.ae_attr = char_attr;
8100 }
8101
Bram Moolenaar30abd282005-06-22 22:35:10 +00008102 if (prim_attr <= HL_ALL)
8103 new_en.ae_attr |= prim_attr;
Bram Moolenaar217ad922005-03-20 22:37:15 +00008104 else
8105 {
Bram Moolenaar3b181812005-12-17 22:10:02 +00008106 spell_aep = syn_term_attr2entry(prim_attr);
Bram Moolenaar217ad922005-03-20 22:37:15 +00008107 if (spell_aep != NULL)
8108 {
8109 new_en.ae_attr |= spell_aep->ae_attr;
8110 if (spell_aep->ae_u.term.start != NULL)
8111 {
8112 new_en.ae_u.term.start = spell_aep->ae_u.term.start;
8113 new_en.ae_u.term.stop = spell_aep->ae_u.term.stop;
8114 }
8115 }
8116 }
8117 return get_attr_entry(&term_attr_table, &new_en);
8118}
8119#endif
8120
Bram Moolenaar071d4272004-06-13 20:20:40 +00008121#ifdef FEAT_GUI
8122
8123 attrentry_T *
8124syn_gui_attr2entry(attr)
8125 int attr;
8126{
8127 attr -= ATTR_OFF;
8128 if (attr >= gui_attr_table.ga_len) /* did ":syntax clear" */
8129 return NULL;
8130 return &(GUI_ATTR_ENTRY(attr));
8131}
Bram Moolenaar071d4272004-06-13 20:20:40 +00008132#endif /* FEAT_GUI */
8133
Bram Moolenaar1c8f93f2006-03-12 22:10:07 +00008134/*
8135 * Get the highlight attributes (HL_BOLD etc.) from an attribute nr.
8136 * Only to be used when "attr" > HL_ALL.
8137 */
8138 int
8139syn_attr2attr(attr)
8140 int attr;
8141{
8142 attrentry_T *aep;
8143
8144#ifdef FEAT_GUI
8145 if (gui.in_use)
8146 aep = syn_gui_attr2entry(attr);
8147 else
8148#endif
8149 if (t_colors > 1)
8150 aep = syn_cterm_attr2entry(attr);
8151 else
8152 aep = syn_term_attr2entry(attr);
8153
8154 if (aep == NULL) /* highlighting not set */
8155 return 0;
8156 return aep->ae_attr;
8157}
8158
8159
Bram Moolenaar071d4272004-06-13 20:20:40 +00008160 attrentry_T *
8161syn_term_attr2entry(attr)
8162 int attr;
8163{
8164 attr -= ATTR_OFF;
8165 if (attr >= term_attr_table.ga_len) /* did ":syntax clear" */
8166 return NULL;
8167 return &(TERM_ATTR_ENTRY(attr));
8168}
8169
8170 attrentry_T *
8171syn_cterm_attr2entry(attr)
8172 int attr;
8173{
8174 attr -= ATTR_OFF;
8175 if (attr >= cterm_attr_table.ga_len) /* did ":syntax clear" */
8176 return NULL;
8177 return &(CTERM_ATTR_ENTRY(attr));
8178}
8179
8180#define LIST_ATTR 1
8181#define LIST_STRING 2
8182#define LIST_INT 3
8183
8184 static void
8185highlight_list_one(id)
8186 int id;
8187{
8188 struct hl_group *sgp;
8189 int didh = FALSE;
8190
8191 sgp = &HL_TABLE()[id - 1]; /* index is ID minus one */
8192
8193 didh = highlight_list_arg(id, didh, LIST_ATTR,
8194 sgp->sg_term, NULL, "term");
8195 didh = highlight_list_arg(id, didh, LIST_STRING,
8196 0, sgp->sg_start, "start");
8197 didh = highlight_list_arg(id, didh, LIST_STRING,
8198 0, sgp->sg_stop, "stop");
8199
8200 didh = highlight_list_arg(id, didh, LIST_ATTR,
8201 sgp->sg_cterm, NULL, "cterm");
8202 didh = highlight_list_arg(id, didh, LIST_INT,
8203 sgp->sg_cterm_fg, NULL, "ctermfg");
8204 didh = highlight_list_arg(id, didh, LIST_INT,
8205 sgp->sg_cterm_bg, NULL, "ctermbg");
8206
8207#ifdef FEAT_GUI
8208 didh = highlight_list_arg(id, didh, LIST_ATTR,
8209 sgp->sg_gui, NULL, "gui");
8210 didh = highlight_list_arg(id, didh, LIST_STRING,
8211 0, sgp->sg_gui_fg_name, "guifg");
8212 didh = highlight_list_arg(id, didh, LIST_STRING,
8213 0, sgp->sg_gui_bg_name, "guibg");
8214 didh = highlight_list_arg(id, didh, LIST_STRING,
Bram Moolenaar75c50c42005-06-04 22:06:24 +00008215 0, sgp->sg_gui_sp_name, "guisp");
8216 didh = highlight_list_arg(id, didh, LIST_STRING,
Bram Moolenaar071d4272004-06-13 20:20:40 +00008217 0, sgp->sg_font_name, "font");
8218#endif
8219
Bram Moolenaar661b1822005-07-28 22:36:45 +00008220 if (sgp->sg_link && !got_int)
Bram Moolenaar071d4272004-06-13 20:20:40 +00008221 {
8222 (void)syn_list_header(didh, 9999, id);
Bram Moolenaar661b1822005-07-28 22:36:45 +00008223 didh = TRUE;
Bram Moolenaar071d4272004-06-13 20:20:40 +00008224 msg_puts_attr((char_u *)"links to", hl_attr(HLF_D));
8225 msg_putchar(' ');
8226 msg_outtrans(HL_TABLE()[HL_TABLE()[id - 1].sg_link - 1].sg_name);
8227 }
Bram Moolenaar661b1822005-07-28 22:36:45 +00008228
Bram Moolenaar5e3cb7e2006-02-27 23:58:35 +00008229 if (!didh)
8230 highlight_list_arg(id, didh, LIST_STRING, 0, (char_u *)"cleared", "");
Bram Moolenaar661b1822005-07-28 22:36:45 +00008231#ifdef FEAT_EVAL
Bram Moolenaar5e3cb7e2006-02-27 23:58:35 +00008232 if (p_verbose > 0)
Bram Moolenaar661b1822005-07-28 22:36:45 +00008233 last_set_msg(sgp->sg_scriptID);
8234#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00008235}
8236
8237 static int
8238highlight_list_arg(id, didh, type, iarg, sarg, name)
8239 int id;
8240 int didh;
8241 int type;
8242 int iarg;
8243 char_u *sarg;
8244 char *name;
8245{
8246 char_u buf[100];
8247 char_u *ts;
8248 int i;
8249
Bram Moolenaar661b1822005-07-28 22:36:45 +00008250 if (got_int)
8251 return FALSE;
Bram Moolenaar071d4272004-06-13 20:20:40 +00008252 if (type == LIST_STRING ? (sarg != NULL) : (iarg != 0))
8253 {
8254 ts = buf;
8255 if (type == LIST_INT)
8256 sprintf((char *)buf, "%d", iarg - 1);
8257 else if (type == LIST_STRING)
8258 ts = sarg;
8259 else /* type == LIST_ATTR */
8260 {
8261 buf[0] = NUL;
8262 for (i = 0; hl_attr_table[i] != 0; ++i)
8263 {
8264 if (iarg & hl_attr_table[i])
8265 {
8266 if (buf[0] != NUL)
8267 STRCAT(buf, ",");
8268 STRCAT(buf, hl_name_table[i]);
8269 iarg &= ~hl_attr_table[i]; /* don't want "inverse" */
8270 }
8271 }
8272 }
8273
8274 (void)syn_list_header(didh,
8275 (int)(vim_strsize(ts) + STRLEN(name) + 1), id);
8276 didh = TRUE;
Bram Moolenaar661b1822005-07-28 22:36:45 +00008277 if (!got_int)
8278 {
Bram Moolenaar5e3cb7e2006-02-27 23:58:35 +00008279 if (*name != NUL)
8280 {
8281 MSG_PUTS_ATTR(name, hl_attr(HLF_D));
8282 MSG_PUTS_ATTR("=", hl_attr(HLF_D));
8283 }
Bram Moolenaar661b1822005-07-28 22:36:45 +00008284 msg_outtrans(ts);
8285 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00008286 }
8287 return didh;
8288}
8289
8290#if (((defined(FEAT_EVAL) || defined(FEAT_PRINTER))) && defined(FEAT_SYN_HL)) || defined(PROTO)
8291/*
8292 * Return "1" if highlight group "id" has attribute "flag".
8293 * Return NULL otherwise.
8294 */
8295 char_u *
8296highlight_has_attr(id, flag, modec)
8297 int id;
8298 int flag;
8299 int modec; /* 'g' for GUI, 'c' for cterm, 't' for term */
8300{
8301 int attr;
8302
8303 if (id <= 0 || id > highlight_ga.ga_len)
8304 return NULL;
8305
8306#ifdef FEAT_GUI
8307 if (modec == 'g')
8308 attr = HL_TABLE()[id - 1].sg_gui;
8309 else
8310#endif
8311 if (modec == 'c')
8312 attr = HL_TABLE()[id - 1].sg_cterm;
8313 else
8314 attr = HL_TABLE()[id - 1].sg_term;
8315
8316 if (attr & flag)
8317 return (char_u *)"1";
8318 return NULL;
8319}
8320#endif
8321
8322#if (defined(FEAT_SYN_HL) && defined(FEAT_EVAL)) || defined(PROTO)
8323/*
8324 * Return color name of highlight group "id".
8325 */
8326 char_u *
8327highlight_color(id, what, modec)
8328 int id;
Bram Moolenaare2cc9702005-03-15 22:43:58 +00008329 char_u *what; /* "fg", "bg", "sp", "fg#", "bg#" or "sp#" */
Bram Moolenaar071d4272004-06-13 20:20:40 +00008330 int modec; /* 'g' for GUI, 'c' for cterm, 't' for term */
8331{
8332 static char_u name[20];
8333 int n;
Bram Moolenaare2cc9702005-03-15 22:43:58 +00008334 int fg = FALSE;
8335# ifdef FEAT_GUI
8336 int sp = FALSE;
8337# endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00008338
8339 if (id <= 0 || id > highlight_ga.ga_len)
8340 return NULL;
8341
8342 if (TOLOWER_ASC(what[0]) == 'f')
8343 fg = TRUE;
Bram Moolenaare2cc9702005-03-15 22:43:58 +00008344# ifdef FEAT_GUI
8345 else if (TOLOWER_ASC(what[0]) == 's')
8346 sp = TRUE;
Bram Moolenaar071d4272004-06-13 20:20:40 +00008347 if (modec == 'g')
8348 {
8349 /* return #RRGGBB form (only possible when GUI is running) */
8350 if (gui.in_use && what[1] && what[2] == '#')
8351 {
8352 guicolor_T color;
8353 long_u rgb;
8354 static char_u buf[10];
8355
8356 if (fg)
8357 color = HL_TABLE()[id - 1].sg_gui_fg;
Bram Moolenaare2cc9702005-03-15 22:43:58 +00008358 else if (sp)
8359 color = HL_TABLE()[id - 1].sg_gui_sp;
Bram Moolenaar071d4272004-06-13 20:20:40 +00008360 else
8361 color = HL_TABLE()[id - 1].sg_gui_bg;
8362 if (color == INVALCOLOR)
8363 return NULL;
8364 rgb = gui_mch_get_rgb(color);
8365 sprintf((char *)buf, "#%02x%02x%02x",
8366 (unsigned)(rgb >> 16),
8367 (unsigned)(rgb >> 8) & 255,
8368 (unsigned)rgb & 255);
8369 return buf;
8370 }
8371 if (fg)
8372 return (HL_TABLE()[id - 1].sg_gui_fg_name);
Bram Moolenaare2cc9702005-03-15 22:43:58 +00008373 if (sp)
8374 return (HL_TABLE()[id - 1].sg_gui_sp_name);
Bram Moolenaar071d4272004-06-13 20:20:40 +00008375 return (HL_TABLE()[id - 1].sg_gui_bg_name);
8376 }
Bram Moolenaare2cc9702005-03-15 22:43:58 +00008377# endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00008378 if (modec == 'c')
8379 {
8380 if (fg)
8381 n = HL_TABLE()[id - 1].sg_cterm_fg - 1;
8382 else
8383 n = HL_TABLE()[id - 1].sg_cterm_bg - 1;
8384 sprintf((char *)name, "%d", n);
8385 return name;
8386 }
8387 /* term doesn't have color */
8388 return NULL;
8389}
8390#endif
8391
8392#if (defined(FEAT_SYN_HL) && defined(FEAT_GUI) && defined(FEAT_PRINTER)) \
8393 || defined(PROTO)
8394/*
8395 * Return color name of highlight group "id" as RGB value.
8396 */
8397 long_u
8398highlight_gui_color_rgb(id, fg)
8399 int id;
8400 int fg; /* TRUE = fg, FALSE = bg */
8401{
8402 guicolor_T color;
8403
8404 if (id <= 0 || id > highlight_ga.ga_len)
8405 return 0L;
8406
8407 if (fg)
8408 color = HL_TABLE()[id - 1].sg_gui_fg;
8409 else
8410 color = HL_TABLE()[id - 1].sg_gui_bg;
8411
8412 if (color == INVALCOLOR)
8413 return 0L;
8414
8415 return gui_mch_get_rgb(color);
8416}
8417#endif
8418
8419/*
8420 * Output the syntax list header.
8421 * Return TRUE when started a new line.
8422 */
8423 static int
8424syn_list_header(did_header, outlen, id)
8425 int did_header; /* did header already */
8426 int outlen; /* length of string that comes */
8427 int id; /* highlight group id */
8428{
8429 int endcol = 19;
8430 int newline = TRUE;
8431
8432 if (!did_header)
8433 {
8434 msg_putchar('\n');
Bram Moolenaar661b1822005-07-28 22:36:45 +00008435 if (got_int)
8436 return TRUE;
Bram Moolenaar071d4272004-06-13 20:20:40 +00008437 msg_outtrans(HL_TABLE()[id - 1].sg_name);
8438 endcol = 15;
8439 }
8440 else if (msg_col + outlen + 1 >= Columns)
Bram Moolenaar661b1822005-07-28 22:36:45 +00008441 {
Bram Moolenaar071d4272004-06-13 20:20:40 +00008442 msg_putchar('\n');
Bram Moolenaar661b1822005-07-28 22:36:45 +00008443 if (got_int)
8444 return TRUE;
8445 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00008446 else
8447 {
8448 if (msg_col >= endcol) /* wrap around is like starting a new line */
8449 newline = FALSE;
8450 }
8451
8452 if (msg_col >= endcol) /* output at least one space */
8453 endcol = msg_col + 1;
8454 if (Columns <= endcol) /* avoid hang for tiny window */
8455 endcol = Columns - 1;
8456
8457 msg_advance(endcol);
8458
8459 /* Show "xxx" with the attributes. */
8460 if (!did_header)
8461 {
8462 msg_puts_attr((char_u *)"xxx", syn_id2attr(id));
8463 msg_putchar(' ');
8464 }
8465
8466 return newline;
8467}
8468
8469/*
8470 * Set the attribute numbers for a highlight group.
8471 * Called after one of the attributes has changed.
8472 */
8473 static void
8474set_hl_attr(idx)
8475 int idx; /* index in array */
8476{
8477 attrentry_T at_en;
8478 struct hl_group *sgp = HL_TABLE() + idx;
8479
8480 /* The "Normal" group doesn't need an attribute number */
8481 if (sgp->sg_name_u != NULL && STRCMP(sgp->sg_name_u, "NORMAL") == 0)
8482 return;
8483
8484#ifdef FEAT_GUI
8485 /*
8486 * For the GUI mode: If there are other than "normal" highlighting
8487 * attributes, need to allocate an attr number.
8488 */
8489 if (sgp->sg_gui_fg == INVALCOLOR
8490 && sgp->sg_gui_bg == INVALCOLOR
Bram Moolenaare2cc9702005-03-15 22:43:58 +00008491 && sgp->sg_gui_sp == INVALCOLOR
Bram Moolenaar071d4272004-06-13 20:20:40 +00008492 && sgp->sg_font == NOFONT
8493# ifdef FEAT_XFONTSET
8494 && sgp->sg_fontset == NOFONTSET
8495# endif
8496 )
8497 {
8498 sgp->sg_gui_attr = sgp->sg_gui;
8499 }
8500 else
8501 {
8502 at_en.ae_attr = sgp->sg_gui;
8503 at_en.ae_u.gui.fg_color = sgp->sg_gui_fg;
8504 at_en.ae_u.gui.bg_color = sgp->sg_gui_bg;
Bram Moolenaare2cc9702005-03-15 22:43:58 +00008505 at_en.ae_u.gui.sp_color = sgp->sg_gui_sp;
Bram Moolenaar071d4272004-06-13 20:20:40 +00008506 at_en.ae_u.gui.font = sgp->sg_font;
8507# ifdef FEAT_XFONTSET
8508 at_en.ae_u.gui.fontset = sgp->sg_fontset;
8509# endif
8510 sgp->sg_gui_attr = get_attr_entry(&gui_attr_table, &at_en);
8511 }
8512#endif
8513 /*
8514 * For the term mode: If there are other than "normal" highlighting
8515 * attributes, need to allocate an attr number.
8516 */
8517 if (sgp->sg_start == NULL && sgp->sg_stop == NULL)
8518 sgp->sg_term_attr = sgp->sg_term;
8519 else
8520 {
8521 at_en.ae_attr = sgp->sg_term;
8522 at_en.ae_u.term.start = sgp->sg_start;
8523 at_en.ae_u.term.stop = sgp->sg_stop;
8524 sgp->sg_term_attr = get_attr_entry(&term_attr_table, &at_en);
8525 }
8526
8527 /*
8528 * For the color term mode: If there are other than "normal"
8529 * highlighting attributes, need to allocate an attr number.
8530 */
8531 if (sgp->sg_cterm_fg == 0 && sgp->sg_cterm_bg == 0)
8532 sgp->sg_cterm_attr = sgp->sg_cterm;
8533 else
8534 {
8535 at_en.ae_attr = sgp->sg_cterm;
8536 at_en.ae_u.cterm.fg_color = sgp->sg_cterm_fg;
8537 at_en.ae_u.cterm.bg_color = sgp->sg_cterm_bg;
8538 sgp->sg_cterm_attr = get_attr_entry(&cterm_attr_table, &at_en);
8539 }
8540}
8541
8542/*
8543 * Lookup a highlight group name and return it's ID.
8544 * If it is not found, 0 is returned.
8545 */
8546 int
8547syn_name2id(name)
8548 char_u *name;
8549{
8550 int i;
8551 char_u name_u[200];
8552
8553 /* Avoid using stricmp() too much, it's slow on some systems */
8554 /* Avoid alloc()/free(), these are slow too. ID names over 200 chars
8555 * don't deserve to be found! */
Bram Moolenaarce0842a2005-07-18 21:58:11 +00008556 vim_strncpy(name_u, name, 199);
Bram Moolenaar071d4272004-06-13 20:20:40 +00008557 vim_strup(name_u);
8558 for (i = highlight_ga.ga_len; --i >= 0; )
8559 if (HL_TABLE()[i].sg_name_u != NULL
8560 && STRCMP(name_u, HL_TABLE()[i].sg_name_u) == 0)
8561 break;
8562 return i + 1;
8563}
8564
8565#if defined(FEAT_EVAL) || defined(PROTO)
8566/*
8567 * Return TRUE if highlight group "name" exists.
8568 */
8569 int
8570highlight_exists(name)
8571 char_u *name;
8572{
8573 return (syn_name2id(name) > 0);
8574}
Bram Moolenaar910f66f2006-04-05 20:41:53 +00008575
8576# if defined(FEAT_SEARCH_EXTRA) || defined(PROTO)
8577/*
8578 * Return the name of highlight group "id".
8579 * When not a valid ID return an empty string.
8580 */
8581 char_u *
8582syn_id2name(id)
8583 int id;
8584{
Bram Moolenaar6ee10162007-07-26 20:58:42 +00008585 if (id <= 0 || id > highlight_ga.ga_len)
Bram Moolenaar910f66f2006-04-05 20:41:53 +00008586 return (char_u *)"";
8587 return HL_TABLE()[id - 1].sg_name;
8588}
8589# endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00008590#endif
8591
8592/*
8593 * Like syn_name2id(), but take a pointer + length argument.
8594 */
8595 int
8596syn_namen2id(linep, len)
8597 char_u *linep;
8598 int len;
8599{
8600 char_u *name;
8601 int id = 0;
8602
8603 name = vim_strnsave(linep, len);
8604 if (name != NULL)
8605 {
8606 id = syn_name2id(name);
8607 vim_free(name);
8608 }
8609 return id;
8610}
8611
8612/*
8613 * Find highlight group name in the table and return it's ID.
8614 * The argument is a pointer to the name and the length of the name.
8615 * If it doesn't exist yet, a new entry is created.
8616 * Return 0 for failure.
8617 */
8618 int
8619syn_check_group(pp, len)
8620 char_u *pp;
8621 int len;
8622{
8623 int id;
8624 char_u *name;
8625
8626 name = vim_strnsave(pp, len);
8627 if (name == NULL)
8628 return 0;
8629
8630 id = syn_name2id(name);
8631 if (id == 0) /* doesn't exist yet */
8632 id = syn_add_group(name);
8633 else
8634 vim_free(name);
8635 return id;
8636}
8637
8638/*
8639 * Add new highlight group and return it's ID.
8640 * "name" must be an allocated string, it will be consumed.
8641 * Return 0 for failure.
8642 */
8643 static int
8644syn_add_group(name)
8645 char_u *name;
8646{
8647 char_u *p;
8648
8649 /* Check that the name is ASCII letters, digits and underscore. */
8650 for (p = name; *p != NUL; ++p)
8651 {
8652 if (!vim_isprintc(*p))
8653 {
8654 EMSG(_("E669: Unprintable character in group name"));
Bram Moolenaarf5b63862009-12-16 17:13:44 +00008655 vim_free(name);
Bram Moolenaar071d4272004-06-13 20:20:40 +00008656 return 0;
8657 }
8658 else if (!ASCII_ISALNUM(*p) && *p != '_')
8659 {
8660 /* This is an error, but since there previously was no check only
8661 * give a warning. */
Bram Moolenaar2df6dcc2004-07-12 15:53:54 +00008662 msg_source(hl_attr(HLF_W));
Bram Moolenaar071d4272004-06-13 20:20:40 +00008663 MSG(_("W18: Invalid character in group name"));
8664 break;
8665 }
8666 }
8667
8668 /*
8669 * First call for this growarray: init growing array.
8670 */
8671 if (highlight_ga.ga_data == NULL)
8672 {
8673 highlight_ga.ga_itemsize = sizeof(struct hl_group);
8674 highlight_ga.ga_growsize = 10;
8675 }
8676
8677 /*
8678 * Make room for at least one other syntax_highlight entry.
8679 */
8680 if (ga_grow(&highlight_ga, 1) == FAIL)
8681 {
8682 vim_free(name);
8683 return 0;
8684 }
8685
8686 vim_memset(&(HL_TABLE()[highlight_ga.ga_len]), 0, sizeof(struct hl_group));
8687 HL_TABLE()[highlight_ga.ga_len].sg_name = name;
8688 HL_TABLE()[highlight_ga.ga_len].sg_name_u = vim_strsave_up(name);
8689#ifdef FEAT_GUI
8690 HL_TABLE()[highlight_ga.ga_len].sg_gui_bg = INVALCOLOR;
8691 HL_TABLE()[highlight_ga.ga_len].sg_gui_fg = INVALCOLOR;
Bram Moolenaare2cc9702005-03-15 22:43:58 +00008692 HL_TABLE()[highlight_ga.ga_len].sg_gui_sp = INVALCOLOR;
Bram Moolenaar071d4272004-06-13 20:20:40 +00008693#endif
8694 ++highlight_ga.ga_len;
Bram Moolenaar071d4272004-06-13 20:20:40 +00008695
8696 return highlight_ga.ga_len; /* ID is index plus one */
8697}
8698
8699/*
8700 * When, just after calling syn_add_group(), an error is discovered, this
8701 * function deletes the new name.
8702 */
8703 static void
8704syn_unadd_group()
8705{
8706 --highlight_ga.ga_len;
Bram Moolenaar071d4272004-06-13 20:20:40 +00008707 vim_free(HL_TABLE()[highlight_ga.ga_len].sg_name);
8708 vim_free(HL_TABLE()[highlight_ga.ga_len].sg_name_u);
8709}
8710
8711/*
8712 * Translate a group ID to highlight attributes.
8713 */
8714 int
8715syn_id2attr(hl_id)
8716 int hl_id;
8717{
8718 int attr;
8719 struct hl_group *sgp;
8720
8721 hl_id = syn_get_final_id(hl_id);
8722 sgp = &HL_TABLE()[hl_id - 1]; /* index is ID minus one */
8723
8724#ifdef FEAT_GUI
8725 /*
8726 * Only use GUI attr when the GUI is being used.
8727 */
8728 if (gui.in_use)
8729 attr = sgp->sg_gui_attr;
8730 else
8731#endif
8732 if (t_colors > 1)
8733 attr = sgp->sg_cterm_attr;
8734 else
8735 attr = sgp->sg_term_attr;
8736
8737 return attr;
8738}
8739
8740#ifdef FEAT_GUI
8741/*
8742 * Get the GUI colors and attributes for a group ID.
8743 * NOTE: the colors will be INVALCOLOR when not set, the color otherwise.
8744 */
8745 int
8746syn_id2colors(hl_id, fgp, bgp)
8747 int hl_id;
8748 guicolor_T *fgp;
8749 guicolor_T *bgp;
8750{
8751 struct hl_group *sgp;
8752
8753 hl_id = syn_get_final_id(hl_id);
8754 sgp = &HL_TABLE()[hl_id - 1]; /* index is ID minus one */
8755
8756 *fgp = sgp->sg_gui_fg;
8757 *bgp = sgp->sg_gui_bg;
8758 return sgp->sg_gui;
8759}
8760#endif
8761
8762/*
8763 * Translate a group ID to the final group ID (following links).
8764 */
8765 int
8766syn_get_final_id(hl_id)
8767 int hl_id;
8768{
8769 int count;
8770 struct hl_group *sgp;
8771
8772 if (hl_id > highlight_ga.ga_len || hl_id < 1)
8773 return 0; /* Can be called from eval!! */
8774
8775 /*
8776 * Follow links until there is no more.
8777 * Look out for loops! Break after 100 links.
8778 */
8779 for (count = 100; --count >= 0; )
8780 {
8781 sgp = &HL_TABLE()[hl_id - 1]; /* index is ID minus one */
8782 if (sgp->sg_link == 0 || sgp->sg_link > highlight_ga.ga_len)
8783 break;
8784 hl_id = sgp->sg_link;
8785 }
8786
8787 return hl_id;
8788}
8789
8790#ifdef FEAT_GUI
8791/*
8792 * Call this function just after the GUI has started.
8793 * It finds the font and color handles for the highlighting groups.
8794 */
8795 void
8796highlight_gui_started()
8797{
8798 int idx;
8799
8800 /* First get the colors from the "Normal" and "Menu" group, if set */
8801 set_normal_colors();
8802
8803 for (idx = 0; idx < highlight_ga.ga_len; ++idx)
8804 gui_do_one_color(idx, FALSE, FALSE);
8805
8806 highlight_changed();
8807}
8808
8809 static void
8810gui_do_one_color(idx, do_menu, do_tooltip)
8811 int idx;
8812 int do_menu; /* TRUE: might set the menu font */
8813 int do_tooltip; /* TRUE: might set the tooltip font */
8814{
8815 int didit = FALSE;
8816
8817 if (HL_TABLE()[idx].sg_font_name != NULL)
8818 {
8819 hl_do_font(idx, HL_TABLE()[idx].sg_font_name, FALSE, do_menu,
8820 do_tooltip);
8821 didit = TRUE;
8822 }
8823 if (HL_TABLE()[idx].sg_gui_fg_name != NULL)
8824 {
8825 HL_TABLE()[idx].sg_gui_fg =
8826 color_name2handle(HL_TABLE()[idx].sg_gui_fg_name);
8827 didit = TRUE;
8828 }
8829 if (HL_TABLE()[idx].sg_gui_bg_name != NULL)
8830 {
8831 HL_TABLE()[idx].sg_gui_bg =
8832 color_name2handle(HL_TABLE()[idx].sg_gui_bg_name);
8833 didit = TRUE;
8834 }
Bram Moolenaare2cc9702005-03-15 22:43:58 +00008835 if (HL_TABLE()[idx].sg_gui_sp_name != NULL)
8836 {
8837 HL_TABLE()[idx].sg_gui_sp =
8838 color_name2handle(HL_TABLE()[idx].sg_gui_sp_name);
8839 didit = TRUE;
8840 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00008841 if (didit) /* need to get a new attr number */
8842 set_hl_attr(idx);
8843}
8844
8845#endif
8846
8847/*
8848 * Translate the 'highlight' option into attributes in highlight_attr[] and
8849 * set up the user highlights User1..9. If FEAT_STL_OPT is in use, a set of
8850 * corresponding highlights to use on top of HLF_SNC is computed.
8851 * Called only when the 'highlight' option has been changed and upon first
8852 * screen redraw after any :highlight command.
8853 * Return FAIL when an invalid flag is found in 'highlight'. OK otherwise.
8854 */
8855 int
8856highlight_changed()
8857{
8858 int hlf;
8859 int i;
8860 char_u *p;
8861 int attr;
8862 char_u *end;
8863 int id;
8864#ifdef USER_HIGHLIGHT
8865 char_u userhl[10];
8866# ifdef FEAT_STL_OPT
8867 int id_SNC = -1;
8868 int id_S = -1;
8869 int hlcnt;
8870# endif
8871#endif
8872 static int hl_flags[HLF_COUNT] = HL_FLAGS;
8873
8874 need_highlight_changed = FALSE;
8875
8876 /*
8877 * Clear all attributes.
8878 */
8879 for (hlf = 0; hlf < (int)HLF_COUNT; ++hlf)
8880 highlight_attr[hlf] = 0;
8881
8882 /*
8883 * First set all attributes to their default value.
8884 * Then use the attributes from the 'highlight' option.
8885 */
8886 for (i = 0; i < 2; ++i)
8887 {
8888 if (i)
8889 p = p_hl;
8890 else
8891 p = get_highlight_default();
8892 if (p == NULL) /* just in case */
8893 continue;
8894
8895 while (*p)
8896 {
8897 for (hlf = 0; hlf < (int)HLF_COUNT; ++hlf)
8898 if (hl_flags[hlf] == *p)
8899 break;
8900 ++p;
8901 if (hlf == (int)HLF_COUNT || *p == NUL)
8902 return FAIL;
8903
8904 /*
8905 * Allow several hl_flags to be combined, like "bu" for
8906 * bold-underlined.
8907 */
8908 attr = 0;
8909 for ( ; *p && *p != ','; ++p) /* parse upto comma */
8910 {
8911 if (vim_iswhite(*p)) /* ignore white space */
8912 continue;
8913
8914 if (attr > HL_ALL) /* Combination with ':' is not allowed. */
8915 return FAIL;
8916
8917 switch (*p)
8918 {
8919 case 'b': attr |= HL_BOLD;
8920 break;
8921 case 'i': attr |= HL_ITALIC;
8922 break;
8923 case '-':
8924 case 'n': /* no highlighting */
8925 break;
8926 case 'r': attr |= HL_INVERSE;
8927 break;
8928 case 's': attr |= HL_STANDOUT;
8929 break;
8930 case 'u': attr |= HL_UNDERLINE;
8931 break;
Bram Moolenaare2cc9702005-03-15 22:43:58 +00008932 case 'c': attr |= HL_UNDERCURL;
8933 break;
Bram Moolenaar071d4272004-06-13 20:20:40 +00008934 case ':': ++p; /* highlight group name */
8935 if (attr || *p == NUL) /* no combinations */
8936 return FAIL;
8937 end = vim_strchr(p, ',');
8938 if (end == NULL)
8939 end = p + STRLEN(p);
8940 id = syn_check_group(p, (int)(end - p));
8941 if (id == 0)
8942 return FAIL;
8943 attr = syn_id2attr(id);
8944 p = end - 1;
8945#if defined(FEAT_STL_OPT) && defined(USER_HIGHLIGHT)
8946 if (hlf == (int)HLF_SNC)
8947 id_SNC = syn_get_final_id(id);
8948 else if (hlf == (int)HLF_S)
8949 id_S = syn_get_final_id(id);
8950#endif
8951 break;
8952 default: return FAIL;
8953 }
8954 }
8955 highlight_attr[hlf] = attr;
8956
8957 p = skip_to_option_part(p); /* skip comma and spaces */
8958 }
8959 }
8960
8961#ifdef USER_HIGHLIGHT
8962 /* Setup the user highlights
8963 *
8964 * Temporarily utilize 10 more hl entries. Have to be in there
8965 * simultaneously in case of table overflows in get_attr_entry()
8966 */
8967# ifdef FEAT_STL_OPT
8968 if (ga_grow(&highlight_ga, 10) == FAIL)
8969 return FAIL;
8970 hlcnt = highlight_ga.ga_len;
8971 if (id_S == 0)
8972 { /* Make sure id_S is always valid to simplify code below */
8973 memset(&HL_TABLE()[hlcnt + 9], 0, sizeof(struct hl_group));
8974 HL_TABLE()[hlcnt + 9].sg_term = highlight_attr[HLF_S];
8975 id_S = hlcnt + 10;
8976 }
8977# endif
8978 for (i = 0; i < 9; i++)
8979 {
8980 sprintf((char *)userhl, "User%d", i + 1);
8981 id = syn_name2id(userhl);
8982 if (id == 0)
8983 {
8984 highlight_user[i] = 0;
8985# ifdef FEAT_STL_OPT
8986 highlight_stlnc[i] = 0;
8987# endif
8988 }
8989 else
8990 {
8991# ifdef FEAT_STL_OPT
8992 struct hl_group *hlt = HL_TABLE();
8993# endif
8994
8995 highlight_user[i] = syn_id2attr(id);
8996# ifdef FEAT_STL_OPT
8997 if (id_SNC == 0)
8998 {
8999 memset(&hlt[hlcnt + i], 0, sizeof(struct hl_group));
9000 hlt[hlcnt + i].sg_term = highlight_attr[HLF_SNC];
9001 hlt[hlcnt + i].sg_cterm = highlight_attr[HLF_SNC];
9002# ifdef FEAT_GUI
9003 hlt[hlcnt + i].sg_gui = highlight_attr[HLF_SNC];
9004# endif
9005 }
9006 else
9007 mch_memmove(&hlt[hlcnt + i],
9008 &hlt[id_SNC - 1],
9009 sizeof(struct hl_group));
9010 hlt[hlcnt + i].sg_link = 0;
9011
9012 /* Apply difference between UserX and HLF_S to HLF_SNC */
9013 hlt[hlcnt + i].sg_term ^=
9014 hlt[id - 1].sg_term ^ hlt[id_S - 1].sg_term;
9015 if (hlt[id - 1].sg_start != hlt[id_S - 1].sg_start)
9016 hlt[hlcnt + i].sg_start = hlt[id - 1].sg_start;
9017 if (hlt[id - 1].sg_stop != hlt[id_S - 1].sg_stop)
9018 hlt[hlcnt + i].sg_stop = hlt[id - 1].sg_stop;
9019 hlt[hlcnt + i].sg_cterm ^=
9020 hlt[id - 1].sg_cterm ^ hlt[id_S - 1].sg_cterm;
9021 if (hlt[id - 1].sg_cterm_fg != hlt[id_S - 1].sg_cterm_fg)
9022 hlt[hlcnt + i].sg_cterm_fg = hlt[id - 1].sg_cterm_fg;
9023 if (hlt[id - 1].sg_cterm_bg != hlt[id_S - 1].sg_cterm_bg)
9024 hlt[hlcnt + i].sg_cterm_bg = hlt[id - 1].sg_cterm_bg;
9025# ifdef FEAT_GUI
9026 hlt[hlcnt + i].sg_gui ^=
9027 hlt[id - 1].sg_gui ^ hlt[id_S - 1].sg_gui;
9028 if (hlt[id - 1].sg_gui_fg != hlt[id_S - 1].sg_gui_fg)
9029 hlt[hlcnt + i].sg_gui_fg = hlt[id - 1].sg_gui_fg;
9030 if (hlt[id - 1].sg_gui_bg != hlt[id_S - 1].sg_gui_bg)
9031 hlt[hlcnt + i].sg_gui_bg = hlt[id - 1].sg_gui_bg;
Bram Moolenaare2cc9702005-03-15 22:43:58 +00009032 if (hlt[id - 1].sg_gui_sp != hlt[id_S - 1].sg_gui_sp)
9033 hlt[hlcnt + i].sg_gui_sp = hlt[id - 1].sg_gui_sp;
Bram Moolenaar071d4272004-06-13 20:20:40 +00009034 if (hlt[id - 1].sg_font != hlt[id_S - 1].sg_font)
9035 hlt[hlcnt + i].sg_font = hlt[id - 1].sg_font;
9036# ifdef FEAT_XFONTSET
9037 if (hlt[id - 1].sg_fontset != hlt[id_S - 1].sg_fontset)
9038 hlt[hlcnt + i].sg_fontset = hlt[id - 1].sg_fontset;
9039# endif
9040# endif
9041 highlight_ga.ga_len = hlcnt + i + 1;
9042 set_hl_attr(hlcnt + i); /* At long last we can apply */
9043 highlight_stlnc[i] = syn_id2attr(hlcnt + i + 1);
9044# endif
9045 }
9046 }
9047# ifdef FEAT_STL_OPT
9048 highlight_ga.ga_len = hlcnt;
9049# endif
9050
9051#endif /* USER_HIGHLIGHT */
9052
9053 return OK;
9054}
9055
Bram Moolenaar4f688582007-07-24 12:34:30 +00009056#if defined(FEAT_CMDL_COMPL) || defined(PROTO)
Bram Moolenaar071d4272004-06-13 20:20:40 +00009057
9058static void highlight_list __ARGS((void));
9059static void highlight_list_two __ARGS((int cnt, int attr));
9060
9061/*
9062 * Handle command line completion for :highlight command.
9063 */
9064 void
9065set_context_in_highlight_cmd(xp, arg)
9066 expand_T *xp;
9067 char_u *arg;
9068{
9069 char_u *p;
9070
9071 /* Default: expand group names */
9072 xp->xp_context = EXPAND_HIGHLIGHT;
9073 xp->xp_pattern = arg;
Bram Moolenaar4f688582007-07-24 12:34:30 +00009074 include_link = 2;
9075 include_default = 1;
Bram Moolenaar071d4272004-06-13 20:20:40 +00009076
9077 /* (part of) subcommand already typed */
9078 if (*arg != NUL)
9079 {
9080 p = skiptowhite(arg);
9081 if (*p != NUL) /* past "default" or group name */
9082 {
Bram Moolenaar4f688582007-07-24 12:34:30 +00009083 include_default = 0;
Bram Moolenaar071d4272004-06-13 20:20:40 +00009084 if (STRNCMP("default", arg, p - arg) == 0)
9085 {
9086 arg = skipwhite(p);
9087 xp->xp_pattern = arg;
9088 p = skiptowhite(arg);
9089 }
9090 if (*p != NUL) /* past group name */
9091 {
Bram Moolenaar4f688582007-07-24 12:34:30 +00009092 include_link = 0;
Bram Moolenaar071d4272004-06-13 20:20:40 +00009093 if (arg[1] == 'i' && arg[0] == 'N')
9094 highlight_list();
9095 if (STRNCMP("link", arg, p - arg) == 0
9096 || STRNCMP("clear", arg, p - arg) == 0)
9097 {
9098 xp->xp_pattern = skipwhite(p);
9099 p = skiptowhite(xp->xp_pattern);
9100 if (*p != NUL) /* past first group name */
9101 {
9102 xp->xp_pattern = skipwhite(p);
9103 p = skiptowhite(xp->xp_pattern);
9104 }
9105 }
9106 if (*p != NUL) /* past group name(s) */
9107 xp->xp_context = EXPAND_NOTHING;
9108 }
9109 }
9110 }
9111}
9112
9113/*
9114 * List highlighting matches in a nice way.
9115 */
9116 static void
9117highlight_list()
9118{
9119 int i;
9120
9121 for (i = 10; --i >= 0; )
9122 highlight_list_two(i, hl_attr(HLF_D));
9123 for (i = 40; --i >= 0; )
9124 highlight_list_two(99, 0);
9125}
9126
9127 static void
9128highlight_list_two(cnt, attr)
9129 int cnt;
9130 int attr;
9131{
9132 msg_puts_attr((char_u *)("N \bI \b! \b" + cnt / 11), attr);
9133 msg_clr_eos();
9134 out_flush();
9135 ui_delay(cnt == 99 ? 40L : (long)cnt * 50L, FALSE);
9136}
9137
9138#endif /* FEAT_CMDL_COMPL */
9139
9140#if defined(FEAT_CMDL_COMPL) || (defined(FEAT_SYN_HL) && defined(FEAT_EVAL)) \
9141 || defined(FEAT_SIGNS) || defined(PROTO)
9142/*
9143 * Function given to ExpandGeneric() to obtain the list of group names.
9144 * Also used for synIDattr() function.
9145 */
Bram Moolenaar071d4272004-06-13 20:20:40 +00009146 char_u *
9147get_highlight_name(xp, idx)
Bram Moolenaar2c4278f2009-05-17 11:33:22 +00009148 expand_T *xp UNUSED;
Bram Moolenaar071d4272004-06-13 20:20:40 +00009149 int idx;
9150{
Bram Moolenaar071d4272004-06-13 20:20:40 +00009151#ifdef FEAT_CMDL_COMPL
Bram Moolenaar4f688582007-07-24 12:34:30 +00009152 if (idx == highlight_ga.ga_len && include_none != 0)
9153 return (char_u *)"none";
9154 if (idx == highlight_ga.ga_len + include_none && include_default != 0)
Bram Moolenaar071d4272004-06-13 20:20:40 +00009155 return (char_u *)"default";
Bram Moolenaar4f688582007-07-24 12:34:30 +00009156 if (idx == highlight_ga.ga_len + include_none + include_default
9157 && include_link != 0)
9158 return (char_u *)"link";
9159 if (idx == highlight_ga.ga_len + include_none + include_default + 1
9160 && include_link != 0)
9161 return (char_u *)"clear";
9162#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00009163 if (idx < 0 || idx >= highlight_ga.ga_len)
9164 return NULL;
9165 return HL_TABLE()[idx].sg_name;
9166}
9167#endif
9168
Bram Moolenaar4f688582007-07-24 12:34:30 +00009169#if defined(FEAT_GUI) || defined(PROTO)
Bram Moolenaar071d4272004-06-13 20:20:40 +00009170/*
9171 * Free all the highlight group fonts.
9172 * Used when quitting for systems which need it.
9173 */
9174 void
9175free_highlight_fonts()
9176{
9177 int idx;
9178
9179 for (idx = 0; idx < highlight_ga.ga_len; ++idx)
9180 {
9181 gui_mch_free_font(HL_TABLE()[idx].sg_font);
9182 HL_TABLE()[idx].sg_font = NOFONT;
9183# ifdef FEAT_XFONTSET
9184 gui_mch_free_fontset(HL_TABLE()[idx].sg_fontset);
9185 HL_TABLE()[idx].sg_fontset = NOFONTSET;
9186# endif
9187 }
9188
9189 gui_mch_free_font(gui.norm_font);
9190# ifdef FEAT_XFONTSET
9191 gui_mch_free_fontset(gui.fontset);
9192# endif
9193# ifndef HAVE_GTK2
9194 gui_mch_free_font(gui.bold_font);
9195 gui_mch_free_font(gui.ital_font);
9196 gui_mch_free_font(gui.boldital_font);
9197# endif
9198}
9199#endif
9200
9201/**************************************
9202 * End of Highlighting stuff *
9203 **************************************/