2016-02-16 09:14:05 +01:00

194 lines
6.6 KiB
EmacsLisp

;;; xlicense.el --- Insert a pre-defined license text
;; Copyright (C) 2010 Seong-Kook Shin
;; Author: Seong-Kook Shin <cinsky@gmail.com>
;; Keywords: abbrev, convenience
;; 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:
;; $Id: xlicense.el,v 1.1 2010/12/16 08:26:26 cinsk Exp $
;;; Code:
(require 'skeleton)
;;(eval-when-compile (require 'cl))
(defvar license-directory (concat
(file-name-directory load-file-name)
"licenses")
"Directory for license templates")
(defvar license-types '((gpl . "GPL-2.0")
(gpl2 . "GPL-2.0")
(gpl3 . "GPL-3.0")
(lgpl . "LGPL")
(apache . "APACHE-2.0")
(boost . "BOOST")
(bsd-old . "BSD-old")
(bsd-new . "BSD-new")
(bsd . "BSD-new")
(freebsd . "FREEBSD")
(mit . "MIT")
;(tmp . "TMP")
)
"Alist of licenses. CAR of each item is a symbol represents the license,
CDR of each item is a filename of the license template")
(defvar license-default-summary
"Description: "
"Short description of what it does.")
(defvar license-eol-text "!@#$EOL!@#$"
"Text to mark blank lines -- used internally")
(defun license-file (type)
"Return the pathname of the given license file"
(let ((tp (assoc type license-types)))
(if tp
(concat (file-name-as-directory license-directory) (cdr tp))
tp)))
(defvar license-keywords-alist '(("@author@" . user-full-name)
("@email@" . user-email-address)
("@year@" . (lambda ()
(substring (current-time-string)
-4)))
("@organization@" .
(lambda ()
(getenv "ORGANIZATION"))))
"Keywords that need to be substituted by `license-substitute-keywords'.
The CAR of an item is a keyword and CDR is a replacement. If the
CDR of an item is a function, the return value(string) is used as
a replacement. If the returned value is nil, no substitution for
that keyword.")
(defun license-substitute-keywords (&optional record)
"Substitute all occurences of keywords to their replacement and returns
the replacement positions in markers."
(let (markers)
(dolist (i license-keywords-alist)
(let ((keyword (regexp-quote (car i)))
(what (if (functionp (cdr i)) (funcall (cdr i)) (cdr i))))
(if what
(progn
(goto-char (point-min))
(while (re-search-forward keyword nil t)
(if record
(setq markers (cons (point-marker) markers)))
(replace-match what))))))
markers))
(defun license-fill-paragraphs (lst)
"Fill paragraphs at markers in LST."
(dolist (i lst)
(goto-char i)
(fill-paragraph)))
(defun create-license (type &optional comments summary author)
"Create a license paragraphs according to current buffer's major mode.
IF COMMENTS is non-nil, comment the license text.
If SUMMARY is non-nil, it is inserted as a header of the comment.
If AUTHOR is non-nil, all occurrence of the author keyword are
replaced to AUTHOR.
See `license-keywords-alist' for keywords and their meaning."
(let (;(buffer (get-buffer-create "*LICENSE*"))
(desc (or (and summary (> (length summary) 0))
license-default-summary))
(auth (or author (user-full-name)))
(lfile (license-file type))
(mode major-mode)
(fill-points nil))
(with-temp-buffer
;; BEGIN Common
(insert "\n")
(insert (format "Author:: %s" auth))
(if user-mail-address
(insert (format " <%s>" user-mail-address)))
(insert "\n")
(insert (format "Copyright:: Copyright (c) %d, %s" (nth 5 (decode-time)) auth))
(insert "\n")
;; END Common
(insert-file-contents lfile)
(goto-char (point-min))
(while (re-search-forward "^$" nil t)
(replace-match license-eol-text))
(let ((case-fold-search t)
(markers (license-substitute-keywords t)))
(funcall mode)
(if (and comments comment-start)
(let ((comment-style 'extra-line))
(comment-region (point-min) (point-max))))
(goto-char (point-min))
(let ((re-eol (concat (regexp-quote license-eol-text) "$")))
(while (re-search-forward re-eol nil t)
(replace-match "")))
(license-fill-paragraphs markers)
;;(print markers)
;;(pop-marker)
(goto-char (point-max))
(insert "\n"))
(buffer-substring-no-properties (point-min) (point-max)))))
(defun insert-license (&optional type)
"Insert a license template into the current buffer."
(interactive)
(let ((text (create-license
(or type (intern (completing-read "Choose a license type: "
license-types nil t))) t)))
(if (called-interactively-p 'any)
(insert text)
text)))
(define-skeleton license-skeleton
"Insert a license template into the current buffer."
""
(insert-license)
"\n"
_)
(define-skeleton gpl-interactive-skeleton
"Insert an Interactive GPL banner."
""
\n > "static const char *gpl_banner[] = {" \n
> "\"" (file-name-nondirectory (file-name-sans-extension buffer-file-name))
> " version XXX, Copyright (C) "
(substring (current-time-string) -4) " " (user-full-name) "\"," \n
> "\"" (file-name-nondirectory (file-name-sans-extension buffer-file-name))
"comes with ABSOLUTELY NO WARRANTY; for details type `show w'.\"," \n
"\"This is free software, and you are welcome to redistribute it\"," \n
"\"under certain conditions; type `show c' for details.\"," \n
> > "};" \n
> _)
(provide 'xlicense)
;;; license.el ends here