diff options
author | Bram Moolenaar <Bram@vim.org> | 2005-01-08 21:49:45 +0000 |
---|---|---|
committer | Bram Moolenaar <Bram@vim.org> | 2005-01-08 21:49:45 +0000 |
commit | 6cc1619799b39016acea738c1ae7ea17d2588b9c (patch) | |
tree | 63c490f800930f83d129eadf8450d3309eab5052 /src | |
parent | 9588a0f72bc1f72962e3e327d29775d3a970e579 (diff) | |
download | vim-git-6cc1619799b39016acea738c1ae7ea17d2588b9c.tar.gz |
updated for version 7.0035v7.0035
Diffstat (limited to 'src')
-rw-r--r-- | src/eval.c | 263 | ||||
-rw-r--r-- | src/gui_gtk_x11.c | 21 | ||||
-rw-r--r-- | src/gui_x11.c | 30 | ||||
-rw-r--r-- | src/proto/gui_riscos.pro | 3 | ||||
-rw-r--r-- | src/proto/gui_x11.pro | 3 |
5 files changed, 263 insertions, 57 deletions
diff --git a/src/eval.c b/src/eval.c index ba4fad4ea..3bc05c987 100644 --- a/src/eval.c +++ b/src/eval.c @@ -324,6 +324,7 @@ static long list_len __ARGS((listvar *l)); static int list_equal __ARGS((listvar *l1, listvar *l2, int ic)); static int tv_equal __ARGS((typeval *tv1, typeval *tv2, int ic)); static listitem *list_find __ARGS((listvar *l, long n)); +static long list_idx_of_item __ARGS((listvar *l, listitem *item)); static listitem *list_find_ext __ARGS((listvar *l, long *ip)); static void list_append __ARGS((listvar *l, listitem *item)); static int list_append_tv __ARGS((listvar *l, typeval *tv)); @@ -440,6 +441,8 @@ static void f_mapcheck __ARGS((typeval *argvars, typeval *rettv)); static void f_match __ARGS((typeval *argvars, typeval *rettv)); static void f_matchend __ARGS((typeval *argvars, typeval *rettv)); static void f_matchstr __ARGS((typeval *argvars, typeval *rettv)); +static void f_max __ARGS((typeval *argvars, typeval *rettv)); +static void f_min __ARGS((typeval *argvars, typeval *rettv)); static void f_mode __ARGS((typeval *argvars, typeval *rettv)); static void f_nextnonblank __ARGS((typeval *argvars, typeval *rettv)); static void f_nr2char __ARGS((typeval *argvars, typeval *rettv)); @@ -1540,8 +1543,8 @@ ex_let_one(arg, tv, copy, endchars) } /* - * Set a variable with an index: "name[expr]", "name[expr][expr]", etc. - * Only works if "name" is an existing List. + * Set a variable with an index: "name[expr]", "name[expr:expr]", + * "name[expr][expr]", etc. Only works if "name" is an existing List. * "ip" points to the first '['. * Returns a pointer to just after the last used ']'; NULL for error. */ @@ -1557,9 +1560,15 @@ set_var_idx(name, ip, rettv, copy, endchars) int c1; char_u *p; typeval var1; + typeval var2; + int range = FALSE; typeval *tv; - long n; - listitem *item; + long n1 = 0, n2 = 0; + int empty1, empty2 = FALSE; + listitem *item = NULL; + listitem *ni; + listitem *ri; + listvar *l = NULL; c1 = *ip; *ip = NUL; @@ -1579,28 +1588,122 @@ set_var_idx(name, ip, rettv, copy, endchars) p = NULL; break; } - p = skipwhite(p + 1); - if (eval1(&p, &var1, TRUE) == FAIL) /* recursive! */ + if (range) { + EMSG(_("E708: [:] must come last")); p = NULL; break; } + + /* Get the index [expr] or the first index [expr: ]. */ + p = skipwhite(p + 1); + if (*p == ':') + empty1 = TRUE; + else + { + empty1 = FALSE; + if (eval1(&p, &var1, TRUE) == FAIL) /* recursive! */ + { + p = NULL; + break; + } + } + + /* Optionally get the second index [ :expr]. */ + if (*p == ':') + { + if (rettv->v_type != VAR_LIST || rettv->vval.v_list == NULL) + { + EMSG(_("E709: [:] requires a List value")); + p = NULL; + if (!empty1) + clear_tv(&var1); + break; + } + p = skipwhite(p + 1); + if (*p == ']') + empty2 = TRUE; + else + { + empty2 = FALSE; + if (eval1(&p, &var2, TRUE) == FAIL) /* recursive! */ + { + p = NULL; + if (!empty1) + clear_tv(&var1); + break; + } + } + range = TRUE; + } + else + range = FALSE; + if (*p != ']') { EMSG(_(e_missbrac)); - clear_tv(&var1); + if (!empty1) + clear_tv(&var1); + if (range && !empty2) + clear_tv(&var2); p = NULL; break; } - n = get_tv_number(&var1); - clear_tv(&var1); - item = list_find(tv->vval.v_list, n); + + /* + * Get the number and item for the only or first index. + */ + if (empty1) + n1 = 0; + else + { + n1 = get_tv_number(&var1); + clear_tv(&var1); + } + l = tv->vval.v_list; + item = list_find(l, n1); if (item == NULL) { - EMSGN(_(e_listidx), n); + EMSGN(_(e_listidx), n1); p = NULL; + if (range && !empty2) + clear_tv(&var2); break; } + + /* + * May need to find the item or absolute index for the second index of + * a range. + * When no index given: "empty2" is TRUE. + * Otherwise "n2" is set to the second index. + */ + if (range && !empty2) + { + n2 = get_tv_number(&var2); + clear_tv(&var2); + if (n2 < 0) + { + ni = list_find(l, n2); + if (ni == NULL) + { + EMSGN(_(e_listidx), n2); + p = NULL; + break; + } + n2 = list_idx_of_item(l, ni); + } + + /* Check that n2 isn't before n1. */ + if (n1 < 0) + n1 = list_idx_of_item(l, item); + if (n2 < n1) + { + EMSGN(_(e_listidx), n2); + p = NULL; + break; + } + } + tv = &item->li_tv; } @@ -1611,11 +1714,47 @@ set_var_idx(name, ip, rettv, copy, endchars) EMSG(_(e_letunexp)); p = NULL; } + else if (range) + { + /* + * Assign the List values to the list items. + */ + for (ri = rettv->vval.v_list->lv_first; ri != NULL; ) + { + clear_tv(&item->li_tv); + copy_tv(&ri->li_tv, &item->li_tv); + ri = ri->li_next; + if (ri == NULL || (!empty2 && n2 == n1)) + break; + if (item->li_next == NULL) + { + /* Need to add an empty item. */ + ni = listitem_alloc(); + if (ni == NULL) + { + ri = NULL; + break; + } + ni->li_tv.v_type = VAR_NUMBER; + ni->li_tv.vval.v_number = 0; + list_append(l, ni); + } + item = item->li_next; + ++n1; + } + if (ri != NULL) + EMSG(_("E710: List value has more items than target")); + else if (empty2 ? item != NULL && item->li_next != NULL : n1 != n2) + EMSG(_("E711: List value has not enough items")); + } else { + /* + * Assign the value to the variable or list item. + */ clear_tv(tv); if (copy) - copy_tv(tv, rettv); + copy_tv(rettv, tv); else { *tv = *rettv; @@ -3785,6 +3924,28 @@ list_find(l, n) } /* + * Locate "item" list "l" and return its index. + * Returns -1 when "item" is not in the list. + */ + static long +list_idx_of_item(l, item) + listvar *l; + listitem *item; +{ + long idx = 0; + listitem *li; + + if (l == NULL) + return -1; + idx = 0; + for (li = l->lv_first; li != NULL && li != item; li = li->li_next) + ++idx; + if (li == NULL) + return -1; + return idx;; +} + +/* * Like list_find(), but also find an item just past the end. * "*ip" is the item to find. * When found "*ip" is set to zero, when not found "*ip" is non-zero. @@ -4253,6 +4414,8 @@ static struct fst {"match", 2, 4, f_match}, {"matchend", 2, 4, f_matchend}, {"matchstr", 2, 4, f_matchstr}, + {"max", 1, 1, f_max}, + {"min", 1, 1, f_min}, {"mode", 0, 0, f_mode}, {"nextnonblank", 1, 1, f_nextnonblank}, {"nr2char", 1, 1, f_nr2char}, @@ -8085,6 +8248,67 @@ f_matchstr(argvars, rettv) find_some_match(argvars, rettv, 2); } +static void max_min __ARGS((typeval *argvars, typeval *rettv, int domax)); + + static void +max_min(argvars, rettv, domax) + typeval *argvars; + typeval *rettv; + int domax; +{ + listvar *l; + listitem *li; + long n = 0; + long i; + + if (argvars[0].v_type == VAR_LIST) + { + l = argvars[0].vval.v_list; + if (l != NULL) + { + li = l->lv_first; + if (li != NULL) + { + n = get_tv_number(&li->li_tv); + while (1) + { + li = li->li_next; + if (li == NULL) + break; + i = get_tv_number(&li->li_tv); + if (domax ? i > n : i < n) + n = i; + } + } + } + } + else + EMSG(_(e_listreq)); + rettv->vval.v_number = n; +} + +/* + * "max()" function + */ + static void +f_max(argvars, rettv) + typeval *argvars; + typeval *rettv; +{ + max_min(argvars, rettv, TRUE); +} + +/* + * "min()" function + */ + static void +f_min(argvars, rettv) + typeval *argvars; + typeval *rettv; +{ + max_min(argvars, rettv, FALSE); +} + /* * "mode()" function */ @@ -10247,10 +10471,17 @@ f_type(argvars, rettv) typeval *argvars; typeval *rettv; { - if (argvars[0].v_type == VAR_NUMBER) - rettv->vval.v_number = 0; - else - rettv->vval.v_number = 1; + int n; + + switch (argvars[0].v_type) + { + case VAR_NUMBER: n = 0; break; + case VAR_STRING: n = 1; break; + case VAR_FUNC: n = 2; break; + case VAR_LIST: n = 3; break; + default: EMSG2(_(e_intern2), "f_type()"); n = 0; break; + } + rettv->vval.v_number = n; } /* diff --git a/src/gui_gtk_x11.c b/src/gui_gtk_x11.c index abf3fbeb7..0ff685aea 100644 --- a/src/gui_gtk_x11.c +++ b/src/gui_gtk_x11.c @@ -6249,25 +6249,12 @@ gui_mch_get_rgb(guicolor_T pixel) } /* - * Get current y mouse coordinate in text window. - * Return -1 when unknown. + * Get current mouse coordinates in text window. */ - int -gui_mch_get_mouse_x(void) -{ - int win_x; - - gdk_window_get_pointer(gui.drawarea->window, &win_x, NULL, NULL); - return win_x; -} - - int -gui_mch_get_mouse_y(void) + void +gui_mch_getmouse(int *x, int *y) { - int win_y; - - gdk_window_get_pointer(gui.drawarea->window, NULL, &win_y, NULL); - return win_y; + gdk_window_get_pointer(gui.drawarea->window, x, y, NULL); } void diff --git a/src/gui_x11.c b/src/gui_x11.c index 88ee91c0c..6d990c5d1 100644 --- a/src/gui_x11.c +++ b/src/gui_x11.c @@ -3276,33 +3276,23 @@ gui_x11_callbacks(textArea, vimForm) } /* - * Get current y mouse coordinate in text window. - * Return -1 when unknown. + * Get current mouse coordinates in text window. */ - int -gui_mch_get_mouse_x() -{ - int rootx, rooty, winx, winy; - Window root, child; - unsigned int mask; - - if (gui.wid && XQueryPointer(gui.dpy, gui.wid, &root, &child, - &rootx, &rooty, &winx, &winy, &mask)) - return winx; - return -1; -} - - int -gui_mch_get_mouse_y() + void +gui_mch_getmouse(int *x, int *y) { int rootx, rooty, winx, winy; Window root, child; unsigned int mask; if (gui.wid && XQueryPointer(gui.dpy, gui.wid, &root, &child, - &rootx, &rooty, &winx, &winy, &mask)) - return winy; - return -1; + &rootx, &rooty, &winx, &winy, &mask)) { + *x = winx; + *y = winy; + } else { + *x = -1; + *y = -1; + } } void diff --git a/src/proto/gui_riscos.pro b/src/proto/gui_riscos.pro index 3c23fe004..ab4e8bc9b 100644 --- a/src/proto/gui_riscos.pro +++ b/src/proto/gui_riscos.pro @@ -53,8 +53,7 @@ void gui_mch_start_blink __ARGS((void)); void process_event __ARGS((int event, int *block)); void gui_mch_show_popupmenu __ARGS((vimmenu_T *menu)); long_u gui_mch_get_rgb __ARGS((guicolor_T pixel)); -int gui_mch_get_mouse_x __ARGS((void)); -int gui_mch_get_mouse_y __ARGS((void)); +void gui_mch_getmouse __ARGS((int *x, int *y)); void gui_mch_setmouse __ARGS((int x, int y)); void gui_mch_drawsign __ARGS((int row, int col, int typenr)); void gui_mch_destroy_sign __ARGS((XImage *sign)); diff --git a/src/proto/gui_x11.pro b/src/proto/gui_x11.pro index f14e26f27..7308cd3de 100644 --- a/src/proto/gui_x11.pro +++ b/src/proto/gui_x11.pro @@ -57,8 +57,7 @@ void gui_mch_stop_blink __ARGS((void)); void gui_mch_start_blink __ARGS((void)); long_u gui_mch_get_rgb __ARGS((guicolor_T pixel)); void gui_x11_callbacks __ARGS((Widget textArea, Widget vimForm)); -int gui_mch_get_mouse_x __ARGS((void)); -int gui_mch_get_mouse_y __ARGS((void)); +void gui_mch_getmouse __ARGS((int *x, int *y)); void gui_mch_setmouse __ARGS((int x, int y)); XButtonPressedEvent *gui_x11_get_last_mouse_event __ARGS((void)); void gui_mch_drawsign __ARGS((int row, int col, int typenr)); |