flutter-embedding-native-views
Compare original and translation side by side
🇺🇸
Original
English🇨🇳
Translation
ChineseIntegrating Platform Views and Web Content
平台视图与Web内容集成
Contents
目录
Platform Views Architecture
平台视图架构
Platform Views allow embedding native views (Android, iOS, macOS) directly into a Flutter application, enabling the application of transforms, clips, and opacity from Dart.
Platform Views支持将原生视图(Android、iOS、macOS)直接嵌入Flutter应用,能够通过Dart应用变换、裁剪和透明度效果。
Android Implementations (API 23+)
Android实现(API 23+)
Choose the appropriate implementation based on your performance and fidelity requirements:
- Hybrid Composition: Renders Flutter content into a texture and uses to compose both.
SurfaceFlinger- Pros: Best performance and fidelity for Android views.
- Cons: Lowers overall application FPS. Certain Flutter widget transformations will not work.
- Texture Layer (Texture Layer Hybrid Composition): Renders Platform Views into a texture. Flutter draws them via the texture and renders its own content directly into a Surface.
- Pros: Best performance for Flutter rendering. All transformations work correctly.
- Cons: Quick scrolling (e.g., WebViews) can be janky. is problematic (breaks accessibility). Text magnifiers break unless Flutter is rendered into a
SurfaceView.TextureView
根据性能和保真度需求选择合适的实现方式:
- Hybrid Composition:将Flutter内容渲染为纹理,使用进行合成。
SurfaceFlinger- 优点:对Android视图来说性能和保真度最佳。
- 缺点:会降低应用整体FPS。部分Flutter组件变换无法生效。
- Texture Layer(Texture Layer Hybrid Composition):将Platform Views渲染为纹理。Flutter通过该纹理绘制内容,并直接将自身内容渲染到Surface中。
- 优点:Flutter渲染性能最佳。所有变换均可正常工作。
- 缺点:快速滚动(如WebViews)时可能出现卡顿。存在兼容性问题(会破坏无障碍功能)。除非Flutter渲染到
SurfaceView中,否则文本放大镜功能会失效。TextureView
iOS & macOS Implementations
iOS & macOS实现
- iOS: Uses Hybrid Composition exclusively. The native is appended to the view hierarchy.
UIView- Limitations: and
ShaderMaskwidgets are not supported.ColorFilteredhas composition limitations.BackdropFilter
- Limitations:
- macOS: Uses Hybrid Composition ().
NSView- Limitations: Not fully functional in current releases (e.g., gesture support is unavailable).
- iOS:仅使用Hybrid Composition。原生会被添加到视图层级中。
UIView- 限制:不支持和
ShaderMask组件。ColorFiltered存在合成限制。BackdropFilter
- 限制:不支持
- macOS:使用Hybrid Composition()。
NSView- 限制:当前版本中功能未完全实现(例如手势支持缺失)。
Performance Mitigation
性能优化
Mitigate performance drops during complex Dart animations by rendering a screenshot of the native view as a placeholder texture while the animation runs.
在复杂Dart动画运行期间,可将原生视图的截图作为占位纹理,以缓解性能下降问题。
Web Embedding Architecture
Web嵌入架构
Embed Flutter into existing web applications (Vanilla JS, React, Angular, etc.) using either Full Page mode or Embedded (Multi-view) mode.
- Full Page Mode: Flutter takes over the entire browser window. Use an if you need to constrain the Flutter app without modifying the Flutter bootstrap process.
iframe - Embedded Mode (Multi-view): Render Flutter into specific HTML elements (s). Requires
divduring engine initialization.multiViewEnabled: true- Manage views from JavaScript using and
app.addView().app.removeView() - In Dart, replace with
runApp.runWidget - Manage the dynamic list of views using and render them using
WidgetsBinding.instance.platformDispatcher.viewsandViewCollectionwidgets.View
- Manage views from JavaScript using
将Flutter嵌入现有Web应用(Vanilla JS、React、Angular等),支持全屏模式或嵌入(多视图)模式。
- 全屏模式:Flutter接管整个浏览器窗口。若需在不修改Flutter引导流程的情况下限制Flutter应用范围,可使用。
iframe - 嵌入模式(多视图):将Flutter渲染到特定HTML元素()中。引擎初始化时需设置
div。multiViewEnabled: true- 通过JavaScript使用和
app.addView()管理视图。app.removeView() - 在Dart中,用替代
runWidget。runApp - 使用管理动态视图列表,并通过
WidgetsBinding.instance.platformDispatcher.views和ViewCollection组件进行渲染。View
- 通过JavaScript使用
Workflow: Implementing Android Platform Views
工作流:实现Android平台视图
Follow this sequential workflow to implement a Platform View on Android.
Task Progress:
- 1. Determine the composition mode (Hybrid vs. Texture Layer).
- 2. Implement the Dart widget.
- 3. Implement the native Android View and Factory.
- 4. Register the Platform View in the Android host.
- 5. Run validator -> review rendering -> fix manual invalidation issues.
按照以下步骤在Android上实现Platform View。
任务进度:
- 1. 确定合成模式(Hybrid vs Texture Layer)。
- 2. 实现Dart组件。
- 3. 实现原生Android View与Factory。
- 4. 在Android宿主中注册Platform View。
- 5. 运行验证器 -> 检查渲染效果 -> 修复手动失效问题。
1. Dart Implementation
1. Dart实现
If using Hybrid Composition, use , , and .
If using Texture Layer, use the widget.
PlatformViewLinkAndroidViewSurfacePlatformViewsService.initSurfaceAndroidViewAndroidView若使用Hybrid Composition,需使用、和。
若使用Texture Layer,则使用组件。
PlatformViewLinkAndroidViewSurfacePlatformViewsService.initSurfaceAndroidViewAndroidView2. Native Implementation
2. 原生实现
Create a class implementing that returns your native .
Create a factory extending to instantiate your view.
io.flutter.plugin.platform.PlatformViewandroid.view.ViewPlatformViewFactory创建实现的类,返回你的原生。
创建继承自的工厂类,用于实例化视图。
io.flutter.plugin.platform.PlatformViewandroid.view.ViewPlatformViewFactory3. Registration
3. 注册
Register the factory in your (or plugin) using .
MainActivity.ktflutterEngine.platformViewsController.registry.registerViewFactoryNote: If your native view uses or , manually call on the View or its parent when content changes, as they do not invalidate themselves automatically.
SurfaceViewSurfaceTextureinvalidate在(或插件)中,通过注册工厂类。
MainActivity.ktflutterEngine.platformViewsController.registry.registerViewFactory注意:若你的原生视图使用或,当内容变化时需手动调用View或其父视图的方法,因为它们不会自动触发失效。
SurfaceViewSurfaceTextureinvalidateWorkflow: Implementing iOS Platform Views
工作流:实现iOS平台视图
Follow this sequential workflow to implement a Platform View on iOS.
Task Progress:
- 1. Implement the Dart widget using .
UiKitView - 2. Implement the native iOS View () and Factory (
FlutterPlatformView).FlutterPlatformViewFactory - 3. Register the Platform View in or the plugin registrar.
AppDelegate.swift - 4. Run validator -> review composition limitations -> fix unsupported filters.
按照以下步骤在iOS上实现Platform View。
任务进度:
- 1. 使用实现Dart组件。
UiKitView - 2. 实现原生iOS View()与Factory(
FlutterPlatformView)。FlutterPlatformViewFactory - 3. 在或插件注册器中注册Platform View。
AppDelegate.swift - 4. 运行验证器 -> 检查合成限制 -> 修复不支持的滤镜问题。
Workflow: Embedding Flutter in Web Applications
工作流:在Web应用中嵌入Flutter
Follow this sequential workflow to embed Flutter into an existing web DOM.
Task Progress:
- 1. Update to enable multi-view.
flutter_bootstrap.js - 2. Update to use
main.dartandrunWidget.ViewCollection - 3. Implement JavaScript logic to add/remove host elements.
- 4. Run validator -> review view constraints -> fix CSS conflicts.
按照以下步骤将Flutter嵌入现有Web DOM中。
任务进度:
- 1. 更新以启用多视图。
flutter_bootstrap.js - 2. 更新以使用
main.dart和runWidget。ViewCollection - 3. 实现JavaScript逻辑以添加/移除宿主元素。
- 4. 运行验证器 -> 检查视图约束 -> 修复CSS冲突。
1. JavaScript Configuration
1. JavaScript配置
In , initialize the engine with .
Use the returned object to add views: .
flutter_bootstrap.jsmultiViewEnabled: trueappapp.addView({ hostElement: document.getElementById('my-div') })在中,初始化引擎时设置。
使用返回的对象添加视图:。
flutter_bootstrap.jsmultiViewEnabled: trueappapp.addView({ hostElement: document.getElementById('my-div') })2. Dart Configuration
2. Dart配置
Replace with .
Create a root widget that listens to .
Map over to create a widget for each attached , and wrap them all in a .
runApp()runWidget()WidgetsBindingObserver.didChangeMetricsWidgetsBinding.instance.platformDispatcher.viewsViewFlutterViewViewCollection用替代。
创建根组件,监听。
遍历,为每个附加的创建组件,并将所有组件包裹在中。
runWidget()runApp()WidgetsBindingObserver.didChangeMetricsWidgetsBinding.instance.platformDispatcher.viewsFlutterViewViewViewCollectionExamples
示例
Example: Android Texture Layer (Dart)
示例:Android Texture Layer(Dart)
dart
import 'package:flutter/material.dart';
import 'package:flutter/services.dart';
class NativeAndroidView extends StatelessWidget {
Widget build(BuildContext context) {
const String viewType = 'my_native_view';
final Map<String, dynamic> creationParams = <String, dynamic>{};
return AndroidView(
viewType: viewType,
layoutDirection: TextDirection.ltr,
creationParams: creationParams,
creationParamsCodec: const StandardMessageCodec(),
);
}
}dart
import 'package:flutter/material.dart';
import 'package:flutter/services.dart';
class NativeAndroidView extends StatelessWidget {
Widget build(BuildContext context) {
const String viewType = 'my_native_view';
final Map<String, dynamic> creationParams = <String, dynamic>{};
return AndroidView(
viewType: viewType,
layoutDirection: TextDirection.ltr,
creationParams: creationParams,
creationParamsCodec: const StandardMessageCodec(),
);
}
}Example: Web Multi-View Initialization (JavaScript)
示例:Web多视图初始化(JavaScript)
javascript
_flutter.loader.load({
onEntrypointLoaded: async function(engineInitializer) {
let engine = await engineInitializer.initializeEngine({
multiViewEnabled: true,
});
let app = await engine.runApp();
// Add a view to a specific DOM element
let viewId = app.addView({
hostElement: document.querySelector('#flutter-host-container'),
initialData: { customData: 'Hello from JS' }
});
}
});javascript
_flutter.loader.load({
onEntrypointLoaded: async function(engineInitializer) {
let engine = await engineInitializer.initializeEngine({
multiViewEnabled: true,
});
let app = await engine.runApp();
// Add a view to a specific DOM element
let viewId = app.addView({
hostElement: document.querySelector('#flutter-host-container'),
initialData: { customData: 'Hello from JS' }
});
}
});Example: Web Multi-View Root Widget (Dart)
示例:Web多视图根组件(Dart)
dart
import 'dart:ui' show FlutterView;
import 'package:flutter/widgets.dart';
void main() {
runWidget(MultiViewApp(viewBuilder: (context) => const MyEmbeddedWidget()));
}
class MultiViewApp extends StatefulWidget {
final WidgetBuilder viewBuilder;
const MultiViewApp({super.key, required this.viewBuilder});
State<MultiViewApp> createState() => _MultiViewAppState();
}
class _MultiViewAppState extends State<MultiViewApp> with WidgetsBindingObserver {
Map<Object, Widget> _views = {};
void initState() {
super.initState();
WidgetsBinding.instance.addObserver(this);
_updateViews();
}
void didChangeMetrics() => _updateViews();
void _updateViews() {
final newViews = <Object, Widget>{};
for (final FlutterView view in WidgetsBinding.instance.platformDispatcher.views) {
newViews[view.viewId] = _views[view.viewId] ?? View(
view: view,
child: Builder(builder: widget.viewBuilder),
);
}
setState(() => _views = newViews);
}
void dispose() {
WidgetsBinding.instance.removeObserver(this);
super.dispose();
}
Widget build(BuildContext context) {
return ViewCollection(views: _views.values.toList(growable: false));
}
}dart
import 'dart:ui' show FlutterView;
import 'package:flutter/widgets.dart';
void main() {
runWidget(MultiViewApp(viewBuilder: (context) => const MyEmbeddedWidget()));
}
class MultiViewApp extends StatefulWidget {
final WidgetBuilder viewBuilder;
const MultiViewApp({super.key, required this.viewBuilder});
State<MultiViewApp> createState() => _MultiViewAppState();
}
class _MultiViewAppState extends State<MultiViewApp> with WidgetsBindingObserver {
Map<Object, Widget> _views = {};
void initState() {
super.initState();
WidgetsBinding.instance.addObserver(this);
_updateViews();
}
void didChangeMetrics() => _updateViews();
void _updateViews() {
final newViews = <Object, Widget>{};
for (final FlutterView view in WidgetsBinding.instance.platformDispatcher.views) {
newViews[view.viewId] = _views[view.viewId] ?? View(
view: view,
child: Builder(builder: widget.viewBuilder),
);
}
setState(() => _views = newViews);
}
void dispose() {
WidgetsBinding.instance.removeObserver(this);
super.dispose();
}
Widget build(BuildContext context) {
return ViewCollection(views: _views.values.toList(growable: false));
}
}