<?xml version="1.0" encoding="utf-8" standalone="yes"?><rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom" xmlns:content="http://purl.org/rss/1.0/modules/content/"><channel><title>Lisp on Xerxes II's Blog</title><link>/tags/lisp/</link><description>Recent content in Lisp on Xerxes II's Blog</description><generator>Hugo</generator><language>zh-cn</language><lastBuildDate>Sun, 03 May 2026 00:00:00 +1000</lastBuildDate><atom:link href="/tags/lisp/index.xml" rel="self" type="application/rss+xml"/><item><title>用 Scheme 给 Helix 写 Wakatime 插件</title><link>/posts/helix-wakatime-plugin-in-scheme/</link><pubDate>Sun, 03 May 2026 00:00:00 +1000</pubDate><guid>/posts/helix-wakatime-plugin-in-scheme/</guid><description>&lt;p&gt;我日常用的编辑器是 &lt;a href="https://helix-editor.com/"&gt;Helix&lt;/a&gt;。我从来没深入用过 Vim，所以转到 Helix 并没有什么适应成本——反而更自然：Helix 开箱即用的体验很好，而且&amp;quot;先选择再操作&amp;quot;（select-then-act）的编辑模型比 Vim 的&amp;quot;先动词再名词&amp;quot;对我来说更直觉。&lt;/p&gt;
&lt;p&gt;Helix 主线目前没有插件系统，这一直是社区里最大的痛点之一。不过 &lt;a href="https://github.com/mattwparas"&gt;mattwparas&lt;/a&gt; 维护了一个 fork（&lt;a href="https://github.com/mattwparas/helix/tree/steel-event-system"&gt;&lt;code&gt;steel-event-system&lt;/code&gt; 分支&lt;/a&gt;），集成了 &lt;a href="https://github.com/mattwparas/steel"&gt;Steel&lt;/a&gt;——一个嵌入式 Scheme 实现——作为插件运行时。Scheme 是 Lisp 的一个方言，而 Steel 是 Scheme 的一个方言，所以你大概能猜到写插件的体验是什么画风：很多括号。&lt;/p&gt;
&lt;p&gt;我在 AUR 上打包了这个 fork：&lt;a href="https://aur.archlinux.org/packages/helix-steel-git"&gt;helix-steel-git&lt;/a&gt;，Arch 用户可以直接装。&lt;/p&gt;
&lt;p&gt;其实我很早就知道这个 fork 的存在，但一直没有真正去用——原因很简单，我是真的怕括号。然而最近我彻底抛弃了 VSCode，全面转向 Helix 作为唯一的编辑器，随之而来的问题是：对插件的需求越来越强烈。而 Steel 生态目前能用的插件屈指可数，想要什么功能基本只能自己动手。于是我终于硬着头皮开始写 Scheme 了。&lt;/p&gt;
&lt;h2 id="第一版让-llm-来"&gt;第一版：让 LLM 来&lt;/h2&gt;
&lt;p&gt;第一步是从社区已有的插件里复制了一套模板，建好 &lt;a href="https://github.com/Xerxes-2/wakatime.hx"&gt;wakatime.hx&lt;/a&gt; 仓库，然后直接把需求丢给 Claude，让它生成第一版代码。毕竟我对 Lisp 一无所知，连基本语法都得现查。&lt;/p&gt;
&lt;p&gt;Claude 确实给了我一个能跑的版本。问题是——LLM 生成的 Lisp 代码，人类读起来比手写的还要痛苦。比如获取编辑器版本号这段：&lt;/p&gt;
&lt;div class="highlight"&gt;&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;-webkit-text-size-adjust:none;"&gt;&lt;code class="language-scheme" data-lang="scheme"&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;(&lt;span style="color:#66d9ef"&gt;define &lt;/span&gt;(&lt;span style="color:#a6e22e"&gt;editor-version&lt;/span&gt;)
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (&lt;span style="color:#66d9ef"&gt;if &lt;/span&gt;*wakatime-editor-version*
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; *wakatime-editor-version*
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (&lt;span style="color:#a6e22e"&gt;begin&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (&lt;span style="color:#66d9ef"&gt;let &lt;/span&gt;([spawned (&lt;span style="color:#a6e22e"&gt;spawn-process&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (&lt;span style="color:#a6e22e"&gt;with-stdout-piped&lt;/span&gt; (&lt;span style="color:#a6e22e"&gt;command&lt;/span&gt; &lt;span style="color:#e6db74"&gt;&amp;#34;hx&amp;#34;&lt;/span&gt; (list &lt;span style="color:#e6db74"&gt;&amp;#34;--version&amp;#34;&lt;/span&gt;))))])
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (&lt;span style="color:#66d9ef"&gt;if &lt;/span&gt;(&lt;span style="color:#a6e22e"&gt;Ok?&lt;/span&gt; spawned)
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (&lt;span style="color:#66d9ef"&gt;let &lt;/span&gt;([output (&lt;span style="color:#a6e22e"&gt;wait-&amp;gt;stdout&lt;/span&gt; (&lt;span style="color:#a6e22e"&gt;Ok-&amp;gt;value&lt;/span&gt; spawned))])
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (&lt;span style="color:#66d9ef"&gt;if &lt;/span&gt;(&lt;span style="color:#a6e22e"&gt;Ok?&lt;/span&gt; output)
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (&lt;span style="color:#66d9ef"&gt;let &lt;/span&gt;([version (&lt;span style="color:#a6e22e"&gt;extract-version&lt;/span&gt; (&lt;span style="color:#a6e22e"&gt;Ok-&amp;gt;value&lt;/span&gt; output))])
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; (&lt;span style="color:#66d9ef"&gt;set! &lt;/span&gt;*wakatime-editor-version* version)
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; version)
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; &lt;span style="color:#e6db74"&gt;&amp;#34;unknown&amp;#34;&lt;/span&gt;))
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; &lt;span style="color:#e6db74"&gt;&amp;#34;unknown&amp;#34;&lt;/span&gt;)))))
&lt;/span&gt;&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;p&gt;嵌套的 &lt;code&gt;if&lt;/code&gt;-&lt;code&gt;let&lt;/code&gt;-&lt;code&gt;if&lt;/code&gt;-&lt;code&gt;let&lt;/code&gt;-&lt;code&gt;if&lt;/code&gt;，层层剥洋葱式地处理 &lt;code&gt;Ok?&lt;/code&gt; 结果——典型的&amp;quot;LLM 不知道这门语言的惯用写法，只好用最保守的方式堆逻辑&amp;quot;。能跑，但读起来让人头大。&lt;/p&gt;</description></item></channel></rss>