【发布时间】:2013-11-04 22:21:57
【问题描述】:
我在 Python 2.7.5 中使用 wxPython 2.9.5、ObjectListView 1.2、lxml 2.3 和 SQLAlchemy 0.8.2 编写并使用 py2exe 编译成 exe 时遇到了困难。
我遇到的问题是在将程序编译为 exe 后,我不再能够使用 Ctrl-C 从 ObjectListView 复制数据并将其粘贴到其他程序中,例如 Excel、记事本甚至 Notepad++。无论选择要复制/粘贴多少行,都会发生这种情况。粘贴到 Excel 时得到的结果是“Microsoft Excel 无法粘贴数据”。当我尝试记事本时,我得到了第一列和第一行的第一个字母(我认为它是第一行),但没有别的。
这是我遇到问题的输出窗口之一的代码 sn-p(总共有 2 个,但它们几乎是彼此的完美镜像)。这由主脚本导入,然后作为页面添加到 agw AUI Notebook。
import wx
from ObjectListView import ObjectListView, ColumnDefn
from wx.lib.pubsub import pub
import ectworker as EW
TabLabel = 'Primary Output'
outputOne = []
class TabPanel(wx.Panel):
def __init__(self, parent):
wx.Panel.__init__(self, parent)
headerFont = wx.Font(12, wx.MODERN, wx.SLANT, wx.NORMAL)
title = wx.StaticText(self, -1, " Primary Output:")
title.SetFont(headerFont)
self.countDisplay = wx.StaticText(self, -1, '%s'%len(invOutput))
self.countDisplay.SetFont(headerFont)
self.dataOLV = ObjectListView(self, wx.ID_ANY,
style=wx.LC_REPORT|wx.SUNKEN_BORDER)
self.setColumns()
self.dataOLV.SetEmptyListMsg('Please click the show button.\n\
If nothing appears then there is nothing to display')
self.dataOLV.cellEditMode = ObjectListView.CELLEDIT_NONE
updateBtn = wx.Button(self, wx.ID_ANY, 'Show')
updateBtn.Bind(wx.EVT_BUTTON, self.updateControl)
filterBtn = wx.Button(self, label='Filter')
filterBtn.Bind(wx.EVT_BUTTON, self.filterInfo)
lookupBtn = wx.Button(self, wx.ID_ANY, 'Look Up')
lookupBtn.Bind(wx.EVT_BUTTON, self.onLookUp)
vbox = wx.BoxSizer(wx.VERTICAL)
hbox0 = wx.BoxSizer(wx.HORIZONTAL)
hbox0.Add(title, 0, wx.ALL, 5)
hbox0.Add(self.countDisplay, 0, wx.ALL, 5)
hbox1 = wx.BoxSizer(wx.HORIZONTAL)
hbox1.Add(updateBtn, 0, wx.ALL, 5)
hbox1.Add(filterBtn, 0, wx.ALL, 5)
hbox1.Add(lookupBtn, 0, wx.ALL, 5)
vbox.Add(hbox0, 0, wx.EXPAND, 5)
vbox.Add(self.dataOLV, 1, wx.ALL|wx.EXPAND, 5)
vbox.Add(hbox1, 0, wx.ALL|wx.CENTER, 5)
self.SetSizer(vbox)
def setColumns(self, data=None):
self.idColumn = ColumnDefn('Dealer ID', 'left', -1, 'id')
self.nameColumn = ColumnDefn('Dealer Name', 'left', -1, 'name')
self.thirdColumn = ColumnDefn('Third', 'left', -1, 'third')
self.fourthColumn = ColumnDefn('Fourth', 'left', -1, 'fourth')
self.fifthColumn = ColumnDefn('Fifth', 'left', -1, 'fifth')
self.sixthColumn = ColumnDefn('Sixth', 'right', -1, 'sixth')
self.seventhColumn = ColumnDefn('Seventh', 'right', -1, 'seventh')
self.olvColumns = [
self.idColumn,
self.nameColumn,
self.thirdColumn,
self.fourthColumn,
self.fifthColumn,
self.sixthColumn,
self.seventhColumn]
self.dataOLV.SetColumns(self.olvColumns)
def updateControl(self, event):
outputOne = EW.filterOne()
self.updateOLV(outputOne)
def filterInfo(self, event):
pub.sendMessage('dialog.filter', event=None)
while EW.dataVars.theFilter == []:
pub.sendMessage('dialog.filter', event=None)
outputOne = EW.filterOne()
self.updateOLV(outputOne)
def updateOLV(self, object):
self.dataOLV.SetObjects(object)
self.countDisplay.SetLabel('%s'%len(object))
def onLookUp(self, event):
# This needs proper error handling
selectedRow = self.dataOLV.GetSelectedObject()
try:
webbrowser.open('sc/%s' % selectedRow.id)
except AttributeError:
pass
if __name__ == '__main__':
print('File file is not meant to be run as a stand alone application.')
print(' Please try again by running the main program.')
我的主要问题是如何在使用 py2exe 编译后让键盘快捷键 Ctrl+C 正常工作?
在我收到“尝试 PyInstaller”的回复之前,我每次都会收到一条错误消息,但这是另一个可能的问题发布。
提前致谢。
--编辑--
我认为这段代码是我的解决方案,但我的最终结果是外语混搭。
def onCtrlC(self, event):
selectedRows = self.dataOLV.GetSelectedObjects()
self.dataObj = wx.TextDataObject()
self.dataObj.SetData('%s'%str(selectedRows))
if wx.TheClipboard.Open():
wx.TheClipboard.SetData(self.dataObj)
wx.TheClipboard.Flush()
else:
pub.sendMessage('dialog.error', message='Unable to open the clipboard',
caption='Error')
我在做一个有根据的猜测,我错过了一些东西。
--EDIT2--
感谢 Mike Driscoll,让它按预期工作,谢谢! 我确实改变了将数据放入剪贴板的方式,我认为它不是最有效的,但它确实有效。我还发现了为什么我在粘贴时收到一团乱麻。代码如下,我正在使用多行 TextCtrl 来临时存储数据并在完成后将其清除。
self.hiddenTxt = wx.TextCtrl(self, style=wx.TE_MULTILINE)
self.hiddenTxt.Hide()
def onCtrlC(self, event):
selectedRows = self.dataOLV.GetSelectedObjects()
for x in selectedRows:
self.hiddenTxt.AppendText('%s\t%s\n'%(x.id, x.name))
self.dataObj = wx.TextDataObject()
self.dataObj.SetText(self.hiddenTxt.GetValue())
if wx.TheClipboard.Open():
wx.TheClipboard.SetData(self.dataObj)
wx.TheClipboard.Flush()
else:
pub.sendMessage('dialog.error', message='Unable to open the clipboard',
caption='Error')
self.hiddenTxt.SetValue('')
我搞砸的部分是:
self.dataObj.SetData
实际上应该是:
self.dataObj.SetText
通过在每一行的花絮之间使用 \t,我可以毫无问题地粘贴到 Excel 中!
非常感谢 Mike D。
【问题讨论】:
-
我会将数据对象更改为逗号分隔列表或带有逗号的字符串以伪造逗号分隔列表。然后将列表或字符串添加到剪贴板
-
@MikeD 谢谢,我确实尝试过逗号分隔,但我仍然遇到粘贴到 Excel 的问题,但我能够使用 \t 粘贴到 Excel 中没有问题。我还注意到,在将文本添加到 dataObj 时,我使用的是 SetData 而不是 SetText,这似乎是我的多语言输出的原因。
-
我很高兴你弄明白了。
标签: wxpython py2exe objectlistview