mirror of
https://github.com/SpaceVim/SpaceVim.git
synced 2025-01-24 02:50:03 +08:00
8151 lines
290 KiB
VimL
Vendored
8151 lines
290 KiB
VimL
Vendored
scriptencoding utf-8
|
||
|
||
set foldmethod=manual
|
||
|
||
call org#buffer#init()
|
||
call org#windows#init()
|
||
|
||
set noswapfile
|
||
|
||
setlocal ignorecase " searches ignore case
|
||
setlocal smartcase " searches use smart case
|
||
setlocal autoindent
|
||
setlocal fileformat=unix
|
||
setlocal backspace=2
|
||
setlocal nowrap
|
||
setlocal tw=78
|
||
setlocal expandtab
|
||
setlocal nosmarttab
|
||
setlocal softtabstop=0
|
||
setlocal foldcolumn=1
|
||
setlocal tabstop=4
|
||
setlocal shiftwidth=4
|
||
setlocal formatlistpat=^\\s*\\d\\+\\.\\s\\+\\\|^\\s*\\-\\s\\+
|
||
setlocal indentexpr=
|
||
setlocal foldexpr=org#fold#level(v:lnum)
|
||
"setlocal iskeyword+=<
|
||
setlocal nocindent
|
||
setlocal iskeyword=@,39,45,48-57,_,129-255
|
||
|
||
|
||
" LINE BELOW IS MAJOR IF THAT ENCOMPASSES MOST OF org.vim
|
||
" endif is near bottom of document
|
||
" everything in between is executed only the first time an
|
||
" org file is opened
|
||
if !exists('g:org_loaded')
|
||
" Load the checkbox plugin
|
||
execute 'runtime ftplugins/vo_checkbox.vim'
|
||
|
||
" set calfunc depending on which calendar version installed
|
||
if exists(':Calendar')==2
|
||
if exists('*Calendar')>0
|
||
let s:Calfunc=function('Calendar')
|
||
else " we have to assume it's more recent version
|
||
let s:Calfunc=function('calendar#show')
|
||
endif
|
||
if !exists('g:calendar_navi')
|
||
let g:calendar_navi=''
|
||
endif
|
||
endif
|
||
|
||
function! s:SID()
|
||
return matchstr(expand('<sfile>'), '<SNR>\zs\d\+\ze_SID$')
|
||
endfun
|
||
let g:org_sid = s:SID()
|
||
let sid='<SNR>' . g:org_sid . '_'
|
||
function! OrgSID(func)
|
||
execute 'call <SNR>'.s:SID().'_'.a:func
|
||
endfunction
|
||
|
||
if has('win32') || has('win64')
|
||
let s:cmd_line_quote_fix = '^'
|
||
else
|
||
let s:cmd_line_quote_fix = ''
|
||
endif
|
||
let g:org_filename_wildcards = ['*.org']
|
||
let s:sfile = expand("<sfile>:p:h")
|
||
let s:last_refile_point = []
|
||
let g:org_todos_done_dict = {}
|
||
let g:org_todos_notdone_dict = {}
|
||
let g:org_agenda_todos_done_pattern = ''
|
||
let g:org_agenda_todos_notdone_pattern = ''
|
||
let g:org_clock_history=[]
|
||
let g:org_reverse_note_order = 0
|
||
let g:org_html_app=''
|
||
let g:org_pdf_app=''
|
||
let s:org_headMatch = '^\*\+\s'
|
||
let s:org_cal_date = '2000-01-01'
|
||
let g:org_export_babel_evaluate = 1
|
||
let g:org_tag_group_arrange = 0
|
||
let g:org_first_sparse=1
|
||
let g:org_clocks_in_agenda = 0
|
||
let s:remstring = '^\s*:\S'
|
||
let s:block_line = '^\s*\(:\|DEADLINE\|SCHEDULED\|CLOSED\|<\d\d\d\d-\|[\d\d\d\d-\)'
|
||
"let s:remstring = '^\s*\(:\|DEADLINE:\|SCHEDULED:\|CLOSED:\|<\d\d\d\d-\)'
|
||
let g:org_use_calendar = 1
|
||
let g:org_todoitems=[]
|
||
let s:headline = ''
|
||
let g:org_ColumnHead = 'Lines'
|
||
let g:org_gray_agenda = 0
|
||
let g:org_sparse_lines_after = 10
|
||
let g:org_log_todos=0
|
||
let g:org_timegrid=[8,20,2]
|
||
let w:v.org_colview_list = []
|
||
let s:firsttext = ''
|
||
let g:org_supported_link_types = '\(http\|file\|mailto\)'
|
||
let g:org_unsupported_link_types = '\(vm\|wl\|mhe\|rmail\|gnus\|bbdb\|irc\|info\|shell\|elisp\)'
|
||
|
||
|
||
let w:v.org_item_len=100
|
||
let w:sparse_on = 0
|
||
let g:org_folds = 1
|
||
let g:org_show_fold_lines = 1
|
||
let g:org_columns_default_width = 15
|
||
let s:org_columns_master_heading = 0
|
||
let w:v.org_colview_list=[]
|
||
let g:org_show_fold_dots = 0
|
||
let g:org_show_matches_folded=1
|
||
let g:org_indent_from_head = 0
|
||
let g:org_agenda_skip_gap = 2
|
||
let g:org_agenda_days=7
|
||
let g:org_agenda_minforskip = 8
|
||
|
||
let g:org_show_balloon_tips=1
|
||
let g:org_datelist = []
|
||
let g:org_search_spec = ''
|
||
let g:org_deadline_warning_days = 3
|
||
let s:org_weekdays = ['mon','tue','wed','thu','fri','sat','sun']
|
||
let s:org_weekdaystring = '\cmon\|tue\|wed\|thu\|fri\|sat\|sun'
|
||
let s:org_months = ['jan','feb','mar','apr','may','jun','jul','aug','sep','oct','nov','dec']
|
||
let s:org_monthstring = '\c\(jan\|feb\|mar\|apr\|may\|jun\|jul\|aug\|sep\|oct\|nov\|dec\)\S*'
|
||
let s:include_inherited_props=0
|
||
let s:AgendaBufferName = "__Agenda__"
|
||
let s:sparse_lines = {}
|
||
let s:search_list = []
|
||
let s:last_search_list = []
|
||
|
||
"testing stuff
|
||
function! CustomSearchesSetup()
|
||
let g:org_custom_searches = [
|
||
\ { 'name':"Next week's agenda", 'type':'agenda',
|
||
\ 'agenda_date':'+1w','agenda_duration':'w'}
|
||
\ ,{ 'name':"Next week's TODOS", 'type':'agenda',
|
||
\ 'agenda_date':'+1w','agenda_duration':'w','spec':'+UNFINISHED_TODOS'}
|
||
\ , { 'name':'Home tags', 'type':'tags', 'spec':'+HOME'}
|
||
\ , { 'name':'Home tags', 'type':'sparse_tree', 'spec':'+HOME'}
|
||
\ , [ { 'name':"Next week's agenda", 'type':'agenda',
|
||
\ 'agenda_date':'+1w','agenda_duration':'w'}
|
||
\ ,{ 'name':'Home tags', 'type':'tags', 'spec':'+HOME'}
|
||
\ ]
|
||
\ , [ { 'name':'Home tags', 'type':'tags', 'spec':'+HOME'}
|
||
\ ,{ 'name':"Next week's agenda", 'type':'agenda',
|
||
\ 'agenda_date':'+1w','agenda_duration':'w'}
|
||
\ ]
|
||
\ ]
|
||
endfunction
|
||
function! s:RunCustom(search)
|
||
" search arg is either number of predefined custom search,
|
||
" or full spec for search from agenda dashboard
|
||
if type(a:search) == type(1)
|
||
let search_spec = g:org_custom_searches[a:search]
|
||
else
|
||
let search_spec = a:search
|
||
endif
|
||
|
||
if !exists("g:agenda_files") || (g:agenda_files == [])
|
||
if has('dialog_con') || has('dialog_gui')
|
||
unsilent call confirm("No agenda files defined. Will add current file to agenda files.")
|
||
endif
|
||
call s:CurfileAgenda()
|
||
endif
|
||
|
||
if type(search_spec) == type({})
|
||
"single spec
|
||
let s:search_list = [ search_spec ]
|
||
else
|
||
" block agenda specs
|
||
let s:search_list = search_spec
|
||
endif
|
||
|
||
let this_time_list = s:search_list
|
||
if this_time_list[0].type !~ 'sparse_tree'
|
||
let curfile = expand("%:p")
|
||
let curtab = tabpagenr()
|
||
let curwin = winnr()
|
||
:AAgenda
|
||
" delete redo block (always just one) if any
|
||
if get(this_time_list[0],'redo_num') > 0
|
||
"redo_num is block in agenda to redo
|
||
let redo_num = s:AgendaBlockNum(this_time_list[0].redo_num)
|
||
normal gg
|
||
if redo_num > 1
|
||
for i in range(1,redo_num - 1)
|
||
call search('^==============','','')
|
||
endfor
|
||
endif
|
||
let start_line = (line('.') == 1) ? 1 : line('.') + 1
|
||
let test_end = search('^=========','W','')
|
||
let end_line = test_end > 0 ? test_end - 1 : line('$')
|
||
silent exec start_line . ',' . end_line . 'delete'
|
||
call append(start_line - 1,['','']) " append to agenda buf
|
||
let s:agenda_insert_point = start_line
|
||
elseif len(this_time_list) == 1
|
||
silent exec '%d'
|
||
let s:agenda_insert_point = 1
|
||
else
|
||
let s:agenda_insert_point = line('$')
|
||
endif
|
||
endif
|
||
|
||
let i = 0
|
||
for item in this_time_list
|
||
if i > 0
|
||
call append(line('$'),['',repeat('=',70),'','']) " append to agenda buf
|
||
let s:agenda_insert_point = line('$')
|
||
endif
|
||
let mydict = item
|
||
if mydict.type ==? 'agenda'
|
||
let mydate = DateCueResult( get(mydict,'agenda_date',''), s:Today())
|
||
let mydur = get(mydict, 'agenda_duration','w')
|
||
let mydur = (mydur == 'w') ? '7' : (mydur == 'd' ? '1' : mydur)
|
||
|
||
call OrgRunAgenda( mydate,
|
||
\ get(mydict, 'agenda_duration', 'w'),
|
||
\ get(mydict, 'spec','') )
|
||
let save_line = line('.')
|
||
let last_line = search('^============','W','') - 1
|
||
exec (last_line <= 0) ? line('$') : last_line
|
||
if (g:org_agenda_include_clocktable == 1)
|
||
call append(line('.'),OrgGetClocktable(g:agenda_files,' :narrow 40! :tstart ' . '"<' . mydate . ' xxx 00:00>" :tend "<' . calutil#cal(calutil#jul(mydate) + mydur ) . ' xxx 00:00>" :link ')[3:-3])
|
||
endif
|
||
exec save_line
|
||
elseif mydict.type ==? 'sparse_tree'
|
||
call OrgRunSearch( mydict.spec, 1 )
|
||
elseif mydict.type ==? 'sparse_tree_regex'
|
||
silent call s:SparseTreeRun(mydict.spec)
|
||
elseif mydict.type ==? 'tags'
|
||
call OrgRunSearch( mydict.spec )
|
||
elseif mydict.type ==? 'tags-todo'
|
||
" add todos to spec
|
||
call OrgRunSearch( mydict.spec,'agenda_todo' )
|
||
endif
|
||
let i += 1
|
||
endfor
|
||
set foldlevel=999
|
||
execute s:agenda_insert_point
|
||
nohl
|
||
endfunction
|
||
"Section Tag and Todo Funcs
|
||
function! OrgProcessConfigLines()
|
||
let b:v.org_config_lines = []
|
||
let b:v.todoitems = []
|
||
silent g/^#+/call add( b:v.org_config_lines, getline(line('.')) )
|
||
|
||
" clear out for new tag settings
|
||
let b:v.tagdict = {}
|
||
let b:v.buf_tags_static_spec = ''
|
||
let b:v.tagchars=''
|
||
let b:v.tags_order = []
|
||
if g:org_tags_alist ==# ''
|
||
let b:v.dynamic_tags=1
|
||
else
|
||
let b:v.dynamic_tags=0
|
||
endif
|
||
|
||
for line in b:v.org_config_lines
|
||
if line =~ '^#+CATEGORY'
|
||
let b:v.buffer_category = matchstr( line ,'^#+CATEGORY:\s*\zs.*')
|
||
let b:v.org_inherited_defaults['CATEGORY'] = b:v.buffer_category
|
||
elseif line =~ '^#+COLUMNS'
|
||
let b:v.buffer_columns = matchstr( line ,'^#+COLUMNS:\s*\zs.*')
|
||
let b:v.org_inherited_defaults['COLUMNS'] = b:v.buffer_columns
|
||
"let w:v.org_current_columns = b:v.buffer_columns
|
||
elseif line =~ '#+STARTUP:'
|
||
let startup_list = split(matchstr( line, '#+STARTUP:\s*\zs.*') )
|
||
for item in startup_list
|
||
silent! exec "let b:v." . item . "=1"
|
||
endfor
|
||
elseif line =~ '#+TAGS:'
|
||
let newtags = matchstr( line, '#+TAGS:\s*\zs.*')
|
||
if newtags ==# ''
|
||
let b:v.dynamic_tags = 1
|
||
else
|
||
let b:v.buf_tags_static_spec .= newtags . ' \n '
|
||
endif
|
||
elseif line =~ '\(#+TODO:\|#+SEQ_TODO:\)'
|
||
call OrgTodoSetup(matchstr(line,'\(#+TODO:\|#+SEQ_TODO:\)\s*\zs.*'))
|
||
elseif line =~ '#+PROPERTY:'
|
||
let m = matchlist(line,'.\{-}:\s*\(.\{-}\)_ALL \(.*\)')
|
||
if m[2] ># ''
|
||
let b:v.prop_all_dict[toupper(m[1])] = m[2]
|
||
endif
|
||
endif
|
||
endfor
|
||
if empty(b:v.todoitems)
|
||
call OrgTodoSetup(g:org_todo_setup)
|
||
endif
|
||
|
||
call OrgTagSetup( b:v.buf_tags_static_spec )
|
||
|
||
" get rid of b:v.buffer_category (and columns also) and just use o_i_d var???
|
||
if b:v.buffer_category ==# ''
|
||
let b:v.buffer_category = expand("%:t:r")
|
||
endif
|
||
|
||
normal gg
|
||
endfunction
|
||
function! s:EditProp(property)
|
||
let s:edit_prop = toupper(a:property)
|
||
let propval = get(s:GetProperties(line('.'),0), s:edit_prop,'')
|
||
function! s:PropCompleteFunc(arghead, sd, gf)
|
||
let prop = matchstr(a:sd,'^.\{-}\ze:')
|
||
let valuelist = split(get(b:v.prop_all_dict, s:edit_prop, ''))
|
||
let matches = filter(valuelist, 'v:val =~ a:arghead')
|
||
return join(matches, "\n")
|
||
endfunction
|
||
let myfunc = '<SNR>' . s:SID() . '_PropCompleteFunc'
|
||
let newval = input("Enter value for " . s:edit_prop . ": ", propval, 'custom,' . myfunc)
|
||
if newval != propval
|
||
call s:SetProp(s:edit_prop, newval)
|
||
echo s:edit_prop . " changed to: " newval
|
||
else
|
||
echo s:edit_prop . " not changed."
|
||
endif
|
||
endfunction
|
||
function! OrgTodoConvert(orgtodo)
|
||
let todolist = []
|
||
let sublist = []
|
||
let b:v.tododict = {}
|
||
" let templist = []
|
||
let temp_list = split(a:orgtodo,'\s\+')
|
||
|
||
for item in temp_list
|
||
if item == '|'
|
||
continue
|
||
endif
|
||
let b:v.tododict[item] = {}
|
||
if matchstr(item,'.*(.)') ==# ''
|
||
let b:v.tododict[item].todotext = item
|
||
let b:v.tododict[item].todochar = ''
|
||
else
|
||
let b:v.tododict[item].todotext = matchstr(item,'.*\ze(.)')
|
||
let b:v.tododict[item].todochar = matchstr(item,'.*(\zs.\ze)')
|
||
endif
|
||
endfor
|
||
" count '|' chars in list, if 0 or 1 then
|
||
" it is like Org-mode format, otherwise
|
||
" sublists are used in non-done slot"
|
||
let bar_count = count(split(a:orgtodo,'\zs'),'|')
|
||
let after_bar = 0
|
||
if bar_count >= 2
|
||
for item in temp_list
|
||
if item != '|'
|
||
call add(sublist,item)
|
||
elseif (item ==? '|')
|
||
call add(todolist,sublist)
|
||
let sublist = []
|
||
endif
|
||
endfor
|
||
else
|
||
for item in temp_list
|
||
if (item != '|') && (after_bar == 1)
|
||
call add(sublist,item)
|
||
elseif (item != '|') && (after_bar == 0)
|
||
call add(todolist,item)
|
||
elseif (item ==? '|')
|
||
let sublist = []
|
||
let after_bar = 1
|
||
endif
|
||
endfor
|
||
endif
|
||
if sublist != []
|
||
call add(todolist,sublist)
|
||
endif
|
||
return todolist
|
||
endfunction
|
||
|
||
function! OrgTodoSetup(todolist_str)
|
||
let todolist = OrgTodoConvert(a:todolist_str)
|
||
let b:v.todo_setup = a:todolist_str
|
||
"set up list and patterns for use throughout
|
||
let b:v.todoitems=[]
|
||
let b:v.todo_first_letters = ''
|
||
"let b:v.fulltodos=todolist
|
||
let b:v.todocycle=[]
|
||
let b:v.todoMatch=''
|
||
let b:v.todoNotDoneMatch=''
|
||
let b:v.todoDoneMatch=''
|
||
let i = 0
|
||
while i < len(todolist)
|
||
if type(todolist[i]) == type('abc')
|
||
let thistodo = matchstr(todolist[i],'.*ze(.)')
|
||
let thistodo = b:v.tododict[todolist[i]].todotext
|
||
let todolist[i] = substitute(todolist[i],'(.)','','')
|
||
call add(b:v.todoitems,thistodo)
|
||
call add(b:v.todocycle,thistodo)
|
||
" add to patterns
|
||
"let newtodo = b:v.todoitems[len(b:v.todoitems)-1]
|
||
let newtodo = thistodo
|
||
let b:v.todoMatch .= newtodo . '\|'
|
||
if i < len(todolist) - 1
|
||
let b:v.todoNotDoneMatch .= newtodo . '\|'
|
||
let g:org_todos_notdone_dict[newtodo] = 1
|
||
else
|
||
let b:v.todoDoneMatch .= newtodo . '\|'
|
||
let g:org_todos_done_dict[newtodo] = 1
|
||
endif
|
||
else
|
||
"item is itself a list
|
||
let j = 0
|
||
while j < len(todolist[i])
|
||
let thisitem = b:v.tododict[todolist[i][j]].todotext
|
||
let todolist[i][j] = substitute(todolist[i][j],'(.)','','')
|
||
call add(b:v.todoitems,thisitem )
|
||
if j == 0
|
||
call add(b:v.todocycle,thisitem)
|
||
endif
|
||
" add to patterns
|
||
let b:v.todoMatch .= thisitem . '\|'
|
||
if i < len(todolist) - 1
|
||
let b:v.todoNotDoneMatch .= thisitem . '\|'
|
||
let g:org_todos_notdone_dict[thisitem] = 1
|
||
else
|
||
let b:v.todoDoneMatch .= thisitem . '\|'
|
||
let g:org_todos_done_dict[thisitem] = 1
|
||
endif
|
||
let j += 1
|
||
endwhile
|
||
endif
|
||
let i += 1
|
||
endwhile
|
||
let b:v.todoMatch = '^\*\+\s*\zs\('.b:v.todoMatch[:-2] . ')'
|
||
let b:v.todoDoneMatch = '^\*\+\s*\zs\('.b:v.todoDoneMatch[:-2] . ')'
|
||
let b:v.todoNotDoneMatch = '^\*\+\s*\zs\('.b:v.todoNotDoneMatch[:-2] . ')'
|
||
let b:v.fulltodos = todolist
|
||
|
||
syntax clear DONETODO
|
||
exec 'syntax match DONETODO /' . b:v.todoDoneMatch[1:] . '/ containedin=OL1,OL2,OL3,OL4,OL5,OL6'
|
||
syntax clear NOTDONETODO
|
||
exec 'syntax match NOTDONETODO /' . b:v.todoNotDoneMatch[1:] . '/ containedin=OL1,OL2,OL3,OL4,OL5,OL6'
|
||
|
||
call s:OrgCustomTodoHighlights()
|
||
for item in keys( b:v.tododict )
|
||
let item_char = tolower( b:v.tododict[item].todochar)
|
||
if item_char ==# ''
|
||
let item_char = tolower(item[0])
|
||
endif
|
||
endfor
|
||
|
||
endfunction
|
||
function! s:CurfileAgenda()
|
||
exec "let g:agenda_files=['".expand("%:p")."']"
|
||
endfunction
|
||
|
||
function! OrgTagSetup(tagspec)
|
||
let b:v.tags = split(tr(a:tagspec,'{}',' '),'\s\+')
|
||
for item in b:v.tags
|
||
if item =~ '('
|
||
let char = matchstr(item,'(\zs.\ze)')
|
||
let tag = matchstr(item,'.*\ze(')
|
||
else
|
||
"find an unused character
|
||
let char = ''
|
||
let tag = item
|
||
let i = 0
|
||
while i < len(item)
|
||
"if !has_key(chardict, item[i])
|
||
" find char that isn't in tagchars yet
|
||
if b:v.tagchars !~ item[i]
|
||
let char = item[i]
|
||
"let chardict[item[i]] = 1
|
||
break
|
||
endif
|
||
let i += 1
|
||
endwhile
|
||
if char ==# ''
|
||
for i in range(65,90)
|
||
if b:v.tagchars !~ nr2char(i)
|
||
let char = nr2char(i)
|
||
break
|
||
endif
|
||
endfor
|
||
endif
|
||
endif
|
||
let b:v.tagdict[item] = {'char':char, 'tag':tag, 'exclude':'', 'exgroup':0}
|
||
call add(b:v.tags_order,item)
|
||
if char != ''
|
||
let b:v.tagchars .= char
|
||
endif
|
||
endfor
|
||
|
||
let templist = a:tagspec
|
||
let i = 1
|
||
while templist =~ '{.\{}}'
|
||
"cycle through groups and add exclude chars for any group members
|
||
let strikeout = matchstr(templist,'{.\{-}}')
|
||
let exclusive = matchstr(templist,'{\zs.\{-}\ze}')
|
||
let templist = substitute(templist,strikeout,'','')
|
||
let xlist = split(exclusive,'\s\+')
|
||
for item in xlist
|
||
let b:v.tagdict[item].exgroup = i
|
||
for x in xlist
|
||
if x != item
|
||
let b:v.tagdict[item].exclude .= b:v.tagdict[x].char
|
||
endif
|
||
endfor
|
||
endfor
|
||
let i += 1
|
||
endwhile
|
||
endfunction
|
||
|
||
|
||
function! OrgTagsEdit(...)
|
||
let line_file_str = ''
|
||
let lineno=line('.')
|
||
let file = expand("%")
|
||
if bufname("%") ==? ('__Agenda__')
|
||
" new file and lineno below to test with new line marker in agenda
|
||
let file = s:filedict[str2nr(matchstr(getline(line('.')), '^\d\d\d'))]
|
||
let lineno = str2nr(matchstr(getline(line('.')),'^\d\d\d\zs\d*'))
|
||
|
||
call org#SaveLocation()
|
||
call org#LocateFile(file)
|
||
call s:SetDynamicTags()
|
||
call org#RestoreLocation()
|
||
|
||
let b:v.tagdict = getbufvar(file,'v').tagdict
|
||
let b:v.tags_order = getbufvar(file,'v').tags_order
|
||
else
|
||
call s:SetDynamicTags()
|
||
endif
|
||
|
||
let heading_tags = get(s:GetProperties(lineno,0,file),'TAGS','')
|
||
|
||
let new_heading_tags = s:TagMenu(heading_tags)
|
||
if new_heading_tags != heading_tags
|
||
silent call s:SetProp('tags',new_heading_tags,lineno, file)
|
||
endif
|
||
endfunction
|
||
|
||
function! s:TagMenu(heading_tags)
|
||
let heading_tags = a:heading_tags
|
||
|
||
let tagstring = ''
|
||
let tagchars = ''
|
||
for item in b:v.tags_order
|
||
let tagchars .= b:v.tagdict[item].char
|
||
if match(heading_tags,':'.b:v.tagdict[item].tag .':') >= 0
|
||
let tagstring .= b:v.tagdict[item].char
|
||
endif
|
||
endfor
|
||
|
||
hi Cursor guibg=black
|
||
let cue = ''
|
||
set nomore
|
||
while 1
|
||
echo repeat('-',winwidth(0)-1)
|
||
echohl Title | echo 'Choose tags: ' | echohl None | echon '( <enter> to accept, <esc> to cancel )'
|
||
echo '------------'
|
||
let oldgroup = 0
|
||
let items_in_row = 1
|
||
for item in b:v.tags_order
|
||
if item ==? '\n'
|
||
continue
|
||
endif
|
||
let curindex = index(b:v.tags_order,item)
|
||
let newgroup = b:v.tagdict[item].exgroup
|
||
let select = ' '
|
||
if match(tagstring,b:v.tagdict[item].char) >= 0
|
||
let select = 'X'
|
||
echohl Question
|
||
else
|
||
echohl None
|
||
endif
|
||
"if (g:org_tag_group_arrange == 0) || (newgroup != oldgroup) || (newgroup == 0 ) || (b:v.tags_order[curindex+1] ==? '\n')
|
||
if (curindex == 0) || (b:v.tags_order[curindex-1] ==? '\n') || (winwidth(0) - (items_in_row*20) < 20)
|
||
echo repeat(' ',3) . '[' | echohl Question | echon select | echohl None | echon '] '
|
||
echohl None | echon b:v.tagdict[item].tag | echohl Title | echon '('.b:v.tagdict[item].char.')' | echohl None
|
||
let nextindent = repeat(' ',12-len(b:v.tagdict[item].tag))
|
||
let items_in_row = 1
|
||
else
|
||
"echon repeat(' ',3) .
|
||
echon nextindent
|
||
echon '[' | echohl Question | echon select | echohl None | echon '] '
|
||
echohl None | echon b:v.tagdict[item].tag | echohl Title | echon '('.b:v.tagdict[item].char.')' | echohl None
|
||
let nextindent = repeat(' ',12-len(b:v.tagdict[item].tag))
|
||
let items_in_row += 1
|
||
"echon repeat(' ', 12-len(b:v.tagdict[item]))
|
||
endif
|
||
let oldgroup = b:v.tagdict[item].exgroup
|
||
endfor
|
||
echo ""
|
||
"echohl LineNr | echon 'Date+time ['.basedate . ' '.basetime.']: '
|
||
"echohl None | echon cue.'_ =>' | echohl WildMenu | echon ' '.newdate.' '.newtime
|
||
let nchar = getchar()
|
||
let newchar = nr2char(nchar)
|
||
if (nchar ==? "\<BS>") && (len(cue)>0)
|
||
let cue = cue[:-2]
|
||
elseif nchar ==? "\<s-c-up>"
|
||
let cue = ((curdif-365>=0) ?'+':'').(curdif-365).'d'
|
||
elseif newchar ==? "\<s-cr>"
|
||
" add new tag . . . todo . . .
|
||
elseif newchar ==? "\<cr>"
|
||
break
|
||
elseif newchar ==? "\<Esc>"
|
||
hi Cursor guibg=gray
|
||
redraw
|
||
return a:heading_tags
|
||
elseif (match(tagchars,newchar) >= 0)
|
||
if (match(tagstring,newchar) == -1)
|
||
let tagstring .= newchar
|
||
" check for mutually exclusve tags
|
||
for item in keys(b:v.tagdict)
|
||
if b:v.tagdict[item].char ==? newchar
|
||
let exclude_str = b:v.tagdict[item].exclude
|
||
let tagstring = tr(tagstring,exclude_str,repeat(' ',len(exclude_str)))
|
||
break
|
||
endif
|
||
endfor
|
||
else
|
||
let tagstring = tr(tagstring,newchar,' ')
|
||
endif
|
||
endif
|
||
call substitute(tagstring,' ','','')
|
||
echon repeat(' ',72)
|
||
redraw
|
||
endwhile
|
||
|
||
hi Cursor guibg=gray
|
||
redraw
|
||
echo
|
||
set more
|
||
|
||
let heading_tags = ''
|
||
for item in keys(b:v.tagdict)
|
||
if (item!='\n') && (match(tagstring, b:v.tagdict[item].char) >= 0)
|
||
let heading_tags .= b:v.tagdict[item].tag . ':'
|
||
endif
|
||
endfor
|
||
if heading_tags ># '' | let heading_tags = ':' . heading_tags | endif
|
||
return heading_tags
|
||
endfunction
|
||
|
||
function! s:SetDynamicTags()
|
||
let taglist = s:GetBufferTags()
|
||
let chardict = {}
|
||
let b:v.tagdict = {}
|
||
let b:v.tagchars = ''
|
||
let b:v.tags_order = []
|
||
|
||
if b:v.buf_tags_static_spec ==# ''
|
||
let static_tags = g:org_tags_alist . ' ' . g:org_tags_persistent_alist
|
||
if static_tags ==# ''
|
||
let b:v.dynamic_tags_only = 1
|
||
endif
|
||
elseif exists('b:v.noptags')
|
||
let static_tags = b:v.buf_tags_static_spec
|
||
else
|
||
let static_tags = b:v.buf_tags_static_spec . ' ' . g:org_tags_persistent_alist
|
||
endif
|
||
|
||
if exists('b:v.dynamic_tags_only') && (b:v.dynamic_tags_only == 1)
|
||
let setup_string = join(taglist)
|
||
elseif exists('b:v.dynamic_tags') && (b:v.dynamic_tags == 1)
|
||
"first need to remove dups in dynamic taglist
|
||
let temp_list = split(static_tags)
|
||
for i in range(0,len(temp_list)-1)
|
||
if temp_list[i] =~ '(.)'
|
||
let temp_list[i] = matchstr(temp_list[i],'^.*\ze(')
|
||
endif
|
||
endfor
|
||
let dup_list = s:Intersect( temp_list, taglist )
|
||
for item in dup_list
|
||
call remove( taglist, index(taglist, item) )
|
||
endfor
|
||
let setup_string = static_tags . ' ' . join(taglist)
|
||
else
|
||
let setup_string = static_tags
|
||
endif
|
||
|
||
call OrgTagSetup( setup_string )
|
||
|
||
endfunction
|
||
|
||
function! s:GetBufferTags()
|
||
let save_cursor = getpos(".")
|
||
let b:v.buftagdict = {}
|
||
" call addtags for each headline in buffer
|
||
g/^\*/call s:AddTagsToDict(line("."))
|
||
call setpos('.',save_cursor)
|
||
return sort(keys(b:v.buftagdict))
|
||
endfunction
|
||
inoremap <F5> <C-R>=OrgEffort()<CR>
|
||
noremap <F5> A<C-R>=OrgEffort()<CR>
|
||
function! OrgEffort()
|
||
if getline(line('.'))=~':Effort:'
|
||
call setline(line('.'), substitute(getline(line('.')),'ort:\zs.*','',''))
|
||
normal A
|
||
call complete(col('.'),b:v.effort)
|
||
endif
|
||
return ''
|
||
endfunction
|
||
function! s:AddTagsToDict(line)
|
||
let taglist = s:GetTagList(a:line)
|
||
if !empty(taglist)
|
||
for item in taglist
|
||
execute "let b:v.buftagdict['" . item . "'] = 1"
|
||
endfor
|
||
endif
|
||
endfunction
|
||
|
||
function! s:GetTagList(line)
|
||
let text = getline(a:line+1)
|
||
if (text !~ b:v.drawerMatch) && (text !~ b:v.dateMatch) && (text =~ s:remstring)
|
||
let tags = matchlist(text,':\(\S*\):\s*$')
|
||
" include replacement of mistaken double colons with single colon
|
||
let tagstr = substitute(tags[1],'::',':','g')
|
||
if !empty(tags)
|
||
return split(tagstr,':')
|
||
else
|
||
return []
|
||
endif
|
||
else
|
||
return []
|
||
endif
|
||
endfunction
|
||
function! s:IsTagLine(line)
|
||
let text = getline(a:line)
|
||
return (text !~ b:v.drawerMatch) && (text !~ b:v.dateMatch) && (text =~ s:remstring)
|
||
endfunction
|
||
function! s:GetTags(line)
|
||
if s:IsTagLine(a:line+1)
|
||
return matchstr(getline(a:line+1),':.*$')
|
||
else
|
||
return ''
|
||
endif
|
||
endfunction
|
||
function! s:AddTag(tag,line)
|
||
if s:IsTagLine(a:line + 1)
|
||
if matchstr(getline(a:line+1),':'.a:tag.':') ==# ''
|
||
call setline(a:line+1,getline(a:line+1) . ':' .a:tag. ':')
|
||
endif
|
||
else
|
||
call append(a:line, ' :' . a:tag . ':')
|
||
endif
|
||
endfunction
|
||
function! s:TagInput(line)
|
||
let linetags = s:GetTagList(a:line)
|
||
if empty(linetags)
|
||
call append(a:line,':')
|
||
endif
|
||
let buftags = s:GetBufferTags()
|
||
let displaytags = deepcopy(buftags)
|
||
call insert(displaytags,' Exit Menu')
|
||
while 1
|
||
let curstatus = []
|
||
call add(curstatus,0)
|
||
let i = 1
|
||
let linetags = s:GetTagList(a:line)
|
||
while i < len(buftags) + 1
|
||
if index(linetags, buftags[i-1]) >= 0
|
||
let cbox = '[ X ]'
|
||
call add(curstatus,1)
|
||
else
|
||
let cbox = ' '
|
||
call add(curstatus,0)
|
||
endif
|
||
|
||
let displaytags[i] = cbox . s:PrePad('&'.buftags[i-1],28)
|
||
let i += 1
|
||
endwhile
|
||
|
||
let @/=''
|
||
if foldclosed(a:line) > 0
|
||
let b:v.sparse_list = [a:line]
|
||
else
|
||
normal V
|
||
endif
|
||
redraw
|
||
if foldclosed(a:line) > 0
|
||
let b:v.sparse_list = []
|
||
else
|
||
normal V
|
||
endif
|
||
"call insert(displaytags,'Choose tags below:')
|
||
"let key = inputlist(displaytags) - 1
|
||
let taglist = join(displaytags,"\n")
|
||
set guioptions+=v
|
||
let key = confirm('Choose tags:',taglist)-1
|
||
set guioptions-=v
|
||
"call remove(displaytags,0)
|
||
if (key == 0) " || (key == 1)
|
||
" need setline for final redraw
|
||
call setline(a:line+1,getline(a:line+1))
|
||
redraw
|
||
break
|
||
endif
|
||
let curstatus[key] = 1 - curstatus[key]
|
||
let newtags = ''
|
||
let i = 1
|
||
while i < len(curstatus)
|
||
if curstatus[i] == 1
|
||
let newtags .= ':' . buftags[i-1] . ':'
|
||
endif
|
||
let i += 1
|
||
endwhile
|
||
let newtags = substitute(newtags, '::',':','g')
|
||
call setline(a:line+1, repeat(' ',s:Starcount(a:line)+1) . newtags)
|
||
|
||
endwhile
|
||
if empty(s:GetTagList(a:line))
|
||
execute a:line+1 .'d'
|
||
execute a:line
|
||
endif
|
||
endfunction
|
||
|
||
function! s:UnconvertTags(line)
|
||
if s:IsTagLine(a:line+1)
|
||
normal J
|
||
endif
|
||
endfunction
|
||
function! <SID>GlobalUnconvertTags(state)
|
||
if exists('g:org_emacs_autoconvert') && (g:org_emacs_autoconvert != 0)
|
||
let s:save_cursor = getpos(".")
|
||
let s:last_changenr = a:state
|
||
mkview
|
||
normal A
|
||
g/^\*\+\s/call s:UnconvertTags(line("."))
|
||
silent! %s/^\(\s*\):\(DEADLINE\|SCHEDULED\|CLOSED\|CLOCK\|<\d\d\d\d-\d\d-\d\d\)/\1\2/
|
||
endif
|
||
endfunction
|
||
function! <SID>UndoUnconvertTags()
|
||
if exists('g:org_emacs_autoconvert') && (g:org_emacs_autoconvert != 0)
|
||
silent exec 'undo ' . s:last_changenr
|
||
silent undo
|
||
loadview
|
||
call setpos(".",s:save_cursor)
|
||
endif
|
||
endfunction
|
||
|
||
function! s:ConvertTags(line)
|
||
let tags = matchstr(getline(a:line), '\(:\S*:\)\s*$')
|
||
if tags ># ''
|
||
s/\s\+:.*:\s*$//
|
||
call append(a:line, repeat(' ',s:Starcount(a:line)+1) . tags)
|
||
endif
|
||
endfunction
|
||
function! <SID>GlobalConvertTags()
|
||
"if exists('g:org_emacs_autoconvert') && (g:org_emacs_autoconvert != 0)
|
||
let save_cursor = getpos(".")
|
||
g/^\*\+\s/call s:ConvertTags(line("."))
|
||
silent! %s/^\(\s*\)\(DEADLINE:\|SCHEDULED:\|CLOSED:\|CLOCK:\|<\d\d\d\d-\d\d-\d\d\)/\1:\2/
|
||
call setpos(".",save_cursor)
|
||
"endif
|
||
endfunction
|
||
function! s:GlobalFormatTags()
|
||
let save_cursor = getpos(".")
|
||
g/^\*\+\s/call s:FormatTags(line("."))
|
||
call setpos(".",save_cursor)
|
||
endfunction
|
||
function! s:FormatTags(line)
|
||
let tagmatch = matchlist(getline(a:line),'\(:\S*:\)\s*$')
|
||
if !empty(tagmatch)
|
||
let linetags = tagmatch[1]
|
||
s/\s\+:.*:\s*$//
|
||
" add newtags back in, including new tag
|
||
call setline(a:line,getline(a:line) . ' '
|
||
\ . repeat(' ', winwidth(0) - len(getline(a:line)) - len(linetags) - 15)
|
||
\ . linetags)
|
||
endif
|
||
endfunction
|
||
|
||
function! s:FCTest(line)
|
||
if foldclosed(a:line) != a:line
|
||
return a:line . ' --- ' . foldclosed(a:line)
|
||
endif
|
||
endfunction
|
||
|
||
function! OrgSequenceTodo(line,...)
|
||
if a:0 == 1
|
||
if a:1 ==? 'x'
|
||
let newtodo = ''
|
||
else
|
||
for item in b:v.todoitems
|
||
if item[0] ==? a:1
|
||
let newtodo = item
|
||
endif
|
||
endfor
|
||
endif
|
||
endif
|
||
let linetext = getline(a:line)
|
||
if (linetext =~ s:org_headMatch)
|
||
" get first word in line and its index in todoitems
|
||
let tword = matchstr(linetext,'\*\+\s\+\zs\S\+\ze')
|
||
if a:0 == 1
|
||
call s:ReplaceTodo(newtodo)
|
||
else
|
||
call s:ReplaceTodo()
|
||
endif
|
||
endif
|
||
endfunction
|
||
function! s:NextTodo(curtodo)
|
||
let curtodo = a:curtodo
|
||
" check whether word is in todoitems and make appropriate
|
||
" substitution
|
||
let i = index(b:v.todoitems, curtodo)
|
||
if i == -1
|
||
" Not found -> start with first todo
|
||
let newtodo = b:v.todoitems[0]
|
||
elseif i == len(b:v.todoitems) - 1
|
||
" All cycled -> next is empty
|
||
let newtodo = ''
|
||
else
|
||
let newtodo = b:v.todoitems[i+1]
|
||
endif
|
||
return newtodo
|
||
endfunction
|
||
function! s:PreviousTodo(curtodo)
|
||
let curtodo = a:curtodo
|
||
" check whether word is in todoitems and make appropriate
|
||
" substitution
|
||
let i = index(b:v.todoitems, curtodo)
|
||
if i == -1
|
||
" Not found -> Start with last todo
|
||
let last = len(b:v.todoitems) - 1
|
||
let newtodo = b:v.todoitems[last]
|
||
elseif i == 0
|
||
" All cycled -> next is empty
|
||
let newtodo = ''
|
||
else
|
||
let newtodo = b:v.todoitems[i-1]
|
||
endif
|
||
return newtodo
|
||
endfunction
|
||
|
||
function! OrgTodoDashboard(...)
|
||
let key = (a:0==1) ? a:1 : ''
|
||
let save_cursor = getpos('.')
|
||
let save_window = winnr()
|
||
if bufname("%") ==? ('__Agenda__')
|
||
let file = s:filedict[str2nr(matchstr(getline(line('.')), '^\d\d\d'))]
|
||
let lineno = str2nr(matchstr(getline(line('.')),'^\d\d\d\zs\d*'))
|
||
let buffer_lineno = s:ActualBufferLine(lineno,bufnr(file))
|
||
let b:v.todoitems = getbufvar(file,'v').todoitems
|
||
let b:v.todo_setup = getbufvar(file,'v').todo_setup
|
||
let props = s:GetProperties(buffer_lineno, 0, file)
|
||
let Replace_func = function('s:AgendaReplaceTodo')
|
||
else
|
||
exec s:OrgGetHead()
|
||
let props = s:GetProperties(line('.'),0)
|
||
let Replace_func = function('s:ReplaceTodo')
|
||
endif
|
||
|
||
if key ==# ''
|
||
echohl MoreMsg
|
||
echo " ================================="
|
||
echo " Todos defined in this " . (bufname("%") ==? ('__Agenda__') ? "heading's" : "" ) . " document are:"
|
||
echo " " . b:v.todo_setup
|
||
echo " ================================="
|
||
echo " Press key for a todo command:"
|
||
echo " ---------------------------------"
|
||
echo " f (or n) cycle current heading's todo Forward/Next"
|
||
echo " b (or p) cycle current heading's todo Backward/Previous"
|
||
echo " t mark current heading with initial 'unfinished' state"
|
||
echo " d mark current heading with main 'finished' state"
|
||
"if bufname("%") !=? ('__Agenda__')
|
||
let i = 1
|
||
for item in b:v.todoitems
|
||
echo ' ' . i . ' mark current heading as ' . item
|
||
let i += 1
|
||
endfor
|
||
"endif
|
||
echo " "
|
||
echohl Question
|
||
let key = nr2char(getchar())
|
||
redraw
|
||
endif
|
||
"let thisline = getline(line('.'))
|
||
"let curTodo = matchstr(thisline, '\*\+ \zs\S\+')
|
||
|
||
if key =~? 'f\|n'
|
||
call Replace_func()
|
||
elseif key =~? 'b\|p'
|
||
call Replace_func('todo-bkwd')
|
||
elseif key ==? 't'
|
||
"call Replace_func(b:v.todoitems[0])
|
||
call Replace_func('first-todo')
|
||
elseif key ==? 'd'
|
||
"let done_state = (type(b:v.fulltodos[-1])==type([])) ? b:v.fulltodos[-1][0] : b:v.fulltodos[-1]
|
||
"call Replace_func(done_state)
|
||
call Replace_func('first-done')
|
||
elseif key =~ '[1-9]'
|
||
call Replace_func(b:v.todoitems[key-1])
|
||
else
|
||
echo "No todo action selected."
|
||
endif
|
||
echohl None
|
||
exe save_window . 'wincmd w'
|
||
call setpos('.',save_cursor)
|
||
endfunction
|
||
function! s:AgendaReplaceTodo(...)
|
||
" wrapper to call OrgAgendaGetText to do todo operation
|
||
" OrgAgendaGetText does double duty (needs to be
|
||
" refactored) and both retrieves text from main buffer
|
||
" and handles todo replacements
|
||
if bufname('%') != '__Agenda__'
|
||
echo "Not in agenda, can't use AgendaReplaceTodo"
|
||
return
|
||
endif
|
||
if empty(b:v.heading_marks_dict)
|
||
" just mark and do current item
|
||
let b:v.heading_marks_dict[line('.')] = 1
|
||
endif
|
||
|
||
for item in sort(keys(b:v.heading_marks_dict), 's:ReverseSort')
|
||
" go to the line first, then process
|
||
exec item
|
||
let file = s:filedict[str2nr(matchstr(getline(line('.')), '^\d\d\d'))]
|
||
let b:v.fulltodos = getbufvar(file,'v').fulltodos
|
||
let b:v.todoitems = getbufvar(file,'v').todoitems
|
||
let todoword = matchstr(getline(line('.')), '.* \*\+ \zs\S\+')
|
||
if a:0 == 0
|
||
let newtodo = 'todo-fwd'
|
||
else
|
||
let newtodo = a:1
|
||
endif
|
||
if newtodo == 'todo-fwd'
|
||
let newtodo = s:NextTodo(todoword)
|
||
elseif newtodo == 'todo-bkwd'
|
||
let newtodo = s:PreviousTodo(todoword)
|
||
elseif newtodo == 'first-todo'
|
||
let newtodo = b:v.todoitems[0]
|
||
elseif newtodo == 'first-done'
|
||
let newtodo = (type(b:v.fulltodos[-1])==type([])) ? b:v.fulltodos[-1][0] : b:v.fulltodos[-1]
|
||
else
|
||
let newtodo = a:1
|
||
endif
|
||
call OrgAgendaGetText(1,newtodo)
|
||
execute 'sign unplace ' . item . ' buffer=' . bufnr('%')
|
||
endfor
|
||
let b:v.heading_marks_dict = {}
|
||
|
||
endfunction
|
||
function! s:ReplaceTodo(...)
|
||
"a:1 would be newtodo word
|
||
let save_cursor = getpos('.')
|
||
if getline(line('.'))[0] == '*'
|
||
exec s:OrgGetHead()
|
||
endif
|
||
let thisline = getline(line('.'))
|
||
if bufname("%") !=? '__Agenda__'
|
||
let todoword = matchstr(thisline, '\*\+ \zs\S\+')
|
||
else
|
||
let file = s:filedict[str2nr(matchstr(getline(line('.')), '^\d\d\d'))]
|
||
" fulltodos needed for s:NewTodo()
|
||
let b:v.fulltodos = getbufvar(file,'v').fulltodos
|
||
let b:v.todoitems = getbufvar(file,'v').todoitems
|
||
let todoword = matchstr(thisline, '.* \*\+ \zs\S\+')
|
||
endif
|
||
|
||
if a:0 == 0
|
||
let newtodo = 'todo-fwd'
|
||
else
|
||
let newtodo = a:1
|
||
endif
|
||
if newtodo == 'todo-fwd'
|
||
let newtodo = s:NextTodo(todoword)
|
||
elseif newtodo == 'todo-bkwd'
|
||
let newtodo = s:PreviousTodo(todoword)
|
||
elseif newtodo == 'first-todo'
|
||
let newtodo = b:v.todoitems[0]
|
||
elseif newtodo == 'first-done'
|
||
let newtodo = (type(b:v.fulltodos[-1])==type([])) ? b:v.fulltodos[-1][0] : b:v.fulltodos[-1]
|
||
else
|
||
let newtodo = a:1
|
||
endif
|
||
|
||
" if going to main done state check for repeater and change date if necessary
|
||
if (bufnr("%") != bufnr('Agenda')) && (newtodo =~ b:v.todoDoneMatch[11:])
|
||
let newtodo = s:CheckDateRepeaterDone(todoword, newtodo)
|
||
endif
|
||
let s:last_newtodo = newtodo " used to set agenda line in next pass from agenda
|
||
|
||
if newtodo ># ''
|
||
let newtodo .= ' '
|
||
endif
|
||
if (index(b:v.todoitems,todoword) >= 0)
|
||
if newtodo ># ''
|
||
let newline = substitute(getline(line(".")),
|
||
\ '\* ' . todoword.' ',
|
||
\ '\* ' . newtodo,'g')
|
||
else
|
||
let newline = substitute(getline(line(".")),
|
||
\ '\* ' . todoword.' ',
|
||
\ '\* ' . '','g')
|
||
endif
|
||
else
|
||
let newline = substitute(getline(line(".")),
|
||
\ '\zs\* \ze\S\+',
|
||
\ '\* ' . newtodo ,'g')
|
||
endif
|
||
|
||
|
||
call setline(line("."),newline)
|
||
if exists("*Org_after_todo_state_change_hook") && (bufnr("%") != bufnr('Agenda'))
|
||
let Hook = function("Org_after_todo_state_change_hook")
|
||
call Hook(line('.'),todoword,newtodo)
|
||
endif
|
||
|
||
call setpos('.',save_cursor)
|
||
endfunction
|
||
function! s:CheckDateRepeaterDone(state1,state2)
|
||
"check for date repeater on change of todo to done state
|
||
" and handle logging and resetting of date"
|
||
let newtodo = a:state2
|
||
let props = s:GetProperties(line('.'),0)
|
||
let repeat_pattern = '\d\d\d\d-\d\d-\d\d.*[ +.]+\d\+\S\+.*'
|
||
for dateprop in ['DEADLINE','SCHEDULED','TIMESTAMP']
|
||
let thisdate = get(props,dateprop)
|
||
if thisdate =~ repeat_pattern
|
||
"put in log note
|
||
call OrgConfirmDrawer("LOGBOOK")
|
||
let str = ":- State: " . printf('%.10s','"'.a:state2.'"') . " from: " . printf('%.10s','"'.a:state1.'"') .
|
||
\ ' [' . org#Timestamp() . ']'
|
||
call append(line("."), repeat(' ',len(matchstr(getline(line(".")),'^\s*'))) . str)
|
||
exec s:OrgGetHead()
|
||
let newtodo = b:v.todocycle[0]
|
||
"change date as appropriate
|
||
let basedate = matchstr(thisdate,'\d\d\d\d-\d\d-\d\d')
|
||
let cue = '+' . matchstr(thisdate,'+\d*[dwmy]')
|
||
if thisdate =~ ' +\d*[dwmy]'
|
||
let newdate = DateCueResult(cue,basedate)
|
||
elseif thisdate =~ '\.+\d*[dwmy]'
|
||
let newdate = DateCueResult(cue,org#Timestamp()[0:9])
|
||
elseif thisdate =~ '++\d*[dwmy]'
|
||
let newdate = DateCueResult(cue,basedate)
|
||
let i = 0
|
||
while newdate <= org#Timestamp()[0:9]
|
||
if i == 9
|
||
unsilent call confirm('Ten adjustments failed to bring to future date.')
|
||
break
|
||
endif
|
||
let newdate = DateCueResult(cue,newdate)
|
||
let i += 1
|
||
endwhile
|
||
endif
|
||
let mydow = calutil#dayname(newdate)
|
||
call s:SetProp(dateprop,'<' . newdate . ' ' . mydow . thisdate[14:] . '>')
|
||
" break as soon as one repeater is found
|
||
unsilent call confirm('Repeater date: entering log and resetting date.')
|
||
break
|
||
endif
|
||
endfor
|
||
return newtodo
|
||
endfunction
|
||
|
||
"Section Navigation Funcs
|
||
"
|
||
function! s:OrgSubtreeLastLine()
|
||
" Return the line number of the next head at same level, 0 for none
|
||
return s:OrgSubtreeLastLine_l(line("."))
|
||
endfunction
|
||
|
||
function! s:OrgSubtreeLastLine_l(line)
|
||
if a:line == 0
|
||
return line("$")
|
||
endif
|
||
let l:starthead = s:OrgGetHead_l(a:line)
|
||
let l:stars = s:Starcount(l:starthead)
|
||
let l:mypattern = substitute(b:v.headMatchLevel,'level', '1,'.l:stars, "")
|
||
let l:lastline = s:Range_Search(l:mypattern,'nW', line("$"), l:starthead)
|
||
" lastline now has NextHead on abs basis so return end of subtree
|
||
if l:lastline != 0
|
||
let l:lastline -= 1
|
||
else
|
||
let l:lastline = line("$")
|
||
endif
|
||
return l:lastline
|
||
|
||
endfunction
|
||
|
||
function! s:HasAncestorHeadOf(line,ancestor)
|
||
let ultimate = s:OrgUltimateParentHead_l(a:line)
|
||
if (a:line < a:ancestor) || (a:ancestor < ultimate)
|
||
let result = 0
|
||
elseif (a:line == a:ancestor)
|
||
let result = 1
|
||
else
|
||
let test_ancestor = s:OrgParentHead_l(a:line)
|
||
while 1
|
||
if (test_ancestor == a:ancestor)
|
||
let result = 1
|
||
break
|
||
elseif test_ancestor < ultimate
|
||
let result = 0
|
||
break
|
||
endif
|
||
let test_ancestor = s:OrgParentHead_l(test_ancestor)
|
||
endwhile
|
||
endif
|
||
|
||
return result
|
||
endfunction
|
||
function! s:OrgUltimateParentHead()
|
||
" Return the line number of the parent heading, 0 for none
|
||
return s:OrgUltimateParentHead_l(line("."))
|
||
endfunction
|
||
|
||
function! s:OrgUltimateParentHead_l(line)
|
||
" returns 0 for main headings, main heading otherwise
|
||
let l:starthead = s:OrgGetHead_l(a:line)
|
||
|
||
if s:Ind(l:starthead) > 1
|
||
return s:Range_Search('^* ','bnW',1,l:starthead)
|
||
else
|
||
return 0
|
||
endif
|
||
endfunction
|
||
|
||
function! s:OrgParentHead()
|
||
" Return the line number of the parent heading, 0 for none
|
||
return s:OrgParentHead_l(line("."))
|
||
endfunction
|
||
|
||
function! s:OrgParentHead_l(line)
|
||
" todo -- get b:v.levelstars in here
|
||
let l:starthead = s:OrgGetHead_l(a:line)
|
||
let l:parentheadlevel = s:Starcount(l:starthead) - b:v.levelstars
|
||
if l:parentheadlevel <= 0
|
||
return 0
|
||
else
|
||
let l:mypattern = substitute(b:v.headMatchLevel,'level',l:parentheadlevel,'')
|
||
return s:Range_Search(l:mypattern,'bnW',1,l:starthead)
|
||
endif
|
||
endfunction
|
||
|
||
|
||
function! s:Range_Search(stext, flags, ...)
|
||
" searches range, restores cursor to
|
||
" beginning position, and returns
|
||
" first occurrence of pattern
|
||
let save_cursor = getpos(".")
|
||
" a:1 and a:2 are stopline and startline
|
||
if a:0 == 2
|
||
let l:stopline = a:1
|
||
" go to startline
|
||
execute a:2
|
||
normal! $
|
||
elseif a:0 == 1
|
||
let l:stopline = a:1
|
||
else
|
||
let l:stopline = line("$")
|
||
endif
|
||
let l:result = search(a:stext, a:flags, l:stopline)
|
||
call setpos('.',save_cursor)
|
||
return l:result
|
||
endfunction
|
||
|
||
function! s:OrgGetHead()
|
||
return s:OrgGetHead_l(line("."))
|
||
endfunction
|
||
|
||
function! s:OrgGetHead_l(line)
|
||
if s:IsText(a:line)
|
||
return s:Range_Search(b:v.headMatch,'nb', 1, a:line)
|
||
else
|
||
return a:line
|
||
endif
|
||
endfunction
|
||
|
||
function! s:OrgPrevSiblingHead()
|
||
return s:OrgPrevSiblingHead_l(line("."))
|
||
endfunction
|
||
function! s:OrgPrevSiblingHead_l(line)
|
||
if s:Ind(a:line) > 0
|
||
let upperline = s:OrgParentHead_l(a:line)
|
||
else
|
||
let upperline = 0
|
||
endif
|
||
let sibline = s:OrgPrevHeadSameLevel_l(a:line)
|
||
if (sibline <= upperline)
|
||
let sibline = 0
|
||
endif
|
||
return sibline
|
||
endfunction
|
||
|
||
function! s:OrgNextSiblingHead()
|
||
return s:OrgNextSiblingHead_l(line("."))
|
||
endfunction
|
||
function! s:OrgNextSiblingHead_l(line)
|
||
if s:Ind(a:line) > 0
|
||
let lastline = s:OrgSubtreeLastLine_l(s:OrgParentHead_l(a:line))
|
||
else
|
||
let lastline = line("$")
|
||
endif
|
||
let sibline = s:OrgNextHeadSameLevel_l(a:line)
|
||
if (sibline > lastline)
|
||
let sibline = 0
|
||
endif
|
||
return sibline
|
||
endfunction
|
||
|
||
function! s:OrgNextHead()
|
||
" Return the line number of the next heading, 0 for none
|
||
return s:OrgNextHead_l(line("."))
|
||
endfunction
|
||
function! s:OrgNextHead_l(line)
|
||
return s:Range_Search(b:v.headMatch,'n', line("$"),a:line)
|
||
endfunction
|
||
|
||
function! s:OrgPrevHead()
|
||
" Return the line number of the previous heading, 0 for none
|
||
|
||
return s:OrgPrevHead_l(line("."))
|
||
|
||
endfunction
|
||
|
||
function! s:OrgPrevHead_l(line)
|
||
|
||
return s:Range_Search(b:v.headMatch,'nb', 1, a:line-1)
|
||
|
||
endfunction
|
||
|
||
function! s:OrgNextHeadSameLevel()
|
||
" Return the line number of the next head at same level, 0 for none
|
||
return s:OrgNextHeadSameLevel_l(line("."))
|
||
endfunction
|
||
|
||
function! s:OrgNextHeadSameLevel_l(line)
|
||
let level = s:Starcount(a:line)
|
||
let mypattern = substitute(b:v.headMatchLevel,'level', level, "")
|
||
let foundline = s:Range_Search(mypattern,'nW', line("$"), a:line)
|
||
if foundline < line ("$")
|
||
return foundline
|
||
else
|
||
if s:Starcount(foundline) > 0
|
||
return foundline
|
||
else
|
||
return 0
|
||
endif
|
||
endif
|
||
endfunction
|
||
|
||
function! s:OrgPrevHeadSameLevel()
|
||
" Return the line number of the previous heading, 0 for none
|
||
return s:OrgPrevHeadSameLevel_l(line("."))
|
||
endfunction
|
||
function! s:OrgPrevHeadSameLevel_l(line)
|
||
let l:level = s:Starcount(a:line)
|
||
let l:mypattern = substitute(b:v.headMatchLevel,'level', l:level, "")
|
||
let foundline = s:Range_Search(mypattern,'nbW', 1, a:line-1)
|
||
if foundline > 1
|
||
return foundline
|
||
else
|
||
if (s:Starcount(foundline) > 0) && (a:line != 1)
|
||
return 1
|
||
else
|
||
return 0
|
||
endif
|
||
endif
|
||
|
||
endfunction
|
||
|
||
function! s:OrgFirstChildHead()
|
||
" Return the line number of first child, 0 for none
|
||
return s:OrgFirstChildHead_l(line("."))
|
||
endfunction
|
||
function! s:OrgFirstChildHead_l(line)
|
||
let l:starthead = s:OrgGetHead_l(a:line)
|
||
|
||
let l:level = s:Starcount(l:starthead) + 1
|
||
let l:nexthead = s:OrgNextHeadSameLevel_l(l:starthead)
|
||
if l:nexthead == 0
|
||
let l:nexthead = line("$")
|
||
endif
|
||
let l:mypattern = substitute(b:v.headMatchLevel,'level', l:level, "")
|
||
return s:Range_Search(l:mypattern,'nW',l:nexthead, l:starthead)
|
||
endfunction
|
||
|
||
function! s:OrgLastChildHead()
|
||
" Return the line number of the last child, 0 for none
|
||
return s:OrgLastChildHead_l(line("."))
|
||
endfunction
|
||
|
||
function! s:OrgLastChildHead_l(line)
|
||
" returns line number of last immediate child, 0 if none
|
||
let l:starthead = s:OrgGetHead_l(a:line)
|
||
|
||
let l:level = s:Starcount(l:starthead) + 1
|
||
|
||
let l:nexthead = s:OrgNextHeadSameLevel_l(l:starthead)
|
||
if l:nexthead == 0
|
||
let l:nexthead = line("$")
|
||
endif
|
||
|
||
let l:mypattern = substitute(b:v.headMatchLevel,'level', l:level, "")
|
||
return s:Range_Search(l:mypattern,'nbW',l:starthead, l:nexthead)
|
||
|
||
endfunction
|
||
|
||
function! s:MyLastChild(line)
|
||
" Return the line number of the last decendent of parent line
|
||
let l:parentindent = s:Ind(a:line)
|
||
if s:IsText(a:line+1)
|
||
let l:searchline = s:NextLevelLine(a:line+1)
|
||
else
|
||
let l:searchline = a:line+1
|
||
endif
|
||
while s:Ind(l:searchline) > l:parentindent
|
||
let l:searchline = l:searchline+1
|
||
endwhile
|
||
return l:searchline-1
|
||
endfunction
|
||
|
||
function! s:NextVisibleHead(line)
|
||
" Return line of next visible heanding, 0 if none
|
||
let save_cursor = getpos(".")
|
||
|
||
while 1
|
||
let nh = s:OrgNextHead()
|
||
if (nh == 0) || s:IsVisibleHeading(nh)
|
||
break
|
||
endif
|
||
execute nh
|
||
endwhile
|
||
|
||
call setpos('.',save_cursor)
|
||
return nh
|
||
|
||
endfunction
|
||
|
||
function! s:FoldStatus(line)
|
||
" adds new heading or text level depending on type
|
||
let l:fc = foldclosed(a:line)
|
||
if l:fc == -1
|
||
let l:status = 'unfolded'
|
||
elseif l:fc > 0 && l:fc < a:line
|
||
let l:status = 'infold'
|
||
elseif l:fc == a:line
|
||
let l:status = 'foldhead'
|
||
endif
|
||
return l:status
|
||
endfunction
|
||
|
||
function! OrgEnterFunc()
|
||
let syn_items = synstack(line('.'),col('.'))
|
||
call map(syn_items, "synIDattr(v:val,'name')")
|
||
if (index(syn_items,'Org_Full_Link') >= 0) || ( index(syn_items,'Org_Half_Link') >= 0)
|
||
call FollowLink( OrgGetLink() )
|
||
else
|
||
call OrgNewHead('same')
|
||
endif
|
||
endfunction
|
||
|
||
function! OrgNewHead(type,...)
|
||
" adds new heading or text level depending on type
|
||
if a:0 == 1
|
||
normal
|
||
endif
|
||
execute s:OrgGetHead()
|
||
let l:org_line = line(".")
|
||
let l:linebegin = matchlist(getline(line(".")),'^\(\**\s*\)')[1]
|
||
if s:IsText(line(".")) == 0
|
||
|
||
let l:lastline = s:OrgSubtreeLastLine()
|
||
if a:type ==? 'levelup'
|
||
let l:linebegin = substitute(l:linebegin,'^\*\{'.b:v.levelstars.'}','','')
|
||
elseif a:type ==? 'leveldown'
|
||
let l:linebegin = substitute(l:linebegin,'^\*',repeat('*',b:v.levelstars+1),'')
|
||
endif
|
||
call append( l:lastline ,l:linebegin)
|
||
execute l:lastline + 1
|
||
startinsert!
|
||
|
||
endif
|
||
return ''
|
||
endfunction
|
||
|
||
function! s:IsText(line)
|
||
" checks for whether line is any kind of text block
|
||
" test if line matches all-inclusive text block pattern
|
||
return (getline(a:line) !~ b:v.headMatch) && (a:line <= line('$'))
|
||
endfunction
|
||
|
||
function! s:NextLevelAbs(line)
|
||
" Return line of next heading
|
||
" in absolute terms, not just visible headings
|
||
let l:i = 1
|
||
" go down to next non-text line
|
||
while s:IsText(a:line + l:i)
|
||
let l:i = l:i + 1
|
||
"if (a:line + l:i) == line("$")
|
||
:" return 0
|
||
"endif
|
||
endwhile
|
||
return a:line + l:i
|
||
endfunction
|
||
|
||
function! s:NextLevelLine(line)
|
||
" Return line of next heading
|
||
let l:fend = foldclosedend(a:line)
|
||
if l:fend == -1
|
||
let l:i = 1
|
||
" go down to next non-text line
|
||
while s:IsText(a:line + l:i)
|
||
let l:i = l:i + 1
|
||
endwhile
|
||
return a:line + l:i
|
||
else
|
||
return l:fend+1
|
||
endif
|
||
endfunction
|
||
|
||
function! s:HasChild(line)
|
||
" checks for whether heading line has
|
||
" a sublevel
|
||
" checks to see if heading has a non-text sublevel
|
||
let nh = s:OrgNextHead_l(a:line)
|
||
if nh == 0
|
||
return 0
|
||
else
|
||
return (s:Ind(nh) > s:Ind(a:line))
|
||
endif
|
||
|
||
" if s:IsText(a:line + 1) &&
|
||
" \ (s:Ind(s:NextLevelLine(a:line+1)) > s:Ind(a:line))
|
||
" return 1
|
||
" elseif s:IsText(a:line + 1) == 0 &&
|
||
" \ (s:Ind(s:NextLevelLine(a:line)) > s:Ind(a:line))
|
||
" return 1
|
||
" else
|
||
" return 0
|
||
" endif
|
||
endfunction
|
||
|
||
function! s:DoFullCollapse(line)
|
||
let lastline = s:OrgSubtreeLastLine_l(a:line)
|
||
if lastline == a:line
|
||
return
|
||
else
|
||
while foldclosedend(a:line) < lastline
|
||
normal! zc
|
||
endwhile
|
||
endif
|
||
" make sure headline is not just
|
||
" text collapse
|
||
" test if line matches all-inclusive text block pattern
|
||
" while foldclosed(a:line) == -1 && (s:HasChild(a:line) || s:IsText(a:line+1))
|
||
" normal! zc
|
||
" endwhile
|
||
" if s:IsTextOnlyFold(a:line) && s:HasChild(a:line)
|
||
" normal! zc
|
||
" if s:IsTextOnlyFold(a:line) && s:HasChild(a:line)
|
||
" normal! zc
|
||
" if s:IsTextOnlyFold(a:line) && s:HasChild(a:line)
|
||
" normal! zc
|
||
" endif
|
||
" endif
|
||
" endif
|
||
endfunction
|
||
|
||
function! s:IsTextOnlyFold(line)
|
||
" checks for whether heading line has full fold
|
||
" or merely a text fold
|
||
"if s:IsText(a:line + 1) && (foldclosed(a:line + 1) == a:line)
|
||
if s:IsText(a:line + 1) && (foldclosedend(a:line) > 0)
|
||
\ && (s:Ind(foldclosedend(a:line)) <= s:Ind(a:line))
|
||
return 1
|
||
else
|
||
return 0
|
||
endif
|
||
endfunction
|
||
|
||
function! s:MaxVisIndent(headingline)
|
||
" returns max indent for
|
||
" visible lines in a heading's subtree
|
||
" used by ShowSubs
|
||
let l:line = a:headingline
|
||
let l:endline = s:OrgSubtreeLastLine()
|
||
"let l:endline = s:MyLastChild(l:line)
|
||
let l:maxi = s:Ind(l:line)
|
||
let l:textflag = 0
|
||
while l:line <= l:endline
|
||
if (s:Ind(l:line) > l:maxi) &&
|
||
\ ( foldclosed(l:line) == l:line
|
||
\ || foldclosed(l:line) == -1 )
|
||
let l:maxi = s:Ind(l:line)
|
||
if s:IsText(l:line)
|
||
let l:textflag = 1
|
||
endif
|
||
endif
|
||
let l:line = l:line + 1
|
||
endwhile
|
||
return l:maxi + l:textflag
|
||
endfunction
|
||
|
||
function! OrgShowLess(headingline)
|
||
" collapses headings at farthest out visible level
|
||
let l:maxi = s:MaxVisIndent(a:headingline)
|
||
let l:offset = l:maxi - s:Ind(a:headingline)
|
||
echo 'offset: ' . l:offset
|
||
if l:offset > 1
|
||
call s:ShowSubs(l:offset - 1,0)
|
||
elseif l:offset == 1
|
||
normal zc
|
||
"normal! zc
|
||
endif
|
||
endfunction
|
||
|
||
|
||
function! OrgShowMore(headingline)
|
||
" expands headings at furthest out
|
||
" visible level in a heading's subtree
|
||
let l:maxi = s:MaxVisIndent(a:headingline)
|
||
let l:offset = l:maxi - s:Ind(a:headingline)
|
||
if l:offset >= 0
|
||
call s:ShowSubs(l:offset + 1,0)
|
||
if l:maxi == s:MaxVisIndent(a:headingline)
|
||
"call OrgSingleHeadingText('expand')
|
||
endif
|
||
endif
|
||
endfunction
|
||
|
||
function! OrgShowSubs(number,withtext)
|
||
" used by comma-num mapping
|
||
" expands/collapses individual heading to level visibility equal to a:number
|
||
if getline(line('.'))[0] != '*'
|
||
exec s:OrgPrevHead()
|
||
endif
|
||
let cur_level = s:Ind(line('.')) - 1
|
||
if a:number > cur_level
|
||
let rel_level = a:number - cur_level
|
||
if rel_level >= 1
|
||
call s:ShowSubs(rel_level ,0)
|
||
endif
|
||
else
|
||
call s:DoFullCollapse(line('.'))
|
||
endif
|
||
normal ztkj
|
||
endfunction
|
||
|
||
function! s:ShowSubs(number,withtext)
|
||
" shows specif number of levels down from current
|
||
" heading, includes text
|
||
" or merely a text fold
|
||
let save_cursor = getpos(".")
|
||
|
||
call s:DoFullCollapse(line("."))
|
||
let l:start = foldclosed(line("."))
|
||
if l:start != -1
|
||
let l:end = foldclosedend(line("."))
|
||
exec "" . l:start . "," . l:end . "foldc!"
|
||
exec "normal! zv"
|
||
let to_level = 2
|
||
for to_level in range( 2 , a:number )
|
||
exec "" . l:start . "," . l:end . "foldo"
|
||
endfor
|
||
endif
|
||
if a:withtext == 0
|
||
call OrgSingleHeadingText('collapse')
|
||
endif
|
||
|
||
call setpos(".",save_cursor)
|
||
endfunction
|
||
|
||
" 2 args of start line num and direction ('up' or 'down')
|
||
"command -nargs=* OrgMoveLevel :call OrgMoveLevel(<f-args>,v:count1)
|
||
nmap <buffer> <localleader>,q :<C-U>call OrgMoveLevel(line('.'),'up',v:count1)<cr>
|
||
|
||
function! OrgMoveLevel(line, direction,...)
|
||
if (getline(a:line) !~ b:v.headMatch)
|
||
echo "Must be on a heading line to move subtrees."
|
||
return
|
||
endif
|
||
if a:0>=1
|
||
let mycount = a:1
|
||
else
|
||
let mycount = 1
|
||
endif
|
||
" move a heading tree up, down, left, or right
|
||
let lastline = s:OrgSubtreeLastLine_l(a:line)
|
||
if a:direction ==? 'up'
|
||
let l:headabove = a:line
|
||
let count_message = ''
|
||
for i in range( 1, mycount)
|
||
let lasthead = l:headabove
|
||
let l:headabove = s:OrgPrevSiblingHead_l(l:headabove)
|
||
if l:headabove > 0
|
||
let count_message = 'Moved up ' . i . ' levels.'
|
||
elseif i == 1
|
||
" break with no message here
|
||
break
|
||
else
|
||
let l:headabove = lasthead
|
||
if i <= mycount | let count_message .= ' No more siblings above.' | endif
|
||
break
|
||
endif
|
||
endfor
|
||
if l:headabove > 0
|
||
let l:lines = getline(line("."), lastline)
|
||
call s:DoFullCollapse(a:line)
|
||
silent normal! dd
|
||
call append(l:headabove-1,l:lines)
|
||
execute l:headabove
|
||
call s:ShowSubs(1,0)
|
||
echo count_message
|
||
else
|
||
echo "No sibling heading above in this subtree."
|
||
endif
|
||
elseif a:direction ==? 'down'
|
||
let l:headbelow = a:line
|
||
let count_message = ''
|
||
for i in range(1, mycount)
|
||
let lasthead = l:headbelow
|
||
let l:headbelow = s:OrgNextSiblingHead_l(l:headbelow)
|
||
if l:headbelow > 0
|
||
let count_message = 'Moved down ' . i . ' levels.'
|
||
elseif i == 1
|
||
" break with no message here
|
||
break
|
||
else
|
||
let l:headbelow = lasthead
|
||
if i <= mycount | let count_message .= ' No more siblings below.' | endif
|
||
break
|
||
endif
|
||
endfor
|
||
if l:headbelow > 0
|
||
let endofnext = s:OrgSubtreeLastLine_l(l:headbelow)
|
||
let lines = getline(line("."),lastline)
|
||
silent call append(endofnext,lines)
|
||
execute endofnext + 1
|
||
" set mark and go back to delete original subtree
|
||
normal ma
|
||
execute a:line
|
||
call s:DoFullCollapse(a:line)
|
||
silent normal! dd
|
||
normal g'a
|
||
call s:ShowSubs(1,0)
|
||
echo count_message
|
||
else
|
||
echo "No sibling below in this subtree."
|
||
endif
|
||
elseif a:direction ==? 'left'
|
||
if s:Ind(a:line) > 2
|
||
" first move to be last sibling
|
||
let movetoline = s:OrgSubtreeLastLine_l(s:OrgParentHead_l(a:line))
|
||
let lines = getline(line("."),lastline)
|
||
call append(movetoline,lines)
|
||
execute movetoline + 1
|
||
" set mark and go back to delete original subtree
|
||
normal ma
|
||
execute a:line
|
||
call s:DoFullCollapse(a:line)
|
||
silent exe 'normal! dd'
|
||
normal g'a
|
||
" now move tree to the left
|
||
normal ma
|
||
silent execute line(".") ',' . s:OrgSubtreeLastLine() . 's/^' . repeat('\*',b:v.levelstars) .'//'
|
||
call s:DoFullCollapse(a:line)
|
||
normal g'a
|
||
call s:ShowSubs(1,0)
|
||
execute line(".")
|
||
else
|
||
echo "You're already at main heading level."
|
||
endif
|
||
elseif a:direction ==? 'right'
|
||
if s:Ind(s:OrgPrevHead_l(a:line)) >= s:Ind(a:line)
|
||
execute a:line . ',' . lastline . 's/^\*/'.repeat('\*',b:v.levelstars+1).'/'
|
||
call s:DoFullCollapse(a:line)
|
||
execute a:line
|
||
call s:ShowSubs(1,0)
|
||
else
|
||
echo "Already at lowest level of this subtree."
|
||
endif
|
||
endif
|
||
endfunction
|
||
|
||
function! OrgNavigateLevels(direction)
|
||
" Move among headings
|
||
" direction: "up", "down", "right", "left","end", or 'home'
|
||
if s:IsText(line("."))
|
||
exec s:OrgGetHead()
|
||
return
|
||
endif
|
||
|
||
if s:Ind(line(".")) > 0
|
||
let lowerlimit = s:OrgParentHead()
|
||
let upperlimit = s:OrgSubtreeLastLine_l(lowerlimit)
|
||
else
|
||
let lowerlimit = 0
|
||
let upperlimit = line("$")
|
||
endif
|
||
|
||
if a:direction ==? "left"
|
||
let dest = s:OrgParentHead()
|
||
let msg = "At highest level."
|
||
elseif a:direction ==? "home"
|
||
let dest = s:OrgParentHead()
|
||
let msg = "At highest level."
|
||
elseif a:direction ==? "right"
|
||
let dest = s:OrgFirstChildHead()
|
||
let msg = (dest > 0 ? "Has subheadings, but none visible."
|
||
\ : "No more subheadings.")
|
||
elseif a:direction ==? 'end'
|
||
let dest = s:OrgLastChildHead()
|
||
let msg = (dest > 0 ? "Has subheadings, but none visible."
|
||
\ : "No more subheadings.")
|
||
elseif a:direction ==? 'up'
|
||
let dest = s:OrgPrevHeadSameLevel()
|
||
let msg = "Can't go up more here."
|
||
elseif a:direction ==? 'down'
|
||
let dest = s:OrgNextHeadSameLevel()
|
||
let msg = "Can't go down more."
|
||
endif
|
||
|
||
let visible = s:IsVisibleHeading(dest)
|
||
if (dest > 0) && visible && (dest >= lowerlimit) && (dest <= upperlimit)
|
||
execute dest
|
||
else
|
||
echo msg
|
||
endif
|
||
endfunction
|
||
|
||
function! OrgHeadingFirstText(headline)
|
||
exec a:headline + 1
|
||
let found = 0
|
||
while 1
|
||
let thisline = getline(line('.'))
|
||
if thisline =~ b:v.headMatch
|
||
break
|
||
else
|
||
if (thisline !~ s:remstring) && (thisline !~ b:v.dateMatch)
|
||
\ && (thisline !~ b:v.drawerMatch)
|
||
let found = line('.')
|
||
break
|
||
elseif line('.') == line('$')
|
||
break
|
||
endif
|
||
endif
|
||
exec line('.') + 1
|
||
endwhile
|
||
return found
|
||
endfunction
|
||
|
||
function! OrgUnfoldBodyText(headline)
|
||
if OrgHeadingFirstText(a:headline) > 0
|
||
normal zv
|
||
endif
|
||
endfunction
|
||
|
||
function! OrgExpandWithoutText(tolevel)
|
||
" expand all headings but leave Body Text collapsed
|
||
" tolevel: number, 0 to 9, of level to expand to
|
||
" expand levels to 'tolevel' with all body text collapsed
|
||
let l:startline = 1
|
||
let l:endline = line("$")
|
||
let l:execstr = "set foldlevel=" . string(a:tolevel )
|
||
"let l:execstr = "set foldlevel=" . (a:tolevel - 1)
|
||
exec l:execstr
|
||
call OrgBodyTextOperation(l:startline,l:endline,"collapse")
|
||
endfunction
|
||
function! s:OrgExpandSubtree(headline,...)
|
||
if a:0 > 0
|
||
let withtext = a:1
|
||
endif
|
||
let save_cursor = getpos(".")
|
||
call s:DoFullFold(a:headline)
|
||
"let end = foldclosedend(a:headline)
|
||
"normal! zO
|
||
"call OrgBodyTextOperation(a:headline, end, 'collapse')
|
||
call s:ShowSubs(3,withtext)
|
||
call setpos(".",save_cursor)
|
||
endfunction
|
||
function! s:OrgExpandHead(headline)
|
||
let save_cursor = getpos(".")
|
||
call s:DoFullFold(a:headline)
|
||
"let end = foldclosedend(a:headline)
|
||
"normal! zO
|
||
"call OrgBodyTextOperation(a:headline, end, 'collapse')
|
||
call s:ShowSubs(1,0)
|
||
while foldclosed(a:headline) != -1
|
||
normal! zo
|
||
endwhile
|
||
call setpos(".",save_cursor)
|
||
endfunction
|
||
function! s:DoFullFold(headline)
|
||
let save_cursor = getpos(".")
|
||
"normal! zo
|
||
call s:DoAllTextFold(a:headline)
|
||
let fend = foldclosedend(a:headline)
|
||
if ((fend > a:headline) && (s:Ind(fend+1) > s:Ind(a:headline)))
|
||
\ || (s:Ind(a:headline+1) > s:Ind(a:headline))
|
||
normal zc
|
||
endif
|
||
call setpos(".",save_cursor)
|
||
endfunction
|
||
function! s:OrgCycle(headline)
|
||
let save_cursor = getpos(".")
|
||
let topline = line("w0")
|
||
|
||
let end = foldclosedend(a:headline)
|
||
if (end>0) && (s:Ind(end+1) <= s:Ind(a:headline))
|
||
call s:OrgExpandHead(a:headline)
|
||
let endline = end
|
||
elseif ((end == -1) && (s:Ind(s:OrgNextHead_l(a:headline)) > s:Ind(a:headline))
|
||
\ && (foldclosed(s:OrgNextHead_l(a:headline)) > 0))
|
||
let nextsamelevel = s:OrgNextHeadSameLevel_l(a:headline)
|
||
"let nextuplevel = s:OrgNextHeadSameLevel_l(s:OrgParentHead_l(a:headline))
|
||
let nextuplevel = s:OrgNextHeadSameLevel_l(a:headline)
|
||
if (nextsamelevel > 0) && (nextsamelevel > nextuplevel)
|
||
let endline = nextsamelevel
|
||
elseif nextuplevel > a:headline
|
||
let endline = nextuplevel - 1
|
||
else
|
||
let endline = line('$')
|
||
endif
|
||
if b:v.cycle_with_text
|
||
call OrgBodyTextOperation(a:headline+1,endline,'expand')
|
||
else
|
||
call s:OrgExpandSubtree(a:headline,0)
|
||
endif
|
||
else
|
||
call s:DoFullFold(a:headline)
|
||
endif
|
||
|
||
exe "normal! " . topline . "G"
|
||
normal zt
|
||
call setpos(".",save_cursor)
|
||
if exists('endline') && line('w$') < endline
|
||
normal ztkj
|
||
endif
|
||
endfunction
|
||
function! OrgCycle()
|
||
if getline(line(".")) =~ b:v.headMatch
|
||
call s:OrgCycle(line("."))
|
||
elseif getline(line(".")) =~ b:v.drawerMatch
|
||
normal! za
|
||
elseif getline(line('.')) =~ '^\s*|.*|\s*$'
|
||
" we're in a table, do tab and short circuit
|
||
exec "normal i\tl"
|
||
return
|
||
endif
|
||
endfunction
|
||
let s:orgskipthirdcycle = 0
|
||
function! OrgGlobalCycle()
|
||
if getline(line('.')) =~ '^\s*|.*|\s*$'
|
||
"short circuit if we're in table
|
||
exec "normal i\<s-tab>l"
|
||
return
|
||
endif
|
||
if exists('w:sparse_on') && w:sparse_on
|
||
call s:ClearSparseTree()
|
||
endif
|
||
if (&foldlevel > 1) && (&foldlevel != b:v.global_cycle_levels_to_show)
|
||
call OrgExpandWithoutText(1)
|
||
elseif &foldlevel == 1
|
||
call OrgExpandWithoutText(b:v.global_cycle_levels_to_show)
|
||
"elseif (&foldlevel > 1) && ( s:orgskipthirdcycle == 0 )
|
||
" let s = getpos('.')
|
||
" g/^\*\+ /call OrgUnfoldBodyText(line('.'))
|
||
" call setpos('.',s)
|
||
" let s:orgskipthirdcycle = 1
|
||
else
|
||
let save_cursor = getpos('.')
|
||
set foldlevel=9999
|
||
silent exec 'g/' . b:v.drawerMatch . '/normal! zc'
|
||
let s:orgskipthirdcycle = 0
|
||
call setpos('.',save_cursor)
|
||
endif
|
||
endfunction
|
||
function! s:LastTextLine(headingline)
|
||
" returns last text line of text under
|
||
" a heading, or 0 if no text
|
||
let l:retval = 0
|
||
if s:IsText(a:line + 1)
|
||
let l:i = a:line + 1
|
||
while s:IsText(l:i)
|
||
let l:i = l:i + 1
|
||
endwhile
|
||
let l:retval = l:i - 1
|
||
endif
|
||
return l:retval
|
||
endfunction
|
||
|
||
function! s:ShowSynStack()
|
||
for id in synstack(line("."),col("."))
|
||
echo synIDattr(id,"name")
|
||
endfor
|
||
endfunction
|
||
function! s:SignList()
|
||
let signlist = ''
|
||
redir => signlist
|
||
silent execute "sign list"
|
||
redir END
|
||
return split(signlist,'\n')
|
||
endfunction
|
||
function! s:DeleteSigns()
|
||
" first delete all placed signs
|
||
sign unplace *
|
||
let signs = s:SignList()
|
||
for item in signs
|
||
silent execute "sign undefine " . matchstr(item,'\S\+ \zs\S\+\ze ')
|
||
endfor
|
||
sign define piet text=>>
|
||
sign define fbegin text=>
|
||
sign define fend text=<
|
||
sign define marked text=> linehl=Org_Marked
|
||
endfunction
|
||
|
||
function! s:GetPlacedSignsString(buffer)
|
||
let placedstr = ''
|
||
redir => placedstr
|
||
silent execute "sign place buffer=".a:buffer
|
||
redir END
|
||
return placedstr
|
||
|
||
endfunction
|
||
function! s:GetProperties(hl,withtextinfo,...)
|
||
let save_cursor = getpos(".")
|
||
if a:0 >=1
|
||
let curtab = tabpagenr()
|
||
let curwin = winnr()
|
||
" optional args are: a:1 - lineno, a:2 - file
|
||
call org#LocateFile(a:1)
|
||
endif
|
||
let datesdone = 0
|
||
let result1 = {}
|
||
let result = {}
|
||
|
||
let linetext = getline(a:hl)
|
||
if linetext[0] == '*'
|
||
let hl = a:hl
|
||
else
|
||
let hl = s:OrgGetHead_l(a:hl)
|
||
let linetext = getline(hl)
|
||
endif
|
||
|
||
let result1['LINE'] = hl
|
||
let result1['LEVEL'] = s:Ind(hl) - 1
|
||
"let linetext = getline(hl)
|
||
let result1['ITEM'] = linetext
|
||
let result1['FILE'] = expand("%:t")
|
||
" get date on headline, if any
|
||
if linetext =~ b:v.dateMatch
|
||
let result1['ld'] = matchlist(linetext,b:v.dateMatch)[1]
|
||
endif
|
||
if (getline(hl+1) =~ b:v.tagMatch) && (getline(hl+1) !~ b:v.drawerMatch)
|
||
let result1['TAGS'] = matchstr(getline(hl+1),b:v.tagMatch)
|
||
endif
|
||
if linetext =~ b:v.todoMatch
|
||
let result1['TODO'] = matchstr(linetext,b:v.todoMatch)
|
||
else
|
||
let result1['TODO'] = ''
|
||
endif
|
||
|
||
let line = hl + 1
|
||
"let firsttext=0
|
||
while 1
|
||
let ltext = getline(line)
|
||
if ltext =~ b:v.propMatch
|
||
let result = s:GetPropVals(line+1)
|
||
elseif (ltext =~ '^\s*:\s*CLOCK')
|
||
" do nothing
|
||
elseif (ltext !~ s:block_line) || (ltext =~ b:v.headMatch)
|
||
call extend(result, result1)
|
||
if datesdone
|
||
call extend(result, dateresult)
|
||
endif
|
||
let result['BLOCK_END'] = line - 1
|
||
break
|
||
elseif (ltext =~ b:v.dateMatch) && !datesdone
|
||
let dateresult = s:GetDateVals(line)
|
||
let datesdone = 1
|
||
" no break, go back around to check for props
|
||
"elseif (ltext =~ '^\s*$') || (ltext =~ '^\s*:\s*CLOCK')
|
||
endif
|
||
let line += 1
|
||
endwhile
|
||
" *****************************************
|
||
" get inherited properties
|
||
if s:include_inherited_props == 1
|
||
for item in b:v.org_inherited_properties
|
||
if index(keys(result), item) == -1
|
||
let result[item] = s:IProp(hl , item)
|
||
endif
|
||
endfor
|
||
endif
|
||
" *****************************************
|
||
" get last line
|
||
if a:withtextinfo
|
||
"let result['tbegin'] = line
|
||
let result['TEND'] = s:OrgNextHead_l(hl) - 1
|
||
endif
|
||
if a:0 >= 1
|
||
execute "tabnext ".curtab
|
||
execute curwin . "wincmd w"
|
||
endif
|
||
call setpos(".",save_cursor)
|
||
"debugging
|
||
let g:org_result = result
|
||
return result
|
||
endfunction
|
||
|
||
function! s:GetDateVals(line)
|
||
"result is dict with all date vals
|
||
let myline = a:line
|
||
let result = {}
|
||
while 1
|
||
let ltext = getline(myline)
|
||
let mtest1 = '<\zs'.b:v.dateMatch.'.*\ze>'
|
||
let mtest2 = '\[\zs'.b:v.dateMatch.'.*\ze\]'
|
||
if ltext =~ mtest1
|
||
"let mymatch = matchlist(ltext, '.\{-}\(<\d\d\d\d-\d\d-\d\d\) \S\S\S\( \d\d:\d\d\)*')
|
||
"let mydate = mymatch[1] . mymatch[2] . '>'
|
||
let mymatch = '^\s*\(:DEADLINE:\|:SCHEDULED:\|:CLOSED:\|:<\)\s*\zs.*'
|
||
let mydate = matchstr(ltext,mymatch)
|
||
let mydate = (mydate[0]=='<') ? mydate[1:-2] : mydate[:-2]
|
||
if ltext =~ 'DEADLINE'
|
||
let dtype = 'DEADLINE'
|
||
elseif ltext =~ 'SCHEDULED'
|
||
let dtype = 'SCHEDULED'
|
||
elseif ltext =~ 'CLOSED'
|
||
let dtype = 'CLOSED'
|
||
else
|
||
let dtype = 'TIMESTAMP'
|
||
endif
|
||
elseif ltext =~ mtest2
|
||
let mydate = matchstr(ltext, mtest2)
|
||
"let mydate = substitute(ltext, '\(\[\d\d\d\d-\d\d-\d\d\) \S\S\S\( \d\d:\d\d\)*','\1\2','')
|
||
let dtype = 'TIMESTAMP_IA'
|
||
else
|
||
break
|
||
endif
|
||
|
||
try
|
||
"only add if first of dtype encountered
|
||
if get(result,dtype) == 0
|
||
let result[dtype] = mydate
|
||
endif
|
||
catch /^Vim\%((\a\+)\)\=:E/
|
||
endtry
|
||
let myline += 1
|
||
endwhile
|
||
return result
|
||
endfunction
|
||
|
||
function! s:GetPropVals(line)
|
||
"result is dict with all prop vals
|
||
let myline = a:line
|
||
let result = {}
|
||
while 1
|
||
let ltext = getline(myline)
|
||
if ltext =~ b:v.propvalMatch
|
||
let mtch = matchlist(ltext, b:v.propvalMatch)
|
||
" mtch[1] is now property, mtch[2] is its value
|
||
try
|
||
let result[toupper(mtch[1])] = mtch[2]
|
||
catch /^Vim\%((\a\+)\)\=:E/
|
||
endtry
|
||
else
|
||
break
|
||
endif
|
||
let myline += 1
|
||
endwhile
|
||
return result
|
||
endfunction
|
||
|
||
|
||
function! s:RedoTextIndent()
|
||
set fdm=manual
|
||
set foldlevel=9999
|
||
exec 1
|
||
let myindent = 0
|
||
while line(".") < line("$")
|
||
let line = getline(line("."))
|
||
if matchstr(line,'^\*\+') ># ''
|
||
let myindent = len(matchstr(line,'^\*\+')) + g:org_indent_from_head
|
||
normal j
|
||
else
|
||
let text = matchstr(line,'^ *\zs.*')
|
||
let spaces = len(matchstr(line,'^ *'))
|
||
if (spaces != (myindent + 1)) && (text != '')
|
||
call setline(line("."),repeat(' ',myindent+1) . text)
|
||
endif
|
||
normal j
|
||
endif
|
||
endwhile
|
||
exec 1
|
||
set fdm=expr
|
||
endfunction
|
||
|
||
function! s:LoremIpsum()
|
||
let lines = ['Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum.", "Sed ut perspiciatis unde omnis iste natus error sit voluptatem accusantium doloremque laudantium, totam rem aperiam, eaque ipsa quae ab illo inventore veritatis et quasi architecto beatae vitae dicta sunt explicabo. Nemo enim ipsam voluptatem quia voluptas sit aspernatur aut odit aut fugit, sed quia consequuntur magni dolores eos qui ratione voluptatem sequi nesciunt. Neque porro quisquam est, qui dolorem ipsum quia dolor sit amet, consectetur, adipisci velit, sed quia non numquam eius modi tempora incidunt ut labore et dolore magnam aliquam quaerat voluptatem. Ut enim ad minima veniam, quis nostrum exercitationem ullam corporis suscipit laboriosam, nisi ut aliquid ex ea commodi consequatur? Quis autem vel eum iure reprehenderit qui in ea voluptate velit esse quam nihil molestiae consequatur, vel illum qui dolorem eum fugiat quo voluptas nulla pariatur?","At vero eos et accusamus et iusto odio dignissimos ducimus qui blanditiis praesentium voluptatum deleniti atque corrupti quos dolores et quas molestias excepturi sint occaecati cupiditate non provident, similique sunt in culpa qui officia deserunt mollitia animi, id est laborum et dolorum fuga. Et harum quidem rerum facilis est et expedita distinctio. Nam libero tempore, cum soluta nobis est eligendi optio cumque nihil impedit quo minus id quod maxime placeat facere possimus, omnis voluptas assumenda est, omnis dolor repellendus. Temporibus autem quibusdam et aut officiis debitis aut rerum necessitatibus saepe eveniet ut et voluptates repudiandae sint et molestiae non recusandae. Itaque earum rerum hic tenetur a sapiente delectus, ut aut reiciendis voluptatibus maiores alias consequatur aut perferendis doloribus asperiores repellat.']
|
||
return split(lines[org#util#random(3)-1],'\%70c\S*\zs \ze')
|
||
endfunction
|
||
|
||
function! s:SetRandomDate(...)
|
||
call s:OrgGetHead()
|
||
if a:0 == 1
|
||
let date_type = a:1
|
||
else
|
||
let date_type = ['DEADLINE','TIMESTAMP','SCHEDULED'][org#util#random(3)-1]
|
||
endif
|
||
if date_type != ''
|
||
call s:SetProp(date_type,org#randomData())
|
||
else
|
||
let hl = line('.')
|
||
let dmatch = match(getline(hl),'\s*<\d\d\d\d-\d\d-\d\d')
|
||
if dmatch > 0
|
||
let dmatch = dmatch - 1
|
||
call setline(hl,getline(hl)[:dmatch])
|
||
endif
|
||
let newd = org#randomData()
|
||
execute hl
|
||
execute "normal A ". newd
|
||
endif
|
||
endfunction
|
||
function! s:SetRandomTodo()
|
||
let newtodo = b:v.todoitems[org#util#random(3)-1]
|
||
if index(b:v.todoitems,matchstr(getline(line('.')),'^\*\+ \zs\S*\ze ')) >= 0
|
||
call setline(line('.'),matchstr(getline(line('.')),'^\*\+ ') . newtodo .
|
||
\ ' '. matchstr(getline(line('.')),'^\*\+ \S* \zs.*'))
|
||
else
|
||
call setline(line('.'),matchstr(getline(line('.')),'^\*\+ ') . newtodo .
|
||
\ ' '. matchstr(getline(line('.')),'^\*\+ \zs.*'))
|
||
endif
|
||
|
||
endfunction
|
||
|
||
function! s:UpdateHeadlineSums()
|
||
g/^\s*:TOTALCLOCKTIME/d
|
||
call OrgMakeDict()
|
||
let g:tempdict = {}
|
||
g/^\*\+ /let g:tempdict[line('.')] = b:v.org_dict.SumTime(line('.'),'ITEMCLOCKTIME')
|
||
let items = sort(map(copy(keys(g:tempdict)),"str2nr(v:val)"),'s:NumCompare')
|
||
let i = len(items) - 1
|
||
while i >= 0
|
||
if g:tempdict[items[i]] != '0:00'
|
||
call s:SetProp('TOTALCLOCKTIME',g:tempdict[items[i]],items[i])
|
||
endif
|
||
let i = i-1
|
||
endwhile
|
||
endfunction
|
||
|
||
function! s:IProp(headline,property)
|
||
let prop = a:property
|
||
let parent = s:OrgParentHead_l(a:headline)
|
||
if parent == 0
|
||
return get(b:v.org_inherited_defaults,prop)
|
||
else
|
||
return s:GetProperties(parent,0)[prop]
|
||
endif
|
||
endfunction
|
||
|
||
function! OrgMakeDictInherited(...)
|
||
let get_tags = ((a:0==1) && (a:1 =='get_tags_too')) ? 1 : 0
|
||
if (b:v.last_idict_time >= getftime(expand('%:p')) && (&modified==0))
|
||
" now check and return if last run had tags or this run doesn't need them
|
||
if (b:v.last_idict_with_tags == 1) || (get_tags == 0)
|
||
return
|
||
endif
|
||
endif
|
||
let b:v.last_idict_time = localtime()
|
||
let b:v.last_idict_with_tags = get_tags
|
||
if s:OrgVal('g:org_save_when_searched') > 0 | write! | endif
|
||
let b:v.org_dict = {}
|
||
call OrgProcessConfigLines()
|
||
let b:v.org_dict = {'0':{'c':[],'CATEGORY':b:v.org_inherited_defaults['CATEGORY'] }}
|
||
function! b:v.org_dict.iprop(ndx,property) dict
|
||
let prop = a:property
|
||
let ndx = a:ndx
|
||
let result = get(self[ndx] , prop,'')
|
||
if (result ==# '') && (ndx != 0)
|
||
"recurse up through parents in tree
|
||
let result = b:v.org_dict.iprop(self[ndx].parent,prop)
|
||
endif
|
||
return result
|
||
endfunction
|
||
execute 1
|
||
let next = 1
|
||
if s:IsText(line('.'))
|
||
let next = s:OrgNextHead()
|
||
endif
|
||
while next > 0
|
||
execute next
|
||
if getline(line('.'))[1] ==? ' '
|
||
let parent = 0
|
||
else
|
||
let parent = s:OrgParentHead()
|
||
endif
|
||
let b:v.org_dict[line('.')] = {'parent': parent}
|
||
let next = s:OrgNextHead()
|
||
endwhile
|
||
" parent properties assigned above, now explicity record CATEGORY for
|
||
" any headlines where CATEGORY won't be inherited
|
||
if get_tags == 0
|
||
silent execute 'g/^\s*:CATEGORY:/let b:v.org_dict[s:OrgGetHead()].CATEGORY = matchstr(getline(line(".")),":CATEGORY:\\s*\\zs.*")'
|
||
else
|
||
silent execute 'g/^\s*:CATEGORY:/let b:v.org_dict[s:OrgGetHead()].CATEGORY = matchstr(getline(line(".")),":CATEGORY:\\s*\\zs.*")'
|
||
silent g/^\*\+\s*\S/call s:GetBasicMeta(line('.'))
|
||
endif
|
||
endfunction
|
||
function! s:GetBasicMeta(line)
|
||
"let b:v.org_dict[a:line].CATEGORY = matchstr(getline(a:line),":CATEGORY:\\s*\\zs.*")
|
||
let this_line = getline(a:line)
|
||
let todo = matchstr(this_line,b:v.todoMatch)
|
||
"let b:v.org_dict[a:line].TODO = matchstr(getline(a:line),b:v.todoMatch)
|
||
if s:IsTagLine(a:line + 1)
|
||
let tags = matchstr(getline(a:line+1),'^\s*\zs.*')
|
||
else
|
||
let tags = ''
|
||
endif
|
||
let b:v.org_dict[a:line].props = {'ITEM': this_line,'TODO':todo,'TAGS':tags}
|
||
endfunction
|
||
function! OrgMakeDict()
|
||
if (b:v.last_dict_time >= getftime(expand('%:p')) && (&modified==0))
|
||
return
|
||
endif
|
||
" save new dict time to use in future searches
|
||
let b:v.last_dict_time = localtime()
|
||
"let b:v.org_dict = {}
|
||
"save buffer changes since last_dict
|
||
if s:OrgVal('g:org_save_when_searched') > 0 | write! | endif
|
||
" and recreate the dict
|
||
call OrgMakeDictInherited()
|
||
function! b:v.org_dict.SumTime(ndx,property) dict
|
||
let prop = a:property
|
||
let result = get(self[a:ndx].props , prop,'0:00')
|
||
" now recursion down the subtree of children in c
|
||
for item in self[a:ndx].c
|
||
let result = s:AddTime(result,b:v.org_dict.SumTime(item,prop))
|
||
endfor
|
||
return result
|
||
endfunction
|
||
function! b:v.org_dict.Sum(ndx,property) dict
|
||
let prop = a:property
|
||
let result = get(self[a:ndx].props , prop)
|
||
" now recursion down the subtree of children in c
|
||
for item in self[a:ndx].c
|
||
let result += b:v.org_dict.Sum(item,prop)
|
||
endfor
|
||
return result
|
||
endfunction
|
||
execute 1
|
||
let next = 1
|
||
if s:IsText(line('.'))
|
||
let next = s:OrgNextHead()
|
||
endif
|
||
while next > 0
|
||
execute next
|
||
let b:v.org_dict[line('.')].c = []
|
||
let b:v.org_dict[line('.')].props = s:GetProperties(line('.'),1)
|
||
let parent = b:v.org_dict[line('.')].parent
|
||
call add(b:v.org_dict[parent].c ,line('.'))
|
||
let next = s:OrgNextHead()
|
||
endwhile
|
||
endfunction
|
||
|
||
function! s:ClearSparseTreeOld()
|
||
set fdm=manual
|
||
silent exe '%s/^*x//'
|
||
silent exe 'undojoin | %s/^*o//'
|
||
"g/^*x/call substitute(getline(line(".")),'^*x',''))
|
||
"g/^*o/call substitute(getline(line(".")),'^*o',''))
|
||
|
||
call clearmatches()
|
||
set fdm=expr
|
||
echo "sparse tree view cleared"
|
||
endfunction
|
||
|
||
function! s:SparseTreeRun(term)
|
||
|
||
call s:ClearSparseLists()
|
||
let w:sparse_on = 1
|
||
execute 'g/' . a:term . '/call add(b:v.sparse_list,line("."))'
|
||
call s:SparseTreeDoFolds()
|
||
call clearmatches()
|
||
let g:org_first_sparse=1
|
||
let b:v.signstring= s:GetPlacedSignsString(bufnr("%"))
|
||
set fdm=expr
|
||
set foldlevel=0
|
||
let g:org_first_sparse=1
|
||
execute 'let @/ ="' . a:term .'"'
|
||
execute 'g/' . a:term . '/normal zv'
|
||
set hlsearch
|
||
execute 1
|
||
endfunction
|
||
|
||
function! s:SparseTreeDoFolds()
|
||
let i = len(b:v.sparse_list) - 1
|
||
while i >= 0
|
||
"if b:v.sparse_list[i] + g:org_sparse_lines_after > line("$")
|
||
if b:v.sparse_list[i] + 10 > line("$")
|
||
call remove(b:v.sparse_list, i) "insert(b:v.fold_list,0)
|
||
let i -= 1
|
||
continue
|
||
"elseif (i>0) && (b:v.sparse_list[i] < b:v.sparse_list[i-1] + g:org_sparse_lines_after)
|
||
elseif (i>0) && (b:v.sparse_list[i] < b:v.sparse_list[i-1] + 10)
|
||
call remove(b:v.sparse_list, i) "insert(b:v.fold_list,0)
|
||
let i -= 1
|
||
continue
|
||
else
|
||
let phead = s:OrgUltimateParentHead_l(b:v.sparse_list[i])
|
||
if phead >= 1
|
||
call insert(b:v.fold_list,phead-1)
|
||
else
|
||
" match is already on level 1 head
|
||
call insert(b:v.fold_list,b:v.sparse_list[i]-1)
|
||
endif
|
||
endif
|
||
|
||
let i -= 1
|
||
endwhile
|
||
"call map(b:v.sparse_list,"v:val + g:org_sparse_lines_after")
|
||
call map(b:v.sparse_list,"v:val + 10")
|
||
call insert(b:v.sparse_list, 1)
|
||
call add(b:v.fold_list, line("$"))
|
||
|
||
" sign method to potentially supersede list based method above
|
||
call s:DeleteSigns()
|
||
for item in b:v.sparse_list
|
||
execute "sign place " . item ." line=".item." name=fbegin buffer=".bufnr("%")
|
||
endfor
|
||
for item in b:v.fold_list
|
||
execute "sign place " . item ." line=".item." name=fend buffer=".bufnr("%")
|
||
endfor
|
||
let s:sparse_lines = {}
|
||
for item in b:v.sparse_list
|
||
let s:sparse_lines[item] = 1
|
||
let s:sparse_lines[item-1] = 1
|
||
endfor
|
||
for item in b:v.fold_list
|
||
let s:sparse_lines[item] = 1
|
||
endfor
|
||
" FoldTouch below instead of fdm line above to save time
|
||
" updating folds for just newly changed foldlevel lines
|
||
"call s:FoldTouch()
|
||
|
||
endfunction
|
||
|
||
function! s:ClearSparseLists()
|
||
" mylist with lines of matches
|
||
let b:v.sparse_list = []
|
||
" foldlist with line before previous level 1 parent
|
||
let b:v.fold_list = []
|
||
let b:v.sparse_heads = []
|
||
endfunction
|
||
function! s:ClearSparseTree()
|
||
" mylist with lines of matches
|
||
let w:sparse_on = 0
|
||
let b:v.sparse_list = []
|
||
" foldlist with line before previous level 1 parent
|
||
let b:v.fold_list = []
|
||
set fdm=expr
|
||
set foldlevel=1
|
||
execute 1
|
||
endfunction
|
||
|
||
function! s:FoldTouch()
|
||
" not used right now, since speed increase over
|
||
" set fdm=expr is uncertain, and was having problems
|
||
" in cleanly undoing it.
|
||
"
|
||
" touch each line in lists to update their fold levels
|
||
let i = 0
|
||
while i < len(b:v.sparse_list)
|
||
execute b:v.sparse_list[i]
|
||
" insert letter 'b' to force level update and then undo
|
||
silent execute "normal! ib"
|
||
silent execute "normal! u"
|
||
execute b:v.fold_list[i]
|
||
silent execute "normal! ib"
|
||
silent execute "normal! u"
|
||
let i += 1
|
||
endwhile
|
||
endfunction
|
||
|
||
function! s:OrgIfExpr()
|
||
let mypattern = ''
|
||
" two wrapper subst statements around middle
|
||
" subst are to make dates work properly with substitute/split
|
||
" operation
|
||
let ifstring_list = [[]]
|
||
let test_str = g:org_search_spec
|
||
if test_str[0] !~ '[+-]'
|
||
let test_str = '+' . test_str
|
||
endif
|
||
|
||
let ndx=0
|
||
let result_if_list = []
|
||
"try
|
||
while 1
|
||
" text string
|
||
" curly bracket reg ex string
|
||
" numeric comparison
|
||
" single operand -- TAG or TODO
|
||
|
||
let m = matchlist(test_str,'^\(|'
|
||
\ . '\|[+-]\w\{-}[!<>\=]=*".\{-}"'
|
||
\ . '\|[+-]\w\{-}[!\=]=*{.\{-}}'
|
||
\ . '\|[+-]\w\{-}[=<>!]=*[0-9+-.][0-9.]*'
|
||
\ . '\|[+-]\w*\)'
|
||
\ . '\(.*\)')
|
||
if m[1] == '|'
|
||
call add(ifstring_list,[])
|
||
let ndx += 1
|
||
let test_str = m[2]
|
||
if test_str !~ '+\|-'
|
||
let test_str = '+' . test_str
|
||
endif
|
||
elseif m[1] ># ''
|
||
call add(ifstring_list[ndx],m[1])
|
||
let test_str = m[2]
|
||
if test_str == ''
|
||
break
|
||
endif
|
||
else
|
||
break
|
||
endif
|
||
endwhile
|
||
|
||
for ifstr in ifstring_list
|
||
|
||
let b:v.my_if_list = ifstr
|
||
let ifexpr = ''
|
||
" okay, right now we have split list with each item prepended by + or -
|
||
" now change each item to be a pattern match equation in parens
|
||
" e.g.,'( prop1 =~ propval) && (prop2 =~ propval) && (thisline =~tag)
|
||
let i = 0
|
||
"using while structure because for structure doesn't allow changing
|
||
" items?
|
||
while i < len(b:v.my_if_list)
|
||
let item = b:v.my_if_list[i]
|
||
if item[0] !~ '+\|-'
|
||
let item = '+' . item
|
||
endif
|
||
" Propmatch has '=' sign and something before and after
|
||
if item =~ 'TEXT=\S.*'
|
||
let mtch = matchlist(item[1:],'\(\S.*\)=\(\S.*\)')
|
||
let b:v.my_if_list[i] = "(s:Range_Search('" . mtch[2][1:-2] . "','nbW',"
|
||
let b:v.my_if_list[i] .= 'tbegin,tend)> 0)'
|
||
let i += 1
|
||
" loop to next item
|
||
continue
|
||
endif
|
||
if item =~ '\S.*[=><]\S.*'
|
||
if item =~ '[^<>!]=\\('
|
||
let item = substitute(item,'=','=~','')
|
||
elseif item =~ '[^!]={'
|
||
let item = substitute(item,'[^!]\zs=','=~','')
|
||
let item = substitute(item,'{','"','')
|
||
let item = substitute(item,'}','"','')
|
||
elseif item =~ '!={'
|
||
let item = substitute(item,'!=','!~','')
|
||
let item = substitute(item,'{','"','')
|
||
let item = substitute(item,'}','"','')
|
||
elseif item =~ '[^<>!]=[^=]'
|
||
let item = substitute(item,'=','==','')
|
||
endif
|
||
let pat = '\(\S\{-}\)\(==\|=\~\|!\~\|>=\|<=\|!=\|<\|>\)\(\S.*\)'
|
||
let mtch = matchlist(item[1:],pat)
|
||
let mtch[1] = toupper(mtch[1])
|
||
if mtch[3] =~ '^[+\-0-9.][0-9.]*$'
|
||
" numeric comparison
|
||
let b:v.my_if_list[i] = (item[0]=='-' ? '!' : '') . '(get(lineprops,"' . mtch[1] . '") ' . mtch[2]. mtch[3] . ')'
|
||
else
|
||
" string comparison
|
||
let rightside="'".mtch[3][1:-2]."'"
|
||
let b:v.my_if_list[i] = (item[0]=='-' ? '!' : '') . '(get(lineprops,"' . mtch[1] . '","") ' . mtch[2]. rightside. ')'
|
||
" line below is addd on to exclude headings not
|
||
" having an entry at all from the comparison
|
||
" \ . '&& (get(lineprops,"' . mtch[1] . '","") != "")'
|
||
endif
|
||
let i += 1
|
||
" loop to next item
|
||
continue
|
||
endif
|
||
|
||
" it must be a todo or tag item
|
||
if item[0] ==? '+'
|
||
let op = '=~'
|
||
elseif item[0] ==? '-'
|
||
let op = '!~'
|
||
endif
|
||
" if it's in todoitems assume it's a todo search item, not a tag
|
||
if index(b:v.todoitems,item[1:]) >= 0
|
||
let item = '(lineprops.TODO ' . op . " '" . item[1:] . "')"
|
||
"let item = '(lineprops.TODO ' . op . " '^\\*\\+\\s*" . item[1:] . "')"
|
||
let b:v.my_if_list[i] = item
|
||
elseif item[1:] =~? 'UNFINISHED_TODO\|UNDONE_TODO'
|
||
let item = '(lineprops.TODO ' . op . " '" . b:v.todoNotDoneMatch[11:] . "')"
|
||
let b:v.my_if_list[i] = item
|
||
elseif item[1:] =~? 'FINISHED_TODO\|DONE_TODO'
|
||
let item = '(lineprops.TODO ' . op . " '" . b:v.todoDoneMatch[11:] . "')"
|
||
let b:v.my_if_list[i] = item
|
||
elseif item[1:] ==? 'ANY_TODO'
|
||
let item = '(lineprops.TODO ' . op . " '" . b:v.todoMatch[11:] . "')"
|
||
let b:v.my_if_list[i] = item
|
||
else
|
||
"not a todo so we treat it as a tag item
|
||
let item = '(get(lineprops,"TAGS","") ' . op . " ':" . item[1:] . ":')"
|
||
let b:v.my_if_list[i] = item
|
||
endif
|
||
let i += 1
|
||
endwhile
|
||
let i = 0
|
||
let b:v.check1 = b:v.my_if_list
|
||
let ifexpr = ''
|
||
while i < len(b:v.my_if_list)
|
||
let ifexpr .= b:v.my_if_list[i]
|
||
if i < len(b:v.my_if_list) - 1
|
||
let ifexpr .= ' && '
|
||
endif
|
||
let i += 1
|
||
endwhile
|
||
|
||
"return ifexpr
|
||
call add(result_if_list, ifexpr)
|
||
endfor
|
||
" let succeeded = 1
|
||
"finally
|
||
" if !exists('succeeded')
|
||
" return []
|
||
" else
|
||
return result_if_list
|
||
" endif
|
||
endfunction
|
||
|
||
function! s:CheckIfExpr(line,ifexpr,...)
|
||
" this is 'ifexpr' eval func used for agenda date searches
|
||
let headline = s:OrgGetHead_l(a:line)
|
||
" 0 arg is to not get start and end line numbers
|
||
let lineprops=s:GetProperties(headline,0)
|
||
" _thisline_ is variable evaluated in myifexpr
|
||
let thisline = getline(headline)
|
||
if s:IsTagLine(headline + 1)
|
||
let thisline .= ' ' . getline(headline+1)
|
||
endif
|
||
let result = 0
|
||
for item in a:ifexpr
|
||
if eval(item) == 1
|
||
let result = 1
|
||
break
|
||
endif
|
||
endfor
|
||
return result
|
||
|
||
endfunction
|
||
|
||
function! FileDict()
|
||
return s:filedict
|
||
endfunction
|
||
|
||
function! s:OrgIfExprResults(ifexpr,...)
|
||
" ifexpr has single compound expression that will evaluate
|
||
" as true only for desired lines
|
||
let sparse_search = 0
|
||
if a:0 > 0
|
||
let sparse_search = a:1
|
||
endif
|
||
|
||
"let myifexpr = a:ifexpr
|
||
|
||
execute 1
|
||
if getline(line('.'))!~ '^\*\+ '
|
||
let headline = s:OrgNextHead()
|
||
else
|
||
let headline = 1
|
||
endif
|
||
let g:checkexpr = a:ifexpr
|
||
while 1
|
||
if headline > 0
|
||
execute headline
|
||
" _thisline_ is variable evaluated in myifexpr
|
||
let thisline = getline(headline)
|
||
if s:IsTagLine(headline + 1)
|
||
let thisline .= ' ' . getline(headline+1)
|
||
endif
|
||
" lineprops is main variable tested in 'ifexpr'
|
||
" expression that gets evaluated
|
||
"let lineprops = s:GetProperties(headline,1)
|
||
let lineprops = b:v.org_dict[headline].props
|
||
|
||
for if_item in a:ifexpr
|
||
" next line is to fix for text area search
|
||
" now that we can reference tbegin and tend
|
||
let myifexpr = substitute(if_item,'\cTBEGIN,TEND',get(lineprops,'LINE') .','. get(lineprops,'TEND'),"")
|
||
"
|
||
"******** eval() is what does it all ***************
|
||
if eval(myifexpr)
|
||
if sparse_search
|
||
let keyval = headline
|
||
else
|
||
"let keyval = s:PrePad(index(s:agenda_files_copy, lineprops.file . '.org'),3,'0') . s:PrePad(headline,5,'0')
|
||
"let keyval = s:PrePad(lineprops.file,3,'0') . s:PrePad(headline,5,'0')
|
||
let keyval = s:PrePad(s:filenum,3,'0') . s:PrePad(headline,5,'0')
|
||
endif
|
||
|
||
let g:adict[keyval]=lineprops
|
||
if !exists('g:adict[keyval].CATEGORY')
|
||
let g:adict[keyval].CATEGORY = b:v.org_dict.iprop(headline,'CATEGORY')
|
||
endif
|
||
break
|
||
endif
|
||
endfor
|
||
normal l
|
||
let headline = s:OrgNextHead()
|
||
else
|
||
break
|
||
endif
|
||
endwhile
|
||
endfunction
|
||
|
||
function! s:MakeResults(search_spec,...)
|
||
let s:filedict = copy(g:agenda_files)
|
||
let sparse_search = 0
|
||
if a:0 > 0
|
||
let sparse_search = a:1
|
||
endif
|
||
let save_cursor = getpos(".")
|
||
"let curfile = substitute(expand("%"),' ','\\ ','g')
|
||
let curfile = substitute(expand("%:p"),' ','\\ ','g')
|
||
let curtab = tabpagenr()
|
||
let curwin = winnr()
|
||
|
||
let g:org_search_spec = a:search_spec
|
||
let g:org_todoitems=[]
|
||
let g:adict = {}
|
||
let g:datedict = {}
|
||
let s:agenda_files_copy = copy(g:agenda_files)
|
||
" fix so copy doesn't have full path. . .
|
||
"call map(s:agenda_files_copy, 'matchstr(v:val,"[\\/]") > "" ? matchstr(v:val,"[^/\\\\]*$") : v:val')
|
||
if sparse_search
|
||
call OrgMakeDict()
|
||
let ifexpr = s:OrgIfExpr()
|
||
call s:OrgIfExprResults(ifexpr,sparse_search)
|
||
else
|
||
let g:in_agenda_search = 1
|
||
for file in g:agenda_files
|
||
"execute 'tab drop ' . file
|
||
call org#LocateFile(file)
|
||
call org#SaveLocation()
|
||
" let bnum = bufnr(file)
|
||
" if bnum == -1
|
||
" execute 'tabedit ' . file
|
||
" else
|
||
" execute 'b' . bnum
|
||
" endif
|
||
let s:filenum = index(g:agenda_files,file)
|
||
if g:org_search_spec =~ '[<>\=!]'
|
||
call OrgMakeDict()
|
||
else
|
||
" it's just todos and/or tags
|
||
call OrgMakeDictInherited('get_tags_too')
|
||
endif
|
||
let ifexpr = s:OrgIfExpr()
|
||
let g:org_todoitems = extend(g:org_todoitems,b:v.todoitems)
|
||
call s:OrgIfExprResults(ifexpr,sparse_search)
|
||
call org#RestoreLocation()
|
||
endfor
|
||
unlet g:in_agenda_search
|
||
"call org#LocateFile(curfile)
|
||
execute 'tabnext ' . curtab
|
||
execute curwin . 'wincmd w'
|
||
endif
|
||
call setpos(".",save_cursor)
|
||
endfunction
|
||
function! s:DaysInMonth(date)
|
||
let month = str2nr(a:date[5:6])
|
||
let year = str2nr(a:date[0:3])
|
||
if (month == 2) && (year % 4)
|
||
let days = 28
|
||
elseif month == 2
|
||
let days = 29
|
||
elseif index([1,3,5,7,8,10,12],month) >= 0
|
||
let days = 31
|
||
else
|
||
let days = 30
|
||
endif
|
||
return days
|
||
endfunction
|
||
|
||
function! s:MakeAgenda(date,count,...)
|
||
if a:0 >= 1
|
||
let g:org_search_spec = a:1
|
||
else
|
||
let g:org_search_spec = ''
|
||
endif
|
||
let as_today = ''
|
||
if a:0 >= 2
|
||
let as_today = a:2
|
||
endif
|
||
|
||
"call org#SaveLocation()
|
||
let curtab = tabpagenr()
|
||
let curwin = winnr()
|
||
|
||
let l:count = a:count
|
||
if l:count ==? 'd' | let l:count = 1 | endif
|
||
if l:count ==? 'w'
|
||
let g:agenda_startdate = calutil#cal(calutil#jul(a:date) - calutil#dow(a:date))
|
||
let g:org_agenda_days=7
|
||
elseif l:count ==? 'm'
|
||
let g:agenda_startdate = a:date[0:7].'01'
|
||
let g:org_agenda_days = s:DaysInMonth(a:date)
|
||
elseif l:count ==? 'y'
|
||
let g:agenda_startdate = a:date[0:3].'-01-01'
|
||
let g:org_agenda_days = ( a:date[0:3] % 4 == 0 ) ? 366 : 365
|
||
else
|
||
let g:agenda_startdate = a:date
|
||
let g:org_agenda_days = l:count
|
||
endif
|
||
if l:count == 1 | let as_today = g:agenda_startdate | endif
|
||
let g:adict = {}
|
||
let s:filedict = copy(g:agenda_files)
|
||
let s:agenda_files_copy = copy(g:agenda_files)
|
||
let g:datedict = {}
|
||
call s:MakeCalendar(g:agenda_startdate,g:org_agenda_days)
|
||
let g:in_agenda_search=1
|
||
for file in g:agenda_files
|
||
call org#LocateFile(file)
|
||
call org#SaveLocation()
|
||
" for now g:org_search_spec is limited to tags and
|
||
" todo prop which are part of OrgMakeDictInherited
|
||
" If ever want to expand to general props can
|
||
" add OrgMakeDict()
|
||
call OrgMakeDictInherited()
|
||
let s:filenum = index(g:agenda_files,file)
|
||
let t:agenda_date=a:date
|
||
if as_today ># ''
|
||
call s:GetDateHeads(g:agenda_startdate,g:org_agenda_days,as_today)
|
||
else
|
||
call s:GetDateHeads(g:agenda_startdate,g:org_agenda_days)
|
||
endif
|
||
call org#RestoreLocation()
|
||
endfor
|
||
unlet g:in_agenda_search
|
||
|
||
"call org#RestoreLocation()
|
||
exec 'tabnext ' . curtab
|
||
exec curwin . 'wincmd w'
|
||
|
||
endfunction
|
||
|
||
function! s:NumCompare(i1, i2)
|
||
return a:i1 == a:i2 ? 0 : a:i1 > a:i2 ? 1 : -1
|
||
endfunc
|
||
|
||
function! OrgRunSearch(search_spec,...)
|
||
"set mouseshape-=n:busy,v:busy,i:busy
|
||
if bufnr('Calendar') > 0
|
||
execute 'bw!' . bufnr('Calendar')
|
||
endif
|
||
|
||
try
|
||
|
||
if bufnr('__Agenda__') >= 0
|
||
"bwipeout __Agenda__
|
||
endif
|
||
|
||
let g:agenda_head_lookup={}
|
||
let sparse_search = 0
|
||
let search_type = ''
|
||
if a:0 > 0
|
||
if a:1 == 1
|
||
let sparse_search = a:1
|
||
else
|
||
let search_type=a:1
|
||
endif
|
||
endif
|
||
let g:adict={}
|
||
let g:agenda_date_dict={}
|
||
if !exists("g:agenda_files") || (g:agenda_files == [])
|
||
if has('dialog_con') || has('dialog_gui')
|
||
unsilent call confirm("No agenda files defined. Will add current file to agenda files.")
|
||
endif
|
||
call s:CurfileAgenda()
|
||
endif
|
||
if exists('b:v.sparse_list') && (len(b:v.sparse_list) > 0)
|
||
call s:ClearSparseTree()
|
||
endif
|
||
call s:MakeResults(a:search_spec,sparse_search)
|
||
|
||
if sparse_search
|
||
call s:ResultsToSparseTree()
|
||
else
|
||
call s:ResultsToAgenda( search_type )
|
||
endif
|
||
|
||
finally
|
||
"set mouseshape-=n:busy,v:busy,i:busy
|
||
endtry
|
||
endfunction
|
||
function! s:ResultsToAgenda( search_type )
|
||
" FOR OrgRunSearch()
|
||
":AAgenda
|
||
:EditAgenda
|
||
let b:v={}
|
||
call s:AgendaBufSyntax()
|
||
set nowrap
|
||
|
||
call s:DoAgendaMaps()
|
||
|
||
let i = 0
|
||
call s:ADictPlaceSigns()
|
||
let b:v.heading_marks_dict = {}
|
||
"call setline(line('$'), ["Headlines matching search spec: ".g:org_search_spec,''])
|
||
let lines = []
|
||
let lines = lines + ["Headlines matching search spec: ".g:org_search_spec,'']
|
||
if a:search_type ==? 'agenda_todo'
|
||
let msg = "Press <num>r to re-search: "
|
||
let numstr= ''
|
||
nmap <buffer> r :call OrgRunSearch(g:org_search_spec,'agenda_todo')<cr>
|
||
let tlist = ['ANY_TODO','UNFINISHED_TODOS', 'FINISHED_TODOS'] + s:Union(g:org_todoitems,[])
|
||
for item in tlist
|
||
let num = index(tlist,item)
|
||
let numstr .= '('.num.')'.item.' '
|
||
execute "nmap <buffer> ".num."r :silent call OrgRunCustom({'redo_num':line('.'), 'type':'tags-todo', 'spec':'". tlist[num] . "'})<CR>"
|
||
endfor
|
||
"call add(lines,split(msg.numstr,'\%72c\S*\zs '))
|
||
let lines = lines + split(msg.numstr,'\%72c\S*\zs ')
|
||
call add(lines,'')
|
||
endif
|
||
for key in sort(keys(g:adict))
|
||
call add(lines , key . ' ' .
|
||
\ printf("%-12.12s",g:adict[key].CATEGORY ) . ' ' .
|
||
\ s:PrePad(matchstr(g:adict[key].ITEM,'^\*\+ '),8) .
|
||
\ matchstr(g:adict[key].ITEM,'\* \zs.*$'))
|
||
let i += 1
|
||
endfor
|
||
call append(s:agenda_insert_point,lines)
|
||
endfunction
|
||
function! s:ToFromAgenda()
|
||
if bufname('%') == '__Agenda__'
|
||
for i in range(1, winnr('$'))
|
||
exe i . 'wincmd w'
|
||
if bufname('%') !~ '__Agenda__\|__Calendar'
|
||
break
|
||
endif
|
||
endfor
|
||
"windo if bufname('%') !~ '__Agenda__\|__Calendar' | let return | endif
|
||
elseif bufwinnr('__Agenda__') > 0
|
||
exec bufwinnr('__Agenda__') . 'wincmd w'
|
||
endif
|
||
endfunction
|
||
function! s:DoAgendaMaps()
|
||
execute "source " . s:sfile . '/vimorg-agenda-mappings.vim'
|
||
|
||
command! -buffer -nargs=* Agenda :call OrgAgendaCommand(<f-args>)
|
||
|
||
" user can have a function in their vimrc to have their own
|
||
" agenda mappings
|
||
if exists('*OrgCustomAgendaMaps')
|
||
call OrgCustomAgendaMaps()
|
||
endif
|
||
endfunction
|
||
|
||
function! s:OrgAgendaTab()
|
||
if getline(line(".")) !~ '^\d\+'
|
||
return
|
||
endif
|
||
let thisline = getline(line('.'))
|
||
let file = s:agenda_files_copy[str2nr(matchstr(thisline, '^\d\d\d'))]
|
||
if bufwinnr(file) == -1
|
||
call s:OrgAgendaToBuf()
|
||
else
|
||
call s:MoveToHeadingFromAgenda(line('.'))
|
||
if b:v.chosen_agenda_heading != line('.')
|
||
"back to agenda and do agendatobuf from there
|
||
call org#LocateFile('__Agenda__')
|
||
call s:OrgAgendaToBuf()
|
||
else
|
||
" we're in file buffer, ready to cycle
|
||
call s:OrgCycle(line('.'))
|
||
call org#LocateFile('__Agenda__')
|
||
endif
|
||
endif
|
||
endfunction
|
||
function! s:ToggleHeadingMark(line)
|
||
let line = a:line
|
||
if (bufname('%') == '__Agenda__' && getline(line) !~ '^\d\{8}\s')
|
||
\ || ((&filetype == 'org') && getline(line) !~ b:v.headMatch)
|
||
return
|
||
endif
|
||
let list_item = get(b:v.heading_marks_dict, line)
|
||
if list_item == 0
|
||
let b:v.heading_marks_dict[line] = 1
|
||
execute "sign place " . line('.') . " line=" . line('.') . " name=marked buffer=".bufnr("%")
|
||
else
|
||
unlet b:v.heading_marks_dict[line]
|
||
execute "sign unplace " . line('.') . " buffer=".bufnr("%")
|
||
endif
|
||
endfunction
|
||
function! s:DeleteHeadingMarks()
|
||
if bufname('%')=='__Agenda__'
|
||
for item in keys(b:v.heading_marks_dict)
|
||
execute "sign unplace " . item . " buffer=".bufnr("%")
|
||
endfor
|
||
else
|
||
sign unplace *
|
||
endif
|
||
let b:v.heading_marks = []
|
||
let b:v.heading_marks_dict = {}
|
||
endfunction
|
||
function! s:ResultsToSparseTree()
|
||
"call s:ClearSparseTree()
|
||
let w:sparse_on = 1
|
||
let temp = []
|
||
for key in keys(g:adict)
|
||
call add(temp,g:adict[key].LINE)
|
||
endfor
|
||
let b:v.sparse_list = sort(temp,'s:NumCompare')
|
||
"for key in keys(g:adict)
|
||
" call add(b:v.sparse_heads,str2nr(key))
|
||
"endfor
|
||
"for item in sort(b:v.sparse_heads,'NumCompare')
|
||
call sort(b:v.fold_list,"s:NumCompare")
|
||
call s:SparseTreeDoFolds()
|
||
"for item in sort(b:v.fold_list,'NumCompare')
|
||
set fdm=expr
|
||
set foldlevel=0
|
||
call clearmatches()
|
||
for item in b:v.sparse_list
|
||
if item > 11
|
||
execute item - g:org_sparse_lines_after
|
||
normal! zv
|
||
call matchadd('Search','\%' . (item - g:org_sparse_lines_after) . 'l')
|
||
endif
|
||
endfor
|
||
execute 1
|
||
endfunction
|
||
|
||
function! s:TestTime()
|
||
let g:timestart=join(reltime(), ' ')
|
||
let g:start = strftime("%")
|
||
let i = 0
|
||
set fdm=expr
|
||
let g:timefinish=join(reltime(), ' ')
|
||
echo g:timestart . ' --- ' . g:timefinish
|
||
endfunction
|
||
function! s:TestTime2(fu)
|
||
let g:timestart=join(reltime(), ' ')
|
||
let g:start = strftime("%")
|
||
let i = 0
|
||
execute a:fu
|
||
let g:timefinish=join(reltime(), ' ')
|
||
echo g:timestart . ' --- ' . g:timefinish
|
||
endfunction
|
||
|
||
function! s:ADictPlaceSigns()
|
||
let myl=[]
|
||
call s:DeleteSigns() " signs were placed during search
|
||
for key in keys(g:adict)
|
||
let headline = matchstr(key,'^\d\d\d\zs\d\+')
|
||
let filenum = str2nr(key[0:2])
|
||
let buf = bufnr(s:agenda_files_copy[filenum])
|
||
try
|
||
silent execute "sign place " . headline . " line="
|
||
\ . headline . " name=piet buffer=" . buf
|
||
catch
|
||
echo "ERROR: headline " . headline . ' and buf ' .buf
|
||
echo key .', '. matchstr(key,'^.*\ze_\d\+$')
|
||
finally
|
||
endtry
|
||
endfor
|
||
endfunction
|
||
function! s:DateDictPlaceSigns()
|
||
let myl=[]
|
||
call s:DeleteSigns() " signs were placed in GetDateHeads
|
||
for key in keys(g:agenda_date_dict)
|
||
let myl = get(g:agenda_date_dict[key], 'l')
|
||
if len(myl) > 0
|
||
for item in myl
|
||
let dateline = matchstr(item,'^\d\d\d\zs\d\+')
|
||
let filenum = str2nr(item[0:2])
|
||
let buf = bufnr(s:agenda_files_copy[filenum])
|
||
try
|
||
silent execute "sign place " . dateline . " line="
|
||
\ . dateline . " name=piet buffer=" . buf
|
||
catch
|
||
echo "ERROR: headline " . headline . ' and buf ' . buf . ' and dateline ' . dateline
|
||
|
||
echo (matchstr(item,'^\d\+\s\+\zs\S\+') . '.org')
|
||
finally
|
||
endtry
|
||
endfor
|
||
endif
|
||
endfor
|
||
endfunction
|
||
|
||
function! s:DateDictToScreen()
|
||
":%d "delete all lines
|
||
let lines = ["Press <f> or <b> for next or previous period, q to close agenda," ,
|
||
\ "<Enter> on a heading to synch main file, <ctl-Enter> to goto line," ,
|
||
\ "<tab> to cycle heading text, <shift-Enter> to cycle Todos.",'']
|
||
let search_spec = g:org_search_spec ># '' ? g:org_search_spec : 'None - include all heads'
|
||
let d = g:org_agenda_days
|
||
let start_week = 'W' . org#ISODateToYWD(g:agenda_startdate)[1]
|
||
let end_date = calutil#cal(calutil#jul(g:agenda_startdate)+g:org_agenda_days-1)
|
||
let end_week = 'W' . org#ISODateToYWD(end_date)[1]
|
||
if d==1 | let type = 'Day-agenda (' . start_week . '):'
|
||
elseif d==7 | let type = 'Week-agenda (' . start_week . '):'
|
||
elseif (d>27 && d<32) | let type = 'Month-agenda (' . start_week . '-' . end_week . '):'
|
||
elseif (d>364 && d<366) | let type = 'Year-agenda:'
|
||
else | let type = 'Agenda:'
|
||
endif
|
||
|
||
call add(lines,"Agenda view for " . g:agenda_startdate
|
||
\ . " to ". calutil#cal(calutil#jul(g:agenda_startdate)+g:org_agenda_days-1)
|
||
\ . ' matching FILTER: ' . search_spec )
|
||
call add(lines,'')
|
||
call add(lines,type)
|
||
|
||
call s:DateDictPlaceSigns()
|
||
let b:v.heading_marks_dict = {}
|
||
let gap = 0
|
||
let mycount = len(keys(g:agenda_date_dict))
|
||
for key in sort(keys(g:agenda_date_dict))
|
||
if empty(g:agenda_date_dict[key].l)
|
||
let gap +=1
|
||
call add(lines, g:agenda_date_dict[key].marker)
|
||
else
|
||
if (gap > g:org_agenda_skip_gap) && (g:org_agenda_minforskip <= mycount)
|
||
call remove(lines, len(lines)-gap, -1 )
|
||
let lines = lines + ['',' [. . . ' .gap. ' empty days omitted ]','']
|
||
endif
|
||
let gap = 0
|
||
call add(lines, g:agenda_date_dict[key].marker)
|
||
if ((g:org_agenda_days == 1) || (key == strftime("%Y-%m-%d"))) && exists('g:org_timegrid') && (g:org_timegrid != [])
|
||
let lines = lines + s:PlaceTimeGrid(g:agenda_date_dict[key].l)
|
||
else
|
||
let lines = lines + g:agenda_date_dict[key].l
|
||
endif
|
||
endif
|
||
endfor
|
||
|
||
if (gap > g:org_agenda_skip_gap) && (g:org_agenda_minforskip <= mycount)
|
||
call remove(lines, len(lines)-gap, -1 )
|
||
let lines = lines + ['',' [. . . ' .gap. ' empty days omitted ]','']
|
||
endif
|
||
" finally, place all the prepared result lines in agenda buffer
|
||
call append(s:agenda_insert_point-1, lines)
|
||
endfunction
|
||
function! s:TimeGridSort(s1, s2)
|
||
return (a:s1[23:] == a:s2[23:]) ? 0 : (a:s1[23:] > a:s2[23:]) ? 1 : -1
|
||
endfunction
|
||
function! s:PlaceTimeGrid(lines)
|
||
let lines = a:lines
|
||
" assemble timegrid for agenda view
|
||
if lines[0] =~ '\%24c\d\d:\d\d'
|
||
"if, at least one time item put grid lines in and sort with other time items
|
||
let grid = s:TimeGrid(g:org_timegrid[0],g:org_timegrid[1],g:org_timegrid[2])
|
||
let lines = grid + lines
|
||
let i = len(grid) - 1
|
||
while ((i < len(lines)) && matchstr(lines[i],'\%24c\d\d:\d\d') )
|
||
let i += 1
|
||
endwhile
|
||
let lines = sort(lines[0:i-1], 's:TimeGridSort') + lines[i :]
|
||
" now delete duplicates where grid is same as actual entry
|
||
" skip this part until rest is working. . .
|
||
"let i = 0
|
||
"while i < len(lines)
|
||
" let match1 = matchstr(getline(line('.')),'\%24c.*\%29c')
|
||
" let match2 = matchstr(getline(line('.')-1),'\%24c.*\%29c')
|
||
" if match1 ==? match2
|
||
" if match1[0] ==? ' '
|
||
" normal ddk
|
||
" else
|
||
" normal kdd
|
||
" endif
|
||
" endif
|
||
" exec line('.')-1
|
||
"endwhile
|
||
endif
|
||
return lines
|
||
endfunction
|
||
function! OrgRunAgenda(date,count,...)
|
||
try
|
||
|
||
if bufname('%') ==? '__Agenda__'
|
||
"wincmd k
|
||
endif
|
||
let g:agenda_head_lookup={}
|
||
let win = bufwinnr('Calendar')
|
||
let file = expand('%:p')
|
||
if win >= 0
|
||
execute win . 'wincmd w'
|
||
normal ggjjj
|
||
wincmd l
|
||
execute 'bw!' . bufnr('Calendar')
|
||
|
||
endif
|
||
if !exists("g:agenda_files") || (g:agenda_files == [])
|
||
unsilent call confirm("No agenda files defined. Will add current file to agenda files.")
|
||
call s:CurfileAgenda()
|
||
endif
|
||
if exists('b:v.sparse_list') && (len(b:v.sparse_list) > 0)
|
||
call s:ClearSparseTree()
|
||
endif
|
||
" a:1 is search_spec, a:2 is "today" for search
|
||
if a:0 == 1
|
||
call s:MakeAgenda(a:date,a:count,a:1)
|
||
elseif a:0 == 2
|
||
call s:MakeAgenda(a:date,a:count,a:1,a:2)
|
||
else
|
||
call s:MakeAgenda(a:date,a:count)
|
||
endif
|
||
AAgenda
|
||
call s:SetupDateAgendaWin()
|
||
"we are no in newly created agenda buf/window
|
||
|
||
for key in keys(g:agenda_date_dict)
|
||
call sort(g:agenda_date_dict[key].l, 's:AgendaCompare')
|
||
endfor
|
||
call s:DateDictToScreen()
|
||
if win >= 0
|
||
let year = matchstr(t:agenda_date,'\d\d\d\d')
|
||
let month = matchstr(t:agenda_date,'\d-\zs\d\d\ze-')
|
||
execute 'Calendar ' . year .' '. str2nr(month)
|
||
execute bufwinnr('Agenda').'wincmd w'
|
||
endif
|
||
" we're in agenda, do rigamarole to get top column heading window,
|
||
" if any, back to zero height
|
||
let curheight=winheight(0)
|
||
call org#LocateFile(file)
|
||
"wincmd k
|
||
resize
|
||
call org#LocateFile('__Agenda__')
|
||
"wincmd j
|
||
execute 1
|
||
execute 'resize ' . curheight
|
||
|
||
finally
|
||
"set mouseshape-=n:busy,v:busy,i:busy
|
||
endtry
|
||
|
||
endfunction
|
||
function! OrgRunCustom(arg)
|
||
call s:RunCustom(a:arg)
|
||
endfunction
|
||
function! s:SetupDateAgendaWin()
|
||
EditAgenda
|
||
let b:v={}
|
||
call s:AgendaBufSyntax()
|
||
set nowrap
|
||
call s:DoAgendaMaps()
|
||
|
||
command! -buffer -nargs=* Agenda :call OrgAgendaCommand(<f-args>)
|
||
|
||
endfunction
|
||
|
||
function! OrgRefreshCalendarAgenda()
|
||
let g:org_search_spec = matchstr(getline(5),'FILTER:\s*\zs.*$')
|
||
if g:org_search_spec =~ '\c^None'
|
||
let g:org_search_spec = ''
|
||
endif
|
||
call OrgRunCustom({'redo_num': line('.'),'type':'agenda', 'agenda_date': g:agenda_startdate, 'agenda_duration': g:org_agenda_days, 'spec': g:org_search_spec})
|
||
endfunction
|
||
function! s:Resize()
|
||
let cur = winheight(0)
|
||
resize
|
||
resize cur
|
||
endfunction
|
||
|
||
function! s:GetDateHeads(date1,count,...)
|
||
" a:1 is date for warnings
|
||
let save_cursor=getpos(".")
|
||
if g:org_search_spec ># ''
|
||
let b:v.agenda_ifexpr = s:OrgIfExpr()
|
||
endif
|
||
let g:date1 = a:date1
|
||
let date1 = a:date1
|
||
let date2 = calutil#Jul2Cal(calutil#Cal2Jul(split(date1,'-')[0],split(date1,'-')[1],split(date1,'-')[2]) + a:count)
|
||
execute 1
|
||
while search('[^-][[<]\d\d\d\d-\d\d-\d\d','W') > 0
|
||
"if ((g:org_search_spec ># '') && (s:CheckIfExpr(line("."),b:v.agenda_ifexpr)==0))
|
||
" continue
|
||
"endif
|
||
let repeatlist = []
|
||
let line = getline(line("."))
|
||
let datematch = matchstr(line,'[[<]\d\d\d\d-\d\d-\d\d\ze')
|
||
let repeatmatch = matchstr(line, '<\d\d\d\d-\d\d-\d\d.*[ +.]+\d\+\S\+.*>\ze')
|
||
if repeatmatch != ''
|
||
" if date has repeater then call once for each repeat in period
|
||
let repeatlist = s:RepeatMatch(repeatmatch[1:],date1,date2)
|
||
for dateitem in repeatlist
|
||
if a:0 == 1
|
||
call s:ProcessDateMatch(dateitem,date1,date2,a:1)
|
||
else
|
||
call s:ProcessDateMatch(dateitem,date1,date2)
|
||
endif
|
||
endfor
|
||
else
|
||
if (datematch[0]!='[') || g:org_clocks_in_agenda
|
||
if a:0 == 1
|
||
call s:ProcessDateMatch(datematch[1:],date1,date2,a:1)
|
||
else
|
||
call s:ProcessDateMatch(datematch[1:],date1,date2)
|
||
endif
|
||
endif
|
||
endif
|
||
"endif
|
||
endwhile
|
||
call setpos(".",save_cursor)
|
||
endfunction
|
||
|
||
function! s:ProcessDateMatch(datematch,date1,date2,...)
|
||
if a:0 > 0
|
||
let today = a:1
|
||
else
|
||
let today = strftime("%Y-%m-%d")
|
||
endif
|
||
let datematch = a:datematch
|
||
let rangedate = matchstr(getline(line(".")),'--<\zs\d\d\d\d-\d\d-\d\d')
|
||
|
||
let locator = s:PrePad(s:filenum,3,'0') . s:PrePad(line('.'),5,'0') . ' '
|
||
|
||
let g:myline = s:OrgGetHead_l(line('.'))
|
||
"let g:myline = s:OrgParentHead_l(line('.'))
|
||
"if g:myline == 0
|
||
" let filename = org#Pad(b:v.org_dict[g:myline].CATEGORY,13)
|
||
"else
|
||
"let filename = org#Pad(b:v.org_dict.iprop(g:myline,'CATEGORY'),12) . ' '
|
||
let filename = printf("%-12.12s",b:v.org_dict.iprop(g:myline,'CATEGORY')) . ' '
|
||
"endif
|
||
let line = getline(line("."))
|
||
let date1 = a:date1
|
||
let date2 = a:date2
|
||
let s:headline=0
|
||
if (datematch >= date1) && (datematch < date2)
|
||
\ && ((g:org_search_spec ==# '') || (s:CheckIfExpr(line("."),b:v.agenda_ifexpr)))
|
||
let mlist = matchlist(line,'\(DEADLINE\|SCHEDULED\|CLOSED\)')
|
||
call s:SetHeadInfo()
|
||
if empty(mlist)
|
||
" it's a regular date, first check for time parts
|
||
let tmatch = matchstr(line,' \zs\d\d:\d\d\ze.*[[>]')
|
||
if tmatch ># ''
|
||
let tmatch2 = matchstr(line,'<.\{-}-\zs\d\d:\d\d\ze.*>')
|
||
if tmatch2 ># ''
|
||
let tmatch .= '-' . tmatch2
|
||
else
|
||
if match(line,':\s*CLOCK\s*:') >= 0
|
||
let tmatch .= '-'.matchstr(line,'--\[.\{-}\zs\d\d:\d\d\ze\]')
|
||
let s:headtext = s:headtext[0:6] . 'Clocked: ('.matchstr(line,'->\s*\zs.*$') .') '.s:headtext[7:]
|
||
else
|
||
let tmatch .= '......'
|
||
endif
|
||
endif
|
||
endif
|
||
call add(g:agenda_date_dict[datematch].l, locator . filename . org#Pad(tmatch,11) . s:headtext)
|
||
"call add(g:agenda_date_dict[datematch].l, line(".") . repeat(' ',6-len(line("."))) . filename . org#Pad(tmatch,11) . s:headtext)
|
||
if rangedate != ''
|
||
"let startdate = matchstr(line,'<\zs\d\d\d\d-\d\d-\d\d\ze')
|
||
"let thisday = calutil#jul(datematch) - calutil#jul(startdate) + 1
|
||
let days_in_range = calutil#jul(rangedate) - calutil#jul(datematch) + 1
|
||
"let rangestr = '('.thisday.'/'.days_in_range.')'
|
||
let i = days_in_range
|
||
"while (rangedate < date2) && (rangedate > datematch)
|
||
while (rangedate > datematch)
|
||
let rangestr = '('.i.'/'.days_in_range.')'
|
||
if exists("g:agenda_date_dict['".rangedate."']")
|
||
"call add(g:agenda_date_dict[rangedate].l, line(".") . repeat(' ',6-len(line("."))) .
|
||
call add(g:agenda_date_dict[rangedate].l, locator .
|
||
\ filename . org#Pad(rangestr,11) . s:headtext)
|
||
endif
|
||
let rangedate = calutil#cal(calutil#jul(rangedate) - 1)
|
||
let i = i - 1
|
||
endwhile
|
||
" to end of line to avoid double
|
||
" treatment
|
||
normal $
|
||
endif
|
||
else
|
||
" it's a deadline/scheduled/closed date
|
||
let type = org#Pad(mlist[1][0] . tolower(mlist[1][1:]) . ':' , 11)
|
||
call add(g:agenda_date_dict[datematch].l, locator . filename . type . s:headtext)
|
||
endif
|
||
endif
|
||
" Now test for late and upcoming warnings if 'today' is in range and not
|
||
" a done-type todo
|
||
if (today >= date1) && (today < date2) && (getline(g:myline) !~ b:v.todoDoneMatch)
|
||
if (datematch < today) && (match(line,'\(DEADLINE\|SCHEDULED\)')>-1)
|
||
\ && ((g:org_search_spec ==# '') || (s:CheckIfExpr(line("."),b:v.agenda_ifexpr)))
|
||
let mlist = matchlist(line,'\(DEADLINE\|SCHEDULED\)')
|
||
call s:SetHeadInfo()
|
||
if !empty(mlist)
|
||
let dayspast = calutil#jul(today) - calutil#jul(datematch)
|
||
if mlist[1] ==? 'DEADLINE'
|
||
let newpart = org#Pad('In',6-len(dayspast)) . '-' . dayspast . ' d.:'
|
||
else
|
||
let newpart = org#Pad('Sched:',9-len(dayspast)) . dayspast . 'X:'
|
||
endif
|
||
call add(g:agenda_date_dict[today].l, locator . filename . newpart . s:headtext)
|
||
endif
|
||
" also put in warning entry for deadlines when appropriate
|
||
elseif (datematch > today) && (match(line,'DEADLINE')>-1)
|
||
\ && ((g:org_search_spec ==# '') || (s:CheckIfExpr(line("."),b:v.agenda_ifexpr)))
|
||
let mlist = matchlist(line,'DEADLINE')
|
||
call s:SetHeadInfo()
|
||
if !empty(mlist)
|
||
let daysahead = calutil#jul(datematch) - calutil#jul(today)
|
||
let g:specific_warning = str2nr(matchstr(line,'<\S*\d\d.*-\zs\d\+\zed.*>'))
|
||
if (daysahead <= g:org_deadline_warning_days) || (daysahead <= g:specific_warning)
|
||
let newpart = org#Pad('In',7-len(daysahead)) . daysahead . ' d.:'
|
||
call add(g:agenda_date_dict[today].l, locator . filename . newpart . s:headtext)
|
||
endif
|
||
endif
|
||
endif
|
||
endif
|
||
" finally handle things for a range that began before date1
|
||
if (rangedate != '') && (datematch < date1)
|
||
\ && ((g:org_search_spec ==# '') || (s:CheckIfExpr(line("."),b:v.agenda_ifexpr)))
|
||
let days_in_range = calutil#jul(rangedate) - calutil#jul(datematch) + 1
|
||
if rangedate >= date2
|
||
let last_day_to_add = calutil#jul(date2) - calutil#jul(datematch)
|
||
let rangedate = calutil#cal(calutil#jul(date2)-1)
|
||
else
|
||
let last_day_to_add = days_in_range
|
||
endif
|
||
|
||
call s:SetHeadInfo()
|
||
let i = last_day_to_add
|
||
while (rangedate >= date1)
|
||
let rangestr = '('.i.'/'.days_in_range.')'
|
||
call add(g:agenda_date_dict[rangedate].l, locator .
|
||
\ filename . org#Pad(rangestr,11) . s:headtext)
|
||
let rangedate = calutil#cal(calutil#jul(rangedate) - 1)
|
||
let i = i - 1
|
||
endwhile
|
||
" go past match to avoid double treatment
|
||
normal $
|
||
endif
|
||
if s:headline > 0
|
||
let g:agenda_head_lookup[line(".")]=s:headline
|
||
endif
|
||
endfunction
|
||
|
||
function! s:SetHeadInfo()
|
||
let s:headline = s:OrgGetHead_l(line("."))
|
||
let s:headtext = getline(s:headline)
|
||
let s:mystars = matchstr(s:headtext,'^\*\+')
|
||
let s:headstars = s:PrePad(s:mystars,6)
|
||
let s:headtext = s:headstars . ' ' . s:headtext[len(s:mystars)+1:]
|
||
endfunction
|
||
|
||
function! s:RepeatMatch(rptdate, date1, date2)
|
||
let yearflag = 0
|
||
let basedate = matchstr(a:rptdate,'\d\d\d\d-\d\d-\d\d')
|
||
if basedate >= a:date2
|
||
" no need for repeat, rturn to check fo deadlien warnings
|
||
return [basedate]
|
||
endif
|
||
let date1 = a:date1
|
||
if basedate > date1
|
||
let date1 = basedate
|
||
endif
|
||
let baserpt = matchstr(a:rptdate, ' \S\S\S [.+]\{0,1}+\zs\S\+\ze.*>')
|
||
let rptnum = matchstr(baserpt, '^\d\+')
|
||
let rpttype = matchstr(baserpt, '^\d\+\zs.')
|
||
let g:rptlist = []
|
||
let date1jul = calutil#jul(date1)
|
||
let date2jul = calutil#jul(a:date2)
|
||
if rpttype ==? 'w'
|
||
let rpttype = 'd'
|
||
let rptnum = str2nr(rptnum)*7
|
||
endif
|
||
if rpttype ==? 'y'
|
||
let rpttype = 'm'
|
||
let rptnum = str2nr(rptnum)*12
|
||
let yearflag = 1
|
||
endif
|
||
if rpttype ==? 'd'
|
||
let dmod = (date1jul - calutil#jul(basedate)) % rptnum
|
||
let i = 0
|
||
while 1
|
||
let testjul = date1jul - dmod + (i*rptnum)
|
||
if testjul < date2jul
|
||
call add(g:rptlist, calutil#cal(testjul))
|
||
else
|
||
break
|
||
endif
|
||
let i += 1
|
||
endwhile
|
||
elseif rpttype ==? 'm'
|
||
let g:special = baserpt[-1:]
|
||
let monthday = str2nr(basedate[8:])
|
||
let baseclone = basedate
|
||
" this if-structure assigns begin test month as
|
||
" first repeat month _before_ date1
|
||
if yearflag
|
||
if (date1[:6]) >= (date1[:3] . baseclone[4:6])
|
||
let baseclone = date1[:3] . baseclone[4:]
|
||
else
|
||
let baseclone = string(str2nr(date1[:3]) - 1) . baseclone[4:]
|
||
endif
|
||
let first_of_month_jul = calutil#jul(baseclone[:7]. '01')
|
||
else
|
||
let first_of_month_jul = calutil#jul(date1[:4] .
|
||
\ s:Pre0( date1[5:6] ) . '-01')
|
||
"\ s:Pre0( date1[5:6] - 1) . '-01')
|
||
endif
|
||
|
||
if g:special ==? '*'
|
||
let specialnum = (monthday / 7) + 1
|
||
let specialdaynum = calutil#dow(basedate)
|
||
endif
|
||
while 1
|
||
if g:special != '*'
|
||
let testjul = first_of_month_jul - 1 + monthday
|
||
else
|
||
" process for 'xth weekday of month' type
|
||
let fdow = calutil#dow(calutil#cal(first_of_month_jul))
|
||
if fdow == specialdaynum
|
||
let testjul = first_of_month_jul + (specialnum-1)*7
|
||
elseif fdow < specialdaynum
|
||
let testjul = first_of_month_jul + (specialnum-1)*7
|
||
\ + (specialdaynum - fdow)
|
||
elseif fdow > specialdaynum
|
||
let testjul = first_of_month_jul + (specialnum*7)
|
||
\ - (fdow - specialdaynum)
|
||
endif
|
||
endif
|
||
|
||
if (testjul < date2jul) && (testjul >= first_of_month_jul)
|
||
call add(g:rptlist, calutil#cal(testjul))
|
||
else
|
||
"put in this one to check for deadline warnings
|
||
"if len(g:rptlist)>0
|
||
call add(g:rptlist, calutil#cal(testjul))
|
||
"endif
|
||
break
|
||
endif
|
||
let first_cal = calutil#cal(first_of_month_jul)
|
||
let nextmonth = str2nr(first_cal[5:6]) + rptnum
|
||
let year = str2nr(first_cal[0:3])
|
||
if nextmonth >= 13
|
||
let nextmonth = (nextmonth-12)
|
||
let year += 1
|
||
endif
|
||
let first_of_month_jul = calutil#jul(string(year) . '-' . s:Pre0(nextmonth) . '-01')
|
||
endwhile
|
||
endif
|
||
|
||
return g:rptlist
|
||
|
||
endfunction
|
||
|
||
function! s:BufMinMaxDate()
|
||
let b:v.MinMaxDate=['2099-12-31','1900-01-01']
|
||
g/<\d\d\d\d-\d\d-\d\d/call s:CheckMinMax()
|
||
|
||
endfunction
|
||
function! s:CheckMinMax()
|
||
let date = matchstr(getline(line(".")),'<\zs\d\d\d\d-\d\d-\d\d')
|
||
if (date < b:v.MinMaxDate[0])
|
||
let b:v.MinMaxDate[0] = date
|
||
endif
|
||
if (date > b:v.MinMaxDate[1])
|
||
let b:v.MinMaxDate[1] = date
|
||
endif
|
||
endfunction
|
||
function! s:Timeline(...)
|
||
if a:0 > 0
|
||
let spec = a:1
|
||
else
|
||
let spec = ''
|
||
endif
|
||
if bufname("%") ==? '__Agenda__'
|
||
"go back up to main org buffer
|
||
"wincmd k
|
||
endif
|
||
if exists('g:org_search_spec')
|
||
let prev_spec = g:org_search_spec
|
||
endif
|
||
if exists('g:agenda_files')
|
||
let prev_files = g:agenda_files
|
||
endif
|
||
exec "let g:agenda_files=['".substitute(expand("%:p"),' ','\\ ','g')."']"
|
||
call s:BufMinMaxDate()
|
||
let num_days = 1 + calutil#jul(b:v.MinMaxDate[1]) - calutil#jul(b:v.MinMaxDate[0])
|
||
try
|
||
"AAgenda
|
||
call OrgRunAgenda(b:v.MinMaxDate[0], num_days,spec)
|
||
finally
|
||
if exists('prev_spec')
|
||
let g:org_search_spec = prev_spec
|
||
endif
|
||
if exists('prev_files')
|
||
let g:agenda_files = prev_files
|
||
endif
|
||
endtry
|
||
endfunction
|
||
|
||
function! s:Pre0(s)
|
||
return repeat('0',2 - len(a:s)) . a:s
|
||
endfunction
|
||
|
||
function! s:PrePad(s,amt,...)
|
||
if a:0 > 0
|
||
let char = a:1
|
||
else
|
||
let char = ' '
|
||
endif
|
||
return repeat(char,a:amt - len(a:s)) . a:s
|
||
endfunction
|
||
function! s:AgendaCompare(i0, i1)
|
||
let mymstr = '^\(\d\+\)\s\+\(\S\+\)\s\+\(\%24c.\{11}\).*\(\*\+\)\s\(.*$\)'
|
||
" mymstr below would be better match string regex, but generic dates
|
||
" have no text at position 24 to match \S . . . "
|
||
"let mymstr = '^\(\d\+\)\s\+\(\S\+\)\s\+\(\S.\{10}\).*\(\*\+\)\s\(.*$\)'
|
||
" [1] is lineno, [2] is file, [3] is scheduling, [4] is levelstarts,
|
||
" [5] is headtext
|
||
let cp0 = matchlist(a:i0,mymstr)
|
||
let cp1 = matchlist(a:i1,mymstr)
|
||
let myitems = [cp0, cp1]
|
||
let sched_comp = []
|
||
let i = 0
|
||
while i < 2
|
||
let item = myitems[i]
|
||
if item[3][0] ==? 'S'
|
||
if item[3][5] ==? ':'
|
||
"let str_ord = 'a' . substitute(item[3][6:8],' ', '0','')
|
||
let str_ord = 'aa' . s:PrePad(1000-str2nr(item[3][6:8]),' ', '0')
|
||
else
|
||
let str_ord = 'ab000'
|
||
endif
|
||
elseif item[3][0] ==? 'I'
|
||
if matchstr(item[3],'-') ># ''
|
||
let str_ord = 'd-'.s:PrePad(1000-str2nr(matchstr(item[3],'\d\+')),3,'0')
|
||
else
|
||
let str_ord = 'da'.s:PrePad(matchstr(item[3],'\d\+'),3,'0')
|
||
endif
|
||
elseif item[3][0] ==? 'D'
|
||
let str_ord = 'd0000'
|
||
elseif item[3][0] ==? ' '
|
||
let str_ord = 'zzzzz'
|
||
else
|
||
let str_ord = item[3][:4]
|
||
endif
|
||
call add(sched_comp,str_ord.item[2].s:PrePad(item[1],5,'0'))
|
||
let i += 1
|
||
endwhile
|
||
|
||
return sched_comp[0] ==? sched_comp[1] ? 0 : sched_comp[0] > sched_comp[1] ? 1 : -1
|
||
|
||
" let num1 = str2nr(matchstr(a:i1,'In *\zs[ -]\d\+\ze d.:'))
|
||
" let num2 = str2nr(matchstr(a:i2,'In *\zs[ -]\d\+\ze d.:'))
|
||
" if num1 == 0
|
||
" let num1 = str2nr(matchstr(a:i1,'Sched: *\zs\d\+\zeX:'))
|
||
" if num1 !=0
|
||
" let num1 = -num1 - 10000
|
||
" endif
|
||
" endif
|
||
" if num2 == 0
|
||
" let num2 = str2nr(matchstr(a:i2,'Sched: *\zs\d\+\zeX:'))
|
||
" if num2 !=0
|
||
" let num2 = -num2 - 10000
|
||
" endif
|
||
" endif
|
||
" if (a:i1 =~ '^\d\+\s\+\S\+\s\+\d')
|
||
" let num1=num1-20000
|
||
" endif
|
||
" if (a:i2 =~ '^\d\+\s\+\S\+\s\+\d')
|
||
" let num2=num2-20000
|
||
" endif
|
||
|
||
" return num1 == num2 ? 0 : num1 > num2 ? 1 : -1
|
||
|
||
endfunc
|
||
|
||
function! s:DateListAdd(valdict)
|
||
let namelist = [' GENERAL','SCHEDULED','CLOSED','DEADLINE']
|
||
let templist = []
|
||
call add(templist, get(a:valdict,'ud',0))
|
||
call add(templist, get(a:valdict,'sd',0))
|
||
call add(templist, get(a:valdict,'cd',0))
|
||
call add(templist, get(a:valdict,'dd',0))
|
||
let i = 0
|
||
while i < 4
|
||
if templist[i] != 0
|
||
call add(g:org_datelist, templist[i] . ' ' . namelist[i] . ' ' . a:valdict.l )
|
||
endif
|
||
let i += 1
|
||
endwhile
|
||
return a:valdict
|
||
endfunction
|
||
|
||
function! OrgAgendaMove(direction,count)
|
||
" need to add functionaity for count, refactor
|
||
let cnt = (a:count == 0) ? 1 : a:count
|
||
let cnt = (a:direction ==? 'forward') ? cnt : -cnt
|
||
let date_jul = calutil#jul(g:agenda_startdate)
|
||
let ymd_list = split(g:agenda_startdate,'-')
|
||
|
||
if g:org_agenda_days == 1
|
||
let g:agenda_startdate = calutil#cal(date_jul + (cnt * 1))
|
||
elseif g:org_agenda_days == 7
|
||
let g:agenda_startdate = calutil#cal(date_jul + (cnt * 7))
|
||
elseif g:org_agenda_days >= 360
|
||
let g:agenda_startdate = string(ymd_list[0] + (cnt * 1)) .'-01-01'
|
||
else
|
||
let ymd_list[1] = ymd_list[1] + (cnt * 1)
|
||
if ymd_list[1] > 0
|
||
let ymd_list[0] += ymd_list[1] / 12
|
||
let ymd_list[1] = ymd_list[1] % 12
|
||
else
|
||
let ymd_list[0] += (ymd_list[1] / 12) - 1
|
||
let ymd_list[1] = 12 + (ymd_list[1] % 12)
|
||
endif
|
||
let g:agenda_startdate = ymd_list[0] . '-' .
|
||
\ s:Pre0(string(ymd_list[1])) .'-01'
|
||
let g:org_agenda_days = s:DaysInMonth(g:agenda_startdate)
|
||
endif
|
||
|
||
if g:org_agenda_days == 1
|
||
"call OrgRunAgenda(g:agenda_startdate,g:org_agenda_days,g:org_search_spec,g:agenda_startdate)
|
||
call OrgRunCustom({'redo_num': line('.'), 'type':'agenda', 'agenda_date': g:agenda_startdate, 'agenda_duration': g:org_agenda_days, 'spec': g:org_search_spec})
|
||
else
|
||
"call OrgRunAgenda(g:agenda_startdate,g:org_agenda_days,g:org_search_spec)
|
||
call s:RunCustom({'redo_num': line('.'), 'type':'agenda', 'agenda_date': g:agenda_startdate, 'agenda_duration': g:org_agenda_days ,'spec': g:org_search_spec})
|
||
endif
|
||
endfunction
|
||
|
||
function! s:TimeGrid(starthour,endhour,inc)
|
||
let result = []
|
||
for i in range(a:starthour, a:endhour,a:inc)
|
||
call add(result,repeat(' ',23).s:Pre0(i).':00...... ------------')
|
||
endfor
|
||
return result
|
||
endfunction
|
||
|
||
function! s:MakeCalendar(date, daycount)
|
||
"function! s:MakeCalendar(year, month, day, daycount)
|
||
" this function is taken from vim tip by Siegfried Bublitz
|
||
" at: http://vim.wikia.com/wiki/Generate_calendar_file
|
||
" with many mods to 1.output to list rather than to buffer
|
||
" and 2. get weekday and weekno from calutils
|
||
let g:agenda_date_dict = {}
|
||
let g:agenda_head_lookup = {}
|
||
"let startdate = calutil#Jul2Cal((calutil#Cal2Jul(a:year,a:month,a:day) - calutil#DayOfWeek(a:year,a:month,a:day)))
|
||
let year = split(a:date,'-')[0]
|
||
let month = split(a:date,'-')[1]
|
||
let day = split(a:date,'-')[2]
|
||
let day = str2nr(day)
|
||
|
||
if a:daycount == 7
|
||
let wd = 1
|
||
elseif (a:daycount>=28) && (a:daycount <=31)
|
||
let wd = calutil#dow(a:date[0:7].'01') + 1
|
||
else
|
||
let wd = calutil#dow(a:date) + 1
|
||
endif
|
||
let week = 1 + (calutil#Cal2Jul(year,month,day) - calutil#Cal2Jul(year,1,1)) / 7
|
||
let index = 0
|
||
let datetext = ''
|
||
let diy = 777 " day in year, wrong before next year
|
||
while (index < a:daycount) " no of days to output
|
||
let diy = diy + 1
|
||
if (wd > 7)
|
||
let wd = 1
|
||
let week = week + 1
|
||
if (week >= 53)
|
||
if (week >= 54)
|
||
let week = 1
|
||
elseif (day >= 28 || day <= 3)
|
||
let week = 1
|
||
endif
|
||
endif
|
||
endif
|
||
let monthnames=['January','February','March','April','May','June','July',
|
||
\ 'August','September','October','November','December']
|
||
"let daynames = ['Mon','Tue','Wed','Thu','Fri','Sat','Sun']
|
||
let daynames = ['Monday','Tuesday','Wednesday','Thursday','Friday','Saturday','Sunday']
|
||
let dn = daynames[wd-1]
|
||
if ((day > 31) || (month == 2 && (day > 29 || day > 28 && year % 4))
|
||
\ || (month == 4 && day > 30) || (month == 6 && day > 30)
|
||
\ || (month == 9 && day > 30) || (month == 11 && day > 30))
|
||
let day = 1
|
||
let month = month + 1
|
||
if (month > 12)
|
||
let month = 1
|
||
let diy = 1
|
||
let year = year + 1
|
||
if (wd <= 3)
|
||
let week = 1
|
||
endif
|
||
endif
|
||
endif
|
||
|
||
let datetext = dn . repeat(' ',10-len(dn)) . (day<10?' ':' ') .
|
||
\ day . ' ' . monthnames[month-1] . ' ' . year . (wd == 1 ? ' Wk' . week : '' )
|
||
|
||
let g:agenda_date_dict[year . '-' . s:Pre0(month) . '-' . (day<10 ? '0'.day : day) ]
|
||
\ = {'marker': datetext, 'l': [] }
|
||
let index = index + 1
|
||
let day = day + 1
|
||
let wd = wd + 1
|
||
endwhile
|
||
|
||
endfunction
|
||
|
||
function! s:ActualBufferLine(lineref_in_agenda,bufnumber)
|
||
let actual_line = matchstr(s:GetPlacedSignsString(a:bufnumber),'line=\zs\d\+\ze\s\+id='.a:lineref_in_agenda)
|
||
return actual_line
|
||
endfunction
|
||
|
||
function! s:AgendaPutText(...)
|
||
let save_cursor = getpos(".")
|
||
let thisline = getline(line("."))
|
||
if thisline =~ '^\d\+\s\+'
|
||
if (getline(line(".") + 1) =~ '^\*\+ ')
|
||
"let file = matchstr(thisline,'^\d\+\s\+\zs\S\+\ze')
|
||
"let file = s:filedict[str2nr(matchstr(thisline, '^\d\d\d'))]
|
||
let file = s:agenda_files_copy[str2nr(matchstr(thisline, '^\d\d\d'))]
|
||
"let lineno = matchstr(thisline,'^\d\+\ze\s\+')
|
||
let lineno = str2nr(matchstr(thisline,'^\d\d\d\zs\d*'))
|
||
let starttab = tabpagenr()
|
||
|
||
"call org#LocateFile(file.'.org')
|
||
call org#LocateFile(file)
|
||
if g:agenda_date_dict != {}
|
||
"let confirmhead = g:agenda_head_lookup[lineno]
|
||
let confirmhead = lineno
|
||
elseif g:adict != {}
|
||
let confirmhead = lineno
|
||
endif
|
||
let newhead = matchstr(s:GetPlacedSignsString(bufnr("%")),'line=\zs\d\+\ze\s\+id='.confirmhead)
|
||
let newhead = s:OrgGetHead_l(newhead)
|
||
execute newhead
|
||
let lastline = s:OrgNextHead_l(newhead) - 1
|
||
if lastline > newhead
|
||
let g:text = getline(newhead,lastline)
|
||
elseif lastline == -1
|
||
let g:text = getline(newhead,line('$'))
|
||
else
|
||
let g:text = []
|
||
endif
|
||
|
||
execute 'tabnext ' . starttab
|
||
execute bufwinnr('Agenda').'wincmd w'
|
||
|
||
call setpos(".",save_cursor)
|
||
" okay, we're back in agenda and have main buffer's
|
||
" text in g:text, now need to compare it
|
||
|
||
normal j
|
||
let firstline = line(".")
|
||
let daytextpat = '^\S\+\s\+\d\{1,2}\s\S\+\s\d\d\d\d'
|
||
while (getline(line(".")) !~ '^\d\+\s\+') && (line(".") != line("$"))
|
||
\ && (getline(line(".")) !~ daytextpat)
|
||
\ && (getline(line(".")) !~ '\d empty days omitted')
|
||
normal j
|
||
endwhile
|
||
let lastline = line(".")
|
||
if (lastline < line("$")) ||
|
||
\ ( (getline(line(".")) =~ '^\d\+\s\+')
|
||
\ || (getline(line(".")) =~ daytextpat)
|
||
\ || (getline(line(".")) =~ '\d empty days omitted') )
|
||
let lastline = line(".") - 1
|
||
endif
|
||
"execute firstline . ', ' . lastline . 'd'
|
||
if g:text ==? getline(firstline, lastline)
|
||
echo "headings are identical"
|
||
else
|
||
if g:adict != {}
|
||
let resp = confirm("Heading's text has changed, save changes?","&Save\n&Cancel",1)
|
||
if resp == 1
|
||
call s:SaveHeadline(file, newhead,getline(firstline,lastline))
|
||
"call s:SaveHeadline(file, confirmhead,getline(firstline,lastline))
|
||
else
|
||
echo "Changes were _not_ saved."
|
||
endif
|
||
else
|
||
call confirm("Heading's text has changed, but saving is\n"
|
||
\ . "temporarily disabled for date-based agenda views.\n"
|
||
\ . "No changes are being saved, original buffer text remains as it was.")
|
||
endif
|
||
endif
|
||
endif
|
||
else
|
||
echo "You're not on a headline line."
|
||
endif
|
||
call setpos(".",save_cursor)
|
||
endfunction
|
||
function! s:SaveHeadline(file, headline, lines)
|
||
let file = a:file
|
||
let headline = a:headline
|
||
let lines=a:lines
|
||
let starttab = tabpagenr()
|
||
|
||
"call org#LocateFile(file.'.org')
|
||
call org#LocateFile(file)
|
||
"let newhead = matchstr(s:GetPlacedSignsString(bufnr("%")),'line=\zs\d\+\ze\s\+id='.headline)
|
||
let newhead = a:headline
|
||
execute newhead
|
||
|
||
let lastline = s:OrgNextHead_l(newhead) - 1
|
||
execute newhead+1.','.lastline.'d'
|
||
" don't delete orig headline b/c that's where sign is placed
|
||
call setline(newhead,lines[0])
|
||
call append(newhead,lines[1:])
|
||
|
||
execute 'tabnext ' . starttab
|
||
execute bufwinnr('Agenda').'wincmd w'
|
||
|
||
endfunction
|
||
function! OrgAgendaGetText(...)
|
||
"type: 'datedict' for date agenda, 'adict' for regular search
|
||
let cycle_todo = 0
|
||
if a:0 >= 1
|
||
let cycle_todo = 1
|
||
if a:0 == 2
|
||
let newtodo = a:2
|
||
endif
|
||
endif
|
||
" called by <TAB> map to toggle view of heading's body text in agenda
|
||
" view
|
||
let save_cursor = getpos(".")
|
||
let thisline = getline(line("."))
|
||
let curTodo = matchstr(thisline, '\*\+ \zs\S\+')
|
||
if thisline =~ '^\d\+\s\+'
|
||
if (getline(line(".") + 1) =~ '^\d\+\s\+') || (line(".") == line("$")) ||
|
||
\ (getline(line(".") + 1 ) =~ '^\S\+\s\+\d\{1,2}\s\S\+\s\d\d\d\d')
|
||
\ || (getline(line(".") + 1 ) =~ '\d empty days omitted')
|
||
|
||
let starttab = tabpagenr()
|
||
|
||
" go to the heading's file
|
||
let file = s:agenda_files_copy[str2nr(matchstr(thisline, '^\d\d\d'))]
|
||
let lineno = str2nr(matchstr(thisline,'^\d\d\d\zs\d*'))
|
||
call org#LocateFile(file)
|
||
|
||
let save_cursor2 = getpos(".")
|
||
|
||
let newhead = matchstr(s:GetPlacedSignsString(bufnr("%")),'line=\zs\d\+\ze\s\+id=' . lineno)
|
||
let newhead = s:OrgGetHead_l(newhead)
|
||
|
||
execute newhead
|
||
|
||
if cycle_todo
|
||
" do todo operation
|
||
if a:0 >= 2
|
||
silent call s:ReplaceTodo(newtodo)
|
||
else
|
||
silent call s:ReplaceTodo()
|
||
endif
|
||
"normal V
|
||
"redraw
|
||
"sleep 100m
|
||
"normal V
|
||
else
|
||
" get the heading's text block
|
||
let lastline = s:OrgNextHead_l(newhead) - 1
|
||
if lastline > newhead
|
||
let g:text = getline(newhead,lastline)
|
||
elseif lastline == -1
|
||
let g:text = getline(newhead,line('$'))
|
||
else
|
||
let g:text = []
|
||
endif
|
||
endif
|
||
call setpos(".",save_cursor2)
|
||
" now go back to agenda
|
||
execute 'tabnext ' . starttab
|
||
execute bufwinnr('Agenda').'wincmd w'
|
||
if !cycle_todo
|
||
call append(line("."),g:text)
|
||
endif
|
||
else
|
||
" we're dealing with toggle of agenda heading that already
|
||
" has main buffer heading text in agenda. . .
|
||
normal j
|
||
let firstline = line(".")
|
||
let daytextpat = '^\S\+\s\+\d\{1,2}\s\S\+\s\d\d\d\d'
|
||
while (getline(line(".")) !~ '^\d\+\s\+') && (line(".") != line("$"))
|
||
\ && (getline(line(".")) !~ daytextpat)
|
||
\ && (getline(line(".")) !~ '\d empty days omitted')
|
||
normal j
|
||
endwhile
|
||
let lastline = line(".")
|
||
if (lastline < line("$")) ||
|
||
\ ( (getline(line(".")) =~ '^\d\+\s\+')
|
||
\ || (getline(line(".")) =~ daytextpat)
|
||
\ || (getline(line(".")) =~ '\d empty days omitted'))
|
||
let lastline = line(".") - 1
|
||
endif
|
||
call setpos(".",save_cursor)
|
||
" see if it there are changes that need to be
|
||
" pushed back to main buffer
|
||
call s:AgendaPutText()
|
||
silent execute firstline . ', ' . lastline . 'd'
|
||
endif
|
||
else
|
||
echo "You're not on a headline line."
|
||
endif
|
||
call setpos(".",save_cursor)
|
||
if cycle_todo
|
||
" we did cycle in main buffer, now do in agenda
|
||
if a:0 >= 2
|
||
silent call s:ReplaceTodo(s:last_newtodo)
|
||
else
|
||
silent call s:ReplaceTodo(s:last_newtodo)
|
||
endif
|
||
echo "Todo cycled."
|
||
endif
|
||
endfunction
|
||
|
||
function! s:MoveToHeadingFromAgenda(agenda_line)
|
||
"given line from agenda, go to the associated file and heading
|
||
let thisline = getline(a:agenda_line)
|
||
let file = s:agenda_files_copy[str2nr(matchstr(thisline, '^\d\d\d'))]
|
||
let lineno = str2nr(matchstr(thisline,'^\d\d\d\zs\d*'))
|
||
call org#LocateFile(file)
|
||
" need to check sign at this lineno, since for date agenda we're not necessarily at heading
|
||
let newhead = matchstr(s:GetPlacedSignsString(bufnr("%")),'line=\zs\d\+\ze\s\+id=' . lineno)
|
||
let newhead = s:OrgGetHead_l(newhead)
|
||
execute newhead
|
||
endfunction
|
||
|
||
|
||
function! s:IsVisibleHeading(line)
|
||
" returns 1 if line is of a visible heading,
|
||
" 0 if not
|
||
" a heading is visible if foldclosed = -1
|
||
" (i.e., it's not in a fold)
|
||
" OR if it's not in an earlier-started fold
|
||
" (i.e. start of fold heading is in is
|
||
" same as line of heading)
|
||
" *************************** the second and third lines of if
|
||
" statement are necessary because of bug where foldclosed is less
|
||
" than a head even though it is the fold head ***************
|
||
let fc = foldclosed(a:line)
|
||
if ((a:line > 0) && (fc == -1)) || (fc == a:line)
|
||
\ || ((fc < a:line) && s:IsText(fc) )
|
||
\ || ((fc < a:line) && (foldclosedend(fc) < a:line) )
|
||
" \ || (s:Ind(a:line) == 2)
|
||
return 1
|
||
else
|
||
return 0
|
||
endif
|
||
endfunction
|
||
|
||
function! OrgSingleHeadingText(operation)
|
||
" expand or collapse all visible Body Text
|
||
" under Heading fold that cursor is in
|
||
" operation: "collapse" or "expand"
|
||
" expand or collapse all Body Text
|
||
" currently visible under current heading
|
||
let l:startline = line(".")
|
||
let l:endline = s:OrgSubtreeLastLine_l(l:startline) - 1
|
||
call OrgBodyTextOperation(l:startline,l:endline,a:operation)
|
||
endfunction
|
||
|
||
function! s:StarLevelFromTo(from, to)
|
||
let save_cursor = getpos(".")
|
||
set fdm=manual
|
||
let b:v.levelstars = a:to
|
||
ChangeSyn
|
||
g/^\*\+/call setline(line("."),substitute(getline(line(".")),'^\*\+','*' .
|
||
\ repeat('*',(len(matchstr(getline(line(".")),'^\*\+')) - 1) * a:to / a:from),''))
|
||
set fdm=expr
|
||
call setpos(".",save_cursor)
|
||
endfunction
|
||
|
||
function! s:StarsForLevel(level)
|
||
return 1 + (a:level - 1) * b:v.levelstars
|
||
endfunction
|
||
|
||
function! s:OrgExpandLevelText(startlevel, endlevel)
|
||
" expand regular text for headings by level
|
||
let save_cursor = getpos(".")
|
||
|
||
normal gg
|
||
let startlevel = s:StarsForLevel(a:startlevel)
|
||
let endlevel = s:StarsForLevel(a:endlevel)
|
||
let l:mypattern = substitute(b:v.headMatchLevel,'level', startlevel . ',' . endlevel, "")
|
||
while search(l:mypattern, 'cW') > 0
|
||
execute line(".") + 1
|
||
while getline(line(".")) =~ b:v.drawerMatch
|
||
execute line(".") + 1
|
||
normal! j
|
||
endwhile
|
||
if s:IsText(line("."))
|
||
normal zv
|
||
endif
|
||
"normal l
|
||
endwhile
|
||
|
||
call setpos('.',save_cursor)
|
||
|
||
endfunction
|
||
|
||
" just an idea using 'global' not used anywhere yet
|
||
" visible is problem, must operate only on visible, doesn't do ths now
|
||
function! s:BodyTextOperation3(startline,endline, operation)
|
||
let l:oldcursor = line(".")
|
||
let nh = 0
|
||
call cursor(a:startline,0)
|
||
g/\*\{4,}/s:DoAllTextFold(line("."))
|
||
call cursor(l:oldcursor,0)
|
||
|
||
endfunction
|
||
|
||
|
||
function! OrgBodyTextOperation(startline,endline, operation)
|
||
" expand or collapse all Body Text from startline to endline
|
||
" operation: "collapse" or "expand"
|
||
" save original line
|
||
let l:oldcursor = line(".")
|
||
let nh = 0
|
||
" go to startline
|
||
call cursor(a:startline,0)
|
||
" travel from start to end operating on any
|
||
while 1
|
||
if getline(line(".")) =~ b:v.headMatch
|
||
if a:operation ==? "collapse"
|
||
call s:DoAllTextFold(line("."))
|
||
elseif a:operation ==? 'expand'
|
||
normal zv
|
||
endif
|
||
"elseif s:IsText(line(".")+1) && foldclosed(line(".")) == line(".")
|
||
"elseif foldclosed(line(".")) == line(".")
|
||
" "echo 'in expand area'
|
||
" if a:operation ==? 'expand'
|
||
" normal zv
|
||
" endif
|
||
endif
|
||
let lastnh = nh
|
||
let nh = s:NextVisibleHead(line("."))
|
||
"echo 'last ' . lastnh . ' now ' . nh
|
||
if (nh == 0) || (nh >= a:endline) || (nh == lastnh)
|
||
"echo "hit break"
|
||
break
|
||
elseif lastnh == nh
|
||
break
|
||
echo "bad exit from BodyTextOp"
|
||
else
|
||
"echo "hit ex"
|
||
execute nh
|
||
endif
|
||
|
||
endwhile
|
||
" now go back to original line position in buffer
|
||
call cursor(l:oldcursor,0)
|
||
endfunction
|
||
|
||
let g:calendar_sign = 'OrgCalSign'
|
||
function! OrgCalSign(day, month, year)
|
||
if a:year .'-'.s:Pre0(a:month).'-'.s:Pre0(a:day) ==? s:org_cal_date
|
||
return 1
|
||
else
|
||
return 0
|
||
endif
|
||
endfunction
|
||
function! OrgSetLine(line, file, newtext)
|
||
let save_cursor = getpos(".")
|
||
let curfile = expand("%:t")
|
||
|
||
call org#LocateFile(a:file)
|
||
call setline(a:line,a:newtext)
|
||
|
||
call org#LocateFile(curfile)
|
||
call setpos('.',save_cursor)
|
||
endfunction
|
||
function! OrgGetLine(line, file)
|
||
let save_cursor = getpos(".")
|
||
let curfile = expand("%:t")
|
||
|
||
call org#LocateFile(a:file)
|
||
let result = getline(a:line)
|
||
|
||
call org#LocateFile(curfile)
|
||
call setpos('.',save_cursor)
|
||
return result
|
||
endfunction
|
||
function! OrgAgendaDateType()
|
||
" return type of date line in Agenda
|
||
let text = getline(line('.'))[19:29]
|
||
if text =~ 'Sched'
|
||
let result = 'Scheduled'
|
||
elseif text =~ '\(In \|DEADLINE\)'
|
||
let result = 'Deadline'
|
||
elseif text =~ 'Closed'
|
||
let result = 'Closed'
|
||
elseif text =~ '('
|
||
let result = 'Range'
|
||
else
|
||
let result = 'Regular'
|
||
endif
|
||
return result
|
||
endfunction
|
||
|
||
function! GetDateAtCursor()
|
||
|
||
return matchstr( GetDateSpecAtCursor() , '^[[<]\zs\d\d\d\d-\d\d-\d\d' )
|
||
|
||
endfunction
|
||
|
||
function! GetDateSpecAtCursor()
|
||
let savecursor = getpos(".")
|
||
" save visual bell settings so no bell
|
||
" when not found"
|
||
let orig_vb = &vb
|
||
let orig_t_vb = &t_vb
|
||
set vb t_vb=
|
||
|
||
" check for date string within brackets
|
||
normal! va<
|
||
silent! normal! "xy
|
||
call setpos('.',savecursor)
|
||
if len(@x) < 7
|
||
"normal! vi["xy
|
||
normal! va[
|
||
silent! normal! "xy
|
||
endif
|
||
|
||
if (len(@x)>=15) && (len(@x)<41)
|
||
let date = matchstr(@x,'^[[<]\d\d\d\d-\d\d-\d\d')
|
||
else
|
||
let date = ''
|
||
endif
|
||
|
||
" restore visual bell settings
|
||
let &vb = orig_vb
|
||
let &t_vb = orig_t_vb
|
||
|
||
if date ># ''
|
||
" return with only opening '<' or '['
|
||
return @x
|
||
else
|
||
return ''
|
||
endif
|
||
|
||
call setpos(".", savecursor)
|
||
|
||
endfunction
|
||
|
||
function! CalEdit( sdate, stime )
|
||
" bring up calendar to edit and return a date value
|
||
call org#SaveLocation()
|
||
let basedate = a:sdate ==# '' ? s:Today() : a:sdate
|
||
let basetime = a:stime
|
||
let newdate = '<' . basedate[0:9] . ' ' . calutil#dayname(basedate[0:9]) . (basetime ># '' ? ' ' . basetime : '') . '>'
|
||
let newtime = basetime
|
||
|
||
hi Cursor guibg=black
|
||
let s:org_cal_date = newdate[1:10]
|
||
call s:Calfunc(1,newdate[1:4],str2nr(newdate[6:7]))
|
||
" highlight chosen dates in calendar
|
||
hi Ag_Date guifg=red
|
||
call matchadd('Ag_Date','+\s\{0,1}\d\+')
|
||
redraw
|
||
let g:calendar_action='<SNR>'.s:SID().'_CalendarInsertDate'
|
||
let cue = ''
|
||
while 1
|
||
echohl LineNr | echon 'Date+time ['.basedate . ' '.basetime.']: '
|
||
"echohl None | echon cue.'_ =>' | echohl WildMenu | echon ' '.newdate[:-2] . ' ' . newtime
|
||
echohl None | echon cue.'_ =>' | echohl WildMenu | echon ' '.newdate[:-2] . '>'
|
||
let nchar = getchar()
|
||
let newchar = nr2char(nchar)
|
||
if newdate !~ 'interpret'
|
||
let curdif = calutil#jul(newdate[1:10])-calutil#jul(s:Today())
|
||
endif
|
||
if (nchar ==? "\<BS>") && (len(cue)>0)
|
||
let cue = cue[:-2]
|
||
elseif nchar ==? "\<s-right>"
|
||
let cue = ((curdif+1>=0) ?'+':'').(curdif+1).'d'
|
||
elseif nchar ==? "\<s-left>"
|
||
let cue = ((curdif-1>=0) ?'+':'').(curdif-1).'d'
|
||
elseif nchar ==? "\<s-down>"
|
||
let cue = ((curdif+7>=0) ?'+':'').(curdif+7).'d'
|
||
elseif nchar ==? "\<s-up>"
|
||
let cue = ((curdif-7>=0) ?'+':'').(curdif-7).'d'
|
||
elseif nchar ==? "\<c-down>"
|
||
let cue = ((curdif+30>=0) ?'+':'').(curdif+30).'d'
|
||
elseif nchar ==? "\<c-up>"
|
||
let cue = ((curdif-30>=0) ?'+':'').(curdif-30).'d'
|
||
elseif nchar ==? "\<s-c-down>"
|
||
let cue = ((curdif+365>=0) ?'+':'').(curdif+365).'d'
|
||
elseif nchar ==? "\<s-c-up>"
|
||
let cue = ((curdif-365>=0) ?'+':'').(curdif-365).'d'
|
||
elseif newchar ==? "\<cr>"
|
||
break
|
||
elseif newchar ==? "\<Esc>"
|
||
hi Cursor guibg=gray
|
||
if bufwinnr('__Calendar') > 0
|
||
bdelete Calendar
|
||
endif
|
||
redraw
|
||
return ''
|
||
elseif (nchar ==? "\<LeftMouse>") && (v:mouse_win > 0) && (bufwinnr('__Calendar') == v:mouse_win)
|
||
let g:cal_list=[]
|
||
exe v:mouse_win . "wincmd w"
|
||
exe v:mouse_lnum
|
||
exe "normal " . v:mouse_col."|"
|
||
normal
|
||
if g:cal_list != []
|
||
if newtime ># ''
|
||
let timespec = newtime
|
||
else
|
||
let timespec = matchstr(newdate,'\S\+:.*>')
|
||
endif
|
||
let newdate = '<'.g:cal_list[0].'-'.s:Pre0(g:cal_list[1]).'-'.s:Pre0(g:cal_list[2]) . ' '
|
||
let newdate .= calutil#dayname( g:cal_list[0].'-'.g:cal_list[1].'-'.g:cal_list[2])
|
||
let newdate .= timespec ># '' ? ' ' . timespec : ''.'>'
|
||
break
|
||
endif
|
||
else
|
||
let cue .= newchar
|
||
endif
|
||
try
|
||
let newdate = '<' . s:GetNewDate(cue,basedate,basetime) . '>'
|
||
catch
|
||
" don't raise error if user mistypes cue. . .
|
||
" or if last char makes cue uninterpretable
|
||
let newdate = "can't interpret date cue "
|
||
endtry
|
||
if g:org_use_calendar && (match(newdate,'\d\d\d\d-\d\d')>=0)
|
||
let s:org_cal_date = newdate[1:10]
|
||
call s:Calfunc(1,newdate[1:4],str2nr(newdate[6:7]))
|
||
endif
|
||
echon repeat(' ',72)
|
||
redraw
|
||
endwhile
|
||
hi Cursor guibg=gray
|
||
bdelete __Calendar
|
||
call org#RestoreLocation()
|
||
return newdate
|
||
endfunction
|
||
|
||
command! OrgColumns :call OrgColumnsDashboard()
|
||
function! OrgColumnsDashboard(...)
|
||
let key = (a:0==1) ? a:1 : ''
|
||
let save_cursor = getpos('.')
|
||
if !exists('w:v.columnview')
|
||
let w:v={'columnview':0}
|
||
let w:v.org_item_len=100
|
||
let w:v.org_colview_list = []
|
||
let w:v.org_current_columns = ''
|
||
let w:v.org_column_item_head = ''
|
||
endif
|
||
if !exists('b:v.org_columns_show_headings')
|
||
let b:v.org_columns_show_headings = 0
|
||
endif
|
||
let force_all = 0
|
||
|
||
echohl WarningMsg
|
||
let save_more = &more
|
||
set nomore
|
||
while 1
|
||
if key ==# ''
|
||
echohl MoreMsg
|
||
echo "=========================================="
|
||
echo " Buffer default columns: " . b:v.buffer_columns
|
||
echo " Current default columns: " . w:v.org_current_columns
|
||
echo " Column view is currently: " . (w:v.columnview==1 ? 'ON' : 'OFF')
|
||
echo " Show column headers is currently: " . (b:v.org_columns_show_headings ? 'ON' : 'OFF')
|
||
echo " Heading line count is currently: " . (g:org_show_fold_lines==1 ? 'ON' : 'OFF')
|
||
if (w:v.columnview == 0) && (force_all == 1)
|
||
echo " NEXT CHOICE WILL BE APPLIED TO ENTIRE BUFFER"
|
||
endif
|
||
echo " "
|
||
echo " Press key to enter a columns command"
|
||
echo " ------------------------------------"
|
||
if (w:v.columnview == 0) && (force_all == 0)
|
||
echo " f force all of buffer to use chosen columns"
|
||
endif
|
||
if w:v.org_current_columns != b:v.buffer_columns
|
||
echo " r revert to buffer default columns"
|
||
endif
|
||
echo " t toggle column view on/off"
|
||
echo " h toggle show headings on/off"
|
||
echo " l line count on/off"
|
||
if len(g:org_custom_column_options) > 0
|
||
echo " Custom columns settings:"
|
||
endif
|
||
let i = 0
|
||
while i < len(g:org_custom_column_options)
|
||
echo " " . i . " " . g:org_custom_column_options[i]
|
||
let i += 1
|
||
endwhile
|
||
echo " ------------------------------------"
|
||
echo " "
|
||
echohl Question
|
||
let key = nr2char(getchar())
|
||
redraw
|
||
endif
|
||
|
||
if key ==? 'f'
|
||
let force_all = 1
|
||
let key = ''
|
||
redraw
|
||
continue
|
||
endif
|
||
|
||
let master_head = (force_all == 1 ) ? 0 : line('.')
|
||
|
||
if key ==? 'r'
|
||
let w:v.org_current_columns = b:v.buffer_columns
|
||
if w:v.columnview == 1
|
||
"turn off col view
|
||
call ToggleColumnView(master_head, w:v.org_current_columns)
|
||
endif
|
||
call ToggleColumnView(master_head, w:v.org_current_columns)
|
||
elseif key ==? 't'
|
||
" current columns will get set in SetColumnHeads()
|
||
call ToggleColumnView(master_head,'')
|
||
elseif key ==? 'h'
|
||
let b:v.org_columns_show_headings = 1 - b:v.org_columns_show_headings
|
||
elseif key ==? 'l'
|
||
let g:org_show_fold_lines = 1 - g:org_show_fold_lines
|
||
elseif key =~ '[0-9]'
|
||
let w:v.org_current_columns = g:org_custom_column_options[key]
|
||
if w:v.columnview == 1
|
||
" turn off
|
||
call ToggleColumnView(master_head, w:v.org_current_columns)
|
||
endif
|
||
call ToggleColumnView(master_head, w:v.org_current_columns)
|
||
else
|
||
echo "No column option selected."
|
||
endif
|
||
break
|
||
endwhile
|
||
|
||
if b:v.org_columns_show_headings == 0
|
||
call s:ColHeadWindow('',0)
|
||
elseif (w:v.columnview == 1) && (bufnr('ColHeadBuffer') == -1)
|
||
call s:ColHeadWindow(w:v.org_column_item_head)
|
||
endif
|
||
echohl None
|
||
let &more = save_more
|
||
call s:AdjustItemLen()
|
||
" redraw folded headings
|
||
setlocal foldtext=OrgFoldText()
|
||
call setpos('.',save_cursor)
|
||
endfunction
|
||
function! OrgDateDashboard(...)
|
||
let key = (a:0==1) ? a:1 : ''
|
||
let save_cursor = getpos('.')
|
||
let save_window = winnr()
|
||
if bufname("%") ==? ('__Agenda__')
|
||
let file = s:filedict[str2nr(matchstr(getline(line('.')), '^\d\d\d'))]
|
||
let lineno = str2nr(matchstr(getline(line('.')),'^\d\d\d\zs\d*'))
|
||
let buffer_lineno = s:ActualBufferLine(lineno,bufnr(file))
|
||
let props = s:GetProperties(buffer_lineno, 0, file)
|
||
else
|
||
let props = s:GetProperties(line('.'),0)
|
||
endif
|
||
|
||
if key ==# ''
|
||
echohl MoreMsg
|
||
echo " ================================="
|
||
echo " Press key, for a date command:"
|
||
echo " ---------------------------------"
|
||
echo " d set DEADLINE for current heading (currently: " . get(props,'DEADLINE','NONE') . ')'
|
||
echo " s set SCHEDULED for current heading (currently: " . get(props,'SCHEDULED','NONE') . ')'
|
||
echo " c set CLOSED for current heading (currently: " . get(props,'CLOSED','NONE') . ')'
|
||
echo " t set TIMESTAMP for current heading (currently: " . get(props,'TIMESTAMP','NONE') . ')'
|
||
echo " g set date at cursor"
|
||
echo " "
|
||
echo " "
|
||
echohl Question
|
||
let key = nr2char(getchar())
|
||
redraw
|
||
endif
|
||
if key ==? 'd'
|
||
call OrgDateEdit('DEADLINE')
|
||
elseif key ==? 's'
|
||
call OrgDateEdit('SCHEDULED')
|
||
elseif key ==? 'c'
|
||
call OrgDateEdit('CLOSED')
|
||
elseif key ==? 't'
|
||
call OrgDateEdit('TIMESTAMP')
|
||
elseif key ==? 'g' && (bufname('%') != '__Agenda__')
|
||
call OrgDateEdit('ATCURSOR')
|
||
else
|
||
echo "No date command selected."
|
||
endif
|
||
echohl None
|
||
exe save_window . 'wincmd w'
|
||
call setpos('.',save_cursor)
|
||
endfunction
|
||
function! OrgAgendaDateInc(datecue)
|
||
let agline = getline(line('.'))
|
||
if agline =~ 'Deadline:\|In.*\dd\.:'
|
||
call OrgDateEdit('DEADLINE',a:datecue)
|
||
elseif agline =~ 'Scheduled:\|Sched:'
|
||
call OrgDateEdit('SCHEDULED',a:datecue)
|
||
let type = 'SCHEDULED'
|
||
elseif agline =~ 'Closed:'
|
||
call OrgDateEdit('CLOSED',a:datecue)
|
||
else
|
||
call OrgDateEdit('TIMESTAMP',a:datecue)
|
||
endif
|
||
|
||
endfunction
|
||
function! OrgDateEdit(type,...)
|
||
if a:0 == 1
|
||
"use this to bypass calendar ui
|
||
let datecue = a:1
|
||
endif
|
||
" type can equal DEADLINE/CLOSED/SCHEDULED/TIMESTAMP/ATCURSOR
|
||
let save_cursor = getpos('.')
|
||
let old_cal_navi = g:calendar_navi
|
||
unlet g:calendar_navi
|
||
try
|
||
let dtype = a:type
|
||
if bufname("%") ==? ('__Agenda__')
|
||
"get file, lineno, and other data if in Agenda
|
||
let from_agenda=1
|
||
let file = s:filedict[str2nr(matchstr(getline(line('.')), '^\d\d\d'))]
|
||
let lineno = str2nr(matchstr(getline(line('.')),'^\d\d\d\zs\d*'))
|
||
let buffer_lineno = s:ActualBufferLine(lineno,bufnr(file))
|
||
let bufline = OrgGetLine(buffer_lineno,file)
|
||
else
|
||
let from_agenda=0
|
||
let buffer_lineno = line('.')
|
||
let bufline = getline(buffer_lineno)
|
||
let file = expand("%:t")
|
||
endif
|
||
if dtype =~ '\(DEADLINE\|SCHEDULED\|CLOSED\|TIMESTAMP\|ATCURSOR\)'
|
||
if dtype == 'ATCURSOR'
|
||
let my_date = GetDateSpecAtCursor()
|
||
else
|
||
let my_date = s:GetProp(dtype,buffer_lineno, file)
|
||
endif
|
||
|
||
let bracket = my_date[0]
|
||
let orig_date = matchstr( my_date, '[[<]\zs\d\d\d\d-\d\d-\d\d' )
|
||
let orig_time = matchstr( my_date, '[[<]\d\d\d\d-\d\d-\d\d ... \zs\d\d:\d\d' )
|
||
if orig_time ># ''
|
||
let rpt_or_warning = matchstr( my_date, ' \d\d:\d\d\zs .*\ze.$')
|
||
else
|
||
let rpt_or_warning = matchstr( my_date, '-\d\d-\d\d \S\S\S\zs .*\ze.$')
|
||
endif
|
||
|
||
if !exists('datecue')
|
||
let cal_result = CalEdit(orig_date, orig_time)
|
||
if cal_result ==# ''
|
||
return
|
||
endif
|
||
else
|
||
let cal_result = s:GetNewDate(datecue, orig_date, orig_time)
|
||
let cal_result = bracket . cal_result . (bracket=='<'?'>':']')
|
||
endif
|
||
if bracket == '['
|
||
let cal_result = '[' . cal_result[1:-2] . rpt_or_warning . ']'
|
||
else
|
||
let cal_result = '<' . cal_result[1:-2] . rpt_or_warning . '>'
|
||
endif
|
||
|
||
" back to main window if agenda is present and cursor is in it
|
||
if (from_agenda == 0) && bufname("%") ==? '__Agenda__'
|
||
"wincmd k
|
||
exe bufwinnr(file) . 'wincmd w'
|
||
endif
|
||
|
||
if dtype == 'ATCURSOR'
|
||
" put new date into text
|
||
call setpos('.', save_cursor)
|
||
if cal_result =~ '^.\d\d\d\d-\d\d'
|
||
let @x = cal_result[1:-2]
|
||
if my_date ># ''
|
||
"replace existing date within delimiters
|
||
exec 'normal vi' . my_date[0] . 'd'
|
||
normal h"xpll
|
||
else
|
||
"paste in brand new date
|
||
exec 'normal i <> '
|
||
normal hh"xpll
|
||
endif
|
||
endif
|
||
else
|
||
" set the prop with new date . . .
|
||
if (from_agenda == 0)
|
||
"just set the prop if we're in org file buffer
|
||
call s:SetProp(dtype,cal_result,buffer_lineno, file)
|
||
else
|
||
if empty(b:v.heading_marks_dict)
|
||
" just mark and do current item
|
||
"let b:v.heading_marks = [line('.')]
|
||
let b:v.heading_marks_dict[line('.')] = 1
|
||
endif
|
||
for item in sort(keys(b:v.heading_marks_dict), 's:ReverseSort')
|
||
exec item
|
||
let file = s:filedict[str2nr(matchstr(getline(line('.')), '^\d\d\d'))]
|
||
let lineno = str2nr(matchstr(getline(line('.')),'^\d\d\d\zs\d*'))
|
||
let buffer_lineno = s:ActualBufferLine(lineno,bufnr(file))
|
||
let bufline = OrgGetLine(buffer_lineno,file)
|
||
call s:SetProp(dtype,cal_result,buffer_lineno, file)
|
||
"put prop change indicator on line(s)
|
||
let cur_line = getline(line('.'))
|
||
let mylen = winwidth(0) - 29
|
||
let cur_line = (len(cur_line) > mylen) ? cur_line[:mylen] : cur_line . repeat(' ', mylen-len(cur_line))
|
||
let change_indicator = ' ' . dtype[0] . ' => ' . dtype . ' on ' . cal_result
|
||
call setline(line('.'), cur_line . change_indicator)
|
||
execute 'sign unplace ' . item . ' buffer=' . bufnr('%')
|
||
endfor
|
||
let b:v.heading_marks = []
|
||
let b:v.heading_marks_dict = {}
|
||
endif
|
||
endif
|
||
|
||
redraw
|
||
echo
|
||
redraw
|
||
else
|
||
echo "Date type wasn't DEADLINE, SCHEDULED, CLOSED, TIMESTAMP, or ATCURSOR."
|
||
endif
|
||
finally
|
||
let g:calendar_navi = old_cal_navi
|
||
call setpos('.',save_cursor)
|
||
endtry
|
||
endfunction
|
||
|
||
function! s:GetNewTime(cue, basetime)
|
||
" called from caledit()
|
||
let timecue = a:cue
|
||
if timecue =~ '\d\d:\d\d'
|
||
let mytime = ' '.timecue
|
||
else
|
||
let mytime = ''
|
||
endif
|
||
return mytime
|
||
|
||
endfunction
|
||
|
||
function! s:GetNewDate(cue,basedate,basetime)
|
||
" called from caledit()
|
||
if match(a:cue,'\S:') >= 0
|
||
let mlist = matchlist(a:cue, '^\(.*\) \(\S\+:.*\)')
|
||
let cue = mlist[1]
|
||
let timecue = mlist[2]
|
||
else
|
||
let cue = a:cue
|
||
let timecue = ''
|
||
endif
|
||
let basedate = a:basedate
|
||
let newdate = DateCueResult( cue , basedate )
|
||
let tmatch = matchlist( timecue , '\(\d\{1,2}\):\(\d\d\)\(am\|pm\)*')
|
||
if !empty(tmatch)
|
||
let hours = tmatch[1]
|
||
if (tmatch[3]=='pm') && (hours < 12)
|
||
let hours = hours + 12
|
||
elseif (tmatch[3]=='am') && (hours == 12)
|
||
let hours = 0
|
||
endif
|
||
let hours = s:Pre0(hours)
|
||
let mytime = ' ' . hours . ':' . tmatch[2]
|
||
else
|
||
let mytime = a:basetime ># '' ? ' ' . a:basetime : ''
|
||
endif
|
||
let mydow = calutil#dayname(newdate)
|
||
return newdate . ' ' . mydow . mytime
|
||
endfunction
|
||
function! DateCueResult( cue, basedate)
|
||
let cue = a:cue
|
||
let basedate = a:basedate
|
||
if cue =~ '^\(+\|++\|-\|--\)$'
|
||
let cue = cue . '1d'
|
||
elseif cue =~ '^\(+\|++\|-\|--\)\d\+$'
|
||
let cue = cue .'d'
|
||
endif
|
||
if cue ==? '.'
|
||
let newdate = strftime('%Y-%m-%d')
|
||
elseif cue ==# ''
|
||
let newdate = a:basedate
|
||
elseif (cue =~ '^\d\+$') && (str2nr(cue) <= 31)
|
||
" day of month string
|
||
if str2nr(cue) > str2nr(basedate[8:9])
|
||
let newdate = calutil#cal(calutil#jul(basedate[0:7].s:Pre0(cue)))
|
||
else
|
||
let newmonth = s:Pre0(basedate[5:6]+1)
|
||
let newdate = calutil#cal(calutil#jul(basedate[0:4].newmonth.'-'.s:Pre0(cue)))
|
||
endif
|
||
elseif cue =~ '^\d\+[-/]\d\+$'
|
||
" month/day string
|
||
let month = matchstr(cue,'^\d\+')
|
||
let day = matchstr(cue,'\d\+$')
|
||
let year = basedate[0:3]
|
||
if basedate[0:4] . s:Pre0(month) . '-' . s:Pre0(day) < basedate
|
||
let year = year + 1
|
||
endif
|
||
let newdate = calutil#cal(calutil#Cal2Jul(year,month,day))
|
||
elseif cue =~ '\d\+/\d\+/\d\+'
|
||
" m/d/y string
|
||
let month = matchstr(cue,'^\d\+\ze/.*/')
|
||
let day = matchstr(cue,'/\zs\d\+\ze/')
|
||
let year = matchstr(cue,'/\zs\d\+\ze$')
|
||
if len(year) < 3
|
||
let year +=2000
|
||
endif
|
||
let newdate = calutil#cal(calutil#Cal2Jul(year,month,day))
|
||
elseif cue =~ '\d\+-\d\+-\d\+'
|
||
" y-m-d string
|
||
let year = matchstr(cue,'^\d\+\ze-.*-')
|
||
if year < 100
|
||
let year +=2000
|
||
endif
|
||
let month = matchstr(cue,'-\zs\d\+\ze-')
|
||
let day = matchstr(cue,'-\zs\d\+\ze$')
|
||
let newdate = calutil#cal(calutil#Cal2Jul(year,month,day))
|
||
|
||
elseif cue =~ s:org_monthstring . ' \d\+$'
|
||
let month_str = matchstr(cue,'^' . s:org_monthstring)
|
||
let month = 1 + index(s:org_months, tolower(month_str[:2]))
|
||
let day = matchstr(cue,' \zs\d\+$')
|
||
let year = basedate[0:3]
|
||
if basedate[0:4] . s:Pre0(month) . '-' . s:Pre0(day) < basedate
|
||
let year = year + 1
|
||
endif
|
||
let newdate = calutil#cal(calutil#Cal2Jul(year,month,day))
|
||
elseif cue =~ s:org_monthstring . ' \d\+ \d\+$'
|
||
let month_str = matchstr(cue,'^' . s:org_monthstring)
|
||
let month = 1 + index(s:org_months, tolower(month_str[:2]))
|
||
let day = matchstr(cue,' \zs\d\+\ze ')
|
||
let year = str2nr(matchstr(cue,' \d\+ \zs\d\+$'))
|
||
let year = (year < 1800) ? 2000 + year : year
|
||
let newdate = calutil#cal(calutil#Cal2Jul(year,month,day))
|
||
elseif cue =~ s:org_weekdaystring
|
||
" wed, 3tue, 5fri, i.e., dow string
|
||
let mycount = matchstr(cue,'^\d\+')
|
||
let myday = matchstr(cue,s:org_weekdaystring)
|
||
let newday = index(s:org_weekdays,tolower(myday))
|
||
let oldday = calutil#dow(matchstr(basedate,'\d\d\d\d-\d\d-\d\d'))
|
||
if newday > oldday
|
||
let amt=newday-oldday
|
||
elseif newday < oldday
|
||
let amt =7-oldday+newday
|
||
else
|
||
let amt = 7
|
||
endif
|
||
let amt = amt + (mycount*7)
|
||
let newdate=calutil#cal(calutil#jul(basedate)+amt)
|
||
elseif cue =~ '\c\([-+]\|[-+][-+]\)\d\+[ dwmy]'
|
||
" plus minus count of dwmy
|
||
let mlist = matchlist(cue,'\c\([-+]\|[-+][-+]\)\(\d\+\)\([ wdmy]\)')
|
||
let op = mlist[1]
|
||
let mycount = mlist[2]
|
||
let type = mlist[3]
|
||
if len(op) == 1
|
||
let mydate = strftime('%Y-%m-%d')
|
||
else
|
||
let mydate = basedate
|
||
endif
|
||
let op = op[0]
|
||
let year = mydate[0:3]
|
||
let month = mydate[5:6]
|
||
let day = mydate[8:9]
|
||
if type ==? 'y'
|
||
let type = 'm'
|
||
let mycount = mycount * 12
|
||
elseif type ==? 'w'
|
||
let type='d'
|
||
let mycount = mycount * 7
|
||
endif
|
||
if type ==? 'm'
|
||
if (op ==? '+')
|
||
let yplus = mycount / 12
|
||
let mplus = mycount % 12
|
||
let year += yplus
|
||
let month += mplus
|
||
if month > 12
|
||
let month = month - 12
|
||
let year = year + 1
|
||
endif
|
||
elseif ((mycount % 12) >= month) && (op ==? '-')
|
||
let yminus = mycount/12
|
||
let year = year - yminus - 1
|
||
let month = (month + 12 - (mycount % 12))
|
||
else " '-' with month greater
|
||
let month = month - (mycount % 12)
|
||
let year = year - (mycount / 12)
|
||
endif
|
||
" correct for bad dates
|
||
while calutil#cal(calutil#jul(year.'-'.s:Pre0(month).'-'.s:Pre0(day)))[5:6] != month
|
||
let day = day - 1
|
||
endwhile
|
||
elseif (type ==? 'd') || (type ==? ' ')
|
||
let newjul = calutil#jul(mydate)
|
||
if op ==? '+'
|
||
let newjul = newjul + mycount
|
||
else
|
||
let newjul = newjul - mycount
|
||
endif
|
||
"execute 'let newjul = newjul ' . op . mycount
|
||
let mydate = calutil#cal(newjul)
|
||
let year = mydate[0:3]
|
||
let month = mydate[5:6]
|
||
let day = mydate[8:9]
|
||
endif
|
||
|
||
let newdate = year . '-' . s:Pre0(month) . '-' . s:Pre0(day)
|
||
else
|
||
return " ?? can't interpret your spec"
|
||
endif
|
||
return newdate
|
||
endfunction
|
||
function! s:TimeInc(direction)
|
||
let save_cursor = getpos(".")
|
||
let i = 0
|
||
let col = save_cursor[2] - 1
|
||
let line = getline(line("."))
|
||
if line[col] =~ '\d'
|
||
let i = 1
|
||
while i < 6
|
||
let start = col - i
|
||
let end = col - i + 6
|
||
silent execute 'let timetest = line[' . start . ':' . end .']'
|
||
if timetest =~ ' \d\d:\d\d[>\]]'
|
||
break
|
||
endif
|
||
let i += 1
|
||
endwhile
|
||
else
|
||
let i = 6
|
||
endif
|
||
if i == 6
|
||
execute "normal! \<s-up>"
|
||
return
|
||
else
|
||
let start = col - i + 1
|
||
let end = col - i + 5
|
||
execute 'let time = line[' . start . ':' . end .']'
|
||
if i > 3
|
||
let newminutes = (time[3:4] + (a:direction *5))
|
||
let newminutes = newminutes - (newminutes % 5)
|
||
if (newminutes >= 60)
|
||
let newminutes = 0
|
||
let newhours = time[0:1] + 1
|
||
elseif (newminutes == -5) && (a:direction == -1)
|
||
let newminutes = 55
|
||
let newhours = time[0:1] - 1
|
||
else
|
||
let newhours = time[0:1]
|
||
endif
|
||
else
|
||
let newhours = time[0:1] + (1 * a:direction)
|
||
let newminutes = time[3:4]
|
||
endif
|
||
if newhours >= 24
|
||
let newhours = 0
|
||
"let tempsave = getpos(".")
|
||
elseif newhours < 0
|
||
let newhours = 23
|
||
"execute "normal ".start-6."|"
|
||
"call OrgDateInc(a:direction)
|
||
"call setpos(".",tempsave)
|
||
endif
|
||
let matchcol = col-i+2
|
||
execute 's/\%'.matchcol.'c\zs\d\d:\d\d/' . s:Pre0(newhours) . ':' . s:Pre0(newminutes).'/'
|
||
endif
|
||
call setpos(".",save_cursor)
|
||
endfunction
|
||
function! OrgDateInc(direction)
|
||
" <dddd-dd-dd
|
||
" 01234567890
|
||
" 09876543210
|
||
let save_cursor = getpos(".")
|
||
let i = 0
|
||
let col = save_cursor[2] - 1
|
||
let line = getline(line("."))
|
||
if line[col] =~ '\d'
|
||
let i = 1
|
||
while i < 21
|
||
let start = col - i
|
||
let end = col - i + 11
|
||
silent execute 'let datetest = line[' . start . ':' . end .']'
|
||
if datetest =~ '[<[]\d\d\d\d-\d\d-\d\d'
|
||
break
|
||
endif
|
||
let i += 1
|
||
endwhile
|
||
else
|
||
let i = 21
|
||
endif
|
||
if i == 21
|
||
execute "normal! \<s-up>"
|
||
return
|
||
else
|
||
if i > 12
|
||
call setpos(".",save_cursor)
|
||
call s:TimeInc(a:direction)
|
||
return
|
||
endif
|
||
let start = col - i + 1
|
||
let end = col - i + 11
|
||
execute 'let date = line[' . start . ':' . end .']'
|
||
if i > 7
|
||
let newdate = calutil#cal(calutil#jul(date) + a:direction)
|
||
let newyear = newdate[0:3]
|
||
let newmonth = newdate[5:6]
|
||
let newday = newdate[8:9]
|
||
elseif i < 5
|
||
let spot = 'year'
|
||
let newyear = date[0:3] + a:direction
|
||
let newmonth = date[5:6]
|
||
let newday = date[8:9]
|
||
"execute 's/\d\d\d\d/' . newyear . '/'
|
||
else
|
||
let spot = 'month'
|
||
let newmonth = date[5:6] + a:direction
|
||
let newday = date[8:9]
|
||
if newmonth > 12
|
||
let newyear = date[0:3] + 1
|
||
let newmonth = '01'
|
||
let newday = '01'
|
||
elseif newmonth < 1
|
||
let newyear = date[0:3] - 1
|
||
let newmonth = '12'
|
||
let newday = '31'
|
||
else
|
||
let newyear = date[0:3]
|
||
let newday = date[8:9]
|
||
endif
|
||
endif
|
||
" correct for bad dates
|
||
while calutil#cal(calutil#jul(newyear.'-'.newmonth.'-'.newday))[5:6] != newmonth
|
||
let newday = newday - 1
|
||
endwhile
|
||
let matchcol = col-i+2
|
||
execute 's/\%'.matchcol.'c\zs\d\d\d\d-\d\d-\d\d/' . newyear . '-' . s:Pre0(newmonth) . '-' . s:Pre0(newday).'/'
|
||
" update dow if there is one
|
||
let end +=5
|
||
silent execute 'let datetest = line[' . start . ':' . end .']'
|
||
if datetest =~ '\d\d\d\d-\d\d-\d\d \S\S\S'
|
||
let dow = calutil#DayOfWeek(newyear,newmonth,newday,2)
|
||
silent execute 's/\%'.matchcol.'c\(\d\d\d\d-\d\d-\d\d \)\S\S\S/\1' . dow.'/'
|
||
endif
|
||
endif
|
||
call setpos(".",save_cursor)
|
||
endfunction
|
||
|
||
function! s:GetClock()
|
||
return '['.strftime("%Y-%m-%d %a %H:%M").']'
|
||
endfunction
|
||
function! OrgClockIn(...)
|
||
let save_cursor=getpos(".")
|
||
let lineno=line('.')
|
||
if bufname("%") ==? ('__Agenda__')
|
||
let lineno = matchstr(getline(line('.')),'^\d\+')
|
||
let file = matchstr(getline(line('.')),'^\d\+\s*\zs\S\+').'.org'
|
||
let str = ','.lineno.',"'.file.'"'
|
||
call s:SetProp('CLOCKIN','',lineno,file)
|
||
else
|
||
|
||
if a:0 > 1
|
||
execute a:1
|
||
endif
|
||
execute s:OrgGetHead()
|
||
if s:IsTagLine(line(".")+1)
|
||
execute line('.')+1
|
||
endif
|
||
call append(line('.'),' :CLOCK: '.s:GetClock())
|
||
let dict={'file':expand("%"),'line':line('.'),'Timestamp':org#Timestamp()}
|
||
call add(g:org_clock_history,dict)
|
||
endif
|
||
|
||
|
||
call setpos(".",save_cursor)
|
||
endfunction
|
||
function! s:GetOpenClock()
|
||
let found_line = 0
|
||
let file = ''
|
||
if !exists('g:agenda_files') || (g:agenda_files == [])
|
||
unsilent call confirm("No agenda files defined, will search only this buffer for open clocks.")
|
||
let found_line = search('CLOCK: \[\d\d\d\d-\d\d-\d\d \S\S\S \d\d:\d\d\]\($\|\s\)','w')
|
||
else
|
||
let g:in_agenda_search = 1
|
||
for file in g:agenda_files
|
||
call org#LocateFile(file)
|
||
let found_line = search('CLOCK: \[\d\d\d\d-\d\d-\d\d \S\S\S \d\d:\d\d\]\($\|\s\)','w')
|
||
let file = expand("%")
|
||
if found_line > 0
|
||
break
|
||
endif
|
||
endfor
|
||
unlet g:in_agenda_search
|
||
endif
|
||
return [file,found_line]
|
||
endfunction
|
||
function! OrgClockOut(...)
|
||
let cur_file=expand("%")
|
||
let save_cursor= getpos('.')
|
||
if a:0 > 1
|
||
execute a:1
|
||
else
|
||
let oc = s:GetOpenClock()
|
||
if oc[0] ># ''
|
||
call org#LocateFile(oc[0])
|
||
execute oc[1]
|
||
endif
|
||
endif
|
||
execute s:OrgGetHead()
|
||
let bottom = s:OrgNextHead() > 0 ? s:OrgNextHead() - 1 : line("$")
|
||
let str = 'CLOCK: \[\d\d\d\d-\d\d-\d\d \S\S\S \d\d:\d\d\]\($\|\s\)'
|
||
let found = s:Range_Search(str,'n',bottom,line("."))
|
||
if found
|
||
execute found
|
||
execute 'normal A--' . s:GetClock()
|
||
if b:v.clock_to_logbook
|
||
let headline = s:OrgGetHead()
|
||
let clockline = getline(line(".")) . ' -> ' . s:ClockTime(line("."))
|
||
normal! dd
|
||
call OrgConfirmDrawer("LOGBOOK",headline)
|
||
let clockline = matchstr(getline(line(".")),'^\s*') . matchstr(clockline,'\S.*')
|
||
call append(line("."),clockline )
|
||
endif
|
||
let msg = "Open clock found and clocked out in \n"
|
||
let msg .= "file: ".expand("%")."\n"
|
||
let msg .= "in headline at line number: ".headline
|
||
call confirm(msg)
|
||
else
|
||
echo 'No open clock found. . . .'
|
||
endif
|
||
call org#LocateFile(cur_file)
|
||
call setpos(".",save_cursor)
|
||
endfunction
|
||
function! s:UpdateAllClocks()
|
||
%g/^\s*:CLOCK:/call s:AddClockTime(line("."))
|
||
endfunction
|
||
function! s:AddClockTime(line)
|
||
call setline(a:line,matchstr(getline(a:line),'.*\]') . ' -> ' . s:ClockTime(a:line))
|
||
endfunction
|
||
|
||
function! s:UpdateClockSums()
|
||
let save_cursor = getpos(".")
|
||
g/^\s*:ItemClockTime/d
|
||
call s:UpdateAllClocks()
|
||
g/^\s*:CLOCK:/call s:SetProp('ItemClockTime', s:SumClockLines(line(".")))
|
||
call setpos(".",save_cursor)
|
||
endfunction
|
||
|
||
function! s:SumClockLines(line)
|
||
let save_cursor = getpos(".")
|
||
execute s:OrgGetHead_l(a:line) + 1
|
||
"execute a:line + 1
|
||
let hours = 0
|
||
let minutes = 0
|
||
while 1
|
||
let text = getline(line("."))
|
||
if text !~ s:remstring
|
||
break
|
||
endif
|
||
let time = matchstr(text,'CLOCK.*->\s*\zs\d\+:\d\+')
|
||
if time ># ''
|
||
let hours += str2nr(split(time,':')[0])
|
||
let minutes += str2nr(split(time,':')[1])
|
||
endif
|
||
|
||
if line('.') == line('$')
|
||
break
|
||
else
|
||
execute line('.') + 1
|
||
endif
|
||
|
||
endwhile
|
||
let totalminutes = (60 * hours) + minutes
|
||
call setpos(".",save_cursor)
|
||
return (totalminutes/60) . ':' . s:Pre0(totalminutes % 60)
|
||
|
||
endfunction
|
||
function! s:UpdateBlock()
|
||
normal j
|
||
?^#+BEGIN:
|
||
let block_type = matchstr(getline(line('.')),'\S\+\s\+\zs\S\+')
|
||
if matchstr(getline(line('.')+1),'^#+END') ==# ''
|
||
normal jV/^#+END/-1
|
||
dk
|
||
endif
|
||
if block_type ==? 'clocktable'
|
||
let block_type='ClockTable'
|
||
endif
|
||
let mycommand = block_type.'()'
|
||
execute "call append(line('.'),".mycommand.")"
|
||
endfunction
|
||
function! ClockTable()
|
||
let save_cursor = getpos(".")
|
||
|
||
call s:UpdateClockSums()
|
||
call s:UpdateHeadlineSums()
|
||
call OrgMakeDict()
|
||
let g:ctable_dict = {}
|
||
let mycommand = "let g:ctable_dict[line('.')] = "
|
||
\ . "{'text':s:GetProperties(line('.'),0)['ITEM']"
|
||
\ . " , 'time':s:GetProperties(line('.'),0)['TOTALCLOCKTIME']}"
|
||
g/:TOTALCLOCKTIME/execute mycommand
|
||
let total='00:00'
|
||
for item in keys(g:ctable_dict)
|
||
"let test = g:ctable_dict[item].text
|
||
if g:ctable_dict[item].text[0:1] ==? '* '
|
||
"if test[0:1] ==? '* '
|
||
let total = s:AddTime(total,g:ctable_dict[item].time)
|
||
endif
|
||
endfor
|
||
let result = ['Clock summary at ['.org#Timestamp().']','',
|
||
\ '|Lev| Heading | ClockTime',
|
||
\ '|---+------------------------------+-------+--------' ,
|
||
\ '| | *TOTAL* | '.total ]
|
||
for item in sort(keys(g:ctable_dict),'s:NumCompare')
|
||
let level = len(matchstr(g:ctable_dict[item].text,'^\*\+'))
|
||
let treesym = repeat(' ',level-2) . (level > 1 ? '\_ ' : '')
|
||
let str = '| '.level.' | '
|
||
\ . org#Pad(treesym . matchstr(g:ctable_dict[item].text,'^\*\+ \zs.*')[:20],28) . ' | '
|
||
\ . repeat(' | ',level-1)
|
||
\ . s:PrePad(g:ctable_dict[item].time,5) . ' |'
|
||
if g:ctable_dict[item].text[0:1] ==? '* '
|
||
call add(result, '|---+------------------------------+-------+-------+' )
|
||
endif
|
||
call add(result, str)
|
||
endfor
|
||
call setpos(".",save_cursor)
|
||
|
||
unlet b:v.org_dict
|
||
return result
|
||
|
||
endfunction
|
||
|
||
function! s:NumCompare(i1,i2)
|
||
let i1 = str2nr(a:i1)
|
||
let i2 = str2nr(a:i2)
|
||
return i1 == i2 ? 0 : i1>i2 ? 1 : -1
|
||
endfunction
|
||
|
||
function! s:ClockTime(line)
|
||
let ctext = getline(a:line)
|
||
let start = matchstr(ctext,'CLOCK:\s*\[\zs\S\+\s\S\+\s\S\+\ze\]')
|
||
let end = matchstr(ctext,'--\[\zs.*\ze\]')
|
||
let daydifference = calutil#jul(end[0:9])-calutil#jul(start[0:9])
|
||
let startmin = 60*start[15:16] + start[18:19]
|
||
let endmin = 60*end[15:16] + end[18:19]
|
||
let totalmin = (daydifference * 1440) + (endmin - startmin)
|
||
return string(totalmin/60) . ':' . s:Pre0(totalmin % 60)
|
||
endfunction
|
||
function! s:AddTime(time1, time2)
|
||
let time1 = a:time1
|
||
let time2 = a:time2
|
||
if match(time1,':') == -1 | let time1 = '00:00' | endif
|
||
if match(time2,':') == -1 | let time2 = '00:00' | endif
|
||
let hours = str2nr(matchstr(time1,'^.*\ze:')) + str2nr(matchstr(time2,'^.*\ze:'))
|
||
let minutes = (60*hours) + time1[-2:] + time2[-2:]
|
||
return (minutes/60) . ':' . s:Pre0(minutes % 60)
|
||
endfunction
|
||
function! s:GetProp(key,...)
|
||
let save_cursor = getpos(".")
|
||
if a:0 >=2
|
||
let curtab = tabpagenr()
|
||
let curwin = winnr()
|
||
" optional args are: a:1 - lineno, a:2 - file
|
||
call org#LocateFile(a:2)
|
||
endif
|
||
if (a:0 >= 1) && (a:1 > 0)
|
||
execute a:1
|
||
endif
|
||
execute s:OrgGetHead() + 1
|
||
let myval = ''
|
||
while 1
|
||
let text = getline(line("."))
|
||
if text !~ s:remstring
|
||
break
|
||
endif
|
||
let mymatch = matchstr(text,':\s*'.a:key.'\s*:')
|
||
if mymatch ># ''
|
||
let myval = matchstr(text,':\s*'.a:key.'\s*:\s*\zs.*$')
|
||
break
|
||
endif
|
||
execute line(".") + 1
|
||
if line(".") == line("$")
|
||
break
|
||
endif
|
||
endwhile
|
||
if a:0 >= 2
|
||
execute "tabnext ".curtab
|
||
execute curwin . "wincmd w"
|
||
endif
|
||
call setpos(".",save_cursor)
|
||
return myval
|
||
|
||
endfunction
|
||
function! s:SetDateProp(type,newdate,...)
|
||
" almost identical to s:GetProp() above, need to refactor
|
||
let save_cursor = getpos(".")
|
||
if a:0 == 1
|
||
execute a:1 + 1
|
||
else
|
||
execute line(".") + 1
|
||
endif
|
||
let myval = ''
|
||
while 1
|
||
let text = getline(line("."))
|
||
if text !~ s:remstring
|
||
break
|
||
endif
|
||
let mymatch = matchstr(text,'\s*'.a:type.'\s*:')
|
||
if mymatch ># ''
|
||
execute 's/'.a:type.'.*$/'.a:type.':<'.a:newdate.'>/'
|
||
break
|
||
endif
|
||
execute line(".") + 1
|
||
endwhile
|
||
call setpos(".",save_cursor)
|
||
return myval
|
||
endfunction
|
||
function! s:SetProp(key, val,...)
|
||
let save_cursor = getpos(".")
|
||
" optional args are: a:1 - lineno, a:2 - file
|
||
if a:0 >=2
|
||
let curtab = tabpagenr()
|
||
let curwin = winnr()
|
||
call org#LocateFile(a:2)
|
||
endif
|
||
if (a:0 >= 1) && (a:1 > 0)
|
||
execute a:1
|
||
endif
|
||
let key = a:key
|
||
let val = a:val
|
||
execute s:OrgGetHead()
|
||
" block_end was end of properties block, but getting that
|
||
" from GetProperties(line('.'),0) creates problems with
|
||
" line numbers having changed from previous run of OrgMakeDict
|
||
" So, just use next head as end of block for now.
|
||
let block_end = s:OrgNextHead()
|
||
let block_end = (block_end == 0) ? line('$') : block_end
|
||
if key =~ 'DEADLINE\|SCHEDULED\|CLOSED\|TIMESTAMP'
|
||
" it's one of the five date props
|
||
" find existing date line if there is one
|
||
if key ==? 'TIMESTAMP'
|
||
let key = ''
|
||
let foundline = s:Range_Search('^\s*:\s*<\d\d\d\d-\d\d-\d\d','n',block_end,line("."))
|
||
elseif key ==? 'TIMESTAMP_IA'
|
||
let key = ''
|
||
let foundline = s:Range_Search('^\s*:\s*[\d\d\d\d-\d\d-\d\d','n',block_end,line("."))
|
||
else
|
||
let foundline = s:Range_Search('^\s*\(:\)\{}'.key.'\s*:','n',block_end,line("."))
|
||
endif
|
||
if foundline > 0
|
||
"exec foundline
|
||
"exec 's/:\s*<\d\d\d\d-\d\d-\d\d \S\S\S\( \d\d:\d\d\)*/' . (key ==# '' ? ':' : ': ') . a:val[:-2]
|
||
call setline(foundline,substitute(getline(foundline),
|
||
\ ':\s*<\d\d\d\d-\d\d-\d\d \S\S\S.*$', (key ==# '' ? ':' : ': ') . a:val,''))
|
||
"\ ':\s*<\d\d\d\d-\d\d-\d\d \S\S\S.*$', (key ==# '' ? ':' : ': ') . a:val[:-2],''))
|
||
else
|
||
let line_ind = len(matchstr(getline(line(".")),'^\**'))+1 + g:org_indent_from_head
|
||
if s:IsTagLine(line('.')+1)
|
||
execute line('.') + 1
|
||
endif
|
||
call append(line("."),org#Pad(' ',line_ind)
|
||
\ .':'.key.(key ==# '' ? '' : ': ').a:val)
|
||
endif
|
||
elseif key ==? 'tags'
|
||
let indent_count = len(matchstr(getline(line(".")),'^\*\+ '))
|
||
let curindent = repeat(' ',indent_count)
|
||
let newval = curindent . a:val
|
||
if s:IsTagLine(line('.') + 1)
|
||
call setline(line('.') + 1, newval)
|
||
else
|
||
call append(line('.'), newval)
|
||
endif
|
||
"execute line('.') + 1
|
||
"normal =$
|
||
"execute line('.') - 1
|
||
elseif key ==? 'CLOCKIN'
|
||
call OrgClockIn()
|
||
elseif key ==? 'CLOCKOUT'
|
||
call OrgClockOut(a:val)
|
||
else
|
||
" it's a regular key/val pair in properties drawer
|
||
call OrgConfirmDrawer("PROPERTIES")
|
||
while (getline(line(".")) !~ '^\s*:\s*' . key) &&
|
||
\ (getline(line(".")) =~ s:remstring) &&
|
||
\ (line('.') != line('$'))
|
||
execute line(".") + 1
|
||
endwhile
|
||
|
||
if (getline(line(".")) =~ s:remstring) && (getline(line('.')) !~ '^\s*:END:')
|
||
call setline(line("."), matchstr(getline(line(".")),'^\s*:') .
|
||
\ key . ': ' . val)
|
||
else
|
||
execute line(".") - 1
|
||
call OrgConfirmDrawer("PROPERTIES")
|
||
let curindent = matchstr(getline(line(".")),'^\s*')
|
||
let newline = curindent . ':' . key . ': ' . val
|
||
call append(line("."),newline)
|
||
endif
|
||
endif
|
||
|
||
if exists("*Org_property_changed_functions") && (bufnr("%") != bufnr('Agenda'))
|
||
let Hook = function("Org_property_changed_functions")
|
||
silent execute "call Hook(line('.'),a:key, a:val)"
|
||
endif
|
||
if a:0 >=2
|
||
"back to tab/window where setprop call was made
|
||
execute "tabnext ".curtab
|
||
execute curwin . "wincmd w"
|
||
endif
|
||
call setpos(".",save_cursor)
|
||
endfunction
|
||
function! s:CurrentRemoveFromAgendaFiles()
|
||
let cur_file1 = fnamemodify(expand("%:p"), ":~")[1:]
|
||
let cur_file = substitute(cur_file1,'\','\\\\','g')
|
||
let cur_file = substitute(cur_file,'\\ ','\ ','g')
|
||
|
||
let file_count = len(g:agenda_files)
|
||
|
||
let ndx = -1
|
||
for i in range(0, file_count - 1 )
|
||
if g:agenda_files[i] =~ '\c' . cur_file
|
||
let ndx = i
|
||
break
|
||
endif
|
||
endfor
|
||
if ndx > -1
|
||
call remove(g:agenda_files,ndx)
|
||
echo cur_file1 . ' was removed from agenda files list.'
|
||
else
|
||
echo cur_file1 . ' was not in agenda files list.'
|
||
endif
|
||
endfunction
|
||
|
||
function! s:CurrentToAgendaFiles(top_or_bottom)
|
||
" argument should be either 'top' or 'bottom'
|
||
let cur_file1 = fnamemodify(expand("%:p"), ":~")
|
||
let cur_file = substitute(cur_file1[1:],'\','\\\\','g')
|
||
let cur_file = substitute(cur_file,'\\ ','\ ','g')
|
||
|
||
if !exists('g:agenda_files')
|
||
let g:agenda_files = []
|
||
endif
|
||
let file_count = len(g:agenda_files)
|
||
|
||
let ndx = -1
|
||
for i in range(0, file_count - 1 )
|
||
if g:agenda_files[i] =~ '\c' . cur_file
|
||
let ndx = i
|
||
break
|
||
endif
|
||
endfor
|
||
if ndx > -1
|
||
call remove(g:agenda_files,ndx)
|
||
endif
|
||
if a:top_or_bottom == 'top'
|
||
let g:agenda_files = [cur_file1] + g:agenda_files
|
||
else
|
||
let g:agenda_files = g:agenda_files + [cur_file1]
|
||
endif
|
||
echo cur_file1 . ' is at ' . a:top_or_bottom . ' of agenda files list.'
|
||
endfunction
|
||
|
||
function! s:OrgGotoChosenFile(...)
|
||
if a:0 == 1
|
||
" type is 'agenda' or 'all'
|
||
let type = a:1
|
||
else
|
||
let type = 'all'
|
||
endif
|
||
let fdict = {}
|
||
let bufnums= []
|
||
let bufnums_nonagenda = []
|
||
if exists('g:agenda_files') && !empty(g:agenda_files)
|
||
for i in range(0, len(g:agenda_files) - 1)
|
||
let buf = bufnr(g:agenda_files[i])
|
||
let buf = (buf == -1 ? 10000 + i : buf)
|
||
let fdict[buf] = g:agenda_files[i]
|
||
call add(bufnums,buf)
|
||
endfor
|
||
endif
|
||
if type == 'all'
|
||
let buffers = split(org#redir('buffers'), "\n")
|
||
for item in buffers
|
||
let m = matchlist(item,'\s*\(\d\+\).*"\(.*\)"')
|
||
if get(fdict, m[1], -1 ) == -1
|
||
"put buffer as key and fname as value
|
||
let fdict[m[1]] = m[2]
|
||
call add(bufnums_nonagenda,str2nr(m[1]))
|
||
endif
|
||
endfor
|
||
" get rid of non org buffers
|
||
for bufnum in keys(fdict)
|
||
if (getbufvar(str2nr(bufnum), "&filetype") !=# 'org') && (bufnum < 10000)
|
||
unlet fdict[bufnum]
|
||
call remove(bufnums_nonagenda,index(bufnums_nonagenda,str2nr(bufnum)))
|
||
endif
|
||
endfor
|
||
endif
|
||
|
||
redraw
|
||
echo "Agenda files:"
|
||
echo "============================="
|
||
let nums = range(char2nr('a'),char2nr('z')) + range(char2nr('0'), char2nr('9'))
|
||
|
||
for i in range(0,len(bufnums)-1)
|
||
echo ' ' . nr2char(nums[i]) . ' ' . g:agenda_files[i]
|
||
endfor
|
||
echo "Non-agenda org files:"
|
||
echo "============================="
|
||
for i in range(0,len(bufnums_nonagenda)-1)
|
||
echo ' ' . nr2char(nums[i + len(bufnums)]) . ' ' . fdict[bufnums_nonagenda[i]]
|
||
endfor
|
||
let bufnums = bufnums + bufnums_nonagenda
|
||
echo ""
|
||
echo "Press a key to go to chosen file: "
|
||
silent let x = nr2char(getchar())
|
||
redraw
|
||
|
||
if x =~ '[0-9a-z]'
|
||
silent call org#LocateFile(fdict[bufnums[index(nums,char2nr(x))]])
|
||
endif
|
||
|
||
endfunction
|
||
function! s:CycleAgendaFiles(direction)
|
||
if exists('g:agenda_files') && !empty('g:agenda_files')
|
||
let cur_file = fnamemodify(expand("%:p"), ":~")[1:]
|
||
let cur_file = substitute(cur_file,'\','\\\\','g')
|
||
let cur_file = substitute(cur_file,'\\ ','\ ','g')
|
||
|
||
let file_count = len(g:agenda_files)
|
||
"let ndx = index(g:agenda_files,cur_file)
|
||
let ndx = -1
|
||
for i in range(0, file_count - 1 )
|
||
if g:agenda_files[i] =~ '\c' . cur_file
|
||
let ndx = i
|
||
break
|
||
endif
|
||
endfor
|
||
if ndx > -1
|
||
"let ndx = (ndx == len(g:agenda_files)) ? 0 : ndx
|
||
let ndx += (a:direction ==? 'backward') ? -1 : 1
|
||
if a:direction == 'backward'
|
||
let ndx = (ndx == -1) ? (len(g:agenda_files) - 1) : ndx
|
||
elseif a:direction == 'forward'
|
||
let ndx = (ndx == len(g:agenda_files) ? 0 : ndx)
|
||
endif
|
||
|
||
let filename = g:agenda_files[ndx]
|
||
else
|
||
let filename = g:agenda_files[0]
|
||
endif
|
||
call org#LocateFile(filename)
|
||
else
|
||
echo "No agenda files defined."
|
||
endif
|
||
|
||
endfunction
|
||
|
||
function! OrgConfirmDrawer(type,...)
|
||
let line = s:OrgGetHead()
|
||
if a:0 == 1
|
||
let line = a:1
|
||
endif
|
||
execute line
|
||
let indent = repeat(' ', len(matchstr(getline(line('.')),'^\*\+ ')))
|
||
let bottom = s:OrgNextHead() > 0 ? s:OrgNextHead() - 1 : line("$")
|
||
let found = s:Range_Search(':\s*'. a:type . '\s*:','n',bottom,line)
|
||
if !found
|
||
while getline(line(".") + 1) =~ s:remstring
|
||
execute line('.')+1
|
||
endwhile
|
||
"execute 'normal o:'. a:type . ':'
|
||
"execute 'normal o:END:'
|
||
"normal k
|
||
call append(line('.'),[ indent . ':' . a:type . ':' , indent . ':END:'] )
|
||
execute line('.') + 1
|
||
else
|
||
execute found
|
||
endif
|
||
endfunction
|
||
|
||
function! OrgGetLink()
|
||
let savecursor = getpos('.')
|
||
|
||
let linkdict = {'link':'','desc':''}
|
||
let curpos = getpos('.')[2]
|
||
call search('\[\[','bc',line('.'))
|
||
let startpos = getpos('.')[2] - 1
|
||
call search(']]','ce',line('.'))
|
||
let endpos = getpos('.')[2] - 1
|
||
if (curpos >= startpos) && (curpos <= endpos)
|
||
let linktext = getline(line("."))[ startpos : endpos ]
|
||
if linktext =~ ']\['
|
||
let linkdict.link = matchstr(linktext,'\[\[\zs.*\ze]\[')
|
||
let linkdict.desc = matchstr(linktext,']\[\zs.*\ze]]')
|
||
else
|
||
let linkdict.link = matchstr(linktext,'\[\[\zs.*\ze]]')
|
||
endif
|
||
endif
|
||
call setpos('.',savecursor)
|
||
return linkdict
|
||
endfunction
|
||
function! FollowLink(ldict)
|
||
let ld = a:ldict
|
||
let ld.suffix = ''
|
||
"process things so org-compatible while still calling Utl
|
||
let prefix = matchstr(ld.link,'\S\{-1,}\ze:')
|
||
if exists(":Utl") == 0
|
||
echo "The Vim plugin Utl.vim must be installed to follow links."
|
||
echo "You can find a copy at:"
|
||
echo 'http://www.vim.org/scripts/script.php?script_id=293'
|
||
return
|
||
endif
|
||
if prefix =~ g:org_unsupported_link_types
|
||
echo 'Link type "' . prefix '" not supported in VimOrganizer.'
|
||
return
|
||
endif
|
||
" now have to translate from org format to Utl format
|
||
if ld.link[0] ==# '#'
|
||
let ld.link = '#tn=:CUSTOM_ID:\s\*' . ld.link[1:]
|
||
let prefix = '#'
|
||
elseif prefix ==? 'file' && (ld.link[5] ==# '.')
|
||
" || ld.link[5:6] ==# '\S:')
|
||
" take file prefix out b/c Utl can't handle relative file paths
|
||
let ld.link = ld.link[5:]
|
||
endif
|
||
|
||
if prefix ==? 'file' && (ld.link =~ '::')
|
||
let mylist = split(ld.link,'::')
|
||
let ld.link = mylist[0]
|
||
let ld.suffix = mylist[1]
|
||
endif
|
||
|
||
if (prefix ==# '') || ((prefix !~ g:org_supported_link_types) && (prefix != '#'))
|
||
" we have org text search that needs different treatment
|
||
call FollowTextLink(a:ldict.link)
|
||
"for search_type in ['dedicated', 'headline1', 'headline2', 'general']
|
||
" let savecursor = getpos('.')
|
||
" if search_type is 'dedicated'
|
||
" let newlink = '#tn=<<' . a:ldict.link . '>>'
|
||
" elseif search_type is 'headline1'
|
||
" let newlink = '#tn=' . b:v.todoMatch . a:ldict.link
|
||
" elseif search_type is 'headline2'
|
||
" let newlink = '#tn=^*\+ ' . a:ldict.link
|
||
" else
|
||
" let newlink = '#tn=' . a:ldict.link
|
||
" endif
|
||
|
||
" let newlink = substitute(newlink,' ','\\ ','g')
|
||
" let g:newlink = newlink
|
||
" silent! exec 'Utl o '. newlink . ' split'
|
||
|
||
" if line('.') != savecursor[1]
|
||
" break
|
||
" endif
|
||
"endfor
|
||
else
|
||
exec 'Utl o ' . ld.link . ' split'
|
||
if ld.suffix ># ''
|
||
call FollowTextLink(ld.suffix)
|
||
endif
|
||
end
|
||
endfunction
|
||
function! FollowTextLink(link)
|
||
for search_type in ['dedicated', 'headline1', 'headline2', 'general']
|
||
let savecursor = getpos('.')
|
||
if search_type is 'dedicated'
|
||
let newlink = '#tn=<<' . a:link . '>>'
|
||
elseif search_type is 'headline1'
|
||
let newlink = '#tn=' . b:v.todoMatch . a:link
|
||
elseif search_type is 'headline2'
|
||
let newlink = '#tn=^*\+ ' . a:link
|
||
else
|
||
let newlink = '#tn=' . a:link
|
||
endif
|
||
|
||
let newlink = substitute(newlink,' ','\\ ','g')
|
||
let g:newlink = newlink
|
||
silent! exec 'Utl o '. newlink . ' split'
|
||
|
||
if line('.') != savecursor[1]
|
||
break
|
||
endif
|
||
endfor
|
||
endfunction
|
||
function! EditLink()
|
||
"is this here: and is this there:
|
||
let thislink = OrgGetLink()
|
||
|
||
let link = input('Link: ', thislink.link)
|
||
let desc = input('Description: ', thislink.desc)
|
||
|
||
if thislink.link !=# ''
|
||
"delete existing hyperlink
|
||
call search('\[\[','b',line('.'))
|
||
normal v/]]/e
|
||
xx
|
||
endif
|
||
|
||
silent exec 'normal i[[' . link . ']' . (desc ># '' ? '[' . desc . ']' : '') . ']'
|
||
echo ''
|
||
endfunction
|
||
function! OrgMouseDate()
|
||
let @x=''
|
||
let date=''
|
||
let save_cursor = getpos(".")
|
||
let found = ''
|
||
silent! let date = GetDateAtCursor()
|
||
call setpos('.',save_cursor)
|
||
let linkdict = OrgGetLink()
|
||
if date ># ''
|
||
let found='date'
|
||
let date = date[0:9]
|
||
elseif linkdict.link ># ''
|
||
let found= 'link'
|
||
else
|
||
call setpos(".",save_cursor)
|
||
" get area between colons, if any, in @x
|
||
normal T:vt:"xy
|
||
if (matchstr(@x,'\S\+') ># '') && (len(@x)<25)
|
||
let found = 'tag'
|
||
endif
|
||
endif
|
||
call setpos(".",save_cursor)
|
||
if found ==? 'date'
|
||
silent call s:RunCustom({'type':'agenda','agenda_date':date, 'agenda_duration':'d'})
|
||
"call OrgRunAgenda(date,1,'',date)
|
||
" go to 8th line in agenda buf
|
||
execute 8
|
||
elseif found ==? 'link'
|
||
call FollowLink(linkdict)
|
||
elseif found ==? 'tag'
|
||
"call OrgRunSearch('+'.@x)
|
||
silent call s:RunCustom({'type':'tags','spec':@x})
|
||
else
|
||
echo 'Nothing found to search for.'
|
||
endif
|
||
|
||
endfunction
|
||
function! s:SetColumnHead()
|
||
" NOT USED NOW, NEEDS to be redone since switch to using orgmode-style col
|
||
" specs
|
||
"let i = 0
|
||
"while i < len(w:v.org_colview_list)
|
||
" let result .= '|' . s:PrePad(w:v.org_colview_list[i] , w:v.org_colview_list[i+1]) . ' '
|
||
" let i += 2
|
||
"endwhile
|
||
"let g:org_ColumnHead = result[:-2]
|
||
endfunction
|
||
|
||
function! s:OrgSetColumnList(line_for_cols,...)
|
||
" call GetProperties making sure it gets inherited props (viz. COLUMNS)
|
||
let save_inherit_setting = s:include_inherited_props
|
||
let s:include_inherited_props = 1
|
||
try
|
||
let column_prop = s:GetProperties(a:line_for_cols,0)['COLUMNS']
|
||
finally
|
||
let s:include_inherited_props = save_inherit_setting
|
||
endtry
|
||
|
||
if (a:0 >= 1) && (a:1==0)
|
||
" use 0 for master head, i.e., columns for entire doc
|
||
let w:v.org_columns_master_heading = a:1
|
||
else
|
||
let w:v.org_columns_master_heading = s:OrgGetHead_l(a:line_for_cols)
|
||
endif
|
||
if (a:0 >= 2) && (a:2 ># '')
|
||
" use column spec that was passed in
|
||
let column_prop = a:2
|
||
else
|
||
let w:v.org_current_columns = column_prop
|
||
endif
|
||
|
||
let result = ''
|
||
let g:org_column_headers = ''
|
||
let i = 0
|
||
|
||
if column_prop ># ''
|
||
let w:v.org_colview_list=split(column_prop,' ')
|
||
else
|
||
let w:v.org_colview_list=[]
|
||
endif
|
||
|
||
call s:SetColumnHeaders()
|
||
|
||
endfunction
|
||
function! s:SetColumnHeaders()
|
||
" build g:org_column_headers
|
||
let g:org_column_headers = ''
|
||
let w:v.org_column_item_head = ''
|
||
for item in (w:v.org_colview_list)
|
||
let [ fmt, field, hdr ] = matchlist(item,'%\(\d*\)\(\S\{-}[^({]*\)(*\([^ )]*\)')[1:3]
|
||
let fmt = (fmt ==# '') ? '%-' . g:org_columns_default_width . 's' : ('%-' . fmt . 's')
|
||
if field ==# 'ITEM'
|
||
let w:v.org_column_item_head = (hdr=='') ? 'ITEM' : hdr
|
||
continue
|
||
endif
|
||
let g:org_column_headers .= printf('|' . fmt, (hdr ==# '') ? field : hdr )
|
||
endfor
|
||
|
||
endfunction
|
||
function! s:GetFoldColumns(line)
|
||
let save_inherit_setting = s:include_inherited_props
|
||
let s:include_inherited_props = 1
|
||
try
|
||
let props = s:GetProperties(a:line,0)
|
||
finally
|
||
let s:include_inherited_props = save_inherit_setting
|
||
endtry
|
||
" build text string with column values
|
||
let result = ''
|
||
for item in (w:v.org_colview_list)
|
||
let [ fmt, field, hdr ] = matchlist(item,'%\(\d*\)\(\S\{-}[^({]*\)(*\(\S*\))*')[1:3]
|
||
if field ==# 'ITEM' | continue | endif
|
||
let fldtext = get(props,toupper(field),'')
|
||
let fmt = (fmt ==# '') ? g:org_columns_default_width : fmt
|
||
" truncate text if too long
|
||
"let fldtext = (len(fldtext)<=fmt) ? fldtext : (fldtext[:fmt-3] . '..')
|
||
let fldtext = (strchars(fldtext)<=fmt) ? fldtext : (fldtext[:fmt-3] . '..')
|
||
let result .= printf( '|%-'.fmt.'s', fldtext,'')
|
||
endfor
|
||
|
||
return result
|
||
|
||
endfunction
|
||
function! ToggleColumnView(master_head,col_spec)
|
||
|
||
if w:v.columnview
|
||
let winnum = bufwinnr('ColHeadBuffer')
|
||
if winnum > 0
|
||
execute "bw!" . bufnr('ColHeadBuffer')
|
||
endif
|
||
let w:v.columnview = 0
|
||
else
|
||
call s:OrgSetColumnList(line('.'),a:master_head,a:col_spec)
|
||
call s:ColHeadWindow(w:v.org_column_item_head)
|
||
let w:v.columnview = 1
|
||
endif
|
||
endfunction
|
||
function! <SID>ColumnStatusLine()
|
||
if exists('g:org_column_headers')
|
||
let part2 = s:PrePad(g:org_column_headers, winwidth(0)-13)
|
||
|
||
return ' ' . w:v.org_column_item_head . part2
|
||
endif
|
||
endfunction
|
||
function! s:AdjustItemLen()
|
||
" called on VimResized event, adjusts length of heading when folded
|
||
if &filetype != 'org'
|
||
return
|
||
endif
|
||
|
||
if !exists('w:v.columnview')
|
||
let w:v={'columnview':0, 'org_item_len':100, 'org_colview_list':[],'org_current_columns':'','org_column_item_head':''}
|
||
endif
|
||
let i = 1
|
||
let w:v.total_columns_width = 3
|
||
let colspec = split(w:v.org_current_columns, ' ')
|
||
|
||
for item in colspec
|
||
let [ flen, field ] = matchlist(item,'%\(\d*\)\(\S\{-}[^({]*\)')[1:2]
|
||
if field == 'ITEM' | continue | endif
|
||
let w:v.total_columns_width += (flen > 0) ? flen : g:org_columns_default_width
|
||
endfor
|
||
|
||
let w:v.org_item_len = winwidth(0) - 10 - ((w:v.columnview==1) ? w:v.total_columns_width : 0)
|
||
endfunction
|
||
au VimResized * :call s:ResizedWin()
|
||
function! s:ResizedWin()
|
||
let curwin = winnr()
|
||
""avoid using 'windo' b/c it screws up colheadbuffer's 0 height
|
||
for i in range(1,winnr('$'))
|
||
if getbufvar(winbufnr(i),'&filetype') == 'org'
|
||
exec i . 'wincmd w'
|
||
call s:AdjustItemLen()
|
||
endif
|
||
endfor
|
||
exec curwin . 'wincmd w'
|
||
endfunction
|
||
|
||
function! <SID>CalendarChoice(day, month, year, week, dir)
|
||
let g:agenda_startdate = a:year.'-' . s:Pre0(a:month).'-'.s:Pre0(a:day)
|
||
"call OrgRunAgenda(g:agenda_startdate, g:org_agenda_days,g:org_search_spec)
|
||
silent call s:RunCustom({'type':'agenda','agenda_date':g:agenda_startdate, 'agenda_duration':'d', 'spec':g:org_search_spec})
|
||
endfunction
|
||
function! <SID>CalendarInsertDate(day, month, year, week, dir)
|
||
if (a:year > 0) && (a:month>0) && (a:day>0)
|
||
let g:cal_list=[a:year,a:month,a:day]
|
||
endif
|
||
|
||
"call confirm('got here')
|
||
endfunction
|
||
function! OrgFunc(func,...)
|
||
"not working, itnended to be general way to
|
||
" call script-local functions
|
||
let myfunc = function('<SNR>'.s:SID().'_'.a:func)
|
||
if a:000 > 0
|
||
let myargs = split(a:000,',')
|
||
else
|
||
let myargs = ''
|
||
endif
|
||
endfunction
|
||
|
||
function! s:MyPopup()
|
||
call feedkeys("i\<c-x>\<c-u>")
|
||
endfunction
|
||
|
||
let g:calendar_action = '<SNR>' . s:SID() .'_CalendarChoice'
|
||
let b:v.ColorList=['purple', 'green', 'white', 'black','blue','red','orange','green']
|
||
function! s:CompleteOrg(findstart, base)
|
||
if a:findstart
|
||
" locate the start of the word
|
||
let line = getline('.')
|
||
let start = col('.') - 1
|
||
while (start > 0) && (line[start - 1] =~ '\a')
|
||
let start -= 1
|
||
endwhile
|
||
return start
|
||
else
|
||
let prop = matchstr(getline(line(".")),'^\s*:\zs\s*\S\+\s*\ze:')
|
||
" find months matching with "a:base"
|
||
let res = []
|
||
execute "let proplist = b:v." . prop . 'List'
|
||
"for m in split("Jan Feb Mar Apr May Jun Jul Aug Sep Oct Nov Dec")
|
||
for m in proplist
|
||
if m =~ '^' . a:base
|
||
call add(res, m)
|
||
endif
|
||
endfor
|
||
return res
|
||
endif
|
||
endfunction
|
||
"set completefunc=CompleteOrg
|
||
|
||
|
||
function! OrgFoldText(...)
|
||
" Create string used for folded text blocks
|
||
if a:0 == 1
|
||
let l:line = getline(line("."))
|
||
let foldstart = line(".")
|
||
else
|
||
let l:line = getline(v:foldstart)
|
||
let foldstart = v:foldstart
|
||
endif
|
||
let origline = l:line
|
||
let l:nextline = getline(foldstart + 1)
|
||
let myind = s:Ind(foldstart)
|
||
|
||
"let level_highlight = hlID(b:v.foldcolors[myind])
|
||
let level_highlight = hlID('OL' . (myind-1) . 'Folded')
|
||
|
||
" get rid of header prefix
|
||
let l:line = substitute(l:line,'^\*\+\s*','','g')
|
||
let l:line = repeat(' ', s:Starcount(foldstart)+1) . l:line
|
||
let line_count = v:foldend - v:foldstart
|
||
|
||
if l:line =~ b:v.drawerMatch
|
||
"let level_highlight = hlID('Title')
|
||
let level_highlight = hlID('Org_Drawer_Folded')
|
||
let l:line = repeat(' ', strchars(matchstr(l:line,'^ *'))-1)
|
||
\ . matchstr(l:line,'\S.*$')
|
||
let line_count = line_count - 1
|
||
elseif l:line[0] ==? '#'
|
||
let level_highlight = hlID('VisualNOS')
|
||
elseif w:v.columnview==1
|
||
let mytrim = w:v.org_item_len
|
||
let line = line[:mytrim]
|
||
else
|
||
let mytrim = w:v.org_item_len
|
||
let line = line[:mytrim]
|
||
endif
|
||
if exists('w:sparse_on') && w:sparse_on && (a:0 == 0)
|
||
let b:v.signstring= s:GetPlacedSignsString(bufnr("%"))
|
||
if match(b:v.signstring,'line='.v:foldstart.'\s\sid=\d\+\s\sname=fbegin') >=0
|
||
"if index(b:v.sparse_list,v:foldstart) > -1 "v:foldstart == 10
|
||
let l:line = '* * * * * * * * * * * ' . (v:foldend - v:foldstart) . ' lines skipped here * * * * * * *'
|
||
let l:line .= repeat(' ', winwidth(0)-strchars(l:line)-28) . 'SPARSETREE SKIP >>'
|
||
let level_highlight = hlID('TabLineFill')
|
||
endif
|
||
endif
|
||
if g:org_show_fold_dots
|
||
let l:line .= '...'
|
||
endif
|
||
let offset = &fdc + 5*(&number) + (w:v.columnview ? 7 : 1)
|
||
if w:v.columnview && (origline =~ b:v.headMatch)
|
||
if (w:v.org_columns_master_heading == 0) || s:HasAncestorHeadOf(foldstart,w:v.org_columns_master_heading)
|
||
let l:line .= s:PrePad(s:GetFoldColumns(foldstart), winwidth(0)-strchars(l:line) - offset)
|
||
else
|
||
let offset -= 6
|
||
endif
|
||
endif
|
||
if a:0 && (foldclosed(line('.')) > 0)
|
||
let l:line .= s:PrePad("("
|
||
\ . s:PrePad( (foldclosedend(line('.'))-foldclosed(line('.'))) . ")",5),
|
||
\ winwidth(0)-strchars(l:line) - offset)
|
||
elseif (g:org_show_fold_lines ) || (l:line =~ b:v.drawerMatch)
|
||
let offset = (w:v.columnview && l:line =~ b:v.drawerMatch) ? offset - 6 : offset
|
||
let l:line .= s:PrePad("|" . s:PrePad( line_count . "|",5),
|
||
\ winwidth(0)-strchars(l:line) - offset)
|
||
endif
|
||
if exists('v:foldhighlight')
|
||
if get(b:v.heading_marks_dict, v:foldstart) == 1
|
||
let v:foldhighlight = hlID('CursorLine')
|
||
elseif foldstart == b:v.chosen_agenda_heading
|
||
let v:foldhighlight = hlID('Org_Chosen_Agenda_Heading' . (myind>6 ? '' : myind-1))
|
||
else
|
||
let v:foldhighlight = level_highlight
|
||
endif
|
||
if exists('v:todohighlight')
|
||
if matchstr(origline, b:v.todoMatch) ># ''
|
||
let this_todo = matchstr(origline, '^\*\+ \zs\S*')
|
||
if hlID(this_todo) > 55 " > 55 avoids built-in todo group
|
||
let v:todohighlight = hlID(this_todo)
|
||
else
|
||
let v:todohighlight = ('* ' . this_todo =~ b:v.todoDoneMatch) ? hlID('DONETODO') : hlID('NOTDONETODO')
|
||
endif
|
||
else
|
||
let v:todohighlight=0
|
||
endif
|
||
endif
|
||
endif
|
||
return l:line
|
||
endfunction
|
||
function! s:MySort(comppattern) range
|
||
let b:v.sortcompare = a:comppattern
|
||
let b:v.complist = ['\s*\S\+','\s*\S\+\s\+\zs\S\+','\s*\(\S\+\s\+\)\{2}\zs\S\+'
|
||
\ , '\s*\(\S\+\s\+\)\{3}\zs\S\+'
|
||
\ , '\s*\(\S\+\s\+\)\{4}\zs\S\+'
|
||
\ , '\s*\(\S\+\s\+\)\{5}\zs\S\+'
|
||
\ , '\s*\(\S\+\s\+\)\{6}\zs\S\+']
|
||
let mylines = getline(a:firstline, a:lastline)
|
||
let mylines = sort(mylines,"s:BCompare")
|
||
call setline(a:firstline, mylines)
|
||
unlet b:v.sortcompare
|
||
unlet b:v.complist
|
||
endfunction
|
||
|
||
function! s:BCompare(i1,i2)
|
||
if !exists('b:v.sortcompare')
|
||
echo 'b:v.sortcompare is not defined'
|
||
return
|
||
endif
|
||
let i = 0
|
||
|
||
while i < len(b:v.sortcompare)
|
||
" prefix an item by 'n' if you want numeric sorting
|
||
if (i < len(b:v.sortcompare) - 1) && (b:v.sortcompare[i] ==? 'n')
|
||
let i = i + 1
|
||
let m1 = str2nr(matchstr(a:i1,b:v.complist[b:v.sortcompare[i]-1]))
|
||
let m2 = str2nr(matchstr(a:i2,b:v.complist[b:v.sortcompare[i]-1]))
|
||
else
|
||
let m1 = matchstr(a:i1,b:v.complist[b:v.sortcompare[i]-1])
|
||
let m2 = matchstr(a:i2,b:v.complist[b:v.sortcompare[i]-1])
|
||
endif
|
||
if m1 == m2
|
||
if i == len(b:v.sortcompare) - 1
|
||
return 0
|
||
else
|
||
let i += 1
|
||
continue
|
||
endif
|
||
elseif m1 > m2
|
||
return 1
|
||
else
|
||
return -1
|
||
endif
|
||
endwhile
|
||
endfunction
|
||
|
||
function! s:OrgShowMatch(cycleflag)
|
||
"wincmd k
|
||
" first, make sure agenda buffer has same heading pattern
|
||
" and todo list as main buffer
|
||
call s:GotoMainWindow()
|
||
let l:headMatch = b:v.headMatch
|
||
let l:todoitems = b:v.todoitems
|
||
"wincmd j
|
||
call s:GotoAgendaWindow()
|
||
let b:v.headMatch = l:headMatch
|
||
let b:v.todoitems = l:todoitems
|
||
if a:cycleflag
|
||
call OrgSequenceTodo(line("."))
|
||
endif
|
||
"let g:showndx = line(".")-1
|
||
if getline(line(".")) =~ '^\d\+'
|
||
let g:showndx = matchlist(getline(line(".")),'^\d\+')[0]
|
||
execute "let b:v.sparse_list = [" . g:showndx . ']'
|
||
endif
|
||
"wincmd k
|
||
call s:GotoMainWindow()
|
||
call OrgExpandWithoutText(1)
|
||
execute g:showndx
|
||
"execute g:alines[g:showndx]
|
||
normal zv
|
||
if a:cycleflag
|
||
call OrgSequenceTodo(line("."))
|
||
endif
|
||
if getline(line(".")) =~ b:v.headMatch
|
||
call OrgBodyTextOperation(line("."),s:OrgNextHead(),'collapse')
|
||
endif
|
||
"wincmd j
|
||
call s:GotoAgendaWindow()
|
||
endfunction
|
||
command! MySynch call <SID>OrgShowMatch(0)
|
||
command! MySynchCycle call <SID>OrgShowMatch(1)
|
||
command! MyAgendaToBuf call <SID>OrgAgendaToBufTest()
|
||
command! AgendaMoveToBuf call s:OrgAgendaToBuf()
|
||
|
||
command! -range CodeEval :call <SID>CodeEval
|
||
command! -buffer -nargs=* Agenda :call OrgAgendaCommand(<f-args>)
|
||
function! CodeEval() range
|
||
|
||
endfunction
|
||
|
||
function! OrgAgendaCommand(...)
|
||
if exists('a:1')
|
||
let mydate = a:1
|
||
else
|
||
let mydate = s:Today()
|
||
endif
|
||
if exists('a:2')
|
||
let viewdays = a:2
|
||
else
|
||
let viewdays = 'w'
|
||
endif
|
||
if exists('a:3')
|
||
let search_spec = a:3
|
||
else
|
||
let search_spec = ''
|
||
endif
|
||
if mydate =~ '\d\d\d\d-\d\d-\d\d'
|
||
call OrgRunAgenda(mydate,viewdays,search_spec)
|
||
else
|
||
call OrgRunAgenda(DateCueResult(mydate,s:Today()),viewdays,search_spec)
|
||
endif
|
||
endfunction
|
||
|
||
function! s:OrgAgendaToBufTest()
|
||
" this loads unfolded buffer into same window as Agenda
|
||
if getline(line(".")) =~ '^\d\+'
|
||
let thisline = getline(line('.'))
|
||
let g:tofile = s:filedict[str2nr(matchstr(thisline, '^\d\d\d'))]
|
||
let g:showndx = str2nr(matchstr(thisline,'^\d\d\d\zs\d*'))
|
||
"let g:showndx = matchlist(getline(line(".")),'^\d\+')[0]
|
||
"let g:tofile = matchlist(getline(line(".")),'^\d\+\s*\(\S\+\)')[1]
|
||
endif
|
||
let cur_buf = bufnr("%")
|
||
"let g:org_folds=0
|
||
let newbuf = bufnr(g:tofile)
|
||
execute "b"newbuf
|
||
set ft=org
|
||
execute g:showndx
|
||
normal zv
|
||
"let g:org_folds=1
|
||
endfunction
|
||
function! s:OrgAgendaToBuf()
|
||
let win = bufwinnr('Calendar')
|
||
|
||
if win >= 0
|
||
execute win . 'wincmd w'
|
||
wincmd c
|
||
execute bufwinnr('Agenda').'wincmd w'
|
||
endif
|
||
|
||
if getline(line(".")) =~ '^\d\+'
|
||
let thisline = getline(line('.'))
|
||
let g:tofile = s:filedict[str2nr(matchstr(thisline, '^\d\d\d'))]
|
||
let g:showndx = str2nr(matchstr(thisline,'^\d\d\d\zs\d*'))
|
||
endif
|
||
let ag_line = line(".")
|
||
let ag_height = winheight(0)
|
||
let cur_buf = bufnr("%") " should be Agenda
|
||
close!
|
||
call org#LocateFile(g:tofile )
|
||
|
||
AAgenda
|
||
|
||
call org#LocateFile(g:tofile )
|
||
"if &fdm != 'expr'
|
||
" set fdm=expr
|
||
"endif
|
||
set foldlevel=1
|
||
let newhead = matchstr(s:GetPlacedSignsString(bufnr("%")),'line=\zs\d\+\ze\s\+id=' . g:showndx)
|
||
let newhead = s:OrgGetHead_l(newhead)
|
||
execute newhead
|
||
|
||
normal! zv
|
||
if getline(line('.')) =~ b:v.headMatch
|
||
"restrict to headings only
|
||
call s:OrgExpandSubtree(line('.'),0)
|
||
"call s:OrgExpandSubtree(g:showndx,0)
|
||
endif
|
||
|
||
let b:v.chosen_agenda_heading = s:OrgGetHead()
|
||
call clearmatches()
|
||
let headlevel = s:Ind(b:v.chosen_agenda_heading)
|
||
let headlevel = (headlevel > 6) ? '' : headlevel-1
|
||
call matchadd('Org_Chosen_Agenda_Heading' . headlevel,'\%' . b:v.chosen_agenda_heading .'l')
|
||
|
||
execute bufwinnr('Agenda').'wincmd w'
|
||
|
||
execute ag_line
|
||
resize
|
||
execute "resize " . ag_height
|
||
|
||
if win >= 0
|
||
Calendar
|
||
execute 1
|
||
call org#LocateFile('__Agenda__')
|
||
endif
|
||
endfunction
|
||
|
||
function! s:OrgSource()
|
||
unlet g:org_loaded
|
||
source $VIM/vimfiles/ftplugin/org.vim
|
||
endfunction
|
||
|
||
function! s:OrgSetLevel(startlevel, endlevel)
|
||
"call OrgExpandWithoutText(a:endlevel)
|
||
call s:OrgExpandLevelText(a:startlevel, a:endlevel)
|
||
endfunction
|
||
|
||
function! s:Starcount(line)
|
||
" used to get number of stars for a heading
|
||
return (len(matchstr(getline(a:line),'^\**\s'))-1)
|
||
endfunction
|
||
|
||
function! s:GotoAgendaWindow()
|
||
"wincmd b
|
||
silent execute "b __Agenda__"
|
||
endfunction
|
||
|
||
function! s:GotoMainWindow()
|
||
wincmd t
|
||
endfunction
|
||
|
||
function! s:Ind(line)
|
||
" used to get level of a heading (todo : rename this function)
|
||
"return 1 + (len(matchstr(getline(a:line),'^\**\s'))-1)/b:v.levelstars
|
||
return 2 + (len(matchstr(getline(a:line),'^\**\s'))-2)/b:v.levelstars
|
||
|
||
endfunction
|
||
|
||
function! s:DoAllTextFold(line)
|
||
"let d = inputdialog('in fullfold')
|
||
if s:IsText(a:line+1) == 0
|
||
return
|
||
endif
|
||
while ((s:NextVisibleHead(a:line) != foldclosedend(a:line) + 1)
|
||
\ && (foldclosedend(a:line) <= line("$"))
|
||
\ && (s:NextVisibleHead(a:line) != 0)
|
||
\ && (org#fold#level(a:line) =~ '>'))
|
||
\ || (foldclosedend(a:line) < 0)
|
||
\ || ((s:NextVisibleHead(a:line) == 0) && (s:OrgSubtreeLastLine() == line('$')) && (foldclosedend(a:line)!=line('$')))
|
||
call OrgDoSingleFold(a:line)
|
||
endwhile
|
||
endfunction
|
||
|
||
function! OrgDoSingleFold(line)
|
||
if (foldclosed(a:line) == -1) "&& (getline(a:line+1) !~ b:v.headMatch)
|
||
if (getline(a:line+1) !~ b:v.headMatch) || (s:Ind(a:line+1) > s:Ind(a:line))
|
||
while (foldclosed(a:line) == -1) && (a:line != line('$'))
|
||
normal! zc
|
||
endwhile
|
||
endif
|
||
"elseif (foldclosed(a:line) < a:line)
|
||
" do nothing, line is not visible
|
||
else
|
||
let cur_end = foldclosedend(a:line)
|
||
" I know runaway can happen if at last heading in document,
|
||
" not sure where else
|
||
let runaway_count = 0
|
||
if (cur_end >= line("$")) "|| (org#fold#level(cur_end+1) ==? '<0')
|
||
return
|
||
endif
|
||
if getline(cur_end+1) =~ b:v.drawerMatch
|
||
"while (foldclosedend(a:line) == cur_end) && (runaway_count < 10)
|
||
while (foldclosedend(a:line) == cur_end) && (cur_end != line("$"))
|
||
let runaway_count += 1
|
||
normal! zc
|
||
endwhile
|
||
elseif getline(cur_end+1) !~ b:v.headMatch
|
||
"while (foldclosedend(a:line) == cur_end) && (runaway_count < 10)
|
||
while (foldclosedend(a:line) == cur_end) && (cur_end <= line("$"))
|
||
let runaway_count += 1
|
||
normal! zc
|
||
endwhile
|
||
elseif (getline(cur_end+1) =~ b:v.headMatch) && (s:Ind(cur_end+1) > s:Ind(a:line))
|
||
while (foldclosedend(a:line) == cur_end) && (cur_end != line("$"))
|
||
" let runaway_count += 1
|
||
normal! zc
|
||
endwhile
|
||
endif
|
||
endif
|
||
endfunction
|
||
|
||
|
||
function! s:AlignSection(regex,skip,extra) range
|
||
" skip is first part of regex, 'regex' is part to match
|
||
" they must work together so that 'skip.regex' is matched
|
||
" and the point where they connect is where space is inserted
|
||
let extra = a:extra
|
||
let sep = empty(a:regex) ? '=' : a:regex
|
||
let minst = 999
|
||
let maxst = 0
|
||
let b:v.stposd = {}
|
||
let section = getline(a:firstline, a:lastline)
|
||
for line in section
|
||
let stpos = matchend(line,a:skip)
|
||
let b:v.stposd[index(section,line)]=stpos
|
||
if maxst < stpos
|
||
let maxst = stpos
|
||
endif
|
||
let stpos = len(matchstr(matchstr(line,a:skip),'\s*$'))
|
||
if minst > stpos
|
||
let minst = stpos
|
||
endif
|
||
endfor
|
||
call map(section, 's:AlignLine(v:val, sep, a:skip, minst, maxst - matchend(v:val,a:skip), extra)')
|
||
call setline(a:firstline, section)
|
||
endfunction
|
||
|
||
function! s:AlignLine(line, sep, skip, maxpos, offset, extra)
|
||
let b:v.m = matchlist(a:line, '\(' .a:skip . '\)\('.a:sep.'.*\)')
|
||
if empty(b:v.m)
|
||
return a:line
|
||
endif
|
||
let spaces = repeat(' ', a:offset + a:extra)
|
||
exec 'return b:v.m[1][:-' . a:maxpos .'] . spaces . b:v.m[3]'
|
||
endfunction
|
||
function! s:AlignSectionR(regex,skip,extra) range
|
||
let extra = a:extra
|
||
let sep = empty(a:regex) ? '=' : a:regex
|
||
let minst = 999
|
||
let maxpos = 0
|
||
let maxst = 0
|
||
let b:v.stposd = {}
|
||
let section = getline(a:firstline, a:lastline)
|
||
for line in section
|
||
execute 'let pos = matchend(line, a:skip ." *".sep)'
|
||
if maxpos < pos
|
||
let maxpos = pos
|
||
endif
|
||
let stpos = len(matchstr(matchstr(line,a:skip),'\s*$'))
|
||
if minst > stpos
|
||
let minst = stpos
|
||
endif
|
||
endfor
|
||
call map(section, 's:AlignLine(v:val, sep, a:skip, minst, maxpos - matchend(v:val,a:skip.sep) , extra)')
|
||
call setline(a:firstline, section)
|
||
endfunction
|
||
function! s:ColHeadWindow(itemhead,...)
|
||
if (a:0 >= 1) && (a:1 == 0)
|
||
if bufnr('ColHeadBuffer') > -1
|
||
bw ColHeadBuffer
|
||
endif
|
||
return
|
||
endif
|
||
|
||
au! BufEnter ColHeadBuffer
|
||
"let s:AgendaBufferName = 'ColHeadBuffer'
|
||
"call s:AgendaBufferOpen(1)
|
||
"let s:AgendaBufferName = '__Agenda__'
|
||
1split ColHeadBuffer
|
||
call s:ScratchBufSetup()
|
||
|
||
execute "setlocal statusline=%#OrgColumnHeadings#%{<SNR>" . s:SID() . '_ColumnStatusLine()}'
|
||
set winfixheight
|
||
set winminheight=0
|
||
let w:v = {'org_column_item_head': a:itemhead}
|
||
|
||
wincmd j
|
||
" make lower window as big as possible to shrink
|
||
" ColHeadWindow to zero height
|
||
let curheight = winheight(0)
|
||
resize 100
|
||
if bufwinnr('Agenda') > 0
|
||
execute "resize " . curheight
|
||
endif
|
||
au BufEnter ColHeadBuffer call s:ColHeadBufferEnter()
|
||
endfunction
|
||
|
||
function! s:ColHeadBufferEnter()
|
||
"prevents user from entering this buffer
|
||
"wincmd j
|
||
endfunction
|
||
" AgendaBufferOpen
|
||
" Open the scratch buffer
|
||
function! s:AgendaBufferOpen(new_win)
|
||
let save_sb = &splitbelow
|
||
let save_sr = &splitright
|
||
let split_win = a:new_win
|
||
let t = g:org_agenda_window_position
|
||
if t == 'top'
|
||
let g:vsplit=0
|
||
set nosplitbelow
|
||
elseif t == 'bottom'
|
||
let g:vsplit = 0
|
||
set splitbelow
|
||
elseif t == 'left'
|
||
let g:vsplit = 1
|
||
set nosplitright
|
||
elseif t == 'right'
|
||
let g:vsplit = 1
|
||
set splitright
|
||
endif
|
||
" If the current buffer is modified then open the scratch buffer in a new
|
||
" window
|
||
"if !split_win && &modified
|
||
" let split_win = 1
|
||
"endif
|
||
|
||
" Check whether the scratch buffer is already created
|
||
let scr_bufnum = bufnr(s:AgendaBufferName)
|
||
"if scr_bufnum == -1
|
||
if scr_bufnum == -1
|
||
" open a new scratch buffer
|
||
if split_win
|
||
if exists('g:vsplit') && (g:vsplit==1)
|
||
exe "vsplit __Agenda__"
|
||
else
|
||
exe "split __Agenda__"
|
||
endif
|
||
else
|
||
exe "edit " . s:AgendaBufferName
|
||
endif
|
||
else
|
||
" Agenda buffer is already created. Check whether it is open
|
||
" in one of the windows
|
||
let scr_winnum = bufwinnr(scr_bufnum)
|
||
if scr_winnum != -1
|
||
" Jump to the window which has the scratch buffer if we are not
|
||
" already in that window
|
||
if winnr() != scr_winnum
|
||
exe scr_winnum . "wincmd w"
|
||
endif
|
||
else
|
||
" open a window and put existing Agenda in
|
||
if split_win
|
||
if exists('g:vsplit') && (g:vsplit==1)
|
||
exe "vsplit +buffer __Agenda__"
|
||
else
|
||
exe "split +buffer __Agenda__"
|
||
endif
|
||
|
||
else
|
||
exe "buffer " . scr_bufnum
|
||
endif
|
||
endif
|
||
endif
|
||
let &splitbelow = save_sb
|
||
let &splitright = save_sr
|
||
endfunction
|
||
|
||
command! EditAgendaFiles :call <SID>EditAgendaFiles()
|
||
command! OrgCycleAgendaForward :call <SID>CycleAgendaFiles('forward')
|
||
command! OrgCycleAgendaBackward :call <SID>CycleAgendaFiles('backward')
|
||
command! OrgChooseOrgBuffer :call <SID>OrgGotoChosenFile()
|
||
function! s:EditAgendaFiles()
|
||
call org#SaveLocation()
|
||
if !exists("g:agenda_files") || (g:agenda_files == [])
|
||
call s:CurfileAgenda()
|
||
endif
|
||
tabnew Edit_Agenda_Files
|
||
call s:ScratchBufSetup()
|
||
set buftype=acwrite
|
||
autocmd BufLeave <buffer> :call <SID>SaveAgendaFiles(0)
|
||
autocmd BufWriteCmd <buffer> :call <SID>SaveAgendaFiles(1)
|
||
command! W :call s:SaveAgendaFiles()
|
||
let msg = "These are your current agenda files:"
|
||
let msg2 = "Org files in your 'g:org_agenda_select_dirs' are below."
|
||
call setline(1,[msg])
|
||
call append(1, repeat('-',winwidth(0)-5))
|
||
call append("$",g:agenda_files + ['',''])
|
||
" change '\ ' to plain ' ' for current text in buffer
|
||
silent! execute '%s/\\ / /g'
|
||
let line = repeat('-',winwidth(0)-5)
|
||
call append("$",[line] + [msg2,"To add files to 'g:agenda_files' copy or move them ","to between the preceding lines and press :W to save (or :q to cancel):","",""])
|
||
for item in g:org_agenda_select_dirs
|
||
for wild_card in g:org_filename_wildcards
|
||
let dir_files = split(globpath(item,wild_card),"\n")
|
||
endfor
|
||
let heading_path = substitute(expand('~'),'\','\\\\','g')
|
||
call map(dir_files,'substitute( v:val , ' . "'" . heading_path . "'" . ' , ''~'','''')')
|
||
call append("$",dir_files)
|
||
endfor
|
||
endfunction
|
||
function! s:SaveAgendaFiles(save)
|
||
" saves edited file list into g:agenda_files
|
||
" yank files into @a
|
||
set nomodified
|
||
if a:save == 1
|
||
normal gg/^--
|
||
jV/^--
|
||
?^\S
|
||
"ay
|
||
let @a = substitute(@a,' ','\\ ','g')
|
||
if @a[0] != '-'
|
||
let g:agenda_files = split(@a,"\n")
|
||
else
|
||
let g:agenda_files=[]
|
||
endif
|
||
endif
|
||
:bw
|
||
delcommand W
|
||
call org#RestoreLocation()
|
||
endfunction
|
||
"function! s:CycleAgendaFiles()
|
||
" let i = 0
|
||
" while i < len(g:agenda_files)
|
||
" if g:agenda_files[i] =~ bufname('%') . '$'
|
||
" let i += 1
|
||
" break
|
||
" endif
|
||
" let i += 1
|
||
" endwhile
|
||
" if i >= len(g:agenda_files) - 1
|
||
" let i = 0
|
||
" endif
|
||
"
|
||
" call org#LocateFile(g:agenda_files[i])
|
||
"endfunction
|
||
function! s:ScratchBufSetup()
|
||
"setlocal buftype=nofile
|
||
setlocal buftype=nofile
|
||
setlocal bufhidden=hide
|
||
setlocal noswapfile
|
||
setlocal buflisted
|
||
setlocal fdc=1
|
||
endfunction
|
||
function! s:Emacs2PDF()
|
||
silent !"c:program files (x86)\emacs\emacs\bin\emacs.exe" -batch --visit=newtest3.org --funcall org-export-as-pdf
|
||
"silent !c:\sumatra.exe newtest3.org
|
||
endfunction
|
||
function! s:Today()
|
||
return strftime("%Y-%m-%d")
|
||
endfunction
|
||
|
||
function! OrgCustomSearchMenu()
|
||
if !exists('g:org_custom_searches') || empty(g:org_custom_searches)
|
||
echo "No custom searches defined."
|
||
else
|
||
echo " Press number to run custom search:"
|
||
echo " ----------------------------------"
|
||
let i = 1
|
||
for item in g:org_custom_searches
|
||
if type(item)==type({})
|
||
echo printf(" (%d) %-25s %10s", i, item.name, item.type )
|
||
elseif type(item) == type([])
|
||
echo printf(" (%d) %-25s %10s", i, item[0].name, item[0].type )
|
||
for each in item[1:]
|
||
"echo printf(" (%d) %-25s %10s", i, each.name, each.type )
|
||
echo printf(" + %-25s %10s", each.name, each.type )
|
||
endfor
|
||
endif
|
||
let i += 1
|
||
unlet item
|
||
endfor
|
||
echo " "
|
||
let key = nr2char(getchar())
|
||
let itemnum = str2nr(key)
|
||
if itemnum > 0 && itemnum <= len(g:org_custom_searches)
|
||
call s:RunCustom( itemnum - 1 )
|
||
else
|
||
echo 'No search was chosen.'
|
||
endif
|
||
endif
|
||
endfunction
|
||
|
||
function! s:AgendaBlockNum(line)
|
||
let block_num = 0
|
||
let sep_list = []
|
||
silent g/^========/call add(sep_list,line('.'))
|
||
call add(sep_list,line('$'))
|
||
for i in range(1,len(sep_list))
|
||
if a:line < sep_list[i-1]
|
||
let block_num = i
|
||
break
|
||
endif
|
||
endfor
|
||
return block_num
|
||
endfunction
|
||
|
||
function! OrgAgendaFilesDashboard()
|
||
echohl MoreMsg
|
||
echo ""
|
||
echo " ======================================"
|
||
echo " Press key for an agenda files command:"
|
||
echo " --------------------------------------"
|
||
echo " e Edit agenda files"
|
||
if (bufname('%') != '__Agenda__')
|
||
echo " t Current file to top of agenda file list"
|
||
echo " b Current file to bottom of agenda file list"
|
||
echo " r Remove Current file from agenda file list"
|
||
endif
|
||
echo " n or f Cycle to next file in agenda files"
|
||
echo " p or b Cycle to previous file in agenda files"
|
||
echo " c or g Choose agenda file to goto"
|
||
echo ""
|
||
echohl None
|
||
let key = nr2char(getchar())
|
||
redraw
|
||
if key =~? 'e'
|
||
EditAgendaFiles
|
||
elseif key =~? 't'
|
||
call <SID>CurrentToAgendaFiles('top')
|
||
elseif key =~? 'b'
|
||
call <SID>CurrentToAgendaFiles('bottom')
|
||
elseif key =~? 'r'
|
||
call <SID>CurrentRemoveFromAgendaFiles()
|
||
elseif key =~? 'n\|f'
|
||
call <SID>CycleAgendaFiles('forward')
|
||
elseif key =~? 'p\|b'
|
||
call <SID>CycleAgendaFiles('backward')
|
||
elseif key =~? 'c\|g'
|
||
call <SID>OrgGotoChosenFile()
|
||
else
|
||
echo "No option chosen."
|
||
endif
|
||
"endif
|
||
endfunction
|
||
function! OrgAgendaDashboard()
|
||
if (bufnr('__Agenda__') >= 0) && (bufwinnr('__Agenda__') == -1)
|
||
" move agenda to cur tab if it exists and is on a different tab
|
||
"let curtab = tabpagenr()
|
||
"" go to current agenda win and close it
|
||
"call org#LocateFile('__Agenda__')
|
||
"wincmd c
|
||
""back to start tab and open
|
||
"execute "tabnext ".curtab
|
||
"split
|
||
"winc j
|
||
"buffer __Agenda__
|
||
AAgenda
|
||
else
|
||
" show dashboard if there is no agenda buffer or it's
|
||
" already on this tab page
|
||
let restrict = 0
|
||
let saved_afiles = []
|
||
while 1
|
||
echohl MoreMsg
|
||
echo ""
|
||
echo " ================================"
|
||
echo " Press key for an agenda command:"
|
||
echo " --------------------------------"
|
||
echo " a Agenda for current week"
|
||
echo " t List of all TODO entries"
|
||
echo " m Match a TAGS/PROP/TODO query"
|
||
echo " L Timeline for current buffer"
|
||
"echo ' s Freeform regex search, not heading-metadata'
|
||
echo " "
|
||
echo " c Show custom search menu"
|
||
echo " "
|
||
echo " h Headline-metadata-based sparse tree search"
|
||
echo " f Freeform (i.e., regex) sparse tree search"
|
||
echo " < restrict to current buffer"
|
||
if restrict == 1
|
||
echo " Will restrict to current buffer. Press a key to choose search..."
|
||
endif
|
||
echo ""
|
||
echohl None
|
||
let key = nr2char(getchar())
|
||
redraw
|
||
if key == '<'
|
||
let restrict = 1
|
||
continue
|
||
else
|
||
break
|
||
endif
|
||
endwhile
|
||
if restrict == 1
|
||
let save_win = winnr()
|
||
for winnum in range(1,winnr('$'))
|
||
exec winnum . 'wincmd w'
|
||
if expand('%') =~ '\.org$'
|
||
let saved_afiles = copy(g:agenda_files)
|
||
let g:agenda_files = [expand('%:p')]
|
||
break
|
||
endif
|
||
endfor
|
||
exec save_win . 'wincmd w'
|
||
endif
|
||
try
|
||
if key =~ '[tTaALcCmM]' && bufnr('__Agenda__') >= 0
|
||
bwipeout __Agenda__
|
||
endif
|
||
if key ==? 't'
|
||
"silent execute "call OrgRunSearch('+ANY_TODO','agenda_todo')"
|
||
silent call s:RunCustom({'type':'tags-todo','spec':'+ANY_TODO'})
|
||
elseif key ==? 'a'
|
||
"if (g:org_search_spec ==# '')
|
||
"let g:org_search_spec = g:agenda_default_search_spec
|
||
"endif
|
||
"silent execute "call OrgRunAgenda(s:Today(),'w', g:org_agenda_default_search_spec)"
|
||
let today = s:Today()
|
||
silent call s:RunCustom({'type':'agenda','agenda_date':today, 'agenda_duration':'w'})
|
||
|
||
elseif key ==? 'L'
|
||
silent execute "call s:Timeline()"
|
||
elseif key ==? 'c'
|
||
execute "call OrgCustomSearchMenu()"
|
||
elseif key ==? 'm'
|
||
let mysearch = input("Enter search string: ")
|
||
silent call s:RunCustom({'type':'tags','spec':mysearch})
|
||
"silent execute "call OrgRunSearch(mysearch)"
|
||
elseif key ==? 'h'
|
||
let g:org_sparse_spec = input("Enter search string: ")
|
||
if bufname("%") ==? '__Agenda__'
|
||
:bd
|
||
endif
|
||
silent execute "call OrgRunSearch(g:org_sparse_spec,1)"
|
||
elseif key ==? 'f'
|
||
let g:org_sparse_spec = input("Enter search string: ")
|
||
if bufname("%") ==? '__Agenda__'
|
||
:bd
|
||
endif
|
||
silent call s:SparseTreeRun(g:org_sparse_spec)
|
||
endif
|
||
finally
|
||
if bufname('%') == '__Agenda__'
|
||
if !exists("b:v")
|
||
let b:v = {}
|
||
endif
|
||
let b:v.agenda_files = copy(g:agenda_files)
|
||
let @/=''
|
||
endif
|
||
if len(saved_afiles) > 0
|
||
let g:agenda_files = copy(saved_afiles)
|
||
endif
|
||
endtry
|
||
endif
|
||
endfunction
|
||
|
||
function! s:AgendaBufSyntax()
|
||
"called once whenever new agenda buffer is opened from regular or date agenda
|
||
"searches
|
||
call s:AgendaHighlight()
|
||
syntax clear
|
||
if has("conceal")
|
||
syntax match Locator '^\d\+' conceal containedin=AOL1,AOL2,AOL3,AOL4,AOL5,agenda_todo,agenda_done,agenda_scheduled,agenda_scheduled_previous
|
||
syntax match TimeGridSpace '^ \{8}\ze *\d\d:\d\d' conceal containedin=agenda_timegrid
|
||
"for agenda clocktable
|
||
syn region Org_Full_Link concealends matchgroup=linkends start='\[\[\(.\{-1,}\)]\[' end=']]'
|
||
endif
|
||
syntax match agenda_timegrid '^\s*\d\d:\d\d.\{14}--.*'
|
||
syntax match AOL1 '^\d\+\s\+\S\+\s\{-1,16}\*\{1} .*$'hs=s+23
|
||
syntax match AOL2 '^\d\+\s\+\S\+\s\{-1,16}\*\{2} .*$'hs=s+23
|
||
syntax match AOL3 '^\d\+\s\+\S\+\s\{-1,16}\*\{3} .*$'hs=s+23
|
||
syntax match AOL4 '^\d\+\s\+\S\+\s\{-1,16}\*\{4} .*$'hs=s+23
|
||
syntax match AOL5 '^\d\+\s\+\S\+\s\{-1,16}\*\{5} .*$'hs=s+23
|
||
|
||
let donepat_fragment = '\(' . join(keys(g:org_todos_done_dict),'\|') . '\)'
|
||
let donepat = ' \*\+ \zs' . donepat_fragment . ' '
|
||
exec "syntax match DONETODO /" . donepat . '/ containedin=AOL1,AOL2,AOL3,AOL4,AOL5'
|
||
let notdonepat_fragment = '\(' . join(keys(g:org_todos_notdone_dict),'\|') . '\)'
|
||
let notdonepat = ' \*\+ \zs' . notdonepat_fragment . ' '
|
||
exec "syntax match NOTDONETODO /" . notdonepat . '/ containedin=AOL1,AOL2,AOL3,AOL4,AOL5'
|
||
|
||
syntax match agenda_scheduled_previous '^\d\+\s\+\S\+\s\+Sched.\{-}:.*$'
|
||
syntax match agenda_scheduled '^\d\+\s\+\S\+\s\+Scheduled:.*$'
|
||
|
||
exec 'syntax match agenda_todo /^\d\+\s\+\S\+\s\+\(In\|Deadline\|(\d\).\{-}:.\{-}\* ' . notdonepat_fragment . ' .*$/ contains=Locator'
|
||
exec 'syntax match agenda_done /^\d\+\s\+\S\+\s\+\(In\|Deadline\|(\d\).\{-}:.\{-}\* ' . donepat_fragment . ' .*$/'
|
||
let daytextpat = '/^[^S]\S\+\s\+\d\{1,2}\s\S\+\s\d\d\d\d.*/'
|
||
let wkendtextpat = '/^S\S\+\s\+\d\{1,2}\s\S\+\s\d\d\d\d.*/'
|
||
exec 'syntax match agenda_date ' . daytextpat
|
||
|
||
exec 'syntax match agenda_weekenddate ' . wkendtextpat
|
||
syntax match agenda_omitted_days ' \[\. \..\{-}empty days omitted \]'
|
||
|
||
" call to set up syntax and highlights for custom todos in agenda
|
||
call s:OrgCustomTodoHighlights()
|
||
|
||
endfunction
|
||
function! s:AgendaHighlight()
|
||
if g:org_gray_agenda
|
||
hi link AOL1 NONE
|
||
hi link AOL2 NONE
|
||
hi link AOL3 NONE
|
||
hi link AOL4 NONE
|
||
hi link AOL5 NONE
|
||
hi Deadline guifg=lightred
|
||
hi Scheduled guifg=lightyellow
|
||
else
|
||
hi link AOL1 OL1
|
||
hi link AOL2 OL2
|
||
hi link AOL3 OL3
|
||
hi link AOL4 OL4
|
||
hi link AOL5 OL5
|
||
hi Deadline guifg=NONE
|
||
hi Scheduled guifg=NONE
|
||
endif
|
||
endfunction
|
||
|
||
function! OrgScreenLines() range
|
||
" returns lines as
|
||
" seen on screen, including folded text overlays
|
||
" Call with visual selection set, or will
|
||
" use last selection
|
||
let save_cursor = getpos('.')
|
||
let newline=0
|
||
let oldline=1
|
||
let mylines=[]
|
||
normal '>
|
||
let endline = line('.')
|
||
" go to first line of selection
|
||
normal '<
|
||
while (line('.') <= endline) && (newline != oldline)
|
||
let oldline=line('.')
|
||
let newline=oldline
|
||
call add(mylines,OrgFoldText(line('.')))
|
||
normal j
|
||
let newline=line('.')
|
||
endwhile
|
||
call setpos('.',save_cursor)
|
||
return mylines
|
||
endfunction
|
||
|
||
function! s:CurTodo(line)
|
||
let result = matchstr(getline(a:line),'.*\* \zs\S\+\ze ')
|
||
if index(b:v.todoitems,curtodo) == -1
|
||
let result = ''
|
||
endif
|
||
return result
|
||
endfunction
|
||
|
||
"autocmd CursorHold * call s:Timer()
|
||
function! s:Timer()
|
||
call feedkeys("f\e")
|
||
" K_IGNORE keycode does not work after version 7.2.025)
|
||
echo strftime("%c")
|
||
" there are numerous other keysequences that you can use
|
||
endfunction
|
||
|
||
autocmd BufNewFile __Agenda__ call s:ScratchBufSetup()
|
||
"autocmd BufWinEnter __Agenda__ call s:AgendaBufSyntax()
|
||
" Command to edit the scratch buffer in the current window
|
||
"command! -nargs=0 Agenda call s:AgendaBufferOpen(0)
|
||
" Command to open the scratch buffer in a new split window
|
||
command! -nargs=0 AAgenda call s:AgendaBufferOpen(1)
|
||
command! -nargs=0 EditAgenda call s:AgendaBufferOpen(0)
|
||
|
||
command! -nargs=0 OrgToPDF :call s:ExportToPDF()
|
||
command! -nargs=0 OrgToHTML :call s:ExportToHTML()
|
||
command! -nargs=0 OrgToAscii :call s:ExportToAscii()
|
||
command! -nargs=0 OrgToDocBook :call s:ExportToDocBook()
|
||
function! s:OrgHasEmacsVar()
|
||
let result = 1
|
||
if !exists('g:org_command_for_emacsclient')
|
||
let msg = "=============================================== \n"
|
||
\ . "You're trying to call out to Emacs but \n"
|
||
\ . "you haven't set an Emacs command variable. \n"
|
||
\ . "You should set this in your vimrc by including \n"
|
||
\ . "a line like: \n\n"
|
||
\ . " let g:org_command_for_emacsclient=[put command to start emacs here] \n\n"
|
||
\ . "See :h vimorg-emacs-setup for more info. \n\n"
|
||
\ . "The call you attempted to Emacs will now be aborted. \n"
|
||
\ . "Revise your vimrc and restart Vim to use this feature.\n"
|
||
\ . "==============================================\n"
|
||
\ . "Press <enter> to continue."
|
||
call input(msg)
|
||
let result = 0
|
||
endif
|
||
return result
|
||
endfunction
|
||
function! OrgGetClocktable(filelist, options)
|
||
let i = 0
|
||
let filelist = copy(a:filelist)
|
||
let filestr = ''
|
||
while i < len(filelist)
|
||
let filelist[i] = substitute(expand(filelist[i]),'\','/','g')
|
||
let filelist[i] = '"' . substitute(filelist[i],' ','\ ','g') . '" '
|
||
let filestr .= filelist[i]
|
||
let i += 1
|
||
endwhile
|
||
let clkfile = '~/cblock.org'
|
||
|
||
let mylist = ['#+BEGIN: clocktable :scope (' . filestr . ') ' . a:options ,'','#+END:','']
|
||
|
||
call writefile(mylist, expand(clkfile) )
|
||
|
||
let part1 = '(let ((org-confirm-babel-evaluate nil)(buf (find-file \' . s:cmd_line_quote_fix . '"' . clkfile . '\' . s:cmd_line_quote_fix . '"' . '))) (progn (org-dblock-update)(save-buffer buf)(kill-buffer buf)))'
|
||
let orgcmd = g:org_command_for_emacsclient . ' --eval ' . s:cmd_line_quote_fix . '"' . part1 . s:cmd_line_quote_fix . '"'
|
||
"redraw
|
||
"unsilent echo "Calculating in Emacs. . . "
|
||
if exists('*xolox#shell#execute')
|
||
silent call xolox#shell#execute(orgcmd, 1)
|
||
else
|
||
silent exe '!' . orgcmd
|
||
endif
|
||
|
||
return readfile(expand('~/cblock.org'))
|
||
endfunction
|
||
function! OrgEvalBlock()
|
||
let savecursor = getpos('.')
|
||
let save_showcmd = &showcmd | set noshowcmd
|
||
|
||
let start = search('^#+begin:','bnW','')
|
||
let prev_end = search('^#+end','bnW','')
|
||
let end = search('^#+end','nW','')
|
||
if (start == 0) || (end == 0) || ( ( prev_end > start ) && (prev_end < line('.') ) )
|
||
unsilent echo "You aren't in a dynamic block."
|
||
return
|
||
endif
|
||
|
||
exec start
|
||
let block_name = matchstr(getline(line('.')),'\c^#+BEGIN:\s*\zs\S\+')
|
||
|
||
if start < end - 1
|
||
exec (start+1) . ',' . (end-1) . 'delete'
|
||
endif
|
||
exec start
|
||
let line_mark = '@@@@@' . start . '@e@f@g@h'
|
||
exec 'normal o' . line_mark
|
||
|
||
silent write!
|
||
let this_file = substitute(expand("%:p"),'\','/','g')
|
||
let this_file = substitute(this_file,' ','\ ','g')
|
||
|
||
let part1 = '(let ((org-confirm-babel-evaluate nil)(buf (find-file \' . s:cmd_line_quote_fix . '"' . this_file . '\' . s:cmd_line_quote_fix . '"' . '))) (progn (search-forward \^"' . line_mark . '\^" )(forward-line -1)(org-dblock-update)(beginning-of-line)(set-mark (point))(re-search-forward \^"^#\\+END\^")(end-of-line)(write-region (mark) (point) \' . s:cmd_line_quote_fix . '"~/org-block.org\' . s:cmd_line_quote_fix . '")(set-buffer buf) (not-modified) (kill-this-buffer)))'
|
||
" line below was using org-narrow-to-block, which may use again
|
||
"let part1 = '(let ((org-confirm-babel-evaluate nil)(buf (find-file \' . s:cmd_line_quote_fix . '"' . this_file . '\' . s:cmd_line_quote_fix . '"' . '))) (progn (search-forward \^"' . line_mark . '\^" )(forward-line -1)(org-dblock-update)(org-narrow-to-block)(write-region (point-min) (point-max) \' . s:cmd_line_quote_fix . '"~/org-block.org\' . s:cmd_line_quote_fix . '")(set-buffer buf) (not-modified) (kill-this-buffer)))'
|
||
let orgcmd = g:org_command_for_emacsclient . ' --eval ' . s:cmd_line_quote_fix . '"' . part1 . s:cmd_line_quote_fix . '"'
|
||
redraw
|
||
unsilent echo "Calculating in Emacs. . . "
|
||
if exists('*xolox#shell#execute')
|
||
silent call xolox#shell#execute(orgcmd, 1)
|
||
else
|
||
silent exe '!' . orgcmd
|
||
endif
|
||
let g:orgcmd = orgcmd
|
||
exec start
|
||
normal 3ddk
|
||
silent exe 'read ~/org-block.org'
|
||
redraw
|
||
unsilent echo "Block is being evaluated in Emacs. . . Evaluation complete."
|
||
|
||
let &showcmd = save_showcmd
|
||
call setpos('.',savecursor)
|
||
"return readfile(expand('~/org-block.org'))
|
||
endfunction
|
||
|
||
function! s:OrgTableOptionList(A,L,P)
|
||
return keys(s:OrgTableEvalOptions())
|
||
endfunction
|
||
command! -buffer -nargs=? -complete=customlist,s:OrgTableOptions OrgTblEval :call OrgEvalTable(<f-args>)
|
||
function! s:OrgTableEvalOptions()
|
||
return { 'col_right':'org-table-move-column-right',
|
||
\ 'col_left': 'org-table-move-column-left',
|
||
\ 'col_delete': 'org-table-delete-column',
|
||
\ 'col_insert': 'org-table-insert-column',
|
||
\ 'row_down': 'org-table-move-row-down',
|
||
\ 'row_up': 'org-table-move-row-up',
|
||
\ 'row_delete': 'org-table-kill-row',
|
||
\ 'row_insert': 'org-table-insert-row',
|
||
\ 'row_sort_region_alpha': 'org-table-sort-lines nil ?a',
|
||
\ 'row_sort_region_numeric': 'org-table-sort-lines nil ?n',
|
||
\ 'row_sort_region_alpha_reverse': 'org-table-sort-lines nil ?A',
|
||
\ 'row_sort_region_numeric_reverse': 'org-table-sort-lines nil ?N',
|
||
\ 'row_hline_insert': 'org-table-insert-hline',
|
||
\ 'convert_region_to_table':'org-table-convert-region (point-min) (point-max)' }
|
||
endfunction
|
||
|
||
command! -buffer -nargs=0 OrgTableDashboard :call OrgTableDashboard()
|
||
function! OrgTableDashboard()
|
||
if s:OrgHasEmacsVar() == 0
|
||
return
|
||
endif
|
||
let save_more = &more | set nomore
|
||
let save_showcmd = &showcmd | set noshowcmd
|
||
" different dashboard for "in table" and "not in table"
|
||
" show export dashboard
|
||
if getline(line('.')) =~ '^\s*$'
|
||
let rows_cols = input("Create new table (enter rows, columns): ")
|
||
if rows_cols =~ '^\d\+\s*,\s*\d\+$'
|
||
let [rows,cols] = split(rows_cols,',')
|
||
call org#tbl#create(cols,rows)
|
||
elseif rows_cols =~ '^\d\+\s\+\d\+$'
|
||
let [rows,cols] = split(rows_cols,' ')
|
||
call org#tbl#create(cols,rows)
|
||
endif
|
||
return
|
||
endif
|
||
echohl MoreMsg
|
||
echo " --------------------------------"
|
||
echo " Press key for table operation:"
|
||
echo " --------------------------------"
|
||
if getline(line('.')) !~ b:v.tableMatch
|
||
let mydict = { 't' : 'convert_region_to_table'}
|
||
echo " [t] Create (t)able from current block"
|
||
else
|
||
let mydict = { 'l':'col_left', 'r':'col_right', 'e':'col_delete', 'o':'col_insert',
|
||
\ 'd':'row_down', 'u':'row_up', 'x':'row_delete',
|
||
\ 'i':'row_insert', 'a':'row_sort_region_alpha', 'A':'row_sort_region_alpha_reverse',
|
||
\ 'n':'row_sort_region_numeric', 'N':'row_sort_region_numeric', 'h':'row_hline_insert'
|
||
\ }
|
||
echo " COLUMN: [l] Move left [r] Move right [e] Delete [o] Insert"
|
||
echo " ROW: [d] Move down [u] Move up [x] Delete [i] Insert"
|
||
echo " "
|
||
echo " RowSort: [a] alpha(a-z) [A] alpha(z-a)"
|
||
echo " [n] numeric(1..9) [N] numeric(9-1)"
|
||
echo ""
|
||
echo " [h] insert horizontal line"
|
||
endif
|
||
echo " "
|
||
echohl None
|
||
let key = nr2char(getchar())
|
||
for item in keys(mydict)
|
||
if key == 't'
|
||
let thisline = getline(line('.'))
|
||
if thisline !~ '^\s*$'
|
||
let firstline = search('^\s*$','nb','') + 1
|
||
let lastline = search('^\s*$','n','') - 1
|
||
exec firstline . ',' . lastline . 'call OrgEvalTable(mydict[item])'
|
||
else
|
||
echo "You aren't in a block of text."
|
||
endif
|
||
break
|
||
elseif (key =~# item)
|
||
exec 'OrgTblEval ' . mydict[item]
|
||
break
|
||
endif
|
||
endfor
|
||
let &more = save_more
|
||
let &showcmd = save_showcmd
|
||
|
||
endfunction
|
||
|
||
function! OrgEvalTable(...) range
|
||
let options = s:OrgTableEvalOptions()
|
||
if a:0 == 1
|
||
let opt = a:1
|
||
let opt_cmd = '(' . options[opt] . ')'
|
||
else
|
||
let opt = 'just_eval'
|
||
let opt_cmd = ''
|
||
endif
|
||
let savecursor = getpos('.')
|
||
if a:firstline == a:lastline
|
||
" get start, end for whole table
|
||
"call search('^\s*[^|]','b','')
|
||
call search('^\(\s*|\)\@!','b','')
|
||
let start=line('.')
|
||
call search('^\(\s*|\)\@!','','')
|
||
let end=line('.')
|
||
else
|
||
let start=a:firstline
|
||
let end =a:lastline
|
||
endif
|
||
let line_offset = savecursor[1] - start + 1
|
||
"let line_offset = savecursor[1] - start
|
||
" find first line after table block and check for formulas
|
||
exe start . ',' . end . 'w! ~/org-tbl-block.org'
|
||
if opt != 'convert_region_to_table'
|
||
let part1 = '(let ((org-confirm-babel-evaluate nil)'
|
||
\ . '(buf (find-file \' . s:cmd_line_quote_fix . '"~/org-tbl-block.org\' . s:cmd_line_quote_fix . '"' . ')))'
|
||
\ . '(progn (beginning-of-line ' . line_offset . ')(forward-char ' . savecursor[2] .')'
|
||
\ . '(org-table-maybe-eval-formula)'
|
||
\ . ((opt=='just_eval') ? '' : '(unwind-protect ') . opt_cmd
|
||
\ . '(org-table-recalculate-buffer-tables)(save-buffer buf)(kill-buffer buf))))'
|
||
else
|
||
let part1 = '(let ((org-confirm-babel-evaluate nil)'
|
||
\ . '(buf (find-file \' . s:cmd_line_quote_fix . '"~/org-tbl-block.org\' . s:cmd_line_quote_fix . '"' . ')))'
|
||
\ . '(progn (beginning-of-line ' . line_offset . ')(forward-char ' . savecursor[2] .')'
|
||
\ . '(goto-char (point-min))(set-mark (point))(goto-char (point-max))' . opt_cmd
|
||
\ . '(save-buffer buf)(kill-buffer buf)))'
|
||
endif
|
||
let orgcmd = g:org_command_for_emacsclient . ' --eval ' . s:cmd_line_quote_fix . '"' . part1 . s:cmd_line_quote_fix . '"'
|
||
redraw
|
||
unsilent echo "Calculating in Emacs. . . "
|
||
|
||
let g:orgcmd = orgcmd
|
||
|
||
"if exists('*xolox#shell#execute')
|
||
" silent let myx = xolox#shell#execute(orgcmd . '| cat', 1)
|
||
"else
|
||
silent exe '!' . orgcmd
|
||
"endif
|
||
exe start .',' . end . 'read ~/org-tbl-block.org'
|
||
exe start . ',' . end . 'd'
|
||
redraw
|
||
unsilent echo "Calculating in Emacs. . . Calculations complete. "
|
||
"else
|
||
" unsilent echo "No #+TBLFM line at end of table, so no calculations necessary."
|
||
"endif
|
||
call setpos('.',savecursor)
|
||
endfunction
|
||
function! OrgEval()
|
||
if s:OrgHasEmacsVar() == 0
|
||
call confirm('VimOrganizer has not been configured to make calls to Emacs.'
|
||
\ . "\nPlease see :h vimorg-emacs-setup.")
|
||
return
|
||
endif
|
||
" check if we're in source block
|
||
let line = getline(line('.'))
|
||
let start = search('\c^#+begin_src','cbnW','')
|
||
let prev_end = search('\c^#+end_src','bnW','')
|
||
let end = search('\c^#+end_src','cnW','')
|
||
if (start > 0) && (end > start) && ( prev_end < start )
|
||
call OrgEvalSource()
|
||
return
|
||
endif
|
||
" check if we're in dynamic block
|
||
let start = search('^#+begin:','bnW','')
|
||
let prev_end = search('^#+end','bnW','')
|
||
let end = search('^#+end','nW','')
|
||
if (start > 0) && (end > start) && ( prev_end < start )
|
||
call OrgEvalBlock()
|
||
return
|
||
endif
|
||
"check if we're in a table
|
||
if line =~ '^\s*|.*|\s*$'
|
||
call OrgEvalTable()
|
||
return
|
||
endif
|
||
|
||
unsilent echo "No evaluation done. You must be in a table, or on an initial "
|
||
\ . "\nblock line that begins in col 0 with #+BEGIN . . ."
|
||
|
||
endfunction
|
||
|
||
function! OrgEvalSource()
|
||
let savecursor = getpos('.')
|
||
let start = search('^#+begin_src','cbnW','') - 1
|
||
let prev_end = search('^#+end_src','bnW','')
|
||
let end = search('^#+end_src','cnW','')
|
||
if (start == -1) || (end == 0) || ( ( prev_end > start ) && (prev_end < line('.') ) )
|
||
unsilent echo "You aren't in a code block."
|
||
return
|
||
endif
|
||
exec end
|
||
" include results if there is result block w/in a couple of lines
|
||
if ( search('^\s*#+results','nW','') - end ) <= 3
|
||
call search('^\s*#+results','','')
|
||
if getline(line('.')+1) =~ '#+BEGIN_RESULT'
|
||
call search('^#+END_RESULT')
|
||
normal j
|
||
else
|
||
" :'s used as linebegins w/first blank line as end of result block
|
||
call search('^\s*$','','')
|
||
endif
|
||
normal k
|
||
let end = line('.')
|
||
endif
|
||
exe start . ',' . end . 'w! ~/org-src-block.org'
|
||
let part1 = '(let ((org-confirm-babel-evaluate nil)) (progn (find-file \' . s:cmd_line_quote_fix . '"~/org-src-block.org\' . s:cmd_line_quote_fix . '"' . ')(org-babel-next-src-block)(org-babel-execute-src-block)(save-buffer)(kill-buffer)))'
|
||
let orgcmd = g:org_command_for_emacsclient . ' --eval ' . s:cmd_line_quote_fix . '"' . part1 . s:cmd_line_quote_fix . '"'
|
||
if exists('*xolox#shell#execute')
|
||
silent call xolox#shell#execute(orgcmd, 1)
|
||
else
|
||
silent exe "!" . orgcmd
|
||
endif
|
||
exe start .',' . end . 'read ~/org-src-block.org'
|
||
exe start . ',' . end . 'd'
|
||
call setpos('.',savecursor)
|
||
endfunction
|
||
function! MyExpTest()
|
||
let g:orgpath='c:\users\herbert\emacsclientw.exe --eval '
|
||
let g:myfilename = substitute(expand("%:p"),'\','/','g')
|
||
let g:myfilename = substitute(g:myfilename, '/ ','\ ','g')
|
||
let g:myvar = '(let ((org-export-babel-evaluate nil)) (progn (find-file \^' . '"' . g:myfilename . '\^' . '"' . ') (org-export-as-html-and-open 3) (kill-buffer) ))'
|
||
let g:myc = '!' . g:orgpath . '^"' . g:myvar . '^"'
|
||
silent exec g:myc
|
||
endfunction
|
||
function! OrgExportDashboard()
|
||
if s:OrgHasEmacsVar() == 0
|
||
return
|
||
endif
|
||
let save_more = &more | set nomore
|
||
let save_showcmd = &showcmd | set noshowcmd
|
||
" show export dashboard
|
||
"let mydict = { 't':'template', 'a':'ascii', 'n':'latin1', 'u':'utf8',
|
||
let mydict = { 't':'template', 'a':'ascii', 'A':'ascii', 'o':'odt', 'O':'odt-and-open',
|
||
\ 'n':'latin1', 'N':'latin1', 'u':'utf8','U':'utf8',
|
||
\ 'h':'html', 'b':'html-and-open', 'l':'latex',
|
||
\ 'f':'freemind', 'j':'taskjuggler', 'k':'taskjuggler-and-open',
|
||
\ 'p':'pdf', 'd':'pdf-and-open', 'D':'docbook', 'g':'tangle',
|
||
\ 'F':'current-file', 'P':'current-project', 'E':'all' }
|
||
echohl MoreMsg
|
||
echo " Press key for export operation:"
|
||
echo " --------------------------------"
|
||
echo " [t] insert the export options template block"
|
||
echo " "
|
||
echo " [a/n/u] export as ASCII/Latin1/utf8 [A/N/U] ...and open in buffer"
|
||
echo " "
|
||
echo " [h] export as HTML"
|
||
echo " [b] export as HTML and open in browser"
|
||
echo " "
|
||
echo " [l] export as LaTeX"
|
||
echo " [p] export as LaTeX and process to PDF"
|
||
echo " [d] . . . and open PDF file"
|
||
echo " "
|
||
echo " [o] export as ODT [O] as ODT and open"
|
||
echo " [D] export as DocBook"
|
||
echo " [V] export as DocBook, process to PDF, and open"
|
||
echo " [x] export as XOXO [j] export as TaskJuggler"
|
||
echo " [m] export as Freemind [k] export as TaskJuggler and open"
|
||
|
||
echo " [g] tangle file"
|
||
echo " "
|
||
echo " [F] publish current file"
|
||
echo " [P] publish current project"
|
||
echo " [E] publish all projects"
|
||
echo " "
|
||
echohl None
|
||
let key = nr2char(getchar())
|
||
for item in keys(mydict)
|
||
if (item ==# key) && (item !=# 't')
|
||
"let g:org_emacs_autoconvert = 1
|
||
"call s:GlobalUnconvertTags(changenr())
|
||
let exportfile = expand('%:t')
|
||
silent exec 'write'
|
||
|
||
let orgpath = g:org_command_for_emacsclient . ' -n --eval '
|
||
let g:myfilename = substitute(expand("%:p"),'\','/','g')
|
||
let g:myfilename = substitute(g:myfilename, '/ ','\ ','g')
|
||
" set org-mode to either auto-evaluate all exec blocks or evaluate none w/o
|
||
" confirming each with yes/no
|
||
if g:org_export_babel_evaluate == 1
|
||
let g:mypart1 = '(let ((org-export-babel-evaluate t)(org-confirm-babel-evaluate nil)'
|
||
else
|
||
let g:mypart1 = '(let ((org-export-babel-evaluate nil)'
|
||
endif
|
||
let g:mypart1 .= '(buf (find-file \' . s:cmd_line_quote_fix . '"' . g:myfilename . '\' . s:cmd_line_quote_fix . '"))) (progn ('
|
||
|
||
if item =~? 'g'
|
||
let g:mypart3 = ' ) (set-buffer buf) (not-modified) (kill-this-buffer) ))'
|
||
else
|
||
let g:mypart3 = ' nil ) (set-buffer buf) (not-modified) (kill-this-buffer) ))'
|
||
endif
|
||
|
||
if item =~# 'F\|P\|E'
|
||
let command_part2 = ' org-publish-' . mydict[key]
|
||
elseif item == 'g'
|
||
let command_part2 = ' org-babel-tangle'
|
||
else
|
||
let command_part2 = ' org-export-as-' . mydict[key]
|
||
endif
|
||
|
||
let orgcmd = orgpath . s:cmd_line_quote_fix . '"' . g:mypart1 . command_part2 . g:mypart3 . s:cmd_line_quote_fix . '"'
|
||
let g:orgcmd = orgcmd
|
||
" execute the call out to emacs
|
||
redraw
|
||
echo "Export in progress. . . "
|
||
if exists('*xolox#shell#execute')
|
||
"silent! let g:expmsg = xolox#shell#execute(orgcmd . ' | cat ', 1)
|
||
silent! call xolox#shell#execute(orgcmd , 1)
|
||
else
|
||
"execute '!' . orgcmd
|
||
silent! execute '!' . orgcmd
|
||
endif
|
||
redraw
|
||
echo "Export in progress. . . Export complete."
|
||
break
|
||
endif
|
||
endfor
|
||
if key ==# 't'
|
||
let template = [
|
||
\ '#+TITLE: ' . expand("%p")
|
||
\ ,'#+AUTHOR: '
|
||
\ ,'#+EMAIL: '
|
||
\ ,'#+DATE: ' . strftime("%Y %b %d %H:%M")
|
||
\ ,'#+DESCRIPTION: '
|
||
\ ,'#+KEYWORDS: '
|
||
\ ,'#+LANGUAGE: en'
|
||
\ ,'#+OPTIONS: H:3 num:t toc:t \n:nil @:t ::t |:t ^:t -:t f:t *:t <:t'
|
||
\ ,'#+OPTIONS: TeX:t LaTeX:t skip:nil d:nil todo:t pri:nil tags:not-in-toc'
|
||
\ ,'#+INFOJS_OPT: view:nil toc:nil ltoc:t mouse:underline buttons:0 path:http://orgmode.org/org-info.js'
|
||
\ ,'#+EXPORT_SELECT_TAGS: export'
|
||
\ ,'#+EXPORT_EXCLUDE_TAGS: noexport'
|
||
\ ,'#+LINK_UP: '
|
||
\ ,'#+LINK_HOME: '
|
||
\ ,'#+XSLT: '
|
||
\ ]
|
||
silent call append(line('.')-1,template)
|
||
elseif key =~# 'A\|N\|U'
|
||
exec 'split ' . expand('%:r') . '.txt'
|
||
normal gg
|
||
endif
|
||
|
||
let &more = save_more
|
||
let &showcmd = save_showcmd
|
||
|
||
endfunction
|
||
|
||
function! s:MailLookup()
|
||
Utl openlink https://mail.google.com/mail/?hl=en&shva=1#search/after:2010-10-24+before:2010-10-26
|
||
"https://mail.google.com/mail/?hl=en&shva=1#search/after%3A2010-10-24+before%3A2010-10-26
|
||
endfunction
|
||
function! s:UniqueList(list)
|
||
" returns the union of two lists
|
||
" (some algo ...)
|
||
let rdict = {}
|
||
for item in a:list
|
||
let rdict[item] = 1
|
||
endfor
|
||
return sort(map(keys(rdict),'str2nr(v:val)'))
|
||
endfunc
|
||
function! s:Union(list1, list2)
|
||
" returns the union of two lists
|
||
" (some algo ...)
|
||
let rdict = {}
|
||
for item in a:list1
|
||
let rdict[item] = 1
|
||
endfor
|
||
for item in a:list2
|
||
let rdict[item] = 1
|
||
endfor
|
||
return sort(keys(rdict))
|
||
endfunc
|
||
function! s:Intersect(list1, list2)
|
||
" returns the intersection of two lists
|
||
" (some algo ...)
|
||
" fro andy wokula on vim-use mailing list
|
||
let rdict = {}
|
||
for item in a:list1
|
||
if has_key(rdict, item)
|
||
let rdict[item] += 1
|
||
else
|
||
let rdict[item] = 1
|
||
endif
|
||
endfor
|
||
for item in a:list2
|
||
if has_key(rdict, item)
|
||
let rdict[item] += 1
|
||
else
|
||
let rdict[item] = 1
|
||
endif
|
||
endfor
|
||
call filter(rdict, 'v:val == 2')
|
||
return sort(keys(rdict))
|
||
endfunc
|
||
|
||
function! OrgSetEmphasis( emph_char ) range
|
||
let emph_char = a:emph_char
|
||
let my_mode = mode()
|
||
if my_mode ==? 'v'
|
||
exe 'normal oi' . emph_char
|
||
exe 'normal gvoi' . emph_char
|
||
else
|
||
exe 'normal i' . emph_char . emph_char
|
||
endif
|
||
endfunction
|
||
|
||
function! s:OrgCustomTodoHighlights()
|
||
if !exists('g:org_todo_custom_highlights')
|
||
return
|
||
endif
|
||
for item in keys(g:org_todo_custom_highlights)
|
||
let d = g:org_todo_custom_highlights
|
||
if has('gui_running')
|
||
let fg = get(d[item], 'guifg')
|
||
let bg = get(d[item], 'guibg')
|
||
exec 'hi! ' . item . ((fg>#'') ? ' guifg=' . fg : '') . ((bg>#'') ? ' guibg=' . bg : '')
|
||
else
|
||
let fg = get(d[item], 'ctermfg')
|
||
let bg = get(d[item], 'ctermfg')
|
||
exec 'hi! ' . item . ((fg>#'') ? ' ctermfg=' . fg : '') . ((bg>#'') ? ' ctermbg=' . bg : '')
|
||
endif
|
||
|
||
" xxxx todo put back in containedins, do synclears? check order?
|
||
if bufname('%')=='__Agenda__'
|
||
exec 'syntax match ' . item . ' ' . '+ \*\+ \zs' . item . ' + containedin=AOL1,AOL2,AOL3,AOL4,AOL5,AOL6'
|
||
else
|
||
" delete current match if it already exists
|
||
let mymatches = getmatches()
|
||
let tempdict = {}
|
||
for i in range(0, len(mymatches)-1)
|
||
let tempdict[i] = mymatches[i]
|
||
endfor
|
||
for i in keys(tempdict)
|
||
if tempdict[i].group == item
|
||
exec 'syntax clear ' . tempdict[i].group
|
||
break
|
||
endif
|
||
endfor
|
||
"now put new match in
|
||
exec 'syntax match ' . item . ' ' . '+\*\+ \zs' . item . ' + containedin=OL1,OL2,OL3,OL4,OL5,OL6'
|
||
endif
|
||
|
||
endfor
|
||
endfunction
|
||
|
||
function! OrgSetColors()
|
||
" Set highlights for outline headings. These are set from existing
|
||
" highlights in a colorscheme:
|
||
" OL1 from Statement
|
||
" OL2 from Identifier
|
||
" OL3 from Constant
|
||
" OL4 from Comment
|
||
" OL5 from Special
|
||
for pair in [ ['OL1','Statement'], ['OL2','Identifier'], ['OL3','Constant'],
|
||
\ ['OL4','Comment'], ['OL5','Special'] ]
|
||
execute 'hi clear ' . pair[0]
|
||
execute 'hi clear ' . pair[0] .'Folded'
|
||
execute 'hi ' . pair[0] . ' ' . org#GetGroupHighlight( pair[1] )
|
||
execute 'hi ' . pair[0] . 'Folded ' . org#GetGroupHighlight( pair[1] )
|
||
execute 'hi ' . pair[0] . ' gui=NONE'
|
||
execute 'hi ' . pair[0] . 'Folded gui=bold'
|
||
endfor
|
||
|
||
if has('gui_running')
|
||
hi! FoldColumn guifg=bg guibg=bg
|
||
else
|
||
try
|
||
hi! FoldColumn ctermfg=bg ctermbg=bg
|
||
catch
|
||
hi! FoldColumn ctermfg=0 ctermbg=0
|
||
endtry
|
||
endif
|
||
|
||
"show text on SignColumn
|
||
hi! SignColumn guibg=fg guibg=bg
|
||
|
||
|
||
" various text item "highlightings" are below
|
||
" change to suit your taste and put in OrgCustomColors() (see below)
|
||
hi! Org_Drawer guifg=pink ctermfg=magenta
|
||
hi! Org_Drawer_Folded guifg=pink ctermfg=magenta gui=bold cterm=bold
|
||
hi! Org_Property_Value guifg=pink ctermfg=magenta
|
||
hi! Org_Block guifg=#555555 ctermfg=magenta
|
||
hi! Org_Src_Block guifg=#555555 ctermfg=magenta
|
||
hi! Org_Table guifg=#888888 guibg=#333333 ctermfg=magenta
|
||
hi! Org_Config_Line guifg=darkgray ctermfg=magenta
|
||
hi! Org_Tag guifg=lightgreen ctermfg=blue
|
||
hi! Org_Date guifg=magenta ctermfg=magenta gui=underline cterm=underline
|
||
hi! Org_Star guifg=#444444 ctermfg=darkgray
|
||
hi! Props guifg=#ffa0a0 ctermfg=gray
|
||
hi! Org_Code guifg=darkgray gui=bold ctermfg=14
|
||
hi! Org_Itals gui=italic guifg=#aaaaaa ctermfg=lightgray
|
||
hi! Org_Bold gui=bold guifg=#aaaaaa ctermfg=lightgray
|
||
hi! Org_Underline gui=underline guifg=#aaaaaa ctermfg=lightgray
|
||
hi! Org_Marked gui=bold guibg=#bbaacc ctermbg=lightgray
|
||
hi! Org_Lnumber guifg=#999999 ctermfg=gray
|
||
|
||
" agenda highlight groups below
|
||
hi Overdue guifg=red
|
||
hi Upcoming guifg=yellow
|
||
hi DateType guifg=#dd66bb
|
||
hi Locator guifg=#333333
|
||
|
||
"hi agenda_dayline guifg=#44aa44 gui=underline
|
||
"hi agenda_weekendline guifg=#55ee55 gui=underline
|
||
hi agenda_omitted_days guifg=#555555 ctermfg=gray
|
||
hi agenda_todo guifg=lightred gui=bold ctermfg=lightred cterm=bold
|
||
hi agenda_done guifg=lightgreen ctermfg=lightgreen
|
||
hi agenda_date guifg=lightblue ctermfg=lightblue
|
||
hi agenda_weekenddate guifg=lightblue gui=bold ctermfg=lightblue cterm=bold
|
||
hi agenda_scheduled guifg=lightyellow ctermfg=lightyellow
|
||
hi agenda_scheduled_previous guifg=lightmagenta ctermfg=lightmagenta
|
||
hi agenda_timegrid guifg=#666666 ctermfg=gray
|
||
" end agenda highlights
|
||
|
||
if has("conceal")
|
||
hi! default linkends guifg=blue ctermfg=blue
|
||
endif
|
||
hi! Org_Full_Link guifg=cyan gui=underline ctermfg=lightblue cterm=underline
|
||
hi! Org_Half_Link guifg=cyan gui=underline ctermfg=lightblue cterm=underline
|
||
highlight OrgColumnHeadings guibg=#444444 guifg=#aaaaaa gui=underline
|
||
|
||
hi! DONETODO guifg=green ctermfg=green
|
||
hi! NOTDONETODO guifg=red ctermfg=lightred
|
||
|
||
"user can define OrgCustomColors() in vimrc for above items, these will be executed
|
||
"here and override the defaults above.
|
||
if exists('*OrgCustomColors')
|
||
call OrgCustomColors()
|
||
endif
|
||
|
||
call s:OrgCustomTodoHighlights()
|
||
|
||
" below is for setting highlight groups for when agenda is used to
|
||
" select a headline in main doc, or when items are marked.
|
||
" Use g:org_selector_ctermbg and/or g:org_selector_guibg
|
||
" to customize.
|
||
for i in range(1,5)
|
||
let hlstring = org#GetGroupHighlight('OL' . i)
|
||
let ctermbg = 'ctermbg=' . (exists('g:org_selector_ctermbg') ? g:org_selector_ctermbg : 'gray')
|
||
let guibg = 'guibg=' . (exists('g:org_selector_guibg') ? g:org_selector_guibg : '#333333')
|
||
if hlstring =~ 'ctermbg'
|
||
let hlstring = substitute(hlstring,'ctermbg=\S+',ctermbg,'')
|
||
else
|
||
let hlstring = hlstring . ' ' . ctermbg
|
||
endif
|
||
if hlstring =~ 'guibg'
|
||
let hlstring = substitute(hlstring,'guibg=\S+',guibg,'')
|
||
else
|
||
let hlstring = hlstring . ' ' . guibg
|
||
endif
|
||
exec 'hi! Org_Chosen_Agenda_Heading' . i . ' ' . hlstring
|
||
endfor
|
||
exec 'hi! Org_Chosen_Agenda_Heading ' . guibg . ' ' . ctermbg
|
||
endfunction
|
||
autocmd ColorScheme * :silent! call OrgSetColors()
|
||
call OrgSetColors()
|
||
|
||
function! s:OrgPromoteSubheads()
|
||
let line = s:OrgGetHead()
|
||
let star_count = 1 + len(matchstr(getline(line('.')),'^\*\+'))
|
||
let start_line = line + 1
|
||
let end_line = s:OrgSubtreeLastLine()
|
||
let subheads = []
|
||
execute start_line . ',' . end_line . 'g/^\*\{' . star_count . '} /call add(subheads,line("."))'
|
||
for i in sort(subheads,'<SID>ReverseSort')
|
||
exec i
|
||
call OrgMoveLevel(i,'left')
|
||
endfor
|
||
execute line
|
||
echo "Subheads promoted."
|
||
endfunction
|
||
"Section for refile and archive funcs
|
||
function! OrgRefileDashboard()
|
||
echohl MoreMsg
|
||
echo " ================================"
|
||
echo " Persisted point is: " . (!exists('persistent_refile_point') ? 'None' : join(s:persistent_refile_point,'/'))
|
||
echo " Last point used: " . (!exists('s:last_refile_point') ? 'None' : join(s:last_refile_point,'/'))
|
||
echo " ================================"
|
||
echo " Press key for a refile command:"
|
||
echo " --------------------------------"
|
||
echo " r / l / p refile subtree to new point/last point/persisted point"
|
||
echo " j / z / t jump to new point/last point/persistent point"
|
||
echo " s set persistent refile point "
|
||
echo " a archive to _archive file"
|
||
echo " v jump to _archive file"
|
||
echo ""
|
||
echohl Question
|
||
let key = nr2char(getchar())
|
||
redraw
|
||
if key ==? 'r'
|
||
call s:OrgRefile(line('.'))
|
||
elseif key ==? 'l'
|
||
call s:OrgRefileToLastPoint(line('.'))
|
||
elseif key ==? 'p'
|
||
call s:OrgRefileToPermPoint(line('.'))
|
||
elseif key ==? 'j'
|
||
call s:OrgJumpToRefilePoint()
|
||
elseif key ==? 'z'
|
||
call s:OrgJumpToLastRefilePoint()
|
||
elseif key ==? 't'
|
||
call s:OrgJumpToRefilePointPersistent()
|
||
elseif key ==? 's'
|
||
call s:OrgSetRefilePoint()
|
||
elseif key ==? 'v'
|
||
call org#LocateFile(expand('%:p') . '_archive')
|
||
elseif key ==? 'a'
|
||
if confirm('Confirm that you want to archive subtree(s)',"&Yes\n&Cancel")
|
||
call s:DoRefile(['_archive'],[line('.')])
|
||
endif
|
||
else
|
||
echo "No refile option selected."
|
||
endif
|
||
echohl None
|
||
endfunction
|
||
function! s:OrgGatherDashboard()
|
||
echohl MoreMsg
|
||
echo " ================================"
|
||
echo " Press key for a mark/gather/sort command:"
|
||
echo " --------------------------------"
|
||
echo " <space> toggle mark on/off for current heading"
|
||
echo " d delete all marks"
|
||
echo " h gather marked headings to current heading"
|
||
echo " s sort subheads of current heading"
|
||
echo ""
|
||
echohl Question
|
||
let key = nr2char(getchar())
|
||
redraw
|
||
if key ==? ' '
|
||
call s:ToggleHeadingMark(line('.'))
|
||
elseif key ==? 'd'
|
||
call s:DeleteHeadingMarks()
|
||
elseif key ==? 'h'
|
||
call s:GatherMarks()
|
||
elseif key ==? 's'
|
||
call s:OrgSortSubheads()
|
||
else
|
||
echo "No gather/mark option selected."
|
||
endif
|
||
echohl None
|
||
endfunction
|
||
|
||
let g:org_heading_temp=['','','','','','','','']
|
||
function! s:OutlineHeads()
|
||
let level = s:Ind(line('.'))
|
||
let g:org_heading_temp[level-1] = matchstr(getline(line('.')),'^\*\+ \zs.*')
|
||
" put level 1 head in result
|
||
let result = expand("%:t") . g:org_heading_temp[0]
|
||
" now add full tree to level of current heading
|
||
for item in g:org_heading_temp[1: level-1]
|
||
let result .= '/' . item
|
||
endfor
|
||
return result
|
||
endfunction
|
||
function! s:GetMyItems(arghead)
|
||
" assemble and return list of subheads of argument file/heading
|
||
let arghead = a:arghead
|
||
let result = []
|
||
if a:arghead[-1:] == '*'
|
||
let arghead = arghead[:-2]
|
||
endif
|
||
let ilist = split(arghead,'/')
|
||
call org#SaveLocation()
|
||
if expand("%:t") != ilist[0]
|
||
"call org#LocateFile( '~\Desktop\org_files\' . ilist[0] )
|
||
call org#LocateFile( fnamemodify(s:refile_file,":p:h:") . '/' . ilist[0] )
|
||
if &ft != 'org'
|
||
set ft=org
|
||
endif
|
||
endif
|
||
if a:arghead[-1:] == '*'
|
||
let tolevel = 3
|
||
else
|
||
let tolevel = len(ilist)
|
||
endif
|
||
exec 'g/^\*\{1,' . tolevel . '} /call add(result, s:OutlineHeads())'
|
||
call org#RestoreLocation()
|
||
return result
|
||
endfunction
|
||
function! OrgFileList(arghead,sd,gf)
|
||
let arghead = substitute(a:arghead,'\~','\\\~','g')
|
||
let s:myheads = ['[current file]'] + copy(g:agenda_files)
|
||
let matches = filter( copy( s:myheads ),'v:val =~ arghead')
|
||
redraw!
|
||
return join( matches, "\n" )
|
||
endfunction
|
||
|
||
function! OrgHeadingList(arghead,sd,gf)
|
||
let arghead = a:arghead
|
||
let s:myheads = s:GetMyItems(arghead)
|
||
let matches = filter( copy( s:myheads ),'v:val =~ a:arghead')
|
||
redraw!
|
||
return join( matches, "\n" )
|
||
endfunction
|
||
|
||
function! s:GetTarget()
|
||
let orig_wildmode = &wildmode
|
||
set wildmode=list:full
|
||
try
|
||
" need to modify getcmdline to strip back on bs
|
||
cmap <c-BS> <C-\>egetcmdline()[-1:] == '/' ? matchstr(getcmdline()[:-2], '.*\ze/.*' ) . '/' : matchstr(getcmdline(), '.*\ze/.*') . '/'<CR>
|
||
while 1
|
||
let s:refile_file = ''
|
||
"let s:refile_file = input("Target file: ","[current file]",'custom,s:FileList')
|
||
let file_default = (bufname('%')==#'__Agenda__') ? g:agenda_files[0] : ""
|
||
"[current file]"
|
||
let s:refile_file = input("Target file: ",file_default,'custom,OrgFileList')
|
||
if s:refile_file ==# '[current file]'
|
||
let s:refile_file = expand("%")
|
||
elseif index(g:agenda_files,s:refile_file) == -1
|
||
break
|
||
endif
|
||
let heading = input('Outline heading: ', fnamemodify(s:refile_file,':t:') . "\t",'custom,OrgHeadingList')
|
||
let head = matchstr(heading,'.\{-}/\zs.*')
|
||
if head ==# ''
|
||
continue
|
||
else
|
||
return [ s:refile_file, head]
|
||
endif
|
||
endwhile
|
||
finally
|
||
let &wildmode = orig_wildmode
|
||
cunmap <c-BS>
|
||
endtry
|
||
endfunction
|
||
function! s:OrgJumpToRefilePointPersistent()
|
||
if exists('s:persistent_refile_point') && (len(s:persistent_refile_point)==2)
|
||
call s:OrgGotoHeading( s:persistent_refile_point[0], s:persistent_refile_point[1] )
|
||
normal zv
|
||
else
|
||
echo 'No persistent refile point assigned.'
|
||
endif
|
||
endfunction
|
||
function! s:OrgJumpToLastRefilePoint()
|
||
if exists('s:last_refile_point') && (len(s:last_refile_point)==2)
|
||
let p = s:last_refile_point
|
||
call s:OrgGotoHeading( p[0], p[1])
|
||
normal zv
|
||
else
|
||
echo 'No last refile point yet.'
|
||
endif
|
||
endfunction
|
||
function! s:OrgJumpToRefilePoint()
|
||
let my_refile_point = s:GetTarget()
|
||
if len(my_refile_point)==2
|
||
call s:OrgGotoHeading( my_refile_point[0], my_refile_point[1] )
|
||
normal zv
|
||
else
|
||
echo "Jump aborted."
|
||
endif
|
||
endfunction
|
||
function! s:OrgSetRefilePoint()
|
||
let s:persistent_refile_point = s:GetTarget()
|
||
endfunction
|
||
function! OrgSetArchivePoint()
|
||
let s:archive_refile_point = s:GetTarget()
|
||
endfunction
|
||
|
||
function! OrgRefileToArchive(headline)
|
||
if exists('s:archive_refile_point') && (s:archive_refile_point[1] !=# '')
|
||
silent call s:DoRefile( s:archive_refile_point, [a:headline] )
|
||
redraw!
|
||
let p = s:archive_refile_point
|
||
echo "Tree(s) refiled to: " . p[0] . '/' . p[1]
|
||
else
|
||
echo 'Refile aborted, archive point not assigned.'
|
||
endif
|
||
endfunction
|
||
function! s:OrgRefileToPermPoint(headline)
|
||
if exists('s:persistent_refile_point') && (s:persistent_refile_point[1] !=# '')
|
||
silent call s:DoRefile( s:persistent_refile_point, [a:headline] )
|
||
redraw!
|
||
echo "Tree(s) refiled to: \n" . s:persistent_refile_point[0] . '/' . s:persistent_refile_point[1]
|
||
else
|
||
echo 'Refile aborted, persistent point not defined.'
|
||
endif
|
||
endfunction
|
||
function! s:OrgRefileToLastPoint(headline)
|
||
if exists('s:last_refile_point') && (s:last_refile_point[1] !=# '')
|
||
silent call s:DoRefile( s:last_refile_point, [a:headline] )
|
||
redraw!
|
||
echo "Tree(s) refiled to: " . s:last_refile_point[0] . '/' . s:last_refile_point[1]
|
||
else
|
||
echo 'Refile aborted, no last point yet.'
|
||
endif
|
||
endfunction
|
||
function! s:OrgSortSubheads()
|
||
let start = line('.')
|
||
if getline(start) !~ b:v.headMatch
|
||
echo "You must be on a heading to sort subheads."
|
||
return
|
||
endif
|
||
|
||
" FIRST DELETE ALL SIGNS
|
||
sign unplace *
|
||
let b:v.heading_marks_dict = {}
|
||
let end = s:OrgSubtreeLastLine()
|
||
execute start . ',' . end . 'g/^\*\{' . s:Ind(line('.')) . '} /call s:PlaceSignsForSort(line("."))'
|
||
exec start
|
||
call s:GatherMarks('forward')
|
||
|
||
endfunction
|
||
function! s:PlaceSignsForSort(line)
|
||
exec 'sign place ' . a:line . " line=" . a:line . " name=marked buffer=".bufnr("%")
|
||
endfunction
|
||
command! -buffer OrgGatherMarks :call <SID>GatherMarks()
|
||
command! -buffer OrgGatherAgendaMarks :call <SID>GatherMarks('agenda')
|
||
function! s:GatherMarks(...)
|
||
if getline(line('.')) !~ b:v.headMatch
|
||
echo "You must be on a heading to gather."
|
||
return
|
||
endif
|
||
let targ_list = [expand('%:p'), getline(line('.'))]
|
||
if (a:0 == 1) && (a:1 ==# 'agenda')
|
||
call org#LocateFile('__Agenda__')
|
||
endif
|
||
if org#redir('sign place') =~ 'name=marked'
|
||
if (a:0 == 1) && (a:1 !=# 'agenda')
|
||
silent call s:DoRefile( targ_list, [line('.')], a:1 )
|
||
else
|
||
silent call s:DoRefile( targ_list, [line('.')])
|
||
endif
|
||
call s:OrgGotoHeading(targ_list[0], targ_list[1])
|
||
call OrgShowSubs(s:Ind(line('.')), 0)
|
||
redraw!
|
||
echo "Gathered marked headings to subheads of current heading."
|
||
else
|
||
echo 'No headings marked, nothing gathered.'
|
||
endif
|
||
endfunction
|
||
function! s:OrgRefile(headline)
|
||
let targ_list = s:GetTarget()
|
||
if targ_list[1] !=# ''
|
||
silent call s:DoRefile( targ_list, [a:headline] )
|
||
redraw!
|
||
echo "Tree(s) refiled to: " . targ_list[0] . '/' . targ_list[1]
|
||
else
|
||
echo 'Refile aborted.'
|
||
endif
|
||
endfunction
|
||
func! GetMarks()
|
||
echo b:v.heading_marks
|
||
endfunction
|
||
function! s:ChangeLevel( text_lines, change_val )
|
||
let mylines = split( a:text_lines, "\n")
|
||
let change_val = a:change_val
|
||
let i = 0
|
||
while i < len(mylines)
|
||
if mylines[i][0] == '*'
|
||
if change_val > 0
|
||
let mylines[i] = repeat('*',change_val) . mylines[i]
|
||
else
|
||
let abs_change = -(change_val)
|
||
let mylines[i] = mylines[i][ abs_change :]
|
||
endif
|
||
endif
|
||
let i += 1
|
||
endwhile
|
||
return mylines
|
||
endfunction
|
||
function! s:DoRefile(targ_list,heading_list,...)
|
||
|
||
if a:0 == 1
|
||
let sort_type = a:1
|
||
endif
|
||
let targ_list = a:targ_list
|
||
let heading_list = a:heading_list
|
||
let file_lines_dict = {}
|
||
let first_lines = {}
|
||
let held_lines = {}
|
||
call org#SaveLocation()
|
||
|
||
|
||
if bufname('%') == '__Agenda__'
|
||
if len(b:v.heading_marks_dict) >= 1
|
||
let heading_list = keys(b:v.heading_marks_dict)
|
||
endif
|
||
" go through and assemble dict of files and lists of lines for each
|
||
for item in heading_list
|
||
exec item
|
||
let file = s:filedict[str2nr(matchstr(getline(line('.')), '^\d\d\d'))]
|
||
let lineno = str2nr(matchstr(getline(line('.')),'^\d\d\d\zs\d*'))
|
||
let buffer_lineno = s:ActualBufferLine(lineno,bufnr(file))
|
||
|
||
if type(get(file_lines_dict,file)) == 0
|
||
" add first line item
|
||
let file_lines_dict[file] = [buffer_lineno]
|
||
else
|
||
" add subsequent lines
|
||
call add(file_lines_dict[file], buffer_lineno)
|
||
endif
|
||
endfor
|
||
else
|
||
let file_lines_dict = s:AssembleFileHeadingsDict()
|
||
if empty(file_lines_dict)
|
||
let file_lines_dict = { expand('%:p'):heading_list}
|
||
endif
|
||
endif
|
||
|
||
" now that we have dict of file(s) and linelists in file_lines_dict
|
||
" we can go through and refile/archive
|
||
let archiving = (targ_list[0] == '_archive')
|
||
if archiving
|
||
call s:ArchiveSubtrees(targ_list, file_lines_dict)
|
||
else
|
||
for afile in keys(file_lines_dict)
|
||
call org#LocateFile(afile)
|
||
" need to get heads and avoid duplicates in case we're in date agenda view
|
||
let linelist = file_lines_dict[afile]
|
||
call map( linelist, 's:OrgGetHead_l(v:val)')
|
||
let linelist = s:UniqueList(linelist)
|
||
" now linelist has list of heading line numbers, go through
|
||
for aline in sort(linelist,'<SID>ReverseSort')
|
||
call org#LocateFile(afile)
|
||
exec aline
|
||
let refile_stars = s:Ind(aline) - 1
|
||
" delete subhead, but first guarantee it's not in a fold
|
||
normal! zv
|
||
|
||
let aline_text = getline(aline)
|
||
silent execute aline . ',' . s:OrgSubtreeLastLine_l(aline) . 'delete x'
|
||
|
||
" now go to refile point and put back in
|
||
call s:OrgGotoHeading(targ_list[0],targ_list[1])
|
||
let target_stars = s:Ind(line('.')) " don't subtract 1 b/c refile will be subhead
|
||
if refile_stars != target_stars
|
||
let x = s:ChangeLevel( @x, target_stars - refile_stars )
|
||
else
|
||
let x = split( @x, "\n")
|
||
endif
|
||
|
||
let line_buf = string(aline) . bufname('%')
|
||
if g:org_sort_with_todo_words
|
||
let first_lines[aline_text . bufname('%')] = line_buf
|
||
else
|
||
let first_lines[matchstr(aline_text,b:v.todoMatch . '\{0,1}\s\{0,1}\zs.*') . bufname('%')] = line_buf
|
||
endif
|
||
let held_lines[line_buf] = x
|
||
"let held_lines[x[0]] = x
|
||
echo "Refiled " . x[0] . ' to: ' . join(targ_list,'/')
|
||
endfor " for lines in this file
|
||
"now go back and clear out its heading marks
|
||
call org#LocateFile(afile)
|
||
let b:v.heading_marks_dict = {}
|
||
endfor " for files in file_lines_dict
|
||
|
||
" now put all the lines in . . .
|
||
" lines below were when each subtree was appended individually
|
||
" that's not case anymore, may need to rework and associate
|
||
" them with append of 'held_lines' to get ordering option. . .
|
||
for heading_line in (exists('sort_type') ? sort(keys(first_lines)) : keys(first_lines))
|
||
"for heading_line in keys(held_lines)
|
||
call s:OrgGotoHeading(targ_list[0],targ_list[1])
|
||
if g:org_reverse_note_order
|
||
let insert_line = s:OrgNextHead() - 1
|
||
if insert_line == -1
|
||
let insert_line = line('$')
|
||
endif
|
||
exec insert_line
|
||
else
|
||
exec s:OrgSubtreeLastLine()
|
||
endif
|
||
silent call append(line('.') , held_lines[first_lines[heading_line]])
|
||
endfor
|
||
endif
|
||
if !archiving
|
||
let s:last_refile_point = targ_list
|
||
endif
|
||
|
||
call org#RestoreLocation()
|
||
if bufname('%') == '__Agenda__'
|
||
if !empty(b:v.heading_marks_dict)
|
||
"need reverse list of str2nr(keys)
|
||
for line in sort(map(keys(b:v.heading_marks_dict), 'str2nr(v:val)'),'s:ReverseSort')
|
||
exec line . 'delete'
|
||
endfor
|
||
else "just delete this line
|
||
delete
|
||
endif
|
||
endif
|
||
call s:DeleteHeadingMarks()
|
||
|
||
endfunction
|
||
function! s:AssembleFileHeadingsDict()
|
||
call org#SaveLocation()
|
||
let file_lines_dict = {}
|
||
let signlist = split(org#redir('sign place'),"\n")
|
||
for line in signlist
|
||
let ftest = matchstr(line, '^Signs for \zs.*\ze:$')
|
||
if (ftest ># '') "&& (getbufvar(ftest,'&filetype') == 'org')
|
||
let file = ftest
|
||
continue
|
||
endif
|
||
let markmatch = matchlist(line,'\s*line=\(\d\+\).*name=\(.*$\)')
|
||
if !empty(markmatch) && (markmatch[2] == 'marked')
|
||
if get(file_lines_dict, file, []) == []
|
||
let file_lines_dict[file] = [ str2nr(markmatch[1]) ]
|
||
else
|
||
call add(file_lines_dict[file] , str2nr(markmatch[1]))
|
||
endif
|
||
endif
|
||
endfor
|
||
call org#RestoreLocation()
|
||
return file_lines_dict
|
||
endfunction
|
||
func! s:ReverseSort(i1, i2)
|
||
return a:i1 == a:i2 ? 0 : a:i1 > a:i2 ? -1 : 1
|
||
endfunc
|
||
function! s:ArchiveSubtrees(targ_list, file_lines_dict)
|
||
let targ_list = a:targ_list
|
||
let file_lines_dict = a:file_lines_dict
|
||
let msg = ''
|
||
|
||
for afile in keys(file_lines_dict)
|
||
call org#LocateFile(afile)
|
||
"need this dict for category
|
||
call OrgMakeDictInherited()
|
||
" need to get heads and avoid duplicates in case we're in date agenda view
|
||
let linelist = file_lines_dict[afile]
|
||
call map( linelist, 's:OrgGetHead_l(v:val)')
|
||
let linelist = s:UniqueList(linelist)
|
||
" reverse sort avoids problems with deleting as we go . . .
|
||
for aline in sort(linelist,'<SID>ReverseSort')
|
||
call org#LocateFile(afile)
|
||
exec aline
|
||
let refile_stars = s:Ind(aline) - 1
|
||
" we will delete subhead, but first guarantee it's not in a fold
|
||
normal! zv
|
||
|
||
call s:SetProp('ARCHIVE_TIME', strftime('%Y-%m-%d %a %H:%M'))
|
||
call s:SetProp('ARCHIVE_FILE', afile)
|
||
call s:SetProp('ARCHIVE_OLPATH', matchstr(getline(s:OrgParentHead()),'^\*\+\s\+\zs.*$') )
|
||
call s:SetProp('ARCHIVE_CATEGORY', b:v.org_dict.iprop(aline,'CATEGORY'))
|
||
|
||
silent execute aline . ',' . s:OrgSubtreeLastLine_l(aline) . 'delete x'
|
||
|
||
call org#LocateFile(afile . '_archive')
|
||
if line('$') == 1
|
||
let prefix_lines = [' # -*- mode: org -*-',
|
||
\ ' vim:ft=org:','','',
|
||
\ 'Archived entries from file ' . afile ]
|
||
call setline(1,prefix_lines)
|
||
endif
|
||
let target_stars = 1
|
||
if refile_stars != target_stars
|
||
let x = s:ChangeLevel( @x, target_stars - refile_stars )
|
||
else
|
||
let x = split( @x, "\n")
|
||
endif
|
||
silent call append(line('$') , x)
|
||
let msg .= "Archived " . x[0] . " to: " . afile . "_archive\n"
|
||
endfor " for lines in this file
|
||
" now write changes in this archive file and quit
|
||
silent wq
|
||
echo msg
|
||
echo " Press any key to continue..."
|
||
call getchar()
|
||
endfor " for files in file list
|
||
endfunction
|
||
function! s:OrgGotoHeading(target_file, target_head, ...)
|
||
call org#LocateFile( a:target_file )
|
||
normal gg
|
||
let head_list = split(a:target_head,'/')
|
||
let head_list[0] = (head_list[0][0] ==# '*' ? head_list[0] : '\* ' . head_list[0])
|
||
call search( head_list[0], 'c', '')
|
||
"call search( '^\* ' . head_list[0], 'c', '')
|
||
let heading_line = line('.')
|
||
let last_subline = s:OrgSubtreeLastLine_l(line('.'))
|
||
let i = 1
|
||
while i < len(head_list)
|
||
let stars = repeat('\*', i + 1)
|
||
call search( '^' . stars . ' ' . head_list[i], '', last_subline)
|
||
let i += 1
|
||
endwhile
|
||
endfunction
|
||
|
||
function! s:OrgVal(variable_str)
|
||
" takes variable as an argument and tests whether it's defined
|
||
" if not returns -1, otherwise return variable value
|
||
if exists(a:variable_str)
|
||
execute 'let val=' . a:variable_str
|
||
else
|
||
let val = -1
|
||
endif
|
||
return val
|
||
endfunction
|
||
|
||
command! PreLoadTags :silent call <SID>GlobalConvertTags()
|
||
command! PreWriteTags :silent call <SID>GlobalUnconvertTags(changenr())
|
||
command! PostWriteTags :silent call <SID>UndoUnconvertTags()
|
||
au BufRead *.org :PreLoadTags
|
||
au BufWrite *.org :PreWriteTags
|
||
au BufWritePost *.org :PostWriteTags
|
||
|
||
setlocal fillchars=|,
|
||
|
||
"Section Narrow Region
|
||
let g:nrrw_rgn_vert=1
|
||
let g:nrrw_custom_options={'wrap':0}
|
||
command! -buffer Narrow :call NarrowCodeBlock(line('.'))
|
||
|
||
function! NarrowCodeBlock(line)
|
||
if exists(":NarrowRegion") == 0
|
||
echo "The Vim plugin NrrwRgn.vim must be installed for"
|
||
echo "narrowing to work. You can find a copy at:"
|
||
echo 'http://www.vim.org/scripts/script.php?script_id=3075'
|
||
return
|
||
endif
|
||
" function first tests if inside src block, and if so
|
||
" narrows the code block. If not, then
|
||
" tests for headline and narrows the heading subtree
|
||
" save buf vars to put in new org buffer if subtree narrowing
|
||
let main_buf_vars = b:v
|
||
execute a:line
|
||
call search('^#+begin_src','b','')
|
||
let start=line('.') + 1
|
||
let language = matchstr(getline(line('.')), '^#+begin_src \zs\S\+')
|
||
if language == 'emacs-lisp'
|
||
let language = 'lisp'
|
||
endif
|
||
call search('^#+end_src','','')
|
||
let end=line('.') - 1
|
||
|
||
let start_width = winwidth(0)
|
||
let &winwidth = winwidth(0) / 3
|
||
if (start <= a:line) && (end >= a:line)
|
||
execute start ',' . end . 'call nrrwrgn#NrrwRgn()'
|
||
set nomodified
|
||
if filereadable($VIMRUNTIME . '/ftplugin/' . language . '.vim')
|
||
execute 'set ft=' . language
|
||
endif
|
||
if filereadable($VIMRUNTIME. '/syntax/' . language . '.vim')
|
||
execute 'set syntax=' . language
|
||
endif
|
||
|
||
let &winwidth=start_width*2/3
|
||
else
|
||
let start = s:OrgGetHead_l(a:line)
|
||
if start > 0
|
||
let end = s:OrgSubtreeLastLine_l(start)
|
||
execute start . ',' . end . 'call nrrwrgn#NrrwRgn()'
|
||
" then set ftype in new buffer
|
||
set ft=org
|
||
set nomodified
|
||
let b:v = main_buf_vars
|
||
let &winwidth=start_width*2/3
|
||
else
|
||
execute a:line
|
||
echo "You're not in a source code block or an outline heading."
|
||
endif
|
||
endif
|
||
endfunction
|
||
" Org Menu Entries
|
||
amenu &Org.&View.Entire\ &Document.To\ Level\ &1<tab>,1 :set foldlevel=1<cr>
|
||
amenu &Org.&View.Entire\ &Document.To\ Level\ &2<tab>,2 :set foldlevel=2<cr>
|
||
amenu &Org.&View.Entire\ &Document.To\ Level\ &3<tab>,3 :set foldlevel=3<cr>
|
||
amenu &Org.&View.Entire\ &Document.To\ Level\ &4<tab>,4 :set foldlevel=4<cr>
|
||
amenu &Org.&View.Entire\ &Document.To\ Level\ &5<tab>,5 :set foldlevel=5<cr>
|
||
amenu &Org.&View.Entire\ &Document.To\ Level\ &6<tab>,6 :set foldlevel=6<cr>
|
||
amenu &Org.&View.Entire\ &Document.To\ Level\ &7<tab>,7 :set foldlevel=7<cr>
|
||
amenu &Org.&View.Entire\ &Document.To\ Level\ &8<tab>,8 :set foldlevel=8<cr>
|
||
amenu &Org.&View.Entire\ &Document.To\ Level\ &9<tab>,9 :set foldlevel=9<cr>
|
||
amenu &Org.&View.Entire\ &Document.Expand\ Level\ &All :set foldlevel=99999<cr>
|
||
amenu &Org.&View.&Subtree.To\ Level\ &1<tab>,,1 :silent call OrgShowSubs(1,0)<cr>
|
||
amenu &Org.&View.&Subtree.To\ Level\ &2<tab>,,2 :silent call OrgShowSubs(2,0)<cr>
|
||
amenu &Org.&View.&Subtree.To\ Level\ &3<tab>,,3 :silent call OrgShowSubs(3,0)<cr>
|
||
amenu &Org.&View.&Subtree.To\ Level\ &4<tab>,,4 :silent call OrgShowSubs(4,0)<cr>
|
||
amenu &Org.&View.&Subtree.To\ Level\ &5<tab>,,5 :silent call OrgShowSubs(5,0)<cr>
|
||
amenu &Org.&View.&Subtree.To\ Level\ &6<tab>,,6 :silent call OrgShowSubs(6,0)<cr>
|
||
amenu &Org.&View.&Subtree.To\ Level\ &7<tab>,,7 :silent call OrgShowSubs(7,0)<cr>
|
||
amenu &Org.&View.&Subtree.To\ Level\ &8<tab>,,8 :silent call OrgShowSubs(8,0)<cr>
|
||
amenu &Org.&View.&Subtree.To\ Level\ &9\ \ \ \ \ \ <tab>,,9 :silent call OrgShowSubs(9,0)cr>
|
||
amenu &Org.-Sep1- :
|
||
amenu &Org.&New\ Heading.New\ Head\ Same\ Level<tab><cr>(or\ <s-cr>) :call OrgNewHead('same')<cr>
|
||
amenu &Org.&New\ Heading.New\ Subhead<tab><c-cr> :call OrgNewHead('leveldown')<cr>
|
||
amenu &Org.&New\ Heading.New\ Head\ Parent\ Level<tab><s-c-cr> :call OrgNewHead('levelup')<cr>
|
||
amenu &Org.&Navigate\ Headings.&Up\ to\ Parent\ Heading<tab><a-left> :exec <SID>OrgParentHead()<cr>
|
||
amenu &Org.&Navigate\ Headings.&First\ Child\ Heading<tab><a-right> :exec <SID>OrgFirstChildHead()<cr>
|
||
amenu &Org.&Navigate\ Headings.&Last\ Child\ Heading :exec <SID>OrgLastChildHead()<cr>
|
||
amenu &Org.&Navigate\ Headings.&Next\ Heading :exec <SID>OrgNextHead()<cr>
|
||
amenu &Org.&Navigate\ Headings.&Previous\ Heading :exec <SID>OrgPrevHead()<cr>
|
||
amenu &Org.&Navigate\ Headings.Next\ &Same\ Level :exec <SID>OrgNextHeadSameLevel()<cr>
|
||
amenu &Org.&Navigate\ Headings.Previous\ Same\ Level :exec <SID>OrgPrevHeadSameLevel()<cr>
|
||
amenu &Org.&Navigate\ Headings.Next\ &Sibling<tab><a-down> :exec <SID>OrgNextSiblingHead()<cr>
|
||
amenu &Org.&Navigate\ Headings.Previous\ Sibling<tab><a-up> :exec <SID>OrgPrevSiblingHead()<cr>
|
||
amenu &Org.Edit\ &Structure.Move\ Subtree\ &Up<tab><c-a-up> :call OrgMoveLevel(line('.'),'up')<cr>
|
||
amenu &Org.Edit\ &Structure.Move\ Subtree\ &Down<tab><c-a-down> :call OrgMoveLevel(line('.'),'down')<cr>
|
||
amenu &Org.Edit\ &Structure.&Promote\ Subtree<tab><c-a-left> :call OrgMoveLevel(line('.'),'left')<cr>
|
||
amenu &Org.Edit\ &Structure.&Demote\ Subtree<tab><c-a-right> :call OrgMoveLevel(line('.'),'right')<cr>
|
||
vmenu &Org.&Editing.&Bold\ (*)<tab>,cb "zdi*<C-R>z*<ESC>l
|
||
vmenu &Org.&Editing.&Italic\ (/)<tab>,ci "zdi/<C-R>z/<ESC>l
|
||
vmenu &Org.&Editing.&Underline\ (_)<tab>,cu "zdi_<C-R>z_<ESC>l
|
||
vmenu &Org.&Editing.&Code\ (=)<tab>,cc "zdi=<C-R>z=<ESC>l
|
||
amenu &Org.&Editing.-Sep22- :
|
||
amenu &Org.&Editing.&Narrow<tab>,na :silent call NarrowCodeBlock(line('.'))<cr>
|
||
"amenu &Org.&Editing.Narrow\ &Codeblock<tab>,nc :silent call NarrowCodeBlock(line('.'))<cr>
|
||
"amenu &Org.&Editing.Narrow\ Outline\ &Subtree<tab>,ns :silent call NarrowOutline(line('.'))<cr>
|
||
amenu &Org.&Refile.&Refile\ to\ Point<tab>,rh :call OrgRefile(line('.'))<cr>
|
||
amenu &Org.&Refile.&Jump\ to\ Point<tab>,rj :call OrgJumpToRefilePoint()<cr>
|
||
amenu &Org.&Refile.&Jump\ to\ Persistent\ Point<tab>,rx :call OrgJumpToRefilePointPersistent()<cr>
|
||
amenu &Org.&Refile.&Jump\ to\ Point<tab>,rj :call OrgJumpToRefilePoint()<cr>
|
||
amenu &Org.&Refile.&Set\ Persistent\ Refile\ Point<tab>,rs :call OrgSetRefilePoint()<cr>
|
||
amenu &Org.&Refile.Refile\ to\ Persistent\ Point<tab>,rp :call OrgRefileToPermPoint(line('.'))<cr>
|
||
amenu &Org.Open\ Capture\ File :call org#OpenCaptureFile()<cr>
|
||
amenu &Org.&Mark/Gather/Sort.&Mark/Unmark\ Heading<tab>,<space> :call <SID>ToggleHeadingMark(line('.'))<cr>
|
||
amenu &Org.&Mark/Gather/Sort.&Unmark\ all<tab>,<c-space> :call <SID>DeleteHeadingMarks()<cr>
|
||
amenu &Org.&Mark/Gather/Sort.&Gather\ to\ current\ heading<tab>,gh :call <SID>GatherMarks()<cr>
|
||
amenu &Org.&Mark/Gather/Sort.&Sort\ subheads<tab>,gs :call <SID>OrgSortSubheads()<cr>
|
||
amenu &Org.-Sep2- :
|
||
amenu &Org.&Columns\ Menu :call OrgColumnsDashboard()<cr>
|
||
amenu &Org.&Hyperlinks.Add/&edit\ link<tab>,le :call EditLink()<cr>
|
||
amenu &Org.&Hyperlinks.&Follow\ link<tab>,lf :call FollowLink(OrgGetLink())<cr>
|
||
amenu &Org.&Hyperlinks.&Next\ link<tab>,ln :/]]<cr>
|
||
amenu &Org.&Hyperlinks.&Previous\ link<tab>,lp :?]]<cr>
|
||
amenu &Org.&Hyperlinks.Perma-compre&ss\ links<tab>,lc :set conceallevel=3\|set concealcursor=nc<cr>
|
||
amenu &Org.&Hyperlinks.&Autocompress\ links<tab>,la :set conceallevel=3\|set concealcursor=c<cr>
|
||
amenu &Org.&Hyperlinks.No\ auto&compress\ links<tab>,lx :set conceallevel=0<cr>
|
||
amenu &Org.&Table.$Table\ Dashboard<tab>,b :call OrgTableDashboard()<cr>
|
||
amenu &Org.&Table.E$valuate\ Table<tab>,v :call OrgTableDashboard()<cr>
|
||
amenu &Org.-Sep3- :
|
||
amenu <silent> &Org.TODO\ &Dashboard<tab>,t :call OrgTodoDashboard()<CR>
|
||
amenu <silent> &Org.TODO\ &Cycle<tab><s-cr> :call <SID>ReplaceTodo()<CR>
|
||
"amenu <silent> &Org.TODO\ Cycle\ &Backward<tab><c-s-cr> :call <SID>ReplaceTodo('todo-bkwd')<CR>
|
||
amenu &Org.Edit\ TA&GS<tab>,et :call OrgTagsEdit()<cr>
|
||
amenu &Org.&Dates\ and\ Scheduling.Add/Edit\ &Deadline<tab>,dd :call OrgDateEdit('DEADLINE')<cr>
|
||
amenu &Org.&Dates\ and\ Scheduling.Add/Edit\ &Scheduled<tab>,ds :call OrgDateEdit('SCHEDULED')<cr>
|
||
amenu &Org.&Dates\ and\ Scheduling.Add/Edit\ &Closed<tab>,dc :call OrgDateEdit('CLOSED')<cr>
|
||
amenu &Org.&Dates\ and\ Scheduling.Add/Edit\ &Timestamp<tab>,dt :call OrgDateEdit('TIMESTAMP')<cr>
|
||
amenu &Org.&Dates\ and\ Scheduling.Add/Edit\ &GenericDate<tab>,dg :call OrgDateEdit('ATCURSOR')<cr>
|
||
amenu &Org.&Logging\ work.Clock\ in<tab>,ci :call OrgClockIn(line('.'))<cr>
|
||
amenu &Org.&Logging\ work.Clock\ out<tab>,co :call OrgClockOut()<cr>
|
||
amenu &Org.-Sep4- :
|
||
amenu &Org.Agenda\ command<tab>,ag :call OrgAgendaDashboard()<cr>
|
||
amenu &Org.Special\ &views\ current\ file :call OrgCustomSearchMenu()<cr>
|
||
amenu &Org.Agenda\ &files<tab>,af.&Edit\ Agenda\ Files :EditAgendaFiles<cr>
|
||
amenu &Org.Agenda\ &files<tab>,af.Current\ file\ to\ &top :call {sid}CurrentToAgendaFiles('top')<cr>
|
||
amenu &Org.Agenda\ &files<tab>,af.Current\ file\ to\ &bottom :call {sid}CurrentToAgendaFiles('bottom')<cr>
|
||
amenu &Org.Agenda\ &files<tab>,af.Current\ file\ &remove :call {sid}CurrentRemoveFromAgendaFiles()<cr>
|
||
amenu &Org.Agenda\ &files<tab>,af.Cycle\ to\ &next :call {sid}CycleAgendaFiles('forward')<cr>
|
||
amenu &Org.Agenda\ &files<tab>,af.Cycle\ to\ &previous :call {sid}CycleAgendaFiles('backward')<cr>
|
||
amenu &Org.Agenda\ &files<tab>,af.&Choose\ file\ to\ goto :call {sid}OrgGotoChosenFile()<cr>
|
||
amenu &Org.-Sep45- :
|
||
amenu <silent> &Org.&Do\ Emacs\ Eval<tab>,v :call OrgEval()<cr>
|
||
amenu &Org.-Sep5- :
|
||
amenu &Org.Narro&w.Outline\ &Subtree<tab>,ns :call NarrowCodeBlock(line('.'))<cr>
|
||
amenu &Org.Narro&w.&Code\ Block<tab>,nc :call NarrowCodeBlock(line('.'))<cr>
|
||
amenu &Org.-Sep6- :
|
||
amenu &Org.Open\ Capture\ Buffer :call org#CaptureBuffer()<cr>
|
||
amenu &Org.-Sep7- :
|
||
amenu &Org.Export/Publish\ w/Emacs :call OrgExportDashboard()<cr>
|
||
amenu &Org.-Sep8- :
|
||
amenu <silent> &Org.R&e-read\ Config\ Lines :call OrgProcessConfigLines()<cr>
|
||
|
||
"*********************************************************************
|
||
"*********************************************************************
|
||
" 'endif' below is special 'endif' closing the 'if !exists(org_loaded)
|
||
" line near top of file. Thus the main functions are loaded
|
||
" only once for all buffers, with settngs at begin of file
|
||
" and mappings below this line executed for each buffer
|
||
" having org filetype
|
||
"*********************************************************************
|
||
"*********************************************************************
|
||
endif
|
||
|
||
let g:org_loaded=1
|
||
let b:v.org_loaded=1
|
||
"*********************************************************************
|
||
"*********************************************************************
|
||
"*********************************************************************
|
||
"*********************************************************************
|
||
" convert to VimOrganizer tag format and add colon (:) before dates
|
||
PreLoadTags
|
||
" below is default todo setup, anything different can be done
|
||
" in vimrc (or in future using a config line in the org file itself)
|
||
if !exists('g:in_agenda_search') && ( &foldmethod!= 'expr') && !exists('b:v.bufloaded')
|
||
setlocal foldmethod=expr
|
||
"setlocal foldexpr=org#fold#level(v:lnum)
|
||
set foldlevel=1
|
||
let b:v.bufloaded=1
|
||
else
|
||
setlocal foldmethod=manual
|
||
endif
|
||
"if !exists('b:v.todoitems')
|
||
" call OrgTodoSetup('TODO | DONE')
|
||
"endif
|
||
if !exists('g:org_todo_setup')
|
||
let g:org_todo_setup = 'TODO | DONE'
|
||
endif
|
||
if !exists('g:org_tag_setup')
|
||
let g:org_tag_setup = '{home(h) work(w)}'
|
||
endif
|
||
|
||
call OrgProcessConfigLines()
|
||
"syntax match OL1 +^\(*\)\{1}\s.*+
|
||
"syntax match OL2 +^\(*\)\{2}\s.*+
|
||
"syntax match OL3 +^\(*\)\{3}\s.*+ contains=CONTAINED
|
||
"exec "syntax match DONETODO '" . b:v.todoDoneMatch . "' containedin=OL1,OL2,OL3,OL4,OL5,OL6"
|
||
"exec "syntax match DONETODO '" . b:v.todoDoneMatch . "' containedin=OL1,OL2,OL3,OL4,OL5,OL6"
|
||
"exec "syntax match NOTDONETODO '" . b:v.todoNotDoneMatch . "' containedin=OL1,OL2,OL3,OL4,OL5,OL6"
|
||
"using matchadd b/c syntax match wouldn't show up as contained in OL group
|
||
"with start of '^'
|
||
|
||
"Menu stuff
|
||
function! MenuCycle()
|
||
if foldclosed(line('.')) > -1
|
||
exec foldclosed(line('.'))
|
||
else
|
||
exec s:OrgGetHead()
|
||
endif
|
||
call OrgCycle()
|
||
endfunction
|
||
|
||
nmap <silent> <buffer> <localleader>t :call OrgTodoDashboard()<CR>
|
||
nmap <silent> <buffer> <s-CR> :call <SID>ReplaceTodo()<CR>
|
||
nmap <silent> <buffer> <s-right> :call <SID>ReplaceTodo()<CR>
|
||
nmap <silent> <buffer> <localleader><space> :call <SID>ToggleHeadingMark(line('.'))<CR>
|
||
nmap <silent> <buffer> <localleader><c-space> :call <SID>DeleteHeadingMarks()<CR>
|
||
" c-s-cr already taken
|
||
nmap <silent> <buffer> <s-left> :call <SID>ReplaceTodo('todo-bkwd')<CR>
|
||
if !has('gui_running')
|
||
nmap <silent> <buffer> <localleader>nt :call <SID>ReplaceTodo()<CR>
|
||
endif
|
||
execute "source " . expand("<sfile>:p:h") . '/vimorg-main-mappings.vim'
|
||
" next line call is to get signs defined
|
||
call s:DeleteSigns()
|
||
" below is autocmd to change tw for lines that have comments on them
|
||
" I think this should go in vimrc so i runs for each buffer load
|
||
" :autocmd CursorMoved,CursorMovedI * :if match(getline(line(".")), '^*\*\s') == 0 | :setlocal textwidth=99 | :else | :setlocal textwidth=79 | :endif
|
||
set com=sO::\ -,mO::\ \ ,eO:::,::,sO:>\ -,mO:>\ \ ,eO:>>,:>
|
||
set fo=qtcwn
|
||
let b:v.current_syntax = "org"
|
||
setlocal foldtext=OrgFoldText()
|
||
|
||
"user can define OrgCustomSettings() in vimrc to change default settings
|
||
"in org buffers. If syntax or highlight related, though, should
|
||
"make changes in OrgCustomColors()
|
||
if exists('*OrgCustomSettings')
|
||
call OrgCustomSettings()
|
||
endif
|
||
|
||
|
||
" vim:set nofoldenable:
|