我的第一个flutter项目(Android Webview)

news2025/1/22 15:58:01

前言:flutter开发环境搭建Flutter的开发环境搭建-图解_☆七年的博客-CSDN博客

第一个flutter简单项目,内容是一个主界面,其中:

    1.内容点击数字自增

    2.跳转一个空页,

    3.跳转一个WebView界面

其中涉及添加主键,新建界面类,导入依赖,使用WebView, 兼容http网络安全

 一、主界面 main.dart文件

import 'package:flutter/material.dart';

import 'package:flutterdemo/newpage.dart';

import 'package:flutterdemo/webviewExamlpe.dart';





void main() {

  runApp(const MyApp());

}





/// 这里我们的MyApp是一个类,继承了StatelessWidget

class MyApp extends StatelessWidget {

  const MyApp({super.key});





  /// 这个组件是这个app的根 这是一个无状态部件,然后实现构造方法,

  @override

  Widget build(BuildContext context) {

    ///构造方法里面通过MaterialApp()函数定义风格,然后是标题、主题和主页面信息,

    return MaterialApp(

      title: 'Flutter Demo',

      theme: ThemeData(

        ///这里有一个Colors.blue,你试一下改成red,或者green。

        ///如果你这时候项目是运行在模拟器 或者真机上的话,你可以修改后Ctrl + S 进行保存。就能同步展示

        colorScheme: ColorScheme.fromSeed(seedColor: Colors.green),

        useMaterial3: true,

      ),





      ///  这里主页面home中调用MyHomePage()函数,也就是我们当前页面所显示的内容。

      home: const MyHomePage(title: '主页'),

    );

  }

}





/// 这里MyHomePage继承StatefulWidget,

/// 这是一个有状态的部件,这里就需要一个状态了,

/// 通过createState()得到一个_MyHomePageState,

/// 这个_MyHomePageState()就是这个页面的主要内容了,它里面是

class MyHomePage extends StatefulWidget {

  const MyHomePage({super.key, required this.title});





  final String title;





  @override

  State<MyHomePage> createState() => _MyHomePageState();

}





class _MyHomePageState extends State<MyHomePage> {

  int _counter = 0;





  void _incrementCounter() {

    setState(() {

      _counter++;

    });

  }





  void _goToNewPage() {

    Navigator.push(

      context,

      MaterialPageRoute(builder: (context) => NewPage()),

    );

  }





  // 例如,在主页的某个按钮点击事件中进行页面跳转

  void _goToWebViewPage() {

    Navigator.push(

      context,

      MaterialPageRoute(builder: (context) => WebViewExample()),

    );

  }





  /// 在 build 方法中,我们通常通过对基础 Widget 进行相应的 UI 配置,或是组合各类基础 Widget 的方式进行 UI 的定制化。

  @override

  Widget build(BuildContext context) {

    ///这里返回一个Scaffold,这是一个脚手架,用来构建页面

    return Scaffold(

      ///然后我们看Scaffold中的内容,AppBar 是页面的导航栏,我们直接将 MyHomePage 中的 title 属性作为标题使用。

      appBar: AppBar(

        backgroundColor: Theme.of(context).colorScheme.inversePrimary,





        ///这里我们从App.build方法创建的MyHomePage对象中获取值,并使用它来设置appbar的标题。

        title: Text(widget.title),

      ),





      ///body 主题内容

      body: Center(

        child: Column(

          mainAxisAlignment: MainAxisAlignment.center,

          children: <Widget>[

            const Text(

              '你可以点击按钮增加数字:',

            ),

            Text(

              '$_counter',

              style: Theme.of(context).textTheme.headlineMedium,

            ),

            ///按钮,_incrementCounter 作为其点击处理函数,数字自增。

            ElevatedButton(

              onPressed: () {

                _incrementCounter();

              },

              // child: const Icon(Icons.add),

              child: const Icon(Icons.add),

            ),

            ///按钮。我们将  _goToNewPage 作为其点击处理函数,跳转空白页

            ElevatedButton(

              onPressed: () {

                _goToNewPage();

              },

              // child: const Icon(Icons.add),

              child: Text('click me ,go to newPage'),

            ),

          ],

        ),

      ),





      ///悬浮按钮,则是页面右下角的带“->”的悬浮按钮。我们将  _goToWebViewPager 作为其点击处理函数。

      floatingActionButton: FloatingActionButton(

        onPressed: _goToWebViewPage,

        tooltip: 'Go to New Page',

        child: const Icon(Icons.arrow_forward),

      ),

    );

  }

}




 

二、空页面 newpager.dart

import 'package:flutter/material.dart';

class NewPage extends StatelessWidget {

  const NewPage({super.key});





  @override

  Widget build(BuildContext context) {

    return Scaffold(

      appBar: AppBar(

        title: const Text('New Page'),

      ),

      body: const Center(

        child: Text(

          'This is a new page!',

          style: TextStyle(fontSize: 20),

        ),

      ),

    );

  }

}



三、Webview界面 webViewExample.dart

import 'package:flutter/material.dart';

import 'package:webview_flutter/webview_flutter.dart';





class WebViewExample extends StatefulWidget {

  @override

  _WebViewExampleState createState() => _WebViewExampleState();

}





class _WebViewExampleState extends State<WebViewExample> {

  @override

  Widget build(BuildContext context) {

    return Scaffold(

      appBar: AppBar(

        title: const Text('Web View Example'),

      ),

      body: const WebView(

        initialUrl: 'http://www.baidu.com', // 替换为你想要嵌套的网页地址

        javascriptMode: JavascriptMode.unrestricted, // 允许执行JavaScript代码

      ),

    );

  }

}

其中在pubspec.yaml文件中添加webview_flutter插件依赖: 然后运行flutter pub get来获取插件依赖。

四.可能遇到的问题

1.flutter 打开网页ERR_CLEARTEXT_NOT_PERMITTED

方案: 在`android/app/src/main/AndroidManifest.xml`文件中的`application`标签下添加以下行:

```xml

android:usesCleartextTraffic="true"

```

- 保存文件并重新编译Flutter应用。

https://blog.csdn.net/ly_xiamu/article/details/131931357

2.minSdkVersion is 16 不匹配,直接修改android/app/src/build.gradle 改成对应的如 19

五.编译出apk

1.你可以直接在AS中运行或者项目cmd命令行中flutter build apk命令来生成未签名的APK文件,它将位于Flutter项目的/build/app/outputs/flutter-apk目录下。

2.签名apk

给APK签名—两种方式(flutter android 安装包)_☆七年的博客-CSDN博客

创造价值,乐哉分享!776147358

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/791010.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

选择合适明星代言:确保品牌传播与销售成功的关键一步

在当今激烈的市场竞争中&#xff0c;企业需要不断探索新的营销策略来吸引消费者的关注和忠诚度。其中一种被广泛采用的方法是邀请明星代言产品或品牌。判断想请的明星与自己的产品是否相合适是十分重要的步骤&#xff0c;这关系到代言活动的成功与否。以下是一些方法可以帮助你…

DuckDB全面挑战SQLite

概要 当我们想要在具有嵌入式数据库的本地环境中工作时&#xff0c;我们倾向于默认使用 SQLite。虽然大多数情况下这都很好&#xff0c;但这就像骑自行车去 100 公里之外&#xff1a;可能不是最好的选择。 这篇文章中将讨论以下要点&#xff1a; • DuckDB 简介&#xff1a;它…

企业邮箱默认发信账户用途说明及设置方法

有的时候&#xff0c;企业有多个子公司&#xff0c;或者对内和对外需要用不同的邮箱地址&#xff0c;或者发给不同的人需要用不同的邮箱地址&#xff0c;这个时候企业或用户一般会设置别名邮箱用来区分。 那么问题来了&#xff0c;这么多邮箱账号&#xff0c;我发信的时候默认…

大模型的淘金时代,HPE给出了一份智能经济“奇点”攻略

进入2023年&#xff0c;ChatGPT引发了一个新的AI时代——大模型时代。陆奇说&#xff1a;“我已经跟不上大模型时代的狂飙速度了&#xff01;”大模型引发了AI产业整体升级换代&#xff0c;各种大模型层出不穷&#xff0c;科技公司纷纷入局&#xff0c;AI创业公司再次雨后春笋般…

【Redis深度专题】「核心技术提升」探究Redis服务启动的过程机制的技术原理和流程分析的指南(持久化功能分析)

探究Redis服务启动的过程机制的技术原理和流程分析的指南&#xff08;持久化功能分析&#xff09; Redis提供的持久化机制Redis持久化如何工作Redis持久化的故障分析持久化频率操作分析数据库多久调用一次write&#xff0c;将数据写入内核缓冲区&#xff1f;内核多久将系统缓冲…

算法空间复杂度详解

如果您觉得文章不错&#xff0c;期待你的一键三连哦&#xff0c;你的鼓励是我创作的动力之源&#xff0c;让我们一起加油&#xff0c;一起奔跑&#xff0c;让我们顶峰相见&#xff01;&#xff01;&#xff01; 前言 避免在处理大规模问题时出现效率低下&#xff0c;耗费较多…

STM32 Flash学习(一)

STM32 FLASH简介 不同型号的STM32&#xff0c;其Flash容量也不同。 MiniSTM32开发板选择的STM32F103RCT6的FLASH容量为256K字节&#xff0c;属于大容量产品。 STM32的闪存模块由&#xff1a;主存储器、信息块和闪存存储器接口寄存器等3部分组成。 主存储器&#xff0c;该部分…

服务器被爬虫恶意攻击怎么办?

在有预算的情况可以采购第三方服务防火墙&#xff0c;没钱就使用开源的WAF进行防护。 # WAF防火墙的基本防护原理 WAF&#xff08;Web 应用防火墙&#xff09;可以使用多种技术来防止恶意爬虫攻击&#xff0c;例如&#xff1a; 1. 黑名单&#xff1a;WAF 可以使用黑名单技术来…

