124 lines
3.1 KiB
Vue
124 lines
3.1 KiB
Vue
<template>
|
||
<!--
|
||
转让群主:选 1 位新群主 → 二次确认 → transferGroupOwner
|
||
- dialog 壳本组件持有;选择 UI 委托 GroupMemberPickerPanel
|
||
- 当前用户从候选里隐藏(不能转给自己)
|
||
- maxSize=1 限定单选
|
||
- 对外接口:ref + open({ groupId, members, hideIds }) + emit reload()
|
||
-->
|
||
<el-dialog
|
||
v-model="visible"
|
||
title="选择新群主"
|
||
width="700px"
|
||
:close-on-click-modal="false"
|
||
class="im-picker-dialog"
|
||
>
|
||
<div class="h-[480px]">
|
||
<GroupMemberPickerPanel
|
||
v-model:selected-ids="selectedIds"
|
||
:members="members"
|
||
:hide-ids="hideIds"
|
||
:max-size="1"
|
||
/>
|
||
</div>
|
||
|
||
<template #footer>
|
||
<el-button @click="visible = false">取消</el-button>
|
||
<el-button
|
||
type="primary"
|
||
:loading="submitting"
|
||
:disabled="selectedIds.length === 0"
|
||
@click="handleOk"
|
||
>
|
||
确定
|
||
</el-button>
|
||
</template>
|
||
</el-dialog>
|
||
</template>
|
||
|
||
<script lang="ts" setup>
|
||
import { computed, ref } from 'vue'
|
||
import { useMessage } from '@/hooks/web/useMessage'
|
||
|
||
import { transferGroupOwner } from '@/api/im/group'
|
||
import GroupMemberPickerPanel from '../picker/GroupMemberPickerPanel.vue'
|
||
import type { GroupMemberLite } from './GroupMember.vue'
|
||
|
||
defineOptions({ name: 'ImGroupOwnerTransferDialog' })
|
||
|
||
const emit = defineEmits<{
|
||
/** 转让成功;父侧通常用来 reload 群数据 */
|
||
reload: []
|
||
}>()
|
||
|
||
const message = useMessage()
|
||
|
||
const visible = ref(false)
|
||
const submitting = ref(false)
|
||
const groupId = ref(0)
|
||
const members = ref<GroupMemberLite[]>([])
|
||
const hideIds = ref<number[]>([])
|
||
const selectedIds = ref<number[]>([])
|
||
|
||
defineExpose({
|
||
/** 打开转让群主弹窗:reset → 灌参 → visible=true */
|
||
open(opts: {
|
||
groupId: number
|
||
members: GroupMemberLite[]
|
||
/** 隐藏 userId:当前用户(不能转给自己) */
|
||
hideIds?: number[]
|
||
}) {
|
||
groupId.value = opts.groupId
|
||
members.value = opts.members
|
||
hideIds.value = opts.hideIds ? [...opts.hideIds] : []
|
||
selectedIds.value = []
|
||
submitting.value = false
|
||
visible.value = true
|
||
}
|
||
})
|
||
|
||
/** 选中的新群主对象(取数组首项) */
|
||
const newOwner = computed<GroupMemberLite | undefined>(() => {
|
||
if (selectedIds.value.length === 0) {
|
||
return undefined
|
||
}
|
||
return members.value.find((member) => member.userId === selectedIds.value[0])
|
||
})
|
||
|
||
/** 二次确认转让:转让后旧群主降为普通成员,无法撤销 */
|
||
async function handleOk() {
|
||
if (!groupId.value || !newOwner.value) {
|
||
return
|
||
}
|
||
try {
|
||
await message.confirm(
|
||
`确定将群主转让给 ${newOwner.value.showName}?转让后你将变为普通成员,无法撤销。`,
|
||
'确认转让群主'
|
||
)
|
||
} catch {
|
||
return
|
||
}
|
||
submitting.value = true
|
||
try {
|
||
await transferGroupOwner({
|
||
groupId: groupId.value,
|
||
newOwnerUserId: newOwner.value.userId
|
||
})
|
||
message.success('群主转让成功')
|
||
emit('reload')
|
||
visible.value = false
|
||
} finally {
|
||
submitting.value = false
|
||
}
|
||
}
|
||
</script>
|
||
|
||
<style scoped lang="scss">
|
||
@use '../picker/picker-dialog' as picker;
|
||
|
||
.im-picker-dialog {
|
||
@include picker.styles;
|
||
}
|
||
</style>
|
||
|