from wxPython.wx import *
 
+import string
+
+#---------------------------------------------------------------------------
+
+class wxFindPrefixListBox(wxListBox):
+    def __init__(self, parent, id, pos=wxDefaultPosition, size=wxDefaultSize,
+                 choices=[], style=0, validator=wxDefaultValidator):
+        wxListBox.__init__(self, parent, id, pos, size, choices, style, validator)
+        self.typedText = ''
+        EVT_KEY_UP(self, self.OnKey)
+
+
+    def FindPrefix(self, prefix):
+        if prefix:
+            prefix = string.lower(prefix)
+            length = len(prefix)
+            for x in range(self.Number()):
+                text = self.GetString(x)
+                text = string.lower(text)
+                if text[:length] == prefix:
+                    return x
+        return -1
+
+
+    def OnKey(self, evt):
+        key = evt.GetKeyCode()
+        if key >= 32 and key <= 127:
+            self.typedText = self.typedText + chr(key)
+            item = self.FindPrefix(self.typedText)
+            if item != -1:
+                self.SetSelection(item)
+
+        elif key == WXK_BACK:   # backspace removes one character and backs up
+            self.typedText = self.typedText[:-1]
+            if not self.typedText:
+                self.SetSelection(0)
+            else:
+                item = self.FindPrefix(self.typedText)
+                if item != -1:
+                    self.SetSelection(item)
+
+        else:
+            evt.Skip()
+
+
 #---------------------------------------------------------------------------
 
 class TestListBox(wxPanel):
         EVT_LISTBOX(self, 60, self.EvtListBox)
         EVT_LISTBOX_DCLICK(self, 60, self.EvtListBoxDClick)
         EVT_RIGHT_UP(self.lb1, self.EvtRightButton)
-        self.lb1.SetSelection(0)
+        self.lb1.SetSelection(3)
 
 
         wxStaticText(self, -1, "Select many:", wxPoint(200, 50), wxSize(65, 18))
         self.lb2.SetSelection(0)
 
 
+        sampleList = sampleList + ['test a', 'test aa', 'test aab',
+                                   'test ab', 'test abc', 'test abcc',
+                                   'test abcd' ]
+        sampleList.sort()
+        wxStaticText(self, -1, "Find Prefix:", wxPoint(15, 250))
+        fp = wxFindPrefixListBox(self, -1, wxPoint(80, 250), wxSize(80, 120),
+                                 sampleList, wxLB_SINGLE)
+        fp.SetSelection(0)
+
+
     def EvtListBox(self, event):
         self.log.WriteText('EvtListBox: %s\n' % event.GetString())
 
 overview = """\
 A listbox is used to select one or more of a list of strings. The strings are displayed in a scrolling box, with the selected string(s) marked in reverse video. A listbox can be single selection (if an item is selected, the previous selection is removed) or multiple selection (clicking an item toggles the item on or off independently of other selections).
 
-List box elements are numbered from zero.
-
-wxListBox()
----------------------
-
-Default constructor.
-
-wxListBox(wxWindow* parent, wxWindowID id, const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxDefaultSize, int n, const wxString choices[] = NULL, long style = 0, const wxValidator& validator = wxDefaultValidator, const wxString& name = "listBox")
-
-Constructor, creating and showing a list box.
-
-Parameters
--------------------
-
-parent = Parent window. Must not be NULL.
-
-id = Window identifier. A value of -1 indicates a default value.
-
-pos = Window position.
-
-size = Window size. If the default size (-1, -1) is specified then the window is sized appropriately.
-
-n = Number of strings with which to initialise the control.
-
-choices = An array of strings with which to initialise the control.
-
-style = Window style. See wxListBox.
-
-validator = Window validator.
-
-name = Window name.
 """