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

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

View File

@@ -232,6 +232,46 @@ const routes = [
name: 'management-modules',
component: () => import('../views/smartcontracts/ModulesView.vue')
},
{
path: '/management/modules/deploy/treasury',
name: 'module-deploy-treasury',
component: () => import('../views/smartcontracts/modules/TreasuryModuleDeployView.vue')
},
{
path: '/management/modules/deploy/timelock',
name: 'module-deploy-timelock',
component: () => import('../views/smartcontracts/modules/TimelockModuleDeployView.vue')
},
{
path: '/management/modules/deploy/communication',
name: 'module-deploy-communication',
component: () => import('../views/smartcontracts/modules/CommunicationModuleDeployView.vue')
},
{
path: '/management/modules/deploy/application',
name: 'module-deploy-application',
component: () => import('../views/smartcontracts/modules/ApplicationModuleDeployView.vue')
},
{
path: '/management/modules/deploy/mint',
name: 'module-deploy-mint',
component: () => import('../views/smartcontracts/modules/MintModuleDeploy.vue')
},
{
path: '/management/modules/deploy/burn',
name: 'module-deploy-burn',
component: () => import('../views/smartcontracts/modules/BurnModuleDeploy.vue')
},
{
path: '/management/modules/deploy/oracle',
name: 'module-deploy-oracle',
component: () => import('../views/smartcontracts/modules/OracleModuleDeploy.vue')
},
{
path: '/management/modules/deploy/custom',
name: 'module-deploy-custom',
component: () => import('../views/smartcontracts/modules/ModuleDeployFormView.vue')
},
// {
// path: '/management/multisig',
// name: 'management-multisig',

View File

@@ -751,4 +751,79 @@ export async function loadDeactivationProposals(dleAddress) {
console.error('Ошибка загрузки предложений деактивации:', error);
return [];
}
}
/**
* Создать предложение о переводе токенов через governance
* @param {string} dleAddress - Адрес DLE контракта
* @param {Object} transferData - Данные перевода
* @param {string} transferData.recipient - Адрес получателя
* @param {number} transferData.amount - Количество токенов
* @param {string} transferData.description - Описание предложения
* @param {number} transferData.duration - Длительность голосования в секундах
* @param {number} transferData.governanceChainId - ID сети для голосования
* @param {Array<number>} transferData.targetChains - Целевые сети для исполнения
* @returns {Promise<Object>} - Результат создания предложения
*/
export async function createTransferTokensProposal(dleAddress, transferData) {
try {
// Проверяем наличие браузерного кошелька
if (!window.ethereum) {
throw new Error('Браузерный кошелек не установлен');
}
// Запрашиваем подключение к кошельку
const accounts = await window.ethereum.request({ method: 'eth_requestAccounts' });
const provider = new ethers.BrowserProvider(window.ethereum);
const signer = await provider.getSigner();
// ABI для создания предложения
const dleAbi = [
"function createProposal(string memory _description, uint256 _duration, bytes memory _operation, uint256 _governanceChainId, uint256[] memory _targetChains, uint256 _timelockDelay) external returns (uint256)"
];
const dle = new ethers.Contract(dleAddress, dleAbi, signer);
// Кодируем операцию перевода токенов
const transferFunctionSelector = ethers.id("_transferTokens(address,uint256)");
const transferDataEncoded = ethers.AbiCoder.defaultAbiCoder().encode(
["address", "uint256"],
[transferData.recipient, ethers.parseUnits(transferData.amount.toString(), 18)]
);
// Объединяем селектор и данные
const operation = ethers.concat([transferFunctionSelector, transferDataEncoded]);
console.log('Создание предложения о переводе токенов:', {
recipient: transferData.recipient,
amount: transferData.amount,
description: transferData.description,
operation: operation
});
// Создаем предложение
const tx = await dle.createProposal(
transferData.description,
transferData.duration,
operation,
transferData.governanceChainId,
transferData.targetChains || [],
0 // timelockDelay
);
// Ждем подтверждения транзакции
const receipt = await tx.wait();
console.log('Предложение о переводе токенов создано, tx hash:', tx.hash);
return {
proposalId: receipt.logs[0]?.topics[1] || '0', // Извлекаем ID предложения из события
txHash: tx.hash,
blockNumber: receipt.blockNumber
};
} catch (error) {
console.error('Ошибка создания предложения о переводе токенов:', error);
throw error;
}
}

View File

