]> code.delx.au - gnu-emacs/blobdiff - lisp/org/org-inlinetask.el
Merge from emacs-23
[gnu-emacs] / lisp / org / org-inlinetask.el
index 5911acb88a5d9bbc580936f9e4b018e0795926fe..12b5b5655a129b9d5f53e7d50df435e304c5b607 100644 (file)
@@ -5,7 +5,7 @@
 ;; Author: Carsten Dominik <carsten at orgmode dot org>
 ;; Keywords: outlines, hypermedia, calendar, wp
 ;; Homepage: http://orgmode.org
-;; Version: 6.33x
+;; Version: 7.4
 
 ;; This file is part of GNU Emacs.
 
@@ -33,7 +33,7 @@
 ;; and properties.  However, these nodes are treated specially by the
 ;; visibility cycling and export commands.
 ;;
-;; Visibility cycling exempts these nodes from cycling. So whenever their
+;; Visibility cycling exempts these nodes from cycling.  So whenever their
 ;; parent is opened, so are these tasks.  This will only work with
 ;; `org-cycle', so if you are also using other commands to show/hide
 ;; entries, you will occasionally find these tasks to behave like
@@ -74,8 +74,7 @@
 ;;
 ;; C-c C-x t      Insert a new inline task with END line
 
-
-;;; Code
+;;; Code:
 
 (require 'org)
 
@@ -91,31 +90,151 @@ or to a number smaller than this one.  In fact, when `org-cycle-max-level' is
 not set, it will be assumed to be one less than the value of smaller than
 the value of this variable."
   :group 'org-inlinetask
-  :type 'boolean)
+  :type '(choice
+         (const :tag "Off" nil)
+         (integer)))
 
 (defcustom org-inlinetask-export t
-  "Non-nil means, export inline tasks.
+  "Non-nil means export inline tasks.
 When nil, they will not be exported."
   :group 'org-inlinetask
   :type 'boolean)
 
+(defvar org-inlinetask-export-templates
+  '((html "<pre class=\"inlinetask\"><b>%s%s</b><br>%s</pre>"
+         '((unless (eq todo "")
+             (format "<span class=\"%s %s\">%s%s</span> "
+                     class todo todo priority))
+           heading content))
+    (latex "\\begin\{description\}\\item[%s%s]%s\\end\{description\}"
+          '((unless (eq todo "") (format "\\textsc\{%s%s\} " todo priority))
+            heading content))
+    (ascii "     -- %s%s%s"
+          '((unless (eq todo "") (format "%s%s " todo priority))
+            heading
+            (unless (eq content "")
+              (format "\n         ¦ %s"
+                      (mapconcat 'identity (org-split-string content "\n")
+                                 "\n         ¦ ")))))
+    (docbook "<variablelist>
+<varlistentry>
+<term>%s%s</term>
+<listitem><para>%s</para></listitem>
+</varlistentry>
+</variablelist>"
+            '((unless (eq todo "") (format "%s%s " todo priority))
+              heading content)))
+  "Templates for inline tasks in various exporters.
+
+This variable is an alist in the shape of (BACKEND STRING OBJECTS).
+
+BACKEND is the name of the backend for the template (ascii, html...).
+
+STRING is a format control string.
+
+OBJECTS is a list of elements to be substituted into the format
+string.  They can be of any type, from a string to a form
+returning a value (thus allowing conditional insertion).  A nil
+object will be substituted as the empty string.  Obviously, there
+must be at least as many objects as %-sequences in the format
+string.
+
+Moreover, the following special keywords are provided: `todo',
+`priority', `heading', `content', `tags'.  If some of them are not
+defined in an inline task, their value is the empty string.
+
+As an example, valid associations are:
+
+(html \"<ul><li>%s <p>%s</p></li></ul>\" (heading content))
+
+or, with the additional package \"todonotes\" for LaTeX,
+
+(latex \"\\todo[inline]{\\textbf{\\textsf{%s %s}}\\linebreak{} %s}\"
+       '((unless (eq todo \"\")
+          (format \"\\textsc{%s%s}\" todo priority))
+        heading content)))")
+
 (defvar org-odd-levels-only)
 (defvar org-keyword-time-regexp)
 (defvar org-drawer-regexp)
 (defvar org-complex-heading-regexp)
 (defvar org-property-end-re)
 
