主頁 > 移動端開發 > 一統天下 flutter - 插件: flutter 使用 web 原生控制元件,并做資料通信

一統天下 flutter - 插件: flutter 使用 web 原生控制元件,并做資料通信

2023-05-11 08:43:32 移動端開發

原始碼 https://github.com/webabcd/flutter_demo
作者 webabcd

一統天下 flutter - 插件: flutter 使用 web 原生控制元件,并做資料通信

示例如下:

lib\plugin\plugin2.dart

/*
 * 插件
 * 本例用于演示 flutter 使用 android/ios/web 原生控制元件,并做資料通信
 *
 * 一、android 插件開發
 * 1、主 flutter 專案要先在 android 平臺中運行一下
 * 2、在 android 檔案夾上,使用右鍵選單,然后選擇 Flutter -> Open Android module in Android Studio 即可開發插件
 * 3、參見 /android/app/src/main/kotlin/com/example/flutter_demo/MainActivity.kt
 *
 * 二、ios 插件開發
 * 1、主 flutter 專案要先在 ios 平臺中運行一下
 * 2、在 android studio 或 visual studio code 中執行如下邏輯
 *    cd ios
 *    pod install
 * 3、用 xcode 中打開 /ios/Runner.xcworkspace 即可開發插件
 * 4、參見 /ios/Runner/AppDelegate.swift
 *
 * 三、web 原生控制元件,以及 flutter 與 js 的通信
 * 1、參見 /lib/plugin/flutter_plugin_web2.dart
 *
 *
 * 注:插件中實作的功能(非 .dart 實作的)不支持 flutter 的 hot reload
 */

import 'dart:async';
import 'dart:io';

import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart';
import 'package:flutter/services.dart';

import '../helper.dart';

/// 這里要注意,如果編譯的時候,目標平臺不是 web 環境,那么如果專案中 import 了 dart:js, dart:ui, dart:html 之類的庫,則會報類似如下的錯誤
/// FileSystemException(uri=org-dartlang-untranslatable-uri:dart%3Ahtml; message=StandardFileSystem only supports file:* and data:* URIs)
/// 此時,就需要用如下的方式 import
/// 下面的 import 的意思是:匯入 flutter_plugin_web2_stub.dart,但是編譯為 web 時(即 dart.library.js 為真)則匯入 flutter_plugin_web2.dart
/// flutter_plugin_web2_stub.dart 里的對外的方法定義與 flutter_plugin_web2.dart 是一樣的
/// 但是 flutter_plugin_web2_stub.dart 中沒有具體的邏輯,不會匯入 dart:js, dart:ui, dart:html 之類的庫,這樣就保證了編譯為非 web 時不會報錯
/// 而 flutter_plugin_web2.dart 有具體的邏輯,會匯入 dart:js, dart:ui, dart:html 之類的庫,這樣就保證了編譯為 web 時會包括相關的邏輯
import 'flutter_plugin_web2_stub.dart' if (dart.library.js) "flutter_plugin_web2.dart";

class Plugin2Demo extends StatefulWidget {
  const Plugin2Demo({Key? key}) : super(key: key);

  @override
  _Plugin2DemoState createState() => _Plugin2DemoState();
}

class _Plugin2DemoState extends State<Plugin2Demo> {

  @override
  Widget build(BuildContext context) {

    return Scaffold(
      appBar: AppBar(
        title: const Text('title'),
      ),
      body: const _MyWidget()
    );
  }
}


class _MyWidget extends StatefulWidget {
  const _MyWidget({Key? key}) : super(key: key);

  @override
  _MyWidgetState createState() => _MyWidgetState();
}

class _MyWidgetState extends State<_MyWidget> {

  /// 用于保存從 android/ios 發送到 flutter 的資料
  String _nativeToFlutterMessage = '';
  /// 用于控制 android/ios 和 flutter 通信的 controller
  final _controller = _MyViewController();

  @override
  void initState() {

    _controller.addListener(() {
      setState(() {
        _nativeToFlutterMessage = _controller.nativeToFlutterMessage;
      });
    });

    super.initState();
  }

