一統天下 flutter - 插件: flutter 使用 web 原生控制項,並做數據通信

来源:https://www.cnblogs.com/webabcd/archive/2023/05/09/flutter_lib_plugin_plugin2.html
-Advertisement-
Play Games

源碼 https://github.com/webabcd/flutter_demo 作者 webabcd 一統天下 flutter - 插件: flutter 使用 web 原生控制項,並做數據通信 示例如下: lib\plugin\plugin2.dart /* * 插件 * 本例用於演示 flu ...


源碼 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


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 隨著世界經濟由工業經濟向數字經濟轉型,數據逐步成為關鍵的生產要素,企業開始將數據作為一種戰略資產進行管理。數據從業務中產生,在IT系統中承載,要對數據進行有效治理,需要業務充分參與,IT系統確保遵從,這是一個非常複雜的系統工程。 數據治理架構 實踐證明,企業只有構築一套企業級的數據治理綜合體系,明確 ...
  • 開源之夏 2023 學生報名已經正式開啟!Apache DolphinScheduler 今年繼續參與開源之夏的活動,2023 年 4 月 29 日-6 月 3 日 15:00 UTC+8,同學們可以在開源之夏官網 https://summer-ospp.ac.cn/ 找到 Apache Dolph ...
  • 摘要:金山辦公攜手華為雲完成金山辦公自主研發的“WPS文檔中心系統”與華為雲GaussDB相互相容性測試認證,並獲得華為雲授予的《技術認證書》。 本文分享自華為雲社區《共築數字化未來 金山辦公攜手華為雲完成文檔中心和GaussDB適配》,作者:GaussDB 資料庫。 近日,金山辦公攜手華為雲完成金 ...
  • 更多技術交流、求職機會,歡迎關註位元組跳動數據平臺微信公眾號,回覆【1】進入官方交流群 近期,火山引擎 DataLeap 上線“動態探查”能力,為用戶提供全局數據視角、完善的抽樣策略,提高數據探查的靈活度以及響應速率。 傳統的數據探查是基於庫表的全量探查,由後端引擎執行,通過自動化檢查數據成分、關係、 ...
  • (Oracle 定時任務job實際應用) 一、Oracle定時任務簡介 Oracle定時任務是在oracle系統中一個非常重要的子系統,運用得當,可以大大提高我們系統運行和維護能力。oracle定時任務的功能,可以在指定的時間點自行執行任務。 那麼在實際工作中,什麼樣的場景會用到定時任務呢?下麵是在 ...
  • HUAWEI Health Kit為開發者提供用戶自定義的跑步課程導入介面,便於用戶在華為運動健康App和華為智能穿戴設備上查看來自生態應用的訓練課表,開啟科學、適度的運動訓練。 跑步課程導入能力支持生態應用在獲取用戶的華為帳號授權後,將跑步課程數據寫入至華為運動健康App,併在已有的華為智能穿戴設 ...
  • 京喜APP早期開發主要是快速原生化迭代替代原有H5,提高用戶體驗,在這期間也積累了不少性能問題。之後我們開始進行一些性能優化相關的工作,本文主要是介紹京喜圖片庫相關優化策略以及關於圖片相關的一些關聯知識。 ...
  • 廣告是App開發者最常用的流量變現方法之一,當App擁有一定數量用戶時,開發者就需要考慮如何進行流量變現,幫助App實現商業可持續增長。 鯨鴻動能流量變現服務是廣告服務依托華為終端強大的平臺與數據能力為開發者提供的App流量變現服務,開發者通過該服務可以在自己的App中獲取並向用戶展示精美的、高價值 ...
一周排行
    -Advertisement-
    Play Games
  • 在一些複雜的業務表中間查詢數據,有時候操作會比較複雜一些,不過基於SqlSugar的相關操作,處理的代碼會比較簡單一些,以前我在隨筆《基於SqlSugar的開發框架循序漸進介紹(2)-- 基於中間表的查詢處理》介紹過基於主表和中間表的聯合查詢,而往往實際會比這個會複雜一些。本篇隨筆介紹聯合多個表進行... ...
  • 從按鈕、文本框到下拉框、列表框,WPF提供了一系列常用控制項,每個控制項都有自己獨特的特性和用途。通過靈活的佈局容器,如網格、堆棧面板和換行面板,我們可以將這些控制項組合在一起,實現複雜的界面佈局。而通過樣式和模板,我們可以輕鬆地定製控制項的外觀和行為,以符合我們的設計需求。本篇記錄WPF入門需要瞭解的樣式... ...
  • 以MySQL資料庫為例 # 一. 安裝 NuGet搜索Dapper.Lite並安裝最新版本。 ![](https://img2023.cnblogs.com/blog/174862/202306/174862-20230602155913303-757935399.jpg) NuGet搜索MySql ...
  • # 圖片介面JWT鑒權實現 # 前言 之前做了個返回圖片鏈接的介面,然後沒做授權,然後今天鍵盤到了,也是用JWT來做介面的許可權控制。 然後JTW網上已經有很多文章來說怎麼用了,這裡就不做多的解釋了,如果不懂的可以參考下列鏈接的 文章。 圖片介面文章:[還在愁個人博客沒有圖片放?](https://w ...
  • ![線程各屬性縱覽](https://img2023.cnblogs.com/blog/1220983/202306/1220983-20230603114109107-477345835.png) 如上圖所示,線程有四個屬性: - 線程ID - 線程名稱 - 守護線程 - 線程優先順序 ### 1. ...
  • 本次主要介紹golang中的標準庫`bytes`,基本上參考了 [位元組 | bytes](https://cloud.tencent.com/developer/section/1140520) 、[Golang標準庫——bytes](https://www.jianshu.com/p/e6f7f2 ...
  • 歡迎來到本篇文章!通過上一篇什麼是 Spring?為什麼學它?的學習,我們知道了 Spring 的基本概念,知道什麼是 Spring,以及為什麼學習 Spring。今天,這篇就來說說 Spring 中的核心概念之一 IoC。 ...
  • # 2022版本IDEA+Maven+Tomcat的第一個程式(傻瓜教學) ​ 作為學習Javaweb的一個重要環節,如何實現在IDEA中利用Maven工具創建一個Javaweb程式模版並連接Tomcat發佈是非常重要的。我比較愚鈍(小白),而且自身電腦先前運行過spring或maven的程式,系統 ...
  • 本篇專門扯一下有關 QCheckBox 組件的一個問題。老周不水字數,直接上程式,你看了就明白。 #include <QApplication> #include <QWidget> #include <QPushButton> #include <QCheckBox> #include <QVBo ...
  • # 1.列表數據元素排序 在創建的列表中,數據元素的排列順序常常是無法預測的。這雖然在大多數情況下都是不可避免的,但經常需要以特定的順序呈現信息。有時候希望保留列表數據元素最初的排列順序,而有時候又需要調整排列順序。python提供了很多列表數據元素排序的方式,可根據情況選用。 ## 1.永久性排序 ...