-(defun org-inlinetask-insert-task ()
-  "Insert an inline task."
-  (interactive)
+(defcustom org-inlinetask-default-state nil
+  "Non-nil means make inline tasks have a TODO keyword initially.
+This should be the state `org-inlinetask-insert-task' should use by
+default, or nil of no state should be assigned."
+  :group 'org-inlinetask
+  :type '(choice
+         (const :tag "No state" nil)
+         (string :tag "Specific state")))
+
+(defun org-inlinetask-insert-task (&optional no-state)
+  "Insert an inline task.
+If prefix arg NO-STATE is set, ignore `org-inlinetask-default-state'."
+  (interactive "P")
   (or (bolp) (newline))
-  (insert (make-string org-inlinetask-min-level ?*) " \n"
-         (make-string org-inlinetask-min-level ?*) " END\n")
+  (let ((indent org-inlinetask-min-level))
+    (if org-odd-levels-only
+        (setq indent (- (* 2 indent) 1)))
+    (insert (make-string indent ?*)
+            (if (or no-state (not org-inlinetask-default-state))
+               " \n"
+             (concat " " org-inlinetask-default-state " \n"))
+            (make-string indent ?*) " END\n"))
   (end-of-line -1))
 (define-key org-mode-map "\C-c\C-xt" 'org-inlinetask-insert-task)
 
-(defvar htmlp)  ; dynamically scoped into the next function
-(defvar latexp) ; dynamically scoped into the next function
+(defun org-inlinetask-outline-regexp ()
+  "Return string matching an inline task heading.
+The number of levels is controlled by `org-inlinetask-min-level'."
+  (let ((nstars (if org-odd-levels-only
+                   (1- (* org-inlinetask-min-level 2))
+                 org-inlinetask-min-level)))
+    (format "^\\(\\*\\{%d,\\}\\)[ \t]+" nstars)))
+
+(defun org-inlinetask-in-task-p ()
+  "Return true if point is inside an inline task."
+  (save-excursion
+    (let* ((stars-re (org-inlinetask-outline-regexp))
+          (task-beg-re (concat stars-re "\\(?:.*\\)"))
+          (task-end-re (concat stars-re "\\(?:END\\|end\\)[ \t]*$")))
+      (beginning-of-line)
+      (or (looking-at task-beg-re)
+         (and (re-search-forward "^\\*+[ \t]+" nil t)
+              (progn (beginning-of-line) (looking-at task-end-re)))))))
+
+(defun org-inlinetask-goto-beginning ()
+  "Go to the beginning of the inline task at point."
+  (end-of-line)
+  (re-search-backward (org-inlinetask-outline-regexp) nil t)
+  (when (org-looking-at-p (concat (org-inlinetask-outline-regexp) "END[ \t]*$"))
+    (re-search-backward (org-inlinetask-outline-regexp) nil t)))
+
+(defun org-inlinetask-goto-end ()
+  "Go to the end of the inline task at point."
+  (beginning-of-line)
+  (cond
+   ((org-looking-at-p (concat (org-inlinetask-outline-regexp) "END[ \t]*$"))
+    (forward-line 1))
+   ((org-looking-at-p (org-inlinetask-outline-regexp))
+    (forward-line 1)
+    (when (org-inlinetask-in-task-p)
+      (re-search-forward (org-inlinetask-outline-regexp) nil t)
+      (forward-line 1)))
+   (t
+    (re-search-forward (org-inlinetask-outline-regexp) nil t)
+    (forward-line 1))))
+
+(defun org-inlinetask-get-task-level ()
+  "Get the level of the inline task around.
+This assumes the point is inside an inline task."
+  (save-excursion
+    (end-of-line)
+    (re-search-backward (org-inlinetask-outline-regexp) nil t)
+    (- (match-end 1) (match-beginning 1))))
+
+(defvar backend) ; dynamically scoped into the next function
 (defun org-inlinetask-export-handler ()
   "Handle headlines with level larger or equal to `org-inlinetask-min-level'.
 Either remove headline and meta data, or do special formatting."
@@ -125,7 +244,7 @@ Either remove headline and meta data, or do special formatting."
                   (or org-inlinetask-min-level 200)))
         (re1 (format "^\\(\\*\\{%d,\\}\\) .*\n" nstars))
         (re2 (concat "^[ \t]*" org-keyword-time-regexp))
-        headline beg end stars content indent)
+        headline beg end stars content)
     (while (re-search-forward re1 nil t)
       (setq headline (match-string 0)
            stars (match-string 1)
@@ -146,30 +265,34 @@ Either remove headline and meta data, or do special formatting."
        (delete-region beg (1+ (match-end 0))))
       (goto-char beg)
       (when org-inlinetask-export
-       (when (string-match org-complex-heading-regexp headline)
-         (setq headline (concat
-                         (if (match-end 2)
-                             (concat (match-string 2 headline) " ") "")
-                         (match-string 4 headline)))
-         (when content
+       ;; content formatting
+       (when content
            (if (not (string-match "\\S-" content))
                (setq content nil)
              (if (string-match "[ \t\n]+\\'" content)
                  (setq content (substring content 0 (match-beginning 0))))
-             (setq content (org-remove-indentation content))
-             (if latexp (setq content (concat "\\quad \\\\\n" content)))))
-         (insert (make-string (org-inlinetask-get-current-indentation) ?\ )
-                 "- ")
-         (setq indent (make-string (current-column) ?\ ))
-         (insert headline " ::")
-         (if content
-             (insert (if htmlp " " (concat "\n" indent))
-                     (mapconcat 'identity (org-split-string content "\n")
-                                (concat "\n" indent)) "\n")
-           (insert "\n"))
-         (insert indent)
-         (backward-delete-char 2)
-         (insert "THISISTHEINLINELISTTEMINATOR\n"))))))
+             (setq content (org-remove-indentation content))))
+       (setq content (or content ""))
+       ;; grab elements to export
+       (when (string-match org-complex-heading-regexp headline)
+         (let* ((todo (or (match-string 2 headline) ""))
+                (class (or (and (eq "" todo) "")
+                           (if (member todo org-done-keywords) "done" "todo")))
+                (priority (or (match-string 3 headline) ""))
+                (heading (or (match-string 4 headline) ""))
+                (tags (or (match-string 5 headline) ""))
+                (backend-spec (assq backend org-inlinetask-export-templates))
+                (format-str (nth 1 backend-spec))
+                (tokens (cadr (nth 2 backend-spec)))
+                ;; change nil arguments into empty strings
+                (nil-to-str (lambda (el) (or (eval el) "")))
+                ;; build and protect export string
+                (export-str (org-add-props
+                                (eval (append '(format format-str)
+                                              (mapcar nil-to-str tokens)))
+                                nil 'org-protected t)))
+           ;; eventually insert it
+           (insert export-str "\n")))))))
 
 (defun org-inlinetask-get-current-indentation ()
   "Get the indentation of the last non-while line above this one."
@@ -204,33 +327,12 @@ Either remove headline and meta data, or do special formatting."
                            org-inlinetask-min-level))
     (replace-match "")))
 
