[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]
Re: yank-media: allow users to limit image types that can be inserted
From: |
Visuwesh |
Subject: |
Re: yank-media: allow users to limit image types that can be inserted |
Date: |
Thu, 12 Dec 2024 21:48:30 +0530 |
User-agent: |
Gnus/5.13 (Gnus v5.13) |
[திங்கள் நவம்பர் 04, 2024] Visuwesh wrote:
To recap,
The OP asked for a way to choose a specific image type out of the
multiple types available in their clipboard. We decided that Emacs
should auto-select the best type itself and have a way for major-modes
to customise this selection process. Auto-selection should also be
skipped with a prefix arg.
> [ஞாயிறு நவம்பர் 03, 2024] Ihor Radchenko wrote:
>
>> Visuwesh <visuweshm@gmail.com> writes:
>>
>>>> And I'm saying that I don't understand why users would need such
>>>> control, in addition to being able to select a format in each case.
>>>
>>> OK, I think I finally understand your stance: we let yank-media pick the
>>> best format out of all in the clipboard based on rules that a major-mode
>>> can tweak. Apart from this, we also have C-u or somesuch that asks the
>>> user to select the data type instead of auto-selecting it. Since the
>>> C-u is good enough to override a case-by-case basis, you do not see
>>> value in having a defcustom to let the user customise the selection
>>> rules for the best data type, correct?
>>
>> So, what will be the plan here? A patch?
>
> I just want to inform that I probably won't be able to get to writing a
> patch until mid December since I hardly have time with my semester end
> nearing.
Here's the promised patch (I can create a bug report if that is
preferred):
diff --git a/lisp/yank-media.el b/lisp/yank-media.el
index 17981c37c0e..a5ad913c24a 100644
--- a/lisp/yank-media.el
+++ b/lisp/yank-media.el
@@ -29,19 +29,49 @@
(defvar yank-media--registered-handlers nil)
+(defvar yank-media-autoselect-function #'yank-media-autoselect-function
+ "Function to auto select the best mime type when many are available.
+The function is called with a list of mime types that have handler in
+the current buffer, and should return the type to use or nil if no
+preferred type is found.")
+
+(defun yank-media-autoselect-function (mimetypes)
+ (cond
+ ;; Check first since LibreOffice also puts a PNG image in the
+ ;; clipboard when a table cell is copied.
+ ((memq 'application/x-libreoffice-tsvc mimetypes)
+ 'application/x-libreoffice-tsvc)
+ ;; Give PNG more priority.
+ ((memq 'image/png mimetypes)
+ 'image/png)
+ ((memq 'image/jpeg mimetypes)
+ 'image/jpeg)
+ ;; These are files copied/cut to the clipboard from a file manager.
+ ((seq-find (lambda (type)
+ (string-match-p "x-special/\\(gnome\|KDE\|mate\)-files"
+ (symbol-name type)))
+ mimetypes))
+ ;; FIXME: We should have a way to handle text/rtf.
+ ((memq 'text/html mimetypes)
+ 'text/html)))
+
;;;###autoload
-(defun yank-media ()
+(defun yank-media (&optional noselect)
"Yank media (images, HTML and the like) from the clipboard.
This command depends on the current major mode having support for
accepting the media type. The mode has to register itself using
the `yank-media-handler' mechanism.
+Optional argument NOSELECT non-nil (interactively, with a prefix
+argument) means to skip auto-selecting the best mimetype and ask
+for the media type to use when multiple are available.
Also see `yank-media-types' for a command that lets you explore
all the different selection types."
- (interactive)
+ (interactive "P")
(unless yank-media--registered-handlers
(user-error "The `%s' mode hasn't registered any handlers" major-mode))
- (let ((all-types nil))
+ (let ((all-types nil)
+ pref-type)
(pcase-dolist (`(,handled-type . ,handler)
yank-media--registered-handlers)
(dolist (type (yank-media--find-matching-media handled-type))
@@ -49,18 +79,27 @@ yank-media
(unless all-types
(user-error
"No handler in the current buffer for anything on the clipboard"))
- ;; We have a handler in the current buffer; if there's just
- ;; matching type, just call the handler.
- (if (length= all-types 1)
- (funcall (cdar all-types) (caar all-types)
- (yank-media--get-selection (caar all-types)))
- ;; More than one type the user for what type to insert.
+ (setq pref-type (and (null noselect)
+ (funcall yank-media-autoselect-function
+ (mapcar #'car all-types))))
+ (cond
+ ;; We have a preferred mime type so use it unconditionally.
+ ((and (null noselect) pref-type)
+ (funcall (cdr (assq pref-type all-types)) pref-type
+ (yank-media--get-selection pref-type)))
+ ;; The user chose to not autoselet and there's just a single type,
+ ;; just call the handler.
+ ((length= all-types 1)
+ (funcall (cdar all-types) (caar all-types)
+ (yank-media--get-selection (caar all-types))))
+ ;; More than one type the user for what type to insert.
+ (t
(let ((type
(intern
(completing-read "Several types available, choose one: "
(mapcar #'car all-types) nil t))))
(funcall (alist-get type all-types)
- type (yank-media--get-selection type))))))
+ type (yank-media--get-selection type)))))))
(defun yank-media--find-matching-media (handled-type)
(seq-filter
As planned, yank-media auto-selects the best media type out of the
available types by default. This auto-selection can be skipped by
giving a prefix arg. Some details about the selection process itself:
1. The autoselect function can return only one preferred type.
2. There is no user option to control this selection process. I
cannot envision such an option myself. Major mode authors can
control the selection rules by using add-function (like they
already do with e.g., filter-buffer-substring-function):
(add-function :before/:after/:around/...
(local 'yank-media-autoselect-function)
#'major-mode-ymaf)
An informed user could do the same if she desires.
3. The autoselect function _can_ return nil. If it does, then we
proceed as if the prefix arg was given. This makes sense to me
since we cannot hope to be exhaustive since the mime type can be
_anything_. For e.g., Avogadro puts chemical/x-mdl-molfile when
you select and copy atoms but I doubt many Emacs user would ever
come across such a type in their clipboard. So it is best we fall
over to the NOSELECT=t when the autoselect function returns nil
instead of signalling an error.
4. We select application/x-libreoffice-tsvc first since LibreOffice
also puts image/png when you copy table cell(s).
5. We prefer PNG images over other types.
6. We prefer text/html but I believe the plan is to eventually prefer
text/rtf?
Now some questions:
1. Should we allow the autoselect function to return _multiple_
preferred types?
2. Should we prefer image/svg over image/png and image/jpeg? Should
we prefer it over image/jpeg at least? Or do we leave it to the
major-mode to take care of this (e.g., as in Robert's example)?
3. We prefer images over files cut/copied to the clipboard. I don't
know if any software puts both image/png and
x-special/gnome-copied-files to the clipboard. If it does, which
do we prefer? The list of file:// links or raw image/png data?
4. The mimetype used for cut/copied files only works in Linux
environments. If other platforms can present such file:// links in
the clipboard and Emacs supports it, we would need to add it to the
list too.
- Re: yank-media: allow users to limit image types that can be inserted,
Visuwesh <=