mirror of
https://git.savannah.gnu.org/git/emacs.git
synced 2025-01-19 18:13:55 +00:00
bf247b6ed0
Change development version from 21.3.50 to 22.0.50.
737 lines
28 KiB
EmacsLisp
737 lines
28 KiB
EmacsLisp
;;; tree-widget.el --- Tree widget
|
||
|
||
;; Copyright (C) 2004 Free Software Foundation, Inc.
|
||
|
||
;; Author: David Ponce <david@dponce.com>
|
||
;; Maintainer: David Ponce <david@dponce.com>
|
||
;; Created: 16 Feb 2001
|
||
;; Keywords: extensions
|
||
|
||
;; This file is part of GNU Emacs
|
||
|
||
;; This program is free software; you can redistribute it and/or
|
||
;; modify it under the terms of the GNU General Public License as
|
||
;; published by the Free Software Foundation; either version 2, or (at
|
||
;; your option) any later version.
|
||
|
||
;; This program is distributed in the hope that it will be useful, but
|
||
;; WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||
;; General Public License for more details.
|
||
|
||
;; You should have received a copy of the GNU General Public License
|
||
;; along with this program; see the file COPYING. If not, write to
|
||
;; the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
||
;; Boston, MA 02111-1307, USA.
|
||
|
||
;;; Commentary:
|
||
;;
|
||
;; This library provide a tree widget useful to display data
|
||
;; structures organized in a hierarchical order.
|
||
;;
|
||
;; The following properties are specific to the tree widget:
|
||
;;
|
||
;; :open
|
||
;; Set to non-nil to unfold the tree. By default the tree is
|
||
;; folded.
|
||
;;
|
||
;; :node
|
||
;; Specify the widget used to represent a tree node. By default
|
||
;; this is an `item' widget which displays the tree-widget :tag
|
||
;; property value if defined or a string representation of the
|
||
;; tree-widget value.
|
||
;;
|
||
;; :keep
|
||
;; Specify a list of properties to keep when the tree is
|
||
;; folded so they can be recovered when the tree is unfolded.
|
||
;; This property can be used in child widgets too.
|
||
;;
|
||
;; :dynargs
|
||
;; Specify a function to be called when the tree is unfolded, to
|
||
;; dynamically provide the tree children in response to an unfold
|
||
;; request. This function will be passed the tree widget and
|
||
;; must return a list of child widgets. That list will be stored
|
||
;; as the :args property of the parent tree.
|
||
|
||
;; To speed up successive unfold requests, the :dynargs function
|
||
;; can directly return the :args value if non-nil. Refreshing
|
||
;; child values can be achieved by giving the :args property the
|
||
;; value nil, then redrawing the tree.
|
||
;;
|
||
;; :has-children
|
||
;; Specify if this tree has children. This property has meaning
|
||
;; only when used with the above :dynargs one. It indicates that
|
||
;; child widgets exist but will be dynamically provided when
|
||
;; unfolding the node.
|
||
;;
|
||
;; :open-control (default `tree-widget-open-control')
|
||
;; :close-control (default `tree-widget-close-control')
|
||
;; :empty-control (default `tree-widget-empty-control')
|
||
;; :leaf-control (default `tree-widget-leaf-control')
|
||
;; :guide (default `tree-widget-guide')
|
||
;; :end-guide (default `tree-widget-end-guide')
|
||
;; :no-guide (default `tree-widget-no-guide')
|
||
;; :handle (default `tree-widget-handle')
|
||
;; :no-handle (default `tree-widget-no-handle')
|
||
;;
|
||
;; The above nine properties define the widgets used to draw the tree.
|
||
;; For example, using widgets that display this values:
|
||
;;
|
||
;; open-control "[-] "
|
||
;; close-control "[+] "
|
||
;; empty-control "[X] "
|
||
;; leaf-control "[>] "
|
||
;; guide " |"
|
||
;; noguide " "
|
||
;; end-guide " `"
|
||
;; handle "-"
|
||
;; no-handle " "
|
||
;;
|
||
;; A tree will look like this:
|
||
;;
|
||
;; [-] 1 open-control
|
||
;; |-[+] 1.0 guide+handle+close-control
|
||
;; |-[X] 1.1 guide+handle+empty-control
|
||
;; `-[-] 1.2 end-guide+handle+open-control
|
||
;; |-[>] 1.2.1 no-guide+no-handle+guide+handle+leaf-control
|
||
;; `-[>] 1.2.2 no-guide+no-handle+end-guide+handle+leaf-control
|
||
;;
|
||
;; By default, the tree widget try to use images instead of strings to
|
||
;; draw a nice-looking tree. See the `tree-widget-themes-directory'
|
||
;; and `tree-widget-theme' options for more details.
|
||
;;
|
||
|
||
;;; History:
|
||
;;
|
||
|
||
;;; Code:
|
||
(eval-when-compile (require 'cl))
|
||
(require 'wid-edit)
|
||
|
||
;;; Customization
|
||
;;
|
||
(defgroup tree-widget nil
|
||
"Customization support for the Tree Widget Library."
|
||
:version "22.1"
|
||
:group 'widgets)
|
||
|
||
(defcustom tree-widget-image-enable
|
||
(not (or (featurep 'xemacs) (< emacs-major-version 21)))
|
||
"*non-nil means that tree-widget will try to use images."
|
||
:type 'boolean
|
||
:group 'tree-widget)
|
||
|
||
(defcustom tree-widget-themes-directory "tree-widget"
|
||
"*Name of the directory where to lookup for image themes.
|
||
When nil use the directory where the tree-widget library is located.
|
||
When a relative name is specified, try to locate that sub-directory in
|
||
`load-path', then in the data directory, and use the first one found.
|
||
Default is to search for a \"tree-widget\" sub-directory.
|
||
|
||
The data directory is the value of:
|
||
- the variable `data-directory' on GNU Emacs;
|
||
- `(locate-data-directory \"tree-widget\")' on XEmacs."
|
||
:type '(choice (const :tag "Default" "tree-widget")
|
||
(const :tag "With the library" nil)
|
||
(directory :format "%{%t%}:\n%v"))
|
||
:group 'tree-widget)
|
||
|
||
(defcustom tree-widget-theme nil
|
||
"*Name of the theme to use to lookup for images.
|
||
The theme name must be a subdirectory in `tree-widget-themes-directory'.
|
||
If nil use the \"default\" theme.
|
||
When a image is not found in the current theme, the \"default\" theme
|
||
is searched too.
|
||
A complete theme should contain images with these file names:
|
||
|
||
Name Represents
|
||
----------- ------------------------------------------------
|
||
open opened node (for example an open folder)
|
||
close closed node (for example a close folder)
|
||
empty empty node (a node without children)
|
||
leaf leaf node (for example a document)
|
||
guide a vertical guide line
|
||
no-guide an invisible guide line
|
||
end-guide the end of a vertical guide line
|
||
handle an horizontal line drawn before a node control
|
||
no-handle an invisible handle
|
||
----------- ------------------------------------------------"
|
||
:type '(choice (const :tag "Default" nil)
|
||
(string :tag "Name"))
|
||
:group 'tree-widget)
|
||
|
||
(defcustom tree-widget-image-properties-emacs
|
||
'(:ascent center :mask (heuristic t))
|
||
"*Properties of GNU Emacs images."
|
||
:type 'plist
|
||
:group 'tree-widget)
|
||
|
||
(defcustom tree-widget-image-properties-xemacs
|
||
nil
|
||
"*Properties of XEmacs images."
|
||
:type 'plist
|
||
:group 'tree-widget)
|
||
|
||
;;; Image support
|
||
;;
|
||
(eval-when-compile ;; GNU Emacs/XEmacs compatibility stuff
|
||
(cond
|
||
;; XEmacs
|
||
((featurep 'xemacs)
|
||
(defsubst tree-widget-use-image-p ()
|
||
"Return non-nil if image support is currently enabled."
|
||
(and tree-widget-image-enable
|
||
widget-glyph-enable
|
||
(console-on-window-system-p)))
|
||
(defsubst tree-widget-create-image (type file &optional props)
|
||
"Create an image of type TYPE from FILE.
|
||
Give the image the specified properties PROPS.
|
||
Return the new image."
|
||
(apply 'make-glyph `([,type :file ,file ,@props])))
|
||
(defsubst tree-widget-image-formats ()
|
||
"Return the list of image formats, file name suffixes associations.
|
||
See also the option `widget-image-file-name-suffixes'."
|
||
(delq nil
|
||
(mapcar
|
||
#'(lambda (fmt)
|
||
(and (valid-image-instantiator-format-p (car fmt)) fmt))
|
||
widget-image-file-name-suffixes)))
|
||
)
|
||
;; GNU Emacs
|
||
(t
|
||
(defsubst tree-widget-use-image-p ()
|
||
"Return non-nil if image support is currently enabled."
|
||
(and tree-widget-image-enable
|
||
widget-image-enable
|
||
(display-images-p)))
|
||
(defsubst tree-widget-create-image (type file &optional props)
|
||
"Create an image of type TYPE from FILE.
|
||
Give the image the specified properties PROPS.
|
||
Return the new image."
|
||
(apply 'create-image `(,file ,type nil ,@props)))
|
||
(defsubst tree-widget-image-formats ()
|
||
"Return the list of image formats, file name suffixes associations.
|
||
See also the option `widget-image-conversion'."
|
||
(delq nil
|
||
(mapcar
|
||
#'(lambda (fmt)
|
||
(and (image-type-available-p (car fmt)) fmt))
|
||
widget-image-conversion)))
|
||
))
|
||
)
|
||
|
||
;; Buffer local cache of theme data.
|
||
(defvar tree-widget--theme nil)
|
||
|
||
(defsubst tree-widget-theme-name ()
|
||
"Return the current theme name, or nil if no theme is active."
|
||
(and tree-widget--theme (aref tree-widget--theme 0)))
|
||
|
||
(defsubst tree-widget-set-theme (&optional name)
|
||
"In the current buffer, set the theme to use for images.
|
||
The current buffer should be where the tree widget is drawn.
|
||
Optional argument NAME is the name of the theme to use, which defaults
|
||
to the value of the variable `tree-widget-theme'.
|
||
Does nothing if NAME is the name of the current theme."
|
||
(or name (setq name (or tree-widget-theme "default")))
|
||
(unless (equal name (tree-widget-theme-name))
|
||
(set (make-local-variable 'tree-widget--theme)
|
||
(make-vector 4 nil))
|
||
(aset tree-widget--theme 0 name)))
|
||
|
||
(defun tree-widget-themes-directory ()
|
||
"Locate the directory where to search for a theme.
|
||
It is defined in variable `tree-widget-themes-directory'.
|
||
Return the absolute name of the directory found, or nil if the
|
||
specified directory is not accessible."
|
||
(let ((found (aref tree-widget--theme 1)))
|
||
(if found
|
||
;; The directory is available in the cache.
|
||
(unless (eq found 'void) found)
|
||
(cond
|
||
;; Use the directory where tree-widget is located.
|
||
((null tree-widget-themes-directory)
|
||
(setq found (locate-library "tree-widget"))
|
||
(when found
|
||
(setq found (file-name-directory found))
|
||
(or (file-accessible-directory-p found)
|
||
(setq found nil))))
|
||
;; Check accessibility of absolute directory name.
|
||
((file-name-absolute-p tree-widget-themes-directory)
|
||
(setq found (expand-file-name tree-widget-themes-directory))
|
||
(or (file-accessible-directory-p found)
|
||
(setq found nil)))
|
||
;; Locate a sub-directory in `load-path' and data directory.
|
||
(t
|
||
(let ((path
|
||
(append load-path
|
||
;; The data directory depends on which, GNU
|
||
;; Emacs or XEmacs, is running.
|
||
(list (if (fboundp 'locate-data-directory)
|
||
(locate-data-directory "tree-widget")
|
||
data-directory)))))
|
||
(while (and path (not found))
|
||
(when (car path)
|
||
(setq found (expand-file-name
|
||
tree-widget-themes-directory (car path)))
|
||
(or (file-accessible-directory-p found)
|
||
(setq found nil)))
|
||
(setq path (cdr path))))))
|
||
;; Store the result in the cache for later use.
|
||
(aset tree-widget--theme 1 (or found 'void))
|
||
found)))
|
||
|
||
(defsubst tree-widget-set-image-properties (props)
|
||
"In current theme, set images properties to PROPS."
|
||
(aset tree-widget--theme 2 props))
|
||
|
||
(defun tree-widget-image-properties (file)
|
||
"Return properties of images in current theme.
|
||
If the \"tree-widget-theme-setup.el\" file exists in the directory
|
||
where is located the image FILE, load it to setup theme images
|
||
properties. Typically that file should contain something like this:
|
||
|
||
(tree-widget-set-image-properties
|
||
(if (featurep 'xemacs)
|
||
'(:ascent center)
|
||
'(:ascent center :mask (heuristic t))
|
||
))
|
||
|
||
By default, use the global properties provided in variables
|
||
`tree-widget-image-properties-emacs' or
|
||
`tree-widget-image-properties-xemacs'."
|
||
;; If properties are in the cache, use them.
|
||
(or (aref tree-widget--theme 2)
|
||
(progn
|
||
;; Load tree-widget-theme-setup if available.
|
||
(load (expand-file-name
|
||
"tree-widget-theme-setup"
|
||
(file-name-directory file)) t t)
|
||
;; If properties have been setup, use them.
|
||
(or (aref tree-widget--theme 2)
|
||
;; By default, use supplied global properties.
|
||
(tree-widget-set-image-properties
|
||
(if (featurep 'xemacs)
|
||
tree-widget-image-properties-xemacs
|
||
tree-widget-image-properties-emacs))))))
|
||
|
||
(defun tree-widget-find-image (name)
|
||
"Find the image with NAME in current theme.
|
||
NAME is an image file name sans extension.
|
||
Search first in current theme, then in default theme.
|
||
A theme is a sub-directory of the root theme directory specified in
|
||
variable `tree-widget-themes-directory'.
|
||
Return the first image found having a supported format in those
|
||
returned by the function `tree-widget-image-formats', or nil if not
|
||
found."
|
||
(when (tree-widget-use-image-p)
|
||
;; Ensure there is an active theme.
|
||
(tree-widget-set-theme (tree-widget-theme-name))
|
||
;; If the image is in the cache, return it.
|
||
(or (cdr (assoc name (aref tree-widget--theme 3)))
|
||
;; Search the image in the current, then default themes.
|
||
(let ((default-directory (tree-widget-themes-directory)))
|
||
(when default-directory
|
||
(let* ((theme (tree-widget-theme-name))
|
||
(path (mapcar 'expand-file-name
|
||
(if (equal theme "default")
|
||
'("default")
|
||
(list theme "default"))))
|
||
(formats (tree-widget-image-formats))
|
||
(found
|
||
(catch 'found
|
||
(dolist (dir path)
|
||
(dolist (fmt formats)
|
||
(dolist (ext (cdr fmt))
|
||
(let ((file (expand-file-name
|
||
(concat name ext) dir)))
|
||
(and (file-readable-p file)
|
||
(file-regular-p file)
|
||
(throw 'found
|
||
(cons (car fmt) file)))))))
|
||
nil)))
|
||
(when found
|
||
(let ((image
|
||
(tree-widget-create-image
|
||
(car found) (cdr found)
|
||
(tree-widget-image-properties (cdr found)))))
|
||
;; Store image in the cache for later use.
|
||
(push (cons name image) (aref tree-widget--theme 3))
|
||
image))))))))
|
||
|
||
;;; Widgets
|
||
;;
|
||
(defvar tree-widget-button-keymap
|
||
(let (parent-keymap mouse-button1 keymap)
|
||
(if (featurep 'xemacs)
|
||
(setq parent-keymap widget-button-keymap
|
||
mouse-button1 [button1])
|
||
(setq parent-keymap widget-keymap
|
||
mouse-button1 [down-mouse-1]))
|
||
(setq keymap (copy-keymap parent-keymap))
|
||
(define-key keymap mouse-button1 'widget-button-click)
|
||
keymap)
|
||
"Keymap used inside node handle buttons.")
|
||
|
||
(define-widget 'tree-widget-control 'push-button
|
||
"Base `tree-widget' control."
|
||
:format "%[%t%]"
|
||
:button-keymap tree-widget-button-keymap ; XEmacs
|
||
:keymap tree-widget-button-keymap ; Emacs
|
||
)
|
||
|
||
(define-widget 'tree-widget-open-control 'tree-widget-control
|
||
"Control widget that represents a opened `tree-widget' node."
|
||
:tag "[-] "
|
||
;;:tag-glyph (tree-widget-find-image "open")
|
||
:notify 'tree-widget-close-node
|
||
:help-echo "Hide node"
|
||
)
|
||
|
||
(define-widget 'tree-widget-empty-control 'tree-widget-open-control
|
||
"Control widget that represents an empty opened `tree-widget' node."
|
||
:tag "[X] "
|
||
;;:tag-glyph (tree-widget-find-image "empty")
|
||
)
|
||
|
||
(define-widget 'tree-widget-close-control 'tree-widget-control
|
||
"Control widget that represents a closed `tree-widget' node."
|
||
:tag "[+] "
|
||
;;:tag-glyph (tree-widget-find-image "close")
|
||
:notify 'tree-widget-open-node
|
||
:help-echo "Show node"
|
||
)
|
||
|
||
(define-widget 'tree-widget-leaf-control 'item
|
||
"Control widget that represents a leaf node."
|
||
:tag " " ;; Need at least a char to display the image :-(
|
||
;;:tag-glyph (tree-widget-find-image "leaf")
|
||
:format "%t"
|
||
)
|
||
|
||
(define-widget 'tree-widget-guide 'item
|
||
"Widget that represents a guide line."
|
||
:tag " |"
|
||
;;:tag-glyph (tree-widget-find-image "guide")
|
||
:format "%t"
|
||
)
|
||
|
||
(define-widget 'tree-widget-end-guide 'item
|
||
"Widget that represents the end of a guide line."
|
||
:tag " `"
|
||
;;:tag-glyph (tree-widget-find-image "end-guide")
|
||
:format "%t"
|
||
)
|
||
|
||
(define-widget 'tree-widget-no-guide 'item
|
||
"Widget that represents an invisible guide line."
|
||
:tag " "
|
||
;;:tag-glyph (tree-widget-find-image "no-guide")
|
||
:format "%t"
|
||
)
|
||
|
||
(define-widget 'tree-widget-handle 'item
|
||
"Widget that represent a node handle."
|
||
:tag " "
|
||
;;:tag-glyph (tree-widget-find-image "handle")
|
||
:format "%t"
|
||
)
|
||
|
||
(define-widget 'tree-widget-no-handle 'item
|
||
"Widget that represent an invisible node handle."
|
||
:tag " "
|
||
;;:tag-glyph (tree-widget-find-image "no-handle")
|
||
:format "%t"
|
||
)
|
||
|
||
(define-widget 'tree-widget 'default
|
||
"Tree widget."
|
||
:format "%v"
|
||
:convert-widget 'widget-types-convert-widget
|
||
:value-get 'widget-value-value-get
|
||
:value-create 'tree-widget-value-create
|
||
:value-delete 'tree-widget-value-delete
|
||
)
|
||
|
||
;;; Widget support functions
|
||
;;
|
||
(defun tree-widget-p (widget)
|
||
"Return non-nil if WIDGET is a `tree-widget' widget."
|
||
(let ((type (widget-type widget)))
|
||
(while (and type (not (eq type 'tree-widget)))
|
||
(setq type (widget-type (get type 'widget-type))))
|
||
(eq type 'tree-widget)))
|
||
|
||
(defsubst tree-widget-get-super (widget property)
|
||
"Return WIDGET's inherited PROPERTY value."
|
||
(widget-get (get (widget-type (get (widget-type widget)
|
||
'widget-type))
|
||
'widget-type)
|
||
property))
|
||
|
||
(defsubst tree-widget-super-format-handler (widget escape)
|
||
"Call WIDGET's inherited format handler to process ESCAPE character."
|
||
(let ((handler (tree-widget-get-super widget :format-handler)))
|
||
(and handler (funcall handler widget escape))))
|
||
|
||
(defun tree-widget-format-handler (widget escape)
|
||
"For WIDGET, signal that the %p format template is obsolete.
|
||
Call WIDGET's inherited format handler to process other ESCAPE
|
||
characters."
|
||
(if (eq escape ?p)
|
||
(message "The %%p format template is obsolete and ignored")
|
||
(tree-widget-super-format-handler widget escape)))
|
||
(make-obsolete 'tree-widget-format-handler
|
||
'tree-widget-super-format-handler)
|
||
|
||
(defsubst tree-widget-node (widget)
|
||
"Return the tree WIDGET :node value.
|
||
If not found setup a default 'item' widget."
|
||
(let ((node (widget-get widget :node)))
|
||
(unless node
|
||
(setq node `(item :tag ,(or (widget-get widget :tag)
|
||
(widget-princ-to-string
|
||
(widget-value widget)))))
|
||
(widget-put widget :node node))
|
||
node))
|
||
|
||
(defsubst tree-widget-open-control (widget)
|
||
"Return the opened node control specified in WIDGET."
|
||
(or (widget-get widget :open-control)
|
||
'tree-widget-open-control))
|
||
|
||
(defsubst tree-widget-close-control (widget)
|
||
"Return the closed node control specified in WIDGET."
|
||
(or (widget-get widget :close-control)
|
||
'tree-widget-close-control))
|
||
|
||
(defsubst tree-widget-empty-control (widget)
|
||
"Return the empty node control specified in WIDGET."
|
||
(or (widget-get widget :empty-control)
|
||
'tree-widget-empty-control))
|
||
|
||
(defsubst tree-widget-leaf-control (widget)
|
||
"Return the leaf node control specified in WIDGET."
|
||
(or (widget-get widget :leaf-control)
|
||
'tree-widget-leaf-control))
|
||
|
||
(defsubst tree-widget-guide (widget)
|
||
"Return the guide line widget specified in WIDGET."
|
||
(or (widget-get widget :guide)
|
||
'tree-widget-guide))
|
||
|
||
(defsubst tree-widget-end-guide (widget)
|
||
"Return the end of guide line widget specified in WIDGET."
|
||
(or (widget-get widget :end-guide)
|
||
'tree-widget-end-guide))
|
||
|
||
(defsubst tree-widget-no-guide (widget)
|
||
"Return the invisible guide line widget specified in WIDGET."
|
||
(or (widget-get widget :no-guide)
|
||
'tree-widget-no-guide))
|
||
|
||
(defsubst tree-widget-handle (widget)
|
||
"Return the node handle line widget specified in WIDGET."
|
||
(or (widget-get widget :handle)
|
||
'tree-widget-handle))
|
||
|
||
(defsubst tree-widget-no-handle (widget)
|
||
"Return the node invisible handle line widget specified in WIDGET."
|
||
(or (widget-get widget :no-handle)
|
||
'tree-widget-no-handle))
|
||
|
||
(defun tree-widget-keep (arg widget)
|
||
"Save in ARG the WIDGET properties specified by :keep."
|
||
(dolist (prop (widget-get widget :keep))
|
||
(widget-put arg prop (widget-get widget prop))))
|
||
|
||
(defun tree-widget-children-value-save (widget &optional args node)
|
||
"Save WIDGET children values.
|
||
Children properties and values are saved in ARGS if non-nil else in
|
||
WIDGET :args property value. Data node properties and value are saved
|
||
in NODE if non-nil else in WIDGET :node property value."
|
||
(let ((args (or args (widget-get widget :args)))
|
||
(node (or node (tree-widget-node widget)))
|
||
(children (widget-get widget :children))
|
||
(node-child (widget-get widget :tree-widget--node))
|
||
arg child)
|
||
(while (and args children)
|
||
(setq arg (car args)
|
||
args (cdr args)
|
||
child (car children)
|
||
children (cdr children))
|
||
(if (tree-widget-p child)
|
||
;;;; The child is a tree node.
|
||
(progn
|
||
;; Backtrack :args and :node properties.
|
||
(widget-put arg :args (widget-get child :args))
|
||
(widget-put arg :node (tree-widget-node child))
|
||
;; Save :open property.
|
||
(widget-put arg :open (widget-get child :open))
|
||
;; The node is open.
|
||
(when (widget-get child :open)
|
||
;; Save the widget value.
|
||
(widget-put arg :value (widget-value child))
|
||
;; Save properties specified in :keep.
|
||
(tree-widget-keep arg child)
|
||
;; Save children.
|
||
(tree-widget-children-value-save
|
||
child (widget-get arg :args) (widget-get arg :node))))
|
||
;;;; Another non tree node.
|
||
;; Save the widget value
|
||
(widget-put arg :value (widget-value child))
|
||
;; Save properties specified in :keep.
|
||
(tree-widget-keep arg child)))
|
||
(when (and node node-child)
|
||
;; Assume that the node child widget is not a tree!
|
||
;; Save the node child widget value.
|
||
(widget-put node :value (widget-value node-child))
|
||
;; Save the node child properties specified in :keep.
|
||
(tree-widget-keep node node-child))
|
||
))
|
||
|
||
(defvar tree-widget-after-toggle-functions nil
|
||
"Hooks run after toggling a `tree-widget' folding.
|
||
Each function will receive the `tree-widget' as its unique argument.
|
||
This variable should be local to each buffer used to display
|
||
widgets.")
|
||
|
||
(defun tree-widget-close-node (widget &rest ignore)
|
||
"Close the `tree-widget' node associated to this control WIDGET.
|
||
WIDGET's parent should be a `tree-widget'.
|
||
IGNORE other arguments."
|
||
(let ((tree (widget-get widget :parent)))
|
||
;; Before folding the node up, save children values so next open
|
||
;; can recover them.
|
||
(tree-widget-children-value-save tree)
|
||
(widget-put tree :open nil)
|
||
(widget-value-set tree nil)
|
||
(run-hook-with-args 'tree-widget-after-toggle-functions tree)))
|
||
|
||
(defun tree-widget-open-node (widget &rest ignore)
|
||
"Open the `tree-widget' node associated to this control WIDGET.
|
||
WIDGET's parent should be a `tree-widget'.
|
||
IGNORE other arguments."
|
||
(let ((tree (widget-get widget :parent)))
|
||
(widget-put tree :open t)
|
||
(widget-value-set tree t)
|
||
(run-hook-with-args 'tree-widget-after-toggle-functions tree)))
|
||
|
||
(defun tree-widget-value-delete (widget)
|
||
"Delete tree WIDGET children."
|
||
;; Delete children
|
||
(widget-children-value-delete widget)
|
||
;; Delete node child
|
||
(widget-delete (widget-get widget :tree-widget--node))
|
||
(widget-put widget :tree-widget--node nil))
|
||
|
||
(defun tree-widget-value-create (tree)
|
||
"Create the TREE widget."
|
||
(let* ((widget-image-enable (tree-widget-use-image-p)) ; Emacs
|
||
(widget-glyph-enable widget-image-enable) ; XEmacs
|
||
(node (tree-widget-node tree))
|
||
children buttons)
|
||
(if (widget-get tree :open)
|
||
;;;; Unfolded node.
|
||
(let* ((args (widget-get tree :args))
|
||
(dynargs (widget-get tree :dynargs))
|
||
(flags (widget-get tree :tree-widget--guide-flags))
|
||
(rflags (reverse flags))
|
||
(guide (tree-widget-guide tree))
|
||
(noguide (tree-widget-no-guide tree))
|
||
(endguide (tree-widget-end-guide tree))
|
||
(handle (tree-widget-handle tree))
|
||
(nohandle (tree-widget-no-handle tree))
|
||
;; Lookup for images and set widgets' tag-glyphs here,
|
||
;; to allow to dynamically change the image theme.
|
||
(guidi (tree-widget-find-image "guide"))
|
||
(noguidi (tree-widget-find-image "no-guide"))
|
||
(endguidi (tree-widget-find-image "end-guide"))
|
||
(handli (tree-widget-find-image "handle"))
|
||
(nohandli (tree-widget-find-image "no-handle"))
|
||
child)
|
||
(when dynargs
|
||
;; Request the definition of dynamic children
|
||
(setq dynargs (funcall dynargs tree))
|
||
;; Unless children have changed, reuse the widgets
|
||
(unless (eq args dynargs)
|
||
(setq args (mapcar 'widget-convert dynargs))
|
||
(widget-put tree :args args)))
|
||
;; Insert the node control
|
||
(push (widget-create-child-and-convert
|
||
tree (if args (tree-widget-open-control tree)
|
||
(tree-widget-empty-control tree))
|
||
:tag-glyph (tree-widget-find-image
|
||
(if args "open" "empty")))
|
||
buttons)
|
||
;; Insert the node element
|
||
(widget-put tree :tree-widget--node
|
||
(widget-create-child-and-convert tree node))
|
||
;; Insert children
|
||
(while args
|
||
(setq child (car args)
|
||
args (cdr args))
|
||
;; Insert guide lines elements
|
||
(dolist (f rflags)
|
||
(widget-create-child-and-convert
|
||
tree (if f guide noguide)
|
||
:tag-glyph (if f guidi noguidi))
|
||
(widget-create-child-and-convert
|
||
tree nohandle :tag-glyph nohandli)
|
||
)
|
||
(widget-create-child-and-convert
|
||
tree (if args guide endguide)
|
||
:tag-glyph (if args guidi endguidi))
|
||
;; Insert the node handle line
|
||
(widget-create-child-and-convert
|
||
tree handle :tag-glyph handli)
|
||
;; If leaf node, insert a leaf node control
|
||
(unless (tree-widget-p child)
|
||
(push (widget-create-child-and-convert
|
||
tree (tree-widget-leaf-control tree)
|
||
:tag-glyph (tree-widget-find-image "leaf"))
|
||
buttons))
|
||
;; Insert the child element
|
||
(push (widget-create-child-and-convert
|
||
tree child
|
||
:tree-widget--guide-flags (cons (if args t) flags))
|
||
children)))
|
||
;;;; Folded node.
|
||
;; Insert the closed node control
|
||
(push (widget-create-child-and-convert
|
||
tree (tree-widget-close-control tree)
|
||
:tag-glyph (tree-widget-find-image "close"))
|
||
buttons)
|
||
;; Insert the node element
|
||
(widget-put tree :tree-widget--node
|
||
(widget-create-child-and-convert tree node)))
|
||
;; Save widget children and buttons
|
||
(widget-put tree :children (nreverse children))
|
||
(widget-put tree :buttons buttons)
|
||
))
|
||
|
||
;;; Utilities
|
||
;;
|
||
(defun tree-widget-map (widget fun)
|
||
"For each WIDGET displayed child call function FUN.
|
||
FUN is called with three arguments like this:
|
||
|
||
(FUN CHILD IS-NODE WIDGET)
|
||
|
||
where:
|
||
- - CHILD is the child widget.
|
||
- - IS-NODE is non-nil if CHILD is WIDGET node widget."
|
||
(when (widget-get widget :tree-widget--node)
|
||
(funcall fun (widget-get widget :tree-widget--node) t widget)
|
||
(dolist (child (widget-get widget :children))
|
||
(if (tree-widget-p child)
|
||
;; The child is a tree node.
|
||
(tree-widget-map child fun)
|
||
;; Another non tree node.
|
||
(funcall fun child nil widget)))))
|
||
|
||
(provide 'tree-widget)
|
||
|
||
;;; arch-tag: c3a1ada2-1663-41dc-9d16-2479ed8320e8
|
||
;;; tree-widget.el ends here
|