-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathidomenu.el
81 lines (73 loc) · 2.93 KB
/
idomenu.el
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
;;; idomenu.el --- imenu tag selection a la ido
;;
;; Copyright (C) 2010 Georg Brandl
;;
;; Author: Georg Brandl <[email protected]>
;; Version: 0.1
;;
;; This file is NOT 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
;; of the License, 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. If not, see <http://www.gnu.org/licenses/>.
;;
;;; Commentary:
;;
;; This package provides the `idomenu' command for selecting an imenu tag using
;; ido completion. The buffer needs to have support for imenu already enabled.
;;
;; Add something like the following to your .emacs:
;;
;; (autoload 'idomenu "idomenu" nil t)
;;
;;; Code:
(require 'ido)
(require 'imenu)
(defun idomenu--guess-default (index-alist symbol)
"Guess a default choice from the given symbol."
(catch 'found
(let ((regex (concat "\\_<" (regexp-quote symbol) "\\_>")))
(dolist (item index-alist)
(if (string-match regex (car item)) (throw 'found (car item)))))))
(defun idomenu--read (index-alist &optional prompt guess)
"Read a choice from an Imenu alist via Ido."
(let* ((symatpt (thing-at-point 'symbol))
(default (and guess symatpt (idomenu--guess-default index-alist symatpt)))
(names (mapcar 'car index-alist))
(name (ido-completing-read (or prompt "imenu ") names
nil t nil nil default))
(choice (assoc name index-alist)))
(if (imenu--subalist-p choice)
(idomenu--read (cdr choice) prompt nil)
choice)))
(defun idomenu--trim (str)
"Trim leading and tailing whitespace from STR."
(let ((s (if (symbolp str) (symbol-name str) str)))
(replace-regexp-in-string "\\(^[[:space:]\n]*\\|[[:space:]\n]*$\\)" "" s)))
(defun idomenu--trim-alist (index-alist)
"There must be a better way to apply a function to all cars of an alist"
(mapcar (lambda (pair) (cons (idomenu--trim (car pair)) (cdr pair)))
index-alist))
(defun idomenu ()
"Switch to a buffer-local tag from Imenu via Ido."
(interactive)
;; ido initialization
(ido-init-completion-maps)
(add-hook 'minibuffer-setup-hook 'ido-minibuffer-setup)
(add-hook 'choose-completion-string-functions 'ido-choose-completion-string)
(add-hook 'kill-emacs-hook 'ido-kill-emacs-hook)
;; set up ido completion list
(let ((index-alist (cdr (imenu--make-index-alist))))
(if (equal index-alist '(nil))
(message "No imenu tags in buffer")
(imenu (idomenu--read (idomenu--trim-alist index-alist) nil t)))))
(provide 'idomenu)