branch: externals/taxy commit 2f18dce603e4a66b1fa8cc2bdc7398a08031ec5a Author: Adam Porter <a...@alphapapa.net> Commit: Adam Porter <a...@alphapapa.net>
Examples: Add taxy-org-ql-view This will eventually be part of org-ql itself, but until then the WIP can live here. --- .elpaignore | 3 +- examples/taxy-org-ql-view.el | 166 +++++++++++++++++++++++++++++++++++++++++++ 2 files changed, 168 insertions(+), 1 deletion(-) diff --git a/.elpaignore b/.elpaignore index 68ef0e6..a8ea314 100644 --- a/.elpaignore +++ b/.elpaignore @@ -1,2 +1,3 @@ examples/musicy.el -examples/magit-loggy.el \ No newline at end of file +examples/magit-loggy.el +examples/taxy-org-ql-view.el \ No newline at end of file diff --git a/examples/taxy-org-ql-view.el b/examples/taxy-org-ql-view.el new file mode 100644 index 0000000..69a6aff --- /dev/null +++ b/examples/taxy-org-ql-view.el @@ -0,0 +1,166 @@ +;;; taxy-org-ql-view.el --- -*- lexical-binding: t; -*- + +;; Copyright (C) 2021 Adam Porter + +;; Author: Adam Porter <a...@alphapapa.net> +;; Keywords: + +;; This program is free software; you can redistribute it and/or modify +;; it under the terms of the GNU General Public License as published by +;; the Free Software Foundation, either version 3 of the License, or +;; (at your option) any later version. + +;; This program is distributed in the hope that it will be useful, +;; but WITHOUT ANY WARRANTY; without even the implied warranty of +;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +;; GNU General Public License for more details. + +;; You should have received a copy of the GNU General Public License +;; along with this program. If not, see <https://www.gnu.org/licenses/>. + +;;; Commentary: + +;; + +;;; Code: + +;;;; Requirements + +(require 'org-ql-view) + +(require 'taxy) +(require 'taxy-magit-section) + +;;;; Structs + +(cl-defstruct (taxy-org-ql-view-section + (:include taxy-magit-section + (format-fn #'org-ql-view--format-element) + (indent 2) + (make #'make-taxy-org-ql-view-section)))) + +;;;; Macros + +;;;; Defining taxy keys with macro + +(defvar taxy-org-ql-view-keys nil) + +(defmacro taxy-org-ql-view-define-key (name &rest body) + "Define a `taxy-org-ql-view' key function by NAME having BODY. +Within BODY, `element' is bound to the `org-element' element +being tested. + +Defines a function named `taxy-org-ql--predicate-NAME', and adds +an entry to `taxy-org-ql-view-keys' mapping NAME to the new +function symbol." + (declare (indent defun)) + (let ((fn-symbol (intern (format "taxy-org-ql--predicate-%s" name))) + (fn `(lambda (element) + ,@body))) + `(progn + (fset ',fn-symbol ,fn) + (setf (map-elt taxy-org-ql-view-keys ',name) ',fn-symbol)))) + +(taxy-org-ql-view-define-key todo + "Return the to-do keyword for ELEMENT." + (org-element-property :todo-keyword element)) + +(taxy-org-ql-view-define-key priority + "Return ELEMENT's priority as a string." + (when-let ((priority-number (org-element-property :priority element))) + ;; FIXME: Priority numbers may be wildly larger, right? + (char-to-string priority-number))) + +(taxy-org-ql-view-define-key planning-month + "Return ELEMENT's planning-date month, or nil. +Returns in format \"%Y-%m (%B)\"." + (when-let ((planning-element (or (org-element-property :deadline element) + (org-element-property :scheduled element) + (org-element-property :closed element)))) + (ts-format "%Y-%m (%B)" (ts-parse-org-element planning-element)))) + +(taxy-org-ql-view-define-key planning-year + "Return ELEMENT's planning-date year, or nil. +Returns in format \"%Y\"." + (when-let ((planning-element (or (org-element-property :deadline element) + (org-element-property :scheduled element) + (org-element-property :closed element)))) + (ts-format "%Y" (ts-parse-org-element planning-element)))) + +(taxy-org-ql-view-define-key planning-date + "Return ELEMENT's planning date, or nil. +Returns in format \"%Y-%m-%d\"." + (when-let ((planning-element (or (org-element-property :deadline element) + (org-element-property :scheduled element) + (org-element-property :closed element)))) + (ts-format "%Y-%m-%d" (ts-parse-org-element planning-element)))) + +(defun taxy-org-ql-view-take-fn (keys) + "Return a `taxy' \"take\" function for KEYS. +Each of KEYS should be a function alias defined in +`taxy-org-ql-view-keys', or a list of such KEY-FNS (recursively, +ad infinitum, approximately)." + (cl-labels ((quote-fn + (fn) (cl-typecase fn + (symbol fn) + (list (cons 'list (mapcar #'quote-fn fn)))))) + (setf keys (mapcar #'quote-fn keys))) + (let ((macrolets (cl-loop for (name . fn) in taxy-org-ql-view-keys + collect `(,name ',fn)))) + ;; Is using (cadr (macroexpand-all ...)) really better than `eval'? + (cadr (macroexpand-all `(cl-symbol-macrolet (,@macrolets) + (lambda (item taxy) + (taxy-take-keyed (list ,@keys) item taxy))))))) + +(defun taxy-org-ql-view-make-taxy (name keys) + "Return a dynamic `taxy-org-ql-view-section' taxy named NAME having KEYS. +KEYS is passed to `taxy-org-ql-view-take-fn', which see." + (declare (indent defun)) + (make-taxy-org-ql-view-section + :name name + :take (taxy-org-ql-view-take-fn keys))) + +;;;; Variables + +;;;; Customization + +;;;; Commands + +;;;; Functions + +(cl-defun taxy-org-ql-search + (buffers-or-files query &key taxy-keys sort) + "Show Org QL QUERY on BUFFERS-OR-FILES with `taxy-org-ql-view'." + (declare (indent defun)) + (let* ((title (format "Query:%S In:%s" query buffers-or-files)) + (taxy (taxy-org-ql-view-make-taxy title + taxy-keys)) + (items (org-ql-select buffers-or-files query + :action 'element-with-markers + :sort sort)) + (buffer-name (format "*Taxy Org QL View: %s" title))) + (when (get-buffer buffer-name) + ;; Reusing an existing magit-section buffer seems to cause a lot + ;; of GC, so just kill it if it already exists. + (kill-buffer buffer-name)) + (with-current-buffer (get-buffer-create buffer-name) + (let ((inhibit-read-only t)) + (erase-buffer) + (delete-all-overlays) + (magit-section-mode) + (use-local-map (make-composed-keymap (list org-ql-view-map magit-section-mode-map))) + (taxy-magit-section-insert + (thread-last taxy + (taxy-fill items) + (taxy-mapc* (lambda (taxy) + (setf (taxy-taxys taxy) + (cl-sort (taxy-taxys taxy) #'string< + :key #'taxy-name))))) + :objects 'last)) + (pop-to-buffer (current-buffer))))) + +;;;; Footer + +(provide 'taxy-org-ql-view) + +;;; taxy-org-ql-view.el ends here