平时使用过很多Emacs插件,看过很多插件的源码,也自己写过一些小函数和插件,来提高自己日常使用Emacs的效率。
在折腾的过程中,越来越有这样的感觉:作为开发Emacs插件的主力语言,Emacs Lisp很强大,但也不是无所不能的,也有无能为力的场景,比如和操作系统底层交互、调用OS的API、跨平台能力等,这个时候就在思考,怎样扩展Emacs插件的能力,实现Emacs Lisp不能实现或难以实现的功能。
目前想到的有这样几个途径:
使用Common Lisp的能力
Emacs Lisp就是基于Common Lisp的,Common Lisp提供的能力在Emacs Lisp中大部分都可以使用。
网上有一本Common Lisp的cookbook,值得看看:http://cl-cookbook.sourceforge.net/
使用其他Emacs插件已经提供的能力
Emacs的历史非常悠久,在这么多年里,全世界无数的黑客为Emacs贡献了成千上万的插件,这是一个巨大的宝库,多学习优秀的插件,学习其中的编程技巧、代码结构、设计思想,非常有助于自己技术的提升,在自己开发插件的时候,也可以避免重复造轮子。
emacs-china网站有一个国内的插件镜像仓库,可以经常去搜一搜、折腾一下各种插件:https://elpa.emacs-china.org/
调用C/Python/Java等语言的外部接口
如果经常看Emacs自带的各种函数的帮助文档,会发现有大量的函数是来源于C source code
的built-in
函数,比如base64-encode-string
这个函数的帮助文档中就有这样一行:
base64-encode-string is a built-in function in ‘C source code’.
这说明可以用C语言来编写Emacs Lisp函数。那么更进一步,可以用Python、Java等其他语言吗?我目前不了解,但我相信是有可能做到的。
调用其他可执行程序的命令行接口(Command Line Interface, CLI)
在Windows/Mac/Linux上都有很多操作系统支持的命令,也有很多优秀的软件,有些软件提供了命令行调用的接口,这就为在Emacs Lisp中调用这些软件提供了方便,在Emacs Lisp中可以直接使用shell-command
函数调用一个命令行命令或调用一个可执行文件。
最近写的一个一键往Emacs中粘贴、插入图片的小插件pasteex-mode就是通过调用PasteEx
这个软件的命令行接口实现的,PasteEx
是基于C#的一款用于复制粘贴图片的开源软件。
不过这种方式有一个缺点,就是无法做到跨平台。如果可执行文件只支持某一个平台(比如PasteEx
就只支持Windows平台),那么在其他平台就无法使用这个功能,还需要考虑其他的实现方式。
目前用到的非常好用的其他几个调用CLI实现功能的插件:sift
,grep-dired
,grip
等。
调用网络API
我当前使用的插件中,youdao-dictionary
这个插件就是调用网络API实现插件功能的典范,通过调用有道词典的API,实现查词和翻译的插件功能。这种方式刚好可以解决上面说到的无法跨平台的缺点,但是也有一个缺点就是必须连网。
借助socket技术和其他程序交互(比如chrome浏览器插件)
这是在使用atomic-chrome
这个插件时学习到的一种技术,通过socket技术和chrome浏览器交互,而chrome浏览器加上JS语言的能力,又几乎是无所不能的了,所以这种方式也有很大的想象空间。这种方式也可以解决跨平台的问题。
总结
上面总结了6种扩展Emacs Lisp能力的途径,也许还有更多我现在所不知道的途径,有待继续挖掘、学习和思考,每种技术都值得深入去学习,并且相当有趣,just enjoy it.