Reflex/src/views/admin/AdminReportDetail.vue

1007 lines
26 KiB
Vue
Raw Normal View History

2025-05-26 00:57:20 +07:00
<template>
<div class="admin-report-detail">
<div class="header-controls">
<button @click="goBack" class="back-btn">
2025-05-26 01:28:53 +07:00
<i class="bi-arrow-left"></i>
<span class="btn-text">Назад к списку жалоб</span>
2025-05-26 00:57:20 +07:00
</button>
</div>
<div v-if="loading" class="loading-indicator">
2025-05-26 01:28:53 +07:00
<div class="loading-spinner"></div>
<span>Загрузка информации о жалобе...</span>
2025-05-26 00:57:20 +07:00
</div>
<div v-if="error" class="error-message">
2025-05-26 01:28:53 +07:00
<i class="bi-exclamation-triangle"></i>
2025-05-26 00:57:20 +07:00
{{ error }}
</div>
<div v-if="report && !loading" class="report-container">
2025-05-26 01:28:53 +07:00
<div class="page-header">
<h2>Детали жалобы</h2>
<div class="status-display">
<span class="status-badge" :class="getStatusClass(report.status)">
{{ getStatusText(report.status) }}
</span>
</div>
</div>
2025-05-26 00:57:20 +07:00
<div class="report-card">
<div class="report-header">
<div class="report-info">
2025-05-26 01:28:53 +07:00
<div class="report-id-section">
<i class="bi-file-text"></i>
<div class="id-info">
<h3>Жалоба #{{ report._id.substring(0, 8) }}</h3>
<span class="full-id">ID: {{ report._id }}</span>
</div>
</div>
2025-05-26 00:57:20 +07:00
</div>
<div class="report-date">
2025-05-26 01:28:53 +07:00
<i class="bi-calendar3"></i>
<span>Подана {{ formatDate(report.createdAt) }}</span>
2025-05-26 00:57:20 +07:00
</div>
</div>
<div class="report-content">
<div class="info-grid">
2025-05-26 01:28:53 +07:00
<div class="info-section">
<div class="section-header">
<i class="bi-info-circle"></i>
<h4>Информация о жалобе</h4>
2025-05-26 01:48:25 +07:00
</div
2025-05-26 01:28:53 +07:00
<div class="info-item">
<label>Причина жалобы:</label>
<span class="reason-badge" :class="getReasonClass(report.reason)">
{{ getReasonText(report.reason) }}
</span>
</div>
<div class="info-item" v-if="report.description">
<label>Описание:</label>
<div class="description-container">
<p class="description">{{ report.description }}</p>
</div>
2025-05-26 00:57:20 +07:00
</div>
</div>
2025-05-26 01:28:53 +07:00
<div class="info-section">
<div class="section-header">
<i class="bi-people"></i>
<h4>Участники</h4>
</div>
<div class="users-grid">
<div class="user-card reporter">
<div class="user-header">
<i class="bi-person-exclamation"></i>
<span class="user-role">Жалующийся</span>
</div>
<div class="user-details">
<div class="user-name">{{ report.reporter?.name || 'Пользователь удален' }}</div>
<div class="user-email">{{ report.reporter?.email || '' }}</div>
</div>
</div>
2025-05-26 01:45:39 +07:00
<!-- Индикатор стрелки адаптирован для мобильных устройств -->
2025-05-26 01:28:53 +07:00
<div class="arrow-divider">
2025-05-26 01:45:39 +07:00
<i class="bi-arrow-right arrow-icon"></i>
<span class="vs-text">vs</span>
2025-05-26 01:28:53 +07:00
</div>
<div class="user-card reported">
<div class="user-header">
<i class="bi-person-x"></i>
<span class="user-role">Обжалуемый</span>
</div>
<div class="user-details">
<div class="user-name">{{ report.reportedUser?.name || 'Пользователь удален' }}</div>
<div class="user-email">{{ report.reportedUser?.email || '' }}</div>
<button
v-if="report.reportedUser?._id"
@click="viewUser(report.reportedUser._id)"
class="btn view-user-btn"
>
<i class="bi-eye"></i>
<span>Просмотр профиля</span>
</button>
</div>
</div>
</div>
2025-05-26 00:57:20 +07:00
</div>
2025-05-26 01:28:53 +07:00
<div class="info-section" v-if="report.adminNotes || report.resolvedAt">
<div class="section-header">
<i class="bi-shield-check"></i>
<h4>Административная информация</h4>
</div>
<div class="info-item" v-if="report.adminNotes">
<label>Заметки администратора:</label>
<div class="admin-notes-container">
<p class="admin-notes">{{ report.adminNotes }}</p>
</div>
</div>
<div class="info-item" v-if="report.resolvedAt">
<label>Дата рассмотрения:</label>
<div class="resolve-date">
<i class="bi-check-circle"></i>
<span>{{ formatDate(report.resolvedAt) }}</span>
</div>
</div>
2025-05-26 00:57:20 +07:00
</div>
</div>
</div>
<div class="report-actions" v-if="report.status === 'pending'">
2025-05-26 01:28:53 +07:00
<div class="actions-header">
<i class="bi-gear"></i>
<h4>Действия с жалобой</h4>
</div>
2025-05-26 00:57:20 +07:00
<div class="action-buttons">
<button @click="showResolveModal = true" class="btn resolve-btn">
<i class="bi-check-circle"></i>
2025-05-26 01:28:53 +07:00
<span>Рассмотреть</span>
2025-05-26 00:57:20 +07:00
</button>
<button @click="showDismissModal = true" class="btn dismiss-btn">
<i class="bi-x-circle"></i>
2025-05-26 01:28:53 +07:00
<span>Отклонить</span>
2025-05-26 00:57:20 +07:00
</button>
</div>
</div>
</div>
</div>
<!-- Модальное окно для рассмотрения жалобы -->
<div v-if="showResolveModal" class="modal-overlay" @click="showResolveModal = false">
<div class="modal" @click.stop>
2025-05-26 01:28:53 +07:00
<div class="modal-header">
<h3>Рассмотрение жалобы</h3>
<button @click="showResolveModal = false" class="modal-close">
<i class="bi-x"></i>
</button>
</div>
<div class="modal-content">
<p>Вы собираетесь отметить эту жалобу как рассмотренную.</p>
<div class="form-group">
<label>Заметки администратора (необязательно):</label>
<textarea
v-model="adminNotes"
placeholder="Укажите принятые меры или комментарии..."
rows="4"
></textarea>
</div>
2025-05-26 00:57:20 +07:00
</div>
<div class="modal-actions">
<button @click="updateReportStatus('resolved')" class="btn resolve-btn" :disabled="updating">
2025-05-26 01:28:53 +07:00
<i class="bi-check-circle"></i>
<span>{{ updating ? 'Обработка...' : 'Рассмотреть' }}</span>
</button>
<button @click="showResolveModal = false" class="btn cancel-btn">
<i class="bi-x"></i>
<span>Отмена</span>
2025-05-26 00:57:20 +07:00
</button>
</div>
</div>
</div>
<!-- Модальное окно для отклонения жалобы -->
<div v-if="showDismissModal" class="modal-overlay" @click="showDismissModal = false">
<div class="modal" @click.stop>
2025-05-26 01:28:53 +07:00
<div class="modal-header">
<h3>Отклонение жалобы</h3>
<button @click="showDismissModal = false" class="modal-close">
<i class="bi-x"></i>
</button>
</div>
<div class="modal-content">
<p>Вы собираетесь отклонить эту жалобу.</p>
<div class="form-group">
<label>Заметки администратора (необязательно):</label>
<textarea
v-model="adminNotes"
placeholder="Укажите причину отклонения..."
rows="4"
></textarea>
</div>
2025-05-26 00:57:20 +07:00
</div>
<div class="modal-actions">
<button @click="updateReportStatus('dismissed')" class="btn dismiss-btn" :disabled="updating">
2025-05-26 01:28:53 +07:00
<i class="bi-x-circle"></i>
<span>{{ updating ? 'Обработка...' : 'Отклонить' }}</span>
</button>
<button @click="showDismissModal = false" class="btn cancel-btn">
<i class="bi-arrow-left"></i>
<span>Отмена</span>
2025-05-26 00:57:20 +07:00
</button>
</div>
</div>
</div>
</div>
</template>
<script>
import { ref, onMounted } from 'vue';
import { useRouter, useRoute } from 'vue-router';
2025-05-26 01:28:53 +07:00
import axios from 'axios';
2025-05-26 00:57:20 +07:00
export default {
name: 'AdminReportDetail',
props: {
id: {
type: String,
required: true
}
},
setup(props) {
const router = useRouter();
const route = useRoute();
const report = ref(null);
const loading = ref(false);
const error = ref(null);
const updating = ref(false);
const showResolveModal = ref(false);
const showDismissModal = ref(false);
const adminNotes = ref('');
// Загрузка информации о жалобе
const loadReport = async () => {
loading.value = true;
error.value = null;
try {
2025-05-26 01:28:53 +07:00
const token = localStorage.getItem('userToken');
const response = await axios.get(`/api/admin/reports/${props.id}`, {
headers: {
Authorization: `Bearer ${token}`
}
});
2025-05-26 00:57:20 +07:00
report.value = response.data;
} catch (err) {
console.error('Ошибка при загрузке жалобы:', err);
2025-05-26 01:28:53 +07:00
error.value = 'Ошибка при загрузке информации о жалобе.';
2025-05-26 00:57:20 +07:00
} finally {
loading.value = false;
}
};
// Обновление статуса жалобы
const updateReportStatus = async (status) => {
updating.value = true;
try {
2025-05-26 01:28:53 +07:00
const token = localStorage.getItem('userToken');
const response = await axios.put(`/api/admin/reports/${props.id}`, {
2025-05-26 00:57:20 +07:00
status,
adminNotes: adminNotes.value
2025-05-26 01:28:53 +07:00
}, {
headers: {
Authorization: `Bearer ${token}`
}
2025-05-26 00:57:20 +07:00
});
// Обновляем локальные данные
report.value = response.data;
// Закрываем модальные окна
showResolveModal.value = false;
showDismissModal.value = false;
adminNotes.value = '';
alert(`Жалоба успешно ${status === 'resolved' ? 'рассмотрена' : 'отклонена'}`);
} catch (err) {
console.error('Ошибка при обновлении статуса жалобы:', err);
2025-05-26 01:28:53 +07:00
alert('Ошибка при обновлении статуса жалобы');
2025-05-26 00:57:20 +07:00
} finally {
updating.value = false;
}
};
// Возврат к списку жалоб
const goBack = () => {
router.push({ name: 'AdminReports' });
};
// Просмотр профиля пользователя
const viewUser = (userId) => {
router.push({ name: 'AdminUserDetail', params: { id: userId } });
};
// Форматирование даты
const formatDate = (dateString) => {
if (!dateString) return '';
const date = new Date(dateString);
return date.toLocaleString('ru-RU', {
day: '2-digit',
month: '2-digit',
year: 'numeric',
hour: '2-digit',
minute: '2-digit'
});
};
// Получение текста причины жалобы
const getReasonText = (reason) => {
const reasons = {
inappropriate_content: 'Неподходящий контент',
fake_profile: 'Фальшивый профиль',
harassment: 'Домогательства',
spam: 'Спам',
2025-05-26 01:05:41 +07:00
offensive_behavior: 'Оскорбительное поведение',
2025-05-26 00:57:20 +07:00
underage: 'Несовершеннолетний',
other: 'Другое'
};
return reasons[reason] || reason;
};
// Получение CSS класса для причины
const getReasonClass = (reason) => {
const classes = {
inappropriate_content: 'reason-inappropriate',
fake_profile: 'reason-fake',
harassment: 'reason-harassment',
spam: 'reason-spam',
2025-05-26 01:05:41 +07:00
offensive_behavior: 'reason-offensive',
2025-05-26 00:57:20 +07:00
underage: 'reason-underage',
other: 'reason-other'
};
return classes[reason] || 'reason-other';
};
// Получение текста статуса
const getStatusText = (status) => {
const statuses = {
pending: 'На рассмотрении',
resolved: 'Рассмотрена',
dismissed: 'Отклонена'
};
return statuses[status] || status;
};
// Получение CSS класса для статуса
const getStatusClass = (status) => {
const classes = {
pending: 'status-pending',
resolved: 'status-resolved',
dismissed: 'status-dismissed'
};
return classes[status] || 'status-pending';
};
onMounted(() => {
loadReport();
});
return {
report,
loading,
error,
updating,
showResolveModal,
showDismissModal,
adminNotes,
loadReport,
updateReportStatus,
goBack,
viewUser,
formatDate,
getReasonText,
getReasonClass,
getStatusText,
getStatusClass
};
}
}
</script>
<style scoped>
.admin-report-detail {
2025-05-26 01:37:24 +07:00
/* Base padding, applied if no media query overrides a specific side */
padding-top: 1.5rem;
padding-left: 1rem;
padding-right: 1rem;
padding-bottom: 2rem; /* Default bottom padding */
2025-05-26 00:57:20 +07:00
max-width: 100%;
2025-05-26 01:41:32 +07:00
overflow-x: hidden; /* Предотвращение горизонтальной прокрутки */
box-sizing: border-box; /* Гарантия, что padding учитывается в ширине */
2025-05-26 00:57:20 +07:00
}
.header-controls {
margin-bottom: 1.5rem;
}
.back-btn {
background: #6c757d;
color: white;
border: none;
padding: 0.6rem 1rem;
border-radius: 0.375rem;
cursor: pointer;
font-weight: 500;
transition: all 0.2s ease;
text-decoration: none;
display: inline-flex;
align-items: center;
gap: 0.5rem;
}
.back-btn:hover {
background: #545b62;
transform: translateY(-1px);
}
.loading-indicator {
text-align: center;
padding: 2rem;
color: #666;
font-style: italic;
}
2025-05-26 01:28:53 +07:00
.loading-spinner {
border: 4px solid rgba(255, 255, 255, 0.3);
border-top: 4px solid #007bff;
border-radius: 50%;
width: 2rem;
height: 2rem;
animation: spin 0.8s linear infinite;
margin: 0 auto 1rem auto;
}
@keyframes spin {
0% { transform: rotate(0deg); }
100% { transform: rotate(360deg); }
}
2025-05-26 00:57:20 +07:00
.error-message {
background-color: #f8d7da;
color: #721c24;
padding: 0.75rem 1rem;
border-radius: 0.375rem;
border: 1px solid #f5c6cb;
margin-bottom: 1rem;
2025-05-26 01:28:53 +07:00
display: flex;
align-items: center;
gap: 0.5rem;
2025-05-26 00:57:20 +07:00
}
h2 {
margin-top: 0;
margin-bottom: 1.5rem;
color: #333;
font-weight: 600;
}
2025-05-26 01:28:53 +07:00
.page-header {
display: flex;
justify-content: space-between;
align-items: center;
margin-bottom: 1.5rem;
}
.status-display {
display: flex;
align-items: center;
gap: 0.5rem;
}
2025-05-26 01:41:32 +07:00
.report-container {
width: 100%;
overflow-x: hidden; /* Предотвращение горизонтальной прокрутки */
}
2025-05-26 00:57:20 +07:00
.report-card {
background: white;
border-radius: 0.75rem;
box-shadow: 0 2px 10px rgba(0, 0, 0, 0.1);
overflow: hidden;
2025-05-26 01:41:32 +07:00
width: 100%;
box-sizing: border-box; /* Гарантия, что padding учитывается в ширине */
2025-05-26 00:57:20 +07:00
}
.report-header {
padding: 1.5rem;
background: linear-gradient(to right, #f8f9fa, #e9ecef);
border-bottom: 1px solid #dee2e6;
display: flex;
justify-content: space-between;
align-items: center;
}
.report-info {
display: flex;
align-items: center;
gap: 1rem;
}
2025-05-26 01:28:53 +07:00
.report-id-section {
display: flex;
align-items: center;
gap: 0.5rem;
}
2025-05-26 00:57:20 +07:00
.report-info h3 {
margin: 0;
color: #333;
font-size: 1.25rem;
}
2025-05-26 01:28:53 +07:00
.full-id {
color: #6c757d;
font-size: 0.9rem;
2025-05-26 01:41:32 +07:00
word-break: break-all; /* Разрешить перенос длинных строк везде */
max-width: 100%; /* Ограничение ширины */
display: inline-block; /* Для корректного применения max-width */
2025-05-26 01:28:53 +07:00
}
2025-05-26 00:57:20 +07:00
.report-date {
color: #6c757d;
font-size: 0.9rem;
2025-05-26 01:28:53 +07:00
display: flex;
align-items: center;
gap: 0.25rem;
2025-05-26 00:57:20 +07:00
}
.report-content {
padding: 1.5rem;
2025-05-26 01:41:32 +07:00
width: 100%;
box-sizing: border-box; /* Учитывать padding в общей ширине */
2025-05-26 00:57:20 +07:00
}
.info-grid {
display: grid;
gap: 1.5rem;
2025-05-26 01:41:32 +07:00
width: 100%; /* Обеспечение заполнения всей доступной ширины */
2025-05-26 00:57:20 +07:00
}
2025-05-26 01:28:53 +07:00
.info-section {
background: #f8f9fa;
padding: 1.5rem;
border-radius: 0.375rem;
border: 1px solid #e9ecef;
2025-05-26 01:41:32 +07:00
width: 100%;
box-sizing: border-box; /* Учитывать padding в общей ширине */
overflow-wrap: break-word; /* Гарантия переноса длинных слов */
word-wrap: break-word; /* Для поддержки старых браузеров */
2025-05-26 01:28:53 +07:00
}
.section-header {
display: flex;
align-items: center;
gap: 0.5rem;
margin-bottom: 1rem;
}
.section-header h4 {
margin: 0;
color: #495057;
font-weight: 500;
}
2025-05-26 00:57:20 +07:00
.info-item label {
display: block;
font-weight: 600;
color: #495057;
margin-bottom: 0.5rem;
}
.description,
.admin-notes {
background: #f8f9fa;
padding: 1rem;
border-radius: 0.375rem;
border: 1px solid #e9ecef;
margin: 0;
line-height: 1.6;
2025-05-26 01:41:32 +07:00
overflow-wrap: break-word; /* Гарантия переноса длинных слов */
word-wrap: break-word; /* Для поддержки старых браузеров */
white-space: pre-line; /* Сохранение переносов строк в тексте */
max-width: 100%; /* Ограничение ширины */
2025-05-26 00:57:20 +07:00
}
.user-info {
display: flex;
flex-direction: column;
gap: 0.25rem;
}
.email {
color: #6c757d;
font-size: 0.9rem;
}
.view-user-btn {
background: #007bff;
color: white;
border: none;
padding: 0.4rem 0.75rem;
border-radius: 0.375rem;
cursor: pointer;
font-size: 0.85rem;
font-weight: 500;
margin-top: 0.5rem;
align-self: flex-start;
transition: all 0.2s ease;
2025-05-26 01:28:53 +07:00
display: inline-flex;
align-items: center;
gap: 0.5rem;
2025-05-26 00:57:20 +07:00
}
.view-user-btn:hover {
background: #0056b3;
transform: translateY(-1px);
}
.reason-badge,
.status-badge {
padding: 0.25rem 0.75rem;
border-radius: 1rem;
font-size: 0.85rem;
font-weight: 500;
display: inline-block;
}
.reason-inappropriate { background-color: #f8d7da; color: #721c24; }
.reason-fake { background-color: #fff3cd; color: #856404; }
.reason-harassment { background-color: #d1ecf1; color: #0c5460; }
.reason-spam { background-color: #d4edda; color: #155724; }
2025-05-26 01:05:41 +07:00
.reason-offensive { background-color: #f8d7da; color: #721c24; }
2025-05-26 00:57:20 +07:00
.reason-underage { background-color: #f8d7da; color: #721c24; }
.reason-other { background-color: #e2e3e5; color: #383d41; }
.status-pending { background-color: #fff3cd; color: #856404; }
.status-resolved { background-color: #d4edda; color: #155724; }
.status-dismissed { background-color: #e2e3e5; color: #383d41; }
.report-actions {
padding: 1.5rem;
border-top: 1px solid #dee2e6;
background: #f8f9fa;
}
2025-05-26 01:28:53 +07:00
.actions-header {
display: flex;
align-items: center;
gap: 0.5rem;
margin-bottom: 1rem;
2025-05-26 00:57:20 +07:00
}
.action-buttons {
display: flex;
gap: 1rem;
}
.btn {
display: inline-flex;
align-items: center;
gap: 0.5rem;
padding: 0.6rem 1rem;
border: none;
border-radius: 0.375rem;
cursor: pointer;
font-weight: 500;
transition: all 0.2s ease;
text-decoration: none;
}
.resolve-btn {
background: #28a745;
color: white;
}
.resolve-btn:hover:not(:disabled) {
background: #218838;
transform: translateY(-1px);
}
.dismiss-btn {
background: #dc3545;
color: white;
}
.dismiss-btn:hover:not(:disabled) {
background: #c82333;
transform: translateY(-1px);
}
.cancel-btn {
background: #6c757d;
color: white;
}
.cancel-btn:hover {
background: #545b62;
}
.btn:disabled {
opacity: 0.6;
cursor: not-allowed;
transform: none;
}
/* Модальные окна */
.modal-overlay {
position: fixed;
top: 0;
left: 0;
right: 0;
bottom: 0;
background: rgba(0, 0, 0, 0.5);
display: flex;
align-items: center;
justify-content: center;
z-index: 1000;
}
.modal {
background: white;
border-radius: 0.75rem;
padding: 1.5rem;
max-width: 500px;
width: 90%;
max-height: 90vh;
overflow-y: auto;
}
2025-05-26 01:28:53 +07:00
.modal-header {
display: flex;
justify-content: space-between;
align-items: center;
margin-bottom: 1rem;
}
.modal-header h3 {
margin: 0;
color: #333;
}
.modal-close {
background: transparent;
border: none;
cursor: pointer;
font-size: 1.25rem;
color: #6c757d;
}
.modal-close:hover {
2025-05-26 00:57:20 +07:00
color: #333;
}
.form-group {
margin: 1rem 0;
}
.form-group label {
display: block;
font-weight: 600;
color: #495057;
margin-bottom: 0.5rem;
}
.form-group textarea {
width: 100%;
min-height: 100px;
padding: 0.75rem;
border: 1px solid #ced4da;
border-radius: 0.375rem;
font-family: inherit;
resize: vertical;
}
.modal-actions {
display: flex;
gap: 1rem;
justify-content: flex-end;
margin-top: 1.5rem;
}
2025-05-26 01:45:39 +07:00
/* Пользовательские карточки в секции участников */
.users-grid {
display: grid;
grid-template-columns: 1fr auto 1fr;
gap: 1rem;
align-items: center;
width: 100%;
box-sizing: border-box;
2025-05-26 00:57:20 +07:00
}
2025-05-26 01:45:39 +07:00
.user-card {
background: white;
border: 1px solid #e9ecef;
border-radius: 0.5rem;
padding: 1rem;
transition: all 0.2s ease;
width: 100%;
box-sizing: border-box;
height: 100%; /* Одинаковая высота карточек */
display: flex;
flex-direction: column;
}
.user-card:hover {
border-color: #007bff;
box-shadow: 0 2px 8px rgba(0, 123, 255, 0.1);
}
.user-header {
display: flex;
align-items: center;
gap: 0.5rem;
margin-bottom: 0.75rem;
flex-wrap: wrap; /* Разрешить перенос на новую строку */
}
.user-role {
font-weight: 600;
font-size: 0.85rem;
color: #495057;
white-space: nowrap; /* Предотвращение разрыва метки роли */
}
.user-details {
display: flex;
flex-direction: column;
gap: 0.25rem;
overflow-wrap: break-word;
word-wrap: break-word;
flex: 1; /* Растягивание для выравнивания высоты */
}
.user-name {
font-weight: 500;
color: #333;
line-height: 1.3;
margin-bottom: 0.25rem;
}
.user-email {
color: #6c757d;
font-size: 0.85rem;
word-break: break-all;
margin-bottom: 0.5rem;
}
.arrow-divider {
display: flex;
justify-content: center;
align-items: center;
color: #007bff;
font-size: 1.5rem;
position: relative;
}
.arrow-icon {
display: block;
}
.vs-text {
display: none; /* По умолчанию скрыт, показывается на мобильных */
font-weight: 600;
font-size: 0.9rem;
color: #495057;
text-transform: uppercase;
}
/* Специальные стили для очень маленьких экранов */
@media (max-width: 360px) {
.users-grid {
gap: 0.5rem;
2025-05-26 01:28:53 +07:00
}
2025-05-26 01:45:39 +07:00
.user-card {
padding: 0.75rem;
2025-05-26 01:28:53 +07:00
}
2025-05-26 01:45:39 +07:00
.user-header {
margin-bottom: 0.5rem;
2025-05-26 01:28:53 +07:00
}
2025-05-26 01:45:39 +07:00
.user-role {
font-size: 0.8rem;
2025-05-26 01:28:53 +07:00
}
2025-05-26 01:45:39 +07:00
.user-name {
font-size: 0.9rem;
2025-05-26 01:28:53 +07:00
}
2025-05-26 01:45:39 +07:00
.user-email {
font-size: 0.75rem;
2025-05-26 01:28:53 +07:00
}
2025-05-26 01:45:39 +07:00
.view-user-btn {
padding: 0.3rem 0.5rem;
2025-05-26 01:28:53 +07:00
font-size: 0.8rem;
}
2025-05-26 01:45:39 +07:00
.view-user-btn span {
display: none;
2025-05-26 01:28:53 +07:00
}
2025-05-26 01:45:39 +07:00
}
/* Улучшенная мобильная адаптация */
@media (max-width: 480px) {
2025-05-26 01:28:53 +07:00
.users-grid {
grid-template-columns: 1fr;
gap: 1rem;
2025-05-26 01:45:39 +07:00
position: relative;
padding-bottom: 0.5rem;
2025-05-26 01:48:25 +07:00
padding-top: 1rem;
2025-05-26 01:28:53 +07:00
}
.arrow-divider {
2025-05-26 01:45:39 +07:00
position: absolute;
2025-05-26 01:48:25 +07:00
top: calc(50% - 0.75rem); /* Смещение вверх для точного центрирования */
2025-05-26 01:45:39 +07:00
left: 50%;
transform: translate(-50%, -50%);
background-color: white;
border: 1px solid #e9ecef;
border-radius: 50%;
2025-05-26 01:48:25 +07:00
width: 2.2rem; /* Увеличил размер для лучшей видимости */
height: 2.2rem;
2025-05-26 01:45:39 +07:00
box-shadow: 0 2px 5px rgba(0,0,0,0.1);
z-index: 2;
margin: 0;
display: flex;
align-items: center;
justify-content: center;
2025-05-26 01:28:53 +07:00
}
2025-05-26 01:45:39 +07:00
.vs-text {
display: block; /* Показать текст VS на мобильных */
2025-05-26 01:48:25 +07:00
font-size: 0.85rem;
line-height: 1;
margin: 0;
padding: 0;
2025-05-26 01:28:53 +07:00
}
2025-05-26 01:45:39 +07:00
.reporter {
2025-05-26 01:48:25 +07:00
margin-bottom: 2rem; /* Увеличенное пространство для разделителя */
2025-05-26 01:28:53 +07:00
}
2025-05-26 01:45:39 +07:00
.reported {
2025-05-26 01:48:25 +07:00
margin-top: 1rem; /* Увеличенное пространство после разделителя */
2025-05-26 01:28:53 +07:00
}
}
/* Маленькие экраны (481px - 767px) */
@media (min-width: 481px) and (max-width: 767px) {
.users-grid {
grid-template-columns: 1fr;
gap: 1.5rem;
2025-05-26 01:45:39 +07:00
position: relative;
2025-05-26 01:28:53 +07:00
}
.arrow-divider {
2025-05-26 01:45:39 +07:00
position: absolute;
top: 50%;
left: -0.5rem;
transform: translateY(-50%);
z-index: 2;
width: 2rem;
height: 2rem;
background: #f8f9fa;
border-radius: 50%;
border: 1px solid #dee2e6;
box-shadow: 0 2px 5px rgba(0,0,0,0.1);
margin: 0;
}
.arrow-icon {
transform: rotate(90deg);
2025-05-26 01:41:32 +07:00
}
.user-card {
width: 100%;
2025-05-26 01:45:39 +07:00
margin-left: 1rem;
2025-05-26 01:28:53 +07:00
}
2025-05-26 01:45:39 +07:00
.view-user-btn {
align-self: flex-start;
margin-top: 0.75rem;
2025-05-26 01:28:53 +07:00
}
}
2025-05-26 00:57:20 +07:00
</style>