介绍

Ewoc 是 Emacs’s Widget for Object Collections的简写,它可以根据lisp对象的结构绘制和更新buffer的文本,就像MVC设计模式中的视图层。其中,生成的buffer的文本分为三个部分:特定的头部文本,lisp对象代表的数据元素的文本,特定的底部文本。一个ewoc对象的信息包含在以下内容中:

  • 用于产生文本的buffer
  • buffer中文本的起始位置
  • 头部和底部文本的内容
  • 一个双向链接的结点链,每一个结点包含:
    • 一个数据元素(单个lisp对象)
    • 结点链中上一个和下一个结点的链接
  • 一个将数据元素的文本插入到buffer中的打印函数

使用 ewoc-create 来定义一个ewoc对象,然后用其他的ewoc函数构建结点的结构并在buffer中显示。一旦在buffer中显示了文本,便可使用其他函数负责buffer的光标位置和结点间的数据通信。

让结点包含数据元素就像给变量设置值。通常结点包含数据元素的行为发生在将结点加入到ewoc对象过程中。可以通过下面两个函数获取和设置数据元素的值:

(ewoc-data node)
⇒ value

(ewoc-set-data node new-value)
⇒ new-value

也可以使用lisp对象(list or vector)或其他结构的索引作为数据元素的值。

当数据改变时,buffer中文本会相应的更新。使用 ewoc-update 函数更新所有结点,或者用 ewoc-invalidate ,使用 ewoc-map 函数给所有的结点增加条件判断。同时,可以使用 ewoc-deleteewoc-filter 函数删除无效的结点并设置新的结点。删除一个结点的同时会删掉该结点在buffer中展示的文本。

函数

这一部分的术语中, ewocnode 分别代表前面提到的两种结构,而 data 代表作为数据元素的lisp对象。三者的关系是:ewoc包含node,node包含data。下面详细介绍ewoc的每个函数。

ewoc-create pretty-printer &optional header footer nosep

创建并返回一个不包含结点的ewoc对象。 pretty-printer 是包含一个数据元素参数的函数,该数据元素的文本值将被插入到buffer中。(使用 insert 函数插入,不要用 insert-before-markers ,因为它对ewoc包的内部机制有干扰)

通常,该函数会自动在头部文本,底部文本和每一个node的文本后插入新的一行。设置 nosep 参数为 non-nil 可以不插入新行,这在需要将整个ewoc的文本显示在一行时很有用。ewoc创建时会在当前buffer下维护文本,所以在调用创建函数前要先切换到目标buffer。

ewoc-buffer ewoc

返回维护ewoc文本的buffer。

ewoc-get-hf ewoc

返回一个由头部文本和底部文本构成的 cons cell (header . footer)。

ewoc-enter-first ewoc data
ewoc-enter-last ewoc data

分别在ewoc的结点链的开头和结尾添加包含数据元素的新结点。

ewoc-enter-before ewoc node data
ewoc-enter-after ewoc node data

分别在指定结点的前面和后面添加一个包含数据元素的新结点。

ewoc-prev ewoc node
ewoc-next ewoc node

分别返回指定结点的前一个和后一个结点。

ewoc-nth ewoc n

返回以0开始的索引n的结点,n为负值时从最后开始索引。n超出范围时返回nil。

ewoc-data node

获取并返回node包含的数据元素。

ewoc-set-data node

设置node包含的数据元素值为data。

ewoc-locate ewoc &optional pos guess

确定并返回包含光标位置的结点。如果ewoc没有结点,返回nil。如果 pos 在第一个结点之前,返回第一个结点;在最后一个结点之后,返回最后一个结点。可选参数 guess 是有可能在pos附近的结点,它不会改变函数的结果但会让函数执行的更快。

ewoc-location node

返回结点的起始位置。

ewoc-goto-prev ewoc arg
ewoc-goto-next ewoc arg

分别移动光标到上arg个或下arg个结点。如果已经在第一个结点或ewoc为空, ewoc-goto-prev 不移动光标并返回nil,同样 ewoc-goto-next 不超过最后一个结点。除了以上特殊情况,函数返回移动到的结点。

ewoc-goto-node ewoc node

移动光标到结点的起始位置。

ewoc-refresh ewoc

重新生成ewoc的文本。执行过程是:删除header和footer之间的文本,然后为每一个结点依次调用 pretty-printer 函数生成文本。

ewoc-invalidate ewoc &rest nodes

ewoc-refresh 类似,但只更新 nodes 列表内的结点而不是所有结点。

ewoc-delete ewoc &rest nodes

删除所有 nodes 列表内的结点。

ewoc-filter ewoc predicate &rest args

为ewoc中的每一个数据元素调用 predicate 函数,删除断言为nil的结点。 args 是传递给断言函数的参数。

