Improvements to contrib/lisp/org-eval.el

This works now reasonably stable.
This commit is contained in:
Carsten Dominik 2008-05-03 16:29:48 +02:00
parent 7436274350
commit af0b9e7405
3 changed files with 101 additions and 47 deletions

View File

@ -4,7 +4,7 @@
;; Author: Carsten Dominik <carsten at orgmode dot org> ;; Author: Carsten Dominik <carsten at orgmode dot org>
;; Keywords: outlines, hypermedia, calendar, wp ;; Keywords: outlines, hypermedia, calendar, wp
;; Homepage: http://orgmode.org ;; Homepage: http://orgmode.org
;; Version: 0.01 ;; Version: 0.03
;; ;;
;; This file is not yet part of GNU Emacs. ;; This file is not yet part of GNU Emacs.
;; ;;
@ -25,46 +25,97 @@
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;; ;;
;;; Commentary: ;;; Commentary:
;;
;; This modules allows to include output from various commands into an
;; Org-mode buffer. This technique has been copied from Emacs-Muse, and
;; we try to make it work here in a way as simila as possible to
;; Muse.
(require 'org) (require 'org)
;;; Customization ;;; Customization
(defgroup org-eval nil (defgroup org-eval nil
"Options concerning global entry identifiers in Org-mode." "Options concerning including output from commands into the Org-mode buffer."
:tag "Org ID" :tag "Org Eval"
:group 'org) :group 'org)
(defface org-eval (defface org-eval
(org-compatible-face nil (org-compatible-face nil
'((((class color grayscale) (min-colors 88) (background light)) '((((class color grayscale) (min-colors 88) (background light))
(:foreground "grey20")) (:foreground "grey40"))
(((class color grayscale) (min-colors 88) (background dark)) (((class color grayscale) (min-colors 88) (background dark))
(:foreground "grey80")) (:foreground "grey60"))
(((class color) (min-colors 8) (background light)) (((class color) (min-colors 8) (background light))
(:foreground "green")) (:foreground "green"))
(((class color) (min-colors 8) (background dark)) (((class color) (min-colors 8) (background dark))
(:foreground "yellow")))) (:foreground "yellow"))))
"Face for fixed-with text like code snippets." "Face for command output that is included into an Org-mode buffer."
:group 'org-eval :group 'org-eval
:group 'org-faces :group 'org-faces
:version "22.1") :version "22.1")
(defvar org-eval-regexp nil)
(defun org-eval-set-interpreters (var value)
(set-default var value)
(setq org-eval-regexp
(concat "<\\("
(mapconcat 'regexp-quote value "\\|")
"\\)"
"\\([^>]\\{0,50\\}?\\)>"
"\\([^\000]+?\\)</\\1>")))
(defcustom org-eval-interpreters '("lisp")
"Interpreters allows for evaluation tags.
This is a list of program names (as strings) that can evaluate code and
insert the output into an Org-mode buffer. Valid choices are
lisp Interpret Emacs Lisp code and display the result
shell Pass command to the shell and display the result
perl The perl interpreter
python Thy python interpreter
ruby The ruby interpreter"
:group 'org-eval
:set 'org-eval-set-interpreters
:type '(set :greedy t
(const "lisp")
(const "perl")
(const "python")
(const "ruby")
(const "shell")))
(defun org-eval-handle-snippets (limit &optional replace) (defun org-eval-handle-snippets (limit &optional replace)
"Evaluate code nisppets and display the results as display property.
When REPLACE is non-nil, replace the code region with the result (used
for export)."
(let (a) (let (a)
(while (setq a (text-property-any (point) (or limit (point-max)) (while (setq a (text-property-any (point) (or limit (point-max))
'org-eval t)) 'org-eval t))
(remove-text-properties (remove-text-properties
a (next-single-property-change a 'org-eval nil limit) a (next-single-property-change a 'org-eval nil limit)
'(display t intangible t org-eval t)))) '(display t intangible t org-eval t))))
(while (re-search-forward "<\\(lisp\\)>\\([^\000]+?\\)</\\1>" limit t) (while (re-search-forward org-eval-regexp limit t)
(let* ((beg (match-beginning 0)) (let* ((beg (match-beginning 0))
(end (match-end 0)) (end (match-end 0))
(kind (match-string 1)) (kind (match-string 1))
(code (match-string 2)) (attr (match-string 2))
(value (org-eval-code kind code))) (code (match-string 3))
(value (org-eval-code kind code))
markup lang)
(if replace (if replace
(replace-match value t t) (progn
(setq attr (save-match-data (org-eval-get-attributes attr))
markup (cdr (assoc "markup" attr))
lang (cdr (assoc "lang" attr)))
(replace-match
(concat (if markup (format "#+BEGIN_%s" (upcase markup)))
(if (and markup (equal (downcase markup) "src"))
(concat " " (or lang "fundamental")))
"\n"
value
(if markup (format "\n#+END_%s\n" (upcase markup))))
t t))
(add-text-properties (add-text-properties
beg end beg end
(list 'display value 'intangible t 'font-lock-multiline t (list 'display value 'intangible t 'font-lock-multiline t
@ -80,10 +131,23 @@ This should go into the `org-export-preprocess-hook'."
(add-hook 'org-export-preprocess-hook 'org-eval-replace-snippts) (add-hook 'org-export-preprocess-hook 'org-eval-replace-snippts)
(add-hook 'org-font-lock-hook 'org-eval-handle-snippets) (add-hook 'org-font-lock-hook 'org-eval-handle-snippets)
(defun org-eval-get-attributes (str)
(let ((start 0) key value rtn)
(while (string-match "\\<\\([a-zA-Z]+\\)\\>=\"\\([^\"]+\\)\"" str start)
(setq key (match-string 1 str)
value (match-string 2 str)
start (match-end 0))
(push (cons key value) rtn))
rtn))
(defun org-eval-code (interpreter code) (defun org-eval-code (interpreter code)
(cond (cond
((equal interpreter "lisp") ((equal interpreter "lisp")
(org-eval-lisp (concat "(progn\n" code "\n)"))) (org-eval-lisp (concat "(progn\n" code "\n)")))
((equal interpreter "shell")
(shell-command-to-string code))
((member interpreter '("perl" "python" "ruby"))
(org-eval-run (executable-find interpreter) code))
(t (error "Cannot evaluate code type %s" interpreter)))) (t (error "Cannot evaluate code type %s" interpreter))))
(defun org-eval-lisp (form) (defun org-eval-lisp (form)
@ -108,21 +172,11 @@ This should go into the `org-export-preprocess-hook'."
"???" form err)) "???" form err))
"; INVALID LISP CODE")))) "; INVALID LISP CODE"))))
(defun org-display-warning (message) (defun org-eval-run (cmd code)
"Display the given MESSAGE as a warning." (with-temp-buffer
(if (fboundp 'display-warning) (insert code)
(display-warning 'org message (shell-command-on-region (point-min) (point-max) cmd nil 'replace)
(if (featurep 'xemacs) (buffer-string)))
'warning
:warning))
(let ((buf (get-buffer-create "*Org warnings*")))
(with-current-buffer buf
(goto-char (point-max))
(insert "Warning (Org): " message)
(unless (bolp)
(newline)))
(display-buffer buf)
(sit-for 0))))
(provide 'org-eval) (provide 'org-eval)

View File

@ -1586,6 +1586,7 @@ When LEVEL is non-nil, increase section numbers on that level."
"Format CODE from language LANG and return it formatted for export. "Format CODE from language LANG and return it formatted for export.
Currently, this only does something for HTML export, for all other Currently, this only does something for HTML export, for all other
backends, it converts the segment into an EXAMPLE segment." backends, it converts the segment into an EXAMPLE segment."
(save-match-data
(cond (cond
(htmlp (htmlp
;; We are exporting to HTML ;; We are exporting to HTML
@ -1599,7 +1600,6 @@ backends, it converts the segment into an EXAMPLE segment."
(if (string-match "\n\\'" code) "" "\n") (if (string-match "\n\\'" code) "" "\n")
"#+END_EXAMPLE\n")) "#+END_EXAMPLE\n"))
;; ok, we are good to go ;; ok, we are good to go
(save-match-data
(let* ((mode (and lang (intern (concat lang "-mode")))) (let* ((mode (and lang (intern (concat lang "-mode"))))
(org-startup-folded nil) (org-startup-folded nil)
(htmltext (htmltext
@ -1622,12 +1622,12 @@ backends, it converts the segment into an EXAMPLE segment."
(if (string-match "<pre\\([^>]*\\)>\n?" htmltext) (if (string-match "<pre\\([^>]*\\)>\n?" htmltext)
(setq htmltext (replace-match "<pre class=\"src\">" (setq htmltext (replace-match "<pre class=\"src\">"
t t htmltext))) t t htmltext)))
(concat "#+BEGIN_HTML\n" htmltext "\n#+END_HTML\n"))))) (concat "#+BEGIN_HTML\n" htmltext "\n#+END_HTML\n"))))
(t (t
;; This is not HTML, so just make it an example. ;; This is not HTML, so just make it an example.
(concat "#+BEGIN_EXAMPLE\n" code (concat "#+BEGIN_EXAMPLE\n" code
(if (string-match "\n\\'" code) "" "\n") (if (string-match "\n\\'" code) "" "\n")
"#+END_EXAMPLE\n")))) "#+END_EXAMPLE\n")))))
;;; ASCII export ;;; ASCII export

View File

@ -181,7 +181,7 @@ calendar | %:type %:date"
(defcustom org-remember-clock-out-on-exit 'query (defcustom org-remember-clock-out-on-exit 'query
"Non-nil means, stop the clock when exiting a clocking remember buffer. "Non-nil means, stop the clock when exiting a clocking remember buffer.
This only applies of the clock is running in the remember buffer. If the This only applies if the clock is running in the remember buffer. If the
clock is not stopped, it continues to run in the storage location. clock is not stopped, it continues to run in the storage location.
Instead of nil or t, this may also be the symbol `query' to prompt the Instead of nil or t, this may also be the symbol `query' to prompt the
user each time a remember buffer with a running clock is filed away. " user each time a remember buffer with a running clock is filed away. "