1158 lines
33 KiB
Vue
1158 lines
33 KiB
Vue
<script setup lang="ts">
|
||
import { computed, onBeforeUnmount, onMounted, ref, watch } from 'vue';
|
||
import { PRODUCT_MANAGER_ROLE_CODE } from '@/constants/business';
|
||
import { fetchGetDeptSimpleList, fetchGetUserManagementRelationTree } from '@/service/api';
|
||
import BusinessFormDialog from '@/components/custom/business-form-dialog.vue';
|
||
import { buildMenuTree } from '@/views/system/shared/menu-tree';
|
||
import IconEpUser from '~icons/ep/user';
|
||
import IconEpOfficeBuilding from '~icons/ep/office-building';
|
||
|
||
defineOptions({ name: 'ProductTeamBatchDialog' });
|
||
|
||
type Source = 'dept' | 'chain' | 'all';
|
||
|
||
export interface BatchMemberPayload {
|
||
userId: string;
|
||
roleId: string;
|
||
remark: string;
|
||
}
|
||
|
||
interface Props {
|
||
userOptions: Api.SystemManage.UserSimple[];
|
||
roleOptions: Api.SystemManage.RoleSimple[];
|
||
/** 已加入产品的 userId(包含产品经理 locked 行),作为禁选 */
|
||
disabledUserIds: readonly string[];
|
||
}
|
||
|
||
interface Emits {
|
||
(e: 'submit', payloads: BatchMemberPayload[]): void;
|
||
}
|
||
|
||
const props = defineProps<Props>();
|
||
const emit = defineEmits<Emits>();
|
||
|
||
const visible = defineModel<boolean>('visible', { default: false });
|
||
|
||
const source = ref<Source>('dept');
|
||
const currentNodeId = ref<string | null>(null);
|
||
const selected = ref<Set<string>>(new Set());
|
||
const roleOverrides = ref<Map<string, string>>(new Map());
|
||
const hideAdded = ref(false);
|
||
const treeSearch = ref('');
|
||
const userSearch = ref('');
|
||
const defaultRoleId = ref('');
|
||
const batchRemark = ref('');
|
||
|
||
const deptTree = ref<Api.SystemManage.DeptSimple[]>([]);
|
||
const deptTreeLoading = ref(false);
|
||
let deptTreeLoaded = false;
|
||
|
||
async function ensureDeptTree() {
|
||
if (deptTreeLoaded) return;
|
||
deptTreeLoading.value = true;
|
||
const { data } = await fetchGetDeptSimpleList();
|
||
deptTreeLoading.value = false;
|
||
// 后端返回平铺数组(每条带 parentId),前端按 parentId 组装为嵌套树
|
||
deptTree.value = data ? buildMenuTree(data) : [];
|
||
deptTreeLoaded = true;
|
||
}
|
||
|
||
const chainTree = ref<Api.SystemManage.UserManagementRelationTreeRespVO[]>([]);
|
||
const chainTreeLoading = ref(false);
|
||
let chainTreeLoaded = false;
|
||
|
||
async function ensureChainTree() {
|
||
if (chainTreeLoaded) return;
|
||
chainTreeLoading.value = true;
|
||
const { data } = await fetchGetUserManagementRelationTree({ fromUserIndex: false });
|
||
chainTreeLoading.value = false;
|
||
chainTree.value = data ?? [];
|
||
chainTreeLoaded = true;
|
||
}
|
||
|
||
const disabledUserIdSet = computed(() => new Set(props.disabledUserIds.map(String)));
|
||
|
||
const selectableRoles = computed(() =>
|
||
props.roleOptions.filter(role => role.code !== PRODUCT_MANAGER_ROLE_CODE && role.visible !== 0)
|
||
);
|
||
|
||
const selectedCount = computed(() => selected.value.size);
|
||
const visibleSelectedIds = computed(() => [...selected.value].slice(0, 4));
|
||
const overflowSelectedCount = computed(() => Math.max(0, selected.value.size - 4));
|
||
|
||
const overflowPopoverVisible = ref(false);
|
||
const overflowReferenceEl = ref<HTMLElement | null>(null);
|
||
|
||
function handleOverflowOutsideClick(e: MouseEvent) {
|
||
if (!overflowPopoverVisible.value) return;
|
||
const target = e.target as HTMLElement | null;
|
||
if (!target) return;
|
||
// 点击在 popover 自身 / 任意 popper / dropdown 菜单 / reference 按钮上,都不关
|
||
if (target.closest('.team-batch__overflow-popper')) return;
|
||
if (target.closest('.el-popper')) return;
|
||
if (target.closest('.el-dropdown-menu')) return;
|
||
if (target.closest('.el-dropdown__popper')) return;
|
||
if (overflowReferenceEl.value?.contains(target)) return;
|
||
overflowPopoverVisible.value = false;
|
||
}
|
||
|
||
onMounted(() => document.addEventListener('mousedown', handleOverflowOutsideClick, true));
|
||
onBeforeUnmount(() => document.removeEventListener('mousedown', handleOverflowOutsideClick, true));
|
||
const overrideCount = computed(() => {
|
||
let count = 0;
|
||
for (const uid of roleOverrides.value.keys()) {
|
||
if (selected.value.has(uid)) count += 1;
|
||
}
|
||
return count;
|
||
});
|
||
|
||
const confirmDisabled = computed(() => selectedCount.value === 0 || !defaultRoleId.value);
|
||
|
||
function resetState() {
|
||
source.value = 'dept';
|
||
currentNodeId.value = null;
|
||
selected.value = new Set();
|
||
roleOverrides.value = new Map();
|
||
hideAdded.value = false;
|
||
treeSearch.value = '';
|
||
userSearch.value = '';
|
||
defaultRoleId.value = '';
|
||
batchRemark.value = '';
|
||
overflowPopoverVisible.value = false;
|
||
}
|
||
|
||
function collectDeptIds(node: Api.SystemManage.DeptSimple): string[] {
|
||
const ids: string[] = [String(node.id)];
|
||
if (node.children) {
|
||
for (const c of node.children) ids.push(...collectDeptIds(c));
|
||
}
|
||
return ids;
|
||
}
|
||
|
||
function getDeptUserIds(node: Api.SystemManage.DeptSimple): string[] {
|
||
const deptIds = new Set(collectDeptIds(node));
|
||
return props.userOptions
|
||
.filter(u => u.deptId !== null && u.deptId !== undefined && deptIds.has(String(u.deptId)))
|
||
.map(u => String(u.id));
|
||
}
|
||
|
||
type TreeCheckState = 'none' | 'partial' | 'all';
|
||
|
||
function getDeptCheckState(node: Api.SystemManage.DeptSimple): TreeCheckState {
|
||
const ids = getDeptUserIds(node).filter(id => !disabledUserIdSet.value.has(id));
|
||
if (!ids.length) return 'none';
|
||
const sel = ids.filter(id => selected.value.has(id)).length;
|
||
if (sel === 0) return 'none';
|
||
if (sel === ids.length) return 'all';
|
||
return 'partial';
|
||
}
|
||
|
||
function findDeptNode(list: Api.SystemManage.DeptSimple[], key: string): Api.SystemManage.DeptSimple | null {
|
||
for (const n of list) {
|
||
if (String(n.id) === key) return n;
|
||
if (n.children) {
|
||
const r = findDeptNode(n.children, key);
|
||
if (r) return r;
|
||
}
|
||
}
|
||
return null;
|
||
}
|
||
|
||
function chainNodeKey(node: Api.SystemManage.UserManagementRelationTreeRespVO): string {
|
||
return node.id ?? `chain_${node.userId}`;
|
||
}
|
||
|
||
function getChainUserIds(node: Api.SystemManage.UserManagementRelationTreeRespVO): string[] {
|
||
const ids = new Set<string>([String(node.userId)]);
|
||
if (node.children) {
|
||
for (const c of node.children) {
|
||
for (const id of getChainUserIds(c)) ids.add(id);
|
||
}
|
||
}
|
||
return [...ids];
|
||
}
|
||
|
||
function getChainCheckState(node: Api.SystemManage.UserManagementRelationTreeRespVO): TreeCheckState {
|
||
const ids = getChainUserIds(node).filter(id => !disabledUserIdSet.value.has(id));
|
||
if (!ids.length) return 'none';
|
||
const sel = ids.filter(id => selected.value.has(id)).length;
|
||
if (sel === 0) return 'none';
|
||
if (sel === ids.length) return 'all';
|
||
return 'partial';
|
||
}
|
||
|
||
function findChainNode(
|
||
list: Api.SystemManage.UserManagementRelationTreeRespVO[],
|
||
key: string
|
||
): Api.SystemManage.UserManagementRelationTreeRespVO | null {
|
||
for (const n of list) {
|
||
if (chainNodeKey(n) === key) return n;
|
||
if (n.children) {
|
||
const r = findChainNode(n.children, key);
|
||
if (r) return r;
|
||
}
|
||
}
|
||
return null;
|
||
}
|
||
|
||
function toggleChainCheck(node: Api.SystemManage.UserManagementRelationTreeRespVO) {
|
||
const ids = getChainUserIds(node).filter(id => !disabledUserIdSet.value.has(id));
|
||
const state = getChainCheckState(node);
|
||
if (state === 'all') {
|
||
for (const id of ids) {
|
||
selected.value.delete(id);
|
||
roleOverrides.value.delete(id);
|
||
}
|
||
} else {
|
||
for (const id of ids) selected.value.add(id);
|
||
}
|
||
selected.value = new Set(selected.value);
|
||
roleOverrides.value = new Map(roleOverrides.value);
|
||
}
|
||
|
||
function activateNode(key: string) {
|
||
currentNodeId.value = key;
|
||
}
|
||
|
||
function handleDeptNodeClick(data: Api.SystemManage.DeptSimple) {
|
||
activateNode(String(data.id));
|
||
}
|
||
|
||
function handleChainNodeClick(data: Api.SystemManage.UserManagementRelationTreeRespVO) {
|
||
activateNode(chainNodeKey(data));
|
||
}
|
||
|
||
function toggleDeptCheck(node: Api.SystemManage.DeptSimple) {
|
||
const ids = getDeptUserIds(node).filter(id => !disabledUserIdSet.value.has(id));
|
||
const state = getDeptCheckState(node);
|
||
if (state === 'all') {
|
||
for (const id of ids) {
|
||
selected.value.delete(id);
|
||
roleOverrides.value.delete(id);
|
||
}
|
||
} else {
|
||
for (const id of ids) selected.value.add(id);
|
||
}
|
||
selected.value = new Set(selected.value);
|
||
roleOverrides.value = new Map(roleOverrides.value);
|
||
}
|
||
|
||
const deptTreeProps = { children: 'children', label: 'name' } as const;
|
||
const chainTreeProps = { children: 'children', label: 'userNickname' } as const;
|
||
|
||
function getDeptCheckStateClass(node: Api.SystemManage.DeptSimple) {
|
||
const s = getDeptCheckState(node);
|
||
return { 'is-checked': s === 'all', 'is-partial': s === 'partial' };
|
||
}
|
||
|
||
function getDeptMetaText(node: Api.SystemManage.DeptSimple): string {
|
||
const total = getDeptUserIds(node).length;
|
||
return total > 0 ? `${total} 人` : '';
|
||
}
|
||
|
||
function getChainCheckStateClass(node: Api.SystemManage.UserManagementRelationTreeRespVO) {
|
||
const s = getChainCheckState(node);
|
||
return { 'is-checked': s === 'all', 'is-partial': s === 'partial' };
|
||
}
|
||
|
||
function getChainMetaText(node: Api.SystemManage.UserManagementRelationTreeRespVO): string {
|
||
const total = getChainUserIds(node).length;
|
||
return total > 1 ? `${total} 人` : '';
|
||
}
|
||
|
||
function visibleUserIds(): string[] {
|
||
if (source.value === 'all') return props.userOptions.map(u => String(u.id));
|
||
if (!currentNodeId.value) return props.userOptions.map(u => String(u.id));
|
||
if (source.value === 'dept') {
|
||
const node = findDeptNode(deptTree.value, currentNodeId.value);
|
||
return node ? getDeptUserIds(node) : props.userOptions.map(u => String(u.id));
|
||
}
|
||
if (source.value === 'chain') {
|
||
const node = findChainNode(chainTree.value, currentNodeId.value);
|
||
return node ? getChainUserIds(node) : props.userOptions.map(u => String(u.id));
|
||
}
|
||
return props.userOptions.map(u => String(u.id));
|
||
}
|
||
|
||
function matchDeptKeyword(node: Api.SystemManage.DeptSimple, kw: string): boolean {
|
||
if (!kw) return true;
|
||
if (node.name.toLowerCase().includes(kw)) return true;
|
||
if (node.children) return node.children.some(c => matchDeptKeyword(c, kw));
|
||
return false;
|
||
}
|
||
|
||
function matchChainKeyword(node: Api.SystemManage.UserManagementRelationTreeRespVO, kw: string): boolean {
|
||
if (!kw) return true;
|
||
if (node.userNickname.toLowerCase().includes(kw)) return true;
|
||
if (node.children) return node.children.some(c => matchChainKeyword(c, kw));
|
||
return false;
|
||
}
|
||
|
||
const visibleDeptTree = computed(() => {
|
||
const kw = treeSearch.value.trim().toLowerCase();
|
||
if (!kw) return deptTree.value;
|
||
return deptTree.value.filter(n => matchDeptKeyword(n, kw));
|
||
});
|
||
|
||
const visibleChainTree = computed(() => {
|
||
const kw = treeSearch.value.trim().toLowerCase();
|
||
if (!kw) return chainTree.value;
|
||
return chainTree.value.filter(n => matchChainKeyword(n, kw));
|
||
});
|
||
|
||
const filteredUserIds = computed(() => {
|
||
let ids = visibleUserIds();
|
||
if (hideAdded.value) ids = ids.filter(id => !disabledUserIdSet.value.has(id));
|
||
const kw = userSearch.value.trim().toLowerCase();
|
||
if (kw) {
|
||
ids = ids.filter(id => {
|
||
const u = getUserById(id);
|
||
if (!u) return false;
|
||
return u.nickname.toLowerCase().includes(kw) || (u.username ?? '').toLowerCase().includes(kw);
|
||
});
|
||
}
|
||
return ids;
|
||
});
|
||
|
||
async function switchSource(next: Source) {
|
||
if (source.value === next) return;
|
||
source.value = next;
|
||
currentNodeId.value = null;
|
||
treeSearch.value = '';
|
||
if (next === 'dept') await ensureDeptTree();
|
||
else if (next === 'chain') await ensureChainTree();
|
||
}
|
||
|
||
function toggleUser(uid: string) {
|
||
if (disabledUserIdSet.value.has(uid)) return;
|
||
if (selected.value.has(uid)) {
|
||
selected.value.delete(uid);
|
||
roleOverrides.value.delete(uid);
|
||
} else {
|
||
selected.value.add(uid);
|
||
}
|
||
selected.value = new Set(selected.value);
|
||
roleOverrides.value = new Map(roleOverrides.value);
|
||
}
|
||
|
||
function clearAll() {
|
||
selected.value = new Set();
|
||
roleOverrides.value = new Map();
|
||
}
|
||
|
||
function getUserById(uid: string) {
|
||
return props.userOptions.find(u => String(u.id) === uid);
|
||
}
|
||
|
||
function getRoleNameById(roleId: string): string {
|
||
return props.roleOptions.find(r => r.id === roleId)?.name ?? '';
|
||
}
|
||
|
||
function setRoleOverride(uid: string, roleId: string | null) {
|
||
if (roleId === null) roleOverrides.value.delete(uid);
|
||
else roleOverrides.value.set(uid, roleId);
|
||
roleOverrides.value = new Map(roleOverrides.value);
|
||
}
|
||
|
||
function clearUserFilter() {
|
||
userSearch.value = '';
|
||
hideAdded.value = false;
|
||
}
|
||
|
||
async function handleConfirm() {
|
||
if (confirmDisabled.value) return;
|
||
const payloads: BatchMemberPayload[] = [...selected.value].map(uid => ({
|
||
userId: uid,
|
||
roleId: roleOverrides.value.get(uid) ?? defaultRoleId.value,
|
||
remark: batchRemark.value.trim()
|
||
}));
|
||
emit('submit', payloads);
|
||
}
|
||
|
||
watch(visible, async value => {
|
||
if (!value) {
|
||
resetState();
|
||
return;
|
||
}
|
||
if (source.value === 'dept') await ensureDeptTree();
|
||
else if (source.value === 'chain') await ensureChainTree();
|
||
});
|
||
</script>
|
||
|
||
<template>
|
||
<BusinessFormDialog
|
||
v-model="visible"
|
||
title="新增团队成员"
|
||
preset="lg"
|
||
width="820px"
|
||
max-body-height="540px"
|
||
:confirm-disabled="confirmDisabled"
|
||
:confirm-text="`确认添加(${selectedCount})`"
|
||
@confirm="handleConfirm"
|
||
>
|
||
<div class="team-batch">
|
||
<div class="team-batch__tabs">
|
||
<button
|
||
v-for="tab in [
|
||
{ key: 'dept', label: '部门' },
|
||
{ key: 'chain', label: '团队' },
|
||
{ key: 'all', label: '全部用户' }
|
||
]"
|
||
:key="tab.key"
|
||
class="team-batch__tab"
|
||
:class="{ 'is-active': source === tab.key }"
|
||
type="button"
|
||
@click="switchSource(tab.key as Source)"
|
||
>
|
||
{{ tab.label }}
|
||
</button>
|
||
</div>
|
||
|
||
<div class="team-batch__picker" :class="{ 'is-single': source === 'all' }">
|
||
<!-- 左栏:dept / chain 树,all 时隐藏 -->
|
||
<div v-if="source !== 'all'" class="team-batch__col team-batch__col--tree">
|
||
<div class="team-batch__col-head">{{ source === 'dept' ? '部门' : '团队' }}</div>
|
||
<div class="team-batch__search">
|
||
<ElInput
|
||
v-model="treeSearch"
|
||
size="small"
|
||
clearable
|
||
:placeholder="source === 'dept' ? '搜索部门…' : '搜索成员…'"
|
||
/>
|
||
</div>
|
||
<div v-loading="source === 'dept' ? deptTreeLoading : chainTreeLoading" class="team-batch__col-body">
|
||
<ElTree
|
||
v-if="source === 'dept'"
|
||
:data="visibleDeptTree"
|
||
:props="deptTreeProps"
|
||
node-key="id"
|
||
:expand-on-click-node="false"
|
||
:default-expand-all="true"
|
||
:indent="14"
|
||
class="team-batch__tree"
|
||
@node-click="handleDeptNodeClick"
|
||
>
|
||
<template #default="{ data }">
|
||
<div class="team-batch__node" :class="{ 'is-active': currentNodeId === String(data.id) }">
|
||
<span
|
||
class="team-batch__node-check"
|
||
:class="getDeptCheckStateClass(data)"
|
||
@click.stop="toggleDeptCheck(data)"
|
||
/>
|
||
<IconEpOfficeBuilding class="team-batch__node-icon" />
|
||
<span class="team-batch__node-label">{{ data.name }}</span>
|
||
<span v-if="getDeptMetaText(data)" class="team-batch__node-meta">
|
||
{{ getDeptMetaText(data) }}
|
||
</span>
|
||
</div>
|
||
</template>
|
||
</ElTree>
|
||
<ElTree
|
||
v-else
|
||
:data="visibleChainTree"
|
||
:props="chainTreeProps"
|
||
node-key="userId"
|
||
:expand-on-click-node="false"
|
||
:default-expand-all="true"
|
||
:indent="14"
|
||
class="team-batch__tree"
|
||
@node-click="handleChainNodeClick"
|
||
>
|
||
<template #default="{ data }">
|
||
<div class="team-batch__node" :class="{ 'is-active': currentNodeId === chainNodeKey(data) }">
|
||
<span
|
||
class="team-batch__node-check"
|
||
:class="getChainCheckStateClass(data)"
|
||
@click.stop="toggleChainCheck(data)"
|
||
/>
|
||
<IconEpUser class="team-batch__node-icon" />
|
||
<span class="team-batch__node-label">{{ data.userNickname }}</span>
|
||
<span v-if="getChainMetaText(data)" class="team-batch__node-meta">
|
||
{{ getChainMetaText(data) }}
|
||
</span>
|
||
</div>
|
||
</template>
|
||
</ElTree>
|
||
</div>
|
||
</div>
|
||
<!-- 右栏:候选用户 -->
|
||
<div class="team-batch__col team-batch__col--users">
|
||
<div class="team-batch__col-head team-batch__col-head--user">
|
||
<span>
|
||
候选用户(
|
||
<span>{{ filteredUserIds.length }}</span>
|
||
人)
|
||
</span>
|
||
<label class="team-batch__hide-added">
|
||
<ElCheckbox v-model="hideAdded">隐藏已添加</ElCheckbox>
|
||
</label>
|
||
</div>
|
||
<div class="team-batch__search">
|
||
<ElInput v-model="userSearch" size="small" clearable placeholder="搜索用户名 / 部门…" />
|
||
</div>
|
||
<div class="team-batch__col-body">
|
||
<div v-if="!filteredUserIds.length" class="team-batch__empty">
|
||
该节点下没有匹配用户
|
||
<button
|
||
v-if="userSearch || hideAdded"
|
||
type="button"
|
||
class="team-batch__link team-batch__empty-action"
|
||
@click="clearUserFilter"
|
||
>
|
||
清除筛选条件
|
||
</button>
|
||
</div>
|
||
<div
|
||
v-for="uid in filteredUserIds"
|
||
:key="uid"
|
||
class="team-batch__user-row"
|
||
:class="{ 'is-disabled': disabledUserIdSet.has(uid) }"
|
||
@click="toggleUser(uid)"
|
||
>
|
||
<span class="team-batch__node-check" :class="{ 'is-checked': selected.has(uid) }" />
|
||
<span class="team-batch__user-avatar">{{ (getUserById(uid)?.nickname ?? '?').slice(0, 1) }}</span>
|
||
<div class="team-batch__user-main">
|
||
<div class="team-batch__user-name">{{ getUserById(uid)?.nickname }}</div>
|
||
</div>
|
||
<span v-if="disabledUserIdSet.has(uid)" class="team-batch__user-tag">已添加</span>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
<div class="team-batch__selected">
|
||
<div class="team-batch__selected-head">
|
||
<span>
|
||
已选
|
||
<strong>{{ selectedCount }}</strong>
|
||
人
|
||
<span v-if="overrideCount > 0">
|
||
· 其中
|
||
<strong class="team-batch__override-cnt">{{ overrideCount }}</strong>
|
||
人已自定义角色
|
||
</span>
|
||
</span>
|
||
<button type="button" class="team-batch__link team-batch__link--danger" @click="clearAll">清空</button>
|
||
</div>
|
||
<div v-if="selectedCount === 0" class="team-batch__selected-empty">从左侧勾选用户后会出现在这里</div>
|
||
<div v-else class="team-batch__chips">
|
||
<span v-for="uid in visibleSelectedIds" :key="uid" class="team-batch__chip">
|
||
<span class="team-batch__chip-name">{{ getUserById(uid)?.nickname }}</span>
|
||
<ElDropdown
|
||
trigger="click"
|
||
@command="roleId => setRoleOverride(uid, roleId === '__default__' ? null : (roleId as string))"
|
||
>
|
||
<button type="button" class="team-batch__chip-role" :class="{ 'is-override': roleOverrides.has(uid) }">
|
||
{{
|
||
roleOverrides.get(uid)
|
||
? getRoleNameById(roleOverrides.get(uid)!)
|
||
: defaultRoleId
|
||
? getRoleNameById(defaultRoleId)
|
||
: '默认'
|
||
}}
|
||
▾
|
||
</button>
|
||
<template #dropdown>
|
||
<ElDropdownMenu>
|
||
<ElDropdownItem command="__default__">使用默认</ElDropdownItem>
|
||
<ElDropdownItem v-for="role in selectableRoles" :key="role.id" :command="role.id">
|
||
<div class="team-batch__role-option">
|
||
<span class="team-batch__role-option-name">{{ role.name }}</span>
|
||
<ElTooltip v-if="role.remark" :content="role.remark" placement="right" :show-after="120">
|
||
<icon-ep:info-filled class="team-batch__role-option-info" @click.stop />
|
||
</ElTooltip>
|
||
</div>
|
||
</ElDropdownItem>
|
||
</ElDropdownMenu>
|
||
</template>
|
||
</ElDropdown>
|
||
<button type="button" class="team-batch__chip-x" @click="toggleUser(uid)">×</button>
|
||
</span>
|
||
<ElPopover
|
||
v-if="overflowSelectedCount > 0"
|
||
:visible="overflowPopoverVisible"
|
||
placement="top-end"
|
||
:width="360"
|
||
popper-class="team-batch__overflow-popper"
|
||
>
|
||
<template #reference>
|
||
<button
|
||
ref="overflowReferenceEl"
|
||
type="button"
|
||
class="team-batch__chip-more"
|
||
@click="overflowPopoverVisible = !overflowPopoverVisible"
|
||
>
|
||
+{{ overflowSelectedCount }} 更多
|
||
</button>
|
||
</template>
|
||
<div class="team-batch__overflow-head">
|
||
<span>
|
||
另外
|
||
<strong>{{ overflowSelectedCount }}</strong>
|
||
人
|
||
</span>
|
||
<button type="button" class="team-batch__link team-batch__link--danger" @click="clearAll">
|
||
清空全部
|
||
</button>
|
||
</div>
|
||
<div class="team-batch__overflow-chips">
|
||
<span v-for="uid in [...selected].slice(4)" :key="uid" class="team-batch__chip">
|
||
<span class="team-batch__chip-name">{{ getUserById(uid)?.nickname }}</span>
|
||
<ElDropdown
|
||
trigger="click"
|
||
@command="roleId => setRoleOverride(uid, roleId === '__default__' ? null : (roleId as string))"
|
||
>
|
||
<button
|
||
type="button"
|
||
class="team-batch__chip-role"
|
||
:class="{ 'is-override': roleOverrides.has(uid) }"
|
||
>
|
||
{{
|
||
roleOverrides.get(uid)
|
||
? getRoleNameById(roleOverrides.get(uid)!)
|
||
: defaultRoleId
|
||
? getRoleNameById(defaultRoleId)
|
||
: '默认'
|
||
}}
|
||
▾
|
||
</button>
|
||
<template #dropdown>
|
||
<ElDropdownMenu>
|
||
<ElDropdownItem command="__default__">使用默认</ElDropdownItem>
|
||
<ElDropdownItem v-for="role in selectableRoles" :key="role.id" :command="role.id">
|
||
<div class="team-batch__role-option">
|
||
<span class="team-batch__role-option-name">{{ role.name }}</span>
|
||
<ElTooltip v-if="role.remark" :content="role.remark" placement="right" :show-after="120">
|
||
<icon-ep:info-filled class="team-batch__role-option-info" @click.stop />
|
||
</ElTooltip>
|
||
</div>
|
||
</ElDropdownItem>
|
||
</ElDropdownMenu>
|
||
</template>
|
||
</ElDropdown>
|
||
<button type="button" class="team-batch__chip-x" @click="toggleUser(uid)">×</button>
|
||
</span>
|
||
</div>
|
||
</ElPopover>
|
||
</div>
|
||
</div>
|
||
|
||
<div class="team-batch__params">
|
||
<div class="team-batch__params-grid">
|
||
<div class="team-batch__field">
|
||
<label class="team-batch__field-label">
|
||
默认角色
|
||
<span class="team-batch__required">*</span>
|
||
</label>
|
||
<ElSelect v-model="defaultRoleId" placeholder="请选择默认角色">
|
||
<ElOption v-for="role in selectableRoles" :key="role.id" :label="role.name" :value="role.id">
|
||
<div class="team-batch__role-option">
|
||
<span class="team-batch__role-option-name">{{ role.name }}</span>
|
||
<ElTooltip v-if="role.remark" :content="role.remark" placement="right" :show-after="120">
|
||
<icon-ep:info-filled class="team-batch__role-option-info" @click.stop />
|
||
</ElTooltip>
|
||
</div>
|
||
</ElOption>
|
||
</ElSelect>
|
||
</div>
|
||
<div class="team-batch__field">
|
||
<label class="team-batch__field-label">备注(可选,统一应用)</label>
|
||
<ElInput v-model="batchRemark" maxlength="200" placeholder="如:本次批量加入" />
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</BusinessFormDialog>
|
||
</template>
|
||
|
||
<style scoped>
|
||
.team-batch {
|
||
display: flex;
|
||
flex-direction: column;
|
||
gap: 10px;
|
||
}
|
||
|
||
.team-batch__tabs {
|
||
display: flex;
|
||
gap: 2px;
|
||
border-bottom: 1px solid var(--el-border-color);
|
||
}
|
||
|
||
.team-batch__tab {
|
||
padding: 6px 14px;
|
||
border: none;
|
||
background: transparent;
|
||
cursor: pointer;
|
||
font-size: 12.5px;
|
||
color: var(--el-text-color-regular);
|
||
border-bottom: 2px solid transparent;
|
||
margin-bottom: -1px;
|
||
}
|
||
|
||
.team-batch__tab.is-active {
|
||
color: var(--el-color-primary);
|
||
border-bottom-color: var(--el-color-primary);
|
||
font-weight: 600;
|
||
}
|
||
|
||
.team-batch__picker {
|
||
display: grid;
|
||
grid-template-columns: 240px 1fr;
|
||
gap: 12px;
|
||
height: min(260px, 40vh);
|
||
min-height: 240px;
|
||
}
|
||
|
||
.team-batch__col {
|
||
display: flex;
|
||
flex-direction: column;
|
||
border: 1px solid var(--el-border-color);
|
||
border-radius: 8px;
|
||
overflow: hidden;
|
||
background: #fff;
|
||
}
|
||
|
||
.team-batch__col-head {
|
||
padding: 6px 10px;
|
||
border-bottom: 1px solid var(--el-border-color);
|
||
background: #fafbfc;
|
||
font-size: 12px;
|
||
color: var(--el-text-color-regular);
|
||
}
|
||
|
||
.team-batch__col-body {
|
||
flex: 1;
|
||
overflow-y: auto;
|
||
}
|
||
|
||
.team-batch__placeholder {
|
||
padding: 24px;
|
||
color: var(--el-text-color-placeholder);
|
||
font-size: 12px;
|
||
text-align: center;
|
||
}
|
||
|
||
.team-batch__params {
|
||
border: 1px solid var(--el-border-color);
|
||
border-radius: 6px;
|
||
background: #fff;
|
||
}
|
||
|
||
.team-batch__tree {
|
||
padding: 4px;
|
||
background: transparent;
|
||
}
|
||
|
||
.team-batch__tree :deep(.el-tree-node__content) {
|
||
height: 32px;
|
||
padding-right: 8px !important;
|
||
border-radius: 4px;
|
||
transition: background-color 0.15s ease;
|
||
}
|
||
|
||
.team-batch__tree :deep(.el-tree-node__content:hover) {
|
||
background: var(--el-fill-color-light);
|
||
}
|
||
|
||
.team-batch__tree :deep(.el-tree-node__expand-icon) {
|
||
padding: 4px;
|
||
color: var(--el-text-color-placeholder);
|
||
font-size: 14px;
|
||
}
|
||
|
||
.team-batch__tree :deep(.el-tree-node__expand-icon.is-leaf) {
|
||
color: transparent;
|
||
}
|
||
|
||
.team-batch__node {
|
||
flex: 1;
|
||
display: flex;
|
||
align-items: center;
|
||
gap: 8px;
|
||
min-width: 0;
|
||
height: 100%;
|
||
font-size: 13px;
|
||
color: var(--el-text-color-primary);
|
||
}
|
||
|
||
.team-batch__node.is-active {
|
||
color: var(--el-color-primary);
|
||
font-weight: 500;
|
||
}
|
||
|
||
.team-batch__node-check {
|
||
position: relative;
|
||
flex-shrink: 0;
|
||
width: 14px;
|
||
height: 14px;
|
||
border: 1px solid var(--el-border-color);
|
||
border-radius: 2px;
|
||
background: var(--el-bg-color);
|
||
cursor: pointer;
|
||
transition:
|
||
border-color 0.15s ease,
|
||
background-color 0.15s ease;
|
||
}
|
||
|
||
.team-batch__node-check:hover {
|
||
border-color: var(--el-color-primary);
|
||
}
|
||
|
||
.team-batch__node-check.is-checked {
|
||
background: var(--el-color-primary);
|
||
border-color: var(--el-color-primary);
|
||
}
|
||
|
||
.team-batch__node-check.is-checked::after {
|
||
content: '';
|
||
position: absolute;
|
||
top: 1px;
|
||
left: 4px;
|
||
width: 3px;
|
||
height: 7px;
|
||
border: solid #fff;
|
||
border-width: 0 1px 1px 0;
|
||
transform: rotate(45deg);
|
||
}
|
||
|
||
.team-batch__node-check.is-partial {
|
||
background: var(--el-color-primary);
|
||
border-color: var(--el-color-primary);
|
||
}
|
||
|
||
.team-batch__node-check.is-partial::after {
|
||
content: '';
|
||
position: absolute;
|
||
top: 50%;
|
||
left: 50%;
|
||
width: 8px;
|
||
height: 2px;
|
||
margin: -1px 0 0 -4px;
|
||
background: #fff;
|
||
border-radius: 1px;
|
||
}
|
||
|
||
.team-batch__node-icon {
|
||
flex-shrink: 0;
|
||
font-size: 15px;
|
||
color: var(--el-text-color-secondary);
|
||
transition: color 0.15s ease;
|
||
}
|
||
|
||
.team-batch__node.is-active .team-batch__node-icon {
|
||
color: var(--el-color-primary);
|
||
}
|
||
|
||
.team-batch__node-label {
|
||
flex: 1;
|
||
min-width: 0;
|
||
overflow: hidden;
|
||
text-overflow: ellipsis;
|
||
white-space: nowrap;
|
||
}
|
||
|
||
.team-batch__node-meta {
|
||
flex-shrink: 0;
|
||
padding-left: 6px;
|
||
font-size: 12px;
|
||
color: var(--el-text-color-placeholder);
|
||
font-variant-numeric: tabular-nums;
|
||
}
|
||
|
||
.team-batch__node.is-active .team-batch__node-meta {
|
||
color: var(--el-color-primary);
|
||
opacity: 0.7;
|
||
}
|
||
|
||
.team-batch__user-row {
|
||
display: flex;
|
||
align-items: center;
|
||
gap: 10px;
|
||
padding: 0 10px;
|
||
height: 36px;
|
||
border-bottom: 1px solid var(--el-border-color-lighter);
|
||
cursor: pointer;
|
||
}
|
||
|
||
.team-batch__user-row:hover {
|
||
background: var(--el-fill-color);
|
||
}
|
||
|
||
.team-batch__user-row.is-disabled {
|
||
opacity: 0.55;
|
||
cursor: not-allowed;
|
||
}
|
||
|
||
.team-batch__user-row.is-disabled:hover {
|
||
background: transparent;
|
||
}
|
||
|
||
.team-batch__user-avatar {
|
||
display: inline-flex;
|
||
align-items: center;
|
||
justify-content: center;
|
||
width: 24px;
|
||
height: 24px;
|
||
border-radius: 50%;
|
||
background: linear-gradient(135deg, #c7d2fe, #93c5fd);
|
||
color: #fff;
|
||
font-size: 12px;
|
||
font-weight: 600;
|
||
flex-shrink: 0;
|
||
}
|
||
|
||
.team-batch__user-main {
|
||
flex: 1;
|
||
min-width: 0;
|
||
overflow: hidden;
|
||
}
|
||
|
||
.team-batch__user-name {
|
||
font-size: 13px;
|
||
overflow: hidden;
|
||
text-overflow: ellipsis;
|
||
white-space: nowrap;
|
||
}
|
||
|
||
.team-batch__user-tag {
|
||
flex-shrink: 0;
|
||
padding: 1px 7px;
|
||
border-radius: 999px;
|
||
font-size: 11px;
|
||
background: var(--el-color-warning-light-7);
|
||
color: var(--el-color-warning-dark-2);
|
||
}
|
||
|
||
.team-batch__empty {
|
||
padding: 40px 0;
|
||
text-align: center;
|
||
color: var(--el-text-color-placeholder);
|
||
font-size: 12px;
|
||
}
|
||
|
||
.team-batch__selected {
|
||
padding: 8px 12px;
|
||
background: #f8fafc;
|
||
border: 1px solid var(--el-border-color);
|
||
border-radius: 6px;
|
||
}
|
||
|
||
.team-batch__selected-head {
|
||
display: flex;
|
||
justify-content: space-between;
|
||
align-items: center;
|
||
margin-bottom: 6px;
|
||
font-size: 11.5px;
|
||
color: var(--el-text-color-regular);
|
||
}
|
||
|
||
.team-batch__selected-head strong {
|
||
color: var(--el-color-primary);
|
||
font-weight: 700;
|
||
font-size: 12.5px;
|
||
}
|
||
|
||
.team-batch__selected-empty {
|
||
display: flex;
|
||
align-items: center;
|
||
min-height: 26px;
|
||
color: var(--el-text-color-placeholder);
|
||
font-size: 11.5px;
|
||
}
|
||
|
||
.team-batch__chips {
|
||
display: flex;
|
||
flex-wrap: wrap;
|
||
align-items: center;
|
||
gap: 6px;
|
||
min-height: 26px;
|
||
}
|
||
|
||
.team-batch__chip {
|
||
display: inline-flex;
|
||
align-items: center;
|
||
gap: 4px;
|
||
padding: 2px 4px 2px 8px;
|
||
background: #fff;
|
||
border: 1px solid var(--el-border-color-darker);
|
||
border-radius: 999px;
|
||
font-size: 11.5px;
|
||
}
|
||
|
||
.team-batch__chip-x {
|
||
width: 16px;
|
||
height: 16px;
|
||
border-radius: 50%;
|
||
background: var(--el-fill-color);
|
||
color: var(--el-text-color-regular);
|
||
border: none;
|
||
cursor: pointer;
|
||
font-size: 11px;
|
||
}
|
||
|
||
.team-batch__chip-x:hover {
|
||
background: var(--el-color-danger);
|
||
color: #fff;
|
||
}
|
||
|
||
.team-batch__chip-more {
|
||
display: inline-flex;
|
||
align-items: center;
|
||
height: 24px;
|
||
padding: 0 10px;
|
||
border-radius: 999px;
|
||
border: 1px dashed var(--el-border-color-darker);
|
||
background: transparent;
|
||
color: var(--el-color-primary);
|
||
font-size: 11.5px;
|
||
cursor: pointer;
|
||
transition:
|
||
border-color 0.15s ease,
|
||
background-color 0.15s ease;
|
||
}
|
||
|
||
.team-batch__chip-more:hover {
|
||
border-color: var(--el-color-primary);
|
||
background: var(--el-color-primary-light-9);
|
||
}
|
||
|
||
.team-batch__overflow-head {
|
||
display: flex;
|
||
justify-content: space-between;
|
||
align-items: center;
|
||
margin-bottom: 8px;
|
||
font-size: 12px;
|
||
color: var(--el-text-color-regular);
|
||
}
|
||
|
||
.team-batch__overflow-head strong {
|
||
color: var(--el-color-primary);
|
||
font-weight: 700;
|
||
}
|
||
|
||
.team-batch__overflow-chips {
|
||
display: flex;
|
||
flex-wrap: wrap;
|
||
gap: 6px;
|
||
max-height: 260px;
|
||
overflow-y: auto;
|
||
}
|
||
|
||
.team-batch__link {
|
||
background: transparent;
|
||
border: none;
|
||
cursor: pointer;
|
||
font-size: 11.5px;
|
||
padding: 0;
|
||
}
|
||
|
||
.team-batch__link--danger {
|
||
color: var(--el-color-danger);
|
||
}
|
||
|
||
.team-batch__link:hover {
|
||
text-decoration: underline;
|
||
}
|
||
|
||
.team-batch__params {
|
||
padding: 10px 12px;
|
||
}
|
||
|
||
.team-batch__params-grid {
|
||
display: grid;
|
||
grid-template-columns: 240px 1fr;
|
||
gap: 12px;
|
||
}
|
||
|
||
.team-batch__field {
|
||
display: flex;
|
||
flex-direction: column;
|
||
gap: 4px;
|
||
}
|
||
|
||
.team-batch__field-label {
|
||
font-size: 11.5px;
|
||
color: var(--el-text-color-primary);
|
||
font-weight: 500;
|
||
}
|
||
|
||
.team-batch__required {
|
||
color: var(--el-color-danger);
|
||
}
|
||
|
||
.team-batch__picker.is-single {
|
||
grid-template-columns: 1fr;
|
||
}
|
||
|
||
.team-batch__chip-role {
|
||
display: inline-flex;
|
||
align-items: center;
|
||
gap: 2px;
|
||
padding: 0 6px;
|
||
height: 18px;
|
||
border-radius: 999px;
|
||
background: var(--el-color-primary-light-9);
|
||
color: var(--el-color-primary);
|
||
font-size: 11px;
|
||
cursor: pointer;
|
||
border: none;
|
||
}
|
||
|
||
.team-batch__chip-role.is-override {
|
||
background: var(--el-color-warning-light-7);
|
||
color: var(--el-color-warning-dark-2);
|
||
}
|
||
|
||
.team-batch__override-cnt {
|
||
color: var(--el-color-warning-dark-2);
|
||
font-weight: 700;
|
||
}
|
||
|
||
.team-batch__search {
|
||
padding: 6px 10px;
|
||
border-bottom: 1px solid var(--el-border-color);
|
||
}
|
||
|
||
.team-batch__col-head--user {
|
||
display: flex;
|
||
justify-content: space-between;
|
||
align-items: center;
|
||
}
|
||
|
||
.team-batch__hide-added {
|
||
font-size: 11.5px;
|
||
}
|
||
|
||
.team-batch__empty-action {
|
||
display: block;
|
||
margin: 6px auto 0;
|
||
}
|
||
|
||
.team-batch__role-option {
|
||
display: flex;
|
||
align-items: center;
|
||
justify-content: space-between;
|
||
gap: 8px;
|
||
width: 100%;
|
||
}
|
||
|
||
.team-batch__role-option-name {
|
||
flex: 1;
|
||
min-width: 0;
|
||
overflow: hidden;
|
||
text-overflow: ellipsis;
|
||
white-space: nowrap;
|
||
}
|
||
|
||
.team-batch__role-option-info {
|
||
flex-shrink: 0;
|
||
font-size: 14px;
|
||
color: var(--el-text-color-placeholder);
|
||
cursor: help;
|
||
}
|
||
|
||
.team-batch__role-option-info:hover {
|
||
color: var(--el-color-primary);
|
||
}
|
||
</style>
|