【uni-app】常用组件(自定义 navbar )和 API

常用组件

uni-app 为开发者提供了一系列基础组件,类似 HTML 里的基础标签元素,但 uni-app 的组件与 HTML 不同,而是与小程序相同,更适合手机端使用。

虽然不推荐使用 HTML 标签,但实际上如果开发者写了div等标签,在编译到非H5平台时也会被编译器转换为 view 标签,类似的还有 spantextanavigator等,包括 css 里的元素选择器也会转,但为了管理方便、策略统一,新写代码时仍然建议使用view等组件。

开发者可以通过组合这些基础组件进行快速开发, 基于内置的基础组件,可以开发各种扩展组件,组件规范与vue组件相同。内置组件


view 视图容器。 它类似于传统html中的div,用于包裹各种元素内容。 如果使用nvue,则需注意,包裹文字应该使用 组件。

swiper 滑块视图容器。 一般用于左右滑动或上下滑动,比如banner轮播图。 注意滑动切换和滚动的区别,滑动切换是一屏一屏的切换。swiper下的每个swiper-item是一个滑动切换区域,不能停留在2个滑动区域之间。

uni-app 支持在 template 模板中嵌套 <template/> <block/>,用来进行 条件渲染 和 列表渲染。 <template/> <block/> 并不是一个组件,它们仅仅是一个包装元素,不会在页面中做任何渲染,只接受控制属性。<block/>在不同的平台表现存在一定差异,推荐统一使用 <template/>

其他常用组件见 uni-app 组件

常用 API

uni-app的 js 代码,h5 端运行于浏览器中,非 h5 端 Android 平台运行在 v8 引擎中,iOS 平台运行在 iOS 自带的 jscore 引擎中。所以,uni-app的 jsAPI 由标准 ECMAScript 的 js API 和 uni 扩展 API 这两部分组成。

ECMAScript 由 Ecma 国际管理,是基础 js 语法。浏览器基于标准 js 扩充了window、document 等 js API;Node.js 基于标准 js 扩充了 fs 等模块;小程序也基于标准 js 扩展了各种 wx.xx、my.xx、swan.xx 的 API。

标准 ecmascript 的 API 非常多,比如:console、settimeout等等。

非 H5 端,虽然不支持 window、document、navigator 等浏览器的 js API,但也支持标准 ECMAScript。

开发者不要把浏览器里的 js 等价于标准 js。

所以 uni-app 的非 H5 端,一样支持标准 js,支持 if、for 等语法,支持字符串、数组、时间等变量及各种处理方法,仅仅是不支持浏览器专用对象。


实例: 自定义 navbar

  • uni.getSystemInfoSync() 同步获取系统数据
  • rpx 即响应式 px,一种根据屏幕宽度自适应的动态单位。以 750 宽的屏幕为基准,750rpx 恰好为屏幕宽度。屏幕变宽,rpx 实际显示效果会等比放大,但在 App(vue2 不含 nvue) 端和 H5(vue2) 端屏幕宽度达到 960px 时,默认将按照 375px 的屏幕宽度进行计算,具体配置参考:rpx 计算配置 。(简单理解为 1px = 2rpx)
  • navigationStyle 默认default展示默认样式。 导航栏样式,仅支持 default/custom。custom即取消默认的原生导航栏,需看使用注意
  • getCurrentPages() 函数用于获取当前页面栈的实例,以数组形式按栈的顺序给出,数组中的元素为页面实例,第一个元素为首页,最后一个元素为当前页面。
  • uni.navigateTo(OBJECT) 保留当前页面,跳转到应用内的某个页面,使用uni.navigateBack可以返回到原页面。
  • uni.switchTab(OBJECT) 跳转到 tabBar 页面,并关闭其他所有非 tabBar 页面。
  • uni.navigateBack(OBJECT) 关闭当前页面,返回上一页面或多级页面。可通过 getCurrentPages() 获取当前的页面栈,决定需要返回几层。
  • uni.getMenuButtonBoundingClientRect() 在小程序平台,如果原生导航栏被隐藏,仍然在右上角会有一个悬浮按钮,微信下也被称为胶囊按钮。本API用于获取小程序下该菜单按钮的布局位置信息,方便开发者布局顶部内容时避开该按钮。
  • navigator 页面跳转。 该组件类似HTML中的<a>组件,但只能跳转本地页面。目标页面必须在pages.json中注册。
  • onLoad比较适合的操作是:接受上页传递过来的参数对象,联网取数据,更新data。
    onLoad((options)=>{ console.log(options, 'options') })
  • uni.getStorageSync(KEY) 从本地缓存(localStorage)中同步获取指定 key 对应的内容。
  • onShow和onHide 注意页面显示,是一个会重复触发的事件。a页面刚进入时,会触发a页面的onShow。当a跳转到b页面时,a会触发onHide,而b会触发onShow。但当b被关闭时,b会触发onUnload,此时a再次显示出现,会再次触发onShow。在tabbar页面(指 pages.json 里配置的tabbar),不同tab页面互相切换时,会触发各自的onShow和onHide。
  • uni.makePhoneCall(OBJECT) 拨打电话。
