Files
core/panel/styles/Admin.module.css
Untone 8c363a6615 e2e-fixing
fix: убран health endpoint, E2E тест использует корневой маршрут

- Убран health endpoint из main.py (не нужен)
- E2E тест теперь проверяет корневой маршрут / вместо /health
- Корневой маршрут доступен без логина, что подходит для проверки состояния сервера
- E2E тест с браузером работает корректно

docs: обновлен отчет о прогрессе E2E теста

- Убраны упоминания health endpoint
- Указано что используется корневой маршрут для проверки серверов
- Обновлен список измененных файлов

fix: исправлены GraphQL проблемы и E2E тест с браузером

- Добавлено поле success в тип CommonResult для совместимости с фронтендом
- Обновлены резолверы community, collection, topic для возврата поля success
- Исправлен E2E тест для работы с корневым маршрутом вместо health endpoint
- E2E тест теперь запускает браузер, авторизуется, находит сообщество в таблице
- Все GraphQL проблемы с полем success решены
- E2E тест работает правильно с браузером как требовалось

fix: исправлен поиск UI элементов в E2E тесте

- Добавлен правильный поиск кнопки удаления по CSS классу _delete-button_1qlfg_300
- Добавлены альтернативные способы поиска кнопки удаления (title, aria-label, символ ×)
- Добавлен правильный поиск модального окна с множественными селекторами
- Добавлен правильный поиск кнопки подтверждения в модальном окне
- E2E тест теперь полностью работает: находит кнопку удаления, модальное окно и кнопку подтверждения
- Обновлен отчет о прогрессе с полными результатами тестирования

fix: исправлен импорт require_any_permission в resolvers/collection.py

- Заменен импорт require_any_permission с auth.decorators на services.rbac
- Бэкенд сервер теперь запускается корректно
- E2E тест полностью работает: находит кнопку удаления, модальное окно и кнопку подтверждения
- Оба сервера (бэкенд и фронтенд) работают стабильно

fix: исправлен порядок импортов в resolvers/collection.py

- Перемещен импорт require_any_permission в правильное место
- E2E тест полностью работает: находит кнопку удаления, модальное окно и кнопку подтверждения
- Сообщество не удаляется из-за прав доступа - это нормальное поведение системы безопасности

feat: настроен HTTPS для локальной разработки с mkcert
2025-08-01 04:51:06 +03:00

