Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Submit feedback
Sign in
Toggle navigation
F
Front-End
Project
Project
Details
Activity
Releases
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
abdullh.alsoleman
Front-End
Commits
896c19a2
Unverified
Commit
896c19a2
authored
Oct 07, 2020
by
Justin McCandless
Committed by
GitHub
Oct 07, 2020
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Characters docs (#67361)
parent
b5562c1b
Changes
7
Hide whitespace changes
Inline
Side-by-side
Showing
7 changed files
with
55 additions
and
4 deletions
+55
-4
tabs_demo.dart
...on_tests/flutter_gallery/lib/demo/material/tabs_demo.dart
+1
-1
text_field.dart
packages/flutter/lib/src/cupertino/text_field.dart
+2
-0
search.dart
packages/flutter/lib/src/material/search.dart
+6
-0
text_field.dart
packages/flutter/lib/src/material/text_field.dart
+9
-1
text_formatter.dart
packages/flutter/lib/src/services/text_formatter.dart
+11
-2
editable_text.dart
packages/flutter/lib/src/widgets/editable_text.dart
+16
-0
navigator.dart
packages/flutter/lib/src/widgets/navigator.dart
+10
-0
No files found.
dev/integration_tests/flutter_gallery/lib/demo/material/tabs_demo.dart
View file @
896c19a2
...
@@ -14,7 +14,7 @@ const String _kGalleryAssetsPackage = 'flutter_gallery_assets';
...
@@ -14,7 +14,7 @@ const String _kGalleryAssetsPackage = 'flutter_gallery_assets';
class
_Page
{
class
_Page
{
_Page
({
this
.
label
});
_Page
({
this
.
label
});
final
String
label
;
final
String
label
;
String
get
id
=>
label
[
0
]
;
String
get
id
=>
label
.
characters
.
first
;
@override
@override
String
toString
()
=>
'
$runtimeType
("
$label
")'
;
String
toString
()
=>
'
$runtimeType
("
$label
")'
;
}
}
...
...
packages/flutter/lib/src/cupertino/text_field.dart
View file @
896c19a2
...
@@ -131,6 +131,8 @@ class _CupertinoTextFieldSelectionGestureDetectorBuilder extends TextSelectionGe
...
@@ -131,6 +131,8 @@ class _CupertinoTextFieldSelectionGestureDetectorBuilder extends TextSelectionGe
/// field (e.g., by pressing a button on the soft keyboard), the text field
/// field (e.g., by pressing a button on the soft keyboard), the text field
/// calls the [onSubmitted] callback.
/// calls the [onSubmitted] callback.
///
///
/// {@macro flutter.widgets.editableText.complexCharacters}
///
/// To control the text that is displayed in the text field, use the
/// To control the text that is displayed in the text field, use the
/// [controller]. For example, to set the initial value of the text field, use
/// [controller]. For example, to set the initial value of the text field, use
/// a [controller] that already contains some text such as:
/// a [controller] that already contains some text such as:
...
...
packages/flutter/lib/src/material/search.dart
View file @
896c19a2
...
@@ -43,6 +43,9 @@ import 'theme.dart';
...
@@ -43,6 +43,9 @@ import 'theme.dart';
/// the [AppBar.leading] position e.g. from the hamburger menu to the back arrow
/// the [AppBar.leading] position e.g. from the hamburger menu to the back arrow
/// used to exit the search page.
/// used to exit the search page.
///
///
/// ## Handling emojis and other complex characters
/// {@macro flutter.widgets.editableText.complexCharacters}
///
/// See also:
/// See also:
///
///
/// * [SearchDelegate] to define the content of the search page.
/// * [SearchDelegate] to define the content of the search page.
...
@@ -85,6 +88,9 @@ Future<T> showSearch<T>({
...
@@ -85,6 +88,9 @@ Future<T> showSearch<T>({
/// A given [SearchDelegate] can only be associated with one active [showSearch]
/// A given [SearchDelegate] can only be associated with one active [showSearch]
/// call. Call [SearchDelegate.close] before re-using the same delegate instance
/// call. Call [SearchDelegate.close] before re-using the same delegate instance
/// for another [showSearch] call.
/// for another [showSearch] call.
///
/// ## Handling emojis and other complex characters
/// {@macro flutter.widgets.editableText.complexCharacters}
abstract
class
SearchDelegate
<
T
>
{
abstract
class
SearchDelegate
<
T
>
{
/// Constructor to be called by subclasses which may specify [searchFieldLabel], [keyboardType] and/or
/// Constructor to be called by subclasses which may specify [searchFieldLabel], [keyboardType] and/or
...
...
packages/flutter/lib/src/material/text_field.dart
View file @
896c19a2
...
@@ -227,7 +227,7 @@ class _TextFieldSelectionGestureDetectorBuilder extends TextSelectionGestureDete
...
@@ -227,7 +227,7 @@ class _TextFieldSelectionGestureDetectorBuilder extends TextSelectionGestureDete
/// builder: (BuildContext context) {
/// builder: (BuildContext context) {
/// return AlertDialog(
/// return AlertDialog(
/// title: const Text('Thanks!'),
/// title: const Text('Thanks!'),
/// content: Text ('You typed "$value".'),
/// content: Text ('You typed "$value"
, which has length ${value.characters.length}
.'),
/// actions: <Widget>[
/// actions: <Widget>[
/// TextButton(
/// TextButton(
/// onPressed: () { Navigator.pop(context); },
/// onPressed: () { Navigator.pop(context); },
...
@@ -256,6 +256,14 @@ class _TextFieldSelectionGestureDetectorBuilder extends TextSelectionGestureDete
...
@@ -256,6 +256,14 @@ class _TextFieldSelectionGestureDetectorBuilder extends TextSelectionGestureDete
///
///
/// Keep in mind you can also always read the current string from a TextField's
/// Keep in mind you can also always read the current string from a TextField's
/// [TextEditingController] using [TextEditingController.text].
/// [TextEditingController] using [TextEditingController.text].
///
/// ## Handling emojis and other complex characters
/// {@macro flutter.widgets.editableText.complexCharacters}
///
/// In the live Dartpad example above, try typing the emoji 👨👩👦
/// into the field and submitting. Because the example code measures the length
/// with `value.characters.length`, the emoji is correctly counted as a single
/// character.
///
///
/// See also:
/// See also:
///
///
...
...
packages/flutter/lib/src/services/text_formatter.dart
View file @
896c19a2
...
@@ -25,6 +25,9 @@ import 'text_input.dart';
...
@@ -25,6 +25,9 @@ import 'text_input.dart';
/// To create custom formatters, extend the [TextInputFormatter] class and
/// To create custom formatters, extend the [TextInputFormatter] class and
/// implement the [formatEditUpdate] method.
/// implement the [formatEditUpdate] method.
///
///
/// ## Handling emojis and other complex characters
/// {@macro flutter.widgets.editableText.complexCharacters}
///
/// See also:
/// See also:
///
///
/// * [EditableText] on which the formatting apply.
/// * [EditableText] on which the formatting apply.
...
@@ -300,13 +303,19 @@ class WhitelistingTextInputFormatter extends FilteringTextInputFormatter {
...
@@ -300,13 +303,19 @@ class WhitelistingTextInputFormatter extends FilteringTextInputFormatter {
}
}
/// A [TextInputFormatter] that prevents the insertion of more characters
/// A [TextInputFormatter] that prevents the insertion of more characters
///
(currently defined as Unicode scalar values)
than allowed.
/// than allowed.
///
///
/// Since this formatter only prevents new characters from being added to the
/// Since this formatter only prevents new characters from being added to the
/// text, it preserves the existing [TextEditingValue.selection].
/// text, it preserves the existing [TextEditingValue.selection].
///
///
/// Characters are counted as user-perceived characters using the
/// [characters](https://pub.dev/packages/characters) package, so even complex
/// characters like extended grapheme clusters and surrogate pairs are counted
/// as single characters.
///
/// See also:
/// * [maxLength], which discusses the precise meaning of "number of
/// * [maxLength], which discusses the precise meaning of "number of
/// characters"
and how it may differ from the intuitive meaning
.
/// characters".
class
LengthLimitingTextInputFormatter
extends
TextInputFormatter
{
class
LengthLimitingTextInputFormatter
extends
TextInputFormatter
{
/// Creates a formatter that prevents the insertion of more characters than a
/// Creates a formatter that prevents the insertion of more characters than a
/// limit.
/// limit.
...
...
packages/flutter/lib/src/widgets/editable_text.dart
View file @
896c19a2
...
@@ -946,6 +946,22 @@ class EditableText extends StatefulWidget {
...
@@ -946,6 +946,22 @@ class EditableText extends StatefulWidget {
/// {@end-tool}
/// {@end-tool}
/// {@endtemplate}
/// {@endtemplate}
///
///
/// ## Handling emojis and other complex characters
/// {@template flutter.widgets.editableText.complexCharacters}
/// It's important to always use
/// [characters](https://pub.dev/packages/characters) when dealing with user
/// input text that may contain complex characters. This will ensure that
/// extended grapheme clusters and surrogate pairs are treated as single
/// characters, as they appear to the user.
///
/// For example, when finding the length of some user input, use
/// `string.characters.length`. Do NOT use `string.length` or even
/// `string.runes.length`. For the complex character "👨👩👦", this
/// appears to the user as a single character, and `string.characters.length`
/// intuitively returns 1. On the other hand, `string.length` returns 8, and
/// `string.runes.length` returns 5!
/// {@endtemplate}
///
/// See also:
/// See also:
///
///
/// * [inputFormatters], which are called before [onChanged]
/// * [inputFormatters], which are called before [onChanged]
...
...
packages/flutter/lib/src/widgets/navigator.dart
View file @
896c19a2
...
@@ -1571,6 +1571,16 @@ class Navigator extends StatefulWidget {
...
@@ -1571,6 +1571,16 @@ class Navigator extends StatefulWidget {
/// The [NavigatorState] and [initialRoute] will be passed to the callback.
/// The [NavigatorState] and [initialRoute] will be passed to the callback.
/// The callback must return a list of [Route] objects with which the history
/// The callback must return a list of [Route] objects with which the history
/// will be primed.
/// will be primed.
///
/// When parsing the initialRoute, if there's any chance that the it may
/// contain complex characters, it's best to use the
/// [characters](https://pub.dev/packages/characters) API. This will ensure
/// that extended grapheme clusters and surrogate pairs are treated as single
/// characters by the code, the same way that they appear to the user. For
/// example, the string "👨👩👦" appears to the user as a single
/// character and `string.characters.length` intuitively returns 1. On the
/// other hand, `string.length` returns 8, and `string.runes.length` returns
/// 5!
final
RouteListFactory
onGenerateInitialRoutes
;
final
RouteListFactory
onGenerateInitialRoutes
;
/// Whether this navigator should report route update message back to the
/// Whether this navigator should report route update message back to the
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment