Можно ли ограничить TextCtrl принимать числа только в wxPython?
Я хочу иметь текстовый элемент управления, который принимает только числа. (Только целочисленные значения, такие как 45 или 366)
каков наилучший способ сделать это?
6 ответов
IntCtrl
, Masked Edit Control
и NumCtrl
все предназначены для этого, с различными уровнями контроля. Проверьте демонстрацию wx в разделе "Больше окон / элементов управления", чтобы увидеть, как они работают.
(или, если вы вместо этого действительно хотите сделать это напрямую с помощью необработанного TextCtrl, я думаю, вы захотите поймать события EVT_CHAR, проверить символы и вызвать evt.Пропустить (), если это разрешенный символ.)
Я должен был сделать что-то подобное, проверяя буквенно-цифровые коды. Подсказка на EVT_CHAR была правильной:
class TestPanel(wx.Panel):
def __init__(self, parent):
wx.Panel.__init__(self, parent, -1)
self.entry = wx.TextCtrl(self, -1)
self.entry.Bind(wx.EVT_CHAR, self.handle_keypress)
def handle_keypress(self, event):
keycode = event.GetKeyCode()
if keycode < 255:
# valid ASCII
if chr(keycode).isalnum():
# Valid alphanumeric character
event.Skip()
Вы можете попробовать IntCtrl
, EVT_CHAR
или реализовать новый / существующий валидатор (например,IntValidator). Валидаторы могут использоваться для проверки поля (полезно при попытке проверить несколько вещей в диалоговом окне/панели), а также могут использоваться с EVT_CHAR для ограничения ввода в поле.
как отмечают другие ответы, это можно сделать с помощью EVT_CHAR
обработчик. Вы захотите позвонить event.Skip()
для символов, которые вы хотите пропустить, а не вызывать его для тех, которые вы хотите заблокировать. Один нюанс заключается в том, что вы, вероятно, также хотите позвонить event.Skip()
для символов вкладки; нажатие tab запускает EVT_CHAR
событие, и если вы не называете event.Skip()
, вы эффективно отключите обход вкладки между TextCtrl
s.
вот минимальное приложение, показывающее два TextCtrl
s, которые принимают целочисленные или десятичные числа с рабочим обходом табуляции:
import wx
app = wx.App()
frame = wx.Frame(None, -1, 'simple.py')
panel = wx.Panel(frame)
text_ctrl_1 = wx.TextCtrl(panel, value='123')
text_ctrl_2 = wx.TextCtrl(panel, value='456', pos=(0, 30))
def block_non_numbers(event):
key_code = event.GetKeyCode()
# Allow ASCII numerics
if ord('0') <= key_code <= ord('9'):
event.Skip()
return
# Allow decimal points
if key_code == ord('.'):
event.Skip()
return
# Allow tabs, for tab navigation between TextCtrls
if key_code == ord('\t'):
event.Skip()
return
# Block everything else
return
text_ctrl_1.Bind(wx.EVT_CHAR, block_non_numbers)
text_ctrl_2.Bind(wx.EVT_CHAR, block_non_numbers)
frame.Show()
app.MainLoop()
NumCtrl имеет некоторые странные причуды для меня. Вот моя попытка создать элемент управления Number на основе EVT_CHAR и keycode.
этот элемент управления позволяет номера, а также все специальные keycode (ctrl combo, стрелка, backspace и т.д...), так что copy-paste, undo-redo, select-all etc все еще работает. Он будет блокировать только другие печатаемые символы (используя строку.для печати) и символы юникода (с помощью WXK_NONE)
другой способ проверить и разрешить все специальный код можно найти ответ. Это лучший подход, но требует больше кода.
import string
MyNumCtrl = wx.TextCtrl()
MyNumCtrl.Bind(EVT_CHAR, onChar)
def onChar(self, event):
keycode = event.GetKeyCode()
obj = event.GetEventObject()
val = obj.GetValue()
# filter unicode characters
if keycode == wx.WXK_NONE:
pass
# allow digits
elif chr(keycode) in string.digits:
event.Skip()
# allow special, non-printable keycodes
elif chr(keycode) not in string.printable:
event.Skip() # allow all other special keycode
# allow '-' for negative numbers
elif chr(keycode) == '-':
if val[0] == '-':
obj.SetValue(val[1:])
else:
obj.SetValue('-' + val)
# allow '.' for float numbers
elif chr(keycode) == '.' and '.' not in val:
event.Skip()
return