我正在使用 GtkSheet
PyGTK 中的小部件为我的应用程序的电子表格提供支持,它为我提供了一个 API 来从单元格中提取和推送数据。 (我查看了使用 GtkTreeView,但似乎工作量太大)
我不明白的是如何拦截粘贴请求(通过 ie.CTRL+V)以便我可以处理它们而不是将其传递给小部件。目前,从电子表格粘贴时,数据显示如下:
是否有我应该拦截的信号?
我在 Ubuntu 9.10、Python 2.6 上。
最佳答案
要捕获粘贴事件,您需要首先创建一个继承自gtksheet.ItemEntry
的自定义条目类(本例中为PastableEntry
)。在初始化期间,我们连接到 paste-clipboard
信号以捕获粘贴事件:
class PastableEntry(gtksheet.ItemEntry):
def __init__(self):
gtksheet.ItemEntry.__init__(self)
self.connect('paste-clipboard', self.__on_paste)
困难的工作是在事件处理程序中。首先我们需要获取剪贴板内容。在 Unix 中,剪贴板源可以公布多种数据格式。根据您的屏幕截图,我假设您正在尝试从 Gnumeric 复制数据。 Gnumeric 支持 application/x-gnumeric
、text/html
、UTF8_STRING
、COMPOUND_TEXT
和 STRING
。对于此示例,我们将使用 UTF8_STRING 格式,如下所示:
1,1 <tab> 1,2 <tab> 1,3 <newline>
2,1 <tab> 2,2 <tab> 2,3 <newline>
3,1 <tab> 3,2 <tab> 3,3
显然,如果任何单元格包含制表符或换行符,这将严重失败,但为了简单起见,我们将使用它。在现实世界的应用程序中,您可能希望解析 application/x-gnumeric
或 text/html
格式的数据。
回到我们的 PastatableEntry 类,现在我们定义粘贴事件处理程序:
def __on_paste(self, entry):
clip = gtk.Clipboard()
data = clip.wait_for_contents('UTF8_STRING')
text = data.get_text()
sheet = self.parent
o_row, o_col = sheet.get_active_cell()
for i_row, row in enumerate(text.split('\n')):
for i_col, cell in enumerate(row.split('\t')):
sheet.set_cell_text(o_row + i_row, o_col + i_col, cell)
self.stop_emission('paste-clipboard')
它应该是不言自明的。我们将剪贴板数据拆分为行(通过换行符),然后拆分为单元格(通过制表符),并相应地设置工作表单元格值。
stop_emission
用于阻止 GTK+ 运行粘贴操作的默认处理程序。如果没有该行,所选单元格将被原始数据覆盖。
然后我们用 GObject 注册这个类:
gobject.type_register(PastableEntry)
最后,要实际使用我们的自定义入口类,将其传递给 gtksheet.Sheet
的构造函数:
s = gtksheet.Sheet(20, 20, "Sheet 1", entry_type=PastableEntry)
关于python - 在 GTK 中通过剪贴板处理电子表格数据,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/2022594/