@@ -11,160 +11,460 @@
-->
<template>
<BaseLayout>
<div class="content-page-block">
<div class="content-header-nav">
<button class="nav-btn" @click="goToCreate">Создать</button>
<button class="nav-btn" @click="goToList">Список страниц</button>
<button class="nav-btn" @click="goToSettings">Настройки</button>
<BaseLayout
:is-authenticated="isAuthenticated"
:identities="identities"
:token-balances="tokenBalances"
:is-loading-tokens="isLoadingTokens"
@auth-action-completed="$emit('auth-action-completed')"
>
<div class="content-create-page">
<!-- Заголовок страницы -->
<div class="page-header">
<div class="header-content">
<h1>📝 Создание страницы</h1>
<p>Создайте новую страницу для вашего DLE</p>
</div>
<div class="header-actions">
<button class="close-btn" @click="goBack">×</button>
</div>
</div>
<!-- Основной контент с тенью -->
<div class="content-block">
<form class="content-form" @submit.prevent="handleSubmit">
<!-- Основная информация -->
<div class="form-section">
<h2>Основная информация</h2>
<div class="form-group">
<label for="title">Заголовок страницы *</label>
<input
v-model="form.title"
id="title"
type="text"
required
placeholder="Введите заголовок страницы"
class="form-input"
/>
</div>
<div class="form-group">
<label for="summary">Краткое описание *</label>
<textarea
v-model="form.summary"
id="summary"
required
rows="3"
placeholder="Краткое описание страницы"
class="form-textarea"
/>
</div>
</div>
<!-- Контент -->
<div class="form-section">
<h2>Содержание</h2>
<div class="form-group">
<label for="content">Основной контент *</label>
<textarea
v-model="form.content"
id="content"
required
rows="10"
placeholder="Введите основной контент страницы"
class="form-textarea"
/>
<div class="content-stats">
<span>Слов: {{ wordCount }}</span>
<span>Символов: {{ characterCount }}</span>
</div>
</div>
</div>
<!-- SEO настройки -->
<div class="form-section">
<h2>SEO настройки</h2>
<div class="form-group">
<label for="seo-title">Meta Title</label>
<input
v-model="form.seo.title"
id="seo-title"
type="text"
placeholder="SEO заголовок (если отличается от основного)"
class="form-input"
/>
</div>
<div class="form-group">
<label for="seo-description">Meta Description</label>
<textarea
v-model="form.seo.description"
id="seo-description"
rows="3"
placeholder="SEO описание для поисковых систем"
class="form-textarea"
/>
</div>
<div class="form-group">
<label for="seo-keywords">Keywords</label>
<input
v-model="form.seo.keywords"
id="seo-keywords"
type="text"
placeholder="Ключевые слова через запятую"
class="form-input"
/>
</div>
</div>
<!-- Настройки публикации -->
<div class="form-section">
<h2>Настройки публикации</h2>
<div class="form-group">
<label class="checkbox-label">
<input
type="checkbox"
v-model="form.settings.autoPublish"
class="form-checkbox"
/>
<span>Опубликовать сразу после создания</span>
</label>
</div>
<div class="form-group">
<label for="status">Статус</label>
<select v-model="form.status" id="status" class="form-select">
<option value="draft">Черновик</option>
<option value="published">Опубликовано</option>
<option value="pending">На модерации</option>
</select>
</div>
</div>
<!-- Кнопки действий -->
<div class="form-actions">
<button type="button" class="btn btn-outline" @click="goBack">
<i class="fas fa-times"></i>
Отмена
</button>
<button type="submit" class="btn btn-primary" :disabled="isSubmitting">
<i class="fas fa-save"></i>
{{ isSubmitting ? 'Сохранение...' : 'Создать страницу' }}
</button>
</div>
</form>
</div>
<router-view />
<form class="content-form" @submit.prevent="handleSubmit">
<div class="form-group">
<label for="title">Заголовок страницы *</label>
<input v-model="form.title" id="title" type="text" required />
</div>
<div class="form-group">
<label for="summary">Краткое описание *</label>
<textarea v-model="form.summary" id="summary" required rows="2" />
</div>
<div class="form-group">
<label for="content">Основной контент *</label>
<textarea v-model="form.content" id="content" required rows="6" />
</div>
<button class="submit-btn" type="submit">Сохранить</button>
</form>
</div>
</BaseLayout>
</template>
<script setup>
import { ref } from 'vue';
import { ref, computed } from 'vue';
import { useRouter } from 'vue-router';
import BaseLayout from '../components/BaseLayout.vue';
import pagesService from '../services/pagesService';
const router = useRouter();
function goToCreate() { router.push({ name: 'content-create' }); }
function goToList() { router.push({ name: 'content-list' }); }
function goToSettings() { router.push({ name: 'content-settings' }); }
// Props
const props = defineProps({
isAuthenticated: {
type: Boolean,
default: false
},
identities: {
type: Array,
default: () => []
},
tokenBalances: {
type: Object,
default: () => ({})
},
isLoadingTokens: {
type: Boolean,
default: false
}
});
// Emits
const emit = defineEmits(['auth-action-completed']);
const router = useRouter();
// Состояние формы
const form = ref({
title: '',
summary: '',
content: ''
content: '',
seo: {
title: '',
description: '',
keywords: ''
},
settings: {
autoPublish: false
},
status: 'draft'
});
const isSubmitting = ref(false);
// Вычисляемые свойства
const wordCount = computed(() => {
return form.value.content ? form.value.content.split(/\s+/).length : 0;
});
const characterCount = computed(() => {
return form.value.content ? form.value.content.length : 0;
});
// Методы
function goBack() {
router.push({ name: 'content-list' });
}
async function handleSubmit() {
console.log('handleSubmit called', form.value);
if (!form.value.title.trim()) {
alert('Заполните заголовок страницы!');
return;
}
if (!form.value.summary.trim()) {
alert('Заполните описание страницы!');
return;
}
if (!form.value.content.trim()) {
alert('Заполните контент страницы!');
return;
}
try {
if (!form.value.title) {
alert('Заполните заголовок страницы!');
return;
}
// Создаём страницу через pagesService
const page = await pagesService.createPage({
title: form.value.title,
summary: form.value.summary,
content: form.value.content
});
console.log('createPage result:', page);
isSubmitting.value = true;
const pageData = {
title: form.value.title.trim(),
summary: form.value.summary.trim(),
content: form.value.content.trim(),
seo: form.value.seo,
status: form.value.status,
settings: form.value.settings
};
const page = await pagesService.createPage(pageData);
if (!page || !page.id) {
alert('Ошибка: страница не создана!');
return;
throw new Error('Страница не была создана');
}
// Перенаправляем на список страниц
router.push({ name: 'content-list' });
} catch (e) {
alert('Ошибка при создании страницы: ' + (e?.message || e));
console.error('Ошибка при создании страницы:', e);
} catch (error) {
console.error('Ошибка при создании страницы:', error);
alert('Ошибка при создании страницы: ' + (error?.message || error));
} finally {
isSubmitting.value = false;
}
}
</script>
<style scoped>
.content-page-block {
background: #fff;
border-radius: 16px;
box-shadow: 0 4px 32px rgba(0,0,0,0.12);
padding: 32px 24px 24px 24px;
width: 100%;
margin-top: 40px;
position: relative;
overflow-x: auto;
}
.content-form {
display: flex;
flex-direction: column;
gap: 18px;
margin-top: 24px;
}
.form-group {
display: flex;
flex-direction: column;
gap: 6px;
}
input[type="text"], textarea, select {
border: 1px solid #d0d0d0;
border-radius: 6px;
padding: 8px 10px;
font-size: 1rem;
.content-create-page {
padding: 20px;
width: 100%;
}
.tags-input {
.page-header {
display: flex;
flex-direction: column;
gap: 6px;
justify-content: space-between;
align-items: flex-start;
margin-bottom: 30px;
padding-bottom: 20px;
border-bottom: 2px solid #f0f0f0;
}
.tags-list {
display: flex;
flex-wrap: wrap;
gap: 6px;
.header-content {
flex: 1;
}
.tag {
background: #f0f0f0;
border-radius: 4px;
padding: 2px 8px;
display: flex;
align-items: center;
font-size: 0.95em;
.header-content h1 {
color: var(--color-primary);
font-size: 2.5rem;
margin: 0 0 10px 0;
}
.tag button {
.header-content p {
color: var(--color-grey-dark);
font-size: 1.1rem;
margin: 0;
}
.close-btn {
background: none;
border: none;
color: #888;
margin-left: 4px;
font-size: 2rem;
color: var(--color-grey-dark);
cursor: pointer;
font-size: 1.1em;
padding: 0 10px;
transition: color 0.3s ease;
}
.submit-btn {
background: #2d72d9;
color: #fff;
border: none;
border-radius: 6px;
padding: 10px 0;
font-size: 1.1em;
cursor: pointer;
margin-top: 12px;
transition: background 0.2s;
.close-btn:hover {
color: var(--color-primary);
}
.submit-btn:hover {
background: #1a4e96;
.content-block {
background: #f8f9fa;
border-radius: var(--radius-lg);
padding: 25px;
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.1);
}
.content-header-nav {
.content-form {
background: white;
border-radius: var(--radius-sm);
padding: 30px;
border: 1px solid #e9ecef;
max-width: 1000px;
margin: 0 auto;
}
.form-section {
margin-bottom: 30px;
}
.form-section:last-child {
margin-bottom: 0;
}
.form-section h2 {
color: var(--color-primary);
margin: 0 0 20px 0;
font-size: 1.3rem;
border-bottom: 1px solid #f0f0f0;
padding-bottom: 10px;
}
.form-group {
margin-bottom: 20px;
}
.form-group:last-child {
margin-bottom: 0;
}
.form-group label {
display: block;
margin-bottom: 8px;
font-weight: 500;
color: var(--color-grey-dark);
}
.form-input,
.form-textarea,
.form-select {
width: 100%;
padding: 12px 15px;
border: 1px solid #e9ecef;
border-radius: var(--radius-sm);
font-size: 1rem;
transition: border-color 0.3s ease;
box-sizing: border-box;
}
.form-input:focus,
.form-textarea:focus,
.form-select:focus {
outline: none;
border-color: var(--color-primary);
box-shadow: 0 0 0 3px rgba(45, 114, 217, 0.1);
}
.form-textarea {
resize: vertical;
min-height: 100px;
}
.content-stats {
display: flex;
gap: 12px;
margin-bottom: 18px;
gap: 20px;
margin-top: 8px;
font-size: 0.9rem;
color: var(--color-grey-dark);
}
.nav-btn {
background: #f5f5f5;
border: 1px solid #d0d0d0;
border-radius: 6px;
padding: 7px 18px;
font-size: 1em;
.checkbox-label {
display: flex;
align-items: center;
gap: 10px;
cursor: pointer;
transition: background 0.2s;
font-weight: normal;
}
.nav-btn:hover {
background: #e0e0e0;
.form-checkbox {
width: auto;
margin: 0;
}
.form-actions {
display: flex;
gap: 15px;
justify-content: flex-end;
margin-top: 30px;
padding-top: 20px;
border-top: 1px solid #f0f0f0;
}
.btn {
padding: 12px 24px;
border: none;
border-radius: var(--radius-sm);
font-size: 1rem;
cursor: pointer;
transition: all 0.3s ease;
text-decoration: none;
display: inline-flex;
align-items: center;
gap: 8px;
}
.btn:disabled {
opacity: 0.6;
cursor: not-allowed;
}
.btn-primary {
background: var(--color-primary);
color: white;
}
.btn-primary:hover:not(:disabled) {
background: var(--color-primary-dark);
}
.btn-outline {
background: white;
color: var(--color-primary);
border: 1px solid var(--color-primary);
}
.btn-outline:hover {
background: var(--color-primary);
color: white;
}
@media (max-width: 768px) {
.page-header {
flex-direction: column;
gap: 15px;
}
.header-content h1 {
font-size: 2rem;
}
.form-actions {
flex-direction: column;
}
.content-stats {
flex-direction: column;
gap: 5px;
}
}
</style>

View File

@@ -11,84 +11,748 @@
-->
<template>
<BaseLayout>
<div class="content-list-block">
<div class="content-header-nav">
<button class="nav-btn" @click="goToCreate">Создать</button>
<button class="nav-btn" @click="goToList">Список страниц</button>
<button class="nav-btn" @click="goToSettings">Настройки</button>
<BaseLayout
:is-authenticated="isAuthenticated"
:identities="identities"
:token-balances="tokenBalances"
:is-loading-tokens="isLoadingTokens"
@auth-action-completed="$emit('auth-action-completed')"
>
<div class="content-management-page">
<!-- Заголовок страницы -->
<div class="page-header">
<div class="header-content">
<h1>📄 Управление контентом</h1>
<p>Создавайте и управляйте страницами вашего DLE</p>
<button class="btn btn-primary" @click="goToCreate">
<i class="fas fa-plus"></i>
Создать страницу
</button>
</div>
<div class="header-actions">
<button class="close-btn" @click="goBack">×</button>
</div>
</div>
<!-- Основной контент с тенью -->
<div class="content-block">
<!-- Навигация -->
<div class="content-navigation">
<div class="nav-tabs">
<button
class="nav-tab"
:class="{ active: activeTab === 'pages' }"
@click="activeTab = 'pages'"
>
<i class="fas fa-file-alt"></i>
Страницы
</button>
<button
class="nav-tab"
:class="{ active: activeTab === 'templates' }"
@click="activeTab = 'templates'"
>
<i class="fas fa-layer-group"></i>
Шаблоны
</button>
<button
class="nav-tab"
:class="{ active: activeTab === 'settings' }"
@click="activeTab = 'settings'"
>
<i class="fas fa-cog"></i>
Настройки
</button>
</div>
</div>
<!-- Контент в зависимости от активной вкладки -->
<div class="content-section">
<!-- Вкладка Страницы -->
<div v-if="activeTab === 'pages'" class="pages-section">
<div class="section-header">
<h2>Созданные страницы</h2>
<div class="search-box">
<input
v-model="searchQuery"
type="text"
placeholder="Поиск страниц..."
class="search-input"
>
<i class="fas fa-search search-icon"></i>
</div>
</div>
<!-- Список страниц -->
<div v-if="filteredPages.length" class="pages-grid">
<div
v-for="page in filteredPages"
:key="page.id"
class="page-card"
@click="goToPage(page.id)"
>
<div class="page-card-header">
<h3>{{ page.title }}</h3>
<div class="page-actions">
<button
class="action-btn edit-btn"
@click.stop="goToEdit(page.id)"
title="Редактировать"
>
<i class="fas fa-edit"></i>
</button>
<button
class="action-btn delete-btn"
@click.stop="deletePage(page.id)"
title="Удалить"
>
<i class="fas fa-trash"></i>
</button>
</div>
</div>
<div class="page-card-content">
<p class="page-summary">{{ page.summary || 'Без описания' }}</p>
<div class="page-meta">
<span class="page-date">
<i class="fas fa-calendar"></i>
{{ formatDate(page.createdAt) }}
</span>
<span class="page-status" :class="page.status">
<i class="fas fa-circle"></i>
{{ getStatusText(page.status) }}
</span>
</div>
</div>
</div>
</div>
<!-- Пустое состояние -->
<div v-else-if="!isLoading" class="empty-state">
<div class="empty-icon">
<i class="fas fa-file-alt"></i>
</div>
<h3>Нет созданных страниц</h3>
<p>Создайте первую страницу для вашего DLE</p>
<button class="btn btn-primary" @click="goToCreate">
<i class="fas fa-plus"></i>
Создать страницу
</button>
</div>
<!-- Загрузка -->
<div v-else class="loading-state">
<div class="loading-spinner"></div>
<p>Загрузка страниц...</p>
</div>
</div>
<!-- Вкладка Шаблоны -->
<div v-if="activeTab === 'templates'" class="templates-section">
<div class="section-header">
<h2>Шаблоны страниц</h2>
<p>Готовые шаблоны для быстрого создания контента</p>
</div>
<div class="templates-grid">
<div
v-for="template in templates"
:key="template.id"
class="template-card"
@click="useTemplate(template)"
>
<div class="template-icon">
<i :class="template.icon"></i>
</div>
<h3>{{ template.name }}</h3>
<p>{{ template.description }}</p>
<button class="btn btn-outline">Использовать шаблон</button>
</div>
</div>
</div>
<!-- Вкладка Настройки -->
<div v-if="activeTab === 'settings'" class="settings-section">
<div class="section-header">
<h2>Настройки контента</h2>
</div>
<div class="settings-grid">
<div class="setting-card">
<h3>SEO настройки</h3>
<div class="setting-item">
<label>Мета-теги по умолчанию</label>
<textarea v-model="seoSettings.defaultMeta" placeholder="Введите мета-теги..."></textarea>
</div>
</div>
<div class="setting-card">
<h3>Настройки публикации</h3>
<div class="setting-item">
<label>
<input type="checkbox" v-model="publishSettings.autoPublish">
Автоматическая публикация
</label>
</div>
</div>
</div>
</div>
</div>
</div>
<h2>Список страниц</h2>
<ul v-if="pages.length" class="pages-list">
<li v-for="page in pages" :key="page.id">
<router-link :to="{ name: 'page-view', params: { id: page.id } }">{{ page.title }}</router-link>
</li>
</ul>
<div v-else class="empty-list-placeholder">Нет созданных страниц.</div>
</div>
</BaseLayout>
</template>
<script setup>
import { onMounted, ref } from 'vue';
import { ref, computed, onMounted } from 'vue';
import { useRouter } from 'vue-router';
import BaseLayout from '../../components/BaseLayout.vue';
import pagesService from '../../services/pagesService';
const router = useRouter();
function goToCreate() { router.push({ name: 'content-create' }); }
function goToList() { router.push({ name: 'content-list' }); }
function goToSettings() { router.push({ name: 'content-settings' }); }
// Props
const props = defineProps({
isAuthenticated: {
type: Boolean,
default: false
},
identities: {
type: Array,
default: () => []
},
tokenBalances: {
type: Object,
default: () => ({})
},
isLoadingTokens: {
type: Boolean,
default: false
}
});
// Emits
const emit = defineEmits(['auth-action-completed']);
const router = useRouter();
// Состояние
const activeTab = ref('pages');
const pages = ref([]);
onMounted(async () => {
pages.value = await pagesService.getPages();
const isLoading = ref(false);
const searchQuery = ref('');
// Настройки
const seoSettings = ref({
defaultMeta: ''
});
const publishSettings = ref({
autoPublish: false
});
// Шаблоны
const templates = ref([
{
id: 1,
name: 'О компании',
description: 'Стандартная страница с информацией о компании',
icon: 'fas fa-building'
},
{
id: 2,
name: 'Услуги',
description: 'Страница с описанием услуг и сервисов',
icon: 'fas fa-cogs'
},
{
id: 3,
name: 'Контакты',
description: 'Контактная информация и форма обратной связи',
icon: 'fas fa-address-book'
},
{
id: 4,
name: 'Блог',
description: 'Шаблон для ведения блога и новостей',
icon: 'fas fa-blog'
}
]);
// Вычисляемые свойства
const filteredPages = computed(() => {
if (!searchQuery.value) return pages.value;
return pages.value.filter(page =>
page.title.toLowerCase().includes(searchQuery.value.toLowerCase()) ||
page.summary?.toLowerCase().includes(searchQuery.value.toLowerCase())
);
});
// Методы
function goToCreate() {
router.push({ name: 'content-create' });
}
function goBack() {
router.go(-1);
}
function goToPage(id) {
router.push({ name: 'page-view', params: { id } });
}
function goToEdit(id) {
router.push({ name: 'page-edit', params: { id } });
}
async function deletePage(id) {
if (confirm('Вы уверены, что хотите удалить эту страницу?')) {
try {
await pagesService.deletePage(id);
await loadPages();
} catch (error) {
console.error('Ошибка удаления страницы:', error);
}
}
}
function useTemplate(template) {
router.push({
name: 'content-create',
query: { template: template.id }
});
}
function formatDate(date) {
if (!date) return 'Не указана';
return new Date(date).toLocaleDateString('ru-RU');
}
function getStatusText(status) {
const statusMap = {
draft: 'Черновик',
published: 'Опубликовано',
archived: 'Архив'
};
return statusMap[status] || 'Неизвестно';
}
async function loadPages() {
try {
isLoading.value = true;
pages.value = await pagesService.getPages();
} catch (error) {
console.error('Ошибка загрузки страниц:', error);
pages.value = [];
} finally {
isLoading.value = false;
}
}
// Загрузка данных
onMounted(() => {
loadPages();
});
</script>
<style scoped>
.content-list-block {
background: #fff;
border-radius: 16px;
box-shadow: 0 4px 32px rgba(0,0,0,0.12);
padding: 32px 24px 24px 24px;
.content-management-page {
padding: 20px;
width: 100%;
margin-top: 40px;
position: relative;
overflow-x: auto;
}
.content-header-nav {
.page-header {
display: flex;
gap: 12px;
margin-bottom: 18px;
justify-content: space-between;
align-items: flex-start;
margin-bottom: 30px;
padding-bottom: 20px;
border-bottom: 2px solid #f0f0f0;
}
.nav-btn {
background: #f5f5f5;
border: 1px solid #d0d0d0;
border-radius: 6px;
padding: 7px 18px;
font-size: 1em;
.header-content h1 {
color: var(--color-primary);
font-size: 2.5rem;
margin: 0 0 10px 0;
}
.header-content p {
color: var(--color-grey-dark);
font-size: 1.1rem;
margin: 0 0 20px 0;
}
.header-content .btn {
margin-top: 10px;
}
.content-navigation {
margin-bottom: 30px;
}
.nav-tabs {
display: flex;
gap: 10px;
border-bottom: 1px solid #e9ecef;
}
.nav-tab {
background: none;
border: none;
padding: 15px 20px;
font-size: 1rem;
cursor: pointer;
border-bottom: 3px solid transparent;
transition: all 0.3s ease;
color: var(--color-grey-dark);
}
.nav-tab:hover {
color: var(--color-primary);
}
.nav-tab.active {
color: var(--color-primary);
border-bottom-color: var(--color-primary);
}
.nav-tab i {
margin-right: 8px;
}
.content-block {
background: #f8f9fa;
border-radius: var(--radius-lg);
padding: 25px;
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.1);
}
.content-section {
background: #f8f9fa;
border-radius: var(--radius-lg);
padding: 25px;
}
.section-header {
display: flex;
justify-content: space-between;
align-items: center;
margin-bottom: 25px;
}
.section-header h2 {
color: var(--color-primary);
margin: 0;
}
.search-box {
position: relative;
width: 300px;
}
.search-input {
width: 100%;
padding: 10px 40px 10px 15px;
border: 1px solid #e9ecef;
border-radius: var(--radius-sm);
font-size: 1rem;
}
.search-icon {
position: absolute;
right: 15px;
top: 50%;
transform: translateY(-50%);
color: var(--color-grey-dark);
}
.pages-grid {
display: grid;
grid-template-columns: repeat(auto-fill, minmax(350px, 1fr));
gap: 20px;
margin-top: 20px;
}
.page-card {
background: white;
border-radius: var(--radius-sm);
padding: 20px;
border: 1px solid #e9ecef;
cursor: pointer;
transition: all 0.3s ease;
}
.page-card:hover {
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.1);
transform: translateY(-2px);
}
.page-card-header {
display: flex;
justify-content: space-between;
align-items: flex-start;
margin-bottom: 15px;
}
.page-card-header h3 {
color: var(--color-primary);
margin: 0;
font-size: 1.2rem;
}
.page-actions {
display: flex;
gap: 8px;
}
.action-btn {
background: none;
border: none;
padding: 5px;
cursor: pointer;
border-radius: var(--radius-sm);
transition: background 0.2s;
}
.nav-btn:hover {
background: #e0e0e0;
.edit-btn:hover {
background: #e3f2fd;
color: #2196f3;
}
.empty-list-placeholder {
color: #888;
font-size: 1.1em;
margin-top: 2em;
.delete-btn:hover {
background: #ffebee;
color: #f44336;
}
.pages-list {
margin-top: 1.5em;
padding-left: 0;
list-style: none;
.page-summary {
color: var(--color-grey-dark);
margin: 0 0 15px 0;
line-height: 1.5;
}
.pages-list li {
padding: 0.5em 0;
border-bottom: 1px solid #f0f0f0;
font-size: 1.08em;
.page-meta {
display: flex;
gap: 15px;
font-size: 0.9rem;
color: var(--color-grey-dark);
}
.pages-list li:last-child {
border-bottom: none;
.page-date i,
.page-status i {
margin-right: 5px;
}
.page-status.draft i {
color: #ff9800;
}
.page-status.published i {
color: #4caf50;
}
.page-status.archived i {
color: #9e9e9e;
}
.empty-state {
text-align: center;
padding: 60px 20px;
}
.empty-icon {
font-size: 4rem;
color: var(--color-grey-dark);
margin-bottom: 20px;
}
.empty-state h3 {
color: var(--color-primary);
margin: 0 0 10px 0;
}
.empty-state p {
color: var(--color-grey-dark);
margin: 0 0 25px 0;
}
.loading-state {
text-align: center;
padding: 60px 20px;
}
.loading-spinner {
border: 3px solid #f3f3f3;
border-top: 3px solid var(--color-primary);
border-radius: 50%;
width: 40px;
height: 40px;
animation: spin 1s linear infinite;
margin: 0 auto 20px;
}
@keyframes spin {
0% { transform: rotate(0deg); }
100% { transform: rotate(360deg); }
}
.templates-grid {
display: grid;
grid-template-columns: repeat(auto-fill, minmax(300px, 1fr));
gap: 20px;
margin-top: 20px;
}
.template-card {
background: white;
border-radius: var(--radius-sm);
padding: 25px;
text-align: center;
border: 1px solid #e9ecef;
cursor: pointer;
transition: all 0.3s ease;
}
.template-card:hover {
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.1);
transform: translateY(-2px);
}
.template-icon {
font-size: 3rem;
color: var(--color-primary);
margin-bottom: 15px;
}
.template-card h3 {
color: var(--color-primary);
margin: 0 0 10px 0;
}
.template-card p {
color: var(--color-grey-dark);
margin: 0 0 20px 0;
line-height: 1.5;
}
.settings-grid {
display: grid;
grid-template-columns: repeat(auto-fill, minmax(400px, 1fr));
gap: 20px;
margin-top: 20px;
}
.setting-card {
background: white;
border-radius: var(--radius-sm);
padding: 20px;
border: 1px solid #e9ecef;
}
.setting-card h3 {
color: var(--color-primary);
margin: 0 0 15px 0;
}
.setting-item {
margin-bottom: 15px;
}
.setting-item label {
display: block;
margin-bottom: 8px;
font-weight: 500;
color: var(--color-grey-dark);
}
.setting-item textarea {
width: 100%;
min-height: 100px;
padding: 10px;
border: 1px solid #e9ecef;
border-radius: var(--radius-sm);
resize: vertical;
}
.setting-item input[type="checkbox"] {
margin-right: 8px;
}
.btn {
padding: 10px 20px;
border: none;
border-radius: var(--radius-sm);
font-size: 1rem;
cursor: pointer;
transition: all 0.3s ease;
text-decoration: none;
display: inline-flex;
align-items: center;
gap: 8px;
}
.btn-primary {
background: var(--color-primary);
color: white;
}
.btn-primary:hover {
background: var(--color-primary-dark);
}
.btn-outline {
background: white;
color: var(--color-primary);
border: 1px solid var(--color-primary);
}
.btn-outline:hover {
background: var(--color-primary);
color: white;
}
.close-btn {
background: none;
border: none;
font-size: 1.5rem;
cursor: pointer;
color: var(--color-grey-dark);
padding: 5px;
border-radius: var(--radius-sm);
transition: background 0.2s;
}
.close-btn:hover {
background: #f0f0f0;
color: var(--color-primary);
}
@media (max-width: 768px) {
.page-header {
flex-direction: column;
gap: 15px;
}
.nav-tabs {
flex-wrap: wrap;
}
.section-header {
flex-direction: column;
gap: 15px;
align-items: stretch;
}
.search-box {
width: 100%;
}
.pages-grid {
grid-template-columns: 1fr;
}
.templates-grid {
grid-template-columns: 1fr;
}
.settings-grid {
grid-template-columns: 1fr;
}
}
</style>

View File

@@ -11,48 +11,474 @@
-->
<template>
<BaseLayout>
<div v-if="page" class="page-edit-block">
<h2>Редактировать страницу</h2>
<form @submit.prevent="save">
<label>Заголовок</label>
<input v-model="page.title" required />
<label>Описание</label>
<textarea v-model="page.summary" />
<label>Контент</label>
<textarea v-model="page.content" />
<button type="submit">Сохранить</button>
<button type="button" @click="goBack">Отмена</button>
</form>
<BaseLayout
:is-authenticated="isAuthenticated"
:identities="identities"
:token-balances="tokenBalances"
:is-loading-tokens="isLoadingTokens"
@auth-action-completed="$emit('auth-action-completed')"
>
<div v-if="page" class="page-edit-page">
<!-- Заголовок страницы -->
<div class="page-header">
<div class="header-content">
<h1> Редактирование страницы</h1>
<p v-if="page">Редактируйте содержимое страницы "{{ page.title }}"</p>
<p v-else>Загрузка страницы...</p>
</div>
<div class="header-actions">
<button class="close-btn" @click="goBack">×</button>
</div>
</div>
<!-- Основной контент с тенью -->
<div class="content-block">
<form class="content-form" @submit.prevent="save">
<!-- Основная информация -->
<div class="form-section">
<h2>Основная информация</h2>
<div class="form-group">
<label for="title">Заголовок страницы *</label>
<input
v-model="page.title"
id="title"
type="text"
required
placeholder="Введите заголовок страницы"
class="form-input"
/>
</div>
<div class="form-group">
<label for="summary">Краткое описание *</label>
<textarea
v-model="page.summary"
id="summary"
required
rows="3"
placeholder="Краткое описание страницы"
class="form-textarea"
/>
</div>
</div>
<!-- Контент -->
<div class="form-section">
<h2>Содержание</h2>
<div class="form-group">
<label for="content">Основной контент *</label>
<textarea
v-model="page.content"
id="content"
required
rows="10"
placeholder="Введите основной контент страницы"
class="form-textarea"
/>
<div class="content-stats">
<span>Слов: {{ wordCount }}</span>
<span>Символов: {{ characterCount }}</span>
</div>
</div>
</div>
<!-- SEO настройки -->
<div class="form-section">
<h2>SEO настройки</h2>
<div class="form-group">
<label for="seo-title">Meta Title</label>
<input
v-model="page.seo.title"
id="seo-title"
type="text"
placeholder="SEO заголовок (если отличается от основного)"
class="form-input"
/>
</div>
<div class="form-group">
<label for="seo-description">Meta Description</label>
<textarea
v-model="page.seo.description"
id="seo-description"
rows="3"
placeholder="SEO описание для поисковых систем"
class="form-textarea"
/>
</div>
<div class="form-group">
<label for="seo-keywords">Keywords</label>
<input
v-model="page.seo.keywords"
id="seo-keywords"
type="text"
placeholder="Ключевые слова через запятую"
class="form-input"
/>
</div>
</div>
<!-- Настройки публикации -->
<div class="form-section">
<h2>Настройки публикации</h2>
<div class="form-group">
<label for="status">Статус</label>
<select v-model="page.status" id="status" class="form-select">
<option value="draft">Черновик</option>
<option value="published">Опубликовано</option>
<option value="pending">На модерации</option>
<option value="archived">Архив</option>
</select>
</div>
</div>
<!-- Кнопки действий -->
<div class="form-actions">
<button type="button" class="btn btn-outline" @click="goBack">
<i class="fas fa-times"></i>
Отмена
</button>
<button type="submit" class="btn btn-primary" :disabled="isSubmitting">
<i class="fas fa-save"></i>
{{ isSubmitting ? 'Сохранение...' : 'Сохранить изменения' }}
</button>
</div>
</form>
</div>
</div>
<!-- Загрузка -->
<div v-else class="loading-state">
<div class="loading-spinner"></div>
<p>Загрузка страницы...</p>
</div>
<div v-else>Загрузка...</div>
</BaseLayout>
</template>
<script setup>
import { ref, onMounted, computed } from 'vue';
import { ref, computed, onMounted } from 'vue';
import { useRoute, useRouter } from 'vue-router';
import BaseLayout from '../../components/BaseLayout.vue';
import pagesService from '../../services/pagesService';
// Props
const props = defineProps({
isAuthenticated: {
type: Boolean,
default: false
},
identities: {
type: Array,
default: () => []
},
tokenBalances: {
type: Object,
default: () => ({})
},
isLoadingTokens: {
type: Boolean,
default: false
}
});
// Emits
const emit = defineEmits(['auth-action-completed']);
const route = useRoute();
const router = useRouter();
const page = ref(null);
const isSubmitting = ref(false);
onMounted(async () => {
page.value = await pagesService.getPage(route.params.id);
// Вычисляемые свойства
const wordCount = computed(() => {
return page.value?.content ? page.value.content.split(/\s+/).length : 0;
});
async function save() {
await pagesService.updatePage(route.params.id, {
title: page.value.title,
summary: page.value.summary,
content: page.value.content
});
router.push({ name: 'page-view', params: { id: route.params.id } });
}
const characterCount = computed(() => {
return page.value?.content ? page.value.content.length : 0;
});
// Методы
function goBack() {
router.push({ name: 'page-view', params: { id: route.params.id } });
}
</script>
async function save() {
if (!page.value.title.trim()) {
alert('Заполните заголовок страницы!');
return;
}
if (!page.value.summary.trim()) {
alert('Заполните описание страницы!');
return;
}
if (!page.value.content.trim()) {
alert('Заполните контент страницы!');
return;
}
try {
isSubmitting.value = true;
const pageData = {
title: page.value.title.trim(),
summary: page.value.summary.trim(),
content: page.value.content.trim(),
seo: page.value.seo || {},
status: page.value.status || 'draft'
};
await pagesService.updatePage(route.params.id, pageData);
// Перенаправляем на просмотр страницы
router.push({ name: 'page-view', params: { id: route.params.id } });
} catch (error) {
console.error('Ошибка при сохранении страницы:', error);
alert('Ошибка при сохранении страницы: ' + (error?.message || error));
} finally {
isSubmitting.value = false;
}
}
// Загрузка данных
onMounted(async () => {
try {
page.value = await pagesService.getPage(route.params.id);
// Инициализируем SEO объект если его нет
if (!page.value.seo) {
page.value.seo = {
title: '',
description: '',
keywords: ''
};
}
} catch (error) {
console.error('Ошибка при загрузке страницы:', error);
alert('Ошибка при загрузке страницы: ' + (error?.message || error));
}
});
</script>
<style scoped>
.page-edit-page {
padding: 20px;
width: 100%;
}
.page-header {
display: flex;
justify-content: space-between;
align-items: flex-start;
margin-bottom: 30px;
padding-bottom: 20px;
border-bottom: 2px solid #f0f0f0;
}
.header-content {
flex: 1;
}
.header-content h1 {
color: var(--color-primary);
font-size: 2.5rem;
margin: 0 0 10px 0;
}
.header-content p {
color: var(--color-grey-dark);
font-size: 1.1rem;
margin: 0;
}
.close-btn {
background: none;
border: none;
font-size: 2rem;
color: var(--color-grey-dark);
cursor: pointer;
padding: 0 10px;
transition: color 0.3s ease;
}
.close-btn:hover {
color: var(--color-primary);
}
.content-block {
background: #f8f9fa;
border-radius: var(--radius-lg);
padding: 25px;
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.1);
}
.content-form {
background: white;
border-radius: var(--radius-sm);
padding: 30px;
border: 1px solid #e9ecef;
max-width: 1000px;
margin: 0 auto;
}
.form-section {
margin-bottom: 30px;
}
.form-section:last-child {
margin-bottom: 0;
}
.form-section h2 {
color: var(--color-primary);
margin: 0 0 20px 0;
font-size: 1.3rem;
border-bottom: 1px solid #f0f0f0;
padding-bottom: 10px;
}
.form-group {
margin-bottom: 20px;
}
.form-group:last-child {
margin-bottom: 0;
}
.form-group label {
display: block;
margin-bottom: 8px;
font-weight: 500;
color: var(--color-grey-dark);
}
.form-input,
.form-textarea,
.form-select {
width: 100%;
padding: 12px 15px;
border: 1px solid #e9ecef;
border-radius: var(--radius-sm);
font-size: 1rem;
transition: border-color 0.3s ease;
box-sizing: border-box;
}
.form-input:focus,
.form-textarea:focus,
.form-select:focus {
outline: none;
border-color: var(--color-primary);
box-shadow: 0 0 0 3px rgba(45, 114, 217, 0.1);
}
.form-textarea {
resize: vertical;
min-height: 100px;
}
.content-stats {
display: flex;
gap: 20px;
margin-top: 8px;
font-size: 0.9rem;
color: var(--color-grey-dark);
}
.form-actions {
display: flex;
gap: 15px;
justify-content: flex-end;
margin-top: 30px;
padding-top: 20px;
border-top: 1px solid #f0f0f0;
}
.btn {
padding: 12px 24px;
border: none;
border-radius: var(--radius-sm);
font-size: 1rem;
cursor: pointer;
transition: all 0.3s ease;
text-decoration: none;
display: inline-flex;
align-items: center;
gap: 8px;
}
.btn:disabled {
opacity: 0.6;
cursor: not-allowed;
}
.btn-primary {
background: var(--color-primary);
color: white;
}
.btn-primary:hover:not(:disabled) {
background: var(--color-primary-dark);
}
.btn-outline {
background: white;
color: var(--color-primary);
border: 1px solid var(--color-primary);
}
.btn-outline:hover {
background: var(--color-primary);
color: white;
}
.loading-state {
display: flex;
flex-direction: column;
align-items: center;
justify-content: center;
padding: 60px 20px;
text-align: center;
}
.loading-spinner {
width: 40px;
height: 40px;
border: 4px solid #f3f3f3;
border-top: 4px solid var(--color-primary);
border-radius: 50%;
animation: spin 1s linear infinite;
margin-bottom: 20px;
}
@keyframes spin {
0% { transform: rotate(0deg); }
100% { transform: rotate(360deg); }
}
.loading-state p {
color: var(--color-grey-dark);
font-size: 1.1rem;
margin: 0;
}
@media (max-width: 768px) {
.page-header {
flex-direction: column;
gap: 15px;
}
.header-content h1 {
font-size: 2rem;
}
.form-actions {
flex-direction: column;
}
.content-stats {
flex-direction: column;
gap: 5px;
}
}
</style>

View File

@@ -11,15 +11,128 @@
-->
<template>
<BaseLayout>
<div v-if="page" class="page-view-block">
<h2>{{ page.title }}</h2>
<p><b>Описание:</b> {{ page.summary }}</p>
<div><b>Контент:</b> {{ page.content }}</div>
<button @click="goToEdit">Редактировать</button>
<button @click="deletePage" style="color:red">Удалить</button>
<BaseLayout
:is-authenticated="isAuthenticated"
:identities="identities"
:token-balances="tokenBalances"
:is-loading-tokens="isLoadingTokens"
@auth-action-completed="$emit('auth-action-completed')"
>
<div class="page-view-container">
<!-- Заголовок страницы -->
<div v-if="page" class="page-header">
<div class="header-content">
<h1>📄 {{ page.title }}</h1>
<div class="page-meta">
<span class="page-status" :class="page.status">
<i class="fas fa-circle"></i>
{{ getStatusText(page.status) }}
</span>
<span class="page-date">
<i class="fas fa-calendar"></i>
Создано: {{ formatDate(page.createdAt) }}
</span>
<span v-if="page.updatedAt" class="page-date">
<i class="fas fa-edit"></i>
Обновлено: {{ formatDate(page.updatedAt) }}
</span>
</div>
</div>
<div class="header-actions">
<button class="btn btn-outline" @click="goToEdit">
<i class="fas fa-edit"></i>
Редактировать
</button>
<button class="btn btn-danger" @click="deletePage">
<i class="fas fa-trash"></i>
Удалить
</button>
<button class="close-btn" @click="goBack">×</button>
</div>
</div>
<!-- Контент страницы -->
<div v-if="page" class="page-content-block">
<div class="page-content">
<!-- Описание -->
<div v-if="page.summary" class="content-section">
<h2>Описание</h2>
<div class="summary-content">
{{ page.summary }}
</div>
</div>
<!-- Основной контент -->
<div class="content-section">
<h2>Содержание</h2>
<div class="main-content">
<div v-if="page.content" v-html="formatContent(page.content)"></div>
<div v-else class="empty-content">
<i class="fas fa-file-alt"></i>
<p>Контент не добавлен</p>
</div>
</div>
</div>
<!-- SEO информация -->
<div v-if="page.seo" class="content-section">
<h2>SEO информация</h2>
<div class="seo-info">
<div class="seo-item">
<label>Meta Title:</label>
<span>{{ page.seo.title || 'Не указан' }}</span>
</div>
<div class="seo-item">
<label>Meta Description:</label>
<span>{{ page.seo.description || 'Не указан' }}</span>
</div>
<div class="seo-item">
<label>Keywords:</label>
<span>{{ page.seo.keywords || 'Не указаны' }}</span>
</div>
</div>
</div>
<!-- Статистика -->
<div class="content-section">
<h2>Статистика</h2>
<div class="stats-grid">
<div class="stat-item">
<div class="stat-value">{{ page.views || 0 }}</div>
<div class="stat-label">Просмотров</div>
</div>
<div class="stat-item">
<div class="stat-value">{{ page.wordCount || 0 }}</div>
<div class="stat-label">Слов</div>
</div>
<div class="stat-item">
<div class="stat-value">{{ page.characterCount || 0 }}</div>
<div class="stat-label">Символов</div>
</div>
</div>
</div>
</div>
</div>
<!-- Загрузка -->
<div v-else-if="isLoading" class="loading-state">
<div class="loading-spinner"></div>
<p>Загрузка страницы...</p>
</div>
<!-- Ошибка -->
<div v-else class="error-state">
<div class="error-icon">
<i class="fas fa-exclamation-triangle"></i>
</div>
<h3>Страница не найдена</h3>
<p>Запрашиваемая страница не существует или была удалена</p>
<button class="btn btn-primary" @click="goBack">
<i class="fas fa-arrow-left"></i>
Вернуться назад
</button>
</div>
</div>
<div v-else>Загрузка...</div>
</BaseLayout>
</template>
@@ -29,22 +142,427 @@ import { useRoute, useRouter } from 'vue-router';
import BaseLayout from '../../components/BaseLayout.vue';
import pagesService from '../../services/pagesService';
const route = useRoute();
const router = useRouter();
const page = ref(null);
onMounted(async () => {
page.value = await pagesService.getPage(route.params.id);
// Props
const props = defineProps({
isAuthenticated: {
type: Boolean,
default: false
},
identities: {
type: Array,
default: () => []
},
tokenBalances: {
type: Object,
default: () => ({})
},
isLoadingTokens: {
type: Boolean,
default: false
}
});
// Emits
const emit = defineEmits(['auth-action-completed']);
const route = useRoute();
const router = useRouter();
// Состояние
const page = ref(null);
const isLoading = ref(false);
// Методы
function goToEdit() {
router.push({ name: 'page-edit', params: { id: route.params.id } });
}
async function deletePage() {
if (confirm('Удалить страницу?')) {
await pagesService.deletePage(route.params.id);
router.push({ name: 'content-list' });
if (confirm('Вы уверены, что хотите удалить эту страницу? Это действие нельзя отменить.')) {
try {
await pagesService.deletePage(route.params.id);
router.push({ name: 'content-list' });
} catch (error) {
console.error('Ошибка удаления страницы:', error);
alert('Ошибка при удалении страницы');
}
}
}
</script>
function goBack() {
router.go(-1);
}
function formatDate(date) {
if (!date) return 'Не указана';
return new Date(date).toLocaleDateString('ru-RU', {
year: 'numeric',
month: 'long',
day: 'numeric',
hour: '2-digit',
minute: '2-digit'
});
}
function getStatusText(status) {
const statusMap = {
draft: 'Черновик',
published: 'Опубликовано',
archived: 'Архив',
pending: 'На модерации'
};
return statusMap[status] || 'Неизвестно';
}
function formatContent(content) {
// Простое форматирование контента
if (!content) return '';
// Заменяем переносы строк на <br>
return content.replace(/\n/g, '<br>');
}
async function loadPage() {
try {
isLoading.value = true;
page.value = await pagesService.getPage(route.params.id);
// Подсчитываем статистику
if (page.value) {
page.value.wordCount = page.value.content ? page.value.content.split(/\s+/).length : 0;
page.value.characterCount = page.value.content ? page.value.content.length : 0;
}
} catch (error) {
console.error('Ошибка загрузки страницы:', error);
page.value = null;
} finally {
isLoading.value = false;
}
}
// Загрузка данных
onMounted(() => {
loadPage();
});
</script>
<style scoped>
.page-view-container {
padding: 20px;
width: 100%;
}
.page-header {
display: flex;
justify-content: space-between;
align-items: flex-start;
margin-bottom: 30px;
padding-bottom: 20px;
border-bottom: 2px solid #f0f0f0;
}
.header-content {
flex: 1;
}
.header-content h1 {
color: var(--color-primary);
font-size: 2.5rem;
margin: 0 0 15px 0;
}
.page-meta {
display: flex;
gap: 20px;
align-items: center;
font-size: 0.9rem;
color: var(--color-grey-dark);
}
.page-status {
display: flex;
align-items: center;
gap: 5px;
padding: 4px 12px;
border-radius: 20px;
font-weight: 500;
}
.page-status.draft {
background: #fff3e0;
color: #e65100;
}
.page-status.published {
background: #e8f5e8;
color: #2e7d32;
}
.page-status.archived {
background: #f5f5f5;
color: #616161;
}
.page-status.pending {
background: #e3f2fd;
color: #1565c0;
}
.page-date,
.page-updated {
display: flex;
align-items: center;
gap: 5px;
}
.header-actions {
display: flex;
gap: 10px;
}
.page-content-block {
background: #f8f9fa;
border-radius: var(--radius-lg);
padding: 25px;
box-shadow: 0 4px 15px rgba(0, 0, 0, 0.08);
}
.page-content {
background: white;
border-radius: var(--radius-sm);
padding: 25px;
border: 1px solid #e9ecef;
}
.content-section {
background: white;
border-radius: var(--radius-sm);
padding: 25px;
margin-bottom: 20px;
border: 1px solid #e9ecef;
}
.content-section:last-child {
margin-bottom: 0;
}
.content-section h2 {
color: var(--color-primary);
margin: 0 0 20px 0;
font-size: 1.5rem;
}
.summary-content {
color: var(--color-grey-dark);
line-height: 1.6;
font-size: 1.1rem;
}
.main-content {
line-height: 1.8;
color: #333;
}
.empty-content {
text-align: center;
padding: 40px 20px;
color: var(--color-grey-dark);
}
.empty-content i {
font-size: 3rem;
margin-bottom: 15px;
opacity: 0.5;
}
.seo-info {
display: grid;
gap: 15px;
}
.seo-item {
display: flex;
justify-content: space-between;
align-items: flex-start;
padding: 10px 0;
border-bottom: 1px solid #f0f0f0;
}
.seo-item:last-child {
border-bottom: none;
}
.seo-item label {
font-weight: 500;
color: var(--color-grey-dark);
min-width: 150px;
}
.seo-item span {
color: #333;
flex: 1;
margin-left: 20px;
}
.stats-grid {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(150px, 1fr));
gap: 20px;
}
.stat-item {
text-align: center;
padding: 20px;
background: #f8f9fa;
border-radius: var(--radius-sm);
}
.stat-value {
font-size: 2rem;
font-weight: bold;
color: var(--color-primary);
margin-bottom: 5px;
}
.stat-label {
color: var(--color-grey-dark);
font-size: 0.9rem;
}
.loading-state,
.error-state {
text-align: center;
padding: 60px 20px;
}
.loading-spinner {
border: 3px solid #f3f3f3;
border-top: 3px solid var(--color-primary);
border-radius: 50%;
width: 40px;
height: 40px;
animation: spin 1s linear infinite;
margin: 0 auto 20px;
}
@keyframes spin {
0% { transform: rotate(0deg); }
100% { transform: rotate(360deg); }
}
.error-icon {
font-size: 4rem;
color: #f44336;
margin-bottom: 20px;
}
.error-state h3 {
color: var(--color-primary);
margin: 0 0 10px 0;
}
.error-state p {
color: var(--color-grey-dark);
margin: 0 0 25px 0;
}
.btn {
padding: 10px 20px;
border: none;
border-radius: var(--radius-sm);
font-size: 1rem;
cursor: pointer;
transition: all 0.3s ease;
text-decoration: none;
display: inline-flex;
align-items: center;
gap: 8px;
}
.btn-outline {
background: white;
color: var(--color-primary);
border: 1px solid var(--color-primary);
}
.btn-outline:hover {
background: var(--color-primary);
color: white;
}
.btn-danger {
background: #f44336;
color: white;
}
.btn-danger:hover {
background: #d32f2f;
}
.btn-primary {
background: var(--color-primary);
color: white;
}
.btn-primary:hover {
background: var(--color-primary-dark);
}
.close-btn {
background: none;
border: none;
font-size: 2rem;
color: var(--color-grey-dark);
cursor: pointer;
padding: 0 10px;
transition: color 0.3s ease;
}
.close-btn:hover {
color: var(--color-primary);
}
@media (max-width: 768px) {
.page-header {
flex-direction: column;
gap: 15px;
}
.page-title-section h1 {
font-size: 2rem;
}
.page-meta {
flex-direction: column;
gap: 10px;
align-items: flex-start;
}
.header-actions {
width: 100%;
justify-content: stretch;
}
.header-actions .btn {
flex: 1;
justify-content: center;
}
.seo-item {
flex-direction: column;
gap: 5px;
}
.seo-item label {
min-width: auto;
}
.seo-item span {
margin-left: 0;
}
.stats-grid {
grid-template-columns: 1fr;
}
}
</style>

View File

@@ -48,6 +48,332 @@
</div>
</div>
<!-- Блоки для деплоя стандартных модулей -->
<div class="standard-modules">
<div class="modules-header">
<h3>🚀 Деплой стандартных модулей</h3>
<p>Быстрый деплой предустановленных модулей DLE</p>
</div>
<div class="modules-grid">
<!-- TreasuryModule -->
<div class="module-deploy-card">
<div class="module-content">
<h4>TreasuryModule</h4>
<p>Казначейство DLE - управление финансами, депозиты, выводы, дивиденды</p>
<div class="module-features">
<span class="feature-tag">Финансы</span>
<span class="feature-tag">Бюджет</span>
<span class="feature-tag">Дивиденды</span>
</div>
</div>
<div class="module-actions">
<button
class="btn btn-primary btn-deploy"
@click="router.push(`/management/modules/deploy/treasury?address=${route.query.address}`)"
>
<i class="fas fa-rocket"></i>
Деплой
</button>
</div>
</div>
<!-- TimelockModule -->
<div class="module-deploy-card">
<div class="module-content">
<h4>TimelockModule</h4>
<p>Задержки исполнения - безопасность критических операций через таймлоки</p>
<div class="module-features">
<span class="feature-tag">Безопасность</span>
<span class="feature-tag">Таймлок</span>
<span class="feature-tag">Аудит</span>
</div>
</div>
<div class="module-actions">
<button
class="btn btn-primary btn-deploy"
@click="router.push(`/management/modules/deploy/timelock?address=${route.query.address}`)"
>
<i class="fas fa-rocket"></i>
Деплой
</button>
</div>
</div>
<!-- CommunicationModule -->
<div class="module-deploy-card">
<div class="module-content">
<h4>CommunicationModule</h4>
<p>Коммуникации - сообщения, звонки, история общения между участниками</p>
<div class="module-features">
<span class="feature-tag">Сообщения</span>
<span class="feature-tag">Звонки</span>
<span class="feature-tag">История</span>
</div>
</div>
<div class="module-actions">
<button
class="btn btn-primary btn-deploy"
@click="router.push(`/management/modules/deploy/communication?address=${route.query.address}`)"
>
<i class="fas fa-rocket"></i>
Деплой
</button>
</div>
</div>
<!-- ApplicationModule -->
<div class="module-deploy-card">
<div class="module-content">
<h4>ApplicationModule</h4>
<p>Управление вызовом функций приложения через предложения и голосование</p>
<div class="module-features">
<span class="feature-tag">API</span>
<span class="feature-tag">Голосование</span>
<span class="feature-tag">Управление</span>
</div>
</div>
<div class="module-actions">
<button
class="btn btn-primary btn-deploy"
@click="router.push(`/management/modules/deploy/application?address=${route.query.address}`)"
>
<i class="fas fa-rocket"></i>
Деплой
</button>
</div>
</div>
<!-- MintModule -->
<div class="module-deploy-card">
<div class="module-content">
<h4>MintModule</h4>
<p>Выпуск новых токенов DLE - создание дополнительных токенов через governance</p>
<div class="module-features">
<span class="feature-tag">Минтинг</span>
<span class="feature-tag">Токены</span>
<span class="feature-tag">Governance</span>
</div>
</div>
<div class="module-actions">
<button
class="btn btn-primary btn-deploy"
@click="router.push(`/management/modules/deploy/mint?address=${route.query.address}`)"
>
<i class="fas fa-rocket"></i>
Деплой
</button>
</div>
</div>
<!-- BurnModule -->
<div class="module-deploy-card">
<div class="module-content">
<h4>BurnModule</h4>
<p>Сжигание токенов DLE - уменьшение общего предложения через governance</p>
<div class="module-features">
<span class="feature-tag">Сжигание</span>
<span class="feature-tag">Токены</span>
<span class="feature-tag">Governance</span>
</div>
</div>
<div class="module-actions">
<button
class="btn btn-primary btn-deploy"
@click="router.push(`/management/modules/deploy/burn?address=${route.query.address}`)"
>
<i class="fas fa-rocket"></i>
Деплой
</button>
</div>
</div>
<!-- OracleModule -->
<div class="module-deploy-card">
<div class="module-content">
<h4>OracleModule</h4>
<p>Интеграция с внешними данными - автоматизация на основе IoT, API, датчиков</p>
<div class="module-features">
<span class="feature-tag">Оракулы</span>
<span class="feature-tag">Автоматизация</span>
<span class="feature-tag">IoT</span>
<span class="feature-tag">API</span>
</div>
</div>
<div class="module-actions">
<button
class="btn btn-primary btn-deploy"
@click="router.push(`/management/modules/deploy/oracle?address=${route.query.address}`)"
>
<i class="fas fa-rocket"></i>
Деплой
</button>
</div>
</div>
<!-- InheritanceModule -->
<div class="module-deploy-card">
<div class="module-content">
<h4>InheritanceModule</h4>
<p>Наследование токенов - автоматическая передача токенов наследникам</p>
<div class="module-features">
<span class="feature-tag">Наследование</span>
<span class="feature-tag">Безопасность</span>
<span class="feature-tag">Юридические</span>
<span class="feature-tag">Автоматизация</span>
</div>
</div>
<div class="module-actions">
<button
class="btn btn-primary btn-deploy"
@click="router.push(`/management/modules/deploy/inheritance?address=${route.query.address}`)"
>
<i class="fas fa-rocket"></i>
Деплой
</button>
</div>
</div>
<!-- VestingModule -->
<div class="module-deploy-card">
<div class="module-content">
<h4>VestingModule</h4>
<p>Вестинг токенов - постепенное разблокирование токенов по расписанию</p>
<div class="module-features">
<span class="feature-tag">Вестинг</span>
<span class="feature-tag">Мотивация</span>
<span class="feature-tag">Удержание</span>
<span class="feature-tag">Расписание</span>
</div>
</div>
<div class="module-actions">
<button
class="btn btn-primary btn-deploy"
@click="router.push(`/management/modules/deploy/vesting?address=${route.query.address}`)"
>
<i class="fas fa-rocket"></i>
Деплой
</button>
</div>
</div>
<!-- StakingModule -->
<div class="module-deploy-card">
<div class="module-content">
<h4>StakingModule</h4>
<p>Стейкинг токенов - заработок на удержании токенов</p>
<div class="module-features">
<span class="feature-tag">Стейкинг</span>
<span class="feature-tag">Доход</span>
<span class="feature-tag">Ликвидность</span>
<span class="feature-tag">APY</span>
</div>
</div>
<div class="module-actions">
<button
class="btn btn-primary btn-deploy"
@click="router.push(`/management/modules/deploy/staking?address=${route.query.address}`)"
>
<i class="fas fa-rocket"></i>
Деплой
</button>
</div>
</div>
<!-- InsuranceModule -->
<div class="module-deploy-card">
<div class="module-content">
<h4>InsuranceModule</h4>
<p>Страхование токенов - защита от рисков и потерь</p>
<div class="module-features">
<span class="feature-tag">Страхование</span>
<span class="feature-tag">Защита</span>
<span class="feature-tag">Риски</span>
<span class="feature-tag">Безопасность</span>
</div>
</div>
<div class="module-actions">
<button
class="btn btn-primary btn-deploy"
@click="router.push(`/management/modules/deploy/insurance?address=${route.query.address}`)"
>
<i class="fas fa-rocket"></i>
Деплой
</button>
</div>
</div>
<!-- ComplianceModule -->
<div class="module-deploy-card">
<div class="module-content">
<h4>ComplianceModule</h4>
<p>Соответствие требованиям - KYC/AML, налоги, аудит</p>
<div class="module-features">
<span class="feature-tag">KYC/AML</span>
<span class="feature-tag">Налоги</span>
<span class="feature-tag">Аудит</span>
<span class="feature-tag">Регуляторы</span>
</div>
</div>
<div class="module-actions">
<button
class="btn btn-primary btn-deploy"
@click="router.push(`/management/modules/deploy/compliance?address=${route.query.address}`)"
>
<i class="fas fa-rocket"></i>
Деплой
</button>
</div>
</div>
<!-- SupplyChainModule -->
<div class="module-deploy-card">
<div class="module-content">
<h4>SupplyChainModule</h4>
<p>Цепочка поставок - отслеживание и токенизация логистики</p>
<div class="module-features">
<span class="feature-tag">Логистика</span>
<span class="feature-tag">Отслеживание</span>
<span class="feature-tag">Качество</span>
<span class="feature-tag">Прозрачность</span>
</div>
</div>
<div class="module-actions">
<button
class="btn btn-primary btn-deploy"
@click="router.push(`/management/modules/deploy/supplychain?address=${route.query.address}`)"
>
<i class="fas fa-rocket"></i>
Деплой
</button>
</div>
</div>
<!-- EventModule -->
<div class="module-deploy-card">
<div class="module-content">
<h4>EventModule</h4>
<p>Событийный модуль - токенизация мероприятий и событий</p>
<div class="module-features">
<span class="feature-tag">События</span>
<span class="feature-tag">NFT-билеты</span>
<span class="feature-tag">Мероприятия</span>
<span class="feature-tag">VR/AR</span>
</div>
</div>
<div class="module-actions">
<button
class="btn btn-primary btn-deploy"
@click="router.push(`/management/modules/deploy/event?address=${route.query.address}`)"
>
<i class="fas fa-rocket"></i>
Деплой
</button>
</div>
</div>
</div>
</div>
<!-- Форма добавления модуля -->
<div class="add-module-form">
<div class="form-header">
@@ -521,6 +847,116 @@ onMounted(() => {
border: 1px solid #dee2e6;
}
/* Блоки для деплоя стандартных модулей */
.standard-modules {
background: #f8f9fa;
border-radius: var(--radius-md);
padding: 20px;
margin-bottom: 30px;
border: 1px solid #e9ecef;
}
.modules-header {
margin-bottom: 20px;
padding-bottom: 15px;
border-bottom: 1px solid #dee2e6;
}
.modules-header h3 {
margin: 0 0 10px 0;
color: var(--color-primary);
}
.modules-header p {
margin: 0 0 15px 0;
color: #666;
}
.module-deploy-card {
display: flex;
flex-direction: column;
padding: 20px;
background: white;
border: 1px solid #e9ecef;
border-radius: var(--radius-md);
margin-bottom: 15px;
transition: all 0.2s;
box-shadow: 0 1px 3px rgba(0, 0, 0, 0.1);
}
.module-deploy-card:hover {
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.15);
transform: translateY(-2px);
}
.module-content {
flex: 1;
margin-bottom: 20px;
}
.module-content h4 {
margin: 0 0 8px 0;
color: var(--color-primary);
font-size: 1.2rem;
font-weight: 600;
}
.module-content p {
margin: 0 0 12px 0;
color: #666;
font-size: 14px;
line-height: 1.4;
}
.module-features {
display: flex;
flex-wrap: wrap;
gap: 6px;
}
.feature-tag {
background: linear-gradient(135deg, #e3f2fd, #bbdefb);
color: #1976d2;
padding: 4px 10px;
border-radius: 12px;
font-size: 11px;
font-weight: 500;
border: 1px solid #90caf9;
}
.module-actions {
display: flex;
justify-content: center;
}
.btn-deploy {
background: linear-gradient(135deg, var(--color-primary), var(--color-primary-dark));
color: white;
padding: 10px 20px;
border: none;
border-radius: var(--radius-md);
cursor: pointer;
font-size: 14px;
font-weight: 500;
transition: all 0.3s;
display: inline-flex;
align-items: center;
gap: 8px;
box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1);
}
.btn-deploy:hover:not(:disabled) {
background: linear-gradient(135deg, var(--color-primary-dark), var(--color-primary));
box-shadow: 0 4px 8px rgba(0, 0, 0, 0.2);
transform: translateY(-1px);
}
.btn-deploy:disabled {
opacity: 0.6;
cursor: not-allowed;
transform: none;
}
/* Форма добавления модуля */
.add-module-form {
background: #f8f9fa;
@@ -770,4 +1206,20 @@ onMounted(() => {
grid-template-columns: 1fr;
}
}
/* Адаптивность для блоков деплоя */
@media (max-width: 768px) {
.module-deploy-card {
padding: 15px;
}
.module-content {
margin-bottom: 15px;
}
.btn-deploy {
width: 100%;
justify-content: center;
}
}
</style>

View File

@@ -22,7 +22,8 @@
<!-- Заголовок -->
<div class="page-header">
<div class="header-content">
<h1>Токены DLE</h1>
<h1>Управление токенами DLE</h1>
<p>Создание предложений для перевода токенов через систему голосования</p>
<div v-if="selectedDle" class="dle-info">
<span class="dle-name">{{ selectedDle.name }} ({{ selectedDle.symbol }})</span>
<span class="dle-address">{{ shortenAddress(selectedDle.dleAddress) }}</span>
@@ -48,10 +49,8 @@
<div class="info-card">
<h3>Ваш баланс</h3>
<p class="token-amount">{{ userBalance }} {{ tokenSymbol }}</p>
</div>
<div class="info-card">
<h3>Кворум</h3>
<p class="token-amount">{{ quorumPercentage }}%</p>
<p v-if="currentUserAddress" class="user-address">{{ shortenAddress(currentUserAddress) }}</p>
<p v-else class="no-wallet">Кошелек не подключен</p>
</div>
<div class="info-card">
<h3>Цена токена</h3>
@@ -60,110 +59,70 @@
</div>
</div>
<!-- Трансфер токенов -->
<!-- Перевод токенов через governance -->
<div class="transfer-section">
<h2>Перевод токенов</h2>
<form @submit.prevent="transferTokens" class="transfer-form">
<div class="form-row">
<div class="form-group">
<label for="recipient">Получатель:</label>
<input
id="recipient"
v-model="transferData.recipient"
type="text"
placeholder="0x..."
required
>
</div>
<div class="form-group">
<label for="amount">Количество токенов:</label>
<input
id="amount"
v-model="transferData.amount"
type="number"
min="0.01"
step="0.01"
placeholder="0.00"
required
>
</div>
</div>
<h2>Перевод токенов через Governance</h2>
<p class="section-description">
Создайте предложение для перевода токенов через систему голосования.
Токены будут переведены от имени DLE после одобрения кворумом.
<strong>Важно:</strong> Перевод через governance будет выполнен во всех поддерживаемых сетях DLE.
</p>
<form @submit.prevent="createTransferProposal" class="transfer-form">
<div class="form-group">
<label for="transferDescription">Описание (опционально):</label>
<label for="proposal-recipient">Адрес получателя:</label>
<input
id="proposal-recipient"
v-model="proposalData.recipient"
type="text"
placeholder="0x..."
required
/>
</div>
<div class="form-group">
<label for="proposal-amount">Количество токенов:</label>
<input
id="proposal-amount"
v-model="proposalData.amount"
type="number"
step="0.000001"
placeholder="0.0"
required
/>
</div>
<div class="form-group">
<label for="proposal-description">Описание предложения:</label>
<textarea
id="transferDescription"
v-model="transferData.description"
placeholder="Укажите причину перевода..."
rows="3"
id="proposal-description"
v-model="proposalData.description"
placeholder="Опишите причину перевода токенов..."
required
></textarea>
</div>
<button type="submit" class="btn-primary" :disabled="isTransferring">
{{ isTransferring ? 'Перевод...' : 'Перевести токены' }}
</button>
</form>
</div>
<!-- Распределение токенов -->
<div class="distribution-section">
<h2>Распределение токенов</h2>
<form @submit.prevent="distributeTokens" class="distribution-form">
<div class="form-group">
<label for="distributionType">Тип распределения:</label>
<select id="distributionType" v-model="distributionData.type" required>
<option value="">Выберите тип</option>
<option value="partners">Партнерам</option>
<option value="employees">Сотрудникам</option>
<option value="investors">Инвесторам</option>
<option value="custom">Пользовательское</option>
</select>
<label for="proposal-duration">Длительность голосования (часы):</label>
<input
id="proposal-duration"
v-model="proposalData.duration"
type="number"
min="1"
max="168"
placeholder="24"
required
/>
</div>
<div class="form-group">
<label>Получатели:</label>
<div class="recipients-list">
<div
v-for="(recipient, index) in distributionData.recipients"
:key="index"
class="recipient-item"
>
<input
v-model="recipient.address"
type="text"
placeholder="Адрес получателя"
required
>
<input
v-model="recipient.amount"
type="number"
placeholder="Количество"
min="0.01"
step="0.01"
required
>
<button
type="button"
@click="removeRecipient(index)"
class="btn-remove"
>
</button>
</div>
</div>
<button
type="button"
@click="addRecipient"
class="btn-secondary"
>
+ Добавить получателя
</button>
</div>
<button type="submit" class="btn-primary" :disabled="isDistributing">
{{ isDistributing ? 'Распределение...' : 'Распределить токены' }}
<button type="submit" class="btn-primary" :disabled="isCreatingProposal">
{{ isCreatingProposal ? 'Создание предложения...' : 'Создать предложение' }}
</button>
<!-- Статус предложения -->
<div v-if="proposalStatus" class="proposal-status">
<p class="status-message">{{ proposalStatus }}</p>
</div>
</form>
</div>
@@ -199,6 +158,8 @@ import { useRouter, useRoute } from 'vue-router';
import BaseLayout from '../../components/BaseLayout.vue';
import { getTokenBalance, getTotalSupply, getTokenHolders } from '../../services/tokensService.js';
import api from '../../api/axios';
import { ethers } from 'ethers';
import { createTransferTokensProposal } from '../../utils/dle-contract.js';
// Определяем props
const props = defineProps({
@@ -225,30 +186,48 @@ const dleAddress = computed(() => {
const selectedDle = ref(null);
const isLoadingDle = ref(false);
// Состояние
const isTransferring = ref(false);
const isDistributing = ref(false);
// Состояние для предложения о переводе токенов через governance
const isCreatingProposal = ref(false);
const proposalStatus = ref('');
// Данные токенов (загружаются из блокчейна)
const tokenSymbol = computed(() => selectedDle.value?.symbol || '');
const totalSupply = computed(() => selectedDle.value?.totalSupply || 0);
const userBalance = computed(() => selectedDle.value?.deployerBalance || 0);
const quorumPercentage = computed(() => selectedDle.value?.quorumPercentage || 0);
const totalSupply = ref(0);
const userBalance = ref(0);
const deployerBalance = ref(0);
const quorumPercentage = ref(0);
const tokenPrice = ref(0);
// Данные трансфера
const transferData = ref({
// Данные для формы
const proposalData = ref({
recipient: '',
amount: '',
description: ''
description: '',
duration: 86400, // 24 часа по умолчанию
governanceChainId: 11155111, // Sepolia по умолчанию
targetChains: [11155111] // Sepolia по умолчанию
});
// Данные распределения
const distributionData = ref({
type: '',
recipients: [
{ address: '', amount: '' }
]
// Получаем адрес текущего пользователя
const currentUserAddress = computed(() => {
console.log('Проверяем identities:', props.identities);
// Получаем адрес из props или из window.ethereum
if (props.identities && props.identities.length > 0) {
const walletIdentity = props.identities.find(id => id.provider === 'wallet');
console.log('Найден wallet identity:', walletIdentity);
if (walletIdentity) {
return walletIdentity.provider_id;
}
}
// Fallback: пытаемся получить из window.ethereum
if (window.ethereum && window.ethereum.selectedAddress) {
console.log('Получаем адрес из window.ethereum:', window.ethereum.selectedAddress);
return window.ethereum.selectedAddress;
}
console.log('Адрес пользователя не найден');
return null;
});
// Держатели токенов (загружаются из блокчейна)
@@ -273,6 +252,9 @@ async function loadDleData() {
selectedDle.value = blockchainData;
console.log('Загружены данные DLE из блокчейна:', blockchainData);
// Загружаем баланс текущего пользователя
await loadUserBalance();
// Загружаем держателей токенов (если есть API)
await loadTokenHolders();
} else {
@@ -285,6 +267,35 @@ async function loadDleData() {
}
}
// Новая функция для загрузки баланса текущего пользователя
async function loadUserBalance() {
if (!currentUserAddress.value || !dleAddress.value) {
userBalance.value = 0;
console.log('Не удается загрузить баланс: нет адреса пользователя или DLE');
return;
}
try {
console.log('Загружаем баланс для пользователя:', currentUserAddress.value);
const response = await api.post('/blockchain/get-token-balance', {
dleAddress: dleAddress.value,
account: currentUserAddress.value
});
if (response.data.success) {
userBalance.value = parseFloat(response.data.data.balance);
console.log('Баланс пользователя загружен:', userBalance.value);
} else {
console.warn('Не удалось загрузить баланс пользователя:', response.data.error);
userBalance.value = 0;
}
} catch (error) {
console.error('Ошибка загрузки баланса пользователя:', error);
userBalance.value = 0;
}
}
async function loadTokenHolders() {
try {
// Здесь можно добавить загрузку держателей токенов из блокчейна
@@ -301,99 +312,132 @@ function shortenAddress(address) {
}
// Методы
const transferTokens = async () => {
if (isTransferring.value) return;
try {
isTransferring.value = true;
// Здесь будет логика трансфера токенов
// console.log('Трансфер токенов:', transferData.value);
// Временная логика
const amount = parseFloat(transferData.value.amount);
if (amount > userBalance.value) {
alert('Недостаточно токенов для перевода');
return;
}
userBalance.value -= amount;
// Сброс формы
transferData.value = {
recipient: '',
amount: '',
description: ''
};
alert('Токены успешно переведены!');
} catch (error) {
// console.error('Ошибка трансфера токенов:', error);
alert('Ошибка при переводе токенов');
} finally {
isTransferring.value = false;
}
};
const distributeTokens = async () => {
if (isDistributing.value) return;
try {
isDistributing.value = true;
// Здесь будет логика распределения токенов
// console.log('Распределение токенов:', distributionData.value);
// Временная логика
const totalAmount = distributionData.value.recipients.reduce((sum, recipient) => {
return sum + parseFloat(recipient.amount || 0);
}, 0);
if (totalAmount > userBalance.value) {
alert('Недостаточно токенов для распределения');
return;
}
userBalance.value -= totalAmount;
// Сброс формы
distributionData.value = {
type: '',
recipients: [{ address: '', amount: '' }]
};
alert('Токены успешно распределены!');
} catch (error) {
// console.error('Ошибка распределения токенов:', error);
alert('Ошибка при распределении токенов');
} finally {
isDistributing.value = false;
}
};
const addRecipient = () => {
distributionData.value.recipients.push({ address: '', amount: '' });
};
const removeRecipient = (index) => {
if (distributionData.value.recipients.length > 1) {
distributionData.value.recipients.splice(index, 1);
}
};
const formatAddress = (address) => {
if (!address) return '';
return address.substring(0, 6) + '...' + address.substring(address.length - 4);
};
// Функция создания предложения о переводе токенов через governance
const createTransferProposal = async () => {
if (isCreatingProposal.value) return;
try {
// Проверяем подключение к кошельку
if (!window.ethereum) {
alert('Пожалуйста, установите MetaMask или другой Web3 кошелек');
return;
}
// Проверяем, что пользователь подключен
if (!currentUserAddress.value) {
alert('Пожалуйста, подключите кошелек');
return;
}
// Валидация данных
const recipient = proposalData.value.recipient.trim();
const amount = parseFloat(proposalData.value.amount);
const description = proposalData.value.description.trim();
if (!recipient) {
alert('Пожалуйста, укажите адрес получателя');
return;
}
// Проверяем, что адрес получателя является корректным Ethereum адресом
if (!ethers.isAddress(recipient)) {
alert('Пожалуйста, укажите корректный Ethereum адрес получателя');
return;
}
if (!amount || amount <= 0) {
alert('Пожалуйста, укажите корректное количество токенов');
return;
}
if (!description) {
alert('Пожалуйста, укажите описание предложения');
return;
}
// Проверяем, что получатель не является отправителем
if (recipient.toLowerCase() === currentUserAddress.value.toLowerCase()) {
alert('Нельзя отправить токены самому себе');
return;
}
isCreatingProposal.value = true;
proposalStatus.value = 'Создание предложения...';
// Создаем предложение
const result = await createTransferTokensProposal(dleAddress.value, {
recipient: recipient,
amount: amount,
description: description,
duration: proposalData.value.duration * 3600, // Конвертируем часы в секунды
governanceChainId: proposalData.value.governanceChainId,
targetChains: proposalData.value.targetChains
});
proposalStatus.value = 'Предложение создано!';
console.log('Предложение о переводе токенов создано:', result);
// Сброс формы
proposalData.value = {
recipient: '',
amount: '',
description: '',
duration: 86400,
governanceChainId: 11155111,
targetChains: [11155111]
};
// Очищаем статус через 5 секунд
setTimeout(() => {
proposalStatus.value = '';
}, 5000);
alert(`Предложение о переводе токенов создано!\nID предложения: ${result.proposalId}\nХеш транзакции: ${result.txHash}`);
} catch (error) {
console.error('Ошибка создания предложения о переводе токенов:', error);
// Очищаем статус предложения
proposalStatus.value = '';
let errorMessage = 'Ошибка создания предложения о переводе токенов';
if (error.code === 4001) {
errorMessage = 'Транзакция отменена пользователем';
} else if (error.message && error.message.includes('insufficient funds')) {
errorMessage = 'Недостаточно ETH для оплаты газа';
} else if (error.message && error.message.includes('execution reverted')) {
errorMessage = 'Ошибка выполнения транзакции. Проверьте данные и попробуйте снова';
} else if (error.message) {
errorMessage = error.message;
}
alert(errorMessage);
} finally {
isCreatingProposal.value = false;
}
};
// Отслеживаем изменения в адресе DLE
watch(dleAddress, (newAddress) => {
if (newAddress) {
loadDleData();
}
}, { immediate: true });
// Отслеживаем изменения адреса пользователя
watch(currentUserAddress, (newAddress) => {
if (newAddress && dleAddress.value) {
loadUserBalance();
} else {
userBalance.value = 0;
}
}, { immediate: true });
</script>
<style scoped>
@@ -488,14 +532,12 @@ watch(dleAddress, (newAddress) => {
/* Секции */
.token-info-section,
.transfer-section,
.distribution-section,
.holders-section {
margin-bottom: 40px;
}
.token-info-section h2,
.transfer-section h2,
.distribution-section h2,
.holders-section h2 {
color: var(--color-primary);
margin-bottom: 20px;
@@ -532,9 +574,26 @@ watch(dleAddress, (newAddress) => {
color: var(--color-primary);
}
.user-address {
font-family: monospace;
font-size: 0.75rem;
color: #666;
margin: 5px 0 0 0;
background: #f8f9fa;
padding: 2px 6px;
border-radius: 3px;
display: inline-block;
}
.no-wallet {
font-size: 0.75rem;
color: #dc3545;
margin: 5px 0 0 0;
font-style: italic;
}
/* Формы */
.transfer-form,
.distribution-form {
.transfer-form {
background: #f8f9fa;
padding: 25px;
border-radius: var(--radius-lg);
@@ -574,81 +633,25 @@ watch(dleAddress, (newAddress) => {
min-height: 80px;
}
/* Получатели */
.recipients-list {
display: grid;
gap: 15px;
margin-bottom: 15px;
/* Статус предложения */
.proposal-status {
margin-top: 20px;
padding: 15px;
background: #e8f5e8;
border-radius: var(--radius-sm);
border-left: 4px solid #28a745;
}
.recipient-item {
display: grid;
grid-template-columns: 1fr 1fr auto;
gap: 15px;
align-items: center;
.proposal-status .status-message {
color: #28a745;
}
.btn-remove {
background: #dc3545;
color: white;
border: none;
border-radius: 50%;
width: 30px;
height: 30px;
cursor: pointer;
font-size: 14px;
display: flex;
align-items: center;
justify-content: center;
}
.btn-remove:hover {
background: #c82333;
}
/* Держатели токенов */
.holders-list {
display: grid;
gap: 15px;
}
.holder-item {
display: flex;
justify-content: space-between;
align-items: center;
padding: 20px;
background: white;
border-radius: var(--radius-lg);
border: 1px solid #e9ecef;
transition: all 0.3s ease;
}
.holder-item:hover {
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.1);
}
.holder-info {
display: flex;
flex-direction: column;
gap: 5px;
}
.holder-address {
font-family: monospace;
font-size: 0.9rem;
/* Описание секции */
.section-description {
color: var(--color-grey-dark);
}
.holder-balance {
font-size: 1.1rem;
font-weight: 600;
color: var(--color-primary);
}
.holder-percentage {
font-size: 0.9rem;
color: var(--color-grey-dark);
font-weight: 600;
font-size: 0.95rem;
margin-bottom: 20px;
line-height: 1.5;
}
/* Кнопки */

View File

@@ -0,0 +1,329 @@
<!--
Copyright (c) 2024-2025 Тарабанов Александр Викторович
All rights reserved.
This software is proprietary and confidential.
Unauthorized copying, modification, or distribution is prohibited.
For licensing inquiries: info@hb3-accelerator.com
Website: https://hb3-accelerator.com
GitHub: https://github.com/HB3-ACCELERATOR
-->
<template>
<BaseLayout
:is-authenticated="isAuthenticated"
:identities="identities"
:token-balances="tokenBalances"
:is-loading-tokens="isLoadingTokens"
@auth-action-completed="$emit('auth-action-completed')"
>
<div class="application-module-deploy">
<!-- Заголовок -->
<div class="page-header">
<div class="header-content">
<h1>Деплой ApplicationModule</h1>
<p>Управление вызовом функций приложения через предложения и голосование</p>
<p v-if="dleAddress" class="dle-address">
<strong>DLE:</strong> {{ dleAddress }}
</p>
</div>
<button class="close-btn" @click="router.push('/management/modules')">×</button>
</div>
<!-- Информация о модуле -->
<div class="module-info">
<div class="info-card">
<h3>🖥 ApplicationModule</h3>
<div class="info-grid">
<div class="info-item">
<strong>Назначение:</strong> Управление функциями приложения через DLE
</div>
<div class="info-item">
<strong>Функции:</strong> Создание предложений для вызова API, голосование за операции
</div>
<div class="info-item">
<strong>Безопасность:</strong> Все операции приложения через кворум токен-холдеров
</div>
<div class="info-item">
<strong>Примеры:</strong> Удаление пользователей, изменение настроек, обновление данных
</div>
</div>
</div>
</div>
<!-- Детальное описание -->
<div class="module-details">
<div class="details-card">
<h3>📋 Как работает ApplicationModule</h3>
<div class="details-content">
<div class="detail-step">
<div class="step-number">1</div>
<div class="step-content">
<h4>Создание предложения</h4>
<p>Токен-холдер создает предложение для выполнения операции в приложении (например, удаление пользователя, изменение настроек)</p>
</div>
</div>
<div class="detail-step">
<div class="step-number">2</div>
<div class="step-content">
<h4>Голосование</h4>
<p>Все токен-холдеры голосуют за или против предложения в течение установленного времени</p>
</div>
</div>
<div class="detail-step">
<div class="step-number">3</div>
<div class="step-content">
<h4>Исполнение</h4>
<p>При достижении кворума предложение исполняется - вызывается соответствующая функция приложения</p>
</div>
</div>
<div class="detail-step">
<div class="step-number">4</div>
<div class="step-content">
<h4>Аудит</h4>
<p>Все операции логируются в блокчейне для полной прозрачности и подотчетности</p>
</div>
</div>
</div>
</div>
</div>
<!-- Форма деплоя будет добавлена позже -->
<div class="deploy-form-placeholder">
<div class="placeholder-content">
<h3>🚧 Форма деплоя в разработке</h3>
<p>Здесь будет форма для деплоя ApplicationModule</p>
</div>
</div>
</div>
</BaseLayout>
</template>
<script setup>
import { defineProps, defineEmits, ref, onMounted } from 'vue';
import { useRouter, useRoute } from 'vue-router';
import BaseLayout from '../../../components/BaseLayout.vue';
// Определяем props
const props = defineProps({
isAuthenticated: { type: Boolean, default: false },
identities: { type: Array, default: () => [] },
tokenBalances: { type: Object, default: () => ({}) },
isLoadingTokens: { type: Boolean, default: false }
});
// Определяем emits
const emit = defineEmits(['auth-action-completed']);
const router = useRouter();
const route = useRoute();
// Состояние
const isLoading = ref(false);
const dleAddress = ref(route.query.address || null);
// Инициализация
onMounted(() => {
console.log('[ApplicationModuleDeployView] Страница загружена');
});
</script>
<style scoped>
.application-module-deploy {
padding: 20px;
background-color: var(--color-white);
border-radius: var(--radius-lg);
box-shadow: 0 2px 10px rgba(0, 0, 0, 0.1);
margin-top: 20px;
margin-bottom: 20px;
}
.page-header {
display: flex;
justify-content: space-between;
align-items: center;
margin-bottom: 30px;
padding-bottom: 20px;
border-bottom: 2px solid #f0f0f0;
}
.page-header h1 {
color: var(--color-primary);
font-size: 2rem;
margin: 0;
}
.page-header p {
margin: 10px 0 0 0;
color: #666;
}
.dle-address {
margin-top: 10px !important;
font-family: monospace;
background: #f8f9fa;
padding: 8px 12px;
border-radius: var(--radius-sm);
border: 1px solid #dee2e6;
}
.close-btn {
background: none;
border: none;
font-size: 1.5rem;
cursor: pointer;
color: #666;
padding: 0;
width: 30px;
height: 30px;
display: flex;
align-items: center;
justify-content: center;
border-radius: 50%;
transition: all 0.2s;
}
.close-btn:hover {
background: #f0f0f0;
color: #333;
}
/* Информация о модуле */
.module-info {
margin-bottom: 30px;
}
.info-card {
background: #f8f9fa;
border-radius: var(--radius-md);
padding: 20px;
border: 1px solid #e9ecef;
}
.info-card h3 {
margin: 0 0 15px 0;
color: var(--color-primary);
}
.info-grid {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(250px, 1fr));
gap: 15px;
}
.info-item {
padding: 10px;
background: white;
border-radius: var(--radius-sm);
border: 1px solid #dee2e6;
}
.info-item strong {
color: var(--color-primary);
}
/* Детальное описание */
.module-details {
margin-bottom: 30px;
}
.details-card {
background: #f8f9fa;
border-radius: var(--radius-md);
padding: 20px;
border: 1px solid #e9ecef;
}
.details-card h3 {
margin: 0 0 20px 0;
color: var(--color-primary);
}
.details-content {
display: flex;
flex-direction: column;
gap: 20px;
}
.detail-step {
display: flex;
align-items: flex-start;
gap: 15px;
padding: 15px;
background: white;
border-radius: var(--radius-sm);
border: 1px solid #dee2e6;
}
.step-number {
background: var(--color-primary);
color: white;
width: 30px;
height: 30px;
border-radius: 50%;
display: flex;
align-items: center;
justify-content: center;
font-weight: bold;
font-size: 14px;
flex-shrink: 0;
}
.step-content h4 {
margin: 0 0 8px 0;
color: var(--color-primary);
font-size: 16px;
}
.step-content p {
margin: 0;
color: #666;
line-height: 1.5;
}
/* Плейсхолдер для формы */
.deploy-form-placeholder {
background: #f8f9fa;
border-radius: var(--radius-md);
padding: 40px;
text-align: center;
border: 2px dashed #dee2e6;
}
.placeholder-content h3 {
color: var(--color-primary);
margin-bottom: 10px;
}
.placeholder-content p {
color: #666;
margin: 0;
}
/* Адаптивность */
@media (max-width: 768px) {
.info-grid {
grid-template-columns: 1fr;
}
.page-header {
flex-direction: column;
align-items: flex-start;
gap: 15px;
}
.detail-step {
flex-direction: column;
text-align: center;
}
.step-number {
align-self: center;
}
}
</style>

View File

@@ -0,0 +1,515 @@
<!--
Copyright (c) 2024-2025 Тарабанов Александр Викторович
All rights reserved.
This software is proprietary and confidential.
Unauthorized copying, modification, or distribution is prohibited.
For licensing inquiries: info@hb3-accelerator.com
Website: https://hb3-accelerator.com
GitHub: https://github.com/HB3-ACCELERATOR
-->
<template>
<BaseLayout
:is-authenticated="isAuthenticated"
:identities="identities"
:token-balances="tokenBalances"
:is-loading-tokens="isLoadingTokens"
@auth-action-completed="$emit('auth-action-completed')"
>
<div class="module-deploy-page">
<!-- Заголовок -->
<div class="page-header">
<div class="header-content">
<h1>🔥 Деплой BurnModule</h1>
<p>Модуль для сжигания токенов DLE через governance</p>
<div v-if="selectedDle" class="dle-info">
<span class="dle-name">{{ selectedDle.name }} ({{ selectedDle.symbol }})</span>
<span class="dle-address">{{ selectedDle.dleAddress }}</span>
</div>
</div>
<button class="close-btn" @click="router.push(`/management/modules?address=${route.query.address}`)">×</button>
</div>
<!-- Описание модуля -->
<div class="module-description">
<div class="description-card">
<h3>📋 Описание BurnModule</h3>
<div class="description-content">
<p><strong>BurnModule</strong> - это модуль для управления сжиганием токенов DLE через систему governance.</p>
<h4>🔧 Функциональность:</h4>
<ul>
<li><strong>Сжигание токенов:</strong> Уменьшение общего предложения токенов DLE</li>
<li><strong>Governance:</strong> Все операции требуют голосования и кворума</li>
<li><strong>Безопасность:</strong> Контролируемое сжигание через коллективные решения</li>
<li><strong>Прозрачность:</strong> Все операции записываются в блокчейн</li>
</ul>
<h4> Важные особенности:</h4>
<ul>
<li>Сжигание токенов возможно только через предложения и голосование</li>
<li>Можно сжигать токены из казны DLE или от имени участников</li>
<li>Все операции требуют достижения кворума</li>
<li>История всех сжиганий сохраняется в блокчейне</li>
<li>Сжигание необратимо - токены уничтожаются навсегда</li>
</ul>
</div>
</div>
</div>
<!-- Форма деплоя -->
<div class="deploy-form-section">
<div class="form-header">
<h3> Настройки деплоя</h3>
<p>Настройте параметры для деплоя BurnModule</p>
</div>
<form @submit.prevent="deployModule" class="deploy-form">
<div class="form-row">
<div class="form-group">
<label for="moduleName">Название модуля:</label>
<input
id="moduleName"
v-model="deployData.moduleName"
type="text"
placeholder="BurnModule"
required
/>
</div>
<div class="form-group">
<label for="moduleVersion">Версия модуля:</label>
<input
id="moduleVersion"
v-model="deployData.moduleVersion"
type="text"
placeholder="1.0.0"
required
/>
</div>
</div>
<div class="form-group">
<label for="moduleDescription">Описание модуля:</label>
<textarea
id="moduleDescription"
v-model="deployData.moduleDescription"
placeholder="Модуль для сжигания токенов DLE через governance..."
rows="3"
required
></textarea>
</div>
<div class="form-group">
<label for="maxBurnPerProposal">Максимальное сжигание за одно предложение:</label>
<input
id="maxBurnPerProposal"
v-model="deployData.maxBurnPerProposal"
type="number"
min="1"
step="1"
placeholder="1000000"
required
/>
<small class="form-help">Максимальное количество токенов, которое можно сжечь за одно предложение</small>
</div>
<div class="form-group">
<label for="burnCooldown">Кулдаун между сжиганиями (часы):</label>
<input
id="burnCooldown"
v-model="deployData.burnCooldown"
type="number"
min="0"
step="1"
placeholder="24"
required
/>
<small class="form-help">Минимальное время между успешными сжиганиями токенов</small>
</div>
<div class="form-group">
<label for="allowTreasuryBurn">Разрешить сжигание из казны:</label>
<select
id="allowTreasuryBurn"
v-model="deployData.allowTreasuryBurn"
required
>
<option value="true">Да</option>
<option value="false">Нет</option>
</select>
<small class="form-help">Разрешить сжигание токенов из казны DLE</small>
</div>
<div class="form-group">
<label for="allowUserBurn">Разрешить сжигание от участников:</label>
<select
id="allowUserBurn"
v-model="deployData.allowUserBurn"
required
>
<option value="true">Да</option>
<option value="false">Нет</option>
</select>
<small class="form-help">Разрешить сжигание токенов от имени участников DLE</small>
</div>
<div class="form-group">
<label for="deployDescription">Описание предложения для деплоя:</label>
<textarea
id="deployDescription"
v-model="deployData.deployDescription"
placeholder="Предложение о деплое BurnModule для управления сжиганием токенов DLE..."
rows="3"
required
></textarea>
</div>
<div class="form-group">
<label for="votingDuration">Длительность голосования (часы):</label>
<input
id="votingDuration"
v-model="deployData.votingDuration"
type="number"
min="1"
max="168"
placeholder="24"
required
/>
<small class="form-help">Время для голосования по предложению деплоя (1-168 часов)</small>
</div>
<button type="submit" class="btn-primary" :disabled="isDeploying">
{{ isDeploying ? 'Деплой...' : 'Деплой BurnModule' }}
</button>
<!-- Статус деплоя -->
<div v-if="deployStatus" class="deploy-status">
<p class="status-message">{{ deployStatus }}</p>
</div>
</form>
</div>
</div>
</BaseLayout>
</template>
<script setup>
import { ref, computed, onMounted } from 'vue';
import { useRoute, useRouter } from 'vue-router';
import BaseLayout from '../../../components/BaseLayout.vue';
import api from '../../../api/axios';
// Props
const props = defineProps({
isAuthenticated: {
type: Boolean,
default: false
},
identities: {
type: Array,
default: () => []
},
tokenBalances: {
type: Object,
default: () => ({})
},
isLoadingTokens: {
type: Boolean,
default: false
}
});
// Emits
const emit = defineEmits(['auth-action-completed']);
// Router
const route = useRoute();
const router = useRouter();
// Состояние
const isDeploying = ref(false);
const deployStatus = ref('');
const selectedDle = ref(null);
const isLoadingDle = ref(false);
// Данные для деплоя
const deployData = ref({
moduleName: 'BurnModule',
moduleVersion: '1.0.0',
moduleDescription: 'Модуль для сжигания токенов DLE через governance',
maxBurnPerProposal: 1000000,
burnCooldown: 24,
allowTreasuryBurn: 'true',
allowUserBurn: 'true',
deployDescription: 'Предложение о деплое BurnModule для управления сжиганием токенов DLE',
votingDuration: 24
});
// Получаем адрес DLE из URL
const dleAddress = computed(() => route.query.address);
// Загрузка данных DLE
const loadDleData = async () => {
if (!dleAddress.value) return;
try {
isLoadingDle.value = true;
const response = await api.post('/blockchain/read-dle-info', {
dleAddress: dleAddress.value
});
if (response.data.success) {
selectedDle.value = response.data.data;
}
} catch (error) {
console.error('Ошибка загрузки данных DLE:', error);
} finally {
isLoadingDle.value = false;
}
};
// Функция деплоя модуля
const deployModule = async () => {
if (isDeploying.value) return;
try {
isDeploying.value = true;
deployStatus.value = 'Подготовка к деплою...';
// Здесь будет логика деплоя модуля
console.log('Деплой BurnModule:', deployData.value);
// Временная заглушка
await new Promise(resolve => setTimeout(resolve, 2000));
deployStatus.value = 'Модуль успешно развернут!';
// Очищаем статус через 3 секунды
setTimeout(() => {
deployStatus.value = '';
}, 3000);
alert('BurnModule успешно развернут!');
} catch (error) {
console.error('Ошибка деплоя модуля:', error);
deployStatus.value = 'Ошибка деплоя модуля';
setTimeout(() => {
deployStatus.value = '';
}, 3000);
alert('Ошибка при деплое модуля');
} finally {
isDeploying.value = false;
}
};
// Загружаем данные при монтировании
onMounted(() => {
loadDleData();
});
</script>
<style scoped>
.module-deploy-page {
padding: 20px;
max-width: 1200px;
margin: 0 auto;
}
.page-header {
display: flex;
justify-content: space-between;
align-items: flex-start;
margin-bottom: 30px;
padding-bottom: 20px;
border-bottom: 2px solid #f0f0f0;
}
.header-content h1 {
color: var(--color-primary);
font-size: 2.5rem;
margin: 0 0 10px 0;
}
.header-content p {
color: var(--color-grey-dark);
font-size: 1.1rem;
margin: 0 0 15px 0;
}
.dle-info {
display: flex;
gap: 15px;
align-items: center;
}
.dle-name {
font-weight: 600;
color: var(--color-primary);
}
.dle-address {
font-family: monospace;
color: var(--color-grey-dark);
font-size: 0.9rem;
}
.close-btn {
background: none;
border: none;
font-size: 2rem;
color: var(--color-grey-dark);
cursor: pointer;
padding: 5px;
}
.close-btn:hover {
color: var(--color-primary);
}
.module-description {
margin-bottom: 30px;
}
.description-card {
background: #f8f9fa;
padding: 25px;
border-radius: var(--radius-lg);
border: 1px solid #e9ecef;
}
.description-card h3 {
color: var(--color-primary);
margin: 0 0 20px 0;
}
.description-content h4 {
color: var(--color-grey-dark);
margin: 20px 0 10px 0;
}
.description-content ul {
margin: 10px 0;
padding-left: 20px;
}
.description-content li {
margin: 5px 0;
line-height: 1.5;
}
.deploy-form-section {
background: white;
padding: 30px;
border-radius: var(--radius-lg);
box-shadow: 0 2px 10px rgba(0, 0, 0, 0.1);
}
.form-header {
margin-bottom: 25px;
}
.form-header h3 {
color: var(--color-primary);
margin: 0 0 10px 0;
}
.form-header p {
color: var(--color-grey-dark);
margin: 0;
}
.deploy-form {
display: flex;
flex-direction: column;
gap: 20px;
}
.form-row {
display: grid;
grid-template-columns: 1fr 1fr;
gap: 20px;
}
.form-group {
display: flex;
flex-direction: column;
gap: 8px;
}
.form-group label {
font-weight: 600;
color: var(--color-grey-dark);
}
.form-group input,
.form-group select,
.form-group textarea {
padding: 12px;
border: 1px solid #ddd;
border-radius: var(--radius-sm);
font-size: 1rem;
}
.form-group textarea {
resize: vertical;
min-height: 80px;
}
.form-help {
font-size: 0.85rem;
color: var(--color-grey-dark);
font-style: italic;
}
.btn-primary {
background: var(--color-primary);
color: white;
border: none;
padding: 15px 30px;
border-radius: var(--radius-sm);
font-size: 1.1rem;
font-weight: 600;
cursor: pointer;
transition: background-color 0.3s;
}
.btn-primary:hover:not(:disabled) {
background: var(--color-primary-dark);
}
.btn-primary:disabled {
opacity: 0.6;
cursor: not-allowed;
}
.deploy-status {
margin-top: 20px;
padding: 15px;
background: #e8f5e8;
border-radius: var(--radius-sm);
border-left: 4px solid #28a745;
}
.status-message {
margin: 0;
font-weight: 600;
color: #28a745;
}
@media (max-width: 768px) {
.form-row {
grid-template-columns: 1fr;
}
.dle-info {
flex-direction: column;
align-items: flex-start;
gap: 5px;
}
}
</style>

View File

@@ -0,0 +1,218 @@
<!--
Copyright (c) 2024-2025 Тарабанов Александр Викторович
All rights reserved.
This software is proprietary and confidential.
Unauthorized copying, modification, or distribution is prohibited.
For licensing inquiries: info@hb3-accelerator.com
Website: https://hb3-accelerator.com
GitHub: https://github.com/HB3-ACCELERATOR
-->
<template>
<BaseLayout
:is-authenticated="isAuthenticated"
:identities="identities"
:token-balances="tokenBalances"
:is-loading-tokens="isLoadingTokens"
@auth-action-completed="$emit('auth-action-completed')"
>
<div class="communication-module-deploy">
<!-- Заголовок -->
<div class="page-header">
<div class="header-content">
<h1>Деплой CommunicationModule</h1>
<p>Коммуникации - сообщения, звонки, история общения между участниками</p>
<p v-if="dleAddress" class="dle-address">
<strong>DLE:</strong> {{ dleAddress }}
</p>
</div>
<button class="close-btn" @click="router.push('/management/modules')">×</button>
</div>
<!-- Информация о модуле -->
<div class="module-info">
<div class="info-card">
<h3>💬 CommunicationModule</h3>
<div class="info-grid">
<div class="info-item">
<strong>Назначение:</strong> Коммуникации между участниками DLE
</div>
<div class="info-item">
<strong>Функции:</strong> Сообщения, аудио/видео звонки, история общения
</div>
<div class="info-item">
<strong>Безопасность:</strong> Кворум для коммуникационных операций
</div>
</div>
</div>
</div>
<!-- Форма деплоя будет добавлена позже -->
<div class="deploy-form-placeholder">
<div class="placeholder-content">
<h3>🚧 Форма деплоя в разработке</h3>
<p>Здесь будет форма для деплоя CommunicationModule</p>
</div>
</div>
</div>
</BaseLayout>
</template>
<script setup>
import { defineProps, defineEmits, ref, onMounted } from 'vue';
import { useRouter, useRoute } from 'vue-router';
import BaseLayout from '../../../components/BaseLayout.vue';
// Определяем props
const props = defineProps({
isAuthenticated: { type: Boolean, default: false },
identities: { type: Array, default: () => [] },
tokenBalances: { type: Object, default: () => ({}) },
isLoadingTokens: { type: Boolean, default: false }
});
// Определяем emits
const emit = defineEmits(['auth-action-completed']);
const router = useRouter();
const route = useRoute();
// Состояние
const isLoading = ref(false);
const dleAddress = ref(route.query.address || null);
// Инициализация
onMounted(() => {
console.log('[CommunicationModuleDeployView] Страница загружена');
});
</script>
<style scoped>
.communication-module-deploy {
padding: 20px;
background-color: var(--color-white);
border-radius: var(--radius-lg);
box-shadow: 0 2px 10px rgba(0, 0, 0, 0.1);
margin-top: 20px;
margin-bottom: 20px;
}
.page-header {
display: flex;
justify-content: space-between;
align-items: center;
margin-bottom: 30px;
padding-bottom: 20px;
border-bottom: 2px solid #f0f0f0;
}
.page-header h1 {
color: var(--color-primary);
font-size: 2rem;
margin: 0;
}
.page-header p {
margin: 10px 0 0 0;
color: #666;
}
.dle-address {
margin-top: 10px !important;
font-family: monospace;
background: #f8f9fa;
padding: 8px 12px;
border-radius: var(--radius-sm);
border: 1px solid #dee2e6;
}
.close-btn {
background: none;
border: none;
font-size: 1.5rem;
cursor: pointer;
color: #666;
padding: 0;
width: 30px;
height: 30px;
display: flex;
align-items: center;
justify-content: center;
border-radius: 50%;
transition: all 0.2s;
}
.close-btn:hover {
background: #f0f0f0;
color: #333;
}
/* Информация о модуле */
.module-info {
margin-bottom: 30px;
}
.info-card {
background: #f8f9fa;
border-radius: var(--radius-md);
padding: 20px;
border: 1px solid #e9ecef;
}
.info-card h3 {
margin: 0 0 15px 0;
color: var(--color-primary);
}
.info-grid {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(250px, 1fr));
gap: 15px;
}
.info-item {
padding: 10px;
background: white;
border-radius: var(--radius-sm);
border: 1px solid #dee2e6;
}
.info-item strong {
color: var(--color-primary);
}
/* Плейсхолдер для формы */
.deploy-form-placeholder {
background: #f8f9fa;
border-radius: var(--radius-md);
padding: 40px;
text-align: center;
border: 2px dashed #dee2e6;
}
.placeholder-content h3 {
color: var(--color-primary);
margin-bottom: 10px;
}
.placeholder-content p {
color: #666;
margin: 0;
}
/* Адаптивность */
@media (max-width: 768px) {
.info-grid {
grid-template-columns: 1fr;
}
.page-header {
flex-direction: column;
align-items: flex-start;
gap: 15px;
}
}
</style>

View File

@@ -0,0 +1,663 @@
<!--
Copyright (c) 2024-2025 Тарабанов Александр Викторович
All rights reserved.
This software is proprietary and confidential.
Unauthorized copying, modification, or distribution is prohibited.
For licensing inquiries: info@hb3-accelerator.com
Website: https://hb3-accelerator.com
GitHub: https://github.com/HB3-ACCELERATOR
-->
<template>
<BaseLayout
:is-authenticated="isAuthenticated"
:identities="identities"
:token-balances="tokenBalances"
:is-loading-tokens="isLoadingTokens"
@auth-action-completed="$emit('auth-action-completed')"
>
<div class="module-deploy-page">
<!-- Заголовок -->
<div class="page-header">
<div class="header-content">
<h1>🏛 Деплой InheritanceModule</h1>
<p>Модуль наследования токенов DLE - защита активов и автоматическая передача наследникам</p>
<div v-if="selectedDle" class="dle-info">
<span class="dle-name">{{ selectedDle.name }} ({{ selectedDle.symbol }})</span>
<span class="dle-address">{{ selectedDle.dleAddress }}</span>
</div>
</div>
<button class="close-btn" @click="router.push(`/management/modules?address=${route.query.address}`)">×</button>
</div>
<!-- Описание модуля -->
<div class="module-description">
<div class="description-card">
<h3>📋 Описание InheritanceModule</h3>
<div class="description-content">
<p><strong>InheritanceModule</strong> - это модуль для автоматической передачи токенов DLE наследникам в случае смерти или недееспособности токенхолдера.</p>
<h4>🔧 Основная функциональность:</h4>
<ul>
<li><strong>Назначение наследников:</strong> Токенхолдеры могут указать один или несколько наследников</li>
<li><strong>Распределение долей:</strong> Настройка процентного распределения токенов между наследниками</li>
<li><strong>Условия активации:</strong> Настройка условий для передачи токенов (смерть, недееспособность)</li>
<li><strong>Временные ограничения:</strong> Установка минимального периода владения токенами</li>
<li><strong>Множественные наследники:</strong> Поддержка сложных схем наследования</li>
<li><strong>Отзыв и изменение:</strong> Возможность изменения наследников в любое время</li>
</ul>
<h4>🏛 Юридические аспекты:</h4>
<ul>
<li><strong>Соответствие законам:</strong> Интеграция с юридическими системами наследования</li>
<li><strong>Документооборот:</strong> Автоматическое создание юридических документов</li>
<li><strong>Подтверждение смерти:</strong> Интеграция с государственными реестрами</li>
<li><strong>Споры и оспаривание:</strong> Механизмы разрешения споров о наследстве</li>
<li><strong>Налоговые обязательства:</strong> Автоматический расчет налогов на наследство</li>
</ul>
<h4>🔐 Безопасность и контроль:</h4>
<ul>
<li>Все изменения наследников требуют подтверждения через governance</li>
<li>Криптографическая защита данных о наследниках</li>
<li>Аудит всех операций наследования</li>
<li>Возможность экстренной блокировки в случае споров</li>
<li>Интеграция с системой идентификации для подтверждения личности</li>
</ul>
</div>
</div>
</div>
<!-- Архитектура модуля -->
<div class="module-architecture">
<div class="architecture-card">
<h3>🏗 Архитектура InheritanceModule</h3>
<div class="architecture-content">
<div class="architecture-diagram">
<div class="diagram-row">
<div class="diagram-item tokenholder">
<h5>👤 Токенхолдер</h5>
<ul>
<li>Назначает наследников</li>
<li>Устанавливает доли</li>
<li>Управляет условиями</li>
<li>Может отозвать</li>
</ul>
</div>
<div class="diagram-arrow"></div>
<div class="diagram-item inheritance">
<h5>🏛 InheritanceModule</h5>
<ul>
<li>Хранит данные наследников</li>
<li>Проверяет условия</li>
<li>Выполняет передачу</li>
<li>Ведет аудит</li>
</ul>
</div>
<div class="diagram-arrow"></div>
<div class="diagram-item heirs">
<h5>👥 Наследники</h5>
<ul>
<li>Получают токены</li>
<li>Подтверждают получение</li>
<li>Управляют наследством</li>
<li>Планируют налоги</li>
</ul>
</div>
</div>
</div>
</div>
</div>
</div>
<!-- Типы наследования -->
<div class="inheritance-types">
<div class="types-card">
<h3>📊 Типы наследования</h3>
<div class="types-grid">
<div class="type-item">
<h4>👨👩👧👦 Семейное наследование</h4>
<p>Передача токенов членам семьи согласно традиционным схемам</p>
<ul>
<li>Супруг/супруга (50%)</li>
<li>Дети (равные доли)</li>
<li>Родители (при отсутствии детей)</li>
<li>Братья/сестры (при отсутствии родителей)</li>
</ul>
</div>
<div class="type-item">
<h4>🏢 Корпоративное наследование</h4>
<p>Передача токенов в рамках бизнес-структур и организаций</p>
<ul>
<li>Партнеры по бизнесу</li>
<li>Ключевые сотрудники</li>
<li>Дочерние компании</li>
<li>Благотворительные фонды</li>
</ul>
</div>
<div class="type-item">
<h4>🎯 Целевое наследование</h4>
<p>Передача токенов для достижения конкретных целей</p>
<ul>
<li>Образовательные учреждения</li>
<li>Исследовательские проекты</li>
<li>Экологические инициативы</li>
<li>Социальные программы</li>
</ul>
</div>
<div class="type-item">
<h4> Условное наследование</h4>
<p>Передача токенов при выполнении определенных условий</p>
<ul>
<li>Достижение определенного возраста</li>
<li>Завершение образования</li>
<li>Создание семьи</li>
<li>Достижение карьерных целей</li>
</ul>
</div>
</div>
</div>
</div>
<!-- Примеры использования -->
<div class="usage-examples">
<div class="examples-card">
<h3>💡 Примеры использования</h3>
<div class="examples-content">
<div class="example-item">
<h4>👨👩👧👦 Семейное планирование</h4>
<div class="example-code">
<pre><code>// Назначение наследников для семьи
function setFamilyInheritance() {
setHeir(spouse, 50); // Супруг 50%
setHeir(son, 25); // Сын 25%
setHeir(daughter, 25); // Дочь 25%
setActivationCondition("death");
}</code></pre>
</div>
</div>
<div class="example-item">
<h4>🏢 Бизнес-преемственность</h4>
<div class="example-code">
<pre><code>// Передача бизнеса партнеру
function setBusinessInheritance() {
setHeir(businessPartner, 100); // Партнер 100%
setActivationCondition("death");
setTimeLock(365 days); // Минимум 1 год владения
}</code></pre>
</div>
</div>
<div class="example-item">
<h4>🎯 Благотворительное наследование</h4>
<div class="example-code">
<pre><code>// Передача в благотворительный фонд
function setCharityInheritance() {
setHeir(environmentalFund, 70); // Экологический фонд 70%
setHeir(educationFund, 30); // Образовательный фонд 30%
setActivationCondition("death");
}</code></pre>
</div>
</div>
</div>
</div>
</div>
<!-- Юридические аспекты -->
<div class="legal-aspects">
<div class="legal-card">
<h3> Юридические аспекты</h3>
<div class="legal-content">
<div class="legal-section">
<h4>📜 Соответствие законодательству</h4>
<ul>
<li><strong>Гражданский кодекс:</strong> Соответствие нормам наследования</li>
<li><strong>Налоговый кодекс:</strong> Правильный расчет налогов на наследство</li>
<li><strong>Семейный кодекс:</strong> Учет семейных обязательств</li>
<li><strong>Международное право:</strong> Наследование в разных юрисдикциях</li>
</ul>
</div>
<div class="legal-section">
<h4>🔍 Процедура подтверждения</h4>
<ul>
<li><strong>Свидетельство о смерти:</strong> Официальное подтверждение</li>
<li><strong>Медицинское заключение:</strong> При недееспособности</li>
<li><strong>Судебное решение:</strong> При спорах о наследстве</li>
<li><strong>Нотариальное заверение:</strong> Документов о наследниках</li>
</ul>
</div>
<div class="legal-section">
<h4>💰 Налоговые обязательства</h4>
<ul>
<li><strong>Налог на наследство:</strong> Автоматический расчет</li>
<li><strong>НДФЛ:</strong> При получении токенов</li>
<li><strong>Отчетность:</strong> Автоматическая подача деклараций</li>
<li><strong>Льготы:</strong> Учет налоговых льгот для наследников</li>
</ul>
</div>
</div>
</div>
</div>
<!-- Статус разработки -->
<div class="development-status">
<div class="status-card">
<h3>🚧 Статус разработки</h3>
<div class="status-content">
<p><strong>InheritanceModule находится в стадии планирования.</strong></p>
<p>Модуль будет включать:</p>
<ul>
<li> Систему назначения наследников</li>
<li> Управление долями и условиями</li>
<li> Интеграцию с юридическими системами</li>
<li> Автоматическую передачу токенов</li>
<li> Налоговые расчеты</li>
<li> Аудит и мониторинг</li>
<li> Разрешение споров</li>
</ul>
<p><em>Модуль будет доступен в следующих обновлениях DLE.</em></p>
</div>
</div>
</div>
</div>
</BaseLayout>
</template>
<script setup>
import { ref, computed, onMounted } from 'vue';
import { useRoute, useRouter } from 'vue-router';
import BaseLayout from '../../../components/BaseLayout.vue';
import api from '../../../api/axios';
// Props
const props = defineProps({
isAuthenticated: {
type: Boolean,
default: false
},
identities: {
type: Array,
default: () => []
},
tokenBalances: {
type: Object,
default: () => ({})
},
isLoadingTokens: {
type: Boolean,
default: false
}
});
// Emits
const emit = defineEmits(['auth-action-completed']);
// Router
const route = useRoute();
const router = useRouter();
// Состояние
const selectedDle = ref(null);
const isLoadingDle = ref(false);
// Получаем адрес DLE из URL
const dleAddress = computed(() => route.query.address);
// Загрузка данных DLE
const loadDleData = async () => {
if (!dleAddress.value) return;
try {
isLoadingDle.value = true;
const response = await api.post('/blockchain/read-dle-info', {
dleAddress: dleAddress.value
});
if (response.data.success) {
selectedDle.value = response.data.data;
}
} catch (error) {
console.error('Ошибка загрузки данных DLE:', error);
} finally {
isLoadingDle.value = false;
}
};
// Загружаем данные при монтировании
onMounted(() => {
loadDleData();
});
</script>
<style scoped>
.module-deploy-page {
padding: 20px;
max-width: 1200px;
margin: 0 auto;
}
.page-header {
display: flex;
justify-content: space-between;
align-items: flex-start;
margin-bottom: 30px;
padding-bottom: 20px;
border-bottom: 2px solid #f0f0f0;
}
.header-content h1 {
color: var(--color-primary);
font-size: 2.5rem;
margin: 0 0 10px 0;
}
.header-content p {
color: var(--color-grey-dark);
font-size: 1.1rem;
margin: 0 0 15px 0;
}
.dle-info {
display: flex;
gap: 15px;
align-items: center;
}
.dle-name {
font-weight: 600;
color: var(--color-primary);
}
.dle-address {
font-family: monospace;
color: var(--color-grey-dark);
font-size: 0.9rem;
}
.close-btn {
background: none;
border: none;
font-size: 2rem;
color: var(--color-grey-dark);
cursor: pointer;
padding: 5px;
}
.close-btn:hover {
color: var(--color-primary);
}
.module-description,
.module-architecture,
.inheritance-types,
.usage-examples,
.legal-aspects,
.development-status {
margin-bottom: 30px;
}
.description-card,
.architecture-card,
.types-card,
.examples-card,
.legal-card,
.status-card {
background: #f8f9fa;
padding: 25px;
border-radius: var(--radius-lg);
border: 1px solid #e9ecef;
}
.description-card h3,
.architecture-card h3,
.types-card h3,
.examples-card h3,
.legal-card h3,
.status-card h3 {
color: var(--color-primary);
margin: 0 0 20px 0;
}
.description-content h4 {
color: var(--color-grey-dark);
margin: 20px 0 10px 0;
}
.description-content ul {
margin: 10px 0;
padding-left: 20px;
}
.description-content li {
margin: 5px 0;
line-height: 1.5;
}
/* Архитектурная диаграмма */
.architecture-diagram {
margin: 20px 0;
}
.diagram-row {
display: flex;
align-items: center;
justify-content: space-between;
gap: 20px;
margin-bottom: 20px;
}
.diagram-item {
flex: 1;
padding: 20px;
border-radius: var(--radius-sm);
text-align: center;
min-height: 150px;
display: flex;
flex-direction: column;
justify-content: center;
}
.diagram-item.tokenholder {
background: #e8f5e8;
border: 2px solid #4caf50;
}
.diagram-item.inheritance {
background: #fff3e0;
border: 2px solid #ff9800;
}
.diagram-item.heirs {
background: #f3e5f5;
border: 2px solid #9c27b0;
}
.diagram-item h5 {
margin: 0 0 15px 0;
font-weight: 600;
}
.diagram-item ul {
margin: 0;
padding: 0;
list-style: none;
font-size: 0.9rem;
}
.diagram-item li {
margin: 5px 0;
}
.diagram-arrow {
font-size: 2rem;
color: var(--color-primary);
font-weight: bold;
}
/* Типы наследования */
.types-grid {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(280px, 1fr));
gap: 20px;
margin-top: 20px;
}
.type-item {
background: white;
padding: 20px;
border-radius: var(--radius-sm);
border: 1px solid #e9ecef;
}
.type-item h4 {
color: var(--color-primary);
margin: 0 0 10px 0;
}
.type-item p {
color: var(--color-grey-dark);
margin: 0 0 15px 0;
font-size: 0.9rem;
}
.type-item ul {
margin: 0;
padding-left: 20px;
font-size: 0.9rem;
}
.type-item li {
margin: 5px 0;
color: var(--color-grey-dark);
}
/* Примеры использования */
.examples-content {
display: flex;
flex-direction: column;
gap: 20px;
}
.example-item {
background: white;
padding: 20px;
border-radius: var(--radius-sm);
border: 1px solid #e9ecef;
}
.example-item h4 {
color: var(--color-primary);
margin: 0 0 15px 0;
}
.example-code {
background: #f8f9fa;
border: 1px solid #e9ecef;
border-radius: var(--radius-sm);
padding: 15px;
overflow-x: auto;
}
.example-code pre {
margin: 0;
font-family: 'Courier New', monospace;
font-size: 0.9rem;
color: #333;
}
.example-code code {
background: none;
padding: 0;
}
/* Юридические аспекты */
.legal-content {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(300px, 1fr));
gap: 20px;
}
.legal-section {
background: white;
padding: 20px;
border-radius: var(--radius-sm);
border: 1px solid #e9ecef;
}
.legal-section h4 {
color: var(--color-primary);
margin: 0 0 15px 0;
}
.legal-section ul {
margin: 0;
padding-left: 20px;
}
.legal-section li {
margin: 8px 0;
line-height: 1.5;
color: var(--color-grey-dark);
}
/* Статус разработки */
.status-content {
background: white;
padding: 20px;
border-radius: var(--radius-sm);
border: 1px solid #e9ecef;
}
.status-content p {
margin: 0 0 15px 0;
line-height: 1.6;
}
.status-content ul {
margin: 15px 0;
padding-left: 20px;
}
.status-content li {
margin: 5px 0;
color: var(--color-grey-dark);
}
.status-content em {
color: var(--color-primary);
font-style: italic;
}
@media (max-width: 768px) {
.diagram-row {
flex-direction: column;
gap: 15px;
}
.diagram-arrow {
transform: rotate(90deg);
}
.types-grid {
grid-template-columns: 1fr;
}
.legal-content {
grid-template-columns: 1fr;
}
.dle-info {
flex-direction: column;
align-items: flex-start;
gap: 5px;
}
}
</style>

View File

@@ -0,0 +1,485 @@
<!--
Copyright (c) 2024-2025 Тарабанов Александр Викторович
All rights reserved.
This software is proprietary and confidential.
Unauthorized copying, modification, or distribution is prohibited.
For licensing inquiries: info@hb3-accelerator.com
Website: https://hb3-accelerator.com
GitHub: https://github.com/HB3-ACCELERATOR
-->
<template>
<BaseLayout
:is-authenticated="isAuthenticated"
:identities="identities"
:token-balances="tokenBalances"
:is-loading-tokens="isLoadingTokens"
@auth-action-completed="$emit('auth-action-completed')"
>
<div class="module-deploy-page">
<!-- Заголовок -->
<div class="page-header">
<div class="header-content">
<h1>🚀 Деплой MintModule</h1>
<p>Модуль для выпуска новых токенов DLE через governance</p>
<div v-if="selectedDle" class="dle-info">
<span class="dle-name">{{ selectedDle.name }} ({{ selectedDle.symbol }})</span>
<span class="dle-address">{{ selectedDle.dleAddress }}</span>
</div>
</div>
<button class="close-btn" @click="router.push(`/management/modules?address=${route.query.address}`)">×</button>
</div>
<!-- Описание модуля -->
<div class="module-description">
<div class="description-card">
<h3>📋 Описание MintModule</h3>
<div class="description-content">
<p><strong>MintModule</strong> - это модуль для управления выпуском новых токенов DLE через систему governance.</p>
<h4>🔧 Функциональность:</h4>
<ul>
<li><strong>Выпуск токенов:</strong> Создание дополнительных токенов DLE</li>
<li><strong>Governance:</strong> Все операции требуют голосования и кворума</li>
<li><strong>Безопасность:</strong> Контролируемый выпуск через коллективные решения</li>
<li><strong>Прозрачность:</strong> Все операции записываются в блокчейн</li>
</ul>
<h4> Важные особенности:</h4>
<ul>
<li>Выпуск токенов возможен только через предложения и голосование</li>
<li>Новые токены могут быть распределены между участниками или добавлены в казну DLE</li>
<li>Все операции требуют достижения кворума</li>
<li>История всех выпусков сохраняется в блокчейне</li>
</ul>
</div>
</div>
</div>
<!-- Форма деплоя -->
<div class="deploy-form-section">
<div class="form-header">
<h3> Настройки деплоя</h3>
<p>Настройте параметры для деплоя MintModule</p>
</div>
<form @submit.prevent="deployModule" class="deploy-form">
<div class="form-row">
<div class="form-group">
<label for="moduleName">Название модуля:</label>
<input
id="moduleName"
v-model="deployData.moduleName"
type="text"
placeholder="MintModule"
required
/>
</div>
<div class="form-group">
<label for="moduleVersion">Версия модуля:</label>
<input
id="moduleVersion"
v-model="deployData.moduleVersion"
type="text"
placeholder="1.0.0"
required
/>
</div>
</div>
<div class="form-group">
<label for="moduleDescription">Описание модуля:</label>
<textarea
id="moduleDescription"
v-model="deployData.moduleDescription"
placeholder="Модуль для выпуска новых токенов DLE через governance..."
rows="3"
required
></textarea>
</div>
<div class="form-group">
<label for="maxMintPerProposal">Максимальный выпуск за одно предложение:</label>
<input
id="maxMintPerProposal"
v-model="deployData.maxMintPerProposal"
type="number"
min="1"
step="1"
placeholder="1000000"
required
/>
<small class="form-help">Максимальное количество токенов, которое можно выпустить за одно предложение</small>
</div>
<div class="form-group">
<label for="mintCooldown">Кулдаун между выпусками (часы):</label>
<input
id="mintCooldown"
v-model="deployData.mintCooldown"
type="number"
min="0"
step="1"
placeholder="24"
required
/>
<small class="form-help">Минимальное время между успешными выпусками токенов</small>
</div>
<div class="form-group">
<label for="deployDescription">Описание предложения для деплоя:</label>
<textarea
id="deployDescription"
v-model="deployData.deployDescription"
placeholder="Предложение о деплое MintModule для управления выпуском токенов DLE..."
rows="3"
required
></textarea>
</div>
<div class="form-group">
<label for="votingDuration">Длительность голосования (часы):</label>
<input
id="votingDuration"
v-model="deployData.votingDuration"
type="number"
min="1"
max="168"
placeholder="24"
required
/>
<small class="form-help">Время для голосования по предложению деплоя (1-168 часов)</small>
</div>
<button type="submit" class="btn-primary" :disabled="isDeploying">
{{ isDeploying ? 'Деплой...' : 'Деплой MintModule' }}
</button>
<!-- Статус деплоя -->
<div v-if="deployStatus" class="deploy-status">
<p class="status-message">{{ deployStatus }}</p>
</div>
</form>
</div>
</div>
</BaseLayout>
</template>
<script setup>
import { ref, computed, onMounted } from 'vue';
import { useRoute, useRouter } from 'vue-router';
import BaseLayout from '../../../components/BaseLayout.vue';
import api from '../../../api/axios';
// Props
const props = defineProps({
isAuthenticated: {
type: Boolean,
default: false
},
identities: {
type: Array,
default: () => []
},
tokenBalances: {
type: Object,
default: () => ({})
},
isLoadingTokens: {
type: Boolean,
default: false
}
});
// Emits
const emit = defineEmits(['auth-action-completed']);
// Router
const route = useRoute();
const router = useRouter();
// Состояние
const isDeploying = ref(false);
const deployStatus = ref('');
const selectedDle = ref(null);
const isLoadingDle = ref(false);
// Данные для деплоя
const deployData = ref({
moduleName: 'MintModule',
moduleVersion: '1.0.0',
moduleDescription: 'Модуль для выпуска новых токенов DLE через governance',
maxMintPerProposal: 1000000,
mintCooldown: 24,
deployDescription: 'Предложение о деплое MintModule для управления выпуском токенов DLE',
votingDuration: 24
});
// Получаем адрес DLE из URL
const dleAddress = computed(() => route.query.address);
// Загрузка данных DLE
const loadDleData = async () => {
if (!dleAddress.value) return;
try {
isLoadingDle.value = true;
const response = await api.post('/blockchain/read-dle-info', {
dleAddress: dleAddress.value
});
if (response.data.success) {
selectedDle.value = response.data.data;
}
} catch (error) {
console.error('Ошибка загрузки данных DLE:', error);
} finally {
isLoadingDle.value = false;
}
};
// Функция деплоя модуля
const deployModule = async () => {
if (isDeploying.value) return;
try {
isDeploying.value = true;
deployStatus.value = 'Подготовка к деплою...';
// Здесь будет логика деплоя модуля
console.log('Деплой MintModule:', deployData.value);
// Временная заглушка
await new Promise(resolve => setTimeout(resolve, 2000));
deployStatus.value = 'Модуль успешно развернут!';
// Очищаем статус через 3 секунды
setTimeout(() => {
deployStatus.value = '';
}, 3000);
alert('MintModule успешно развернут!');
} catch (error) {
console.error('Ошибка деплоя модуля:', error);
deployStatus.value = 'Ошибка деплоя модуля';
setTimeout(() => {
deployStatus.value = '';
}, 3000);
alert('Ошибка при деплое модуля');
} finally {
isDeploying.value = false;
}
};
// Загружаем данные при монтировании
onMounted(() => {
loadDleData();
});
</script>
<style scoped>
.module-deploy-page {
padding: 20px;
max-width: 1200px;
margin: 0 auto;
}
.page-header {
display: flex;
justify-content: space-between;
align-items: flex-start;
margin-bottom: 30px;
padding-bottom: 20px;
border-bottom: 2px solid #f0f0f0;
}
.header-content h1 {
color: var(--color-primary);
font-size: 2.5rem;
margin: 0 0 10px 0;
}
.header-content p {
color: var(--color-grey-dark);
font-size: 1.1rem;
margin: 0 0 15px 0;
}
.dle-info {
display: flex;
gap: 15px;
align-items: center;
}
.dle-name {
font-weight: 600;
color: var(--color-primary);
}
.dle-address {
font-family: monospace;
color: var(--color-grey-dark);
font-size: 0.9rem;
}
.close-btn {
background: none;
border: none;
font-size: 2rem;
color: var(--color-grey-dark);
cursor: pointer;
padding: 5px;
}
.close-btn:hover {
color: var(--color-primary);
}
.module-description {
margin-bottom: 30px;
}
.description-card {
background: #f8f9fa;
padding: 25px;
border-radius: var(--radius-lg);
border: 1px solid #e9ecef;
}
.description-card h3 {
color: var(--color-primary);
margin: 0 0 20px 0;
}
.description-content h4 {
color: var(--color-grey-dark);
margin: 20px 0 10px 0;
}
.description-content ul {
margin: 10px 0;
padding-left: 20px;
}
.description-content li {
margin: 5px 0;
line-height: 1.5;
}
.deploy-form-section {
background: white;
padding: 30px;
border-radius: var(--radius-lg);
box-shadow: 0 2px 10px rgba(0, 0, 0, 0.1);
}
.form-header {
margin-bottom: 25px;
}
.form-header h3 {
color: var(--color-primary);
margin: 0 0 10px 0;
}
.form-header p {
color: var(--color-grey-dark);
margin: 0;
}
.deploy-form {
display: flex;
flex-direction: column;
gap: 20px;
}
.form-row {
display: grid;
grid-template-columns: 1fr 1fr;
gap: 20px;
}
.form-group {
display: flex;
flex-direction: column;
gap: 8px;
}
.form-group label {
font-weight: 600;
color: var(--color-grey-dark);
}
.form-group input,
.form-group textarea {
padding: 12px;
border: 1px solid #ddd;
border-radius: var(--radius-sm);
font-size: 1rem;
}
.form-group textarea {
resize: vertical;
min-height: 80px;
}
.form-help {
font-size: 0.85rem;
color: var(--color-grey-dark);
font-style: italic;
}
.btn-primary {
background: var(--color-primary);
color: white;
border: none;
padding: 15px 30px;
border-radius: var(--radius-sm);
font-size: 1.1rem;
font-weight: 600;
cursor: pointer;
transition: background-color 0.3s;
}
.btn-primary:hover:not(:disabled) {
background: var(--color-primary-dark);
}
.btn-primary:disabled {
opacity: 0.6;
cursor: not-allowed;
}
.deploy-status {
margin-top: 20px;
padding: 15px;
background: #e8f5e8;
border-radius: var(--radius-sm);
border-left: 4px solid #28a745;
}
.status-message {
margin: 0;
font-weight: 600;
color: #28a745;
}
@media (max-width: 768px) {
.form-row {
grid-template-columns: 1fr;
}
.dle-info {
flex-direction: column;
align-items: flex-start;
gap: 5px;
}
}
</style>

View File

@@ -0,0 +1,205 @@
<!--
Copyright (c) 2024-2025 Тарабанов Александр Викторович
All rights reserved.
This software is proprietary and confidential.
Unauthorized copying, modification, or distribution is prohibited.
For licensing inquiries: info@hb3-accelerator.com
Website: https://hb3-accelerator.com
GitHub: https://github.com/HB3-ACCELERATOR
-->
<template>
<BaseLayout
:is-authenticated="isAuthenticated"
:identities="identities"
:token-balances="tokenBalances"
:is-loading-tokens="isLoadingTokens"
@auth-action-completed="$emit('auth-action-completed')"
>
<div class="module-deploy-form">
<!-- Заголовок -->
<div class="page-header">
<div class="header-content">
<h1>Деплой пользовательского модуля</h1>
<p>Создание и деплой кастомного модуля для DLE</p>
</div>
<button class="close-btn" @click="router.push('/management/modules')">×</button>
</div>
<!-- Информация о пользовательских модулях -->
<div class="module-info">
<div class="info-card">
<h3>🔧 Пользовательские модули</h3>
<div class="info-grid">
<div class="info-item">
<strong>Назначение:</strong> Расширение функциональности DLE
</div>
<div class="info-item">
<strong>Возможности:</strong> Любая кастомная логика, интеграции, API
</div>
<div class="info-item">
<strong>Безопасность:</strong> Проверка через голосование токен-холдеров
</div>
</div>
</div>
</div>
<!-- Форма деплоя будет добавлена позже -->
<div class="deploy-form-placeholder">
<div class="placeholder-content">
<h3>🚧 Форма деплоя в разработке</h3>
<p>Здесь будет универсальная форма для деплоя пользовательских модулей</p>
</div>
</div>
</div>
</BaseLayout>
</template>
<script setup>
import { defineProps, defineEmits, ref, onMounted } from 'vue';
import { useRouter, useRoute } from 'vue-router';
import BaseLayout from '../../../components/BaseLayout.vue';
// Определяем props
const props = defineProps({
isAuthenticated: { type: Boolean, default: false },
identities: { type: Array, default: () => [] },
tokenBalances: { type: Object, default: () => ({}) },
isLoadingTokens: { type: Boolean, default: false }
});
// Определяем emits
const emit = defineEmits(['auth-action-completed']);
const router = useRouter();
const route = useRoute();
// Состояние
const isLoading = ref(false);
// Инициализация
onMounted(() => {
console.log('[ModuleDeployFormView] Страница загружена');
});
</script>
<style scoped>
.module-deploy-form {
padding: 20px;
background-color: var(--color-white);
border-radius: var(--radius-lg);
box-shadow: 0 2px 10px rgba(0, 0, 0, 0.1);
margin-top: 20px;
margin-bottom: 20px;
}
.page-header {
display: flex;
justify-content: space-between;
align-items: center;
margin-bottom: 30px;
padding-bottom: 20px;
border-bottom: 2px solid #f0f0f0;
}
.page-header h1 {
color: var(--color-primary);
font-size: 2rem;
margin: 0;
}
.page-header p {
margin: 10px 0 0 0;
color: #666;
}
.close-btn {
background: none;
border: none;
font-size: 1.5rem;
cursor: pointer;
color: #666;
padding: 0;
width: 30px;
height: 30px;
display: flex;
align-items: center;
justify-content: center;
border-radius: 50%;
transition: all 0.2s;
}
.close-btn:hover {
background: #f0f0f0;
color: #333;
}
/* Информация о модуле */
.module-info {
margin-bottom: 30px;
}
.info-card {
background: #f8f9fa;
border-radius: var(--radius-md);
padding: 20px;
border: 1px solid #e9ecef;
}
.info-card h3 {
margin: 0 0 15px 0;
color: var(--color-primary);
}
.info-grid {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(250px, 1fr));
gap: 15px;
}
.info-item {
padding: 10px;
background: white;
border-radius: var(--radius-sm);
border: 1px solid #dee2e6;
}
.info-item strong {
color: var(--color-primary);
}
/* Плейсхолдер для формы */
.deploy-form-placeholder {
background: #f8f9fa;
border-radius: var(--radius-md);
padding: 40px;
text-align: center;
border: 2px dashed #dee2e6;
}
.placeholder-content h3 {
color: var(--color-primary);
margin-bottom: 10px;
}
.placeholder-content p {
color: #666;
margin: 0;
}
/* Адаптивность */
@media (max-width: 768px) {
.info-grid {
grid-template-columns: 1fr;
}
.page-header {
flex-direction: column;
align-items: flex-start;
gap: 15px;
}
}
</style>

View File

@@ -0,0 +1,584 @@
<!--
Copyright (c) 2024-2025 Тарабанов Александр Викторович
All rights reserved.
This software is proprietary and confidential.
Unauthorized copying, modification, or distribution is prohibited.
For licensing inquiries: info@hb3-accelerator.com
Website: https://hb3-accelerator.com
GitHub: https://github.com/HB3-ACCELERATOR
-->
<template>
<BaseLayout
:is-authenticated="isAuthenticated"
:identities="identities"
:token-balances="tokenBalances"
:is-loading-tokens="isLoadingTokens"
@auth-action-completed="$emit('auth-action-completed')"
>
<div class="module-deploy-page">
<!-- Заголовок -->
<div class="page-header">
<div class="header-content">
<h1>🔗 Деплой OracleModule</h1>
<p>Модуль для интеграции с внешними данными и автоматизации DLE</p>
<div v-if="selectedDle" class="dle-info">
<span class="dle-name">{{ selectedDle.name }} ({{ selectedDle.symbol }})</span>
<span class="dle-address">{{ selectedDle.dleAddress }}</span>
</div>
</div>
<button class="close-btn" @click="router.push(`/management/modules?address=${route.query.address}`)">×</button>
</div>
<!-- Описание модуля -->
<div class="module-description">
<div class="description-card">
<h3>📋 Описание OracleModule</h3>
<div class="description-content">
<p><strong>OracleModule</strong> - это модуль для интеграции DLE с внешними источниками данных и автоматизации процессов на основе реальных событий.</p>
<h4>🔧 Основная функциональность:</h4>
<ul>
<li><strong>Интеграция с IoT:</strong> Получение данных от датчиков, производственных линий, оборудования</li>
<li><strong>API интеграция:</strong> Подключение к внешним системам, ERP, CRM, аналитическим платформам</li>
<li><strong>Автоматические триггеры:</strong> Создание предложений на основе внешних событий</li>
<li><strong>Валидация данных:</strong> Проверка достоверности полученной информации</li>
<li><strong>Множественные оракулы:</strong> Подтверждение данных от нескольких источников</li>
</ul>
<h4>🏭 Примеры применения:</h4>
<ul>
<li><strong>Производственные токены:</strong> Автоматический выпуск токенов за завершение партий продукции</li>
<li><strong>Качественные бонусы:</strong> Токены за высокое качество продукции на основе данных датчиков</li>
<li><strong>Экологические токены:</strong> Вознаграждения за снижение энергопотребления и выбросов</li>
<li><strong>Инновационные токены:</strong> Токены за внедрение новых технологий и процессов</li>
<li><strong>Рыночные корректировки:</strong> Автоматическая адаптация стратегии на основе рыночных данных</li>
</ul>
<h4>🔐 Безопасность и контроль:</h4>
<ul>
<li>Все оракулы должны быть авторизованы через governance</li>
<li>Данные валидируются перед обработкой</li>
<li>Критические решения требуют множественного подтверждения</li>
<li>История всех оракульных данных сохраняется в блокчейне</li>
<li>Возможность отключения оракулов в экстренных случаях</li>
</ul>
</div>
</div>
</div>
<!-- Архитектура модуля -->
<div class="module-architecture">
<div class="architecture-card">
<h3>🏗 Архитектура OracleModule</h3>
<div class="architecture-content">
<div class="architecture-diagram">
<div class="diagram-row">
<div class="diagram-item external">
<h5>🌐 Внешние источники</h5>
<ul>
<li>IoT датчики</li>
<li>Производственные линии</li>
<li>ERP/CRM системы</li>
<li>Аналитические платформы</li>
<li>Рыночные данные</li>
</ul>
</div>
<div class="diagram-arrow"></div>
<div class="diagram-item oracle">
<h5>🔗 OracleModule</h5>
<ul>
<li>Получение данных</li>
<li>Валидация</li>
<li>Обработка</li>
<li>Создание предложений</li>
</ul>
</div>
<div class="diagram-arrow"></div>
<div class="diagram-item dle">
<h5>🏛 DLE Governance</h5>
<ul>
<li>Предложения</li>
<li>Голосование</li>
<li>Исполнение</li>
<li>Выпуск токенов</li>
</ul>
</div>
</div>
</div>
</div>
</div>
</div>
<!-- Типы оракулов -->
<div class="oracle-types">
<div class="types-card">
<h3>📊 Типы оракулов</h3>
<div class="types-grid">
<div class="type-item">
<h4>🏭 Производственные оракулы</h4>
<p>Данные от производственных линий, оборудования, датчиков качества</p>
<ul>
<li>Завершение партий продукции</li>
<li>Показатели качества</li>
<li>Энергопотребление</li>
<li>Время простоя</li>
</ul>
</div>
<div class="type-item">
<h4>📈 Рыночные оракулы</h4>
<p>Рыночные данные, цены, спрос, конкуренция</p>
<ul>
<li>Цены на сырье</li>
<li>Спрос на продукцию</li>
<li>Конкурентные данные</li>
<li>Экономические индикаторы</li>
</ul>
</div>
<div class="type-item">
<h4>🌱 Экологические оракулы</h4>
<p>Данные об экологическом воздействии и устойчивости</p>
<ul>
<li>Выбросы CO2</li>
<li>Потребление энергии</li>
<li>Переработка отходов</li>
<li>Использование возобновляемых ресурсов</li>
</ul>
</div>
<div class="type-item">
<h4>💼 Бизнес-оракулы</h4>
<p>Бизнес-метрики, продажи, удовлетворенность клиентов</p>
<ul>
<li>Объемы продаж</li>
<li>Удовлетворенность клиентов</li>
<li>Эффективность процессов</li>
<li>Финансовые показатели</li>
</ul>
</div>
</div>
</div>
</div>
<!-- Примеры использования -->
<div class="usage-examples">
<div class="examples-card">
<h3>💡 Примеры использования</h3>
<div class="examples-content">
<div class="example-item">
<h4>🏭 Автоматические производственные токены</h4>
<div class="example-code">
<pre><code>// При завершении партии продукции
function onBatchCompleted(uint256 quantity, uint256 quality) {
uint256 tokens = quantity * quality * 10;
createMintProposal(tokens, "Production reward");
}</code></pre>
</div>
</div>
<div class="example-item">
<h4>🌱 Экологические бонусы</h4>
<div class="example-code">
<pre><code>// При снижении энергопотребления
function onEnergyReduction(uint256 savedEnergy) {
uint256 tokens = savedEnergy * 5;
createMintProposal(tokens, "Energy efficiency bonus");
}</code></pre>
</div>
</div>
<div class="example-item">
<h4>📈 Рыночная адаптация</h4>
<div class="example-code">
<pre><code>// При изменении рыночных условий
function onMarketChange(uint256 priceChange) {
if (priceChange > 10) {
createStrategyProposal("Increase production");
}
}</code></pre>
</div>
</div>
</div>
</div>
</div>
<!-- Статус разработки -->
<div class="development-status">
<div class="status-card">
<h3>🚧 Статус разработки</h3>
<div class="status-content">
<p><strong>OracleModule находится в стадии планирования.</strong></p>
<p>Модуль будет включать:</p>
<ul>
<li> Систему авторизации оракулов</li>
<li> Валидацию и обработку данных</li>
<li> Интеграцию с IoT и API</li>
<li> Автоматические триггеры</li>
<li> Множественное подтверждение данных</li>
<li> Аудит и мониторинг</li>
</ul>
<p><em>Модуль будет доступен в следующих обновлениях DLE.</em></p>
</div>
</div>
</div>
</div>
</BaseLayout>
</template>
<script setup>
import { ref, computed, onMounted } from 'vue';
import { useRoute, useRouter } from 'vue-router';
import BaseLayout from '../../../components/BaseLayout.vue';
import api from '../../../api/axios';
// Props
const props = defineProps({
isAuthenticated: {
type: Boolean,
default: false
},
identities: {
type: Array,
default: () => []
},
tokenBalances: {
type: Object,
default: () => ({})
},
isLoadingTokens: {
type: Boolean,
default: false
}
});
// Emits
const emit = defineEmits(['auth-action-completed']);
// Router
const route = useRoute();
const router = useRouter();
// Состояние
const selectedDle = ref(null);
const isLoadingDle = ref(false);
// Получаем адрес DLE из URL
const dleAddress = computed(() => route.query.address);
// Загрузка данных DLE
const loadDleData = async () => {
if (!dleAddress.value) return;
try {
isLoadingDle.value = true;
const response = await api.post('/blockchain/read-dle-info', {
dleAddress: dleAddress.value
});
if (response.data.success) {
selectedDle.value = response.data.data;
}
} catch (error) {
console.error('Ошибка загрузки данных DLE:', error);
} finally {
isLoadingDle.value = false;
}
};
// Загружаем данные при монтировании
onMounted(() => {
loadDleData();
});
</script>
<style scoped>
.module-deploy-page {
padding: 20px;
max-width: 1200px;
margin: 0 auto;
}
.page-header {
display: flex;
justify-content: space-between;
align-items: flex-start;
margin-bottom: 30px;
padding-bottom: 20px;
border-bottom: 2px solid #f0f0f0;
}
.header-content h1 {
color: var(--color-primary);
font-size: 2.5rem;
margin: 0 0 10px 0;
}
.header-content p {
color: var(--color-grey-dark);
font-size: 1.1rem;
margin: 0 0 15px 0;
}
.dle-info {
display: flex;
gap: 15px;
align-items: center;
}
.dle-name {
font-weight: 600;
color: var(--color-primary);
}
.dle-address {
font-family: monospace;
color: var(--color-grey-dark);
font-size: 0.9rem;
}
.close-btn {
background: none;
border: none;
font-size: 2rem;
color: var(--color-grey-dark);
cursor: pointer;
padding: 5px;
}
.close-btn:hover {
color: var(--color-primary);
}
.module-description,
.module-architecture,
.oracle-types,
.usage-examples,
.development-status {
margin-bottom: 30px;
}
.description-card,
.architecture-card,
.types-card,
.examples-card,
.status-card {
background: #f8f9fa;
padding: 25px;
border-radius: var(--radius-lg);
border: 1px solid #e9ecef;
}
.description-card h3,
.architecture-card h3,
.types-card h3,
.examples-card h3,
.status-card h3 {
color: var(--color-primary);
margin: 0 0 20px 0;
}
.description-content h4 {
color: var(--color-grey-dark);
margin: 20px 0 10px 0;
}
.description-content ul {
margin: 10px 0;
padding-left: 20px;
}
.description-content li {
margin: 5px 0;
line-height: 1.5;
}
/* Архитектурная диаграмма */
.architecture-diagram {
margin: 20px 0;
}
.diagram-row {
display: flex;
align-items: center;
justify-content: space-between;
gap: 20px;
margin-bottom: 20px;
}
.diagram-item {
flex: 1;
padding: 20px;
border-radius: var(--radius-sm);
text-align: center;
min-height: 150px;
display: flex;
flex-direction: column;
justify-content: center;
}
.diagram-item.external {
background: #e3f2fd;
border: 2px solid #2196f3;
}
.diagram-item.oracle {
background: #f3e5f5;
border: 2px solid #9c27b0;
}
.diagram-item.dle {
background: #e8f5e8;
border: 2px solid #4caf50;
}
.diagram-item h5 {
margin: 0 0 15px 0;
font-weight: 600;
}
.diagram-item ul {
margin: 0;
padding: 0;
list-style: none;
font-size: 0.9rem;
}
.diagram-item li {
margin: 5px 0;
}
.diagram-arrow {
font-size: 2rem;
color: var(--color-primary);
font-weight: bold;
}
/* Типы оракулов */
.types-grid {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(280px, 1fr));
gap: 20px;
margin-top: 20px;
}
.type-item {
background: white;
padding: 20px;
border-radius: var(--radius-sm);
border: 1px solid #e9ecef;
}
.type-item h4 {
color: var(--color-primary);
margin: 0 0 10px 0;
}
.type-item p {
color: var(--color-grey-dark);
margin: 0 0 15px 0;
font-size: 0.9rem;
}
.type-item ul {
margin: 0;
padding-left: 20px;
font-size: 0.9rem;
}
.type-item li {
margin: 5px 0;
color: var(--color-grey-dark);
}
/* Примеры использования */
.examples-content {
display: flex;
flex-direction: column;
gap: 20px;
}
.example-item {
background: white;
padding: 20px;
border-radius: var(--radius-sm);
border: 1px solid #e9ecef;
}
.example-item h4 {
color: var(--color-primary);
margin: 0 0 15px 0;
}
.example-code {
background: #f8f9fa;
border: 1px solid #e9ecef;
border-radius: var(--radius-sm);
padding: 15px;
overflow-x: auto;
}
.example-code pre {
margin: 0;
font-family: 'Courier New', monospace;
font-size: 0.9rem;
color: #333;
}
.example-code code {
background: none;
padding: 0;
}
/* Статус разработки */
.status-content {
background: white;
padding: 20px;
border-radius: var(--radius-sm);
border: 1px solid #e9ecef;
}
.status-content p {
margin: 0 0 15px 0;
line-height: 1.6;
}
.status-content ul {
margin: 15px 0;
padding-left: 20px;
}
.status-content li {
margin: 5px 0;
color: var(--color-grey-dark);
}
.status-content em {
color: var(--color-primary);
font-style: italic;
}
@media (max-width: 768px) {
.diagram-row {
flex-direction: column;
gap: 15px;
}
.diagram-arrow {
transform: rotate(90deg);
}
.types-grid {
grid-template-columns: 1fr;
}
.dle-info {
flex-direction: column;
align-items: flex-start;
gap: 5px;
}
}
</style>

View File

@@ -0,0 +1,218 @@
<!--
Copyright (c) 2024-2025 Тарабанов Александр Викторович
All rights reserved.
This software is proprietary and confidential.
Unauthorized copying, modification, or distribution is prohibited.
For licensing inquiries: info@hb3-accelerator.com
Website: https://hb3-accelerator.com
GitHub: https://github.com/HB3-ACCELERATOR
-->
<template>
<BaseLayout
:is-authenticated="isAuthenticated"
:identities="identities"
:token-balances="tokenBalances"
:is-loading-tokens="isLoadingTokens"
@auth-action-completed="$emit('auth-action-completed')"
>
<div class="timelock-module-deploy">
<!-- Заголовок -->
<div class="page-header">
<div class="header-content">
<h1>Деплой TimelockModule</h1>
<p>Задержки исполнения - безопасность критических операций через таймлоки</p>
<p v-if="dleAddress" class="dle-address">
<strong>DLE:</strong> {{ dleAddress }}
</p>
</div>
<button class="close-btn" @click="router.push('/management/modules')">×</button>
</div>
<!-- Информация о модуле -->
<div class="module-info">
<div class="info-card">
<h3> TimelockModule</h3>
<div class="info-grid">
<div class="info-item">
<strong>Назначение:</strong> Безопасность критических операций
</div>
<div class="info-item">
<strong>Функции:</strong> Настраиваемые таймлоки, отмена предложений, аудит
</div>
<div class="info-item">
<strong>Безопасность:</strong> Задержки исполнения для защиты от атак
</div>
</div>
</div>
</div>
<!-- Форма деплоя будет добавлена позже -->
<div class="deploy-form-placeholder">
<div class="placeholder-content">
<h3>🚧 Форма деплоя в разработке</h3>
<p>Здесь будет форма для деплоя TimelockModule</p>
</div>
</div>
</div>
</BaseLayout>
</template>
<script setup>
import { defineProps, defineEmits, ref, onMounted } from 'vue';
import { useRouter, useRoute } from 'vue-router';
import BaseLayout from '../../../components/BaseLayout.vue';
// Определяем props
const props = defineProps({
isAuthenticated: { type: Boolean, default: false },
identities: { type: Array, default: () => [] },
tokenBalances: { type: Object, default: () => ({}) },
isLoadingTokens: { type: Boolean, default: false }
});
// Определяем emits
const emit = defineEmits(['auth-action-completed']);
const router = useRouter();
const route = useRoute();
// Состояние
const isLoading = ref(false);
const dleAddress = ref(route.query.address || null);
// Инициализация
onMounted(() => {
console.log('[TimelockModuleDeployView] Страница загружена');
});
</script>
<style scoped>
.timelock-module-deploy {
padding: 20px;
background-color: var(--color-white);
border-radius: var(--radius-lg);
box-shadow: 0 2px 10px rgba(0, 0, 0, 0.1);
margin-top: 20px;
margin-bottom: 20px;
}
.page-header {
display: flex;
justify-content: space-between;
align-items: center;
margin-bottom: 30px;
padding-bottom: 20px;
border-bottom: 2px solid #f0f0f0;
}
.page-header h1 {
color: var(--color-primary);
font-size: 2rem;
margin: 0;
}
.page-header p {
margin: 10px 0 0 0;
color: #666;
}
.dle-address {
margin-top: 10px !important;
font-family: monospace;
background: #f8f9fa;
padding: 8px 12px;
border-radius: var(--radius-sm);
border: 1px solid #dee2e6;
}
.close-btn {
background: none;
border: none;
font-size: 1.5rem;
cursor: pointer;
color: #666;
padding: 0;
width: 30px;
height: 30px;
display: flex;
align-items: center;
justify-content: center;
border-radius: 50%;
transition: all 0.2s;
}
.close-btn:hover {
background: #f0f0f0;
color: #333;
}
/* Информация о модуле */
.module-info {
margin-bottom: 30px;
}
.info-card {
background: #f8f9fa;
border-radius: var(--radius-md);
padding: 20px;
border: 1px solid #e9ecef;
}
.info-card h3 {
margin: 0 0 15px 0;
color: var(--color-primary);
}
.info-grid {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(250px, 1fr));
gap: 15px;
}
.info-item {
padding: 10px;
background: white;
border-radius: var(--radius-sm);
border: 1px solid #dee2e6;
}
.info-item strong {
color: var(--color-primary);
}
/* Плейсхолдер для формы */
.deploy-form-placeholder {
background: #f8f9fa;
border-radius: var(--radius-md);
padding: 40px;
text-align: center;
border: 2px dashed #dee2e6;
}
.placeholder-content h3 {
color: var(--color-primary);
margin-bottom: 10px;
}
.placeholder-content p {
color: #666;
margin: 0;
}
/* Адаптивность */
@media (max-width: 768px) {
.info-grid {
grid-template-columns: 1fr;
}
.page-header {
flex-direction: column;
align-items: flex-start;
gap: 15px;
}
}
</style>

View File

@@ -0,0 +1,218 @@
<!--
Copyright (c) 2024-2025 Тарабанов Александр Викторович
All rights reserved.
This software is proprietary and confidential.
Unauthorized copying, modification, or distribution is prohibited.
For licensing inquiries: info@hb3-accelerator.com
Website: https://hb3-accelerator.com
GitHub: https://github.com/HB3-ACCELERATOR
-->
<template>
<BaseLayout
:is-authenticated="isAuthenticated"
:identities="identities"
:token-balances="tokenBalances"
:is-loading-tokens="isLoadingTokens"
@auth-action-completed="$emit('auth-action-completed')"
>
<div class="treasury-module-deploy">
<!-- Заголовок -->
<div class="page-header">
<div class="header-content">
<h1>Деплой TreasuryModule</h1>
<p>Казначейство DLE - управление финансами, депозиты, выводы, дивиденды</p>
<p v-if="dleAddress" class="dle-address">
<strong>DLE:</strong> {{ dleAddress }}
</p>
</div>
<button class="close-btn" @click="router.push('/management/modules')">×</button>
</div>
<!-- Информация о модуле -->
<div class="module-info">
<div class="info-card">
<h3>💰 TreasuryModule</h3>
<div class="info-grid">
<div class="info-item">
<strong>Назначение:</strong> Управление финансами DLE
</div>
<div class="info-item">
<strong>Функции:</strong> Депозиты, выводы, дивиденды, бюджетирование
</div>
<div class="info-item">
<strong>Безопасность:</strong> Все операции через голосование
</div>
</div>
</div>
</div>
<!-- Форма деплоя будет добавлена позже -->
<div class="deploy-form-placeholder">
<div class="placeholder-content">
<h3>🚧 Форма деплоя в разработке</h3>
<p>Здесь будет форма для деплоя TreasuryModule</p>
</div>
</div>
</div>
</BaseLayout>
</template>
<script setup>
import { defineProps, defineEmits, ref, onMounted } from 'vue';
import { useRouter, useRoute } from 'vue-router';
import BaseLayout from '../../../components/BaseLayout.vue';
// Определяем props
const props = defineProps({
isAuthenticated: { type: Boolean, default: false },
identities: { type: Array, default: () => [] },
tokenBalances: { type: Object, default: () => ({}) },
isLoadingTokens: { type: Boolean, default: false }
});
// Определяем emits
const emit = defineEmits(['auth-action-completed']);
const router = useRouter();
const route = useRoute();
// Состояние
const isLoading = ref(false);
const dleAddress = ref(route.query.address || null);
// Инициализация
onMounted(() => {
console.log('[TreasuryModuleDeployView] Страница загружена');
});
</script>
<style scoped>
.treasury-module-deploy {
padding: 20px;
background-color: var(--color-white);
border-radius: var(--radius-lg);
box-shadow: 0 2px 10px rgba(0, 0, 0, 0.1);
margin-top: 20px;
margin-bottom: 20px;
}
.page-header {
display: flex;
justify-content: space-between;
align-items: center;
margin-bottom: 30px;
padding-bottom: 20px;
border-bottom: 2px solid #f0f0f0;
}
.page-header h1 {
color: var(--color-primary);
font-size: 2rem;
margin: 0;
}
.page-header p {
margin: 10px 0 0 0;
color: #666;
}
.dle-address {
margin-top: 10px !important;
font-family: monospace;
background: #f8f9fa;
padding: 8px 12px;
border-radius: var(--radius-sm);
border: 1px solid #dee2e6;
}
.close-btn {
background: none;
border: none;
font-size: 1.5rem;
cursor: pointer;
color: #666;
padding: 0;
width: 30px;
height: 30px;
display: flex;
align-items: center;
justify-content: center;
border-radius: 50%;
transition: all 0.2s;
}
.close-btn:hover {
background: #f0f0f0;
color: #333;
}
/* Информация о модуле */
.module-info {
margin-bottom: 30px;
}
.info-card {
background: #f8f9fa;
border-radius: var(--radius-md);
padding: 20px;
border: 1px solid #e9ecef;
}
.info-card h3 {
margin: 0 0 15px 0;
color: var(--color-primary);
}
.info-grid {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(250px, 1fr));
gap: 15px;
}
.info-item {
padding: 10px;
background: white;
border-radius: var(--radius-sm);
border: 1px solid #dee2e6;
}
.info-item strong {
color: var(--color-primary);
}
/* Плейсхолдер для формы */
.deploy-form-placeholder {
background: #f8f9fa;
border-radius: var(--radius-md);
padding: 40px;
text-align: center;
border: 2px dashed #dee2e6;
}
.placeholder-content h3 {
color: var(--color-primary);
margin-bottom: 10px;
}
.placeholder-content p {
color: #666;
margin: 0;
}
/* Адаптивность */
@media (max-width: 768px) {
.info-grid {
grid-template-columns: 1fr;
}
.page-header {
flex-direction: column;
align-items: flex-start;
gap: 15px;
}
}
</style>