183 lines
5.4 KiB
TypeScript
183 lines
5.4 KiB
TypeScript
import { Component, createEffect, createSignal, For, Show } from 'solid-js'
|
||
import { useData } from '../context/data'
|
||
import formStyles from '../styles/Form.module.css'
|
||
import styles from '../styles/Modal.module.css'
|
||
import Button from '../ui/Button'
|
||
import Modal from '../ui/Modal'
|
||
|
||
interface Author {
|
||
id: number
|
||
name: string
|
||
email: string
|
||
slug: string
|
||
}
|
||
|
||
interface Community {
|
||
id: number
|
||
name: string
|
||
slug: string
|
||
}
|
||
|
||
interface Role {
|
||
id: string
|
||
name: string
|
||
description?: string
|
||
}
|
||
|
||
interface CommunityRolesModalProps {
|
||
isOpen: boolean
|
||
author: Author | null
|
||
community: Community | null
|
||
onClose: () => void
|
||
onSave: (authorId: number, communityId: number, roles: string[]) => Promise<void>
|
||
}
|
||
|
||
const CommunityRolesModal: Component<CommunityRolesModalProps> = (props) => {
|
||
const { queryGraphQL } = useData()
|
||
const [roles, setRoles] = createSignal<Role[]>([])
|
||
const [userRoles, setUserRoles] = createSignal<string[]>([])
|
||
const [loading, setLoading] = createSignal(false)
|
||
|
||
// Загружаем доступные роли при открытии модала
|
||
createEffect(() => {
|
||
if (props.isOpen && props.community) {
|
||
void loadRolesData()
|
||
}
|
||
})
|
||
|
||
const loadRolesData = async () => {
|
||
setLoading(true)
|
||
try {
|
||
// Получаем доступные роли
|
||
const rolesData = await queryGraphQL(
|
||
`
|
||
query GetRoles($community: Int) {
|
||
adminGetRoles(community: $community) {
|
||
id
|
||
name
|
||
description
|
||
}
|
||
}
|
||
`,
|
||
{ community: props.community?.id }
|
||
)
|
||
|
||
if (rolesData?.adminGetRoles) {
|
||
setRoles(rolesData.adminGetRoles)
|
||
}
|
||
|
||
// Получаем текущие роли пользователя
|
||
if (props.author) {
|
||
const membersData = await queryGraphQL(
|
||
`
|
||
query GetCommunityMembers($community_id: Int!) {
|
||
adminGetCommunityMembers(community_id: $community_id, limit: 1000) {
|
||
members {
|
||
id
|
||
roles
|
||
}
|
||
}
|
||
}
|
||
`,
|
||
{ community_id: props.community?.id }
|
||
)
|
||
|
||
const members = membersData?.adminGetCommunityMembers?.members || []
|
||
const currentUser = members.find((m: { id: number }) => m.id === props.author?.id)
|
||
setUserRoles(currentUser?.roles || [])
|
||
}
|
||
} catch (error) {
|
||
console.error('Ошибка загрузки ролей:', error)
|
||
} finally {
|
||
setLoading(false)
|
||
}
|
||
}
|
||
|
||
const handleRoleToggle = (roleId: string) => {
|
||
const currentRoles = userRoles()
|
||
if (currentRoles.includes(roleId)) {
|
||
setUserRoles(currentRoles.filter((r) => r !== roleId))
|
||
} else {
|
||
setUserRoles([...currentRoles, roleId])
|
||
}
|
||
}
|
||
|
||
const handleSave = async () => {
|
||
if (!props.author || !props.community) return
|
||
|
||
setLoading(true)
|
||
try {
|
||
await props.onSave(props.author.id, props.community.id, userRoles())
|
||
props.onClose()
|
||
} catch (error) {
|
||
console.error('Ошибка сохранения ролей:', error)
|
||
} finally {
|
||
setLoading(false)
|
||
}
|
||
}
|
||
|
||
return (
|
||
<Modal
|
||
isOpen={props.isOpen}
|
||
onClose={props.onClose}
|
||
title={`Роли пользователя: ${props.author?.name || ''}`}
|
||
>
|
||
<div class={styles.content}>
|
||
<Show when={props.community && props.author}>
|
||
<div class={formStyles.field}>
|
||
<label class={formStyles.label}>
|
||
Сообщество: <strong>{props.community?.name}</strong>
|
||
</label>
|
||
</div>
|
||
|
||
<div class={formStyles.field}>
|
||
<label class={formStyles.label}>
|
||
Пользователь: <strong>{props.author?.name}</strong> ({props.author?.email})
|
||
</label>
|
||
</div>
|
||
|
||
<div class={formStyles.field}>
|
||
<label class={formStyles.label}>Роли:</label>
|
||
<Show when={!loading()} fallback={<div>Загрузка ролей...</div>}>
|
||
<div class={formStyles.checkboxGroup}>
|
||
<For each={roles()}>
|
||
{(role) => (
|
||
<div class={formStyles.checkboxItem}>
|
||
<input
|
||
type="checkbox"
|
||
id={`role-${role.id}`}
|
||
checked={userRoles().includes(role.id)}
|
||
onChange={() => handleRoleToggle(role.id)}
|
||
class={formStyles.checkbox}
|
||
/>
|
||
<label for={`role-${role.id}`} class={formStyles.checkboxLabel}>
|
||
<div>
|
||
<strong>{role.name}</strong>
|
||
<Show when={role.description}>
|
||
<div class={formStyles.description}>{role.description}</div>
|
||
</Show>
|
||
</div>
|
||
</label>
|
||
</div>
|
||
)}
|
||
</For>
|
||
</div>
|
||
</Show>
|
||
</div>
|
||
</Show>
|
||
|
||
<div class={styles.actions}>
|
||
<Button variant="secondary" onClick={props.onClose}>
|
||
Отмена
|
||
</Button>
|
||
<Button variant="primary" onClick={handleSave} disabled={loading()}>
|
||
{loading() ? 'Сохранение...' : 'Сохранить'}
|
||
</Button>
|
||
</div>
|
||
</div>
|
||
</Modal>
|
||
)
|
||
}
|
||
|
||
export default CommunityRolesModal
|