【uniapp】APP打包上架应用商-注意事项

news2024/11/19 0:24:07

初雪云-uniapp启动图自定义生成(支持一键生成storyboard)

一、修改App端上传图片/视频 uni.uploadFile

在这里插入图片描述

let that=this;
uni.chooseImage({
	count: 1,
	sourceType: ['camera','album'],
	sizeType: ['compressed', 'original'],
	success: rey => {
		uni.showLoading({ title: '上传中',mask: true });
		uni.uploadFile({
			url: that.uploadUrl, // 你的上传图片接口
			filePath: rey.tempFilePaths[0],
			name: 'file',
			success:(uploadFileRes) =>{
				let res = JSON.parse(uploadFileRes.data)
				uni.hideLoading();
				if(res.code==1){
					
				} else{
					that.toast(res.msg)
				}
			}
		})
	},
	fail: (error) => {}
});

二、配置隐私政策提示框

在 manifest.json 中配置,在app-plus中添加

uni-app【Android平台隐私与政策提示框】

"privacy" : {
    "prompt" : "template",
    "template" : {
        "title" : "用户协议和隐私政策",
        "message" : "  请你务必审慎阅读、充分理解“用户协议”和“隐私政策”各条款,包括但不限于:为了更好的向你提供服务,我们需要收集你的设备标识、操作日志等信息用于分析、优化应用性能。<br/>  你可阅读<a href='https://https://xieyi.com'>《用户协议》</a>和<a href='https://https://xieyi.com'>《隐私政策》</a>了解详细信息。如果你同意,请点击下面按钮开始接受我们的服务。",
        "buttonAccept" : "同意并接受",
        "buttonRefuse" : "暂不同意",
        "second" : {
            "title" : "温馨提示",
            "message" : "  进入应用前,你需先同意<a href='https://https://xieyi.com'>《用户协议》</a>和<a href='https://https://xieyi.com'>《隐私政策》</a>,否则将退出应用。",
            "buttonAccept" : "同意并继续",
            "buttonRefuse" : "退出应用"
        }
    }
},

效果图
在这里插入图片描述

三、华为应用市场审核

用户告知权限申请的目的

【插件市场】解决软件在运行时,未见向用户告知权限申请的目的,华为等上架被拒问题

在这里插入图片描述
3.1 在上传图片中使用

chooseImage选择图片的sourceType要自己定义,来判断选择的权限

<template>
	<view class="page">
		<view class="authItem" @tap.stop="judgment">上传图片</view>
		<!-- 用户告知权限申请的目的 -->
		<yk-authpup ref="authpup" type="top" @changeAuth="changeAuth" :permissionID="permissionID"></yk-authpup>
	</view>
</template>

<script>
	var sourceType = [ ['camera'],['album'],['camera','album'] ]
	import ykAuthpup from "@/components/yk-authpup/yk-authpup";
	export default {
		components: { ykAuthpup },
		data() {
			return {
				// 用户告知权限申请的目的
				soIndex: 0,
				permissionID:''
			}
		},
		methods: {
			judgment() { // 判断H5还是App
				// #ifdef APP-PLUS
				let that=this;
				uni.showActionSheet({
					itemList: ['拍摄', '从相册选择'],
					success: function (res) {
						that.soIndex = res.tapIndex;
						if(res.tapIndex==0){ // 相机
							this.permissionID = 'CAMERA';
						} else if(res.tapIndex==1){ // 相册选择
							this.permissionID = 'WRITE_EXTERNAL_STORAGE';
						}
						setTimeout(()=>{ this.$refs['authpup'].open(); },200)
					},
					fail: function (res) { console.log(res.errMsg); }
				});
				// #endif
				// #ifdef H5
				this.choose_img();
				// #endif
			},
			choose_img(){
				var that=this;
				uni.chooseImage({
					count: 1,
					// #ifdef H5
					sourceType: ['camera','album'],
					// #endif
					// #ifdef APP-PLUS
					sourceType: sourceType[this.soIndex],
					// #endif
					sizeType: ['compressed', 'original'],
					success: rey => {
						uni.showLoading({ title: '上传中',mask: true });
						uni.uploadFile({
							url: that.uploadUrl,
							filePath: rey.tempFilePaths[0],
							name: 'file',
							success:(res) =>{
								var data=JSON.parse(res.data);
								if(data.code==1){
								
								} else{
									that.toast(data.msg)
								}
							}
						})
					},
					fail: (error) => {}
				});
			},
		}
	}
