【问题标题】:Python and multiple select with a ListCtrlPython 和使用 ListCtrl 进行多选
【发布时间】:2013-11-02 07:17:30
【问题描述】:

我的应用程序中有一个自定义的 listctrl,我希望能够像在 ListBox 中那样选择多行(和取消选择)。目前我有一个 listctrl 可以抓取单个选择;但是,一旦我单击 listctrl 中的另一行,之前的“取消突出显示”。我希望它保持突出显示,除非我再次单击它——这样用户就知道他/她选择了哪些行(与 ListBox 的工作方式完全相同)。我尝试将wx.LC_MULTIPLE_SEL 添加到 listctrl 行,但这不起作用。

帮助?谢谢!

我从 Mike Driscoll Python Blog 的以下站点中获取了这个示例。我适应了它来抓取行选择。本质上,我选择了一行并将索引附加到名为InfoList 的列表中。当我选择一个新行时,它会正确附加,但我希望该行在实际列表中保持“突出显示”。然后,如果我在 listctrl 中再次选择它,我可以添加另一行以从列表中删除一个项目。

import wx
import wx.lib.mixins.listctrl as listmix

InfoList = []

musicdata = {
0 : ("Bad English", "The Price Of Love", "Rock"),
1 : ("DNA featuring Suzanne Vega", "Tom's Diner", "Rock"),
2 : ("George Michael", "Praying For Time", "Rock"),
3 : ("Gloria Estefan", "Here We Are", "Rock"),
4 : ("Linda Ronstadt", "Don't Know Much", "Rock"),
5 : ("Michael Bolton", "How Am I Supposed To Live Without You", "Blues"),
6 : ("Paul Young", "Oh Girl", "Rock"),
}

########################################################################
class TestListCtrl(wx.ListCtrl):

    #----------------------------------------------------------------------
    def __init__(self, parent, ID=wx.ID_ANY, pos=wx.DefaultPosition,
                 size=wx.DefaultSize, style=0):
        wx.ListCtrl.__init__(self, parent, ID, pos, size, style)

########################################################################
class TestListCtrlPanel(wx.Panel, listmix.ColumnSorterMixin):

    #----------------------------------------------------------------------
    def __init__(self, parent):
        wx.Panel.__init__(self, parent, -1, style=wx.WANTS_CHARS)

        self.list_ctrl = TestListCtrl(self, size=(-1,100),
                         style=wx.LC_REPORT
                         |wx.BORDER_SUNKEN
                         |wx.LC_SORT_ASCENDING
                         )
        self.list_ctrl.InsertColumn(0, "Artist")
        self.list_ctrl.InsertColumn(1, "Title", wx.LIST_FORMAT_RIGHT)
        self.list_ctrl.InsertColumn(2, "Genre")

        items = musicdata.items()
        index = 0
        for key, data in items:
            self.list_ctrl.InsertStringItem(index, data[0])
            self.list_ctrl.SetStringItem(index, 1, data[1])
            self.list_ctrl.SetStringItem(index, 2, data[2])
            self.list_ctrl.SetItemData(index, key)
            index += 1

        # Now that the list exists we can init the other base class,
        # see wx/lib/mixins/listctrl.py
        self.itemDataMap = musicdata
        listmix.ColumnSorterMixin.__init__(self, 3)
        self.Bind(wx.EVT_LIST_COL_CLICK, self.OnColClick, self.list_ctrl)

        ### I ADDED THIS ###
        self.Bind(wx.EVT_LIST_ITEM_SELECTED, self.OnSelectMustHave, self.list_ctrl)

        sizer = wx.BoxSizer(wx.VERTICAL)
        sizer.Add(self.list_ctrl, 0, wx.ALL|wx.EXPAND, 5)
        self.SetSizer(sizer)

    ### I ADDED THIS ###
    def OnSelectMustHave(self,event):        
        info = event.GetData()
        InfoList.append(info)
        print info,InfoList

    #----------------------------------------------------------------------
    # Used by the ColumnSorterMixin, see wx/lib/mixins/listctrl.py
    def GetListCtrl(self):
        return self.list_ctrl

    #----------------------------------------------------------------------
    def OnColClick(self, event):
        print "column clicked"
        event.Skip()

########################################################################
class MyForm(wx.Frame):

    #----------------------------------------------------------------------
    def __init__(self):
        wx.Frame.__init__(self, None, wx.ID_ANY, "List Control Tutorial")

        # Add a panel so it looks the correct on all platforms
        panel = TestListCtrlPanel(self)

#----------------------------------------------------------------------
# Run the program
if __name__ == "__main__":
    app = wx.App(False)
    frame = MyForm()
    frame.Show()
    app.MainLoop()

【问题讨论】:

  • 我在上面添加了工作代码(我的代码太长了,无法简单地添加)。
  • 如果我直接复制/粘贴你的代码(上图),我可以在 windows7 上选择多个...你在什么操作系统上?
  • Windows 8...请不要告诉我这是我不喜欢 Windows 8 的另一件事!
  • 您需要按住 control 或 shift 才能选择多个...
  • 我想...希望有另一种方法。谢谢!

标签: python select wxpython listctrl


