anki-editor/anki-editor.el

1371 lines
53 KiB
EmacsLisp
Raw Normal View History

2018-06-10 15:13:00 +02:00
;;; anki-editor.el --- Minor mode for making Anki cards with Org -*- lexical-binding: t; -*-
2022-11-03 19:18:04 +01:00
;; Copyright (C) 2018-2022 Lei Tan <louietanlei[at]gmail[dot]com>
2018-06-10 15:13:00 +02:00
;; Author: Lei Tan
;; Version: 0.3.3
2018-01-15 15:11:02 +01:00
;; URL: https://github.com/louietan/anki-editor
2022-11-03 19:18:04 +01:00
;; Package-Requires: ((emacs "25.1"))
;; 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.
2018-01-15 15:11:02 +01:00
;;
2022-11-03 19:18:04 +01:00
;; 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 GNU Emacs. If not, see <http://www.gnu.org/licenses/>.
2018-01-15 15:11:02 +01:00
;;; Commentary:
2019-11-01 15:59:15 +01:00
;;
2019-09-23 11:13:10 +02:00
;; This package is for users of both Emacs and Anki, who'd like to
;; make Anki cards in Org mode. With this package, Anki cards can be
2022-11-03 19:18:04 +01:00
;; made from an Org buffer like below (inspired by org-drill):
2018-01-15 15:11:02 +01:00
;;
2019-09-23 11:13:10 +02:00
;; * Sample :emacs:lisp:programming:
2018-06-10 15:13:00 +02:00
;; :PROPERTIES:
;; :ANKI_DECK: Computing
;; :ANKI_NOTE_TYPE: Basic
;; :END:
;; ** Front
2019-09-23 11:13:10 +02:00
;; How to say "hello world" in elisp?
2018-06-10 15:13:00 +02:00
;; ** Back
;; #+BEGIN_SRC emacs-lisp
;; (message "Hello, world!")
;; #+END_SRC
;;
2018-06-10 15:13:00 +02:00
;; This package extends Org-mode's built-in HTML backend to generate
;; HTML for contents of note fields with specific syntax (e.g. latex)
2019-09-23 11:13:10 +02:00
;; translated to Anki style.
;;
;; For this package to work, you have to setup these external dependencies:
2022-11-03 19:18:04 +01:00
;; - Anki
2019-09-23 11:13:10 +02:00
;; - AnkiConnect, an Anki addon that runs an RPC server over HTTP to expose
2022-11-03 19:18:04 +01:00
;; Anki functions as APIs, for installation instructions see
;; https://github.com/FooSoft/anki-connect#installation
;; - curl
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
2018-01-15 15:11:02 +01:00
;;; Code:
2017-12-27 17:20:01 +01:00
2018-05-07 15:09:52 +02:00
(require 'cl-lib)
2017-12-27 17:20:01 +01:00
(require 'json)
(require 'org-element)
2018-02-04 02:10:40 +01:00
(require 'ox)
2018-05-22 15:09:36 +02:00
(require 'ox-html)
2017-12-27 17:20:01 +01:00
2018-01-21 12:22:42 +01:00
(defgroup anki-editor nil
"Customizations for anki-editor."
:group 'org)
2022-11-03 19:18:04 +01:00
(defcustom anki-editor-break-consecutive-braces-in-latex nil
"If non-nil, automatically separate consecutive `}' in latex by spaces.
This prevents early closing of cloze."
2020-02-23 08:37:40 +01:00
:type 'boolean)
2022-11-03 19:18:04 +01:00
(defcustom anki-editor-org-tags-as-anki-tags t
2020-02-23 08:37:40 +01:00
"If nil, tags of entries won't be counted as Anki tags."
:type 'boolean)
2022-11-03 19:18:04 +01:00
(defcustom anki-editor-protected-tags '("marked" "leech")
"A list of protected tags to not delete from Anki.
These won't be deleted from Anki even when they're absent in Org entries.
Useful for special tags like `marked' and `leech'."
:type '(repeat string))
(defcustom anki-editor-ignored-org-tags
(append org-export-select-tags org-export-exclude-tags)
2022-11-03 19:18:04 +01:00
"A list of Org tags that are ignored when constructing notes."
:type '(repeat string))
2022-11-03 19:18:04 +01:00
(defcustom anki-editor-api-host "127.0.0.1"
"The network address AnkiConnect is listening on."
2020-02-23 08:37:40 +01:00
:type 'string)
2018-01-18 10:21:49 +01:00
2022-11-03 19:18:04 +01:00
(defcustom anki-editor-api-port "8765"
"The port number AnkiConnect is listening on."
2020-02-23 08:37:40 +01:00
:type 'string)
2017-12-27 17:20:01 +01:00
2019-11-11 16:03:55 +01:00
(defcustom anki-editor-latex-style 'builtin
"The style of latex to translate into."
:type '(radio (const :tag "Built-in" builtin)
(const :tag "MathJax" mathjax)))
2017-12-27 17:20:01 +01:00
2020-03-01 17:10:53 +01:00
(defcustom anki-editor-include-default-style t
2022-11-03 19:18:04 +01:00
"Whether to include the default style with `anki-editor-copy-styles'.
The default style is specified in `org-html-style-default'.
2020-03-01 17:10:53 +01:00
For example, you might want to turn this off when you are going to
provide your custom styles in `anki-editor-html-head'."
:type 'boolean)
(defcustom anki-editor-html-head nil
2022-11-03 19:18:04 +01:00
"Additional html tags to append with `anki-editor-copy-styles'.
Can be used to add custom styles and scripts to card styles."
2020-03-01 17:10:53 +01:00
:type 'string)
2020-03-09 14:36:27 +01:00
(defcustom anki-editor-note-match nil
2022-11-03 19:18:04 +01:00
"Additional matching string for mapping through Anki note headings.
2020-03-09 14:36:27 +01:00
A leading logical operator like `+' or `&' is required."
:type 'string)
(defcustom anki-editor-prepend-note-heading nil
"Prepend note heading to contents before first (sub)heading.
This is only in effect when exactly one note-type field is not found
among the note subheadings and there is content before the first subheading."
:type 'boolean)
(defcustom anki-editor-insert-note-always-use-content nil
"Whether to always make use of content before (sub)heading.
See `anki-editor-insert-note', whose behavior this controls."
:type 'boolean)
(defcustom anki-editor-default-note-type "Basic"
"Default note type when creating anki-editor notes in org.
2022-11-03 19:18:04 +01:00
Only used when no ANKI_DEFAULT_NOTE_TYPE property is inherited."
:type 'string)
(defcustom anki-editor-gui-browse-ensure-foreground t
"Ensure that `anki-editor-gui-browse' opens in foreground."
:type 'boolean)
2018-01-19 13:32:28 +01:00
2019-11-01 15:59:15 +01:00
;;; AnkiConnect
2020-03-09 11:08:33 +01:00
(defconst anki-editor-api-version 6)
2019-11-03 10:24:00 +01:00
2020-02-10 12:51:02 +01:00
(cl-defun anki-editor--fetch (url
&rest settings
2020-02-24 06:39:31 +01:00
&key
(type "GET")
data success _error
(parser 'buffer-string)
2020-02-10 12:51:02 +01:00
&allow-other-keys)
2022-11-03 19:18:04 +01:00
"Fetch URL using curl.
The api is borrowed from request.el."
;; This exists because request.el's sync mode calls curl asynchronously under
;; the hood, which doesn't work on some machines (like mine) where the process
;; sentinel never gets called. After some debugging of Emacs, it seems that in
;; 'process.c' the pselect syscall to the file descriptor of inotify used by
;; 'autorevert' always returns a nonzero value and causes 'status_notify' never
;; being called. To determine whether it's a bug in Emacs and make a patch
;; requires more digging.
2020-02-10 12:51:02 +01:00
(let ((tempfile (make-temp-file "emacs-anki-editor"))
(responsebuf (generate-new-buffer " *anki-editor-curl*")))
2020-02-24 06:39:31 +01:00
(when data
(with-temp-file tempfile
(setq buffer-file-coding-system 'utf-8)
(set-buffer-multibyte t)
(insert data)))
2020-02-10 12:51:02 +01:00
(unwind-protect
(with-current-buffer responsebuf
(apply #'call-process "curl" nil t nil (list
url
"--silent"
"-X" type
"--data-binary"
(concat "@" tempfile)))
(goto-char (point-min))
2020-02-24 06:39:31 +01:00
(when success
(apply success (list :data (funcall parser)))))
2020-02-10 12:51:02 +01:00
(kill-buffer responsebuf)
(delete-file tempfile))))
2019-11-02 17:22:27 +01:00
(defun anki-editor-api-call (action &rest params)
2018-06-10 15:13:00 +02:00
"Invoke AnkiConnect with ACTION and PARAMS."
2019-11-02 17:22:27 +01:00
(let ((payload (list :action action :version anki-editor-api-version))
2020-02-23 08:37:40 +01:00
(_request-backend 'curl)
(json-array-type 'list)
reply err)
2019-11-02 17:22:27 +01:00
(when params
(plist-put payload :params params))
2020-02-10 12:51:02 +01:00
(anki-editor--fetch (format "http://%s:%s"
anki-editor-api-host
anki-editor-api-port)
2022-11-03 19:18:04 +01:00
:type "POST"
2020-02-10 12:51:02 +01:00
:parser 'json-read
:data (json-encode payload)
2022-11-03 19:18:04 +01:00
:success (cl-function
(lambda (&key data &allow-other-keys)
(setq reply data)))
:error (cl-function
(lambda (&key error-thrown &allow-other-keys)
(setq err (string-trim (cdr error-thrown)))))
2020-02-10 12:51:02 +01:00
:sync t)
2022-11-03 19:18:04 +01:00
(when err
(error "Error communicating with AnkiConnect using cURL: %s" err))
2018-05-13 09:52:37 +02:00
(or reply (error "Got empty reply from AnkiConnect"))))
2018-01-19 13:32:28 +01:00
2019-11-02 17:22:27 +01:00
(defun anki-editor-api-call-result (&rest args)
2022-11-03 19:18:04 +01:00
"Invoke AnkiConnect with ARGS and return the result from response.
Raise an error if applicable."
2019-11-02 17:22:27 +01:00
(let-alist (apply #'anki-editor-api-call args)
(when .error (error .error))
.result))
2018-01-19 13:32:28 +01:00
2019-11-01 15:59:15 +01:00
(defmacro anki-editor-api-with-multi (&rest body)
"Use in combination with `anki-editor-api-enqueue' to combine
multiple api calls into a single 'multi' call, return the results
of these calls in the same order."
`(let (--anki-editor-var-multi-actions--
--anki-editor-var-multi-results--)
,@body
(setq --anki-editor-var-multi-results--
(anki-editor-api-call-result
'multi
:actions (nreverse
;; Here we make a vector from the action list,
2022-11-03 19:18:04 +01:00
;; or `json-encode' will consider it as an alist.
2019-11-01 15:59:15 +01:00
(vconcat
--anki-editor-var-multi-actions--))))
(cl-loop for result in --anki-editor-var-multi-results--
2020-02-23 08:37:40 +01:00
do (when-let ((pred (listp result))
2019-11-01 15:59:15 +01:00
(err (alist-get 'error result)))
(error err))
collect result)))
(defmacro anki-editor-api-enqueue (action &rest params)
"Like `anki-editor-api-call', but is only used in combination
with `anki-editor-api-with-multi'. Instead of sending the
request directly, it simply queues the request."
`(let ((action (list :action ,action))
(params (list ,@params)))
(when params
(plist-put action :params params))
(push action --anki-editor-var-multi-actions--)))
(defun anki-editor-api--note (note)
2018-02-04 02:10:40 +01:00
"Convert NOTE to the form that AnkiConnect accepts."
2019-11-01 15:59:15 +01:00
(list
:id (string-to-number (or (anki-editor-note-id note) "0"))
:deckName (anki-editor-note-deck note)
:modelName (anki-editor-note-model note)
:fields (anki-editor-note-fields note)
;; Convert tags to a vector since empty list is identical to nil
;; which will become None in Python, but AnkiConnect requires it
;; to be type of list.
:tags (vconcat (anki-editor-note-tags note))))
2018-01-19 13:32:28 +01:00
2019-11-02 17:22:27 +01:00
(defun anki-editor-api--store-media-file (path)
2018-02-07 17:27:44 +01:00
"Store media file for PATH, which is an absolute file name.
The result is the path to the newly stored media file."
(let* ((bytes (with-temp-buffer
(insert-file-contents-literally path)
(buffer-string)))
(hash (secure-hash 'sha1 bytes))
2018-02-07 17:27:44 +01:00
(media-file-name (format "%s-%s%s"
(file-name-base path)
hash
2019-11-02 17:22:27 +01:00
(file-name-extension path t))))
(when (eq :json-false
(anki-editor-api-call-result 'retrieveMediaFile
:filename media-file-name))
2019-09-23 09:41:07 +02:00
(message "Storing media file %s to Anki, this might take a while" path)
2019-11-02 17:22:27 +01:00
(anki-editor-api-call-result 'storeMediaFile
:filename media-file-name
:data (base64-encode-string bytes)))
2018-02-07 17:27:44 +01:00
media-file-name))
2019-11-03 10:24:00 +01:00
;;; Org export backend
2018-05-07 15:09:52 +02:00
2018-06-11 12:00:19 +02:00
(defconst anki-editor--ox-anki-html-backend
2019-11-03 10:24:00 +01:00
(org-export-create-backend
:parent 'html
:transcoders '((latex-fragment . anki-editor--ox-latex)
(latex-environment . anki-editor--ox-latex))))
(defconst anki-editor--ox-export-ext-plist
2020-02-24 06:39:31 +01:00
'(:with-toc nil :with-properties nil :with-planning nil :anki-editor-mode t))
(defconst anki-editor--audio-extensions
'(".mp3" ".3gp" ".flac" ".m4a" ".oga" ".ogg" ".opus" ".spx" ".wav"))
2022-11-03 19:18:04 +01:00
(cl-macrolet
((with-table (table)
`(cl-loop for delims in ,table
collect
(list (concat "^" (regexp-quote (cl-first delims)))
(cl-second delims)
(concat (regexp-quote (cl-third delims)) "$")
(cl-fourth delims)))))
2019-11-03 10:24:00 +01:00
(defconst anki-editor--native-latex-delimiters
(with-table '(("$$" "[$$]"
"$$" "[/$$]")
("$" "[$]"
"$" "[/$]")
("\\(" "[$]"
"\\)" "[/$]")
("\\[" "[$$]"
"\\]" "[/$$]"))))
(defconst anki-editor--mathjax-delimiters
(with-table '(("$$" "\\["
"$$" "\\]")
("$" "\\("
"$" "\\)")))))
(defun anki-editor--translate-latex-fragment (latex-code)
2022-11-03 19:18:04 +01:00
"Translate LATEX-CODE fragment to html."
2019-11-11 16:03:55 +01:00
(cl-loop for delims in (cl-ecase anki-editor-latex-style
(builtin anki-editor--native-latex-delimiters)
(mathjax anki-editor--mathjax-delimiters))
for matches = (string-match (cl-first delims) latex-code)
when matches
do
(setq latex-code (replace-match (cl-second delims) t t latex-code))
(string-match (cl-third delims) latex-code)
(setq latex-code (replace-match (cl-fourth delims) t t latex-code))
until matches
finally return latex-code))
2019-11-03 10:24:00 +01:00
(defun anki-editor--translate-latex-env (latex-code)
2022-11-03 19:18:04 +01:00
"Translate LATEX-CODE environment to html."
(setq latex-code (replace-regexp-in-string
"\n" "<br>" (org-html-encode-plain-text latex-code)))
2019-11-11 16:03:55 +01:00
(cl-ecase anki-editor-latex-style
(builtin (concat "[latex]<br>" latex-code "[/latex]"))
(mathjax (concat "\\[<br>" latex-code "\\]"))))
2018-05-07 15:09:52 +02:00
2018-06-11 12:00:19 +02:00
(defun anki-editor--ox-latex (latex _contents _info)
"Transcode LATEX from Org to HTML.
CONTENTS is nil. INFO is a plist holding contextual information."
2018-06-16 10:59:32 +02:00
(let ((code (org-remove-indentation (org-element-property :value latex))))
2019-11-03 10:24:00 +01:00
(setq code (cl-ecase (org-element-type latex)
(latex-fragment (anki-editor--translate-latex-fragment code))
(latex-environment (anki-editor--translate-latex-env code))))
(if anki-editor-break-consecutive-braces-in-latex
(replace-regexp-in-string "}}" "} } " code)
code)))
2018-06-11 12:00:19 +02:00
(defun anki-editor--ox-html-link (oldfun link desc info)
2022-11-03 19:18:04 +01:00
"Export LINK and its target.
When LINK is a link to local file, transcode it to html
and store the target file to Anki, otherwise call OLDFUN for help.
2018-06-11 12:00:19 +02:00
The implementation is borrowed and simplified from ox-html."
2022-11-03 19:18:04 +01:00
(or
(catch 'giveup
(unless (plist-get info :anki-editor-mode)
(throw 'giveup nil))
(let* ((type (org-element-property :type link))
(raw-path (org-element-property :path link))
(desc (org-string-nw-p desc))
(path
(cond
((string= type "file")
;; Possibly append `:html-link-home' to relative file
;; name.
(let ((inhibit-message nil)
(home (and (plist-get info :html-link-home)
(org-trim (plist-get info :html-link-home)))))
(when (and home
(plist-get info :html-link-use-abs-url)
(file-name-absolute-p raw-path))
(setq raw-path
(concat (file-name-as-directory home) raw-path)))
;; storing file to Anki and return the modified path
(anki-editor-api--store-media-file
(expand-file-name (url-unhex-string raw-path)))))
(t (throw 'giveup nil))))
(attributes-plist
(let* ((parent (org-export-get-parent-element link))
(link (let ((container (org-export-get-parent link)))
(if (and (eq (org-element-type container) 'link)
(org-html-inline-image-p link info))
container
link))))
(and (eq (org-element-map parent 'link 'identity info t) link)
(org-export-read-attribute :attr_html parent))))
(attributes
(let ((attr (org-html--make-attribute-string attributes-plist)))
(if (org-string-nw-p attr) (concat " " attr) ""))))
(cond
;; Image file.
((and (plist-get info :html-inline-images)
(org-export-inline-image-p
link (plist-get info :html-inline-image-rules)))
(org-html--format-image path attributes-plist info))
;; Audio file.
((cl-some (lambda (string) (string-suffix-p string path t))
anki-editor--audio-extensions)
(format "[sound:%s]" path))
;; External link with a description part.
((and path desc) (format "<a href=\"%s\"%s>%s</a>"
(org-html-encode-plain-text path)
attributes
desc))
;; External link without a description part.
(path (let ((path (org-html-encode-plain-text path)))
(format "<a href=\"%s\"%s>%s</a>"
path
attributes
(org-link-unescape path))))
(t (throw 'giveup nil)))))
(funcall oldfun link desc info)))
2018-01-31 16:08:24 +01:00
2020-02-24 06:39:31 +01:00
(defun anki-editor--export-string (src fmt)
2022-11-03 19:18:04 +01:00
"Export string SRC and format it if FMT."
2020-02-24 06:39:31 +01:00
(cl-ecase fmt
('nil src)
('t (or (org-export-string-as src
anki-editor--ox-anki-html-backend
t
anki-editor--ox-export-ext-plist)
;; 8.2.10 version of
;; `org-export-filter-apply-functions'
;; returns nil for an input of empty string,
;; which will cause AnkiConnect to fail
""))))
2019-11-03 10:24:00 +01:00
;;; Core primitives
(defconst anki-editor-prop-note-type "ANKI_NOTE_TYPE")
(defconst anki-editor-prop-note-id "ANKI_NOTE_ID")
2020-02-24 06:39:31 +01:00
(defconst anki-editor-prop-format "ANKI_FORMAT")
2019-11-03 10:24:00 +01:00
(defconst anki-editor-prop-deck "ANKI_DECK")
(defconst anki-editor-prop-tags "ANKI_TAGS")
(defconst anki-editor-prop-tags-plus (concat anki-editor-prop-tags "+"))
(defconst anki-editor-prop-failure-reason "ANKI_FAILURE_REASON")
(defconst anki-editor-prop-default-note-type "ANKI_DEFAULT_NOTE_TYPE")
2019-11-03 10:24:00 +01:00
(defconst anki-editor-org-tag-regexp "^\\([[:alnum:]_@#%]+\\)+$")
2017-12-27 17:20:01 +01:00
2019-11-01 15:59:15 +01:00
(cl-defstruct anki-editor-note
id model deck fields tags)
2020-02-24 06:39:31 +01:00
(defvar anki-editor--collection-data-updated nil
2022-11-03 19:18:04 +01:00
"Whether or not collection data is updated from Anki.
Used by `anki-editor--with-collection-data-updated'
to avoid unnecessary updates.")
2020-02-24 06:39:31 +01:00
2022-11-03 19:18:04 +01:00
;; The following variables should only be used inside
;; `anki-editor--with-collection-data-updated'.
2020-02-24 06:39:31 +01:00
(defvar anki-editor--model-names nil
"Note types from Anki.")
(defvar anki-editor--model-fields nil
"Alist of (NOTE-TYPE . FIELDS).")
(defmacro anki-editor--with-collection-data-updated (&rest body)
"Execute BODY with collection data updated from Anki.
Note that since we have no idea of whether BODY will update collection
data, BODY might read out-dated data. This doesn't matter right now
as note types won't change in BODY."
(declare (indent defun) (debug t))
`(if anki-editor--collection-data-updated
(progn ,@body)
(cl-destructuring-bind (models)
(anki-editor-api-with-multi
(anki-editor-api-enqueue 'modelNames))
(unwind-protect
(progn
(setq anki-editor--collection-data-updated t
anki-editor--model-names models
anki-editor--model-fields
2022-11-03 19:18:04 +01:00
(cl-loop
for flds in (eval `(anki-editor-api-with-multi
,@(cl-loop
for mod in models
collect `(anki-editor-api-enqueue
'modelFieldNames
:modelName ,mod))))
for mod in models
collect (cons mod flds)))
2020-02-24 06:39:31 +01:00
,@body)
(setq anki-editor--collection-data-updated nil)))))
2018-07-06 15:10:03 +02:00
(defun anki-editor-map-note-entries (func &optional match scope &rest skip)
2022-11-03 19:18:04 +01:00
"Apply FUNC to each anki-editor note matching MATCH in SCOPE.
Simple wrapper that calls `org-map-entries' with entries that match
2020-03-09 14:36:27 +01:00
`ANKI_NOTE_TYPE<>\"\"', `anki-editor-note-match' and MATCH.
A leading logical operator like `+' or `&' is required in MATCH."
2018-07-06 15:10:03 +02:00
;; disable property inheritance temporarily, or all subheadings of a
;; note heading will be counted as note headings as well
(let ((org-use-property-inheritance nil))
(apply #'org-map-entries
func
(concat "+" anki-editor-prop-note-type "<>\"\""
match
anki-editor-note-match)
scope
skip)))
2020-02-24 06:39:31 +01:00
(defun anki-editor--insert-note-skeleton (prefix deck heading type fields)
"Insert a note subtree (skeleton) with HEADING, TYPE and FIELDS.
DECK is only inserted if not already inherited. For PREFIX and more
see `anki-editor-insert-note' which wraps this function."
(org-insert-heading-respect-content)
2018-01-22 14:57:35 +01:00
(insert heading)
(org-set-property anki-editor-prop-note-type type)
(unless (save-excursion
(org-up-heading-safe)
(and (not (string-blank-p deck))
(string= deck (org-entry-get-with-inheritance
anki-editor-prop-deck))))
(org-set-property anki-editor-prop-deck deck))
(when (string-blank-p heading)
(setq fields (cdr fields)))
(when (equal 1 (length fields))
(setq fields nil))
(let ((use-content (if prefix
(not anki-editor-insert-note-always-use-content)
anki-editor-insert-note-always-use-content)))
(when use-content
(setq fields (cdr fields)))
(dolist (field fields)
(save-excursion
(org-insert-heading-respect-content)
(org-do-demote)
(insert field)))
(when (and (not (string-blank-p heading))
(not use-content))
(org-goto-first-child)
(end-of-line))))
2018-01-22 14:57:35 +01:00
(defun anki-editor--push-note (note)
2018-02-04 02:10:40 +01:00
"Request AnkiConnect for updating or creating NOTE."
2019-11-01 15:59:15 +01:00
(cond
((null (anki-editor-note-id note))
(anki-editor--create-note note))
(t
(anki-editor--update-note note))))
(defun anki-editor--set-note-id (id)
2022-11-03 19:18:04 +01:00
"Set note-id of anki-editor note at point to ID."
(unless id
(error "Note creation failed for unknown reason"))
(org-set-property anki-editor-prop-note-id (number-to-string id)))
(defun anki-editor--create-note (note)
2018-02-04 02:10:40 +01:00
"Request AnkiConnect for creating NOTE."
2019-11-01 15:59:15 +01:00
(thread-last
2022-11-03 19:18:04 +01:00
(anki-editor-api-with-multi
(anki-editor-api-enqueue 'createDeck
:deck (anki-editor-note-deck note))
(anki-editor-api-enqueue 'addNote
:note (anki-editor-api--note note)))
2019-11-01 15:59:15 +01:00
(nth 1)
(anki-editor--set-note-id)))
(defun anki-editor--update-note (note)
"Request AnkiConnect for updating fields, deck, and tags of NOTE."
2019-11-01 15:59:15 +01:00
(let* ((oldnote (caar (anki-editor-api-with-multi
2022-11-03 19:18:04 +01:00
(anki-editor-api-enqueue
'notesInfo
:notes (list (string-to-number
(anki-editor-note-id note))))
(anki-editor-api-enqueue
'updateNoteFields
:note (anki-editor-api--note note)))))
2019-11-01 15:59:15 +01:00
(tagsadd (cl-set-difference (anki-editor-note-tags note)
(alist-get 'tags oldnote)
:test 'string=))
(tagsdel (thread-first (alist-get 'tags oldnote)
2022-11-03 19:18:04 +01:00
(cl-set-difference (anki-editor-note-tags note)
:test 'string=)
(cl-set-difference anki-editor-protected-tags
:test 'string=))))
2019-11-01 15:59:15 +01:00
(anki-editor-api-with-multi
(anki-editor-api-enqueue 'changeDeck
:cards (alist-get 'cards oldnote)
:deck (anki-editor-note-deck note))
2019-11-01 15:59:15 +01:00
(when tagsadd
(anki-editor-api-enqueue 'addTags
:notes (list (string-to-number
(anki-editor-note-id note)))
2019-11-01 15:59:15 +01:00
:tags (mapconcat #'identity tagsadd " ")))
(when tagsdel
(anki-editor-api-enqueue 'removeTags
:notes (list (string-to-number
(anki-editor-note-id note)))
2019-11-01 15:59:15 +01:00
:tags (mapconcat #'identity tagsdel " "))))))
(defun anki-editor--set-failure-reason (reason)
2018-01-19 13:21:06 +01:00
"Set failure reason to REASON in property drawer at point."
(org-entry-put nil anki-editor-prop-failure-reason reason))
(defun anki-editor--clear-failure-reason ()
2018-01-19 13:21:06 +01:00
"Clear failure reason in property drawer at point."
(org-entry-delete nil anki-editor-prop-failure-reason))
2018-04-15 09:31:45 +02:00
(defun anki-editor--get-allowed-values-for-property (property)
"Get allowed values for PROPERTY."
(pcase property
((pred (string= anki-editor-prop-deck)) (anki-editor-deck-names))
((pred (string= anki-editor-prop-note-type)) (anki-editor-note-types))
2020-02-24 06:39:31 +01:00
((pred (string= anki-editor-prop-format)) (list "t" "nil"))
2022-11-03 19:18:04 +01:00
((pred (string-match-p (format "%s\\+?" anki-editor-prop-tags)))
(anki-editor-all-tags))
(_ nil)))
(defun anki-editor-is-valid-org-tag (tag)
"Check if string TAG can be used as an Org tag."
(string-match-p anki-editor-org-tag-regexp tag))
2018-05-06 08:01:06 +02:00
(defun anki-editor-all-tags ()
"Get all tags from Anki."
2019-11-02 17:22:27 +01:00
(anki-editor-api-call-result 'getTags))
2018-05-06 08:01:06 +02:00
(defun anki-editor-deck-names ()
"Get all decks names from Anki."
2019-11-02 17:22:27 +01:00
(anki-editor-api-call-result 'deckNames))
(defun anki-editor--enable-tag-completion ()
(and anki-editor-mode anki-editor-org-tags-as-anki-tags))
2018-05-06 08:01:06 +02:00
(defun anki-editor--before-set-tags (&optional _ just-align)
"Fetch and cache tags from Anki."
(when (and (anki-editor--enable-tag-completion)
(not just-align))
2018-07-29 11:18:59 +02:00
(setq anki-editor--anki-tags-cache (anki-editor-all-tags))
2022-11-03 19:18:04 +01:00
(when (cl-notevery #'anki-editor-is-valid-org-tag
anki-editor--anki-tags-cache)
(warn (concat "Some tags from Anki contain characters that are not"
"valid in Org tags.")))))
2018-05-06 08:01:06 +02:00
(defun anki-editor--get-buffer-tags (oldfun)
"Append tags from Anki to the result of applying OLDFUN."
(append (funcall oldfun)
(when (anki-editor--enable-tag-completion)
(mapcar #'list anki-editor--anki-tags-cache))))
2018-05-06 08:01:06 +02:00
(defun anki-editor-note-types ()
"Get note types from Anki."
2019-11-02 17:22:27 +01:00
(anki-editor-api-call-result 'modelNames))
2020-02-24 06:39:31 +01:00
(defun anki-editor-entry-format ()
(read (or (org-entry-get-with-inheritance anki-editor-prop-format t) "t")))
(defun anki-editor-toggle-format ()
"Cycle ANKI_FORMAT through \"nil\" and \"t\"."
2020-02-24 06:39:31 +01:00
(interactive)
(let ((val (pcase (org-entry-get nil anki-editor-prop-format nil t)
('nil "nil")
("nil" "t")
("t" nil)
(_ "nil"))))
(if val
(org-entry-put nil anki-editor-prop-format val)
(org-entry-delete nil anki-editor-prop-format))))
(defun anki-editor-note-at-point ()
2019-11-01 15:59:15 +01:00
"Make a note struct from current entry."
2022-11-01 12:47:59 +01:00
(let* ((org-trust-scanner-tags t)
(deck (org-entry-get-with-inheritance anki-editor-prop-deck))
(format (anki-editor-entry-format))
(note-id (org-entry-get nil anki-editor-prop-note-id))
(note-type (org-entry-get nil anki-editor-prop-note-type))
(tags (cl-set-difference (anki-editor--get-tags)
anki-editor-ignored-org-tags
:test #'string=))
(heading (substring-no-properties (org-get-heading t t t)))
(level (org-current-level))
2022-11-01 12:47:59 +01:00
(content-before-subheading
(anki-editor--note-contents-before-subheading))
(subheading-fields (anki-editor--build-fields))
(fields (anki-editor--map-fields heading
2022-11-01 12:47:59 +01:00
content-before-subheading
subheading-fields
note-type
level))
(exported-fields (mapcar (lambda (x)
(cons
(car x)
2022-11-03 19:18:04 +01:00
(anki-editor--export-string (cdr x)
format)))
fields)))
(unless deck (user-error "Missing deck"))
(unless note-type (user-error "Missing note type"))
2019-11-01 15:59:15 +01:00
(make-anki-editor-note :id note-id
:model note-type
:deck deck
:tags tags
:fields exported-fields)))
2017-12-27 17:20:01 +01:00
(defun anki-editor--get-tags ()
2018-09-23 20:41:15 +02:00
(let ((tags (anki-editor--entry-get-multivalued-property-with-inheritance
nil
anki-editor-prop-tags)))
(if anki-editor-org-tags-as-anki-tags
2019-11-01 15:59:15 +01:00
(append tags (org-get-tags))
tags)))
2022-11-03 19:18:04 +01:00
(defun anki-editor--entry-get-multivalued-property-with-inheritance (pom
property)
2018-09-23 20:41:15 +02:00
"Return a list of values in a multivalued property with inheritance."
(let* ((value (org-entry-get pom property t))
2022-11-03 19:18:04 +01:00
(values (and value (split-string value))))
2018-09-23 20:41:15 +02:00
(mapcar #'org-entry-restore-space values)))
(defun anki-editor--build-fields ()
2020-02-10 16:23:44 +01:00
"Build a list of fields from subheadings of current heading.
2022-11-01 12:47:59 +01:00
Return a list of cons of (FIELD-NAME . FIELD-CONTENT)."
(save-excursion
2022-11-03 19:18:04 +01:00
(cl-loop with inhibit-message = t
;; suppress echo message from `org-babel-exp-src-block'
2020-02-10 16:23:44 +01:00
initially (unless (org-goto-first-child)
(cl-return))
for last-pt = (point)
for element = (org-element-at-point)
for heading = (substring-no-properties
(org-element-property :raw-value element))
2020-02-24 06:39:31 +01:00
;; contents-begin includes drawers and scheduling data,
;; which we'd like to ignore, here we skip these
;; elements and reset contents-begin.
for begin = (save-excursion
2022-11-03 19:18:04 +01:00
(cl-loop for eoh = (org-element-property
:contents-begin element)
then (org-element-property :end subelem)
while eoh
for subelem = (progn
(goto-char eoh)
(org-element-context))
2022-11-03 19:18:04 +01:00
while (memq
(org-element-type subelem)
'(drawer planning property-drawer))
finally return (and eoh
(org-element-property
:begin subelem))))
2020-02-10 16:23:44 +01:00
for end = (org-element-property :contents-end element)
for raw = (or (and begin
end
(buffer-substring-no-properties
begin
;; in case the buffer is narrowed,
;; e.g. by `org-map-entries' when
;; scope is `tree'
(min (point-max) end)))
"")
;; for content = (anki-editor--export-string raw format)
;; collect (cons heading content)
collect (cons heading raw)
2020-02-10 16:23:44 +01:00
;; proceed to next field entry and check last-pt to
;; see if it's already the last entry
do (org-forward-heading-same-level nil t)
until (= last-pt (point)))))
2018-02-04 02:10:40 +01:00
2022-11-01 12:47:59 +01:00
(defun anki-editor--note-contents-before-subheading ()
"Get content between heading at point and next sub/heading.
Leading whitespace, drawers, and planning content is skipped."
(save-excursion
(let* ((element (org-element-at-point))
2022-11-03 19:18:04 +01:00
(begin (cl-loop for eoh = (org-element-property
:contents-begin element)
2022-11-01 12:47:59 +01:00
then (org-element-property :end subelem)
while eoh
for subelem = (progn
(goto-char eoh)
(org-element-context))
while (memq (org-element-type subelem)
'(drawer planning property-drawer))
2022-11-03 19:18:04 +01:00
finally return (and eoh (org-element-property
:begin subelem))))
(end (cl-loop for eoh = (org-element-property
:contents-begin element)
2022-11-01 12:47:59 +01:00
then (org-element-property :end nextelem)
while eoh
for nextelem = (progn
(goto-char eoh)
(org-element-at-point))
2022-11-03 19:18:04 +01:00
while (not (memq (org-element-type nextelem)
'(headline)))
finally return (and eoh (org-element-property
:begin nextelem))))
2022-11-01 12:47:59 +01:00
(contents-raw (or (and begin
end
(buffer-substring-no-properties
begin
;; in case the buffer is narrowed,
;; e.g. by `org-map-entries' when
;; scope is `tree'
(min (point-max) end)))
"")))
contents-raw)))
(defun anki-editor--map-fields (heading
content-before-subheading
subheading-fields
note-type
level)
2022-11-01 12:47:59 +01:00
"Map `heading', pre-subheading content, and subheadings to fields.
When the `subheading-fields' don't match the `note-type's fields,
map missing fields to the `heading' and/or `content-before-subheading'.
Return a list of cons of (FIELD-NAME . FIELD-CONTENT)."
(anki-editor--with-collection-data-updated
(let* ((fields-matching (cl-intersection
(alist-get note-type
anki-editor--model-fields
nil nil #'string=)
(mapcar #'car subheading-fields)
:test #'string=))
(fields-missing (cl-set-difference
(alist-get note-type
anki-editor--model-fields
nil nil #'string=)
(mapcar #'car subheading-fields)
:test #'string=))
(fields-extra (cl-set-difference
(mapcar #'car subheading-fields)
(alist-get note-type
anki-editor--model-fields
nil nil #'string=)
:test #'string=))
(fields (cl-loop for f in fields-matching
collect (cons f (alist-get
f subheading-fields
nil nil #'string=)))))
(cond ((equal 0 (length fields-missing))
(when (< 0 (length fields-extra))
2022-11-03 19:18:04 +01:00
(user-error "Failed to map all subheadings to a field")))
((equal 1 (length fields-missing))
(if (equal 0 (length fields-extra))
(if (equal "" (string-trim content-before-subheading))
(push (cons (car fields-missing) heading)
fields)
(if anki-editor-prepend-note-heading
(push (cons (car fields-missing)
(concat heading "\n\n"
content-before-subheading))
fields)
(push (cons (car fields-missing)
content-before-subheading)
fields)))
(if (equal "" (string-trim content-before-subheading))
(push (cons (car fields-missing)
(anki-editor--concat-fields
fields-extra subheading-fields level))
fields)
(if anki-editor-prepend-note-heading
(push (cons (car fields-missing)
(concat heading "\n\n"
content-before-subheading
(anki-editor--concat-fields
fields-extra subheading-fields
level)))
fields)
(push (cons (car fields-missing)
(concat content-before-subheading
(anki-editor--concat-fields
fields-extra subheading-fields
level)))
fields)))))
((equal 2 (length fields-missing))
(if (equal 0 (length fields-extra))
(progn
(push (cons (nth 1 fields-missing)
content-before-subheading)
fields)
(push (cons (car fields-missing)
heading)
fields))
(if (equal "" (string-trim content-before-subheading))
(progn
(push (cons (nth 1 fields-missing)
(anki-editor--concat-fields
fields-extra subheading-fields level))
fields)
(push (cons (car fields-missing)
heading)
fields))
(progn
(push (cons (nth 1 fields-missing)
(concat content-before-subheading
(anki-editor--concat-fields
fields-extra subheading-fields level)))
fields)
(push (cons (car fields-missing)
heading)
fields)))))
((< 2 (length fields-missing))
2022-11-03 19:18:04 +01:00
(user-error (concaat "Cannot map note fields: "
"more than two fields missing"))))
2022-11-01 12:47:59 +01:00
fields)))
(defun anki-editor--concat-fields (field-names field-alist level)
"Concat field names and content of fields in list `field-names'."
(let ((format (anki-editor-entry-format)))
(cl-loop for f in field-names
concat (concat (make-string (+ 1 level) ?*) " " f "\n\n"
2022-11-03 19:18:04 +01:00
(string-trim (alist-get f field-alist nil nil
#'string=))
"\n\n"))))
2018-06-11 12:00:19 +02:00
;;; Minor mode
2018-02-04 02:10:40 +01:00
(defvar-local anki-editor--anki-tags-cache nil)
2018-09-23 20:41:15 +02:00
(defun anki-editor--concat-multivalued-property-value (prop value)
(let ((old-values (org-entry-get-multivalued-property nil prop)))
(unless (string-suffix-p prop "+")
2019-11-01 15:59:15 +01:00
(setq old-values (cl-set-difference old-values
(org-entry-get-multivalued-property
nil (concat prop "+"))
:test 'string=)))
2018-09-23 20:41:15 +02:00
(mapconcat #'org-entry-protect-space
(append old-values (list value))
" ")))
(setq org-properties-postprocess-alist
(append org-properties-postprocess-alist
(list (cons anki-editor-prop-tags
(lambda (value)
2022-11-03 19:18:04 +01:00
(anki-editor--concat-multivalued-property-value
anki-editor-prop-tags value)))
2018-09-23 20:41:15 +02:00
(cons anki-editor-prop-tags-plus
(lambda (value)
2022-11-03 19:18:04 +01:00
(anki-editor--concat-multivalued-property-value
anki-editor-prop-tags-plus value))))))
2018-09-23 20:41:15 +02:00
2018-06-11 12:00:19 +02:00
;;;###autoload
(define-minor-mode anki-editor-mode
2022-11-03 19:18:04 +01:00
"A minor mode for making Anki cards with Org."
2018-06-11 12:00:19 +02:00
:lighter " anki-editor"
:keymap (make-sparse-keymap)
2018-06-11 12:00:19 +02:00
(if anki-editor-mode (anki-editor-setup-minor-mode)
(anki-editor-teardown-minor-mode)))
2018-02-04 02:10:40 +01:00
2018-06-11 12:00:19 +02:00
(defun anki-editor-setup-minor-mode ()
"Set up this minor mode."
2019-11-02 17:22:27 +01:00
(anki-editor-api-check)
2022-11-03 19:18:04 +01:00
(add-hook 'org-property-allowed-value-functions
#'anki-editor--get-allowed-values-for-property nil t)
2018-06-11 12:00:19 +02:00
(advice-add 'org-set-tags :before #'anki-editor--before-set-tags)
(advice-add 'org-get-buffer-tags :around #'anki-editor--get-buffer-tags)
2018-06-11 12:00:19 +02:00
(advice-add 'org-html-link :around #'anki-editor--ox-html-link))
2017-12-27 17:20:01 +01:00
2018-06-11 12:00:19 +02:00
(defun anki-editor-teardown-minor-mode ()
"Tear down this minor mode."
2022-11-03 19:18:04 +01:00
(remove-hook 'org-property-allowed-value-functions
#'anki-editor--get-allowed-values-for-property t))
2018-01-08 11:20:11 +01:00
2017-12-27 17:20:01 +01:00
2018-06-11 12:00:19 +02:00
;;; Commands
2020-02-24 10:16:20 +01:00
(defvar anki-editor--note-markers nil)
(defun anki-editor--collect-note-marker ()
(message "Scanning notes %d (%s@%d), wait a moment..."
(length anki-editor--note-markers) (buffer-name) (point))
(push (point-marker) anki-editor--note-markers))
(defun anki-editor-push-notes (&optional scope match &rest skip)
2022-11-03 19:18:04 +01:00
"Build notes from headings that MATCH within SCOPE and push them to Anki.
2017-12-27 17:20:01 +01:00
2018-06-11 12:00:19 +02:00
The default search condition `&ANKI_NOTE_TYPE<>\"\"' will always
be appended to MATCH.
2018-05-22 15:09:36 +02:00
2018-06-11 12:00:19 +02:00
For notes that already exist in Anki (i.e. has `ANKI_NOTE_ID'
property), only their fields, tags and deck will be updated,
change of note type is currently not supported.
2018-05-22 15:09:36 +02:00
2018-06-11 12:00:19 +02:00
If SCOPE is not specified, the following rules are applied to
determine the scope:
2018-05-22 15:09:36 +02:00
2018-06-11 12:00:19 +02:00
- If there's an active region, it will be set to `region'
- If called with prefix `C-u', it will be set to `tree'
- If called with prefix double `C-u', it will be set to `file'
- If called with prefix triple `C-u', will be set to `agenda'
2018-02-07 17:27:44 +01:00
2018-06-11 12:00:19 +02:00
See doc string of `org-map-entries' for what these different options mean.
2018-06-11 12:00:19 +02:00
If one fails, the failure reason will be set in property drawer
of that heading."
2019-09-23 09:41:07 +02:00
(interactive (list (cond
((region-active-p) 'region)
((equal current-prefix-arg '(4)) 'tree)
((equal current-prefix-arg '(16)) 'file)
((equal current-prefix-arg '(64)) 'agenda)
(t nil))))
2020-02-24 10:16:20 +01:00
(unwind-protect
(progn
(apply #'anki-editor-map-note-entries
#'anki-editor--collect-note-marker match scope skip)
2020-02-24 10:16:20 +01:00
(setq anki-editor--note-markers (reverse anki-editor--note-markers))
(let ((count 0)
(failed 0))
(save-excursion
(anki-editor--with-collection-data-updated
(cl-loop with bar-width = 30
for marker in anki-editor--note-markers
2022-11-03 19:18:04 +01:00
for progress = (/ (float (cl-incf count))
(length anki-editor--note-markers))
2020-02-24 10:16:20 +01:00
do
(goto-char marker)
2022-11-03 19:18:04 +01:00
(message
"Uploading notes in buffer %s%s [%s%s] %d/%d (%.2f%%)"
(marker-buffer marker)
(if (zerop failed)
""
(propertize (format " %d failed" failed)
'face `(:foreground "red")))
(make-string (truncate (* bar-width progress))
?#)
(make-string (- bar-width
(truncate (* bar-width
progress)))
?.)
count
(length anki-editor--note-markers)
(* 100 progress))
2020-02-24 10:16:20 +01:00
(anki-editor--clear-failure-reason)
(condition-case-unless-debug err
(anki-editor--push-note (anki-editor-note-at-point))
(error (cl-incf failed)
2022-11-03 19:18:04 +01:00
(anki-editor--set-failure-reason
(error-message-string err))))
2020-02-24 10:16:20 +01:00
;; free marker
(set-marker marker nil))))
(message
(cond
2022-11-03 19:18:04 +01:00
((zerop (length anki-editor--note-markers))
"Nothing to push")
((zerop failed)
(format "Successfully pushed %d notes to Anki" count))
(t
(format (concat "Pushed %d notes to Anki, with %d failed. "
"Check property drawers for details. "
"\nWhen you have fixed those issues, "
"try re-push the failed ones with "
"\n`anki-editor-retry-failed-notes'.")
count failed))))))
2020-02-24 10:16:20 +01:00
;; clean up markers
(cl-loop for m in anki-editor--note-markers
2022-11-03 19:18:04 +01:00
do (set-marker m nil)
finally do (setq anki-editor--note-markers nil))))
2019-09-23 09:41:07 +02:00
(defun anki-editor-push-note-at-point ()
"Push note at point to Anki.
If point is not at a heading with an `ANKI_NOTE_TYPE' property,
go up one heading at a time, until heading level 1, and push the
subtree associated with the first heading that has one."
(interactive)
(save-excursion
(let ((note-type))
(while
(and (org-back-to-heading)
(not (setq note-type
(org-entry-get nil anki-editor-prop-note-type)))
(org-up-heading-safe)))
(if (not note-type)
2022-11-03 19:18:04 +01:00
(user-error "No note to push found")
(anki-editor--push-note (anki-editor-note-at-point))
(message "Successfully pushed note at point to Anki.")))))
2019-09-23 09:41:07 +02:00
(defun anki-editor-push-new-notes (&optional scope)
"Push note entries without ANKI_NOTE_ID in SCOPE to Anki."
(interactive)
(anki-editor-push-notes scope (concat anki-editor-prop-note-id "=\"\"")))
2018-06-11 12:00:19 +02:00
2019-09-23 09:41:07 +02:00
(defun anki-editor-retry-failed-notes (&optional scope)
"Retry pushing notes marked as failed.
2022-11-03 19:18:04 +01:00
This command just calls `anki-editor-push-notes' with match string
2018-06-11 12:00:19 +02:00
matching non-empty `ANKI_FAILURE_REASON' properties."
2019-09-23 09:41:07 +02:00
(interactive)
2022-11-03 19:18:04 +01:00
(anki-editor-push-notes scope
(concat anki-editor-prop-failure-reason "<>\"\"")))
2019-09-23 09:41:07 +02:00
(defun anki-editor-delete-note-at-point (&optional prefix)
"Delete the note at point from Anki.
With PREFIX also delete it from Org."
(interactive "P")
(save-excursion
(let (note-type note-id)
(while
(and (org-back-to-heading)
(not (setq note-type
(org-entry-get nil anki-editor-prop-note-type)))
(org-up-heading-safe)))
(when (not note-type)
(user-error "No note to delete found"))
(setq note-id (condition-case nil
(string-to-number
(org-entry-get nil anki-editor-prop-note-id))
(error nil)))
(if (not note-id)
(if prefix
(message "Note at point is not in Anki (no note-id)")
(user-error "Note at point is not in Anki (no note-id)"))
(when (yes-or-no-p
(format (concat "Do you really want to delete note %s "
"from Anki?")
note-id))
(anki-editor-api-call-result 'deleteNotes
:notes (list note-id))
(org-entry-delete nil anki-editor-prop-note-id)
(message "Deleted note %s from Anki" note-id)))
(when prefix
(org-mark-subtree)
(kill-region nil nil t)
(message "Deleted note at point from Org")))))
2018-06-11 12:00:19 +02:00
(defun anki-editor-insert-note (&optional prefix note-type)
2018-06-11 12:00:19 +02:00
"Insert a note interactively.
The note is placed after the current subtree, at the same level
as the heading closest before point.
When note heading is not provided, it is used as the first field;
when additionally the note-type only has two fields, the content
after the heading is used for the second field and no subheading
is created.
2020-02-24 06:39:31 +01:00
With `anki-editor-insert-note-always-use-content' the content
after the note heading and before the first subheading is always
used for a field (the second or first field, depending on whether
the heading is used for the first field or not). PREFIX temporarily
inverts the value of `anki-editor-insert-note-always-use-content'.
When NOTE-TYPE is nil, prompt for one."
2018-06-11 12:00:19 +02:00
(interactive "P")
(let* ((deck (or (org-entry-get-with-inheritance anki-editor-prop-deck)
(completing-read "Deck: " (sort (anki-editor-deck-names)
#'string-lessp))))
(type (or note-type
(completing-read "Note type: " (sort
(anki-editor-note-types)
#'string-lessp))))
(fields (anki-editor-api-call-result 'modelFieldNames
:modelName type))
2020-02-24 06:39:31 +01:00
(heading (read-from-minibuffer "Note heading (optional): ")))
(anki-editor--insert-note-skeleton prefix deck heading type fields)))
2018-06-11 12:00:19 +02:00
(defun anki-editor-insert-default-note (&optional prefix)
"Insert a note with default note type interactively.
The note type is taken from the ANKI_DEFAULT_NOTE_TYPE property,
with inheritance, or from `anki-editor-default-note-type'.
Otherwise this command is like `anki-editor-insert-note'."
(interactive "P")
(let ((note-type
(or (org-entry-get-with-inheritance
anki-editor-prop-default-note-type)
anki-editor-default-note-type
(user-error "No default note type set"))))
(anki-editor-insert-note prefix note-type)))
(defun anki-editor-set-note-type (&optional prefix note-type)
"Set note type for current or closest previous heading.
With PREFIX set note type for all top-level headings in subtree.
When NOTE-TYPE is nil, prompt for one."
(interactive "P")
(let ((note-type
(or note-type
(completing-read "Note type: " (sort
(anki-editor-note-types)
#'string-lessp))))
(level
(if prefix
(+ 1 (or (org-current-level) 0))
(or (org-current-level) 0))))
(org-map-entries
(lambda () (org-set-property anki-editor-prop-note-type note-type))
(concat "LEVEL=" (number-to-string level))
(if (and prefix
(equal 1 level))
nil
'tree))))
(defun anki-editor-set-default-note-type (&optional prefix)
"Set default note type for current or closest previous heading.
The note type is taken from the ANKI_DEFAULT_NOTE_TYPE property,
with inheritance, or from `anki-editor-default-note-type'.
Otherwise this command is like `anki-editor-set-note-type'."
(interactive "P")
(let ((note-type
(or (org-entry-get-with-inheritance
anki-editor-prop-default-note-type)
anki-editor-default-note-type
(user-error "No default note type set"))))
2022-11-03 19:18:04 +01:00
(anki-editor-set-note-type prefix note-type)))
(defun anki-editor-cloze-region (&optional arg hint)
2018-06-11 12:00:19 +02:00
"Cloze region with number ARG."
(interactive "p\nsHint (optional): ")
(unless (region-active-p) (user-error "No active region"))
(anki-editor-cloze (region-beginning) (region-end) arg hint))
(defun anki-editor-cloze-dwim (&optional arg hint)
2022-11-03 19:18:04 +01:00
"Cloze current active region or a word the under the cursor."
(interactive "p\nsHint (optional): ")
(cond
2022-11-03 19:18:04 +01:00
((region-active-p)
(anki-editor-cloze (region-beginning) (region-end) arg hint))
((thing-at-point 'word)
(let ((bounds (bounds-of-thing-at-point 'word)))
(anki-editor-cloze (car bounds) (cdr bounds) arg hint)))
(t (user-error "Nothing to create cloze from"))))
(defun anki-editor-cloze (begin end arg hint)
"Cloze region from BEGIN to END with number ARG."
(let ((region (buffer-substring begin end)))
2018-06-11 12:00:19 +02:00
(save-excursion
(delete-region begin end)
2018-06-11 12:00:19 +02:00
(insert (with-output-to-string
(princ (format "{{c%d::%s" (or arg 1) region))
(unless (string-blank-p hint) (princ (format "::%s" hint)))
(princ "}}"))))))
(defun anki-editor-export-subtree-to-html ()
"Export subtree of the element at point to HTML."
(interactive)
(org-export-to-buffer
anki-editor--ox-anki-html-backend
2022-11-03 19:18:04 +01:00
"*AnkiEditor HTML Output*" nil t nil t
anki-editor--ox-export-ext-plist #'html-mode))
2018-06-11 12:00:19 +02:00
(defun anki-editor-convert-region-to-html ()
"Convert and replace region to HTML."
(interactive)
(org-export-replace-region-by anki-editor--ox-anki-html-backend))
2019-09-23 09:41:07 +02:00
;;; More utilities
2019-11-02 17:22:27 +01:00
(defun anki-editor-api-check ()
2019-09-23 09:41:07 +02:00
"Check if correct version of AnkiConnect is serving."
(interactive)
2020-02-24 10:40:41 +01:00
(let ((ver (condition-case err
(anki-editor-api-call-result 'version)
(error (error (concat "Failed to connect to Anki: %s"
"\nIs Anki running with the "
"AnkiConnect add-on enabled?")
(error-message-string err))))))
2019-11-02 17:22:27 +01:00
(if (<= anki-editor-api-version ver)
2019-09-23 09:41:07 +02:00
(when (called-interactively-p 'interactive)
(message "AnkiConnect v.%d is running" ver))
2022-11-03 19:18:04 +01:00
(user-error "anki-editor requires at least version %d of AnkiConnect"
anki-editor-api-version))))
2019-09-23 09:41:07 +02:00
2022-11-13 19:46:22 +01:00
(defun anki-editor-sync-collection ()
"Synchronize the local Anki collection with AnkiWeb."
2019-09-23 09:41:07 +02:00
(interactive)
2022-11-13 19:46:22 +01:00
(anki-editor-api-call-result 'sync)
(message "Synced local Anki collection with AnkiWeb."))
2019-09-23 09:41:07 +02:00
(defun anki-editor-gui-browse (&optional query)
"Open Anki Browser with QUERY.
When called interactively, it will try to set QUERY to current
note or deck."
2022-11-03 19:18:04 +01:00
(interactive
(list
(pcase (org-entry-get-with-inheritance anki-editor-prop-note-id)
((and (pred stringp) nid) (format "nid:%s" nid))
(_ (format "deck:%s"
(or (org-entry-get-with-inheritance anki-editor-prop-deck)
"current"))))))
(anki-editor-api-call 'guiBrowse :query (or query ""))
(when anki-editor-gui-browse-ensure-foreground
(anki-editor-api-call 'guiBrowse :query (or query ""))))
2019-09-23 09:41:07 +02:00
(defun anki-editor-gui-add-cards ()
2022-11-03 19:18:04 +01:00
"Open Anki Add Cards dialog with presets from current note entry."
2019-09-23 09:41:07 +02:00
(interactive)
2019-11-02 17:22:27 +01:00
(anki-editor-api-call-result 'guiAddCards
2019-11-01 15:59:15 +01:00
:note (append
(anki-editor-api--note
(anki-editor-note-at-point))
(list :options '(:closeAfterAdding t)))))
2019-09-23 09:41:07 +02:00
(defun anki-editor-find-notes (&optional query)
"Find notes with QUERY."
(interactive "sQuery: ")
2019-11-02 17:22:27 +01:00
(let ((nids (anki-editor-api-call-result 'findNotes
2019-11-01 15:59:15 +01:00
:query (or query ""))))
2019-09-23 09:41:07 +02:00
(if (called-interactively-p 'interactive)
(message "%S" nids)
nids)))
2018-01-22 14:57:35 +01:00
2020-03-01 17:10:53 +01:00
(defvar anki-editor--style-start "</style>\n<!-- {{ Emacs Org-mode -->")
(defvar anki-editor--style-end "<!-- Emacs Org-mode }} -->\n<style>")
(defun anki-editor-copy-styles ()
2022-11-03 19:18:04 +01:00
"Copy `org-html-style-default' and `anki-editor-html-head' to Anki."
2020-03-01 17:10:53 +01:00
(interactive)
(let ((head (concat (org-element-normalize-string anki-editor--style-start)
2022-11-03 19:18:04 +01:00
(org-element-normalize-string
(format "<!-- Updated: %s -->" (current-time-string)))
2020-03-01 17:10:53 +01:00
(when anki-editor-include-default-style
(org-element-normalize-string org-html-style-default))
(org-element-normalize-string anki-editor-html-head)
anki-editor--style-end)))
(cl-loop for model in (anki-editor-note-types)
2022-11-03 19:18:04 +01:00
for style = (let* ((css (alist-get
'css
(anki-editor-api-call-result
'modelStyling :modelName model)))
2020-03-01 17:10:53 +01:00
(start (string-match
(regexp-quote anki-editor--style-start)
css))
(end (string-match
(regexp-quote anki-editor--style-end)
css)))
(if (and start end)
(progn
(cl-incf end (length anki-editor--style-end))
;; skip whitespaces
2022-11-03 19:18:04 +01:00
(when-let ((newend (string-match
"[[:graph:]]" css end)))
2020-03-01 17:10:53 +01:00
(setq end newend))
(concat
(substring css 0 start)
(substring css end)))
css))
do
(message "Updating styles for \"%s\"..." model)
2022-11-03 19:18:04 +01:00
(anki-editor-api-call-result
'updateModelStyling
:model (list :name model
:css (concat (concat head "\n\n") style)))
2020-03-01 17:10:53 +01:00
finally do (message "Updating styles...Done"))))
(defun anki-editor-remove-styles ()
2022-11-03 19:18:04 +01:00
"Remove html tags generated by this mode from card styles."
2020-03-01 17:10:53 +01:00
(interactive)
(cl-loop for model in (anki-editor-note-types)
2022-11-03 19:18:04 +01:00
for css = (alist-get 'css (anki-editor-api-call-result
'modelStyling :modelName model))
2020-03-01 17:10:53 +01:00
for start = (string-match
(regexp-quote anki-editor--style-start)
css)
for end = (string-match
(regexp-quote anki-editor--style-end)
css)
if (and start end)
do
(cl-incf end (length anki-editor--style-end))
;; also remove whitespaces
(when-let ((newend (string-match "[[:graph:]]" css end)))
(setq end newend))
(message "Resetting styles for \"%s\"..." model)
(anki-editor-api-call-result
'updateModelStyling
:model (list :name model
:css (concat
(substring css 0 start)
(substring css end))))
finally do (message "Resetting styles...Done")))
2017-12-27 17:20:01 +01:00
(provide 'anki-editor)
2022-11-03 19:18:04 +01:00
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
2017-12-27 17:20:01 +01:00
;;; anki-editor.el ends here