223 lines
7.4 KiB
VimL
223 lines
7.4 KiB
VimL
" ============================================================================
|
|
" File: wakatime.vim
|
|
" Description: Automatic time tracking for Vim.
|
|
" Maintainer: WakaTime <support@wakatime.com>
|
|
" License: BSD, see LICENSE.txt for more details.
|
|
" Website: https://wakatime.com/
|
|
" ============================================================================
|
|
|
|
let s:VERSION = '4.0.12'
|
|
|
|
|
|
" Init {{{
|
|
|
|
" Check Vim version
|
|
if v:version < 700
|
|
echoerr "This plugin requires vim >= 7."
|
|
finish
|
|
endif
|
|
|
|
" Only load plugin once
|
|
if exists("g:loaded_wakatime")
|
|
finish
|
|
endif
|
|
let g:loaded_wakatime = 1
|
|
|
|
" Backup & Override cpoptions
|
|
let s:old_cpo = &cpo
|
|
set cpo&vim
|
|
|
|
" Script Globals
|
|
let s:cli_location = expand("<sfile>:p:h") . '/packages/wakatime/cli.py'
|
|
let s:config_file = expand("$HOME/.wakatime.cfg")
|
|
let s:data_file = expand("$HOME/.wakatime.data")
|
|
let s:config_file_already_setup = 0
|
|
let s:local_cache_expire = 10
|
|
let s:last_heartbeat = [0, 0, '']
|
|
|
|
" For backwards compatibility, rename wakatime.conf to wakatime.cfg
|
|
if !filereadable(s:config_file)
|
|
if filereadable(expand("$HOME/.wakatime"))
|
|
exec "silent !mv" expand("$HOME/.wakatime") expand("$HOME/.wakatime.conf")
|
|
endif
|
|
if filereadable(expand("$HOME/.wakatime.conf"))
|
|
if !filereadable(s:config_file)
|
|
let contents = ['[settings]'] + readfile(expand("$HOME/.wakatime.conf"), '')
|
|
call writefile(contents, s:config_file)
|
|
call delete(expand("$HOME/.wakatime.conf"))
|
|
endif
|
|
endif
|
|
endif
|
|
|
|
" Set default python binary location
|
|
if !exists("g:wakatime_PythonBinary")
|
|
let g:wakatime_PythonBinary = 'python'
|
|
endif
|
|
|
|
" Set default heartbeat frequency in minutes
|
|
if !exists("g:wakatime_HeartbeatFrequency")
|
|
let g:wakatime_HeartbeatFrequency = 2
|
|
endif
|
|
|
|
" }}}
|
|
|
|
|
|
" Function Definitions {{{
|
|
|
|
function! s:StripWhitespace(str)
|
|
return substitute(a:str, '^\s*\(.\{-}\)\s*$', '\1', '')
|
|
endfunction
|
|
|
|
function! s:SetupConfigFile()
|
|
if !s:config_file_already_setup
|
|
|
|
" Create config file if does not exist
|
|
if !filereadable(s:config_file)
|
|
let key = input("[WakaTime] Enter your wakatime.com api key: ")
|
|
if key != ''
|
|
call writefile(['[settings]', 'debug = false', printf("api_key = %s", key), 'hidefilenames = false', 'ignore =', ' COMMIT_EDITMSG$', ' PULLREQ_EDITMSG$', ' MERGE_MSG$', ' TAG_EDITMSG$'], s:config_file)
|
|
echo "[WakaTime] Setup complete! Visit http://wakatime.com to view your logged time."
|
|
endif
|
|
|
|
" Make sure config file has api_key
|
|
else
|
|
let found_api_key = 0
|
|
let lines = readfile(s:config_file)
|
|
for line in lines
|
|
let group = split(line, '=')
|
|
if len(group) == 2 && s:StripWhitespace(group[0]) == 'api_key' && s:StripWhitespace(group[1]) != ''
|
|
let found_api_key = 1
|
|
endif
|
|
endfor
|
|
if !found_api_key
|
|
let key = input("[WakaTime] Enter your wakatime.com api key: ")
|
|
let lines = lines + [join(['api_key', key], '=')]
|
|
call writefile(lines, s:config_file)
|
|
echo "[WakaTime] Setup complete! Visit http://wakatime.com to view your logged time."
|
|
endif
|
|
endif
|
|
|
|
let s:config_file_already_setup = 1
|
|
endif
|
|
endfunction
|
|
|
|
function! s:GetCurrentFile()
|
|
return expand("%:p")
|
|
endfunction
|
|
|
|
function! s:EscapeArg(arg)
|
|
return substitute(shellescape(a:arg), '!', '\\!', '')
|
|
endfunction
|
|
|
|
function! s:JoinArgs(args)
|
|
let safeArgs = []
|
|
for arg in a:args
|
|
let safeArgs = safeArgs + [s:EscapeArg(arg)]
|
|
endfor
|
|
return join(safeArgs, ' ')
|
|
endfunction
|
|
|
|
function! s:SendHeartbeat(file, time, is_write, last)
|
|
let file = a:file
|
|
if file == ''
|
|
let file = a:last[2]
|
|
endif
|
|
if file != ''
|
|
let python_bin = g:wakatime_PythonBinary
|
|
if has('win32') || has('win64')
|
|
if python_bin == 'python'
|
|
let python_bin = 'pythonw'
|
|
endif
|
|
endif
|
|
let cmd = [python_bin, '-W', 'ignore', s:cli_location]
|
|
let cmd = cmd + ['--entity', file]
|
|
let cmd = cmd + ['--plugin', printf('vim/%d vim-wakatime/%s', v:version, s:VERSION)]
|
|
if a:is_write
|
|
let cmd = cmd + ['--write']
|
|
endif
|
|
if !empty(&syntax)
|
|
let cmd = cmd + ['--alternate-language', &syntax]
|
|
else
|
|
if !empty(&filetype)
|
|
let cmd = cmd + ['--alternate-language', &filetype]
|
|
endif
|
|
endif
|
|
"let cmd = cmd + ['--verbose']
|
|
if has('win32') || has('win64')
|
|
exec 'silent !start /min cmd /c "' . s:JoinArgs(cmd) . '"'
|
|
else
|
|
exec 'silent !' . s:JoinArgs(cmd) . ' &'
|
|
endif
|
|
call s:SetLastHeartbeat(a:time, a:time, file)
|
|
endif
|
|
endfunction
|
|
|
|
function! s:GetLastHeartbeat()
|
|
if !s:last_heartbeat[0] || localtime() - s:last_heartbeat[0] > s:local_cache_expire
|
|
if !filereadable(s:data_file)
|
|
return [0, 0, '']
|
|
endif
|
|
let last = readfile(s:data_file, '', 3)
|
|
if len(last) == 3
|
|
let s:last_heartbeat = [s:last_heartbeat[0], last[1], last[2]]
|
|
endif
|
|
endif
|
|
return s:last_heartbeat
|
|
endfunction
|
|
|
|
function! s:SetLastHeartbeatLocally(time, last_update, file)
|
|
let s:last_heartbeat = [a:time, a:last_update, a:file]
|
|
endfunction
|
|
|
|
function! s:SetLastHeartbeat(time, last_update, file)
|
|
call s:SetLastHeartbeatLocally(a:time, a:last_update, a:file)
|
|
call writefile([substitute(printf('%d', a:time), ',', '.', ''), substitute(printf('%d', a:last_update), ',', '.', ''), a:file], s:data_file)
|
|
endfunction
|
|
|
|
function! s:EnoughTimePassed(now, last)
|
|
let prev = a:last[1]
|
|
if a:now - prev > g:wakatime_HeartbeatFrequency * 60
|
|
return 1
|
|
endif
|
|
return 0
|
|
endfunction
|
|
|
|
" }}}
|
|
|
|
|
|
" Event Handlers {{{
|
|
|
|
function! s:handleActivity(is_write)
|
|
call s:SetupConfigFile()
|
|
let file = s:GetCurrentFile()
|
|
let now = localtime()
|
|
let last = s:GetLastHeartbeat()
|
|
if !empty(file) && file !~ "-MiniBufExplorer-" && file !~ "--NO NAME--" && file !~ "^term:"
|
|
if a:is_write || s:EnoughTimePassed(now, last) || file != last[2]
|
|
call s:SendHeartbeat(file, now, a:is_write, last)
|
|
else
|
|
if now - s:last_heartbeat[0] > s:local_cache_expire
|
|
call s:SetLastHeartbeatLocally(now, last[1], last[2])
|
|
endif
|
|
endif
|
|
endif
|
|
endfunction
|
|
|
|
" }}}
|
|
|
|
|
|
" Autocommand Events {{{
|
|
|
|
augroup Wakatime
|
|
autocmd!
|
|
autocmd BufEnter * call s:handleActivity(0)
|
|
autocmd VimEnter * call s:handleActivity(0)
|
|
autocmd BufWritePost * call s:handleActivity(1)
|
|
autocmd CursorMoved,CursorMovedI * call s:handleActivity(0)
|
|
augroup END
|
|
|
|
" }}}
|
|
|
|
|
|
" Restore cpoptions
|
|
let &cpo = s:old_cpo
|