You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
370 lines
12 KiB
370 lines
12 KiB
9 years ago
|
;;; wrap-region.el --- Wrap text with punctation or tag
|
||
|
|
||
|
;; Copyright (C) 2008-2012 Johan Andersson
|
||
|
|
||
|
;; Author: Johan Andersson <johan.rejeep@gmail.com>
|
||
|
;; Maintainer: Johan Andersson <johan.rejeep@gmail.com>
|
||
|
;; Version: 0.7.3
|
||
|
;; Package-Version: 20140116.2320
|
||
|
;; Keywords: speed, convenience
|
||
|
;; URL: http://github.com/rejeep/wrap-region
|
||
|
;; Package-Requires: ((dash "1.0.3"))
|
||
|
|
||
|
;; This file is NOT part of GNU Emacs.
|
||
|
|
||
|
|
||
|
;;; License:
|
||
|
|
||
|
;; 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, 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 GNU Emacs; see the file COPYING. If not, write to the
|
||
|
;; Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
|
||
|
;; Boston, MA 02110-1301, USA.
|
||
|
|
||
|
|
||
|
;;; Commentary:
|
||
|
|
||
|
;; wrap-region is a minor mode that wraps a region with
|
||
|
;; punctuations. For tagged markup modes, such as HTML and XML, it
|
||
|
;; wraps with tags.
|
||
|
;;
|
||
|
;; To use wrap-region, make sure that this file is in Emacs load-path:
|
||
|
;; (add-to-list 'load-path "/path/to/directory/or/file")
|
||
|
;;
|
||
|
;; Then require wrap-region:
|
||
|
;; (require 'wrap-region)
|
||
|
|
||
|
;; To start wrap-region:
|
||
|
;; (wrap-region-mode t) or M-x wrap-region-mode
|
||
|
;;
|
||
|
;; If you only want wrap-region active in some mode, use hooks:
|
||
|
;; (add-hook 'ruby-mode-hook 'wrap-region-mode)
|
||
|
;;
|
||
|
;; Or if you want to activate it in all buffers, use the global mode:
|
||
|
;; (wrap-region-global-mode t)
|
||
|
|
||
|
;; To wrap a region, select that region and hit one of the punctuation
|
||
|
;; keys. In "tag-modes"" (see `wrap-region-tag-active-modes'), "<" is
|
||
|
;; replaced and wraps the region with a tag. To activate this behavior
|
||
|
;; in a mode that is not default:
|
||
|
;; (add-to-list 'wrap-region-tag-active-modes 'some-tag-mode)
|
||
|
;;
|
||
|
;; `wrap-region-table' contains the default punctuations
|
||
|
;; that wraps. You can add and remove new wrappers by using the
|
||
|
;; functions `wrap-region-add-wrapper' and
|
||
|
;; `wrap-region-remove-wrapper' respectively.
|
||
|
;; (wrap-region-add-wrapper "`" "'") ; hit ` then region -> `region'
|
||
|
;; (wrap-region-add-wrapper "/*" "*/" "/") ; hit / then region -> /*region*/
|
||
|
;; (wrap-region-add-wrapper "$" "$" nil 'latex-mode) ; hit $ then region -> $region$ in latex-mode
|
||
|
;; (wrap-region-remove-wrapper "(")
|
||
|
;; (wrap-region-remove-wrapper "$" 'latex-mode)
|
||
|
;;
|
||
|
;; Some modes may have conflicting key bindings with wrap-region. To
|
||
|
;; avoid conflicts, the list `wrap-region-except-modes' contains names
|
||
|
;; of modes where wrap-region should not be activated (note, only in
|
||
|
;; the global mode). You can add new modes like this:
|
||
|
;; (add-to-list 'wrap-region-except-modes 'conflicting-mode)
|
||
|
|
||
|
|
||
|
;;; Code:
|
||
|
|
||
|
(require 'edmacro)
|
||
|
(require 'dash)
|
||
|
(eval-when-compile
|
||
|
(require 'cl))
|
||
|
|
||
|
(defstruct wrap-region-wrapper key left right modes)
|
||
|
|
||
|
(defgroup wrap-region nil
|
||
|
"Wrap region with delimiters."
|
||
|
:group 'editing
|
||
|
:link '(url-link :tag "Github" "https://github.com/rejeep/wrap-region"))
|
||
|
|
||
|
(defcustom wrap-region-except-modes '(calc-mode dired-mode)
|
||
|
"Major modes which should not use `wrap-region-mode'."
|
||
|
:group 'wrap-region
|
||
|
:type '(repeat (symbol :tag "Major mode")))
|
||
|
|
||
|
(defcustom wrap-region-tag-active-modes
|
||
|
'(html-mode sgml-mode rhtml-mode nxml-mode nxhtml-mode handlebars-mode web-mode)
|
||
|
"Major modes that use tags."
|
||
|
:group 'wrap-region
|
||
|
:type '(repeat (symbol :tag "Major mode")))
|
||
|
|
||
|
(define-obsolete-variable-alias 'wrap-region-hook 'wrap-region-mode-hook "0.8")
|
||
|
(defcustom wrap-region-mode-hook nil
|
||
|
"Functions to run after `wrap-region-mode' is enabled.
|
||
|
|
||
|
This variable is a normal hook."
|
||
|
:group 'wrap-region
|
||
|
:type 'hook)
|
||
|
|
||
|
(defcustom wrap-region-before-wrap-hook nil
|
||
|
"Functions to run before wrapping.
|
||
|
|
||
|
This variable is a normal hook."
|
||
|
:group 'wrap-region
|
||
|
:type 'hook)
|
||
|
|
||
|
(defcustom wrap-region-after-wrap-hook nil
|
||
|
"Functions to run after wrapping.
|
||
|
|
||
|
This variable is a normal hook."
|
||
|
:group 'wrap-region
|
||
|
:type 'hook)
|
||
|
|
||
|
(defcustom wrap-region-only-with-negative-prefix nil
|
||
|
"If non-nil only wrap with negative prefix.
|
||
|
|
||
|
If this variable is not nil, only wrap the region if the trigger
|
||
|
key is given a negative prefix argument. Otherwise do not wrap.
|
||
|
|
||
|
If nil, always wrap the region."
|
||
|
:group 'wrap-region
|
||
|
:type 'boolean)
|
||
|
|
||
|
(defcustom wrap-region-keep-mark nil
|
||
|
"If non-nil, keep the wrapped region active."
|
||
|
:group 'wrap-region
|
||
|
:type 'boolean)
|
||
|
|
||
|
(defvar wrap-region-mode-map (make-sparse-keymap)
|
||
|
"Keymap for `wrap-region-mode'.")
|
||
|
|
||
|
(defvar wrap-region-table (make-hash-table :test 'equal)
|
||
|
"Table with wrapper pairs.")
|
||
|
|
||
|
(defun wrap-region-trigger (arg key)
|
||
|
"Called when trigger key is pressed."
|
||
|
(let* ((wrapper (wrap-region-find key)))
|
||
|
(if (and wrapper
|
||
|
(region-active-p)
|
||
|
(if wrap-region-only-with-negative-prefix (< arg 0) t))
|
||
|
(if (wrap-region-insert-tag-p key)
|
||
|
(wrap-region-with-tag)
|
||
|
(wrap-region-with-punctuations
|
||
|
(wrap-region-wrapper-left wrapper)
|
||
|
(wrap-region-wrapper-right wrapper)))
|
||
|
(wrap-region-fallback key))))
|
||
|
|
||
|
(defun wrap-region-find (key)
|
||
|
"Find first wrapper with trigger KEY that should be active in MAJOR-MODE."
|
||
|
(let ((wrappers (gethash key wrap-region-table)))
|
||
|
(or
|
||
|
(-first
|
||
|
(lambda (wrapper)
|
||
|
(member major-mode (wrap-region-wrapper-modes wrapper)))
|
||
|
wrappers)
|
||
|
(-first
|
||
|
(lambda (wrapper)
|
||
|
(not (wrap-region-wrapper-modes wrapper)))
|
||
|
wrappers))))
|
||
|
|
||
|
(defun wrap-region-insert-tag-p (key)
|
||
|
"Check if tag should be inserted or not."
|
||
|
(and
|
||
|
(equal key "<")
|
||
|
(member major-mode wrap-region-tag-active-modes)))
|
||
|
|
||
|
(defun wrap-region-with-tag ()
|
||
|
"Wraps region with tag."
|
||
|
(let* ((tag (read-string "Enter Tag (with optional attributes): "))
|
||
|
(split (split-string tag " "))
|
||
|
(tag-name (car split))
|
||
|
(left (concat "<" tag ">"))
|
||
|
(right (concat "</" tag-name ">")))
|
||
|
(wrap-region-with left right)))
|
||
|
|
||
|
(defun wrap-region-with-punctuations (left right)
|
||
|
"Wraps region with LEFT and RIGHT punctuations."
|
||
|
(wrap-region-with left right))
|
||
|
|
||
|
(defun wrap-region-with (left right)
|
||
|
"Wraps region with LEFT and RIGHT."
|
||
|
(run-hooks 'wrap-region-before-wrap-hook)
|
||
|
(let ((beg (region-beginning))
|
||
|
(end (region-end))
|
||
|
(pos (point))
|
||
|
(deactivate-mark nil))
|
||
|
(save-excursion
|
||
|
(goto-char beg)
|
||
|
(insert left)
|
||
|
(goto-char (+ end (length left)))
|
||
|
(insert right))
|
||
|
(if (= pos end) (forward-char 1))
|
||
|
(if wrap-region-keep-mark
|
||
|
(let* ((beg-p (eq beg pos))
|
||
|
(beg* (+ beg (length left)))
|
||
|
(end* (+ end (length left))))
|
||
|
(push-mark (if beg-p end* beg*) nil t)
|
||
|
(goto-char (if beg-p beg* end*)))
|
||
|
(deactivate-mark)))
|
||
|
(run-hooks 'wrap-region-after-wrap-hook))
|
||
|
|
||
|
(defun wrap-region-fallback (key)
|
||
|
"Execute function that KEY was bound to before `wrap-region-mode'."
|
||
|
(let ((wrap-region-mode nil))
|
||
|
(call-interactively (key-binding key))))
|
||
|
|
||
|
(defun wrap-region-add-wrappers (wrappers)
|
||
|
"Add WRAPPERS by calling `wrap-region-add-wrapper' for each one."
|
||
|
(mapc
|
||
|
(lambda (wrapper)
|
||
|
(apply 'wrap-region-add-wrapper wrapper))
|
||
|
wrappers))
|
||
|
|
||
|
(defun wrap-region-add-wrapper (left right &optional key mode-or-modes)
|
||
|
"Add new LEFT and RIGHT wrapper.
|
||
|
|
||
|
Optional KEY is the trigger key and MODE-OR-MODES is a single
|
||
|
mode or multiple modes that the wrapper should trigger in."
|
||
|
(or key (setq key left))
|
||
|
(let ((wrappers (gethash key wrap-region-table))
|
||
|
(modes
|
||
|
(if mode-or-modes
|
||
|
(if (listp mode-or-modes)
|
||
|
mode-or-modes
|
||
|
(list mode-or-modes)))))
|
||
|
(if wrappers
|
||
|
(let ((wrapper-exactly-same
|
||
|
(-first
|
||
|
(lambda (wrapper)
|
||
|
(and
|
||
|
(equal (wrap-region-wrapper-key wrapper) key)
|
||
|
(equal (wrap-region-wrapper-left wrapper) left)
|
||
|
(equal (wrap-region-wrapper-right wrapper) right)))
|
||
|
wrappers)))
|
||
|
(if wrapper-exactly-same
|
||
|
(when (wrap-region-wrapper-modes wrapper-exactly-same)
|
||
|
(if modes
|
||
|
(setf
|
||
|
(wrap-region-wrapper-modes wrapper-exactly-same)
|
||
|
(-union modes (wrap-region-wrapper-modes wrapper-exactly-same)))
|
||
|
(let ((new-wrapper (make-wrap-region-wrapper :key key :left left :right right)))
|
||
|
(puthash key (cons new-wrapper wrappers) wrap-region-table))))
|
||
|
(let* ((new-wrapper (make-wrap-region-wrapper :key key :left left :right right :modes modes))
|
||
|
(wrapper-same-trigger
|
||
|
(-first
|
||
|
(lambda (wrapper)
|
||
|
(equal (wrap-region-wrapper-key wrapper) key))
|
||
|
wrappers))
|
||
|
(wrapper-same-trigger-modes
|
||
|
(wrap-region-wrapper-modes wrapper-same-trigger)))
|
||
|
(when (and wrapper-same-trigger wrapper-same-trigger-modes)
|
||
|
(let ((new-modes (-difference (wrap-region-wrapper-modes wrapper-same-trigger) modes)))
|
||
|
(if new-modes
|
||
|
(setf (wrap-region-wrapper-modes wrapper-same-trigger) new-modes)
|
||
|
(setq wrappers (delete wrapper-same-trigger wrappers)))))
|
||
|
(puthash key (cons new-wrapper wrappers) wrap-region-table))))
|
||
|
(let ((new-wrapper (make-wrap-region-wrapper :key key :left left :right right :modes modes)))
|
||
|
(puthash key (list new-wrapper) wrap-region-table))))
|
||
|
(wrap-region-define-trigger key))
|
||
|
|
||
|
(defun wrap-region-remove-wrapper (key &optional mode-or-modes)
|
||
|
"Remove wrapper with trigger KEY or exclude from MODE-OR-MODES.
|
||
|
|
||
|
If MODE-OR-MODES is not present, all wrappers for KEY are removed."
|
||
|
(if mode-or-modes
|
||
|
(let ((wrappers (gethash key wrap-region-table))
|
||
|
(modes
|
||
|
(if mode-or-modes
|
||
|
(if (listp mode-or-modes)
|
||
|
mode-or-modes
|
||
|
(list mode-or-modes)))))
|
||
|
(mapc
|
||
|
(lambda (mode)
|
||
|
(let ((wrapper-including-mode
|
||
|
(-first
|
||
|
(lambda (wrapper)
|
||
|
(member mode (wrap-region-wrapper-modes wrapper)))
|
||
|
wrappers)))
|
||
|
(when wrapper-including-mode
|
||
|
(let ((new-modes (delete mode (wrap-region-wrapper-modes wrapper-including-mode))))
|
||
|
(if new-modes
|
||
|
(setf (wrap-region-wrapper-modes wrapper-including-mode) new-modes)
|
||
|
(puthash key (delete wrapper-including-mode wrappers) wrap-region-table))))))
|
||
|
modes))
|
||
|
(wrap-region-destroy-wrapper key)))
|
||
|
|
||
|
(defun wrap-region-destroy-wrapper (key)
|
||
|
"Remove the wrapper bound to KEY, no questions asked."
|
||
|
(remhash key wrap-region-table)
|
||
|
(wrap-region-unset-key key))
|
||
|
|
||
|
(defun wrap-region-define-wrappers ()
|
||
|
"Defines defaults wrappers."
|
||
|
(mapc
|
||
|
(lambda (pair)
|
||
|
(apply 'wrap-region-add-wrapper pair))
|
||
|
'(("\"" "\"")
|
||
|
("'" "'")
|
||
|
("(" ")")
|
||
|
("{" "}")
|
||
|
("[" "]")
|
||
|
("<" ">"))))
|
||
|
|
||
|
(defun wrap-region-define-trigger (key)
|
||
|
"Defines KEY as wrapper."
|
||
|
(wrap-region-define-key
|
||
|
key
|
||
|
`(lambda (arg)
|
||
|
(interactive "p")
|
||
|
(wrap-region-trigger arg ,key))))
|
||
|
|
||
|
(defun wrap-region-unset-key (key)
|
||
|
"Remove KEY from `wrap-region-mode-map'."
|
||
|
(wrap-region-define-key key))
|
||
|
|
||
|
(defun wrap-region-define-key (key &optional fn)
|
||
|
"Binds KEY to FN in `wrap-region-mode-map'."
|
||
|
(define-key wrap-region-mode-map (read-kbd-macro key) fn))
|
||
|
|
||
|
|
||
|
;;;###autoload
|
||
|
(define-minor-mode wrap-region-mode
|
||
|
"Wrap region with stuff."
|
||
|
:init-value nil
|
||
|
:lighter " wr"
|
||
|
:keymap wrap-region-mode-map
|
||
|
:group 'wrap-region
|
||
|
:require 'wrap-region
|
||
|
(when wrap-region-mode
|
||
|
(wrap-region-define-wrappers)))
|
||
|
|
||
|
;;;###autoload
|
||
|
(defun turn-on-wrap-region-mode ()
|
||
|
"Turn on `wrap-region-mode'."
|
||
|
(interactive)
|
||
|
(unless (member major-mode wrap-region-except-modes)
|
||
|
(wrap-region-mode +1)))
|
||
|
|
||
|
;;;###autoload
|
||
|
(defun turn-off-wrap-region-mode ()
|
||
|
"Turn off `wrap-region-mode'."
|
||
|
(interactive)
|
||
|
(wrap-region-mode -1))
|
||
|
|
||
|
;;;###autoload
|
||
|
(define-globalized-minor-mode wrap-region-global-mode
|
||
|
wrap-region-mode
|
||
|
turn-on-wrap-region-mode
|
||
|
:group 'wrap-region
|
||
|
:require 'wrap-region)
|
||
|
|
||
|
(provide 'wrap-region)
|
||
|
|
||
|
;; Local Variables:
|
||
|
;; byte-compile-warnings: (not cl-functions)
|
||
|
;; End:
|
||
|
|
||
|
;;; wrap-region.el ends here
|