根据要求,我提供了将 URI 写入剪贴板和从剪贴板获取 URI 的示例。这些示例基本上是立即获取/设置剪贴板的命令行程序。在实际的 GUI 应用程序中,您可能会对按下按钮或捕捉 CtrlC / CtrlV 做出反应事件,使用Gtk.Widget.add_events() 并在处理Gtk.Widget.event 信号时获取/设置剪贴板。
获取剪贴板
您可以使用Gtk.Clipboard.request_uris () 从 X11 剪贴板请求 URI。此函数接受一个回调,一旦 URI 可用,就会调用该回调。
例子:
public void main (string[] args) {
Gtk.init (ref args);
Gdk.Display display = Gdk.Display.get_default ();
Gtk.Clipboard clipboard = Gtk.Clipboard.get_for_display (display, Gdk.SELECTION_CLIPBOARD);
clipboard.request_uris (recieved_func);
Gtk.main ();
}
/* Gtk.ClipboardURIRecievedFunc */
private void recieved_func (Gtk.Clipboard clipboard, string[] uris) {
foreach (var uri in uris) {
print (uri + "\n");
}
Gtk.main_quit ();
}
用valac clipget.vala --pkg=gtk+-3.0编译
设置剪贴板
理论:
来自Qt4 documentation:
由于没有标准的方法来复制和粘贴文件
X11 上的应用程序,目前各种 MIME 类型和约定
正在使用。例如,Nautilus 期望文件提供一个
x-special/gnome-copied-files MIME 类型,数据以
剪切/复制操作、换行符和文件的 URL。
Gtk.Clipboard 没有预先设置剪贴板来复制/剪切文件。正如你所说,没有这样的Gtk.Clipboard.set_uris()。
相反,您应该通过提供回调来设置剪贴板,让 X11 从请求中获取剪贴板内容。
这些是所需的步骤:
创建一组Gtk.TargetEntrys,指定您的应用程序可以处理哪些剪贴板协议。您需要处理协议text/uri-list、x-special/gnome-copied-files 和UTF8_STRING。每个TargetEntry 都由其info 字段标识,因此该数字应该是唯一的(请参阅下面示例中的enum ClipboardProtocol)
实现Gtk.ClipboardGetFunc 类型的方法。此方法应填充与要复制/剪切的文件路径一起传递的Gtk.SelectionData 对象。检查info 参数以根据指定的协议设置SelectionData 参数。
使用Gtk.Clipboard.set_with_owner或Gtk.Clipboard.set_with_data注册回调和X11实现的协议
示例:
enum ClipboardProtocol {
TEXT_URI_LIST,
GNOME_COPIED_FILES,
UTF8_STRING
}
public void main (string[] args) {
Gtk.init (ref args);
Gdk.Display display = Gdk.Display.get_default ();
Gtk.Clipboard clipboard = Gtk.Clipboard.get_for_display (display, Gdk.SELECTION_CLIPBOARD);
var clipboard_targets = new Gtk.TargetEntry[3];
Gtk.TargetEntry target_entry = { "text/uri-list", 0, ClipboardProtocol.TEXT_URI_LIST };
clipboard_targets[0] = target_entry;
target_entry = { "x-special/gnome-copied-files", 0, ClipboardProtocol.GNOME_COPIED_FILES };
clipboard_targets[1] = target_entry;
target_entry = { "UTF8_STRING", 0, ClipboardProtocol.UTF8_STRING };
clipboard_targets[2] = target_entry;
var owner = new Object ();
var rc = clipboard.set_with_owner (
clipboard_targets,
get_func,
clear_func,
owner
);
assert (rc);
clipboard.store ();
Gtk.main ();
}
/* Gtk.ClipboardGetFunc */
private void get_func (
Gtk.Clipboard clipboard,
Gtk.SelectionData selection_data,
uint info,
void* user_data_or_owner
) {
print ("GET FUNC!\n");
File my_file = File.new_for_path ("/home/lukas/tmp/test.txt");
File my_2nd_file = File.new_for_path ("/home/lukas/tmp/test2.txt");
File[] files = { my_file, my_2nd_file };
switch (info) {
case ClipboardProtocol.TEXT_URI_LIST:
string[] uris = {};
foreach (var file in files) {
uris += file.get_uri ();
}
selection_data.set_uris (uris);
break;
case ClipboardProtocol.GNOME_COPIED_FILES:
var prefix = "copy\n";
//var prefix = "cut\n";
/* use one of the above */
var builder = new StringBuilder (prefix);
for (int i = 0; i < files.length; i++) {
builder.append (files[i].get_uri ());
/* dont put the newline if this is the last file */
if (i != files.length - 1)
builder.append_c ('\n');
}
selection_data.set (
selection_data.get_target (),
8,
builder.data
);
break;
case ClipboardProtocol.UTF8_STRING:
var builder = new StringBuilder ();
foreach (var file in files) {
builder.append (file.get_parse_name ());
}
builder.append_c ('\n');
selection_data.set_text (builder.str, -1);
break;
default:
assert_not_reached ();
}
Gtk.main_quit ();
}
/* Gtk.ClipboardClearFunc */
private void clear_func (Gtk.Clipboard clipboard, void* data) {
;
}
用valac clipset.vala --pkg=gtk+-3.0编译
几点说明:
来源: