flutter 获取通话记录和通讯录
Dart SDK version is 3.7.01
dependencies:flutter:sdk: flutterpermission_handler: ^11.0.1 # 权限管理flutter_contacts: ^1.1.9+2call_log: ^5.0.5cupertino_icons: ^1.0.8dev_dependencies:flutter_test:sdk: flutterflutter_lints: ^5.0.0
2 contact_and_calls_page.dartimport 'package:flutter/material.dart'; import 'package:permission_handler/permission_handler.dart'; import 'package:flutter_contacts/flutter_contacts.dart'; import 'package:call_log/call_log.dart';class ContactAndCallsPage extends StatefulWidget {@override_ContactAndCallsPageState createState() => _ContactAndCallsPageState(); }class _ContactAndCallsPageState extends State<ContactAndCallsPage>with SingleTickerProviderStateMixin {List<Contact> _contacts = [];Iterable<CallLogEntry> _callLogs = [];bool _isLoading = true;String _errorMessage = '';late TabController _tabController;@overridevoid initState() {super.initState();_tabController = TabController(length: 2, vsync: this);_requestPermissions();}@overridevoid dispose() {_tabController.dispose();super.dispose();}// 请求权限Future<void> _requestPermissions() async {setState(() {_isLoading = true;_errorMessage = '';});try {// Android 特定权限处理var contactStatus = await Permission.contacts.request();var phoneStatus = await Permission.phone.request();// 检查是否获得权限if (contactStatus.isGranted || phoneStatus.isGranted) {await _fetchContacts();await _fetchCallLogs();} else {setState(() {_errorMessage = '需要通讯录和通话记录权限以正常使用功能';});_showPermissionDeniedDialog();}} catch (e) {setState(() {_errorMessage = '权限请求发生错误: $e';});print('权限请求错误: $e');} finally {setState(() {_isLoading = false;});}}// 显示权限被拒绝的对话框void _showPermissionDeniedDialog() {showDialog(context: context,builder: (context) => AlertDialog(title: Text('权限受限'),content: Text(_errorMessage),actions: [TextButton(onPressed: () {Navigator.of(context).pop();openAppSettings(); // 打开应用设置},child: Text('打开设置'),),TextButton(onPressed: () => Navigator.of(context).pop(),child: Text('取消'),),],),);}// 获取通讯录联系人Future<void> _fetchContacts() async {try {// 多重权限检查bool permission = await FlutterContacts.requestPermission(readonly: true);if (permission) {// 添加详细配置获取联系人final contacts = await FlutterContacts.getContacts(withProperties: true, // 获取联系人属性withPhoto: false, // 不获取照片以提高性能sorted: true, // 按名称排序);// 过滤掉没有电话号码的联系人final filteredContacts = contacts.where((contact) =>contact.phones.isNotEmpty).toList();setState(() {_contacts = filteredContacts;});print('获取到联系人数量: ${_contacts.length}');} else {setState(() {_errorMessage = '未获得通讯录权限';});}} catch (e) {print('获取通讯录错误: $e');setState(() {_errorMessage = '获取通讯录失败: $e';});}}// 获取通话记录Future<void> _fetchCallLogs() async {try {// 获取通话记录Iterable<CallLogEntry> callLogs = await CallLog.get();// 过滤最近30天的通话记录final now = DateTime.now();final thirtyDaysAgo = now.subtract(Duration(days: 30));setState(() {_callLogs = callLogs.where((log) {final logTime = DateTime.fromMillisecondsSinceEpoch(log.timestamp ?? 0);return logTime.isAfter(thirtyDaysAgo);}).take(100) // 限制最多100条记录.toList();});print('获取到通话记录数量: ${_callLogs.length}');} catch (e) {print('获取通话记录错误: $e');setState(() {_errorMessage = '获取通话记录失败: $e';});}}// 转换通话类型String _getCallType(CallType? callType) {switch (callType) {case CallType.incoming:return '呼入';case CallType.outgoing:return '呼出';case CallType.missed:return '未接';default:return '未知';}}// 格式化通话记录时间String _formatCallLogTime(int? timestamp) {if (timestamp == null) return '未知时间';final dateTime = DateTime.fromMillisecondsSinceEpoch(timestamp);return '${dateTime.year}-${_twoDigits(dateTime.month)}-${_twoDigits(dateTime.day)} ''${_twoDigits(dateTime.hour)}:${_twoDigits(dateTime.minute)}:${_twoDigits(dateTime.second)}';}// 补充两位数方法String _twoDigits(int n) {return n.toString().padLeft(2, '0');}// 格式化通话时长String _formatCallDuration(int? duration) {if (duration == null || duration == 0) return '未接通';final minutes = duration ~/ 60;final seconds = duration % 60;return '$minutes分${_twoDigits(seconds)}秒';}@overrideWidget build(BuildContext context) {return Scaffold(appBar: AppBar(title: Text('通讯录和通话记录'),bottom: TabBar(controller: _tabController,tabs: [Tab(icon: Icon(Icons.contacts), text: '通讯录'),Tab(icon: Icon(Icons.call), text: '通话记录'),],),actions: [IconButton(icon: Icon(Icons.refresh),onPressed: _requestPermissions, // 直接调用权限请求方法),],),body: _isLoading? Center(child: CircularProgressIndicator()): _errorMessage.isNotEmpty? Center(child: Column(mainAxisAlignment: MainAxisAlignment.center,children: [Icon(Icons.error_outline, color: Colors.red, size: 100),SizedBox(height: 20),Text(_errorMessage,style: TextStyle(color: Colors.red),textAlign: TextAlign.center,),SizedBox(height: 20),ElevatedButton(onPressed: _requestPermissions,child: Text('重新获取权限'),)],),): TabBarView(controller: _tabController,children: [_buildContactsList(),_buildCallLogsList(),],),);}// 构建通讯录列表Widget _buildContactsList() {if (_contacts.isEmpty) {return Center(child: Column(mainAxisAlignment: MainAxisAlignment.center,children: [Icon(Icons.contacts, size: 100, color: Colors.grey),SizedBox(height: 20),Text('暂无联系人',style: TextStyle(fontSize: 18, color: Colors.grey),),],),);}return ListView.builder(itemCount: _contacts.length,itemBuilder: (context, index) {Contact contact = _contacts[index];return ListTile(leading: CircleAvatar(backgroundColor: Colors.primaries[index % Colors.primaries.length],child: Text(contact.name.first.isNotEmpty ? contact.name.first[0] : '?',style: TextStyle(fontWeight: FontWeight.bold,color: Colors.white,),),),title: Text(contact.name.first.isNotEmpty ? contact.name.first : '未命名',style: TextStyle(fontWeight: FontWeight.bold),),subtitle: Text(contact.phones.isNotEmpty? contact.phones.first.number: '无电话号码',style: TextStyle(color: Colors.grey[600]),),);},);}// 构建通话记录列表Widget _buildCallLogsList() {if (_callLogs.isEmpty) {return Center(child: Column(mainAxisAlignment: MainAxisAlignment.center,children: [Icon(Icons.call, size: 100, color: Colors.grey),SizedBox(height: 20),Text('暂无通话记录',style: TextStyle(fontSize: 18, color: Colors.grey),),],),);}return ListView.builder(itemCount: _callLogs.length,itemBuilder: (context, index) {CallLogEntry callLog = _callLogs.elementAt(index);return ListTile(leading: Icon(callLog.callType == CallType.missed? Icons.call_missed: (callLog.callType == CallType.incoming? Icons.call_received: Icons.call_made),color: callLog.callType == CallType.missed? Colors.red: Colors.green,),title: Text(callLog.number ?? '未知号码'),subtitle: Column(crossAxisAlignment: CrossAxisAlignment.start,children: [Text('时间: ${_formatCallLogTime(callLog.timestamp)}'),Text('类型: ${_getCallType(callLog.callType)}'),Text('通话时长: ${_formatCallDuration(callLog.duration)}'),],),);},);} }
3 main.dart
import 'package:flutter/material.dart'; import 'contact_and_calls_page.dart';void main() {runApp(MyApp()); }class MyApp extends StatelessWidget {@overrideWidget build(BuildContext context) {return MaterialApp(title: '通讯录和通话记录',theme: ThemeData(primarySwatch: Colors.blue,visualDensity: VisualDensity.adaptivePlatformDensity,),home: ContactAndCallsPage(),);} }
4 降级java sdk到1.8 --- build.gradle.kts
plugins {id("com.android.application")id("kotlin-android")id("dev.flutter.flutter-gradle-plugin") }android {namespace = "com.example.contactlist"compileSdk = flutter.compileSdkVersionndkVersion = flutter.ndkVersioncompileOptions {// Java 8 配置sourceCompatibility = JavaVersion.VERSION_1_8targetCompatibility = JavaVersion.VERSION_1_8// 启用 Core Library DesugaringisCoreLibraryDesugaringEnabled = true}kotlinOptions {// 匹配 Java 版本jvmTarget = JavaVersion.VERSION_1_8.toString()}defaultConfig {applicationId = "com.example.contactlist"minSdk = 21 // 确保 minSdk 不低于 21targetSdk = flutter.targetSdkVersionversionCode = flutter.versionCodeversionName = flutter.versionName// 启用 MultiDexmultiDexEnabled = true}buildTypes {release {signingConfig = signingConfigs.getByName("debug")}} }dependencies {// 升级到 2.1.4 或更高版本coreLibraryDesugaring("com.android.tools:desugar_jdk_libs:2.1.4")// MultiDex 支持implementation("androidx.multidex:multidex:2.0.1") }flutter {source = "../.." }
相关文章:
flutter 获取通话记录和通讯录
Dart SDK version is 3.7.01 dependencies:flutter:sdk: flutterpermission_handler: ^11.0.1 # 权限管理flutter_contacts: ^1.1.92call_log: ^5.0.5cupertino_icons: ^1.0.8dev_dependencies:flutter_test:sdk: flutterflutter_lints: ^5.0.0 2 contact_and_calls_page.da…...
AICon 2024年全球人工智能与大模型开发与应用大会(脱敏)PPT汇总(36份).zip
AICon 2024年全球人工智能与大模型开发与应用大会(脱敏)PPT汇总(36份).zip 1、面向开放域的大模型智能体.pdf 2、企业一站式 AI 智能体构建平台演进实践.pdf 3、PPIO 模型平台出海实战,跨地域业务扩展中的技术优化之道…...
swift菜鸟教程6-10(运算符,条件,循环,字符串,字符)
一个朴实无华的目录 今日学习内容:1.Swift 运算符算术运算符比较运算符逻辑运算符位运算符赋值运算区间运算符其他运算符 2.Swift 条件语句3.Swift 循环4.Swift 字符串字符串属性 isEmpty字符串常量let 变量var字符串中插入值字符串连接字符串长度 String.count使用…...
【14】RUST高级特性
文章目录 不安全操作裸指针应用 不安全函数or方法extern调用外部函数调用C语言函数创建供C调用的接口 全局变量(静态变量)不安全的trait访问联合体中的字段 不安全操作 裸指针 需要程序员保证有效性 从引用创建 let mut num 5; let r1 &num as …...
Linux 系统中 `echo`、`cat`、`tail`、`grep` 四个常用命令介绍
以下是 Linux 系统中 echo、cat、tail、grep 四个常用命令的详细介绍,涵盖其功能、常用选项及实际示例: 1. echo - 输出文本 作用:将文本或变量的值输出到终端或文件。常用于脚本中的信息提示或日志记录。 常用选项: 选项说明-…...
Python 根据多个下标向列表中插入对应的值的巧妙方法:逆序插入
例如根据多个下标(比如2, 5, 8)向列表中插入对应的值,即: 在位置2插入一个值A,在位置5插入一个值B,在位置8插入一个值C, 而且每次插入都会改变列表长度,所以后续位置也会发生偏移。…...
“实时滚动”插件:一个简单的基于vue.js的无缝滚动
1、参考连接: 安装 | vue-seamless-scroll 2、使用步骤: 第一步:安装 yarn add vue-seamless-scroll 第二步:引入 import vueSeamlessScroll from vue-seamless-scroll/src 第三步:注册 components: { vueSeamless…...
【Vue3 + Element-Plus】TreeTransfer树形穿梭框组件
基于 Element Plus 实现高效树形穿梭框组件 组件概述 本组件实现了一个基于 Element Plus 的双树形结构穿梭框,支持以下核心功能: 树形结构数据展示节点多选与批量转移展开状态记忆双向数据同步节点禁用与过滤全选/全不选功能(待完善&#…...
014_多线程
多线程 多线程创建线程方式一:继承Thread类方式二:实现Runable接口方式三:实现Callbale接口 Thread的常用方法线程安全线程同步方式一:同步代码块同步方法方式三:Lock锁 线性池创建线程池处理Runnable任务处理Callable…...
vue自定义颜色选择器
vue自定义颜色选择器 效果图: step0: 默认写法 调用系统自带的颜色选择器 <input type"color">step1:C:\Users\wangrusheng\PycharmProjects\untitled18\src\views\Home.vue <template><div class"container"><!-- 颜…...
(十五)安卓开发中不同类型的view之间继承关系详解
在安卓开发中,View 是所有 UI 组件的基类,不同类别的 View 通过继承关系扩展和特化功能,以满足多样化的界面需求。以下将详细讲解常见 View 类别的继承关系,并结合代码示例和使用场景进行说明。 1. View 继承关系: java.lang.Obj…...
Linux 入门七:从基础到进阶的文件操作
一、Linux 文件系统基础:一切皆文件的哲学 在 Linux 的世界里,“一切皆文件” 是核心设计哲学。无论是普通文件、目录、设备(如硬盘、串口),还是网络套接字,都被抽象为文件模型,通过统一的接口…...
DeepSeek的神经元革命:穿透搜索引擎算法的下一代内容基建
DeepSeek的神经元革命:穿透搜索引擎算法的下一代内容基建 ——从语义网络到价值共识的范式重构 一、搜索引擎的“内容饥渴症”与AI的基建使命 2024年Q1数据显示,百度索引网页总数突破3500亿,但用户点击集中在0.78%的高价值页面。这种“数据…...
【时时三省】(C语言基础)用switch语句实现多分支选择结构 例题
山不在高,有仙则名。水不在深,有龙则灵。 ----CSDN 时时三省 例题: 用switch语句处理菜单命令。在许多应用程序中,用菜单对流程进行控制,例如从键盘输入一个 A 或 a 字符,就会执行A操作,输入一…...
CMake macro
CMake中的macro主要用于在调用处直接展开代码,类似于文本替换,其作用类似于C语言的#define宏,但具备更复杂的结构。以下是详细分析: 1. macro的作用 代码展开:调用宏时,其内容会原地展开,如同…...
高防服务器防御DDoS全解析——从架构设计到实战对抗
本文系统阐述高防服务器对抗DDoS攻击的技术原理与实施路径,深度剖析BGP线路、流量清洗、协议栈优化等关键技术,结合2024年最新攻击案例与Gartner防御框架,提供企业级防御体系构建指南,涵盖硬件选型、策略配置、合规审计等全生命周…...
高防IP如何构筑DDoS防线?_解析抗攻击核心技术体系
本文深度解析高防IP防御DDoS攻击的技术实现路径,涵盖流量清洗机制、智能调度策略、混合防护架构三大核心技术体系。通过2023年某金融平台800Gbps混合攻击实战案例,结合Gartner最新防护成熟度模型,给出高防IP部署的六步实施框架与成本优化方案…...
RDD行动算子和累加器
RDD行动算子: 是能触发真正计算数据的算子 reduce:聚集RDD元素 collect:返回数据集所有元素 foreach:分布式遍历元素 count:返回元素个数: first:返回首个元素 take:返回前n个元素 takeOrdered:返回排序后的前n个元素 aggregate:分区和分区间数据聚合 fol…...
【计算机网络】同步操作 vs 异步操作:核心区别与实战场景解析
📌 引言 在网络通信和分布式系统中,**同步(Synchronous)和异步(Asynchronous)**是两种基础却易混淆的操作模式。本文将通过代码示例、生活类比和对比表格,帮你彻底理解它们的区别与应用场景。 1…...
iframe学习与应用场景指南
一、iframe核心原理与学习路径 1. 嵌套网站的本质原理 技术特性: • 浏览器为每个iframe创建独立的window对象和DOM环境 • 资源独立加载:子页面拥有自己的CSS/JS/Cookie作用域 • 跨域限制:同源策略下无法直接访问DOM(需CORS或…...
基于SSM的线上花店鲜花销售商城网站系统
作者:计算机学姐 开发技术:SpringBoot、SSM、Vue、MySQL、JSP、ElementUI、Python、小程序等,“文末源码”。 专栏推荐:前后端分离项目源码、SpringBoot项目源码、Vue项目源码、SSM项目源码、微信小程序源码 精品专栏:…...
【大模型理论篇】Search-R1: 通过强化学习训练LLM推理与利⽤搜索引擎
最近基于强化学习框架来实现大模型在推理和检索能力增强的项目很多,也是Deep Research技术持续演进的缩影。之前我们讨论过《R1-Searcher:通过强化学习激励llm的搜索能⼒》,今天我们分析下Search-R1【1】。 1. 研究背景与问题 ⼤模型(LLM&a…...
错误码code:9568282 error: install releaseType target not same怎么处理?
目录 1.背景 2.解决方案 1.背景 当前是由于应用从4.1版本升级到5.0版本,然后安装应用会报错9568282 ,如果签名是一致的&#...
qt联动其他库实现一个客户端(本章主要是概述如何实现)
一.服务器功能 1.能连接多个客户端通信 2.负责统计与手机客户端的数据 3.遇到客户端请求数据时能检索数据库并发送对应数据 4.服务器需要能连接到公网 5.服务器需要有账号密码登录功能 6.服务器要有日志与管理员系统能统计信息 二.客户端 1.客户端需要有登录界面 2.客户端需要…...
爱普生FC1610AN5G手机中替代传统晶振的理想之选
在 5G 技术引领的通信新时代,手机性能面临前所未有的挑战与机遇。从高速数据传输到多任务高效处理,从长时间续航到紧凑轻薄设计,每一项提升都离不开内部精密组件的协同优化。晶振,作为为手机各系统提供稳定时钟信号的关键元件&…...
SpringMVC基础二(RestFul、接收数据、视图跳转)
ReauestMapping ReauestMapping注解用于映射url到控制器类或一个特定的处理程序方法。可用于类或方法上,用于类上,表示类中的所有响应请求的方法都是以该地址作为父路径。 创建一个新项目:设置为web项目 编写web.xml(此配置也几…...
BERT - 段嵌入(Segment Embedding)
1. 段嵌入(Segment Embedding)的作用 在BERT模型中,段嵌入的主要作用是区分不同的句子。具体来说: 单句任务:所有位置的段嵌入都是0。 句子对任务:第一个句子的所有位置使用段嵌入0,第二个句子…...
Kaggle-Disaster Tweets-(二分类+NLP+模型融合)
Disaster Tweets 题意: 就是给出一个dataframe包含text这一列代表着文本,文本会有一些词,问对于每条记录中的text是真关于灾难的还是假关于灾难的。 比如我们说今天作业真多,这真是一场灾难。实际上这个灾难只是我们调侃而言的。…...
关于哈希冲突的讨论
文章目录 1. 什么是哈希冲突?2. 为什么会产生哈希冲突?3. 如何解决哈希冲突?4. 为什么哈希算法一定会产生冲突?5. 存在不发生冲突的哈希算法吗?6. 为什么不用无冲突的哈希算法(如完美哈希)&…...
傅利叶发布首款开源人形机器人N1:开发者可实现完整复刻
2025年4月11日,上海——通用机器人公司傅利叶正式发布首款开源人形机器人 Fourier N1,并同步开放涵盖物料清单、设计图纸、装配指南、基础操作软件在内的完整本体资源包。作为傅利叶 “Nexus 开源生态矩阵” 的首个落地项目(“N1” 即 “Nexu…...
2020年INS SCI1区TOP:平衡复合运动优化算法BCMO,深度解析+性能实测
目录 1.摘要2.算法原理3.结果展示4.参考文献5.代码获取 1.摘要 元启发式算法因其强大的鲁棒性和简便的编程方式,在优化领域中发挥着重要作用。本文提出了一种基于平衡复合运动优化算法BCMO,其核心思想是在解空间中平衡个体的复合运动特性。通过概率选择…...
2022年全国职业院校技能大赛 高职组 “大数据技术与应用” 赛项赛卷(3卷)任务书
2022年全国职业院校技能大赛 高职组 “大数据技术与应用” 赛项赛卷(3卷)任务书 背景描述:模块A:大数据平台搭建(容器环境)(15分)任务一:Hadoop HA安装部署任务二&#x…...
Express中间件(Middleware)详解:从零开始掌握(4)
下面我将为你提供四个实战项目的完整实现代码,每个项目都展示了Express中间件的实际应用场景。 1. API网关实现 const express require(express); const rateLimit require(express-rate-limit); const helmet require(helmet); const morgan require(morgan)…...
Ubuntu22环境下,Docker部署阿里FunASR的gpu版本
番外: 随着deepseek的爆火,人工智能相关的开发变得异常火爆,相关的大模型开发很常见的agent智能体需要ASR语音识别的功能,阿里开源的FunASR几乎是把一个商业的项目放给我们使用了。那么我们项目中的生产环境怎么部署gpu版本的语音识别服务呢?经过跟deepseek的一上午的极限…...
vue springboot 案例 收集
vue springboot 案例 收集 SpringbootVue前后端分离项目-管理系统 https://blog.csdn.net/m0_56308072/article/details/130893828...
Windows环境下本地部署deepseek-r1或其他大模型 【保姆级教程】
目录 背景准备工作开始部署下载olloma安装olloma下载deepseek-r1模型使用如何使用 结束语 背景 最近deepseek本地部署的概念越来越火,勾起了我学习的兴趣。 我就在思考如何使用家用机或者平时打游戏的机器来本地部署deepseek,给自己开发个智能体来辅佐…...
ubuntu20.04系统安装apollo10.0系统
文章目录 前言一、安装基础软件1、更新相关软件2 安装 Docker Engine 二、获取 GPU 支持1、安装显卡驱动2、安装 Nvidia container toolkit 三、安装 Apollo 环境管理工具1、安装依赖软件2、在宿主机添加 Apollo 软件源的 gpg key,并设置好源和更新3、安装aem 四、安…...
图片文本识别OCR+DeepSeekapi实现提取图片关键信息
用到的技术: 通过腾讯OCR文字识别,deepseek的api实现 目录 需求分析: 文字识别(OCR)具体实现步骤 起步工作 代码编写 deepseek整合消息,返回文本关键信息 起步工作 编写工具类 具体调用实现 具体…...
minio改成https+域名访问
思路有两个: 方式一:通过nginx反向代理,将https配置在nginx,内部的MinIO还是使用HTTP;方式二:MinIO服务端直接配置成HTTPS; 注意: 私钥需要命名为:private.key 公钥需要…...
unity与usb串口通信(web版)
一、本文介绍在web环境下unity与usb串口进行通信的代码 本篇使用本地服务器作为unity与串口的中介,unity发送数据到服务器,服务器发送给串口收到响应并解析返回给uinty。 使用websocket协议。 注: 1.我的硬件是检测磁阻液位,用…...
UE5每次都打开上一次的工程文件 , 如何取消?
点击左上角 - 文件 点击 打开项目 取消勾选 - 启动时固定加载上次打开的项目...
AI大模型与人类未来的协作图景:从工具到“数字共生体”
📝个人主页🌹:慌ZHANG-CSDN博客 🌹🌹期待您的关注 🌹🌹 一、引言:技术跃迁与文明重塑的十字路口 2020年代以来,人工智能特别是**AI大语言模型(Large Language Models, LLMs)**的迅猛发展,正在从根本上改变人类与技术的关系。从最初的“智能写作助手”到今日…...
C++ I/O 性能优化指南
在高性能计算和大规模数据处理中,I/O 性能优化是提升系统整体效率的关键环节。C 作为一种高性能编程语言,提供了丰富的工具和机制来优化 I/O 操作。本文将详细介绍在 Linux 环境下,如何通过代码层面的优化、系统调用的选择以及多线程技术等手…...
Idea忽略已提交文件
全局忽略 项目根目录下新增.gitignore文件,写入想要忽略的信息,以下可参考 **/src/main/resources/application-local.yamltarget/ !.mvn/wrapper/maven-wrapper.jar !**/src/main/**/target/ !**/src/test/**/target/### IntelliJ IDEA ### .idea/mod…...
Mamba原理及在low-level vision的工作[持续更新]
文章目录 Mamba原理选择性扫描(Selective Retain Information):选择有关/无关信息状态空间模型(SSM)Mamba的选择性保留信息Mamba的扫描操作(The Scan Operation) 硬件感知(Hardware-…...
openlayers入门02 -- 地图控件
地图控件 1.视图跳转控件(ZoomToExtent) 视图跳转控件用于将地图快速跳转到指定的范围。示例: // 视图跳转控件(extent这里用的是学校的经纬度范围,可以按照需要修改) const ZoomToExtent new ol.contro…...
Python 装饰器(Decorator)
文章目录 代码解析1. 装饰器定义 timer(func)2. 应用装饰器 timer **执行流程****关键点****实际应用场景****改进版本(带 functools.wraps)** 这是一个 Python 装饰器(Decorator) 的示例,用于测量函数的执行时间。下…...
UE的AI判断队伍归属的机制:IGenericTeamAgentInterface接口
从官方论坛老哥那学来的,优点在于使用项目设置,像配置碰撞一样,能配置碰撞通道对其他碰撞通道的反应,如阻挡,忽略,重叠,全局配置队伍归属,也能配置当前队伍对其他队伍的身份识别&…...
安宝特新闻丨Vuzix Core™波导助力AR,视角可调、高效传输,优化开发流程
Vuzix Core™ 光波导技术 近期,Vuzix Core™光波导技术赋能AR新视界!该系列镜片支持定制化宽高比调节及20至40视场角范围,可灵活适配各类显示引擎。通过创新的衍射光波导架构,Vuzix Core™实现了光学传输效率与图像质量的双重突破…...
基于springboot留守儿童网站的设计与实现 docx
收藏关注不迷路!! 🌟文末获取源码数据库🌟 感兴趣的可以先收藏起来,还有大家在毕设选题(免费咨询指导选题),项目以及论文编写等相关问题都可以给我留言咨询,希望帮助更多…...