]> git.eshelyaron.com Git - emacs.git/commitdiff
* todos.el: Reorganize file structure again, to pacify byte-compiler.
authorStephen Berman <stephen.berman@gmx.net>
Fri, 7 Jun 2013 22:55:42 +0000 (00:55 +0200)
committerStephen Berman <stephen.berman@gmx.net>
Fri, 7 Jun 2013 22:55:42 +0000 (00:55 +0200)
lisp/calendar/ChangeLog
lisp/calendar/todos.el

index 829b60c5637fa4280bac8c98d2c81d01707f5c22..86f268816cb35619ea407e16412f08f052a03085 100644 (file)
@@ -1,3 +1,14 @@
+2013-06-07  Stephen Berman  <stephen.berman@gmx.net>
+
+       * todos.el: Reorganize file structure again, to pacify byte-compiler.
+
+2013-06-06  Stephen Berman  <stephen.berman@gmx.net>
+
+       * todos.el: Fix more byte-compiler warnings.
+       (todos-jump-to-category): Let-bind variable that was mistakenly free.
+       (todos-toggle-item-highlighting, todos-convert-legacy-files):
+       Use eval-when-compile.
+
 2013-06-05  Stephen Berman  <stephen.berman@gmx.net>
 
        * todos.el: Fix byte-compiler warnings.
index a345fdcfd08fde89d2661a01fefa359ca39d8f8f..f2c31a33aeafca036e2f5dd04ed088510f6dca2a 100644 (file)
 
 ;;; Commentary:
 
-;; This package provides facilities for making, displaying, navigating
-;; and editing todo lists, which are prioritized lists of todo items.
-;; Todo lists are identified with named categories, so you can group
-;; together thematically related todo items.  Each category is stored
-;; in a file, which thus provides a further level of organization.
-;; You can create as many todo files, and in each as many categories,
-;; as you want.
-
-;; With Todos you can navigate among the items of a category, and
-;; between categories in the same and in different todo files.  You
-;; can edit todo items, reprioritize them within their category, move
-;; them to another category, delete them, or mark items as done and
-;; store them separately from the not yet done items in a category.
-;; You can add new todo files and categories, rename categories, move
-;; them to another file or delete them.  You can also display summary
-;; tables of the categories in a file and the types of items they
-;; contain.  And you can build cross-categorial lists of items that
-;; satisfy various criteria.
-
-;; To get started, load this package and type `M-x todos-show'.  This
-;; will prompt you for the name of the first todo file, its first
-;; category and the category's first item, create these and display
-;; them in Todos mode.  Now you can insert further items into the list
-;; (i.e., the category) and assign them priorities by typing `i i'.
-
-;; You will probably find it convenient to give `todos-show' a global
-;; key binding in your init file, since it is one of the entry points
-;; to Todos mode; a good choice is `C-c t', since `todos-show' is
-;; bound to `t' in Todos mode.
-
-;; To see a list of all Todos mode commands and their key bindings,
-;; including other entry points, type `C-h m' in Todos mode.  Consult
-;; the document strings of the commands for details of their use.  The
-;; `todos' customization group and its subgroups list the options you
-;; can set to alter the behavior of many commands and various aspects
-;; of the display.
-
-;; This package is a new version of Oliver Seidel's todo-mode.el,
-;; which retains the same basic organization and handling of todo
-;; lists and the basic UI, but extends these in many ways and
-;; reimplements most of the internals.
+;; This package provides facilities for making, displaying, navigating and
+;; editing todo lists, which are prioritized lists of todo items.  Todo lists
+;; are identified with named categories, so you can group together and
+;; separately prioritize thematically related todo items.  Each category is
+;; stored in a file, which thus provides a further level of organization.  You
+;; can create as many todo files, and in each as many categories, as you want.
+
+;; With Todos you can navigate among the items of a category, and between
+;; categories in the same and in different todo files.  You can edit todo
+;; items, reprioritize them within their category, move them to another
+;; category, delete them, or mark items as done and store them separately from
+;; the not yet done items in a category.  You can add new todo files and
+;; categories, rename categories, move them to another file or delete them.
+;; You can also display summary tables of the categories in a file and the
+;; types of items they contain.  And you can build cross-categorial lists of
+;; items that satisfy various criteria.
+
+;; To get started, load this package and type `M-x todos-show'.  This will
+;; prompt you for the name of the first todo file, its first category and the
+;; category's first item, create these and display them in Todos mode.  Now
+;; you can insert further items into the list (i.e., the category) and assign
+;; them priorities by typing `i i'.
+
+;; You will probably find it convenient to give `todos-show' a global key
+;; binding in your init file, since it is one of the entry points to Todos
+;; mode; a good choice is `C-c t', since `todos-show' is bound to `t' in Todos
+;; mode.
+
+;; To see a list of all Todos mode commands and their key bindings, including
+;; other entry points, type `C-h m' in Todos mode.  Consult the document
+;; strings of the commands for details of their use.  The `todos'
+;; customization group and its subgroups list the options you can set to alter
+;; the behavior of many commands and various aspects of the display.
+
+;; This package is a new version of Oliver Seidel's todo-mode.el, which
+;; retains the same basic organization and handling of todo lists and the
+;; basic UI, but extends these in many ways and reimplements most of the
+;; internals.
 
 ;;; Code:
 
 ;; For cl-remove-duplicates (in todos-insertion-commands-args) and cl-oddp.
 (require 'cl-lib)
 
-;; =============================================================================
-;;; User interface
-;; =============================================================================
 ;; -----------------------------------------------------------------------------
-;;; Options for file and category selection
+;;; Setting up Todos files, categories, and items
 ;; -----------------------------------------------------------------------------
 
 (defcustom todos-directory (locate-user-emacs-file "todos/")
@@ -103,5612 +97,5550 @@ makes it return the value of the variable `todos-archives'."
   :type 'function
   :group 'todos)
 
-(defun todos-short-file-name (file)
-  "Return short form of Todos FILE.
-This lacks the extension and directory components."
-  (when (stringp file)
-    (file-name-sans-extension (file-name-nondirectory file))))
+(defvar todos-files (funcall todos-files-function)
+  "List of truenames of user's Todos files.")
 
-(defcustom todos-visit-files-commands (list 'find-file 'dired-find-file)
-  "List of file finding commands for `todos-display-as-todos-file'.
-Invoking these commands to visit a Todos or Todos Archive file
-calls `todos-show' or `todos-find-archive', so that the file is
-displayed correctly."
-  :type '(repeat function)
-  :group 'todos)
+(defvar todos-archives (funcall todos-files-function t)
+  "List of truenames of user's Todos archives.")
 
-(defcustom todos-default-todos-file (todos-short-file-name
-                                    (car (funcall todos-files-function)))
-  "Todos file visited by first session invocation of `todos-show'."
-  :type `(radio ,@(mapcar (lambda (f) (list 'const f))
-                         (mapcar 'todos-short-file-name
-                                 (funcall todos-files-function))))
-  :group 'todos)
+(defvar todos-visited nil
+  "List of Todos files visited in this session by `todos-show'.
+Used to determine initial display according to the value of
+`todos-show-first'.")
 
-(defcustom todos-show-current-file t
-  "Non-nil to make `todos-show' visit the current Todos file.
-Otherwise, `todos-show' always visits `todos-default-todos-file'."
-  :type 'boolean
-  :initialize 'custom-initialize-default
-  :set 'todos-set-show-current-file
-  :group 'todos)
+(defvar todos-file-buffers nil
+  "List of file names of live Todos mode buffers.")
 
-(defcustom todos-show-first 'first
-  "What action to take on first use of `todos-show' on a file."
-  :type '(choice (const :tag "Show first category" first)
-                (const :tag "Show table of categories" table)
-                (const :tag "Show top priorities" top)
-                (const :tag "Show diary items" diary)
-                (const :tag "Show regexp items" regexp))
-  :group 'todos)
+(defvar todos-global-current-todos-file nil
+  "Variable holding name of current Todos file.
+Used by functions called from outside of Todos mode to visit the
+current Todos file rather than the default Todos file (i.e. when
+users option `todos-show-current-file' is non-nil).")
 
-(defcustom todos-initial-file "Todo"
-  "Default file name offered on adding first Todos file."
+(defvar todos-current-todos-file nil
+  "Variable holding the name of the currently active Todos file.")
+
+(defvar todos-categories nil
+  "Alist of categories in the current Todos file.
+The elements are cons cells whose car is a category name and
+whose cdr is a vector of the category's item counts.  These are,
+in order, the numbers of todo items, of todo items included in
+the Diary, of done items and of archived items.")
+
+(defvar todos-category-number 1
+  "Variable holding the number of the current Todos category.
+Todos categories are numbered starting from 1.")
+
+(defvar todos-categories-with-marks nil
+  "Alist of categories and number of marked items they contain.")
+
+(defconst todos-category-beg "--==-- "
+  "String marking beginning of category (inserted with its name).")
+
+(defconst todos-category-done "==--== DONE "
+  "String marking beginning of category's done items.")
+
+(defcustom todos-done-separator-string "="
+  "String determining the value of variable `todos-done-separator'.
+If the string consists of a single character,
+`todos-done-separator' will be the string made by repeating this
+character for the width of the window, and the length is
+automatically recalculated when the window width changes.  If the
+string consists of more (or less) than one character, it will be
+the value of `todos-done-separator'."
   :type 'string
-  :group 'todos)
+  :initialize 'custom-initialize-default
+  :set 'todos-reset-done-separator-string
+  :group 'todos-display)
 
-(defcustom todos-initial-category "Todo"
-  "Default category name offered on initializing a new Todos file."
+(defun todos-done-separator ()
+  "Return string used as value of variable `todos-done-separator'."
+  (let ((sep todos-done-separator-string))
+    (propertize (if (= 1 (length sep))
+                   ;; Until bug#2749 is fixed, if separator's length
+                   ;; is window-width and todos-wrap-lines is
+                   ;; non-nil, an indented empty line appears between
+                   ;; the separator and the first done item.
+                   ;; (make-string (window-width) (string-to-char sep))
+                   (make-string (1- (window-width)) (string-to-char sep))
+                 todos-done-separator-string)
+               'face 'todos-done-sep)))
+
+(defvar todos-done-separator (todos-done-separator)
+  "String used to visually separate done from not done items.
+Displayed as an overlay instead of `todos-category-done' when
+done items are shown.  Its value is determined by user option
+`todos-done-separator-string'.")
+
+(defvar todos-show-done-only nil
+  "If non-nil display only done items in current category.
+Set by the command `todos-toggle-view-done-only' and used by
+`todos-category-select'.")
+
+(defcustom todos-nondiary-marker '("[" "]")
+  "List of strings surrounding item date to block diary inclusion.
+The first string is inserted before the item date and must be a
+non-empty string that does not match a diary date in order to
+have its intended effect.  The second string is inserted after
+the diary date."
+  :type '(list string string)
+  :group 'todos-edit
+  :initialize 'custom-initialize-default
+  :set 'todos-reset-nondiary-marker)
+
+(defconst todos-nondiary-start (nth 0 todos-nondiary-marker)
+  "String inserted before item date to block diary inclusion.")
+
+(defconst todos-nondiary-end (nth 1 todos-nondiary-marker)
+  "String inserted after item date matching `todos-nondiary-start'.")
+
+(defconst todos-month-name-array
+  (vconcat calendar-month-name-array (vector "*"))
+  "Array of month names, in order.
+The final element is \"*\", indicating an unspecified month.")
+
+(defconst todos-month-abbrev-array
+  (vconcat calendar-month-abbrev-array (vector "*"))
+  "Array of abbreviated month names, in order.
+The final element is \"*\", indicating an unspecified month.")
+
+(defconst todos-date-pattern
+  (let ((dayname (diary-name-pattern calendar-day-name-array nil t)))
+    (concat "\\(?5:" dayname "\\|"
+           (let ((dayname)
+                 (monthname (format "\\(?6:%s\\)" (diary-name-pattern
+                                                   todos-month-name-array
+                                                   todos-month-abbrev-array)))
+                 (month "\\(?7:[0-9]+\\|\\*\\)")
+                 (day "\\(?8:[0-9]+\\|\\*\\)")
+                 (year "-?\\(?9:[0-9]+\\|\\*\\)"))
+             (mapconcat 'eval calendar-date-display-form ""))
+           "\\)"))
+  "Regular expression matching a Todos date header.")
+
+;; By itself this matches anything, because of the `?'; however, it's only
+;; used in the context of `todos-date-pattern' (but Emacs Lisp lacks
+;; lookahead).
+(defconst todos-date-string-start
+  (concat "^\\(" (regexp-quote todos-nondiary-start) "\\|"
+         (regexp-quote diary-nonmarking-symbol) "\\)?")
+  "Regular expression matching part of item header before the date.")
+
+(defcustom todos-done-string "DONE "
+  "Identifying string appended to the front of done todos items."
   :type 'string
-  :group 'todos)
+  :initialize 'custom-initialize-default
+  :set 'todos-reset-done-string
+  :group 'todos-edit)
 
-(defcustom todos-category-completions-files nil
-  "List of files for building `todos-read-category' completions."
-  :type `(set ,@(mapcar (lambda (f) (list 'const f))
-                       (mapcar 'todos-short-file-name
-                               (funcall todos-files-function))))
-  :group 'todos)
+(defconst todos-done-string-start
+  (concat "^\\[" (regexp-quote todos-done-string))
+  "Regular expression matching start of done item.")
 
-(defcustom todos-completion-ignore-case nil
-  "Non-nil means case is ignored by `todos-read-*' functions."
-  :type 'boolean
-  :group 'todos)
+(defconst todos-item-start (concat "\\(" todos-date-string-start "\\|"
+                                todos-done-string-start "\\)"
+                                todos-date-pattern)
+  "String identifying start of a Todos item.")
 
 ;; -----------------------------------------------------------------------------
-;;; Entering and exiting Todos mode
+;;; Todos mode display options
 ;; -----------------------------------------------------------------------------
 
-(defun todos-show (&optional solicit-file)
-  "Visit a Todos file and display one of its categories.
+(defcustom todos-prefix ""
+  "String prefixed to todo items for visual distinction."
+  :type '(string :validate
+                (lambda (widget)
+                  (when (string= (widget-value widget) todos-item-mark)
+                    (widget-put
+                     widget :error
+                     "Invalid value: must be distinct from `todos-item-mark'")
+                    widget)))
+  :initialize 'custom-initialize-default
+  :set 'todos-reset-prefix
+  :group 'todos-display)
 
-When invoked in Todos mode, prompt for which todo file to visit.
-When invoked outside of Todos mode with non-nil prefix argument
-SOLICIT-FILE prompt for which todo file to visit; otherwise visit
-`todos-default-todos-file'.  Subsequent invocations from outside
-of Todos mode revisit this file or, with option
-`todos-show-current-file' non-nil (the default), whichever Todos
-file was last visited.
+(defcustom todos-number-prefix t
+  "Non-nil to prefix items with consecutively increasing integers.
+These reflect the priorities of the items in each category."
+  :type 'boolean
+  :initialize 'custom-initialize-default
+  :set 'todos-reset-prefix
+  :group 'todos-display)
 
-Calling this command before any Todos file exists prompts for a
-file name and an initial category (defaulting to
-`todos-initial-file' and `todos-initial-category'), creates both
-of these, visits the file and displays the category, and if
-option `todos-add-item-if-new-category' is non-nil (the default),
-prompts for the first item.
+(defun todos-mode-line-control (cat)
+  "Return a mode line control for todo or archive file buffers.
+Argument CAT is the name of the current Todos category.
+This function is the value of the user variable
+`todos-mode-line-function'."
+  (let ((file (todos-short-file-name todos-current-todos-file)))
+    (format "%s category %d: %s" file todos-category-number cat)))
 
-The first invocation of this command on an existing Todos file
-interacts with the option `todos-show-first': if its value is
-`first' (the default), show the first category in the file; if
-its value is `table', show the table of categories in the file;
-if its value is one of `top', `diary' or `regexp', show the
-corresponding saved top priorities, diary items, or regexp items
-file, if any.  Subsequent invocations always show the file's
-current (i.e., last displayed) category.
+(defcustom todos-mode-line-function 'todos-mode-line-control
+  "Function that returns a mode line control for Todos buffers.
+The function expects one argument holding the name of the current
+Todos category.  The resulting control becomes the local value of
+`mode-line-buffer-identification' in each Todos buffer."
+  :type 'function
+  :group 'todos-display)
 
-In Todos mode just the category's unfinished todo items are shown
-by default.  The done items are hidden, but typing
-`\\[todos-toggle-view-done-items]' displays them below the todo
-items.  With non-nil user option `todos-show-with-done' both todo
-and done items are always shown on visiting a category.
+(defcustom todos-highlight-item nil
+  "Non-nil means highlight items at point."
+  :type 'boolean
+  :initialize 'custom-initialize-default
+  :set 'todos-reset-highlight-item
+  :group 'todos-display)
 
-Invoking this command in Todos Archive mode visits the
-corresponding Todos file, displaying the corresponding category."
-  (interactive "P")
-  (let* ((cat)
-        (show-first todos-show-first)
-        (file (cond ((or solicit-file
-                         (and (called-interactively-p 'any)
-                              (memq major-mode '(todos-mode
-                                                 todos-archive-mode
-                                                 todos-filtered-items-mode))))
-                     (if (funcall todos-files-function)
-                         (todos-read-file-name "Choose a Todos file to visit: "
-                                               nil t)
-                       (user-error "There are no Todos files")))
-                    ((and (eq major-mode 'todos-archive-mode)
-                          ;; Called noninteractively via todos-quit
-                          ;; to jump to corresponding category in
-                          ;; todo file.
-                          (not (called-interactively-p 'any)))
-                     (setq cat (todos-current-category))
-                     (concat (file-name-sans-extension
-                              todos-current-todos-file) ".todo"))
-                    (t
-                     (or todos-current-todos-file
-                         (and todos-show-current-file
-                              todos-global-current-todos-file)
-                         (todos-absolute-file-name todos-default-todos-file)
-                         (todos-add-file)))))
-        add-item first-file)
-    (unless todos-default-todos-file
-      ;; We just initialized the first todo file, so make it the default.
-      (setq todos-default-todos-file (todos-short-file-name file)
-           first-file t)
-      (todos-reevaluate-default-file-defcustom))
-    (unless (member file todos-visited)
-      ;; Can't setq t-c-t-f here, otherwise wrong file shown when
-      ;; todos-show is called from todos-show-categories-table.
-      (let ((todos-current-todos-file file))
-       (cond ((eq todos-show-first 'table)
-              (todos-show-categories-table))
-             ((memq todos-show-first '(top diary regexp))
-              (let* ((shortf (todos-short-file-name file))
-                     (fi-file (todos-absolute-file-name
-                               shortf todos-show-first)))
-                (when (eq todos-show-first 'regexp)
-                  (let ((rxfiles (directory-files todos-directory t
-                                                  ".*\\.todr$" t)))
-                    (when (and rxfiles (> (length rxfiles) 1))
-                      (let ((rxf (mapcar 'todos-short-file-name rxfiles)))
-                        (setq fi-file (todos-absolute-file-name
-                                       (completing-read
-                                        "Choose a regexp items file: "
-                                        rxf) 'regexp))))))
-                (if (file-exists-p fi-file)
-                    (set-window-buffer
-                     (selected-window)
-                     (set-buffer (find-file-noselect fi-file 'nowarn)))
-                  (message "There is no %s file for %s"
-                           (cond ((eq todos-show-first 'top)
-                                  "top priorities")
-                                 ((eq todos-show-first 'diary)
-                                  "diary items")
-                                 ((eq todos-show-first 'regexp)
-                                  "regexp items"))
-                           shortf)
-                  (setq todos-show-first 'first)))))))
-    (when (or (member file todos-visited)
-             (eq todos-show-first 'first))
-      (set-window-buffer (selected-window)
-                        (set-buffer (find-file-noselect file 'nowarn)))
-      ;; When quitting archive file, show corresponding category in
-      ;; Todos file, if it exists.
-      (when (assoc cat todos-categories)
-       (setq todos-category-number (todos-category-number cat)))
-      ;; If this is a new Todos file, add its first category.
-      (when (zerop (buffer-size))
-       (let (cat-added)
-         (unwind-protect
-             (setq todos-category-number
-                   (todos-add-category todos-current-todos-file "")
-                   add-item todos-add-item-if-new-category
-                   cat-added t)
-           (if cat-added
-               ;; If the category was added, save the file now, so we
-               ;; don't risk having an empty todo file, which would
-               ;; signal an error if we tried to visit it later,
-               ;; since doing that looks for category boundaries.
-               (save-buffer 0)
-             ;; If user cancels before adding the category, clean up
-             ;; and exit, so we have a fresh slate the next time.
-             (delete-file file)
-             (setq todos-files (delete file todos-files))
-             (when first-file
-               (setq todos-default-todos-file nil
-                     todos-current-todos-file nil))
-             (kill-buffer)
-             (keyboard-quit)))))
-      (save-excursion (todos-category-select))
-      (when add-item (todos-basic-insert-item)))
-    (setq todos-show-first show-first)
-    (add-to-list 'todos-visited file)))
+(defcustom todos-wrap-lines t
+  "Non-nil to activate Visual Line mode and use wrap prefix."
+  :type 'boolean
+  :group 'todos-display)
 
-(defun todos-save ()
-  "Save the current Todos file."
-  (interactive)
-  (cond ((eq major-mode 'todos-filtered-items-mode)
-        (todos-check-filtered-items-file)
-        (todos-save-filtered-items-buffer))
-       (t
-        (save-buffer))))
+(defcustom todos-indent-to-here 3
+  "Number of spaces to indent continuation lines of items.
+This must be a positive number to ensure such items are fully
+shown in the Fancy Diary display."
+  :type '(integer :validate
+                 (lambda (widget)
+                   (unless (> (widget-value widget) 0)
+                     (widget-put widget :error
+                                 "Invalid value: must be a positive integer")
+                     widget)))
+  :group 'todos-display)
 
-(defun todos-quit ()
-  "Exit the current Todos-related buffer.
-Depending on the specific mode, this either kills the buffer or
-buries it and restores state as needed."
-  (interactive)
-  (let ((buf (current-buffer)))
-    (cond ((eq major-mode 'todos-categories-mode)
-          ;; Postpone killing buffer till after calling todos-show, to
-          ;; prevent killing todos-mode buffer.
-          (setq todos-descending-counts nil)
-          ;; Ensure todos-show calls todos-show-categories-table only on
-          ;; first invocation per file.
-          (when (eq todos-show-first 'table)
-            (add-to-list 'todos-visited todos-current-todos-file))
-          (todos-show)
-          (kill-buffer buf))
-         ((eq major-mode 'todos-filtered-items-mode)
-          (kill-buffer)
-          (unless (eq major-mode 'todos-mode) (todos-show)))
-         ((eq major-mode 'todos-archive-mode)
-          ;; Have to write a newly created archive to file to avoid
-          ;; subsequent errors.
-          (todos-save)
-          (todos-show)
-          (bury-buffer buf))
-         ((eq major-mode 'todos-mode)
-          (todos-save)
-          ;; If we just quit archive mode, just burying the buffer
-          ;; in todos-mode would return to archive.
-          (set-window-buffer (selected-window)
-                             (set-buffer (other-buffer)))
-          (bury-buffer buf)))))
+(defun todos-indent ()
+  "Indent from point to `todos-indent-to-here'."
+  (indent-to todos-indent-to-here todos-indent-to-here))
+
+(defcustom todos-show-with-done nil
+  "Non-nil to display done items in all categories."
+  :type 'boolean
+  :group 'todos-display)
 
 ;; -----------------------------------------------------------------------------
-;;; Navigation commands
+;;; Faces
 ;; -----------------------------------------------------------------------------
 
-(defun todos-forward-category (&optional back)
-  "Visit the numerically next category in this Todos file.
-If the current category is the highest numbered, visit the first
-category.  With non-nil argument BACK, visit the numerically
-previous category (the highest numbered one, if the current
-category is the first)."
-  (interactive)
-  (setq todos-category-number
-        (1+ (mod (- todos-category-number (if back 2 0))
-                (length todos-categories))))
-  (when todos-skip-archived-categories
-    (while (and (zerop (todos-get-count 'todo))
-               (zerop (todos-get-count 'done))
-               (not (zerop (todos-get-count 'archived))))
-      (setq todos-category-number
-           (apply (if back '1- '1+) (list todos-category-number)))))
-  (todos-category-select)
-  (goto-char (point-min)))
-
-(defun todos-backward-category ()
-  "Visit the numerically previous category in this Todos file.
-If the current category is the highest numbered, visit the first
-category."
-  (interactive)
-  (todos-forward-category t))
-
-(defun todos-jump-to-category (&optional file where)
-  "Prompt for a category in a Todos file and jump to it.
+(defface todos-mark
+  ;; '((t :inherit font-lock-warning-face))
+  '((((class color)
+      (min-colors 88)
+      (background light))
+     (:weight bold :foreground "Red1"))
+    (((class color)
+      (min-colors 88)
+      (background dark))
+     (:weight bold :foreground "Pink"))
+    (((class color)
+      (min-colors 16)
+      (background light))
+     (:weight bold :foreground "Red1"))
+    (((class color)
+      (min-colors 16)
+      (background dark))
+     (:weight bold :foreground "Pink"))
+    (((class color)
+      (min-colors 8))
+     (:foreground "red"))
+    (t
+     (:weight bold :inverse-video t)))
+  "Face for marks on marked items."
+  :group 'todos-faces)
 
-With non-nil FILE (interactively a prefix argument), prompt for a
-specific Todos file and choose (with TAB completion) a category
-in it to jump to; otherwise, choose and jump to any category in
-either the current Todos file or a file in
-`todos-category-completions-files'.
+(defface todos-prefix-string
+  ;; '((t :inherit font-lock-constant-face))
+  '((((class grayscale) (background light))
+     (:foreground "LightGray" :weight bold :underline t))
+    (((class grayscale) (background dark))
+     (:foreground "Gray50" :weight bold :underline t))
+    (((class color) (min-colors 88) (background light)) (:foreground "dark cyan"))
+    (((class color) (min-colors 88) (background dark)) (:foreground "Aquamarine"))
+    (((class color) (min-colors 16) (background light)) (:foreground "CadetBlue"))
+    (((class color) (min-colors 16) (background dark)) (:foreground "Aquamarine"))
+    (((class color) (min-colors 8)) (:foreground "magenta"))
+    (t (:weight bold :underline t)))
+  "Face for Todos prefix or numerical priority string."
+  :group 'todos-faces)
 
-Also accept a non-existing category name and ask whether to add a
-new category by that name; on confirmation, add it and jump to
-that category, and if option `todos-add-item-if-new-category' is
-non-nil (the default), then prompt for the first item.
+(defface todos-top-priority
+  ;; bold font-lock-comment-face
+  '((default :weight bold)
+    (((class grayscale) (background light)) :foreground "DimGray" :slant italic)
+    (((class grayscale) (background dark)) :foreground "LightGray" :slant italic)
+    (((class color) (min-colors 88) (background light)) :foreground "Firebrick")
+    (((class color) (min-colors 88) (background dark)) :foreground "chocolate1")
+    (((class color) (min-colors 16) (background light)) :foreground "red")
+    (((class color) (min-colors 16) (background dark)) :foreground "red1")
+    (((class color) (min-colors 8) (background light)) :foreground "red")
+    (((class color) (min-colors 8) (background dark)) :foreground "yellow")
+    (t :slant italic))
+  "Face for top priority Todos item numerical priority string.
+The item's priority number string has this face if the number is
+less than or equal the category's top priority setting."
+  :group 'todos-faces)
 
-In noninteractive calls non-nil WHERE specifies either the goal
-category or its file.  If its value is `archive', the choice of
-categories is restricted to the current archive file or the
-archive you were prompted to choose; this is used by
-`todos-jump-to-archive-category'.  If its value is the name of a
-category, jump directly to that category; this is used in Todos
-Categories mode."
-  (interactive "P")
-  ;; If invoked outside of Todos mode and there is not yet any Todos
-  ;; file, initialize one.
-  (if (null todos-files)
-      (todos-show)
-    (let* ((archive (eq where 'archive))
-          (cat (unless archive where))
-          (file0 (when cat             ; We're in Todos Categories mode.
-                   ;; With non-nil `todos-skip-archived-categories'
-                   ;; jump to archive file of a category with only
-                   ;; archived items.
-                   (if (and todos-skip-archived-categories
-                            (zerop (todos-get-count 'todo cat))
-                            (zerop (todos-get-count 'done cat))
-                            (not (zerop (todos-get-count 'archived cat))))
-                       (concat (file-name-sans-extension
-                                todos-current-todos-file) ".toda")
-                     ;; Otherwise, jump to current todos file.
-                     todos-current-todos-file)))
-          (len (length todos-categories))
-          (cat+file (unless cat
-                      (todos-read-category "Jump to category: "
-                                           (if archive 'archive) file)))
-          (add-item (and todos-add-item-if-new-category
-                         (> (length todos-categories) len))))
-      (setq category (or cat (car cat+file)))
-      (unless cat (setq file0 (cdr cat+file)))
-      (with-current-buffer (find-file-noselect file0 'nowarn)
-       (setq todos-current-todos-file file0)
-       ;; If called from Todos Categories mode, clean up before jumping.
-       (if (string= (buffer-name) todos-categories-buffer)
-           (kill-buffer))
-       (set-window-buffer (selected-window)
-                          (set-buffer (find-buffer-visiting file0)))
-       (unless todos-global-current-todos-file
-         (setq todos-global-current-todos-file todos-current-todos-file))
-       (todos-category-number category)
-       (todos-category-select)
-       (goto-char (point-min))
-       (when add-item (todos-basic-insert-item))))))
-
-(defun todos-next-item (&optional count)
-  "Move point down to the beginning of the next item.
-With positive numerical prefix COUNT, move point COUNT items
-downward.
-
-If the category's done items are hidden, this command also moves
-point to the empty line below the last todo item from any higher
-item in the category, i.e., when invoked with or without a prefix
-argument.  If the category's done items are visible, this command
-called with a prefix argument only moves point to a lower item,
-e.g., with point on the last todo item and called with prefix 1,
-it moves point to the first done item; but if called with point
-on the last todo item without a prefix argument, it moves point
-the the empty line above the done items separator."
-  (interactive "p")
-  ;; It's not worth the trouble to allow prefix arg value < 1, since we have
-  ;; the corresponding command.
-  (cond ((and current-prefix-arg (< count 1))
-        (user-error "The prefix argument must be a positive number"))
-       (current-prefix-arg
-        (todos-forward-item count))
-       (t
-        (todos-forward-item))))
-
-(defun todos-previous-item (&optional count)
-  "Move point up to start of item with next higher priority.
-With positive numerical prefix COUNT, move point COUNT items
-upward.
+(defface todos-nondiary
+  ;; '((t :inherit font-lock-type-face))
+  '((((class grayscale) (background light)) :foreground "Gray90" :weight bold)
+    (((class grayscale) (background dark))  :foreground "DimGray" :weight bold)
+    (((class color) (min-colors 88) (background light)) :foreground "ForestGreen")
+    (((class color) (min-colors 88) (background dark))  :foreground "PaleGreen")
+    (((class color) (min-colors 16) (background light)) :foreground "ForestGreen")
+    (((class color) (min-colors 16) (background dark))  :foreground "PaleGreen")
+    (((class color) (min-colors 8)) :foreground "green")
+    (t :weight bold :underline t))
+  "Face for non-diary markers around todo item date/time header."
+  :group 'todos-faces)
 
-If the category's done items are visible, this command called
-with a prefix argument only moves point to a higher item, e.g.,
-with point on the first done item and called with prefix 1, it
-moves to the last todo item; but if called with point on the
-first done item without a prefix argument, it moves point the the
-empty line above the done items separator."
-  (interactive "p")
-  ;; Avoid moving to bob if on the first item but not at bob.
-  (when (> (line-number-at-pos) 1)
-    ;; It's not worth the trouble to allow prefix arg value < 1, since we have
-    ;; the corresponding command.
-    (cond ((and current-prefix-arg (< count 1))
-          (user-error "The prefix argument must be a positive number"))
-         (current-prefix-arg
-          (todos-backward-item count))
-         (t
-          (todos-backward-item)))))
+(defface todos-date
+  '((t :inherit diary))
+  "Face for the date string of a Todos item."
+  :group 'todos-faces)
 
-;; -----------------------------------------------------------------------------
-;;; File editing commands
-;; -----------------------------------------------------------------------------
+(defface todos-time
+  '((t :inherit diary-time))
+  "Face for the time string of a Todos item."
+  :group 'todos-faces)
 
-(defun todos-add-file ()
-  "Name and initialize a new Todos file.
-Interactively, prompt for a category and display it, and if
-option `todos-add-item-if-new-category' is non-nil (the default),
-prompt for the first item.
-Noninteractively, return the name of the new file.
+(defface todos-diary-expired
+  ;; Doesn't contrast enough with todos-date (= diary) face.
+  ;; ;; '((t :inherit warning))
+  ;; '((default :weight bold)
+  ;;   (((class color) (min-colors 16)) :foreground "DarkOrange")
+  ;;   (((class color)) :foreground "yellow"))
+  ;; bold font-lock-function-name-face
+  '((default :weight bold)
+    (((class color) (min-colors 88) (background light)) :foreground "Blue1")
+    (((class color) (min-colors 88) (background dark))  :foreground "LightSkyBlue")
+    (((class color) (min-colors 16) (background light)) :foreground "Blue")
+    (((class color) (min-colors 16) (background dark))  :foreground "LightSkyBlue")
+    (((class color) (min-colors 8)) :foreground "blue")
+    (t :inverse-video t))
+  "Face for expired dates of diary items."
+  :group 'todos-faces)
 
-This command does not save the file to disk; to do that type
-\\[todos-save] or \\[todos-quit]."
-  (interactive)
-  (let ((prompt (concat "Enter name of new Todos file "
-                       "(TAB or SPC to see current names): "))
-       file)
-    (setq file (todos-read-file-name prompt))
-    (with-current-buffer (get-buffer-create file)
-      (erase-buffer)
-      (write-region (point-min) (point-max) file nil 'nomessage nil t)
-      (kill-buffer file))
-    (setq todos-files (funcall todos-files-function))
-    (todos-reevaluate-filelist-defcustoms)
-    (if (called-interactively-p 'any)
-       (progn
-         (set-window-buffer (selected-window)
-                            (set-buffer (find-file-noselect file)))
-         (setq todos-current-todos-file file)
-         (todos-show))
-      file)))
+(defface todos-done-sep
+  ;; '((t :inherit font-lock-builtin-face))
+  '((((class grayscale) (background light)) :foreground "LightGray" :weight bold)
+    (((class grayscale) (background dark))  :foreground "DimGray" :weight bold)
+    (((class color) (min-colors 88) (background light)) :foreground "dark slate blue")
+    (((class color) (min-colors 88) (background dark))  :foreground "LightSteelBlue")
+    (((class color) (min-colors 16) (background light)) :foreground "Orchid")
+    (((class color) (min-colors 16) (background dark)) :foreground "LightSteelBlue")
+    (((class color) (min-colors 8)) :foreground "blue" :weight bold)
+    (t :weight bold))
+  "Face for separator string bewteen done and not done Todos items."
+  :group 'todos-faces)
 
-(defvar todos-edit-buffer "*Todos Edit*"
-  "Name of current buffer in Todos Edit mode.")
+(defface todos-done
+  ;; '((t :inherit font-lock-keyword-face))
+  '((((class grayscale) (background light)) :foreground "LightGray" :weight bold)
+    (((class grayscale) (background dark))  :foreground "DimGray" :weight bold)
+    (((class color) (min-colors 88) (background light)) :foreground "Purple")
+    (((class color) (min-colors 88) (background dark))  :foreground "Cyan1")
+    (((class color) (min-colors 16) (background light)) :foreground "Purple")
+    (((class color) (min-colors 16) (background dark))  :foreground "Cyan")
+    (((class color) (min-colors 8)) :foreground "cyan" :weight bold)
+    (t :weight bold))
+  "Face for done Todos item header string."
+  :group 'todos-faces)
 
-(defun todos-edit-file ()
-  "Put current buffer in `todos-edit-mode'.
-This makes the entire file visible and the buffer writeable and
-you can use the self-insertion keys and standard Emacs editing
-commands to make changes.  To return to Todos mode, type
-\\[todos-edit-quit].  This runs a file format check, signalling
-an error if the format has become invalid.  However, this check
-cannot tell if the number of items changed, which could result in
-the file containing inconsistent information.  For this reason
-this command should be used with caution."
-  (interactive)
-  (widen)
-  (todos-edit-mode)
-  (remove-overlays)
-  (message "%s" (substitute-command-keys
-                (concat "Type \\[todos-edit-quit] to check file format "
-                        "validity and return to Todos mode.\n"))))
+(defface todos-comment
+  ;; '((t :inherit font-lock-comment-face))
+  '((((class grayscale) (background light))
+     :foreground "DimGray" :weight bold :slant italic)
+    (((class grayscale) (background dark))
+     :foreground "LightGray" :weight bold :slant italic)
+    (((class color) (min-colors 88) (background light))
+     :foreground "Firebrick")
+    (((class color) (min-colors 88) (background dark))
+     :foreground "chocolate1")
+    (((class color) (min-colors 16) (background light))
+     :foreground "red")
+    (((class color) (min-colors 16) (background dark))
+     :foreground "red1")
+    (((class color) (min-colors 8) (background light))
+     :foreground "red")
+    (((class color) (min-colors 8) (background dark))
+     :foreground "yellow")
+    (t :weight bold :slant italic))
+  "Face for comments appended to done Todos items."
+  :group 'todos-faces)
 
-;; -----------------------------------------------------------------------------
-;;; Category editing commands
-;; -----------------------------------------------------------------------------
+(defface todos-search
+  ;; '((t :inherit match))
+  '((((class color)
+      (min-colors 88)
+      (background light))
+     (:background "yellow1"))
+    (((class color)
+      (min-colors 88)
+      (background dark))
+     (:background "RoyalBlue3"))
+    (((class color)
+      (min-colors 8)
+      (background light))
+     (:foreground "black" :background "yellow"))
+    (((class color)
+      (min-colors 8)
+      (background dark))
+     (:foreground "white" :background "blue"))
+    (((type tty)
+      (class mono))
+     (:inverse-video t))
+    (t
+     (:background "gray")))
+  "Face for matches found by `todos-search'."
+  :group 'todos-faces)
 
-(defun todos-add-category (&optional file cat)
-  "Add a new category to a Todos file.
+(defface todos-button
+  ;; '((t :inherit widget-field))
+  '((((type tty))
+     (:foreground "black" :background "yellow3"))
+    (((class grayscale color)
+      (background light))
+     (:background "gray85"))
+    (((class grayscale color)
+      (background dark))
+     (:background "dim gray"))
+    (t
+     (:slant italic)))
+  "Face for buttons in table of categories."
+  :group 'todos-faces)
 
-Called interactively with prefix argument FILE, prompt for a file
-and then for a new category to add to that file, otherwise prompt
-just for a category to add to the current Todos file.  After
-adding the category, visit it in Todos mode and if option
-`todos-add-item-if-new-category' is non-nil (the default), prompt
-for the first item.
+(defface todos-sorted-column
+  '((((type tty))
+     (:inverse-video t))
+    (((class color)
+      (background light))
+     (:background "grey85"))
+    (((class color)
+      (background dark))
+      (:background "grey85" :foreground "grey10"))
+    (t
+     (:background "gray")))
+  "Face for sorted column in table of categories."
+  :group 'todos-faces)
 
-Non-interactively, add category CAT to file FILE; if FILE is nil,
-add CAT to the current Todos file.  After adding the category,
-return the new category number."
-  (interactive "P")
-  (let (catfil file0)
-    ;; If cat is passed from caller, don't prompt, unless it is "",
-    ;; which means the file was just added and has no category yet.
-    (if (and cat (> (length cat) 0))
-       (setq file0 (or (and (stringp file) file)
-                       todos-current-todos-file))
-      (setq catfil (todos-read-category "Enter a new category name: "
-                                       'add (when (called-interactively-p 'any)
-                                              file))
-           cat (car catfil)
-           file0 (if (called-interactively-p 'any)
-                     (cdr catfil)
-                   file)))
-    (find-file file0)
-    (let ((counts (make-vector 4 0))   ; [todo diary done archived]
-         (num (1+ (length todos-categories)))
-         (buffer-read-only nil))
-      (setq todos-current-todos-file file0)
-      (setq todos-categories (append todos-categories
-                                    (list (cons cat counts))))
-      (widen)
-      (goto-char (point-max))
-      (save-excursion                  ; Save point for todos-category-select.
-       (insert todos-category-beg cat "\n\n" todos-category-done "\n"))
-      (todos-update-categories-sexp)
-      ;; If invoked by user, display the newly added category, if
-      ;; called programmatically return the category number to the
-      ;; caller.
-      (if (called-interactively-p 'any)
-         (progn
-           (setq todos-category-number num)
-           (todos-category-select)
-           (when todos-add-item-if-new-category
-             (todos-basic-insert-item)))
-       num))))
+(defface todos-archived-only
+  ;; '((t (:inherit (shadow))))
+  '((((class color)
+      (background light))
+     (:foreground "grey50"))
+    (((class color)
+      (background dark))
+     (:foreground "grey70"))
+    (t
+     (:foreground "gray")))
+  "Face for archived-only category names in table of categories."
+  :group 'todos-faces)
 
-(defun todos-rename-category ()
-  "Rename current Todos category.
-If this file has an archive containing this category, rename the
-category there as well."
-  (interactive)
-  (let* ((cat (todos-current-category))
-        (new (read-from-minibuffer
-              (format "Rename category \"%s\" to: " cat))))
-    (setq new (todos-validate-name new 'category))
-    (let* ((ofile todos-current-todos-file)
-          (archive (concat (file-name-sans-extension ofile) ".toda"))
-          (buffers (append (list ofile)
-                           (unless (zerop (todos-get-count 'archived cat))
-                             (list archive)))))
-      (dolist (buf buffers)
-       (with-current-buffer (find-file-noselect buf)
-         (let (buffer-read-only)
-           (setq todos-categories (todos-set-categories))
-           (save-excursion
-             (save-restriction
-               (setcar (assoc cat todos-categories) new)
-               (widen)
-               (goto-char (point-min))
-               (todos-update-categories-sexp)
-               (re-search-forward (concat (regexp-quote todos-category-beg)
-                                          "\\(" (regexp-quote cat) "\\)\n")
-                                  nil t)
-               (replace-match new t t nil 1)))))))
-    (force-mode-line-update))
-  (save-excursion (todos-category-select)))
+(defface todos-category-string
+    ;; '((t :inherit font-lock-type-face))
+  '((((class grayscale) (background light)) :foreground "Gray90" :weight bold)
+    (((class grayscale) (background dark))  :foreground "DimGray" :weight bold)
+    (((class color) (min-colors 88) (background light)) :foreground "ForestGreen")
+    (((class color) (min-colors 88) (background dark))  :foreground "PaleGreen")
+    (((class color) (min-colors 16) (background light)) :foreground "ForestGreen")
+    (((class color) (min-colors 16) (background dark))  :foreground "PaleGreen")
+    (((class color) (min-colors 8)) :foreground "green")
+    (t :weight bold :underline t))
+  "Face for category-file header in Todos Filtered Items mode."
+  :group 'todos-faces)
 
-(defun todos-delete-category (&optional arg)
-  "Delete current Todos category provided it is empty.
-With ARG non-nil delete the category unconditionally,
-i.e. including all existing todo and done items."
-  (interactive "P")
-  (let* ((file todos-current-todos-file)
-        (cat (todos-current-category))
-        (todo (todos-get-count 'todo cat))
-        (done (todos-get-count 'done cat))
-        (archived (todos-get-count 'archived cat)))
-    (if (and (not arg)
-            (or (> todo 0) (> done 0)))
-       (message "%s" (substitute-command-keys
-                      (concat "To delete a non-empty category, "
-                              "type C-u \\[todos-delete-category].")))
-      (when (cond ((= (length todos-categories) 1)
-                  (todos-y-or-n-p
-                   (concat "This is the only category in this file; "
-                           "deleting it will also delete the file.\n"
-                           "Do you want to proceed? ")))
-                 ((> archived 0)
-                  (todos-y-or-n-p (concat "This category has archived items; "
-                                    "the archived category will remain\n"
-                                    "after deleting the todo category.  "
-                                    "Do you still want to delete it\n"
-                                    "(see `todos-skip-archived-categories' "
-                                    "for another option)? ")))
-                 (t
-                  (todos-y-or-n-p (concat "Permanently remove category \"" cat
-                                    "\"" (and arg " and all its entries")
-                                    "? "))))
-       (widen)
-       (let ((buffer-read-only)
-             (beg (re-search-backward
-                   (concat "^" (regexp-quote (concat todos-category-beg cat))
-                           "\n") nil t))
-             (end (if (re-search-forward
-                       (concat "\n\\(" (regexp-quote todos-category-beg)
-                               ".*\n\\)") nil t)
-                      (match-beginning 1)
-                    (point-max))))
-         (remove-overlays beg end)
-         (delete-region beg end)
-         (if (= (length todos-categories) 1)
-             ;; If deleted category was the only one, delete the file.
-             (progn
-               (todos-reevaluate-filelist-defcustoms)
-               ;; Skip confirming killing the archive buffer if it has been
-               ;; modified and not saved.
-               (set-buffer-modified-p nil)
-               (delete-file file)
-               (kill-buffer)
-               (message "Deleted Todos file %s." file))
-           (setq todos-categories (delete (assoc cat todos-categories)
-                                              todos-categories))
-           (todos-update-categories-sexp)
-           (setq todos-category-number
-                 (1+ (mod todos-category-number (length todos-categories))))
-           (todos-category-select)
-           (goto-char (point-min))
-           (message "Deleted category %s." cat)))))))
+;; -----------------------------------------------------------------------------
+;;; Entering and exiting Todos
+;; -----------------------------------------------------------------------------
 
-(defun todos-move-category ()
-  "Move current category to a different Todos file.
-If current category has archived items, also move those to the
-archive of the file moved to, creating it if it does not exist."
-  (interactive)
-  (when (or (> (length todos-categories) 1)
-           (todos-y-or-n-p (concat "This is the only category in this file; "
-                                   "moving it will also delete the file.\n"
-                                   "Do you want to proceed? ")))
-    (let* ((ofile todos-current-todos-file)
-          (cat (todos-current-category))
-          (nfile (todos-read-file-name
-                  "Choose a Todos file to move this category to: " nil t))
-          (archive (concat (file-name-sans-extension ofile) ".toda"))
-          (buffers (append (list ofile)
-                           (unless (zerop (todos-get-count 'archived cat))
-                             (list archive))))
-          new)
-      (while (equal (file-truename nfile) (file-truename ofile))
-       (setq nfile (todos-read-file-name
-                    "Choose a file distinct from this file: " nil t)))
-      (dolist (buf buffers)
-       (with-current-buffer (find-file-noselect buf)
-         (widen)
-         (goto-char (point-max))
-         (let* ((beg (re-search-backward
-                      (concat "^"
-                              (regexp-quote (concat todos-category-beg cat))
-                              "$")
-                      nil t))
-                (end (if (re-search-forward
-                          (concat "^" (regexp-quote todos-category-beg))
-                          nil t 2)
-                         (match-beginning 0)
-                       (point-max)))
-                (content (buffer-substring-no-properties beg end))
-                (counts (cdr (assoc cat todos-categories)))
-                buffer-read-only)
-           ;; Move the category to the new file.  Also update or create
-           ;; archive file if necessary.
-           (with-current-buffer
-               (find-file-noselect
-                ;; Regenerate todos-archives in case there
-                ;; is a newly created archive.
-                (if (member buf (funcall todos-files-function t))
-                    (concat (file-name-sans-extension nfile) ".toda")
-                  nfile))
-             (let* ((nfile-short (todos-short-file-name nfile))
-                    (prompt (concat
-                             (format "Todos file \"%s\" already has "
-                                     nfile-short)
-                             (format "the category \"%s\";\n" cat)
-                             "enter a new category name: "))
-                    buffer-read-only)
-               (widen)
-               (goto-char (point-max))
-               (insert content)
-               ;; If the file moved to has a category with the same
-               ;; name, rename the moved category.
-               (when (assoc cat todos-categories)
-                 (unless (member (file-truename (buffer-file-name))
-                                 (funcall todos-files-function t))
-                   (setq new (read-from-minibuffer prompt))
-                   (setq new (todos-validate-name new 'category))))
-               ;; Replace old with new name in Todos and archive files.
-               (when new
-                 (goto-char (point-max))
-                 (re-search-backward
-                  (concat "^" (regexp-quote todos-category-beg)
-                          "\\(" (regexp-quote cat) "\\)$") nil t)
-                 (replace-match new nil nil nil 1)))
-             (setq todos-categories
-                   (append todos-categories (list (cons new counts))))
-             (todos-update-categories-sexp)
-             ;; If archive was just created, save it to avoid "File
-             ;; <xyz> no longer exists!" message on invoking
-             ;; `todos-view-archived-items'.
-             (unless (file-exists-p (buffer-file-name))
-               (save-buffer))
-             (todos-category-number (or new cat))
-             (todos-category-select))
-           ;; Delete the category from the old file, and if that was the
-           ;; last category, delete the file.  Also handle archive file
-           ;; if necessary.
-           (remove-overlays beg end)
-           (delete-region beg end)
-           (goto-char (point-min))
-           ;; Put point after todos-categories sexp.
-           (forward-line)
-           (if (eobp)          ; Aside from sexp, file is empty.
-               (progn
-                 ;; Skip confirming killing the archive buffer.
-                 (set-buffer-modified-p nil)
-                 (delete-file todos-current-todos-file)
-                 (kill-buffer)
-                 (when (member todos-current-todos-file todos-files)
-                   (todos-reevaluate-filelist-defcustoms)))
-             (setq todos-categories (delete (assoc cat todos-categories)
-                                            todos-categories))
-             (todos-update-categories-sexp)
-             (todos-category-select)))))
-      (set-window-buffer (selected-window)
-                        (set-buffer (find-file-noselect nfile)))
-      (todos-category-number (or new cat))
-      (todos-category-select))))
-
-(defun todos-merge-category (&optional file)
-  "Merge current category into another existing category.
+(defcustom todos-visit-files-commands (list 'find-file 'dired-find-file)
+  "List of file finding commands for `todos-display-as-todos-file'.
+Invoking these commands to visit a Todos or Todos Archive file
+calls `todos-show' or `todos-find-archive', so that the file is
+displayed correctly."
+  :type '(repeat function)
+  :group 'todos)
 
-With prefix argument FILE, prompt for a specific Todos file and
-choose (with TAB completion) a category in it to merge into;
-otherwise, choose and merge into a category in either the
-current Todos file or a file in `todos-category-completions-files'.
+(defun todos-short-file-name (file)
+  "Return short form of Todos FILE.
+This lacks the extension and directory components."
+  (when (stringp file)
+    (file-name-sans-extension (file-name-nondirectory file))))
 
-After merging, the current category's todo and done items are
-appended to the chosen goal category's todo and done items,
-respectively.  The goal category becomes the current category,
-and the previous current category is deleted.
+(defcustom todos-default-todos-file (todos-short-file-name
+                                    (car (funcall todos-files-function)))
+  "Todos file visited by first session invocation of `todos-show'."
+  :type `(radio ,@(mapcar (lambda (f) (list 'const f))
+                         (mapcar 'todos-short-file-name
+                                 (funcall todos-files-function))))
+  :group 'todos)
 
-If both the first and goal categories also have archived items,
-the former are merged to the latter.  If only the first category
-has archived items, the archived category is renamed to the goal
-category."
-  (interactive "P")
-  (let* ((tfile todos-current-todos-file)
-        (cat (todos-current-category))
-        (cat+file (todos-read-category "Merge into category: " 'todo file))
-        (goal (car cat+file))
-        (gfile  (cdr cat+file))
-        (archive (concat (file-name-sans-extension (if file gfile tfile))
-                         ".toda"))
-        archived-count here)
-    ;; Merge in todo file.
-    (with-current-buffer (get-buffer (find-file-noselect tfile))
-      (widen)
-      (let* ((buffer-read-only nil)
-            (cbeg (progn
-                    (re-search-backward
-                     (concat "^" (regexp-quote todos-category-beg)) nil t)
-                    (point-marker)))
-            (tbeg (progn (forward-line) (point-marker)))
-            (dbeg (progn
-                    (re-search-forward
-                     (concat "^" (regexp-quote todos-category-done)) nil t)
-                    (forward-line) (point-marker)))
-            ;; Omit empty line between todo and done items.
-            (tend (progn (forward-line -2) (point-marker)))
-            (cend (progn
-                    (if (re-search-forward
-                         (concat "^" (regexp-quote todos-category-beg)) nil t)
-                        (progn
-                          (goto-char (match-beginning 0))
-                          (point-marker))
-                      (point-max-marker))))
-            (todo (buffer-substring-no-properties tbeg tend))
-            (done (buffer-substring-no-properties dbeg cend)))
-       (goto-char (point-min))
-       ;; Merge any todo items.
-       (unless (zerop (length todo))
-         (re-search-forward
-          (concat "^" (regexp-quote (concat todos-category-beg goal)) "$")
-          nil t)
-         (re-search-forward
-          (concat "^" (regexp-quote todos-category-done)) nil t)
-         (forward-line -1)
-         (setq here (point-marker))
-         (insert todo)
-         (todos-update-count 'todo (todos-get-count 'todo cat) goal))
-       ;; Merge any done items.
-       (unless (zerop (length done))
-         (goto-char (if (re-search-forward
-                         (concat "^" (regexp-quote todos-category-beg)) nil t)
-                        (match-beginning 0)
-                      (point-max)))
-         (when (zerop (length todo)) (setq here (point-marker)))
-         (insert done)
-         (todos-update-count 'done (todos-get-count 'done cat) goal))
-       (remove-overlays cbeg cend)
-       (delete-region cbeg cend)
-       (setq todos-categories (delete (assoc cat todos-categories)
-                                      todos-categories))
-       (todos-update-categories-sexp)
-       (mapc (lambda (m) (set-marker m nil)) (list cbeg tbeg dbeg tend cend))))
-    (when (file-exists-p archive)
-      ;; Merge in archive file.
-      (with-current-buffer (get-buffer (find-file-noselect archive))
-       (widen)
-       (goto-char (point-min))
-       (let ((buffer-read-only nil)
-             (cbeg (save-excursion
-                     (when (re-search-forward
-                            (concat "^" (regexp-quote
-                                         (concat todos-category-beg cat)) "$")
-                            nil t)
-                       (goto-char (match-beginning 0))
-                       (point-marker))))
-             (gbeg (save-excursion
-                     (when (re-search-forward
-                            (concat "^" (regexp-quote
-                                         (concat todos-category-beg goal)) "$")
-                            nil t)
-                       (goto-char (match-beginning 0))
-                       (point-marker))))
-             cend carch)
-         (when cbeg
-           (setq archived-count (todos-get-count 'done cat))
-           (setq cend (save-excursion
-                        (if (re-search-forward
-                             (concat "^" (regexp-quote todos-category-beg))
-                             nil t)
-                            (match-beginning 0)
-                          (point-max))))
-           (setq carch (save-excursion (goto-char cbeg) (forward-line)
-                         (buffer-substring-no-properties (point) cend)))
-           ;; If both categories of the merge have archived items, merge the
-           ;; source items to the goal items, else "merge" by renaming the
-           ;; source category to goal.
-           (if gbeg
-               (progn
-                 (goto-char (if (re-search-forward
-                                 (concat "^" (regexp-quote todos-category-beg))
-                                 nil t)
-                                (match-beginning 0)
-                              (point-max)))
-                 (insert carch)
-                 (remove-overlays cbeg cend)
-                 (delete-region cbeg cend))
-             (goto-char cbeg)
-             (search-forward cat)
-             (replace-match goal))
-           (setq todos-categories (todos-make-categories-list t))
-           (todos-update-categories-sexp)))))
-    (with-current-buffer (get-file-buffer tfile)
-      (when archived-count
-       (unless (zerop archived-count)
-         (todos-update-count 'archived archived-count goal)
-         (todos-update-categories-sexp)))
-      (todos-category-number goal)
-      ;; If there are only merged done items, show them.
-      (let ((todos-show-with-done (zerop (todos-get-count 'todo goal))))
-       (todos-category-select)
-       ;; Put point on the first merged item.
-       (goto-char here)))
-    (set-marker here nil)))
+(defcustom todos-show-current-file t
+  "Non-nil to make `todos-show' visit the current Todos file.
+Otherwise, `todos-show' always visits `todos-default-todos-file'."
+  :type 'boolean
+  :initialize 'custom-initialize-default
+  :set 'todos-set-show-current-file
+  :group 'todos)
 
-;; -----------------------------------------------------------------------------
-;;; Item marking
-;; -----------------------------------------------------------------------------
+(defcustom todos-show-first 'first
+  "What action to take on first use of `todos-show' on a file."
+  :type '(choice (const :tag "Show first category" first)
+                (const :tag "Show table of categories" table)
+                (const :tag "Show top priorities" top)
+                (const :tag "Show diary items" diary)
+                (const :tag "Show regexp items" regexp))
+  :group 'todos)
 
-(defcustom todos-item-mark "*"
-  "String used to mark items.
-To ensure item marking works, change the value of this option
-only when no items are marked."
-  :type '(string :validate
-                (lambda (widget)
-                  (when (string= (widget-value widget) todos-prefix)
-                    (widget-put
-                     widget :error
-                     "Invalid value: must be distinct from `todos-prefix'")
-                    widget)))
-  :set (lambda (symbol value)
-        (custom-set-default symbol (propertize value 'face 'todos-mark)))
+(defcustom todos-add-item-if-new-category t
+  "Non-nil to prompt for an item after adding a new category."
+  :type 'boolean
   :group 'todos-edit)
 
-(defun todos-toggle-mark-item (&optional n)
-  "Mark item with `todos-item-mark' if unmarked, otherwise unmark it.
-With a positive numerical prefix argument N, change the
-marking of the next N items."
-  (interactive "p")
-  (when (todos-item-string)
-    (unless (> n 1) (setq n 1))
-    (dotimes (i n)
-      (let* ((cat (todos-current-category))
-            (marks (assoc cat todos-categories-with-marks))
-            (ov (progn
-                  (unless (looking-at todos-item-start)
-                    (todos-item-start))
-                  (todos-get-overlay 'prefix)))
-            (pref (overlay-get ov 'before-string)))
-       (if (todos-marked-item-p)
-           (progn
-             (overlay-put ov 'before-string (substring pref 1))
-             (if (= (cdr marks) 1)     ; Deleted last mark in this category.
-                 (setq todos-categories-with-marks
-                       (assq-delete-all cat todos-categories-with-marks))
-               (setcdr marks (1- (cdr marks)))))
-         (overlay-put ov 'before-string (concat todos-item-mark pref))
-         (if marks
-             (setcdr marks (1+ (cdr marks)))
-           (push (cons cat 1) todos-categories-with-marks))))
-      (todos-forward-item))))
+(defcustom todos-initial-file "Todo"
+  "Default file name offered on adding first Todos file."
+  :type 'string
+  :group 'todos)
 
-(defun todos-mark-category ()
-  "Mark all visiblw items in this category with `todos-item-mark'."
-  (interactive)
-  (let* ((cat (todos-current-category))
-        (marks (assoc cat todos-categories-with-marks)))
-    (save-excursion
-      (goto-char (point-min))
-      (while (not (eobp))
-       (let* ((ov (todos-get-overlay 'prefix))
-              (pref (overlay-get ov 'before-string)))
-         (unless (todos-marked-item-p)
-           (overlay-put ov 'before-string (concat todos-item-mark pref))
-           (if marks
-               (setcdr marks (1+ (cdr marks)))
-             (push (cons cat 1) todos-categories-with-marks))))
-       (todos-forward-item)))))
-
-(defun todos-unmark-category ()
-  "Remove `todos-item-mark' from all visible items in this category."
-  (interactive)
-  (let* ((cat (todos-current-category))
-        (marks (assoc cat todos-categories-with-marks)))
-    (save-excursion
-      (goto-char (point-min))
-      (while (not (eobp))
-       (let* ((ov (todos-get-overlay 'prefix))
-              ;; No overlay on empty line between todo and done items.
-              (pref (when ov (overlay-get ov 'before-string))))
-         (when (todos-marked-item-p)
-           (overlay-put ov 'before-string (substring pref 1)))
-         (todos-forward-item))))
-    (setq todos-categories-with-marks
-         (delq marks todos-categories-with-marks))))
+(defcustom todos-initial-category "Todo"
+  "Default category name offered on initializing a new Todos file."
+  :type 'string
+  :group 'todos)
 
-;; -----------------------------------------------------------------------------
-;;; Item editing options
-;; -----------------------------------------------------------------------------
+(defcustom todos-category-completions-files nil
+  "List of files for building `todos-read-category' completions."
+  :type `(set ,@(mapcar (lambda (f) (list 'const f))
+                       (mapcar 'todos-short-file-name
+                               (funcall todos-files-function))))
+  :group 'todos)
 
-(defcustom todos-add-item-if-new-category t
-  "Non-nil to prompt for an item after adding a new category."
+(defcustom todos-completion-ignore-case nil
+  "Non-nil means case is ignored by `todos-read-*' functions."
   :type 'boolean
-  :group 'todos-edit)
+  :group 'todos)
 
-(defcustom todos-include-in-diary nil
-  "Non-nil to allow new Todo items to be included in the diary."
-  :type 'boolean
-  :group 'todos-edit)
+(defun todos-show (&optional solicit-file)
+  "Visit a Todos file and display one of its categories.
 
-(defcustom todos-diary-nonmarking nil
-  "Non-nil to insert new Todo diary items as nonmarking by default.
-This appends `diary-nonmarking-symbol' to the front of an item on
-insertion provided it doesn't begin with `todos-nondiary-marker'."
-  :type 'boolean
-  :group 'todos-edit)
+When invoked in Todos mode, prompt for which todo file to visit.
+When invoked outside of Todos mode with non-nil prefix argument
+SOLICIT-FILE prompt for which todo file to visit; otherwise visit
+`todos-default-todos-file'.  Subsequent invocations from outside
+of Todos mode revisit this file or, with option
+`todos-show-current-file' non-nil (the default), whichever Todos
+file was last visited.
 
-(defcustom todos-nondiary-marker '("[" "]")
-  "List of strings surrounding item date to block diary inclusion.
-The first string is inserted before the item date and must be a
-non-empty string that does not match a diary date in order to
-have its intended effect.  The second string is inserted after
-the diary date."
-  :type '(list string string)
-  :group 'todos-edit
-  :initialize 'custom-initialize-default
-  :set 'todos-reset-nondiary-marker)
+Calling this command before any Todos file exists prompts for a
+file name and an initial category (defaulting to
+`todos-initial-file' and `todos-initial-category'), creates both
+of these, visits the file and displays the category, and if
+option `todos-add-item-if-new-category' is non-nil (the default),
+prompts for the first item.
 
-(defcustom todos-always-add-time-string nil
-  "Non-nil adds current time to a new item's date header by default.
-When the Todos insertion commands have a non-nil \"maybe-notime\"
-argument, this reverses the effect of
-`todos-always-add-time-string': if t, these commands omit the
-current time, if nil, they include it."
-  :type 'boolean
-  :group 'todos-edit)
+The first invocation of this command on an existing Todos file
+interacts with the option `todos-show-first': if its value is
+`first' (the default), show the first category in the file; if
+its value is `table', show the table of categories in the file;
+if its value is one of `top', `diary' or `regexp', show the
+corresponding saved top priorities, diary items, or regexp items
+file, if any.  Subsequent invocations always show the file's
+current (i.e., last displayed) category.
 
-(defcustom todos-use-only-highlighted-region t
-  "Non-nil to enable inserting only highlighted region as new item."
-  :type 'boolean
-  :group 'todos-edit)
+In Todos mode just the category's unfinished todo items are shown
+by default.  The done items are hidden, but typing
+`\\[todos-toggle-view-done-items]' displays them below the todo
+items.  With non-nil user option `todos-show-with-done' both todo
+and done items are always shown on visiting a category.
 
-(defcustom todos-undo-item-omit-comment 'ask
-  "Whether to omit done item comment on undoing the item.
-Nil means never omit the comment, t means always omit it, `ask'
-means prompt user and omit comment only on confirmation."
-  :type '(choice (const :tag "Never" nil)
-                (const :tag "Always" t)
-                (const :tag "Ask" ask))
-  :group 'todos-edit)
+Invoking this command in Todos Archive mode visits the
+corresponding Todos file, displaying the corresponding category."
+  (interactive "P")
+  (let* ((cat)
+        (show-first todos-show-first)
+        (file (cond ((or solicit-file
+                         (and (called-interactively-p 'any)
+                              (memq major-mode '(todos-mode
+                                                 todos-archive-mode
+                                                 todos-filtered-items-mode))))
+                     (if (funcall todos-files-function)
+                         (todos-read-file-name "Choose a Todos file to visit: "
+                                               nil t)
+                       (user-error "There are no Todos files")))
+                    ((and (eq major-mode 'todos-archive-mode)
+                          ;; Called noninteractively via todos-quit
+                          ;; to jump to corresponding category in
+                          ;; todo file.
+                          (not (called-interactively-p 'any)))
+                     (setq cat (todos-current-category))
+                     (concat (file-name-sans-extension
+                              todos-current-todos-file) ".todo"))
+                    (t
+                     (or todos-current-todos-file
+                         (and todos-show-current-file
+                              todos-global-current-todos-file)
+                         (todos-absolute-file-name todos-default-todos-file)
+                         (todos-add-file)))))
+        add-item first-file)
+    (unless todos-default-todos-file
+      ;; We just initialized the first todo file, so make it the default.
+      (setq todos-default-todos-file (todos-short-file-name file)
+           first-file t)
+      (todos-reevaluate-default-file-defcustom))
+    (unless (member file todos-visited)
+      ;; Can't setq t-c-t-f here, otherwise wrong file shown when
+      ;; todos-show is called from todos-show-categories-table.
+      (let ((todos-current-todos-file file))
+       (cond ((eq todos-show-first 'table)
+              (todos-show-categories-table))
+             ((memq todos-show-first '(top diary regexp))
+              (let* ((shortf (todos-short-file-name file))
+                     (fi-file (todos-absolute-file-name
+                               shortf todos-show-first)))
+                (when (eq todos-show-first 'regexp)
+                  (let ((rxfiles (directory-files todos-directory t
+                                                  ".*\\.todr$" t)))
+                    (when (and rxfiles (> (length rxfiles) 1))
+                      (let ((rxf (mapcar 'todos-short-file-name rxfiles)))
+                        (setq fi-file (todos-absolute-file-name
+                                       (completing-read
+                                        "Choose a regexp items file: "
+                                        rxf) 'regexp))))))
+                (if (file-exists-p fi-file)
+                    (set-window-buffer
+                     (selected-window)
+                     (set-buffer (find-file-noselect fi-file 'nowarn)))
+                  (message "There is no %s file for %s"
+                           (cond ((eq todos-show-first 'top)
+                                  "top priorities")
+                                 ((eq todos-show-first 'diary)
+                                  "diary items")
+                                 ((eq todos-show-first 'regexp)
+                                  "regexp items"))
+                           shortf)
+                  (setq todos-show-first 'first)))))))
+    (when (or (member file todos-visited)
+             (eq todos-show-first 'first))
+      (set-window-buffer (selected-window)
+                        (set-buffer (find-file-noselect file 'nowarn)))
+      ;; When quitting archive file, show corresponding category in
+      ;; Todos file, if it exists.
+      (when (assoc cat todos-categories)
+       (setq todos-category-number (todos-category-number cat)))
+      ;; If this is a new Todos file, add its first category.
+      (when (zerop (buffer-size))
+       (let (cat-added)
+         (unwind-protect
+             (setq todos-category-number
+                   (todos-add-category todos-current-todos-file "")
+                   add-item todos-add-item-if-new-category
+                   cat-added t)
+           (if cat-added
+               ;; If the category was added, save the file now, so we
+               ;; don't risk having an empty todo file, which would
+               ;; signal an error if we tried to visit it later,
+               ;; since doing that looks for category boundaries.
+               (save-buffer 0)
+             ;; If user cancels before adding the category, clean up
+             ;; and exit, so we have a fresh slate the next time.
+             (delete-file file)
+             (setq todos-files (delete file todos-files))
+             (when first-file
+               (setq todos-default-todos-file nil
+                     todos-current-todos-file nil))
+             (kill-buffer)
+             (keyboard-quit)))))
+      (save-excursion (todos-category-select))
+      (when add-item (todos-basic-insert-item)))
+    (setq todos-show-first show-first)
+    (add-to-list 'todos-visited file)))
+
+(defun todos-save ()
+  "Save the current Todos file."
+  (interactive)
+  (cond ((eq major-mode 'todos-filtered-items-mode)
+        (todos-check-filtered-items-file)
+        (todos-save-filtered-items-buffer))
+       (t
+        (save-buffer))))
+
+(defvar todos-descending-counts)
+
+(defun todos-quit ()
+  "Exit the current Todos-related buffer.
+Depending on the specific mode, this either kills the buffer or
+buries it and restores state as needed."
+  (interactive)
+  (let ((buf (current-buffer)))
+    (cond ((eq major-mode 'todos-categories-mode)
+          ;; Postpone killing buffer till after calling todos-show, to
+          ;; prevent killing todos-mode buffer.
+          (setq todos-descending-counts nil)
+          ;; Ensure todos-show calls todos-show-categories-table only on
+          ;; first invocation per file.
+          (when (eq todos-show-first 'table)
+            (add-to-list 'todos-visited todos-current-todos-file))
+          (todos-show)
+          (kill-buffer buf))
+         ((eq major-mode 'todos-filtered-items-mode)
+          (kill-buffer)
+          (unless (eq major-mode 'todos-mode) (todos-show)))
+         ((eq major-mode 'todos-archive-mode)
+          ;; Have to write a newly created archive to file to avoid
+          ;; subsequent errors.
+          (todos-save)
+          (todos-show)
+          (bury-buffer buf))
+         ((eq major-mode 'todos-mode)
+          (todos-save)
+          ;; If we just quit archive mode, just burying the buffer
+          ;; in todos-mode would return to archive.
+          (set-window-buffer (selected-window)
+                             (set-buffer (other-buffer)))
+          (bury-buffer buf)))))
 
 ;; -----------------------------------------------------------------------------
-;;; Item editing commands
+;;; Navigation between and within categories
 ;; -----------------------------------------------------------------------------
 
-(defun todos-basic-insert-item (&optional arg diary nonmarking date-type time
-                                   region-or-here)
-  "Insert a new Todo item into a category.
-This is the function from which the generated Todos item
-insertion commands derive.
-
-The generated commands have mnenomic key bindings based on the
-arguments' values and their order in the command's argument list,
-as follows: (1) for DIARY `d', (2) for NONMARKING `k', (3) for
-DATE-TYPE either `c' for calendar or `d' for date or `n' for
-weekday name, (4) for TIME `t', (5) for REGION-OR-HERE either `r'
-for region or `h' for here.  Sequences of these keys are appended
-to the insertion prefix key `i'.  Keys that allow a following
-key (i.e., any but `r' or `h') must be doubled when used finally.
-For example, the command bound to the key sequence `i y h' will
-insert a new item with today's date, marked according to the
-DIARY argument described below, and with priority according to
-the HERE argument; `i y y' does the same except that the priority
-is not given by HERE but by prompting.
+(defcustom todos-skip-archived-categories nil
+  "Non-nil to handle categories with only archived items specially.
 
-In command invocations, ARG is passed as a prefix argument as
-follows.  With no prefix argument, add the item to the current
-category; with one prefix argument (`C-u'), prompt for a category
-from the current Todos file; with two prefix arguments (`C-u C-u'),
-first prompt for a Todos file, then a category in that file.  If
-a non-existing category is entered, ask whether to add it to the
-Todos file; if answered affirmatively, add the category and
-insert the item there.
+Sequential category navigation using \\[todos-forward-category]
+or \\[todos-backward-category] skips categories that contain only
+archived items.  Other commands still recognize these categories.
+In Todos Categories mode (\\[todos-show-categories-table]) these
+categories shown in `todos-archived-only' face and pressing the
+category button visits the category in the archive instead of the
+todo file."
+  :type 'boolean
+  :group 'todos-display)
 
-The remaining arguments are set or left nil by the generated item
-insertion commands; their meanings are described in the follows
-paragraphs.
+(defun todos-forward-category (&optional back)
+  "Visit the numerically next category in this Todos file.
+If the current category is the highest numbered, visit the first
+category.  With non-nil argument BACK, visit the numerically
+previous category (the highest numbered one, if the current
+category is the first)."
+  (interactive)
+  (setq todos-category-number
+        (1+ (mod (- todos-category-number (if back 2 0))
+                (length todos-categories))))
+  (when todos-skip-archived-categories
+    (while (and (zerop (todos-get-count 'todo))
+               (zerop (todos-get-count 'done))
+               (not (zerop (todos-get-count 'archived))))
+      (setq todos-category-number
+           (apply (if back '1- '1+) (list todos-category-number)))))
+  (todos-category-select)
+  (goto-char (point-min)))
 
-When argument DIARY is non-nil, this overrides the intent of the
-user option `todos-include-in-diary' for this item: if
-`todos-include-in-diary' is nil, include the item in the Fancy
-Diary display, and if it is non-nil, exclude the item from the
-Fancy Diary display.  When DIARY is nil, `todos-include-in-diary'
-has its intended effect.
+(defun todos-backward-category ()
+  "Visit the numerically previous category in this Todos file.
+If the current category is the highest numbered, visit the first
+category."
+  (interactive)
+  (todos-forward-category t))
 
-When the item is included in the Fancy Diary display and the
-argument NONMARKING is non-nil, this overrides the intent of the
-user option `todos-diary-nonmarking' for this item: if
-`todos-diary-nonmarking' is nil, append `diary-nonmarking-symbol'
-to the item, and if it is non-nil, omit `diary-nonmarking-symbol'.
+(defvar todos-categories-buffer)
 
-The argument DATE-TYPE determines the content of the item's
-mandatory date header string and how it is added:
-- If DATE-TYPE is the symbol `calendar', the Calendar pops up and
-  when the user puts the cursor on a date and hits RET, that
-  date, in the format set by `calendar-date-display-form',
-  becomes the date in the header.
-- If DATE-TYPE is a string matching the regexp
-  `todos-date-pattern', that string becomes the date in the
-  header.  This case is for the command
-  `todos-insert-item-from-calendar' which is called from the
-  Calendar.
-- If DATE-TYPE is the symbol `date', the header contains the date
-  in the format set by `calendar-date-display-form', with year,
-  month and day individually prompted for (month with tab
-  completion).
-- If DATE-TYPE is the symbol `dayname' the header contains a
-  weekday name instead of a date, prompted for with tab
-  completion.
-- If DATE-TYPE has any other value (including nil or none) the
-  header contains the current date (in the format set by
-  `calendar-date-display-form').
+(defun todos-jump-to-category (&optional file where)
+  "Prompt for a category in a Todos file and jump to it.
 
-With non-nil argument TIME prompt for a time string, which must
-match `diary-time-regexp'.  Typing `<return>' at the prompt
-returns the current time, if the user option
-`todos-always-add-time-string' is non-nil, otherwise the empty
-string (i.e., no time string).  If TIME is absent or nil, add or
-omit the current time string according as
-`todos-always-add-time-string' is non-nil or nil, respectively.
+With non-nil FILE (interactively a prefix argument), prompt for a
+specific Todos file and choose (with TAB completion) a category
+in it to jump to; otherwise, choose and jump to any category in
+either the current Todos file or a file in
+`todos-category-completions-files'.
 
-The argument REGION-OR-HERE determines the source and location of
-the new item:
-- If the REGION-OR-HERE is the symbol `here', prompt for the text of
-  the new item and, if the command was invoked with point in the todo
-  items section of the current category, give the new item the
-  priority of the item at point, lowering the latter's priority and
-  the priority of the remaining items.  If point is in the done items
-  section of the category, insert the new item as the first todo item
-  in the category.  Likewise, if the command with `here' is invoked
-  outside of the current category, jump to the chosen category and
-  insert the new item as the first item in the category.
-- If REGION-OR-HERE is the symbol `region', use the region of the
-  current buffer as the text of the new item, depending on the
-  value of user option `todos-use-only-highlighted-region': if
-  this is non-nil, then use the region only when it is
-  highlighted; otherwise, use the region regardless of
-  highlighting.  An error is signalled if there is no region in
-  the current buffer.  Prompt for the item's priority in the
-  category (an integer between 1 and one more than the number of
-  items in the category), and insert the item accordingly.
-- If REGION-OR-HERE has any other value (in particular, nil or
-  none), prompt for the text and the item's priority, and insert
-  the item accordingly."
+Also accept a non-existing category name and ask whether to add a
+new category by that name; on confirmation, add it and jump to
+that category, and if option `todos-add-item-if-new-category' is
+non-nil (the default), then prompt for the first item.
+
+In noninteractive calls non-nil WHERE specifies either the goal
+category or its file.  If its value is `archive', the choice of
+categories is restricted to the current archive file or the
+archive you were prompted to choose; this is used by
+`todos-jump-to-archive-category'.  If its value is the name of a
+category, jump directly to that category; this is used in Todos
+Categories mode."
+  (interactive "P")
   ;; If invoked outside of Todos mode and there is not yet any Todos
   ;; file, initialize one.
   (if (null todos-files)
       (todos-show)
-    (let ((region (eq region-or-here 'region))
-         (here (eq region-or-here 'here)))
-      (when region
-       (let (use-empty-active-region)
-         (unless (and todos-use-only-highlighted-region (use-region-p))
-           (user-error "There is no active region"))))
-      (let* ((obuf (current-buffer))
-            (ocat (todos-current-category))
-            (opoint (point))
-            (todos-mm (eq major-mode 'todos-mode))
-            (cat+file (cond ((equal arg '(4))
-                             (todos-read-category "Insert in category: "))
-                            ((equal arg '(16))
-                             (todos-read-category "Insert in category: "
-                                                  nil 'file))
-                            (t
-                             (cons (todos-current-category)
-                                   (or todos-current-todos-file
-                                       (and todos-show-current-file
-                                            todos-global-current-todos-file)
-                                       (todos-absolute-file-name
-                                        todos-default-todos-file))))))
-            (cat (car cat+file))
-            (file (cdr cat+file))
-            (new-item (if region
-                          (buffer-substring-no-properties
-                           (region-beginning) (region-end))
-                        (read-from-minibuffer "Todo item: ")))
-            (date-string (cond
-                          ((eq date-type 'date)
-                           (todos-read-date))
-                          ((eq date-type 'dayname)
-                           (todos-read-dayname))
-                          ((eq date-type 'calendar)
-                           (setq todos-date-from-calendar t)
-                           (or (todos-set-date-from-calendar)
-                               ;; If user exits Calendar before choosing
-                               ;; a date, cancel item insertion.
-                               (keyboard-quit)))
-                          ((and (stringp date-type)
-                                (string-match todos-date-pattern date-type))
-                           (setq todos-date-from-calendar date-type)
-                           (todos-set-date-from-calendar))
-                          (t
-                           (calendar-date-string
-                            (calendar-current-date) t t))))
-            (time-string (or (and time (todos-read-time))
-                             (and todos-always-add-time-string
-                                  (substring (current-time-string) 11 16)))))
-       (setq todos-date-from-calendar nil)
-       (find-file-noselect file 'nowarn)
+    (let* ((archive (eq where 'archive))
+          (cat (unless archive where))
+          (file0 (when cat             ; We're in Todos Categories mode.
+                   ;; With non-nil `todos-skip-archived-categories'
+                   ;; jump to archive file of a category with only
+                   ;; archived items.
+                   (if (and todos-skip-archived-categories
+                            (zerop (todos-get-count 'todo cat))
+                            (zerop (todos-get-count 'done cat))
+                            (not (zerop (todos-get-count 'archived cat))))
+                       (concat (file-name-sans-extension
+                                todos-current-todos-file) ".toda")
+                     ;; Otherwise, jump to current todos file.
+                     todos-current-todos-file)))
+          (len (length todos-categories))
+          (cat+file (unless cat
+                      (todos-read-category "Jump to category: "
+                                           (if archive 'archive) file)))
+          (add-item (and todos-add-item-if-new-category
+                         (> (length todos-categories) len)))
+          (category (or cat (car cat+file))))
+      (unless cat (setq file0 (cdr cat+file)))
+      (with-current-buffer (find-file-noselect file0 'nowarn)
+       (setq todos-current-todos-file file0)
+       ;; If called from Todos Categories mode, clean up before jumping.
+       (if (string= (buffer-name) todos-categories-buffer)
+           (kill-buffer))
        (set-window-buffer (selected-window)
-                          (set-buffer (find-buffer-visiting file)))
-       ;; If this command was invoked outside of a Todos buffer, the
-       ;; call to todos-current-category above returned nil.  If we
-       ;; just entered Todos mode now, then cat was set to the file's
-       ;; first category, but if todos-mode was already enabled, cat
-       ;; did not get set, so we have to set it explicitly.
-       (unless cat
-         (setq cat (todos-current-category)))
-       (setq todos-current-todos-file file)
+                          (set-buffer (find-buffer-visiting file0)))
        (unless todos-global-current-todos-file
          (setq todos-global-current-todos-file todos-current-todos-file))
-       (let ((buffer-read-only nil)
-             (called-from-outside (not (and todos-mm (equal cat ocat))))
-             done-only item-added)
-         (setq new-item
-               ;; Add date, time and diary marking as required.
-               (concat (if (not (and diary (not todos-include-in-diary)))
-                           todos-nondiary-start
-                         (when (and nonmarking (not todos-diary-nonmarking))
-                           diary-nonmarking-symbol))
-                       date-string (when (and time-string ; Can be empty.
-                                              (not (zerop (length
-                                                           time-string))))
-                                     (concat " " time-string))
-                       (when (not (and diary (not todos-include-in-diary)))
-                         todos-nondiary-end)
-                       " " new-item))
-         ;; Indent newlines inserted by C-q C-j if nonspace char follows.
-         (setq new-item (replace-regexp-in-string "\\(\n\\)[^[:blank:]]"
-                                                  "\n\t" new-item nil nil 1))
-         (unwind-protect
-             (progn
-               ;; Make sure the correct category is selected.  There
-               ;; are two cases: (i) we just visited the file, so no
-               ;; category is selected yet, or (ii) we invoked
-               ;; insertion "here" from outside the category we want
-               ;; to insert in (with priority insertion, category
-               ;; selection is done by todos-set-item-priority).
-               (when (or (= (- (point-max) (point-min)) (buffer-size))
-                         (and here called-from-outside))
-                 (todos-category-number cat)
-                 (todos-category-select))
-               ;; If only done items are displayed in category,
-               ;; toggle to todo items before inserting new item.
-               (when (save-excursion
-                       (goto-char (point-min))
-                       (looking-at todos-done-string-start))
-                 (setq done-only t)
-                 (todos-toggle-view-done-only))
-               (if here
-                   (progn
-                     ;; If command was invoked with point in done
-                     ;; items section or outside of the current
-                     ;; category, can't insert "here", so to be
-                     ;; useful give new item top priority.
-                     (when (or (todos-done-item-section-p)
-                               called-from-outside
-                               done-only)
-                       (goto-char (point-min)))
-                     (todos-insert-with-overlays new-item))
-                 (todos-set-item-priority new-item cat t))
-               (setq item-added t))
-           ;; If user cancels before setting priority, restore
-           ;; display.
-           (unless item-added
-             (if ocat
-                 (progn
-                   (unless (equal cat ocat)
-                     (todos-category-number ocat)
-                     (todos-category-select))
-                   (and done-only (todos-toggle-view-done-only)))
-               (set-window-buffer (selected-window) (set-buffer obuf)))
-             (goto-char opoint))
-           ;; If the todo items section is not visible when the
-           ;; insertion command is called (either because only done
-           ;; items were shown or because the category was not in the
-           ;; current buffer), then if the item is inserted at the
-           ;; end of the category, point is at eob and eob at
-           ;; window-start, so that higher priority todo items are
-           ;; out of view.  So we recenter to make sure the todo
-           ;; items are displayed in the window.
-           (when item-added (recenter)))
-         (todos-update-count 'todo 1)
-         (if (or diary todos-include-in-diary) (todos-update-count 'diary 1))
-         (todos-update-categories-sexp))))))
+       (todos-category-number category)
+       (todos-category-select)
+       (goto-char (point-min))
+       (when add-item (todos-basic-insert-item))))))
 
-(defvar todos-date-from-calendar nil
-  "Helper variable for setting item date from the Emacs Calendar.")
+(defun todos-next-item (&optional count)
+  "Move point down to the beginning of the next item.
+With positive numerical prefix COUNT, move point COUNT items
+downward.
 
-(defun todos-set-date-from-calendar ()
-  "Return string of date chosen from Calendar."
-  (cond ((and (stringp todos-date-from-calendar)
-             (string-match todos-date-pattern todos-date-from-calendar))
-        todos-date-from-calendar)
-       (todos-date-from-calendar
-        (let (calendar-view-diary-initially-flag)
-          (calendar))                  ; *Calendar* is now current buffer.
-        (define-key calendar-mode-map [remap newline] 'exit-recursive-edit)
-        ;; If user exits Calendar before choosing a date, clean up properly.
-        (define-key calendar-mode-map
-          [remap calendar-exit] (lambda ()
-                                   (interactive)
-                                   (progn
-                                     (calendar-exit)
-                                     (exit-recursive-edit))))
-        (message "Put cursor on a date and type <return> to set it.")
-        (recursive-edit)
-        (unwind-protect
-            (when (equal (buffer-name) calendar-buffer)
-              (setq todos-date-from-calendar
-                    (calendar-date-string (calendar-cursor-to-date t) t t))
-              (calendar-exit)
-              todos-date-from-calendar)
-          (define-key calendar-mode-map [remap newline] nil)
-          (define-key calendar-mode-map [remap calendar-exit] nil)
-          (unless (zerop (recursion-depth)) (exit-recursive-edit))
-          (when (stringp todos-date-from-calendar)
-            todos-date-from-calendar)))))
+If the category's done items are hidden, this command also moves
+point to the empty line below the last todo item from any higher
+item in the category, i.e., when invoked with or without a prefix
+argument.  If the category's done items are visible, this command
+called with a prefix argument only moves point to a lower item,
+e.g., with point on the last todo item and called with prefix 1,
+it moves point to the first done item; but if called with point
+on the last todo item without a prefix argument, it moves point
+the the empty line above the done items separator."
+  (interactive "p")
+  ;; It's not worth the trouble to allow prefix arg value < 1, since we have
+  ;; the corresponding command.
+  (cond ((and current-prefix-arg (< count 1))
+        (user-error "The prefix argument must be a positive number"))
+       (current-prefix-arg
+        (todos-forward-item count))
+       (t
+        (todos-forward-item))))
 
-(defun todos-insert-item-from-calendar (&optional arg)
-  "Prompt for and insert a new item with date selected from calendar.
-Invoked without prefix argument ARG, insert the item into the
-current category, without one prefix argument, prompt for the
-category from the current todo file or from one listed in
-`todos-category-completions-files'; with two prefix arguments,
-prompt for a todo file and then for a category in it."
-  (interactive "P")
-  (setq todos-date-from-calendar
-       (calendar-date-string (calendar-cursor-to-date t) t t))
-  (calendar-exit)
-  (todos-basic-insert-item arg nil nil todos-date-from-calendar))
+(defun todos-previous-item (&optional count)
+  "Move point up to start of item with next higher priority.
+With positive numerical prefix COUNT, move point COUNT items
+upward.
 
-(define-key calendar-mode-map "it" 'todos-insert-item-from-calendar)
+If the category's done items are visible, this command called
+with a prefix argument only moves point to a higher item, e.g.,
+with point on the first done item and called with prefix 1, it
+moves to the last todo item; but if called with point on the
+first done item without a prefix argument, it moves point the the
+empty line above the done items separator."
+  (interactive "p")
+  ;; Avoid moving to bob if on the first item but not at bob.
+  (when (> (line-number-at-pos) 1)
+    ;; It's not worth the trouble to allow prefix arg value < 1, since we have
+    ;; the corresponding command.
+    (cond ((and current-prefix-arg (< count 1))
+          (user-error "The prefix argument must be a positive number"))
+         (current-prefix-arg
+          (todos-backward-item count))
+         (t
+          (todos-backward-item)))))
 
-(defun todos-copy-item ()
-  "Copy item at point and insert the copy as a new item."
-  (interactive)
-  (unless (or (todos-done-item-p) (looking-at "^$"))
-    (let ((copy (todos-item-string))
-         (diary-item (todos-diary-item-p)))
-      (todos-set-item-priority copy (todos-current-category) t)
-      (todos-update-count 'todo 1)
-      (when diary-item (todos-update-count 'diary 1))
-      (todos-update-categories-sexp))))
+;; -----------------------------------------------------------------------------
+;;; Display toggle commands
+;; -----------------------------------------------------------------------------
 
-(defun todos-delete-item ()
-  "Delete at least one item in this category.
+(defun todos-toggle-prefix-numbers ()
+  "Hide item numbering if shown, show if hidden."
+  (interactive)
+  (save-excursion
+    (save-restriction
+      (goto-char (point-min))
+      (let* ((ov (todos-get-overlay 'prefix))
+            (show-done (re-search-forward todos-done-string-start nil t))
+            (todos-show-with-done show-done)
+            (todos-number-prefix (not (equal (overlay-get ov 'before-string)
+                                             "1 "))))
+       (if (eq major-mode 'todos-filtered-items-mode)
+           (todos-prefix-overlays)
+         (todos-category-select))))))
 
-If there are marked items, delete all of these; otherwise, delete
-the item at point."
+(defun todos-toggle-view-done-items ()
+  "Show hidden or hide visible done items in current category."
   (interactive)
-  (let (ov)
-    (unwind-protect
-       (let* ((cat (todos-current-category))
-              (marked (assoc cat todos-categories-with-marks))
-              (item (unless marked (todos-item-string)))
-              (answer (if marked
-                          (todos-y-or-n-p
-                           "Permanently delete all marked items? ")
-                        (when item
-                          (setq ov (make-overlay
-                                    (save-excursion (todos-item-start))
-                                    (save-excursion (todos-item-end))))
-                          (overlay-put ov 'face 'todos-search)
-                          (todos-y-or-n-p "Permanently delete this item? "))))
-              buffer-read-only)
-         (when answer
-           (and marked (goto-char (point-min)))
-           (catch 'done
-             (while (not (eobp))
-               (if (or (and marked (todos-marked-item-p)) item)
-                   (progn
-                     (if (todos-done-item-p)
-                         (todos-update-count 'done -1)
-                       (todos-update-count 'todo -1 cat)
-                       (and (todos-diary-item-p)
-                            (todos-update-count 'diary -1)))
-                     (if ov (delete-overlay ov))
-                     (todos-remove-item)
-                     ;; Don't leave point below last item.
-                     (and item (bolp) (eolp) (< (point-min) (point-max))
-                          (todos-backward-item))
-                     (when item
-                       (throw 'done (setq item nil))))
-                 (todos-forward-item))))
-           (when marked
-             (setq todos-categories-with-marks
-                   (assq-delete-all cat todos-categories-with-marks)))
-           (todos-update-categories-sexp)
-           (todos-prefix-overlays)))
-      (if ov (delete-overlay ov)))))
+  (if (zerop (todos-get-count 'done (todos-current-category)))
+      (message "There are no done items in this category.")
+    (let ((opoint (point)))
+      (goto-char (point-min))
+      (let* ((shown (re-search-forward todos-done-string-start nil t))
+            (todos-show-with-done (not shown)))
+       (todos-category-select)
+       (goto-char opoint)
+       ;; If start of done items sections is below the bottom of the
+       ;; window, make it visible.
+       (unless shown
+         (setq shown (progn
+                       (goto-char (point-min))
+                       (re-search-forward todos-done-string-start nil t)))
+         (if (not (pos-visible-in-window-p shown))
+             (recenter)
+           (goto-char opoint)))))))
 
-(defun todos-edit-item (&optional arg)
-  "Edit the Todo item at point.
+(defun todos-toggle-view-done-only ()
+  "Switch between displaying only done or only todo items."
+  (interactive)
+  (setq todos-show-done-only (not todos-show-done-only))
+  (todos-category-select))
 
-With non-nil prefix argument ARG, include the item's date/time
-header, making it also editable; otherwise, include only the item
-content.
+(defun todos-toggle-item-highlighting ()
+  "Highlight or unhighlight the todo item the cursor is on."
+  (interactive)
+  (eval-when-compile (require 'hl-line))
+  (when (memq major-mode
+             '(todos-mode todos-archive-mode todos-filtered-items-mode))
+    (if hl-line-mode
+       (hl-line-mode -1)
+      (hl-line-mode 1))))
 
-If the item consists of only one logical line, edit it in the
-minibuffer; otherwise, edit it in Todos Edit mode."
-  (interactive "P")
-  (when (todos-item-string)
-    (let* ((opoint (point))
-          (start (todos-item-start))
-          (item-beg (progn
-                      (re-search-forward
-                       (concat todos-date-string-start todos-date-pattern
-                               "\\( " diary-time-regexp "\\)?"
-                               (regexp-quote todos-nondiary-end) "?")
-                       (line-end-position) t)
-                      (1+ (- (point) start))))
-          (header (substring (todos-item-string) 0 item-beg))
-          (item (if arg (todos-item-string)
-                  (substring (todos-item-string) item-beg)))
-          (multiline (> (length (split-string item "\n")) 1))
-          (buffer-read-only nil))
-      (if multiline
-         (todos-edit-multiline-item)
-       (let ((new (concat (if arg "" header)
-                          (read-string "Edit: " (if arg
-                                                    (cons item item-beg)
-                                                  (cons item 0))))))
-         (when arg
-           (while (not (string-match (concat todos-date-string-start
-                                             todos-date-pattern) new))
-             (setq new (read-from-minibuffer
-                        "Item must start with a date: " new))))
-         ;; Ensure lines following hard newlines are indented.
-         (setq new (replace-regexp-in-string "\\(\n\\)[^[:blank:]]"
-                                             "\n\t" new nil nil 1))
-         ;; If user moved point during editing, make sure it moves back.
-         (goto-char opoint)
-         (todos-remove-item)
-         (todos-insert-with-overlays new)
-         (move-to-column item-beg))))))
+(defun todos-toggle-item-header ()
+  "Hide or show item date-time headers in the current file.
+With done items, this hides only the done date-time string, not
+the the original date-time string."
+  (interactive)
+  (save-excursion
+    (save-restriction
+      (goto-char (point-min))
+      (let ((ov (todos-get-overlay 'header)))
+       (if ov
+           (remove-overlays 1 (1+ (buffer-size)) 'todos 'header)
+         (widen)
+         (goto-char (point-min))
+         (while (not (eobp))
+           (when (re-search-forward
+                  (concat todos-item-start
+                          "\\( " diary-time-regexp "\\)?"
+                          (regexp-quote todos-nondiary-end) "? ")
+                  nil t)
+             (setq ov (make-overlay (match-beginning 0) (match-end 0) nil t))
+             (overlay-put ov 'todos 'header)
+             (overlay-put ov 'display ""))
+           (todos-forward-item)))))))
 
-(defun todos-edit-multiline-item ()
-  "Edit current Todo item in Todos Edit mode.
-Use of newlines invokes `todos-indent' to insure compliance with
-the format of Diary entries."
+;; -----------------------------------------------------------------------------
+;;; File and category editing
+;; -----------------------------------------------------------------------------
+
+(defun todos-add-file ()
+  "Name and initialize a new Todos file.
+Interactively, prompt for a category and display it, and if
+option `todos-add-item-if-new-category' is non-nil (the default),
+prompt for the first item.
+Noninteractively, return the name of the new file."
   (interactive)
-  (when (todos-item-string)
-    (let ((buf todos-edit-buffer))
-      (set-window-buffer (selected-window)
-                        (set-buffer (make-indirect-buffer (buffer-name) buf)))
-      (narrow-to-region (todos-item-start) (todos-item-end))
-      (todos-edit-mode)
-      (message "%s" (substitute-command-keys
-                    (concat "Type \\[todos-edit-quit] "
-                            "to return to Todos mode.\n"))))))
+  (let ((prompt (concat "Enter name of new Todos file "
+                       "(TAB or SPC to see current names): "))
+       file)
+    (setq file (todos-read-file-name prompt))
+    (with-current-buffer (get-buffer-create file)
+      (erase-buffer)
+      (write-region (point-min) (point-max) file nil 'nomessage nil t)
+      (kill-buffer file))
+    (setq todos-files (funcall todos-files-function))
+    (todos-reevaluate-filelist-defcustoms)
+    (if (called-interactively-p 'any)
+       (progn
+         (set-window-buffer (selected-window)
+                            (set-buffer (find-file-noselect file)))
+         (setq todos-current-todos-file file)
+         (todos-show))
+      file)))
 
-(defun todos-edit-quit ()
-  "Return from Todos Edit mode to Todos mode.
-If the item contains hard line breaks, make sure the following
-lines are indented by `todos-indent-to-here' to conform to diary
-format.
+(defvar todos-edit-buffer "*Todos Edit*"
+  "Name of current buffer in Todos Edit mode.")
 
-If the whole file was in Todos Edit mode, check before returning
-whether the file is still a valid Todos file and if so, also
-recalculate the Todos categories sexp, in case changes were made
-in the number or names of categories."
+(defun todos-edit-file ()
+  "Put current buffer in `todos-edit-mode'.
+This makes the entire file visible and the buffer writeable and
+you can use the self-insertion keys and standard Emacs editing
+commands to make changes.  To return to Todos mode, type
+\\[todos-edit-quit].  This runs a file format check, signalling
+an error if the format has become invalid.  However, this check
+cannot tell if the number of items changed, which could result in
+the file containing inconsistent information.  For this reason
+this command should be used with caution."
   (interactive)
-  (if (> (buffer-size) (- (point-max) (point-min)))
-      ;; We got here via `e m'.
-      (let ((item (buffer-string))
-           (regex "\\(\n\\)[^[:blank:]]")
-           (buf (buffer-base-buffer)))
-       (while (not (string-match (concat todos-date-string-start
-                                         todos-date-pattern) item))
-         (setq item (read-from-minibuffer
-                    "Item must start with a date: " item)))
-       ;; Ensure lines following hard newlines are indented.
-       (when (string-match regex (buffer-string))
-         (setq item (replace-regexp-in-string regex "\n\t" item nil nil 1))
-         (delete-region (point-min) (point-max))
-         (insert item))
-       (kill-buffer)
-       (unless (eq (current-buffer) buf)
-         (set-window-buffer (selected-window) (set-buffer buf))))
-    ;; We got here via `F e'.
-    (when (todos-check-format)
-      ;; FIXME: separate out sexp check?
-      ;; If manual editing makes e.g. item counts change, have to
-      ;; call this to update todos-categories, but it restores
-      ;; category order to list order.
-      ;; (todos-repair-categories-sexp)
-      ;; Compare (todos-make-categories-list t) with sexp and if
-      ;; different ask (todos-update-categories-sexp) ?
-      (todos-mode)
-      (let* ((cat-beg (concat "^" (regexp-quote todos-category-beg)
-                             "\\(.*\\)$"))
-            (curline (buffer-substring-no-properties
-                      (line-beginning-position) (line-end-position)))
-            (cat (cond ((string-match cat-beg curline)
-                        (match-string-no-properties 1 curline))
-                       ((or (re-search-backward cat-beg nil t)
-                            (re-search-forward cat-beg nil t))
-                        (match-string-no-properties 1)))))
-       (todos-category-number cat)
-       (todos-category-select)
-       (goto-char (point-min))))))
+  (widen)
+  (todos-edit-mode)
+  (remove-overlays)
+  (message "%s" (substitute-command-keys
+                (concat "Type \\[todos-edit-quit] to check file format "
+                        "validity and return to Todos mode.\n"))))
 
-(defun todos-basic-edit-item-header (what &optional inc)
-  "Function underlying commands to edit item date/time header.
+(defun todos-add-category (&optional file cat)
+  "Add a new category to a Todos file.
 
-The argument WHAT (passed by invoking commands) specifies what
-part of the header to edit; possible values are these symbols:
-`date', to edit the year, month, and day of the date string;
-`time', to edit just the time string; `calendar', to select the
-date from the Calendar; `today', to set the date to today's date;
-`dayname', to set the date string to the name of a day or to
-change the day name; and `year', `month' or `day', to edit only
-these respective parts of the date string (`day' is the number of
-the given day of the month, and `month' is either the name of the
-given month or its number, depending on the value of
-`calendar-date-display-form').
+Called interactively with prefix argument FILE, prompt for a file
+and then for a new category to add to that file, otherwise prompt
+just for a category to add to the current Todos file.  After
+adding the category, visit it in Todos mode and if option
+`todos-add-item-if-new-category' is non-nil (the default), prompt
+for the first item.
 
-The optional argument INC is a positive or negative integer
-\(passed by invoking commands as a numerical prefix argument)
-that in conjunction with the WHAT values `year', `month' or
-`day', increments or decrements the specified date string
-component by the specified number of suitable units, i.e., years,
-months, or days, with automatic adjustment of the other date
-string components as necessary.
+Non-interactively, add category CAT to file FILE; if FILE is nil,
+add CAT to the current Todos file.  After adding the category,
+return the new category number."
+  (interactive "P")
+  (let (catfil file0)
+    ;; If cat is passed from caller, don't prompt, unless it is "",
+    ;; which means the file was just added and has no category yet.
+    (if (and cat (> (length cat) 0))
+       (setq file0 (or (and (stringp file) file)
+                       todos-current-todos-file))
+      (setq catfil (todos-read-category "Enter a new category name: "
+                                       'add (when (called-interactively-p 'any)
+                                              file))
+           cat (car catfil)
+           file0 (if (called-interactively-p 'any)
+                     (cdr catfil)
+                   file)))
+    (find-file file0)
+    (let ((counts (make-vector 4 0))   ; [todo diary done archived]
+         (num (1+ (length todos-categories)))
+         (buffer-read-only nil))
+      (setq todos-current-todos-file file0)
+      (setq todos-categories (append todos-categories
+                                    (list (cons cat counts))))
+      (widen)
+      (goto-char (point-max))
+      (save-excursion                  ; Save point for todos-category-select.
+       (insert todos-category-beg cat "\n\n" todos-category-done "\n"))
+      (todos-update-categories-sexp)
+      ;; If invoked by user, display the newly added category, if
+      ;; called programmatically return the category number to the
+      ;; caller.
+      (if (called-interactively-p 'any)
+         (progn
+           (setq todos-category-number num)
+           (todos-category-select)
+           (when todos-add-item-if-new-category
+             (todos-basic-insert-item)))
+       num))))
 
-If there are marked items, apply the same edit to all of these;
-otherwise, edit just the item at point."
+(defun todos-rename-category ()
+  "Rename current Todos category.
+If this file has an archive containing this category, rename the
+category there as well."
+  (interactive)
   (let* ((cat (todos-current-category))
-        (marked (assoc cat todos-categories-with-marks))
-        (first t)
-        (todos-date-from-calendar t)
-        (buffer-read-only nil)
-        ndate ntime year monthname month day
-        dayname)       ; Needed by calendar-date-display-form.
-    (save-excursion
-      (or (and marked (goto-char (point-min))) (todos-item-start))
-      (catch 'end
-       (while (not (eobp))
-         (and marked
-              (while (not (todos-marked-item-p))
-                (todos-forward-item)
-                (and (eobp) (throw 'end nil))))
-         (re-search-forward (concat todos-date-string-start "\\(?1:"
-                                    todos-date-pattern
-                                    "\\)\\(?2: " diary-time-regexp "\\)?"
-                                    (regexp-quote todos-nondiary-end) "?")
-                            (line-end-position) t)
-         (let* ((odate (match-string-no-properties 1))
-                (otime (match-string-no-properties 2))
-                (omonthname (match-string-no-properties 6))
-                (omonth (match-string-no-properties 7))
-                (oday (match-string-no-properties 8))
-                (oyear (match-string-no-properties 9))
-                (tmn-array todos-month-name-array)
-                (mlist (append tmn-array nil))
-                (tma-array todos-month-abbrev-array)
-                (mablist (append tma-array nil))
-                (yy (and oyear (unless (string= oyear "*")
-                                 (string-to-number oyear))))
-                (mm (or (and omonth (unless (string= omonth "*")
-                                      (string-to-number omonth)))
-                        (1+ (- (length mlist)
-                               (length (or (member omonthname mlist)
-                                           (member omonthname mablist)))))))
-                (dd (and oday (unless (string= oday "*")
-                                (string-to-number oday)))))
-           ;; If there are marked items, use only the first to set
-           ;; header changes, and apply these to all marked items.
-           (when first
-             (cond
-              ((eq what 'date)
-               (setq ndate (todos-read-date)))
-              ((eq what 'calendar)
-               (setq ndate (save-match-data (todos-set-date-from-calendar))))
-              ((eq what 'today)
-               (setq ndate (calendar-date-string (calendar-current-date) t t)))
-              ((eq what 'dayname)
-               (setq ndate (todos-read-dayname)))
-              ((eq what 'time)
-               (setq ntime (save-match-data (todos-read-time)))
-               (when (> (length ntime) 0)
-                 (setq ntime (concat " " ntime))))
-              ;; When date string consists only of a day name,
-              ;; passing other date components is a NOP.
-              ((and (memq what '(year month day))
-                    (not (or oyear omonth oday))))
-              ((eq what 'year)
-               (setq day oday
-                     monthname omonthname
-                     month omonth
-                     year (cond ((not current-prefix-arg)
-                                 (todos-read-date 'year))
-                                ((string= oyear "*")
-                                 (user-error "Cannot increment *"))
-                                (t
-                                 (number-to-string (+ yy inc))))))
-              ((eq what 'month)
-               (setf day oday
-                     year oyear
-                     (if (memq 'month calendar-date-display-form)
-                         month
-                       monthname)
-                     (cond ((not current-prefix-arg)
-                            (todos-read-date 'month))
-                           ((or (string= omonth "*") (= mm 13))
-                            (user-error "Cannot increment *"))
-                           (t
-                            (let ((mminc (+ mm inc)))
-                              ;; Increment or decrement month by INC
-                              ;; modulo 12.
-                              (setq mm (% mminc 12))
-                              ;; If result is 0, make month December.
-                              (setq mm (if (= mm 0) 12 (abs mm)))
-                              ;; Adjust year if necessary.
-                              (setq year (or (and (cond ((> mminc 12)
-                                                         (+ yy (/ mminc 12)))
-                                                        ((< mminc 1)
-                                                         (- yy (/ mminc 12) 1))
-                                                        (t yy))
-                                                  (number-to-string yy))
-                                             oyear)))
-                            ;; Return the changed numerical month as
-                            ;; a string or the corresponding month name.
-                            (if omonth
-                                (number-to-string mm)
-                              (aref tma-array (1- mm))))))
-               (let ((yy (string-to-number year)) ; 0 if year is "*".
-                     ;; When mm is 13 (corresponding to "*" as value
-                     ;; of month), this raises an args-out-of-range
-                     ;; error in calendar-last-day-of-month, so use 1
-                     ;; (corresponding to January) to get 31 days.
-                     (mm (if (= mm 13) 1 mm)))
-                 (if (> (string-to-number day)
-                        (calendar-last-day-of-month mm yy))
-                     (user-error "%s %s does not have %s days"
-                            (aref tmn-array (1- mm))
-                            (if (= mm 2) yy "") day))))
-              ((eq what 'day)
-               (setq year oyear
-                     month omonth
-                     monthname omonthname
-                     day (cond
-                          ((not current-prefix-arg)
-                           (todos-read-date 'day mm oyear))
-                          ((string= oday "*")
-                           (user-error "Cannot increment *"))
-                          ((or (string= omonth "*") (string= omonthname "*"))
-                           (setq dd (+ dd inc))
-                           (if (> dd 31)
-                               (user-error "A month cannot have more than 31 days")
-                             (number-to-string dd)))
-                          ;; Increment or decrement day by INC,
-                          ;; adjusting month and year if necessary
-                          ;; (if year is "*" assume current year to
-                          ;; calculate adjustment).
-                          (t
-                           (let* ((yy (or yy (calendar-extract-year
-                                              (calendar-current-date))))
-                                  (date (calendar-gregorian-from-absolute
-                                         (+ (calendar-absolute-from-gregorian
-                                             (list mm dd yy)) inc)))
-                                  (adjmm (nth 0 date)))
-                             ;; Set year and month(name) to adjusted values.
-                             (unless (string= year "*")
-                               (setq year (number-to-string (nth 2 date))))
-                             (if month
-                                 (setq month (number-to-string adjmm))
-                               (setq monthname (aref tma-array (1- adjmm))))
-                             ;; Return changed numerical day as a string.
-                             (number-to-string (nth 1 date)))))))))
-           ;; If new year, month or day date string components were
-           ;; calculated, rebuild the whole date string from them.
-           (when (memq what '(year month day))
-             (if (or oyear omonth omonthname oday)
-                 (setq ndate (mapconcat 'eval calendar-date-display-form ""))
-               (message "Cannot edit date component of empty date string")))
-           (when ndate (replace-match ndate nil nil nil 1))
-           ;; Add new time string to the header, if it was supplied.
-           (when ntime
-             (if otime
-                 (replace-match ntime nil nil nil 2)
-               (goto-char (match-end 1))
-               (insert ntime)))
-           (setq todos-date-from-calendar nil)
-           (setq first nil))
-         ;; Apply the changes to the first marked item header to the
-         ;; remaining marked items.  If there are no marked items,
-         ;; we're finished.
-         (if marked
-             (todos-forward-item)
-           (goto-char (point-max))))))))
-
-(defun todos-edit-item-header ()
-  "Interactively edit at least the date of item's date/time header.
-If user option `todos-always-add-time-string' is non-nil, also
-edit item's time string."
-  (interactive)
-  (todos-basic-edit-item-header 'date)
-  (when todos-always-add-time-string
-    (todos-edit-item-time)))
-
-(defun todos-edit-item-time ()
-  "Interactively edit the time string of item's date/time header."
-  (interactive)
-  (todos-basic-edit-item-header 'time))
-
-(defun todos-edit-item-date-from-calendar ()
-  "Interactively edit item's date using the Calendar."
-  (interactive)
-  (todos-basic-edit-item-header 'calendar))
-
-(defun todos-edit-item-date-to-today ()
-  "Set item's date to today's date."
-  (interactive)
-  (todos-basic-edit-item-header 'today))
-
-(defun todos-edit-item-date-day-name ()
-  "Replace item's date with the name of a day of the week."
-  (interactive)
-  (todos-basic-edit-item-header 'dayname))
-
-(defun todos-edit-item-date-year (&optional inc)
-  "Interactively edit the year of item's date string.
-With prefix argument INC a positive or negative integer,
-increment or decrement the year by INC."
-  (interactive "p")
-  (todos-basic-edit-item-header 'year inc))
-
-(defun todos-edit-item-date-month (&optional inc)
-  "Interactively edit the month of item's date string.
-With prefix argument INC a positive or negative integer,
-increment or decrement the month by INC."
-  (interactive "p")
-  (todos-basic-edit-item-header 'month inc))
-
-(defun todos-edit-item-date-day (&optional inc)
-  "Interactively edit the day of the month of item's date string.
-With prefix argument INC a positive or negative integer,
-increment or decrement the day by INC."
-  (interactive "p")
-  (todos-basic-edit-item-header 'day inc))
-
-(defun todos-edit-item-diary-inclusion ()
-  "Change diary status of one or more todo items in this category.
-That is, insert `todos-nondiary-marker' if the candidate items
-lack this marking; otherwise, remove it.
-
-If there are marked todo items, change the diary status of all
-and only these, otherwise change the diary status of the item at
-point."
-  (interactive)
-  (let ((buffer-read-only)
-       (marked (assoc (todos-current-category)
-                      todos-categories-with-marks)))
-    (catch 'stop
-      (save-excursion
-       (when marked (goto-char (point-min)))
-       (while (not (eobp))
-         (if (todos-done-item-p)
-             (throw 'stop (message "Done items cannot be edited"))
-           (unless (and marked (not (todos-marked-item-p)))
-             (let* ((beg (todos-item-start))
-                    (lim (save-excursion (todos-item-end)))
-                    (end (save-excursion
-                           (or (todos-time-string-matcher lim)
-                               (todos-date-string-matcher lim)))))
-               (if (looking-at (regexp-quote todos-nondiary-start))
-                   (progn
-                     (replace-match "")
-                     (search-forward todos-nondiary-end (1+ end) t)
-                     (replace-match "")
-                     (todos-update-count 'diary 1))
-                 (when end
-                   (insert todos-nondiary-start)
-                   (goto-char (1+ end))
-                   (insert todos-nondiary-end)
-                   (todos-update-count 'diary -1)))))
-           (unless marked (throw 'stop nil))
-           (todos-forward-item)))))
-    (todos-update-categories-sexp)))
-
-(defun todos-edit-category-diary-inclusion (arg)
-  "Make all items in this category diary items.
-With prefix ARG, make all items in this category non-diary
-items."
-  (interactive "P")
-  (save-excursion
-    (goto-char (point-min))
-    (let ((todo-count (todos-get-count 'todo))
-         (diary-count (todos-get-count 'diary))
-         (buffer-read-only))
-      (catch 'stop
-       (while (not (eobp))
-         (if (todos-done-item-p)       ; We've gone too far.
-             (throw 'stop nil)
-           (let* ((beg (todos-item-start))
-                  (lim (save-excursion (todos-item-end)))
-                  (end (save-excursion
-                         (or (todos-time-string-matcher lim)
-                             (todos-date-string-matcher lim)))))
-             (if arg
-                 (unless (looking-at (regexp-quote todos-nondiary-start))
-                   (insert todos-nondiary-start)
-                   (goto-char (1+ end))
-                   (insert todos-nondiary-end))
-               (when (looking-at (regexp-quote todos-nondiary-start))
-                 (replace-match "")
-                 (search-forward todos-nondiary-end (1+ end) t)
-                 (replace-match "")))))
-         (todos-forward-item))
-       (unless (if arg (zerop diary-count) (= diary-count todo-count))
-         (todos-update-count 'diary (if arg
-                                     (- diary-count)
-                                   (- todo-count diary-count))))
-       (todos-update-categories-sexp)))))
-
-(defun todos-edit-item-diary-nonmarking ()
-  "Change non-marking of one or more diary items in this category.
-That is, insert `diary-nonmarking-symbol' if the candidate items
-lack this marking; otherwise, remove it.
-
-If there are marked todo items, change the non-marking status of
-all and only these, otherwise change the non-marking status of
-the item at point."
-  (interactive)
-  (let ((buffer-read-only)
-       (marked (assoc (todos-current-category)
-                      todos-categories-with-marks)))
-    (catch 'stop
-      (save-excursion
-       (when marked (goto-char (point-min)))
-       (while (not (eobp))
-         (if (todos-done-item-p)
-             (throw 'stop (message "Done items cannot be edited"))
-           (unless (and marked (not (todos-marked-item-p)))
-             (todos-item-start)
-             (unless (looking-at (regexp-quote todos-nondiary-start))
-               (if (looking-at (regexp-quote diary-nonmarking-symbol))
-                   (replace-match "")
-                 (insert diary-nonmarking-symbol))))
-           (unless marked (throw 'stop nil))
-           (todos-forward-item)))))))
-
-(defun todos-edit-category-diary-nonmarking (arg)
-  "Add `diary-nonmarking-symbol' to all diary items in this category.
-With prefix ARG, remove `diary-nonmarking-symbol' from all diary
-items in this category."
-  (interactive "P")
-  (save-excursion
-    (goto-char (point-min))
-    (let (buffer-read-only)
-      (catch 'stop
-      (while (not (eobp))
-       (if (todos-done-item-p)         ; We've gone too far.
-           (throw 'stop nil)
-         (unless (looking-at (regexp-quote todos-nondiary-start))
-           (if arg
-               (when (looking-at (regexp-quote diary-nonmarking-symbol))
-                 (replace-match ""))
-             (unless (looking-at (regexp-quote diary-nonmarking-symbol))
-               (insert diary-nonmarking-symbol))))
-       (todos-forward-item)))))))
-
-(defun todos-set-item-priority (&optional item cat new arg)
-  "Prompt for and set ITEM's priority in CATegory.
-
-Interactively, ITEM is the todo item at point, CAT is the current
-category, and the priority is a number between 1 and the number
-of items in the category.  Non-interactively, non-nil NEW means
-ITEM is a new item and the lowest priority is one more than the
-number of items in CAT.
-
-The new priority is set either interactively by prompt or by a
-numerical prefix argument, or noninteractively by argument ARG,
-whose value can be either of the symbols `raise' or `lower',
-meaning to raise or lower the item's priority by one."
-  (interactive)
-  (unless (and (called-interactively-p 'any)
-              (or (todos-done-item-p) (looking-at "^$")))
-    (let* ((item (or item (todos-item-string)))
-          (marked (todos-marked-item-p))
-          (cat (or cat (cond ((eq major-mode 'todos-mode)
-                              (todos-current-category))
-                             ((eq major-mode 'todos-filtered-items-mode)
-                              (let* ((regexp1
-                                      (concat todos-date-string-start
-                                              todos-date-pattern
-                                              "\\( " diary-time-regexp "\\)?"
-                                              (regexp-quote todos-nondiary-end)
-                                              "?\\(?1: \\[\\(.+:\\)?.+\\]\\)")))
-                                (save-excursion
-                                  (re-search-forward regexp1 nil t)
-                                  (match-string-no-properties 1)))))))
-          curnum
-          (todo (cond ((or (eq arg 'raise) (eq arg 'lower)
-                           (eq major-mode 'todos-filtered-items-mode))
-                       (save-excursion
-                         (let ((curstart (todos-item-start))
-                               (count 0))
-                           (goto-char (point-min))
-                           (while (looking-at todos-item-start)
-                             (setq count (1+ count))
-                             (when (= (point) curstart) (setq curnum count))
-                             (todos-forward-item))
-                           count)))
-                      ((eq major-mode 'todos-mode)
-                       (todos-get-count 'todo cat))))
-          (maxnum (if new (1+ todo) todo))
-          (prompt (format "Set item priority (1-%d): " maxnum))
-          (priority (cond ((and (not arg) (numberp current-prefix-arg))
-                           current-prefix-arg)
-                          ((and (eq arg 'raise) (>= curnum 1))
-                           (1- curnum))
-                          ((and (eq arg 'lower) (<= curnum maxnum))
-                           (1+ curnum))))
-          candidate
-          buffer-read-only)
-      (unless (and priority
-                  (or (and (eq arg 'raise) (zerop priority))
-                      (and (eq arg 'lower) (> priority maxnum))))
-       ;; When moving item to another category, show the category before
-       ;; prompting for its priority.
-       (unless (or arg (called-interactively-p 'any))
-         (todos-category-number cat)
-         ;; If done items in category are visible, keep them visible.
-         (let ((done todos-show-with-done))
-           (when (> (buffer-size) (- (point-max) (point-min)))
-             (save-excursion
-               (goto-char (point-min))
-               (setq done (re-search-forward todos-done-string-start nil t))))
-           (let ((todos-show-with-done done))
-             (todos-category-select)
-             ;; Keep top of category in view while setting priority.
-             (goto-char (point-min)))))
-       ;; Prompt for priority only when the category has at least one
-       ;; todo item.
-       (when (> maxnum 1)
-         (while (not priority)
-           (setq candidate (read-number prompt))
-           (setq prompt (when (or (< candidate 1) (> candidate maxnum))
-                          (format "Priority must be an integer between 1 and %d.\n"
-                                  maxnum)))
-           (unless prompt (setq priority candidate))))
-       ;; In Top Priorities buffer, an item's priority can be changed
-       ;; wrt items in another category, but not wrt items in the same
-       ;; category.
-       (when (eq major-mode 'todos-filtered-items-mode)
-         (let* ((regexp2 (concat todos-date-string-start todos-date-pattern
-                                 "\\( " diary-time-regexp "\\)?"
-                                 (regexp-quote todos-nondiary-end)
-                                 "?\\(?1:" (regexp-quote cat) "\\)"))
-                (end (cond ((< curnum priority)
-                            (save-excursion (todos-item-end)))
-                           ((> curnum priority)
-                            (save-excursion (todos-item-start)))))
-                (match (save-excursion
-                         (cond ((< curnum priority)
-                                (todos-forward-item (1+ (- priority curnum)))
-                                (when (re-search-backward regexp2 end t)
-                                  (match-string-no-properties 1)))
-                               ((> curnum priority)
-                                (todos-backward-item (- curnum priority))
-                                (when (re-search-forward regexp2 end t)
-                                  (match-string-no-properties 1)))))))
-           (when match
-             (user-error (concat "Cannot reprioritize items from the same "
-                            "category in this mode, only in Todos mode")))))
-       ;; Interactively or with non-nil ARG, relocate the item within its
-       ;; category.
-       (when (or arg (called-interactively-p 'any))
-         (todos-remove-item))
-       (goto-char (point-min))
-       (when priority
-         (unless (= priority 1)
-           (todos-forward-item (1- priority))
-           ;; When called from todos-item-undone and the highest priority
-           ;; is chosen, this advances point to the first done item, so
-           ;; move it up to the empty line above the done items
-           ;; separator.
-           (when (looking-back (concat "^"
-                                       (regexp-quote todos-category-done)
-                                       "\n"))
-             (todos-backward-item))))
-       (todos-insert-with-overlays item)
-       ;; If item was marked, restore the mark.
-       (and marked
-            (let* ((ov (todos-get-overlay 'prefix))
-                   (pref (overlay-get ov 'before-string)))
-              (overlay-put ov 'before-string
-                           (concat todos-item-mark pref))))))))
-
-(defun todos-raise-item-priority ()
-  "Raise priority of current item by moving it up by one item."
-  (interactive)
-  (todos-set-item-priority nil nil nil 'raise))
-
-(defun todos-lower-item-priority ()
-  "Lower priority of current item by moving it down by one item."
-  (interactive)
-  (todos-set-item-priority nil nil nil 'lower))
-
-(defun todos-move-item (&optional file)
-  "Move at least one todo or done item to another category.
-If there are marked items, move all of these; otherwise, move
-the item at point.
-
-With prefix argument FILE, prompt for a specific Todos file and
-choose (with TAB completion) a category in it to move the item or
-items to; otherwise, choose and move to any category in either
-the current Todos file or one of the files in
-`todos-category-completions-files'.  If the chosen category is
-not an existing categories, then it is created and the item(s)
-become(s) the first entry/entries in that category.
-
-With moved Todo items, prompt to set the priority in the category
-moved to (with multiple todos items, the one that had the highest
-priority in the category moved from gets the new priority and the
-rest of the moved todo items are inserted in sequence below it).
-Moved done items are appended to the top of the done items
-section in the category moved to."
-  (interactive "P")
-  (let* ((cat1 (todos-current-category))
-        (marked (assoc cat1 todos-categories-with-marks)))
-    ;; Noop if point is not on an item and there are no marked items.
-    (unless (and (looking-at "^$")
-                (not marked))
-      (let* ((buffer-read-only)
-            (file1 todos-current-todos-file)
-            (num todos-category-number)
-            (item (todos-item-string))
-            (diary-item (todos-diary-item-p))
-            (done-item (and (todos-done-item-p) (concat item "\n")))
-            (omark (save-excursion (todos-item-start) (point-marker)))
-            (todo 0)
-            (diary 0)
-            (done 0)
-            ov cat2 file2 moved nmark todo-items done-items)
-       (unwind-protect
-           (progn
-             (unless marked
-               (setq ov (make-overlay (save-excursion (todos-item-start))
-                                      (save-excursion (todos-item-end))))
-               (overlay-put ov 'face 'todos-search))
-             (let* ((pl (if (and marked (> (cdr marked) 1)) "s" ""))
-                    (cat+file (todos-read-category (concat "Move item" pl
-                                                           " to category: ")
-                                                   nil file)))
-               (while (and (equal (car cat+file) cat1)
-                           (equal (cdr cat+file) file1))
-                 (setq cat+file (todos-read-category
-                                 "Choose a different category: ")))
-               (setq cat2 (car cat+file)
-                     file2 (cdr cat+file))))
-         (if ov (delete-overlay ov)))
-       (set-buffer (find-buffer-visiting file1))
-       (if marked
-           (progn
-             (goto-char (point-min))
-             (while (not (eobp))
-               (when (todos-marked-item-p)
-                 (if (todos-done-item-p)
-                     (setq done-items (concat done-items
-                                              (todos-item-string) "\n")
-                           done (1+ done))
-                   (setq todo-items (concat todo-items
-                                            (todos-item-string) "\n")
-                         todo (1+ todo))
-                   (when (todos-diary-item-p)
-                     (setq diary (1+ diary)))))
-               (todos-forward-item))
-             ;; Chop off last newline of multiple todo item string,
-             ;; since it will be reinserted when setting priority
-             ;; (but with done items priority is not set, so keep
-             ;; last newline).
-             (and todo-items
-                  (setq todo-items (substring todo-items 0 -1))))
-         (if (todos-done-item-p)
-             (setq done 1)
-           (setq todo 1)
-           (when (todos-diary-item-p) (setq diary 1))))
-       (set-window-buffer (selected-window)
-                          (set-buffer (find-file-noselect file2 'nowarn)))
-       (unwind-protect
-           (progn
-             (when (or todo-items (and item (not done-item)))
-               (todos-set-item-priority (or todo-items item) cat2 t))
-             ;; Move done items en bloc to top of done items section.
-             (when (or done-items done-item)
-               (todos-category-number cat2)
-               (widen)
-               (goto-char (point-min))
-               (re-search-forward
-                (concat "^" (regexp-quote (concat todos-category-beg cat2))
-                        "$") nil t)
-               (re-search-forward
-                (concat "^" (regexp-quote todos-category-done)) nil t)
-               (forward-line)
-               (insert (or done-items done-item)))
-             (setq moved t))
-         (cond
-          ;; Move succeeded, so remove item from starting category,
-          ;; update item counts and display the category containing
-          ;; the moved item.
-          (moved
-           (setq nmark (point-marker))
-           (when todo (todos-update-count 'todo todo))
-           (when diary (todos-update-count 'diary diary))
-           (when done (todos-update-count 'done done))
-           (todos-update-categories-sexp)
-           (with-current-buffer (find-buffer-visiting file1)
-             (save-excursion
-               (save-restriction
-                 (widen)
-                 (goto-char omark)
-                 (if marked
-                     (let (beg end)
-                       (setq item nil)
-                       (re-search-backward
-                        (concat "^" (regexp-quote todos-category-beg)) nil t)
-                       (forward-line)
-                       (setq beg (point))
-                       (setq end (if (re-search-forward
-                                      (concat "^" (regexp-quote
-                                                   todos-category-beg)) nil t)
-                                     (match-beginning 0)
-                                   (point-max)))
-                       (goto-char beg)
-                       (while (< (point) end)
-                         (if (todos-marked-item-p)
-                             (todos-remove-item)
-                           (todos-forward-item)))
-                       (setq todos-categories-with-marks
-                             (assq-delete-all cat1 todos-categories-with-marks)))
-                   (if ov (delete-overlay ov))
-                   (todos-remove-item))))
-             (when todo (todos-update-count 'todo (- todo) cat1))
-             (when diary (todos-update-count 'diary (- diary) cat1))
-             (when done (todos-update-count 'done (- done) cat1))
-             (todos-update-categories-sexp))
-           (set-window-buffer (selected-window)
-                              (set-buffer (find-file-noselect file2 'nowarn)))
-           (setq todos-category-number (todos-category-number cat2))
-           (let ((todos-show-with-done (or done-items done-item)))
-             (todos-category-select))
-           (goto-char nmark)
-           ;; If item is moved to end of (just first?) category, make
-           ;; sure the items above it are displayed in the window.
-           (recenter))
-          ;; User quit before setting priority of todo item(s), so
-          ;; return to starting category.
-          (t
-           (set-window-buffer (selected-window)
-                              (set-buffer (find-file-noselect file1 'nowarn)))
-           (todos-category-number cat1)
-           (todos-category-select)
-           (goto-char omark))))))))
-
-(defun todos-item-done (&optional arg)
-  "Tag a todo item in this category as done and relocate it.
-
-With prefix argument ARG prompt for a comment and append it to
-the done item; this is only possible if there are no marked
-items.  If there are marked items, tag all of these with
-`todos-done-string' plus the current date and, if
-`todos-always-add-time-string' is non-nil, the current time;
-otherwise, just tag the item at point.  Items tagged as done are
-relocated to the category's (by default hidden) done section.  If
-done items are visible on invoking this command, they remain
-visible."
-  (interactive "P")
-  (let* ((cat (todos-current-category))
-        (marked (assoc cat todos-categories-with-marks)))
-    (when marked
-      (save-excursion
-       (save-restriction
-         (goto-char (point-max))
-         (todos-backward-item)
-         (unless (todos-done-item-p)
-           (widen)
-           (unless (re-search-forward
-                    (concat "^" (regexp-quote todos-category-beg)) nil t)
-             (goto-char (point-max)))
-           (forward-line -1))
-         (while (todos-done-item-p)
-           (when (todos-marked-item-p)
-             (user-error "This command does not apply to done items"))
-           (todos-backward-item)))))
-    (unless (and (not marked)
-                (or (todos-done-item-p)
-                    ;; Point is between todo and done items.
-                    (looking-at "^$")))
-      (let* ((date-string (calendar-date-string (calendar-current-date) t t))
-            (time-string (if todos-always-add-time-string
-                             (concat " " (substring (current-time-string)
-                                                    11 16))
-                           ""))
-            (done-prefix (concat "[" todos-done-string date-string time-string
-                                 "] "))
-            (comment (and arg (read-string "Enter a comment: ")))
-            (item-count 0)
-            (diary-count 0)
-            (show-done (save-excursion
-                         (goto-char (point-min))
-                         (re-search-forward todos-done-string-start nil t)))
-            (buffer-read-only nil)
-            item done-item opoint)
-       ;; Don't add empty comment to done item.
-       (setq comment (unless (zerop (length comment))
-                       (concat " [" todos-comment-string ": " comment "]")))
-       (and marked (goto-char (point-min)))
-       (catch 'done
-         ;; Stop looping when we hit the empty line below the last
-         ;; todo item (this is eobp if only done items are hidden).
-         (while (not (looking-at "^$"))
-           (if (or (not marked) (and marked (todos-marked-item-p)))
-               (progn
-                 (setq item (todos-item-string))
-                 (setq done-item (concat done-item done-prefix item
-                                         comment (and marked "\n")))
-                 (setq item-count (1+ item-count))
-                 (when (todos-diary-item-p)
-                   (setq diary-count (1+ diary-count)))
-                 (todos-remove-item)
-                 (unless marked (throw 'done nil)))
-             (todos-forward-item))))
-       (when marked
-         ;; Chop off last newline of done item string.
-         (setq done-item (substring done-item 0 -1))
-         (setq todos-categories-with-marks
-               (assq-delete-all cat todos-categories-with-marks)))
-       (save-excursion
-         (widen)
-         (re-search-forward
-          (concat "^" (regexp-quote todos-category-done)) nil t)
-         (forward-char)
-         (when show-done (setq opoint (point)))
-         (insert done-item "\n"))
-       (todos-update-count 'todo (- item-count))
-       (todos-update-count 'done item-count)
-       (todos-update-count 'diary (- diary-count))
-       (todos-update-categories-sexp)
-       (let ((todos-show-with-done show-done))
-         (todos-category-select)
-         ;; When done items are shown, put cursor on first just done item.
-         (when opoint (goto-char opoint)))))))
-
-(defun todos-done-item-add-edit-or-delete-comment (&optional arg)
-  "Add a comment to this done item or edit an existing comment.
-With prefix ARG delete an existing comment."
-  (interactive "P")
-  (when (todos-done-item-p)
-    (let ((item (todos-item-string))
-         (opoint (point))
-         (end (save-excursion (todos-item-end)))
-         comment buffer-read-only)
-      (save-excursion
-       (todos-item-start)
-       (if (re-search-forward (concat " \\["
-                                      (regexp-quote todos-comment-string)
-                                      ": \\([^]]+\\)\\]") end t)
-           (if arg
-               (when (todos-y-or-n-p "Delete comment? ")
-                 (delete-region (match-beginning 0) (match-end 0)))
-             (setq comment (read-string "Edit comment: "
-                                        (cons (match-string 1) 1)))
-             (replace-match comment nil nil nil 1))
-         (setq comment (read-string "Enter a comment: "))
-         ;; If user moved point during editing, make sure it moves back.
-         (goto-char opoint)
-         (todos-item-end)
-         (insert " [" todos-comment-string ": " comment "]"))))))
-
-(defun todos-item-undone ()
-  "Restore at least one done item to this category's todo section.
-Prompt for the new priority.  If there are marked items, undo all
-of these, giving the first undone item the new priority and the
-rest following directly in sequence; otherwise, undo just the
-item at point.
-
-If the done item has a comment, ask whether to omit the comment
-from the restored item.  With multiple marked done items with
-comments, only ask once, and if affirmed, omit subsequent
-comments without asking."
-  (interactive)
-  (let* ((cat (todos-current-category))
-        (marked (assoc cat todos-categories-with-marks))
-        (pl (if (and marked (> (cdr marked) 1)) "s" "")))
-    (when (or marked (todos-done-item-p))
-      (let ((buffer-read-only)
-           (opoint (point))
-           (omark (point-marker))
-           (first 'first)
-           (item-count 0)
-           (diary-count 0)
-           start end item ov npoint undone)
-       (and marked (goto-char (point-min)))
-       (catch 'done
-         (while (not (eobp))
-           (when (or (not marked) (and marked (todos-marked-item-p)))
-             (if (not (todos-done-item-p))
-                 (user-error "Only done items can be undone")
-               (todos-item-start)
-               (unless marked
-                 (setq ov (make-overlay (save-excursion (todos-item-start))
-                                        (save-excursion (todos-item-end))))
-                 (overlay-put ov 'face 'todos-search))
-               ;; Find the end of the date string added upon tagging item as
-               ;; done.
-               (setq start (search-forward "] "))
-               (setq item-count (1+ item-count))
-               (unless (looking-at (regexp-quote todos-nondiary-start))
-                 (setq diary-count (1+ diary-count)))
-               (setq end (save-excursion (todos-item-end)))
-               ;; Ask (once) whether to omit done item's comment.  If
-               ;; affirmed, omit subsequent comments without asking.
-               (when (re-search-forward
-                      (concat " \\[" (regexp-quote todos-comment-string)
-                              ": [^]]+\\]") end t)
-                 (unwind-protect
-                     (if (eq first 'first)
-                         (setq first
-                               (if (eq todos-undo-item-omit-comment 'ask)
-                                   (when (todos-y-or-n-p
-                                          (concat "Omit comment" pl
-                                                  " from restored item"
-                                                  pl "? "))
-                                     'omit)
-                                 (when todos-undo-item-omit-comment 'omit)))
-                       t)
-                   (when (and (eq first 'first) ov) (delete-overlay ov)))
-                 (when (eq first 'omit)
-                   (setq end (match-beginning 0))))
-               (setq item (concat item
-                                  (buffer-substring-no-properties start end)
-                                  (when marked "\n")))
-               (unless marked (throw 'done nil))))
-           (todos-forward-item)))
-       (unwind-protect
-           (progn
-             ;; Chop off last newline of multiple items string, since
-             ;; it will be reinserted on setting priority.
-             (and marked (setq item (substring item 0 -1)))
-             (todos-set-item-priority item cat t)
-             (setq npoint (point))
-             (setq undone t))
-         (when ov (delete-overlay ov))
-         (if (not undone)
-             (goto-char opoint)
-           (if marked
-               (progn
-                 (setq item nil)
-                 (re-search-forward
-                  (concat "^" (regexp-quote todos-category-done)) nil t)
-                 (while (not (eobp))
-                   (if (todos-marked-item-p)
-                       (todos-remove-item)
-                     (todos-forward-item)))
-                 (setq todos-categories-with-marks
-                       (assq-delete-all cat todos-categories-with-marks)))
-             (goto-char omark)
-             (todos-remove-item))
-           (todos-update-count 'todo item-count)
-           (todos-update-count 'done (- item-count))
-           (when diary-count (todos-update-count 'diary diary-count))
-           (todos-update-categories-sexp)
-           (let ((todos-show-with-done (> (todos-get-count 'done) 0)))
-             (todos-category-select))
-           ;; Put cursor on undone item.
-           (goto-char npoint)))
-       (set-marker omark nil)))))
-
-;; -----------------------------------------------------------------------------
-;;; Done Item Archives
-;; -----------------------------------------------------------------------------
-
-(defcustom todos-skip-archived-categories nil
-  "Non-nil to handle categories with only archived items specially.
-
-Sequential category navigation using \\[todos-forward-category]
-or \\[todos-backward-category] skips categories that contain only
-archived items.  Other commands still recognize these categories.
-In Todos Categories mode (\\[todos-show-categories-table]) these
-categories shown in `todos-archived-only' face and pressing the
-category button visits the category in the archive instead of the
-todo file."
-  :type 'boolean
-  :group 'todos-display)
+        (new (read-from-minibuffer
+              (format "Rename category \"%s\" to: " cat))))
+    (setq new (todos-validate-name new 'category))
+    (let* ((ofile todos-current-todos-file)
+          (archive (concat (file-name-sans-extension ofile) ".toda"))
+          (buffers (append (list ofile)
+                           (unless (zerop (todos-get-count 'archived cat))
+                             (list archive)))))
+      (dolist (buf buffers)
+       (with-current-buffer (find-file-noselect buf)
+         (let (buffer-read-only)
+           (setq todos-categories (todos-set-categories))
+           (save-excursion
+             (save-restriction
+               (setcar (assoc cat todos-categories) new)
+               (widen)
+               (goto-char (point-min))
+               (todos-update-categories-sexp)
+               (re-search-forward (concat (regexp-quote todos-category-beg)
+                                          "\\(" (regexp-quote cat) "\\)\n")
+                                  nil t)
+               (replace-match new t t nil 1)))))))
+    (force-mode-line-update))
+  (save-excursion (todos-category-select)))
 
-(defun todos-find-archive (&optional ask)
-  "Visit the archive of the current Todos category, if it exists.
-If the category has no archived items, prompt to visit the
-archive anyway.  If there is no archive for this file or with
-non-nil argument ASK, prompt to visit another archive.
+(defun todos-delete-category (&optional arg)
+  "Delete current Todos category provided it is empty.
+With ARG non-nil delete the category unconditionally,
+i.e. including all existing todo and done items."
+  (interactive "P")
+  (let* ((file todos-current-todos-file)
+        (cat (todos-current-category))
+        (todo (todos-get-count 'todo cat))
+        (done (todos-get-count 'done cat))
+        (archived (todos-get-count 'archived cat)))
+    (if (and (not arg)
+            (or (> todo 0) (> done 0)))
+       (message "%s" (substitute-command-keys
+                      (concat "To delete a non-empty category, "
+                              "type C-u \\[todos-delete-category].")))
+      (when (cond ((= (length todos-categories) 1)
+                  (todos-y-or-n-p
+                   (concat "This is the only category in this file; "
+                           "deleting it will also delete the file.\n"
+                           "Do you want to proceed? ")))
+                 ((> archived 0)
+                  (todos-y-or-n-p (concat "This category has archived items; "
+                                    "the archived category will remain\n"
+                                    "after deleting the todo category.  "
+                                    "Do you still want to delete it\n"
+                                    "(see `todos-skip-archived-categories' "
+                                    "for another option)? ")))
+                 (t
+                  (todos-y-or-n-p (concat "Permanently remove category \"" cat
+                                    "\"" (and arg " and all its entries")
+                                    "? "))))
+       (widen)
+       (let ((buffer-read-only)
+             (beg (re-search-backward
+                   (concat "^" (regexp-quote (concat todos-category-beg cat))
+                           "\n") nil t))
+             (end (if (re-search-forward
+                       (concat "\n\\(" (regexp-quote todos-category-beg)
+                               ".*\n\\)") nil t)
+                      (match-beginning 1)
+                    (point-max))))
+         (remove-overlays beg end)
+         (delete-region beg end)
+         (if (= (length todos-categories) 1)
+             ;; If deleted category was the only one, delete the file.
+             (progn
+               (todos-reevaluate-filelist-defcustoms)
+               ;; Skip confirming killing the archive buffer if it has been
+               ;; modified and not saved.
+               (set-buffer-modified-p nil)
+               (delete-file file)
+               (kill-buffer)
+               (message "Deleted Todos file %s." file))
+           (setq todos-categories (delete (assoc cat todos-categories)
+                                              todos-categories))
+           (todos-update-categories-sexp)
+           (setq todos-category-number
+                 (1+ (mod todos-category-number (length todos-categories))))
+           (todos-category-select)
+           (goto-char (point-min))
+           (message "Deleted category %s." cat)))))))
 
-The buffer showing the archive is in Todos Archive mode.  The
-first visit in a session displays the first category in the
-archive, subsequent visits return to the last category
-displayed."
+(defun todos-move-category ()
+  "Move current category to a different Todos file.
+If current category has archived items, also move those to the
+archive of the file moved to, creating it if it does not exist."
   (interactive)
-  (let* ((cat (todos-current-category))
-        (count (todos-get-count 'archived cat))
-        (archive (concat (file-name-sans-extension todos-current-todos-file)
-                         ".toda"))
-        place)
-    (setq place (cond (ask 'other-archive)
-                     ((file-exists-p archive) 'this-archive)
-                     (t (when (todos-y-or-n-p
-                               (concat "This file has no archive; "
-                                       "visit another archive? "))
-                          'other-archive))))
-    (when (eq place 'other-archive)
-      (setq archive (todos-read-file-name "Choose a Todos archive: " t t)))
-    (when (and (eq place 'this-archive) (zerop count))
-      (setq place (when (todos-y-or-n-p
-                         (concat "This category has no archived items;"
-                                 " visit archive anyway? "))
-                    'other-cat)))
-    (when place
+  (when (or (> (length todos-categories) 1)
+           (todos-y-or-n-p (concat "This is the only category in this file; "
+                                   "moving it will also delete the file.\n"
+                                   "Do you want to proceed? ")))
+    (let* ((ofile todos-current-todos-file)
+          (cat (todos-current-category))
+          (nfile (todos-read-file-name
+                  "Choose a Todos file to move this category to: " nil t))
+          (archive (concat (file-name-sans-extension ofile) ".toda"))
+          (buffers (append (list ofile)
+                           (unless (zerop (todos-get-count 'archived cat))
+                             (list archive))))
+          new)
+      (while (equal (file-truename nfile) (file-truename ofile))
+       (setq nfile (todos-read-file-name
+                    "Choose a file distinct from this file: " nil t)))
+      (dolist (buf buffers)
+       (with-current-buffer (find-file-noselect buf)
+         (widen)
+         (goto-char (point-max))
+         (let* ((beg (re-search-backward
+                      (concat "^"
+                              (regexp-quote (concat todos-category-beg cat))
+                              "$")
+                      nil t))
+                (end (if (re-search-forward
+                          (concat "^" (regexp-quote todos-category-beg))
+                          nil t 2)
+                         (match-beginning 0)
+                       (point-max)))
+                (content (buffer-substring-no-properties beg end))
+                (counts (cdr (assoc cat todos-categories)))
+                buffer-read-only)
+           ;; Move the category to the new file.  Also update or create
+           ;; archive file if necessary.
+           (with-current-buffer
+               (find-file-noselect
+                ;; Regenerate todos-archives in case there
+                ;; is a newly created archive.
+                (if (member buf (funcall todos-files-function t))
+                    (concat (file-name-sans-extension nfile) ".toda")
+                  nfile))
+             (let* ((nfile-short (todos-short-file-name nfile))
+                    (prompt (concat
+                             (format "Todos file \"%s\" already has "
+                                     nfile-short)
+                             (format "the category \"%s\";\n" cat)
+                             "enter a new category name: "))
+                    buffer-read-only)
+               (widen)
+               (goto-char (point-max))
+               (insert content)
+               ;; If the file moved to has a category with the same
+               ;; name, rename the moved category.
+               (when (assoc cat todos-categories)
+                 (unless (member (file-truename (buffer-file-name))
+                                 (funcall todos-files-function t))
+                   (setq new (read-from-minibuffer prompt))
+                   (setq new (todos-validate-name new 'category))))
+               ;; Replace old with new name in Todos and archive files.
+               (when new
+                 (goto-char (point-max))
+                 (re-search-backward
+                  (concat "^" (regexp-quote todos-category-beg)
+                          "\\(" (regexp-quote cat) "\\)$") nil t)
+                 (replace-match new nil nil nil 1)))
+             (setq todos-categories
+                   (append todos-categories (list (cons new counts))))
+             (todos-update-categories-sexp)
+             ;; If archive was just created, save it to avoid "File
+             ;; <xyz> no longer exists!" message on invoking
+             ;; `todos-view-archived-items'.
+             (unless (file-exists-p (buffer-file-name))
+               (save-buffer))
+             (todos-category-number (or new cat))
+             (todos-category-select))
+           ;; Delete the category from the old file, and if that was the
+           ;; last category, delete the file.  Also handle archive file
+           ;; if necessary.
+           (remove-overlays beg end)
+           (delete-region beg end)
+           (goto-char (point-min))
+           ;; Put point after todos-categories sexp.
+           (forward-line)
+           (if (eobp)          ; Aside from sexp, file is empty.
+               (progn
+                 ;; Skip confirming killing the archive buffer.
+                 (set-buffer-modified-p nil)
+                 (delete-file todos-current-todos-file)
+                 (kill-buffer)
+                 (when (member todos-current-todos-file todos-files)
+                   (todos-reevaluate-filelist-defcustoms)))
+             (setq todos-categories (delete (assoc cat todos-categories)
+                                            todos-categories))
+             (todos-update-categories-sexp)
+             (todos-category-select)))))
       (set-window-buffer (selected-window)
-                        (set-buffer (find-file-noselect archive)))
-      (if (member place '(other-archive other-cat))
-         (setq todos-category-number 1)
-       (todos-category-number cat))
+                        (set-buffer (find-file-noselect nfile)))
+      (todos-category-number (or new cat))
       (todos-category-select))))
 
-(defun todos-choose-archive ()
-  "Choose an archive and visit it."
-  (interactive)
-  (todos-find-archive t))
+(defun todos-merge-category (&optional file)
+  "Merge current category into another existing category.
 
-(defun todos-archive-done-item (&optional all)
-  "Archive at least one done item in this category.
+With prefix argument FILE, prompt for a specific Todos file and
+choose (with TAB completion) a category in it to merge into;
+otherwise, choose and merge into a category in either the
+current Todos file or a file in `todos-category-completions-files'.
 
-With prefix argument ALL, prompt whether to archive all done
-items in this category and on confirmation archive them.
-Otherwise, if there are marked done items (and no marked todo
-items), archive all of these; otherwise, archive the done item at
-point.
+After merging, the current category's todo and done items are
+appended to the chosen goal category's todo and done items,
+respectively.  The goal category becomes the current category,
+and the previous current category is deleted.
 
-If the archive of this file does not exist, it is created.  If
-this category does not exist in the archive, it is created."
+If both the first and goal categories also have archived items,
+the former are merged to the latter.  If only the first category
+has archived items, the archived category is renamed to the goal
+category."
   (interactive "P")
-  (when (eq major-mode 'todos-mode)
-    (if (and all (zerop (todos-get-count 'done)))
-       (message "No done items in this category")
-      (catch 'end
-       (let* ((cat (todos-current-category))
-              (tbuf (current-buffer))
-              (marked (assoc cat todos-categories-with-marks))
-              (afile (concat (file-name-sans-extension
-                              todos-current-todos-file) ".toda"))
-              (archive (if (file-exists-p afile)
-                           (find-file-noselect afile t)
-                         (get-buffer-create afile)))
-              (item (and (todos-done-item-p)
-                         (concat (todos-item-string) "\n")))
-              (count 0)
-              (opoint (unless (todos-done-item-p) (point)))
-              marked-items beg end all-done
-              buffer-read-only)
-         (cond
-          (all
-           (if (todos-y-or-n-p "Archive all done items in this category? ")
-               (save-excursion
-                 (save-restriction
-                   (goto-char (point-min))
-                   (widen)
-                   (setq beg (progn
-                               (re-search-forward todos-done-string-start
-                                                  nil t)
-                               (match-beginning 0))
-                         end (if (re-search-forward
-                                  (concat "^"
-                                          (regexp-quote todos-category-beg))
-                                  nil t)
-                                 (match-beginning 0)
-                               (point-max))
-                         all-done (buffer-substring-no-properties beg end)
-                         count (todos-get-count 'done))
-                   ;; Restore starting point, unless it was on a done
-                   ;; item, since they will all be deleted.
-                   (when opoint (goto-char opoint))))
-             (throw 'end nil)))
-          (marked
-           (save-excursion
-             (goto-char (point-min))
-             (while (not (eobp))
-               (when (todos-marked-item-p)
-                 (if (not (todos-done-item-p))
-                     (throw 'end (message "Only done items can be archived"))
-                   (setq marked-items
-                         (concat marked-items (todos-item-string) "\n"))
-                   (setq count (1+ count))))
-               (todos-forward-item)))))
-         (if (not (or marked all item))
-             (throw 'end (message "Only done items can be archived"))
-           (with-current-buffer archive
-             (unless buffer-file-name (erase-buffer))
-             (let (buffer-read-only)
-               (widen)
-               (goto-char (point-min))
-               (if (and (re-search-forward
-                         (concat "^" (regexp-quote
-                                      (concat todos-category-beg cat)) "$")
-                         nil t)
-                        (re-search-forward (regexp-quote todos-category-done)
-                                           nil t))
-                   ;; Start of done items section in existing category.
-                   (forward-char)
-                 (todos-add-category nil cat)
-                 ;; Start of done items section in new category.
-                 (goto-char (point-max)))
-               (insert (cond (marked marked-items)
-                             (all all-done)
-                             (item)))
-               (todos-update-count 'done (if (or marked all) count 1) cat)
-               (todos-update-categories-sexp)
-               ;; If archive is new, save to file now (using write-region in
-               ;; order not to get prompted for file to save to), to let
-               ;; auto-mode-alist take effect below.
-               (unless buffer-file-name
-                 (write-region nil nil afile)
-                 (kill-buffer))))
-           (with-current-buffer tbuf
-             (cond
-              (all
-               (save-excursion
-                 (save-restriction
-                   ;; Make sure done items are accessible.
-                   (widen)
-                   (remove-overlays beg end)
-                   (delete-region beg end)
-                   (todos-update-count 'done (- count))
-                   (todos-update-count 'archived count))))
-              ((or marked
-                   ;; If we're archiving all done items, can't
-                   ;; first archive item point was on, since
-                   ;; that will short-circuit the rest.
-                   (and item (not all)))
-               (and marked (goto-char (point-min)))
-               (catch 'done
-                 (while (not (eobp))
-                   (if (or (and marked (todos-marked-item-p)) item)
-                       (progn
-                         (todos-remove-item)
-                         (todos-update-count 'done -1)
-                         (todos-update-count 'archived 1)
-                         ;; Don't leave point below last item.
-                         (and item (bolp) (eolp) (< (point-min) (point-max))
-                              (todos-backward-item))
-                         (when item
-                           (throw 'done (setq item nil))))
-                     (todos-forward-item))))))
-             (when marked
-               (setq todos-categories-with-marks
-                     (assq-delete-all cat todos-categories-with-marks)))
-             (todos-update-categories-sexp)
-             (todos-prefix-overlays)))
-         (find-file afile)
-         (todos-category-number cat)
-         (todos-category-select)
-         (split-window-below)
-         (set-window-buffer (selected-window) tbuf)
-         ;; Make todo file current to select category.
-         (find-file (buffer-file-name tbuf))
-         ;; Make sure done item separator is hidden (if done items
-         ;; were initially visible).
-         (let (todos-show-with-done) (todos-category-select)))))))
-
-(defun todos-unarchive-items ()
-  "Unarchive at least one item in this archive category.
-If there are marked items, unarchive all of these; otherwise,
-unarchive the item at point.
-
-Unarchived items are restored as done items to the corresponding
-category in the Todos file, inserted at the top of done items
-section.  If all items in the archive category have been
-restored, the category is deleted from the archive.  If this was
-the only category in the archive, the archive file is deleted."
-  (interactive)
-  (when (eq major-mode 'todos-archive-mode)
-    (let* ((cat (todos-current-category))
-          (tbuf (find-file-noselect
-                 (concat (file-name-sans-extension todos-current-todos-file)
-                         ".todo") t))
-          (marked (assoc cat todos-categories-with-marks))
-          (item (concat (todos-item-string) "\n"))
-          (marked-count 0)
-          marked-items
-          buffer-read-only)
-      (when marked
-       (save-excursion
-         (goto-char (point-min))
-         (while (not (eobp))
-           (when (todos-marked-item-p)
-             (setq marked-items (concat marked-items (todos-item-string) "\n"))
-             (setq marked-count (1+ marked-count)))
-           (todos-forward-item))))
-      ;; Restore items to top of category's done section and update counts.
-      (with-current-buffer tbuf
-       (let (buffer-read-only newcat)
-         (widen)
-         (goto-char (point-min))
-         ;; Find the corresponding todo category, or if there isn't
-         ;; one, add it.
-         (unless (re-search-forward
-                  (concat "^" (regexp-quote (concat todos-category-beg cat))
-                          "$") nil t)
-           (todos-add-category nil cat)
-           (setq newcat t))
-         ;; Go to top of category's done section.
+  (let* ((tfile todos-current-todos-file)
+        (cat (todos-current-category))
+        (cat+file (todos-read-category "Merge into category: " 'todo file))
+        (goal (car cat+file))
+        (gfile  (cdr cat+file))
+        (archive (concat (file-name-sans-extension (if file gfile tfile))
+                         ".toda"))
+        archived-count here)
+    ;; Merge in todo file.
+    (with-current-buffer (get-buffer (find-file-noselect tfile))
+      (widen)
+      (let* ((buffer-read-only nil)
+            (cbeg (progn
+                    (re-search-backward
+                     (concat "^" (regexp-quote todos-category-beg)) nil t)
+                    (point-marker)))
+            (tbeg (progn (forward-line) (point-marker)))
+            (dbeg (progn
+                    (re-search-forward
+                     (concat "^" (regexp-quote todos-category-done)) nil t)
+                    (forward-line) (point-marker)))
+            ;; Omit empty line between todo and done items.
+            (tend (progn (forward-line -2) (point-marker)))
+            (cend (progn
+                    (if (re-search-forward
+                         (concat "^" (regexp-quote todos-category-beg)) nil t)
+                        (progn
+                          (goto-char (match-beginning 0))
+                          (point-marker))
+                      (point-max-marker))))
+            (todo (buffer-substring-no-properties tbeg tend))
+            (done (buffer-substring-no-properties dbeg cend)))
+       (goto-char (point-min))
+       ;; Merge any todo items.
+       (unless (zerop (length todo))
+         (re-search-forward
+          (concat "^" (regexp-quote (concat todos-category-beg goal)) "$")
+          nil t)
          (re-search-forward
           (concat "^" (regexp-quote todos-category-done)) nil t)
-         (forward-line)
-         (cond (marked
-                (insert marked-items)
-                (todos-update-count 'done marked-count cat)
-                (unless newcat         ; Newly added category has no archive.
-                  (todos-update-count 'archived (- marked-count) cat)))
-               (t
-                (insert item)
-                (todos-update-count 'done 1 cat)
-                (unless newcat         ; Newly added category has no archive.
-                  (todos-update-count 'archived -1 cat))))
-         (todos-update-categories-sexp)))
-      ;; Delete restored items from archive.
-      (when marked
-       (setq item nil)
-       (goto-char (point-min)))
-      (catch 'done
-       (while (not (eobp))
-         (if (or (todos-marked-item-p) item)
-             (progn
-               (todos-remove-item)
-               (when item
-                 (throw 'done (setq item nil))))
-           (todos-forward-item))))
-      (todos-update-count 'done (if marked (- marked-count) -1) cat)
-      ;; If that was the last category in the archive, delete the whole file.
-      (if (= (length todos-categories) 1)
-         (progn
-           (delete-file todos-current-todos-file)
-           ;; Kill the archive buffer silently.
-           (set-buffer-modified-p nil)
-           (kill-buffer))
-       ;; Otherwise, if the archive category is now empty, delete it.
-       (when (eq (point-min) (point-max))
-         (widen)
-         (let ((beg (re-search-backward
-                     (concat "^" (regexp-quote todos-category-beg) cat "$")
-                     nil t))
-               (end (if (re-search-forward
-                         (concat "^" (regexp-quote todos-category-beg))
-                         nil t 2)
+         (forward-line -1)
+         (setq here (point-marker))
+         (insert todo)
+         (todos-update-count 'todo (todos-get-count 'todo cat) goal))
+       ;; Merge any done items.
+       (unless (zerop (length done))
+         (goto-char (if (re-search-forward
+                         (concat "^" (regexp-quote todos-category-beg)) nil t)
                         (match-beginning 0)
-                      (point-max))))
-           (remove-overlays beg end)
-           (delete-region beg end)
-           (setq todos-categories (delete (assoc cat todos-categories)
-                                          todos-categories))
-           (todos-update-categories-sexp))))
-      ;; Visit category in Todos file and show restored done items.
-      (let ((tfile (buffer-file-name tbuf))
-           (todos-show-with-done t))
-       (set-window-buffer (selected-window)
-                          (set-buffer (find-file-noselect tfile)))
-       (todos-category-number cat)
+                      (point-max)))
+         (when (zerop (length todo)) (setq here (point-marker)))
+         (insert done)
+         (todos-update-count 'done (todos-get-count 'done cat) goal))
+       (remove-overlays cbeg cend)
+       (delete-region cbeg cend)
+       (setq todos-categories (delete (assoc cat todos-categories)
+                                      todos-categories))
+       (todos-update-categories-sexp)
+       (mapc (lambda (m) (set-marker m nil)) (list cbeg tbeg dbeg tend cend))))
+    (when (file-exists-p archive)
+      ;; Merge in archive file.
+      (with-current-buffer (get-buffer (find-file-noselect archive))
+       (widen)
+       (goto-char (point-min))
+       (let ((buffer-read-only nil)
+             (cbeg (save-excursion
+                     (when (re-search-forward
+                            (concat "^" (regexp-quote
+                                         (concat todos-category-beg cat)) "$")
+                            nil t)
+                       (goto-char (match-beginning 0))
+                       (point-marker))))
+             (gbeg (save-excursion
+                     (when (re-search-forward
+                            (concat "^" (regexp-quote
+                                         (concat todos-category-beg goal)) "$")
+                            nil t)
+                       (goto-char (match-beginning 0))
+                       (point-marker))))
+             cend carch)
+         (when cbeg
+           (setq archived-count (todos-get-count 'done cat))
+           (setq cend (save-excursion
+                        (if (re-search-forward
+                             (concat "^" (regexp-quote todos-category-beg))
+                             nil t)
+                            (match-beginning 0)
+                          (point-max))))
+           (setq carch (save-excursion (goto-char cbeg) (forward-line)
+                         (buffer-substring-no-properties (point) cend)))
+           ;; If both categories of the merge have archived items, merge the
+           ;; source items to the goal items, else "merge" by renaming the
+           ;; source category to goal.
+           (if gbeg
+               (progn
+                 (goto-char (if (re-search-forward
+                                 (concat "^" (regexp-quote todos-category-beg))
+                                 nil t)
+                                (match-beginning 0)
+                              (point-max)))
+                 (insert carch)
+                 (remove-overlays cbeg cend)
+                 (delete-region cbeg cend))
+             (goto-char cbeg)
+             (search-forward cat)
+             (replace-match goal))
+           (setq todos-categories (todos-make-categories-list t))
+           (todos-update-categories-sexp)))))
+    (with-current-buffer (get-file-buffer tfile)
+      (when archived-count
+       (unless (zerop archived-count)
+         (todos-update-count 'archived archived-count goal)
+         (todos-update-categories-sexp)))
+      (todos-category-number goal)
+      ;; If there are only merged done items, show them.
+      (let ((todos-show-with-done (zerop (todos-get-count 'todo goal))))
        (todos-category-select)
-       (message "Items unarchived.")))))
-
-(defun todos-jump-to-archive-category (&optional file)
-  "Prompt for a category in a Todos archive and jump to it.
-With prefix argument FILE, prompt for an archive and choose (with
-TAB completion) a category in it to jump to; otherwise, choose
-and jump to any category in the current archive."
-  (interactive "P")
-  (todos-jump-to-category file 'archive))
+       ;; Put point on the first merged item.
+       (goto-char here)))
+    (set-marker here nil)))
 
 ;; -----------------------------------------------------------------------------
-;;; Todos mode display options
+;;; Item editing
 ;; -----------------------------------------------------------------------------
 
-(defcustom todos-prefix ""
-  "String prefixed to todo items for visual distinction."
-  :type '(string :validate
-                (lambda (widget)
-                  (when (string= (widget-value widget) todos-item-mark)
-                    (widget-put
-                     widget :error
-                     "Invalid value: must be distinct from `todos-item-mark'")
-                    widget)))
-  :initialize 'custom-initialize-default
-  :set 'todos-reset-prefix
-  :group 'todos-display)
-
-(defcustom todos-number-prefix t
-  "Non-nil to prefix items with consecutively increasing integers.
-These reflect the priorities of the items in each category."
+(defcustom todos-include-in-diary nil
+  "Non-nil to allow new Todo items to be included in the diary."
   :type 'boolean
-  :initialize 'custom-initialize-default
-  :set 'todos-reset-prefix
-  :group 'todos-display)
-
-(defcustom todos-done-separator-string "="
-  "String determining the value of variable `todos-done-separator'.
-
-If the string consists of a single character,
-`todos-done-separator' will be the string made by repeating this
-character for the width of the window, and the length is
-automatically recalculated when the window width changes.  If the
-string consists of more (or less) than one character, it will be
-the value of `todos-done-separator'."
-  :type 'string
-  :initialize 'custom-initialize-default
-  :set 'todos-reset-done-separator-string
-  :group 'todos-display)
-
-(defcustom todos-done-string "DONE "
-  "Identifying string appended to the front of done todos items."
-  :type 'string
-  :initialize 'custom-initialize-default
-  :set 'todos-reset-done-string
-  :group 'todos-display)
-
-(defcustom todos-comment-string "COMMENT"
-  "String inserted before optional comment appended to done item."
-  :type 'string
-  :initialize 'custom-initialize-default
-  :set 'todos-reset-comment-string
-  :group 'todos-display)
+  :group 'todos-edit)
 
-(defcustom todos-show-with-done nil
-  "Non-nil to display done items in all categories."
+(defcustom todos-diary-nonmarking nil
+  "Non-nil to insert new Todo diary items as nonmarking by default.
+This appends `diary-nonmarking-symbol' to the front of an item on
+insertion provided it doesn't begin with `todos-nondiary-marker'."
   :type 'boolean
-  :group 'todos-display)
-
-(defun todos-mode-line-control (cat)
-  "Return a mode line control for todo or archive file buffers.
-Argument CAT is the name of the current Todos category.
-This function is the value of the user variable
-`todos-mode-line-function'."
-  (let ((file (todos-short-file-name todos-current-todos-file)))
-    (format "%s category %d: %s" file todos-category-number cat)))
-
-(defcustom todos-mode-line-function 'todos-mode-line-control
-  "Function that returns a mode line control for Todos buffers.
-The function expects one argument holding the name of the current
-Todos category.  The resulting control becomes the local value of
-`mode-line-buffer-identification' in each Todos buffer."
-  :type 'function
-  :group 'todos-display)
+  :group 'todos-edit)
 
-(defcustom todos-highlight-item nil
-  "Non-nil means highlight items at point."
+(defcustom todos-always-add-time-string nil
+  "Non-nil adds current time to a new item's date header by default.
+When the Todos insertion commands have a non-nil \"maybe-notime\"
+argument, this reverses the effect of
+`todos-always-add-time-string': if t, these commands omit the
+current time, if nil, they include it."
   :type 'boolean
-  :initialize 'custom-initialize-default
-  :set 'todos-reset-highlight-item
-  :group 'todos-display)
+  :group 'todos-edit)
 
-(defcustom todos-wrap-lines t
-  "Non-nil to activate Visual Line mode and use wrap prefix."
+(defcustom todos-use-only-highlighted-region t
+  "Non-nil to enable inserting only highlighted region as new item."
   :type 'boolean
-  :group 'todos-display)
+  :group 'todos-edit)
 
-(defcustom todos-indent-to-here 3
-  "Number of spaces to indent continuation lines of items.
-This must be a positive number to ensure such items are fully
-shown in the Fancy Diary display."
-  :type '(integer :validate
-                 (lambda (widget)
-                   (unless (> (widget-value widget) 0)
-                     (widget-put widget :error
-                                 "Invalid value: must be a positive integer")
-                     widget)))
-  :group 'todos-display)
+(defcustom todos-item-mark "*"
+  "String used to mark items.
+To ensure item marking works, change the value of this option
+only when no items are marked."
+  :type '(string :validate
+                (lambda (widget)
+                  (when (string= (widget-value widget) todos-prefix)
+                    (widget-put
+                     widget :error
+                     "Invalid value: must be distinct from `todos-prefix'")
+                    widget)))
+  :set (lambda (symbol value)
+        (custom-set-default symbol (propertize value 'face 'todos-mark)))
+  :group 'todos-edit)
 
-(defun todos-indent ()
-  "Indent from point to `todos-indent-to-here'."
-  (indent-to todos-indent-to-here todos-indent-to-here))
+(defcustom todos-comment-string "COMMENT"
+  "String inserted before optional comment appended to done item."
+  :type 'string
+  :initialize 'custom-initialize-default
+  :set 'todos-reset-comment-string
+  :group 'todos-edit)
 
-;; -----------------------------------------------------------------------------
-;;; Display Commands
-;; -----------------------------------------------------------------------------
+(defcustom todos-undo-item-omit-comment 'ask
+  "Whether to omit done item comment on undoing the item.
+Nil means never omit the comment, t means always omit it, `ask'
+means prompt user and omit comment only on confirmation."
+  :type '(choice (const :tag "Never" nil)
+                (const :tag "Always" t)
+                (const :tag "Ask" ask))
+  :group 'todos-edit)
 
-(defun todos-toggle-prefix-numbers ()
-  "Hide item numbering if shown, show if hidden."
-  (interactive)
-  (save-excursion
-    (save-restriction
-      (goto-char (point-min))
-      (let* ((ov (todos-get-overlay 'prefix))
-            (show-done (re-search-forward todos-done-string-start nil t))
-            (todos-show-with-done show-done)
-            (todos-number-prefix (not (equal (overlay-get ov 'before-string)
-                                             "1 "))))
-       (if (eq major-mode 'todos-filtered-items-mode)
-           (todos-prefix-overlays)
-         (todos-category-select))))))
+(defun todos-toggle-mark-item (&optional n)
+  "Mark item with `todos-item-mark' if unmarked, otherwise unmark it.
+With a positive numerical prefix argument N, change the
+marking of the next N items."
+  (interactive "p")
+  (when (todos-item-string)
+    (unless (> n 1) (setq n 1))
+    (dotimes (i n)
+      (let* ((cat (todos-current-category))
+            (marks (assoc cat todos-categories-with-marks))
+            (ov (progn
+                  (unless (looking-at todos-item-start)
+                    (todos-item-start))
+                  (todos-get-overlay 'prefix)))
+            (pref (overlay-get ov 'before-string)))
+       (if (todos-marked-item-p)
+           (progn
+             (overlay-put ov 'before-string (substring pref 1))
+             (if (= (cdr marks) 1)     ; Deleted last mark in this category.
+                 (setq todos-categories-with-marks
+                       (assq-delete-all cat todos-categories-with-marks))
+               (setcdr marks (1- (cdr marks)))))
+         (overlay-put ov 'before-string (concat todos-item-mark pref))
+         (if marks
+             (setcdr marks (1+ (cdr marks)))
+           (push (cons cat 1) todos-categories-with-marks))))
+      (todos-forward-item))))
 
-(defun todos-toggle-view-done-items ()
-  "Show hidden or hide visible done items in current category."
+(defun todos-mark-category ()
+  "Mark all visiblw items in this category with `todos-item-mark'."
   (interactive)
-  (if (zerop (todos-get-count 'done (todos-current-category)))
-      (message "There are no done items in this category.")
-    (let ((opoint (point)))
+  (let* ((cat (todos-current-category))
+        (marks (assoc cat todos-categories-with-marks)))
+    (save-excursion
       (goto-char (point-min))
-      (let* ((shown (re-search-forward todos-done-string-start nil t))
-            (todos-show-with-done (not shown)))
-       (todos-category-select)
-       (goto-char opoint)
-       ;; If start of done items sections is below the bottom of the
-       ;; window, make it visible.
-       (unless shown
-         (setq shown (progn
-                       (goto-char (point-min))
-                       (re-search-forward todos-done-string-start nil t)))
-         (if (not (pos-visible-in-window-p shown))
-             (recenter)
-           (goto-char opoint)))))))
+      (while (not (eobp))
+       (let* ((ov (todos-get-overlay 'prefix))
+              (pref (overlay-get ov 'before-string)))
+         (unless (todos-marked-item-p)
+           (overlay-put ov 'before-string (concat todos-item-mark pref))
+           (if marks
+               (setcdr marks (1+ (cdr marks)))
+             (push (cons cat 1) todos-categories-with-marks))))
+       (todos-forward-item)))))
 
-(defun todos-toggle-view-done-only ()
-  "Switch between displaying only done or only todo items."
+(defun todos-unmark-category ()
+  "Remove `todos-item-mark' from all visible items in this category."
   (interactive)
-  (setq todos-show-done-only (not todos-show-done-only))
-  (todos-category-select))
+  (let* ((cat (todos-current-category))
+        (marks (assoc cat todos-categories-with-marks)))
+    (save-excursion
+      (goto-char (point-min))
+      (while (not (eobp))
+       (let* ((ov (todos-get-overlay 'prefix))
+              ;; No overlay on empty line between todo and done items.
+              (pref (when ov (overlay-get ov 'before-string))))
+         (when (todos-marked-item-p)
+           (overlay-put ov 'before-string (substring pref 1)))
+         (todos-forward-item))))
+    (setq todos-categories-with-marks
+         (delq marks todos-categories-with-marks))))
 
-(defun todos-toggle-item-highlighting ()
-  "Highlight or unhighlight the todo item the cursor is on."
-  (interactive)
-  (require 'hl-line)
-  (when (memq major-mode '(todos-mode todos-archive-mode
-                                     todos-filtered-items-mode))
-    (if hl-line-mode
-       (hl-line-mode -1)
-      (hl-line-mode 1))))
+(defvar todos-date-from-calendar nil
+  "Helper variable for setting item date from the Emacs Calendar.")
 
-(defun todos-toggle-item-header ()
-  "Hide or show item date-time headers in the current file.
-With done items, this hides only the done date-time string, not
-the the original date-time string."
-  (interactive)
-  (save-excursion
-    (save-restriction
-      (goto-char (point-min))
-      (let ((ov (todos-get-overlay 'header)))
-       (if ov
-           (remove-overlays 1 (1+ (buffer-size)) 'todos 'header)
-         (widen)
-         (goto-char (point-min))
-         (while (not (eobp))
-           (when (re-search-forward
-                  (concat todos-item-start
-                          "\\( " diary-time-regexp "\\)?"
-                          (regexp-quote todos-nondiary-end) "? ")
-                  nil t)
-             (setq ov (make-overlay (match-beginning 0) (match-end 0) nil t))
-             (overlay-put ov 'todos 'header)
-             (overlay-put ov 'display ""))
-           (todos-forward-item)))))))
+(defun todos-basic-insert-item (&optional arg diary nonmarking date-type time
+                                   region-or-here)
+  "Insert a new Todo item into a category.
+This is the function from which the generated Todos item
+insertion commands derive.
 
-;; -----------------------------------------------------------------------------
-;;; Faces
-;; -----------------------------------------------------------------------------
+The generated commands have mnenomic key bindings based on the
+arguments' values and their order in the command's argument list,
+as follows: (1) for DIARY `d', (2) for NONMARKING `k', (3) for
+DATE-TYPE either `c' for calendar or `d' for date or `n' for
+weekday name, (4) for TIME `t', (5) for REGION-OR-HERE either `r'
+for region or `h' for here.  Sequences of these keys are appended
+to the insertion prefix key `i'.  Keys that allow a following
+key (i.e., any but `r' or `h') must be doubled when used finally.
+For example, the command bound to the key sequence `i y h' will
+insert a new item with today's date, marked according to the
+DIARY argument described below, and with priority according to
+the HERE argument; `i y y' does the same except that the priority
+is not given by HERE but by prompting.
 
-(defface todos-prefix-string
-  ;; '((t :inherit font-lock-constant-face))
-  '((((class grayscale) (background light))
-     (:foreground "LightGray" :weight bold :underline t))
-    (((class grayscale) (background dark))
-     (:foreground "Gray50" :weight bold :underline t))
-    (((class color) (min-colors 88) (background light)) (:foreground "dark cyan"))
-    (((class color) (min-colors 88) (background dark)) (:foreground "Aquamarine"))
-    (((class color) (min-colors 16) (background light)) (:foreground "CadetBlue"))
-    (((class color) (min-colors 16) (background dark)) (:foreground "Aquamarine"))
-    (((class color) (min-colors 8)) (:foreground "magenta"))
-    (t (:weight bold :underline t)))
-  "Face for Todos prefix or numerical priority string."
-  :group 'todos-faces)
+In command invocations, ARG is passed as a prefix argument as
+follows.  With no prefix argument, add the item to the current
+category; with one prefix argument (`C-u'), prompt for a category
+from the current Todos file; with two prefix arguments (`C-u C-u'),
+first prompt for a Todos file, then a category in that file.  If
+a non-existing category is entered, ask whether to add it to the
+Todos file; if answered affirmatively, add the category and
+insert the item there.
 
-(defface todos-top-priority
-  ;; bold font-lock-comment-face
-  '((default :weight bold)
-    (((class grayscale) (background light)) :foreground "DimGray" :slant italic)
-    (((class grayscale) (background dark)) :foreground "LightGray" :slant italic)
-    (((class color) (min-colors 88) (background light)) :foreground "Firebrick")
-    (((class color) (min-colors 88) (background dark)) :foreground "chocolate1")
-    (((class color) (min-colors 16) (background light)) :foreground "red")
-    (((class color) (min-colors 16) (background dark)) :foreground "red1")
-    (((class color) (min-colors 8) (background light)) :foreground "red")
-    (((class color) (min-colors 8) (background dark)) :foreground "yellow")
-    (t :slant italic))
-  "Face for top priority Todos item numerical priority string.
-The item's priority number string has this face if the number is
-less than or equal the category's top priority setting."
-  :group 'todos-faces)
+The remaining arguments are set or left nil by the generated item
+insertion commands; their meanings are described in the follows
+paragraphs.
 
-(defface todos-mark
-  ;; '((t :inherit font-lock-warning-face))
-  '((((class color)
-      (min-colors 88)
-      (background light))
-     (:weight bold :foreground "Red1"))
-    (((class color)
-      (min-colors 88)
-      (background dark))
-     (:weight bold :foreground "Pink"))
-    (((class color)
-      (min-colors 16)
-      (background light))
-     (:weight bold :foreground "Red1"))
-    (((class color)
-      (min-colors 16)
-      (background dark))
-     (:weight bold :foreground "Pink"))
-    (((class color)
-      (min-colors 8))
-     (:foreground "red"))
-    (t
-     (:weight bold :inverse-video t)))
-  "Face for marks on marked items."
-  :group 'todos-faces)
+When argument DIARY is non-nil, this overrides the intent of the
+user option `todos-include-in-diary' for this item: if
+`todos-include-in-diary' is nil, include the item in the Fancy
+Diary display, and if it is non-nil, exclude the item from the
+Fancy Diary display.  When DIARY is nil, `todos-include-in-diary'
+has its intended effect.
 
-(defface todos-button
-  ;; '((t :inherit widget-field))
-  '((((type tty))
-     (:foreground "black" :background "yellow3"))
-    (((class grayscale color)
-      (background light))
-     (:background "gray85"))
-    (((class grayscale color)
-      (background dark))
-     (:background "dim gray"))
-    (t
-     (:slant italic)))
-  "Face for buttons in table of categories."
-  :group 'todos-faces)
+When the item is included in the Fancy Diary display and the
+argument NONMARKING is non-nil, this overrides the intent of the
+user option `todos-diary-nonmarking' for this item: if
+`todos-diary-nonmarking' is nil, append `diary-nonmarking-symbol'
+to the item, and if it is non-nil, omit `diary-nonmarking-symbol'.
 
-(defface todos-sorted-column
-  '((((type tty))
-     (:inverse-video t))
-    (((class color)
-      (background light))
-     (:background "grey85"))
-    (((class color)
-      (background dark))
-      (:background "grey85" :foreground "grey10"))
-    (t
-     (:background "gray")))
-  "Face for sorted column in table of categories."
-  :group 'todos-faces)
+The argument DATE-TYPE determines the content of the item's
+mandatory date header string and how it is added:
+- If DATE-TYPE is the symbol `calendar', the Calendar pops up and
+  when the user puts the cursor on a date and hits RET, that
+  date, in the format set by `calendar-date-display-form',
+  becomes the date in the header.
+- If DATE-TYPE is a string matching the regexp
+  `todos-date-pattern', that string becomes the date in the
+  header.  This case is for the command
+  `todos-insert-item-from-calendar' which is called from the
+  Calendar.
+- If DATE-TYPE is the symbol `date', the header contains the date
+  in the format set by `calendar-date-display-form', with year,
+  month and day individually prompted for (month with tab
+  completion).
+- If DATE-TYPE is the symbol `dayname' the header contains a
+  weekday name instead of a date, prompted for with tab
+  completion.
+- If DATE-TYPE has any other value (including nil or none) the
+  header contains the current date (in the format set by
+  `calendar-date-display-form').
 
-(defface todos-archived-only
-  ;; '((t (:inherit (shadow))))
-  '((((class color)
-      (background light))
-     (:foreground "grey50"))
-    (((class color)
-      (background dark))
-     (:foreground "grey70"))
-    (t
-     (:foreground "gray")))
-  "Face for archived-only category names in table of categories."
-  :group 'todos-faces)
+With non-nil argument TIME prompt for a time string, which must
+match `diary-time-regexp'.  Typing `<return>' at the prompt
+returns the current time, if the user option
+`todos-always-add-time-string' is non-nil, otherwise the empty
+string (i.e., no time string).  If TIME is absent or nil, add or
+omit the current time string according as
+`todos-always-add-time-string' is non-nil or nil, respectively.
 
-(defface todos-search
-  ;; '((t :inherit match))
-  '((((class color)
-      (min-colors 88)
-      (background light))
-     (:background "yellow1"))
-    (((class color)
-      (min-colors 88)
-      (background dark))
-     (:background "RoyalBlue3"))
-    (((class color)
-      (min-colors 8)
-      (background light))
-     (:foreground "black" :background "yellow"))
-    (((class color)
-      (min-colors 8)
-      (background dark))
-     (:foreground "white" :background "blue"))
-    (((type tty)
-      (class mono))
-     (:inverse-video t))
-    (t
-     (:background "gray")))
-  "Face for matches found by `todos-search'."
-  :group 'todos-faces)
+The argument REGION-OR-HERE determines the source and location of
+the new item:
+- If the REGION-OR-HERE is the symbol `here', prompt for the text of
+  the new item and, if the command was invoked with point in the todo
+  items section of the current category, give the new item the
+  priority of the item at point, lowering the latter's priority and
+  the priority of the remaining items.  If point is in the done items
+  section of the category, insert the new item as the first todo item
+  in the category.  Likewise, if the command with `here' is invoked
+  outside of the current category, jump to the chosen category and
+  insert the new item as the first item in the category.
+- If REGION-OR-HERE is the symbol `region', use the region of the
+  current buffer as the text of the new item, depending on the
+  value of user option `todos-use-only-highlighted-region': if
+  this is non-nil, then use the region only when it is
+  highlighted; otherwise, use the region regardless of
+  highlighting.  An error is signalled if there is no region in
+  the current buffer.  Prompt for the item's priority in the
+  category (an integer between 1 and one more than the number of
+  items in the category), and insert the item accordingly.
+- If REGION-OR-HERE has any other value (in particular, nil or
+  none), prompt for the text and the item's priority, and insert
+  the item accordingly."
+  ;; If invoked outside of Todos mode and there is not yet any Todos
+  ;; file, initialize one.
+  (if (null todos-files)
+      (todos-show)
+    (let ((region (eq region-or-here 'region))
+         (here (eq region-or-here 'here)))
+      (when region
+       (let (use-empty-active-region)
+         (unless (and todos-use-only-highlighted-region (use-region-p))
+           (user-error "There is no active region"))))
+      (let* ((obuf (current-buffer))
+            (ocat (todos-current-category))
+            (opoint (point))
+            (todos-mm (eq major-mode 'todos-mode))
+            (cat+file (cond ((equal arg '(4))
+                             (todos-read-category "Insert in category: "))
+                            ((equal arg '(16))
+                             (todos-read-category "Insert in category: "
+                                                  nil 'file))
+                            (t
+                             (cons (todos-current-category)
+                                   (or todos-current-todos-file
+                                       (and todos-show-current-file
+                                            todos-global-current-todos-file)
+                                       (todos-absolute-file-name
+                                        todos-default-todos-file))))))
+            (cat (car cat+file))
+            (file (cdr cat+file))
+            (new-item (if region
+                          (buffer-substring-no-properties
+                           (region-beginning) (region-end))
+                        (read-from-minibuffer "Todo item: ")))
+            (date-string (cond
+                          ((eq date-type 'date)
+                           (todos-read-date))
+                          ((eq date-type 'dayname)
+                           (todos-read-dayname))
+                          ((eq date-type 'calendar)
+                           (setq todos-date-from-calendar t)
+                           (or (todos-set-date-from-calendar)
+                               ;; If user exits Calendar before choosing
+                               ;; a date, cancel item insertion.
+                               (keyboard-quit)))
+                          ((and (stringp date-type)
+                                (string-match todos-date-pattern date-type))
+                           (setq todos-date-from-calendar date-type)
+                           (todos-set-date-from-calendar))
+                          (t
+                           (calendar-date-string
+                            (calendar-current-date) t t))))
+            (time-string (or (and time (todos-read-time))
+                             (and todos-always-add-time-string
+                                  (substring (current-time-string) 11 16)))))
+       (setq todos-date-from-calendar nil)
+       (find-file-noselect file 'nowarn)
+       (set-window-buffer (selected-window)
+                          (set-buffer (find-buffer-visiting file)))
+       ;; If this command was invoked outside of a Todos buffer, the
+       ;; call to todos-current-category above returned nil.  If we
+       ;; just entered Todos mode now, then cat was set to the file's
+       ;; first category, but if todos-mode was already enabled, cat
+       ;; did not get set, so we have to set it explicitly.
+       (unless cat
+         (setq cat (todos-current-category)))
+       (setq todos-current-todos-file file)
+       (unless todos-global-current-todos-file
+         (setq todos-global-current-todos-file todos-current-todos-file))
+       (let ((buffer-read-only nil)
+             (called-from-outside (not (and todos-mm (equal cat ocat))))
+             done-only item-added)
+         (setq new-item
+               ;; Add date, time and diary marking as required.
+               (concat (if (not (and diary (not todos-include-in-diary)))
+                           todos-nondiary-start
+                         (when (and nonmarking (not todos-diary-nonmarking))
+                           diary-nonmarking-symbol))
+                       date-string (when (and time-string ; Can be empty.
+                                              (not (zerop (length
+                                                           time-string))))
+                                     (concat " " time-string))
+                       (when (not (and diary (not todos-include-in-diary)))
+                         todos-nondiary-end)
+                       " " new-item))
+         ;; Indent newlines inserted by C-q C-j if nonspace char follows.
+         (setq new-item (replace-regexp-in-string "\\(\n\\)[^[:blank:]]"
+                                                  "\n\t" new-item nil nil 1))
+         (unwind-protect
+             (progn
+               ;; Make sure the correct category is selected.  There
+               ;; are two cases: (i) we just visited the file, so no
+               ;; category is selected yet, or (ii) we invoked
+               ;; insertion "here" from outside the category we want
+               ;; to insert in (with priority insertion, category
+               ;; selection is done by todos-set-item-priority).
+               (when (or (= (- (point-max) (point-min)) (buffer-size))
+                         (and here called-from-outside))
+                 (todos-category-number cat)
+                 (todos-category-select))
+               ;; If only done items are displayed in category,
+               ;; toggle to todo items before inserting new item.
+               (when (save-excursion
+                       (goto-char (point-min))
+                       (looking-at todos-done-string-start))
+                 (setq done-only t)
+                 (todos-toggle-view-done-only))
+               (if here
+                   (progn
+                     ;; If command was invoked with point in done
+                     ;; items section or outside of the current
+                     ;; category, can't insert "here", so to be
+                     ;; useful give new item top priority.
+                     (when (or (todos-done-item-section-p)
+                               called-from-outside
+                               done-only)
+                       (goto-char (point-min)))
+                     (todos-insert-with-overlays new-item))
+                 (todos-set-item-priority new-item cat t))
+               (setq item-added t))
+           ;; If user cancels before setting priority, restore
+           ;; display.
+           (unless item-added
+             (if ocat
+                 (progn
+                   (unless (equal cat ocat)
+                     (todos-category-number ocat)
+                     (todos-category-select))
+                   (and done-only (todos-toggle-view-done-only)))
+               (set-window-buffer (selected-window) (set-buffer obuf)))
+             (goto-char opoint))
+           ;; If the todo items section is not visible when the
+           ;; insertion command is called (either because only done
+           ;; items were shown or because the category was not in the
+           ;; current buffer), then if the item is inserted at the
+           ;; end of the category, point is at eob and eob at
+           ;; window-start, so that higher priority todo items are
+           ;; out of view.  So we recenter to make sure the todo
+           ;; items are displayed in the window.
+           (when item-added (recenter)))
+         (todos-update-count 'todo 1)
+         (if (or diary todos-include-in-diary) (todos-update-count 'diary 1))
+         (todos-update-categories-sexp))))))
+
+(defun todos-set-date-from-calendar ()
+  "Return string of date chosen from Calendar."
+  (cond ((and (stringp todos-date-from-calendar)
+             (string-match todos-date-pattern todos-date-from-calendar))
+        todos-date-from-calendar)
+       (todos-date-from-calendar
+        (let (calendar-view-diary-initially-flag)
+          (calendar))                  ; *Calendar* is now current buffer.
+        (define-key calendar-mode-map [remap newline] 'exit-recursive-edit)
+        ;; If user exits Calendar before choosing a date, clean up properly.
+        (define-key calendar-mode-map
+          [remap calendar-exit] (lambda ()
+                                   (interactive)
+                                   (progn
+                                     (calendar-exit)
+                                     (exit-recursive-edit))))
+        (message "Put cursor on a date and type <return> to set it.")
+        (recursive-edit)
+        (unwind-protect
+            (when (equal (buffer-name) calendar-buffer)
+              (setq todos-date-from-calendar
+                    (calendar-date-string (calendar-cursor-to-date t) t t))
+              (calendar-exit)
+              todos-date-from-calendar)
+          (define-key calendar-mode-map [remap newline] nil)
+          (define-key calendar-mode-map [remap calendar-exit] nil)
+          (unless (zerop (recursion-depth)) (exit-recursive-edit))
+          (when (stringp todos-date-from-calendar)
+            todos-date-from-calendar)))))
 
-(defface todos-diary-expired
-  ;; Doesn't contrast enough with todos-date (= diary) face.
-  ;; ;; '((t :inherit warning))
-  ;; '((default :weight bold)
-  ;;   (((class color) (min-colors 16)) :foreground "DarkOrange")
-  ;;   (((class color)) :foreground "yellow"))
-  ;; bold font-lock-function-name-face
-  '((default :weight bold)
-    (((class color) (min-colors 88) (background light)) :foreground "Blue1")
-    (((class color) (min-colors 88) (background dark))  :foreground "LightSkyBlue")
-    (((class color) (min-colors 16) (background light)) :foreground "Blue")
-    (((class color) (min-colors 16) (background dark))  :foreground "LightSkyBlue")
-    (((class color) (min-colors 8)) :foreground "blue")
-    (t :inverse-video t))
-  "Face for expired dates of diary items."
-  :group 'todos-faces)
+(defun todos-insert-item-from-calendar (&optional arg)
+  "Prompt for and insert a new item with date selected from calendar.
+Invoked without prefix argument ARG, insert the item into the
+current category, without one prefix argument, prompt for the
+category from the current todo file or from one listed in
+`todos-category-completions-files'; with two prefix arguments,
+prompt for a todo file and then for a category in it."
+  (interactive "P")
+  (setq todos-date-from-calendar
+       (calendar-date-string (calendar-cursor-to-date t) t t))
+  (calendar-exit)
+  (todos-basic-insert-item arg nil nil todos-date-from-calendar))
 
-(defface todos-date
-  '((t :inherit diary))
-  "Face for the date string of a Todos item."
-  :group 'todos-faces)
+(define-key calendar-mode-map "it" 'todos-insert-item-from-calendar)
 
-(defface todos-time
-  '((t :inherit diary-time))
-  "Face for the time string of a Todos item."
-  :group 'todos-faces)
+(defun todos-copy-item ()
+  "Copy item at point and insert the copy as a new item."
+  (interactive)
+  (unless (or (todos-done-item-p) (looking-at "^$"))
+    (let ((copy (todos-item-string))
+         (diary-item (todos-diary-item-p)))
+      (todos-set-item-priority copy (todos-current-category) t)
+      (todos-update-count 'todo 1)
+      (when diary-item (todos-update-count 'diary 1))
+      (todos-update-categories-sexp))))
 
-(defface todos-nondiary
-  ;; '((t :inherit font-lock-type-face))
-  '((((class grayscale) (background light)) :foreground "Gray90" :weight bold)
-    (((class grayscale) (background dark))  :foreground "DimGray" :weight bold)
-    (((class color) (min-colors 88) (background light)) :foreground "ForestGreen")
-    (((class color) (min-colors 88) (background dark))  :foreground "PaleGreen")
-    (((class color) (min-colors 16) (background light)) :foreground "ForestGreen")
-    (((class color) (min-colors 16) (background dark))  :foreground "PaleGreen")
-    (((class color) (min-colors 8)) :foreground "green")
-    (t :weight bold :underline t))
-  "Face for non-diary markers around todo item date/time header."
-  :group 'todos-faces)
+(defun todos-delete-item ()
+  "Delete at least one item in this category.
+If there are marked items, delete all of these; otherwise, delete
+the item at point."
+  (interactive)
+  (let (ov)
+    (unwind-protect
+       (let* ((cat (todos-current-category))
+              (marked (assoc cat todos-categories-with-marks))
+              (item (unless marked (todos-item-string)))
+              (answer (if marked
+                          (todos-y-or-n-p
+                           "Permanently delete all marked items? ")
+                        (when item
+                          (setq ov (make-overlay
+                                    (save-excursion (todos-item-start))
+                                    (save-excursion (todos-item-end))))
+                          (overlay-put ov 'face 'todos-search)
+                          (todos-y-or-n-p "Permanently delete this item? "))))
+              buffer-read-only)
+         (when answer
+           (and marked (goto-char (point-min)))
+           (catch 'done
+             (while (not (eobp))
+               (if (or (and marked (todos-marked-item-p)) item)
+                   (progn
+                     (if (todos-done-item-p)
+                         (todos-update-count 'done -1)
+                       (todos-update-count 'todo -1 cat)
+                       (and (todos-diary-item-p)
+                            (todos-update-count 'diary -1)))
+                     (if ov (delete-overlay ov))
+                     (todos-remove-item)
+                     ;; Don't leave point below last item.
+                     (and item (bolp) (eolp) (< (point-min) (point-max))
+                          (todos-backward-item))
+                     (when item
+                       (throw 'done (setq item nil))))
+                 (todos-forward-item))))
+           (when marked
+             (setq todos-categories-with-marks
+                   (assq-delete-all cat todos-categories-with-marks)))
+           (todos-update-categories-sexp)
+           (todos-prefix-overlays)))
+      (if ov (delete-overlay ov)))))
 
-(defface todos-category-string
-    ;; '((t :inherit font-lock-type-face))
-  '((((class grayscale) (background light)) :foreground "Gray90" :weight bold)
-    (((class grayscale) (background dark))  :foreground "DimGray" :weight bold)
-    (((class color) (min-colors 88) (background light)) :foreground "ForestGreen")
-    (((class color) (min-colors 88) (background dark))  :foreground "PaleGreen")
-    (((class color) (min-colors 16) (background light)) :foreground "ForestGreen")
-    (((class color) (min-colors 16) (background dark))  :foreground "PaleGreen")
-    (((class color) (min-colors 8)) :foreground "green")
-    (t :weight bold :underline t))
-  "Face for category-file header in Todos Filtered Items mode."
-  :group 'todos-faces)
+(defun todos-edit-item (&optional arg)
+  "Edit the Todo item at point.
+With non-nil prefix argument ARG, include the item's date/time
+header, making it also editable; otherwise, include only the item
+content.
 
-(defface todos-done
-  ;; '((t :inherit font-lock-keyword-face))
-  '((((class grayscale) (background light)) :foreground "LightGray" :weight bold)
-    (((class grayscale) (background dark))  :foreground "DimGray" :weight bold)
-    (((class color) (min-colors 88) (background light)) :foreground "Purple")
-    (((class color) (min-colors 88) (background dark))  :foreground "Cyan1")
-    (((class color) (min-colors 16) (background light)) :foreground "Purple")
-    (((class color) (min-colors 16) (background dark))  :foreground "Cyan")
-    (((class color) (min-colors 8)) :foreground "cyan" :weight bold)
-    (t :weight bold))
-  "Face for done Todos item header string."
-  :group 'todos-faces)
+If the item consists of only one logical line, edit it in the
+minibuffer; otherwise, edit it in Todos Edit mode."
+  (interactive "P")
+  (when (todos-item-string)
+    (let* ((opoint (point))
+          (start (todos-item-start))
+          (item-beg (progn
+                      (re-search-forward
+                       (concat todos-date-string-start todos-date-pattern
+                               "\\( " diary-time-regexp "\\)?"
+                               (regexp-quote todos-nondiary-end) "?")
+                       (line-end-position) t)
+                      (1+ (- (point) start))))
+          (header (substring (todos-item-string) 0 item-beg))
+          (item (if arg (todos-item-string)
+                  (substring (todos-item-string) item-beg)))
+          (multiline (> (length (split-string item "\n")) 1))
+          (buffer-read-only nil))
+      (if multiline
+         (todos-edit-multiline-item)
+       (let ((new (concat (if arg "" header)
+                          (read-string "Edit: " (if arg
+                                                    (cons item item-beg)
+                                                  (cons item 0))))))
+         (when arg
+           (while (not (string-match (concat todos-date-string-start
+                                             todos-date-pattern) new))
+             (setq new (read-from-minibuffer
+                        "Item must start with a date: " new))))
+         ;; Ensure lines following hard newlines are indented.
+         (setq new (replace-regexp-in-string "\\(\n\\)[^[:blank:]]"
+                                             "\n\t" new nil nil 1))
+         ;; If user moved point during editing, make sure it moves back.
+         (goto-char opoint)
+         (todos-remove-item)
+         (todos-insert-with-overlays new)
+         (move-to-column item-beg))))))
 
-(defface todos-comment
-  ;; '((t :inherit font-lock-comment-face))
-  '((((class grayscale) (background light))
-     :foreground "DimGray" :weight bold :slant italic)
-    (((class grayscale) (background dark))
-     :foreground "LightGray" :weight bold :slant italic)
-    (((class color) (min-colors 88) (background light))
-     :foreground "Firebrick")
-    (((class color) (min-colors 88) (background dark))
-     :foreground "chocolate1")
-    (((class color) (min-colors 16) (background light))
-     :foreground "red")
-    (((class color) (min-colors 16) (background dark))
-     :foreground "red1")
-    (((class color) (min-colors 8) (background light))
-     :foreground "red")
-    (((class color) (min-colors 8) (background dark))
-     :foreground "yellow")
-    (t :weight bold :slant italic))
-  "Face for comments appended to done Todos items."
-  :group 'todos-faces)
+(defun todos-edit-multiline-item ()
+  "Edit current Todo item in Todos Edit mode.
+Use of newlines invokes `todos-indent' to insure compliance with
+the format of Diary entries."
+  (interactive)
+  (when (todos-item-string)
+    (let ((buf todos-edit-buffer))
+      (set-window-buffer (selected-window)
+                        (set-buffer (make-indirect-buffer (buffer-name) buf)))
+      (narrow-to-region (todos-item-start) (todos-item-end))
+      (todos-edit-mode)
+      (message "%s" (substitute-command-keys
+                    (concat "Type \\[todos-edit-quit] "
+                            "to return to Todos mode.\n"))))))
 
-(defface todos-done-sep
-  ;; '((t :inherit font-lock-builtin-face))
-  '((((class grayscale) (background light)) :foreground "LightGray" :weight bold)
-    (((class grayscale) (background dark))  :foreground "DimGray" :weight bold)
-    (((class color) (min-colors 88) (background light)) :foreground "dark slate blue")
-    (((class color) (min-colors 88) (background dark))  :foreground "LightSteelBlue")
-    (((class color) (min-colors 16) (background light)) :foreground "Orchid")
-    (((class color) (min-colors 16) (background dark)) :foreground "LightSteelBlue")
-    (((class color) (min-colors 8)) :foreground "blue" :weight bold)
-    (t :weight bold))
-  "Face for separator string bewteen done and not done Todos items."
-  :group 'todos-faces)
+(defun todos-edit-quit ()
+  "Return from Todos Edit mode to Todos mode.
+If the item contains hard line breaks, make sure the following
+lines are indented by `todos-indent-to-here' to conform to diary
+format.
+
+If the whole file was in Todos Edit mode, check before returning
+whether the file is still a valid Todos file and if so, also
+recalculate the Todos categories sexp, in case changes were made
+in the number or names of categories."
+  (interactive)
+  (if (> (buffer-size) (- (point-max) (point-min)))
+      ;; We got here via `e m'.
+      (let ((item (buffer-string))
+           (regex "\\(\n\\)[^[:blank:]]")
+           (buf (buffer-base-buffer)))
+       (while (not (string-match (concat todos-date-string-start
+                                         todos-date-pattern) item))
+         (setq item (read-from-minibuffer
+                    "Item must start with a date: " item)))
+       ;; Ensure lines following hard newlines are indented.
+       (when (string-match regex (buffer-string))
+         (setq item (replace-regexp-in-string regex "\n\t" item nil nil 1))
+         (delete-region (point-min) (point-max))
+         (insert item))
+       (kill-buffer)
+       (unless (eq (current-buffer) buf)
+         (set-window-buffer (selected-window) (set-buffer buf))))
+    ;; We got here via `F e'.
+    (when (todos-check-format)
+      ;; FIXME: separate out sexp check?
+      ;; If manual editing makes e.g. item counts change, have to
+      ;; call this to update todos-categories, but it restores
+      ;; category order to list order.
+      ;; (todos-repair-categories-sexp)
+      ;; Compare (todos-make-categories-list t) with sexp and if
+      ;; different ask (todos-update-categories-sexp) ?
+      (todos-mode)
+      (let* ((cat-beg (concat "^" (regexp-quote todos-category-beg)
+                             "\\(.*\\)$"))
+            (curline (buffer-substring-no-properties
+                      (line-beginning-position) (line-end-position)))
+            (cat (cond ((string-match cat-beg curline)
+                        (match-string-no-properties 1 curline))
+                       ((or (re-search-backward cat-beg nil t)
+                            (re-search-forward cat-beg nil t))
+                        (match-string-no-properties 1)))))
+       (todos-category-number cat)
+       (todos-category-select)
+       (goto-char (point-min))))))
 
-;; -----------------------------------------------------------------------------
-;;; Todos Categories mode options
-;; -----------------------------------------------------------------------------
+(defun todos-basic-edit-item-header (what &optional inc)
+  "Function underlying commands to edit item date/time header.
 
-(defcustom todos-categories-category-label "Category"
-  "Category button label in Todos Categories mode."
-  :type 'string
-  :group 'todos-categories)
+The argument WHAT (passed by invoking commands) specifies what
+part of the header to edit; possible values are these symbols:
+`date', to edit the year, month, and day of the date string;
+`time', to edit just the time string; `calendar', to select the
+date from the Calendar; `today', to set the date to today's date;
+`dayname', to set the date string to the name of a day or to
+change the day name; and `year', `month' or `day', to edit only
+these respective parts of the date string (`day' is the number of
+the given day of the month, and `month' is either the name of the
+given month or its number, depending on the value of
+`calendar-date-display-form').
 
-(defcustom todos-categories-todo-label "Todo"
-  "Todo button label in Todos Categories mode."
-  :type 'string
-  :group 'todos-categories)
+The optional argument INC is a positive or negative integer
+\(passed by invoking commands as a numerical prefix argument)
+that in conjunction with the WHAT values `year', `month' or
+`day', increments or decrements the specified date string
+component by the specified number of suitable units, i.e., years,
+months, or days, with automatic adjustment of the other date
+string components as necessary.
 
-(defcustom todos-categories-diary-label "Diary"
-  "Diary button label in Todos Categories mode."
-  :type 'string
-  :group 'todos-categories)
+If there are marked items, apply the same edit to all of these;
+otherwise, edit just the item at point."
+  (let* ((cat (todos-current-category))
+        (marked (assoc cat todos-categories-with-marks))
+        (first t)
+        (todos-date-from-calendar t)
+        (buffer-read-only nil)
+        ndate ntime year monthname month day
+        dayname)       ; Needed by calendar-date-display-form.
+    (save-excursion
+      (or (and marked (goto-char (point-min))) (todos-item-start))
+      (catch 'end
+       (while (not (eobp))
+         (and marked
+              (while (not (todos-marked-item-p))
+                (todos-forward-item)
+                (and (eobp) (throw 'end nil))))
+         (re-search-forward (concat todos-date-string-start "\\(?1:"
+                                    todos-date-pattern
+                                    "\\)\\(?2: " diary-time-regexp "\\)?"
+                                    (regexp-quote todos-nondiary-end) "?")
+                            (line-end-position) t)
+         (let* ((odate (match-string-no-properties 1))
+                (otime (match-string-no-properties 2))
+                (omonthname (match-string-no-properties 6))
+                (omonth (match-string-no-properties 7))
+                (oday (match-string-no-properties 8))
+                (oyear (match-string-no-properties 9))
+                (tmn-array todos-month-name-array)
+                (mlist (append tmn-array nil))
+                (tma-array todos-month-abbrev-array)
+                (mablist (append tma-array nil))
+                (yy (and oyear (unless (string= oyear "*")
+                                 (string-to-number oyear))))
+                (mm (or (and omonth (unless (string= omonth "*")
+                                      (string-to-number omonth)))
+                        (1+ (- (length mlist)
+                               (length (or (member omonthname mlist)
+                                           (member omonthname mablist)))))))
+                (dd (and oday (unless (string= oday "*")
+                                (string-to-number oday)))))
+           ;; If there are marked items, use only the first to set
+           ;; header changes, and apply these to all marked items.
+           (when first
+             (cond
+              ((eq what 'date)
+               (setq ndate (todos-read-date)))
+              ((eq what 'calendar)
+               (setq ndate (save-match-data (todos-set-date-from-calendar))))
+              ((eq what 'today)
+               (setq ndate (calendar-date-string (calendar-current-date) t t)))
+              ((eq what 'dayname)
+               (setq ndate (todos-read-dayname)))
+              ((eq what 'time)
+               (setq ntime (save-match-data (todos-read-time)))
+               (when (> (length ntime) 0)
+                 (setq ntime (concat " " ntime))))
+              ;; When date string consists only of a day name,
+              ;; passing other date components is a NOP.
+              ((and (memq what '(year month day))
+                    (not (or oyear omonth oday))))
+              ((eq what 'year)
+               (setq day oday
+                     monthname omonthname
+                     month omonth
+                     year (cond ((not current-prefix-arg)
+                                 (todos-read-date 'year))
+                                ((string= oyear "*")
+                                 (user-error "Cannot increment *"))
+                                (t
+                                 (number-to-string (+ yy inc))))))
+              ((eq what 'month)
+               (setf day oday
+                     year oyear
+                     (if (memq 'month calendar-date-display-form)
+                         month
+                       monthname)
+                     (cond ((not current-prefix-arg)
+                            (todos-read-date 'month))
+                           ((or (string= omonth "*") (= mm 13))
+                            (user-error "Cannot increment *"))
+                           (t
+                            (let ((mminc (+ mm inc)))
+                              ;; Increment or decrement month by INC
+                              ;; modulo 12.
+                              (setq mm (% mminc 12))
+                              ;; If result is 0, make month December.
+                              (setq mm (if (= mm 0) 12 (abs mm)))
+                              ;; Adjust year if necessary.
+                              (setq year (or (and (cond ((> mminc 12)
+                                                         (+ yy (/ mminc 12)))
+                                                        ((< mminc 1)
+                                                         (- yy (/ mminc 12) 1))
+                                                        (t yy))
+                                                  (number-to-string yy))
+                                             oyear)))
+                            ;; Return the changed numerical month as
+                            ;; a string or the corresponding month name.
+                            (if omonth
+                                (number-to-string mm)
+                              (aref tma-array (1- mm))))))
+               (let ((yy (string-to-number year)) ; 0 if year is "*".
+                     ;; When mm is 13 (corresponding to "*" as value
+                     ;; of month), this raises an args-out-of-range
+                     ;; error in calendar-last-day-of-month, so use 1
+                     ;; (corresponding to January) to get 31 days.
+                     (mm (if (= mm 13) 1 mm)))
+                 (if (> (string-to-number day)
+                        (calendar-last-day-of-month mm yy))
+                     (user-error "%s %s does not have %s days"
+                            (aref tmn-array (1- mm))
+                            (if (= mm 2) yy "") day))))
+              ((eq what 'day)
+               (setq year oyear
+                     month omonth
+                     monthname omonthname
+                     day (cond
+                          ((not current-prefix-arg)
+                           (todos-read-date 'day mm oyear))
+                          ((string= oday "*")
+                           (user-error "Cannot increment *"))
+                          ((or (string= omonth "*") (string= omonthname "*"))
+                           (setq dd (+ dd inc))
+                           (if (> dd 31)
+                               (user-error "A month cannot have more than 31 days")
+                             (number-to-string dd)))
+                          ;; Increment or decrement day by INC,
+                          ;; adjusting month and year if necessary
+                          ;; (if year is "*" assume current year to
+                          ;; calculate adjustment).
+                          (t
+                           (let* ((yy (or yy (calendar-extract-year
+                                              (calendar-current-date))))
+                                  (date (calendar-gregorian-from-absolute
+                                         (+ (calendar-absolute-from-gregorian
+                                             (list mm dd yy)) inc)))
+                                  (adjmm (nth 0 date)))
+                             ;; Set year and month(name) to adjusted values.
+                             (unless (string= year "*")
+                               (setq year (number-to-string (nth 2 date))))
+                             (if month
+                                 (setq month (number-to-string adjmm))
+                               (setq monthname (aref tma-array (1- adjmm))))
+                             ;; Return changed numerical day as a string.
+                             (number-to-string (nth 1 date)))))))))
+           ;; If new year, month or day date string components were
+           ;; calculated, rebuild the whole date string from them.
+           (when (memq what '(year month day))
+             (if (or oyear omonth omonthname oday)
+                 (setq ndate (mapconcat 'eval calendar-date-display-form ""))
+               (message "Cannot edit date component of empty date string")))
+           (when ndate (replace-match ndate nil nil nil 1))
+           ;; Add new time string to the header, if it was supplied.
+           (when ntime
+             (if otime
+                 (replace-match ntime nil nil nil 2)
+               (goto-char (match-end 1))
+               (insert ntime)))
+           (setq todos-date-from-calendar nil)
+           (setq first nil))
+         ;; Apply the changes to the first marked item header to the
+         ;; remaining marked items.  If there are no marked items,
+         ;; we're finished.
+         (if marked
+             (todos-forward-item)
+           (goto-char (point-max))))))))
 
-(defcustom todos-categories-done-label "Done"
-  "Done button label in Todos Categories mode."
-  :type 'string
-  :group 'todos-categories)
+(defun todos-edit-item-header ()
+  "Interactively edit at least the date of item's date/time header.
+If user option `todos-always-add-time-string' is non-nil, also
+edit item's time string."
+  (interactive)
+  (todos-basic-edit-item-header 'date)
+  (when todos-always-add-time-string
+    (todos-edit-item-time)))
 
-(defcustom todos-categories-archived-label "Archived"
-  "Archived button label in Todos Categories mode."
-  :type 'string
-  :group 'todos-categories)
+(defun todos-edit-item-time ()
+  "Interactively edit the time string of item's date/time header."
+  (interactive)
+  (todos-basic-edit-item-header 'time))
 
-(defcustom todos-categories-totals-label "Totals"
-  "String to label total item counts in Todos Categories mode."
-  :type 'string
-  :group 'todos-categories)
+(defun todos-edit-item-date-from-calendar ()
+  "Interactively edit item's date using the Calendar."
+  (interactive)
+  (todos-basic-edit-item-header 'calendar))
 
-(defcustom todos-categories-number-separator " | "
-  "String between number and category in Todos Categories mode.
-This separates the number from the category name in the default
-categories display according to priority."
-  :type 'string
-  :group 'todos-categories)
+(defun todos-edit-item-date-to-today ()
+  "Set item's date to today's date."
+  (interactive)
+  (todos-basic-edit-item-header 'today))
 
-(defcustom todos-categories-align 'center
-  "Alignment of category names in Todos Categories mode."
-  :type '(radio (const left) (const center) (const right))
-  :group 'todos-categories)
+(defun todos-edit-item-date-day-name ()
+  "Replace item's date with the name of a day of the week."
+  (interactive)
+  (todos-basic-edit-item-header 'dayname))
 
-;; -----------------------------------------------------------------------------
-;;; Entering and using Todos Categories mode
-;; -----------------------------------------------------------------------------
+(defun todos-edit-item-date-year (&optional inc)
+  "Interactively edit the year of item's date string.
+With prefix argument INC a positive or negative integer,
+increment or decrement the year by INC."
+  (interactive "p")
+  (todos-basic-edit-item-header 'year inc))
 
-(defun todos-show-categories-table ()
-  "Display a table of the current file's categories and item counts.
+(defun todos-edit-item-date-month (&optional inc)
+  "Interactively edit the month of item's date string.
+With prefix argument INC a positive or negative integer,
+increment or decrement the month by INC."
+  (interactive "p")
+  (todos-basic-edit-item-header 'month inc))
 
-In the initial display the categories are numbered, indicating
-their current order for navigating by \\[todos-forward-category]
-and \\[todos-backward-category].  You can persistantly change the
-order of the category at point by typing
-\\[todos-set-category-number], \\[todos-raise-category] or
-\\[todos-lower-category].
+(defun todos-edit-item-date-day (&optional inc)
+  "Interactively edit the day of the month of item's date string.
+With prefix argument INC a positive or negative integer,
+increment or decrement the day by INC."
+  (interactive "p")
+  (todos-basic-edit-item-header 'day inc))
 
-The labels above the category names and item counts are buttons,
-and clicking these changes the display: sorted by category name
-or by the respective item counts (alternately descending or
-ascending).  In these displays the categories are not numbered
-and \\[todos-set-category-number], \\[todos-raise-category] and
-\\[todos-lower-category] are disabled.  (Programmatically, the
-sorting is triggered by passing a non-nil SORTKEY argument.)
+(defun todos-edit-item-diary-inclusion ()
+  "Change diary status of one or more todo items in this category.
+That is, insert `todos-nondiary-marker' if the candidate items
+lack this marking; otherwise, remove it.
 
-In addition, the lines with the category names and item counts
-are buttonized, and pressing one of these button jumps to the
-category in Todos mode (or Todos Archive mode, for categories
-containing only archived items, provided user option
-`todos-skip-archived-categories' is non-nil.  These categories
-are shown in `todos-archived-only' face."
+If there are marked todo items, change the diary status of all
+and only these, otherwise change the diary status of the item at
+point."
   (interactive)
-  (todos-display-categories)
-  (let (sortkey)
-    (todos-update-categories-display sortkey)))
+  (let ((buffer-read-only)
+       (marked (assoc (todos-current-category)
+                      todos-categories-with-marks)))
+    (catch 'stop
+      (save-excursion
+       (when marked (goto-char (point-min)))
+       (while (not (eobp))
+         (if (todos-done-item-p)
+             (throw 'stop (message "Done items cannot be edited"))
+           (unless (and marked (not (todos-marked-item-p)))
+             (let* ((beg (todos-item-start))
+                    (lim (save-excursion (todos-item-end)))
+                    (end (save-excursion
+                           (or (todos-time-string-matcher lim)
+                               (todos-date-string-matcher lim)))))
+               (if (looking-at (regexp-quote todos-nondiary-start))
+                   (progn
+                     (replace-match "")
+                     (search-forward todos-nondiary-end (1+ end) t)
+                     (replace-match "")
+                     (todos-update-count 'diary 1))
+                 (when end
+                   (insert todos-nondiary-start)
+                   (goto-char (1+ end))
+                   (insert todos-nondiary-end)
+                   (todos-update-count 'diary -1)))))
+           (unless marked (throw 'stop nil))
+           (todos-forward-item)))))
+    (todos-update-categories-sexp)))
 
-(defun todos-sort-categories-alphabetically-or-numerically ()
-  "Sort table of categories alphabetically or numerically."
-  (interactive)
+(defun todos-edit-category-diary-inclusion (arg)
+  "Make all items in this category diary items.
+With prefix ARG, make all items in this category non-diary
+items."
+  (interactive "P")
   (save-excursion
     (goto-char (point-min))
-    (forward-line 2)
-    (if (member 'alpha todos-descending-counts)
-       (progn
-         (todos-update-categories-display nil)
-         (setq todos-descending-counts
-               (delete 'alpha todos-descending-counts)))
-      (todos-update-categories-display 'alpha))))
+    (let ((todo-count (todos-get-count 'todo))
+         (diary-count (todos-get-count 'diary))
+         (buffer-read-only))
+      (catch 'stop
+       (while (not (eobp))
+         (if (todos-done-item-p)       ; We've gone too far.
+             (throw 'stop nil)
+           (let* ((beg (todos-item-start))
+                  (lim (save-excursion (todos-item-end)))
+                  (end (save-excursion
+                         (or (todos-time-string-matcher lim)
+                             (todos-date-string-matcher lim)))))
+             (if arg
+                 (unless (looking-at (regexp-quote todos-nondiary-start))
+                   (insert todos-nondiary-start)
+                   (goto-char (1+ end))
+                   (insert todos-nondiary-end))
+               (when (looking-at (regexp-quote todos-nondiary-start))
+                 (replace-match "")
+                 (search-forward todos-nondiary-end (1+ end) t)
+                 (replace-match "")))))
+         (todos-forward-item))
+       (unless (if arg (zerop diary-count) (= diary-count todo-count))
+         (todos-update-count 'diary (if arg
+                                     (- diary-count)
+                                   (- todo-count diary-count))))
+       (todos-update-categories-sexp)))))
 
-(defun todos-sort-categories-by-todo ()
-  "Sort table of categories by number of todo items."
+(defun todos-edit-item-diary-nonmarking ()
+  "Change non-marking of one or more diary items in this category.
+That is, insert `diary-nonmarking-symbol' if the candidate items
+lack this marking; otherwise, remove it.
+
+If there are marked todo items, change the non-marking status of
+all and only these, otherwise change the non-marking status of
+the item at point."
   (interactive)
+  (let ((buffer-read-only)
+       (marked (assoc (todos-current-category)
+                      todos-categories-with-marks)))
+    (catch 'stop
+      (save-excursion
+       (when marked (goto-char (point-min)))
+       (while (not (eobp))
+         (if (todos-done-item-p)
+             (throw 'stop (message "Done items cannot be edited"))
+           (unless (and marked (not (todos-marked-item-p)))
+             (todos-item-start)
+             (unless (looking-at (regexp-quote todos-nondiary-start))
+               (if (looking-at (regexp-quote diary-nonmarking-symbol))
+                   (replace-match "")
+                 (insert diary-nonmarking-symbol))))
+           (unless marked (throw 'stop nil))
+           (todos-forward-item)))))))
+
+(defun todos-edit-category-diary-nonmarking (arg)
+  "Add `diary-nonmarking-symbol' to all diary items in this category.
+With prefix ARG, remove `diary-nonmarking-symbol' from all diary
+items in this category."
+  (interactive "P")
   (save-excursion
     (goto-char (point-min))
-    (forward-line 2)
-    (todos-update-categories-display 'todo)))
+    (let (buffer-read-only)
+      (catch 'stop
+      (while (not (eobp))
+       (if (todos-done-item-p)         ; We've gone too far.
+           (throw 'stop nil)
+         (unless (looking-at (regexp-quote todos-nondiary-start))
+           (if arg
+               (when (looking-at (regexp-quote diary-nonmarking-symbol))
+                 (replace-match ""))
+             (unless (looking-at (regexp-quote diary-nonmarking-symbol))
+               (insert diary-nonmarking-symbol))))
+       (todos-forward-item)))))))
+
+(defun todos-set-item-priority (&optional item cat new arg)
+  "Prompt for and set ITEM's priority in CATegory.
+
+Interactively, ITEM is the todo item at point, CAT is the current
+category, and the priority is a number between 1 and the number
+of items in the category.  Non-interactively, non-nil NEW means
+ITEM is a new item and the lowest priority is one more than the
+number of items in CAT.
+
+The new priority is set either interactively by prompt or by a
+numerical prefix argument, or noninteractively by argument ARG,
+whose value can be either of the symbols `raise' or `lower',
+meaning to raise or lower the item's priority by one."
+  (interactive)
+  (unless (and (called-interactively-p 'any)
+              (or (todos-done-item-p) (looking-at "^$")))
+    (let* ((item (or item (todos-item-string)))
+          (marked (todos-marked-item-p))
+          (cat (or cat (cond ((eq major-mode 'todos-mode)
+                              (todos-current-category))
+                             ((eq major-mode 'todos-filtered-items-mode)
+                              (let* ((regexp1
+                                      (concat todos-date-string-start
+                                              todos-date-pattern
+                                              "\\( " diary-time-regexp "\\)?"
+                                              (regexp-quote todos-nondiary-end)
+                                              "?\\(?1: \\[\\(.+:\\)?.+\\]\\)")))
+                                (save-excursion
+                                  (re-search-forward regexp1 nil t)
+                                  (match-string-no-properties 1)))))))
+          curnum
+          (todo (cond ((or (eq arg 'raise) (eq arg 'lower)
+                           (eq major-mode 'todos-filtered-items-mode))
+                       (save-excursion
+                         (let ((curstart (todos-item-start))
+                               (count 0))
+                           (goto-char (point-min))
+                           (while (looking-at todos-item-start)
+                             (setq count (1+ count))
+                             (when (= (point) curstart) (setq curnum count))
+                             (todos-forward-item))
+                           count)))
+                      ((eq major-mode 'todos-mode)
+                       (todos-get-count 'todo cat))))
+          (maxnum (if new (1+ todo) todo))
+          (prompt (format "Set item priority (1-%d): " maxnum))
+          (priority (cond ((and (not arg) (numberp current-prefix-arg))
+                           current-prefix-arg)
+                          ((and (eq arg 'raise) (>= curnum 1))
+                           (1- curnum))
+                          ((and (eq arg 'lower) (<= curnum maxnum))
+                           (1+ curnum))))
+          candidate
+          buffer-read-only)
+      (unless (and priority
+                  (or (and (eq arg 'raise) (zerop priority))
+                      (and (eq arg 'lower) (> priority maxnum))))
+       ;; When moving item to another category, show the category before
+       ;; prompting for its priority.
+       (unless (or arg (called-interactively-p 'any))
+         (todos-category-number cat)
+         ;; If done items in category are visible, keep them visible.
+         (let ((done todos-show-with-done))
+           (when (> (buffer-size) (- (point-max) (point-min)))
+             (save-excursion
+               (goto-char (point-min))
+               (setq done (re-search-forward todos-done-string-start nil t))))
+           (let ((todos-show-with-done done))
+             (todos-category-select)
+             ;; Keep top of category in view while setting priority.
+             (goto-char (point-min)))))
+       ;; Prompt for priority only when the category has at least one
+       ;; todo item.
+       (when (> maxnum 1)
+         (while (not priority)
+           (setq candidate (read-number prompt))
+           (setq prompt (when (or (< candidate 1) (> candidate maxnum))
+                          (format "Priority must be an integer between 1 and %d.\n"
+                                  maxnum)))
+           (unless prompt (setq priority candidate))))
+       ;; In Top Priorities buffer, an item's priority can be changed
+       ;; wrt items in another category, but not wrt items in the same
+       ;; category.
+       (when (eq major-mode 'todos-filtered-items-mode)
+         (let* ((regexp2 (concat todos-date-string-start todos-date-pattern
+                                 "\\( " diary-time-regexp "\\)?"
+                                 (regexp-quote todos-nondiary-end)
+                                 "?\\(?1:" (regexp-quote cat) "\\)"))
+                (end (cond ((< curnum priority)
+                            (save-excursion (todos-item-end)))
+                           ((> curnum priority)
+                            (save-excursion (todos-item-start)))))
+                (match (save-excursion
+                         (cond ((< curnum priority)
+                                (todos-forward-item (1+ (- priority curnum)))
+                                (when (re-search-backward regexp2 end t)
+                                  (match-string-no-properties 1)))
+                               ((> curnum priority)
+                                (todos-backward-item (- curnum priority))
+                                (when (re-search-forward regexp2 end t)
+                                  (match-string-no-properties 1)))))))
+           (when match
+             (user-error (concat "Cannot reprioritize items from the same "
+                            "category in this mode, only in Todos mode")))))
+       ;; Interactively or with non-nil ARG, relocate the item within its
+       ;; category.
+       (when (or arg (called-interactively-p 'any))
+         (todos-remove-item))
+       (goto-char (point-min))
+       (when priority
+         (unless (= priority 1)
+           (todos-forward-item (1- priority))
+           ;; When called from todos-item-undone and the highest priority
+           ;; is chosen, this advances point to the first done item, so
+           ;; move it up to the empty line above the done items
+           ;; separator.
+           (when (looking-back (concat "^"
+                                       (regexp-quote todos-category-done)
+                                       "\n"))
+             (todos-backward-item))))
+       (todos-insert-with-overlays item)
+       ;; If item was marked, restore the mark.
+       (and marked
+            (let* ((ov (todos-get-overlay 'prefix))
+                   (pref (overlay-get ov 'before-string)))
+              (overlay-put ov 'before-string
+                           (concat todos-item-mark pref))))))))
 
-(defun todos-sort-categories-by-diary ()
-  "Sort table of categories by number of diary items."
+(defun todos-raise-item-priority ()
+  "Raise priority of current item by moving it up by one item."
   (interactive)
-  (save-excursion
-    (goto-char (point-min))
-    (forward-line 2)
-    (todos-update-categories-display 'diary)))
+  (todos-set-item-priority nil nil nil 'raise))
 
-(defun todos-sort-categories-by-done ()
-  "Sort table of categories by number of non-archived done items."
+(defun todos-lower-item-priority ()
+  "Lower priority of current item by moving it down by one item."
   (interactive)
-  (save-excursion
-    (goto-char (point-min))
-    (forward-line 2)
-    (todos-update-categories-display 'done)))
+  (todos-set-item-priority nil nil nil 'lower))
 
-(defun todos-sort-categories-by-archived ()
-  "Sort table of categories by number of archived items."
-  (interactive)
-  (save-excursion
-    (goto-char (point-min))
-    (forward-line 2)
-    (todos-update-categories-display 'archived)))
+(defun todos-move-item (&optional file)
+  "Move at least one todo or done item to another category.
+If there are marked items, move all of these; otherwise, move
+the item at point.
 
-(defun todos-next-button (n)
-  "Move point to the Nth next button in the table of categories."
-  (interactive "p")
-  (forward-button n 'wrap 'display-message)
-  (and (bolp) (button-at (point))
-       ;; Align with beginning of category label.
-       (forward-char (+ 4 (length todos-categories-number-separator)))))
+With prefix argument FILE, prompt for a specific Todos file and
+choose (with TAB completion) a category in it to move the item or
+items to; otherwise, choose and move to any category in either
+the current Todos file or one of the files in
+`todos-category-completions-files'.  If the chosen category is
+not an existing categories, then it is created and the item(s)
+become(s) the first entry/entries in that category.
 
-(defun todos-previous-button (n)
-  "Move point to the Nth previous button in the table of categories."
-  (interactive "p")
-  (backward-button n 'wrap 'display-message)
-  (and (bolp) (button-at (point))
-       ;; Align with beginning of category label.
-       (forward-char (+ 4 (length todos-categories-number-separator)))))
+With moved Todo items, prompt to set the priority in the category
+moved to (with multiple todos items, the one that had the highest
+priority in the category moved from gets the new priority and the
+rest of the moved todo items are inserted in sequence below it).
+Moved done items are appended to the top of the done items
+section in the category moved to."
+  (interactive "P")
+  (let* ((cat1 (todos-current-category))
+        (marked (assoc cat1 todos-categories-with-marks)))
+    ;; Noop if point is not on an item and there are no marked items.
+    (unless (and (looking-at "^$")
+                (not marked))
+      (let* ((buffer-read-only)
+            (file1 todos-current-todos-file)
+            (num todos-category-number)
+            (item (todos-item-string))
+            (diary-item (todos-diary-item-p))
+            (done-item (and (todos-done-item-p) (concat item "\n")))
+            (omark (save-excursion (todos-item-start) (point-marker)))
+            (todo 0)
+            (diary 0)
+            (done 0)
+            ov cat2 file2 moved nmark todo-items done-items)
+       (unwind-protect
+           (progn
+             (unless marked
+               (setq ov (make-overlay (save-excursion (todos-item-start))
+                                      (save-excursion (todos-item-end))))
+               (overlay-put ov 'face 'todos-search))
+             (let* ((pl (if (and marked (> (cdr marked) 1)) "s" ""))
+                    (cat+file (todos-read-category (concat "Move item" pl
+                                                           " to category: ")
+                                                   nil file)))
+               (while (and (equal (car cat+file) cat1)
+                           (equal (cdr cat+file) file1))
+                 (setq cat+file (todos-read-category
+                                 "Choose a different category: ")))
+               (setq cat2 (car cat+file)
+                     file2 (cdr cat+file))))
+         (if ov (delete-overlay ov)))
+       (set-buffer (find-buffer-visiting file1))
+       (if marked
+           (progn
+             (goto-char (point-min))
+             (while (not (eobp))
+               (when (todos-marked-item-p)
+                 (if (todos-done-item-p)
+                     (setq done-items (concat done-items
+                                              (todos-item-string) "\n")
+                           done (1+ done))
+                   (setq todo-items (concat todo-items
+                                            (todos-item-string) "\n")
+                         todo (1+ todo))
+                   (when (todos-diary-item-p)
+                     (setq diary (1+ diary)))))
+               (todos-forward-item))
+             ;; Chop off last newline of multiple todo item string,
+             ;; since it will be reinserted when setting priority
+             ;; (but with done items priority is not set, so keep
+             ;; last newline).
+             (and todo-items
+                  (setq todo-items (substring todo-items 0 -1))))
+         (if (todos-done-item-p)
+             (setq done 1)
+           (setq todo 1)
+           (when (todos-diary-item-p) (setq diary 1))))
+       (set-window-buffer (selected-window)
+                          (set-buffer (find-file-noselect file2 'nowarn)))
+       (unwind-protect
+           (progn
+             (when (or todo-items (and item (not done-item)))
+               (todos-set-item-priority (or todo-items item) cat2 t))
+             ;; Move done items en bloc to top of done items section.
+             (when (or done-items done-item)
+               (todos-category-number cat2)
+               (widen)
+               (goto-char (point-min))
+               (re-search-forward
+                (concat "^" (regexp-quote (concat todos-category-beg cat2))
+                        "$") nil t)
+               (re-search-forward
+                (concat "^" (regexp-quote todos-category-done)) nil t)
+               (forward-line)
+               (insert (or done-items done-item)))
+             (setq moved t))
+         (cond
+          ;; Move succeeded, so remove item from starting category,
+          ;; update item counts and display the category containing
+          ;; the moved item.
+          (moved
+           (setq nmark (point-marker))
+           (when todo (todos-update-count 'todo todo))
+           (when diary (todos-update-count 'diary diary))
+           (when done (todos-update-count 'done done))
+           (todos-update-categories-sexp)
+           (with-current-buffer (find-buffer-visiting file1)
+             (save-excursion
+               (save-restriction
+                 (widen)
+                 (goto-char omark)
+                 (if marked
+                     (let (beg end)
+                       (setq item nil)
+                       (re-search-backward
+                        (concat "^" (regexp-quote todos-category-beg)) nil t)
+                       (forward-line)
+                       (setq beg (point))
+                       (setq end (if (re-search-forward
+                                      (concat "^" (regexp-quote
+                                                   todos-category-beg)) nil t)
+                                     (match-beginning 0)
+                                   (point-max)))
+                       (goto-char beg)
+                       (while (< (point) end)
+                         (if (todos-marked-item-p)
+                             (todos-remove-item)
+                           (todos-forward-item)))
+                       (setq todos-categories-with-marks
+                             (assq-delete-all cat1 todos-categories-with-marks)))
+                   (if ov (delete-overlay ov))
+                   (todos-remove-item))))
+             (when todo (todos-update-count 'todo (- todo) cat1))
+             (when diary (todos-update-count 'diary (- diary) cat1))
+             (when done (todos-update-count 'done (- done) cat1))
+             (todos-update-categories-sexp))
+           (set-window-buffer (selected-window)
+                              (set-buffer (find-file-noselect file2 'nowarn)))
+           (setq todos-category-number (todos-category-number cat2))
+           (let ((todos-show-with-done (or done-items done-item)))
+             (todos-category-select))
+           (goto-char nmark)
+           ;; If item is moved to end of (just first?) category, make
+           ;; sure the items above it are displayed in the window.
+           (recenter))
+          ;; User quit before setting priority of todo item(s), so
+          ;; return to starting category.
+          (t
+           (set-window-buffer (selected-window)
+                              (set-buffer (find-file-noselect file1 'nowarn)))
+           (todos-category-number cat1)
+           (todos-category-select)
+           (goto-char omark))))))))
 
-(defun todos-set-category-number (&optional arg)
-  "Change number of category at point in the table of categories.
+(defun todos-item-done (&optional arg)
+  "Tag a todo item in this category as done and relocate it.
 
-With ARG nil, prompt for the new number.  Alternatively, the
-enter the new number with numerical prefix ARG.  Otherwise, if
-ARG is either of the symbols `raise' or `lower', raise or lower
-the category line in the table by one, respectively, thereby
-decreasing or increasing its number."
+With prefix argument ARG prompt for a comment and append it to
+the done item; this is only possible if there are no marked
+items.  If there are marked items, tag all of these with
+`todos-done-string' plus the current date and, if
+`todos-always-add-time-string' is non-nil, the current time;
+otherwise, just tag the item at point.  Items tagged as done are
+relocated to the category's (by default hidden) done section.  If
+done items are visible on invoking this command, they remain
+visible."
   (interactive "P")
-  (let ((curnum (save-excursion
-                 ;; Get the number representing the priority of the category
-                 ;; on the current line.
-                 (forward-line 0) (skip-chars-forward " ") (number-at-point))))
-    (when curnum               ; Do nothing if we're not on a category line.
-      (let* ((maxnum (length todos-categories))
-            (prompt (format "Set category priority (1-%d): " maxnum))
-            (col (current-column))
+  (let* ((cat (todos-current-category))
+        (marked (assoc cat todos-categories-with-marks)))
+    (when marked
+      (save-excursion
+       (save-restriction
+         (goto-char (point-max))
+         (todos-backward-item)
+         (unless (todos-done-item-p)
+           (widen)
+           (unless (re-search-forward
+                    (concat "^" (regexp-quote todos-category-beg)) nil t)
+             (goto-char (point-max)))
+           (forward-line -1))
+         (while (todos-done-item-p)
+           (when (todos-marked-item-p)
+             (user-error "This command does not apply to done items"))
+           (todos-backward-item)))))
+    (unless (and (not marked)
+                (or (todos-done-item-p)
+                    ;; Point is between todo and done items.
+                    (looking-at "^$")))
+      (let* ((date-string (calendar-date-string (calendar-current-date) t t))
+            (time-string (if todos-always-add-time-string
+                             (concat " " (substring (current-time-string)
+                                                    11 16))
+                           ""))
+            (done-prefix (concat "[" todos-done-string date-string time-string
+                                 "] "))
+            (comment (and arg (read-string "Enter a comment: ")))
+            (item-count 0)
+            (diary-count 0)
+            (show-done (save-excursion
+                         (goto-char (point-min))
+                         (re-search-forward todos-done-string-start nil t)))
             (buffer-read-only nil)
-            (priority (cond ((and (eq arg 'raise) (> curnum 1))
-                             (1- curnum))
-                            ((and (eq arg 'lower) (< curnum maxnum))
-                             (1+ curnum))))
-            candidate)
-       (while (not priority)
-         (setq candidate (or arg (read-number prompt)))
-         (setq arg nil)
-         (setq prompt
-               (cond ((or (< candidate 1) (> candidate maxnum))
-                      (format "Priority must be an integer between 1 and %d: "
-                              maxnum))
-                     ((= candidate curnum)
-                      "Choose a different priority than the current one: ")))
-         (unless prompt (setq priority candidate)))
-       (let* ((lower (< curnum priority)) ; Priority is being lowered.
-              (head (butlast todos-categories
-                             (apply (if lower 'identity '1+)
-                                    (list (- maxnum priority)))))
-              (tail (nthcdr (apply (if lower 'identity '1-) (list priority))
-                            todos-categories))
-              ;; Category's name and items counts list.
-              (catcons (nth (1- curnum) todos-categories))
-              (todos-categories (nconc head (list catcons) tail))
-              newcats)
-         (when lower (setq todos-categories (nreverse todos-categories)))
-         (setq todos-categories (delete-dups todos-categories))
-         (when lower (setq todos-categories (nreverse todos-categories)))
-         (setq newcats todos-categories)
-         (kill-buffer)
-         (with-current-buffer (find-buffer-visiting todos-current-todos-file)
-           (setq todos-categories newcats)
-           (todos-update-categories-sexp))
-         (todos-show-categories-table)
-         (forward-line (1+ priority))
-         (forward-char col))))))
+            item done-item opoint)
+       ;; Don't add empty comment to done item.
+       (setq comment (unless (zerop (length comment))
+                       (concat " [" todos-comment-string ": " comment "]")))
+       (and marked (goto-char (point-min)))
+       (catch 'done
+         ;; Stop looping when we hit the empty line below the last
+         ;; todo item (this is eobp if only done items are hidden).
+         (while (not (looking-at "^$"))
+           (if (or (not marked) (and marked (todos-marked-item-p)))
+               (progn
+                 (setq item (todos-item-string))
+                 (setq done-item (concat done-item done-prefix item
+                                         comment (and marked "\n")))
+                 (setq item-count (1+ item-count))
+                 (when (todos-diary-item-p)
+                   (setq diary-count (1+ diary-count)))
+                 (todos-remove-item)
+                 (unless marked (throw 'done nil)))
+             (todos-forward-item))))
+       (when marked
+         ;; Chop off last newline of done item string.
+         (setq done-item (substring done-item 0 -1))
+         (setq todos-categories-with-marks
+               (assq-delete-all cat todos-categories-with-marks)))
+       (save-excursion
+         (widen)
+         (re-search-forward
+          (concat "^" (regexp-quote todos-category-done)) nil t)
+         (forward-char)
+         (when show-done (setq opoint (point)))
+         (insert done-item "\n"))
+       (todos-update-count 'todo (- item-count))
+       (todos-update-count 'done item-count)
+       (todos-update-count 'diary (- diary-count))
+       (todos-update-categories-sexp)
+       (let ((todos-show-with-done show-done))
+         (todos-category-select)
+         ;; When done items are shown, put cursor on first just done item.
+         (when opoint (goto-char opoint)))))))
 
-(defun todos-raise-category ()
-  "Raise priority of category at point in Todos Categories buffer."
-  (interactive)
-  (todos-set-category-number 'raise))
+(defun todos-done-item-add-edit-or-delete-comment (&optional arg)
+  "Add a comment to this done item or edit an existing comment.
+With prefix ARG delete an existing comment."
+  (interactive "P")
+  (when (todos-done-item-p)
+    (let ((item (todos-item-string))
+         (opoint (point))
+         (end (save-excursion (todos-item-end)))
+         comment buffer-read-only)
+      (save-excursion
+       (todos-item-start)
+       (if (re-search-forward (concat " \\["
+                                      (regexp-quote todos-comment-string)
+                                      ": \\([^]]+\\)\\]") end t)
+           (if arg
+               (when (todos-y-or-n-p "Delete comment? ")
+                 (delete-region (match-beginning 0) (match-end 0)))
+             (setq comment (read-string "Edit comment: "
+                                        (cons (match-string 1) 1)))
+             (replace-match comment nil nil nil 1))
+         (setq comment (read-string "Enter a comment: "))
+         ;; If user moved point during editing, make sure it moves back.
+         (goto-char opoint)
+         (todos-item-end)
+         (insert " [" todos-comment-string ": " comment "]"))))))
 
-(defun todos-lower-category ()
-  "Lower priority of category at point in Todos Categories buffer."
+(defun todos-item-undone ()
+  "Restore at least one done item to this category's todo section.
+Prompt for the new priority.  If there are marked items, undo all
+of these, giving the first undone item the new priority and the
+rest following directly in sequence; otherwise, undo just the
+item at point.
+
+If the done item has a comment, ask whether to omit the comment
+from the restored item.  With multiple marked done items with
+comments, only ask once, and if affirmed, omit subsequent
+comments without asking."
   (interactive)
-  (todos-set-category-number 'lower))
+  (let* ((cat (todos-current-category))
+        (marked (assoc cat todos-categories-with-marks))
+        (pl (if (and marked (> (cdr marked) 1)) "s" "")))
+    (when (or marked (todos-done-item-p))
+      (let ((buffer-read-only)
+           (opoint (point))
+           (omark (point-marker))
+           (first 'first)
+           (item-count 0)
+           (diary-count 0)
+           start end item ov npoint undone)
+       (and marked (goto-char (point-min)))
+       (catch 'done
+         (while (not (eobp))
+           (when (or (not marked) (and marked (todos-marked-item-p)))
+             (if (not (todos-done-item-p))
+                 (user-error "Only done items can be undone")
+               (todos-item-start)
+               (unless marked
+                 (setq ov (make-overlay (save-excursion (todos-item-start))
+                                        (save-excursion (todos-item-end))))
+                 (overlay-put ov 'face 'todos-search))
+               ;; Find the end of the date string added upon tagging item as
+               ;; done.
+               (setq start (search-forward "] "))
+               (setq item-count (1+ item-count))
+               (unless (looking-at (regexp-quote todos-nondiary-start))
+                 (setq diary-count (1+ diary-count)))
+               (setq end (save-excursion (todos-item-end)))
+               ;; Ask (once) whether to omit done item's comment.  If
+               ;; affirmed, omit subsequent comments without asking.
+               (when (re-search-forward
+                      (concat " \\[" (regexp-quote todos-comment-string)
+                              ": [^]]+\\]") end t)
+                 (unwind-protect
+                     (if (eq first 'first)
+                         (setq first
+                               (if (eq todos-undo-item-omit-comment 'ask)
+                                   (when (todos-y-or-n-p
+                                          (concat "Omit comment" pl
+                                                  " from restored item"
+                                                  pl "? "))
+                                     'omit)
+                                 (when todos-undo-item-omit-comment 'omit)))
+                       t)
+                   (when (and (eq first 'first) ov) (delete-overlay ov)))
+                 (when (eq first 'omit)
+                   (setq end (match-beginning 0))))
+               (setq item (concat item
+                                  (buffer-substring-no-properties start end)
+                                  (when marked "\n")))
+               (unless marked (throw 'done nil))))
+           (todos-forward-item)))
+       (unwind-protect
+           (progn
+             ;; Chop off last newline of multiple items string, since
+             ;; it will be reinserted on setting priority.
+             (and marked (setq item (substring item 0 -1)))
+             (todos-set-item-priority item cat t)
+             (setq npoint (point))
+             (setq undone t))
+         (when ov (delete-overlay ov))
+         (if (not undone)
+             (goto-char opoint)
+           (if marked
+               (progn
+                 (setq item nil)
+                 (re-search-forward
+                  (concat "^" (regexp-quote todos-category-done)) nil t)
+                 (while (not (eobp))
+                   (if (todos-marked-item-p)
+                       (todos-remove-item)
+                     (todos-forward-item)))
+                 (setq todos-categories-with-marks
+                       (assq-delete-all cat todos-categories-with-marks)))
+             (goto-char omark)
+             (todos-remove-item))
+           (todos-update-count 'todo item-count)
+           (todos-update-count 'done (- item-count))
+           (when diary-count (todos-update-count 'diary diary-count))
+           (todos-update-categories-sexp)
+           (let ((todos-show-with-done (> (todos-get-count 'done) 0)))
+             (todos-category-select))
+           ;; Put cursor on undone item.
+           (goto-char npoint)))
+       (set-marker omark nil)))))
 
 ;; -----------------------------------------------------------------------------
-;;; Searching
+;;; Done item archives
 ;; -----------------------------------------------------------------------------
 
-(defun todos-search ()
-  "Search for a regular expression in this Todos file.
-The search runs through the whole file and encompasses all and
-only todo and done items; it excludes category names.  Multiple
-matches are shown sequentially, highlighted in `todos-search'
-face."
+(defun todos-find-archive (&optional ask)
+  "Visit the archive of the current Todos category, if it exists.
+If the category has no archived items, prompt to visit the
+archive anyway.  If there is no archive for this file or with
+non-nil argument ASK, prompt to visit another archive.
+
+The buffer showing the archive is in Todos Archive mode.  The
+first visit in a session displays the first category in the
+archive, subsequent visits return to the last category
+displayed."
   (interactive)
-  (let ((regex (read-from-minibuffer "Enter a search string (regexp): "))
-       (opoint (point))
-       matches match cat in-done ov mlen msg)
-    (widen)
-    (goto-char (point-min))
-    (while (not (eobp))
-      (setq match (re-search-forward regex nil t))
-      (goto-char (line-beginning-position))
-      (unless (or (equal (point) 1)
-                 (looking-at (concat "^" (regexp-quote todos-category-beg))))
-       (if match (push match matches)))
-      (forward-line))
-    (setq matches (reverse matches))
-    (if matches
-       (catch 'stop
-         (while matches
-           (setq match (pop matches))
-           (goto-char match)
-           (todos-item-start)
-           (when (looking-at todos-done-string-start)
-             (setq in-done t))
-           (re-search-backward (concat "^" (regexp-quote todos-category-beg)
-                                       "\\(.*\\)\n") nil t)
-           (setq cat (match-string-no-properties 1))
-           (todos-category-number cat)
-           (todos-category-select)
-           (if in-done
-               (unless todos-show-with-done (todos-toggle-view-done-items)))
-           (goto-char match)
-           (setq ov (make-overlay (- (point) (length regex)) (point)))
-           (overlay-put ov 'face 'todos-search)
-           (when matches
-             (setq mlen (length matches))
-             (if (todos-y-or-n-p
-                  (if (> mlen 1)
-                      (format "There are %d more matches; go to next match? "
-                              mlen)
-                    "There is one more match; go to it? "))
-                 (widen)
-               (throw 'stop (setq msg (if (> mlen 1)
-                                          (format "There are %d more matches."
-                                                  mlen)
-                                        "There is one more match."))))))
-         (setq msg "There are no more matches."))
-      (todos-category-select)
-      (goto-char opoint)
-      (message "No match for \"%s\"" regex))
-    (when msg
-      (if (todos-y-or-n-p (concat msg "\nUnhighlight matches? "))
-         (todos-clear-matches)
-       (message "You can unhighlight the matches later by typing %s"
-                (key-description (car (where-is-internal
-                                       'todos-clear-matches))))))))
+  (let* ((cat (todos-current-category))
+        (count (todos-get-count 'archived cat))
+        (archive (concat (file-name-sans-extension todos-current-todos-file)
+                         ".toda"))
+        place)
+    (setq place (cond (ask 'other-archive)
+                     ((file-exists-p archive) 'this-archive)
+                     (t (when (todos-y-or-n-p
+                               (concat "This file has no archive; "
+                                       "visit another archive? "))
+                          'other-archive))))
+    (when (eq place 'other-archive)
+      (setq archive (todos-read-file-name "Choose a Todos archive: " t t)))
+    (when (and (eq place 'this-archive) (zerop count))
+      (setq place (when (todos-y-or-n-p
+                         (concat "This category has no archived items;"
+                                 " visit archive anyway? "))
+                    'other-cat)))
+    (when place
+      (set-window-buffer (selected-window)
+                        (set-buffer (find-file-noselect archive)))
+      (if (member place '(other-archive other-cat))
+         (setq todos-category-number 1)
+       (todos-category-number cat))
+      (todos-category-select))))
 
-(defun todos-clear-matches ()
-  "Remove highlighting on matches found by todos-search."
+(defun todos-choose-archive ()
+  "Choose an archive and visit it."
   (interactive)
-  (remove-overlays 1 (1+ (buffer-size)) 'face 'todos-search))
+  (todos-find-archive t))
 
-;; -----------------------------------------------------------------------------
-;;; Item filtering options
-;; -----------------------------------------------------------------------------
+(defun todos-archive-done-item (&optional all)
+  "Archive at least one done item in this category.
 
-(defcustom todos-top-priorities-overrides nil
-  "List of rules specifying number of top priority items to show.
-These rules override `todos-top-priorities' on invocations of
-`\\[todos-filter-top-priorities]' and
-`\\[todos-filter-top-priorities-multifile]'.  Each rule is a list
-of the form (FILE NUM ALIST), where FILE is a member of
-`todos-files', NUM is a number specifying the default number of
-top priority items for each category in that file, and ALIST,
-when non-nil, consists of conses of a category name in FILE and a
-number specifying the default number of top priority items in
-that category, which overrides NUM.
+With prefix argument ALL, prompt whether to archive all done
+items in this category and on confirmation archive them.
+Otherwise, if there are marked done items (and no marked todo
+items), archive all of these; otherwise, archive the done item at
+point.
 
-This variable should be set interactively by
-`\\[todos-set-top-priorities-in-file]' or
-`\\[todos-set-top-priorities-in-category]'."
-  :type 'sexp
-  :group 'todos-filtered)
+If the archive of this file does not exist, it is created.  If
+this category does not exist in the archive, it is created."
+  (interactive "P")
+  (when (eq major-mode 'todos-mode)
+    (if (and all (zerop (todos-get-count 'done)))
+       (message "No done items in this category")
+      (catch 'end
+       (let* ((cat (todos-current-category))
+              (tbuf (current-buffer))
+              (marked (assoc cat todos-categories-with-marks))
+              (afile (concat (file-name-sans-extension
+                              todos-current-todos-file) ".toda"))
+              (archive (if (file-exists-p afile)
+                           (find-file-noselect afile t)
+                         (get-buffer-create afile)))
+              (item (and (todos-done-item-p)
+                         (concat (todos-item-string) "\n")))
+              (count 0)
+              (opoint (unless (todos-done-item-p) (point)))
+              marked-items beg end all-done
+              buffer-read-only)
+         (cond
+          (all
+           (if (todos-y-or-n-p "Archive all done items in this category? ")
+               (save-excursion
+                 (save-restriction
+                   (goto-char (point-min))
+                   (widen)
+                   (setq beg (progn
+                               (re-search-forward todos-done-string-start
+                                                  nil t)
+                               (match-beginning 0))
+                         end (if (re-search-forward
+                                  (concat "^"
+                                          (regexp-quote todos-category-beg))
+                                  nil t)
+                                 (match-beginning 0)
+                               (point-max))
+                         all-done (buffer-substring-no-properties beg end)
+                         count (todos-get-count 'done))
+                   ;; Restore starting point, unless it was on a done
+                   ;; item, since they will all be deleted.
+                   (when opoint (goto-char opoint))))
+             (throw 'end nil)))
+          (marked
+           (save-excursion
+             (goto-char (point-min))
+             (while (not (eobp))
+               (when (todos-marked-item-p)
+                 (if (not (todos-done-item-p))
+                     (throw 'end (message "Only done items can be archived"))
+                   (setq marked-items
+                         (concat marked-items (todos-item-string) "\n"))
+                   (setq count (1+ count))))
+               (todos-forward-item)))))
+         (if (not (or marked all item))
+             (throw 'end (message "Only done items can be archived"))
+           (with-current-buffer archive
+             (unless buffer-file-name (erase-buffer))
+             (let (buffer-read-only)
+               (widen)
+               (goto-char (point-min))
+               (if (and (re-search-forward
+                         (concat "^" (regexp-quote
+                                      (concat todos-category-beg cat)) "$")
+                         nil t)
+                        (re-search-forward (regexp-quote todos-category-done)
+                                           nil t))
+                   ;; Start of done items section in existing category.
+                   (forward-char)
+                 (todos-add-category nil cat)
+                 ;; Start of done items section in new category.
+                 (goto-char (point-max)))
+               (insert (cond (marked marked-items)
+                             (all all-done)
+                             (item)))
+               (todos-update-count 'done (if (or marked all) count 1) cat)
+               (todos-update-categories-sexp)
+               ;; If archive is new, save to file now (using write-region in
+               ;; order not to get prompted for file to save to), to let
+               ;; auto-mode-alist take effect below.
+               (unless buffer-file-name
+                 (write-region nil nil afile)
+                 (kill-buffer))))
+           (with-current-buffer tbuf
+             (cond
+              (all
+               (save-excursion
+                 (save-restriction
+                   ;; Make sure done items are accessible.
+                   (widen)
+                   (remove-overlays beg end)
+                   (delete-region beg end)
+                   (todos-update-count 'done (- count))
+                   (todos-update-count 'archived count))))
+              ((or marked
+                   ;; If we're archiving all done items, can't
+                   ;; first archive item point was on, since
+                   ;; that will short-circuit the rest.
+                   (and item (not all)))
+               (and marked (goto-char (point-min)))
+               (catch 'done
+                 (while (not (eobp))
+                   (if (or (and marked (todos-marked-item-p)) item)
+                       (progn
+                         (todos-remove-item)
+                         (todos-update-count 'done -1)
+                         (todos-update-count 'archived 1)
+                         ;; Don't leave point below last item.
+                         (and item (bolp) (eolp) (< (point-min) (point-max))
+                              (todos-backward-item))
+                         (when item
+                           (throw 'done (setq item nil))))
+                     (todos-forward-item))))))
+             (when marked
+               (setq todos-categories-with-marks
+                     (assq-delete-all cat todos-categories-with-marks)))
+             (todos-update-categories-sexp)
+             (todos-prefix-overlays)))
+         (find-file afile)
+         (todos-category-number cat)
+         (todos-category-select)
+         (split-window-below)
+         (set-window-buffer (selected-window) tbuf)
+         ;; Make todo file current to select category.
+         (find-file (buffer-file-name tbuf))
+         ;; Make sure done item separator is hidden (if done items
+         ;; were initially visible).
+         (let (todos-show-with-done) (todos-category-select)))))))
 
-(defcustom todos-top-priorities 1
-  "Default number of top priorities shown by `todos-filter-top-priorities'."
-  :type 'integer
-  :group 'todos-filtered)
+(defun todos-unarchive-items ()
+  "Unarchive at least one item in this archive category.
+If there are marked items, unarchive all of these; otherwise,
+unarchive the item at point.
 
-(defcustom todos-filter-files nil
-  "List of default files for multifile item filtering."
-  :type `(set ,@(mapcar (lambda (f) (list 'const f))
-                       (mapcar 'todos-short-file-name
-                               (funcall todos-files-function))))
-  :group 'todos-filtered)
+Unarchived items are restored as done items to the corresponding
+category in the Todos file, inserted at the top of done items
+section.  If all items in the archive category have been
+restored, the category is deleted from the archive.  If this was
+the only category in the archive, the archive file is deleted."
+  (interactive)
+  (when (eq major-mode 'todos-archive-mode)
+    (let* ((cat (todos-current-category))
+          (tbuf (find-file-noselect
+                 (concat (file-name-sans-extension todos-current-todos-file)
+                         ".todo") t))
+          (marked (assoc cat todos-categories-with-marks))
+          (item (concat (todos-item-string) "\n"))
+          (marked-count 0)
+          marked-items
+          buffer-read-only)
+      (when marked
+       (save-excursion
+         (goto-char (point-min))
+         (while (not (eobp))
+           (when (todos-marked-item-p)
+             (setq marked-items (concat marked-items (todos-item-string) "\n"))
+             (setq marked-count (1+ marked-count)))
+           (todos-forward-item))))
+      ;; Restore items to top of category's done section and update counts.
+      (with-current-buffer tbuf
+       (let (buffer-read-only newcat)
+         (widen)
+         (goto-char (point-min))
+         ;; Find the corresponding todo category, or if there isn't
+         ;; one, add it.
+         (unless (re-search-forward
+                  (concat "^" (regexp-quote (concat todos-category-beg cat))
+                          "$") nil t)
+           (todos-add-category nil cat)
+           (setq newcat t))
+         ;; Go to top of category's done section.
+         (re-search-forward
+          (concat "^" (regexp-quote todos-category-done)) nil t)
+         (forward-line)
+         (cond (marked
+                (insert marked-items)
+                (todos-update-count 'done marked-count cat)
+                (unless newcat         ; Newly added category has no archive.
+                  (todos-update-count 'archived (- marked-count) cat)))
+               (t
+                (insert item)
+                (todos-update-count 'done 1 cat)
+                (unless newcat         ; Newly added category has no archive.
+                  (todos-update-count 'archived -1 cat))))
+         (todos-update-categories-sexp)))
+      ;; Delete restored items from archive.
+      (when marked
+       (setq item nil)
+       (goto-char (point-min)))
+      (catch 'done
+       (while (not (eobp))
+         (if (or (todos-marked-item-p) item)
+             (progn
+               (todos-remove-item)
+               (when item
+                 (throw 'done (setq item nil))))
+           (todos-forward-item))))
+      (todos-update-count 'done (if marked (- marked-count) -1) cat)
+      ;; If that was the last category in the archive, delete the whole file.
+      (if (= (length todos-categories) 1)
+         (progn
+           (delete-file todos-current-todos-file)
+           ;; Kill the archive buffer silently.
+           (set-buffer-modified-p nil)
+           (kill-buffer))
+       ;; Otherwise, if the archive category is now empty, delete it.
+       (when (eq (point-min) (point-max))
+         (widen)
+         (let ((beg (re-search-backward
+                     (concat "^" (regexp-quote todos-category-beg) cat "$")
+                     nil t))
+               (end (if (re-search-forward
+                         (concat "^" (regexp-quote todos-category-beg))
+                         nil t 2)
+                        (match-beginning 0)
+                      (point-max))))
+           (remove-overlays beg end)
+           (delete-region beg end)
+           (setq todos-categories (delete (assoc cat todos-categories)
+                                          todos-categories))
+           (todos-update-categories-sexp))))
+      ;; Visit category in Todos file and show restored done items.
+      (let ((tfile (buffer-file-name tbuf))
+           (todos-show-with-done t))
+       (set-window-buffer (selected-window)
+                          (set-buffer (find-file-noselect tfile)))
+       (todos-category-number cat)
+       (todos-category-select)
+       (message "Items unarchived.")))))
 
-(defcustom todos-filter-done-items nil
-  "Non-nil to include done items when processing regexp filters.
-Done items from corresponding archive files are also included."
-  :type 'boolean
-  :group 'todos-filtered)
+(defun todos-jump-to-archive-category (&optional file)
+  "Prompt for a category in a Todos archive and jump to it.
+With prefix argument FILE, prompt for an archive and choose (with
+TAB completion) a category in it to jump to; otherwise, choose
+and jump to any category in the current archive."
+  (interactive "P")
+  (todos-jump-to-category file 'archive))
 
 ;; -----------------------------------------------------------------------------
-;;; Item filtering commands
+;;; Displaying and sorting tables of categories
 ;; -----------------------------------------------------------------------------
 
-(defun todos-set-top-priorities-in-file ()
-  "Set number of top priorities for this file.
-See `todos-set-top-priorities' for more details."
-  (interactive)
-  (todos-set-top-priorities))
-
-(defun todos-set-top-priorities-in-category ()
-  "Set number of top priorities for this category.
-See `todos-set-top-priorities' for more details."
-  (interactive)
-  (todos-set-top-priorities t))
-
-(defun todos-filter-top-priorities (&optional arg)
-  "Display a list of top priority items from different categories.
-The categories can be any of those in the current Todos file.
-
-With numerical prefix ARG show at most ARG top priority items
-from each category.  With `C-u' as prefix argument show the
-numbers of top priority items specified by category in
-`todos-top-priorities-overrides', if this has an entry for the file(s);
-otherwise show `todos-top-priorities' items per category in the
-file(s).  With no prefix argument, if a top priorities file for
-the current Todos file has previously been saved (see
-`todos-save-filtered-items-buffer'), visit this file; if there is
-no such file, build the list as with prefix argument `C-u'.
-
-  The prefix ARG regulates how many top priorities from
-each category to show, as described above."
-  (interactive "P")
-  (todos-filter-items 'top arg))
+(defcustom todos-categories-category-label "Category"
+  "Category button label in Todos Categories mode."
+  :type 'string
+  :group 'todos-categories)
 
-(defun todos-filter-top-priorities-multifile (&optional arg)
-  "Display a list of top priority items from different categories.
-The categories are a subset of the categories in the files listed
-in `todos-filter-files', or if this nil, in the files chosen from
-a file selection dialog that pops up in this case.
+(defcustom todos-categories-todo-label "Todo"
+  "Todo button label in Todos Categories mode."
+  :type 'string
+  :group 'todos-categories)
 
-With numerical prefix ARG show at most ARG top priority items
-from each category in each file.  With `C-u' as prefix argument
-show the numbers of top priority items specified in
-`todos-top-priorities-overrides', if this is non-nil; otherwise show
-`todos-top-priorities' items per category.  With no prefix
-argument, if a top priorities file for the chosen Todos files
-exists (see `todos-save-filtered-items-buffer'), visit this file;
-if there is no such file, do the same as with prefix argument
-`C-u'."
-  (interactive "P")
-  (todos-filter-items 'top arg t))
+(defcustom todos-categories-diary-label "Diary"
+  "Diary button label in Todos Categories mode."
+  :type 'string
+  :group 'todos-categories)
 
-(defun todos-filter-diary-items (&optional arg)
-  "Display a list of todo diary items from different categories.
-The categories can be any of those in the current Todos file.
+(defcustom todos-categories-done-label "Done"
+  "Done button label in Todos Categories mode."
+  :type 'string
+  :group 'todos-categories)
 
-Called with no prefix ARG, if a diary items file for the current
-Todos file has previously been saved (see
-`todos-save-filtered-items-buffer'), visit this file; if there is
-no such file, build the list of diary items.  Called with a
-prefix argument, build the list even if there is a saved file of
-diary items."
-  (interactive "P")
-  (todos-filter-items 'diary arg))
+(defcustom todos-categories-archived-label "Archived"
+  "Archived button label in Todos Categories mode."
+  :type 'string
+  :group 'todos-categories)
 
-(defun todos-filter-diary-items-multifile (&optional arg)
-  "Display a list of todo diary items from different categories.
-The categories are a subset of the categories in the files listed
-in `todos-filter-files', or if this nil, in the files chosen from
-a file selection dialog that pops up in this case.
+(defcustom todos-categories-totals-label "Totals"
+  "String to label total item counts in Todos Categories mode."
+  :type 'string
+  :group 'todos-categories)
 
-Called with no prefix ARG, if a diary items file for the chosen
-Todos files has previously been saved (see
-`todos-save-filtered-items-buffer'), visit this file; if there is
-no such file, build the list of diary items.  Called with a
-prefix argument, build the list even if there is a saved file of
-diary items."
-  (interactive "P")
-  (todos-filter-items 'diary arg t))
+(defcustom todos-categories-number-separator " | "
+  "String between number and category in Todos Categories mode.
+This separates the number from the category name in the default
+categories display according to priority."
+  :type 'string
+  :group 'todos-categories)
 
-(defun todos-filter-regexp-items (&optional arg)
-  "Prompt for a regular expression and display items that match it.
-The matches can be from any categories in the current Todos file
-and with non-nil option `todos-filter-done-items', can include
-not only todo items but also done items, including those in
-Archive files.
+(defcustom todos-categories-align 'center
+  "Alignment of category names in Todos Categories mode."
+  :type '(radio (const left) (const center) (const right))
+  :group 'todos-categories)
 
-Called with no prefix ARG, if a regexp items file for the current
-Todos file has previously been saved (see
-`todos-save-filtered-items-buffer'), visit this file; if there is
-no such file, build the list of regexp items.  Called with a
-prefix argument, build the list even if there is a saved file of
-regexp items."
-  (interactive "P")
-  (todos-filter-items 'regexp arg))
+(defun todos-show-categories-table ()
+  "Display a table of the current file's categories and item counts.
 
-(defun todos-filter-regexp-items-multifile (&optional arg)
-  "Prompt for a regular expression and display items that match it.
-The matches can be from any categories in the files listed in
-`todos-filter-files', or if this nil, in the files chosen from a
-file selection dialog that pops up in this case.  With non-nil
-option `todos-filter-done-items', the matches can include not
-only todo items but also done items, including those in Archive
-files.
+In the initial display the categories are numbered, indicating
+their current order for navigating by \\[todos-forward-category]
+and \\[todos-backward-category].  You can persistantly change the
+order of the category at point by typing
+\\[todos-set-category-number], \\[todos-raise-category] or
+\\[todos-lower-category].
 
-Called with no prefix ARG, if a regexp items file for the current
-Todos file has previously been saved (see
-`todos-save-filtered-items-buffer'), visit this file; if there is
-no such file, build the list of regexp items.  Called with a
-prefix argument, build the list even if there is a saved file of
-regexp items."
-  (interactive "P")
-  (todos-filter-items 'regexp arg t))
+The labels above the category names and item counts are buttons,
+and clicking these changes the display: sorted by category name
+or by the respective item counts (alternately descending or
+ascending).  In these displays the categories are not numbered
+and \\[todos-set-category-number], \\[todos-raise-category] and
+\\[todos-lower-category] are disabled.  (Programmatically, the
+sorting is triggered by passing a non-nil SORTKEY argument.)
 
-(defun todos-find-filtered-items-file ()
-  "Choose a filtered items file and visit it."
+In addition, the lines with the category names and item counts
+are buttonized, and pressing one of these button jumps to the
+category in Todos mode (or Todos Archive mode, for categories
+containing only archived items, provided user option
+`todos-skip-archived-categories' is non-nil.  These categories
+are shown in `todos-archived-only' face."
   (interactive)
-  (let ((files (directory-files todos-directory t "\.tod[rty]$" t))
-       falist file)
-    (dolist (f files)
-      (let ((type (cond ((equal (file-name-extension f) "todr") "regexp")
-                       ((equal (file-name-extension f) "todt") "top")
-                       ((equal (file-name-extension f) "tody") "diary"))))
-       (push (cons (concat (todos-short-file-name f) " (" type ")") f)
-             falist)))
-    (setq file (completing-read "Choose a filtered items file: "
-                               falist nil t nil nil (car falist)))
-    (setq file (cdr (assoc-string file falist)))
-    (find-file file)))
+  (todos-display-categories)
+  (let (sortkey)
+    (todos-update-categories-display sortkey)))
 
-(defun todos-go-to-source-item ()
-  "Display the file and category of the filtered item at point."
-  (interactive)
-  (let* ((str (todos-item-string))
-        (buf (current-buffer))
-        (res (todos-find-item str))
-        (found (nth 0 res))
-        (file (nth 1 res))
-        (cat (nth 2 res)))
-    (if (not found)
-       (message "Category %s does not contain this item." cat)
-      (kill-buffer buf)
-      (set-window-buffer (selected-window)
-                        (set-buffer (find-buffer-visiting file)))
-      (setq todos-current-todos-file file)
-      (setq todos-category-number (todos-category-number cat))
-      (let ((todos-show-with-done (if (or todos-filter-done-items
-                                         (eq (cdr found) 'done))
-                                     t
-                                   todos-show-with-done)))
-       (todos-category-select))
-      (goto-char (car found)))))
+(defun todos-next-button (n)
+  "Move point to the Nth next button in the table of categories."
+  (interactive "p")
+  (forward-button n 'wrap 'display-message)
+  (and (bolp) (button-at (point))
+       ;; Align with beginning of category label.
+       (forward-char (+ 4 (length todos-categories-number-separator)))))
 
-;; -----------------------------------------------------------------------------
-;;; Printing Todos Buffers
-;; -----------------------------------------------------------------------------
+(defun todos-previous-button (n)
+  "Move point to the Nth previous button in the table of categories."
+  (interactive "p")
+  (backward-button n 'wrap 'display-message)
+  (and (bolp) (button-at (point))
+       ;; Align with beginning of category label.
+       (forward-char (+ 4 (length todos-categories-number-separator)))))
 
-(defcustom todos-print-buffer-function 'ps-print-buffer-with-faces
-  "Function called by the command `todos-print-buffer'."
-  :type 'symbol
-  :group 'todos)
+(defun todos-set-category-number (&optional arg)
+  "Change number of category at point in the table of categories.
 
-(defvar todos-print-buffer "*Todos Print*"
-  "Name of buffer containing printable Todos text.")
+With ARG nil, prompt for the new number.  Alternatively, the
+enter the new number with numerical prefix ARG.  Otherwise, if
+ARG is either of the symbols `raise' or `lower', raise or lower
+the category line in the table by one, respectively, thereby
+decreasing or increasing its number."
+  (interactive "P")
+  (let ((curnum (save-excursion
+                 ;; Get the number representing the priority of the category
+                 ;; on the current line.
+                 (forward-line 0) (skip-chars-forward " ") (number-at-point))))
+    (when curnum               ; Do nothing if we're not on a category line.
+      (let* ((maxnum (length todos-categories))
+            (prompt (format "Set category priority (1-%d): " maxnum))
+            (col (current-column))
+            (buffer-read-only nil)
+            (priority (cond ((and (eq arg 'raise) (> curnum 1))
+                             (1- curnum))
+                            ((and (eq arg 'lower) (< curnum maxnum))
+                             (1+ curnum))))
+            candidate)
+       (while (not priority)
+         (setq candidate (or arg (read-number prompt)))
+         (setq arg nil)
+         (setq prompt
+               (cond ((or (< candidate 1) (> candidate maxnum))
+                      (format "Priority must be an integer between 1 and %d: "
+                              maxnum))
+                     ((= candidate curnum)
+                      "Choose a different priority than the current one: ")))
+         (unless prompt (setq priority candidate)))
+       (let* ((lower (< curnum priority)) ; Priority is being lowered.
+              (head (butlast todos-categories
+                             (apply (if lower 'identity '1+)
+                                    (list (- maxnum priority)))))
+              (tail (nthcdr (apply (if lower 'identity '1-) (list priority))
+                            todos-categories))
+              ;; Category's name and items counts list.
+              (catcons (nth (1- curnum) todos-categories))
+              (todos-categories (nconc head (list catcons) tail))
+              newcats)
+         (when lower (setq todos-categories (nreverse todos-categories)))
+         (setq todos-categories (delete-dups todos-categories))
+         (when lower (setq todos-categories (nreverse todos-categories)))
+         (setq newcats todos-categories)
+         (kill-buffer)
+         (with-current-buffer (find-buffer-visiting todos-current-todos-file)
+           (setq todos-categories newcats)
+           (todos-update-categories-sexp))
+         (todos-show-categories-table)
+         (forward-line (1+ priority))
+         (forward-char col))))))
 
-(defun todos-print-buffer (&optional to-file)
-  "Produce a printable version of the current Todos buffer.
-This converts overlays and soft line wrapping and, depending on
-the value of `todos-print-buffer-function', includes faces.  With
-non-nil argument TO-FILE write the printable version to a file;
-otherwise, send it to the default printer."
+(defun todos-raise-category ()
+  "Raise priority of category at point in Todos Categories buffer."
   (interactive)
-  (let ((buf todos-print-buffer)
-       (header (cond
-                ((eq major-mode 'todos-mode)
-                 (concat "Todos File: "
-                         (todos-short-file-name todos-current-todos-file)
-                         "\nCategory: " (todos-current-category)))
-                ((eq major-mode 'todos-filtered-items-mode)
-                 (buffer-name))))
-       (prefix (propertize (concat todos-prefix " ")
-                           'face 'todos-prefix-string))
-       (num 0)
-       (fill-prefix (make-string todos-indent-to-here 32))
-       (content (buffer-string))
-       file)
-    (with-current-buffer (get-buffer-create buf)
-      (insert content)
-      (goto-char (point-min))
-      (while (not (eobp))
-       (let ((beg (point))
-             (end (save-excursion (todos-item-end))))
-         (when todos-number-prefix
-           (setq num (1+ num))
-           (setq prefix (propertize (concat (number-to-string num) " ")
-                                    'face 'todos-prefix-string)))
-         (insert prefix)
-         (fill-region beg end))
-       ;; Calling todos-forward-item infloops at todos-item-start due to
-       ;; non-overlay prefix, so search for item start instead.
-       (if (re-search-forward todos-item-start nil t)
-           (beginning-of-line)
-         (goto-char (point-max))))
-      (if (re-search-backward (concat "^" (regexp-quote todos-category-done))
-                             nil t)
-         (replace-match todos-done-separator))
-      (goto-char (point-min))
-      (insert header)
-      (newline 2)
-      (if to-file
-         (let ((file (read-file-name "Print to file: ")))
-           (funcall todos-print-buffer-function file))
-       (funcall todos-print-buffer-function)))
-    (kill-buffer buf)))
+  (todos-set-category-number 'raise))
 
-(defun todos-print-buffer-to-file ()
-  "Save printable version of this Todos buffer to a file."
+(defun todos-lower-category ()
+  "Lower priority of category at point in Todos Categories buffer."
   (interactive)
-  (todos-print-buffer t))
-
-;; -----------------------------------------------------------------------------
-;;; Legacy Todo Mode Files
-;; -----------------------------------------------------------------------------
+  (todos-set-category-number 'lower))
 
-(defcustom todos-todo-mode-date-time-regexp
-  (concat "\\(?1:[0-9]\\{4\\}\\)-\\(?2:[0-9]\\{2\\}\\)-"
-         "\\(?3:[0-9]\\{2\\}\\) \\(?4:[0-9]\\{2\\}:[0-9]\\{2\\}\\)")
-  "Regexp matching legacy todo-mode.el item date-time strings.
-In order for `todos-convert-legacy-files' to correctly convert this
-string to the current Todos format, the regexp must contain four
-explicitly numbered groups (see `(elisp) Regexp Backslash'),
-where group 1 matches a string for the year, group 2 a string for
-the month, group 3 a string for the day and group 4 a string for
-the time.  The default value converts date-time strings built
-using the default value of `todo-time-string-format' from
-todo-mode.el."
-  :type 'regexp
-  :group 'todos)
+(defun todos-sort-categories-alphabetically-or-numerically ()
+  "Sort table of categories alphabetically or numerically."
+  (interactive)
+  (save-excursion
+    (goto-char (point-min))
+    (forward-line 2)
+    (if (member 'alpha todos-descending-counts)
+       (progn
+         (todos-update-categories-display nil)
+         (setq todos-descending-counts
+               (delete 'alpha todos-descending-counts)))
+      (todos-update-categories-display 'alpha))))
 
-(defun todos-convert-legacy-date-time ()
-  "Return converted date-time string.
-Helper function for `todos-convert-legacy-files'."
-  (let* ((year (match-string 1))
-        (month (match-string 2))
-        (monthname (calendar-month-name (string-to-number month) t))
-        (day (match-string 3))
-        (time (match-string 4))
-        dayname)
-    (replace-match "")
-    (insert (mapconcat 'eval calendar-date-display-form "")
-           (when time (concat " " time)))))
+(defun todos-sort-categories-by-todo ()
+  "Sort table of categories by number of todo items."
+  (interactive)
+  (save-excursion
+    (goto-char (point-min))
+    (forward-line 2)
+    (todos-update-categories-display 'todo)))
 
-(defun todos-convert-legacy-files ()
-  "Convert legacy Todo files to the current Todos format.
-The files `todo-file-do' and `todo-file-done' are converted and
-saved (the latter as a Todos Archive file) with a new name in
-`todos-directory'.  See also the documentation string of
-`todos-todo-mode-date-time-regexp' for further details."
+(defun todos-sort-categories-by-diary ()
+  "Sort table of categories by number of diary items."
+  (interactive)
+  (save-excursion
+    (goto-char (point-min))
+    (forward-line 2)
+    (todos-update-categories-display 'diary)))
+
+(defun todos-sort-categories-by-done ()
+  "Sort table of categories by number of non-archived done items."
   (interactive)
-  (require 'todo-mode)
-  ;; Convert `todo-file-do'.
-  (if (file-exists-p todo-file-do)
-      (let ((default "todo-do-conv")
-           file archive-sexp)
-       (with-temp-buffer
-         (insert-file-contents todo-file-do)
-         (let ((end (search-forward ")" (line-end-position) t))
-               (beg (search-backward "(" (line-beginning-position) t)))
-           (setq todo-categories
-                 (read (buffer-substring-no-properties beg end))))
-         (todo-mode)
-         (delete-region (line-beginning-position) (1+ (line-end-position)))
-         (while (not (eobp))
-           (cond
-            ((looking-at (regexp-quote (concat todo-prefix todo-category-beg)))
-             (replace-match todos-category-beg))
-            ((looking-at (regexp-quote todo-category-end))
-             (replace-match ""))
-            ((looking-at (regexp-quote (concat todo-prefix " "
-                                               todo-category-sep)))
-             (replace-match todos-category-done))
-            ((looking-at (concat (regexp-quote todo-prefix) " "
-                                 todos-todo-mode-date-time-regexp " "
-                                 (regexp-quote todo-initials) ":"))
-             (todos-convert-legacy-date-time)))
-           (forward-line))
-         (setq file (concat todos-directory
-                            (read-string
-                             (format "Save file as (default \"%s\"): " default)
-                             nil nil default)
-                            ".todo"))
-         (write-region (point-min) (point-max) file nil 'nomessage nil t))
-       (with-temp-buffer
-         (insert-file-contents file)
-         (let ((todos-categories (todos-make-categories-list t)))
-           (todos-update-categories-sexp))
-         (write-region (point-min) (point-max) file nil 'nomessage))
-       ;; Convert `todo-file-done'.
-       (when (file-exists-p todo-file-done)
-         (with-temp-buffer
-           (insert-file-contents todo-file-done)
-           (let ((beg (make-marker))
-                 (end (make-marker))
-                 cat cats comment item)
-             (while (not (eobp))
-               (when (looking-at todos-todo-mode-date-time-regexp)
-                 (set-marker beg (point))
-                 (todos-convert-legacy-date-time)
-                 (set-marker end (point))
-                 (goto-char beg)
-                 (insert "[" todos-done-string)
-                 (goto-char end)
-                 (insert "]")
-                 (forward-char)
-                 (when (looking-at todos-todo-mode-date-time-regexp)
-                   (todos-convert-legacy-date-time))
-                 (when (looking-at (concat " "
-                                           (regexp-quote todo-initials) ":"))
-                   (replace-match "")))
-               (if (re-search-forward
-                    (concat "^" todos-todo-mode-date-time-regexp) nil t)
-                   (goto-char (match-beginning 0))
-                 (goto-char (point-max)))
-               (backward-char)
-               (when (looking-back "\\[\\([^][]+\\)\\]")
-                 (setq cat (match-string 1))
-                 (goto-char (match-beginning 0))
-                 (replace-match ""))
-               ;; If the item ends with a non-comment parenthesis not
-               ;; followed by a period, we lose (but we inherit that problem
-               ;; from todo-mode.el).
-               (when (looking-back "(\\(.*\\)) ")
-                 (setq comment (match-string 1))
-                 (replace-match "")
-                 (insert "[" todos-comment-string ": " comment "]"))
-               (set-marker end (point))
-               (if (member cat cats)
-                   ;; If item is already in its category, leave it there.
-                   (unless (save-excursion
-                             (re-search-backward
-                              (concat "^" (regexp-quote todos-category-beg)
-                                      "\\(.*\\)$") nil t)
-                             (string= (match-string 1) cat))
-                     ;; Else move it to its category.
-                     (setq item (buffer-substring-no-properties beg end))
-                     (delete-region beg (1+ end))
-                     (set-marker beg (point))
-                     (re-search-backward
-                      (concat "^"
-                              (regexp-quote (concat todos-category-beg cat))
-                              "$")
-                      nil t)
-                     (forward-line)
-                     (if (re-search-forward
-                          (concat "^" (regexp-quote todos-category-beg)
-                                  "\\(.*\\)$") nil t)
-                         (progn (goto-char (match-beginning 0))
-                                (newline)
-                                (forward-line -1))
-                       (goto-char (point-max)))
-                     (insert item "\n")
-                     (goto-char beg))
-                 (push cat cats)
-                 (goto-char beg)
-                 (insert todos-category-beg cat "\n\n" todos-category-done "\n"))
-               (forward-line))
-             (set-marker beg nil)
-             (set-marker end nil))
-           (setq file (concat (file-name-sans-extension file) ".toda"))
-           (write-region (point-min) (point-max) file nil 'nomessage nil t))
-         (with-temp-buffer
-           (insert-file-contents file)
-           (let ((todos-categories (todos-make-categories-list t)))
-             (todos-update-categories-sexp))
-           (write-region (point-min) (point-max) file nil 'nomessage)
-           (setq archive-sexp (read (buffer-substring-no-properties
-                                     (line-beginning-position)
-                                     (line-end-position)))))
-         (setq file (concat (file-name-sans-extension file) ".todo"))
-         ;; Update categories sexp of converted Todos file again, adding
-         ;; counts of archived items.
-         (with-temp-buffer
-           (insert-file-contents file)
-           (let ((sexp (read (buffer-substring-no-properties
-                              (line-beginning-position)
-                              (line-end-position)))))
-             (dolist (cat sexp)
-               (let ((archive-cat (assoc (car cat) archive-sexp)))
-                 (if archive-cat
-                     (aset (cdr cat) 3 (aref (cdr archive-cat) 2)))))
-             (delete-region (line-beginning-position) (line-end-position))
-             (prin1 sexp (current-buffer)))
-           (write-region (point-min) (point-max) file nil 'nomessage)))
-       (todos-reevaluate-filelist-defcustoms)
-       (message "Format conversion done."))
-    (user-error "No legacy Todo file exists")))
+  (save-excursion
+    (goto-char (point-min))
+    (forward-line 2)
+    (todos-update-categories-display 'done)))
 
-;; =============================================================================
-;;; Todos utilities and internals
-;; =============================================================================
+(defun todos-sort-categories-by-archived ()
+  "Sort table of categories by number of archived items."
+  (interactive)
+  (save-excursion
+    (goto-char (point-min))
+    (forward-line 2)
+    (todos-update-categories-display 'archived)))
 
-(defcustom todos-y-with-space nil
-  "Non-nil means allow SPC to affirm a \"y or n\" question."
-  :type 'boolean
-  :group 'todos)
+(defvar todos-categories-buffer "*Todos Categories*"
+  "Name of buffer in Todos Categories mode.")
 
-(defun todos-y-or-n-p (prompt)
-  "Ask \"y or n\" question PROMPT and return t if answer is \"y\".
-Also return t if answer is \"Y\", but unlike `y-or-n-p', allow
-SPC to affirm the question only if option `todos-y-with-space' is
-non-nil."
-  (unless todos-y-with-space
-    (define-key query-replace-map " " 'ignore))
-  (prog1
-   (y-or-n-p prompt)
-   (define-key query-replace-map " " 'act)))
+(defun todos-longest-category-name-length (categories)
+  "Return the length of the longest name in list CATEGORIES."
+  (let ((longest 0))
+    (dolist (c categories longest)
+      (setq longest (max longest (length c))))))
 
-;; -----------------------------------------------------------------------------
-;;; File-level global variables and support functions
-;; -----------------------------------------------------------------------------
+(defun todos-adjusted-category-label-length ()
+  "Return adjusted length of category label button.
+The adjustment ensures proper tabular alignment in Todos
+Categories mode."
+  (let* ((categories (mapcar 'car todos-categories))
+        (longest (todos-longest-category-name-length categories))
+        (catlablen (length todos-categories-category-label))
+        (lc-diff (- longest catlablen)))
+    (if (and (natnump lc-diff) (cl-oddp lc-diff))
+       (1+ longest)
+      (max longest catlablen))))
 
-(defvar todos-files (funcall todos-files-function)
-  "List of truenames of user's Todos files.")
+(defun todos-padded-string (str)
+  "Return category name or label string STR padded with spaces.
+The placement of the padding is determined by the value of user
+option `todos-categories-align'."
+  (let* ((len (todos-adjusted-category-label-length))
+        (strlen (length str))
+        (strlen-odd (eq (logand strlen 1) 1))
+        (padding (max 0 (/ (- len strlen) 2)))
+        (padding-left (cond ((eq todos-categories-align 'left) 0)
+                            ((eq todos-categories-align 'center) padding)
+                            ((eq todos-categories-align 'right)
+                             (if strlen-odd (1+ (* padding 2)) (* padding 2)))))
+        (padding-right (cond ((eq todos-categories-align 'left)
+                              (if strlen-odd (1+ (* padding 2)) (* padding 2)))
+                             ((eq todos-categories-align 'center)
+                              (if strlen-odd (1+ padding) padding))
+                             ((eq todos-categories-align 'right) 0))))
+    (concat (make-string padding-left 32) str (make-string padding-right 32))))
 
-(defvar todos-archives (funcall todos-files-function t)
-  "List of truenames of user's Todos archives.")
+(defvar todos-descending-counts nil
+  "List of keys for category counts sorted in descending order.")
 
-(defvar todos-visited nil
-  "List of Todos files visited in this session by `todos-show'.
-Used to determine initial display according to the value of
-`todos-show-first'.")
+(defun todos-sort (list &optional key)
+  "Return a copy of LIST, possibly sorted according to KEY."
+  (let* ((l (copy-sequence list))
+        (fn (if (eq key 'alpha)
+                  (lambda (x) (upcase x)) ; Alphabetize case insensitively.
+                (lambda (x) (todos-get-count key x))))
+        ;; Keep track of whether the last sort by key was descending or
+        ;; ascending.
+        (descending (member key todos-descending-counts))
+        (cmp (if (eq key 'alpha)
+                 'string<
+               (if descending '< '>)))
+        (pred (lambda (s1 s2) (let ((t1 (funcall fn (car s1)))
+                                    (t2 (funcall fn (car s2))))
+                                (funcall cmp t1 t2)))))
+    (when key
+      (setq l (sort l pred))
+      ;; Switch between descending and ascending sort order.
+      (if descending
+         (setq todos-descending-counts
+               (delete key todos-descending-counts))
+       (push key todos-descending-counts)))
+    l))
 
-(defvar todos-file-buffers nil
-  "List of file names of live Todos mode buffers.")
+(defun todos-display-sorted (type)
+  "Keep point on the TYPE count sorting button just clicked."
+  (let ((opoint (point)))
+    (todos-update-categories-display type)
+    (goto-char opoint)))
 
-(defvar todos-global-current-todos-file nil
-  "Variable holding name of current Todos file.
-Used by functions called from outside of Todos mode to visit the
-current Todos file rather than the default Todos file (i.e. when
-users option `todos-show-current-file' is non-nil).")
+(defun todos-label-to-key (label)
+  "Return symbol for sort key associated with LABEL."
+  (let (key)
+    (cond ((string= label todos-categories-category-label)
+          (setq key 'alpha))
+         ((string= label todos-categories-todo-label)
+          (setq key 'todo))
+         ((string= label todos-categories-diary-label)
+          (setq key 'diary))
+         ((string= label todos-categories-done-label)
+          (setq key 'done))
+         ((string= label todos-categories-archived-label)
+          (setq key 'archived)))
+    key))
 
-(defun todos-absolute-file-name (name &optional type)
-  "Return the absolute file name of short Todos file NAME.
-With TYPE `archive' or `top' return the absolute file name of the
-short Todos Archive or Top Priorities file name, respectively."
-  ;; NOP if there is no Todos file yet (i.e. don't concatenate nil).
-  (when name
-    (file-truename
-     (concat todos-directory name
-            (cond ((eq type 'archive) ".toda")
-                  ((eq type 'top) ".todt")
-                  ((eq type 'diary) ".tody")
-                  ((eq type 'regexp) ".todr")
-                  (t ".todo"))))))
+(defun todos-insert-sort-button (label)
+  "Insert button for displaying categories sorted by item counts.
+LABEL determines which type of count is sorted."
+  (let* ((str (if (string= label todos-categories-category-label)
+                 (todos-padded-string label)
+               label))
+        (beg (point))
+        (end (+ beg (length str)))
+        ov)
+    (insert-button str 'face nil
+                  'action
+                  `(lambda (button)
+                     (let ((key (todos-label-to-key ,label)))
+                       (if (and (member key todos-descending-counts)
+                                (eq key 'alpha))
+                           (progn
+                             ;; If display is alphabetical, switch back to
+                             ;; category priority order.
+                             (todos-display-sorted nil)
+                             (setq todos-descending-counts
+                                   (delete key todos-descending-counts)))
+                         (todos-display-sorted key)))))
+    (setq ov (make-overlay beg end))
+    (overlay-put ov 'face 'todos-button)))
 
-(defun todos-check-format ()
-  "Signal an error if the current Todos file is ill-formatted.
-Otherwise return t.  Display a message if the file is well-formed
-but the categories sexp differs from the current value of
-`todos-categories'."
-  (save-excursion
-    (save-restriction
-      (widen)
-      (goto-char (point-min))
-      (let* ((cats (prin1-to-string todos-categories))
-            (ssexp (buffer-substring-no-properties (line-beginning-position)
-                                                   (line-end-position)))
-            (sexp (read ssexp)))
-       ;; Check the first line for `todos-categories' sexp.
-       (dolist (c sexp)
-         (let ((v (cdr c)))
-           (unless (and (stringp (car c))
-                        (vectorp v)
-                        (= 4 (length v)))
-             (user-error "Invalid or missing todos-categories sexp"))))
-       (forward-line)
-       ;; Check well-formedness of categories.
-       (let ((legit (concat
-                     "\\(^" (regexp-quote todos-category-beg) "\\)"
-                     "\\|\\(" todos-date-string-start todos-date-pattern "\\)"
-                     "\\|\\(^[ \t]+[^ \t]*\\)"
-                     "\\|^$"
-                     "\\|\\(^" (regexp-quote todos-category-done) "\\)"
-                     "\\|\\(" todos-done-string-start "\\)")))
-         (while (not (eobp))
-           (unless (looking-at legit)
-             (user-error "Illegitimate Todos file format at line %d"
-                    (line-number-at-pos (point))))
-           (forward-line)))
-       ;; Warn user if categories sexp has changed.
-       (unless (string= ssexp cats)
-         (message (concat "The sexp at the beginning of the file differs "
-                          "from the value of `todos-categories.\n"
-                          "If the sexp is wrong, you can fix it with "
-                          "M-x todos-repair-categories-sexp,\n"
-                          "but note this reverts any changes you have "
-                          "made in the order of the categories."))))))
-  t)
+(defun todos-total-item-counts ()
+  "Return a list of total item counts for the current file."
+  (mapcar (lambda (i) (apply '+ (mapcar (lambda (l) (aref l i))
+                                       (mapcar 'cdr todos-categories))))
+         (list 0 1 2 3)))
 
-(defun todos-reevaluate-filelist-defcustoms ()
-  "Reevaluate defcustoms that provide choice list of Todos files."
-  (custom-set-default 'todos-default-todos-file
-                     (symbol-value 'todos-default-todos-file))
-  (todos-reevaluate-default-file-defcustom)
-  (custom-set-default 'todos-filter-files (symbol-value 'todos-filter-files))
-  (todos-reevaluate-filter-files-defcustom)
-  (custom-set-default 'todos-category-completions-files
-                     (symbol-value 'todos-category-completions-files))
-  (todos-reevaluate-category-completions-files-defcustom))
+(defvar todos-categories-category-number 0
+  "Variable for numbering categories in Todos Categories mode.")
 
-(defun todos-reevaluate-default-file-defcustom ()
-  "Reevaluate defcustom of `todos-default-todos-file'.
-Called after adding or deleting a Todos file."
-  (eval (defcustom todos-default-todos-file (car (funcall todos-files-function))
-         "Todos file visited by first session invocation of `todos-show'."
-         :type `(radio ,@(mapcar (lambda (f) (list 'const f))
-                                 (mapcar 'todos-short-file-name
-                                         (funcall todos-files-function))))
-         :group 'todos)))
+(defun todos-insert-category-line (cat &optional nonum)
+  "Insert button with category CAT's name and item counts.
+With non-nil argument NONUM show only these; otherwise, insert a
+number in front of the button indicating the category's priority.
+The number and the category name are separated by the string
+which is the value of the user option
+`todos-categories-number-separator'."
+  (let ((archive (member todos-current-todos-file todos-archives))
+       (num todos-categories-category-number)
+       (str (todos-padded-string cat))
+       (opoint (point)))
+    (setq num (1+ num) todos-categories-category-number num)
+    (insert-button
+     (concat (if nonum
+                (make-string (+ 4 (length todos-categories-number-separator))
+                             32)
+              (format " %3d%s" num todos-categories-number-separator))
+            str
+            (mapconcat (lambda (elt)
+                         (concat
+                          (make-string (1+ (/ (length (car elt)) 2)) 32) ; label
+                          (format "%3d" (todos-get-count (cdr elt) cat)) ; count
+                          ;; Add an extra space if label length is odd.
+                          (when (cl-oddp (length (car elt))) " ")))
+                       (if archive
+                           (list (cons todos-categories-done-label 'done))
+                         (list (cons todos-categories-todo-label 'todo)
+                               (cons todos-categories-diary-label 'diary)
+                               (cons todos-categories-done-label 'done)
+                               (cons todos-categories-archived-label
+                                     'archived)))
+                         "")
+            " ") ; Make highlighting on last column look better.
+     'face (if (and todos-skip-archived-categories
+                   (zerop (todos-get-count 'todo cat))
+                   (zerop (todos-get-count 'done cat))
+                   (not (zerop (todos-get-count 'archived cat))))
+              'todos-archived-only
+            nil)
+     'action `(lambda (button) (let ((buf (current-buffer)))
+                                (todos-jump-to-category nil ,cat)
+                                (kill-buffer buf))))
+    ;; Highlight the sorted count column.
+    (let* ((beg (+ opoint 7 (length str)))
+          end ovl)
+      (cond ((eq nonum 'todo)
+            (setq beg (+ beg 1 (/ (length todos-categories-todo-label) 2))))
+           ((eq nonum 'diary)
+            (setq beg (+ beg 1 (length todos-categories-todo-label)
+                          2 (/ (length todos-categories-diary-label) 2))))
+           ((eq nonum 'done)
+            (setq beg (+ beg 1 (length todos-categories-todo-label)
+                          2 (length todos-categories-diary-label)
+                          2 (/ (length todos-categories-done-label) 2))))
+           ((eq nonum 'archived)
+            (setq beg (+ beg 1 (length todos-categories-todo-label)
+                          2 (length todos-categories-diary-label)
+                          2 (length todos-categories-done-label)
+                          2 (/ (length todos-categories-archived-label) 2)))))
+      (unless (= beg (+ opoint 7 (length str))) ; Don't highlight categories.
+       (setq end (+ beg 4))
+       (setq ovl (make-overlay beg end))
+       (overlay-put ovl 'face 'todos-sorted-column)))
+    (newline)))
 
-(defun todos-reevaluate-category-completions-files-defcustom ()
-  "Reevaluate defcustom of `todos-category-completions-files'.
-Called after adding or deleting a Todos file."
-  (eval (defcustom todos-category-completions-files nil
-  "List of files for building `todos-read-category' completions."
-         :type `(set ,@(mapcar (lambda (f) (list 'const f))
-                               (mapcar 'todos-short-file-name
-                                       (funcall todos-files-function))))
-         :group 'todos)))
+(defun todos-display-categories ()
+  "Prepare buffer for displaying table of categories and item counts."
+  (unless (eq major-mode 'todos-categories-mode)
+    (setq todos-global-current-todos-file
+         (or todos-current-todos-file
+             (todos-absolute-file-name todos-default-todos-file)))
+    (set-window-buffer (selected-window)
+                      (set-buffer (get-buffer-create todos-categories-buffer)))
+    (kill-all-local-variables)
+    (todos-categories-mode)
+    (let ((archive (member todos-current-todos-file todos-archives))
+         buffer-read-only)
+      (erase-buffer)
+      (insert (format (concat "Category counts for Todos "
+                             (if archive "archive" "file")
+                             " \"%s\".")
+                     (todos-short-file-name todos-current-todos-file)))
+      (newline 2)
+      ;; Make space for the column of category numbers.
+      (insert (make-string (+ 4 (length todos-categories-number-separator)) 32))
+      ;; Add the category and item count buttons (if this is the list of
+      ;; categories in an archive, show only done item counts).
+      (todos-insert-sort-button todos-categories-category-label)
+      (if archive
+         (progn
+           (insert (make-string 3 32))
+           (todos-insert-sort-button todos-categories-done-label))
+       (insert (make-string 3 32))
+       (todos-insert-sort-button todos-categories-todo-label)
+       (insert (make-string 2 32))
+       (todos-insert-sort-button todos-categories-diary-label)
+       (insert (make-string 2 32))
+       (todos-insert-sort-button todos-categories-done-label)
+       (insert (make-string 2 32))
+       (todos-insert-sort-button todos-categories-archived-label))
+      (newline 2))))
 
-(defun todos-reevaluate-filter-files-defcustom ()
-  "Reevaluate defcustom of `todos-filter-files'.
-Called after adding or deleting a Todos file."
-  (eval (defcustom todos-filter-files nil
-         "List of files for multifile item filtering."
-         :type `(set ,@(mapcar (lambda (f) (list 'const f))
-                               (mapcar 'todos-short-file-name
-                                       (funcall todos-files-function))))
-         :group 'todos)))
+(defun todos-update-categories-display (sortkey)
+  "Populate table of categories and sort by SORTKEY."
+  (let* ((cats0 todos-categories)
+        (cats (todos-sort cats0 sortkey))
+        (archive (member todos-current-todos-file todos-archives))
+        (todos-categories-category-number 0)
+        ;; Find start of Category button if we just entered Todos Categories
+        ;; mode.
+        (pt (if (eq (point) (point-max))
+                (save-excursion
+                  (forward-line -2)
+                  (goto-char (next-single-char-property-change
+                              (point) 'face nil (line-end-position))))))
+        (buffer-read-only))
+    (forward-line 2)
+    (delete-region (point) (point-max))
+    ;; Fill in the table with buttonized lines, each showing a category and
+    ;; its item counts.
+    (mapc (lambda (cat) (todos-insert-category-line cat sortkey))
+         (mapcar 'car cats))
+    (newline)
+    ;; Add a line showing item count totals.
+    (insert (make-string (+ 4 (length todos-categories-number-separator)) 32)
+           (todos-padded-string todos-categories-totals-label)
+           (mapconcat
+            (lambda (elt)
+              (concat
+               (make-string (1+ (/ (length (car elt)) 2)) 32)
+               (format "%3d" (nth (cdr elt) (todos-total-item-counts)))
+               ;; Add an extra space if label length is odd.
+               (when (cl-oddp (length (car elt))) " ")))
+            (if archive
+                (list (cons todos-categories-done-label 2))
+              (list (cons todos-categories-todo-label 0)
+                    (cons todos-categories-diary-label 1)
+                    (cons todos-categories-done-label 2)
+                    (cons todos-categories-archived-label 3)))
+            ""))
+    ;; Put cursor on Category button initially.
+    (if pt (goto-char pt))
+    (setq buffer-read-only t)))
 
 ;; -----------------------------------------------------------------------------
-;;; Category-level global variables and support functions
+;;; Searching and item filtering
 ;; -----------------------------------------------------------------------------
 
-(defun todos-category-number (cat)
-  "Return the number of category CAT in this Todos file.
-The buffer-local variable `todos-category-number' holds this
-number as its value."
-  (let ((categories (mapcar 'car todos-categories)))
-    (setq todos-category-number
-         ;; Increment by one, so that the highest priority category in Todos
-         ;; Categories mode is numbered one rather than zero.
-         (1+ (- (length categories)
-                (length (member cat categories)))))))
-
-(defun todos-current-category ()
-  "Return the name of the current category."
-  (car (nth (1- todos-category-number) todos-categories)))
-
-(defun todos-category-select ()
-  "Display the current category correctly."
-  (let ((name (todos-current-category))
-       cat-begin cat-end done-start done-sep-start done-end)
+(defun todos-search ()
+  "Search for a regular expression in this Todos file.
+The search runs through the whole file and encompasses all and
+only todo and done items; it excludes category names.  Multiple
+matches are shown sequentially, highlighted in `todos-search'
+face."
+  (interactive)
+  (let ((regex (read-from-minibuffer "Enter a search string (regexp): "))
+       (opoint (point))
+       matches match cat in-done ov mlen msg)
     (widen)
     (goto-char (point-min))
-    (re-search-forward
-     (concat "^" (regexp-quote (concat todos-category-beg name)) "$") nil t)
-    (setq cat-begin (1+ (line-end-position)))
-    (setq cat-end (if (re-search-forward
-                      (concat "^" (regexp-quote todos-category-beg)) nil t)
-                     (match-beginning 0)
-                   (point-max)))
-    (setq mode-line-buffer-identification
-         (funcall todos-mode-line-function name))
-    (narrow-to-region cat-begin cat-end)
-    (todos-prefix-overlays)
-    (goto-char (point-min))
-    (if (re-search-forward (concat "\n\\(" (regexp-quote todos-category-done)
-                                  "\\)") nil t)
-       (progn
-         (setq done-start (match-beginning 0))
-         (setq done-sep-start (match-beginning 1))
-         (setq done-end (match-end 0)))
-      (error "Category %s is missing todos-category-done string" name))
-    (if todos-show-done-only
-       (narrow-to-region (1+ done-end) (point-max))
-      (when (and todos-show-with-done
-                (re-search-forward todos-done-string-start nil t))
-       ;; Now we want to see the done items, so reset displayed end to end of
-       ;; done items.
-       (setq done-start cat-end)
-       ;; Make display overlay for done items separator string, unless there
-       ;; already is one.
-       (let* ((done-sep todos-done-separator)
-              (ov (progn (goto-char done-sep-start)
-                         (todos-get-overlay 'separator))))
-         (unless ov
-           (setq ov (make-overlay done-sep-start done-end))
-           (overlay-put ov 'todos 'separator)
-           (overlay-put ov 'display done-sep))))
-      (narrow-to-region (point-min) done-start)
-      ;; Loading this from todos-mode, or adding it to the mode hook, causes
-      ;; Emacs to hang in todos-item-start, at (looking-at todos-item-start).
-      (when todos-highlight-item
-       (require 'hl-line)
-       (hl-line-mode 1)))))
+    (while (not (eobp))
+      (setq match (re-search-forward regex nil t))
+      (goto-char (line-beginning-position))
+      (unless (or (equal (point) 1)
+                 (looking-at (concat "^" (regexp-quote todos-category-beg))))
+       (if match (push match matches)))
+      (forward-line))
+    (setq matches (reverse matches))
+    (if matches
+       (catch 'stop
+         (while matches
+           (setq match (pop matches))
+           (goto-char match)
+           (todos-item-start)
+           (when (looking-at todos-done-string-start)
+             (setq in-done t))
+           (re-search-backward (concat "^" (regexp-quote todos-category-beg)
+                                       "\\(.*\\)\n") nil t)
+           (setq cat (match-string-no-properties 1))
+           (todos-category-number cat)
+           (todos-category-select)
+           (if in-done
+               (unless todos-show-with-done (todos-toggle-view-done-items)))
+           (goto-char match)
+           (setq ov (make-overlay (- (point) (length regex)) (point)))
+           (overlay-put ov 'face 'todos-search)
+           (when matches
+             (setq mlen (length matches))
+             (if (todos-y-or-n-p
+                  (if (> mlen 1)
+                      (format "There are %d more matches; go to next match? "
+                              mlen)
+                    "There is one more match; go to it? "))
+                 (widen)
+               (throw 'stop (setq msg (if (> mlen 1)
+                                          (format "There are %d more matches."
+                                                  mlen)
+                                        "There is one more match."))))))
+         (setq msg "There are no more matches."))
+      (todos-category-select)
+      (goto-char opoint)
+      (message "No match for \"%s\"" regex))
+    (when msg
+      (if (todos-y-or-n-p (concat msg "\nUnhighlight matches? "))
+         (todos-clear-matches)
+       (message "You can unhighlight the matches later by typing %s"
+                (key-description (car (where-is-internal
+                                       'todos-clear-matches))))))))
 
-(defconst todos-category-beg "--==-- "
-  "String marking beginning of category (inserted with its name).")
+(defun todos-clear-matches ()
+  "Remove highlighting on matches found by todos-search."
+  (interactive)
+  (remove-overlays 1 (1+ (buffer-size)) 'face 'todos-search))
 
-(defconst todos-category-done "==--== DONE "
-  "String marking beginning of category's done items.")
+(defcustom todos-top-priorities-overrides nil
+  "List of rules specifying number of top priority items to show.
+These rules override `todos-top-priorities' on invocations of
+`\\[todos-filter-top-priorities]' and
+`\\[todos-filter-top-priorities-multifile]'.  Each rule is a list
+of the form (FILE NUM ALIST), where FILE is a member of
+`todos-files', NUM is a number specifying the default number of
+top priority items for each category in that file, and ALIST,
+when non-nil, consists of conses of a category name in FILE and a
+number specifying the default number of top priority items in
+that category, which overrides NUM.
 
-(defun todos-done-separator ()
-  "Return string used as value of variable `todos-done-separator'."
-  (let ((sep todos-done-separator-string))
-    (propertize (if (= 1 (length sep))
-                   ;; Until bug#2749 is fixed, if separator's length
-                   ;; is window-width and todos-wrap-lines is
-                   ;; non-nil, an indented empty line appears between
-                   ;; the separator and the first done item.
-                   ;; (make-string (window-width) (string-to-char sep))
-                   (make-string (1- (window-width)) (string-to-char sep))
-                 todos-done-separator-string)
-               'face 'todos-done-sep)))
+This variable should be set interactively by
+`\\[todos-set-top-priorities-in-file]' or
+`\\[todos-set-top-priorities-in-category]'."
+  :type 'sexp
+  :group 'todos-filtered)
 
-(defvar todos-done-separator (todos-done-separator)
-  "String used to visually separate done from not done items.
-Displayed as an overlay instead of `todos-category-done' when
-done items are shown.  Its value is determined by user option
-`todos-done-separator-string'.")
+(defcustom todos-top-priorities 1
+  "Default number of top priorities shown by `todos-filter-top-priorities'."
+  :type 'integer
+  :group 'todos-filtered)
 
-(defun todos-reset-done-separator (sep)
-  "Replace existing overlays of done items separator string SEP."
-  (save-excursion
-    (save-restriction
-      (widen)
-      (goto-char (point-min))
-      (while (re-search-forward
-             (concat "\n\\(" (regexp-quote todos-category-done) "\\)") nil t)
-       (let* ((beg (match-beginning 1))
-              (end (match-end 0))
-              (ov (progn (goto-char beg)
-                         (todos-get-overlay 'separator)))
-              (old-sep (when ov (overlay-get ov 'display)))
-              new-ov)
-         (when old-sep
-           (unless (string= old-sep sep)
-             (setq new-ov (make-overlay beg end))
-             (overlay-put new-ov 'todos 'separator)
-             (overlay-put new-ov 'display todos-done-separator)
-             (delete-overlay ov))))))))
+(defcustom todos-filter-files nil
+  "List of default files for multifile item filtering."
+  :type `(set ,@(mapcar (lambda (f) (list 'const f))
+                       (mapcar 'todos-short-file-name
+                               (funcall todos-files-function))))
+  :group 'todos-filtered)
 
-(defun todos-get-count (type &optional category)
-  "Return count of TYPE items in CATEGORY.
-If CATEGORY is nil, default to the current category."
-  (let* ((cat (or category (todos-current-category)))
-        (counts (cdr (assoc cat todos-categories)))
-        (idx (cond ((eq type 'todo) 0)
-                   ((eq type 'diary) 1)
-                   ((eq type 'done) 2)
-                   ((eq type 'archived) 3))))
-    (aref counts idx)))
+(defcustom todos-filter-done-items nil
+  "Non-nil to include done items when processing regexp filters.
+Done items from corresponding archive files are also included."
+  :type 'boolean
+  :group 'todos-filtered)
 
-(defun todos-update-count (type increment &optional category)
-  "Change count of TYPE items in CATEGORY by integer INCREMENT.
-With nil or omitted CATEGORY, default to the current category."
-  (let* ((cat (or category (todos-current-category)))
-        (counts (cdr (assoc cat todos-categories)))
-        (idx (cond ((eq type 'todo) 0)
-                   ((eq type 'diary) 1)
-                   ((eq type 'done) 2)
-                   ((eq type 'archived) 3))))
-    (aset counts idx (+ increment (aref counts idx)))))
+(defun todos-set-top-priorities-in-file ()
+  "Set number of top priorities for this file.
+See `todos-set-top-priorities' for more details."
+  (interactive)
+  (todos-set-top-priorities))
 
-(defun todos-set-categories ()
-  "Set `todos-categories' from the sexp at the top of the file."
-  ;; New archive files created by `todos-move-category' are empty, which would
-  ;; make the sexp test fail and raise an error, so in this case we skip it.
-  (unless (zerop (buffer-size))
-    (save-excursion
-      (save-restriction
-       (widen)
-       (goto-char (point-min))
-       (setq todos-categories
-             (if (looking-at "\(\(\"")
-                 (read (buffer-substring-no-properties
-                        (line-beginning-position)
-                        (line-end-position)))
-               (error "Invalid or missing todos-categories sexp")))))))
+(defun todos-set-top-priorities-in-category ()
+  "Set number of top priorities for this category.
+See `todos-set-top-priorities' for more details."
+  (interactive)
+  (todos-set-top-priorities t))
+
+(defun todos-filter-top-priorities (&optional arg)
+  "Display a list of top priority items from different categories.
+The categories can be any of those in the current Todos file.
 
-(defun todos-update-categories-sexp ()
-  "Update the `todos-categories' sexp at the top of the file."
-  (let (buffer-read-only)
-    (save-excursion
-      (save-restriction
-       (widen)
-       (goto-char (point-min))
-       (if (looking-at (concat "^" (regexp-quote todos-category-beg)))
-           (progn (newline) (goto-char (point-min)) ; Make space for sexp.
-                  (setq todos-categories (todos-make-categories-list t)))
-         (delete-region (line-beginning-position) (line-end-position)))
-       (prin1 todos-categories (current-buffer))))))
+With numerical prefix ARG show at most ARG top priority items
+from each category.  With `C-u' as prefix argument show the
+numbers of top priority items specified by category in
+`todos-top-priorities-overrides', if this has an entry for the file(s);
+otherwise show `todos-top-priorities' items per category in the
+file(s).  With no prefix argument, if a top priorities file for
+the current Todos file has previously been saved (see
+`todos-save-filtered-items-buffer'), visit this file; if there is
+no such file, build the list as with prefix argument `C-u'.
 
-(defun todos-make-categories-list (&optional force)
-  "Return an alist of Todos categories and their item counts.
-With non-nil argument FORCE parse the entire file to build the
-list; otherwise, get the value by reading the sexp at the top of
-the file."
-  (setq todos-categories nil)
-  (save-excursion
-    (save-restriction
-      (widen)
-      (goto-char (point-min))
-      (let (counts cat archive)
-       ;; If the file is a todo file and has archived items, identify the
-       ;; archive, in order to count its items.  But skip this with
-       ;; `todos-convert-legacy-files', since that converts filed items to
-       ;; archived items.
-       (when buffer-file-name   ; During conversion there is no file yet.
-         ;; If the file is an archive, it doesn't have an archive.
-         (unless (member (file-truename buffer-file-name)
-                         (funcall todos-files-function t))
-           (setq archive (concat (file-name-sans-extension
-                                  todos-current-todos-file) ".toda"))))
-       (while (not (eobp))
-         (cond ((looking-at (concat (regexp-quote todos-category-beg)
-                                    "\\(.*\\)\n"))
-                (setq cat (match-string-no-properties 1))
-                ;; Counts for each category: [todo diary done archive]
-                (setq counts (make-vector 4 0))
-                (setq todos-categories
-                      (append todos-categories (list (cons cat counts))))
-                ;; Add archived item count to the todo file item counts.
-                ;; Make sure to include newly created archives, e.g. due to
-                ;; todos-move-category.
-                (when (member archive (funcall todos-files-function t))
-                  (let ((archive-count 0))
-                    (with-current-buffer (find-file-noselect archive)
-                      (widen)
-                      (goto-char (point-min))
-                      (when (re-search-forward
-                             (concat "^" (regexp-quote todos-category-beg)
-                                     cat "$")
-                             (point-max) t)
-                        (forward-line)
-                        (while (not (or (looking-at
-                                         (concat
-                                          (regexp-quote todos-category-beg)
-                                          "\\(.*\\)\n"))
-                                        (eobp)))
-                          (when (looking-at todos-done-string-start)
-                            (setq archive-count (1+ archive-count)))
-                          (forward-line))))
-                    (todos-update-count 'archived archive-count cat))))
-               ((looking-at todos-done-string-start)
-                (todos-update-count 'done 1 cat))
-               ((looking-at (concat "^\\("
-                                    (regexp-quote diary-nonmarking-symbol)
-                                    "\\)?" todos-date-pattern))
-                (todos-update-count 'diary 1 cat)
-                (todos-update-count 'todo 1 cat))
-               ((looking-at (concat todos-date-string-start todos-date-pattern))
-                (todos-update-count 'todo 1 cat))
-               ;; If first line is todos-categories list, use it and end loop
-               ;; -- unless FORCEd to scan whole file.
-               ((bobp)
-                (unless force
-                  (setq todos-categories (read (buffer-substring-no-properties
-                                                (line-beginning-position)
-                                                (line-end-position))))
-                  (goto-char (1- (point-max))))))
-         (forward-line)))))
-  todos-categories)
+  The prefix ARG regulates how many top priorities from
+each category to show, as described above."
+  (interactive "P")
+  (todos-filter-items 'top arg))
 
-(defun todos-repair-categories-sexp ()
-  "Repair corrupt Todos categories sexp.
-This should only be needed as a consequence of careless manual
-editing or a bug in todos.el.
+(defun todos-filter-top-priorities-multifile (&optional arg)
+  "Display a list of top priority items from different categories.
+The categories are a subset of the categories in the files listed
+in `todos-filter-files', or if this nil, in the files chosen from
+a file selection dialog that pops up in this case.
 
-*Warning*: Calling this command restores the category order to
-the list element order in the Todos categories sexp, so any order
-changes made in Todos Categories mode will have to be made again."
-  (interactive)
-  (let ((todos-categories (todos-make-categories-list t)))
-    (todos-update-categories-sexp)))
+With numerical prefix ARG show at most ARG top priority items
+from each category in each file.  With `C-u' as prefix argument
+show the numbers of top priority items specified in
+`todos-top-priorities-overrides', if this is non-nil; otherwise show
+`todos-top-priorities' items per category.  With no prefix
+argument, if a top priorities file for the chosen Todos files
+exists (see `todos-save-filtered-items-buffer'), visit this file;
+if there is no such file, do the same as with prefix argument
+`C-u'."
+  (interactive "P")
+  (todos-filter-items 'top arg t))
 
-;; -----------------------------------------------------------------------------
-;;; Item-level global variables and support functions
-;; -----------------------------------------------------------------------------
+(defun todos-filter-diary-items (&optional arg)
+  "Display a list of todo diary items from different categories.
+The categories can be any of those in the current Todos file.
 
-(defconst todos-month-name-array
-  (vconcat calendar-month-name-array (vector "*"))
-  "Array of month names, in order.
-The final element is \"*\", indicating an unspecified month.")
+Called with no prefix ARG, if a diary items file for the current
+Todos file has previously been saved (see
+`todos-save-filtered-items-buffer'), visit this file; if there is
+no such file, build the list of diary items.  Called with a
+prefix argument, build the list even if there is a saved file of
+diary items."
+  (interactive "P")
+  (todos-filter-items 'diary arg))
 
-(defconst todos-month-abbrev-array
-  (vconcat calendar-month-abbrev-array (vector "*"))
-  "Array of abbreviated month names, in order.
-The final element is \"*\", indicating an unspecified month.")
+(defun todos-filter-diary-items-multifile (&optional arg)
+  "Display a list of todo diary items from different categories.
+The categories are a subset of the categories in the files listed
+in `todos-filter-files', or if this nil, in the files chosen from
+a file selection dialog that pops up in this case.
 
-(defconst todos-date-pattern
-  (let ((dayname (diary-name-pattern calendar-day-name-array nil t)))
-    (concat "\\(?5:" dayname "\\|"
-           (let ((dayname)
-                 (monthname (format "\\(?6:%s\\)" (diary-name-pattern
-                                                   todos-month-name-array
-                                                   todos-month-abbrev-array)))
-                 (month "\\(?7:[0-9]+\\|\\*\\)")
-                 (day "\\(?8:[0-9]+\\|\\*\\)")
-                 (year "-?\\(?9:[0-9]+\\|\\*\\)"))
-             (mapconcat 'eval calendar-date-display-form ""))
-           "\\)"))
-  "Regular expression matching a Todos date header.")
+Called with no prefix ARG, if a diary items file for the chosen
+Todos files has previously been saved (see
+`todos-save-filtered-items-buffer'), visit this file; if there is
+no such file, build the list of diary items.  Called with a
+prefix argument, build the list even if there is a saved file of
+diary items."
+  (interactive "P")
+  (todos-filter-items 'diary arg t))
 
-(defconst todos-nondiary-start (nth 0 todos-nondiary-marker)
-  "String inserted before item date to block diary inclusion.")
+(defun todos-filter-regexp-items (&optional arg)
+  "Prompt for a regular expression and display items that match it.
+The matches can be from any categories in the current Todos file
+and with non-nil option `todos-filter-done-items', can include
+not only todo items but also done items, including those in
+Archive files.
 
-(defconst todos-nondiary-end (nth 1 todos-nondiary-marker)
-  "String inserted after item date matching `todos-nondiary-start'.")
+Called with no prefix ARG, if a regexp items file for the current
+Todos file has previously been saved (see
+`todos-save-filtered-items-buffer'), visit this file; if there is
+no such file, build the list of regexp items.  Called with a
+prefix argument, build the list even if there is a saved file of
+regexp items."
+  (interactive "P")
+  (todos-filter-items 'regexp arg))
 
-;; By itself this matches anything, because of the `?'; however, it's only
-;; used in the context of `todos-date-pattern' (but Emacs Lisp lacks
-;; lookahead).
-(defconst todos-date-string-start
-  (concat "^\\(" (regexp-quote todos-nondiary-start) "\\|"
-         (regexp-quote diary-nonmarking-symbol) "\\)?")
-  "Regular expression matching part of item header before the date.")
+(defun todos-filter-regexp-items-multifile (&optional arg)
+  "Prompt for a regular expression and display items that match it.
+The matches can be from any categories in the files listed in
+`todos-filter-files', or if this nil, in the files chosen from a
+file selection dialog that pops up in this case.  With non-nil
+option `todos-filter-done-items', the matches can include not
+only todo items but also done items, including those in Archive
+files.
 
-(defconst todos-done-string-start
-  (concat "^\\[" (regexp-quote todos-done-string))
-  "Regular expression matching start of done item.")
+Called with no prefix ARG, if a regexp items file for the current
+Todos file has previously been saved (see
+`todos-save-filtered-items-buffer'), visit this file; if there is
+no such file, build the list of regexp items.  Called with a
+prefix argument, build the list even if there is a saved file of
+regexp items."
+  (interactive "P")
+  (todos-filter-items 'regexp arg t))
 
-(defconst todos-item-start (concat "\\(" todos-date-string-start "\\|"
-                                todos-done-string-start "\\)"
-                                todos-date-pattern)
-  "String identifying start of a Todos item.")
+(defun todos-find-filtered-items-file ()
+  "Choose a filtered items file and visit it."
+  (interactive)
+  (let ((files (directory-files todos-directory t "\.tod[rty]$" t))
+       falist file)
+    (dolist (f files)
+      (let ((type (cond ((equal (file-name-extension f) "todr") "regexp")
+                       ((equal (file-name-extension f) "todt") "top")
+                       ((equal (file-name-extension f) "tody") "diary"))))
+       (push (cons (concat (todos-short-file-name f) " (" type ")") f)
+             falist)))
+    (setq file (completing-read "Choose a filtered items file: "
+                               falist nil t nil nil (car falist)))
+    (setq file (cdr (assoc-string file falist)))
+    (find-file file)))
+
+(defun todos-go-to-source-item ()
+  "Display the file and category of the filtered item at point."
+  (interactive)
+  (let* ((str (todos-item-string))
+        (buf (current-buffer))
+        (res (todos-find-item str))
+        (found (nth 0 res))
+        (file (nth 1 res))
+        (cat (nth 2 res)))
+    (if (not found)
+       (message "Category %s does not contain this item." cat)
+      (kill-buffer buf)
+      (set-window-buffer (selected-window)
+                        (set-buffer (find-buffer-visiting file)))
+      (setq todos-current-todos-file file)
+      (setq todos-category-number (todos-category-number cat))
+      (let ((todos-show-with-done (if (or todos-filter-done-items
+                                         (eq (cdr found) 'done))
+                                     t
+                                   todos-show-with-done)))
+       (todos-category-select))
+      (goto-char (car found)))))
+
+(defvar todos-multiple-filter-files nil
+  "List of files selected from `todos-multiple-filter-files' widget.")
 
-(defun todos-item-start ()
-  "Move to start of current Todos item and return its position."
-  (unless (or
-          ;; Buffer is empty (invocation possible e.g. via todos-forward-item
-          ;; from todos-filter-items when processing category with no todo
-          ;; items).
-          (eq (point-min) (point-max))
-          ;; Point is on the empty line below category's last todo item...
-          (and (looking-at "^$")
-               (or (eobp)              ; ...and done items are hidden...
-                   (save-excursion     ; ...or done items are visible.
-                     (forward-line)
-                     (looking-at (concat "^"
-                                         (regexp-quote todos-category-done))))))
-          ;; Buffer is widened.
-          (looking-at (regexp-quote todos-category-beg)))
-    (goto-char (line-beginning-position))
-    (while (not (looking-at todos-item-start))
-      (forward-line -1))
-    (point)))
+(defvar todos-multiple-filter-files-widget nil
+  "Variable holding widget created by `todos-multiple-filter-files'.")
 
-(defun todos-item-end ()
-  "Move to end of current Todos item and return its position."
-  ;; Items cannot end with a blank line.
-  (unless (looking-at "^$")
-    (let* ((done (todos-done-item-p))
-          (to-lim nil)
-          ;; For todo items, end is before the done items section, for done
-          ;; items, end is before the next category.  If these limits are
-          ;; missing or inaccessible, end it before the end of the buffer.
-          (lim (if (save-excursion
-                     (re-search-forward
-                      (concat "^" (regexp-quote (if done
-                                                    todos-category-beg
-                                                  todos-category-done)))
-                      nil t))
-                   (progn (setq to-lim t) (match-beginning 0))
-                 (point-max))))
-      (when (bolp) (forward-char))     ; Find start of next item.
-      (goto-char (if (re-search-forward todos-item-start lim t)
-                    (match-beginning 0)
-                  (if to-lim lim (point-max))))
-      ;; For last todo item, skip back over the empty line before the done
-      ;; items section, else just back to the end of the previous line.
-      (backward-char (when (and to-lim (not done) (eq (point) lim)) 2))
-      (point))))
+(defun todos-multiple-filter-files ()
+  "Pop to a buffer with a widget for choosing multiple filter files."
+  (require 'widget)
+  (eval-when-compile
+    (require 'wid-edit))
+  (with-current-buffer (get-buffer-create "*Todos Filter Files*")
+    (pop-to-buffer (current-buffer))
+    (erase-buffer)
+    (kill-all-local-variables)
+    (widget-insert "Select files for generating the top priorities list.\n\n")
+    (setq todos-multiple-filter-files-widget
+         (widget-create
+          `(set ,@(mapcar (lambda (x) (list 'const x))
+                          (mapcar 'todos-short-file-name
+                                  (funcall todos-files-function))))))
+    (widget-insert "\n")
+    (widget-create 'push-button
+                  :notify (lambda (widget &rest ignore)
+                            (setq todos-multiple-filter-files 'quit)
+                            (quit-window t)
+                            (exit-recursive-edit))
+                  "Cancel")
+    (widget-insert "   ")
+    (widget-create 'push-button
+                  :notify (lambda (&rest ignore)
+                            (setq todos-multiple-filter-files
+                                  (mapcar (lambda (f)
+                                            (file-truename
+                                             (concat todos-directory
+                                                     f ".todo")))
+                                          (widget-value
+                                           todos-multiple-filter-files-widget)))
+                            (quit-window t)
+                            (exit-recursive-edit))
+                  "Apply")
+    (use-local-map widget-keymap)
+    (widget-setup))
+  (message "Click \"Apply\" after selecting files.")
+  (recursive-edit))
 
-(defun todos-item-string ()
-  "Return bare text of current item as a string."
-  (let ((opoint (point))
-       (start (todos-item-start))
-       (end (todos-item-end)))
-    (goto-char opoint)
-    (and start end (buffer-substring-no-properties start end))))
+(defconst todos-filtered-items-buffer "Todos filtered items"
+  "Initial name of buffer in Todos Filter Items mode.")
 
-(defun todos-forward-item (&optional count)
-  "Move point COUNT items down (by default, move down by one item)."
-  (let* ((not-done (not (or (todos-done-item-p) (looking-at "^$"))))
-        (start (line-end-position)))
-    (goto-char start)
-    (if (re-search-forward todos-item-start nil t (or count 1))
-       (goto-char (match-beginning 0))
-      (goto-char (point-max)))
-    ;; If points advances by one from a todo to a done item, go back
-    ;; to the space above todos-done-separator, since that is a
-    ;; legitimate place to insert an item.  But skip this space if
-    ;; count > 1, since that should only stop on an item.
-    (when (and not-done (todos-done-item-p) (not count))
-      ;; (if (or (not count) (= count 1))
-         (re-search-backward "^$" start t))));)
-    ;; The preceding sexp is insufficient when buffer is not narrowed,
-    ;; since there could be no done items in this category, so the
-    ;; search puts us on first todo item of next category.  Does this
-    ;; ever happen?  If so:
-    ;; (let ((opoint) (point))
-    ;;   (forward-line -1)
-    ;;   (when (or (not count) (= count 1))
-    ;;         (cond ((looking-at (concat "^" (regexp-quote todos-category-beg)))
-    ;;                (forward-line -2))
-    ;;               ((looking-at (concat "^" (regexp-quote todos-category-done)))
-    ;;                (forward-line -1))
-    ;;               (t
-    ;;                (goto-char opoint)))))))
+(defconst todos-top-priorities-buffer "Todos top priorities"
+  "Buffer type string for `todos-filter-items'.")
 
-(defun todos-backward-item (&optional count)
-  "Move point up to start of item with next higher priority.
-With positive numerical prefix COUNT, move point COUNT items
-upward.
+(defconst todos-diary-items-buffer "Todos diary items"
+  "Buffer type string for `todos-filter-items'.")
 
-If the category's done items are visible, this command called
-with a prefix argument only moves point to a higher item, e.g.,
-with point on the first done item and called with prefix 1, it
-moves to the last todo item; but if called with point on the
-first done item without a prefix argument, it moves point the the
-empty line above the done items separator."
-  (let* ((done (todos-done-item-p)))
-    (todos-item-start)
-    (unless (bobp)
-      (re-search-backward todos-item-start nil t (or count 1)))
-    ;; Unless this is a regexp filtered items buffer (which can contain
-    ;; intermixed todo and done items), if points advances by one from a
-    ;; done to a todo item, go back to the space above
-    ;; todos-done-separator, since that is a legitimate place to insert an
-    ;; item.  But skip this space if count > 1, since that should only
-    ;; stop on an item.
-    (when (and done (not (todos-done-item-p)) (not count)
-                                   ;(or (not count) (= count 1))
-              (not (equal (buffer-name) todos-regexp-items-buffer)))
-      (re-search-forward (concat "^" (regexp-quote todos-category-done))
-                        nil t)
-      (forward-line -1))))
+(defconst todos-regexp-items-buffer "Todos regexp items"
+  "Buffer type string for `todos-filter-items'.")
 
-(defun todos-remove-item ()
-  "Internal function called in editing, deleting or moving items."
-  (let* ((end (progn (todos-item-end) (1+ (point))))
-        (beg (todos-item-start))
-        (ov (todos-get-overlay 'prefix)))
-    (when ov (delete-overlay ov))
-    (delete-region beg end)))
+(defun todos-filter-items (filter &optional new multifile)
+  "Display a cross-categorial list of items filtered by FILTER.
+The values of FILTER can be `top' for top priority items, a cons
+of `top' and a number passed by the caller, `diary' for diary
+items, or `regexp' for items matching a regular expresion entered
+by the user.  The items can be from any categories in the current
+todo file or, with non-nil MULTIFILE, from several files.  If NEW
+is nil, visit an appropriate file containing the list of filtered
+items; if there is no such file, or with non-nil NEW, build the
+list and display it.
 
-(defun todos-diary-item-p ()
-  "Return non-nil if item at point has diary entry format."
-  (save-excursion
-    (when (todos-item-string)          ; Exclude empty lines.
-      (todos-item-start)
-      (not (looking-at (regexp-quote todos-nondiary-start))))))
+See the document strings of the commands
+`todos-filter-top-priorities', `todos-filter-diary-items',
+`todos-filter-regexp-items', and those of the corresponding
+multifile commands for further details."
+  (let* ((top (eq filter 'top))
+        (diary (eq filter 'diary))
+        (regexp (eq filter 'regexp))
+        (buf (cond (top todos-top-priorities-buffer)
+                   (diary todos-diary-items-buffer)
+                   (regexp todos-regexp-items-buffer)))
+        (flist (if multifile
+                   (or todos-filter-files
+                       (progn (todos-multiple-filter-files)
+                              todos-multiple-filter-files))
+                 (list todos-current-todos-file)))
+        (multi (> (length flist) 1))
+        (fname (if (equal flist 'quit)
+                   ;; Pressed `cancel' in t-m-f-f file selection dialog.
+                   (keyboard-quit)
+                 (concat todos-directory
+                         (mapconcat 'todos-short-file-name flist "-")
+                         (cond (top ".todt")
+                               (diary ".tody")
+                               (regexp ".todr")))))
+        (rxfiles (when regexp
+                   (directory-files todos-directory t ".*\\.todr$" t)))
+        (file-exists (or (file-exists-p fname) rxfiles)))
+    (cond ((and top new (natnump new))
+          (todos-filter-items-1 (cons 'top new) flist))
+         ((and (not new) file-exists)
+          (when (and rxfiles (> (length rxfiles) 1))
+            (let ((rxf (mapcar 'todos-short-file-name rxfiles)))
+              (setq fname (todos-absolute-file-name
+                           (completing-read "Choose a regexp items file: "
+                                            rxf) 'regexp))))
+          (find-file fname)
+          (todos-prefix-overlays)
+          (todos-check-filtered-items-file))
+         (t
+          (todos-filter-items-1 filter flist)))
+    (setq fname (replace-regexp-in-string "-" ", "
+                                         (todos-short-file-name fname)))
+    (rename-buffer (format (concat "%s for file" (if multi "s" "")
+                                  " \"%s\"") buf fname))))
 
-(defun todos-diary-goto-entry ()
-  "Jump to todo item included in Fancy Diary display.
-Helper function for `diary-goto-entry'."
-  (when (eq major-mode 'todos-mode)
-    (let ((opoint (point)))
-      (re-search-backward (concat "^" (regexp-quote todos-category-beg)
-                                 "\\(.*\\)\n") nil t)
-      (todos-category-number (match-string 1))
-      (todos-category-select)
-      (goto-char opoint))))
+(defun todos-filter-items-1 (filter file-list)
+  "Build a list of items by applying FILTER to FILE-LIST.
+Internal subroutine called by `todos-filter-items', which passes
+the values of FILTER and FILE-LIST."
+  (let ((num (if (consp filter) (cdr filter) todos-top-priorities))
+       (buf (get-buffer-create todos-filtered-items-buffer))
+       (multifile (> (length file-list) 1))
+       regexp fname bufstr cat beg end done)
+    (if (null file-list)
+       (user-error "No files have been chosen for filtering")
+      (with-current-buffer buf
+       (erase-buffer)
+       (kill-all-local-variables)
+       (todos-filtered-items-mode))
+      (when (eq filter 'regexp)
+       (setq regexp (read-string "Enter a regular expression: ")))
+      (save-current-buffer
+       (dolist (f file-list)
+         ;; Before inserting file contents into temp buffer, save a modified
+         ;; buffer visiting it.
+         (let ((bf (find-buffer-visiting f)))
+           (when (buffer-modified-p bf)
+             (with-current-buffer bf (save-buffer))))
+         (setq fname (todos-short-file-name f))
+         (with-temp-buffer
+           (when (and todos-filter-done-items (eq filter 'regexp))
+             ;; If there is a corresponding archive file for the
+             ;; Todos file, insert it first and add identifiers for
+             ;; todos-go-to-source-item.
+             (let ((arch (concat (file-name-sans-extension f) ".toda")))
+               (when (file-exists-p arch)
+                 (insert-file-contents arch)
+                 ;; Delete Todos archive file categories sexp.
+                 (delete-region (line-beginning-position)
+                                (1+ (line-end-position)))
+                 (save-excursion
+                   (while (not (eobp))
+                     (when (re-search-forward
+                            (concat (if todos-filter-done-items
+                                        (concat "\\(?:" todos-done-string-start
+                                                "\\|" todos-date-string-start
+                                                "\\)")
+                                      todos-date-string-start)
+                                    todos-date-pattern "\\(?: "
+                                    diary-time-regexp "\\)?"
+                                    (if todos-filter-done-items
+                                        "\\]"
+                                      (regexp-quote todos-nondiary-end)) "?")
+                            nil t)
+                       (insert "(archive) "))
+                     (forward-line))))))
+           (insert-file-contents f)
+           ;; Delete Todos file categories sexp.
+           (delete-region (line-beginning-position) (1+ (line-end-position)))
+           (let (fnum)
+             ;; Unless the number of top priorities to show was
+             ;; passed by the caller, the file-wide value from
+             ;; `todos-top-priorities-overrides', if non-nil, overrides
+             ;; `todos-top-priorities'.
+             (unless (consp filter)
+               (setq fnum (or (nth 1 (assoc f todos-top-priorities-overrides))
+                              todos-top-priorities)))
+             (while (re-search-forward
+                     (concat "^" (regexp-quote todos-category-beg)
+                             "\\(.+\\)\n") nil t)
+               (setq cat (match-string 1))
+               (let (cnum)
+                 ;; Unless the number of top priorities to show was
+                 ;; passed by the caller, the category-wide value
+                 ;; from `todos-top-priorities-overrides', if non-nil,
+                 ;; overrides a non-nil file-wide value from
+                 ;; `todos-top-priorities-overrides' as well as
+                 ;; `todos-top-priorities'.
+                 (unless (consp filter)
+                   (let ((cats (nth 2 (assoc f todos-top-priorities-overrides))))
+                     (setq cnum (or (cdr (assoc cat cats)) fnum))))
+                 (delete-region (match-beginning 0) (match-end 0))
+                 (setq beg (point))    ; First item in the current category.
+                 (setq end (if (re-search-forward
+                                (concat "^" (regexp-quote todos-category-beg))
+                                nil t)
+                               (match-beginning 0)
+                             (point-max)))
+                 (goto-char beg)
+                 (setq done
+                       (if (re-search-forward
+                            (concat "\n" (regexp-quote todos-category-done))
+                            end t)
+                           (match-beginning 0)
+                         end))
+                 (unless (and todos-filter-done-items (eq filter 'regexp))
+                   ;; Leave done items.
+                   (delete-region done end)
+                   (setq end done))
+                 (narrow-to-region beg end)    ; Process only current category.
+                 (goto-char (point-min))
+                 ;; Apply the filter.
+                 (cond ((eq filter 'diary)
+                        (while (not (eobp))
+                          (if (looking-at (regexp-quote todos-nondiary-start))
+                              (todos-remove-item)
+                            (todos-forward-item))))
+                       ((eq filter 'regexp)
+                        (while (not (eobp))
+                          (if (looking-at todos-item-start)
+                              (if (string-match regexp (todos-item-string))
+                                  (todos-forward-item)
+                                (todos-remove-item))
+                            ;; Kill lines that aren't part of a todo or done
+                            ;; item (empty or todos-category-done).
+                            (delete-region (line-beginning-position)
+                                           (1+ (line-end-position))))
+                          ;; If last todo item in file matches regexp and
+                          ;; there are no following done items,
+                          ;; todos-category-done string is left dangling,
+                          ;; because todos-forward-item jumps over it.
+                          (if (and (eobp)
+                                   (looking-back
+                                    (concat (regexp-quote todos-done-string)
+                                            "\n")))
+                              (delete-region (point) (progn
+                                                       (forward-line -2)
+                                                       (point))))))
+                       (t ; Filter top priority items.
+                        (setq num (or cnum fnum num))
+                        (unless (zerop num)
+                          (todos-forward-item num))))
+                 (setq beg (point))
+                 ;; Delete non-top-priority items.
+                 (unless (member filter '(diary regexp))
+                   (delete-region beg end))
+                 (goto-char (point-min))
+                 ;; Add file (if using multiple files) and category tags to
+                 ;; item.
+                 (while (not (eobp))
+                   (when (re-search-forward
+                          (concat (if todos-filter-done-items
+                                      (concat "\\(?:" todos-done-string-start
+                                              "\\|" todos-date-string-start
+                                              "\\)")
+                                    todos-date-string-start)
+                                  todos-date-pattern "\\(?: " diary-time-regexp
+                                  "\\)?" (if todos-filter-done-items
+                                             "\\]"
+                                           (regexp-quote todos-nondiary-end))
+                                  "?")
+                          nil t)
+                     (insert " [")
+                     (when (looking-at "(archive) ") (goto-char (match-end 0)))
+                     (insert (if multifile (concat fname ":") "") cat "]"))
+                   (forward-line))
+                 (widen)))
+               (setq bufstr (buffer-string))
+               (with-current-buffer buf
+                 (let (buffer-read-only)
+                   (insert bufstr)))))))
+      (set-window-buffer (selected-window) (set-buffer buf))
+      (todos-prefix-overlays)
+      (goto-char (point-min)))))
 
-(defun todos-done-item-p ()
-  "Return non-nil if item at point is a done item."
-  (save-excursion
-    (todos-item-start)
-    (looking-at todos-done-string-start)))
+(defun todos-set-top-priorities (&optional arg)
+  "Set number of top priorities shown by `todos-filter-top-priorities'.
+With non-nil ARG, set the number only for the current Todos
+category; otherwise, set the number for all categories in the
+current Todos file.
 
-(defun todos-done-item-section-p ()
-  "Return non-nil if point is in category's done items section."
-  (save-excursion
-    (or (re-search-backward (concat "^" (regexp-quote todos-category-done))
-                           nil t)
-       (progn (goto-char (point-min))
-              (looking-at todos-done-string-start)))))
+Calling this function via either of the commands
+`todos-set-top-priorities-in-file' or
+`todos-set-top-priorities-in-category' is the recommended way to
+set the user customizable option `todos-top-priorities-overrides'."
+  (let* ((cat (todos-current-category))
+        (file todos-current-todos-file)
+        (rules todos-top-priorities-overrides)
+        (frule (assoc-string file rules))
+        (crule (assoc-string cat (nth 2 frule)))
+        (crules (nth 2 frule))
+        (cur (or (if arg (cdr crule) (nth 1 frule))
+                 todos-top-priorities))
+        (prompt (if arg (concat "Number of top priorities in this category"
+                                " (currently %d): ")
+                  (concat "Default number of top priorities per category"
+                                " in this file (currently %d): ")))
+        (new -1)
+        nrule)
+    (while (< new 0)
+      (let ((cur0 cur))
+       (setq new (read-number (format prompt cur0))
+             prompt "Enter a non-negative number: "
+             cur0 nil)))
+    (setq nrule (if arg
+                   (append (delete crule crules) (list (cons cat new)))
+                 (append (list file new) (list crules))))
+    (setq rules (cons (if arg
+                         (list file cur nrule)
+                       nrule)
+                     (delete frule rules)))
+    (customize-save-variable 'todos-top-priorities-overrides rules)
+    (todos-prefix-overlays)))
 
-(defun todos-get-overlay (val)
-  "Return the overlay at point whose `todos' property has value VAL."
-  ;; Use overlays-in to find prefix overlays and check over two
-  ;; positions to find done separator overlay.
-  (let ((ovs (overlays-in (point) (1+ (point))))
-       ov)
-    (catch 'done
-      (while ovs
-       (setq ov (pop ovs))
-       (when (eq (overlay-get ov 'todos) val)
-         (throw 'done ov))))))
+(defun todos-find-item (str)
+  "Search for filtered item STR in its saved Todos file.
+Return the list (FOUND FILE CAT), where CAT and FILE are the
+item's category and file, and FOUND is a cons cell if the search
+succeeds, whose car is the start of the item in FILE and whose
+cdr is `done', if the item is now a done item, `changed', if its
+text was truncated or augmented or, for a top priority item, if
+its priority has changed, and `same' otherwise."
+  (string-match (concat (if todos-filter-done-items
+                           (concat "\\(?:" todos-done-string-start "\\|"
+                                   todos-date-string-start "\\)")
+                         todos-date-string-start)
+                       todos-date-pattern "\\(?: " diary-time-regexp "\\)?"
+                       (if todos-filter-done-items
+                           "\\]"
+                         (regexp-quote todos-nondiary-end)) "?"
+                       "\\(?4: \\[\\(?3:(archive) \\)?\\(?2:.*:\\)?"
+                       "\\(?1:.*\\)\\]\\).*$") str)
+  (let ((cat (match-string 1 str))
+       (file (match-string 2 str))
+       (archive (string= (match-string 3 str) "(archive) "))
+       (filcat (match-string 4 str))
+       (tpriority 1)
+       (tpbuf (save-match-data (string-match "top" (buffer-name))))
+       found)
+    (setq str (replace-match "" nil nil str 4))
+    (when tpbuf
+      ;; Calculate priority of STR wrt its category.
+      (save-excursion
+       (while (search-backward filcat nil t)
+           (setq tpriority (1+ tpriority)))))
+    (setq file (if file
+                  (concat todos-directory (substring file 0 -1)
+                          (if archive ".toda" ".todo"))
+                (if archive
+                    (concat (file-name-sans-extension
+                             todos-global-current-todos-file) ".toda")
+                  todos-global-current-todos-file)))
+    (find-file-noselect file)
+    (with-current-buffer (find-buffer-visiting file)
+      (save-restriction
+       (widen)
+       (goto-char (point-min))
+       (let ((beg (re-search-forward
+                   (concat "^" (regexp-quote (concat todos-category-beg cat))
+                           "$")
+                   nil t))
+             (done (save-excursion
+                     (re-search-forward
+                      (concat "^" (regexp-quote todos-category-done)) nil t)))
+             (end (save-excursion
+                    (or (re-search-forward
+                         (concat "^" (regexp-quote todos-category-beg))
+                         nil t)
+                        (point-max)))))
+         (setq found (when (search-forward str end t)
+                       (goto-char (match-beginning 0))))
+         (when found
+           (setq found
+                 (cons found (if (> (point) done)
+                                 'done
+                               (let ((cpriority 1))
+                                 (when tpbuf
+                                   (save-excursion
+                                     ;; Not top item in category.
+                                     (while (> (point) (1+ beg))
+                                       (let ((opoint (point)))
+                                         (todos-backward-item)
+                                         ;; Can't move backward beyond
+                                         ;; first item in file.
+                                         (unless (= (point) opoint)
+                                           (setq cpriority (1+ cpriority)))))))
+                                 (if (and (= tpriority cpriority)
+                                          ;; Proper substring is not the same.
+                                          (string= (todos-item-string)
+                                                   str))
+                                     'same
+                                   'changed)))))))))
+      (list found file cat)))
 
-(defun todos-marked-item-p ()
-  "Non-nil if this item begins with `todos-item-mark'.
-In that case, return the item's prefix overlay."
-  (let* ((ov (todos-get-overlay 'prefix))
-        ;; If an item insertion command is called on a Todos file
-        ;; before it is visited, it has no prefix overlays yet, so
-        ;; check for this.
-        (pref (when ov (overlay-get ov 'before-string)))
-        (marked (when pref
-                  (string-match (concat "^" (regexp-quote todos-item-mark))
-                                pref))))
-    (when marked ov)))
+(defun todos-check-filtered-items-file ()
+  "Check if filtered items file is up to date and a show suitable message."
+  ;; (catch 'old
+  (let ((count 0))
+    (while (not (eobp))
+      (let* ((item (todos-item-string))
+            (found (car (todos-find-item item))))
+       (unless (eq (cdr found) 'same)
+         (save-excursion
+           (overlay-put (make-overlay (todos-item-start) (todos-item-end))
+                        'face 'todos-search))
+         (setq count (1+ count))))
+         ;; (throw 'old (message "The marked item is not up to date.")))
+      (todos-forward-item))
+    (if (zerop count)
+       (message "Filtered items file is up to date.")
+      (message (concat "The highlighted item" (if (= count 1) " is " "s are ")
+                      "not up to date."
+                      ;; "\nType <return> on item for details."
+                      )))))
 
-(defun todos-insert-with-overlays (item)
-  "Insert ITEM at point and update prefix/priority number overlays."
-  (todos-item-start)
-  ;; Insertion pushes item down but not its prefix overlay.  When the
-  ;; overlay includes a mark, this would now mark the inserted ITEM,
-  ;; so move it to the pushed down item.
-  (let ((ov (todos-get-overlay 'prefix))
-       (marked (todos-marked-item-p)))
-    (insert item "\n")
-    (when marked (move-overlay ov (point) (point))))
-  (todos-backward-item)
-  (todos-prefix-overlays))
+(defun todos-filter-items-filename ()
+  "Return absolute file name for saving this Filtered Items buffer."
+  (let ((bufname (buffer-name)))
+    (string-match "\"\\([^\"]+\\)\"" bufname)
+    (let* ((filename-str (substring bufname (match-beginning 1) (match-end 1)))
+          (filename-base (replace-regexp-in-string ", " "-" filename-str))
+          (top-priorities (string-match "top priorities" bufname))
+          (diary-items (string-match "diary items" bufname))
+          (regexp-items (string-match "regexp items" bufname)))
+      (when regexp-items
+       (let ((prompt (concat "Enter a short identifying string"
+                             " to make this file name unique: ")))
+         (setq filename-base (concat filename-base "-" (read-string prompt)))))
+      (concat todos-directory filename-base
+             (cond (top-priorities ".todt")
+                   (diary-items ".tody")
+                   (regexp-items ".todr"))))))
 
-(defun todos-prefix-overlays ()
-  "Update the prefix overlays of the current category's items.
-The overlay's value is the string `todos-prefix' or with non-nil
-`todos-number-prefix' an integer in the sequence from 1 to
-the number of todo or done items in the category indicating the
-item's priority.  Todo and done items are numbered independently
-of each other."
-  (let ((num 0)
-       (cat-tp (or (cdr (assoc-string
-                         (todos-current-category)
-                         (nth 2 (assoc-string todos-current-todos-file
-                                              todos-top-priorities-overrides))))
-                   todos-top-priorities))
-       done prefix)
-    (save-excursion
-      (goto-char (point-min))
-      (while (not (eobp))
-       (when (or (todos-date-string-matcher (line-end-position))
-                 (todos-done-string-matcher (line-end-position)))
-         (goto-char (match-beginning 0))
-         (setq num (1+ num))
-         ;; Reset number to 1 for first done item.
-         (when (and (eq major-mode 'todos-mode)
-                    (looking-at todos-done-string-start)
-                    (looking-back (concat "^"
-                                          (regexp-quote todos-category-done)
-                                          "\n")))
-           (setq num 1
-                 done t))
-         (setq prefix (concat (propertize
-                               (if todos-number-prefix
-                                   (number-to-string num)
-                                 todos-prefix)
-                               'face
-                               ;; Prefix of top priority items has a
-                               ;; distinct face in Todos mode.
-                               (if (and (eq major-mode 'todos-mode)
-                                        (not done)
-                                        (<= num cat-tp))
-                                   'todos-top-priority
-                                 'todos-prefix-string))
-                              " "))
-         (let ((ov (todos-get-overlay 'prefix))
-               (marked (todos-marked-item-p)))
-           ;; Prefix overlay must be at a single position so its
-           ;; bounds aren't changed when (re)moving an item.
-           (unless ov (setq ov (make-overlay (point) (point))))
-           (overlay-put ov 'todos 'prefix)
-           (overlay-put ov 'before-string (if marked
-                                              (concat todos-item-mark prefix)
-                                            prefix))))
-       (forward-line)))))
+(defun todos-save-filtered-items-buffer ()
+  "Save current Filtered Items buffer to a file.
+If the file already exists, overwrite it only on confirmation."
+  (let ((filename (or (buffer-file-name) (todos-filter-items-filename))))
+    (write-file filename t)))
 
 ;; -----------------------------------------------------------------------------
-;;; Generation of item insertion commands and key bindings
+;;; Printing Todos buffers
 ;; -----------------------------------------------------------------------------
 
-;; These two powerset definitions are adaptations of code published at
-;; http://rosettacode.org, whose content is licensed under GFDL 1.2.
-;; The recursive definition is a slight reformulation of
-;; http://rosettacode.org/wiki/Power_set#Common_Lisp.  The iterative
-;; definition is my Elisp implementation of
-;; http://rosettacode.org/wiki/Power_set#C.  Can either of these be
-;; included in Emacs, or is there no need to concerned about copyright
-;; here?
-
-;; (defun todos-powerset (list)
-;;   "Return the powerset of LIST."
-;;   (cond ((null list)
-;;      (list nil))
-;;     (t
-;;      (let ((recur (todos-powerset-recursive (cdr list)))
-;;            pset)
-;;        (dolist (elt recur pset)
-;;          (push (cons (car list) elt) pset))
-;;        (append pset recur)))))
-
-(defun todos-powerset (list)
-  "Return the powerset of LIST."
-  (let ((card (expt 2 (length list)))
-        pset elt)
-    (dotimes (n card)
-      (let ((i n)
-           (l list))
-       (while (not (zerop i))
-         (let ((arg (pop l)))
-           (when (cl-oddp i)
-             (setq elt (append elt (list arg))))
-           (setq i (/ i 2))))
-       (setq pset (append pset (list elt)))
-       (setq elt nil)))
-    pset))
-
-(defun todos-gen-arglists (arglist)
-  "Return list of lists of non-nil atoms produced from ARGLIST.
-The elements of ARGLIST may be atoms or lists."
-  (let (arglists)
-    (while arglist
-      (let ((arg (pop arglist)))
-       (cond ((symbolp arg)
-              (setq arglists (if arglists
-                                 (mapcar (lambda (l) (push arg l)) arglists)
-                               (list (push arg arglists)))))
-             ((listp arg)
-              (setq arglists
-                    (mapcar (lambda (a)
-                              (if (= 1 (length arglists))
-                                  (apply (lambda (l) (push a l)) arglists)
-                                (mapcar (lambda (l) (push a l)) arglists)))
-                            arg))))))
-    (setq arglists (mapcar 'reverse (apply 'append (mapc 'car arglists))))))
-
-(defvar todos-insertion-commands-args-genlist
-  '(diary nonmarking (calendar date dayname) time (here region))
-  "Generator list for argument lists of item insertion commands.")
+(defcustom todos-print-buffer-function 'ps-print-buffer-with-faces
+  "Function called by the command `todos-print-buffer'."
+  :type 'symbol
+  :group 'todos)
 
-(defvar todos-insertion-commands-args
-  (let ((argslist (todos-gen-arglists todos-insertion-commands-args-genlist))
-       res new)
-    (setq res (cl-remove-duplicates
-              (apply 'append (mapcar 'todos-powerset argslist)) :test 'equal))
-    (dolist (l res)
-      (unless (= 5 (length l))
-       (let ((v (make-vector 5 nil)) elt)
-         (while l
-           (setq elt (pop l))
-           (cond ((eq elt 'diary)
-                  (aset v 0 elt))
-                 ((eq elt 'nonmarking)
-                  (aset v 1 elt))
-                 ((or (eq elt 'calendar)
-                      (eq elt 'date)
-                      (eq elt 'dayname))
-                  (aset v 2 elt))
-                 ((eq elt 'time)
-                  (aset v 3 elt))
-                 ((or (eq elt 'here)
-                      (eq elt 'region))
-                  (aset v 4 elt))))
-         (setq l (append v nil))))
-      (setq new (append new (list l))))
-    new)
-  "List of all argument lists for Todos item insertion commands.")
+(defvar todos-print-buffer "*Todos Print*"
+  "Name of buffer containing printable Todos text.")
 
-(defun todos-insertion-command-name (arglist)
-  "Generate Todos item insertion command name from ARGLIST."
-  (replace-regexp-in-string
-   "-\\_>" ""
-   (replace-regexp-in-string
-    "-+" "-"
-    ;; (concat "todos-item-insert-"
-    (concat "todos-insert-item-"
-           (mapconcat (lambda (e) (if e (symbol-name e))) arglist "-")))))
+(defun todos-print-buffer (&optional to-file)
+  "Produce a printable version of the current Todos buffer.
+This converts overlays and soft line wrapping and, depending on
+the value of `todos-print-buffer-function', includes faces.  With
+non-nil argument TO-FILE write the printable version to a file;
+otherwise, send it to the default printer."
+  (interactive)
+  (let ((buf todos-print-buffer)
+       (header (cond
+                ((eq major-mode 'todos-mode)
+                 (concat "Todos File: "
+                         (todos-short-file-name todos-current-todos-file)
+                         "\nCategory: " (todos-current-category)))
+                ((eq major-mode 'todos-filtered-items-mode)
+                 (buffer-name))))
+       (prefix (propertize (concat todos-prefix " ")
+                           'face 'todos-prefix-string))
+       (num 0)
+       (fill-prefix (make-string todos-indent-to-here 32))
+       (content (buffer-string))
+       file)
+    (with-current-buffer (get-buffer-create buf)
+      (insert content)
+      (goto-char (point-min))
+      (while (not (eobp))
+       (let ((beg (point))
+             (end (save-excursion (todos-item-end))))
+         (when todos-number-prefix
+           (setq num (1+ num))
+           (setq prefix (propertize (concat (number-to-string num) " ")
+                                    'face 'todos-prefix-string)))
+         (insert prefix)
+         (fill-region beg end))
+       ;; Calling todos-forward-item infloops at todos-item-start due to
+       ;; non-overlay prefix, so search for item start instead.
+       (if (re-search-forward todos-item-start nil t)
+           (beginning-of-line)
+         (goto-char (point-max))))
+      (if (re-search-backward (concat "^" (regexp-quote todos-category-done))
+                             nil t)
+         (replace-match todos-done-separator))
+      (goto-char (point-min))
+      (insert header)
+      (newline 2)
+      (if to-file
+         (let ((file (read-file-name "Print to file: ")))
+           (funcall todos-print-buffer-function file))
+       (funcall todos-print-buffer-function)))
+    (kill-buffer buf)))
 
-(defvar todos-insertion-commands-names
-  (mapcar (lambda (l)
-          (todos-insertion-command-name l))
-         todos-insertion-commands-args)
-  "List of names of Todos item insertion commands.")
+(defun todos-print-buffer-to-file ()
+  "Save printable version of this Todos buffer to a file."
+  (interactive)
+  (todos-print-buffer t))
 
-(defmacro todos-define-insertion-command (&rest args)
-  "Generate item insertion command definitions from ARGS."
-  (let ((name (intern (todos-insertion-command-name args)))
-       (arg0 (nth 0 args))
-       (arg1 (nth 1 args))
-       (arg2 (nth 2 args))
-       (arg3 (nth 3 args))
-       (arg4 (nth 4 args)))
-    `(defun ,name (&optional arg &rest args)
-       "Todos item insertion command generated from ARGS.
-For descriptions of the individual arguments, their values, and
-their relation to key bindings, see `todos-basic-insert-item'."
-       (interactive (list current-prefix-arg))
-       (todos-basic-insert-item arg ',arg0 ',arg1 ',arg2 ',arg3 ',arg4))))
+;; -----------------------------------------------------------------------------
+;;; Legacy Todo mode files
+;; -----------------------------------------------------------------------------
 
-(defvar todos-insertion-commands
-  (mapcar (lambda (c)
-           (eval `(todos-define-insertion-command ,@c)))
-         todos-insertion-commands-args)
-  "List of Todos item insertion commands.")
+(defcustom todos-todo-mode-date-time-regexp
+  (concat "\\(?1:[0-9]\\{4\\}\\)-\\(?2:[0-9]\\{2\\}\\)-"
+         "\\(?3:[0-9]\\{2\\}\\) \\(?4:[0-9]\\{2\\}:[0-9]\\{2\\}\\)")
+  "Regexp matching legacy todo-mode.el item date-time strings.
+In order for `todos-convert-legacy-files' to correctly convert this
+string to the current Todos format, the regexp must contain four
+explicitly numbered groups (see `(elisp) Regexp Backslash'),
+where group 1 matches a string for the year, group 2 a string for
+the month, group 3 a string for the day and group 4 a string for
+the time.  The default value converts date-time strings built
+using the default value of `todo-time-string-format' from
+todo-mode.el."
+  :type 'regexp
+  :group 'todos)
 
-(defvar todos-insertion-commands-arg-key-list
-  '(("diary" "y" "yy")
-    ("nonmarking" "k" "kk")
-    ("calendar" "c" "cc")
-    ("date" "d" "dd")
-    ("dayname" "n" "nn")
-    ("time" "t" "tt")
-    ("here" "h" "h")
-    ("region" "r" "r"))
-  "List of mappings of insertion command arguments to key sequences.")
+(defun todos-convert-legacy-date-time ()
+  "Return converted date-time string.
+Helper function for `todos-convert-legacy-files'."
+  (let* ((year (match-string 1))
+        (month (match-string 2))
+        (monthname (calendar-month-name (string-to-number month) t))
+        (day (match-string 3))
+        (time (match-string 4))
+        dayname)
+    (replace-match "")
+    (insert (mapconcat 'eval calendar-date-display-form "")
+           (when time (concat " " time)))))
 
-(defun todos-insertion-key-bindings (map)
-  "Generate key binding definitions for item insertion keymap MAP."
-  (dolist (c todos-insertion-commands)
-    (let* ((key "")
-          (cname (symbol-name c)))
-      (mapc (lambda (l)
-             (let ((arg (nth 0 l))
-                   (key1 (nth 1 l))
-                   (key2 (nth 2 l)))
-               (if (string-match (concat (regexp-quote arg) "\\_>") cname)
-                   (setq key (concat key key2)))
-               (if (string-match (concat (regexp-quote arg) ".+") cname)
-                   (setq key (concat key key1)))))
-           todos-insertion-commands-arg-key-list)
-      (if (string-match (concat (regexp-quote "todos-insert-item") "\\_>") cname)
-         (setq key (concat key "i")))
-      (define-key map key c))))
+(defun todos-convert-legacy-files ()
+  "Convert legacy Todo files to the current Todos format.
+The files `todo-file-do' and `todo-file-done' are converted and
+saved (the latter as a Todos Archive file) with a new name in
+`todos-directory'.  See also the documentation string of
+`todos-todo-mode-date-time-regexp' for further details."
+  (interactive)
+  (eval-when-compile (require 'todo-mode))
+  ;; Convert `todo-file-do'.
+  (if (file-exists-p todo-file-do)
+      (let ((default "todo-do-conv")
+           file archive-sexp)
+       (with-temp-buffer
+         (insert-file-contents todo-file-do)
+         (let ((end (search-forward ")" (line-end-position) t))
+               (beg (search-backward "(" (line-beginning-position) t)))
+           (setq todo-categories
+                 (read (buffer-substring-no-properties beg end))))
+         (todo-mode)
+         (delete-region (line-beginning-position) (1+ (line-end-position)))
+         (while (not (eobp))
+           (cond
+            ((looking-at (regexp-quote (concat todo-prefix todo-category-beg)))
+             (replace-match todos-category-beg))
+            ((looking-at (regexp-quote todo-category-end))
+             (replace-match ""))
+            ((looking-at (regexp-quote (concat todo-prefix " "
+                                               todo-category-sep)))
+             (replace-match todos-category-done))
+            ((looking-at (concat (regexp-quote todo-prefix) " "
+                                 todos-todo-mode-date-time-regexp " "
+                                 (regexp-quote todo-initials) ":"))
+             (todos-convert-legacy-date-time)))
+           (forward-line))
+         (setq file (concat todos-directory
+                            (read-string
+                             (format "Save file as (default \"%s\"): " default)
+                             nil nil default)
+                            ".todo"))
+         (write-region (point-min) (point-max) file nil 'nomessage nil t))
+       (with-temp-buffer
+         (insert-file-contents file)
+         (let ((todos-categories (todos-make-categories-list t)))
+           (todos-update-categories-sexp))
+         (write-region (point-min) (point-max) file nil 'nomessage))
+       ;; Convert `todo-file-done'.
+       (when (file-exists-p todo-file-done)
+         (with-temp-buffer
+           (insert-file-contents todo-file-done)
+           (let ((beg (make-marker))
+                 (end (make-marker))
+                 cat cats comment item)
+             (while (not (eobp))
+               (when (looking-at todos-todo-mode-date-time-regexp)
+                 (set-marker beg (point))
+                 (todos-convert-legacy-date-time)
+                 (set-marker end (point))
+                 (goto-char beg)
+                 (insert "[" todos-done-string)
+                 (goto-char end)
+                 (insert "]")
+                 (forward-char)
+                 (when (looking-at todos-todo-mode-date-time-regexp)
+                   (todos-convert-legacy-date-time))
+                 (when (looking-at (concat " "
+                                           (regexp-quote todo-initials) ":"))
+                   (replace-match "")))
+               (if (re-search-forward
+                    (concat "^" todos-todo-mode-date-time-regexp) nil t)
+                   (goto-char (match-beginning 0))
+                 (goto-char (point-max)))
+               (backward-char)
+               (when (looking-back "\\[\\([^][]+\\)\\]")
+                 (setq cat (match-string 1))
+                 (goto-char (match-beginning 0))
+                 (replace-match ""))
+               ;; If the item ends with a non-comment parenthesis not
+               ;; followed by a period, we lose (but we inherit that problem
+               ;; from todo-mode.el).
+               (when (looking-back "(\\(.*\\)) ")
+                 (setq comment (match-string 1))
+                 (replace-match "")
+                 (insert "[" todos-comment-string ": " comment "]"))
+               (set-marker end (point))
+               (if (member cat cats)
+                   ;; If item is already in its category, leave it there.
+                   (unless (save-excursion
+                             (re-search-backward
+                              (concat "^" (regexp-quote todos-category-beg)
+                                      "\\(.*\\)$") nil t)
+                             (string= (match-string 1) cat))
+                     ;; Else move it to its category.
+                     (setq item (buffer-substring-no-properties beg end))
+                     (delete-region beg (1+ end))
+                     (set-marker beg (point))
+                     (re-search-backward
+                      (concat "^"
+                              (regexp-quote (concat todos-category-beg cat))
+                              "$")
+                      nil t)
+                     (forward-line)
+                     (if (re-search-forward
+                          (concat "^" (regexp-quote todos-category-beg)
+                                  "\\(.*\\)$") nil t)
+                         (progn (goto-char (match-beginning 0))
+                                (newline)
+                                (forward-line -1))
+                       (goto-char (point-max)))
+                     (insert item "\n")
+                     (goto-char beg))
+                 (push cat cats)
+                 (goto-char beg)
+                 (insert todos-category-beg cat "\n\n" todos-category-done "\n"))
+               (forward-line))
+             (set-marker beg nil)
+             (set-marker end nil))
+           (setq file (concat (file-name-sans-extension file) ".toda"))
+           (write-region (point-min) (point-max) file nil 'nomessage nil t))
+         (with-temp-buffer
+           (insert-file-contents file)
+           (let ((todos-categories (todos-make-categories-list t)))
+             (todos-update-categories-sexp))
+           (write-region (point-min) (point-max) file nil 'nomessage)
+           (setq archive-sexp (read (buffer-substring-no-properties
+                                     (line-beginning-position)
+                                     (line-end-position)))))
+         (setq file (concat (file-name-sans-extension file) ".todo"))
+         ;; Update categories sexp of converted Todos file again, adding
+         ;; counts of archived items.
+         (with-temp-buffer
+           (insert-file-contents file)
+           (let ((sexp (read (buffer-substring-no-properties
+                              (line-beginning-position)
+                              (line-end-position)))))
+             (dolist (cat sexp)
+               (let ((archive-cat (assoc (car cat) archive-sexp)))
+                 (if archive-cat
+                     (aset (cdr cat) 3 (aref (cdr archive-cat) 2)))))
+             (delete-region (line-beginning-position) (line-end-position))
+             (prin1 sexp (current-buffer)))
+           (write-region (point-min) (point-max) file nil 'nomessage)))
+       (todos-reevaluate-filelist-defcustoms)
+       (message "Format conversion done."))
+    (user-error "No legacy Todo file exists")))
 
 ;; -----------------------------------------------------------------------------
-;;; Todos minibuffer completion
+;;; Utility functions for Todos files, categories and items
 ;; -----------------------------------------------------------------------------
 
-(defun todos-category-completions (&optional archive)
-  "Return a list of completions for `todos-read-category'.
-Each element of the list is a cons of a category name and the
-file or list of files (as short file names) it is in.  The files
-are either the current (or if there is none, the default) todo
-file plus the files listed in `todos-category-completions-files',
-or, with non-nil ARCHIVE, the current archive file."
-  (let* ((curfile (or todos-current-todos-file
-                     (and todos-show-current-file
-                          todos-global-current-todos-file)
-                     (todos-absolute-file-name todos-default-todos-file)))
-        (files (or (unless archive
-                     (mapcar 'todos-absolute-file-name
-                             todos-category-completions-files))
-                   (list curfile)))
-        listall listf)
-    ;; If file was just added, it has no category completions.
-    (unless (zerop (buffer-size (find-buffer-visiting curfile)))
-      (unless (member curfile todos-archives)
-       (add-to-list 'files curfile))
-      (dolist (f files listall)
-       (with-current-buffer (find-file-noselect f 'nowarn)
-         ;; Ensure category is properly displayed in case user
-         ;; switches to file via a non-Todos command.  And if done
-         ;; items in category are visible, keep them visible.
-         (let ((done todos-show-with-done))
-           (when (> (buffer-size) (- (point-max) (point-min)))
-             (save-excursion
-               (goto-char (point-min))
-               (setq done (re-search-forward todos-done-string-start nil t))))
-           (let ((todos-show-with-done done))
-             (save-excursion (todos-category-select))))
-         (save-excursion
-           (save-restriction
-             (widen)
-             (goto-char (point-min))
-             (setq listf (read (buffer-substring-no-properties
-                                (line-beginning-position)
-                                (line-end-position)))))))
-       (mapc (lambda (elt) (let* ((cat (car elt))
-                                  (la-elt (assoc cat listall)))
-                             (if la-elt
-                                 (setcdr la-elt (append (list (cdr la-elt))
-                                                        (list f)))
-                               (push (cons cat f) listall))))
-             listf)))))
+(defun todos-absolute-file-name (name &optional type)
+  "Return the absolute file name of short Todos file NAME.
+With TYPE `archive' or `top' return the absolute file name of the
+short Todos Archive or Top Priorities file name, respectively."
+  ;; NOP if there is no Todos file yet (i.e. don't concatenate nil).
+  (when name
+    (file-truename
+     (concat todos-directory name
+            (cond ((eq type 'archive) ".toda")
+                  ((eq type 'top) ".todt")
+                  ((eq type 'diary) ".tody")
+                  ((eq type 'regexp) ".todr")
+                  (t ".todo"))))))
 
-(defun todos-read-file-name (prompt &optional archive mustmatch)
-  "Choose and return the name of a Todos file, prompting with PROMPT.
+(defun todos-category-number (cat)
+  "Return the number of category CAT in this Todos file.
+The buffer-local variable `todos-category-number' holds this
+number as its value."
+  (let ((categories (mapcar 'car todos-categories)))
+    (setq todos-category-number
+         ;; Increment by one, so that the highest priority category in Todos
+         ;; Categories mode is numbered one rather than zero.
+         (1+ (- (length categories)
+                (length (member cat categories)))))))
 
-Show completions with TAB or SPC; the names are shown in short
-form but the absolute truename is returned.  With non-nil ARCHIVE
-return the absolute truename of a Todos archive file.  With non-nil
-MUSTMATCH the name of an existing file must be chosen;
-otherwise, a new file name is allowed."
-  (let* ((completion-ignore-case todos-completion-ignore-case)
-        (files (mapcar 'todos-short-file-name
-                       (if archive todos-archives todos-files)))
-        (file (completing-read prompt files nil mustmatch nil nil
-                               (if files
-                                   ;; If user hit RET without
-                                   ;; choosing a file, default to
-                                   ;; current or default file.
-                                   (todos-short-file-name
-                                    (or todos-current-todos-file
-                                        (and todos-show-current-file
-                                             todos-global-current-todos-file)
-                                        (todos-absolute-file-name
-                                         todos-default-todos-file)))
-                                 ;; Trigger prompt for initial file.
-                                 ""))))
-    (unless (file-exists-p todos-directory)
-      (make-directory todos-directory))
-    (unless mustmatch
-      (setq file (todos-validate-name file 'file)))
-    (setq file (file-truename (concat todos-directory file
-                                     (if archive ".toda" ".todo"))))))
+(defun todos-current-category ()
+  "Return the name of the current category."
+  (car (nth (1- todos-category-number) todos-categories)))
 
-(defun todos-read-category (prompt &optional match-type file)
-  "Choose and return a category name, prompting with PROMPT.
-Show completions for existing categories with TAB or SPC.
+(defun todos-category-select ()
+  "Display the current category correctly."
+  (let ((name (todos-current-category))
+       cat-begin cat-end done-start done-sep-start done-end)
+    (widen)
+    (goto-char (point-min))
+    (re-search-forward
+     (concat "^" (regexp-quote (concat todos-category-beg name)) "$") nil t)
+    (setq cat-begin (1+ (line-end-position)))
+    (setq cat-end (if (re-search-forward
+                      (concat "^" (regexp-quote todos-category-beg)) nil t)
+                     (match-beginning 0)
+                   (point-max)))
+    (setq mode-line-buffer-identification
+         (funcall todos-mode-line-function name))
+    (narrow-to-region cat-begin cat-end)
+    (todos-prefix-overlays)
+    (goto-char (point-min))
+    (if (re-search-forward (concat "\n\\(" (regexp-quote todos-category-done)
+                                  "\\)") nil t)
+       (progn
+         (setq done-start (match-beginning 0))
+         (setq done-sep-start (match-beginning 1))
+         (setq done-end (match-end 0)))
+      (error "Category %s is missing todos-category-done string" name))
+    (if todos-show-done-only
+       (narrow-to-region (1+ done-end) (point-max))
+      (when (and todos-show-with-done
+                (re-search-forward todos-done-string-start nil t))
+       ;; Now we want to see the done items, so reset displayed end to end of
+       ;; done items.
+       (setq done-start cat-end)
+       ;; Make display overlay for done items separator string, unless there
+       ;; already is one.
+       (let* ((done-sep todos-done-separator)
+              (ov (progn (goto-char done-sep-start)
+                         (todos-get-overlay 'separator))))
+         (unless ov
+           (setq ov (make-overlay done-sep-start done-end))
+           (overlay-put ov 'todos 'separator)
+           (overlay-put ov 'display done-sep))))
+      (narrow-to-region (point-min) done-start)
+      ;; Loading this from todos-mode, or adding it to the mode hook, causes
+      ;; Emacs to hang in todos-item-start, at (looking-at todos-item-start).
+      (when todos-highlight-item
+       (require 'hl-line)
+       (hl-line-mode 1)))))
 
-The argument MATCH-TYPE specifies the matching requirements on
-the category name: with the value `todo' or `archive' the name
-must complete to that of an existing todo or archive category,
-respectively; with the value `add' the name must not be that of
-an existing category; with all other values both existing and new
-valid category names are accepted.
+(defun todos-get-count (type &optional category)
+  "Return count of TYPE items in CATEGORY.
+If CATEGORY is nil, default to the current category."
+  (let* ((cat (or category (todos-current-category)))
+        (counts (cdr (assoc cat todos-categories)))
+        (idx (cond ((eq type 'todo) 0)
+                   ((eq type 'diary) 1)
+                   ((eq type 'done) 2)
+                   ((eq type 'archived) 3))))
+    (aref counts idx)))
 
-With non-nil argument FILE prompt for a file and complete only
-against categories in that file; otherwise complete against all
-categories from `todos-category-completions-files'."
-  ;; Allow SPC to insert spaces, for adding new category names.
-  (let ((map minibuffer-local-completion-map))
-    (define-key map " " nil)
-    (let* ((add (eq match-type 'add))
-          (archive (eq match-type 'archive))
-          (file0 (when (and file (> (length todos-files) 1))
-                   (todos-read-file-name (concat "Choose a" (if archive
-                                                                "n archive"
-                                                              " todo")
-                                                 " file: ") archive t)))
-          (completions (unless file0 (todos-category-completions archive)))
-          (categories (cond (file0
-                             (with-current-buffer
-                                 (find-file-noselect file0 'nowarn)
-                               (let ((todos-current-todos-file file0))
-                                 todos-categories)))
-                            ((and add (not file))
-                             (with-current-buffer
-                                 (find-file-noselect todos-current-todos-file)
-                               todos-categories))
-                            (t
-                             completions)))
-          (completion-ignore-case todos-completion-ignore-case)
-          (cat (completing-read prompt categories nil
-                                (eq match-type 'todo) nil nil
-                                ;; Unless we're adding a category via
-                                ;; todos-add-category, set default
-                                ;; for existing categories to the
-                                ;; current category of the chosen
-                                ;; file or else of the current file.
-                                (if (and categories (not add))
-                                    (with-current-buffer
-                                        (find-file-noselect
-                                         (or file0
-                                             todos-current-todos-file
-                                             (todos-absolute-file-name
-                                              todos-default-todos-file)))
-                                      (todos-current-category))
-                                  ;; Trigger prompt for initial category.
-                                  "")))
-          (catfil (cdr (assoc cat completions)))
-          (str "Category \"%s\" from which file (TAB for choices)? "))
-      ;; If we do category completion and the chosen category name
-      ;; occurs in more than one file, prompt to choose one file.
-      (unless (or file0 add (not catfil))
-       (setq file0 (file-truename
-                    (if (atom catfil)
-                        catfil
-                      (todos-absolute-file-name
-                       (let ((files (mapcar 'todos-short-file-name catfil)))
-                         (completing-read (format str cat) files)))))))
-      ;; Default to the current file.
-      (unless file0 (setq file0 todos-current-todos-file))
-      ;; First validate only a name passed interactively from
-      ;; todos-add-category, which must be of a nonexisting category.
-      (unless (and (assoc cat categories) (not add))
-       ;; Validate only against completion categories.
-       (let ((todos-categories categories))
-         (setq cat (todos-validate-name cat 'category)))
-       ;; When user enters a nonexisting category name by jumping or
-       ;; moving, confirm that it should be added, then validate.
-       (unless add
-         (if (todos-y-or-n-p (format "Add new category \"%s\" to file \"%s\"? "
-                               cat (todos-short-file-name file0)))
-             (progn
-               (when (assoc cat categories)
-                 (let ((todos-categories categories))
-                   (setq cat (todos-validate-name cat 'category))))
-               ;; Restore point and narrowing after adding new
-               ;; category, to avoid moving to beginning of file when
-               ;; moving marked items to a new category
-               ;; (todos-move-item).
-               (save-excursion
-                 (save-restriction
-                   (todos-add-category file0 cat))))
-           ;; If we decide not to add a category, exit without returning.
-           (keyboard-quit))))
-      (cons cat file0))))
+(defun todos-update-count (type increment &optional category)
+  "Change count of TYPE items in CATEGORY by integer INCREMENT.
+With nil or omitted CATEGORY, default to the current category."
+  (let* ((cat (or category (todos-current-category)))
+        (counts (cdr (assoc cat todos-categories)))
+        (idx (cond ((eq type 'todo) 0)
+                   ((eq type 'diary) 1)
+                   ((eq type 'done) 2)
+                   ((eq type 'archived) 3))))
+    (aset counts idx (+ increment (aref counts idx)))))
 
-(defun todos-validate-name (name type)
-  "Prompt for new NAME for TYPE until it is valid, then return it.
-TYPE can be either of the symbols `file' or `category'."
-  (let ((categories todos-categories)
-       (files (mapcar 'todos-short-file-name todos-files))
-       prompt)
-    (while
-       (and
-        (cond ((string= "" name)
-               (setq prompt
-                     (cond ((eq type 'file)
-                            (if files
-                                "Enter a non-empty file name: "
-                              ;; Empty string passed by todos-show to
-                              ;; prompt for initial Todos file.
-                              (concat "Initial file name ["
-                                      todos-initial-file "]: ")))
-                           ((eq type 'category)
-                            (if categories
-                                "Enter a non-empty category name: "
-                              ;; Empty string passed by todos-show to
-                              ;; prompt for initial category of a new
-                              ;; Todos file.
-                              (concat "Initial category name ["
-                                      todos-initial-category "]: "))))))
-              ((string-match "\\`\\s-+\\'" name)
-               (setq prompt
-                     "Enter a name that does not contain only white space: "))
-              ((and (eq type 'file) (member name files))
-               (setq prompt "Enter a non-existing file name: "))
-              ((and (eq type 'category) (assoc name categories))
-               (setq prompt "Enter a non-existing category name: ")))
-        (setq name (if (or (and (eq type 'file) files)
-                           (and (eq type 'category) categories))
-                       (completing-read prompt (cond ((eq type 'file)
-                                                      files)
-                                                     ((eq type 'category)
-                                                      categories)))
-                     ;; Offer default initial name.
-                     (completing-read prompt (if (eq type 'file)
-                                                 files
-                                               categories)
-                                      nil nil (if (eq type 'file)
-                                                  todos-initial-file
-                                                todos-initial-category))))))
-    name))
+(defun todos-set-categories ()
+  "Set `todos-categories' from the sexp at the top of the file."
+  ;; New archive files created by `todos-move-category' are empty, which would
+  ;; make the sexp test fail and raise an error, so in this case we skip it.
+  (unless (zerop (buffer-size))
+    (save-excursion
+      (save-restriction
+       (widen)
+       (goto-char (point-min))
+       (setq todos-categories
+             (if (looking-at "\(\(\"")
+                 (read (buffer-substring-no-properties
+                        (line-beginning-position)
+                        (line-end-position)))
+               (error "Invalid or missing todos-categories sexp")))))))
 
-;; Adapted from calendar-read-date and calendar-date-string.
-(defun todos-read-date (&optional arg mo yr)
-  "Prompt for Gregorian date and return it in the current format.
+(defun todos-update-categories-sexp ()
+  "Update the `todos-categories' sexp at the top of the file."
+  (let (buffer-read-only)
+    (save-excursion
+      (save-restriction
+       (widen)
+       (goto-char (point-min))
+       (if (looking-at (concat "^" (regexp-quote todos-category-beg)))
+           (progn (newline) (goto-char (point-min)) ; Make space for sexp.
+                  (setq todos-categories (todos-make-categories-list t)))
+         (delete-region (line-beginning-position) (line-end-position)))
+       (prin1 todos-categories (current-buffer))))))
 
-With non-nil ARG, prompt for and return only the date component
-specified by ARG, which can be one of these symbols:
-`month' (prompt for name, return name or number according to
-value of `calendar-date-display-form'), `day' of month, or
-`year'.  The value of each of these components can be `*',
-indicating an unspecified month, day, or year.
+(defun todos-make-categories-list (&optional force)
+  "Return an alist of Todos categories and their item counts.
+With non-nil argument FORCE parse the entire file to build the
+list; otherwise, get the value by reading the sexp at the top of
+the file."
+  (setq todos-categories nil)
+  (save-excursion
+    (save-restriction
+      (widen)
+      (goto-char (point-min))
+      (let (counts cat archive)
+       ;; If the file is a todo file and has archived items, identify the
+       ;; archive, in order to count its items.  But skip this with
+       ;; `todos-convert-legacy-files', since that converts filed items to
+       ;; archived items.
+       (when buffer-file-name   ; During conversion there is no file yet.
+         ;; If the file is an archive, it doesn't have an archive.
+         (unless (member (file-truename buffer-file-name)
+                         (funcall todos-files-function t))
+           (setq archive (concat (file-name-sans-extension
+                                  todos-current-todos-file) ".toda"))))
+       (while (not (eobp))
+         (cond ((looking-at (concat (regexp-quote todos-category-beg)
+                                    "\\(.*\\)\n"))
+                (setq cat (match-string-no-properties 1))
+                ;; Counts for each category: [todo diary done archive]
+                (setq counts (make-vector 4 0))
+                (setq todos-categories
+                      (append todos-categories (list (cons cat counts))))
+                ;; Add archived item count to the todo file item counts.
+                ;; Make sure to include newly created archives, e.g. due to
+                ;; todos-move-category.
+                (when (member archive (funcall todos-files-function t))
+                  (let ((archive-count 0))
+                    (with-current-buffer (find-file-noselect archive)
+                      (widen)
+                      (goto-char (point-min))
+                      (when (re-search-forward
+                             (concat "^" (regexp-quote todos-category-beg)
+                                     cat "$")
+                             (point-max) t)
+                        (forward-line)
+                        (while (not (or (looking-at
+                                         (concat
+                                          (regexp-quote todos-category-beg)
+                                          "\\(.*\\)\n"))
+                                        (eobp)))
+                          (when (looking-at todos-done-string-start)
+                            (setq archive-count (1+ archive-count)))
+                          (forward-line))))
+                    (todos-update-count 'archived archive-count cat))))
+               ((looking-at todos-done-string-start)
+                (todos-update-count 'done 1 cat))
+               ((looking-at (concat "^\\("
+                                    (regexp-quote diary-nonmarking-symbol)
+                                    "\\)?" todos-date-pattern))
+                (todos-update-count 'diary 1 cat)
+                (todos-update-count 'todo 1 cat))
+               ((looking-at (concat todos-date-string-start todos-date-pattern))
+                (todos-update-count 'todo 1 cat))
+               ;; If first line is todos-categories list, use it and end loop
+               ;; -- unless FORCEd to scan whole file.
+               ((bobp)
+                (unless force
+                  (setq todos-categories (read (buffer-substring-no-properties
+                                                (line-beginning-position)
+                                                (line-end-position))))
+                  (goto-char (1- (point-max))))))
+         (forward-line)))))
+  todos-categories)
 
-When ARG is `day', non-nil arguments MO and YR determine the
-number of the last the day of the month."
-  (let (year monthname month day
-            dayname)                   ; Needed by calendar-date-display-form.
-    (when (or (not arg) (eq arg 'year))
-      (while (if (natnump year) (< year 1) (not (eq year '*)))
-       (setq year (read-from-minibuffer
-                   "Year (>0 or RET for this year or * for any year): "
-                   nil nil t nil (number-to-string
-                                  (calendar-extract-year
-                                   (calendar-current-date)))))))
-    (when (or (not arg) (eq arg 'month))
-      (let* ((marray todos-month-name-array)
-            (mlist (append marray nil))
-            (mabarray todos-month-abbrev-array)
-            (mablist (append mabarray nil))
-            (completion-ignore-case todos-completion-ignore-case))
-       (setq monthname (completing-read
-                        "Month name (RET for current month, * for any month): "
-                        ;; (mapcar 'list (append marray nil))
-                        mlist nil t nil nil
-                        (calendar-month-name (calendar-extract-month
-                                              (calendar-current-date)) t))
-             ;; month (cdr (assoc-string
-             ;;                  monthname (calendar-make-alist marray nil nil
-             ;;                                                 abbrevs))))))
-             month (1+ (- (length mlist)
-                          (length (or (member monthname mlist)
-                                      (member monthname mablist))))))
-       (setq monthname (aref mabarray (1- month)))))
-    (when (or (not arg) (eq arg 'day))
-      (let ((last (let ((mm (or month mo))
-                       (yy (or year yr)))
-                   ;; If month is unspecified, use a month with 31
-                   ;; days for checking day of month input.  Does
-                   ;; Calendar do anything special when * is
-                   ;; currently a shorter month?
-                   (if (= mm 13) (setq mm 1))
-                   ;; If year is unspecified, use a leap year to
-                   ;; allow Feb. 29.
-                   (if (eq year '*) (setq yy 2012))
-                   (calendar-last-day-of-month mm yy))))
-       (while (if (natnump day) (or (< day 1) (> day last)) (not (eq day '*)))
-         (setq day (read-from-minibuffer
-                    (format "Day (1-%d or RET for today or * for any day): "
-                            last)
-                    nil nil t nil (number-to-string
-                                   (calendar-extract-day
-                                    (calendar-current-date))))))))
-    ;; Stringify read values (monthname is already a string).
-    (and year (setq year (if (eq year '*)
-                            (symbol-name '*)
-                          (number-to-string year))))
-    (and day (setq day (if (eq day '*)
-                          (symbol-name '*)
-                        (number-to-string day))))
-    (and month (setq month (if (eq month '*)
-                              (symbol-name '*)
-                            (number-to-string month))))
-    (if arg
-       (cond ((eq arg 'year) year)
-             ((eq arg 'day) day)
-             ((eq arg 'month)
-              (if (memq 'month calendar-date-display-form)
-                  month
-                monthname)))
-      (mapconcat 'eval calendar-date-display-form ""))))
+(defun todos-repair-categories-sexp ()
+  "Repair corrupt Todos categories sexp.
+This should only be needed as a consequence of careless manual
+editing or a bug in todos.el.
 
-(defun todos-read-dayname ()
-  "Choose name of a day of the week with completion and return it."
-  (let ((completion-ignore-case todos-completion-ignore-case))
-    (completing-read "Enter a day name: "
-                    (append calendar-day-name-array nil)
-                    nil t)))
-  
-(defun todos-read-time ()
-  "Prompt for and return a valid clock time as a string.
+*Warning*: Calling this command restores the category order to
+the list element order in the Todos categories sexp, so any order
+changes made in Todos Categories mode will have to be made again."
+  (interactive)
+  (let ((todos-categories (todos-make-categories-list t)))
+    (todos-update-categories-sexp)))
+
+(defun todos-check-format ()
+  "Signal an error if the current Todos file is ill-formatted.
+Otherwise return t.  Display a message if the file is well-formed
+but the categories sexp differs from the current value of
+`todos-categories'."
+  (save-excursion
+    (save-restriction
+      (widen)
+      (goto-char (point-min))
+      (let* ((cats (prin1-to-string todos-categories))
+            (ssexp (buffer-substring-no-properties (line-beginning-position)
+                                                   (line-end-position)))
+            (sexp (read ssexp)))
+       ;; Check the first line for `todos-categories' sexp.
+       (dolist (c sexp)
+         (let ((v (cdr c)))
+           (unless (and (stringp (car c))
+                        (vectorp v)
+                        (= 4 (length v)))
+             (user-error "Invalid or missing todos-categories sexp"))))
+       (forward-line)
+       ;; Check well-formedness of categories.
+       (let ((legit (concat
+                     "\\(^" (regexp-quote todos-category-beg) "\\)"
+                     "\\|\\(" todos-date-string-start todos-date-pattern "\\)"
+                     "\\|\\(^[ \t]+[^ \t]*\\)"
+                     "\\|^$"
+                     "\\|\\(^" (regexp-quote todos-category-done) "\\)"
+                     "\\|\\(" todos-done-string-start "\\)")))
+         (while (not (eobp))
+           (unless (looking-at legit)
+             (user-error "Illegitimate Todos file format at line %d"
+                    (line-number-at-pos (point))))
+           (forward-line)))
+       ;; Warn user if categories sexp has changed.
+       (unless (string= ssexp cats)
+         (message (concat "The sexp at the beginning of the file differs "
+                          "from the value of `todos-categories.\n"
+                          "If the sexp is wrong, you can fix it with "
+                          "M-x todos-repair-categories-sexp,\n"
+                          "but note this reverts any changes you have "
+                          "made in the order of the categories."))))))
+  t)
 
-Valid time strings are those matching `diary-time-regexp'.
-Typing `<return>' at the prompt returns the current time, if the
-user option `todos-always-add-time-string' is non-nil, otherwise
-the empty string (i.e., no time string)."
-  (let (valid answer)
-    (while (not valid)
-      (setq answer (read-string "Enter a clock time: " nil nil
-                               (when todos-always-add-time-string
-                                 (substring (current-time-string) 11 16))))
-      (when (or (string= "" answer)
-               (string-match diary-time-regexp answer))
-       (setq valid t)))
-    answer))
+(defun todos-item-start ()
+  "Move to start of current Todos item and return its position."
+  (unless (or
+          ;; Buffer is empty (invocation possible e.g. via todos-forward-item
+          ;; from todos-filter-items when processing category with no todo
+          ;; items).
+          (eq (point-min) (point-max))
+          ;; Point is on the empty line below category's last todo item...
+          (and (looking-at "^$")
+               (or (eobp)              ; ...and done items are hidden...
+                   (save-excursion     ; ...or done items are visible.
+                     (forward-line)
+                     (looking-at (concat "^"
+                                         (regexp-quote todos-category-done))))))
+          ;; Buffer is widened.
+          (looking-at (regexp-quote todos-category-beg)))
+    (goto-char (line-beginning-position))
+    (while (not (looking-at todos-item-start))
+      (forward-line -1))
+    (point)))
 
-;; -----------------------------------------------------------------------------
-;;; Todos Categories mode tabulation and sorting
-;; -----------------------------------------------------------------------------
+(defun todos-item-end ()
+  "Move to end of current Todos item and return its position."
+  ;; Items cannot end with a blank line.
+  (unless (looking-at "^$")
+    (let* ((done (todos-done-item-p))
+          (to-lim nil)
+          ;; For todo items, end is before the done items section, for done
+          ;; items, end is before the next category.  If these limits are
+          ;; missing or inaccessible, end it before the end of the buffer.
+          (lim (if (save-excursion
+                     (re-search-forward
+                      (concat "^" (regexp-quote (if done
+                                                    todos-category-beg
+                                                  todos-category-done)))
+                      nil t))
+                   (progn (setq to-lim t) (match-beginning 0))
+                 (point-max))))
+      (when (bolp) (forward-char))     ; Find start of next item.
+      (goto-char (if (re-search-forward todos-item-start lim t)
+                    (match-beginning 0)
+                  (if to-lim lim (point-max))))
+      ;; For last todo item, skip back over the empty line before the done
+      ;; items section, else just back to the end of the previous line.
+      ;; (When byte-comiled, backward-char barfs on an argument that evaluates
+      ;; to nil (bug#14565).)
+      ;; (backward-char (when (and to-lim (not done) (eq (point) lim)) 2))
+      (backward-char (if (and to-lim (not done) (eq (point) lim)) 2 1))
+      (point))))
 
-(defvar todos-categories-buffer "*Todos Categories*"
-  "Name of buffer in Todos Categories mode.")
+(defun todos-item-string ()
+  "Return bare text of current item as a string."
+  (let ((opoint (point))
+       (start (todos-item-start))
+       (end (todos-item-end)))
+    (goto-char opoint)
+    (and start end (buffer-substring-no-properties start end))))
 
-(defun todos-longest-category-name-length (categories)
-  "Return the length of the longest name in list CATEGORIES."
-  (let ((longest 0))
-    (dolist (c categories longest)
-      (setq longest (max longest (length c))))))
+(defun todos-forward-item (&optional count)
+  "Move point COUNT items down (by default, move down by one item)."
+  (let* ((not-done (not (or (todos-done-item-p) (looking-at "^$"))))
+        (start (line-end-position)))
+    (goto-char start)
+    (if (re-search-forward todos-item-start nil t (or count 1))
+       (goto-char (match-beginning 0))
+      (goto-char (point-max)))
+    ;; If points advances by one from a todo to a done item, go back
+    ;; to the space above todos-done-separator, since that is a
+    ;; legitimate place to insert an item.  But skip this space if
+    ;; count > 1, since that should only stop on an item.
+    (when (and not-done (todos-done-item-p) (not count))
+      ;; (if (or (not count) (= count 1))
+         (re-search-backward "^$" start t))));)
+    ;; The preceding sexp is insufficient when buffer is not narrowed,
+    ;; since there could be no done items in this category, so the
+    ;; search puts us on first todo item of next category.  Does this
+    ;; ever happen?  If so:
+    ;; (let ((opoint) (point))
+    ;;   (forward-line -1)
+    ;;   (when (or (not count) (= count 1))
+    ;;         (cond ((looking-at (concat "^" (regexp-quote todos-category-beg)))
+    ;;                (forward-line -2))
+    ;;               ((looking-at (concat "^" (regexp-quote todos-category-done)))
+    ;;                (forward-line -1))
+    ;;               (t
+    ;;                (goto-char opoint)))))))
 
-(defun todos-adjusted-category-label-length ()
-  "Return adjusted length of category label button.
-The adjustment ensures proper tabular alignment in Todos
-Categories mode."
-  (let* ((categories (mapcar 'car todos-categories))
-        (longest (todos-longest-category-name-length categories))
-        (catlablen (length todos-categories-category-label))
-        (lc-diff (- longest catlablen)))
-    (if (and (natnump lc-diff) (cl-oddp lc-diff))
-       (1+ longest)
-      (max longest catlablen))))
+(defun todos-backward-item (&optional count)
+  "Move point up to start of item with next higher priority.
+With positive numerical prefix COUNT, move point COUNT items
+upward.
 
-(defun todos-padded-string (str)
-  "Return category name or label string STR padded with spaces.
-The placement of the padding is determined by the value of user
-option `todos-categories-align'."
-  (let* ((len (todos-adjusted-category-label-length))
-        (strlen (length str))
-        (strlen-odd (eq (logand strlen 1) 1))
-        (padding (max 0 (/ (- len strlen) 2)))
-        (padding-left (cond ((eq todos-categories-align 'left) 0)
-                            ((eq todos-categories-align 'center) padding)
-                            ((eq todos-categories-align 'right)
-                             (if strlen-odd (1+ (* padding 2)) (* padding 2)))))
-        (padding-right (cond ((eq todos-categories-align 'left)
-                              (if strlen-odd (1+ (* padding 2)) (* padding 2)))
-                             ((eq todos-categories-align 'center)
-                              (if strlen-odd (1+ padding) padding))
-                             ((eq todos-categories-align 'right) 0))))
-    (concat (make-string padding-left 32) str (make-string padding-right 32))))
+If the category's done items are visible, this command called
+with a prefix argument only moves point to a higher item, e.g.,
+with point on the first done item and called with prefix 1, it
+moves to the last todo item; but if called with point on the
+first done item without a prefix argument, it moves point the the
+empty line above the done items separator."
+  (let* ((done (todos-done-item-p)))
+    (todos-item-start)
+    (unless (bobp)
+      (re-search-backward todos-item-start nil t (or count 1)))
+    ;; Unless this is a regexp filtered items buffer (which can contain
+    ;; intermixed todo and done items), if points advances by one from a
+    ;; done to a todo item, go back to the space above
+    ;; todos-done-separator, since that is a legitimate place to insert an
+    ;; item.  But skip this space if count > 1, since that should only
+    ;; stop on an item.
+    (when (and done (not (todos-done-item-p)) (not count)
+                                   ;(or (not count) (= count 1))
+              (not (equal (buffer-name) todos-regexp-items-buffer)))
+      (re-search-forward (concat "^" (regexp-quote todos-category-done))
+                        nil t)
+      (forward-line -1))))
 
-(defvar todos-descending-counts nil
-  "List of keys for category counts sorted in descending order.")
+(defun todos-remove-item ()
+  "Internal function called in editing, deleting or moving items."
+  (let* ((end (progn (todos-item-end) (1+ (point))))
+        (beg (todos-item-start))
+        (ov (todos-get-overlay 'prefix)))
+    (when ov (delete-overlay ov))
+    (delete-region beg end)))
 
-(defun todos-sort (list &optional key)
-  "Return a copy of LIST, possibly sorted according to KEY."
-  (let* ((l (copy-sequence list))
-        (fn (if (eq key 'alpha)
-                  (lambda (x) (upcase x)) ; Alphabetize case insensitively.
-                (lambda (x) (todos-get-count key x))))
-        ;; Keep track of whether the last sort by key was descending or
-        ;; ascending.
-        (descending (member key todos-descending-counts))
-        (cmp (if (eq key 'alpha)
-                 'string<
-               (if descending '< '>)))
-        (pred (lambda (s1 s2) (let ((t1 (funcall fn (car s1)))
-                                    (t2 (funcall fn (car s2))))
-                                (funcall cmp t1 t2)))))
-    (when key
-      (setq l (sort l pred))
-      ;; Switch between descending and ascending sort order.
-      (if descending
-         (setq todos-descending-counts
-               (delete key todos-descending-counts))
-       (push key todos-descending-counts)))
-    l))
+(defun todos-diary-item-p ()
+  "Return non-nil if item at point has diary entry format."
+  (save-excursion
+    (when (todos-item-string)          ; Exclude empty lines.
+      (todos-item-start)
+      (not (looking-at (regexp-quote todos-nondiary-start))))))
 
-(defun todos-display-sorted (type)
-  "Keep point on the TYPE count sorting button just clicked."
-  (let ((opoint (point)))
-    (todos-update-categories-display type)
-    (goto-char opoint)))
+(defun todos-diary-goto-entry ()
+  "Jump to todo item included in Fancy Diary display.
+Helper function for `diary-goto-entry'."
+  (when (eq major-mode 'todos-mode)
+    (let ((opoint (point)))
+      (re-search-backward (concat "^" (regexp-quote todos-category-beg)
+                                 "\\(.*\\)\n") nil t)
+      (todos-category-number (match-string 1))
+      (todos-category-select)
+      (goto-char opoint))))
 
-(defun todos-label-to-key (label)
-  "Return symbol for sort key associated with LABEL."
-  (let (key)
-    (cond ((string= label todos-categories-category-label)
-          (setq key 'alpha))
-         ((string= label todos-categories-todo-label)
-          (setq key 'todo))
-         ((string= label todos-categories-diary-label)
-          (setq key 'diary))
-         ((string= label todos-categories-done-label)
-          (setq key 'done))
-         ((string= label todos-categories-archived-label)
-          (setq key 'archived)))
-    key))
+(defun todos-done-item-p ()
+  "Return non-nil if item at point is a done item."
+  (save-excursion
+    (todos-item-start)
+    (looking-at todos-done-string-start)))
 
-(defun todos-insert-sort-button (label)
-  "Insert button for displaying categories sorted by item counts.
-LABEL determines which type of count is sorted."
-  (let* ((str (if (string= label todos-categories-category-label)
-                 (todos-padded-string label)
-               label))
-        (beg (point))
-        (end (+ beg (length str)))
-        ov)
-    (insert-button str 'face nil
-                  'action
-                  `(lambda (button)
-                     (let ((key (todos-label-to-key ,label)))
-                       (if (and (member key todos-descending-counts)
-                                (eq key 'alpha))
-                           (progn
-                             ;; If display is alphabetical, switch back to
-                             ;; category priority order.
-                             (todos-display-sorted nil)
-                             (setq todos-descending-counts
-                                   (delete key todos-descending-counts)))
-                         (todos-display-sorted key)))))
-    (setq ov (make-overlay beg end))
-    (overlay-put ov 'face 'todos-button)))
+(defun todos-done-item-section-p ()
+  "Return non-nil if point is in category's done items section."
+  (save-excursion
+    (or (re-search-backward (concat "^" (regexp-quote todos-category-done))
+                           nil t)
+       (progn (goto-char (point-min))
+              (looking-at todos-done-string-start)))))
 
-(defun todos-total-item-counts ()
-  "Return a list of total item counts for the current file."
-  (mapcar (lambda (i) (apply '+ (mapcar (lambda (l) (aref l i))
-                                       (mapcar 'cdr todos-categories))))
-         (list 0 1 2 3)))
+(defun todos-reset-done-separator (sep)
+  "Replace existing overlays of done items separator string SEP."
+  (save-excursion
+    (save-restriction
+      (widen)
+      (goto-char (point-min))
+      (while (re-search-forward
+             (concat "\n\\(" (regexp-quote todos-category-done) "\\)") nil t)
+       (let* ((beg (match-beginning 1))
+              (end (match-end 0))
+              (ov (progn (goto-char beg)
+                         (todos-get-overlay 'separator)))
+              (old-sep (when ov (overlay-get ov 'display)))
+              new-ov)
+         (when old-sep
+           (unless (string= old-sep sep)
+             (setq new-ov (make-overlay beg end))
+             (overlay-put new-ov 'todos 'separator)
+             (overlay-put new-ov 'display todos-done-separator)
+             (delete-overlay ov))))))))
 
-(defvar todos-categories-category-number 0
-  "Variable for numbering categories in Todos Categories mode.")
+(defun todos-get-overlay (val)
+  "Return the overlay at point whose `todos' property has value VAL."
+  ;; Use overlays-in to find prefix overlays and check over two
+  ;; positions to find done separator overlay.
+  (let ((ovs (overlays-in (point) (1+ (point))))
+       ov)
+    (catch 'done
+      (while ovs
+       (setq ov (pop ovs))
+       (when (eq (overlay-get ov 'todos) val)
+         (throw 'done ov))))))
 
-(defun todos-insert-category-line (cat &optional nonum)
-  "Insert button with category CAT's name and item counts.
-With non-nil argument NONUM show only these; otherwise, insert a
-number in front of the button indicating the category's priority.
-The number and the category name are separated by the string
-which is the value of the user option
-`todos-categories-number-separator'."
-  (let ((archive (member todos-current-todos-file todos-archives))
-       (num todos-categories-category-number)
-       (str (todos-padded-string cat))
-       (opoint (point)))
-    (setq num (1+ num) todos-categories-category-number num)
-    (insert-button
-     (concat (if nonum
-                (make-string (+ 4 (length todos-categories-number-separator))
-                             32)
-              (format " %3d%s" num todos-categories-number-separator))
-            str
-            (mapconcat (lambda (elt)
-                         (concat
-                          (make-string (1+ (/ (length (car elt)) 2)) 32) ; label
-                          (format "%3d" (todos-get-count (cdr elt) cat)) ; count
-                          ;; Add an extra space if label length is odd.
-                          (when (cl-oddp (length (car elt))) " ")))
-                       (if archive
-                           (list (cons todos-categories-done-label 'done))
-                         (list (cons todos-categories-todo-label 'todo)
-                               (cons todos-categories-diary-label 'diary)
-                               (cons todos-categories-done-label 'done)
-                               (cons todos-categories-archived-label
-                                     'archived)))
-                         "")
-            " ") ; Make highlighting on last column look better.
-     'face (if (and todos-skip-archived-categories
-                   (zerop (todos-get-count 'todo cat))
-                   (zerop (todos-get-count 'done cat))
-                   (not (zerop (todos-get-count 'archived cat))))
-              'todos-archived-only
-            nil)
-     'action `(lambda (button) (let ((buf (current-buffer)))
-                                (todos-jump-to-category nil ,cat)
-                                (kill-buffer buf))))
-    ;; Highlight the sorted count column.
-    (let* ((beg (+ opoint 7 (length str)))
-          end ovl)
-      (cond ((eq nonum 'todo)
-            (setq beg (+ beg 1 (/ (length todos-categories-todo-label) 2))))
-           ((eq nonum 'diary)
-            (setq beg (+ beg 1 (length todos-categories-todo-label)
-                          2 (/ (length todos-categories-diary-label) 2))))
-           ((eq nonum 'done)
-            (setq beg (+ beg 1 (length todos-categories-todo-label)
-                          2 (length todos-categories-diary-label)
-                          2 (/ (length todos-categories-done-label) 2))))
-           ((eq nonum 'archived)
-            (setq beg (+ beg 1 (length todos-categories-todo-label)
-                          2 (length todos-categories-diary-label)
-                          2 (length todos-categories-done-label)
-                          2 (/ (length todos-categories-archived-label) 2)))))
-      (unless (= beg (+ opoint 7 (length str))) ; Don't highlight categories.
-       (setq end (+ beg 4))
-       (setq ovl (make-overlay beg end))
-       (overlay-put ovl 'face 'todos-sorted-column)))
-    (newline)))
+(defun todos-marked-item-p ()
+  "Non-nil if this item begins with `todos-item-mark'.
+In that case, return the item's prefix overlay."
+  (let* ((ov (todos-get-overlay 'prefix))
+        ;; If an item insertion command is called on a Todos file
+        ;; before it is visited, it has no prefix overlays yet, so
+        ;; check for this.
+        (pref (when ov (overlay-get ov 'before-string)))
+        (marked (when pref
+                  (string-match (concat "^" (regexp-quote todos-item-mark))
+                                pref))))
+    (when marked ov)))
 
-(defun todos-display-categories ()
-  "Prepare buffer for displaying table of categories and item counts."
-  (unless (eq major-mode 'todos-categories-mode)
-    (setq todos-global-current-todos-file
-         (or todos-current-todos-file
-             (todos-absolute-file-name todos-default-todos-file)))
-    (set-window-buffer (selected-window)
-                      (set-buffer (get-buffer-create todos-categories-buffer)))
-    (kill-all-local-variables)
-    (todos-categories-mode)
-    (let ((archive (member todos-current-todos-file todos-archives))
-         buffer-read-only)
-      (erase-buffer)
-      (insert (format (concat "Category counts for Todos "
-                             (if archive "archive" "file")
-                             " \"%s\".")
-                     (todos-short-file-name todos-current-todos-file)))
-      (newline 2)
-      ;; Make space for the column of category numbers.
-      (insert (make-string (+ 4 (length todos-categories-number-separator)) 32))
-      ;; Add the category and item count buttons (if this is the list of
-      ;; categories in an archive, show only done item counts).
-      (todos-insert-sort-button todos-categories-category-label)
-      (if archive
-         (progn
-           (insert (make-string 3 32))
-           (todos-insert-sort-button todos-categories-done-label))
-       (insert (make-string 3 32))
-       (todos-insert-sort-button todos-categories-todo-label)
-       (insert (make-string 2 32))
-       (todos-insert-sort-button todos-categories-diary-label)
-       (insert (make-string 2 32))
-       (todos-insert-sort-button todos-categories-done-label)
-       (insert (make-string 2 32))
-       (todos-insert-sort-button todos-categories-archived-label))
-      (newline 2))))
+(defun todos-insert-with-overlays (item)
+  "Insert ITEM at point and update prefix/priority number overlays."
+  (todos-item-start)
+  ;; Insertion pushes item down but not its prefix overlay.  When the
+  ;; overlay includes a mark, this would now mark the inserted ITEM,
+  ;; so move it to the pushed down item.
+  (let ((ov (todos-get-overlay 'prefix))
+       (marked (todos-marked-item-p)))
+    (insert item "\n")
+    (when marked (move-overlay ov (point) (point))))
+  (todos-backward-item)
+  (todos-prefix-overlays))
 
-(defun todos-update-categories-display (sortkey)
-  "Populate table of categories and sort by SORTKEY."
-  (let* ((cats0 todos-categories)
-        (cats (todos-sort cats0 sortkey))
-        (archive (member todos-current-todos-file todos-archives))
-        (todos-categories-category-number 0)
-        ;; Find start of Category button if we just entered Todos Categories
-        ;; mode.
-        (pt (if (eq (point) (point-max))
-                (save-excursion
-                  (forward-line -2)
-                  (goto-char (next-single-char-property-change
-                              (point) 'face nil (line-end-position))))))
-        (buffer-read-only))
-    (forward-line 2)
-    (delete-region (point) (point-max))
-    ;; Fill in the table with buttonized lines, each showing a category and
-    ;; its item counts.
-    (mapc (lambda (cat) (todos-insert-category-line cat sortkey))
-         (mapcar 'car cats))
-    (newline)
-    ;; Add a line showing item count totals.
-    (insert (make-string (+ 4 (length todos-categories-number-separator)) 32)
-           (todos-padded-string todos-categories-totals-label)
-           (mapconcat
-            (lambda (elt)
-              (concat
-               (make-string (1+ (/ (length (car elt)) 2)) 32)
-               (format "%3d" (nth (cdr elt) (todos-total-item-counts)))
-               ;; Add an extra space if label length is odd.
-               (when (cl-oddp (length (car elt))) " ")))
-            (if archive
-                (list (cons todos-categories-done-label 2))
-              (list (cons todos-categories-todo-label 0)
-                    (cons todos-categories-diary-label 1)
-                    (cons todos-categories-done-label 2)
-                    (cons todos-categories-archived-label 3)))
-            ""))
-    ;; Put cursor on Category button initially.
-    (if pt (goto-char pt))
-    (setq buffer-read-only t)))
+(defun todos-prefix-overlays ()
+  "Update the prefix overlays of the current category's items.
+The overlay's value is the string `todos-prefix' or with non-nil
+`todos-number-prefix' an integer in the sequence from 1 to
+the number of todo or done items in the category indicating the
+item's priority.  Todo and done items are numbered independently
+of each other."
+  (let ((num 0)
+       (cat-tp (or (cdr (assoc-string
+                         (todos-current-category)
+                         (nth 2 (assoc-string todos-current-todos-file
+                                              todos-top-priorities-overrides))))
+                   todos-top-priorities))
+       done prefix)
+    (save-excursion
+      (goto-char (point-min))
+      (while (not (eobp))
+       (when (or (todos-date-string-matcher (line-end-position))
+                 (todos-done-string-matcher (line-end-position)))
+         (goto-char (match-beginning 0))
+         (setq num (1+ num))
+         ;; Reset number to 1 for first done item.
+         (when (and (eq major-mode 'todos-mode)
+                    (looking-at todos-done-string-start)
+                    (looking-back (concat "^"
+                                          (regexp-quote todos-category-done)
+                                          "\n")))
+           (setq num 1
+                 done t))
+         (setq prefix (concat (propertize
+                               (if todos-number-prefix
+                                   (number-to-string num)
+                                 todos-prefix)
+                               'face
+                               ;; Prefix of top priority items has a
+                               ;; distinct face in Todos mode.
+                               (if (and (eq major-mode 'todos-mode)
+                                        (not done)
+                                        (<= num cat-tp))
+                                   'todos-top-priority
+                                 'todos-prefix-string))
+                              " "))
+         (let ((ov (todos-get-overlay 'prefix))
+               (marked (todos-marked-item-p)))
+           ;; Prefix overlay must be at a single position so its
+           ;; bounds aren't changed when (re)moving an item.
+           (unless ov (setq ov (make-overlay (point) (point))))
+           (overlay-put ov 'todos 'prefix)
+           (overlay-put ov 'before-string (if marked
+                                              (concat todos-item-mark prefix)
+                                            prefix))))
+       (forward-line)))))
 
 ;; -----------------------------------------------------------------------------
-;;; Item filtering selection and display
+;;; Utilities for generating item insertion commands and key bindings
 ;; -----------------------------------------------------------------------------
 
-(defvar todos-multiple-filter-files nil
-  "List of files selected from `todos-multiple-filter-files' widget.")
+;; These two powerset definitions are adaptations of code published at
+;; http://rosettacode.org, whose content is licensed under GFDL 1.2.
+;; The recursive definition is a slight reformulation of
+;; http://rosettacode.org/wiki/Power_set#Common_Lisp.  The iterative
+;; definition is my Elisp implementation of
+;; http://rosettacode.org/wiki/Power_set#C.  Can either of these be
+;; included in Emacs, or is there no need to concerned about copyright
+;; here?
 
-(defvar todos-multiple-filter-files-widget nil
-  "Variable holding widget created by `todos-multiple-filter-files'.")
+;; (defun todos-powerset (list)
+;;   "Return the powerset of LIST."
+;;   (cond ((null list)
+;;      (list nil))
+;;     (t
+;;      (let ((recur (todos-powerset-recursive (cdr list)))
+;;            pset)
+;;        (dolist (elt recur pset)
+;;          (push (cons (car list) elt) pset))
+;;        (append pset recur)))))
 
-(defun todos-multiple-filter-files ()
-  "Pop to a buffer with a widget for choosing multiple filter files."
-  (require 'widget)
-  (eval-when-compile
-    (require 'wid-edit))
-  (with-current-buffer (get-buffer-create "*Todos Filter Files*")
-    (pop-to-buffer (current-buffer))
-    (erase-buffer)
-    (kill-all-local-variables)
-    (widget-insert "Select files for generating the top priorities list.\n\n")
-    (setq todos-multiple-filter-files-widget
-         (widget-create
-          `(set ,@(mapcar (lambda (x) (list 'const x))
-                          (mapcar 'todos-short-file-name
-                                  (funcall todos-files-function))))))
-    (widget-insert "\n")
-    (widget-create 'push-button
-                  :notify (lambda (widget &rest ignore)
-                            (setq todos-multiple-filter-files 'quit)
-                            (quit-window t)
-                            (exit-recursive-edit))
-                  "Cancel")
-    (widget-insert "   ")
-    (widget-create 'push-button
-                  :notify (lambda (&rest ignore)
-                            (setq todos-multiple-filter-files
-                                  (mapcar (lambda (f)
-                                            (file-truename
-                                             (concat todos-directory
-                                                     f ".todo")))
-                                          (widget-value
-                                           todos-multiple-filter-files-widget)))
-                            (quit-window t)
-                            (exit-recursive-edit))
-                  "Apply")
-    (use-local-map widget-keymap)
-    (widget-setup))
-  (message "Click \"Apply\" after selecting files.")
-  (recursive-edit))
+(defun todos-powerset (list)
+  "Return the powerset of LIST."
+  (let ((card (expt 2 (length list)))
+        pset elt)
+    (dotimes (n card)
+      (let ((i n)
+           (l list))
+       (while (not (zerop i))
+         (let ((arg (pop l)))
+           (when (cl-oddp i)
+             (setq elt (append elt (list arg))))
+           (setq i (/ i 2))))
+       (setq pset (append pset (list elt)))
+       (setq elt nil)))
+    pset))
 
-(defun todos-filter-items (filter &optional new multifile)
-  "Display a cross-categorial list of items filtered by FILTER.
-The values of FILTER can be `top' for top priority items, a cons
-of `top' and a number passed by the caller, `diary' for diary
-items, or `regexp' for items matching a regular expresion entered
-by the user.  The items can be from any categories in the current
-todo file or, with non-nil MULTIFILE, from several files.  If NEW
-is nil, visit an appropriate file containing the list of filtered
-items; if there is no such file, or with non-nil NEW, build the
-list and display it.
+(defun todos-gen-arglists (arglist)
+  "Return list of lists of non-nil atoms produced from ARGLIST.
+The elements of ARGLIST may be atoms or lists."
+  (let (arglists)
+    (while arglist
+      (let ((arg (pop arglist)))
+       (cond ((symbolp arg)
+              (setq arglists (if arglists
+                                 (mapcar (lambda (l) (push arg l)) arglists)
+                               (list (push arg arglists)))))
+             ((listp arg)
+              (setq arglists
+                    (mapcar (lambda (a)
+                              (if (= 1 (length arglists))
+                                  (apply (lambda (l) (push a l)) arglists)
+                                (mapcar (lambda (l) (push a l)) arglists)))
+                            arg))))))
+    (setq arglists (mapcar 'reverse (apply 'append (mapc 'car arglists))))))
 
-See the document strings of the commands
-`todos-filter-top-priorities', `todos-filter-diary-items',
-`todos-filter-regexp-items', and those of the corresponding
-multifile commands for further details."
-  (let* ((top (eq filter 'top))
-        (diary (eq filter 'diary))
-        (regexp (eq filter 'regexp))
-        (buf (cond (top todos-top-priorities-buffer)
-                   (diary todos-diary-items-buffer)
-                   (regexp todos-regexp-items-buffer)))
-        (flist (if multifile
-                   (or todos-filter-files
-                       (progn (todos-multiple-filter-files)
-                              todos-multiple-filter-files))
-                 (list todos-current-todos-file)))
-        (multi (> (length flist) 1))
-        (fname (if (equal flist 'quit)
-                   ;; Pressed `cancel' in t-m-f-f file selection dialog.
-                   (keyboard-quit)
-                 (concat todos-directory
-                         (mapconcat 'todos-short-file-name flist "-")
-                         (cond (top ".todt")
-                               (diary ".tody")
-                               (regexp ".todr")))))
-        (rxfiles (when regexp
-                   (directory-files todos-directory t ".*\\.todr$" t)))
-        (file-exists (or (file-exists-p fname) rxfiles)))
-    (cond ((and top new (natnump new))
-          (todos-filter-items-1 (cons 'top new) flist))
-         ((and (not new) file-exists)
-          (when (and rxfiles (> (length rxfiles) 1))
-            (let ((rxf (mapcar 'todos-short-file-name rxfiles)))
-              (setq fname (todos-absolute-file-name
-                           (completing-read "Choose a regexp items file: "
-                                            rxf) 'regexp))))
-          (find-file fname)
-          (todos-prefix-overlays)
-          (todos-check-filtered-items-file))
-         (t
-          (todos-filter-items-1 filter flist)))
-    (setq fname (replace-regexp-in-string "-" ", "
-                                         (todos-short-file-name fname)))
-    (rename-buffer (format (concat "%s for file" (if multi "s" "")
-                                  " \"%s\"") buf fname))))
+(defvar todos-insertion-commands-args-genlist
+  '(diary nonmarking (calendar date dayname) time (here region))
+  "Generator list for argument lists of item insertion commands.")
+
+(defvar todos-insertion-commands-args
+  (let ((argslist (todos-gen-arglists todos-insertion-commands-args-genlist))
+       res new)
+    (setq res (cl-remove-duplicates
+              (apply 'append (mapcar 'todos-powerset argslist)) :test 'equal))
+    (dolist (l res)
+      (unless (= 5 (length l))
+       (let ((v (make-vector 5 nil)) elt)
+         (while l
+           (setq elt (pop l))
+           (cond ((eq elt 'diary)
+                  (aset v 0 elt))
+                 ((eq elt 'nonmarking)
+                  (aset v 1 elt))
+                 ((or (eq elt 'calendar)
+                      (eq elt 'date)
+                      (eq elt 'dayname))
+                  (aset v 2 elt))
+                 ((eq elt 'time)
+                  (aset v 3 elt))
+                 ((or (eq elt 'here)
+                      (eq elt 'region))
+                  (aset v 4 elt))))
+         (setq l (append v nil))))
+      (setq new (append new (list l))))
+    new)
+  "List of all argument lists for Todos item insertion commands.")
+
+(defun todos-insertion-command-name (arglist)
+  "Generate Todos item insertion command name from ARGLIST."
+  (replace-regexp-in-string
+   "-\\_>" ""
+   (replace-regexp-in-string
+    "-+" "-"
+    ;; (concat "todos-item-insert-"
+    (concat "todos-insert-item-"
+           (mapconcat (lambda (e) (if e (symbol-name e))) arglist "-")))))
+
+(defvar todos-insertion-commands-names
+  (mapcar (lambda (l)
+          (todos-insertion-command-name l))
+         todos-insertion-commands-args)
+  "List of names of Todos item insertion commands.")
+
+(defmacro todos-define-insertion-command (&rest args)
+  "Generate item insertion command definitions from ARGS."
+  (let ((name (intern (todos-insertion-command-name args)))
+       (arg0 (nth 0 args))
+       (arg1 (nth 1 args))
+       (arg2 (nth 2 args))
+       (arg3 (nth 3 args))
+       (arg4 (nth 4 args)))
+    `(defun ,name (&optional arg &rest args)
+       "Todos item insertion command generated from ARGS.
+For descriptions of the individual arguments, their values, and
+their relation to key bindings, see `todos-basic-insert-item'."
+       (interactive (list current-prefix-arg))
+       (todos-basic-insert-item arg ',arg0 ',arg1 ',arg2 ',arg3 ',arg4))))
+
+(defvar todos-insertion-commands
+  (mapcar (lambda (c)
+           (eval `(todos-define-insertion-command ,@c)))
+         todos-insertion-commands-args)
+  "List of Todos item insertion commands.")
+
+(defvar todos-insertion-commands-arg-key-list
+  '(("diary" "y" "yy")
+    ("nonmarking" "k" "kk")
+    ("calendar" "c" "cc")
+    ("date" "d" "dd")
+    ("dayname" "n" "nn")
+    ("time" "t" "tt")
+    ("here" "h" "h")
+    ("region" "r" "r"))
+  "List of mappings of insertion command arguments to key sequences.")
 
-(defun todos-filter-items-1 (filter file-list)
-  "Build a list of items by applying FILTER to FILE-LIST.
-Internal subroutine called by `todos-filter-items', which passes
-the values of FILTER and FILE-LIST."
-  (let ((num (if (consp filter) (cdr filter) todos-top-priorities))
-       (buf (get-buffer-create todos-filtered-items-buffer))
-       (multifile (> (length file-list) 1))
-       regexp fname bufstr cat beg end done)
-    (if (null file-list)
-       (user-error "No files have been chosen for filtering")
-      (with-current-buffer buf
-       (erase-buffer)
-       (kill-all-local-variables)
-       (todos-filtered-items-mode))
-      (when (eq filter 'regexp)
-       (setq regexp (read-string "Enter a regular expression: ")))
-      (save-current-buffer
-       (dolist (f file-list)
-         ;; Before inserting file contents into temp buffer, save a modified
-         ;; buffer visiting it.
-         (let ((bf (find-buffer-visiting f)))
-           (when (buffer-modified-p bf)
-             (with-current-buffer bf (save-buffer))))
-         (setq fname (todos-short-file-name f))
-         (with-temp-buffer
-           (when (and todos-filter-done-items (eq filter 'regexp))
-             ;; If there is a corresponding archive file for the
-             ;; Todos file, insert it first and add identifiers for
-             ;; todos-go-to-source-item.
-             (let ((arch (concat (file-name-sans-extension f) ".toda")))
-               (when (file-exists-p arch)
-                 (insert-file-contents arch)
-                 ;; Delete Todos archive file categories sexp.
-                 (delete-region (line-beginning-position)
-                                (1+ (line-end-position)))
-                 (save-excursion
-                   (while (not (eobp))
-                     (when (re-search-forward
-                            (concat (if todos-filter-done-items
-                                        (concat "\\(?:" todos-done-string-start
-                                                "\\|" todos-date-string-start
-                                                "\\)")
-                                      todos-date-string-start)
-                                    todos-date-pattern "\\(?: "
-                                    diary-time-regexp "\\)?"
-                                    (if todos-filter-done-items
-                                        "\\]"
-                                      (regexp-quote todos-nondiary-end)) "?")
-                            nil t)
-                       (insert "(archive) "))
-                     (forward-line))))))
-           (insert-file-contents f)
-           ;; Delete Todos file categories sexp.
-           (delete-region (line-beginning-position) (1+ (line-end-position)))
-           (let (fnum)
-             ;; Unless the number of top priorities to show was
-             ;; passed by the caller, the file-wide value from
-             ;; `todos-top-priorities-overrides', if non-nil, overrides
-             ;; `todos-top-priorities'.
-             (unless (consp filter)
-               (setq fnum (or (nth 1 (assoc f todos-top-priorities-overrides))
-                              todos-top-priorities)))
-             (while (re-search-forward
-                     (concat "^" (regexp-quote todos-category-beg)
-                             "\\(.+\\)\n") nil t)
-               (setq cat (match-string 1))
-               (let (cnum)
-                 ;; Unless the number of top priorities to show was
-                 ;; passed by the caller, the category-wide value
-                 ;; from `todos-top-priorities-overrides', if non-nil,
-                 ;; overrides a non-nil file-wide value from
-                 ;; `todos-top-priorities-overrides' as well as
-                 ;; `todos-top-priorities'.
-                 (unless (consp filter)
-                   (let ((cats (nth 2 (assoc f todos-top-priorities-overrides))))
-                     (setq cnum (or (cdr (assoc cat cats)) fnum))))
-                 (delete-region (match-beginning 0) (match-end 0))
-                 (setq beg (point))    ; First item in the current category.
-                 (setq end (if (re-search-forward
-                                (concat "^" (regexp-quote todos-category-beg))
-                                nil t)
-                               (match-beginning 0)
-                             (point-max)))
-                 (goto-char beg)
-                 (setq done
-                       (if (re-search-forward
-                            (concat "\n" (regexp-quote todos-category-done))
-                            end t)
-                           (match-beginning 0)
-                         end))
-                 (unless (and todos-filter-done-items (eq filter 'regexp))
-                   ;; Leave done items.
-                   (delete-region done end)
-                   (setq end done))
-                 (narrow-to-region beg end)    ; Process only current category.
-                 (goto-char (point-min))
-                 ;; Apply the filter.
-                 (cond ((eq filter 'diary)
-                        (while (not (eobp))
-                          (if (looking-at (regexp-quote todos-nondiary-start))
-                              (todos-remove-item)
-                            (todos-forward-item))))
-                       ((eq filter 'regexp)
-                        (while (not (eobp))
-                          (if (looking-at todos-item-start)
-                              (if (string-match regexp (todos-item-string))
-                                  (todos-forward-item)
-                                (todos-remove-item))
-                            ;; Kill lines that aren't part of a todo or done
-                            ;; item (empty or todos-category-done).
-                            (delete-region (line-beginning-position)
-                                           (1+ (line-end-position))))
-                          ;; If last todo item in file matches regexp and
-                          ;; there are no following done items,
-                          ;; todos-category-done string is left dangling,
-                          ;; because todos-forward-item jumps over it.
-                          (if (and (eobp)
-                                   (looking-back
-                                    (concat (regexp-quote todos-done-string)
-                                            "\n")))
-                              (delete-region (point) (progn
-                                                       (forward-line -2)
-                                                       (point))))))
-                       (t ; Filter top priority items.
-                        (setq num (or cnum fnum num))
-                        (unless (zerop num)
-                          (todos-forward-item num))))
-                 (setq beg (point))
-                 ;; Delete non-top-priority items.
-                 (unless (member filter '(diary regexp))
-                   (delete-region beg end))
-                 (goto-char (point-min))
-                 ;; Add file (if using multiple files) and category tags to
-                 ;; item.
-                 (while (not (eobp))
-                   (when (re-search-forward
-                          (concat (if todos-filter-done-items
-                                      (concat "\\(?:" todos-done-string-start
-                                              "\\|" todos-date-string-start
-                                              "\\)")
-                                    todos-date-string-start)
-                                  todos-date-pattern "\\(?: " diary-time-regexp
-                                  "\\)?" (if todos-filter-done-items
-                                             "\\]"
-                                           (regexp-quote todos-nondiary-end))
-                                  "?")
-                          nil t)
-                     (insert " [")
-                     (when (looking-at "(archive) ") (goto-char (match-end 0)))
-                     (insert (if multifile (concat fname ":") "") cat "]"))
-                   (forward-line))
-                 (widen)))
-               (setq bufstr (buffer-string))
-               (with-current-buffer buf
-                 (let (buffer-read-only)
-                   (insert bufstr)))))))
-      (set-window-buffer (selected-window) (set-buffer buf))
-      (todos-prefix-overlays)
-      (goto-char (point-min)))))
+(defun todos-insertion-key-bindings (map)
+  "Generate key binding definitions for item insertion keymap MAP."
+  (dolist (c todos-insertion-commands)
+    (let* ((key "")
+          (cname (symbol-name c)))
+      (mapc (lambda (l)
+             (let ((arg (nth 0 l))
+                   (key1 (nth 1 l))
+                   (key2 (nth 2 l)))
+               (if (string-match (concat (regexp-quote arg) "\\_>") cname)
+                   (setq key (concat key key2)))
+               (if (string-match (concat (regexp-quote arg) ".+") cname)
+                   (setq key (concat key key1)))))
+           todos-insertion-commands-arg-key-list)
+      (if (string-match (concat (regexp-quote "todos-insert-item") "\\_>") cname)
+         (setq key (concat key "i")))
+      (define-key map key c))))
 
-(defun todos-set-top-priorities (&optional arg)
-  "Set number of top priorities shown by `todos-filter-top-priorities'.
-With non-nil ARG, set the number only for the current Todos
-category; otherwise, set the number for all categories in the
-current Todos file.
+;; -----------------------------------------------------------------------------
+;;; Todos minibuffer utilities
+;; -----------------------------------------------------------------------------
 
-Calling this function via either of the commands
-`todos-set-top-priorities-in-file' or
-`todos-set-top-priorities-in-category' is the recommended way to
-set the user customizable option `todos-top-priorities-overrides'."
-  (let* ((cat (todos-current-category))
-        (file todos-current-todos-file)
-        (rules todos-top-priorities-overrides)
-        (frule (assoc-string file rules))
-        (crule (assoc-string cat (nth 2 frule)))
-        (crules (nth 2 frule))
-        (cur (or (if arg (cdr crule) (nth 1 frule))
-                 todos-top-priorities))
-        (prompt (if arg (concat "Number of top priorities in this category"
-                                " (currently %d): ")
-                  (concat "Default number of top priorities per category"
-                                " in this file (currently %d): ")))
-        (new -1)
-        nrule)
-    (while (< new 0)
-      (let ((cur0 cur))
-       (setq new (read-number (format prompt cur0))
-             prompt "Enter a non-negative number: "
-             cur0 nil)))
-    (setq nrule (if arg
-                   (append (delete crule crules) (list (cons cat new)))
-                 (append (list file new) (list crules))))
-    (setq rules (cons (if arg
-                         (list file cur nrule)
-                       nrule)
-                     (delete frule rules)))
-    (customize-save-variable 'todos-top-priorities-overrides rules)
-    (todos-prefix-overlays)))
+(defcustom todos-y-with-space nil
+  "Non-nil means allow SPC to affirm a \"y or n\" question."
+  :type 'boolean
+  :group 'todos)
+
+(defun todos-y-or-n-p (prompt)
+  "Ask \"y or n\" question PROMPT and return t if answer is \"y\".
+Also return t if answer is \"Y\", but unlike `y-or-n-p', allow
+SPC to affirm the question only if option `todos-y-with-space' is
+non-nil."
+  (unless todos-y-with-space
+    (define-key query-replace-map " " 'ignore))
+  (prog1
+   (y-or-n-p prompt)
+   (define-key query-replace-map " " 'act)))
+
+(defun todos-category-completions (&optional archive)
+  "Return a list of completions for `todos-read-category'.
+Each element of the list is a cons of a category name and the
+file or list of files (as short file names) it is in.  The files
+are either the current (or if there is none, the default) todo
+file plus the files listed in `todos-category-completions-files',
+or, with non-nil ARCHIVE, the current archive file."
+  (let* ((curfile (or todos-current-todos-file
+                     (and todos-show-current-file
+                          todos-global-current-todos-file)
+                     (todos-absolute-file-name todos-default-todos-file)))
+        (files (or (unless archive
+                     (mapcar 'todos-absolute-file-name
+                             todos-category-completions-files))
+                   (list curfile)))
+        listall listf)
+    ;; If file was just added, it has no category completions.
+    (unless (zerop (buffer-size (find-buffer-visiting curfile)))
+      (unless (member curfile todos-archives)
+       (add-to-list 'files curfile))
+      (dolist (f files listall)
+       (with-current-buffer (find-file-noselect f 'nowarn)
+         ;; Ensure category is properly displayed in case user
+         ;; switches to file via a non-Todos command.  And if done
+         ;; items in category are visible, keep them visible.
+         (let ((done todos-show-with-done))
+           (when (> (buffer-size) (- (point-max) (point-min)))
+             (save-excursion
+               (goto-char (point-min))
+               (setq done (re-search-forward todos-done-string-start nil t))))
+           (let ((todos-show-with-done done))
+             (save-excursion (todos-category-select))))
+         (save-excursion
+           (save-restriction
+             (widen)
+             (goto-char (point-min))
+             (setq listf (read (buffer-substring-no-properties
+                                (line-beginning-position)
+                                (line-end-position)))))))
+       (mapc (lambda (elt) (let* ((cat (car elt))
+                                  (la-elt (assoc cat listall)))
+                             (if la-elt
+                                 (setcdr la-elt (append (list (cdr la-elt))
+                                                        (list f)))
+                               (push (cons cat f) listall))))
+             listf)))))
 
-(defconst todos-filtered-items-buffer "Todos filtered items"
-  "Initial name of buffer in Todos Filter Items mode.")
+(defun todos-read-file-name (prompt &optional archive mustmatch)
+  "Choose and return the name of a Todos file, prompting with PROMPT.
 
-(defconst todos-top-priorities-buffer "Todos top priorities"
-  "Buffer type string for `todos-filter-items'.")
+Show completions with TAB or SPC; the names are shown in short
+form but the absolute truename is returned.  With non-nil ARCHIVE
+return the absolute truename of a Todos archive file.  With non-nil
+MUSTMATCH the name of an existing file must be chosen;
+otherwise, a new file name is allowed."
+  (let* ((completion-ignore-case todos-completion-ignore-case)
+        (files (mapcar 'todos-short-file-name
+                       (if archive todos-archives todos-files)))
+        (file (completing-read prompt files nil mustmatch nil nil
+                               (if files
+                                   ;; If user hit RET without
+                                   ;; choosing a file, default to
+                                   ;; current or default file.
+                                   (todos-short-file-name
+                                    (or todos-current-todos-file
+                                        (and todos-show-current-file
+                                             todos-global-current-todos-file)
+                                        (todos-absolute-file-name
+                                         todos-default-todos-file)))
+                                 ;; Trigger prompt for initial file.
+                                 ""))))
+    (unless (file-exists-p todos-directory)
+      (make-directory todos-directory))
+    (unless mustmatch
+      (setq file (todos-validate-name file 'file)))
+    (setq file (file-truename (concat todos-directory file
+                                     (if archive ".toda" ".todo"))))))
 
-(defconst todos-diary-items-buffer "Todos diary items"
-  "Buffer type string for `todos-filter-items'.")
+(defun todos-read-category (prompt &optional match-type file)
+  "Choose and return a category name, prompting with PROMPT.
+Show completions for existing categories with TAB or SPC.
 
-(defconst todos-regexp-items-buffer "Todos regexp items"
-  "Buffer type string for `todos-filter-items'.")
+The argument MATCH-TYPE specifies the matching requirements on
+the category name: with the value `todo' or `archive' the name
+must complete to that of an existing todo or archive category,
+respectively; with the value `add' the name must not be that of
+an existing category; with all other values both existing and new
+valid category names are accepted.
 
-(defun todos-find-item (str)
-  "Search for filtered item STR in its saved Todos file.
-Return the list (FOUND FILE CAT), where CAT and FILE are the
-item's category and file, and FOUND is a cons cell if the search
-succeeds, whose car is the start of the item in FILE and whose
-cdr is `done', if the item is now a done item, `changed', if its
-text was truncated or augmented or, for a top priority item, if
-its priority has changed, and `same' otherwise."
-  (string-match (concat (if todos-filter-done-items
-                           (concat "\\(?:" todos-done-string-start "\\|"
-                                   todos-date-string-start "\\)")
-                         todos-date-string-start)
-                       todos-date-pattern "\\(?: " diary-time-regexp "\\)?"
-                       (if todos-filter-done-items
-                           "\\]"
-                         (regexp-quote todos-nondiary-end)) "?"
-                       "\\(?4: \\[\\(?3:(archive) \\)?\\(?2:.*:\\)?"
-                       "\\(?1:.*\\)\\]\\).*$") str)
-  (let ((cat (match-string 1 str))
-       (file (match-string 2 str))
-       (archive (string= (match-string 3 str) "(archive) "))
-       (filcat (match-string 4 str))
-       (tpriority 1)
-       (tpbuf (save-match-data (string-match "top" (buffer-name))))
-       found)
-    (setq str (replace-match "" nil nil str 4))
-    (when tpbuf
-      ;; Calculate priority of STR wrt its category.
-      (save-excursion
-       (while (search-backward filcat nil t)
-           (setq tpriority (1+ tpriority)))))
-    (setq file (if file
-                  (concat todos-directory (substring file 0 -1)
-                          (if archive ".toda" ".todo"))
-                (if archive
-                    (concat (file-name-sans-extension
-                             todos-global-current-todos-file) ".toda")
-                  todos-global-current-todos-file)))
-    (find-file-noselect file)
-    (with-current-buffer (find-buffer-visiting file)
-      (save-restriction
-       (widen)
-       (goto-char (point-min))
-       (let ((beg (re-search-forward
-                   (concat "^" (regexp-quote (concat todos-category-beg cat))
-                           "$")
-                   nil t))
-             (done (save-excursion
-                     (re-search-forward
-                      (concat "^" (regexp-quote todos-category-done)) nil t)))
-             (end (save-excursion
-                    (or (re-search-forward
-                         (concat "^" (regexp-quote todos-category-beg))
-                         nil t)
-                        (point-max)))))
-         (setq found (when (search-forward str end t)
-                       (goto-char (match-beginning 0))))
-         (when found
-           (setq found
-                 (cons found (if (> (point) done)
-                                 'done
-                               (let ((cpriority 1))
-                                 (when tpbuf
-                                   (save-excursion
-                                     ;; Not top item in category.
-                                     (while (> (point) (1+ beg))
-                                       (let ((opoint (point)))
-                                         (todos-backward-item)
-                                         ;; Can't move backward beyond
-                                         ;; first item in file.
-                                         (unless (= (point) opoint)
-                                           (setq cpriority (1+ cpriority)))))))
-                                 (if (and (= tpriority cpriority)
-                                          ;; Proper substring is not the same.
-                                          (string= (todos-item-string)
-                                                   str))
-                                     'same
-                                   'changed)))))))))
-      (list found file cat)))
+With non-nil argument FILE prompt for a file and complete only
+against categories in that file; otherwise complete against all
+categories from `todos-category-completions-files'."
+  ;; Allow SPC to insert spaces, for adding new category names.
+  (let ((map minibuffer-local-completion-map))
+    (define-key map " " nil)
+    (let* ((add (eq match-type 'add))
+          (archive (eq match-type 'archive))
+          (file0 (when (and file (> (length todos-files) 1))
+                   (todos-read-file-name (concat "Choose a" (if archive
+                                                                "n archive"
+                                                              " todo")
+                                                 " file: ") archive t)))
+          (completions (unless file0 (todos-category-completions archive)))
+          (categories (cond (file0
+                             (with-current-buffer
+                                 (find-file-noselect file0 'nowarn)
+                               (let ((todos-current-todos-file file0))
+                                 todos-categories)))
+                            ((and add (not file))
+                             (with-current-buffer
+                                 (find-file-noselect todos-current-todos-file)
+                               todos-categories))
+                            (t
+                             completions)))
+          (completion-ignore-case todos-completion-ignore-case)
+          (cat (completing-read prompt categories nil
+                                (eq match-type 'todo) nil nil
+                                ;; Unless we're adding a category via
+                                ;; todos-add-category, set default
+                                ;; for existing categories to the
+                                ;; current category of the chosen
+                                ;; file or else of the current file.
+                                (if (and categories (not add))
+                                    (with-current-buffer
+                                        (find-file-noselect
+                                         (or file0
+                                             todos-current-todos-file
+                                             (todos-absolute-file-name
+                                              todos-default-todos-file)))
+                                      (todos-current-category))
+                                  ;; Trigger prompt for initial category.
+                                  "")))
+          (catfil (cdr (assoc cat completions)))
+          (str "Category \"%s\" from which file (TAB for choices)? "))
+      ;; If we do category completion and the chosen category name
+      ;; occurs in more than one file, prompt to choose one file.
+      (unless (or file0 add (not catfil))
+       (setq file0 (file-truename
+                    (if (atom catfil)
+                        catfil
+                      (todos-absolute-file-name
+                       (let ((files (mapcar 'todos-short-file-name catfil)))
+                         (completing-read (format str cat) files)))))))
+      ;; Default to the current file.
+      (unless file0 (setq file0 todos-current-todos-file))
+      ;; First validate only a name passed interactively from
+      ;; todos-add-category, which must be of a nonexisting category.
+      (unless (and (assoc cat categories) (not add))
+       ;; Validate only against completion categories.
+       (let ((todos-categories categories))
+         (setq cat (todos-validate-name cat 'category)))
+       ;; When user enters a nonexisting category name by jumping or
+       ;; moving, confirm that it should be added, then validate.
+       (unless add
+         (if (todos-y-or-n-p (format "Add new category \"%s\" to file \"%s\"? "
+                               cat (todos-short-file-name file0)))
+             (progn
+               (when (assoc cat categories)
+                 (let ((todos-categories categories))
+                   (setq cat (todos-validate-name cat 'category))))
+               ;; Restore point and narrowing after adding new
+               ;; category, to avoid moving to beginning of file when
+               ;; moving marked items to a new category
+               ;; (todos-move-item).
+               (save-excursion
+                 (save-restriction
+                   (todos-add-category file0 cat))))
+           ;; If we decide not to add a category, exit without returning.
+           (keyboard-quit))))
+      (cons cat file0))))
+
+(defun todos-validate-name (name type)
+  "Prompt for new NAME for TYPE until it is valid, then return it.
+TYPE can be either of the symbols `file' or `category'."
+  (let ((categories todos-categories)
+       (files (mapcar 'todos-short-file-name todos-files))
+       prompt)
+    (while
+       (and
+        (cond ((string= "" name)
+               (setq prompt
+                     (cond ((eq type 'file)
+                            (if files
+                                "Enter a non-empty file name: "
+                              ;; Empty string passed by todos-show to
+                              ;; prompt for initial Todos file.
+                              (concat "Initial file name ["
+                                      todos-initial-file "]: ")))
+                           ((eq type 'category)
+                            (if categories
+                                "Enter a non-empty category name: "
+                              ;; Empty string passed by todos-show to
+                              ;; prompt for initial category of a new
+                              ;; Todos file.
+                              (concat "Initial category name ["
+                                      todos-initial-category "]: "))))))
+              ((string-match "\\`\\s-+\\'" name)
+               (setq prompt
+                     "Enter a name that does not contain only white space: "))
+              ((and (eq type 'file) (member name files))
+               (setq prompt "Enter a non-existing file name: "))
+              ((and (eq type 'category) (assoc name categories))
+               (setq prompt "Enter a non-existing category name: ")))
+        (setq name (if (or (and (eq type 'file) files)
+                           (and (eq type 'category) categories))
+                       (completing-read prompt (cond ((eq type 'file)
+                                                      files)
+                                                     ((eq type 'category)
+                                                      categories)))
+                     ;; Offer default initial name.
+                     (completing-read prompt (if (eq type 'file)
+                                                 files
+                                               categories)
+                                      nil nil (if (eq type 'file)
+                                                  todos-initial-file
+                                                todos-initial-category))))))
+    name))
 
-(defun todos-check-filtered-items-file ()
-  "Check if filtered items file is up to date and a show suitable message."
-  ;; (catch 'old
-  (let ((count 0))
-    (while (not (eobp))
-      (let* ((item (todos-item-string))
-            (found (car (todos-find-item item))))
-       (unless (eq (cdr found) 'same)
-         (save-excursion
-           (overlay-put (make-overlay (todos-item-start) (todos-item-end))
-                        'face 'todos-search))
-         (setq count (1+ count))))
-         ;; (throw 'old (message "The marked item is not up to date.")))
-      (todos-forward-item))
-    (if (zerop count)
-       (message "Filtered items file is up to date.")
-      (message (concat "The highlighted item" (if (= count 1) " is " "s are ")
-                      "not up to date."
-                      ;; "\nType <return> on item for details."
-                      )))))
+;; Adapted from calendar-read-date and calendar-date-string.
+(defun todos-read-date (&optional arg mo yr)
+  "Prompt for Gregorian date and return it in the current format.
 
-(defun todos-filter-items-filename ()
-  "Return absolute file name for saving this Filtered Items buffer."
-  (let ((bufname (buffer-name)))
-    (string-match "\"\\([^\"]+\\)\"" bufname)
-    (let* ((filename-str (substring bufname (match-beginning 1) (match-end 1)))
-          (filename-base (replace-regexp-in-string ", " "-" filename-str))
-          (top-priorities (string-match "top priorities" bufname))
-          (diary-items (string-match "diary items" bufname))
-          (regexp-items (string-match "regexp items" bufname)))
-      (when regexp-items
-       (let ((prompt (concat "Enter a short identifying string"
-                             " to make this file name unique: ")))
-         (setq filename-base (concat filename-base "-" (read-string prompt)))))
-      (concat todos-directory filename-base
-             (cond (top-priorities ".todt")
-                   (diary-items ".tody")
-                   (regexp-items ".todr"))))))
+With non-nil ARG, prompt for and return only the date component
+specified by ARG, which can be one of these symbols:
+`month' (prompt for name, return name or number according to
+value of `calendar-date-display-form'), `day' of month, or
+`year'.  The value of each of these components can be `*',
+indicating an unspecified month, day, or year.
 
-(defun todos-save-filtered-items-buffer ()
-  "Save current Filtered Items buffer to a file.
-If the file already exists, overwrite it only on confirmation."
-  (let ((filename (or (buffer-file-name) (todos-filter-items-filename))))
-    (write-file filename t)))
+When ARG is `day', non-nil arguments MO and YR determine the
+number of the last the day of the month."
+  (let (year monthname month day
+            dayname)                   ; Needed by calendar-date-display-form.
+    (when (or (not arg) (eq arg 'year))
+      (while (if (natnump year) (< year 1) (not (eq year '*)))
+       (setq year (read-from-minibuffer
+                   "Year (>0 or RET for this year or * for any year): "
+                   nil nil t nil (number-to-string
+                                  (calendar-extract-year
+                                   (calendar-current-date)))))))
+    (when (or (not arg) (eq arg 'month))
+      (let* ((marray todos-month-name-array)
+            (mlist (append marray nil))
+            (mabarray todos-month-abbrev-array)
+            (mablist (append mabarray nil))
+            (completion-ignore-case todos-completion-ignore-case))
+       (setq monthname (completing-read
+                        "Month name (RET for current month, * for any month): "
+                        ;; (mapcar 'list (append marray nil))
+                        mlist nil t nil nil
+                        (calendar-month-name (calendar-extract-month
+                                              (calendar-current-date)) t))
+             ;; month (cdr (assoc-string
+             ;;                  monthname (calendar-make-alist marray nil nil
+             ;;                                                 abbrevs))))))
+             month (1+ (- (length mlist)
+                          (length (or (member monthname mlist)
+                                      (member monthname mablist))))))
+       (setq monthname (aref mabarray (1- month)))))
+    (when (or (not arg) (eq arg 'day))
+      (let ((last (let ((mm (or month mo))
+                       (yy (or year yr)))
+                   ;; If month is unspecified, use a month with 31
+                   ;; days for checking day of month input.  Does
+                   ;; Calendar do anything special when * is
+                   ;; currently a shorter month?
+                   (if (= mm 13) (setq mm 1))
+                   ;; If year is unspecified, use a leap year to
+                   ;; allow Feb. 29.
+                   (if (eq year '*) (setq yy 2012))
+                   (calendar-last-day-of-month mm yy))))
+       (while (if (natnump day) (or (< day 1) (> day last)) (not (eq day '*)))
+         (setq day (read-from-minibuffer
+                    (format "Day (1-%d or RET for today or * for any day): "
+                            last)
+                    nil nil t nil (number-to-string
+                                   (calendar-extract-day
+                                    (calendar-current-date))))))))
+    ;; Stringify read values (monthname is already a string).
+    (and year (setq year (if (eq year '*)
+                            (symbol-name '*)
+                          (number-to-string year))))
+    (and day (setq day (if (eq day '*)
+                          (symbol-name '*)
+                        (number-to-string day))))
+    (and month (setq month (if (eq month '*)
+                              (symbol-name '*)
+                            (number-to-string month))))
+    (if arg
+       (cond ((eq arg 'year) year)
+             ((eq arg 'day) day)
+             ((eq arg 'month)
+              (if (memq 'month calendar-date-display-form)
+                  month
+                monthname)))
+      (mapconcat 'eval calendar-date-display-form ""))))
+
+(defun todos-read-dayname ()
+  "Choose name of a day of the week with completion and return it."
+  (let ((completion-ignore-case todos-completion-ignore-case))
+    (completing-read "Enter a day name: "
+                    (append calendar-day-name-array nil)
+                    nil t)))
+  
+(defun todos-read-time ()
+  "Prompt for and return a valid clock time as a string.
+
+Valid time strings are those matching `diary-time-regexp'.
+Typing `<return>' at the prompt returns the current time, if the
+user option `todos-always-add-time-string' is non-nil, otherwise
+the empty string (i.e., no time string)."
+  (let (valid answer)
+    (while (not valid)
+      (setq answer (read-string "Enter a clock time: " nil nil
+                               (when todos-always-add-time-string
+                                 (substring (current-time-string) 11 16))))
+      (when (or (string= "" answer)
+               (string-match diary-time-regexp answer))
+       (setq valid t)))
+    answer))
 
 ;; -----------------------------------------------------------------------------
-;;; Customization groups and set functions
+;;; Customization groups and utilities
 ;; -----------------------------------------------------------------------------
 
 (defgroup todos nil
@@ -5866,20 +5798,51 @@ If the file already exists, overwrite it only on confirmation."
                  (hl-line-mode 1)
                (hl-line-mode -1)))))))))
 
+(defun todos-reevaluate-filelist-defcustoms ()
+  "Reevaluate defcustoms that provide choice list of Todos files."
+  (custom-set-default 'todos-default-todos-file
+                     (symbol-value 'todos-default-todos-file))
+  (todos-reevaluate-default-file-defcustom)
+  (custom-set-default 'todos-filter-files (symbol-value 'todos-filter-files))
+  (todos-reevaluate-filter-files-defcustom)
+  (custom-set-default 'todos-category-completions-files
+                     (symbol-value 'todos-category-completions-files))
+  (todos-reevaluate-category-completions-files-defcustom))
+
+(defun todos-reevaluate-default-file-defcustom ()
+  "Reevaluate defcustom of `todos-default-todos-file'.
+Called after adding or deleting a Todos file."
+  (eval (defcustom todos-default-todos-file (car (funcall todos-files-function))
+         "Todos file visited by first session invocation of `todos-show'."
+         :type `(radio ,@(mapcar (lambda (f) (list 'const f))
+                                 (mapcar 'todos-short-file-name
+                                         (funcall todos-files-function))))
+         :group 'todos)))
+
+(defun todos-reevaluate-category-completions-files-defcustom ()
+  "Reevaluate defcustom of `todos-category-completions-files'.
+Called after adding or deleting a Todos file."
+  (eval (defcustom todos-category-completions-files nil
+  "List of files for building `todos-read-category' completions."
+         :type `(set ,@(mapcar (lambda (f) (list 'const f))
+                               (mapcar 'todos-short-file-name
+                                       (funcall todos-files-function))))
+         :group 'todos)))
+
+(defun todos-reevaluate-filter-files-defcustom ()
+  "Reevaluate defcustom of `todos-filter-files'.
+Called after adding or deleting a Todos file."
+  (eval (defcustom todos-filter-files nil
+         "List of files for multifile item filtering."
+         :type `(set ,@(mapcar (lambda (f) (list 'const f))
+                               (mapcar 'todos-short-file-name
+                                       (funcall todos-files-function))))
+         :group 'todos)))
+
 ;; -----------------------------------------------------------------------------
 ;;; Font locking
 ;; -----------------------------------------------------------------------------
 
-(defun todos-date-string-matcher (lim)
-  "Search for Todos date string within LIM for font-locking."
-  (re-search-forward
-   (concat todos-date-string-start "\\(?1:" todos-date-pattern "\\)") lim t))
-
-(defun todos-time-string-matcher (lim)
-  "Search for Todos time string within LIM for font-locking."
-  (re-search-forward (concat todos-date-string-start todos-date-pattern
-                            " \\(?1:" diary-time-regexp "\\)") lim t))
-
 (defun todos-nondiary-marker-matcher (lim)
   "Search for Todos nondiary markers within LIM for font-locking."
   (re-search-forward (concat "^\\(?1:" (regexp-quote todos-nondiary-start) "\\)"
@@ -5892,6 +5855,16 @@ If the file already exists, overwrite it only on confirmation."
   (re-search-forward (concat "^\\(?1:" (regexp-quote diary-nonmarking-symbol)
                             "\\)" todos-date-pattern) lim t))
 
+(defun todos-date-string-matcher (lim)
+  "Search for Todos date string within LIM for font-locking."
+  (re-search-forward
+   (concat todos-date-string-start "\\(?1:" todos-date-pattern "\\)") lim t))
+
+(defun todos-time-string-matcher (lim)
+  "Search for Todos time string within LIM for font-locking."
+  (re-search-forward (concat todos-date-string-start todos-date-pattern
+                            " \\(?1:" diary-time-regexp "\\)") lim t))
+
 (defun todos-diary-expired-matcher (lim)
   "Search for expired diary item date within LIM for font-locking."
   (when (re-search-forward (concat "^\\(?:"
@@ -5941,15 +5914,14 @@ Filtered Items mode following todo (not done) items."
                                 "\\)? \\(?1:\\[.+\\]\\)")
                         lim t)))
 
-(defvar todos-diary-expired-face 'todos-diary-expired)
+(defvar todos-nondiary-face 'todos-nondiary)
 (defvar todos-date-face 'todos-date)
 (defvar todos-time-face 'todos-time)
-(defvar todos-nondiary-face 'todos-nondiary)
-(defvar todos-category-string-face 'todos-category-string)
+(defvar todos-diary-expired-face 'todos-diary-expired)
+(defvar todos-done-sep-face 'todos-done-sep)
 (defvar todos-done-face 'todos-done)
 (defvar todos-comment-face 'todos-comment)
-(defvar todos-done-sep-face 'todos-done-sep)
-
+(defvar todos-category-string-face 'todos-category-string)
 (defvar todos-font-lock-keywords
   (list
    '(todos-nondiary-marker-matcher 1 todos-nondiary-face t)
@@ -5968,7 +5940,7 @@ Filtered Items mode following todo (not done) items."
   "Font-locking for Todos modes.")
 
 ;; -----------------------------------------------------------------------------
-;;; Key maps and menus
+;;; Key binding
 ;; -----------------------------------------------------------------------------
 
 (defvar todos-insertion-map
@@ -6133,6 +6105,7 @@ Filtered Items mode following todo (not done) items."
     map)
   "Todos Filtered Items mode keymap.")
 
+;; FIXME: Is it worth having a menu and if so, which commands?
 ;; (easy-menu-define
 ;;   todos-menu todos-mode-map "Todos Menu"
 ;;   '("Todos"
@@ -6192,12 +6165,9 @@ Filtered Items mode following todo (not done) items."
 ;;     ))
 
 ;; -----------------------------------------------------------------------------
-;;; Mode local variables and hook functions
+;;; Hook functions and mode definitions 
 ;; -----------------------------------------------------------------------------
 
-(defvar todos-current-todos-file nil
-  "Variable holding the name of the currently active Todos file.")
-
 (defun todos-show-current-file ()
   "Visit current instead of default Todos file with `todos-show'.
 This function is added to `pre-command-hook' when user option
@@ -6237,25 +6207,6 @@ This function is added to `kill-buffer-hook' in Todos mode."
          (or (car todos-file-buffers)
              (todos-absolute-file-name todos-default-todos-file)))))
 
-(defvar todos-categories nil
-  "Alist of categories in the current Todos file.
-The elements are cons cells whose car is a category name and
-whose cdr is a vector of the category's item counts.  These are,
-in order, the numbers of todo items, of todo items included in
-the Diary, of done items and of archived items.")
-
-(defvar todos-categories-with-marks nil
-  "Alist of categories and number of marked items they contain.")
-
-(defvar todos-category-number 1
-  "Variable holding the number of the current Todos category.
-Todos categories are numbered starting from 1.")
-
-(defvar todos-show-done-only nil
-  "If non-nil display only done items in current category.
-Set by the command `todos-toggle-view-done-only' and used by
-`todos-category-select'.")
-
 (defun todos-reset-and-enable-done-separator ()
   "Show resized done items separator overlay after window change.
 Added to `window-configuration-change-hook' in `todos-mode'."
@@ -6264,10 +6215,6 @@ Added to `window-configuration-change-hook' in `todos-mode'."
       (setq todos-done-separator (todos-done-separator))
       (save-match-data (todos-reset-done-separator sep)))))
 
-;; -----------------------------------------------------------------------------
-;;; Mode definitions
-;; -----------------------------------------------------------------------------
-
 (defun todos-modes-set-1 ()
   "Make some settings that apply to multiple Todos modes."
   (setq-local font-lock-defaults '(todos-font-lock-keywords t))