CNCEC_APP/uni_modules/uview-pro/components/u-collapse-item/u-collapse-item.vue

291 lines
7.4 KiB
Vue
Raw Normal View History

2026-03-25 14:54:15 +08:00
<template>
<view class="u-collapse-item" :style="$u.toStyle(itemStyle, customStyle)" :class="customClass">
<view
:hover-stay-time="200"
class="u-collapse-head"
@tap.stop="headClick"
:hover-class="hoverClass"
:style="headStyle"
>
<template v-if="!slots['title-all']">
<view v-if="!slots['title']" class="u-collapse-title u-line-1" :style="titleStyle">
{{ title }}
</view>
<slot v-else name="title" />
<view class="u-icon-wrap">
<u-icon v-if="showArrow" :color="arrowColor" :name="isShow ? 'arrow-up' : 'arrow-down'" />
</view>
</template>
<slot v-else name="title-all" />
</view>
<view class="u-collapse-body" :style="{ height: isShow ? height : '0' }">
<view class="u-collapse-content" :id="elId" :style="bodyStyle">
<slot></slot>
</view>
</view>
</view>
</template>
<script lang="ts">
export default {
name: 'u-collapse-item',
options: {
addGlobalClass: true,
// #ifndef MP-TOUTIAO
virtualHost: true,
// #endif
styleIsolation: 'shared'
}
};
</script>
<script setup lang="ts">
import { ref, watch, onMounted, useSlots, getCurrentInstance, nextTick, computed } from 'vue';
import { $u, useChildren } from '../..';
import { CollapseItemProps } from './types';
/**
* collapseItem 手风琴Item
* @description 通过折叠面板收纳内容区域搭配u-collapse使用
* @tutorial https://uviewpro.cn/zh/components/collapse.html
* @property {String} title 面板标题
* @property {String Number} index 主要用于事件的回调标识那个Item被点击
* @property {Boolean} disabled 面板是否可以打开或收起默认false
* @property {Boolean} open 设置某个面板的初始状态是否打开默认false
* @property {String Number} name 唯一标识符如不设置默认用当前collapse-item的索引值
* @property {String} align 标题的对齐方式默认left
* @property {Object} active-style 不显示箭头时可以添加当前选择的collapse-item活动样式对象形式
* @event {Function} change 某个item被打开或者收起时触发
* @example <u-collapse-item :title="item.head" v-for="(item, index) in itemList" :key="index">{{item.body}}</u-collapse-item>
*/
const props = defineProps(CollapseItemProps);
const emit = defineEmits(['change']);
const slots = useSlots();
const instance = getCurrentInstance();
const isShow = ref(false);
const elId = ref('');
const height = ref('0');
const headStyle = ref<Record<string, any>>({});
const bodyStyle = ref<Record<string, any>>({});
const itemStyle = ref<Record<string, any>>({});
const arrowColor = ref('');
const hoverClass = ref('');
// 使用通信库的子组件Hook
const { childId, parentExposed } = useChildren('u-collapse-item', 'u-collapse');
// 计算属性
const showArrow = computed(() => {
return parentExposed.value?.props ? parentExposed.value.props.arrow : true;
});
const titleStyle = computed(() => {
let style = { textAlign: props.align ? props.align : 'left' };
if (isShow.value && props.activeStyle && !showArrow.value) {
style = $u.deepMerge(style, props.activeStyle);
}
return $u.toStyle(style);
});
// 获取唯一标识符
const itemName = computed(() => {
// 优先级name > index > childId
if (props.name !== undefined && props.name !== '') {
return props.name;
} else if (props.index !== undefined && props.index !== '') {
return props.index;
} else {
return childId;
}
});
/**
* 设置显示状态
*/
function setShowState(show: boolean) {
if (isShow.value !== show) {
isShow.value = show;
// 如果展开,需要重新计算高度
if (show) {
nextTick(() => {
queryRect();
});
}
// 本地触发change事件
emit('change', {
index: props.index,
name: itemName.value,
show: isShow.value
});
}
}
/**
* 异步获取内容或者动态修改了内容时需要重新初始化
*/
function init() {
if (parentExposed.value?.props) {
headStyle.value = parentExposed.value.props.headStyle || {};
bodyStyle.value = parentExposed.value.props.bodyStyle || {};
arrowColor.value = parentExposed.value.props.arrowColor || 'var(--u-tips-color)';
hoverClass.value = parentExposed.value.props.hoverClass || 'u-hover-class';
itemStyle.value = parentExposed.value.props.itemStyle || {};
}
elId.value = $u.guid();
nextTick(() => {
queryRect();
});
}
/**
* 点击collapse head头部
*/
function headClick() {
if (props.disabled) return;
// 通知父组件状态变化
parentExposed?.value?.onChange(itemName.value);
}
/**
* 查询内容高度
*/
function queryRect() {
$u.getRect('#' + elId.value, instance)
.then((res: any) => {
if (res && res.height) {
height.value = res.height + 'px';
}
// #ifdef MP-TOUTIAO
if (isShow.value) {
height.value = 'auto';
}
// #endif
})
.catch((err: any) => {
console.warn('queryRect error:', err);
height.value = 'auto';
});
}
// 单选
function openSingle(data: any) {
// 只有目标项展开,其他都关闭
const shouldShow = data.targetName === itemName.value;
setShowState(shouldShow);
}
// 关闭所有
function closeAll() {
setShowState(false);
}
// 多选
function setMultiple(data: any) {
const shouldShow = data.targetNames.includes(itemName.value);
setShowState(shouldShow);
}
// 切换单个
function toggleSingle(data: any) {
// 只有目标项才切换状态
if (data.targetName === itemName.value) {
setShowState(!isShow.value);
}
}
onMounted(() => {
// 关键修复:根据 open 属性设置初始状态
setShowState(props.open);
// 初始化
init();
});
// 监听 open 属性变化
watch(
() => props.open,
newVal => {
setShowState(newVal);
}
);
// 监听父组件exposed的变化
watch(
parentExposed,
newExposed => {
if (newExposed) {
init();
}
},
{ deep: true, immediate: true }
);
defineExpose({
init,
isShow,
elId,
height,
headStyle,
bodyStyle,
itemStyle,
arrowColor,
hoverClass,
itemName: itemName.value,
queryRect,
setShowState,
openSingle,
closeAll,
setMultiple,
toggleSingle
});
</script>
<style lang="scss" scoped>
@import '../../libs/css/style.components.scss';
.u-collapse-head {
position: relative;
@include vue-flex;
justify-content: space-between;
align-items: center;
color: $u-main-color;
font-size: 30rpx;
line-height: 1;
padding: 24rpx 0;
text-align: left;
}
.u-collapse-title {
flex: 1;
overflow: hidden;
}
.u-arrow-down-icon {
transition: all 0.3s;
margin-right: 20rpx;
margin-left: 14rpx;
}
.u-arrow-down-icon-active {
transform: rotate(180deg);
transform-origin: center center;
}
.u-collapse-body {
overflow: hidden;
transition: all 0.3s;
}
.u-collapse-content {
overflow: hidden;
font-size: 28rpx;
color: $u-tips-color;
text-align: left;
}
</style>