何も構築する必要はありません (少なくともサーバー側では) - WTForms は既に必要なものをサポートしています - それらは「フィールド エンクロージャ」と呼ばれます。探している動作は と にwtforms.fields.FormField
ありますwtforms.fields.FieldList
class ContactForm(Form):
name = TextField("Name", validators=[Required()])
contact_type = SelectField("Contact Type",
validators=[Required()],
choices=[
("email", "Email"),
("phone", "Phone Number"),
("im", "Instant Message")
])
# `If` is a custom validator - see below
email_address = TextField("Email",
validators=[If("contact_type",
"email",
[Required(), Email()])
])
phone_number = TextField("Phone #",
validators=[If("contact_type",
"phone", [Required()])
])
im_handle = TextField("IM Handle",
validators=[If("contact_type",
"im", [Required()])
])
class SignUpForm(Form):
# Other fields go here
contacts = FieldList(FormField(ContactForm))
ユーザーの選択に応じて、適切なフィールドを検証するためのカスタム バリデーターも必要になります。
# CAUTION: Untested code ahead
class If(object):
def __init__(self,
parent,
run_validation=None,
extra_validators=None,
msg=None):
self.parent = parent
self.msg = msg if msg is not None else u"Invalid"
if callable(run_validation):
self.run_validation = run_validation
else:
_run_validation = lambda self, parent, form: parent.data == run_validation
self.run_validation = _run_validation
self.extra_validators = extra_validators if extra_validators is not None \
else []
def __call__(self, field, form):
parent = getattr(form, self.parent)
if self.run_validation(parent, form):
return field.validate(form, extra_validators=self.extra_validators)
サーバー側で呼び出すform.validate()
と、フィールドは要件に対して自動的にチェックされ、エラーが適切に入力されるため、クライアント側でそれらをレンダリングできます。
クライアント側で新しいフィールドを作成するのは簡単で、WTForms が使用するのと同じ命名規則を使用して名前を付ける限り、WTForms はバックエンドでそれらを取得しますfield.short_name + '-' + index
。