1.uniapp如何使用vue3编写页面
<template>
<view class="content">
<navbar name="navbar组件"></navbar>
<image class="logo" src="/static/logo.png"></image>
<view class="text-area">
<text class="title">{{title}}</text>
</view>
<view class="">
11111
</view>
<button @click="handleClick">点我</button>
<text>总共购买的水果数量{{totalNum}}</text>
<my-component></my-component>
<aComponent></aComponent>
<navbar></navbar>
<view v-for="item in list" :key="item.name">
<view>
1111
</view>
<text>{{item.name}}</text>
<text>{{item.num}}</text>
</view>
</view>
</template>
<script setup>
import aComponent from '../../../project/component/component.vue';
import{ref,reactive,computed} from 'vue'
import{onLoad} from '@dcloudio/uni-app'
const title=ref('Hello')
const list=reactive([
{name:'apple',num:1},
{name:'orange',num:2},
{name:'banana',num:3}
])
const handleClick=()=>{
list.forEach(item=>{
item.num++
})
}
onLoad(()=>{
console.log('onLode生命周期')
})
const totalNum=computed(()=>{
return list.reduce((total,cur)=>total+cur.num,0)
})
</script>
<style>
.content {
display: flex;
flex-direction: column;
align-items: center;
justify-content: center;
}
.logo {
height: 200rpx;
width: 200rpx;
margin-top: 200rpx;
margin-left: auto;
margin-right: auto;
margin-bottom: 50rpx;
}
.text-area {
display: flex;
justify-content: center;
}
.title {
font-size: 36rpx;
color: #8f8f94;
}
</style>
引入组件的三种方式
全局引入
1.引入
import componentVue from './component/component.vue'
2.全局注册
export function createApp() {
const app = createSSRApp(App)
app.component('my-component',componentVue)
return {
app
}
}
3.页面中引入
<my-component></my-component>
局部引入
<script setup>
import aComponent from '../../../project/component/component.vue';
import{ref,reactive,computed} from 'vue'
import{onLoad} from '@dcloudio/uni-app'
const title=ref('Hello')
const list=reactive([
{name:'apple',num:1},
{name:'orange',num:2},
{name:'banana',num:3}
])
const handleClick=()=>{
list.forEach(item=>{
item.num++
})
}
onLoad(()=>{
console.log('onLode生命周期')
})
const totalNum=computed(()=>{
return list.reduce((total,cur)=>total+cur.num,0)
})
</script>
<aComponent></aComponent>
自动引入
新建文件夹然后新建组件
组件引入
<navbar></navbar>
2.uniapp组件通信props和$emit和插槽语法
props
组件
<template>
<view>
navbar组件
</view>
</template>
<script setup>
import { defineProps } from 'vue';
defineProps(['name','content'])
</script>
<style>
</style>
组件引入数据
<navbar name="navbar组件" :content="data"></navbar>
父组件(页面)向子组件传值,如果没传就用默认值
<template>
<view>
navbar组件
<view>组件的name属性{{name}}</view>
<view>组件的content属性{{content}}</view>
</view>
</template>
<script setup>
import { defineProps } from 'vue';
// defineProps(['name','content'])
defineProps({
name:String,
content:{
type:String,
default:()=>{
return '默认值';
}
}
})
</script>
<style>
</style>
$emit
navbar.vue
<template>
<view>
navbar组件
<view>组件的name属性{{name}}</view>
<view>组件的content属性{{content}}</view>
<button @click="handleChange">修改content</button>
</view>
</template><script setup>
import { defineProps,defineEmits } from 'vue';
// defineProps(['name','content'])
defineProps({
name:String,
content:{
type:String,
default:()=>{
return '默认值';
}
}
})
const emit=defineEmits(['changeData'])
const handleChange=()=>{
emit('changeData','修改后的数据')
}</script>
<style>
</style>
<navbar :name="navbar组件" :content="data" @changeData="changeData"></navbar>
插槽语法
<navbar :name="navbar组件" :content="data" @changeData="changeData">
<view>我是插槽的内容</view>
</navbar>
<template>
<view>组件</view>
<view>组件的name属性{{name}}</view>
<view>组件的content属性{{content}}</view>
<slot></slot>
<button @click="handleChange">修改content</button>
</template><script setup>
import { defineProps,defineEmits } from 'vue';
// defineProps(['name','content'])
defineProps({
name:String,
content:{
type:String,
default:()=>{
return '默认值';
}
}
})
const emit=defineEmits(['changeData'])
const handleChange=()=>{
emit('changeData','修改后的数据')
}</script>
<style>
</style>
3.系统参数获取和navBar组件样式动态设计
<template>
<view class="nav">
<view :style="'height:'+status+'rpx;'+containerStyle">
</view>
<view class="navbar" :style="'height:'+navHeight+'rpx;'+containerStyle"></view>
</view>
</template>
<script setup>
import { ref,onBeforeMount,defineProps } 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
},
icoWidth:{
type:String,
default:116
},
iconHeight:{
type:String,
default:38
}
})
onBeforeMount(()=>{
setNavSize()
setStyle()
})
//状态栏高度
const status=ref(0)
//内容高度
const navHeight=ref(0)
//背景颜色
const containerStyle=ref('')
//字体样式
const textStyle=ref('')
//图标的样式
const iconStyle=ref('')
//计算状态栏高度
const setNavSize=()=>{
const {system,statusBarHeight}=uni.getSystemInfoSync()
status.value=statusBarHeight*2
// console.log(res)
const isiOS=system.indexOf('iOS')>-1
if(!isiOS){
navHeight.value=96
}else{
navHeight.value=88
}
// console.log(res)
}
const setStyle=()=>{
containerStyle.value=['background:'+props.background].join(";")
textStyle.value=['color:'+props.color,'font-size'+props.fontSize+'rpx'].join(';')
iconStyle.value=['width:'+props.icoWidth+'rpx','height:'+props.iconHeight+'rpx'].join(';')
}
</script>
<style>
.nav{
position: fixed;
width: 100%;
top: 0;
left: 0;
z-index: 2;
}
</style>
4.页面栈获取和navBar跳转逻辑实现
index.vue
<template>
<view class="content">
<navbar titleText="首页"></navbar>
<button style="margin-top: 130rpx;" @click="navigateTo">跳转</button>
</view>
</template>
<script setup>
import aComponent from '../../../project/component/component.vue';
import{ref,reactive,computed} from 'vue'
import{onLoad} from '@dcloudio/uni-app'
const data=ref("动态数组")
const title=ref('Hello')
const list=reactive([
{name:'apple',num:1},
{name:'orange',num:2},
{name:'banana',num:3}
])
const handleClick=()=>{
list.forEach(item=>{
item.num++
})
}
const changeData=(val)=>{
data.value=val
}
onLoad(()=>{
console.log('onLode生命周期')
})
const totalNum=computed(()=>{
return list.reduce((total,cur)=>total+cur.num,0)
})
const navigateTo=()=>{
console.log(1111)
uni.navigateTo({
url:'/pages/search/index'
})
}
</script>
<style>
.content {
display: flex;
flex-direction: column;
align-items: center;
justify-content: center;
}
.logo {
height: 200rpx;
width: 200rpx;
margin-top: 200rpx;
margin-left: auto;
margin-right: auto;
margin-bottom: 50rpx;
}
.text-area {
display: flex;
justify-content: center;
}
.title {
font-size: 36rpx;
color: #8f8f94;
}
</style>
navbar.vue
<template>
<view class="nav">
<view :style="'height:'+status+'rpx;'+containerStyle"></view>
<view 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" mode=""></image>
<image v-else src="../../static/resource/navbar/ic_home.png" mode=""></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 { ref,onBeforeMount,defineProps } 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
},
icoWidth:{
type:String,
default:116
},
iconHeight:{
type:String,
default:38
},
titleText:{
type:String,
default:''
}
})
onBeforeMount(()=>{
setNavSize()
setStyle()
// const pages = getCurrentPages().length;
// console.log('当前页面栈的长度:', pages);
})
//状态栏高度
const status=ref(0)
//内容高度
const navHeight=ref(0)
//背景颜色
const containerStyle=ref('')
//字体样式
const textStyle=ref('')
//图标的样式
const iconStyle=ref('')
//页面栈的数量
const pages=ref(getCurrentPages().length)
console.log(pages.value,'page')
//计算状态栏高度
const setNavSize=()=>{
const {system,statusBarHeight}=uni.getSystemInfoSync()
status.value=statusBarHeight*2
// console.log(res)
const isiOS=system.indexOf('iOS')>-1
if(!isiOS){
navHeight.value=96
}else{
navHeight.value=88
}
// console.log(res)
}
const setStyle=()=>{
containerStyle.value=['background:'+props.background].join(";")
textStyle.value=['color:'+props.color,'font-size'+props.fontSize+'rpx'].join(';')
iconStyle.value=['width:'+props.icoWidth+'rpx','height:'+props.iconHeight+'rpx'].join(';')
}
const backOrHome=()=>{
if(pages.value>1){
uni.navigateBack();
}else{
uni.switchTab({
url:'/pages/index/index'
})
}
}
</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%);
}
</style>
5.胶囊位置计算和首页navBar显示效果
navbar.vue
<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:0.6">
<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" mode=""></image>
<image v-else src="../../static/resource/navbar/ic_home.png" mode=""></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 { ref,reactive,onBeforeMount,defineProps } 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
},
icoWidth:{
type:String,
default:116
},
iconHeight:{
type:String,
default:38
},
titleText:{
type:String,
default:''
},
isHome:{
type:Boolean,
default:false
}
})
onBeforeMount(()=>{
setNavSize()
setStyle()
// const pages = getCurrentPages().length;
// console.log('当前页面栈的长度:', pages);
})
//状态栏高度
const status=ref(0)
//内容高度
const navHeight=ref(0)
//背景颜色
const containerStyle=ref('')
//字体样式
const textStyle=ref('')
//图标的样式
const iconStyle=ref('')
//页面栈的数量
const pages=ref(getCurrentPages().length)
console.log(pages.value,'page')
//获取胶囊的位置
const menu=reactive(uni.getMenuButtonBoundingClientRect())
//计算状态栏高度
const setNavSize=()=>{
const {system,statusBarHeight}=uni.getSystemInfoSync()
status.value=statusBarHeight*2
// console.log(res)
const isiOS=system.indexOf('iOS')>-1
if(!isiOS){
navHeight.value=96
}else{
navHeight.value=88
}
// console.log(res)
}
//样式设置
const setStyle=()=>{
containerStyle.value=['background:'+props.background].join(";")
textStyle.value=['color:'+props.color,'font-size'+props.fontSize+'rpx'].join(';')
iconStyle.value=['width:'+props.icoWidth+'rpx','height:'+props.iconHeight+'rpx'].join(';')
}
const backOrHome=()=>{
if(pages.value>1){
uni.navigateBack();
}else{
uni.switchTab({
url:'/pages/index/index'
})
}
}
</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>
6.公告效果和utils封装与AppId注册
index.vue
<template>
<view class="content">
<navbar :isHome="true"></navbar>
<view style="margin-top: 160rpx;">
<view class="weui-cell" style="background: #fff9eb;display: flex;flex-direction: row;">
<view class="weui-cell_hd">
<image src="/static/resource/images/ic_myapp.png" style="display: block;width: 40rpx;height: 40rpx;margin-right: 14rpx;"></image>
</view>
<view class="weui-cell_bd">
<text style="color: #be9719;font-size: 13px;">点击右上角"添加到我的小程序",方便下次找到!</text>
</view>
<view class="weui-cell_ft">
<image src="/static/resource/images/modal_closer.png" style="display: block;width: 40rpx;height: 40rpx;margin-right: 14rpx;"></image>
</view>
</view>
</view>
</view>
</template>
<script setup>
import{ref,reactive,computed}from 'vue'
import{onLode}from '@dcloudio/uni-app'
const app=getApp()
onLode(()=>{
app.globalData.utils.getUserInfo()
app.globalData.utils.request({
url:'/app/init',
success:res=>{
console.log(res)
const {id}=res.data.area
app.globalData.utils.request({
url:'/Index/index',
data:{
aid:id
},
success:({data})=>{
console.log(data)
}
})
}
})
})
</script>
<style>
page{
background-color: #fff;
}
</style>
utils.js
class Utils{
constructor(){
this.baseUrl='http://159.75.169.224:7300/pz'
}
//获取用户信息
getUserInfo(){
//调用登录的api
uni.login({
success:res=>{
console.log(res)
this.request({
url:'/auth/wxLogin',
data:{
code:res.code
},
success:res=>{
console.log(res,'res')
}
})
}
})
}
request(option={
showLoadding:false
}){
//判断url是否存在
if(!option.url){
return false
}
//http://159.75.169.224:7300/pz/auth/wxLogin
if(option.showLoadding){
this.showLoading()
}
uni.request({
url:this.baseUrl+option.url,
data:option.data?option.data:{},
header:option.header?option.header:{},
method:option.method?option.method:'GET',
success:(response)=>{
uni.hideLoading()
//后端返回的数据异常
if(response.data.code!=10000){
//将失败的结果返回出去
if(option.fail && typeof option.fail=='function'){
option.fail(response)
}
}
else{
//将成功的结果返回
if(option.success && typeof option.success=='function'){
option.success(response.data)
}
}
},
fail:response=>{
uni.hideLoading()
console.log(response)
}
})
}
showLoading(){
const isShowLoading=uni.getStorageInfoSync('isShowLoading')
if(isShowLoading){
uni.hideLoading()
uni.setStorageSync('isShowLoading',false)
}
uni.showLoading({
title:'加载中...',
complete:function(){
uni.setStorageSync('isShowLoading',true)
},
fail:function(){
uni.setStorageSync('isShowLoading',false)
}
})
}
}
export default new Utils()