[flutter_tools] register service worker after first frame event (#66082)
Registering the service worker immediately after the documented has loaded may cause SW initialization to compete with framework initialization. It was recommended to us that we defer the service worker setup until after the framework is done with setup, which should be sometime after the first frame. To implement this, I augmented the binding setup to dispatch an event on the document after the binding has initialized. I don't see any obvious risks with this setup. Fixes #66066
This commit is contained in:
parent
9bfab77835
commit
90d83e37a6
@ -24,7 +24,7 @@ found in the LICENSE file. -->
|
||||
https://developers.google.com/web/fundamentals/primers/service-workers -->
|
||||
<script>
|
||||
if ('serviceWorker' in navigator) {
|
||||
window.addEventListener('load', function () {
|
||||
window.addEventListener('flutter-first-frame', function () {
|
||||
navigator.serviceWorker.register('flutter_service_worker.js');
|
||||
});
|
||||
}
|
||||
|
@ -44,6 +44,9 @@ mixin RendererBinding on BindingBase, ServicesBinding, SchedulerBinding, Gesture
|
||||
assert(renderView != null);
|
||||
addPersistentFrameCallback(_handlePersistentFrameCallback);
|
||||
initMouseTracker();
|
||||
if (kIsWeb) {
|
||||
addPostFrameCallback(_handleWebFirstFrame);
|
||||
}
|
||||
}
|
||||
|
||||
/// The current [RendererBinding], if one has been created.
|
||||
@ -281,6 +284,12 @@ mixin RendererBinding on BindingBase, ServicesBinding, SchedulerBinding, Gesture
|
||||
}
|
||||
}
|
||||
|
||||
void _handleWebFirstFrame(Duration _) {
|
||||
assert(kIsWeb);
|
||||
const MethodChannel methodChannel = MethodChannel('flutter/service_worker');
|
||||
methodChannel.invokeMethod<void>('first-frame');
|
||||
}
|
||||
|
||||
void _handleSemanticsAction(int id, SemanticsAction action, ByteData? args) {
|
||||
_pipelineOwner.semanticsOwner?.performAction(
|
||||
id,
|
||||
|
38
packages/flutter/test/rendering/first_frame_test.dart
Normal file
38
packages/flutter/test/rendering/first_frame_test.dart
Normal file
@ -0,0 +1,38 @@
|
||||
// Copyright 2014 The Flutter Authors. All rights reserved.
|
||||
// Use of this source code is governed by a BSD-style license that can be
|
||||
// found in the LICENSE file.
|
||||
|
||||
// @dart = 2.8
|
||||
|
||||
import 'dart:async';
|
||||
|
||||
import 'package:flutter/foundation.dart';
|
||||
import 'package:flutter/gestures.dart';
|
||||
import 'package:flutter/rendering.dart';
|
||||
import 'package:flutter/scheduler.dart';
|
||||
import 'package:flutter/services.dart';
|
||||
|
||||
import '../flutter_test_alternative.dart';
|
||||
|
||||
void main() {
|
||||
test('Flutter dispatches first frame event on the web only', () async {
|
||||
final Completer<void> completer = Completer<void>();
|
||||
final TestRenderBinding binding = TestRenderBinding();
|
||||
const MethodChannel firstFrameChannel = MethodChannel('flutter/service_worker');
|
||||
firstFrameChannel.setMockMethodCallHandler((MethodCall methodCall) async {
|
||||
completer.complete();
|
||||
});
|
||||
|
||||
binding.handleBeginFrame(Duration.zero);
|
||||
binding.handleDrawFrame();
|
||||
|
||||
await expectLater(completer.future, completes);
|
||||
}, skip: !kIsWeb);
|
||||
}
|
||||
|
||||
class TestRenderBinding extends BindingBase with SchedulerBinding, ServicesBinding, GestureBinding, SemanticsBinding, RendererBinding {
|
||||
@override
|
||||
void initInstances() {
|
||||
super.initInstances();
|
||||
}
|
||||
}
|
@ -56,7 +56,7 @@ class TestMouseTrackerFlutterBinding extends BindingBase
|
||||
_overridePhase = lastPhase;
|
||||
}
|
||||
|
||||
List<void Function(Duration)> postFrameCallbacks;
|
||||
List<void Function(Duration)> postFrameCallbacks = <void Function(Duration)>[];
|
||||
|
||||
// Proxy post-frame callbacks.
|
||||
@override
|
||||
|
@ -23,7 +23,7 @@
|
||||
https://developers.google.com/web/fundamentals/primers/service-workers -->
|
||||
<script>
|
||||
if ('serviceWorker' in navigator) {
|
||||
window.addEventListener('load', function () {
|
||||
window.addEventListener('flutter-first-frame', function () {
|
||||
navigator.serviceWorker.register('flutter_service_worker.js');
|
||||
});
|
||||
}
|
||||
|
Loading…
x
Reference in New Issue
Block a user