vue3 uniapp h5 安卓和iOS开发适配踩坑记录

文章介绍了在uni-app开发中如何处理字体大小以适应不同设备和操作系统,包括利用`--global-font-size`和CSS变量调整字体大小,以及兼容iOS和安卓的状态栏高度和底部安全区域。此外,还提到了uni-app中的常见问题,如position:fixed在iOS中的行为和高度计算方法的注意事项。
摘要由CSDN通过智能技术生成

在这里插入图片描述

font-size适配屏幕大小及iOS和安卓状态栏及安全距离的处理

在这里插入图片描述
在这里插入图片描述
App.vue

<script setup lang="ts">
import { onLaunch, onShow, onHide } from "@dcloudio/uni-app";
import './main.scss'
onLaunch(() => {
  console.log("App Launch");
  var  width = document.documentElement.clientWidth;
  const fontSize = width / 750 + 'px';
  // 这样写是不生效的h5编译运行的文件是有默认的style样式的
  // 一个标签上不能同时出现两个style,不符合编译原则
  // document.documentElement.style.fontSize = fontSize;

  // 解决方法
  // 1. 利用html的默认style来实现
  // 2. 在html上添加--gobal-font-size(注意-gobal-font-size变量仅适用于css文件),设置文字的大小
        // 1. 在  static 文件夹下创建 css/public.css 文件
        // 2. 在 main.ts 文件中引入 public.css (import './static/css/public.css';)
    document.documentElement.className = 'gobal-font-size';
  // 3. 然后将设置文字大小的类添加到html标签上
    document.documentElement.style.setProperty("--gobal-font-size", fontSize);

  // 处理系统不同兼容性问题
  uni.getSystemInfo({
      success: function (res) {
          const statusBarHeight = res.statusBarHeight;
          document.documentElement.style.setProperty("--statusBarHeight", statusBarHeight + 'px');
          console.log('状态栏的高度', statusBarHeight)

          if (res.platform === "android") {
            document.body.classList.add('android');
          } else if (res.platform === "ios") {
            document.body.classList.add('ios');

            // 处理安全区域的距离
            const safeArea = res.safeArea;
            const bottomDistance = res.screenHeight - safeArea.bottom;
            document.documentElement.style.setProperty("--safe-area", bottomDistance + 'px');
            console.log('底部安全区域距离:', bottomDistance);

          } else {
            document.body.classList.add('other');
          }
      }
  })
});
onShow(() => {
  console.log("App Show");
});
onHide(() => {
  console.log("App Hide");
});
</script>
<style>
@import "@/static/fonts/iconfont.css";
</style>

public.css

html {
    --gobal-font-size: 0.45px;
    /*状态栏的高度*/
    --statusBarHeight: 44px;
    /*在竖屏正方向下的安全区域*/
    --safe-area: 34px;
}
.gobal-font-size {
    font-size: var(--gobal-font-size) !important;
}

#app {
    padding-bottom: var(--safe-area);
}

/* 安卓兼容性单独处理*/
.android {
    /* 处理状态栏的间距 */
    padding-top: var(--statusBarHeight);
}
/* ios兼容性单独处理*/
.ios {
    /* 处理状态栏的间距 */
    padding-top: var(--statusBarHeight);
}
/* 其它兼容性单独处理*/
.other {

}

记得在main.ts中引入public.css哟~
在这里插入图片描述
提醒:
能做全局的就千万不要单独写,后续维护是个麻烦。

iOS position:fixed失效

iOS的position: fixed是根据距离最近的scroll来定位的, 如果自定义弹窗是通过position: fixed来实现的,就不要在scroll-view里面调用,可以通过自定义样式(overflow: auto)来实现滚动。
在这里插入图片描述

calc 100vh 计算高度

最好不要使用height: calc(100vh - 44px)这种方式来计算高度,系统不一样,高度的计算方法也不一样,iOS有时候没问题,安卓会莫名的出现滚动条。

body 上使用 padding-top: env(safe-area-inset-top)

只针对主页有用,对tabbar页面无效,tabbar页面要单独处理。

页面设置100vh高度后Android会出现滚动条(body上添加padding处理状态栏的距离后出现的问题)

解决方法:

body {
	box-sizing: border-box;
}

在安卓上浏览器会将body元素的宽高设置为视口的宽高,并将任何添加到body元素上的padding会增加在总宽高上面,就会出现滚动条。

pinyin包实现省市区的选择

使用pinyin-pro包比pinyin包小
在这里插入图片描述
在这里插入图片描述
实现方法:

  1. 调用接口查看省市区的数据
  2. 处理省市区的数据
