vim-patch:8.2.0868: trim() always trims both ends

Problem:    trim() always trims both ends.
Solution:   Add an argument to only trim the beginning or end. (Yegappan
            Lakshmanan, closes vim/vim#6126)
2245ae18e3
This commit is contained in:
skippi
2020-10-10 00:24:52 -05:00
parent 0af18a6a43
commit 8c9d9fa304
4 changed files with 71 additions and 37 deletions

View File

@@ -2468,7 +2468,8 @@ tolower({expr}) String the String {expr} switched to lowercase
toupper({expr}) String the String {expr} switched to uppercase toupper({expr}) String the String {expr} switched to uppercase
tr({src}, {fromstr}, {tostr}) String translate chars of {src} in {fromstr} tr({src}, {fromstr}, {tostr}) String translate chars of {src} in {fromstr}
to chars in {tostr} to chars in {tostr}
trim({text} [, {mask}]) String trim characters in {mask} from {text} trim({text} [, {mask} [, {dir}]])
String trim characters in {mask} from {text}
trunc({expr}) Float truncate Float {expr} trunc({expr}) Float truncate Float {expr}
type({name}) Number type of variable {name} type({name}) Number type of variable {name}
undofile({name}) String undo file name for {name} undofile({name}) String undo file name for {name}
@@ -9048,21 +9049,28 @@ tr({src}, {fromstr}, {tostr}) *tr()*
echo tr("<blob>", "<>", "{}") echo tr("<blob>", "<>", "{}")
< returns "{blob}" < returns "{blob}"
trim({text} [, {mask}]) *trim()* trim({text} [, {mask} [, {dir}]]) *trim()*
Return {text} as a String where any character in {mask} is Return {text} as a String where any character in {mask} is
removed from the beginning and end of {text}. removed from the beginning and/or end of {text}.
If {mask} is not given, {mask} is all characters up to 0x20, If {mask} is not given, {mask} is all characters up to 0x20,
which includes Tab, space, NL and CR, plus the non-breaking which includes Tab, space, NL and CR, plus the non-breaking
space character 0xa0. space character 0xa0.
This code deals with multibyte characters properly. The optional {dir} argument specifies where to remove the
characters:
0 remove from the beginning and end of {text}
1 remove only at the beginning of {text}
2 remove only at the end of {text}
When omitted both ends are trimmed.
This function deals with multibyte characters properly.
Examples: > Examples: >
echo trim(" some text ") echo trim(" some text ")
< returns "some text" > < returns "some text" >
echo trim(" \r\t\t\r RESERVE \t\n\x0B\xA0") . "_TAIL" echo trim(" \r\t\t\r RESERVE \t\n\x0B\xA0") . "_TAIL"
< returns "RESERVE_TAIL" > < returns "RESERVE_TAIL" >
echo trim("rm<Xrm<>X>rrm", "rm<>") echo trim("rm<Xrm<>X>rrm", "rm<>")
< returns "Xrm<>X" (characters in the middle are not removed) < returns "Xrm<>X" (characters in the middle are not removed) >
echo trim(" vim ", " ", 2)
< returns " vim"
trunc({expr}) *trunc()* trunc({expr}) *trunc()*
Return the largest integral value with magnitude less than or Return the largest integral value with magnitude less than or

View File

@@ -371,7 +371,7 @@ return {
tolower={args=1}, tolower={args=1},
toupper={args=1}, toupper={args=1},
tr={args=3}, tr={args=3},
trim={args={1,2}}, trim={args={1,3}},
trunc={args=1, func="float_op_wrapper", data="&trunc"}, trunc={args=1, func="float_op_wrapper", data="&trunc"},
type={args=1}, type={args=1},
undofile={args=1}, undofile={args=1},

View File

@@ -10823,17 +10823,32 @@ static void f_trim(typval_T *argvars, typval_T *rettv, FunPtr fptr)
const char_u *prev; const char_u *prev;
const char_u *p; const char_u *p;
int c1; int c1;
int dir = 0;
rettv->v_type = VAR_STRING; rettv->v_type = VAR_STRING;
if (head == NULL) {
rettv->vval.v_string = NULL; rettv->vval.v_string = NULL;
if (head == NULL) {
return; return;
} }
if (argvars[1].v_type == VAR_STRING) { if (argvars[1].v_type == VAR_STRING) {
mask = (const char_u *)tv_get_string_buf_chk(&argvars[1], buf2); mask = (const char_u *)tv_get_string_buf_chk(&argvars[1], buf2);
if (argvars[2].v_type != VAR_UNKNOWN) {
bool error = false;
// leading or trailing characters to trim
dir = (int)tv_get_number_chk(&argvars[2], &error);
if (error) {
return;
}
if (dir < 0 || dir > 2) {
emsgf(_(e_invarg2), tv_get_string(&argvars[2]));
return;
}
}
} }
if (dir == 0 || dir == 1) {
// Trim leading characters
while (*head != NUL) { while (*head != NUL) {
c1 = PTR2CHAR(head); c1 = PTR2CHAR(head);
if (mask == NULL) { if (mask == NULL) {
@@ -10852,8 +10867,12 @@ static void f_trim(typval_T *argvars, typval_T *rettv, FunPtr fptr)
} }
MB_PTR_ADV(head); MB_PTR_ADV(head);
} }
}
for (tail = head + STRLEN(head); tail > head; tail = prev) { tail = head + STRLEN(head);
if (dir == 0 || dir == 2) {
// Trim trailing characters
for (; tail > head; tail = prev) {
prev = tail; prev = tail;
MB_PTR_BACK(head, prev); MB_PTR_BACK(head, prev);
c1 = PTR2CHAR(prev); c1 = PTR2CHAR(prev);
@@ -10872,6 +10891,7 @@ static void f_trim(typval_T *argvars, typval_T *rettv, FunPtr fptr)
} }
} }
} }
}
rettv->vval.v_string = vim_strnsave(head, (int)(tail - head)); rettv->vval.v_string = vim_strnsave(head, (int)(tail - head));
} }

View File

@@ -1083,6 +1083,12 @@ func Test_trim()
call assert_equal("", trim("", "")) call assert_equal("", trim("", ""))
call assert_equal("a", trim("a", "")) call assert_equal("a", trim("a", ""))
call assert_equal("", trim("", "a")) call assert_equal("", trim("", "a"))
call assert_equal("vim", trim(" vim ", " ", 0))
call assert_equal("vim ", trim(" vim ", " ", 1))
call assert_equal(" vim", trim(" vim ", " ", 2))
call assert_fails('call trim(" vim ", " ", [])', 'E745:')
call assert_fails('call trim(" vim ", " ", -1)', 'E475:')
call assert_fails('call trim(" vim ", " ", 3)', 'E475:')
let chars = join(map(range(1, 0x20) + [0xa0], {n -> nr2char(n)}), '') let chars = join(map(range(1, 0x20) + [0xa0], {n -> nr2char(n)}), '')
call assert_equal("x", trim(chars . "x" . chars)) call assert_equal("x", trim(chars . "x" . chars))