成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

利用CTags開(kāi)發(fā)一個(gè)Sublime Text代碼補(bǔ)完插件

系統(tǒng) Linux
喜歡使用 Sublime Text 的朋友們都知道,Sublime Text 相當(dāng)于 Linux 上的 Vim,它們都具有很強(qiáng)的可擴(kuò)展功能,功能多樣的同時(shí)速度也很快,對(duì)于處理小型文件和項(xiàng)目效率特別高,因此如果不是特別復(fù)雜的項(xiàng)目,我一般都是用 Sublime Text 編寫(xiě)以及編譯的。

利用CTags開(kāi)發(fā)一個(gè)Sublime Text代碼補(bǔ)完插件

喜歡使用 Sublime Text 的朋友們都知道,Sublime Text 相當(dāng)于 Linux 上的 Vim,它們都具有很強(qiáng)的可擴(kuò)展功能,功能多樣的同時(shí)速度也很快,對(duì)于處理小型文件和項(xiàng)目效率特別高,因此如果不是特別復(fù)雜的項(xiàng)目,我一般都是用 Sublime Text 編寫(xiě)以及編譯的。

然而在用 Sublime Text 開(kāi)發(fā)的過(guò)程中,我發(fā)現(xiàn)了一個(gè)問(wèn)題:Sublime Text 本身的自動(dòng)完成功能只搜索當(dāng)前視圖中正在編輯文件的函數(shù),當(dāng)我想用其他文件中自定義的函數(shù)時(shí),是沒(méi)有自動(dòng)完成功能的。而當(dāng)自定義函數(shù)過(guò)多時(shí),效率會(huì)大大降低,于是我開(kāi)始尋找具有相關(guān)功能的插件。

一開(kāi)始我用了非常熱門(mén)的 “SublimeCodeIntel” 插件,試了一下的確非常好用,但是可惜的是,這個(gè)插件不支持 C/C++,而且占用的空間非常大,追求簡(jiǎn)潔輕便的我不得不另辟蹊徑。后來(lái)又找到一款 “All AutoComplete” 插件,這款插件擴(kuò)展了 Sublime Text 默認(rèn)的自動(dòng)完成功能,可以在當(dāng)前視圖打開(kāi)的所有文件里面尋找定義的函數(shù)和變量,盡管用起來(lái)效果不錯(cuò),但是它的問(wèn)題也很明顯,必須要同時(shí)打開(kāi)多個(gè)文件才行,非常不方便,于是我又放棄了。

在 Package Control 上找了許久,也沒(méi)能找到我想要的插件,于是我開(kāi)始考慮不如自己寫(xiě)一個(gè)這樣的插件,剛好借此機(jī)會(huì)入門(mén) Python。這時(shí)我剛好想到能不能利用 CTags,它能把當(dāng)前項(xiàng)目中的所有自定義函數(shù)提取出來(lái),生成 .tags 文件,并提供符號(hào)跳轉(zhuǎn)功能,只要提取 .tags 文件里面的信息,用正則匹配,然后添加到 Sublime Text 的自動(dòng)完成函數(shù)中不就行了。

為了完成這個(gè)插件,我在網(wǎng)上搜索相關(guān)信息,找到相關(guān)素材并重新構(gòu)思了一下,同時(shí)參考了 All Complete 插件的源碼。

需要提一下,在 Sublime Text 下安裝 CTags 的方法這里不會(huì)提到,因此麻煩各位自行查詢(xún)。

 

插件構(gòu)思

  • 讀取設(shè)置,設(shè)置中添加的語(yǔ)言禁用插件功能

  • 檢測(cè) .tag 文件是否存在,不存在則直接 return

  • 讀取當(dāng)前文件夾中的 .tag 文件

  • 正則匹配函數(shù)名

  • 正則匹配函數(shù)體

  • 添加到自動(dòng)完成的接口上

 

開(kāi)始編寫(xiě)

 

新建插件

剛開(kāi)始接觸 Sublime Text 插件的編寫(xiě),當(dāng)然需要先了解 Sublime Text 提供的各種接口,為此,我去 Sublime Text 的官網(wǎng)找到了相關(guān)文檔:How to Create a Sublime Text Plugin,以及 Sublime Text Unofficial Documentation

首先,在 Sublime Text  中選擇 “Tools -> Developer -> New Plugin” 新建一個(gè)最基本的插件文檔:

  1. import sublime
  2. import sublime_plugin
  3.  
  4. class ExampleCommand(sublime_plugin.TextCommand):
  5. def run(self, edit):
  6. self.view.insert(edit, 0, "Hello, World!")