  @override
  Widget build(BuildContext context) {
    return Column(
      crossAxisAlignment: CrossAxisAlignment.stretch,
      children: [
        Expanded(
          child: Container(
            color: Colors.red,
            child: _buildNativeView(),
          ),
        ),
        Expanded(
          child: Container(
            color: Colors.green,
            child: _buildFlutterView(),
          ),
        ),
      ],
    );
  }

  /// 嵌入到 flutter 中的 android/ios 的 view
  Widget _buildNativeView() {
    return _MyNativeView(
      controller: _controller,
    );
  }

  Widget _buildFlutterView() {
    return Stack(
      alignment: AlignmentDirectional.bottomCenter,
      children: [
        Column(
          mainAxisAlignment: MainAxisAlignment.center,
          children: [
            Text("native to flutter: $_nativeToFlutterMessage"),
            const SizedBox(height: 10),
            ElevatedButton(
              onPressed: () {
                _controller.flutterToNative("${DateTime.now().millisecondsSinceEpoch}");
              },
              child: const Text('發送資料給 Native'),
            ),
          ],
        ),
        const Padding(
          padding: EdgeInsets.only(bottom: 15),
          child: Text(
            'Flutter - View',
            style: TextStyle(
              fontSize: 24,
              fontWeight: FontWeight.bold,
            ),
          ),
        ),
      ],
    );
  }
}


/// 嵌入到 flutter 中的 android/ios 的 view
class _MyNativeView extends StatefulWidget {
  final _MyViewController controller;

  const _MyNativeView({required this.controller, Key? key}) : super(key: key);

  @override
  _MyNativeViewState createState() => _MyNativeViewState();
}

class _MyNativeViewState extends State<_MyNativeView> {

  @override
  Widget build(BuildContext context) {

    /// 判斷是否為 web 環境要用 kIsWeb
    /// 如果在 web 環境使用 Platform.xxx 的話會報錯的
    if (kIsWeb) {
      /// 嵌入到 flutter 中的 web 的 view(相關的插件在 /lib/plugin/flutter_plugin_web2.dart)
      /// 這是一個 HtmlElementView 型別的組件
      return FlutterPluginWeb2().getHtmlElementView(widget.controller.jsToFlutter);
    }

    if (Platform.isAndroid) {
      /// 嵌入到 flutter 中的 android 的 view(相關的插件在 android/app/src/main/kotlin/com/example/flutter_demo/MyFlutterPlugin2.kt)
      return AndroidView(
        viewType: 'com.webabcd.flutter/myview',                           /// 需要嵌入的 view 的標識(這是在插件中定義的)
        onPlatformViewCreated: _onPlatformViewCreated,                    /// 傳給插件的初始引數
        creationParams: const <String, dynamic>{'k1': 'p1', 'k2': 'p2'},  /// 傳給插件的初始引數的編碼方式
        creationParamsCodec: const StandardMessageCodec(),                /// 需要嵌入的 view 創建后觸發的事件
      );
    }

    if (Platform.isIOS) {
      /// 嵌入到 flutter 中的 ios 的 view(相關的插件在 ios/Runner/MyFlutterPlugin2.swift)
      return UiKitView(
        viewType: 'com.webabcd.flutter/myview',                           /// 需要嵌入的 view 的標識(這是在插件中定義的)
        creationParams: const <String, dynamic>{'k1': 'p1', 'k2': 'p2'},  /// 傳給插件的初始引數
        creationParamsCodec: const StandardMessageCodec(),                /// 傳給插件的初始引數的編碼方式
        onPlatformViewCreated: _onPlatformViewCreated,                    /// 需要嵌入的 view 創建后觸發的事件
      );
    }

    return const MyText("不支持當前環境");
  }

  /// 對于 android 來說,這個 id 是插件中 PlatformViewFactory 的 create() 中生成的 viewId
  /// 對于 ios 來說,這個 id 是插件中 FlutterPlatformViewFactory 的 create() 中生成的 viewId
  void _onPlatformViewCreated(int id) {
    var methodChannel = MethodChannel('com.webabcd.flutter/channel2_view$id');
    widget.controller.setMethodChannel(methodChannel);
  }
}

