Можно ли ограничить 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(), вы эффективно отключите обход вкладки между TextCtrls.

вот минимальное приложение, показывающее два TextCtrls, которые принимают целочисленные или десятичные числа с рабочим обходом табуляции:

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

пожалуйста, проверьте "Validator.py" сценарий в демо-версии wxpython. это именно то, что вам нужно