/aquamacs/src/site-lisp/visual-line.el

http://github.com/davidswelt/aquamacs-emacs · Emacs Lisp · 513 lines · 339 code · 69 blank · 105 comment · 6 complexity · 302b3888ea3105e0ce5d973f8c6c7c51 MD5 · raw file

  1. ;;; visual-line.el
  2. ;; Copyright (C) 2008 Free Software Foundation
  3. ;; Maintainer: David Reitter <david.reitter@gmail.com>
  4. ;; Authors: David Reitter
  5. ;; Keywords: mail
  6. ;; This file is part of GNU Emacs.
  7. ;; GNU Emacs is free software; you can redistribute it and/or modify
  8. ;; it under the terms of the GNU General Public License as published by
  9. ;; the Free Software Foundation; either version 3, or (at your option)
  10. ;; any later version.
  11. ;; GNU Emacs is distributed in the hope that it will be useful,
  12. ;; but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. ;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  14. ;; GNU General Public License for more details.
  15. ;; You should have received a copy of the GNU General Public License
  16. ;; along with GNU Emacs; see the file COPYING. If not, write to the
  17. ;; Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
  18. ;; Boston, MA 02110-1301, USA.
  19. ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
  20. ;;
  21. ;; Overview:
  22. ;;
  23. ;; `visual-line-mode' and `global-visual-line-mode' enable
  24. ;; navigation by visual lines. Vertical movement commands such as
  25. ;; `next-line' and `previous-line' (normally bound to up/down arrow
  26. ;; keys) will move the point to the next line as shown on the
  27. ;; screen, even if that is the same line in the underlying buffer.
  28. ;; The point is moved to a position that is located (on the screen)
  29. ;; horizontally close (pixel-wise), rather than to an equivalent
  30. ;; by-character column.
  31. ;;
  32. ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
  33. ;;
  34. ;; Notable changes:
  35. ;;
  36. ;; Initial version:
  37. ;; This file was adapted from Aquamacs Emacs.
  38. ;; Lennart Borgmann contributed the code that creates a minor mode
  39. ;; for this.
  40. ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
  41. ;; Code Comments:
  42. ;; Note that `visual-line-up' and friends use two different methods to
  43. ;; figure out the best position to move to because of a slowness with
  44. ;; outline-(minor-)mode. One of the methods (basically binary search) is
  45. ;; much faster when a lot of hidden text is present, but a bit slower in
  46. ;; all other cases.
  47. (defun visual-col-at-point ()
  48. "Returns the visual column at point.
  49. The visual column is relative to the left window edge, not
  50. to the beginning of the (unwrapped) line."
  51. (- (point)
  52. (save-excursion
  53. (vertical-motion 0)
  54. (point))))
  55. ;; seems slower (in situations with very long lines)
  56. ;;(or (car (nth 6 (posn-at-point))) 0))
  57. (defun visual-pixel-col-at-point ()
  58. "Returns the pixel column at point.
  59. This is the distance from the left edge of the window
  60. to the character at point."
  61. (or (car-safe
  62. (pos-visible-in-window-p (point) nil 'partial))
  63. 0))
  64. (defvar visual-movement-temporary-goal-column nil)
  65. (make-variable-buffer-local 'visual-movement-temporary-goal-column)
  66. (defvar visual-previous-scroll-margin 'none)
  67. (defun visual-restore-scroll-margin ()
  68. "Restore the scroll margin."
  69. (if (integerp visual-previous-scroll-margin)
  70. (setq scroll-margin visual-previous-scroll-margin))
  71. (remove-hook 'pre-command-hook 'visual-restore-scroll-margin))
  72. (defcustom visual-scroll-margin nil
  73. "Number of lines of margin at top and bottom of a window.
  74. For visual scrolling with up and down keys, this value
  75. applies instead of `scroll-margin' if it is non-nil.
  76. The reason this variable exists is that clicks in the first and last
  77. line of a window will set the cursor within the standard scroll-margin,
  78. causing the buffer to scroll immediately. This is usually undesired.
  79. In this case, set `scroll-margin' to zero and `visual-scroll-margin'
  80. to the desired margin."
  81. :group 'Windows)
  82. (defun visual-line-up (num-lines)
  83. (interactive "p")
  84. (unless (bobp)
  85. (let ((to-set-point)
  86. (old-point (point)))
  87. (let ((inhibit-point-motion-hooks t))
  88. (let ((visual-pixel-col (visual-pixel-col-at-point))
  89. (end-of-old-line))
  90. ;; temporary binding of scroll-margin
  91. ;; cannot do this with a temporary let binding
  92. (setq visual-previous-scroll-margin scroll-margin)
  93. (if visual-scroll-margin
  94. (setq scroll-margin visual-scroll-margin))
  95. (add-hook 'pre-command-hook 'visual-restore-scroll-margin)
  96. (let ((x-char) ;; jump to this char position (x). takes precedence
  97. (x (car (posn-x-y (posn-at-point))))) ;; jump to this pixel pos
  98. (save-excursion
  99. (vertical-motion 1) ;; trying going one down, to left
  100. (setq end-of-old-line (point)))
  101. (vertical-motion 0)
  102. (let* ((beg-of-old-line
  103. ;; move right, but not further than to end of line
  104. (prog1 (point)
  105. (vertical-motion (- num-lines)))) ;; one up again
  106. (beg-of-new-line (point))
  107. (rel-beg-of-old-line (- beg-of-old-line (point) 1)))
  108. ;; handle track-eol...
  109. (if (and track-eol (= old-point (1- end-of-old-line))
  110. ;; Don't count beg of empty line as end of line
  111. ;; unless we just did explicit end-of-line.
  112. (or (not (= old-point beg-of-old-line))
  113. (eq last-command 'end-of-line)))
  114. (setq visual-movement-temporary-goal-column 9999))
  115. ;; approximate positioning
  116. (if (and (or goal-column visual-movement-temporary-goal-column)
  117. (memq last-command '(visual-line-up
  118. visual-line-down))
  119. ;;(= old-point (1- end-of-old-line))
  120. )
  121. (if goal-column
  122. (setq x-char goal-column)
  123. (setq x visual-movement-temporary-goal-column))
  124. ;; else, do complete positioning
  125. ;; save original position
  126. (setq visual-movement-temporary-goal-column visual-pixel-col))
  127. (if x-char
  128. (forward-char (min x-char rel-beg-of-old-line))
  129. (unless (pos-visible-in-window-p (point) nil 'partial)
  130. (redisplay t))
  131. (let ((y (cdr (posn-x-y (posn-at-point)))))
  132. (when y
  133. (if (> x 2)
  134. ;; at low x, posn-at-x-y likes to change windows if
  135. ;; windows are vertically split.
  136. (setq to-set-point (posn-at-x-y x y))))))))))
  137. ;; point motion hooks aren't inhibited any longer
  138. (and to-set-point (posn-set-point to-set-point))
  139. (if (eq (point) old-point)
  140. ;; got stuck, perhaps at the end of
  141. ;; several visual lines of intangible text?
  142. (beginning-of-line)))))
  143. (defun visual-line-down (num-lines)
  144. (interactive "p")
  145. (if (and next-line-add-newlines (= num-lines 1))
  146. (if (save-excursion (end-of-line) (eobp))
  147. ;; When adding a newline, don't expand an abbrev.
  148. (let ((abbrev-mode nil))
  149. (end-of-line)
  150. (insert hard-newline))))
  151. (unless (eobp)
  152. (let ((to-set-point))
  153. (let ((inhibit-point-motion-hooks t))
  154. (let ((old-point (point))
  155. (visual-pixel-col (visual-pixel-col-at-point))
  156. (beg-of-line)
  157. (next-line-start)
  158. (rel-next-line-start))
  159. ;; temporary binding of scroll-margin
  160. ;; cannot do this with a temporary let binding
  161. (setq visual-previous-scroll-margin scroll-margin)
  162. (if visual-scroll-margin
  163. (setq scroll-margin visual-scroll-margin))
  164. (add-hook 'pre-command-hook 'visual-restore-scroll-margin)
  165. (let ((x-char) ;; jump to this char position (x). takes precedence
  166. (x (car (posn-x-y (posn-at-point))))) ;; jump to this pixel pos
  167. (vertical-motion num-lines) ;; down
  168. (save-excursion
  169. (setq beg-of-line (point))
  170. (vertical-motion +1) ;; down
  171. (setq next-line-start (point))
  172. (setq rel-next-line-start (- (point) beg-of-line 1)))
  173. (unless (= beg-of-line (point-max))
  174. ;; handle track-eol...
  175. (if (and track-eol (= old-point (1- next-line-start))
  176. ;; Don't count beg of empty line as end of line
  177. ;; unless we just did explicit end-of-line.
  178. (or (not (= 0 visual-pixel-col))
  179. (eq last-command 'end-of-line)))
  180. (setq visual-movement-temporary-goal-column 9999))
  181. ;; approximate positioning
  182. (if (and (or goal-column visual-movement-temporary-goal-column)
  183. (memq last-command '(visual-line-up
  184. visual-line-down))
  185. ;(= old-point (- beg-of-line 1))
  186. ;; do not re-set temp column but jump to the old one
  187. ;; in case of repeated movement commands
  188. )
  189. (if goal-column
  190. (setq x-char goal-column)
  191. (setq x visual-movement-temporary-goal-column))
  192. ;; else, do complete positioning
  193. ;; save original position
  194. (setq visual-movement-temporary-goal-column visual-pixel-col))
  195. (if x-char
  196. (forward-char (min x-char rel-next-line-start))
  197. (unless (pos-visible-in-window-p (point) nil 'partial)
  198. (redisplay t))
  199. (let ((y (cdr (posn-x-y (posn-at-point)))))
  200. (when y
  201. (if (> x 2)
  202. ;; at low x, posn-at-x-y likes to change windows if
  203. ;; windows are vertically split.
  204. (setq to-set-point (posn-at-x-y x y)))))))
  205. )))
  206. ;; point motion hooks aren't inhibited any longer
  207. (and to-set-point (posn-set-point to-set-point)))))
  208. (defun beginning-of-visual-line (&optional n)
  209. "Move point to the beginning of the current line.
  210. If `word-wrap' is nil, we move to the beginning of the buffer
  211. line (as in `beginning-of-line'); otherwise, point is moved to
  212. the beginning of the visual line."
  213. (interactive)
  214. (if word-wrap
  215. (progn
  216. (if (and n (/= n 1))
  217. (vertical-motion (1- n))
  218. ;; the following would need Emacs 23
  219. ;; (let ((line-move-visual t))
  220. ;; (line-move (1- n) t)))
  221. (vertical-motion 0))
  222. (skip-read-only-prompt))
  223. (beginning-of-line n)))
  224. (defun end-of-visual-line (&optional n)
  225. "Move point to the end of the current line.
  226. If `word-wrap' is nil, we move to the end of the line (as in
  227. `beginning-of-line'); otherwise, point is moved to the end of the
  228. visual line."
  229. (interactive)
  230. (if word-wrap
  231. (unless (eobp)
  232. (progn
  233. (if (and n (/= n 1))
  234. (vertical-motion (1- n))
  235. (vertical-motion 1))
  236. (skip-chars-backward " \r\n" (- (point) 1))))
  237. (end-of-line n)))
  238. ;; this code based on simple.el
  239. (defun kill-visual-line (&optional arg)
  240. "Kill the rest of the visual line; if no nonblanks there, kill thru
  241. newline.
  242. With prefix argument, kill that many lines from point.
  243. Negative arguments kill lines backward.
  244. With zero argument, kills the text before point on hthe current line.
  245. When calling from a program, nil means \"no arg\",
  246. a number counts as a prefix arg.
  247. To kill a whole line, when point is not at the beginning, type \
  248. \\[beginning-of-line] \\[kill-line] \\[kill-line].
  249. If `kill-whole-line' is non-nil, then this command kills the whole line
  250. including its terminating newline, when used at the beginning of a line
  251. with no argument. As a consequence, you can always kill a whole line
  252. by typing \\[beginning-of-line] \\[kill-line].
  253. If you want to append the killed line to the last killed text,
  254. use \\[append-next-kill] before \\[kill-line].
  255. If the buffer is read-only, Emacs will beep and refrain from deleting
  256. the line, but put the line in the kill ring anyway. This means that
  257. you can use this command to copy text from a read-only buffer.
  258. \(If the variable `kill-read-only-ok' is non-nil, then this won't
  259. even beep.)
  260. ``Line'' is defined as visual line, from the leftmost to the
  261. rightmost position of a single visual line, if `word-wrap' is
  262. non-nil. Otherwise, this function behaves exactly like
  263. `kill-line'."
  264. (interactive "P")
  265. (if word-wrap
  266. (kill-region
  267. (point)
  268. ;; It is better to move point to the other end of the
  269. ;; kill before killing. That way, in a read-only
  270. ;; buffer, point moves across the text that is copied
  271. ;; to the kill ring. The choice has no ef
  272. ;; now that undo records the value of point from before
  273. ;; the command was run.
  274. (progn
  275. (if arg
  276. (vertical-motion (prefix-numeric-value arg))
  277. (if (eobp)
  278. (signal 'end-of-buffer nil))
  279. (let ((end
  280. (save-excursion
  281. (vertical-motion 1)
  282. ; we're possibly one too far
  283. (skip-chars-backward "\r\n" (- (point) 1))
  284. (point))))
  285. (if (or (save-excursion
  286. ;; If trailing whitespace is visible,
  287. ;; don't treat it as nothing.
  288. (unless show-trailing-whitespace
  289. (skip-chars-forward " \t" end))
  290. (= (point) end))
  291. (and kill-whole-line (bolp)))
  292. (vertical-motion 1)
  293. (goto-char end))))
  294. (point)))
  295. (kill-line arg)))
  296. ;; to do: we should really delete everything
  297. ;; that is not read-only, rather than just
  298. ;; exclude a prompt
  299. (defun skip-read-only-prompt (&optional max)
  300. (while (and (get-char-property (point) 'read-only)
  301. (< (point) (or max (point-max)))
  302. (forward-char))))
  303. (defun kill-whole-visual-line (&optional arg)
  304. "Kill current visual line.
  305. With prefix arg, kill that many lines starting from the current
  306. line. If arg is negative, kill backward. Also kill the
  307. preceding newline. \(This is meant to make \\[repeat] work well
  308. with negative arguments.\)
  309. If arg is zero, kill current line but exclude the trailing
  310. newline.
  311. In `transient-mark-mode', if arg is one and the mark is
  312. active (a region is set), kill the region.
  313. ``Line'' is defined as visual line, from the leftmost to the
  314. rightmost position of a single visual line, if `word-wrap' is
  315. non-nil. Otherwise, lines are treated just like `kill-line'
  316. would do."
  317. (interactive "p")
  318. (if (and transient-mark-mode
  319. (= arg 1) mark-active) ;; region defined?
  320. (call-interactively #'kill-region)
  321. (if (not word-wrap)
  322. (progn
  323. (beginning-of-line)
  324. (skip-read-only-prompt)
  325. (kill-line arg))
  326. (if (and (> arg 0) (eobp) (save-excursion (vertical-motion 0) (eobp)))
  327. (signal 'end-of-buffer nil))
  328. (if (and (< arg 0) (bobp) (save-excursion (vertical-motion 1) (bobp)))
  329. (signal 'beginning-of-buffer nil))
  330. (unless (eq last-command 'kill-region)
  331. (kill-new "")
  332. (setq last-command 'kill-region))
  333. (cond ((zerop arg)
  334. ;; We need to kill in two steps, because the previous command
  335. ;; could have been a kill command, in which case the text
  336. ;; before point needs to be prepended to the current kill
  337. ;; ring entry and the text after point appended. Also, we
  338. ;; neehd to use save-excursion to avoid copying the same text
  339. ;; twice to the kill ring in read-only buffers.
  340. (save-excursion
  341. ;; delete in one go
  342. (kill-region (progn (vertical-motion 0)
  343. (skip-read-only-prompt) (point))
  344. (progn (vertical-motion 1) (point)))
  345. ))
  346. ((< arg 0)
  347. (save-excursion
  348. (kill-region (point) (progn (end-of-visual-line) (point))))
  349. (kill-region (point)
  350. (progn
  351. (vertical-motion (1+ arg))
  352. (unless (bobp) (backward-char))
  353. (point))))
  354. (t
  355. (save-excursion
  356. (kill-region (let ((ep (point)))
  357. (vertical-motion 0)
  358. (skip-read-only-prompt ep)
  359. (point))
  360. (progn
  361. (vertical-motion arg)
  362. (point)))))))))
  363. ;; mark functions for CUA
  364. (dolist (cmd
  365. '( beginning-of-visual-line
  366. end-of-visual-line
  367. visual-line-down visual-line-up))
  368. (put cmd 'CUA 'move))
  369. (defalias 'original-kill-line 'kill-line)
  370. (defalias 'original-next-line 'next-line)
  371. (defalias 'original-previous-line 'previous-line)
  372. (defalias 'original-move-beginning-of-line 'move-beginning-of-line)
  373. (defalias 'original-move-end-of-line 'move-end-of-line)
  374. (defun line-wrapped-p ()
  375. "Return non-nil if the current line is wrapped."
  376. (let ((here (point))
  377. result)
  378. (vertical-motion 0)
  379. (setq result (/= (line-beginning-position) (point)))
  380. (unless result
  381. (let ((line-end-pos (line-end-position)))
  382. (vertical-motion 1)
  383. (setq result (/= line-end-pos (- (point) 1)))))
  384. (goto-char here)
  385. result))
  386. (when (< emacs-major-version 23)
  387. (defvar visual-line-map
  388. (let ((map (make-sparse-keymap)))
  389. (define-key map [remap next-line] 'visual-line-down)
  390. (define-key map [remap previous-line] 'visual-line-up)
  391. (define-key map [remap kill-line] 'kill-visual-line)
  392. (define-key map [(control shift ?k)] 'original-kill-line)
  393. (define-key map [remap move-beginning-of-line] 'beginning-of-visual-line)
  394. (define-key map [remap move-end-of-line] 'end-of-visual-line)
  395. map))
  396. (define-minor-mode visual-line-mode
  397. "Define key binding for visual line moves."
  398. :keymap visual-line-map
  399. :group 'convenience
  400. (setq line-move-visual visual-line-mode))
  401. (defun turn-on-visual-line-mode ()
  402. (visual-line-mode 1))
  403. (define-globalized-minor-mode global-visual-line-mode
  404. visual-line-mode turn-on-visual-line-mode))
  405. (defface blank-newline
  406. '((((class color) (background dark))
  407. (:foreground "lightgrey" :bold nil))
  408. (((class color) (background light))
  409. ( :foreground "lightgrey" :bold nil))
  410. (t (:bold nil :underline t)))
  411. "Face used to visualize NEWLINE char mapping.
  412. See `blank-display-mappings'."
  413. :group 'blank)
  414. ;; Pilcrow Unicode 00b6
  415. (defvar show-newlines-newline-code
  416. (vector (make-glyph-code ?\x00B6 'blank-newline) 10))
  417. (define-minor-mode show-newlines-mode
  418. "Mark newlines in current buffer"
  419. :group 'convenience
  420. (unless buffer-display-table
  421. (setq buffer-display-table (or standard-display-table (make-display-table))))
  422. (if show-newlines-mode
  423. (aset buffer-display-table 10 show-newlines-newline-code)
  424. (aset buffer-display-table 10 nil)))
  425. (define-minor-mode global-show-newlines-mode
  426. "Mark newlines in all buffers"
  427. :group 'convenience
  428. :global t
  429. (unless standard-display-table
  430. (setq standard-display-table (make-display-table)))
  431. (if global-show-newlines-mode
  432. (aset standard-display-table 10 show-newlines-newline-code)
  433. (aset standard-display-table 10 nil))
  434. (dolist (buffer (buffer-list))
  435. (with-current-buffer buffer
  436. (if buffer-display-table
  437. (show-newlines-mode (if global-show-newlines-mode 1 -1))))))
  438. ;;(setq show-newlines-newline-code (vector (make-glyph-code 2230 'blank-newline) 10))
  439. ;;(setf (aref show-newlines-newline-code 0) (make-glyph-code 34 'blank-newline))
  440. (define-key-after menu-bar-showhide-menu [show-newlines-mode]
  441. (menu-bar-make-mm-toggle global-show-newlines-mode
  442. "Show Newlines"
  443. "Show hard newlines") 'highlight-paren-mode)
  444. (provide 'visual-line)