Support password fields for a11y (#15497)
* Support password fields for a11y
* rename to obscured
* Roll engine to c3ab0c9143029f0267a05b99effbfbd280a4901b
diff --git a/bin/internal/engine.version b/bin/internal/engine.version
index 4052075..2bae9f9 100644
--- a/bin/internal/engine.version
+++ b/bin/internal/engine.version
@@ -1 +1 @@
-1348ab5b63adc18148f161876a4b1cacd5ec0779
+c3ab0c9143029f0267a05b99effbfbd280a4901b
diff --git a/packages/flutter/lib/src/rendering/custom_paint.dart b/packages/flutter/lib/src/rendering/custom_paint.dart
index 5bd9f42..1b6b6d0 100644
--- a/packages/flutter/lib/src/rendering/custom_paint.dart
+++ b/packages/flutter/lib/src/rendering/custom_paint.dart
@@ -831,6 +831,9 @@
if (properties.inMutuallyExclusiveGroup != null) {
config.isInMutuallyExclusiveGroup = properties.inMutuallyExclusiveGroup;
}
+ if (properties.obscured != null) {
+ config.isObscured = properties.obscured;
+ }
if (properties.header != null) {
config.isHeader = properties.header;
}
diff --git a/packages/flutter/lib/src/rendering/editable.dart b/packages/flutter/lib/src/rendering/editable.dart
index 7cd6b63..2e62705 100644
--- a/packages/flutter/lib/src/rendering/editable.dart
+++ b/packages/flutter/lib/src/rendering/editable.dart
@@ -132,12 +132,14 @@
this.onSelectionChanged,
this.onCaretChanged,
this.ignorePointer: false,
+ bool obscureText: false,
}) : assert(textAlign != null),
assert(textDirection != null, 'RenderEditable created without a textDirection.'),
assert(maxLines == null || maxLines > 0),
assert(textScaleFactor != null),
assert(offset != null),
assert(ignorePointer != null),
+ assert(obscureText != null),
_textPainter = new TextPainter(
text: text,
textAlign: textAlign,
@@ -150,7 +152,8 @@
_maxLines = maxLines,
_selectionColor = selectionColor,
_selection = selection,
- _offset = offset {
+ _offset = offset,
+ _obscureText = obscureText {
assert(_showCursor != null);
assert(!_showCursor.value || cursorColor != null);
_tap = new TapGestureRecognizer(debugOwner: this)
@@ -160,6 +163,9 @@
..onLongPress = _handleLongPress;
}
+ /// Character used to obscure text if [obscureText] is true.
+ static const String obscuringCharacter = '•';
+
/// Called when the selection changes.
SelectionChangedHandler onSelectionChanged;
@@ -175,6 +181,16 @@
/// The default value of this property is false.
bool ignorePointer;
+ /// Whether to hide the text being edited (e.g., for passwords).
+ bool get obscureText => _obscureText;
+ bool _obscureText;
+ set obscureText(bool value) {
+ if (_obscureText == value)
+ return;
+ _obscureText = value;
+ markNeedsSemanticsUpdate();
+ }
+
Rect _lastCaretRect;
/// Marks the render object as needing to be laid out again and have its text
@@ -351,7 +367,10 @@
super.describeSemanticsConfiguration(config);
config
- ..value = text.toPlainText()
+ ..value = obscureText
+ ? obscuringCharacter * text.toPlainText().length
+ : text.toPlainText()
+ ..isObscured = obscureText
..textDirection = textDirection
..isFocused = hasFocus
..isTextField = true;
diff --git a/packages/flutter/lib/src/rendering/proxy_box.dart b/packages/flutter/lib/src/rendering/proxy_box.dart
index f5c499a..67ae971 100644
--- a/packages/flutter/lib/src/rendering/proxy_box.dart
+++ b/packages/flutter/lib/src/rendering/proxy_box.dart
@@ -3019,6 +3019,7 @@
bool textField,
bool focused,
bool inMutuallyExclusiveGroup,
+ bool obscured,
String label,
String value,
String increasedValue,
@@ -3053,6 +3054,7 @@
_textField = textField,
_focused = focused,
_inMutuallyExclusiveGroup = inMutuallyExclusiveGroup,
+ _obscured = obscured,
_label = label,
_value = value,
_increasedValue = increasedValue,
@@ -3201,6 +3203,17 @@
markNeedsSemanticsUpdate();
}
+ /// If non-null, sets the [SemanticsNode.isObscured] semantic to the given
+ /// value.
+ bool get obscured => _obscured;
+ bool _obscured;
+ set obscured(bool value) {
+ if (obscured == value)
+ return;
+ _obscured = value;
+ markNeedsSemanticsUpdate();
+ }
+
/// If non-null, sets the [SemanticsNode.label] semantic to the given value.
///
/// The reading direction is given by [textDirection].
@@ -3638,6 +3651,8 @@
config.isFocused = focused;
if (inMutuallyExclusiveGroup != null)
config.isInMutuallyExclusiveGroup = inMutuallyExclusiveGroup;
+ if (obscured != null)
+ config.isObscured = obscured;
if (label != null)
config.label = label;
if (value != null)
diff --git a/packages/flutter/lib/src/semantics/semantics.dart b/packages/flutter/lib/src/semantics/semantics.dart
index 09fabd1..e2ea34c 100644
--- a/packages/flutter/lib/src/semantics/semantics.dart
+++ b/packages/flutter/lib/src/semantics/semantics.dart
@@ -319,6 +319,7 @@
this.textField,
this.focused,
this.inMutuallyExclusiveGroup,
+ this.obscured,
this.label,
this.value,
this.increasedValue,
@@ -398,6 +399,13 @@
/// For example, a radio button is in a mutually exclusive group because only
/// one radio button in that group can be marked as [checked].
final bool inMutuallyExclusiveGroup;
+
+ /// If non-null, whether [value] should be obscured.
+ ///
+ /// This option is usually set in combination with [textField] to indicate
+ /// that the text field contains a password (or other sensitive information).
+ /// Doing so instructs screen readers to not read out the [value].
+ final bool obscured;
/// Provides a textual description of the widget.
///
@@ -2405,6 +2413,16 @@
_setFlag(SemanticsFlag.isTextField, value);
}
+ /// Whether the [value] should be obscured.
+ ///
+ /// This option is usually set in combination with [textField] to indicate
+ /// that the text field contains a password (or other sensitive information).
+ /// Doing so instructs screen readers to not read out the [value].
+ bool get isObscured => _hasFlag(SemanticsFlag.isObscured);
+ set isObscured(bool value) {
+ _setFlag(SemanticsFlag.isObscured, value);
+ }
+
/// The currently selected text (or the position of the cursor) within [value]
/// if this node represents a text field.
TextSelection get textSelection => _textSelection;
diff --git a/packages/flutter/lib/src/widgets/basic.dart b/packages/flutter/lib/src/widgets/basic.dart
index ce383b4..948bde4 100644
--- a/packages/flutter/lib/src/widgets/basic.dart
+++ b/packages/flutter/lib/src/widgets/basic.dart
@@ -4892,6 +4892,7 @@
bool textField,
bool focused,
bool inMutuallyExclusiveGroup,
+ bool obscured,
String label,
String value,
String increasedValue,
@@ -4929,6 +4930,7 @@
textField: textField,
focused: focused,
inMutuallyExclusiveGroup: inMutuallyExclusiveGroup,
+ obscured: obscured,
label: label,
value: value,
increasedValue: increasedValue,
@@ -5007,6 +5009,7 @@
textField: properties.textField,
focused: properties.focused,
inMutuallyExclusiveGroup: properties.inMutuallyExclusiveGroup,
+ obscured: properties.obscured,
label: properties.label,
value: properties.value,
increasedValue: properties.increasedValue,
diff --git a/packages/flutter/lib/src/widgets/editable_text.dart b/packages/flutter/lib/src/widgets/editable_text.dart
index fd820f3..b77e837 100644
--- a/packages/flutter/lib/src/widgets/editable_text.dart
+++ b/packages/flutter/lib/src/widgets/editable_text.dart
@@ -759,6 +759,7 @@
onSelectionChanged: onSelectionChanged,
onCaretChanged: onCaretChanged,
ignorePointer: rendererIgnoresPointer,
+ obscureText: obscureText,
);
}
@@ -778,7 +779,8 @@
..offset = offset
..onSelectionChanged = onSelectionChanged
..onCaretChanged = onCaretChanged
- ..ignorePointer = rendererIgnoresPointer;
+ ..ignorePointer = rendererIgnoresPointer
+ ..obscureText = obscureText;
}
TextSpan get _styledTextSpan {
@@ -801,7 +803,7 @@
String text = value.text;
if (obscureText) {
- text = new String.fromCharCodes(new List<int>.filled(text.length, 0x2022));
+ text = RenderEditable.obscuringCharacter * text.length;
final int o = obscureShowCharacterAtIndex;
if (o != null && o >= 0 && o < text.length)
text = text.replaceRange(o, o + 1, value.text.substring(o, o + 1));
diff --git a/packages/flutter/test/widgets/custom_painter_test.dart b/packages/flutter/test/widgets/custom_painter_test.dart
index 5b114ab..4cea57d 100644
--- a/packages/flutter/test/widgets/custom_painter_test.dart
+++ b/packages/flutter/test/widgets/custom_painter_test.dart
@@ -415,6 +415,7 @@
focused: true,
inMutuallyExclusiveGroup: true,
header: true,
+ obscured: true,
),
),
),
diff --git a/packages/flutter/test/widgets/editable_text_test.dart b/packages/flutter/test/widgets/editable_text_test.dart
index 006ea14..ea17a81 100644
--- a/packages/flutter/test/widgets/editable_text_test.dart
+++ b/packages/flutter/test/widgets/editable_text_test.dart
@@ -575,6 +575,38 @@
semantics.dispose();
});
+ testWidgets('password fields have correct semantics', (WidgetTester tester) async {
+ final SemanticsTester semantics = new SemanticsTester(tester);
+
+ controller.text = 'super-secret-password!!1';
+
+ await tester.pumpWidget(new MaterialApp(
+ home: new EditableText(
+ obscureText: true,
+ controller: controller,
+ focusNode: focusNode,
+ style: textStyle,
+ cursorColor: cursorColor,
+ ),
+ ));
+
+ final String expectedValue = '•' * controller.text.length;
+
+ expect(semantics, hasSemantics(new TestSemantics(
+ children: <TestSemantics>[
+ new TestSemantics(
+ flags: <SemanticsFlag>[SemanticsFlag.isTextField, SemanticsFlag.isObscured],
+ value: expectedValue,
+ textDirection: TextDirection.ltr,
+ nextNodeId: -1,
+ previousNodeId: -1,
+ ),
+ ],
+ ), ignoreTransform: true, ignoreRect: true, ignoreId: true));
+
+ semantics.dispose();
+ });
+
group('a11y copy/cut/paste', () {
Future<Null> _buildApp(MockTextSelectionControls controls, WidgetTester tester) {
return tester.pumpWidget(new MaterialApp(
diff --git a/packages/flutter/test/widgets/semantics_test.dart b/packages/flutter/test/widgets/semantics_test.dart
index e31159f..cdaa8c5 100644
--- a/packages/flutter/test/widgets/semantics_test.dart
+++ b/packages/flutter/test/widgets/semantics_test.dart
@@ -470,6 +470,7 @@
focused: true,
inMutuallyExclusiveGroup: true,
header: true,
+ obscured: true,
)
);