這里的 sublimesublime_plugin 是 Sublime 必需的模塊,其中具體的類(lèi)和方法可以參考官方的 API Reference

接著,把這個(gè)文件保存到 Package文件夾(默認(rèn)的保存位置 User 文件夾的上一層)的 CTagsAutoComplete 文件夾(新建)下,并命名為 CTagsAutoComplete.py。盡管命名并沒(méi)有什么限制,但***還是以插件的名稱(chēng)來(lái)統(tǒng)一命名。

然后回到 Sublime Text 中,通過(guò)快捷鍵 Ctrl+` 進(jìn)入 Sublime Text 的 Command Console,然后輸入 view.run_command('example'),如果下方顯示 “Hello World”,說(shuō)明插件已經(jīng)正常加載。

這里之所以直接用 'example',是因?yàn)?Command 命令的名稱(chēng)是根據(jù)大寫(xiě)字符進(jìn)行拆分的,例子中的 ExampleCommand 在 Command 中 為 'example_command',直接輸入 'example' 也可以訪(fǎng)問(wèn)。

 

文中的術(shù)語(yǔ)

  • Window:Sublime Text 的當(dāng)前窗口對(duì)象

  • View:Sublime Text 當(dāng)前窗口中打開(kāi)的視圖對(duì)象

  • Command Palette:Sublime Text 中通過(guò)快捷鍵 Ctrl+Shift+P 打開(kāi)的交互式列表

 

確定插件接口類(lèi)型

Sublime Text 下的插件命令有 3 種命令類(lèi)型(都來(lái)自于 sublime_plugin 模塊):

  • TextCommand Class:通過(guò) View 對(duì)象提供對(duì)選定文件/緩沖區(qū)的內(nèi)容的訪(fǎng)問(wèn)。

  • WindowCommand Class:通過(guò) Window 對(duì)象提供當(dāng)前窗口的引用

  • ApplicationCommand Class:這個(gè)類(lèi)沒(méi)有引用任何特定窗口或文件/緩沖區(qū),因此很少使用

2 種事件監(jiān)聽(tīng)類(lèi)型:

  • EventListener Class:監(jiān)聽(tīng) Sublime Text 中各種事件并執(zhí)行一次命令

  • ViewEventListener Class:為 EventListener 提供類(lèi)似事件處理的類(lèi),但綁定到特定的 view。

2 種輸入處理程序:

因?yàn)槲乙獙?shí)現(xiàn)的功能比較簡(jiǎn)單,只需要監(jiān)聽(tīng)輸入事件并觸發(fā)自動(dòng)完成功能,因此需要用到 EventListener Class。在該類(lèi)下面找到了 on_query_completions 方法用來(lái)處理觸發(fā)自動(dòng)完成時(shí)執(zhí)行的命令。接著修改一下剛才的代碼:

  1. import sublime
  2. import sublime_plugin
  3.  
  4. class CTagsAutoComplete(sublime_plugin.EventListener):
  5. def on_query_completions(self, view, prefix, locations):
  • view:當(dāng)前視圖

  • prefix:觸發(fā)自動(dòng)完成時(shí)輸入的文字

  • locations: 觸發(fā)自動(dòng)完成時(shí)輸入在緩存區(qū)中的位置,可以通過(guò)這個(gè)參數(shù)判斷語(yǔ)言來(lái)執(zhí)行不同命令

  • 返回類(lèi)型:

    • return None
    • return [["trigger \t hint", "contents"]...],其中 \t hint 為可選內(nèi)容,給自動(dòng)完成的函數(shù)名稱(chēng)添加一個(gè)提示
    • return (results, flag),其中 results 是包含自動(dòng)完成語(yǔ)句的 list,如上;flag 是一個(gè)額外參數(shù),可用來(lái)控制是否顯示 Sublime Text 自帶的自動(dòng)完成功能

 

讀取 CTags 文件

為了讀取 .tag 文件,首先得判斷當(dāng)前項(xiàng)目是否打開(kāi),同時(shí) .tag 文件是否存在,然后讀取 .tag 文件中的所有內(nèi)容:

  1. import sublime
  2. import sublime_plugin
  3. import os
  4. import re
  5.  
  6. class CTagsAutoComplete(sublime_plugin.EventListener):
  7. def on_query_completions(self, view, prefix, locations):
  8. results = []
  9.  
  10. ctags_paths = [folder + '\.tags' for folder in view.window().folders()]
  11. ctags_rows = []
  12.  
  13. for ctags_path in ctags_paths:
  14. if not is_file_exist(view, ctags_path):
  15. return []
  16. ctags_path = str(ctags_path)
  17. ctags_file = open(ctags_path, encoding = 'utf-8')
  18. ctags_rows += ctags_file.readlines()
  19. ctags_file.close()
  20.  
  21. def is_file_exist(view, file):
  22. if (not view.window().folders() or not os.path.exists(file)):
  23. return False
  24. return True

通過(guò)上述操作,即可讀取當(dāng)前項(xiàng)目下所有的 .tag 文件中的內(nèi)容。

 

分析 CTags 文件

首先是獲取 .tags 文件中,包含 prefix 的行:

  1. for rows in ctags_rows:
  2. target = re.findall('^' + prefix + '.*', rows)

一旦找到,就通過(guò)正則表達(dá)式對(duì)該行數(shù)據(jù)進(jìn)行處理:

  1. if target:
  2. matched = re.split('\t', str(target[0]))
  3. trigger = matched[0] # 返回的***個(gè)參數(shù),函數(shù)名稱(chēng)
  4. trigger += '\t(%s)' % 'CTags' # 給函數(shù)名稱(chēng)后加上標(biāo)識(shí) 'CTags'
  5. contents = re.findall(prefix + '[0-9a-zA-Z_]*\(.*\)', str(matched[2])) # 返回的第二個(gè)參數(shù),函數(shù)的具體定義
  6. if (len(matched) > 1 and contents):
  7. results.append((trigger, contents[0]))
  8. results = list(set(results)) # 去除重復(fù)的函數(shù)
  9. results.sort() # 排序

處理完成之后就可以返回了,考慮到***只顯示 .tags 中的函數(shù),我不需要顯示 Sublime Text 自帶的自動(dòng)完成功能(提取當(dāng)前頁(yè)面中的變量和函數(shù)),因此我的返回結(jié)果如下:

  1. return (results, sublime.INHIBIT_WORD_COMPLETIONS | sublime.INHIBIT_EXPLICIT_COMPLETIONS)

 

添加配置文件

考慮到能夠關(guān)閉插件的功能,因此需要添加一個(gè)配置文件,用來(lái)指定不開(kāi)啟插件功能的語(yǔ)言,這里我參考了 “All AutoComplete” 的代碼:

  1. def plugin_loaded():
  2. global settings
  3. settings = sublime.load_settings('CTagsAutoComplete.sublime-settings')
  4.  
  5. def is_disabled_in(scope):
  6. excluded_scopes = settings.get("exclude_from_completion", [])
  7. for excluded_scope in excluded_scopes:
  8. if scope.find(excluded_scope) != -1:
  9. return True
  10. return False
  11.  
  12. if is_disabled_in(view.scope_name(locations[0])):
  13. return []

這里用到的配置文件需要添加到插件所在的文件夾中,名稱(chēng)為 CTagsAutoComplete.sublime-settings,其內(nèi)容為:

  1. {
  2. // An array of syntax names to exclude from being autocompleted.
  3. "exclude_from_completion": [
  4. "css",
  5. "html"
  6. ]
  7. }

 

添加設(shè)置文件

有了配置文件,還需要在 Sublime Text 的 “Preferences -> Package settings” 下添加相應(yīng)的設(shè)置,同樣也是放在插件所在文件夾中,名稱(chēng)為 Main.sublime-menu

  1. [
  2. {
  3. "caption": "Preferences",
  4. "mnemonic": "n",
  5. "id": "preferences",
  6. "children": [
  7. {
  8. "caption": "Package Settings",
  9. "mnemonic": "P",
  10. "id": "package-settings",
  11. "children": [
  12. {
  13. "caption": "CTagsAutoComplete",
  14. "children": [
  15. {
  16. "command": "open_file",
  17. "args": {
  18. "file": "${packages}/CTagsAutoComplete/CTagsAutoComplete.sublime-settings"
  19. },
  20. "caption": "Settings"
  21. }
  22. ]
  23. }
  24. ]
  25. }
  26. ]
  27. }
  28. ]

 

總結(jié)

首先給出插件的完整源碼:

  1. import sublime
  2. import sublime_plugin
  3. import os
  4. import re
  5.  
  6. def plugin_loaded():
  7. global settings
  8. settings = sublime.load_settings('CTagsAutoComplete.sublime-settings')
  9.  
  10. class CTagsAutoComplete(sublime_plugin.EventListener):
  11. def on_query_completions(self, view, prefix, locations):
  12. if is_disabled_in(view.scope_name(locations[0])):
  13. return []
  14.  
  15. results = []
  16.  
  17. ctags_paths = [folder + '\.tags' for folder in view.window().folders()]
  18. ctags_rows = []
  19.  
  20. for ctags_path in ctags_paths:
  21. if not is_file_exist(view, ctags_path):
  22. return []
  23. ctags_path = str(ctags_path)
  24. ctags_file = open(ctags_path, encoding = 'utf-8')
  25. ctags_rows += ctags_file.readlines()
  26. ctags_file.close()
  27.  
  28. for rows in ctags_rows:
  29. target = re.findall('^' + prefix + '.*', rows)
  30. if target:
  31. matched = re.split('\t', str(target[0]))
  32. trigger = matched[0]
  33. trigger += '\t(%s)' % 'CTags'
  34. contents = re.findall(prefix + '[0-9a-zA-Z_]*\(.*\)', str(matched[2]))
  35. if (len(matched) > 1 and contents):
  36. results.append((trigger, contents[0]))
  37. results = list(set(results))
  38. results.sort()
  39.  
  40. return (results, sublime.INHIBIT_WORD_COMPLETIONS | sublime.INHIBIT_EXPLICIT_COMPLETIONS)
  41.  
  42. def is_disabled_in(scope):
  43. excluded_scopes = settings.get("exclude_from_completion", [])
  44. for excluded_scope in excluded_scopes:
  45. if scope.find(excluded_scope) != -1:
  46. return True
  47. return False
  48.  
  49. def is_file_exist(view, file):
  50. if (not view.window().folders() or not os.path.exists(file)):
  51. return False
  52. return True
  53.  
  54. plugin_loaded()

之后我會(huì)把這個(gè)插件整合好后,上傳到 Package Control 上,從而方便更多人使用。通過(guò)這次入門(mén),我嘗到了甜頭,未來(lái)的開(kāi)發(fā)過(guò)程中,可能會(huì)出現(xiàn)各種各樣獨(dú)特的需求,如果已有的插件無(wú)法提供幫助,那就自己上吧。 

責(zé)任編輯:龐桂玉 來(lái)源: Linux中國(guó)
相關(guān)推薦

2020-05-09 09:31:15

前端開(kāi)發(fā)Sublime Tex擴(kuò)展

2015-03-10 09:23:21

前端開(kāi)發(fā)Sublime插件Sublime

2013-05-29 00:28:24

Sublime Tex開(kāi)發(fā)工具開(kāi)發(fā)

2018-01-08 14:31:09

Electron桌面APP前端

2014-08-20 10:47:58

Lime Text開(kāi)源

2020-06-23 17:30:44

前端Sublime

2022-06-13 06:33:04

瀏覽器瀏覽器插件

2025-04-01 02:00:00

2016-03-08 09:52:22

xcode插件開(kāi)發(fā)

2023-10-26 11:22:32

2010-09-02 13:32:52

jQueryjQuery插件

2020-10-10 06:32:43

IDEA插件環(huán)境

2015-07-15 10:46:00

Sublime Tex

2015-04-30 08:03:36

2020-09-02 07:22:17

JavaScript插件框架

2011-10-19 09:30:23

jQuery

2016-09-30 10:16:39

sublimeswift

2015-09-16 10:21:46

SublimeAPICloudAPP

2017-11-13 13:33:09

MySQL全備份恢復(fù)表

2022-07-20 09:38:31

Python語(yǔ)音合成代碼
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)

主站蜘蛛池模板: www.天堂av.com | 国产精品欧美一区二区 | 国产精品久久久久久福利一牛影视 | 97精品久久 | 精品久久久久久久久久久久久久 | 国产一级毛片精品完整视频版 | 国产日韩精品在线 | 国产精品久久久久久久久久 | 亚洲有码转帖 | 亚洲精品在线播放 | 亚洲狠狠 | 国产成人精品一区二区三 | 亚洲一视频 | av天天干| 久久久久久国产精品免费免费 | 日本三级网站在线 | 久久伊人精品 | 91久久夜色 | 中国大陆高清aⅴ毛片 | 国产亚洲二区 | 久久久久亚洲 | 欧美区日韩区 | 国产成人精品久久二区二区91 | 天天干夜夜操 | 日韩图区| 亚洲免费在线 | 亚洲高清在线 | 免费看a | 国产成人精品一区二区在线 | 九九在线视频 | 婷婷久久五月 | 在线色网址 | 夜夜爽99久久国产综合精品女不卡 | 欧产日产国产精品v | 久久av影院| 在线a视频| 成人性生交大片 | 欧美在线播放一区 | 精精久久 | 久久久久久久久久久丰满 | 羞羞视频网站免费观看 |