123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432 |
- ;;; ob-lilypond.el --- Babel Functions for Lilypond -*- lexical-binding: t; -*-
- ;; Copyright (C) 2010-2022 Free Software Foundation, Inc.
- ;; Author: Martyn Jago
- ;; Keywords: babel language, literate programming
- ;; URL: https://orgmode.org/worg/org-contrib/babel/languages/ob-doc-lilypond.html
- ;; This file is part of GNU Emacs.
- ;; GNU Emacs 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.
- ;; GNU Emacs 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 <https://www.gnu.org/licenses/>.
- ;;; Commentary:
- ;; Installation, ob-lilypond documentation, and examples are available at
- ;; https://orgmode.org/worg/org-contrib/babel/languages/ob-doc-lilypond.html
- ;;
- ;; Lilypond documentation can be found at
- ;; https://lilypond.org/manuals.html
- ;;
- ;; This depends on epstopdf --- See https://www.ctan.org/pkg/epstopdf.
- ;;; Code:
- (require 'org-macs)
- (org-assert-version)
- (require 'ob)
- (declare-function org-fold-show-all "org-fold" (&optional types))
- (defalias 'lilypond-mode 'LilyPond-mode)
- (add-to-list 'org-babel-tangle-lang-exts '("LilyPond" . "ly"))
- (defvar org-babel-default-header-args:lilypond '()
- "Default header arguments for lilypond code blocks.
- NOTE: The arguments are determined at lilypond compile time.
- See `org-babel-lilypond-set-header-args'
- To configure, see `ob-lilypond-header-args'
- .")
- (defvar ob-lilypond-header-args
- '((:results . "file") (:exports . "results"))
- "User-configurable header arguments for lilypond code blocks.
- NOTE: The final value used by org-babel is computed at compile-time
- and stored in `org-babel-default-header-args:lilypond'
- See `org-babel-lilypond-set-header-args'.")
- (defvar org-babel-lilypond-compile-post-tangle t
- "Following the org-babel-tangle (C-c C-v t) command,
- org-babel-lilypond-compile-post-tangle determines whether ob-lilypond should
- automatically attempt to compile the resultant tangled file.
- If the value is nil, no automated compilation takes place.
- Default value is t.")
- (defvar org-babel-lilypond-display-pdf-post-tangle t
- "Following a successful LilyPond compilation
- org-babel-lilypond-display-pdf-post-tangle determines whether to automate the
- drawing / redrawing of the resultant pdf. If the value is nil,
- the pdf is not automatically redrawn. Default value is t.")
- (defvar org-babel-lilypond-play-midi-post-tangle t
- "Following a successful LilyPond compilation
- org-babel-lilypond-play-midi-post-tangle determines whether to automate the
- playing of the resultant midi file. If the value is nil,
- the midi file is not automatically played. Default value is t")
- (defvar org-babel-lilypond-ly-command ""
- "Command to execute lilypond on your system.
- Do not set it directly. Customize `org-babel-lilypond-commands' instead.")
- (defvar org-babel-lilypond-pdf-command ""
- "Command to show a PDF file on your system.
- Do not set it directly. Customize `org-babel-lilypond-commands' instead.")
- (defvar org-babel-lilypond-midi-command ""
- "Command to play a MIDI file on your system.
- Do not set it directly. Customize `org-babel-lilypond-commands' instead.")
- (defcustom org-babel-lilypond-commands
- (cond
- ((eq system-type 'darwin)
- '("/Applications/lilypond.app/Contents/Resources/bin/lilypond" "open" "open"))
- ((eq system-type 'windows-nt)
- '("lilypond" "" ""))
- (t
- '("lilypond" "xdg-open" "xdg-open")))
- "Commands to run lilypond and view or play the results.
- These should be executables that take a filename as an argument.
- On some system it is possible to specify the filename directly
- and the viewer or player will be determined from the file type;
- you can leave the string empty on this case."
- :group 'org-babel
- :type '(list
- (string :tag "Lilypond ")
- (string :tag "PDF Viewer ")
- (string :tag "MIDI Player"))
- :version "24.4"
- :package-version '(Org . "8.2.7")
- :set
- (lambda (symbol value)
- (set-default-toplevel-value symbol value)
- (setq
- org-babel-lilypond-ly-command (nth 0 value)
- org-babel-lilypond-pdf-command (nth 1 value)
- org-babel-lilypond-midi-command (nth 2 value))))
- (defvar org-babel-lilypond-gen-png nil
- "Non-nil means image generation (PNG) is turned on by default.")
- (defvar org-babel-lilypond-gen-svg nil
- "Non-nil means image generation (SVG) is be turned on by default.")
- (defvar org-babel-lilypond-gen-html nil
- "Non-nil means HTML generation is turned on by default.")
- (defvar org-babel-lilypond-gen-pdf nil
- "Non-nil means PDF generation is be turned on by default.")
- (defvar org-babel-lilypond-use-eps nil
- "Non-nil forces the compiler to use the EPS backend.")
- (defvar org-babel-lilypond-arrange-mode nil
- "Non-nil turns Arrange mode on.
- In Arrange mode the following settings are altered from default:
- :tangle yes, :noweb yes
- :results silent :comments yes.
- In addition lilypond block execution causes tangling of all lilypond
- blocks.")
- (defun org-babel-expand-body:lilypond (body params)
- "Expand BODY according to PARAMS, return the expanded body."
- (let ((vars (org-babel--get-vars params)))
- (mapc
- (lambda (pair)
- (let ((name (symbol-name (car pair)))
- (value (cdr pair)))
- (setq body
- (replace-regexp-in-string
- (concat "$" (regexp-quote name))
- (if (stringp value) value (format "%S" value))
- body))))
- vars)
- body))
- (defun org-babel-execute:lilypond (body params)
- "This function is called by `org-babel-execute-src-block'.
- Depending on whether we are in arrange mode either:
- 1. Attempt to execute lilypond block according to header settings
- (This is the default basic mode)
- 2. Tangle all lilypond blocks and process the result (arrange mode)"
- (org-babel-lilypond-set-header-args org-babel-lilypond-arrange-mode)
- (if org-babel-lilypond-arrange-mode
- (org-babel-lilypond-tangle)
- (org-babel-lilypond-process-basic body params)))
- (defun org-babel-lilypond-tangle ()
- "ob-lilypond specific tangle, attempts to invoke
- =ly-execute-tangled-ly= if tangle is successful. Also passes
- specific arguments to =org-babel-tangle=."
- (interactive)
- (if (org-babel-tangle nil "yes" "lilypond")
- (org-babel-lilypond-execute-tangled-ly) nil))
- (defun org-babel-lilypond-process-basic (body params)
- "Execute a lilypond block in basic mode."
- (let* ((out-file (cdr (assq :file params)))
- (cmdline (or (cdr (assq :cmdline params))
- ""))
- (in-file (org-babel-temp-file "lilypond-")))
- (with-temp-file in-file
- (insert (org-babel-expand-body:generic body params)))
- (org-babel-eval
- (concat
- org-babel-lilypond-ly-command
- " -dbackend=eps "
- "-dno-gs-load-fonts "
- "-dinclude-eps-fonts "
- (or (cdr (assoc (file-name-extension out-file)
- '(("pdf" . "--pdf ")
- ("ps" . "--ps ")
- ("png" . "--png "))))
- "--png ")
- "--output="
- (file-name-sans-extension out-file)
- " "
- cmdline
- in-file) "")) nil)
- (defun org-babel-prep-session:lilypond (_session _params)
- "Return an error because LilyPond exporter does not support sessions."
- (error "Sorry, LilyPond does not currently support sessions!"))
- (defun org-babel-lilypond-execute-tangled-ly ()
- "Compile result of block tangle with lilypond.
- If error in compilation, attempt to mark the error in lilypond org file."
- (when org-babel-lilypond-compile-post-tangle
- (let ((org-babel-lilypond-tangled-file (org-babel-lilypond-switch-extension
- (buffer-file-name) ".lilypond"))
- (org-babel-lilypond-temp-file (org-babel-lilypond-switch-extension
- (buffer-file-name) ".ly")))
- (if (not (file-exists-p org-babel-lilypond-tangled-file))
- (error "Error: Tangle Failed!")
- (when (file-exists-p org-babel-lilypond-temp-file)
- (delete-file org-babel-lilypond-temp-file))
- (rename-file org-babel-lilypond-tangled-file
- org-babel-lilypond-temp-file))
- (org-switch-to-buffer-other-window "*lilypond*")
- (erase-buffer)
- (org-babel-lilypond-compile-lilyfile org-babel-lilypond-temp-file)
- (goto-char (point-min))
- (if (org-babel-lilypond-check-for-compile-error org-babel-lilypond-temp-file)
- (error "Error in Compilation!")
- (other-window -1)
- (org-babel-lilypond-attempt-to-open-pdf org-babel-lilypond-temp-file)
- (org-babel-lilypond-attempt-to-play-midi org-babel-lilypond-temp-file)))))
- (defun org-babel-lilypond-compile-lilyfile (file-name &optional test)
- "Compile lilypond file and check for compile errors.
- FILE-NAME is full path to lilypond (.ly) file."
- (message "Compiling LilyPond...")
- (let ((arg-1 org-babel-lilypond-ly-command) ;program
- ;; (arg-2 nil) ;infile
- (arg-3 "*lilypond*") ;buffer
- (arg-4 t) ;display
- (arg-5 (if org-babel-lilypond-gen-png "--png" "")) ;&rest...
- (arg-6 (if org-babel-lilypond-gen-html "--html" ""))
- (arg-7 (if org-babel-lilypond-gen-pdf "--pdf" ""))
- (arg-8 (if org-babel-lilypond-use-eps "-dbackend=eps" ""))
- (arg-9 (if org-babel-lilypond-gen-svg "-dbackend=svg" ""))
- (arg-10 (concat "--output=" (file-name-sans-extension file-name)))
- (arg-11 file-name))
- (if test
- `(,arg-1 ,nil ,arg-3 ,arg-4 ,arg-5 ,arg-6 ;; arg-2
- ,arg-7 ,arg-8 ,arg-9 ,arg-10 ,arg-11)
- (call-process
- arg-1 nil arg-3 arg-4 arg-5 arg-6 ;; arg-2
- arg-7 arg-8 arg-9 arg-10 arg-11))))
- (defun org-babel-lilypond-check-for-compile-error (file-name &optional test)
- "Check for compile error.
- This is performed by parsing the *lilypond* buffer
- containing the output message from the compilation.
- FILE-NAME is full path to lilypond file.
- If TEST is t just return nil if no error found, and pass
- nil as file-name since it is unused in this context."
- (let ((is-error (search-forward "error:" nil t)))
- (if test
- is-error
- (when is-error
- (org-babel-lilypond-process-compile-error file-name)))))
- (defun org-babel-lilypond-process-compile-error (file-name)
- "Process the compilation error that has occurred.
- FILE-NAME is full path to lilypond file."
- (let ((line-num (org-babel-lilypond-parse-line-num)))
- (let ((error-lines (org-babel-lilypond-parse-error-line file-name line-num)))
- (org-babel-lilypond-mark-error-line file-name error-lines)
- (error "Error: Compilation Failed!"))))
- (defun org-babel-lilypond-mark-error-line (file-name line)
- "Mark the erroneous lines in the lilypond org buffer.
- FILE-NAME is full path to lilypond file.
- LINE is the erroneous line."
- (org-switch-to-buffer-other-window
- (concat (file-name-nondirectory
- (org-babel-lilypond-switch-extension file-name ".org"))))
- (let ((temp (point)))
- (goto-char (point-min))
- (setq case-fold-search nil)
- (if (search-forward line nil t)
- (progn
- (org-fold-show-all)
- (set-mark (point))
- (goto-char (- (point) (length line))))
- (goto-char temp))))
- (defun org-babel-lilypond-parse-line-num (&optional buffer)
- "Extract error line number."
- (when buffer (set-buffer buffer))
- (let ((start
- (and (search-backward ":" nil t)
- (search-backward ":" nil t)
- (search-backward ":" nil t)
- (search-backward ":" nil t))))
- (when start
- (forward-char)
- (let ((num (string-to-number
- (buffer-substring
- (+ 1 start)
- (- (search-forward ":" nil t) 1)))))
- (and (numberp num) num)))))
- (defun org-babel-lilypond-parse-error-line (file-name lineNo)
- "Extract the erroneous line from the tangled .ly file.
- FILE-NAME is full path to lilypond file.
- LINENO is the number of the erroneous line."
- (with-temp-buffer
- (insert-file-contents (org-babel-lilypond-switch-extension file-name ".ly")
- nil nil nil t)
- (if (> lineNo 0)
- (progn
- (goto-char (point-min))
- (forward-line (- lineNo 1))
- (buffer-substring (point) (line-end-position)))
- nil)))
- (defun org-babel-lilypond-attempt-to-open-pdf (file-name &optional test)
- "Attempt to display the generated pdf file.
- FILE-NAME is full path to lilypond file.
- If TEST is non-nil, the shell command is returned and is not run."
- (when org-babel-lilypond-display-pdf-post-tangle
- (let ((pdf-file (org-babel-lilypond-switch-extension file-name ".pdf")))
- (if (file-exists-p pdf-file)
- (let ((cmd-string
- (concat org-babel-lilypond-pdf-command " " pdf-file)))
- (if test
- cmd-string
- (start-process
- "\"Audition pdf\""
- "*lilypond*"
- org-babel-lilypond-pdf-command
- pdf-file)))
- (message "No pdf file generated so can't display!")))))
- (defun org-babel-lilypond-attempt-to-play-midi (file-name &optional test)
- "Attempt to play the generated MIDI file.
- FILE-NAME is full path to lilypond file.
- If TEST is non-nil, the shell command is returned and is not run."
- (when org-babel-lilypond-play-midi-post-tangle
- (let* ((ext (if (eq system-type 'windows-nt)
- ".mid" ".midi"))
- (midi-file (org-babel-lilypond-switch-extension file-name ext)))
- (if (file-exists-p midi-file)
- (let ((cmd-string
- (concat org-babel-lilypond-midi-command " " midi-file)))
- (if test
- cmd-string
- (start-process
- "\"Audition midi\""
- "*lilypond*"
- org-babel-lilypond-midi-command
- midi-file)))
- (message "No midi file generated so can't play!")))))
- (defun org-babel-lilypond-toggle-midi-play ()
- "Toggle whether midi will be played following a successful compilation."
- (interactive)
- (setq org-babel-lilypond-play-midi-post-tangle
- (not org-babel-lilypond-play-midi-post-tangle))
- (message (concat "Post-Tangle MIDI play has been "
- (if org-babel-lilypond-play-midi-post-tangle
- "ENABLED." "DISABLED."))))
- (defun org-babel-lilypond-toggle-pdf-display ()
- "Toggle whether pdf will be displayed following a successful compilation."
- (interactive)
- (setq org-babel-lilypond-display-pdf-post-tangle
- (not org-babel-lilypond-display-pdf-post-tangle))
- (message (concat "Post-Tangle PDF display has been "
- (if org-babel-lilypond-display-pdf-post-tangle
- "ENABLED." "DISABLED."))))
- (defun org-babel-lilypond-toggle-png-generation ()
- "Toggle whether png image will be generated by compilation."
- (interactive)
- (setq org-babel-lilypond-gen-png (not org-babel-lilypond-gen-png))
- (message (concat "PNG image generation has been "
- (if org-babel-lilypond-gen-png "ENABLED." "DISABLED."))))
- (defun org-babel-lilypond-toggle-html-generation ()
- "Toggle whether html will be generated by compilation."
- (interactive)
- (setq org-babel-lilypond-gen-html (not org-babel-lilypond-gen-html))
- (message (concat "HTML generation has been "
- (if org-babel-lilypond-gen-html "ENABLED." "DISABLED."))))
- (defun org-babel-lilypond-toggle-pdf-generation ()
- "Toggle whether pdf will be generated by compilation."
- (interactive)
- (setq org-babel-lilypond-gen-pdf (not org-babel-lilypond-gen-pdf))
- (message (concat "PDF generation has been "
- (if org-babel-lilypond-gen-pdf "ENABLED." "DISABLED."))))
- (defun org-babel-lilypond-toggle-arrange-mode ()
- "Toggle whether in Arrange mode or Basic mode."
- (interactive)
- (setq org-babel-lilypond-arrange-mode
- (not org-babel-lilypond-arrange-mode))
- (message (concat "Arrange mode has been "
- (if org-babel-lilypond-arrange-mode "ENABLED." "DISABLED."))))
- (defun org-babel-lilypond-switch-extension (file-name ext)
- "Utility command to swap current FILE-NAME extension with EXT."
- (concat (file-name-sans-extension
- file-name)
- ext))
- (defun org-babel-lilypond-get-header-args (mode)
- "Default arguments to use when evaluating a lilypond source block.
- These depend upon whether we are in Arrange mode i.e. MODE is t."
- (cond (mode
- '((:tangle . "yes")
- (:noweb . "yes")
- (:results . "silent")
- (:cache . "yes")
- (:comments . "yes")))
- (t
- ob-lilypond-header-args)))
- (defun org-babel-lilypond-set-header-args (mode)
- "Set org-babel-default-header-args:lilypond
- dependent on ORG-BABEL-LILYPOND-ARRANGE-MODE."
- (setq org-babel-default-header-args:lilypond
- (org-babel-lilypond-get-header-args mode)))
- (provide 'ob-lilypond)
- ;;; ob-lilypond.el ends here
|