【解决方案1】:

普通的 ListCtrl 可以工作:

import wx

########################################################################
class MyForm(wx.Frame):

    #----------------------------------------------------------------------
    def __init__(self):
        wx.Frame.__init__(self, None, wx.ID_ANY, "List Control Tutorial")

        # Add a panel so it looks the correct on all platforms
        panel = wx.Panel(self, wx.ID_ANY)
        self.index = 0

        self.list_ctrl = wx.ListCtrl(panel, size=(-1,100),
                         style=wx.LC_REPORT
                         |wx.BORDER_SUNKEN
                         )
        self.list_ctrl.InsertColumn(0, 'Subject')
        self.list_ctrl.InsertColumn(1, 'Due')
        self.list_ctrl.InsertColumn(2, 'Location', width=125)

        btn = wx.Button(panel, label="Add Line")
        btn.Bind(wx.EVT_BUTTON, self.add_line)

        sizer = wx.BoxSizer(wx.VERTICAL)
        sizer.Add(self.list_ctrl, 1, wx.ALL|wx.EXPAND, 5)
        sizer.Add(btn, 0, wx.ALL|wx.CENTER, 5)
        panel.SetSizer(sizer)

    #----------------------------------------------------------------------
    def add_line(self, event):
        line = "Line %s" % self.index
        self.list_ctrl.InsertStringItem(self.index, line)
        self.list_ctrl.SetStringItem(self.index, 1, "01/19/2010")
        self.list_ctrl.SetStringItem(self.index, 2, "USA")
        self.index += 1

#----------------------------------------------------------------------
# Run the program
if __name__ == "__main__":
    app = wx.App(False)
    frame = MyForm()
    frame.Show()
    app.MainLoop()

如果我修改它以使用您提到的两个 mixin,它也可以工作:

import wx
import wx.lib.mixins.listctrl as listmix

########################################################################
class MyListCtrl(wx.ListCtrl, listmix.TextEditMixin, listmix.ColumnSorterMixin):
    """"""

    #----------------------------------------------------------------------
    def __init__(self, parent):
        """Constructor"""
        wx.ListCtrl.__init__(self, parent, style=wx.LC_REPORT|wx.BORDER_SUNKEN)
        listmix.ColumnSorterMixin.__init__(self, 3)
        listmix.TextEditMixin.__init__(self)

    #----------------------------------------------------------------------
    def GetListCtrl(self):
        return self

########################################################################
class MyForm(wx.Frame):

    #----------------------------------------------------------------------
    def __init__(self):
        wx.Frame.__init__(self, None, wx.ID_ANY, "List Control Tutorial")

        # Add a panel so it looks the correct on all platforms
        panel = wx.Panel(self, wx.ID_ANY)
        self.index = 0

        self.list_ctrl = MyListCtrl(panel)
        self.list_ctrl.InsertColumn(0, 'Subject')
        self.list_ctrl.InsertColumn(1, 'Due')
        self.list_ctrl.InsertColumn(2, 'Location', width=125)

        btn = wx.Button(panel, label="Add Line")
        btn.Bind(wx.EVT_BUTTON, self.add_line)

        sizer = wx.BoxSizer(wx.VERTICAL)
        sizer.Add(self.list_ctrl, 1, wx.ALL|wx.EXPAND, 5)
        sizer.Add(btn, 0, wx.ALL|wx.CENTER, 5)
        panel.SetSizer(sizer)

    #----------------------------------------------------------------------
    def add_line(self, event):
        line = "Line %s" % self.index
        self.list_ctrl.InsertStringItem(self.index, line)
        self.list_ctrl.SetStringItem(self.index, 1, "01/19/2010")
        self.list_ctrl.SetStringItem(self.index, 2, "USA")
        self.index += 1

#----------------------------------------------------------------------
# Run the program
if __name__ == "__main__":
    app = wx.App(False)
    frame = MyForm()
    frame.Show()
    app.MainLoop()

我在 Windows 7 上使用 wxPython 2.6 和 wxPython 2.8.12。我可以一次选择多个项目并取消选择它们。

【讨论】:

  • 我应该更详细。我不确定您如何选择多行。我认为他们的方式是按住“ctrl”并选择。我希望有一种不同的方法可以解决这个问题,我可以单独左键单击每个,它们都保持突出显示。
【解决方案2】:

它很恶心,但这会做你想做的事

def OnSelectMustHave(self,event):        
    info = event.GetData()
    InfoList.append(info)
    def do_select():
          map(lambda x:self.list_ctrl.Select(x,1) ,set(InfoList))

    wx.CallLater(100,do_select)
    print info,InfoList
    return

但是,由于您在 InfoList 中有选定元素的列表,因此类似以下解决方案可能会起作用

def OnSelectMustHave(self,event):        
    info = event.GetData()
    InfoList.append(info)
    map(lambda x:self.list_ctrl.SetItemBackgroundColour(x,wx.LIGHT_GREY) ,set(InfoList))
    print info,InfoList
    return

【讨论】:

    猜你喜欢
    • 2013-08-06
    • 2017-08-06
    • 1970-01-01
    • 1970-01-01
    • 2020-09-28
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多