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

This commit is contained in:
2025-08-16 22:12:00 +03:00
parent fe0dc6509b
commit 1d636c113f
28 changed files with 6809 additions and 525 deletions

View File

@@ -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

View File

@@ -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

View File

@@ -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",

View File

@@ -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.");
}
}

View File

@@ -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
View 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;

View File

@@ -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;

View File

@@ -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;
}
}

View 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();

View File

@@ -18,6 +18,7 @@
- Безопасность: удалены уязвимые Merkleмеханизмы crosschain; нет внешних мостов/оракулов.
- Голосующая сила: OpenZeppelin `ERC20Votes` (снимки `getPastVotes`, `getPastTotalSupply`).
- Делегирование: жестко ограничено «только на себя»; третьим лицам делегировать нельзя (1 токен = 1 голос).
- Переводы токенов: ЗАБЛОКИРОВАНЫ прямые переводы (transfer, transferFrom, approve); переводы возможны ТОЛЬКО через governance предложения.
- SingleChain Governance: голосование происходит в одной выбранной сети (`governanceChainId`), время снапшота фиксируется на создании предложения и используется во всех сетях.
- MultiChain исполнение: выполнение в целевых сетях по EIP712 подписям холдеров, проверяется суммарная голосующая сила на зафиксированном `timepoint` (без доверия к мостам).
- «100% или ничего»: операции считаются успешными только при готовности/успешности всех целевых сетей.
@@ -94,6 +95,8 @@ DLE.sol (Один контракт)
- Распределение токенов между участниками
- **Голосующая сила = количество токенов**
- Проверка баланса токенов при каждой операции
- **Прямые переводы ЗАБЛОКИРОВАНЫ** - токены служат только для голосования
- **Переводы возможны ТОЛЬКО через governance предложения**
#### 2. Настраиваемый кворум
- **Описание**: Процент от общего количества токенов для принятия решений

View File

@@ -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',

View File

@@ -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;
}
}

View File

@@ -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>

View File

@@ -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>

View File

@@ -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>

View File

@@ -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>

View File

@@ -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>

View File

@@ -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;
}
/* Кнопки */

View File

@@ -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>

View 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>

View File

@@ -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>

View File

@@ -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>

View 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>

View File

@@ -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>

View 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>

View File

@@ -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>

View File

@@ -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>