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
a47b3ccc
Unverified
Commit
a47b3ccc
authored
Aug 11, 2022
by
stuartmorgan
Committed by
GitHub
Aug 11, 2022
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Fix Android platform view creation flow (#109232)
parent
e3b42e5e
Changes
5
Hide whitespace changes
Inline
Side-by-side
Showing
5 changed files
with
181 additions
and
42 deletions
+181
-42
platform_view.dart
packages/flutter/lib/src/rendering/platform_view.dart
+3
-11
platform_views.dart
packages/flutter/lib/src/services/platform_views.dart
+50
-24
platform_view.dart
packages/flutter/lib/src/widgets/platform_view.dart
+10
-4
fake_platform_views.dart
packages/flutter/test/services/fake_platform_views.dart
+13
-2
platform_view_test.dart
packages/flutter/test/widgets/platform_view_test.dart
+105
-1
No files found.
packages/flutter/lib/src/rendering/platform_view.dart
View file @
a47b3ccc
...
@@ -179,17 +179,9 @@ class RenderAndroidView extends PlatformViewRenderBox {
...
@@ -179,17 +179,9 @@ class RenderAndroidView extends PlatformViewRenderBox {
Size
targetSize
;
Size
targetSize
;
do
{
do
{
targetSize
=
size
;
targetSize
=
size
;
if
(
_viewController
.
isCreated
)
{
_currentTextureSize
=
await
_viewController
.
setSize
(
targetSize
);
_currentTextureSize
=
await
_viewController
.
setSize
(
targetSize
);
if
(
_isDisposed
)
{
if
(
_isDisposed
)
{
return
;
return
;
}
}
else
{
await
_viewController
.
create
(
size:
targetSize
);
if
(
_isDisposed
)
{
return
;
}
_currentTextureSize
=
targetSize
;
}
}
// We've resized the platform view to targetSize, but it is possible that
// We've resized the platform view to targetSize, but it is possible that
// while we were resizing the render object's size was changed again.
// while we were resizing the render object's size was changed again.
...
...
packages/flutter/lib/src/services/platform_views.dart
View file @
a47b3ccc
...
@@ -764,17 +764,33 @@ abstract class AndroidViewController extends PlatformViewController {
...
@@ -764,17 +764,33 @@ abstract class AndroidViewController extends PlatformViewController {
Future
<
void
>
_sendDisposeMessage
();
Future
<
void
>
_sendDisposeMessage
();
/// Sends the message to create the platform view with an initial [size].
/// Sends the message to create the platform view with an initial [size].
Future
<
void
>
_sendCreateMessage
({
Size
?
size
});
///
/// Returns true if the view was actually created. In some cases (e.g.,
/// trying to create a texture-based view with a null size) creation will
/// fail and need to be re-attempted later.
Future
<
bool
>
_sendCreateMessage
({
Size
?
size
});
/// Sends the message to resize the platform view to [size].
Future
<
Size
>
_sendResizeMessage
(
Size
size
);
@override
bool
get
awaitingCreation
=>
_state
==
_AndroidViewState
.
waitingForSize
;
@override
@override
Future
<
void
>
create
({
Size
?
size
})
async
{
Future
<
void
>
create
({
Size
?
size
})
async
{
assert
(
_state
!=
_AndroidViewState
.
disposed
,
'trying to create a disposed Android view'
);
assert
(
_state
!=
_AndroidViewState
.
disposed
,
'trying to create a disposed Android view'
);
await
_sendCreateMessage
(
size:
size
);
assert
(
_state
==
_AndroidViewState
.
waitingForSize
,
'Android view is already sized. View id:
$viewId
'
);
_state
=
_AndroidViewState
.
creating
;
_state
=
_AndroidViewState
.
created
;
final
bool
created
=
await
_sendCreateMessage
(
size:
size
);
for
(
final
PlatformViewCreatedCallback
callback
in
_platformViewCreatedCallbacks
)
{
callback
(
viewId
);
if
(
created
)
{
_state
=
_AndroidViewState
.
created
;
for
(
final
PlatformViewCreatedCallback
callback
in
_platformViewCreatedCallbacks
)
{
callback
(
viewId
);
}
}
else
{
_state
=
_AndroidViewState
.
waitingForSize
;
}
}
}
}
...
@@ -792,7 +808,17 @@ abstract class AndroidViewController extends PlatformViewController {
...
@@ -792,7 +808,17 @@ abstract class AndroidViewController extends PlatformViewController {
///
///
/// As a result, consumers are expected to clip the texture using [size], while using
/// As a result, consumers are expected to clip the texture using [size], while using
/// the return value to size the texture.
/// the return value to size the texture.
Future
<
Size
>
setSize
(
Size
size
);
Future
<
Size
>
setSize
(
Size
size
)
async
{
assert
(
_state
!=
_AndroidViewState
.
disposed
,
'Android view is disposed. View id:
$viewId
'
);
if
(
_state
==
_AndroidViewState
.
waitingForSize
)
{
// Either `create` hasn't been called, or it couldn't run due to missing
// size information, so create the view now.
await
create
(
size:
size
);
return
size
;
}
else
{
return
_sendResizeMessage
(
size
);
}
}
/// Sets the offset of the platform view.
/// Sets the offset of the platform view.
///
///
...
@@ -972,7 +998,7 @@ class ExpensiveAndroidViewController extends AndroidViewController {
...
@@ -972,7 +998,7 @@ class ExpensiveAndroidViewController extends AndroidViewController {
})
:
super
.
_
();
})
:
super
.
_
();
@override
@override
Future
<
void
>
_sendCreateMessage
({
Size
?
size
})
{
Future
<
bool
>
_sendCreateMessage
({
Size
?
size
})
async
{
final
Map
<
String
,
dynamic
>
args
=
<
String
,
dynamic
>{
final
Map
<
String
,
dynamic
>
args
=
<
String
,
dynamic
>{
'id'
:
viewId
,
'id'
:
viewId
,
'viewType'
:
_viewType
,
'viewType'
:
_viewType
,
...
@@ -988,7 +1014,8 @@ class ExpensiveAndroidViewController extends AndroidViewController {
...
@@ -988,7 +1014,8 @@ class ExpensiveAndroidViewController extends AndroidViewController {
paramsByteData
.
lengthInBytes
,
paramsByteData
.
lengthInBytes
,
);
);
}
}
return
SystemChannels
.
platform_views
.
invokeMethod
<
void
>(
'create'
,
args
);
await
SystemChannels
.
platform_views
.
invokeMethod
<
void
>(
'create'
,
args
);
return
true
;
}
}
@override
@override
...
@@ -1005,7 +1032,7 @@ class ExpensiveAndroidViewController extends AndroidViewController {
...
@@ -1005,7 +1032,7 @@ class ExpensiveAndroidViewController extends AndroidViewController {
}
}
@override
@override
Future
<
Size
>
setSiz
e
(
Size
size
)
{
Future
<
Size
>
_sendResizeMessag
e
(
Size
size
)
{
throw
UnimplementedError
(
'Not supported for
$SurfaceAndroidViewController
.'
);
throw
UnimplementedError
(
'Not supported for
$SurfaceAndroidViewController
.'
);
}
}
...
@@ -1044,8 +1071,7 @@ class TextureAndroidViewController extends AndroidViewController {
...
@@ -1044,8 +1071,7 @@ class TextureAndroidViewController extends AndroidViewController {
Offset
_off
=
Offset
.
zero
;
Offset
_off
=
Offset
.
zero
;
@override
@override
Future
<
Size
>
setSize
(
Size
size
)
async
{
Future
<
Size
>
_sendResizeMessage
(
Size
size
)
async
{
assert
(
_state
!=
_AndroidViewState
.
disposed
,
'Android view is disposed. View id:
$viewId
'
);
assert
(
_state
!=
_AndroidViewState
.
waitingForSize
,
'Android view must have an initial size. View id:
$viewId
'
);
assert
(
_state
!=
_AndroidViewState
.
waitingForSize
,
'Android view must have an initial size. View id:
$viewId
'
);
assert
(
size
!=
null
);
assert
(
size
!=
null
);
assert
(!
size
.
isEmpty
);
assert
(!
size
.
isEmpty
);
...
@@ -1064,16 +1090,6 @@ class TextureAndroidViewController extends AndroidViewController {
...
@@ -1064,16 +1090,6 @@ class TextureAndroidViewController extends AndroidViewController {
return
Size
(
meta
![
'width'
]!
as
double
,
meta
[
'height'
]!
as
double
);
return
Size
(
meta
![
'width'
]!
as
double
,
meta
[
'height'
]!
as
double
);
}
}
@override
Future
<
void
>
create
({
Size
?
size
})
async
{
if
(
size
==
null
)
{
return
;
}
assert
(
_state
==
_AndroidViewState
.
waitingForSize
,
'Android view is already sized. View id:
$viewId
'
);
assert
(!
size
.
isEmpty
);
return
super
.
create
(
size:
size
);
}
@override
@override
Future
<
void
>
setOffset
(
Offset
off
)
async
{
Future
<
void
>
setOffset
(
Offset
off
)
async
{
if
(
off
==
_off
)
{
if
(
off
==
_off
)
{
...
@@ -1100,9 +1116,9 @@ class TextureAndroidViewController extends AndroidViewController {
...
@@ -1100,9 +1116,9 @@ class TextureAndroidViewController extends AndroidViewController {
}
}
@override
@override
Future
<
void
>
_sendCreateMessage
({
Size
?
size
})
async
{
Future
<
bool
>
_sendCreateMessage
({
Size
?
size
})
async
{
if
(
size
==
null
)
{
if
(
size
==
null
)
{
return
;
return
false
;
}
}
assert
(!
size
.
isEmpty
,
'trying to create
$TextureAndroidViewController
without setting a valid size.'
);
assert
(!
size
.
isEmpty
,
'trying to create
$TextureAndroidViewController
without setting a valid size.'
);
...
@@ -1123,6 +1139,7 @@ class TextureAndroidViewController extends AndroidViewController {
...
@@ -1123,6 +1139,7 @@ class TextureAndroidViewController extends AndroidViewController {
);
);
}
}
_textureId
=
await
SystemChannels
.
platform_views
.
invokeMethod
<
int
>(
'create'
,
args
);
_textureId
=
await
SystemChannels
.
platform_views
.
invokeMethod
<
int
>(
'create'
,
args
);
return
true
;
}
}
@override
@override
...
@@ -1219,6 +1236,15 @@ abstract class PlatformViewController {
...
@@ -1219,6 +1236,15 @@ abstract class PlatformViewController {
/// * [PlatformViewsRegistry], which is a helper for managing platform view IDs.
/// * [PlatformViewsRegistry], which is a helper for managing platform view IDs.
int
get
viewId
;
int
get
viewId
;
/// True if [create] has not been successfully called the platform view.
///
/// This can indicate either that [create] was never called, or that [create]
/// was deferred for implementation-specific reasons.
///
/// A `false` return value does not necessarily indicate that the [Future]
/// returned by [create] has completed, only that creation has been started.
bool
get
awaitingCreation
=>
false
;
/// Dispatches the `event` to the platform view.
/// Dispatches the `event` to the platform view.
Future
<
void
>
dispatchPointerEvent
(
PointerEvent
event
);
Future
<
void
>
dispatchPointerEvent
(
PointerEvent
event
);
...
...
packages/flutter/lib/src/widgets/platform_view.dart
View file @
a47b3ccc
...
@@ -871,14 +871,20 @@ class _PlatformViewLinkState extends State<PlatformViewLink> {
...
@@ -871,14 +871,20 @@ class _PlatformViewLinkState extends State<PlatformViewLink> {
@override
@override
Widget
build
(
BuildContext
context
)
{
Widget
build
(
BuildContext
context
)
{
if
(
_controller
==
null
)
{
final
PlatformViewController
?
controller
=
_controller
;
if
(
controller
==
null
)
{
return
const
SizedBox
.
expand
();
return
const
SizedBox
.
expand
();
}
}
if
(!
_platformViewCreated
)
{
if
(!
_platformViewCreated
)
{
// Depending on the platform, the initial size can be used to size the platform view.
// Depending on the implementation, the initial size can be used to size
return
_PlatformViewPlaceHolder
(
onLayout:
(
Size
size
)
=>
_controller
!.
create
(
size:
size
));
// the platform view.
return
_PlatformViewPlaceHolder
(
onLayout:
(
Size
size
)
{
if
(
controller
.
awaitingCreation
)
{
controller
.
create
(
size:
size
);
}
});
}
}
_surface
??=
widget
.
_surfaceFactory
(
context
,
_controller
!
);
_surface
??=
widget
.
_surfaceFactory
(
context
,
controller
);
return
Focus
(
return
Focus
(
focusNode:
_focusNode
,
focusNode:
_focusNode
,
onFocusChange:
_handleFrameworkFocusChanged
,
onFocusChange:
_handleFrameworkFocusChanged
,
...
...
packages/flutter/test/services/fake_platform_views.dart
View file @
a47b3ccc
...
@@ -45,11 +45,16 @@ class FakePlatformViewController extends PlatformViewController {
...
@@ -45,11 +45,16 @@ class FakePlatformViewController extends PlatformViewController {
}
}
class
FakeAndroidViewController
implements
AndroidViewController
{
class
FakeAndroidViewController
implements
AndroidViewController
{
FakeAndroidViewController
(
this
.
viewId
);
FakeAndroidViewController
(
this
.
viewId
,
{
this
.
requiresSize
=
false
}
);
bool
disposed
=
false
;
bool
disposed
=
false
;
bool
focusCleared
=
false
;
bool
focusCleared
=
false
;
bool
created
=
false
;
bool
created
=
false
;
// If true, [create] won't be considered to have been called successfully
// unless it includes a size.
bool
requiresSize
;
bool
_createCalledSuccessfully
=
false
;
/// Events that are dispatched.
/// Events that are dispatched.
List
<
PointerEvent
>
dispatchedPointerEvents
=
<
PointerEvent
>[];
List
<
PointerEvent
>
dispatchedPointerEvents
=
<
PointerEvent
>[];
...
@@ -92,6 +97,9 @@ class FakeAndroidViewController implements AndroidViewController {
...
@@ -92,6 +97,9 @@ class FakeAndroidViewController implements AndroidViewController {
@override
@override
int
get
textureId
=>
0
;
int
get
textureId
=>
0
;
@override
bool
get
awaitingCreation
=>
!
_createCalledSuccessfully
;
@override
@override
bool
get
isCreated
=>
created
;
bool
get
isCreated
=>
created
;
...
@@ -114,7 +122,10 @@ class FakeAndroidViewController implements AndroidViewController {
...
@@ -114,7 +122,10 @@ class FakeAndroidViewController implements AndroidViewController {
}
}
@override
@override
Future
<
void
>
create
({
Size
?
size
})
async
{}
Future
<
void
>
create
({
Size
?
size
})
async
{
assert
(!
_createCalledSuccessfully
);
_createCalledSuccessfully
=
size
!=
null
||
!
requiresSize
;
}
@override
@override
List
<
PlatformViewCreatedCallback
>
get
createdCallbacks
=>
<
PlatformViewCreatedCallback
>[];
List
<
PlatformViewCreatedCallback
>
get
createdCallbacks
=>
<
PlatformViewCreatedCallback
>[];
...
...
packages/flutter/test/widgets/platform_view_test.dart
View file @
a47b3ccc
...
@@ -1026,6 +1026,7 @@ void main() {
...
@@ -1026,6 +1026,7 @@ void main() {
containerFocusNode
.
requestFocus
();
containerFocusNode
.
requestFocus
();
viewsController
.
createCompleter
!.
complete
();
await
tester
.
pump
();
await
tester
.
pump
();
expect
(
containerFocusNode
.
hasFocus
,
isTrue
);
expect
(
containerFocusNode
.
hasFocus
,
isTrue
);
...
@@ -2423,7 +2424,110 @@ void main() {
...
@@ -2423,7 +2424,110 @@ void main() {
onCreatePlatformView:
(
PlatformViewCreationParams
params
)
{
onCreatePlatformView:
(
PlatformViewCreationParams
params
)
{
onPlatformViewCreatedCallBack
=
params
.
onPlatformViewCreated
;
onPlatformViewCreatedCallBack
=
params
.
onPlatformViewCreated
;
createdPlatformViewId
=
params
.
id
;
createdPlatformViewId
=
params
.
id
;
return
FakePlatformViewController
(
params
.
id
);
return
FakePlatformViewController
(
params
.
id
)..
create
();
},
surfaceFactory:
(
BuildContext
context
,
PlatformViewController
controller
)
{
return
PlatformViewSurface
(
gestureRecognizers:
const
<
Factory
<
OneSequenceGestureRecognizer
>>{},
controller:
controller
,
hitTestBehavior:
PlatformViewHitTestBehavior
.
opaque
,
);
},
);
await
tester
.
pumpWidget
(
platformViewLink
);
expect
(
tester
.
allWidgets
.
map
((
Widget
widget
)
=>
widget
.
runtimeType
.
toString
()).
toList
(),
equals
(<
String
>[
'PlatformViewLink'
,
'_PlatformViewPlaceHolder'
]),
);
onPlatformViewCreatedCallBack
(
createdPlatformViewId
);
await
tester
.
pump
();
expect
(
tester
.
allWidgets
.
map
((
Widget
widget
)
=>
widget
.
runtimeType
.
toString
()).
toList
(),
equals
(<
String
>[
'PlatformViewLink'
,
'Focus'
,
'_FocusMarker'
,
'Semantics'
,
'PlatformViewSurface'
]),
);
expect
(
createdPlatformViewId
,
currentViewId
+
1
);
},
);
testWidgets
(
'PlatformViewLink calls create when needed for Android texture display modes'
,
(
WidgetTester
tester
)
async
{
final
int
currentViewId
=
platformViewsRegistry
.
getNextPlatformViewId
();
late
int
createdPlatformViewId
;
late
PlatformViewCreatedCallback
onPlatformViewCreatedCallBack
;
late
PlatformViewController
controller
;
final
PlatformViewLink
platformViewLink
=
PlatformViewLink
(
viewType:
'webview'
,
onCreatePlatformView:
(
PlatformViewCreationParams
params
)
{
onPlatformViewCreatedCallBack
=
params
.
onPlatformViewCreated
;
createdPlatformViewId
=
params
.
id
;
controller
=
FakeAndroidViewController
(
params
.
id
,
requiresSize:
true
);
controller
.
create
();
// This test should be simulating one of the texture-based display
// modes, where `create` is a no-op when not provided a size, and
// creation is triggered via a later call to setSize, or to `create`
// with a size.
expect
(
controller
.
awaitingCreation
,
true
);
return
controller
;
},
surfaceFactory:
(
BuildContext
context
,
PlatformViewController
controller
)
{
return
PlatformViewSurface
(
gestureRecognizers:
const
<
Factory
<
OneSequenceGestureRecognizer
>>{},
controller:
controller
,
hitTestBehavior:
PlatformViewHitTestBehavior
.
opaque
,
);
},
);
await
tester
.
pumpWidget
(
platformViewLink
);
expect
(
tester
.
allWidgets
.
map
((
Widget
widget
)
=>
widget
.
runtimeType
.
toString
()).
toList
(),
equals
(<
String
>[
'PlatformViewLink'
,
'_PlatformViewPlaceHolder'
]),
);
onPlatformViewCreatedCallBack
(
createdPlatformViewId
);
await
tester
.
pump
();
expect
(
tester
.
allWidgets
.
map
((
Widget
widget
)
=>
widget
.
runtimeType
.
toString
()).
toList
(),
equals
(<
String
>[
'PlatformViewLink'
,
'Focus'
,
'_FocusMarker'
,
'Semantics'
,
'PlatformViewSurface'
]),
);
expect
(
createdPlatformViewId
,
currentViewId
+
1
);
expect
(
controller
.
awaitingCreation
,
false
);
},
);
testWidgets
(
'PlatformViewLink does not double-call create for Android Hybrid Composition'
,
(
WidgetTester
tester
)
async
{
final
int
currentViewId
=
platformViewsRegistry
.
getNextPlatformViewId
();
late
int
createdPlatformViewId
;
late
PlatformViewCreatedCallback
onPlatformViewCreatedCallBack
;
late
PlatformViewController
controller
;
final
PlatformViewLink
platformViewLink
=
PlatformViewLink
(
viewType:
'webview'
,
onCreatePlatformView:
(
PlatformViewCreationParams
params
)
{
onPlatformViewCreatedCallBack
=
params
.
onPlatformViewCreated
;
createdPlatformViewId
=
params
.
id
;
controller
=
FakeAndroidViewController
(
params
.
id
);
controller
.
create
();
// This test should be simulating Hybrid Composition mode, where
// `create` takes effect immidately.
expect
(
controller
.
awaitingCreation
,
false
);
return
controller
;
},
},
surfaceFactory:
(
BuildContext
context
,
PlatformViewController
controller
)
{
surfaceFactory:
(
BuildContext
context
,
PlatformViewController
controller
)
{
return
PlatformViewSurface
(
return
PlatformViewSurface
(
...
...
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