[Mobile]/[Flutter]

[Flutter] 다양한 FormBuilder,FormValidator

HiSmith 2023. 8. 26. 18:40
반응형

다양한 폼 빌더를 구현했다.

사실 UI 와, 폼 빌더의 동작방식에 대한 내용인데. 어느 폼같이 비슷비슷하기때문에 정리용으로 작성한다.

 

폼은 formkey로 동작한다.

final _formKey = GlobalKey<FormBuilderState>();
@override
  Widget build(BuildContext context) {
    return  SingleChildScrollView(
      child: Column(
        children: <Widget>[
          FormBuilder(
            key: _formKey,
            // enabled: false,
            onChanged: () {
              _formKey.currentState!.save();
            },
            autovalidateMode: AutovalidateMode.disabled,
            initialValue: const {
              'gender': '상관없음'
            },
            skipDisabled: true,
            child:

위와 같이 선언하여 사용할 수 있고,

폼 빌더의 onChanged메소드와 연결하면, 해당 값이 변경될때마다 form 자체적으로 관리하는

state를 저장하게 된다.

  FormBuilderTextField(
              autovalidateMode: AutovalidateMode.always,
                name: 'title',
                decoration: InputDecoration(
                  labelText: '제목',
                ),
                onChanged: (val) {
                  setState(() {
                    _titleHasError =
                    !(_formKey.currentState?.fields['title']?.validate() ??
                        false);
                  });
                },
                validator: FormBuilderValidators.compose([
                  FormBuilderValidators.required(errorText: '제목은 필수 값 입니다.'),
                  FormBuilderValidators.max(20,errorText: '제목은 20자 이하여야 합니다.'),
                ]),
                keyboardType: TextInputType.number,
                textInputAction: TextInputAction.next,),

폼 텍스트 필드이다.

벨리데이터를 달면, 각 조건을 자동으로 검사해주는데,

기본으로 제공해주는 errorText도 있으나, 커스텀하게 다는게 좋다 (왜냐면 디폴트가 영어임)

 

FormBuilderDateTimePicker(
                  name: 'date',
                  initialEntryMode: DatePickerEntryMode.calendar,
                  initialValue: DateTime.now(),
                  inputType: InputType.both,
                  format: DateFormat("yyyy-MM-dd hh:mm"),
                  decoration: InputDecoration(
                    icon : const Icon(Icons.access_time,color: Colors.green,),
                    labelText: '시간',
                    suffixIcon: IconButton(
                      icon: const Icon(Icons.close),
                      onPressed: () {
                        _formKey.currentState!.fields['date']?.didChange(null);
                      },
                    ),
                  ),
                  initialTime: const TimeOfDay(hour: 8, minute: 0),
                  // locale: const Locale.fromSubtags(languageCode: 'fr'),
                ),

폼 날짜 피커이다.

이것도 뭐 사용방식은 비슷하다.

더많은 속성은 라이브러리 소스를 보면 바로 이해가 된다.

FormBuilderSlider(
                  name: 'slider',
                  validator: FormBuilderValidators.compose([
                    FormBuilderValidators.min(1,errorText: "인원수는 1명 이상이여야 합니다."),
                  ]),
                  onChanged: _onChanged,
                  min: 0.0,
                  max: 20.0,
                  initialValue: 2.0,
                  divisions: 20,
                  activeColor: Colors.green,
                  inactiveColor: Colors.green[100],
                  decoration: const InputDecoration(
                    icon: const Icon(Icons.people_alt_rounded,color: Colors.green,),
                    labelText: '인원 수',
                  ),
                ),

 

슬라이더이다.

인원에 대한 제한을 입력받는게 아니라, 슬라이더로 조정하도록 하고 싶어서 구현했다.

max값과 벨리데이터를 구현할 수 있다.

FormBuilderDropdown<String>(
                  name: 'gender',
                  decoration: InputDecoration(
                    icon: const Icon(Icons.back_hand_sharp,color: Colors.green,)
                  ),
                  items: genderOptions
                      .map((gender) => DropdownMenuItem(
                    alignment: AlignmentDirectional.center,
                    value: gender,
                    child: Text(gender),
                  ))
                      .toList(),
                  onChanged: (val) {
                    setState(() {
                      _genderHasError = !(_formKey
                          .currentState?.fields['gender']
                          ?.validate() ??
                          false);
                    });
                  },
                  valueTransformer: (val) => val?.toString(),
                ),

드롭다운 버튼이다.

List<String>을 아이템에 넣게 되면 해당 아이템들이 노출된다.

이렇게  표현이 가능하다.

  var genderOptions = ['남자', '여자', '상관없음'];

마지막에 해당 값을 String으로 바꿔야 json에 대한 관리가 가능하다.

FormBuilderFilterChip<String>(
                  autovalidateMode: AutovalidateMode.onUserInteraction,
                  decoration: const InputDecoration(
                      labelText: '소개 태그'),
                  name: 'introduce_tag',
                  selectedColor: Colors.white,
                  options: const [
                    FormBuilderChipOption(
                      value: '요금은 더치',
                    ),
                    FormBuilderChipOption(
                      value: '요금은 주최자',
                    ),
                    FormBuilderChipOption(
                      value: '게임하고 식사도',
                    ),
                    FormBuilderChipOption(
                      value: '게임만',
                    ),
                    FormBuilderChipOption(
                      value: '초보만',
                    ),
                    FormBuilderChipOption(
                      value: '고수만',
                    ),
                  ],
                  onChanged: _onChanged,
                )

태그에 대한 내용이다.

해당 태그는 체크와 체크해제를 할 수있다. 해당 값도 리스트 형태로 전달 된다.

 

마지막으로해당 값을 저장하기 위한 액션 버튼을 구현해야한다.

Row(
            children: <Widget>[
              Expanded(
                child: ElevatedButton(
                  onPressed: () {
                    if (_formKey.currentState?.saveAndValidate() ?? false) {
                      debugPrint(_formKey.currentState?.value.toString());
                    } else {
                      debugPrint(_formKey.currentState?.value.toString());
                      debugPrint('validation failed');
                    }
                  },
                  child: const Text(
                    '생성하기',
                    style: TextStyle(color: Colors.white),
                  ),
                ),
              ),
              const SizedBox(width: 20),
            ],

 

해당 같이하면, 폼에 있는 것을 키값으로 가져와서, value값을 확인할 수 있다.

즉 마지막에 저장된 최종 currentState를 저장하고 보여준다.

 

 

이로써 폼 빌더는 끝났다.

반응형