blob: 370bc761b6acbf736ad4ceded84e222dad7948cc [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 */
Bram Moolenaar860cae12010-06-05 23:22:07 +0200143 int sp_flags; /* see HL_ defines below */
144#ifdef FEAT_CONCEAL
145 int sp_char; /* conceal substitute character */
146#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +0000147 struct sp_syn sp_syn; /* struct passed to in_id_list() */
148 short sp_syn_match_id; /* highlight group ID of pattern */
149 char_u *sp_pattern; /* regexp to match, pattern */
150 regprog_T *sp_prog; /* regexp to match, program */
151 int sp_ic; /* ignore-case flag for sp_prog */
152 short sp_off_flags; /* see below */
153 int sp_offsets[SPO_COUNT]; /* offsets */
154 short *sp_cont_list; /* cont. group IDs, if non-zero */
155 short *sp_next_list; /* next group IDs, if non-zero */
156 int sp_sync_idx; /* sync item index (syncing only) */
157 int sp_line_id; /* ID of last line where tried */
158 int sp_startcol; /* next match in sp_line_id line */
159} synpat_T;
160
161/* The sp_off_flags are computed like this:
162 * offset from the start of the matched text: (1 << SPO_XX_OFF)
163 * offset from the end of the matched text: (1 << (SPO_XX_OFF + SPO_COUNT))
164 * When both are present, only one is used.
165 */
166
167#define SPTYPE_MATCH 1 /* match keyword with this group ID */
168#define SPTYPE_START 2 /* match a regexp, start of item */
169#define SPTYPE_END 3 /* match a regexp, end of item */
170#define SPTYPE_SKIP 4 /* match a regexp, skip within item */
171
Bram Moolenaar071d4272004-06-13 20:20:40 +0000172
173#define SYN_ITEMS(buf) ((synpat_T *)((buf)->b_syn_patterns.ga_data))
174
175#define NONE_IDX -2 /* value of sp_sync_idx for "NONE" */
176
177/*
178 * Flags for b_syn_sync_flags:
179 */
180#define SF_CCOMMENT 0x01 /* sync on a C-style comment */
181#define SF_MATCH 0x02 /* sync by matching a pattern */
182
183#define SYN_STATE_P(ssp) ((bufstate_T *)((ssp)->ga_data))
184
Bram Moolenaar071d4272004-06-13 20:20:40 +0000185#define MAXKEYWLEN 80 /* maximum length of a keyword */
186
187/*
188 * The attributes of the syntax item that has been recognized.
189 */
190static int current_attr = 0; /* attr of current syntax word */
191#ifdef FEAT_EVAL
192static int current_id = 0; /* ID of current char for syn_get_id() */
Bram Moolenaarf5b63862009-12-16 17:13:44 +0000193static int current_trans_id = 0; /* idem, transparency removed */
Bram Moolenaar071d4272004-06-13 20:20:40 +0000194#endif
Bram Moolenaar860cae12010-06-05 23:22:07 +0200195#ifdef FEAT_CONCEAL
196static int current_flags = 0;
197static int current_sub_char = 0;
198#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +0000199
Bram Moolenaar217ad922005-03-20 22:37:15 +0000200typedef struct syn_cluster_S
Bram Moolenaar071d4272004-06-13 20:20:40 +0000201{
202 char_u *scl_name; /* syntax cluster name */
203 char_u *scl_name_u; /* uppercase of scl_name */
204 short *scl_list; /* IDs in this syntax cluster */
Bram Moolenaar217ad922005-03-20 22:37:15 +0000205} syn_cluster_T;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000206
207/*
208 * Methods of combining two clusters
209 */
210#define CLUSTER_REPLACE 1 /* replace first list with second */
211#define CLUSTER_ADD 2 /* add second list to first */
212#define CLUSTER_SUBTRACT 3 /* subtract second list from first */
213
Bram Moolenaar217ad922005-03-20 22:37:15 +0000214#define SYN_CLSTR(buf) ((syn_cluster_T *)((buf)->b_syn_clusters.ga_data))
Bram Moolenaar071d4272004-06-13 20:20:40 +0000215
216/*
217 * Syntax group IDs have different types:
218 * 0 - 9999 normal syntax groups
219 * 10000 - 14999 ALLBUT indicator (current_syn_inc_tag added)
220 * 15000 - 19999 TOP indicator (current_syn_inc_tag added)
221 * 20000 - 24999 CONTAINED indicator (current_syn_inc_tag added)
222 * >= 25000 cluster IDs (subtract SYNID_CLUSTER for the cluster ID)
223 */
224#define SYNID_ALLBUT 10000 /* syntax group ID for contains=ALLBUT */
225#define SYNID_TOP 15000 /* syntax group ID for contains=TOP */
226#define SYNID_CONTAINED 20000 /* syntax group ID for contains=CONTAINED */
227#define SYNID_CLUSTER 25000 /* first syntax group ID for clusters */
228
229/*
230 * Annoying Hack(TM): ":syn include" needs this pointer to pass to
231 * expand_filename(). Most of the other syntax commands don't need it, so
232 * instead of passing it to them, we stow it here.
233 */
234static char_u **syn_cmdlinep;
235
236/*
237 * Another Annoying Hack(TM): To prevent rules from other ":syn include"'d
Bram Moolenaar56be9502010-06-06 14:20:26 +0200238 * files from leaking into ALLBUT lists, we assign a unique ID to the
Bram Moolenaar071d4272004-06-13 20:20:40 +0000239 * rules in each ":syn include"'d file.
240 */
241static int current_syn_inc_tag = 0;
242static int running_syn_inc_tag = 0;
243
244/*
Bram Moolenaardad6b692005-01-25 22:14:34 +0000245 * In a hashtable item "hi_key" points to "keyword" in a keyentry.
246 * This avoids adding a pointer to the hashtable item.
247 * KE2HIKEY() converts a var pointer to a hashitem key pointer.
248 * HIKEY2KE() converts a hashitem key pointer to a var pointer.
249 * HI2KE() converts a hashitem pointer to a var pointer.
250 */
251static keyentry_T dumkey;
252#define KE2HIKEY(kp) ((kp)->keyword)
253#define HIKEY2KE(p) ((keyentry_T *)((p) - (dumkey.keyword - (char_u *)&dumkey)))
254#define HI2KE(hi) HIKEY2KE((hi)->hi_key)
255
256/*
Bram Moolenaar071d4272004-06-13 20:20:40 +0000257 * To reduce the time spent in keepend(), remember at which level in the state
258 * stack the first item with "keepend" is present. When "-1", there is no
259 * "keepend" on the stack.
260 */
261static int keepend_level = -1;
262
263/*
264 * For the current state we need to remember more than just the idx.
265 * When si_m_endpos.lnum is 0, the items other than si_idx are unknown.
266 * (The end positions have the column number of the next char)
267 */
268typedef struct state_item
269{
Bram Moolenaar3a36cf72007-08-21 15:29:56 +0000270 int si_idx; /* index of syntax pattern or
271 KEYWORD_IDX */
Bram Moolenaar071d4272004-06-13 20:20:40 +0000272 int si_id; /* highlight group ID for keywords */
Bram Moolenaarf5b63862009-12-16 17:13:44 +0000273 int si_trans_id; /* idem, transparency removed */
Bram Moolenaar071d4272004-06-13 20:20:40 +0000274 int si_m_lnum; /* lnum of the match */
275 int si_m_startcol; /* starting column of the match */
276 lpos_T si_m_endpos; /* just after end posn of the match */
277 lpos_T si_h_startpos; /* start position of the highlighting */
278 lpos_T si_h_endpos; /* end position of the highlighting */
279 lpos_T si_eoe_pos; /* end position of end pattern */
280 int si_end_idx; /* group ID for end pattern or zero */
281 int si_ends; /* if match ends before si_m_endpos */
282 int si_attr; /* attributes in this state */
283 long si_flags; /* HL_HAS_EOL flag in this state, and
284 * HL_SKIP* for si_next_list */
Bram Moolenaar860cae12010-06-05 23:22:07 +0200285#ifdef FEAT_CONCEAL
286 int si_char; /* substitution character for conceal */
287#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +0000288 short *si_cont_list; /* list of contained groups */
289 short *si_next_list; /* nextgroup IDs after this item ends */
290 reg_extmatch_T *si_extmatch; /* \z(...\) matches from start
291 * pattern */
292} stateitem_T;
293
294#define KEYWORD_IDX -1 /* value of si_idx for keywords */
295#define ID_LIST_ALL (short *)-1 /* valid of si_cont_list for containing all
296 but contained groups */
297
298/*
Bram Moolenaar6ac54292005-02-02 23:07:25 +0000299 * Struct to reduce the number of arguments to get_syn_options(), it's used
300 * very often.
301 */
302typedef struct
303{
Bram Moolenaarb8017e72007-05-10 18:59:07 +0000304 int flags; /* flags for contained and transparent */
Bram Moolenaar6ac54292005-02-02 23:07:25 +0000305 int keyword; /* TRUE for ":syn keyword" */
306 int *sync_idx; /* syntax item for "grouphere" argument, NULL
307 if not allowed */
308 char has_cont_list; /* TRUE if "cont_list" can be used */
309 short *cont_list; /* group IDs for "contains" argument */
310 short *cont_in_list; /* group IDs for "containedin" argument */
311 short *next_list; /* group IDs for "nextgroup" argument */
312} syn_opt_arg_T;
313
314/*
Bram Moolenaar071d4272004-06-13 20:20:40 +0000315 * The next possible match in the current line for any pattern is remembered,
316 * to avoid having to try for a match in each column.
317 * If next_match_idx == -1, not tried (in this line) yet.
318 * If next_match_col == MAXCOL, no match found in this line.
319 * (All end positions have the column of the char after the end)
320 */
321static int next_match_col; /* column for start of next match */
322static lpos_T next_match_m_endpos; /* position for end of next match */
323static lpos_T next_match_h_startpos; /* pos. for highl. start of next match */
324static lpos_T next_match_h_endpos; /* pos. for highl. end of next match */
325static int next_match_idx; /* index of matched item */
326static long next_match_flags; /* flags for next match */
327static lpos_T next_match_eos_pos; /* end of start pattn (start region) */
328static lpos_T next_match_eoe_pos; /* pos. for end of end pattern */
329static int next_match_end_idx; /* ID of group for end pattn or zero */
330static reg_extmatch_T *next_match_extmatch = NULL;
331
332/*
333 * A state stack is an array of integers or stateitem_T, stored in a
334 * garray_T. A state stack is invalid if it's itemsize entry is zero.
335 */
336#define INVALID_STATE(ssp) ((ssp)->ga_itemsize == 0)
337#define VALID_STATE(ssp) ((ssp)->ga_itemsize != 0)
338
339/*
340 * The current state (within the line) of the recognition engine.
341 * When current_state.ga_itemsize is 0 the current state is invalid.
342 */
343static win_T *syn_win; /* current window for highlighting */
344static buf_T *syn_buf; /* current buffer for highlighting */
Bram Moolenaar860cae12010-06-05 23:22:07 +0200345static synblock_T *syn_block; /* current buffer for highlighting */
Bram Moolenaar071d4272004-06-13 20:20:40 +0000346static linenr_T current_lnum = 0; /* lnum of current state */
347static colnr_T current_col = 0; /* column of current state */
348static int current_state_stored = 0; /* TRUE if stored current state
349 * after setting current_finished */
350static int current_finished = 0; /* current line has been finished */
351static garray_T current_state /* current stack of state_items */
352 = {0, 0, 0, 0, NULL};
353static short *current_next_list = NULL; /* when non-zero, nextgroup list */
354static int current_next_flags = 0; /* flags for current_next_list */
355static int current_line_id = 0; /* unique number for current line */
356
357#define CUR_STATE(idx) ((stateitem_T *)(current_state.ga_data))[idx]
358
359static void syn_sync __ARGS((win_T *wp, linenr_T lnum, synstate_T *last_valid));
360static int syn_match_linecont __ARGS((linenr_T lnum));
361static void syn_start_line __ARGS((void));
362static void syn_update_ends __ARGS((int startofline));
363static void syn_stack_alloc __ARGS((void));
364static int syn_stack_cleanup __ARGS((void));
Bram Moolenaar860cae12010-06-05 23:22:07 +0200365static void syn_stack_free_entry __ARGS((synblock_T *block, synstate_T *p));
Bram Moolenaar071d4272004-06-13 20:20:40 +0000366static synstate_T *syn_stack_find_entry __ARGS((linenr_T lnum));
Bram Moolenaardbe31752008-01-13 16:40:19 +0000367static synstate_T *store_current_state __ARGS((void));
Bram Moolenaar071d4272004-06-13 20:20:40 +0000368static void load_current_state __ARGS((synstate_T *from));
369static void invalidate_current_state __ARGS((void));
370static int syn_stack_equal __ARGS((synstate_T *sp));
371static void validate_current_state __ARGS((void));
372static int syn_finish_line __ARGS((int syncing));
Bram Moolenaar56cefaf2008-01-12 15:47:10 +0000373static int syn_current_attr __ARGS((int syncing, int displaying, int *can_spell, int keep_state));
Bram Moolenaar071d4272004-06-13 20:20:40 +0000374static int did_match_already __ARGS((int idx, garray_T *gap));
375static stateitem_T *push_next_match __ARGS((stateitem_T *cur_si));
376static void check_state_ends __ARGS((void));
377static void update_si_attr __ARGS((int idx));
378static void check_keepend __ARGS((void));
379static void update_si_end __ARGS((stateitem_T *sip, int startcol, int force));
380static short *copy_id_list __ARGS((short *list));
381static int in_id_list __ARGS((stateitem_T *item, short *cont_list, struct sp_syn *ssp, int contained));
382static int push_current_state __ARGS((int idx));
383static void pop_current_state __ARGS((void));
384
Bram Moolenaar860cae12010-06-05 23:22:07 +0200385static void syn_stack_apply_changes_block __ARGS((synblock_T *block, buf_T *buf));
Bram Moolenaar071d4272004-06-13 20:20:40 +0000386static 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));
387static void clear_syn_state __ARGS((synstate_T *p));
388static void clear_current_state __ARGS((void));
389
390static void limit_pos __ARGS((lpos_T *pos, lpos_T *limit));
391static void limit_pos_zero __ARGS((lpos_T *pos, lpos_T *limit));
392static void syn_add_end_off __ARGS((lpos_T *result, regmmatch_T *regmatch, synpat_T *spp, int idx, int extra));
393static void syn_add_start_off __ARGS((lpos_T *result, regmmatch_T *regmatch, synpat_T *spp, int idx, int extra));
394static char_u *syn_getcurline __ARGS((void));
395static int syn_regexec __ARGS((regmmatch_T *rmp, linenr_T lnum, colnr_T col));
Bram Moolenaar860cae12010-06-05 23:22:07 +0200396static int check_keyword_id __ARGS((char_u *line, int startcol, int *endcol, long *flags, short **next_list, stateitem_T *cur_si, int *ccharp));
Bram Moolenaar071d4272004-06-13 20:20:40 +0000397static void syn_cmd_case __ARGS((exarg_T *eap, int syncing));
Bram Moolenaarce0842a2005-07-18 21:58:11 +0000398static void syn_cmd_spell __ARGS((exarg_T *eap, int syncing));
Bram Moolenaar071d4272004-06-13 20:20:40 +0000399static void syntax_sync_clear __ARGS((void));
Bram Moolenaar860cae12010-06-05 23:22:07 +0200400static void syn_remove_pattern __ARGS((synblock_T *block, int idx));
401static void syn_clear_pattern __ARGS((synblock_T *block, int i));
402static void syn_clear_cluster __ARGS((synblock_T *block, int i));
Bram Moolenaar071d4272004-06-13 20:20:40 +0000403static void syn_cmd_clear __ARGS((exarg_T *eap, int syncing));
Bram Moolenaar860cae12010-06-05 23:22:07 +0200404static void syn_cmd_conceal __ARGS((exarg_T *eap, int syncing));
Bram Moolenaar071d4272004-06-13 20:20:40 +0000405static void syn_clear_one __ARGS((int id, int syncing));
406static void syn_cmd_on __ARGS((exarg_T *eap, int syncing));
407static void syn_cmd_enable __ARGS((exarg_T *eap, int syncing));
408static void syn_cmd_reset __ARGS((exarg_T *eap, int syncing));
409static void syn_cmd_manual __ARGS((exarg_T *eap, int syncing));
410static void syn_cmd_off __ARGS((exarg_T *eap, int syncing));
411static void syn_cmd_onoff __ARGS((exarg_T *eap, char *name));
412static void syn_cmd_list __ARGS((exarg_T *eap, int syncing));
413static void syn_lines_msg __ARGS((void));
414static void syn_match_msg __ARGS((void));
Bram Moolenaar860cae12010-06-05 23:22:07 +0200415static void syn_stack_free_block __ARGS((synblock_T *block));
Bram Moolenaar071d4272004-06-13 20:20:40 +0000416static void syn_list_one __ARGS((int id, int syncing, int link_only));
417static void syn_list_cluster __ARGS((int id));
418static void put_id_list __ARGS((char_u *name, short *list, int attr));
419static void put_pattern __ARGS((char *s, int c, synpat_T *spp, int attr));
Bram Moolenaardad6b692005-01-25 22:14:34 +0000420static int syn_list_keywords __ARGS((int id, hashtab_T *ht, int did_header, int attr));
421static void syn_clear_keyword __ARGS((int id, hashtab_T *ht));
422static void clear_keywtab __ARGS((hashtab_T *ht));
Bram Moolenaar860cae12010-06-05 23:22:07 +0200423static void add_keyword __ARGS((char_u *name, int id, int flags, short *cont_in_list, short *next_list, int conceal_char));
Bram Moolenaar071d4272004-06-13 20:20:40 +0000424static char_u *get_group_name __ARGS((char_u *arg, char_u **name_end));
Bram Moolenaar860cae12010-06-05 23:22:07 +0200425static char_u *get_syn_options __ARGS((char_u *arg, syn_opt_arg_T *opt, int *conceal_char));
Bram Moolenaar071d4272004-06-13 20:20:40 +0000426static void syn_cmd_include __ARGS((exarg_T *eap, int syncing));
427static void syn_cmd_keyword __ARGS((exarg_T *eap, int syncing));
428static void syn_cmd_match __ARGS((exarg_T *eap, int syncing));
429static void syn_cmd_region __ARGS((exarg_T *eap, int syncing));
430#ifdef __BORLANDC__
431static int _RTLENTRYF syn_compare_stub __ARGS((const void *v1, const void *v2));
432#else
433static int syn_compare_stub __ARGS((const void *v1, const void *v2));
434#endif
435static void syn_cmd_cluster __ARGS((exarg_T *eap, int syncing));
436static int syn_scl_name2id __ARGS((char_u *name));
437static int syn_scl_namen2id __ARGS((char_u *linep, int len));
438static int syn_check_cluster __ARGS((char_u *pp, int len));
439static int syn_add_cluster __ARGS((char_u *name));
440static void init_syn_patterns __ARGS((void));
441static char_u *get_syn_pattern __ARGS((char_u *arg, synpat_T *ci));
442static void syn_cmd_sync __ARGS((exarg_T *eap, int syncing));
443static int get_id_list __ARGS((char_u **arg, int keylen, short **list));
444static void syn_combine_list __ARGS((short **clstr1, short **clstr2, int list_op));
445static void syn_incl_toplevel __ARGS((int id, int *flagsp));
446
447/*
448 * Start the syntax recognition for a line. This function is normally called
449 * from the screen updating, once for each displayed line.
450 * The buffer is remembered in syn_buf, because get_syntax_attr() doesn't get
451 * it. Careful: curbuf and curwin are likely to point to another buffer and
452 * window.
453 */
454 void
455syntax_start(wp, lnum)
456 win_T *wp;
457 linenr_T lnum;
458{
459 synstate_T *p;
460 synstate_T *last_valid = NULL;
461 synstate_T *last_min_valid = NULL;
Bram Moolenaardbe31752008-01-13 16:40:19 +0000462 synstate_T *sp, *prev = NULL;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000463 linenr_T parsed_lnum;
464 linenr_T first_stored;
465 int dist;
Bram Moolenaar3fdfa4a2004-10-07 21:02:47 +0000466 static int changedtick = 0; /* remember the last change ID */
Bram Moolenaar071d4272004-06-13 20:20:40 +0000467
Bram Moolenaar071d4272004-06-13 20:20:40 +0000468 /*
469 * After switching buffers, invalidate current_state.
Bram Moolenaar3fdfa4a2004-10-07 21:02:47 +0000470 * Also do this when a change was made, the current state may be invalid
471 * then.
Bram Moolenaar071d4272004-06-13 20:20:40 +0000472 */
Bram Moolenaar860cae12010-06-05 23:22:07 +0200473 if (syn_block != wp->w_s || changedtick != syn_buf->b_changedtick)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000474 {
475 invalidate_current_state();
476 syn_buf = wp->w_buffer;
Bram Moolenaar860cae12010-06-05 23:22:07 +0200477 syn_block = wp->w_s;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000478 }
Bram Moolenaar3fdfa4a2004-10-07 21:02:47 +0000479 changedtick = syn_buf->b_changedtick;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000480 syn_win = wp;
481
482 /*
483 * Allocate syntax stack when needed.
484 */
485 syn_stack_alloc();
Bram Moolenaar860cae12010-06-05 23:22:07 +0200486 if (syn_block->b_sst_array == NULL)
Bram Moolenaar3b56eb32005-07-11 22:40:32 +0000487 return; /* out of memory */
Bram Moolenaar860cae12010-06-05 23:22:07 +0200488 syn_block->b_sst_lasttick = display_tick;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000489
490 /*
491 * If the state of the end of the previous line is useful, store it.
492 */
493 if (VALID_STATE(&current_state)
494 && current_lnum < lnum
495 && current_lnum < syn_buf->b_ml.ml_line_count)
496 {
497 (void)syn_finish_line(FALSE);
498 if (!current_state_stored)
499 {
500 ++current_lnum;
Bram Moolenaardbe31752008-01-13 16:40:19 +0000501 (void)store_current_state();
Bram Moolenaar071d4272004-06-13 20:20:40 +0000502 }
503
504 /*
505 * If the current_lnum is now the same as "lnum", keep the current
506 * state (this happens very often!). Otherwise invalidate
507 * current_state and figure it out below.
508 */
509 if (current_lnum != lnum)
510 invalidate_current_state();
511 }
512 else
513 invalidate_current_state();
514
515 /*
516 * Try to synchronize from a saved state in b_sst_array[].
517 * Only do this if lnum is not before and not to far beyond a saved state.
518 */
Bram Moolenaar860cae12010-06-05 23:22:07 +0200519 if (INVALID_STATE(&current_state) && syn_block->b_sst_array != NULL)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000520 {
521 /* Find last valid saved state before start_lnum. */
Bram Moolenaar860cae12010-06-05 23:22:07 +0200522 for (p = syn_block->b_sst_first; p != NULL; p = p->sst_next)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000523 {
524 if (p->sst_lnum > lnum)
525 break;
526 if (p->sst_lnum <= lnum && p->sst_change_lnum == 0)
527 {
528 last_valid = p;
Bram Moolenaar860cae12010-06-05 23:22:07 +0200529 if (p->sst_lnum >= lnum - syn_block->b_syn_sync_minlines)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000530 last_min_valid = p;
531 }
532 }
533 if (last_min_valid != NULL)
534 load_current_state(last_min_valid);
535 }
536
537 /*
538 * If "lnum" is before or far beyond a line with a saved state, need to
539 * re-synchronize.
540 */
541 if (INVALID_STATE(&current_state))
542 {
543 syn_sync(wp, lnum, last_valid);
Bram Moolenaar860cae12010-06-05 23:22:07 +0200544 first_stored = current_lnum + syn_block->b_syn_sync_minlines;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000545 }
546 else
547 first_stored = current_lnum;
548
549 /*
550 * Advance from the sync point or saved state until the current line.
551 * Save some entries for syncing with later on.
552 */
Bram Moolenaar860cae12010-06-05 23:22:07 +0200553 if (syn_block->b_sst_len <= Rows)
Bram Moolenaar910f66f2006-04-05 20:41:53 +0000554 dist = 999999;
555 else
Bram Moolenaar860cae12010-06-05 23:22:07 +0200556 dist = syn_buf->b_ml.ml_line_count / (syn_block->b_sst_len - Rows) + 1;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000557 while (current_lnum < lnum)
558 {
559 syn_start_line();
560 (void)syn_finish_line(FALSE);
561 ++current_lnum;
562
563 /* If we parsed at least "minlines" lines or started at a valid
564 * state, the current state is considered valid. */
565 if (current_lnum >= first_stored)
566 {
567 /* Check if the saved state entry is for the current line and is
568 * equal to the current state. If so, then validate all saved
569 * states that depended on a change before the parsed line. */
570 if (prev == NULL)
Bram Moolenaardbe31752008-01-13 16:40:19 +0000571 prev = syn_stack_find_entry(current_lnum - 1);
572 if (prev == NULL)
Bram Moolenaar860cae12010-06-05 23:22:07 +0200573 sp = syn_block->b_sst_first;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000574 else
Bram Moolenaardbe31752008-01-13 16:40:19 +0000575 sp = prev;
576 while (sp != NULL && sp->sst_lnum < current_lnum)
577 sp = sp->sst_next;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000578 if (sp != NULL
579 && sp->sst_lnum == current_lnum
580 && syn_stack_equal(sp))
581 {
582 parsed_lnum = current_lnum;
583 prev = sp;
584 while (sp != NULL && sp->sst_change_lnum <= parsed_lnum)
585 {
586 if (sp->sst_lnum <= lnum)
587 /* valid state before desired line, use this one */
588 prev = sp;
589 else if (sp->sst_change_lnum == 0)
590 /* past saved states depending on change, break here. */
591 break;
592 sp->sst_change_lnum = 0;
593 sp = sp->sst_next;
594 }
595 load_current_state(prev);
596 }
597 /* Store the state at this line when it's the first one, the line
598 * where we start parsing, or some distance from the previously
599 * saved state. But only when parsed at least 'minlines'. */
600 else if (prev == NULL
601 || current_lnum == lnum
602 || current_lnum >= prev->sst_lnum + dist)
Bram Moolenaardbe31752008-01-13 16:40:19 +0000603 prev = store_current_state();
Bram Moolenaar071d4272004-06-13 20:20:40 +0000604 }
605
606 /* This can take a long time: break when CTRL-C pressed. The current
607 * state will be wrong then. */
608 line_breakcheck();
609 if (got_int)
610 {
611 current_lnum = lnum;
612 break;
613 }
614 }
615
616 syn_start_line();
Bram Moolenaar071d4272004-06-13 20:20:40 +0000617}
618
619/*
620 * We cannot simply discard growarrays full of state_items or buf_states; we
621 * have to manually release their extmatch pointers first.
622 */
623 static void
624clear_syn_state(p)
625 synstate_T *p;
626{
627 int i;
628 garray_T *gap;
629
630 if (p->sst_stacksize > SST_FIX_STATES)
631 {
632 gap = &(p->sst_union.sst_ga);
633 for (i = 0; i < gap->ga_len; i++)
634 unref_extmatch(SYN_STATE_P(gap)[i].bs_extmatch);
635 ga_clear(gap);
636 }
637 else
638 {
639 for (i = 0; i < p->sst_stacksize; i++)
640 unref_extmatch(p->sst_union.sst_stack[i].bs_extmatch);
641 }
642}
643
644/*
645 * Cleanup the current_state stack.
646 */
647 static void
648clear_current_state()
649{
650 int i;
651 stateitem_T *sip;
652
653 sip = (stateitem_T *)(current_state.ga_data);
654 for (i = 0; i < current_state.ga_len; i++)
655 unref_extmatch(sip[i].si_extmatch);
656 ga_clear(&current_state);
657}
658
659/*
660 * Try to find a synchronisation point for line "lnum".
661 *
662 * This sets current_lnum and the current state. One of three methods is
663 * used:
664 * 1. Search backwards for the end of a C-comment.
665 * 2. Search backwards for given sync patterns.
666 * 3. Simply start on a given number of lines above "lnum".
667 */
668 static void
669syn_sync(wp, start_lnum, last_valid)
670 win_T *wp;
671 linenr_T start_lnum;
672 synstate_T *last_valid;
673{
674 buf_T *curbuf_save;
675 win_T *curwin_save;
676 pos_T cursor_save;
677 int idx;
678 linenr_T lnum;
679 linenr_T end_lnum;
680 linenr_T break_lnum;
681 int had_sync_point;
682 stateitem_T *cur_si;
683 synpat_T *spp;
684 char_u *line;
685 int found_flags = 0;
686 int found_match_idx = 0;
687 linenr_T found_current_lnum = 0;
688 int found_current_col= 0;
689 lpos_T found_m_endpos;
Bram Moolenaar81366db2005-07-24 21:16:51 +0000690 colnr_T prev_current_col;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000691
692 /*
693 * Clear any current state that might be hanging around.
694 */
695 invalidate_current_state();
696
697 /*
698 * Start at least "minlines" back. Default starting point for parsing is
699 * there.
700 * Start further back, to avoid that scrolling backwards will result in
701 * resyncing for every line. Now it resyncs only one out of N lines,
702 * where N is minlines * 1.5, or minlines * 2 if minlines is small.
703 * Watch out for overflow when minlines is MAXLNUM.
704 */
Bram Moolenaar860cae12010-06-05 23:22:07 +0200705 if (syn_block->b_syn_sync_minlines > start_lnum)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000706 start_lnum = 1;
707 else
708 {
Bram Moolenaar860cae12010-06-05 23:22:07 +0200709 if (syn_block->b_syn_sync_minlines == 1)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000710 lnum = 1;
Bram Moolenaar860cae12010-06-05 23:22:07 +0200711 else if (syn_block->b_syn_sync_minlines < 10)
712 lnum = syn_block->b_syn_sync_minlines * 2;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000713 else
Bram Moolenaar860cae12010-06-05 23:22:07 +0200714 lnum = syn_block->b_syn_sync_minlines * 3 / 2;
715 if (syn_block->b_syn_sync_maxlines != 0
716 && lnum > syn_block->b_syn_sync_maxlines)
717 lnum = syn_block->b_syn_sync_maxlines;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000718 if (lnum >= start_lnum)
719 start_lnum = 1;
720 else
721 start_lnum -= lnum;
722 }
723 current_lnum = start_lnum;
724
725 /*
726 * 1. Search backwards for the end of a C-style comment.
727 */
Bram Moolenaar860cae12010-06-05 23:22:07 +0200728 if (syn_block->b_syn_sync_flags & SF_CCOMMENT)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000729 {
730 /* Need to make syn_buf the current buffer for a moment, to be able to
731 * use find_start_comment(). */
732 curwin_save = curwin;
733 curwin = wp;
734 curbuf_save = curbuf;
735 curbuf = syn_buf;
736
737 /*
738 * Skip lines that end in a backslash.
739 */
740 for ( ; start_lnum > 1; --start_lnum)
741 {
742 line = ml_get(start_lnum - 1);
743 if (*line == NUL || *(line + STRLEN(line) - 1) != '\\')
744 break;
745 }
746 current_lnum = start_lnum;
747
748 /* set cursor to start of search */
749 cursor_save = wp->w_cursor;
750 wp->w_cursor.lnum = start_lnum;
751 wp->w_cursor.col = 0;
752
753 /*
754 * If the line is inside a comment, need to find the syntax item that
755 * defines the comment.
756 * Restrict the search for the end of a comment to b_syn_sync_maxlines.
757 */
Bram Moolenaar860cae12010-06-05 23:22:07 +0200758 if (find_start_comment((int)syn_block->b_syn_sync_maxlines) != NULL)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000759 {
Bram Moolenaar860cae12010-06-05 23:22:07 +0200760 for (idx = syn_block->b_syn_patterns.ga_len; --idx >= 0; )
761 if (SYN_ITEMS(syn_block)[idx].sp_syn.id
762 == syn_block->b_syn_sync_id
763 && SYN_ITEMS(syn_block)[idx].sp_type == SPTYPE_START)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000764 {
765 validate_current_state();
766 if (push_current_state(idx) == OK)
767 update_si_attr(current_state.ga_len - 1);
768 break;
769 }
770 }
771
772 /* restore cursor and buffer */
773 wp->w_cursor = cursor_save;
774 curwin = curwin_save;
775 curbuf = curbuf_save;
776 }
777
778 /*
779 * 2. Search backwards for given sync patterns.
780 */
Bram Moolenaar860cae12010-06-05 23:22:07 +0200781 else if (syn_block->b_syn_sync_flags & SF_MATCH)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000782 {
Bram Moolenaar860cae12010-06-05 23:22:07 +0200783 if (syn_block->b_syn_sync_maxlines != 0
784 && start_lnum > syn_block->b_syn_sync_maxlines)
785 break_lnum = start_lnum - syn_block->b_syn_sync_maxlines;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000786 else
787 break_lnum = 0;
788
Bram Moolenaarfd2ac762006-03-01 22:09:21 +0000789 found_m_endpos.lnum = 0;
790 found_m_endpos.col = 0;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000791 end_lnum = start_lnum;
792 lnum = start_lnum;
793 while (--lnum > break_lnum)
794 {
795 /* This can take a long time: break when CTRL-C pressed. */
796 line_breakcheck();
797 if (got_int)
798 {
799 invalidate_current_state();
800 current_lnum = start_lnum;
801 break;
802 }
803
804 /* Check if we have run into a valid saved state stack now. */
805 if (last_valid != NULL && lnum == last_valid->sst_lnum)
806 {
807 load_current_state(last_valid);
808 break;
809 }
810
811 /*
812 * Check if the previous line has the line-continuation pattern.
813 */
814 if (lnum > 1 && syn_match_linecont(lnum - 1))
815 continue;
816
817 /*
818 * Start with nothing on the state stack
819 */
820 validate_current_state();
821
822 for (current_lnum = lnum; current_lnum < end_lnum; ++current_lnum)
823 {
824 syn_start_line();
825 for (;;)
826 {
827 had_sync_point = syn_finish_line(TRUE);
828 /*
829 * When a sync point has been found, remember where, and
830 * continue to look for another one, further on in the line.
831 */
832 if (had_sync_point && current_state.ga_len)
833 {
834 cur_si = &CUR_STATE(current_state.ga_len - 1);
835 if (cur_si->si_m_endpos.lnum > start_lnum)
836 {
837 /* ignore match that goes to after where started */
838 current_lnum = end_lnum;
839 break;
840 }
Bram Moolenaar3a36cf72007-08-21 15:29:56 +0000841 if (cur_si->si_idx < 0)
842 {
843 /* Cannot happen? */
844 found_flags = 0;
845 found_match_idx = KEYWORD_IDX;
846 }
847 else
848 {
Bram Moolenaar860cae12010-06-05 23:22:07 +0200849 spp = &(SYN_ITEMS(syn_block)[cur_si->si_idx]);
Bram Moolenaar3a36cf72007-08-21 15:29:56 +0000850 found_flags = spp->sp_flags;
851 found_match_idx = spp->sp_sync_idx;
852 }
Bram Moolenaar071d4272004-06-13 20:20:40 +0000853 found_current_lnum = current_lnum;
854 found_current_col = current_col;
855 found_m_endpos = cur_si->si_m_endpos;
856 /*
857 * Continue after the match (be aware of a zero-length
858 * match).
859 */
860 if (found_m_endpos.lnum > current_lnum)
861 {
862 current_lnum = found_m_endpos.lnum;
863 current_col = found_m_endpos.col;
864 if (current_lnum >= end_lnum)
865 break;
866 }
867 else if (found_m_endpos.col > current_col)
868 current_col = found_m_endpos.col;
869 else
870 ++current_col;
871
872 /* syn_current_attr() will have skipped the check for
Bram Moolenaar81366db2005-07-24 21:16:51 +0000873 * an item that ends here, need to do that now. Be
874 * careful not to go past the NUL. */
875 prev_current_col = current_col;
876 if (syn_getcurline()[current_col] != NUL)
877 ++current_col;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000878 check_state_ends();
Bram Moolenaar81366db2005-07-24 21:16:51 +0000879 current_col = prev_current_col;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000880 }
881 else
882 break;
883 }
884 }
885
886 /*
887 * If a sync point was encountered, break here.
888 */
889 if (found_flags)
890 {
891 /*
892 * Put the item that was specified by the sync point on the
893 * state stack. If there was no item specified, make the
894 * state stack empty.
895 */
896 clear_current_state();
897 if (found_match_idx >= 0
898 && push_current_state(found_match_idx) == OK)
899 update_si_attr(current_state.ga_len - 1);
900
901 /*
902 * When using "grouphere", continue from the sync point
903 * match, until the end of the line. Parsing starts at
904 * the next line.
905 * For "groupthere" the parsing starts at start_lnum.
906 */
907 if (found_flags & HL_SYNC_HERE)
908 {
909 if (current_state.ga_len)
910 {
911 cur_si = &CUR_STATE(current_state.ga_len - 1);
912 cur_si->si_h_startpos.lnum = found_current_lnum;
913 cur_si->si_h_startpos.col = found_current_col;
914 update_si_end(cur_si, (int)current_col, TRUE);
915 check_keepend();
916 }
917 current_col = found_m_endpos.col;
918 current_lnum = found_m_endpos.lnum;
919 (void)syn_finish_line(FALSE);
920 ++current_lnum;
921 }
922 else
923 current_lnum = start_lnum;
924
925 break;
926 }
927
928 end_lnum = lnum;
929 invalidate_current_state();
930 }
931
932 /* Ran into start of the file or exceeded maximum number of lines */
933 if (lnum <= break_lnum)
934 {
935 invalidate_current_state();
936 current_lnum = break_lnum + 1;
937 }
938 }
939
940 validate_current_state();
941}
942
943/*
944 * Return TRUE if the line-continuation pattern matches in line "lnum".
945 */
946 static int
947syn_match_linecont(lnum)
948 linenr_T lnum;
949{
950 regmmatch_T regmatch;
951
Bram Moolenaar860cae12010-06-05 23:22:07 +0200952 if (syn_block->b_syn_linecont_prog != NULL)
Bram Moolenaar071d4272004-06-13 20:20:40 +0000953 {
Bram Moolenaar860cae12010-06-05 23:22:07 +0200954 regmatch.rmm_ic = syn_block->b_syn_linecont_ic;
955 regmatch.regprog = syn_block->b_syn_linecont_prog;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000956 return syn_regexec(&regmatch, lnum, (colnr_T)0);
957 }
958 return FALSE;
959}
960
961/*
962 * Prepare the current state for the start of a line.
963 */
964 static void
965syn_start_line()
966{
967 current_finished = FALSE;
968 current_col = 0;
969
970 /*
971 * Need to update the end of a start/skip/end that continues from the
972 * previous line and regions that have "keepend".
973 */
974 if (current_state.ga_len > 0)
975 syn_update_ends(TRUE);
976
977 next_match_idx = -1;
978 ++current_line_id;
979}
980
981/*
982 * Check for items in the stack that need their end updated.
983 * When "startofline" is TRUE the last item is always updated.
984 * When "startofline" is FALSE the item with "keepend" is forcefully updated.
985 */
986 static void
987syn_update_ends(startofline)
988 int startofline;
989{
990 stateitem_T *cur_si;
991 int i;
Bram Moolenaar7bd2cd82006-10-03 15:04:36 +0000992 int seen_keepend;
Bram Moolenaar071d4272004-06-13 20:20:40 +0000993
994 if (startofline)
995 {
996 /* Check for a match carried over from a previous line with a
997 * contained region. The match ends as soon as the region ends. */
998 for (i = 0; i < current_state.ga_len; ++i)
999 {
1000 cur_si = &CUR_STATE(i);
1001 if (cur_si->si_idx >= 0
Bram Moolenaar860cae12010-06-05 23:22:07 +02001002 && (SYN_ITEMS(syn_block)[cur_si->si_idx]).sp_type
Bram Moolenaar071d4272004-06-13 20:20:40 +00001003 == SPTYPE_MATCH
1004 && cur_si->si_m_endpos.lnum < current_lnum)
1005 {
1006 cur_si->si_flags |= HL_MATCHCONT;
1007 cur_si->si_m_endpos.lnum = 0;
1008 cur_si->si_m_endpos.col = 0;
1009 cur_si->si_h_endpos = cur_si->si_m_endpos;
1010 cur_si->si_ends = TRUE;
1011 }
1012 }
1013 }
1014
1015 /*
1016 * Need to update the end of a start/skip/end that continues from the
1017 * previous line. And regions that have "keepend", because they may
Bram Moolenaar7bd2cd82006-10-03 15:04:36 +00001018 * influence contained items. If we've just removed "extend"
1019 * (startofline == 0) then we should update ends of normal regions
1020 * contained inside "keepend" because "extend" could have extended
1021 * these "keepend" regions as well as contained normal regions.
Bram Moolenaar071d4272004-06-13 20:20:40 +00001022 * Then check for items ending in column 0.
1023 */
1024 i = current_state.ga_len - 1;
1025 if (keepend_level >= 0)
1026 for ( ; i > keepend_level; --i)
1027 if (CUR_STATE(i).si_flags & HL_EXTEND)
1028 break;
Bram Moolenaar7bd2cd82006-10-03 15:04:36 +00001029
1030 seen_keepend = FALSE;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001031 for ( ; i < current_state.ga_len; ++i)
1032 {
1033 cur_si = &CUR_STATE(i);
1034 if ((cur_si->si_flags & HL_KEEPEND)
Bram Moolenaar7bd2cd82006-10-03 15:04:36 +00001035 || (seen_keepend && !startofline)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001036 || (i == current_state.ga_len - 1 && startofline))
1037 {
1038 cur_si->si_h_startpos.col = 0; /* start highl. in col 0 */
1039 cur_si->si_h_startpos.lnum = current_lnum;
1040
1041 if (!(cur_si->si_flags & HL_MATCHCONT))
1042 update_si_end(cur_si, (int)current_col, !startofline);
Bram Moolenaar7bd2cd82006-10-03 15:04:36 +00001043
1044 if (!startofline && (cur_si->si_flags & HL_KEEPEND))
1045 seen_keepend = TRUE;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001046 }
1047 }
1048 check_keepend();
1049 check_state_ends();
1050}
1051
1052/****************************************
1053 * Handling of the state stack cache.
1054 */
1055
1056/*
1057 * EXPLANATION OF THE SYNTAX STATE STACK CACHE
1058 *
1059 * To speed up syntax highlighting, the state stack for the start of some
1060 * lines is cached. These entries can be used to start parsing at that point.
1061 *
1062 * The stack is kept in b_sst_array[] for each buffer. There is a list of
1063 * valid entries. b_sst_first points to the first one, then follow sst_next.
1064 * The entries are sorted on line number. The first entry is often for line 2
1065 * (line 1 always starts with an empty stack).
1066 * There is also a list for free entries. This construction is used to avoid
1067 * having to allocate and free memory blocks too often.
1068 *
1069 * When making changes to the buffer, this is logged in b_mod_*. When calling
1070 * update_screen() to update the display, it will call
1071 * syn_stack_apply_changes() for each displayed buffer to adjust the cached
1072 * entries. The entries which are inside the changed area are removed,
1073 * because they must be recomputed. Entries below the changed have their line
1074 * number adjusted for deleted/inserted lines, and have their sst_change_lnum
1075 * set to indicate that a check must be made if the changed lines would change
1076 * the cached entry.
1077 *
1078 * When later displaying lines, an entry is stored for each line. Displayed
1079 * lines are likely to be displayed again, in which case the state at the
1080 * start of the line is needed.
1081 * For not displayed lines, an entry is stored for every so many lines. These
1082 * entries will be used e.g., when scrolling backwards. The distance between
1083 * entries depends on the number of lines in the buffer. For small buffers
1084 * the distance is fixed at SST_DIST, for large buffers there is a fixed
1085 * number of entries SST_MAX_ENTRIES, and the distance is computed.
1086 */
1087
Bram Moolenaar860cae12010-06-05 23:22:07 +02001088 static void
1089syn_stack_free_block(block)
1090 synblock_T *block;
1091{
1092 synstate_T *p;
1093
1094 if (block->b_sst_array != NULL)
1095 {
1096 for (p = block->b_sst_first; p != NULL; p = p->sst_next)
1097 clear_syn_state(p);
1098 vim_free(block->b_sst_array);
1099 block->b_sst_array = NULL;
1100 block->b_sst_len = 0;
1101 }
1102}
Bram Moolenaar071d4272004-06-13 20:20:40 +00001103/*
1104 * Free b_sst_array[] for buffer "buf".
1105 * Used when syntax items changed to force resyncing everywhere.
1106 */
1107 void
Bram Moolenaar860cae12010-06-05 23:22:07 +02001108syn_stack_free_all(block)
1109 synblock_T *block;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001110{
Bram Moolenaar071d4272004-06-13 20:20:40 +00001111 win_T *wp;
1112
Bram Moolenaar860cae12010-06-05 23:22:07 +02001113 syn_stack_free_block(block);
1114
1115
Bram Moolenaar071d4272004-06-13 20:20:40 +00001116#ifdef FEAT_FOLDING
1117 /* When using "syntax" fold method, must update all folds. */
1118 FOR_ALL_WINDOWS(wp)
1119 {
Bram Moolenaar860cae12010-06-05 23:22:07 +02001120 if (wp->w_s == block && foldmethodIsSyntax(wp))
Bram Moolenaar071d4272004-06-13 20:20:40 +00001121 foldUpdateAll(wp);
1122 }
1123#endif
1124}
1125
1126/*
1127 * Allocate the syntax state stack for syn_buf when needed.
1128 * If the number of entries in b_sst_array[] is much too big or a bit too
1129 * small, reallocate it.
1130 * Also used to allocate b_sst_array[] for the first time.
1131 */
1132 static void
1133syn_stack_alloc()
1134{
1135 long len;
1136 synstate_T *to, *from;
1137 synstate_T *sstp;
1138
1139 len = syn_buf->b_ml.ml_line_count / SST_DIST + Rows * 2;
1140 if (len < SST_MIN_ENTRIES)
1141 len = SST_MIN_ENTRIES;
1142 else if (len > SST_MAX_ENTRIES)
1143 len = SST_MAX_ENTRIES;
Bram Moolenaar860cae12010-06-05 23:22:07 +02001144 if (syn_block->b_sst_len > len * 2 || syn_block->b_sst_len < len)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001145 {
1146 /* Allocate 50% too much, to avoid reallocating too often. */
1147 len = syn_buf->b_ml.ml_line_count;
1148 len = (len + len / 2) / SST_DIST + Rows * 2;
1149 if (len < SST_MIN_ENTRIES)
1150 len = SST_MIN_ENTRIES;
1151 else if (len > SST_MAX_ENTRIES)
1152 len = SST_MAX_ENTRIES;
1153
Bram Moolenaar860cae12010-06-05 23:22:07 +02001154 if (syn_block->b_sst_array != NULL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001155 {
1156 /* When shrinking the array, cleanup the existing stack.
1157 * Make sure that all valid entries fit in the new array. */
Bram Moolenaar860cae12010-06-05 23:22:07 +02001158 while (syn_block->b_sst_len - syn_block->b_sst_freecount + 2 > len
Bram Moolenaar071d4272004-06-13 20:20:40 +00001159 && syn_stack_cleanup())
1160 ;
Bram Moolenaar860cae12010-06-05 23:22:07 +02001161 if (len < syn_block->b_sst_len - syn_block->b_sst_freecount + 2)
1162 len = syn_block->b_sst_len - syn_block->b_sst_freecount + 2;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001163 }
1164
1165 sstp = (synstate_T *)alloc_clear((unsigned)(len * sizeof(synstate_T)));
1166 if (sstp == NULL) /* out of memory! */
1167 return;
1168
1169 to = sstp - 1;
Bram Moolenaar860cae12010-06-05 23:22:07 +02001170 if (syn_block->b_sst_array != NULL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001171 {
1172 /* Move the states from the old array to the new one. */
Bram Moolenaar860cae12010-06-05 23:22:07 +02001173 for (from = syn_block->b_sst_first; from != NULL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001174 from = from->sst_next)
1175 {
1176 ++to;
1177 *to = *from;
1178 to->sst_next = to + 1;
1179 }
1180 }
1181 if (to != sstp - 1)
1182 {
1183 to->sst_next = NULL;
Bram Moolenaar860cae12010-06-05 23:22:07 +02001184 syn_block->b_sst_first = sstp;
1185 syn_block->b_sst_freecount = len - (int)(to - sstp) - 1;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001186 }
1187 else
1188 {
Bram Moolenaar860cae12010-06-05 23:22:07 +02001189 syn_block->b_sst_first = NULL;
1190 syn_block->b_sst_freecount = len;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001191 }
1192
1193 /* Create the list of free entries. */
Bram Moolenaar860cae12010-06-05 23:22:07 +02001194 syn_block->b_sst_firstfree = to + 1;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001195 while (++to < sstp + len)
1196 to->sst_next = to + 1;
1197 (sstp + len - 1)->sst_next = NULL;
1198
Bram Moolenaar860cae12010-06-05 23:22:07 +02001199 vim_free(syn_block->b_sst_array);
1200 syn_block->b_sst_array = sstp;
1201 syn_block->b_sst_len = len;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001202 }
1203}
1204
1205/*
1206 * Check for changes in a buffer to affect stored syntax states. Uses the
1207 * b_mod_* fields.
1208 * Called from update_screen(), before screen is being updated, once for each
1209 * displayed buffer.
1210 */
1211 void
1212syn_stack_apply_changes(buf)
1213 buf_T *buf;
1214{
Bram Moolenaar860cae12010-06-05 23:22:07 +02001215 win_T *wp;
1216
1217 syn_stack_apply_changes_block(&buf->b_s, buf);
1218
1219 FOR_ALL_WINDOWS(wp)
1220 {
1221 if ((wp->w_buffer == buf) && (wp->w_s != &buf->b_s))
1222 syn_stack_apply_changes_block(wp->w_s, buf);
1223 }
1224}
1225
1226 static void
1227syn_stack_apply_changes_block(block, buf)
1228 synblock_T *block;
1229 buf_T *buf;
1230{
Bram Moolenaar071d4272004-06-13 20:20:40 +00001231 synstate_T *p, *prev, *np;
1232 linenr_T n;
1233
Bram Moolenaar860cae12010-06-05 23:22:07 +02001234 if (block->b_sst_array == NULL) /* nothing to do */
Bram Moolenaar071d4272004-06-13 20:20:40 +00001235 return;
1236
1237 prev = NULL;
Bram Moolenaar860cae12010-06-05 23:22:07 +02001238 for (p = block->b_sst_first; p != NULL; )
Bram Moolenaar071d4272004-06-13 20:20:40 +00001239 {
Bram Moolenaar860cae12010-06-05 23:22:07 +02001240 if (p->sst_lnum + block->b_syn_sync_linebreaks > buf->b_mod_top)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001241 {
1242 n = p->sst_lnum + buf->b_mod_xlines;
1243 if (n <= buf->b_mod_bot)
1244 {
1245 /* this state is inside the changed area, remove it */
1246 np = p->sst_next;
1247 if (prev == NULL)
Bram Moolenaar860cae12010-06-05 23:22:07 +02001248 block->b_sst_first = np;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001249 else
1250 prev->sst_next = np;
Bram Moolenaar860cae12010-06-05 23:22:07 +02001251 syn_stack_free_entry(block, p);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001252 p = np;
1253 continue;
1254 }
1255 /* This state is below the changed area. Remember the line
1256 * that needs to be parsed before this entry can be made valid
1257 * again. */
1258 if (p->sst_change_lnum != 0 && p->sst_change_lnum > buf->b_mod_top)
1259 {
1260 if (p->sst_change_lnum + buf->b_mod_xlines > buf->b_mod_top)
1261 p->sst_change_lnum += buf->b_mod_xlines;
1262 else
1263 p->sst_change_lnum = buf->b_mod_top;
1264 }
1265 if (p->sst_change_lnum == 0
1266 || p->sst_change_lnum < buf->b_mod_bot)
1267 p->sst_change_lnum = buf->b_mod_bot;
1268
1269 p->sst_lnum = n;
1270 }
1271 prev = p;
1272 p = p->sst_next;
1273 }
1274}
1275
1276/*
1277 * Reduce the number of entries in the state stack for syn_buf.
1278 * Returns TRUE if at least one entry was freed.
1279 */
1280 static int
1281syn_stack_cleanup()
1282{
1283 synstate_T *p, *prev;
1284 disptick_T tick;
1285 int above;
1286 int dist;
1287 int retval = FALSE;
1288
Bram Moolenaar860cae12010-06-05 23:22:07 +02001289 if (syn_block->b_sst_array == NULL || syn_block->b_sst_first == NULL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001290 return retval;
1291
1292 /* Compute normal distance between non-displayed entries. */
Bram Moolenaar860cae12010-06-05 23:22:07 +02001293 if (syn_block->b_sst_len <= Rows)
Bram Moolenaar910f66f2006-04-05 20:41:53 +00001294 dist = 999999;
1295 else
Bram Moolenaar860cae12010-06-05 23:22:07 +02001296 dist = syn_buf->b_ml.ml_line_count / (syn_block->b_sst_len - Rows) + 1;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001297
1298 /*
Bram Moolenaarf5b63862009-12-16 17:13:44 +00001299 * Go through the list to find the "tick" for the oldest entry that can
Bram Moolenaar071d4272004-06-13 20:20:40 +00001300 * be removed. Set "above" when the "tick" for the oldest entry is above
1301 * "b_sst_lasttick" (the display tick wraps around).
1302 */
Bram Moolenaar860cae12010-06-05 23:22:07 +02001303 tick = syn_block->b_sst_lasttick;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001304 above = FALSE;
Bram Moolenaar860cae12010-06-05 23:22:07 +02001305 prev = syn_block->b_sst_first;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001306 for (p = prev->sst_next; p != NULL; prev = p, p = p->sst_next)
1307 {
1308 if (prev->sst_lnum + dist > p->sst_lnum)
1309 {
Bram Moolenaar860cae12010-06-05 23:22:07 +02001310 if (p->sst_tick > syn_block->b_sst_lasttick)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001311 {
1312 if (!above || p->sst_tick < tick)
1313 tick = p->sst_tick;
1314 above = TRUE;
1315 }
1316 else if (!above && p->sst_tick < tick)
1317 tick = p->sst_tick;
1318 }
1319 }
1320
1321 /*
1322 * Go through the list to make the entries for the oldest tick at an
1323 * interval of several lines.
1324 */
Bram Moolenaar860cae12010-06-05 23:22:07 +02001325 prev = syn_block->b_sst_first;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001326 for (p = prev->sst_next; p != NULL; prev = p, p = p->sst_next)
1327 {
1328 if (p->sst_tick == tick && prev->sst_lnum + dist > p->sst_lnum)
1329 {
1330 /* Move this entry from used list to free list */
1331 prev->sst_next = p->sst_next;
Bram Moolenaar860cae12010-06-05 23:22:07 +02001332 syn_stack_free_entry(syn_block, p);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001333 p = prev;
1334 retval = TRUE;
1335 }
1336 }
1337 return retval;
1338}
1339
1340/*
1341 * Free the allocated memory for a syn_state item.
1342 * Move the entry into the free list.
1343 */
1344 static void
Bram Moolenaar860cae12010-06-05 23:22:07 +02001345syn_stack_free_entry(block, p)
1346 synblock_T *block;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001347 synstate_T *p;
1348{
1349 clear_syn_state(p);
Bram Moolenaar860cae12010-06-05 23:22:07 +02001350 p->sst_next = block->b_sst_firstfree;
1351 block->b_sst_firstfree = p;
1352 ++block->b_sst_freecount;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001353}
1354
1355/*
1356 * Find an entry in the list of state stacks at or before "lnum".
1357 * Returns NULL when there is no entry or the first entry is after "lnum".
1358 */
1359 static synstate_T *
1360syn_stack_find_entry(lnum)
1361 linenr_T lnum;
1362{
1363 synstate_T *p, *prev;
1364
1365 prev = NULL;
Bram Moolenaar860cae12010-06-05 23:22:07 +02001366 for (p = syn_block->b_sst_first; p != NULL; prev = p, p = p->sst_next)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001367 {
1368 if (p->sst_lnum == lnum)
1369 return p;
1370 if (p->sst_lnum > lnum)
1371 break;
1372 }
1373 return prev;
1374}
1375
1376/*
1377 * Try saving the current state in b_sst_array[].
1378 * The current state must be valid for the start of the current_lnum line!
1379 */
1380 static synstate_T *
Bram Moolenaardbe31752008-01-13 16:40:19 +00001381store_current_state()
Bram Moolenaar071d4272004-06-13 20:20:40 +00001382{
1383 int i;
1384 synstate_T *p;
1385 bufstate_T *bp;
1386 stateitem_T *cur_si;
Bram Moolenaardbe31752008-01-13 16:40:19 +00001387 synstate_T *sp = syn_stack_find_entry(current_lnum);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001388
1389 /*
1390 * If the current state contains a start or end pattern that continues
1391 * from the previous line, we can't use it. Don't store it then.
1392 */
1393 for (i = current_state.ga_len - 1; i >= 0; --i)
1394 {
1395 cur_si = &CUR_STATE(i);
1396 if (cur_si->si_h_startpos.lnum >= current_lnum
1397 || cur_si->si_m_endpos.lnum >= current_lnum
1398 || cur_si->si_h_endpos.lnum >= current_lnum
1399 || (cur_si->si_end_idx
1400 && cur_si->si_eoe_pos.lnum >= current_lnum))
1401 break;
1402 }
1403 if (i >= 0)
1404 {
1405 if (sp != NULL)
1406 {
1407 /* find "sp" in the list and remove it */
Bram Moolenaar860cae12010-06-05 23:22:07 +02001408 if (syn_block->b_sst_first == sp)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001409 /* it's the first entry */
Bram Moolenaar860cae12010-06-05 23:22:07 +02001410 syn_block->b_sst_first = sp->sst_next;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001411 else
1412 {
1413 /* find the entry just before this one to adjust sst_next */
Bram Moolenaar860cae12010-06-05 23:22:07 +02001414 for (p = syn_block->b_sst_first; p != NULL; p = p->sst_next)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001415 if (p->sst_next == sp)
1416 break;
Bram Moolenaareb3593b2006-04-22 22:33:57 +00001417 if (p != NULL) /* just in case */
1418 p->sst_next = sp->sst_next;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001419 }
Bram Moolenaar860cae12010-06-05 23:22:07 +02001420 syn_stack_free_entry(syn_block, sp);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001421 sp = NULL;
1422 }
1423 }
1424 else if (sp == NULL || sp->sst_lnum != current_lnum)
1425 {
1426 /*
1427 * Add a new entry
1428 */
1429 /* If no free items, cleanup the array first. */
Bram Moolenaar860cae12010-06-05 23:22:07 +02001430 if (syn_block->b_sst_freecount == 0)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001431 {
1432 (void)syn_stack_cleanup();
1433 /* "sp" may have been moved to the freelist now */
1434 sp = syn_stack_find_entry(current_lnum);
1435 }
1436 /* Still no free items? Must be a strange problem... */
Bram Moolenaar860cae12010-06-05 23:22:07 +02001437 if (syn_block->b_sst_freecount == 0)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001438 sp = NULL;
1439 else
1440 {
1441 /* Take the first item from the free list and put it in the used
1442 * list, after *sp */
Bram Moolenaar860cae12010-06-05 23:22:07 +02001443 p = syn_block->b_sst_firstfree;
1444 syn_block->b_sst_firstfree = p->sst_next;
1445 --syn_block->b_sst_freecount;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001446 if (sp == NULL)
1447 {
1448 /* Insert in front of the list */
Bram Moolenaar860cae12010-06-05 23:22:07 +02001449 p->sst_next = syn_block->b_sst_first;
1450 syn_block->b_sst_first = p;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001451 }
1452 else
1453 {
1454 /* insert in list after *sp */
1455 p->sst_next = sp->sst_next;
1456 sp->sst_next = p;
1457 }
1458 sp = p;
1459 sp->sst_stacksize = 0;
1460 sp->sst_lnum = current_lnum;
1461 }
1462 }
1463 if (sp != NULL)
1464 {
1465 /* When overwriting an existing state stack, clear it first */
1466 clear_syn_state(sp);
1467 sp->sst_stacksize = current_state.ga_len;
1468 if (current_state.ga_len > SST_FIX_STATES)
1469 {
1470 /* Need to clear it, might be something remaining from when the
1471 * length was less than SST_FIX_STATES. */
1472 ga_init2(&sp->sst_union.sst_ga, (int)sizeof(bufstate_T), 1);
1473 if (ga_grow(&sp->sst_union.sst_ga, current_state.ga_len) == FAIL)
1474 sp->sst_stacksize = 0;
1475 else
Bram Moolenaar071d4272004-06-13 20:20:40 +00001476 sp->sst_union.sst_ga.ga_len = current_state.ga_len;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001477 bp = SYN_STATE_P(&(sp->sst_union.sst_ga));
1478 }
1479 else
1480 bp = sp->sst_union.sst_stack;
1481 for (i = 0; i < sp->sst_stacksize; ++i)
1482 {
1483 bp[i].bs_idx = CUR_STATE(i).si_idx;
1484 bp[i].bs_flags = CUR_STATE(i).si_flags;
1485 bp[i].bs_extmatch = ref_extmatch(CUR_STATE(i).si_extmatch);
1486 }
1487 sp->sst_next_flags = current_next_flags;
1488 sp->sst_next_list = current_next_list;
1489 sp->sst_tick = display_tick;
1490 sp->sst_change_lnum = 0;
1491 }
1492 current_state_stored = TRUE;
1493 return sp;
1494}
1495
1496/*
1497 * Copy a state stack from "from" in b_sst_array[] to current_state;
1498 */
1499 static void
1500load_current_state(from)
1501 synstate_T *from;
1502{
1503 int i;
1504 bufstate_T *bp;
1505
1506 clear_current_state();
1507 validate_current_state();
1508 keepend_level = -1;
1509 if (from->sst_stacksize
1510 && ga_grow(&current_state, from->sst_stacksize) != FAIL)
1511 {
1512 if (from->sst_stacksize > SST_FIX_STATES)
1513 bp = SYN_STATE_P(&(from->sst_union.sst_ga));
1514 else
1515 bp = from->sst_union.sst_stack;
1516 for (i = 0; i < from->sst_stacksize; ++i)
1517 {
1518 CUR_STATE(i).si_idx = bp[i].bs_idx;
1519 CUR_STATE(i).si_flags = bp[i].bs_flags;
1520 CUR_STATE(i).si_extmatch = ref_extmatch(bp[i].bs_extmatch);
1521 if (keepend_level < 0 && (CUR_STATE(i).si_flags & HL_KEEPEND))
1522 keepend_level = i;
1523 CUR_STATE(i).si_ends = FALSE;
1524 CUR_STATE(i).si_m_lnum = 0;
1525 if (CUR_STATE(i).si_idx >= 0)
1526 CUR_STATE(i).si_next_list =
Bram Moolenaar860cae12010-06-05 23:22:07 +02001527 (SYN_ITEMS(syn_block)[CUR_STATE(i).si_idx]).sp_next_list;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001528 else
1529 CUR_STATE(i).si_next_list = NULL;
1530 update_si_attr(i);
1531 }
1532 current_state.ga_len = from->sst_stacksize;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001533 }
1534 current_next_list = from->sst_next_list;
1535 current_next_flags = from->sst_next_flags;
1536 current_lnum = from->sst_lnum;
1537}
1538
1539/*
1540 * Compare saved state stack "*sp" with the current state.
1541 * Return TRUE when they are equal.
1542 */
1543 static int
1544syn_stack_equal(sp)
1545 synstate_T *sp;
1546{
1547 int i, j;
1548 bufstate_T *bp;
1549 reg_extmatch_T *six, *bsx;
1550
1551 /* First a quick check if the stacks have the same size end nextlist. */
1552 if (sp->sst_stacksize == current_state.ga_len
1553 && sp->sst_next_list == current_next_list)
1554 {
1555 /* Need to compare all states on both stacks. */
1556 if (sp->sst_stacksize > SST_FIX_STATES)
1557 bp = SYN_STATE_P(&(sp->sst_union.sst_ga));
1558 else
1559 bp = sp->sst_union.sst_stack;
1560
1561 for (i = current_state.ga_len; --i >= 0; )
1562 {
1563 /* If the item has another index the state is different. */
1564 if (bp[i].bs_idx != CUR_STATE(i).si_idx)
1565 break;
1566 if (bp[i].bs_extmatch != CUR_STATE(i).si_extmatch)
1567 {
1568 /* When the extmatch pointers are different, the strings in
1569 * them can still be the same. Check if the extmatch
1570 * references are equal. */
1571 bsx = bp[i].bs_extmatch;
1572 six = CUR_STATE(i).si_extmatch;
1573 /* If one of the extmatch pointers is NULL the states are
1574 * different. */
1575 if (bsx == NULL || six == NULL)
1576 break;
1577 for (j = 0; j < NSUBEXP; ++j)
1578 {
1579 /* Check each referenced match string. They must all be
1580 * equal. */
1581 if (bsx->matches[j] != six->matches[j])
1582 {
1583 /* If the pointer is different it can still be the
1584 * same text. Compare the strings, ignore case when
1585 * the start item has the sp_ic flag set. */
1586 if (bsx->matches[j] == NULL
1587 || six->matches[j] == NULL)
1588 break;
Bram Moolenaar860cae12010-06-05 23:22:07 +02001589 if ((SYN_ITEMS(syn_block)[CUR_STATE(i).si_idx]).sp_ic
Bram Moolenaar071d4272004-06-13 20:20:40 +00001590 ? MB_STRICMP(bsx->matches[j],
1591 six->matches[j]) != 0
1592 : STRCMP(bsx->matches[j], six->matches[j]) != 0)
1593 break;
1594 }
1595 }
1596 if (j != NSUBEXP)
1597 break;
1598 }
1599 }
1600 if (i < 0)
1601 return TRUE;
1602 }
1603 return FALSE;
1604}
1605
1606/*
1607 * We stop parsing syntax above line "lnum". If the stored state at or below
1608 * this line depended on a change before it, it now depends on the line below
1609 * the last parsed line.
1610 * The window looks like this:
1611 * line which changed
1612 * displayed line
1613 * displayed line
1614 * lnum -> line below window
1615 */
1616 void
1617syntax_end_parsing(lnum)
1618 linenr_T lnum;
1619{
1620 synstate_T *sp;
1621
1622 sp = syn_stack_find_entry(lnum);
1623 if (sp != NULL && sp->sst_lnum < lnum)
1624 sp = sp->sst_next;
1625
1626 if (sp != NULL && sp->sst_change_lnum != 0)
1627 sp->sst_change_lnum = lnum;
1628}
1629
1630/*
1631 * End of handling of the state stack.
1632 ****************************************/
1633
1634 static void
1635invalidate_current_state()
1636{
1637 clear_current_state();
1638 current_state.ga_itemsize = 0; /* mark current_state invalid */
1639 current_next_list = NULL;
1640 keepend_level = -1;
1641}
1642
1643 static void
1644validate_current_state()
1645{
1646 current_state.ga_itemsize = sizeof(stateitem_T);
1647 current_state.ga_growsize = 3;
1648}
1649
1650/*
1651 * Return TRUE if the syntax at start of lnum changed since last time.
1652 * This will only be called just after get_syntax_attr() for the previous
1653 * line, to check if the next line needs to be redrawn too.
1654 */
1655 int
1656syntax_check_changed(lnum)
1657 linenr_T lnum;
1658{
1659 int retval = TRUE;
1660 synstate_T *sp;
1661
Bram Moolenaar071d4272004-06-13 20:20:40 +00001662 /*
1663 * Check the state stack when:
1664 * - lnum is just below the previously syntaxed line.
1665 * - lnum is not before the lines with saved states.
1666 * - lnum is not past the lines with saved states.
1667 * - lnum is at or before the last changed line.
1668 */
1669 if (VALID_STATE(&current_state) && lnum == current_lnum + 1)
1670 {
1671 sp = syn_stack_find_entry(lnum);
1672 if (sp != NULL && sp->sst_lnum == lnum)
1673 {
1674 /*
1675 * finish the previous line (needed when not all of the line was
1676 * drawn)
1677 */
1678 (void)syn_finish_line(FALSE);
1679
1680 /*
1681 * Compare the current state with the previously saved state of
1682 * the line.
1683 */
1684 if (syn_stack_equal(sp))
1685 retval = FALSE;
1686
1687 /*
1688 * Store the current state in b_sst_array[] for later use.
1689 */
1690 ++current_lnum;
Bram Moolenaardbe31752008-01-13 16:40:19 +00001691 (void)store_current_state();
Bram Moolenaar071d4272004-06-13 20:20:40 +00001692 }
1693 }
1694
Bram Moolenaar071d4272004-06-13 20:20:40 +00001695 return retval;
1696}
1697
1698/*
1699 * Finish the current line.
1700 * This doesn't return any attributes, it only gets the state at the end of
1701 * the line. It can start anywhere in the line, as long as the current state
1702 * is valid.
1703 */
1704 static int
1705syn_finish_line(syncing)
1706 int syncing; /* called for syncing */
1707{
1708 stateitem_T *cur_si;
Bram Moolenaar81366db2005-07-24 21:16:51 +00001709 colnr_T prev_current_col;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001710
1711 if (!current_finished)
1712 {
1713 while (!current_finished)
1714 {
Bram Moolenaar56cefaf2008-01-12 15:47:10 +00001715 (void)syn_current_attr(syncing, FALSE, NULL, FALSE);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001716 /*
1717 * When syncing, and found some item, need to check the item.
1718 */
1719 if (syncing && current_state.ga_len)
1720 {
1721 /*
1722 * Check for match with sync item.
1723 */
1724 cur_si = &CUR_STATE(current_state.ga_len - 1);
1725 if (cur_si->si_idx >= 0
Bram Moolenaar860cae12010-06-05 23:22:07 +02001726 && (SYN_ITEMS(syn_block)[cur_si->si_idx].sp_flags
Bram Moolenaar071d4272004-06-13 20:20:40 +00001727 & (HL_SYNC_HERE|HL_SYNC_THERE)))
1728 return TRUE;
1729
1730 /* syn_current_attr() will have skipped the check for an item
Bram Moolenaar81366db2005-07-24 21:16:51 +00001731 * that ends here, need to do that now. Be careful not to go
1732 * past the NUL. */
1733 prev_current_col = current_col;
1734 if (syn_getcurline()[current_col] != NUL)
1735 ++current_col;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001736 check_state_ends();
Bram Moolenaar81366db2005-07-24 21:16:51 +00001737 current_col = prev_current_col;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001738 }
1739 ++current_col;
1740 }
1741 }
1742 return FALSE;
1743}
1744
1745/*
1746 * Return highlight attributes for next character.
1747 * Must first call syntax_start() once for the line.
1748 * "col" is normally 0 for the first use in a line, and increments by one each
1749 * time. It's allowed to skip characters and to stop before the end of the
1750 * line. But only a "col" after a previously used column is allowed.
Bram Moolenaar217ad922005-03-20 22:37:15 +00001751 * When "can_spell" is not NULL set it to TRUE when spell-checking should be
1752 * done.
Bram Moolenaar071d4272004-06-13 20:20:40 +00001753 */
1754 int
Bram Moolenaar860cae12010-06-05 23:22:07 +02001755get_syntax_attr(col, p_flags, can_spell, keep_state)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001756 colnr_T col;
Bram Moolenaar860cae12010-06-05 23:22:07 +02001757 int *p_flags UNUSED;
Bram Moolenaar217ad922005-03-20 22:37:15 +00001758 int *can_spell;
Bram Moolenaar56cefaf2008-01-12 15:47:10 +00001759 int keep_state; /* keep state of char at "col" */
Bram Moolenaar071d4272004-06-13 20:20:40 +00001760{
1761 int attr = 0;
1762
Bram Moolenaar349955a2007-08-14 21:07:36 +00001763 if (can_spell != NULL)
1764 /* Default: Only do spelling when there is no @Spell cluster or when
1765 * ":syn spell toplevel" was used. */
Bram Moolenaar860cae12010-06-05 23:22:07 +02001766 *can_spell = syn_block->b_syn_spell == SYNSPL_DEFAULT
1767 ? (syn_block->b_spell_cluster_id == 0)
1768 : (syn_block->b_syn_spell == SYNSPL_TOP);
Bram Moolenaar349955a2007-08-14 21:07:36 +00001769
Bram Moolenaar071d4272004-06-13 20:20:40 +00001770 /* check for out of memory situation */
Bram Moolenaar860cae12010-06-05 23:22:07 +02001771 if (syn_block->b_sst_array == NULL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001772 return 0;
1773
Bram Moolenaarce0842a2005-07-18 21:58:11 +00001774 /* After 'synmaxcol' the attribute is always zero. */
Bram Moolenaar84fb85a2005-07-20 22:02:14 +00001775 if (syn_buf->b_p_smc > 0 && col >= (colnr_T)syn_buf->b_p_smc)
Bram Moolenaarce0842a2005-07-18 21:58:11 +00001776 {
1777 clear_current_state();
1778#ifdef FEAT_EVAL
1779 current_id = 0;
1780 current_trans_id = 0;
1781#endif
1782 return 0;
1783 }
1784
Bram Moolenaar071d4272004-06-13 20:20:40 +00001785 /* Make sure current_state is valid */
1786 if (INVALID_STATE(&current_state))
1787 validate_current_state();
1788
1789 /*
1790 * Skip from the current column to "col", get the attributes for "col".
1791 */
1792 while (current_col <= col)
1793 {
Bram Moolenaar56cefaf2008-01-12 15:47:10 +00001794 attr = syn_current_attr(FALSE, TRUE, can_spell,
1795 current_col == col ? keep_state : FALSE);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001796 ++current_col;
1797 }
1798
Bram Moolenaar860cae12010-06-05 23:22:07 +02001799#ifdef FEAT_CONCEAL
1800 if (p_flags != NULL)
1801 *p_flags = current_flags;
1802#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00001803 return attr;
1804}
1805
1806/*
1807 * Get syntax attributes for current_lnum, current_col.
1808 */
1809 static int
Bram Moolenaar56cefaf2008-01-12 15:47:10 +00001810syn_current_attr(syncing, displaying, can_spell, keep_state)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001811 int syncing; /* When 1: called for syncing */
1812 int displaying; /* result will be displayed */
Bram Moolenaar217ad922005-03-20 22:37:15 +00001813 int *can_spell; /* return: do spell checking */
Bram Moolenaar56cefaf2008-01-12 15:47:10 +00001814 int keep_state; /* keep syntax stack afterwards */
Bram Moolenaar071d4272004-06-13 20:20:40 +00001815{
1816 int syn_id;
1817 lpos_T endpos; /* was: char_u *endp; */
1818 lpos_T hl_startpos; /* was: int hl_startcol; */
1819 lpos_T hl_endpos;
1820 lpos_T eos_pos; /* end-of-start match (start region) */
1821 lpos_T eoe_pos; /* end-of-end pattern */
1822 int end_idx; /* group ID for end pattern */
1823 int idx;
1824 synpat_T *spp;
Bram Moolenaar217ad922005-03-20 22:37:15 +00001825 stateitem_T *cur_si, *sip = NULL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001826 int startcol;
1827 int endcol;
1828 long flags;
Bram Moolenaar860cae12010-06-05 23:22:07 +02001829 int cchar;
Bram Moolenaar071d4272004-06-13 20:20:40 +00001830 short *next_list;
1831 int found_match; /* found usable match */
1832 static int try_next_column = FALSE; /* must try in next col */
1833 int do_keywords;
1834 regmmatch_T regmatch;
1835 lpos_T pos;
1836 int lc_col;
1837 reg_extmatch_T *cur_extmatch = NULL;
1838 char_u *line; /* current line. NOTE: becomes invalid after
1839 looking for a pattern match! */
1840
1841 /* variables for zero-width matches that have a "nextgroup" argument */
1842 int keep_next_list;
1843 int zero_width_next_list = FALSE;
1844 garray_T zero_width_next_ga;
1845
1846 /*
1847 * No character, no attributes! Past end of line?
1848 * Do try matching with an empty line (could be the start of a region).
1849 */
1850 line = syn_getcurline();
1851 if (line[current_col] == NUL && current_col != 0)
1852 {
1853 /*
1854 * If we found a match after the last column, use it.
1855 */
1856 if (next_match_idx >= 0 && next_match_col >= (int)current_col
1857 && next_match_col != MAXCOL)
1858 (void)push_next_match(NULL);
1859
1860 current_finished = TRUE;
1861 current_state_stored = FALSE;
1862 return 0;
1863 }
1864
1865 /* if the current or next character is NUL, we will finish the line now */
1866 if (line[current_col] == NUL || line[current_col + 1] == NUL)
1867 {
1868 current_finished = TRUE;
1869 current_state_stored = FALSE;
1870 }
1871
1872 /*
1873 * When in the previous column there was a match but it could not be used
1874 * (empty match or already matched in this column) need to try again in
1875 * the next column.
1876 */
1877 if (try_next_column)
1878 {
1879 next_match_idx = -1;
1880 try_next_column = FALSE;
1881 }
1882
1883 /* Only check for keywords when not syncing and there are some. */
1884 do_keywords = !syncing
Bram Moolenaar860cae12010-06-05 23:22:07 +02001885 && (syn_block->b_keywtab.ht_used > 0
1886 || syn_block->b_keywtab_ic.ht_used > 0);
Bram Moolenaar071d4272004-06-13 20:20:40 +00001887
1888 /* Init the list of zero-width matches with a nextlist. This is used to
1889 * avoid matching the same item in the same position twice. */
1890 ga_init2(&zero_width_next_ga, (int)sizeof(int), 10);
1891
1892 /*
1893 * Repeat matching keywords and patterns, to find contained items at the
1894 * same column. This stops when there are no extra matches at the current
1895 * column.
1896 */
1897 do
1898 {
1899 found_match = FALSE;
1900 keep_next_list = FALSE;
1901 syn_id = 0;
1902
1903 /*
1904 * 1. Check for a current state.
1905 * Only when there is no current state, or if the current state may
1906 * contain other things, we need to check for keywords and patterns.
1907 * Always need to check for contained items if some item has the
1908 * "containedin" argument (takes extra time!).
1909 */
1910 if (current_state.ga_len)
1911 cur_si = &CUR_STATE(current_state.ga_len - 1);
1912 else
1913 cur_si = NULL;
1914
Bram Moolenaar860cae12010-06-05 23:22:07 +02001915 if (syn_block->b_syn_containedin || cur_si == NULL
Bram Moolenaar071d4272004-06-13 20:20:40 +00001916 || cur_si->si_cont_list != NULL)
1917 {
1918 /*
1919 * 2. Check for keywords, if on a keyword char after a non-keyword
1920 * char. Don't do this when syncing.
1921 */
1922 if (do_keywords)
1923 {
1924 line = syn_getcurline();
1925 if (vim_iswordc_buf(line + current_col, syn_buf)
1926 && (current_col == 0
1927 || !vim_iswordc_buf(line + current_col - 1
1928#ifdef FEAT_MBYTE
1929 - (has_mbyte
1930 ? (*mb_head_off)(line, line + current_col - 1)
1931 : 0)
1932#endif
1933 , syn_buf)))
1934 {
1935 syn_id = check_keyword_id(line, (int)current_col,
Bram Moolenaar860cae12010-06-05 23:22:07 +02001936 &endcol, &flags, &next_list, cur_si,
1937 &cchar);
Bram Moolenaare2cc9702005-03-15 22:43:58 +00001938 if (syn_id != 0)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001939 {
1940 if (push_current_state(KEYWORD_IDX) == OK)
1941 {
1942 cur_si = &CUR_STATE(current_state.ga_len - 1);
1943 cur_si->si_m_startcol = current_col;
1944 cur_si->si_h_startpos.lnum = current_lnum;
1945 cur_si->si_h_startpos.col = 0; /* starts right away */
1946 cur_si->si_m_endpos.lnum = current_lnum;
1947 cur_si->si_m_endpos.col = endcol;
1948 cur_si->si_h_endpos.lnum = current_lnum;
1949 cur_si->si_h_endpos.col = endcol;
1950 cur_si->si_ends = TRUE;
1951 cur_si->si_end_idx = 0;
1952 cur_si->si_flags = flags;
Bram Moolenaar860cae12010-06-05 23:22:07 +02001953#ifdef FEAT_CONCEAL
1954 cur_si->si_char = cchar;
1955 if (current_state.ga_len > 1)
1956 cur_si->si_flags |=
1957 CUR_STATE(current_state.ga_len - 2).si_flags
1958 & HL_CONCEAL;
1959#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00001960 cur_si->si_id = syn_id;
1961 cur_si->si_trans_id = syn_id;
1962 if (flags & HL_TRANSP)
1963 {
1964 if (current_state.ga_len < 2)
1965 {
1966 cur_si->si_attr = 0;
1967 cur_si->si_trans_id = 0;
1968 }
1969 else
1970 {
1971 cur_si->si_attr = CUR_STATE(
1972 current_state.ga_len - 2).si_attr;
1973 cur_si->si_trans_id = CUR_STATE(
1974 current_state.ga_len - 2).si_trans_id;
1975 }
1976 }
1977 else
1978 cur_si->si_attr = syn_id2attr(syn_id);
1979 cur_si->si_cont_list = NULL;
1980 cur_si->si_next_list = next_list;
1981 check_keepend();
1982 }
1983 else
1984 vim_free(next_list);
1985 }
1986 }
1987 }
1988
1989 /*
Bram Moolenaare2cc9702005-03-15 22:43:58 +00001990 * 3. Check for patterns (only if no keyword found).
Bram Moolenaar071d4272004-06-13 20:20:40 +00001991 */
Bram Moolenaar860cae12010-06-05 23:22:07 +02001992 if (syn_id == 0 && syn_block->b_syn_patterns.ga_len)
Bram Moolenaar071d4272004-06-13 20:20:40 +00001993 {
1994 /*
1995 * If we didn't check for a match yet, or we are past it, check
1996 * for any match with a pattern.
1997 */
1998 if (next_match_idx < 0 || next_match_col < (int)current_col)
1999 {
2000 /*
2001 * Check all relevant patterns for a match at this
2002 * position. This is complicated, because matching with a
2003 * pattern takes quite a bit of time, thus we want to
2004 * avoid doing it when it's not needed.
2005 */
2006 next_match_idx = 0; /* no match in this line yet */
2007 next_match_col = MAXCOL;
Bram Moolenaar860cae12010-06-05 23:22:07 +02002008 for (idx = syn_block->b_syn_patterns.ga_len; --idx >= 0; )
Bram Moolenaar071d4272004-06-13 20:20:40 +00002009 {
Bram Moolenaar860cae12010-06-05 23:22:07 +02002010 spp = &(SYN_ITEMS(syn_block)[idx]);
Bram Moolenaar071d4272004-06-13 20:20:40 +00002011 if ( spp->sp_syncing == syncing
2012 && (displaying || !(spp->sp_flags & HL_DISPLAY))
2013 && (spp->sp_type == SPTYPE_MATCH
2014 || spp->sp_type == SPTYPE_START)
2015 && (current_next_list != NULL
2016 ? in_id_list(NULL, current_next_list,
2017 &spp->sp_syn, 0)
2018 : (cur_si == NULL
2019 ? !(spp->sp_flags & HL_CONTAINED)
2020 : in_id_list(cur_si,
2021 cur_si->si_cont_list, &spp->sp_syn,
2022 spp->sp_flags & HL_CONTAINED))))
2023 {
2024 /* If we already tried matching in this line, and
2025 * there isn't a match before next_match_col, skip
2026 * this item. */
2027 if (spp->sp_line_id == current_line_id
2028 && spp->sp_startcol >= next_match_col)
2029 continue;
2030 spp->sp_line_id = current_line_id;
2031
2032 lc_col = current_col - spp->sp_offsets[SPO_LC_OFF];
2033 if (lc_col < 0)
2034 lc_col = 0;
2035
2036 regmatch.rmm_ic = spp->sp_ic;
2037 regmatch.regprog = spp->sp_prog;
2038 if (!syn_regexec(&regmatch, current_lnum,
2039 (colnr_T)lc_col))
2040 {
2041 /* no match in this line, try another one */
2042 spp->sp_startcol = MAXCOL;
2043 continue;
2044 }
2045
2046 /*
2047 * Compute the first column of the match.
2048 */
2049 syn_add_start_off(&pos, &regmatch,
2050 spp, SPO_MS_OFF, -1);
2051 if (pos.lnum > current_lnum)
2052 {
2053 /* must have used end of match in a next line,
2054 * we can't handle that */
2055 spp->sp_startcol = MAXCOL;
2056 continue;
2057 }
2058 startcol = pos.col;
2059
2060 /* remember the next column where this pattern
2061 * matches in the current line */
2062 spp->sp_startcol = startcol;
2063
2064 /*
2065 * If a previously found match starts at a lower
2066 * column number, don't use this one.
2067 */
2068 if (startcol >= next_match_col)
2069 continue;
2070
2071 /*
2072 * If we matched this pattern at this position
2073 * before, skip it. Must retry in the next
2074 * column, because it may match from there.
2075 */
2076 if (did_match_already(idx, &zero_width_next_ga))
2077 {
2078 try_next_column = TRUE;
2079 continue;
2080 }
2081
2082 endpos.lnum = regmatch.endpos[0].lnum;
2083 endpos.col = regmatch.endpos[0].col;
2084
2085 /* Compute the highlight start. */
2086 syn_add_start_off(&hl_startpos, &regmatch,
2087 spp, SPO_HS_OFF, -1);
2088
2089 /* Compute the region start. */
2090 /* Default is to use the end of the match. */
2091 syn_add_end_off(&eos_pos, &regmatch,
2092 spp, SPO_RS_OFF, 0);
2093
2094 /*
2095 * Grab the external submatches before they get
2096 * overwritten. Reference count doesn't change.
2097 */
2098 unref_extmatch(cur_extmatch);
2099 cur_extmatch = re_extmatch_out;
2100 re_extmatch_out = NULL;
2101
2102 flags = 0;
2103 eoe_pos.lnum = 0; /* avoid warning */
2104 eoe_pos.col = 0;
2105 end_idx = 0;
2106 hl_endpos.lnum = 0;
2107
2108 /*
2109 * For a "oneline" the end must be found in the
2110 * same line too. Search for it after the end of
2111 * the match with the start pattern. Set the
2112 * resulting end positions at the same time.
2113 */
2114 if (spp->sp_type == SPTYPE_START
2115 && (spp->sp_flags & HL_ONELINE))
2116 {
2117 lpos_T startpos;
2118
2119 startpos = endpos;
2120 find_endpos(idx, &startpos, &endpos, &hl_endpos,
2121 &flags, &eoe_pos, &end_idx, cur_extmatch);
2122 if (endpos.lnum == 0)
2123 continue; /* not found */
2124 }
2125
2126 /*
2127 * For a "match" the size must be > 0 after the
2128 * end offset needs has been added. Except when
2129 * syncing.
2130 */
2131 else if (spp->sp_type == SPTYPE_MATCH)
2132 {
2133 syn_add_end_off(&hl_endpos, &regmatch, spp,
2134 SPO_HE_OFF, 0);
2135 syn_add_end_off(&endpos, &regmatch, spp,
2136 SPO_ME_OFF, 0);
2137 if (endpos.lnum == current_lnum
2138 && (int)endpos.col + syncing < startcol)
2139 {
2140 /*
2141 * If an empty string is matched, may need
2142 * to try matching again at next column.
2143 */
2144 if (regmatch.startpos[0].col
2145 == regmatch.endpos[0].col)
2146 try_next_column = TRUE;
2147 continue;
2148 }
2149 }
2150
2151 /*
2152 * keep the best match so far in next_match_*
2153 */
2154 /* Highlighting must start after startpos and end
2155 * before endpos. */
2156 if (hl_startpos.lnum == current_lnum
2157 && (int)hl_startpos.col < startcol)
2158 hl_startpos.col = startcol;
2159 limit_pos_zero(&hl_endpos, &endpos);
2160
2161 next_match_idx = idx;
2162 next_match_col = startcol;
2163 next_match_m_endpos = endpos;
2164 next_match_h_endpos = hl_endpos;
2165 next_match_h_startpos = hl_startpos;
2166 next_match_flags = flags;
2167 next_match_eos_pos = eos_pos;
2168 next_match_eoe_pos = eoe_pos;
2169 next_match_end_idx = end_idx;
2170 unref_extmatch(next_match_extmatch);
2171 next_match_extmatch = cur_extmatch;
2172 cur_extmatch = NULL;
2173 }
2174 }
2175 }
2176
2177 /*
2178 * If we found a match at the current column, use it.
2179 */
2180 if (next_match_idx >= 0 && next_match_col == (int)current_col)
2181 {
2182 synpat_T *lspp;
2183
2184 /* When a zero-width item matched which has a nextgroup,
2185 * don't push the item but set nextgroup. */
Bram Moolenaar860cae12010-06-05 23:22:07 +02002186 lspp = &(SYN_ITEMS(syn_block)[next_match_idx]);
Bram Moolenaar071d4272004-06-13 20:20:40 +00002187 if (next_match_m_endpos.lnum == current_lnum
2188 && next_match_m_endpos.col == current_col
2189 && lspp->sp_next_list != NULL)
2190 {
2191 current_next_list = lspp->sp_next_list;
2192 current_next_flags = lspp->sp_flags;
2193 keep_next_list = TRUE;
2194 zero_width_next_list = TRUE;
2195
2196 /* Add the index to a list, so that we can check
2197 * later that we don't match it again (and cause an
2198 * endless loop). */
2199 if (ga_grow(&zero_width_next_ga, 1) == OK)
2200 {
2201 ((int *)(zero_width_next_ga.ga_data))
2202 [zero_width_next_ga.ga_len++] = next_match_idx;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002203 }
2204 next_match_idx = -1;
2205 }
2206 else
2207 cur_si = push_next_match(cur_si);
2208 found_match = TRUE;
2209 }
2210 }
2211 }
2212
2213 /*
2214 * Handle searching for nextgroup match.
2215 */
2216 if (current_next_list != NULL && !keep_next_list)
2217 {
2218 /*
2219 * If a nextgroup was not found, continue looking for one if:
2220 * - this is an empty line and the "skipempty" option was given
2221 * - we are on white space and the "skipwhite" option was given
2222 */
2223 if (!found_match)
2224 {
2225 line = syn_getcurline();
2226 if (((current_next_flags & HL_SKIPWHITE)
2227 && vim_iswhite(line[current_col]))
2228 || ((current_next_flags & HL_SKIPEMPTY)
2229 && *line == NUL))
2230 break;
2231 }
2232
2233 /*
2234 * If a nextgroup was found: Use it, and continue looking for
2235 * contained matches.
2236 * If a nextgroup was not found: Continue looking for a normal
2237 * match.
2238 * When did set current_next_list for a zero-width item and no
2239 * match was found don't loop (would get stuck).
2240 */
2241 current_next_list = NULL;
2242 next_match_idx = -1;
2243 if (!zero_width_next_list)
2244 found_match = TRUE;
2245 }
2246
2247 } while (found_match);
2248
2249 /*
2250 * Use attributes from the current state, if within its highlighting.
2251 * If not, use attributes from the current-but-one state, etc.
2252 */
2253 current_attr = 0;
2254#ifdef FEAT_EVAL
2255 current_id = 0;
2256 current_trans_id = 0;
2257#endif
Bram Moolenaar860cae12010-06-05 23:22:07 +02002258#ifdef FEAT_CONCEAL
2259 current_flags = 0;
2260#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00002261 if (cur_si != NULL)
2262 {
Bram Moolenaar217ad922005-03-20 22:37:15 +00002263#ifndef FEAT_EVAL
2264 int current_trans_id = 0;
2265#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00002266 for (idx = current_state.ga_len - 1; idx >= 0; --idx)
2267 {
2268 sip = &CUR_STATE(idx);
2269 if ((current_lnum > sip->si_h_startpos.lnum
2270 || (current_lnum == sip->si_h_startpos.lnum
2271 && current_col >= sip->si_h_startpos.col))
2272 && (sip->si_h_endpos.lnum == 0
2273 || current_lnum < sip->si_h_endpos.lnum
2274 || (current_lnum == sip->si_h_endpos.lnum
2275 && current_col < sip->si_h_endpos.col)))
2276 {
2277 current_attr = sip->si_attr;
2278#ifdef FEAT_EVAL
2279 current_id = sip->si_id;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002280#endif
Bram Moolenaar217ad922005-03-20 22:37:15 +00002281 current_trans_id = sip->si_trans_id;
Bram Moolenaar860cae12010-06-05 23:22:07 +02002282#ifdef FEAT_CONCEAL
2283 current_flags = sip->si_flags;
2284 current_sub_char = sip->si_char;
2285#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00002286 break;
2287 }
2288 }
2289
Bram Moolenaar217ad922005-03-20 22:37:15 +00002290 if (can_spell != NULL)
2291 {
2292 struct sp_syn sps;
2293
2294 /*
2295 * set "can_spell" to TRUE if spell checking is supposed to be
2296 * done in the current item.
2297 */
Bram Moolenaar860cae12010-06-05 23:22:07 +02002298 if (syn_block->b_spell_cluster_id == 0)
Bram Moolenaar6bb68362005-03-22 23:03:44 +00002299 {
Bram Moolenaar3638c682005-06-08 22:05:14 +00002300 /* There is no @Spell cluster: Do spelling for items without
2301 * @NoSpell cluster. */
Bram Moolenaar860cae12010-06-05 23:22:07 +02002302 if (syn_block->b_nospell_cluster_id == 0
2303 || current_trans_id == 0)
2304 *can_spell = (syn_block->b_syn_spell != SYNSPL_NOTOP);
Bram Moolenaar6bb68362005-03-22 23:03:44 +00002305 else
2306 {
2307 sps.inc_tag = 0;
Bram Moolenaar860cae12010-06-05 23:22:07 +02002308 sps.id = syn_block->b_nospell_cluster_id;
Bram Moolenaar6bb68362005-03-22 23:03:44 +00002309 sps.cont_in_list = NULL;
2310 *can_spell = !in_id_list(sip, sip->si_cont_list, &sps, 0);
2311 }
2312 }
Bram Moolenaar217ad922005-03-20 22:37:15 +00002313 else
2314 {
Bram Moolenaar3638c682005-06-08 22:05:14 +00002315 /* The @Spell cluster is defined: Do spelling in items with
Bram Moolenaarce0842a2005-07-18 21:58:11 +00002316 * the @Spell cluster. But not when @NoSpell is also there.
2317 * At the toplevel only spell check when ":syn spell toplevel"
2318 * was used. */
Bram Moolenaar3638c682005-06-08 22:05:14 +00002319 if (current_trans_id == 0)
Bram Moolenaar860cae12010-06-05 23:22:07 +02002320 *can_spell = (syn_block->b_syn_spell == SYNSPL_TOP);
Bram Moolenaar3638c682005-06-08 22:05:14 +00002321 else
2322 {
2323 sps.inc_tag = 0;
Bram Moolenaar860cae12010-06-05 23:22:07 +02002324 sps.id = syn_block->b_spell_cluster_id;
Bram Moolenaar3638c682005-06-08 22:05:14 +00002325 sps.cont_in_list = NULL;
2326 *can_spell = in_id_list(sip, sip->si_cont_list, &sps, 0);
2327
Bram Moolenaar860cae12010-06-05 23:22:07 +02002328 if (syn_block->b_nospell_cluster_id != 0)
Bram Moolenaar3638c682005-06-08 22:05:14 +00002329 {
Bram Moolenaar860cae12010-06-05 23:22:07 +02002330 sps.id = syn_block->b_nospell_cluster_id;
Bram Moolenaar3638c682005-06-08 22:05:14 +00002331 if (in_id_list(sip, sip->si_cont_list, &sps, 0))
2332 *can_spell = FALSE;
2333 }
2334 }
Bram Moolenaar217ad922005-03-20 22:37:15 +00002335 }
2336 }
2337
2338
Bram Moolenaar071d4272004-06-13 20:20:40 +00002339 /*
2340 * Check for end of current state (and the states before it) at the
2341 * next column. Don't do this for syncing, because we would miss a
2342 * single character match.
2343 * First check if the current state ends at the current column. It
2344 * may be for an empty match and a containing item might end in the
2345 * current column.
2346 */
Bram Moolenaar56cefaf2008-01-12 15:47:10 +00002347 if (!syncing && !keep_state)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002348 {
2349 check_state_ends();
Bram Moolenaar81366db2005-07-24 21:16:51 +00002350 if (current_state.ga_len > 0
2351 && syn_getcurline()[current_col] != NUL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002352 {
2353 ++current_col;
2354 check_state_ends();
2355 --current_col;
2356 }
2357 }
2358 }
Bram Moolenaar217ad922005-03-20 22:37:15 +00002359 else if (can_spell != NULL)
Bram Moolenaarce0842a2005-07-18 21:58:11 +00002360 /* Default: Only do spelling when there is no @Spell cluster or when
2361 * ":syn spell toplevel" was used. */
Bram Moolenaar860cae12010-06-05 23:22:07 +02002362 *can_spell = syn_block->b_syn_spell == SYNSPL_DEFAULT
2363 ? (syn_block->b_spell_cluster_id == 0)
2364 : (syn_block->b_syn_spell == SYNSPL_TOP);
Bram Moolenaar071d4272004-06-13 20:20:40 +00002365
Bram Moolenaarf5b63862009-12-16 17:13:44 +00002366 /* nextgroup ends at end of line, unless "skipnl" or "skipempty" present */
Bram Moolenaar071d4272004-06-13 20:20:40 +00002367 if (current_next_list != NULL
2368 && syn_getcurline()[current_col + 1] == NUL
2369 && !(current_next_flags & (HL_SKIPNL | HL_SKIPEMPTY)))
2370 current_next_list = NULL;
2371
2372 if (zero_width_next_ga.ga_len > 0)
2373 ga_clear(&zero_width_next_ga);
2374
2375 /* No longer need external matches. But keep next_match_extmatch. */
2376 unref_extmatch(re_extmatch_out);
2377 re_extmatch_out = NULL;
2378 unref_extmatch(cur_extmatch);
2379
2380 return current_attr;
2381}
2382
2383
2384/*
2385 * Check if we already matched pattern "idx" at the current column.
2386 */
2387 static int
2388did_match_already(idx, gap)
2389 int idx;
2390 garray_T *gap;
2391{
2392 int i;
2393
2394 for (i = current_state.ga_len; --i >= 0; )
2395 if (CUR_STATE(i).si_m_startcol == (int)current_col
2396 && CUR_STATE(i).si_m_lnum == (int)current_lnum
2397 && CUR_STATE(i).si_idx == idx)
2398 return TRUE;
2399
2400 /* Zero-width matches with a nextgroup argument are not put on the syntax
2401 * stack, and can only be matched once anyway. */
2402 for (i = gap->ga_len; --i >= 0; )
2403 if (((int *)(gap->ga_data))[i] == idx)
2404 return TRUE;
2405
2406 return FALSE;
2407}
2408
2409/*
2410 * Push the next match onto the stack.
2411 */
2412 static stateitem_T *
2413push_next_match(cur_si)
2414 stateitem_T *cur_si;
2415{
2416 synpat_T *spp;
Bram Moolenaar860cae12010-06-05 23:22:07 +02002417#ifdef FEAT_CONCEAL
2418 int save_flags;
2419#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00002420
Bram Moolenaar860cae12010-06-05 23:22:07 +02002421 spp = &(SYN_ITEMS(syn_block)[next_match_idx]);
Bram Moolenaar071d4272004-06-13 20:20:40 +00002422
2423 /*
2424 * Push the item in current_state stack;
2425 */
2426 if (push_current_state(next_match_idx) == OK)
2427 {
2428 /*
2429 * If it's a start-skip-end type that crosses lines, figure out how
2430 * much it continues in this line. Otherwise just fill in the length.
2431 */
2432 cur_si = &CUR_STATE(current_state.ga_len - 1);
2433 cur_si->si_h_startpos = next_match_h_startpos;
2434 cur_si->si_m_startcol = current_col;
2435 cur_si->si_m_lnum = current_lnum;
2436 cur_si->si_flags = spp->sp_flags;
Bram Moolenaar860cae12010-06-05 23:22:07 +02002437#ifdef FEAT_CONCEAL
2438 cur_si->si_char = spp->sp_char;
2439 if (current_state.ga_len > 1)
2440 cur_si->si_flags |=
2441 CUR_STATE(current_state.ga_len - 2).si_flags & HL_CONCEAL;
2442#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00002443 cur_si->si_next_list = spp->sp_next_list;
2444 cur_si->si_extmatch = ref_extmatch(next_match_extmatch);
2445 if (spp->sp_type == SPTYPE_START && !(spp->sp_flags & HL_ONELINE))
2446 {
2447 /* Try to find the end pattern in the current line */
2448 update_si_end(cur_si, (int)(next_match_m_endpos.col), TRUE);
2449 check_keepend();
2450 }
2451 else
2452 {
2453 cur_si->si_m_endpos = next_match_m_endpos;
2454 cur_si->si_h_endpos = next_match_h_endpos;
2455 cur_si->si_ends = TRUE;
2456 cur_si->si_flags |= next_match_flags;
2457 cur_si->si_eoe_pos = next_match_eoe_pos;
2458 cur_si->si_end_idx = next_match_end_idx;
2459 }
2460 if (keepend_level < 0 && (cur_si->si_flags & HL_KEEPEND))
2461 keepend_level = current_state.ga_len - 1;
2462 check_keepend();
2463 update_si_attr(current_state.ga_len - 1);
2464
Bram Moolenaar860cae12010-06-05 23:22:07 +02002465#ifdef FEAT_CONCEAL
2466 save_flags = cur_si->si_flags & (HL_CONCEAL | HL_CONCEALENDS);
2467#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00002468 /*
2469 * If the start pattern has another highlight group, push another item
2470 * on the stack for the start pattern.
2471 */
2472 if ( spp->sp_type == SPTYPE_START
2473 && spp->sp_syn_match_id != 0
2474 && push_current_state(next_match_idx) == OK)
2475 {
2476 cur_si = &CUR_STATE(current_state.ga_len - 1);
2477 cur_si->si_h_startpos = next_match_h_startpos;
2478 cur_si->si_m_startcol = current_col;
2479 cur_si->si_m_lnum = current_lnum;
2480 cur_si->si_m_endpos = next_match_eos_pos;
2481 cur_si->si_h_endpos = next_match_eos_pos;
2482 cur_si->si_ends = TRUE;
2483 cur_si->si_end_idx = 0;
2484 cur_si->si_flags = HL_MATCH;
Bram Moolenaar860cae12010-06-05 23:22:07 +02002485#ifdef FEAT_CONCEAL
2486 cur_si->si_flags |= save_flags;
2487 if (cur_si->si_flags & HL_CONCEALENDS)
2488 cur_si->si_flags |= HL_CONCEAL;
2489#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00002490 cur_si->si_next_list = NULL;
2491 check_keepend();
2492 update_si_attr(current_state.ga_len - 1);
2493 }
2494 }
2495
2496 next_match_idx = -1; /* try other match next time */
2497
2498 return cur_si;
2499}
2500
2501/*
2502 * Check for end of current state (and the states before it).
2503 */
2504 static void
2505check_state_ends()
2506{
2507 stateitem_T *cur_si;
2508 int had_extend = FALSE;
2509
2510 cur_si = &CUR_STATE(current_state.ga_len - 1);
2511 for (;;)
2512 {
2513 if (cur_si->si_ends
2514 && (cur_si->si_m_endpos.lnum < current_lnum
2515 || (cur_si->si_m_endpos.lnum == current_lnum
2516 && cur_si->si_m_endpos.col <= current_col)))
2517 {
2518 /*
2519 * If there is an end pattern group ID, highlight the end pattern
2520 * now. No need to pop the current item from the stack.
2521 * Only do this if the end pattern continues beyond the current
2522 * position.
2523 */
2524 if (cur_si->si_end_idx
2525 && (cur_si->si_eoe_pos.lnum > current_lnum
2526 || (cur_si->si_eoe_pos.lnum == current_lnum
2527 && cur_si->si_eoe_pos.col > current_col)))
2528 {
2529 cur_si->si_idx = cur_si->si_end_idx;
2530 cur_si->si_end_idx = 0;
2531 cur_si->si_m_endpos = cur_si->si_eoe_pos;
2532 cur_si->si_h_endpos = cur_si->si_eoe_pos;
2533 cur_si->si_flags |= HL_MATCH;
Bram Moolenaar860cae12010-06-05 23:22:07 +02002534#ifdef FEAT_CONCEAL
2535 if (cur_si->si_flags & HL_CONCEALENDS)
2536 cur_si->si_flags |= HL_CONCEAL;
2537#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00002538 update_si_attr(current_state.ga_len - 1);
Bram Moolenaar293ee4d2004-12-09 21:34:53 +00002539
2540 /* what matches next may be different now, clear it */
2541 next_match_idx = 0;
2542 next_match_col = MAXCOL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002543 break;
2544 }
2545 else
2546 {
2547 /* handle next_list, unless at end of line and no "skipnl" or
2548 * "skipempty" */
2549 current_next_list = cur_si->si_next_list;
2550 current_next_flags = cur_si->si_flags;
2551 if (!(current_next_flags & (HL_SKIPNL | HL_SKIPEMPTY))
2552 && syn_getcurline()[current_col] == NUL)
2553 current_next_list = NULL;
2554
2555 /* When the ended item has "extend", another item with
2556 * "keepend" now needs to check for its end. */
2557 if (cur_si->si_flags & HL_EXTEND)
2558 had_extend = TRUE;
2559
2560 pop_current_state();
2561
2562 if (current_state.ga_len == 0)
2563 break;
2564
Bram Moolenaar81993f42008-01-11 20:27:45 +00002565 if (had_extend && keepend_level >= 0)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002566 {
2567 syn_update_ends(FALSE);
2568 if (current_state.ga_len == 0)
2569 break;
2570 }
2571
2572 cur_si = &CUR_STATE(current_state.ga_len - 1);
2573
2574 /*
2575 * Only for a region the search for the end continues after
2576 * the end of the contained item. If the contained match
2577 * included the end-of-line, break here, the region continues.
2578 * Don't do this when:
2579 * - "keepend" is used for the contained item
2580 * - not at the end of the line (could be end="x$"me=e-1).
2581 * - "excludenl" is used (HL_HAS_EOL won't be set)
2582 */
2583 if (cur_si->si_idx >= 0
Bram Moolenaar860cae12010-06-05 23:22:07 +02002584 && SYN_ITEMS(syn_block)[cur_si->si_idx].sp_type
Bram Moolenaar071d4272004-06-13 20:20:40 +00002585 == SPTYPE_START
2586 && !(cur_si->si_flags & (HL_MATCH | HL_KEEPEND)))
2587 {
2588 update_si_end(cur_si, (int)current_col, TRUE);
2589 check_keepend();
2590 if ((current_next_flags & HL_HAS_EOL)
2591 && keepend_level < 0
2592 && syn_getcurline()[current_col] == NUL)
2593 break;
2594 }
2595 }
2596 }
2597 else
2598 break;
2599 }
2600}
2601
2602/*
2603 * Update an entry in the current_state stack for a match or region. This
2604 * fills in si_attr, si_next_list and si_cont_list.
2605 */
2606 static void
2607update_si_attr(idx)
2608 int idx;
2609{
2610 stateitem_T *sip = &CUR_STATE(idx);
2611 synpat_T *spp;
2612
Bram Moolenaar3a36cf72007-08-21 15:29:56 +00002613 /* This should not happen... */
2614 if (sip->si_idx < 0)
2615 return;
2616
Bram Moolenaar860cae12010-06-05 23:22:07 +02002617 spp = &(SYN_ITEMS(syn_block)[sip->si_idx]);
Bram Moolenaar071d4272004-06-13 20:20:40 +00002618 if (sip->si_flags & HL_MATCH)
2619 sip->si_id = spp->sp_syn_match_id;
2620 else
2621 sip->si_id = spp->sp_syn.id;
2622 sip->si_attr = syn_id2attr(sip->si_id);
2623 sip->si_trans_id = sip->si_id;
2624 if (sip->si_flags & HL_MATCH)
2625 sip->si_cont_list = NULL;
2626 else
2627 sip->si_cont_list = spp->sp_cont_list;
2628
2629 /*
2630 * For transparent items, take attr from outer item.
2631 * Also take cont_list, if there is none.
2632 * Don't do this for the matchgroup of a start or end pattern.
2633 */
2634 if ((spp->sp_flags & HL_TRANSP) && !(sip->si_flags & HL_MATCH))
2635 {
2636 if (idx == 0)
2637 {
2638 sip->si_attr = 0;
2639 sip->si_trans_id = 0;
2640 if (sip->si_cont_list == NULL)
2641 sip->si_cont_list = ID_LIST_ALL;
2642 }
2643 else
2644 {
2645 sip->si_attr = CUR_STATE(idx - 1).si_attr;
2646 sip->si_trans_id = CUR_STATE(idx - 1).si_trans_id;
Bram Moolenaar293ee4d2004-12-09 21:34:53 +00002647 sip->si_h_startpos = CUR_STATE(idx - 1).si_h_startpos;
2648 sip->si_h_endpos = CUR_STATE(idx - 1).si_h_endpos;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002649 if (sip->si_cont_list == NULL)
2650 {
2651 sip->si_flags |= HL_TRANS_CONT;
2652 sip->si_cont_list = CUR_STATE(idx - 1).si_cont_list;
2653 }
2654 }
2655 }
2656}
2657
2658/*
2659 * Check the current stack for patterns with "keepend" flag.
2660 * Propagate the match-end to contained items, until a "skipend" item is found.
2661 */
2662 static void
2663check_keepend()
2664{
2665 int i;
2666 lpos_T maxpos;
Bram Moolenaar7bd2cd82006-10-03 15:04:36 +00002667 lpos_T maxpos_h;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002668 stateitem_T *sip;
2669
2670 /*
2671 * This check can consume a lot of time; only do it from the level where
2672 * there really is a keepend.
2673 */
2674 if (keepend_level < 0)
2675 return;
2676
2677 /*
2678 * Find the last index of an "extend" item. "keepend" items before that
2679 * won't do anything. If there is no "extend" item "i" will be
2680 * "keepend_level" and all "keepend" items will work normally.
2681 */
2682 for (i = current_state.ga_len - 1; i > keepend_level; --i)
2683 if (CUR_STATE(i).si_flags & HL_EXTEND)
2684 break;
2685
2686 maxpos.lnum = 0;
Bram Moolenaared39e1d2008-08-09 17:55:22 +00002687 maxpos.col = 0;
Bram Moolenaar7bd2cd82006-10-03 15:04:36 +00002688 maxpos_h.lnum = 0;
Bram Moolenaared39e1d2008-08-09 17:55:22 +00002689 maxpos_h.col = 0;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002690 for ( ; i < current_state.ga_len; ++i)
2691 {
2692 sip = &CUR_STATE(i);
2693 if (maxpos.lnum != 0)
2694 {
2695 limit_pos_zero(&sip->si_m_endpos, &maxpos);
Bram Moolenaar7bd2cd82006-10-03 15:04:36 +00002696 limit_pos_zero(&sip->si_h_endpos, &maxpos_h);
Bram Moolenaar071d4272004-06-13 20:20:40 +00002697 limit_pos_zero(&sip->si_eoe_pos, &maxpos);
2698 sip->si_ends = TRUE;
2699 }
Bram Moolenaar7bd2cd82006-10-03 15:04:36 +00002700 if (sip->si_ends && (sip->si_flags & HL_KEEPEND))
2701 {
2702 if (maxpos.lnum == 0
Bram Moolenaar071d4272004-06-13 20:20:40 +00002703 || maxpos.lnum > sip->si_m_endpos.lnum
2704 || (maxpos.lnum == sip->si_m_endpos.lnum
Bram Moolenaar7bd2cd82006-10-03 15:04:36 +00002705 && maxpos.col > sip->si_m_endpos.col))
2706 maxpos = sip->si_m_endpos;
2707 if (maxpos_h.lnum == 0
2708 || maxpos_h.lnum > sip->si_h_endpos.lnum
2709 || (maxpos_h.lnum == sip->si_h_endpos.lnum
2710 && maxpos_h.col > sip->si_h_endpos.col))
2711 maxpos_h = sip->si_h_endpos;
2712 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00002713 }
2714}
2715
2716/*
2717 * Update an entry in the current_state stack for a start-skip-end pattern.
2718 * This finds the end of the current item, if it's in the current line.
2719 *
2720 * Return the flags for the matched END.
2721 */
2722 static void
2723update_si_end(sip, startcol, force)
2724 stateitem_T *sip;
2725 int startcol; /* where to start searching for the end */
2726 int force; /* when TRUE overrule a previous end */
2727{
2728 lpos_T startpos;
2729 lpos_T endpos;
2730 lpos_T hl_endpos;
2731 lpos_T end_endpos;
2732 int end_idx;
2733
Bram Moolenaar3a36cf72007-08-21 15:29:56 +00002734 /* return quickly for a keyword */
2735 if (sip->si_idx < 0)
2736 return;
2737
Bram Moolenaar071d4272004-06-13 20:20:40 +00002738 /* Don't update when it's already done. Can be a match of an end pattern
2739 * that started in a previous line. Watch out: can also be a "keepend"
2740 * from a containing item. */
2741 if (!force && sip->si_m_endpos.lnum >= current_lnum)
2742 return;
2743
2744 /*
2745 * We need to find the end of the region. It may continue in the next
2746 * line.
2747 */
2748 end_idx = 0;
2749 startpos.lnum = current_lnum;
2750 startpos.col = startcol;
2751 find_endpos(sip->si_idx, &startpos, &endpos, &hl_endpos,
2752 &(sip->si_flags), &end_endpos, &end_idx, sip->si_extmatch);
2753
2754 if (endpos.lnum == 0)
2755 {
2756 /* No end pattern matched. */
Bram Moolenaar860cae12010-06-05 23:22:07 +02002757 if (SYN_ITEMS(syn_block)[sip->si_idx].sp_flags & HL_ONELINE)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002758 {
2759 /* a "oneline" never continues in the next line */
2760 sip->si_ends = TRUE;
2761 sip->si_m_endpos.lnum = current_lnum;
2762 sip->si_m_endpos.col = (colnr_T)STRLEN(syn_getcurline());
2763 }
2764 else
2765 {
2766 /* continues in the next line */
2767 sip->si_ends = FALSE;
2768 sip->si_m_endpos.lnum = 0;
2769 }
2770 sip->si_h_endpos = sip->si_m_endpos;
2771 }
2772 else
2773 {
2774 /* match within this line */
2775 sip->si_m_endpos = endpos;
2776 sip->si_h_endpos = hl_endpos;
2777 sip->si_eoe_pos = end_endpos;
2778 sip->si_ends = TRUE;
2779 sip->si_end_idx = end_idx;
2780 }
2781}
2782
2783/*
2784 * Add a new state to the current state stack.
2785 * It is cleared and the index set to "idx".
2786 * Return FAIL if it's not possible (out of memory).
2787 */
2788 static int
2789push_current_state(idx)
2790 int idx;
2791{
2792 if (ga_grow(&current_state, 1) == FAIL)
2793 return FAIL;
2794 vim_memset(&CUR_STATE(current_state.ga_len), 0, sizeof(stateitem_T));
2795 CUR_STATE(current_state.ga_len).si_idx = idx;
2796 ++current_state.ga_len;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002797 return OK;
2798}
2799
2800/*
2801 * Remove a state from the current_state stack.
2802 */
2803 static void
2804pop_current_state()
2805{
2806 if (current_state.ga_len)
2807 {
2808 unref_extmatch(CUR_STATE(current_state.ga_len - 1).si_extmatch);
2809 --current_state.ga_len;
Bram Moolenaar071d4272004-06-13 20:20:40 +00002810 }
2811 /* after the end of a pattern, try matching a keyword or pattern */
2812 next_match_idx = -1;
2813
2814 /* if first state with "keepend" is popped, reset keepend_level */
2815 if (keepend_level >= current_state.ga_len)
2816 keepend_level = -1;
2817}
2818
2819/*
2820 * Find the end of a start/skip/end syntax region after "startpos".
2821 * Only checks one line.
2822 * Also handles a match item that continued from a previous line.
2823 * If not found, the syntax item continues in the next line. m_endpos->lnum
2824 * will be 0.
2825 * If found, the end of the region and the end of the highlighting is
2826 * computed.
2827 */
2828 static void
2829find_endpos(idx, startpos, m_endpos, hl_endpos, flagsp, end_endpos,
2830 end_idx, start_ext)
2831 int idx; /* index of the pattern */
2832 lpos_T *startpos; /* where to start looking for an END match */
2833 lpos_T *m_endpos; /* return: end of match */
2834 lpos_T *hl_endpos; /* return: end of highlighting */
2835 long *flagsp; /* return: flags of matching END */
2836 lpos_T *end_endpos; /* return: end of end pattern match */
2837 int *end_idx; /* return: group ID for end pat. match, or 0 */
2838 reg_extmatch_T *start_ext; /* submatches from the start pattern */
2839{
2840 colnr_T matchcol;
2841 synpat_T *spp, *spp_skip;
2842 int start_idx;
2843 int best_idx;
2844 regmmatch_T regmatch;
2845 regmmatch_T best_regmatch; /* startpos/endpos of best match */
2846 lpos_T pos;
2847 char_u *line;
2848 int had_match = FALSE;
2849
Bram Moolenaar3a36cf72007-08-21 15:29:56 +00002850 /* just in case we are invoked for a keyword */
2851 if (idx < 0)
2852 return;
2853
Bram Moolenaar071d4272004-06-13 20:20:40 +00002854 /*
2855 * Check for being called with a START pattern.
2856 * Can happen with a match that continues to the next line, because it
2857 * contained a region.
2858 */
Bram Moolenaar860cae12010-06-05 23:22:07 +02002859 spp = &(SYN_ITEMS(syn_block)[idx]);
Bram Moolenaar071d4272004-06-13 20:20:40 +00002860 if (spp->sp_type != SPTYPE_START)
2861 {
2862 *hl_endpos = *startpos;
2863 return;
2864 }
2865
2866 /*
2867 * Find the SKIP or first END pattern after the last START pattern.
2868 */
2869 for (;;)
2870 {
Bram Moolenaar860cae12010-06-05 23:22:07 +02002871 spp = &(SYN_ITEMS(syn_block)[idx]);
Bram Moolenaar071d4272004-06-13 20:20:40 +00002872 if (spp->sp_type != SPTYPE_START)
2873 break;
2874 ++idx;
2875 }
2876
2877 /*
2878 * Lookup the SKIP pattern (if present)
2879 */
2880 if (spp->sp_type == SPTYPE_SKIP)
2881 {
2882 spp_skip = spp;
2883 ++idx;
2884 }
2885 else
2886 spp_skip = NULL;
2887
2888 /* Setup external matches for syn_regexec(). */
2889 unref_extmatch(re_extmatch_in);
2890 re_extmatch_in = ref_extmatch(start_ext);
2891
2892 matchcol = startpos->col; /* start looking for a match at sstart */
2893 start_idx = idx; /* remember the first END pattern. */
2894 best_regmatch.startpos[0].col = 0; /* avoid compiler warning */
2895 for (;;)
2896 {
2897 /*
2898 * Find end pattern that matches first after "matchcol".
2899 */
2900 best_idx = -1;
Bram Moolenaar860cae12010-06-05 23:22:07 +02002901 for (idx = start_idx; idx < syn_block->b_syn_patterns.ga_len; ++idx)
Bram Moolenaar071d4272004-06-13 20:20:40 +00002902 {
2903 int lc_col = matchcol;
2904
Bram Moolenaar860cae12010-06-05 23:22:07 +02002905 spp = &(SYN_ITEMS(syn_block)[idx]);
Bram Moolenaar071d4272004-06-13 20:20:40 +00002906 if (spp->sp_type != SPTYPE_END) /* past last END pattern */
2907 break;
2908 lc_col -= spp->sp_offsets[SPO_LC_OFF];
2909 if (lc_col < 0)
2910 lc_col = 0;
2911
2912 regmatch.rmm_ic = spp->sp_ic;
2913 regmatch.regprog = spp->sp_prog;
2914 if (syn_regexec(&regmatch, startpos->lnum, lc_col))
2915 {
2916 if (best_idx == -1 || regmatch.startpos[0].col
2917 < best_regmatch.startpos[0].col)
2918 {
2919 best_idx = idx;
2920 best_regmatch.startpos[0] = regmatch.startpos[0];
2921 best_regmatch.endpos[0] = regmatch.endpos[0];
2922 }
2923 }
2924 }
2925
2926 /*
2927 * If all end patterns have been tried, and there is no match, the
2928 * item continues until end-of-line.
2929 */
2930 if (best_idx == -1)
2931 break;
2932
2933 /*
2934 * If the skip pattern matches before the end pattern,
2935 * continue searching after the skip pattern.
2936 */
2937 if (spp_skip != NULL)
2938 {
2939 int lc_col = matchcol - spp_skip->sp_offsets[SPO_LC_OFF];
2940
2941 if (lc_col < 0)
2942 lc_col = 0;
2943 regmatch.rmm_ic = spp_skip->sp_ic;
2944 regmatch.regprog = spp_skip->sp_prog;
2945 if (syn_regexec(&regmatch, startpos->lnum, lc_col)
2946 && regmatch.startpos[0].col
2947 <= best_regmatch.startpos[0].col)
2948 {
2949 /* Add offset to skip pattern match */
2950 syn_add_end_off(&pos, &regmatch, spp_skip, SPO_ME_OFF, 1);
2951
2952 /* If the skip pattern goes on to the next line, there is no
2953 * match with an end pattern in this line. */
2954 if (pos.lnum > startpos->lnum)
2955 break;
2956
2957 line = ml_get_buf(syn_buf, startpos->lnum, FALSE);
2958
2959 /* take care of an empty match or negative offset */
2960 if (pos.col <= matchcol)
2961 ++matchcol;
2962 else if (pos.col <= regmatch.endpos[0].col)
2963 matchcol = pos.col;
2964 else
2965 /* Be careful not to jump over the NUL at the end-of-line */
2966 for (matchcol = regmatch.endpos[0].col;
2967 line[matchcol] != NUL && matchcol < pos.col;
2968 ++matchcol)
2969 ;
2970
2971 /* if the skip pattern includes end-of-line, break here */
2972 if (line[matchcol] == NUL)
2973 break;
2974
2975 continue; /* start with first end pattern again */
2976 }
2977 }
2978
2979 /*
2980 * Match from start pattern to end pattern.
2981 * Correct for match and highlight offset of end pattern.
2982 */
Bram Moolenaar860cae12010-06-05 23:22:07 +02002983 spp = &(SYN_ITEMS(syn_block)[best_idx]);
Bram Moolenaar071d4272004-06-13 20:20:40 +00002984 syn_add_end_off(m_endpos, &best_regmatch, spp, SPO_ME_OFF, 1);
2985 /* can't end before the start */
2986 if (m_endpos->lnum == startpos->lnum && m_endpos->col < startpos->col)
2987 m_endpos->col = startpos->col;
2988
2989 syn_add_end_off(end_endpos, &best_regmatch, spp, SPO_HE_OFF, 1);
2990 /* can't end before the start */
2991 if (end_endpos->lnum == startpos->lnum
2992 && end_endpos->col < startpos->col)
2993 end_endpos->col = startpos->col;
2994 /* can't end after the match */
2995 limit_pos(end_endpos, m_endpos);
2996
2997 /*
2998 * If the end group is highlighted differently, adjust the pointers.
2999 */
3000 if (spp->sp_syn_match_id != spp->sp_syn.id && spp->sp_syn_match_id != 0)
3001 {
3002 *end_idx = best_idx;
3003 if (spp->sp_off_flags & (1 << (SPO_RE_OFF + SPO_COUNT)))
3004 {
3005 hl_endpos->lnum = best_regmatch.endpos[0].lnum;
3006 hl_endpos->col = best_regmatch.endpos[0].col;
3007 }
3008 else
3009 {
3010 hl_endpos->lnum = best_regmatch.startpos[0].lnum;
3011 hl_endpos->col = best_regmatch.startpos[0].col;
3012 }
3013 hl_endpos->col += spp->sp_offsets[SPO_RE_OFF];
3014
3015 /* can't end before the start */
3016 if (hl_endpos->lnum == startpos->lnum
3017 && hl_endpos->col < startpos->col)
3018 hl_endpos->col = startpos->col;
3019 limit_pos(hl_endpos, m_endpos);
3020
3021 /* now the match ends where the highlighting ends, it is turned
3022 * into the matchgroup for the end */
3023 *m_endpos = *hl_endpos;
3024 }
3025 else
3026 {
3027 *end_idx = 0;
3028 *hl_endpos = *end_endpos;
3029 }
3030
3031 *flagsp = spp->sp_flags;
3032
3033 had_match = TRUE;
3034 break;
3035 }
3036
3037 /* no match for an END pattern in this line */
3038 if (!had_match)
3039 m_endpos->lnum = 0;
3040
3041 /* Remove external matches. */
3042 unref_extmatch(re_extmatch_in);
3043 re_extmatch_in = NULL;
3044}
3045
3046/*
3047 * Limit "pos" not to be after "limit".
3048 */
3049 static void
3050limit_pos(pos, limit)
3051 lpos_T *pos;
3052 lpos_T *limit;
3053{
3054 if (pos->lnum > limit->lnum)
3055 *pos = *limit;
3056 else if (pos->lnum == limit->lnum && pos->col > limit->col)
3057 pos->col = limit->col;
3058}
3059
3060/*
3061 * Limit "pos" not to be after "limit", unless pos->lnum is zero.
3062 */
3063 static void
3064limit_pos_zero(pos, limit)
3065 lpos_T *pos;
3066 lpos_T *limit;
3067{
3068 if (pos->lnum == 0)
3069 *pos = *limit;
3070 else
3071 limit_pos(pos, limit);
3072}
3073
3074/*
3075 * Add offset to matched text for end of match or highlight.
3076 */
3077 static void
3078syn_add_end_off(result, regmatch, spp, idx, extra)
3079 lpos_T *result; /* returned position */
3080 regmmatch_T *regmatch; /* start/end of match */
3081 synpat_T *spp; /* matched pattern */
3082 int idx; /* index of offset */
3083 int extra; /* extra chars for offset to start */
3084{
3085 int col;
Bram Moolenaar8c8de832008-06-24 22:58:06 +00003086 int off;
3087 char_u *base;
3088 char_u *p;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003089
3090 if (spp->sp_off_flags & (1 << idx))
3091 {
3092 result->lnum = regmatch->startpos[0].lnum;
Bram Moolenaar8c8de832008-06-24 22:58:06 +00003093 col = regmatch->startpos[0].col;
3094 off = spp->sp_offsets[idx] + extra;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003095 }
3096 else
3097 {
3098 result->lnum = regmatch->endpos[0].lnum;
3099 col = regmatch->endpos[0].col;
Bram Moolenaar8c8de832008-06-24 22:58:06 +00003100 off = spp->sp_offsets[idx];
Bram Moolenaar071d4272004-06-13 20:20:40 +00003101 }
Bram Moolenaar8c8de832008-06-24 22:58:06 +00003102 /* Don't go past the end of the line. Matters for "rs=e+2" when there
3103 * is a matchgroup. Watch out for match with last NL in the buffer. */
3104 if (result->lnum > syn_buf->b_ml.ml_line_count)
3105 col = 0;
3106 else if (off != 0)
Bram Moolenaara40ceaf2006-01-13 22:35:40 +00003107 {
Bram Moolenaar8c8de832008-06-24 22:58:06 +00003108 base = ml_get_buf(syn_buf, result->lnum, FALSE);
3109 p = base + col;
3110 if (off > 0)
3111 {
3112 while (off-- > 0 && *p != NUL)
3113 mb_ptr_adv(p);
3114 }
3115 else if (off < 0)
3116 {
3117 while (off++ < 0 && base < p)
3118 mb_ptr_back(base, p);
3119 }
3120 col = (int)(p - base);
Bram Moolenaara40ceaf2006-01-13 22:35:40 +00003121 }
Bram Moolenaar8c8de832008-06-24 22:58:06 +00003122 result->col = col;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003123}
3124
3125/*
3126 * Add offset to matched text for start of match or highlight.
3127 * Avoid resulting column to become negative.
3128 */
3129 static void
3130syn_add_start_off(result, regmatch, spp, idx, extra)
3131 lpos_T *result; /* returned position */
3132 regmmatch_T *regmatch; /* start/end of match */
3133 synpat_T *spp;
3134 int idx;
3135 int extra; /* extra chars for offset to end */
3136{
3137 int col;
Bram Moolenaar8c8de832008-06-24 22:58:06 +00003138 int off;
3139 char_u *base;
3140 char_u *p;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003141
3142 if (spp->sp_off_flags & (1 << (idx + SPO_COUNT)))
3143 {
3144 result->lnum = regmatch->endpos[0].lnum;
Bram Moolenaar8c8de832008-06-24 22:58:06 +00003145 col = regmatch->endpos[0].col;
3146 off = spp->sp_offsets[idx] + extra;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003147 }
3148 else
3149 {
3150 result->lnum = regmatch->startpos[0].lnum;
3151 col = regmatch->startpos[0].col;
Bram Moolenaar8c8de832008-06-24 22:58:06 +00003152 off = spp->sp_offsets[idx];
Bram Moolenaar071d4272004-06-13 20:20:40 +00003153 }
Bram Moolenaar72b73c12010-02-24 17:22:20 +01003154 if (result->lnum > syn_buf->b_ml.ml_line_count)
3155 {
3156 /* a "\n" at the end of the pattern may take us below the last line */
3157 result->lnum = syn_buf->b_ml.ml_line_count;
Bram Moolenaar8b9c05f2010-03-02 17:54:33 +01003158 col = (int)STRLEN(ml_get_buf(syn_buf, result->lnum, FALSE));
Bram Moolenaar72b73c12010-02-24 17:22:20 +01003159 }
Bram Moolenaar8c8de832008-06-24 22:58:06 +00003160 if (off != 0)
3161 {
3162 base = ml_get_buf(syn_buf, result->lnum, FALSE);
3163 p = base + col;
3164 if (off > 0)
3165 {
3166 while (off-- && *p != NUL)
3167 mb_ptr_adv(p);
3168 }
3169 else if (off < 0)
3170 {
3171 while (off++ && base < p)
3172 mb_ptr_back(base, p);
3173 }
3174 col = (int)(p - base);
3175 }
3176 result->col = col;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003177}
3178
3179/*
3180 * Get current line in syntax buffer.
3181 */
3182 static char_u *
3183syn_getcurline()
3184{
3185 return ml_get_buf(syn_buf, current_lnum, FALSE);
3186}
3187
3188/*
Bram Moolenaar3b56eb32005-07-11 22:40:32 +00003189 * Call vim_regexec() to find a match with "rmp" in "syn_buf".
Bram Moolenaar071d4272004-06-13 20:20:40 +00003190 * Returns TRUE when there is a match.
3191 */
3192 static int
3193syn_regexec(rmp, lnum, col)
3194 regmmatch_T *rmp;
3195 linenr_T lnum;
3196 colnr_T col;
3197{
Bram Moolenaar3b56eb32005-07-11 22:40:32 +00003198 rmp->rmm_maxcol = syn_buf->b_p_smc;
Bram Moolenaar91a4e822008-01-19 14:59:58 +00003199 if (vim_regexec_multi(rmp, syn_win, syn_buf, lnum, col, NULL) > 0)
Bram Moolenaar071d4272004-06-13 20:20:40 +00003200 {
3201 rmp->startpos[0].lnum += lnum;
3202 rmp->endpos[0].lnum += lnum;
3203 return TRUE;
3204 }
3205 return FALSE;
3206}
3207
3208/*
3209 * Check one position in a line for a matching keyword.
3210 * The caller must check if a keyword can start at startcol.
3211 * Return it's ID if found, 0 otherwise.
3212 */
3213 static int
Bram Moolenaar860cae12010-06-05 23:22:07 +02003214check_keyword_id(line, startcol, endcolp, flagsp, next_listp, cur_si, ccharp)
Bram Moolenaar071d4272004-06-13 20:20:40 +00003215 char_u *line;
3216 int startcol; /* position in line to check for keyword */
3217 int *endcolp; /* return: character after found keyword */
3218 long *flagsp; /* return: flags of matching keyword */
3219 short **next_listp; /* return: next_list of matching keyword */
3220 stateitem_T *cur_si; /* item at the top of the stack */
Bram Moolenaar860cae12010-06-05 23:22:07 +02003221 int *ccharp UNUSED; /* conceal substitution char */
Bram Moolenaar071d4272004-06-13 20:20:40 +00003222{
Bram Moolenaardad6b692005-01-25 22:14:34 +00003223 keyentry_T *kp;
3224 char_u *kwp;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003225 int round;
Bram Moolenaardad6b692005-01-25 22:14:34 +00003226 int kwlen;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003227 char_u keyword[MAXKEYWLEN + 1]; /* assume max. keyword len is 80 */
Bram Moolenaardad6b692005-01-25 22:14:34 +00003228 hashtab_T *ht;
3229 hashitem_T *hi;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003230
3231 /* Find first character after the keyword. First character was already
3232 * checked. */
Bram Moolenaardad6b692005-01-25 22:14:34 +00003233 kwp = line + startcol;
3234 kwlen = 0;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003235 do
3236 {
3237#ifdef FEAT_MBYTE
3238 if (has_mbyte)
Bram Moolenaar0fa313a2005-08-10 21:07:57 +00003239 kwlen += (*mb_ptr2len)(kwp + kwlen);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003240 else
3241#endif
Bram Moolenaardad6b692005-01-25 22:14:34 +00003242 ++kwlen;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003243 }
Bram Moolenaardad6b692005-01-25 22:14:34 +00003244 while (vim_iswordc_buf(kwp + kwlen, syn_buf));
Bram Moolenaar071d4272004-06-13 20:20:40 +00003245
Bram Moolenaardad6b692005-01-25 22:14:34 +00003246 if (kwlen > MAXKEYWLEN)
Bram Moolenaar071d4272004-06-13 20:20:40 +00003247 return 0;
3248
3249 /*
3250 * Must make a copy of the keyword, so we can add a NUL and make it
3251 * lowercase.
3252 */
Bram Moolenaarce0842a2005-07-18 21:58:11 +00003253 vim_strncpy(keyword, kwp, kwlen);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003254
3255 /*
3256 * Try twice:
3257 * 1. matching case
3258 * 2. ignoring case
3259 */
3260 for (round = 1; round <= 2; ++round)
3261 {
Bram Moolenaar860cae12010-06-05 23:22:07 +02003262 ht = round == 1 ? &syn_block->b_keywtab : &syn_block->b_keywtab_ic;
Bram Moolenaardad6b692005-01-25 22:14:34 +00003263 if (ht->ht_used == 0)
Bram Moolenaar071d4272004-06-13 20:20:40 +00003264 continue;
Bram Moolenaardad6b692005-01-25 22:14:34 +00003265 if (round == 2) /* ignore case */
3266 (void)str_foldcase(kwp, kwlen, keyword, MAXKEYWLEN + 1);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003267
3268 /*
Bram Moolenaardad6b692005-01-25 22:14:34 +00003269 * Find keywords that match. There can be several with different
3270 * attributes.
Bram Moolenaar071d4272004-06-13 20:20:40 +00003271 * When current_next_list is non-zero accept only that group, otherwise:
3272 * Accept a not-contained keyword at toplevel.
3273 * Accept a keyword at other levels only if it is in the contains list.
3274 */
Bram Moolenaardad6b692005-01-25 22:14:34 +00003275 hi = hash_find(ht, keyword);
3276 if (!HASHITEM_EMPTY(hi))
3277 for (kp = HI2KE(hi); kp != NULL; kp = kp->ke_next)
Bram Moolenaar071d4272004-06-13 20:20:40 +00003278 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00003279 if (current_next_list != 0
3280 ? in_id_list(NULL, current_next_list, &kp->k_syn, 0)
3281 : (cur_si == NULL
3282 ? !(kp->flags & HL_CONTAINED)
3283 : in_id_list(cur_si, cur_si->si_cont_list,
3284 &kp->k_syn, kp->flags & HL_CONTAINED)))
3285 {
3286 *endcolp = startcol + kwlen;
3287 *flagsp = kp->flags;
3288 *next_listp = kp->next_list;
Bram Moolenaar860cae12010-06-05 23:22:07 +02003289#ifdef FEAT_CONCEAL
3290 *ccharp = kp->k_char;
3291#endif
Bram Moolenaardad6b692005-01-25 22:14:34 +00003292 return kp->k_syn.id;
3293 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00003294 }
3295 }
3296 return 0;
3297}
3298
3299/*
Bram Moolenaar860cae12010-06-05 23:22:07 +02003300 * Handle ":syntax conceal" command.
3301 */
3302 static void
3303syn_cmd_conceal(eap, syncing)
3304 exarg_T *eap UNUSED;
3305 int syncing UNUSED;
3306{
3307#ifdef FEAT_CONCEAL
3308 char_u *arg = eap->arg;
3309 char_u *next;
3310
3311 eap->nextcmd = find_nextcmd(arg);
3312 if (eap->skip)
3313 return;
3314
3315 next = skiptowhite(arg);
3316 if (STRNICMP(arg, "on", 2) == 0 && next - arg == 2)
3317 curwin->w_s->b_syn_conceal = TRUE;
3318 else if (STRNICMP(arg, "off", 3) == 0 && next - arg == 3)
3319 curwin->w_s->b_syn_conceal = FALSE;
3320 else
3321 EMSG2(_("E390: Illegal argument: %s"), arg);
3322#endif
3323}
3324
3325/*
Bram Moolenaar071d4272004-06-13 20:20:40 +00003326 * Handle ":syntax case" command.
3327 */
Bram Moolenaar071d4272004-06-13 20:20:40 +00003328 static void
3329syn_cmd_case(eap, syncing)
3330 exarg_T *eap;
Bram Moolenaar2c4278f2009-05-17 11:33:22 +00003331 int syncing UNUSED;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003332{
3333 char_u *arg = eap->arg;
3334 char_u *next;
3335
3336 eap->nextcmd = find_nextcmd(arg);
3337 if (eap->skip)
3338 return;
3339
3340 next = skiptowhite(arg);
3341 if (STRNICMP(arg, "match", 5) == 0 && next - arg == 5)
Bram Moolenaar860cae12010-06-05 23:22:07 +02003342 curwin->w_s->b_syn_ic = FALSE;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003343 else if (STRNICMP(arg, "ignore", 6) == 0 && next - arg == 6)
Bram Moolenaar860cae12010-06-05 23:22:07 +02003344 curwin->w_s->b_syn_ic = TRUE;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003345 else
3346 EMSG2(_("E390: Illegal argument: %s"), arg);
3347}
3348
3349/*
Bram Moolenaarce0842a2005-07-18 21:58:11 +00003350 * Handle ":syntax spell" command.
3351 */
Bram Moolenaarce0842a2005-07-18 21:58:11 +00003352 static void
3353syn_cmd_spell(eap, syncing)
3354 exarg_T *eap;
Bram Moolenaar2c4278f2009-05-17 11:33:22 +00003355 int syncing UNUSED;
Bram Moolenaarce0842a2005-07-18 21:58:11 +00003356{
3357 char_u *arg = eap->arg;
3358 char_u *next;
3359
3360 eap->nextcmd = find_nextcmd(arg);
3361 if (eap->skip)
3362 return;
3363
3364 next = skiptowhite(arg);
3365 if (STRNICMP(arg, "toplevel", 8) == 0 && next - arg == 8)
Bram Moolenaar860cae12010-06-05 23:22:07 +02003366 curwin->w_s->b_syn_spell = SYNSPL_TOP;
Bram Moolenaarce0842a2005-07-18 21:58:11 +00003367 else if (STRNICMP(arg, "notoplevel", 10) == 0 && next - arg == 10)
Bram Moolenaar860cae12010-06-05 23:22:07 +02003368 curwin->w_s->b_syn_spell = SYNSPL_NOTOP;
Bram Moolenaar86ea7642007-02-04 01:48:10 +00003369 else if (STRNICMP(arg, "default", 7) == 0 && next - arg == 7)
Bram Moolenaar860cae12010-06-05 23:22:07 +02003370 curwin->w_s->b_syn_spell = SYNSPL_DEFAULT;
Bram Moolenaarce0842a2005-07-18 21:58:11 +00003371 else
3372 EMSG2(_("E390: Illegal argument: %s"), arg);
3373}
3374
3375/*
Bram Moolenaar071d4272004-06-13 20:20:40 +00003376 * Clear all syntax info for one buffer.
3377 */
3378 void
Bram Moolenaar860cae12010-06-05 23:22:07 +02003379syntax_clear(block)
3380 synblock_T *block;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003381{
3382 int i;
3383
Bram Moolenaar860cae12010-06-05 23:22:07 +02003384 block->b_syn_error = FALSE; /* clear previous error */
3385 block->b_syn_ic = FALSE; /* Use case, by default */
3386 block->b_syn_spell = SYNSPL_DEFAULT; /* default spell checking */
3387 block->b_syn_containedin = FALSE;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003388
3389 /* free the keywords */
Bram Moolenaar860cae12010-06-05 23:22:07 +02003390 clear_keywtab(&block->b_keywtab);
3391 clear_keywtab(&block->b_keywtab_ic);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003392
3393 /* free the syntax patterns */
Bram Moolenaar860cae12010-06-05 23:22:07 +02003394 for (i = block->b_syn_patterns.ga_len; --i >= 0; )
3395 syn_clear_pattern(block, i);
3396 ga_clear(&block->b_syn_patterns);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003397
3398 /* free the syntax clusters */
Bram Moolenaar860cae12010-06-05 23:22:07 +02003399 for (i = block->b_syn_clusters.ga_len; --i >= 0; )
3400 syn_clear_cluster(block, i);
3401 ga_clear(&block->b_syn_clusters);
3402 block->b_spell_cluster_id = 0;
3403 block->b_nospell_cluster_id = 0;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003404
Bram Moolenaar860cae12010-06-05 23:22:07 +02003405 block->b_syn_sync_flags = 0;
3406 block->b_syn_sync_minlines = 0;
3407 block->b_syn_sync_maxlines = 0;
3408 block->b_syn_sync_linebreaks = 0;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003409
Bram Moolenaar860cae12010-06-05 23:22:07 +02003410 vim_free(block->b_syn_linecont_prog);
3411 block->b_syn_linecont_prog = NULL;
3412 vim_free(block->b_syn_linecont_pat);
3413 block->b_syn_linecont_pat = NULL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003414#ifdef FEAT_FOLDING
Bram Moolenaar860cae12010-06-05 23:22:07 +02003415 block->b_syn_folditems = 0;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003416#endif
3417
3418 /* free the stored states */
Bram Moolenaar860cae12010-06-05 23:22:07 +02003419 syn_stack_free_all(block);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003420 invalidate_current_state();
3421}
3422
3423/*
3424 * Clear syncing info for one buffer.
3425 */
3426 static void
3427syntax_sync_clear()
3428{
3429 int i;
3430
3431 /* free the syntax patterns */
Bram Moolenaar860cae12010-06-05 23:22:07 +02003432 for (i = curwin->w_s->b_syn_patterns.ga_len; --i >= 0; )
3433 if (SYN_ITEMS(curwin->w_s)[i].sp_syncing)
3434 syn_remove_pattern(curwin->w_s, i);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003435
Bram Moolenaar860cae12010-06-05 23:22:07 +02003436 curwin->w_s->b_syn_sync_flags = 0;
3437 curwin->w_s->b_syn_sync_minlines = 0;
3438 curwin->w_s->b_syn_sync_maxlines = 0;
3439 curwin->w_s->b_syn_sync_linebreaks = 0;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003440
Bram Moolenaar860cae12010-06-05 23:22:07 +02003441 vim_free(curwin->w_s->b_syn_linecont_prog);
3442 curwin->w_s->b_syn_linecont_prog = NULL;
3443 vim_free(curwin->w_s->b_syn_linecont_pat);
3444 curwin->w_s->b_syn_linecont_pat = NULL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003445
Bram Moolenaar860cae12010-06-05 23:22:07 +02003446 syn_stack_free_all(curwin->w_s); /* Need to recompute all syntax. */
Bram Moolenaar071d4272004-06-13 20:20:40 +00003447}
3448
3449/*
3450 * Remove one pattern from the buffer's pattern list.
3451 */
3452 static void
Bram Moolenaar860cae12010-06-05 23:22:07 +02003453syn_remove_pattern(block, idx)
3454 synblock_T *block;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003455 int idx;
3456{
3457 synpat_T *spp;
3458
Bram Moolenaar860cae12010-06-05 23:22:07 +02003459 spp = &(SYN_ITEMS(block)[idx]);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003460#ifdef FEAT_FOLDING
3461 if (spp->sp_flags & HL_FOLD)
Bram Moolenaar860cae12010-06-05 23:22:07 +02003462 --block->b_syn_folditems;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003463#endif
Bram Moolenaar860cae12010-06-05 23:22:07 +02003464 syn_clear_pattern(block, idx);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003465 mch_memmove(spp, spp + 1,
Bram Moolenaar860cae12010-06-05 23:22:07 +02003466 sizeof(synpat_T) * (block->b_syn_patterns.ga_len - idx - 1));
3467 --block->b_syn_patterns.ga_len;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003468}
3469
3470/*
3471 * Clear and free one syntax pattern. When clearing all, must be called from
3472 * last to first!
3473 */
3474 static void
Bram Moolenaar860cae12010-06-05 23:22:07 +02003475syn_clear_pattern(block, i)
3476 synblock_T *block;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003477 int i;
3478{
Bram Moolenaar860cae12010-06-05 23:22:07 +02003479 vim_free(SYN_ITEMS(block)[i].sp_pattern);
3480 vim_free(SYN_ITEMS(block)[i].sp_prog);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003481 /* Only free sp_cont_list and sp_next_list of first start pattern */
Bram Moolenaar860cae12010-06-05 23:22:07 +02003482 if (i == 0 || SYN_ITEMS(block)[i - 1].sp_type != SPTYPE_START)
Bram Moolenaar071d4272004-06-13 20:20:40 +00003483 {
Bram Moolenaar860cae12010-06-05 23:22:07 +02003484 vim_free(SYN_ITEMS(block)[i].sp_cont_list);
3485 vim_free(SYN_ITEMS(block)[i].sp_next_list);
3486 vim_free(SYN_ITEMS(block)[i].sp_syn.cont_in_list);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003487 }
3488}
3489
3490/*
3491 * Clear and free one syntax cluster.
3492 */
3493 static void
Bram Moolenaar860cae12010-06-05 23:22:07 +02003494syn_clear_cluster(block, i)
3495 synblock_T *block;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003496 int i;
3497{
Bram Moolenaar860cae12010-06-05 23:22:07 +02003498 vim_free(SYN_CLSTR(block)[i].scl_name);
3499 vim_free(SYN_CLSTR(block)[i].scl_name_u);
3500 vim_free(SYN_CLSTR(block)[i].scl_list);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003501}
3502
3503/*
3504 * Handle ":syntax clear" command.
3505 */
3506 static void
3507syn_cmd_clear(eap, syncing)
3508 exarg_T *eap;
3509 int syncing;
3510{
3511 char_u *arg = eap->arg;
3512 char_u *arg_end;
3513 int id;
3514
3515 eap->nextcmd = find_nextcmd(arg);
3516 if (eap->skip)
3517 return;
3518
3519 /*
3520 * We have to disable this within ":syn include @group filename",
3521 * because otherwise @group would get deleted.
3522 * Only required for Vim 5.x syntax files, 6.0 ones don't contain ":syn
3523 * clear".
3524 */
Bram Moolenaar860cae12010-06-05 23:22:07 +02003525 if (curwin->w_s->b_syn_topgrp != 0)
Bram Moolenaar071d4272004-06-13 20:20:40 +00003526 return;
3527
3528 if (ends_excmd(*arg))
3529 {
3530 /*
3531 * No argument: Clear all syntax items.
3532 */
3533 if (syncing)
3534 syntax_sync_clear();
3535 else
3536 {
Bram Moolenaar860cae12010-06-05 23:22:07 +02003537 syntax_clear(curwin->w_s);
3538 if (curwin->w_s == &curwin->w_buffer->b_s)
3539 do_unlet((char_u *)"b:current_syntax", TRUE);
3540 else
3541 do_unlet((char_u *)"w:current_syntax", TRUE);
3542
Bram Moolenaar071d4272004-06-13 20:20:40 +00003543 }
3544 }
3545 else
3546 {
3547 /*
3548 * Clear the group IDs that are in the argument.
3549 */
3550 while (!ends_excmd(*arg))
3551 {
3552 arg_end = skiptowhite(arg);
3553 if (*arg == '@')
3554 {
3555 id = syn_scl_namen2id(arg + 1, (int)(arg_end - arg - 1));
3556 if (id == 0)
3557 {
3558 EMSG2(_("E391: No such syntax cluster: %s"), arg);
3559 break;
3560 }
3561 else
3562 {
3563 /*
3564 * We can't physically delete a cluster without changing
3565 * the IDs of other clusters, so we do the next best thing
3566 * and make it empty.
3567 */
3568 short scl_id = id - SYNID_CLUSTER;
3569
Bram Moolenaar860cae12010-06-05 23:22:07 +02003570 vim_free(SYN_CLSTR(curwin->w_s)[scl_id].scl_list);
3571 SYN_CLSTR(curwin->w_s)[scl_id].scl_list = NULL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003572 }
3573 }
3574 else
3575 {
3576 id = syn_namen2id(arg, (int)(arg_end - arg));
3577 if (id == 0)
3578 {
3579 EMSG2(_(e_nogroup), arg);
3580 break;
3581 }
3582 else
3583 syn_clear_one(id, syncing);
3584 }
3585 arg = skipwhite(arg_end);
3586 }
3587 }
Bram Moolenaar1c8f93f2006-03-12 22:10:07 +00003588 redraw_curbuf_later(SOME_VALID);
Bram Moolenaar860cae12010-06-05 23:22:07 +02003589 syn_stack_free_all(curwin->w_s); /* Need to recompute all syntax. */
Bram Moolenaar071d4272004-06-13 20:20:40 +00003590}
3591
3592/*
3593 * Clear one syntax group for the current buffer.
3594 */
3595 static void
3596syn_clear_one(id, syncing)
3597 int id;
3598 int syncing;
3599{
3600 synpat_T *spp;
3601 int idx;
3602
3603 /* Clear keywords only when not ":syn sync clear group-name" */
3604 if (!syncing)
3605 {
Bram Moolenaar860cae12010-06-05 23:22:07 +02003606 (void)syn_clear_keyword(id, &curwin->w_s->b_keywtab);
3607 (void)syn_clear_keyword(id, &curwin->w_s->b_keywtab_ic);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003608 }
3609
3610 /* clear the patterns for "id" */
Bram Moolenaar860cae12010-06-05 23:22:07 +02003611 for (idx = curwin->w_s->b_syn_patterns.ga_len; --idx >= 0; )
Bram Moolenaar071d4272004-06-13 20:20:40 +00003612 {
Bram Moolenaar860cae12010-06-05 23:22:07 +02003613 spp = &(SYN_ITEMS(curwin->w_s)[idx]);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003614 if (spp->sp_syn.id != id || spp->sp_syncing != syncing)
3615 continue;
Bram Moolenaar860cae12010-06-05 23:22:07 +02003616 syn_remove_pattern(curwin->w_s, idx);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003617 }
3618}
3619
3620/*
3621 * Handle ":syntax on" command.
3622 */
Bram Moolenaar071d4272004-06-13 20:20:40 +00003623 static void
3624syn_cmd_on(eap, syncing)
3625 exarg_T *eap;
Bram Moolenaar2c4278f2009-05-17 11:33:22 +00003626 int syncing UNUSED;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003627{
3628 syn_cmd_onoff(eap, "syntax");
3629}
3630
3631/*
3632 * Handle ":syntax enable" command.
3633 */
Bram Moolenaar071d4272004-06-13 20:20:40 +00003634 static void
3635syn_cmd_enable(eap, syncing)
3636 exarg_T *eap;
Bram Moolenaar2c4278f2009-05-17 11:33:22 +00003637 int syncing UNUSED;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003638{
3639 set_internal_string_var((char_u *)"syntax_cmd", (char_u *)"enable");
3640 syn_cmd_onoff(eap, "syntax");
Bram Moolenaar2ce06f62005-01-31 19:19:04 +00003641 do_unlet((char_u *)"g:syntax_cmd", TRUE);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003642}
3643
3644/*
3645 * Handle ":syntax reset" command.
3646 */
Bram Moolenaar071d4272004-06-13 20:20:40 +00003647 static void
3648syn_cmd_reset(eap, syncing)
3649 exarg_T *eap;
Bram Moolenaar2c4278f2009-05-17 11:33:22 +00003650 int syncing UNUSED;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003651{
3652 eap->nextcmd = check_nextcmd(eap->arg);
3653 if (!eap->skip)
3654 {
3655 set_internal_string_var((char_u *)"syntax_cmd", (char_u *)"reset");
3656 do_cmdline_cmd((char_u *)"runtime! syntax/syncolor.vim");
Bram Moolenaar2ce06f62005-01-31 19:19:04 +00003657 do_unlet((char_u *)"g:syntax_cmd", TRUE);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003658 }
3659}
3660
3661/*
3662 * Handle ":syntax manual" command.
3663 */
Bram Moolenaar071d4272004-06-13 20:20:40 +00003664 static void
3665syn_cmd_manual(eap, syncing)
3666 exarg_T *eap;
Bram Moolenaar2c4278f2009-05-17 11:33:22 +00003667 int syncing UNUSED;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003668{
3669 syn_cmd_onoff(eap, "manual");
3670}
3671
3672/*
3673 * Handle ":syntax off" command.
3674 */
Bram Moolenaar071d4272004-06-13 20:20:40 +00003675 static void
3676syn_cmd_off(eap, syncing)
3677 exarg_T *eap;
Bram Moolenaar2c4278f2009-05-17 11:33:22 +00003678 int syncing UNUSED;
Bram Moolenaar071d4272004-06-13 20:20:40 +00003679{
3680 syn_cmd_onoff(eap, "nosyntax");
3681}
3682
3683 static void
3684syn_cmd_onoff(eap, name)
3685 exarg_T *eap;
3686 char *name;
3687{
3688 char_u buf[100];
3689
3690 eap->nextcmd = check_nextcmd(eap->arg);
3691 if (!eap->skip)
3692 {
3693 STRCPY(buf, "so ");
Bram Moolenaar555b2802005-05-19 21:08:39 +00003694 vim_snprintf((char *)buf + 3, sizeof(buf) - 3, SYNTAX_FNAME, name);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003695 do_cmdline_cmd(buf);
3696 }
3697}
3698
3699/*
3700 * Handle ":syntax [list]" command: list current syntax words.
3701 */
3702 static void
3703syn_cmd_list(eap, syncing)
3704 exarg_T *eap;
3705 int syncing; /* when TRUE: list syncing items */
3706{
3707 char_u *arg = eap->arg;
3708 int id;
3709 char_u *arg_end;
3710
3711 eap->nextcmd = find_nextcmd(arg);
3712 if (eap->skip)
3713 return;
3714
Bram Moolenaar860cae12010-06-05 23:22:07 +02003715 if (!syntax_present(curwin))
Bram Moolenaar071d4272004-06-13 20:20:40 +00003716 {
3717 MSG(_("No Syntax items defined for this buffer"));
3718 return;
3719 }
3720
3721 if (syncing)
3722 {
Bram Moolenaar860cae12010-06-05 23:22:07 +02003723 if (curwin->w_s->b_syn_sync_flags & SF_CCOMMENT)
Bram Moolenaar071d4272004-06-13 20:20:40 +00003724 {
3725 MSG_PUTS(_("syncing on C-style comments"));
3726 syn_lines_msg();
3727 syn_match_msg();
3728 return;
3729 }
Bram Moolenaar860cae12010-06-05 23:22:07 +02003730 else if (!(curwin->w_s->b_syn_sync_flags & SF_MATCH))
Bram Moolenaar071d4272004-06-13 20:20:40 +00003731 {
Bram Moolenaar860cae12010-06-05 23:22:07 +02003732 if (curwin->w_s->b_syn_sync_minlines == 0)
Bram Moolenaar071d4272004-06-13 20:20:40 +00003733 MSG_PUTS(_("no syncing"));
3734 else
3735 {
3736 MSG_PUTS(_("syncing starts "));
Bram Moolenaar860cae12010-06-05 23:22:07 +02003737 msg_outnum(curwin->w_s->b_syn_sync_minlines);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003738 MSG_PUTS(_(" lines before top line"));
3739 syn_match_msg();
3740 }
3741 return;
3742 }
3743 MSG_PUTS_TITLE(_("\n--- Syntax sync items ---"));
Bram Moolenaar860cae12010-06-05 23:22:07 +02003744 if (curwin->w_s->b_syn_sync_minlines > 0
3745 || curwin->w_s->b_syn_sync_maxlines > 0
3746 || curwin->w_s->b_syn_sync_linebreaks > 0)
Bram Moolenaar071d4272004-06-13 20:20:40 +00003747 {
3748 MSG_PUTS(_("\nsyncing on items"));
3749 syn_lines_msg();
3750 syn_match_msg();
3751 }
3752 }
3753 else
3754 MSG_PUTS_TITLE(_("\n--- Syntax items ---"));
3755 if (ends_excmd(*arg))
3756 {
3757 /*
3758 * No argument: List all group IDs and all syntax clusters.
3759 */
3760 for (id = 1; id <= highlight_ga.ga_len && !got_int; ++id)
3761 syn_list_one(id, syncing, FALSE);
Bram Moolenaar860cae12010-06-05 23:22:07 +02003762 for (id = 0; id < curwin->w_s->b_syn_clusters.ga_len && !got_int; ++id)
Bram Moolenaar071d4272004-06-13 20:20:40 +00003763 syn_list_cluster(id);
3764 }
3765 else
3766 {
3767 /*
3768 * List the group IDs and syntax clusters that are in the argument.
3769 */
3770 while (!ends_excmd(*arg) && !got_int)
3771 {
3772 arg_end = skiptowhite(arg);
3773 if (*arg == '@')
3774 {
3775 id = syn_scl_namen2id(arg + 1, (int)(arg_end - arg - 1));
3776 if (id == 0)
3777 EMSG2(_("E392: No such syntax cluster: %s"), arg);
3778 else
3779 syn_list_cluster(id - SYNID_CLUSTER);
3780 }
3781 else
3782 {
3783 id = syn_namen2id(arg, (int)(arg_end - arg));
3784 if (id == 0)
3785 EMSG2(_(e_nogroup), arg);
3786 else
3787 syn_list_one(id, syncing, TRUE);
3788 }
3789 arg = skipwhite(arg_end);
3790 }
3791 }
3792 eap->nextcmd = check_nextcmd(arg);
3793}
3794
3795 static void
3796syn_lines_msg()
3797{
Bram Moolenaar860cae12010-06-05 23:22:07 +02003798 if (curwin->w_s->b_syn_sync_maxlines > 0
3799 || curwin->w_s->b_syn_sync_minlines > 0)
Bram Moolenaar071d4272004-06-13 20:20:40 +00003800 {
3801 MSG_PUTS("; ");
Bram Moolenaar860cae12010-06-05 23:22:07 +02003802 if (curwin->w_s->b_syn_sync_minlines > 0)
Bram Moolenaar071d4272004-06-13 20:20:40 +00003803 {
3804 MSG_PUTS(_("minimal "));
Bram Moolenaar860cae12010-06-05 23:22:07 +02003805 msg_outnum(curwin->w_s->b_syn_sync_minlines);
3806 if (curwin->w_s->b_syn_sync_maxlines)
Bram Moolenaar071d4272004-06-13 20:20:40 +00003807 MSG_PUTS(", ");
3808 }
Bram Moolenaar860cae12010-06-05 23:22:07 +02003809 if (curwin->w_s->b_syn_sync_maxlines > 0)
Bram Moolenaar071d4272004-06-13 20:20:40 +00003810 {
3811 MSG_PUTS(_("maximal "));
Bram Moolenaar860cae12010-06-05 23:22:07 +02003812 msg_outnum(curwin->w_s->b_syn_sync_maxlines);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003813 }
3814 MSG_PUTS(_(" lines before top line"));
3815 }
3816}
3817
3818 static void
3819syn_match_msg()
3820{
Bram Moolenaar860cae12010-06-05 23:22:07 +02003821 if (curwin->w_s->b_syn_sync_linebreaks > 0)
Bram Moolenaar071d4272004-06-13 20:20:40 +00003822 {
3823 MSG_PUTS(_("; match "));
Bram Moolenaar860cae12010-06-05 23:22:07 +02003824 msg_outnum(curwin->w_s->b_syn_sync_linebreaks);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003825 MSG_PUTS(_(" line breaks"));
3826 }
3827}
3828
3829static int last_matchgroup;
3830
3831struct name_list
3832{
3833 int flag;
3834 char *name;
3835};
3836
3837static void syn_list_flags __ARGS((struct name_list *nl, int flags, int attr));
3838
3839/*
3840 * List one syntax item, for ":syntax" or "syntax list syntax_name".
3841 */
3842 static void
3843syn_list_one(id, syncing, link_only)
3844 int id;
3845 int syncing; /* when TRUE: list syncing items */
3846 int link_only; /* when TRUE; list link-only too */
3847{
3848 int attr;
3849 int idx;
3850 int did_header = FALSE;
3851 synpat_T *spp;
3852 static struct name_list namelist1[] =
3853 {
3854 {HL_DISPLAY, "display"},
3855 {HL_CONTAINED, "contained"},
3856 {HL_ONELINE, "oneline"},
3857 {HL_KEEPEND, "keepend"},
3858 {HL_EXTEND, "extend"},
3859 {HL_EXCLUDENL, "excludenl"},
3860 {HL_TRANSP, "transparent"},
3861 {HL_FOLD, "fold"},
Bram Moolenaar860cae12010-06-05 23:22:07 +02003862#ifdef FEAT_CONCEAL
3863 {HL_CONCEAL, "conceal"},
3864 {HL_CONCEALENDS, "concealends"},
3865#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00003866 {0, NULL}
3867 };
3868 static struct name_list namelist2[] =
3869 {
3870 {HL_SKIPWHITE, "skipwhite"},
3871 {HL_SKIPNL, "skipnl"},
3872 {HL_SKIPEMPTY, "skipempty"},
3873 {0, NULL}
3874 };
3875
3876 attr = hl_attr(HLF_D); /* highlight like directories */
3877
3878 /* list the keywords for "id" */
3879 if (!syncing)
3880 {
Bram Moolenaar860cae12010-06-05 23:22:07 +02003881 did_header = syn_list_keywords(id, &curwin->w_s->b_keywtab, FALSE, attr);
3882 did_header = syn_list_keywords(id, &curwin->w_s->b_keywtab_ic,
Bram Moolenaar071d4272004-06-13 20:20:40 +00003883 did_header, attr);
3884 }
3885
3886 /* list the patterns for "id" */
Bram Moolenaar860cae12010-06-05 23:22:07 +02003887 for (idx = 0; idx < curwin->w_s->b_syn_patterns.ga_len && !got_int; ++idx)
Bram Moolenaar071d4272004-06-13 20:20:40 +00003888 {
Bram Moolenaar860cae12010-06-05 23:22:07 +02003889 spp = &(SYN_ITEMS(curwin->w_s)[idx]);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003890 if (spp->sp_syn.id != id || spp->sp_syncing != syncing)
3891 continue;
3892
3893 (void)syn_list_header(did_header, 999, id);
3894 did_header = TRUE;
3895 last_matchgroup = 0;
3896 if (spp->sp_type == SPTYPE_MATCH)
3897 {
3898 put_pattern("match", ' ', spp, attr);
3899 msg_putchar(' ');
3900 }
3901 else if (spp->sp_type == SPTYPE_START)
3902 {
Bram Moolenaar860cae12010-06-05 23:22:07 +02003903 while (SYN_ITEMS(curwin->w_s)[idx].sp_type == SPTYPE_START)
3904 put_pattern("start", '=', &SYN_ITEMS(curwin->w_s)[idx++], attr);
3905 if (SYN_ITEMS(curwin->w_s)[idx].sp_type == SPTYPE_SKIP)
3906 put_pattern("skip", '=', &SYN_ITEMS(curwin->w_s)[idx++], attr);
3907 while (idx < curwin->w_s->b_syn_patterns.ga_len
3908 && SYN_ITEMS(curwin->w_s)[idx].sp_type == SPTYPE_END)
3909 put_pattern("end", '=', &SYN_ITEMS(curwin->w_s)[idx++], attr);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003910 --idx;
3911 msg_putchar(' ');
3912 }
3913 syn_list_flags(namelist1, spp->sp_flags, attr);
3914
3915 if (spp->sp_cont_list != NULL)
3916 put_id_list((char_u *)"contains", spp->sp_cont_list, attr);
3917
3918 if (spp->sp_syn.cont_in_list != NULL)
3919 put_id_list((char_u *)"containedin",
3920 spp->sp_syn.cont_in_list, attr);
3921
3922 if (spp->sp_next_list != NULL)
3923 {
3924 put_id_list((char_u *)"nextgroup", spp->sp_next_list, attr);
3925 syn_list_flags(namelist2, spp->sp_flags, attr);
3926 }
3927 if (spp->sp_flags & (HL_SYNC_HERE|HL_SYNC_THERE))
3928 {
3929 if (spp->sp_flags & HL_SYNC_HERE)
3930 msg_puts_attr((char_u *)"grouphere", attr);
3931 else
3932 msg_puts_attr((char_u *)"groupthere", attr);
3933 msg_putchar(' ');
3934 if (spp->sp_sync_idx >= 0)
Bram Moolenaar860cae12010-06-05 23:22:07 +02003935 msg_outtrans(HL_TABLE()[SYN_ITEMS(curwin->w_s)
Bram Moolenaar071d4272004-06-13 20:20:40 +00003936 [spp->sp_sync_idx].sp_syn.id - 1].sg_name);
3937 else
3938 MSG_PUTS("NONE");
3939 msg_putchar(' ');
3940 }
3941 }
3942
3943 /* list the link, if there is one */
3944 if (HL_TABLE()[id - 1].sg_link && (did_header || link_only) && !got_int)
3945 {
3946 (void)syn_list_header(did_header, 999, id);
3947 msg_puts_attr((char_u *)"links to", attr);
3948 msg_putchar(' ');
3949 msg_outtrans(HL_TABLE()[HL_TABLE()[id - 1].sg_link - 1].sg_name);
3950 }
3951}
3952
3953 static void
3954syn_list_flags(nl, flags, attr)
3955 struct name_list *nl;
3956 int flags;
3957 int attr;
3958{
3959 int i;
3960
3961 for (i = 0; nl[i].flag != 0; ++i)
3962 if (flags & nl[i].flag)
3963 {
3964 msg_puts_attr((char_u *)nl[i].name, attr);
3965 msg_putchar(' ');
3966 }
3967}
3968
3969/*
3970 * List one syntax cluster, for ":syntax" or "syntax list syntax_name".
3971 */
3972 static void
3973syn_list_cluster(id)
3974 int id;
3975{
3976 int endcol = 15;
3977
3978 /* slight hack: roughly duplicate the guts of syn_list_header() */
3979 msg_putchar('\n');
Bram Moolenaar860cae12010-06-05 23:22:07 +02003980 msg_outtrans(SYN_CLSTR(curwin->w_s)[id].scl_name);
Bram Moolenaar071d4272004-06-13 20:20:40 +00003981
3982 if (msg_col >= endcol) /* output at least one space */
3983 endcol = msg_col + 1;
3984 if (Columns <= endcol) /* avoid hang for tiny window */
3985 endcol = Columns - 1;
3986
3987 msg_advance(endcol);
Bram Moolenaar860cae12010-06-05 23:22:07 +02003988 if (SYN_CLSTR(curwin->w_s)[id].scl_list != NULL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00003989 {
Bram Moolenaar860cae12010-06-05 23:22:07 +02003990 put_id_list((char_u *)"cluster", SYN_CLSTR(curwin->w_s)[id].scl_list,
Bram Moolenaar071d4272004-06-13 20:20:40 +00003991 hl_attr(HLF_D));
3992 }
3993 else
3994 {
3995 msg_puts_attr((char_u *)"cluster", hl_attr(HLF_D));
3996 msg_puts((char_u *)"=NONE");
3997 }
3998}
3999
4000 static void
4001put_id_list(name, list, attr)
4002 char_u *name;
4003 short *list;
4004 int attr;
4005{
4006 short *p;
4007
4008 msg_puts_attr(name, attr);
4009 msg_putchar('=');
4010 for (p = list; *p; ++p)
4011 {
4012 if (*p >= SYNID_ALLBUT && *p < SYNID_TOP)
4013 {
4014 if (p[1])
4015 MSG_PUTS("ALLBUT");
4016 else
4017 MSG_PUTS("ALL");
4018 }
4019 else if (*p >= SYNID_TOP && *p < SYNID_CONTAINED)
4020 {
4021 MSG_PUTS("TOP");
4022 }
4023 else if (*p >= SYNID_CONTAINED && *p < SYNID_CLUSTER)
4024 {
4025 MSG_PUTS("CONTAINED");
4026 }
4027 else if (*p >= SYNID_CLUSTER)
4028 {
4029 short scl_id = *p - SYNID_CLUSTER;
4030
4031 msg_putchar('@');
Bram Moolenaar860cae12010-06-05 23:22:07 +02004032 msg_outtrans(SYN_CLSTR(curwin->w_s)[scl_id].scl_name);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004033 }
4034 else
4035 msg_outtrans(HL_TABLE()[*p - 1].sg_name);
4036 if (p[1])
4037 msg_putchar(',');
4038 }
4039 msg_putchar(' ');
4040}
4041
4042 static void
4043put_pattern(s, c, spp, attr)
4044 char *s;
4045 int c;
4046 synpat_T *spp;
4047 int attr;
4048{
4049 long n;
4050 int mask;
4051 int first;
4052 static char *sepchars = "/+=-#@\"|'^&";
4053 int i;
4054
4055 /* May have to write "matchgroup=group" */
4056 if (last_matchgroup != spp->sp_syn_match_id)
4057 {
4058 last_matchgroup = spp->sp_syn_match_id;
4059 msg_puts_attr((char_u *)"matchgroup", attr);
4060 msg_putchar('=');
4061 if (last_matchgroup == 0)
4062 msg_outtrans((char_u *)"NONE");
4063 else
4064 msg_outtrans(HL_TABLE()[last_matchgroup - 1].sg_name);
4065 msg_putchar(' ');
4066 }
4067
4068 /* output the name of the pattern and an '=' or ' ' */
4069 msg_puts_attr((char_u *)s, attr);
4070 msg_putchar(c);
4071
4072 /* output the pattern, in between a char that is not in the pattern */
4073 for (i = 0; vim_strchr(spp->sp_pattern, sepchars[i]) != NULL; )
4074 if (sepchars[++i] == NUL)
4075 {
4076 i = 0; /* no good char found, just use the first one */
4077 break;
4078 }
4079 msg_putchar(sepchars[i]);
4080 msg_outtrans(spp->sp_pattern);
4081 msg_putchar(sepchars[i]);
4082
4083 /* output any pattern options */
4084 first = TRUE;
4085 for (i = 0; i < SPO_COUNT; ++i)
4086 {
4087 mask = (1 << i);
4088 if (spp->sp_off_flags & (mask + (mask << SPO_COUNT)))
4089 {
4090 if (!first)
4091 msg_putchar(','); /* separate with commas */
4092 msg_puts((char_u *)spo_name_tab[i]);
4093 n = spp->sp_offsets[i];
4094 if (i != SPO_LC_OFF)
4095 {
4096 if (spp->sp_off_flags & mask)
4097 msg_putchar('s');
4098 else
4099 msg_putchar('e');
4100 if (n > 0)
4101 msg_putchar('+');
4102 }
4103 if (n || i == SPO_LC_OFF)
4104 msg_outnum(n);
4105 first = FALSE;
4106 }
4107 }
4108 msg_putchar(' ');
4109}
4110
4111/*
4112 * List or clear the keywords for one syntax group.
4113 * Return TRUE if the header has been printed.
4114 */
4115 static int
Bram Moolenaardad6b692005-01-25 22:14:34 +00004116syn_list_keywords(id, ht, did_header, attr)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004117 int id;
Bram Moolenaardad6b692005-01-25 22:14:34 +00004118 hashtab_T *ht;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004119 int did_header; /* header has already been printed */
4120 int attr;
4121{
Bram Moolenaar071d4272004-06-13 20:20:40 +00004122 int outlen;
Bram Moolenaardad6b692005-01-25 22:14:34 +00004123 hashitem_T *hi;
4124 keyentry_T *kp;
4125 int todo;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004126 int prev_contained = 0;
4127 short *prev_next_list = NULL;
4128 short *prev_cont_in_list = NULL;
4129 int prev_skipnl = 0;
4130 int prev_skipwhite = 0;
4131 int prev_skipempty = 0;
4132
Bram Moolenaar071d4272004-06-13 20:20:40 +00004133 /*
4134 * Unfortunately, this list of keywords is not sorted on alphabet but on
4135 * hash value...
4136 */
Bram Moolenaara93fa7e2006-04-17 22:14:47 +00004137 todo = (int)ht->ht_used;
Bram Moolenaardad6b692005-01-25 22:14:34 +00004138 for (hi = ht->ht_array; todo > 0 && !got_int; ++hi)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004139 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00004140 if (!HASHITEM_EMPTY(hi))
Bram Moolenaar071d4272004-06-13 20:20:40 +00004141 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00004142 --todo;
4143 for (kp = HI2KE(hi); kp != NULL && !got_int; kp = kp->ke_next)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004144 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00004145 if (kp->k_syn.id == id)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004146 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00004147 if (prev_contained != (kp->flags & HL_CONTAINED)
4148 || prev_skipnl != (kp->flags & HL_SKIPNL)
4149 || prev_skipwhite != (kp->flags & HL_SKIPWHITE)
4150 || prev_skipempty != (kp->flags & HL_SKIPEMPTY)
4151 || prev_cont_in_list != kp->k_syn.cont_in_list
4152 || prev_next_list != kp->next_list)
4153 outlen = 9999;
4154 else
4155 outlen = (int)STRLEN(kp->keyword);
4156 /* output "contained" and "nextgroup" on each line */
4157 if (syn_list_header(did_header, outlen, id))
Bram Moolenaar071d4272004-06-13 20:20:40 +00004158 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00004159 prev_contained = 0;
4160 prev_next_list = NULL;
4161 prev_cont_in_list = NULL;
4162 prev_skipnl = 0;
4163 prev_skipwhite = 0;
4164 prev_skipempty = 0;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004165 }
Bram Moolenaardad6b692005-01-25 22:14:34 +00004166 did_header = TRUE;
4167 if (prev_contained != (kp->flags & HL_CONTAINED))
Bram Moolenaar071d4272004-06-13 20:20:40 +00004168 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00004169 msg_puts_attr((char_u *)"contained", attr);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004170 msg_putchar(' ');
Bram Moolenaardad6b692005-01-25 22:14:34 +00004171 prev_contained = (kp->flags & HL_CONTAINED);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004172 }
Bram Moolenaardad6b692005-01-25 22:14:34 +00004173 if (kp->k_syn.cont_in_list != prev_cont_in_list)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004174 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00004175 put_id_list((char_u *)"containedin",
4176 kp->k_syn.cont_in_list, attr);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004177 msg_putchar(' ');
Bram Moolenaardad6b692005-01-25 22:14:34 +00004178 prev_cont_in_list = kp->k_syn.cont_in_list;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004179 }
Bram Moolenaardad6b692005-01-25 22:14:34 +00004180 if (kp->next_list != prev_next_list)
4181 {
4182 put_id_list((char_u *)"nextgroup", kp->next_list, attr);
4183 msg_putchar(' ');
4184 prev_next_list = kp->next_list;
4185 if (kp->flags & HL_SKIPNL)
4186 {
4187 msg_puts_attr((char_u *)"skipnl", attr);
4188 msg_putchar(' ');
4189 prev_skipnl = (kp->flags & HL_SKIPNL);
4190 }
4191 if (kp->flags & HL_SKIPWHITE)
4192 {
4193 msg_puts_attr((char_u *)"skipwhite", attr);
4194 msg_putchar(' ');
4195 prev_skipwhite = (kp->flags & HL_SKIPWHITE);
4196 }
4197 if (kp->flags & HL_SKIPEMPTY)
4198 {
4199 msg_puts_attr((char_u *)"skipempty", attr);
4200 msg_putchar(' ');
4201 prev_skipempty = (kp->flags & HL_SKIPEMPTY);
4202 }
4203 }
4204 msg_outtrans(kp->keyword);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004205 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00004206 }
4207 }
4208 }
4209
4210 return did_header;
4211}
4212
4213 static void
Bram Moolenaardad6b692005-01-25 22:14:34 +00004214syn_clear_keyword(id, ht)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004215 int id;
Bram Moolenaardad6b692005-01-25 22:14:34 +00004216 hashtab_T *ht;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004217{
Bram Moolenaardad6b692005-01-25 22:14:34 +00004218 hashitem_T *hi;
4219 keyentry_T *kp;
4220 keyentry_T *kp_prev;
4221 keyentry_T *kp_next;
4222 int todo;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004223
Bram Moolenaardad6b692005-01-25 22:14:34 +00004224 hash_lock(ht);
Bram Moolenaara93fa7e2006-04-17 22:14:47 +00004225 todo = (int)ht->ht_used;
Bram Moolenaardad6b692005-01-25 22:14:34 +00004226 for (hi = ht->ht_array; todo > 0; ++hi)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004227 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00004228 if (!HASHITEM_EMPTY(hi))
Bram Moolenaar071d4272004-06-13 20:20:40 +00004229 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00004230 --todo;
4231 kp_prev = NULL;
4232 for (kp = HI2KE(hi); kp != NULL; )
Bram Moolenaar071d4272004-06-13 20:20:40 +00004233 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00004234 if (kp->k_syn.id == id)
4235 {
4236 kp_next = kp->ke_next;
4237 if (kp_prev == NULL)
4238 {
4239 if (kp_next == NULL)
4240 hash_remove(ht, hi);
4241 else
4242 hi->hi_key = KE2HIKEY(kp_next);
4243 }
4244 else
4245 kp_prev->ke_next = kp_next;
4246 vim_free(kp->next_list);
4247 vim_free(kp->k_syn.cont_in_list);
4248 vim_free(kp);
4249 kp = kp_next;
4250 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00004251 else
Bram Moolenaardad6b692005-01-25 22:14:34 +00004252 {
4253 kp_prev = kp;
4254 kp = kp->ke_next;
4255 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00004256 }
4257 }
4258 }
Bram Moolenaardad6b692005-01-25 22:14:34 +00004259 hash_unlock(ht);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004260}
4261
4262/*
Bram Moolenaardad6b692005-01-25 22:14:34 +00004263 * Clear a whole keyword table.
Bram Moolenaar071d4272004-06-13 20:20:40 +00004264 */
4265 static void
Bram Moolenaardad6b692005-01-25 22:14:34 +00004266clear_keywtab(ht)
4267 hashtab_T *ht;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004268{
Bram Moolenaardad6b692005-01-25 22:14:34 +00004269 hashitem_T *hi;
4270 int todo;
4271 keyentry_T *kp;
4272 keyentry_T *kp_next;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004273
Bram Moolenaara93fa7e2006-04-17 22:14:47 +00004274 todo = (int)ht->ht_used;
Bram Moolenaardad6b692005-01-25 22:14:34 +00004275 for (hi = ht->ht_array; todo > 0; ++hi)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004276 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00004277 if (!HASHITEM_EMPTY(hi))
4278 {
4279 --todo;
Bram Moolenaardad6b692005-01-25 22:14:34 +00004280 for (kp = HI2KE(hi); kp != NULL; kp = kp_next)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004281 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00004282 kp_next = kp->ke_next;
4283 vim_free(kp->next_list);
4284 vim_free(kp->k_syn.cont_in_list);
4285 vim_free(kp);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004286 }
Bram Moolenaardad6b692005-01-25 22:14:34 +00004287 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00004288 }
Bram Moolenaardad6b692005-01-25 22:14:34 +00004289 hash_clear(ht);
4290 hash_init(ht);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004291}
4292
4293/*
4294 * Add a keyword to the list of keywords.
4295 */
4296 static void
Bram Moolenaar860cae12010-06-05 23:22:07 +02004297add_keyword(name, id, flags, cont_in_list, next_list, conceal_char)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004298 char_u *name; /* name of keyword */
4299 int id; /* group ID for this keyword */
4300 int flags; /* flags for this keyword */
4301 short *cont_in_list; /* containedin for this keyword */
4302 short *next_list; /* nextgroup for this keyword */
Bram Moolenaar860cae12010-06-05 23:22:07 +02004303 int conceal_char;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004304{
Bram Moolenaardad6b692005-01-25 22:14:34 +00004305 keyentry_T *kp;
4306 hashtab_T *ht;
4307 hashitem_T *hi;
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004308 char_u *name_ic;
Bram Moolenaardad6b692005-01-25 22:14:34 +00004309 long_u hash;
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004310 char_u name_folded[MAXKEYWLEN + 1];
Bram Moolenaar071d4272004-06-13 20:20:40 +00004311
Bram Moolenaar860cae12010-06-05 23:22:07 +02004312 if (curwin->w_s->b_syn_ic)
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004313 name_ic = str_foldcase(name, (int)STRLEN(name),
4314 name_folded, MAXKEYWLEN + 1);
4315 else
4316 name_ic = name;
Bram Moolenaardad6b692005-01-25 22:14:34 +00004317 kp = (keyentry_T *)alloc((int)(sizeof(keyentry_T) + STRLEN(name_ic)));
4318 if (kp == NULL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004319 return;
Bram Moolenaardad6b692005-01-25 22:14:34 +00004320 STRCPY(kp->keyword, name_ic);
Bram Moolenaardad6b692005-01-25 22:14:34 +00004321 kp->k_syn.id = id;
4322 kp->k_syn.inc_tag = current_syn_inc_tag;
4323 kp->flags = flags;
Bram Moolenaar860cae12010-06-05 23:22:07 +02004324 kp->k_char = conceal_char;
Bram Moolenaardad6b692005-01-25 22:14:34 +00004325 kp->k_syn.cont_in_list = copy_id_list(cont_in_list);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004326 if (cont_in_list != NULL)
Bram Moolenaar860cae12010-06-05 23:22:07 +02004327 curwin->w_s->b_syn_containedin = TRUE;
Bram Moolenaardad6b692005-01-25 22:14:34 +00004328 kp->next_list = copy_id_list(next_list);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004329
Bram Moolenaar860cae12010-06-05 23:22:07 +02004330 if (curwin->w_s->b_syn_ic)
4331 ht = &curwin->w_s->b_keywtab_ic;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004332 else
Bram Moolenaar860cae12010-06-05 23:22:07 +02004333 ht = &curwin->w_s->b_keywtab;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004334
Bram Moolenaardad6b692005-01-25 22:14:34 +00004335 hash = hash_hash(kp->keyword);
4336 hi = hash_lookup(ht, kp->keyword, hash);
4337 if (HASHITEM_EMPTY(hi))
Bram Moolenaar071d4272004-06-13 20:20:40 +00004338 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00004339 /* new keyword, add to hashtable */
4340 kp->ke_next = NULL;
4341 hash_add_item(ht, hi, kp->keyword, hash);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004342 }
Bram Moolenaardad6b692005-01-25 22:14:34 +00004343 else
Bram Moolenaar071d4272004-06-13 20:20:40 +00004344 {
Bram Moolenaardad6b692005-01-25 22:14:34 +00004345 /* keyword already exists, prepend to list */
4346 kp->ke_next = HI2KE(hi);
4347 hi->hi_key = KE2HIKEY(kp);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004348 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00004349}
4350
4351/*
4352 * Get the start and end of the group name argument.
4353 * Return a pointer to the first argument.
4354 * Return NULL if the end of the command was found instead of further args.
4355 */
4356 static char_u *
4357get_group_name(arg, name_end)
4358 char_u *arg; /* start of the argument */
4359 char_u **name_end; /* pointer to end of the name */
4360{
4361 char_u *rest;
4362
4363 *name_end = skiptowhite(arg);
4364 rest = skipwhite(*name_end);
4365
4366 /*
4367 * Check if there are enough arguments. The first argument may be a
4368 * pattern, where '|' is allowed, so only check for NUL.
4369 */
4370 if (ends_excmd(*arg) || *rest == NUL)
4371 return NULL;
4372 return rest;
4373}
4374
4375/*
4376 * Check for syntax command option arguments.
4377 * This can be called at any place in the list of arguments, and just picks
4378 * out the arguments that are known. Can be called several times in a row to
4379 * collect all options in between other arguments.
4380 * Return a pointer to the next argument (which isn't an option).
4381 * Return NULL for any error;
4382 */
4383 static char_u *
Bram Moolenaar860cae12010-06-05 23:22:07 +02004384get_syn_options(arg, opt, conceal_char)
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004385 char_u *arg; /* next argument to be checked */
4386 syn_opt_arg_T *opt; /* various things */
Bram Moolenaar860cae12010-06-05 23:22:07 +02004387 int *conceal_char UNUSED;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004388{
Bram Moolenaar071d4272004-06-13 20:20:40 +00004389 char_u *gname_start, *gname;
4390 int syn_id;
4391 int len;
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004392 char *p;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004393 int i;
4394 int fidx;
4395 static struct flag
4396 {
4397 char *name;
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004398 int argtype;
4399 int flags;
4400 } flagtab[] = { {"cCoOnNtTaAiInNeEdD", 0, HL_CONTAINED},
4401 {"oOnNeElLiInNeE", 0, HL_ONELINE},
4402 {"kKeEeEpPeEnNdD", 0, HL_KEEPEND},
4403 {"eExXtTeEnNdD", 0, HL_EXTEND},
4404 {"eExXcClLuUdDeEnNlL", 0, HL_EXCLUDENL},
4405 {"tTrRaAnNsSpPaArReEnNtT", 0, HL_TRANSP},
4406 {"sSkKiIpPnNlL", 0, HL_SKIPNL},
4407 {"sSkKiIpPwWhHiItTeE", 0, HL_SKIPWHITE},
4408 {"sSkKiIpPeEmMpPtTyY", 0, HL_SKIPEMPTY},
4409 {"gGrRoOuUpPhHeErReE", 0, HL_SYNC_HERE},
4410 {"gGrRoOuUpPtThHeErReE", 0, HL_SYNC_THERE},
4411 {"dDiIsSpPlLaAyY", 0, HL_DISPLAY},
4412 {"fFoOlLdD", 0, HL_FOLD},
Bram Moolenaar860cae12010-06-05 23:22:07 +02004413 {"cCoOnNcCeEaAlL", 0, HL_CONCEAL},
4414 {"cCoOnNcCeEaAlLeEnNdDsS", 0, HL_CONCEALENDS},
4415 {"cCcChHaArR", 11, 0},
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004416 {"cCoOnNtTaAiInNsS", 1, 0},
4417 {"cCoOnNtTaAiInNeEdDiInN", 2, 0},
4418 {"nNeExXtTgGrRoOuUpP", 3, 0},
Bram Moolenaar071d4272004-06-13 20:20:40 +00004419 };
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004420 static char *first_letters = "cCoOkKeEtTsSgGdDfFnN";
Bram Moolenaar071d4272004-06-13 20:20:40 +00004421
4422 if (arg == NULL) /* already detected error */
4423 return NULL;
4424
Bram Moolenaar860cae12010-06-05 23:22:07 +02004425#ifdef FEAT_CONCEAL
4426 if (curwin->w_s->b_syn_conceal)
4427 opt->flags |= HL_CONCEAL;
4428#endif
4429
Bram Moolenaar071d4272004-06-13 20:20:40 +00004430 for (;;)
4431 {
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004432 /*
4433 * This is used very often when a large number of keywords is defined.
4434 * Need to skip quickly when no option name is found.
4435 * Also avoid tolower(), it's slow.
4436 */
4437 if (strchr(first_letters, *arg) == NULL)
4438 break;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004439
4440 for (fidx = sizeof(flagtab) / sizeof(struct flag); --fidx >= 0; )
4441 {
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004442 p = flagtab[fidx].name;
4443 for (i = 0, len = 0; p[i] != NUL; i += 2, ++len)
4444 if (arg[len] != p[i] && arg[len] != p[i + 1])
4445 break;
4446 if (p[i] == NUL && (vim_iswhite(arg[len])
4447 || (flagtab[fidx].argtype > 0
4448 ? arg[len] == '='
4449 : ends_excmd(arg[len]))))
Bram Moolenaar071d4272004-06-13 20:20:40 +00004450 {
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004451 if (opt->keyword
4452 && (flagtab[fidx].flags == HL_DISPLAY
4453 || flagtab[fidx].flags == HL_FOLD
4454 || flagtab[fidx].flags == HL_EXTEND))
Bram Moolenaar071d4272004-06-13 20:20:40 +00004455 /* treat "display", "fold" and "extend" as a keyword */
4456 fidx = -1;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004457 break;
4458 }
4459 }
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004460 if (fidx < 0) /* no match found */
4461 break;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004462
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004463 if (flagtab[fidx].argtype == 1)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004464 {
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004465 if (!opt->has_cont_list)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004466 {
4467 EMSG(_("E395: contains argument not accepted here"));
4468 return NULL;
4469 }
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004470 if (get_id_list(&arg, 8, &opt->cont_list) == FAIL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004471 return NULL;
4472 }
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004473 else if (flagtab[fidx].argtype == 2)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004474 {
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004475#if 0 /* cannot happen */
4476 if (opt->cont_in_list == NULL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004477 {
4478 EMSG(_("E396: containedin argument not accepted here"));
4479 return NULL;
4480 }
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004481#endif
4482 if (get_id_list(&arg, 11, &opt->cont_in_list) == FAIL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004483 return NULL;
4484 }
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004485 else if (flagtab[fidx].argtype == 3)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004486 {
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004487 if (get_id_list(&arg, 9, &opt->next_list) == FAIL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004488 return NULL;
4489 }
Bram Moolenaar860cae12010-06-05 23:22:07 +02004490 else if (flagtab[fidx].argtype == 11 && arg[5] == '=')
4491 {
4492#ifdef FEAT_MBYTE
4493 /* cchar=? */
4494 if (has_mbyte)
4495 {
4496# ifdef FEAT_CONCEAL
4497 *conceal_char = mb_ptr2char(arg + 6);
4498# endif
4499 arg += mb_ptr2len(arg + 6) - 1;
4500 }
4501 else
4502#endif
Bram Moolenaar56be9502010-06-06 14:20:26 +02004503 {
Bram Moolenaar860cae12010-06-05 23:22:07 +02004504#ifdef FEAT_CONCEAL
4505 *conceal_char = arg[6];
4506#else
4507 ;
4508#endif
Bram Moolenaar56be9502010-06-06 14:20:26 +02004509 }
Bram Moolenaar860cae12010-06-05 23:22:07 +02004510 arg = skipwhite(arg + 7);
4511 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00004512 else
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004513 {
4514 opt->flags |= flagtab[fidx].flags;
4515 arg = skipwhite(arg + len);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004516
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004517 if (flagtab[fidx].flags == HL_SYNC_HERE
4518 || flagtab[fidx].flags == HL_SYNC_THERE)
4519 {
4520 if (opt->sync_idx == NULL)
4521 {
4522 EMSG(_("E393: group[t]here not accepted here"));
4523 return NULL;
4524 }
4525 gname_start = arg;
4526 arg = skiptowhite(arg);
4527 if (gname_start == arg)
4528 return NULL;
4529 gname = vim_strnsave(gname_start, (int)(arg - gname_start));
4530 if (gname == NULL)
4531 return NULL;
4532 if (STRCMP(gname, "NONE") == 0)
4533 *opt->sync_idx = NONE_IDX;
4534 else
4535 {
4536 syn_id = syn_name2id(gname);
Bram Moolenaar860cae12010-06-05 23:22:07 +02004537 for (i = curwin->w_s->b_syn_patterns.ga_len; --i >= 0; )
4538 if (SYN_ITEMS(curwin->w_s)[i].sp_syn.id == syn_id
4539 && SYN_ITEMS(curwin->w_s)[i].sp_type == SPTYPE_START)
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004540 {
4541 *opt->sync_idx = i;
4542 break;
4543 }
4544 if (i < 0)
4545 {
4546 EMSG2(_("E394: Didn't find region item for %s"), gname);
4547 vim_free(gname);
4548 return NULL;
4549 }
4550 }
4551
4552 vim_free(gname);
4553 arg = skipwhite(arg);
4554 }
4555#ifdef FEAT_FOLDING
4556 else if (flagtab[fidx].flags == HL_FOLD
4557 && foldmethodIsSyntax(curwin))
4558 /* Need to update folds later. */
4559 foldUpdateAll(curwin);
4560#endif
4561 }
4562 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00004563
4564 return arg;
4565}
4566
4567/*
4568 * Adjustments to syntax item when declared in a ":syn include"'d file.
4569 * Set the contained flag, and if the item is not already contained, add it
4570 * to the specified top-level group, if any.
4571 */
4572 static void
4573syn_incl_toplevel(id, flagsp)
4574 int id;
4575 int *flagsp;
4576{
Bram Moolenaar860cae12010-06-05 23:22:07 +02004577 if ((*flagsp & HL_CONTAINED) || curwin->w_s->b_syn_topgrp == 0)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004578 return;
4579 *flagsp |= HL_CONTAINED;
Bram Moolenaar860cae12010-06-05 23:22:07 +02004580 if (curwin->w_s->b_syn_topgrp >= SYNID_CLUSTER)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004581 {
4582 /* We have to alloc this, because syn_combine_list() will free it. */
4583 short *grp_list = (short *)alloc((unsigned)(2 * sizeof(short)));
Bram Moolenaar860cae12010-06-05 23:22:07 +02004584 int tlg_id = curwin->w_s->b_syn_topgrp - SYNID_CLUSTER;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004585
4586 if (grp_list != NULL)
4587 {
4588 grp_list[0] = id;
4589 grp_list[1] = 0;
Bram Moolenaar860cae12010-06-05 23:22:07 +02004590 syn_combine_list(&SYN_CLSTR(curwin->w_s)[tlg_id].scl_list, &grp_list,
Bram Moolenaar071d4272004-06-13 20:20:40 +00004591 CLUSTER_ADD);
4592 }
4593 }
4594}
4595
4596/*
4597 * Handle ":syntax include [@{group-name}] filename" command.
4598 */
Bram Moolenaar071d4272004-06-13 20:20:40 +00004599 static void
4600syn_cmd_include(eap, syncing)
4601 exarg_T *eap;
Bram Moolenaar2c4278f2009-05-17 11:33:22 +00004602 int syncing UNUSED;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004603{
4604 char_u *arg = eap->arg;
4605 int sgl_id = 1;
4606 char_u *group_name_end;
4607 char_u *rest;
4608 char_u *errormsg = NULL;
4609 int prev_toplvl_grp;
4610 int prev_syn_inc_tag;
4611 int source = FALSE;
4612
4613 eap->nextcmd = find_nextcmd(arg);
4614 if (eap->skip)
4615 return;
4616
4617 if (arg[0] == '@')
4618 {
4619 ++arg;
4620 rest = get_group_name(arg, &group_name_end);
4621 if (rest == NULL)
4622 {
4623 EMSG((char_u *)_("E397: Filename required"));
4624 return;
4625 }
4626 sgl_id = syn_check_cluster(arg, (int)(group_name_end - arg));
4627 /* separate_nextcmd() and expand_filename() depend on this */
4628 eap->arg = rest;
4629 }
4630
4631 /*
4632 * Everything that's left, up to the next command, should be the
4633 * filename to include.
4634 */
4635 eap->argt |= (XFILE | NOSPC);
4636 separate_nextcmd(eap);
4637 if (*eap->arg == '<' || *eap->arg == '$' || mch_isFullName(eap->arg))
4638 {
4639 /* For an absolute path, "$VIM/..." or "<sfile>.." we ":source" the
4640 * file. Need to expand the file name first. In other cases
4641 * ":runtime!" is used. */
4642 source = TRUE;
4643 if (expand_filename(eap, syn_cmdlinep, &errormsg) == FAIL)
4644 {
4645 if (errormsg != NULL)
4646 EMSG(errormsg);
4647 return;
4648 }
4649 }
4650
4651 /*
4652 * Save and restore the existing top-level grouplist id and ":syn
4653 * include" tag around the actual inclusion.
4654 */
4655 prev_syn_inc_tag = current_syn_inc_tag;
4656 current_syn_inc_tag = ++running_syn_inc_tag;
Bram Moolenaar860cae12010-06-05 23:22:07 +02004657 prev_toplvl_grp = curwin->w_s->b_syn_topgrp;
4658 curwin->w_s->b_syn_topgrp = sgl_id;
Bram Moolenaar786b1dc2007-06-28 10:03:45 +00004659 if (source ? do_source(eap->arg, FALSE, DOSO_NONE) == FAIL
4660 : source_runtime(eap->arg, TRUE) == FAIL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00004661 EMSG2(_(e_notopen), eap->arg);
Bram Moolenaar860cae12010-06-05 23:22:07 +02004662 curwin->w_s->b_syn_topgrp = prev_toplvl_grp;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004663 current_syn_inc_tag = prev_syn_inc_tag;
4664}
4665
4666/*
4667 * Handle ":syntax keyword {group-name} [{option}] keyword .." command.
4668 */
Bram Moolenaar071d4272004-06-13 20:20:40 +00004669 static void
4670syn_cmd_keyword(eap, syncing)
4671 exarg_T *eap;
Bram Moolenaar2c4278f2009-05-17 11:33:22 +00004672 int syncing UNUSED;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004673{
4674 char_u *arg = eap->arg;
4675 char_u *group_name_end;
4676 int syn_id;
4677 char_u *rest;
4678 char_u *keyword_copy;
4679 char_u *p;
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004680 char_u *kw;
4681 syn_opt_arg_T syn_opt_arg;
4682 int cnt;
Bram Moolenaar860cae12010-06-05 23:22:07 +02004683 int conceal_char = NUL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004684
4685 rest = get_group_name(arg, &group_name_end);
4686
4687 if (rest != NULL)
4688 {
4689 syn_id = syn_check_group(arg, (int)(group_name_end - arg));
4690
4691 /* allocate a buffer, for removing the backslashes in the keyword */
4692 keyword_copy = alloc((unsigned)STRLEN(rest) + 1);
4693 if (keyword_copy != NULL)
4694 {
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004695 syn_opt_arg.flags = 0;
4696 syn_opt_arg.keyword = TRUE;
4697 syn_opt_arg.sync_idx = NULL;
4698 syn_opt_arg.has_cont_list = FALSE;
4699 syn_opt_arg.cont_in_list = NULL;
4700 syn_opt_arg.next_list = NULL;
4701
Bram Moolenaar071d4272004-06-13 20:20:40 +00004702 /*
4703 * The options given apply to ALL keywords, so all options must be
4704 * found before keywords can be created.
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004705 * 1: collect the options and copy the keywords to keyword_copy.
Bram Moolenaar071d4272004-06-13 20:20:40 +00004706 */
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004707 cnt = 0;
4708 p = keyword_copy;
4709 for ( ; rest != NULL && !ends_excmd(*rest); rest = skipwhite(rest))
Bram Moolenaar071d4272004-06-13 20:20:40 +00004710 {
Bram Moolenaar860cae12010-06-05 23:22:07 +02004711 rest = get_syn_options(rest, &syn_opt_arg, &conceal_char);
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004712 if (rest == NULL || ends_excmd(*rest))
4713 break;
4714 /* Copy the keyword, removing backslashes, and add a NUL. */
4715 while (*rest != NUL && !vim_iswhite(*rest))
Bram Moolenaar071d4272004-06-13 20:20:40 +00004716 {
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004717 if (*rest == '\\' && rest[1] != NUL)
4718 ++rest;
4719 *p++ = *rest++;
4720 }
4721 *p++ = NUL;
4722 ++cnt;
4723 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00004724
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004725 if (!eap->skip)
4726 {
4727 /* Adjust flags for use of ":syn include". */
4728 syn_incl_toplevel(syn_id, &syn_opt_arg.flags);
4729
4730 /*
4731 * 2: Add an entry for each keyword.
4732 */
4733 for (kw = keyword_copy; --cnt >= 0; kw += STRLEN(kw) + 1)
4734 {
4735 for (p = vim_strchr(kw, '['); ; )
4736 {
4737 if (p != NULL)
4738 *p = NUL;
4739 add_keyword(kw, syn_id, syn_opt_arg.flags,
Bram Moolenaar860cae12010-06-05 23:22:07 +02004740 syn_opt_arg.cont_in_list,
4741 syn_opt_arg.next_list, conceal_char);
Bram Moolenaar26a60b42005-02-22 08:49:11 +00004742 if (p == NULL)
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004743 break;
Bram Moolenaar26a60b42005-02-22 08:49:11 +00004744 if (p[1] == NUL)
4745 {
Bram Moolenaarfc1421e2006-04-20 22:17:20 +00004746 EMSG2(_("E789: Missing ']': %s"), kw);
Bram Moolenaar26a60b42005-02-22 08:49:11 +00004747 kw = p + 2; /* skip over the NUL */
4748 break;
4749 }
4750 if (p[1] == ']')
4751 {
4752 kw = p + 1; /* skip over the "]" */
4753 break;
4754 }
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004755#ifdef FEAT_MBYTE
4756 if (has_mbyte)
4757 {
Bram Moolenaar0fa313a2005-08-10 21:07:57 +00004758 int l = (*mb_ptr2len)(p + 1);
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004759
4760 mch_memmove(p, p + 1, l);
4761 p += l;
4762 }
4763 else
Bram Moolenaar071d4272004-06-13 20:20:40 +00004764#endif
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004765 {
4766 p[0] = p[1];
4767 ++p;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004768 }
4769 }
4770 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00004771 }
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004772
Bram Moolenaar071d4272004-06-13 20:20:40 +00004773 vim_free(keyword_copy);
Bram Moolenaar26a60b42005-02-22 08:49:11 +00004774 vim_free(syn_opt_arg.cont_in_list);
4775 vim_free(syn_opt_arg.next_list);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004776 }
4777 }
4778
4779 if (rest != NULL)
4780 eap->nextcmd = check_nextcmd(rest);
4781 else
4782 EMSG2(_(e_invarg2), arg);
4783
Bram Moolenaar1c8f93f2006-03-12 22:10:07 +00004784 redraw_curbuf_later(SOME_VALID);
Bram Moolenaar860cae12010-06-05 23:22:07 +02004785 syn_stack_free_all(curwin->w_s); /* Need to recompute all syntax. */
Bram Moolenaar071d4272004-06-13 20:20:40 +00004786}
4787
4788/*
4789 * Handle ":syntax match {name} [{options}] {pattern} [{options}]".
4790 *
4791 * Also ":syntax sync match {name} [[grouphere | groupthere] {group-name}] .."
4792 */
4793 static void
4794syn_cmd_match(eap, syncing)
4795 exarg_T *eap;
4796 int syncing; /* TRUE for ":syntax sync match .. " */
4797{
4798 char_u *arg = eap->arg;
4799 char_u *group_name_end;
4800 char_u *rest;
4801 synpat_T item; /* the item found in the line */
4802 int syn_id;
4803 int idx;
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004804 syn_opt_arg_T syn_opt_arg;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004805 int sync_idx = 0;
Bram Moolenaar860cae12010-06-05 23:22:07 +02004806 int conceal_char = NUL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004807
4808 /* Isolate the group name, check for validity */
4809 rest = get_group_name(arg, &group_name_end);
4810
4811 /* Get options before the pattern */
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004812 syn_opt_arg.flags = 0;
4813 syn_opt_arg.keyword = FALSE;
4814 syn_opt_arg.sync_idx = syncing ? &sync_idx : NULL;
4815 syn_opt_arg.has_cont_list = TRUE;
4816 syn_opt_arg.cont_list = NULL;
4817 syn_opt_arg.cont_in_list = NULL;
4818 syn_opt_arg.next_list = NULL;
Bram Moolenaar860cae12010-06-05 23:22:07 +02004819 rest = get_syn_options(rest, &syn_opt_arg, &conceal_char);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004820
4821 /* get the pattern. */
4822 init_syn_patterns();
4823 vim_memset(&item, 0, sizeof(item));
4824 rest = get_syn_pattern(rest, &item);
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004825 if (vim_regcomp_had_eol() && !(syn_opt_arg.flags & HL_EXCLUDENL))
4826 syn_opt_arg.flags |= HL_HAS_EOL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004827
4828 /* Get options after the pattern */
Bram Moolenaar860cae12010-06-05 23:22:07 +02004829 rest = get_syn_options(rest, &syn_opt_arg, &conceal_char);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004830
4831 if (rest != NULL) /* all arguments are valid */
4832 {
4833 /*
4834 * Check for trailing command and illegal trailing arguments.
4835 */
4836 eap->nextcmd = check_nextcmd(rest);
4837 if (!ends_excmd(*rest) || eap->skip)
4838 rest = NULL;
Bram Moolenaar860cae12010-06-05 23:22:07 +02004839 else if (ga_grow(&curwin->w_s->b_syn_patterns, 1) != FAIL
Bram Moolenaar071d4272004-06-13 20:20:40 +00004840 && (syn_id = syn_check_group(arg,
4841 (int)(group_name_end - arg))) != 0)
4842 {
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004843 syn_incl_toplevel(syn_id, &syn_opt_arg.flags);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004844 /*
4845 * Store the pattern in the syn_items list
4846 */
Bram Moolenaar860cae12010-06-05 23:22:07 +02004847 idx = curwin->w_s->b_syn_patterns.ga_len;
4848 SYN_ITEMS(curwin->w_s)[idx] = item;
4849 SYN_ITEMS(curwin->w_s)[idx].sp_syncing = syncing;
4850 SYN_ITEMS(curwin->w_s)[idx].sp_type = SPTYPE_MATCH;
4851 SYN_ITEMS(curwin->w_s)[idx].sp_syn.id = syn_id;
4852 SYN_ITEMS(curwin->w_s)[idx].sp_syn.inc_tag = current_syn_inc_tag;
4853 SYN_ITEMS(curwin->w_s)[idx].sp_flags = syn_opt_arg.flags;
4854 SYN_ITEMS(curwin->w_s)[idx].sp_sync_idx = sync_idx;
4855 SYN_ITEMS(curwin->w_s)[idx].sp_cont_list = syn_opt_arg.cont_list;
4856 SYN_ITEMS(curwin->w_s)[idx].sp_syn.cont_in_list =
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004857 syn_opt_arg.cont_in_list;
Bram Moolenaar860cae12010-06-05 23:22:07 +02004858#ifdef FEAT_CONCEAL
4859 SYN_ITEMS(curwin->w_s)[idx].sp_char = conceal_char;
4860#endif
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004861 if (syn_opt_arg.cont_in_list != NULL)
Bram Moolenaar860cae12010-06-05 23:22:07 +02004862 curwin->w_s->b_syn_containedin = TRUE;
4863 SYN_ITEMS(curwin->w_s)[idx].sp_next_list = syn_opt_arg.next_list;
4864 ++curwin->w_s->b_syn_patterns.ga_len;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004865
4866 /* remember that we found a match for syncing on */
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004867 if (syn_opt_arg.flags & (HL_SYNC_HERE|HL_SYNC_THERE))
Bram Moolenaar860cae12010-06-05 23:22:07 +02004868 curwin->w_s->b_syn_sync_flags |= SF_MATCH;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004869#ifdef FEAT_FOLDING
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004870 if (syn_opt_arg.flags & HL_FOLD)
Bram Moolenaar860cae12010-06-05 23:22:07 +02004871 ++curwin->w_s->b_syn_folditems;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004872#endif
4873
Bram Moolenaar1c8f93f2006-03-12 22:10:07 +00004874 redraw_curbuf_later(SOME_VALID);
Bram Moolenaar860cae12010-06-05 23:22:07 +02004875 syn_stack_free_all(curwin->w_s); /* Need to recompute all syntax. */
Bram Moolenaar071d4272004-06-13 20:20:40 +00004876 return; /* don't free the progs and patterns now */
4877 }
4878 }
4879
4880 /*
4881 * Something failed, free the allocated memory.
4882 */
4883 vim_free(item.sp_prog);
4884 vim_free(item.sp_pattern);
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004885 vim_free(syn_opt_arg.cont_list);
4886 vim_free(syn_opt_arg.cont_in_list);
4887 vim_free(syn_opt_arg.next_list);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004888
4889 if (rest == NULL)
4890 EMSG2(_(e_invarg2), arg);
4891}
4892
4893/*
4894 * Handle ":syntax region {group-name} [matchgroup={group-name}]
4895 * start {start} .. [skip {skip}] end {end} .. [{options}]".
4896 */
4897 static void
4898syn_cmd_region(eap, syncing)
4899 exarg_T *eap;
4900 int syncing; /* TRUE for ":syntax sync region .." */
4901{
4902 char_u *arg = eap->arg;
4903 char_u *group_name_end;
4904 char_u *rest; /* next arg, NULL on error */
4905 char_u *key_end;
4906 char_u *key = NULL;
4907 char_u *p;
4908 int item;
4909#define ITEM_START 0
4910#define ITEM_SKIP 1
4911#define ITEM_END 2
4912#define ITEM_MATCHGROUP 3
4913 struct pat_ptr
4914 {
4915 synpat_T *pp_synp; /* pointer to syn_pattern */
4916 int pp_matchgroup_id; /* matchgroup ID */
4917 struct pat_ptr *pp_next; /* pointer to next pat_ptr */
4918 } *(pat_ptrs[3]);
4919 /* patterns found in the line */
4920 struct pat_ptr *ppp;
4921 struct pat_ptr *ppp_next;
4922 int pat_count = 0; /* nr of syn_patterns found */
4923 int syn_id;
4924 int matchgroup_id = 0;
4925 int not_enough = FALSE; /* not enough arguments */
4926 int illegal = FALSE; /* illegal arguments */
4927 int success = FALSE;
4928 int idx;
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004929 syn_opt_arg_T syn_opt_arg;
Bram Moolenaar860cae12010-06-05 23:22:07 +02004930 int conceal_char = NUL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00004931
4932 /* Isolate the group name, check for validity */
4933 rest = get_group_name(arg, &group_name_end);
4934
4935 pat_ptrs[0] = NULL;
4936 pat_ptrs[1] = NULL;
4937 pat_ptrs[2] = NULL;
4938
4939 init_syn_patterns();
4940
Bram Moolenaar6ac54292005-02-02 23:07:25 +00004941 syn_opt_arg.flags = 0;
4942 syn_opt_arg.keyword = FALSE;
4943 syn_opt_arg.sync_idx = NULL;
4944 syn_opt_arg.has_cont_list = TRUE;
4945 syn_opt_arg.cont_list = NULL;
4946 syn_opt_arg.cont_in_list = NULL;
4947 syn_opt_arg.next_list = NULL;
4948
Bram Moolenaar071d4272004-06-13 20:20:40 +00004949 /*
4950 * get the options, patterns and matchgroup.
4951 */
4952 while (rest != NULL && !ends_excmd(*rest))
4953 {
4954 /* Check for option arguments */
Bram Moolenaar860cae12010-06-05 23:22:07 +02004955 rest = get_syn_options(rest, &syn_opt_arg, &conceal_char);
Bram Moolenaar071d4272004-06-13 20:20:40 +00004956 if (rest == NULL || ends_excmd(*rest))
4957 break;
4958
4959 /* must be a pattern or matchgroup then */
4960 key_end = rest;
4961 while (*key_end && !vim_iswhite(*key_end) && *key_end != '=')
4962 ++key_end;
4963 vim_free(key);
4964 key = vim_strnsave_up(rest, (int)(key_end - rest));
4965 if (key == NULL) /* out of memory */
4966 {
4967 rest = NULL;
4968 break;
4969 }
4970 if (STRCMP(key, "MATCHGROUP") == 0)
4971 item = ITEM_MATCHGROUP;
4972 else if (STRCMP(key, "START") == 0)
4973 item = ITEM_START;
4974 else if (STRCMP(key, "END") == 0)
4975 item = ITEM_END;
4976 else if (STRCMP(key, "SKIP") == 0)
4977 {
4978 if (pat_ptrs[ITEM_SKIP] != NULL) /* one skip pattern allowed */
4979 {
4980 illegal = TRUE;
4981 break;
4982 }
4983 item = ITEM_SKIP;
4984 }
4985 else
4986 break;
4987 rest = skipwhite(key_end);
4988 if (*rest != '=')
4989 {
4990 rest = NULL;
4991 EMSG2(_("E398: Missing '=': %s"), arg);
4992 break;
4993 }
4994 rest = skipwhite(rest + 1);
4995 if (*rest == NUL)
4996 {
4997 not_enough = TRUE;
4998 break;
4999 }
5000
5001 if (item == ITEM_MATCHGROUP)
5002 {
5003 p = skiptowhite(rest);
5004 if ((p - rest == 4 && STRNCMP(rest, "NONE", 4) == 0) || eap->skip)
5005 matchgroup_id = 0;
5006 else
5007 {
5008 matchgroup_id = syn_check_group(rest, (int)(p - rest));
5009 if (matchgroup_id == 0)
5010 {
5011 illegal = TRUE;
5012 break;
5013 }
5014 }
5015 rest = skipwhite(p);
5016 }
5017 else
5018 {
5019 /*
5020 * Allocate room for a syn_pattern, and link it in the list of
5021 * syn_patterns for this item, at the start (because the list is
5022 * used from end to start).
5023 */
5024 ppp = (struct pat_ptr *)alloc((unsigned)sizeof(struct pat_ptr));
5025 if (ppp == NULL)
5026 {
5027 rest = NULL;
5028 break;
5029 }
5030 ppp->pp_next = pat_ptrs[item];
5031 pat_ptrs[item] = ppp;
5032 ppp->pp_synp = (synpat_T *)alloc_clear((unsigned)sizeof(synpat_T));
5033 if (ppp->pp_synp == NULL)
5034 {
5035 rest = NULL;
5036 break;
5037 }
5038
5039 /*
5040 * Get the syntax pattern and the following offset(s).
5041 */
5042 /* Enable the appropriate \z specials. */
5043 if (item == ITEM_START)
5044 reg_do_extmatch = REX_SET;
5045 else if (item == ITEM_SKIP || item == ITEM_END)
5046 reg_do_extmatch = REX_USE;
5047 rest = get_syn_pattern(rest, ppp->pp_synp);
5048 reg_do_extmatch = 0;
5049 if (item == ITEM_END && vim_regcomp_had_eol()
Bram Moolenaar6ac54292005-02-02 23:07:25 +00005050 && !(syn_opt_arg.flags & HL_EXCLUDENL))
Bram Moolenaar071d4272004-06-13 20:20:40 +00005051 ppp->pp_synp->sp_flags |= HL_HAS_EOL;
5052 ppp->pp_matchgroup_id = matchgroup_id;
5053 ++pat_count;
5054 }
5055 }
5056 vim_free(key);
5057 if (illegal || not_enough)
5058 rest = NULL;
5059
5060 /*
5061 * Must have a "start" and "end" pattern.
5062 */
5063 if (rest != NULL && (pat_ptrs[ITEM_START] == NULL ||
5064 pat_ptrs[ITEM_END] == NULL))
5065 {
5066 not_enough = TRUE;
5067 rest = NULL;
5068 }
5069
5070 if (rest != NULL)
5071 {
5072 /*
5073 * Check for trailing garbage or command.
5074 * If OK, add the item.
5075 */
5076 eap->nextcmd = check_nextcmd(rest);
5077 if (!ends_excmd(*rest) || eap->skip)
5078 rest = NULL;
Bram Moolenaar860cae12010-06-05 23:22:07 +02005079 else if (ga_grow(&(curwin->w_s->b_syn_patterns), pat_count) != FAIL
Bram Moolenaar071d4272004-06-13 20:20:40 +00005080 && (syn_id = syn_check_group(arg,
5081 (int)(group_name_end - arg))) != 0)
5082 {
Bram Moolenaar6ac54292005-02-02 23:07:25 +00005083 syn_incl_toplevel(syn_id, &syn_opt_arg.flags);
Bram Moolenaar071d4272004-06-13 20:20:40 +00005084 /*
5085 * Store the start/skip/end in the syn_items list
5086 */
Bram Moolenaar860cae12010-06-05 23:22:07 +02005087 idx = curwin->w_s->b_syn_patterns.ga_len;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005088 for (item = ITEM_START; item <= ITEM_END; ++item)
5089 {
5090 for (ppp = pat_ptrs[item]; ppp != NULL; ppp = ppp->pp_next)
5091 {
Bram Moolenaar860cae12010-06-05 23:22:07 +02005092 SYN_ITEMS(curwin->w_s)[idx] = *(ppp->pp_synp);
5093 SYN_ITEMS(curwin->w_s)[idx].sp_syncing = syncing;
5094 SYN_ITEMS(curwin->w_s)[idx].sp_type =
Bram Moolenaar071d4272004-06-13 20:20:40 +00005095 (item == ITEM_START) ? SPTYPE_START :
5096 (item == ITEM_SKIP) ? SPTYPE_SKIP : SPTYPE_END;
Bram Moolenaar860cae12010-06-05 23:22:07 +02005097 SYN_ITEMS(curwin->w_s)[idx].sp_flags |= syn_opt_arg.flags;
5098 SYN_ITEMS(curwin->w_s)[idx].sp_syn.id = syn_id;
5099 SYN_ITEMS(curwin->w_s)[idx].sp_syn.inc_tag = current_syn_inc_tag;
5100 SYN_ITEMS(curwin->w_s)[idx].sp_syn_match_id =
Bram Moolenaar071d4272004-06-13 20:20:40 +00005101 ppp->pp_matchgroup_id;
Bram Moolenaar860cae12010-06-05 23:22:07 +02005102#ifdef FEAT_CONCEAL
5103 SYN_ITEMS(curwin->w_s)[idx].sp_char = conceal_char;
5104#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00005105 if (item == ITEM_START)
5106 {
Bram Moolenaar860cae12010-06-05 23:22:07 +02005107 SYN_ITEMS(curwin->w_s)[idx].sp_cont_list =
Bram Moolenaar6ac54292005-02-02 23:07:25 +00005108 syn_opt_arg.cont_list;
Bram Moolenaar860cae12010-06-05 23:22:07 +02005109 SYN_ITEMS(curwin->w_s)[idx].sp_syn.cont_in_list =
Bram Moolenaar6ac54292005-02-02 23:07:25 +00005110 syn_opt_arg.cont_in_list;
5111 if (syn_opt_arg.cont_in_list != NULL)
Bram Moolenaar860cae12010-06-05 23:22:07 +02005112 curwin->w_s->b_syn_containedin = TRUE;
5113 SYN_ITEMS(curwin->w_s)[idx].sp_next_list =
Bram Moolenaar6ac54292005-02-02 23:07:25 +00005114 syn_opt_arg.next_list;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005115 }
Bram Moolenaar860cae12010-06-05 23:22:07 +02005116 ++curwin->w_s->b_syn_patterns.ga_len;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005117 ++idx;
5118#ifdef FEAT_FOLDING
Bram Moolenaar6ac54292005-02-02 23:07:25 +00005119 if (syn_opt_arg.flags & HL_FOLD)
Bram Moolenaar860cae12010-06-05 23:22:07 +02005120 ++curwin->w_s->b_syn_folditems;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005121#endif
5122 }
5123 }
5124
Bram Moolenaar1c8f93f2006-03-12 22:10:07 +00005125 redraw_curbuf_later(SOME_VALID);
Bram Moolenaar860cae12010-06-05 23:22:07 +02005126 syn_stack_free_all(curwin->w_s); /* Need to recompute all syntax. */
Bram Moolenaar071d4272004-06-13 20:20:40 +00005127 success = TRUE; /* don't free the progs and patterns now */
5128 }
5129 }
5130
5131 /*
5132 * Free the allocated memory.
5133 */
5134 for (item = ITEM_START; item <= ITEM_END; ++item)
5135 for (ppp = pat_ptrs[item]; ppp != NULL; ppp = ppp_next)
5136 {
5137 if (!success)
5138 {
5139 vim_free(ppp->pp_synp->sp_prog);
5140 vim_free(ppp->pp_synp->sp_pattern);
5141 }
5142 vim_free(ppp->pp_synp);
5143 ppp_next = ppp->pp_next;
5144 vim_free(ppp);
5145 }
5146
5147 if (!success)
5148 {
Bram Moolenaar6ac54292005-02-02 23:07:25 +00005149 vim_free(syn_opt_arg.cont_list);
5150 vim_free(syn_opt_arg.cont_in_list);
5151 vim_free(syn_opt_arg.next_list);
Bram Moolenaar071d4272004-06-13 20:20:40 +00005152 if (not_enough)
5153 EMSG2(_("E399: Not enough arguments: syntax region %s"), arg);
5154 else if (illegal || rest == NULL)
5155 EMSG2(_(e_invarg2), arg);
5156 }
5157}
5158
5159/*
5160 * A simple syntax group ID comparison function suitable for use in qsort()
5161 */
5162 static int
5163#ifdef __BORLANDC__
5164_RTLENTRYF
5165#endif
5166syn_compare_stub(v1, v2)
5167 const void *v1;
5168 const void *v2;
5169{
5170 const short *s1 = v1;
5171 const short *s2 = v2;
5172
5173 return (*s1 > *s2 ? 1 : *s1 < *s2 ? -1 : 0);
5174}
5175
5176/*
5177 * Combines lists of syntax clusters.
5178 * *clstr1 and *clstr2 must both be allocated memory; they will be consumed.
5179 */
5180 static void
5181syn_combine_list(clstr1, clstr2, list_op)
5182 short **clstr1;
5183 short **clstr2;
5184 int list_op;
5185{
5186 int count1 = 0;
5187 int count2 = 0;
5188 short *g1;
5189 short *g2;
5190 short *clstr = NULL;
5191 int count;
5192 int round;
5193
5194 /*
5195 * Handle degenerate cases.
5196 */
5197 if (*clstr2 == NULL)
5198 return;
5199 if (*clstr1 == NULL || list_op == CLUSTER_REPLACE)
5200 {
5201 if (list_op == CLUSTER_REPLACE)
5202 vim_free(*clstr1);
5203 if (list_op == CLUSTER_REPLACE || list_op == CLUSTER_ADD)
5204 *clstr1 = *clstr2;
5205 else
5206 vim_free(*clstr2);
5207 return;
5208 }
5209
5210 for (g1 = *clstr1; *g1; g1++)
5211 ++count1;
5212 for (g2 = *clstr2; *g2; g2++)
5213 ++count2;
5214
5215 /*
5216 * For speed purposes, sort both lists.
5217 */
5218 qsort(*clstr1, (size_t)count1, sizeof(short), syn_compare_stub);
5219 qsort(*clstr2, (size_t)count2, sizeof(short), syn_compare_stub);
5220
5221 /*
5222 * We proceed in two passes; in round 1, we count the elements to place
5223 * in the new list, and in round 2, we allocate and populate the new
5224 * list. For speed, we use a mergesort-like method, adding the smaller
5225 * of the current elements in each list to the new list.
5226 */
5227 for (round = 1; round <= 2; round++)
5228 {
5229 g1 = *clstr1;
5230 g2 = *clstr2;
5231 count = 0;
5232
5233 /*
5234 * First, loop through the lists until one of them is empty.
5235 */
5236 while (*g1 && *g2)
5237 {
5238 /*
5239 * We always want to add from the first list.
5240 */
5241 if (*g1 < *g2)
5242 {
5243 if (round == 2)
5244 clstr[count] = *g1;
5245 count++;
5246 g1++;
5247 continue;
5248 }
5249 /*
5250 * We only want to add from the second list if we're adding the
5251 * lists.
5252 */
5253 if (list_op == CLUSTER_ADD)
5254 {
5255 if (round == 2)
5256 clstr[count] = *g2;
5257 count++;
5258 }
5259 if (*g1 == *g2)
5260 g1++;
5261 g2++;
5262 }
5263
5264 /*
5265 * Now add the leftovers from whichever list didn't get finished
5266 * first. As before, we only want to add from the second list if
5267 * we're adding the lists.
5268 */
5269 for (; *g1; g1++, count++)
5270 if (round == 2)
5271 clstr[count] = *g1;
5272 if (list_op == CLUSTER_ADD)
5273 for (; *g2; g2++, count++)
5274 if (round == 2)
5275 clstr[count] = *g2;
5276
5277 if (round == 1)
5278 {
5279 /*
5280 * If the group ended up empty, we don't need to allocate any
5281 * space for it.
5282 */
5283 if (count == 0)
5284 {
5285 clstr = NULL;
5286 break;
5287 }
5288 clstr = (short *)alloc((unsigned)((count + 1) * sizeof(short)));
5289 if (clstr == NULL)
5290 break;
5291 clstr[count] = 0;
5292 }
5293 }
5294
5295 /*
5296 * Finally, put the new list in place.
5297 */
5298 vim_free(*clstr1);
5299 vim_free(*clstr2);
5300 *clstr1 = clstr;
5301}
5302
5303/*
5304 * Lookup a syntax cluster name and return it's ID.
5305 * If it is not found, 0 is returned.
5306 */
5307 static int
5308syn_scl_name2id(name)
5309 char_u *name;
5310{
5311 int i;
5312 char_u *name_u;
5313
5314 /* Avoid using stricmp() too much, it's slow on some systems */
5315 name_u = vim_strsave_up(name);
5316 if (name_u == NULL)
5317 return 0;
Bram Moolenaar860cae12010-06-05 23:22:07 +02005318 for (i = curwin->w_s->b_syn_clusters.ga_len; --i >= 0; )
5319 if (SYN_CLSTR(curwin->w_s)[i].scl_name_u != NULL
5320 && STRCMP(name_u, SYN_CLSTR(curwin->w_s)[i].scl_name_u) == 0)
Bram Moolenaar071d4272004-06-13 20:20:40 +00005321 break;
5322 vim_free(name_u);
5323 return (i < 0 ? 0 : i + SYNID_CLUSTER);
5324}
5325
5326/*
5327 * Like syn_scl_name2id(), but take a pointer + length argument.
5328 */
5329 static int
5330syn_scl_namen2id(linep, len)
5331 char_u *linep;
5332 int len;
5333{
5334 char_u *name;
5335 int id = 0;
5336
5337 name = vim_strnsave(linep, len);
5338 if (name != NULL)
5339 {
5340 id = syn_scl_name2id(name);
5341 vim_free(name);
5342 }
5343 return id;
5344}
5345
5346/*
5347 * Find syntax cluster name in the table and return it's ID.
5348 * The argument is a pointer to the name and the length of the name.
5349 * If it doesn't exist yet, a new entry is created.
5350 * Return 0 for failure.
5351 */
5352 static int
5353syn_check_cluster(pp, len)
5354 char_u *pp;
5355 int len;
5356{
5357 int id;
5358 char_u *name;
5359
5360 name = vim_strnsave(pp, len);
5361 if (name == NULL)
5362 return 0;
5363
5364 id = syn_scl_name2id(name);
5365 if (id == 0) /* doesn't exist yet */
5366 id = syn_add_cluster(name);
5367 else
5368 vim_free(name);
5369 return id;
5370}
5371
5372/*
5373 * Add new syntax cluster and return it's ID.
5374 * "name" must be an allocated string, it will be consumed.
5375 * Return 0 for failure.
5376 */
5377 static int
5378syn_add_cluster(name)
Bram Moolenaar217ad922005-03-20 22:37:15 +00005379 char_u *name;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005380{
Bram Moolenaar217ad922005-03-20 22:37:15 +00005381 int len;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005382
5383 /*
5384 * First call for this growarray: init growing array.
5385 */
Bram Moolenaar860cae12010-06-05 23:22:07 +02005386 if (curwin->w_s->b_syn_clusters.ga_data == NULL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00005387 {
Bram Moolenaar860cae12010-06-05 23:22:07 +02005388 curwin->w_s->b_syn_clusters.ga_itemsize = sizeof(syn_cluster_T);
5389 curwin->w_s->b_syn_clusters.ga_growsize = 10;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005390 }
5391
5392 /*
5393 * Make room for at least one other cluster entry.
5394 */
Bram Moolenaar860cae12010-06-05 23:22:07 +02005395 if (ga_grow(&curwin->w_s->b_syn_clusters, 1) == FAIL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00005396 {
5397 vim_free(name);
5398 return 0;
5399 }
Bram Moolenaar860cae12010-06-05 23:22:07 +02005400 len = curwin->w_s->b_syn_clusters.ga_len;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005401
Bram Moolenaar860cae12010-06-05 23:22:07 +02005402 vim_memset(&(SYN_CLSTR(curwin->w_s)[len]), 0, sizeof(syn_cluster_T));
5403 SYN_CLSTR(curwin->w_s)[len].scl_name = name;
5404 SYN_CLSTR(curwin->w_s)[len].scl_name_u = vim_strsave_up(name);
5405 SYN_CLSTR(curwin->w_s)[len].scl_list = NULL;
5406 ++curwin->w_s->b_syn_clusters.ga_len;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005407
Bram Moolenaar217ad922005-03-20 22:37:15 +00005408 if (STRICMP(name, "Spell") == 0)
Bram Moolenaar860cae12010-06-05 23:22:07 +02005409 curwin->w_s->b_spell_cluster_id = len + SYNID_CLUSTER;
Bram Moolenaar6bb68362005-03-22 23:03:44 +00005410 if (STRICMP(name, "NoSpell") == 0)
Bram Moolenaar860cae12010-06-05 23:22:07 +02005411 curwin->w_s->b_nospell_cluster_id = len + SYNID_CLUSTER;
Bram Moolenaar217ad922005-03-20 22:37:15 +00005412
Bram Moolenaar071d4272004-06-13 20:20:40 +00005413 return len + SYNID_CLUSTER;
5414}
5415
5416/*
5417 * Handle ":syntax cluster {cluster-name} [contains={groupname},..]
5418 * [add={groupname},..] [remove={groupname},..]".
5419 */
Bram Moolenaar071d4272004-06-13 20:20:40 +00005420 static void
5421syn_cmd_cluster(eap, syncing)
5422 exarg_T *eap;
Bram Moolenaar2c4278f2009-05-17 11:33:22 +00005423 int syncing UNUSED;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005424{
5425 char_u *arg = eap->arg;
5426 char_u *group_name_end;
5427 char_u *rest;
5428 int scl_id;
5429 short *clstr_list;
5430 int got_clstr = FALSE;
5431 int opt_len;
5432 int list_op;
5433
5434 eap->nextcmd = find_nextcmd(arg);
5435 if (eap->skip)
5436 return;
5437
5438 rest = get_group_name(arg, &group_name_end);
5439
5440 if (rest != NULL)
5441 {
5442 scl_id = syn_check_cluster(arg, (int)(group_name_end - arg))
Bram Moolenaar217ad922005-03-20 22:37:15 +00005443 - SYNID_CLUSTER;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005444
5445 for (;;)
5446 {
5447 if (STRNICMP(rest, "add", 3) == 0
5448 && (vim_iswhite(rest[3]) || rest[3] == '='))
5449 {
5450 opt_len = 3;
5451 list_op = CLUSTER_ADD;
5452 }
5453 else if (STRNICMP(rest, "remove", 6) == 0
5454 && (vim_iswhite(rest[6]) || rest[6] == '='))
5455 {
5456 opt_len = 6;
5457 list_op = CLUSTER_SUBTRACT;
5458 }
5459 else if (STRNICMP(rest, "contains", 8) == 0
5460 && (vim_iswhite(rest[8]) || rest[8] == '='))
5461 {
5462 opt_len = 8;
5463 list_op = CLUSTER_REPLACE;
5464 }
5465 else
5466 break;
5467
5468 clstr_list = NULL;
5469 if (get_id_list(&rest, opt_len, &clstr_list) == FAIL)
5470 {
5471 EMSG2(_(e_invarg2), rest);
5472 break;
5473 }
Bram Moolenaar860cae12010-06-05 23:22:07 +02005474 syn_combine_list(&SYN_CLSTR(curwin->w_s)[scl_id].scl_list,
Bram Moolenaar071d4272004-06-13 20:20:40 +00005475 &clstr_list, list_op);
5476 got_clstr = TRUE;
5477 }
5478
5479 if (got_clstr)
5480 {
Bram Moolenaar1c8f93f2006-03-12 22:10:07 +00005481 redraw_curbuf_later(SOME_VALID);
Bram Moolenaar860cae12010-06-05 23:22:07 +02005482 syn_stack_free_all(curwin->w_s); /* Need to recompute all syntax. */
Bram Moolenaar071d4272004-06-13 20:20:40 +00005483 }
5484 }
5485
5486 if (!got_clstr)
5487 EMSG(_("E400: No cluster specified"));
5488 if (rest == NULL || !ends_excmd(*rest))
5489 EMSG2(_(e_invarg2), arg);
5490}
5491
5492/*
5493 * On first call for current buffer: Init growing array.
5494 */
5495 static void
5496init_syn_patterns()
5497{
Bram Moolenaar860cae12010-06-05 23:22:07 +02005498 curwin->w_s->b_syn_patterns.ga_itemsize = sizeof(synpat_T);
5499 curwin->w_s->b_syn_patterns.ga_growsize = 10;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005500}
5501
5502/*
5503 * Get one pattern for a ":syntax match" or ":syntax region" command.
5504 * Stores the pattern and program in a synpat_T.
5505 * Returns a pointer to the next argument, or NULL in case of an error.
5506 */
5507 static char_u *
5508get_syn_pattern(arg, ci)
5509 char_u *arg;
5510 synpat_T *ci;
5511{
5512 char_u *end;
5513 int *p;
5514 int idx;
5515 char_u *cpo_save;
5516
5517 /* need at least three chars */
5518 if (arg == NULL || arg[1] == NUL || arg[2] == NUL)
5519 return NULL;
5520
5521 end = skip_regexp(arg + 1, *arg, TRUE, NULL);
5522 if (*end != *arg) /* end delimiter not found */
5523 {
5524 EMSG2(_("E401: Pattern delimiter not found: %s"), arg);
5525 return NULL;
5526 }
5527 /* store the pattern and compiled regexp program */
5528 if ((ci->sp_pattern = vim_strnsave(arg + 1, (int)(end - arg - 1))) == NULL)
5529 return NULL;
5530
5531 /* Make 'cpoptions' empty, to avoid the 'l' flag */
5532 cpo_save = p_cpo;
5533 p_cpo = (char_u *)"";
5534 ci->sp_prog = vim_regcomp(ci->sp_pattern, RE_MAGIC);
5535 p_cpo = cpo_save;
5536
5537 if (ci->sp_prog == NULL)
5538 return NULL;
Bram Moolenaar860cae12010-06-05 23:22:07 +02005539 ci->sp_ic = curwin->w_s->b_syn_ic;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005540
5541 /*
5542 * Check for a match, highlight or region offset.
5543 */
5544 ++end;
5545 do
5546 {
5547 for (idx = SPO_COUNT; --idx >= 0; )
5548 if (STRNCMP(end, spo_name_tab[idx], 3) == 0)
5549 break;
5550 if (idx >= 0)
5551 {
5552 p = &(ci->sp_offsets[idx]);
5553 if (idx != SPO_LC_OFF)
5554 switch (end[3])
5555 {
5556 case 's': break;
5557 case 'b': break;
5558 case 'e': idx += SPO_COUNT; break;
5559 default: idx = -1; break;
5560 }
5561 if (idx >= 0)
5562 {
5563 ci->sp_off_flags |= (1 << idx);
5564 if (idx == SPO_LC_OFF) /* lc=99 */
5565 {
5566 end += 3;
5567 *p = getdigits(&end);
5568
5569 /* "lc=" offset automatically sets "ms=" offset */
5570 if (!(ci->sp_off_flags & (1 << SPO_MS_OFF)))
5571 {
5572 ci->sp_off_flags |= (1 << SPO_MS_OFF);
5573 ci->sp_offsets[SPO_MS_OFF] = *p;
5574 }
5575 }
5576 else /* yy=x+99 */
5577 {
5578 end += 4;
5579 if (*end == '+')
5580 {
5581 ++end;
5582 *p = getdigits(&end); /* positive offset */
5583 }
5584 else if (*end == '-')
5585 {
5586 ++end;
5587 *p = -getdigits(&end); /* negative offset */
5588 }
5589 }
5590 if (*end != ',')
5591 break;
5592 ++end;
5593 }
5594 }
5595 } while (idx >= 0);
5596
5597 if (!ends_excmd(*end) && !vim_iswhite(*end))
5598 {
5599 EMSG2(_("E402: Garbage after pattern: %s"), arg);
5600 return NULL;
5601 }
5602 return skipwhite(end);
5603}
5604
5605/*
5606 * Handle ":syntax sync .." command.
5607 */
Bram Moolenaar071d4272004-06-13 20:20:40 +00005608 static void
5609syn_cmd_sync(eap, syncing)
5610 exarg_T *eap;
Bram Moolenaar2c4278f2009-05-17 11:33:22 +00005611 int syncing UNUSED;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005612{
5613 char_u *arg_start = eap->arg;
5614 char_u *arg_end;
5615 char_u *key = NULL;
5616 char_u *next_arg;
5617 int illegal = FALSE;
5618 int finished = FALSE;
5619 long n;
5620 char_u *cpo_save;
5621
5622 if (ends_excmd(*arg_start))
5623 {
5624 syn_cmd_list(eap, TRUE);
5625 return;
5626 }
5627
5628 while (!ends_excmd(*arg_start))
5629 {
5630 arg_end = skiptowhite(arg_start);
5631 next_arg = skipwhite(arg_end);
5632 vim_free(key);
5633 key = vim_strnsave_up(arg_start, (int)(arg_end - arg_start));
5634 if (STRCMP(key, "CCOMMENT") == 0)
5635 {
5636 if (!eap->skip)
Bram Moolenaar860cae12010-06-05 23:22:07 +02005637 curwin->w_s->b_syn_sync_flags |= SF_CCOMMENT;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005638 if (!ends_excmd(*next_arg))
5639 {
5640 arg_end = skiptowhite(next_arg);
5641 if (!eap->skip)
Bram Moolenaar860cae12010-06-05 23:22:07 +02005642 curwin->w_s->b_syn_sync_id = syn_check_group(next_arg,
Bram Moolenaar071d4272004-06-13 20:20:40 +00005643 (int)(arg_end - next_arg));
5644 next_arg = skipwhite(arg_end);
5645 }
5646 else if (!eap->skip)
Bram Moolenaar860cae12010-06-05 23:22:07 +02005647 curwin->w_s->b_syn_sync_id = syn_name2id((char_u *)"Comment");
Bram Moolenaar071d4272004-06-13 20:20:40 +00005648 }
5649 else if ( STRNCMP(key, "LINES", 5) == 0
5650 || STRNCMP(key, "MINLINES", 8) == 0
5651 || STRNCMP(key, "MAXLINES", 8) == 0
5652 || STRNCMP(key, "LINEBREAKS", 10) == 0)
5653 {
5654 if (key[4] == 'S')
5655 arg_end = key + 6;
5656 else if (key[0] == 'L')
5657 arg_end = key + 11;
5658 else
5659 arg_end = key + 9;
5660 if (arg_end[-1] != '=' || !VIM_ISDIGIT(*arg_end))
5661 {
5662 illegal = TRUE;
5663 break;
5664 }
5665 n = getdigits(&arg_end);
5666 if (!eap->skip)
5667 {
5668 if (key[4] == 'B')
Bram Moolenaar860cae12010-06-05 23:22:07 +02005669 curwin->w_s->b_syn_sync_linebreaks = n;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005670 else if (key[1] == 'A')
Bram Moolenaar860cae12010-06-05 23:22:07 +02005671 curwin->w_s->b_syn_sync_maxlines = n;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005672 else
Bram Moolenaar860cae12010-06-05 23:22:07 +02005673 curwin->w_s->b_syn_sync_minlines = n;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005674 }
5675 }
5676 else if (STRCMP(key, "FROMSTART") == 0)
5677 {
5678 if (!eap->skip)
5679 {
Bram Moolenaar860cae12010-06-05 23:22:07 +02005680 curwin->w_s->b_syn_sync_minlines = MAXLNUM;
5681 curwin->w_s->b_syn_sync_maxlines = 0;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005682 }
5683 }
5684 else if (STRCMP(key, "LINECONT") == 0)
5685 {
Bram Moolenaar860cae12010-06-05 23:22:07 +02005686 if (curwin->w_s->b_syn_linecont_pat != NULL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00005687 {
5688 EMSG(_("E403: syntax sync: line continuations pattern specified twice"));
5689 finished = TRUE;
5690 break;
5691 }
5692 arg_end = skip_regexp(next_arg + 1, *next_arg, TRUE, NULL);
5693 if (*arg_end != *next_arg) /* end delimiter not found */
5694 {
5695 illegal = TRUE;
5696 break;
5697 }
5698
5699 if (!eap->skip)
5700 {
5701 /* store the pattern and compiled regexp program */
Bram Moolenaar860cae12010-06-05 23:22:07 +02005702 if ((curwin->w_s->b_syn_linecont_pat = vim_strnsave(next_arg + 1,
Bram Moolenaar071d4272004-06-13 20:20:40 +00005703 (int)(arg_end - next_arg - 1))) == NULL)
5704 {
5705 finished = TRUE;
5706 break;
5707 }
Bram Moolenaar860cae12010-06-05 23:22:07 +02005708 curwin->w_s->b_syn_linecont_ic = curwin->w_s->b_syn_ic;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005709
5710 /* Make 'cpoptions' empty, to avoid the 'l' flag */
5711 cpo_save = p_cpo;
5712 p_cpo = (char_u *)"";
Bram Moolenaar860cae12010-06-05 23:22:07 +02005713 curwin->w_s->b_syn_linecont_prog =
5714 vim_regcomp(curwin->w_s->b_syn_linecont_pat, RE_MAGIC);
Bram Moolenaar071d4272004-06-13 20:20:40 +00005715 p_cpo = cpo_save;
5716
Bram Moolenaar860cae12010-06-05 23:22:07 +02005717 if (curwin->w_s->b_syn_linecont_prog == NULL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00005718 {
Bram Moolenaar860cae12010-06-05 23:22:07 +02005719 vim_free(curwin->w_s->b_syn_linecont_pat);
5720 curwin->w_s->b_syn_linecont_pat = NULL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00005721 finished = TRUE;
5722 break;
5723 }
5724 }
5725 next_arg = skipwhite(arg_end + 1);
5726 }
5727 else
5728 {
5729 eap->arg = next_arg;
5730 if (STRCMP(key, "MATCH") == 0)
5731 syn_cmd_match(eap, TRUE);
5732 else if (STRCMP(key, "REGION") == 0)
5733 syn_cmd_region(eap, TRUE);
5734 else if (STRCMP(key, "CLEAR") == 0)
5735 syn_cmd_clear(eap, TRUE);
5736 else
5737 illegal = TRUE;
5738 finished = TRUE;
5739 break;
5740 }
5741 arg_start = next_arg;
5742 }
5743 vim_free(key);
5744 if (illegal)
5745 EMSG2(_("E404: Illegal arguments: %s"), arg_start);
5746 else if (!finished)
5747 {
5748 eap->nextcmd = check_nextcmd(arg_start);
Bram Moolenaar1c8f93f2006-03-12 22:10:07 +00005749 redraw_curbuf_later(SOME_VALID);
Bram Moolenaar860cae12010-06-05 23:22:07 +02005750 syn_stack_free_all(curwin->w_s); /* Need to recompute all syntax. */
Bram Moolenaar071d4272004-06-13 20:20:40 +00005751 }
5752}
5753
5754/*
5755 * Convert a line of highlight group names into a list of group ID numbers.
5756 * "arg" should point to the "contains" or "nextgroup" keyword.
5757 * "arg" is advanced to after the last group name.
5758 * Careful: the argument is modified (NULs added).
5759 * returns FAIL for some error, OK for success.
5760 */
5761 static int
5762get_id_list(arg, keylen, list)
5763 char_u **arg;
5764 int keylen; /* length of keyword */
5765 short **list; /* where to store the resulting list, if not
5766 NULL, the list is silently skipped! */
5767{
5768 char_u *p = NULL;
5769 char_u *end;
5770 int round;
5771 int count;
5772 int total_count = 0;
5773 short *retval = NULL;
5774 char_u *name;
5775 regmatch_T regmatch;
5776 int id;
5777 int i;
5778 int failed = FALSE;
5779
5780 /*
5781 * We parse the list twice:
5782 * round == 1: count the number of items, allocate the array.
5783 * round == 2: fill the array with the items.
5784 * In round 1 new groups may be added, causing the number of items to
5785 * grow when a regexp is used. In that case round 1 is done once again.
5786 */
5787 for (round = 1; round <= 2; ++round)
5788 {
5789 /*
5790 * skip "contains"
5791 */
5792 p = skipwhite(*arg + keylen);
5793 if (*p != '=')
5794 {
5795 EMSG2(_("E405: Missing equal sign: %s"), *arg);
5796 break;
5797 }
5798 p = skipwhite(p + 1);
5799 if (ends_excmd(*p))
5800 {
5801 EMSG2(_("E406: Empty argument: %s"), *arg);
5802 break;
5803 }
5804
5805 /*
5806 * parse the arguments after "contains"
5807 */
5808 count = 0;
5809 while (!ends_excmd(*p))
5810 {
5811 for (end = p; *end && !vim_iswhite(*end) && *end != ','; ++end)
5812 ;
5813 name = alloc((int)(end - p + 3)); /* leave room for "^$" */
5814 if (name == NULL)
5815 {
5816 failed = TRUE;
5817 break;
5818 }
Bram Moolenaarce0842a2005-07-18 21:58:11 +00005819 vim_strncpy(name + 1, p, end - p);
Bram Moolenaar071d4272004-06-13 20:20:40 +00005820 if ( STRCMP(name + 1, "ALLBUT") == 0
5821 || STRCMP(name + 1, "ALL") == 0
5822 || STRCMP(name + 1, "TOP") == 0
5823 || STRCMP(name + 1, "CONTAINED") == 0)
5824 {
5825 if (TOUPPER_ASC(**arg) != 'C')
5826 {
5827 EMSG2(_("E407: %s not allowed here"), name + 1);
5828 failed = TRUE;
5829 vim_free(name);
5830 break;
5831 }
5832 if (count != 0)
5833 {
5834 EMSG2(_("E408: %s must be first in contains list"), name + 1);
5835 failed = TRUE;
5836 vim_free(name);
5837 break;
5838 }
5839 if (name[1] == 'A')
5840 id = SYNID_ALLBUT;
5841 else if (name[1] == 'T')
5842 id = SYNID_TOP;
5843 else
5844 id = SYNID_CONTAINED;
5845 id += current_syn_inc_tag;
5846 }
5847 else if (name[1] == '@')
5848 {
5849 id = syn_check_cluster(name + 2, (int)(end - p - 1));
5850 }
5851 else
5852 {
5853 /*
5854 * Handle full group name.
5855 */
5856 if (vim_strpbrk(name + 1, (char_u *)"\\.*^$~[") == NULL)
5857 id = syn_check_group(name + 1, (int)(end - p));
5858 else
5859 {
5860 /*
5861 * Handle match of regexp with group names.
5862 */
5863 *name = '^';
5864 STRCAT(name, "$");
5865 regmatch.regprog = vim_regcomp(name, RE_MAGIC);
5866 if (regmatch.regprog == NULL)
5867 {
5868 failed = TRUE;
5869 vim_free(name);
5870 break;
5871 }
5872
5873 regmatch.rm_ic = TRUE;
5874 id = 0;
5875 for (i = highlight_ga.ga_len; --i >= 0; )
5876 {
5877 if (vim_regexec(&regmatch, HL_TABLE()[i].sg_name,
5878 (colnr_T)0))
5879 {
5880 if (round == 2)
5881 {
5882 /* Got more items than expected; can happen
5883 * when adding items that match:
5884 * "contains=a.*b,axb".
5885 * Go back to first round */
5886 if (count >= total_count)
5887 {
5888 vim_free(retval);
5889 round = 1;
5890 }
5891 else
5892 retval[count] = i + 1;
5893 }
5894 ++count;
5895 id = -1; /* remember that we found one */
5896 }
5897 }
5898 vim_free(regmatch.regprog);
5899 }
5900 }
5901 vim_free(name);
5902 if (id == 0)
5903 {
5904 EMSG2(_("E409: Unknown group name: %s"), p);
5905 failed = TRUE;
5906 break;
5907 }
5908 if (id > 0)
5909 {
5910 if (round == 2)
5911 {
5912 /* Got more items than expected, go back to first round */
5913 if (count >= total_count)
5914 {
5915 vim_free(retval);
5916 round = 1;
5917 }
5918 else
5919 retval[count] = id;
5920 }
5921 ++count;
5922 }
5923 p = skipwhite(end);
5924 if (*p != ',')
5925 break;
5926 p = skipwhite(p + 1); /* skip comma in between arguments */
5927 }
5928 if (failed)
5929 break;
5930 if (round == 1)
5931 {
5932 retval = (short *)alloc((unsigned)((count + 1) * sizeof(short)));
5933 if (retval == NULL)
5934 break;
5935 retval[count] = 0; /* zero means end of the list */
5936 total_count = count;
5937 }
5938 }
5939
5940 *arg = p;
5941 if (failed || retval == NULL)
5942 {
5943 vim_free(retval);
5944 return FAIL;
5945 }
5946
5947 if (*list == NULL)
5948 *list = retval;
5949 else
5950 vim_free(retval); /* list already found, don't overwrite it */
5951
5952 return OK;
5953}
5954
5955/*
5956 * Make a copy of an ID list.
5957 */
5958 static short *
5959copy_id_list(list)
5960 short *list;
5961{
5962 int len;
5963 int count;
5964 short *retval;
5965
5966 if (list == NULL)
5967 return NULL;
5968
5969 for (count = 0; list[count]; ++count)
5970 ;
5971 len = (count + 1) * sizeof(short);
5972 retval = (short *)alloc((unsigned)len);
5973 if (retval != NULL)
5974 mch_memmove(retval, list, (size_t)len);
5975
5976 return retval;
5977}
5978
5979/*
5980 * Check if syntax group "ssp" is in the ID list "list" of "cur_si".
5981 * "cur_si" can be NULL if not checking the "containedin" list.
5982 * Used to check if a syntax item is in the "contains" or "nextgroup" list of
5983 * the current item.
5984 * This function is called very often, keep it fast!!
5985 */
5986 static int
5987in_id_list(cur_si, list, ssp, contained)
5988 stateitem_T *cur_si; /* current item or NULL */
5989 short *list; /* id list */
5990 struct sp_syn *ssp; /* group id and ":syn include" tag of group */
5991 int contained; /* group id is contained */
5992{
5993 int retval;
5994 short *scl_list;
5995 short item;
5996 short id = ssp->id;
5997 static int depth = 0;
5998 int r;
5999
6000 /* If spp has a "containedin" list and "cur_si" is in it, return TRUE. */
Bram Moolenaar293ee4d2004-12-09 21:34:53 +00006001 if (cur_si != NULL && ssp->cont_in_list != NULL
6002 && !(cur_si->si_flags & HL_MATCH))
Bram Moolenaar071d4272004-06-13 20:20:40 +00006003 {
6004 /* Ignore transparent items without a contains argument. Double check
6005 * that we don't go back past the first one. */
6006 while ((cur_si->si_flags & HL_TRANS_CONT)
6007 && cur_si > (stateitem_T *)(current_state.ga_data))
6008 --cur_si;
6009 /* cur_si->si_idx is -1 for keywords, these never contain anything. */
6010 if (cur_si->si_idx >= 0 && in_id_list(NULL, ssp->cont_in_list,
Bram Moolenaar860cae12010-06-05 23:22:07 +02006011 &(SYN_ITEMS(syn_block)[cur_si->si_idx].sp_syn),
6012 SYN_ITEMS(syn_block)[cur_si->si_idx].sp_flags & HL_CONTAINED))
Bram Moolenaar071d4272004-06-13 20:20:40 +00006013 return TRUE;
6014 }
6015
6016 if (list == NULL)
6017 return FALSE;
6018
6019 /*
6020 * If list is ID_LIST_ALL, we are in a transparent item that isn't
6021 * inside anything. Only allow not-contained groups.
6022 */
6023 if (list == ID_LIST_ALL)
6024 return !contained;
6025
6026 /*
6027 * If the first item is "ALLBUT", return TRUE if "id" is NOT in the
6028 * contains list. We also require that "id" is at the same ":syn include"
6029 * level as the list.
6030 */
6031 item = *list;
6032 if (item >= SYNID_ALLBUT && item < SYNID_CLUSTER)
6033 {
6034 if (item < SYNID_TOP)
6035 {
6036 /* ALL or ALLBUT: accept all groups in the same file */
6037 if (item - SYNID_ALLBUT != ssp->inc_tag)
6038 return FALSE;
6039 }
6040 else if (item < SYNID_CONTAINED)
6041 {
6042 /* TOP: accept all not-contained groups in the same file */
6043 if (item - SYNID_TOP != ssp->inc_tag || contained)
6044 return FALSE;
6045 }
6046 else
6047 {
6048 /* CONTAINED: accept all contained groups in the same file */
6049 if (item - SYNID_CONTAINED != ssp->inc_tag || !contained)
6050 return FALSE;
6051 }
6052 item = *++list;
6053 retval = FALSE;
6054 }
6055 else
6056 retval = TRUE;
6057
6058 /*
6059 * Return "retval" if id is in the contains list.
6060 */
6061 while (item != 0)
6062 {
6063 if (item == id)
6064 return retval;
6065 if (item >= SYNID_CLUSTER)
6066 {
Bram Moolenaar860cae12010-06-05 23:22:07 +02006067 scl_list = SYN_CLSTR(syn_block)[item - SYNID_CLUSTER].scl_list;
Bram Moolenaar071d4272004-06-13 20:20:40 +00006068 /* restrict recursiveness to 30 to avoid an endless loop for a
6069 * cluster that includes itself (indirectly) */
6070 if (scl_list != NULL && depth < 30)
6071 {
6072 ++depth;
6073 r = in_id_list(NULL, scl_list, ssp, contained);
6074 --depth;
6075 if (r)
6076 return retval;
6077 }
6078 }
6079 item = *++list;
6080 }
6081 return !retval;
6082}
6083
6084struct subcommand
6085{
6086 char *name; /* subcommand name */
6087 void (*func)__ARGS((exarg_T *, int)); /* function to call */
6088};
6089
6090static struct subcommand subcommands[] =
6091{
6092 {"case", syn_cmd_case},
6093 {"clear", syn_cmd_clear},
6094 {"cluster", syn_cmd_cluster},
Bram Moolenaar860cae12010-06-05 23:22:07 +02006095 {"conceal", syn_cmd_conceal},
Bram Moolenaar071d4272004-06-13 20:20:40 +00006096 {"enable", syn_cmd_enable},
6097 {"include", syn_cmd_include},
6098 {"keyword", syn_cmd_keyword},
6099 {"list", syn_cmd_list},
6100 {"manual", syn_cmd_manual},
6101 {"match", syn_cmd_match},
6102 {"on", syn_cmd_on},
6103 {"off", syn_cmd_off},
6104 {"region", syn_cmd_region},
6105 {"reset", syn_cmd_reset},
Bram Moolenaarce0842a2005-07-18 21:58:11 +00006106 {"spell", syn_cmd_spell},
Bram Moolenaar071d4272004-06-13 20:20:40 +00006107 {"sync", syn_cmd_sync},
6108 {"", syn_cmd_list},
6109 {NULL, NULL}
6110};
6111
6112/*
6113 * ":syntax".
6114 * This searches the subcommands[] table for the subcommand name, and calls a
6115 * syntax_subcommand() function to do the rest.
6116 */
6117 void
6118ex_syntax(eap)
6119 exarg_T *eap;
6120{
6121 char_u *arg = eap->arg;
6122 char_u *subcmd_end;
6123 char_u *subcmd_name;
6124 int i;
6125
6126 syn_cmdlinep = eap->cmdlinep;
6127
6128 /* isolate subcommand name */
6129 for (subcmd_end = arg; ASCII_ISALPHA(*subcmd_end); ++subcmd_end)
6130 ;
6131 subcmd_name = vim_strnsave(arg, (int)(subcmd_end - arg));
6132 if (subcmd_name != NULL)
6133 {
6134 if (eap->skip) /* skip error messages for all subcommands */
6135 ++emsg_skip;
6136 for (i = 0; ; ++i)
6137 {
6138 if (subcommands[i].name == NULL)
6139 {
6140 EMSG2(_("E410: Invalid :syntax subcommand: %s"), subcmd_name);
6141 break;
6142 }
6143 if (STRCMP(subcmd_name, (char_u *)subcommands[i].name) == 0)
6144 {
6145 eap->arg = skipwhite(subcmd_end);
6146 (subcommands[i].func)(eap, FALSE);
6147 break;
6148 }
6149 }
6150 vim_free(subcmd_name);
6151 if (eap->skip)
6152 --emsg_skip;
6153 }
6154}
6155
Bram Moolenaar860cae12010-06-05 23:22:07 +02006156 void
6157ex_ownsyntax(eap)
6158 exarg_T *eap;
Bram Moolenaar071d4272004-06-13 20:20:40 +00006159{
Bram Moolenaar860cae12010-06-05 23:22:07 +02006160 if (curwin->w_s == &curwin->w_buffer->b_s)
6161 {
6162 curwin->w_s = (synblock_T *)alloc(sizeof(synblock_T));
6163 memset(curwin->w_s, 0, sizeof(synblock_T));
6164#ifdef FEAT_SPELL
6165 curwin->w_p_spell = FALSE; /* No spell checking */
6166 clear_string_option(&curwin->w_s->b_p_spc);
6167 clear_string_option(&curwin->w_s->b_p_spf);
6168 vim_free(curwin->w_s->b_cap_prog);
6169 curwin->w_s->b_cap_prog = NULL;
6170 clear_string_option(&curwin->w_s->b_p_spl);
6171#endif
6172 }
6173 apply_autocmds(EVENT_SYNTAX, eap->arg, curbuf->b_fname, TRUE, curbuf);
6174}
6175
6176 int
6177syntax_present(win)
6178 win_T *win;
6179{
6180 return (win->w_s->b_syn_patterns.ga_len != 0
6181 || win->w_s->b_syn_clusters.ga_len != 0
6182 || win->w_s->b_keywtab.ht_used > 0
6183 || win->w_s->b_keywtab_ic.ht_used > 0);
Bram Moolenaar071d4272004-06-13 20:20:40 +00006184}
6185
6186#if defined(FEAT_CMDL_COMPL) || defined(PROTO)
6187
6188static enum
6189{
6190 EXP_SUBCMD, /* expand ":syn" sub-commands */
6191 EXP_CASE /* expand ":syn case" arguments */
6192} expand_what;
6193
Bram Moolenaar4f688582007-07-24 12:34:30 +00006194/*
6195 * Reset include_link, include_default, include_none to 0.
6196 * Called when we are done expanding.
6197 */
6198 void
6199reset_expand_highlight()
6200{
6201 include_link = include_default = include_none = 0;
6202}
6203
6204/*
6205 * Handle command line completion for :match and :echohl command: Add "None"
6206 * as highlight group.
6207 */
6208 void
6209set_context_in_echohl_cmd(xp, arg)
6210 expand_T *xp;
6211 char_u *arg;
6212{
6213 xp->xp_context = EXPAND_HIGHLIGHT;
6214 xp->xp_pattern = arg;
6215 include_none = 1;
6216}
Bram Moolenaar071d4272004-06-13 20:20:40 +00006217
6218/*
6219 * Handle command line completion for :syntax command.
6220 */
6221 void
6222set_context_in_syntax_cmd(xp, arg)
6223 expand_T *xp;
6224 char_u *arg;
6225{
6226 char_u *p;
6227
6228 /* Default: expand subcommands */
6229 xp->xp_context = EXPAND_SYNTAX;
6230 expand_what = EXP_SUBCMD;
6231 xp->xp_pattern = arg;
Bram Moolenaar4f688582007-07-24 12:34:30 +00006232 include_link = 0;
6233 include_default = 0;
Bram Moolenaar071d4272004-06-13 20:20:40 +00006234
6235 /* (part of) subcommand already typed */
6236 if (*arg != NUL)
6237 {
6238 p = skiptowhite(arg);
6239 if (*p != NUL) /* past first word */
6240 {
6241 xp->xp_pattern = skipwhite(p);
6242 if (*skiptowhite(xp->xp_pattern) != NUL)
6243 xp->xp_context = EXPAND_NOTHING;
6244 else if (STRNICMP(arg, "case", p - arg) == 0)
6245 expand_what = EXP_CASE;
6246 else if ( STRNICMP(arg, "keyword", p - arg) == 0
6247 || STRNICMP(arg, "region", p - arg) == 0
6248 || STRNICMP(arg, "match", p - arg) == 0
6249 || STRNICMP(arg, "list", p - arg) == 0)
6250 xp->xp_context = EXPAND_HIGHLIGHT;
6251 else
6252 xp->xp_context = EXPAND_NOTHING;
6253 }
6254 }
6255}
6256
6257static char *(case_args[]) = {"match", "ignore", NULL};
6258
6259/*
6260 * Function given to ExpandGeneric() to obtain the list syntax names for
6261 * expansion.
6262 */
Bram Moolenaar071d4272004-06-13 20:20:40 +00006263 char_u *
6264get_syntax_name(xp, idx)
Bram Moolenaar2c4278f2009-05-17 11:33:22 +00006265 expand_T *xp UNUSED;
Bram Moolenaar071d4272004-06-13 20:20:40 +00006266 int idx;
6267{
6268 if (expand_what == EXP_SUBCMD)
6269 return (char_u *)subcommands[idx].name;
6270 return (char_u *)case_args[idx];
6271}
6272
6273#endif /* FEAT_CMDL_COMPL */
6274
Bram Moolenaar071d4272004-06-13 20:20:40 +00006275/*
6276 * Function called for expression evaluation: get syntax ID at file position.
6277 */
6278 int
Bram Moolenaar56cefaf2008-01-12 15:47:10 +00006279syn_get_id(wp, lnum, col, trans, spellp, keep_state)
Bram Moolenaar81f1ecb2005-08-25 21:27:31 +00006280 win_T *wp;
Bram Moolenaar071d4272004-06-13 20:20:40 +00006281 long lnum;
Bram Moolenaar9d0ec2e2005-04-20 19:45:58 +00006282 colnr_T col;
Bram Moolenaarf5b63862009-12-16 17:13:44 +00006283 int trans; /* remove transparency */
Bram Moolenaar56cefaf2008-01-12 15:47:10 +00006284 int *spellp; /* return: can do spell checking */
6285 int keep_state; /* keep state of char at "col" */
Bram Moolenaar071d4272004-06-13 20:20:40 +00006286{
6287 /* When the position is not after the current position and in the same
6288 * line of the same buffer, need to restart parsing. */
Bram Moolenaar81f1ecb2005-08-25 21:27:31 +00006289 if (wp->w_buffer != syn_buf
Bram Moolenaar071d4272004-06-13 20:20:40 +00006290 || lnum != current_lnum
Bram Moolenaar9d0ec2e2005-04-20 19:45:58 +00006291 || col < current_col)
Bram Moolenaar81f1ecb2005-08-25 21:27:31 +00006292 syntax_start(wp, lnum);
Bram Moolenaar071d4272004-06-13 20:20:40 +00006293
Bram Moolenaar860cae12010-06-05 23:22:07 +02006294 (void)get_syntax_attr(col, NULL, spellp, keep_state);
Bram Moolenaar071d4272004-06-13 20:20:40 +00006295
6296 return (trans ? current_trans_id : current_id);
6297}
Bram Moolenaar071d4272004-06-13 20:20:40 +00006298
Bram Moolenaar860cae12010-06-05 23:22:07 +02006299#if defined(FEAT_CONCEAL) || defined(PROTO)
6300/*
6301 * Return conceal substitution character
6302 */
6303 int
6304syn_get_sub_char()
6305{
6306 return current_sub_char;
6307}
6308#endif
6309
Bram Moolenaar9d188ab2008-01-10 21:24:39 +00006310#if defined(FEAT_EVAL) || defined(PROTO)
6311/*
6312 * Return the syntax ID at position "i" in the current stack.
6313 * The caller must have called syn_get_id() before to fill the stack.
6314 * Returns -1 when "i" is out of range.
6315 */
6316 int
6317syn_get_stack_item(i)
6318 int i;
6319{
Bram Moolenaar56cefaf2008-01-12 15:47:10 +00006320 if (i >= current_state.ga_len)
6321 {
6322 /* Need to invalidate the state, because we didn't properly finish it
6323 * for the last character, "keep_state" was TRUE. */
6324 invalidate_current_state();
6325 current_col = MAXCOL;
Bram Moolenaar9d188ab2008-01-10 21:24:39 +00006326 return -1;
Bram Moolenaar56cefaf2008-01-12 15:47:10 +00006327 }
Bram Moolenaar9d188ab2008-01-10 21:24:39 +00006328 return CUR_STATE(i).si_id;
6329}
6330#endif
6331
Bram Moolenaar071d4272004-06-13 20:20:40 +00006332#if defined(FEAT_FOLDING) || defined(PROTO)
6333/*
6334 * Function called to get folding level for line "lnum" in window "wp".
6335 */
6336 int
6337syn_get_foldlevel(wp, lnum)
6338 win_T *wp;
6339 long lnum;
6340{
6341 int level = 0;
6342 int i;
6343
6344 /* Return quickly when there are no fold items at all. */
Bram Moolenaar860cae12010-06-05 23:22:07 +02006345 if (wp->w_s->b_syn_folditems != 0)
Bram Moolenaar071d4272004-06-13 20:20:40 +00006346 {
6347 syntax_start(wp, lnum);
6348
6349 for (i = 0; i < current_state.ga_len; ++i)
6350 if (CUR_STATE(i).si_flags & HL_FOLD)
6351 ++level;
6352 }
6353 if (level > wp->w_p_fdn)
Bram Moolenaar74c596b2006-11-01 11:44:31 +00006354 {
Bram Moolenaar071d4272004-06-13 20:20:40 +00006355 level = wp->w_p_fdn;
Bram Moolenaar74c596b2006-11-01 11:44:31 +00006356 if (level < 0)
6357 level = 0;
6358 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00006359 return level;
6360}
6361#endif
6362
6363#endif /* FEAT_SYN_HL */
6364
6365
6366/**************************************
6367 * Highlighting stuff *
6368 **************************************/
6369
6370/*
6371 * The default highlight groups. These are compiled-in for fast startup and
6372 * they still work when the runtime files can't be found.
6373 * When making changes here, also change runtime/colors/default.vim!
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006374 * The #ifdefs are needed to reduce the amount of static data. Helps to make
6375 * the 16 bit DOS (museum) version compile.
Bram Moolenaar071d4272004-06-13 20:20:40 +00006376 */
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006377#ifdef FEAT_GUI
6378# define CENT(a, b) b
6379#else
6380# define CENT(a, b) a
6381#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00006382static char *(highlight_init_both[]) =
6383 {
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006384 CENT("ErrorMsg term=standout ctermbg=DarkRed ctermfg=White",
6385 "ErrorMsg term=standout ctermbg=DarkRed ctermfg=White guibg=Red guifg=White"),
6386 CENT("IncSearch term=reverse cterm=reverse",
6387 "IncSearch term=reverse cterm=reverse gui=reverse"),
6388 CENT("ModeMsg term=bold cterm=bold",
6389 "ModeMsg term=bold cterm=bold gui=bold"),
6390 CENT("NonText term=bold ctermfg=Blue",
6391 "NonText term=bold ctermfg=Blue gui=bold guifg=Blue"),
6392 CENT("StatusLine term=reverse,bold cterm=reverse,bold",
6393 "StatusLine term=reverse,bold cterm=reverse,bold gui=reverse,bold"),
6394 CENT("StatusLineNC term=reverse cterm=reverse",
6395 "StatusLineNC term=reverse cterm=reverse gui=reverse"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006396#ifdef FEAT_VERTSPLIT
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006397 CENT("VertSplit term=reverse cterm=reverse",
6398 "VertSplit term=reverse cterm=reverse gui=reverse"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006399#endif
6400#ifdef FEAT_CLIPBOARD
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006401 CENT("VisualNOS term=underline,bold cterm=underline,bold",
6402 "VisualNOS term=underline,bold cterm=underline,bold gui=underline,bold"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006403#endif
6404#ifdef FEAT_DIFF
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006405 CENT("DiffText term=reverse cterm=bold ctermbg=Red",
6406 "DiffText term=reverse cterm=bold ctermbg=Red gui=bold guibg=Red"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006407#endif
6408#ifdef FEAT_INS_EXPAND
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006409 CENT("PmenuThumb cterm=reverse",
6410 "PmenuThumb cterm=reverse gui=reverse"),
6411 CENT("PmenuSbar ctermbg=Grey",
6412 "PmenuSbar ctermbg=Grey guibg=Grey"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006413#endif
6414#ifdef FEAT_WINDOWS
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006415 CENT("TabLineSel term=bold cterm=bold",
6416 "TabLineSel term=bold cterm=bold gui=bold"),
6417 CENT("TabLineFill term=reverse cterm=reverse",
6418 "TabLineFill term=reverse cterm=reverse gui=reverse"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006419#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00006420#ifdef FEAT_GUI
6421 "Cursor guibg=fg guifg=bg",
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006422 "lCursor guibg=fg guifg=bg", /* should be different, but what? */
Bram Moolenaar071d4272004-06-13 20:20:40 +00006423#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00006424 NULL
6425 };
6426
6427static char *(highlight_init_light[]) =
6428 {
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006429 CENT("Directory term=bold ctermfg=DarkBlue",
6430 "Directory term=bold ctermfg=DarkBlue guifg=Blue"),
6431 CENT("LineNr term=underline ctermfg=Brown",
6432 "LineNr term=underline ctermfg=Brown guifg=Brown"),
6433 CENT("MoreMsg term=bold ctermfg=DarkGreen",
6434 "MoreMsg term=bold ctermfg=DarkGreen gui=bold guifg=SeaGreen"),
6435 CENT("Question term=standout ctermfg=DarkGreen",
6436 "Question term=standout ctermfg=DarkGreen gui=bold guifg=SeaGreen"),
6437 CENT("Search term=reverse ctermbg=Yellow ctermfg=NONE",
6438 "Search term=reverse ctermbg=Yellow ctermfg=NONE guibg=Yellow guifg=NONE"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006439#ifdef FEAT_SPELL
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006440 CENT("SpellBad term=reverse ctermbg=LightRed",
6441 "SpellBad term=reverse ctermbg=LightRed guisp=Red gui=undercurl"),
6442 CENT("SpellCap term=reverse ctermbg=LightBlue",
6443 "SpellCap term=reverse ctermbg=LightBlue guisp=Blue gui=undercurl"),
6444 CENT("SpellRare term=reverse ctermbg=LightMagenta",
6445 "SpellRare term=reverse ctermbg=LightMagenta guisp=Magenta gui=undercurl"),
6446 CENT("SpellLocal term=underline ctermbg=Cyan",
6447 "SpellLocal term=underline ctermbg=Cyan guisp=DarkCyan gui=undercurl"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006448#endif
6449#ifdef FEAT_INS_EXPAND
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006450 CENT("Pmenu ctermbg=LightMagenta",
6451 "Pmenu ctermbg=LightMagenta guibg=LightMagenta"),
6452 CENT("PmenuSel ctermbg=LightGrey",
6453 "PmenuSel ctermbg=LightGrey guibg=Grey"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006454#endif
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006455 CENT("SpecialKey term=bold ctermfg=DarkBlue",
6456 "SpecialKey term=bold ctermfg=DarkBlue guifg=Blue"),
6457 CENT("Title term=bold ctermfg=DarkMagenta",
6458 "Title term=bold ctermfg=DarkMagenta gui=bold guifg=Magenta"),
6459 CENT("WarningMsg term=standout ctermfg=DarkRed",
6460 "WarningMsg term=standout ctermfg=DarkRed guifg=Red"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006461#ifdef FEAT_WILDMENU
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006462 CENT("WildMenu term=standout ctermbg=Yellow ctermfg=Black",
6463 "WildMenu term=standout ctermbg=Yellow ctermfg=Black guibg=Yellow guifg=Black"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006464#endif
6465#ifdef FEAT_FOLDING
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006466 CENT("Folded term=standout ctermbg=Grey ctermfg=DarkBlue",
6467 "Folded term=standout ctermbg=Grey ctermfg=DarkBlue guibg=LightGrey guifg=DarkBlue"),
6468 CENT("FoldColumn term=standout ctermbg=Grey ctermfg=DarkBlue",
6469 "FoldColumn term=standout ctermbg=Grey ctermfg=DarkBlue guibg=Grey guifg=DarkBlue"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006470#endif
6471#ifdef FEAT_SIGNS
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006472 CENT("SignColumn term=standout ctermbg=Grey ctermfg=DarkBlue",
6473 "SignColumn term=standout ctermbg=Grey ctermfg=DarkBlue guibg=Grey guifg=DarkBlue"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006474#endif
6475#ifdef FEAT_VISUAL
Bram Moolenaarc1e37902006-04-18 21:55:01 +00006476 CENT("Visual term=reverse",
6477 "Visual term=reverse guibg=LightGrey"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006478#endif
6479#ifdef FEAT_DIFF
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006480 CENT("DiffAdd term=bold ctermbg=LightBlue",
6481 "DiffAdd term=bold ctermbg=LightBlue guibg=LightBlue"),
6482 CENT("DiffChange term=bold ctermbg=LightMagenta",
6483 "DiffChange term=bold ctermbg=LightMagenta guibg=LightMagenta"),
6484 CENT("DiffDelete term=bold ctermfg=Blue ctermbg=LightCyan",
6485 "DiffDelete term=bold ctermfg=Blue ctermbg=LightCyan gui=bold guifg=Blue guibg=LightCyan"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006486#endif
6487#ifdef FEAT_WINDOWS
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006488 CENT("TabLine term=underline cterm=underline ctermfg=black ctermbg=LightGrey",
6489 "TabLine term=underline cterm=underline ctermfg=black ctermbg=LightGrey gui=underline guibg=LightGrey"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006490#endif
6491#ifdef FEAT_SYN_HL
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006492 CENT("CursorColumn term=reverse ctermbg=LightGrey",
Bram Moolenaare2f98b92006-03-29 21:18:24 +00006493 "CursorColumn term=reverse ctermbg=LightGrey guibg=Grey90"),
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006494 CENT("CursorLine term=underline cterm=underline",
Bram Moolenaare2f98b92006-03-29 21:18:24 +00006495 "CursorLine term=underline cterm=underline guibg=Grey90"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006496#endif
Bram Moolenaar860cae12010-06-05 23:22:07 +02006497#ifdef FEAT_CONCEAL
6498 CENT("Conceal ctermbg=DarkGrey ctermfg=LightGrey",
6499 "Conceal ctermbg=DarkGrey ctermfg=LightGrey guibg=DarkGrey guifg=LightGrey"),
6500#endif
Bram Moolenaarb21e5842006-04-16 18:30:08 +00006501#ifdef FEAT_AUTOCMD
6502 CENT("MatchParen term=reverse ctermbg=Cyan",
6503 "MatchParen term=reverse ctermbg=Cyan guibg=Cyan"),
6504#endif
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006505#ifdef FEAT_GUI
Bram Moolenaar071d4272004-06-13 20:20:40 +00006506 "Normal gui=NONE",
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006507#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00006508 NULL
6509 };
6510
6511static char *(highlight_init_dark[]) =
6512 {
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006513 CENT("Directory term=bold ctermfg=LightCyan",
6514 "Directory term=bold ctermfg=LightCyan guifg=Cyan"),
6515 CENT("LineNr term=underline ctermfg=Yellow",
6516 "LineNr term=underline ctermfg=Yellow guifg=Yellow"),
6517 CENT("MoreMsg term=bold ctermfg=LightGreen",
6518 "MoreMsg term=bold ctermfg=LightGreen gui=bold guifg=SeaGreen"),
6519 CENT("Question term=standout ctermfg=LightGreen",
6520 "Question term=standout ctermfg=LightGreen gui=bold guifg=Green"),
6521 CENT("Search term=reverse ctermbg=Yellow ctermfg=Black",
6522 "Search term=reverse ctermbg=Yellow ctermfg=Black guibg=Yellow guifg=Black"),
6523 CENT("SpecialKey term=bold ctermfg=LightBlue",
6524 "SpecialKey term=bold ctermfg=LightBlue guifg=Cyan"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006525#ifdef FEAT_SPELL
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006526 CENT("SpellBad term=reverse ctermbg=Red",
6527 "SpellBad term=reverse ctermbg=Red guisp=Red gui=undercurl"),
6528 CENT("SpellCap term=reverse ctermbg=Blue",
6529 "SpellCap term=reverse ctermbg=Blue guisp=Blue gui=undercurl"),
6530 CENT("SpellRare term=reverse ctermbg=Magenta",
6531 "SpellRare term=reverse ctermbg=Magenta guisp=Magenta gui=undercurl"),
6532 CENT("SpellLocal term=underline ctermbg=Cyan",
6533 "SpellLocal term=underline ctermbg=Cyan guisp=Cyan gui=undercurl"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006534#endif
6535#ifdef FEAT_INS_EXPAND
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006536 CENT("Pmenu ctermbg=Magenta",
6537 "Pmenu ctermbg=Magenta guibg=Magenta"),
6538 CENT("PmenuSel ctermbg=DarkGrey",
6539 "PmenuSel ctermbg=DarkGrey guibg=DarkGrey"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006540#endif
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006541 CENT("Title term=bold ctermfg=LightMagenta",
6542 "Title term=bold ctermfg=LightMagenta gui=bold guifg=Magenta"),
6543 CENT("WarningMsg term=standout ctermfg=LightRed",
6544 "WarningMsg term=standout ctermfg=LightRed guifg=Red"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006545#ifdef FEAT_WILDMENU
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006546 CENT("WildMenu term=standout ctermbg=Yellow ctermfg=Black",
6547 "WildMenu term=standout ctermbg=Yellow ctermfg=Black guibg=Yellow guifg=Black"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006548#endif
6549#ifdef FEAT_FOLDING
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006550 CENT("Folded term=standout ctermbg=DarkGrey ctermfg=Cyan",
6551 "Folded term=standout ctermbg=DarkGrey ctermfg=Cyan guibg=DarkGrey guifg=Cyan"),
6552 CENT("FoldColumn term=standout ctermbg=DarkGrey ctermfg=Cyan",
6553 "FoldColumn term=standout ctermbg=DarkGrey ctermfg=Cyan guibg=Grey guifg=Cyan"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006554#endif
6555#ifdef FEAT_SIGNS
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006556 CENT("SignColumn term=standout ctermbg=DarkGrey ctermfg=Cyan",
6557 "SignColumn term=standout ctermbg=DarkGrey ctermfg=Cyan guibg=Grey guifg=Cyan"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006558#endif
6559#ifdef FEAT_VISUAL
Bram Moolenaarc1e37902006-04-18 21:55:01 +00006560 CENT("Visual term=reverse",
6561 "Visual term=reverse guibg=DarkGrey"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006562#endif
6563#ifdef FEAT_DIFF
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006564 CENT("DiffAdd term=bold ctermbg=DarkBlue",
6565 "DiffAdd term=bold ctermbg=DarkBlue guibg=DarkBlue"),
6566 CENT("DiffChange term=bold ctermbg=DarkMagenta",
6567 "DiffChange term=bold ctermbg=DarkMagenta guibg=DarkMagenta"),
6568 CENT("DiffDelete term=bold ctermfg=Blue ctermbg=DarkCyan",
6569 "DiffDelete term=bold ctermfg=Blue ctermbg=DarkCyan gui=bold guifg=Blue guibg=DarkCyan"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006570#endif
6571#ifdef FEAT_WINDOWS
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006572 CENT("TabLine term=underline cterm=underline ctermfg=white ctermbg=DarkGrey",
6573 "TabLine term=underline cterm=underline ctermfg=white ctermbg=DarkGrey gui=underline guibg=DarkGrey"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006574#endif
6575#ifdef FEAT_SYN_HL
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006576 CENT("CursorColumn term=reverse ctermbg=DarkGrey",
Bram Moolenaarb21e5842006-04-16 18:30:08 +00006577 "CursorColumn term=reverse ctermbg=DarkGrey guibg=Grey40"),
Bram Moolenaar899dddf2006-03-26 21:06:50 +00006578 CENT("CursorLine term=underline cterm=underline",
Bram Moolenaarb21e5842006-04-16 18:30:08 +00006579 "CursorLine term=underline cterm=underline guibg=Grey40"),
6580#endif
6581#ifdef FEAT_AUTOCMD
6582 CENT("MatchParen term=reverse ctermbg=DarkCyan",
6583 "MatchParen term=reverse ctermbg=DarkCyan guibg=DarkCyan"),
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006584#endif
Bram Moolenaar860cae12010-06-05 23:22:07 +02006585#ifdef FEAT_CONCEAL
6586 CENT("Conceal ctermbg=DarkGrey ctermfg=LightGrey",
6587 "Conceal ctermbg=DarkGrey ctermfg=LightGrey guibg=DarkGrey guifg=LightGrey"),
6588#endif
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006589#ifdef FEAT_GUI
Bram Moolenaar071d4272004-06-13 20:20:40 +00006590 "Normal gui=NONE",
Bram Moolenaar42bbef42006-03-25 22:02:07 +00006591#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00006592 NULL
6593 };
6594
6595 void
6596init_highlight(both, reset)
6597 int both; /* include groups where 'bg' doesn't matter */
6598 int reset; /* clear group first */
6599{
6600 int i;
6601 char **pp;
6602 static int had_both = FALSE;
6603#ifdef FEAT_EVAL
6604 char_u *p;
6605
6606 /*
6607 * Try finding the color scheme file. Used when a color file was loaded
6608 * and 'background' or 't_Co' is changed.
6609 */
6610 p = get_var_value((char_u *)"g:colors_name");
6611 if (p != NULL && load_colors(p) == OK)
6612 return;
6613#endif
6614
6615 /*
6616 * Didn't use a color file, use the compiled-in colors.
6617 */
6618 if (both)
6619 {
6620 had_both = TRUE;
6621 pp = highlight_init_both;
6622 for (i = 0; pp[i] != NULL; ++i)
6623 do_highlight((char_u *)pp[i], reset, TRUE);
6624 }
6625 else if (!had_both)
6626 /* Don't do anything before the call with both == TRUE from main().
6627 * Not everything has been setup then, and that call will overrule
6628 * everything anyway. */
6629 return;
6630
6631 if (*p_bg == 'l')
6632 pp = highlight_init_light;
6633 else
6634 pp = highlight_init_dark;
6635 for (i = 0; pp[i] != NULL; ++i)
6636 do_highlight((char_u *)pp[i], reset, TRUE);
6637
Bram Moolenaarc1e37902006-04-18 21:55:01 +00006638 /* Reverse looks ugly, but grey may not work for 8 colors. Thus let it
Bram Moolenaarf193fff2006-04-27 00:02:13 +00006639 * depend on the number of colors available.
6640 * With 8 colors brown is equal to yellow, need to use black for Search fg
Bram Moolenaarda40c852008-08-06 13:28:57 +00006641 * to avoid Statement highlighted text disappears.
6642 * Clear the attributes, needed when changing the t_Co value. */
Bram Moolenaarab194812005-09-14 21:40:12 +00006643 if (t_colors > 8)
Bram Moolenaarda40c852008-08-06 13:28:57 +00006644 do_highlight((char_u *)(*p_bg == 'l'
6645 ? "Visual cterm=NONE ctermbg=LightGrey"
6646 : "Visual cterm=NONE ctermbg=DarkGrey"), FALSE, TRUE);
Bram Moolenaarc1e37902006-04-18 21:55:01 +00006647 else
Bram Moolenaarf193fff2006-04-27 00:02:13 +00006648 {
Bram Moolenaarda40c852008-08-06 13:28:57 +00006649 do_highlight((char_u *)"Visual cterm=reverse ctermbg=NONE",
6650 FALSE, TRUE);
Bram Moolenaarf193fff2006-04-27 00:02:13 +00006651 if (*p_bg == 'l')
6652 do_highlight((char_u *)"Search ctermfg=black", FALSE, TRUE);
6653 }
Bram Moolenaarab194812005-09-14 21:40:12 +00006654
Bram Moolenaar071d4272004-06-13 20:20:40 +00006655#ifdef FEAT_SYN_HL
6656 /*
6657 * If syntax highlighting is enabled load the highlighting for it.
6658 */
6659 if (get_var_value((char_u *)"g:syntax_on") != NULL)
Bram Moolenaarc0197e22004-09-13 20:26:32 +00006660 {
6661 static int recursive = 0;
6662
6663 if (recursive >= 5)
6664 EMSG(_("E679: recursive loop loading syncolor.vim"));
6665 else
6666 {
6667 ++recursive;
Bram Moolenaar90cfdbe2005-08-12 19:59:19 +00006668 (void)source_runtime((char_u *)"syntax/syncolor.vim", TRUE);
Bram Moolenaarc0197e22004-09-13 20:26:32 +00006669 --recursive;
6670 }
6671 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00006672#endif
6673}
6674
6675/*
Bram Moolenaarcfbc5ee2004-07-02 15:38:35 +00006676 * Load color file "name".
Bram Moolenaar071d4272004-06-13 20:20:40 +00006677 * Return OK for success, FAIL for failure.
6678 */
6679 int
Bram Moolenaarcfbc5ee2004-07-02 15:38:35 +00006680load_colors(name)
6681 char_u *name;
Bram Moolenaar071d4272004-06-13 20:20:40 +00006682{
6683 char_u *buf;
6684 int retval = FAIL;
6685 static int recursive = FALSE;
6686
6687 /* When being called recursively, this is probably because setting
6688 * 'background' caused the highlighting to be reloaded. This means it is
6689 * working, thus we should return OK. */
6690 if (recursive)
6691 return OK;
6692
6693 recursive = TRUE;
Bram Moolenaarcfbc5ee2004-07-02 15:38:35 +00006694 buf = alloc((unsigned)(STRLEN(name) + 12));
Bram Moolenaar071d4272004-06-13 20:20:40 +00006695 if (buf != NULL)
6696 {
Bram Moolenaarcfbc5ee2004-07-02 15:38:35 +00006697 sprintf((char *)buf, "colors/%s.vim", name);
Bram Moolenaar90cfdbe2005-08-12 19:59:19 +00006698 retval = source_runtime(buf, FALSE);
Bram Moolenaar071d4272004-06-13 20:20:40 +00006699 vim_free(buf);
Bram Moolenaarcfbc5ee2004-07-02 15:38:35 +00006700#ifdef FEAT_AUTOCMD
6701 apply_autocmds(EVENT_COLORSCHEME, NULL, NULL, FALSE, curbuf);
6702#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00006703 }
6704 recursive = FALSE;
6705
6706 return retval;
6707}
6708
6709/*
6710 * Handle the ":highlight .." command.
6711 * When using ":hi clear" this is called recursively for each group with
6712 * "forceit" and "init" both TRUE.
6713 */
6714 void
6715do_highlight(line, forceit, init)
6716 char_u *line;
6717 int forceit;
6718 int init; /* TRUE when called for initializing */
6719{
6720 char_u *name_end;
6721 char_u *p;
6722 char_u *linep;
6723 char_u *key_start;
6724 char_u *arg_start;
6725 char_u *key = NULL, *arg = NULL;
6726 long i;
6727 int off;
6728 int len;
6729 int attr;
6730 int id;
6731 int idx;
6732 int dodefault = FALSE;
6733 int doclear = FALSE;
6734 int dolink = FALSE;
6735 int error = FALSE;
6736 int color;
6737 int is_normal_group = FALSE; /* "Normal" group */
6738#ifdef FEAT_GUI_X11
6739 int is_menu_group = FALSE; /* "Menu" group */
6740 int is_scrollbar_group = FALSE; /* "Scrollbar" group */
6741 int is_tooltip_group = FALSE; /* "Tooltip" group */
6742 int do_colors = FALSE; /* need to update colors? */
6743#else
6744# define is_menu_group 0
6745# define is_tooltip_group 0
6746#endif
6747
6748 /*
6749 * If no argument, list current highlighting.
6750 */
6751 if (ends_excmd(*line))
6752 {
6753 for (i = 1; i <= highlight_ga.ga_len && !got_int; ++i)
6754 /* TODO: only call when the group has attributes set */
6755 highlight_list_one((int)i);
6756 return;
6757 }
6758
6759 /*
6760 * Isolate the name.
6761 */
6762 name_end = skiptowhite(line);
6763 linep = skipwhite(name_end);
6764
6765 /*
6766 * Check for "default" argument.
6767 */
6768 if (STRNCMP(line, "default", name_end - line) == 0)
6769 {
6770 dodefault = TRUE;
6771 line = linep;
6772 name_end = skiptowhite(line);
6773 linep = skipwhite(name_end);
6774 }
6775
6776 /*
6777 * Check for "clear" or "link" argument.
6778 */
6779 if (STRNCMP(line, "clear", name_end - line) == 0)
6780 doclear = TRUE;
6781 if (STRNCMP(line, "link", name_end - line) == 0)
6782 dolink = TRUE;
6783
6784 /*
6785 * ":highlight {group-name}": list highlighting for one group.
6786 */
6787 if (!doclear && !dolink && ends_excmd(*linep))
6788 {
6789 id = syn_namen2id(line, (int)(name_end - line));
6790 if (id == 0)
6791 EMSG2(_("E411: highlight group not found: %s"), line);
6792 else
6793 highlight_list_one(id);
6794 return;
6795 }
6796
6797 /*
6798 * Handle ":highlight link {from} {to}" command.
6799 */
6800 if (dolink)
6801 {
6802 char_u *from_start = linep;
6803 char_u *from_end;
6804 char_u *to_start;
6805 char_u *to_end;
6806 int from_id;
6807 int to_id;
6808
6809 from_end = skiptowhite(from_start);
6810 to_start = skipwhite(from_end);
6811 to_end = skiptowhite(to_start);
6812
6813 if (ends_excmd(*from_start) || ends_excmd(*to_start))
6814 {
6815 EMSG2(_("E412: Not enough arguments: \":highlight link %s\""),
6816 from_start);
6817 return;
6818 }
6819
6820 if (!ends_excmd(*skipwhite(to_end)))
6821 {
6822 EMSG2(_("E413: Too many arguments: \":highlight link %s\""), from_start);
6823 return;
6824 }
6825
6826 from_id = syn_check_group(from_start, (int)(from_end - from_start));
6827 if (STRNCMP(to_start, "NONE", 4) == 0)
6828 to_id = 0;
6829 else
6830 to_id = syn_check_group(to_start, (int)(to_end - to_start));
6831
6832 if (from_id > 0 && (!init || HL_TABLE()[from_id - 1].sg_set == 0))
6833 {
6834 /*
6835 * Don't allow a link when there already is some highlighting
6836 * for the group, unless '!' is used
6837 */
6838 if (to_id > 0 && !forceit && !init
6839 && hl_has_settings(from_id - 1, dodefault))
6840 {
6841 if (sourcing_name == NULL && !dodefault)
6842 EMSG(_("E414: group has settings, highlight link ignored"));
6843 }
6844 else
6845 {
6846 if (!init)
6847 HL_TABLE()[from_id - 1].sg_set |= SG_LINK;
6848 HL_TABLE()[from_id - 1].sg_link = to_id;
Bram Moolenaar661b1822005-07-28 22:36:45 +00006849#ifdef FEAT_EVAL
6850 HL_TABLE()[from_id - 1].sg_scriptID = current_SID;
6851#endif
Bram Moolenaar1c8f93f2006-03-12 22:10:07 +00006852 redraw_all_later(SOME_VALID);
Bram Moolenaar071d4272004-06-13 20:20:40 +00006853 }
6854 }
6855
6856 /* Only call highlight_changed() once, after sourcing a syntax file */
6857 need_highlight_changed = TRUE;
6858
6859 return;
6860 }
6861
6862 if (doclear)
6863 {
6864 /*
6865 * ":highlight clear [group]" command.
6866 */
6867 line = linep;
6868 if (ends_excmd(*line))
6869 {
6870#ifdef FEAT_GUI
6871 /* First, we do not destroy the old values, but allocate the new
6872 * ones and update the display. THEN we destroy the old values.
6873 * If we destroy the old values first, then the old values
6874 * (such as GuiFont's or GuiFontset's) will still be displayed but
6875 * invalid because they were free'd.
6876 */
6877 if (gui.in_use)
6878 {
6879# ifdef FEAT_BEVAL_TIP
6880 gui_init_tooltip_font();
6881# endif
6882# if defined(FEAT_MENU) && (defined(FEAT_GUI_ATHENA) || defined(FEAT_GUI_MOTIF))
6883 gui_init_menu_font();
6884# endif
6885 }
6886# if defined(FEAT_GUI_MSWIN) || defined(FEAT_GUI_X11)
6887 gui_mch_def_colors();
6888# endif
6889# ifdef FEAT_GUI_X11
6890# ifdef FEAT_MENU
6891
6892 /* This only needs to be done when there is no Menu highlight
6893 * group defined by default, which IS currently the case.
6894 */
6895 gui_mch_new_menu_colors();
6896# endif
6897 if (gui.in_use)
6898 {
6899 gui_new_scrollbar_colors();
6900# ifdef FEAT_BEVAL
6901 gui_mch_new_tooltip_colors();
6902# endif
6903# ifdef FEAT_MENU
6904 gui_mch_new_menu_font();
6905# endif
6906 }
6907# endif
6908
6909 /* Ok, we're done allocating the new default graphics items.
6910 * The screen should already be refreshed at this point.
6911 * It is now Ok to clear out the old data.
6912 */
6913#endif
6914#ifdef FEAT_EVAL
Bram Moolenaar2ce06f62005-01-31 19:19:04 +00006915 do_unlet((char_u *)"colors_name", TRUE);
Bram Moolenaar071d4272004-06-13 20:20:40 +00006916#endif
6917 restore_cterm_colors();
6918
6919 /*
6920 * Clear all default highlight groups and load the defaults.
6921 */
6922 for (idx = 0; idx < highlight_ga.ga_len; ++idx)
6923 highlight_clear(idx);
6924 init_highlight(TRUE, TRUE);
6925#ifdef FEAT_GUI
6926 if (gui.in_use)
6927 highlight_gui_started();
6928#endif
6929 highlight_changed();
6930 redraw_later_clear();
6931 return;
6932 }
6933 name_end = skiptowhite(line);
6934 linep = skipwhite(name_end);
6935 }
6936
6937 /*
6938 * Find the group name in the table. If it does not exist yet, add it.
6939 */
6940 id = syn_check_group(line, (int)(name_end - line));
6941 if (id == 0) /* failed (out of memory) */
6942 return;
6943 idx = id - 1; /* index is ID minus one */
6944
6945 /* Return if "default" was used and the group already has settings. */
6946 if (dodefault && hl_has_settings(idx, TRUE))
6947 return;
6948
6949 if (STRCMP(HL_TABLE()[idx].sg_name_u, "NORMAL") == 0)
6950 is_normal_group = TRUE;
6951#ifdef FEAT_GUI_X11
6952 else if (STRCMP(HL_TABLE()[idx].sg_name_u, "MENU") == 0)
6953 is_menu_group = TRUE;
6954 else if (STRCMP(HL_TABLE()[idx].sg_name_u, "SCROLLBAR") == 0)
6955 is_scrollbar_group = TRUE;
6956 else if (STRCMP(HL_TABLE()[idx].sg_name_u, "TOOLTIP") == 0)
6957 is_tooltip_group = TRUE;
6958#endif
6959
6960 /* Clear the highlighting for ":hi clear {group}" and ":hi clear". */
6961 if (doclear || (forceit && init))
6962 {
6963 highlight_clear(idx);
6964 if (!doclear)
6965 HL_TABLE()[idx].sg_set = 0;
6966 }
6967
6968 if (!doclear)
6969 while (!ends_excmd(*linep))
6970 {
6971 key_start = linep;
6972 if (*linep == '=')
6973 {
6974 EMSG2(_("E415: unexpected equal sign: %s"), key_start);
6975 error = TRUE;
6976 break;
6977 }
6978
6979 /*
6980 * Isolate the key ("term", "ctermfg", "ctermbg", "font", "guifg" or
6981 * "guibg").
6982 */
6983 while (*linep && !vim_iswhite(*linep) && *linep != '=')
6984 ++linep;
6985 vim_free(key);
6986 key = vim_strnsave_up(key_start, (int)(linep - key_start));
6987 if (key == NULL)
6988 {
6989 error = TRUE;
6990 break;
6991 }
6992 linep = skipwhite(linep);
6993
6994 if (STRCMP(key, "NONE") == 0)
6995 {
6996 if (!init || HL_TABLE()[idx].sg_set == 0)
6997 {
6998 if (!init)
6999 HL_TABLE()[idx].sg_set |= SG_TERM+SG_CTERM+SG_GUI;
7000 highlight_clear(idx);
7001 }
7002 continue;
7003 }
7004
7005 /*
7006 * Check for the equal sign.
7007 */
7008 if (*linep != '=')
7009 {
7010 EMSG2(_("E416: missing equal sign: %s"), key_start);
7011 error = TRUE;
7012 break;
7013 }
7014 ++linep;
7015
7016 /*
7017 * Isolate the argument.
7018 */
7019 linep = skipwhite(linep);
7020 if (*linep == '\'') /* guifg='color name' */
7021 {
7022 arg_start = ++linep;
7023 linep = vim_strchr(linep, '\'');
7024 if (linep == NULL)
7025 {
7026 EMSG2(_(e_invarg2), key_start);
7027 error = TRUE;
7028 break;
7029 }
7030 }
7031 else
7032 {
7033 arg_start = linep;
7034 linep = skiptowhite(linep);
7035 }
7036 if (linep == arg_start)
7037 {
7038 EMSG2(_("E417: missing argument: %s"), key_start);
7039 error = TRUE;
7040 break;
7041 }
7042 vim_free(arg);
7043 arg = vim_strnsave(arg_start, (int)(linep - arg_start));
7044 if (arg == NULL)
7045 {
7046 error = TRUE;
7047 break;
7048 }
7049 if (*linep == '\'')
7050 ++linep;
7051
7052 /*
7053 * Store the argument.
7054 */
7055 if ( STRCMP(key, "TERM") == 0
7056 || STRCMP(key, "CTERM") == 0
7057 || STRCMP(key, "GUI") == 0)
7058 {
7059 attr = 0;
7060 off = 0;
7061 while (arg[off] != NUL)
7062 {
7063 for (i = sizeof(hl_attr_table) / sizeof(int); --i >= 0; )
7064 {
7065 len = (int)STRLEN(hl_name_table[i]);
7066 if (STRNICMP(arg + off, hl_name_table[i], len) == 0)
7067 {
7068 attr |= hl_attr_table[i];
7069 off += len;
7070 break;
7071 }
7072 }
7073 if (i < 0)
7074 {
7075 EMSG2(_("E418: Illegal value: %s"), arg);
7076 error = TRUE;
7077 break;
7078 }
7079 if (arg[off] == ',') /* another one follows */
7080 ++off;
7081 }
7082 if (error)
7083 break;
7084 if (*key == 'T')
7085 {
7086 if (!init || !(HL_TABLE()[idx].sg_set & SG_TERM))
7087 {
7088 if (!init)
7089 HL_TABLE()[idx].sg_set |= SG_TERM;
7090 HL_TABLE()[idx].sg_term = attr;
7091 }
7092 }
7093 else if (*key == 'C')
7094 {
7095 if (!init || !(HL_TABLE()[idx].sg_set & SG_CTERM))
7096 {
7097 if (!init)
7098 HL_TABLE()[idx].sg_set |= SG_CTERM;
7099 HL_TABLE()[idx].sg_cterm = attr;
7100 HL_TABLE()[idx].sg_cterm_bold = FALSE;
7101 }
7102 }
7103#ifdef FEAT_GUI
7104 else
7105 {
7106 if (!init || !(HL_TABLE()[idx].sg_set & SG_GUI))
7107 {
7108 if (!init)
7109 HL_TABLE()[idx].sg_set |= SG_GUI;
7110 HL_TABLE()[idx].sg_gui = attr;
7111 }
7112 }
7113#endif
7114 }
7115 else if (STRCMP(key, "FONT") == 0)
7116 {
7117 /* in non-GUI fonts are simply ignored */
7118#ifdef FEAT_GUI
7119 if (!gui.shell_created)
7120 {
7121 /* GUI not started yet, always accept the name. */
7122 vim_free(HL_TABLE()[idx].sg_font_name);
7123 HL_TABLE()[idx].sg_font_name = vim_strsave(arg);
7124 }
7125 else
7126 {
7127 GuiFont temp_sg_font = HL_TABLE()[idx].sg_font;
7128# ifdef FEAT_XFONTSET
7129 GuiFontset temp_sg_fontset = HL_TABLE()[idx].sg_fontset;
7130# endif
7131 /* First, save the current font/fontset.
7132 * Then try to allocate the font/fontset.
7133 * If the allocation fails, HL_TABLE()[idx].sg_font OR
7134 * sg_fontset will be set to NOFONT or NOFONTSET respectively.
7135 */
7136
7137 HL_TABLE()[idx].sg_font = NOFONT;
7138# ifdef FEAT_XFONTSET
7139 HL_TABLE()[idx].sg_fontset = NOFONTSET;
7140# endif
7141 hl_do_font(idx, arg, is_normal_group, is_menu_group,
7142 is_tooltip_group);
7143
7144# ifdef FEAT_XFONTSET
7145 if (HL_TABLE()[idx].sg_fontset != NOFONTSET)
7146 {
7147 /* New fontset was accepted. Free the old one, if there was
7148 * one.
7149 */
7150 gui_mch_free_fontset(temp_sg_fontset);
7151 vim_free(HL_TABLE()[idx].sg_font_name);
7152 HL_TABLE()[idx].sg_font_name = vim_strsave(arg);
7153 }
7154 else
7155 HL_TABLE()[idx].sg_fontset = temp_sg_fontset;
7156# endif
7157 if (HL_TABLE()[idx].sg_font != NOFONT)
7158 {
7159 /* New font was accepted. Free the old one, if there was
7160 * one.
7161 */
7162 gui_mch_free_font(temp_sg_font);
7163 vim_free(HL_TABLE()[idx].sg_font_name);
7164 HL_TABLE()[idx].sg_font_name = vim_strsave(arg);
7165 }
7166 else
7167 HL_TABLE()[idx].sg_font = temp_sg_font;
7168 }
7169#endif
7170 }
7171 else if (STRCMP(key, "CTERMFG") == 0 || STRCMP(key, "CTERMBG") == 0)
7172 {
7173 if (!init || !(HL_TABLE()[idx].sg_set & SG_CTERM))
7174 {
7175 if (!init)
7176 HL_TABLE()[idx].sg_set |= SG_CTERM;
7177
7178 /* When setting the foreground color, and previously the "bold"
7179 * flag was set for a light color, reset it now */
7180 if (key[5] == 'F' && HL_TABLE()[idx].sg_cterm_bold)
7181 {
7182 HL_TABLE()[idx].sg_cterm &= ~HL_BOLD;
7183 HL_TABLE()[idx].sg_cterm_bold = FALSE;
7184 }
7185
7186 if (VIM_ISDIGIT(*arg))
7187 color = atoi((char *)arg);
7188 else if (STRICMP(arg, "fg") == 0)
7189 {
7190 if (cterm_normal_fg_color)
7191 color = cterm_normal_fg_color - 1;
7192 else
7193 {
7194 EMSG(_("E419: FG color unknown"));
7195 error = TRUE;
7196 break;
7197 }
7198 }
7199 else if (STRICMP(arg, "bg") == 0)
7200 {
7201 if (cterm_normal_bg_color > 0)
7202 color = cterm_normal_bg_color - 1;
7203 else
7204 {
7205 EMSG(_("E420: BG color unknown"));
7206 error = TRUE;
7207 break;
7208 }
7209 }
7210 else
7211 {
7212 static char *(color_names[28]) = {
7213 "Black", "DarkBlue", "DarkGreen", "DarkCyan",
7214 "DarkRed", "DarkMagenta", "Brown", "DarkYellow",
7215 "Gray", "Grey",
7216 "LightGray", "LightGrey", "DarkGray", "DarkGrey",
7217 "Blue", "LightBlue", "Green", "LightGreen",
7218 "Cyan", "LightCyan", "Red", "LightRed", "Magenta",
7219 "LightMagenta", "Yellow", "LightYellow", "White", "NONE"};
7220 static int color_numbers_16[28] = {0, 1, 2, 3,
7221 4, 5, 6, 6,
7222 7, 7,
7223 7, 7, 8, 8,
7224 9, 9, 10, 10,
7225 11, 11, 12, 12, 13,
7226 13, 14, 14, 15, -1};
7227 /* for xterm with 88 colors... */
7228 static int color_numbers_88[28] = {0, 4, 2, 6,
7229 1, 5, 32, 72,
7230 84, 84,
7231 7, 7, 82, 82,
7232 12, 43, 10, 61,
7233 14, 63, 9, 74, 13,
7234 75, 11, 78, 15, -1};
7235 /* for xterm with 256 colors... */
7236 static int color_numbers_256[28] = {0, 4, 2, 6,
7237 1, 5, 130, 130,
7238 248, 248,
7239 7, 7, 242, 242,
7240 12, 81, 10, 121,
7241 14, 159, 9, 224, 13,
7242 225, 11, 229, 15, -1};
7243 /* for terminals with less than 16 colors... */
7244 static int color_numbers_8[28] = {0, 4, 2, 6,
7245 1, 5, 3, 3,
7246 7, 7,
7247 7, 7, 0+8, 0+8,
7248 4+8, 4+8, 2+8, 2+8,
7249 6+8, 6+8, 1+8, 1+8, 5+8,
7250 5+8, 3+8, 3+8, 7+8, -1};
7251#if defined(__QNXNTO__)
7252 static int *color_numbers_8_qansi = color_numbers_8;
7253 /* On qnx, the 8 & 16 color arrays are the same */
7254 if (STRNCMP(T_NAME, "qansi", 5) == 0)
7255 color_numbers_8_qansi = color_numbers_16;
7256#endif
7257
7258 /* reduce calls to STRICMP a bit, it can be slow */
7259 off = TOUPPER_ASC(*arg);
7260 for (i = (sizeof(color_names) / sizeof(char *)); --i >= 0; )
7261 if (off == color_names[i][0]
7262 && STRICMP(arg + 1, color_names[i] + 1) == 0)
7263 break;
7264 if (i < 0)
7265 {
7266 EMSG2(_("E421: Color name or number not recognized: %s"), key_start);
7267 error = TRUE;
7268 break;
7269 }
7270
7271 /* Use the _16 table to check if its a valid color name. */
7272 color = color_numbers_16[i];
7273 if (color >= 0)
7274 {
7275 if (t_colors == 8)
7276 {
7277 /* t_Co is 8: use the 8 colors table */
7278#if defined(__QNXNTO__)
7279 color = color_numbers_8_qansi[i];
7280#else
7281 color = color_numbers_8[i];
7282#endif
7283 if (key[5] == 'F')
7284 {
7285 /* set/reset bold attribute to get light foreground
7286 * colors (on some terminals, e.g. "linux") */
7287 if (color & 8)
7288 {
7289 HL_TABLE()[idx].sg_cterm |= HL_BOLD;
7290 HL_TABLE()[idx].sg_cterm_bold = TRUE;
7291 }
7292 else
7293 HL_TABLE()[idx].sg_cterm &= ~HL_BOLD;
7294 }
7295 color &= 7; /* truncate to 8 colors */
7296 }
7297 else if (t_colors == 16 || t_colors == 88
7298 || t_colors == 256)
7299 {
7300 /*
7301 * Guess: if the termcap entry ends in 'm', it is
7302 * probably an xterm-like terminal. Use the changed
7303 * order for colors.
7304 */
7305 if (*T_CAF != NUL)
7306 p = T_CAF;
7307 else
7308 p = T_CSF;
7309 if (*p != NUL && *(p + STRLEN(p) - 1) == 'm')
7310 switch (t_colors)
7311 {
7312 case 16:
7313 color = color_numbers_8[i];
7314 break;
7315 case 88:
7316 color = color_numbers_88[i];
7317 break;
7318 case 256:
7319 color = color_numbers_256[i];
7320 break;
7321 }
7322 }
7323 }
7324 }
Bram Moolenaarccbab932010-05-13 15:40:30 +02007325 /* Add one to the argument, to avoid zero. Zero is used for
7326 * "NONE", then "color" is -1. */
Bram Moolenaar071d4272004-06-13 20:20:40 +00007327 if (key[5] == 'F')
7328 {
7329 HL_TABLE()[idx].sg_cterm_fg = color + 1;
7330 if (is_normal_group)
7331 {
7332 cterm_normal_fg_color = color + 1;
7333 cterm_normal_fg_bold = (HL_TABLE()[idx].sg_cterm & HL_BOLD);
7334#ifdef FEAT_GUI
7335 /* Don't do this if the GUI is used. */
7336 if (!gui.in_use && !gui.starting)
7337#endif
7338 {
7339 must_redraw = CLEAR;
Bram Moolenaarccbab932010-05-13 15:40:30 +02007340 if (termcap_active && color >= 0)
Bram Moolenaar071d4272004-06-13 20:20:40 +00007341 term_fg_color(color);
7342 }
7343 }
7344 }
7345 else
7346 {
7347 HL_TABLE()[idx].sg_cterm_bg = color + 1;
7348 if (is_normal_group)
7349 {
7350 cterm_normal_bg_color = color + 1;
7351#ifdef FEAT_GUI
7352 /* Don't mess with 'background' if the GUI is used. */
7353 if (!gui.in_use && !gui.starting)
7354#endif
7355 {
7356 must_redraw = CLEAR;
Bram Moolenaarccbab932010-05-13 15:40:30 +02007357 if (color >= 0)
7358 {
7359 if (termcap_active)
7360 term_bg_color(color);
7361 if (t_colors < 16)
7362 i = (color == 0 || color == 4);
7363 else
7364 i = (color < 7 || color == 8);
7365 /* Set the 'background' option if the value is
7366 * wrong. */
7367 if (i != (*p_bg == 'd'))
7368 set_option_value((char_u *)"bg", 0L,
7369 i ? (char_u *)"dark"
7370 : (char_u *)"light", 0);
7371 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00007372 }
7373 }
7374 }
7375 }
7376 }
7377 else if (STRCMP(key, "GUIFG") == 0)
7378 {
7379#ifdef FEAT_GUI /* in non-GUI guifg colors are simply ignored */
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007380 if (!init || !(HL_TABLE()[idx].sg_set & SG_GUI))
Bram Moolenaar071d4272004-06-13 20:20:40 +00007381 {
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007382 if (!init)
7383 HL_TABLE()[idx].sg_set |= SG_GUI;
7384
7385 i = color_name2handle(arg);
7386 if (i != INVALCOLOR || STRCMP(arg, "NONE") == 0 || !gui.in_use)
7387 {
7388 HL_TABLE()[idx].sg_gui_fg = i;
7389 vim_free(HL_TABLE()[idx].sg_gui_fg_name);
7390 if (STRCMP(arg, "NONE"))
7391 HL_TABLE()[idx].sg_gui_fg_name = vim_strsave(arg);
7392 else
7393 HL_TABLE()[idx].sg_gui_fg_name = NULL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007394# ifdef FEAT_GUI_X11
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007395 if (is_menu_group)
7396 gui.menu_fg_pixel = i;
7397 if (is_scrollbar_group)
7398 gui.scroll_fg_pixel = i;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007399# ifdef FEAT_BEVAL
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007400 if (is_tooltip_group)
7401 gui.tooltip_fg_pixel = i;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007402# endif
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007403 do_colors = TRUE;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007404# endif
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007405 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00007406 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00007407#endif
7408 }
7409 else if (STRCMP(key, "GUIBG") == 0)
7410 {
7411#ifdef FEAT_GUI /* in non-GUI guibg colors are simply ignored */
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007412 if (!init || !(HL_TABLE()[idx].sg_set & SG_GUI))
Bram Moolenaar071d4272004-06-13 20:20:40 +00007413 {
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007414 if (!init)
7415 HL_TABLE()[idx].sg_set |= SG_GUI;
7416
7417 i = color_name2handle(arg);
7418 if (i != INVALCOLOR || STRCMP(arg, "NONE") == 0 || !gui.in_use)
7419 {
7420 HL_TABLE()[idx].sg_gui_bg = i;
7421 vim_free(HL_TABLE()[idx].sg_gui_bg_name);
7422 if (STRCMP(arg, "NONE") != 0)
7423 HL_TABLE()[idx].sg_gui_bg_name = vim_strsave(arg);
7424 else
7425 HL_TABLE()[idx].sg_gui_bg_name = NULL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007426# ifdef FEAT_GUI_X11
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007427 if (is_menu_group)
7428 gui.menu_bg_pixel = i;
7429 if (is_scrollbar_group)
7430 gui.scroll_bg_pixel = i;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007431# ifdef FEAT_BEVAL
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007432 if (is_tooltip_group)
7433 gui.tooltip_bg_pixel = i;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007434# endif
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007435 do_colors = TRUE;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007436# endif
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007437 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00007438 }
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007439#endif
7440 }
7441 else if (STRCMP(key, "GUISP") == 0)
7442 {
7443#ifdef FEAT_GUI /* in non-GUI guisp colors are simply ignored */
7444 if (!init || !(HL_TABLE()[idx].sg_set & SG_GUI))
7445 {
7446 if (!init)
7447 HL_TABLE()[idx].sg_set |= SG_GUI;
7448
7449 i = color_name2handle(arg);
7450 if (i != INVALCOLOR || STRCMP(arg, "NONE") == 0 || !gui.in_use)
7451 {
7452 HL_TABLE()[idx].sg_gui_sp = i;
7453 vim_free(HL_TABLE()[idx].sg_gui_sp_name);
7454 if (STRCMP(arg, "NONE") != 0)
7455 HL_TABLE()[idx].sg_gui_sp_name = vim_strsave(arg);
7456 else
7457 HL_TABLE()[idx].sg_gui_sp_name = NULL;
7458 }
7459 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00007460#endif
7461 }
7462 else if (STRCMP(key, "START") == 0 || STRCMP(key, "STOP") == 0)
7463 {
7464 char_u buf[100];
7465 char_u *tname;
7466
7467 if (!init)
7468 HL_TABLE()[idx].sg_set |= SG_TERM;
7469
7470 /*
7471 * The "start" and "stop" arguments can be a literal escape
Bram Moolenaarb8017e72007-05-10 18:59:07 +00007472 * sequence, or a comma separated list of terminal codes.
Bram Moolenaar071d4272004-06-13 20:20:40 +00007473 */
7474 if (STRNCMP(arg, "t_", 2) == 0)
7475 {
7476 off = 0;
7477 buf[0] = 0;
7478 while (arg[off] != NUL)
7479 {
7480 /* Isolate one termcap name */
7481 for (len = 0; arg[off + len] &&
7482 arg[off + len] != ','; ++len)
7483 ;
7484 tname = vim_strnsave(arg + off, len);
7485 if (tname == NULL) /* out of memory */
7486 {
7487 error = TRUE;
7488 break;
7489 }
7490 /* lookup the escape sequence for the item */
7491 p = get_term_code(tname);
7492 vim_free(tname);
7493 if (p == NULL) /* ignore non-existing things */
7494 p = (char_u *)"";
7495
7496 /* Append it to the already found stuff */
7497 if ((int)(STRLEN(buf) + STRLEN(p)) >= 99)
7498 {
7499 EMSG2(_("E422: terminal code too long: %s"), arg);
7500 error = TRUE;
7501 break;
7502 }
7503 STRCAT(buf, p);
7504
7505 /* Advance to the next item */
7506 off += len;
7507 if (arg[off] == ',') /* another one follows */
7508 ++off;
7509 }
7510 }
7511 else
7512 {
7513 /*
7514 * Copy characters from arg[] to buf[], translating <> codes.
7515 */
7516 for (p = arg, off = 0; off < 100 && *p; )
7517 {
7518 len = trans_special(&p, buf + off, FALSE);
7519 if (len) /* recognized special char */
7520 off += len;
7521 else /* copy as normal char */
7522 buf[off++] = *p++;
7523 }
7524 buf[off] = NUL;
7525 }
7526 if (error)
7527 break;
7528
7529 if (STRCMP(buf, "NONE") == 0) /* resetting the value */
7530 p = NULL;
7531 else
7532 p = vim_strsave(buf);
7533 if (key[2] == 'A')
7534 {
7535 vim_free(HL_TABLE()[idx].sg_start);
7536 HL_TABLE()[idx].sg_start = p;
7537 }
7538 else
7539 {
7540 vim_free(HL_TABLE()[idx].sg_stop);
7541 HL_TABLE()[idx].sg_stop = p;
7542 }
7543 }
7544 else
7545 {
7546 EMSG2(_("E423: Illegal argument: %s"), key_start);
7547 error = TRUE;
7548 break;
7549 }
7550
7551 /*
7552 * When highlighting has been given for a group, don't link it.
7553 */
7554 if (!init || !(HL_TABLE()[idx].sg_set & SG_LINK))
7555 HL_TABLE()[idx].sg_link = 0;
7556
7557 /*
7558 * Continue with next argument.
7559 */
7560 linep = skipwhite(linep);
7561 }
7562
7563 /*
7564 * If there is an error, and it's a new entry, remove it from the table.
7565 */
7566 if (error && idx == highlight_ga.ga_len)
7567 syn_unadd_group();
7568 else
7569 {
7570 if (is_normal_group)
7571 {
7572 HL_TABLE()[idx].sg_term_attr = 0;
7573 HL_TABLE()[idx].sg_cterm_attr = 0;
7574#ifdef FEAT_GUI
7575 HL_TABLE()[idx].sg_gui_attr = 0;
7576 /*
7577 * Need to update all groups, because they might be using "bg"
7578 * and/or "fg", which have been changed now.
7579 */
7580 if (gui.in_use)
7581 highlight_gui_started();
7582#endif
7583 }
7584#ifdef FEAT_GUI_X11
7585# ifdef FEAT_MENU
7586 else if (is_menu_group)
7587 {
7588 if (gui.in_use && do_colors)
7589 gui_mch_new_menu_colors();
7590 }
7591# endif
7592 else if (is_scrollbar_group)
7593 {
7594 if (gui.in_use && do_colors)
7595 gui_new_scrollbar_colors();
7596 }
7597# ifdef FEAT_BEVAL
7598 else if (is_tooltip_group)
7599 {
7600 if (gui.in_use && do_colors)
7601 gui_mch_new_tooltip_colors();
7602 }
7603# endif
7604#endif
7605 else
7606 set_hl_attr(idx);
Bram Moolenaar661b1822005-07-28 22:36:45 +00007607#ifdef FEAT_EVAL
7608 HL_TABLE()[idx].sg_scriptID = current_SID;
7609#endif
Bram Moolenaar910f66f2006-04-05 20:41:53 +00007610 redraw_all_later(NOT_VALID);
Bram Moolenaar071d4272004-06-13 20:20:40 +00007611 }
7612 vim_free(key);
7613 vim_free(arg);
7614
7615 /* Only call highlight_changed() once, after sourcing a syntax file */
7616 need_highlight_changed = TRUE;
7617}
7618
Bram Moolenaar1ec484f2005-06-24 23:07:47 +00007619#if defined(EXITFREE) || defined(PROTO)
7620 void
7621free_highlight()
7622{
7623 int i;
7624
7625 for (i = 0; i < highlight_ga.ga_len; ++i)
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00007626 {
Bram Moolenaar1ec484f2005-06-24 23:07:47 +00007627 highlight_clear(i);
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00007628 vim_free(HL_TABLE()[i].sg_name);
7629 vim_free(HL_TABLE()[i].sg_name_u);
7630 }
Bram Moolenaar1ec484f2005-06-24 23:07:47 +00007631 ga_clear(&highlight_ga);
7632}
7633#endif
7634
Bram Moolenaar071d4272004-06-13 20:20:40 +00007635/*
7636 * Reset the cterm colors to what they were before Vim was started, if
7637 * possible. Otherwise reset them to zero.
7638 */
7639 void
7640restore_cterm_colors()
7641{
7642#if defined(MSDOS) || (defined(WIN3264) && !defined(FEAT_GUI_W32))
7643 /* Since t_me has been set, this probably means that the user
7644 * wants to use this as default colors. Need to reset default
7645 * background/foreground colors. */
7646 mch_set_normal_colors();
7647#else
7648 cterm_normal_fg_color = 0;
7649 cterm_normal_fg_bold = 0;
7650 cterm_normal_bg_color = 0;
7651#endif
7652}
7653
7654/*
7655 * Return TRUE if highlight group "idx" has any settings.
7656 * When "check_link" is TRUE also check for an existing link.
7657 */
7658 static int
7659hl_has_settings(idx, check_link)
7660 int idx;
7661 int check_link;
7662{
7663 return ( HL_TABLE()[idx].sg_term_attr != 0
7664 || HL_TABLE()[idx].sg_cterm_attr != 0
7665#ifdef FEAT_GUI
7666 || HL_TABLE()[idx].sg_gui_attr != 0
7667#endif
7668 || (check_link && (HL_TABLE()[idx].sg_set & SG_LINK)));
7669}
7670
7671/*
7672 * Clear highlighting for one group.
7673 */
7674 static void
7675highlight_clear(idx)
7676 int idx;
7677{
7678 HL_TABLE()[idx].sg_term = 0;
7679 vim_free(HL_TABLE()[idx].sg_start);
7680 HL_TABLE()[idx].sg_start = NULL;
7681 vim_free(HL_TABLE()[idx].sg_stop);
7682 HL_TABLE()[idx].sg_stop = NULL;
7683 HL_TABLE()[idx].sg_term_attr = 0;
7684 HL_TABLE()[idx].sg_cterm = 0;
7685 HL_TABLE()[idx].sg_cterm_bold = FALSE;
7686 HL_TABLE()[idx].sg_cterm_fg = 0;
7687 HL_TABLE()[idx].sg_cterm_bg = 0;
7688 HL_TABLE()[idx].sg_cterm_attr = 0;
7689#ifdef FEAT_GUI /* in non-GUI fonts are simply ignored */
7690 HL_TABLE()[idx].sg_gui = 0;
7691 HL_TABLE()[idx].sg_gui_fg = INVALCOLOR;
7692 vim_free(HL_TABLE()[idx].sg_gui_fg_name);
7693 HL_TABLE()[idx].sg_gui_fg_name = NULL;
7694 HL_TABLE()[idx].sg_gui_bg = INVALCOLOR;
7695 vim_free(HL_TABLE()[idx].sg_gui_bg_name);
7696 HL_TABLE()[idx].sg_gui_bg_name = NULL;
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007697 HL_TABLE()[idx].sg_gui_sp = INVALCOLOR;
7698 vim_free(HL_TABLE()[idx].sg_gui_sp_name);
7699 HL_TABLE()[idx].sg_gui_sp_name = NULL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00007700 gui_mch_free_font(HL_TABLE()[idx].sg_font);
7701 HL_TABLE()[idx].sg_font = NOFONT;
7702# ifdef FEAT_XFONTSET
7703 gui_mch_free_fontset(HL_TABLE()[idx].sg_fontset);
7704 HL_TABLE()[idx].sg_fontset = NOFONTSET;
7705# endif
7706 vim_free(HL_TABLE()[idx].sg_font_name);
7707 HL_TABLE()[idx].sg_font_name = NULL;
7708 HL_TABLE()[idx].sg_gui_attr = 0;
7709#endif
Bram Moolenaar661b1822005-07-28 22:36:45 +00007710#ifdef FEAT_EVAL
7711 /* Clear the script ID only when there is no link, since that is not
7712 * cleared. */
7713 if (HL_TABLE()[idx].sg_link == 0)
7714 HL_TABLE()[idx].sg_scriptID = 0;
7715#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00007716}
7717
7718#if defined(FEAT_GUI) || defined(PROTO)
7719/*
7720 * Set the normal foreground and background colors according to the "Normal"
Bram Moolenaarf5b63862009-12-16 17:13:44 +00007721 * highlighting group. For X11 also set "Menu", "Scrollbar", and
Bram Moolenaar071d4272004-06-13 20:20:40 +00007722 * "Tooltip" colors.
7723 */
7724 void
7725set_normal_colors()
7726{
7727 if (set_group_colors((char_u *)"Normal",
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007728 &gui.norm_pixel, &gui.back_pixel,
7729 FALSE, TRUE, FALSE))
Bram Moolenaar071d4272004-06-13 20:20:40 +00007730 {
7731 gui_mch_new_colors();
7732 must_redraw = CLEAR;
7733 }
7734#ifdef FEAT_GUI_X11
7735 if (set_group_colors((char_u *)"Menu",
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007736 &gui.menu_fg_pixel, &gui.menu_bg_pixel,
7737 TRUE, FALSE, FALSE))
Bram Moolenaar071d4272004-06-13 20:20:40 +00007738 {
7739# ifdef FEAT_MENU
7740 gui_mch_new_menu_colors();
7741# endif
7742 must_redraw = CLEAR;
7743 }
7744# ifdef FEAT_BEVAL
7745 if (set_group_colors((char_u *)"Tooltip",
7746 &gui.tooltip_fg_pixel, &gui.tooltip_bg_pixel,
7747 FALSE, FALSE, TRUE))
7748 {
7749# ifdef FEAT_TOOLBAR
7750 gui_mch_new_tooltip_colors();
7751# endif
7752 must_redraw = CLEAR;
7753 }
7754#endif
7755 if (set_group_colors((char_u *)"Scrollbar",
Bram Moolenaare2cc9702005-03-15 22:43:58 +00007756 &gui.scroll_fg_pixel, &gui.scroll_bg_pixel,
7757 FALSE, FALSE, FALSE))
Bram Moolenaar071d4272004-06-13 20:20:40 +00007758 {
7759 gui_new_scrollbar_colors();
7760 must_redraw = CLEAR;
7761 }
7762#endif
7763}
7764
7765/*
7766 * Set the colors for "Normal", "Menu", "Tooltip" or "Scrollbar".
7767 */
7768 static int
7769set_group_colors(name, fgp, bgp, do_menu, use_norm, do_tooltip)
7770 char_u *name;
7771 guicolor_T *fgp;
7772 guicolor_T *bgp;
7773 int do_menu;
7774 int use_norm;
7775 int do_tooltip;
7776{
7777 int idx;
7778
7779 idx = syn_name2id(name) - 1;
7780 if (idx >= 0)
7781 {
7782 gui_do_one_color(idx, do_menu, do_tooltip);
7783
7784 if (HL_TABLE()[idx].sg_gui_fg != INVALCOLOR)
7785 *fgp = HL_TABLE()[idx].sg_gui_fg;
7786 else if (use_norm)
7787 *fgp = gui.def_norm_pixel;
7788 if (HL_TABLE()[idx].sg_gui_bg != INVALCOLOR)
7789 *bgp = HL_TABLE()[idx].sg_gui_bg;
7790 else if (use_norm)
7791 *bgp = gui.def_back_pixel;
7792 return TRUE;
7793 }
7794 return FALSE;
7795}
7796
7797/*
7798 * Get the font of the "Normal" group.
7799 * Returns "" when it's not found or not set.
7800 */
7801 char_u *
7802hl_get_font_name()
7803{
7804 int id;
7805 char_u *s;
7806
7807 id = syn_name2id((char_u *)"Normal");
7808 if (id > 0)
7809 {
7810 s = HL_TABLE()[id - 1].sg_font_name;
7811 if (s != NULL)
7812 return s;
7813 }
7814 return (char_u *)"";
7815}
7816
7817/*
7818 * Set font for "Normal" group. Called by gui_mch_init_font() when a font has
7819 * actually chosen to be used.
7820 */
7821 void
7822hl_set_font_name(font_name)
7823 char_u *font_name;
7824{
7825 int id;
7826
7827 id = syn_name2id((char_u *)"Normal");
7828 if (id > 0)
7829 {
7830 vim_free(HL_TABLE()[id - 1].sg_font_name);
7831 HL_TABLE()[id - 1].sg_font_name = vim_strsave(font_name);
7832 }
7833}
7834
7835/*
7836 * Set background color for "Normal" group. Called by gui_set_bg_color()
7837 * when the color is known.
7838 */
7839 void
7840hl_set_bg_color_name(name)
7841 char_u *name; /* must have been allocated */
7842{
7843 int id;
7844
7845 if (name != NULL)
7846 {
7847 id = syn_name2id((char_u *)"Normal");
7848 if (id > 0)
7849 {
7850 vim_free(HL_TABLE()[id - 1].sg_gui_bg_name);
7851 HL_TABLE()[id - 1].sg_gui_bg_name = name;
7852 }
7853 }
7854}
7855
7856/*
7857 * Set foreground color for "Normal" group. Called by gui_set_fg_color()
7858 * when the color is known.
7859 */
7860 void
7861hl_set_fg_color_name(name)
7862 char_u *name; /* must have been allocated */
7863{
7864 int id;
7865
7866 if (name != NULL)
7867 {
7868 id = syn_name2id((char_u *)"Normal");
7869 if (id > 0)
7870 {
7871 vim_free(HL_TABLE()[id - 1].sg_gui_fg_name);
7872 HL_TABLE()[id - 1].sg_gui_fg_name = name;
7873 }
7874 }
7875}
7876
7877/*
7878 * Return the handle for a color name.
7879 * Returns INVALCOLOR when failed.
7880 */
7881 static guicolor_T
7882color_name2handle(name)
7883 char_u *name;
7884{
7885 if (STRCMP(name, "NONE") == 0)
7886 return INVALCOLOR;
7887
7888 if (STRICMP(name, "fg") == 0 || STRICMP(name, "foreground") == 0)
7889 return gui.norm_pixel;
7890 if (STRICMP(name, "bg") == 0 || STRICMP(name, "background") == 0)
7891 return gui.back_pixel;
7892
7893 return gui_get_color(name);
7894}
7895
7896/*
7897 * Return the handle for a font name.
7898 * Returns NOFONT when failed.
7899 */
7900 static GuiFont
7901font_name2handle(name)
7902 char_u *name;
7903{
7904 if (STRCMP(name, "NONE") == 0)
7905 return NOFONT;
7906
7907 return gui_mch_get_font(name, TRUE);
7908}
7909
7910# ifdef FEAT_XFONTSET
7911/*
7912 * Return the handle for a fontset name.
7913 * Returns NOFONTSET when failed.
7914 */
7915 static GuiFontset
7916fontset_name2handle(name, fixed_width)
7917 char_u *name;
7918 int fixed_width;
7919{
7920 if (STRCMP(name, "NONE") == 0)
7921 return NOFONTSET;
7922
7923 return gui_mch_get_fontset(name, TRUE, fixed_width);
7924}
7925# endif
7926
7927/*
7928 * Get the font or fontset for one highlight group.
7929 */
Bram Moolenaar071d4272004-06-13 20:20:40 +00007930 static void
7931hl_do_font(idx, arg, do_normal, do_menu, do_tooltip)
7932 int idx;
7933 char_u *arg;
Bram Moolenaar2c4278f2009-05-17 11:33:22 +00007934 int do_normal; /* set normal font */
7935 int do_menu UNUSED; /* set menu font */
7936 int do_tooltip UNUSED; /* set tooltip font */
Bram Moolenaar071d4272004-06-13 20:20:40 +00007937{
7938# ifdef FEAT_XFONTSET
7939 /* If 'guifontset' is not empty, first try using the name as a
7940 * fontset. If that doesn't work, use it as a font name. */
7941 if (*p_guifontset != NUL
7942# ifdef FONTSET_ALWAYS
7943 || do_menu
7944# endif
7945# ifdef FEAT_BEVAL_TIP
7946 /* In Athena & Motif, the Tooltip highlight group is always a fontset */
7947 || do_tooltip
7948# endif
7949 )
7950 HL_TABLE()[idx].sg_fontset = fontset_name2handle(arg, 0
7951# ifdef FONTSET_ALWAYS
7952 || do_menu
7953# endif
7954# ifdef FEAT_BEVAL_TIP
7955 || do_tooltip
7956# endif
7957 );
7958 if (HL_TABLE()[idx].sg_fontset != NOFONTSET)
7959 {
7960 /* If it worked and it's the Normal group, use it as the
7961 * normal fontset. Same for the Menu group. */
7962 if (do_normal)
7963 gui_init_font(arg, TRUE);
7964# if (defined(FEAT_GUI_MOTIF) || defined(FEAT_GUI_ATHENA)) && defined(FEAT_MENU)
7965 if (do_menu)
7966 {
7967# ifdef FONTSET_ALWAYS
7968 gui.menu_fontset = HL_TABLE()[idx].sg_fontset;
7969# else
7970 /* YIKES! This is a bug waiting to crash the program */
7971 gui.menu_font = HL_TABLE()[idx].sg_fontset;
7972# endif
7973 gui_mch_new_menu_font();
7974 }
7975# ifdef FEAT_BEVAL
7976 if (do_tooltip)
7977 {
7978 /* The Athena widget set cannot currently handle switching between
7979 * displaying a single font and a fontset.
7980 * If the XtNinternational resource is set to True at widget
Bram Moolenaarb8017e72007-05-10 18:59:07 +00007981 * creation, then a fontset is always used, otherwise an
Bram Moolenaar071d4272004-06-13 20:20:40 +00007982 * XFontStruct is used.
7983 */
7984 gui.tooltip_fontset = (XFontSet)HL_TABLE()[idx].sg_fontset;
7985 gui_mch_new_tooltip_font();
7986 }
7987# endif
7988# endif
7989 }
7990 else
7991# endif
7992 {
7993 HL_TABLE()[idx].sg_font = font_name2handle(arg);
7994 /* If it worked and it's the Normal group, use it as the
7995 * normal font. Same for the Menu group. */
7996 if (HL_TABLE()[idx].sg_font != NOFONT)
7997 {
7998 if (do_normal)
7999 gui_init_font(arg, FALSE);
8000#ifndef FONTSET_ALWAYS
8001# if (defined(FEAT_GUI_MOTIF) || defined(FEAT_GUI_ATHENA)) && defined(FEAT_MENU)
8002 if (do_menu)
8003 {
8004 gui.menu_font = HL_TABLE()[idx].sg_font;
8005 gui_mch_new_menu_font();
8006 }
8007# endif
8008#endif
8009 }
8010 }
8011}
8012
8013#endif /* FEAT_GUI */
8014
8015/*
8016 * Table with the specifications for an attribute number.
8017 * Note that this table is used by ALL buffers. This is required because the
8018 * GUI can redraw at any time for any buffer.
8019 */
Bram Moolenaar6c0b44b2005-06-01 21:56:33 +00008020static garray_T term_attr_table = {0, 0, 0, 0, NULL};
Bram Moolenaar071d4272004-06-13 20:20:40 +00008021
8022#define TERM_ATTR_ENTRY(idx) ((attrentry_T *)term_attr_table.ga_data)[idx]
8023
Bram Moolenaar6c0b44b2005-06-01 21:56:33 +00008024static garray_T cterm_attr_table = {0, 0, 0, 0, NULL};
Bram Moolenaar071d4272004-06-13 20:20:40 +00008025
8026#define CTERM_ATTR_ENTRY(idx) ((attrentry_T *)cterm_attr_table.ga_data)[idx]
8027
8028#ifdef FEAT_GUI
Bram Moolenaar6c0b44b2005-06-01 21:56:33 +00008029static garray_T gui_attr_table = {0, 0, 0, 0, NULL};
Bram Moolenaar071d4272004-06-13 20:20:40 +00008030
8031#define GUI_ATTR_ENTRY(idx) ((attrentry_T *)gui_attr_table.ga_data)[idx]
8032#endif
8033
8034/*
8035 * Return the attr number for a set of colors and font.
8036 * Add a new entry to the term_attr_table, cterm_attr_table or gui_attr_table
8037 * if the combination is new.
8038 * Return 0 for error (no more room).
8039 */
8040 static int
8041get_attr_entry(table, aep)
8042 garray_T *table;
8043 attrentry_T *aep;
8044{
8045 int i;
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00008046 attrentry_T *taep;
Bram Moolenaar071d4272004-06-13 20:20:40 +00008047 static int recursive = FALSE;
8048
8049 /*
8050 * Init the table, in case it wasn't done yet.
8051 */
8052 table->ga_itemsize = sizeof(attrentry_T);
8053 table->ga_growsize = 7;
8054
8055 /*
8056 * Try to find an entry with the same specifications.
8057 */
8058 for (i = 0; i < table->ga_len; ++i)
8059 {
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00008060 taep = &(((attrentry_T *)table->ga_data)[i]);
8061 if ( aep->ae_attr == taep->ae_attr
Bram Moolenaar071d4272004-06-13 20:20:40 +00008062 && (
8063#ifdef FEAT_GUI
8064 (table == &gui_attr_table
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00008065 && (aep->ae_u.gui.fg_color == taep->ae_u.gui.fg_color
8066 && aep->ae_u.gui.bg_color
8067 == taep->ae_u.gui.bg_color
8068 && aep->ae_u.gui.sp_color
8069 == taep->ae_u.gui.sp_color
8070 && aep->ae_u.gui.font == taep->ae_u.gui.font
Bram Moolenaar071d4272004-06-13 20:20:40 +00008071# ifdef FEAT_XFONTSET
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00008072 && aep->ae_u.gui.fontset == taep->ae_u.gui.fontset
Bram Moolenaar071d4272004-06-13 20:20:40 +00008073# endif
8074 ))
8075 ||
8076#endif
8077 (table == &term_attr_table
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00008078 && (aep->ae_u.term.start == NULL)
8079 == (taep->ae_u.term.start == NULL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00008080 && (aep->ae_u.term.start == NULL
8081 || STRCMP(aep->ae_u.term.start,
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00008082 taep->ae_u.term.start) == 0)
8083 && (aep->ae_u.term.stop == NULL)
8084 == (taep->ae_u.term.stop == NULL)
Bram Moolenaar071d4272004-06-13 20:20:40 +00008085 && (aep->ae_u.term.stop == NULL
8086 || STRCMP(aep->ae_u.term.stop,
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00008087 taep->ae_u.term.stop) == 0))
Bram Moolenaar071d4272004-06-13 20:20:40 +00008088 || (table == &cterm_attr_table
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00008089 && aep->ae_u.cterm.fg_color
8090 == taep->ae_u.cterm.fg_color
8091 && aep->ae_u.cterm.bg_color
8092 == taep->ae_u.cterm.bg_color)
Bram Moolenaar071d4272004-06-13 20:20:40 +00008093 ))
8094
8095 return i + ATTR_OFF;
8096 }
8097
Bram Moolenaar34cdc3e2005-05-18 22:24:46 +00008098 if (table->ga_len + ATTR_OFF > MAX_TYPENR)
Bram Moolenaar071d4272004-06-13 20:20:40 +00008099 {
8100 /*
8101 * Running out of attribute entries! remove all attributes, and
8102 * compute new ones for all groups.
8103 * When called recursively, we are really out of numbers.
8104 */
8105 if (recursive)
8106 {
8107 EMSG(_("E424: Too many different highlighting attributes in use"));
8108 return 0;
8109 }
8110 recursive = TRUE;
8111
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00008112 clear_hl_tables();
8113
Bram Moolenaar071d4272004-06-13 20:20:40 +00008114 must_redraw = CLEAR;
8115
8116 for (i = 0; i < highlight_ga.ga_len; ++i)
8117 set_hl_attr(i);
8118
8119 recursive = FALSE;
8120 }
8121
8122 /*
8123 * This is a new combination of colors and font, add an entry.
8124 */
8125 if (ga_grow(table, 1) == FAIL)
8126 return 0;
8127
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00008128 taep = &(((attrentry_T *)table->ga_data)[table->ga_len]);
8129 vim_memset(taep, 0, sizeof(attrentry_T));
8130 taep->ae_attr = aep->ae_attr;
Bram Moolenaar071d4272004-06-13 20:20:40 +00008131#ifdef FEAT_GUI
8132 if (table == &gui_attr_table)
8133 {
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00008134 taep->ae_u.gui.fg_color = aep->ae_u.gui.fg_color;
8135 taep->ae_u.gui.bg_color = aep->ae_u.gui.bg_color;
8136 taep->ae_u.gui.sp_color = aep->ae_u.gui.sp_color;
8137 taep->ae_u.gui.font = aep->ae_u.gui.font;
Bram Moolenaar071d4272004-06-13 20:20:40 +00008138# ifdef FEAT_XFONTSET
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00008139 taep->ae_u.gui.fontset = aep->ae_u.gui.fontset;
Bram Moolenaar071d4272004-06-13 20:20:40 +00008140# endif
8141 }
8142#endif
8143 if (table == &term_attr_table)
8144 {
8145 if (aep->ae_u.term.start == NULL)
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00008146 taep->ae_u.term.start = NULL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00008147 else
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00008148 taep->ae_u.term.start = vim_strsave(aep->ae_u.term.start);
Bram Moolenaar071d4272004-06-13 20:20:40 +00008149 if (aep->ae_u.term.stop == NULL)
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00008150 taep->ae_u.term.stop = NULL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00008151 else
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00008152 taep->ae_u.term.stop = vim_strsave(aep->ae_u.term.stop);
Bram Moolenaar071d4272004-06-13 20:20:40 +00008153 }
8154 else if (table == &cterm_attr_table)
8155 {
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00008156 taep->ae_u.cterm.fg_color = aep->ae_u.cterm.fg_color;
8157 taep->ae_u.cterm.bg_color = aep->ae_u.cterm.bg_color;
Bram Moolenaar071d4272004-06-13 20:20:40 +00008158 }
8159 ++table->ga_len;
Bram Moolenaar071d4272004-06-13 20:20:40 +00008160 return (table->ga_len - 1 + ATTR_OFF);
8161}
8162
Bram Moolenaarf461c8e2005-06-25 23:04:51 +00008163/*
8164 * Clear all highlight tables.
8165 */
8166 void
8167clear_hl_tables()
8168{
8169 int i;
8170 attrentry_T *taep;
8171
8172#ifdef FEAT_GUI
8173 ga_clear(&gui_attr_table);
8174#endif
8175 for (i = 0; i < term_attr_table.ga_len; ++i)
8176 {
8177 taep = &(((attrentry_T *)term_attr_table.ga_data)[i]);
8178 vim_free(taep->ae_u.term.start);
8179 vim_free(taep->ae_u.term.stop);
8180 }
8181 ga_clear(&term_attr_table);
8182 ga_clear(&cterm_attr_table);
8183}
8184
Bram Moolenaar1c8f93f2006-03-12 22:10:07 +00008185#if defined(FEAT_SYN_HL) || defined(FEAT_SPELL) || defined(PROTO)
Bram Moolenaar217ad922005-03-20 22:37:15 +00008186/*
Bram Moolenaar30abd282005-06-22 22:35:10 +00008187 * Combine special attributes (e.g., for spelling) with other attributes
8188 * (e.g., for syntax highlighting).
8189 * "prim_attr" overrules "char_attr".
Bram Moolenaar217ad922005-03-20 22:37:15 +00008190 * This creates a new group when required.
8191 * Since we expect there to be few spelling mistakes we don't cache the
8192 * result.
8193 * Return the resulting attributes.
8194 */
8195 int
Bram Moolenaar30abd282005-06-22 22:35:10 +00008196hl_combine_attr(char_attr, prim_attr)
Bram Moolenaar217ad922005-03-20 22:37:15 +00008197 int char_attr;
Bram Moolenaar30abd282005-06-22 22:35:10 +00008198 int prim_attr;
Bram Moolenaar217ad922005-03-20 22:37:15 +00008199{
8200 attrentry_T *char_aep = NULL;
8201 attrentry_T *spell_aep;
8202 attrentry_T new_en;
8203
8204 if (char_attr == 0)
Bram Moolenaar30abd282005-06-22 22:35:10 +00008205 return prim_attr;
8206 if (char_attr <= HL_ALL && prim_attr <= HL_ALL)
8207 return char_attr | prim_attr;
Bram Moolenaar217ad922005-03-20 22:37:15 +00008208#ifdef FEAT_GUI
8209 if (gui.in_use)
8210 {
8211 if (char_attr > HL_ALL)
8212 char_aep = syn_gui_attr2entry(char_attr);
8213 if (char_aep != NULL)
8214 new_en = *char_aep;
8215 else
8216 {
8217 vim_memset(&new_en, 0, sizeof(new_en));
Bram Moolenaard5cdbeb2005-10-10 20:59:28 +00008218 new_en.ae_u.gui.fg_color = INVALCOLOR;
8219 new_en.ae_u.gui.bg_color = INVALCOLOR;
8220 new_en.ae_u.gui.sp_color = INVALCOLOR;
Bram Moolenaar217ad922005-03-20 22:37:15 +00008221 if (char_attr <= HL_ALL)
8222 new_en.ae_attr = char_attr;
8223 }
8224
Bram Moolenaar30abd282005-06-22 22:35:10 +00008225 if (prim_attr <= HL_ALL)
8226 new_en.ae_attr |= prim_attr;
Bram Moolenaar217ad922005-03-20 22:37:15 +00008227 else
8228 {
Bram Moolenaar30abd282005-06-22 22:35:10 +00008229 spell_aep = syn_gui_attr2entry(prim_attr);
Bram Moolenaar217ad922005-03-20 22:37:15 +00008230 if (spell_aep != NULL)
8231 {
8232 new_en.ae_attr |= spell_aep->ae_attr;
8233 if (spell_aep->ae_u.gui.fg_color != INVALCOLOR)
8234 new_en.ae_u.gui.fg_color = spell_aep->ae_u.gui.fg_color;
8235 if (spell_aep->ae_u.gui.bg_color != INVALCOLOR)
8236 new_en.ae_u.gui.bg_color = spell_aep->ae_u.gui.bg_color;
8237 if (spell_aep->ae_u.gui.sp_color != INVALCOLOR)
8238 new_en.ae_u.gui.sp_color = spell_aep->ae_u.gui.sp_color;
8239 if (spell_aep->ae_u.gui.font != NOFONT)
8240 new_en.ae_u.gui.font = spell_aep->ae_u.gui.font;
8241# ifdef FEAT_XFONTSET
8242 if (spell_aep->ae_u.gui.fontset != NOFONTSET)
8243 new_en.ae_u.gui.fontset = spell_aep->ae_u.gui.fontset;
8244# endif
8245 }
8246 }
8247 return get_attr_entry(&gui_attr_table, &new_en);
8248 }
8249#endif
8250
8251 if (t_colors > 1)
8252 {
8253 if (char_attr > HL_ALL)
8254 char_aep = syn_cterm_attr2entry(char_attr);
8255 if (char_aep != NULL)
8256 new_en = *char_aep;
8257 else
8258 {
8259 vim_memset(&new_en, 0, sizeof(new_en));
8260 if (char_attr <= HL_ALL)
8261 new_en.ae_attr = char_attr;
8262 }
8263
Bram Moolenaar30abd282005-06-22 22:35:10 +00008264 if (prim_attr <= HL_ALL)
8265 new_en.ae_attr |= prim_attr;
Bram Moolenaar217ad922005-03-20 22:37:15 +00008266 else
8267 {
Bram Moolenaar30abd282005-06-22 22:35:10 +00008268 spell_aep = syn_cterm_attr2entry(prim_attr);
Bram Moolenaar217ad922005-03-20 22:37:15 +00008269 if (spell_aep != NULL)
8270 {
8271 new_en.ae_attr |= spell_aep->ae_attr;
8272 if (spell_aep->ae_u.cterm.fg_color > 0)
8273 new_en.ae_u.cterm.fg_color = spell_aep->ae_u.cterm.fg_color;
8274 if (spell_aep->ae_u.cterm.bg_color > 0)
8275 new_en.ae_u.cterm.bg_color = spell_aep->ae_u.cterm.bg_color;
8276 }
8277 }
8278 return get_attr_entry(&cterm_attr_table, &new_en);
8279 }
8280
8281 if (char_attr > HL_ALL)
8282 char_aep = syn_term_attr2entry(char_attr);
8283 if (char_aep != NULL)
8284 new_en = *char_aep;
8285 else
8286 {
8287 vim_memset(&new_en, 0, sizeof(new_en));
8288 if (char_attr <= HL_ALL)
8289 new_en.ae_attr = char_attr;
8290 }
8291
Bram Moolenaar30abd282005-06-22 22:35:10 +00008292 if (prim_attr <= HL_ALL)
8293 new_en.ae_attr |= prim_attr;
Bram Moolenaar217ad922005-03-20 22:37:15 +00008294 else
8295 {
Bram Moolenaar3b181812005-12-17 22:10:02 +00008296 spell_aep = syn_term_attr2entry(prim_attr);
Bram Moolenaar217ad922005-03-20 22:37:15 +00008297 if (spell_aep != NULL)
8298 {
8299 new_en.ae_attr |= spell_aep->ae_attr;
8300 if (spell_aep->ae_u.term.start != NULL)
8301 {
8302 new_en.ae_u.term.start = spell_aep->ae_u.term.start;
8303 new_en.ae_u.term.stop = spell_aep->ae_u.term.stop;
8304 }
8305 }
8306 }
8307 return get_attr_entry(&term_attr_table, &new_en);
8308}
8309#endif
8310
Bram Moolenaar071d4272004-06-13 20:20:40 +00008311#ifdef FEAT_GUI
8312
8313 attrentry_T *
8314syn_gui_attr2entry(attr)
8315 int attr;
8316{
8317 attr -= ATTR_OFF;
8318 if (attr >= gui_attr_table.ga_len) /* did ":syntax clear" */
8319 return NULL;
8320 return &(GUI_ATTR_ENTRY(attr));
8321}
Bram Moolenaar071d4272004-06-13 20:20:40 +00008322#endif /* FEAT_GUI */
8323
Bram Moolenaar1c8f93f2006-03-12 22:10:07 +00008324/*
8325 * Get the highlight attributes (HL_BOLD etc.) from an attribute nr.
8326 * Only to be used when "attr" > HL_ALL.
8327 */
8328 int
8329syn_attr2attr(attr)
8330 int attr;
8331{
8332 attrentry_T *aep;
8333
8334#ifdef FEAT_GUI
8335 if (gui.in_use)
8336 aep = syn_gui_attr2entry(attr);
8337 else
8338#endif
8339 if (t_colors > 1)
8340 aep = syn_cterm_attr2entry(attr);
8341 else
8342 aep = syn_term_attr2entry(attr);
8343
8344 if (aep == NULL) /* highlighting not set */
8345 return 0;
8346 return aep->ae_attr;
8347}
8348
8349
Bram Moolenaar071d4272004-06-13 20:20:40 +00008350 attrentry_T *
8351syn_term_attr2entry(attr)
8352 int attr;
8353{
8354 attr -= ATTR_OFF;
8355 if (attr >= term_attr_table.ga_len) /* did ":syntax clear" */
8356 return NULL;
8357 return &(TERM_ATTR_ENTRY(attr));
8358}
8359
8360 attrentry_T *
8361syn_cterm_attr2entry(attr)
8362 int attr;
8363{
8364 attr -= ATTR_OFF;
8365 if (attr >= cterm_attr_table.ga_len) /* did ":syntax clear" */
8366 return NULL;
8367 return &(CTERM_ATTR_ENTRY(attr));
8368}
8369
8370#define LIST_ATTR 1
8371#define LIST_STRING 2
8372#define LIST_INT 3
8373
8374 static void
8375highlight_list_one(id)
8376 int id;
8377{
8378 struct hl_group *sgp;
8379 int didh = FALSE;
8380
8381 sgp = &HL_TABLE()[id - 1]; /* index is ID minus one */
8382
8383 didh = highlight_list_arg(id, didh, LIST_ATTR,
8384 sgp->sg_term, NULL, "term");
8385 didh = highlight_list_arg(id, didh, LIST_STRING,
8386 0, sgp->sg_start, "start");
8387 didh = highlight_list_arg(id, didh, LIST_STRING,
8388 0, sgp->sg_stop, "stop");
8389
8390 didh = highlight_list_arg(id, didh, LIST_ATTR,
8391 sgp->sg_cterm, NULL, "cterm");
8392 didh = highlight_list_arg(id, didh, LIST_INT,
8393 sgp->sg_cterm_fg, NULL, "ctermfg");
8394 didh = highlight_list_arg(id, didh, LIST_INT,
8395 sgp->sg_cterm_bg, NULL, "ctermbg");
8396
8397#ifdef FEAT_GUI
8398 didh = highlight_list_arg(id, didh, LIST_ATTR,
8399 sgp->sg_gui, NULL, "gui");
8400 didh = highlight_list_arg(id, didh, LIST_STRING,
8401 0, sgp->sg_gui_fg_name, "guifg");
8402 didh = highlight_list_arg(id, didh, LIST_STRING,
8403 0, sgp->sg_gui_bg_name, "guibg");
8404 didh = highlight_list_arg(id, didh, LIST_STRING,
Bram Moolenaar75c50c42005-06-04 22:06:24 +00008405 0, sgp->sg_gui_sp_name, "guisp");
8406 didh = highlight_list_arg(id, didh, LIST_STRING,
Bram Moolenaar071d4272004-06-13 20:20:40 +00008407 0, sgp->sg_font_name, "font");
8408#endif
8409
Bram Moolenaar661b1822005-07-28 22:36:45 +00008410 if (sgp->sg_link && !got_int)
Bram Moolenaar071d4272004-06-13 20:20:40 +00008411 {
8412 (void)syn_list_header(didh, 9999, id);
Bram Moolenaar661b1822005-07-28 22:36:45 +00008413 didh = TRUE;
Bram Moolenaar071d4272004-06-13 20:20:40 +00008414 msg_puts_attr((char_u *)"links to", hl_attr(HLF_D));
8415 msg_putchar(' ');
8416 msg_outtrans(HL_TABLE()[HL_TABLE()[id - 1].sg_link - 1].sg_name);
8417 }
Bram Moolenaar661b1822005-07-28 22:36:45 +00008418
Bram Moolenaar5e3cb7e2006-02-27 23:58:35 +00008419 if (!didh)
8420 highlight_list_arg(id, didh, LIST_STRING, 0, (char_u *)"cleared", "");
Bram Moolenaar661b1822005-07-28 22:36:45 +00008421#ifdef FEAT_EVAL
Bram Moolenaar5e3cb7e2006-02-27 23:58:35 +00008422 if (p_verbose > 0)
Bram Moolenaar661b1822005-07-28 22:36:45 +00008423 last_set_msg(sgp->sg_scriptID);
8424#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00008425}
8426
8427 static int
8428highlight_list_arg(id, didh, type, iarg, sarg, name)
8429 int id;
8430 int didh;
8431 int type;
8432 int iarg;
8433 char_u *sarg;
8434 char *name;
8435{
8436 char_u buf[100];
8437 char_u *ts;
8438 int i;
8439
Bram Moolenaar661b1822005-07-28 22:36:45 +00008440 if (got_int)
8441 return FALSE;
Bram Moolenaar071d4272004-06-13 20:20:40 +00008442 if (type == LIST_STRING ? (sarg != NULL) : (iarg != 0))
8443 {
8444 ts = buf;
8445 if (type == LIST_INT)
8446 sprintf((char *)buf, "%d", iarg - 1);
8447 else if (type == LIST_STRING)
8448 ts = sarg;
8449 else /* type == LIST_ATTR */
8450 {
8451 buf[0] = NUL;
8452 for (i = 0; hl_attr_table[i] != 0; ++i)
8453 {
8454 if (iarg & hl_attr_table[i])
8455 {
8456 if (buf[0] != NUL)
8457 STRCAT(buf, ",");
8458 STRCAT(buf, hl_name_table[i]);
8459 iarg &= ~hl_attr_table[i]; /* don't want "inverse" */
8460 }
8461 }
8462 }
8463
8464 (void)syn_list_header(didh,
8465 (int)(vim_strsize(ts) + STRLEN(name) + 1), id);
8466 didh = TRUE;
Bram Moolenaar661b1822005-07-28 22:36:45 +00008467 if (!got_int)
8468 {
Bram Moolenaar5e3cb7e2006-02-27 23:58:35 +00008469 if (*name != NUL)
8470 {
8471 MSG_PUTS_ATTR(name, hl_attr(HLF_D));
8472 MSG_PUTS_ATTR("=", hl_attr(HLF_D));
8473 }
Bram Moolenaar661b1822005-07-28 22:36:45 +00008474 msg_outtrans(ts);
8475 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00008476 }
8477 return didh;
8478}
8479
8480#if (((defined(FEAT_EVAL) || defined(FEAT_PRINTER))) && defined(FEAT_SYN_HL)) || defined(PROTO)
8481/*
8482 * Return "1" if highlight group "id" has attribute "flag".
8483 * Return NULL otherwise.
8484 */
8485 char_u *
8486highlight_has_attr(id, flag, modec)
8487 int id;
8488 int flag;
8489 int modec; /* 'g' for GUI, 'c' for cterm, 't' for term */
8490{
8491 int attr;
8492
8493 if (id <= 0 || id > highlight_ga.ga_len)
8494 return NULL;
8495
8496#ifdef FEAT_GUI
8497 if (modec == 'g')
8498 attr = HL_TABLE()[id - 1].sg_gui;
8499 else
8500#endif
8501 if (modec == 'c')
8502 attr = HL_TABLE()[id - 1].sg_cterm;
8503 else
8504 attr = HL_TABLE()[id - 1].sg_term;
8505
8506 if (attr & flag)
8507 return (char_u *)"1";
8508 return NULL;
8509}
8510#endif
8511
8512#if (defined(FEAT_SYN_HL) && defined(FEAT_EVAL)) || defined(PROTO)
8513/*
8514 * Return color name of highlight group "id".
8515 */
8516 char_u *
8517highlight_color(id, what, modec)
8518 int id;
Bram Moolenaar12682fd2010-03-10 13:43:49 +01008519 char_u *what; /* "font", "fg", "bg", "sp", "fg#", "bg#" or "sp#" */
Bram Moolenaar071d4272004-06-13 20:20:40 +00008520 int modec; /* 'g' for GUI, 'c' for cterm, 't' for term */
8521{
8522 static char_u name[20];
8523 int n;
Bram Moolenaare2cc9702005-03-15 22:43:58 +00008524 int fg = FALSE;
8525# ifdef FEAT_GUI
8526 int sp = FALSE;
Bram Moolenaar12682fd2010-03-10 13:43:49 +01008527 int font = FALSE;
Bram Moolenaare2cc9702005-03-15 22:43:58 +00008528# endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00008529
8530 if (id <= 0 || id > highlight_ga.ga_len)
8531 return NULL;
8532
Bram Moolenaar12682fd2010-03-10 13:43:49 +01008533 if (TOLOWER_ASC(what[0]) == 'f' && TOLOWER_ASC(what[1]) == 'g')
Bram Moolenaar071d4272004-06-13 20:20:40 +00008534 fg = TRUE;
Bram Moolenaare2cc9702005-03-15 22:43:58 +00008535# ifdef FEAT_GUI
Bram Moolenaar12682fd2010-03-10 13:43:49 +01008536 else if (TOLOWER_ASC(what[0]) == 'f' && TOLOWER_ASC(what[1]) == 'o'
8537 && TOLOWER_ASC(what[2]) == 'n' && TOLOWER_ASC(what[3]) == 't')
8538 font = TRUE;
8539 else if (TOLOWER_ASC(what[0]) == 's' && TOLOWER_ASC(what[1]) == 'p')
Bram Moolenaare2cc9702005-03-15 22:43:58 +00008540 sp = TRUE;
Bram Moolenaar12682fd2010-03-10 13:43:49 +01008541 else if (!(TOLOWER_ASC(what[0]) == 'b' && TOLOWER_ASC(what[1]) == 'g'))
8542 return NULL;
Bram Moolenaar071d4272004-06-13 20:20:40 +00008543 if (modec == 'g')
8544 {
Bram Moolenaar12682fd2010-03-10 13:43:49 +01008545 /* return font name */
8546 if (font)
8547 return HL_TABLE()[id - 1].sg_font_name;
8548
Bram Moolenaar071d4272004-06-13 20:20:40 +00008549 /* return #RRGGBB form (only possible when GUI is running) */
Bram Moolenaar12682fd2010-03-10 13:43:49 +01008550 if (gui.in_use && what[2] == '#')
Bram Moolenaar071d4272004-06-13 20:20:40 +00008551 {
8552 guicolor_T color;
8553 long_u rgb;
8554 static char_u buf[10];
8555
8556 if (fg)
8557 color = HL_TABLE()[id - 1].sg_gui_fg;
Bram Moolenaare2cc9702005-03-15 22:43:58 +00008558 else if (sp)
8559 color = HL_TABLE()[id - 1].sg_gui_sp;
Bram Moolenaar071d4272004-06-13 20:20:40 +00008560 else
8561 color = HL_TABLE()[id - 1].sg_gui_bg;
8562 if (color == INVALCOLOR)
8563 return NULL;
8564 rgb = gui_mch_get_rgb(color);
8565 sprintf((char *)buf, "#%02x%02x%02x",
8566 (unsigned)(rgb >> 16),
8567 (unsigned)(rgb >> 8) & 255,
8568 (unsigned)rgb & 255);
8569 return buf;
8570 }
8571 if (fg)
8572 return (HL_TABLE()[id - 1].sg_gui_fg_name);
Bram Moolenaare2cc9702005-03-15 22:43:58 +00008573 if (sp)
8574 return (HL_TABLE()[id - 1].sg_gui_sp_name);
Bram Moolenaar071d4272004-06-13 20:20:40 +00008575 return (HL_TABLE()[id - 1].sg_gui_bg_name);
8576 }
Bram Moolenaar12682fd2010-03-10 13:43:49 +01008577 if (font || sp)
8578 return NULL;
Bram Moolenaare2cc9702005-03-15 22:43:58 +00008579# endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00008580 if (modec == 'c')
8581 {
8582 if (fg)
8583 n = HL_TABLE()[id - 1].sg_cterm_fg - 1;
8584 else
8585 n = HL_TABLE()[id - 1].sg_cterm_bg - 1;
8586 sprintf((char *)name, "%d", n);
8587 return name;
8588 }
8589 /* term doesn't have color */
8590 return NULL;
8591}
8592#endif
8593
8594#if (defined(FEAT_SYN_HL) && defined(FEAT_GUI) && defined(FEAT_PRINTER)) \
8595 || defined(PROTO)
8596/*
8597 * Return color name of highlight group "id" as RGB value.
8598 */
8599 long_u
8600highlight_gui_color_rgb(id, fg)
8601 int id;
8602 int fg; /* TRUE = fg, FALSE = bg */
8603{
8604 guicolor_T color;
8605
8606 if (id <= 0 || id > highlight_ga.ga_len)
8607 return 0L;
8608
8609 if (fg)
8610 color = HL_TABLE()[id - 1].sg_gui_fg;
8611 else
8612 color = HL_TABLE()[id - 1].sg_gui_bg;
8613
8614 if (color == INVALCOLOR)
8615 return 0L;
8616
8617 return gui_mch_get_rgb(color);
8618}
8619#endif
8620
8621/*
8622 * Output the syntax list header.
8623 * Return TRUE when started a new line.
8624 */
8625 static int
8626syn_list_header(did_header, outlen, id)
8627 int did_header; /* did header already */
8628 int outlen; /* length of string that comes */
8629 int id; /* highlight group id */
8630{
8631 int endcol = 19;
8632 int newline = TRUE;
8633
8634 if (!did_header)
8635 {
8636 msg_putchar('\n');
Bram Moolenaar661b1822005-07-28 22:36:45 +00008637 if (got_int)
8638 return TRUE;
Bram Moolenaar071d4272004-06-13 20:20:40 +00008639 msg_outtrans(HL_TABLE()[id - 1].sg_name);
8640 endcol = 15;
8641 }
8642 else if (msg_col + outlen + 1 >= Columns)
Bram Moolenaar661b1822005-07-28 22:36:45 +00008643 {
Bram Moolenaar071d4272004-06-13 20:20:40 +00008644 msg_putchar('\n');
Bram Moolenaar661b1822005-07-28 22:36:45 +00008645 if (got_int)
8646 return TRUE;
8647 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00008648 else
8649 {
8650 if (msg_col >= endcol) /* wrap around is like starting a new line */
8651 newline = FALSE;
8652 }
8653
8654 if (msg_col >= endcol) /* output at least one space */
8655 endcol = msg_col + 1;
8656 if (Columns <= endcol) /* avoid hang for tiny window */
8657 endcol = Columns - 1;
8658
8659 msg_advance(endcol);
8660
8661 /* Show "xxx" with the attributes. */
8662 if (!did_header)
8663 {
8664 msg_puts_attr((char_u *)"xxx", syn_id2attr(id));
8665 msg_putchar(' ');
8666 }
8667
8668 return newline;
8669}
8670
8671/*
8672 * Set the attribute numbers for a highlight group.
8673 * Called after one of the attributes has changed.
8674 */
8675 static void
8676set_hl_attr(idx)
8677 int idx; /* index in array */
8678{
8679 attrentry_T at_en;
8680 struct hl_group *sgp = HL_TABLE() + idx;
8681
8682 /* The "Normal" group doesn't need an attribute number */
8683 if (sgp->sg_name_u != NULL && STRCMP(sgp->sg_name_u, "NORMAL") == 0)
8684 return;
8685
8686#ifdef FEAT_GUI
8687 /*
8688 * For the GUI mode: If there are other than "normal" highlighting
8689 * attributes, need to allocate an attr number.
8690 */
8691 if (sgp->sg_gui_fg == INVALCOLOR
8692 && sgp->sg_gui_bg == INVALCOLOR
Bram Moolenaare2cc9702005-03-15 22:43:58 +00008693 && sgp->sg_gui_sp == INVALCOLOR
Bram Moolenaar071d4272004-06-13 20:20:40 +00008694 && sgp->sg_font == NOFONT
8695# ifdef FEAT_XFONTSET
8696 && sgp->sg_fontset == NOFONTSET
8697# endif
8698 )
8699 {
8700 sgp->sg_gui_attr = sgp->sg_gui;
8701 }
8702 else
8703 {
8704 at_en.ae_attr = sgp->sg_gui;
8705 at_en.ae_u.gui.fg_color = sgp->sg_gui_fg;
8706 at_en.ae_u.gui.bg_color = sgp->sg_gui_bg;
Bram Moolenaare2cc9702005-03-15 22:43:58 +00008707 at_en.ae_u.gui.sp_color = sgp->sg_gui_sp;
Bram Moolenaar071d4272004-06-13 20:20:40 +00008708 at_en.ae_u.gui.font = sgp->sg_font;
8709# ifdef FEAT_XFONTSET
8710 at_en.ae_u.gui.fontset = sgp->sg_fontset;
8711# endif
8712 sgp->sg_gui_attr = get_attr_entry(&gui_attr_table, &at_en);
8713 }
8714#endif
8715 /*
8716 * For the term mode: If there are other than "normal" highlighting
8717 * attributes, need to allocate an attr number.
8718 */
8719 if (sgp->sg_start == NULL && sgp->sg_stop == NULL)
8720 sgp->sg_term_attr = sgp->sg_term;
8721 else
8722 {
8723 at_en.ae_attr = sgp->sg_term;
8724 at_en.ae_u.term.start = sgp->sg_start;
8725 at_en.ae_u.term.stop = sgp->sg_stop;
8726 sgp->sg_term_attr = get_attr_entry(&term_attr_table, &at_en);
8727 }
8728
8729 /*
8730 * For the color term mode: If there are other than "normal"
8731 * highlighting attributes, need to allocate an attr number.
8732 */
8733 if (sgp->sg_cterm_fg == 0 && sgp->sg_cterm_bg == 0)
8734 sgp->sg_cterm_attr = sgp->sg_cterm;
8735 else
8736 {
8737 at_en.ae_attr = sgp->sg_cterm;
8738 at_en.ae_u.cterm.fg_color = sgp->sg_cterm_fg;
8739 at_en.ae_u.cterm.bg_color = sgp->sg_cterm_bg;
8740 sgp->sg_cterm_attr = get_attr_entry(&cterm_attr_table, &at_en);
8741 }
8742}
8743
8744/*
8745 * Lookup a highlight group name and return it's ID.
8746 * If it is not found, 0 is returned.
8747 */
8748 int
8749syn_name2id(name)
8750 char_u *name;
8751{
8752 int i;
8753 char_u name_u[200];
8754
8755 /* Avoid using stricmp() too much, it's slow on some systems */
8756 /* Avoid alloc()/free(), these are slow too. ID names over 200 chars
8757 * don't deserve to be found! */
Bram Moolenaarce0842a2005-07-18 21:58:11 +00008758 vim_strncpy(name_u, name, 199);
Bram Moolenaar071d4272004-06-13 20:20:40 +00008759 vim_strup(name_u);
8760 for (i = highlight_ga.ga_len; --i >= 0; )
8761 if (HL_TABLE()[i].sg_name_u != NULL
8762 && STRCMP(name_u, HL_TABLE()[i].sg_name_u) == 0)
8763 break;
8764 return i + 1;
8765}
8766
8767#if defined(FEAT_EVAL) || defined(PROTO)
8768/*
8769 * Return TRUE if highlight group "name" exists.
8770 */
8771 int
8772highlight_exists(name)
8773 char_u *name;
8774{
8775 return (syn_name2id(name) > 0);
8776}
Bram Moolenaar910f66f2006-04-05 20:41:53 +00008777
8778# if defined(FEAT_SEARCH_EXTRA) || defined(PROTO)
8779/*
8780 * Return the name of highlight group "id".
8781 * When not a valid ID return an empty string.
8782 */
8783 char_u *
8784syn_id2name(id)
8785 int id;
8786{
Bram Moolenaar6ee10162007-07-26 20:58:42 +00008787 if (id <= 0 || id > highlight_ga.ga_len)
Bram Moolenaar910f66f2006-04-05 20:41:53 +00008788 return (char_u *)"";
8789 return HL_TABLE()[id - 1].sg_name;
8790}
8791# endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00008792#endif
8793
8794/*
8795 * Like syn_name2id(), but take a pointer + length argument.
8796 */
8797 int
8798syn_namen2id(linep, len)
8799 char_u *linep;
8800 int len;
8801{
8802 char_u *name;
8803 int id = 0;
8804
8805 name = vim_strnsave(linep, len);
8806 if (name != NULL)
8807 {
8808 id = syn_name2id(name);
8809 vim_free(name);
8810 }
8811 return id;
8812}
8813
8814/*
8815 * Find highlight group name in the table and return it's ID.
8816 * The argument is a pointer to the name and the length of the name.
8817 * If it doesn't exist yet, a new entry is created.
8818 * Return 0 for failure.
8819 */
8820 int
8821syn_check_group(pp, len)
8822 char_u *pp;
8823 int len;
8824{
8825 int id;
8826 char_u *name;
8827
8828 name = vim_strnsave(pp, len);
8829 if (name == NULL)
8830 return 0;
8831
8832 id = syn_name2id(name);
8833 if (id == 0) /* doesn't exist yet */
8834 id = syn_add_group(name);
8835 else
8836 vim_free(name);
8837 return id;
8838}
8839
8840/*
8841 * Add new highlight group and return it's ID.
8842 * "name" must be an allocated string, it will be consumed.
8843 * Return 0 for failure.
8844 */
8845 static int
8846syn_add_group(name)
8847 char_u *name;
8848{
8849 char_u *p;
8850
8851 /* Check that the name is ASCII letters, digits and underscore. */
8852 for (p = name; *p != NUL; ++p)
8853 {
8854 if (!vim_isprintc(*p))
8855 {
8856 EMSG(_("E669: Unprintable character in group name"));
Bram Moolenaarf5b63862009-12-16 17:13:44 +00008857 vim_free(name);
Bram Moolenaar071d4272004-06-13 20:20:40 +00008858 return 0;
8859 }
8860 else if (!ASCII_ISALNUM(*p) && *p != '_')
8861 {
8862 /* This is an error, but since there previously was no check only
8863 * give a warning. */
Bram Moolenaar2df6dcc2004-07-12 15:53:54 +00008864 msg_source(hl_attr(HLF_W));
Bram Moolenaar071d4272004-06-13 20:20:40 +00008865 MSG(_("W18: Invalid character in group name"));
8866 break;
8867 }
8868 }
8869
8870 /*
8871 * First call for this growarray: init growing array.
8872 */
8873 if (highlight_ga.ga_data == NULL)
8874 {
8875 highlight_ga.ga_itemsize = sizeof(struct hl_group);
8876 highlight_ga.ga_growsize = 10;
8877 }
8878
8879 /*
8880 * Make room for at least one other syntax_highlight entry.
8881 */
8882 if (ga_grow(&highlight_ga, 1) == FAIL)
8883 {
8884 vim_free(name);
8885 return 0;
8886 }
8887
8888 vim_memset(&(HL_TABLE()[highlight_ga.ga_len]), 0, sizeof(struct hl_group));
8889 HL_TABLE()[highlight_ga.ga_len].sg_name = name;
8890 HL_TABLE()[highlight_ga.ga_len].sg_name_u = vim_strsave_up(name);
8891#ifdef FEAT_GUI
8892 HL_TABLE()[highlight_ga.ga_len].sg_gui_bg = INVALCOLOR;
8893 HL_TABLE()[highlight_ga.ga_len].sg_gui_fg = INVALCOLOR;
Bram Moolenaare2cc9702005-03-15 22:43:58 +00008894 HL_TABLE()[highlight_ga.ga_len].sg_gui_sp = INVALCOLOR;
Bram Moolenaar071d4272004-06-13 20:20:40 +00008895#endif
8896 ++highlight_ga.ga_len;
Bram Moolenaar071d4272004-06-13 20:20:40 +00008897
8898 return highlight_ga.ga_len; /* ID is index plus one */
8899}
8900
8901/*
8902 * When, just after calling syn_add_group(), an error is discovered, this
8903 * function deletes the new name.
8904 */
8905 static void
8906syn_unadd_group()
8907{
8908 --highlight_ga.ga_len;
Bram Moolenaar071d4272004-06-13 20:20:40 +00008909 vim_free(HL_TABLE()[highlight_ga.ga_len].sg_name);
8910 vim_free(HL_TABLE()[highlight_ga.ga_len].sg_name_u);
8911}
8912
8913/*
8914 * Translate a group ID to highlight attributes.
8915 */
8916 int
8917syn_id2attr(hl_id)
8918 int hl_id;
8919{
8920 int attr;
8921 struct hl_group *sgp;
8922
8923 hl_id = syn_get_final_id(hl_id);
8924 sgp = &HL_TABLE()[hl_id - 1]; /* index is ID minus one */
8925
8926#ifdef FEAT_GUI
8927 /*
8928 * Only use GUI attr when the GUI is being used.
8929 */
8930 if (gui.in_use)
8931 attr = sgp->sg_gui_attr;
8932 else
8933#endif
8934 if (t_colors > 1)
8935 attr = sgp->sg_cterm_attr;
8936 else
8937 attr = sgp->sg_term_attr;
8938
8939 return attr;
8940}
8941
8942#ifdef FEAT_GUI
8943/*
8944 * Get the GUI colors and attributes for a group ID.
8945 * NOTE: the colors will be INVALCOLOR when not set, the color otherwise.
8946 */
8947 int
8948syn_id2colors(hl_id, fgp, bgp)
8949 int hl_id;
8950 guicolor_T *fgp;
8951 guicolor_T *bgp;
8952{
8953 struct hl_group *sgp;
8954
8955 hl_id = syn_get_final_id(hl_id);
8956 sgp = &HL_TABLE()[hl_id - 1]; /* index is ID minus one */
8957
8958 *fgp = sgp->sg_gui_fg;
8959 *bgp = sgp->sg_gui_bg;
8960 return sgp->sg_gui;
8961}
8962#endif
8963
8964/*
8965 * Translate a group ID to the final group ID (following links).
8966 */
8967 int
8968syn_get_final_id(hl_id)
8969 int hl_id;
8970{
8971 int count;
8972 struct hl_group *sgp;
8973
8974 if (hl_id > highlight_ga.ga_len || hl_id < 1)
8975 return 0; /* Can be called from eval!! */
8976
8977 /*
8978 * Follow links until there is no more.
8979 * Look out for loops! Break after 100 links.
8980 */
8981 for (count = 100; --count >= 0; )
8982 {
8983 sgp = &HL_TABLE()[hl_id - 1]; /* index is ID minus one */
8984 if (sgp->sg_link == 0 || sgp->sg_link > highlight_ga.ga_len)
8985 break;
8986 hl_id = sgp->sg_link;
8987 }
8988
8989 return hl_id;
8990}
8991
8992#ifdef FEAT_GUI
8993/*
8994 * Call this function just after the GUI has started.
8995 * It finds the font and color handles for the highlighting groups.
8996 */
8997 void
8998highlight_gui_started()
8999{
9000 int idx;
9001
9002 /* First get the colors from the "Normal" and "Menu" group, if set */
9003 set_normal_colors();
9004
9005 for (idx = 0; idx < highlight_ga.ga_len; ++idx)
9006 gui_do_one_color(idx, FALSE, FALSE);
9007
9008 highlight_changed();
9009}
9010
9011 static void
9012gui_do_one_color(idx, do_menu, do_tooltip)
9013 int idx;
9014 int do_menu; /* TRUE: might set the menu font */
9015 int do_tooltip; /* TRUE: might set the tooltip font */
9016{
9017 int didit = FALSE;
9018
9019 if (HL_TABLE()[idx].sg_font_name != NULL)
9020 {
9021 hl_do_font(idx, HL_TABLE()[idx].sg_font_name, FALSE, do_menu,
9022 do_tooltip);
9023 didit = TRUE;
9024 }
9025 if (HL_TABLE()[idx].sg_gui_fg_name != NULL)
9026 {
9027 HL_TABLE()[idx].sg_gui_fg =
9028 color_name2handle(HL_TABLE()[idx].sg_gui_fg_name);
9029 didit = TRUE;
9030 }
9031 if (HL_TABLE()[idx].sg_gui_bg_name != NULL)
9032 {
9033 HL_TABLE()[idx].sg_gui_bg =
9034 color_name2handle(HL_TABLE()[idx].sg_gui_bg_name);
9035 didit = TRUE;
9036 }
Bram Moolenaare2cc9702005-03-15 22:43:58 +00009037 if (HL_TABLE()[idx].sg_gui_sp_name != NULL)
9038 {
9039 HL_TABLE()[idx].sg_gui_sp =
9040 color_name2handle(HL_TABLE()[idx].sg_gui_sp_name);
9041 didit = TRUE;
9042 }
Bram Moolenaar071d4272004-06-13 20:20:40 +00009043 if (didit) /* need to get a new attr number */
9044 set_hl_attr(idx);
9045}
9046
9047#endif
9048
9049/*
9050 * Translate the 'highlight' option into attributes in highlight_attr[] and
9051 * set up the user highlights User1..9. If FEAT_STL_OPT is in use, a set of
9052 * corresponding highlights to use on top of HLF_SNC is computed.
9053 * Called only when the 'highlight' option has been changed and upon first
9054 * screen redraw after any :highlight command.
9055 * Return FAIL when an invalid flag is found in 'highlight'. OK otherwise.
9056 */
9057 int
9058highlight_changed()
9059{
9060 int hlf;
9061 int i;
9062 char_u *p;
9063 int attr;
9064 char_u *end;
9065 int id;
9066#ifdef USER_HIGHLIGHT
9067 char_u userhl[10];
9068# ifdef FEAT_STL_OPT
9069 int id_SNC = -1;
9070 int id_S = -1;
9071 int hlcnt;
9072# endif
9073#endif
9074 static int hl_flags[HLF_COUNT] = HL_FLAGS;
9075
9076 need_highlight_changed = FALSE;
9077
9078 /*
9079 * Clear all attributes.
9080 */
9081 for (hlf = 0; hlf < (int)HLF_COUNT; ++hlf)
9082 highlight_attr[hlf] = 0;
9083
9084 /*
9085 * First set all attributes to their default value.
9086 * Then use the attributes from the 'highlight' option.
9087 */
9088 for (i = 0; i < 2; ++i)
9089 {
9090 if (i)
9091 p = p_hl;
9092 else
9093 p = get_highlight_default();
9094 if (p == NULL) /* just in case */
9095 continue;
9096
9097 while (*p)
9098 {
9099 for (hlf = 0; hlf < (int)HLF_COUNT; ++hlf)
9100 if (hl_flags[hlf] == *p)
9101 break;
9102 ++p;
9103 if (hlf == (int)HLF_COUNT || *p == NUL)
9104 return FAIL;
9105
9106 /*
9107 * Allow several hl_flags to be combined, like "bu" for
9108 * bold-underlined.
9109 */
9110 attr = 0;
9111 for ( ; *p && *p != ','; ++p) /* parse upto comma */
9112 {
9113 if (vim_iswhite(*p)) /* ignore white space */
9114 continue;
9115
9116 if (attr > HL_ALL) /* Combination with ':' is not allowed. */
9117 return FAIL;
9118
9119 switch (*p)
9120 {
9121 case 'b': attr |= HL_BOLD;
9122 break;
9123 case 'i': attr |= HL_ITALIC;
9124 break;
9125 case '-':
9126 case 'n': /* no highlighting */
9127 break;
9128 case 'r': attr |= HL_INVERSE;
9129 break;
9130 case 's': attr |= HL_STANDOUT;
9131 break;
9132 case 'u': attr |= HL_UNDERLINE;
9133 break;
Bram Moolenaare2cc9702005-03-15 22:43:58 +00009134 case 'c': attr |= HL_UNDERCURL;
9135 break;
Bram Moolenaar071d4272004-06-13 20:20:40 +00009136 case ':': ++p; /* highlight group name */
9137 if (attr || *p == NUL) /* no combinations */
9138 return FAIL;
9139 end = vim_strchr(p, ',');
9140 if (end == NULL)
9141 end = p + STRLEN(p);
9142 id = syn_check_group(p, (int)(end - p));
9143 if (id == 0)
9144 return FAIL;
9145 attr = syn_id2attr(id);
9146 p = end - 1;
9147#if defined(FEAT_STL_OPT) && defined(USER_HIGHLIGHT)
9148 if (hlf == (int)HLF_SNC)
9149 id_SNC = syn_get_final_id(id);
9150 else if (hlf == (int)HLF_S)
9151 id_S = syn_get_final_id(id);
9152#endif
9153 break;
9154 default: return FAIL;
9155 }
9156 }
9157 highlight_attr[hlf] = attr;
9158
9159 p = skip_to_option_part(p); /* skip comma and spaces */
9160 }
9161 }
9162
9163#ifdef USER_HIGHLIGHT
9164 /* Setup the user highlights
9165 *
9166 * Temporarily utilize 10 more hl entries. Have to be in there
9167 * simultaneously in case of table overflows in get_attr_entry()
9168 */
9169# ifdef FEAT_STL_OPT
9170 if (ga_grow(&highlight_ga, 10) == FAIL)
9171 return FAIL;
9172 hlcnt = highlight_ga.ga_len;
9173 if (id_S == 0)
9174 { /* Make sure id_S is always valid to simplify code below */
Bram Moolenaar7db5fc82010-05-24 11:59:29 +02009175 vim_memset(&HL_TABLE()[hlcnt + 9], 0, sizeof(struct hl_group));
Bram Moolenaar071d4272004-06-13 20:20:40 +00009176 HL_TABLE()[hlcnt + 9].sg_term = highlight_attr[HLF_S];
9177 id_S = hlcnt + 10;
9178 }
9179# endif
9180 for (i = 0; i < 9; i++)
9181 {
9182 sprintf((char *)userhl, "User%d", i + 1);
9183 id = syn_name2id(userhl);
9184 if (id == 0)
9185 {
9186 highlight_user[i] = 0;
9187# ifdef FEAT_STL_OPT
9188 highlight_stlnc[i] = 0;
9189# endif
9190 }
9191 else
9192 {
9193# ifdef FEAT_STL_OPT
9194 struct hl_group *hlt = HL_TABLE();
9195# endif
9196
9197 highlight_user[i] = syn_id2attr(id);
9198# ifdef FEAT_STL_OPT
9199 if (id_SNC == 0)
9200 {
Bram Moolenaar7db5fc82010-05-24 11:59:29 +02009201 vim_memset(&hlt[hlcnt + i], 0, sizeof(struct hl_group));
Bram Moolenaar071d4272004-06-13 20:20:40 +00009202 hlt[hlcnt + i].sg_term = highlight_attr[HLF_SNC];
9203 hlt[hlcnt + i].sg_cterm = highlight_attr[HLF_SNC];
9204# ifdef FEAT_GUI
9205 hlt[hlcnt + i].sg_gui = highlight_attr[HLF_SNC];
9206# endif
9207 }
9208 else
9209 mch_memmove(&hlt[hlcnt + i],
9210 &hlt[id_SNC - 1],
9211 sizeof(struct hl_group));
9212 hlt[hlcnt + i].sg_link = 0;
9213
9214 /* Apply difference between UserX and HLF_S to HLF_SNC */
9215 hlt[hlcnt + i].sg_term ^=
9216 hlt[id - 1].sg_term ^ hlt[id_S - 1].sg_term;
9217 if (hlt[id - 1].sg_start != hlt[id_S - 1].sg_start)
9218 hlt[hlcnt + i].sg_start = hlt[id - 1].sg_start;
9219 if (hlt[id - 1].sg_stop != hlt[id_S - 1].sg_stop)
9220 hlt[hlcnt + i].sg_stop = hlt[id - 1].sg_stop;
9221 hlt[hlcnt + i].sg_cterm ^=
9222 hlt[id - 1].sg_cterm ^ hlt[id_S - 1].sg_cterm;
9223 if (hlt[id - 1].sg_cterm_fg != hlt[id_S - 1].sg_cterm_fg)
9224 hlt[hlcnt + i].sg_cterm_fg = hlt[id - 1].sg_cterm_fg;
9225 if (hlt[id - 1].sg_cterm_bg != hlt[id_S - 1].sg_cterm_bg)
9226 hlt[hlcnt + i].sg_cterm_bg = hlt[id - 1].sg_cterm_bg;
9227# ifdef FEAT_GUI
9228 hlt[hlcnt + i].sg_gui ^=
9229 hlt[id - 1].sg_gui ^ hlt[id_S - 1].sg_gui;
9230 if (hlt[id - 1].sg_gui_fg != hlt[id_S - 1].sg_gui_fg)
9231 hlt[hlcnt + i].sg_gui_fg = hlt[id - 1].sg_gui_fg;
9232 if (hlt[id - 1].sg_gui_bg != hlt[id_S - 1].sg_gui_bg)
9233 hlt[hlcnt + i].sg_gui_bg = hlt[id - 1].sg_gui_bg;
Bram Moolenaare2cc9702005-03-15 22:43:58 +00009234 if (hlt[id - 1].sg_gui_sp != hlt[id_S - 1].sg_gui_sp)
9235 hlt[hlcnt + i].sg_gui_sp = hlt[id - 1].sg_gui_sp;
Bram Moolenaar071d4272004-06-13 20:20:40 +00009236 if (hlt[id - 1].sg_font != hlt[id_S - 1].sg_font)
9237 hlt[hlcnt + i].sg_font = hlt[id - 1].sg_font;
9238# ifdef FEAT_XFONTSET
9239 if (hlt[id - 1].sg_fontset != hlt[id_S - 1].sg_fontset)
9240 hlt[hlcnt + i].sg_fontset = hlt[id - 1].sg_fontset;
9241# endif
9242# endif
9243 highlight_ga.ga_len = hlcnt + i + 1;
9244 set_hl_attr(hlcnt + i); /* At long last we can apply */
9245 highlight_stlnc[i] = syn_id2attr(hlcnt + i + 1);
9246# endif
9247 }
9248 }
9249# ifdef FEAT_STL_OPT
9250 highlight_ga.ga_len = hlcnt;
9251# endif
9252
9253#endif /* USER_HIGHLIGHT */
9254
9255 return OK;
9256}
9257
Bram Moolenaar4f688582007-07-24 12:34:30 +00009258#if defined(FEAT_CMDL_COMPL) || defined(PROTO)
Bram Moolenaar071d4272004-06-13 20:20:40 +00009259
9260static void highlight_list __ARGS((void));
9261static void highlight_list_two __ARGS((int cnt, int attr));
9262
9263/*
9264 * Handle command line completion for :highlight command.
9265 */
9266 void
9267set_context_in_highlight_cmd(xp, arg)
9268 expand_T *xp;
9269 char_u *arg;
9270{
9271 char_u *p;
9272
9273 /* Default: expand group names */
9274 xp->xp_context = EXPAND_HIGHLIGHT;
9275 xp->xp_pattern = arg;
Bram Moolenaar4f688582007-07-24 12:34:30 +00009276 include_link = 2;
9277 include_default = 1;
Bram Moolenaar071d4272004-06-13 20:20:40 +00009278
9279 /* (part of) subcommand already typed */
9280 if (*arg != NUL)
9281 {
9282 p = skiptowhite(arg);
9283 if (*p != NUL) /* past "default" or group name */
9284 {
Bram Moolenaar4f688582007-07-24 12:34:30 +00009285 include_default = 0;
Bram Moolenaar071d4272004-06-13 20:20:40 +00009286 if (STRNCMP("default", arg, p - arg) == 0)
9287 {
9288 arg = skipwhite(p);
9289 xp->xp_pattern = arg;
9290 p = skiptowhite(arg);
9291 }
9292 if (*p != NUL) /* past group name */
9293 {
Bram Moolenaar4f688582007-07-24 12:34:30 +00009294 include_link = 0;
Bram Moolenaar071d4272004-06-13 20:20:40 +00009295 if (arg[1] == 'i' && arg[0] == 'N')
9296 highlight_list();
9297 if (STRNCMP("link", arg, p - arg) == 0
9298 || STRNCMP("clear", arg, p - arg) == 0)
9299 {
9300 xp->xp_pattern = skipwhite(p);
9301 p = skiptowhite(xp->xp_pattern);
9302 if (*p != NUL) /* past first group name */
9303 {
9304 xp->xp_pattern = skipwhite(p);
9305 p = skiptowhite(xp->xp_pattern);
9306 }
9307 }
9308 if (*p != NUL) /* past group name(s) */
9309 xp->xp_context = EXPAND_NOTHING;
9310 }
9311 }
9312 }
9313}
9314
9315/*
9316 * List highlighting matches in a nice way.
9317 */
9318 static void
9319highlight_list()
9320{
9321 int i;
9322
9323 for (i = 10; --i >= 0; )
9324 highlight_list_two(i, hl_attr(HLF_D));
9325 for (i = 40; --i >= 0; )
9326 highlight_list_two(99, 0);
9327}
9328
9329 static void
9330highlight_list_two(cnt, attr)
9331 int cnt;
9332 int attr;
9333{
9334 msg_puts_attr((char_u *)("N \bI \b! \b" + cnt / 11), attr);
9335 msg_clr_eos();
9336 out_flush();
9337 ui_delay(cnt == 99 ? 40L : (long)cnt * 50L, FALSE);
9338}
9339
9340#endif /* FEAT_CMDL_COMPL */
9341
9342#if defined(FEAT_CMDL_COMPL) || (defined(FEAT_SYN_HL) && defined(FEAT_EVAL)) \
9343 || defined(FEAT_SIGNS) || defined(PROTO)
9344/*
9345 * Function given to ExpandGeneric() to obtain the list of group names.
9346 * Also used for synIDattr() function.
9347 */
Bram Moolenaar071d4272004-06-13 20:20:40 +00009348 char_u *
9349get_highlight_name(xp, idx)
Bram Moolenaar2c4278f2009-05-17 11:33:22 +00009350 expand_T *xp UNUSED;
Bram Moolenaar071d4272004-06-13 20:20:40 +00009351 int idx;
9352{
Bram Moolenaar071d4272004-06-13 20:20:40 +00009353#ifdef FEAT_CMDL_COMPL
Bram Moolenaar4f688582007-07-24 12:34:30 +00009354 if (idx == highlight_ga.ga_len && include_none != 0)
9355 return (char_u *)"none";
9356 if (idx == highlight_ga.ga_len + include_none && include_default != 0)
Bram Moolenaar071d4272004-06-13 20:20:40 +00009357 return (char_u *)"default";
Bram Moolenaar4f688582007-07-24 12:34:30 +00009358 if (idx == highlight_ga.ga_len + include_none + include_default
9359 && include_link != 0)
9360 return (char_u *)"link";
9361 if (idx == highlight_ga.ga_len + include_none + include_default + 1
9362 && include_link != 0)
9363 return (char_u *)"clear";
9364#endif
Bram Moolenaar071d4272004-06-13 20:20:40 +00009365 if (idx < 0 || idx >= highlight_ga.ga_len)
9366 return NULL;
9367 return HL_TABLE()[idx].sg_name;
9368}
9369#endif
9370
Bram Moolenaar4f688582007-07-24 12:34:30 +00009371#if defined(FEAT_GUI) || defined(PROTO)
Bram Moolenaar071d4272004-06-13 20:20:40 +00009372/*
9373 * Free all the highlight group fonts.
9374 * Used when quitting for systems which need it.
9375 */
9376 void
9377free_highlight_fonts()
9378{
9379 int idx;
9380
9381 for (idx = 0; idx < highlight_ga.ga_len; ++idx)
9382 {
9383 gui_mch_free_font(HL_TABLE()[idx].sg_font);
9384 HL_TABLE()[idx].sg_font = NOFONT;
9385# ifdef FEAT_XFONTSET
9386 gui_mch_free_fontset(HL_TABLE()[idx].sg_fontset);
9387 HL_TABLE()[idx].sg_fontset = NOFONTSET;
9388# endif
9389 }
9390
9391 gui_mch_free_font(gui.norm_font);
9392# ifdef FEAT_XFONTSET
9393 gui_mch_free_fontset(gui.fontset);
9394# endif
9395# ifndef HAVE_GTK2
9396 gui_mch_free_font(gui.bold_font);
9397 gui_mch_free_font(gui.ital_font);
9398 gui_mch_free_font(gui.boldital_font);
9399# endif
9400}
9401#endif
9402
9403/**************************************
9404 * End of Highlighting stuff *
9405 **************************************/