Skip to content

Commit c5209e1

Browse files
committed
♻️ Refactor completions to allow aliases to have priority.
This also actually *adds* support for aliases properly. We still need to eventually handle the case where we asynchronously complete what is being typed by the user, but for all intents and purposes, we are now in a 0.1.0 state.
1 parent 7561cc1 commit c5209e1

File tree

2 files changed

+57
-33
lines changed

2 files changed

+57
-33
lines changed

after/ftplugin/gitcommit.vim

+4-6
Original file line numberDiff line numberDiff line change
@@ -13,11 +13,9 @@ let s:aliases = gitmoji#aliases()
1313
if g:gitmoji_abbreviations && g:gitmoji_insert_emoji
1414
for [name, gitmoji] in s:builtins->items()
1515
execute 'iabbrev' gitmoji.code gitmoji.emoji
16-
for alias in s:aliases->get(name, [])
17-
" NOTE: For some reason not assigning this to a variable before use
18-
" results in an invalid argument error, with no additional information :/
19-
let code = printf(":%s:", alias)
20-
execute 'iabbrev' code gitmoji.emoji
21-
endfor
16+
endfor
17+
for [name, gitmoji] in s:aliases->items()
18+
let code = printf(":%s:", name)
19+
execute 'iabbrev' code gitmoji.emoji
2220
endfor
2321
endif

autoload/gitmoji.vim

+53-27
Original file line numberDiff line numberDiff line change
@@ -10,12 +10,12 @@ function s:warn(...)
1010
endfunction
1111

1212
function s:compare(lhs, rhs)
13-
if a:lhs.name == a:rhs.name
14-
return 0
15-
elseif a:lhs.name < a:rhs.name
13+
if a:lhs.kind < a:rhs.kind
1614
return -1
15+
elseif a:lhs.kind > a:rhs.kind
16+
return 1
1717
endif
18-
return 1
18+
return 0
1919
endfunction
2020

2121
function s:readjson(filename)
@@ -29,29 +29,35 @@ endfunction
2929
" TODO: Look into applying iabbrev(s) during the CompleteDonePre or
3030
" CompleteDone event.
3131
function s:builtin(idx, name)
32-
let gitmoji = s:gitmoji[a:name]
33-
let word = g:gitmoji_insert_emoji ? gitmoji.emoji : gitmoji.code
34-
let abbr = printf('%s %s', gitmoji.emoji, gitmoji.name)
35-
let menu = gitmoji.description
36-
let kind = 'builtin'
37-
return #{ word: word, abbr: abbr, menu: menu, kind: kind }
32+
return s:completion(s:builtins[a:name], 'builtin')
3833
endfunction
3934

40-
function s:builtins()
35+
function s:alias(idx, name)
36+
let gitmoji = s:aliases[a:name]
37+
return s:completion(s:aliases[a:name], 'alias')
38+
endfunction
39+
40+
function s:completion(gitmoji, type)
41+
let word = g:gitmoji_insert_emoji ? a:gitmoji.emoji : a:gitmoji.code
42+
let abbr = printf('%s %s', a:gitmoji.emoji, a:gitmoji.name)
43+
let menu = a:gitmoji.description
44+
return #{ word: word, abbr: abbr, menu: menu, kind: a:type }
45+
endfunction
46+
47+
function s:getbuiltins()
4148
let data = s:findlocal('gitmojis.json')->s:readjson()
4249
let result = {}
4350
if !has_key(data, 'gitmojis')
44-
s:warn('gitmojis.json', "is missing the 'gitmojis' key")
51+
s:warn('gitmojis.json', "is missing the 'gitmojis' key.")
4552
return {}
4653
endif
47-
let data.gitmojis = data.gitmojis->sort(function('s:compare'))
4854
for item in data.gitmojis
4955
let result[item.name] = item
5056
endfor
5157
return result
5258
endfunction
5359

54-
function s:aliases()
60+
function s:getaliases()
5561
if !exists('g:gitmoji_aliases')
5662
return {}
5763
endif
@@ -64,38 +70,58 @@ function s:aliases()
6470
elseif type == v:t_func
6571
let data = call g:gitmoji_aliases
6672
if type(data) != v:t_dict
67-
s:warn('gitmoji.vim', 'g:gitmoji_aliases function did not return a dictionary')
68-
return {}
73+
s:warn('gitmoji.vim', 'g:gitmoji_aliases function did not return a dictionary.')
74+
let data = {}
6975
endif
7076
endif
71-
return data
77+
let builtins = gitmoji#builtins()
78+
let results = {}
79+
for [name, aliases] in data->items()
80+
let gitmoji = builtins[name]->deepcopy()
81+
for alias in aliases
82+
let results[alias] = gitmoji->extend(#{ name: alias }, 'force')
83+
endfor
84+
endfor
85+
return results
7286
endfunction
7387

7488
function gitmoji#builtins()
75-
if !exists('s:gitmoji')
76-
let s:gitmoji = s:builtins()
89+
if !exists('s:builtins')
90+
let s:builtins = s:getbuiltins()
7791
endif
78-
return s:gitmoji
92+
return s:builtins
7993
endfunction
8094

8195
function gitmoji#aliases()
8296
if !exists('s:aliases')
83-
let s:aliases = s:aliases()
97+
let s:aliases = s:getaliases()
8498
endif
8599
return s:aliases
86100
endfunction
87101

88102
function gitmoji#complete(findstart, base)
103+
" Ensure that the dictionaries have been loaded.
89104
call gitmoji#builtins()
90-
" TODO: Permit configuration setting to allow 'matching' if the line is
91-
" empty
105+
call gitmoji#aliases()
92106
if a:base->empty() && a:findstart == 1
93107
let line = getline('.')[0:col('.') - 1]
94-
return line->match(':[^: \t]*$')
108+
let column = line->match(':[^: \t]*$')
109+
if column < 0 && g:gitmoji_complete_anywhere
110+
return col('.')
111+
endif
112+
return column
95113
endif
96-
let keys = s:gitmoji->keys()->sort()
114+
let builtins = s:builtins->keys()
115+
let aliases = s:aliases->keys()
116+
" In this case, there is 'a match' of sorts, and so we need to weed out the
117+
" names. The issue is, we need to *then* also return both the builtins and
118+
" the aliases defined. Thus our 'simple' code path will no longer work, and
119+
" we must branch for when the a:base is... empty.
97120
if !a:base->empty() && a:findstart == 0
98-
let keys = keys->matchfuzzy(a:base[1:])
121+
let builtins = builtins->matchfuzzy(a:base[1:])
122+
let aliases = aliases->matchfuzzy(a:base[1:])
99123
endif
100-
return keys->map(function('s:builtin'))
124+
let completions = builtins->map(function('s:builtin'))
125+
let completions += aliases->map(function('s:alias'))
126+
return completions->sort(function('s:compare'))
101127
endfunction

0 commit comments

Comments
 (0)