291 lines
7.4 KiB
Vue
291 lines
7.4 KiB
Vue
|
|
<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>
|