-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathmove-lines.el
145 lines (131 loc) · 4.8 KB
/
move-lines.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
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
;;; move-lines.el --- move current line or lines surrounding region up or down
;;
;; Copyright (C) 2014-2017 Emanuele Tomasi <[email protected]>
;;
;; Author: Emanuele Tomasi <[email protected]>
;; URL: https://github.com/targzeta/move-lines
;; Maintainer: Emanuele Tomasi <[email protected]>
;; Keywords: convenience
;; Version: 2.0
;;
;; 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;
;;
;; There are two entry points: `move-lines-up' moves the text up and
;; `move-lines-down' that moves the text down.
;;
;; Copy this file in a directory which is in the Emacs `load-path'. Then,
;; execute the following code either directly or in your .emacs file:
;;
;; (require 'move-lines)
;; (move-lines-binding)
;;
;; Now, you can move the line(s) up by M-p or M-<up> or down by M-n or
;; M-<down>.
;;
;;; Code:
(defun move-lines--internal (n)
"Moves the current line or, if region is actives, the lines surrounding
region, of N lines. Down if N is positive, up if is negative"
;; The text area spans from the beginning of the first line (text-start) to
;; the end of the last line, '\n' included (text-end). Its coordinates are
;; the number of chars from the beginning of buffer.
;; The region is within the text area and its coordinates are the (negative)
;; numbers of chars from text-end.
;;
;; E.g.:
;; Lorem ipsum dolor sit amet, consectetur adipisci elit,\n
;; ^ ^ ^ ^
;; text-start(1) region-start(-35) region-end(-14) text-end(55)
;;
;; We assume that point is always ahead the mark, else temporarily we swap
;; them.
;; If we act on the latest line of the buffer and it hasn't a newline, we
;; temporarily add one.
(let* (text-start
text-end
(region-start (point))
(region-end region-start)
swap-point-mark
delete-latest-newline)
;; STEP 1: identifying the text to cut.
(when (region-active-p)
(if (> (point) (mark))
(setq region-start (mark))
(exchange-point-and-mark)
(setq swap-point-mark t
region-end (point))))
;; text-end and region-end
(end-of-line)
;; If point !< point-max, this buffers doesn't have the trailing newline.
(if (< (point) (point-max))
(forward-char 1)
(setq delete-latest-newline t)
(insert-char ?\n))
(setq text-end (point)
region-end (- region-end text-end))
;; text-start and region-start
(goto-char region-start)
(beginning-of-line)
(setq text-start (point)
region-start (- region-start text-end))
;; STEP 2: cut and paste.
(let ((text (delete-and-extract-region text-start text-end)))
(forward-line n)
;; If the current-column != 0, I have moved the region at the bottom of a
;; buffer doesn't have the trailing newline.
(when (not (= (current-column) 0))
(insert-char ?\n)
(setq delete-latest-newline t))
(insert text))
;; STEP 3: Restoring.
(forward-char region-end)
(when delete-latest-newline
(save-excursion
(goto-char (point-max))
(delete-char -1)))
(when (region-active-p)
(setq deactivate-mark nil)
(set-mark (+ (point) (- region-start region-end)))
(if swap-point-mark
(exchange-point-and-mark)))))
;;;###autoload
(defun move-lines-up (n)
"Moves the current line or, if region is actives, the lines surrounding
region, up by N lines, or 1 line if N is nil."
(interactive "p")
(if (eq n nil)
(setq n 1))
(move-lines--internal (- n)))
;;;###autoload
(defun move-lines-down (n)
"Moves the current line or, if region is actives, the lines surrounding
region, down by N lines, or 1 line if N is nil."
(interactive "p")
(if (eq n nil)
(setq n 1))
(move-lines--internal n))
;;;###autoload
(defun move-lines-binding ()
"Sets the default key binding for moving lines. M-p or M-<up> for moving up
and M-n or M-<down> for moving down."
(global-set-key (kbd "M-p") 'move-lines-up)
(global-set-key (kbd "M-<up>") 'move-lines-up)
(global-set-key (kbd "M-n") 'move-lines-down)
(global-set-key (kbd "M-<down>") 'move-lines-down))
(provide 'move-lines)
;; move-lines.el ends here