<template>
	<view class="nav">
		<view :style="'height:' + status + 'rpx;' + containerStyle"></view>
		<view v-if="isHome" class="headNav"
			:style="'height:' + navHeight + 'rpx;line-height:' + navHeight +'rpx;padding-left:20rpx;'">
			<text class="city">中部地区</text>
			<view style="flex: 1">
				<navigator url="../../pages/search/index" :style="
						'height:' +
						menu.height*2 +
						'rpx;line-height:' +
						menu.height*2 +
						'rpx;margin-top:' + 
						(menu.top*2 - status) +
						'rpx;margin-left:32rpx;margin-right:' +
						(menu.width*2 + 24) + 
						'rpx;background: #f4f4f4;border-radius:200rpx;text-align:center'
					">
					<text class="search-text">找医院</text>
				</navigator>
			</view>
		</view>
		<view v-else class="navbar" :style="'height:' + navHeight + 'rpx;' + containerStyle">
			<view class="back-icon" @click="backOrHome">
				<image v-if="pages > 1" src="../../static/resource/navbar/ic_back.png"></image>
				<image v-else src="../../static/resource/navbar/ic_home.png"></image>
			</view>
			<view class="nav-title" v-if="titleText">
				<view :style="'height:' + navHeight + 'rpx;line-height:'+ navHeight + 'rpx;' + textStyle">{{titleText}}
				</view>
			</view>
		</view>
	</view>
</template>

<script setup>
	import {
		onBeforeMount,
		ref,
		reactive
	} from 'vue';
	const props = defineProps({
		background: {
			type: String,
			default: 'rgba(255,255,255,1)'
		},
		color: {
			type: String,
			default: 'rgba(0,0,0,1)'
		},
		fontSize: {
			type: String,
			default: '32'
		},
		iconWidth: {
			type: String,
			default: '116'
		},
		iconHeight: {
			type: String,
			default: '38'
		},
		titleText: {
			type: String,
			default: ''
		},
		isHome: {
			type: Boolean,
			default: false
		}
	})
	const emit = defineEmits(['navBarAttached'])
	onBeforeMount(() => {
		setNavSize()
		setStyle()
		emit('navBarAttached', {
			detail: {
				statusHeight: status.value,
				navHeight: navHeight.value,
				navBarHeight: status.value + navHeight.value
			}
		})
	})
	// 状态栏高度
	const status = ref(0)
	// 内容高度
	const navHeight = ref(0)
	// 计算状态栏的高度
	const setNavSize = () => {
		const {
			system,
			statusBarHeight
		} = uni.getSystemInfoSync()
		// 设计稿宽度为375px,使用 rpx 需要以750px 为基准
		status.value = statusBarHeight * 2
		const isiOS = system.indexOf('iOS') > -1
		if (!isiOS) {
			navHeight.value = 96
		} else {
			navHeight.value = 88
		}
	}
	// 背景颜色
	const containerStyle = ref('')
	// 字体样式
	const textStyle = ref('')
	// 图标样式
	const iconStyle = ref('')
	// 样式设置
	const setStyle = () => {
		containerStyle.value = ['background:' + props.background].join(';')
		textStyle.value = ['color:' + props.color, 'font-size:' + props.fontSize + 'rpx'].join(';')
		iconStyle.value = ['height:' + props.iconHeight + 'rpx', 'width:' + props.iconWidth + 'rpx'].join(';')
	}
	// 页面栈的数量
	const pages = ref(getCurrentPages().length)
	const backOrHome = () => {
		if (pages.value > 1) {
			uni.navigateBack()
		} else {
			uni.switchTab({
				url: '/pages/index/index'
			})
		}
	}
	// 获取胶囊的位置
	const menu = reactive(uni.getMenuButtonBoundingClientRect())
</script>
	
	<style>
		.nav {
			position: fixed;
			width: 100%;
			top: 0;
			left: 0;
			z-index: 2;
		}
	
		.back-icon {
			display: flex;
			align-items: center;
			width: 64rpx;
			height: 100%;
			margin-left: 20rpx;
		}
	
		.back-icon image {
			width: 64rpx;
			height: 64rpx;
		}
	
		.navbar {
			position: relative;
		}
	
		.nav-title {
			position: absolute;
			top: 0;
			left: 50%;
			transform: translate(-50%);
		}
	
		.headNav {
			display: flex;
		}
	
		.city {
			display: inline-block;
			position: relative;
			font-size: 30rpx;
			font-weight: bold;
			padding-left: 55rpx;
			background: url() no-repeat left center;
			background-size: 40rpx;
		}
	
		.city:after {
			content: ' ';
			display: inline-block;
			height: 6px;
			width: 6px;
			border-width: 1px 1px 0 0;
			border-color: #353535;
			border-style: solid;
			-webkit-transform: matrix(0.71, 0.71, -0.71, 0.71, 0, 0);
			transform: matrix(0.71, 0.71, -0.71, 0.71, 0, 0);
			position: relative;
			top: -2px;
			position: absolute;
			top: 50%;
			margin-top: -4px;
			right: -10px;
		}
	
		.search-text {
			display: inline-block;
			padding-left: 30rpx;
			color: #bbbbbb;
			font-size: 26rpx;
			background: url() no-repeat left center;
			background-size: 23rpx;
		}
	</style>

data-*属性是HTML5中增加的元素全局属性,它能让我们在所有HTML元素上嵌入自定义数据属性的能力。并且可以通过js来获取,在Vue的template代码里边用data-变量名使用。

 <text data-title="name" @click="postParams">NAME</text>

data-*通过Js事件可以在dataset中拿到数据:

 postParams(e) {
     console.log(e.currentTarget.dataset.title);
 }

uni.$on(eventName,callback) 监听全局的自定义事件。事件可以由 uni.$emit 触发,回调函数会接收所有传入事件触发函数的额外参数。

    uni.$on('userChange', data => {
        console.log(data)
    })
    //....
    uni.$emit('userChange', res)

uni.chooseAddress(OBJECT) 获取用户收货地址。调起用户编辑收货地址原生界面,并在编辑完成后返回用户选择的地址。(开启此功能需要在 manifest.json 中 mp-weixin 微信小程序相关配置中配置 requiredPrivateInfos Array 地理位置相关接口

在这里插入图片描述

uni.showToast(OBJECT) 显示消息提示框。

  • 9
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

小秀_heo

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值