?? namespaces.vim
字號:
let szBeginPart = remove(listNamespace, 0) " Is 'Ns1' an alias ? if has_key(a:mapNamespaceAlias, szBeginPart) " Resolving alias 'Ns1' " eg: Ns1 = NsResolved let szResult = a:mapNamespaceAlias[szBeginPart] " szEndPart = 'Ns2::Ns3' let szEndPart = join(listNamespace, '::') if szEndPart != '' " Concatenation => szResult = 'NsResolved::Ns2::Ns3' let szResult .= '::' . szEndPart endif " If a:szNamespace starts with '::' we add '::' to the beginning " of the result if match(a:szNamespace, '^::')>=0 let szResult = omni#cpp#utils#SimplifyScope('::' . szResult) endif endif endif return szResultendfunc" Resolve namespace aliasfunction! s:ResolveAliasInNamespaceList(mapNamespaceAlias, listNamespaces) call map(a:listNamespaces, 'omni#cpp#namespaces#ResolveAlias(a:mapNamespaceAlias, v:val)')endfunc" Get namespaces used at the cursor postion in a vim buffer" Note: The result depends on the current cursor position" @return" - List of namespace used in the reverse orderfunction! omni#cpp#namespaces#GetUsingNamespaces() " We have to get local using namespace declarations " We need the current cursor position and the position of the start of the " current scope " We store the cursor position because searchpairpos() moves the cursor let result = [] let originalPos = getpos('.') let origPos = originalPos[1:2] let stopPos = s:GetStopPositionForLocalSearch() let stopLine = stopPos[0] let curPos = origPos let lastLine = 0 let nextStopLine = origPos[0] while curPos !=[0,0] let curPos = searchpos('\C}\|\(using\s\+namespace\)', 'bW',stopLine) if curPos!=[0,0] && curPos[0]!=lastLine let lastLine = curPos[0] let szLine = getline('.') if origPos[0] == curPos[0] " We get the line until cursor position let szLine = szLine[:origPos[1]] endif let szLine = omni#cpp#utils#GetCodeFromLine(szLine) if match(szLine, '\Cusing\s\+namespace')<0 " We found a '}' let curPos = searchpairpos('{', '', '}', 'bW', g:omni#cpp#utils#expIgnoreComments) else " We get the namespace list from the line let result = s:GetNamespaceListFromLine(szLine) + result let nextStopLine = curPos[0] endif endif endwhile " Setting the cursor to the original position call setpos('.', originalPos) " 2) Now we can get all global using namespace declaration from the " beginning of the file to nextStopLine let result = omni#cpp#namespaces#GetListFromCurrentBuffer(nextStopLine) + result " Resolving alias in the namespace list " TODO: For the next release "let g:omni#cpp#namespaces#CacheAlias= s:GetNamespaceAliasMap() "call s:ResolveAliasInNamespaceList(g:omni#cpp#namespaces#CacheAlias, result) return ['::'] + resultendfunc" Resolve a using namespace regarding the current context" For each namespace used:" - We get all possible contexts where the namespace" can be define" - We do a comparison test of each parent contexts with the current" context list" - If one and only one parent context is present in the" current context list we add the namespace in the current" context" - If there is more than one of parent contexts in the" current context the namespace is ambiguous" @return" - result item" - kind = 0|1" - 0 = unresolved or error" - 1 = resolved" - value = resolved namespacefunction! s:ResolveNamespace(namespace, mapCurrentContexts) let result = {'kind':0, 'value': ''} " If the namespace is already resolved we add it in the list of " current contexts if match(a:namespace, '^::')>=0 let result.kind = 1 let result.value = a:namespace return result elseif match(a:namespace, '\w\+::\w\+')>=0 let mapCurrentContextsTmp = copy(a:mapCurrentContexts) let resolvedItem = {} for nsTmp in split(a:namespace, '::') let resolvedItem = s:ResolveNamespace(nsTmp, mapCurrentContextsTmp) if resolvedItem.kind " Note: We don't extend the map let mapCurrentContextsTmp = {resolvedItem.value : 1} else break endif endfor if resolvedItem!={} && resolvedItem.kind let result.kind = 1 let result.value = resolvedItem.value endif return result endif " We get all possible parent contexts of this namespace let listTagsOfNamespace = [] if has_key(g:omni#cpp#namespaces#CacheResolve, a:namespace) let listTagsOfNamespace = g:omni#cpp#namespaces#CacheResolve[a:namespace] else let listTagsOfNamespace = omni#common#utils#TagList('^'.a:namespace.'$') let g:omni#cpp#namespaces#CacheResolve[a:namespace] = listTagsOfNamespace endif if len(listTagsOfNamespace)==0 return result endif call filter(listTagsOfNamespace, 'v:val.kind[0]=="n"') " We extract parent context from tags " We use a map to avoid multiple entries let mapContext = {} for tagItem in listTagsOfNamespace let szParentContext = omni#cpp#utils#ExtractScope(tagItem) let mapContext[szParentContext] = 1 endfor let listParentContext = keys(mapContext) " Now for each parent context we test if the context is in the current " contexts list let listResolvedNamespace = [] for szParentContext in listParentContext if has_key(a:mapCurrentContexts, szParentContext) call extend(listResolvedNamespace, [omni#cpp#utils#SimplifyScope(szParentContext.'::'.a:namespace)]) endif endfor " Now we know if the namespace is ambiguous or not let len = len(listResolvedNamespace) if len==1 " Namespace resolved let result.kind = 1 let result.value = listResolvedNamespace[0] elseif len > 1 " Ambiguous namespace, possible matches are in listResolvedNamespace else " Other cases endif return resultendfunc" Resolve namespaces"@return" - List of resolved namespacesfunction! omni#cpp#namespaces#ResolveAll(namespacesUsed) " We add the default context '::' let contextOrder = 0 let mapCurrentContexts = {} " For each namespace used: " - We get all possible contexts where the namespace " can be define " - We do a comparison test of each parent contexts with the current " context list " - If one and only one parent context is present in the " current context list we add the namespace in the current " context " - If there is more than one of parent contexts in the " current context the namespace is ambiguous for ns in a:namespacesUsed let resolvedItem = s:ResolveNamespace(ns, mapCurrentContexts) if resolvedItem.kind let contextOrder+=1 let mapCurrentContexts[resolvedItem.value] = contextOrder endif endfor " Build the list of current contexts from the map, we have to keep the " order let mapReorder = {} for key in keys(mapCurrentContexts) let mapReorder[ mapCurrentContexts[key] ] = key endfor let result = [] for key in sort(keys(mapReorder)) call extend(result, [mapReorder[key]]) endfor return resultendfunc" Build the context stackfunction! s:BuildContextStack(namespaces, szCurrentScope) let result = copy(a:namespaces) if a:szCurrentScope != '::' let tagItem = omni#cpp#utils#GetResolvedTagItem(a:namespaces, omni#cpp#utils#CreateTypeInfo(a:szCurrentScope)) if has_key(tagItem, 'inherits') let listBaseClass = omni#cpp#utils#GetClassInheritanceList(a:namespaces, omni#cpp#utils#CreateTypeInfo(a:szCurrentScope)) let result = listBaseClass + result elseif has_key(tagItem, 'kind') && index(['c', 's', 'u'], tagItem.kind[0])>=0 call insert(result, omni#cpp#utils#ExtractTypeInfoFromTag(tagItem)) endif endif return resultendfunc" Returns the class scope at the current position of the cursor" @return a string that represents the class scope" eg: ::NameSpace1::Class1" The returned string always starts with '::'" Note: In term of performance it's the weak point of the scriptfunction! s:GetClassScopeAtCursor() " We store the cursor position because searchpairpos() moves the cursor let originalPos = getpos('.') let endPos = originalPos[1:2] let listCode = [] let result = {'namespaces': [], 'scope': ''} while endPos!=[0,0] let endPos = searchpairpos('{', '', '}', 'bW', g:omni#cpp#utils#expIgnoreComments) let szReStartPos = '[;{}]\|\%^' let startPos = searchpairpos(szReStartPos, '', '{', 'bWn', g:omni#cpp#utils#expIgnoreComments) " If the file starts with a comment so the startPos can be [0,0] " we change it to [1,1] if startPos==[0,0] let startPos = [1,1] endif " Get lines backward from cursor position to last ; or { or } " or when we are at the beginning of the file. " We store lines in listCode if endPos!=[0,0] " We remove the last character which is a '{' " We also remove starting { or } or ; if exits let szCodeWithoutComments = substitute(omni#cpp#utils#GetCode(startPos, endPos)[:-2], '^[;{}]', '', 'g') call insert(listCode, {'startLine' : startPos[0], 'code' : szCodeWithoutComments}) endif endwhile " Setting the cursor to the original position call setpos('.', originalPos) let listClassScope = [] let bResolved = 0 let startLine = 0 " Now we can check in the list of code if there is a function for code in listCode " We get the name of the namespace, class, struct or union " and we store it in listClassScope let tokens = omni#cpp#tokenizer#Tokenize(code.code) let bContinue=0 let bAddNamespace = 0 let state=0 for token in tokens if state==0 if index(['namespace', 'class', 'struct', 'union'], token.value)>=0 if token.value == 'namespace' let bAddNamespace = 1 endif let state= 1 " Maybe end of tokens endif elseif state==1 if token.kind == 'cppWord' " eg: namespace MyNs { class MyCl {}; } " => listClassScope = [MyNs, MyCl] call extend( listClassScope , [token.value] ) " Add the namespace in result if bAddNamespace call extend(result.namespaces, [token.value]) let bAddNamespace = 0 endif let bContinue=1 break endif endif endfor if bContinue==1 continue endif " Simple test to check if we have a chance to find a " class method let aPos = matchend(code.code, '::\s*\~*\s*\w\+\s*(') if aPos ==-1 continue endif let startLine = code.startLine let listTmp = [] " eg: 'void MyNamespace::MyClass::foo(' " => tokens = ['MyClass', '::', 'MyNamespace', 'void'] let tokens = reverse(omni#cpp#tokenizer#Tokenize(code.code[:aPos-1])[:-4]) let state = 0 " Reading tokens backward for token in tokens if state==0 if token.kind=='cppWord' call insert(listTmp, token.value) let state=1 endif elseif state==1 if token.value=='::' let state=2 else break endif elseif state==2 if token.kind=='cppWord' call insert(listTmp, token.value) let state=1 else break endif endif endfor if len(listTmp) if len(listClassScope) let bResolved = 1 " Merging class scopes " eg: current class scope = 'MyNs::MyCl1' " method class scope = 'MyCl1::MyCl2' " If we add the method class scope to current class scope " we'll have MyNs::MyCl1::MyCl1::MyCl2 => it's wrong " we want MyNs::MyCl1::MyCl2 let index = 0 for methodClassScope in listTmp if methodClassScope==listClassScope[-1] let listTmp = listTmp[index+1:] break else let index+=1 endif endfor endif call extend(listClassScope, listTmp) break endif endfor let szClassScope = '::' if len(listClassScope) if bResolved let szClassScope .= join(listClassScope, '::') else let szClassScope = join(listClassScope, '::') " The class scope is not resolved, we have to check using " namespace declarations and search the class scope in each " namespace if startLine != 0 let namespaces = ['::'] + omni#cpp#namespaces#GetListFromCurrentBuffer(startLine) let namespaces = omni#cpp#namespaces#ResolveAll(namespaces) let tagItem = omni#cpp#utils#GetResolvedTagItem(namespaces, omni#cpp#utils#CreateTypeInfo(szClassScope)) if tagItem != {} let szClassScope = omni#cpp#utils#ExtractTypeInfoFromTag(tagItem) endif endif endif endif let result.scope = szClassScope return resultendfunc" Get all contexts at the cursor positionfunction! omni#cpp#namespaces#GetContexts() " Get the current class scope at the cursor, the result depends on the current cursor position let scopeItem = s:GetClassScopeAtCursor() let listUsingNamespace = copy(g:OmniCpp_DefaultNamespaces) call extend(listUsingNamespace, scopeItem.namespaces) if g:OmniCpp_NamespaceSearch && &filetype != 'c' " Get namespaces used in the file until the cursor position let listUsingNamespace = omni#cpp#namespaces#GetUsingNamespaces() + listUsingNamespace " Resolving namespaces, removing ambiguous namespaces let namespaces = omni#cpp#namespaces#ResolveAll(listUsingNamespace) else let namespaces = ['::'] + listUsingNamespace endif call reverse(namespaces) " Building context stack from namespaces and the current class scope return s:BuildContextStack(namespaces, scopeItem.scope)endfunc
?? 快捷鍵說明
復制代碼
Ctrl + C
搜索代碼
Ctrl + F
全屏模式
F11
切換主題
Ctrl + Shift + D
顯示快捷鍵
?
增大字號
Ctrl + =
減小字號
Ctrl + -