-(defun org-inlinetask-remove-terminator ()
-  (let (beg end)
-    (save-excursion
-      (goto-char (point-min))
-      (while (re-search-forward "THISISTHEINLINELISTTEMINATOR\n" nil t)
-       (setq beg (match-beginning 0) end (match-end 0))
-       (save-excursion
-         (beginning-of-line 1)
-         (and (looking-at "<p\\(ara\\)?>THISISTHEINLINELISTTEMINATOR[ \t\n]*</p\\(ara\\)?>")
-              (setq beg (point) end (match-end 0))))
-       (delete-region beg end)))))
-
 (eval-after-load "org-exp"
   '(add-hook 'org-export-preprocess-after-tree-selection-hook
             'org-inlinetask-export-handler))
 (eval-after-load "org"
   '(add-hook 'org-font-lock-hook 'org-inlinetask-fontify))
-(eval-after-load "org-html"
-  '(add-hook 'org-export-html-final-hook 'org-inlinetask-remove-terminator))
-(eval-after-load "org-latex"
-  '(add-hook 'org-export-latex-final-hook 'org-inlinetask-remove-terminator))
-(eval-after-load "org-ascii"
-  '(add-hook 'org-export-ascii-final-hook 'org-inlinetask-remove-terminator))
-(eval-after-load "org-docbook"
-  '(add-hook 'org-export-docbook-final-hook 'org-inlinetask-remove-terminator))
 
 (provide 'org-inlinetask)
 
-;; arch-tag: 59fdac51-8bcc-469e-a21e-6897dd6697bb
 ;;; org-inlinetask.el ends here