ваше сообщение коммита

This commit is contained in:
2025-07-27 03:30:13 +03:00
parent 057fe6254c
commit 1835632be9
141 changed files with 32514 additions and 6661 deletions

View File

@@ -0,0 +1,705 @@
<!--
Copyright (c) 2024-2025 Тарабанов Александр Викторович
All rights reserved.
This software is proprietary and confidential.
Unauthorized copying, modification, or distribution is prohibited.
For licensing inquiries: info@hb3-accelerator.com
Website: https://hb3-accelerator.com
GitHub: https://github.com/HB3-ACCELERATOR
-->
<template>
<BaseLayout
:is-authenticated="isAuthenticated"
:identities="identities"
:token-balances="tokenBalances"
:is-loading-tokens="isLoadingTokens"
@auth-action-completed="$emit('auth-action-completed')"
>
<div class="analytics-container">
<!-- Заголовок -->
<div class="page-header">
<div class="header-content">
<h1>Аналитика</h1>
<p>Графики, статистика и отчеты DLE</p>
</div>
<button class="close-btn" @click="router.push('/management')">×</button>
</div>
<!-- Ключевые метрики -->
<div class="metrics-section">
<h2>Ключевые метрики</h2>
<div class="metrics-grid">
<div class="metric-card">
<h3>Общая стоимость</h3>
<p class="metric-value">${{ totalValue.toLocaleString() }}</p>
<p class="metric-change positive">+{{ valueChange }}% (30д)</p>
</div>
<div class="metric-card">
<h3>Активные участники</h3>
<p class="metric-value">{{ activeParticipants }}</p>
<p class="metric-change positive">+{{ participantsChange }} (30д)</p>
</div>
<div class="metric-card">
<h3>Предложения</h3>
<p class="metric-value">{{ totalProposals }}</p>
<p class="metric-change positive">+{{ proposalsChange }} (30д)</p>
</div>
<div class="metric-card">
<h3>Доходность</h3>
<p class="metric-value">{{ yieldRate }}%</p>
<p class="metric-change positive">+{{ yieldChange }}% (30д)</p>
</div>
</div>
</div>
<!-- Графики -->
<div class="charts-section">
<h2>Графики</h2>
<div class="charts-grid">
<!-- График стоимости токенов -->
<div class="chart-card">
<h3>Стоимость токенов</h3>
<div class="chart-placeholder">
<div class="chart-line">
<div class="chart-point" style="left: 10%; top: 80%"></div>
<div class="chart-point" style="left: 25%; top: 60%"></div>
<div class="chart-point" style="left: 40%; top: 40%"></div>
<div class="chart-point" style="left: 55%; top: 30%"></div>
<div class="chart-point" style="left: 70%; top: 20%"></div>
<div class="chart-point" style="left: 85%; top: 10%"></div>
<div class="chart-point" style="left: 100%; top: 5%"></div>
</div>
</div>
<div class="chart-legend">
<span class="legend-item">
<span class="legend-color" style="background: var(--color-primary)"></span>
Стоимость токена
</span>
</div>
</div>
<!-- График активности -->
<div class="chart-card">
<h3>Активность участников</h3>
<div class="activity-chart">
<div class="activity-bar" style="height: 60%">
<span class="bar-label">Пн</span>
</div>
<div class="activity-bar" style="height: 80%">
<span class="bar-label">Вт</span>
</div>
<div class="activity-bar" style="height: 45%">
<span class="bar-label">Ср</span>
</div>
<div class="activity-bar" style="height: 90%">
<span class="bar-label">Чт</span>
</div>
<div class="activity-bar" style="height: 75%">
<span class="bar-label">Пт</span>
</div>
<div class="activity-bar" style="height: 55%">
<span class="bar-label">Сб</span>
</div>
<div class="activity-bar" style="height: 40%">
<span class="bar-label">Вс</span>
</div>
</div>
<div class="chart-legend">
<span class="legend-item">
<span class="legend-color" style="background: var(--color-secondary)"></span>
Количество операций
</span>
</div>
</div>
</div>
</div>
<!-- Статистика -->
<div class="statistics-section">
<h2>Статистика</h2>
<div class="stats-grid">
<div class="stats-card">
<h3>Распределение токенов</h3>
<div class="distribution-chart">
<div class="pie-segment" style="--percentage: 40; --color: #007bff">
<span class="segment-label">Крупные держатели</span>
<span class="segment-value">40%</span>
</div>
<div class="pie-segment" style="--percentage: 30; --color: #28a745">
<span class="segment-label">Средние держатели</span>
<span class="segment-value">30%</span>
</div>
<div class="pie-segment" style="--percentage: 20; --color: #ffc107">
<span class="segment-label">Малые держатели</span>
<span class="segment-value">20%</span>
</div>
<div class="pie-segment" style="--percentage: 10; --color: #dc3545">
<span class="segment-label">Резерв</span>
<span class="segment-value">10%</span>
</div>
</div>
</div>
<div class="stats-card">
<h3>Топ участников</h3>
<div class="top-participants">
<div
v-for="participant in topParticipants"
:key="participant.address"
class="participant-item"
>
<div class="participant-info">
<span class="participant-rank">#{{ participant.rank }}</span>
<span class="participant-address">{{ formatAddress(participant.address) }}</span>
</div>
<div class="participant-stats">
<span class="participant-balance">{{ participant.balance }} токенов</span>
<span class="participant-percentage">{{ participant.percentage }}%</span>
</div>
</div>
</div>
</div>
</div>
</div>
<!-- Отчеты -->
<div class="reports-section">
<h2>Отчеты</h2>
<div class="reports-grid">
<div class="report-card">
<h3>Ежемесячный отчет</h3>
<p>Подробный анализ деятельности DLE за последний месяц</p>
<div class="report-actions">
<button class="btn-secondary">Просмотреть</button>
<button class="btn-secondary">Скачать PDF</button>
</div>
</div>
<div class="report-card">
<h3>Финансовый отчет</h3>
<p>Анализ финансового состояния и доходности</p>
<div class="report-actions">
<button class="btn-secondary">Просмотреть</button>
<button class="btn-secondary">Скачать PDF</button>
</div>
</div>
<div class="report-card">
<h3>Отчет по предложениям</h3>
<p>Статистика и анализ предложений за период</p>
<div class="report-actions">
<button class="btn-secondary">Просмотреть</button>
<button class="btn-secondary">Скачать PDF</button>
</div>
</div>
<div class="report-card">
<h3>Отчет по активности</h3>
<p>Анализ активности участников и операций</p>
<div class="report-actions">
<button class="btn-secondary">Просмотреть</button>
<button class="btn-secondary">Скачать PDF</button>
</div>
</div>
</div>
</div>
</div>
</BaseLayout>
</template>
<script setup>
import { ref, defineProps, defineEmits } from 'vue';
import { useRouter } from 'vue-router';
import BaseLayout from '../../components/BaseLayout.vue';
// Определяем props
const props = defineProps({
isAuthenticated: Boolean,
identities: Array,
tokenBalances: Object,
isLoadingTokens: Boolean
});
// Определяем emits
const emit = defineEmits(['auth-action-completed']);
const router = useRouter();
// Ключевые метрики
const totalValue = ref(2500000);
const valueChange = ref(12.5);
const activeParticipants = ref(156);
const participantsChange = ref(23);
const totalProposals = ref(45);
const proposalsChange = ref(8);
const yieldRate = ref(8.7);
const yieldChange = ref(1.2);
// Топ участников (временные данные)
const topParticipants = ref([
{
rank: 1,
address: '0x1234567890123456789012345678901234567890',
balance: 2500,
percentage: 25.0
},
{
rank: 2,
address: '0x2345678901234567890123456789012345678901',
balance: 1800,
percentage: 18.0
},
{
rank: 3,
address: '0x3456789012345678901234567890123456789012',
balance: 1200,
percentage: 12.0
},
{
rank: 4,
address: '0x4567890123456789012345678901234567890123',
balance: 800,
percentage: 8.0
},
{
rank: 5,
address: '0x5678901234567890123456789012345678901234',
balance: 600,
percentage: 6.0
}
]);
// Методы
const formatAddress = (address) => {
if (!address) return '';
return address.substring(0, 6) + '...' + address.substring(address.length - 4);
};
</script>
<style scoped>
.analytics-container {
padding: 20px;
background-color: var(--color-white);
border-radius: var(--radius-lg);
box-shadow: 0 2px 10px rgba(0, 0, 0, 0.1);
margin-top: 20px;
margin-bottom: 20px;
}
.page-header {
display: flex;
justify-content: space-between;
align-items: flex-start;
margin-bottom: 40px;
padding-bottom: 20px;
border-bottom: 2px solid #f0f0f0;
}
.header-content {
flex-grow: 1;
}
.page-header h1 {
color: var(--color-primary);
font-size: 2.5rem;
margin: 0 0 10px 0;
}
.page-header p {
color: var(--color-grey-dark);
font-size: 1.1rem;
margin: 0;
}
.close-btn {
background: none;
border: none;
font-size: 1.5rem;
cursor: pointer;
color: #666;
padding: 0;
width: 30px;
height: 30px;
display: flex;
align-items: center;
justify-content: center;
border-radius: 50%;
transition: all 0.2s;
flex-shrink: 0;
}
.close-btn:hover {
background: #f0f0f0;
color: #333;
}
/* Секции */
.metrics-section,
.charts-section,
.statistics-section,
.reports-section {
margin-bottom: 40px;
}
.metrics-section h2,
.charts-section h2,
.statistics-section h2,
.reports-section h2 {
color: var(--color-primary);
margin-bottom: 20px;
font-size: 1.8rem;
}
/* Ключевые метрики */
.metrics-grid {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(250px, 1fr));
gap: 20px;
}
.metric-card {
background: #f8f9fa;
padding: 25px;
border-radius: var(--radius-lg);
border-left: 4px solid var(--color-primary);
text-align: center;
}
.metric-card h3 {
color: var(--color-primary);
margin-bottom: 15px;
font-size: 1rem;
text-transform: uppercase;
font-weight: 600;
}
.metric-value {
font-size: 2rem;
font-weight: 700;
margin: 0 0 10px 0;
color: var(--color-primary);
}
.metric-change {
font-size: 1rem;
font-weight: 600;
margin: 0;
}
.metric-change.positive {
color: #28a745;
}
.metric-change.negative {
color: #dc3545;
}
/* Графики */
.charts-grid {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(400px, 1fr));
gap: 30px;
}
.chart-card {
background: white;
padding: 25px;
border-radius: var(--radius-lg);
border: 1px solid #e9ecef;
}
.chart-card h3 {
color: var(--color-primary);
margin-bottom: 20px;
font-size: 1.3rem;
}
/* График стоимости токенов */
.chart-placeholder {
height: 200px;
background: #f8f9fa;
border-radius: var(--radius-sm);
position: relative;
margin-bottom: 15px;
border: 1px solid #e9ecef;
}
.chart-line {
position: relative;
height: 100%;
width: 100%;
}
.chart-point {
position: absolute;
width: 8px;
height: 8px;
background: var(--color-primary);
border-radius: 50%;
transform: translate(-50%, -50%);
}
.chart-point::before {
content: '';
position: absolute;
top: 50%;
left: 50%;
width: 12px;
height: 12px;
background: rgba(0, 123, 255, 0.2);
border-radius: 50%;
transform: translate(-50%, -50%);
}
/* График активности */
.activity-chart {
height: 200px;
display: flex;
align-items: end;
justify-content: space-between;
gap: 10px;
margin-bottom: 15px;
}
.activity-bar {
background: var(--color-secondary);
border-radius: 4px 4px 0 0;
min-width: 30px;
position: relative;
transition: all 0.3s ease;
}
.activity-bar:hover {
background: var(--color-secondary-dark);
}
.bar-label {
position: absolute;
bottom: -25px;
left: 50%;
transform: translateX(-50%);
font-size: 0.8rem;
color: var(--color-grey-dark);
}
/* Легенда графиков */
.chart-legend {
display: flex;
justify-content: center;
gap: 20px;
}
.legend-item {
display: flex;
align-items: center;
gap: 8px;
font-size: 0.9rem;
color: var(--color-grey-dark);
}
.legend-color {
width: 12px;
height: 12px;
border-radius: 2px;
}
/* Статистика */
.stats-grid {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(400px, 1fr));
gap: 30px;
}
.stats-card {
background: white;
padding: 25px;
border-radius: var(--radius-lg);
border: 1px solid #e9ecef;
}
.stats-card h3 {
color: var(--color-primary);
margin-bottom: 20px;
font-size: 1.3rem;
}
/* Круговая диаграмма */
.distribution-chart {
display: flex;
flex-direction: column;
gap: 15px;
}
.pie-segment {
display: flex;
justify-content: space-between;
align-items: center;
padding: 10px 15px;
background: #f8f9fa;
border-radius: var(--radius-sm);
border-left: 4px solid var(--color);
}
.segment-label {
font-weight: 600;
color: var(--color-grey-dark);
}
.segment-value {
font-weight: 700;
color: var(--color);
}
/* Топ участников */
.top-participants {
display: flex;
flex-direction: column;
gap: 15px;
}
.participant-item {
display: flex;
justify-content: space-between;
align-items: center;
padding: 15px;
background: #f8f9fa;
border-radius: var(--radius-sm);
transition: all 0.3s ease;
}
.participant-item:hover {
background: #e9ecef;
}
.participant-info {
display: flex;
align-items: center;
gap: 15px;
}
.participant-rank {
background: var(--color-primary);
color: white;
width: 24px;
height: 24px;
border-radius: 50%;
display: flex;
align-items: center;
justify-content: center;
font-size: 0.8rem;
font-weight: 600;
}
.participant-address {
font-family: monospace;
font-size: 0.9rem;
color: var(--color-grey-dark);
}
.participant-stats {
display: flex;
flex-direction: column;
align-items: flex-end;
gap: 5px;
}
.participant-balance {
font-weight: 600;
color: var(--color-primary);
font-size: 0.9rem;
}
.participant-percentage {
font-size: 0.8rem;
color: var(--color-grey-dark);
}
/* Отчеты */
.reports-grid {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(300px, 1fr));
gap: 20px;
}
.report-card {
background: white;
padding: 25px;
border-radius: var(--radius-lg);
border: 1px solid #e9ecef;
transition: all 0.3s ease;
}
.report-card:hover {
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.1);
}
.report-card h3 {
color: var(--color-primary);
margin-bottom: 15px;
font-size: 1.2rem;
}
.report-card p {
color: var(--color-grey-dark);
margin-bottom: 20px;
line-height: 1.5;
}
.report-actions {
display: flex;
gap: 10px;
}
/* Кнопки */
.btn-secondary {
background: var(--color-secondary);
color: white;
border: none;
padding: 8px 16px;
border-radius: var(--radius-sm);
cursor: pointer;
font-size: 0.9rem;
font-weight: 600;
transition: all 0.3s ease;
}
.btn-secondary:hover {
background: var(--color-secondary-dark);
}
/* Адаптивность */
@media (max-width: 768px) {
.charts-grid {
grid-template-columns: 1fr;
}
.stats-grid {
grid-template-columns: 1fr;
}
.reports-grid {
grid-template-columns: 1fr;
}
.report-actions {
flex-direction: column;
}
.participant-item {
flex-direction: column;
align-items: flex-start;
gap: 10px;
}
.participant-stats {
align-items: flex-start;
}
.metrics-grid {
grid-template-columns: 1fr;
}
}
</style>