Bram Moolenaar | 1735bc9 | 2016-03-14 23:05:14 +0100 | [diff] [blame] | 1 | " Test binding arguments to a Funcref. |
| 2 | |
| 3 | func MyFunc(arg1, arg2, arg3) |
| 4 | return a:arg1 . '/' . a:arg2 . '/' . a:arg3 |
| 5 | endfunc |
| 6 | |
| 7 | func MySort(up, one, two) |
| 8 | if a:one == a:two |
| 9 | return 0 |
| 10 | endif |
| 11 | if a:up |
Bram Moolenaar | 790500a | 2016-03-15 11:05:45 +0100 | [diff] [blame] | 12 | return a:one > a:two ? 1 : -1 |
Bram Moolenaar | 1735bc9 | 2016-03-14 23:05:14 +0100 | [diff] [blame] | 13 | endif |
Bram Moolenaar | 790500a | 2016-03-15 11:05:45 +0100 | [diff] [blame] | 14 | return a:one < a:two ? 1 : -1 |
Bram Moolenaar | 1735bc9 | 2016-03-14 23:05:14 +0100 | [diff] [blame] | 15 | endfunc |
| 16 | |
| 17 | func Test_partial_args() |
| 18 | let Cb = function('MyFunc', ["foo", "bar"]) |
Bram Moolenaar | 6563903 | 2016-03-16 21:40:30 +0100 | [diff] [blame] | 19 | |
| 20 | call Cb("zzz") |
Bram Moolenaar | 1735bc9 | 2016-03-14 23:05:14 +0100 | [diff] [blame] | 21 | call assert_equal("foo/bar/xxx", Cb("xxx")) |
| 22 | call assert_equal("foo/bar/yyy", call(Cb, ["yyy"])) |
Bram Moolenaar | 8a1bb04 | 2016-03-17 21:11:53 +0100 | [diff] [blame] | 23 | let Cb2 = function(Cb) |
| 24 | call assert_equal("foo/bar/zzz", Cb2("zzz")) |
| 25 | let Cb3 = function(Cb, ["www"]) |
| 26 | call assert_equal("foo/bar/www", Cb3()) |
Bram Moolenaar | 1735bc9 | 2016-03-14 23:05:14 +0100 | [diff] [blame] | 27 | |
Bram Moolenaar | 346418c | 2016-03-15 12:36:08 +0100 | [diff] [blame] | 28 | let Cb = function('MyFunc', []) |
| 29 | call assert_equal("a/b/c", Cb("a", "b", "c")) |
Bram Moolenaar | 8a1bb04 | 2016-03-17 21:11:53 +0100 | [diff] [blame] | 30 | let Cb2 = function(Cb, []) |
| 31 | call assert_equal("a/b/d", Cb2("a", "b", "d")) |
| 32 | let Cb3 = function(Cb, ["a", "b"]) |
| 33 | call assert_equal("a/b/e", Cb3("e")) |
Bram Moolenaar | 346418c | 2016-03-15 12:36:08 +0100 | [diff] [blame] | 34 | |
Bram Moolenaar | 1735bc9 | 2016-03-14 23:05:14 +0100 | [diff] [blame] | 35 | let Sort = function('MySort', [1]) |
| 36 | call assert_equal([1, 2, 3], sort([3, 1, 2], Sort)) |
| 37 | let Sort = function('MySort', [0]) |
| 38 | call assert_equal([3, 2, 1], sort([3, 1, 2], Sort)) |
| 39 | endfunc |
| 40 | |
| 41 | func MyDictFunc(arg1, arg2) dict |
| 42 | return self.name . '/' . a:arg1 . '/' . a:arg2 |
| 43 | endfunc |
| 44 | |
| 45 | func Test_partial_dict() |
| 46 | let dict = {'name': 'hello'} |
| 47 | let Cb = function('MyDictFunc', ["foo", "bar"], dict) |
| 48 | call assert_equal("hello/foo/bar", Cb()) |
| 49 | call assert_fails('Cb("xxx")', 'E492:') |
Bram Moolenaar | 346418c | 2016-03-15 12:36:08 +0100 | [diff] [blame] | 50 | |
Bram Moolenaar | 1735bc9 | 2016-03-14 23:05:14 +0100 | [diff] [blame] | 51 | let Cb = function('MyDictFunc', ["foo"], dict) |
| 52 | call assert_equal("hello/foo/xxx", Cb("xxx")) |
| 53 | call assert_fails('Cb()', 'E492:') |
Bram Moolenaar | 346418c | 2016-03-15 12:36:08 +0100 | [diff] [blame] | 54 | |
| 55 | let Cb = function('MyDictFunc', [], dict) |
| 56 | call assert_equal("hello/ttt/xxx", Cb("ttt", "xxx")) |
| 57 | call assert_fails('Cb("yyy")', 'E492:') |
| 58 | |
Bram Moolenaar | 1735bc9 | 2016-03-14 23:05:14 +0100 | [diff] [blame] | 59 | let Cb = function('MyDictFunc', dict) |
| 60 | call assert_equal("hello/xxx/yyy", Cb("xxx", "yyy")) |
Bram Moolenaar | 346418c | 2016-03-15 12:36:08 +0100 | [diff] [blame] | 61 | call assert_fails('Cb("fff")', 'E492:') |
Bram Moolenaar | 6563903 | 2016-03-16 21:40:30 +0100 | [diff] [blame] | 62 | |
| 63 | let dict = {"tr": function('tr', ['hello', 'h', 'H'])} |
| 64 | call assert_equal("Hello", dict.tr()) |
Bram Moolenaar | 1735bc9 | 2016-03-14 23:05:14 +0100 | [diff] [blame] | 65 | endfunc |
Bram Moolenaar | ab1fa39 | 2016-03-15 19:33:34 +0100 | [diff] [blame] | 66 | |
| 67 | func Test_partial_implicit() |
| 68 | let dict = {'name': 'foo'} |
| 69 | func dict.MyFunc(arg) dict |
| 70 | return self.name . '/' . a:arg |
| 71 | endfunc |
| 72 | |
| 73 | call assert_equal('foo/bar', dict.MyFunc('bar')) |
| 74 | |
| 75 | call assert_fails('let func = dict.MyFunc', 'E704:') |
| 76 | let Func = dict.MyFunc |
| 77 | call assert_equal('foo/aaa', Func('aaa')) |
| 78 | |
| 79 | let Func = function(dict.MyFunc, ['bbb']) |
| 80 | call assert_equal('foo/bbb', Func()) |
Bram Moolenaar | ab1fa39 | 2016-03-15 19:33:34 +0100 | [diff] [blame] | 81 | endfunc |
Bram Moolenaar | 7a5c46a | 2016-03-16 20:41:21 +0100 | [diff] [blame] | 82 | |
| 83 | fun InnerCall(funcref) |
| 84 | return a:funcref |
| 85 | endfu |
| 86 | |
| 87 | fun OuterCall() |
| 88 | let opt = { 'func' : function('sin') } |
| 89 | call InnerCall(opt.func) |
| 90 | endfu |
| 91 | |
| 92 | func Test_function_in_dict() |
| 93 | call OuterCall() |
| 94 | endfunc |
| 95 | |
Bram Moolenaar | 6f2e4b3 | 2016-03-16 22:52:12 +0100 | [diff] [blame] | 96 | function! s:cache_clear() dict |
| 97 | return self.name |
| 98 | endfunction |
| 99 | |
| 100 | func Test_script_function_in_dict() |
| 101 | let s:obj = {'name': 'foo'} |
| 102 | let s:obj2 = {'name': 'bar'} |
| 103 | |
| 104 | let s:obj['clear'] = function('s:cache_clear') |
| 105 | |
| 106 | call assert_equal('foo', s:obj.clear()) |
| 107 | let F = s:obj.clear |
| 108 | call assert_equal('foo', F()) |
| 109 | call assert_equal('foo', call(s:obj.clear, [], s:obj)) |
| 110 | call assert_equal('bar', call(s:obj.clear, [], s:obj2)) |
| 111 | |
| 112 | let s:obj2['clear'] = function('s:cache_clear') |
| 113 | call assert_equal('bar', s:obj2.clear()) |
| 114 | let B = s:obj2.clear |
| 115 | call assert_equal('bar', B()) |
| 116 | endfunc |
Bram Moolenaar | d22a189 | 2016-03-17 20:50:47 +0100 | [diff] [blame] | 117 | |
Bram Moolenaar | 9e63f61 | 2016-03-17 23:13:28 +0100 | [diff] [blame] | 118 | function! s:cache_arg(arg) dict |
| 119 | let s:result = self.name . '/' . a:arg |
| 120 | return s:result |
| 121 | endfunction |
| 122 | |
| 123 | func Test_script_function_in_dict_arg() |
| 124 | let s:obj = {'name': 'foo'} |
| 125 | let s:obj['clear'] = function('s:cache_arg') |
| 126 | |
| 127 | call assert_equal('foo/bar', s:obj.clear('bar')) |
| 128 | let F = s:obj.clear |
| 129 | let s:result = '' |
| 130 | call assert_equal('foo/bar', F('bar')) |
| 131 | call assert_equal('foo/bar', s:result) |
| 132 | |
| 133 | let s:obj['clear'] = function('s:cache_arg', ['bar']) |
| 134 | call assert_equal('foo/bar', s:obj.clear()) |
| 135 | let s:result = '' |
| 136 | call s:obj.clear() |
| 137 | call assert_equal('foo/bar', s:result) |
| 138 | |
| 139 | let F = s:obj.clear |
| 140 | call assert_equal('foo/bar', F()) |
| 141 | let s:result = '' |
| 142 | call F() |
| 143 | call assert_equal('foo/bar', s:result) |
| 144 | |
| 145 | call assert_equal('foo/bar', call(s:obj.clear, [], s:obj)) |
| 146 | endfunc |
| 147 | |
Bram Moolenaar | d22a189 | 2016-03-17 20:50:47 +0100 | [diff] [blame] | 148 | func Test_partial_exists() |
| 149 | let F = function('MyFunc') |
| 150 | call assert_true(exists('*F')) |
| 151 | let lF = [F] |
| 152 | call assert_true(exists('*lF[0]')) |
| 153 | |
| 154 | let F = function('MyFunc', ['arg']) |
| 155 | call assert_true(exists('*F')) |
| 156 | let lF = [F] |
| 157 | call assert_true(exists('*lF[0]')) |
| 158 | endfunc |
Bram Moolenaar | 5c29154 | 2016-03-19 20:05:45 +0100 | [diff] [blame] | 159 | |
| 160 | func Test_partial_string() |
| 161 | let F = function('MyFunc') |
| 162 | call assert_equal("function('MyFunc')", string(F)) |
| 163 | let F = function('MyFunc', ['foo']) |
| 164 | call assert_equal("function('MyFunc', ['foo'])", string(F)) |
| 165 | let F = function('MyFunc', ['foo', 'bar']) |
| 166 | call assert_equal("function('MyFunc', ['foo', 'bar'])", string(F)) |
| 167 | let d = {'one': 1} |
| 168 | let F = function('MyFunc', d) |
| 169 | call assert_equal("function('MyFunc', {'one': 1})", string(F)) |
| 170 | let F = function('MyFunc', ['foo'], d) |
| 171 | call assert_equal("function('MyFunc', ['foo'], {'one': 1})", string(F)) |
| 172 | endfunc |
Bram Moolenaar | e4eb6ff | 2016-03-22 21:00:09 +0100 | [diff] [blame] | 173 | |
| 174 | func Test_func_unref() |
| 175 | let obj = {} |
| 176 | function! obj.func() abort |
| 177 | endfunction |
| 178 | let funcnumber = matchstr(string(obj.func), '^function(''\zs.\{-}\ze''') |
| 179 | call assert_true(exists('*{' . funcnumber . '}')) |
| 180 | unlet obj |
| 181 | call assert_false(exists('*{' . funcnumber . '}')) |
| 182 | endfunc |
Bram Moolenaar | 24c77a1 | 2016-03-24 21:23:06 +0100 | [diff] [blame] | 183 | |
| 184 | func Test_tostring() |
| 185 | let d = {} |
| 186 | let d.d = d |
| 187 | function d.test3() |
| 188 | echo 42 |
| 189 | endfunction |
| 190 | try |
| 191 | call string(d.test3) |
| 192 | catch |
| 193 | call assert_true(v:false, v:exception) |
| 194 | endtry |
| 195 | endfunc |
Bram Moolenaar | c5fbe8a | 2016-03-24 21:42:09 +0100 | [diff] [blame] | 196 | |
| 197 | func Test_redefine_dict_func() |
| 198 | let d = {} |
| 199 | function d.test4() |
| 200 | endfunction |
| 201 | let d.test4 = d.test4 |
| 202 | try |
| 203 | function! d.test4(name) |
| 204 | endfunction |
| 205 | catch |
| 206 | call assert_true(v:errmsg, v:exception) |
| 207 | endtry |
| 208 | endfunc |
Bram Moolenaar | 4c90861 | 2016-03-24 21:58:12 +0100 | [diff] [blame] | 209 | |
| 210 | func Test_bind_in_python() |
| 211 | if has('python') |
| 212 | let g:d = {} |
| 213 | function g:d.test2() |
| 214 | endfunction |
| 215 | python import vim |
| 216 | try |
| 217 | call assert_equal(pyeval('vim.bindeval("g:d.test2")'), g:d.test2) |
| 218 | catch |
| 219 | call assert_true(v:false, v:exception) |
| 220 | endtry |
| 221 | endif |
| 222 | endfunc |
Bram Moolenaar | ddecc25 | 2016-04-06 22:59:37 +0200 | [diff] [blame] | 223 | |
Bram Moolenaar | 107e1ee | 2016-04-08 17:07:19 +0200 | [diff] [blame] | 224 | " This caused double free on exit if EXITFREE is defined. |
Bram Moolenaar | ddecc25 | 2016-04-06 22:59:37 +0200 | [diff] [blame] | 225 | func Test_cyclic_list_arg() |
| 226 | let l = [] |
| 227 | let Pt = function('string', [l]) |
| 228 | call add(l, Pt) |
| 229 | unlet l |
| 230 | unlet Pt |
| 231 | endfunc |
| 232 | |
Bram Moolenaar | 107e1ee | 2016-04-08 17:07:19 +0200 | [diff] [blame] | 233 | " This caused double free on exit if EXITFREE is defined. |
Bram Moolenaar | ddecc25 | 2016-04-06 22:59:37 +0200 | [diff] [blame] | 234 | func Test_cyclic_dict_arg() |
| 235 | let d = {} |
| 236 | let Pt = function('string', [d]) |
| 237 | let d.Pt = Pt |
| 238 | unlet d |
| 239 | unlet Pt |
| 240 | endfunc |
Bram Moolenaar | 107e1ee | 2016-04-08 17:07:19 +0200 | [diff] [blame] | 241 | |
| 242 | func Ignored(job1, job2, status) |
| 243 | endfunc |
| 244 | |
| 245 | func Test_cycle_partial_job() |
Bram Moolenaar | 9e40437 | 2016-04-08 17:25:19 +0200 | [diff] [blame] | 246 | if has('job') |
| 247 | let job = job_start('echo') |
| 248 | call job_setoptions(job, {'exit_cb': function('Ignored', [job])}) |
| 249 | unlet job |
| 250 | endif |
Bram Moolenaar | 107e1ee | 2016-04-08 17:07:19 +0200 | [diff] [blame] | 251 | endfunc |
| 252 | |
| 253 | func Test_ref_job_partial_dict() |
Bram Moolenaar | 9e40437 | 2016-04-08 17:25:19 +0200 | [diff] [blame] | 254 | if has('job') |
| 255 | let g:ref_job = job_start('echo') |
| 256 | let d = {'a': 'b'} |
| 257 | call job_setoptions(g:ref_job, {'exit_cb': function('string', [], d)}) |
| 258 | endif |
Bram Moolenaar | 107e1ee | 2016-04-08 17:07:19 +0200 | [diff] [blame] | 259 | endfunc |
Bram Moolenaar | 1d42961 | 2016-05-24 15:44:17 +0200 | [diff] [blame] | 260 | |
| 261 | func Test_auto_partial_rebind() |
| 262 | let dict1 = {'name': 'dict1'} |
| 263 | func! dict1.f1() |
| 264 | return self.name |
| 265 | endfunc |
| 266 | let dict1.f2 = function(dict1.f1, dict1) |
| 267 | |
| 268 | call assert_equal('dict1', dict1.f1()) |
| 269 | call assert_equal('dict1', dict1['f1']()) |
| 270 | call assert_equal('dict1', dict1.f2()) |
| 271 | call assert_equal('dict1', dict1['f2']()) |
| 272 | |
| 273 | let dict2 = {'name': 'dict2'} |
| 274 | let dict2.f1 = dict1.f1 |
| 275 | let dict2.f2 = dict1.f2 |
| 276 | |
| 277 | call assert_equal('dict2', dict2.f1()) |
| 278 | call assert_equal('dict2', dict2['f1']()) |
| 279 | call assert_equal('dict1', dict2.f2()) |
| 280 | call assert_equal('dict1', dict2['f2']()) |
| 281 | endfunc |
Bram Moolenaar | 2bbf8ef | 2016-05-24 18:37:12 +0200 | [diff] [blame] | 282 | |
| 283 | func Test_get_partial_items() |
| 284 | let dict = {'name': 'hello'} |
| 285 | let Cb = function('MyDictFunc', ["foo", "bar"], dict) |
| 286 | call assert_equal('MyDictFunc', get(Cb, 'func')) |
| 287 | call assert_equal(["foo", "bar"], get(Cb, 'args')) |
| 288 | call assert_equal(dict, get(Cb, 'dict')) |
| 289 | call assert_fails('call get(Cb, "xxx")', 'E475:') |
| 290 | endfunc |