-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathzzz-to-char.el
104 lines (89 loc) · 3.43 KB
/
zzz-to-char.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
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
;;; zzz-to-char.el --- Fancy version of `zap-to-char' command -*- lexical-binding: t; -*-
;;
;; Copyright © 2015–present Mark Karpov <markkarpov92@gmail.com>
;;
;; Author: Mark Karpov <markkarpov92@gmail.com>
;; URL: https://github.com/mrkkrp/zzz-to-char
;; Version: 1.0.0
;; Package-Requires: ((emacs "24.4") (avy "0.3.0"))
;; Keywords: convenience
;;
;; This file is not part of GNU Emacs.
;;
;; This program is free software: you can redistribute it and/or modify it
;; under the terms of the GNU General Public License as published by the
;; Free Software Foundation, either version 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:
;; This package provides two new commands: `zzz-to-char' and
;; `zzz-up-to-char' which work like the built-ins `zap-to-char' and
;; `zap-up-to-char', but allow the user to quickly select the exact
;; character they want to zzz to. The commands work like the built-ins when
;; there is only one occurrence of the target character, excepting that they
;; automatically work in the backward direction, too. One can specify how
;; many characters to scan from each side of the point, see
;; `zzz-to-char-reach'.
;;; Code:
(require 'avy)
(require 'cl-lib)
(defgroup zzz-to-char nil
"A fancy version of `zap-to-char' command."
:group 'convenience
:tag "Zzz to char"
:prefix "zzz-to-char-"
:link '(url-link :tag "GitHub" "https://github.com/mrkkrp/zzz-to-char"))
(defcustom zzz-to-char-reach 80
"Number of characters to scan on each side of the point."
:tag "How many characters to scan"
:type 'integer)
(defun zzz-to-char--base (char n-shift)
"Kill text between the point and the character CHAR.
Boundary of text to kill that doesn't coincide with point
position can be shifted with help of the N-SHIFT argument.
This is an internal function, see also `zzz-to-char' and
`zzz-up-to-char'."
(let ((p (point))
(avy-all-windows nil))
(avy-with zzz-to-char
(avy-jump
(if (= 13 char)
"\n"
(regexp-quote (string char)))
:window-flip nil
:beg (max (- p zzz-to-char-reach)
(point-min))
:end (min (+ p zzz-to-char-reach)
(point-max))
:pred (lambda () (> (abs (- p (point))) n-shift))))
(let ((n (point)))
(when (/= n p)
(cl-destructuring-bind (beg . end)
(if (> n p)
(cons p (- (1+ n) n-shift))
(cons (+ n n-shift) p))
(goto-char end)
(kill-region beg end))))))
;;;###autoload
(defun zzz-to-char (char)
"Kill text between the point and the character CHAR.
This command is similar to `zap-to-char', it kills the target
character too."
(interactive (list (read-char "Zzz to: " t)))
(zzz-to-char--base char 0))
;;;###autoload
(defun zzz-to-char-up-to-char (char)
"Kill text between the point and the character CHAR.
This command is similar to `zap-up-to-char', it doesn't kill the
target character."
(interactive (list (read-char "Zzz up to: " t)))
(zzz-to-char--base char 1))
(provide 'zzz-to-char)
;;; zzz-to-char.el ends here