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
d3a8b035
Unverified
Commit
d3a8b035
authored
Oct 27, 2020
by
Daniel Edrisian
Committed by
GitHub
Oct 27, 2020
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Adaptive TextField (#68918)
parent
9e5e763e
Changes
2
Hide whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
395 additions
and
3 deletions
+395
-3
text_field.dart
packages/flutter/lib/src/material/text_field.dart
+194
-3
text_field_test.dart
packages/flutter/test/material/text_field_test.dart
+201
-0
No files found.
packages/flutter/lib/src/material/text_field.dart
View file @
d3a8b035
...
...
@@ -25,6 +25,8 @@ import 'theme.dart';
export
'package:flutter/services.dart'
show
TextInputType
,
TextInputAction
,
TextCapitalization
,
SmartQuotesType
,
SmartDashesType
;
enum
_TextFieldType
{
material
,
adaptive
}
/// Signature for the [TextField.buildCounter] callback.
typedef
InputCounterWidgetBuilder
=
Widget
?
Function
(
/// The build context for the TextField.
...
...
@@ -380,7 +382,8 @@ class TextField extends StatefulWidget {
this
.
scrollPhysics
,
this
.
autofillHints
,
this
.
restorationId
,
})
:
assert
(
textAlign
!=
null
),
})
:
_textFieldType
=
_TextFieldType
.
material
,
assert
(
textAlign
!=
null
),
assert
(
readOnly
!=
null
),
assert
(
autofocus
!=
null
),
assert
(
obscuringCharacter
!=
null
&&
obscuringCharacter
.
length
==
1
),
...
...
@@ -427,6 +430,119 @@ class TextField extends StatefulWidget {
)),
super
(
key:
key
);
/// Creates a [CupertinoTextField] if the target platform is iOS, creates a
/// material design text field otherwise.
///
/// To retain the standard look of [CupertinoTextField], this constructor only
/// uses the [decoration] property's [decoration.hintText],
/// [decoration.hintStyle], [decoration.suffix], and [decoration.prefix] for
/// the iOS platform, and all else is ignored in support of the default iOS
/// look. For instance, the [decoration.border] cannot override the default
/// iOS-style border.
///
/// The target platform is based on the current [Theme]: [ThemeData.platform].
const
TextField
.
adaptive
({
Key
?
key
,
this
.
controller
,
this
.
focusNode
,
this
.
decoration
=
const
InputDecoration
(),
TextInputType
?
keyboardType
,
this
.
textInputAction
,
this
.
textCapitalization
=
TextCapitalization
.
none
,
this
.
style
,
this
.
strutStyle
,
this
.
textAlign
=
TextAlign
.
start
,
this
.
textAlignVertical
,
this
.
textDirection
,
this
.
readOnly
=
false
,
ToolbarOptions
?
toolbarOptions
,
this
.
showCursor
,
this
.
autofocus
=
false
,
this
.
obscuringCharacter
=
'•'
,
this
.
obscureText
=
false
,
this
.
autocorrect
=
true
,
SmartDashesType
?
smartDashesType
,
SmartQuotesType
?
smartQuotesType
,
this
.
enableSuggestions
=
true
,
this
.
maxLines
=
1
,
this
.
minLines
,
this
.
expands
=
false
,
this
.
maxLength
,
this
.
maxLengthEnforced
=
true
,
this
.
onChanged
,
this
.
onEditingComplete
,
this
.
onSubmitted
,
this
.
onAppPrivateCommand
,
this
.
inputFormatters
,
this
.
enabled
,
this
.
cursorWidth
=
2.0
,
this
.
cursorHeight
,
this
.
cursorRadius
,
this
.
cursorColor
,
this
.
selectionControls
,
this
.
selectionHeightStyle
=
ui
.
BoxHeightStyle
.
tight
,
this
.
selectionWidthStyle
=
ui
.
BoxWidthStyle
.
tight
,
this
.
keyboardAppearance
,
this
.
scrollPadding
=
const
EdgeInsets
.
all
(
20.0
),
this
.
dragStartBehavior
=
DragStartBehavior
.
start
,
this
.
enableInteractiveSelection
=
true
,
this
.
onTap
,
this
.
mouseCursor
,
this
.
buildCounter
,
this
.
scrollController
,
this
.
scrollPhysics
,
this
.
autofillHints
,
this
.
restorationId
,
})
:
_textFieldType
=
_TextFieldType
.
adaptive
,
assert
(
textAlign
!=
null
),
assert
(
readOnly
!=
null
),
assert
(
autofocus
!=
null
),
assert
(
obscuringCharacter
!=
null
&&
obscuringCharacter
.
length
==
1
),
assert
(
obscureText
!=
null
),
assert
(
autocorrect
!=
null
),
smartDashesType
=
smartDashesType
??
(
obscureText
?
SmartDashesType
.
disabled
:
SmartDashesType
.
enabled
),
smartQuotesType
=
smartQuotesType
??
(
obscureText
?
SmartQuotesType
.
disabled
:
SmartQuotesType
.
enabled
),
assert
(
enableSuggestions
!=
null
),
assert
(
enableInteractiveSelection
!=
null
),
assert
(
maxLengthEnforced
!=
null
),
assert
(
scrollPadding
!=
null
),
assert
(
dragStartBehavior
!=
null
),
assert
(
selectionHeightStyle
!=
null
),
assert
(
selectionWidthStyle
!=
null
),
assert
(
maxLines
==
null
||
maxLines
>
0
),
assert
(
minLines
==
null
||
minLines
>
0
),
assert
(
(
maxLines
==
null
)
||
(
minLines
==
null
)
||
(
maxLines
>=
minLines
),
"minLines can't be greater than maxLines"
,
),
assert
(
expands
!=
null
),
assert
(
!
expands
||
(
maxLines
==
null
&&
minLines
==
null
),
'minLines and maxLines must be null when expands is true.'
,
),
assert
(!
obscureText
||
maxLines
==
1
,
'Obscured fields cannot be multiline.'
),
assert
(
maxLength
==
null
||
maxLength
==
TextField
.
noMaxLength
||
maxLength
>
0
),
// Assert the following instead of setting it directly to avoid surprising the user by silently changing the value they set.
assert
(!
identical
(
textInputAction
,
TextInputAction
.
newline
)
||
maxLines
==
1
||
!
identical
(
keyboardType
,
TextInputType
.
text
),
'Use keyboardType TextInputType.multiline when using TextInputAction.newline on a multiline TextField.'
),
keyboardType
=
keyboardType
??
(
maxLines
==
1
?
TextInputType
.
text
:
TextInputType
.
multiline
),
toolbarOptions
=
toolbarOptions
??
(
obscureText
?
const
ToolbarOptions
(
selectAll:
true
,
paste:
true
,
)
:
const
ToolbarOptions
(
copy:
true
,
cut:
true
,
selectAll:
true
,
paste:
true
,
)),
super
(
key:
key
);
final
_TextFieldType
_textFieldType
;
/// Controls the text being edited.
///
/// If null, this widget will create its own [TextEditingController].
...
...
@@ -1076,8 +1192,60 @@ class _TextFieldState extends State<TextField> with RestorationMixin implements
}
}
@override
Widget
build
(
BuildContext
context
)
{
Widget
_buildCupertinoTextField
(
BuildContext
context
)
{
return
CupertinoTextField
(
key:
widget
.
key
,
controller:
widget
.
controller
,
focusNode:
widget
.
focusNode
,
placeholder:
widget
.
decoration
?.
hintText
,
placeholderStyle:
widget
.
decoration
?.
hintStyle
??
const
TextStyle
(
fontWeight:
FontWeight
.
w400
,
color:
CupertinoColors
.
placeholderText
),
prefix:
widget
.
decoration
?.
prefix
,
suffix:
widget
.
decoration
?.
suffix
,
keyboardType:
widget
.
keyboardType
,
textInputAction:
widget
.
textInputAction
,
textCapitalization:
widget
.
textCapitalization
,
style:
widget
.
style
,
strutStyle:
widget
.
strutStyle
,
textAlign:
widget
.
textAlign
,
textAlignVertical:
widget
.
textAlignVertical
,
readOnly:
widget
.
readOnly
,
toolbarOptions:
widget
.
toolbarOptions
,
showCursor:
widget
.
showCursor
,
autofocus:
widget
.
autofocus
,
obscuringCharacter:
widget
.
obscuringCharacter
,
obscureText:
widget
.
obscureText
,
autocorrect:
widget
.
autocorrect
,
smartDashesType:
widget
.
smartDashesType
,
smartQuotesType:
widget
.
smartQuotesType
,
enableSuggestions:
widget
.
enableSuggestions
,
maxLines:
widget
.
maxLines
,
minLines:
widget
.
minLines
,
expands:
widget
.
expands
,
maxLength:
widget
.
maxLength
,
maxLengthEnforced:
widget
.
maxLengthEnforced
,
onChanged:
widget
.
onChanged
,
onEditingComplete:
widget
.
onEditingComplete
,
onSubmitted:
widget
.
onSubmitted
,
inputFormatters:
widget
.
inputFormatters
,
enabled:
widget
.
enabled
,
cursorWidth:
widget
.
cursorWidth
,
cursorHeight:
widget
.
cursorHeight
,
cursorColor:
widget
.
cursorColor
,
selectionHeightStyle:
widget
.
selectionHeightStyle
,
selectionWidthStyle:
widget
.
selectionWidthStyle
,
keyboardAppearance:
widget
.
keyboardAppearance
,
scrollPadding:
widget
.
scrollPadding
,
dragStartBehavior:
widget
.
dragStartBehavior
,
enableInteractiveSelection:
widget
.
enableInteractiveSelection
,
onTap:
widget
.
onTap
,
scrollController:
widget
.
scrollController
,
scrollPhysics:
widget
.
scrollPhysics
,
autofillHints:
widget
.
autofillHints
,
restorationId:
widget
.
restorationId
,
);
}
Widget
_buildMaterialTextField
(
BuildContext
context
)
{
assert
(
debugCheckHasMaterial
(
context
));
assert
(
debugCheckHasMaterialLocalizations
(
context
));
assert
(
debugCheckHasDirectionality
(
context
));
...
...
@@ -1255,4 +1423,27 @@ class _TextFieldState extends State<TextField> with RestorationMixin implements
),
);
}
@override
Widget
build
(
BuildContext
context
)
{
switch
(
widget
.
_textFieldType
)
{
case
_TextFieldType
.
material
:
return
_buildMaterialTextField
(
context
);
case
_TextFieldType
.
adaptive
:
{
final
ThemeData
theme
=
Theme
.
of
(
context
)!;
assert
(
theme
.
platform
!=
null
);
switch
(
theme
.
platform
)
{
case
TargetPlatform
.
iOS
:
case
TargetPlatform
.
macOS
:
return
_buildCupertinoTextField
(
context
);
case
TargetPlatform
.
android
:
case
TargetPlatform
.
fuchsia
:
case
TargetPlatform
.
linux
:
case
TargetPlatform
.
windows
:
return
_buildMaterialTextField
(
context
);
}
}
}
}
}
packages/flutter/test/material/text_field_test.dart
View file @
d3a8b035
...
...
@@ -8528,4 +8528,205 @@ void main() {
expect
(
inputWidth
,
wideWidth
);
expect
(
cursorRight
,
inputWidth
-
kCaretGap
);
});
testWidgets
(
'Adaptive TextField displays CupertinoTextField in iOS'
,
(
WidgetTester
tester
)
async
{
await
tester
.
pumpWidget
(
const
MaterialApp
(
home:
Scaffold
(
body:
Material
(
child:
TextField
.
adaptive
(),
),
),
),
);
expect
(
find
.
byType
(
CupertinoTextField
),
findsOneWidget
);
},
variant:
const
TargetPlatformVariant
(<
TargetPlatform
>
{
TargetPlatform
.
iOS
,
TargetPlatform
.
macOS
,
})
);
testWidgets
(
'Adaptive TextField does not display CupertinoTextField in non-iOS'
,
(
WidgetTester
tester
)
async
{
await
tester
.
pumpWidget
(
const
MaterialApp
(
home:
Scaffold
(
body:
Material
(
child:
TextField
.
adaptive
(),
),
),
),
);
expect
(
find
.
byType
(
CupertinoTextField
),
findsNothing
);
},
variant:
const
TargetPlatformVariant
(<
TargetPlatform
>
{
TargetPlatform
.
android
,
TargetPlatform
.
fuchsia
,
TargetPlatform
.
windows
,
TargetPlatform
.
linux
,
}),
);
testWidgets
(
'Adaptive TextField in iOS with specified hintText'
,
(
WidgetTester
tester
)
async
{
await
tester
.
pumpWidget
(
const
MaterialApp
(
home:
Scaffold
(
body:
Material
(
child:
TextField
.
adaptive
(
decoration:
InputDecoration
(
hintText:
'Hint'
,
),
),
),
),
),
);
expect
(
find
.
text
(
'Hint'
),
findsOneWidget
);
},
variant:
const
TargetPlatformVariant
(<
TargetPlatform
>
{
TargetPlatform
.
iOS
,
TargetPlatform
.
macOS
,
})
);
testWidgets
(
'Adaptive TextField in iOS cannot override iOS-specific decoration border'
,
(
WidgetTester
tester
)
async
{
final
BorderRadius
borderRadius
=
BorderRadius
.
circular
(
0
);
await
tester
.
pumpWidget
(
MaterialApp
(
home:
Scaffold
(
body:
Material
(
child:
TextField
.
adaptive
(
decoration:
InputDecoration
(
hintText:
'Hint'
,
border:
OutlineInputBorder
(
borderRadius:
borderRadius
,
),
),
),
),
),
),
);
final
CupertinoTextField
textField
=
tester
.
widget
(
find
.
byType
(
CupertinoTextField
));
expect
(
textField
.
decoration
!.
borderRadius
!=
borderRadius
,
isTrue
);
},
variant:
const
TargetPlatformVariant
(<
TargetPlatform
>
{
TargetPlatform
.
iOS
,
TargetPlatform
.
macOS
,
})
);
testWidgets
(
'Adaptive TextField in non-iOS can override decoration border'
,
(
WidgetTester
tester
)
async
{
final
OutlineInputBorder
border
=
OutlineInputBorder
(
borderRadius:
BorderRadius
.
circular
(
0
),
);
await
tester
.
pumpWidget
(
MaterialApp
(
home:
Scaffold
(
body:
Material
(
child:
TextField
.
adaptive
(
decoration:
InputDecoration
(
hintText:
'Hint'
,
border:
border
,
),
),
),
),
),
);
final
TextField
textField
=
tester
.
widget
(
find
.
byType
(
TextField
));
expect
(
textField
.
decoration
!.
border
,
border
);
},
variant:
const
TargetPlatformVariant
(<
TargetPlatform
>
{
TargetPlatform
.
android
,
TargetPlatform
.
fuchsia
,
TargetPlatform
.
windows
,
TargetPlatform
.
linux
,
})
);
testWidgets
(
'Adaptive TextField in iOS with specified hintStyle'
,
(
WidgetTester
tester
)
async
{
final
TextStyle
hintStyle
=
TextStyle
(
inherit:
false
,
color:
Colors
.
pink
[
500
],
fontSize:
10.0
,
);
await
tester
.
pumpWidget
(
MaterialApp
(
home:
Scaffold
(
body:
Material
(
child:
TextField
.
adaptive
(
decoration:
InputDecoration
(
hintText:
'Hint'
,
hintStyle:
hintStyle
,
),
),
),
),
),
);
final
Text
hintText
=
tester
.
widget
(
find
.
text
(
'Hint'
));
expect
(
hintText
.
style
,
hintStyle
);
},
variant:
const
TargetPlatformVariant
(<
TargetPlatform
>
{
TargetPlatform
.
iOS
,
TargetPlatform
.
macOS
,
})
);
testWidgets
(
'Adaptive TextField in iOS with custom text style'
,
(
WidgetTester
tester
)
async
{
final
TextStyle
style
=
TextStyle
(
color:
Colors
.
pink
[
500
],
fontSize:
2.0
,
);
await
tester
.
pumpWidget
(
overlay
(
child:
TextField
.
adaptive
(
controller:
TextEditingController
(
text:
'Text'
),
style:
style
,
),
),
);
final
EditableText
text
=
tester
.
widget
(
find
.
text
(
'Text'
));
expect
(
text
.
style
.
color
,
style
.
color
);
expect
(
text
.
style
.
fontSize
,
style
.
fontSize
);
},
variant:
const
TargetPlatformVariant
(<
TargetPlatform
>
{
TargetPlatform
.
iOS
,
TargetPlatform
.
macOS
,
})
);
testWidgets
(
'Adaptive TextField in iOS with suffix'
,
(
WidgetTester
tester
)
async
{
await
tester
.
pumpWidget
(
overlay
(
child:
TextField
.
adaptive
(
controller:
TextEditingController
(
text:
'Text'
),
decoration:
const
InputDecoration
(
suffix:
Icon
(
Icons
.
phone
),
prefix:
Icon
(
Icons
.
message
),
),
),
),
);
expect
(
find
.
byIcon
(
Icons
.
phone
),
findsOneWidget
);
expect
(
find
.
byIcon
(
Icons
.
message
),
findsOneWidget
);
},
variant:
const
TargetPlatformVariant
(<
TargetPlatform
>
{
TargetPlatform
.
iOS
,
TargetPlatform
.
macOS
,
})
);
}
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