diff options
author | Alexander Sulfrian <alexander@sulfrian.net> | 2009-06-01 19:13:30 +0200 |
---|---|---|
committer | Alexander Sulfrian <alexander@sulfrian.net> | 2009-06-01 19:13:30 +0200 |
commit | 3513d65b8b63f639059eb1909eb490cd3dcf2c43 (patch) | |
tree | ca7ad9d737520969ed1fdbfa454164a3ac1f97ff /emacs.d/lisp | |
parent | a0dcc9055d0822e024fab9c1fe5344f0cd1a29ae (diff) | |
download | dotfiles-3513d65b8b63f639059eb1909eb490cd3dcf2c43.tar.gz dotfiles-3513d65b8b63f639059eb1909eb490cd3dcf2c43.tar.xz dotfiles-3513d65b8b63f639059eb1909eb490cd3dcf2c43.zip |
updated emacs config: rect-mark, saveposition, gitignore
added rect-mark to make it posible to visual select rectangle regions
added saveposition to jump to the position of the last visit of the file
added gitignore file to ignore the dynamic files by emacs
Diffstat (limited to '')
-rw-r--r-- | emacs.d/lisp/rect-mark.el | 445 |
1 files changed, 445 insertions, 0 deletions
diff --git a/emacs.d/lisp/rect-mark.el b/emacs.d/lisp/rect-mark.el new file mode 100644 index 0000000..6373230 --- /dev/null +++ b/emacs.d/lisp/rect-mark.el @@ -0,0 +1,445 @@ +;;; rect-mark.el --- Mark a rectangle of text with highlighting. + +;;; Copyright (C) 1994, 1995 Rick Sladkey <jrs@world.std.com> + +;;; This file is not part of GNU Emacs but it is distributed under the +;;; same conditions as GNU Emacs. + +;;; This is free software. +;;; GNU Emacs 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 2, or (at your +;;; option) any later version. + +;;; GNU Emacs 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 GNU Emacs; see the file COPYING. If not, write to the +;;; Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA. + +;; Author: Rick Sladkey <jrs@world.std.com> +;; Version: 1.4 + +;;; Commentary: + +;; If you use both transient-mark-mode and picture-mode, you will +;; probably realize how convenient it would be to be able to highlight +;; the region between point and mark as a rectangle. Have you ever +;; wished you could see where exactly those other two corners fell +;; before you operated on a rectangle? If so, then this program is +;; for you. + +;; For example, you can set the mark in preparation for a rectangle +;; command with `C-x r C-SPC', watch the highlighted rectangle grow as +;; you move the cursor to the other corner, and then issue the command +;; and the rectangle disappears. Or if point and mark are already set +;; but you want to see what the region would look like as a rectangle, +;; try `C-x r C-x' which exchanges point and mark and makes the +;; highlighted region rectangular. + +;; The default Emacs key-bindings put `point-to-register' on +;; `C-x r C-SPC' but since that command it is already on `C-x r SPC' +;; and since it is irresistably intuitive to put `rm-set-mark' on +;; `C-x r C-SPC', I have taken the liberty of recommending that you +;; override the default key-bindings. + +;; You can also kill or copy rectangles onto the kill ring which is +;; convenient for yanking rectangles into ordinary buffers (i.e. ones +;; not in picture mode) and for pasting rectangles into other window +;; system programs (e.g. xterm). These keys are by default bound to +;; `C-x r C-w' and `C-x r M-w' by analogy to the normal kill and copy +;; counterparts. + +;; Finally, there is mouse support for rectangle highlighting by +;; dragging the mouse while holding down the shift key. The idea is +;; that this behaves exactly like normal mouse dragging except that +;; the region is treated as a rectangle. + +;;; Usage: + +;; Use this section in your "~/.emacs" when rect-mark isn't included +;; as an integral part of Emacs. Don't forget to remove the first +;; three columns. + +;; ;; Support for marking a rectangle of text with highlighting. +;; (define-key ctl-x-map "r\C-@" 'rm-set-mark) +;; (define-key ctl-x-map [?r ?\C-\ ] 'rm-set-mark) +;; (define-key ctl-x-map "r\C-x" 'rm-exchange-point-and-mark) +;; (define-key ctl-x-map "r\C-w" 'rm-kill-region) +;; (define-key ctl-x-map "r\M-w" 'rm-kill-ring-save) +;; (define-key global-map [S-down-mouse-1] 'rm-mouse-drag-region) +;; (autoload 'rm-set-mark "rect-mark" +;; "Set mark for rectangle." t) +;; (autoload 'rm-exchange-point-and-mark "rect-mark" +;; "Exchange point and mark for rectangle." t) +;; (autoload 'rm-kill-region "rect-mark" +;; "Kill a rectangular region and save it in the kill ring." t) +;; (autoload 'rm-kill-ring-save "rect-mark" +;; "Copy a rectangular region to the kill ring." t) +;; (autoload 'rm-mouse-drag-region "rect-mark" +;; "Drag out a rectangular region with the mouse." t) + +;; Use this section in your "~/.emacs" to modify picture mode so that +;; it automatically uses the rect-mark equivalents of many commands. + +;; ;; One vision of a better picture mode. +;; (add-hook 'picture-mode-hook 'rm-example-picture-mode-bindings) +;; (autoload 'rm-example-picture-mode-bindings "rect-mark" +;; "Example rect-mark key and mouse bindings for picture mode.") + +;;; Code: + + +;;;###autoload (define-key ctl-x-map "r\C-@" 'rm-set-mark) +;;;###autoload (define-key ctl-x-map [?r ?\C-\ ] 'rm-set-mark) +;;;###autoload (define-key ctl-x-map "r\C-x" 'rm-exchange-point-and-mark) +;;;###autoload (define-key ctl-x-map "r\C-w" 'rm-kill-region) +;;;###autoload (define-key ctl-x-map "r\M-w" 'rm-kill-ring-save) +;;;###autoload (define-key global-map [S-down-mouse-1] 'rm-mouse-drag-region) + +;; Our state variables, each internal and buffer local. +(defvar rm-mark-active nil) +(defvar rm-overlay-list) +(defvar rm-old-transient-mark-mode) +(defvar rm-force) +(defvar rm-old-global-variables) + +;; A list of our buffer local variables. +(defconst rm-our-local-variables + '(rm-mark-active + rm-overlay-list + rm-old-transient-mark-mode + rm-force + rm-old-global-variables)) + +;; System variables which must temorarily be buffer local. +(defconst rm-temporary-local-variables + '(transient-mark-mode + ;; Alas, we can no longer uninstall a post command hook from a post + ;; command hook (as of 19.28 at least) so we must leave it installed + ;; globally. + ;post-command-hook + deactivate-mark-hook)) + +;; Those commands which don't necessarily deactivate the mark but +;; should. This is a partial list as of Emacs 19.22. Most problems +;; are the result of the pathological case of a zero-width rectangle. +(defconst rm-deactivate-mark-commands + '(clear-rectangle + copy-rectangle + copy-rectangle-to-register + kill-rectangle + open-rectangle + string-rectangle + yank-rectangle + keyboard-quit)) + +;;; Quiet the byte-compiler. +(defvar killed-rectangle) +(defvar picture-mode-map) +(defvar deactivate-mark-hook) + + +;;;###autoload +(defun rm-example-picture-mode-bindings () + "Example rect-mark keyboard and mouse bindings for picture mode." + (define-key picture-mode-map "\C-@" 'rm-set-mark) + (define-key picture-mode-map [?\C-\ ] 'rm-set-mark) + (define-key picture-mode-map [down-mouse-1] 'rm-mouse-drag-region) + (define-key picture-mode-map "\C-x\C-x" 'rm-exchange-point-and-mark) + (define-key picture-mode-map "\C-w" 'rm-kill-region) + (define-key picture-mode-map "\M-w" 'rm-kill-ring-save) + (define-key picture-mode-map "\C-y" 'yank-rectangle) + ;; Prevent `move-to-column-force' from deactivating the mark. + (defun move-to-column-force (column) + (let ((deactivate-mark deactivate-mark)) + (move-to-column (max column 0) t) + (hscroll-point-visible)))) + +;;;###autoload +(defun rm-set-mark (force) + "Set mark like `set-mark-command' but anticipates a rectangle. +This arranges for the rectangular region between point and mark +to be highlighted using the same face that is used to highlight +the region in `transient-mark-mode'. This special state lasts only +until the mark is deactivated, usually by executing a text-modifying +command like \\[kill-rectangle], by inserting text, or by typing \\[keyboard-quit]. + +With optional argument FORCE, arrange for tabs to be expanded and +for spaces to inserted as necessary to keep the region perfectly +rectangular. This is the default in `picture-mode'." + (interactive "P") + (rm-activate-mark force) + (push-mark nil nil t)) + +;;;###autoload +(defun rm-exchange-point-and-mark (force) + "Like `exchange-point-and-mark' but treats region as a rectangle. +See `rm-set-mark' for more details. + +With optional argument FORCE, tabs are expanded and spaces are +inserted as necessary to keep the region perfectly rectangular. +This is the default in `picture-mode'." + (interactive "P") + (rm-activate-mark force) + (exchange-point-and-mark)) + +;;;###autoload +(defun rm-kill-region (start end) + "Like kill-rectangle except the rectangle is also saved in the kill ring. +Since rectangles are not ordinary text, the killed rectangle is saved +in the kill ring as a series of lines, one for each row of the rectangle. +The rectangle is also saved as the killed rectangle so it is available for +insertion with yank-rectangle." + (interactive "r") + (rm-kill-ring-save start end) + (delete-rectangle start end) + (and (interactive-p) + rm-mark-active + (rm-deactivate-mark))) + +;;;###autoload +(defun rm-kill-ring-save (start end) + "Copies the region like rm-kill-region would but the rectangle isn't killed." + (interactive "r") + (setq killed-rectangle (extract-rectangle start end)) + (kill-new (mapconcat (function + (lambda (row) + (concat row "\n"))) + killed-rectangle "")) + (and (interactive-p) + rm-mark-active + (rm-deactivate-mark))) + +;;;###autoload +(defun rm-mouse-drag-region (start-event) + "Highlight a rectangular region of text as the the mouse is dragged over it. +This must be bound to a button-down mouse event." + (interactive "e") + (let* ((start-posn (event-start start-event)) + (start-point (posn-point start-posn)) + (start-window (posn-window start-posn)) + (start-frame (window-frame start-window)) + (bounds (window-edges start-window)) + (top (nth 1 bounds)) + (bottom (if (window-minibuffer-p start-window) + (nth 3 bounds) + ;; Don't count the mode line. + (1- (nth 3 bounds)))) + (click-count (1- (event-click-count start-event)))) + (setq mouse-selection-click-count click-count) + (mouse-set-point start-event) + (rm-activate-mark) + (let (end-event + end-posn + end-point + end-window) + (track-mouse + (while (progn + (setq end-event (read-event) + end-posn (event-end end-event) + end-point (posn-point end-posn) + end-window (posn-window end-posn)) + (or (mouse-movement-p end-event) + (eq (car-safe end-event) 'switch-frame))) + (cond + ;; Ignore switch-frame events. + ((eq (car-safe end-event) 'switch-frame) + nil) + ;; Are we moving within the original window? + ((and (eq end-window start-window) + (integer-or-marker-p end-point)) + (goto-char end-point) + (rm-highlight-rectangle start-point end-point)) + ;; Are we moving on a different window on the same frame? + ((and (windowp end-window) + (eq (window-frame end-window) start-frame)) + (let ((mouse-row (+ (nth 1 (window-edges end-window)) + (cdr (posn-col-row end-posn))))) + (cond + ((< mouse-row top) + (mouse-scroll-subr (- mouse-row top) + nil start-point)) + ((and (not (eobp)) + (>= mouse-row bottom)) + (mouse-scroll-subr (1+ (- mouse-row bottom)) + nil start-point))))) + (t + (let ((mouse-y (cdr (cdr (mouse-position)))) + (menu-bar-lines (or (cdr (assq 'menu-bar-lines + (frame-parameters))) + 0))) + ;; Are we on the menu bar? + (and (integerp mouse-y) (< mouse-y menu-bar-lines) + (mouse-scroll-subr (- mouse-y menu-bar-lines) + nil start-point))))))) + (and (eq (get (event-basic-type end-event) 'event-kind) 'mouse-click) + (eq end-window start-window) + (numberp end-point) + (if (= start-point end-point) + (setq deactivate-mark t) + (push-mark start-point t t) + (goto-char end-point) + (rm-kill-ring-save start-point end-point))) + ))) + + +(defun rm-activate-mark (&optional force) + ;; Turn on rectangular marking mode by temporarily (and in a buffer + ;; local way) disabling transient mark mode and manually handling + ;; highlighting from a post command hook. + (setq rm-force (and (not buffer-read-only) + (or force + (eq major-mode 'picture-mode)))) + ;; Be careful if we are already marking a rectangle. + (if rm-mark-active + nil + ;; Make each of our state variables buffer local. + (mapcar (function make-local-variable) rm-our-local-variables) + (setq rm-mark-active t + rm-overlay-list nil + rm-old-transient-mark-mode transient-mark-mode) + ;; Remember which system variables weren't buffer local. + (setq rm-old-global-variables + (apply (function nconc) + (mapcar (function + (lambda (variable) + (and (not (assoc variable + (buffer-local-variables))) + (list variable)))) + rm-temporary-local-variables))) + ;; Then make them all buffer local too. + (mapcar (function make-local-variable) rm-temporary-local-variables) + ;; Making transient-mark-mode buffer local doesn't really work + ;; correctly as of 19.22: the current buffer's value affects all + ;; displayed buffers. + (setq transient-mark-mode nil) + (add-hook 'post-command-hook 'rm-post-command) + (add-hook 'deactivate-mark-hook 'rm-deactivate-mark))) + +(defun rm-post-command () + ;; An error in a post-command function can be fatal if it re-occurs + ;; on each call, thus the condition-case safety nets. + ;; We have to do things this way because deactivate-mark doesn't + ;; (in general) get called if transient-mark-mode isn't turned on. + (if rm-mark-active + (if (or (not mark-active) + deactivate-mark + (memq this-command rm-deactivate-mark-commands)) + (condition-case nil + (rm-deactivate-mark) + (error nil)) + (condition-case info + (rm-highlight-rectangle (mark) (point)) + (error + (ding) + (message "rect-mark trouble: %s" info) + (condition-case nil + (rm-deactivate-mark) + (error nil))))) + (and (boundp 'rm-overlay-list) + (condition-case nil + (rm-deactivate-mark) + (error nil))))) + +(defun rm-highlight-rectangle (start end) + ;; This function is used to highlight the rectangular region from + ;; START to END. We do this by putting an overlay on each line + ;; within the rectangle. Each overlay extends across all the + ;; columns of the rectangle. We try to reuse overlays where + ;; possible because this is more efficient and results in less + ;; flicker. If rm-force is nil and the buffer contains tabs or + ;; short lines, the higlighted region may not be perfectly + ;; rectangular. + (save-excursion + ;; Calculate the rectangular region represented by point and mark, + ;; putting start in the north-west corner and end in the + ;; south-east corner. We can't effectively use + ;; operate-on-rectangle because it doesn't work for zero-width + ;; rectangles as of 19.22. + (and (> start end) + (setq start (prog1 + end + (setq end start)))) + (let ((start-col (save-excursion + (goto-char start) + (current-column))) + (end-col (save-excursion + (goto-char end) + (current-column))) + (deactivate-mark deactivate-mark)) + (and (> start-col end-col) + (setq start-col (prog1 + end-col + (setq end-col start-col)) + start (save-excursion + (goto-char start) + (move-to-column start-col rm-force) + (point)) + end (save-excursion + (goto-char end) + (move-to-column end-col rm-force) + (point)))) + ;; Force a redisplay so we can do reliable window start/end + ;; calculations. + (sit-for 0) + (let ((old rm-overlay-list) + (new nil) + overlay + (window-start (max (window-start) start)) + (window-end (min (window-end) end))) + ;; Iterate over those lines of the rectangle which are visible + ;; in the currently selected window. + (goto-char window-start) + (while (< (point) window-end) + (let ((row-start (progn + (move-to-column start-col rm-force) + (point))) + (row-end (progn + (move-to-column end-col rm-force) + (point)))) + ;; Trim old leading overlays. + (while (and old + (setq overlay (car old)) + (< (overlay-start overlay) row-start) + (/= (overlay-end overlay) row-end)) + (delete-overlay overlay) + (setq old (cdr old))) + ;; Reuse an overlay if possible, otherwise create one. + (if (and old + (setq overlay (car old)) + (or (= (overlay-start overlay) row-start) + (= (overlay-end overlay) row-end))) + (progn + (move-overlay overlay row-start row-end) + (setq new (cons overlay new) + old (cdr old))) + (setq overlay (make-overlay row-start row-end)) + (overlay-put overlay 'face 'region) + (setq new (cons overlay new))) + (forward-line 1))) + ;; Trim old trailing overlays. + (mapcar (function delete-overlay) old) + (setq rm-overlay-list (nreverse new)))))) + +(defun rm-deactivate-mark () + ;; This is used to clean up after `rm-activate-mark'. + ;; Alas, we can no longer uninstall a post command hook from a post + ;; command hook (as of 19.28 at least) so we must leave it installed + ;; globally. + ;(setq post-command-hook (delq 'rm-post-command post-command-hook)) + (setq deactivate-mark-hook (delq 'rm-deactivate-mark deactivate-mark-hook)) + (setq transient-mark-mode rm-old-transient-mark-mode) + (mapcar (function delete-overlay) rm-overlay-list) + (mapcar (function kill-local-variable) rm-old-global-variables) + (mapcar (function kill-local-variable) rm-our-local-variables) + (and transient-mark-mode + mark-active + (deactivate-mark))) + +(provide 'rect-mark) + +;;; rect-mark.el ends here |