1126 lines
46 KiB
EmacsLisp
1126 lines
46 KiB
EmacsLisp
;;; helm-utils.el --- Utilities Functions for helm. -*- lexical-binding: t -*-
|
||
|
||
;; Copyright (C) 2012 ~ 2021 Thierry Volpiatto
|
||
|
||
;; 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)
|
||
|
||
(declare-function helm-find-files-1 "helm-files" (fname &optional preselect))
|
||
(declare-function helm-grep-split-line "helm-grep" (line))
|
||
(declare-function popup-tip "ext:popup")
|
||
(declare-function markdown-show-entry "ext:markdown-mode.el")
|
||
(declare-function outline-show-subtree "outline")
|
||
(declare-function org-reveal "org")
|
||
(declare-function hs-show-block "hideshow.el")
|
||
(declare-function tab-bar-tabs "tab-bar")
|
||
(declare-function tab-bar-select-tab "tab-bar")
|
||
(declare-function dired-goto-file "dired")
|
||
(declare-function bookmark-get-filename "bookmark")
|
||
(declare-function package-installed-p "package")
|
||
(declare-function package-desc-dir "package")
|
||
|
||
(defvar hs-minor-mode)
|
||
(defvar hs-show-hook)
|
||
(defvar org-directory)
|
||
(defvar winner-boring-buffers)
|
||
(defvar bookmark-alist)
|
||
(defvar dired-buffers)
|
||
(defvar helm-show-completion-overlay)
|
||
(defvar helm-buffers-maybe-switch-to-tab)
|
||
(defvar helm-ff-transformer-show-only-basename)
|
||
(defvar helm-popup-tip-mode)
|
||
(defvar helm-ff-last-expanded-candidate-regexp)
|
||
|
||
|
||
(defgroup helm-utils nil
|
||
"Utilities routines for Helm."
|
||
:group 'helm)
|
||
|
||
(defcustom helm-su-or-sudo "sudo"
|
||
"What command to use for root access."
|
||
:type 'string
|
||
:group 'helm-utils)
|
||
|
||
(defcustom helm-default-kbsize 1024.0
|
||
"Default Kbsize to use for showing files size.
|
||
It is a float, usually 1024.0 but could be 1000.0 on some systems."
|
||
:group 'helm-utils
|
||
:type 'float)
|
||
|
||
(define-obsolete-variable-alias
|
||
'helm-highlight-number-lines-around-point
|
||
'helm-highlight-matches-around-point-max-lines
|
||
"20160119")
|
||
|
||
(defcustom helm-highlight-matches-around-point-max-lines '(15 . 15)
|
||
"Number of lines around point where matched items are highlighted.
|
||
|
||
Possible value are:
|
||
- A cons cell (x . y)
|
||
Match x lines before point and y lines after point.
|
||
- An integer
|
||
Positive means this number lines after point.
|
||
Negative means this number lines before point.
|
||
A zero value means highlight only inside matched lines.
|
||
- The symbol never
|
||
Means do not highlight matched items. "
|
||
:group 'helm-utils
|
||
:type '(choice (cons (integer :tag "Match before")
|
||
(integer :tag "Match after"))
|
||
(const :tag "Match in line only" 0)
|
||
(integer :tag "Match after or before (+/-)")
|
||
(const :tag "Never match" never)))
|
||
|
||
(defcustom helm-highlight-only-all-matches nil
|
||
"Highlight only when all items match on the line when non nil.
|
||
See `helm-highlight-current-line'."
|
||
:group 'helm-utils
|
||
:type 'boolean)
|
||
|
||
(defcustom helm-buffers-to-resize-on-pa nil
|
||
"A list of helm buffers where the helm-window should be reduced on PA.
|
||
Where PA means persistent action."
|
||
:group 'helm-utils
|
||
:type '(repeat (choice string)))
|
||
|
||
(defcustom helm-resize-on-pa-text-height 12
|
||
"The size of the helm-window when resizing on persistent action."
|
||
:group 'helm-utils
|
||
:type 'integer)
|
||
|
||
(defcustom helm-sources-using-help-echo-popup '("Ack-Grep" "AG" "RG" "Gid" "Git-Grep")
|
||
"Show the buffer name or the filename in a popup at selection."
|
||
:group 'helm-utils
|
||
:type '(repeat (choice string)))
|
||
|
||
(defcustom helm-html-decode-entities-function #'helm-html-decode-entities-string
|
||
"Function used to decode HTML entities in HTML bookmarks.
|
||
Helm comes by default with `helm-html-decode-entities-string', if
|
||
you need something more sophisticated you can use
|
||
`w3m-decode-entities-string' if available.
|
||
|
||
In Emacs itself org-entities seem broken and `xml-substitute-numeric-entities'
|
||
supports only numeric entities."
|
||
:group 'helm-utils
|
||
:type 'function)
|
||
|
||
|
||
(defvar helm-goto-line-before-hook '(helm-save-current-pos-to-mark-ring)
|
||
"Run before jumping to line.
|
||
This hook runs when jumping from `helm-goto-line', `helm-etags-default-action',
|
||
and `helm-imenu-default-action'.
|
||
This allows you to retrieve a previous position after using the different helm
|
||
tools for searching (etags, grep, gid, (m)occur etc...).
|
||
By default positions are added to `mark-ring'.
|
||
You can also add to register by using (or adding)
|
||
`helm-save-pos-to-register-before-jump' instead. In this case
|
||
last position is added to the register `helm-save-pos-before-jump-register'.")
|
||
|
||
(defvar helm-save-pos-before-jump-register ?_
|
||
"The register where `helm-save-pos-to-register-before-jump' saves position.")
|
||
|
||
(defconst helm-html-entities-alist
|
||
'((""" . 34) ;; "
|
||
(">" . 62) ;; >
|
||
("<" . 60) ;; <
|
||
("&" . 38) ;; &
|
||
("€" . 8364) ;; €
|
||
("Ÿ" . 89) ;; Y
|
||
("¡" . 161) ;; ¡
|
||
("¢" . 162) ;; ¢
|
||
("£" . 163) ;; £
|
||
("¤" . 164) ;; ¤
|
||
("¥" . 165) ;; ¥
|
||
("¦" . 166) ;; ¦
|
||
("§" . 167) ;; §
|
||
("¨" . 32) ;; SPC
|
||
(" " . 160) ;; (non breaking space)
|
||
("©" . 169) ;; ©
|
||
("ª" . 97) ;; a
|
||
("«" . 171) ;; «
|
||
("¬" . 172) ;; ¬
|
||
("&masr;" . 174) ;; ®
|
||
("°" . 176) ;; °
|
||
("±" . 177) ;; ±
|
||
("²" . 50) ;; 2
|
||
("³" . 51) ;; 3
|
||
("´" . 39) ;; '
|
||
("µ" . 956) ;; μ
|
||
("¶" . 182) ;; ¶
|
||
("·" . 183) ;; ·
|
||
("¸" . 32) ;; SPC
|
||
("¹" . 49) ;; 1
|
||
("º" . 111) ;; o
|
||
("»" . 187) ;; »
|
||
("¼" . 49) ;; 1
|
||
("½" . 49) ;; 1
|
||
("¾" . 51) ;; 3
|
||
("¿" . 191) ;; ¿
|
||
("À" . 192) ;; À
|
||
("Á" . 193) ;; Á
|
||
("Â" . 194) ;; Â
|
||
("Ã" . 195) ;; Ã
|
||
("Ä" . 196) ;; Ä
|
||
("Å" . 197) ;; Å
|
||
("&Aelig" . 198) ;; Æ
|
||
("Ç" . 199) ;; Ç
|
||
("È" . 200) ;; È
|
||
("É" . 201) ;; É
|
||
("Ê" . 202) ;; Ê
|
||
("Ë" . 203) ;; Ë
|
||
("Ì" . 204) ;; Ì
|
||
("Í" . 205) ;; Í
|
||
("Î" . 206) ;; Î
|
||
("Ï" . 207) ;; Ï
|
||
("ð" . 208) ;; Ð
|
||
("Ñ" . 209) ;; Ñ
|
||
("Ò" . 210) ;; Ò
|
||
("Ó" . 211) ;; Ó
|
||
("Ô" . 212) ;; Ô
|
||
("Õ" . 213) ;; Õ
|
||
("Ö" . 214) ;; Ö
|
||
("×" . 215) ;; ×
|
||
("Ø" . 216) ;; Ø
|
||
("Ù" . 217) ;; Ù
|
||
("Ú" . 218) ;; Ú
|
||
("Û" . 219) ;; Û
|
||
("Ü" . 220) ;; Ü
|
||
("Ý" . 221) ;; Ý
|
||
("þ" . 222) ;; Þ
|
||
("ß" . 223) ;; ß
|
||
("à" . 224) ;; à
|
||
("á" . 225) ;; á
|
||
("â" . 226) ;; â
|
||
("ã" . 227) ;; ã
|
||
("ä" . 228) ;; ä
|
||
("å" . 229) ;; å
|
||
("æ" . 230) ;; æ
|
||
("ç" . 231) ;; ç
|
||
("è" . 232) ;; è
|
||
("é" . 233) ;; é
|
||
("ê" . 234) ;; ê
|
||
("ë" . 235) ;; ë
|
||
("ì" . 236) ;; ì
|
||
("í" . 237) ;; í
|
||
("î" . 238) ;; î
|
||
("ï" . 239) ;; ï
|
||
("ð" . 240) ;; ð
|
||
("ñ" . 241) ;; ñ
|
||
("ò" . 242) ;; ò
|
||
("ó" . 243) ;; ó
|
||
("ô" . 244) ;; ô
|
||
("õ" . 245) ;; õ
|
||
("ö" . 246) ;; ö
|
||
("÷" . 247) ;; ÷
|
||
("ø" . 248) ;; ø
|
||
("ù" . 249) ;; ù
|
||
("ú" . 250) ;; ú
|
||
("û" . 251) ;; û
|
||
("ü" . 252) ;; ü
|
||
("ý" . 253) ;; ý
|
||
("þ" . 254) ;; þ
|
||
("ÿ" . 255) ;; ÿ
|
||
("®" . 174) ;; ®
|
||
("­" . 173)) ;;
|
||
|
||
"Table of html character entities and values.
|
||
See https://www.freeformatter.com/html-entities.html")
|
||
|
||
(defvar helm-find-many-files-after-hook nil
|
||
"Hook that runs at end of `helm-find-many-files'.")
|
||
|
||
;;; Faces.
|
||
;;
|
||
(defface helm-selection-line
|
||
`((t ,@(and (>= emacs-major-version 27) '(:extend t))
|
||
:inherit highlight :distant-foreground "black"))
|
||
"Face used in the `helm-current-buffer' when jumping to a candidate."
|
||
:group 'helm-faces)
|
||
|
||
(defface helm-match-item
|
||
`((t ,@(and (>= emacs-major-version 27) '(:extend t))
|
||
:inherit isearch))
|
||
"Face used to highlight the item matched in a selected line."
|
||
:group 'helm-faces)
|
||
|
||
|
||
;;; Utils functions
|
||
;;
|
||
;;
|
||
(defcustom helm-window-prefer-horizontal-split nil
|
||
"Maybe switch to other window vertically when non nil.
|
||
|
||
Possible values are t, nil and `decide'.
|
||
|
||
When t switch vertically.
|
||
When nil switch horizontally.
|
||
When `decide' try to guess if it is possible to switch vertically
|
||
according to the setting of `split-width-threshold' and the size of
|
||
the window from where splitting is done.
|
||
|
||
Note that when using `decide' and `split-width-threshold' is nil, the
|
||
behavior is the same as with a nil value."
|
||
:group 'helm-utils
|
||
:type '(choice
|
||
(const :tag "Split window vertically" t)
|
||
(const :tag "Split window horizontally" nil)
|
||
(symbol :tag "Guess how to split window" 'decide)))
|
||
|
||
(defcustom helm-window-show-buffers-function #'helm-window-default-split-fn
|
||
"The default function to use when opening several buffers at once.
|
||
It is typically used to rearrange windows."
|
||
:group 'helm-utils
|
||
:type '(choice
|
||
(function :tag "Split windows vertically or horizontally"
|
||
helm-window-default-split-fn)
|
||
(function :tag "Split in alternate windows"
|
||
helm-window-alternate-split-fn)
|
||
(function :tag "Split windows in mosaic"
|
||
helm-window-mosaic-fn)))
|
||
|
||
(defun helm-window-show-buffers (buffers &optional other-window)
|
||
"Show BUFFERS.
|
||
|
||
With more than one buffer marked switch to these buffers in separate windows.
|
||
If OTHER-WINDOW is non-nil, keep current buffer and switch to other buffers
|
||
in separate windows.
|
||
If a prefix arg is given split windows vertically."
|
||
(let ((initial-ow-fn (if (cdr (window-list))
|
||
#'switch-to-buffer-other-window
|
||
#'helm-window-other-window)))
|
||
(if (cdr buffers)
|
||
(funcall helm-window-show-buffers-function buffers
|
||
(and other-window initial-ow-fn))
|
||
(if other-window
|
||
(funcall initial-ow-fn (car buffers))
|
||
(helm-buffers-switch-to-buffer-or-tab (car buffers))))))
|
||
|
||
(defvar tab-bar-tab-name-function)
|
||
(declare-function tab-bar-switch-to-tab "tab-bar.el")
|
||
(declare-function tab-bar-tab-name-all "tab-bar.el")
|
||
|
||
(defun helm-buffers-switch-to-buffer-or-tab (buffer)
|
||
"Switch to BUFFER in its tab if some."
|
||
(if (and (fboundp 'tab-bar-mode)
|
||
helm-buffers-maybe-switch-to-tab)
|
||
(let* ((tab-bar-tab-name-function #'tab-bar-tab-name-all)
|
||
(tabs (tab-bar-tabs))
|
||
(tab-names (mapcar (lambda (tab)
|
||
(cdr (assq 'name tab)))
|
||
tabs))
|
||
(bname (buffer-name (get-buffer buffer)))
|
||
(tab (helm-buffers--get-tab-from-name bname tabs)))
|
||
(if (helm-buffers--buffer-in-tab-p bname tab-names)
|
||
(progn
|
||
(tab-bar-switch-to-tab (alist-get 'name tab))
|
||
(select-window (get-buffer-window bname)))
|
||
(switch-to-buffer buffer)))
|
||
(switch-to-buffer buffer)))
|
||
|
||
(defun helm-buffers--get-tab-from-name (tab-name tabs)
|
||
"Return tab from TABS when it contains TAB-NAME."
|
||
(cl-loop for tab in tabs
|
||
when (member tab-name (split-string (cdr (assq 'name tab)) ", " t))
|
||
return tab))
|
||
|
||
(defun helm-buffers--buffer-in-tab-p (buffer-name tab-names)
|
||
"Check if BUFFER-NAME is in TAB-NAMES list."
|
||
(cl-loop for name in tab-names
|
||
;; Buf names are separated with "," in TAB-NAMES
|
||
;; e.g. '("tab-bar.el" "*scratch*, helm-buffers.el").
|
||
thereis (member buffer-name (split-string name ", " t))))
|
||
|
||
(defun helm-window-default-split-fn (candidates &optional other-window-fn)
|
||
"Split windows in one direction and balance them.
|
||
|
||
Direction can be controlled via `helm-window-prefer-horizontal-split'.
|
||
If a prefix arg is given split windows the other direction.
|
||
This function is suitable for `helm-window-show-buffers-function'."
|
||
(if other-window-fn
|
||
(funcall other-window-fn (car candidates))
|
||
(switch-to-buffer (car candidates)))
|
||
(save-selected-window
|
||
(cl-loop with nosplit
|
||
for b in (cdr candidates)
|
||
when nosplit return
|
||
(message "Too many buffers to visit simultaneously")
|
||
do (condition-case _err
|
||
(helm-window-other-window b 'balance)
|
||
(error (setq nosplit t) nil)))))
|
||
|
||
(defun helm-window-alternate-split-fn (candidates &optional other-window-fn)
|
||
"Split windows horizontally and vertically in alternate fashion.
|
||
|
||
Direction can be controlled via `helm-window-prefer-horizontal-split'.
|
||
If a prefix arg is given split windows the other direction.
|
||
This function is suitable for `helm-window-show-buffers-function'."
|
||
(if other-window-fn
|
||
(funcall other-window-fn (car candidates))
|
||
(switch-to-buffer (car candidates)))
|
||
(let (right-side)
|
||
(save-selected-window
|
||
(cl-loop with nosplit
|
||
for b in (cdr candidates)
|
||
when nosplit return
|
||
(message "Too many buffers to visit simultaneously")
|
||
do (condition-case _err
|
||
(let ((helm-current-prefix-arg right-side))
|
||
(helm-window-other-window b)
|
||
(setq right-side (not right-side)))
|
||
(error (setq nosplit t) nil))))))
|
||
|
||
(defun helm-window-mosaic-fn (candidates &optional other-window-fn)
|
||
"Make an as-square-as-possible window mosaic of the CANDIDATES buffers.
|
||
|
||
If rectangular, the long side is in the direction given by
|
||
`helm-window-prefer-horizontal-split': if non-nil, it is horizontal, vertical
|
||
otherwise.
|
||
If OTHER-WINDOW-FN is non-nil, current windows are included in the mosaic.
|
||
This function is suitable for `helm-window-show-buffers-function'."
|
||
(when other-window-fn
|
||
(setq candidates (append (mapcar 'window-buffer (window-list)) candidates)))
|
||
(delete-other-windows)
|
||
(let* ((helm-window-prefer-horizontal-split
|
||
(if (eq helm-window-prefer-horizontal-split 'decide)
|
||
(and (numberp split-width-threshold)
|
||
(>= (window-width (selected-window))
|
||
split-width-threshold))
|
||
helm-window-prefer-horizontal-split))
|
||
mosaic-length-tile-count
|
||
mosaic-width-tile-count
|
||
mosaic-length-tile-size
|
||
mosaic-width-tile-size
|
||
next-window)
|
||
;; If 4 tiles, make 2x2 mosaic.
|
||
;; If 5-6 tiles, make 2x3 mosaic with direction depending on `helm-window-prefer-horizontal-split'.
|
||
;; If 7-9 tiles, make 3x3 mosaic. And so on.
|
||
(setq mosaic-length-tile-count (ceiling (sqrt (length candidates))))
|
||
(setq mosaic-width-tile-count
|
||
(if (<= (length candidates) (* mosaic-length-tile-count (1- mosaic-length-tile-count)))
|
||
(1- mosaic-length-tile-count)
|
||
mosaic-length-tile-count))
|
||
;; We lower-bound the tile size, otherwise the function would
|
||
;; fail during the first inner split.
|
||
;; There is consequently no need to check for errors when
|
||
;; splitting.
|
||
(let ((frame-mosaic-length-direction-size (frame-height))
|
||
(frame-mosaic-width-direction-size (frame-width))
|
||
(window-mosaic-length-direction-min-size window-min-height)
|
||
(window-mosaic-width-direction-min-size window-min-width))
|
||
(if helm-window-prefer-horizontal-split
|
||
(setq frame-mosaic-length-direction-size (frame-width)
|
||
frame-mosaic-width-direction-size (frame-height)
|
||
window-mosaic-length-direction-min-size window-min-width
|
||
window-mosaic-width-direction-min-size window-min-height))
|
||
(setq mosaic-length-tile-size (max
|
||
(/ frame-mosaic-length-direction-size mosaic-length-tile-count)
|
||
window-mosaic-length-direction-min-size)
|
||
mosaic-width-tile-size (max
|
||
(/ frame-mosaic-width-direction-size mosaic-width-tile-count)
|
||
window-mosaic-width-direction-min-size))
|
||
;; Shorten `candidates' to `max-tiles' elements.
|
||
(let ((max-tiles (* (/ frame-mosaic-length-direction-size mosaic-length-tile-size)
|
||
(/ frame-mosaic-width-direction-size mosaic-width-tile-size))))
|
||
(when (> (length candidates) max-tiles)
|
||
(message "Too many buffers to visit simultaneously")
|
||
(setcdr (nthcdr (- max-tiles 1) candidates) nil))))
|
||
;; Make the mosaic.
|
||
(while candidates
|
||
(when (> (length candidates) mosaic-length-tile-count)
|
||
(setq next-window (split-window nil
|
||
mosaic-width-tile-size
|
||
(not helm-window-prefer-horizontal-split))))
|
||
(switch-to-buffer (pop candidates))
|
||
(dotimes (_ (min (1- mosaic-length-tile-count) (length candidates)))
|
||
(select-window (split-window nil
|
||
mosaic-length-tile-size
|
||
helm-window-prefer-horizontal-split))
|
||
(switch-to-buffer (pop candidates)))
|
||
(when next-window
|
||
(select-window next-window)))))
|
||
|
||
(defun helm-window-other-window (buffer-or-name &optional balance)
|
||
"Switch to BUFFER-OR-NAME in other window.
|
||
Direction can be controlled via `helm-window-prefer-horizontal-split'.
|
||
If a prefix arg is given split windows the other direction.
|
||
When argument BALANCE is provided `balance-windows'."
|
||
(let* ((helm-window-prefer-horizontal-split
|
||
(if (eq helm-window-prefer-horizontal-split 'decide)
|
||
(and (numberp split-width-threshold)
|
||
(>= (window-width (selected-window))
|
||
split-width-threshold))
|
||
helm-window-prefer-horizontal-split))
|
||
(right-side (if helm-window-prefer-horizontal-split
|
||
(not helm-current-prefix-arg)
|
||
helm-current-prefix-arg)))
|
||
(select-window (split-window nil nil right-side))
|
||
(and balance (balance-windows))
|
||
(switch-to-buffer buffer-or-name)))
|
||
|
||
(cl-defun helm-current-buffer-narrowed-p (&optional
|
||
(buffer helm-current-buffer))
|
||
"Check if BUFFER is narrowed.
|
||
Default is `helm-current-buffer'."
|
||
(with-current-buffer buffer
|
||
(let ((beg (point-min))
|
||
(end (point-max))
|
||
(total (buffer-size)))
|
||
(or (/= beg 1) (/= end (1+ total))))))
|
||
|
||
(defun helm-goto-char (loc)
|
||
"Go to char, revealing if necessary."
|
||
(goto-char loc)
|
||
(let ((fn (cond ((eq major-mode 'org-mode)
|
||
;; On some old Emacs versions org may not be loaded.
|
||
(require 'org)
|
||
#'org-reveal)
|
||
((and (boundp 'outline-minor-mode)
|
||
outline-minor-mode)
|
||
#'outline-show-subtree)
|
||
((and (boundp 'hs-minor-mode)
|
||
hs-minor-mode)
|
||
#'hs-show-block)
|
||
((and (boundp 'markdown-mode-map)
|
||
(derived-mode-p 'markdown-mode))
|
||
#'markdown-show-entry)))
|
||
(hs-show-hook (list (lambda () (goto-char loc)))))
|
||
;; outline may fail in some conditions e.g. with markdown enabled
|
||
;; (Bug#1919).
|
||
(condition-case-unless-debug nil
|
||
(and fn (funcall fn))
|
||
(error nil))))
|
||
|
||
(defun helm-goto-line (lineno &optional noanim)
|
||
"Goto LINENO opening only outline headline if needed.
|
||
Animation is used unless NOANIM is non--nil."
|
||
(helm-log-run-hook 'helm-goto-line-before-hook)
|
||
(helm-match-line-cleanup)
|
||
(unless helm-alive-p
|
||
(with-helm-current-buffer
|
||
(unless helm-yank-point (setq helm-yank-point (point)))))
|
||
(goto-char (point-min))
|
||
(helm-goto-char (point-at-bol lineno))
|
||
(unless noanim
|
||
(helm-highlight-current-line)))
|
||
|
||
(defun helm-save-pos-to-register-before-jump ()
|
||
"Save current buffer position to `helm-save-pos-before-jump-register'.
|
||
To use this add it to `helm-goto-line-before-hook'."
|
||
(with-helm-current-buffer
|
||
(unless helm-in-persistent-action
|
||
(point-to-register helm-save-pos-before-jump-register))))
|
||
|
||
(defun helm-save-current-pos-to-mark-ring ()
|
||
"Save current buffer position to mark ring.
|
||
To use this add it to `helm-goto-line-before-hook'."
|
||
(with-helm-current-buffer
|
||
(unless helm-in-persistent-action
|
||
(set-marker (mark-marker) (point))
|
||
(push-mark (point) 'nomsg))))
|
||
|
||
(defun helm-displaying-source-names ()
|
||
"Return the list of sources name for this helm session."
|
||
(with-current-buffer helm-buffer
|
||
(goto-char (point-min))
|
||
(cl-loop with pos
|
||
while (setq pos (next-single-property-change (point) 'helm-header))
|
||
do (goto-char pos)
|
||
collect (buffer-substring-no-properties (point-at-bol)(point-at-eol))
|
||
do (forward-line 1))))
|
||
|
||
(defun helm-handle-winner-boring-buffers ()
|
||
"Add `helm-buffer' to `winner-boring-buffers' when quitting/exiting helm.
|
||
Add this function to `helm-cleanup-hook' when you don't want to see helm buffers
|
||
after running winner-undo/redo."
|
||
(require 'winner)
|
||
(cl-pushnew helm-buffer winner-boring-buffers :test 'equal))
|
||
(add-hook 'helm-cleanup-hook #'helm-handle-winner-boring-buffers)
|
||
|
||
(defun helm-quit-and-find-file ()
|
||
"Drop into `helm-find-files' from `helm'.
|
||
If current selection is a buffer or a file, `helm-find-files'
|
||
from its directory."
|
||
(interactive)
|
||
(with-helm-alive-p
|
||
(require 'helm-grep)
|
||
(require 'helm-elisp)
|
||
(require 'bookmark) ; For bookmark-alist
|
||
(let ((src (helm-get-current-source)))
|
||
(helm-run-after-exit
|
||
(lambda (f)
|
||
;; Ensure specifics `helm-execute-action-at-once-if-one'
|
||
;; fns don't run here.
|
||
(let (helm-execute-action-at-once-if-one
|
||
helm-actions-inherit-frame-settings) ; use this-command
|
||
(if (file-exists-p f)
|
||
(helm-find-files-1 (file-name-directory f)
|
||
(format
|
||
helm-ff-last-expanded-candidate-regexp
|
||
(regexp-quote
|
||
(if helm-ff-transformer-show-only-basename
|
||
(helm-basename f) f))))
|
||
(helm-find-files-1 f))))
|
||
(helm--quit-and-find-file-default-file src)))))
|
||
(put 'helm-quit-and-find-file 'helm-only t)
|
||
|
||
(defun helm--quit-and-find-file-default-file (source)
|
||
(let ((target-fn (helm-get-attr 'find-file-target)))
|
||
;; target-fn function may return nil, in this case fallback to default.
|
||
(helm-aif (and target-fn (funcall target-fn source))
|
||
it
|
||
(let* ((sel (helm-get-selection nil nil source))
|
||
(default-preselection (or (helm-default-directory)
|
||
(buffer-file-name helm-current-buffer)
|
||
default-directory)))
|
||
(cond
|
||
((and (stringp sel) (or (file-remote-p sel)
|
||
(file-exists-p sel)))
|
||
(expand-file-name sel))
|
||
;; Url.
|
||
((and (stringp sel)
|
||
helm--url-regexp
|
||
(string-match helm--url-regexp sel))
|
||
sel)
|
||
;; Exit brutally from a `with-helm-show-completion'
|
||
((and helm-show-completion-overlay
|
||
(overlayp helm-show-completion-overlay))
|
||
(delete-overlay helm-show-completion-overlay)
|
||
(remove-hook 'helm-move-selection-after-hook 'helm-show-completion)
|
||
(expand-file-name default-preselection))
|
||
;; Default.
|
||
(t (expand-file-name default-preselection)))))))
|
||
|
||
(defun helm-generic-sort-fn (s1 s2)
|
||
"Sort predicate function for helm candidates.
|
||
Args S1 and S2 can be single or (display . real) candidates,
|
||
that is sorting is done against real value of candidate."
|
||
(let* ((qpattern (regexp-quote helm-pattern))
|
||
(reg1 (concat "\\_<" qpattern "\\_>"))
|
||
(reg2 (concat "\\_<" qpattern))
|
||
(reg3 helm-pattern)
|
||
(split (helm-remove-if-match
|
||
"\\`!" (helm-mm-split-pattern helm-pattern)))
|
||
(str1 (if (consp s1) (cdr s1) s1))
|
||
(str2 (if (consp s2) (cdr s2) s2))
|
||
(score (lambda (str r1 r2 r3 lst)
|
||
(+ (if (string-match (concat "\\`" qpattern) str) 1 0)
|
||
(cond ((string-match r1 str) 5)
|
||
((and (string-match " " qpattern)
|
||
(string-match
|
||
(concat "\\_<" (regexp-quote (car lst))) str)
|
||
(cl-loop for r in (cdr lst)
|
||
always (string-match r str)))
|
||
4)
|
||
((and (string-match " " qpattern)
|
||
(cl-loop for r in lst
|
||
always (string-match r str)))
|
||
3)
|
||
((string-match r2 str) 2)
|
||
((string-match r3 str) 1)
|
||
(t 0)))))
|
||
(sc1 (get-text-property 0 'completion-score str1))
|
||
(sc2 (get-text-property 0 'completion-score str2))
|
||
(sc3 (if sc1 0 (funcall score str1 reg1 reg2 reg3 split)))
|
||
(sc4 (if sc2 0 (funcall score str2 reg1 reg2 reg3 split))))
|
||
(cond ((and sc1 sc2) ; helm-flex style.
|
||
(> sc1 sc2))
|
||
((or (zerop (string-width qpattern))
|
||
(and (zerop sc3) (zerop sc4)))
|
||
(string-lessp str1 str2))
|
||
((= sc3 sc4)
|
||
(< (length str1) (length str2)))
|
||
(t (> sc3 sc4)))))
|
||
|
||
(cl-defun helm-file-human-size (size &optional (kbsize helm-default-kbsize))
|
||
"Return a string showing SIZE of a file in human readable form.
|
||
SIZE can be an integer or a float depending on it's value.
|
||
`file-attributes' will take care of that to avoid overflow error.
|
||
KBSIZE is a floating point number, defaulting to `helm-default-kbsize'."
|
||
(cl-loop with result = (cons "B" size)
|
||
for i in '("k" "M" "G" "T" "P" "E" "Z" "Y")
|
||
while (>= (cdr result) kbsize)
|
||
do (setq result (cons i (/ (cdr result) kbsize)))
|
||
finally return
|
||
(helm-acase (car result)
|
||
("B" (format "%s" size))
|
||
(t (format "%.1f%s" (cdr result) it)))))
|
||
|
||
(defun helm-directory-size (directory &optional recursive human)
|
||
"Return the resulting size of the sum of all files in DIRECTORY.
|
||
|
||
If RECURSIVE is non nil return the size of all files in DIRECTORY and
|
||
its subdirectories. With arg HUMAN format the size in a human
|
||
readable format,see `helm-file-human-size'."
|
||
(cl-loop with files = (if recursive
|
||
(helm-walk-directory
|
||
directory
|
||
:path 'full
|
||
:directories t)
|
||
(directory-files directory t))
|
||
for file in files
|
||
sum (nth 7 (file-attributes file)) into total
|
||
finally return (if human
|
||
(helm-file-human-size total)
|
||
total)))
|
||
|
||
(cl-defun helm-file-attributes
|
||
(file &key type links uid gid access-time modif-time
|
||
status size mode gid-change inode device-num dired human-size
|
||
mode-type mode-owner mode-group mode-other (string t))
|
||
"Return `file-attributes' elements of FILE separately according to key value.
|
||
Availables keys are:
|
||
- TYPE: Same as nth 0 `files-attributes' if STRING is nil
|
||
otherwise return either symlink, directory or file (default).
|
||
- LINKS: See nth 1 `files-attributes'.
|
||
- UID: See nth 2 `files-attributes'.
|
||
- GID: See nth 3 `files-attributes'.
|
||
- ACCESS-TIME: See nth 4 `files-attributes', however format time
|
||
when STRING is non--nil (the default).
|
||
- MODIF-TIME: See nth 5 `files-attributes', same as above.
|
||
- STATUS: See nth 6 `files-attributes', same as above.
|
||
- SIZE: See nth 7 `files-attributes'.
|
||
- MODE: See nth 8 `files-attributes'.
|
||
- GID-CHANGE: See nth 9 `files-attributes'.
|
||
- INODE: See nth 10 `files-attributes'.
|
||
- DEVICE-NUM: See nth 11 `files-attributes'.
|
||
- DIRED: A line similar to what \\='ls -l' return.
|
||
- HUMAN-SIZE: The size in human form, see `helm-file-human-size'.
|
||
- MODE-TYPE, mode-owner,mode-group, mode-other: Split what
|
||
nth 7 `files-attributes' return in four categories.
|
||
- STRING: When non--nil (default) `helm-file-attributes' return
|
||
more friendly values.
|
||
If you want the same behavior as `files-attributes' ,
|
||
\(but with return values in proplist) use a nil value for STRING.
|
||
However when STRING is non--nil, time and type value are different from what
|
||
you have in `file-attributes'."
|
||
(helm-aif (file-attributes file string)
|
||
(let* ((all (cl-destructuring-bind
|
||
(type links uid gid access-time modif-time
|
||
status size mode gid-change inode device-num)
|
||
it
|
||
(list :type (if string
|
||
(cond ((stringp type) "symlink") ; fname
|
||
(type "directory") ; t
|
||
(t "file")) ; nil
|
||
type)
|
||
:links links
|
||
:uid uid
|
||
:gid gid
|
||
:access-time (if string
|
||
(format-time-string
|
||
"%Y-%m-%d %R" access-time)
|
||
access-time)
|
||
:modif-time (if string
|
||
(format-time-string
|
||
"%Y-%m-%d %R" modif-time)
|
||
modif-time)
|
||
:status (if string
|
||
(format-time-string
|
||
"%Y-%m-%d %R" status)
|
||
status)
|
||
:size size
|
||
:mode mode
|
||
:gid-change gid-change
|
||
:inode inode
|
||
:device-num device-num)))
|
||
(modes (helm-split-mode-file-attributes (cl-getf all :mode))))
|
||
(cond (type (cl-getf all :type))
|
||
(links (cl-getf all :links))
|
||
(uid (cl-getf all :uid))
|
||
(gid (cl-getf all :gid))
|
||
(access-time (cl-getf all :access-time))
|
||
(modif-time (cl-getf all :modif-time))
|
||
(status (cl-getf all :status))
|
||
(size (cl-getf all :size))
|
||
(mode (cl-getf all :mode))
|
||
(gid-change (cl-getf all :gid-change))
|
||
(inode (cl-getf all :inode))
|
||
(device-num (cl-getf all :device-num))
|
||
(dired (concat
|
||
(helm-split-mode-file-attributes
|
||
(cl-getf all :mode) t) " "
|
||
(number-to-string (cl-getf all :links)) " "
|
||
(cl-getf all :uid) ":"
|
||
(cl-getf all :gid) " "
|
||
(if human-size
|
||
(helm-file-human-size (cl-getf all :size))
|
||
(int-to-string (cl-getf all :size))) " "
|
||
(cl-getf all :modif-time)))
|
||
(human-size (helm-file-human-size (cl-getf all :size)))
|
||
(mode-type (cl-getf modes :mode-type))
|
||
(mode-owner (cl-getf modes :user))
|
||
(mode-group (cl-getf modes :group))
|
||
(mode-other (cl-getf modes :other))
|
||
(t (append all modes))))))
|
||
|
||
(defun helm-split-mode-file-attributes (str &optional string)
|
||
"Split mode file attributes STR into a proplist.
|
||
If STRING is non--nil return instead a space separated string."
|
||
(cl-loop with type = (substring str 0 1)
|
||
with cdr = (substring str 1)
|
||
for i across cdr
|
||
for count from 1
|
||
if (<= count 3)
|
||
concat (string i) into user
|
||
if (and (> count 3) (<= count 6))
|
||
concat (string i) into group
|
||
if (and (> count 6) (<= count 9))
|
||
concat (string i) into other
|
||
finally return
|
||
(if string
|
||
(mapconcat 'identity (list type user group other) " ")
|
||
(list :mode-type type :user user :group group :other other))))
|
||
|
||
(defun helm-format-columns-of-files (files)
|
||
"Same as `dired-format-columns-of-files'.
|
||
Inlined here for compatibility."
|
||
(let ((beg (point)))
|
||
(completion--insert-strings files)
|
||
(put-text-property beg (point) 'mouse-face nil)))
|
||
|
||
(defmacro with-helm-display-marked-candidates (buffer-or-name candidates &rest body)
|
||
(declare (indent 0) (debug t))
|
||
(helm-with-gensyms (buffer window winconf)
|
||
`(let* ((,buffer (temp-buffer-window-setup ,buffer-or-name))
|
||
(,winconf helm-last-frame-or-window-configuration)
|
||
(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-inside-p
|
||
helm-reuse-last-window-split-state
|
||
,window)
|
||
(with-current-buffer ,buffer
|
||
(helm-format-columns-of-files ,candidates))
|
||
(unwind-protect
|
||
(with-selected-window
|
||
(setq ,window (temp-buffer-window-show
|
||
,buffer
|
||
'(display-buffer-below-selected
|
||
(window-height . fit-window-to-buffer))))
|
||
(progn ,@body))
|
||
(quit-window 'kill ,window)
|
||
(and ,winconf (set-window-configuration ,winconf))))))
|
||
|
||
;;; Persistent Action Helpers
|
||
;;
|
||
;;
|
||
;; Internal
|
||
(defvar helm-match-line-overlay nil)
|
||
(defvar helm--match-item-overlays nil)
|
||
|
||
(cl-defun helm-highlight-current-line (&optional start end buf face)
|
||
"Highlight current line and all matching items around it.
|
||
|
||
The number of lines around matched line where the matching items are
|
||
highlighted are defined by `helm-highlight-matches-around-point-max-lines'.
|
||
When the variable `helm-highlight-only-all-matches' is non nil only
|
||
the lines containing all matches (in case of multi match) are highlighted.
|
||
|
||
Optional arguments START, END and FACE are only here for debugging purpose."
|
||
(let* ((start (or start (line-beginning-position)))
|
||
(end (or end (1+ (line-end-position))))
|
||
start-match end-match
|
||
(args (list start end buf))
|
||
(case-fold-search (if helm-alive-p
|
||
(helm-set-case-fold-search)
|
||
case-fold-search)))
|
||
;; Highlight the current line.
|
||
(if (not helm-match-line-overlay)
|
||
(setq helm-match-line-overlay (apply 'make-overlay args))
|
||
(apply 'move-overlay helm-match-line-overlay args))
|
||
(overlay-put helm-match-line-overlay
|
||
'face (or face 'helm-selection-line))
|
||
;; Now highlight matches only if we are in helm session, we are
|
||
;; maybe coming from helm-grep-mode or helm-moccur-mode buffers.
|
||
(when helm-alive-p
|
||
(cond (;; These 2 clauses have to be the first otherwise
|
||
;; `helm-highlight-matches-around-point-max-lines' is
|
||
;; compared as a number by other clauses and return an error.
|
||
(eq helm-highlight-matches-around-point-max-lines 'never)
|
||
(cl-return-from helm-highlight-current-line))
|
||
((consp helm-highlight-matches-around-point-max-lines)
|
||
(setq start-match
|
||
(save-excursion
|
||
(forward-line
|
||
(- (car helm-highlight-matches-around-point-max-lines)))
|
||
(point-at-bol))
|
||
end-match
|
||
(save-excursion
|
||
(forward-line
|
||
(cdr helm-highlight-matches-around-point-max-lines))
|
||
(point-at-bol))))
|
||
((or (null helm-highlight-matches-around-point-max-lines)
|
||
(zerop helm-highlight-matches-around-point-max-lines))
|
||
(setq start-match start
|
||
end-match end))
|
||
((< helm-highlight-matches-around-point-max-lines 0)
|
||
(setq start-match
|
||
(save-excursion
|
||
(forward-line
|
||
helm-highlight-matches-around-point-max-lines)
|
||
(point-at-bol))
|
||
end-match start))
|
||
((> helm-highlight-matches-around-point-max-lines 0)
|
||
(setq start-match start
|
||
end-match
|
||
(save-excursion
|
||
(forward-line
|
||
helm-highlight-matches-around-point-max-lines)
|
||
(point-at-bol)))))
|
||
(catch 'empty-line
|
||
(let* ((regex-list (helm-remove-if-match
|
||
"\\`!" (helm-mm-split-pattern
|
||
(if (with-helm-buffer
|
||
;; Needed for highlighting AG matches.
|
||
(assq 'pcre (helm-get-current-source)))
|
||
(helm--translate-pcre-to-elisp helm-input)
|
||
helm-input))))
|
||
(num-regex (length regex-list)))
|
||
(save-excursion
|
||
(goto-char start-match)
|
||
(while (< (point) end-match)
|
||
(let* ((start-line (line-beginning-position))
|
||
(end-line (line-end-position))
|
||
all-matches)
|
||
(dolist (r regex-list)
|
||
(let ((match-list '()))
|
||
(save-excursion
|
||
(goto-char start-line)
|
||
(while (condition-case _err
|
||
(and (not (= start-line end-line))
|
||
(if helm-migemo-mode
|
||
(helm-mm-migemo-forward r end-line t)
|
||
(re-search-forward r end-line t)))
|
||
(invalid-regexp nil))
|
||
(let ((s (match-beginning 0))
|
||
(e (match-end 0)))
|
||
(if (= s e)
|
||
(throw 'empty-line nil)
|
||
(push (cons s e) match-list)))))
|
||
(when match-list
|
||
(push match-list all-matches))))
|
||
(when (and all-matches
|
||
(or (not helm-highlight-only-all-matches)
|
||
(eql (length all-matches) num-regex)))
|
||
(cl-loop for ml in all-matches
|
||
do (cl-loop for (s . e) in ml
|
||
for ov = (make-overlay s e)
|
||
do (progn
|
||
(push ov helm--match-item-overlays)
|
||
(overlay-put ov 'face 'helm-match-item)
|
||
(overlay-put ov 'priority 1))))))
|
||
(forward-line 1))))))
|
||
(recenter)))
|
||
|
||
(defun helm--translate-pcre-to-elisp (regexp)
|
||
"Should translate pcre REGEXP to elisp regexp.
|
||
Assume regexp is a pcre based regexp."
|
||
(with-temp-buffer
|
||
(insert " " regexp " ")
|
||
(goto-char (point-min))
|
||
(save-excursion
|
||
;; match (){}| unquoted
|
||
(helm-awhile (and (re-search-forward "\\([(){}|]\\)" nil t)
|
||
(match-string 1))
|
||
(let ((pos (match-beginning 1)))
|
||
(if (eql (char-before pos) ?\\)
|
||
(delete-region pos (1- pos))
|
||
(replace-match (concat "\\" it) t t nil 1)))))
|
||
;; match \s or \S
|
||
(helm-awhile (and (re-search-forward "\\S\\?\\(\\s\\[sS]\\)[^-]" nil t)
|
||
(match-string 1))
|
||
(replace-match (concat it "-") t t nil 1))
|
||
(buffer-substring (1+ (point-min)) (1- (point-max)))))
|
||
|
||
(defun helm-match-line-cleanup ()
|
||
(when helm-match-line-overlay
|
||
(delete-overlay helm-match-line-overlay)
|
||
(setq helm-match-line-overlay nil))
|
||
(when helm--match-item-overlays
|
||
(mapc 'delete-overlay helm--match-item-overlays)))
|
||
|
||
(defun helm-match-line-cleanup-maybe ()
|
||
(when (helm-empty-buffer-p)
|
||
(helm-match-line-cleanup)))
|
||
|
||
(defun helm-match-line-update ()
|
||
(when helm--match-item-overlays
|
||
(mapc 'delete-overlay helm--match-item-overlays))
|
||
(when helm-match-line-overlay
|
||
(delete-overlay helm-match-line-overlay)
|
||
(helm-highlight-current-line)))
|
||
|
||
(defun helm-persistent-autoresize-hook ()
|
||
(when (and helm-buffers-to-resize-on-pa
|
||
(member helm-buffer helm-buffers-to-resize-on-pa)
|
||
(eq helm-split-window-state 'vertical))
|
||
(set-window-text-height (helm-window) helm-resize-on-pa-text-height)))
|
||
|
||
(defun helm-match-line-cleanup-pulse ()
|
||
(run-with-timer 0.3 nil #'helm-match-line-cleanup))
|
||
|
||
(add-hook 'helm-after-update-hook 'helm-match-line-cleanup-maybe)
|
||
(add-hook 'helm-after-persistent-action-hook 'helm-persistent-autoresize-hook)
|
||
(add-hook 'helm-cleanup-hook 'helm-match-line-cleanup)
|
||
(add-hook 'helm-after-action-hook 'helm-match-line-cleanup-pulse)
|
||
(add-hook 'helm-after-persistent-action-hook 'helm-match-line-update)
|
||
|
||
;;; Popup buffer-name or filename in grep/moccur/imenu-all.
|
||
;;
|
||
(defvar helm--show-help-echo-timer nil)
|
||
|
||
(defun helm-cancel-help-echo-timer ()
|
||
(when helm--show-help-echo-timer
|
||
(cancel-timer helm--show-help-echo-timer)
|
||
(setq helm--show-help-echo-timer nil)))
|
||
|
||
(defun helm-maybe-show-help-echo ()
|
||
(when helm--show-help-echo-timer
|
||
(cancel-timer helm--show-help-echo-timer)
|
||
(setq helm--show-help-echo-timer nil))
|
||
(when (and helm-alive-p
|
||
helm-popup-tip-mode
|
||
(member (assoc-default 'name (helm-get-current-source))
|
||
helm-sources-using-help-echo-popup))
|
||
(setq helm--show-help-echo-timer
|
||
(run-with-timer
|
||
1 nil
|
||
(lambda ()
|
||
(save-selected-window
|
||
(with-helm-window
|
||
(helm-aif (get-text-property (point-at-bol) 'help-echo)
|
||
(popup-tip (concat " " (abbreviate-file-name
|
||
(replace-regexp-in-string "\n.*" "" it)))
|
||
:around nil
|
||
:point (save-excursion
|
||
(end-of-visual-line) (point)))))))))))
|
||
|
||
;;;###autoload
|
||
(define-minor-mode helm-popup-tip-mode
|
||
"Show help-echo informations in a popup tip at end of line."
|
||
:global t
|
||
(require 'popup)
|
||
(if helm-popup-tip-mode
|
||
(progn
|
||
(add-hook 'helm-move-selection-after-hook 'helm-maybe-show-help-echo)
|
||
(add-hook 'helm-cleanup-hook 'helm-cancel-help-echo-timer))
|
||
(remove-hook 'helm-move-selection-after-hook 'helm-maybe-show-help-echo)
|
||
(remove-hook 'helm-cleanup-hook 'helm-cancel-help-echo-timer)))
|
||
|
||
(defun helm-open-file-with-default-tool (file)
|
||
"Open FILE with the default tool on this platform."
|
||
(let (process-connection-type)
|
||
(if (eq system-type 'windows-nt)
|
||
(helm-w32-shell-execute-open-file file)
|
||
(start-process "helm-open-file-with-default-tool"
|
||
nil
|
||
(cond ((eq system-type 'gnu/linux)
|
||
"xdg-open")
|
||
((or (eq system-type 'darwin) ;; Mac OS X
|
||
(eq system-type 'macos)) ;; Mac OS 9
|
||
"open"))
|
||
file))))
|
||
|
||
(defun helm-open-dired (file)
|
||
"Open a dired buffer in FILE's directory.
|
||
If FILE is a directory, open this directory."
|
||
(require 'dired)
|
||
(if (file-directory-p file)
|
||
(dired file)
|
||
(dired (file-name-directory file))
|
||
(dired-goto-file file)))
|
||
|
||
(defun helm-find-file-as-root (candidate)
|
||
(let* ((buf (helm-basename candidate))
|
||
(host (file-remote-p candidate 'host))
|
||
(remote-path (format "/%s:%s:%s"
|
||
helm-su-or-sudo
|
||
(or host "")
|
||
(expand-file-name
|
||
(if host
|
||
(file-remote-p candidate 'localname)
|
||
candidate))))
|
||
non-essential)
|
||
(if (buffer-live-p (get-buffer buf))
|
||
(progn
|
||
(set-buffer buf)
|
||
(find-alternate-file remote-path))
|
||
(find-file remote-path))))
|
||
|
||
(defun helm-find-many-files (_ignore)
|
||
"Simple action that run `find-file' on marked candidates.
|
||
Run `helm-find-many-files-after-hook' at end."
|
||
(let ((helm--reading-passwd-or-string t))
|
||
(mapc 'find-file (helm-marked-candidates))
|
||
(helm-log-run-hook 'helm-find-many-files-after-hook)))
|
||
|
||
(defun helm-read-repeat-string (prompt &optional count)
|
||
"Prompt as many time PROMPT is not empty.
|
||
If COUNT is non--nil add a number after each prompt."
|
||
(cl-loop with elm
|
||
while (not (string= elm ""))
|
||
for n from 1
|
||
do (when count
|
||
(setq prompt (concat prompt (int-to-string n) ": ")))
|
||
collect (setq elm (helm-read-string prompt)) into lis
|
||
finally return (remove "" lis)))
|
||
|
||
(defun helm-html-bookmarks-to-alist (file url-regexp bmk-regexp)
|
||
"Parse HTML bookmark FILE and return an alist with (title . url) as elements."
|
||
(let (bookmarks-alist url title)
|
||
(with-temp-buffer
|
||
(insert-file-contents file)
|
||
(goto-char (point-min))
|
||
(while (re-search-forward "href=\\|^ *<DT><A HREF=" nil t)
|
||
(forward-line 0)
|
||
(when (re-search-forward url-regexp nil t)
|
||
(setq url (match-string 0)))
|
||
(when (re-search-forward bmk-regexp nil t)
|
||
(setq title (url-unhex-string
|
||
(funcall helm-html-decode-entities-function
|
||
(match-string 1)))))
|
||
(push (cons title url) bookmarks-alist)
|
||
(forward-line)))
|
||
(nreverse bookmarks-alist)))
|
||
|
||
(defun helm-html-entity-to-string (entity)
|
||
"Replace an HTML ENTITY with its string value.
|
||
When unable to decode ENTITY returns nil."
|
||
(helm-aif (assoc entity helm-html-entities-alist)
|
||
(string (cdr it))
|
||
(save-match-data
|
||
(when (string-match "[0-9]+" entity)
|
||
(string (string-to-number (match-string 0 entity)))))))
|
||
|
||
(defun helm-html-decode-entities-string (str)
|
||
"Decode entities in the string STR."
|
||
(save-match-data
|
||
(with-temp-buffer
|
||
(insert str)
|
||
(goto-char (point-min))
|
||
(while (re-search-forward "&#?\\([^;]*\\);" nil t)
|
||
(helm-aif (helm-html-entity-to-string (match-string 0))
|
||
(replace-match it)))
|
||
(buffer-string))))
|
||
|
||
(provide 'helm-utils)
|
||
|
||
;;; helm-utils.el ends here
|