</script>

<style></style>

3.2 对yk-authpup插件进行修改- components/yk-authpup/yk-authpup

由于权限拒绝之后,再次点击 权限说明弹窗总是会一闪一闪,所以进行了修改。
[/** ====新增==== */] 是我修改的内容

<template>
	<!-- 用户告知权限申请的目的 -->
	<view v-if="showPopup" class="uni-popup" :style="{top:isNativeHead?'':StatusBar}">
		<view :class="[type, ani, animation ? 'ani' : '']" class="uni-custom uni-popup__wrapper" @click="close(true)">
			<view class="uni-popup__wrapper-box">
				<view class="title">{{authList[permissionID].title}}</view>
				<view class="content">{{authList[permissionID].content}}</view>
			</view>
		</view>
	</view>
</template>

<script>
	export default {
		name: 'YkAuthpup',
		props: {
			// 开启动画
			animation: {
				type: Boolean,
				default: true
			},
			type: {
				type: String,
				default: 'top'
			},
			show: {
				type: Boolean,
				default: true
			},
			//是否是原生头部
			isNativeHead:{
				type: Boolean,
				default: false
			},
			permissionID: {
				type: [String, Number],
				default: ''
			}
		},
		data() {
			return {
				ani: '',
				showPopup: false,
				StatusBar:'',
				refuseNum:'',//拒绝次数,
				authList: {
					'WRITE_EXTERNAL_STORAGE': {
						title: "存储空间/照片权限申请说明",
						content: "便于您使用该功能拍照上传您的照片/视频及用于更换头像、XXXX、XXXX等场景中使用,请您确认授权,否则无法使用该功能"
					},
					'CAMERA':{
						title: "相机/摄像头权限申请说明",
						content: "便于您使用该功能拍照上传您的照片/视频及用于更换头像、XXXX、XXXX等场景中使用,请您确认授权,否则无法使用该功能"
					},
					'ACCESS_FINE_LOCATION': {
						title: "地理位置权限申请说明",
						content: "便于应用程序可以提供基于位置的服务、计算距离、获取经纬度、定位导航、附近搜索等功能,请您确认授权,否则无法使用该功能"
					},
					'CALL_PHONE': {
						title: "拨打电话权限申请说明",
						content: "便于您使用该功能联系、XXXX、XXXX,请您确认授权,否则无法使用该功能"
					}
				},
				/** ====新增==== */
				quanx: {
					'WRITE_EXTERNAL_STORAGE': 0,
					'CAMERA': 0,
					'ACCESS_FINE_LOCATION': 0,
					'CALL_PHONE': 0,
				}
				/** ====新增==== */
			}
		},
		created() {
			// #ifdef APP-PLUS
			this.getSystemInfo();
			// #endif
		},
		methods: {
			//获取状态栏高度
			getSystemInfo() {
				let _this = this;
				uni.getSystemInfo({
					success: function(e) {
						_this.StatusBar = e.statusBarHeight + 'px'; //用于自定义头部时,给手机状态栏留出位置,可通过isNativeHead这个参数控制
					}
				})
			},
			open() {
				this.requestPermissions(this.permissionID);
			},
			close(type) {
				this.ani = '';
				this.$nextTick(() => {
					setTimeout(() => {
						this.showPopup = false;
					}, 300)
				})
			},
			//权限检测
			requestPermissions(permissionID) {
				let _this = this;
				/** ====新增==== */
				let _quanx = uni.getStorageSync('quanx')||_this.quanx;
				/** ====新增==== */
				// #ifdef APP-PLUS
				//判断安卓与ios设备
				if (plus.os.name == 'Android') {
					let _permissionID = 'android.permission.' + permissionID;
					plus.android.checkPermission(_permissionID,
						granted => {
							if (granted.checkResult == -1) {
								console.log(uni.getStorageSync('quanx'))
								//还未授权当前查询的权限,打开权限申请目的自定义弹框
								/** ====新增判断==== */
								if(_quanx[permissionID]==0){
									_this.showPopup = true;
									_this.$nextTick(() => {
										setTimeout(() => {
											_this.ani = 'uni-' + _this.type
										},30)
									})
								}
								/** ====新增==== */
							}
						},
						error => {
							console.log(error.message);
						}
					);
					plus.android.requestPermissions([_permissionID], (e) => {
						//关闭权限申请目的自定义弹框
						_this.$nextTick(() => {
							setTimeout(() => {
								_this.ani = '';
								_this.showPopup = false
							}, 0)
						})
						// console.log(e,'kkkkk')
						if (e.granted.length > 0) { // 已获取的权限
							/** ====新增==== */
							_quanx[permissionID]=1;
							/** ====新增==== */
							//当前查询权限已授权,此时可以通知页面执行接下来的操作
							_this.$emit('changeAuth');
						}
						if (e.deniedAlways.length > 0) { // 永久拒绝申请的权限
							/** ====新增==== */
							_quanx[permissionID]=-1;
							/** ====新增==== */
							//当前查询权限已被永久禁用,此时需要引导用户跳转手机系统设置去开启
							// uni.showModal({
							// 	title: '温馨提示', content: '还没有该权限,立即去设置开启?',
							// 	cancelText: "取消", confirmText: "去设置",
							// 	showCancel: true, confirmColor: '#000', cancelColor: '#666',
							// 	success: (res) => {
							// 		if (res.confirm) {
							// 			_this.goSetting();
							// 		}
							// 	}
							// })
						}
						/** ====新增==== */
						uni.setStorageSync('quanx', _quanx);
						/** ====新增==== */
					})
				} else {
					//IOS不需要添加自定义弹框来描述权限目的,因为在配置文件的隐私信息访问的许可描述里可添加
					//正常可以直接调用uni的API调起权限询问弹框使用各种权限,下面的判断使用场景主要是在IOS禁用某权限后,这个可以判断有无权限,进而引导用户跳转设置开启,仅列出了位置、相册、通讯录、相机、录音等权限,其他IOS权限可具体参考 https://ext.dcloud.net.cn/plugin?id=15787
					let result = 0;
					if (permissionID == 'ACCESS_FINE_LOCATION') {
						//IOS检测位置权限
						let cLLocationManager = plus.ios.importClass("CLLocationManager"),
							authStatus = cLLocationManager.authorizationStatus(),
							enable = cLLocationManager.locationServicesEnabled();
						if (enable && authStatus != 2) {
							result = 1;
						} else {
							result = 0;
						}
						plus.ios.deleteObject(cLLocationManager);
					} else if (permissionID == 'WRITE_EXTERNAL_STORAGE') {
						//IOS检测相册权限
						let PHPhotoLibrary = plus.ios.importClass("PHPhotoLibrary"),
							authStatus = PHPhotoLibrary.authorizationStatus();
						if (authStatus === 3) {
							result = 1;
						} else {
							result = 0;
						}
						plus.ios.deleteObject(PHPhotoLibrary);
					} else if (permissionID == 'CAMERA') {
						//IOS检测相机/摄像头权限
						let avCaptureDevice = plus.ios.importClass("AVCaptureDevice"),
						    authStatus = avCaptureDevice.authorizationStatusForMediaType("vide");
						if (authStatus === 3) {
						  result = 1;
						} else {
						  result = 0;
						}
						plus.ios.deleteObject(avCaptureDevice);
					} else if (permissionID == 'CALL_PHONE') {
						//IOS检测通讯录权限
						let contactStore = plus.ios.importClass("CNContactStore"),
							authStatus = contactStore.authorizationStatusForEntityType(0);
						if (authStatus === 3) {
							result = 1;
						} else {
							result = 0;
						}
						plus.ios.deleteObject(contactStore);
					}else if(permissionID == 'RECORD_AUDIO'){
						//IOS检测麦克风权限
						let aVAudioSession = plus.ios.importClass("AVAudioSession"),
						  aVAudio = aVAudioSession.sharedInstance(),
						  authStatus = aVAudio.recordPermission();
						if ([1684369017, 1970168948].includes(authStatus)) {
						  result = 0;
						} else {
						  result = 1;
						}
						plus.ios.deleteObject(aVAudioSession);
					}
					if (result) {
						//当前查询权限已授权,此时可以通知页面执行接下来的操作
						that.$emit('changeAuth')
					} else {
						//当前查询的权限已禁用,引导用户跳转手机系统设置去开启
						// uni.showModal({
						// 	title: '温馨提示', content: '还没有该权限,立即去设置开启?',
						// 	cancelText: "取消", confirmText: "去设置",
						// 	showCancel: true, confirmColor: '#000', cancelColor: '#666',
						// 	success: (res) => {
						// 		if (res.confirm) {
						// 			_this.goSetting();
						// 		}
						// 	}
						// })
					}
				}
				// #endif
			},
			//跳转手机系统设置
			goSetting() {
				if (plus.os.name == "iOS") {
					var UIApplication = plus.ios.import("UIApplication");
					var application2 = UIApplication.sharedApplication();
					var NSURL2 = plus.ios.import("NSURL");
					var setting2 = NSURL2.URLWithString("app-settings:");
					application2.openURL(setting2);
					plus.ios.deleteObject(setting2);
					plus.ios.deleteObject(NSURL2);
					plus.ios.deleteObject(application2);
				} else {
					var Intent = plus.android.importClass("android.content.Intent");
					var Settings = plus.android.importClass("android.provider.Settings");
					var Uri = plus.android.importClass("android.net.Uri");
					var mainActivity = plus.android.runtimeMainActivity();
					var intent = new Intent();
					intent.setAction(Settings.ACTION_APPLICATION_DETAILS_SETTINGS);
					var uri = Uri.fromParts("package", mainActivity.getPackageName(), null);
					intent.setData(uri);
					mainActivity.startActivity(intent);
				}
			}
		}
	}
