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

肥肥呀呀呀 2024-09-09 16:03:02 阅读 78

一直报错

[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 

<code>const box_bottom =

"";

const box_middle =

"";

const box_top =

"";

const Dot =

"";code>

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),

),

);

}

}

直接加载本地file形式的网页不推荐 ,这样图片只能转换成base64的,很长而且前端打包出的js很大。

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

 使用另外一种思路。再flutter 的app端内建立一个http服务器,使用http服务器来托管flutter的asset文件。这样做到本地webview内部的网页也能够共享flutter的静态资源。

实现  jump 是html的网页文件路径(图片资源还可以转成webp的)

main.dart 开启http服务

<code> static initLocalWebServer() async {

final server = Jaguar();

server.addRoute(serveFlutterAssets());

await server.serve();

}

webview 设置加载

..loadRequest(Uri.http("127.0.0.1:8080", "jump/index.html"));

https://pub.dev/packages/jaguar_flutter_asset



声明

本文内容仅代表作者观点,或转载于其他网站,本站不以此文作为商业用途
如有涉及侵权,请联系本站进行删除
转载本站原创文章,请注明来源及作者。