会点C++还需要再学Python吗?

提到的C、数据结构与算法、操作系统、计算机网络和数据库技术等确实是计算机科学中非常重要的基础知识领域&#xff0c;对于软件开发和计算机工程师来说&#xff0c;它们是必备的核心知识。掌握这些知识对于开发高性能、可靠和安全的应用程序非常重要。Python作为一种脚本语言&…

Spring Cloud+Spring Boot+Mybatis+uniapp+前后端分离实现知识付费平台免费搭建

Java版知识付费-轻松拥有知识付费平台 多种直播形式&#xff0c;全面满足直播场景需求 公开课、小班课、独立直播间等类型&#xff0c;满足讲师个性化直播场景需求&#xff1b;低延迟、双向视频&#xff0c;亲密互动&#xff0c;无论是互动、答疑&#xff0c;还是打赏、带货、…

【C++初阶】介绍stack_queue及OJ题

介绍stack_queue及OJ题 前言一、简单了解1、stack2、queue 二、OJ题&#xff08;前三个栈&#xff0c;第四、五个队列&#xff09;1、最小栈&#xff08;1&#xff09;题目描述&#xff08;2&#xff09;解题思路&#xff08;3&#xff09;解题代码 2、栈的压入、弹出序列&…

【C#】并行编程实战:异步流

本来这章该讲的是 ASP .NET Core 中的 IIS 和 Kestrel &#xff0c;但是我看了下这个是给服务器用的。而我只是个 Unity 客户端程序&#xff0c;对于服务器的了解趋近于零。 鉴于我对服务器知识和需求的匮乏&#xff0c;这里就不讲原书&#xff08;大部分&#xff09;内容了。本…

基于RPA的自动化流程治理方案探索及应用实践

编者荐语&#xff1a; 随着企业数字化转型进程加快&#xff0c;信息系统大量上线&#xff0c;但流程运营管理问题逐渐显现出来。为提升企业流程运营能力&#xff0c;亚信科技联合某省运营商推出智能化流程治理运营模式&#xff0c;尝试基于RPA&#xff08;机器人流程自动化&am…

IRIS搭建docker

之前把web实现了docker&#xff0c;开发或测试环境可能需要开发自己搭数据库&#xff0c;为了方便使用&#xff0c;把数据库也做一个docker。 由于原生的CentOS我还有改yum仓库&#xff0c;所以这次从之前lis搞的改好yum的镜像开始&#xff08;从改好yum的lisnew的镜像创建lis…

SaaS到底是什么,如何做?这份笔记讲明白了

阅读本篇文章&#xff0c;您将可以了解&#xff1a;1、什么是SaaS&#xff1b;2、SaaS的商业模式&#xff1b;3、SaaS的技术架构&#xff1b;4、国内比较好的SaaS平台。 一、什么是SaaS SaaS即软件即服务&#xff08;Software as a Service&#xff09;&#xff0c;是一种通过…

【数据结构】AVL树/红黑树

目录 1.AVL树&#xff08;高度平衡二叉搜索树&#xff09; 10.1.基本概念 10.2.实现 10.2.1.AVL树节点的定义 10.2.2.AVL树的插入 10.2.3.AVL树的旋转 1.新节点插入较高左子树的左侧---左左&#xff1a;右单旋 2.新节点插入较高右子树的右侧---右右&#xff1a;左单旋 3.新节点…

Python Flask构建微信小程序订餐系统 (十二)

🔥 创建切换商品分类状态的JS文件 🔥 ; var food_act_ops={init:function(){this.eventBind();},eventBind:function(){//表示作用域var that = this;$(".wrap_search select[name=status]").change(function(){$(".wrap_search").submit();});$(&qu…

对ai绘画二次元生成器你有多少了解?

在一个小镇上&#xff0c;有一位年轻的艺术家名叫艾米莉。她是个富有创意的女孩&#xff0c;总是追求着新奇和美妙的艺术体验。然而&#xff0c;她最近遇到了一些创作上的障碍&#xff0c;感觉自己的绘画已经陷入了瓶颈。在艾米莉寻找灵感的过程中&#xff0c;她听说了神秘的ai…

SQL语句(三十二)

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 目录 前言 一、SQL语句类型 二、数据库操作 ​三、数据表操作 1. 数据类型 2. 查看 3. 创建 4. 删除 5. 更改 5.1 表 5.2 列 四、数据操作 4.1 增 4.2 删 4.3 改 4.4 查…

13.5.4 【Linux】常用模块简介

登陆所需要的PAM流程&#xff1a; 上面这个表格当中使用到非常多的 PAM 模块&#xff0c;每个模块的功能都不太相同&#xff0c;详细的模块情报可以在你的系统中找到&#xff1a; /etc/pam.d/*&#xff1a;每个程序个别的 PAM 配置文件&#xff1b; /lib64/security/*&#x…