2016-04-21 23:27:19 +02:00
|
|
|
|
;;; helm-elisp.el --- Elisp symbols completion for helm. -*- lexical-binding: t -*-
|
|
|
|
|
|
|
|
|
|
;; Copyright (C) 2012 ~ 2016 Thierry Volpiatto <thierry.volpiatto@gmail.com>
|
|
|
|
|
|
|
|
|
|
;; 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 3 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/>.
|
|
|
|
|
|
|
|
|
|
;;; Code:
|
|
|
|
|
(require 'cl-lib)
|
|
|
|
|
(require 'helm)
|
|
|
|
|
(require 'helm-help)
|
|
|
|
|
(require 'helm-types)
|
|
|
|
|
(require 'helm-utils)
|
|
|
|
|
(require 'helm-info)
|
|
|
|
|
(require 'helm-eval)
|
|
|
|
|
(require 'helm-files)
|
|
|
|
|
(require 'advice)
|
|
|
|
|
|
|
|
|
|
(declare-function 'helm-describe-function "helm-lib")
|
|
|
|
|
(declare-function 'helm-describe-variable "helm-lib")
|
|
|
|
|
(declare-function 'helm-describe-face "helm-lib")
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
;;; Customizable values
|
|
|
|
|
|
|
|
|
|
(defgroup helm-elisp nil
|
|
|
|
|
"Elisp related Applications and libraries for Helm."
|
|
|
|
|
:group 'helm)
|
|
|
|
|
|
|
|
|
|
(defcustom helm-turn-on-show-completion t
|
|
|
|
|
"Display candidate in buffer while moving selection when non--nil."
|
|
|
|
|
:group 'helm-elisp
|
|
|
|
|
:type 'boolean)
|
|
|
|
|
|
|
|
|
|
(defcustom helm-show-completion-use-special-display t
|
|
|
|
|
"A special display will be used in Lisp completion if non--nil.
|
|
|
|
|
All functions that are wrapped in macro `with-helm-show-completion'
|
|
|
|
|
will be affected."
|
|
|
|
|
:group 'helm-elisp
|
|
|
|
|
:type 'boolean)
|
|
|
|
|
|
|
|
|
|
(defcustom helm-show-completion-min-window-height 7
|
|
|
|
|
"Minimum completion window height used in show completion.
|
|
|
|
|
This is used in macro `with-helm-show-completion'."
|
|
|
|
|
:group 'helm-elisp
|
|
|
|
|
:type 'integer)
|
|
|
|
|
|
|
|
|
|
(defcustom helm-lisp-quoted-function-list
|
|
|
|
|
'(funcall apply mapc cl-mapc mapcar cl-mapcar
|
|
|
|
|
callf callf2 cl-callf cl-callf2 fset
|
|
|
|
|
fboundp fmakunbound symbol-function)
|
|
|
|
|
"List of function where quoted function completion happen.
|
|
|
|
|
e.g give only function names after \(funcall '."
|
|
|
|
|
:group 'helm-elisp
|
|
|
|
|
:type '(repeat (choice symbol)))
|
|
|
|
|
|
|
|
|
|
(defcustom helm-lisp-unquoted-function-list
|
|
|
|
|
'(function defadvice)
|
|
|
|
|
"List of function where unquoted function completion happen.
|
|
|
|
|
e.g give only function names after \(function ."
|
|
|
|
|
:group 'helm-elisp
|
|
|
|
|
:type '(repeat (choice symbol)))
|
|
|
|
|
|
|
|
|
|
(defcustom helm-apropos-fuzzy-match nil
|
|
|
|
|
"Enable fuzzy matching for `helm-apropos' when non-nil."
|
|
|
|
|
:group 'helm-elisp
|
|
|
|
|
:type 'boolean)
|
|
|
|
|
|
|
|
|
|
(defcustom helm-lisp-fuzzy-completion nil
|
|
|
|
|
"Enable fuzzy matching in emacs-lisp completion when non-nil.
|
|
|
|
|
NOTE: This enable fuzzy matching in helm native implementation of
|
|
|
|
|
elisp completion, but not on helmized elisp completion, i.e
|
|
|
|
|
fuzzy completion is not available in `completion-at-point'."
|
|
|
|
|
:group 'helm-elisp
|
|
|
|
|
:type 'boolean)
|
|
|
|
|
|
|
|
|
|
(defcustom helm-apropos-function-list '(helm-def-source--emacs-commands
|
|
|
|
|
helm-def-source--emacs-functions
|
|
|
|
|
helm-def-source--eieio-classes
|
|
|
|
|
helm-def-source--eieio-generic
|
|
|
|
|
helm-def-source--emacs-variables
|
2016-07-12 15:08:56 +02:00
|
|
|
|
helm-def-source--emacs-faces)
|
2016-04-21 23:27:19 +02:00
|
|
|
|
"A list of functions that build helm sources to use in `helm-apropos'."
|
|
|
|
|
:group 'helm-elisp
|
|
|
|
|
:type '(repeat (choice symbol)))
|
|
|
|
|
|
2016-09-15 11:18:17 +02:00
|
|
|
|
(defcustom helm-apropos-defaut-info-lookup-sources '(helm-source-info-elisp
|
|
|
|
|
helm-source-info-cl
|
|
|
|
|
helm-source-info-eieio)
|
|
|
|
|
"A list of sources to look into when searching info page of a symbol."
|
|
|
|
|
:group 'helm-elisp
|
|
|
|
|
:type '(repeat (choice symbol)))
|
|
|
|
|
|
2016-04-21 23:27:19 +02:00
|
|
|
|
|
|
|
|
|
;;; Faces
|
|
|
|
|
;;
|
|
|
|
|
;;
|
|
|
|
|
(defgroup helm-elisp-faces nil
|
|
|
|
|
"Customize the appearance of helm-elisp."
|
|
|
|
|
:prefix "helm-"
|
|
|
|
|
:group 'helm-elisp
|
|
|
|
|
:group 'helm-faces)
|
|
|
|
|
|
|
|
|
|
(defface helm-lisp-show-completion
|
|
|
|
|
'((t (:background "DarkSlateGray")))
|
|
|
|
|
"Face used for showing candidates in `helm-lisp-completion'."
|
|
|
|
|
:group 'helm-elisp-faces)
|
|
|
|
|
|
|
|
|
|
(defface helm-lisp-completion-info
|
|
|
|
|
'((t (:foreground "red")))
|
|
|
|
|
"Face used for showing info in `helm-lisp-completion'."
|
|
|
|
|
:group 'helm-elisp-faces)
|
|
|
|
|
|
|
|
|
|
(defcustom helm-elisp-help-function
|
|
|
|
|
'helm-elisp-show-help
|
|
|
|
|
"Function for displaying help for Lisp symbols."
|
|
|
|
|
:group 'helm-elisp
|
|
|
|
|
:type '(choice (function :tag "Open help for the symbol."
|
|
|
|
|
helm-elisp-show-help)
|
|
|
|
|
(function :tag "Show one liner in modeline."
|
|
|
|
|
helm-elisp-show-doc-modeline)))
|
|
|
|
|
|
2016-09-26 19:08:35 +02:00
|
|
|
|
(defcustom helm-locate-library-fuzzy-match t
|
|
|
|
|
"Enable fuzzy-matching in `helm-locate-library' when non--nil."
|
|
|
|
|
:type 'boolean
|
|
|
|
|
:group 'helm-elisp)
|
|
|
|
|
|
2016-04-21 23:27:19 +02:00
|
|
|
|
|
|
|
|
|
;;; Show completion.
|
|
|
|
|
;;
|
|
|
|
|
;; Provide show completion with macro `with-helm-show-completion'.
|
|
|
|
|
|
|
|
|
|
(defvar helm-show-completion-overlay nil)
|
|
|
|
|
|
|
|
|
|
;; Called each time cursor move in helm-buffer.
|
|
|
|
|
(defun helm-show-completion ()
|
|
|
|
|
(with-helm-current-buffer
|
|
|
|
|
(overlay-put helm-show-completion-overlay
|
|
|
|
|
'display (substring-no-properties
|
|
|
|
|
(helm-get-selection)))))
|
|
|
|
|
|
|
|
|
|
(defun helm-show-completion-init-overlay (beg end)
|
|
|
|
|
(when (and helm-turn-on-show-completion beg end)
|
|
|
|
|
(setq helm-show-completion-overlay (make-overlay beg end))
|
|
|
|
|
(overlay-put helm-show-completion-overlay
|
|
|
|
|
'face 'helm-lisp-show-completion)))
|
|
|
|
|
|
|
|
|
|
(defun helm-show-completion-display-function (buffer &rest _args)
|
|
|
|
|
"A special resized helm window is used depending on position in BUFFER."
|
|
|
|
|
(with-selected-window (selected-window)
|
|
|
|
|
(if (window-dedicated-p)
|
|
|
|
|
(helm-default-display-buffer buffer)
|
|
|
|
|
(let* ((screen-size (+ (count-screen-lines (window-start) (point) t)
|
|
|
|
|
1 ; mode-line
|
|
|
|
|
(if header-line-format 1 0))) ; header-line
|
|
|
|
|
(def-size (- (window-height)
|
|
|
|
|
helm-show-completion-min-window-height))
|
|
|
|
|
(upper-height (max window-min-height (min screen-size def-size)))
|
|
|
|
|
split-window-keep-point)
|
|
|
|
|
(recenter -1)
|
|
|
|
|
(set-window-buffer (if (active-minibuffer-window)
|
|
|
|
|
(minibuffer-selected-window)
|
|
|
|
|
(split-window nil upper-height
|
|
|
|
|
helm-split-window-default-side))
|
|
|
|
|
buffer)))))
|
|
|
|
|
|
|
|
|
|
(defmacro with-helm-show-completion (beg end &rest body)
|
|
|
|
|
"Show helm candidate in an overlay at point.
|
|
|
|
|
BEG and END are the beginning and end position of the current completion
|
|
|
|
|
in `helm-current-buffer'.
|
|
|
|
|
BODY is an helm call where we want to enable show completion.
|
|
|
|
|
If `helm-turn-on-show-completion' is nil just do nothing."
|
|
|
|
|
(declare (indent 2) (debug t))
|
|
|
|
|
`(let ((helm-move-selection-after-hook
|
|
|
|
|
(and helm-turn-on-show-completion
|
|
|
|
|
(append (list 'helm-show-completion)
|
|
|
|
|
helm-move-selection-after-hook)))
|
|
|
|
|
(helm-always-two-windows t)
|
|
|
|
|
(helm-split-window-default-side
|
|
|
|
|
(if (eq helm-split-window-default-side 'same)
|
|
|
|
|
'below helm-split-window-default-side))
|
|
|
|
|
helm-split-window-in-side-p
|
|
|
|
|
helm-reuse-last-window-split-state)
|
|
|
|
|
(helm-set-local-variable
|
|
|
|
|
'helm-display-function
|
|
|
|
|
(if helm-show-completion-use-special-display
|
|
|
|
|
'helm-show-completion-display-function
|
|
|
|
|
'helm-default-display-buffer))
|
|
|
|
|
(unwind-protect
|
|
|
|
|
(progn
|
|
|
|
|
(helm-show-completion-init-overlay ,beg ,end)
|
|
|
|
|
,@body)
|
|
|
|
|
(when (and helm-turn-on-show-completion
|
|
|
|
|
helm-show-completion-overlay
|
|
|
|
|
(overlayp helm-show-completion-overlay))
|
|
|
|
|
(delete-overlay helm-show-completion-overlay)))))
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
;;; Lisp symbol completion.
|
|
|
|
|
;;
|
|
|
|
|
;;
|
|
|
|
|
(defun helm-lisp-completion--predicate-at-point (beg)
|
|
|
|
|
;; Return a predicate for `all-completions'.
|
|
|
|
|
(let ((fn-sym-p (lambda ()
|
|
|
|
|
(or
|
|
|
|
|
(and (eq (char-before) ?\ )
|
|
|
|
|
(save-excursion
|
|
|
|
|
(skip-syntax-backward " " (point-at-bol))
|
|
|
|
|
(memq (symbol-at-point)
|
|
|
|
|
helm-lisp-unquoted-function-list)))
|
|
|
|
|
(and (eq (char-before) ?\')
|
|
|
|
|
(save-excursion
|
|
|
|
|
(forward-char -1)
|
|
|
|
|
(eq (char-before) ?\#)))))))
|
|
|
|
|
(save-excursion
|
|
|
|
|
(goto-char beg)
|
|
|
|
|
(if (or
|
|
|
|
|
;; Complete on all symbols in non--lisp modes (logs mail etc..)
|
|
|
|
|
(not (memq major-mode '(emacs-lisp-mode
|
|
|
|
|
lisp-interaction-mode
|
|
|
|
|
inferior-emacs-lisp-mode)))
|
|
|
|
|
(not (or (funcall fn-sym-p)
|
|
|
|
|
(and (eq (char-before) ?\')
|
|
|
|
|
(save-excursion
|
|
|
|
|
(forward-char (if (funcall fn-sym-p) -2 -1))
|
|
|
|
|
(skip-syntax-backward " " (point-at-bol))
|
|
|
|
|
(memq (symbol-at-point)
|
|
|
|
|
helm-lisp-quoted-function-list)))
|
|
|
|
|
(eq (char-before) ?\())) ; no paren before str.
|
|
|
|
|
;; Looks like we are in a let statement.
|
|
|
|
|
(condition-case nil
|
|
|
|
|
(progn (up-list -2) (forward-char 1)
|
|
|
|
|
(eq (char-after) ?\())
|
|
|
|
|
(error nil)))
|
|
|
|
|
(lambda (sym)
|
|
|
|
|
(or (boundp sym) (fboundp sym) (symbol-plist sym)))
|
|
|
|
|
#'fboundp))))
|
|
|
|
|
|
|
|
|
|
(defun helm-thing-before-point (&optional limits regexp)
|
|
|
|
|
"Return symbol name before point.
|
|
|
|
|
If REGEXP is specified return what REGEXP find before point.
|
|
|
|
|
By default match the beginning of symbol before point.
|
|
|
|
|
With LIMITS arg specified return the beginning and end position
|
|
|
|
|
of symbol before point."
|
|
|
|
|
(save-excursion
|
|
|
|
|
(let (beg
|
|
|
|
|
(end (point))
|
|
|
|
|
(boundary (field-beginning nil nil (point-at-bol))))
|
|
|
|
|
(if (re-search-backward (or regexp "\\_<") boundary t)
|
|
|
|
|
(setq beg (match-end 0))
|
|
|
|
|
(setq beg boundary))
|
|
|
|
|
(unless (= beg end)
|
|
|
|
|
(if limits
|
|
|
|
|
(cons beg end)
|
|
|
|
|
(buffer-substring-no-properties beg end))))))
|
|
|
|
|
|
|
|
|
|
(defun helm-bounds-of-thing-before-point (&optional regexp)
|
|
|
|
|
"Get the beginning and end position of `helm-thing-before-point'.
|
|
|
|
|
Return a cons \(beg . end\)."
|
|
|
|
|
(helm-thing-before-point 'limits regexp))
|
|
|
|
|
|
|
|
|
|
(defun helm-insert-completion-at-point (beg end str)
|
|
|
|
|
;; When there is no space after point
|
|
|
|
|
;; we are completing inside a symbol or
|
|
|
|
|
;; after a partial symbol with the next arg aside
|
|
|
|
|
;; without space, in this case mark the region.
|
|
|
|
|
;; deleting it would remove the
|
|
|
|
|
;; next arg which is unwanted.
|
|
|
|
|
(delete-region beg end)
|
|
|
|
|
(insert str)
|
|
|
|
|
(let ((pos (cdr (or (bounds-of-thing-at-point 'symbol)
|
|
|
|
|
;; needed for helm-dabbrev.
|
|
|
|
|
(bounds-of-thing-at-point 'filename)))))
|
|
|
|
|
(when (and pos (< (point) pos))
|
|
|
|
|
(push-mark pos t t))))
|
|
|
|
|
|
|
|
|
|
(defvar helm-lisp-completion--cache nil)
|
|
|
|
|
(defvar helm-lgst-len nil)
|
|
|
|
|
;;;###autoload
|
|
|
|
|
(defun helm-lisp-completion-at-point ()
|
|
|
|
|
"Preconfigured helm for lisp symbol completion at point."
|
|
|
|
|
(interactive)
|
|
|
|
|
(setq helm-lgst-len 0)
|
|
|
|
|
(let* ((target (helm-thing-before-point))
|
|
|
|
|
(beg (car (helm-bounds-of-thing-before-point)))
|
|
|
|
|
(end (point))
|
|
|
|
|
(pred (and beg (helm-lisp-completion--predicate-at-point beg)))
|
|
|
|
|
(loc-vars (and (fboundp 'elisp--local-variables)
|
|
|
|
|
(ignore-errors
|
|
|
|
|
(mapcar #'symbol-name (elisp--local-variables)))))
|
|
|
|
|
(glob-syms (and target pred (all-completions target obarray pred)))
|
|
|
|
|
(candidates (append loc-vars glob-syms))
|
|
|
|
|
(helm-quit-if-no-candidate t)
|
|
|
|
|
(helm-execute-action-at-once-if-one t)
|
|
|
|
|
(enable-recursive-minibuffers t))
|
|
|
|
|
(setq helm-lisp-completion--cache (cl-loop for sym in candidates
|
|
|
|
|
for len = (length sym)
|
|
|
|
|
when (> len helm-lgst-len)
|
|
|
|
|
do (setq helm-lgst-len len)
|
|
|
|
|
collect sym))
|
|
|
|
|
(if candidates
|
|
|
|
|
(with-helm-show-completion beg end
|
|
|
|
|
;; Overlay is initialized now in helm-current-buffer.
|
|
|
|
|
(helm
|
|
|
|
|
:sources (helm-build-in-buffer-source "Lisp completion"
|
|
|
|
|
:data helm-lisp-completion--cache
|
|
|
|
|
:persistent-action 'helm-lisp-completion-persistent-action
|
|
|
|
|
:nomark t
|
2016-08-18 22:01:20 +02:00
|
|
|
|
:match-part (lambda (c) (car (split-string c)))
|
2016-04-21 23:27:19 +02:00
|
|
|
|
:fuzzy-match helm-lisp-fuzzy-completion
|
|
|
|
|
:persistent-help (helm-lisp-completion-persistent-help)
|
|
|
|
|
:filtered-candidate-transformer
|
|
|
|
|
'helm-lisp-completion-transformer
|
2016-09-15 11:18:17 +02:00
|
|
|
|
:action (lambda (candidate)
|
|
|
|
|
(with-helm-current-buffer
|
|
|
|
|
(run-with-timer
|
|
|
|
|
0.01 nil
|
|
|
|
|
'helm-insert-completion-at-point
|
|
|
|
|
beg end candidate))))
|
2016-04-21 23:27:19 +02:00
|
|
|
|
:input (if helm-lisp-fuzzy-completion
|
|
|
|
|
target (concat target " "))
|
|
|
|
|
:resume 'noresume
|
|
|
|
|
:buffer "*helm lisp completion*"
|
|
|
|
|
:allow-nest t))
|
|
|
|
|
(message "[No Match]"))))
|
|
|
|
|
|
|
|
|
|
(defun helm-lisp-completion-persistent-action (candidate &optional name)
|
|
|
|
|
"Show documentation for the function.
|
|
|
|
|
Documentation is shown briefly in mode-line or completely
|
|
|
|
|
in other window according to the value of `helm-elisp-help-function'."
|
|
|
|
|
(funcall helm-elisp-help-function candidate name))
|
|
|
|
|
|
|
|
|
|
(defun helm-lisp-completion-persistent-help ()
|
|
|
|
|
"Return persistent-help according to the value of `helm-elisp-help-function'"
|
|
|
|
|
(cl-ecase helm-elisp-help-function
|
|
|
|
|
(helm-elisp-show-doc-modeline "Show brief doc in mode-line")
|
|
|
|
|
(helm-elisp-show-help "Toggle show help for the symbol")))
|
|
|
|
|
|
|
|
|
|
(defun helm-elisp--show-help-1 (candidate &optional name)
|
|
|
|
|
(let ((sym (intern-soft candidate)))
|
|
|
|
|
(cl-typecase sym
|
|
|
|
|
((and fboundp boundp)
|
|
|
|
|
(if (member name '("describe-function" "describe-variable"))
|
|
|
|
|
(funcall (intern (format "helm-%s" name)) sym)
|
|
|
|
|
;; When there is no way to know what to describe
|
|
|
|
|
;; prefer describe-function.
|
|
|
|
|
(helm-describe-function sym)))
|
|
|
|
|
(fbound (helm-describe-function sym))
|
|
|
|
|
(bound (helm-describe-variable sym))
|
|
|
|
|
(face (helm-describe-face sym)))))
|
|
|
|
|
|
|
|
|
|
(defun helm-elisp-show-help (candidate &optional name)
|
|
|
|
|
"Show full help for the function CANDIDATE.
|
|
|
|
|
Arg NAME specify the name of the top level function
|
|
|
|
|
calling helm generic completion (e.g \"describe-function\")."
|
|
|
|
|
(helm-elisp--persistent-help
|
|
|
|
|
candidate 'helm-elisp--show-help-1 name))
|
|
|
|
|
|
|
|
|
|
(defun helm-elisp-show-doc-modeline (candidate &optional name)
|
|
|
|
|
"Show brief documentation for the function in modeline."
|
|
|
|
|
(let ((cursor-in-echo-area t)
|
|
|
|
|
mode-line-in-non-selected-windows)
|
|
|
|
|
(helm-show-info-in-mode-line
|
|
|
|
|
(propertize
|
|
|
|
|
(helm-get-first-line-documentation
|
|
|
|
|
(intern candidate) name)
|
|
|
|
|
'face 'helm-lisp-completion-info))))
|
|
|
|
|
|
|
|
|
|
(defun helm-lisp-completion-transformer (candidates _source)
|
|
|
|
|
"Helm candidates transformer for lisp completion."
|
|
|
|
|
(cl-loop for c in candidates
|
|
|
|
|
for sym = (intern c)
|
|
|
|
|
for annot = (cl-typecase sym
|
|
|
|
|
(command " (Com)")
|
|
|
|
|
(class " (Class)")
|
|
|
|
|
(generic " (Gen)")
|
|
|
|
|
(fbound " (Fun)")
|
|
|
|
|
(bound " (Var)")
|
|
|
|
|
(face " (Face)"))
|
|
|
|
|
for spaces = (make-string (- helm-lgst-len (length c)) ? )
|
|
|
|
|
collect (cons (concat c spaces annot) c) into lst
|
|
|
|
|
finally return (sort lst #'helm-generic-sort-fn)))
|
|
|
|
|
|
|
|
|
|
(defun helm-get-first-line-documentation (sym &optional name)
|
|
|
|
|
"Return first line documentation of symbol SYM.
|
|
|
|
|
If SYM is not documented, return \"Not documented\"."
|
|
|
|
|
(let ((doc (cl-typecase sym
|
|
|
|
|
((and fboundp boundp)
|
|
|
|
|
(cond ((string= name "describe-function")
|
|
|
|
|
(documentation sym t))
|
|
|
|
|
((string= name "describe-variable")
|
|
|
|
|
(documentation-property sym 'variable-documentation t))
|
|
|
|
|
(t (documentation sym t))))
|
|
|
|
|
(fbound (documentation sym t))
|
|
|
|
|
(bound (documentation-property sym 'variable-documentation t))
|
|
|
|
|
(face (face-documentation sym)))))
|
|
|
|
|
(if (and doc (not (string= doc ""))
|
|
|
|
|
;; `documentation' return "\n\n(args...)"
|
|
|
|
|
;; for CL-style functions.
|
|
|
|
|
(not (string-match-p "^\n\n" doc)))
|
|
|
|
|
(car (split-string doc "\n"))
|
|
|
|
|
"Not documented")))
|
|
|
|
|
|
|
|
|
|
;;; File completion.
|
|
|
|
|
;;
|
|
|
|
|
;; Complete file name at point.
|
|
|
|
|
|
|
|
|
|
;;;###autoload
|
|
|
|
|
(defun helm-complete-file-name-at-point (&optional force)
|
|
|
|
|
"Preconfigured helm to complete file name at point."
|
|
|
|
|
(interactive)
|
|
|
|
|
(require 'helm-mode)
|
|
|
|
|
(let* ((tap (thing-at-point 'filename))
|
|
|
|
|
beg
|
|
|
|
|
(init (and tap
|
|
|
|
|
(or force
|
|
|
|
|
(save-excursion
|
|
|
|
|
(end-of-line)
|
|
|
|
|
(search-backward tap (point-at-bol) t)
|
|
|
|
|
(setq beg (point))
|
|
|
|
|
(looking-back "[^'`( ]" (1- (point)))))
|
|
|
|
|
(expand-file-name
|
|
|
|
|
(substring-no-properties tap))))
|
|
|
|
|
(end (point))
|
|
|
|
|
(helm-quit-if-no-candidate t)
|
|
|
|
|
(helm-execute-action-at-once-if-one t)
|
|
|
|
|
completion)
|
|
|
|
|
(with-helm-show-completion beg end
|
|
|
|
|
(setq completion (helm-read-file-name "FileName: "
|
|
|
|
|
:initial-input init)))
|
|
|
|
|
(when (and completion (not (string= completion "")))
|
|
|
|
|
(delete-region beg end) (insert (if (string-match "^~" tap)
|
|
|
|
|
(abbreviate-file-name completion)
|
|
|
|
|
completion)))))
|
|
|
|
|
|
|
|
|
|
;;;###autoload
|
|
|
|
|
(defun helm-lisp-indent ()
|
|
|
|
|
;; It is meant to use with `helm-define-multi-key' which
|
|
|
|
|
;; does not support args for functions yet, so use `current-prefix-arg'
|
|
|
|
|
;; for now instead of (interactive "P").
|
|
|
|
|
(interactive)
|
|
|
|
|
(let ((tab-always-indent (or (eq tab-always-indent 'complete)
|
|
|
|
|
tab-always-indent)))
|
|
|
|
|
(indent-for-tab-command current-prefix-arg)))
|
|
|
|
|
|
|
|
|
|
;;;###autoload
|
|
|
|
|
(defun helm-lisp-completion-or-file-name-at-point ()
|
|
|
|
|
"Preconfigured helm to complete lisp symbol or filename at point.
|
|
|
|
|
Filename completion happen if string start after or between a double quote."
|
|
|
|
|
(interactive)
|
|
|
|
|
(let* ((tap (thing-at-point 'filename)))
|
|
|
|
|
(if (and tap (save-excursion
|
|
|
|
|
(end-of-line)
|
|
|
|
|
(search-backward tap (point-at-bol) t)
|
|
|
|
|
(looking-back "[^'`( ]" (1- (point)))))
|
|
|
|
|
(helm-complete-file-name-at-point)
|
|
|
|
|
(helm-lisp-completion-at-point))))
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
;;; Apropos
|
|
|
|
|
;;
|
|
|
|
|
;;
|
2016-06-29 09:21:54 +02:00
|
|
|
|
(defvar helm-apropos-history nil)
|
|
|
|
|
|
2016-04-21 23:27:19 +02:00
|
|
|
|
(defun helm-apropos-init (test default)
|
|
|
|
|
"Init candidates buffer for `helm-apropos' sources."
|
|
|
|
|
(require 'helm-help)
|
|
|
|
|
(helm-init-candidates-in-buffer 'global
|
|
|
|
|
(let ((default-symbol (and (stringp default)
|
|
|
|
|
(intern-soft default)))
|
|
|
|
|
(symbols (all-completions "" obarray test)))
|
|
|
|
|
(if (and default-symbol (funcall test default-symbol))
|
|
|
|
|
(cons default-symbol symbols)
|
|
|
|
|
symbols))))
|
|
|
|
|
|
|
|
|
|
(defun helm-apropos-init-faces (default)
|
|
|
|
|
"Init candidates buffer for faces for `helm-apropos'."
|
|
|
|
|
(require 'helm-help)
|
|
|
|
|
(with-current-buffer (helm-candidate-buffer 'global)
|
|
|
|
|
(goto-char (point-min))
|
|
|
|
|
(let ((default-symbol (and (stringp default)
|
|
|
|
|
(intern-soft default)))
|
|
|
|
|
(faces (face-list)))
|
|
|
|
|
(when (and default-symbol (facep default-symbol))
|
|
|
|
|
(insert (concat default "\n")))
|
|
|
|
|
(insert
|
|
|
|
|
(mapconcat #'prin1-to-string
|
|
|
|
|
(if default
|
|
|
|
|
(cl-remove-if (lambda (sym) (string= sym default)) faces)
|
|
|
|
|
faces)
|
|
|
|
|
"\n")))))
|
|
|
|
|
|
|
|
|
|
(defun helm-apropos-default-sort-fn (candidates _source)
|
|
|
|
|
(if (string= helm-pattern "")
|
|
|
|
|
candidates
|
|
|
|
|
(sort candidates #'helm-generic-sort-fn)))
|
|
|
|
|
|
|
|
|
|
(defun helm-def-source--emacs-variables (&optional default)
|
|
|
|
|
(helm-build-in-buffer-source "Variables"
|
2016-09-15 11:18:17 +02:00
|
|
|
|
:init (lambda ()
|
|
|
|
|
(helm-apropos-init
|
|
|
|
|
(lambda (x) (and (boundp x) (not (keywordp x)))) default))
|
2016-04-21 23:27:19 +02:00
|
|
|
|
:fuzzy-match helm-apropos-fuzzy-match
|
|
|
|
|
:filtered-candidate-transformer (and (null helm-apropos-fuzzy-match)
|
|
|
|
|
'helm-apropos-default-sort-fn)
|
|
|
|
|
:nomark t
|
|
|
|
|
:persistent-action (lambda (candidate)
|
|
|
|
|
(helm-elisp--persistent-help
|
|
|
|
|
candidate 'helm-describe-variable))
|
|
|
|
|
:persistent-help "Describe variable"
|
|
|
|
|
:action '(("Describe variable" . helm-describe-variable)
|
|
|
|
|
("Find variable" . helm-find-variable)
|
|
|
|
|
("Info lookup" . helm-info-lookup-symbol)
|
|
|
|
|
("Set variable" . helm-set-variable))
|
|
|
|
|
:action-transformer
|
|
|
|
|
(lambda (actions candidate)
|
|
|
|
|
(let ((sym (helm-symbolify candidate)))
|
|
|
|
|
(if (custom-variable-p sym)
|
|
|
|
|
(append
|
|
|
|
|
actions
|
|
|
|
|
(let ((standard-value (eval (car (get sym 'standard-value)))))
|
|
|
|
|
(unless (equal standard-value (symbol-value sym))
|
|
|
|
|
`(("Reset Variable to default value" .
|
|
|
|
|
,(lambda (candidate)
|
|
|
|
|
(let ((sym (helm-symbolify candidate)))
|
|
|
|
|
(set sym standard-value)))))))
|
|
|
|
|
'(("Customize variable" .
|
|
|
|
|
(lambda (candidate)
|
|
|
|
|
(customize-option (helm-symbolify candidate))))))
|
|
|
|
|
actions)))))
|
|
|
|
|
|
|
|
|
|
(defun helm-def-source--emacs-faces (&optional default)
|
|
|
|
|
"Create `helm' source for faces to be displayed with
|
|
|
|
|
`helm-apropos'."
|
|
|
|
|
(helm-build-in-buffer-source "Faces"
|
|
|
|
|
:init (lambda () (helm-apropos-init-faces default))
|
|
|
|
|
:fuzzy-match helm-apropos-fuzzy-match
|
|
|
|
|
:filtered-candidate-transformer
|
|
|
|
|
(append (and (null helm-apropos-fuzzy-match)
|
|
|
|
|
'(helm-apropos-default-sort-fn))
|
|
|
|
|
(list
|
|
|
|
|
(lambda (candidates _source)
|
|
|
|
|
(cl-loop for c in candidates
|
|
|
|
|
collect (propertize c 'face (intern c))))))
|
|
|
|
|
:persistent-action (lambda (candidate)
|
|
|
|
|
(helm-elisp--persistent-help
|
|
|
|
|
candidate 'helm-describe-face))
|
|
|
|
|
:persistent-help "Describe face"
|
|
|
|
|
:action '(("Describe face" . helm-describe-face)
|
|
|
|
|
("Find face" . helm-find-face-definition)
|
|
|
|
|
("Customize face" . (lambda (candidate)
|
|
|
|
|
(customize-face (helm-symbolify candidate)))))))
|
|
|
|
|
|
|
|
|
|
(defun helm-def-source--emacs-commands (&optional default)
|
|
|
|
|
(helm-build-in-buffer-source "Commands"
|
2016-09-15 11:18:17 +02:00
|
|
|
|
:init (lambda ()
|
|
|
|
|
(helm-apropos-init 'commandp default))
|
2016-04-21 23:27:19 +02:00
|
|
|
|
:fuzzy-match helm-apropos-fuzzy-match
|
|
|
|
|
:filtered-candidate-transformer (and (null helm-apropos-fuzzy-match)
|
|
|
|
|
'helm-apropos-default-sort-fn)
|
|
|
|
|
:nomark t
|
|
|
|
|
:persistent-action (lambda (candidate)
|
|
|
|
|
(helm-elisp--persistent-help
|
|
|
|
|
candidate 'helm-describe-function))
|
|
|
|
|
:persistent-help "Describe command"
|
|
|
|
|
:action '(("Describe function" . helm-describe-function)
|
|
|
|
|
("Find function" . helm-find-function)
|
|
|
|
|
("Info lookup" . helm-info-lookup-symbol))))
|
|
|
|
|
|
|
|
|
|
(defun helm-def-source--emacs-functions (&optional default)
|
|
|
|
|
(helm-build-in-buffer-source "Functions"
|
2016-09-15 11:18:17 +02:00
|
|
|
|
:init (lambda ()
|
|
|
|
|
(helm-apropos-init (lambda (x)
|
|
|
|
|
(and (fboundp x)
|
|
|
|
|
(not (commandp x))
|
|
|
|
|
(not (generic-p x))
|
|
|
|
|
(not (class-p x))))
|
|
|
|
|
default))
|
2016-04-21 23:27:19 +02:00
|
|
|
|
:fuzzy-match helm-apropos-fuzzy-match
|
|
|
|
|
:filtered-candidate-transformer (and (null helm-apropos-fuzzy-match)
|
|
|
|
|
'helm-apropos-default-sort-fn)
|
|
|
|
|
:persistent-action (lambda (candidate)
|
|
|
|
|
(helm-elisp--persistent-help
|
|
|
|
|
candidate 'helm-describe-function))
|
|
|
|
|
:persistent-help "Describe function"
|
|
|
|
|
:nomark t
|
|
|
|
|
:action '(("Describe function" . helm-describe-function)
|
|
|
|
|
("Find function" . helm-find-function)
|
|
|
|
|
("Info lookup" . helm-info-lookup-symbol))))
|
|
|
|
|
|
|
|
|
|
(defun helm-def-source--eieio-classes (&optional default)
|
|
|
|
|
(helm-build-in-buffer-source "Classes"
|
2016-09-15 11:18:17 +02:00
|
|
|
|
:init (lambda ()
|
|
|
|
|
(helm-apropos-init (lambda (x)
|
|
|
|
|
(class-p x))
|
|
|
|
|
default))
|
2016-04-21 23:27:19 +02:00
|
|
|
|
:fuzzy-match helm-apropos-fuzzy-match
|
|
|
|
|
:filtered-candidate-transformer (and (null helm-apropos-fuzzy-match)
|
|
|
|
|
'helm-apropos-default-sort-fn)
|
|
|
|
|
:nomark t
|
|
|
|
|
:persistent-action (lambda (candidate)
|
|
|
|
|
(helm-elisp--persistent-help
|
|
|
|
|
candidate 'helm-describe-function))
|
|
|
|
|
:persistent-help "Describe class"
|
|
|
|
|
:action '(("Describe function" . helm-describe-function)
|
|
|
|
|
("Find function" . helm-find-function)
|
|
|
|
|
("Info lookup" . helm-info-lookup-symbol))))
|
|
|
|
|
|
|
|
|
|
(defun helm-def-source--eieio-generic (&optional default)
|
|
|
|
|
(helm-build-in-buffer-source "Generic functions"
|
2016-09-15 11:18:17 +02:00
|
|
|
|
:init (lambda ()
|
|
|
|
|
(helm-apropos-init (lambda (x)
|
|
|
|
|
(generic-p x))
|
|
|
|
|
default))
|
2016-04-21 23:27:19 +02:00
|
|
|
|
:fuzzy-match helm-apropos-fuzzy-match
|
|
|
|
|
:filtered-candidate-transformer (and (null helm-apropos-fuzzy-match)
|
|
|
|
|
'helm-apropos-default-sort-fn)
|
|
|
|
|
:nomark t
|
|
|
|
|
:persistent-action (lambda (candidate)
|
|
|
|
|
(helm-elisp--persistent-help
|
|
|
|
|
candidate 'helm-describe-function))
|
|
|
|
|
:persistent-help "Describe generic function"
|
|
|
|
|
:action '(("Describe function" . helm-describe-function)
|
|
|
|
|
("Find function" . helm-find-function)
|
|
|
|
|
("Info lookup" . helm-info-lookup-symbol))))
|
|
|
|
|
|
2016-09-15 11:18:17 +02:00
|
|
|
|
(defun helm-info-lookup-fallback-source (candidate)
|
|
|
|
|
(let ((sym (helm-symbolify candidate))
|
|
|
|
|
src-name fn)
|
|
|
|
|
(cond ((class-p sym)
|
|
|
|
|
(setq fn #'helm-describe-function
|
|
|
|
|
src-name "Describe class"))
|
|
|
|
|
((generic-p sym)
|
|
|
|
|
(setq fn #'helm-describe-function
|
|
|
|
|
src-name "Describe generic function"))
|
|
|
|
|
((fboundp sym)
|
|
|
|
|
(setq fn #'helm-describe-function
|
|
|
|
|
src-name "Describe function"))
|
|
|
|
|
((facep sym)
|
|
|
|
|
(setq fn #'helm-describe-face
|
|
|
|
|
src-name "Describe face"))
|
|
|
|
|
(t
|
|
|
|
|
(setq fn #'helm-describe-variable
|
|
|
|
|
src-name "Describe variable")))
|
|
|
|
|
(helm-build-sync-source src-name
|
|
|
|
|
:candidates (list candidate)
|
|
|
|
|
:persistent-action (lambda (candidate)
|
|
|
|
|
(helm-elisp--persistent-help
|
|
|
|
|
candidate fn))
|
|
|
|
|
:persistent-help src-name
|
|
|
|
|
:nomark t
|
|
|
|
|
:action fn)))
|
|
|
|
|
|
2016-04-21 23:27:19 +02:00
|
|
|
|
(defun helm-info-lookup-symbol-1 (c)
|
2016-09-15 11:18:17 +02:00
|
|
|
|
(let ((helm-execute-action-at-once-if-one 'current-source))
|
|
|
|
|
(helm :sources (append helm-apropos-defaut-info-lookup-sources
|
|
|
|
|
(list (helm-info-lookup-fallback-source c)))
|
2016-04-21 23:27:19 +02:00
|
|
|
|
:resume 'noresume
|
|
|
|
|
:buffer "*helm lookup*"
|
|
|
|
|
:input c)))
|
|
|
|
|
|
|
|
|
|
(defun helm-info-lookup-symbol (candidate)
|
2016-09-15 11:18:17 +02:00
|
|
|
|
;; Running an idle-timer allow not catching RET
|
|
|
|
|
;; when exiting with the fallback source.
|
|
|
|
|
(run-with-idle-timer 0.01 nil #'helm-info-lookup-symbol-1 candidate))
|
2016-04-21 23:27:19 +02:00
|
|
|
|
|
|
|
|
|
(defun helm-elisp--persistent-help (candidate fun &optional name)
|
|
|
|
|
(let ((hbuf (get-buffer (help-buffer))))
|
2016-09-15 11:18:17 +02:00
|
|
|
|
(cond ((helm-follow-mode-p)
|
|
|
|
|
(if name
|
|
|
|
|
(funcall fun candidate name)
|
|
|
|
|
(funcall fun candidate)))
|
|
|
|
|
((or (and (helm-attr 'help-running-p)
|
|
|
|
|
(string= candidate (helm-attr 'help-current-symbol))))
|
|
|
|
|
(progn
|
|
|
|
|
;; When started from a help buffer,
|
|
|
|
|
;; Don't kill this buffer as it is helm-current-buffer.
|
|
|
|
|
(unless (equal hbuf helm-current-buffer)
|
|
|
|
|
(kill-buffer hbuf)
|
|
|
|
|
(set-window-buffer (get-buffer-window hbuf)
|
|
|
|
|
helm-current-buffer))
|
|
|
|
|
(helm-attrset 'help-running-p nil)))
|
|
|
|
|
(t
|
|
|
|
|
(if name
|
|
|
|
|
(funcall fun candidate name)
|
|
|
|
|
(funcall fun candidate))
|
|
|
|
|
(helm-attrset 'help-running-p t)))
|
2016-04-21 23:27:19 +02:00
|
|
|
|
(helm-attrset 'help-current-symbol candidate)))
|
|
|
|
|
|
|
|
|
|
;;;###autoload
|
|
|
|
|
(defun helm-apropos (default)
|
|
|
|
|
"Preconfigured helm to describe commands, functions, variables and faces.
|
|
|
|
|
In non interactives calls DEFAULT argument should be provided as a string,
|
|
|
|
|
i.e the `symbol-name' of any existing symbol."
|
|
|
|
|
(interactive (list (thing-at-point 'symbol)))
|
|
|
|
|
(helm :sources
|
|
|
|
|
(mapcar (lambda (func)
|
|
|
|
|
(funcall func default))
|
|
|
|
|
helm-apropos-function-list)
|
2016-06-29 09:21:54 +02:00
|
|
|
|
:history 'helm-apropos-history
|
2016-04-21 23:27:19 +02:00
|
|
|
|
:buffer "*helm apropos*"
|
|
|
|
|
:preselect (and default (concat "\\_<" (regexp-quote default) "\\_>"))))
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
;;; Advices
|
|
|
|
|
;;
|
|
|
|
|
;;
|
|
|
|
|
(defvar helm-source-advice
|
2016-08-18 22:01:20 +02:00
|
|
|
|
(helm-build-sync-source "Function Advice"
|
|
|
|
|
:candidates 'helm-advice-candidates
|
|
|
|
|
:action (helm-make-actions "Toggle Enable/Disable" 'helm-advice-toggle)
|
|
|
|
|
:persistent-action 'helm-advice-persistent-action
|
|
|
|
|
:nomark t
|
|
|
|
|
:multiline t
|
|
|
|
|
:persistent-help "Describe function / C-u C-j: Toggle advice"))
|
2016-04-21 23:27:19 +02:00
|
|
|
|
|
|
|
|
|
(defun helm-advice-candidates ()
|
|
|
|
|
(cl-loop for (fname) in ad-advised-functions
|
|
|
|
|
for function = (intern fname)
|
|
|
|
|
append
|
|
|
|
|
(cl-loop for class in ad-advice-classes append
|
|
|
|
|
(cl-loop for advice in (ad-get-advice-info-field function class)
|
|
|
|
|
for enabled = (ad-advice-enabled advice)
|
|
|
|
|
collect
|
|
|
|
|
(cons (format
|
|
|
|
|
"%s %s %s"
|
|
|
|
|
(if enabled "Enabled " "Disabled")
|
|
|
|
|
(propertize fname 'face 'font-lock-function-name-face)
|
|
|
|
|
(ad-make-single-advice-docstring advice class nil))
|
|
|
|
|
(list function class advice))))))
|
|
|
|
|
|
|
|
|
|
(defun helm-advice-persistent-action (func-class-advice)
|
|
|
|
|
(if current-prefix-arg
|
|
|
|
|
(helm-advice-toggle func-class-advice)
|
|
|
|
|
(describe-function (car func-class-advice))))
|
|
|
|
|
|
|
|
|
|
(defun helm-advice-toggle (func-class-advice)
|
|
|
|
|
(cl-destructuring-bind (function _class advice) func-class-advice
|
|
|
|
|
(cond ((ad-advice-enabled advice)
|
|
|
|
|
(ad-advice-set-enabled advice nil)
|
|
|
|
|
(message "Disabled"))
|
|
|
|
|
(t
|
|
|
|
|
(ad-advice-set-enabled advice t)
|
|
|
|
|
(message "Enabled")))
|
|
|
|
|
(ad-activate function)
|
|
|
|
|
(and helm-in-persistent-action
|
|
|
|
|
(helm-advice-update-current-display-string))))
|
|
|
|
|
|
|
|
|
|
(defun helm-advice-update-current-display-string ()
|
|
|
|
|
(helm-edit-current-selection
|
|
|
|
|
(let ((newword (cond ((looking-at "Disabled") "Enabled")
|
|
|
|
|
((looking-at "Enabled") "Disabled"))))
|
|
|
|
|
(when newword
|
|
|
|
|
(delete-region (point) (progn (forward-word 1) (point)))
|
|
|
|
|
(insert newword)))))
|
|
|
|
|
|
|
|
|
|
;;;###autoload
|
|
|
|
|
(defun helm-manage-advice ()
|
|
|
|
|
"Preconfigured `helm' to disable/enable function advices."
|
|
|
|
|
(interactive)
|
|
|
|
|
(helm-other-buffer 'helm-source-advice "*helm advice*"))
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
;;; Locate elisp library
|
|
|
|
|
;;
|
|
|
|
|
;;
|
|
|
|
|
(defun helm-locate-library-scan-list ()
|
|
|
|
|
(cl-loop for dir in load-path
|
2016-09-15 11:18:17 +02:00
|
|
|
|
with load-suffixes = '(".el")
|
|
|
|
|
when (file-directory-p dir)
|
|
|
|
|
append (directory-files
|
|
|
|
|
dir t (concat (regexp-opt (get-load-suffixes))
|
|
|
|
|
"\\'"))))
|
2016-04-21 23:27:19 +02:00
|
|
|
|
|
|
|
|
|
;;;###autoload
|
|
|
|
|
(defun helm-locate-library ()
|
|
|
|
|
"Preconfigured helm to locate elisp libraries."
|
|
|
|
|
(interactive)
|
|
|
|
|
(helm :sources (helm-build-in-buffer-source "Elisp libraries (Scan)"
|
2016-09-26 19:08:35 +02:00
|
|
|
|
:data #'helm-locate-library-scan-list
|
|
|
|
|
:fuzzy-match helm-locate-library-fuzzy-match
|
2016-04-21 23:27:19 +02:00
|
|
|
|
:keymap helm-generic-files-map
|
2016-09-26 19:08:35 +02:00
|
|
|
|
:search (unless helm-locate-library-fuzzy-match
|
|
|
|
|
(lambda (regexp)
|
|
|
|
|
(re-search-forward
|
|
|
|
|
(if helm-ff-transformer-show-only-basename
|
|
|
|
|
(replace-regexp-in-string
|
|
|
|
|
"\\`\\^" "" regexp)
|
|
|
|
|
regexp)
|
|
|
|
|
nil t)))
|
2016-04-21 23:27:19 +02:00
|
|
|
|
:match-part (lambda (candidate)
|
|
|
|
|
(if helm-ff-transformer-show-only-basename
|
|
|
|
|
(helm-basename candidate) candidate))
|
|
|
|
|
:filter-one-by-one (lambda (c)
|
|
|
|
|
(if helm-ff-transformer-show-only-basename
|
|
|
|
|
(cons (helm-basename c) c) c))
|
|
|
|
|
:action (helm-actions-from-type-file))
|
|
|
|
|
:buffer "*helm locate library*"))
|
|
|
|
|
|
|
|
|
|
(defun helm-set-variable (var)
|
|
|
|
|
"Set value to VAR interactively."
|
|
|
|
|
(let* ((sym (helm-symbolify var))
|
|
|
|
|
(val (default-value sym)))
|
|
|
|
|
(set-default sym (eval-minibuffer (format "Set `%s': " var)
|
|
|
|
|
(if (or (stringp val) (memq val '(nil t)))
|
|
|
|
|
(prin1-to-string val)
|
|
|
|
|
(format "'%s" (prin1-to-string val)))))))
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
;;; Elisp Timers.
|
|
|
|
|
;;
|
|
|
|
|
;;
|
|
|
|
|
(defclass helm-absolute-time-timers-class (helm-source-sync helm-type-timers)
|
|
|
|
|
((candidates :initform timer-list)
|
|
|
|
|
(allow-dups :initform t)
|
|
|
|
|
(candidate-transformer
|
|
|
|
|
:initform
|
|
|
|
|
(lambda (candidates)
|
|
|
|
|
(cl-loop for timer in candidates
|
|
|
|
|
collect (cons (helm-elisp--format-timer timer) timer))))))
|
|
|
|
|
|
|
|
|
|
(defvar helm-source-absolute-time-timers
|
|
|
|
|
(helm-make-source "Absolute Time Timers" 'helm-absolute-time-timers-class))
|
|
|
|
|
|
|
|
|
|
(defclass helm-idle-time-timers-class (helm-source-sync helm-type-timers)
|
|
|
|
|
((candidates :initform timer-idle-list)
|
|
|
|
|
(allow-dups :initform t)
|
|
|
|
|
(candidate-transformer
|
|
|
|
|
:initform
|
|
|
|
|
(lambda (candidates)
|
|
|
|
|
(cl-loop for timer in candidates
|
|
|
|
|
collect (cons (helm-elisp--format-timer timer) timer))))))
|
|
|
|
|
|
|
|
|
|
(defvar helm-source-idle-time-timers
|
|
|
|
|
(helm-make-source "Idle Time Timers" 'helm-idle-time-timers-class))
|
|
|
|
|
|
|
|
|
|
(defun helm-elisp--format-timer (timer)
|
|
|
|
|
(format "%s repeat=%s %s(%s)"
|
|
|
|
|
(let ((time (timer--time timer)))
|
|
|
|
|
(if (timer--idle-delay timer)
|
|
|
|
|
(format-time-string "idle-for=%5s" time)
|
|
|
|
|
(format-time-string "%m/%d %T" time)))
|
|
|
|
|
(or (timer--repeat-delay timer) "nil")
|
|
|
|
|
(mapconcat 'identity (split-string
|
|
|
|
|
(prin1-to-string (timer--function timer))
|
|
|
|
|
"\n") " ")
|
|
|
|
|
(mapconcat 'prin1-to-string (timer--args timer) " ")))
|
|
|
|
|
|
|
|
|
|
;;;###autoload
|
|
|
|
|
(defun helm-timers ()
|
|
|
|
|
"Preconfigured `helm' for timers."
|
|
|
|
|
(interactive)
|
|
|
|
|
(helm :sources '(helm-source-absolute-time-timers
|
|
|
|
|
helm-source-idle-time-timers)
|
|
|
|
|
:buffer "*helm timers*"))
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
;;; Complex command history
|
|
|
|
|
;;
|
|
|
|
|
;;
|
|
|
|
|
(defun helm-btf--usable-p ()
|
|
|
|
|
"Return t if current version of `backtrace-frame' accept 2 arguments."
|
|
|
|
|
(condition-case nil
|
|
|
|
|
(progn (backtrace-frame 1 'condition-case) t)
|
|
|
|
|
(wrong-number-of-arguments nil)))
|
|
|
|
|
|
|
|
|
|
(if (helm-btf--usable-p) ; Check if BTF accept more than one arg.
|
|
|
|
|
;; Emacs 24.4.
|
|
|
|
|
(dont-compile
|
|
|
|
|
(defvar helm-sexp--last-sexp nil)
|
|
|
|
|
;; This wont work compiled.
|
|
|
|
|
(defun helm-sexp-eval-1 ()
|
|
|
|
|
(interactive)
|
|
|
|
|
(unwind-protect
|
|
|
|
|
(progn
|
|
|
|
|
;; Trick called-interactively-p into thinking that `cand' is
|
|
|
|
|
;; an interactive call, See `repeat-complex-command'.
|
|
|
|
|
(add-hook 'called-interactively-p-functions
|
|
|
|
|
#'helm-complex-command-history--called-interactively-skip)
|
|
|
|
|
(eval (read helm-sexp--last-sexp)))
|
|
|
|
|
(remove-hook 'called-interactively-p-functions
|
|
|
|
|
#'helm-complex-command-history--called-interactively-skip)))
|
|
|
|
|
|
|
|
|
|
(defun helm-complex-command-history--called-interactively-skip (i _frame1 frame2)
|
|
|
|
|
(and (eq 'eval (cadr frame2))
|
|
|
|
|
(eq 'helm-sexp-eval-1
|
|
|
|
|
(cadr (backtrace-frame (+ i 2) #'called-interactively-p)))
|
|
|
|
|
1))
|
|
|
|
|
|
|
|
|
|
(defun helm-sexp-eval (_candidate)
|
|
|
|
|
(call-interactively #'helm-sexp-eval-1)))
|
|
|
|
|
;; Emacs 24.3
|
|
|
|
|
(defun helm-sexp-eval (cand)
|
|
|
|
|
(let ((sexp (read cand)))
|
|
|
|
|
(condition-case err
|
|
|
|
|
(if (> (length (remove nil sexp)) 1)
|
|
|
|
|
(eval sexp)
|
|
|
|
|
(apply 'call-interactively sexp))
|
|
|
|
|
(error (message "Evaluating gave an error: %S" err)
|
|
|
|
|
nil)))))
|
|
|
|
|
|
|
|
|
|
(defvar helm-source-complex-command-history
|
|
|
|
|
(helm-build-sync-source "Complex Command History"
|
|
|
|
|
:candidates (lambda ()
|
|
|
|
|
;; Use cdr to avoid adding
|
|
|
|
|
;; `helm-complex-command-history' here.
|
|
|
|
|
(cl-loop for i in command-history
|
|
|
|
|
unless (equal i '(helm-complex-command-history))
|
|
|
|
|
collect (prin1-to-string i)))
|
|
|
|
|
:action (helm-make-actions
|
|
|
|
|
"Eval" (lambda (candidate)
|
|
|
|
|
(and (boundp 'helm-sexp--last-sexp)
|
|
|
|
|
(setq helm-sexp--last-sexp candidate))
|
|
|
|
|
(let ((command (read candidate)))
|
|
|
|
|
(unless (equal command (car command-history))
|
|
|
|
|
(setq command-history (cons command command-history))))
|
|
|
|
|
(run-with-timer 0.1 nil #'helm-sexp-eval candidate))
|
|
|
|
|
"Edit and eval" (lambda (candidate)
|
|
|
|
|
(edit-and-eval-command "Eval: " (read candidate))))
|
|
|
|
|
:persistent-action #'helm-sexp-eval
|
|
|
|
|
:multiline t))
|
|
|
|
|
|
|
|
|
|
;;;###autoload
|
|
|
|
|
(defun helm-complex-command-history ()
|
|
|
|
|
"Preconfigured helm for complex command history."
|
|
|
|
|
(interactive)
|
|
|
|
|
(helm :sources 'helm-source-complex-command-history
|
|
|
|
|
:buffer "*helm complex commands*"))
|
|
|
|
|
|
|
|
|
|
(provide 'helm-elisp)
|
|
|
|
|
|
|
|
|
|
;; Local Variables:
|
|
|
|
|
;; byte-compile-warnings: (not cl-functions obsolete)
|
|
|
|
|
;; coding: utf-8
|
|
|
|
|
;; indent-tabs-mode: nil
|
|
|
|
|
;; End:
|
|
|
|
|
|
|
|
|
|
;;; helm-elisp.el ends here
|