import {pinyin} from 'pinyin-pro';
// 声明变量存储处理的数据
let list = [];
// 数据处理
if (res.data?.length > 0) {
	res.data?.map(item => {
		// 将中文字符串转换成拼音数组
		const pinyinArray = pinyin(item?.name, {
			// 设置拼音首字母的风格
			style: pinyin.STYLE_FIRST_LETTER
		})
		// 获取拼音数组的第一个字母(这里有些多音字翻译的不准确要单独处理)
		const firstLetter = pinyinArray[0][0];
		// 将获取的第一个字母转换成大写字母
		const upFont = firstLetter.toUpperCase();
		// 判断拼音数组的第一个字母是否存在在list中
		const listIndex = list.findIndex(i => i?.letter === upFont);
		if (listIndex > 0) {
			// 相同首字母的数据合并在一起
			list[listIndex].data = [item, ...list[listIndex].data];
		} else {
			list.push({
				letter: upFont,
				data: [item]
			})
		}
	})
}
// 将处理过的数据从小到大排列
const sortList = list.sort((a, b) => {
	return a.letter.localeCompare(b.letter);
})

手机号实现344格式

在这里插入图片描述
注意:必须用input才能实现时候号码格式化。

<input v-model="mobileText" @input="mobileChange"/>
<div class="clear" v-if="mobileText.length > 0"></div>

mobileChange() {
	// 手机号码
	this.mobile = e.detail.value.replace(/[^0-9_]/g, '');
	// 页面显示的数据
	this.mobileText = this.mobile;
	// 实现344格式
	if (this.mobileText.length > 3 && this.mobileText.length < 8) {
		this.mobileText = this.mobileText.replace(/^(\d{3})/g, '$1 ');
	} else if this.mobileClear.length > 7) {
		this.mobileText = this.mobileText.replace(/^(\d{3})(\d{4})/g, '$1 $2 ')
	}
}

uniapp 微信小程序和h5的小区别之编译内容

最近碰到个微信小程序要改造成h5的项目,微信小程序的文件编译成h5文件之后样式丢失,就nth-child的样式,我本来还以为是配置错了,结果后面发现是编译的标签不一样,下面是一些编译的差别:
微信小程序编译后的文件:
在这里插入图片描述
h5 编译后的文件(直接将所有的uniapp组件都编译成原生的view形式):
在这里插入图片描述
按钮微信小程序编译之后是button,h5编译之后还是uni-button。

表单 Uncaught (in promise) TypeError: Converting circular structure to JSON

这个问题我找了好久,要命~
在这里插入图片描述

透明度只能用小数不能用%

uniapp的opacity不正常识别%,只能用小数,而且安卓真机调试都正常,打包安装后透明度就会极低,70%几乎看不到。

uni-calendar 的使用

在这里插入图片描述

<uni-calendar v-if="showCalendar" class="mt-4" :selected="selectedInfo" :showMonth="false" @change="change"
                  @monthSwitch="monthSwitch"/>
selectedInfo = [];
tasks = null;
showCalendar = false;

created() {
    this.selectedInfo = [];
    onLoad(option => {
        // 查询数据(掉接口查询数据)
        this.getData();
        // 等数据加载完了再显示日历 不然 日历打点加载不出来(要切换月份或者点今日按钮之后才会显示)
        setTimeout(() => {
            this.showCalendar = true;
        }, 1000)
    })
}

getData() {
	this.tasks = [...];
	this.selectedInfo = [
		{
			date: '2024-03-15',
            info: '2条消息',
		}
	];
	console.log('数据查询''日历打点''日历上的数据添加')
}

change(e) {
     console.log('点击的是哪一天', e.fulldate)
}

monthSwitch(e) {
   // 将查询的数据切换成切换的月份和年份
   // 切换月份改变数据
}

页面头部添加操作按钮

在这里插入图片描述
pages.json中配置:在这里插入图片描述
在这里插入图片描述

在这里插入图片描述

自定义tabbar

pages的第一个页面是底部导航栏页面,在App.vue的onShow方法中调用

export default {
    onShow: function () {
        console.log('App Show')
        // pages的第一个页面是导航栏,直接使用这个就行
        // 隐藏tabbar
        uni.hideTabBar();
    }
}

pages的第一个页面不是底部导航栏页面,在每个tabbar页面的的onShow方法中调用

export default class Home extends Vue {
	created(): void {
        // 只加载一次
        onLoad((option) => {

        });
        // 每次都加载
        onShow((option) => {
            // 隐藏tabbar
            uni.hideTabBar();
        })
    }
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值