2025-06-30 18:25:26 +00:00
|
|
|
|
import { Component, createSignal, For, onMount, Show } from 'solid-js'
|
2025-06-30 19:19:46 +00:00
|
|
|
|
import {
|
|
|
|
|
CREATE_COMMUNITY_MUTATION,
|
|
|
|
|
DELETE_COMMUNITY_MUTATION,
|
|
|
|
|
UPDATE_COMMUNITY_MUTATION
|
|
|
|
|
} from '../graphql/mutations'
|
2025-06-30 18:25:26 +00:00
|
|
|
|
import { GET_COMMUNITIES_QUERY } from '../graphql/queries'
|
2025-06-30 19:19:46 +00:00
|
|
|
|
import CommunityEditModal from '../modals/CommunityEditModal'
|
2025-06-30 18:25:26 +00:00
|
|
|
|
import styles from '../styles/Table.module.css'
|
|
|
|
|
import Button from '../ui/Button'
|
|
|
|
|
import Modal from '../ui/Modal'
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Интерфейс для сообщества (используем локальный интерфейс для совместимости)
|
|
|
|
|
*/
|
|
|
|
|
interface Community {
|
|
|
|
|
id: number
|
|
|
|
|
slug: string
|
|
|
|
|
name: string
|
|
|
|
|
desc?: string
|
|
|
|
|
pic: string
|
|
|
|
|
created_at: number
|
|
|
|
|
created_by: {
|
|
|
|
|
id: number
|
|
|
|
|
name: string
|
|
|
|
|
email: string
|
|
|
|
|
}
|
|
|
|
|
stat: {
|
|
|
|
|
shouts: number
|
|
|
|
|
followers: number
|
|
|
|
|
authors: number
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
interface CommunitiesRouteProps {
|
|
|
|
|
onError: (error: string) => void
|
|
|
|
|
onSuccess: (message: string) => void
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Компонент для управления сообществами
|
|
|
|
|
*/
|
|
|
|
|
const CommunitiesRoute: Component<CommunitiesRouteProps> = (props) => {
|
|
|
|
|
const [communities, setCommunities] = createSignal<Community[]>([])
|
|
|
|
|
const [loading, setLoading] = createSignal(false)
|
|
|
|
|
const [editModal, setEditModal] = createSignal<{ show: boolean; community: Community | null }>({
|
|
|
|
|
show: false,
|
|
|
|
|
community: null
|
|
|
|
|
})
|
|
|
|
|
const [deleteModal, setDeleteModal] = createSignal<{ show: boolean; community: Community | null }>({
|
|
|
|
|
show: false,
|
|
|
|
|
community: null
|
|
|
|
|
})
|
2025-06-30 19:19:46 +00:00
|
|
|
|
const [createModal, setCreateModal] = createSignal<{ show: boolean }>({
|
|
|
|
|
show: false
|
2025-06-30 18:25:26 +00:00
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Загружает список всех сообществ
|
|
|
|
|
*/
|
|
|
|
|
const loadCommunities = async () => {
|
|
|
|
|
setLoading(true)
|
|
|
|
|
try {
|
|
|
|
|
const response = await fetch('/graphql', {
|
|
|
|
|
method: 'POST',
|
|
|
|
|
headers: {
|
|
|
|
|
'Content-Type': 'application/json'
|
|
|
|
|
},
|
|
|
|
|
body: JSON.stringify({
|
|
|
|
|
query: GET_COMMUNITIES_QUERY
|
|
|
|
|
})
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
const result = await response.json()
|
|
|
|
|
|
|
|
|
|
if (result.errors) {
|
|
|
|
|
throw new Error(result.errors[0].message)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
setCommunities(result.data.get_communities_all || [])
|
|
|
|
|
} catch (error) {
|
|
|
|
|
props.onError(`Ошибка загрузки сообществ: ${(error as Error).message}`)
|
|
|
|
|
} finally {
|
|
|
|
|
setLoading(false)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Форматирует дату
|
|
|
|
|
*/
|
|
|
|
|
const formatDate = (timestamp: number): string => {
|
|
|
|
|
return new Date(timestamp * 1000).toLocaleDateString('ru-RU')
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-30 19:19:46 +00:00
|
|
|
|
/**
|
|
|
|
|
* Открывает модалку создания
|
|
|
|
|
*/
|
|
|
|
|
const openCreateModal = () => {
|
|
|
|
|
setCreateModal({ show: true })
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-30 18:25:26 +00:00
|
|
|
|
/**
|
|
|
|
|
* Открывает модалку редактирования
|
|
|
|
|
*/
|
|
|
|
|
const openEditModal = (community: Community) => {
|
|
|
|
|
setEditModal({ show: true, community })
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
2025-06-30 19:19:46 +00:00
|
|
|
|
* Обрабатывает сохранение сообщества (создание или обновление)
|
2025-06-30 18:25:26 +00:00
|
|
|
|
*/
|
2025-06-30 19:19:46 +00:00
|
|
|
|
const handleSaveCommunity = async (communityData: Partial<Community>) => {
|
2025-06-30 18:25:26 +00:00
|
|
|
|
try {
|
2025-06-30 19:19:46 +00:00
|
|
|
|
const isCreating = !editModal().community && createModal().show
|
|
|
|
|
const mutation = isCreating ? CREATE_COMMUNITY_MUTATION : UPDATE_COMMUNITY_MUTATION
|
|
|
|
|
|
2025-06-30 18:25:26 +00:00
|
|
|
|
const response = await fetch('/graphql', {
|
|
|
|
|
method: 'POST',
|
|
|
|
|
headers: {
|
|
|
|
|
'Content-Type': 'application/json'
|
|
|
|
|
},
|
|
|
|
|
body: JSON.stringify({
|
2025-06-30 19:19:46 +00:00
|
|
|
|
query: mutation,
|
|
|
|
|
variables: { community_input: communityData }
|
2025-06-30 18:25:26 +00:00
|
|
|
|
})
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
const result = await response.json()
|
|
|
|
|
|
|
|
|
|
if (result.errors) {
|
|
|
|
|
throw new Error(result.errors[0].message)
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-30 19:19:46 +00:00
|
|
|
|
const resultData = isCreating ? result.data.create_community : result.data.update_community
|
|
|
|
|
if (resultData.error) {
|
|
|
|
|
throw new Error(resultData.error)
|
2025-06-30 18:25:26 +00:00
|
|
|
|
}
|
|
|
|
|
|
2025-06-30 19:19:46 +00:00
|
|
|
|
props.onSuccess(isCreating ? 'Сообщество успешно создано' : 'Сообщество успешно обновлено')
|
|
|
|
|
setCreateModal({ show: false })
|
2025-06-30 18:25:26 +00:00
|
|
|
|
setEditModal({ show: false, community: null })
|
|
|
|
|
await loadCommunities()
|
|
|
|
|
} catch (error) {
|
2025-06-30 19:19:46 +00:00
|
|
|
|
props.onError(
|
|
|
|
|
`Ошибка ${createModal().show ? 'создания' : 'обновления'} сообщества: ${(error as Error).message}`
|
|
|
|
|
)
|
2025-06-30 18:25:26 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Удаляет сообщество
|
|
|
|
|
*/
|
|
|
|
|
const deleteCommunity = async (slug: string) => {
|
|
|
|
|
try {
|
|
|
|
|
const response = await fetch('/graphql', {
|
|
|
|
|
method: 'POST',
|
|
|
|
|
headers: {
|
|
|
|
|
'Content-Type': 'application/json'
|
|
|
|
|
},
|
|
|
|
|
body: JSON.stringify({
|
|
|
|
|
query: DELETE_COMMUNITY_MUTATION,
|
|
|
|
|
variables: { slug }
|
|
|
|
|
})
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
const result = await response.json()
|
|
|
|
|
|
|
|
|
|
if (result.errors) {
|
|
|
|
|
throw new Error(result.errors[0].message)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (result.data.delete_community.error) {
|
|
|
|
|
throw new Error(result.data.delete_community.error)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
props.onSuccess('Сообщество успешно удалено')
|
|
|
|
|
setDeleteModal({ show: false, community: null })
|
|
|
|
|
await loadCommunities()
|
|
|
|
|
} catch (error) {
|
|
|
|
|
props.onError(`Ошибка удаления сообщества: ${(error as Error).message}`)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Загружаем сообщества при монтировании компонента
|
|
|
|
|
onMount(() => {
|
|
|
|
|
void loadCommunities()
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
return (
|
|
|
|
|
<div class={styles.container}>
|
|
|
|
|
<div class={styles.header}>
|
|
|
|
|
<Button onClick={loadCommunities} disabled={loading()}>
|
|
|
|
|
{loading() ? 'Загрузка...' : 'Обновить'}
|
|
|
|
|
</Button>
|
2025-06-30 19:19:46 +00:00
|
|
|
|
<Button variant="primary" onClick={openCreateModal}>
|
|
|
|
|
Создать сообщество
|
|
|
|
|
</Button>
|
2025-06-30 18:25:26 +00:00
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<Show
|
|
|
|
|
when={!loading()}
|
|
|
|
|
fallback={
|
|
|
|
|
<div class="loading-screen">
|
|
|
|
|
<div class="loading-spinner" />
|
|
|
|
|
<div>Загрузка сообществ...</div>
|
|
|
|
|
</div>
|
|
|
|
|
}
|
|
|
|
|
>
|
|
|
|
|
<table class={styles.table}>
|
|
|
|
|
<thead>
|
|
|
|
|
<tr>
|
|
|
|
|
<th>ID</th>
|
|
|
|
|
<th>Название</th>
|
|
|
|
|
<th>Slug</th>
|
|
|
|
|
<th>Описание</th>
|
|
|
|
|
<th>Создатель</th>
|
|
|
|
|
<th>Публикации</th>
|
|
|
|
|
<th>Подписчики</th>
|
|
|
|
|
<th>Авторы</th>
|
|
|
|
|
<th>Создано</th>
|
|
|
|
|
<th>Действия</th>
|
|
|
|
|
</tr>
|
|
|
|
|
</thead>
|
|
|
|
|
<tbody>
|
|
|
|
|
<For each={communities()}>
|
|
|
|
|
{(community) => (
|
|
|
|
|
<tr
|
|
|
|
|
onClick={() => openEditModal(community)}
|
|
|
|
|
style={{ cursor: 'pointer' }}
|
|
|
|
|
class={styles['clickable-row']}
|
|
|
|
|
>
|
|
|
|
|
<td>{community.id}</td>
|
|
|
|
|
<td>{community.name}</td>
|
|
|
|
|
<td>{community.slug}</td>
|
|
|
|
|
<td>
|
|
|
|
|
<div
|
|
|
|
|
style={{
|
|
|
|
|
'max-width': '200px',
|
|
|
|
|
overflow: 'hidden',
|
|
|
|
|
'text-overflow': 'ellipsis',
|
|
|
|
|
'white-space': 'nowrap'
|
|
|
|
|
}}
|
|
|
|
|
title={community.desc}
|
|
|
|
|
>
|
|
|
|
|
{community.desc || '—'}
|
|
|
|
|
</div>
|
|
|
|
|
</td>
|
|
|
|
|
<td>{community.created_by.name || community.created_by.email}</td>
|
|
|
|
|
<td>{community.stat.shouts}</td>
|
|
|
|
|
<td>{community.stat.followers}</td>
|
|
|
|
|
<td>{community.stat.authors}</td>
|
|
|
|
|
<td>{formatDate(community.created_at)}</td>
|
|
|
|
|
<td onClick={(e) => e.stopPropagation()}>
|
|
|
|
|
<button
|
|
|
|
|
onClick={(e) => {
|
|
|
|
|
e.stopPropagation()
|
|
|
|
|
setDeleteModal({ show: true, community })
|
|
|
|
|
}}
|
|
|
|
|
class={styles['delete-button']}
|
|
|
|
|
title="Удалить сообщество"
|
|
|
|
|
aria-label="Удалить сообщество"
|
|
|
|
|
>
|
|
|
|
|
×
|
|
|
|
|
</button>
|
|
|
|
|
</td>
|
|
|
|
|
</tr>
|
|
|
|
|
)}
|
|
|
|
|
</For>
|
|
|
|
|
</tbody>
|
|
|
|
|
</table>
|
|
|
|
|
</Show>
|
|
|
|
|
|
2025-06-30 19:19:46 +00:00
|
|
|
|
{/* Модальное окно создания */}
|
|
|
|
|
<CommunityEditModal
|
|
|
|
|
isOpen={createModal().show}
|
|
|
|
|
community={null}
|
|
|
|
|
onClose={() => setCreateModal({ show: false })}
|
|
|
|
|
onSave={handleSaveCommunity}
|
|
|
|
|
/>
|
|
|
|
|
|
2025-06-30 18:25:26 +00:00
|
|
|
|
{/* Модальное окно редактирования */}
|
2025-06-30 19:19:46 +00:00
|
|
|
|
<CommunityEditModal
|
2025-06-30 18:25:26 +00:00
|
|
|
|
isOpen={editModal().show}
|
2025-06-30 19:19:46 +00:00
|
|
|
|
community={editModal().community}
|
2025-06-30 18:25:26 +00:00
|
|
|
|
onClose={() => setEditModal({ show: false, community: null })}
|
2025-06-30 19:19:46 +00:00
|
|
|
|
onSave={handleSaveCommunity}
|
|
|
|
|
/>
|
2025-06-30 18:25:26 +00:00
|
|
|
|
|
|
|
|
|
{/* Модальное окно подтверждения удаления */}
|
|
|
|
|
<Modal
|
|
|
|
|
isOpen={deleteModal().show}
|
|
|
|
|
onClose={() => setDeleteModal({ show: false, community: null })}
|
|
|
|
|
title="Подтверждение удаления"
|
|
|
|
|
>
|
|
|
|
|
<div>
|
|
|
|
|
<p>
|
|
|
|
|
Вы уверены, что хотите удалить сообщество "<strong>{deleteModal().community?.name}</strong>"?
|
|
|
|
|
</p>
|
|
|
|
|
<p class={styles['warning-text']}>
|
|
|
|
|
Это действие нельзя отменить. Все публикации и темы сообщества могут быть затронуты.
|
|
|
|
|
</p>
|
|
|
|
|
<div class={styles['modal-actions']}>
|
|
|
|
|
<Button variant="secondary" onClick={() => setDeleteModal({ show: false, community: null })}>
|
|
|
|
|
Отмена
|
|
|
|
|
</Button>
|
|
|
|
|
<Button
|
|
|
|
|
variant="danger"
|
|
|
|
|
onClick={() => deleteModal().community && deleteCommunity(deleteModal().community!.slug)}
|
|
|
|
|
>
|
|
|
|
|
Удалить
|
|
|
|
|
</Button>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</Modal>
|
|
|
|
|
</div>
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export default CommunitiesRoute
|