</script>

<style lang="scss">
	.uni-popup {
		position: fixed;
		top: 0;
		bottom: 0;
		left: 0;
		right: 0;
		z-index: 99999;
		overflow: hidden;
		&__wrapper {
			position: absolute;
			z-index: 999;
			/* #ifndef APP-NVUE */
			box-sizing: border-box;
			/* #endif */
			&.ani {
				/* #ifndef APP-NVUE */
				transition: all 0.3s;
				/* #endif */
			}
			&.top {
				top: 0;
				width:705rpx;
				/* #ifdef APP-NVUE */
				left:22.5rpx;
				/* #endif */
				/* #ifndef APP-NVUE */
				left:0;
				transform: translateY(-705rpx);
				/* #endif */
			}
			&-box {
				position: relative;
				/* #ifndef APP-NVUE */
				box-sizing: border-box;
				/* #endif */
			}
			&.uni-custom {
				& .uni-popup__wrapper-box {
					width: 705rpx;
					/* #ifndef APP-NVUE */
					margin: 0 22.5rpx;
					/* #endif */
					padding: 30upx;
					background: #fff;
					border: solid 2rpx #ddd;
					/* #ifndef APP-NVUE */
					box-sizing: border-box;
					/* #endif */
					border-radius: 16rpx;
					.title{
						font-size: 32rpx;
						font-weight: bold;
					}
					.content{
						margin-top: 16rpx;
						line-height: 1.6;
					}
				}
				&.top{
					& .uni-popup__wrapper-box {
						width: 705rpx;
					}
				}
			}
			&.uni-top{
				transform: translateY(0);
			}
		}
	}
