Skip to content

Commit a23309a

Browse files
chrisbobbegnprice
authored andcommitted
compose: Enforce max topic/content length by code points, following API
Fixes zulip#1238
1 parent e67b014 commit a23309a

File tree

2 files changed

+54
-32
lines changed

2 files changed

+54
-32
lines changed

lib/widgets/compose_box.dart

+37-10
Original file line numberDiff line numberDiff line change
@@ -28,10 +28,31 @@ const double _composeButtonSize = 44;
2828
///
2929
/// Subclasses must ensure that [_update] is called in all exposed constructors.
3030
abstract class ComposeController<ErrorT> extends TextEditingController {
31+
int get maxLengthUnicodeCodePoints;
32+
3133
String get textNormalized => _textNormalized;
3234
late String _textNormalized;
3335
String _computeTextNormalized();
3436

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+
3556
List<ErrorT> get validationErrors => _validationErrors;
3657
late List<ErrorT> _validationErrors;
3758
List<ErrorT> _computeValidationErrors();
@@ -40,6 +61,8 @@ abstract class ComposeController<ErrorT> extends TextEditingController {
4061

4162
void _update() {
4263
_textNormalized = _computeTextNormalized();
64+
// uses _textNormalized, so comes after _computeTextNormalized()
65+
_lengthUnicodeCodePointsIfLong = _computeLengthUnicodeCodePointsIfLong();
4366
_validationErrors = _computeValidationErrors();
4467
hasValidationErrors.value = _validationErrors.isNotEmpty;
4568
}
@@ -74,6 +97,9 @@ class ComposeTopicController extends ComposeController<TopicValidationError> {
7497
// https://zulip.com/help/require-topics
7598
final mandatory = true;
7699

100+
// TODO(#307) use `max_topic_length` instead of hardcoded limit
101+
@override final maxLengthUnicodeCodePoints = kMaxTopicLengthCodePoints;
102+
77103
@override
78104
String _computeTextNormalized() {
79105
String trimmed = text.trim();
@@ -86,11 +112,10 @@ class ComposeTopicController extends ComposeController<TopicValidationError> {
86112
if (mandatory && textNormalized == kNoTopicTopic)
87113
TopicValidationError.mandatoryButEmpty,
88114

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+
)
94119
TopicValidationError.tooLong,
95120
];
96121
}
@@ -125,6 +150,9 @@ class ComposeContentController extends ComposeController<ContentValidationError>
125150
_update();
126151
}
127152

153+
// TODO(#1237) use `max_message_length` instead of hardcoded limit
154+
@override final maxLengthUnicodeCodePoints = kMaxMessageLengthCodePoints;
155+
128156
int _nextQuoteAndReplyTag = 0;
129157
int _nextUploadTag = 0;
130158

@@ -266,11 +294,10 @@ class ComposeContentController extends ComposeController<ContentValidationError>
266294
if (textNormalized.isEmpty)
267295
ContentValidationError.empty,
268296

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+
)
274301
ContentValidationError.tooLong,
275302

276303
if (_quoteAndReplies.isNotEmpty)

test/widgets/compose_box_test.dart

+17-22
Original file line numberDiff line numberDiff line change
@@ -256,20 +256,17 @@ void main() {
256256
await checkErrorResponse(tester);
257257
});
258258

259-
// TODO(#1238) unskip
260-
// testWidgets('max-length content not rejected', (tester) async {
261-
// await prepareWithContent(tester,
262-
// makeStringWithCodePoints(kMaxMessageLengthCodePoints));
263-
// await tapSendButton(tester);
264-
// checkNoErrorDialog(tester);
265-
// });
266-
267-
// TODO(#1238) replace with above commented-out test
268-
testWidgets('some content not rejected', (tester) async {
269-
await prepareWithContent(tester, 'a' * kMaxMessageLengthCodePoints);
259+
testWidgets('max-length content not rejected', (tester) async {
260+
await prepareWithContent(tester,
261+
makeStringWithCodePoints(kMaxMessageLengthCodePoints));
270262
await tapSendButton(tester);
271263
checkNoErrorDialog(tester);
272264
});
265+
266+
testWidgets('code points not counted unnecessarily', (tester) async {
267+
await prepareWithContent(tester, 'a' * kMaxMessageLengthCodePoints);
268+
check(controller!.content.debugLengthUnicodeCodePointsIfLong).isNull();
269+
});
273270
});
274271

275272
group('topic', () {
@@ -296,20 +293,18 @@ void main() {
296293
await checkErrorResponse(tester);
297294
});
298295

299-
// TODO(#1238) unskip
300-
// testWidgets('max-length topic not rejected', (tester) async {
301-
// await prepareWithTopic(tester,
302-
// makeStringWithCodePoints(kMaxTopicLengthCodePoints));
303-
// await tapSendButton(tester);
304-
// checkNoErrorDialog(tester);
305-
// });
306-
307-
// TODO(#1238) replace with above commented-out test
308-
testWidgets('some topic not rejected', (tester) async {
309-
await prepareWithTopic(tester, 'a' * kMaxTopicLengthCodePoints);
296+
testWidgets('max-length topic not rejected', (tester) async {
297+
await prepareWithTopic(tester,
298+
makeStringWithCodePoints(kMaxTopicLengthCodePoints));
310299
await tapSendButton(tester);
311300
checkNoErrorDialog(tester);
312301
});
302+
303+
testWidgets('code points not counted unnecessarily', (tester) async {
304+
await prepareWithTopic(tester, 'a' * kMaxTopicLengthCodePoints);
305+
check((controller as StreamComposeBoxController)
306+
.topic.debugLengthUnicodeCodePointsIfLong).isNull();
307+
});
313308
});
314309
});
315310

0 commit comments

Comments
 (0)