123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532 |
- scriptencoding utf-8
- let s:is_vim = !has('nvim')
- let s:utf = &encoding =~# '^utf'
- let s:error_icon = get(g:, 'coc_notify_error_icon', s:utf ? '' : 'E')
- let s:warning_icon = get(g:, 'coc_notify_warning_icon', s:utf ? '⚠' : 'W')
- let s:info_icon = get(g:, 'coc_notify_info_icon', s:utf ? '' : 'I')
- let s:interval = get(g:, 'coc_notify_interval', s:is_vim ? 50 : 20)
- let s:phl = 'CocNotificationProgress'
- let s:progress_char = '─'
- let s:duration = 300.0
- let s:winids = []
- " Valid notify winids on current tab
- function! coc#notify#win_list() abort
- call filter(s:winids, 'coc#float#valid(v:val)')
- return filter(copy(s:winids), '!empty(getwinvar(v:val,"float"))')
- endfunction
- function! coc#notify#close_all() abort
- for winid in coc#notify#win_list()
- call coc#notify#close(winid)
- endfor
- endfunction
- " Do action for winid or first notify window with actions.
- function! coc#notify#do_action(...) abort
- let winids = a:0 > 0 ? a:000 : coc#notify#win_list()
- for winid in winids
- if coc#float#valid(winid) && getwinvar(winid, 'closing', 0) != 1
- let actions = getwinvar(winid, 'actions', [])
- if !empty(actions)
- let items = map(copy(actions), '(v:key + 1).". ".v:val')
- let msg = join(getbufline(winbufnr(winid), 1, '$'), ' ')
- call coc#ui#quickpick(msg, items, {err, res -> s:on_action(err, res, winid) })
- break
- endif
- endif
- endfor
- endfunction
- " Copy notification contents
- function! coc#notify#copy() abort
- let lines = []
- for winid in coc#notify#win_list()
- let key = getwinvar(winid, 'key', v:null)
- if type(key) == v:t_string
- call extend(lines, json_decode(key)['lines'])
- endif
- endfor
- if empty(lines)
- echohl WarningMsg | echon 'No content to copy' | echohl None
- return
- endif
- call setreg('*', join(lines, "\n"))
- endfunction
- " Show source name in window
- function! coc#notify#show_sources() abort
- if !exists('*getbufline') || !exists('*appendbufline')
- throw "getbufline and appendbufline functions required, please upgrade your vim."
- endif
- let winids = filter(coc#notify#win_list(), 'coc#window#get_var(v:val,"closing") != 1')
- for winid in winids
- let key = getwinvar(winid, 'key', v:null)
- if type(key) == v:t_string
- let bufnr = winbufnr(winid)
- let obj = json_decode(key)
- let sourcename = get(obj, 'source', '')
- let lnum = get(obj, 'kind', '') ==# 'progress' ? 1 : 0
- let content = get(getbufline(bufnr, lnum + 1), 0, '')
- if empty(sourcename) || content ==# sourcename
- continue
- endif
- call appendbufline(bufnr, lnum, sourcename)
- call coc#highlight#add_highlight(bufnr, -1, 'Title', lnum, 0, -1)
- call coc#float#scroll_win(winid, 0, 1)
- endif
- endfor
- redra
- endfunction
- function! coc#notify#close_by_source(source) abort
- let winids = filter(coc#notify#win_list(), 'coc#window#get_var(v:val,"closing") != 1')
- for winid in winids
- let key = getwinvar(winid, 'key', v:null)
- if type(key) == v:t_string
- let obj = json_decode(key)
- if get(obj, 'source', '') ==# a:source
- call coc#notify#close(winid)
- endif
- endif
- endfor
- endfunction
- " Cancel auto hide
- function! coc#notify#keep() abort
- for winid in coc#notify#win_list()
- call s:cancel(winid, 'close_timer')
- endfor
- endfunction
- " borderhighlight - border highlight [string]
- " maxWidth - max content width, default 60 [number]
- " minWidth - minimal width [number]
- " maxHeight - max content height, default 10 [number]
- " highlight - default highlight [string]
- " winblend - winblend [number]
- " timeout - auto close timeout, default 5000 [number]
- " title - title text
- " marginRight - margin right, default 10 [number]
- " focusable - focusable [number]
- " source - source name [string]
- " kind - kind for create icon [string]
- " actions - action names [string[]]
- function! coc#notify#create(lines, config) abort
- let actions = get(a:config, 'actions', [])
- let key = json_encode(extend({'lines': a:lines}, a:config))
- let winid = s:find_win(key)
- let kind = get(a:config, 'kind', '')
- let row = 0
- if winid != -1
- let row = getwinvar(winid, 'top', 0)
- call filter(s:winids, 'v:val != '.winid)
- call coc#float#close(winid)
- let winid = v:null
- endif
- let opts = coc#dict#pick(a:config, ['highlight', 'borderhighlight', 'focusable', 'shadow'])
- let border = has_key(opts, 'borderhighlight') ? [1, 1, 1, 1] : []
- let icon = s:get_icon(kind, get(a:config, 'highlight', 'CocFloating'))
- let margin = get(a:config, 'marginRight', 10)
- let maxWidth = min([&columns - margin - 2, get(a:config, 'maxWidth', 80)])
- if maxWidth <= 0
- throw 'No enough spaces for notification'
- endif
- let lines = map(copy(a:lines), 'tr(v:val, "\t", " ")')
- if has_key(a:config, 'title')
- if !empty(border)
- let opts['title'] = a:config['title']
- else
- let lines = [a:config['title']] + lines
- endif
- endif
- let width = max(map(copy(lines), 'strwidth(v:val)')) + (empty(icon) ? 1 : 3)
- if width > maxWidth
- let lines = coc#string#reflow(lines, maxWidth)
- let width = max(map(copy(lines), 'strwidth(v:val)')) + (empty(icon) ? 1 : 3)
- endif
- let highlights = []
- if !empty(icon)
- let ic = icon['text']
- if empty(lines)
- call add(lines, ic)
- else
- let lines[0] = ic.' '.lines[0]
- endif
- call add(highlights, {'lnum': 0, 'hlGroup': icon['hl'], 'colStart': 0, 'colEnd': strlen(ic)})
- endif
- let actionText = join(actions, ' ')
- call map(lines, 'v:key == 0 ? v:val : repeat(" ", '.(empty(icon) ? 0 : 2).').v:val')
- let minWidth = get(a:config, 'minWidth', kind ==# 'progress' ? 30 : 10)
- let width = max(extend(map(lines + [get(opts, 'title', '').' '], 'strwidth(v:val)'), [minWidth, strwidth(actionText) + 1]))
- let width = min([maxWidth, width])
- let height = min([get(a:config, 'maxHeight', 3), len(lines)])
- if kind ==# 'progress'
- let lines = [repeat(s:progress_char, width)] + lines
- let height = height + 1
- endif
- if !empty(actions)
- let before = max([width - strwidth(actionText), 0])
- let lines = lines + [repeat(' ', before).actionText]
- let height = height + 1
- call s:add_action_highlights(before, height - 1, highlights, actions)
- endif
- if row == 0
- let wintop = coc#notify#get_top()
- let row = wintop - height - (empty(border) ? 0 : 2) - 1
- if !s:is_vim && !empty(border)
- let row = row + 1
- endif
- endif
- let col = &columns - margin - width
- if s:is_vim && !empty(border)
- let col = col - 2
- endif
- let winblend = 60
- " Avoid animate for transparent background.
- if get(a:config, 'winblend', 30) == 0 && empty(synIDattr(synIDtrans(hlID(get(opts, 'highlight', 'CocFloating'))), 'bg', 'gui'))
- let winblend = 0
- endif
- call extend(opts, {
- \ 'relative': 'editor',
- \ 'width': width,
- \ 'height': height,
- \ 'col': col,
- \ 'row': row + 1,
- \ 'lines': lines,
- \ 'rounded': 1,
- \ 'highlights': highlights,
- \ 'winblend': winblend,
- \ 'border': border,
- \ })
- let result = coc#float#create_float_win(0, 0, opts)
- if empty(result)
- throw 'Unable to create notify window'
- endif
- let winid = result[0]
- let bufnr = result[1]
- call setwinvar(winid, 'right', 1)
- call setwinvar(winid, 'kind', 'notification')
- call setwinvar(winid, 'top', row)
- call setwinvar(winid, 'key', key)
- call setwinvar(winid, 'actions', actions)
- call setwinvar(winid, 'source', get(a:config, 'source', ''))
- call setwinvar(winid, 'border', !empty(border))
- call coc#float#nvim_scrollbar(winid)
- call add(s:winids, winid)
- let from = {'row': opts['row'], 'winblend': opts['winblend']}
- let to = {'row': row, 'winblend': get(a:config, 'winblend', 30)}
- call timer_start(s:interval, { -> s:animate(winid, from, to, 0)})
- if kind ==# 'progress'
- call timer_start(s:interval, { -> s:progress(winid, width, 0, -1)})
- endif
- if !s:is_vim
- call coc#compat#buf_add_keymap(bufnr, 'n', '<LeftRelease>', ':call coc#notify#nvim_click('.winid.')<CR>', {
- \ 'silent': v:true,
- \ 'nowait': v:true
- \ })
- endif
- " Enable auto close
- if empty(actions) && kind !=# 'progress'
- let timer = timer_start(get(a:config, 'timeout', 10000), { -> coc#notify#close(winid)})
- call setwinvar(winid, 'close_timer', timer)
- endif
- return [winid, bufnr]
- endfunction
- function! coc#notify#nvim_click(winid) abort
- if getwinvar(a:winid, 'closing', 0)
- return
- endif
- call s:cancel(a:winid, 'close_timer')
- let actions = getwinvar(a:winid, 'actions', [])
- if !empty(actions)
- let character = strpart(getline('.'), col('.') - 1, 1)
- if character =~# '^\k'
- let word = expand('<cword>')
- let idx = index(actions, word)
- if idx != -1
- call coc#rpc#notify('FloatBtnClick', [winbufnr(a:winid), idx])
- call coc#notify#close(a:winid)
- endif
- endif
- endif
- endfunction
- function! coc#notify#on_close(winid) abort
- if index(s:winids, a:winid) >= 0
- call filter(s:winids, 'v:val != '.a:winid)
- call coc#notify#reflow()
- endif
- endfunction
- function! coc#notify#get_top() abort
- let mintop = min(map(coc#notify#win_list(), 'coc#notify#get_win_top(v:val)'))
- if mintop != 0
- return mintop
- endif
- return &lines - &cmdheight - (&laststatus == 0 ? 0 : 1 )
- endfunction
- function! coc#notify#get_win_top(winid) abort
- let row = getwinvar(a:winid, 'top', 0)
- if row == 0
- return row
- endif
- return row - (s:is_vim ? 0 : getwinvar(a:winid, 'border', 0))
- endfunction
- " Close with timer
- function! coc#notify#close(winid) abort
- if !coc#float#valid(a:winid) || coc#window#get_var(a:winid, 'closing', 0) == 1
- return
- endif
- if !coc#window#visible(a:winid)
- call coc#float#close(a:winid)
- return
- endif
- let row = coc#window#get_var(a:winid, 'top')
- if type(row) != v:t_number
- call coc#float#close(a:winid)
- return
- endif
- call coc#window#set_var(a:winid, 'closing', 1)
- call s:cancel(a:winid)
- let winblend = coc#window#get_var(a:winid, 'winblend', 0)
- let curr = s:is_vim ? {'row': row} : {'winblend': winblend}
- let dest = s:is_vim ? {'row': row + 1} : {'winblend': winblend == 0 ? 0 : 60}
- call s:animate(a:winid, curr, dest, 0, 1)
- endfunction
- function! s:add_action_highlights(before, lnum, highlights, actions) abort
- let colStart = a:before
- for text in a:actions
- let w = strwidth(text)
- call add(a:highlights, {
- \ 'lnum': a:lnum,
- \ 'hlGroup': 'CocNotificationButton',
- \ 'colStart': colStart,
- \ 'colEnd': colStart + w
- \ })
- let colStart = colStart + w + 1
- endfor
- endfunction
- function! s:on_action(err, idx, winid) abort
- if !empty(a:err)
- throw a:err
- endif
- if a:idx > 0
- call coc#rpc#notify('FloatBtnClick', [winbufnr(a:winid), a:idx - 1])
- call coc#notify#close(a:winid)
- endif
- endfunction
- function! s:cancel(winid, ...) abort
- let name = get(a:, 1, 'timer')
- let timer = coc#window#get_var(a:winid, name)
- if !empty(timer)
- call timer_stop(timer)
- call coc#window#set_var(a:winid, name, v:null)
- endif
- endfunction
- function! s:progress(winid, total, curr, index) abort
- if !coc#float#valid(a:winid)
- return
- endif
- if coc#window#visible(a:winid)
- let total = a:total
- let idx = float2nr(a:curr/5.0)%total
- if idx != a:index
- " update percent
- let bufnr = winbufnr(a:winid)
- let percent = coc#window#get_var(a:winid, 'percent')
- if !empty(percent)
- let width = strchars(getbufline(bufnr, 1)[0])
- let line = repeat(s:progress_char, width - 4).printf('%4s', percent)
- let total = width - 4
- call setbufline(bufnr, 1, line)
- endif
- let message = coc#window#get_var(a:winid, 'message')
- if !empty(message)
- let linecount = coc#compat#buf_line_count(bufnr)
- let hasAction = !empty(coc#window#get_var(a:winid, 'actions', []))
- if getbufvar(bufnr, 'message', 0) == 0
- call appendbufline(bufnr, linecount - hasAction, message)
- call setbufvar(bufnr, 'message', 1)
- call coc#float#change_height(a:winid, 1)
- let tabnr = coc#window#tabnr(a:winid)
- call coc#notify#reflow(tabnr)
- else
- call setbufline(bufnr, linecount - hasAction, message)
- endif
- endif
- let bytes = strlen(s:progress_char)
- call coc#highlight#clear_highlight(bufnr, -1, 0, 1)
- let colStart = bytes * idx
- if idx + 4 <= total
- let colEnd = bytes * (idx + 4)
- call coc#highlight#add_highlight(bufnr, -1, s:phl, 0, colStart, colEnd)
- else
- let colEnd = bytes * total
- call coc#highlight#add_highlight(bufnr, -1, s:phl, 0, colStart, colEnd)
- call coc#highlight#add_highlight(bufnr, -1, s:phl, 0, 0, bytes * (idx + 4 - total))
- endif
- endif
- call timer_start(s:interval, { -> s:progress(a:winid, total, a:curr + 1, idx)})
- else
- " Not block CursorHold event
- call timer_start(&updatetime + 100, { -> s:progress(a:winid, a:total, a:curr, a:index)})
- endif
- endfunction
- " Optional row & winblend
- function! s:config_win(winid, props) abort
- let change_row = has_key(a:props, 'row')
- if s:is_vim
- if change_row
- call popup_move(a:winid, {'line': a:props['row'] + 1})
- endif
- else
- if change_row
- let [row, column] = nvim_win_get_position(a:winid)
- call nvim_win_set_config(a:winid, {
- \ 'row': a:props['row'],
- \ 'col': column,
- \ 'relative': 'editor',
- \ })
- call s:nvim_move_related(a:winid, a:props['row'])
- endif
- call coc#float#nvim_set_winblend(a:winid, get(a:props, 'winblend', v:null))
- call coc#float#nvim_refresh_scrollbar(a:winid)
- endif
- endfunction
- function! s:nvim_move_related(winid, row) abort
- let winids = coc#window#get_var(a:winid, 'related')
- if empty(winids)
- return
- endif
- for winid in winids
- if nvim_win_is_valid(winid)
- let [row, column] = nvim_win_get_position(winid)
- let delta = coc#window#get_var(winid, 'delta', 0)
- call nvim_win_set_config(winid, {
- \ 'row': a:row + delta,
- \ 'col': column,
- \ 'relative': 'editor',
- \ })
- endif
- endfor
- endfunction
- function! s:animate(winid, from, to, prev, ...) abort
- if !coc#float#valid(a:winid)
- return
- endif
- let curr = a:prev + s:interval
- let percent = coc#math#min(curr / s:duration, 1)
- let props = s:get_props(a:from, a:to, percent)
- call s:config_win(a:winid, props)
- let close = get(a:, 1, 0)
- if percent < 1
- call timer_start(s:interval, { -> s:animate(a:winid, a:from, a:to, curr, close)})
- elseif close
- call filter(s:winids, 'v:val != '.a:winid)
- let tabnr = coc#window#tabnr(a:winid)
- if tabnr != -1
- call coc#float#close(a:winid)
- call coc#notify#reflow(tabnr)
- endif
- endif
- endfunction
- function! coc#notify#reflow(...) abort
- let tabnr = get(a:, 1, tabpagenr())
- let winids = filter(copy(s:winids), 'coc#window#tabnr(v:val) == '.tabnr.' && coc#window#get_var(v:val,"closing") != 1')
- if empty(winids)
- return
- endif
- let animate = tabnr == tabpagenr()
- let wins = map(copy(winids), {_, val -> {
- \ 'winid': val,
- \ 'row': coc#window#get_var(val,'top',0),
- \ 'top': coc#window#get_var(val,'top',0) - (s:is_vim ? 0 : coc#window#get_var(val, 'border', 0)),
- \ 'height': coc#float#get_height(val),
- \ }})
- call sort(wins, {a, b -> b['top'] - a['top']})
- let bottom = &lines - &cmdheight - (&laststatus == 0 ? 0 : 1 )
- let moved = 0
- for item in wins
- let winid = item['winid']
- let delta = bottom - (item['top'] + item['height'] + 1)
- if delta != 0
- call s:cancel(winid)
- let dest = item['row'] + delta
- call coc#window#set_var(winid, 'top', dest)
- if animate
- call s:move_win_timer(winid, {'row': item['row']}, {'row': dest}, 0)
- else
- call s:config_win(winid, {'row': dest})
- endif
- let moved = moved + delta
- endif
- let bottom = item['top'] + delta
- endfor
- endfunction
- function! s:move_win_timer(winid, from, to, curr) abort
- if !coc#float#valid(a:winid)
- return
- endif
- if coc#window#get_var(a:winid, 'closing', 0) == 1
- return
- endif
- let percent = coc#math#min(a:curr / s:duration, 1)
- let next = a:curr + s:interval
- if a:curr > 0
- call s:config_win(a:winid, s:get_props(a:from, a:to, percent))
- endif
- if percent < 1
- let timer = timer_start(s:interval, { -> s:move_win_timer(a:winid, a:from, a:to, next)})
- call coc#window#set_var(a:winid, 'timer', timer)
- endif
- endfunction
- function! s:find_win(key) abort
- for winid in coc#notify#win_list()
- if getwinvar(winid, 'key', '') ==# a:key
- return winid
- endif
- endfor
- return -1
- endfunction
- function! s:get_icon(kind, bg) abort
- if a:kind ==# 'info'
- return {'text': s:info_icon, 'hl': coc#highlight#compose_hlgroup('CocInfoSign', a:bg)}
- endif
- if a:kind ==# 'warning'
- return {'text': s:warning_icon, 'hl': coc#highlight#compose_hlgroup('CocWarningSign', a:bg)}
- endif
- if a:kind ==# 'error'
- return {'text': s:error_icon, 'hl': coc#highlight#compose_hlgroup('CocErrorSign', a:bg)}
- endif
- return v:null
- endfunction
- " percent should be float
- function! s:get_props(from, to, percent) abort
- let obj = {}
- for key in keys(a:from)
- let changed = a:to[key] - a:from[key]
- if !s:is_vim && key ==# 'row'
- " Could be float
- let obj[key] = a:from[key] + changed * a:percent
- else
- let obj[key] = a:from[key] + float2nr(round(changed * a:percent))
- endif
- endfor
- return obj
- endfunction
|