2011-10-08 23:05:06 +03:00
|
|
|
|
;;; prelude-core.el --- Emacs Prelude: core Prelude defuns.
|
|
|
|
|
;;
|
2012-04-17 17:35:21 +03:00
|
|
|
|
;; Copyright (c) 2011-2012 Bozhidar Batsov
|
2011-10-08 23:05:06 +03:00
|
|
|
|
;;
|
2012-04-17 17:35:21 +03:00
|
|
|
|
;; Author: Bozhidar Batsov <bozhidar@batsov.com>
|
|
|
|
|
;; URL: http://batsov.com/emacs-prelude
|
2011-10-08 23:05:06 +03:00
|
|
|
|
;; Version: 1.0.0
|
|
|
|
|
;; Keywords: convenience
|
|
|
|
|
|
|
|
|
|
;; This file is not part of GNU Emacs.
|
|
|
|
|
|
|
|
|
|
;;; Commentary:
|
|
|
|
|
|
|
|
|
|
;; Here are the definitions of most of the functions added by Prelude.
|
|
|
|
|
|
|
|
|
|
;;; 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
|
|
|
|
|
;; 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 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.
|
|
|
|
|
|
|
|
|
|
;;; Code:
|
|
|
|
|
|
|
|
|
|
(require 'thingatpt)
|
|
|
|
|
|
|
|
|
|
(defun prelude-open-with ()
|
2011-10-10 17:29:52 +03:00
|
|
|
|
"Simple function that allows us to open the underlying
|
2011-10-08 23:05:06 +03:00
|
|
|
|
file of a buffer in an external program."
|
|
|
|
|
(interactive)
|
|
|
|
|
(when buffer-file-name
|
|
|
|
|
(shell-command (concat
|
2011-11-01 18:48:06 +02:00
|
|
|
|
(if (eq system-type 'darwin)
|
|
|
|
|
"open"
|
|
|
|
|
(read-shell-command "Open current file with: "))
|
2011-10-08 23:05:06 +03:00
|
|
|
|
" "
|
|
|
|
|
buffer-file-name))))
|
|
|
|
|
|
|
|
|
|
(defun prelude-buffer-mode (buffer-or-name)
|
|
|
|
|
(with-current-buffer buffer-or-name major-mode))
|
|
|
|
|
|
|
|
|
|
(defun prelude-visit-term-buffer ()
|
|
|
|
|
(interactive)
|
|
|
|
|
(if (not (get-buffer "*ansi-term*"))
|
2012-06-16 00:15:09 -06:00
|
|
|
|
(ansi-term (getenv "SHELL"))
|
2011-10-08 23:05:06 +03:00
|
|
|
|
(switch-to-buffer "*ansi-term*")))
|
|
|
|
|
|
2011-10-10 17:29:52 +03:00
|
|
|
|
(defun prelude-google ()
|
|
|
|
|
"Googles a query or region if any."
|
|
|
|
|
(interactive)
|
|
|
|
|
(browse-url
|
|
|
|
|
(concat
|
|
|
|
|
"http://www.google.com/search?ie=utf-8&oe=utf-8&q="
|
2012-03-28 14:40:50 +03:00
|
|
|
|
(url-hexify-string (if mark-active
|
|
|
|
|
(buffer-substring (region-beginning) (region-end))
|
|
|
|
|
(read-string "Google: "))))))
|
2011-10-10 17:29:52 +03:00
|
|
|
|
|
2011-10-08 23:05:06 +03:00
|
|
|
|
(defun prelude-indent-rigidly-and-copy-to-clipboard (begin end indent)
|
|
|
|
|
"Copy the selected code region to the clipboard, indented according
|
|
|
|
|
to Markdown blockquote rules."
|
|
|
|
|
(let ((buffer (current-buffer)))
|
|
|
|
|
(with-temp-buffer
|
|
|
|
|
(insert-buffer-substring-no-properties buffer begin end)
|
|
|
|
|
(indent-rigidly (point-min) (point-max) indent)
|
|
|
|
|
(clipboard-kill-ring-save (point-min) (point-max)))))
|
|
|
|
|
|
|
|
|
|
(defun prelude-indent-blockquote-and-copy-to-clipboard (begin end)
|
|
|
|
|
"Copy the selected code region to the clipboard, indented according
|
|
|
|
|
to markdown blockquote rules (useful to copy snippets to StackOverflow, Assembla, Github."
|
|
|
|
|
(interactive "r")
|
|
|
|
|
(prelude-indent-rigidly-and-copy-to-clipboard begin end 4))
|
|
|
|
|
|
|
|
|
|
(defun prelude-indent-nested-blockquote-and-copy-to-clipboard (begin end)
|
|
|
|
|
"Copy the selected code region to the clipboard, indented according
|
|
|
|
|
to markdown blockquote rules. Useful to add snippets under bullet points."
|
|
|
|
|
(interactive "r")
|
2011-10-10 17:50:23 +03:00
|
|
|
|
(prelude-indent-rigidly-and-copy-to-clipboard begin end 6))
|
2011-10-08 23:05:06 +03:00
|
|
|
|
|
|
|
|
|
(defun prelude-insert-empty-line ()
|
|
|
|
|
"Insert an empty line after the current line and positon
|
|
|
|
|
the curson at its beginning, according to the current mode."
|
|
|
|
|
(interactive)
|
|
|
|
|
(move-end-of-line nil)
|
|
|
|
|
(open-line 1)
|
2012-08-23 10:02:44 +03:00
|
|
|
|
(forward-line 1)
|
2011-10-08 23:05:06 +03:00
|
|
|
|
(indent-according-to-mode))
|
|
|
|
|
|
|
|
|
|
(defun prelude-move-line-up ()
|
|
|
|
|
"Move up the current line."
|
|
|
|
|
(interactive)
|
|
|
|
|
(transpose-lines 1)
|
2012-08-23 10:02:44 +03:00
|
|
|
|
(forward-line -2))
|
2011-10-08 23:05:06 +03:00
|
|
|
|
|
|
|
|
|
(defun prelude-move-line-down ()
|
|
|
|
|
"Move down the current line."
|
|
|
|
|
(interactive)
|
2012-08-23 10:02:44 +03:00
|
|
|
|
(forward-line 1)
|
2011-10-08 23:05:06 +03:00
|
|
|
|
(transpose-lines 1)
|
2012-08-23 10:02:44 +03:00
|
|
|
|
(forward-line -1))
|
2011-10-08 23:05:06 +03:00
|
|
|
|
|
|
|
|
|
(defun prelude-indent-buffer ()
|
|
|
|
|
"Indents the entire buffer."
|
|
|
|
|
(interactive)
|
|
|
|
|
(indent-region (point-min) (point-max)))
|
|
|
|
|
|
|
|
|
|
(defun prelude-indent-region-or-buffer ()
|
|
|
|
|
"Indents a region if selected, otherwise the whole buffer."
|
|
|
|
|
(interactive)
|
|
|
|
|
(save-excursion
|
|
|
|
|
(if (region-active-p)
|
|
|
|
|
(progn
|
|
|
|
|
(indent-region (region-beginning) (region-end))
|
|
|
|
|
(message "Indented selected region."))
|
|
|
|
|
(progn
|
2011-10-10 17:50:23 +03:00
|
|
|
|
(prelude-indent-buffer)
|
2011-10-08 23:05:06 +03:00
|
|
|
|
(message "Indented buffer.")))))
|
|
|
|
|
|
|
|
|
|
(defun prelude-annotate-todo ()
|
|
|
|
|
"Put fringe marker on TODO: lines in the curent buffer."
|
|
|
|
|
(interactive)
|
|
|
|
|
(save-excursion
|
|
|
|
|
(goto-char (point-min))
|
|
|
|
|
(while (re-search-forward "TODO:" nil t)
|
|
|
|
|
(let ((overlay (make-overlay (- (point) 5) (point))))
|
|
|
|
|
(overlay-put overlay
|
|
|
|
|
'before-string
|
|
|
|
|
(propertize (format "A")
|
|
|
|
|
'display '(left-fringe right-triangle)))))))
|
|
|
|
|
|
|
|
|
|
(defun prelude-copy-file-name-to-clipboard ()
|
2012-02-23 16:21:19 +02:00
|
|
|
|
"Copy the current buffer file name to the clipboard."
|
2011-10-08 23:05:06 +03:00
|
|
|
|
(interactive)
|
|
|
|
|
(let ((filename (if (equal major-mode 'dired-mode)
|
|
|
|
|
default-directory
|
|
|
|
|
(buffer-file-name))))
|
|
|
|
|
(when filename
|
2012-02-23 16:21:19 +02:00
|
|
|
|
(kill-new filename)
|
|
|
|
|
(message "Copied buffer file name '%s' to the clipboard." filename))))
|
2011-10-08 23:05:06 +03:00
|
|
|
|
|
|
|
|
|
(defun prelude-duplicate-current-line-or-region (arg)
|
|
|
|
|
"Duplicates the current line or region ARG times.
|
|
|
|
|
If there's no region, the current line will be duplicated. However, if
|
|
|
|
|
there's a region, all lines that region covers will be duplicated."
|
|
|
|
|
(interactive "p")
|
|
|
|
|
(let (beg end (origin (point)))
|
|
|
|
|
(if (and mark-active (> (point) (mark)))
|
|
|
|
|
(exchange-point-and-mark))
|
|
|
|
|
(setq beg (line-beginning-position))
|
|
|
|
|
(if mark-active
|
|
|
|
|
(exchange-point-and-mark))
|
|
|
|
|
(setq end (line-end-position))
|
|
|
|
|
(let ((region (buffer-substring-no-properties beg end)))
|
2012-12-15 20:42:19 +02:00
|
|
|
|
(-dotimes arg
|
|
|
|
|
(lambda ()
|
|
|
|
|
(goto-char end)
|
|
|
|
|
(newline)
|
|
|
|
|
(insert region)
|
|
|
|
|
(setq end (point))))
|
2011-10-08 23:05:06 +03:00
|
|
|
|
(goto-char (+ origin (* (length region) arg) arg)))))
|
|
|
|
|
|
|
|
|
|
;; TODO doesn't work with uniquify
|
|
|
|
|
(defun prelude-rename-file-and-buffer ()
|
|
|
|
|
"Renames current buffer and file it is visiting."
|
|
|
|
|
(interactive)
|
|
|
|
|
(let ((name (buffer-name))
|
|
|
|
|
(filename (buffer-file-name)))
|
|
|
|
|
(if (not (and filename (file-exists-p filename)))
|
|
|
|
|
(message "Buffer '%s' is not visiting a file!" name)
|
|
|
|
|
(let ((new-name (read-file-name "New name: " filename)))
|
|
|
|
|
(cond ((get-buffer new-name)
|
|
|
|
|
(message "A buffer named '%s' already exists!" new-name))
|
|
|
|
|
(t
|
|
|
|
|
(rename-file name new-name 1)
|
|
|
|
|
(rename-buffer new-name)
|
|
|
|
|
(set-visited-file-name new-name)
|
|
|
|
|
(set-buffer-modified-p nil)))))))
|
|
|
|
|
|
|
|
|
|
(defun prelude-delete-file-and-buffer ()
|
|
|
|
|
"Kills the current buffer and deletes the file it is visiting"
|
|
|
|
|
(interactive)
|
|
|
|
|
(let ((filename (buffer-file-name)))
|
|
|
|
|
(when filename
|
|
|
|
|
(delete-file filename)
|
|
|
|
|
(message "Deleted file %s" filename)))
|
|
|
|
|
(kill-buffer))
|
|
|
|
|
|
|
|
|
|
(defun prelude-view-url ()
|
|
|
|
|
"Open a new buffer containing the contents of URL."
|
|
|
|
|
(interactive)
|
|
|
|
|
(let* ((default (thing-at-point-url-at-point))
|
|
|
|
|
(url (read-from-minibuffer "URL: " default)))
|
|
|
|
|
(switch-to-buffer (url-retrieve-synchronously url))
|
|
|
|
|
(rename-buffer url t)
|
|
|
|
|
;; TODO: switch to nxml/nxhtml mode
|
|
|
|
|
(cond ((search-forward "<?xml" nil t) (xml-mode))
|
|
|
|
|
((search-forward "<html" nil t) (html-mode)))))
|
|
|
|
|
|
|
|
|
|
(defun prelude-untabify-buffer ()
|
|
|
|
|
(interactive)
|
|
|
|
|
(untabify (point-min) (point-max)))
|
|
|
|
|
|
|
|
|
|
(defun prelude-cleanup-buffer ()
|
|
|
|
|
"Perform a bunch of operations on the whitespace content of a buffer."
|
|
|
|
|
(interactive)
|
|
|
|
|
(prelude-indent-buffer)
|
|
|
|
|
(prelude-untabify-buffer)
|
2011-11-23 22:44:49 +02:00
|
|
|
|
(whitespace-cleanup))
|
2011-10-08 23:05:06 +03:00
|
|
|
|
|
|
|
|
|
(defun prelude-eval-and-replace ()
|
|
|
|
|
"Replace the preceding sexp with its value."
|
|
|
|
|
(interactive)
|
|
|
|
|
(backward-kill-sexp)
|
|
|
|
|
(condition-case nil
|
|
|
|
|
(prin1 (eval (read (current-kill 0)))
|
|
|
|
|
(current-buffer))
|
|
|
|
|
(error (message "Invalid expression")
|
|
|
|
|
(insert (current-kill 0)))))
|
|
|
|
|
|
|
|
|
|
(defun prelude-recompile-init ()
|
|
|
|
|
"Byte-compile all your dotfiles again."
|
|
|
|
|
(interactive)
|
2012-04-17 17:47:05 +03:00
|
|
|
|
(byte-recompile-directory prelude-dir 0))
|
2011-10-08 23:05:06 +03:00
|
|
|
|
|
|
|
|
|
(defun prelude-sudo-edit (&optional arg)
|
|
|
|
|
(interactive "p")
|
|
|
|
|
(if (or arg (not buffer-file-name))
|
|
|
|
|
(find-file (concat "/sudo:root@localhost:" (ido-read-file-name "File: ")))
|
|
|
|
|
(find-alternate-file (concat "/sudo:root@localhost:" buffer-file-name))))
|
|
|
|
|
|
|
|
|
|
(defun prelude-switch-or-start (function buffer)
|
|
|
|
|
"If the buffer is current, bury it, otherwise invoke the function."
|
|
|
|
|
(if (equal (buffer-name (current-buffer)) buffer)
|
|
|
|
|
(bury-buffer)
|
|
|
|
|
(if (get-buffer buffer)
|
|
|
|
|
(switch-to-buffer buffer)
|
|
|
|
|
(funcall function))))
|
|
|
|
|
|
|
|
|
|
(defun prelude-insert-date ()
|
|
|
|
|
"Insert a time-stamp according to locale's date and time format."
|
|
|
|
|
(interactive)
|
|
|
|
|
(insert (format-time-string "%c" (current-time))))
|
|
|
|
|
|
|
|
|
|
(defun prelude-conditionally-enable-paredit-mode ()
|
|
|
|
|
"Enable paredit-mode in the minibuffer, during eval-expression."
|
|
|
|
|
(if (eq this-command 'eval-expression)
|
|
|
|
|
(paredit-mode 1)))
|
|
|
|
|
|
|
|
|
|
(add-hook 'minibuffer-setup-hook 'prelude-conditionally-enable-paredit-mode)
|
|
|
|
|
|
|
|
|
|
(defun prelude-recentf-ido-find-file ()
|
|
|
|
|
"Find a recent file using ido."
|
|
|
|
|
(interactive)
|
|
|
|
|
(let ((file (ido-completing-read "Choose recent file: " recentf-list nil t)))
|
|
|
|
|
(when file
|
|
|
|
|
(find-file file))))
|
|
|
|
|
|
|
|
|
|
(defun prelude-swap-windows ()
|
|
|
|
|
"If you have 2 windows, it swaps them."
|
|
|
|
|
(interactive)
|
2011-10-10 17:50:23 +03:00
|
|
|
|
(if (/= (count-windows) 2)
|
|
|
|
|
(message "You need exactly 2 windows to do this.")
|
2012-12-17 17:43:35 +02:00
|
|
|
|
(let* ((w1 (car (window-list)))
|
|
|
|
|
(w2 (cadr (window-list)))
|
2011-10-10 17:50:23 +03:00
|
|
|
|
(b1 (window-buffer w1))
|
|
|
|
|
(b2 (window-buffer w2))
|
|
|
|
|
(s1 (window-start w1))
|
|
|
|
|
(s2 (window-start w2)))
|
|
|
|
|
(set-window-buffer w1 b2)
|
|
|
|
|
(set-window-buffer w2 b1)
|
|
|
|
|
(set-window-start w1 s2)
|
|
|
|
|
(set-window-start w2 s1)))
|
2011-10-08 23:05:06 +03:00
|
|
|
|
(other-window 1))
|
|
|
|
|
|
2012-01-26 14:44:02 +02:00
|
|
|
|
(defun prelude-kill-other-buffers ()
|
2012-01-26 16:57:51 +02:00
|
|
|
|
"Kill all buffers but the current one. Doesn't mess with special buffers."
|
2012-01-26 14:44:02 +02:00
|
|
|
|
(interactive)
|
2012-12-15 20:42:19 +02:00
|
|
|
|
(-each
|
|
|
|
|
(->> (buffer-list)
|
|
|
|
|
(-filter #'buffer-file-name)
|
|
|
|
|
(--remove (eql (current-buffer) it)))
|
|
|
|
|
#'kill-buffer))
|
2012-01-26 14:44:02 +02:00
|
|
|
|
|
2012-03-09 16:20:46 +02:00
|
|
|
|
(require 'repeat)
|
|
|
|
|
|
|
|
|
|
(defun make-repeatable-command (cmd)
|
|
|
|
|
"Returns a new command that is a repeatable version of CMD.
|
|
|
|
|
The new command is named CMD-repeat. CMD should be a quoted
|
|
|
|
|
command.
|
|
|
|
|
|
2012-09-06 22:42:07 +03:00
|
|
|
|
This allows you to bind the command to a compound keystroke andб
|
2012-03-09 16:20:46 +02:00
|
|
|
|
repeat it with just the final key. For example:
|
|
|
|
|
|
|
|
|
|
(global-set-key (kbd \"C-c a\") (make-repeatable-command 'foo))
|
|
|
|
|
|
|
|
|
|
will create a new command called foo-repeat. Typing C-c a will
|
|
|
|
|
just invoke foo. Typing C-c a a a will invoke foo three times,
|
|
|
|
|
and so on."
|
|
|
|
|
(fset (intern (concat (symbol-name cmd) "-repeat"))
|
|
|
|
|
`(lambda ,(help-function-arglist cmd) ;; arg list
|
2012-09-06 22:42:07 +03:00
|
|
|
|
,(format "A repeatable version of `%s'."
|
|
|
|
|
(symbol-name cmd)) ;; doc string
|
2012-03-09 16:20:46 +02:00
|
|
|
|
,(interactive-form cmd) ;; interactive form
|
|
|
|
|
;; see also repeat-message-function
|
|
|
|
|
(setq last-repeatable-command ',cmd)
|
|
|
|
|
(repeat nil)))
|
|
|
|
|
(intern (concat (symbol-name cmd) "-repeat")))
|
|
|
|
|
|
2012-09-06 14:09:47 +03:00
|
|
|
|
(defun prelude-create-scratch-buffer ()
|
|
|
|
|
"Create a new scratch buffer."
|
|
|
|
|
(interactive)
|
|
|
|
|
(progn
|
2012-09-06 22:42:07 +03:00
|
|
|
|
(switch-to-buffer
|
|
|
|
|
(get-buffer-create (generate-new-buffer-name "*scratch*")))
|
2012-09-06 14:09:47 +03:00
|
|
|
|
(emacs-lisp-mode)))
|
|
|
|
|
|
2012-05-07 18:04:03 +03:00
|
|
|
|
(defvar prelude-tips
|
|
|
|
|
'("Press <C-c o> to open a file with external program."
|
|
|
|
|
"Press <C-c p f> to navigate a project's files with ido."
|
|
|
|
|
"Press <C-c h> to navigate a project in Helm."
|
|
|
|
|
"Press <C-c g> to search in Google."
|
|
|
|
|
"Press <C-c r> to rename the current buffer and file it's visiting."
|
|
|
|
|
"Press <C-c t> to open a terminal in Emacs."
|
2012-05-07 18:06:32 +03:00
|
|
|
|
"Explore the Prelude menu to find out about some of Prelude extensions to Emacs."
|
|
|
|
|
"Access the official Emacs manual by pressing <C-h r>."
|
|
|
|
|
"Visit WikEmacs at http://wikemacs.org to find out even more about Emacs."))
|
2012-05-07 18:04:03 +03:00
|
|
|
|
|
|
|
|
|
(defun prelude-tip-of-the-day ()
|
|
|
|
|
(interactive)
|
2012-09-06 22:42:07 +03:00
|
|
|
|
(message
|
|
|
|
|
(concat "Prelude tip: " (nth (random (length prelude-tips)) prelude-tips))))
|
2012-05-07 18:04:03 +03:00
|
|
|
|
|
2012-05-16 14:50:30 +03:00
|
|
|
|
(defun prelude-eval-after-init (form)
|
|
|
|
|
"Add `(lambda () FORM)' to `after-init-hook'.
|
|
|
|
|
|
|
|
|
|
If Emacs has already finished initialization, also eval FORM immediately."
|
|
|
|
|
(let ((func (list 'lambda nil form)))
|
|
|
|
|
(add-hook 'after-init-hook func)
|
|
|
|
|
(when after-init-time
|
|
|
|
|
(eval form))))
|
|
|
|
|
|
2012-09-06 22:42:07 +03:00
|
|
|
|
(defun prelude-exchange-point-and-mark ()
|
|
|
|
|
"Identical to `exchange-point-and-mark' but will not activate the region."
|
|
|
|
|
(interactive)
|
|
|
|
|
(exchange-point-and-mark)
|
|
|
|
|
(deactivate-mark nil))
|
|
|
|
|
|
2011-10-08 23:05:06 +03:00
|
|
|
|
(provide 'prelude-core)
|
|
|
|
|
;;; prelude-core.el ends here
|