From mboxrd@z Thu Jan 1 00:00:00 1970 Return-Path: Received: from mp12.migadu.com ([2001:41d0:306:2d92::]) (using TLSv1.3 with cipher TLS_AES_256_GCM_SHA384 (256/256 bits)) by ms8.migadu.com with LMTPS id 4C5qGFi/NGWpwwAAauVa8A:P1 (envelope-from ) for ; Sun, 22 Oct 2023 08:21:12 +0200 Received: from aspmx1.migadu.com ([2001:41d0:306:2d92::]) (using TLSv1.3 with cipher TLS_AES_256_GCM_SHA384 (256/256 bits)) by mp12.migadu.com with LMTPS id 4C5qGFi/NGWpwwAAauVa8A (envelope-from ) for ; Sun, 22 Oct 2023 08:21:12 +0200 Received: from lists.gnu.org (lists.gnu.org [209.51.188.17]) (using TLSv1.2 with cipher ECDHE-RSA-AES256-GCM-SHA384 (256/256 bits)) (No client certificate requested) by aspmx1.migadu.com (Postfix) with ESMTPS id C1C1A63DAE for ; Sun, 22 Oct 2023 08:21:11 +0200 (CEST) Authentication-Results: aspmx1.migadu.com; dkim=pass header.d=gmail.com header.s=20230601 header.b=eeeWJMkC; spf=pass (aspmx1.migadu.com: domain of "emacs-orgmode-bounces+larch=yhetil.org@gnu.org" designates 209.51.188.17 as permitted sender) smtp.mailfrom="emacs-orgmode-bounces+larch=yhetil.org@gnu.org"; dmarc=pass (policy=none) header.from=gmail.com ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=yhetil.org; s=key1; t=1697955672; h=from:from:sender:sender:reply-to:subject:subject:date:date: message-id:message-id:to:to:cc:cc:mime-version:mime-version: content-type:content-type:in-reply-to:in-reply-to: references:references:list-id:list-help:list-unsubscribe: list-subscribe:list-post:dkim-signature; bh=rd2OLc+Wj00W6erSkiU1ILGwC3dqm7NcpW+PnDvWvJI=; b=n2ntSXg/wFt7GxYh1uZyv0ZdNWzohtwjpgX37V9LDkNHCH75IapKOri1wRu2T9aOMUzp+b 0lKpbz70flktzpQRStcfgqCNQzxPoR0zY1rrXJi5+oP2H54M4JY/ydUeMEV7cTMoWZ36eJ FIy0pJR3QYA/NHhNJ6fHKzAY7/WDDeCmHBUSa4bk6EIs09q3M4buMbguZBggioalOLe/Tq 07ruOEQb6QSbGcQRFEaHyMFqwRrNkpNqSqre/+mDqImA485kuK61X/XBG+RvXxJCmJrrSy M7JxZbVbqk4VHBYqx+7uY6N1eJKoAtDvVOcf5PY5YWs2z0DIvqmiOrMzFyWs0Q== ARC-Authentication-Results: i=1; aspmx1.migadu.com; dkim=pass header.d=gmail.com header.s=20230601 header.b=eeeWJMkC; spf=pass (aspmx1.migadu.com: domain of "emacs-orgmode-bounces+larch=yhetil.org@gnu.org" designates 209.51.188.17 as permitted sender) smtp.mailfrom="emacs-orgmode-bounces+larch=yhetil.org@gnu.org"; dmarc=pass (policy=none) header.from=gmail.com ARC-Seal: i=1; s=key1; d=yhetil.org; t=1697955672; a=rsa-sha256; cv=none; b=EdVSS9WvMGpj+vR72ZbD5Nj6U9B5nxPV0PBf5JGuVx9VsG7xxXIu5AilAd0EbbgVVy/nUJ 6F3AlCmTP4g1A19axXxAqV9xGEus0u2UjkZIW6yNsBj/rAhRap+1tDPJOx7OlItngu6pZd dGBqoRSzM8IRAfNjAgVLgYX7heMh82zv/uvD2cvxHkD/QQcwR3eDO4+QKVp1MpE3doGKet 2yTs8aXcjq4pezD+BeGC45zfzAW0tQhOu0h3O4xlQP3Fl5GEbIpDTPaCut+zfsi/UlUIwG g6GJkEtva1o+4E8Cv+88z/CazvIgKoK3+bKdo1HkvZk6zsKN6UmRaiaMWXhhxQ== Received: from localhost ([::1] helo=lists1p.gnu.org) by lists.gnu.org with esmtp (Exim 4.90_1) (envelope-from ) id 1quRoo-0000Jy-IZ; Sun, 22 Oct 2023 02:20:11 -0400 Received: from eggs.gnu.org ([2001:470:142:3::10]) by lists.gnu.org with esmtps (TLS1.2:ECDHE_RSA_AES_256_GCM_SHA384:256) (Exim 4.90_1) (envelope-from ) id 1quRok-0000I1-DN for emacs-orgmode@gnu.org; Sun, 22 Oct 2023 02:20:07 -0400 Received: from mail-il1-x141.google.com ([2607:f8b0:4864:20::141]) by eggs.gnu.org with esmtps (TLS1.2:ECDHE_RSA_AES_128_GCM_SHA256:128) (Exim 4.90_1) (envelope-from ) id 1quRog-0005RG-GE for emacs-orgmode@gnu.org; Sun, 22 Oct 2023 02:20:05 -0400 Received: by mail-il1-x141.google.com with SMTP id e9e14a558f8ab-3579f5392e7so9884435ab.3 for ; Sat, 21 Oct 2023 23:20:01 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=gmail.com; s=20230601; t=1697955601; x=1698560401; darn=gnu.org; h=mime-version:user-agent:references:message-id:date:in-reply-to :subject:cc:to:from:from:to:cc:subject:date:message-id:reply-to; bh=rd2OLc+Wj00W6erSkiU1ILGwC3dqm7NcpW+PnDvWvJI=; b=eeeWJMkCZqhNIIErjEw6d75ZpxBz6zmUU3VunsEFfRT2p70IoBz34wFvIzqB9g7j9o UohmRa1/kVtByyGbnKfeXmo1X9UXdGezwoNQIvTliKH6OmYR6//VO8qsNf5MgAASdvLZ C3If8nxLcLfoCSzp+2dSP5qpbcAlrn6NBR2nr+Bekm5pe+CYEC5fEbyXLQoRC3PYReZt xFPQlIZGYIKT8NJYXGO1utJTXa4XEjRbAjyRPND9o2n5FBtj+oNgj/wAqA3g9sI4y66T puT2j2M+xGYM80tPmua5Lb6xFwl3sZH57plB/1B/dCWQlqKHIGcODw7kw/9LZ4lJshKU qTRg== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20230601; t=1697955601; x=1698560401; h=mime-version:user-agent:references:message-id:date:in-reply-to :subject:cc:to:from:x-gm-message-state:from:to:cc:subject:date :message-id:reply-to; bh=rd2OLc+Wj00W6erSkiU1ILGwC3dqm7NcpW+PnDvWvJI=; b=rtjAQUOAG2hz2YuOL2TBtBRhUS8+yFId0j13fNFWkh7f7DYwN5far5Ah5Mor/JFl2R l7CxNNwc9LUHvn9YAkQoeLiygZLl34pKiR1bTMSCgsPuAI/crp1rhx2K6y2pIr6Oh+nX tPdv7JM9umIxsEjMwU1tAME8D5YSRJLI5krN6emvDoKEj324Q2xiRP3tMEvKQWE3qbxE 6bfOoCLbK4b/GPF/axJ51KDyJ1M0NVrIrI8DkM/y2WYk3r0FwjJ3nRybP8F9cUPqE6n6 HZ1mnL7c+hOm0JlEeEKXf/J2i8++/mFGdAy62kOin5iEM4v2qLH0t4oRkB8cAlDgNh8I gVTg== X-Gm-Message-State: AOJu0YwcLr1L/E3f7C1/FwU2+eNrIIqXpvnU01gYbFx+G/XgaLWexNWw GcZB8l1Vpy5qx0mFoBaaYb57znPVulccwQ== X-Google-Smtp-Source: AGHT+IGq85QzDIsp5YcmPkn9j7nigoB446AWm2bk1o0Z7ylbNghlgh1Sc6iydhZO1uEyF5bUEzkrJw== X-Received: by 2002:a05:6e02:1889:b0:351:22a2:bbe7 with SMTP id o9-20020a056e02188900b0035122a2bbe7mr9732017ilu.0.1697955600792; Sat, 21 Oct 2023 23:20:00 -0700 (PDT) Received: from localhost ([115.240.90.130]) by smtp.gmail.com with ESMTPSA id gg20-20020a17090b0a1400b00276cb03a0e9sm3709575pjb.46.2023.10.21.23.19.59 (version=TLS1_3 cipher=TLS_AES_256_GCM_SHA384 bits=256/256); Sat, 21 Oct 2023 23:19:59 -0700 (PDT) From: Visuwesh To: Ihor Radchenko Cc: emacs-orgmode@gnu.org, Po Lu Subject: Re: [BUG] [PATCH] Add yank-media and DND handler [9.6.7 (9.6.7-g6eb773 @ /home/viz/lib/emacs/straight/build/org/)] In-Reply-To: <87edi4m4my.fsf@localhost> (Ihor Radchenko's message of "Mon, 09 Oct 2023 11:12:05 +0000") Date: Sun, 22 Oct 2023 11:49:45 +0530 Message-ID: <87wmvf89ha.fsf@gmail.com> References: <87jzsintv0.fsf@gmail.com> <87lecx2nff.fsf@localhost> <87bkdsomm2.fsf@gmail.com> <87a5tamm3j.fsf@gmail.com> <877cnyzlda.fsf@localhost> <87a5sur51r.fsf@gmail.com> <875y3ir0lu.fsf@gmail.com> <87ttr1pild.fsf@localhost> <87v8bhpdg8.fsf@gmail.com> <87edi4m4my.fsf@localhost> User-Agent: Gnus/5.13 (Gnus v5.13) MIME-Version: 1.0 Content-Type: multipart/mixed; boundary="=-=-=" Received-SPF: pass client-ip=2607:f8b0:4864:20::141; envelope-from=visuweshm@gmail.com; helo=mail-il1-x141.google.com X-Spam_score_int: -20 X-Spam_score: -2.1 X-Spam_bar: -- X-Spam_report: (-2.1 / 5.0 requ) BAYES_00=-1.9, DKIM_SIGNED=0.1, DKIM_VALID=-0.1, DKIM_VALID_AU=-0.1, DKIM_VALID_EF=-0.1, FREEMAIL_FROM=0.001, RCVD_IN_DNSWL_NONE=-0.0001, SPF_HELO_NONE=0.001, SPF_PASS=-0.001 autolearn=ham autolearn_force=no X-Spam_action: no action X-BeenThere: emacs-orgmode@gnu.org X-Mailman-Version: 2.1.29 Precedence: list List-Id: "General discussions about Org-mode." List-Unsubscribe: , List-Archive: List-Post: List-Help: List-Subscribe: , Errors-To: emacs-orgmode-bounces+larch=yhetil.org@gnu.org Sender: emacs-orgmode-bounces+larch=yhetil.org@gnu.org X-Migadu-Country: US X-Migadu-Flow: FLOW_IN X-Migadu-Spam-Score: -6.40 X-Spam-Score: -6.40 X-Migadu-Queue-Id: C1C1A63DAE X-Migadu-Scanner: mx2.migadu.com X-TUID: 25q7Z9uWhyaG --=-=-= Content-Type: text/plain; charset=utf-8 Content-Transfer-Encoding: quoted-printable [=E0=AE=A4=E0=AE=BF=E0=AE=99=E0=AF=8D=E0=AE=95=E0=AE=B3=E0=AF=8D =E0=AE=85= =E0=AE=95=E0=AF=8D=E0=AE=9F=E0=AF=8B=E0=AE=AA=E0=AE=B0=E0=AF=8D 09, 2023] I= hor Radchenko wrote: > Thanks! > >> +(defun org--dnd-attach-file (url action) >> ... >> + (insert >> + (org-link-make-string >> + (concat (if separatep >> + "file:" >> + "attachment:") >> + (if separatep >> + (expand-file-name (file-name-nondirectory filename) >> + org-yank-image-save-type) >> + (file-name-nondirectory filename)))) >> + "\n") >> + 'private)) > >> + (pcase org--dnd-xds-method >> + (`attach (insert (org-link-make-string >> + (concat "attachment:" (file-name-nondirectory f= ilename))) > >> + (`file-link (insert (org-link-make-string (concat "file:" filenam= e)) >> + "\n")) > > Is there any particular reason why you insert a newline after the image > link? I think this can stay here until Po Lu writes the platform-agnostic code that you requested further down this thread? > Also, we might want to add a subsection describing the new customization = to > 17 Miscellaneous section of the manual. Otherwise, users might have > difficulties discovering relevant settings to customize dnd and yank > behavior. I have now added these to the manual with a few relevant concept indices. Please check if they are understandable as I am terrible at writing documentation. Thanks. --=-=-= Content-Type: text/x-diff Content-Disposition: attachment; filename=0001-Add-support-for-yank-media-and-DND.patch >From e8b142db0d91e305cbefbb2c945fee7c40ed03f3 Mon Sep 17 00:00:00 2001 From: Visuwesh Date: Fri, 22 Sep 2023 20:11:41 +0530 Subject: [PATCH] Add support for yank-media and DND * lisp/org.el (org-mode): Call the setup function for yank-media and DND. (org-setup-yank-dnd-handlers): Register yank-media-handler and DND handler. (org-yank-image-save-type, org-yank-image-file-name-function) (org-dnd-default-attach-method, org-dnd-method): New defcustoms. (org--image-yank-media-handler, org--copied-files-yank-media-handler) (org--dnd-attach-file, org--dnd-local-file-handler, org--dnd-xds-method) (org--dnd-xds-function, org--dnd-rmc): Add yank-media and DND handlers. * doc/org-manual.org: (Drag and Drop & ~yank-media~): Describe the new feature in the manual. * etc/ORG-NEWS: Advertise the new features. --- doc/org-manual.org | 40 ++++++++ etc/ORG-NEWS | 20 ++++ lisp/org.el | 243 ++++++++++++++++++++++++++++++++++++++++++++- 3 files changed, 298 insertions(+), 5 deletions(-) diff --git a/doc/org-manual.org b/doc/org-manual.org index 3e9d42f55..dcdaa98fa 100644 --- a/doc/org-manual.org +++ b/doc/org-manual.org @@ -21095,6 +21095,46 @@ most recent since the mobile application searches files that were last pulled. To get an updated agenda view with changes since the last pull, pull again. +** Drag and Drop & ~yank-media~ +:PROPERTIES: +:DESCRIPTION: Dropping and pasting files and images +:END: + +#+cindex: dropping files +#+vindex: org-dnd-method +Org can attach, insert =file:= links, or open files dropped onto an +Emacs frame. By default, Org asks the user what must be done with the +dropped file but this can be changed by customizing the user option +~org-dnd-method~. Changing the variable can make Org do any of the +above mentioned actions without prompting. + +#+vindex: org-dnd-default-attach-method +If Org cannot figure out which attachment method to use automatically, +it defaults to using the attachment method mentioned in +~org-dnd-default-attach-method~. Org uses the attachment method +mentioned in ~org-attach-method~ by default, so Org using the =cp= +method to attach the dropped files. + +#+cindex: pasting files, images from clipboard +From Emacs 29, Org can deal with images copied to the clipboard, and +files copied from a file manager when pasted using the command +~yank-media~. Org deals with them differently. + +#+vindex: org-yank-image-save-type +#+vindex: org-yank-image-file-name-function +For images, Org attaches the image when pasting but this can be +changed by customizing ~org-yank-image-save-type~ to save them under +another directory instead. The name given to these images are +autogenerated by default but you can customize what name should be +given to the pasted image by customizing +~org-yank-image-file-name-function~. This function takes no argument +and should return the filename without extension to use as the image's +filename. + +For files copied from a file manager, Org attaches them to the current +heading using the =mv= or the =cp= method if the files were cut or +copied in the file manager respectively. + * Hacking :PROPERTIES: :DESCRIPTION: How to hack your way around. diff --git a/etc/ORG-NEWS b/etc/ORG-NEWS index 252c5a9f9..c4a58dd4d 100644 --- a/etc/ORG-NEWS +++ b/etc/ORG-NEWS @@ -596,6 +596,26 @@ return a matplotlib Figure object to plot. For output results, the current figure (as returned by =pyplot.gcf()=) is cleared before evaluation, and then plotted afterwards. +*** Images and files in clipboard can be attached + +Org can now attach images in clipboard and files copied/cut to the +clipboard from file managers using the ~yank-media~ command which also +inserts a link to the attached file. This command was added in Emacs 29. + +Images can be saved to a separate directory instead of being attached, +customize ~org-yank-image-save-type~. + +Image filename chosen can be customized by setting +~org-yank-image-file-name-function~ which by default autogenerates a +filename based on the current time. + +*** Files and images can be attached by dropping onto Emacs + +Attachment method other than ~org-attach-method~ for dropped files can +be specified using ~org-dnd-default-attach-method~. + +Images dropped also respect the value of ~org-yank-image-save-type~. + ** New functions and changes in function arguments *** =TYPES= argument in ~org-element-lineage~ can now be a symbol diff --git a/lisp/org.el b/lisp/org.el index d0b2355ea..855897e7d 100644 --- a/lisp/org.el +++ b/lisp/org.el @@ -4999,7 +4999,10 @@ The following commands are available: (org--set-faces-extend '(org-block-begin-line org-block-end-line) org-fontify-whole-block-delimiter-line) (org--set-faces-extend org-level-faces org-fontify-whole-heading-line) - (setq-local org-mode-loading nil)) + (setq-local org-mode-loading nil) + + ;; `yank-media' handler and DND support. + (org-setup-yank-dnd-handlers)) ;; Update `customize-package-emacs-version-alist' (add-to-list 'customize-package-emacs-version-alist @@ -15125,20 +15128,20 @@ INCREMENT-STEP divisor." (setq hour (mod hour 24)) (setq pos-match-group 1 new (format "-%02d:%02d" hour minute))) - + ((org-pos-in-match-range pos 6) ;; POS on "dmwy" repeater char. (setq pos-match-group 6 new (car (rassoc (+ nincrements (cdr (assoc (match-string 6 ts-string) idx))) idx)))) - + ((org-pos-in-match-range pos 5) ;; POS on X in "Xd" repeater. (setq pos-match-group 5 ;; Never drop below X=1. new (format "%d" (max 1 (+ nincrements (string-to-number (match-string 5 ts-string))))))) - + ((org-pos-in-match-range pos 9) ;; POS on "dmwy" repeater in warning interval. (setq pos-match-group 9 new (car (rassoc (+ nincrements (cdr (assoc (match-string 9 ts-string) idx))) idx)))) - + ((org-pos-in-match-range pos 8) ;; POS on X in "Xd" in warning interval. (setq pos-match-group 8 ;; Never drop below X=0. @@ -20254,6 +20257,236 @@ it has a `diary' type." (org-format-timestamp timestamp fmt t)) (org-format-timestamp timestamp fmt (eq boundary 'end))))))) +;;; Yank media handler and DND +(defun org-setup-yank-dnd-handlers () + "Setup the `yank-media' and DND handlers for buffer." + (setq-local dnd-protocol-alist + (cons '("^file:///" . org--dnd-local-file-handler) + dnd-protocol-alist)) + (when (fboundp 'yank-media-handler) + (yank-media-handler "image/.*" #'org--image-yank-media-handler) + ;; Looks like different DEs go for different handler names, + ;; https://larsee.com/blog/2019/05/clipboard-files/. + (yank-media-handler "x/special-\\(?:gnome\|KDE\|mate\\)-files" + #'org--copied-files-yank-media-handler)) + (when (boundp 'x-dnd-direct-save-function) + (setq-local x-dnd-direct-save-function #'org--dnd-xds-function))) + +(defcustom org-yank-image-save-type 'attach + "Method to save images yanked from clipboard and dropped to Emacs. +It can be the symbol `attach' to add it as an attachment, or a +directory name to copy/cut the image to that directory." + :group 'org + :package-version '(Org . "9.7") + :type '(choice (const :tag "Add it as attachment" attach) + (directory :tag "Save it in directory")) + :safe (lambda (x) (eq x 'attach))) + +(defcustom org-yank-image-file-name-function #'org-yank-image-autogen-filename + "Function to generate filename for image yanked from clipboard. +By default, this autogenerates a filename based on the current +time. +It is called with no arguments and should return a string without +any extension which is used as the filename." + :group 'org + :package-version '(Org . "9.7") + :type '(radio (function-item :doc "Autogenerate filename" + org-yank-image-autogen-filename) + (function-item :doc "Ask for filename" + org-yank-image-read-filename) + function)) + +(defun org-yank-image-autogen-filename () + "Autogenerate filename for image in clipboard." + (format-time-string "clipboard-%Y%m%dT%H%M%S.%6N")) + +(defun org-yank-image-read-filename () + "Read filename for image in clipboard." + (read-string "Basename for image file without extension: ")) + +(declare-function org-attach-attach "org-attach" (file &optional visit-dir method)) + +(defun org--image-yank-media-handler (mimetype data) + "Save image DATA of mime-type MIMETYPE and insert link at point. +It is saved as per `org-yank-image-save-type'. The name for the +image is prompted and the extension is automatically added to the +end." + (let* ((ext (symbol-name (mailcap-mime-type-to-extension mimetype))) + (iname (funcall org-yank-image-file-name-function)) + (filename (file-name-with-extension iname ext)) + (absname (expand-file-name + filename + (if (eq org-yank-image-save-type 'attach) + temporary-file-directory + org-yank-image-save-type))) + link) + (when (and (not (eq org-yank-image-save-type 'attach)) + (not (file-directory-p org-yank-image-save-type))) + (make-directory org-yank-image-save-type t)) + (with-temp-file absname + (insert data)) + (if (null (eq org-yank-image-save-type 'attach)) + (setq link (org-link-make-string (concat "file:" (file-relative-name absname)))) + (require 'org-attach) + (org-attach-attach absname nil 'mv) + (setq link (org-link-make-string (concat "attachment:" filename)))) + (insert link))) + +;; I cannot find a spec for this but +;; https://indigo.re/posts/2021-12-21-clipboard-data.html and pcmanfm +;; suggests that this is the format. +(defun org--copied-files-yank-media-handler (_mimetype data) + "Attach copied or cut files from file manager. +If the files were cut from the file manager, then the `mv' attach +method is used; `cp' otherwise. + +DATA is a string where the first line is the operation to +perform: copy or cut. Rest of the lines are file: links to the +concerned files." + (require 'org-attach) + ;; pcmanfm adds a null byte at the end for some reason. + (let* ((data (split-string data "[\0\n\r]" t "^file://")) + (files (cdr data)) + (method (if (equal (car data) "cut") + 'mv + 'cp))) + (dolist (f files) + (setq f (url-unhex-string f)) + (if (file-readable-p f) + (org-attach-attach f nil method) + (message "File `%s' is not readable, skipping" f))))) + +(defcustom org-dnd-method 'ask + "Action to perform on the dropped file. +When the value is the symbol, + . `attach' -- attach the dropped file + . `open' -- visit/open the dropped file in Emacs + . `file-link' -- insert file: link to the dropped file + . `ask' -- ask what to do out of the above." + :group 'org + :package-version '(Org . "9.7") + :type '(choice (const :tag "Attach" attach) + (const :tag "Open/Visit file" open) + (const :tag "Insert file: link" file-link) + (const :tag "Ask what to do" ask))) + +(defcustom org-dnd-default-attach-method nil + "Default attach method to use when DND action is unspecified. +This attach method is used when the DND action is `private'. +This is also used when `org-yank-image-save-type' is nil. +When nil, use `org-attach-method'." + :group 'org + :package-version '(Org . "9.7") + :type '(choice (const :tag "Default attach method" nil) + (const :tag "Copy" cp) + (const :tag "Move" mv) + (const :tag "Hard link" ln) + (const :tag "Symbolic link" lns))) + +(declare-function mailcap-file-name-to-mime-type "mailcap" (file-name)) +(defvar org-attach-method) + +(defun org--dnd-rmc (prompt choices) + (if (null (use-dialog-box-p)) + (caddr (read-multiple-choice prompt choices)) + (setq choices + (mapcar + (pcase-lambda (`(_key ,message ,val)) + (cons (capitalize message) val)) + choices)) + (x-popup-menu t (list prompt (cons "" choices))))) + +(defun org--dnd-local-file-handler (url action) + "Handle file URL as per ACTION." + (let ((method (if (eq org-dnd-method 'ask) + (org--dnd-rmc + "What to do with dropped file?" + '((?a "attach" attach) + (?o "open" open) + (?f "insert file: link" file-link))) + org-dnd-method))) + (pcase method + (`attach (org--dnd-attach-file url action)) + (`open (dnd-open-local-file url action)) + (`file-link + (let ((filename (dnd-get-local-file-name url))) + (insert (org-link-make-string (concat "file:" filename)))))))) + +(defun org--dnd-attach-file (url action) + "Attach filename given by URL using method pertaining to ACTION. +If ACTION is `move', use `mv' attach method. +If `copy', use `cp' attach method. +If `ask', ask the user. +If `private', use the method denoted in `org-dnd-default-attach-action'. +The action `private' is always returned." + (require 'mailcap) + (let* ((filename (dnd-get-local-file-name url)) + (mimetype (mailcap-file-name-to-mime-type filename)) + (separatep (and (string-prefix-p "image/" mimetype) + (not (eq 'attach org-yank-image-save-type)))) + (method (pcase action + ('copy 'cp) + ('move 'mv) + ('ask (org--dnd-rmc + "Attach using method" + '((?c "copy" cp) + (?m "move" mv) + (?l "hard link" ln) + (?s "symbolic link" lns)))) + ('private (or org-dnd-default-attach-method + org-attach-method))))) + (if separatep + (funcall + (pcase method + ('cp #'copy-file) + ('mv #'rename-file) + ('ln #'add-name-to-file) + ('lns #'make-symbolic-link)) + filename + (expand-file-name (file-name-nondirectory filename) + org-yank-image-save-type)) + (org-attach-attach filename nil method)) + (insert + (org-link-make-string + (concat (if separatep + "file:" + "attachment:") + (if separatep + (expand-file-name (file-name-nondirectory filename) + org-yank-image-save-type) + (file-name-nondirectory filename)))) + "\n") + 'private)) + +(defvar-local org--dnd-xds-method nil + "The method to use for dropped file.") +(defun org--dnd-xds-function (need-name filename) + "Handle file with FILENAME dropped via XDS protocol. +When NEED-NAME is t, FILNAME is the base name of the file to be +saved. +When NEED-NAME is nil, the drop is complete." + (if need-name + (let ((method (if (eq org-dnd-method 'ask) + (org--dnd-rmc + "What to do with dropped file?" + '((?a "attach" attach) + (?o "open" open) + (?f "insert file: link" file-link))) + org-dnd-method))) + (setq-local org--dnd-xds-method method) + (pcase method + (`attach (expand-file-name filename (org-attach-dir 'create))) + (`open (expand-file-name (make-temp-name "emacs.") temporary-file-directory)) + (`file-link (read-file-name "Write file to: " nil nil nil filename)))) + (pcase org--dnd-xds-method + (`attach (insert (org-link-make-string + (concat "attachment:" (file-name-nondirectory filename))) + "\n")) + (`file-link (insert (org-link-make-string (concat "file:" filename)) + "\n")) + (`open (find-file filename))) + (setq-local org--dnd-xds-method nil))) + ;;; Other stuff (defvar reftex-docstruct-symbol) -- 2.42.0 --=-=-=--