</style>

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

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

相关文章

VS code的使用介绍

VS code的使用介绍 简介下载和安装常用的插件使用教程快捷键 集成Git未找到 Git。请安装 Git&#xff0c;或在 "git.path" 设置中配置。操作步骤打开文件夹初始化仓库文件版本控制状态提交文件到git打开git操作栏位 好用的插件ChineseDraw.io Integration实体关系 Gi…

案例099:基于微信小程序的外卖小程序的研究与开发

文末获取源码 开发语言&#xff1a;Java 框架&#xff1a;SSM JDK版本&#xff1a;JDK1.8 数据库&#xff1a;mysql 5.7 开发软件&#xff1a;eclipse/myeclipse/idea Maven包&#xff1a;Maven3.5.4 小程序框架&#xff1a;uniapp 小程序开发软件&#xff1a;HBuilder X 小程序…

leetcode 每日一题 2023年12月30日 一周中的第几天

题目 给你一个日期&#xff0c;请你设计一个算法来判断它是对应一周中的哪一天。 输入为三个整数&#xff1a;day、month 和 year&#xff0c;分别表示日、月、年。 您返回的结果必须是这几个值中的一个 {"Sunday", "Monday", "Tuesday", &qu…

数字图像处理 Harris 角点和边缘检测器

一、简述 Harris角点和边缘检测器是一项古老的技术,说它古老是因为该技术从1988年被发明。下面是论文地址,技术出现的时间虽然很久,但是并不代表没有用处了,很多神经网络无法发挥作用的场景下,类似的技术还是会大行其道。 https://web.stanford.edu/class/cs231m/referen…

