" Copyright (c) 1998-2002 " Michael Sharpe " " We grant permission to use, copy modify, distribute, and sell this " software for any purpose without fee, provided that the above copyright " notice and this text are not removed. We make no guarantee about the " suitability of this software for any purpose and we are not liable " for any damages resulting from its use. Further, we are under no " obligation to maintain or extend this software. It is provided on an " "as is" basis without any expressed or implied warranty. if exists("loaded_alternateFile") finish endif let loaded_alternateFile = 1 " setup the default set of alternate extensions. The user can override in thier " .vimrc if the defaults are not suitable. To override in a .vimrc simply set a " g:alternateExtensions_ variable to a comma separated list of alternates, " where is the extension to map. " E.g. let g:alternateExtensions_CPP = "inc,h,H,HPP,hpp" " Function : AddAlternateExtensionMapping (PRIVATE) " Purpose : simple helper function to add the default alternate extension " mappings. " Args : extension -- the extension to map " alternates -- comma separated list of alternates extensions " Returns : nothing " Author : Michael Sharpe function! AddAlternateExtensionMapping(extension, alternates) let varName = "g:alternateExtensions_" . a:extension if (!exists(varName)) let g:alternateExtensions_{a:extension} = a:alternates endif endfunction " Add all the default extensions call AddAlternateExtensionMapping('h',"c,cpp,cxx,cc,CC") call AddAlternateExtensionMapping('H',"C,CPP,CXX,CC") call AddAlternateExtensionMapping('hpp',"cpp,c") call AddAlternateExtensionMapping('HPP',"CPP,C") call AddAlternateExtensionMapping('c',"h") call AddAlternateExtensionMapping('C',"H") call AddAlternateExtensionMapping('cpp',"h,hpp") call AddAlternateExtensionMapping('CPP',"H,HPP") call AddAlternateExtensionMapping('cc',"h") call AddAlternateExtensionMapping('CC',"H,h") call AddAlternateExtensionMapping('cxx',"h") call AddAlternateExtensionMapping('CXX',"H") call AddAlternateExtensionMapping('psl',"ph") call AddAlternateExtensionMapping('ph',"psl") call AddAlternateExtensionMapping('adb',"ads") call AddAlternateExtensionMapping('ads',"adb") " Function : GetNthExtensionFromSpec (PRIVATE) " Purpose : Use to iterate all the extensions in an extension spec " Args : extSpec -- the extension spec to iterate " n -- the extension to get " Returns : the nth extension from the extension spec, or "" for failure " Author : Michael Sharpe function! GetNthExtensionFromSpec(extSpec, n) let extStart = 0 let extEnd = -1 let pos = 0 let ext = "" let i = 0 while (i != a:n) let extStart = extEnd + 1 let extEnd = match(a:extSpec, ",", extStart) let i = i + 1 if (extEnd == -1) if (i == a:n) let extEnd = strlen(a:extSpec) endif break endif endwhile if (extEnd != -1) let ext = strpart(a:extSpec, extStart, extEnd - extStart) endif return ext endfunction " Function : AlternateFile (PUBLIC) " Purpose : Opens a new buffer by looking at the extension of the current " buffer and finding the corresponding file. E.g. foo.c <--> foo.h " Args : accepts one argument. If present it used the argument as the new " extension. " Returns : nothing " Author : Michael Sharpe function! AlternateFile(splitWindow, ...) let newFilename = "" let baseName = expand("%<") if (a:0 != 0) let newFilename = baseName . "." . a:1 else let currentFile = expand("%") let extension = fnamemodify(currentFile,":e") let extSpec = "" silent! let extSpec = g:alternateExtensions_{extension} if (extSpec != "") let firstFilename = "" let foundMatch = 0 let n = 1 while (!foundMatch) let ext = GetNthExtensionFromSpec(extSpec, n) if (ext != "") let newFilename = baseName . "." . ext let existsCheck = BufferOrFileExists(newFilename) if (existsCheck == 1) let foundMatch = 1 elseif (n == 1) " save the first for posible later use let firstFilename = newFilename endif else break endif let n = n + 1 endwhile if (foundMatch == 0 && firstFilename != "") let newFilename = firstFilename endif endif endif if (newFilename != "") call FindOrCreateBuffer(newFilename, a:splitWindow) else echo "No alternate file available" endif endfunction comm! -nargs=? A call AlternateFile("", ) comm! -nargs=? AS call AlternateFile("h", ) comm! -nargs=? AV call AlternateFile("v", ) " Function : BufferOrFileExists (PRIVATE) " Purpose : determines if a buffer or a readable file exists " Args : name (IN) - name of the buffer/file to check " Returns : TRUE if it exists, FALSE otherwise " Author : Michael Sharpe function! BufferOrFileExists(name) let result = bufexists(a:name) || filereadable(a:name) return result endfunction " Function : FindOrCreateBuffer (PRIVATE) " Purpose : searches the buffer list (:ls) for the specified filename. If " found, checks the window list for the buffer. If the buffer is in " an already open window, it switches to the window. If the buffer " was not in a window, it switches to that buffer. If the buffer did " not exist, it creates it. " Args : filename (IN) -- the name of the file " doSplit (IN) -- indicates whether the window should be split " ("v", "h", "") " Returns : nothing " Author : Michael Sharpe function! FindOrCreateBuffer(filename, doSplit) " Check to see if the buffer is already open before re-opening it. let bufName = bufname(a:filename) if (bufName == "") " Buffer did not exist....create it if (a:doSplit == "h") execute ":split " . a:filename elseif (a:doSplit == "v") execute ":vsplit " . a:filename else execute ":e " . a:filename endif else " Buffer was already open......check to see if it is in a window let bufWindow = bufwinnr(a:filename) if (bufWindow == -1) if (a:doSplit == "h") execute ":sbuffer " . a:filename elseif (a:doSplit == "v") execute ":vert sbuffer " . a:filename else execute ":buffer " . a:filename endif else " search the windows for the target window if bufWindow != winnr() " only search if the current window does not contain the buffer execute "normal \b" let winNum = winnr() while (winNum != bufWindow && winNum > 0) execute "normal \k" let winNum = winNum - 1 endwhile if (0 == winNum) " something wierd happened...open the buffer if (a:doSplit == "h") execute ":split " . a:filename elseif (a:doSplit == "v") execute ":vsplit " . a:filename else execute ":e " . a:filename endif endif endif endif endif endfunction