6824d7ce7d
🚀 Create and publish a Docker image / Detect changes in backend and frontend (push) Successful in 9s
🚀 Create and publish a Docker image / Build & publish backend image (push) Successful in 2m6s
🚀 Create and publish a Docker image / Build & publish frontend image (push) Successful in 26s
🚀 Create and publish a Docker image / Update stack on Portainer (push) Successful in 6s
140 lines
5.5 KiB
Vue
140 lines
5.5 KiB
Vue
<script setup lang="ts">
|
|
import { computed, onMounted, ref } from 'vue'
|
|
import { usersApi } from '@/api'
|
|
import type { UserDto } from '@/api/types'
|
|
import GlassCard from '@/components/ui/GlassCard.vue'
|
|
import DataTable from '@/components/ui/DataTable.vue'
|
|
import EmptyState from '@/components/ui/EmptyState.vue'
|
|
|
|
const search = ref('')
|
|
const roleFilter = ref('Все роли')
|
|
const instituteFilter = ref('Все институты')
|
|
const users = ref<UserDto[]>([])
|
|
const loading = ref(false)
|
|
const error = ref('')
|
|
|
|
const columns = [
|
|
{ key: 'name', label: 'Имя' },
|
|
{ key: 'email', label: 'Email' },
|
|
{ key: 'role', label: 'Роль', align: 'center' },
|
|
{ key: 'institute', label: 'Институт' },
|
|
{ key: 'activity', label: 'Активность', align: 'center' },
|
|
{ key: 'created', label: 'Дата регистрации' },
|
|
{ key: 'actions', label: 'Действия', align: 'right' },
|
|
]
|
|
|
|
const roleLabels = { Student: 'Студент', Teacher: 'Преподаватель', Admin: 'Администратор' } as const
|
|
const roleApi = { Студент: 'Student', Преподаватель: 'Teacher', Администратор: 'Admin' } as const
|
|
type ApiUserRole = 'Student' | 'Teacher' | 'Admin'
|
|
const roleSetSequence: ApiUserRole[][] = [
|
|
['Student'],
|
|
['Student', 'Teacher'],
|
|
['Student', 'Teacher', 'Admin'],
|
|
]
|
|
|
|
const rows = computed(() =>
|
|
users.value.map(user => ({
|
|
id: user.id,
|
|
name: user.displayName || user.email,
|
|
email: user.email,
|
|
role: user.roles.map(role => roleLabels[role]).join(', '),
|
|
apiRoles: user.roles,
|
|
institute: 'ЮФУ',
|
|
activity: user.isActive ? 'Активен' : 'Заблокирован',
|
|
isActive: user.isActive,
|
|
created: new Date(user.createdAt).toLocaleDateString('ru-RU'),
|
|
})),
|
|
)
|
|
|
|
async function fetchUsers() {
|
|
loading.value = true
|
|
error.value = ''
|
|
try {
|
|
users.value = await usersApi.list({
|
|
Search: search.value || undefined,
|
|
Role: roleFilter.value === 'Все роли' ? undefined : roleApi[roleFilter.value as keyof typeof roleApi],
|
|
PageSize: 100,
|
|
})
|
|
} catch (err) {
|
|
error.value = err instanceof Error ? err.message : 'Не удалось загрузить пользователей.'
|
|
} finally {
|
|
loading.value = false
|
|
}
|
|
}
|
|
|
|
async function toggleActive(row: Record<string, unknown>) {
|
|
const id = Number(row.id)
|
|
const isActive = Boolean(row.isActive)
|
|
await usersApi.setActive(id, !isActive)
|
|
await fetchUsers()
|
|
}
|
|
|
|
async function promoteRole(row: Record<string, unknown>) {
|
|
const id = Number(row.id)
|
|
const apiRoles = (Array.isArray(row.apiRoles) ? row.apiRoles : []) as ApiUserRole[]
|
|
const currentKey = [...new Set(apiRoles)].sort().join(',')
|
|
const currentIndex = roleSetSequence.findIndex(set => set.slice().sort().join(',') === currentKey)
|
|
const next: ApiUserRole[] = (
|
|
currentIndex >= 0
|
|
? roleSetSequence[(currentIndex + 1) % roleSetSequence.length]
|
|
: roleSetSequence[0]
|
|
) ?? ['Student']
|
|
await usersApi.setRole(id, next)
|
|
await fetchUsers()
|
|
}
|
|
|
|
onMounted(fetchUsers)
|
|
</script>
|
|
|
|
<template>
|
|
<div class="admin-users page-content">
|
|
<div class="header">
|
|
<h1 class="page-title">Пользователи</h1>
|
|
<button class="btn-primary">Добавить пользователя</button>
|
|
</div>
|
|
|
|
<GlassCard>
|
|
<div class="filters">
|
|
<input v-model="search" class="glass-input" placeholder="Поиск по имени или email" />
|
|
<select v-model="roleFilter" class="glass-input" @change="fetchUsers">
|
|
<option>Все роли</option>
|
|
<option>Студент</option>
|
|
<option>Преподаватель</option>
|
|
<option>Администратор</option>
|
|
</select>
|
|
<select v-model="instituteFilter" class="glass-input">
|
|
<option>Все институты</option>
|
|
<option>ИКТИБ</option>
|
|
<option>ИФиМКН</option>
|
|
<option>АГиС</option>
|
|
<option>ЮФ</option>
|
|
</select>
|
|
</div>
|
|
|
|
<EmptyState v-if="error" title="Не удалось загрузить пользователей" :subtitle="error" />
|
|
<EmptyState v-else-if="!rows.length && !loading" title="Пользователей не найдено" subtitle="Попробуйте изменить фильтры." />
|
|
<DataTable :columns="columns" :rows="rows">
|
|
<template #role="{ value }">
|
|
<span :class="value === 'Студент' ? 'badge badge-green' : value === 'Преподаватель' ? 'badge badge-blue' : 'badge badge-purple'">{{ value }}</span>
|
|
</template>
|
|
<template #activity="{ value }">
|
|
<span class="badge" :class="value === 'Активен' ? 'badge-green' : 'badge-orange'">{{ value }}</span>
|
|
</template>
|
|
<template #actions="{ row }">
|
|
<div class="actions">
|
|
<button class="btn-ghost" @click="promoteRole(row)">Назначить роль</button>
|
|
<button class="btn-ghost" @click="toggleActive(row)">{{ row.isActive ? 'Заблокировать' : 'Активировать' }}</button>
|
|
</div>
|
|
</template>
|
|
</DataTable>
|
|
</GlassCard>
|
|
</div>
|
|
</template>
|
|
|
|
<style scoped>
|
|
.admin-users { display: flex; flex-direction: column; gap: 16px; }
|
|
.header { display: flex; align-items: center; justify-content: space-between; gap: 12px; flex-wrap: wrap; }
|
|
.filters { display: grid; grid-template-columns: repeat(auto-fit, minmax(180px, 1fr)); gap: 12px; margin-bottom: 12px; }
|
|
.actions { display: flex; gap: 6px; justify-content: flex-end; flex-wrap: wrap; }
|
|
</style>
|