org-bibtex.el 8.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204
  1. ;;; org-bibtex.el --- Org links to BibTeX entries
  2. ;;
  3. ;; Copyright 2007, 2008 Free Software Foundation, Inc.
  4. ;;
  5. ;; Author: Bastien Guerry <bzg at altern dot org>
  6. ;; Carsten Dominik <carsten dot dominik at gmail dot com>
  7. ;; Keywords: org, wp, remember
  8. ;; Version: 6.00pre-4
  9. ;;
  10. ;; This file is part of GNU Emacs.
  11. ;;
  12. ;; GNU Emacs is free software; you can redistribute it and/or modify
  13. ;; it under the terms of the GNU General Public License as published by
  14. ;; the Free Software Foundation; either version 3, or (at your option)
  15. ;; any later version.
  16. ;; GNU Emacs is distributed in the hope that it will be useful,
  17. ;; but WITHOUT ANY WARRANTY; without even the implied warranty of
  18. ;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  19. ;; GNU General Public License for more details.
  20. ;; You should have received a copy of the GNU General Public License
  21. ;; along with GNU Emacs; see the file COPYING. If not, write to the
  22. ;; Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
  23. ;; Boston, MA 02110-1301, USA.
  24. ;;
  25. ;;; Commentary:
  26. ;;
  27. ;; This file implements links to database entries in BibTeX files.
  28. ;; Instead of defining a special link prefix, it uses the normal file
  29. ;; links combined with a custom search mechanism to find entries
  30. ;; by reference key. And it constucts a nice description tag for
  31. ;; the link that contains the author name, the year and a short title.
  32. ;;
  33. ;; It also stores detailed information about the entry so that
  34. ;; remember templates can access and enter this information easily.
  35. ;;
  36. ;; The available properties for each entry are listed here:
  37. ;;
  38. ;; :author :publisher :volume :pages
  39. ;; :editor :url :number :journal
  40. ;; :title :year :series :address
  41. ;; :booktitle :month :annote :abstract
  42. ;; :key :btype
  43. ;;
  44. ;; Here is an example of a remember template that use some of this
  45. ;; information (:author :year :title :journal :pages):
  46. ;;
  47. ;; (setq org-remember-templates
  48. ;; '((?b "* READ %?\n\n%a\n\n%:author (%:year): %:title\n \
  49. ;; In %:journal, %:pages.")))
  50. ;;
  51. ;; Let's say you want to remember this BibTeX entry:
  52. ;;
  53. ;; @Article{dolev83,
  54. ;; author = {Danny Dolev and Andrew C. Yao},
  55. ;; title = {On the security of public-key protocols},
  56. ;; journal = {IEEE Transaction on Information Theory},
  57. ;; year = 1983,
  58. ;; volume = 2,
  59. ;; number = 29,
  60. ;; pages = {198--208},
  61. ;; month = {Mars}
  62. ;; }
  63. ;;
  64. ;; M-x `org-remember' on this entry will produce this buffer:
  65. ;;
  66. ;; =====================================================================
  67. ;; * READ <== [point here]
  68. ;;
  69. ;; [[file:/file.bib::dolev83][Dolev & Yao 1983: security of public key protocols]]
  70. ;;
  71. ;; Danny Dolev and Andrew C. Yao (1983): On the security of public-key protocols
  72. ;; In IEEE Transaction on Information Theory, 198--208.
  73. ;; =====================================================================
  74. ;;
  75. ;;; History:
  76. ;;
  77. ;; The link creation part has been part of Org-mode for a long time.
  78. ;;
  79. ;; Creating better remember template information was inspired by a request
  80. ;; of Austin Frank: http://article.gmane.org/gmane.emacs.orgmode/4112
  81. ;; and then imlemented by Bastien Guerry.
  82. ;;
  83. ;; Org-mode loads this module by default - if this is not what you want,
  84. ;; configure the variable `org-modules'.
  85. ;;; Code:
  86. (require 'org)
  87. (defvar description nil) ; dynamically scoped from org.el
  88. (declare-function bibtex-beginning-of-entry "bibtex" ())
  89. (declare-function bibtex-generate-autokey "bibtex" ())
  90. (declare-function bibtex-parse-entry "bibtex" (&optional content))
  91. (declare-function bibtex-url "bibtex" (&optional pos no-browse))
  92. (org-add-link-type "bibtex" 'org-bibtex-open)
  93. (add-hook 'org-store-link-functions 'org-bibtex-store-link)
  94. ;; (defun org-bibtex-publish (path)
  95. ;; "Build the description of the BibTeX entry for publishing."
  96. ;; (let* ((search (when (string-match "::\\(.+\\)\\'" path)
  97. ;; (match-string 1 path)))
  98. ;; (path (substring path 0 (match-beginning 0)))
  99. ;; key)
  100. ;; (with-temp-buffer
  101. ;; (org-open-file path t nil search)
  102. ;; (setq key (org-create-file-search-functions)))
  103. ;; (or description key)))
  104. (defun org-bibtex-open (path)
  105. "Visit the bibliography entry on PATH."
  106. (let* ((search (when (string-match "::\\(.+\\)\\'" path)
  107. (match-string 1 path)))
  108. (path (substring path 0 (match-beginning 0))))
  109. (org-open-file path t nil search)))
  110. (defun org-bibtex-store-link ()
  111. "Store a link to a BibTeX entry."
  112. (when (eq major-mode 'bibtex-mode)
  113. (let* ((search (org-create-file-search-in-bibtex))
  114. (link (concat "file:" (abbreviate-file-name buffer-file-name)
  115. "::" search))
  116. (entry (mapcar ; repair strings enclosed in "..." or {...}
  117. (lambda(c)
  118. (if (string-match
  119. "^\\(?:{\\|\"\\)\\(.*\\)\\(?:}\\|\"\\)$" (cdr c))
  120. (cons (car c) (match-string 1 (cdr c))) c))
  121. (save-excursion
  122. (bibtex-beginning-of-entry)
  123. (bibtex-parse-entry)))))
  124. (org-store-link-props
  125. :key (cdr (assoc "=key=" entry))
  126. :author (or (cdr (assoc "author" entry)) "[no author]")
  127. :editor (or (cdr (assoc "editor" entry)) "[no editor]")
  128. :title (or (cdr (assoc "title" entry)) "[no title]")
  129. :booktitle (or (cdr (assoc "booktitle" entry)) "[no booktitle]")
  130. :journal (or (cdr (assoc "journal" entry)) "[no journal]")
  131. :publisher (or (cdr (assoc "publisher" entry)) "[no publisher]")
  132. :pages (or (cdr (assoc "pages" entry)) "[no pages]")
  133. :url (or (cdr (assoc "url" entry)) "[no url]")
  134. :year (or (cdr (assoc "year" entry)) "[no year]")
  135. :month (or (cdr (assoc "month" entry)) "[no month]")
  136. :address (or (cdr (assoc "address" entry)) "[no address]")
  137. :volume (or (cdr (assoc "volume" entry)) "[no volume]")
  138. :number (or (cdr (assoc "number" entry)) "[no number]")
  139. :annote (or (cdr (assoc "annote" entry)) "[no annotation]")
  140. :series (or (cdr (assoc "series" entry)) "[no series]")
  141. :abstract (or (cdr (assoc "abstract" entry)) "[no abstract]")
  142. :btype (or (cdr (assoc "=type=" entry)) "[no type]")
  143. :type "bibtex"
  144. :link link
  145. :description description))))
  146. (defun org-create-file-search-in-bibtex ()
  147. "Create the search string and description for a BibTeX database entry."
  148. ;; Make a good description for this entry, using names, year and the title
  149. ;; Put it into the `description' variable which is dynamically scoped.
  150. (let ((bibtex-autokey-names 1)
  151. (bibtex-autokey-names-stretch 1)
  152. (bibtex-autokey-name-case-convert-function 'identity)
  153. (bibtex-autokey-name-separator " & ")
  154. (bibtex-autokey-additional-names " et al.")
  155. (bibtex-autokey-year-length 4)
  156. (bibtex-autokey-name-year-separator " ")
  157. (bibtex-autokey-titlewords 3)
  158. (bibtex-autokey-titleword-separator " ")
  159. (bibtex-autokey-titleword-case-convert-function 'identity)
  160. (bibtex-autokey-titleword-length 'infty)
  161. (bibtex-autokey-year-title-separator ": "))
  162. (setq description (bibtex-generate-autokey)))
  163. ;; Now parse the entry, get the key and return it.
  164. (save-excursion
  165. (bibtex-beginning-of-entry)
  166. (cdr (assoc "=key=" (bibtex-parse-entry)))))
  167. (defun org-execute-file-search-in-bibtex (s)
  168. "Find the link search string S as a key for a database entry."
  169. (when (eq major-mode 'bibtex-mode)
  170. ;; Yes, we want to do the search in this file.
  171. ;; We construct a regexp that searches for "@entrytype{" followed by the key
  172. (goto-char (point-min))
  173. (and (re-search-forward (concat "@[a-zA-Z]+[ \t\n]*{[ \t\n]*"
  174. (regexp-quote s) "[ \t\n]*,") nil t)
  175. (goto-char (match-beginning 0)))
  176. (if (and (match-beginning 0) (equal current-prefix-arg '(16)))
  177. ;; Use double prefix to indicate that any web link should be browsed
  178. (let ((b (current-buffer)) (p (point)))
  179. ;; Restore the window configuration because we just use the web link
  180. (set-window-configuration org-window-config-before-follow-link)
  181. (save-excursion (set-buffer b) (goto-char p)
  182. (bibtex-url)))
  183. (recenter 0)) ; Move entry start to beginning of window
  184. ;; return t to indicate that the search is done.
  185. t))
  186. ;; Finally add the link search function to the right hook.
  187. (add-hook 'org-execute-file-search-functions 'org-execute-file-search-in-bibtex)
  188. (provide 'org-bibtex)
  189. ;;; org-bibtex.el ends here