mirror of
https://gitee.com/lab1024/smart-admin.git
synced 2025-09-17 10:56:39 +08:00
【smart-app更新】1、版本更新记录;2、复杂表单‘3、引入tabs组件
This commit is contained in:
parent
3b31558adb
commit
d170a9d189
@ -14,4 +14,11 @@ export const changeLogApi = {
|
||||
queryPage: (param) => {
|
||||
return postRequest('/support/changeLog/queryPage', param);
|
||||
},
|
||||
|
||||
/**
|
||||
* 详情 @author 卓大
|
||||
*/
|
||||
getDetail: (changeLogId) => {
|
||||
return getRequest(`/support/changeLog/getDetail/${changeLogId}`);
|
||||
},
|
||||
};
|
||||
|
@ -2,7 +2,8 @@
|
||||
"easycom": {
|
||||
"autoscan": true,
|
||||
"custom": {
|
||||
"^uni-(.*)": "@dcloudio/uni-ui/lib/uni-$1/uni-$1.vue"
|
||||
"^uni-(.*)": "@dcloudio/uni-ui/lib/uni-$1/uni-$1.vue",
|
||||
"^y-(.*)": "@/uni_modules/y-$1/components/y-$1.vue"
|
||||
}
|
||||
},
|
||||
"pages": [
|
||||
@ -69,7 +70,7 @@
|
||||
"path" : "pages/notice/notice-detail",
|
||||
"style" :
|
||||
{
|
||||
"navigationBarTitleText" : "通知公告",
|
||||
"navigationBarTitleText" : "公告内容",
|
||||
"enablePullDownRefresh" : false,
|
||||
"navigationBarBackgroundColor": "#fff"
|
||||
}
|
||||
@ -92,6 +93,15 @@
|
||||
"navigationBarBackgroundColor": "#fff"
|
||||
}
|
||||
},
|
||||
{
|
||||
"path" : "pages/support/change-log/change-log-detail",
|
||||
"style" :
|
||||
{
|
||||
"navigationBarTitleText" : "版本更新内容",
|
||||
"enablePullDownRefresh" : false,
|
||||
"navigationBarBackgroundColor": "#fff"
|
||||
}
|
||||
},
|
||||
{
|
||||
"path" : "pages/list/list",
|
||||
"style" :
|
||||
|
@ -1,49 +0,0 @@
|
||||
<template>
|
||||
<view class="font-item-box">
|
||||
<text :class="modelValue==0?'active':''" @click="modelValue=0">标准</text>
|
||||
<view></view>
|
||||
<text :class="modelValue==1?'active':''" @click="modelValue=1">大号</text>
|
||||
<view></view>
|
||||
<text :class="modelValue==2?'active':''" @click="modelValue=2">小号</text>
|
||||
</view>
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import { watch } from 'vue';
|
||||
const emits = defineEmits(['update:modelValue'])
|
||||
const props = defineProps({
|
||||
modelValue:{
|
||||
type:Number,
|
||||
default:0
|
||||
}
|
||||
})
|
||||
|
||||
watch(()=>props.modelValue,(newValue,oldValue)=>{
|
||||
emits('update:modelValue',newValue)
|
||||
})
|
||||
</script>
|
||||
|
||||
<style lang="scss" scoped>
|
||||
.font-item-box {
|
||||
width: 100%;
|
||||
display: flex;
|
||||
justify-content: flex-end;
|
||||
align-items: center;
|
||||
|
||||
text {
|
||||
font-size: 30rpx;
|
||||
color: #cccccc;
|
||||
|
||||
&.active {
|
||||
color: #1A9AFF;
|
||||
}
|
||||
}
|
||||
|
||||
view {
|
||||
width: 4rpx;
|
||||
height: 16rpx;
|
||||
background-color: #e6e6e6;
|
||||
margin: 0 42rpx;
|
||||
}
|
||||
}
|
||||
</style>
|
@ -1,59 +0,0 @@
|
||||
<template>
|
||||
<view class="card-content">
|
||||
<view @click="modelValue = index" :class="modelValue == index?'active':''"
|
||||
v-for="(item,index) in list" :key="index">
|
||||
{{item}}
|
||||
</view>
|
||||
</view>
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import { watch } from 'vue';
|
||||
const emits = defineEmits(['update:modelValue'])
|
||||
const props = defineProps({
|
||||
modelValue:{
|
||||
type:Number,
|
||||
default:0
|
||||
},
|
||||
list:{
|
||||
type:Array,
|
||||
default:[]
|
||||
}
|
||||
})
|
||||
|
||||
watch(()=>props.modelValue,(newValue,oldValue)=>{
|
||||
emits('update:modelValue',newValue)
|
||||
})
|
||||
</script>
|
||||
|
||||
<style lang="scss" scoped>
|
||||
.card-content {
|
||||
padding: 0 30rpx 24rpx;
|
||||
display: flex;
|
||||
flex-wrap: wrap;
|
||||
|
||||
view {
|
||||
box-sizing: border-box;
|
||||
width: 197rpx;
|
||||
height: 72rpx;
|
||||
background: #f7f8f9;
|
||||
border-radius: 8rpx;
|
||||
text-align: center;
|
||||
line-height: 72rpx;
|
||||
margin-right: 24rpx;
|
||||
margin-top: 24rpx;
|
||||
font-size: 30rpx;
|
||||
color: #323333;
|
||||
border: 2rpx solid #f7f8f9;
|
||||
|
||||
&:nth-child(3n) {
|
||||
margin-right: 0;
|
||||
}
|
||||
}
|
||||
|
||||
.active {
|
||||
background: #eff8ff;
|
||||
border: 2rpx solid #2291f9;
|
||||
}
|
||||
}
|
||||
</style>
|
@ -1,60 +0,0 @@
|
||||
<template>
|
||||
<view class="sex-box">
|
||||
<view class="sex-item" :class="modelValue?'active':''" @click="modelValue=true">
|
||||
<uni-icons type="circle" v-if="!modelValue" color="#ccc" size="30"></uni-icons>
|
||||
<uni-icons v-else type="checkbox-filled" color="#1A9AFF" size="30"></uni-icons>
|
||||
<view>
|
||||
男
|
||||
</view>
|
||||
</view>
|
||||
<view class="sex-item" :class="!modelValue?'active':''" @click="modelValue=false">
|
||||
<uni-icons type="circle" v-if="modelValue" color="#ccc" size="30"></uni-icons>
|
||||
<uni-icons v-else type="checkbox-filled" color="#1A9AFF" size="30"></uni-icons>
|
||||
<view>
|
||||
女
|
||||
</view>
|
||||
</view>
|
||||
</view>
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import { watch } from 'vue';
|
||||
const emits = defineEmits(['update:modelValue'])
|
||||
const props = defineProps({
|
||||
modelValue:{
|
||||
type:Boolean,
|
||||
default:true
|
||||
}
|
||||
})
|
||||
|
||||
watch(()=>props.modelValue,(newValue,oldValue)=>{
|
||||
emits('update:modelValue',newValue)
|
||||
})
|
||||
</script>
|
||||
|
||||
<style lang="scss" scoped>
|
||||
.sex-box {
|
||||
width: 100%;
|
||||
display: flex;
|
||||
justify-content: flex-end;
|
||||
|
||||
.sex-item {
|
||||
display: flex;
|
||||
width: 112rpx;
|
||||
height: 64rpx;
|
||||
background-color: #f3f3f3;
|
||||
border: 1rpx solid #ededed;
|
||||
align-items: center;
|
||||
margin-left: 40rpx;
|
||||
justify-content: center;
|
||||
border-radius: 8rpx;
|
||||
font-size: 32rpx;
|
||||
}
|
||||
|
||||
.active {
|
||||
background: #f1f9ff;
|
||||
border: 1rpx solid #1a9aff;
|
||||
color: #1A9AFF;
|
||||
}
|
||||
}
|
||||
</style>
|
@ -1,199 +1,94 @@
|
||||
<template>
|
||||
<view class="">
|
||||
<view class="form-card">
|
||||
<view class="title"> 常用功能 </view>
|
||||
<view class="content">
|
||||
<uni-forms :label-width="100" :modelValue="formData" label-position="left">
|
||||
<uni-forms-item class="uni-forms-item" label="姓名" name="name">
|
||||
<input class="input" type="text" v-model="formData.name" placeholder="请输入姓名" />
|
||||
<view class="smart-form">
|
||||
<uni-forms :label-width="100" :modelValue="formData" label-position="left">
|
||||
<view class="smart-form-group">
|
||||
<view class="smart-form-group-title"> 常用功能 </view>
|
||||
<view class="smart-form-group-content">
|
||||
<uni-forms-item class="smart-form-item" label="姓名" name="name" required>
|
||||
<uni-easyinput trim="all" v-model="formData.name" placeholder="请输入姓名" />
|
||||
</uni-forms-item>
|
||||
<uni-forms-item class="uni-forms-item" label="手机号码" name="name">
|
||||
<input class="input" type="text" v-model="formData.name" placeholder="请输入手机号码" />
|
||||
<uni-forms-item class="smart-form-item" label="手机号码" name="name" required>
|
||||
<uni-easyinput trim="all" v-model="formData.name" placeholder="请输入手机号码" />
|
||||
</uni-forms-item>
|
||||
<uni-forms-item class="uni-forms-item" label="邮箱地址" name="name">
|
||||
<input class="input" type="text" v-model="formData.name" placeholder="请输入邮箱地址" />
|
||||
<uni-forms-item class="smart-form-item" label="邮箱地址" name="name">
|
||||
<uni-easyinput trim="all" v-model="formData.name" placeholder="请输入邮箱地址" />
|
||||
</uni-forms-item>
|
||||
<uni-forms-item class="uni-forms-item" label="性别" name="name">
|
||||
<RadioSex v-model="formData.sex"></RadioSex>
|
||||
<uni-forms-item class="smart-form-item" label="性别" required>
|
||||
<uni-data-checkbox v-model="formData.sex" :localdata="sexs" />
|
||||
</uni-forms-item>
|
||||
<uni-forms-item class="uni-forms-item" label="出生日期" name="name">
|
||||
<view class="item-box">
|
||||
<picker ref="datePickerRef" mode="date" @change="bindDateChange">
|
||||
<input ref="dateInputRef" class="input" type="text" v-model="date" placeholder="点击选择时间" />
|
||||
</picker>
|
||||
</view>
|
||||
<uni-forms-item class="smart-form-item" label="出生日期" name="name">
|
||||
<uni-datetime-picker type="date" return-type="timestamp" v-model="formData.datetimesingle" />
|
||||
</uni-forms-item>
|
||||
<uni-forms-item class="uni-forms-item" label="所在地" name="name">
|
||||
<input class="input" disabled type="text" v-model="formData.name" placeholder="点击选择所在地" />
|
||||
</uni-forms-item>
|
||||
</uni-forms>
|
||||
</view>
|
||||
</view>
|
||||
</view>
|
||||
<view class="form-card">
|
||||
<view class="title"> 推送用户 </view>
|
||||
<view class="content">
|
||||
<uni-forms :label-width="100" :modelValue="formData" label-position="left">
|
||||
<uni-forms-item class="uni-forms-item" label="选择用户" name="name">
|
||||
<view class="item-box" @click="openSelectPeople">
|
||||
<image class="user-select-image" src="/src/static/images/form/add.png" mode=""></image>
|
||||
</view>
|
||||
<view class="smart-form-group">
|
||||
<view class="smart-form-group-title"> 兴趣爱好 </view>
|
||||
<view class="smart-form-group-content">
|
||||
<uni-forms-item class="smart-form-item" label="兴趣爱好" name="interest">
|
||||
<uni-data-checkbox mode="button" multiple v-model="formData.interest" :localdata="interestList"></uni-data-checkbox>
|
||||
</uni-forms-item>
|
||||
</uni-forms>
|
||||
</view>
|
||||
</view>
|
||||
</view>
|
||||
<view class="form-card">
|
||||
<view class="title"> 兴趣爱好 </view>
|
||||
<Interest v-model="formData.interest" :list="interestList"></Interest>
|
||||
</view>
|
||||
<view class="form-card">
|
||||
<view class="title"> 推送用户 </view>
|
||||
<view class="content">
|
||||
<uni-forms :label-width="100" :modelValue="formData" label-position="left">
|
||||
<uni-forms-item class="uni-forms-item" label="亮度调整" name="name">
|
||||
<view class="item-box">
|
||||
<slider style="width: 100%" value="50" activeColor="#2291F9" backgroundColor="#f5f6f8" block-color="#2291F9" block-size="20" />
|
||||
</view>
|
||||
<view class="smart-form-group">
|
||||
<view class="smart-form-group-title"> 屏幕设置 </view>
|
||||
<view class="smart-form-group-content">
|
||||
<uni-forms-item class="smart-form-item" label="亮度调整" name="name">
|
||||
<slider style="width: 100%" value="50" activeColor="#2291F9" backgroundColor="#f5f6f8" block-color="#2291F9" block-size="20" />
|
||||
</uni-forms-item>
|
||||
<uni-forms-item class="uni-forms-item" label="字体大小" name="name">
|
||||
<FontSizeSelece v-model="formData.fontType"></FontSizeSelece>
|
||||
</view>
|
||||
</view>
|
||||
<view class="smart-form-group">
|
||||
<view class="smart-form-group-title"> 自我介绍 </view>
|
||||
<view class="smart-form-group-content">
|
||||
<uni-forms-item class="smart-form-item" label="兴趣爱好" name="interest">
|
||||
<uni-easyinput type="textarea" autoHeight v-model="value" placeholder="请输入自我介绍" />
|
||||
</uni-forms-item>
|
||||
</uni-forms>
|
||||
</view>
|
||||
</view>
|
||||
<view class="form-card">
|
||||
<view class="title"> 自我介绍 </view>
|
||||
<view class="content">
|
||||
<uni-forms :modelValue="formData" label-position="left">
|
||||
<view class="textarea">
|
||||
<textarea auto-height style="font-size: 30rpx" placeholder="请输入自我介绍" placeholder-class="textarea-placeholder" />
|
||||
</view>
|
||||
<view class="example-body">
|
||||
<uni-file-picker limit="9" title="上传图片">
|
||||
<image style="width: 100%; height: 100%" src="/static/images/form/add-image.png" mode=""></image>
|
||||
</uni-file-picker>
|
||||
</view>
|
||||
</uni-forms>
|
||||
|
||||
<uni-forms-item class="smart-form-item" label="上传图片" name="interest">
|
||||
<uni-file-picker limit="9" title="最多选择9张图片" />
|
||||
</uni-forms-item>
|
||||
</view>
|
||||
</view>
|
||||
</uni-forms>
|
||||
<view class="smart-form-submit fixed-bottom-button">
|
||||
<button class="smart-form-submit-btn smart-margin-right20" type="default">取消</button>
|
||||
<button class="smart-form-submit-btn" type="warn">重置</button>
|
||||
<button class="smart-form-submit-btn" type="primary">确定</button>
|
||||
</view>
|
||||
</view>
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import RadioSex from './components/radio-sex.vue';
|
||||
import Interest from './components/interest.vue';
|
||||
import FontSizeSelece from './components/font-size-select.vue';
|
||||
import { reactive, ref } from 'vue';
|
||||
import { reactive } from 'vue';
|
||||
|
||||
const interestList = ['唱歌', '跳舞', 'RAP', '篮球', '音乐', '唱歌', '跳舞', 'RAP', '篮球'];
|
||||
const formData = reactive({
|
||||
interest: 4,
|
||||
fontType: 0,
|
||||
});
|
||||
const hobby = ref('');
|
||||
const date = ref();
|
||||
const bindDateChange = (e) => {
|
||||
date.value = e.detail.value;
|
||||
};
|
||||
const sexs = [
|
||||
{
|
||||
text: '男',
|
||||
value: 0,
|
||||
},
|
||||
{
|
||||
text: '女',
|
||||
value: 1,
|
||||
},
|
||||
{
|
||||
text: '你懂的',
|
||||
value: 2,
|
||||
},
|
||||
];
|
||||
|
||||
const openSelectPeople = () => {
|
||||
uni.navigateTo({
|
||||
url: '/pages/select-people/select-people',
|
||||
});
|
||||
};
|
||||
const interestList = [
|
||||
{ text: '唱歌', value: 1 },
|
||||
{ text: '足球', value: 2 },
|
||||
{ text: '篮球', value: 3 },
|
||||
{ text: '跑步', value: 4 },
|
||||
{ text: '写字', value: 5 },
|
||||
{ text: '美术', value: 6 },
|
||||
{ text: '射击', value: 7 },
|
||||
{ text: '健身', value: 8 },
|
||||
{ text: '马术', value: 9 },
|
||||
{ text: '美食', value: 10 },
|
||||
];
|
||||
const formData = reactive({});
|
||||
</script>
|
||||
|
||||
<style lang="scss" scoped>
|
||||
page {
|
||||
background: #f5f6f8;
|
||||
}
|
||||
|
||||
::v-deep .uni-forms-item__content {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
}
|
||||
|
||||
::v-deep .uni-forms-item__label {
|
||||
font-size: 32rpx;
|
||||
color: #000000;
|
||||
padding-top: 28rpx;
|
||||
}
|
||||
|
||||
::v-deep .uni-forms-item {
|
||||
margin-bottom: 0 !important;
|
||||
}
|
||||
|
||||
::v-deep .uni-slider-thumb {
|
||||
background: #fff !important;
|
||||
border: 10rpx solid #1a9aff;
|
||||
box-sizing: border-box;
|
||||
}
|
||||
|
||||
.uni-forms-item {
|
||||
height: 100rpx;
|
||||
border-bottom: 1rpx solid #ededed;
|
||||
|
||||
&:last-child {
|
||||
border: none;
|
||||
}
|
||||
}
|
||||
|
||||
.form-card {
|
||||
box-sizing: border-box;
|
||||
width: 700rpx;
|
||||
margin: 20rpx auto 0;
|
||||
background: #fff;
|
||||
border-radius: 16rpx;
|
||||
|
||||
.title {
|
||||
width: 100%;
|
||||
height: 84rpx;
|
||||
background-image: url('/static/images/list/form-list.png');
|
||||
background-size: 100% 84rpx;
|
||||
line-height: 84rpx;
|
||||
text-indent: 30rpx;
|
||||
font-size: 32rpx;
|
||||
color: #323333;
|
||||
font-weight: bold;
|
||||
}
|
||||
|
||||
.content {
|
||||
padding: 0 30rpx;
|
||||
}
|
||||
|
||||
.input {
|
||||
font-size: 30rpx;
|
||||
text-align: right;
|
||||
width: 100%;
|
||||
}
|
||||
}
|
||||
|
||||
.item-box {
|
||||
width: 100%;
|
||||
display: flex;
|
||||
justify-content: flex-end;
|
||||
}
|
||||
|
||||
.user-select-image {
|
||||
width: 40rpx;
|
||||
height: 40rpx;
|
||||
}
|
||||
|
||||
.textarea {
|
||||
background: #fcfcfc;
|
||||
border: 0.5px solid #ededed;
|
||||
border-radius: 4px;
|
||||
width: 100%;
|
||||
height: 320rpx;
|
||||
margin-top: 24rpx;
|
||||
padding: 24rpx 30rpx;
|
||||
box-sizing: border-box;
|
||||
.textarea-placeholder {
|
||||
color: #cccccc;
|
||||
font-size: 30rpx;
|
||||
}
|
||||
}
|
||||
|
||||
.example-body {
|
||||
padding-bottom: 24rpx;
|
||||
}
|
||||
</style>
|
||||
<style lang="scss" scoped></style>
|
||||
|
@ -38,10 +38,15 @@
|
||||
<uni-grid-item class="menu-grid">
|
||||
<view class="menu-item" @click="navigateTo('/pages/form/form')">
|
||||
<image class="item-image" src="/@/static/images/index/ic_home_menu6.png"></image>
|
||||
<view class="item-text"> 表单 </view>
|
||||
<view class="item-text"> 复杂表单 </view>
|
||||
</view>
|
||||
</uni-grid-item>
|
||||
<uni-grid-item class="menu-grid">
|
||||
<view class="menu-item" @click="switchTab('/pages/list/list')">
|
||||
<image class="item-image" src="/@/static/images/index/ic_home_menu9.png"></image>
|
||||
<view class="item-text"> 常见列表 </view>
|
||||
</view>
|
||||
</uni-grid-item>
|
||||
|
||||
<uni-grid-item class="menu-grid">
|
||||
<view class="menu-item" @click="navigateTo('/pages/order-detail/order-detail')">
|
||||
<image class="item-image" src="/@/static/images/index/ic_home_menu7.png"></image>
|
||||
@ -54,12 +59,6 @@
|
||||
<view class="item-text"> 优惠券 </view>
|
||||
</view>
|
||||
</uni-grid-item>
|
||||
<uni-grid-item class="menu-grid">
|
||||
<view class="menu-item" @click="navigateTo('/pages/list/list')">
|
||||
<image class="item-image" src="/@/static/images/index/ic_home_menu9.png"></image>
|
||||
<view class="item-text"> 精品课程 </view>
|
||||
</view>
|
||||
</uni-grid-item>
|
||||
<uni-grid-item class="menu-grid">
|
||||
<view class="menu-item" @click="navigateTo('/pages/change-log/change-log-list')">
|
||||
<image class="item-image" src="/@/static/images/index/ic_home_menu10.png"></image>
|
||||
@ -82,6 +81,11 @@
|
||||
url,
|
||||
});
|
||||
}
|
||||
function switchTab(url) {
|
||||
uni.switchTab({
|
||||
url,
|
||||
});
|
||||
}
|
||||
</script>
|
||||
|
||||
<style lang="scss" scoped>
|
||||
|
@ -66,5 +66,5 @@ const tabsList = [{
|
||||
color: #ccc;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
</style>
|
||||
|
79
smart-app/src/pages/support/change-log/change-log-detail.vue
Normal file
79
smart-app/src/pages/support/change-log/change-log-detail.vue
Normal file
@ -0,0 +1,79 @@
|
||||
<template>
|
||||
<view class="container">
|
||||
<view class="title">
|
||||
<uni-title type="h1" align="center" :title="detail.title"></uni-title>
|
||||
<uni-title type="h4" align="center" color="#999999" :title="detail.subTitle"></uni-title>
|
||||
</view>
|
||||
<view class="content">
|
||||
<rich-text :nodes="detail.content" />
|
||||
</view>
|
||||
</view>
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import { inject, reactive } from 'vue';
|
||||
import { changeLogApi } from '/@/api/support/change-log-api';
|
||||
import { onLoad } from '@dcloudio/uni-app';
|
||||
import { smartSentry } from '/@/lib/smart-sentry';
|
||||
|
||||
const smartEnumPlugin = inject('smartEnumPlugin');
|
||||
|
||||
const detail = reactive({
|
||||
title: '',
|
||||
subTitle: '',
|
||||
content: '',
|
||||
});
|
||||
|
||||
async function getDetail(changeLogId) {
|
||||
try {
|
||||
uni.showLoading({ title: '加载中' });
|
||||
let res = await changeLogApi.getDetail(changeLogId);
|
||||
detail.title = res.data.version + '版本' + smartEnumPlugin.getDescByValue('CHANGE_LOG_TYPE_ENUM', res.data.type);
|
||||
detail.content =
|
||||
'<pre style="' +
|
||||
'line-height: 18px;\n' +
|
||||
'font-size: 14px;\n' +
|
||||
'white-space: pre-wrap;\n' +
|
||||
' white-space: -moz-pre-wrap;\n' +
|
||||
' white-space: -pre-wrap;\n' +
|
||||
' white-space: -o-pre-wrap;\n' +
|
||||
' word-wrap: break-word;">' +
|
||||
res.data.content +
|
||||
'</pre>';
|
||||
let subTitleArray = [];
|
||||
if (res.data.publishAuthor) {
|
||||
subTitleArray.push(res.data.publishAuthor);
|
||||
}
|
||||
if (res.data.publicDate) {
|
||||
subTitleArray.push(res.data.publicDate);
|
||||
}
|
||||
detail.subTitle = subTitleArray.join(' | ');
|
||||
} catch (e) {
|
||||
smartSentry.captureError(e);
|
||||
} finally {
|
||||
uni.hideLoading();
|
||||
}
|
||||
}
|
||||
|
||||
onLoad((option) => {
|
||||
uni.pageScrollTo({
|
||||
scrollTop: 0,
|
||||
});
|
||||
getDetail(option.changeLogId);
|
||||
});
|
||||
</script>
|
||||
|
||||
<style lang="scss" scoped>
|
||||
.container {
|
||||
height: 100vh;
|
||||
padding: 20rpx;
|
||||
|
||||
.content {
|
||||
border-top: #cccccc 1px solid;
|
||||
margin-top: 50rpx;
|
||||
padding: 50rpx 16rpx 50rpx 16rpx;
|
||||
line-height: 30px;
|
||||
color: #333333;
|
||||
}
|
||||
}
|
||||
</style>
|
@ -25,23 +25,20 @@
|
||||
<view class="list-container">
|
||||
<view class="list-item" @click="gotoDetail(item.changeLogId)" v-for="item in listData" :key="item.changeLogId">
|
||||
<view class="list-item-row">
|
||||
<view class="list-item-content bolder">{{ item.version }}</view>
|
||||
<view class="list-item-content bolder"
|
||||
>{{ item.version }}版本{{ $smartEnumPlugin.getDescByValue('CHANGE_LOG_TYPE_ENUM', item.type) }}</view
|
||||
>
|
||||
<uni-tag
|
||||
:text="$smartEnumPlugin.getDescByValue('CHANGE_LOG_TYPE_ENUM', item.type)"
|
||||
:type="$smartEnumPlugin.getObjectByValue('CHANGE_LOG_TYPE_ENUM', item.type).type"
|
||||
/>
|
||||
</view>
|
||||
<view class="list-item-row">
|
||||
<view class="list-item-label">发布日期:{{ item.publicDate }} - {{ item.publishAuthor }}</view>
|
||||
</view>
|
||||
<view class="list-item-row">
|
||||
<view class="list-item-label">{{ item.content }}</view>
|
||||
<view class="list-item-label">发布日期:{{ item.publicDate }}</view>
|
||||
</view>
|
||||
</view>
|
||||
</view>
|
||||
</mescroll-body>
|
||||
|
||||
<uni-fab ref="fab" :pattern="fabPattern" horizontal="right" @fabClick="gotoAdd" />
|
||||
</view>
|
||||
</template>
|
||||
|
||||
@ -121,7 +118,7 @@
|
||||
// --------------------------- 详情 ---------------------------------
|
||||
|
||||
function gotoDetail(id) {
|
||||
uni.navigateTo({ url: '/pages/enterprise/enterprise-detail?enterpriseId=' + id });
|
||||
uni.navigateTo({ url: '/pages/support/change-log/change-log-detail?changeLogId=' + id });
|
||||
}
|
||||
</script>
|
||||
|
||||
@ -172,9 +169,10 @@
|
||||
font-size: 30rpx;
|
||||
font-weight: 400;
|
||||
text-align: left;
|
||||
color: $uni-text-color-grey;
|
||||
}
|
||||
.bolder {
|
||||
font-weight: 600 !important;
|
||||
font-weight: 500 !important;
|
||||
font-size: 34rpx !important;
|
||||
}
|
||||
.list-item-content {
|
||||
|
@ -136,9 +136,9 @@
|
||||
}
|
||||
|
||||
.smart-form-item {
|
||||
height: 100rpx;
|
||||
min-height: 100rpx;
|
||||
height: auto;
|
||||
padding-bottom: 24rpx;
|
||||
//border-bottom: 1rpx solid #ededed;
|
||||
align-items: center;
|
||||
&:last-child {
|
||||
border: none;
|
||||
@ -191,4 +191,9 @@
|
||||
flex: 1;
|
||||
}
|
||||
}
|
||||
|
||||
.fixed-bottom-button {
|
||||
position: fixed;
|
||||
bottom: 0;
|
||||
}
|
||||
}
|
||||
|
464
smart-app/src/uni_modules/y-tabs/components/css/index.scss
Normal file
464
smart-app/src/uni_modules/y-tabs/components/css/index.scss
Normal file
@ -0,0 +1,464 @@
|
||||
.y-tabs {
|
||||
position: relative;
|
||||
display: block;
|
||||
|
||||
// 标签栏垂直方位下的根容器样式
|
||||
&.is-vertical {
|
||||
display: flex;
|
||||
flex-wrap: nowrap;
|
||||
flex-direction: row;
|
||||
justify-content: flex-end;
|
||||
flex: 1;
|
||||
|
||||
// 垂直时标签栏scroll-view的子项垂直排列
|
||||
.y-tabs__scroll {
|
||||
flex-direction: column;
|
||||
}
|
||||
}
|
||||
|
||||
// 区域滚动下的滚动导航
|
||||
&.is-areaScroll.is-scrollNav {
|
||||
display: flex;
|
||||
height: 100vh;
|
||||
flex-direction: column;
|
||||
|
||||
// 标签栏不收缩
|
||||
.y-tabs__wrap {
|
||||
flex-shrink: 0;
|
||||
}
|
||||
|
||||
.y-tabs__track,
|
||||
.y-tabs__content-scrollview {
|
||||
height: 100%;
|
||||
}
|
||||
}
|
||||
|
||||
// 区域滚动下的侧边栏导航
|
||||
&.is-areaScroll.is-sidebarNav {
|
||||
display: flex;
|
||||
height: 100vh;
|
||||
flex-direction: row;
|
||||
|
||||
.y-tabs__scroll {
|
||||
height: 100%;
|
||||
}
|
||||
|
||||
// 标签栏不收缩
|
||||
.y-tabs__wrap {
|
||||
flex-shrink: 0;
|
||||
}
|
||||
|
||||
.y-tabs__track,
|
||||
.y-tabs__content-scrollview {
|
||||
height: 100%;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// 依赖元素
|
||||
.y-tabs__depend {
|
||||
position: absolute;
|
||||
top: 0;
|
||||
left: 0;
|
||||
height: 1px; //必须保证有高度,否则observer无效
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
// 透明标签栏所需的依赖元素
|
||||
.y-tabs__depend--transparent {
|
||||
position: absolute;
|
||||
top: 0;
|
||||
left: 0;
|
||||
height: 1px; //必须保证有高度,否则observer无效
|
||||
width: 1px;
|
||||
}
|
||||
|
||||
// 模拟标签栏吸顶时设置offset时距屏幕顶部的元素
|
||||
.y-tabs__depend--offset {
|
||||
position: fixed;
|
||||
top: 0;
|
||||
left: 0;
|
||||
z-index: -1;
|
||||
height: 1px;
|
||||
}
|
||||
|
||||
// 标签栏占位元素
|
||||
.y-tabs__placeholder {
|
||||
position: relative;
|
||||
}
|
||||
|
||||
// 标签垂直展示且吸顶时,标签栏占位元素不伸缩
|
||||
.y-tabs.is-fixed.is-vertical .y-tabs__placeholder {
|
||||
flex-shrink: 0;
|
||||
}
|
||||
|
||||
// 文字省略
|
||||
.y-tabs__ellipsis {
|
||||
overflow: hidden;
|
||||
text-overflow: ellipsis;
|
||||
white-space: nowrap;
|
||||
}
|
||||
|
||||
// 导航区域包裹层
|
||||
.y-tabs__wrap {
|
||||
position: relative;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
overflow: hidden;
|
||||
visibility: visible;
|
||||
background: #fff;
|
||||
touch-action: none;
|
||||
|
||||
// 标签栏垂直展示时包裹层样式
|
||||
&.is-vertical {
|
||||
width: 100px;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
flex-shrink: 0;
|
||||
}
|
||||
|
||||
// 粘性定位布局下的导航区域包裹层
|
||||
&.is-fixed {
|
||||
position: fixed;
|
||||
top: 0;
|
||||
right: 0;
|
||||
left: 0;
|
||||
z-index: 99;
|
||||
}
|
||||
|
||||
// 标签垂直展示且吸顶时,给定bottom,否则scroll-view不会滚动
|
||||
&.is-fixed.is-vertical {
|
||||
bottom: 0;
|
||||
}
|
||||
|
||||
// 透明的导航区域包裹层
|
||||
&.is-transparent {
|
||||
background: rgba(255, 255, 255, 0);
|
||||
}
|
||||
|
||||
// 标签栏水平时按钮风格的包裹层
|
||||
&.is-button:not(.is-vertical),
|
||||
&.is-line-button:not(.is-vertical) {
|
||||
padding: 0 8px;
|
||||
}
|
||||
}
|
||||
|
||||
// scroll-view组件样式
|
||||
.y-tabs__scroll {
|
||||
position: relative;
|
||||
width: 100%;
|
||||
white-space: nowrap; // 使用横向滚动时,需要给<scroll-view>添加white-space: nowrap;样式
|
||||
flex-direction: row;
|
||||
}
|
||||
|
||||
// 条件编译不放在样式中,vue3无效
|
||||
|
||||
// H5、APP端去滚动条
|
||||
// 小程序端会报:Some selectors are not allowed in component wxss, including tag name selectors, ID selectors, and attribute selectors.
|
||||
/* #ifdef H5 || APP */
|
||||
.y-tabs__scroll ::-webkit-scrollbar {
|
||||
display: none;
|
||||
width: 0;
|
||||
height: 0;
|
||||
-webkit-appearance: none;
|
||||
background: transparent;
|
||||
color: transparent;
|
||||
}
|
||||
/* #endif */
|
||||
|
||||
// IOS 13 以下的系统,当滚动区域设置了-webkit-overflow-scrolling: touch;时(必须设置,否者几乎无法滚动),::-webkit-scrollbar 相关属性会失效,iOS 13 已经修复了此Bug。
|
||||
// 小程序端: 去除 scroll-view 组件的滚动条
|
||||
/* #ifndef H5 || APP */
|
||||
::-webkit-scrollbar {
|
||||
display: none;
|
||||
width: 0;
|
||||
height: 0;
|
||||
-webkit-appearance: none;
|
||||
background: transparent;
|
||||
color: transparent;
|
||||
}
|
||||
/* #endif */
|
||||
|
||||
// 导航区域
|
||||
.y-tabs__nav {
|
||||
position: relative;
|
||||
box-sizing: border-box;
|
||||
user-select: none;
|
||||
flex: 1;
|
||||
display: flex;
|
||||
|
||||
&.is-shrink{
|
||||
display: inline-flex;
|
||||
}
|
||||
|
||||
// 卡片风格
|
||||
&.is-card {
|
||||
margin: 6px 16px;
|
||||
border-radius: 4px;
|
||||
box-sizing: border-box;
|
||||
border: 1px solid #0022ab;
|
||||
}
|
||||
|
||||
// 标签栏垂直时导航区域样式
|
||||
&.is-vertical {
|
||||
flex-direction: column;
|
||||
height: auto;
|
||||
|
||||
.y-tab {
|
||||
flex: unset;
|
||||
}
|
||||
}
|
||||
|
||||
// 标签左侧、右侧的补充区域
|
||||
&-left,
|
||||
&-right {
|
||||
position: relative;
|
||||
display: inline-flex;
|
||||
white-space: nowrap;
|
||||
}
|
||||
}
|
||||
|
||||
// 导航标签
|
||||
.y-tab {
|
||||
position: relative;
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
box-sizing: border-box;
|
||||
height: 40px;
|
||||
font-size: 28rpx;
|
||||
color: #646566;
|
||||
text-align: center;
|
||||
padding: 0 4px;
|
||||
flex: 1;
|
||||
cursor: pointer;
|
||||
// webkit的css扩展:1、-webkit-tap-highlight-color:这个属性是用于设定元素在移动设备(如Adnroid、iOS)上被触发点击事件时,响应的背景框的颜色。有事件监听的元素被点击的时候会被高亮显示,比如我的android上表现为一个蓝框加上半透明的背景
|
||||
-webkit-tap-highlight-color: transparent;
|
||||
// transition-duration: 0.2s;
|
||||
// transition-property: background;
|
||||
flex-shrink: 0;
|
||||
z-index: 2;
|
||||
|
||||
// 选中状态
|
||||
&.is-active {
|
||||
color: #323233;
|
||||
font-weight: 500;
|
||||
}
|
||||
|
||||
// 禁用状态
|
||||
&.is-disabled {
|
||||
color: #c8c9cc !important;
|
||||
cursor: not-allowed;
|
||||
}
|
||||
|
||||
// 收缩布局
|
||||
&.is-shrink {
|
||||
flex: none;
|
||||
padding: 0 8px;
|
||||
}
|
||||
|
||||
//卡片风格
|
||||
&.is-card {
|
||||
height: 28px;
|
||||
line-height: 28px;
|
||||
}
|
||||
}
|
||||
|
||||
// 标题区域
|
||||
.y-tab__title {
|
||||
overflow: hidden;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
height: inherit;
|
||||
}
|
||||
|
||||
// 标题区域垂直排列
|
||||
.y-tab__title--top,
|
||||
.y-tab__title--bottom {
|
||||
flex-direction: column;
|
||||
}
|
||||
|
||||
// 标题文字
|
||||
.y-tab__text {
|
||||
position: relative;
|
||||
display: block;
|
||||
line-height: 1.2;
|
||||
order: 2;
|
||||
white-space: nowrap; //字节会设置white-space:normal
|
||||
}
|
||||
|
||||
// 标签垂直展示时,未达到文字超出隐藏的条件时
|
||||
.y-tabs__nav.is-vertical .y-tab__text:not(.y-tabs__ellipsis) {
|
||||
white-space: normal;
|
||||
}
|
||||
|
||||
// 使用order排序
|
||||
.y-tab__text--left,
|
||||
.y-tab__text--top {
|
||||
order: 0;
|
||||
}
|
||||
|
||||
// 标题图标/图片包裹层
|
||||
.y-tab__icons {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
order: 1;
|
||||
z-index: 1;
|
||||
}
|
||||
|
||||
//标题图片
|
||||
.y-tab__image {
|
||||
width: 20px;
|
||||
height: 20px;
|
||||
}
|
||||
|
||||
// 右上角信息区域
|
||||
.y-tab__info {
|
||||
display: inline-flex;
|
||||
position: relative;
|
||||
|
||||
&--dot,
|
||||
&--badge {
|
||||
display: inline-block;
|
||||
position: absolute;
|
||||
top: 0;
|
||||
left: 0;
|
||||
box-sizing: border-box;
|
||||
background-color: #e53935;
|
||||
transform-origin: 100%;
|
||||
}
|
||||
|
||||
// 小红点
|
||||
&--dot {
|
||||
width: 6px;
|
||||
height: 6px;
|
||||
border-radius: 100%;
|
||||
transform: translate(0%, -180%);
|
||||
}
|
||||
|
||||
// 徽标
|
||||
&--badge {
|
||||
line-height: 13px;
|
||||
min-width: 18px;
|
||||
border-radius: 18px;
|
||||
padding: 0 2px;
|
||||
transform: translate(0%, -120%);
|
||||
font-size: 18rpx;
|
||||
font-weight: 500;
|
||||
text-align: center;
|
||||
color: #fff;
|
||||
}
|
||||
}
|
||||
|
||||
// 底部条滑块
|
||||
.y-tabs__bar {
|
||||
position: absolute;
|
||||
display: inline-flex;
|
||||
left: 0;
|
||||
z-index: 1;
|
||||
|
||||
width: 20px;
|
||||
height: 3px;
|
||||
border-radius: 3px;
|
||||
background-color: #0022ab;
|
||||
|
||||
// line风格的滑块
|
||||
&.is-line {
|
||||
z-index: 2; //z-index与y-tab一样,避免被遮挡
|
||||
|
||||
// 标签水平展示时
|
||||
&:not(.is-vertical) {
|
||||
bottom: 3px;
|
||||
width: 20px;
|
||||
height: 3px;
|
||||
border-radius: 3px;
|
||||
}
|
||||
|
||||
// 标签垂直展示时
|
||||
&.is-vertical {
|
||||
top: 0;
|
||||
left: 3px;
|
||||
width: 3px;
|
||||
height: 20px;
|
||||
border-radius: 3px;
|
||||
}
|
||||
}
|
||||
|
||||
// button、line-button风格的滑块
|
||||
&.is-button,
|
||||
&.is-line-button {
|
||||
// top: 0;
|
||||
justify-content: center;
|
||||
align-items: center;
|
||||
border-radius: 26px;
|
||||
|
||||
// 标签水平展示时
|
||||
&:not(.is-vertical) {
|
||||
height: calc(100% - 8px);
|
||||
bottom: 4px;
|
||||
}
|
||||
|
||||
// 标签垂直展示时
|
||||
&.is-vertical {
|
||||
width: calc(100% - 8px);
|
||||
height: calc(100% - 8px);
|
||||
}
|
||||
}
|
||||
|
||||
// 线性按钮风格的滑块
|
||||
&.is-line-button {
|
||||
background-color: transparent;
|
||||
border: 2rpx solid transparent;
|
||||
}
|
||||
}
|
||||
|
||||
// 标签内容
|
||||
.y-tabs__content {
|
||||
display: block;
|
||||
position: relative;
|
||||
overflow: hidden; //会导致uni-data-select无法撑开显示下拉选项,最好给pane中的内容设置一个高度(如果包裹select的父元素都没有设置relative,则不会裁剪absolute属性的元素)
|
||||
// 标签栏垂直展示,内容减去标签栏默认宽度
|
||||
&.is-vertical {
|
||||
width: 100%;
|
||||
}
|
||||
}
|
||||
// 标签内容的滑动轨道容器
|
||||
.y-tabs__track {
|
||||
position: relative;
|
||||
display: flex;
|
||||
width: 100%;
|
||||
will-change: left;
|
||||
|
||||
// 滚动导航模式下内容卡片垂直排列
|
||||
&.is-scrollspy {
|
||||
flex-direction: column;
|
||||
}
|
||||
}
|
||||
|
||||
// 标签内容卡片
|
||||
.y-tab__pane {
|
||||
flex-shrink: 0;
|
||||
box-sizing: border-box;
|
||||
width: 100%;
|
||||
height: 0;
|
||||
position: relative;
|
||||
flex-direction: row;
|
||||
display: block;
|
||||
visibility: visible;
|
||||
|
||||
// 选中时
|
||||
&.is-active {
|
||||
height: auto;
|
||||
}
|
||||
// 滚动导航
|
||||
&.is-scrollspy {
|
||||
height: auto;
|
||||
}
|
||||
}
|
||||
.y-tab__pane--wrap {
|
||||
position: relative;
|
||||
}
|
||||
// 区域滚动下的标签内容scroll-view
|
||||
.y-tabs__content-scrollview {
|
||||
flex-direction: column;
|
||||
}
|
194
smart-app/src/uni_modules/y-tabs/components/js/const.js
Normal file
194
smart-app/src/uni_modules/y-tabs/components/js/const.js
Normal file
@ -0,0 +1,194 @@
|
||||
// styleIsolation:组件样式隔离方式,具体配置选项参见:微信小程序自定义组件的样式
|
||||
// 自定义组件 JSON 中的 styleIsolation 选项从基础库版本 2.10.1 开始支持。它支持以下取值:
|
||||
// isolated 表示启用样式隔离,在自定义组件内外,使用 class 指定的样式将不会相互影响(一般情况下的默认值);
|
||||
// apply-shared 表示页面 wxss 样式将影响到自定义组件,但自定义组件 wxss 中指定的样式不会影响页面;
|
||||
// shared 表示页面 wxss 样式将影响到自定义组件,自定义组件 wxss 中指定的样式也会影响页面和其他设置了 apply-shared 或 shared 的自定义组件。(这个选项在插件中不可用。)
|
||||
|
||||
const options = {
|
||||
styleIsolation: 'shared',
|
||||
virtualHost: true // [微信小程序、支付宝小程序(默认值为 true)] 将自定义节点设置成虚拟的,更加接近Vue组件的表现。我们不希望自定义组件的这个节点本身可以设置样式、响应 flex 布局等,而是希望自定义组件内部的第一层节点能够响应 flex 布局或者样式由自定义组件本身完全决定
|
||||
// 微信(可以使用virtualHost配置)/QQ/百度/字节跳动这四家小程序,自定义组件在渲染时会比App/H5端多一级节点,导致flex无效,是否考虑在组件上增加class控制
|
||||
}
|
||||
|
||||
|
||||
|
||||
const emits = [
|
||||
"input",
|
||||
'update:modelValue', // 更新v-model绑定的变量
|
||||
'click', //点击标签时触发 回调参数:name:标识符,title:标题
|
||||
'change', //当前激活的标签改变时触发 回调参数:name:标识符,title:标题
|
||||
'disabled', //点击被禁用的标签时触发 回调参数:name:标识符,title:标题
|
||||
'rendered', //标签内容首次渲染时触发(仅在开启延迟渲染后触发) 回调参数:name:标识符,title:标题
|
||||
'sticky-change', //吸顶时触发,仅在 sticky 模式下生效 回调参数:name:标识符,title:标题
|
||||
'loaded', //组件内部初始化完成后调用 回调参数:{ isFixed: 是否吸顶 }
|
||||
'slide-change', //内容页滑动时触发(仅barAnimateMode为linear、worm、worm-ease时有效) 回调参数:{ dx:滑动距离; rate:当前滑动长度占滑动区域的比例;targetIndex:目标下标;}
|
||||
'slide-end' //内容页滑动结束时触发(仅barAnimateMode为linear、worm、worm-ease时有效) 回调参数:{ targetIndex:目标下标;}
|
||||
];
|
||||
|
||||
const props = {
|
||||
// v-model绑定属性,绑定当前选中标签的标识符(标签的下标)
|
||||
value: {
|
||||
type: [Number, String],
|
||||
default: 0,
|
||||
},
|
||||
modelValue: {
|
||||
type: [Number, String],
|
||||
default: 0,
|
||||
},
|
||||
// 样式风格类型,可选值为 text、card、button、line-button
|
||||
type: {
|
||||
type: String,
|
||||
default: "line",
|
||||
validator(value) {
|
||||
return ['line', 'text', 'card', 'button', 'line-button'].includes(value)
|
||||
}
|
||||
},
|
||||
color: {
|
||||
type: [String, null],
|
||||
default: "#0022AB"
|
||||
}, //标签主题色, 默认值为"#0022AB"
|
||||
background: {
|
||||
type: [String, null],
|
||||
// default: "#fff"
|
||||
}, //标签栏背景色,默认值为"#fff"
|
||||
// 标签栏样式
|
||||
wrapStyle: {
|
||||
type: [Object, null],
|
||||
default: () => {}
|
||||
},
|
||||
// 标签栏的展示方位,可选值:vertical。
|
||||
direction: {
|
||||
type: String,
|
||||
default: "horizontal",
|
||||
validator(value) {
|
||||
return ['horizontal', 'vertical'].includes(value)
|
||||
}
|
||||
},
|
||||
titleActiveColor: String, //标题选中态颜色
|
||||
titleInactiveColor: String, //标题默认态颜色
|
||||
// 是否开启左侧收缩布局,开启后,所有的标签会向左侧收缩对齐。
|
||||
shrink: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
},
|
||||
|
||||
// 动画时间,单位秒,默认为0.3s。仅支持type为line、button、line-button的滑块切换动画,切换标签内容时的转场动画、滚动导航下的内容定位动画。
|
||||
duration: {
|
||||
type: [Number, String],
|
||||
default: 0.2,
|
||||
},
|
||||
// 滑块宽度,默认单位为px, 支持数字、rpx、vh、vw等单位及calc() 函数。 仅支持type为line、button、line-button。
|
||||
// 标签栏水平/垂直展示时,type为line,宽度默认为20px/3px, 而type为button、line-button时,宽度默认为选中标签宽度-8px。
|
||||
barWidth: [Number, String], //inherit:继承tab的宽高
|
||||
// 滑块高度,默认单位为px, 支持数字、rpx、vh、vw等单位及calc() 函数。 仅支持type为line、button、line-button。
|
||||
// 标签栏水平/垂直展示时,type为line,高度默认为3px/20px, 而type为button、line-button时,宽度默认为选中标签高度-8px。
|
||||
barHeight: [Number, String],
|
||||
//滑块样式,仅支持type为line、button、line-button。
|
||||
barStyle: Object,
|
||||
// 滑动切换tab内容时滑块的动画模式,默认值为line,即切换tab时滑块宽度保持不变,线性运动。可选值为worm(毛毛虫效果)、worm-ease(毛毛虫缓动)、none(不设置)。
|
||||
// 可结合swiper组件使用,滑动效果更好。
|
||||
// 仅支持type为line。
|
||||
barAnimateMode: {
|
||||
type: String,
|
||||
default: "linear",
|
||||
validator(value) {
|
||||
return ['none', 'linear', 'worm', 'worm-ease'].includes(value);
|
||||
}
|
||||
},
|
||||
// 标签宽高是否动态变化
|
||||
// 表示标签切换了选中状态后宽高是否有变化,有则需要开启该属性,否则会导致滑块错位
|
||||
isDynamic: {
|
||||
type: Boolean,
|
||||
default: false,
|
||||
},
|
||||
// 是否省略过长的标题文字。标签栏水平展示时,如果标签数量未超过滚动阈值则生效,垂直展示不限制。
|
||||
ellipsis: {
|
||||
type: Boolean,
|
||||
default: true,
|
||||
},
|
||||
// 滚动阈值,标签数量超过阈值且总宽度超过标签栏宽度时开始横向滚动
|
||||
scrollThreshold: {
|
||||
type: [Number, String],
|
||||
default: 5
|
||||
},
|
||||
// 标签栏滚动时当前标签居中
|
||||
scrollToCenter: {
|
||||
type: Boolean,
|
||||
default: true,
|
||||
},
|
||||
// 切换标签前的回调函数,返回 false 可阻止切换,支持返回 Promise
|
||||
beforeChange: Function,
|
||||
// 是否开启延迟渲染(首次切换到标签时才触发内容渲染)
|
||||
isLazyRender: Boolean,
|
||||
// 是否开启切换动画
|
||||
// 用于标签栏滚动动画、切换标签内容时的转场动画、滚动导航下的内容定位动画
|
||||
animated: {
|
||||
type: Boolean,
|
||||
default: true
|
||||
},
|
||||
// 在滚动导航模式下,滚动到最后一个标签内容但其顶部未超过可视区域时,是否激活对应的标签项
|
||||
activeLast: {
|
||||
type: Boolean,
|
||||
default: false,
|
||||
},
|
||||
// ---------------------------------- 用于内容区域左右滑动的配置 ----------------------------------------
|
||||
// 是否开启手势滑动切换
|
||||
swipeable: {
|
||||
type: Boolean,
|
||||
default: false,
|
||||
},
|
||||
// 是否开启标签内容滑动时的拖动动画
|
||||
// swipeable为true时有效,建议设置is-lazy-render=false。(该属性开启时考虑给包裹内容的容器增加一个min-height,因为其他未显示的标签内容会沿用当前显示的高度,拖动切换后由于高度不一致会有回弹)
|
||||
swipeAnimated: {
|
||||
type: Boolean,
|
||||
default: true,
|
||||
},
|
||||
// 滑动切换的滑动距离阈值,单位为px;表示开启手势滑动时,横向滑动多少px切换标签内容(快速滑动时不受限制)
|
||||
swipeThreshold: {
|
||||
type: [Number, String],
|
||||
default: 120,
|
||||
},
|
||||
// ---------------------------------- 用于滚动吸顶的配置 ----------------------------------------
|
||||
// 是否使用粘性定位布局进行滚动吸顶
|
||||
sticky: Boolean,
|
||||
// 粘性布局下与顶部的最小距离,单位为px
|
||||
offsetTop: {
|
||||
type: Number,
|
||||
default: 0
|
||||
},
|
||||
// 粘性布局下标签栏的z-index值
|
||||
zIndex: {
|
||||
type: Number,
|
||||
default: 99
|
||||
},
|
||||
// 粘性布局的判断阈值:表示在页面滚动过程中,标签栏距屏幕顶部多少px时,触发吸顶函数进行吸顶判断
|
||||
stickyThreshold: {
|
||||
type: Number,
|
||||
default: 0
|
||||
},
|
||||
// 页面滚动过程中,标题栏背景色是否透明渐变
|
||||
// background属性值必须为rgba格式
|
||||
transparent: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
},
|
||||
// 标题栏背景色透明渐变的滚动距离
|
||||
transparentOffset: {
|
||||
type: Number,
|
||||
default: 100
|
||||
},
|
||||
// 是否开启滚动导航;该模式下,内容将会平铺展示
|
||||
// 如果标签栏垂直展示,且内容平铺展示,就为侧边栏模式
|
||||
scrollspy: Boolean,
|
||||
// 滚动导航模式下,内容区域是否跟随页面滚动
|
||||
// 为true时,整体区域跟随页面而滚动,为false时,内容区域是放在scroll-view中实现的局部滚动
|
||||
pageScroll: {
|
||||
type: Boolean,
|
||||
default: true
|
||||
},
|
||||
}
|
||||
export {
|
||||
options,
|
||||
emits,
|
||||
props,
|
||||
}
|
75
smart-app/src/uni_modules/y-tabs/components/js/touchMixin.js
Normal file
75
smart-app/src/uni_modules/y-tabs/components/js/touchMixin.js
Normal file
@ -0,0 +1,75 @@
|
||||
import { getDirection, now } from "./uitls"
|
||||
|
||||
export const touchMixin = {
|
||||
data() {
|
||||
return {
|
||||
direction: '', //滑动方向
|
||||
startX: '', //开始滑动的x坐标
|
||||
startY: '', //开始滑动的y坐标
|
||||
nextIndex: -1, //下一个切换的标签下标
|
||||
moved: false, //是否为一次水平滑动
|
||||
startTimestamp: 0,
|
||||
};
|
||||
},
|
||||
methods: {
|
||||
touchStart(event) {
|
||||
if (!this.parent.swipeable) return;
|
||||
this.resetTouchStatus();
|
||||
this.startX = event.touches[0].clientX;
|
||||
this.startY = event.touches[0].clientY;
|
||||
this.startTimestamp = now();
|
||||
},
|
||||
touchMove(event) {
|
||||
if (!this.parent.swipeable) return;
|
||||
const touch = event.touches[0];
|
||||
this.deltaX = touch.clientX < 0 ? 0 : this.startX - touch.clientX;
|
||||
this.deltaY = this.startY - touch.clientY;
|
||||
const offsetX = Math.abs(this.deltaX);
|
||||
const offsetY = Math.abs(this.deltaY);
|
||||
// 当距离大于某个值时锁定方向
|
||||
if (!this.direction || (offsetX < 10 && offsetY < 10)) this.direction = getDirection(offsetX, offsetY);
|
||||
|
||||
if (this.direction === "horizontal") { //水平滑动
|
||||
const { dataLen, contentWidth, currentIndex, tabs, swipeAnimated } = this.parent;
|
||||
const isRight = this.deltaX < 0; //判断是否向右滑动
|
||||
|
||||
// 如果为第一页,则不允许向右滑;为最后一页,则不允许左滑
|
||||
if ((isRight && currentIndex === 0) || (!isRight && currentIndex === dataLen - 1)) return;
|
||||
|
||||
this.nextIndex = currentIndex + (isRight ? -1 : 1); //下一个标签
|
||||
if (tabs[this.nextIndex]?.disabled) return; //禁用的标签不允许滑动
|
||||
|
||||
this.moved = true; //标记为一次水平滑动
|
||||
|
||||
// 改变标签内容滑动轨道样式,模拟拖动动画效果
|
||||
if (swipeAnimated) {
|
||||
const offsetWidth = contentWidth * currentIndex * -1 + offsetX * (isRight ? 1 : -1);
|
||||
this.parent.changeTrackStyle(true, 0, offsetWidth);
|
||||
this.parent.setDx(this.deltaX, false);
|
||||
}
|
||||
}
|
||||
},
|
||||
touchEnd() {
|
||||
if (this.moved) {
|
||||
// 何时可切换标签,当横向滑动距离大于设定阈值,或快速滑动(300ms内)切滑动距离大于18px时
|
||||
const deltaTime = now() - this.startTimestamp;
|
||||
const distance = Math.abs(this.deltaX);
|
||||
const speed = (distance / deltaTime).toFixed(4);
|
||||
const isChange = speed > 0.25 || distance >= this.parent.swipeThreshold;//是否切换
|
||||
const currIndex = this.parent.currentIndex; //当前选中下标
|
||||
const targetIndex = isChange ? this.nextIndex : currIndex; //目标标签的下标
|
||||
this.parent.touchEndForPane(this.deltaX, currIndex, targetIndex, isChange);
|
||||
|
||||
}
|
||||
},
|
||||
// 重置触摸状态
|
||||
resetTouchStatus() {
|
||||
this.direction = '';
|
||||
this.deltaX = 0;
|
||||
this.deltaY = 0;
|
||||
this.nextIndex = -1;
|
||||
this.moved = false;
|
||||
this.startTimestamp = 0;
|
||||
},
|
||||
}
|
||||
}
|
177
smart-app/src/uni_modules/y-tabs/components/js/uitls.js
Normal file
177
smart-app/src/uni_modules/y-tabs/components/js/uitls.js
Normal file
@ -0,0 +1,177 @@
|
||||
/**
|
||||
* 判断传入的值是否为空
|
||||
* @param {*} val
|
||||
* @returns
|
||||
*/
|
||||
export function isNull(val) {
|
||||
if (typeof val == "boolean") {
|
||||
return false;
|
||||
}
|
||||
if (typeof val == "number") {
|
||||
return false;
|
||||
}
|
||||
if (val instanceof Array) {
|
||||
if (val.length == 0) return true;
|
||||
} else if (val instanceof Object) {
|
||||
if (JSON.stringify(val) === "{}") return true;
|
||||
} else {
|
||||
if (
|
||||
val == "null" ||
|
||||
val == null ||
|
||||
val == "undefined" ||
|
||||
val == undefined ||
|
||||
val == ""
|
||||
)
|
||||
return true;
|
||||
return false;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
// 不为空
|
||||
export function isDef(val) {
|
||||
return val !== undefined && val !== null;
|
||||
}
|
||||
|
||||
// 是否是一个数字
|
||||
export function isNumeric(val) {
|
||||
return /^\d+(\.\d+)?$/.test(val);
|
||||
}
|
||||
|
||||
// 是一个对象
|
||||
export function isObject(val) {
|
||||
return val !== null && typeof val === 'object';
|
||||
}
|
||||
// 是一个字符串
|
||||
export function isString(val) {
|
||||
return Object.prototype.toString.call(val) === "[object String]"
|
||||
}
|
||||
|
||||
// 空操作
|
||||
export function noop() {}
|
||||
|
||||
// 是一个函数
|
||||
export function isFunction(val) {
|
||||
return typeof val === 'function';
|
||||
}
|
||||
|
||||
// 是一个promise对象
|
||||
export function isPromise(val) {
|
||||
return isObject(val) && isFunction(val.then) && isFunction(val.catch);
|
||||
}
|
||||
|
||||
|
||||
|
||||
// 添加单位
|
||||
export function addUnit(value) {
|
||||
if (!isDef(value)) {
|
||||
return undefined;
|
||||
}
|
||||
|
||||
value = String(value);
|
||||
return isNumeric(value) ? `${value}px` : value;
|
||||
}
|
||||
|
||||
// 调用拦截器
|
||||
export function callInterceptor(options) {
|
||||
const {
|
||||
interceptor,
|
||||
args,
|
||||
done
|
||||
} = options;
|
||||
|
||||
if (interceptor) {
|
||||
const returnVal = interceptor(...args);
|
||||
if (isPromise(returnVal)) {
|
||||
returnVal.then((value) => {
|
||||
if (value) done();
|
||||
}).catch(noop);
|
||||
} else if (returnVal) {
|
||||
done();
|
||||
}
|
||||
} else {
|
||||
done();
|
||||
}
|
||||
}
|
||||
|
||||
const rgbaRegex = /^rgba\((\d{1,3}),\s*(\d{1,3}),\s*(\d{1,3}),\s*(\d*(?:\.\d+)?)\)$/;
|
||||
export const getColor = function(colorStr) {
|
||||
const matches = colorStr.match(rgbaRegex);
|
||||
if (matches && matches.length === 5) {
|
||||
return [
|
||||
matches[1],
|
||||
matches[2],
|
||||
matches[3],
|
||||
matches[4]
|
||||
];
|
||||
}
|
||||
return [];
|
||||
};
|
||||
|
||||
export function toClass(classObj, ...classArray) {
|
||||
const arr = Object.keys(classObj || {}).filter(key => classObj[key])
|
||||
arr.push(...classArray)
|
||||
return arr.join(" ")
|
||||
}
|
||||
|
||||
|
||||
// 判断是水平滑动还是垂直滑动
|
||||
export function getDirection(x, y) {
|
||||
if (x > y) return 'horizontal';
|
||||
if (y > x) return 'vertical';
|
||||
return '';
|
||||
}
|
||||
|
||||
|
||||
|
||||
// 缓动函数
|
||||
function easingFunction(time, duration, type = "linear") {
|
||||
let pos = time / duration;
|
||||
let value = 0;
|
||||
switch (type) {
|
||||
case "easeOutCubic":
|
||||
value = (Math.pow((pos - 1), 3) + 1)
|
||||
break;
|
||||
case "easeInOutCubic":
|
||||
if ((pos /= 0.5) < 1) value = 0.5 * Math.pow(pos, 3);
|
||||
else value = 0.5 * (Math.pow((pos - 2), 3) + 2);
|
||||
break;
|
||||
default: //linear
|
||||
value = pos;
|
||||
break;
|
||||
}
|
||||
return value;
|
||||
}
|
||||
|
||||
/**
|
||||
* 进度函数
|
||||
* @param {Object} time 当前已经运动的时间
|
||||
* @param {Object} begin 距离的初始值
|
||||
* @param {Object} end 距离的结束值
|
||||
* @param {Object} duration 运动时长
|
||||
*/
|
||||
export function progress(time, begin, end, duration, type) {
|
||||
return begin + (end - begin) * easingFunction(time, duration, type);
|
||||
}
|
||||
|
||||
|
||||
let uid = 0;
|
||||
export function getUid() {
|
||||
return uid++
|
||||
}
|
||||
|
||||
|
||||
const hasOwnProperty = Object.prototype.hasOwnProperty
|
||||
/**
|
||||
* 检查对象是否具有该属性
|
||||
* @param {*} obj 对象
|
||||
* @param {*} key 对象属性名
|
||||
* @returns
|
||||
*/
|
||||
export function hasOwn(obj, key) {
|
||||
return hasOwnProperty.call(obj, key)
|
||||
}
|
||||
|
||||
export const now = Date.now || function() {
|
||||
return +new Date();
|
||||
};
|
@ -0,0 +1,8 @@
|
||||
|
||||
export const utilMixin = function() {
|
||||
return {
|
||||
methods: {
|
||||
|
||||
},
|
||||
}
|
||||
}
|
238
smart-app/src/uni_modules/y-tabs/components/y-tab/y-tab.vue
Normal file
238
smart-app/src/uni_modules/y-tabs/components/y-tab/y-tab.vue
Normal file
@ -0,0 +1,238 @@
|
||||
<template>
|
||||
<view
|
||||
class="y-tab__pane"
|
||||
:data-index="index"
|
||||
:class="[uniquePaneClass, paneClass]"
|
||||
:style="[paneStyle]"
|
||||
@touchstart="touchStart"
|
||||
@touchmove="touchMove"
|
||||
@touchend="touchEnd"
|
||||
>
|
||||
<!-- 渲染过的则不再渲染,未渲染的根据激活状态进行渲染 -->
|
||||
<view class="y-tab__pane--wrap" v-if="rendered ? true : active">
|
||||
<slot></slot>
|
||||
</view>
|
||||
</view>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import { isNull, toClass, getUid } from '../js/uitls';
|
||||
import { touchMixin } from '../js/touchMixin';
|
||||
import { options } from '../js/const';
|
||||
|
||||
export default {
|
||||
name: 'yTab',
|
||||
mixins: [touchMixin],
|
||||
options,
|
||||
props: {
|
||||
title: String, // 标题
|
||||
disabled: Boolean, // 是否禁用标签
|
||||
dot: Boolean, // 是否在标题右上角显示小红点
|
||||
badge: {
|
||||
type: [Number, String],
|
||||
default: '',
|
||||
}, // 图标右上角徽标的内容
|
||||
// 徽标数最大数字限制,超过这个数字将变成badgeMaxCount+,如果传空字符串则不设置
|
||||
badgeMaxCount: {
|
||||
type: [Number, String],
|
||||
default: 99,
|
||||
},
|
||||
name: [Number, String], // 标签名称,作为匹配的标识符
|
||||
titleStyle: Object, // 自定义标题样式
|
||||
titleClass: String, // 自定义标题类名
|
||||
iconType: String, //图标图案,为uniapp扩展组件(uni-ui)下的uni-icons的type值,customPrefix用法等同
|
||||
iconSize: {
|
||||
type: [Number, String],
|
||||
default: 16,
|
||||
}, //图标大小
|
||||
customPrefix: String, //自定义图标
|
||||
imageSrc: String, //图片路径
|
||||
imageMode: {
|
||||
type: String,
|
||||
default: 'scaleToFill',
|
||||
validator(value) {
|
||||
return [
|
||||
'scaleToFill',
|
||||
'aspectFit',
|
||||
'aspectFill',
|
||||
'widthFix',
|
||||
'heightFix',
|
||||
'top',
|
||||
'bottom',
|
||||
'center',
|
||||
'left',
|
||||
'right',
|
||||
'top left',
|
||||
'top right',
|
||||
'bottom left',
|
||||
'bottom right',
|
||||
].includes(value);
|
||||
},
|
||||
}, //图片裁剪、缩放的模式,为uniapp内置组件->媒体组件—>image下的mode值
|
||||
position: {
|
||||
type: String,
|
||||
default: 'right',
|
||||
validator(value) {
|
||||
return ['top', 'bottom', 'left', 'right'].includes(value);
|
||||
},
|
||||
}, //如果存在图片或图标,标题围绕它们的位置
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
isUnmounted: false,
|
||||
index: -1, //内容卡片对应的下标
|
||||
parent: null, //父元素实例
|
||||
active: false, //是否为激活状态
|
||||
rendered: false, //是否渲染过
|
||||
swipeable: false, //是否开启手势滑动切换
|
||||
paneStyle: null, //内容样式
|
||||
scrollspy: false, //是否为滚动导航模式
|
||||
paneObserver: null, //pane交叉观察器
|
||||
isDisjoint: false, //当前pane是否与参照节点布局区域相离
|
||||
isActiveLast: false, // 最后一个pane在滚动导航模式下是否激活对应的标签项
|
||||
};
|
||||
},
|
||||
computed: {
|
||||
computedName() {
|
||||
return !isNull(this.name) ? this.name : this.index;
|
||||
},
|
||||
unqieKey() {
|
||||
return getUid();
|
||||
},
|
||||
// 保证唯一的样式
|
||||
uniquePaneClass() {
|
||||
return 'y-tab__pane' + this.unqieKey;
|
||||
},
|
||||
// 内容class
|
||||
paneClass() {
|
||||
return toClass({ 'is-active': this.active, 'is-scrollspy': this.scrollspy });
|
||||
},
|
||||
},
|
||||
watch: {
|
||||
$props: {
|
||||
deep: true,
|
||||
// immediate: true,
|
||||
handler(newValue, oldValue) {
|
||||
// 更新tab
|
||||
if (this.parent) {
|
||||
this.parent.updateTab({
|
||||
newValue: { ...newValue, badge: this.formatBadge() },
|
||||
oldValue: oldValue && { ...oldValue },
|
||||
index: this.index,
|
||||
});
|
||||
}
|
||||
},
|
||||
},
|
||||
},
|
||||
created() {
|
||||
this.parent = this.getParent();
|
||||
if (this.parent) {
|
||||
this.parent.children.push(this);
|
||||
this.parent.putTab({ newValue: { ...this.$props, key: this.unqieKey, badge: this.formatBadge() } });
|
||||
this.scrollspy = this.parent.scrollspy;
|
||||
this.rendered = !this.parent.isLazyRender || this.parent.scrollspy; //标记是否渲染过,非懒加载与滚动导航模式下默认渲染
|
||||
}
|
||||
},
|
||||
// #ifndef VUE3
|
||||
destroyed() {
|
||||
if (this.isUnmounted) return;
|
||||
this.unInit();
|
||||
},
|
||||
// #endif
|
||||
// #ifdef VUE3
|
||||
unmounted() {
|
||||
this.isUnmounted = true;
|
||||
this.unInit();
|
||||
},
|
||||
// #endif
|
||||
methods: {
|
||||
// 徽标格式化
|
||||
formatBadge() {
|
||||
if (!isNull(this.badge) && !isNull(this.badgeMaxCount) && this.badge > this.badgeMaxCount) {
|
||||
return this.badgeMaxCount + '+';
|
||||
} else {
|
||||
return this.badge;
|
||||
}
|
||||
},
|
||||
// 获取查询节点信息的对象
|
||||
getSelectorQuery() {
|
||||
let query = null;
|
||||
// #ifdef MP-ALIPAY
|
||||
query = uni.createSelectorQuery();
|
||||
// #endif
|
||||
// #ifndef MP-ALIPAY
|
||||
query = uni.createSelectorQuery().in(this);
|
||||
// #endif
|
||||
return query;
|
||||
},
|
||||
// 获取元素位置信息
|
||||
getRect(selector) {
|
||||
return new Promise((resolve, reject) => {
|
||||
selector = `.${this.uniquePaneClass}` + (!isNull(selector) ? ' ' + selector : '');
|
||||
this.getSelectorQuery()
|
||||
.select(selector)
|
||||
.boundingClientRect()
|
||||
.exec((rect) => {
|
||||
resolve(rect[0] || {});
|
||||
});
|
||||
});
|
||||
},
|
||||
// 卸载组件的处理
|
||||
unInit() {
|
||||
this.disconnectObserver(); //销毁观察器
|
||||
if (this.parent) {
|
||||
const index = this.parent.children.findIndex((item) => item === this);
|
||||
this.parent.children.splice(index, 1);
|
||||
this.parent.tabs.splice(index, 1);
|
||||
this.parent.tabRects.splice(index, 1);
|
||||
}
|
||||
},
|
||||
//获取父元素实例
|
||||
getParent(name = 'yTabs') {
|
||||
let parent = this.$parent;
|
||||
let parentName = parent.$options.name;
|
||||
while (parentName !== name) {
|
||||
parent = parent.$parent;
|
||||
if (!parent) return false;
|
||||
parentName = parent.$options.name;
|
||||
}
|
||||
return parent;
|
||||
},
|
||||
// 断掉观察,释放资源
|
||||
disconnectObserver() {
|
||||
this.paneObserver && this.paneObserver?.disconnect();
|
||||
},
|
||||
// 观察 - 标签内容滚动时定位标签项
|
||||
async observePane(top) {
|
||||
this.disconnectObserver();
|
||||
const paneObserver = uni.createIntersectionObserver(this, { thresholds: [0, 0.01, 0.99, 1] });
|
||||
|
||||
// 注意:如果y-tabs使用的区域滚动,整个页面的布局跟随页面滚动,当pane跟随页面移动了之后,
|
||||
// 那么y-tabs__content的top就会变化,导致交互区域位置不准确,可以在onPageScroll使用定时器实现滚动结束的处理重新resize一下组件创建pane的监听
|
||||
// 如果pane内容超过页面的可视区域,最好舍弃这种交互布局,uniapp未实现Android的嵌套滑动机制,页面滑动到底后无法将事件分发给scroll-view,使scroll-view继承滑动
|
||||
|
||||
paneObserver.relativeToViewport({ top: -top }); // 到屏幕顶部的高度时触发
|
||||
// 不能观察根节点 unk-vendors.js:14596 [system] Node .y-tab__pane9 is not found. Intersection observer will not trigger.
|
||||
paneObserver.observe(`.${this.uniquePaneClass} .y-tab__pane--wrap`, (res) => {
|
||||
// console.log('res:', this.title, res);
|
||||
if (!this.isActiveLast) {
|
||||
// 如果目标节点布局区域的top小于参照节点的top,则说明目标节点在参照节点布局区域之上,intersectionRatio不大于0则说明两者不相交
|
||||
this.isDisjoint = res.intersectionRatio <= 0 && res.boundingClientRect.top < res.relativeRect.top;
|
||||
} else {
|
||||
// 滚动导航模式下,最后一个pane完成显示但未超出可视范围顶部时,是否设置相离而激活最后一个标签项
|
||||
this.isDisjoint = res.intersectionRatio > 0 && res.boundingClientRect.bottom <= res.relativeRect.bottom;
|
||||
}
|
||||
|
||||
// 保证组件初始化完成时执行,避免创建时触发一次监听器的回调函数,导致执行顺序先于tabs的init方法,使底部条错位:
|
||||
// 标签栏点击时触发的滚动不允许设置激活下标
|
||||
if (this.parent.isLoaded && !this.parent.lockedScrollspy) this.parent.setActivedIndexToScroll();
|
||||
});
|
||||
this.paneObserver = paneObserver;
|
||||
},
|
||||
},
|
||||
};
|
||||
</script>
|
||||
|
||||
<style lang="scss" scoped>
|
||||
@import '../css/index';
|
||||
</style>
|
1363
smart-app/src/uni_modules/y-tabs/components/y-tabs/y-tabs.vue
Normal file
1363
smart-app/src/uni_modules/y-tabs/components/y-tabs/y-tabs.vue
Normal file
File diff suppressed because it is too large
Load Diff
Loading…
Reference in New Issue
Block a user