From ad7e774fe7056a27ecf28d702cdabd82c62d58a4 Mon Sep 17 00:00:00 2001 From: TobiasZawada Date: Thu, 8 Feb 2024 08:34:41 +0100 Subject: [PATCH] Work/47display images (#48) * #47 Adapt display-images from markdown-mode to adoc-mode * Toggle display of images * Try to get working CI-tests by requiring mouse and image * Correct package version from markdown 0.8.0 to adoc 0.9.0. * Remove dependency on context-menu-mode. * Avoid failing CI-tests through attempt to show images at start * when-let has moved from subr-x to subr * Tell the byte-compiler that imagep and image-flush are defined. * In defcustom booleanp is not a valid type. * Add display of images to README.adoc. - remove more "markdown" and "inline" - add context menu to image links - fontification of images before fontification of general links, since image links can contain urls * require subr-x simplification * Rename adoc-display-images-at-startup to adoc-display-images. Add Usage section to README.adoc with a description of the not so obvious features. --------- Co-authored-by: Tobias Zawada --- README.adoc | 44 +++++++- adoc-mode.el | 305 +++++++++++++++++++++++++++++++++++++++++++++++++-- 2 files changed, 336 insertions(+), 13 deletions(-) diff --git a/README.adoc b/README.adoc index 5dc1aca..5f62579 100644 --- a/README.adoc +++ b/README.adoc @@ -35,6 +35,7 @@ Here are some of the main features of `adoc-mode`: - sophisticated highlighting - native fontification of code blocks +- toggle between the display of image links and the display of the corresponding images - promote / demote title - toggle title type between one line title and two line title - adjust underline length of a two line title to match title text's length @@ -44,9 +45,39 @@ Here are some of the main features of `adoc-mode`: === Demo The highlighting emphasizes on how the output will look like. _All_ -characters are visible, however meta characters are displayed in a faint way. +characters are visible, however meta characters are displayed in a +faint way. An exception are image links. You can toggle between +the display of the link text and the linked image. -image:images/adoc-mode.png[alt=screenshot] +image::images/adoc-mode.png[alt=screenshot] + +== Usage + +Adoc-mode is designed for intuitive use. Most features are available +from the _AsciiDoc_ menu in the menu bar. + +This section only describes some not so obvious features. + +=== Image Preview +If you click kbd:[mouse-3] on an image link like +`image⁣:path/to/image[]` the _Image_ context menu pops up. + +This context menu allows you to generate or remove the preview for +the linked image. + +The menu item _AsciiDoc → Toggle display of images_ runs the command +`adoc-toggle-images`. It removes all image previews from the buffer +if there are any. If there are no image previews in the buffer it +generates them for all image links. + +This may be confusing if all image previews are outside the visible +region of the buffer. In this case, you might expect `adoc-toggle-images` +to generate the previews for the image links in the visible area on +the first run. But it does not, since it first removes all the +previews, even if you have not seen them. + +Just run `adoc-toggle-images` again if it does not what you expect on +the first run. == Installation @@ -111,6 +142,15 @@ This can be tweaked by several configuration options: * All programming languages `XYZ` that have an Emacs major mode `XYZ-mode` and use `font-lock` are automatically supported. Some other languages not fitting into that name scheme are supported through the alist `adoc-code-lang-modes`. You can add your own languages and modes there if they work based on `font-lock` and are not automatically supported. * The fall-back language mode is `prog-mode` without any fontification. You can set your own default by `adoc-fontify-code-block-default-mode`. +=== Display of Image Previews + +Images are shown as preview by default when you open an AsciiDoc file. +You can avoid this by deactivating the option `adoc-display-images`. + +The maximal size for the preview of images can be set by `adoc-max-image-size`. + +An image link can also be given as url to a remote image. The display of remote images is switched off by default. You can activate it by the option `adoc-display-remote-images`. +Thereby, images are only downloaded if the protocol of the url matches one of those in the list `adoc-remote-image-protocols`. This list can be customized. By default, it only contains the entry `https`. === Syntax Highlighting Customization It is possible to customize the way `adoc-mode` renders different text diff --git a/adoc-mode.el b/adoc-mode.el index 9e3b5c8..13aaee7 100644 --- a/adoc-mode.el +++ b/adoc-mode.el @@ -44,6 +44,10 @@ (require 'cl-lib) (require 'tempo) +(require 'subr-x) + +(declare-function imagep "image.c") +(declare-function image-flush "image.c") (defconst adoc-mode-version "0.8.0-snapshot" "adoc mode version number. @@ -301,6 +305,27 @@ Also used to delimit the scan for the end delimiter." :group 'adoc :package-version '(adoc-mode . "0.8.0")) +(defcustom adoc-max-image-size nil + "Maximum width and height for displayed images. +This variable may be nil or a cons cell (MAX-WIDTH . MAX-HEIGHT). +When nil, use the actual size. Otherwise, use ImageMagick to +resize larger images to be of the given maximum dimensions. This +requires Emacs to be built with ImageMagick support." + :group 'adoc + :package-version '(adoc-mode . "0.9.0") + :type '(choice + (const :tag "Use actual image width" nil) + (cons (choice (sexp :tag "Maximum width in pixels") + (const :tag "No maximum width" nil)) + (choice (sexp :tag "Maximum height in pixels") + (const :tag "No maximum height" nil))))) + +(defcustom adoc-display-images t + "Run `adoc-display-images' in function `adoc-mode'." + :group 'adoc + :package-version '(adoc-mode . "0.9.0") + :type 'boolean) + ;;;; faces / font lock (define-obsolete-face-alias 'adoc-orig-default 'adoc-align-face "23.3") @@ -1806,7 +1831,7 @@ When LITERAL-P is non-nil, the contained text is literal text." '(3 nil)) ; grumbl, I don't know how to get rid of it `(4 '(face ,(or del-face adoc-meta-hide-face) adoc-reserved t) t))); close del -(defun adoc-kw-inline-macro (&optional cmd-name unconstrained attribute-list-constraints cmd-face target-faces target-meta-p attribute-list) +(defun adoc-kw-inline-macro (&optional cmd-name unconstrained attribute-list-constraints cmd-face target-faces target-meta-p attribute-list textprops) "Returns a kewyword which highlights an inline macro. For CMD-NAME and UNCONSTRAINED see @@ -1816,20 +1841,22 @@ determines face for the target text. If nil `adoc-meta-face' is used. If a list, the first is used if the attribute list is the empty string, the second is used if its not the empty string. If TARGET-META-P is non-nil, the target text is considered to be -meta characters." +meta characters. +TEXTPROPS is an additional plist with textproperties." (list `(lambda (end) (adoc-kwf-std end ,(adoc-re-inline-macro cmd-name nil unconstrained attribute-list-constraints) '(1 2 4 5) '(0))) + `(0 '(face nil . ,textprops) t) `(1 '(face ,(or cmd-face adoc-command-face) adoc-reserved t) t) ; cmd-name - '(2 '(face adoc-meta-face adoc-reserved t) t) ; : + '(2 '(face adoc-meta-face adoc-reserved t . ,textprops) t) ; : `(3 ,(cond ((not target-faces) adoc-meta-face) ; target ((listp target-faces) `(if (string= (match-string 5) "") ; 5=attribute-list ,(car target-faces) ,(cadr target-faces))) (t target-faces)) ,(if target-meta-p t 'append)) - '(4 '(face adoc-meta-face adoc-reserved t) t) ; [ - `(5 '(face adoc-meta-face adoc-attribute-list ,(or attribute-list t)) t) - '(6 '(face adoc-meta-face adoc-reserved t) t))) ; ] + '(4 '(face adoc-meta-face adoc-reserved t . ,textprops) t) ; [ + `(5 '(face adoc-meta-face adoc-attribute-list ,(or attribute-list t) . ,textprops) t) + '(6 '(face adoc-meta-face adoc-reserved t . ,textprops) t))) ; ] ;; largely copied from adoc-kw-inline-macro ;; TODO: output text should be affected by quotes & co, e.g. bold, emph, ... @@ -2257,7 +2284,7 @@ Use this function as matching function MATCHER in `font-lock-keywords'." ;; image. The first positional attribute is per definition 'alt', see ;; asciidoc manual, sub chapter 'Image macro attributes'. (list `(lambda (end) (adoc-kwf-std end ,(adoc-re-block-macro "image") '(0))) - '(0 '(face adoc-meta-face adoc-reserved block-del) t) ; whole match + '(0 '(face adoc-meta-face adoc-reserved block-del keymap adoc-image-link-map) t) ; whole match '(1 adoc-complex-replacement-face t) ; 'image' '(2 adoc-internal-reference-face t) ; file name '(3 '(face adoc-meta-face adoc-reserved nil adoc-attribute-list ("alt")) t)) ; attribute list @@ -2498,11 +2525,12 @@ Use this function as matching function MATCHER in `font-lock-keywords'." ;; - same order as in asciidoc.conf (is that in 'reverse'? cause 'default syntax' comes first) ;; Macros using default syntax, but having special highlighting in adoc-mode + ;; Remote inline images may contain an url. So they must be fontified before general urls. + (adoc-kw-inline-macro "image" nil nil adoc-complex-replacement-face adoc-internal-reference-face t + '("alt") '(keymap adoc-image-link-map)) (adoc-kw-inline-macro-urls-no-attribute-list) (adoc-kw-inline-macro-urls-attribute-list) (adoc-kw-inline-macro "anchor" nil nil nil adoc-anchor-face t '("xreflabel")) - (adoc-kw-inline-macro "image" nil nil adoc-complex-replacement-face adoc-internal-reference-face t - '("alt")) (adoc-kw-inline-macro "xref" nil nil nil '(adoc-reference-face adoc-internal-reference-face) t '(("caption") (("caption" . adoc-reference-face)))) (adoc-kw-inline-macro "footnote" t nil nil nil nil adoc-secondary-text-face) @@ -2877,6 +2905,257 @@ Is influenced by customization variables such as `adoc-title-style'.")))) (adoc-tempo-define "adoc-pass-+++" '("+++" (r "text" text) "+++") nil adoc-help-pass-+++) (adoc-tempo-define "adoc-pass-$$" '("$$" (r "text" text) "$$") nil adoc-help-pass-$$) ; backticks handled in tempo-template-adoc-monospace-literal + +;;;; Images +;; Disclaimer: Most of the following stuff is copied from `markdown-mode' and adapted to `adoc-mode'. +(require 'url-parse) + +(defvar adoc-image-overlays nil) +(make-variable-buffer-local 'adoc-image-overlays) + +(defcustom adoc-display-remote-images nil + "If non-nil, download and display remote images. +See also `adoc-image-overlays'. + +Only image URLs specified with a protocol listed in +`adoc-remote-image-protocols' are displayed." + :group 'adoc + :type 'boolean) + +(defcustom adoc-remote-image-protocols '("https") + "List of protocols to use to download remote images. +See also `adoc-display-remote-images'." + :group 'adoc + :type '(repeat string)) + +(defvar adoc--remote-image-cache + (make-hash-table :test 'equal) + "A map from URLs to image paths.") + +(defun adoc--get-remote-image (url) + "Retrieve the image path for a given URL." + (or (gethash url adoc--remote-image-cache) + (let ((dl-path (make-temp-file "adoc-mode--image"))) + (require 'url) + (url-copy-file url dl-path t) + (puthash url dl-path adoc--remote-image-cache)))) + +(defconst adoc-re-image "\\") + (lambda () (interactive) (popup-menu adoc-image-link-menu))) + map) + "Keymap for image links.") + +(fset 'adoc-image-link-map adoc-image-link-map) + +(defvar adoc-image-overlay-menu + (let ((map (make-sparse-keymap "Image"))) + (set-keymap-parent map adoc-image-menu) + (easy-menu-add-item map nil ["Remove Preview At Point" + (lambda (e) (interactive "e") (adoc-remove-image-overlay-at e)) + t]) + map) + "Context menu of image overlays.") + +(defvar adoc-image-overlay-map + (let ((map (make-sparse-keymap "Image"))) + (define-key map (kbd "") + (lambda () (interactive) (popup-menu adoc-image-overlay-menu))) + map) + "Keymap for image overlays.") + +(defun adoc-create-image-overlay (file start end) + "Create image overlay with START and END displaying image FILE." + (when (not (zerop (length file))) + (unless (file-exists-p file) + (when (and adoc-display-remote-images + (member (downcase (url-type (url-generic-parse-url file))) + adoc-remote-image-protocols)) + (setq file (adoc--get-remote-image file)))) + (when (file-exists-p file) + (let* ((abspath (if (file-name-absolute-p file) + file + (concat default-directory file))) + (image + (if (and adoc-max-image-size + (image-type-available-p 'imagemagick)) + (create-image + abspath 'imagemagick nil + :max-width (car adoc-max-image-size) + :max-height (cdr adoc-max-image-size)) + (create-image abspath)))) + (when image + (let ((ov (make-overlay start end))) + (overlay-put ov 'adoc-image t) + (overlay-put ov 'display image) + (overlay-put ov 'face 'default) + (overlay-put ov 'keymap adoc-image-overlay-map) + (run-hook-with-args 'adoc-image-overlay-functions ov))))))) + +(defun adoc-display-images () + "Add inline image overlays to image links in the buffer. +This can be toggled with `adoc-toggle-images' +or \\[adoc-toggle-images]." + (interactive) + (unless (display-images-p) + (error "Cannot show images")) + (adoc-remove-images) + (save-excursion + (save-restriction + (widen) + (goto-char (point-min)) + (while (re-search-forward adoc-re-image nil t) + (let ((start (match-beginning 0)) + (end (match-end 0)) + (file (match-string-no-properties 1))) + (adoc-create-image-overlay file start end) + ))))) + +(defun adoc-image-overlays (&optional begin end) + "Return list of image overlays in region from BEGIN to END. +BEGIN and END default to the buffer boundaries +ignoring any restrictions." + (save-restriction + (widen) + (unless begin (setq begin (point-min))) + (unless end (setq end (point-max))) + (let (image-overlays) + (dolist (ov (overlays-in begin end)) + (when (overlay-get ov 'adoc-image) + (push ov image-overlays))) + image-overlays))) + +(defun adoc-remove-images () + "Remove image overlays from image links in the buffer. +This can be toggled with `adoc-toggle-images' +or \\[adoc-toggle-images]." + (interactive) + (let ((image-list (adoc-image-overlays))) + (when image-list + (dolist (ov image-list) + (delete-overlay ov)) + t))) + +(defun adoc-toggle-images () + "Toggle the display of images." + (interactive) + (unless (adoc-remove-images) + (adoc-display-images) + )) + +(defmacro adoc-with-point-at-event (location &rest body) + "Execute BODY like `progn'. +If LOCATION is an event run BODY in the event's buffer +and set LOCATION to the event position. + +If LOCATION is number or marker just run BODY with +LOCATION untouched. + +Otherwise set LOCATION to current point." + (declare (indent 1) (debug (symbolp form body))) + (let ((posn (make-symbol "posn"))) + `(if (mouse-event-p ,location) + (let* ((,posn (event-start ,location))) + (with-current-buffer (window-buffer (posn-window ,posn)) + (setq ,location (posn-point ,posn)) + ,@body)) + (unless (number-or-marker-p ,location) + (setq ,location (point))) + ,@body))) + +(defun adoc-bounds-of-image-link-at (&optional location) + "Get bounds of image link at LOCATION. +Return \\(START . END) giving the start and the end +positions of the image link found. + +If an image link is identified `match-data' is set as follows: +0. whole link inclusive \"image::?\" and attributes +1. image path +2. bracketed attribute list." + (adoc-with-point-at-event location + (save-excursion + (when location (goto-char location)) + (setq location (point)) + (if (looking-at "[[:alpha:]]*::?\\_<") + (skip-chars-backward "[:alpha:]") + (re-search-backward "\\_