浏览代码

feat: limit max character length on select/multiselect options (#1246)

* Add maxLength of 30 characters and truncation to text fields

* Add dynamic height

* Add maxLength param

* Add maxLength in Rounded Field
vortixhead 3 年之前
父节点
当前提交
fa9e32d86a

+ 2 - 1
frontend/app_flowy/lib/plugins/grid/presentation/widgets/cell/select_option_cell/select_option_editor.dart

@@ -142,11 +142,12 @@ class _TextField extends StatelessWidget {
             value: (option) => option);
             value: (option) => option);
 
 
         return SizedBox(
         return SizedBox(
-          height: 42,
+          height: 62,
           child: SelectOptionTextField(
           child: SelectOptionTextField(
             options: state.options,
             options: state.options,
             selectedOptionMap: optionMap,
             selectedOptionMap: optionMap,
             distanceToText: _editorPanelWidth * 0.7,
             distanceToText: _editorPanelWidth * 0.7,
+            maxLength: 30,
             tagController: _tagController,
             tagController: _tagController,
             onClick: () => popoverMutex.close(),
             onClick: () => popoverMutex.close(),
             newText: (text) {
             newText: (text) {

+ 6 - 0
frontend/app_flowy/lib/plugins/grid/presentation/widgets/cell/select_option_cell/text_field.dart

@@ -6,6 +6,7 @@ import 'package:flowy_sdk/protobuf/flowy-grid/select_option.pb.dart';
 import 'package:flutter/material.dart';
 import 'package:flutter/material.dart';
 import 'package:easy_localization/easy_localization.dart';
 import 'package:easy_localization/easy_localization.dart';
 import 'package:app_flowy/generated/locale_keys.g.dart';
 import 'package:app_flowy/generated/locale_keys.g.dart';
+import 'package:flutter/services.dart';
 import 'package:flutter_bloc/flutter_bloc.dart';
 import 'package:flutter_bloc/flutter_bloc.dart';
 import 'package:textfield_tags/textfield_tags.dart';
 import 'package:textfield_tags/textfield_tags.dart';
 
 
@@ -20,6 +21,7 @@ class SelectOptionTextField extends StatefulWidget {
   final Function(String) onSubmitted;
   final Function(String) onSubmitted;
   final Function(String) newText;
   final Function(String) newText;
   final VoidCallback? onClick;
   final VoidCallback? onClick;
+  final int? maxLength;
 
 
   const SelectOptionTextField({
   const SelectOptionTextField({
     required this.options,
     required this.options,
@@ -29,6 +31,7 @@ class SelectOptionTextField extends StatefulWidget {
     required this.onSubmitted,
     required this.onSubmitted,
     required this.newText,
     required this.newText,
     this.onClick,
     this.onClick,
+    this.maxLength,
     TextEditingController? textController,
     TextEditingController? textController,
     FocusNode? focusNode,
     FocusNode? focusNode,
     Key? key,
     Key? key,
@@ -93,6 +96,9 @@ class _SelectOptionTextFieldState extends State<SelectOptionTextField> {
               }
               }
             },
             },
             maxLines: 1,
             maxLines: 1,
+            maxLength: widget.maxLength,
+            maxLengthEnforcement:
+                MaxLengthEnforcement.truncateAfterCompositionEnds,
             style: const TextStyle(fontSize: 14, fontWeight: FontWeight.w500),
             style: const TextStyle(fontSize: 14, fontWeight: FontWeight.w500),
             decoration: InputDecoration(
             decoration: InputDecoration(
               enabledBorder: OutlineInputBorder(
               enabledBorder: OutlineInputBorder(

+ 6 - 1
frontend/app_flowy/lib/plugins/grid/presentation/widgets/common/text_field.dart

@@ -9,6 +9,7 @@ class InputTextField extends StatefulWidget {
   final void Function() onCanceled;
   final void Function() onCanceled;
   final bool autoClearWhenDone;
   final bool autoClearWhenDone;
   final String text;
   final String text;
+  final int? maxLength;
 
 
   const InputTextField({
   const InputTextField({
     required this.text,
     required this.text,
@@ -16,6 +17,7 @@ class InputTextField extends StatefulWidget {
     required this.onCanceled,
     required this.onCanceled,
     this.onChanged,
     this.onChanged,
     this.autoClearWhenDone = false,
     this.autoClearWhenDone = false,
+    this.maxLength,
     Key? key,
     Key? key,
   }) : super(key: key);
   }) : super(key: key);
 
 
@@ -41,11 +43,14 @@ class _InputTextFieldState extends State<InputTextField> {
   Widget build(BuildContext context) {
   Widget build(BuildContext context) {
     final theme = context.watch<AppTheme>();
     final theme = context.watch<AppTheme>();
 
 
+    final height = widget.maxLength == null ? 36.0 : 56.0;
+
     return RoundedInputField(
     return RoundedInputField(
       controller: _controller,
       controller: _controller,
       focusNode: _focusNode,
       focusNode: _focusNode,
       autoFocus: true,
       autoFocus: true,
-      height: 36,
+      height: height,
+      maxLength: widget.maxLength,
       style: const TextStyle(fontSize: 13, fontWeight: FontWeight.w500),
       style: const TextStyle(fontSize: 13, fontWeight: FontWeight.w500),
       normalBorderColor: theme.shader4,
       normalBorderColor: theme.shader4,
       focusBorderColor: theme.main1,
       focusBorderColor: theme.main1,

+ 1 - 0
frontend/app_flowy/lib/plugins/grid/presentation/widgets/header/type_option/select_option.dart

@@ -256,6 +256,7 @@ class _CreateOptionTextField extends StatelessWidget {
         final text = state.newOptionName.foldRight("", (a, previous) => a);
         final text = state.newOptionName.foldRight("", (a, previous) => a);
         return InputTextField(
         return InputTextField(
           autoClearWhenDone: true,
           autoClearWhenDone: true,
+          maxLength: 30,
           text: text,
           text: text,
           onCanceled: () {
           onCanceled: () {
             context
             context

+ 1 - 0
frontend/app_flowy/lib/plugins/grid/presentation/widgets/header/type_option/select_option_editor.dart

@@ -106,6 +106,7 @@ class _OptionNameTextField extends StatelessWidget {
   Widget build(BuildContext context) {
   Widget build(BuildContext context) {
     return InputTextField(
     return InputTextField(
       text: name,
       text: name,
+      maxLength: 30,
       onCanceled: () {},
       onCanceled: () {},
       onDone: (optionName) {
       onDone: (optionName) {
         if (name != optionName) {
         if (name != optionName) {

+ 6 - 0
frontend/app_flowy/packages/flowy_infra_ui/lib/widget/rounded_input_field.dart

@@ -2,6 +2,7 @@ import 'package:flowy_infra/size.dart';
 import 'package:flowy_infra_ui/widget/rounded_button.dart';
 import 'package:flowy_infra_ui/widget/rounded_button.dart';
 import 'package:flutter/material.dart';
 import 'package:flutter/material.dart';
 import 'package:flowy_infra/time/duration.dart';
 import 'package:flowy_infra/time/duration.dart';
+import 'package:flutter/services.dart';
 
 
 class RoundedInputField extends StatefulWidget {
 class RoundedInputField extends StatefulWidget {
   final String? hintText;
   final String? hintText;
@@ -24,6 +25,7 @@ class RoundedInputField extends StatefulWidget {
   final FocusNode? focusNode;
   final FocusNode? focusNode;
   final TextEditingController? controller;
   final TextEditingController? controller;
   final bool autoFocus;
   final bool autoFocus;
+  final int? maxLength;
 
 
   const RoundedInputField({
   const RoundedInputField({
     Key? key,
     Key? key,
@@ -47,6 +49,7 @@ class RoundedInputField extends StatefulWidget {
     this.focusNode,
     this.focusNode,
     this.controller,
     this.controller,
     this.autoFocus = false,
     this.autoFocus = false,
+    this.maxLength,
   }) : super(key: key);
   }) : super(key: key);
 
 
   @override
   @override
@@ -89,6 +92,9 @@ class _RoundedInputFieldState extends State<RoundedInputField> {
           initialValue: widget.initialValue,
           initialValue: widget.initialValue,
           focusNode: widget.focusNode,
           focusNode: widget.focusNode,
           autofocus: widget.autoFocus,
           autofocus: widget.autoFocus,
+          maxLength: widget.maxLength,
+          maxLengthEnforcement:
+              MaxLengthEnforcement.truncateAfterCompositionEnds,
           onChanged: (value) {
           onChanged: (value) {
             inputText = value;
             inputText = value;
             if (widget.onChanged != null) {
             if (widget.onChanged != null) {