250 lines
5.9 KiB
Vue
250 lines
5.9 KiB
Vue
|
<template>
|
|||
|
<view class="tn-collapse-item-class tn-collapse-item" :style="[itemStyle]">
|
|||
|
<!-- 头部 -->
|
|||
|
<view
|
|||
|
class="tn-collapse-item__head"
|
|||
|
:style="[headStyle]"
|
|||
|
:hover-stay-time="200"
|
|||
|
:hover-class="hoverClass"
|
|||
|
@tap.stop="headClick"
|
|||
|
>
|
|||
|
<block v-if="!$slots['title-all'] || !$slots['$title-all']">
|
|||
|
<view
|
|||
|
v-if="!$slots.title || !$slots.$title"
|
|||
|
class="tn-collapse-item__head__title tn-text-ellipsis"
|
|||
|
:style="[
|
|||
|
{ textAlign: align ? align : 'left'},
|
|||
|
isShow && activeStyle && !arrow ? activeStyle : ''
|
|||
|
]"
|
|||
|
>{{ title }}</view>
|
|||
|
<view v-else>
|
|||
|
<slot name="title"></slot>
|
|||
|
</view>
|
|||
|
<view class="tn-collapse-item__head__icon__wrap">
|
|||
|
<view
|
|||
|
v-if="arrow"
|
|||
|
class="tn-icon-down tn-collapse-item__head__icon__arrow"
|
|||
|
:class="{'tn-collapse-item__head__icon__arrow--active': isShow}"
|
|||
|
:style="[arrowIconStyle]"
|
|||
|
></view>
|
|||
|
</view>
|
|||
|
</block>
|
|||
|
<view v-else>
|
|||
|
<slot name="title-all"></slot>
|
|||
|
</view>
|
|||
|
</view>
|
|||
|
<!-- 内容 -->
|
|||
|
<view
|
|||
|
class="tn-collapse-item__body"
|
|||
|
:style="[{
|
|||
|
height: isShow ? height + 'px' : '0'
|
|||
|
}]"
|
|||
|
>
|
|||
|
<view class="tn-collapse-item__body__content" :id="elId" :style="[bodyStyle]">
|
|||
|
<slot></slot>
|
|||
|
</view>
|
|||
|
</view>
|
|||
|
</view>
|
|||
|
</template>
|
|||
|
|
|||
|
<script>
|
|||
|
export default {
|
|||
|
name: 'tn-collapse-item',
|
|||
|
props: {
|
|||
|
// 展开
|
|||
|
open: {
|
|||
|
type: Boolean,
|
|||
|
default: false
|
|||
|
},
|
|||
|
// 唯一标识
|
|||
|
name: {
|
|||
|
type: String,
|
|||
|
default: ''
|
|||
|
},
|
|||
|
// 标题
|
|||
|
title: {
|
|||
|
type: String,
|
|||
|
default: ''
|
|||
|
},
|
|||
|
// 标题对齐方式
|
|||
|
align: {
|
|||
|
type: String,
|
|||
|
default: 'left'
|
|||
|
},
|
|||
|
// 点击不收起
|
|||
|
disabled: {
|
|||
|
type: Boolean,
|
|||
|
default: false
|
|||
|
},
|
|||
|
// 活动时样式
|
|||
|
activeStyle: {
|
|||
|
type: Object,
|
|||
|
default() {
|
|||
|
return {}
|
|||
|
}
|
|||
|
},
|
|||
|
// 标识
|
|||
|
index: {
|
|||
|
type: [Number, String],
|
|||
|
default: ''
|
|||
|
}
|
|||
|
},
|
|||
|
computed: {
|
|||
|
arrowIconStyle() {
|
|||
|
let style = {}
|
|||
|
if (this.arrowColor) {
|
|||
|
style.color = this.arrowColor
|
|||
|
}
|
|||
|
return style
|
|||
|
}
|
|||
|
},
|
|||
|
data() {
|
|||
|
return {
|
|||
|
isShow: false,
|
|||
|
elId: this.$tn.uuid(),
|
|||
|
// body高度
|
|||
|
initCount: 0,
|
|||
|
height: 0,
|
|||
|
// 头部样式
|
|||
|
headStyle: {},
|
|||
|
// 主体样式
|
|||
|
bodyStyle: {},
|
|||
|
// item样式
|
|||
|
itemStyle: {},
|
|||
|
// 显示右边箭头
|
|||
|
arrow: true,
|
|||
|
// 箭头颜色
|
|||
|
arrowColor: '',
|
|||
|
// 点击头部时的效果样式
|
|||
|
hoverClass: ''
|
|||
|
}
|
|||
|
},
|
|||
|
watch: {
|
|||
|
open(value) {
|
|||
|
this.isShow = value
|
|||
|
}
|
|||
|
},
|
|||
|
created() {
|
|||
|
this.parent = false
|
|||
|
this.isShow = this.open
|
|||
|
},
|
|||
|
mounted() {
|
|||
|
this.init()
|
|||
|
},
|
|||
|
methods: {
|
|||
|
// 异步获取内容或者修改了内容时重新获取内容的信息
|
|||
|
init() {
|
|||
|
this.parent = this.$tn.$parent.call(this, 'tn-collapse')
|
|||
|
if (this.parent) {
|
|||
|
this.nameSync = this.name ? this.name : this.parent.childrens.length
|
|||
|
// 不存在才添加对应实例
|
|||
|
!this.parent.childrens.includes(this) && this.parent.childrens.push(this)
|
|||
|
this.headStyle = this.parent.headStyle
|
|||
|
this.bodyStyle = this.parent.bodyStyle
|
|||
|
this.itemStyle = this.parent.itemStyle
|
|||
|
this.arrow = this.parent.arrow
|
|||
|
this.arrowColor = this.parent.arrowColor
|
|||
|
this.hoverClass = this.parent.hoverClass
|
|||
|
}
|
|||
|
this.$nextTick(() => {
|
|||
|
this.queryRect()
|
|||
|
})
|
|||
|
},
|
|||
|
// 点击头部
|
|||
|
headClick() {
|
|||
|
if (this.disabled) return
|
|||
|
if (this.parent && this.parent.accordion) {
|
|||
|
this.parent.childrens.map(child => {
|
|||
|
// 如果是手风琴模式,将其他的item关闭
|
|||
|
if (this !== child) {
|
|||
|
child.isShow = false
|
|||
|
}
|
|||
|
})
|
|||
|
}
|
|||
|
|
|||
|
this.isShow = !this.isShow
|
|||
|
// 触发修改事件
|
|||
|
this.$emit('change', {
|
|||
|
index: this.index,
|
|||
|
show: this.isShow
|
|||
|
})
|
|||
|
// 只有在打开时才触发父元素的change
|
|||
|
if (this.isShow) this.parent && this.parent.onChange()
|
|||
|
this.$forceUpdate()
|
|||
|
},
|
|||
|
// 查询内容高度
|
|||
|
queryRect() {
|
|||
|
this._tGetRect('#'+this.elId).then(res => {
|
|||
|
if (res.height === 0) {
|
|||
|
this.initCount++
|
|||
|
if (this.initCount > 10) {
|
|||
|
this.initCount = 0
|
|||
|
return
|
|||
|
}
|
|||
|
setTimeout(() => {
|
|||
|
this.queryRect()
|
|||
|
}, 250)
|
|||
|
return
|
|||
|
}
|
|||
|
this.height = res.height
|
|||
|
this.initCount = 0
|
|||
|
})
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
</script>
|
|||
|
|
|||
|
<style lang="scss" scoped>
|
|||
|
|
|||
|
.tn-collapse-item {
|
|||
|
|
|||
|
&__head {
|
|||
|
position: relative;
|
|||
|
display: flex;
|
|||
|
flex-direction: row;
|
|||
|
justify-content: space-around;
|
|||
|
align-items: center;
|
|||
|
color: $tn-font-color;
|
|||
|
font-size: 30rpx;
|
|||
|
line-height: 1;
|
|||
|
padding: 24rpx 0;
|
|||
|
padding-left: 24rpx;
|
|||
|
text-align: left;
|
|||
|
background-color: #FFFFFF;
|
|||
|
|
|||
|
&__title {
|
|||
|
flex: 1;
|
|||
|
overflow: hidden;
|
|||
|
}
|
|||
|
|
|||
|
&__icon {
|
|||
|
&__arrow {
|
|||
|
transition: all 0.3s;
|
|||
|
margin-right: 20rpx;
|
|||
|
margin-left: 14rpx;
|
|||
|
font-size: inherit;
|
|||
|
|
|||
|
&--active {
|
|||
|
transform: rotate(180deg);
|
|||
|
transform-origin: center center;
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
&__body {
|
|||
|
transition: all 0.3s;
|
|||
|
overflow: hidden;
|
|||
|
|
|||
|
&__content {
|
|||
|
overflow: hidden;
|
|||
|
font-size: 28rpx;
|
|||
|
color: $tn-font-color;
|
|||
|
text-align: left;
|
|||
|
background-color: #FFFFFF;
|
|||
|
padding-left: 24rpx;
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
</style>
|