/// 用于控制 android/ios 和 flutter 通信的 controller
class _MyViewController extends ChangeNotifier {

  late MethodChannel _methodChannel;

  String nativeToFlutterMessage = "";

  /// 接收從 web 發送到 flutter 的資料
  void jsToFlutter(String message) {
    nativeToFlutterMessage = message;
    notifyListeners();
  }

  /// 接收從 android/ios 發送到 flutter 的資料
  void setMethodChannel(MethodChannel methodChannel) {
    _methodChannel = methodChannel;
    _methodChannel.setMethodCallHandler((call) async {
      switch (call.method) {
        case 'nativeToFlutter':
          final result = call.arguments as String;
          nativeToFlutterMessage = result;
          notifyListeners();
          break;
      }
    });
  }

  /// 從 flutter 發送資料到 android/ios/web
  Future<void> flutterToNative(String message) async {
    if (kIsWeb) {
      /// 從 flutter 發送資料到 web
      var result = FlutterPluginWeb2.flutterToJs(message);
    }
    else {
      /// 從 flutter 發送資料到 android/ios
      await _methodChannel.invokeMethod('flutterToNative', message);
    }
  }
}

lib\plugin\flutter_plugin_web2_stub.dart

/*
 * 此檔案對外的方法定義與 flutter_plugin_web2.dart 是一致的,用于編譯非 web 時使用
 */

import 'package:flutter/material.dart';

class FlutterPluginWeb2 {

  Widget getHtmlElementView(dynamic jsToFlutter) {
    return const Text("不可能到這里");
  }

  static dynamic flutterToJs(String message) {
    return "不可能到這里";
  }
}

lib\plugin\flutter_plugin_web2.dart

/*
 * 本例用于演示 web 插件的開發(flutter 使用 web 原生控制元件,并做資料通信)
 *
 * 本例中用的 flutter 與 js 的通信方式實作起來比較簡單,但是無法和 android/ios 插件的介面保持一致
 * 如果對于 flutter 的開發來說,其想要與 android/ios/web 通信的方法都是一樣的,則可以參見 flutter_plugin_web.dart 中的實作方式
 */

import 'dart:html' as html;
import 'dart:ui' as ui;
import 'dart:js' as js;

import 'package:flutter/material.dart';
import 'package:flutter/services.dart';

class FlutterPluginWeb2 {

