Я подозреваю, что вы используете wx.TextDropTarget
с wx.TextDataObject
, а то, что вы передаете, явно list
.
. Вам нужно будет создать пользовательский объект данных, а затем сериализовать list
при перетаскивании.и десериализовать его по капле.
Для сериализации / десериализации списка вы можете использовать pickle
или marshal
, и я уверен, что вы можете использовать json
, но яна самом деле не смотрел на это.
Кодирование списка drag & drop
может быть довольно сложным, поэтому я собрал этот код, чтобы вы могли визуализировать, как вы можете его реализовать.
Код прокомментирован, так что, надеюсь, вы сможете увидеть, что к чему.
Это немного сложнее, чем должно быть, так как я позволил перетаскивать списки из / внеравное количество столбцов.
Также обратите внимание, что вы можете перетаскивать в тот же список.
import wx
import pickle
#import marshal
class MyTarget(wx.PyDropTarget):
def __init__(self, object):
wx.DropTarget.__init__(self)
self.object = object
# specify the type of data to accept
self.data = wx.CustomDataObject("ListCtrlItems")
self.SetDataObject(self.data)
# Called when OnDrop returns True.
def OnData(self, x, y, opt):
# Find insertion point in the target.
index, flags = self.object.HitTest((x, y))
if self.GetData():
# unpickle data
listdata = self.data.GetData()
dropped_list = pickle.loads(listdata)
#dropped_list = marshal.loads(listdata)
if index == -1: # if not inserting, set index to the end of the listctrl
index = self.object.GetItemCount()
#Insert at drop point
for row in dropped_list:
self.object.InsertItem(index, row[0]) #Insert item
cols = self.object.GetColumnCount()
for pos in range(1,cols):
try:
self.object.SetItem(index, pos, row[pos]) #Add extra columns data
except Exception as e: # run out of columns in target
pass
index +=1
return True
class Mywin(wx.Frame):
def __init__(self,parent,title):
wx.Frame.__init__(self, parent, wx.ID_ANY, title,size= (600,-1))
panel = wx.Panel(self)
box = wx.BoxSizer(wx.HORIZONTAL)
self.listCtrl1 = wx.ListCtrl(panel, -1, style = wx.LC_REPORT|wx.LC_HRULES)
self.listCtrl1.InsertColumn(0, "Item0")
self.listCtrl1.SetColumnWidth(0,100)
self.listCtrl1.InsertColumn(1, "Item1")
self.listCtrl1.SetColumnWidth(1,100)
self.listCtrl2 = wx.ListCtrl(panel, -1, style = wx.LC_REPORT|wx.LC_VRULES|wx.LC_HRULES)
self.listCtrl2.InsertColumn(0, "Item0")
self.listCtrl2.SetColumnWidth(0,100)
self.listCtrl2.InsertColumn(1, "Item1")
self.listCtrl2.SetColumnWidth(0,100)
self.listCtrl2.InsertColumn(2, "Item2")
self.delete = wx.CheckBox(panel, wx.ID_ANY, "Delete on move")
self.delete.SetToolTip("Delete original item when dragged & dropped")
self.delete.SetValue(True)
#load sample data
data = [["abc",1],["def",2],["ghi",3]]
for i in data:
self.listCtrl1.Append((i))
data = [["ABC",1,"first"],["DEF",2,"second"],["GHI",3,"third"]]
for i in data:
self.listCtrl2.Append((i))
#Target Left
tl = MyTarget(self.listCtrl1)
self.listCtrl1.SetDropTarget(tl)
#Target Right
tr = MyTarget(self.listCtrl2)
self.listCtrl2.SetDropTarget(tr)
self.listCtrl1.Bind(wx.EVT_LIST_BEGIN_DRAG, self.OnDrag)
self.listCtrl2.Bind(wx.EVT_LIST_BEGIN_DRAG, self.OnDrag)
box.Add(self.listCtrl1, 0, wx.EXPAND)
box.Add(self.listCtrl2, 0, wx.EXPAND)
box.Add(self.delete, 0, wx.ALIGN_TOP)
panel.SetSizer(box)
panel.Fit()
self.Centre()
self.Show(True)
def OnDrag(self, event):
#create a data object for drag-and-drop
object = event.GetEventObject() # listCtrl1 or listCtrl2
list_data = []
idx = -1
while True: # find all the selected items and put them in a list
idx = object.GetNextSelected(idx)
if idx == -1:
break
item_data = []
for item in range(object.GetColumnCount()): # get data from all columns
item_data.append(object.GetItem(idx, item).GetText())
list_data.append(item_data)
# Pickle the items list.
pickle_data = pickle.dumps(list_data)
#pickle_data = marshal.dumps(list_data)
# create custom data object
cdataobj = wx.CustomDataObject("ListCtrlItems")
cdataobj.SetData(pickle_data)
# Now make a data object for the item list.
data = wx.DataObjectComposite()
data.Add(cdataobj)
# Create drop source and begin drag-and-drop.
dropSource = wx.DropSource(object)
dropSource.SetData(data)
result = dropSource.DoDragDrop(True)
# delete dropped items from source list
if self.delete.GetValue(): # Is delete checkbox ticked
if result == wx.DragCopy: # Was the drag and drop successful
while True:
#For this small sample always start at the beginning (-1)
idx = object.GetNextSelected(-1)
if idx == -1: #No more selected items
break
object.DeleteItem(idx)
demo = wx.App()
Mywin(None,'Drag & Drop ListCtrl Demo')
demo.MainLoop()
В ответ на ваш комментарий о результате, возвращаемом из DoDragDrop
:
Перечисление DragResult предоставляет следующие значения:
Описание Значение
Ошибка DragError помешала завершению операции D & D.
DragNone Цель перетаскивания не приняла данные.
DragCopy Данные были успешно скопированы.
DragMove Данные были успешно перемещены (только MSW).
DragLink Operation - это перетаскиваемая ссылка.
DragCancel Операция была отменена пользователем (не ошибка).
DoDragDrop (self, flags = Drag_CopyOnly) Запускает операцию перетаскивания, которая прекращается, когда пользователь отпускает мышь.
Вызывает это, например, в ответ на нажатие кнопки мыши.
Параметры: flags (int) - Если в флаги включен wx.Drag_AllowMove, данные можно перемещать, а не только копировать, как в случае с wx.Drag_CopyOnly по умолчанию.Если указан wx.Drag_DefaultMove (который включает в себя предыдущий флаг), перемещение не только возможно, но и становится операцией по умолчанию.Тип возвращаемого значения: wx.DragResult. Возвращает: Операция, запрошенная пользователем, может быть wx.DragCopy, wx.DragMove, wx.DragLink, wx.DragCancel или wx.DragNone, если произошла ошибка.