通过聚道云软件连接器实现钉钉与自研主数据系统的完美融合

客户介绍 某知名高校&#xff0c;拥有数千名教职工&#xff0c;日常管理涉及大量的人员异动信息。该高校设有多个学院和研究所&#xff0c;涵盖了工、理、管、文等多个学科领域。该高校是一所充满活力和潜力的学府&#xff0c;致力于为学生提供优质的教育资源和多元化的学习环…

大型语言模型的幻觉问题

1.什么是大模型幻觉&#xff1f; 在语言模型的背景下&#xff0c;幻觉指的是一本正经的胡说八道&#xff1a;看似流畅自然的表述&#xff0c;实则不符合事实或者是错误的。 幻觉现象的存在严重影响LLM应用的可靠性&#xff0c;本文将探讨大型语言模型(LLMs)的幻觉问题&#x…

[嵌入式C][入门篇] 快速掌握基础2 (数据类型、常量、变量)

开发环境&#xff1a; 网页版&#xff1a;跳转本地开发(Vscode)&#xff1a;跳转 文章目录 一、基本变量大小和范围&#xff08;1&#xff09;在8位/32位单⽚机中&#xff1a;测试代码结果&#xff1a;64位机器结果&#xff1a;32位机器&#xff08;单片机&#xff09;无对齐限…

Web爬虫中CAPTCHA挑战的解决方法

CAPTCHA是网站用来区分人类用户和自动机器人的一种安全措施。它包括向用户提出一些挑战&#xff0c;如扭曲的文本、图像或拼图&#xff0c;用户必须解决以证明其真实性。然而&#xff0c;在Web爬取过程中&#xff0c;遇到验证码可能会带来相当大的挑战。本文将探讨在Web爬虫过程…

使用 C# Winfrom编写倒计时功能

在日常生活中&#xff0c;我们经常需要倒计时来提醒自己重要的时间节点&#xff0c;比如倒计时到达一个特定的日期和时间。介绍一个使用 C# 编写的倒计时应用程序的实现。 步骤一&#xff1a;应用程序的功能 它具有以下几个主要特点&#xff1a; 用户输入目标日期和时间&…

Wrk压测发送Post请求的正确姿势

一、Wrk简介 wrk 是一个能够在单个多核 CPU 上产生显著负载的现代 HTTP 基准测试工具。它采用了多线程设计&#xff0c;并使用了像 epoll 和 kqueue 这样的可扩展事件通知机制。此外&#xff0c;用户可以指定 LuaJIT 脚本来完成 HTTP 请求生成、响应处理和自定义报告等功能。 …

