Python GTK拖放 - 获取URL

6

我正在创建一个小应用程序,必须能够接收URL。如果应用程序窗口已打开,则应该能够从浏览器拖动链接并将其放入应用程序中 - 应用程序将URL保存到数据库中。

我正在使用Python/GTk创建此应用程序。但是我对其中的拖放功能有些困惑。那么,如何实现呢?

以下是一些示例代码来实现拖放(我的应用程序使用了一些这些代码)...

import pygtk
pygtk.require('2.0')
import gtk

# function to print out the mime type of the drop item
def drop_cb(wid, context, x, y, time):
    l.set_text('\n'.join([str(t) for t in context.targets]))
    # What should I put here to get the URL of the link?

    context.finish(True, False, time)
    return True

# Create a GTK window and Label, and hook up
# drag n drop signal handlers to the window
w = gtk.Window()
w.set_size_request(200, 150)
w.drag_dest_set(0, [], 0)
w.connect('drag_drop', drop_cb)
w.connect('destroy', lambda w: gtk.main_quit())
l = gtk.Label()
w.add(l)
w.show_all()

# Start the program
gtk.main()
4个回答

8

您必须自己获取数据。下面是一个简单的工作示例,它将把一个标签设置为拖放的url:

#!/usr/local/env python

import pygtk
pygtk.require('2.0')
import gtk

def motion_cb(wid, context, x, y, time):
    l.set_text('\n'.join([str(t) for t in context.targets]))
    context.drag_status(gtk.gdk.ACTION_COPY, time)
    # Returning True which means "I accept this data".
    return True

def drop_cb(wid, context, x, y, time):
    # Some data was dropped, get the data
    wid.drag_get_data(context, context.targets[-1], time)
    return True

def got_data_cb(wid, context, x, y, data, info, time):
    # Got data.
    l.set_text(data.get_text())
    context.finish(True, False, time)

w = gtk.Window()
w.set_size_request(200, 150)
w.drag_dest_set(0, [], 0)
w.connect('drag_motion', motion_cb)
w.connect('drag_drop', drop_cb)
w.connect('drag_data_received', got_data_cb)
w.connect('destroy', lambda w: gtk.main_quit())
l = gtk.Label()
w.add(l)
w.show_all()

gtk.main()

8
要小心,如果数据以URI列表的形式呈现,则可能需要调用data.get_uris()。例如,如果您从Konqueror/Nautilus拖放文件列表到窗口,并且接受“text/uri-list”,则在GtkSelectionData上调用get_data()将返回None。 - Jason Mock
如果我们在拖动保留文本的标签时,将其拖到窗口外并释放鼠标,可能会出现不良行为。清除标签似乎更有意义。 - demongolem

3

如果您希望在从文件资源管理器中拖放文件列表时只获取一个文件或目录的数据,可以使用类似以下内容的方法:

data.get_text().split(None,1)[0]

“got_data_cb”方法的代码应该是这样的:
def got_data_cb(wid, context, x, y, data, info, time):
    # Got data.
    l.set_text(data.get_text().split(None,1)[0])
    context.finish(True, False, time)

这将通过任何空格拆分数据,并返回第一个项目。

2
以下代码是从(旧版)PyGTK教程的一个示例 移植 来的,我猜这也启发了被采纳的答案,但使用了pygi。
#!/usr/local/env python
import gi
gi.require_version("Gtk", "3.0")
from gi.repository import Gtk, Gdk

def motion_cb(wid, context, x, y, time):
    Gdk.drag_status(context, Gdk.DragAction.COPY, time)
    return True

def drop_cb(wid, context, x, y, time):
    l.set_text('\n'.join([str(t) for t in context.list_targets()]))
    context.finish(True, False, time)
    return True

w = Gtk.Window()
w.set_size_request(200, 150)
w.drag_dest_set(0, [], 0)
w.connect('drag-motion', motion_cb)
w.connect('drag-drop', drop_cb)
w.connect('destroy', lambda w: Gtk.main_quit())
l = Gtk.Label()
w.add(l)
w.show_all()

Gtk.main()

1
我唯一有效的解决方案是:


def got_data_cb(wid, context, x, y, data, info, time):
    # Got data.
    l.set_text(data.get_uris()[0])
    context.finish(True, False, time)

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接