  /// 構造一個 HtmlElementView,其用于在 flutter 中顯示指定的 html(僅在 web 環境可用)
  HtmlElementView getHtmlElementView(dynamic jsToFlutter) {
    var view = html.DivElement()
      ..append(html.StyleElement()
        ..text = """
            #myDiv {
              height: 100%;
              font-size: 14px
              color: white;
              display: flex;
              flex-direction: column;
              align-items: center;
            }
          """
      )
      ..append(html.ScriptElement()
        ..text = """
            // 用于演示 flutter 呼叫 js
            function xxx_flutterToJs(message) {
              document.getElementById("txtMessage").innerHTML = "flutter to js: " + message;
            }
            
            // 用于演示 js 呼叫 flutter
            function send() {
              // 通過 xxx_jsToFlutter() 呼叫 flutter
              // 這個 xxx_jsToFlutter() 是通過類似這樣注冊的 js.context["xxx_jsToFlutter"] = jsToFlutter;
              window.xxx_jsToFlutter(new Date().getTime().toString());
            }
          """
      )
      ..append(html.DivElement()
        ..id = 'myDiv'
        ..append(html.DivElement()
          ..id = 'txtMessage'
          ..setAttribute("style", "flex-grow: 1; display: flex; flex-direction: column; justify-content: flex-end; color: black; margin-bottom: 12px")
        )
        ..append(html.DivElement()
          ..setAttribute("style", "flex-grow: 1; display: flex; flex-direction: column; justify-content: flex-start;")
          ..setAttribute("onclick", "send();")
          ..append(html.ButtonElement()
            ..setAttribute("style", "padding: 12px")
            ..setInnerHtml("發送資料給 flutter")
          )
        )
        ..append(html.DivElement()
          ..setAttribute("style", "flex-grow: 0; color: black; font-size: 24px; margin-bottom: 12px")
          ..setInnerHtml("Web - View")
        )
      );

    /// 注冊一個名為 com.webabcd.flutter/myview 的 html
    /// 必須要有下面這行注釋,否則會報錯 The name 'platformViewRegistry' is being referenced through the prefix 'ui', but it isn't defined in any of the libraries imported using that prefix.
    // ignore: undefined_prefixed_name
    ui.platformViewRegistry.registerViewFactory('com.webabcd.flutter/myview', (int viewId) => view);

    /// 通過 js.context[] 在 js 中注冊 js 呼叫 flutter 的方法,并將其映射到 flutter 中指定的方法
    /// 本例的意思是,在 js 中注冊一個名為 xxx_jsToFlutter() 的方法,在 js 中呼叫此方法后,就會呼叫 flutter 中的 jsToFlutter() 方法
    js.context["xxx_jsToFlutter"] = jsToFlutter;

    /// 使用已注冊的名為 com.webabcd.flutter/myview 的 html
    return HtmlElementView(
      viewType: 'com.webabcd.flutter/myview',
      onPlatformViewCreated: _onPlatformViewCreated,  /// 需要嵌入的 view 創建后觸發的事件
    );
  }

  void _onPlatformViewCreated(int id) {
    /// 這里的 id 就是上面 (int viewId) => view 中的 viewId
    /// 一般在這里構造一個 MethodChannel 用于 flutter 和 web 之間的資料通信
    /// 這種方式可以讓 flutter 和 web 之間的資料通信介面與 android/ios 插件的介面保持一致,從而對于 flutter 的開發來說,保證它與 android/ios/web 通信的方法都是一樣的
    /// 具體如何實作可以參看 flutter_plugin_web.dart 中的說明,本例不用這種方式實作,而是用另一種簡單的方式實作 flutter 與 js 的通信(但是無法和 android/ios 插件的介面保持一致)
    var methodChannel = MethodChannel('com.webabcd.flutter/channel2_view$id');
  }

  static dynamic flutterToJs(String message) {
    /// 通過 js.context.callMethod() 呼叫指定的 js 方法,并允許傳遞引數
    return js.context.callMethod('xxx_flutterToJs', [message]);
  }
}

原始碼 https://github.com/webabcd/flutter_demo
作者 webabcd

轉載請註明出處,本文鏈接:https://www.uj5u.com/yidong/552175.html

標籤:其他

上一篇:汽車之家Unity前端通用架構升級實踐

下一篇:返回列表

