如何使用 python 或 CL 将文件复制到剪贴板,然后使用 STRG+V 粘贴它?

Nat*_*tan 6 python linux clipboard ubuntu xfce

我正在尝试将文件复制(使用 python 或 CL 命令,然后可以使用 python 调用)到剪贴板,以便稍后使用 STRG+V 粘贴它。据我了解,文件不会“移动”到剪贴板中,而是剪贴板保存路径和告诉操作系统“这是一个文件”的参数/标志。我对 Linux 特定的答案很满意,但通用的答案将是最重要的。

pyperclip

不是一个解决方案,因为它不允许复制文件,只能复制字符串。

剪辑

不是一个解决方案,因为它只复制文本

xclip-复制文件

不是一个解决方案,因为它只复制到X剪贴板,而不是剪贴板。虽然 xclip 提供了该选项-selection clipboard(但仅复制文本),xclip-copyfile但没有这样的选项。

使用查找

find ${PWD} -name "*.pdf"| xclip -i -selection clipboard -t text/uri-list

是此处描述的命令: https: //askubuntu.com/questions/210413/what-is-the-command-line-equivalent-of-copying-a-file-to-clipboard#answer-210428

但我无法用它复制文件,因此假设它不适用于所有文件。

Ste*_*cht 6

配置

剪贴板是窗口管理的一部分,而不是 Linux 操作系统本身的一部分。具有不同分布的不同配置表现不同,因此需要不同的变体。与此同时,Wayland 正逐渐走上逐步取代 X 的道路,这意味着需要考虑三种配置:

  • 仅限韦兰
  • Wayland 与 XWayland 一起(兼容非适配的 X 软件)
  • X

发送剪贴板内容

当保存到剪贴板时,系统首先仅通知接收者数据可用于剪贴板。仅根据请求发送实际数据。因此,在数据传输之前,不得终止将内容发送到剪贴板的程序。根据环境/配置,程序终止后剪贴板的内容也可能被删除。

xclip那么问题中已经提到的程序如何工作呢?似乎被调用后立即终止。但仔细检查后发现并非如此,因为它执行了分叉,因此它仍然存在于后台(通过查看源代码或命令可以轻松验证ps)。

格式

此外,不同的环境以不同的方式要求内容。x-special/gnome-copied-files例如,GNOME 需要使用特殊目标和特殊格式的内容来复制文件列表,例如copy\nfile:///etc/group,GNOME 文件管理器 Nautilus 才能正确执行复制操作。

另一方面,在 KDE 下,只有一个带有 target 的 URI 列表text/uri-list

确定环境

以下示例程序适用于 Linuxmint 20.2 Cinnamon、带有 Gnome 的 Ubuntu 22.04 和带有 KDE 的 Kubuntu 22.04。其他发行版/配置可能需要一些定制。在这里,建议简单地在适当的文件管理器中复制文件,然后使用程序查看剪贴板内容,然后对脚本进行适当的调整。

根据环境变量XDG_CURRENT_DESKTOPWAYLAND_DISPLAY以下程序尝试确定环境。

如果是 Wayland,wl-copy则使用,否则xclip使用。目标和内容格式会相应调整。该工具启动后subprocess.Popen,内容将发送到stdin该工具。

一旦完成,程序就会退出。然后创建wl-copy一个xclip分叉,确保数据存在于剪贴板中。

import os
import subprocess
import sys
from pathlib import Path

gnome_desktops = ['X-Cinnamon', 'XFCE']


def is_gnome(desktop):
    if desktop.endswith("GNOME") or desktop in gnome_desktops:
        return True
    return False


def target():
    current_desktop = os.environ['XDG_CURRENT_DESKTOP']
    if is_gnome(current_desktop):
        return 'x-special/gnome-copied-files'
    elif current_desktop == 'KDE':
        return 'text/uri-list'
    else:
        sys.exit(f'unsupported desktop {current_desktop}')


def base_copy_cmd():
    if 'WAYLAND_DISPLAY' in os.environ:
        return 'wl-copy'
    return 'xclip -i -selection clipboard'


def copy_clipboard_cmd():
    return f"{base_copy_cmd()} -t '{target()}'"


def content(files_to_copy):
    uris = '\n'.join([Path(f).as_uri() for f in files_to_copy])
    current_desktop = os.environ['XDG_CURRENT_DESKTOP']
    if is_gnome(current_desktop):
        return f"copy\n{uris}".encode("utf-8")
    return uris.encode("utf-8")


def copy_to_clipboard(files_to_copy):
    copy_process = subprocess.Popen(copy_clipboard_cmd(), shell=True, stdin=subprocess.PIPE)
    copy_process.stdin.write(content(files_to_copy))
    copy_process.stdin.close()
    copy_process.wait()


if __name__ == '__main__':
    files = ['/etc/hosts', '/etc/group']
    copy_to_clipboard(files)
Run Code Online (Sandbox Code Playgroud)

如上所述,对于其他环境,只需在本机文件管理器中复制文件,然后检查当前剪贴板内容并对脚本进行适当的调整。

根据环境,xclip或者wl-copy(使用包管理器安装包wl-clipboard)必须在那里。详细信息wl-copy可以在这里找到: https: //github.com/bugaevc/wl-clipboard

检查剪贴板

最后,为了能够转储剪贴板的当前内容,这里有一个小脚本可以做到这一点。因此可以看到其他程序(例如本机文件管理器)将哪些内容放入剪贴板。通常许多程序将同一数据的多个不同表示目标放入剪贴板。

import gi

gi.require_version("Gtk", "3.0")
from gi.repository import Gtk, Gdk


def on_activate(app):
    win = Gtk.ApplicationWindow(application=app)
    win.set_title("GTK Clipboard Util")
    win.set_default_size(256, 192)
    btn = Gtk.Button(label="Dump Clipboard")
    btn.connect('clicked', dump)
    box = Gtk.VBox()
    win.add(box)
    box.add(btn)
    win.show_all()


def dump(button):
    cb_targets = []
    counter = 0

    def print_content(clipboard, data):
        print(data.get_data())
        print()
        print_next_target_and_content(clipboard)

    def print_next_target_and_content(clipboard):
        nonlocal counter
        if counter < len(cb_targets):
            target = cb_targets[counter]
            print(target)
            clipboard.request_contents(target, print_content)
            counter += 1

    def get_targets(clipboard, targets, n_targets):
        nonlocal counter
        nonlocal cb_targets
        counter = 0
        cb_targets = targets
        print_next_target_and_content(clipboard)

    gtk_clipboard = Gtk.Clipboard.get(Gdk.SELECTION_CLIPBOARD)
    gtk_clipboard.request_targets(get_targets)


if __name__ == '__main__':
    app = Gtk.Application(application_id='com.software7.clipboard.formats')
    app.connect('activate', on_activate)
    app.run(None)
Run Code Online (Sandbox Code Playgroud)