202 lines
5.7 KiB
Vue
202 lines
5.7 KiB
Vue
|
|
<script setup lang="ts">
|
||
|
|
import { computed, ref, watch } from 'vue';
|
||
|
|
import { filterProductMembers, formatProductMemberDate, getProductTeamTableHeight } from '../shared';
|
||
|
|
|
||
|
|
defineOptions({ name: 'SettingTeamPanel' });
|
||
|
|
|
||
|
|
interface Props {
|
||
|
|
members: Api.Product.ProductMember[];
|
||
|
|
roleOptions?: Api.SystemManage.RoleSimple[];
|
||
|
|
loading?: boolean;
|
||
|
|
readonly?: boolean;
|
||
|
|
}
|
||
|
|
|
||
|
|
interface Emits {
|
||
|
|
(e: 'create'): void;
|
||
|
|
(e: 'edit', member: Api.Product.ProductMember): void;
|
||
|
|
(e: 'remove', member: Api.Product.ProductMember): void;
|
||
|
|
}
|
||
|
|
|
||
|
|
const props = withDefaults(defineProps<Props>(), {
|
||
|
|
loading: false,
|
||
|
|
readonly: false,
|
||
|
|
roleOptions: () => []
|
||
|
|
});
|
||
|
|
const emit = defineEmits<Emits>();
|
||
|
|
const searchKeyword = ref('');
|
||
|
|
const selectedRoleId = ref('');
|
||
|
|
const teamTableHeight = getProductTeamTableHeight(5);
|
||
|
|
const roleFilterOptions = computed(() => {
|
||
|
|
const roleMap = new Map<string, string>();
|
||
|
|
|
||
|
|
props.roleOptions.forEach(role => {
|
||
|
|
if (!roleMap.has(role.id)) {
|
||
|
|
roleMap.set(role.id, role.name);
|
||
|
|
}
|
||
|
|
});
|
||
|
|
|
||
|
|
return [...roleMap.entries()].map(([value, label]) => ({
|
||
|
|
value,
|
||
|
|
label
|
||
|
|
}));
|
||
|
|
});
|
||
|
|
const filteredMembers = computed(() =>
|
||
|
|
filterProductMembers(props.members, {
|
||
|
|
keyword: searchKeyword.value,
|
||
|
|
roleId: selectedRoleId.value
|
||
|
|
})
|
||
|
|
);
|
||
|
|
const hasFilter = computed(() => Boolean(searchKeyword.value.trim() || selectedRoleId.value));
|
||
|
|
|
||
|
|
watch(roleFilterOptions, options => {
|
||
|
|
if (selectedRoleId.value && !options.some(item => item.value === selectedRoleId.value)) {
|
||
|
|
selectedRoleId.value = '';
|
||
|
|
}
|
||
|
|
});
|
||
|
|
|
||
|
|
function getMemberStatusLabel(status: Api.Product.ProductMemberStatus) {
|
||
|
|
return status === 0 ? '有效' : '失效';
|
||
|
|
}
|
||
|
|
|
||
|
|
function getMemberStatusTagType(status: Api.Product.ProductMemberStatus) {
|
||
|
|
return status === 0 ? 'success' : 'info';
|
||
|
|
}
|
||
|
|
</script>
|
||
|
|
|
||
|
|
<template>
|
||
|
|
<ElCard class="card-wrapper">
|
||
|
|
<template #header>
|
||
|
|
<div class="setting-team-panel__header">
|
||
|
|
<div>
|
||
|
|
<h3 class="text-16px text-[#0f172a] font-700">团队管理</h3>
|
||
|
|
</div>
|
||
|
|
<div class="setting-team-panel__toolbar">
|
||
|
|
<ElSelect v-model="selectedRoleId" clearable placeholder="筛选角色" class="setting-team-panel__role-filter">
|
||
|
|
<ElOption
|
||
|
|
v-for="option in roleFilterOptions"
|
||
|
|
:key="option.value"
|
||
|
|
:label="option.label"
|
||
|
|
:value="option.value"
|
||
|
|
/>
|
||
|
|
</ElSelect>
|
||
|
|
<ElInput v-model="searchKeyword" clearable placeholder="搜索成员姓名" class="setting-team-panel__search" />
|
||
|
|
<ElButton
|
||
|
|
v-if="!props.readonly"
|
||
|
|
v-auth="{ code: 'project:product:update', source: 'object' }"
|
||
|
|
type="primary"
|
||
|
|
plain
|
||
|
|
@click="emit('create')"
|
||
|
|
>
|
||
|
|
新增成员
|
||
|
|
</ElButton>
|
||
|
|
</div>
|
||
|
|
</div>
|
||
|
|
</template>
|
||
|
|
|
||
|
|
<ElTable
|
||
|
|
v-loading="props.loading"
|
||
|
|
:data="filteredMembers"
|
||
|
|
:height="teamTableHeight"
|
||
|
|
:empty-text="hasFilter ? '未找到匹配成员' : '暂无成员'"
|
||
|
|
border
|
||
|
|
row-key="id"
|
||
|
|
>
|
||
|
|
<ElTableColumn type="index" label="序号" width="64" align="center" />
|
||
|
|
<ElTableColumn prop="userNickname" label="成员姓名" min-width="140" />
|
||
|
|
<ElTableColumn prop="roleName" label="当前角色" min-width="140" />
|
||
|
|
<ElTableColumn label="成员状态" width="110" align="center">
|
||
|
|
<template #default="{ row }">
|
||
|
|
<ElTag :type="getMemberStatusTagType(row.status)">{{ getMemberStatusLabel(row.status) }}</ElTag>
|
||
|
|
</template>
|
||
|
|
</ElTableColumn>
|
||
|
|
<ElTableColumn prop="joinedTime" label="加入时间" min-width="132" align="center">
|
||
|
|
<template #default="{ row }">
|
||
|
|
{{ formatProductMemberDate(row.joinedTime) }}
|
||
|
|
</template>
|
||
|
|
</ElTableColumn>
|
||
|
|
<ElTableColumn prop="leftTime" label="退出时间" min-width="170">
|
||
|
|
<template #default="{ row }">
|
||
|
|
{{ formatProductMemberDate(row.leftTime) }}
|
||
|
|
</template>
|
||
|
|
</ElTableColumn>
|
||
|
|
<ElTableColumn prop="remark" label="备注" min-width="180" show-overflow-tooltip>
|
||
|
|
<template #default="{ row }">
|
||
|
|
{{ row.remark || '--' }}
|
||
|
|
</template>
|
||
|
|
</ElTableColumn>
|
||
|
|
<ElTableColumn v-if="!props.readonly" label="操作" width="180" fixed="right" align="center">
|
||
|
|
<template #default="{ row }">
|
||
|
|
<div class="setting-team-panel__actions">
|
||
|
|
<ElButton
|
||
|
|
v-auth="{ code: 'project:product:update', source: 'object' }"
|
||
|
|
link
|
||
|
|
type="primary"
|
||
|
|
:disabled="row.status !== 0 || row.managerFlag"
|
||
|
|
@click="emit('edit', row)"
|
||
|
|
>
|
||
|
|
调整角色
|
||
|
|
</ElButton>
|
||
|
|
<ElButton
|
||
|
|
v-auth="{ code: 'project:product:update', source: 'object' }"
|
||
|
|
link
|
||
|
|
type="danger"
|
||
|
|
:disabled="row.status !== 0 || row.managerFlag"
|
||
|
|
@click="emit('remove', row)"
|
||
|
|
>
|
||
|
|
移出成员
|
||
|
|
</ElButton>
|
||
|
|
</div>
|
||
|
|
</template>
|
||
|
|
</ElTableColumn>
|
||
|
|
</ElTable>
|
||
|
|
</ElCard>
|
||
|
|
</template>
|
||
|
|
|
||
|
|
<style scoped>
|
||
|
|
.setting-team-panel__header {
|
||
|
|
display: flex;
|
||
|
|
align-items: center;
|
||
|
|
justify-content: space-between;
|
||
|
|
gap: 12px;
|
||
|
|
}
|
||
|
|
|
||
|
|
.setting-team-panel__toolbar {
|
||
|
|
display: inline-flex;
|
||
|
|
align-items: center;
|
||
|
|
gap: 12px;
|
||
|
|
}
|
||
|
|
|
||
|
|
.setting-team-panel__search {
|
||
|
|
width: 220px;
|
||
|
|
}
|
||
|
|
|
||
|
|
.setting-team-panel__role-filter {
|
||
|
|
width: 180px;
|
||
|
|
}
|
||
|
|
|
||
|
|
.setting-team-panel__actions {
|
||
|
|
display: inline-flex;
|
||
|
|
align-items: center;
|
||
|
|
gap: 12px;
|
||
|
|
}
|
||
|
|
|
||
|
|
@media (width <= 768px) {
|
||
|
|
.setting-team-panel__header {
|
||
|
|
align-items: flex-start;
|
||
|
|
flex-direction: column;
|
||
|
|
}
|
||
|
|
|
||
|
|
.setting-team-panel__toolbar {
|
||
|
|
width: 100%;
|
||
|
|
}
|
||
|
|
|
||
|
|
.setting-team-panel__search {
|
||
|
|
width: 100%;
|
||
|
|
}
|
||
|
|
|
||
|
|
.setting-team-panel__role-filter {
|
||
|
|
width: 100%;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
</style>
|