標籤雲
其他(158833) Python(38125) JavaScript(25413) Java(18025) C(15225) 區塊鏈(8264) C#(7972) AI(7469) 爪哇(7425) MySQL(7177) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5871) 数组(5741) R(5409) Linux(5338) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4570) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2432) ASP.NET(2402) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) .NET技术(1972) 功能(1967) Web開發(1951) HtmlCss(1935) python-3.x(1918) 弹簧靴(1913) C++(1913) xml(1889) PostgreSQL(1875) .NETCore(1860) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • 【從零開始擼一個App】Dagger2

    Dagger2是一個IOC框架,一般用于Android平臺,第一次接觸的朋友,一定會被搞得暈頭轉向。它延續了Java平臺Spring框架代碼碎片化,注解滿天飛的傳統。嘗試將各處代碼片段串聯起來,理清思緒,真不是件容易的事。更不用說還有各版本細微的差別。 與Spring不同的是,Spring是通過反射 ......

    uj5u.com 2020-09-10 06:57:59 more
  • Flutter Weekly Issue 66

    新聞 Flutter 季度調研結果分享 教程 Flutter+FaaS一體化任務編排的思考與設計 詳解Dart中如何通過注解生成代碼 GitHub 用對了嗎?Flutter 團隊分享如何管理大型開源專案 插件 flutter-bubble-tab-indicator A Flutter librar ......

    uj5u.com 2020-09-10 06:58:52 more
  • Proguard 常用規則

    介紹 Proguard 入口,如何查看輸出,如何使用 keep 設定入口以及使用實體,如何配置壓縮,混淆,校驗等規則。

    ......

    uj5u.com 2020-09-10 06:59:00 more
  • Android 開發技術周報 Issue#292

    新聞 Android即將獲得類AirDrop功能:可向附近設備快速分享檔案 谷歌為安卓檔案管理應用引入可安全隱藏資料的Safe Folder功能 Android TV新主界面將顯示電影、電視節目和應用推薦內容 泄露的Android檔案暗示了傳說中的谷歌Pixel 5a與折疊屏新機 谷歌發布Andro ......

    uj5u.com 2020-09-10 07:00:37 more
  • AutoFitTextureView Error inflating class

    報錯: Binary XML file line #0: Binary XML file line #0: Error inflating class xxx.AutoFitTextureView 解決: <com.example.testy2.AutoFitTextureView android: ......

    uj5u.com 2020-09-10 07:00:41 more
  • 根據Uri,Cursor沒有獲取到對應的屬性

    Android: 背景:呼叫攝像頭,拍攝視頻,指定保存的地址,但是回傳的Cursor檔案,只有名稱和大小的屬性,沒有其他諸如時長,連ID屬性都沒有 使用 cursor.getInt(cursor.getColumnIndexOrThrow(MediaStore.Video.Media.DURATIO ......

    uj5u.com 2020-09-10 07:00:44 more
  • Android連載29-持久化技術

    一、持久化技術 我們平時所使用的APP產生的資料,在記憶體中都是瞬時的,會隨著斷電、關機等丟失資料,因此android系統采用了持久化技術,用于存盤這些“瞬時”資料 持久化技術包括:檔案存盤、SharedPreference存盤以及資料庫存盤,還有更復雜的SD卡記憶體儲。 二、檔案存盤 最基本存盤方式, ......

    uj5u.com 2020-09-10 07:00:47 more
  • Android Camera2Video整合到自己專案里

    背景: Android專案里呼叫攝像頭拍攝視頻,原本使用的 MediaStore.ACTION_VIDEO_CAPTURE, 后來因專案需要,改成了camera2 1.Camera2Video 官方demo有點問題,下載后,不能直接整合到專案 問題1.多次拍攝視頻崩潰 問題2.雙擊record按鈕, ......

    uj5u.com 2020-09-10 07:00:50 more
  • Android 開發技術周報 Issue#293

    新聞 谷歌為Android TV開發者提供多種新功能 Android 11將自動填表功能整合到鍵盤輸入建議中 谷歌宣布Android Auto即將支持更多的導航和數字停車應用 谷歌Pixel 5只有XL版本 搭載驍龍765G且將比Pixel 4更便宜 [圖]Wear OS將迎來重磅更新:應用啟動時間 ......

    uj5u.com 2020-09-10 07:01:38 more
  • 海豚星空掃碼投屏 Android 接收端 SDK 集成 六步驟

    掃碼投屏,開放網路,獨占設備,不需要額外下載軟體,微信掃碼,發現設備。支持標準DLNA協議,支持倍速播放。視頻,音頻,圖片投屏。好點意思。還支持自定義基于 DLNA 擴展的操作動作。好像要收費,沒體驗。 這里簡單記錄一下集成程序。 一 跟目錄的build.gradle添加私有mevan倉庫 mave ......

    uj5u.com 2020-09-10 07:01:43 more
最新发布
  • 一統天下 flutter - 插件: flutter 使用 web 原生控制元件,并做資

    原始碼 https://github.com/webabcd/flutter_demo 作者 webabcd 一統天下 flutter - 插件: flutter 使用 web 原生控制元件,并做資料通信 示例如下: lib\plugin\plugin2.dart /* * 插件 * 本例用于演示 flu ......

    uj5u.com 2023-05-11 08:43:32 more
  • 汽車之家Unity前端通用架構升級實踐

    隨著之家3D虛擬化需求的增加,各產品線使用Unity引擎的專案也越來越多,新老專案共存,代碼維護成本也隨之增加。代碼質量參差加之代碼規范仍沒有完全統一產生高昂學習成本進一步加重了專案維護負擔。
    為應對這些問題,我們決定借助主機廠數科產品線銷冠神器VR版本大升級為貧訓,開發一套移動端通用Unity代碼... ......

    uj5u.com 2023-05-10 10:01:15 more
  • 京東小程式折疊屏適配探索

    京東小程式近年來支持了越來越多的業務和應用,做好小程式的折疊屏的適配也是符合未來的發展趨勢,能為用戶和業務方提供更好的體驗和價值。 ......

    uj5u.com 2023-05-09 08:27:55 more
  • Android Studio相關問題

    下載 去官網下載即可,最新版如果運行不了,可選擇安裝其他版本,我安裝的就是4.0版本 建立專案 一般就是建立一個空專案 如果使用過idea,那么建立專案就很簡單。因為Android Studio和 idea 界面都差不多,功能也類似 步驟: File——New——New Project 會出現以下畫 ......

    uj5u.com 2023-05-09 08:27:46 more
  • 汽車之家Unity前端通用架構升級實踐

    隨著之家3D虛擬化需求的增加,各產品線使用Unity引擎的專案也越來越多,新老專案共存,代碼維護成本也隨之增加。代碼質量參差加之代碼規范仍沒有完全統一產生高昂學習成本進一步加重了專案維護負擔。
    為應對這些問題,我們決定借助主機廠數科產品線銷冠神器VR版本大升級為貧訓,開發一套移動端通用Unity代碼... ......

    uj5u.com 2023-05-09 08:27:05 more
  • 京東小程式折疊屏適配探索

    京東小程式近年來支持了越來越多的業務和應用,做好小程式的折疊屏的適配也是符合未來的發展趨勢,能為用戶和業務方提供更好的體驗和價值。 ......

    uj5u.com 2023-05-09 08:26:52 more
  • Android Studio相關問題

    下載 去官網下載即可,最新版如果運行不了,可選擇安裝其他版本,我安裝的就是4.0版本 建立專案 一般就是建立一個空專案 如果使用過idea,那么建立專案就很簡單。因為Android Studio和 idea 界面都差不多,功能也類似 步驟: File——New——New Project 會出現以下畫 ......

    uj5u.com 2023-05-09 08:26:43 more
  • 當我再次用Kotlin完成五年前已經通過Kotlin完成的專案后

    > 近日來對Kotlin的使用頻率越來越高, 也對自己近年來寫過的Kotlin代碼嘗試進行一個簡單的整理. 翻到了自己五年前第一次使用Kotlin來完成的一個專案([貝塞爾曲線](https://juejin.cn/post/6844903556173004807)), 一時興起, 又用發展到現在的 ......

    uj5u.com 2023-05-08 08:30:58 more
  • AutoGPT:有手就會的安裝教程

    AutoGPT 是什么 Auto-GPT 是一個實驗性開源應用程式,展示了 GPT-4 語言模型的功能。該程式由 GPT-4 驅動,將 LLM 的“思想”鏈接在一起,以自主實作您設定的任何目標。作為 GPT-4 完全自主運行的首批示例之一,Auto-GPT 突破了 AI 的可能性界限。 AutoGP ......

    uj5u.com 2023-05-08 08:30:19 more
  • 當我再次用Kotlin完成五年前已經通過Kotlin完成的專案后

    > 近日來對Kotlin的使用頻率越來越高, 也對自己近年來寫過的Kotlin代碼嘗試進行一個簡單的整理. 翻到了自己五年前第一次使用Kotlin來完成的一個專案([貝塞爾曲線](https://juejin.cn/post/6844903556173004807)), 一時興起, 又用發展到現在的 ......

    uj5u.com 2023-05-08 08:29:26 more