ewoc-collect ewoc predicate &rest args

为ewoc中的每一个数据元素调用 predicate 函数,返回断言为non-nil的元素列表。元素在列表中的顺序和buffer中一致。 args 是传递给断言函数的参数。

ewoc-map map-function ewoc &rest args

为ewoc中的每一个数据元素调用 map-function , 更新map函数返回为non-nil的结点。 args 是传递给map函数的参数。

例子

下面是使用ewoc实现的显示颜色组成的例子,颜色组成由buffer中的三个整数组成的向量表示。

(setq colorcomp-ewoc nil
      colorcomp-data nil
      colorcomp-mode-map nil
      colorcomp-labels ["Red" "Green" "Blue"])

(defun colorcomp-pp (data)
  (if data
      (let ((comp (aref colorcomp-data data)))
        (insert (aref colorcomp-labels data) "\t: #x"
                (format "%02X" comp) " "
                (make-string (ash comp -2) ?#) "\n"))
    (let ((cstr (format "#%02X%02X%02X"
                        (aref colorcomp-data 0)
                        (aref colorcomp-data 1)
                        (aref colorcomp-data 2)))
          (samp " (sample text) "))
      (insert "Color\t: "
              (propertize samp 'face
                          `(foreground-color . ,cstr))
              (propertize samp 'face
                          `(background-color . ,cstr))
              "\n"))))

(defun colorcomp (color)
  "Allow fiddling with COLOR in a new buffer.
     The buffer is in Color Components mode."
  (interactive "sColor (name or #RGB or #RRGGBB): ")
  (when (string= "" color)
    (setq color "green"))
  (unless (color-values color)
    (error "No such color: %S" color))
  (switch-to-buffer
   (generate-new-buffer (format "originally: %s" color)))
  (kill-all-local-variables)
  (setq major-mode 'colorcomp-mode
        mode-name "Color Components")
  (use-local-map colorcomp-mode-map)
  (erase-buffer)
  (buffer-disable-undo)
  (let ((data (apply 'vector (mapcar (lambda (n) (ash n -8))
                                     (color-values color))))
        (ewoc (ewoc-create 'colorcomp-pp
                           "\nColor Components\n\n"
                           (substitute-command-keys
                            "\n\\{colorcomp-mode-map}"))))
    (set (make-local-variable 'colorcomp-data) data)
    (set (make-local-variable 'colorcomp-ewoc) ewoc)
    (ewoc-enter-last ewoc 0)
    (ewoc-enter-last ewoc 1)
    (ewoc-enter-last ewoc 2)
    (ewoc-enter-last ewoc nil)))

通过定义改变 colorcomp-data 的值,完成选择过程和按键绑定,这个例子可以拓展成一个颜色选择的组件(MVC模式中的模型)。

(defun colorcomp-mod (index limit delta)
  (let ((cur (aref colorcomp-data index)))
    (unless (= limit cur)
      (aset colorcomp-data index (+ cur delta)))
    (ewoc-invalidate
     colorcomp-ewoc
     (ewoc-nth colorcomp-ewoc index)
     (ewoc-nth colorcomp-ewoc -1))))

(defun colorcomp-R-more () (interactive) (colorcomp-mod 0 255 1))
(defun colorcomp-G-more () (interactive) (colorcomp-mod 1 255 1))
(defun colorcomp-B-more () (interactive) (colorcomp-mod 2 255 1))
(defun colorcomp-R-less () (interactive) (colorcomp-mod 0 0 -1))
(defun colorcomp-G-less () (interactive) (colorcomp-mod 1 0 -1))
(defun colorcomp-B-less () (interactive) (colorcomp-mod 2 0 -1))

(defun colorcomp-copy-as-kill-and-exit ()
  "Copy the color components into the kill ring and kill the buffer.
     The string is formatted #RRGGBB (hash followed by six hex digits)."
  (interactive)
  (kill-new (format "#%02X%02X%02X"
                    (aref colorcomp-data 0)
                    (aref colorcomp-data 1)
                    (aref colorcomp-data 2)))
  (kill-buffer nil))

(setq colorcomp-mode-map
      (let ((m (make-sparse-keymap)))
        (suppress-keymap m)
        (define-key m "i" 'colorcomp-R-less)
        (define-key m "o" 'colorcomp-R-more)
        (define-key m "k" 'colorcomp-G-less)
        (define-key m "l" 'colorcomp-G-more)
        (define-key m "," 'colorcomp-B-less)
        (define-key m "." 'colorcomp-B-more)
        (define-key m " " 'colorcomp-copy-as-kill-and-exit)
        m))

参考

GNU Emacs Lisp Reference Manual - Abstract-Display