228 lines
9.0 KiB
HTML
228 lines
9.0 KiB
HTML
{% extends "base.html" %}
|
||
|
||
{% block title %}Дашборд - TGLoader Admin{% endblock %}
|
||
|
||
{% block content %}
|
||
<div class="d-flex justify-content-between align-items-center mb-4">
|
||
<h1><i class="bi bi-speedometer2 me-2"></i>Дашборд</h1>
|
||
<div>
|
||
<span class="badge bg-secondary">User ID: {{ current_user.user_id }}</span>
|
||
{% if is_owner %}
|
||
<span class="badge bg-danger ms-2">Owner</span>
|
||
{% else %}
|
||
<span class="badge bg-primary ms-2">Admin</span>
|
||
{% endif %}
|
||
</div>
|
||
</div>
|
||
|
||
<!-- Статистика пользователей -->
|
||
<div class="row mb-4">
|
||
<div class="col-md-12">
|
||
<h5 class="mb-3"><i class="bi bi-people me-2"></i>Пользователи</h5>
|
||
</div>
|
||
<div class="col-md-3 mb-3">
|
||
<div class="card stat-card">
|
||
<div class="card-body">
|
||
<div class="d-flex align-items-center">
|
||
<div class="stat-icon bg-primary text-white me-3">
|
||
<i class="bi bi-people"></i>
|
||
</div>
|
||
<div>
|
||
<h6 class="text-muted mb-0">Всего</h6>
|
||
<h3 class="mb-0">{{ users_stats.total }}</h3>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<div class="col-md-3 mb-3">
|
||
<div class="card stat-card">
|
||
<div class="card-body">
|
||
<div class="d-flex align-items-center">
|
||
<div class="stat-icon bg-success text-white me-3">
|
||
<i class="bi bi-shield-check"></i>
|
||
</div>
|
||
<div>
|
||
<h6 class="text-muted mb-0">Администраторы</h6>
|
||
<h3 class="mb-0">{{ users_stats.admins }}</h3>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<div class="col-md-3 mb-3">
|
||
<div class="card stat-card">
|
||
<div class="card-body">
|
||
<div class="d-flex align-items-center">
|
||
<div class="stat-icon bg-danger text-white me-3">
|
||
<i class="bi bi-ban"></i>
|
||
</div>
|
||
<div>
|
||
<h6 class="text-muted mb-0">Заблокировано</h6>
|
||
<h3 class="mb-0">{{ users_stats.blocked }}</h3>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<div class="col-md-3 mb-3">
|
||
<div class="card stat-card">
|
||
<div class="card-body">
|
||
<div class="d-flex align-items-center">
|
||
<div class="stat-icon bg-info text-white me-3">
|
||
<i class="bi bi-person-plus"></i>
|
||
</div>
|
||
<div>
|
||
<h6 class="text-muted mb-0">Новых (7 дней)</h6>
|
||
<h3 class="mb-0">{{ users_stats.new_week }}</h3>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
<!-- Статистика задач -->
|
||
<div class="row mb-4">
|
||
<div class="col-md-12">
|
||
<h5 class="mb-3"><i class="bi bi-list-task me-2"></i>Задачи</h5>
|
||
</div>
|
||
<div class="col-md-3 mb-3">
|
||
<div class="card stat-card">
|
||
<div class="card-body">
|
||
<div class="d-flex align-items-center">
|
||
<div class="stat-icon bg-primary text-white me-3">
|
||
<i class="bi bi-list-ul"></i>
|
||
</div>
|
||
<div>
|
||
<h6 class="text-muted mb-0">Всего</h6>
|
||
<h3 class="mb-0">{{ tasks_stats.total }}</h3>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<div class="col-md-3 mb-3">
|
||
<div class="card stat-card">
|
||
<div class="card-body">
|
||
<div class="d-flex align-items-center">
|
||
<div class="stat-icon bg-warning text-white me-3">
|
||
<i class="bi bi-hourglass-split"></i>
|
||
</div>
|
||
<div>
|
||
<h6 class="text-muted mb-0">Активные</h6>
|
||
<h3 class="mb-0">{{ tasks_stats.active }}</h3>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<div class="col-md-3 mb-3">
|
||
<div class="card stat-card">
|
||
<div class="card-body">
|
||
<div class="d-flex align-items-center">
|
||
<div class="stat-icon bg-success text-white me-3">
|
||
<i class="bi bi-check-circle"></i>
|
||
</div>
|
||
<div>
|
||
<h6 class="text-muted mb-0">Завершено</h6>
|
||
<h3 class="mb-0">{{ tasks_stats.completed }}</h3>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<div class="col-md-3 mb-3">
|
||
<div class="card stat-card">
|
||
<div class="card-body">
|
||
<div class="d-flex align-items-center">
|
||
<div class="stat-icon bg-danger text-white me-3">
|
||
<i class="bi bi-x-circle"></i>
|
||
</div>
|
||
<div>
|
||
<h6 class="text-muted mb-0">Ошибки</h6>
|
||
<h3 class="mb-0">{{ tasks_stats.failed }}</h3>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
<!-- Последние задачи -->
|
||
<div class="card">
|
||
<div class="card-header">
|
||
<h5 class="mb-0"><i class="bi bi-clock-history me-2"></i>Последние задачи</h5>
|
||
</div>
|
||
<div class="card-body">
|
||
{% if recent_tasks %}
|
||
<div class="table-responsive">
|
||
<table class="table table-hover">
|
||
<thead>
|
||
<tr>
|
||
<th>ID</th>
|
||
<th>Пользователь</th>
|
||
<th>URL</th>
|
||
<th>Статус</th>
|
||
<th>Прогресс</th>
|
||
<th>Создано</th>
|
||
</tr>
|
||
</thead>
|
||
<tbody>
|
||
{% for task in recent_tasks %}
|
||
<tr>
|
||
<td>{{ task.id }}</td>
|
||
<td>
|
||
{% if task.user %}
|
||
{{ task.user.username or task.user.user_id }}
|
||
{% else %}
|
||
{{ task.user_id }}
|
||
{% endif %}
|
||
</td>
|
||
<td>
|
||
<small class="text-muted">
|
||
{{ task.url[:50] + '...' if task.url and task.url|length > 50 else (task.url or '-') }}
|
||
</small>
|
||
</td>
|
||
<td>
|
||
{% if task.status == 'completed' %}
|
||
<span class="badge bg-success">Завершено</span>
|
||
{% elif task.status == 'processing' %}
|
||
<span class="badge bg-warning">В процессе</span>
|
||
{% elif task.status == 'pending' %}
|
||
<span class="badge bg-info">Ожидание</span>
|
||
{% elif task.status == 'failed' %}
|
||
<span class="badge bg-danger">Ошибка</span>
|
||
{% elif task.status == 'cancelled' %}
|
||
<span class="badge bg-secondary">Отменено</span>
|
||
{% else %}
|
||
<span class="badge bg-secondary">{{ task.status }}</span>
|
||
{% endif %}
|
||
</td>
|
||
<td>
|
||
<div class="progress" style="height: 20px; width: 100px;">
|
||
<div class="progress-bar" role="progressbar"
|
||
style="width: {{ task.progress }}%"
|
||
aria-valuenow="{{ task.progress }}"
|
||
aria-valuemin="0"
|
||
aria-valuemax="100">
|
||
{{ task.progress }}%
|
||
</div>
|
||
</div>
|
||
</td>
|
||
<td><small>{{ task.created_at.strftime('%Y-%m-%d %H:%M') if task.created_at else '-' }}</small></td>
|
||
</tr>
|
||
{% endfor %}
|
||
</tbody>
|
||
</table>
|
||
</div>
|
||
{% else %}
|
||
<p class="text-muted text-center">Нет задач</p>
|
||
{% endif %}
|
||
<div class="text-center mt-3">
|
||
<a href="/admin/tasks" class="btn btn-outline-primary">Все задачи</a>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
{% endblock %}
|