ваше сообщение коммита
This commit is contained in:
@@ -48,6 +48,332 @@
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Блоки для деплоя стандартных модулей -->
|
||||
<div class="standard-modules">
|
||||
<div class="modules-header">
|
||||
<h3>🚀 Деплой стандартных модулей</h3>
|
||||
<p>Быстрый деплой предустановленных модулей DLE</p>
|
||||
</div>
|
||||
|
||||
<div class="modules-grid">
|
||||
<!-- TreasuryModule -->
|
||||
<div class="module-deploy-card">
|
||||
<div class="module-content">
|
||||
<h4>TreasuryModule</h4>
|
||||
<p>Казначейство DLE - управление финансами, депозиты, выводы, дивиденды</p>
|
||||
<div class="module-features">
|
||||
<span class="feature-tag">Финансы</span>
|
||||
<span class="feature-tag">Бюджет</span>
|
||||
<span class="feature-tag">Дивиденды</span>
|
||||
</div>
|
||||
</div>
|
||||
<div class="module-actions">
|
||||
<button
|
||||
class="btn btn-primary btn-deploy"
|
||||
@click="router.push(`/management/modules/deploy/treasury?address=${route.query.address}`)"
|
||||
>
|
||||
<i class="fas fa-rocket"></i>
|
||||
Деплой
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- TimelockModule -->
|
||||
<div class="module-deploy-card">
|
||||
<div class="module-content">
|
||||
<h4>TimelockModule</h4>
|
||||
<p>Задержки исполнения - безопасность критических операций через таймлоки</p>
|
||||
<div class="module-features">
|
||||
<span class="feature-tag">Безопасность</span>
|
||||
<span class="feature-tag">Таймлок</span>
|
||||
<span class="feature-tag">Аудит</span>
|
||||
</div>
|
||||
</div>
|
||||
<div class="module-actions">
|
||||
<button
|
||||
class="btn btn-primary btn-deploy"
|
||||
@click="router.push(`/management/modules/deploy/timelock?address=${route.query.address}`)"
|
||||
>
|
||||
<i class="fas fa-rocket"></i>
|
||||
Деплой
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- CommunicationModule -->
|
||||
<div class="module-deploy-card">
|
||||
<div class="module-content">
|
||||
<h4>CommunicationModule</h4>
|
||||
<p>Коммуникации - сообщения, звонки, история общения между участниками</p>
|
||||
<div class="module-features">
|
||||
<span class="feature-tag">Сообщения</span>
|
||||
<span class="feature-tag">Звонки</span>
|
||||
<span class="feature-tag">История</span>
|
||||
</div>
|
||||
</div>
|
||||
<div class="module-actions">
|
||||
<button
|
||||
class="btn btn-primary btn-deploy"
|
||||
@click="router.push(`/management/modules/deploy/communication?address=${route.query.address}`)"
|
||||
>
|
||||
<i class="fas fa-rocket"></i>
|
||||
Деплой
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- ApplicationModule -->
|
||||
<div class="module-deploy-card">
|
||||
<div class="module-content">
|
||||
<h4>ApplicationModule</h4>
|
||||
<p>Управление вызовом функций приложения через предложения и голосование</p>
|
||||
<div class="module-features">
|
||||
<span class="feature-tag">API</span>
|
||||
<span class="feature-tag">Голосование</span>
|
||||
<span class="feature-tag">Управление</span>
|
||||
</div>
|
||||
</div>
|
||||
<div class="module-actions">
|
||||
<button
|
||||
class="btn btn-primary btn-deploy"
|
||||
@click="router.push(`/management/modules/deploy/application?address=${route.query.address}`)"
|
||||
>
|
||||
<i class="fas fa-rocket"></i>
|
||||
Деплой
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- MintModule -->
|
||||
<div class="module-deploy-card">
|
||||
<div class="module-content">
|
||||
<h4>MintModule</h4>
|
||||
<p>Выпуск новых токенов DLE - создание дополнительных токенов через governance</p>
|
||||
<div class="module-features">
|
||||
<span class="feature-tag">Минтинг</span>
|
||||
<span class="feature-tag">Токены</span>
|
||||
<span class="feature-tag">Governance</span>
|
||||
</div>
|
||||
</div>
|
||||
<div class="module-actions">
|
||||
<button
|
||||
class="btn btn-primary btn-deploy"
|
||||
@click="router.push(`/management/modules/deploy/mint?address=${route.query.address}`)"
|
||||
>
|
||||
<i class="fas fa-rocket"></i>
|
||||
Деплой
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- BurnModule -->
|
||||
<div class="module-deploy-card">
|
||||
<div class="module-content">
|
||||
<h4>BurnModule</h4>
|
||||
<p>Сжигание токенов DLE - уменьшение общего предложения через governance</p>
|
||||
<div class="module-features">
|
||||
<span class="feature-tag">Сжигание</span>
|
||||
<span class="feature-tag">Токены</span>
|
||||
<span class="feature-tag">Governance</span>
|
||||
</div>
|
||||
</div>
|
||||
<div class="module-actions">
|
||||
<button
|
||||
class="btn btn-primary btn-deploy"
|
||||
@click="router.push(`/management/modules/deploy/burn?address=${route.query.address}`)"
|
||||
>
|
||||
<i class="fas fa-rocket"></i>
|
||||
Деплой
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- OracleModule -->
|
||||
<div class="module-deploy-card">
|
||||
<div class="module-content">
|
||||
<h4>OracleModule</h4>
|
||||
<p>Интеграция с внешними данными - автоматизация на основе IoT, API, датчиков</p>
|
||||
<div class="module-features">
|
||||
<span class="feature-tag">Оракулы</span>
|
||||
<span class="feature-tag">Автоматизация</span>
|
||||
<span class="feature-tag">IoT</span>
|
||||
<span class="feature-tag">API</span>
|
||||
</div>
|
||||
</div>
|
||||
<div class="module-actions">
|
||||
<button
|
||||
class="btn btn-primary btn-deploy"
|
||||
@click="router.push(`/management/modules/deploy/oracle?address=${route.query.address}`)"
|
||||
>
|
||||
<i class="fas fa-rocket"></i>
|
||||
Деплой
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- InheritanceModule -->
|
||||
<div class="module-deploy-card">
|
||||
<div class="module-content">
|
||||
<h4>InheritanceModule</h4>
|
||||
<p>Наследование токенов - автоматическая передача токенов наследникам</p>
|
||||
<div class="module-features">
|
||||
<span class="feature-tag">Наследование</span>
|
||||
<span class="feature-tag">Безопасность</span>
|
||||
<span class="feature-tag">Юридические</span>
|
||||
<span class="feature-tag">Автоматизация</span>
|
||||
</div>
|
||||
</div>
|
||||
<div class="module-actions">
|
||||
<button
|
||||
class="btn btn-primary btn-deploy"
|
||||
@click="router.push(`/management/modules/deploy/inheritance?address=${route.query.address}`)"
|
||||
>
|
||||
<i class="fas fa-rocket"></i>
|
||||
Деплой
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- VestingModule -->
|
||||
<div class="module-deploy-card">
|
||||
<div class="module-content">
|
||||
<h4>VestingModule</h4>
|
||||
<p>Вестинг токенов - постепенное разблокирование токенов по расписанию</p>
|
||||
<div class="module-features">
|
||||
<span class="feature-tag">Вестинг</span>
|
||||
<span class="feature-tag">Мотивация</span>
|
||||
<span class="feature-tag">Удержание</span>
|
||||
<span class="feature-tag">Расписание</span>
|
||||
</div>
|
||||
</div>
|
||||
<div class="module-actions">
|
||||
<button
|
||||
class="btn btn-primary btn-deploy"
|
||||
@click="router.push(`/management/modules/deploy/vesting?address=${route.query.address}`)"
|
||||
>
|
||||
<i class="fas fa-rocket"></i>
|
||||
Деплой
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- StakingModule -->
|
||||
<div class="module-deploy-card">
|
||||
<div class="module-content">
|
||||
<h4>StakingModule</h4>
|
||||
<p>Стейкинг токенов - заработок на удержании токенов</p>
|
||||
<div class="module-features">
|
||||
<span class="feature-tag">Стейкинг</span>
|
||||
<span class="feature-tag">Доход</span>
|
||||
<span class="feature-tag">Ликвидность</span>
|
||||
<span class="feature-tag">APY</span>
|
||||
</div>
|
||||
</div>
|
||||
<div class="module-actions">
|
||||
<button
|
||||
class="btn btn-primary btn-deploy"
|
||||
@click="router.push(`/management/modules/deploy/staking?address=${route.query.address}`)"
|
||||
>
|
||||
<i class="fas fa-rocket"></i>
|
||||
Деплой
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- InsuranceModule -->
|
||||
<div class="module-deploy-card">
|
||||
<div class="module-content">
|
||||
<h4>InsuranceModule</h4>
|
||||
<p>Страхование токенов - защита от рисков и потерь</p>
|
||||
<div class="module-features">
|
||||
<span class="feature-tag">Страхование</span>
|
||||
<span class="feature-tag">Защита</span>
|
||||
<span class="feature-tag">Риски</span>
|
||||
<span class="feature-tag">Безопасность</span>
|
||||
</div>
|
||||
</div>
|
||||
<div class="module-actions">
|
||||
<button
|
||||
class="btn btn-primary btn-deploy"
|
||||
@click="router.push(`/management/modules/deploy/insurance?address=${route.query.address}`)"
|
||||
>
|
||||
<i class="fas fa-rocket"></i>
|
||||
Деплой
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- ComplianceModule -->
|
||||
<div class="module-deploy-card">
|
||||
<div class="module-content">
|
||||
<h4>ComplianceModule</h4>
|
||||
<p>Соответствие требованиям - KYC/AML, налоги, аудит</p>
|
||||
<div class="module-features">
|
||||
<span class="feature-tag">KYC/AML</span>
|
||||
<span class="feature-tag">Налоги</span>
|
||||
<span class="feature-tag">Аудит</span>
|
||||
<span class="feature-tag">Регуляторы</span>
|
||||
</div>
|
||||
</div>
|
||||
<div class="module-actions">
|
||||
<button
|
||||
class="btn btn-primary btn-deploy"
|
||||
@click="router.push(`/management/modules/deploy/compliance?address=${route.query.address}`)"
|
||||
>
|
||||
<i class="fas fa-rocket"></i>
|
||||
Деплой
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- SupplyChainModule -->
|
||||
<div class="module-deploy-card">
|
||||
<div class="module-content">
|
||||
<h4>SupplyChainModule</h4>
|
||||
<p>Цепочка поставок - отслеживание и токенизация логистики</p>
|
||||
<div class="module-features">
|
||||
<span class="feature-tag">Логистика</span>
|
||||
<span class="feature-tag">Отслеживание</span>
|
||||
<span class="feature-tag">Качество</span>
|
||||
<span class="feature-tag">Прозрачность</span>
|
||||
</div>
|
||||
</div>
|
||||
<div class="module-actions">
|
||||
<button
|
||||
class="btn btn-primary btn-deploy"
|
||||
@click="router.push(`/management/modules/deploy/supplychain?address=${route.query.address}`)"
|
||||
>
|
||||
<i class="fas fa-rocket"></i>
|
||||
Деплой
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- EventModule -->
|
||||
<div class="module-deploy-card">
|
||||
<div class="module-content">
|
||||
<h4>EventModule</h4>
|
||||
<p>Событийный модуль - токенизация мероприятий и событий</p>
|
||||
<div class="module-features">
|
||||
<span class="feature-tag">События</span>
|
||||
<span class="feature-tag">NFT-билеты</span>
|
||||
<span class="feature-tag">Мероприятия</span>
|
||||
<span class="feature-tag">VR/AR</span>
|
||||
</div>
|
||||
</div>
|
||||
<div class="module-actions">
|
||||
<button
|
||||
class="btn btn-primary btn-deploy"
|
||||
@click="router.push(`/management/modules/deploy/event?address=${route.query.address}`)"
|
||||
>
|
||||
<i class="fas fa-rocket"></i>
|
||||
Деплой
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Форма добавления модуля -->
|
||||
<div class="add-module-form">
|
||||
<div class="form-header">
|
||||
@@ -521,6 +847,116 @@ onMounted(() => {
|
||||
border: 1px solid #dee2e6;
|
||||
}
|
||||
|
||||
/* Блоки для деплоя стандартных модулей */
|
||||
.standard-modules {
|
||||
background: #f8f9fa;
|
||||
border-radius: var(--radius-md);
|
||||
padding: 20px;
|
||||
margin-bottom: 30px;
|
||||
border: 1px solid #e9ecef;
|
||||
}
|
||||
|
||||
.modules-header {
|
||||
margin-bottom: 20px;
|
||||
padding-bottom: 15px;
|
||||
border-bottom: 1px solid #dee2e6;
|
||||
}
|
||||
|
||||
.modules-header h3 {
|
||||
margin: 0 0 10px 0;
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
.modules-header p {
|
||||
margin: 0 0 15px 0;
|
||||
color: #666;
|
||||
}
|
||||
|
||||
.module-deploy-card {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
padding: 20px;
|
||||
background: white;
|
||||
border: 1px solid #e9ecef;
|
||||
border-radius: var(--radius-md);
|
||||
margin-bottom: 15px;
|
||||
transition: all 0.2s;
|
||||
box-shadow: 0 1px 3px rgba(0, 0, 0, 0.1);
|
||||
}
|
||||
|
||||
.module-deploy-card:hover {
|
||||
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.15);
|
||||
transform: translateY(-2px);
|
||||
}
|
||||
|
||||
.module-content {
|
||||
flex: 1;
|
||||
margin-bottom: 20px;
|
||||
}
|
||||
|
||||
.module-content h4 {
|
||||
margin: 0 0 8px 0;
|
||||
color: var(--color-primary);
|
||||
font-size: 1.2rem;
|
||||
font-weight: 600;
|
||||
}
|
||||
|
||||
.module-content p {
|
||||
margin: 0 0 12px 0;
|
||||
color: #666;
|
||||
font-size: 14px;
|
||||
line-height: 1.4;
|
||||
}
|
||||
|
||||
.module-features {
|
||||
display: flex;
|
||||
flex-wrap: wrap;
|
||||
gap: 6px;
|
||||
}
|
||||
|
||||
.feature-tag {
|
||||
background: linear-gradient(135deg, #e3f2fd, #bbdefb);
|
||||
color: #1976d2;
|
||||
padding: 4px 10px;
|
||||
border-radius: 12px;
|
||||
font-size: 11px;
|
||||
font-weight: 500;
|
||||
border: 1px solid #90caf9;
|
||||
}
|
||||
|
||||
.module-actions {
|
||||
display: flex;
|
||||
justify-content: center;
|
||||
}
|
||||
|
||||
.btn-deploy {
|
||||
background: linear-gradient(135deg, var(--color-primary), var(--color-primary-dark));
|
||||
color: white;
|
||||
padding: 10px 20px;
|
||||
border: none;
|
||||
border-radius: var(--radius-md);
|
||||
cursor: pointer;
|
||||
font-size: 14px;
|
||||
font-weight: 500;
|
||||
transition: all 0.3s;
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
gap: 8px;
|
||||
box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1);
|
||||
}
|
||||
|
||||
.btn-deploy:hover:not(:disabled) {
|
||||
background: linear-gradient(135deg, var(--color-primary-dark), var(--color-primary));
|
||||
box-shadow: 0 4px 8px rgba(0, 0, 0, 0.2);
|
||||
transform: translateY(-1px);
|
||||
}
|
||||
|
||||
.btn-deploy:disabled {
|
||||
opacity: 0.6;
|
||||
cursor: not-allowed;
|
||||
transform: none;
|
||||
}
|
||||
|
||||
/* Форма добавления модуля */
|
||||
.add-module-form {
|
||||
background: #f8f9fa;
|
||||
@@ -770,4 +1206,20 @@ onMounted(() => {
|
||||
grid-template-columns: 1fr;
|
||||
}
|
||||
}
|
||||
|
||||
/* Адаптивность для блоков деплоя */
|
||||
@media (max-width: 768px) {
|
||||
.module-deploy-card {
|
||||
padding: 15px;
|
||||
}
|
||||
|
||||
.module-content {
|
||||
margin-bottom: 15px;
|
||||
}
|
||||
|
||||
.btn-deploy {
|
||||
width: 100%;
|
||||
justify-content: center;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
|
||||
@@ -22,7 +22,8 @@
|
||||
<!-- Заголовок -->
|
||||
<div class="page-header">
|
||||
<div class="header-content">
|
||||
<h1>Токены DLE</h1>
|
||||
<h1>Управление токенами DLE</h1>
|
||||
<p>Создание предложений для перевода токенов через систему голосования</p>
|
||||
<div v-if="selectedDle" class="dle-info">
|
||||
<span class="dle-name">{{ selectedDle.name }} ({{ selectedDle.symbol }})</span>
|
||||
<span class="dle-address">{{ shortenAddress(selectedDle.dleAddress) }}</span>
|
||||
@@ -48,10 +49,8 @@
|
||||
<div class="info-card">
|
||||
<h3>Ваш баланс</h3>
|
||||
<p class="token-amount">{{ userBalance }} {{ tokenSymbol }}</p>
|
||||
</div>
|
||||
<div class="info-card">
|
||||
<h3>Кворум</h3>
|
||||
<p class="token-amount">{{ quorumPercentage }}%</p>
|
||||
<p v-if="currentUserAddress" class="user-address">{{ shortenAddress(currentUserAddress) }}</p>
|
||||
<p v-else class="no-wallet">Кошелек не подключен</p>
|
||||
</div>
|
||||
<div class="info-card">
|
||||
<h3>Цена токена</h3>
|
||||
@@ -60,110 +59,70 @@
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Трансфер токенов -->
|
||||
<!-- Перевод токенов через governance -->
|
||||
<div class="transfer-section">
|
||||
<h2>Перевод токенов</h2>
|
||||
<form @submit.prevent="transferTokens" class="transfer-form">
|
||||
<div class="form-row">
|
||||
<div class="form-group">
|
||||
<label for="recipient">Получатель:</label>
|
||||
<input
|
||||
id="recipient"
|
||||
v-model="transferData.recipient"
|
||||
type="text"
|
||||
placeholder="0x..."
|
||||
required
|
||||
>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="amount">Количество токенов:</label>
|
||||
<input
|
||||
id="amount"
|
||||
v-model="transferData.amount"
|
||||
type="number"
|
||||
min="0.01"
|
||||
step="0.01"
|
||||
placeholder="0.00"
|
||||
required
|
||||
>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<h2>Перевод токенов через Governance</h2>
|
||||
<p class="section-description">
|
||||
Создайте предложение для перевода токенов через систему голосования.
|
||||
Токены будут переведены от имени DLE после одобрения кворумом.
|
||||
<strong>Важно:</strong> Перевод через governance будет выполнен во всех поддерживаемых сетях DLE.
|
||||
</p>
|
||||
|
||||
<form @submit.prevent="createTransferProposal" class="transfer-form">
|
||||
<div class="form-group">
|
||||
<label for="transferDescription">Описание (опционально):</label>
|
||||
<label for="proposal-recipient">Адрес получателя:</label>
|
||||
<input
|
||||
id="proposal-recipient"
|
||||
v-model="proposalData.recipient"
|
||||
type="text"
|
||||
placeholder="0x..."
|
||||
required
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="proposal-amount">Количество токенов:</label>
|
||||
<input
|
||||
id="proposal-amount"
|
||||
v-model="proposalData.amount"
|
||||
type="number"
|
||||
step="0.000001"
|
||||
placeholder="0.0"
|
||||
required
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="proposal-description">Описание предложения:</label>
|
||||
<textarea
|
||||
id="transferDescription"
|
||||
v-model="transferData.description"
|
||||
placeholder="Укажите причину перевода..."
|
||||
rows="3"
|
||||
id="proposal-description"
|
||||
v-model="proposalData.description"
|
||||
placeholder="Опишите причину перевода токенов..."
|
||||
required
|
||||
></textarea>
|
||||
</div>
|
||||
|
||||
<button type="submit" class="btn-primary" :disabled="isTransferring">
|
||||
{{ isTransferring ? 'Перевод...' : 'Перевести токены' }}
|
||||
</button>
|
||||
</form>
|
||||
</div>
|
||||
|
||||
<!-- Распределение токенов -->
|
||||
<div class="distribution-section">
|
||||
<h2>Распределение токенов</h2>
|
||||
<form @submit.prevent="distributeTokens" class="distribution-form">
|
||||
<div class="form-group">
|
||||
<label for="distributionType">Тип распределения:</label>
|
||||
<select id="distributionType" v-model="distributionData.type" required>
|
||||
<option value="">Выберите тип</option>
|
||||
<option value="partners">Партнерам</option>
|
||||
<option value="employees">Сотрудникам</option>
|
||||
<option value="investors">Инвесторам</option>
|
||||
<option value="custom">Пользовательское</option>
|
||||
</select>
|
||||
<label for="proposal-duration">Длительность голосования (часы):</label>
|
||||
<input
|
||||
id="proposal-duration"
|
||||
v-model="proposalData.duration"
|
||||
type="number"
|
||||
min="1"
|
||||
max="168"
|
||||
placeholder="24"
|
||||
required
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label>Получатели:</label>
|
||||
<div class="recipients-list">
|
||||
<div
|
||||
v-for="(recipient, index) in distributionData.recipients"
|
||||
:key="index"
|
||||
class="recipient-item"
|
||||
>
|
||||
<input
|
||||
v-model="recipient.address"
|
||||
type="text"
|
||||
placeholder="Адрес получателя"
|
||||
required
|
||||
>
|
||||
<input
|
||||
v-model="recipient.amount"
|
||||
type="number"
|
||||
placeholder="Количество"
|
||||
min="0.01"
|
||||
step="0.01"
|
||||
required
|
||||
>
|
||||
<button
|
||||
type="button"
|
||||
@click="removeRecipient(index)"
|
||||
class="btn-remove"
|
||||
>
|
||||
✕
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
<button
|
||||
type="button"
|
||||
@click="addRecipient"
|
||||
class="btn-secondary"
|
||||
>
|
||||
+ Добавить получателя
|
||||
</button>
|
||||
</div>
|
||||
|
||||
<button type="submit" class="btn-primary" :disabled="isDistributing">
|
||||
{{ isDistributing ? 'Распределение...' : 'Распределить токены' }}
|
||||
|
||||
<button type="submit" class="btn-primary" :disabled="isCreatingProposal">
|
||||
{{ isCreatingProposal ? 'Создание предложения...' : 'Создать предложение' }}
|
||||
</button>
|
||||
|
||||
<!-- Статус предложения -->
|
||||
<div v-if="proposalStatus" class="proposal-status">
|
||||
<p class="status-message">{{ proposalStatus }}</p>
|
||||
</div>
|
||||
</form>
|
||||
</div>
|
||||
|
||||
@@ -199,6 +158,8 @@ import { useRouter, useRoute } from 'vue-router';
|
||||
import BaseLayout from '../../components/BaseLayout.vue';
|
||||
import { getTokenBalance, getTotalSupply, getTokenHolders } from '../../services/tokensService.js';
|
||||
import api from '../../api/axios';
|
||||
import { ethers } from 'ethers';
|
||||
import { createTransferTokensProposal } from '../../utils/dle-contract.js';
|
||||
|
||||
// Определяем props
|
||||
const props = defineProps({
|
||||
@@ -225,30 +186,48 @@ const dleAddress = computed(() => {
|
||||
const selectedDle = ref(null);
|
||||
const isLoadingDle = ref(false);
|
||||
|
||||
// Состояние
|
||||
const isTransferring = ref(false);
|
||||
const isDistributing = ref(false);
|
||||
// Состояние для предложения о переводе токенов через governance
|
||||
const isCreatingProposal = ref(false);
|
||||
const proposalStatus = ref('');
|
||||
|
||||
// Данные токенов (загружаются из блокчейна)
|
||||
const tokenSymbol = computed(() => selectedDle.value?.symbol || '');
|
||||
const totalSupply = computed(() => selectedDle.value?.totalSupply || 0);
|
||||
const userBalance = computed(() => selectedDle.value?.deployerBalance || 0);
|
||||
const quorumPercentage = computed(() => selectedDle.value?.quorumPercentage || 0);
|
||||
const totalSupply = ref(0);
|
||||
const userBalance = ref(0);
|
||||
const deployerBalance = ref(0);
|
||||
const quorumPercentage = ref(0);
|
||||
const tokenPrice = ref(0);
|
||||
|
||||
// Данные трансфера
|
||||
const transferData = ref({
|
||||
// Данные для формы
|
||||
const proposalData = ref({
|
||||
recipient: '',
|
||||
amount: '',
|
||||
description: ''
|
||||
description: '',
|
||||
duration: 86400, // 24 часа по умолчанию
|
||||
governanceChainId: 11155111, // Sepolia по умолчанию
|
||||
targetChains: [11155111] // Sepolia по умолчанию
|
||||
});
|
||||
|
||||
// Данные распределения
|
||||
const distributionData = ref({
|
||||
type: '',
|
||||
recipients: [
|
||||
{ address: '', amount: '' }
|
||||
]
|
||||
// Получаем адрес текущего пользователя
|
||||
const currentUserAddress = computed(() => {
|
||||
console.log('Проверяем identities:', props.identities);
|
||||
|
||||
// Получаем адрес из props или из window.ethereum
|
||||
if (props.identities && props.identities.length > 0) {
|
||||
const walletIdentity = props.identities.find(id => id.provider === 'wallet');
|
||||
console.log('Найден wallet identity:', walletIdentity);
|
||||
if (walletIdentity) {
|
||||
return walletIdentity.provider_id;
|
||||
}
|
||||
}
|
||||
|
||||
// Fallback: пытаемся получить из window.ethereum
|
||||
if (window.ethereum && window.ethereum.selectedAddress) {
|
||||
console.log('Получаем адрес из window.ethereum:', window.ethereum.selectedAddress);
|
||||
return window.ethereum.selectedAddress;
|
||||
}
|
||||
|
||||
console.log('Адрес пользователя не найден');
|
||||
return null;
|
||||
});
|
||||
|
||||
// Держатели токенов (загружаются из блокчейна)
|
||||
@@ -273,6 +252,9 @@ async function loadDleData() {
|
||||
selectedDle.value = blockchainData;
|
||||
console.log('Загружены данные DLE из блокчейна:', blockchainData);
|
||||
|
||||
// Загружаем баланс текущего пользователя
|
||||
await loadUserBalance();
|
||||
|
||||
// Загружаем держателей токенов (если есть API)
|
||||
await loadTokenHolders();
|
||||
} else {
|
||||
@@ -285,6 +267,35 @@ async function loadDleData() {
|
||||
}
|
||||
}
|
||||
|
||||
// Новая функция для загрузки баланса текущего пользователя
|
||||
async function loadUserBalance() {
|
||||
if (!currentUserAddress.value || !dleAddress.value) {
|
||||
userBalance.value = 0;
|
||||
console.log('Не удается загрузить баланс: нет адреса пользователя или DLE');
|
||||
return;
|
||||
}
|
||||
|
||||
try {
|
||||
console.log('Загружаем баланс для пользователя:', currentUserAddress.value);
|
||||
|
||||
const response = await api.post('/blockchain/get-token-balance', {
|
||||
dleAddress: dleAddress.value,
|
||||
account: currentUserAddress.value
|
||||
});
|
||||
|
||||
if (response.data.success) {
|
||||
userBalance.value = parseFloat(response.data.data.balance);
|
||||
console.log('Баланс пользователя загружен:', userBalance.value);
|
||||
} else {
|
||||
console.warn('Не удалось загрузить баланс пользователя:', response.data.error);
|
||||
userBalance.value = 0;
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('Ошибка загрузки баланса пользователя:', error);
|
||||
userBalance.value = 0;
|
||||
}
|
||||
}
|
||||
|
||||
async function loadTokenHolders() {
|
||||
try {
|
||||
// Здесь можно добавить загрузку держателей токенов из блокчейна
|
||||
@@ -301,99 +312,132 @@ function shortenAddress(address) {
|
||||
}
|
||||
|
||||
// Методы
|
||||
const transferTokens = async () => {
|
||||
if (isTransferring.value) return;
|
||||
|
||||
try {
|
||||
isTransferring.value = true;
|
||||
|
||||
// Здесь будет логика трансфера токенов
|
||||
// console.log('Трансфер токенов:', transferData.value);
|
||||
|
||||
// Временная логика
|
||||
const amount = parseFloat(transferData.value.amount);
|
||||
if (amount > userBalance.value) {
|
||||
alert('Недостаточно токенов для перевода');
|
||||
return;
|
||||
}
|
||||
|
||||
userBalance.value -= amount;
|
||||
|
||||
// Сброс формы
|
||||
transferData.value = {
|
||||
recipient: '',
|
||||
amount: '',
|
||||
description: ''
|
||||
};
|
||||
|
||||
alert('Токены успешно переведены!');
|
||||
|
||||
} catch (error) {
|
||||
// console.error('Ошибка трансфера токенов:', error);
|
||||
alert('Ошибка при переводе токенов');
|
||||
} finally {
|
||||
isTransferring.value = false;
|
||||
}
|
||||
};
|
||||
|
||||
const distributeTokens = async () => {
|
||||
if (isDistributing.value) return;
|
||||
|
||||
try {
|
||||
isDistributing.value = true;
|
||||
|
||||
// Здесь будет логика распределения токенов
|
||||
// console.log('Распределение токенов:', distributionData.value);
|
||||
|
||||
// Временная логика
|
||||
const totalAmount = distributionData.value.recipients.reduce((sum, recipient) => {
|
||||
return sum + parseFloat(recipient.amount || 0);
|
||||
}, 0);
|
||||
|
||||
if (totalAmount > userBalance.value) {
|
||||
alert('Недостаточно токенов для распределения');
|
||||
return;
|
||||
}
|
||||
|
||||
userBalance.value -= totalAmount;
|
||||
|
||||
// Сброс формы
|
||||
distributionData.value = {
|
||||
type: '',
|
||||
recipients: [{ address: '', amount: '' }]
|
||||
};
|
||||
|
||||
alert('Токены успешно распределены!');
|
||||
|
||||
} catch (error) {
|
||||
// console.error('Ошибка распределения токенов:', error);
|
||||
alert('Ошибка при распределении токенов');
|
||||
} finally {
|
||||
isDistributing.value = false;
|
||||
}
|
||||
};
|
||||
|
||||
const addRecipient = () => {
|
||||
distributionData.value.recipients.push({ address: '', amount: '' });
|
||||
};
|
||||
|
||||
const removeRecipient = (index) => {
|
||||
if (distributionData.value.recipients.length > 1) {
|
||||
distributionData.value.recipients.splice(index, 1);
|
||||
}
|
||||
};
|
||||
|
||||
const formatAddress = (address) => {
|
||||
if (!address) return '';
|
||||
return address.substring(0, 6) + '...' + address.substring(address.length - 4);
|
||||
};
|
||||
|
||||
// Функция создания предложения о переводе токенов через governance
|
||||
const createTransferProposal = async () => {
|
||||
if (isCreatingProposal.value) return;
|
||||
|
||||
try {
|
||||
// Проверяем подключение к кошельку
|
||||
if (!window.ethereum) {
|
||||
alert('Пожалуйста, установите MetaMask или другой Web3 кошелек');
|
||||
return;
|
||||
}
|
||||
|
||||
// Проверяем, что пользователь подключен
|
||||
if (!currentUserAddress.value) {
|
||||
alert('Пожалуйста, подключите кошелек');
|
||||
return;
|
||||
}
|
||||
|
||||
// Валидация данных
|
||||
const recipient = proposalData.value.recipient.trim();
|
||||
const amount = parseFloat(proposalData.value.amount);
|
||||
const description = proposalData.value.description.trim();
|
||||
|
||||
if (!recipient) {
|
||||
alert('Пожалуйста, укажите адрес получателя');
|
||||
return;
|
||||
}
|
||||
|
||||
// Проверяем, что адрес получателя является корректным Ethereum адресом
|
||||
if (!ethers.isAddress(recipient)) {
|
||||
alert('Пожалуйста, укажите корректный Ethereum адрес получателя');
|
||||
return;
|
||||
}
|
||||
|
||||
if (!amount || amount <= 0) {
|
||||
alert('Пожалуйста, укажите корректное количество токенов');
|
||||
return;
|
||||
}
|
||||
|
||||
if (!description) {
|
||||
alert('Пожалуйста, укажите описание предложения');
|
||||
return;
|
||||
}
|
||||
|
||||
// Проверяем, что получатель не является отправителем
|
||||
if (recipient.toLowerCase() === currentUserAddress.value.toLowerCase()) {
|
||||
alert('Нельзя отправить токены самому себе');
|
||||
return;
|
||||
}
|
||||
|
||||
isCreatingProposal.value = true;
|
||||
proposalStatus.value = 'Создание предложения...';
|
||||
|
||||
// Создаем предложение
|
||||
const result = await createTransferTokensProposal(dleAddress.value, {
|
||||
recipient: recipient,
|
||||
amount: amount,
|
||||
description: description,
|
||||
duration: proposalData.value.duration * 3600, // Конвертируем часы в секунды
|
||||
governanceChainId: proposalData.value.governanceChainId,
|
||||
targetChains: proposalData.value.targetChains
|
||||
});
|
||||
|
||||
proposalStatus.value = 'Предложение создано!';
|
||||
console.log('Предложение о переводе токенов создано:', result);
|
||||
|
||||
// Сброс формы
|
||||
proposalData.value = {
|
||||
recipient: '',
|
||||
amount: '',
|
||||
description: '',
|
||||
duration: 86400,
|
||||
governanceChainId: 11155111,
|
||||
targetChains: [11155111]
|
||||
};
|
||||
|
||||
// Очищаем статус через 5 секунд
|
||||
setTimeout(() => {
|
||||
proposalStatus.value = '';
|
||||
}, 5000);
|
||||
|
||||
alert(`Предложение о переводе токенов создано!\nID предложения: ${result.proposalId}\nХеш транзакции: ${result.txHash}`);
|
||||
|
||||
} catch (error) {
|
||||
console.error('Ошибка создания предложения о переводе токенов:', error);
|
||||
|
||||
// Очищаем статус предложения
|
||||
proposalStatus.value = '';
|
||||
|
||||
let errorMessage = 'Ошибка создания предложения о переводе токенов';
|
||||
|
||||
if (error.code === 4001) {
|
||||
errorMessage = 'Транзакция отменена пользователем';
|
||||
} else if (error.message && error.message.includes('insufficient funds')) {
|
||||
errorMessage = 'Недостаточно ETH для оплаты газа';
|
||||
} else if (error.message && error.message.includes('execution reverted')) {
|
||||
errorMessage = 'Ошибка выполнения транзакции. Проверьте данные и попробуйте снова';
|
||||
} else if (error.message) {
|
||||
errorMessage = error.message;
|
||||
}
|
||||
|
||||
alert(errorMessage);
|
||||
} finally {
|
||||
isCreatingProposal.value = false;
|
||||
}
|
||||
};
|
||||
|
||||
// Отслеживаем изменения в адресе DLE
|
||||
watch(dleAddress, (newAddress) => {
|
||||
if (newAddress) {
|
||||
loadDleData();
|
||||
}
|
||||
}, { immediate: true });
|
||||
|
||||
// Отслеживаем изменения адреса пользователя
|
||||
watch(currentUserAddress, (newAddress) => {
|
||||
if (newAddress && dleAddress.value) {
|
||||
loadUserBalance();
|
||||
} else {
|
||||
userBalance.value = 0;
|
||||
}
|
||||
}, { immediate: true });
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
@@ -488,14 +532,12 @@ watch(dleAddress, (newAddress) => {
|
||||
/* Секции */
|
||||
.token-info-section,
|
||||
.transfer-section,
|
||||
.distribution-section,
|
||||
.holders-section {
|
||||
margin-bottom: 40px;
|
||||
}
|
||||
|
||||
.token-info-section h2,
|
||||
.transfer-section h2,
|
||||
.distribution-section h2,
|
||||
.holders-section h2 {
|
||||
color: var(--color-primary);
|
||||
margin-bottom: 20px;
|
||||
@@ -532,9 +574,26 @@ watch(dleAddress, (newAddress) => {
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
.user-address {
|
||||
font-family: monospace;
|
||||
font-size: 0.75rem;
|
||||
color: #666;
|
||||
margin: 5px 0 0 0;
|
||||
background: #f8f9fa;
|
||||
padding: 2px 6px;
|
||||
border-radius: 3px;
|
||||
display: inline-block;
|
||||
}
|
||||
|
||||
.no-wallet {
|
||||
font-size: 0.75rem;
|
||||
color: #dc3545;
|
||||
margin: 5px 0 0 0;
|
||||
font-style: italic;
|
||||
}
|
||||
|
||||
/* Формы */
|
||||
.transfer-form,
|
||||
.distribution-form {
|
||||
.transfer-form {
|
||||
background: #f8f9fa;
|
||||
padding: 25px;
|
||||
border-radius: var(--radius-lg);
|
||||
@@ -574,81 +633,25 @@ watch(dleAddress, (newAddress) => {
|
||||
min-height: 80px;
|
||||
}
|
||||
|
||||
/* Получатели */
|
||||
.recipients-list {
|
||||
display: grid;
|
||||
gap: 15px;
|
||||
margin-bottom: 15px;
|
||||
/* Статус предложения */
|
||||
.proposal-status {
|
||||
margin-top: 20px;
|
||||
padding: 15px;
|
||||
background: #e8f5e8;
|
||||
border-radius: var(--radius-sm);
|
||||
border-left: 4px solid #28a745;
|
||||
}
|
||||
|
||||
.recipient-item {
|
||||
display: grid;
|
||||
grid-template-columns: 1fr 1fr auto;
|
||||
gap: 15px;
|
||||
align-items: center;
|
||||
.proposal-status .status-message {
|
||||
color: #28a745;
|
||||
}
|
||||
|
||||
.btn-remove {
|
||||
background: #dc3545;
|
||||
color: white;
|
||||
border: none;
|
||||
border-radius: 50%;
|
||||
width: 30px;
|
||||
height: 30px;
|
||||
cursor: pointer;
|
||||
font-size: 14px;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
}
|
||||
|
||||
.btn-remove:hover {
|
||||
background: #c82333;
|
||||
}
|
||||
|
||||
/* Держатели токенов */
|
||||
.holders-list {
|
||||
display: grid;
|
||||
gap: 15px;
|
||||
}
|
||||
|
||||
.holder-item {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: center;
|
||||
padding: 20px;
|
||||
background: white;
|
||||
border-radius: var(--radius-lg);
|
||||
border: 1px solid #e9ecef;
|
||||
transition: all 0.3s ease;
|
||||
}
|
||||
|
||||
.holder-item:hover {
|
||||
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.1);
|
||||
}
|
||||
|
||||
.holder-info {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 5px;
|
||||
}
|
||||
|
||||
.holder-address {
|
||||
font-family: monospace;
|
||||
font-size: 0.9rem;
|
||||
/* Описание секции */
|
||||
.section-description {
|
||||
color: var(--color-grey-dark);
|
||||
}
|
||||
|
||||
.holder-balance {
|
||||
font-size: 1.1rem;
|
||||
font-weight: 600;
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
.holder-percentage {
|
||||
font-size: 0.9rem;
|
||||
color: var(--color-grey-dark);
|
||||
font-weight: 600;
|
||||
font-size: 0.95rem;
|
||||
margin-bottom: 20px;
|
||||
line-height: 1.5;
|
||||
}
|
||||
|
||||
/* Кнопки */
|
||||
|
||||
@@ -0,0 +1,329 @@
|
||||
<!--
|
||||
Copyright (c) 2024-2025 Тарабанов Александр Викторович
|
||||
All rights reserved.
|
||||
|
||||
This software is proprietary and confidential.
|
||||
Unauthorized copying, modification, or distribution is prohibited.
|
||||
|
||||
For licensing inquiries: info@hb3-accelerator.com
|
||||
Website: https://hb3-accelerator.com
|
||||
GitHub: https://github.com/HB3-ACCELERATOR
|
||||
-->
|
||||
|
||||
<template>
|
||||
<BaseLayout
|
||||
:is-authenticated="isAuthenticated"
|
||||
:identities="identities"
|
||||
:token-balances="tokenBalances"
|
||||
:is-loading-tokens="isLoadingTokens"
|
||||
@auth-action-completed="$emit('auth-action-completed')"
|
||||
>
|
||||
<div class="application-module-deploy">
|
||||
<!-- Заголовок -->
|
||||
<div class="page-header">
|
||||
<div class="header-content">
|
||||
<h1>Деплой ApplicationModule</h1>
|
||||
<p>Управление вызовом функций приложения через предложения и голосование</p>
|
||||
<p v-if="dleAddress" class="dle-address">
|
||||
<strong>DLE:</strong> {{ dleAddress }}
|
||||
</p>
|
||||
</div>
|
||||
<button class="close-btn" @click="router.push('/management/modules')">×</button>
|
||||
</div>
|
||||
|
||||
<!-- Информация о модуле -->
|
||||
<div class="module-info">
|
||||
<div class="info-card">
|
||||
<h3>🖥️ ApplicationModule</h3>
|
||||
<div class="info-grid">
|
||||
<div class="info-item">
|
||||
<strong>Назначение:</strong> Управление функциями приложения через DLE
|
||||
</div>
|
||||
<div class="info-item">
|
||||
<strong>Функции:</strong> Создание предложений для вызова API, голосование за операции
|
||||
</div>
|
||||
<div class="info-item">
|
||||
<strong>Безопасность:</strong> Все операции приложения через кворум токен-холдеров
|
||||
</div>
|
||||
<div class="info-item">
|
||||
<strong>Примеры:</strong> Удаление пользователей, изменение настроек, обновление данных
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Детальное описание -->
|
||||
<div class="module-details">
|
||||
<div class="details-card">
|
||||
<h3>📋 Как работает ApplicationModule</h3>
|
||||
<div class="details-content">
|
||||
<div class="detail-step">
|
||||
<div class="step-number">1</div>
|
||||
<div class="step-content">
|
||||
<h4>Создание предложения</h4>
|
||||
<p>Токен-холдер создает предложение для выполнения операции в приложении (например, удаление пользователя, изменение настроек)</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="detail-step">
|
||||
<div class="step-number">2</div>
|
||||
<div class="step-content">
|
||||
<h4>Голосование</h4>
|
||||
<p>Все токен-холдеры голосуют за или против предложения в течение установленного времени</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="detail-step">
|
||||
<div class="step-number">3</div>
|
||||
<div class="step-content">
|
||||
<h4>Исполнение</h4>
|
||||
<p>При достижении кворума предложение исполняется - вызывается соответствующая функция приложения</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="detail-step">
|
||||
<div class="step-number">4</div>
|
||||
<div class="step-content">
|
||||
<h4>Аудит</h4>
|
||||
<p>Все операции логируются в блокчейне для полной прозрачности и подотчетности</p>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Форма деплоя будет добавлена позже -->
|
||||
<div class="deploy-form-placeholder">
|
||||
<div class="placeholder-content">
|
||||
<h3>🚧 Форма деплоя в разработке</h3>
|
||||
<p>Здесь будет форма для деплоя ApplicationModule</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
</div>
|
||||
</BaseLayout>
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import { defineProps, defineEmits, ref, onMounted } from 'vue';
|
||||
import { useRouter, useRoute } from 'vue-router';
|
||||
import BaseLayout from '../../../components/BaseLayout.vue';
|
||||
|
||||
// Определяем props
|
||||
const props = defineProps({
|
||||
isAuthenticated: { type: Boolean, default: false },
|
||||
identities: { type: Array, default: () => [] },
|
||||
tokenBalances: { type: Object, default: () => ({}) },
|
||||
isLoadingTokens: { type: Boolean, default: false }
|
||||
});
|
||||
|
||||
// Определяем emits
|
||||
const emit = defineEmits(['auth-action-completed']);
|
||||
|
||||
const router = useRouter();
|
||||
const route = useRoute();
|
||||
|
||||
// Состояние
|
||||
const isLoading = ref(false);
|
||||
const dleAddress = ref(route.query.address || null);
|
||||
|
||||
// Инициализация
|
||||
onMounted(() => {
|
||||
console.log('[ApplicationModuleDeployView] Страница загружена');
|
||||
});
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
.application-module-deploy {
|
||||
padding: 20px;
|
||||
background-color: var(--color-white);
|
||||
border-radius: var(--radius-lg);
|
||||
box-shadow: 0 2px 10px rgba(0, 0, 0, 0.1);
|
||||
margin-top: 20px;
|
||||
margin-bottom: 20px;
|
||||
}
|
||||
|
||||
.page-header {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: center;
|
||||
margin-bottom: 30px;
|
||||
padding-bottom: 20px;
|
||||
border-bottom: 2px solid #f0f0f0;
|
||||
}
|
||||
|
||||
.page-header h1 {
|
||||
color: var(--color-primary);
|
||||
font-size: 2rem;
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
.page-header p {
|
||||
margin: 10px 0 0 0;
|
||||
color: #666;
|
||||
}
|
||||
|
||||
.dle-address {
|
||||
margin-top: 10px !important;
|
||||
font-family: monospace;
|
||||
background: #f8f9fa;
|
||||
padding: 8px 12px;
|
||||
border-radius: var(--radius-sm);
|
||||
border: 1px solid #dee2e6;
|
||||
}
|
||||
|
||||
.close-btn {
|
||||
background: none;
|
||||
border: none;
|
||||
font-size: 1.5rem;
|
||||
cursor: pointer;
|
||||
color: #666;
|
||||
padding: 0;
|
||||
width: 30px;
|
||||
height: 30px;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
border-radius: 50%;
|
||||
transition: all 0.2s;
|
||||
}
|
||||
|
||||
.close-btn:hover {
|
||||
background: #f0f0f0;
|
||||
color: #333;
|
||||
}
|
||||
|
||||
/* Информация о модуле */
|
||||
.module-info {
|
||||
margin-bottom: 30px;
|
||||
}
|
||||
|
||||
.info-card {
|
||||
background: #f8f9fa;
|
||||
border-radius: var(--radius-md);
|
||||
padding: 20px;
|
||||
border: 1px solid #e9ecef;
|
||||
}
|
||||
|
||||
.info-card h3 {
|
||||
margin: 0 0 15px 0;
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
.info-grid {
|
||||
display: grid;
|
||||
grid-template-columns: repeat(auto-fit, minmax(250px, 1fr));
|
||||
gap: 15px;
|
||||
}
|
||||
|
||||
.info-item {
|
||||
padding: 10px;
|
||||
background: white;
|
||||
border-radius: var(--radius-sm);
|
||||
border: 1px solid #dee2e6;
|
||||
}
|
||||
|
||||
.info-item strong {
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
/* Детальное описание */
|
||||
.module-details {
|
||||
margin-bottom: 30px;
|
||||
}
|
||||
|
||||
.details-card {
|
||||
background: #f8f9fa;
|
||||
border-radius: var(--radius-md);
|
||||
padding: 20px;
|
||||
border: 1px solid #e9ecef;
|
||||
}
|
||||
|
||||
.details-card h3 {
|
||||
margin: 0 0 20px 0;
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
.details-content {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 20px;
|
||||
}
|
||||
|
||||
.detail-step {
|
||||
display: flex;
|
||||
align-items: flex-start;
|
||||
gap: 15px;
|
||||
padding: 15px;
|
||||
background: white;
|
||||
border-radius: var(--radius-sm);
|
||||
border: 1px solid #dee2e6;
|
||||
}
|
||||
|
||||
.step-number {
|
||||
background: var(--color-primary);
|
||||
color: white;
|
||||
width: 30px;
|
||||
height: 30px;
|
||||
border-radius: 50%;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
font-weight: bold;
|
||||
font-size: 14px;
|
||||
flex-shrink: 0;
|
||||
}
|
||||
|
||||
.step-content h4 {
|
||||
margin: 0 0 8px 0;
|
||||
color: var(--color-primary);
|
||||
font-size: 16px;
|
||||
}
|
||||
|
||||
.step-content p {
|
||||
margin: 0;
|
||||
color: #666;
|
||||
line-height: 1.5;
|
||||
}
|
||||
|
||||
/* Плейсхолдер для формы */
|
||||
.deploy-form-placeholder {
|
||||
background: #f8f9fa;
|
||||
border-radius: var(--radius-md);
|
||||
padding: 40px;
|
||||
text-align: center;
|
||||
border: 2px dashed #dee2e6;
|
||||
}
|
||||
|
||||
.placeholder-content h3 {
|
||||
color: var(--color-primary);
|
||||
margin-bottom: 10px;
|
||||
}
|
||||
|
||||
.placeholder-content p {
|
||||
color: #666;
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
/* Адаптивность */
|
||||
@media (max-width: 768px) {
|
||||
.info-grid {
|
||||
grid-template-columns: 1fr;
|
||||
}
|
||||
|
||||
.page-header {
|
||||
flex-direction: column;
|
||||
align-items: flex-start;
|
||||
gap: 15px;
|
||||
}
|
||||
|
||||
.detail-step {
|
||||
flex-direction: column;
|
||||
text-align: center;
|
||||
}
|
||||
|
||||
.step-number {
|
||||
align-self: center;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
515
frontend/src/views/smartcontracts/modules/BurnModuleDeploy.vue
Normal file
515
frontend/src/views/smartcontracts/modules/BurnModuleDeploy.vue
Normal file
@@ -0,0 +1,515 @@
|
||||
<!--
|
||||
Copyright (c) 2024-2025 Тарабанов Александр Викторович
|
||||
All rights reserved.
|
||||
|
||||
This software is proprietary and confidential.
|
||||
Unauthorized copying, modification, or distribution is prohibited.
|
||||
|
||||
For licensing inquiries: info@hb3-accelerator.com
|
||||
Website: https://hb3-accelerator.com
|
||||
GitHub: https://github.com/HB3-ACCELERATOR
|
||||
-->
|
||||
|
||||
<template>
|
||||
<BaseLayout
|
||||
:is-authenticated="isAuthenticated"
|
||||
:identities="identities"
|
||||
:token-balances="tokenBalances"
|
||||
:is-loading-tokens="isLoadingTokens"
|
||||
@auth-action-completed="$emit('auth-action-completed')"
|
||||
>
|
||||
<div class="module-deploy-page">
|
||||
<!-- Заголовок -->
|
||||
<div class="page-header">
|
||||
<div class="header-content">
|
||||
<h1>🔥 Деплой BurnModule</h1>
|
||||
<p>Модуль для сжигания токенов DLE через governance</p>
|
||||
<div v-if="selectedDle" class="dle-info">
|
||||
<span class="dle-name">{{ selectedDle.name }} ({{ selectedDle.symbol }})</span>
|
||||
<span class="dle-address">{{ selectedDle.dleAddress }}</span>
|
||||
</div>
|
||||
</div>
|
||||
<button class="close-btn" @click="router.push(`/management/modules?address=${route.query.address}`)">×</button>
|
||||
</div>
|
||||
|
||||
<!-- Описание модуля -->
|
||||
<div class="module-description">
|
||||
<div class="description-card">
|
||||
<h3>📋 Описание BurnModule</h3>
|
||||
<div class="description-content">
|
||||
<p><strong>BurnModule</strong> - это модуль для управления сжиганием токенов DLE через систему governance.</p>
|
||||
|
||||
<h4>🔧 Функциональность:</h4>
|
||||
<ul>
|
||||
<li><strong>Сжигание токенов:</strong> Уменьшение общего предложения токенов DLE</li>
|
||||
<li><strong>Governance:</strong> Все операции требуют голосования и кворума</li>
|
||||
<li><strong>Безопасность:</strong> Контролируемое сжигание через коллективные решения</li>
|
||||
<li><strong>Прозрачность:</strong> Все операции записываются в блокчейн</li>
|
||||
</ul>
|
||||
|
||||
<h4>⚠️ Важные особенности:</h4>
|
||||
<ul>
|
||||
<li>Сжигание токенов возможно только через предложения и голосование</li>
|
||||
<li>Можно сжигать токены из казны DLE или от имени участников</li>
|
||||
<li>Все операции требуют достижения кворума</li>
|
||||
<li>История всех сжиганий сохраняется в блокчейне</li>
|
||||
<li>Сжигание необратимо - токены уничтожаются навсегда</li>
|
||||
</ul>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Форма деплоя -->
|
||||
<div class="deploy-form-section">
|
||||
<div class="form-header">
|
||||
<h3>⚙️ Настройки деплоя</h3>
|
||||
<p>Настройте параметры для деплоя BurnModule</p>
|
||||
</div>
|
||||
|
||||
<form @submit.prevent="deployModule" class="deploy-form">
|
||||
<div class="form-row">
|
||||
<div class="form-group">
|
||||
<label for="moduleName">Название модуля:</label>
|
||||
<input
|
||||
id="moduleName"
|
||||
v-model="deployData.moduleName"
|
||||
type="text"
|
||||
placeholder="BurnModule"
|
||||
required
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="moduleVersion">Версия модуля:</label>
|
||||
<input
|
||||
id="moduleVersion"
|
||||
v-model="deployData.moduleVersion"
|
||||
type="text"
|
||||
placeholder="1.0.0"
|
||||
required
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="moduleDescription">Описание модуля:</label>
|
||||
<textarea
|
||||
id="moduleDescription"
|
||||
v-model="deployData.moduleDescription"
|
||||
placeholder="Модуль для сжигания токенов DLE через governance..."
|
||||
rows="3"
|
||||
required
|
||||
></textarea>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="maxBurnPerProposal">Максимальное сжигание за одно предложение:</label>
|
||||
<input
|
||||
id="maxBurnPerProposal"
|
||||
v-model="deployData.maxBurnPerProposal"
|
||||
type="number"
|
||||
min="1"
|
||||
step="1"
|
||||
placeholder="1000000"
|
||||
required
|
||||
/>
|
||||
<small class="form-help">Максимальное количество токенов, которое можно сжечь за одно предложение</small>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="burnCooldown">Кулдаун между сжиганиями (часы):</label>
|
||||
<input
|
||||
id="burnCooldown"
|
||||
v-model="deployData.burnCooldown"
|
||||
type="number"
|
||||
min="0"
|
||||
step="1"
|
||||
placeholder="24"
|
||||
required
|
||||
/>
|
||||
<small class="form-help">Минимальное время между успешными сжиганиями токенов</small>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="allowTreasuryBurn">Разрешить сжигание из казны:</label>
|
||||
<select
|
||||
id="allowTreasuryBurn"
|
||||
v-model="deployData.allowTreasuryBurn"
|
||||
required
|
||||
>
|
||||
<option value="true">Да</option>
|
||||
<option value="false">Нет</option>
|
||||
</select>
|
||||
<small class="form-help">Разрешить сжигание токенов из казны DLE</small>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="allowUserBurn">Разрешить сжигание от участников:</label>
|
||||
<select
|
||||
id="allowUserBurn"
|
||||
v-model="deployData.allowUserBurn"
|
||||
required
|
||||
>
|
||||
<option value="true">Да</option>
|
||||
<option value="false">Нет</option>
|
||||
</select>
|
||||
<small class="form-help">Разрешить сжигание токенов от имени участников DLE</small>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="deployDescription">Описание предложения для деплоя:</label>
|
||||
<textarea
|
||||
id="deployDescription"
|
||||
v-model="deployData.deployDescription"
|
||||
placeholder="Предложение о деплое BurnModule для управления сжиганием токенов DLE..."
|
||||
rows="3"
|
||||
required
|
||||
></textarea>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="votingDuration">Длительность голосования (часы):</label>
|
||||
<input
|
||||
id="votingDuration"
|
||||
v-model="deployData.votingDuration"
|
||||
type="number"
|
||||
min="1"
|
||||
max="168"
|
||||
placeholder="24"
|
||||
required
|
||||
/>
|
||||
<small class="form-help">Время для голосования по предложению деплоя (1-168 часов)</small>
|
||||
</div>
|
||||
|
||||
<button type="submit" class="btn-primary" :disabled="isDeploying">
|
||||
{{ isDeploying ? 'Деплой...' : 'Деплой BurnModule' }}
|
||||
</button>
|
||||
|
||||
<!-- Статус деплоя -->
|
||||
<div v-if="deployStatus" class="deploy-status">
|
||||
<p class="status-message">{{ deployStatus }}</p>
|
||||
</div>
|
||||
</form>
|
||||
</div>
|
||||
</div>
|
||||
</BaseLayout>
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import { ref, computed, onMounted } from 'vue';
|
||||
import { useRoute, useRouter } from 'vue-router';
|
||||
import BaseLayout from '../../../components/BaseLayout.vue';
|
||||
import api from '../../../api/axios';
|
||||
|
||||
// Props
|
||||
const props = defineProps({
|
||||
isAuthenticated: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
},
|
||||
identities: {
|
||||
type: Array,
|
||||
default: () => []
|
||||
},
|
||||
tokenBalances: {
|
||||
type: Object,
|
||||
default: () => ({})
|
||||
},
|
||||
isLoadingTokens: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
}
|
||||
});
|
||||
|
||||
// Emits
|
||||
const emit = defineEmits(['auth-action-completed']);
|
||||
|
||||
// Router
|
||||
const route = useRoute();
|
||||
const router = useRouter();
|
||||
|
||||
// Состояние
|
||||
const isDeploying = ref(false);
|
||||
const deployStatus = ref('');
|
||||
const selectedDle = ref(null);
|
||||
const isLoadingDle = ref(false);
|
||||
|
||||
// Данные для деплоя
|
||||
const deployData = ref({
|
||||
moduleName: 'BurnModule',
|
||||
moduleVersion: '1.0.0',
|
||||
moduleDescription: 'Модуль для сжигания токенов DLE через governance',
|
||||
maxBurnPerProposal: 1000000,
|
||||
burnCooldown: 24,
|
||||
allowTreasuryBurn: 'true',
|
||||
allowUserBurn: 'true',
|
||||
deployDescription: 'Предложение о деплое BurnModule для управления сжиганием токенов DLE',
|
||||
votingDuration: 24
|
||||
});
|
||||
|
||||
// Получаем адрес DLE из URL
|
||||
const dleAddress = computed(() => route.query.address);
|
||||
|
||||
// Загрузка данных DLE
|
||||
const loadDleData = async () => {
|
||||
if (!dleAddress.value) return;
|
||||
|
||||
try {
|
||||
isLoadingDle.value = true;
|
||||
const response = await api.post('/blockchain/read-dle-info', {
|
||||
dleAddress: dleAddress.value
|
||||
});
|
||||
|
||||
if (response.data.success) {
|
||||
selectedDle.value = response.data.data;
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('Ошибка загрузки данных DLE:', error);
|
||||
} finally {
|
||||
isLoadingDle.value = false;
|
||||
}
|
||||
};
|
||||
|
||||
// Функция деплоя модуля
|
||||
const deployModule = async () => {
|
||||
if (isDeploying.value) return;
|
||||
|
||||
try {
|
||||
isDeploying.value = true;
|
||||
deployStatus.value = 'Подготовка к деплою...';
|
||||
|
||||
// Здесь будет логика деплоя модуля
|
||||
console.log('Деплой BurnModule:', deployData.value);
|
||||
|
||||
// Временная заглушка
|
||||
await new Promise(resolve => setTimeout(resolve, 2000));
|
||||
|
||||
deployStatus.value = 'Модуль успешно развернут!';
|
||||
|
||||
// Очищаем статус через 3 секунды
|
||||
setTimeout(() => {
|
||||
deployStatus.value = '';
|
||||
}, 3000);
|
||||
|
||||
alert('BurnModule успешно развернут!');
|
||||
|
||||
} catch (error) {
|
||||
console.error('Ошибка деплоя модуля:', error);
|
||||
deployStatus.value = 'Ошибка деплоя модуля';
|
||||
|
||||
setTimeout(() => {
|
||||
deployStatus.value = '';
|
||||
}, 3000);
|
||||
|
||||
alert('Ошибка при деплое модуля');
|
||||
} finally {
|
||||
isDeploying.value = false;
|
||||
}
|
||||
};
|
||||
|
||||
// Загружаем данные при монтировании
|
||||
onMounted(() => {
|
||||
loadDleData();
|
||||
});
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
.module-deploy-page {
|
||||
padding: 20px;
|
||||
max-width: 1200px;
|
||||
margin: 0 auto;
|
||||
}
|
||||
|
||||
.page-header {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: flex-start;
|
||||
margin-bottom: 30px;
|
||||
padding-bottom: 20px;
|
||||
border-bottom: 2px solid #f0f0f0;
|
||||
}
|
||||
|
||||
.header-content h1 {
|
||||
color: var(--color-primary);
|
||||
font-size: 2.5rem;
|
||||
margin: 0 0 10px 0;
|
||||
}
|
||||
|
||||
.header-content p {
|
||||
color: var(--color-grey-dark);
|
||||
font-size: 1.1rem;
|
||||
margin: 0 0 15px 0;
|
||||
}
|
||||
|
||||
.dle-info {
|
||||
display: flex;
|
||||
gap: 15px;
|
||||
align-items: center;
|
||||
}
|
||||
|
||||
.dle-name {
|
||||
font-weight: 600;
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
.dle-address {
|
||||
font-family: monospace;
|
||||
color: var(--color-grey-dark);
|
||||
font-size: 0.9rem;
|
||||
}
|
||||
|
||||
.close-btn {
|
||||
background: none;
|
||||
border: none;
|
||||
font-size: 2rem;
|
||||
color: var(--color-grey-dark);
|
||||
cursor: pointer;
|
||||
padding: 5px;
|
||||
}
|
||||
|
||||
.close-btn:hover {
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
.module-description {
|
||||
margin-bottom: 30px;
|
||||
}
|
||||
|
||||
.description-card {
|
||||
background: #f8f9fa;
|
||||
padding: 25px;
|
||||
border-radius: var(--radius-lg);
|
||||
border: 1px solid #e9ecef;
|
||||
}
|
||||
|
||||
.description-card h3 {
|
||||
color: var(--color-primary);
|
||||
margin: 0 0 20px 0;
|
||||
}
|
||||
|
||||
.description-content h4 {
|
||||
color: var(--color-grey-dark);
|
||||
margin: 20px 0 10px 0;
|
||||
}
|
||||
|
||||
.description-content ul {
|
||||
margin: 10px 0;
|
||||
padding-left: 20px;
|
||||
}
|
||||
|
||||
.description-content li {
|
||||
margin: 5px 0;
|
||||
line-height: 1.5;
|
||||
}
|
||||
|
||||
.deploy-form-section {
|
||||
background: white;
|
||||
padding: 30px;
|
||||
border-radius: var(--radius-lg);
|
||||
box-shadow: 0 2px 10px rgba(0, 0, 0, 0.1);
|
||||
}
|
||||
|
||||
.form-header {
|
||||
margin-bottom: 25px;
|
||||
}
|
||||
|
||||
.form-header h3 {
|
||||
color: var(--color-primary);
|
||||
margin: 0 0 10px 0;
|
||||
}
|
||||
|
||||
.form-header p {
|
||||
color: var(--color-grey-dark);
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
.deploy-form {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 20px;
|
||||
}
|
||||
|
||||
.form-row {
|
||||
display: grid;
|
||||
grid-template-columns: 1fr 1fr;
|
||||
gap: 20px;
|
||||
}
|
||||
|
||||
.form-group {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 8px;
|
||||
}
|
||||
|
||||
.form-group label {
|
||||
font-weight: 600;
|
||||
color: var(--color-grey-dark);
|
||||
}
|
||||
|
||||
.form-group input,
|
||||
.form-group select,
|
||||
.form-group textarea {
|
||||
padding: 12px;
|
||||
border: 1px solid #ddd;
|
||||
border-radius: var(--radius-sm);
|
||||
font-size: 1rem;
|
||||
}
|
||||
|
||||
.form-group textarea {
|
||||
resize: vertical;
|
||||
min-height: 80px;
|
||||
}
|
||||
|
||||
.form-help {
|
||||
font-size: 0.85rem;
|
||||
color: var(--color-grey-dark);
|
||||
font-style: italic;
|
||||
}
|
||||
|
||||
.btn-primary {
|
||||
background: var(--color-primary);
|
||||
color: white;
|
||||
border: none;
|
||||
padding: 15px 30px;
|
||||
border-radius: var(--radius-sm);
|
||||
font-size: 1.1rem;
|
||||
font-weight: 600;
|
||||
cursor: pointer;
|
||||
transition: background-color 0.3s;
|
||||
}
|
||||
|
||||
.btn-primary:hover:not(:disabled) {
|
||||
background: var(--color-primary-dark);
|
||||
}
|
||||
|
||||
.btn-primary:disabled {
|
||||
opacity: 0.6;
|
||||
cursor: not-allowed;
|
||||
}
|
||||
|
||||
.deploy-status {
|
||||
margin-top: 20px;
|
||||
padding: 15px;
|
||||
background: #e8f5e8;
|
||||
border-radius: var(--radius-sm);
|
||||
border-left: 4px solid #28a745;
|
||||
}
|
||||
|
||||
.status-message {
|
||||
margin: 0;
|
||||
font-weight: 600;
|
||||
color: #28a745;
|
||||
}
|
||||
|
||||
@media (max-width: 768px) {
|
||||
.form-row {
|
||||
grid-template-columns: 1fr;
|
||||
}
|
||||
|
||||
.dle-info {
|
||||
flex-direction: column;
|
||||
align-items: flex-start;
|
||||
gap: 5px;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
@@ -0,0 +1,218 @@
|
||||
<!--
|
||||
Copyright (c) 2024-2025 Тарабанов Александр Викторович
|
||||
All rights reserved.
|
||||
|
||||
This software is proprietary and confidential.
|
||||
Unauthorized copying, modification, or distribution is prohibited.
|
||||
|
||||
For licensing inquiries: info@hb3-accelerator.com
|
||||
Website: https://hb3-accelerator.com
|
||||
GitHub: https://github.com/HB3-ACCELERATOR
|
||||
-->
|
||||
|
||||
<template>
|
||||
<BaseLayout
|
||||
:is-authenticated="isAuthenticated"
|
||||
:identities="identities"
|
||||
:token-balances="tokenBalances"
|
||||
:is-loading-tokens="isLoadingTokens"
|
||||
@auth-action-completed="$emit('auth-action-completed')"
|
||||
>
|
||||
<div class="communication-module-deploy">
|
||||
<!-- Заголовок -->
|
||||
<div class="page-header">
|
||||
<div class="header-content">
|
||||
<h1>Деплой CommunicationModule</h1>
|
||||
<p>Коммуникации - сообщения, звонки, история общения между участниками</p>
|
||||
<p v-if="dleAddress" class="dle-address">
|
||||
<strong>DLE:</strong> {{ dleAddress }}
|
||||
</p>
|
||||
</div>
|
||||
<button class="close-btn" @click="router.push('/management/modules')">×</button>
|
||||
</div>
|
||||
|
||||
<!-- Информация о модуле -->
|
||||
<div class="module-info">
|
||||
<div class="info-card">
|
||||
<h3>💬 CommunicationModule</h3>
|
||||
<div class="info-grid">
|
||||
<div class="info-item">
|
||||
<strong>Назначение:</strong> Коммуникации между участниками DLE
|
||||
</div>
|
||||
<div class="info-item">
|
||||
<strong>Функции:</strong> Сообщения, аудио/видео звонки, история общения
|
||||
</div>
|
||||
<div class="info-item">
|
||||
<strong>Безопасность:</strong> Кворум для коммуникационных операций
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Форма деплоя будет добавлена позже -->
|
||||
<div class="deploy-form-placeholder">
|
||||
<div class="placeholder-content">
|
||||
<h3>🚧 Форма деплоя в разработке</h3>
|
||||
<p>Здесь будет форма для деплоя CommunicationModule</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
</div>
|
||||
</BaseLayout>
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import { defineProps, defineEmits, ref, onMounted } from 'vue';
|
||||
import { useRouter, useRoute } from 'vue-router';
|
||||
import BaseLayout from '../../../components/BaseLayout.vue';
|
||||
|
||||
// Определяем props
|
||||
const props = defineProps({
|
||||
isAuthenticated: { type: Boolean, default: false },
|
||||
identities: { type: Array, default: () => [] },
|
||||
tokenBalances: { type: Object, default: () => ({}) },
|
||||
isLoadingTokens: { type: Boolean, default: false }
|
||||
});
|
||||
|
||||
// Определяем emits
|
||||
const emit = defineEmits(['auth-action-completed']);
|
||||
|
||||
const router = useRouter();
|
||||
const route = useRoute();
|
||||
|
||||
// Состояние
|
||||
const isLoading = ref(false);
|
||||
const dleAddress = ref(route.query.address || null);
|
||||
|
||||
// Инициализация
|
||||
onMounted(() => {
|
||||
console.log('[CommunicationModuleDeployView] Страница загружена');
|
||||
});
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
.communication-module-deploy {
|
||||
padding: 20px;
|
||||
background-color: var(--color-white);
|
||||
border-radius: var(--radius-lg);
|
||||
box-shadow: 0 2px 10px rgba(0, 0, 0, 0.1);
|
||||
margin-top: 20px;
|
||||
margin-bottom: 20px;
|
||||
}
|
||||
|
||||
.page-header {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: center;
|
||||
margin-bottom: 30px;
|
||||
padding-bottom: 20px;
|
||||
border-bottom: 2px solid #f0f0f0;
|
||||
}
|
||||
|
||||
.page-header h1 {
|
||||
color: var(--color-primary);
|
||||
font-size: 2rem;
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
.page-header p {
|
||||
margin: 10px 0 0 0;
|
||||
color: #666;
|
||||
}
|
||||
|
||||
.dle-address {
|
||||
margin-top: 10px !important;
|
||||
font-family: monospace;
|
||||
background: #f8f9fa;
|
||||
padding: 8px 12px;
|
||||
border-radius: var(--radius-sm);
|
||||
border: 1px solid #dee2e6;
|
||||
}
|
||||
|
||||
.close-btn {
|
||||
background: none;
|
||||
border: none;
|
||||
font-size: 1.5rem;
|
||||
cursor: pointer;
|
||||
color: #666;
|
||||
padding: 0;
|
||||
width: 30px;
|
||||
height: 30px;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
border-radius: 50%;
|
||||
transition: all 0.2s;
|
||||
}
|
||||
|
||||
.close-btn:hover {
|
||||
background: #f0f0f0;
|
||||
color: #333;
|
||||
}
|
||||
|
||||
/* Информация о модуле */
|
||||
.module-info {
|
||||
margin-bottom: 30px;
|
||||
}
|
||||
|
||||
.info-card {
|
||||
background: #f8f9fa;
|
||||
border-radius: var(--radius-md);
|
||||
padding: 20px;
|
||||
border: 1px solid #e9ecef;
|
||||
}
|
||||
|
||||
.info-card h3 {
|
||||
margin: 0 0 15px 0;
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
.info-grid {
|
||||
display: grid;
|
||||
grid-template-columns: repeat(auto-fit, minmax(250px, 1fr));
|
||||
gap: 15px;
|
||||
}
|
||||
|
||||
.info-item {
|
||||
padding: 10px;
|
||||
background: white;
|
||||
border-radius: var(--radius-sm);
|
||||
border: 1px solid #dee2e6;
|
||||
}
|
||||
|
||||
.info-item strong {
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
/* Плейсхолдер для формы */
|
||||
.deploy-form-placeholder {
|
||||
background: #f8f9fa;
|
||||
border-radius: var(--radius-md);
|
||||
padding: 40px;
|
||||
text-align: center;
|
||||
border: 2px dashed #dee2e6;
|
||||
}
|
||||
|
||||
.placeholder-content h3 {
|
||||
color: var(--color-primary);
|
||||
margin-bottom: 10px;
|
||||
}
|
||||
|
||||
.placeholder-content p {
|
||||
color: #666;
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
/* Адаптивность */
|
||||
@media (max-width: 768px) {
|
||||
.info-grid {
|
||||
grid-template-columns: 1fr;
|
||||
}
|
||||
|
||||
.page-header {
|
||||
flex-direction: column;
|
||||
align-items: flex-start;
|
||||
gap: 15px;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
@@ -0,0 +1,663 @@
|
||||
<!--
|
||||
Copyright (c) 2024-2025 Тарабанов Александр Викторович
|
||||
All rights reserved.
|
||||
|
||||
This software is proprietary and confidential.
|
||||
Unauthorized copying, modification, or distribution is prohibited.
|
||||
|
||||
For licensing inquiries: info@hb3-accelerator.com
|
||||
Website: https://hb3-accelerator.com
|
||||
GitHub: https://github.com/HB3-ACCELERATOR
|
||||
-->
|
||||
|
||||
<template>
|
||||
<BaseLayout
|
||||
:is-authenticated="isAuthenticated"
|
||||
:identities="identities"
|
||||
:token-balances="tokenBalances"
|
||||
:is-loading-tokens="isLoadingTokens"
|
||||
@auth-action-completed="$emit('auth-action-completed')"
|
||||
>
|
||||
<div class="module-deploy-page">
|
||||
<!-- Заголовок -->
|
||||
<div class="page-header">
|
||||
<div class="header-content">
|
||||
<h1>🏛️ Деплой InheritanceModule</h1>
|
||||
<p>Модуль наследования токенов DLE - защита активов и автоматическая передача наследникам</p>
|
||||
<div v-if="selectedDle" class="dle-info">
|
||||
<span class="dle-name">{{ selectedDle.name }} ({{ selectedDle.symbol }})</span>
|
||||
<span class="dle-address">{{ selectedDle.dleAddress }}</span>
|
||||
</div>
|
||||
</div>
|
||||
<button class="close-btn" @click="router.push(`/management/modules?address=${route.query.address}`)">×</button>
|
||||
</div>
|
||||
|
||||
<!-- Описание модуля -->
|
||||
<div class="module-description">
|
||||
<div class="description-card">
|
||||
<h3>📋 Описание InheritanceModule</h3>
|
||||
<div class="description-content">
|
||||
<p><strong>InheritanceModule</strong> - это модуль для автоматической передачи токенов DLE наследникам в случае смерти или недееспособности токенхолдера.</p>
|
||||
|
||||
<h4>🔧 Основная функциональность:</h4>
|
||||
<ul>
|
||||
<li><strong>Назначение наследников:</strong> Токенхолдеры могут указать один или несколько наследников</li>
|
||||
<li><strong>Распределение долей:</strong> Настройка процентного распределения токенов между наследниками</li>
|
||||
<li><strong>Условия активации:</strong> Настройка условий для передачи токенов (смерть, недееспособность)</li>
|
||||
<li><strong>Временные ограничения:</strong> Установка минимального периода владения токенами</li>
|
||||
<li><strong>Множественные наследники:</strong> Поддержка сложных схем наследования</li>
|
||||
<li><strong>Отзыв и изменение:</strong> Возможность изменения наследников в любое время</li>
|
||||
</ul>
|
||||
|
||||
<h4>🏛️ Юридические аспекты:</h4>
|
||||
<ul>
|
||||
<li><strong>Соответствие законам:</strong> Интеграция с юридическими системами наследования</li>
|
||||
<li><strong>Документооборот:</strong> Автоматическое создание юридических документов</li>
|
||||
<li><strong>Подтверждение смерти:</strong> Интеграция с государственными реестрами</li>
|
||||
<li><strong>Споры и оспаривание:</strong> Механизмы разрешения споров о наследстве</li>
|
||||
<li><strong>Налоговые обязательства:</strong> Автоматический расчет налогов на наследство</li>
|
||||
</ul>
|
||||
|
||||
<h4>🔐 Безопасность и контроль:</h4>
|
||||
<ul>
|
||||
<li>Все изменения наследников требуют подтверждения через governance</li>
|
||||
<li>Криптографическая защита данных о наследниках</li>
|
||||
<li>Аудит всех операций наследования</li>
|
||||
<li>Возможность экстренной блокировки в случае споров</li>
|
||||
<li>Интеграция с системой идентификации для подтверждения личности</li>
|
||||
</ul>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Архитектура модуля -->
|
||||
<div class="module-architecture">
|
||||
<div class="architecture-card">
|
||||
<h3>🏗️ Архитектура InheritanceModule</h3>
|
||||
<div class="architecture-content">
|
||||
<div class="architecture-diagram">
|
||||
<div class="diagram-row">
|
||||
<div class="diagram-item tokenholder">
|
||||
<h5>👤 Токенхолдер</h5>
|
||||
<ul>
|
||||
<li>Назначает наследников</li>
|
||||
<li>Устанавливает доли</li>
|
||||
<li>Управляет условиями</li>
|
||||
<li>Может отозвать</li>
|
||||
</ul>
|
||||
</div>
|
||||
<div class="diagram-arrow">→</div>
|
||||
<div class="diagram-item inheritance">
|
||||
<h5>🏛️ InheritanceModule</h5>
|
||||
<ul>
|
||||
<li>Хранит данные наследников</li>
|
||||
<li>Проверяет условия</li>
|
||||
<li>Выполняет передачу</li>
|
||||
<li>Ведет аудит</li>
|
||||
</ul>
|
||||
</div>
|
||||
<div class="diagram-arrow">→</div>
|
||||
<div class="diagram-item heirs">
|
||||
<h5>👥 Наследники</h5>
|
||||
<ul>
|
||||
<li>Получают токены</li>
|
||||
<li>Подтверждают получение</li>
|
||||
<li>Управляют наследством</li>
|
||||
<li>Планируют налоги</li>
|
||||
</ul>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Типы наследования -->
|
||||
<div class="inheritance-types">
|
||||
<div class="types-card">
|
||||
<h3>📊 Типы наследования</h3>
|
||||
<div class="types-grid">
|
||||
<div class="type-item">
|
||||
<h4>👨👩👧👦 Семейное наследование</h4>
|
||||
<p>Передача токенов членам семьи согласно традиционным схемам</p>
|
||||
<ul>
|
||||
<li>Супруг/супруга (50%)</li>
|
||||
<li>Дети (равные доли)</li>
|
||||
<li>Родители (при отсутствии детей)</li>
|
||||
<li>Братья/сестры (при отсутствии родителей)</li>
|
||||
</ul>
|
||||
</div>
|
||||
|
||||
<div class="type-item">
|
||||
<h4>🏢 Корпоративное наследование</h4>
|
||||
<p>Передача токенов в рамках бизнес-структур и организаций</p>
|
||||
<ul>
|
||||
<li>Партнеры по бизнесу</li>
|
||||
<li>Ключевые сотрудники</li>
|
||||
<li>Дочерние компании</li>
|
||||
<li>Благотворительные фонды</li>
|
||||
</ul>
|
||||
</div>
|
||||
|
||||
<div class="type-item">
|
||||
<h4>🎯 Целевое наследование</h4>
|
||||
<p>Передача токенов для достижения конкретных целей</p>
|
||||
<ul>
|
||||
<li>Образовательные учреждения</li>
|
||||
<li>Исследовательские проекты</li>
|
||||
<li>Экологические инициативы</li>
|
||||
<li>Социальные программы</li>
|
||||
</ul>
|
||||
</div>
|
||||
|
||||
<div class="type-item">
|
||||
<h4>⏰ Условное наследование</h4>
|
||||
<p>Передача токенов при выполнении определенных условий</p>
|
||||
<ul>
|
||||
<li>Достижение определенного возраста</li>
|
||||
<li>Завершение образования</li>
|
||||
<li>Создание семьи</li>
|
||||
<li>Достижение карьерных целей</li>
|
||||
</ul>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Примеры использования -->
|
||||
<div class="usage-examples">
|
||||
<div class="examples-card">
|
||||
<h3>💡 Примеры использования</h3>
|
||||
<div class="examples-content">
|
||||
<div class="example-item">
|
||||
<h4>👨👩👧👦 Семейное планирование</h4>
|
||||
<div class="example-code">
|
||||
<pre><code>// Назначение наследников для семьи
|
||||
function setFamilyInheritance() {
|
||||
setHeir(spouse, 50); // Супруг 50%
|
||||
setHeir(son, 25); // Сын 25%
|
||||
setHeir(daughter, 25); // Дочь 25%
|
||||
setActivationCondition("death");
|
||||
}</code></pre>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="example-item">
|
||||
<h4>🏢 Бизнес-преемственность</h4>
|
||||
<div class="example-code">
|
||||
<pre><code>// Передача бизнеса партнеру
|
||||
function setBusinessInheritance() {
|
||||
setHeir(businessPartner, 100); // Партнер 100%
|
||||
setActivationCondition("death");
|
||||
setTimeLock(365 days); // Минимум 1 год владения
|
||||
}</code></pre>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="example-item">
|
||||
<h4>🎯 Благотворительное наследование</h4>
|
||||
<div class="example-code">
|
||||
<pre><code>// Передача в благотворительный фонд
|
||||
function setCharityInheritance() {
|
||||
setHeir(environmentalFund, 70); // Экологический фонд 70%
|
||||
setHeir(educationFund, 30); // Образовательный фонд 30%
|
||||
setActivationCondition("death");
|
||||
}</code></pre>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Юридические аспекты -->
|
||||
<div class="legal-aspects">
|
||||
<div class="legal-card">
|
||||
<h3>⚖️ Юридические аспекты</h3>
|
||||
<div class="legal-content">
|
||||
<div class="legal-section">
|
||||
<h4>📜 Соответствие законодательству</h4>
|
||||
<ul>
|
||||
<li><strong>Гражданский кодекс:</strong> Соответствие нормам наследования</li>
|
||||
<li><strong>Налоговый кодекс:</strong> Правильный расчет налогов на наследство</li>
|
||||
<li><strong>Семейный кодекс:</strong> Учет семейных обязательств</li>
|
||||
<li><strong>Международное право:</strong> Наследование в разных юрисдикциях</li>
|
||||
</ul>
|
||||
</div>
|
||||
|
||||
<div class="legal-section">
|
||||
<h4>🔍 Процедура подтверждения</h4>
|
||||
<ul>
|
||||
<li><strong>Свидетельство о смерти:</strong> Официальное подтверждение</li>
|
||||
<li><strong>Медицинское заключение:</strong> При недееспособности</li>
|
||||
<li><strong>Судебное решение:</strong> При спорах о наследстве</li>
|
||||
<li><strong>Нотариальное заверение:</strong> Документов о наследниках</li>
|
||||
</ul>
|
||||
</div>
|
||||
|
||||
<div class="legal-section">
|
||||
<h4>💰 Налоговые обязательства</h4>
|
||||
<ul>
|
||||
<li><strong>Налог на наследство:</strong> Автоматический расчет</li>
|
||||
<li><strong>НДФЛ:</strong> При получении токенов</li>
|
||||
<li><strong>Отчетность:</strong> Автоматическая подача деклараций</li>
|
||||
<li><strong>Льготы:</strong> Учет налоговых льгот для наследников</li>
|
||||
</ul>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Статус разработки -->
|
||||
<div class="development-status">
|
||||
<div class="status-card">
|
||||
<h3>🚧 Статус разработки</h3>
|
||||
<div class="status-content">
|
||||
<p><strong>InheritanceModule находится в стадии планирования.</strong></p>
|
||||
<p>Модуль будет включать:</p>
|
||||
<ul>
|
||||
<li>✅ Систему назначения наследников</li>
|
||||
<li>✅ Управление долями и условиями</li>
|
||||
<li>✅ Интеграцию с юридическими системами</li>
|
||||
<li>✅ Автоматическую передачу токенов</li>
|
||||
<li>✅ Налоговые расчеты</li>
|
||||
<li>✅ Аудит и мониторинг</li>
|
||||
<li>✅ Разрешение споров</li>
|
||||
</ul>
|
||||
<p><em>Модуль будет доступен в следующих обновлениях DLE.</em></p>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</BaseLayout>
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import { ref, computed, onMounted } from 'vue';
|
||||
import { useRoute, useRouter } from 'vue-router';
|
||||
import BaseLayout from '../../../components/BaseLayout.vue';
|
||||
import api from '../../../api/axios';
|
||||
|
||||
// Props
|
||||
const props = defineProps({
|
||||
isAuthenticated: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
},
|
||||
identities: {
|
||||
type: Array,
|
||||
default: () => []
|
||||
},
|
||||
tokenBalances: {
|
||||
type: Object,
|
||||
default: () => ({})
|
||||
},
|
||||
isLoadingTokens: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
}
|
||||
});
|
||||
|
||||
// Emits
|
||||
const emit = defineEmits(['auth-action-completed']);
|
||||
|
||||
// Router
|
||||
const route = useRoute();
|
||||
const router = useRouter();
|
||||
|
||||
// Состояние
|
||||
const selectedDle = ref(null);
|
||||
const isLoadingDle = ref(false);
|
||||
|
||||
// Получаем адрес DLE из URL
|
||||
const dleAddress = computed(() => route.query.address);
|
||||
|
||||
// Загрузка данных DLE
|
||||
const loadDleData = async () => {
|
||||
if (!dleAddress.value) return;
|
||||
|
||||
try {
|
||||
isLoadingDle.value = true;
|
||||
const response = await api.post('/blockchain/read-dle-info', {
|
||||
dleAddress: dleAddress.value
|
||||
});
|
||||
|
||||
if (response.data.success) {
|
||||
selectedDle.value = response.data.data;
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('Ошибка загрузки данных DLE:', error);
|
||||
} finally {
|
||||
isLoadingDle.value = false;
|
||||
}
|
||||
};
|
||||
|
||||
// Загружаем данные при монтировании
|
||||
onMounted(() => {
|
||||
loadDleData();
|
||||
});
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
.module-deploy-page {
|
||||
padding: 20px;
|
||||
max-width: 1200px;
|
||||
margin: 0 auto;
|
||||
}
|
||||
|
||||
.page-header {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: flex-start;
|
||||
margin-bottom: 30px;
|
||||
padding-bottom: 20px;
|
||||
border-bottom: 2px solid #f0f0f0;
|
||||
}
|
||||
|
||||
.header-content h1 {
|
||||
color: var(--color-primary);
|
||||
font-size: 2.5rem;
|
||||
margin: 0 0 10px 0;
|
||||
}
|
||||
|
||||
.header-content p {
|
||||
color: var(--color-grey-dark);
|
||||
font-size: 1.1rem;
|
||||
margin: 0 0 15px 0;
|
||||
}
|
||||
|
||||
.dle-info {
|
||||
display: flex;
|
||||
gap: 15px;
|
||||
align-items: center;
|
||||
}
|
||||
|
||||
.dle-name {
|
||||
font-weight: 600;
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
.dle-address {
|
||||
font-family: monospace;
|
||||
color: var(--color-grey-dark);
|
||||
font-size: 0.9rem;
|
||||
}
|
||||
|
||||
.close-btn {
|
||||
background: none;
|
||||
border: none;
|
||||
font-size: 2rem;
|
||||
color: var(--color-grey-dark);
|
||||
cursor: pointer;
|
||||
padding: 5px;
|
||||
}
|
||||
|
||||
.close-btn:hover {
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
.module-description,
|
||||
.module-architecture,
|
||||
.inheritance-types,
|
||||
.usage-examples,
|
||||
.legal-aspects,
|
||||
.development-status {
|
||||
margin-bottom: 30px;
|
||||
}
|
||||
|
||||
.description-card,
|
||||
.architecture-card,
|
||||
.types-card,
|
||||
.examples-card,
|
||||
.legal-card,
|
||||
.status-card {
|
||||
background: #f8f9fa;
|
||||
padding: 25px;
|
||||
border-radius: var(--radius-lg);
|
||||
border: 1px solid #e9ecef;
|
||||
}
|
||||
|
||||
.description-card h3,
|
||||
.architecture-card h3,
|
||||
.types-card h3,
|
||||
.examples-card h3,
|
||||
.legal-card h3,
|
||||
.status-card h3 {
|
||||
color: var(--color-primary);
|
||||
margin: 0 0 20px 0;
|
||||
}
|
||||
|
||||
.description-content h4 {
|
||||
color: var(--color-grey-dark);
|
||||
margin: 20px 0 10px 0;
|
||||
}
|
||||
|
||||
.description-content ul {
|
||||
margin: 10px 0;
|
||||
padding-left: 20px;
|
||||
}
|
||||
|
||||
.description-content li {
|
||||
margin: 5px 0;
|
||||
line-height: 1.5;
|
||||
}
|
||||
|
||||
/* Архитектурная диаграмма */
|
||||
.architecture-diagram {
|
||||
margin: 20px 0;
|
||||
}
|
||||
|
||||
.diagram-row {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: space-between;
|
||||
gap: 20px;
|
||||
margin-bottom: 20px;
|
||||
}
|
||||
|
||||
.diagram-item {
|
||||
flex: 1;
|
||||
padding: 20px;
|
||||
border-radius: var(--radius-sm);
|
||||
text-align: center;
|
||||
min-height: 150px;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
justify-content: center;
|
||||
}
|
||||
|
||||
.diagram-item.tokenholder {
|
||||
background: #e8f5e8;
|
||||
border: 2px solid #4caf50;
|
||||
}
|
||||
|
||||
.diagram-item.inheritance {
|
||||
background: #fff3e0;
|
||||
border: 2px solid #ff9800;
|
||||
}
|
||||
|
||||
.diagram-item.heirs {
|
||||
background: #f3e5f5;
|
||||
border: 2px solid #9c27b0;
|
||||
}
|
||||
|
||||
.diagram-item h5 {
|
||||
margin: 0 0 15px 0;
|
||||
font-weight: 600;
|
||||
}
|
||||
|
||||
.diagram-item ul {
|
||||
margin: 0;
|
||||
padding: 0;
|
||||
list-style: none;
|
||||
font-size: 0.9rem;
|
||||
}
|
||||
|
||||
.diagram-item li {
|
||||
margin: 5px 0;
|
||||
}
|
||||
|
||||
.diagram-arrow {
|
||||
font-size: 2rem;
|
||||
color: var(--color-primary);
|
||||
font-weight: bold;
|
||||
}
|
||||
|
||||
/* Типы наследования */
|
||||
.types-grid {
|
||||
display: grid;
|
||||
grid-template-columns: repeat(auto-fit, minmax(280px, 1fr));
|
||||
gap: 20px;
|
||||
margin-top: 20px;
|
||||
}
|
||||
|
||||
.type-item {
|
||||
background: white;
|
||||
padding: 20px;
|
||||
border-radius: var(--radius-sm);
|
||||
border: 1px solid #e9ecef;
|
||||
}
|
||||
|
||||
.type-item h4 {
|
||||
color: var(--color-primary);
|
||||
margin: 0 0 10px 0;
|
||||
}
|
||||
|
||||
.type-item p {
|
||||
color: var(--color-grey-dark);
|
||||
margin: 0 0 15px 0;
|
||||
font-size: 0.9rem;
|
||||
}
|
||||
|
||||
.type-item ul {
|
||||
margin: 0;
|
||||
padding-left: 20px;
|
||||
font-size: 0.9rem;
|
||||
}
|
||||
|
||||
.type-item li {
|
||||
margin: 5px 0;
|
||||
color: var(--color-grey-dark);
|
||||
}
|
||||
|
||||
/* Примеры использования */
|
||||
.examples-content {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 20px;
|
||||
}
|
||||
|
||||
.example-item {
|
||||
background: white;
|
||||
padding: 20px;
|
||||
border-radius: var(--radius-sm);
|
||||
border: 1px solid #e9ecef;
|
||||
}
|
||||
|
||||
.example-item h4 {
|
||||
color: var(--color-primary);
|
||||
margin: 0 0 15px 0;
|
||||
}
|
||||
|
||||
.example-code {
|
||||
background: #f8f9fa;
|
||||
border: 1px solid #e9ecef;
|
||||
border-radius: var(--radius-sm);
|
||||
padding: 15px;
|
||||
overflow-x: auto;
|
||||
}
|
||||
|
||||
.example-code pre {
|
||||
margin: 0;
|
||||
font-family: 'Courier New', monospace;
|
||||
font-size: 0.9rem;
|
||||
color: #333;
|
||||
}
|
||||
|
||||
.example-code code {
|
||||
background: none;
|
||||
padding: 0;
|
||||
}
|
||||
|
||||
/* Юридические аспекты */
|
||||
.legal-content {
|
||||
display: grid;
|
||||
grid-template-columns: repeat(auto-fit, minmax(300px, 1fr));
|
||||
gap: 20px;
|
||||
}
|
||||
|
||||
.legal-section {
|
||||
background: white;
|
||||
padding: 20px;
|
||||
border-radius: var(--radius-sm);
|
||||
border: 1px solid #e9ecef;
|
||||
}
|
||||
|
||||
.legal-section h4 {
|
||||
color: var(--color-primary);
|
||||
margin: 0 0 15px 0;
|
||||
}
|
||||
|
||||
.legal-section ul {
|
||||
margin: 0;
|
||||
padding-left: 20px;
|
||||
}
|
||||
|
||||
.legal-section li {
|
||||
margin: 8px 0;
|
||||
line-height: 1.5;
|
||||
color: var(--color-grey-dark);
|
||||
}
|
||||
|
||||
/* Статус разработки */
|
||||
.status-content {
|
||||
background: white;
|
||||
padding: 20px;
|
||||
border-radius: var(--radius-sm);
|
||||
border: 1px solid #e9ecef;
|
||||
}
|
||||
|
||||
.status-content p {
|
||||
margin: 0 0 15px 0;
|
||||
line-height: 1.6;
|
||||
}
|
||||
|
||||
.status-content ul {
|
||||
margin: 15px 0;
|
||||
padding-left: 20px;
|
||||
}
|
||||
|
||||
.status-content li {
|
||||
margin: 5px 0;
|
||||
color: var(--color-grey-dark);
|
||||
}
|
||||
|
||||
.status-content em {
|
||||
color: var(--color-primary);
|
||||
font-style: italic;
|
||||
}
|
||||
|
||||
@media (max-width: 768px) {
|
||||
.diagram-row {
|
||||
flex-direction: column;
|
||||
gap: 15px;
|
||||
}
|
||||
|
||||
.diagram-arrow {
|
||||
transform: rotate(90deg);
|
||||
}
|
||||
|
||||
.types-grid {
|
||||
grid-template-columns: 1fr;
|
||||
}
|
||||
|
||||
.legal-content {
|
||||
grid-template-columns: 1fr;
|
||||
}
|
||||
|
||||
.dle-info {
|
||||
flex-direction: column;
|
||||
align-items: flex-start;
|
||||
gap: 5px;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
485
frontend/src/views/smartcontracts/modules/MintModuleDeploy.vue
Normal file
485
frontend/src/views/smartcontracts/modules/MintModuleDeploy.vue
Normal file
@@ -0,0 +1,485 @@
|
||||
<!--
|
||||
Copyright (c) 2024-2025 Тарабанов Александр Викторович
|
||||
All rights reserved.
|
||||
|
||||
This software is proprietary and confidential.
|
||||
Unauthorized copying, modification, or distribution is prohibited.
|
||||
|
||||
For licensing inquiries: info@hb3-accelerator.com
|
||||
Website: https://hb3-accelerator.com
|
||||
GitHub: https://github.com/HB3-ACCELERATOR
|
||||
-->
|
||||
|
||||
<template>
|
||||
<BaseLayout
|
||||
:is-authenticated="isAuthenticated"
|
||||
:identities="identities"
|
||||
:token-balances="tokenBalances"
|
||||
:is-loading-tokens="isLoadingTokens"
|
||||
@auth-action-completed="$emit('auth-action-completed')"
|
||||
>
|
||||
<div class="module-deploy-page">
|
||||
<!-- Заголовок -->
|
||||
<div class="page-header">
|
||||
<div class="header-content">
|
||||
<h1>🚀 Деплой MintModule</h1>
|
||||
<p>Модуль для выпуска новых токенов DLE через governance</p>
|
||||
<div v-if="selectedDle" class="dle-info">
|
||||
<span class="dle-name">{{ selectedDle.name }} ({{ selectedDle.symbol }})</span>
|
||||
<span class="dle-address">{{ selectedDle.dleAddress }}</span>
|
||||
</div>
|
||||
</div>
|
||||
<button class="close-btn" @click="router.push(`/management/modules?address=${route.query.address}`)">×</button>
|
||||
</div>
|
||||
|
||||
<!-- Описание модуля -->
|
||||
<div class="module-description">
|
||||
<div class="description-card">
|
||||
<h3>📋 Описание MintModule</h3>
|
||||
<div class="description-content">
|
||||
<p><strong>MintModule</strong> - это модуль для управления выпуском новых токенов DLE через систему governance.</p>
|
||||
|
||||
<h4>🔧 Функциональность:</h4>
|
||||
<ul>
|
||||
<li><strong>Выпуск токенов:</strong> Создание дополнительных токенов DLE</li>
|
||||
<li><strong>Governance:</strong> Все операции требуют голосования и кворума</li>
|
||||
<li><strong>Безопасность:</strong> Контролируемый выпуск через коллективные решения</li>
|
||||
<li><strong>Прозрачность:</strong> Все операции записываются в блокчейн</li>
|
||||
</ul>
|
||||
|
||||
<h4>⚠️ Важные особенности:</h4>
|
||||
<ul>
|
||||
<li>Выпуск токенов возможен только через предложения и голосование</li>
|
||||
<li>Новые токены могут быть распределены между участниками или добавлены в казну DLE</li>
|
||||
<li>Все операции требуют достижения кворума</li>
|
||||
<li>История всех выпусков сохраняется в блокчейне</li>
|
||||
</ul>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Форма деплоя -->
|
||||
<div class="deploy-form-section">
|
||||
<div class="form-header">
|
||||
<h3>⚙️ Настройки деплоя</h3>
|
||||
<p>Настройте параметры для деплоя MintModule</p>
|
||||
</div>
|
||||
|
||||
<form @submit.prevent="deployModule" class="deploy-form">
|
||||
<div class="form-row">
|
||||
<div class="form-group">
|
||||
<label for="moduleName">Название модуля:</label>
|
||||
<input
|
||||
id="moduleName"
|
||||
v-model="deployData.moduleName"
|
||||
type="text"
|
||||
placeholder="MintModule"
|
||||
required
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="moduleVersion">Версия модуля:</label>
|
||||
<input
|
||||
id="moduleVersion"
|
||||
v-model="deployData.moduleVersion"
|
||||
type="text"
|
||||
placeholder="1.0.0"
|
||||
required
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="moduleDescription">Описание модуля:</label>
|
||||
<textarea
|
||||
id="moduleDescription"
|
||||
v-model="deployData.moduleDescription"
|
||||
placeholder="Модуль для выпуска новых токенов DLE через governance..."
|
||||
rows="3"
|
||||
required
|
||||
></textarea>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="maxMintPerProposal">Максимальный выпуск за одно предложение:</label>
|
||||
<input
|
||||
id="maxMintPerProposal"
|
||||
v-model="deployData.maxMintPerProposal"
|
||||
type="number"
|
||||
min="1"
|
||||
step="1"
|
||||
placeholder="1000000"
|
||||
required
|
||||
/>
|
||||
<small class="form-help">Максимальное количество токенов, которое можно выпустить за одно предложение</small>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="mintCooldown">Кулдаун между выпусками (часы):</label>
|
||||
<input
|
||||
id="mintCooldown"
|
||||
v-model="deployData.mintCooldown"
|
||||
type="number"
|
||||
min="0"
|
||||
step="1"
|
||||
placeholder="24"
|
||||
required
|
||||
/>
|
||||
<small class="form-help">Минимальное время между успешными выпусками токенов</small>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="deployDescription">Описание предложения для деплоя:</label>
|
||||
<textarea
|
||||
id="deployDescription"
|
||||
v-model="deployData.deployDescription"
|
||||
placeholder="Предложение о деплое MintModule для управления выпуском токенов DLE..."
|
||||
rows="3"
|
||||
required
|
||||
></textarea>
|
||||
</div>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="votingDuration">Длительность голосования (часы):</label>
|
||||
<input
|
||||
id="votingDuration"
|
||||
v-model="deployData.votingDuration"
|
||||
type="number"
|
||||
min="1"
|
||||
max="168"
|
||||
placeholder="24"
|
||||
required
|
||||
/>
|
||||
<small class="form-help">Время для голосования по предложению деплоя (1-168 часов)</small>
|
||||
</div>
|
||||
|
||||
<button type="submit" class="btn-primary" :disabled="isDeploying">
|
||||
{{ isDeploying ? 'Деплой...' : 'Деплой MintModule' }}
|
||||
</button>
|
||||
|
||||
<!-- Статус деплоя -->
|
||||
<div v-if="deployStatus" class="deploy-status">
|
||||
<p class="status-message">{{ deployStatus }}</p>
|
||||
</div>
|
||||
</form>
|
||||
</div>
|
||||
</div>
|
||||
</BaseLayout>
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import { ref, computed, onMounted } from 'vue';
|
||||
import { useRoute, useRouter } from 'vue-router';
|
||||
import BaseLayout from '../../../components/BaseLayout.vue';
|
||||
import api from '../../../api/axios';
|
||||
|
||||
// Props
|
||||
const props = defineProps({
|
||||
isAuthenticated: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
},
|
||||
identities: {
|
||||
type: Array,
|
||||
default: () => []
|
||||
},
|
||||
tokenBalances: {
|
||||
type: Object,
|
||||
default: () => ({})
|
||||
},
|
||||
isLoadingTokens: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
}
|
||||
});
|
||||
|
||||
// Emits
|
||||
const emit = defineEmits(['auth-action-completed']);
|
||||
|
||||
// Router
|
||||
const route = useRoute();
|
||||
const router = useRouter();
|
||||
|
||||
// Состояние
|
||||
const isDeploying = ref(false);
|
||||
const deployStatus = ref('');
|
||||
const selectedDle = ref(null);
|
||||
const isLoadingDle = ref(false);
|
||||
|
||||
// Данные для деплоя
|
||||
const deployData = ref({
|
||||
moduleName: 'MintModule',
|
||||
moduleVersion: '1.0.0',
|
||||
moduleDescription: 'Модуль для выпуска новых токенов DLE через governance',
|
||||
maxMintPerProposal: 1000000,
|
||||
mintCooldown: 24,
|
||||
deployDescription: 'Предложение о деплое MintModule для управления выпуском токенов DLE',
|
||||
votingDuration: 24
|
||||
});
|
||||
|
||||
// Получаем адрес DLE из URL
|
||||
const dleAddress = computed(() => route.query.address);
|
||||
|
||||
// Загрузка данных DLE
|
||||
const loadDleData = async () => {
|
||||
if (!dleAddress.value) return;
|
||||
|
||||
try {
|
||||
isLoadingDle.value = true;
|
||||
const response = await api.post('/blockchain/read-dle-info', {
|
||||
dleAddress: dleAddress.value
|
||||
});
|
||||
|
||||
if (response.data.success) {
|
||||
selectedDle.value = response.data.data;
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('Ошибка загрузки данных DLE:', error);
|
||||
} finally {
|
||||
isLoadingDle.value = false;
|
||||
}
|
||||
};
|
||||
|
||||
// Функция деплоя модуля
|
||||
const deployModule = async () => {
|
||||
if (isDeploying.value) return;
|
||||
|
||||
try {
|
||||
isDeploying.value = true;
|
||||
deployStatus.value = 'Подготовка к деплою...';
|
||||
|
||||
// Здесь будет логика деплоя модуля
|
||||
console.log('Деплой MintModule:', deployData.value);
|
||||
|
||||
// Временная заглушка
|
||||
await new Promise(resolve => setTimeout(resolve, 2000));
|
||||
|
||||
deployStatus.value = 'Модуль успешно развернут!';
|
||||
|
||||
// Очищаем статус через 3 секунды
|
||||
setTimeout(() => {
|
||||
deployStatus.value = '';
|
||||
}, 3000);
|
||||
|
||||
alert('MintModule успешно развернут!');
|
||||
|
||||
} catch (error) {
|
||||
console.error('Ошибка деплоя модуля:', error);
|
||||
deployStatus.value = 'Ошибка деплоя модуля';
|
||||
|
||||
setTimeout(() => {
|
||||
deployStatus.value = '';
|
||||
}, 3000);
|
||||
|
||||
alert('Ошибка при деплое модуля');
|
||||
} finally {
|
||||
isDeploying.value = false;
|
||||
}
|
||||
};
|
||||
|
||||
// Загружаем данные при монтировании
|
||||
onMounted(() => {
|
||||
loadDleData();
|
||||
});
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
.module-deploy-page {
|
||||
padding: 20px;
|
||||
max-width: 1200px;
|
||||
margin: 0 auto;
|
||||
}
|
||||
|
||||
.page-header {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: flex-start;
|
||||
margin-bottom: 30px;
|
||||
padding-bottom: 20px;
|
||||
border-bottom: 2px solid #f0f0f0;
|
||||
}
|
||||
|
||||
.header-content h1 {
|
||||
color: var(--color-primary);
|
||||
font-size: 2.5rem;
|
||||
margin: 0 0 10px 0;
|
||||
}
|
||||
|
||||
.header-content p {
|
||||
color: var(--color-grey-dark);
|
||||
font-size: 1.1rem;
|
||||
margin: 0 0 15px 0;
|
||||
}
|
||||
|
||||
.dle-info {
|
||||
display: flex;
|
||||
gap: 15px;
|
||||
align-items: center;
|
||||
}
|
||||
|
||||
.dle-name {
|
||||
font-weight: 600;
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
.dle-address {
|
||||
font-family: monospace;
|
||||
color: var(--color-grey-dark);
|
||||
font-size: 0.9rem;
|
||||
}
|
||||
|
||||
.close-btn {
|
||||
background: none;
|
||||
border: none;
|
||||
font-size: 2rem;
|
||||
color: var(--color-grey-dark);
|
||||
cursor: pointer;
|
||||
padding: 5px;
|
||||
}
|
||||
|
||||
.close-btn:hover {
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
.module-description {
|
||||
margin-bottom: 30px;
|
||||
}
|
||||
|
||||
.description-card {
|
||||
background: #f8f9fa;
|
||||
padding: 25px;
|
||||
border-radius: var(--radius-lg);
|
||||
border: 1px solid #e9ecef;
|
||||
}
|
||||
|
||||
.description-card h3 {
|
||||
color: var(--color-primary);
|
||||
margin: 0 0 20px 0;
|
||||
}
|
||||
|
||||
.description-content h4 {
|
||||
color: var(--color-grey-dark);
|
||||
margin: 20px 0 10px 0;
|
||||
}
|
||||
|
||||
.description-content ul {
|
||||
margin: 10px 0;
|
||||
padding-left: 20px;
|
||||
}
|
||||
|
||||
.description-content li {
|
||||
margin: 5px 0;
|
||||
line-height: 1.5;
|
||||
}
|
||||
|
||||
.deploy-form-section {
|
||||
background: white;
|
||||
padding: 30px;
|
||||
border-radius: var(--radius-lg);
|
||||
box-shadow: 0 2px 10px rgba(0, 0, 0, 0.1);
|
||||
}
|
||||
|
||||
.form-header {
|
||||
margin-bottom: 25px;
|
||||
}
|
||||
|
||||
.form-header h3 {
|
||||
color: var(--color-primary);
|
||||
margin: 0 0 10px 0;
|
||||
}
|
||||
|
||||
.form-header p {
|
||||
color: var(--color-grey-dark);
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
.deploy-form {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 20px;
|
||||
}
|
||||
|
||||
.form-row {
|
||||
display: grid;
|
||||
grid-template-columns: 1fr 1fr;
|
||||
gap: 20px;
|
||||
}
|
||||
|
||||
.form-group {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 8px;
|
||||
}
|
||||
|
||||
.form-group label {
|
||||
font-weight: 600;
|
||||
color: var(--color-grey-dark);
|
||||
}
|
||||
|
||||
.form-group input,
|
||||
.form-group textarea {
|
||||
padding: 12px;
|
||||
border: 1px solid #ddd;
|
||||
border-radius: var(--radius-sm);
|
||||
font-size: 1rem;
|
||||
}
|
||||
|
||||
.form-group textarea {
|
||||
resize: vertical;
|
||||
min-height: 80px;
|
||||
}
|
||||
|
||||
.form-help {
|
||||
font-size: 0.85rem;
|
||||
color: var(--color-grey-dark);
|
||||
font-style: italic;
|
||||
}
|
||||
|
||||
.btn-primary {
|
||||
background: var(--color-primary);
|
||||
color: white;
|
||||
border: none;
|
||||
padding: 15px 30px;
|
||||
border-radius: var(--radius-sm);
|
||||
font-size: 1.1rem;
|
||||
font-weight: 600;
|
||||
cursor: pointer;
|
||||
transition: background-color 0.3s;
|
||||
}
|
||||
|
||||
.btn-primary:hover:not(:disabled) {
|
||||
background: var(--color-primary-dark);
|
||||
}
|
||||
|
||||
.btn-primary:disabled {
|
||||
opacity: 0.6;
|
||||
cursor: not-allowed;
|
||||
}
|
||||
|
||||
.deploy-status {
|
||||
margin-top: 20px;
|
||||
padding: 15px;
|
||||
background: #e8f5e8;
|
||||
border-radius: var(--radius-sm);
|
||||
border-left: 4px solid #28a745;
|
||||
}
|
||||
|
||||
.status-message {
|
||||
margin: 0;
|
||||
font-weight: 600;
|
||||
color: #28a745;
|
||||
}
|
||||
|
||||
@media (max-width: 768px) {
|
||||
.form-row {
|
||||
grid-template-columns: 1fr;
|
||||
}
|
||||
|
||||
.dle-info {
|
||||
flex-direction: column;
|
||||
align-items: flex-start;
|
||||
gap: 5px;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
@@ -0,0 +1,205 @@
|
||||
<!--
|
||||
Copyright (c) 2024-2025 Тарабанов Александр Викторович
|
||||
All rights reserved.
|
||||
|
||||
This software is proprietary and confidential.
|
||||
Unauthorized copying, modification, or distribution is prohibited.
|
||||
|
||||
For licensing inquiries: info@hb3-accelerator.com
|
||||
Website: https://hb3-accelerator.com
|
||||
GitHub: https://github.com/HB3-ACCELERATOR
|
||||
-->
|
||||
|
||||
<template>
|
||||
<BaseLayout
|
||||
:is-authenticated="isAuthenticated"
|
||||
:identities="identities"
|
||||
:token-balances="tokenBalances"
|
||||
:is-loading-tokens="isLoadingTokens"
|
||||
@auth-action-completed="$emit('auth-action-completed')"
|
||||
>
|
||||
<div class="module-deploy-form">
|
||||
<!-- Заголовок -->
|
||||
<div class="page-header">
|
||||
<div class="header-content">
|
||||
<h1>Деплой пользовательского модуля</h1>
|
||||
<p>Создание и деплой кастомного модуля для DLE</p>
|
||||
</div>
|
||||
<button class="close-btn" @click="router.push('/management/modules')">×</button>
|
||||
</div>
|
||||
|
||||
<!-- Информация о пользовательских модулях -->
|
||||
<div class="module-info">
|
||||
<div class="info-card">
|
||||
<h3>🔧 Пользовательские модули</h3>
|
||||
<div class="info-grid">
|
||||
<div class="info-item">
|
||||
<strong>Назначение:</strong> Расширение функциональности DLE
|
||||
</div>
|
||||
<div class="info-item">
|
||||
<strong>Возможности:</strong> Любая кастомная логика, интеграции, API
|
||||
</div>
|
||||
<div class="info-item">
|
||||
<strong>Безопасность:</strong> Проверка через голосование токен-холдеров
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Форма деплоя будет добавлена позже -->
|
||||
<div class="deploy-form-placeholder">
|
||||
<div class="placeholder-content">
|
||||
<h3>🚧 Форма деплоя в разработке</h3>
|
||||
<p>Здесь будет универсальная форма для деплоя пользовательских модулей</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
</div>
|
||||
</BaseLayout>
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import { defineProps, defineEmits, ref, onMounted } from 'vue';
|
||||
import { useRouter, useRoute } from 'vue-router';
|
||||
import BaseLayout from '../../../components/BaseLayout.vue';
|
||||
|
||||
// Определяем props
|
||||
const props = defineProps({
|
||||
isAuthenticated: { type: Boolean, default: false },
|
||||
identities: { type: Array, default: () => [] },
|
||||
tokenBalances: { type: Object, default: () => ({}) },
|
||||
isLoadingTokens: { type: Boolean, default: false }
|
||||
});
|
||||
|
||||
// Определяем emits
|
||||
const emit = defineEmits(['auth-action-completed']);
|
||||
|
||||
const router = useRouter();
|
||||
const route = useRoute();
|
||||
|
||||
// Состояние
|
||||
const isLoading = ref(false);
|
||||
|
||||
// Инициализация
|
||||
onMounted(() => {
|
||||
console.log('[ModuleDeployFormView] Страница загружена');
|
||||
});
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
.module-deploy-form {
|
||||
padding: 20px;
|
||||
background-color: var(--color-white);
|
||||
border-radius: var(--radius-lg);
|
||||
box-shadow: 0 2px 10px rgba(0, 0, 0, 0.1);
|
||||
margin-top: 20px;
|
||||
margin-bottom: 20px;
|
||||
}
|
||||
|
||||
.page-header {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: center;
|
||||
margin-bottom: 30px;
|
||||
padding-bottom: 20px;
|
||||
border-bottom: 2px solid #f0f0f0;
|
||||
}
|
||||
|
||||
.page-header h1 {
|
||||
color: var(--color-primary);
|
||||
font-size: 2rem;
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
.page-header p {
|
||||
margin: 10px 0 0 0;
|
||||
color: #666;
|
||||
}
|
||||
|
||||
.close-btn {
|
||||
background: none;
|
||||
border: none;
|
||||
font-size: 1.5rem;
|
||||
cursor: pointer;
|
||||
color: #666;
|
||||
padding: 0;
|
||||
width: 30px;
|
||||
height: 30px;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
border-radius: 50%;
|
||||
transition: all 0.2s;
|
||||
}
|
||||
|
||||
.close-btn:hover {
|
||||
background: #f0f0f0;
|
||||
color: #333;
|
||||
}
|
||||
|
||||
/* Информация о модуле */
|
||||
.module-info {
|
||||
margin-bottom: 30px;
|
||||
}
|
||||
|
||||
.info-card {
|
||||
background: #f8f9fa;
|
||||
border-radius: var(--radius-md);
|
||||
padding: 20px;
|
||||
border: 1px solid #e9ecef;
|
||||
}
|
||||
|
||||
.info-card h3 {
|
||||
margin: 0 0 15px 0;
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
.info-grid {
|
||||
display: grid;
|
||||
grid-template-columns: repeat(auto-fit, minmax(250px, 1fr));
|
||||
gap: 15px;
|
||||
}
|
||||
|
||||
.info-item {
|
||||
padding: 10px;
|
||||
background: white;
|
||||
border-radius: var(--radius-sm);
|
||||
border: 1px solid #dee2e6;
|
||||
}
|
||||
|
||||
.info-item strong {
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
/* Плейсхолдер для формы */
|
||||
.deploy-form-placeholder {
|
||||
background: #f8f9fa;
|
||||
border-radius: var(--radius-md);
|
||||
padding: 40px;
|
||||
text-align: center;
|
||||
border: 2px dashed #dee2e6;
|
||||
}
|
||||
|
||||
.placeholder-content h3 {
|
||||
color: var(--color-primary);
|
||||
margin-bottom: 10px;
|
||||
}
|
||||
|
||||
.placeholder-content p {
|
||||
color: #666;
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
/* Адаптивность */
|
||||
@media (max-width: 768px) {
|
||||
.info-grid {
|
||||
grid-template-columns: 1fr;
|
||||
}
|
||||
|
||||
.page-header {
|
||||
flex-direction: column;
|
||||
align-items: flex-start;
|
||||
gap: 15px;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
584
frontend/src/views/smartcontracts/modules/OracleModuleDeploy.vue
Normal file
584
frontend/src/views/smartcontracts/modules/OracleModuleDeploy.vue
Normal file
@@ -0,0 +1,584 @@
|
||||
<!--
|
||||
Copyright (c) 2024-2025 Тарабанов Александр Викторович
|
||||
All rights reserved.
|
||||
|
||||
This software is proprietary and confidential.
|
||||
Unauthorized copying, modification, or distribution is prohibited.
|
||||
|
||||
For licensing inquiries: info@hb3-accelerator.com
|
||||
Website: https://hb3-accelerator.com
|
||||
GitHub: https://github.com/HB3-ACCELERATOR
|
||||
-->
|
||||
|
||||
<template>
|
||||
<BaseLayout
|
||||
:is-authenticated="isAuthenticated"
|
||||
:identities="identities"
|
||||
:token-balances="tokenBalances"
|
||||
:is-loading-tokens="isLoadingTokens"
|
||||
@auth-action-completed="$emit('auth-action-completed')"
|
||||
>
|
||||
<div class="module-deploy-page">
|
||||
<!-- Заголовок -->
|
||||
<div class="page-header">
|
||||
<div class="header-content">
|
||||
<h1>🔗 Деплой OracleModule</h1>
|
||||
<p>Модуль для интеграции с внешними данными и автоматизации DLE</p>
|
||||
<div v-if="selectedDle" class="dle-info">
|
||||
<span class="dle-name">{{ selectedDle.name }} ({{ selectedDle.symbol }})</span>
|
||||
<span class="dle-address">{{ selectedDle.dleAddress }}</span>
|
||||
</div>
|
||||
</div>
|
||||
<button class="close-btn" @click="router.push(`/management/modules?address=${route.query.address}`)">×</button>
|
||||
</div>
|
||||
|
||||
<!-- Описание модуля -->
|
||||
<div class="module-description">
|
||||
<div class="description-card">
|
||||
<h3>📋 Описание OracleModule</h3>
|
||||
<div class="description-content">
|
||||
<p><strong>OracleModule</strong> - это модуль для интеграции DLE с внешними источниками данных и автоматизации процессов на основе реальных событий.</p>
|
||||
|
||||
<h4>🔧 Основная функциональность:</h4>
|
||||
<ul>
|
||||
<li><strong>Интеграция с IoT:</strong> Получение данных от датчиков, производственных линий, оборудования</li>
|
||||
<li><strong>API интеграция:</strong> Подключение к внешним системам, ERP, CRM, аналитическим платформам</li>
|
||||
<li><strong>Автоматические триггеры:</strong> Создание предложений на основе внешних событий</li>
|
||||
<li><strong>Валидация данных:</strong> Проверка достоверности полученной информации</li>
|
||||
<li><strong>Множественные оракулы:</strong> Подтверждение данных от нескольких источников</li>
|
||||
</ul>
|
||||
|
||||
<h4>🏭 Примеры применения:</h4>
|
||||
<ul>
|
||||
<li><strong>Производственные токены:</strong> Автоматический выпуск токенов за завершение партий продукции</li>
|
||||
<li><strong>Качественные бонусы:</strong> Токены за высокое качество продукции на основе данных датчиков</li>
|
||||
<li><strong>Экологические токены:</strong> Вознаграждения за снижение энергопотребления и выбросов</li>
|
||||
<li><strong>Инновационные токены:</strong> Токены за внедрение новых технологий и процессов</li>
|
||||
<li><strong>Рыночные корректировки:</strong> Автоматическая адаптация стратегии на основе рыночных данных</li>
|
||||
</ul>
|
||||
|
||||
<h4>🔐 Безопасность и контроль:</h4>
|
||||
<ul>
|
||||
<li>Все оракулы должны быть авторизованы через governance</li>
|
||||
<li>Данные валидируются перед обработкой</li>
|
||||
<li>Критические решения требуют множественного подтверждения</li>
|
||||
<li>История всех оракульных данных сохраняется в блокчейне</li>
|
||||
<li>Возможность отключения оракулов в экстренных случаях</li>
|
||||
</ul>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Архитектура модуля -->
|
||||
<div class="module-architecture">
|
||||
<div class="architecture-card">
|
||||
<h3>🏗️ Архитектура OracleModule</h3>
|
||||
<div class="architecture-content">
|
||||
<div class="architecture-diagram">
|
||||
<div class="diagram-row">
|
||||
<div class="diagram-item external">
|
||||
<h5>🌐 Внешние источники</h5>
|
||||
<ul>
|
||||
<li>IoT датчики</li>
|
||||
<li>Производственные линии</li>
|
||||
<li>ERP/CRM системы</li>
|
||||
<li>Аналитические платформы</li>
|
||||
<li>Рыночные данные</li>
|
||||
</ul>
|
||||
</div>
|
||||
<div class="diagram-arrow">→</div>
|
||||
<div class="diagram-item oracle">
|
||||
<h5>🔗 OracleModule</h5>
|
||||
<ul>
|
||||
<li>Получение данных</li>
|
||||
<li>Валидация</li>
|
||||
<li>Обработка</li>
|
||||
<li>Создание предложений</li>
|
||||
</ul>
|
||||
</div>
|
||||
<div class="diagram-arrow">→</div>
|
||||
<div class="diagram-item dle">
|
||||
<h5>🏛️ DLE Governance</h5>
|
||||
<ul>
|
||||
<li>Предложения</li>
|
||||
<li>Голосование</li>
|
||||
<li>Исполнение</li>
|
||||
<li>Выпуск токенов</li>
|
||||
</ul>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Типы оракулов -->
|
||||
<div class="oracle-types">
|
||||
<div class="types-card">
|
||||
<h3>📊 Типы оракулов</h3>
|
||||
<div class="types-grid">
|
||||
<div class="type-item">
|
||||
<h4>🏭 Производственные оракулы</h4>
|
||||
<p>Данные от производственных линий, оборудования, датчиков качества</p>
|
||||
<ul>
|
||||
<li>Завершение партий продукции</li>
|
||||
<li>Показатели качества</li>
|
||||
<li>Энергопотребление</li>
|
||||
<li>Время простоя</li>
|
||||
</ul>
|
||||
</div>
|
||||
|
||||
<div class="type-item">
|
||||
<h4>📈 Рыночные оракулы</h4>
|
||||
<p>Рыночные данные, цены, спрос, конкуренция</p>
|
||||
<ul>
|
||||
<li>Цены на сырье</li>
|
||||
<li>Спрос на продукцию</li>
|
||||
<li>Конкурентные данные</li>
|
||||
<li>Экономические индикаторы</li>
|
||||
</ul>
|
||||
</div>
|
||||
|
||||
<div class="type-item">
|
||||
<h4>🌱 Экологические оракулы</h4>
|
||||
<p>Данные об экологическом воздействии и устойчивости</p>
|
||||
<ul>
|
||||
<li>Выбросы CO2</li>
|
||||
<li>Потребление энергии</li>
|
||||
<li>Переработка отходов</li>
|
||||
<li>Использование возобновляемых ресурсов</li>
|
||||
</ul>
|
||||
</div>
|
||||
|
||||
<div class="type-item">
|
||||
<h4>💼 Бизнес-оракулы</h4>
|
||||
<p>Бизнес-метрики, продажи, удовлетворенность клиентов</p>
|
||||
<ul>
|
||||
<li>Объемы продаж</li>
|
||||
<li>Удовлетворенность клиентов</li>
|
||||
<li>Эффективность процессов</li>
|
||||
<li>Финансовые показатели</li>
|
||||
</ul>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Примеры использования -->
|
||||
<div class="usage-examples">
|
||||
<div class="examples-card">
|
||||
<h3>💡 Примеры использования</h3>
|
||||
<div class="examples-content">
|
||||
<div class="example-item">
|
||||
<h4>🏭 Автоматические производственные токены</h4>
|
||||
<div class="example-code">
|
||||
<pre><code>// При завершении партии продукции
|
||||
function onBatchCompleted(uint256 quantity, uint256 quality) {
|
||||
uint256 tokens = quantity * quality * 10;
|
||||
createMintProposal(tokens, "Production reward");
|
||||
}</code></pre>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="example-item">
|
||||
<h4>🌱 Экологические бонусы</h4>
|
||||
<div class="example-code">
|
||||
<pre><code>// При снижении энергопотребления
|
||||
function onEnergyReduction(uint256 savedEnergy) {
|
||||
uint256 tokens = savedEnergy * 5;
|
||||
createMintProposal(tokens, "Energy efficiency bonus");
|
||||
}</code></pre>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="example-item">
|
||||
<h4>📈 Рыночная адаптация</h4>
|
||||
<div class="example-code">
|
||||
<pre><code>// При изменении рыночных условий
|
||||
function onMarketChange(uint256 priceChange) {
|
||||
if (priceChange > 10) {
|
||||
createStrategyProposal("Increase production");
|
||||
}
|
||||
}</code></pre>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Статус разработки -->
|
||||
<div class="development-status">
|
||||
<div class="status-card">
|
||||
<h3>🚧 Статус разработки</h3>
|
||||
<div class="status-content">
|
||||
<p><strong>OracleModule находится в стадии планирования.</strong></p>
|
||||
<p>Модуль будет включать:</p>
|
||||
<ul>
|
||||
<li>✅ Систему авторизации оракулов</li>
|
||||
<li>✅ Валидацию и обработку данных</li>
|
||||
<li>✅ Интеграцию с IoT и API</li>
|
||||
<li>✅ Автоматические триггеры</li>
|
||||
<li>✅ Множественное подтверждение данных</li>
|
||||
<li>✅ Аудит и мониторинг</li>
|
||||
</ul>
|
||||
<p><em>Модуль будет доступен в следующих обновлениях DLE.</em></p>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</BaseLayout>
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import { ref, computed, onMounted } from 'vue';
|
||||
import { useRoute, useRouter } from 'vue-router';
|
||||
import BaseLayout from '../../../components/BaseLayout.vue';
|
||||
import api from '../../../api/axios';
|
||||
|
||||
// Props
|
||||
const props = defineProps({
|
||||
isAuthenticated: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
},
|
||||
identities: {
|
||||
type: Array,
|
||||
default: () => []
|
||||
},
|
||||
tokenBalances: {
|
||||
type: Object,
|
||||
default: () => ({})
|
||||
},
|
||||
isLoadingTokens: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
}
|
||||
});
|
||||
|
||||
// Emits
|
||||
const emit = defineEmits(['auth-action-completed']);
|
||||
|
||||
// Router
|
||||
const route = useRoute();
|
||||
const router = useRouter();
|
||||
|
||||
// Состояние
|
||||
const selectedDle = ref(null);
|
||||
const isLoadingDle = ref(false);
|
||||
|
||||
// Получаем адрес DLE из URL
|
||||
const dleAddress = computed(() => route.query.address);
|
||||
|
||||
// Загрузка данных DLE
|
||||
const loadDleData = async () => {
|
||||
if (!dleAddress.value) return;
|
||||
|
||||
try {
|
||||
isLoadingDle.value = true;
|
||||
const response = await api.post('/blockchain/read-dle-info', {
|
||||
dleAddress: dleAddress.value
|
||||
});
|
||||
|
||||
if (response.data.success) {
|
||||
selectedDle.value = response.data.data;
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('Ошибка загрузки данных DLE:', error);
|
||||
} finally {
|
||||
isLoadingDle.value = false;
|
||||
}
|
||||
};
|
||||
|
||||
// Загружаем данные при монтировании
|
||||
onMounted(() => {
|
||||
loadDleData();
|
||||
});
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
.module-deploy-page {
|
||||
padding: 20px;
|
||||
max-width: 1200px;
|
||||
margin: 0 auto;
|
||||
}
|
||||
|
||||
.page-header {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: flex-start;
|
||||
margin-bottom: 30px;
|
||||
padding-bottom: 20px;
|
||||
border-bottom: 2px solid #f0f0f0;
|
||||
}
|
||||
|
||||
.header-content h1 {
|
||||
color: var(--color-primary);
|
||||
font-size: 2.5rem;
|
||||
margin: 0 0 10px 0;
|
||||
}
|
||||
|
||||
.header-content p {
|
||||
color: var(--color-grey-dark);
|
||||
font-size: 1.1rem;
|
||||
margin: 0 0 15px 0;
|
||||
}
|
||||
|
||||
.dle-info {
|
||||
display: flex;
|
||||
gap: 15px;
|
||||
align-items: center;
|
||||
}
|
||||
|
||||
.dle-name {
|
||||
font-weight: 600;
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
.dle-address {
|
||||
font-family: monospace;
|
||||
color: var(--color-grey-dark);
|
||||
font-size: 0.9rem;
|
||||
}
|
||||
|
||||
.close-btn {
|
||||
background: none;
|
||||
border: none;
|
||||
font-size: 2rem;
|
||||
color: var(--color-grey-dark);
|
||||
cursor: pointer;
|
||||
padding: 5px;
|
||||
}
|
||||
|
||||
.close-btn:hover {
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
.module-description,
|
||||
.module-architecture,
|
||||
.oracle-types,
|
||||
.usage-examples,
|
||||
.development-status {
|
||||
margin-bottom: 30px;
|
||||
}
|
||||
|
||||
.description-card,
|
||||
.architecture-card,
|
||||
.types-card,
|
||||
.examples-card,
|
||||
.status-card {
|
||||
background: #f8f9fa;
|
||||
padding: 25px;
|
||||
border-radius: var(--radius-lg);
|
||||
border: 1px solid #e9ecef;
|
||||
}
|
||||
|
||||
.description-card h3,
|
||||
.architecture-card h3,
|
||||
.types-card h3,
|
||||
.examples-card h3,
|
||||
.status-card h3 {
|
||||
color: var(--color-primary);
|
||||
margin: 0 0 20px 0;
|
||||
}
|
||||
|
||||
.description-content h4 {
|
||||
color: var(--color-grey-dark);
|
||||
margin: 20px 0 10px 0;
|
||||
}
|
||||
|
||||
.description-content ul {
|
||||
margin: 10px 0;
|
||||
padding-left: 20px;
|
||||
}
|
||||
|
||||
.description-content li {
|
||||
margin: 5px 0;
|
||||
line-height: 1.5;
|
||||
}
|
||||
|
||||
/* Архитектурная диаграмма */
|
||||
.architecture-diagram {
|
||||
margin: 20px 0;
|
||||
}
|
||||
|
||||
.diagram-row {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: space-between;
|
||||
gap: 20px;
|
||||
margin-bottom: 20px;
|
||||
}
|
||||
|
||||
.diagram-item {
|
||||
flex: 1;
|
||||
padding: 20px;
|
||||
border-radius: var(--radius-sm);
|
||||
text-align: center;
|
||||
min-height: 150px;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
justify-content: center;
|
||||
}
|
||||
|
||||
.diagram-item.external {
|
||||
background: #e3f2fd;
|
||||
border: 2px solid #2196f3;
|
||||
}
|
||||
|
||||
.diagram-item.oracle {
|
||||
background: #f3e5f5;
|
||||
border: 2px solid #9c27b0;
|
||||
}
|
||||
|
||||
.diagram-item.dle {
|
||||
background: #e8f5e8;
|
||||
border: 2px solid #4caf50;
|
||||
}
|
||||
|
||||
.diagram-item h5 {
|
||||
margin: 0 0 15px 0;
|
||||
font-weight: 600;
|
||||
}
|
||||
|
||||
.diagram-item ul {
|
||||
margin: 0;
|
||||
padding: 0;
|
||||
list-style: none;
|
||||
font-size: 0.9rem;
|
||||
}
|
||||
|
||||
.diagram-item li {
|
||||
margin: 5px 0;
|
||||
}
|
||||
|
||||
.diagram-arrow {
|
||||
font-size: 2rem;
|
||||
color: var(--color-primary);
|
||||
font-weight: bold;
|
||||
}
|
||||
|
||||
/* Типы оракулов */
|
||||
.types-grid {
|
||||
display: grid;
|
||||
grid-template-columns: repeat(auto-fit, minmax(280px, 1fr));
|
||||
gap: 20px;
|
||||
margin-top: 20px;
|
||||
}
|
||||
|
||||
.type-item {
|
||||
background: white;
|
||||
padding: 20px;
|
||||
border-radius: var(--radius-sm);
|
||||
border: 1px solid #e9ecef;
|
||||
}
|
||||
|
||||
.type-item h4 {
|
||||
color: var(--color-primary);
|
||||
margin: 0 0 10px 0;
|
||||
}
|
||||
|
||||
.type-item p {
|
||||
color: var(--color-grey-dark);
|
||||
margin: 0 0 15px 0;
|
||||
font-size: 0.9rem;
|
||||
}
|
||||
|
||||
.type-item ul {
|
||||
margin: 0;
|
||||
padding-left: 20px;
|
||||
font-size: 0.9rem;
|
||||
}
|
||||
|
||||
.type-item li {
|
||||
margin: 5px 0;
|
||||
color: var(--color-grey-dark);
|
||||
}
|
||||
|
||||
/* Примеры использования */
|
||||
.examples-content {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 20px;
|
||||
}
|
||||
|
||||
.example-item {
|
||||
background: white;
|
||||
padding: 20px;
|
||||
border-radius: var(--radius-sm);
|
||||
border: 1px solid #e9ecef;
|
||||
}
|
||||
|
||||
.example-item h4 {
|
||||
color: var(--color-primary);
|
||||
margin: 0 0 15px 0;
|
||||
}
|
||||
|
||||
.example-code {
|
||||
background: #f8f9fa;
|
||||
border: 1px solid #e9ecef;
|
||||
border-radius: var(--radius-sm);
|
||||
padding: 15px;
|
||||
overflow-x: auto;
|
||||
}
|
||||
|
||||
.example-code pre {
|
||||
margin: 0;
|
||||
font-family: 'Courier New', monospace;
|
||||
font-size: 0.9rem;
|
||||
color: #333;
|
||||
}
|
||||
|
||||
.example-code code {
|
||||
background: none;
|
||||
padding: 0;
|
||||
}
|
||||
|
||||
/* Статус разработки */
|
||||
.status-content {
|
||||
background: white;
|
||||
padding: 20px;
|
||||
border-radius: var(--radius-sm);
|
||||
border: 1px solid #e9ecef;
|
||||
}
|
||||
|
||||
.status-content p {
|
||||
margin: 0 0 15px 0;
|
||||
line-height: 1.6;
|
||||
}
|
||||
|
||||
.status-content ul {
|
||||
margin: 15px 0;
|
||||
padding-left: 20px;
|
||||
}
|
||||
|
||||
.status-content li {
|
||||
margin: 5px 0;
|
||||
color: var(--color-grey-dark);
|
||||
}
|
||||
|
||||
.status-content em {
|
||||
color: var(--color-primary);
|
||||
font-style: italic;
|
||||
}
|
||||
|
||||
@media (max-width: 768px) {
|
||||
.diagram-row {
|
||||
flex-direction: column;
|
||||
gap: 15px;
|
||||
}
|
||||
|
||||
.diagram-arrow {
|
||||
transform: rotate(90deg);
|
||||
}
|
||||
|
||||
.types-grid {
|
||||
grid-template-columns: 1fr;
|
||||
}
|
||||
|
||||
.dle-info {
|
||||
flex-direction: column;
|
||||
align-items: flex-start;
|
||||
gap: 5px;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
@@ -0,0 +1,218 @@
|
||||
<!--
|
||||
Copyright (c) 2024-2025 Тарабанов Александр Викторович
|
||||
All rights reserved.
|
||||
|
||||
This software is proprietary and confidential.
|
||||
Unauthorized copying, modification, or distribution is prohibited.
|
||||
|
||||
For licensing inquiries: info@hb3-accelerator.com
|
||||
Website: https://hb3-accelerator.com
|
||||
GitHub: https://github.com/HB3-ACCELERATOR
|
||||
-->
|
||||
|
||||
<template>
|
||||
<BaseLayout
|
||||
:is-authenticated="isAuthenticated"
|
||||
:identities="identities"
|
||||
:token-balances="tokenBalances"
|
||||
:is-loading-tokens="isLoadingTokens"
|
||||
@auth-action-completed="$emit('auth-action-completed')"
|
||||
>
|
||||
<div class="timelock-module-deploy">
|
||||
<!-- Заголовок -->
|
||||
<div class="page-header">
|
||||
<div class="header-content">
|
||||
<h1>Деплой TimelockModule</h1>
|
||||
<p>Задержки исполнения - безопасность критических операций через таймлоки</p>
|
||||
<p v-if="dleAddress" class="dle-address">
|
||||
<strong>DLE:</strong> {{ dleAddress }}
|
||||
</p>
|
||||
</div>
|
||||
<button class="close-btn" @click="router.push('/management/modules')">×</button>
|
||||
</div>
|
||||
|
||||
<!-- Информация о модуле -->
|
||||
<div class="module-info">
|
||||
<div class="info-card">
|
||||
<h3>⏰ TimelockModule</h3>
|
||||
<div class="info-grid">
|
||||
<div class="info-item">
|
||||
<strong>Назначение:</strong> Безопасность критических операций
|
||||
</div>
|
||||
<div class="info-item">
|
||||
<strong>Функции:</strong> Настраиваемые таймлоки, отмена предложений, аудит
|
||||
</div>
|
||||
<div class="info-item">
|
||||
<strong>Безопасность:</strong> Задержки исполнения для защиты от атак
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Форма деплоя будет добавлена позже -->
|
||||
<div class="deploy-form-placeholder">
|
||||
<div class="placeholder-content">
|
||||
<h3>🚧 Форма деплоя в разработке</h3>
|
||||
<p>Здесь будет форма для деплоя TimelockModule</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
</div>
|
||||
</BaseLayout>
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import { defineProps, defineEmits, ref, onMounted } from 'vue';
|
||||
import { useRouter, useRoute } from 'vue-router';
|
||||
import BaseLayout from '../../../components/BaseLayout.vue';
|
||||
|
||||
// Определяем props
|
||||
const props = defineProps({
|
||||
isAuthenticated: { type: Boolean, default: false },
|
||||
identities: { type: Array, default: () => [] },
|
||||
tokenBalances: { type: Object, default: () => ({}) },
|
||||
isLoadingTokens: { type: Boolean, default: false }
|
||||
});
|
||||
|
||||
// Определяем emits
|
||||
const emit = defineEmits(['auth-action-completed']);
|
||||
|
||||
const router = useRouter();
|
||||
const route = useRoute();
|
||||
|
||||
// Состояние
|
||||
const isLoading = ref(false);
|
||||
const dleAddress = ref(route.query.address || null);
|
||||
|
||||
// Инициализация
|
||||
onMounted(() => {
|
||||
console.log('[TimelockModuleDeployView] Страница загружена');
|
||||
});
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
.timelock-module-deploy {
|
||||
padding: 20px;
|
||||
background-color: var(--color-white);
|
||||
border-radius: var(--radius-lg);
|
||||
box-shadow: 0 2px 10px rgba(0, 0, 0, 0.1);
|
||||
margin-top: 20px;
|
||||
margin-bottom: 20px;
|
||||
}
|
||||
|
||||
.page-header {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: center;
|
||||
margin-bottom: 30px;
|
||||
padding-bottom: 20px;
|
||||
border-bottom: 2px solid #f0f0f0;
|
||||
}
|
||||
|
||||
.page-header h1 {
|
||||
color: var(--color-primary);
|
||||
font-size: 2rem;
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
.page-header p {
|
||||
margin: 10px 0 0 0;
|
||||
color: #666;
|
||||
}
|
||||
|
||||
.dle-address {
|
||||
margin-top: 10px !important;
|
||||
font-family: monospace;
|
||||
background: #f8f9fa;
|
||||
padding: 8px 12px;
|
||||
border-radius: var(--radius-sm);
|
||||
border: 1px solid #dee2e6;
|
||||
}
|
||||
|
||||
.close-btn {
|
||||
background: none;
|
||||
border: none;
|
||||
font-size: 1.5rem;
|
||||
cursor: pointer;
|
||||
color: #666;
|
||||
padding: 0;
|
||||
width: 30px;
|
||||
height: 30px;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
border-radius: 50%;
|
||||
transition: all 0.2s;
|
||||
}
|
||||
|
||||
.close-btn:hover {
|
||||
background: #f0f0f0;
|
||||
color: #333;
|
||||
}
|
||||
|
||||
/* Информация о модуле */
|
||||
.module-info {
|
||||
margin-bottom: 30px;
|
||||
}
|
||||
|
||||
.info-card {
|
||||
background: #f8f9fa;
|
||||
border-radius: var(--radius-md);
|
||||
padding: 20px;
|
||||
border: 1px solid #e9ecef;
|
||||
}
|
||||
|
||||
.info-card h3 {
|
||||
margin: 0 0 15px 0;
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
.info-grid {
|
||||
display: grid;
|
||||
grid-template-columns: repeat(auto-fit, minmax(250px, 1fr));
|
||||
gap: 15px;
|
||||
}
|
||||
|
||||
.info-item {
|
||||
padding: 10px;
|
||||
background: white;
|
||||
border-radius: var(--radius-sm);
|
||||
border: 1px solid #dee2e6;
|
||||
}
|
||||
|
||||
.info-item strong {
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
/* Плейсхолдер для формы */
|
||||
.deploy-form-placeholder {
|
||||
background: #f8f9fa;
|
||||
border-radius: var(--radius-md);
|
||||
padding: 40px;
|
||||
text-align: center;
|
||||
border: 2px dashed #dee2e6;
|
||||
}
|
||||
|
||||
.placeholder-content h3 {
|
||||
color: var(--color-primary);
|
||||
margin-bottom: 10px;
|
||||
}
|
||||
|
||||
.placeholder-content p {
|
||||
color: #666;
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
/* Адаптивность */
|
||||
@media (max-width: 768px) {
|
||||
.info-grid {
|
||||
grid-template-columns: 1fr;
|
||||
}
|
||||
|
||||
.page-header {
|
||||
flex-direction: column;
|
||||
align-items: flex-start;
|
||||
gap: 15px;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
@@ -0,0 +1,218 @@
|
||||
<!--
|
||||
Copyright (c) 2024-2025 Тарабанов Александр Викторович
|
||||
All rights reserved.
|
||||
|
||||
This software is proprietary and confidential.
|
||||
Unauthorized copying, modification, or distribution is prohibited.
|
||||
|
||||
For licensing inquiries: info@hb3-accelerator.com
|
||||
Website: https://hb3-accelerator.com
|
||||
GitHub: https://github.com/HB3-ACCELERATOR
|
||||
-->
|
||||
|
||||
<template>
|
||||
<BaseLayout
|
||||
:is-authenticated="isAuthenticated"
|
||||
:identities="identities"
|
||||
:token-balances="tokenBalances"
|
||||
:is-loading-tokens="isLoadingTokens"
|
||||
@auth-action-completed="$emit('auth-action-completed')"
|
||||
>
|
||||
<div class="treasury-module-deploy">
|
||||
<!-- Заголовок -->
|
||||
<div class="page-header">
|
||||
<div class="header-content">
|
||||
<h1>Деплой TreasuryModule</h1>
|
||||
<p>Казначейство DLE - управление финансами, депозиты, выводы, дивиденды</p>
|
||||
<p v-if="dleAddress" class="dle-address">
|
||||
<strong>DLE:</strong> {{ dleAddress }}
|
||||
</p>
|
||||
</div>
|
||||
<button class="close-btn" @click="router.push('/management/modules')">×</button>
|
||||
</div>
|
||||
|
||||
<!-- Информация о модуле -->
|
||||
<div class="module-info">
|
||||
<div class="info-card">
|
||||
<h3>💰 TreasuryModule</h3>
|
||||
<div class="info-grid">
|
||||
<div class="info-item">
|
||||
<strong>Назначение:</strong> Управление финансами DLE
|
||||
</div>
|
||||
<div class="info-item">
|
||||
<strong>Функции:</strong> Депозиты, выводы, дивиденды, бюджетирование
|
||||
</div>
|
||||
<div class="info-item">
|
||||
<strong>Безопасность:</strong> Все операции через голосование
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Форма деплоя будет добавлена позже -->
|
||||
<div class="deploy-form-placeholder">
|
||||
<div class="placeholder-content">
|
||||
<h3>🚧 Форма деплоя в разработке</h3>
|
||||
<p>Здесь будет форма для деплоя TreasuryModule</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
</div>
|
||||
</BaseLayout>
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import { defineProps, defineEmits, ref, onMounted } from 'vue';
|
||||
import { useRouter, useRoute } from 'vue-router';
|
||||
import BaseLayout from '../../../components/BaseLayout.vue';
|
||||
|
||||
// Определяем props
|
||||
const props = defineProps({
|
||||
isAuthenticated: { type: Boolean, default: false },
|
||||
identities: { type: Array, default: () => [] },
|
||||
tokenBalances: { type: Object, default: () => ({}) },
|
||||
isLoadingTokens: { type: Boolean, default: false }
|
||||
});
|
||||
|
||||
// Определяем emits
|
||||
const emit = defineEmits(['auth-action-completed']);
|
||||
|
||||
const router = useRouter();
|
||||
const route = useRoute();
|
||||
|
||||
// Состояние
|
||||
const isLoading = ref(false);
|
||||
const dleAddress = ref(route.query.address || null);
|
||||
|
||||
// Инициализация
|
||||
onMounted(() => {
|
||||
console.log('[TreasuryModuleDeployView] Страница загружена');
|
||||
});
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
.treasury-module-deploy {
|
||||
padding: 20px;
|
||||
background-color: var(--color-white);
|
||||
border-radius: var(--radius-lg);
|
||||
box-shadow: 0 2px 10px rgba(0, 0, 0, 0.1);
|
||||
margin-top: 20px;
|
||||
margin-bottom: 20px;
|
||||
}
|
||||
|
||||
.page-header {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: center;
|
||||
margin-bottom: 30px;
|
||||
padding-bottom: 20px;
|
||||
border-bottom: 2px solid #f0f0f0;
|
||||
}
|
||||
|
||||
.page-header h1 {
|
||||
color: var(--color-primary);
|
||||
font-size: 2rem;
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
.page-header p {
|
||||
margin: 10px 0 0 0;
|
||||
color: #666;
|
||||
}
|
||||
|
||||
.dle-address {
|
||||
margin-top: 10px !important;
|
||||
font-family: monospace;
|
||||
background: #f8f9fa;
|
||||
padding: 8px 12px;
|
||||
border-radius: var(--radius-sm);
|
||||
border: 1px solid #dee2e6;
|
||||
}
|
||||
|
||||
.close-btn {
|
||||
background: none;
|
||||
border: none;
|
||||
font-size: 1.5rem;
|
||||
cursor: pointer;
|
||||
color: #666;
|
||||
padding: 0;
|
||||
width: 30px;
|
||||
height: 30px;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
border-radius: 50%;
|
||||
transition: all 0.2s;
|
||||
}
|
||||
|
||||
.close-btn:hover {
|
||||
background: #f0f0f0;
|
||||
color: #333;
|
||||
}
|
||||
|
||||
/* Информация о модуле */
|
||||
.module-info {
|
||||
margin-bottom: 30px;
|
||||
}
|
||||
|
||||
.info-card {
|
||||
background: #f8f9fa;
|
||||
border-radius: var(--radius-md);
|
||||
padding: 20px;
|
||||
border: 1px solid #e9ecef;
|
||||
}
|
||||
|
||||
.info-card h3 {
|
||||
margin: 0 0 15px 0;
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
.info-grid {
|
||||
display: grid;
|
||||
grid-template-columns: repeat(auto-fit, minmax(250px, 1fr));
|
||||
gap: 15px;
|
||||
}
|
||||
|
||||
.info-item {
|
||||
padding: 10px;
|
||||
background: white;
|
||||
border-radius: var(--radius-sm);
|
||||
border: 1px solid #dee2e6;
|
||||
}
|
||||
|
||||
.info-item strong {
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
/* Плейсхолдер для формы */
|
||||
.deploy-form-placeholder {
|
||||
background: #f8f9fa;
|
||||
border-radius: var(--radius-md);
|
||||
padding: 40px;
|
||||
text-align: center;
|
||||
border: 2px dashed #dee2e6;
|
||||
}
|
||||
|
||||
.placeholder-content h3 {
|
||||
color: var(--color-primary);
|
||||
margin-bottom: 10px;
|
||||
}
|
||||
|
||||
.placeholder-content p {
|
||||
color: #666;
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
/* Адаптивность */
|
||||
@media (max-width: 768px) {
|
||||
.info-grid {
|
||||
grid-template-columns: 1fr;
|
||||
}
|
||||
|
||||
.page-header {
|
||||
flex-direction: column;
|
||||
align-items: flex-start;
|
||||
gap: 15px;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
Reference in New Issue
Block a user