-
Notifications
You must be signed in to change notification settings - Fork 6
/
consult-ag.el
80 lines (66 loc) · 3.03 KB
/
consult-ag.el
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
;;; consult-ag.el --- The silver searcher integration using Consult -*- lexical-binding: t; -*-
;; Copyright (C) 2022 Kanon Kakuno
;; Author: Kanon Kakuno <yadex205@outlook.jp> and contributors
;; Homepage: https://github.com/yadex205/consult-ag
;; Package-Requires: ((emacs "27.1") (consult "0.32"))
;; SPDX-License-Identifier: MIT
;; Version: 0.2.0
;; This file is not part of GNU Emacs.
;;; Commentary:
;; consult-ag provides interfaces for using `ag` (The Silver Searcher).
;; To use this, turn on `consult-ag` in your init-file or interactively.
;;; Code:
(eval-when-compile ; IDK but required for byte-compile
(require 'cl-lib)
(require 'subr-x))
(require 'consult)
(defun consult-ag--builder (input)
"Build command line given INPUT."
(pcase-let* ((cmd (split-string-and-unquote "ag --vimgrep"))
(`(,arg . ,opts) (consult--command-split input)))
`(,@cmd ,@opts ,arg ".")))
(defun consult-ag--format (line)
"Parse LINE into candidate text."
(when (string-match "^\\([^:]+\\):\\([0-9]+\\):\\([0-9]+\\):\\(.*\\)$" line)
(let* ((filename (match-string 1 line))
(row (match-string 2 line))
(column (match-string 3 line))
(body (match-string 4 line))
(candidate (format "%s:%s:%s:%s"
(propertize filename 'face 'consult-file)
(propertize row 'face 'consult-line-number)
(propertize column 'face 'consult-line-number) body)))
(propertize candidate 'filename filename 'row row 'column column))))
(defun consult-ag--grep-position (cand &optional find-file)
"Return the candidate position marker for CAND.
FIND-FILE is the file open function, defaulting to `find-file`."
(when cand
(let ((file (get-text-property 0 'filename cand))
(row (string-to-number (get-text-property 0 'row cand)))
(column (- (string-to-number (get-text-property 0 'column cand)) 1)))
(consult--marker-from-line-column (funcall (or find-file #'find-file) file) row column))))
(defun consult-ag--grep-state ()
"Not documented."
(let ((open (consult--temporary-files))
(jump (consult--jump-state)))
(lambda (action cand)
(unless cand
(funcall open nil))
(funcall jump action (consult-ag--grep-position cand open)))))
;;;###autoload
(defun consult-ag (&optional target initial)
"Consult ag for query in TARGET file(s) with INITIAL input."
(interactive)
(pcase-let* ((`(,prompt ,paths ,dir) (consult--directory-prompt "Consult ag: " target))
(default-directory dir))
(consult--read (consult--async-command #'consult-ag--builder
(consult--async-map #'consult-ag--format))
:prompt prompt
:lookup #'consult--lookup-member
:state (consult-ag--grep-state)
:initial (consult--async-split-initial initial)
:require-match t
:category 'file
:sort nil)))
(provide 'consult-ag)
;;; consult-ag.el ends here