个人笔记:分布式大数据技术原理(二)构建在 Hadoop 框架之上的 Hive 与 Impala

有了 MapReduce&#xff0c;Tez 和 Spark 之后&#xff0c;程序员发现&#xff0c;MapReduce 的程序写起来真麻烦。他们希望简化这个过程。这就好比你有了汇编语言&#xff0c;虽然你几乎什么都能干了&#xff0c;但是你还是觉得繁琐。你希望有个更高层更抽象的语言层来描述算法…

1-01初识C语言

一、概述 C语言是贝尔实验室的Ken Thompson&#xff08;肯汤普逊&#xff09;、Dennis Ritchie&#xff08;丹尼斯里奇&#xff09;等人开发的UNIX 操作系统的“副产品”&#xff0c;诞生于1970年代初。 Thompson和Ritchie共同创作完成了Unix操作系统&#xff0c;他们都被称为…

操作系统----期末单选题

目录 操作系统----期末单选题 一、单选题 操作系统----期末单选题 一、单选题 1&#xff0e; 采用分段存储管理的系统中&#xff0c;若地址用24位表示&#xff0c;其中8位表示段内地址&#xff0c;则允许分段的最大个数是&#xff08; &#xff09;。 A、224 B、216 …

docker安装nodejs,并更改为淘宝源

拉取官方 Node.js 镜像 docker pull node:latest创建 Dockerfile&#xff0c;并更改 NPM 下载源为淘宝源&#xff0c;设置为全局持久化 # 使用最新版本的Node.js作为基础镜像 FROM node:latest# 设置工作目录为/app WORKDIR /app # 更改 NPM 下载源为淘宝源&#xff0c;并设置…

C#不会循环响应的Action设计与实现

目录 一、简述二、测试代码三、测试的输出四、核心代码五、其它 一、简述 特点&#xff1a; 不光是能防止直接的死循环调用&#xff1b;还能防止间接的死循环调用&#xff1b;还支持对不同参数判定&#xff0c;不同参数的调用可以不当循环调用&#xff1b; 消息事件系统中必…

学习笔记——C++运算符之逻辑运算符

作用&#xff1a;用于根据表达式的真值返回真值或假值 逻辑运算符有以下符号&#xff1a; #include<bits/stdc.h> using namespace std; int main(){// 逻辑运算符 非 !int a10;//在c中&#xff0c;除了0均是真 cout<<!a<<endl;//0 cout<<!!a<<…

2023年12月 C/C++(二级)真题解析#中国电子学会#全国青少年软件编程等级考试

C/C++编程(1~8级)全部真题・点这里 第1题:统计指定范围里的数 给定一个数的序列S,以及一个区间[L, R], 求序列中介于该区间的数的个数,即序列中大于等于L且小于等于R的数的个数。 时间限制:1000 内存限制:65536 输入 第一行1个整数n,分别表示序列的长度。(0 < n ≤…

SpringSecurity深度学习

SpringSecurity简介 spring Security是什么&#xff1f; Spring Security 是一个强大且高度可定制的身份验证和访问控制框架&#xff0c;用于保护基于Spring的应用程序。它是Spring项目的一部分&#xff0c;旨在为企业级系统提供全面的安全性解决方案。 一个简单的授权和校验…

idea2023连接gitee远程仓库

目录 1.在gitee创建远程仓库 2.在Idea里配置git 3.初始化本地仓库 4. 提交推送至远程仓库 注意&#xff1a;提前下好git工具、idea2023&#xff0c;注册gitee账号&#xff0c;本文不介绍 1.在gitee创建远程仓库 创建好后&#xff0c;复制远程仓库地址 2.在Idea里配置git ​ …

分布式【Zookeeper】

1.1 ZooKeeper 是什么 ZooKeeper 是 Apache 的顶级项目。ZooKeeper 为分布式应用提供了高效且可靠的分布式协调服务&#xff0c;提供了诸如统一命名服务、配置管理和分布式锁等分布式的基础服务。在解决分布式数据一致性方面&#xff0c;ZooKeeper 并没有直接采用 Paxos 算法&…