Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 1 | " Vim filetype plugin file |
Bram Moolenaar | 42eeac3 | 2005-06-29 22:40:58 +0000 | [diff] [blame] | 2 | " Language: generic Changelog file |
Bram Moolenaar | 57657d8 | 2006-04-21 22:12:41 +0000 | [diff] [blame] | 3 | " Maintainer: Nikolai Weibull <now@bitwi.se> |
| 4 | " Latest Revision: 2006-04-19 |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 5 | " Variables: |
Bram Moolenaar | 57657d8 | 2006-04-21 22:12:41 +0000 | [diff] [blame] | 6 | " g:changelog_timeformat (deprecated: use g:changelog_dateformat instead) - |
Bram Moolenaar | 42eeac3 | 2005-06-29 22:40:58 +0000 | [diff] [blame] | 7 | " description: the timeformat used in ChangeLog entries. |
| 8 | " default: "%Y-%m-%d". |
Bram Moolenaar | 57657d8 | 2006-04-21 22:12:41 +0000 | [diff] [blame] | 9 | " g:changelog_dateformat - |
| 10 | " description: the format sent to strftime() to generate a date string. |
| 11 | " default: "%Y-%m-%d". |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 12 | " g:changelog_username - |
Bram Moolenaar | 42eeac3 | 2005-06-29 22:40:58 +0000 | [diff] [blame] | 13 | " description: the username to use in ChangeLog entries |
| 14 | " default: try to deduce it from environment variables and system files. |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 15 | " Local Mappings: |
| 16 | " <Leader>o - |
Bram Moolenaar | 42eeac3 | 2005-06-29 22:40:58 +0000 | [diff] [blame] | 17 | " adds a new changelog entry for the current user for the current date. |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 18 | " Global Mappings: |
| 19 | " <Leader>o - |
Bram Moolenaar | 42eeac3 | 2005-06-29 22:40:58 +0000 | [diff] [blame] | 20 | " switches to the ChangeLog buffer opened for the current directory, or |
| 21 | " opens it in a new buffer if it exists in the current directory. Then |
| 22 | " it does the same as the local <Leader>o described above. |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 23 | " Notes: |
| 24 | " run 'runtime ftplugin/changelog.vim' to enable the global mapping for |
| 25 | " changelog files. |
| 26 | " TODO: |
| 27 | " should we perhaps open the ChangeLog file even if it doesn't exist already? |
| 28 | " Problem is that you might end up with ChangeLog files all over the place. |
| 29 | |
| 30 | " If 'filetype' isn't "changelog", we must have been to add ChangeLog opener |
Bram Moolenaar | 57657d8 | 2006-04-21 22:12:41 +0000 | [diff] [blame] | 31 | if &filetype == 'changelog' |
| 32 | if exists('b:did_ftplugin') |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 33 | finish |
| 34 | endif |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 35 | let b:did_ftplugin = 1 |
| 36 | |
Bram Moolenaar | 42eeac3 | 2005-06-29 22:40:58 +0000 | [diff] [blame] | 37 | let s:cpo_save = &cpo |
| 38 | set cpo&vim |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 39 | |
Bram Moolenaar | 57657d8 | 2006-04-21 22:12:41 +0000 | [diff] [blame] | 40 | " Set up the format used for dates. |
| 41 | if !exists('g:changelog_dateformat') |
| 42 | if exists('g:changelog_timeformat') |
| 43 | let g:changelog_dateformat = g:changelog_timeformat |
| 44 | else |
| 45 | let g:changelog_dateformat = "%Y-%m-%d" |
| 46 | endif |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 47 | endif |
| 48 | |
| 49 | " Try to figure out a reasonable username of the form: |
Bram Moolenaar | 57657d8 | 2006-04-21 22:12:41 +0000 | [diff] [blame] | 50 | " Full Name <user@host>. |
| 51 | if !exists('g:changelog_username') |
| 52 | if exists('$EMAIL') && $EMAIL != '' |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 53 | let g:changelog_username = $EMAIL |
Bram Moolenaar | 57657d8 | 2006-04-21 22:12:41 +0000 | [diff] [blame] | 54 | elseif exists('$EMAIL_ADDRESS') && $EMAIL_ADDRESS != '' |
| 55 | " This is some Debian junk if I remember correctly. |
| 56 | let g:changelog_username = $EMAIL_ADDRESS |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 57 | else |
Bram Moolenaar | 57657d8 | 2006-04-21 22:12:41 +0000 | [diff] [blame] | 58 | " Get the users login name. |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 59 | let login = system('whoami') |
| 60 | if v:shell_error |
Bram Moolenaar | 42eeac3 | 2005-06-29 22:40:58 +0000 | [diff] [blame] | 61 | let login = 'unknown' |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 62 | else |
Bram Moolenaar | 42eeac3 | 2005-06-29 22:40:58 +0000 | [diff] [blame] | 63 | let newline = stridx(login, "\n") |
| 64 | if newline != -1 |
| 65 | let login = strpart(login, 0, newline) |
| 66 | endif |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 67 | endif |
| 68 | |
Bram Moolenaar | 57657d8 | 2006-04-21 22:12:41 +0000 | [diff] [blame] | 69 | " Try to get the full name from gecos field in /etc/passwd. |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 70 | if filereadable('/etc/passwd') |
Bram Moolenaar | 57657d8 | 2006-04-21 22:12:41 +0000 | [diff] [blame] | 71 | for line in readfile('/etc/passwd') |
| 72 | if line =~ '^' . login |
| 73 | let name = substitute(line,'^\%([^:]*:\)\{4}\([^:]*\):.*$','\1','') |
| 74 | " Only keep stuff before the first comma. |
| 75 | let comma = stridx(name, ',') |
| 76 | if comma != -1 |
| 77 | let name = strpart(name, 0, comma) |
| 78 | endif |
| 79 | " And substitute & in the real name with the login of our user. |
| 80 | let amp = stridx(name, '&') |
| 81 | if amp != -1 |
| 82 | let name = strpart(name, 0, amp) . toupper(login[0]) . |
| 83 | \ strpart(login, 1) . strpart(name, amp + 1) |
| 84 | endif |
| 85 | endif |
| 86 | endfor |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 87 | endif |
| 88 | |
Bram Moolenaar | 57657d8 | 2006-04-21 22:12:41 +0000 | [diff] [blame] | 89 | " If we haven't found a name, try to gather it from other places. |
| 90 | if !exists('name') |
| 91 | " Maybe the environment has something of interest. |
Bram Moolenaar | 42eeac3 | 2005-06-29 22:40:58 +0000 | [diff] [blame] | 92 | if exists("$NAME") |
| 93 | let name = $NAME |
| 94 | else |
| 95 | " No? well, use the login name and capitalize first |
Bram Moolenaar | 57657d8 | 2006-04-21 22:12:41 +0000 | [diff] [blame] | 96 | " character. |
Bram Moolenaar | 42eeac3 | 2005-06-29 22:40:58 +0000 | [diff] [blame] | 97 | let name = toupper(login[0]) . strpart(login, 1) |
| 98 | endif |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 99 | endif |
| 100 | |
Bram Moolenaar | 57657d8 | 2006-04-21 22:12:41 +0000 | [diff] [blame] | 101 | " Get our hostname. |
| 102 | let hostname = system('hostname') |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 103 | if v:shell_error |
Bram Moolenaar | 57657d8 | 2006-04-21 22:12:41 +0000 | [diff] [blame] | 104 | let hostname = 'localhost' |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 105 | else |
Bram Moolenaar | 42eeac3 | 2005-06-29 22:40:58 +0000 | [diff] [blame] | 106 | let newline = stridx(hostname, "\n") |
| 107 | if newline != -1 |
| 108 | let hostname = strpart(hostname, 0, newline) |
| 109 | endif |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 110 | endif |
| 111 | |
Bram Moolenaar | 57657d8 | 2006-04-21 22:12:41 +0000 | [diff] [blame] | 112 | " And finally set the username. |
| 113 | let g:changelog_username = name . ' <' . login . '@' . hostname . '>' |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 114 | endif |
| 115 | endif |
| 116 | |
Bram Moolenaar | 57657d8 | 2006-04-21 22:12:41 +0000 | [diff] [blame] | 117 | " Format used for new date entries. |
| 118 | if !exists('g:changelog_new_date_format') |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 119 | let g:changelog_new_date_format = "%d %u\n\n\t* %c\n\n" |
| 120 | endif |
| 121 | |
Bram Moolenaar | 57657d8 | 2006-04-21 22:12:41 +0000 | [diff] [blame] | 122 | " Format used for new entries to current date entry. |
| 123 | if !exists('g:changelog_new_entry_format') |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 124 | let g:changelog_new_entry_format = "\t* %c" |
| 125 | endif |
| 126 | |
Bram Moolenaar | 57657d8 | 2006-04-21 22:12:41 +0000 | [diff] [blame] | 127 | " Regular expression used to find a given date entry. |
| 128 | if !exists('g:changelog_date_entry_search') |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 129 | let g:changelog_date_entry_search = '^\s*%d\_s*%u' |
| 130 | endif |
| 131 | |
Bram Moolenaar | 57657d8 | 2006-04-21 22:12:41 +0000 | [diff] [blame] | 132 | " Substitutes specific items in new date-entry formats and search strings. |
| 133 | " Can be done with substitute of course, but unclean, and need \@! then. |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 134 | function! s:substitute_items(str, date, user) |
| 135 | let str = a:str |
Bram Moolenaar | 57657d8 | 2006-04-21 22:12:41 +0000 | [diff] [blame] | 136 | let middles = {'%': '%', 'd': a:date, 'u': a:user, 'c': '{cursor}'} |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 137 | let i = stridx(str, '%') |
| 138 | while i != -1 |
Bram Moolenaar | 57657d8 | 2006-04-21 22:12:41 +0000 | [diff] [blame] | 139 | let inc = 0 |
| 140 | if has_key(middles, str[i + 1]) |
| 141 | let mid = middles[str[i + 1]] |
| 142 | let str = strpart(str, 0, i) . mid . strpart(str, i + 2) |
| 143 | let inc = strlen(mid) |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 144 | endif |
Bram Moolenaar | 57657d8 | 2006-04-21 22:12:41 +0000 | [diff] [blame] | 145 | let i = stridx(str, '%', i + 1 + inc) |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 146 | endwhile |
| 147 | return str |
| 148 | endfunction |
| 149 | |
Bram Moolenaar | 57657d8 | 2006-04-21 22:12:41 +0000 | [diff] [blame] | 150 | " Position the cursor once we've done all the funky substitution. |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 151 | function! s:position_cursor() |
| 152 | if search('{cursor}') > 0 |
Bram Moolenaar | 57657d8 | 2006-04-21 22:12:41 +0000 | [diff] [blame] | 153 | let lnum = line('.') |
| 154 | let line = getline(lnum) |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 155 | let cursor = stridx(line, '{cursor}') |
Bram Moolenaar | 57657d8 | 2006-04-21 22:12:41 +0000 | [diff] [blame] | 156 | call setline(lnum, substitute(line, '{cursor}', '', '')) |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 157 | endif |
| 158 | startinsert! |
| 159 | endfunction |
| 160 | |
Bram Moolenaar | 57657d8 | 2006-04-21 22:12:41 +0000 | [diff] [blame] | 161 | " Internal function to create a new entry in the ChangeLog. |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 162 | function! s:new_changelog_entry() |
Bram Moolenaar | 57657d8 | 2006-04-21 22:12:41 +0000 | [diff] [blame] | 163 | " Deal with 'paste' option. |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 164 | let save_paste = &paste |
| 165 | let &paste = 1 |
Bram Moolenaar | 57657d8 | 2006-04-21 22:12:41 +0000 | [diff] [blame] | 166 | call cursor(1, 1) |
| 167 | " Look for an entry for today by our user. |
| 168 | let date = strftime(g:changelog_dateformat) |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 169 | let search = s:substitute_items(g:changelog_date_entry_search, date, |
Bram Moolenaar | 57657d8 | 2006-04-21 22:12:41 +0000 | [diff] [blame] | 170 | \ g:changelog_username) |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 171 | if search(search) > 0 |
Bram Moolenaar | 57657d8 | 2006-04-21 22:12:41 +0000 | [diff] [blame] | 172 | " Ok, now we look for the end of the date entry, and add an entry. |
| 173 | call cursor(nextnonblank(line('.') + 1), 1) |
| 174 | if search('^\s*$', 'W') > 0 |
| 175 | let p = line('.') - 1 |
| 176 | else |
| 177 | let p = line('.') |
| 178 | endif |
| 179 | let ls = split(s:substitute_items(g:changelog_new_entry_format, '', ''), |
| 180 | \ '\n') |
| 181 | call append(p, ls) |
| 182 | call cursor(p + 1, 1) |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 183 | else |
Bram Moolenaar | 57657d8 | 2006-04-21 22:12:41 +0000 | [diff] [blame] | 184 | " Flag for removing empty lines at end of new ChangeLogs. |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 185 | let remove_empty = line('$') == 1 |
| 186 | |
Bram Moolenaar | 57657d8 | 2006-04-21 22:12:41 +0000 | [diff] [blame] | 187 | " No entry today, so create a date-user header and insert an entry. |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 188 | let todays_entry = s:substitute_items(g:changelog_new_date_format, |
Bram Moolenaar | 57657d8 | 2006-04-21 22:12:41 +0000 | [diff] [blame] | 189 | \ date, g:changelog_username) |
| 190 | " Make sure we have a cursor positioning. |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 191 | if stridx(todays_entry, '{cursor}') == -1 |
Bram Moolenaar | 57657d8 | 2006-04-21 22:12:41 +0000 | [diff] [blame] | 192 | let todays_entry = todays_entry . '{cursor}' |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 193 | endif |
| 194 | |
Bram Moolenaar | 57657d8 | 2006-04-21 22:12:41 +0000 | [diff] [blame] | 195 | " Now do the work. |
| 196 | call append(0, split(todays_entry, '\n')) |
| 197 | |
| 198 | " Remove empty lines at end of file. |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 199 | if remove_empty |
Bram Moolenaar | 57657d8 | 2006-04-21 22:12:41 +0000 | [diff] [blame] | 200 | $-/^\s*$/-1,$delete |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 201 | endif |
| 202 | |
Bram Moolenaar | 57657d8 | 2006-04-21 22:12:41 +0000 | [diff] [blame] | 203 | " Reposition cursor once we're done. |
| 204 | call cursor(1, 1) |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 205 | endif |
| 206 | |
| 207 | call s:position_cursor() |
| 208 | |
| 209 | " And reset 'paste' option |
| 210 | let &paste = save_paste |
| 211 | endfunction |
| 212 | |
| 213 | if exists(":NewChangelogEntry") != 2 |
| 214 | map <buffer> <silent> <Leader>o <Esc>:call <SID>new_changelog_entry()<CR> |
| 215 | command! -nargs=0 NewChangelogEntry call s:new_changelog_entry() |
| 216 | endif |
| 217 | |
Bram Moolenaar | 57657d8 | 2006-04-21 22:12:41 +0000 | [diff] [blame] | 218 | let b:undo_ftplugin = "setl com< fo< et< ai<" |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 219 | |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 220 | setlocal comments= |
| 221 | setlocal formatoptions+=t |
| 222 | setlocal noexpandtab |
Bram Moolenaar | 42eeac3 | 2005-06-29 22:40:58 +0000 | [diff] [blame] | 223 | setlocal autoindent |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 224 | |
Bram Moolenaar | 57657d8 | 2006-04-21 22:12:41 +0000 | [diff] [blame] | 225 | if &textwidth == 0 |
| 226 | setlocal textwidth=78 |
| 227 | let b:undo_ftplugin .= " tw<" |
| 228 | endif |
| 229 | |
Bram Moolenaar | 42eeac3 | 2005-06-29 22:40:58 +0000 | [diff] [blame] | 230 | let &cpo = s:cpo_save |
| 231 | unlet s:cpo_save |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 232 | else |
| 233 | " Add the Changelog opening mapping |
| 234 | nmap <silent> <Leader>o :call <SID>open_changelog()<CR> |
| 235 | |
| 236 | function! s:open_changelog() |
Bram Moolenaar | 57657d8 | 2006-04-21 22:12:41 +0000 | [diff] [blame] | 237 | if !filereadable('ChangeLog') |
| 238 | return |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 239 | endif |
Bram Moolenaar | 57657d8 | 2006-04-21 22:12:41 +0000 | [diff] [blame] | 240 | let buf = bufnr('ChangeLog') |
| 241 | if buf != -1 |
| 242 | if bufwinnr(buf) != -1 |
| 243 | execute buf . 'wincmd w' |
| 244 | else |
| 245 | execute 'bsplit' buf |
| 246 | endif |
| 247 | else |
| 248 | split ChangeLog |
| 249 | endif |
| 250 | |
| 251 | call s:new_changelog_entry() |
Bram Moolenaar | 071d427 | 2004-06-13 20:20:40 +0000 | [diff] [blame] | 252 | endfunction |
| 253 | endif |