2016-04-22 09:59:30 +02:00
|
|
|
;;; helm-ag.el --- the silver searcher with helm interface -*- lexical-binding: t; -*-
|
|
|
|
|
|
|
|
;; Copyright (C) 2016 by Syohei YOSHIDA
|
|
|
|
|
|
|
|
;; Author: Syohei YOSHIDA <syohex@gmail.com>
|
|
|
|
;; URL: https://github.com/syohex/emacs-helm-ag
|
2016-09-26 19:08:35 +02:00
|
|
|
;; Package-Version: 20160923.2128
|
2016-06-29 09:21:54 +02:00
|
|
|
;; Version: 0.56
|
2016-09-22 08:19:51 +00:00
|
|
|
;; Package-Requires: ((emacs "24.4") (helm "2.0"))
|
2016-04-22 09:59:30 +02:00
|
|
|
|
|
|
|
;; 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/>.
|
|
|
|
|
|
|
|
;;; Commentary:
|
|
|
|
|
|
|
|
;; helm-ag provides interfaces of the silver searcher(Other search programs can be used
|
|
|
|
;; such as the platinum searcher, ack). And helm-ag provides wgrep like features which
|
|
|
|
;; users can edit from searched result.
|
|
|
|
|
|
|
|
;;; Code:
|
|
|
|
|
|
|
|
(eval-when-compile
|
|
|
|
(require 'grep)
|
|
|
|
(defvar helm-help-message))
|
|
|
|
|
|
|
|
(require 'cl-lib)
|
|
|
|
(require 'helm)
|
|
|
|
(require 'helm-grep)
|
|
|
|
(require 'helm-utils)
|
|
|
|
(require 'compile)
|
2016-09-22 08:19:51 +00:00
|
|
|
(require 'subr-x)
|
2016-04-22 09:59:30 +02:00
|
|
|
|
|
|
|
(declare-function helm-read-file-name "helm-mode")
|
|
|
|
(declare-function helm-grep-get-file-extensions "helm-grep")
|
|
|
|
(declare-function helm-help "helm-help")
|
|
|
|
|
|
|
|
(defgroup helm-ag nil
|
|
|
|
"the silver searcher with helm interface"
|
|
|
|
:group 'helm)
|
|
|
|
|
|
|
|
(defsubst helm-ag--windows-p ()
|
|
|
|
(memq system-type '(ms-dos windows-nt)))
|
|
|
|
|
|
|
|
(defcustom helm-ag-base-command
|
|
|
|
(if (helm-ag--windows-p)
|
|
|
|
"ag --vimgrep"
|
|
|
|
"ag --nocolor --nogroup")
|
|
|
|
"Base command of `ag'"
|
|
|
|
:type 'string)
|
|
|
|
|
|
|
|
(defcustom helm-ag-command-option nil
|
|
|
|
"Command line option of `ag'. This is appended after `helm-ag-base-command'"
|
|
|
|
:type 'string)
|
|
|
|
|
|
|
|
(defcustom helm-ag-insert-at-point nil
|
|
|
|
"Insert thing at point as search pattern.
|
|
|
|
You can set value same as `thing-at-point'"
|
|
|
|
:type 'symbol)
|
|
|
|
|
|
|
|
(defcustom helm-ag-ignore-patterns nil
|
|
|
|
"Ignore patterns for `ag'. This parameters are specified as --ignore"
|
|
|
|
:type '(repeat string))
|
|
|
|
|
|
|
|
(defcustom helm-ag-use-grep-ignore-list nil
|
|
|
|
"Use `grep-find-ignored-files' and `grep-find-ignored-directories' as ignore pattern.
|
|
|
|
They are specified to `--ignore' options."
|
|
|
|
:type 'boolean)
|
|
|
|
|
|
|
|
(defcustom helm-ag-always-set-extra-option nil
|
|
|
|
"Always set `ag' options of `helm-do-ag'."
|
|
|
|
:type 'boolean)
|
|
|
|
|
|
|
|
(defcustom helm-ag-fuzzy-match nil
|
|
|
|
"Enable fuzzy match"
|
|
|
|
:type 'boolean)
|
|
|
|
|
|
|
|
(defcustom helm-ag-edit-save t
|
|
|
|
"Save buffers you edit at completed."
|
|
|
|
:type 'boolean)
|
|
|
|
|
|
|
|
(defcustom helm-ag-use-emacs-lisp-regexp nil
|
|
|
|
"[Experimental] Use Emacs Lisp regexp instead of PCRE."
|
|
|
|
:type 'boolean)
|
|
|
|
|
|
|
|
(defcustom helm-ag-use-agignore nil
|
|
|
|
"Use .agignore where is at project root if it exists."
|
|
|
|
:type 'boolean)
|
|
|
|
|
|
|
|
(defcustom helm-ag-use-temp-buffer nil
|
|
|
|
"Use temporary buffer for persistent action."
|
|
|
|
:type 'boolean)
|
|
|
|
|
2016-09-26 19:08:35 +02:00
|
|
|
(defcustom helm-ag-ignore-buffer-patterns nil
|
|
|
|
"Use temporary buffer for persistent action."
|
|
|
|
:type '(repeat regexp))
|
|
|
|
|
2016-04-22 09:59:30 +02:00
|
|
|
(defcustom helm-ag-show-status-function 'helm-ag-show-status-default-mode-line
|
|
|
|
"Function called after that `ag' process is finished after `helm-do-ag'.
|
|
|
|
Default behaviour shows finish and result in mode-line."
|
|
|
|
:type 'function)
|
|
|
|
|
|
|
|
(defface helm-ag-edit-deleted-line
|
|
|
|
'((t (:inherit font-lock-comment-face :strike-through t)))
|
|
|
|
"Face of deleted line in edit mode.")
|
|
|
|
|
|
|
|
(defvar helm-ag--command-history '())
|
|
|
|
(defvar helm-ag--context-stack nil)
|
|
|
|
(defvar helm-ag--default-directory nil)
|
|
|
|
(defvar helm-ag--last-default-directory nil)
|
|
|
|
(defvar helm-ag--last-query nil)
|
|
|
|
(defvar helm-ag--last-command nil)
|
|
|
|
(defvar helm-ag--elisp-regexp-query nil)
|
|
|
|
(defvar helm-ag--valid-regexp-for-emacs nil)
|
|
|
|
(defvar helm-ag--extra-options nil)
|
|
|
|
(defvar helm-ag--extra-options-history nil)
|
|
|
|
(defvar helm-ag--original-window nil)
|
|
|
|
(defvar helm-ag--search-this-file-p nil)
|
|
|
|
(defvar helm-ag--default-target nil)
|
|
|
|
(defvar helm-ag--buffer-search nil)
|
|
|
|
(defvar helm-ag--command-feature nil)
|
|
|
|
(defvar helm-ag--ignore-case nil)
|
|
|
|
(defvar helm-do-ag--extensions nil)
|
|
|
|
(defvar helm-do-ag--commands nil)
|
|
|
|
|
|
|
|
(defun helm-ag--ignore-case-p (cmds input)
|
|
|
|
(cl-loop for cmd in cmds
|
|
|
|
when (member cmd '("-i" "--ignore-case"))
|
|
|
|
return t
|
|
|
|
|
|
|
|
when (member cmd '("-s" "--case-sensitive"))
|
|
|
|
return nil
|
|
|
|
|
|
|
|
finally
|
|
|
|
return (let ((case-fold-search nil))
|
|
|
|
(not (string-match-p "[A-Z]" input)))))
|
|
|
|
|
|
|
|
(defun helm-ag--save-current-context ()
|
|
|
|
(let ((curpoint (with-helm-current-buffer
|
|
|
|
(point))))
|
|
|
|
(helm-aif (buffer-file-name helm-current-buffer)
|
|
|
|
(push (list :file it :point curpoint) helm-ag--context-stack)
|
|
|
|
(push (list :buffer helm-current-buffer :point curpoint) helm-ag--context-stack))))
|
|
|
|
|
|
|
|
(defsubst helm-ag--insert-thing-at-point (thing)
|
|
|
|
(helm-aif (thing-at-point thing)
|
|
|
|
(substring-no-properties it)
|
|
|
|
""))
|
|
|
|
|
|
|
|
(defun helm-ag--searched-word ()
|
|
|
|
(if helm-ag-insert-at-point
|
|
|
|
(helm-ag--insert-thing-at-point helm-ag-insert-at-point)
|
|
|
|
""))
|
|
|
|
|
|
|
|
(defun helm-ag--construct-ignore-option (pattern)
|
|
|
|
(concat "--ignore=" pattern))
|
|
|
|
|
|
|
|
(defun helm-ag--grep-ignore-list-to-options ()
|
|
|
|
(require 'grep)
|
|
|
|
(cl-loop for ignore in (append grep-find-ignored-files
|
|
|
|
grep-find-ignored-directories)
|
|
|
|
collect (helm-ag--construct-ignore-option ignore)))
|
|
|
|
|
|
|
|
(defun helm-ag--parse-options-and-query (input)
|
|
|
|
(with-temp-buffer
|
|
|
|
(insert input)
|
|
|
|
(let (end options)
|
|
|
|
(goto-char (point-min))
|
|
|
|
(when (re-search-forward "\\s-*--\\s-+" nil t)
|
|
|
|
(setq end (match-end 0)))
|
|
|
|
(goto-char (point-min))
|
|
|
|
(while (re-search-forward "\\(?:^\\|\\s-+\\)\\(-\\S-+\\)\\(?:\\s-+\\|$\\)" end t)
|
|
|
|
(push (match-string-no-properties 1) options)
|
|
|
|
(when end
|
|
|
|
(cl-decf end (- (match-end 0) (match-beginning 0))))
|
|
|
|
(replace-match ""))
|
|
|
|
(cons options (buffer-string)))))
|
|
|
|
|
|
|
|
(defun helm-ag--parse-query (input)
|
|
|
|
(let* ((parsed (helm-ag--parse-options-and-query input))
|
|
|
|
(options (car parsed))
|
|
|
|
(query (cdr parsed)))
|
|
|
|
(when helm-ag-use-emacs-lisp-regexp
|
|
|
|
(setq query (helm-ag--elisp-regexp-to-pcre query)))
|
|
|
|
(setq helm-ag--last-query query
|
|
|
|
helm-ag--elisp-regexp-query (helm-ag--pcre-to-elisp-regexp query))
|
|
|
|
(setq helm-ag--valid-regexp-for-emacs
|
|
|
|
(helm-ag--validate-regexp helm-ag--elisp-regexp-query))
|
|
|
|
(if (not options)
|
|
|
|
(list query)
|
|
|
|
(nconc (nreverse options) (list query)))))
|
|
|
|
|
2016-09-26 19:08:35 +02:00
|
|
|
(defsubst helm-ag--search-buffer-p (bufname)
|
|
|
|
(cl-loop for regexp in helm-ag-ignore-buffer-patterns
|
|
|
|
never (string-match-p regexp bufname)))
|
|
|
|
|
|
|
|
(defun helm-ag--file-visited-buffers ()
|
|
|
|
(let ((bufs (cl-loop for buf in (buffer-list)
|
|
|
|
when (buffer-file-name buf)
|
|
|
|
collect it)))
|
|
|
|
(if (not helm-ag-ignore-buffer-patterns)
|
|
|
|
bufs
|
|
|
|
(cl-loop for buf in bufs
|
|
|
|
when (helm-ag--search-buffer-p buf)
|
|
|
|
collect buf))))
|
2016-04-22 09:59:30 +02:00
|
|
|
|
|
|
|
(defun helm-ag--construct-targets (targets)
|
|
|
|
(let ((default-directory helm-ag--default-directory))
|
|
|
|
(cl-loop for target in targets
|
|
|
|
collect (file-relative-name target))))
|
|
|
|
|
|
|
|
(defun helm-ag--root-agignore ()
|
|
|
|
(let ((root (helm-ag--project-root)))
|
|
|
|
(when root
|
|
|
|
(let ((default-directory root))
|
|
|
|
(when (file-exists-p ".agignore")
|
|
|
|
(expand-file-name (concat default-directory ".agignore")))))))
|
|
|
|
|
|
|
|
(defun helm-ag--construct-command (this-file)
|
|
|
|
(let* ((commands (split-string helm-ag-base-command nil t))
|
|
|
|
(command (car commands))
|
|
|
|
(args (cdr commands)))
|
|
|
|
(when helm-ag-command-option
|
|
|
|
(let ((ag-options (split-string helm-ag-command-option nil t)))
|
|
|
|
(setq args (append args ag-options))))
|
|
|
|
(when helm-ag-use-agignore
|
|
|
|
(helm-aif (helm-ag--root-agignore)
|
|
|
|
(setq args (append args (list "-p" it)))))
|
|
|
|
(when helm-ag-ignore-patterns
|
|
|
|
(setq args (append args (mapcar 'helm-ag--construct-ignore-option
|
|
|
|
helm-ag-ignore-patterns))))
|
|
|
|
(when helm-ag-use-grep-ignore-list
|
|
|
|
(setq args (append args (helm-ag--grep-ignore-list-to-options))))
|
|
|
|
(setq args (append args (helm-ag--parse-query helm-ag--last-query)))
|
|
|
|
(when this-file
|
|
|
|
(setq args (append args (list this-file))))
|
|
|
|
(when helm-ag--buffer-search
|
|
|
|
(setq args (append args (helm-ag--file-visited-buffers))))
|
|
|
|
(when helm-ag--default-target
|
|
|
|
(setq args (append args (helm-ag--construct-targets helm-ag--default-target))))
|
|
|
|
(cons command args)))
|
|
|
|
|
|
|
|
(defun helm-ag--remove-carrige-returns ()
|
|
|
|
(when (helm-ag--windows-p)
|
|
|
|
(save-excursion
|
|
|
|
(goto-char (point-min))
|
|
|
|
(while (re-search-forward "\xd" nil t)
|
|
|
|
(replace-match "")))))
|
|
|
|
|
2016-08-18 22:01:20 +02:00
|
|
|
(defun helm-ag--abbreviate-file-name ()
|
|
|
|
(unless (helm-ag--windows-p)
|
|
|
|
(save-excursion
|
|
|
|
(goto-char (point-min))
|
|
|
|
(forward-line 1)
|
|
|
|
(while (re-search-forward "^\\([^:]+\\)" nil t)
|
|
|
|
(replace-match (abbreviate-file-name (match-string-no-properties 1)))))))
|
|
|
|
|
2016-04-22 09:59:30 +02:00
|
|
|
(defun helm-ag--init ()
|
|
|
|
(let ((buf-coding buffer-file-coding-system))
|
|
|
|
(helm-attrset 'recenter t)
|
|
|
|
(with-current-buffer (helm-candidate-buffer 'global)
|
|
|
|
(let* ((default-directory (or helm-ag--default-directory
|
|
|
|
default-directory))
|
|
|
|
(cmds (helm-ag--construct-command (helm-attr 'search-this-file)))
|
|
|
|
(coding-system-for-read buf-coding)
|
|
|
|
(coding-system-for-write buf-coding))
|
|
|
|
(setq helm-ag--ignore-case (helm-ag--ignore-case-p cmds helm-ag--last-query)
|
|
|
|
helm-ag--last-command cmds)
|
|
|
|
(let ((ret (apply #'process-file (car cmds) nil t nil (cdr cmds))))
|
|
|
|
(if (zerop (length (buffer-string)))
|
|
|
|
(error "No ag output: '%s'" helm-ag--last-query)
|
|
|
|
(unless (zerop ret)
|
|
|
|
(unless (executable-find (car cmds))
|
|
|
|
(error "'ag' is not installed."))
|
|
|
|
(error "Failed: '%s'" helm-ag--last-query))))
|
2016-08-18 22:01:20 +02:00
|
|
|
(when helm-ag--buffer-search
|
|
|
|
(helm-ag--abbreviate-file-name))
|
2016-04-22 09:59:30 +02:00
|
|
|
(helm-ag--remove-carrige-returns)
|
|
|
|
(helm-ag--save-current-context)))))
|
|
|
|
|
|
|
|
(add-to-list 'debug-ignored-errors "^No ag output: ")
|
|
|
|
|
|
|
|
(defun helm-ag--search-only-one-file-p ()
|
|
|
|
(when (and helm-ag--default-target (= (length helm-ag--default-target) 1))
|
|
|
|
(let ((target (car helm-ag--default-target)))
|
|
|
|
(unless (file-directory-p target)
|
|
|
|
target))))
|
|
|
|
|
|
|
|
(defun helm-ag--find-file-action (candidate find-func this-file &optional persistent)
|
|
|
|
(when helm-ag--command-feature
|
|
|
|
;; 'pt' always show filename if matched file is only one.
|
|
|
|
(setq this-file nil))
|
|
|
|
(let* ((file-line (helm-grep-split-line candidate))
|
2016-09-22 08:19:51 +00:00
|
|
|
(filename (or this-file (cl-first file-line) candidate))
|
2016-04-22 09:59:30 +02:00
|
|
|
(line (if this-file
|
|
|
|
(cl-first (split-string candidate ":"))
|
|
|
|
(cl-second file-line)))
|
|
|
|
(default-directory (or helm-ag--default-directory
|
|
|
|
helm-ag--last-default-directory
|
|
|
|
default-directory)))
|
|
|
|
(unless persistent
|
|
|
|
(setq helm-ag--last-default-directory default-directory))
|
|
|
|
(funcall find-func filename)
|
|
|
|
(goto-char (point-min))
|
|
|
|
(when line
|
2016-04-26 13:40:21 +02:00
|
|
|
(forward-line (1- (string-to-number line))))
|
|
|
|
(ignore-errors
|
|
|
|
(and (re-search-forward helm-ag--last-query (line-end-position) t)
|
|
|
|
(goto-char (match-beginning 0))))))
|
2016-04-22 09:59:30 +02:00
|
|
|
|
|
|
|
(defun helm-ag--open-file-with-temp-buffer (filename)
|
|
|
|
(switch-to-buffer (get-buffer-create " *helm-ag persistent*"))
|
|
|
|
(fundamental-mode)
|
|
|
|
(erase-buffer)
|
|
|
|
(insert-file-contents filename)
|
|
|
|
(let ((buffer-file-name filename))
|
|
|
|
(set-auto-mode)
|
|
|
|
(font-lock-fontify-region (point-min) (point-max))))
|
|
|
|
|
|
|
|
(defsubst helm-ag--vimgrep-option ()
|
|
|
|
(member "--vimgrep" helm-ag--last-command))
|
|
|
|
|
|
|
|
(defun helm-ag--search-this-file-p ()
|
|
|
|
(unless (helm-ag--vimgrep-option)
|
|
|
|
(if (eq (helm-get-current-source) 'helm-source-do-ag)
|
|
|
|
(helm-ag--search-only-one-file-p)
|
|
|
|
(helm-attr 'search-this-file))))
|
|
|
|
|
|
|
|
(defun helm-ag--persistent-action (candidate)
|
|
|
|
(let ((find-func (if helm-ag-use-temp-buffer
|
|
|
|
#'helm-ag--open-file-with-temp-buffer
|
|
|
|
#'find-file)))
|
|
|
|
(helm-ag--find-file-action candidate find-func (helm-ag--search-this-file-p) t)
|
|
|
|
(helm-highlight-current-line)))
|
|
|
|
|
|
|
|
(defun helm-ag--validate-regexp (regexp)
|
|
|
|
(condition-case nil
|
|
|
|
(progn
|
|
|
|
(string-match-p regexp "")
|
|
|
|
t)
|
|
|
|
(invalid-regexp nil)))
|
|
|
|
|
|
|
|
(defun helm-ag--pcre-to-elisp-regexp (pcre)
|
|
|
|
;; This is very simple conversion
|
|
|
|
(with-temp-buffer
|
|
|
|
(insert pcre)
|
|
|
|
(goto-char (point-min))
|
|
|
|
;; convert (, ), {, }, |
|
|
|
|
(while (re-search-forward "[(){}|]" nil t)
|
|
|
|
(backward-char 1)
|
|
|
|
(cond ((looking-back "\\\\\\\\" nil))
|
|
|
|
((looking-back "\\\\" nil)
|
|
|
|
(delete-char -1))
|
|
|
|
(t
|
|
|
|
(insert "\\")))
|
|
|
|
(forward-char 1))
|
|
|
|
;; convert \s and \S -> \s- \S-
|
|
|
|
(goto-char (point-min))
|
|
|
|
(while (re-search-forward "\\(\\\\s\\)" nil t)
|
|
|
|
(unless (looking-back "\\\\\\\\s" nil)
|
|
|
|
(insert "-")))
|
|
|
|
(buffer-string)))
|
|
|
|
|
|
|
|
(defun helm-ag--elisp-regexp-to-pcre (regexp)
|
|
|
|
(with-temp-buffer
|
|
|
|
(insert regexp)
|
|
|
|
(goto-char (point-min))
|
|
|
|
(while (re-search-forward "[(){}|]" nil t)
|
|
|
|
(backward-char 1)
|
|
|
|
(cond ((looking-back "\\\\\\\\" nil))
|
|
|
|
((looking-back "\\\\" nil)
|
|
|
|
(delete-char -1))
|
|
|
|
(t
|
|
|
|
(insert "\\")))
|
|
|
|
(forward-char 1))
|
|
|
|
(buffer-string)))
|
|
|
|
|
|
|
|
(defun helm-ag--highlight-candidate (candidate)
|
|
|
|
(let ((limit (1- (length candidate)))
|
|
|
|
(last-pos 0)
|
|
|
|
(case-fold-search helm-ag--ignore-case))
|
|
|
|
(when helm-ag--valid-regexp-for-emacs
|
|
|
|
(while (and (< last-pos limit)
|
|
|
|
(string-match helm-ag--elisp-regexp-query candidate last-pos))
|
|
|
|
(let ((start (match-beginning 0))
|
|
|
|
(end (match-end 0)))
|
|
|
|
(if (= start end)
|
|
|
|
(cl-incf last-pos)
|
|
|
|
(put-text-property start end 'face 'helm-match candidate)
|
|
|
|
(setq last-pos (1+ (match-end 0)))))))
|
|
|
|
candidate))
|
|
|
|
|
|
|
|
(defun helm-ag--candidate-transform-for-this-file (candidate)
|
|
|
|
(when (string-match "\\`\\([^:]+\\):\\(.*\\)" candidate)
|
|
|
|
(format "%s:%s"
|
|
|
|
(propertize (match-string 1 candidate) 'face 'helm-grep-lineno)
|
|
|
|
(helm-ag--highlight-candidate (match-string 2 candidate)))))
|
|
|
|
|
|
|
|
(defun helm-ag--candidate-transform-for-files (candidate)
|
|
|
|
(helm-aif (helm-grep-split-line candidate)
|
|
|
|
(format "%s:%s:%s"
|
|
|
|
(propertize (cl-first it) 'face 'helm-moccur-buffer)
|
|
|
|
(propertize (cl-second it) 'face 'helm-grep-lineno)
|
|
|
|
(helm-ag--highlight-candidate (cl-third it)))))
|
|
|
|
|
|
|
|
(defun helm-ag--candidate-transformer (candidate)
|
|
|
|
(or (if (helm-attr 'search-this-file)
|
|
|
|
(helm-ag--candidate-transform-for-this-file candidate)
|
|
|
|
(helm-ag--candidate-transform-for-files candidate))
|
|
|
|
candidate))
|
|
|
|
|
|
|
|
(defun helm-ag--action-find-file (candidate)
|
|
|
|
(helm-ag--find-file-action candidate 'find-file (helm-ag--search-this-file-p)))
|
|
|
|
|
|
|
|
(defun helm-ag--action-find-file-other-window (candidate)
|
|
|
|
(helm-ag--find-file-action candidate 'find-file-other-window (helm-ag--search-this-file-p)))
|
|
|
|
|
|
|
|
(defvar helm-ag--actions
|
|
|
|
(helm-make-actions
|
|
|
|
"Open file" #'helm-ag--action-find-file
|
|
|
|
"Open file other window" #'helm-ag--action-find-file-other-window
|
2016-06-29 09:21:54 +02:00
|
|
|
"Save results in buffer" #'helm-ag--action-save-buffer
|
|
|
|
"Edit search results" #'helm-ag--edit))
|
|
|
|
|
|
|
|
(defvar helm-ag-map
|
|
|
|
(let ((map (make-sparse-keymap)))
|
|
|
|
(set-keymap-parent map helm-map)
|
|
|
|
(define-key map (kbd "C-c o") 'helm-ag--run-other-window-action)
|
|
|
|
(define-key map (kbd "C-l") 'helm-ag--up-one-level)
|
|
|
|
(define-key map (kbd "C-c C-e") 'helm-ag-edit)
|
|
|
|
(define-key map (kbd "C-x C-s") 'helm-ag--run-save-buffer)
|
|
|
|
(define-key map (kbd "C-c ?") 'helm-ag-help)
|
|
|
|
(define-key map (kbd "C-c >") 'helm-ag--next-file)
|
|
|
|
(define-key map (kbd "<right>") 'helm-ag--next-file)
|
|
|
|
(define-key map (kbd "C-c <") 'helm-ag--previous-file)
|
|
|
|
(define-key map (kbd "<left>") 'helm-ag--previous-file)
|
|
|
|
map)
|
|
|
|
"Keymap for `helm-ag'.")
|
2016-04-22 09:59:30 +02:00
|
|
|
|
|
|
|
(defvar helm-ag-source
|
|
|
|
(helm-build-in-buffer-source "The Silver Searcher"
|
|
|
|
:init 'helm-ag--init
|
|
|
|
:real-to-display 'helm-ag--candidate-transformer
|
|
|
|
:persistent-action 'helm-ag--persistent-action
|
|
|
|
:fuzzy-match helm-ag-fuzzy-match
|
|
|
|
:action helm-ag--actions
|
|
|
|
:candidate-number-limit 9999
|
|
|
|
:follow (and helm-follow-mode-persistent 1)))
|
|
|
|
|
|
|
|
;;;###autoload
|
|
|
|
(defun helm-ag-pop-stack ()
|
|
|
|
(interactive)
|
|
|
|
(let ((context (pop helm-ag--context-stack)))
|
|
|
|
(unless context
|
|
|
|
(error "Context stack is empty !"))
|
|
|
|
(helm-aif (plist-get context :file)
|
|
|
|
(find-file it)
|
|
|
|
(let ((buf (plist-get context :buffer)))
|
|
|
|
(if (buffer-live-p buf)
|
|
|
|
(switch-to-buffer buf)
|
|
|
|
(error "The buffer is already killed."))))
|
|
|
|
(goto-char (plist-get context :point))))
|
|
|
|
|
|
|
|
;;;###autoload
|
|
|
|
(defun helm-ag-clear-stack ()
|
|
|
|
(interactive)
|
|
|
|
(setq helm-ag--context-stack nil))
|
|
|
|
|
|
|
|
(defsubst helm-ag--marked-input ()
|
|
|
|
(when (use-region-p)
|
2016-06-29 09:21:54 +02:00
|
|
|
(prog1 (buffer-substring-no-properties (region-beginning) (region-end))
|
|
|
|
(deactivate-mark))))
|
2016-04-22 09:59:30 +02:00
|
|
|
|
|
|
|
(defun helm-ag--query ()
|
|
|
|
(let* ((searched-word (helm-ag--searched-word))
|
|
|
|
(marked-word (helm-ag--marked-input))
|
|
|
|
(query (read-string "Pattern: " (or marked-word searched-word) 'helm-ag--command-history)))
|
2016-09-22 08:19:51 +00:00
|
|
|
(when (string-empty-p query)
|
2016-04-22 09:59:30 +02:00
|
|
|
(error "Input is empty!!"))
|
|
|
|
(setq helm-ag--last-query query)))
|
|
|
|
|
|
|
|
(defsubst helm-ag--init-state ()
|
|
|
|
(setq helm-ag--original-window (selected-window)
|
|
|
|
helm-ag--last-default-directory nil))
|
|
|
|
|
|
|
|
(defun helm-ag--get-default-directory ()
|
|
|
|
(let ((prefix-val (and current-prefix-arg (abs (prefix-numeric-value current-prefix-arg)))))
|
|
|
|
(cond ((not prefix-val) default-directory)
|
|
|
|
((= prefix-val 4)
|
|
|
|
(file-name-as-directory
|
|
|
|
(read-directory-name "Search directory: " nil nil t)))
|
|
|
|
((= prefix-val 16)
|
|
|
|
(let ((dirs (list (read-directory-name "Search directory: " nil nil t))))
|
|
|
|
(while (y-or-n-p "More directories ?")
|
|
|
|
(push (read-directory-name "Search directory: " nil nil t) dirs))
|
|
|
|
(reverse dirs))))))
|
|
|
|
|
|
|
|
(defsubst helm-ag--helm-header (dir)
|
|
|
|
(if helm-ag--buffer-search
|
|
|
|
"Search Buffers"
|
|
|
|
(concat "Search at " (abbreviate-file-name dir))))
|
|
|
|
|
|
|
|
(defun helm-ag--run-other-window-action ()
|
|
|
|
(interactive)
|
|
|
|
(with-helm-alive-p
|
|
|
|
(helm-exit-and-execute-action #'helm-ag--action-find-file-other-window)))
|
|
|
|
|
|
|
|
(defun helm-ag--exit-from-edit-mode ()
|
|
|
|
(when (window-live-p helm-ag--original-window)
|
|
|
|
(select-window helm-ag--original-window))
|
|
|
|
(kill-buffer (get-buffer "*helm-ag-edit*")))
|
|
|
|
|
|
|
|
(defun helm-ag--match-line-regexp ()
|
|
|
|
;; $1: file name
|
|
|
|
;; $2: line
|
|
|
|
;; $3: match body
|
|
|
|
;; $4: file attributes part(filename, line, column)
|
|
|
|
(cond ((helm-ag--vimgrep-option)
|
|
|
|
"^\\(?4:\\(?1:[^:]+\\):\\(?2:[1-9][0-9]*\\):[^:]+:\\)\\(?3:.*\\)$")
|
|
|
|
(helm-ag--search-this-file-p
|
|
|
|
"^\\(?4:\\(?2:[1-9][0-9]*\\)[:-]\\)\\(?3:.*\\)$")
|
|
|
|
(t
|
|
|
|
"^\\(?4:\\(?1:[^:]+\\):\\(?2:[1-9][0-9]*\\)[:-]\\)\\(?3:.*\\)$")))
|
|
|
|
|
|
|
|
(defun helm-ag--edit-commit ()
|
|
|
|
(interactive)
|
|
|
|
(goto-char (point-min))
|
|
|
|
(let ((read-only-files 0)
|
|
|
|
(saved-buffers nil)
|
|
|
|
(regexp (helm-ag--match-line-regexp))
|
|
|
|
(default-directory helm-ag--default-directory)
|
2016-06-29 09:21:54 +02:00
|
|
|
(line-deletes (make-hash-table :test #'equal))
|
|
|
|
(kept-buffers (buffer-list))
|
|
|
|
open-buffers)
|
2016-04-22 09:59:30 +02:00
|
|
|
(while (re-search-forward regexp nil t)
|
|
|
|
(let* ((file (or (match-string-no-properties 1) helm-ag--search-this-file-p))
|
|
|
|
(line (string-to-number (match-string-no-properties 2)))
|
|
|
|
(body (match-string-no-properties 3))
|
|
|
|
(ovs (overlays-at (line-beginning-position))))
|
|
|
|
(with-current-buffer (find-file-noselect file)
|
2016-06-29 09:21:54 +02:00
|
|
|
(cl-pushnew (current-buffer) open-buffers)
|
2016-04-22 09:59:30 +02:00
|
|
|
(if buffer-read-only
|
|
|
|
(cl-incf read-only-files)
|
|
|
|
(goto-char (point-min))
|
|
|
|
(let ((deleted-lines (gethash file line-deletes 0))
|
|
|
|
(deleted (and ovs (overlay-get (car ovs) 'helm-ag-deleted))))
|
|
|
|
(forward-line (- line 1 deleted-lines))
|
|
|
|
(delete-region (line-beginning-position) (line-end-position))
|
|
|
|
(if (not deleted)
|
|
|
|
(insert body)
|
|
|
|
(let ((beg (point)))
|
|
|
|
(forward-line 1)
|
|
|
|
(delete-region beg (point))
|
|
|
|
(puthash file (1+ deleted-lines) line-deletes)))
|
|
|
|
(cl-pushnew (current-buffer) saved-buffers))))))
|
|
|
|
(when helm-ag-edit-save
|
|
|
|
(dolist (buf saved-buffers)
|
|
|
|
(with-current-buffer buf
|
|
|
|
(save-buffer))))
|
2016-06-29 09:21:54 +02:00
|
|
|
(dolist (buf open-buffers)
|
|
|
|
(unless (memq buf kept-buffers)
|
|
|
|
(kill-buffer buf)))
|
2016-04-22 09:59:30 +02:00
|
|
|
(helm-ag--exit-from-edit-mode)
|
|
|
|
(if (not (zerop read-only-files))
|
|
|
|
(message "%d files are read-only and not editable." read-only-files)
|
|
|
|
(message "Success update"))))
|
|
|
|
|
|
|
|
(defun helm-ag--edit-abort ()
|
|
|
|
(interactive)
|
|
|
|
(when (y-or-n-p "Discard changes ?")
|
|
|
|
(helm-ag--exit-from-edit-mode)
|
|
|
|
(message "Abort edit")))
|
|
|
|
|
|
|
|
(defun helm-ag--mark-line-deleted ()
|
|
|
|
(interactive)
|
|
|
|
(let* ((beg (line-beginning-position))
|
|
|
|
(end (line-end-position))
|
|
|
|
(ov (make-overlay beg end)))
|
|
|
|
(overlay-put ov 'face 'helm-ag-edit-deleted-line)
|
|
|
|
(overlay-put ov 'helm-ag-deleted t)))
|
|
|
|
|
|
|
|
(defun helm-ag--unmark ()
|
|
|
|
(interactive)
|
|
|
|
(dolist (ov (overlays-in (line-beginning-position) (line-end-position)))
|
|
|
|
(when (overlay-get ov 'helm-ag-deleted)
|
|
|
|
(delete-overlay ov))))
|
|
|
|
|
|
|
|
(defvar helm-ag-edit-map
|
|
|
|
(let ((map (make-sparse-keymap)))
|
|
|
|
(define-key map (kbd "C-c C-c") 'helm-ag--edit-commit)
|
|
|
|
(define-key map (kbd "C-c C-k") 'helm-ag--edit-abort)
|
|
|
|
(define-key map (kbd "C-c C-d") 'helm-ag--mark-line-deleted)
|
|
|
|
(define-key map (kbd "C-c C-u") 'helm-ag--unmark)
|
|
|
|
map))
|
|
|
|
|
2016-08-18 22:01:20 +02:00
|
|
|
(defsubst helm-ag--edit-func-to-keys (func)
|
|
|
|
(key-description (car-safe (where-is-internal func helm-ag-edit-map))))
|
|
|
|
|
2016-04-22 09:59:30 +02:00
|
|
|
(defun helm-ag--edit (_candidate)
|
2016-06-29 09:21:54 +02:00
|
|
|
(let* ((helm-buf-dir (or helm-ag--default-directory
|
|
|
|
helm-ag--last-default-directory
|
|
|
|
default-directory))
|
|
|
|
(default-directory helm-buf-dir))
|
2016-04-22 09:59:30 +02:00
|
|
|
(with-current-buffer (get-buffer-create "*helm-ag-edit*")
|
|
|
|
(erase-buffer)
|
2016-06-29 09:21:54 +02:00
|
|
|
(setq-local helm-ag--default-directory helm-buf-dir)
|
2016-04-22 09:59:30 +02:00
|
|
|
(unless (helm-ag--vimgrep-option)
|
|
|
|
(setq-local helm-ag--search-this-file-p
|
|
|
|
(assoc-default 'search-this-file (helm-get-current-source))))
|
|
|
|
(let (buf-content)
|
|
|
|
(with-current-buffer (get-buffer "*helm-ag*")
|
|
|
|
(goto-char (point-min))
|
|
|
|
(forward-line 1)
|
|
|
|
(let* ((body-start (point))
|
|
|
|
(marked-lines (cl-loop for ov in (overlays-in body-start (point-max))
|
|
|
|
when (eq 'helm-visible-mark (overlay-get ov 'face))
|
|
|
|
return (helm-marked-candidates))))
|
|
|
|
(if (not marked-lines)
|
|
|
|
(setq buf-content (buffer-substring-no-properties
|
|
|
|
body-start (point-max)))
|
2016-09-22 08:19:51 +00:00
|
|
|
(setq buf-content (concat (string-join marked-lines "\n") "\n")))))
|
2016-04-22 09:59:30 +02:00
|
|
|
(insert buf-content)
|
|
|
|
(add-text-properties (point-min) (point-max)
|
|
|
|
'(read-only t rear-nonsticky t front-sticky t))
|
|
|
|
(let ((inhibit-read-only t)
|
|
|
|
(regexp (helm-ag--match-line-regexp)))
|
|
|
|
(setq header-line-format
|
2016-08-18 22:01:20 +02:00
|
|
|
(format "[%s] %s: Commit, %s: Abort"
|
|
|
|
(abbreviate-file-name helm-ag--default-directory)
|
|
|
|
(helm-ag--edit-func-to-keys #'helm-ag--edit-commit)
|
|
|
|
(helm-ag--edit-func-to-keys #'helm-ag--edit-abort)))
|
2016-04-22 09:59:30 +02:00
|
|
|
(goto-char (point-min))
|
|
|
|
(while (re-search-forward regexp nil t)
|
|
|
|
(let ((file-line-begin (match-beginning 4))
|
|
|
|
(file-line-end (match-end 4))
|
|
|
|
(body-begin (match-beginning 3))
|
|
|
|
(body-end (match-end 3)))
|
|
|
|
(add-text-properties file-line-begin file-line-end
|
|
|
|
'(face font-lock-function-name-face
|
|
|
|
intangible t))
|
|
|
|
(remove-text-properties body-begin body-end '(read-only t))
|
|
|
|
(set-text-properties body-end (1+ body-end)
|
|
|
|
'(read-only t rear-nonsticky t))))))))
|
|
|
|
(other-window 1)
|
|
|
|
(switch-to-buffer (get-buffer "*helm-ag-edit*"))
|
|
|
|
(goto-char (point-min))
|
|
|
|
(setq next-error-function 'compilation-next-error-function)
|
|
|
|
(setq-local compilation-locs (make-hash-table :test 'equal :weakness 'value))
|
|
|
|
(use-local-map helm-ag-edit-map))
|
|
|
|
|
|
|
|
(defun helm-ag-edit ()
|
|
|
|
(interactive)
|
|
|
|
(helm-exit-and-execute-action 'helm-ag--edit))
|
|
|
|
|
|
|
|
(defconst helm-ag--help-message
|
|
|
|
"\n* Helm Ag\n
|
|
|
|
|
|
|
|
\n** Specific commands for Helm Ag:\n
|
|
|
|
\\<helm-ag-map>
|
|
|
|
\\[helm-ag--run-other-window-action]\t\t-> Open result in other buffer
|
|
|
|
\\[helm-ag--up-one-level]\t\t-> Search in parent directory.
|
|
|
|
\\[helm-ag-edit]\t\t-> Edit search results.
|
|
|
|
\\[helm-ag-help]\t\t-> Show this help.
|
|
|
|
\n** Helm Ag Map\n
|
|
|
|
\\{helm-map}")
|
|
|
|
|
|
|
|
(defun helm-ag-help ()
|
|
|
|
(interactive)
|
|
|
|
(let ((helm-help-message helm-ag--help-message))
|
|
|
|
(helm-help)))
|
|
|
|
|
|
|
|
(defun helm-ag-mode-jump ()
|
|
|
|
(interactive)
|
|
|
|
(let ((line (helm-current-line-contents)))
|
|
|
|
(helm-ag--find-file-action line 'find-file helm-ag--search-this-file-p)))
|
|
|
|
|
|
|
|
(defun helm-ag-mode-jump-other-window ()
|
|
|
|
(interactive)
|
|
|
|
(let ((line (helm-current-line-contents)))
|
|
|
|
(helm-ag--find-file-action line 'find-file-other-window helm-ag--search-this-file-p)))
|
|
|
|
|
|
|
|
(defvar helm-ag-mode-map
|
|
|
|
(let ((map (make-sparse-keymap)))
|
|
|
|
(define-key map (kbd "RET") 'helm-ag-mode-jump)
|
|
|
|
(define-key map (kbd "C-o") 'helm-ag-mode-jump-other-window)
|
|
|
|
(define-key map (kbd "g") 'helm-ag--update-save-results)
|
|
|
|
map))
|
|
|
|
|
|
|
|
(define-derived-mode helm-ag-mode special-mode "helm-ag"
|
|
|
|
"Major mode to provide actions in helm grep saved buffer.
|
|
|
|
|
|
|
|
Special commands:
|
|
|
|
\\{helm-ag-mode-map}")
|
|
|
|
|
|
|
|
(defun helm-ag--put-result-in-save-buffer (result search-this-file-p)
|
|
|
|
(setq buffer-read-only t)
|
|
|
|
(let ((inhibit-read-only t))
|
|
|
|
(erase-buffer)
|
|
|
|
(insert "-*- mode: helm-ag -*-\n\n"
|
|
|
|
(format "Ag Results for `%s':\n\n" helm-ag--last-query))
|
|
|
|
(save-excursion
|
|
|
|
(insert result)))
|
|
|
|
(helm-ag-mode)
|
|
|
|
(unless (helm-ag--vimgrep-option)
|
|
|
|
(setq-local helm-ag--search-this-file-p search-this-file-p))
|
|
|
|
(setq-local helm-ag--default-directory default-directory))
|
|
|
|
|
|
|
|
(defun helm-ag--save-results (use-other-buf)
|
|
|
|
(let* ((search-this-file-p nil)
|
|
|
|
(result (with-current-buffer helm-buffer
|
|
|
|
(goto-char (point-min))
|
|
|
|
(forward-line 1)
|
|
|
|
(buffer-substring (point) (point-max))))
|
|
|
|
(default-directory helm-ag--default-directory)
|
|
|
|
(buf (if use-other-buf
|
|
|
|
(read-string "Results buffer name: "
|
|
|
|
(format "*helm ag results for '%s'*" helm-ag--last-query))
|
|
|
|
"*helm ag results*")))
|
|
|
|
(when (buffer-live-p (get-buffer buf))
|
|
|
|
(kill-buffer buf))
|
|
|
|
(with-current-buffer (get-buffer-create buf)
|
|
|
|
(helm-ag--put-result-in-save-buffer result search-this-file-p)
|
|
|
|
(pop-to-buffer buf)
|
|
|
|
(message "Helm Ag Results saved in `%s' buffer" buf))))
|
|
|
|
|
|
|
|
(defun helm-ag--update-save-results ()
|
|
|
|
(interactive)
|
|
|
|
(let* ((default-directory helm-ag--default-directory)
|
|
|
|
(result (with-temp-buffer
|
|
|
|
(apply #'process-file (car helm-ag--last-command) nil t nil
|
|
|
|
(cdr helm-ag--last-command))
|
|
|
|
(helm-ag--remove-carrige-returns)
|
2016-08-18 22:01:20 +02:00
|
|
|
(when helm-ag--buffer-search
|
|
|
|
(helm-ag--abbreviate-file-name))
|
2016-04-22 09:59:30 +02:00
|
|
|
(helm-ag--propertize-candidates helm-ag--last-query)
|
|
|
|
(buffer-string))))
|
|
|
|
(helm-ag--put-result-in-save-buffer result helm-ag--search-this-file-p)
|
|
|
|
(message "Update Results")))
|
|
|
|
|
|
|
|
(defun helm-ag--action-save-buffer (_arg)
|
|
|
|
(helm-ag--save-results nil))
|
|
|
|
|
|
|
|
(defun helm-ag--run-save-buffer ()
|
|
|
|
(interactive)
|
|
|
|
(let ((use-other-buf-p current-prefix-arg))
|
|
|
|
(with-helm-alive-p
|
|
|
|
(helm-exit-and-execute-action
|
|
|
|
(lambda (_arg)
|
|
|
|
(helm-ag--save-results use-other-buf-p))))))
|
|
|
|
|
|
|
|
(defun helm-ag--file-of-current-file ()
|
|
|
|
(let ((line (helm-current-line-contents)))
|
|
|
|
(when (string-match helm-grep-split-line-regexp line)
|
|
|
|
(match-string-no-properties 1 line))))
|
|
|
|
|
|
|
|
(defun helm-ag--move-file-common (pred move-fn wrap-fn)
|
|
|
|
(with-helm-window
|
|
|
|
(let ((file (helm-ag--file-of-current-file)))
|
|
|
|
(funcall move-fn)
|
|
|
|
(while (and (not (funcall pred)) (string= file (helm-ag--file-of-current-file)))
|
|
|
|
(funcall move-fn))
|
|
|
|
(when (funcall pred)
|
|
|
|
(funcall wrap-fn)))))
|
|
|
|
|
|
|
|
(defun helm-ag--previous-file ()
|
|
|
|
(interactive)
|
|
|
|
(helm-ag--move-file-common
|
|
|
|
#'helm-beginning-of-source-p #'helm-previous-line #'helm-end-of-buffer))
|
|
|
|
|
|
|
|
(defun helm-ag--next-file ()
|
|
|
|
(interactive)
|
|
|
|
(helm-ag--move-file-common
|
|
|
|
#'helm-end-of-source-p #'helm-next-line #'helm-beginning-of-buffer))
|
|
|
|
|
|
|
|
(defsubst helm-ag--root-directory-p ()
|
|
|
|
(cl-loop for dir in '(".git/" ".hg/")
|
|
|
|
thereis (file-directory-p dir)))
|
|
|
|
|
|
|
|
(defun helm-ag--up-one-level ()
|
|
|
|
(interactive)
|
|
|
|
(if (or (not (helm-ag--root-directory-p))
|
|
|
|
(y-or-n-p "Current directory might be the project root. \
|
|
|
|
Continue searching the parent directory? "))
|
|
|
|
(let ((parent (file-name-directory (directory-file-name default-directory))))
|
|
|
|
(helm-run-after-exit
|
|
|
|
(lambda ()
|
|
|
|
(let* ((default-directory parent)
|
|
|
|
(helm-ag--default-directory parent))
|
|
|
|
(setq helm-ag--last-default-directory default-directory)
|
|
|
|
(helm-attrset 'name (helm-ag--helm-header default-directory) helm-ag-source)
|
2016-07-12 15:08:56 +02:00
|
|
|
(helm :sources '(helm-ag-source) :buffer "*helm-ag*" :keymap helm-ag-map)))))
|
2016-04-22 09:59:30 +02:00
|
|
|
(message nil)))
|
|
|
|
|
|
|
|
;;;###autoload
|
|
|
|
(defun helm-ag-this-file ()
|
|
|
|
(interactive)
|
|
|
|
(helm-ag--init-state)
|
|
|
|
(let ((filename (file-name-nondirectory (buffer-file-name)))
|
|
|
|
(helm-ag--default-directory default-directory))
|
|
|
|
(helm-ag--query)
|
|
|
|
(helm-ag--set-command-feature)
|
|
|
|
(helm-attrset 'search-this-file (file-relative-name (buffer-file-name))
|
|
|
|
helm-ag-source)
|
|
|
|
(helm-attrset 'name (format "Search at %s" filename) helm-ag-source)
|
2016-07-12 15:08:56 +02:00
|
|
|
(helm :sources '(helm-ag-source) :buffer "*helm-ag*" :keymap helm-ag-map)))
|
2016-04-22 09:59:30 +02:00
|
|
|
|
|
|
|
;;;###autoload
|
|
|
|
(defun helm-ag (&optional basedir)
|
|
|
|
(interactive)
|
|
|
|
(helm-ag--init-state)
|
|
|
|
(let ((dir (helm-ag--get-default-directory))
|
|
|
|
targets)
|
|
|
|
(when (listp dir)
|
|
|
|
(setq basedir default-directory
|
|
|
|
targets dir))
|
|
|
|
(let ((helm-ag--default-directory (or basedir dir))
|
|
|
|
(helm-ag--default-target targets))
|
|
|
|
(helm-ag--query)
|
|
|
|
(helm-attrset 'search-this-file nil helm-ag-source)
|
|
|
|
(helm-attrset 'name (helm-ag--helm-header helm-ag--default-directory) helm-ag-source)
|
2016-07-12 15:08:56 +02:00
|
|
|
(helm :sources '(helm-ag-source) :buffer "*helm-ag*" :keymap helm-ag-map))))
|
2016-04-22 09:59:30 +02:00
|
|
|
|
|
|
|
(defun helm-ag--split-string (str)
|
|
|
|
(with-temp-buffer
|
|
|
|
(insert str)
|
|
|
|
(goto-char (point-min))
|
|
|
|
(let ((prev (point))
|
|
|
|
patterns)
|
|
|
|
(while (search-forward " " nil 'move)
|
|
|
|
(cond ((looking-back "\\\\\\\\ " nil)
|
|
|
|
(push (buffer-substring-no-properties prev (1- (point))) patterns)
|
|
|
|
(skip-chars-forward " ")
|
|
|
|
(setq prev (point)))
|
|
|
|
((looking-back "\\\\ " nil)
|
|
|
|
(replace-match " "))
|
|
|
|
(t (push (buffer-substring-no-properties prev (1- (point))) patterns)
|
|
|
|
(skip-chars-forward " ")
|
|
|
|
(setq prev (point)))))
|
|
|
|
(push (buffer-substring-no-properties prev (point)) patterns)
|
|
|
|
(reverse (cl-loop for p in patterns unless (string= p "") collect p)))))
|
|
|
|
|
|
|
|
(defsubst helm-ag--convert-invert-pattern (pattern)
|
|
|
|
(when (and (not helm-ag--command-feature)
|
|
|
|
(string-prefix-p "!" pattern) (> (length pattern) 1))
|
|
|
|
(concat "^(?!.*" (substring pattern 1) ").+$")))
|
|
|
|
|
|
|
|
(defun helm-ag--join-patterns (input)
|
|
|
|
(let ((patterns (helm-ag--split-string input)))
|
|
|
|
(if (= (length patterns) 1)
|
|
|
|
(or (helm-ag--convert-invert-pattern (car patterns))
|
|
|
|
(car patterns))
|
|
|
|
(cl-case helm-ag--command-feature
|
|
|
|
(pt input)
|
2016-09-22 08:19:51 +00:00
|
|
|
(pt-regexp (string-join patterns ".*"))
|
2016-04-22 09:59:30 +02:00
|
|
|
(otherwise (cl-loop for s in patterns
|
|
|
|
if (helm-ag--convert-invert-pattern s)
|
|
|
|
concat (concat "(?=" it ")")
|
|
|
|
else
|
|
|
|
concat (concat "(?=.*" s ".*)")))))))
|
|
|
|
|
|
|
|
(defun helm-ag--do-ag-highlight-patterns (input)
|
|
|
|
(if helm-ag--command-feature
|
|
|
|
(list (helm-ag--join-patterns input))
|
|
|
|
(cl-loop with regexp = (helm-ag--pcre-to-elisp-regexp input)
|
|
|
|
for pattern in (helm-ag--split-string regexp)
|
|
|
|
when (helm-ag--validate-regexp pattern)
|
|
|
|
collect pattern)))
|
|
|
|
|
|
|
|
(defun helm-ag--propertize-candidates (input)
|
2016-06-29 09:21:54 +02:00
|
|
|
(save-excursion
|
|
|
|
(goto-char (point-min))
|
|
|
|
(forward-line 1)
|
|
|
|
(let ((patterns (helm-ag--do-ag-highlight-patterns input)))
|
|
|
|
(cl-loop with one-file-p = (and (not (helm-ag--vimgrep-option))
|
|
|
|
(helm-ag--search-only-one-file-p))
|
|
|
|
while (not (eobp))
|
|
|
|
for num = 1 then (1+ num)
|
|
|
|
do
|
|
|
|
(progn
|
|
|
|
(let ((start (point))
|
|
|
|
(bound (line-end-position)))
|
|
|
|
(if (and one-file-p (search-forward ":" bound t))
|
|
|
|
(set-text-properties (line-beginning-position) (1- (point))
|
|
|
|
'(face helm-grep-lineno))
|
|
|
|
(when (re-search-forward helm-grep-split-line-regexp bound t)
|
|
|
|
(set-text-properties (match-beginning 1) (match-end 1) '(face helm-moccur-buffer))
|
|
|
|
(set-text-properties (match-beginning 2) (match-end 2) '(face helm-grep-lineno))
|
|
|
|
(goto-char (match-beginning 3))))
|
|
|
|
(let ((curpoint (point))
|
|
|
|
(case-fold-search helm-ag--ignore-case))
|
|
|
|
(dolist (pattern patterns)
|
|
|
|
(let ((last-point (point)))
|
|
|
|
(while (re-search-forward pattern bound t)
|
|
|
|
(set-text-properties (match-beginning 0) (match-end 0)
|
|
|
|
'(face helm-match))
|
|
|
|
(when (= last-point (point))
|
|
|
|
(forward-char 1))
|
|
|
|
(setq last-point (point)))
|
|
|
|
(goto-char curpoint))))
|
|
|
|
(put-text-property start bound 'helm-cand-num num))
|
|
|
|
(forward-line 1))))))
|
2016-04-22 09:59:30 +02:00
|
|
|
|
|
|
|
(defun helm-ag-show-status-default-mode-line ()
|
|
|
|
(setq mode-line-format
|
|
|
|
'(" " mode-line-buffer-identification " "
|
|
|
|
(:eval (propertize
|
|
|
|
(format
|
|
|
|
"[AG process finished - (%s results)] "
|
|
|
|
(helm-get-candidate-number))
|
|
|
|
'face 'helm-grep-finish)))))
|
|
|
|
|
|
|
|
(defun helm-ag--do-ag-propertize (input)
|
|
|
|
(with-helm-window
|
|
|
|
(helm-ag--remove-carrige-returns)
|
2016-08-18 22:01:20 +02:00
|
|
|
(when helm-ag--buffer-search
|
|
|
|
(helm-ag--abbreviate-file-name))
|
2016-04-22 09:59:30 +02:00
|
|
|
(helm-ag--propertize-candidates input)
|
|
|
|
(when helm-ag-show-status-function
|
|
|
|
(funcall helm-ag-show-status-function)
|
|
|
|
(force-mode-line-update))))
|
|
|
|
|
|
|
|
(defun helm-ag--construct-extension-options ()
|
|
|
|
(cl-loop for ext in helm-do-ag--extensions
|
|
|
|
unless (string= ext "*")
|
|
|
|
collect
|
|
|
|
(concat "-G" (replace-regexp-in-string
|
|
|
|
"\\*" ""
|
|
|
|
(replace-regexp-in-string "\\." "\\\\." ext)))))
|
|
|
|
|
2016-09-22 08:19:51 +00:00
|
|
|
(defun helm-ag--show-result-p (options has-query)
|
|
|
|
(or has-query
|
|
|
|
(cl-loop for opt in options
|
|
|
|
thereis (string-prefix-p "-g" opt))))
|
|
|
|
|
2016-04-22 09:59:30 +02:00
|
|
|
(defun helm-ag--construct-do-ag-command (pattern)
|
|
|
|
(let* ((opt-query (helm-ag--parse-options-and-query pattern))
|
|
|
|
(options (car opt-query))
|
2016-09-22 08:19:51 +00:00
|
|
|
(query (cdr opt-query))
|
|
|
|
(has-query (not (string= query ""))))
|
2016-04-22 09:59:30 +02:00
|
|
|
(when helm-ag-use-emacs-lisp-regexp
|
|
|
|
(setq query (helm-ag--elisp-regexp-to-pcre query)))
|
2016-09-22 08:19:51 +00:00
|
|
|
(when (helm-ag--show-result-p options has-query)
|
2016-04-22 09:59:30 +02:00
|
|
|
(append (car helm-do-ag--commands)
|
2016-09-22 08:19:51 +00:00
|
|
|
options
|
|
|
|
(and has-query (list (helm-ag--join-patterns query)))
|
2016-04-22 09:59:30 +02:00
|
|
|
(cdr helm-do-ag--commands)))))
|
|
|
|
|
|
|
|
(defun helm-ag--do-ag-set-command ()
|
|
|
|
(let ((cmd-opts (split-string helm-ag-base-command nil t)))
|
|
|
|
(when helm-ag-command-option
|
|
|
|
(setq cmd-opts (append cmd-opts (split-string helm-ag-command-option nil t))))
|
|
|
|
(when helm-ag--extra-options
|
|
|
|
(setq cmd-opts (append cmd-opts (split-string helm-ag--extra-options))))
|
|
|
|
(when helm-ag-ignore-patterns
|
|
|
|
(setq cmd-opts
|
|
|
|
(append cmd-opts
|
|
|
|
(mapcar #'helm-ag--construct-ignore-option
|
|
|
|
helm-ag-ignore-patterns))))
|
|
|
|
(when helm-ag-use-agignore
|
|
|
|
(helm-aif (helm-ag--root-agignore)
|
|
|
|
(setq cmd-opts (append cmd-opts (list "-p" it)))))
|
|
|
|
(when helm-do-ag--extensions
|
|
|
|
(setq cmd-opts (append cmd-opts (helm-ag--construct-extension-options))))
|
|
|
|
(let (targets)
|
|
|
|
(when helm-ag--buffer-search
|
|
|
|
(setq targets (helm-ag--file-visited-buffers)))
|
|
|
|
(setq helm-do-ag--commands
|
|
|
|
(cons cmd-opts
|
|
|
|
(if helm-ag--default-target
|
|
|
|
(append targets (helm-ag--construct-targets helm-ag--default-target))
|
|
|
|
targets))))))
|
|
|
|
|
|
|
|
(defun helm-ag--do-ag-candidate-process ()
|
|
|
|
(let* ((non-essential nil)
|
|
|
|
(default-directory (or helm-ag--default-directory
|
|
|
|
helm-ag--last-default-directory
|
|
|
|
default-directory))
|
|
|
|
(cmd-args (helm-ag--construct-do-ag-command helm-pattern)))
|
|
|
|
(when cmd-args
|
|
|
|
(let ((proc (apply #'start-file-process "helm-do-ag" nil cmd-args)))
|
|
|
|
(setq helm-ag--last-query helm-pattern
|
|
|
|
helm-ag--last-command cmd-args
|
|
|
|
helm-ag--ignore-case (helm-ag--ignore-case-p cmd-args helm-pattern)
|
|
|
|
helm-ag--last-default-directory default-directory)
|
|
|
|
(prog1 proc
|
|
|
|
(set-process-sentinel
|
|
|
|
proc
|
|
|
|
(lambda (process event)
|
|
|
|
(helm-process-deferred-sentinel-hook
|
|
|
|
process event (helm-default-directory))
|
|
|
|
(when (string= event "finished\n")
|
|
|
|
(helm-ag--do-ag-propertize helm-input)))))))))
|
|
|
|
|
|
|
|
(defconst helm-do-ag--help-message
|
|
|
|
"\n* Helm Do Ag\n
|
|
|
|
|
|
|
|
\n** Specific commands for Helm Ag:\n
|
|
|
|
\\<helm-do-ag-map>
|
|
|
|
\\[helm-ag--run-other-window-action]\t\t-> Open result in other buffer
|
|
|
|
\\[helm-ag--do-ag-up-one-level]\t\t-> Search in parent directory.
|
|
|
|
\\[helm-ag-edit]\t\t-> Edit search results.
|
|
|
|
\\[helm-ag--do-ag-help]\t\t-> Show this help.
|
|
|
|
\n** Helm Ag Map\n
|
|
|
|
\\{helm-map}")
|
|
|
|
|
|
|
|
(defun helm-ag--do-ag-help ()
|
|
|
|
(interactive)
|
|
|
|
(let ((helm-help-message helm-do-ag--help-message))
|
|
|
|
(helm-help)))
|
|
|
|
|
|
|
|
(defvar helm-do-ag-map
|
|
|
|
(let ((map (make-sparse-keymap)))
|
|
|
|
(set-keymap-parent map helm-ag-map)
|
|
|
|
(define-key map (kbd "C-l") 'helm-ag--do-ag-up-one-level)
|
|
|
|
(define-key map (kbd "C-c ?") 'helm-ag--do-ag-help)
|
|
|
|
map)
|
|
|
|
"Keymap for `helm-do-ag'.")
|
|
|
|
|
|
|
|
(defvar helm-source-do-ag
|
|
|
|
(helm-build-async-source "The Silver Searcher"
|
|
|
|
:init 'helm-ag--do-ag-set-command
|
|
|
|
:candidates-process 'helm-ag--do-ag-candidate-process
|
|
|
|
:persistent-action 'helm-ag--persistent-action
|
|
|
|
:action helm-ag--actions
|
|
|
|
:nohighlight t
|
|
|
|
:requires-pattern 3
|
|
|
|
:candidate-number-limit 9999
|
2016-07-12 15:08:56 +02:00
|
|
|
:follow (and helm-follow-mode-persistent 1)))
|
2016-04-22 09:59:30 +02:00
|
|
|
|
|
|
|
(defun helm-ag--do-ag-up-one-level ()
|
|
|
|
(interactive)
|
|
|
|
(if (or (not (helm-ag--root-directory-p))
|
|
|
|
(y-or-n-p "Current directory might be the project root. \
|
|
|
|
Continue searching the parent directory? "))
|
|
|
|
(let ((parent (file-name-directory (directory-file-name default-directory)))
|
|
|
|
(initial-input helm-input))
|
|
|
|
(helm-run-after-exit
|
|
|
|
(lambda ()
|
|
|
|
(let ((default-directory parent)
|
|
|
|
(helm-ag--default-directory parent))
|
|
|
|
(setq helm-ag--last-default-directory default-directory)
|
|
|
|
(helm-attrset 'name (helm-ag--helm-header parent)
|
|
|
|
helm-source-do-ag)
|
|
|
|
(helm :sources '(helm-source-do-ag) :buffer "*helm-ag*"
|
2016-07-12 15:08:56 +02:00
|
|
|
:keymap helm-do-ag-map :input initial-input)))))
|
2016-04-22 09:59:30 +02:00
|
|
|
(message nil)))
|
|
|
|
|
|
|
|
(defun helm-ag--set-do-ag-option ()
|
|
|
|
(when (or (< (prefix-numeric-value current-prefix-arg) 0)
|
|
|
|
helm-ag-always-set-extra-option)
|
|
|
|
(let ((option (read-string "Extra options: " (or helm-ag--extra-options "")
|
|
|
|
'helm-ag--extra-options-history)))
|
|
|
|
(setq helm-ag--extra-options option))))
|
|
|
|
|
|
|
|
(defun helm-ag--set-command-feature ()
|
|
|
|
(setq helm-ag--command-feature
|
|
|
|
(when (string-prefix-p "pt" helm-ag-base-command)
|
|
|
|
(if (string-match-p "-e" helm-ag-base-command)
|
|
|
|
'pt-regexp
|
|
|
|
'pt))))
|
|
|
|
|
|
|
|
(defun helm-ag--do-ag-searched-extensions ()
|
|
|
|
(when (and current-prefix-arg (= (abs (prefix-numeric-value current-prefix-arg)) 4))
|
|
|
|
(helm-grep-get-file-extensions helm-ag--default-target)))
|
|
|
|
|
|
|
|
(defsubst helm-do-ag--target-one-directory-p (targets)
|
|
|
|
(and (listp targets) (= (length targets) 1) (file-directory-p (car targets))))
|
|
|
|
|
|
|
|
(defun helm-do-ag--helm ()
|
|
|
|
(let ((search-dir (if (not (helm-ag--windows-p))
|
|
|
|
helm-ag--default-directory
|
|
|
|
(if (helm-do-ag--target-one-directory-p helm-ag--default-target)
|
|
|
|
(car helm-ag--default-target)
|
|
|
|
helm-ag--default-directory))))
|
|
|
|
(helm-attrset 'name (helm-ag--helm-header search-dir)
|
|
|
|
helm-source-do-ag)
|
2016-07-12 15:08:56 +02:00
|
|
|
(helm :sources '(helm-source-do-ag) :buffer "*helm-ag*" :keymap helm-do-ag-map
|
2016-06-29 09:21:54 +02:00
|
|
|
:input (or (helm-ag--marked-input)
|
|
|
|
(helm-ag--insert-thing-at-point helm-ag-insert-at-point)))))
|
2016-04-22 09:59:30 +02:00
|
|
|
|
|
|
|
;;;###autoload
|
|
|
|
(defun helm-do-ag-this-file ()
|
|
|
|
(interactive)
|
|
|
|
(helm-aif (buffer-file-name)
|
|
|
|
(helm-do-ag default-directory (list it))
|
|
|
|
(error "Error: This buffer is not visited file.")))
|
|
|
|
|
|
|
|
;;;###autoload
|
|
|
|
(defun helm-do-ag (&optional basedir targets)
|
|
|
|
(interactive)
|
|
|
|
(require 'helm-mode)
|
|
|
|
(helm-ag--init-state)
|
|
|
|
(let* ((helm-ag--default-directory (or basedir default-directory))
|
|
|
|
(helm-ag--default-target (cond (targets targets)
|
|
|
|
((and (helm-ag--windows-p) basedir) (list basedir))
|
|
|
|
(t
|
|
|
|
(when (and (not basedir) (not helm-ag--buffer-search))
|
|
|
|
(helm-read-file-name
|
|
|
|
"Search in file(s): "
|
|
|
|
:default default-directory
|
|
|
|
:marked-candidates t :must-match t)))))
|
|
|
|
(helm-do-ag--extensions (when helm-ag--default-target
|
|
|
|
(helm-ag--do-ag-searched-extensions)))
|
|
|
|
(one-directory-p (helm-do-ag--target-one-directory-p
|
|
|
|
helm-ag--default-target)))
|
|
|
|
(helm-ag--set-do-ag-option)
|
|
|
|
(helm-ag--set-command-feature)
|
|
|
|
(helm-ag--save-current-context)
|
|
|
|
(helm-attrset 'search-this-file
|
|
|
|
(and (= (length helm-ag--default-target) 1)
|
|
|
|
(not (file-directory-p (car helm-ag--default-target)))
|
|
|
|
(car helm-ag--default-target))
|
|
|
|
helm-source-do-ag)
|
|
|
|
(if (or (helm-ag--windows-p) (not one-directory-p)) ;; Path argument must be specified on Windows
|
|
|
|
(helm-do-ag--helm)
|
|
|
|
(let* ((helm-ag--default-directory
|
|
|
|
(file-name-as-directory (car helm-ag--default-target)))
|
|
|
|
(helm-ag--default-target nil))
|
|
|
|
(helm-do-ag--helm)))))
|
|
|
|
|
|
|
|
(defun helm-ag--project-root ()
|
2016-08-18 22:01:20 +02:00
|
|
|
(cl-loop for dir in '(".git/" ".hg/" ".svn/" ".git")
|
2016-04-22 09:59:30 +02:00
|
|
|
when (locate-dominating-file default-directory dir)
|
|
|
|
return it))
|
|
|
|
|
|
|
|
;;;###autoload
|
|
|
|
(defun helm-ag-project-root ()
|
|
|
|
(interactive)
|
|
|
|
(let ((rootdir (helm-ag--project-root)))
|
|
|
|
(unless rootdir
|
|
|
|
(error "Could not find the project root. Create a git, hg, or svn repository there first. "))
|
|
|
|
(helm-ag rootdir)))
|
|
|
|
|
|
|
|
;;;###autoload
|
|
|
|
(defun helm-do-ag-project-root ()
|
|
|
|
(interactive)
|
|
|
|
(let ((rootdir (helm-ag--project-root)))
|
|
|
|
(unless rootdir
|
|
|
|
(error "Could not find the project root. Create a git, hg, or svn repository there first. "))
|
|
|
|
(helm-do-ag rootdir)))
|
|
|
|
|
|
|
|
;;;###autoload
|
|
|
|
(defun helm-ag-buffers ()
|
|
|
|
(interactive)
|
|
|
|
(let ((helm-ag--buffer-search t))
|
|
|
|
(helm-ag)))
|
|
|
|
|
|
|
|
;;;###autoload
|
|
|
|
(defun helm-do-ag-buffers ()
|
|
|
|
(interactive)
|
|
|
|
(let ((helm-ag--buffer-search t))
|
|
|
|
(helm-do-ag)))
|
|
|
|
|
|
|
|
(provide 'helm-ag)
|
|
|
|
|
|
|
|
;;; helm-ag.el ends here
|