2004-04-19 23:24:37 +00:00
|
|
|
#----------------------------------------------------------------------------
|
|
|
|
# Name: masked.combobox.py
|
|
|
|
# Authors: Will Sadkin
|
|
|
|
# Email: wsadkin@nameconnector.com
|
|
|
|
# Created: 02/11/2003
|
|
|
|
# Copyright: (c) 2003 by Will Sadkin, 2003
|
|
|
|
# RCS-ID: $Id$
|
|
|
|
# License: wxWidgets license
|
|
|
|
#----------------------------------------------------------------------------
|
|
|
|
#
|
|
|
|
# This masked edit class allows for the semantics of masked controls
|
|
|
|
# to be applied to combo boxes.
|
|
|
|
#
|
|
|
|
#----------------------------------------------------------------------------
|
|
|
|
|
2004-10-11 22:13:18 +00:00
|
|
|
"""
|
|
|
|
Provides masked edit capabilities within a ComboBox format, as well as
|
|
|
|
a base class from which you can derive masked comboboxes tailored to a specific
|
|
|
|
function. See maskededit module overview for how to configure the control.
|
|
|
|
"""
|
|
|
|
|
|
|
|
import wx, types, string
|
2004-04-19 23:24:37 +00:00
|
|
|
from wx.lib.masked import *
|
|
|
|
|
|
|
|
# jmg 12/9/03 - when we cut ties with Py 2.2 and earlier, this would
|
|
|
|
# be a good place to implement the 2.3 logger class
|
|
|
|
from wx.tools.dbg import Logger
|
2004-10-11 22:13:18 +00:00
|
|
|
##dbg = Logger()
|
2005-04-08 21:31:25 +00:00
|
|
|
##dbg(enable=1)
|
2004-04-19 23:24:37 +00:00
|
|
|
|
|
|
|
## ---------- ---------- ---------- ---------- ---------- ---------- ----------
|
|
|
|
## Because calling SetSelection programmatically does not fire EVT_COMBOBOX
|
|
|
|
## events, we have to do it ourselves when we auto-complete.
|
|
|
|
class MaskedComboBoxSelectEvent(wx.PyCommandEvent):
|
2004-10-11 22:13:18 +00:00
|
|
|
"""
|
|
|
|
Because calling SetSelection programmatically does not fire EVT_COMBOBOX
|
|
|
|
events, the derived control has to do it itself when it auto-completes.
|
|
|
|
"""
|
2004-04-19 23:24:37 +00:00
|
|
|
def __init__(self, id, selection = 0, object=None):
|
|
|
|
wx.PyCommandEvent.__init__(self, wx.wxEVT_COMMAND_COMBOBOX_SELECTED, id)
|
|
|
|
|
|
|
|
self.__selection = selection
|
|
|
|
self.SetEventObject(object)
|
|
|
|
|
|
|
|
def GetSelection(self):
|
|
|
|
"""Retrieve the value of the control at the time
|
|
|
|
this event was generated."""
|
|
|
|
return self.__selection
|
|
|
|
|
|
|
|
|
|
|
|
class BaseMaskedComboBox( wx.ComboBox, MaskedEditMixin ):
|
|
|
|
"""
|
2004-10-11 22:13:18 +00:00
|
|
|
Base class for generic masked edit comboboxes; allows auto-complete of values.
|
|
|
|
It is not meant to be instantiated directly, but rather serves as a base class
|
|
|
|
for any subsequent refinements.
|
2004-04-19 23:24:37 +00:00
|
|
|
"""
|
|
|
|
def __init__( self, parent, id=-1, value = '',
|
|
|
|
pos = wx.DefaultPosition,
|
|
|
|
size = wx.DefaultSize,
|
|
|
|
choices = [],
|
|
|
|
style = wx.CB_DROPDOWN,
|
|
|
|
validator = wx.DefaultValidator,
|
|
|
|
name = "maskedComboBox",
|
|
|
|
setupEventHandling = True, ## setup event handling by default):
|
|
|
|
**kwargs):
|
|
|
|
|
|
|
|
|
|
|
|
kwargs['choices'] = choices ## set up maskededit to work with choice list too
|
|
|
|
|
|
|
|
## Since combobox completion is case-insensitive, always validate same way
|
|
|
|
if not kwargs.has_key('compareNoCase'):
|
|
|
|
kwargs['compareNoCase'] = True
|
|
|
|
|
|
|
|
MaskedEditMixin.__init__( self, name, **kwargs )
|
|
|
|
|
|
|
|
self._choices = self._ctrl_constraints._choices
|
|
|
|
## dbg('self._choices:', self._choices)
|
|
|
|
|
|
|
|
if self._ctrl_constraints._alignRight:
|
|
|
|
choices = [choice.rjust(self._masklength) for choice in choices]
|
|
|
|
else:
|
|
|
|
choices = [choice.ljust(self._masklength) for choice in choices]
|
|
|
|
|
|
|
|
wx.ComboBox.__init__(self, parent, id, value='',
|
|
|
|
pos=pos, size = size,
|
|
|
|
choices=choices, style=style|wx.WANTS_CHARS,
|
|
|
|
validator=validator,
|
|
|
|
name=name)
|
|
|
|
self.controlInitialized = True
|
|
|
|
|
2004-07-22 18:38:34 +00:00
|
|
|
self._PostInit(style=style, setupEventHandling=setupEventHandling,
|
|
|
|
name=name, value=value, **kwargs)
|
|
|
|
|
|
|
|
|
|
|
|
def _PostInit(self, style=wx.CB_DROPDOWN,
|
|
|
|
setupEventHandling = True, ## setup event handling by default):
|
|
|
|
name = "maskedComboBox", value='', **kwargs):
|
|
|
|
|
|
|
|
# This is necessary, because wxComboBox currently provides no
|
|
|
|
# method for determining later if this was specified in the
|
|
|
|
# constructor for the control...
|
|
|
|
self.__readonly = style & wx.CB_READONLY == wx.CB_READONLY
|
|
|
|
|
|
|
|
if not hasattr(self, 'controlInitialized'):
|
|
|
|
|
|
|
|
self.controlInitialized = True ## must have been called via XRC, therefore base class is constructed
|
|
|
|
if not kwargs.has_key('choices'):
|
|
|
|
choices=[]
|
|
|
|
kwargs['choices'] = choices ## set up maskededit to work with choice list too
|
|
|
|
self._choices = []
|
|
|
|
|
|
|
|
## Since combobox completion is case-insensitive, always validate same way
|
|
|
|
if not kwargs.has_key('compareNoCase'):
|
|
|
|
kwargs['compareNoCase'] = True
|
|
|
|
|
|
|
|
MaskedEditMixin.__init__( self, name, **kwargs )
|
|
|
|
|
|
|
|
self._choices = self._ctrl_constraints._choices
|
|
|
|
## dbg('self._choices:', self._choices)
|
|
|
|
|
|
|
|
if self._ctrl_constraints._alignRight:
|
|
|
|
choices = [choice.rjust(self._masklength) for choice in choices]
|
|
|
|
else:
|
|
|
|
choices = [choice.ljust(self._masklength) for choice in choices]
|
|
|
|
wx.ComboBox.Clear(self)
|
|
|
|
wx.ComboBox.AppendItems(self, choices)
|
|
|
|
|
|
|
|
|
2004-04-19 23:24:37 +00:00
|
|
|
# Set control font - fixed width by default
|
|
|
|
self._setFont()
|
|
|
|
|
|
|
|
if self._autofit:
|
|
|
|
self.SetClientSize(self._CalcSize())
|
2004-07-22 18:38:34 +00:00
|
|
|
width = self.GetSize().width
|
|
|
|
height = self.GetBestSize().height
|
2004-08-04 22:24:57 +00:00
|
|
|
self.SetBestFittingSize((width, height))
|
2004-07-22 18:38:34 +00:00
|
|
|
|
2004-04-19 23:24:37 +00:00
|
|
|
|
|
|
|
if value:
|
|
|
|
# ensure value is width of the mask of the control:
|
|
|
|
if self._ctrl_constraints._alignRight:
|
|
|
|
value = value.rjust(self._masklength)
|
|
|
|
else:
|
|
|
|
value = value.ljust(self._masklength)
|
|
|
|
|
|
|
|
if self.__readonly:
|
|
|
|
self.SetStringSelection(value)
|
|
|
|
else:
|
|
|
|
self._SetInitialValue(value)
|
|
|
|
|
|
|
|
|
2004-10-11 22:13:18 +00:00
|
|
|
self._SetKeycodeHandler(wx.WXK_UP, self._OnSelectChoice)
|
|
|
|
self._SetKeycodeHandler(wx.WXK_DOWN, self._OnSelectChoice)
|
2004-04-19 23:24:37 +00:00
|
|
|
|
|
|
|
if setupEventHandling:
|
|
|
|
## Setup event handlers
|
|
|
|
self.Bind(wx.EVT_SET_FOCUS, self._OnFocus ) ## defeat automatic full selection
|
|
|
|
self.Bind(wx.EVT_KILL_FOCUS, self._OnKillFocus ) ## run internal validator
|
|
|
|
self.Bind(wx.EVT_LEFT_DCLICK, self._OnDoubleClick) ## select field under cursor on dclick
|
|
|
|
self.Bind(wx.EVT_RIGHT_UP, self._OnContextMenu ) ## bring up an appropriate context menu
|
|
|
|
self.Bind(wx.EVT_CHAR, self._OnChar ) ## handle each keypress
|
2004-10-11 22:13:18 +00:00
|
|
|
self.Bind(wx.EVT_KEY_DOWN, self._OnKeyDownInComboBox ) ## for special processing of up/down keys
|
2004-04-19 23:24:37 +00:00
|
|
|
self.Bind(wx.EVT_KEY_DOWN, self._OnKeyDown ) ## for processing the rest of the control keys
|
|
|
|
## (next in evt chain)
|
|
|
|
self.Bind(wx.EVT_TEXT, self._OnTextChange ) ## color control appropriately & keep
|
|
|
|
## track of previous value for undo
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def __repr__(self):
|
|
|
|
return "<MaskedComboBox: %s>" % self.GetValue()
|
|
|
|
|
|
|
|
|
|
|
|
def _CalcSize(self, size=None):
|
|
|
|
"""
|
|
|
|
Calculate automatic size if allowed; augment base mixin function
|
|
|
|
to account for the selector button.
|
|
|
|
"""
|
|
|
|
size = self._calcSize(size)
|
|
|
|
return (size[0]+20, size[1])
|
|
|
|
|
|
|
|
|
2004-07-22 18:38:34 +00:00
|
|
|
def SetFont(self, *args, **kwargs):
|
|
|
|
""" Set the font, then recalculate control size, if appropriate. """
|
|
|
|
wx.ComboBox.SetFont(self, *args, **kwargs)
|
|
|
|
if self._autofit:
|
2004-10-11 22:13:18 +00:00
|
|
|
## dbg('calculated size:', self._CalcSize())
|
2004-07-22 18:38:34 +00:00
|
|
|
self.SetClientSize(self._CalcSize())
|
|
|
|
width = self.GetSize().width
|
|
|
|
height = self.GetBestSize().height
|
2004-10-11 22:13:18 +00:00
|
|
|
## dbg('setting client size to:', (width, height))
|
2004-08-04 22:24:57 +00:00
|
|
|
self.SetBestFittingSize((width, height))
|
2004-07-22 18:38:34 +00:00
|
|
|
|
|
|
|
|
2004-04-19 23:24:37 +00:00
|
|
|
def _GetSelection(self):
|
|
|
|
"""
|
|
|
|
Allow mixin to get the text selection of this control.
|
|
|
|
REQUIRED by any class derived from MaskedEditMixin.
|
|
|
|
"""
|
2005-04-08 21:31:25 +00:00
|
|
|
## dbg('MaskedComboBox::_GetSelection()')
|
2004-04-19 23:24:37 +00:00
|
|
|
return self.GetMark()
|
|
|
|
|
|
|
|
def _SetSelection(self, sel_start, sel_to):
|
|
|
|
"""
|
|
|
|
Allow mixin to set the text selection of this control.
|
|
|
|
REQUIRED by any class derived from MaskedEditMixin.
|
|
|
|
"""
|
2005-04-08 21:31:25 +00:00
|
|
|
## dbg('MaskedComboBox::_SetSelection: setting mark to (%d, %d)' % (sel_start, sel_to))
|
2004-04-19 23:24:37 +00:00
|
|
|
return self.SetMark( sel_start, sel_to )
|
|
|
|
|
|
|
|
|
|
|
|
def _GetInsertionPoint(self):
|
2005-04-08 21:31:25 +00:00
|
|
|
## dbg('MaskedComboBox::_GetInsertionPoint()', indent=1)
|
|
|
|
## ret = self.GetInsertionPoint()
|
|
|
|
# work around new bug in 2.5, in which the insertion point
|
|
|
|
# returned is always at the right side of the selection,
|
|
|
|
# rather than the start, as is the case with TextCtrl.
|
|
|
|
ret = self.GetMark()[0]
|
|
|
|
## dbg('returned', ret, indent=0)
|
|
|
|
return ret
|
2004-04-19 23:24:37 +00:00
|
|
|
|
|
|
|
def _SetInsertionPoint(self, pos):
|
2005-04-08 21:31:25 +00:00
|
|
|
## dbg('MaskedComboBox::_SetInsertionPoint(%d)' % pos)
|
2004-04-19 23:24:37 +00:00
|
|
|
self.SetInsertionPoint(pos)
|
|
|
|
|
|
|
|
|
|
|
|
def _GetValue(self):
|
|
|
|
"""
|
|
|
|
Allow mixin to get the raw value of the control with this function.
|
|
|
|
REQUIRED by any class derived from MaskedEditMixin.
|
|
|
|
"""
|
|
|
|
return self.GetValue()
|
|
|
|
|
|
|
|
def _SetValue(self, value):
|
|
|
|
"""
|
|
|
|
Allow mixin to set the raw value of the control with this function.
|
|
|
|
REQUIRED by any class derived from MaskedEditMixin.
|
|
|
|
"""
|
|
|
|
# For wxComboBox, ensure that values are properly padded so that
|
|
|
|
# if varying length choices are supplied, they always show up
|
|
|
|
# in the window properly, and will be the appropriate length
|
|
|
|
# to match the mask:
|
|
|
|
if self._ctrl_constraints._alignRight:
|
|
|
|
value = value.rjust(self._masklength)
|
|
|
|
else:
|
|
|
|
value = value.ljust(self._masklength)
|
|
|
|
|
|
|
|
# Record current selection and insertion point, for undo
|
|
|
|
self._prevSelection = self._GetSelection()
|
|
|
|
self._prevInsertionPoint = self._GetInsertionPoint()
|
|
|
|
wx.ComboBox.SetValue(self, value)
|
|
|
|
# text change events don't always fire, so we check validity here
|
|
|
|
# to make certain formatting is applied:
|
|
|
|
self._CheckValid()
|
|
|
|
|
|
|
|
def SetValue(self, value):
|
|
|
|
"""
|
|
|
|
This function redefines the externally accessible .SetValue to be
|
|
|
|
a smart "paste" of the text in question, so as not to corrupt the
|
|
|
|
masked control. NOTE: this must be done in the class derived
|
|
|
|
from the base wx control.
|
|
|
|
"""
|
|
|
|
if not self._mask:
|
|
|
|
wx.ComboBox.SetValue(value) # revert to base control behavior
|
|
|
|
return
|
|
|
|
# else...
|
|
|
|
# empty previous contents, replacing entire value:
|
|
|
|
self._SetInsertionPoint(0)
|
|
|
|
self._SetSelection(0, self._masklength)
|
|
|
|
|
|
|
|
if( len(value) < self._masklength # value shorter than control
|
|
|
|
and (self._isFloat or self._isInt) # and it's a numeric control
|
|
|
|
and self._ctrl_constraints._alignRight ): # and it's a right-aligned control
|
|
|
|
# try to intelligently "pad out" the value to the right size:
|
|
|
|
value = self._template[0:self._masklength - len(value)] + value
|
|
|
|
## dbg('padded value = "%s"' % value)
|
|
|
|
|
|
|
|
# For wxComboBox, ensure that values are properly padded so that
|
|
|
|
# if varying length choices are supplied, they always show up
|
|
|
|
# in the window properly, and will be the appropriate length
|
|
|
|
# to match the mask:
|
|
|
|
elif self._ctrl_constraints._alignRight:
|
|
|
|
value = value.rjust(self._masklength)
|
|
|
|
else:
|
|
|
|
value = value.ljust(self._masklength)
|
|
|
|
|
|
|
|
|
|
|
|
# make SetValue behave the same as if you had typed the value in:
|
|
|
|
try:
|
2004-06-19 16:25:55 +00:00
|
|
|
value, replace_to = self._Paste(value, raise_on_invalid=True, just_return_value=True)
|
2004-04-19 23:24:37 +00:00
|
|
|
if self._isFloat:
|
|
|
|
self._isNeg = False # (clear current assumptions)
|
|
|
|
value = self._adjustFloat(value)
|
|
|
|
elif self._isInt:
|
|
|
|
self._isNeg = False # (clear current assumptions)
|
|
|
|
value = self._adjustInt(value)
|
|
|
|
elif self._isDate and not self.IsValid(value) and self._4digityear:
|
|
|
|
value = self._adjustDate(value, fixcentury=True)
|
|
|
|
except ValueError:
|
|
|
|
# If date, year might be 2 digits vs. 4; try adjusting it:
|
|
|
|
if self._isDate and self._4digityear:
|
|
|
|
dateparts = value.split(' ')
|
|
|
|
dateparts[0] = self._adjustDate(dateparts[0], fixcentury=True)
|
|
|
|
value = string.join(dateparts, ' ')
|
|
|
|
## dbg('adjusted value: "%s"' % value)
|
|
|
|
value = self._Paste(value, raise_on_invalid=True, just_return_value=True)
|
|
|
|
else:
|
|
|
|
raise
|
|
|
|
|
|
|
|
self._SetValue(value)
|
2004-06-19 16:25:55 +00:00
|
|
|
#### dbg('queuing insertion after .SetValue', replace_to)
|
|
|
|
wx.CallAfter(self._SetInsertionPoint, replace_to)
|
|
|
|
wx.CallAfter(self._SetSelection, replace_to, replace_to)
|
2004-04-19 23:24:37 +00:00
|
|
|
|
|
|
|
|
|
|
|
def _Refresh(self):
|
|
|
|
"""
|
|
|
|
Allow mixin to refresh the base control with this function.
|
|
|
|
REQUIRED by any class derived from MaskedEditMixin.
|
|
|
|
"""
|
|
|
|
wx.ComboBox.Refresh(self)
|
|
|
|
|
|
|
|
def Refresh(self):
|
|
|
|
"""
|
|
|
|
This function redefines the externally accessible .Refresh() to
|
|
|
|
validate the contents of the masked control as it refreshes.
|
|
|
|
NOTE: this must be done in the class derived from the base wx control.
|
|
|
|
"""
|
|
|
|
self._CheckValid()
|
|
|
|
self._Refresh()
|
|
|
|
|
|
|
|
|
|
|
|
def _IsEditable(self):
|
|
|
|
"""
|
|
|
|
Allow mixin to determine if the base control is editable with this function.
|
|
|
|
REQUIRED by any class derived from MaskedEditMixin.
|
|
|
|
"""
|
|
|
|
return not self.__readonly
|
|
|
|
|
|
|
|
|
|
|
|
def Cut(self):
|
|
|
|
"""
|
|
|
|
This function redefines the externally accessible .Cut to be
|
|
|
|
a smart "erase" of the text in question, so as not to corrupt the
|
|
|
|
masked control. NOTE: this must be done in the class derived
|
|
|
|
from the base wx control.
|
|
|
|
"""
|
|
|
|
if self._mask:
|
|
|
|
self._Cut() # call the mixin's Cut method
|
|
|
|
else:
|
|
|
|
wx.ComboBox.Cut(self) # else revert to base control behavior
|
|
|
|
|
|
|
|
|
|
|
|
def Paste(self):
|
|
|
|
"""
|
|
|
|
This function redefines the externally accessible .Paste to be
|
|
|
|
a smart "paste" of the text in question, so as not to corrupt the
|
|
|
|
masked control. NOTE: this must be done in the class derived
|
|
|
|
from the base wx control.
|
|
|
|
"""
|
|
|
|
if self._mask:
|
|
|
|
self._Paste() # call the mixin's Paste method
|
|
|
|
else:
|
|
|
|
wx.ComboBox.Paste(self) # else revert to base control behavior
|
|
|
|
|
|
|
|
|
|
|
|
def Undo(self):
|
|
|
|
"""
|
|
|
|
This function defines the undo operation for the control. (The default
|
|
|
|
undo is 1-deep.)
|
|
|
|
"""
|
|
|
|
if self._mask:
|
|
|
|
self._Undo()
|
|
|
|
else:
|
|
|
|
wx.ComboBox.Undo() # else revert to base control behavior
|
|
|
|
|
|
|
|
def Append( self, choice, clientData=None ):
|
|
|
|
"""
|
2004-10-11 22:13:18 +00:00
|
|
|
This base control function override is necessary so the control can keep track
|
|
|
|
of any additions to the list of choices, because wx.ComboBox doesn't have an
|
|
|
|
accessor for the choice list. The code here is the same as in the
|
|
|
|
SetParameters() mixin function, but is done for the individual value
|
|
|
|
as appended, so the list can be built incrementally without speed penalty.
|
2004-04-19 23:24:37 +00:00
|
|
|
"""
|
|
|
|
if self._mask:
|
|
|
|
if type(choice) not in (types.StringType, types.UnicodeType):
|
|
|
|
raise TypeError('%s: choices must be a sequence of strings' % str(self._index))
|
|
|
|
elif not self.IsValid(choice):
|
|
|
|
raise ValueError('%s: "%s" is not a valid value for the control as specified.' % (str(self._index), choice))
|
|
|
|
|
|
|
|
if not self._ctrl_constraints._choices:
|
|
|
|
self._ctrl_constraints._compareChoices = []
|
|
|
|
self._ctrl_constraints._choices = []
|
|
|
|
self._hasList = True
|
|
|
|
|
|
|
|
compareChoice = choice.strip()
|
|
|
|
|
|
|
|
if self._ctrl_constraints._compareNoCase:
|
|
|
|
compareChoice = compareChoice.lower()
|
|
|
|
|
|
|
|
if self._ctrl_constraints._alignRight:
|
|
|
|
choice = choice.rjust(self._masklength)
|
|
|
|
else:
|
|
|
|
choice = choice.ljust(self._masklength)
|
|
|
|
if self._ctrl_constraints._fillChar != ' ':
|
|
|
|
choice = choice.replace(' ', self._fillChar)
|
|
|
|
## dbg('updated choice:', choice)
|
|
|
|
|
|
|
|
|
|
|
|
self._ctrl_constraints._compareChoices.append(compareChoice)
|
|
|
|
self._ctrl_constraints._choices.append(choice)
|
|
|
|
self._choices = self._ctrl_constraints._choices # (for shorthand)
|
|
|
|
|
|
|
|
if( not self.IsValid(choice) and
|
|
|
|
(not self._ctrl_constraints.IsEmpty(choice) or
|
|
|
|
(self._ctrl_constraints.IsEmpty(choice) and self._ctrl_constraints._validRequired) ) ):
|
|
|
|
raise ValueError('"%s" is not a valid value for the control "%s" as specified.' % (choice, self.name))
|
|
|
|
|
|
|
|
wx.ComboBox.Append(self, choice, clientData)
|
|
|
|
|
|
|
|
|
2004-07-22 18:38:34 +00:00
|
|
|
def AppendItems( self, choices ):
|
|
|
|
"""
|
|
|
|
AppendItems() is handled in terms of Append, to avoid code replication.
|
|
|
|
"""
|
|
|
|
for choice in choices:
|
|
|
|
self.Append(choice)
|
|
|
|
|
2004-04-19 23:24:37 +00:00
|
|
|
|
|
|
|
def Clear( self ):
|
|
|
|
"""
|
2004-10-11 22:13:18 +00:00
|
|
|
This base control function override is necessary so the derived control can
|
|
|
|
keep track of any additions to the list of choices, because wx.ComboBox
|
|
|
|
doesn't have an accessor for the choice list.
|
2004-04-19 23:24:37 +00:00
|
|
|
"""
|
|
|
|
if self._mask:
|
|
|
|
self._choices = []
|
|
|
|
self._ctrl_constraints._autoCompleteIndex = -1
|
|
|
|
if self._ctrl_constraints._choices:
|
|
|
|
self.SetCtrlParameters(choices=[])
|
|
|
|
wx.ComboBox.Clear(self)
|
|
|
|
|
|
|
|
|
|
|
|
def _OnCtrlParametersChanged(self):
|
|
|
|
"""
|
2004-10-11 22:13:18 +00:00
|
|
|
This overrides the mixin's default OnCtrlParametersChanged to detect
|
|
|
|
changes in choice list, so masked.Combobox can update the base control:
|
2004-04-19 23:24:37 +00:00
|
|
|
"""
|
|
|
|
if self.controlInitialized and self._choices != self._ctrl_constraints._choices:
|
|
|
|
wx.ComboBox.Clear(self)
|
|
|
|
self._choices = self._ctrl_constraints._choices
|
|
|
|
for choice in self._choices:
|
|
|
|
wx.ComboBox.Append( self, choice )
|
|
|
|
|
|
|
|
|
2005-03-01 16:22:31 +00:00
|
|
|
# Not all wx platform implementations have .GetMark, so we make the following test,
|
|
|
|
# and fall back to our old hack if they don't...
|
|
|
|
#
|
|
|
|
if not hasattr(wx.ComboBox, 'GetMark'):
|
|
|
|
def GetMark(self):
|
|
|
|
"""
|
|
|
|
This function is a hack to make up for the fact that wx.ComboBox has no
|
|
|
|
method for returning the selected portion of its edit control. It
|
|
|
|
works, but has the nasty side effect of generating lots of intermediate
|
|
|
|
events.
|
|
|
|
"""
|
|
|
|
## dbg(suspend=1) # turn off debugging around this function
|
|
|
|
## dbg('MaskedComboBox::GetMark', indent=1)
|
|
|
|
if self.__readonly:
|
|
|
|
## dbg(indent=0)
|
|
|
|
return 0, 0 # no selection possible for editing
|
|
|
|
## sel_start, sel_to = wxComboBox.GetMark(self) # what I'd *like* to have!
|
|
|
|
sel_start = sel_to = self.GetInsertionPoint()
|
|
|
|
## dbg("current sel_start:", sel_start)
|
|
|
|
value = self.GetValue()
|
|
|
|
## dbg('value: "%s"' % value)
|
2004-04-19 23:24:37 +00:00
|
|
|
|
2005-03-01 16:22:31 +00:00
|
|
|
self._ignoreChange = True # tell _OnTextChange() to ignore next event (if any)
|
2004-04-19 23:24:37 +00:00
|
|
|
|
2005-03-01 16:22:31 +00:00
|
|
|
wx.ComboBox.Cut(self)
|
|
|
|
newvalue = self.GetValue()
|
|
|
|
## dbg("value after Cut operation:", newvalue)
|
2004-04-19 23:24:37 +00:00
|
|
|
|
2005-03-01 16:22:31 +00:00
|
|
|
if newvalue != value: # something was selected; calculate extent
|
|
|
|
## dbg("something selected")
|
|
|
|
sel_to = sel_start + len(value) - len(newvalue)
|
|
|
|
wx.ComboBox.SetValue(self, value) # restore original value and selection (still ignoring change)
|
|
|
|
wx.ComboBox.SetInsertionPoint(self, sel_start)
|
|
|
|
wx.ComboBox.SetMark(self, sel_start, sel_to)
|
2004-04-19 23:24:37 +00:00
|
|
|
|
2005-03-01 16:22:31 +00:00
|
|
|
self._ignoreChange = False # tell _OnTextChange() to pay attn again
|
2004-04-19 23:24:37 +00:00
|
|
|
|
2005-03-01 16:22:31 +00:00
|
|
|
## dbg('computed selection:', sel_start, sel_to, indent=0, suspend=0)
|
|
|
|
return sel_start, sel_to
|
2005-04-08 21:31:25 +00:00
|
|
|
else:
|
|
|
|
def GetMark(self):
|
|
|
|
## dbg('MaskedComboBox::GetMark()', indent = 1)
|
|
|
|
ret = wx.ComboBox.GetMark(self)
|
|
|
|
## dbg('returned', ret, indent=0)
|
|
|
|
return ret
|
2004-04-19 23:24:37 +00:00
|
|
|
|
|
|
|
|
|
|
|
def SetSelection(self, index):
|
|
|
|
"""
|
2004-10-11 22:13:18 +00:00
|
|
|
Necessary override for bookkeeping on choice selection, to keep current value
|
2004-04-19 23:24:37 +00:00
|
|
|
current.
|
|
|
|
"""
|
|
|
|
## dbg('MaskedComboBox::SetSelection(%d)' % index)
|
|
|
|
if self._mask:
|
|
|
|
self._prevValue = self._curValue
|
|
|
|
self._curValue = self._choices[index]
|
|
|
|
self._ctrl_constraints._autoCompleteIndex = index
|
|
|
|
wx.ComboBox.SetSelection(self, index)
|
|
|
|
|
|
|
|
|
2004-10-11 22:13:18 +00:00
|
|
|
def _OnKeyDownInComboBox(self, event):
|
2004-04-19 23:24:37 +00:00
|
|
|
"""
|
|
|
|
This function is necessary because navigation and control key
|
|
|
|
events do not seem to normally be seen by the wxComboBox's
|
|
|
|
EVT_CHAR routine. (Tabs don't seem to be visible no matter
|
|
|
|
what... {:-( )
|
|
|
|
"""
|
|
|
|
if event.GetKeyCode() in self._nav + self._control:
|
|
|
|
self._OnChar(event)
|
|
|
|
return
|
|
|
|
else:
|
|
|
|
event.Skip() # let mixin default KeyDown behavior occur
|
|
|
|
|
|
|
|
|
2004-10-11 22:13:18 +00:00
|
|
|
def _OnSelectChoice(self, event):
|
2004-04-19 23:24:37 +00:00
|
|
|
"""
|
|
|
|
This function appears to be necessary, because the processing done
|
|
|
|
on the text of the control somehow interferes with the combobox's
|
|
|
|
selection mechanism for the arrow keys.
|
|
|
|
"""
|
|
|
|
## dbg('MaskedComboBox::OnSelectChoice', indent=1)
|
|
|
|
|
|
|
|
if not self._mask:
|
|
|
|
event.Skip()
|
|
|
|
return
|
|
|
|
|
|
|
|
value = self.GetValue().strip()
|
|
|
|
|
|
|
|
if self._ctrl_constraints._compareNoCase:
|
|
|
|
value = value.lower()
|
|
|
|
|
|
|
|
if event.GetKeyCode() == wx.WXK_UP:
|
|
|
|
direction = -1
|
|
|
|
else:
|
|
|
|
direction = 1
|
|
|
|
match_index, partial_match = self._autoComplete(
|
|
|
|
direction,
|
|
|
|
self._ctrl_constraints._compareChoices,
|
|
|
|
value,
|
|
|
|
self._ctrl_constraints._compareNoCase,
|
|
|
|
current_index = self._ctrl_constraints._autoCompleteIndex)
|
|
|
|
if match_index is not None:
|
|
|
|
## dbg('setting selection to', match_index)
|
|
|
|
# issue appropriate event to outside:
|
|
|
|
self._OnAutoSelect(self._ctrl_constraints, match_index=match_index)
|
|
|
|
self._CheckValid()
|
|
|
|
keep_processing = False
|
|
|
|
else:
|
|
|
|
pos = self._adjustPos(self._GetInsertionPoint(), event.GetKeyCode())
|
|
|
|
field = self._FindField(pos)
|
|
|
|
if self.IsEmpty() or not field._hasList:
|
|
|
|
## dbg('selecting 1st value in list')
|
|
|
|
self._OnAutoSelect(self._ctrl_constraints, match_index=0)
|
|
|
|
self._CheckValid()
|
|
|
|
keep_processing = False
|
|
|
|
else:
|
|
|
|
# attempt field-level auto-complete
|
|
|
|
## dbg(indent=0)
|
|
|
|
keep_processing = self._OnAutoCompleteField(event)
|
|
|
|
## dbg('keep processing?', keep_processing, indent=0)
|
|
|
|
return keep_processing
|
|
|
|
|
|
|
|
|
|
|
|
def _OnAutoSelect(self, field, match_index):
|
|
|
|
"""
|
|
|
|
Override mixin (empty) autocomplete handler, so that autocompletion causes
|
|
|
|
combobox to update appropriately.
|
|
|
|
"""
|
|
|
|
## dbg('MaskedComboBox::OnAutoSelect', field._index, indent=1)
|
|
|
|
## field._autoCompleteIndex = match_index
|
|
|
|
if field == self._ctrl_constraints:
|
|
|
|
self.SetSelection(match_index)
|
|
|
|
## dbg('issuing combo selection event')
|
|
|
|
self.GetEventHandler().ProcessEvent(
|
|
|
|
MaskedComboBoxSelectEvent( self.GetId(), match_index, self ) )
|
|
|
|
self._CheckValid()
|
|
|
|
## dbg('field._autoCompleteIndex:', match_index)
|
|
|
|
## dbg('self.GetSelection():', self.GetSelection())
|
2004-06-19 16:25:55 +00:00
|
|
|
end = self._goEnd(getPosOnly=True)
|
|
|
|
## dbg('scheduling set of end position to:', end)
|
|
|
|
# work around bug in wx 2.5
|
|
|
|
wx.CallAfter(self.SetInsertionPoint, 0)
|
|
|
|
wx.CallAfter(self.SetInsertionPoint, end)
|
2004-04-19 23:24:37 +00:00
|
|
|
## dbg(indent=0)
|
|
|
|
|
|
|
|
|
|
|
|
def _OnReturn(self, event):
|
|
|
|
"""
|
2004-10-11 22:13:18 +00:00
|
|
|
For wx.ComboBox, it seems that if you hit return when the dropdown is
|
2004-04-19 23:24:37 +00:00
|
|
|
dropped, the event that dismisses the dropdown will also blank the
|
2004-10-11 22:13:18 +00:00
|
|
|
control, because of the implementation of wxComboBox. So this function
|
|
|
|
examines the selection and if it is -1, and the value according to
|
|
|
|
(the base control!) is a value in the list, then it schedules a
|
2004-04-19 23:24:37 +00:00
|
|
|
programmatic wxComboBox.SetSelection() call to pick the appropriate
|
2004-10-11 22:13:18 +00:00
|
|
|
item in the list. (and then does the usual OnReturn bit.)
|
2004-04-19 23:24:37 +00:00
|
|
|
"""
|
|
|
|
## dbg('MaskedComboBox::OnReturn', indent=1)
|
|
|
|
## dbg('current value: "%s"' % self.GetValue(), 'current index:', self.GetSelection())
|
|
|
|
if self.GetSelection() == -1 and self.GetValue().lower().strip() in self._ctrl_constraints._compareChoices:
|
|
|
|
wx.CallAfter(self.SetSelection, self._ctrl_constraints._autoCompleteIndex)
|
|
|
|
|
|
|
|
event.m_keyCode = wx.WXK_TAB
|
|
|
|
event.Skip()
|
|
|
|
## dbg(indent=0)
|
|
|
|
|
|
|
|
|
|
|
|
class ComboBox( BaseMaskedComboBox, MaskedEditAccessorsMixin ):
|
|
|
|
"""
|
2004-10-11 22:13:18 +00:00
|
|
|
The "user-visible" masked combobox control, this class is
|
|
|
|
identical to the BaseMaskedComboBox class it's derived from.
|
|
|
|
(This extra level of inheritance allows us to add the generic
|
|
|
|
set of masked edit parameters only to this class while allowing
|
|
|
|
other classes to derive from the "base" masked combobox control,
|
|
|
|
and provide a smaller set of valid accessor functions.)
|
|
|
|
See BaseMaskedComboBox for available methods.
|
2004-04-19 23:24:37 +00:00
|
|
|
"""
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
2004-07-22 18:38:34 +00:00
|
|
|
class PreMaskedComboBox( BaseMaskedComboBox, MaskedEditAccessorsMixin ):
|
|
|
|
"""
|
2004-10-11 22:13:18 +00:00
|
|
|
This class exists to support the use of XRC subclassing.
|
2004-07-22 18:38:34 +00:00
|
|
|
"""
|
|
|
|
# This should really be wx.EVT_WINDOW_CREATE but it is not
|
|
|
|
# currently delivered for native controls on all platforms, so
|
|
|
|
# we'll use EVT_SIZE instead. It should happen shortly after the
|
|
|
|
# control is created as the control is set to its "best" size.
|
|
|
|
_firstEventType = wx.EVT_SIZE
|
|
|
|
|
|
|
|
def __init__(self):
|
|
|
|
pre = wx.PreComboBox()
|
|
|
|
self.PostCreate(pre)
|
|
|
|
self.Bind(self._firstEventType, self.OnCreate)
|
|
|
|
|
|
|
|
|
|
|
|
def OnCreate(self, evt):
|
|
|
|
self.Unbind(self._firstEventType)
|
|
|
|
self._PostInit()
|
|
|
|
|
2004-10-11 22:13:18 +00:00
|
|
|
__i = 0
|
2004-07-22 18:38:34 +00:00
|
|
|
## CHANGELOG:
|
|
|
|
## ====================
|
2005-03-01 16:22:31 +00:00
|
|
|
## Version 1.3
|
|
|
|
## 1. Made definition of "hack" GetMark conditional on base class not
|
|
|
|
## implementing it properly, to allow for migration in wx code base
|
|
|
|
## while taking advantage of improvements therein for some platforms.
|
|
|
|
##
|
2004-10-11 22:13:18 +00:00
|
|
|
## Version 1.2
|
|
|
|
## 1. Converted docstrings to reST format, added doc for ePyDoc.
|
|
|
|
## 2. Renamed helper functions, vars etc. not intended to be visible in public
|
|
|
|
## interface to code.
|
|
|
|
##
|
2004-07-22 18:38:34 +00:00
|
|
|
## Version 1.1
|
|
|
|
## 1. Added .SetFont() method that properly resizes control
|
|
|
|
## 2. Modified control to support construction via XRC mechanism.
|
|
|
|
## 3. Added AppendItems() to conform with latest combobox.
|