maxkb/ui/src/components/common-list/index.vue
wangdan-fit2cloud 91dbe1907c feat: workspace
2025-06-26 16:36:15 +08:00

102 lines
2.1 KiB
Vue

<template>
<div class="common-list">
<ul v-if="data.length > 0">
<template v-for="(item, index) in data" :key="index">
<li
@click.stop="clickHandle(item, index)"
:class="current === item[props.valueKey] ? 'active color-primary-1' : ''"
class="cursor"
@mouseenter.stop="mouseenter(item)"
@mouseleave.stop="mouseleave()"
>
<slot :row="item" :index="index"> </slot>
</li>
</template>
</ul>
<slot name="empty" v-else>
<el-empty :description="$t('common.noData')" />
</slot>
</div>
</template>
<script setup lang="ts">
import { ref, watch } from 'vue'
defineOptions({ name: 'CommonList' })
const props = withDefaults(
defineProps<{
data: Array<any>
defaultActive?: string
valueKey?: string // 唯一标识的键名
}>(),
{
data: () => [],
defaultActive: '',
valueKey: 'id',
},
)
const current = ref<number | string>(0)
watch(
() => props.defaultActive,
(val) => {
current.value = val
},
{ immediate: true },
)
const emit = defineEmits(['click', 'mouseenter', 'mouseleave'])
function mouseenter(row: any) {
emit('mouseenter', row)
}
function mouseleave() {
emit('mouseleave')
}
function clickHandle(row: any, index: number) {
current.value = row[props.valueKey]
emit('click', row)
}
function clearCurrent() {
current.value = 0
}
defineExpose({
clearCurrent,
})
</script>
<style lang="scss" scoped>
/* 通用 ui li样式 */
.common-list {
li {
padding: 8px;
font-weight: 400;
font-size: 14px;
margin-bottom: 4px;
min-height: 24px;
line-height: 24px;
&.active {
background: var(--el-color-primary-light-9);
border-radius: var(--app-border-radius-base);
color: var(--el-color-primary);
font-weight: 500;
&:hover {
background: var(--el-color-primary-light-9);
}
}
&:hover {
border-radius: var(--app-border-radius-base);
background: var(--app-text-color-light-1);
}
&.is-active {
&:hover {
color: var(--el-color-primary);
background: var(--el-color-primary-light-9);
}
}
}
}
</style>