flutter webview加载本地文件出现跨域解决方案

一直报错

[INFO:CONSOLE(17)] "Access to image at 'file:///android_asset/flutter_assets/assets/jump/box_bottom.png' from origin 'null' has been blocked by CORS policy: Cross origin requests are only supported for protocol schemes: http, data, chrome, chrome-untrusted, https.", source: file:///android_asset/flutter_assets/assets/jump/index.html (17) 

flutter 的webview封装

下面的是加载本地html文件为网页展示

..loadFlutterAsset("assets/jump/index.html");

 但是打包后的网页内图片访问 就出上面的问题了

想要解决

将图片转换成dataUrl 替换进前端代码中去

在线转换生成DataURL\DataUri、图片转为DataURL\DataUri--查错网

去掉file-loader 

const box_bottom =
  "";

const box_middle =
  "";

const box_top =
  "";

const Dot =
  "";

const express =
  "";

export { box_bottom, box_middle, box_top, Dot, express };

使用

import { express } from "../res/dataUrl";

import 'dart:convert';
import 'package:flutter/material.dart';
import 'package:flutter_screenutil/flutter_screenutil.dart';
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';

import 'package:webview_flutter/webview_flutter.dart';

enum GameStatus {
  ready,
  start,
  stop,
}

class JumpGame extends StatefulWidget {
  const JumpGame({super.key});

  @override
  State<JumpGame> createState() => _LocalViewState();
}

class _LocalViewState extends State<JumpGame> {
  late WebViewController controller;

  GameStatus curStatus = GameStatus.ready;

  @override
  void initState() {
    super.initState();
    controller = WebViewController()
      ..setJavaScriptMode(JavaScriptMode.unrestricted)
      ..setBackgroundColor(const Color(0x00000000))
      ..setNavigationDelegate(
        NavigationDelegate(
          onProgress: (int progress) {
            // Update loading bar.
          },
          onPageStarted: (String url) {},
          onPageFinished: (String url) {},
          onWebResourceError: (WebResourceError error) {},
        ),
      )
      ..addJavaScriptChannel('App', onMessageReceived: onMessageReceived)
      ..loadFlutterAsset("assets/jump/index.html");
  }

  void onMessageReceived(JavaScriptMessage message) async {
    final messageJson = json.decode(message.message);
    print(messageJson);
    final messageMethod = messageJson["method"];
    final messageData = messageJson["data"] ?? {};
    switch (messageMethod) {
      case "start":
        show(
          Column(
            mainAxisSize: MainAxisSize.min,
            children: [
              Text("开始"),
              ElevatedButton(
                  onPressed: () {
                    SmartDialog.dismiss();
                  },
                  child: Text("开始"))
            ],
          ),
        );
        break;
      case 'jumpSuccess':
        // await Future.delayed(const Duration(milliseconds: 500));

        show(
          Center(
            child: Column(
              mainAxisSize: MainAxisSize.min,
              children: [
                ElevatedButton(
                  style:
                      ElevatedButton.styleFrom(backgroundColor: Colors.orange),
                  onPressed: () {
                    SmartDialog.dismiss();
                  },
                  child: Text("继续"),
                ),
              ],
            ),
          ),
        );

        break;
      case 'gameOver':
        // await Future.delayed(const Duration(milliseconds: 500));
        // Get.bottomSheet(
        //   isScrollControlled: true,
        //   ignoreSafeArea: true,
        //   isDismissible: false,
        //   backgroundColor: Colors.transparent,
        //   enableDrag: false,
        //   Container(
        //     height: 1.sh,
        //     width: double.infinity,
        //     color: Colors.red,
        //     child:
        //   ),
        // );

        show(Center(
          child: Column(
            mainAxisSize: MainAxisSize.min,
            children: [
              ElevatedButton(
                style: ElevatedButton.styleFrom(backgroundColor: Colors.orange),
                onPressed: () {
                  SmartDialog.dismiss();
                  controller.reload();
                },
                child: Text("重新开始"),
              ),
            ],
          ),
        ));

        break;
    }
    // ScaffoldMessenger.of(context).showSnackBar(
    //   SnackBar(content: Text(message.message)),
    // );
  }

  void show(Widget child) {
    SmartDialog.show(builder: (context) {
      return Container(
        height: 1.sh,
        width: 1.sw,
        color: Colors.black26,
        alignment: Alignment.center,
        child: child,
      );
    });
  }

  @override
  Widget build(BuildContext context) {
    return SafeArea(
      top: false,
      bottom: false,
      child: Scaffold(
        appBar: AppBar(
          title: const Text(
            '跳一跳',
            style: TextStyle(color: Color(0xff333333)),
          ),
        ),
        body: WebViewWidget(controller: controller),
      ),
    );
  }
}

  • 5
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
要在FlutterWebView下载文件,您需要使用flutter_inappwebview插件。该插件提供了一个DownloadListener接口,可以让您拦截文件下载并处理它。 以下是一个示例代码片段,演示如何在FlutterWebView下载文件: ``` import 'dart:io'; import 'package:flutter/material.dart'; import 'package:flutter_inappwebview/flutter_inappwebview.dart'; class MyWebView extends StatefulWidget { final String url; MyWebView({@required this.url}); @override _MyWebViewState createState() => _MyWebViewState(); } class _MyWebViewState extends State<MyWebView> { InAppWebViewController _webViewController; @override Widget build(BuildContext context) { return Scaffold( appBar: AppBar( title: Text(widget.url), ), body: InAppWebView( initialUrl: widget.url, onWebViewCreated: (InAppWebViewController controller) { _webViewController = controller; }, onLoadStart: (InAppWebViewController controller, String url) {}, onLoadStop: (InAppWebViewController controller, String url) {}, onDownloadStart: (InAppWebViewController controller, String url) async { String fileName = url.split('/').last; String path = await ExtStorage.getExternalStoragePublicDirectory( ExtStorage.DIRECTORY_DOWNLOADS); String fullPath = '$path/$fileName'; await downloadFile(url, fullPath); }, ), ); } Future<void> downloadFile(String url, String path) async { try { var httpClient = HttpClient(); var request = await httpClient.getUrl(Uri.parse(url)); var response = await request.close(); var bytes = await consolidateHttpClientResponseBytes(response); File file = File(path); await file.writeAsBytes(bytes); } catch (err) { print(err); } } } ``` 在上面的代码,我们通过使用DownloadListener接口拦截了文件下载事件。当用户点击下载链接时,我们将文件的URL传递给downloadFile()函数,该函数将文件下载到设备上的Downloads目录。 请注意,这个示例需要使用ext_storage插件来访问设备上的存储空间。如果你需要使用它,只需在pubspec.yaml文件添加依赖项即可: ``` dependencies: ext_storage: ^1.0.3 ``` 请确保在使用该插件之前先导入它: ``` import 'package:ext_storage/ext_storage.dart'; ```

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

肥肥呀呀呀

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值