@@ -28,10 +28,31 @@ const double _composeButtonSize = 44;
28
28
///
29
29
/// Subclasses must ensure that [_update] is called in all exposed constructors.
30
30
abstract class ComposeController <ErrorT > extends TextEditingController {
31
+ int get maxLengthUnicodeCodePoints;
32
+
31
33
String get textNormalized => _textNormalized;
32
34
late String _textNormalized;
33
35
String _computeTextNormalized ();
34
36
37
+ /// Length of [textNormalized] in Unicode code points
38
+ /// if it might exceed [maxLengthUnicodeCodePoints] , else null.
39
+ ///
40
+ /// Use this instead of [String.length]
41
+ /// to enforce a max length expressed in code points.
42
+ /// [String.length] is conservative and may cut the user off too short.
43
+ ///
44
+ /// Counting code points ([String.runes] )
45
+ /// is more expensive than getting the number of UTF-16 code units
46
+ /// ([String.length] ), so we avoid it when the result definitely won't exceed
47
+ /// [maxLengthUnicodeCodePoints] .
48
+ late int ? _lengthUnicodeCodePointsIfLong;
49
+ @visibleForTesting
50
+ int ? get debugLengthUnicodeCodePointsIfLong => _lengthUnicodeCodePointsIfLong;
51
+ int ? _computeLengthUnicodeCodePointsIfLong () =>
52
+ _textNormalized.length > maxLengthUnicodeCodePoints
53
+ ? _textNormalized.runes.length
54
+ : null ;
55
+
35
56
List <ErrorT > get validationErrors => _validationErrors;
36
57
late List <ErrorT > _validationErrors;
37
58
List <ErrorT > _computeValidationErrors ();
@@ -40,6 +61,8 @@ abstract class ComposeController<ErrorT> extends TextEditingController {
40
61
41
62
void _update () {
42
63
_textNormalized = _computeTextNormalized ();
64
+ // uses _textNormalized, so comes after _computeTextNormalized()
65
+ _lengthUnicodeCodePointsIfLong = _computeLengthUnicodeCodePointsIfLong ();
43
66
_validationErrors = _computeValidationErrors ();
44
67
hasValidationErrors.value = _validationErrors.isNotEmpty;
45
68
}
@@ -74,6 +97,9 @@ class ComposeTopicController extends ComposeController<TopicValidationError> {
74
97
// https://zulip.com/help/require-topics
75
98
final mandatory = true ;
76
99
100
+ // TODO(#307) use `max_topic_length` instead of hardcoded limit
101
+ @override final maxLengthUnicodeCodePoints = kMaxTopicLengthCodePoints;
102
+
77
103
@override
78
104
String _computeTextNormalized () {
79
105
String trimmed = text.trim ();
@@ -86,11 +112,10 @@ class ComposeTopicController extends ComposeController<TopicValidationError> {
86
112
if (mandatory && textNormalized == kNoTopicTopic)
87
113
TopicValidationError .mandatoryButEmpty,
88
114
89
- // textNormalized.length is the number of UTF-16 code units, while the server
90
- // API expresses the max in Unicode code points. So this comparison will
91
- // be conservative and may cut the user off shorter than necessary.
92
- // TODO(#1238) stop cutting off shorter than necessary
93
- if (textNormalized.length > kMaxTopicLengthCodePoints)
115
+ if (
116
+ _lengthUnicodeCodePointsIfLong != null
117
+ && _lengthUnicodeCodePointsIfLong! > maxLengthUnicodeCodePoints
118
+ )
94
119
TopicValidationError .tooLong,
95
120
];
96
121
}
@@ -125,6 +150,9 @@ class ComposeContentController extends ComposeController<ContentValidationError>
125
150
_update ();
126
151
}
127
152
153
+ // TODO(#1237) use `max_message_length` instead of hardcoded limit
154
+ @override final maxLengthUnicodeCodePoints = kMaxMessageLengthCodePoints;
155
+
128
156
int _nextQuoteAndReplyTag = 0 ;
129
157
int _nextUploadTag = 0 ;
130
158
@@ -266,11 +294,10 @@ class ComposeContentController extends ComposeController<ContentValidationError>
266
294
if (textNormalized.isEmpty)
267
295
ContentValidationError .empty,
268
296
269
- // normalized.length is the number of UTF-16 code units, while the server
270
- // API expresses the max in Unicode code points. So this comparison will
271
- // be conservative and may cut the user off shorter than necessary.
272
- // TODO(#1238) stop cutting off shorter than necessary
273
- if (textNormalized.length > kMaxMessageLengthCodePoints)
297
+ if (
298
+ _lengthUnicodeCodePointsIfLong != null
299
+ && _lengthUnicodeCodePointsIfLong! > maxLengthUnicodeCodePoints
300
+ )
274
301
ContentValidationError .tooLong,
275
302
276
303
if (_quoteAndReplies.isNotEmpty)
0 commit comments