ваше сообщение коммита
This commit is contained in:
@@ -95,6 +95,7 @@ const dleTokensRoutes = require('./routes/dleTokens'); // Функции ток
|
||||
const dleAnalyticsRoutes = require('./routes/dleAnalytics'); // Аналитика и история
|
||||
const dleMultichainRoutes = require('./routes/dleMultichain'); // Мультичейн функции
|
||||
const dleHistoryRoutes = require('./routes/dleHistory'); // Расширенная история
|
||||
const systemRoutes = require('./routes/system'); // Добавляем импорт маршрутов системного мониторинга
|
||||
|
||||
const app = express();
|
||||
|
||||
@@ -233,6 +234,7 @@ app.use('/api/identities', identitiesRoutes);
|
||||
app.use('/api/rag', ragRoutes); // Подключаем роут
|
||||
app.use('/api/monitoring', monitoringRoutes);
|
||||
app.use('/api/pages', pagesRoutes); // Подключаем роутер страниц
|
||||
app.use('/api/system', systemRoutes); // Добавляем маршрут системного мониторинга
|
||||
|
||||
const nonceStore = new Map(); // или любая другая реализация хранилища nonce
|
||||
|
||||
|
||||
@@ -1,4 +1,4 @@
|
||||
{
|
||||
"_format": "hh-sol-dbg-1",
|
||||
"buildInfo": "../../build-info/ab387c71734b3d3e5e7817d328027586.json"
|
||||
"buildInfo": "../../build-info/aa0034b410e4fbe1d1ff90369d480540.json"
|
||||
}
|
||||
|
||||
File diff suppressed because one or more lines are too long
4
backend/cache/solidity-files-cache.json
vendored
4
backend/cache/solidity-files-cache.json
vendored
@@ -2,8 +2,8 @@
|
||||
"_format": "hh-sol-cache-2",
|
||||
"files": {
|
||||
"/home/alex/Digital_Legal_Entity(DLE)/backend/contracts/DLE.sol": {
|
||||
"lastModificationDate": 1755280436490,
|
||||
"contentHash": "f676e9964a39b0fccdc62a9114266863",
|
||||
"lastModificationDate": 1755366617069,
|
||||
"contentHash": "47d6b51ed0025b36c50649b175745512",
|
||||
"sourceName": "contracts/DLE.sol",
|
||||
"solcConfig": {
|
||||
"version": "0.8.20",
|
||||
|
||||
@@ -20,6 +20,12 @@ import "@openzeppelin/contracts/utils/cryptography/ECDSA.sol";
|
||||
* @title DLE (Digital Legal Entity)
|
||||
* @dev Основной контракт DLE с модульной архитектурой, Single-Chain Governance
|
||||
* и безопасной мульти-чейн синхронизацией без сторонних мостов (через подписи холдеров).
|
||||
*
|
||||
* КЛЮЧЕВЫЕ ОСОБЕННОСТИ:
|
||||
* - Прямые переводы токенов ЗАБЛОКИРОВАНЫ (transfer, transferFrom, approve)
|
||||
* - Перевод токенов возможен ТОЛЬКО через governance предложения
|
||||
* - Токены служат только для голосования и управления DLE
|
||||
* - Все операции с токенами требуют коллективного решения
|
||||
*/
|
||||
contract DLE is ERC20, ERC20Permit, ERC20Votes, ReentrancyGuard {
|
||||
using ECDSA for bytes32;
|
||||
@@ -112,6 +118,7 @@ contract DLE is ERC20, ERC20Permit, ERC20Votes, ReentrancyGuard {
|
||||
event DLEInfoUpdated(string name, string symbol, string location, string coordinates, uint256 jurisdiction, string[] okvedCodes, uint256 kpp);
|
||||
event QuorumPercentageUpdated(uint256 oldQuorumPercentage, uint256 newQuorumPercentage);
|
||||
event CurrentChainIdUpdated(uint256 oldChainId, uint256 newChainId);
|
||||
event TokensTransferredByGovernance(address indexed recipient, uint256 amount);
|
||||
|
||||
// EIP712 typehash для подписи одобрения исполнения предложения в целевой сети
|
||||
// ExecutionApproval(uint256 proposalId, bytes32 operationHash, uint256 chainId, uint256 snapshotTimepoint)
|
||||
@@ -531,6 +538,10 @@ contract DLE is ERC20, ERC20Permit, ERC20Votes, ReentrancyGuard {
|
||||
} else if (selector == bytes4(keccak256("_removeSupportedChain(uint256)"))) {
|
||||
(uint256 chainIdToRemove) = abi.decode(data, (uint256));
|
||||
_removeSupportedChain(chainIdToRemove);
|
||||
} else if (selector == bytes4(keccak256("_transferTokens(address,uint256)"))) {
|
||||
// Операция перевода токенов через governance
|
||||
(address recipient, uint256 amount) = abi.decode(data, (address, uint256));
|
||||
_transferTokens(recipient, amount);
|
||||
} else if (selector == bytes4(keccak256("offchainAction(bytes32,string,bytes32)"))) {
|
||||
// Оффчейн операция для приложения: идентификатор, тип, хеш полезной нагрузки
|
||||
// (bytes32 actionId, string memory kind, bytes32 payloadHash) = abi.decode(data, (bytes32, string, bytes32));
|
||||
@@ -604,6 +615,22 @@ contract DLE is ERC20, ERC20Permit, ERC20Votes, ReentrancyGuard {
|
||||
emit CurrentChainIdUpdated(oldChainId, _newChainId);
|
||||
}
|
||||
|
||||
/**
|
||||
* @dev Перевести токены через governance (от имени DLE)
|
||||
* @param _recipient Адрес получателя
|
||||
* @param _amount Количество токенов для перевода
|
||||
*/
|
||||
function _transferTokens(address _recipient, uint256 _amount) internal {
|
||||
require(_recipient != address(0), "Cannot transfer to zero address");
|
||||
require(_amount > 0, "Amount must be positive");
|
||||
require(balanceOf(address(this)) >= _amount, "Insufficient DLE balance");
|
||||
|
||||
// Переводим токены от имени DLE (address(this))
|
||||
_transfer(address(this), _recipient, _amount);
|
||||
|
||||
emit TokensTransferredByGovernance(_recipient, _amount);
|
||||
}
|
||||
|
||||
/**
|
||||
* @dev Создать предложение о добавлении модуля
|
||||
* @param _description Описание предложения
|
||||
@@ -896,4 +923,38 @@ contract DLE is ERC20, ERC20Permit, ERC20Votes, ReentrancyGuard {
|
||||
require(delegator == delegatee, "Delegation disabled");
|
||||
super._delegate(delegator, delegatee);
|
||||
}
|
||||
|
||||
// ===== Блокировка прямых переводов токенов =====
|
||||
// Токены DLE могут быть переведены только через governance
|
||||
|
||||
/**
|
||||
* @dev Блокирует прямые переводы токенов
|
||||
* @param to Адрес получателя (не используется)
|
||||
* @param amount Количество токенов (не используется)
|
||||
* @return Всегда возвращает false
|
||||
*/
|
||||
function transfer(address to, uint256 amount) public override returns (bool) {
|
||||
revert("Direct transfers disabled. Use governance proposals for token transfers.");
|
||||
}
|
||||
|
||||
/**
|
||||
* @dev Блокирует прямые переводы токенов через approve/transferFrom
|
||||
* @param from Адрес отправителя (не используется)
|
||||
* @param to Адрес получателя (не используется)
|
||||
* @param amount Количество токенов (не используется)
|
||||
* @return Всегда возвращает false
|
||||
*/
|
||||
function transferFrom(address from, address to, uint256 amount) public override returns (bool) {
|
||||
revert("Direct transfers disabled. Use governance proposals for token transfers.");
|
||||
}
|
||||
|
||||
/**
|
||||
* @dev Блокирует прямые разрешения на перевод токенов
|
||||
* @param spender Адрес, которому разрешается тратить токены (не используется)
|
||||
* @param amount Количество токенов (не используется)
|
||||
* @return Всегда возвращает false
|
||||
*/
|
||||
function approve(address spender, uint256 amount) public override returns (bool) {
|
||||
revert("Direct approvals disabled. Use governance proposals for token transfers.");
|
||||
}
|
||||
}
|
||||
|
||||
@@ -30,8 +30,39 @@ let pool = new Pool({
|
||||
user: process.env.DB_USER || 'dapp_user',
|
||||
password: process.env.DB_PASSWORD,
|
||||
ssl: false,
|
||||
// Настройки для предотвращения утечек памяти
|
||||
max: 10, // Максимальное количество клиентов в пуле
|
||||
min: 0, // Минимальное количество клиентов в пуле
|
||||
idleTimeoutMillis: 30000, // Время жизни неактивного клиента (30 сек)
|
||||
connectionTimeoutMillis: 2000, // Таймаут подключения (2 сек)
|
||||
maxUses: 7500, // Максимальное количество использований клиента
|
||||
allowExitOnIdle: true, // Разрешить выход при отсутствии активных клиентов
|
||||
});
|
||||
|
||||
// Увеличиваем лимит обработчиков событий для предотвращения предупреждений
|
||||
pool.setMaxListeners(20);
|
||||
|
||||
// Добавляем обработчики для правильного закрытия пула
|
||||
pool.on('error', (err) => {
|
||||
console.error('Unexpected error on idle client', err);
|
||||
process.exit(-1);
|
||||
});
|
||||
|
||||
// Обработчик для очистки при завершении процесса
|
||||
process.on('SIGINT', async () => {
|
||||
console.log('Closing database pool...');
|
||||
await pool.end();
|
||||
process.exit(0);
|
||||
});
|
||||
|
||||
process.on('SIGTERM', async () => {
|
||||
console.log('Closing database pool...');
|
||||
await pool.end();
|
||||
process.exit(0);
|
||||
});
|
||||
|
||||
console.log('Пул создан:', pool.options || pool);
|
||||
|
||||
// Проверяем подключение к базе данных
|
||||
pool.query('SELECT NOW()')
|
||||
.then(res => {
|
||||
@@ -41,8 +72,6 @@ pool.query('SELECT NOW()')
|
||||
console.error('Ошибка подключения к базе данных:', err);
|
||||
});
|
||||
|
||||
console.log('Пул создан:', pool.options || pool);
|
||||
|
||||
function getPool() {
|
||||
return pool;
|
||||
}
|
||||
@@ -69,10 +98,11 @@ async function reinitPoolFromDbSettings() {
|
||||
if (!res.rows.length) throw new Error('DB settings not found');
|
||||
const dbSettings = res.rows[0];
|
||||
|
||||
// Закрываем старый пул
|
||||
// Закрываем старый пул правильно
|
||||
console.log('Закрываем старый пул подключений...');
|
||||
await pool.end();
|
||||
|
||||
// Создаём новый пул с расшифрованными настройками
|
||||
// Создаём новый пул с расшифрованными настройками и теми же параметрами для предотвращения утечек
|
||||
pool = new Pool({
|
||||
host: dbSettings.db_host_encrypted ? await decryptValue(dbSettings.db_host_encrypted) : process.env.DB_HOST || 'postgres',
|
||||
port: parseInt(dbSettings.db_port || process.env.DB_PORT || '5432'),
|
||||
@@ -80,6 +110,22 @@ async function reinitPoolFromDbSettings() {
|
||||
user: dbSettings.db_user_encrypted ? await decryptValue(dbSettings.db_user_encrypted) : process.env.DB_USER || 'dapp_user',
|
||||
password: dbSettings.db_password_encrypted ? await decryptValue(dbSettings.db_password_encrypted) : process.env.DB_PASSWORD,
|
||||
ssl: false,
|
||||
// Те же настройки для предотвращения утечек
|
||||
max: 10,
|
||||
min: 0,
|
||||
idleTimeoutMillis: 30000,
|
||||
connectionTimeoutMillis: 2000,
|
||||
maxUses: 7500,
|
||||
allowExitOnIdle: true,
|
||||
});
|
||||
|
||||
// Устанавливаем лимит обработчиков для нового пула
|
||||
pool.setMaxListeners(20);
|
||||
|
||||
// Добавляем обработчики ошибок для нового пула
|
||||
pool.on('error', (err) => {
|
||||
console.error('Unexpected error on idle client', err);
|
||||
process.exit(-1);
|
||||
});
|
||||
|
||||
// Пересоздаём session middleware
|
||||
|
||||
99
backend/routes/system.js
Normal file
99
backend/routes/system.js
Normal file
@@ -0,0 +1,99 @@
|
||||
/**
|
||||
* 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
|
||||
*/
|
||||
|
||||
const express = require('express');
|
||||
const router = express.Router();
|
||||
const memoryMonitor = require('../utils/memoryMonitor');
|
||||
const logger = require('../utils/logger');
|
||||
const { checkAdminRole } = require('../services/admin-role');
|
||||
|
||||
// Middleware для проверки прав администратора
|
||||
const requireAdmin = async (req, res, next) => {
|
||||
try {
|
||||
if (!req.session || !req.session.userId) {
|
||||
return res.status(401).json({ success: false, error: 'Unauthorized' });
|
||||
}
|
||||
|
||||
const isAdmin = await checkAdminRole(req.session.userId);
|
||||
if (!isAdmin) {
|
||||
return res.status(403).json({ success: false, error: 'Admin access required' });
|
||||
}
|
||||
|
||||
next();
|
||||
} catch (error) {
|
||||
logger.error('Error checking admin role:', error);
|
||||
res.status(500).json({ success: false, error: 'Internal server error' });
|
||||
}
|
||||
};
|
||||
|
||||
// GET /api/system/memory - Получить информацию о памяти
|
||||
router.get('/memory', requireAdmin, (req, res) => {
|
||||
try {
|
||||
const memoryUsage = memoryMonitor.getMemoryUsage();
|
||||
res.json({
|
||||
success: true,
|
||||
data: {
|
||||
memory: memoryUsage,
|
||||
timestamp: new Date().toISOString()
|
||||
}
|
||||
});
|
||||
} catch (error) {
|
||||
logger.error('Error getting memory usage:', error);
|
||||
res.status(500).json({ success: false, error: 'Failed to get memory usage' });
|
||||
}
|
||||
});
|
||||
|
||||
// POST /api/system/memory/start - Запустить мониторинг памяти
|
||||
router.post('/memory/start', requireAdmin, (req, res) => {
|
||||
try {
|
||||
const { interval } = req.body;
|
||||
memoryMonitor.start(interval || 60000);
|
||||
res.json({ success: true, message: 'Memory monitoring started' });
|
||||
} catch (error) {
|
||||
logger.error('Error starting memory monitoring:', error);
|
||||
res.status(500).json({ success: false, error: 'Failed to start memory monitoring' });
|
||||
}
|
||||
});
|
||||
|
||||
// POST /api/system/memory/stop - Остановить мониторинг памяти
|
||||
router.post('/memory/stop', requireAdmin, (req, res) => {
|
||||
try {
|
||||
memoryMonitor.stop();
|
||||
res.json({ success: true, message: 'Memory monitoring stopped' });
|
||||
} catch (error) {
|
||||
logger.error('Error stopping memory monitoring:', error);
|
||||
res.status(500).json({ success: false, error: 'Failed to stop memory monitoring' });
|
||||
}
|
||||
});
|
||||
|
||||
// GET /api/system/health - Проверка здоровья системы
|
||||
router.get('/health', (req, res) => {
|
||||
try {
|
||||
const memoryUsage = memoryMonitor.getMemoryUsage();
|
||||
const uptime = process.uptime();
|
||||
|
||||
res.json({
|
||||
success: true,
|
||||
data: {
|
||||
status: 'healthy',
|
||||
uptime: Math.round(uptime),
|
||||
memory: memoryUsage,
|
||||
timestamp: new Date().toISOString()
|
||||
}
|
||||
});
|
||||
} catch (error) {
|
||||
logger.error('Error getting system health:', error);
|
||||
res.status(500).json({ success: false, error: 'Failed to get system health' });
|
||||
}
|
||||
});
|
||||
|
||||
module.exports = router;
|
||||
@@ -19,6 +19,7 @@ const { getBot } = require('./services/telegramBot');
|
||||
const EmailBotService = require('./services/emailBot');
|
||||
const { initDbPool, seedAIAssistantSettings } = require('./db');
|
||||
const { warmupModel } = require('./scripts/warmup-model'); // Добавляем импорт разогрева модели
|
||||
const memoryMonitor = require('./utils/memoryMonitor');
|
||||
|
||||
const PORT = process.env.PORT || 8000;
|
||||
|
||||
@@ -98,4 +99,25 @@ process.on('uncaughtException', (err) => {
|
||||
logger.error('Uncaught Exception:', err);
|
||||
});
|
||||
|
||||
// Запускаем мониторинг памяти в production
|
||||
if (process.env.NODE_ENV === 'production') {
|
||||
memoryMonitor.start(300000); // Каждые 5 минут
|
||||
logger.info('[Server] Мониторинг памяти запущен в production режиме');
|
||||
}
|
||||
|
||||
// Обработчики для корректного завершения
|
||||
process.on('SIGINT', async () => {
|
||||
logger.info('[Server] Получен сигнал SIGINT, завершаем работу...');
|
||||
memoryMonitor.stop();
|
||||
await initDbPool().then(pool => pool.end()); // Use initDbPool to get the pool
|
||||
process.exit(0);
|
||||
});
|
||||
|
||||
process.on('SIGTERM', async () => {
|
||||
logger.info('[Server] Получен сигнал SIGTERM, завершаем работу...');
|
||||
memoryMonitor.stop();
|
||||
await initDbPool().then(pool => pool.end()); // Use initDbPool to get the pool
|
||||
process.exit(0);
|
||||
});
|
||||
|
||||
module.exports = app;
|
||||
|
||||
@@ -29,6 +29,32 @@ const { isUserBlocked } = require('../utils/userUtils');
|
||||
class EmailBotService {
|
||||
constructor() {
|
||||
// console.log('[EmailBot] constructor called');
|
||||
this.imap = null;
|
||||
this.isChecking = false;
|
||||
this.reconnectAttempts = 0;
|
||||
this.maxReconnectAttempts = 3;
|
||||
}
|
||||
|
||||
// Метод для очистки IMAP соединения
|
||||
cleanupImapConnection() {
|
||||
if (this.imap) {
|
||||
try {
|
||||
// Удаляем все обработчики событий
|
||||
this.imap.removeAllListeners('error');
|
||||
this.imap.removeAllListeners('ready');
|
||||
this.imap.removeAllListeners('end');
|
||||
this.imap.removeAllListeners('close');
|
||||
|
||||
// Закрываем соединение
|
||||
if (this.imap.state !== 'disconnected') {
|
||||
this.imap.end();
|
||||
}
|
||||
} catch (error) {
|
||||
logger.error('[EmailBot] Error cleaning up IMAP connection:', error);
|
||||
} finally {
|
||||
this.imap = null;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
async getSettingsFromDb() {
|
||||
@@ -533,56 +559,50 @@ class EmailBotService {
|
||||
try {
|
||||
// console.log('[EmailBot] start() called');
|
||||
logger.info('[EmailBot] start() called');
|
||||
const imapConfig = await this.getImapConfig();
|
||||
// Логируем IMAP-конфиг (без пароля)
|
||||
const safeConfig = { ...imapConfig };
|
||||
if (safeConfig.password) safeConfig.password = '***';
|
||||
logger.info('[EmailBot] IMAP config:', safeConfig);
|
||||
|
||||
// Очищаем предыдущее соединение если есть
|
||||
this.cleanupImapConnection();
|
||||
|
||||
let attempt = 0;
|
||||
const maxAttempts = 3;
|
||||
this.isChecking = false;
|
||||
const tryConnect = () => {
|
||||
|
||||
const tryConnect = async () => {
|
||||
attempt++;
|
||||
logger.info(`[EmailBot] IMAP connect attempt ${attempt}`);
|
||||
this.imap = new Imap(imapConfig);
|
||||
this.imap = new Imap(await this.getImapConfig());
|
||||
|
||||
// Устанавливаем обработчики событий
|
||||
this.imap.once('ready', () => {
|
||||
logger.info('[EmailBot] IMAP connection ready');
|
||||
this.imap.openBox('INBOX', false, (err, box) => {
|
||||
if (err) {
|
||||
logger.error(`[EmailBot] Error opening INBOX: ${err.message}`);
|
||||
this.imap.end();
|
||||
return;
|
||||
}
|
||||
logger.info('[EmailBot] INBOX opened successfully');
|
||||
});
|
||||
// После успешного подключения — обычная логика
|
||||
this.reconnectAttempts = 0; // Сбрасываем счетчик при успешном подключении
|
||||
this.checkEmails();
|
||||
logger.info('[EmailBot] Email bot started and IMAP connection initiated');
|
||||
// Периодическая проверка почты
|
||||
setInterval(async () => {
|
||||
if (this.isChecking) return;
|
||||
this.isChecking = true;
|
||||
try {
|
||||
await this.checkEmails();
|
||||
} catch (e) {
|
||||
logger.error('[EmailBot] Error in periodic checkEmails:', e);
|
||||
}
|
||||
this.isChecking = false;
|
||||
}, 60000); // 60 секунд
|
||||
});
|
||||
|
||||
this.imap.once('end', () => {
|
||||
logger.info('[EmailBot] IMAP connection ended');
|
||||
this.cleanupImapConnection();
|
||||
});
|
||||
|
||||
this.imap.once('close', () => {
|
||||
logger.info('[EmailBot] IMAP connection closed');
|
||||
this.cleanupImapConnection();
|
||||
});
|
||||
|
||||
this.imap.once('error', (err) => {
|
||||
logger.error(`[EmailBot] IMAP connection error: ${err.message}`);
|
||||
this.cleanupImapConnection();
|
||||
|
||||
if (err.message && err.message.toLowerCase().includes('timed out') && attempt < maxAttempts) {
|
||||
logger.warn(`[EmailBot] IMAP reconnecting in 10 seconds (attempt ${attempt + 1})...`);
|
||||
setTimeout(tryConnect, 10000);
|
||||
}
|
||||
});
|
||||
|
||||
this.imap.connect();
|
||||
};
|
||||
tryConnect();
|
||||
} catch (err) {
|
||||
// console.error('[EmailBot] Ошибка при старте:', err);
|
||||
logger.error('[EmailBot] Ошибка при старте:', err);
|
||||
this.cleanupImapConnection();
|
||||
throw err;
|
||||
}
|
||||
}
|
||||
|
||||
99
backend/utils/memoryMonitor.js
Normal file
99
backend/utils/memoryMonitor.js
Normal file
@@ -0,0 +1,99 @@
|
||||
/**
|
||||
* 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
|
||||
*/
|
||||
|
||||
const logger = require('./logger');
|
||||
|
||||
class MemoryMonitor {
|
||||
constructor() {
|
||||
this.monitoring = false;
|
||||
this.interval = null;
|
||||
this.lastMemoryUsage = null;
|
||||
}
|
||||
|
||||
start(intervalMs = 60000) { // По умолчанию каждую минуту
|
||||
if (this.monitoring) {
|
||||
logger.warn('[MemoryMonitor] Мониторинг уже запущен');
|
||||
return;
|
||||
}
|
||||
|
||||
this.monitoring = true;
|
||||
this.interval = setInterval(() => {
|
||||
this.checkMemoryUsage();
|
||||
}, intervalMs);
|
||||
|
||||
logger.info('[MemoryMonitor] Мониторинг памяти запущен');
|
||||
}
|
||||
|
||||
stop() {
|
||||
if (this.interval) {
|
||||
clearInterval(this.interval);
|
||||
this.interval = null;
|
||||
}
|
||||
this.monitoring = false;
|
||||
logger.info('[MemoryMonitor] Мониторинг памяти остановлен');
|
||||
}
|
||||
|
||||
checkMemoryUsage() {
|
||||
const memUsage = process.memoryUsage();
|
||||
const memUsageMB = {
|
||||
rss: Math.round(memUsage.rss / 1024 / 1024),
|
||||
heapTotal: Math.round(memUsage.heapTotal / 1024 / 1024),
|
||||
heapUsed: Math.round(memUsage.heapUsed / 1024 / 1024),
|
||||
external: Math.round(memUsage.external / 1024 / 1024),
|
||||
arrayBuffers: Math.round(memUsage.arrayBuffers / 1024 / 1024)
|
||||
};
|
||||
|
||||
// Проверяем рост памяти
|
||||
if (this.lastMemoryUsage) {
|
||||
const growth = {
|
||||
rss: memUsageMB.rss - this.lastMemoryUsage.rss,
|
||||
heapUsed: memUsageMB.heapUsed - this.lastMemoryUsage.heapUsed
|
||||
};
|
||||
|
||||
// Логируем если есть значительный рост
|
||||
if (growth.rss > 50 || growth.heapUsed > 20) {
|
||||
logger.warn('[MemoryMonitor] Обнаружен рост памяти:', {
|
||||
current: memUsageMB,
|
||||
growth: growth
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
this.lastMemoryUsage = memUsageMB;
|
||||
|
||||
// Логируем текущее использование памяти
|
||||
logger.info('[MemoryMonitor] Использование памяти:', memUsageMB);
|
||||
}
|
||||
|
||||
getMemoryUsage() {
|
||||
const memUsage = process.memoryUsage();
|
||||
return {
|
||||
rss: Math.round(memUsage.rss / 1024 / 1024),
|
||||
heapTotal: Math.round(memUsage.heapTotal / 1024 / 1024),
|
||||
heapUsed: Math.round(memUsage.heapUsed / 1024 / 1024),
|
||||
external: Math.round(memUsage.external / 1024 / 1024),
|
||||
arrayBuffers: Math.round(memUsage.arrayBuffers / 1024 / 1024)
|
||||
};
|
||||
}
|
||||
|
||||
// Проверка утечек в EventEmitter
|
||||
checkEventEmitterLeaks() {
|
||||
const eventEmitter = require('events');
|
||||
const defaultMaxListeners = eventEmitter.defaultMaxListeners;
|
||||
|
||||
logger.info('[MemoryMonitor] EventEmitter defaultMaxListeners:', defaultMaxListeners);
|
||||
|
||||
// Можно добавить дополнительную логику для проверки конкретных EventEmitter'ов
|
||||
}
|
||||
}
|
||||
|
||||
module.exports = new MemoryMonitor();
|
||||
@@ -18,6 +18,7 @@
|
||||
- Безопасность: удалены уязвимые Merkle‑механизмы cross‑chain; нет внешних мостов/оракулов.
|
||||
- Голосующая сила: OpenZeppelin `ERC20Votes` (снимки `getPastVotes`, `getPastTotalSupply`).
|
||||
- Делегирование: жестко ограничено «только на себя»; третьим лицам делегировать нельзя (1 токен = 1 голос).
|
||||
- Переводы токенов: ЗАБЛОКИРОВАНЫ прямые переводы (transfer, transferFrom, approve); переводы возможны ТОЛЬКО через governance предложения.
|
||||
- Single‑Chain Governance: голосование происходит в одной выбранной сети (`governanceChainId`), время снапшота фиксируется на создании предложения и используется во всех сетях.
|
||||
- Multi‑Chain исполнение: выполнение в целевых сетях по EIP‑712 подписям холдеров, проверяется суммарная голосующая сила на зафиксированном `timepoint` (без доверия к мостам).
|
||||
- «100% или ничего»: операции считаются успешными только при готовности/успешности всех целевых сетей.
|
||||
@@ -94,6 +95,8 @@ DLE.sol (Один контракт)
|
||||
- Распределение токенов между участниками
|
||||
- **Голосующая сила = количество токенов**
|
||||
- Проверка баланса токенов при каждой операции
|
||||
- **Прямые переводы ЗАБЛОКИРОВАНЫ** - токены служат только для голосования
|
||||
- **Переводы возможны ТОЛЬКО через governance предложения**
|
||||
|
||||
#### 2. Настраиваемый кворум
|
||||
- **Описание**: Процент от общего количества токенов для принятия решений
|
||||
|
||||
@@ -232,6 +232,46 @@ const routes = [
|
||||
name: 'management-modules',
|
||||
component: () => import('../views/smartcontracts/ModulesView.vue')
|
||||
},
|
||||
{
|
||||
path: '/management/modules/deploy/treasury',
|
||||
name: 'module-deploy-treasury',
|
||||
component: () => import('../views/smartcontracts/modules/TreasuryModuleDeployView.vue')
|
||||
},
|
||||
{
|
||||
path: '/management/modules/deploy/timelock',
|
||||
name: 'module-deploy-timelock',
|
||||
component: () => import('../views/smartcontracts/modules/TimelockModuleDeployView.vue')
|
||||
},
|
||||
{
|
||||
path: '/management/modules/deploy/communication',
|
||||
name: 'module-deploy-communication',
|
||||
component: () => import('../views/smartcontracts/modules/CommunicationModuleDeployView.vue')
|
||||
},
|
||||
{
|
||||
path: '/management/modules/deploy/application',
|
||||
name: 'module-deploy-application',
|
||||
component: () => import('../views/smartcontracts/modules/ApplicationModuleDeployView.vue')
|
||||
},
|
||||
{
|
||||
path: '/management/modules/deploy/mint',
|
||||
name: 'module-deploy-mint',
|
||||
component: () => import('../views/smartcontracts/modules/MintModuleDeploy.vue')
|
||||
},
|
||||
{
|
||||
path: '/management/modules/deploy/burn',
|
||||
name: 'module-deploy-burn',
|
||||
component: () => import('../views/smartcontracts/modules/BurnModuleDeploy.vue')
|
||||
},
|
||||
{
|
||||
path: '/management/modules/deploy/oracle',
|
||||
name: 'module-deploy-oracle',
|
||||
component: () => import('../views/smartcontracts/modules/OracleModuleDeploy.vue')
|
||||
},
|
||||
{
|
||||
path: '/management/modules/deploy/custom',
|
||||
name: 'module-deploy-custom',
|
||||
component: () => import('../views/smartcontracts/modules/ModuleDeployFormView.vue')
|
||||
},
|
||||
// {
|
||||
// path: '/management/multisig',
|
||||
// name: 'management-multisig',
|
||||
|
||||
@@ -751,4 +751,79 @@ export async function loadDeactivationProposals(dleAddress) {
|
||||
console.error('Ошибка загрузки предложений деактивации:', error);
|
||||
return [];
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Создать предложение о переводе токенов через governance
|
||||
* @param {string} dleAddress - Адрес DLE контракта
|
||||
* @param {Object} transferData - Данные перевода
|
||||
* @param {string} transferData.recipient - Адрес получателя
|
||||
* @param {number} transferData.amount - Количество токенов
|
||||
* @param {string} transferData.description - Описание предложения
|
||||
* @param {number} transferData.duration - Длительность голосования в секундах
|
||||
* @param {number} transferData.governanceChainId - ID сети для голосования
|
||||
* @param {Array<number>} transferData.targetChains - Целевые сети для исполнения
|
||||
* @returns {Promise<Object>} - Результат создания предложения
|
||||
*/
|
||||
export async function createTransferTokensProposal(dleAddress, transferData) {
|
||||
try {
|
||||
// Проверяем наличие браузерного кошелька
|
||||
if (!window.ethereum) {
|
||||
throw new Error('Браузерный кошелек не установлен');
|
||||
}
|
||||
|
||||
// Запрашиваем подключение к кошельку
|
||||
const accounts = await window.ethereum.request({ method: 'eth_requestAccounts' });
|
||||
const provider = new ethers.BrowserProvider(window.ethereum);
|
||||
const signer = await provider.getSigner();
|
||||
|
||||
// ABI для создания предложения
|
||||
const dleAbi = [
|
||||
"function createProposal(string memory _description, uint256 _duration, bytes memory _operation, uint256 _governanceChainId, uint256[] memory _targetChains, uint256 _timelockDelay) external returns (uint256)"
|
||||
];
|
||||
|
||||
const dle = new ethers.Contract(dleAddress, dleAbi, signer);
|
||||
|
||||
// Кодируем операцию перевода токенов
|
||||
const transferFunctionSelector = ethers.id("_transferTokens(address,uint256)");
|
||||
const transferDataEncoded = ethers.AbiCoder.defaultAbiCoder().encode(
|
||||
["address", "uint256"],
|
||||
[transferData.recipient, ethers.parseUnits(transferData.amount.toString(), 18)]
|
||||
);
|
||||
|
||||
// Объединяем селектор и данные
|
||||
const operation = ethers.concat([transferFunctionSelector, transferDataEncoded]);
|
||||
|
||||
console.log('Создание предложения о переводе токенов:', {
|
||||
recipient: transferData.recipient,
|
||||
amount: transferData.amount,
|
||||
description: transferData.description,
|
||||
operation: operation
|
||||
});
|
||||
|
||||
// Создаем предложение
|
||||
const tx = await dle.createProposal(
|
||||
transferData.description,
|
||||
transferData.duration,
|
||||
operation,
|
||||
transferData.governanceChainId,
|
||||
transferData.targetChains || [],
|
||||
0 // timelockDelay
|
||||
);
|
||||
|
||||
// Ждем подтверждения транзакции
|
||||
const receipt = await tx.wait();
|
||||
|
||||
console.log('Предложение о переводе токенов создано, tx hash:', tx.hash);
|
||||
|
||||
return {
|
||||
proposalId: receipt.logs[0]?.topics[1] || '0', // Извлекаем ID предложения из события
|
||||
txHash: tx.hash,
|
||||
blockNumber: receipt.blockNumber
|
||||
};
|
||||
|
||||
} catch (error) {
|
||||
console.error('Ошибка создания предложения о переводе токенов:', error);
|
||||
throw error;
|
||||
}
|
||||
}
|
||||
@@ -11,160 +11,460 @@
|
||||
-->
|
||||
|
||||
<template>
|
||||
<BaseLayout>
|
||||
<div class="content-page-block">
|
||||
<div class="content-header-nav">
|
||||
<button class="nav-btn" @click="goToCreate">Создать</button>
|
||||
<button class="nav-btn" @click="goToList">Список страниц</button>
|
||||
<button class="nav-btn" @click="goToSettings">Настройки</button>
|
||||
<BaseLayout
|
||||
:is-authenticated="isAuthenticated"
|
||||
:identities="identities"
|
||||
:token-balances="tokenBalances"
|
||||
:is-loading-tokens="isLoadingTokens"
|
||||
@auth-action-completed="$emit('auth-action-completed')"
|
||||
>
|
||||
<div class="content-create-page">
|
||||
<!-- Заголовок страницы -->
|
||||
<div class="page-header">
|
||||
<div class="header-content">
|
||||
<h1>📝 Создание страницы</h1>
|
||||
<p>Создайте новую страницу для вашего DLE</p>
|
||||
</div>
|
||||
<div class="header-actions">
|
||||
<button class="close-btn" @click="goBack">×</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Основной контент с тенью -->
|
||||
<div class="content-block">
|
||||
<form class="content-form" @submit.prevent="handleSubmit">
|
||||
<!-- Основная информация -->
|
||||
<div class="form-section">
|
||||
<h2>Основная информация</h2>
|
||||
<div class="form-group">
|
||||
<label for="title">Заголовок страницы *</label>
|
||||
<input
|
||||
v-model="form.title"
|
||||
id="title"
|
||||
type="text"
|
||||
required
|
||||
placeholder="Введите заголовок страницы"
|
||||
class="form-input"
|
||||
/>
|
||||
</div>
|
||||
<div class="form-group">
|
||||
<label for="summary">Краткое описание *</label>
|
||||
<textarea
|
||||
v-model="form.summary"
|
||||
id="summary"
|
||||
required
|
||||
rows="3"
|
||||
placeholder="Краткое описание страницы"
|
||||
class="form-textarea"
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Контент -->
|
||||
<div class="form-section">
|
||||
<h2>Содержание</h2>
|
||||
<div class="form-group">
|
||||
<label for="content">Основной контент *</label>
|
||||
<textarea
|
||||
v-model="form.content"
|
||||
id="content"
|
||||
required
|
||||
rows="10"
|
||||
placeholder="Введите основной контент страницы"
|
||||
class="form-textarea"
|
||||
/>
|
||||
<div class="content-stats">
|
||||
<span>Слов: {{ wordCount }}</span>
|
||||
<span>Символов: {{ characterCount }}</span>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- SEO настройки -->
|
||||
<div class="form-section">
|
||||
<h2>SEO настройки</h2>
|
||||
<div class="form-group">
|
||||
<label for="seo-title">Meta Title</label>
|
||||
<input
|
||||
v-model="form.seo.title"
|
||||
id="seo-title"
|
||||
type="text"
|
||||
placeholder="SEO заголовок (если отличается от основного)"
|
||||
class="form-input"
|
||||
/>
|
||||
</div>
|
||||
<div class="form-group">
|
||||
<label for="seo-description">Meta Description</label>
|
||||
<textarea
|
||||
v-model="form.seo.description"
|
||||
id="seo-description"
|
||||
rows="3"
|
||||
placeholder="SEO описание для поисковых систем"
|
||||
class="form-textarea"
|
||||
/>
|
||||
</div>
|
||||
<div class="form-group">
|
||||
<label for="seo-keywords">Keywords</label>
|
||||
<input
|
||||
v-model="form.seo.keywords"
|
||||
id="seo-keywords"
|
||||
type="text"
|
||||
placeholder="Ключевые слова через запятую"
|
||||
class="form-input"
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Настройки публикации -->
|
||||
<div class="form-section">
|
||||
<h2>Настройки публикации</h2>
|
||||
<div class="form-group">
|
||||
<label class="checkbox-label">
|
||||
<input
|
||||
type="checkbox"
|
||||
v-model="form.settings.autoPublish"
|
||||
class="form-checkbox"
|
||||
/>
|
||||
<span>Опубликовать сразу после создания</span>
|
||||
</label>
|
||||
</div>
|
||||
<div class="form-group">
|
||||
<label for="status">Статус</label>
|
||||
<select v-model="form.status" id="status" class="form-select">
|
||||
<option value="draft">Черновик</option>
|
||||
<option value="published">Опубликовано</option>
|
||||
<option value="pending">На модерации</option>
|
||||
</select>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Кнопки действий -->
|
||||
<div class="form-actions">
|
||||
<button type="button" class="btn btn-outline" @click="goBack">
|
||||
<i class="fas fa-times"></i>
|
||||
Отмена
|
||||
</button>
|
||||
<button type="submit" class="btn btn-primary" :disabled="isSubmitting">
|
||||
<i class="fas fa-save"></i>
|
||||
{{ isSubmitting ? 'Сохранение...' : 'Создать страницу' }}
|
||||
</button>
|
||||
</div>
|
||||
</form>
|
||||
</div>
|
||||
<router-view />
|
||||
<form class="content-form" @submit.prevent="handleSubmit">
|
||||
<div class="form-group">
|
||||
<label for="title">Заголовок страницы *</label>
|
||||
<input v-model="form.title" id="title" type="text" required />
|
||||
</div>
|
||||
<div class="form-group">
|
||||
<label for="summary">Краткое описание *</label>
|
||||
<textarea v-model="form.summary" id="summary" required rows="2" />
|
||||
</div>
|
||||
<div class="form-group">
|
||||
<label for="content">Основной контент *</label>
|
||||
<textarea v-model="form.content" id="content" required rows="6" />
|
||||
</div>
|
||||
<button class="submit-btn" type="submit">Сохранить</button>
|
||||
</form>
|
||||
</div>
|
||||
</BaseLayout>
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import { ref } from 'vue';
|
||||
import { ref, computed } from 'vue';
|
||||
import { useRouter } from 'vue-router';
|
||||
import BaseLayout from '../components/BaseLayout.vue';
|
||||
import pagesService from '../services/pagesService';
|
||||
|
||||
const router = useRouter();
|
||||
function goToCreate() { router.push({ name: 'content-create' }); }
|
||||
function goToList() { router.push({ name: 'content-list' }); }
|
||||
function goToSettings() { router.push({ name: 'content-settings' }); }
|
||||
// Props
|
||||
const props = defineProps({
|
||||
isAuthenticated: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
},
|
||||
identities: {
|
||||
type: Array,
|
||||
default: () => []
|
||||
},
|
||||
tokenBalances: {
|
||||
type: Object,
|
||||
default: () => ({})
|
||||
},
|
||||
isLoadingTokens: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
}
|
||||
});
|
||||
|
||||
// Emits
|
||||
const emit = defineEmits(['auth-action-completed']);
|
||||
|
||||
const router = useRouter();
|
||||
|
||||
// Состояние формы
|
||||
const form = ref({
|
||||
title: '',
|
||||
summary: '',
|
||||
content: ''
|
||||
content: '',
|
||||
seo: {
|
||||
title: '',
|
||||
description: '',
|
||||
keywords: ''
|
||||
},
|
||||
settings: {
|
||||
autoPublish: false
|
||||
},
|
||||
status: 'draft'
|
||||
});
|
||||
|
||||
const isSubmitting = ref(false);
|
||||
|
||||
// Вычисляемые свойства
|
||||
const wordCount = computed(() => {
|
||||
return form.value.content ? form.value.content.split(/\s+/).length : 0;
|
||||
});
|
||||
|
||||
const characterCount = computed(() => {
|
||||
return form.value.content ? form.value.content.length : 0;
|
||||
});
|
||||
|
||||
// Методы
|
||||
function goBack() {
|
||||
router.push({ name: 'content-list' });
|
||||
}
|
||||
|
||||
async function handleSubmit() {
|
||||
console.log('handleSubmit called', form.value);
|
||||
if (!form.value.title.trim()) {
|
||||
alert('Заполните заголовок страницы!');
|
||||
return;
|
||||
}
|
||||
|
||||
if (!form.value.summary.trim()) {
|
||||
alert('Заполните описание страницы!');
|
||||
return;
|
||||
}
|
||||
|
||||
if (!form.value.content.trim()) {
|
||||
alert('Заполните контент страницы!');
|
||||
return;
|
||||
}
|
||||
|
||||
try {
|
||||
if (!form.value.title) {
|
||||
alert('Заполните заголовок страницы!');
|
||||
return;
|
||||
}
|
||||
// Создаём страницу через pagesService
|
||||
const page = await pagesService.createPage({
|
||||
title: form.value.title,
|
||||
summary: form.value.summary,
|
||||
content: form.value.content
|
||||
});
|
||||
console.log('createPage result:', page);
|
||||
isSubmitting.value = true;
|
||||
|
||||
const pageData = {
|
||||
title: form.value.title.trim(),
|
||||
summary: form.value.summary.trim(),
|
||||
content: form.value.content.trim(),
|
||||
seo: form.value.seo,
|
||||
status: form.value.status,
|
||||
settings: form.value.settings
|
||||
};
|
||||
|
||||
const page = await pagesService.createPage(pageData);
|
||||
|
||||
if (!page || !page.id) {
|
||||
alert('Ошибка: страница не создана!');
|
||||
return;
|
||||
throw new Error('Страница не была создана');
|
||||
}
|
||||
|
||||
// Перенаправляем на список страниц
|
||||
router.push({ name: 'content-list' });
|
||||
} catch (e) {
|
||||
alert('Ошибка при создании страницы: ' + (e?.message || e));
|
||||
console.error('Ошибка при создании страницы:', e);
|
||||
} catch (error) {
|
||||
console.error('Ошибка при создании страницы:', error);
|
||||
alert('Ошибка при создании страницы: ' + (error?.message || error));
|
||||
} finally {
|
||||
isSubmitting.value = false;
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
.content-page-block {
|
||||
background: #fff;
|
||||
border-radius: 16px;
|
||||
box-shadow: 0 4px 32px rgba(0,0,0,0.12);
|
||||
padding: 32px 24px 24px 24px;
|
||||
width: 100%;
|
||||
margin-top: 40px;
|
||||
position: relative;
|
||||
overflow-x: auto;
|
||||
}
|
||||
.content-form {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 18px;
|
||||
margin-top: 24px;
|
||||
}
|
||||
.form-group {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 6px;
|
||||
}
|
||||
input[type="text"], textarea, select {
|
||||
border: 1px solid #d0d0d0;
|
||||
border-radius: 6px;
|
||||
padding: 8px 10px;
|
||||
font-size: 1rem;
|
||||
.content-create-page {
|
||||
padding: 20px;
|
||||
width: 100%;
|
||||
}
|
||||
.tags-input {
|
||||
|
||||
.page-header {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 6px;
|
||||
justify-content: space-between;
|
||||
align-items: flex-start;
|
||||
margin-bottom: 30px;
|
||||
padding-bottom: 20px;
|
||||
border-bottom: 2px solid #f0f0f0;
|
||||
}
|
||||
.tags-list {
|
||||
display: flex;
|
||||
flex-wrap: wrap;
|
||||
gap: 6px;
|
||||
|
||||
.header-content {
|
||||
flex: 1;
|
||||
}
|
||||
.tag {
|
||||
background: #f0f0f0;
|
||||
border-radius: 4px;
|
||||
padding: 2px 8px;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
font-size: 0.95em;
|
||||
|
||||
.header-content h1 {
|
||||
color: var(--color-primary);
|
||||
font-size: 2.5rem;
|
||||
margin: 0 0 10px 0;
|
||||
}
|
||||
.tag button {
|
||||
|
||||
.header-content p {
|
||||
color: var(--color-grey-dark);
|
||||
font-size: 1.1rem;
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
.close-btn {
|
||||
background: none;
|
||||
border: none;
|
||||
color: #888;
|
||||
margin-left: 4px;
|
||||
font-size: 2rem;
|
||||
color: var(--color-grey-dark);
|
||||
cursor: pointer;
|
||||
font-size: 1.1em;
|
||||
padding: 0 10px;
|
||||
transition: color 0.3s ease;
|
||||
}
|
||||
.submit-btn {
|
||||
background: #2d72d9;
|
||||
color: #fff;
|
||||
border: none;
|
||||
border-radius: 6px;
|
||||
padding: 10px 0;
|
||||
font-size: 1.1em;
|
||||
cursor: pointer;
|
||||
margin-top: 12px;
|
||||
transition: background 0.2s;
|
||||
|
||||
.close-btn:hover {
|
||||
color: var(--color-primary);
|
||||
}
|
||||
.submit-btn:hover {
|
||||
background: #1a4e96;
|
||||
|
||||
.content-block {
|
||||
background: #f8f9fa;
|
||||
border-radius: var(--radius-lg);
|
||||
padding: 25px;
|
||||
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.1);
|
||||
}
|
||||
.content-header-nav {
|
||||
|
||||
.content-form {
|
||||
background: white;
|
||||
border-radius: var(--radius-sm);
|
||||
padding: 30px;
|
||||
border: 1px solid #e9ecef;
|
||||
max-width: 1000px;
|
||||
margin: 0 auto;
|
||||
}
|
||||
|
||||
.form-section {
|
||||
margin-bottom: 30px;
|
||||
}
|
||||
|
||||
.form-section:last-child {
|
||||
margin-bottom: 0;
|
||||
}
|
||||
|
||||
.form-section h2 {
|
||||
color: var(--color-primary);
|
||||
margin: 0 0 20px 0;
|
||||
font-size: 1.3rem;
|
||||
border-bottom: 1px solid #f0f0f0;
|
||||
padding-bottom: 10px;
|
||||
}
|
||||
|
||||
.form-group {
|
||||
margin-bottom: 20px;
|
||||
}
|
||||
|
||||
.form-group:last-child {
|
||||
margin-bottom: 0;
|
||||
}
|
||||
|
||||
.form-group label {
|
||||
display: block;
|
||||
margin-bottom: 8px;
|
||||
font-weight: 500;
|
||||
color: var(--color-grey-dark);
|
||||
}
|
||||
|
||||
.form-input,
|
||||
.form-textarea,
|
||||
.form-select {
|
||||
width: 100%;
|
||||
padding: 12px 15px;
|
||||
border: 1px solid #e9ecef;
|
||||
border-radius: var(--radius-sm);
|
||||
font-size: 1rem;
|
||||
transition: border-color 0.3s ease;
|
||||
box-sizing: border-box;
|
||||
}
|
||||
|
||||
.form-input:focus,
|
||||
.form-textarea:focus,
|
||||
.form-select:focus {
|
||||
outline: none;
|
||||
border-color: var(--color-primary);
|
||||
box-shadow: 0 0 0 3px rgba(45, 114, 217, 0.1);
|
||||
}
|
||||
|
||||
.form-textarea {
|
||||
resize: vertical;
|
||||
min-height: 100px;
|
||||
}
|
||||
|
||||
.content-stats {
|
||||
display: flex;
|
||||
gap: 12px;
|
||||
margin-bottom: 18px;
|
||||
gap: 20px;
|
||||
margin-top: 8px;
|
||||
font-size: 0.9rem;
|
||||
color: var(--color-grey-dark);
|
||||
}
|
||||
.nav-btn {
|
||||
background: #f5f5f5;
|
||||
border: 1px solid #d0d0d0;
|
||||
border-radius: 6px;
|
||||
padding: 7px 18px;
|
||||
font-size: 1em;
|
||||
|
||||
.checkbox-label {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 10px;
|
||||
cursor: pointer;
|
||||
transition: background 0.2s;
|
||||
font-weight: normal;
|
||||
}
|
||||
.nav-btn:hover {
|
||||
background: #e0e0e0;
|
||||
|
||||
.form-checkbox {
|
||||
width: auto;
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
.form-actions {
|
||||
display: flex;
|
||||
gap: 15px;
|
||||
justify-content: flex-end;
|
||||
margin-top: 30px;
|
||||
padding-top: 20px;
|
||||
border-top: 1px solid #f0f0f0;
|
||||
}
|
||||
|
||||
.btn {
|
||||
padding: 12px 24px;
|
||||
border: none;
|
||||
border-radius: var(--radius-sm);
|
||||
font-size: 1rem;
|
||||
cursor: pointer;
|
||||
transition: all 0.3s ease;
|
||||
text-decoration: none;
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
gap: 8px;
|
||||
}
|
||||
|
||||
.btn:disabled {
|
||||
opacity: 0.6;
|
||||
cursor: not-allowed;
|
||||
}
|
||||
|
||||
.btn-primary {
|
||||
background: var(--color-primary);
|
||||
color: white;
|
||||
}
|
||||
|
||||
.btn-primary:hover:not(:disabled) {
|
||||
background: var(--color-primary-dark);
|
||||
}
|
||||
|
||||
.btn-outline {
|
||||
background: white;
|
||||
color: var(--color-primary);
|
||||
border: 1px solid var(--color-primary);
|
||||
}
|
||||
|
||||
.btn-outline:hover {
|
||||
background: var(--color-primary);
|
||||
color: white;
|
||||
}
|
||||
|
||||
@media (max-width: 768px) {
|
||||
.page-header {
|
||||
flex-direction: column;
|
||||
gap: 15px;
|
||||
}
|
||||
|
||||
.header-content h1 {
|
||||
font-size: 2rem;
|
||||
}
|
||||
|
||||
.form-actions {
|
||||
flex-direction: column;
|
||||
}
|
||||
|
||||
.content-stats {
|
||||
flex-direction: column;
|
||||
gap: 5px;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
@@ -11,84 +11,748 @@
|
||||
-->
|
||||
|
||||
<template>
|
||||
<BaseLayout>
|
||||
<div class="content-list-block">
|
||||
<div class="content-header-nav">
|
||||
<button class="nav-btn" @click="goToCreate">Создать</button>
|
||||
<button class="nav-btn" @click="goToList">Список страниц</button>
|
||||
<button class="nav-btn" @click="goToSettings">Настройки</button>
|
||||
<BaseLayout
|
||||
:is-authenticated="isAuthenticated"
|
||||
:identities="identities"
|
||||
:token-balances="tokenBalances"
|
||||
:is-loading-tokens="isLoadingTokens"
|
||||
@auth-action-completed="$emit('auth-action-completed')"
|
||||
>
|
||||
<div class="content-management-page">
|
||||
<!-- Заголовок страницы -->
|
||||
<div class="page-header">
|
||||
<div class="header-content">
|
||||
<h1>📄 Управление контентом</h1>
|
||||
<p>Создавайте и управляйте страницами вашего DLE</p>
|
||||
<button class="btn btn-primary" @click="goToCreate">
|
||||
<i class="fas fa-plus"></i>
|
||||
Создать страницу
|
||||
</button>
|
||||
</div>
|
||||
<div class="header-actions">
|
||||
<button class="close-btn" @click="goBack">×</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Основной контент с тенью -->
|
||||
<div class="content-block">
|
||||
<!-- Навигация -->
|
||||
<div class="content-navigation">
|
||||
<div class="nav-tabs">
|
||||
<button
|
||||
class="nav-tab"
|
||||
:class="{ active: activeTab === 'pages' }"
|
||||
@click="activeTab = 'pages'"
|
||||
>
|
||||
<i class="fas fa-file-alt"></i>
|
||||
Страницы
|
||||
</button>
|
||||
<button
|
||||
class="nav-tab"
|
||||
:class="{ active: activeTab === 'templates' }"
|
||||
@click="activeTab = 'templates'"
|
||||
>
|
||||
<i class="fas fa-layer-group"></i>
|
||||
Шаблоны
|
||||
</button>
|
||||
<button
|
||||
class="nav-tab"
|
||||
:class="{ active: activeTab === 'settings' }"
|
||||
@click="activeTab = 'settings'"
|
||||
>
|
||||
<i class="fas fa-cog"></i>
|
||||
Настройки
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Контент в зависимости от активной вкладки -->
|
||||
<div class="content-section">
|
||||
<!-- Вкладка Страницы -->
|
||||
<div v-if="activeTab === 'pages'" class="pages-section">
|
||||
<div class="section-header">
|
||||
<h2>Созданные страницы</h2>
|
||||
<div class="search-box">
|
||||
<input
|
||||
v-model="searchQuery"
|
||||
type="text"
|
||||
placeholder="Поиск страниц..."
|
||||
class="search-input"
|
||||
>
|
||||
<i class="fas fa-search search-icon"></i>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Список страниц -->
|
||||
<div v-if="filteredPages.length" class="pages-grid">
|
||||
<div
|
||||
v-for="page in filteredPages"
|
||||
:key="page.id"
|
||||
class="page-card"
|
||||
@click="goToPage(page.id)"
|
||||
>
|
||||
<div class="page-card-header">
|
||||
<h3>{{ page.title }}</h3>
|
||||
<div class="page-actions">
|
||||
<button
|
||||
class="action-btn edit-btn"
|
||||
@click.stop="goToEdit(page.id)"
|
||||
title="Редактировать"
|
||||
>
|
||||
<i class="fas fa-edit"></i>
|
||||
</button>
|
||||
<button
|
||||
class="action-btn delete-btn"
|
||||
@click.stop="deletePage(page.id)"
|
||||
title="Удалить"
|
||||
>
|
||||
<i class="fas fa-trash"></i>
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
<div class="page-card-content">
|
||||
<p class="page-summary">{{ page.summary || 'Без описания' }}</p>
|
||||
<div class="page-meta">
|
||||
<span class="page-date">
|
||||
<i class="fas fa-calendar"></i>
|
||||
{{ formatDate(page.createdAt) }}
|
||||
</span>
|
||||
<span class="page-status" :class="page.status">
|
||||
<i class="fas fa-circle"></i>
|
||||
{{ getStatusText(page.status) }}
|
||||
</span>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Пустое состояние -->
|
||||
<div v-else-if="!isLoading" class="empty-state">
|
||||
<div class="empty-icon">
|
||||
<i class="fas fa-file-alt"></i>
|
||||
</div>
|
||||
<h3>Нет созданных страниц</h3>
|
||||
<p>Создайте первую страницу для вашего DLE</p>
|
||||
<button class="btn btn-primary" @click="goToCreate">
|
||||
<i class="fas fa-plus"></i>
|
||||
Создать страницу
|
||||
</button>
|
||||
</div>
|
||||
|
||||
<!-- Загрузка -->
|
||||
<div v-else class="loading-state">
|
||||
<div class="loading-spinner"></div>
|
||||
<p>Загрузка страниц...</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Вкладка Шаблоны -->
|
||||
<div v-if="activeTab === 'templates'" class="templates-section">
|
||||
<div class="section-header">
|
||||
<h2>Шаблоны страниц</h2>
|
||||
<p>Готовые шаблоны для быстрого создания контента</p>
|
||||
</div>
|
||||
|
||||
<div class="templates-grid">
|
||||
<div
|
||||
v-for="template in templates"
|
||||
:key="template.id"
|
||||
class="template-card"
|
||||
@click="useTemplate(template)"
|
||||
>
|
||||
<div class="template-icon">
|
||||
<i :class="template.icon"></i>
|
||||
</div>
|
||||
<h3>{{ template.name }}</h3>
|
||||
<p>{{ template.description }}</p>
|
||||
<button class="btn btn-outline">Использовать шаблон</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Вкладка Настройки -->
|
||||
<div v-if="activeTab === 'settings'" class="settings-section">
|
||||
<div class="section-header">
|
||||
<h2>Настройки контента</h2>
|
||||
</div>
|
||||
|
||||
<div class="settings-grid">
|
||||
<div class="setting-card">
|
||||
<h3>SEO настройки</h3>
|
||||
<div class="setting-item">
|
||||
<label>Мета-теги по умолчанию</label>
|
||||
<textarea v-model="seoSettings.defaultMeta" placeholder="Введите мета-теги..."></textarea>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="setting-card">
|
||||
<h3>Настройки публикации</h3>
|
||||
<div class="setting-item">
|
||||
<label>
|
||||
<input type="checkbox" v-model="publishSettings.autoPublish">
|
||||
Автоматическая публикация
|
||||
</label>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
<h2>Список страниц</h2>
|
||||
<ul v-if="pages.length" class="pages-list">
|
||||
<li v-for="page in pages" :key="page.id">
|
||||
<router-link :to="{ name: 'page-view', params: { id: page.id } }">{{ page.title }}</router-link>
|
||||
</li>
|
||||
</ul>
|
||||
<div v-else class="empty-list-placeholder">Нет созданных страниц.</div>
|
||||
</div>
|
||||
</BaseLayout>
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import { onMounted, ref } from 'vue';
|
||||
import { ref, computed, onMounted } from 'vue';
|
||||
import { useRouter } from 'vue-router';
|
||||
import BaseLayout from '../../components/BaseLayout.vue';
|
||||
import pagesService from '../../services/pagesService';
|
||||
const router = useRouter();
|
||||
function goToCreate() { router.push({ name: 'content-create' }); }
|
||||
function goToList() { router.push({ name: 'content-list' }); }
|
||||
function goToSettings() { router.push({ name: 'content-settings' }); }
|
||||
|
||||
// Props
|
||||
const props = defineProps({
|
||||
isAuthenticated: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
},
|
||||
identities: {
|
||||
type: Array,
|
||||
default: () => []
|
||||
},
|
||||
tokenBalances: {
|
||||
type: Object,
|
||||
default: () => ({})
|
||||
},
|
||||
isLoadingTokens: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
}
|
||||
});
|
||||
|
||||
// Emits
|
||||
const emit = defineEmits(['auth-action-completed']);
|
||||
|
||||
const router = useRouter();
|
||||
|
||||
// Состояние
|
||||
const activeTab = ref('pages');
|
||||
const pages = ref([]);
|
||||
onMounted(async () => {
|
||||
pages.value = await pagesService.getPages();
|
||||
const isLoading = ref(false);
|
||||
const searchQuery = ref('');
|
||||
|
||||
// Настройки
|
||||
const seoSettings = ref({
|
||||
defaultMeta: ''
|
||||
});
|
||||
|
||||
const publishSettings = ref({
|
||||
autoPublish: false
|
||||
});
|
||||
|
||||
// Шаблоны
|
||||
const templates = ref([
|
||||
{
|
||||
id: 1,
|
||||
name: 'О компании',
|
||||
description: 'Стандартная страница с информацией о компании',
|
||||
icon: 'fas fa-building'
|
||||
},
|
||||
{
|
||||
id: 2,
|
||||
name: 'Услуги',
|
||||
description: 'Страница с описанием услуг и сервисов',
|
||||
icon: 'fas fa-cogs'
|
||||
},
|
||||
{
|
||||
id: 3,
|
||||
name: 'Контакты',
|
||||
description: 'Контактная информация и форма обратной связи',
|
||||
icon: 'fas fa-address-book'
|
||||
},
|
||||
{
|
||||
id: 4,
|
||||
name: 'Блог',
|
||||
description: 'Шаблон для ведения блога и новостей',
|
||||
icon: 'fas fa-blog'
|
||||
}
|
||||
]);
|
||||
|
||||
// Вычисляемые свойства
|
||||
const filteredPages = computed(() => {
|
||||
if (!searchQuery.value) return pages.value;
|
||||
return pages.value.filter(page =>
|
||||
page.title.toLowerCase().includes(searchQuery.value.toLowerCase()) ||
|
||||
page.summary?.toLowerCase().includes(searchQuery.value.toLowerCase())
|
||||
);
|
||||
});
|
||||
|
||||
// Методы
|
||||
function goToCreate() {
|
||||
router.push({ name: 'content-create' });
|
||||
}
|
||||
|
||||
function goBack() {
|
||||
router.go(-1);
|
||||
}
|
||||
|
||||
function goToPage(id) {
|
||||
router.push({ name: 'page-view', params: { id } });
|
||||
}
|
||||
|
||||
function goToEdit(id) {
|
||||
router.push({ name: 'page-edit', params: { id } });
|
||||
}
|
||||
|
||||
async function deletePage(id) {
|
||||
if (confirm('Вы уверены, что хотите удалить эту страницу?')) {
|
||||
try {
|
||||
await pagesService.deletePage(id);
|
||||
await loadPages();
|
||||
} catch (error) {
|
||||
console.error('Ошибка удаления страницы:', error);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
function useTemplate(template) {
|
||||
router.push({
|
||||
name: 'content-create',
|
||||
query: { template: template.id }
|
||||
});
|
||||
}
|
||||
|
||||
function formatDate(date) {
|
||||
if (!date) return 'Не указана';
|
||||
return new Date(date).toLocaleDateString('ru-RU');
|
||||
}
|
||||
|
||||
function getStatusText(status) {
|
||||
const statusMap = {
|
||||
draft: 'Черновик',
|
||||
published: 'Опубликовано',
|
||||
archived: 'Архив'
|
||||
};
|
||||
return statusMap[status] || 'Неизвестно';
|
||||
}
|
||||
|
||||
async function loadPages() {
|
||||
try {
|
||||
isLoading.value = true;
|
||||
pages.value = await pagesService.getPages();
|
||||
} catch (error) {
|
||||
console.error('Ошибка загрузки страниц:', error);
|
||||
pages.value = [];
|
||||
} finally {
|
||||
isLoading.value = false;
|
||||
}
|
||||
}
|
||||
|
||||
// Загрузка данных
|
||||
onMounted(() => {
|
||||
loadPages();
|
||||
});
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
.content-list-block {
|
||||
background: #fff;
|
||||
border-radius: 16px;
|
||||
box-shadow: 0 4px 32px rgba(0,0,0,0.12);
|
||||
padding: 32px 24px 24px 24px;
|
||||
.content-management-page {
|
||||
padding: 20px;
|
||||
width: 100%;
|
||||
margin-top: 40px;
|
||||
position: relative;
|
||||
overflow-x: auto;
|
||||
}
|
||||
.content-header-nav {
|
||||
|
||||
.page-header {
|
||||
display: flex;
|
||||
gap: 12px;
|
||||
margin-bottom: 18px;
|
||||
justify-content: space-between;
|
||||
align-items: flex-start;
|
||||
margin-bottom: 30px;
|
||||
padding-bottom: 20px;
|
||||
border-bottom: 2px solid #f0f0f0;
|
||||
}
|
||||
.nav-btn {
|
||||
background: #f5f5f5;
|
||||
border: 1px solid #d0d0d0;
|
||||
border-radius: 6px;
|
||||
padding: 7px 18px;
|
||||
font-size: 1em;
|
||||
|
||||
.header-content h1 {
|
||||
color: var(--color-primary);
|
||||
font-size: 2.5rem;
|
||||
margin: 0 0 10px 0;
|
||||
}
|
||||
|
||||
.header-content p {
|
||||
color: var(--color-grey-dark);
|
||||
font-size: 1.1rem;
|
||||
margin: 0 0 20px 0;
|
||||
}
|
||||
|
||||
.header-content .btn {
|
||||
margin-top: 10px;
|
||||
}
|
||||
|
||||
.content-navigation {
|
||||
margin-bottom: 30px;
|
||||
}
|
||||
|
||||
.nav-tabs {
|
||||
display: flex;
|
||||
gap: 10px;
|
||||
border-bottom: 1px solid #e9ecef;
|
||||
}
|
||||
|
||||
.nav-tab {
|
||||
background: none;
|
||||
border: none;
|
||||
padding: 15px 20px;
|
||||
font-size: 1rem;
|
||||
cursor: pointer;
|
||||
border-bottom: 3px solid transparent;
|
||||
transition: all 0.3s ease;
|
||||
color: var(--color-grey-dark);
|
||||
}
|
||||
|
||||
.nav-tab:hover {
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
.nav-tab.active {
|
||||
color: var(--color-primary);
|
||||
border-bottom-color: var(--color-primary);
|
||||
}
|
||||
|
||||
.nav-tab i {
|
||||
margin-right: 8px;
|
||||
}
|
||||
|
||||
.content-block {
|
||||
background: #f8f9fa;
|
||||
border-radius: var(--radius-lg);
|
||||
padding: 25px;
|
||||
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.1);
|
||||
}
|
||||
|
||||
.content-section {
|
||||
background: #f8f9fa;
|
||||
border-radius: var(--radius-lg);
|
||||
padding: 25px;
|
||||
}
|
||||
|
||||
.section-header {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: center;
|
||||
margin-bottom: 25px;
|
||||
}
|
||||
|
||||
.section-header h2 {
|
||||
color: var(--color-primary);
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
.search-box {
|
||||
position: relative;
|
||||
width: 300px;
|
||||
}
|
||||
|
||||
.search-input {
|
||||
width: 100%;
|
||||
padding: 10px 40px 10px 15px;
|
||||
border: 1px solid #e9ecef;
|
||||
border-radius: var(--radius-sm);
|
||||
font-size: 1rem;
|
||||
}
|
||||
|
||||
.search-icon {
|
||||
position: absolute;
|
||||
right: 15px;
|
||||
top: 50%;
|
||||
transform: translateY(-50%);
|
||||
color: var(--color-grey-dark);
|
||||
}
|
||||
|
||||
.pages-grid {
|
||||
display: grid;
|
||||
grid-template-columns: repeat(auto-fill, minmax(350px, 1fr));
|
||||
gap: 20px;
|
||||
margin-top: 20px;
|
||||
}
|
||||
|
||||
.page-card {
|
||||
background: white;
|
||||
border-radius: var(--radius-sm);
|
||||
padding: 20px;
|
||||
border: 1px solid #e9ecef;
|
||||
cursor: pointer;
|
||||
transition: all 0.3s ease;
|
||||
}
|
||||
|
||||
.page-card:hover {
|
||||
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.1);
|
||||
transform: translateY(-2px);
|
||||
}
|
||||
|
||||
.page-card-header {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: flex-start;
|
||||
margin-bottom: 15px;
|
||||
}
|
||||
|
||||
.page-card-header h3 {
|
||||
color: var(--color-primary);
|
||||
margin: 0;
|
||||
font-size: 1.2rem;
|
||||
}
|
||||
|
||||
.page-actions {
|
||||
display: flex;
|
||||
gap: 8px;
|
||||
}
|
||||
|
||||
.action-btn {
|
||||
background: none;
|
||||
border: none;
|
||||
padding: 5px;
|
||||
cursor: pointer;
|
||||
border-radius: var(--radius-sm);
|
||||
transition: background 0.2s;
|
||||
}
|
||||
.nav-btn:hover {
|
||||
background: #e0e0e0;
|
||||
|
||||
.edit-btn:hover {
|
||||
background: #e3f2fd;
|
||||
color: #2196f3;
|
||||
}
|
||||
.empty-list-placeholder {
|
||||
color: #888;
|
||||
font-size: 1.1em;
|
||||
margin-top: 2em;
|
||||
|
||||
.delete-btn:hover {
|
||||
background: #ffebee;
|
||||
color: #f44336;
|
||||
}
|
||||
.pages-list {
|
||||
margin-top: 1.5em;
|
||||
padding-left: 0;
|
||||
list-style: none;
|
||||
|
||||
.page-summary {
|
||||
color: var(--color-grey-dark);
|
||||
margin: 0 0 15px 0;
|
||||
line-height: 1.5;
|
||||
}
|
||||
.pages-list li {
|
||||
padding: 0.5em 0;
|
||||
border-bottom: 1px solid #f0f0f0;
|
||||
font-size: 1.08em;
|
||||
|
||||
.page-meta {
|
||||
display: flex;
|
||||
gap: 15px;
|
||||
font-size: 0.9rem;
|
||||
color: var(--color-grey-dark);
|
||||
}
|
||||
.pages-list li:last-child {
|
||||
border-bottom: none;
|
||||
|
||||
.page-date i,
|
||||
.page-status i {
|
||||
margin-right: 5px;
|
||||
}
|
||||
|
||||
.page-status.draft i {
|
||||
color: #ff9800;
|
||||
}
|
||||
|
||||
.page-status.published i {
|
||||
color: #4caf50;
|
||||
}
|
||||
|
||||
.page-status.archived i {
|
||||
color: #9e9e9e;
|
||||
}
|
||||
|
||||
.empty-state {
|
||||
text-align: center;
|
||||
padding: 60px 20px;
|
||||
}
|
||||
|
||||
.empty-icon {
|
||||
font-size: 4rem;
|
||||
color: var(--color-grey-dark);
|
||||
margin-bottom: 20px;
|
||||
}
|
||||
|
||||
.empty-state h3 {
|
||||
color: var(--color-primary);
|
||||
margin: 0 0 10px 0;
|
||||
}
|
||||
|
||||
.empty-state p {
|
||||
color: var(--color-grey-dark);
|
||||
margin: 0 0 25px 0;
|
||||
}
|
||||
|
||||
.loading-state {
|
||||
text-align: center;
|
||||
padding: 60px 20px;
|
||||
}
|
||||
|
||||
.loading-spinner {
|
||||
border: 3px solid #f3f3f3;
|
||||
border-top: 3px solid var(--color-primary);
|
||||
border-radius: 50%;
|
||||
width: 40px;
|
||||
height: 40px;
|
||||
animation: spin 1s linear infinite;
|
||||
margin: 0 auto 20px;
|
||||
}
|
||||
|
||||
@keyframes spin {
|
||||
0% { transform: rotate(0deg); }
|
||||
100% { transform: rotate(360deg); }
|
||||
}
|
||||
|
||||
.templates-grid {
|
||||
display: grid;
|
||||
grid-template-columns: repeat(auto-fill, minmax(300px, 1fr));
|
||||
gap: 20px;
|
||||
margin-top: 20px;
|
||||
}
|
||||
|
||||
.template-card {
|
||||
background: white;
|
||||
border-radius: var(--radius-sm);
|
||||
padding: 25px;
|
||||
text-align: center;
|
||||
border: 1px solid #e9ecef;
|
||||
cursor: pointer;
|
||||
transition: all 0.3s ease;
|
||||
}
|
||||
|
||||
.template-card:hover {
|
||||
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.1);
|
||||
transform: translateY(-2px);
|
||||
}
|
||||
|
||||
.template-icon {
|
||||
font-size: 3rem;
|
||||
color: var(--color-primary);
|
||||
margin-bottom: 15px;
|
||||
}
|
||||
|
||||
.template-card h3 {
|
||||
color: var(--color-primary);
|
||||
margin: 0 0 10px 0;
|
||||
}
|
||||
|
||||
.template-card p {
|
||||
color: var(--color-grey-dark);
|
||||
margin: 0 0 20px 0;
|
||||
line-height: 1.5;
|
||||
}
|
||||
|
||||
.settings-grid {
|
||||
display: grid;
|
||||
grid-template-columns: repeat(auto-fill, minmax(400px, 1fr));
|
||||
gap: 20px;
|
||||
margin-top: 20px;
|
||||
}
|
||||
|
||||
.setting-card {
|
||||
background: white;
|
||||
border-radius: var(--radius-sm);
|
||||
padding: 20px;
|
||||
border: 1px solid #e9ecef;
|
||||
}
|
||||
|
||||
.setting-card h3 {
|
||||
color: var(--color-primary);
|
||||
margin: 0 0 15px 0;
|
||||
}
|
||||
|
||||
.setting-item {
|
||||
margin-bottom: 15px;
|
||||
}
|
||||
|
||||
.setting-item label {
|
||||
display: block;
|
||||
margin-bottom: 8px;
|
||||
font-weight: 500;
|
||||
color: var(--color-grey-dark);
|
||||
}
|
||||
|
||||
.setting-item textarea {
|
||||
width: 100%;
|
||||
min-height: 100px;
|
||||
padding: 10px;
|
||||
border: 1px solid #e9ecef;
|
||||
border-radius: var(--radius-sm);
|
||||
resize: vertical;
|
||||
}
|
||||
|
||||
.setting-item input[type="checkbox"] {
|
||||
margin-right: 8px;
|
||||
}
|
||||
|
||||
.btn {
|
||||
padding: 10px 20px;
|
||||
border: none;
|
||||
border-radius: var(--radius-sm);
|
||||
font-size: 1rem;
|
||||
cursor: pointer;
|
||||
transition: all 0.3s ease;
|
||||
text-decoration: none;
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
gap: 8px;
|
||||
}
|
||||
|
||||
.btn-primary {
|
||||
background: var(--color-primary);
|
||||
color: white;
|
||||
}
|
||||
|
||||
.btn-primary:hover {
|
||||
background: var(--color-primary-dark);
|
||||
}
|
||||
|
||||
.btn-outline {
|
||||
background: white;
|
||||
color: var(--color-primary);
|
||||
border: 1px solid var(--color-primary);
|
||||
}
|
||||
|
||||
.btn-outline:hover {
|
||||
background: var(--color-primary);
|
||||
color: white;
|
||||
}
|
||||
|
||||
.close-btn {
|
||||
background: none;
|
||||
border: none;
|
||||
font-size: 1.5rem;
|
||||
cursor: pointer;
|
||||
color: var(--color-grey-dark);
|
||||
padding: 5px;
|
||||
border-radius: var(--radius-sm);
|
||||
transition: background 0.2s;
|
||||
}
|
||||
|
||||
.close-btn:hover {
|
||||
background: #f0f0f0;
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
@media (max-width: 768px) {
|
||||
.page-header {
|
||||
flex-direction: column;
|
||||
gap: 15px;
|
||||
}
|
||||
|
||||
.nav-tabs {
|
||||
flex-wrap: wrap;
|
||||
}
|
||||
|
||||
.section-header {
|
||||
flex-direction: column;
|
||||
gap: 15px;
|
||||
align-items: stretch;
|
||||
}
|
||||
|
||||
.search-box {
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
.pages-grid {
|
||||
grid-template-columns: 1fr;
|
||||
}
|
||||
|
||||
.templates-grid {
|
||||
grid-template-columns: 1fr;
|
||||
}
|
||||
|
||||
.settings-grid {
|
||||
grid-template-columns: 1fr;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
@@ -11,48 +11,474 @@
|
||||
-->
|
||||
|
||||
<template>
|
||||
<BaseLayout>
|
||||
<div v-if="page" class="page-edit-block">
|
||||
<h2>Редактировать страницу</h2>
|
||||
<form @submit.prevent="save">
|
||||
<label>Заголовок</label>
|
||||
<input v-model="page.title" required />
|
||||
<label>Описание</label>
|
||||
<textarea v-model="page.summary" />
|
||||
<label>Контент</label>
|
||||
<textarea v-model="page.content" />
|
||||
<button type="submit">Сохранить</button>
|
||||
<button type="button" @click="goBack">Отмена</button>
|
||||
</form>
|
||||
<BaseLayout
|
||||
:is-authenticated="isAuthenticated"
|
||||
:identities="identities"
|
||||
:token-balances="tokenBalances"
|
||||
:is-loading-tokens="isLoadingTokens"
|
||||
@auth-action-completed="$emit('auth-action-completed')"
|
||||
>
|
||||
<div v-if="page" class="page-edit-page">
|
||||
<!-- Заголовок страницы -->
|
||||
<div class="page-header">
|
||||
<div class="header-content">
|
||||
<h1>✏️ Редактирование страницы</h1>
|
||||
<p v-if="page">Редактируйте содержимое страницы "{{ page.title }}"</p>
|
||||
<p v-else>Загрузка страницы...</p>
|
||||
</div>
|
||||
<div class="header-actions">
|
||||
<button class="close-btn" @click="goBack">×</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Основной контент с тенью -->
|
||||
<div class="content-block">
|
||||
<form class="content-form" @submit.prevent="save">
|
||||
<!-- Основная информация -->
|
||||
<div class="form-section">
|
||||
<h2>Основная информация</h2>
|
||||
<div class="form-group">
|
||||
<label for="title">Заголовок страницы *</label>
|
||||
<input
|
||||
v-model="page.title"
|
||||
id="title"
|
||||
type="text"
|
||||
required
|
||||
placeholder="Введите заголовок страницы"
|
||||
class="form-input"
|
||||
/>
|
||||
</div>
|
||||
<div class="form-group">
|
||||
<label for="summary">Краткое описание *</label>
|
||||
<textarea
|
||||
v-model="page.summary"
|
||||
id="summary"
|
||||
required
|
||||
rows="3"
|
||||
placeholder="Краткое описание страницы"
|
||||
class="form-textarea"
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Контент -->
|
||||
<div class="form-section">
|
||||
<h2>Содержание</h2>
|
||||
<div class="form-group">
|
||||
<label for="content">Основной контент *</label>
|
||||
<textarea
|
||||
v-model="page.content"
|
||||
id="content"
|
||||
required
|
||||
rows="10"
|
||||
placeholder="Введите основной контент страницы"
|
||||
class="form-textarea"
|
||||
/>
|
||||
<div class="content-stats">
|
||||
<span>Слов: {{ wordCount }}</span>
|
||||
<span>Символов: {{ characterCount }}</span>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- SEO настройки -->
|
||||
<div class="form-section">
|
||||
<h2>SEO настройки</h2>
|
||||
<div class="form-group">
|
||||
<label for="seo-title">Meta Title</label>
|
||||
<input
|
||||
v-model="page.seo.title"
|
||||
id="seo-title"
|
||||
type="text"
|
||||
placeholder="SEO заголовок (если отличается от основного)"
|
||||
class="form-input"
|
||||
/>
|
||||
</div>
|
||||
<div class="form-group">
|
||||
<label for="seo-description">Meta Description</label>
|
||||
<textarea
|
||||
v-model="page.seo.description"
|
||||
id="seo-description"
|
||||
rows="3"
|
||||
placeholder="SEO описание для поисковых систем"
|
||||
class="form-textarea"
|
||||
/>
|
||||
</div>
|
||||
<div class="form-group">
|
||||
<label for="seo-keywords">Keywords</label>
|
||||
<input
|
||||
v-model="page.seo.keywords"
|
||||
id="seo-keywords"
|
||||
type="text"
|
||||
placeholder="Ключевые слова через запятую"
|
||||
class="form-input"
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Настройки публикации -->
|
||||
<div class="form-section">
|
||||
<h2>Настройки публикации</h2>
|
||||
<div class="form-group">
|
||||
<label for="status">Статус</label>
|
||||
<select v-model="page.status" id="status" class="form-select">
|
||||
<option value="draft">Черновик</option>
|
||||
<option value="published">Опубликовано</option>
|
||||
<option value="pending">На модерации</option>
|
||||
<option value="archived">Архив</option>
|
||||
</select>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Кнопки действий -->
|
||||
<div class="form-actions">
|
||||
<button type="button" class="btn btn-outline" @click="goBack">
|
||||
<i class="fas fa-times"></i>
|
||||
Отмена
|
||||
</button>
|
||||
<button type="submit" class="btn btn-primary" :disabled="isSubmitting">
|
||||
<i class="fas fa-save"></i>
|
||||
{{ isSubmitting ? 'Сохранение...' : 'Сохранить изменения' }}
|
||||
</button>
|
||||
</div>
|
||||
</form>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Загрузка -->
|
||||
<div v-else class="loading-state">
|
||||
<div class="loading-spinner"></div>
|
||||
<p>Загрузка страницы...</p>
|
||||
</div>
|
||||
<div v-else>Загрузка...</div>
|
||||
</BaseLayout>
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import { ref, onMounted, computed } from 'vue';
|
||||
import { ref, computed, onMounted } from 'vue';
|
||||
import { useRoute, useRouter } from 'vue-router';
|
||||
import BaseLayout from '../../components/BaseLayout.vue';
|
||||
import pagesService from '../../services/pagesService';
|
||||
|
||||
// Props
|
||||
const props = defineProps({
|
||||
isAuthenticated: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
},
|
||||
identities: {
|
||||
type: Array,
|
||||
default: () => []
|
||||
},
|
||||
tokenBalances: {
|
||||
type: Object,
|
||||
default: () => ({})
|
||||
},
|
||||
isLoadingTokens: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
}
|
||||
});
|
||||
|
||||
// Emits
|
||||
const emit = defineEmits(['auth-action-completed']);
|
||||
|
||||
const route = useRoute();
|
||||
const router = useRouter();
|
||||
const page = ref(null);
|
||||
const isSubmitting = ref(false);
|
||||
|
||||
onMounted(async () => {
|
||||
page.value = await pagesService.getPage(route.params.id);
|
||||
// Вычисляемые свойства
|
||||
const wordCount = computed(() => {
|
||||
return page.value?.content ? page.value.content.split(/\s+/).length : 0;
|
||||
});
|
||||
|
||||
async function save() {
|
||||
await pagesService.updatePage(route.params.id, {
|
||||
title: page.value.title,
|
||||
summary: page.value.summary,
|
||||
content: page.value.content
|
||||
});
|
||||
router.push({ name: 'page-view', params: { id: route.params.id } });
|
||||
}
|
||||
const characterCount = computed(() => {
|
||||
return page.value?.content ? page.value.content.length : 0;
|
||||
});
|
||||
|
||||
// Методы
|
||||
function goBack() {
|
||||
router.push({ name: 'page-view', params: { id: route.params.id } });
|
||||
}
|
||||
</script>
|
||||
|
||||
async function save() {
|
||||
if (!page.value.title.trim()) {
|
||||
alert('Заполните заголовок страницы!');
|
||||
return;
|
||||
}
|
||||
|
||||
if (!page.value.summary.trim()) {
|
||||
alert('Заполните описание страницы!');
|
||||
return;
|
||||
}
|
||||
|
||||
if (!page.value.content.trim()) {
|
||||
alert('Заполните контент страницы!');
|
||||
return;
|
||||
}
|
||||
|
||||
try {
|
||||
isSubmitting.value = true;
|
||||
|
||||
const pageData = {
|
||||
title: page.value.title.trim(),
|
||||
summary: page.value.summary.trim(),
|
||||
content: page.value.content.trim(),
|
||||
seo: page.value.seo || {},
|
||||
status: page.value.status || 'draft'
|
||||
};
|
||||
|
||||
await pagesService.updatePage(route.params.id, pageData);
|
||||
|
||||
// Перенаправляем на просмотр страницы
|
||||
router.push({ name: 'page-view', params: { id: route.params.id } });
|
||||
} catch (error) {
|
||||
console.error('Ошибка при сохранении страницы:', error);
|
||||
alert('Ошибка при сохранении страницы: ' + (error?.message || error));
|
||||
} finally {
|
||||
isSubmitting.value = false;
|
||||
}
|
||||
}
|
||||
|
||||
// Загрузка данных
|
||||
onMounted(async () => {
|
||||
try {
|
||||
page.value = await pagesService.getPage(route.params.id);
|
||||
|
||||
// Инициализируем SEO объект если его нет
|
||||
if (!page.value.seo) {
|
||||
page.value.seo = {
|
||||
title: '',
|
||||
description: '',
|
||||
keywords: ''
|
||||
};
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('Ошибка при загрузке страницы:', error);
|
||||
alert('Ошибка при загрузке страницы: ' + (error?.message || error));
|
||||
}
|
||||
});
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
.page-edit-page {
|
||||
padding: 20px;
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
.page-header {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: flex-start;
|
||||
margin-bottom: 30px;
|
||||
padding-bottom: 20px;
|
||||
border-bottom: 2px solid #f0f0f0;
|
||||
}
|
||||
|
||||
.header-content {
|
||||
flex: 1;
|
||||
}
|
||||
|
||||
.header-content h1 {
|
||||
color: var(--color-primary);
|
||||
font-size: 2.5rem;
|
||||
margin: 0 0 10px 0;
|
||||
}
|
||||
|
||||
.header-content p {
|
||||
color: var(--color-grey-dark);
|
||||
font-size: 1.1rem;
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
.close-btn {
|
||||
background: none;
|
||||
border: none;
|
||||
font-size: 2rem;
|
||||
color: var(--color-grey-dark);
|
||||
cursor: pointer;
|
||||
padding: 0 10px;
|
||||
transition: color 0.3s ease;
|
||||
}
|
||||
|
||||
.close-btn:hover {
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
.content-block {
|
||||
background: #f8f9fa;
|
||||
border-radius: var(--radius-lg);
|
||||
padding: 25px;
|
||||
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.1);
|
||||
}
|
||||
|
||||
.content-form {
|
||||
background: white;
|
||||
border-radius: var(--radius-sm);
|
||||
padding: 30px;
|
||||
border: 1px solid #e9ecef;
|
||||
max-width: 1000px;
|
||||
margin: 0 auto;
|
||||
}
|
||||
|
||||
.form-section {
|
||||
margin-bottom: 30px;
|
||||
}
|
||||
|
||||
.form-section:last-child {
|
||||
margin-bottom: 0;
|
||||
}
|
||||
|
||||
.form-section h2 {
|
||||
color: var(--color-primary);
|
||||
margin: 0 0 20px 0;
|
||||
font-size: 1.3rem;
|
||||
border-bottom: 1px solid #f0f0f0;
|
||||
padding-bottom: 10px;
|
||||
}
|
||||
|
||||
.form-group {
|
||||
margin-bottom: 20px;
|
||||
}
|
||||
|
||||
.form-group:last-child {
|
||||
margin-bottom: 0;
|
||||
}
|
||||
|
||||
.form-group label {
|
||||
display: block;
|
||||
margin-bottom: 8px;
|
||||
font-weight: 500;
|
||||
color: var(--color-grey-dark);
|
||||
}
|
||||
|
||||
.form-input,
|
||||
.form-textarea,
|
||||
.form-select {
|
||||
width: 100%;
|
||||
padding: 12px 15px;
|
||||
border: 1px solid #e9ecef;
|
||||
border-radius: var(--radius-sm);
|
||||
font-size: 1rem;
|
||||
transition: border-color 0.3s ease;
|
||||
box-sizing: border-box;
|
||||
}
|
||||
|
||||
.form-input:focus,
|
||||
.form-textarea:focus,
|
||||
.form-select:focus {
|
||||
outline: none;
|
||||
border-color: var(--color-primary);
|
||||
box-shadow: 0 0 0 3px rgba(45, 114, 217, 0.1);
|
||||
}
|
||||
|
||||
.form-textarea {
|
||||
resize: vertical;
|
||||
min-height: 100px;
|
||||
}
|
||||
|
||||
.content-stats {
|
||||
display: flex;
|
||||
gap: 20px;
|
||||
margin-top: 8px;
|
||||
font-size: 0.9rem;
|
||||
color: var(--color-grey-dark);
|
||||
}
|
||||
|
||||
.form-actions {
|
||||
display: flex;
|
||||
gap: 15px;
|
||||
justify-content: flex-end;
|
||||
margin-top: 30px;
|
||||
padding-top: 20px;
|
||||
border-top: 1px solid #f0f0f0;
|
||||
}
|
||||
|
||||
.btn {
|
||||
padding: 12px 24px;
|
||||
border: none;
|
||||
border-radius: var(--radius-sm);
|
||||
font-size: 1rem;
|
||||
cursor: pointer;
|
||||
transition: all 0.3s ease;
|
||||
text-decoration: none;
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
gap: 8px;
|
||||
}
|
||||
|
||||
.btn:disabled {
|
||||
opacity: 0.6;
|
||||
cursor: not-allowed;
|
||||
}
|
||||
|
||||
.btn-primary {
|
||||
background: var(--color-primary);
|
||||
color: white;
|
||||
}
|
||||
|
||||
.btn-primary:hover:not(:disabled) {
|
||||
background: var(--color-primary-dark);
|
||||
}
|
||||
|
||||
.btn-outline {
|
||||
background: white;
|
||||
color: var(--color-primary);
|
||||
border: 1px solid var(--color-primary);
|
||||
}
|
||||
|
||||
.btn-outline:hover {
|
||||
background: var(--color-primary);
|
||||
color: white;
|
||||
}
|
||||
|
||||
.loading-state {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
padding: 60px 20px;
|
||||
text-align: center;
|
||||
}
|
||||
|
||||
.loading-spinner {
|
||||
width: 40px;
|
||||
height: 40px;
|
||||
border: 4px solid #f3f3f3;
|
||||
border-top: 4px solid var(--color-primary);
|
||||
border-radius: 50%;
|
||||
animation: spin 1s linear infinite;
|
||||
margin-bottom: 20px;
|
||||
}
|
||||
|
||||
@keyframes spin {
|
||||
0% { transform: rotate(0deg); }
|
||||
100% { transform: rotate(360deg); }
|
||||
}
|
||||
|
||||
.loading-state p {
|
||||
color: var(--color-grey-dark);
|
||||
font-size: 1.1rem;
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
@media (max-width: 768px) {
|
||||
.page-header {
|
||||
flex-direction: column;
|
||||
gap: 15px;
|
||||
}
|
||||
|
||||
.header-content h1 {
|
||||
font-size: 2rem;
|
||||
}
|
||||
|
||||
.form-actions {
|
||||
flex-direction: column;
|
||||
}
|
||||
|
||||
.content-stats {
|
||||
flex-direction: column;
|
||||
gap: 5px;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
@@ -11,15 +11,128 @@
|
||||
-->
|
||||
|
||||
<template>
|
||||
<BaseLayout>
|
||||
<div v-if="page" class="page-view-block">
|
||||
<h2>{{ page.title }}</h2>
|
||||
<p><b>Описание:</b> {{ page.summary }}</p>
|
||||
<div><b>Контент:</b> {{ page.content }}</div>
|
||||
<button @click="goToEdit">Редактировать</button>
|
||||
<button @click="deletePage" style="color:red">Удалить</button>
|
||||
<BaseLayout
|
||||
:is-authenticated="isAuthenticated"
|
||||
:identities="identities"
|
||||
:token-balances="tokenBalances"
|
||||
:is-loading-tokens="isLoadingTokens"
|
||||
@auth-action-completed="$emit('auth-action-completed')"
|
||||
>
|
||||
<div class="page-view-container">
|
||||
<!-- Заголовок страницы -->
|
||||
<div v-if="page" class="page-header">
|
||||
<div class="header-content">
|
||||
<h1>📄 {{ page.title }}</h1>
|
||||
<div class="page-meta">
|
||||
<span class="page-status" :class="page.status">
|
||||
<i class="fas fa-circle"></i>
|
||||
{{ getStatusText(page.status) }}
|
||||
</span>
|
||||
<span class="page-date">
|
||||
<i class="fas fa-calendar"></i>
|
||||
Создано: {{ formatDate(page.createdAt) }}
|
||||
</span>
|
||||
<span v-if="page.updatedAt" class="page-date">
|
||||
<i class="fas fa-edit"></i>
|
||||
Обновлено: {{ formatDate(page.updatedAt) }}
|
||||
</span>
|
||||
</div>
|
||||
</div>
|
||||
<div class="header-actions">
|
||||
<button class="btn btn-outline" @click="goToEdit">
|
||||
<i class="fas fa-edit"></i>
|
||||
Редактировать
|
||||
</button>
|
||||
<button class="btn btn-danger" @click="deletePage">
|
||||
<i class="fas fa-trash"></i>
|
||||
Удалить
|
||||
</button>
|
||||
<button class="close-btn" @click="goBack">×</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Контент страницы -->
|
||||
<div v-if="page" class="page-content-block">
|
||||
<div class="page-content">
|
||||
<!-- Описание -->
|
||||
<div v-if="page.summary" class="content-section">
|
||||
<h2>Описание</h2>
|
||||
<div class="summary-content">
|
||||
{{ page.summary }}
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Основной контент -->
|
||||
<div class="content-section">
|
||||
<h2>Содержание</h2>
|
||||
<div class="main-content">
|
||||
<div v-if="page.content" v-html="formatContent(page.content)"></div>
|
||||
<div v-else class="empty-content">
|
||||
<i class="fas fa-file-alt"></i>
|
||||
<p>Контент не добавлен</p>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- SEO информация -->
|
||||
<div v-if="page.seo" class="content-section">
|
||||
<h2>SEO информация</h2>
|
||||
<div class="seo-info">
|
||||
<div class="seo-item">
|
||||
<label>Meta Title:</label>
|
||||
<span>{{ page.seo.title || 'Не указан' }}</span>
|
||||
</div>
|
||||
<div class="seo-item">
|
||||
<label>Meta Description:</label>
|
||||
<span>{{ page.seo.description || 'Не указан' }}</span>
|
||||
</div>
|
||||
<div class="seo-item">
|
||||
<label>Keywords:</label>
|
||||
<span>{{ page.seo.keywords || 'Не указаны' }}</span>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Статистика -->
|
||||
<div class="content-section">
|
||||
<h2>Статистика</h2>
|
||||
<div class="stats-grid">
|
||||
<div class="stat-item">
|
||||
<div class="stat-value">{{ page.views || 0 }}</div>
|
||||
<div class="stat-label">Просмотров</div>
|
||||
</div>
|
||||
<div class="stat-item">
|
||||
<div class="stat-value">{{ page.wordCount || 0 }}</div>
|
||||
<div class="stat-label">Слов</div>
|
||||
</div>
|
||||
<div class="stat-item">
|
||||
<div class="stat-value">{{ page.characterCount || 0 }}</div>
|
||||
<div class="stat-label">Символов</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Загрузка -->
|
||||
<div v-else-if="isLoading" class="loading-state">
|
||||
<div class="loading-spinner"></div>
|
||||
<p>Загрузка страницы...</p>
|
||||
</div>
|
||||
|
||||
<!-- Ошибка -->
|
||||
<div v-else class="error-state">
|
||||
<div class="error-icon">
|
||||
<i class="fas fa-exclamation-triangle"></i>
|
||||
</div>
|
||||
<h3>Страница не найдена</h3>
|
||||
<p>Запрашиваемая страница не существует или была удалена</p>
|
||||
<button class="btn btn-primary" @click="goBack">
|
||||
<i class="fas fa-arrow-left"></i>
|
||||
Вернуться назад
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
<div v-else>Загрузка...</div>
|
||||
</BaseLayout>
|
||||
</template>
|
||||
|
||||
@@ -29,22 +142,427 @@ import { useRoute, useRouter } from 'vue-router';
|
||||
import BaseLayout from '../../components/BaseLayout.vue';
|
||||
import pagesService from '../../services/pagesService';
|
||||
|
||||
const route = useRoute();
|
||||
const router = useRouter();
|
||||
const page = ref(null);
|
||||
|
||||
onMounted(async () => {
|
||||
page.value = await pagesService.getPage(route.params.id);
|
||||
// Props
|
||||
const props = defineProps({
|
||||
isAuthenticated: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
},
|
||||
identities: {
|
||||
type: Array,
|
||||
default: () => []
|
||||
},
|
||||
tokenBalances: {
|
||||
type: Object,
|
||||
default: () => ({})
|
||||
},
|
||||
isLoadingTokens: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
}
|
||||
});
|
||||
|
||||
// Emits
|
||||
const emit = defineEmits(['auth-action-completed']);
|
||||
|
||||
const route = useRoute();
|
||||
const router = useRouter();
|
||||
|
||||
// Состояние
|
||||
const page = ref(null);
|
||||
const isLoading = ref(false);
|
||||
|
||||
// Методы
|
||||
function goToEdit() {
|
||||
router.push({ name: 'page-edit', params: { id: route.params.id } });
|
||||
}
|
||||
|
||||
async function deletePage() {
|
||||
if (confirm('Удалить страницу?')) {
|
||||
await pagesService.deletePage(route.params.id);
|
||||
router.push({ name: 'content-list' });
|
||||
if (confirm('Вы уверены, что хотите удалить эту страницу? Это действие нельзя отменить.')) {
|
||||
try {
|
||||
await pagesService.deletePage(route.params.id);
|
||||
router.push({ name: 'content-list' });
|
||||
} catch (error) {
|
||||
console.error('Ошибка удаления страницы:', error);
|
||||
alert('Ошибка при удалении страницы');
|
||||
}
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
function goBack() {
|
||||
router.go(-1);
|
||||
}
|
||||
|
||||
function formatDate(date) {
|
||||
if (!date) return 'Не указана';
|
||||
return new Date(date).toLocaleDateString('ru-RU', {
|
||||
year: 'numeric',
|
||||
month: 'long',
|
||||
day: 'numeric',
|
||||
hour: '2-digit',
|
||||
minute: '2-digit'
|
||||
});
|
||||
}
|
||||
|
||||
function getStatusText(status) {
|
||||
const statusMap = {
|
||||
draft: 'Черновик',
|
||||
published: 'Опубликовано',
|
||||
archived: 'Архив',
|
||||
pending: 'На модерации'
|
||||
};
|
||||
return statusMap[status] || 'Неизвестно';
|
||||
}
|
||||
|
||||
function formatContent(content) {
|
||||
// Простое форматирование контента
|
||||
if (!content) return '';
|
||||
|
||||
// Заменяем переносы строк на <br>
|
||||
return content.replace(/\n/g, '<br>');
|
||||
}
|
||||
|
||||
async function loadPage() {
|
||||
try {
|
||||
isLoading.value = true;
|
||||
page.value = await pagesService.getPage(route.params.id);
|
||||
|
||||
// Подсчитываем статистику
|
||||
if (page.value) {
|
||||
page.value.wordCount = page.value.content ? page.value.content.split(/\s+/).length : 0;
|
||||
page.value.characterCount = page.value.content ? page.value.content.length : 0;
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('Ошибка загрузки страницы:', error);
|
||||
page.value = null;
|
||||
} finally {
|
||||
isLoading.value = false;
|
||||
}
|
||||
}
|
||||
|
||||
// Загрузка данных
|
||||
onMounted(() => {
|
||||
loadPage();
|
||||
});
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
.page-view-container {
|
||||
padding: 20px;
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
.page-header {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: flex-start;
|
||||
margin-bottom: 30px;
|
||||
padding-bottom: 20px;
|
||||
border-bottom: 2px solid #f0f0f0;
|
||||
}
|
||||
|
||||
.header-content {
|
||||
flex: 1;
|
||||
}
|
||||
|
||||
.header-content h1 {
|
||||
color: var(--color-primary);
|
||||
font-size: 2.5rem;
|
||||
margin: 0 0 15px 0;
|
||||
}
|
||||
|
||||
.page-meta {
|
||||
display: flex;
|
||||
gap: 20px;
|
||||
align-items: center;
|
||||
font-size: 0.9rem;
|
||||
color: var(--color-grey-dark);
|
||||
}
|
||||
|
||||
.page-status {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 5px;
|
||||
padding: 4px 12px;
|
||||
border-radius: 20px;
|
||||
font-weight: 500;
|
||||
}
|
||||
|
||||
.page-status.draft {
|
||||
background: #fff3e0;
|
||||
color: #e65100;
|
||||
}
|
||||
|
||||
.page-status.published {
|
||||
background: #e8f5e8;
|
||||
color: #2e7d32;
|
||||
}
|
||||
|
||||
.page-status.archived {
|
||||
background: #f5f5f5;
|
||||
color: #616161;
|
||||
}
|
||||
|
||||
.page-status.pending {
|
||||
background: #e3f2fd;
|
||||
color: #1565c0;
|
||||
}
|
||||
|
||||
.page-date,
|
||||
.page-updated {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 5px;
|
||||
}
|
||||
|
||||
.header-actions {
|
||||
display: flex;
|
||||
gap: 10px;
|
||||
}
|
||||
|
||||
.page-content-block {
|
||||
background: #f8f9fa;
|
||||
border-radius: var(--radius-lg);
|
||||
padding: 25px;
|
||||
box-shadow: 0 4px 15px rgba(0, 0, 0, 0.08);
|
||||
}
|
||||
|
||||
.page-content {
|
||||
background: white;
|
||||
border-radius: var(--radius-sm);
|
||||
padding: 25px;
|
||||
border: 1px solid #e9ecef;
|
||||
}
|
||||
|
||||
.content-section {
|
||||
background: white;
|
||||
border-radius: var(--radius-sm);
|
||||
padding: 25px;
|
||||
margin-bottom: 20px;
|
||||
border: 1px solid #e9ecef;
|
||||
}
|
||||
|
||||
.content-section:last-child {
|
||||
margin-bottom: 0;
|
||||
}
|
||||
|
||||
.content-section h2 {
|
||||
color: var(--color-primary);
|
||||
margin: 0 0 20px 0;
|
||||
font-size: 1.5rem;
|
||||
}
|
||||
|
||||
.summary-content {
|
||||
color: var(--color-grey-dark);
|
||||
line-height: 1.6;
|
||||
font-size: 1.1rem;
|
||||
}
|
||||
|
||||
.main-content {
|
||||
line-height: 1.8;
|
||||
color: #333;
|
||||
}
|
||||
|
||||
.empty-content {
|
||||
text-align: center;
|
||||
padding: 40px 20px;
|
||||
color: var(--color-grey-dark);
|
||||
}
|
||||
|
||||
.empty-content i {
|
||||
font-size: 3rem;
|
||||
margin-bottom: 15px;
|
||||
opacity: 0.5;
|
||||
}
|
||||
|
||||
.seo-info {
|
||||
display: grid;
|
||||
gap: 15px;
|
||||
}
|
||||
|
||||
.seo-item {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: flex-start;
|
||||
padding: 10px 0;
|
||||
border-bottom: 1px solid #f0f0f0;
|
||||
}
|
||||
|
||||
.seo-item:last-child {
|
||||
border-bottom: none;
|
||||
}
|
||||
|
||||
.seo-item label {
|
||||
font-weight: 500;
|
||||
color: var(--color-grey-dark);
|
||||
min-width: 150px;
|
||||
}
|
||||
|
||||
.seo-item span {
|
||||
color: #333;
|
||||
flex: 1;
|
||||
margin-left: 20px;
|
||||
}
|
||||
|
||||
.stats-grid {
|
||||
display: grid;
|
||||
grid-template-columns: repeat(auto-fit, minmax(150px, 1fr));
|
||||
gap: 20px;
|
||||
}
|
||||
|
||||
.stat-item {
|
||||
text-align: center;
|
||||
padding: 20px;
|
||||
background: #f8f9fa;
|
||||
border-radius: var(--radius-sm);
|
||||
}
|
||||
|
||||
.stat-value {
|
||||
font-size: 2rem;
|
||||
font-weight: bold;
|
||||
color: var(--color-primary);
|
||||
margin-bottom: 5px;
|
||||
}
|
||||
|
||||
.stat-label {
|
||||
color: var(--color-grey-dark);
|
||||
font-size: 0.9rem;
|
||||
}
|
||||
|
||||
.loading-state,
|
||||
.error-state {
|
||||
text-align: center;
|
||||
padding: 60px 20px;
|
||||
}
|
||||
|
||||
.loading-spinner {
|
||||
border: 3px solid #f3f3f3;
|
||||
border-top: 3px solid var(--color-primary);
|
||||
border-radius: 50%;
|
||||
width: 40px;
|
||||
height: 40px;
|
||||
animation: spin 1s linear infinite;
|
||||
margin: 0 auto 20px;
|
||||
}
|
||||
|
||||
@keyframes spin {
|
||||
0% { transform: rotate(0deg); }
|
||||
100% { transform: rotate(360deg); }
|
||||
}
|
||||
|
||||
.error-icon {
|
||||
font-size: 4rem;
|
||||
color: #f44336;
|
||||
margin-bottom: 20px;
|
||||
}
|
||||
|
||||
.error-state h3 {
|
||||
color: var(--color-primary);
|
||||
margin: 0 0 10px 0;
|
||||
}
|
||||
|
||||
.error-state p {
|
||||
color: var(--color-grey-dark);
|
||||
margin: 0 0 25px 0;
|
||||
}
|
||||
|
||||
.btn {
|
||||
padding: 10px 20px;
|
||||
border: none;
|
||||
border-radius: var(--radius-sm);
|
||||
font-size: 1rem;
|
||||
cursor: pointer;
|
||||
transition: all 0.3s ease;
|
||||
text-decoration: none;
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
gap: 8px;
|
||||
}
|
||||
|
||||
.btn-outline {
|
||||
background: white;
|
||||
color: var(--color-primary);
|
||||
border: 1px solid var(--color-primary);
|
||||
}
|
||||
|
||||
.btn-outline:hover {
|
||||
background: var(--color-primary);
|
||||
color: white;
|
||||
}
|
||||
|
||||
.btn-danger {
|
||||
background: #f44336;
|
||||
color: white;
|
||||
}
|
||||
|
||||
.btn-danger:hover {
|
||||
background: #d32f2f;
|
||||
}
|
||||
|
||||
.btn-primary {
|
||||
background: var(--color-primary);
|
||||
color: white;
|
||||
}
|
||||
|
||||
.btn-primary:hover {
|
||||
background: var(--color-primary-dark);
|
||||
}
|
||||
|
||||
.close-btn {
|
||||
background: none;
|
||||
border: none;
|
||||
font-size: 2rem;
|
||||
color: var(--color-grey-dark);
|
||||
cursor: pointer;
|
||||
padding: 0 10px;
|
||||
transition: color 0.3s ease;
|
||||
}
|
||||
|
||||
.close-btn:hover {
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
@media (max-width: 768px) {
|
||||
.page-header {
|
||||
flex-direction: column;
|
||||
gap: 15px;
|
||||
}
|
||||
|
||||
.page-title-section h1 {
|
||||
font-size: 2rem;
|
||||
}
|
||||
|
||||
.page-meta {
|
||||
flex-direction: column;
|
||||
gap: 10px;
|
||||
align-items: flex-start;
|
||||
}
|
||||
|
||||
.header-actions {
|
||||
width: 100%;
|
||||
justify-content: stretch;
|
||||
}
|
||||
|
||||
.header-actions .btn {
|
||||
flex: 1;
|
||||
justify-content: center;
|
||||
}
|
||||
|
||||
.seo-item {
|
||||
flex-direction: column;
|
||||
gap: 5px;
|
||||
}
|
||||
|
||||
.seo-item label {
|
||||
min-width: auto;
|
||||
}
|
||||
|
||||
.seo-item span {
|
||||
margin-left: 0;
|
||||
}
|
||||
|
||||
.stats-grid {
|
||||
grid-template-columns: 1fr;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
@@ -48,6 +48,332 @@
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Блоки для деплоя стандартных модулей -->
|
||||
<div class="standard-modules">
|
||||
<div class="modules-header">
|
||||
<h3>🚀 Деплой стандартных модулей</h3>
|
||||
<p>Быстрый деплой предустановленных модулей DLE</p>
|
||||
</div>
|
||||
|
||||
<div class="modules-grid">
|
||||
<!-- TreasuryModule -->
|
||||
<div class="module-deploy-card">
|
||||
<div class="module-content">
|
||||
<h4>TreasuryModule</h4>
|
||||
<p>Казначейство DLE - управление финансами, депозиты, выводы, дивиденды</p>
|
||||
<div class="module-features">
|
||||
<span class="feature-tag">Финансы</span>
|
||||
<span class="feature-tag">Бюджет</span>
|
||||
<span class="feature-tag">Дивиденды</span>
|
||||
</div>
|
||||
</div>
|
||||
<div class="module-actions">
|
||||
<button
|
||||
class="btn btn-primary btn-deploy"
|
||||
@click="router.push(`/management/modules/deploy/treasury?address=${route.query.address}`)"
|
||||
>
|
||||
<i class="fas fa-rocket"></i>
|
||||
Деплой
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- TimelockModule -->
|
||||
<div class="module-deploy-card">
|
||||
<div class="module-content">
|
||||
<h4>TimelockModule</h4>
|
||||
<p>Задержки исполнения - безопасность критических операций через таймлоки</p>
|
||||
<div class="module-features">
|
||||
<span class="feature-tag">Безопасность</span>
|
||||
<span class="feature-tag">Таймлок</span>
|
||||
<span class="feature-tag">Аудит</span>
|
||||
</div>
|
||||
</div>
|
||||
<div class="module-actions">
|
||||
<button
|
||||
class="btn btn-primary btn-deploy"
|
||||
@click="router.push(`/management/modules/deploy/timelock?address=${route.query.address}`)"
|
||||
>
|
||||
<i class="fas fa-rocket"></i>
|
||||
Деплой
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- CommunicationModule -->
|
||||
<div class="module-deploy-card">
|
||||
<div class="module-content">
|
||||
<h4>CommunicationModule</h4>
|
||||
<p>Коммуникации - сообщения, звонки, история общения между участниками</p>
|
||||
<div class="module-features">
|
||||
<span class="feature-tag">Сообщения</span>
|
||||
<span class="feature-tag">Звонки</span>
|
||||
<span class="feature-tag">История</span>
|
||||
</div>
|
||||
</div>
|
||||
<div class="module-actions">
|
||||
<button
|
||||
class="btn btn-primary btn-deploy"
|
||||
@click="router.push(`/management/modules/deploy/communication?address=${route.query.address}`)"
|
||||
>
|
||||
<i class="fas fa-rocket"></i>
|
||||
Деплой
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- ApplicationModule -->
|
||||
<div class="module-deploy-card">
|
||||
<div class="module-content">
|
||||
<h4>ApplicationModule</h4>
|
||||
<p>Управление вызовом функций приложения через предложения и голосование</p>
|
||||
<div class="module-features">
|
||||
<span class="feature-tag">API</span>
|
||||
<span class="feature-tag">Голосование</span>
|
||||
<span class="feature-tag">Управление</span>
|
||||
</div>
|
||||
</div>
|
||||
<div class="module-actions">
|
||||
<button
|
||||
class="btn btn-primary btn-deploy"
|
||||
@click="router.push(`/management/modules/deploy/application?address=${route.query.address}`)"
|
||||
>
|
||||
<i class="fas fa-rocket"></i>
|
||||
Деплой
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- MintModule -->
|
||||
<div class="module-deploy-card">
|
||||
<div class="module-content">
|
||||
<h4>MintModule</h4>
|
||||
<p>Выпуск новых токенов DLE - создание дополнительных токенов через governance</p>
|
||||
<div class="module-features">
|
||||
<span class="feature-tag">Минтинг</span>
|
||||
<span class="feature-tag">Токены</span>
|
||||
<span class="feature-tag">Governance</span>
|
||||
</div>
|
||||
</div>
|
||||
<div class="module-actions">
|
||||
<button
|
||||
class="btn btn-primary btn-deploy"
|
||||
@click="router.push(`/management/modules/deploy/mint?address=${route.query.address}`)"
|
||||
>
|
||||
<i class="fas fa-rocket"></i>
|
||||
Деплой
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- BurnModule -->
|
||||
<div class="module-deploy-card">
|
||||
<div class="module-content">
|
||||
<h4>BurnModule</h4>
|
||||
<p>Сжигание токенов DLE - уменьшение общего предложения через governance</p>
|
||||
<div class="module-features">
|
||||
<span class="feature-tag">Сжигание</span>
|
||||
<span class="feature-tag">Токены</span>
|
||||
<span class="feature-tag">Governance</span>
|
||||
</div>
|
||||
</div>
|
||||
<div class="module-actions">
|
||||
<button
|
||||
class="btn btn-primary btn-deploy"
|
||||
@click="router.push(`/management/modules/deploy/burn?address=${route.query.address}`)"
|
||||
>
|
||||
<i class="fas fa-rocket"></i>
|
||||
Деплой
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- OracleModule -->
|
||||
<div class="module-deploy-card">
|
||||
<div class="module-content">
|
||||
<h4>OracleModule</h4>
|
||||
<p>Интеграция с внешними данными - автоматизация на основе IoT, API, датчиков</p>
|
||||
<div class="module-features">
|
||||
<span class="feature-tag">Оракулы</span>
|
||||
<span class="feature-tag">Автоматизация</span>
|
||||
<span class="feature-tag">IoT</span>
|
||||
<span class="feature-tag">API</span>
|
||||
</div>
|
||||
</div>
|
||||
<div class="module-actions">
|
||||
<button
|
||||
class="btn btn-primary btn-deploy"
|
||||
@click="router.push(`/management/modules/deploy/oracle?address=${route.query.address}`)"
|
||||
>
|
||||
<i class="fas fa-rocket"></i>
|
||||
Деплой
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- InheritanceModule -->
|
||||
<div class="module-deploy-card">
|
||||
<div class="module-content">
|
||||
<h4>InheritanceModule</h4>
|
||||
<p>Наследование токенов - автоматическая передача токенов наследникам</p>
|
||||
<div class="module-features">
|
||||
<span class="feature-tag">Наследование</span>
|
||||
<span class="feature-tag">Безопасность</span>
|
||||
<span class="feature-tag">Юридические</span>
|
||||
<span class="feature-tag">Автоматизация</span>
|
||||
</div>
|
||||
</div>
|
||||
<div class="module-actions">
|
||||
<button
|
||||
class="btn btn-primary btn-deploy"
|
||||
@click="router.push(`/management/modules/deploy/inheritance?address=${route.query.address}`)"
|
||||
>
|
||||
<i class="fas fa-rocket"></i>
|
||||
Деплой
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- VestingModule -->
|
||||
<div class="module-deploy-card">
|
||||
<div class="module-content">
|
||||
<h4>VestingModule</h4>
|
||||
<p>Вестинг токенов - постепенное разблокирование токенов по расписанию</p>
|
||||
<div class="module-features">
|
||||
<span class="feature-tag">Вестинг</span>
|
||||
<span class="feature-tag">Мотивация</span>
|
||||
<span class="feature-tag">Удержание</span>
|
||||
<span class="feature-tag">Расписание</span>
|
||||
</div>
|
||||
</div>
|
||||
<div class="module-actions">
|
||||
<button
|
||||
class="btn btn-primary btn-deploy"
|
||||
@click="router.push(`/management/modules/deploy/vesting?address=${route.query.address}`)"
|
||||
>
|
||||
<i class="fas fa-rocket"></i>
|
||||
Деплой
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- StakingModule -->
|
||||
<div class="module-deploy-card">
|
||||
<div class="module-content">
|
||||
<h4>StakingModule</h4>
|
||||
<p>Стейкинг токенов - заработок на удержании токенов</p>
|
||||
<div class="module-features">
|
||||
<span class="feature-tag">Стейкинг</span>
|
||||
<span class="feature-tag">Доход</span>
|
||||
<span class="feature-tag">Ликвидность</span>
|
||||
<span class="feature-tag">APY</span>
|
||||
</div>
|
||||
</div>
|
||||
<div class="module-actions">
|
||||
<button
|
||||
class="btn btn-primary btn-deploy"
|
||||
@click="router.push(`/management/modules/deploy/staking?address=${route.query.address}`)"
|
||||
>
|
||||
<i class="fas fa-rocket"></i>
|
||||
Деплой
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- InsuranceModule -->
|
||||
<div class="module-deploy-card">
|
||||
<div class="module-content">
|
||||
<h4>InsuranceModule</h4>
|
||||
<p>Страхование токенов - защита от рисков и потерь</p>
|
||||
<div class="module-features">
|
||||
<span class="feature-tag">Страхование</span>
|
||||
<span class="feature-tag">Защита</span>
|
||||
<span class="feature-tag">Риски</span>
|
||||
<span class="feature-tag">Безопасность</span>
|
||||
</div>
|
||||
</div>
|
||||
<div class="module-actions">
|
||||
<button
|
||||
class="btn btn-primary btn-deploy"
|
||||
@click="router.push(`/management/modules/deploy/insurance?address=${route.query.address}`)"
|
||||
>
|
||||
<i class="fas fa-rocket"></i>
|
||||
Деплой
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- ComplianceModule -->
|
||||
<div class="module-deploy-card">
|
||||
<div class="module-content">
|
||||
<h4>ComplianceModule</h4>
|
||||
<p>Соответствие требованиям - KYC/AML, налоги, аудит</p>
|
||||
<div class="module-features">
|
||||
<span class="feature-tag">KYC/AML</span>
|
||||
<span class="feature-tag">Налоги</span>
|
||||
<span class="feature-tag">Аудит</span>
|
||||
<span class="feature-tag">Регуляторы</span>
|
||||
</div>
|
||||
</div>
|
||||
<div class="module-actions">
|
||||
<button
|
||||
class="btn btn-primary btn-deploy"
|
||||
@click="router.push(`/management/modules/deploy/compliance?address=${route.query.address}`)"
|
||||
>
|
||||
<i class="fas fa-rocket"></i>
|
||||
Деплой
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- SupplyChainModule -->
|
||||
<div class="module-deploy-card">
|
||||
<div class="module-content">
|
||||
<h4>SupplyChainModule</h4>
|
||||
<p>Цепочка поставок - отслеживание и токенизация логистики</p>
|
||||
<div class="module-features">
|
||||
<span class="feature-tag">Логистика</span>
|
||||
<span class="feature-tag">Отслеживание</span>
|
||||
<span class="feature-tag">Качество</span>
|
||||
<span class="feature-tag">Прозрачность</span>
|
||||
</div>
|
||||
</div>
|
||||
<div class="module-actions">
|
||||
<button
|
||||
class="btn btn-primary btn-deploy"
|
||||
@click="router.push(`/management/modules/deploy/supplychain?address=${route.query.address}`)"
|
||||
>
|
||||
<i class="fas fa-rocket"></i>
|
||||
Деплой
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- EventModule -->
|
||||
<div class="module-deploy-card">
|
||||
<div class="module-content">
|
||||
<h4>EventModule</h4>
|
||||
<p>Событийный модуль - токенизация мероприятий и событий</p>
|
||||
<div class="module-features">
|
||||
<span class="feature-tag">События</span>
|
||||
<span class="feature-tag">NFT-билеты</span>
|
||||
<span class="feature-tag">Мероприятия</span>
|
||||
<span class="feature-tag">VR/AR</span>
|
||||
</div>
|
||||
</div>
|
||||
<div class="module-actions">
|
||||
<button
|
||||
class="btn btn-primary btn-deploy"
|
||||
@click="router.push(`/management/modules/deploy/event?address=${route.query.address}`)"
|
||||
>
|
||||
<i class="fas fa-rocket"></i>
|
||||
Деплой
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Форма добавления модуля -->
|
||||
<div class="add-module-form">
|
||||
<div class="form-header">
|
||||
@@ -521,6 +847,116 @@ onMounted(() => {
|
||||
border: 1px solid #dee2e6;
|
||||
}
|
||||
|
||||
/* Блоки для деплоя стандартных модулей */
|
||||
.standard-modules {
|
||||
background: #f8f9fa;
|
||||
border-radius: var(--radius-md);
|
||||
padding: 20px;
|
||||
margin-bottom: 30px;
|
||||
border: 1px solid #e9ecef;
|
||||
}
|
||||
|
||||
.modules-header {
|
||||
margin-bottom: 20px;
|
||||
padding-bottom: 15px;
|
||||
border-bottom: 1px solid #dee2e6;
|
||||
}
|
||||
|
||||
.modules-header h3 {
|
||||
margin: 0 0 10px 0;
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
.modules-header p {
|
||||
margin: 0 0 15px 0;
|
||||
color: #666;
|
||||
}
|
||||
|
||||
.module-deploy-card {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
padding: 20px;
|
||||
background: white;
|
||||
border: 1px solid #e9ecef;
|
||||
border-radius: var(--radius-md);
|
||||
margin-bottom: 15px;
|
||||
transition: all 0.2s;
|
||||
box-shadow: 0 1px 3px rgba(0, 0, 0, 0.1);
|
||||
}
|
||||
|
||||
.module-deploy-card:hover {
|
||||
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.15);
|
||||
transform: translateY(-2px);
|
||||
}
|
||||
|
||||
.module-content {
|
||||
flex: 1;
|
||||
margin-bottom: 20px;
|
||||
}
|
||||
|
||||
.module-content h4 {
|
||||
margin: 0 0 8px 0;
|
||||
color: var(--color-primary);
|
||||
font-size: 1.2rem;
|
||||
font-weight: 600;
|
||||
}
|
||||
|
||||
.module-content p {
|
||||
margin: 0 0 12px 0;
|
||||
color: #666;
|
||||
font-size: 14px;
|
||||
line-height: 1.4;
|
||||
}
|
||||
|
||||
.module-features {
|
||||
display: flex;
|
||||
flex-wrap: wrap;
|
||||
gap: 6px;
|
||||
}
|
||||
|
||||
.feature-tag {
|
||||
background: linear-gradient(135deg, #e3f2fd, #bbdefb);
|
||||
color: #1976d2;
|
||||
padding: 4px 10px;
|
||||
border-radius: 12px;
|
||||
font-size: 11px;
|
||||
font-weight: 500;
|
||||
border: 1px solid #90caf9;
|
||||
}
|
||||
|
||||
.module-actions {
|
||||
display: flex;
|
||||
justify-content: center;
|
||||
}
|
||||
|
||||
.btn-deploy {
|
||||
background: linear-gradient(135deg, var(--color-primary), var(--color-primary-dark));
|
||||
color: white;
|
||||
padding: 10px 20px;
|
||||
border: none;
|
||||
border-radius: var(--radius-md);
|
||||
cursor: pointer;
|
||||
font-size: 14px;
|
||||
font-weight: 500;
|
||||
transition: all 0.3s;
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
gap: 8px;
|
||||
box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1);
|
||||
}
|
||||
|
||||
.btn-deploy:hover:not(:disabled) {
|
||||
background: linear-gradient(135deg, var(--color-primary-dark), var(--color-primary));
|
||||
box-shadow: 0 4px 8px rgba(0, 0, 0, 0.2);
|
||||
transform: translateY(-1px);
|
||||
}
|
||||
|
||||
.btn-deploy:disabled {
|
||||
opacity: 0.6;
|
||||
cursor: not-allowed;
|
||||
transform: none;
|
||||
}
|
||||
|
||||
/* Форма добавления модуля */
|
||||
.add-module-form {
|
||||
background: #f8f9fa;
|
||||
@@ -770,4 +1206,20 @@ onMounted(() => {
|
||||
grid-template-columns: 1fr;
|
||||
}
|
||||
}
|
||||
|
||||
/* Адаптивность для блоков деплоя */
|
||||
@media (max-width: 768px) {
|
||||
.module-deploy-card {
|
||||
padding: 15px;
|
||||
}
|
||||
|
||||
.module-content {
|
||||
margin-bottom: 15px;
|
||||
}
|
||||
|
||||
.btn-deploy {
|
||||
width: 100%;
|
||||
justify-content: center;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
|
||||
@@ -22,7 +22,8 @@
|
||||
<!-- Заголовок -->
|
||||
<div class="page-header">
|
||||
<div class="header-content">
|
||||
<h1>Токены DLE</h1>
|
||||
<h1>Управление токенами DLE</h1>
|
||||
<p>Создание предложений для перевода токенов через систему голосования</p>
|
||||
<div v-if="selectedDle" class="dle-info">
|
||||
<span class="dle-name">{{ selectedDle.name }} ({{ selectedDle.symbol }})</span>
|
||||
<span class="dle-address">{{ shortenAddress(selectedDle.dleAddress) }}</span>
|
||||
@@ -48,10 +49,8 @@
|
||||
<div class="info-card">
|
||||
<h3>Ваш баланс</h3>
|
||||
<p class="token-amount">{{ userBalance }} {{ tokenSymbol }}</p>
|
||||
</div>
|
||||
<div class="info-card">
|
||||
<h3>Кворум</h3>
|
||||
<p class="token-amount">{{ quorumPercentage }}%</p>
|
||||
<p v-if="currentUserAddress" class="user-address">{{ shortenAddress(currentUserAddress) }}</p>
|
||||
<p v-else class="no-wallet">Кошелек не подключен</p>
|
||||
</div>
|
||||
<div class="info-card">
|
||||
<h3>Цена токена</h3>
|
||||
@@ -60,110 +59,70 @@
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Трансфер токенов -->
|
||||
<!-- Перевод токенов через governance -->
|
||||
<div class="transfer-section">
|
||||
<h2>Перевод токенов</h2>
|
||||
<form @submit.prevent="transferTokens" class="transfer-form">
|
||||
<div class="form-row">
|
||||
<div class="form-group">
|
||||
<label for="recipient">Получатель:</label>
|
||||
<input
|
||||
id="recipient"
|
||||
v-model="transferData.recipient"
|
||||
type="text"
|
||||
placeholder="0x..."
|
||||
required
|
||||
>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="amount">Количество токенов:</label>
|
||||
<input
|
||||
id="amount"
|
||||
v-model="transferData.amount"
|
||||
type="number"
|
||||
min="0.01"
|
||||
step="0.01"
|
||||
placeholder="0.00"
|
||||
required
|
||||
>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<h2>Перевод токенов через Governance</h2>
|
||||
<p class="section-description">
|
||||
Создайте предложение для перевода токенов через систему голосования.
|
||||
Токены будут переведены от имени DLE после одобрения кворумом.
|
||||
<strong>Важно:</strong> Перевод через governance будет выполнен во всех поддерживаемых сетях DLE.
|
||||
</p>
|
||||
|
||||
<form @submit.prevent="createTransferProposal" class="transfer-form">
|
||||
<div class="form-group">
|
||||
<label for="transferDescription">Описание (опционально):</label>
|
||||
<label for="proposal-recipient">Адрес получателя:</label>
|
||||
<input
|
||||
id="proposal-recipient"
|
||||
v-model="proposalData.recipient"
|
||||
type="text"
|
||||
placeholder="0x..."
|
||||
required
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="proposal-amount">Количество токенов:</label>
|
||||
<input
|
||||
id="proposal-amount"
|
||||
v-model="proposalData.amount"
|
||||
type="number"
|
||||
step="0.000001"
|
||||
placeholder="0.0"
|
||||
required
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="proposal-description">Описание предложения:</label>
|
||||
<textarea
|
||||
id="transferDescription"
|
||||
v-model="transferData.description"
|
||||
placeholder="Укажите причину перевода..."
|
||||
rows="3"
|
||||
id="proposal-description"
|
||||
v-model="proposalData.description"
|
||||
placeholder="Опишите причину перевода токенов..."
|
||||
required
|
||||
></textarea>
|
||||
</div>
|
||||
|
||||
<button type="submit" class="btn-primary" :disabled="isTransferring">
|
||||
{{ isTransferring ? 'Перевод...' : 'Перевести токены' }}
|
||||
</button>
|
||||
</form>
|
||||
</div>
|
||||
|
||||
<!-- Распределение токенов -->
|
||||
<div class="distribution-section">
|
||||
<h2>Распределение токенов</h2>
|
||||
<form @submit.prevent="distributeTokens" class="distribution-form">
|
||||
<div class="form-group">
|
||||
<label for="distributionType">Тип распределения:</label>
|
||||
<select id="distributionType" v-model="distributionData.type" required>
|
||||
<option value="">Выберите тип</option>
|
||||
<option value="partners">Партнерам</option>
|
||||
<option value="employees">Сотрудникам</option>
|
||||
<option value="investors">Инвесторам</option>
|
||||
<option value="custom">Пользовательское</option>
|
||||
</select>
|
||||
<label for="proposal-duration">Длительность голосования (часы):</label>
|
||||
<input
|
||||
id="proposal-duration"
|
||||
v-model="proposalData.duration"
|
||||
type="number"
|
||||
min="1"
|
||||
max="168"
|
||||
placeholder="24"
|
||||
required
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label>Получатели:</label>
|
||||
<div class="recipients-list">
|
||||
<div
|
||||
v-for="(recipient, index) in distributionData.recipients"
|
||||
:key="index"
|
||||
class="recipient-item"
|
||||
>
|
||||
<input
|
||||
v-model="recipient.address"
|
||||
type="text"
|
||||
placeholder="Адрес получателя"
|
||||
required
|
||||
>
|
||||
<input
|
||||
v-model="recipient.amount"
|
||||
type="number"
|
||||
placeholder="Количество"
|
||||
min="0.01"
|
||||
step="0.01"
|
||||
required
|
||||
>
|
||||
<button
|
||||
type="button"
|
||||
@click="removeRecipient(index)"
|
||||
class="btn-remove"
|
||||
>
|
||||
✕
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
<button
|
||||
type="button"
|
||||
@click="addRecipient"
|
||||
class="btn-secondary"
|
||||
>
|
||||
+ Добавить получателя
|
||||
</button>
|
||||
</div>
|
||||
|
||||
<button type="submit" class="btn-primary" :disabled="isDistributing">
|
||||
{{ isDistributing ? 'Распределение...' : 'Распределить токены' }}
|
||||
|
||||
<button type="submit" class="btn-primary" :disabled="isCreatingProposal">
|
||||
{{ isCreatingProposal ? 'Создание предложения...' : 'Создать предложение' }}
|
||||
</button>
|
||||
|
||||
<!-- Статус предложения -->
|
||||
<div v-if="proposalStatus" class="proposal-status">
|
||||
<p class="status-message">{{ proposalStatus }}</p>
|
||||
</div>
|
||||
</form>
|
||||
</div>
|
||||
|
||||
@@ -199,6 +158,8 @@ import { useRouter, useRoute } from 'vue-router';
|
||||
import BaseLayout from '../../components/BaseLayout.vue';
|
||||
import { getTokenBalance, getTotalSupply, getTokenHolders } from '../../services/tokensService.js';
|
||||
import api from '../../api/axios';
|
||||
import { ethers } from 'ethers';
|
||||
import { createTransferTokensProposal } from '../../utils/dle-contract.js';
|
||||
|
||||
// Определяем props
|
||||
const props = defineProps({
|
||||
@@ -225,30 +186,48 @@ const dleAddress = computed(() => {
|
||||
const selectedDle = ref(null);
|
||||
const isLoadingDle = ref(false);
|
||||
|
||||
// Состояние
|
||||
const isTransferring = ref(false);
|
||||
const isDistributing = ref(false);
|
||||
// Состояние для предложения о переводе токенов через governance
|
||||
const isCreatingProposal = ref(false);
|
||||
const proposalStatus = ref('');
|
||||
|
||||
// Данные токенов (загружаются из блокчейна)
|
||||
const tokenSymbol = computed(() => selectedDle.value?.symbol || '');
|
||||
const totalSupply = computed(() => selectedDle.value?.totalSupply || 0);
|
||||
const userBalance = computed(() => selectedDle.value?.deployerBalance || 0);
|
||||
const quorumPercentage = computed(() => selectedDle.value?.quorumPercentage || 0);
|
||||
const totalSupply = ref(0);
|
||||
const userBalance = ref(0);
|
||||
const deployerBalance = ref(0);
|
||||
const quorumPercentage = ref(0);
|
||||
const tokenPrice = ref(0);
|
||||
|
||||
// Данные трансфера
|
||||
const transferData = ref({
|
||||
// Данные для формы
|
||||
const proposalData = ref({
|
||||
recipient: '',
|
||||
amount: '',
|
||||
description: ''
|
||||
description: '',
|
||||
duration: 86400, // 24 часа по умолчанию
|
||||
governanceChainId: 11155111, // Sepolia по умолчанию
|
||||
targetChains: [11155111] // Sepolia по умолчанию
|
||||
});
|
||||
|
||||
// Данные распределения
|
||||
const distributionData = ref({
|
||||
type: '',
|
||||
recipients: [
|
||||
{ address: '', amount: '' }
|
||||
]
|
||||
// Получаем адрес текущего пользователя
|
||||
const currentUserAddress = computed(() => {
|
||||
console.log('Проверяем identities:', props.identities);
|
||||
|
||||
// Получаем адрес из props или из window.ethereum
|
||||
if (props.identities && props.identities.length > 0) {
|
||||
const walletIdentity = props.identities.find(id => id.provider === 'wallet');
|
||||
console.log('Найден wallet identity:', walletIdentity);
|
||||
if (walletIdentity) {
|
||||
return walletIdentity.provider_id;
|
||||
}
|
||||
}
|
||||
|
||||
// Fallback: пытаемся получить из window.ethereum
|
||||
if (window.ethereum && window.ethereum.selectedAddress) {
|
||||
console.log('Получаем адрес из window.ethereum:', window.ethereum.selectedAddress);
|
||||
return window.ethereum.selectedAddress;
|
||||
}
|
||||
|
||||
console.log('Адрес пользователя не найден');
|
||||
return null;
|
||||
});
|
||||
|
||||
// Держатели токенов (загружаются из блокчейна)
|
||||
@@ -273,6 +252,9 @@ async function loadDleData() {
|
||||
selectedDle.value = blockchainData;
|
||||
console.log('Загружены данные DLE из блокчейна:', blockchainData);
|
||||
|
||||
// Загружаем баланс текущего пользователя
|
||||
await loadUserBalance();
|
||||
|
||||
// Загружаем держателей токенов (если есть API)
|
||||
await loadTokenHolders();
|
||||
} else {
|
||||
@@ -285,6 +267,35 @@ async function loadDleData() {
|
||||
}
|
||||
}
|
||||
|
||||
// Новая функция для загрузки баланса текущего пользователя
|
||||
async function loadUserBalance() {
|
||||
if (!currentUserAddress.value || !dleAddress.value) {
|
||||
userBalance.value = 0;
|
||||
console.log('Не удается загрузить баланс: нет адреса пользователя или DLE');
|
||||
return;
|
||||
}
|
||||
|
||||
try {
|
||||
console.log('Загружаем баланс для пользователя:', currentUserAddress.value);
|
||||
|
||||
const response = await api.post('/blockchain/get-token-balance', {
|
||||
dleAddress: dleAddress.value,
|
||||
account: currentUserAddress.value
|
||||
});
|
||||
|
||||
if (response.data.success) {
|
||||
userBalance.value = parseFloat(response.data.data.balance);
|
||||
console.log('Баланс пользователя загружен:', userBalance.value);
|
||||
} else {
|
||||
console.warn('Не удалось загрузить баланс пользователя:', response.data.error);
|
||||
userBalance.value = 0;
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('Ошибка загрузки баланса пользователя:', error);
|
||||
userBalance.value = 0;
|
||||
}
|
||||
}
|
||||
|
||||
async function loadTokenHolders() {
|
||||
try {
|
||||
// Здесь можно добавить загрузку держателей токенов из блокчейна
|
||||
@@ -301,99 +312,132 @@ function shortenAddress(address) {
|
||||
}
|
||||
|
||||
// Методы
|
||||
const transferTokens = async () => {
|
||||
if (isTransferring.value) return;
|
||||
|
||||
try {
|
||||
isTransferring.value = true;
|
||||
|
||||
// Здесь будет логика трансфера токенов
|
||||
// console.log('Трансфер токенов:', transferData.value);
|
||||
|
||||
// Временная логика
|
||||
const amount = parseFloat(transferData.value.amount);
|
||||
if (amount > userBalance.value) {
|
||||
alert('Недостаточно токенов для перевода');
|
||||
return;
|
||||
}
|
||||
|
||||
userBalance.value -= amount;
|
||||
|
||||
// Сброс формы
|
||||
transferData.value = {
|
||||
recipient: '',
|
||||
amount: '',
|
||||
description: ''
|
||||
};
|
||||
|
||||
alert('Токены успешно переведены!');
|
||||
|
||||
} catch (error) {
|
||||
// console.error('Ошибка трансфера токенов:', error);
|
||||
alert('Ошибка при переводе токенов');
|
||||
} finally {
|
||||
isTransferring.value = false;
|
||||
}
|
||||
};
|
||||
|
||||
const distributeTokens = async () => {
|
||||
if (isDistributing.value) return;
|
||||
|
||||
try {
|
||||
isDistributing.value = true;
|
||||
|
||||
// Здесь будет логика распределения токенов
|
||||
// console.log('Распределение токенов:', distributionData.value);
|
||||
|
||||
// Временная логика
|
||||
const totalAmount = distributionData.value.recipients.reduce((sum, recipient) => {
|
||||
return sum + parseFloat(recipient.amount || 0);
|
||||
}, 0);
|
||||
|
||||
if (totalAmount > userBalance.value) {
|
||||
alert('Недостаточно токенов для распределения');
|
||||
return;
|
||||
}
|
||||
|
||||
userBalance.value -= totalAmount;
|
||||
|
||||
// Сброс формы
|
||||
distributionData.value = {
|
||||
type: '',
|
||||
recipients: [{ address: '', amount: '' }]
|
||||
};
|
||||
|
||||
alert('Токены успешно распределены!');
|
||||
|
||||
} catch (error) {
|
||||
// console.error('Ошибка распределения токенов:', error);
|
||||
alert('Ошибка при распределении токенов');
|
||||
} finally {
|
||||
isDistributing.value = false;
|
||||
}
|
||||
};
|
||||
|
||||
const addRecipient = () => {
|
||||
distributionData.value.recipients.push({ address: '', amount: '' });
|
||||
};
|
||||
|
||||
const removeRecipient = (index) => {
|
||||
if (distributionData.value.recipients.length > 1) {
|
||||
distributionData.value.recipients.splice(index, 1);
|
||||
}
|
||||
};
|
||||
|
||||
const formatAddress = (address) => {
|
||||
if (!address) return '';
|
||||
return address.substring(0, 6) + '...' + address.substring(address.length - 4);
|
||||
};
|
||||
|
||||
// Функция создания предложения о переводе токенов через governance
|
||||
const createTransferProposal = async () => {
|
||||
if (isCreatingProposal.value) return;
|
||||
|
||||
try {
|
||||
// Проверяем подключение к кошельку
|
||||
if (!window.ethereum) {
|
||||
alert('Пожалуйста, установите MetaMask или другой Web3 кошелек');
|
||||
return;
|
||||
}
|
||||
|
||||
// Проверяем, что пользователь подключен
|
||||
if (!currentUserAddress.value) {
|
||||
alert('Пожалуйста, подключите кошелек');
|
||||
return;
|
||||
}
|
||||
|
||||
// Валидация данных
|
||||
const recipient = proposalData.value.recipient.trim();
|
||||
const amount = parseFloat(proposalData.value.amount);
|
||||
const description = proposalData.value.description.trim();
|
||||
|
||||
if (!recipient) {
|
||||
alert('Пожалуйста, укажите адрес получателя');
|
||||
return;
|
||||
}
|
||||
|
||||
// Проверяем, что адрес получателя является корректным Ethereum адресом
|
||||
if (!ethers.isAddress(recipient)) {
|
||||
alert('Пожалуйста, укажите корректный Ethereum адрес получателя');
|
||||
return;
|
||||
}
|
||||
|
||||
if (!amount || amount <= 0) {
|
||||
alert('Пожалуйста, укажите корректное количество токенов');
|
||||
return;
|
||||
}
|
||||
|
||||
if (!description) {
|
||||
alert('Пожалуйста, укажите описание предложения');
|
||||
return;
|
||||
}
|
||||
|
||||
// Проверяем, что получатель не является отправителем
|
||||
if (recipient.toLowerCase() === currentUserAddress.value.toLowerCase()) {
|
||||
alert('Нельзя отправить токены самому себе');
|
||||
return;
|
||||
}
|
||||
|
||||
isCreatingProposal.value = true;
|
||||
proposalStatus.value = 'Создание предложения...';
|
||||
|
||||
// Создаем предложение
|
||||
const result = await createTransferTokensProposal(dleAddress.value, {
|
||||
recipient: recipient,
|
||||
amount: amount,
|
||||
description: description,
|
||||
duration: proposalData.value.duration * 3600, // Конвертируем часы в секунды
|
||||
governanceChainId: proposalData.value.governanceChainId,
|
||||
targetChains: proposalData.value.targetChains
|
||||
});
|
||||
|
||||
proposalStatus.value = 'Предложение создано!';
|
||||
console.log('Предложение о переводе токенов создано:', result);
|
||||
|
||||
// Сброс формы
|
||||
proposalData.value = {
|
||||
recipient: '',
|
||||
amount: '',
|
||||
description: '',
|
||||
duration: 86400,
|
||||
governanceChainId: 11155111,
|
||||
targetChains: [11155111]
|
||||
};
|
||||
|
||||
// Очищаем статус через 5 секунд
|
||||
setTimeout(() => {
|
||||
proposalStatus.value = '';
|
||||
}, 5000);
|
||||
|
||||
alert(`Предложение о переводе токенов создано!\nID предложения: ${result.proposalId}\nХеш транзакции: ${result.txHash}`);
|
||||
|
||||
} catch (error) {
|
||||
console.error('Ошибка создания предложения о переводе токенов:', error);
|
||||
|
||||
// Очищаем статус предложения
|
||||
proposalStatus.value = '';
|
||||
|
||||
let errorMessage = 'Ошибка создания предложения о переводе токенов';
|
||||
|
||||
if (error.code === 4001) {
|
||||
errorMessage = 'Транзакция отменена пользователем';
|
||||
} else if (error.message && error.message.includes('insufficient funds')) {
|
||||
errorMessage = 'Недостаточно ETH для оплаты газа';
|
||||
} else if (error.message && error.message.includes('execution reverted')) {
|
||||
errorMessage = 'Ошибка выполнения транзакции. Проверьте данные и попробуйте снова';
|
||||
} else if (error.message) {
|
||||
errorMessage = error.message;
|
||||
}
|
||||
|
||||
alert(errorMessage);
|
||||
} finally {
|
||||
isCreatingProposal.value = false;
|
||||
}
|
||||
};
|
||||
|
||||
// Отслеживаем изменения в адресе DLE
|
||||
watch(dleAddress, (newAddress) => {
|
||||
if (newAddress) {
|
||||
loadDleData();
|
||||
}
|
||||
}, { immediate: true });
|
||||
|
||||
// Отслеживаем изменения адреса пользователя
|
||||
watch(currentUserAddress, (newAddress) => {
|
||||
if (newAddress && dleAddress.value) {
|
||||
loadUserBalance();
|
||||
} else {
|
||||
userBalance.value = 0;
|
||||
}
|
||||
}, { immediate: true });
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
@@ -488,14 +532,12 @@ watch(dleAddress, (newAddress) => {
|
||||
/* Секции */
|
||||
.token-info-section,
|
||||
.transfer-section,
|
||||
.distribution-section,
|
||||
.holders-section {
|
||||
margin-bottom: 40px;
|
||||
}
|
||||
|
||||
.token-info-section h2,
|
||||
.transfer-section h2,
|
||||
.distribution-section h2,
|
||||
.holders-section h2 {
|
||||
color: var(--color-primary);
|
||||
margin-bottom: 20px;
|
||||
@@ -532,9 +574,26 @@ watch(dleAddress, (newAddress) => {
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
.user-address {
|
||||
font-family: monospace;
|
||||
font-size: 0.75rem;
|
||||
color: #666;
|
||||
margin: 5px 0 0 0;
|
||||
background: #f8f9fa;
|
||||
padding: 2px 6px;
|
||||
border-radius: 3px;
|
||||
display: inline-block;
|
||||
}
|
||||
|
||||
.no-wallet {
|
||||
font-size: 0.75rem;
|
||||
color: #dc3545;
|
||||
margin: 5px 0 0 0;
|
||||
font-style: italic;
|
||||
}
|
||||
|
||||
/* Формы */
|
||||
.transfer-form,
|
||||
.distribution-form {
|
||||
.transfer-form {
|
||||
background: #f8f9fa;
|
||||
padding: 25px;
|
||||
border-radius: var(--radius-lg);
|
||||
@@ -574,81 +633,25 @@ watch(dleAddress, (newAddress) => {
|
||||
min-height: 80px;
|
||||
}
|
||||
|
||||
/* Получатели */
|
||||
.recipients-list {
|
||||
display: grid;
|
||||
gap: 15px;
|
||||
margin-bottom: 15px;
|
||||
/* Статус предложения */
|
||||
.proposal-status {
|
||||
margin-top: 20px;
|
||||
padding: 15px;
|
||||
background: #e8f5e8;
|
||||
border-radius: var(--radius-sm);
|
||||
border-left: 4px solid #28a745;
|
||||
}
|
||||
|
||||
.recipient-item {
|
||||
display: grid;
|
||||
grid-template-columns: 1fr 1fr auto;
|
||||
gap: 15px;
|
||||
align-items: center;
|
||||
.proposal-status .status-message {
|
||||
color: #28a745;
|
||||
}
|
||||
|
||||
.btn-remove {
|
||||
background: #dc3545;
|
||||
color: white;
|
||||
border: none;
|
||||
border-radius: 50%;
|
||||
width: 30px;
|
||||
height: 30px;
|
||||
cursor: pointer;
|
||||
font-size: 14px;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
}
|
||||
|
||||
.btn-remove:hover {
|
||||
background: #c82333;
|
||||
}
|
||||
|
||||
/* Держатели токенов */
|
||||
.holders-list {
|
||||
display: grid;
|
||||
gap: 15px;
|
||||
}
|
||||
|
||||
.holder-item {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: center;
|
||||
padding: 20px;
|
||||
background: white;
|
||||
border-radius: var(--radius-lg);
|
||||
border: 1px solid #e9ecef;
|
||||
transition: all 0.3s ease;
|
||||
}
|
||||
|
||||
.holder-item:hover {
|
||||
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.1);
|
||||
}
|
||||
|
||||
.holder-info {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 5px;
|
||||
}
|
||||
|
||||
.holder-address {
|
||||
font-family: monospace;
|
||||
font-size: 0.9rem;
|
||||
/* Описание секции */
|
||||
.section-description {
|
||||
color: var(--color-grey-dark);
|
||||
}
|
||||
|
||||
.holder-balance {
|
||||
font-size: 1.1rem;
|
||||
font-weight: 600;
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
.holder-percentage {
|
||||
font-size: 0.9rem;
|
||||
color: var(--color-grey-dark);
|
||||
font-weight: 600;
|
||||
font-size: 0.95rem;
|
||||
margin-bottom: 20px;
|
||||
line-height: 1.5;
|
||||
}
|
||||
|
||||
/* Кнопки */
|
||||
|
||||
@@ -0,0 +1,329 @@
|
||||
<!--
|
||||
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="application-module-deploy">
|
||||
<!-- Заголовок -->
|
||||
<div class="page-header">
|
||||
<div class="header-content">
|
||||
<h1>Деплой ApplicationModule</h1>
|
||||
<p>Управление вызовом функций приложения через предложения и голосование</p>
|
||||
<p v-if="dleAddress" class="dle-address">
|
||||
<strong>DLE:</strong> {{ dleAddress }}
|
||||
</p>
|
||||
</div>
|
||||
<button class="close-btn" @click="router.push('/management/modules')">×</button>
|
||||
</div>
|
||||
|
||||
<!-- Информация о модуле -->
|
||||
<div class="module-info">
|
||||
<div class="info-card">
|
||||
<h3>🖥️ ApplicationModule</h3>
|
||||
<div class="info-grid">
|
||||
<div class="info-item">
|
||||
<strong>Назначение:</strong> Управление функциями приложения через DLE
|
||||
</div>
|
||||
<div class="info-item">
|
||||
<strong>Функции:</strong> Создание предложений для вызова API, голосование за операции
|
||||
</div>
|
||||
<div class="info-item">
|
||||
<strong>Безопасность:</strong> Все операции приложения через кворум токен-холдеров
|
||||
</div>
|
||||
<div class="info-item">
|
||||
<strong>Примеры:</strong> Удаление пользователей, изменение настроек, обновление данных
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Детальное описание -->
|
||||
<div class="module-details">
|
||||
<div class="details-card">
|
||||
<h3>📋 Как работает ApplicationModule</h3>
|
||||
<div class="details-content">
|
||||
<div class="detail-step">
|
||||
<div class="step-number">1</div>
|
||||
<div class="step-content">
|
||||
<h4>Создание предложения</h4>
|
||||
<p>Токен-холдер создает предложение для выполнения операции в приложении (например, удаление пользователя, изменение настроек)</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="detail-step">
|
||||
<div class="step-number">2</div>
|
||||
<div class="step-content">
|
||||
<h4>Голосование</h4>
|
||||
<p>Все токен-холдеры голосуют за или против предложения в течение установленного времени</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="detail-step">
|
||||
<div class="step-number">3</div>
|
||||
<div class="step-content">
|
||||
<h4>Исполнение</h4>
|
||||
<p>При достижении кворума предложение исполняется - вызывается соответствующая функция приложения</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="detail-step">
|
||||
<div class="step-number">4</div>
|
||||
<div class="step-content">
|
||||
<h4>Аудит</h4>
|
||||
<p>Все операции логируются в блокчейне для полной прозрачности и подотчетности</p>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Форма деплоя будет добавлена позже -->
|
||||
<div class="deploy-form-placeholder">
|
||||
<div class="placeholder-content">
|
||||
<h3>🚧 Форма деплоя в разработке</h3>
|
||||
<p>Здесь будет форма для деплоя ApplicationModule</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
</div>
|
||||
</BaseLayout>
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import { defineProps, defineEmits, ref, onMounted } from 'vue';
|
||||
import { useRouter, useRoute } from 'vue-router';
|
||||
import BaseLayout from '../../../components/BaseLayout.vue';
|
||||
|
||||
// Определяем props
|
||||
const props = defineProps({
|
||||
isAuthenticated: { type: Boolean, default: false },
|
||||
identities: { type: Array, default: () => [] },
|
||||
tokenBalances: { type: Object, default: () => ({}) },
|
||||
isLoadingTokens: { type: Boolean, default: false }
|
||||
});
|
||||
|
||||
// Определяем emits
|
||||
const emit = defineEmits(['auth-action-completed']);
|
||||
|
||||
const router = useRouter();
|
||||
const route = useRoute();
|
||||
|
||||
// Состояние
|
||||
const isLoading = ref(false);
|
||||
const dleAddress = ref(route.query.address || null);
|
||||
|
||||
// Инициализация
|
||||
onMounted(() => {
|
||||
console.log('[ApplicationModuleDeployView] Страница загружена');
|
||||
});
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
.application-module-deploy {
|
||||
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: center;
|
||||
margin-bottom: 30px;
|
||||
padding-bottom: 20px;
|
||||
border-bottom: 2px solid #f0f0f0;
|
||||
}
|
||||
|
||||
.page-header h1 {
|
||||
color: var(--color-primary);
|
||||
font-size: 2rem;
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
.page-header p {
|
||||
margin: 10px 0 0 0;
|
||||
color: #666;
|
||||
}
|
||||
|
||||
.dle-address {
|
||||
margin-top: 10px !important;
|
||||
font-family: monospace;
|
||||
background: #f8f9fa;
|
||||
padding: 8px 12px;
|
||||
border-radius: var(--radius-sm);
|
||||
border: 1px solid #dee2e6;
|
||||
}
|
||||
|
||||
.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;
|
||||
}
|
||||
|
||||
.close-btn:hover {
|
||||
background: #f0f0f0;
|
||||
color: #333;
|
||||
}
|
||||
|
||||
/* Информация о модуле */
|
||||
.module-info {
|
||||
margin-bottom: 30px;
|
||||
}
|
||||
|
||||
.info-card {
|
||||
background: #f8f9fa;
|
||||
border-radius: var(--radius-md);
|
||||
padding: 20px;
|
||||
border: 1px solid #e9ecef;
|
||||
}
|
||||
|
||||
.info-card h3 {
|
||||
margin: 0 0 15px 0;
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
.info-grid {
|
||||
display: grid;
|
||||
grid-template-columns: repeat(auto-fit, minmax(250px, 1fr));
|
||||
gap: 15px;
|
||||
}
|
||||
|
||||
.info-item {
|
||||
padding: 10px;
|
||||
background: white;
|
||||
border-radius: var(--radius-sm);
|
||||
border: 1px solid #dee2e6;
|
||||
}
|
||||
|
||||
.info-item strong {
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
/* Детальное описание */
|
||||
.module-details {
|
||||
margin-bottom: 30px;
|
||||
}
|
||||
|
||||
.details-card {
|
||||
background: #f8f9fa;
|
||||
border-radius: var(--radius-md);
|
||||
padding: 20px;
|
||||
border: 1px solid #e9ecef;
|
||||
}
|
||||
|
||||
.details-card h3 {
|
||||
margin: 0 0 20px 0;
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
.details-content {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 20px;
|
||||
}
|
||||
|
||||
.detail-step {
|
||||
display: flex;
|
||||
align-items: flex-start;
|
||||
gap: 15px;
|
||||
padding: 15px;
|
||||
background: white;
|
||||
border-radius: var(--radius-sm);
|
||||
border: 1px solid #dee2e6;
|
||||
}
|
||||
|
||||
.step-number {
|
||||
background: var(--color-primary);
|
||||
color: white;
|
||||
width: 30px;
|
||||
height: 30px;
|
||||
border-radius: 50%;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
font-weight: bold;
|
||||
font-size: 14px;
|
||||
flex-shrink: 0;
|
||||
}
|
||||
|
||||
.step-content h4 {
|
||||
margin: 0 0 8px 0;
|
||||
color: var(--color-primary);
|
||||
font-size: 16px;
|
||||
}
|
||||
|
||||
.step-content p {
|
||||
margin: 0;
|
||||
color: #666;
|
||||
line-height: 1.5;
|
||||
}
|
||||
|
||||
/* Плейсхолдер для формы */
|
||||
.deploy-form-placeholder {
|
||||
background: #f8f9fa;
|
||||
border-radius: var(--radius-md);
|
||||
padding: 40px;
|
||||
text-align: center;
|
||||
border: 2px dashed #dee2e6;
|
||||
}
|
||||
|
||||
.placeholder-content h3 {
|
||||
color: var(--color-primary);
|
||||
margin-bottom: 10px;
|
||||
}
|
||||
|
||||
.placeholder-content p {
|
||||
color: #666;
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
/* Адаптивность */
|
||||
@media (max-width: 768px) {
|
||||
.info-grid {
|
||||
grid-template-columns: 1fr;
|
||||
}
|
||||
|
||||
.page-header {
|
||||
flex-direction: column;
|
||||
align-items: flex-start;
|
||||
gap: 15px;
|
||||
}
|
||||
|
||||
.detail-step {
|
||||
flex-direction: column;
|
||||
text-align: center;
|
||||
}
|
||||
|
||||
.step-number {
|
||||
align-self: center;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
515
frontend/src/views/smartcontracts/modules/BurnModuleDeploy.vue
Normal file
515
frontend/src/views/smartcontracts/modules/BurnModuleDeploy.vue
Normal file
@@ -0,0 +1,515 @@
|
||||
<!--
|
||||
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="module-deploy-page">
|
||||
<!-- Заголовок -->
|
||||
<div class="page-header">
|
||||
<div class="header-content">
|
||||
<h1>🔥 Деплой BurnModule</h1>
|
||||
<p>Модуль для сжигания токенов DLE через governance</p>
|
||||
<div v-if="selectedDle" class="dle-info">
|
||||
<span class="dle-name">{{ selectedDle.name }} ({{ selectedDle.symbol }})</span>
|
||||
<span class="dle-address">{{ selectedDle.dleAddress }}</span>
|
||||
</div>
|
||||
</div>
|
||||
<button class="close-btn" @click="router.push(`/management/modules?address=${route.query.address}`)">×</button>
|
||||
</div>
|
||||
|
||||
<!-- Описание модуля -->
|
||||
<div class="module-description">
|
||||
<div class="description-card">
|
||||
<h3>📋 Описание BurnModule</h3>
|
||||
<div class="description-content">
|
||||
<p><strong>BurnModule</strong> - это модуль для управления сжиганием токенов DLE через систему governance.</p>
|
||||
|
||||
<h4>🔧 Функциональность:</h4>
|
||||
<ul>
|
||||
<li><strong>Сжигание токенов:</strong> Уменьшение общего предложения токенов DLE</li>
|
||||
<li><strong>Governance:</strong> Все операции требуют голосования и кворума</li>
|
||||
<li><strong>Безопасность:</strong> Контролируемое сжигание через коллективные решения</li>
|
||||
<li><strong>Прозрачность:</strong> Все операции записываются в блокчейн</li>
|
||||
</ul>
|
||||
|
||||
<h4>⚠️ Важные особенности:</h4>
|
||||
<ul>
|
||||
<li>Сжигание токенов возможно только через предложения и голосование</li>
|
||||
<li>Можно сжигать токены из казны DLE или от имени участников</li>
|
||||
<li>Все операции требуют достижения кворума</li>
|
||||
<li>История всех сжиганий сохраняется в блокчейне</li>
|
||||
<li>Сжигание необратимо - токены уничтожаются навсегда</li>
|
||||
</ul>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Форма деплоя -->
|
||||
<div class="deploy-form-section">
|
||||
<div class="form-header">
|
||||
<h3>⚙️ Настройки деплоя</h3>
|
||||
<p>Настройте параметры для деплоя BurnModule</p>
|
||||
</div>
|
||||
|
||||
<form @submit.prevent="deployModule" class="deploy-form">
|
||||
<div class="form-row">
|
||||
<div class="form-group">
|
||||
<label for="moduleName">Название модуля:</label>
|
||||
<input
|
||||
id="moduleName"
|
||||
v-model="deployData.moduleName"
|
||||
type="text"
|
||||
placeholder="BurnModule"
|
||||
required
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="moduleVersion">Версия модуля:</label>
|
||||
<input
|
||||
id="moduleVersion"
|
||||
v-model="deployData.moduleVersion"
|
||||
type="text"
|
||||
placeholder="1.0.0"
|
||||
required
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="moduleDescription">Описание модуля:</label>
|
||||
<textarea
|
||||
id="moduleDescription"
|
||||
v-model="deployData.moduleDescription"
|
||||
placeholder="Модуль для сжигания токенов DLE через governance..."
|
||||
rows="3"
|
||||
required
|
||||
></textarea>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="maxBurnPerProposal">Максимальное сжигание за одно предложение:</label>
|
||||
<input
|
||||
id="maxBurnPerProposal"
|
||||
v-model="deployData.maxBurnPerProposal"
|
||||
type="number"
|
||||
min="1"
|
||||
step="1"
|
||||
placeholder="1000000"
|
||||
required
|
||||
/>
|
||||
<small class="form-help">Максимальное количество токенов, которое можно сжечь за одно предложение</small>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="burnCooldown">Кулдаун между сжиганиями (часы):</label>
|
||||
<input
|
||||
id="burnCooldown"
|
||||
v-model="deployData.burnCooldown"
|
||||
type="number"
|
||||
min="0"
|
||||
step="1"
|
||||
placeholder="24"
|
||||
required
|
||||
/>
|
||||
<small class="form-help">Минимальное время между успешными сжиганиями токенов</small>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="allowTreasuryBurn">Разрешить сжигание из казны:</label>
|
||||
<select
|
||||
id="allowTreasuryBurn"
|
||||
v-model="deployData.allowTreasuryBurn"
|
||||
required
|
||||
>
|
||||
<option value="true">Да</option>
|
||||
<option value="false">Нет</option>
|
||||
</select>
|
||||
<small class="form-help">Разрешить сжигание токенов из казны DLE</small>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="allowUserBurn">Разрешить сжигание от участников:</label>
|
||||
<select
|
||||
id="allowUserBurn"
|
||||
v-model="deployData.allowUserBurn"
|
||||
required
|
||||
>
|
||||
<option value="true">Да</option>
|
||||
<option value="false">Нет</option>
|
||||
</select>
|
||||
<small class="form-help">Разрешить сжигание токенов от имени участников DLE</small>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="deployDescription">Описание предложения для деплоя:</label>
|
||||
<textarea
|
||||
id="deployDescription"
|
||||
v-model="deployData.deployDescription"
|
||||
placeholder="Предложение о деплое BurnModule для управления сжиганием токенов DLE..."
|
||||
rows="3"
|
||||
required
|
||||
></textarea>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="votingDuration">Длительность голосования (часы):</label>
|
||||
<input
|
||||
id="votingDuration"
|
||||
v-model="deployData.votingDuration"
|
||||
type="number"
|
||||
min="1"
|
||||
max="168"
|
||||
placeholder="24"
|
||||
required
|
||||
/>
|
||||
<small class="form-help">Время для голосования по предложению деплоя (1-168 часов)</small>
|
||||
</div>
|
||||
|
||||
<button type="submit" class="btn-primary" :disabled="isDeploying">
|
||||
{{ isDeploying ? 'Деплой...' : 'Деплой BurnModule' }}
|
||||
</button>
|
||||
|
||||
<!-- Статус деплоя -->
|
||||
<div v-if="deployStatus" class="deploy-status">
|
||||
<p class="status-message">{{ deployStatus }}</p>
|
||||
</div>
|
||||
</form>
|
||||
</div>
|
||||
</div>
|
||||
</BaseLayout>
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import { ref, computed, onMounted } from 'vue';
|
||||
import { useRoute, useRouter } from 'vue-router';
|
||||
import BaseLayout from '../../../components/BaseLayout.vue';
|
||||
import api from '../../../api/axios';
|
||||
|
||||
// Props
|
||||
const props = defineProps({
|
||||
isAuthenticated: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
},
|
||||
identities: {
|
||||
type: Array,
|
||||
default: () => []
|
||||
},
|
||||
tokenBalances: {
|
||||
type: Object,
|
||||
default: () => ({})
|
||||
},
|
||||
isLoadingTokens: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
}
|
||||
});
|
||||
|
||||
// Emits
|
||||
const emit = defineEmits(['auth-action-completed']);
|
||||
|
||||
// Router
|
||||
const route = useRoute();
|
||||
const router = useRouter();
|
||||
|
||||
// Состояние
|
||||
const isDeploying = ref(false);
|
||||
const deployStatus = ref('');
|
||||
const selectedDle = ref(null);
|
||||
const isLoadingDle = ref(false);
|
||||
|
||||
// Данные для деплоя
|
||||
const deployData = ref({
|
||||
moduleName: 'BurnModule',
|
||||
moduleVersion: '1.0.0',
|
||||
moduleDescription: 'Модуль для сжигания токенов DLE через governance',
|
||||
maxBurnPerProposal: 1000000,
|
||||
burnCooldown: 24,
|
||||
allowTreasuryBurn: 'true',
|
||||
allowUserBurn: 'true',
|
||||
deployDescription: 'Предложение о деплое BurnModule для управления сжиганием токенов DLE',
|
||||
votingDuration: 24
|
||||
});
|
||||
|
||||
// Получаем адрес DLE из URL
|
||||
const dleAddress = computed(() => route.query.address);
|
||||
|
||||
// Загрузка данных DLE
|
||||
const loadDleData = async () => {
|
||||
if (!dleAddress.value) return;
|
||||
|
||||
try {
|
||||
isLoadingDle.value = true;
|
||||
const response = await api.post('/blockchain/read-dle-info', {
|
||||
dleAddress: dleAddress.value
|
||||
});
|
||||
|
||||
if (response.data.success) {
|
||||
selectedDle.value = response.data.data;
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('Ошибка загрузки данных DLE:', error);
|
||||
} finally {
|
||||
isLoadingDle.value = false;
|
||||
}
|
||||
};
|
||||
|
||||
// Функция деплоя модуля
|
||||
const deployModule = async () => {
|
||||
if (isDeploying.value) return;
|
||||
|
||||
try {
|
||||
isDeploying.value = true;
|
||||
deployStatus.value = 'Подготовка к деплою...';
|
||||
|
||||
// Здесь будет логика деплоя модуля
|
||||
console.log('Деплой BurnModule:', deployData.value);
|
||||
|
||||
// Временная заглушка
|
||||
await new Promise(resolve => setTimeout(resolve, 2000));
|
||||
|
||||
deployStatus.value = 'Модуль успешно развернут!';
|
||||
|
||||
// Очищаем статус через 3 секунды
|
||||
setTimeout(() => {
|
||||
deployStatus.value = '';
|
||||
}, 3000);
|
||||
|
||||
alert('BurnModule успешно развернут!');
|
||||
|
||||
} catch (error) {
|
||||
console.error('Ошибка деплоя модуля:', error);
|
||||
deployStatus.value = 'Ошибка деплоя модуля';
|
||||
|
||||
setTimeout(() => {
|
||||
deployStatus.value = '';
|
||||
}, 3000);
|
||||
|
||||
alert('Ошибка при деплое модуля');
|
||||
} finally {
|
||||
isDeploying.value = false;
|
||||
}
|
||||
};
|
||||
|
||||
// Загружаем данные при монтировании
|
||||
onMounted(() => {
|
||||
loadDleData();
|
||||
});
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
.module-deploy-page {
|
||||
padding: 20px;
|
||||
max-width: 1200px;
|
||||
margin: 0 auto;
|
||||
}
|
||||
|
||||
.page-header {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: flex-start;
|
||||
margin-bottom: 30px;
|
||||
padding-bottom: 20px;
|
||||
border-bottom: 2px solid #f0f0f0;
|
||||
}
|
||||
|
||||
.header-content h1 {
|
||||
color: var(--color-primary);
|
||||
font-size: 2.5rem;
|
||||
margin: 0 0 10px 0;
|
||||
}
|
||||
|
||||
.header-content p {
|
||||
color: var(--color-grey-dark);
|
||||
font-size: 1.1rem;
|
||||
margin: 0 0 15px 0;
|
||||
}
|
||||
|
||||
.dle-info {
|
||||
display: flex;
|
||||
gap: 15px;
|
||||
align-items: center;
|
||||
}
|
||||
|
||||
.dle-name {
|
||||
font-weight: 600;
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
.dle-address {
|
||||
font-family: monospace;
|
||||
color: var(--color-grey-dark);
|
||||
font-size: 0.9rem;
|
||||
}
|
||||
|
||||
.close-btn {
|
||||
background: none;
|
||||
border: none;
|
||||
font-size: 2rem;
|
||||
color: var(--color-grey-dark);
|
||||
cursor: pointer;
|
||||
padding: 5px;
|
||||
}
|
||||
|
||||
.close-btn:hover {
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
.module-description {
|
||||
margin-bottom: 30px;
|
||||
}
|
||||
|
||||
.description-card {
|
||||
background: #f8f9fa;
|
||||
padding: 25px;
|
||||
border-radius: var(--radius-lg);
|
||||
border: 1px solid #e9ecef;
|
||||
}
|
||||
|
||||
.description-card h3 {
|
||||
color: var(--color-primary);
|
||||
margin: 0 0 20px 0;
|
||||
}
|
||||
|
||||
.description-content h4 {
|
||||
color: var(--color-grey-dark);
|
||||
margin: 20px 0 10px 0;
|
||||
}
|
||||
|
||||
.description-content ul {
|
||||
margin: 10px 0;
|
||||
padding-left: 20px;
|
||||
}
|
||||
|
||||
.description-content li {
|
||||
margin: 5px 0;
|
||||
line-height: 1.5;
|
||||
}
|
||||
|
||||
.deploy-form-section {
|
||||
background: white;
|
||||
padding: 30px;
|
||||
border-radius: var(--radius-lg);
|
||||
box-shadow: 0 2px 10px rgba(0, 0, 0, 0.1);
|
||||
}
|
||||
|
||||
.form-header {
|
||||
margin-bottom: 25px;
|
||||
}
|
||||
|
||||
.form-header h3 {
|
||||
color: var(--color-primary);
|
||||
margin: 0 0 10px 0;
|
||||
}
|
||||
|
||||
.form-header p {
|
||||
color: var(--color-grey-dark);
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
.deploy-form {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 20px;
|
||||
}
|
||||
|
||||
.form-row {
|
||||
display: grid;
|
||||
grid-template-columns: 1fr 1fr;
|
||||
gap: 20px;
|
||||
}
|
||||
|
||||
.form-group {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 8px;
|
||||
}
|
||||
|
||||
.form-group label {
|
||||
font-weight: 600;
|
||||
color: var(--color-grey-dark);
|
||||
}
|
||||
|
||||
.form-group input,
|
||||
.form-group select,
|
||||
.form-group textarea {
|
||||
padding: 12px;
|
||||
border: 1px solid #ddd;
|
||||
border-radius: var(--radius-sm);
|
||||
font-size: 1rem;
|
||||
}
|
||||
|
||||
.form-group textarea {
|
||||
resize: vertical;
|
||||
min-height: 80px;
|
||||
}
|
||||
|
||||
.form-help {
|
||||
font-size: 0.85rem;
|
||||
color: var(--color-grey-dark);
|
||||
font-style: italic;
|
||||
}
|
||||
|
||||
.btn-primary {
|
||||
background: var(--color-primary);
|
||||
color: white;
|
||||
border: none;
|
||||
padding: 15px 30px;
|
||||
border-radius: var(--radius-sm);
|
||||
font-size: 1.1rem;
|
||||
font-weight: 600;
|
||||
cursor: pointer;
|
||||
transition: background-color 0.3s;
|
||||
}
|
||||
|
||||
.btn-primary:hover:not(:disabled) {
|
||||
background: var(--color-primary-dark);
|
||||
}
|
||||
|
||||
.btn-primary:disabled {
|
||||
opacity: 0.6;
|
||||
cursor: not-allowed;
|
||||
}
|
||||
|
||||
.deploy-status {
|
||||
margin-top: 20px;
|
||||
padding: 15px;
|
||||
background: #e8f5e8;
|
||||
border-radius: var(--radius-sm);
|
||||
border-left: 4px solid #28a745;
|
||||
}
|
||||
|
||||
.status-message {
|
||||
margin: 0;
|
||||
font-weight: 600;
|
||||
color: #28a745;
|
||||
}
|
||||
|
||||
@media (max-width: 768px) {
|
||||
.form-row {
|
||||
grid-template-columns: 1fr;
|
||||
}
|
||||
|
||||
.dle-info {
|
||||
flex-direction: column;
|
||||
align-items: flex-start;
|
||||
gap: 5px;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
@@ -0,0 +1,218 @@
|
||||
<!--
|
||||
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="communication-module-deploy">
|
||||
<!-- Заголовок -->
|
||||
<div class="page-header">
|
||||
<div class="header-content">
|
||||
<h1>Деплой CommunicationModule</h1>
|
||||
<p>Коммуникации - сообщения, звонки, история общения между участниками</p>
|
||||
<p v-if="dleAddress" class="dle-address">
|
||||
<strong>DLE:</strong> {{ dleAddress }}
|
||||
</p>
|
||||
</div>
|
||||
<button class="close-btn" @click="router.push('/management/modules')">×</button>
|
||||
</div>
|
||||
|
||||
<!-- Информация о модуле -->
|
||||
<div class="module-info">
|
||||
<div class="info-card">
|
||||
<h3>💬 CommunicationModule</h3>
|
||||
<div class="info-grid">
|
||||
<div class="info-item">
|
||||
<strong>Назначение:</strong> Коммуникации между участниками DLE
|
||||
</div>
|
||||
<div class="info-item">
|
||||
<strong>Функции:</strong> Сообщения, аудио/видео звонки, история общения
|
||||
</div>
|
||||
<div class="info-item">
|
||||
<strong>Безопасность:</strong> Кворум для коммуникационных операций
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Форма деплоя будет добавлена позже -->
|
||||
<div class="deploy-form-placeholder">
|
||||
<div class="placeholder-content">
|
||||
<h3>🚧 Форма деплоя в разработке</h3>
|
||||
<p>Здесь будет форма для деплоя CommunicationModule</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
</div>
|
||||
</BaseLayout>
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import { defineProps, defineEmits, ref, onMounted } from 'vue';
|
||||
import { useRouter, useRoute } from 'vue-router';
|
||||
import BaseLayout from '../../../components/BaseLayout.vue';
|
||||
|
||||
// Определяем props
|
||||
const props = defineProps({
|
||||
isAuthenticated: { type: Boolean, default: false },
|
||||
identities: { type: Array, default: () => [] },
|
||||
tokenBalances: { type: Object, default: () => ({}) },
|
||||
isLoadingTokens: { type: Boolean, default: false }
|
||||
});
|
||||
|
||||
// Определяем emits
|
||||
const emit = defineEmits(['auth-action-completed']);
|
||||
|
||||
const router = useRouter();
|
||||
const route = useRoute();
|
||||
|
||||
// Состояние
|
||||
const isLoading = ref(false);
|
||||
const dleAddress = ref(route.query.address || null);
|
||||
|
||||
// Инициализация
|
||||
onMounted(() => {
|
||||
console.log('[CommunicationModuleDeployView] Страница загружена');
|
||||
});
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
.communication-module-deploy {
|
||||
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: center;
|
||||
margin-bottom: 30px;
|
||||
padding-bottom: 20px;
|
||||
border-bottom: 2px solid #f0f0f0;
|
||||
}
|
||||
|
||||
.page-header h1 {
|
||||
color: var(--color-primary);
|
||||
font-size: 2rem;
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
.page-header p {
|
||||
margin: 10px 0 0 0;
|
||||
color: #666;
|
||||
}
|
||||
|
||||
.dle-address {
|
||||
margin-top: 10px !important;
|
||||
font-family: monospace;
|
||||
background: #f8f9fa;
|
||||
padding: 8px 12px;
|
||||
border-radius: var(--radius-sm);
|
||||
border: 1px solid #dee2e6;
|
||||
}
|
||||
|
||||
.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;
|
||||
}
|
||||
|
||||
.close-btn:hover {
|
||||
background: #f0f0f0;
|
||||
color: #333;
|
||||
}
|
||||
|
||||
/* Информация о модуле */
|
||||
.module-info {
|
||||
margin-bottom: 30px;
|
||||
}
|
||||
|
||||
.info-card {
|
||||
background: #f8f9fa;
|
||||
border-radius: var(--radius-md);
|
||||
padding: 20px;
|
||||
border: 1px solid #e9ecef;
|
||||
}
|
||||
|
||||
.info-card h3 {
|
||||
margin: 0 0 15px 0;
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
.info-grid {
|
||||
display: grid;
|
||||
grid-template-columns: repeat(auto-fit, minmax(250px, 1fr));
|
||||
gap: 15px;
|
||||
}
|
||||
|
||||
.info-item {
|
||||
padding: 10px;
|
||||
background: white;
|
||||
border-radius: var(--radius-sm);
|
||||
border: 1px solid #dee2e6;
|
||||
}
|
||||
|
||||
.info-item strong {
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
/* Плейсхолдер для формы */
|
||||
.deploy-form-placeholder {
|
||||
background: #f8f9fa;
|
||||
border-radius: var(--radius-md);
|
||||
padding: 40px;
|
||||
text-align: center;
|
||||
border: 2px dashed #dee2e6;
|
||||
}
|
||||
|
||||
.placeholder-content h3 {
|
||||
color: var(--color-primary);
|
||||
margin-bottom: 10px;
|
||||
}
|
||||
|
||||
.placeholder-content p {
|
||||
color: #666;
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
/* Адаптивность */
|
||||
@media (max-width: 768px) {
|
||||
.info-grid {
|
||||
grid-template-columns: 1fr;
|
||||
}
|
||||
|
||||
.page-header {
|
||||
flex-direction: column;
|
||||
align-items: flex-start;
|
||||
gap: 15px;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
@@ -0,0 +1,663 @@
|
||||
<!--
|
||||
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="module-deploy-page">
|
||||
<!-- Заголовок -->
|
||||
<div class="page-header">
|
||||
<div class="header-content">
|
||||
<h1>🏛️ Деплой InheritanceModule</h1>
|
||||
<p>Модуль наследования токенов DLE - защита активов и автоматическая передача наследникам</p>
|
||||
<div v-if="selectedDle" class="dle-info">
|
||||
<span class="dle-name">{{ selectedDle.name }} ({{ selectedDle.symbol }})</span>
|
||||
<span class="dle-address">{{ selectedDle.dleAddress }}</span>
|
||||
</div>
|
||||
</div>
|
||||
<button class="close-btn" @click="router.push(`/management/modules?address=${route.query.address}`)">×</button>
|
||||
</div>
|
||||
|
||||
<!-- Описание модуля -->
|
||||
<div class="module-description">
|
||||
<div class="description-card">
|
||||
<h3>📋 Описание InheritanceModule</h3>
|
||||
<div class="description-content">
|
||||
<p><strong>InheritanceModule</strong> - это модуль для автоматической передачи токенов DLE наследникам в случае смерти или недееспособности токенхолдера.</p>
|
||||
|
||||
<h4>🔧 Основная функциональность:</h4>
|
||||
<ul>
|
||||
<li><strong>Назначение наследников:</strong> Токенхолдеры могут указать один или несколько наследников</li>
|
||||
<li><strong>Распределение долей:</strong> Настройка процентного распределения токенов между наследниками</li>
|
||||
<li><strong>Условия активации:</strong> Настройка условий для передачи токенов (смерть, недееспособность)</li>
|
||||
<li><strong>Временные ограничения:</strong> Установка минимального периода владения токенами</li>
|
||||
<li><strong>Множественные наследники:</strong> Поддержка сложных схем наследования</li>
|
||||
<li><strong>Отзыв и изменение:</strong> Возможность изменения наследников в любое время</li>
|
||||
</ul>
|
||||
|
||||
<h4>🏛️ Юридические аспекты:</h4>
|
||||
<ul>
|
||||
<li><strong>Соответствие законам:</strong> Интеграция с юридическими системами наследования</li>
|
||||
<li><strong>Документооборот:</strong> Автоматическое создание юридических документов</li>
|
||||
<li><strong>Подтверждение смерти:</strong> Интеграция с государственными реестрами</li>
|
||||
<li><strong>Споры и оспаривание:</strong> Механизмы разрешения споров о наследстве</li>
|
||||
<li><strong>Налоговые обязательства:</strong> Автоматический расчет налогов на наследство</li>
|
||||
</ul>
|
||||
|
||||
<h4>🔐 Безопасность и контроль:</h4>
|
||||
<ul>
|
||||
<li>Все изменения наследников требуют подтверждения через governance</li>
|
||||
<li>Криптографическая защита данных о наследниках</li>
|
||||
<li>Аудит всех операций наследования</li>
|
||||
<li>Возможность экстренной блокировки в случае споров</li>
|
||||
<li>Интеграция с системой идентификации для подтверждения личности</li>
|
||||
</ul>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Архитектура модуля -->
|
||||
<div class="module-architecture">
|
||||
<div class="architecture-card">
|
||||
<h3>🏗️ Архитектура InheritanceModule</h3>
|
||||
<div class="architecture-content">
|
||||
<div class="architecture-diagram">
|
||||
<div class="diagram-row">
|
||||
<div class="diagram-item tokenholder">
|
||||
<h5>👤 Токенхолдер</h5>
|
||||
<ul>
|
||||
<li>Назначает наследников</li>
|
||||
<li>Устанавливает доли</li>
|
||||
<li>Управляет условиями</li>
|
||||
<li>Может отозвать</li>
|
||||
</ul>
|
||||
</div>
|
||||
<div class="diagram-arrow">→</div>
|
||||
<div class="diagram-item inheritance">
|
||||
<h5>🏛️ InheritanceModule</h5>
|
||||
<ul>
|
||||
<li>Хранит данные наследников</li>
|
||||
<li>Проверяет условия</li>
|
||||
<li>Выполняет передачу</li>
|
||||
<li>Ведет аудит</li>
|
||||
</ul>
|
||||
</div>
|
||||
<div class="diagram-arrow">→</div>
|
||||
<div class="diagram-item heirs">
|
||||
<h5>👥 Наследники</h5>
|
||||
<ul>
|
||||
<li>Получают токены</li>
|
||||
<li>Подтверждают получение</li>
|
||||
<li>Управляют наследством</li>
|
||||
<li>Планируют налоги</li>
|
||||
</ul>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Типы наследования -->
|
||||
<div class="inheritance-types">
|
||||
<div class="types-card">
|
||||
<h3>📊 Типы наследования</h3>
|
||||
<div class="types-grid">
|
||||
<div class="type-item">
|
||||
<h4>👨👩👧👦 Семейное наследование</h4>
|
||||
<p>Передача токенов членам семьи согласно традиционным схемам</p>
|
||||
<ul>
|
||||
<li>Супруг/супруга (50%)</li>
|
||||
<li>Дети (равные доли)</li>
|
||||
<li>Родители (при отсутствии детей)</li>
|
||||
<li>Братья/сестры (при отсутствии родителей)</li>
|
||||
</ul>
|
||||
</div>
|
||||
|
||||
<div class="type-item">
|
||||
<h4>🏢 Корпоративное наследование</h4>
|
||||
<p>Передача токенов в рамках бизнес-структур и организаций</p>
|
||||
<ul>
|
||||
<li>Партнеры по бизнесу</li>
|
||||
<li>Ключевые сотрудники</li>
|
||||
<li>Дочерние компании</li>
|
||||
<li>Благотворительные фонды</li>
|
||||
</ul>
|
||||
</div>
|
||||
|
||||
<div class="type-item">
|
||||
<h4>🎯 Целевое наследование</h4>
|
||||
<p>Передача токенов для достижения конкретных целей</p>
|
||||
<ul>
|
||||
<li>Образовательные учреждения</li>
|
||||
<li>Исследовательские проекты</li>
|
||||
<li>Экологические инициативы</li>
|
||||
<li>Социальные программы</li>
|
||||
</ul>
|
||||
</div>
|
||||
|
||||
<div class="type-item">
|
||||
<h4>⏰ Условное наследование</h4>
|
||||
<p>Передача токенов при выполнении определенных условий</p>
|
||||
<ul>
|
||||
<li>Достижение определенного возраста</li>
|
||||
<li>Завершение образования</li>
|
||||
<li>Создание семьи</li>
|
||||
<li>Достижение карьерных целей</li>
|
||||
</ul>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Примеры использования -->
|
||||
<div class="usage-examples">
|
||||
<div class="examples-card">
|
||||
<h3>💡 Примеры использования</h3>
|
||||
<div class="examples-content">
|
||||
<div class="example-item">
|
||||
<h4>👨👩👧👦 Семейное планирование</h4>
|
||||
<div class="example-code">
|
||||
<pre><code>// Назначение наследников для семьи
|
||||
function setFamilyInheritance() {
|
||||
setHeir(spouse, 50); // Супруг 50%
|
||||
setHeir(son, 25); // Сын 25%
|
||||
setHeir(daughter, 25); // Дочь 25%
|
||||
setActivationCondition("death");
|
||||
}</code></pre>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="example-item">
|
||||
<h4>🏢 Бизнес-преемственность</h4>
|
||||
<div class="example-code">
|
||||
<pre><code>// Передача бизнеса партнеру
|
||||
function setBusinessInheritance() {
|
||||
setHeir(businessPartner, 100); // Партнер 100%
|
||||
setActivationCondition("death");
|
||||
setTimeLock(365 days); // Минимум 1 год владения
|
||||
}</code></pre>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="example-item">
|
||||
<h4>🎯 Благотворительное наследование</h4>
|
||||
<div class="example-code">
|
||||
<pre><code>// Передача в благотворительный фонд
|
||||
function setCharityInheritance() {
|
||||
setHeir(environmentalFund, 70); // Экологический фонд 70%
|
||||
setHeir(educationFund, 30); // Образовательный фонд 30%
|
||||
setActivationCondition("death");
|
||||
}</code></pre>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Юридические аспекты -->
|
||||
<div class="legal-aspects">
|
||||
<div class="legal-card">
|
||||
<h3>⚖️ Юридические аспекты</h3>
|
||||
<div class="legal-content">
|
||||
<div class="legal-section">
|
||||
<h4>📜 Соответствие законодательству</h4>
|
||||
<ul>
|
||||
<li><strong>Гражданский кодекс:</strong> Соответствие нормам наследования</li>
|
||||
<li><strong>Налоговый кодекс:</strong> Правильный расчет налогов на наследство</li>
|
||||
<li><strong>Семейный кодекс:</strong> Учет семейных обязательств</li>
|
||||
<li><strong>Международное право:</strong> Наследование в разных юрисдикциях</li>
|
||||
</ul>
|
||||
</div>
|
||||
|
||||
<div class="legal-section">
|
||||
<h4>🔍 Процедура подтверждения</h4>
|
||||
<ul>
|
||||
<li><strong>Свидетельство о смерти:</strong> Официальное подтверждение</li>
|
||||
<li><strong>Медицинское заключение:</strong> При недееспособности</li>
|
||||
<li><strong>Судебное решение:</strong> При спорах о наследстве</li>
|
||||
<li><strong>Нотариальное заверение:</strong> Документов о наследниках</li>
|
||||
</ul>
|
||||
</div>
|
||||
|
||||
<div class="legal-section">
|
||||
<h4>💰 Налоговые обязательства</h4>
|
||||
<ul>
|
||||
<li><strong>Налог на наследство:</strong> Автоматический расчет</li>
|
||||
<li><strong>НДФЛ:</strong> При получении токенов</li>
|
||||
<li><strong>Отчетность:</strong> Автоматическая подача деклараций</li>
|
||||
<li><strong>Льготы:</strong> Учет налоговых льгот для наследников</li>
|
||||
</ul>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Статус разработки -->
|
||||
<div class="development-status">
|
||||
<div class="status-card">
|
||||
<h3>🚧 Статус разработки</h3>
|
||||
<div class="status-content">
|
||||
<p><strong>InheritanceModule находится в стадии планирования.</strong></p>
|
||||
<p>Модуль будет включать:</p>
|
||||
<ul>
|
||||
<li>✅ Систему назначения наследников</li>
|
||||
<li>✅ Управление долями и условиями</li>
|
||||
<li>✅ Интеграцию с юридическими системами</li>
|
||||
<li>✅ Автоматическую передачу токенов</li>
|
||||
<li>✅ Налоговые расчеты</li>
|
||||
<li>✅ Аудит и мониторинг</li>
|
||||
<li>✅ Разрешение споров</li>
|
||||
</ul>
|
||||
<p><em>Модуль будет доступен в следующих обновлениях DLE.</em></p>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</BaseLayout>
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import { ref, computed, onMounted } from 'vue';
|
||||
import { useRoute, useRouter } from 'vue-router';
|
||||
import BaseLayout from '../../../components/BaseLayout.vue';
|
||||
import api from '../../../api/axios';
|
||||
|
||||
// Props
|
||||
const props = defineProps({
|
||||
isAuthenticated: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
},
|
||||
identities: {
|
||||
type: Array,
|
||||
default: () => []
|
||||
},
|
||||
tokenBalances: {
|
||||
type: Object,
|
||||
default: () => ({})
|
||||
},
|
||||
isLoadingTokens: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
}
|
||||
});
|
||||
|
||||
// Emits
|
||||
const emit = defineEmits(['auth-action-completed']);
|
||||
|
||||
// Router
|
||||
const route = useRoute();
|
||||
const router = useRouter();
|
||||
|
||||
// Состояние
|
||||
const selectedDle = ref(null);
|
||||
const isLoadingDle = ref(false);
|
||||
|
||||
// Получаем адрес DLE из URL
|
||||
const dleAddress = computed(() => route.query.address);
|
||||
|
||||
// Загрузка данных DLE
|
||||
const loadDleData = async () => {
|
||||
if (!dleAddress.value) return;
|
||||
|
||||
try {
|
||||
isLoadingDle.value = true;
|
||||
const response = await api.post('/blockchain/read-dle-info', {
|
||||
dleAddress: dleAddress.value
|
||||
});
|
||||
|
||||
if (response.data.success) {
|
||||
selectedDle.value = response.data.data;
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('Ошибка загрузки данных DLE:', error);
|
||||
} finally {
|
||||
isLoadingDle.value = false;
|
||||
}
|
||||
};
|
||||
|
||||
// Загружаем данные при монтировании
|
||||
onMounted(() => {
|
||||
loadDleData();
|
||||
});
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
.module-deploy-page {
|
||||
padding: 20px;
|
||||
max-width: 1200px;
|
||||
margin: 0 auto;
|
||||
}
|
||||
|
||||
.page-header {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: flex-start;
|
||||
margin-bottom: 30px;
|
||||
padding-bottom: 20px;
|
||||
border-bottom: 2px solid #f0f0f0;
|
||||
}
|
||||
|
||||
.header-content h1 {
|
||||
color: var(--color-primary);
|
||||
font-size: 2.5rem;
|
||||
margin: 0 0 10px 0;
|
||||
}
|
||||
|
||||
.header-content p {
|
||||
color: var(--color-grey-dark);
|
||||
font-size: 1.1rem;
|
||||
margin: 0 0 15px 0;
|
||||
}
|
||||
|
||||
.dle-info {
|
||||
display: flex;
|
||||
gap: 15px;
|
||||
align-items: center;
|
||||
}
|
||||
|
||||
.dle-name {
|
||||
font-weight: 600;
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
.dle-address {
|
||||
font-family: monospace;
|
||||
color: var(--color-grey-dark);
|
||||
font-size: 0.9rem;
|
||||
}
|
||||
|
||||
.close-btn {
|
||||
background: none;
|
||||
border: none;
|
||||
font-size: 2rem;
|
||||
color: var(--color-grey-dark);
|
||||
cursor: pointer;
|
||||
padding: 5px;
|
||||
}
|
||||
|
||||
.close-btn:hover {
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
.module-description,
|
||||
.module-architecture,
|
||||
.inheritance-types,
|
||||
.usage-examples,
|
||||
.legal-aspects,
|
||||
.development-status {
|
||||
margin-bottom: 30px;
|
||||
}
|
||||
|
||||
.description-card,
|
||||
.architecture-card,
|
||||
.types-card,
|
||||
.examples-card,
|
||||
.legal-card,
|
||||
.status-card {
|
||||
background: #f8f9fa;
|
||||
padding: 25px;
|
||||
border-radius: var(--radius-lg);
|
||||
border: 1px solid #e9ecef;
|
||||
}
|
||||
|
||||
.description-card h3,
|
||||
.architecture-card h3,
|
||||
.types-card h3,
|
||||
.examples-card h3,
|
||||
.legal-card h3,
|
||||
.status-card h3 {
|
||||
color: var(--color-primary);
|
||||
margin: 0 0 20px 0;
|
||||
}
|
||||
|
||||
.description-content h4 {
|
||||
color: var(--color-grey-dark);
|
||||
margin: 20px 0 10px 0;
|
||||
}
|
||||
|
||||
.description-content ul {
|
||||
margin: 10px 0;
|
||||
padding-left: 20px;
|
||||
}
|
||||
|
||||
.description-content li {
|
||||
margin: 5px 0;
|
||||
line-height: 1.5;
|
||||
}
|
||||
|
||||
/* Архитектурная диаграмма */
|
||||
.architecture-diagram {
|
||||
margin: 20px 0;
|
||||
}
|
||||
|
||||
.diagram-row {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: space-between;
|
||||
gap: 20px;
|
||||
margin-bottom: 20px;
|
||||
}
|
||||
|
||||
.diagram-item {
|
||||
flex: 1;
|
||||
padding: 20px;
|
||||
border-radius: var(--radius-sm);
|
||||
text-align: center;
|
||||
min-height: 150px;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
justify-content: center;
|
||||
}
|
||||
|
||||
.diagram-item.tokenholder {
|
||||
background: #e8f5e8;
|
||||
border: 2px solid #4caf50;
|
||||
}
|
||||
|
||||
.diagram-item.inheritance {
|
||||
background: #fff3e0;
|
||||
border: 2px solid #ff9800;
|
||||
}
|
||||
|
||||
.diagram-item.heirs {
|
||||
background: #f3e5f5;
|
||||
border: 2px solid #9c27b0;
|
||||
}
|
||||
|
||||
.diagram-item h5 {
|
||||
margin: 0 0 15px 0;
|
||||
font-weight: 600;
|
||||
}
|
||||
|
||||
.diagram-item ul {
|
||||
margin: 0;
|
||||
padding: 0;
|
||||
list-style: none;
|
||||
font-size: 0.9rem;
|
||||
}
|
||||
|
||||
.diagram-item li {
|
||||
margin: 5px 0;
|
||||
}
|
||||
|
||||
.diagram-arrow {
|
||||
font-size: 2rem;
|
||||
color: var(--color-primary);
|
||||
font-weight: bold;
|
||||
}
|
||||
|
||||
/* Типы наследования */
|
||||
.types-grid {
|
||||
display: grid;
|
||||
grid-template-columns: repeat(auto-fit, minmax(280px, 1fr));
|
||||
gap: 20px;
|
||||
margin-top: 20px;
|
||||
}
|
||||
|
||||
.type-item {
|
||||
background: white;
|
||||
padding: 20px;
|
||||
border-radius: var(--radius-sm);
|
||||
border: 1px solid #e9ecef;
|
||||
}
|
||||
|
||||
.type-item h4 {
|
||||
color: var(--color-primary);
|
||||
margin: 0 0 10px 0;
|
||||
}
|
||||
|
||||
.type-item p {
|
||||
color: var(--color-grey-dark);
|
||||
margin: 0 0 15px 0;
|
||||
font-size: 0.9rem;
|
||||
}
|
||||
|
||||
.type-item ul {
|
||||
margin: 0;
|
||||
padding-left: 20px;
|
||||
font-size: 0.9rem;
|
||||
}
|
||||
|
||||
.type-item li {
|
||||
margin: 5px 0;
|
||||
color: var(--color-grey-dark);
|
||||
}
|
||||
|
||||
/* Примеры использования */
|
||||
.examples-content {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 20px;
|
||||
}
|
||||
|
||||
.example-item {
|
||||
background: white;
|
||||
padding: 20px;
|
||||
border-radius: var(--radius-sm);
|
||||
border: 1px solid #e9ecef;
|
||||
}
|
||||
|
||||
.example-item h4 {
|
||||
color: var(--color-primary);
|
||||
margin: 0 0 15px 0;
|
||||
}
|
||||
|
||||
.example-code {
|
||||
background: #f8f9fa;
|
||||
border: 1px solid #e9ecef;
|
||||
border-radius: var(--radius-sm);
|
||||
padding: 15px;
|
||||
overflow-x: auto;
|
||||
}
|
||||
|
||||
.example-code pre {
|
||||
margin: 0;
|
||||
font-family: 'Courier New', monospace;
|
||||
font-size: 0.9rem;
|
||||
color: #333;
|
||||
}
|
||||
|
||||
.example-code code {
|
||||
background: none;
|
||||
padding: 0;
|
||||
}
|
||||
|
||||
/* Юридические аспекты */
|
||||
.legal-content {
|
||||
display: grid;
|
||||
grid-template-columns: repeat(auto-fit, minmax(300px, 1fr));
|
||||
gap: 20px;
|
||||
}
|
||||
|
||||
.legal-section {
|
||||
background: white;
|
||||
padding: 20px;
|
||||
border-radius: var(--radius-sm);
|
||||
border: 1px solid #e9ecef;
|
||||
}
|
||||
|
||||
.legal-section h4 {
|
||||
color: var(--color-primary);
|
||||
margin: 0 0 15px 0;
|
||||
}
|
||||
|
||||
.legal-section ul {
|
||||
margin: 0;
|
||||
padding-left: 20px;
|
||||
}
|
||||
|
||||
.legal-section li {
|
||||
margin: 8px 0;
|
||||
line-height: 1.5;
|
||||
color: var(--color-grey-dark);
|
||||
}
|
||||
|
||||
/* Статус разработки */
|
||||
.status-content {
|
||||
background: white;
|
||||
padding: 20px;
|
||||
border-radius: var(--radius-sm);
|
||||
border: 1px solid #e9ecef;
|
||||
}
|
||||
|
||||
.status-content p {
|
||||
margin: 0 0 15px 0;
|
||||
line-height: 1.6;
|
||||
}
|
||||
|
||||
.status-content ul {
|
||||
margin: 15px 0;
|
||||
padding-left: 20px;
|
||||
}
|
||||
|
||||
.status-content li {
|
||||
margin: 5px 0;
|
||||
color: var(--color-grey-dark);
|
||||
}
|
||||
|
||||
.status-content em {
|
||||
color: var(--color-primary);
|
||||
font-style: italic;
|
||||
}
|
||||
|
||||
@media (max-width: 768px) {
|
||||
.diagram-row {
|
||||
flex-direction: column;
|
||||
gap: 15px;
|
||||
}
|
||||
|
||||
.diagram-arrow {
|
||||
transform: rotate(90deg);
|
||||
}
|
||||
|
||||
.types-grid {
|
||||
grid-template-columns: 1fr;
|
||||
}
|
||||
|
||||
.legal-content {
|
||||
grid-template-columns: 1fr;
|
||||
}
|
||||
|
||||
.dle-info {
|
||||
flex-direction: column;
|
||||
align-items: flex-start;
|
||||
gap: 5px;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
485
frontend/src/views/smartcontracts/modules/MintModuleDeploy.vue
Normal file
485
frontend/src/views/smartcontracts/modules/MintModuleDeploy.vue
Normal file
@@ -0,0 +1,485 @@
|
||||
<!--
|
||||
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="module-deploy-page">
|
||||
<!-- Заголовок -->
|
||||
<div class="page-header">
|
||||
<div class="header-content">
|
||||
<h1>🚀 Деплой MintModule</h1>
|
||||
<p>Модуль для выпуска новых токенов DLE через governance</p>
|
||||
<div v-if="selectedDle" class="dle-info">
|
||||
<span class="dle-name">{{ selectedDle.name }} ({{ selectedDle.symbol }})</span>
|
||||
<span class="dle-address">{{ selectedDle.dleAddress }}</span>
|
||||
</div>
|
||||
</div>
|
||||
<button class="close-btn" @click="router.push(`/management/modules?address=${route.query.address}`)">×</button>
|
||||
</div>
|
||||
|
||||
<!-- Описание модуля -->
|
||||
<div class="module-description">
|
||||
<div class="description-card">
|
||||
<h3>📋 Описание MintModule</h3>
|
||||
<div class="description-content">
|
||||
<p><strong>MintModule</strong> - это модуль для управления выпуском новых токенов DLE через систему governance.</p>
|
||||
|
||||
<h4>🔧 Функциональность:</h4>
|
||||
<ul>
|
||||
<li><strong>Выпуск токенов:</strong> Создание дополнительных токенов DLE</li>
|
||||
<li><strong>Governance:</strong> Все операции требуют голосования и кворума</li>
|
||||
<li><strong>Безопасность:</strong> Контролируемый выпуск через коллективные решения</li>
|
||||
<li><strong>Прозрачность:</strong> Все операции записываются в блокчейн</li>
|
||||
</ul>
|
||||
|
||||
<h4>⚠️ Важные особенности:</h4>
|
||||
<ul>
|
||||
<li>Выпуск токенов возможен только через предложения и голосование</li>
|
||||
<li>Новые токены могут быть распределены между участниками или добавлены в казну DLE</li>
|
||||
<li>Все операции требуют достижения кворума</li>
|
||||
<li>История всех выпусков сохраняется в блокчейне</li>
|
||||
</ul>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Форма деплоя -->
|
||||
<div class="deploy-form-section">
|
||||
<div class="form-header">
|
||||
<h3>⚙️ Настройки деплоя</h3>
|
||||
<p>Настройте параметры для деплоя MintModule</p>
|
||||
</div>
|
||||
|
||||
<form @submit.prevent="deployModule" class="deploy-form">
|
||||
<div class="form-row">
|
||||
<div class="form-group">
|
||||
<label for="moduleName">Название модуля:</label>
|
||||
<input
|
||||
id="moduleName"
|
||||
v-model="deployData.moduleName"
|
||||
type="text"
|
||||
placeholder="MintModule"
|
||||
required
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="moduleVersion">Версия модуля:</label>
|
||||
<input
|
||||
id="moduleVersion"
|
||||
v-model="deployData.moduleVersion"
|
||||
type="text"
|
||||
placeholder="1.0.0"
|
||||
required
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="moduleDescription">Описание модуля:</label>
|
||||
<textarea
|
||||
id="moduleDescription"
|
||||
v-model="deployData.moduleDescription"
|
||||
placeholder="Модуль для выпуска новых токенов DLE через governance..."
|
||||
rows="3"
|
||||
required
|
||||
></textarea>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="maxMintPerProposal">Максимальный выпуск за одно предложение:</label>
|
||||
<input
|
||||
id="maxMintPerProposal"
|
||||
v-model="deployData.maxMintPerProposal"
|
||||
type="number"
|
||||
min="1"
|
||||
step="1"
|
||||
placeholder="1000000"
|
||||
required
|
||||
/>
|
||||
<small class="form-help">Максимальное количество токенов, которое можно выпустить за одно предложение</small>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="mintCooldown">Кулдаун между выпусками (часы):</label>
|
||||
<input
|
||||
id="mintCooldown"
|
||||
v-model="deployData.mintCooldown"
|
||||
type="number"
|
||||
min="0"
|
||||
step="1"
|
||||
placeholder="24"
|
||||
required
|
||||
/>
|
||||
<small class="form-help">Минимальное время между успешными выпусками токенов</small>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="deployDescription">Описание предложения для деплоя:</label>
|
||||
<textarea
|
||||
id="deployDescription"
|
||||
v-model="deployData.deployDescription"
|
||||
placeholder="Предложение о деплое MintModule для управления выпуском токенов DLE..."
|
||||
rows="3"
|
||||
required
|
||||
></textarea>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="votingDuration">Длительность голосования (часы):</label>
|
||||
<input
|
||||
id="votingDuration"
|
||||
v-model="deployData.votingDuration"
|
||||
type="number"
|
||||
min="1"
|
||||
max="168"
|
||||
placeholder="24"
|
||||
required
|
||||
/>
|
||||
<small class="form-help">Время для голосования по предложению деплоя (1-168 часов)</small>
|
||||
</div>
|
||||
|
||||
<button type="submit" class="btn-primary" :disabled="isDeploying">
|
||||
{{ isDeploying ? 'Деплой...' : 'Деплой MintModule' }}
|
||||
</button>
|
||||
|
||||
<!-- Статус деплоя -->
|
||||
<div v-if="deployStatus" class="deploy-status">
|
||||
<p class="status-message">{{ deployStatus }}</p>
|
||||
</div>
|
||||
</form>
|
||||
</div>
|
||||
</div>
|
||||
</BaseLayout>
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import { ref, computed, onMounted } from 'vue';
|
||||
import { useRoute, useRouter } from 'vue-router';
|
||||
import BaseLayout from '../../../components/BaseLayout.vue';
|
||||
import api from '../../../api/axios';
|
||||
|
||||
// Props
|
||||
const props = defineProps({
|
||||
isAuthenticated: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
},
|
||||
identities: {
|
||||
type: Array,
|
||||
default: () => []
|
||||
},
|
||||
tokenBalances: {
|
||||
type: Object,
|
||||
default: () => ({})
|
||||
},
|
||||
isLoadingTokens: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
}
|
||||
});
|
||||
|
||||
// Emits
|
||||
const emit = defineEmits(['auth-action-completed']);
|
||||
|
||||
// Router
|
||||
const route = useRoute();
|
||||
const router = useRouter();
|
||||
|
||||
// Состояние
|
||||
const isDeploying = ref(false);
|
||||
const deployStatus = ref('');
|
||||
const selectedDle = ref(null);
|
||||
const isLoadingDle = ref(false);
|
||||
|
||||
// Данные для деплоя
|
||||
const deployData = ref({
|
||||
moduleName: 'MintModule',
|
||||
moduleVersion: '1.0.0',
|
||||
moduleDescription: 'Модуль для выпуска новых токенов DLE через governance',
|
||||
maxMintPerProposal: 1000000,
|
||||
mintCooldown: 24,
|
||||
deployDescription: 'Предложение о деплое MintModule для управления выпуском токенов DLE',
|
||||
votingDuration: 24
|
||||
});
|
||||
|
||||
// Получаем адрес DLE из URL
|
||||
const dleAddress = computed(() => route.query.address);
|
||||
|
||||
// Загрузка данных DLE
|
||||
const loadDleData = async () => {
|
||||
if (!dleAddress.value) return;
|
||||
|
||||
try {
|
||||
isLoadingDle.value = true;
|
||||
const response = await api.post('/blockchain/read-dle-info', {
|
||||
dleAddress: dleAddress.value
|
||||
});
|
||||
|
||||
if (response.data.success) {
|
||||
selectedDle.value = response.data.data;
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('Ошибка загрузки данных DLE:', error);
|
||||
} finally {
|
||||
isLoadingDle.value = false;
|
||||
}
|
||||
};
|
||||
|
||||
// Функция деплоя модуля
|
||||
const deployModule = async () => {
|
||||
if (isDeploying.value) return;
|
||||
|
||||
try {
|
||||
isDeploying.value = true;
|
||||
deployStatus.value = 'Подготовка к деплою...';
|
||||
|
||||
// Здесь будет логика деплоя модуля
|
||||
console.log('Деплой MintModule:', deployData.value);
|
||||
|
||||
// Временная заглушка
|
||||
await new Promise(resolve => setTimeout(resolve, 2000));
|
||||
|
||||
deployStatus.value = 'Модуль успешно развернут!';
|
||||
|
||||
// Очищаем статус через 3 секунды
|
||||
setTimeout(() => {
|
||||
deployStatus.value = '';
|
||||
}, 3000);
|
||||
|
||||
alert('MintModule успешно развернут!');
|
||||
|
||||
} catch (error) {
|
||||
console.error('Ошибка деплоя модуля:', error);
|
||||
deployStatus.value = 'Ошибка деплоя модуля';
|
||||
|
||||
setTimeout(() => {
|
||||
deployStatus.value = '';
|
||||
}, 3000);
|
||||
|
||||
alert('Ошибка при деплое модуля');
|
||||
} finally {
|
||||
isDeploying.value = false;
|
||||
}
|
||||
};
|
||||
|
||||
// Загружаем данные при монтировании
|
||||
onMounted(() => {
|
||||
loadDleData();
|
||||
});
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
.module-deploy-page {
|
||||
padding: 20px;
|
||||
max-width: 1200px;
|
||||
margin: 0 auto;
|
||||
}
|
||||
|
||||
.page-header {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: flex-start;
|
||||
margin-bottom: 30px;
|
||||
padding-bottom: 20px;
|
||||
border-bottom: 2px solid #f0f0f0;
|
||||
}
|
||||
|
||||
.header-content h1 {
|
||||
color: var(--color-primary);
|
||||
font-size: 2.5rem;
|
||||
margin: 0 0 10px 0;
|
||||
}
|
||||
|
||||
.header-content p {
|
||||
color: var(--color-grey-dark);
|
||||
font-size: 1.1rem;
|
||||
margin: 0 0 15px 0;
|
||||
}
|
||||
|
||||
.dle-info {
|
||||
display: flex;
|
||||
gap: 15px;
|
||||
align-items: center;
|
||||
}
|
||||
|
||||
.dle-name {
|
||||
font-weight: 600;
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
.dle-address {
|
||||
font-family: monospace;
|
||||
color: var(--color-grey-dark);
|
||||
font-size: 0.9rem;
|
||||
}
|
||||
|
||||
.close-btn {
|
||||
background: none;
|
||||
border: none;
|
||||
font-size: 2rem;
|
||||
color: var(--color-grey-dark);
|
||||
cursor: pointer;
|
||||
padding: 5px;
|
||||
}
|
||||
|
||||
.close-btn:hover {
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
.module-description {
|
||||
margin-bottom: 30px;
|
||||
}
|
||||
|
||||
.description-card {
|
||||
background: #f8f9fa;
|
||||
padding: 25px;
|
||||
border-radius: var(--radius-lg);
|
||||
border: 1px solid #e9ecef;
|
||||
}
|
||||
|
||||
.description-card h3 {
|
||||
color: var(--color-primary);
|
||||
margin: 0 0 20px 0;
|
||||
}
|
||||
|
||||
.description-content h4 {
|
||||
color: var(--color-grey-dark);
|
||||
margin: 20px 0 10px 0;
|
||||
}
|
||||
|
||||
.description-content ul {
|
||||
margin: 10px 0;
|
||||
padding-left: 20px;
|
||||
}
|
||||
|
||||
.description-content li {
|
||||
margin: 5px 0;
|
||||
line-height: 1.5;
|
||||
}
|
||||
|
||||
.deploy-form-section {
|
||||
background: white;
|
||||
padding: 30px;
|
||||
border-radius: var(--radius-lg);
|
||||
box-shadow: 0 2px 10px rgba(0, 0, 0, 0.1);
|
||||
}
|
||||
|
||||
.form-header {
|
||||
margin-bottom: 25px;
|
||||
}
|
||||
|
||||
.form-header h3 {
|
||||
color: var(--color-primary);
|
||||
margin: 0 0 10px 0;
|
||||
}
|
||||
|
||||
.form-header p {
|
||||
color: var(--color-grey-dark);
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
.deploy-form {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 20px;
|
||||
}
|
||||
|
||||
.form-row {
|
||||
display: grid;
|
||||
grid-template-columns: 1fr 1fr;
|
||||
gap: 20px;
|
||||
}
|
||||
|
||||
.form-group {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 8px;
|
||||
}
|
||||
|
||||
.form-group label {
|
||||
font-weight: 600;
|
||||
color: var(--color-grey-dark);
|
||||
}
|
||||
|
||||
.form-group input,
|
||||
.form-group textarea {
|
||||
padding: 12px;
|
||||
border: 1px solid #ddd;
|
||||
border-radius: var(--radius-sm);
|
||||
font-size: 1rem;
|
||||
}
|
||||
|
||||
.form-group textarea {
|
||||
resize: vertical;
|
||||
min-height: 80px;
|
||||
}
|
||||
|
||||
.form-help {
|
||||
font-size: 0.85rem;
|
||||
color: var(--color-grey-dark);
|
||||
font-style: italic;
|
||||
}
|
||||
|
||||
.btn-primary {
|
||||
background: var(--color-primary);
|
||||
color: white;
|
||||
border: none;
|
||||
padding: 15px 30px;
|
||||
border-radius: var(--radius-sm);
|
||||
font-size: 1.1rem;
|
||||
font-weight: 600;
|
||||
cursor: pointer;
|
||||
transition: background-color 0.3s;
|
||||
}
|
||||
|
||||
.btn-primary:hover:not(:disabled) {
|
||||
background: var(--color-primary-dark);
|
||||
}
|
||||
|
||||
.btn-primary:disabled {
|
||||
opacity: 0.6;
|
||||
cursor: not-allowed;
|
||||
}
|
||||
|
||||
.deploy-status {
|
||||
margin-top: 20px;
|
||||
padding: 15px;
|
||||
background: #e8f5e8;
|
||||
border-radius: var(--radius-sm);
|
||||
border-left: 4px solid #28a745;
|
||||
}
|
||||
|
||||
.status-message {
|
||||
margin: 0;
|
||||
font-weight: 600;
|
||||
color: #28a745;
|
||||
}
|
||||
|
||||
@media (max-width: 768px) {
|
||||
.form-row {
|
||||
grid-template-columns: 1fr;
|
||||
}
|
||||
|
||||
.dle-info {
|
||||
flex-direction: column;
|
||||
align-items: flex-start;
|
||||
gap: 5px;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
@@ -0,0 +1,205 @@
|
||||
<!--
|
||||
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="module-deploy-form">
|
||||
<!-- Заголовок -->
|
||||
<div class="page-header">
|
||||
<div class="header-content">
|
||||
<h1>Деплой пользовательского модуля</h1>
|
||||
<p>Создание и деплой кастомного модуля для DLE</p>
|
||||
</div>
|
||||
<button class="close-btn" @click="router.push('/management/modules')">×</button>
|
||||
</div>
|
||||
|
||||
<!-- Информация о пользовательских модулях -->
|
||||
<div class="module-info">
|
||||
<div class="info-card">
|
||||
<h3>🔧 Пользовательские модули</h3>
|
||||
<div class="info-grid">
|
||||
<div class="info-item">
|
||||
<strong>Назначение:</strong> Расширение функциональности DLE
|
||||
</div>
|
||||
<div class="info-item">
|
||||
<strong>Возможности:</strong> Любая кастомная логика, интеграции, API
|
||||
</div>
|
||||
<div class="info-item">
|
||||
<strong>Безопасность:</strong> Проверка через голосование токен-холдеров
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Форма деплоя будет добавлена позже -->
|
||||
<div class="deploy-form-placeholder">
|
||||
<div class="placeholder-content">
|
||||
<h3>🚧 Форма деплоя в разработке</h3>
|
||||
<p>Здесь будет универсальная форма для деплоя пользовательских модулей</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
</div>
|
||||
</BaseLayout>
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import { defineProps, defineEmits, ref, onMounted } from 'vue';
|
||||
import { useRouter, useRoute } from 'vue-router';
|
||||
import BaseLayout from '../../../components/BaseLayout.vue';
|
||||
|
||||
// Определяем props
|
||||
const props = defineProps({
|
||||
isAuthenticated: { type: Boolean, default: false },
|
||||
identities: { type: Array, default: () => [] },
|
||||
tokenBalances: { type: Object, default: () => ({}) },
|
||||
isLoadingTokens: { type: Boolean, default: false }
|
||||
});
|
||||
|
||||
// Определяем emits
|
||||
const emit = defineEmits(['auth-action-completed']);
|
||||
|
||||
const router = useRouter();
|
||||
const route = useRoute();
|
||||
|
||||
// Состояние
|
||||
const isLoading = ref(false);
|
||||
|
||||
// Инициализация
|
||||
onMounted(() => {
|
||||
console.log('[ModuleDeployFormView] Страница загружена');
|
||||
});
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
.module-deploy-form {
|
||||
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: center;
|
||||
margin-bottom: 30px;
|
||||
padding-bottom: 20px;
|
||||
border-bottom: 2px solid #f0f0f0;
|
||||
}
|
||||
|
||||
.page-header h1 {
|
||||
color: var(--color-primary);
|
||||
font-size: 2rem;
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
.page-header p {
|
||||
margin: 10px 0 0 0;
|
||||
color: #666;
|
||||
}
|
||||
|
||||
.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;
|
||||
}
|
||||
|
||||
.close-btn:hover {
|
||||
background: #f0f0f0;
|
||||
color: #333;
|
||||
}
|
||||
|
||||
/* Информация о модуле */
|
||||
.module-info {
|
||||
margin-bottom: 30px;
|
||||
}
|
||||
|
||||
.info-card {
|
||||
background: #f8f9fa;
|
||||
border-radius: var(--radius-md);
|
||||
padding: 20px;
|
||||
border: 1px solid #e9ecef;
|
||||
}
|
||||
|
||||
.info-card h3 {
|
||||
margin: 0 0 15px 0;
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
.info-grid {
|
||||
display: grid;
|
||||
grid-template-columns: repeat(auto-fit, minmax(250px, 1fr));
|
||||
gap: 15px;
|
||||
}
|
||||
|
||||
.info-item {
|
||||
padding: 10px;
|
||||
background: white;
|
||||
border-radius: var(--radius-sm);
|
||||
border: 1px solid #dee2e6;
|
||||
}
|
||||
|
||||
.info-item strong {
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
/* Плейсхолдер для формы */
|
||||
.deploy-form-placeholder {
|
||||
background: #f8f9fa;
|
||||
border-radius: var(--radius-md);
|
||||
padding: 40px;
|
||||
text-align: center;
|
||||
border: 2px dashed #dee2e6;
|
||||
}
|
||||
|
||||
.placeholder-content h3 {
|
||||
color: var(--color-primary);
|
||||
margin-bottom: 10px;
|
||||
}
|
||||
|
||||
.placeholder-content p {
|
||||
color: #666;
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
/* Адаптивность */
|
||||
@media (max-width: 768px) {
|
||||
.info-grid {
|
||||
grid-template-columns: 1fr;
|
||||
}
|
||||
|
||||
.page-header {
|
||||
flex-direction: column;
|
||||
align-items: flex-start;
|
||||
gap: 15px;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
584
frontend/src/views/smartcontracts/modules/OracleModuleDeploy.vue
Normal file
584
frontend/src/views/smartcontracts/modules/OracleModuleDeploy.vue
Normal file
@@ -0,0 +1,584 @@
|
||||
<!--
|
||||
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="module-deploy-page">
|
||||
<!-- Заголовок -->
|
||||
<div class="page-header">
|
||||
<div class="header-content">
|
||||
<h1>🔗 Деплой OracleModule</h1>
|
||||
<p>Модуль для интеграции с внешними данными и автоматизации DLE</p>
|
||||
<div v-if="selectedDle" class="dle-info">
|
||||
<span class="dle-name">{{ selectedDle.name }} ({{ selectedDle.symbol }})</span>
|
||||
<span class="dle-address">{{ selectedDle.dleAddress }}</span>
|
||||
</div>
|
||||
</div>
|
||||
<button class="close-btn" @click="router.push(`/management/modules?address=${route.query.address}`)">×</button>
|
||||
</div>
|
||||
|
||||
<!-- Описание модуля -->
|
||||
<div class="module-description">
|
||||
<div class="description-card">
|
||||
<h3>📋 Описание OracleModule</h3>
|
||||
<div class="description-content">
|
||||
<p><strong>OracleModule</strong> - это модуль для интеграции DLE с внешними источниками данных и автоматизации процессов на основе реальных событий.</p>
|
||||
|
||||
<h4>🔧 Основная функциональность:</h4>
|
||||
<ul>
|
||||
<li><strong>Интеграция с IoT:</strong> Получение данных от датчиков, производственных линий, оборудования</li>
|
||||
<li><strong>API интеграция:</strong> Подключение к внешним системам, ERP, CRM, аналитическим платформам</li>
|
||||
<li><strong>Автоматические триггеры:</strong> Создание предложений на основе внешних событий</li>
|
||||
<li><strong>Валидация данных:</strong> Проверка достоверности полученной информации</li>
|
||||
<li><strong>Множественные оракулы:</strong> Подтверждение данных от нескольких источников</li>
|
||||
</ul>
|
||||
|
||||
<h4>🏭 Примеры применения:</h4>
|
||||
<ul>
|
||||
<li><strong>Производственные токены:</strong> Автоматический выпуск токенов за завершение партий продукции</li>
|
||||
<li><strong>Качественные бонусы:</strong> Токены за высокое качество продукции на основе данных датчиков</li>
|
||||
<li><strong>Экологические токены:</strong> Вознаграждения за снижение энергопотребления и выбросов</li>
|
||||
<li><strong>Инновационные токены:</strong> Токены за внедрение новых технологий и процессов</li>
|
||||
<li><strong>Рыночные корректировки:</strong> Автоматическая адаптация стратегии на основе рыночных данных</li>
|
||||
</ul>
|
||||
|
||||
<h4>🔐 Безопасность и контроль:</h4>
|
||||
<ul>
|
||||
<li>Все оракулы должны быть авторизованы через governance</li>
|
||||
<li>Данные валидируются перед обработкой</li>
|
||||
<li>Критические решения требуют множественного подтверждения</li>
|
||||
<li>История всех оракульных данных сохраняется в блокчейне</li>
|
||||
<li>Возможность отключения оракулов в экстренных случаях</li>
|
||||
</ul>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Архитектура модуля -->
|
||||
<div class="module-architecture">
|
||||
<div class="architecture-card">
|
||||
<h3>🏗️ Архитектура OracleModule</h3>
|
||||
<div class="architecture-content">
|
||||
<div class="architecture-diagram">
|
||||
<div class="diagram-row">
|
||||
<div class="diagram-item external">
|
||||
<h5>🌐 Внешние источники</h5>
|
||||
<ul>
|
||||
<li>IoT датчики</li>
|
||||
<li>Производственные линии</li>
|
||||
<li>ERP/CRM системы</li>
|
||||
<li>Аналитические платформы</li>
|
||||
<li>Рыночные данные</li>
|
||||
</ul>
|
||||
</div>
|
||||
<div class="diagram-arrow">→</div>
|
||||
<div class="diagram-item oracle">
|
||||
<h5>🔗 OracleModule</h5>
|
||||
<ul>
|
||||
<li>Получение данных</li>
|
||||
<li>Валидация</li>
|
||||
<li>Обработка</li>
|
||||
<li>Создание предложений</li>
|
||||
</ul>
|
||||
</div>
|
||||
<div class="diagram-arrow">→</div>
|
||||
<div class="diagram-item dle">
|
||||
<h5>🏛️ DLE Governance</h5>
|
||||
<ul>
|
||||
<li>Предложения</li>
|
||||
<li>Голосование</li>
|
||||
<li>Исполнение</li>
|
||||
<li>Выпуск токенов</li>
|
||||
</ul>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Типы оракулов -->
|
||||
<div class="oracle-types">
|
||||
<div class="types-card">
|
||||
<h3>📊 Типы оракулов</h3>
|
||||
<div class="types-grid">
|
||||
<div class="type-item">
|
||||
<h4>🏭 Производственные оракулы</h4>
|
||||
<p>Данные от производственных линий, оборудования, датчиков качества</p>
|
||||
<ul>
|
||||
<li>Завершение партий продукции</li>
|
||||
<li>Показатели качества</li>
|
||||
<li>Энергопотребление</li>
|
||||
<li>Время простоя</li>
|
||||
</ul>
|
||||
</div>
|
||||
|
||||
<div class="type-item">
|
||||
<h4>📈 Рыночные оракулы</h4>
|
||||
<p>Рыночные данные, цены, спрос, конкуренция</p>
|
||||
<ul>
|
||||
<li>Цены на сырье</li>
|
||||
<li>Спрос на продукцию</li>
|
||||
<li>Конкурентные данные</li>
|
||||
<li>Экономические индикаторы</li>
|
||||
</ul>
|
||||
</div>
|
||||
|
||||
<div class="type-item">
|
||||
<h4>🌱 Экологические оракулы</h4>
|
||||
<p>Данные об экологическом воздействии и устойчивости</p>
|
||||
<ul>
|
||||
<li>Выбросы CO2</li>
|
||||
<li>Потребление энергии</li>
|
||||
<li>Переработка отходов</li>
|
||||
<li>Использование возобновляемых ресурсов</li>
|
||||
</ul>
|
||||
</div>
|
||||
|
||||
<div class="type-item">
|
||||
<h4>💼 Бизнес-оракулы</h4>
|
||||
<p>Бизнес-метрики, продажи, удовлетворенность клиентов</p>
|
||||
<ul>
|
||||
<li>Объемы продаж</li>
|
||||
<li>Удовлетворенность клиентов</li>
|
||||
<li>Эффективность процессов</li>
|
||||
<li>Финансовые показатели</li>
|
||||
</ul>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Примеры использования -->
|
||||
<div class="usage-examples">
|
||||
<div class="examples-card">
|
||||
<h3>💡 Примеры использования</h3>
|
||||
<div class="examples-content">
|
||||
<div class="example-item">
|
||||
<h4>🏭 Автоматические производственные токены</h4>
|
||||
<div class="example-code">
|
||||
<pre><code>// При завершении партии продукции
|
||||
function onBatchCompleted(uint256 quantity, uint256 quality) {
|
||||
uint256 tokens = quantity * quality * 10;
|
||||
createMintProposal(tokens, "Production reward");
|
||||
}</code></pre>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="example-item">
|
||||
<h4>🌱 Экологические бонусы</h4>
|
||||
<div class="example-code">
|
||||
<pre><code>// При снижении энергопотребления
|
||||
function onEnergyReduction(uint256 savedEnergy) {
|
||||
uint256 tokens = savedEnergy * 5;
|
||||
createMintProposal(tokens, "Energy efficiency bonus");
|
||||
}</code></pre>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="example-item">
|
||||
<h4>📈 Рыночная адаптация</h4>
|
||||
<div class="example-code">
|
||||
<pre><code>// При изменении рыночных условий
|
||||
function onMarketChange(uint256 priceChange) {
|
||||
if (priceChange > 10) {
|
||||
createStrategyProposal("Increase production");
|
||||
}
|
||||
}</code></pre>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Статус разработки -->
|
||||
<div class="development-status">
|
||||
<div class="status-card">
|
||||
<h3>🚧 Статус разработки</h3>
|
||||
<div class="status-content">
|
||||
<p><strong>OracleModule находится в стадии планирования.</strong></p>
|
||||
<p>Модуль будет включать:</p>
|
||||
<ul>
|
||||
<li>✅ Систему авторизации оракулов</li>
|
||||
<li>✅ Валидацию и обработку данных</li>
|
||||
<li>✅ Интеграцию с IoT и API</li>
|
||||
<li>✅ Автоматические триггеры</li>
|
||||
<li>✅ Множественное подтверждение данных</li>
|
||||
<li>✅ Аудит и мониторинг</li>
|
||||
</ul>
|
||||
<p><em>Модуль будет доступен в следующих обновлениях DLE.</em></p>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</BaseLayout>
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import { ref, computed, onMounted } from 'vue';
|
||||
import { useRoute, useRouter } from 'vue-router';
|
||||
import BaseLayout from '../../../components/BaseLayout.vue';
|
||||
import api from '../../../api/axios';
|
||||
|
||||
// Props
|
||||
const props = defineProps({
|
||||
isAuthenticated: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
},
|
||||
identities: {
|
||||
type: Array,
|
||||
default: () => []
|
||||
},
|
||||
tokenBalances: {
|
||||
type: Object,
|
||||
default: () => ({})
|
||||
},
|
||||
isLoadingTokens: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
}
|
||||
});
|
||||
|
||||
// Emits
|
||||
const emit = defineEmits(['auth-action-completed']);
|
||||
|
||||
// Router
|
||||
const route = useRoute();
|
||||
const router = useRouter();
|
||||
|
||||
// Состояние
|
||||
const selectedDle = ref(null);
|
||||
const isLoadingDle = ref(false);
|
||||
|
||||
// Получаем адрес DLE из URL
|
||||
const dleAddress = computed(() => route.query.address);
|
||||
|
||||
// Загрузка данных DLE
|
||||
const loadDleData = async () => {
|
||||
if (!dleAddress.value) return;
|
||||
|
||||
try {
|
||||
isLoadingDle.value = true;
|
||||
const response = await api.post('/blockchain/read-dle-info', {
|
||||
dleAddress: dleAddress.value
|
||||
});
|
||||
|
||||
if (response.data.success) {
|
||||
selectedDle.value = response.data.data;
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('Ошибка загрузки данных DLE:', error);
|
||||
} finally {
|
||||
isLoadingDle.value = false;
|
||||
}
|
||||
};
|
||||
|
||||
// Загружаем данные при монтировании
|
||||
onMounted(() => {
|
||||
loadDleData();
|
||||
});
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
.module-deploy-page {
|
||||
padding: 20px;
|
||||
max-width: 1200px;
|
||||
margin: 0 auto;
|
||||
}
|
||||
|
||||
.page-header {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: flex-start;
|
||||
margin-bottom: 30px;
|
||||
padding-bottom: 20px;
|
||||
border-bottom: 2px solid #f0f0f0;
|
||||
}
|
||||
|
||||
.header-content h1 {
|
||||
color: var(--color-primary);
|
||||
font-size: 2.5rem;
|
||||
margin: 0 0 10px 0;
|
||||
}
|
||||
|
||||
.header-content p {
|
||||
color: var(--color-grey-dark);
|
||||
font-size: 1.1rem;
|
||||
margin: 0 0 15px 0;
|
||||
}
|
||||
|
||||
.dle-info {
|
||||
display: flex;
|
||||
gap: 15px;
|
||||
align-items: center;
|
||||
}
|
||||
|
||||
.dle-name {
|
||||
font-weight: 600;
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
.dle-address {
|
||||
font-family: monospace;
|
||||
color: var(--color-grey-dark);
|
||||
font-size: 0.9rem;
|
||||
}
|
||||
|
||||
.close-btn {
|
||||
background: none;
|
||||
border: none;
|
||||
font-size: 2rem;
|
||||
color: var(--color-grey-dark);
|
||||
cursor: pointer;
|
||||
padding: 5px;
|
||||
}
|
||||
|
||||
.close-btn:hover {
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
.module-description,
|
||||
.module-architecture,
|
||||
.oracle-types,
|
||||
.usage-examples,
|
||||
.development-status {
|
||||
margin-bottom: 30px;
|
||||
}
|
||||
|
||||
.description-card,
|
||||
.architecture-card,
|
||||
.types-card,
|
||||
.examples-card,
|
||||
.status-card {
|
||||
background: #f8f9fa;
|
||||
padding: 25px;
|
||||
border-radius: var(--radius-lg);
|
||||
border: 1px solid #e9ecef;
|
||||
}
|
||||
|
||||
.description-card h3,
|
||||
.architecture-card h3,
|
||||
.types-card h3,
|
||||
.examples-card h3,
|
||||
.status-card h3 {
|
||||
color: var(--color-primary);
|
||||
margin: 0 0 20px 0;
|
||||
}
|
||||
|
||||
.description-content h4 {
|
||||
color: var(--color-grey-dark);
|
||||
margin: 20px 0 10px 0;
|
||||
}
|
||||
|
||||
.description-content ul {
|
||||
margin: 10px 0;
|
||||
padding-left: 20px;
|
||||
}
|
||||
|
||||
.description-content li {
|
||||
margin: 5px 0;
|
||||
line-height: 1.5;
|
||||
}
|
||||
|
||||
/* Архитектурная диаграмма */
|
||||
.architecture-diagram {
|
||||
margin: 20px 0;
|
||||
}
|
||||
|
||||
.diagram-row {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: space-between;
|
||||
gap: 20px;
|
||||
margin-bottom: 20px;
|
||||
}
|
||||
|
||||
.diagram-item {
|
||||
flex: 1;
|
||||
padding: 20px;
|
||||
border-radius: var(--radius-sm);
|
||||
text-align: center;
|
||||
min-height: 150px;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
justify-content: center;
|
||||
}
|
||||
|
||||
.diagram-item.external {
|
||||
background: #e3f2fd;
|
||||
border: 2px solid #2196f3;
|
||||
}
|
||||
|
||||
.diagram-item.oracle {
|
||||
background: #f3e5f5;
|
||||
border: 2px solid #9c27b0;
|
||||
}
|
||||
|
||||
.diagram-item.dle {
|
||||
background: #e8f5e8;
|
||||
border: 2px solid #4caf50;
|
||||
}
|
||||
|
||||
.diagram-item h5 {
|
||||
margin: 0 0 15px 0;
|
||||
font-weight: 600;
|
||||
}
|
||||
|
||||
.diagram-item ul {
|
||||
margin: 0;
|
||||
padding: 0;
|
||||
list-style: none;
|
||||
font-size: 0.9rem;
|
||||
}
|
||||
|
||||
.diagram-item li {
|
||||
margin: 5px 0;
|
||||
}
|
||||
|
||||
.diagram-arrow {
|
||||
font-size: 2rem;
|
||||
color: var(--color-primary);
|
||||
font-weight: bold;
|
||||
}
|
||||
|
||||
/* Типы оракулов */
|
||||
.types-grid {
|
||||
display: grid;
|
||||
grid-template-columns: repeat(auto-fit, minmax(280px, 1fr));
|
||||
gap: 20px;
|
||||
margin-top: 20px;
|
||||
}
|
||||
|
||||
.type-item {
|
||||
background: white;
|
||||
padding: 20px;
|
||||
border-radius: var(--radius-sm);
|
||||
border: 1px solid #e9ecef;
|
||||
}
|
||||
|
||||
.type-item h4 {
|
||||
color: var(--color-primary);
|
||||
margin: 0 0 10px 0;
|
||||
}
|
||||
|
||||
.type-item p {
|
||||
color: var(--color-grey-dark);
|
||||
margin: 0 0 15px 0;
|
||||
font-size: 0.9rem;
|
||||
}
|
||||
|
||||
.type-item ul {
|
||||
margin: 0;
|
||||
padding-left: 20px;
|
||||
font-size: 0.9rem;
|
||||
}
|
||||
|
||||
.type-item li {
|
||||
margin: 5px 0;
|
||||
color: var(--color-grey-dark);
|
||||
}
|
||||
|
||||
/* Примеры использования */
|
||||
.examples-content {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 20px;
|
||||
}
|
||||
|
||||
.example-item {
|
||||
background: white;
|
||||
padding: 20px;
|
||||
border-radius: var(--radius-sm);
|
||||
border: 1px solid #e9ecef;
|
||||
}
|
||||
|
||||
.example-item h4 {
|
||||
color: var(--color-primary);
|
||||
margin: 0 0 15px 0;
|
||||
}
|
||||
|
||||
.example-code {
|
||||
background: #f8f9fa;
|
||||
border: 1px solid #e9ecef;
|
||||
border-radius: var(--radius-sm);
|
||||
padding: 15px;
|
||||
overflow-x: auto;
|
||||
}
|
||||
|
||||
.example-code pre {
|
||||
margin: 0;
|
||||
font-family: 'Courier New', monospace;
|
||||
font-size: 0.9rem;
|
||||
color: #333;
|
||||
}
|
||||
|
||||
.example-code code {
|
||||
background: none;
|
||||
padding: 0;
|
||||
}
|
||||
|
||||
/* Статус разработки */
|
||||
.status-content {
|
||||
background: white;
|
||||
padding: 20px;
|
||||
border-radius: var(--radius-sm);
|
||||
border: 1px solid #e9ecef;
|
||||
}
|
||||
|
||||
.status-content p {
|
||||
margin: 0 0 15px 0;
|
||||
line-height: 1.6;
|
||||
}
|
||||
|
||||
.status-content ul {
|
||||
margin: 15px 0;
|
||||
padding-left: 20px;
|
||||
}
|
||||
|
||||
.status-content li {
|
||||
margin: 5px 0;
|
||||
color: var(--color-grey-dark);
|
||||
}
|
||||
|
||||
.status-content em {
|
||||
color: var(--color-primary);
|
||||
font-style: italic;
|
||||
}
|
||||
|
||||
@media (max-width: 768px) {
|
||||
.diagram-row {
|
||||
flex-direction: column;
|
||||
gap: 15px;
|
||||
}
|
||||
|
||||
.diagram-arrow {
|
||||
transform: rotate(90deg);
|
||||
}
|
||||
|
||||
.types-grid {
|
||||
grid-template-columns: 1fr;
|
||||
}
|
||||
|
||||
.dle-info {
|
||||
flex-direction: column;
|
||||
align-items: flex-start;
|
||||
gap: 5px;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
@@ -0,0 +1,218 @@
|
||||
<!--
|
||||
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="timelock-module-deploy">
|
||||
<!-- Заголовок -->
|
||||
<div class="page-header">
|
||||
<div class="header-content">
|
||||
<h1>Деплой TimelockModule</h1>
|
||||
<p>Задержки исполнения - безопасность критических операций через таймлоки</p>
|
||||
<p v-if="dleAddress" class="dle-address">
|
||||
<strong>DLE:</strong> {{ dleAddress }}
|
||||
</p>
|
||||
</div>
|
||||
<button class="close-btn" @click="router.push('/management/modules')">×</button>
|
||||
</div>
|
||||
|
||||
<!-- Информация о модуле -->
|
||||
<div class="module-info">
|
||||
<div class="info-card">
|
||||
<h3>⏰ TimelockModule</h3>
|
||||
<div class="info-grid">
|
||||
<div class="info-item">
|
||||
<strong>Назначение:</strong> Безопасность критических операций
|
||||
</div>
|
||||
<div class="info-item">
|
||||
<strong>Функции:</strong> Настраиваемые таймлоки, отмена предложений, аудит
|
||||
</div>
|
||||
<div class="info-item">
|
||||
<strong>Безопасность:</strong> Задержки исполнения для защиты от атак
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Форма деплоя будет добавлена позже -->
|
||||
<div class="deploy-form-placeholder">
|
||||
<div class="placeholder-content">
|
||||
<h3>🚧 Форма деплоя в разработке</h3>
|
||||
<p>Здесь будет форма для деплоя TimelockModule</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
</div>
|
||||
</BaseLayout>
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import { defineProps, defineEmits, ref, onMounted } from 'vue';
|
||||
import { useRouter, useRoute } from 'vue-router';
|
||||
import BaseLayout from '../../../components/BaseLayout.vue';
|
||||
|
||||
// Определяем props
|
||||
const props = defineProps({
|
||||
isAuthenticated: { type: Boolean, default: false },
|
||||
identities: { type: Array, default: () => [] },
|
||||
tokenBalances: { type: Object, default: () => ({}) },
|
||||
isLoadingTokens: { type: Boolean, default: false }
|
||||
});
|
||||
|
||||
// Определяем emits
|
||||
const emit = defineEmits(['auth-action-completed']);
|
||||
|
||||
const router = useRouter();
|
||||
const route = useRoute();
|
||||
|
||||
// Состояние
|
||||
const isLoading = ref(false);
|
||||
const dleAddress = ref(route.query.address || null);
|
||||
|
||||
// Инициализация
|
||||
onMounted(() => {
|
||||
console.log('[TimelockModuleDeployView] Страница загружена');
|
||||
});
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
.timelock-module-deploy {
|
||||
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: center;
|
||||
margin-bottom: 30px;
|
||||
padding-bottom: 20px;
|
||||
border-bottom: 2px solid #f0f0f0;
|
||||
}
|
||||
|
||||
.page-header h1 {
|
||||
color: var(--color-primary);
|
||||
font-size: 2rem;
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
.page-header p {
|
||||
margin: 10px 0 0 0;
|
||||
color: #666;
|
||||
}
|
||||
|
||||
.dle-address {
|
||||
margin-top: 10px !important;
|
||||
font-family: monospace;
|
||||
background: #f8f9fa;
|
||||
padding: 8px 12px;
|
||||
border-radius: var(--radius-sm);
|
||||
border: 1px solid #dee2e6;
|
||||
}
|
||||
|
||||
.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;
|
||||
}
|
||||
|
||||
.close-btn:hover {
|
||||
background: #f0f0f0;
|
||||
color: #333;
|
||||
}
|
||||
|
||||
/* Информация о модуле */
|
||||
.module-info {
|
||||
margin-bottom: 30px;
|
||||
}
|
||||
|
||||
.info-card {
|
||||
background: #f8f9fa;
|
||||
border-radius: var(--radius-md);
|
||||
padding: 20px;
|
||||
border: 1px solid #e9ecef;
|
||||
}
|
||||
|
||||
.info-card h3 {
|
||||
margin: 0 0 15px 0;
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
.info-grid {
|
||||
display: grid;
|
||||
grid-template-columns: repeat(auto-fit, minmax(250px, 1fr));
|
||||
gap: 15px;
|
||||
}
|
||||
|
||||
.info-item {
|
||||
padding: 10px;
|
||||
background: white;
|
||||
border-radius: var(--radius-sm);
|
||||
border: 1px solid #dee2e6;
|
||||
}
|
||||
|
||||
.info-item strong {
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
/* Плейсхолдер для формы */
|
||||
.deploy-form-placeholder {
|
||||
background: #f8f9fa;
|
||||
border-radius: var(--radius-md);
|
||||
padding: 40px;
|
||||
text-align: center;
|
||||
border: 2px dashed #dee2e6;
|
||||
}
|
||||
|
||||
.placeholder-content h3 {
|
||||
color: var(--color-primary);
|
||||
margin-bottom: 10px;
|
||||
}
|
||||
|
||||
.placeholder-content p {
|
||||
color: #666;
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
/* Адаптивность */
|
||||
@media (max-width: 768px) {
|
||||
.info-grid {
|
||||
grid-template-columns: 1fr;
|
||||
}
|
||||
|
||||
.page-header {
|
||||
flex-direction: column;
|
||||
align-items: flex-start;
|
||||
gap: 15px;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
@@ -0,0 +1,218 @@
|
||||
<!--
|
||||
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="treasury-module-deploy">
|
||||
<!-- Заголовок -->
|
||||
<div class="page-header">
|
||||
<div class="header-content">
|
||||
<h1>Деплой TreasuryModule</h1>
|
||||
<p>Казначейство DLE - управление финансами, депозиты, выводы, дивиденды</p>
|
||||
<p v-if="dleAddress" class="dle-address">
|
||||
<strong>DLE:</strong> {{ dleAddress }}
|
||||
</p>
|
||||
</div>
|
||||
<button class="close-btn" @click="router.push('/management/modules')">×</button>
|
||||
</div>
|
||||
|
||||
<!-- Информация о модуле -->
|
||||
<div class="module-info">
|
||||
<div class="info-card">
|
||||
<h3>💰 TreasuryModule</h3>
|
||||
<div class="info-grid">
|
||||
<div class="info-item">
|
||||
<strong>Назначение:</strong> Управление финансами DLE
|
||||
</div>
|
||||
<div class="info-item">
|
||||
<strong>Функции:</strong> Депозиты, выводы, дивиденды, бюджетирование
|
||||
</div>
|
||||
<div class="info-item">
|
||||
<strong>Безопасность:</strong> Все операции через голосование
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Форма деплоя будет добавлена позже -->
|
||||
<div class="deploy-form-placeholder">
|
||||
<div class="placeholder-content">
|
||||
<h3>🚧 Форма деплоя в разработке</h3>
|
||||
<p>Здесь будет форма для деплоя TreasuryModule</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
</div>
|
||||
</BaseLayout>
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import { defineProps, defineEmits, ref, onMounted } from 'vue';
|
||||
import { useRouter, useRoute } from 'vue-router';
|
||||
import BaseLayout from '../../../components/BaseLayout.vue';
|
||||
|
||||
// Определяем props
|
||||
const props = defineProps({
|
||||
isAuthenticated: { type: Boolean, default: false },
|
||||
identities: { type: Array, default: () => [] },
|
||||
tokenBalances: { type: Object, default: () => ({}) },
|
||||
isLoadingTokens: { type: Boolean, default: false }
|
||||
});
|
||||
|
||||
// Определяем emits
|
||||
const emit = defineEmits(['auth-action-completed']);
|
||||
|
||||
const router = useRouter();
|
||||
const route = useRoute();
|
||||
|
||||
// Состояние
|
||||
const isLoading = ref(false);
|
||||
const dleAddress = ref(route.query.address || null);
|
||||
|
||||
// Инициализация
|
||||
onMounted(() => {
|
||||
console.log('[TreasuryModuleDeployView] Страница загружена');
|
||||
});
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
.treasury-module-deploy {
|
||||
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: center;
|
||||
margin-bottom: 30px;
|
||||
padding-bottom: 20px;
|
||||
border-bottom: 2px solid #f0f0f0;
|
||||
}
|
||||
|
||||
.page-header h1 {
|
||||
color: var(--color-primary);
|
||||
font-size: 2rem;
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
.page-header p {
|
||||
margin: 10px 0 0 0;
|
||||
color: #666;
|
||||
}
|
||||
|
||||
.dle-address {
|
||||
margin-top: 10px !important;
|
||||
font-family: monospace;
|
||||
background: #f8f9fa;
|
||||
padding: 8px 12px;
|
||||
border-radius: var(--radius-sm);
|
||||
border: 1px solid #dee2e6;
|
||||
}
|
||||
|
||||
.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;
|
||||
}
|
||||
|
||||
.close-btn:hover {
|
||||
background: #f0f0f0;
|
||||
color: #333;
|
||||
}
|
||||
|
||||
/* Информация о модуле */
|
||||
.module-info {
|
||||
margin-bottom: 30px;
|
||||
}
|
||||
|
||||
.info-card {
|
||||
background: #f8f9fa;
|
||||
border-radius: var(--radius-md);
|
||||
padding: 20px;
|
||||
border: 1px solid #e9ecef;
|
||||
}
|
||||
|
||||
.info-card h3 {
|
||||
margin: 0 0 15px 0;
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
.info-grid {
|
||||
display: grid;
|
||||
grid-template-columns: repeat(auto-fit, minmax(250px, 1fr));
|
||||
gap: 15px;
|
||||
}
|
||||
|
||||
.info-item {
|
||||
padding: 10px;
|
||||
background: white;
|
||||
border-radius: var(--radius-sm);
|
||||
border: 1px solid #dee2e6;
|
||||
}
|
||||
|
||||
.info-item strong {
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
/* Плейсхолдер для формы */
|
||||
.deploy-form-placeholder {
|
||||
background: #f8f9fa;
|
||||
border-radius: var(--radius-md);
|
||||
padding: 40px;
|
||||
text-align: center;
|
||||
border: 2px dashed #dee2e6;
|
||||
}
|
||||
|
||||
.placeholder-content h3 {
|
||||
color: var(--color-primary);
|
||||
margin-bottom: 10px;
|
||||
}
|
||||
|
||||
.placeholder-content p {
|
||||
color: #666;
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
/* Адаптивность */
|
||||
@media (max-width: 768px) {
|
||||
.info-grid {
|
||||
grid-template-columns: 1fr;
|
||||
}
|
||||
|
||||
.page-header {
|
||||
flex-direction: column;
|
||||
align-items: flex-start;
|
||||
gap: 15px;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
Reference in New Issue
Block a user