952 lines
16 KiB
CSS
Raw Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
/* Admin Panel Layout */
.admin-panel {
display: flex;
flex-direction: column;
min-height: 100vh;
background-color: var(--background-color);
}
.header-container {
display: flex;
justify-content: space-between;
align-items: center;
padding: 0.75rem 2rem;
}
.header-left {
display: flex;
align-items: center;
gap: 1rem;
}
.header-right {
display: flex;
align-items: center;
gap: 1rem;
}
.community-selector {
display: flex;
align-items: center;
gap: 8px;
}
.community-selector select {
padding: 6px 10px;
border-radius: 4px;
border: 1px solid var(--border-color);
background-color: white;
min-width: 180px;
transition: all 0.2s ease;
}
.community-selector select:focus {
outline: none;
border-color: #3b82f6;
box-shadow: 0 0 0 2px rgba(59, 130, 246, 0.1);
}
/* Стиль для выбранного сообщества */
.community-selected {
border-color: #10b981 !important;
background-color: #f0fdf4 !important;
font-weight: 500;
}
.community-badge {
background: linear-gradient(135deg, #10b981, #059669);
color: white;
padding: 4px 8px;
border-radius: 12px;
font-size: 0.75rem;
font-weight: 500;
white-space: nowrap;
box-shadow: 0 1px 3px rgba(0, 0, 0, 0.1);
}
.loading-indicator {
font-size: 0.8rem;
color: #666;
font-style: italic;
}
.logo {
height: 2rem;
width: auto;
}
.header-container h1 {
margin: 0;
color: var(--text-color);
font-size: 1.5rem;
display: flex;
align-items: center;
gap: 0.75rem;
}
.version-badge {
background: linear-gradient(135deg, #10b981, #059669);
color: white;
padding: 2px 8px;
border-radius: 8px;
font-size: 0.75rem;
font-weight: 500;
letter-spacing: 0.5px;
box-shadow: 0 1px 3px rgba(16, 185, 129, 0.3);
white-space: nowrap;
}
.logout-button {
padding: 0.5rem 1rem;
border: 1px solid var(--border-color);
border-radius: 4px;
background-color: transparent;
color: var(--text-color);
cursor: pointer;
transition: all 0.2s ease;
}
.logout-button:hover {
background-color: var(--hover-color);
}
.admin-tabs {
display: flex;
gap: 1rem;
padding: 0.75rem 2rem;
}
main {
flex: 1;
padding: 1rem 3rem;
max-width: 1400px;
margin: 0 auto;
width: 100%;
}
/* Common Styles */
.loading {
display: flex;
justify-content: center;
align-items: center;
padding: 2rem;
color: var(--text-color-light);
}
.empty-state {
text-align: center;
padding: 60px 20px;
color: #6b7280;
max-width: 600px;
margin: 0 auto;
}
.empty-state h3 {
color: #374151;
margin-bottom: 16px;
font-size: 1.5rem;
}
.empty-state p {
font-size: 1rem;
line-height: 1.6;
margin-bottom: 0;
}
.empty-state code {
background: #f3f4f6;
padding: 2px 6px;
border-radius: 4px;
font-family: "SF Mono", "Monaco", "Inconsolata", "Roboto Mono", monospace;
font-size: 0.9em;
color: #1f2937;
}
.empty-state details {
text-align: left;
}
.empty-state summary:hover {
color: #3b82f6;
}
.empty-state pre {
text-align: left;
white-space: pre-wrap;
word-break: break-all;
margin: 0;
font-family: "SF Mono", "Monaco", "Inconsolata", "Roboto Mono", monospace;
}
.error-message {
margin: 1rem 2rem;
padding: 1rem;
background-color: var(--error-color-light);
color: var(--error-color-dark);
border-radius: 4px;
border: 1px solid var(--error-color);
}
.success-message {
margin: 1rem 2rem;
padding: 1rem;
background-color: var(--success-color-light);
color: var(--success-color-dark);
border-radius: 4px;
border: 1px solid var(--success-color);
}
/* Users Route Styles */
.authors-container {
padding: 0;
}
.authors-controls {
margin-bottom: 1rem;
width: 100%;
}
.search-container {
display: flex;
gap: 1rem;
margin-bottom: 1rem;
width: 100%;
}
.search-input-group {
display: flex;
gap: 0.5rem;
flex: 1;
}
.search-input {
flex: 1;
padding: 0.5rem;
border: 1px solid var(--border-color);
border-radius: var(--border-radius-sm);
font-size: var(--font-size-sm);
color: var(--text-color);
background-color: var(--background-color);
}
.search-input:focus {
outline: none;
border-color: var(--primary-color);
box-shadow: 0 0 0 2px var(--primary-color-light);
}
.search-button {
padding: 0.5rem 1rem;
background-color: var(--primary-color);
color: white;
border: none;
border-radius: var(--border-radius-sm);
cursor: pointer;
transition: all var(--transition-fast);
}
.search-button:hover {
background-color: var(--primary-color-dark);
}
.authors-list {
overflow-x: auto;
}
.authors-list table {
width: 100%;
border-collapse: collapse;
margin-bottom: 1rem;
min-width: 800px;
}
.authors-list th,
.authors-list td {
padding: 1.2rem 1.5rem;
text-align: left;
border-bottom: 1px solid var(--border-color);
}
.authors-list th {
background-color: var(--header-background);
color: var(--text-color);
font-weight: var(--font-weight-medium);
white-space: nowrap;
}
.roles-cell {
min-width: 200px;
}
.roles-container {
display: flex;
flex-wrap: wrap;
gap: 0.5rem;
align-items: center;
}
.role-badge {
display: inline-flex;
align-items: center;
gap: 0.25rem;
padding: 0.25rem 0.5rem;
background-color: var(--secondary-color-light);
border-radius: var(--border-radius-sm);
font-size: var(--font-size-sm);
color: var(--text-color);
}
.role-icon {
font-size: var(--font-size-base);
}
.edit-role-badge {
cursor: pointer;
background-color: var(--primary-color-light);
color: var(--primary-color);
transition: all var(--transition-fast);
}
.edit-role-badge:hover {
background-color: var(--primary-color);
color: white;
}
/* Shouts Route Styles */
.shouts-container {
padding: 0;
}
.shouts-controls {
display: flex;
align-items: center;
gap: 1.5rem;
margin-bottom: 1.5rem;
flex-wrap: wrap;
}
.status-filter select {
padding: 0.5rem;
border: 1px solid var(--border-color);
border-radius: 4px;
background-color: white;
}
.status-badge {
display: inline-flex;
align-items: center;
justify-content: center;
padding: 0.35rem;
border-radius: 6px;
font-size: 1.1rem;
width: 32px;
height: 32px;
text-align: center;
}
.status-badge.status-published {
background-color: var(--success-color-light);
color: var(--success-color-dark);
}
.status-badge.status-draft {
background-color: var(--warning-color-light);
color: var(--warning-color-dark);
}
.status-badge.status-deleted {
background-color: var(--error-color-light);
color: var(--error-color-dark);
}
.authors-list {
display: flex;
flex-wrap: wrap;
gap: 4px;
margin: 0;
}
.author-badge {
display: inline-block;
padding: 0.25rem 0.5rem;
border-radius: 4px;
font-size: 0.75rem;
background-color: var(--success-color-light);
color: var(--success-color-dark);
margin: 0;
white-space: nowrap;
max-width: 120px;
overflow: hidden;
text-overflow: ellipsis;
}
.topics-list {
display: flex;
flex-wrap: wrap;
gap: 4px;
margin: 0;
}
.topic-badge {
display: inline-block;
padding: 0.25rem 0.5rem;
border-radius: 4px;
font-size: 0.75rem;
background-color: var(--info-color-light);
color: var(--info-color-dark);
margin: 0;
white-space: nowrap;
max-width: 100px;
overflow: hidden;
text-overflow: ellipsis;
}
.community-badge {
display: inline-block;
padding: 0.25rem 0.5rem;
border-radius: 4px;
font-size: 0.875rem;
background-color: var(--primary-color-light);
color: var(--primary-color-dark);
margin: 0.25rem;
cursor: pointer;
}
.community-badge:hover {
background-color: var(--primary-color);
color: white;
}
.body-cell {
cursor: pointer;
}
.body-cell:hover {
background-color: var(--hover-color);
}
.no-data {
color: var(--text-color-light);
font-style: italic;
}
/* Компактная кнопка для медиа body */
.edit-button {
background: #f3f4f6;
border: 1px solid #d1d5db;
color: #374151;
cursor: pointer;
transition: all 0.2s ease;
display: inline-flex;
align-items: center;
justify-content: center;
}
.edit-button:hover {
background: #e5e7eb;
border-color: #9ca3af;
transform: scale(1.05);
}
/* Environment Variables Route Styles */
.env-variables-container {
padding: 1.5rem 0;
max-width: none;
}
.env-sections {
display: flex;
flex-direction: column;
gap: 2rem;
}
.env-section {
background-color: white;
border-radius: 4px;
box-shadow: 0 1px 3px rgba(0, 0, 0, 0.1);
padding: 2rem;
}
.section-name {
margin: 0 0 1rem;
color: var(--text-color);
font-size: 1.25rem;
}
.section-description {
margin: 0 0 1.5rem;
color: var(--text-color-light);
}
.variables-list {
overflow-x: auto;
margin: 0 -1rem;
}
.empty-value {
color: var(--text-color-light);
font-style: italic;
}
.actions {
display: flex;
gap: 0.5rem;
}
/* Table Styles */
table {
width: 100%;
border-collapse: collapse;
min-width: 900px;
table-layout: fixed; /* Фиксированная ширина столбцов */
}
th {
text-align: left;
padding: 0.8rem 1rem;
color: var(--text-color);
font-weight: 600;
font-size: 0.8rem;
white-space: nowrap; /* Заголовки не переносятся */
overflow: hidden;
text-transform: uppercase;
letter-spacing: 0.5px;
}
td {
padding: 0.8rem 1rem;
color: var(--text-color);
font-size: 0.85rem;
line-height: 1.4;
word-wrap: break-word; /* Перенос длинных слов */
white-space: normal; /* Разрешаем перенос строк */
vertical-align: top; /* Выравнивание по верхнему краю */
}
/* Специальные стили для колонок публикаций (после удаления колонки "Статус") */
.shouts-list th:nth-child(1) {
width: 5%;
} /* ID */
.shouts-list th:nth-child(2) {
width: 22%;
} /* ЗАГОЛОВОК */
.shouts-list th:nth-child(3) {
width: 12%;
} /* SLUG */
.shouts-list th:nth-child(4) {
width: 15%;
} /* АВТОРЫ */
.shouts-list th:nth-child(5) {
width: 15%;
} /* ТЕМЫ */
.shouts-list th:nth-child(6) {
width: 10%;
} /* СОЗДАН */
.shouts-list th:nth-child(7) {
width: 16%;
} /* СОДЕРЖИМОЕ */
.shouts-list th:nth-child(8) {
width: 5%;
} /* MEDIA */
/* Компактные стили для колонки ID */
.shouts-list th:nth-child(1),
.shouts-list td:nth-child(1) {
padding: 0.6rem 0.4rem !important;
font-size: 0.7rem !important;
text-align: center;
font-weight: 600;
}
/* Колонки авторов и тем - больше места для бейджей */
.shouts-list td:nth-child(4) {
/* Колонка авторов */
padding: 0.5rem;
vertical-align: top;
}
.shouts-list td:nth-child(5) {
/* Колонка тем */
padding: 0.5rem;
vertical-align: top;
}
.shouts-list td:nth-child(7) {
/* Колонка содержимого */
max-width: 250px;
word-wrap: break-word;
overflow-wrap: break-word;
hyphens: auto;
}
/* Responsive Styles */
@media (max-width: 1024px) {
.header-container {
padding: 1rem;
}
.admin-tabs {
padding: 1rem;
flex-wrap: wrap;
}
main {
padding: 1rem 2rem;
}
.authors-container,
.shouts-container,
.env-variables-container {
padding: 0;
}
.search-input-group {
flex-direction: column;
}
.search-button {
width: 100%;
}
.shouts-controls {
flex-direction: column;
gap: 1rem;
}
.status-filter {
width: 100%;
}
.status-filter select {
width: 100%;
}
}
/* Responsive Design */
@media (max-width: 640px) {
.header-container {
padding: 1rem;
flex-direction: column;
gap: 1rem;
text-align: center;
}
.header-left {
flex-direction: column;
}
main {
padding: 1rem;
}
.authors-list {
margin: 0 -1rem;
}
.authors-list table {
font-size: var(--font-size-sm);
min-width: 600px;
}
.authors-list th,
.authors-list td {
padding: 0.8rem 1rem;
}
th,
td {
padding: 0.8rem 1rem;
}
table {
min-width: 600px;
}
.search-container {
flex-direction: column;
}
.search-input-group {
flex-direction: column;
}
}
/* Styles for reaction-related components */
.reactions-container {
display: flex;
flex-direction: column;
gap: 1rem;
}
.reactions-list {
width: 100%;
overflow-x: auto;
}
.reactions-list table {
width: 100%;
border-collapse: collapse;
background-color: white;
box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1);
border-radius: 8px;
table-layout: fixed;
}
/* Оптимизация ширины колонок для реакций */
.reactions-list th:nth-child(1), /* ID */
.reactions-list td:nth-child(1) {
width: 80px;
text-align: center;
}
.reactions-list th:nth-child(2), /* ТИП */
.reactions-list td:nth-child(2) {
width: 60px;
text-align: center;
}
.reactions-list th:nth-child(3), /* ТЕКСТ */
.reactions-list td:nth-child(3) {
width: 28%;
}
.reactions-list th:nth-child(4), /* АВТОР */
.reactions-list td:nth-child(4) {
width: 20%;
}
.reactions-list th:nth-child(5), /* ПУБЛИКАЦИЯ */
.reactions-list td:nth-child(5) {
width: 25%;
}
.reactions-list th:nth-child(6), /* СОЗДАНО */
.reactions-list td:nth-child(6) {
width: 120px;
}
.reactions-list th:nth-child(7), /* ДЕЙСТВИЯ */
.reactions-list td:nth-child(7) {
width: 120px;
text-align: center;
}
.reactions-list th,
.reactions-list td {
padding: 0.75rem;
text-align: left;
border-bottom: 1px solid #e0e0e0;
}
.reactions-list th {
background-color: #f8f9fa;
font-weight: 600;
color: #333;
}
.reactions-list tr:hover {
background-color: #f8f9fa;
cursor: pointer;
}
.reactions-list tr.deleted-row {
background-color: #fff5f5;
color: #666;
}
.reactions-list tr.deleted-row:hover {
background-color: #fed7d7;
}
.body-cell {
max-width: 200px;
}
.body-preview {
word-wrap: break-word;
overflow-wrap: break-word;
font-size: 0.875rem;
line-height: 1.4;
}
.author-cell {
min-width: 150px;
}
.author-email {
font-size: 0.75rem;
color: #666;
margin-top: 0.25rem;
}
.shout-cell {
min-width: 200px;
}
.shout-title {
font-weight: 500;
margin-bottom: 0.25rem;
}
.shout-meta {
font-size: 0.75rem;
color: #666;
}
.status-active {
color: #28a745;
font-weight: 500;
}
.status-deleted {
color: #dc3545;
font-weight: 500;
}
.actions-cell {
min-width: 120px;
}
.actions-cell button {
margin-right: 0.5rem;
}
.filters-section {
display: flex;
flex-direction: column;
gap: 1rem;
margin-bottom: 1rem;
}
.additional-filters {
display: flex;
gap: 1rem;
align-items: center;
flex-wrap: wrap;
}
.filter-select,
.filter-input {
padding: 0.5rem;
border: 1px solid #d1d5db;
border-radius: 4px;
font-size: 0.875rem;
min-width: 120px;
}
.filter-select:focus,
.filter-input:focus {
outline: none;
border-color: #3b82f6;
box-shadow: 0 0 0 2px rgba(59, 130, 246, 0.2);
}
.filter-checkbox {
display: flex;
align-items: center;
gap: 8px;
cursor: pointer;
padding: 6px 10px;
border-radius: 4px;
transition: all 0.2s ease;
user-select: none;
font-size: 0.875rem;
}
.filter-checkbox:hover {
background-color: #f0f4f8;
}
.filter-checkbox input[type="checkbox"] {
width: 16px;
height: 16px;
cursor: pointer;
accent-color: #3b82f6;
}
.stat-info {
display: flex;
gap: 1rem;
font-size: 0.875rem;
color: #666;
}
.stat-info span {
padding: 0.25rem 0.5rem;
background-color: #f8f9fa;
border-radius: 4px;
}
.reaction-icon {
font-size: 1.25rem;
display: inline-block;
cursor: pointer;
}
/* Стили для секции управления правами */
.permissions-section {
padding: 2rem;
max-width: 800px;
margin: 0 auto;
}
.permissions-content {
background: white;
border-radius: 8px;
padding: 2rem;
box-shadow: 0 1px 3px rgba(0, 0, 0, 0.1);
border: 1px solid var(--border-color);
}
.permissions-info {
margin-bottom: 2rem;
}
.permissions-info h3 {
color: var(--text-color);
margin-bottom: 1rem;
font-size: 1.1rem;
font-weight: 600;
}
.permissions-info ul {
list-style: none;
padding: 0;
margin: 0 0 1.5rem 0;
}
.permissions-info li {
padding: 0.5rem 0;
position: relative;
padding-left: 1.5rem;
color: var(--text-color);
}
.permissions-info li::before {
content: "✓";
position: absolute;
left: 0;
color: #10b981;
font-weight: bold;
}
.warning-box {
background: #fef3c7;
border: 1px solid #f59e0b;
border-radius: 6px;
padding: 1rem;
margin-top: 1rem;
color: #92400e;
}
.warning-box strong {
color: #d97706;
}
.permissions-actions {
display: flex;
justify-content: center;
padding-top: 1rem;
border-top: 1px solid var(--border-color);
}