ваше сообщение коммита
This commit is contained in:
@@ -26,7 +26,8 @@
|
||||
"sortablejs": "^1.15.6",
|
||||
"vue": "^3.2.47",
|
||||
"vue-i18n": "^11.1.2",
|
||||
"vue-router": "^4.1.6"
|
||||
"vue-router": "^4.1.6",
|
||||
"vuex": "^4.1.0"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@vitejs/plugin-vue": "^5.2.1",
|
||||
|
||||
235
frontend/src/composables/useBlockchainNetworks.js
Normal file
235
frontend/src/composables/useBlockchainNetworks.js
Normal file
@@ -0,0 +1,235 @@
|
||||
import { ref, computed } from 'vue';
|
||||
import axios from 'axios';
|
||||
|
||||
/**
|
||||
* Composable для работы с сетями блокчейн
|
||||
* Предоставляет списки доступных сетей, URL RPC и функции для работы с ними
|
||||
*/
|
||||
export default function useBlockchainNetworks() {
|
||||
// Список стандартных URL для популярных сетей
|
||||
const defaultRpcUrls = {
|
||||
ethereum: 'https://mainnet.infura.io/v3/YOUR_API_KEY',
|
||||
sepolia: 'https://sepolia.infura.io/v3/YOUR_API_KEY',
|
||||
goerli: 'https://goerli.infura.io/v3/YOUR_API_KEY',
|
||||
holesky: 'https://holesky.infura.io/v3/YOUR_API_KEY',
|
||||
bsc: 'https://bsc-dataseed1.binance.org',
|
||||
'bsc-testnet': 'https://data-seed-prebsc-1-s1.binance.org:8545',
|
||||
polygon: 'https://polygon-rpc.com',
|
||||
mumbai: 'https://rpc-mumbai.maticvigil.com',
|
||||
arbitrum: 'https://arb1.arbitrum.io/rpc',
|
||||
'arbitrum-goerli': 'https://goerli-rollup.arbitrum.io/rpc',
|
||||
optimism: 'https://mainnet.optimism.io',
|
||||
'optimism-goerli': 'https://goerli.optimism.io',
|
||||
avalanche: 'https://api.avax.network/ext/bc/C/rpc',
|
||||
'avalanche-fuji': 'https://api.avax-test.network/ext/bc/C/rpc',
|
||||
gnosis: 'https://rpc.gnosischain.com',
|
||||
celo: 'https://forno.celo.org',
|
||||
fantom: 'https://rpc.ftm.tools',
|
||||
'fantom-testnet': 'https://rpc.testnet.fantom.network',
|
||||
harmony: 'https://api.harmony.one',
|
||||
metis: 'https://andromeda.metis.io/?owner=1088',
|
||||
aurora: 'https://mainnet.aurora.dev',
|
||||
cronos: 'https://evm.cronos.org',
|
||||
localhost: 'http://localhost:8545',
|
||||
ganache: 'http://localhost:7545'
|
||||
};
|
||||
|
||||
// Группы сетей для отображения в интерфейсе
|
||||
const networkGroups = [
|
||||
{
|
||||
label: 'Основные сети',
|
||||
options: [
|
||||
{ value: 'ethereum', label: 'Ethereum Mainnet', chainId: 1 },
|
||||
{ value: 'bsc', label: 'Binance Smart Chain', chainId: 56 },
|
||||
{ value: 'polygon', label: 'Polygon', chainId: 137 },
|
||||
{ value: 'arbitrum', label: 'Arbitrum One', chainId: 42161 },
|
||||
{ value: 'optimism', label: 'Optimism', chainId: 10 },
|
||||
{ value: 'avalanche', label: 'Avalanche C-Chain', chainId: 43114 },
|
||||
{ value: 'gnosis', label: 'Gnosis Chain (xDai)', chainId: 100 },
|
||||
{ value: 'celo', label: 'Celo', chainId: 42220 },
|
||||
{ value: 'fantom', label: 'Fantom Opera', chainId: 250 },
|
||||
{ value: 'harmony', label: 'Harmony', chainId: 1666600000 },
|
||||
{ value: 'metis', label: 'Metis Andromeda', chainId: 1088 },
|
||||
{ value: 'aurora', label: 'Aurora', chainId: 1313161554 },
|
||||
{ value: 'cronos', label: 'Cronos', chainId: 25 }
|
||||
]
|
||||
},
|
||||
{
|
||||
label: 'Тестовые сети',
|
||||
options: [
|
||||
{ value: 'sepolia', label: 'Sepolia (Ethereum testnet)', chainId: 11155111 },
|
||||
{ value: 'goerli', label: 'Goerli (Ethereum testnet)', chainId: 5 },
|
||||
{ value: 'holesky', label: 'Holesky (Ethereum testnet)', chainId: 17000 },
|
||||
{ value: 'bsc-testnet', label: 'BSC Testnet', chainId: 97 },
|
||||
{ value: 'mumbai', label: 'Mumbai (Polygon testnet)', chainId: 80001 },
|
||||
{ value: 'arbitrum-goerli', label: 'Arbitrum Goerli', chainId: 421613 },
|
||||
{ value: 'optimism-goerli', label: 'Optimism Goerli', chainId: 420 },
|
||||
{ value: 'avalanche-fuji', label: 'Avalanche Fuji', chainId: 43113 },
|
||||
{ value: 'fantom-testnet', label: 'Fantom Testnet', chainId: 4002 }
|
||||
]
|
||||
},
|
||||
{
|
||||
label: 'Локальные сети',
|
||||
options: [
|
||||
{ value: 'localhost', label: 'Localhost (Hardhat)', chainId: 31337 },
|
||||
{ value: 'ganache', label: 'Ganache', chainId: 1337 }
|
||||
]
|
||||
},
|
||||
{
|
||||
label: 'Другое',
|
||||
options: [
|
||||
{ value: 'custom', label: 'Другая сеть (ввести вручную)', chainId: null }
|
||||
]
|
||||
}
|
||||
];
|
||||
|
||||
// Создаем плоский список всех сетей для удобного использования в компонентах
|
||||
const networks = computed(() => {
|
||||
return networkGroups.flatMap(group => group.options);
|
||||
});
|
||||
|
||||
// Объект для хранения выбранной сети и пользовательских значений
|
||||
const networkEntry = ref({
|
||||
networkId: '',
|
||||
rpcUrl: '',
|
||||
customNetworkId: '',
|
||||
customChainId: null
|
||||
});
|
||||
|
||||
// Вычисляемое свойство для предложения URL на основе выбранной сети
|
||||
const defaultRpcUrlSuggestion = computed(() => {
|
||||
if (networkEntry.value.networkId && defaultRpcUrls[networkEntry.value.networkId]) {
|
||||
return defaultRpcUrls[networkEntry.value.networkId];
|
||||
}
|
||||
return '';
|
||||
});
|
||||
|
||||
// Функция для использования предложенного URL
|
||||
const useDefaultRpcUrl = () => {
|
||||
if (defaultRpcUrlSuggestion.value) {
|
||||
networkEntry.value.rpcUrl = defaultRpcUrlSuggestion.value;
|
||||
}
|
||||
};
|
||||
|
||||
// Функция для получения chainId по networkId
|
||||
const getChainIdByNetworkId = (networkId) => {
|
||||
for (const group of networkGroups) {
|
||||
const option = group.options.find(opt => opt.value === networkId);
|
||||
if (option) {
|
||||
return option.chainId;
|
||||
}
|
||||
}
|
||||
return null;
|
||||
};
|
||||
|
||||
// Функция для добавления новой конфигурации сети
|
||||
const validateAndPrepareNetworkConfig = () => {
|
||||
let networkId = networkEntry.value.networkId;
|
||||
const rpcUrl = networkEntry.value.rpcUrl.trim();
|
||||
|
||||
// Если выбрана опция "custom", используем пользовательский ID
|
||||
if (networkId === 'custom') {
|
||||
networkId = networkEntry.value.customNetworkId.trim();
|
||||
if (!networkId) {
|
||||
return { valid: false, error: 'Пожалуйста, введите пользовательский ID сети' };
|
||||
}
|
||||
}
|
||||
|
||||
if (!networkId || !rpcUrl) {
|
||||
return { valid: false, error: 'Пожалуйста, выберите ID Сети и введите RPC URL' };
|
||||
}
|
||||
|
||||
// Определяем chainId
|
||||
let chainId = getChainIdByNetworkId(networkId);
|
||||
if (!chainId && networkId === networkEntry.value.customNetworkId) {
|
||||
chainId = networkEntry.value.customChainId;
|
||||
}
|
||||
|
||||
return {
|
||||
valid: true,
|
||||
networkConfig: {
|
||||
networkId,
|
||||
rpcUrl,
|
||||
chainId
|
||||
}
|
||||
};
|
||||
};
|
||||
|
||||
// Функция сброса формы
|
||||
const resetNetworkEntry = () => {
|
||||
networkEntry.value.networkId = '';
|
||||
networkEntry.value.rpcUrl = '';
|
||||
networkEntry.value.customNetworkId = '';
|
||||
networkEntry.value.customChainId = null;
|
||||
};
|
||||
|
||||
// Функция получения списка всех доступных сетей в плоском формате
|
||||
const getAllNetworks = () => {
|
||||
return networks.value;
|
||||
};
|
||||
|
||||
// Функция получения метаданных сети по ID
|
||||
const getNetworkMetadata = (networkId) => {
|
||||
return networks.value.find(network => network.value === networkId) || null;
|
||||
};
|
||||
|
||||
// Состояние для тестирования RPC
|
||||
const testingRpc = ref(false);
|
||||
const testingRpcId = ref('');
|
||||
|
||||
// Функция для тестирования RPC-соединения
|
||||
const testRpcConnection = async (networkId, rpcUrl) => {
|
||||
testingRpc.value = true;
|
||||
testingRpcId.value = networkId;
|
||||
|
||||
try {
|
||||
// Формируем запрос на бэкенд для проверки RPC
|
||||
const response = await axios.post('/api/settings/rpc-test', {
|
||||
networkId,
|
||||
rpcUrl
|
||||
});
|
||||
|
||||
if (response.data && response.data.success) {
|
||||
return {
|
||||
success: true,
|
||||
message: `Соединение с ${networkId} успешно установлено! Номер блока: ${response.data.blockNumber}`,
|
||||
blockNumber: response.data.blockNumber
|
||||
};
|
||||
} else {
|
||||
return {
|
||||
success: false,
|
||||
error: response.data?.error || 'Не удалось установить соединение'
|
||||
};
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('[useBlockchainNetworks] Ошибка при тестировании RPC:', error);
|
||||
return {
|
||||
success: false,
|
||||
error: error.response?.data?.error || error.message || 'Неизвестная ошибка'
|
||||
};
|
||||
} finally {
|
||||
testingRpc.value = false;
|
||||
testingRpcId.value = '';
|
||||
}
|
||||
};
|
||||
|
||||
return {
|
||||
// Данные
|
||||
defaultRpcUrls,
|
||||
networkGroups,
|
||||
networkEntry,
|
||||
defaultRpcUrlSuggestion,
|
||||
testingRpc,
|
||||
testingRpcId,
|
||||
networks, // Экспортируем плоский список сетей
|
||||
|
||||
// Методы
|
||||
useDefaultRpcUrl,
|
||||
getChainIdByNetworkId,
|
||||
validateAndPrepareNetworkConfig,
|
||||
resetNetworkEntry,
|
||||
getAllNetworks,
|
||||
getNetworkMetadata,
|
||||
testRpcConnection
|
||||
};
|
||||
}
|
||||
92
frontend/src/services/dleService.js
Normal file
92
frontend/src/services/dleService.js
Normal file
@@ -0,0 +1,92 @@
|
||||
import api from '@/api/axios';
|
||||
|
||||
/**
|
||||
* Сервис для работы с DLE (Digital Legal Entity)
|
||||
*/
|
||||
class DLEService {
|
||||
/**
|
||||
* Получает настройки по умолчанию для создания DLE
|
||||
* @returns {Promise<Object>} - Настройки по умолчанию
|
||||
*/
|
||||
async getDefaultSettings() {
|
||||
try {
|
||||
const response = await api.get('/api/dle/settings');
|
||||
return response.data.data;
|
||||
} catch (error) {
|
||||
console.error('Ошибка при получении настроек DLE:', error);
|
||||
throw error;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Создает новое DLE с указанными параметрами
|
||||
* @param {Object} dleParams - Параметры для создания DLE
|
||||
* @returns {Promise<Object>} - Результат создания DLE
|
||||
*/
|
||||
async createDLE(dleParams) {
|
||||
try {
|
||||
const response = await api.post('/api/dle', dleParams);
|
||||
return response.data;
|
||||
} catch (error) {
|
||||
console.error('Ошибка при создании DLE:', error);
|
||||
throw error;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Получает список всех DLE
|
||||
* @returns {Promise<Array>} - Список DLE
|
||||
*/
|
||||
async getAllDLEs() {
|
||||
try {
|
||||
const response = await api.get('/api/dle');
|
||||
|
||||
// Проверяем и нормализуем поля isicCodes для всех DLE
|
||||
if (response.data.data && Array.isArray(response.data.data)) {
|
||||
response.data.data.forEach(dle => {
|
||||
// Если isicCodes отсутствует или не является массивом, инициализируем пустым массивом
|
||||
if (!dle.isicCodes || !Array.isArray(dle.isicCodes)) {
|
||||
dle.isicCodes = [];
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
return response.data.data;
|
||||
} catch (error) {
|
||||
console.error('Ошибка при получении списка DLE:', error);
|
||||
throw error;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Удаляет DLE по адресу токена
|
||||
* @param {string} tokenAddress - Адрес токена DLE
|
||||
* @returns {Promise<Object>} - Результат удаления
|
||||
*/
|
||||
async deleteDLE(tokenAddress) {
|
||||
try {
|
||||
const response = await api.delete(`/api/dle/${tokenAddress}`);
|
||||
return response.data;
|
||||
} catch (error) {
|
||||
console.error('Ошибка при удалении DLE:', error);
|
||||
throw error;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Удаляет пустое DLE по имени файла
|
||||
* @param {string} fileName - Имя файла DLE
|
||||
* @returns {Promise<Object>} - Результат удаления
|
||||
*/
|
||||
async deleteEmptyDLE(fileName) {
|
||||
try {
|
||||
const response = await api.delete(`/api/dle/empty/${fileName}`);
|
||||
return response.data;
|
||||
} catch (error) {
|
||||
console.error('Ошибка при удалении пустого DLE:', error);
|
||||
throw error;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
export default new DLEService();
|
||||
@@ -7,27 +7,260 @@
|
||||
@auth-action-completed="$emit('auth-action-completed')"
|
||||
>
|
||||
<div class="crm-view-container">
|
||||
<h1>CRM Система</h1>
|
||||
<div v-if="isLoading">Загрузка данных пользователя...</div>
|
||||
<h1>Управление DLE</h1>
|
||||
<div v-if="isLoading">
|
||||
<p>Загрузка данных DLE...</p>
|
||||
<div class="loading-spinner"></div>
|
||||
</div>
|
||||
<div v-else-if="!auth.isAuthenticated.value">
|
||||
<p>Для доступа к CRM необходимо <button @click="goToHomeAndShowSidebar">войти</button>.</p>
|
||||
<p>Для доступа к управлению DLE необходимо <button @click="goToHomeAndShowSidebar">войти</button>.</p>
|
||||
</div>
|
||||
<div v-else>
|
||||
<p>Добро пожаловать в CRM!</p>
|
||||
<div v-if="auth.isAdmin.value">
|
||||
<p><strong>У вас полный доступ (Администратор).</strong></p>
|
||||
<!-- Сюда будет добавляться полный функционал CRM -->
|
||||
<p>Здесь будет управление контактами, сделками, задачами и т.д.</p>
|
||||
<!-- Секция со списком DLE -->
|
||||
<div class="dle-list-section">
|
||||
<h2>Ваши DLE</h2>
|
||||
<div v-if="dleList.length === 0" class="no-dle-message">
|
||||
<p>У вас пока нет созданных DLE.</p>
|
||||
<button @click="goToBlockchainSettings" class="btn btn-primary">
|
||||
<i class="fas fa-plus"></i> Создать новое DLE
|
||||
</button>
|
||||
</div>
|
||||
<div v-else>
|
||||
<p><strong>У вас ограниченный доступ.</strong></p>
|
||||
<!-- Сюда будет добавляться ограниченный функционал CRM -->
|
||||
<p>Здесь будет просмотр ваших контактов и задач.</p>
|
||||
<div class="dle-list">
|
||||
<div v-for="(dle, index) in dleList" :key="index" class="dle-card"
|
||||
:class="{ 'active': selectedDleIndex === index }"
|
||||
@click="selectDle(index)">
|
||||
<h3>{{ dle.name }} ({{ dle.symbol }})</h3>
|
||||
<p><strong>Адрес:</strong> {{ shortenAddress(dle.tokenAddress) }}</p>
|
||||
<p><strong>Местонахождение:</strong> {{ dle.location }}</p>
|
||||
<div class="dle-card-actions">
|
||||
<button class="btn btn-sm btn-info">
|
||||
<i class="fas fa-info-circle"></i> Подробнее
|
||||
</button>
|
||||
<button v-if="!dle.name || !dle.name.trim() || !dle.tokenAddress" class="btn btn-sm btn-danger" @click.stop="deleteDLE(index, dle)">
|
||||
<i class="fas fa-trash"></i> Удалить
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
<!-- Демонстрационный блок -->
|
||||
<div class="demo-block">
|
||||
<h3>Демонстрация CRM</h3>
|
||||
<p>Этот раздел будет содержать компоненты CRM...</p>
|
||||
|
||||
<!-- Секция с деталями выбранного DLE -->
|
||||
<div v-if="selectedDle" class="dle-details-section">
|
||||
<h2>Управление "{{ selectedDle.name }}"</h2>
|
||||
|
||||
<div class="dle-tabs">
|
||||
<div class="tab-header">
|
||||
<div class="tab-button"
|
||||
:class="{ 'active': activeTab === 'info' }"
|
||||
@click="activeTab = 'info'">
|
||||
<i class="fas fa-info-circle"></i> Основная информация
|
||||
</div>
|
||||
<div class="tab-button"
|
||||
:class="{ 'active': activeTab === 'proposals' }"
|
||||
@click="activeTab = 'proposals'">
|
||||
<i class="fas fa-tasks"></i> Предложения
|
||||
</div>
|
||||
<div class="tab-button"
|
||||
:class="{ 'active': activeTab === 'governance' }"
|
||||
@click="activeTab = 'governance'">
|
||||
<i class="fas fa-balance-scale"></i> Управление
|
||||
</div>
|
||||
<div class="tab-button"
|
||||
:class="{ 'active': activeTab === 'modules' }"
|
||||
@click="activeTab = 'modules'">
|
||||
<i class="fas fa-puzzle-piece"></i> Модули
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Вкладка информации -->
|
||||
<div class="tab-content" v-if="activeTab === 'info'">
|
||||
<div class="info-card">
|
||||
<h3>Основная информация</h3>
|
||||
<div class="info-row">
|
||||
<span class="info-label">Название:</span>
|
||||
<span class="info-value">{{ selectedDle.name }}</span>
|
||||
</div>
|
||||
<div class="info-row">
|
||||
<span class="info-label">Символ токена:</span>
|
||||
<span class="info-value">{{ selectedDle.symbol }}</span>
|
||||
</div>
|
||||
<div class="info-row">
|
||||
<span class="info-label">Местонахождение:</span>
|
||||
<span class="info-value">{{ selectedDle.location }}</span>
|
||||
</div>
|
||||
<div class="info-row">
|
||||
<span class="info-label">Коды деятельности:</span>
|
||||
<span class="info-value">{{ selectedDle.isicCodes && selectedDle.isicCodes.length ? selectedDle.isicCodes.join(', ') : 'Не указаны' }}</span>
|
||||
</div>
|
||||
<div class="info-row">
|
||||
<span class="info-label">Дата создания:</span>
|
||||
<span class="info-value">{{ formatDate(selectedDle.creationTimestamp) }}</span>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="contract-cards">
|
||||
<div class="contract-card">
|
||||
<h4>Токен управления</h4>
|
||||
<p class="address">{{ selectedDle.tokenAddress }}</p>
|
||||
<div class="contract-actions">
|
||||
<button class="btn btn-sm btn-secondary" @click="copyToClipboard(selectedDle.tokenAddress)">
|
||||
<i class="fas fa-copy"></i> Копировать адрес
|
||||
</button>
|
||||
<button class="btn btn-sm btn-info" @click="viewOnExplorer(selectedDle.tokenAddress)">
|
||||
<i class="fas fa-external-link-alt"></i> Обзор
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="contract-card">
|
||||
<h4>Таймлок</h4>
|
||||
<p class="address">{{ selectedDle.timelockAddress }}</p>
|
||||
<div class="contract-actions">
|
||||
<button class="btn btn-sm btn-secondary" @click="copyToClipboard(selectedDle.timelockAddress)">
|
||||
<i class="fas fa-copy"></i> Копировать адрес
|
||||
</button>
|
||||
<button class="btn btn-sm btn-info" @click="viewOnExplorer(selectedDle.timelockAddress)">
|
||||
<i class="fas fa-external-link-alt"></i> Обзор
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="contract-card">
|
||||
<h4>Governor</h4>
|
||||
<p class="address">{{ selectedDle.governorAddress }}</p>
|
||||
<div class="contract-actions">
|
||||
<button class="btn btn-sm btn-secondary" @click="copyToClipboard(selectedDle.governorAddress)">
|
||||
<i class="fas fa-copy"></i> Копировать адрес
|
||||
</button>
|
||||
<button class="btn btn-sm btn-info" @click="viewOnExplorer(selectedDle.governorAddress)">
|
||||
<i class="fas fa-external-link-alt"></i> Обзор
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Вкладка предложений -->
|
||||
<div class="tab-content" v-if="activeTab === 'proposals'">
|
||||
<h3>Предложения</h3>
|
||||
<div class="proposals-actions">
|
||||
<button class="btn btn-primary" @click="showCreateProposalForm = true">
|
||||
<i class="fas fa-plus"></i> Создать предложение
|
||||
</button>
|
||||
</div>
|
||||
|
||||
<div v-if="showCreateProposalForm" class="create-proposal-form">
|
||||
<h4>Новое предложение</h4>
|
||||
<div class="form-group">
|
||||
<label for="proposalTitle">Заголовок:</label>
|
||||
<input type="text" id="proposalTitle" v-model="newProposal.title" class="form-control">
|
||||
</div>
|
||||
<div class="form-group">
|
||||
<label for="proposalDescription">Описание:</label>
|
||||
<textarea id="proposalDescription" v-model="newProposal.description" class="form-control" rows="3"></textarea>
|
||||
</div>
|
||||
<div class="form-actions">
|
||||
<button class="btn btn-success" @click="createProposal" :disabled="isCreatingProposal">
|
||||
<i class="fas fa-paper-plane"></i> {{ isCreatingProposal ? 'Отправка...' : 'Отправить' }}
|
||||
</button>
|
||||
<button class="btn btn-secondary" @click="showCreateProposalForm = false">
|
||||
<i class="fas fa-times"></i> Отмена
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="proposals-list">
|
||||
<p v-if="proposals.length === 0">Предложений пока нет</p>
|
||||
<div v-else v-for="(proposal, index) in proposals" :key="index" class="proposal-card">
|
||||
<h4>{{ proposal.title }}</h4>
|
||||
<p>{{ proposal.description }}</p>
|
||||
<div class="proposal-status" :class="proposal.status">
|
||||
{{ getProposalStatusText(proposal.status) }}
|
||||
</div>
|
||||
<div class="proposal-actions">
|
||||
<button class="btn btn-sm btn-primary" @click="voteForProposal(proposal.id, true)" :disabled="!canVote(proposal)">
|
||||
<i class="fas fa-thumbs-up"></i> За
|
||||
</button>
|
||||
<button class="btn btn-sm btn-danger" @click="voteForProposal(proposal.id, false)" :disabled="!canVote(proposal)">
|
||||
<i class="fas fa-thumbs-down"></i> Против
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Вкладка управления -->
|
||||
<div class="tab-content" v-if="activeTab === 'governance'">
|
||||
<h3>Управление</h3>
|
||||
<div class="governance-info">
|
||||
<div class="info-card">
|
||||
<h4>Настройки Governor</h4>
|
||||
<div class="info-row">
|
||||
<span class="info-label">Порог предложения:</span>
|
||||
<span class="info-value">100,000 GT</span>
|
||||
</div>
|
||||
<div class="info-row">
|
||||
<span class="info-label">Кворум:</span>
|
||||
<span class="info-value">4%</span>
|
||||
</div>
|
||||
<div class="info-row">
|
||||
<span class="info-label">Задержка голосования:</span>
|
||||
<span class="info-value">1 день</span>
|
||||
</div>
|
||||
<div class="info-row">
|
||||
<span class="info-label">Период голосования:</span>
|
||||
<span class="info-value">7 дней</span>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="info-card">
|
||||
<h4>Статистика голосований</h4>
|
||||
<div class="info-row">
|
||||
<span class="info-label">Всего предложений:</span>
|
||||
<span class="info-value">{{ proposals.length }}</span>
|
||||
</div>
|
||||
<div class="info-row">
|
||||
<span class="info-label">Активных предложений:</span>
|
||||
<span class="info-value">{{ getProposalsByStatus('active').length }}</span>
|
||||
</div>
|
||||
<div class="info-row">
|
||||
<span class="info-label">Успешных предложений:</span>
|
||||
<span class="info-value">{{ getProposalsByStatus('succeeded').length }}</span>
|
||||
</div>
|
||||
<div class="info-row">
|
||||
<span class="info-label">Отклоненных предложений:</span>
|
||||
<span class="info-value">{{ getProposalsByStatus('defeated').length }}</span>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Вкладка модулей -->
|
||||
<div class="tab-content" v-if="activeTab === 'modules'">
|
||||
<h3>Подключение модулей</h3>
|
||||
<p>Здесь вы можете подключить дополнительные модули к вашему DLE.</p>
|
||||
|
||||
<div class="modules-list">
|
||||
<div v-for="(module, index) in availableModules" :key="index" class="module-card">
|
||||
<h4>{{ module.name }}</h4>
|
||||
<p>{{ module.description }}</p>
|
||||
<div class="module-status" :class="{ 'installed': module.installed }">
|
||||
{{ module.installed ? 'Установлен' : 'Доступен' }}
|
||||
</div>
|
||||
<div class="module-actions">
|
||||
<button v-if="!module.installed" class="btn btn-success" @click="installModule(module)">
|
||||
<i class="fas fa-plus"></i> Установить
|
||||
</button>
|
||||
<button v-else class="btn btn-danger" @click="uninstallModule(module)">
|
||||
<i class="fas fa-trash"></i> Удалить
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
@@ -35,12 +268,13 @@
|
||||
</template>
|
||||
|
||||
<script setup>
|
||||
import { ref, onMounted, onBeforeUnmount, defineProps, defineEmits } from 'vue';
|
||||
import { ref, onMounted, onBeforeUnmount, defineProps, defineEmits, computed } from 'vue';
|
||||
import { useAuth } from '../composables/useAuth';
|
||||
import { useRouter } from 'vue-router';
|
||||
import { setToStorage } from '../utils/storage';
|
||||
import BaseLayout from '../components/BaseLayout.vue';
|
||||
import eventBus from '../utils/eventBus';
|
||||
import dleService from '../services/dleService';
|
||||
|
||||
// Определяем props
|
||||
const props = defineProps({
|
||||
@@ -56,6 +290,48 @@ const emit = defineEmits(['auth-action-completed']);
|
||||
const auth = useAuth();
|
||||
const router = useRouter();
|
||||
const isLoading = ref(true);
|
||||
const dleList = ref([]);
|
||||
const selectedDleIndex = ref(null);
|
||||
const activeTab = ref('info');
|
||||
|
||||
// Для создания предложений
|
||||
const showCreateProposalForm = ref(false);
|
||||
const newProposal = ref({ title: '', description: '' });
|
||||
const isCreatingProposal = ref(false);
|
||||
|
||||
// Список доступных модулей
|
||||
const availableModules = ref([
|
||||
{
|
||||
name: 'Контракт на активы',
|
||||
description: 'Позволяет токенизировать физические активы и управлять ими через DLE.',
|
||||
installed: false
|
||||
},
|
||||
{
|
||||
name: 'Мультиподпись',
|
||||
description: 'Добавляет функциональность мультиподписи для повышенной безопасности.',
|
||||
installed: false
|
||||
},
|
||||
{
|
||||
name: 'Дивиденды',
|
||||
description: 'Позволяет распределять дивиденды между держателями токенов.',
|
||||
installed: false
|
||||
},
|
||||
{
|
||||
name: 'Стейкинг',
|
||||
description: 'Добавляет возможность стейкинга токенов для получения наград.',
|
||||
installed: false
|
||||
}
|
||||
]);
|
||||
|
||||
// Список предложений (в реальном приложении будет загружаться из смарт-контракта)
|
||||
const proposals = ref([]);
|
||||
|
||||
const selectedDle = computed(() => {
|
||||
if (selectedDleIndex.value !== null && dleList.value.length > selectedDleIndex.value) {
|
||||
return dleList.value[selectedDleIndex.value];
|
||||
}
|
||||
return null;
|
||||
});
|
||||
|
||||
// Функция для перехода на домашнюю страницу и открытия боковой панели
|
||||
const goToHomeAndShowSidebar = () => {
|
||||
@@ -63,11 +339,168 @@ const goToHomeAndShowSidebar = () => {
|
||||
router.push({ name: 'home' });
|
||||
};
|
||||
|
||||
// Функция для перехода на страницу настроек блокчейна
|
||||
const goToBlockchainSettings = () => {
|
||||
router.push({ name: 'settings-blockchain' });
|
||||
};
|
||||
|
||||
// Функция для выбора DLE
|
||||
const selectDle = (index) => {
|
||||
selectedDleIndex.value = index;
|
||||
activeTab.value = 'info'; // При выборе нового DLE сбрасываем на вкладку информации
|
||||
};
|
||||
|
||||
// Форматирование адреса (сокращение)
|
||||
const shortenAddress = (address) => {
|
||||
if (!address) return '';
|
||||
return `${address.slice(0, 6)}...${address.slice(-4)}`;
|
||||
};
|
||||
|
||||
// Форматирование даты из timestamp
|
||||
const formatDate = (timestamp) => {
|
||||
if (!timestamp) return 'N/A';
|
||||
return new Date(timestamp * 1000).toLocaleString();
|
||||
};
|
||||
|
||||
// Копирование в буфер обмена
|
||||
const copyToClipboard = (text) => {
|
||||
navigator.clipboard.writeText(text)
|
||||
.then(() => {
|
||||
alert('Адрес скопирован в буфер обмена');
|
||||
})
|
||||
.catch(err => {
|
||||
console.error('Ошибка при копировании текста: ', err);
|
||||
});
|
||||
};
|
||||
|
||||
// Открытие адреса в обозревателе блокчейна
|
||||
const viewOnExplorer = (address) => {
|
||||
// Используем Sepolia Etherscan как пример
|
||||
window.open(`https://sepolia.etherscan.io/address/${address}`, '_blank');
|
||||
};
|
||||
|
||||
// Создание нового предложения
|
||||
const createProposal = async () => {
|
||||
if (!newProposal.value.title || !newProposal.value.description) {
|
||||
alert('Пожалуйста, заполните все поля');
|
||||
return;
|
||||
}
|
||||
|
||||
isCreatingProposal.value = true;
|
||||
|
||||
try {
|
||||
// В реальном приложении здесь будет вызов смарт-контракта
|
||||
// Пока просто добавляем в локальный массив
|
||||
proposals.value.push({
|
||||
id: Date.now().toString(),
|
||||
title: newProposal.value.title,
|
||||
description: newProposal.value.description,
|
||||
status: 'pending',
|
||||
votes: { for: 0, against: 0 }
|
||||
});
|
||||
|
||||
showCreateProposalForm.value = false;
|
||||
newProposal.value = { title: '', description: '' };
|
||||
|
||||
alert('Предложение создано!');
|
||||
} catch (error) {
|
||||
console.error('Ошибка при создании предложения:', error);
|
||||
alert('Ошибка при создании предложения');
|
||||
} finally {
|
||||
isCreatingProposal.value = false;
|
||||
}
|
||||
};
|
||||
|
||||
// Голосование за предложение
|
||||
const voteForProposal = async (proposalId, isFor) => {
|
||||
try {
|
||||
// В реальном приложении здесь будет вызов смарт-контракта
|
||||
// Пока просто обновляем локальный массив
|
||||
const proposal = proposals.value.find(p => p.id === proposalId);
|
||||
if (proposal) {
|
||||
if (isFor) {
|
||||
proposal.votes.for += 1;
|
||||
} else {
|
||||
proposal.votes.against += 1;
|
||||
}
|
||||
|
||||
// Обновляем статус в зависимости от голосов
|
||||
if (proposal.votes.for > proposal.votes.against && proposal.votes.for >= 3) {
|
||||
proposal.status = 'succeeded';
|
||||
} else if (proposal.votes.against > proposal.votes.for && proposal.votes.against >= 3) {
|
||||
proposal.status = 'defeated';
|
||||
} else {
|
||||
proposal.status = 'active';
|
||||
}
|
||||
|
||||
alert('Ваш голос учтен!');
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('Ошибка при голосовании:', error);
|
||||
alert('Ошибка при голосовании');
|
||||
}
|
||||
};
|
||||
|
||||
// Проверка возможности голосования
|
||||
const canVote = (proposal) => {
|
||||
return proposal.status === 'active' || proposal.status === 'pending';
|
||||
};
|
||||
|
||||
// Получение текстового статуса предложения
|
||||
const getProposalStatusText = (status) => {
|
||||
const statusMap = {
|
||||
'pending': 'Ожидает',
|
||||
'active': 'Активно',
|
||||
'succeeded': 'Принято',
|
||||
'defeated': 'Отклонено',
|
||||
'executed': 'Выполнено'
|
||||
};
|
||||
return statusMap[status] || status;
|
||||
};
|
||||
|
||||
// Фильтрация предложений по статусу
|
||||
const getProposalsByStatus = (status) => {
|
||||
return proposals.value.filter(p => p.status === status);
|
||||
};
|
||||
|
||||
// Установка модуля
|
||||
const installModule = (module) => {
|
||||
// В реальном приложении здесь будет вызов смарт-контракта
|
||||
module.installed = true;
|
||||
alert(`Модуль "${module.name}" успешно установлен!`);
|
||||
};
|
||||
|
||||
// Удаление модуля
|
||||
const uninstallModule = (module) => {
|
||||
// В реальном приложении здесь будет вызов смарт-контракта
|
||||
module.installed = false;
|
||||
alert(`Модуль "${module.name}" удален.`);
|
||||
};
|
||||
|
||||
// Загрузка списка DLE
|
||||
const loadDLEs = async () => {
|
||||
isLoading.value = true;
|
||||
try {
|
||||
const result = await dleService.getAllDLEs();
|
||||
dleList.value = result || [];
|
||||
|
||||
// Выбираем первый DLE, если есть
|
||||
if (dleList.value.length > 0) {
|
||||
selectedDleIndex.value = 0;
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('Ошибка при загрузке списка DLE:', error);
|
||||
} finally {
|
||||
isLoading.value = false;
|
||||
}
|
||||
};
|
||||
|
||||
// Обработчик события изменения авторизации
|
||||
const handleAuthEvent = (eventData) => {
|
||||
console.log('[CrmView] Получено событие изменения авторизации:', eventData);
|
||||
// Можно обновить данные или состояние, если нужно
|
||||
isLoading.value = false;
|
||||
if (eventData.isAuthenticated) {
|
||||
loadDLEs();
|
||||
}
|
||||
};
|
||||
|
||||
// Регистрация и очистка обработчика событий
|
||||
@@ -75,7 +508,13 @@ let unsubscribe = null;
|
||||
|
||||
onMounted(() => {
|
||||
console.log('[CrmView] Компонент загружен');
|
||||
|
||||
// Если пользователь авторизован, загружаем данные
|
||||
if (auth.isAuthenticated.value) {
|
||||
loadDLEs();
|
||||
} else {
|
||||
isLoading.value = false;
|
||||
}
|
||||
|
||||
// Подписка на события авторизации
|
||||
unsubscribe = eventBus.on('auth-state-changed', handleAuthEvent);
|
||||
@@ -87,6 +526,42 @@ onBeforeUnmount(() => {
|
||||
unsubscribe();
|
||||
}
|
||||
});
|
||||
|
||||
// Функция для удаления DLE
|
||||
const deleteDLE = async (index, dle) => {
|
||||
if (!confirm(`Вы уверены, что хотите удалить DLE "${dle.name || 'без имени'}"?`)) {
|
||||
return;
|
||||
}
|
||||
|
||||
try {
|
||||
if (dle.tokenAddress) {
|
||||
// Если есть адрес токена, удаляем через основной метод
|
||||
await dleService.deleteDLE(dle.tokenAddress);
|
||||
} else if (dle._fileName) {
|
||||
// Если нет адреса токена, но есть имя файла, удаляем как пустое DLE
|
||||
await dleService.deleteEmptyDLE(dle._fileName);
|
||||
} else {
|
||||
// Если нет ни адреса токена, ни имени файла, просто удаляем из списка
|
||||
console.warn('DLE не имеет ни адреса токена, ни имени файла. Удаляется только из локального списка.');
|
||||
}
|
||||
|
||||
// Удаляем из локального списка
|
||||
dleList.value.splice(index, 1);
|
||||
|
||||
// Если был выбран этот DLE, сбрасываем выбор
|
||||
if (selectedDleIndex.value === index) {
|
||||
selectedDleIndex.value = null;
|
||||
} else if (selectedDleIndex.value > index) {
|
||||
// Если был выбран DLE с большим индексом, корректируем индекс
|
||||
selectedDleIndex.value--;
|
||||
}
|
||||
|
||||
alert(`DLE успешно удалено`);
|
||||
} catch (error) {
|
||||
console.error('Ошибка при удалении DLE:', error);
|
||||
alert(`Ошибка при удалении DLE: ${error.message || 'Неизвестная ошибка'}`);
|
||||
}
|
||||
};
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
@@ -99,9 +574,9 @@ onBeforeUnmount(() => {
|
||||
margin-bottom: 20px;
|
||||
}
|
||||
|
||||
h1 {
|
||||
h1, h2, h3, h4 {
|
||||
color: var(--color-dark);
|
||||
margin-bottom: 20px;
|
||||
margin-bottom: 16px;
|
||||
}
|
||||
|
||||
p {
|
||||
@@ -113,23 +588,361 @@ strong {
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
.demo-block {
|
||||
margin-top: 30px;
|
||||
padding: 20px;
|
||||
border: 1px dashed var(--color-grey);
|
||||
border-radius: var(--radius-md);
|
||||
.loading-spinner {
|
||||
width: 40px;
|
||||
height: 40px;
|
||||
border: 4px solid rgba(0, 0, 0, 0.1);
|
||||
border-left-color: var(--color-primary);
|
||||
border-radius: 50%;
|
||||
animation: spin 1s linear infinite;
|
||||
margin: 20px auto;
|
||||
}
|
||||
|
||||
button {
|
||||
padding: 5px 10px;
|
||||
background-color: var(--color-primary);
|
||||
color: var(--color-white);
|
||||
border: none;
|
||||
border-radius: var(--radius-md);
|
||||
cursor: pointer;
|
||||
margin-left: 5px;
|
||||
@keyframes spin {
|
||||
to { transform: rotate(360deg); }
|
||||
}
|
||||
button:hover {
|
||||
opacity: 0.9;
|
||||
|
||||
.btn {
|
||||
display: inline-block;
|
||||
font-weight: 400;
|
||||
text-align: center;
|
||||
white-space: nowrap;
|
||||
vertical-align: middle;
|
||||
user-select: none;
|
||||
border: 1px solid transparent;
|
||||
padding: 0.375rem 0.75rem;
|
||||
font-size: 1rem;
|
||||
line-height: 1.5;
|
||||
border-radius: 0.25rem;
|
||||
transition: all 0.15s ease-in-out;
|
||||
cursor: pointer;
|
||||
}
|
||||
|
||||
.btn-sm {
|
||||
padding: 0.25rem 0.5rem;
|
||||
font-size: 0.875rem;
|
||||
line-height: 1.5;
|
||||
border-radius: 0.2rem;
|
||||
}
|
||||
|
||||
.btn-primary {
|
||||
color: #fff;
|
||||
background-color: var(--color-primary);
|
||||
border-color: var(--color-primary);
|
||||
}
|
||||
|
||||
.btn-secondary {
|
||||
color: #fff;
|
||||
background-color: var(--color-grey-dark);
|
||||
border-color: var(--color-grey-dark);
|
||||
}
|
||||
|
||||
.btn-success {
|
||||
color: #fff;
|
||||
background-color: #28a745;
|
||||
border-color: #28a745;
|
||||
}
|
||||
|
||||
.btn-danger {
|
||||
color: #fff;
|
||||
background-color: #dc3545;
|
||||
border-color: #dc3545;
|
||||
}
|
||||
|
||||
.btn-info {
|
||||
color: #fff;
|
||||
background-color: #17a2b8;
|
||||
border-color: #17a2b8;
|
||||
}
|
||||
|
||||
/* Стили для секции списка DLE */
|
||||
.dle-list-section {
|
||||
margin-bottom: 30px;
|
||||
}
|
||||
|
||||
.dle-list {
|
||||
display: flex;
|
||||
flex-wrap: wrap;
|
||||
gap: 15px;
|
||||
margin-top: 20px;
|
||||
}
|
||||
|
||||
.dle-card {
|
||||
width: 300px;
|
||||
padding: 15px;
|
||||
border: 1px solid var(--color-grey-light);
|
||||
border-radius: var(--radius-md);
|
||||
box-shadow: 0 2px 5px rgba(0, 0, 0, 0.05);
|
||||
cursor: pointer;
|
||||
transition: all 0.2s ease;
|
||||
}
|
||||
|
||||
.dle-card:hover {
|
||||
box-shadow: 0 5px 15px rgba(0, 0, 0, 0.1);
|
||||
}
|
||||
|
||||
.dle-card.active {
|
||||
border-color: var(--color-primary);
|
||||
box-shadow: 0 5px 15px rgba(0, 0, 0, 0.1);
|
||||
}
|
||||
|
||||
.dle-card h3 {
|
||||
margin-top: 0;
|
||||
margin-bottom: 10px;
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
.dle-card-actions {
|
||||
margin-top: 15px;
|
||||
text-align: right;
|
||||
}
|
||||
|
||||
/* Стили для секции деталей DLE */
|
||||
.dle-details-section {
|
||||
margin-top: 30px;
|
||||
border-top: 1px solid var(--color-grey-light);
|
||||
padding-top: 20px;
|
||||
}
|
||||
|
||||
/* Стили для вкладок */
|
||||
.dle-tabs {
|
||||
margin-top: 20px;
|
||||
}
|
||||
|
||||
.tab-header {
|
||||
display: flex;
|
||||
border-bottom: 1px solid var(--color-grey-light);
|
||||
margin-bottom: 20px;
|
||||
}
|
||||
|
||||
.tab-button {
|
||||
padding: 10px 20px;
|
||||
cursor: pointer;
|
||||
border-bottom: 2px solid transparent;
|
||||
transition: all 0.2s ease;
|
||||
}
|
||||
|
||||
.tab-button:hover {
|
||||
background-color: rgba(0, 0, 0, 0.02);
|
||||
}
|
||||
|
||||
.tab-button.active {
|
||||
border-bottom-color: var(--color-primary);
|
||||
color: var(--color-primary);
|
||||
font-weight: 500;
|
||||
}
|
||||
|
||||
.tab-content {
|
||||
padding: 10px;
|
||||
}
|
||||
|
||||
/* Стили для информационных карточек */
|
||||
.info-card {
|
||||
background-color: #f8f9fa;
|
||||
border-radius: var(--radius-md);
|
||||
padding: 15px;
|
||||
margin-bottom: 20px;
|
||||
}
|
||||
|
||||
.info-row {
|
||||
display: flex;
|
||||
margin-bottom: 10px;
|
||||
align-items: baseline;
|
||||
}
|
||||
|
||||
.info-label {
|
||||
font-weight: 500;
|
||||
width: 200px;
|
||||
color: var(--color-dark);
|
||||
}
|
||||
|
||||
.info-value {
|
||||
flex: 1;
|
||||
}
|
||||
|
||||
/* Стили для карточек контрактов */
|
||||
.contract-cards {
|
||||
display: flex;
|
||||
flex-wrap: wrap;
|
||||
gap: 15px;
|
||||
margin-top: 20px;
|
||||
}
|
||||
|
||||
.contract-card {
|
||||
flex: 1;
|
||||
min-width: 250px;
|
||||
padding: 15px;
|
||||
background-color: #f8f9fa;
|
||||
border-radius: var(--radius-md);
|
||||
box-shadow: 0 2px 5px rgba(0, 0, 0, 0.05);
|
||||
}
|
||||
|
||||
.contract-card h4 {
|
||||
margin-top: 0;
|
||||
margin-bottom: 10px;
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
.contract-card .address {
|
||||
font-family: monospace;
|
||||
word-break: break-all;
|
||||
padding: 5px;
|
||||
background-color: rgba(0, 0, 0, 0.03);
|
||||
border-radius: 4px;
|
||||
}
|
||||
|
||||
.contract-actions {
|
||||
margin-top: 15px;
|
||||
display: flex;
|
||||
gap: 10px;
|
||||
}
|
||||
|
||||
/* Стили для формы создания предложения */
|
||||
.create-proposal-form {
|
||||
background-color: #f8f9fa;
|
||||
border-radius: var(--radius-md);
|
||||
padding: 20px;
|
||||
margin-top: 20px;
|
||||
margin-bottom: 20px;
|
||||
}
|
||||
|
||||
.form-group {
|
||||
margin-bottom: 15px;
|
||||
}
|
||||
|
||||
.form-group label {
|
||||
display: block;
|
||||
margin-bottom: 5px;
|
||||
font-weight: 500;
|
||||
}
|
||||
|
||||
.form-control {
|
||||
display: block;
|
||||
width: 100%;
|
||||
padding: 0.375rem 0.75rem;
|
||||
font-size: 1rem;
|
||||
line-height: 1.5;
|
||||
color: #495057;
|
||||
background-color: #fff;
|
||||
background-clip: padding-box;
|
||||
border: 1px solid #ced4da;
|
||||
border-radius: 0.25rem;
|
||||
transition: border-color 0.15s ease-in-out, box-shadow 0.15s ease-in-out;
|
||||
}
|
||||
|
||||
.form-actions {
|
||||
display: flex;
|
||||
gap: 10px;
|
||||
margin-top: 20px;
|
||||
}
|
||||
|
||||
/* Стили для списка предложений */
|
||||
.proposal-card {
|
||||
background-color: #f8f9fa;
|
||||
border-radius: var(--radius-md);
|
||||
padding: 15px;
|
||||
margin-bottom: 15px;
|
||||
box-shadow: 0 2px 5px rgba(0, 0, 0, 0.05);
|
||||
}
|
||||
|
||||
.proposal-status {
|
||||
display: inline-block;
|
||||
padding: 5px 10px;
|
||||
border-radius: 15px;
|
||||
font-size: 0.8rem;
|
||||
margin-top: 10px;
|
||||
margin-bottom: 10px;
|
||||
font-weight: 500;
|
||||
}
|
||||
|
||||
.proposal-status.pending {
|
||||
background-color: #ffeeba;
|
||||
color: #856404;
|
||||
}
|
||||
|
||||
.proposal-status.active {
|
||||
background-color: #d1ecf1;
|
||||
color: #0c5460;
|
||||
}
|
||||
|
||||
.proposal-status.succeeded {
|
||||
background-color: #d4edda;
|
||||
color: #155724;
|
||||
}
|
||||
|
||||
.proposal-status.defeated {
|
||||
background-color: #f8d7da;
|
||||
color: #721c24;
|
||||
}
|
||||
|
||||
.proposal-status.executed {
|
||||
background-color: #d4edda;
|
||||
color: #155724;
|
||||
}
|
||||
|
||||
.proposal-actions {
|
||||
display: flex;
|
||||
gap: 10px;
|
||||
margin-top: 15px;
|
||||
}
|
||||
|
||||
/* Стили для секции модулей */
|
||||
.modules-list {
|
||||
display: flex;
|
||||
flex-wrap: wrap;
|
||||
gap: 15px;
|
||||
margin-top: 20px;
|
||||
}
|
||||
|
||||
.module-card {
|
||||
width: 300px;
|
||||
padding: 15px;
|
||||
background-color: #f8f9fa;
|
||||
border-radius: var(--radius-md);
|
||||
box-shadow: 0 2px 5px rgba(0, 0, 0, 0.05);
|
||||
}
|
||||
|
||||
.module-status {
|
||||
display: inline-block;
|
||||
padding: 3px 8px;
|
||||
border-radius: 12px;
|
||||
font-size: 0.8rem;
|
||||
margin-top: 10px;
|
||||
margin-bottom: 10px;
|
||||
background-color: #f8d7da;
|
||||
color: #721c24;
|
||||
}
|
||||
|
||||
.module-status.installed {
|
||||
background-color: #d4edda;
|
||||
color: #155724;
|
||||
}
|
||||
|
||||
.module-actions {
|
||||
margin-top: 15px;
|
||||
}
|
||||
|
||||
/* Стили для случая отсутствия DLE */
|
||||
.no-dle-message {
|
||||
padding: 20px;
|
||||
background-color: #f8f9fa;
|
||||
border-radius: var(--radius-md);
|
||||
text-align: center;
|
||||
}
|
||||
|
||||
.no-dle-message p {
|
||||
margin-bottom: 15px;
|
||||
}
|
||||
|
||||
/* Стили для секции управления */
|
||||
.governance-info {
|
||||
display: flex;
|
||||
flex-wrap: wrap;
|
||||
gap: 20px;
|
||||
}
|
||||
|
||||
.governance-info .info-card {
|
||||
flex: 1;
|
||||
min-width: 250px;
|
||||
}
|
||||
</style>
|
||||
@@ -7,27 +7,19 @@
|
||||
<div class="setting-form">
|
||||
<p>Настройка и деплой нового DLE (Digital Legal Entity) с токеном управления и контрактом Governor.</p>
|
||||
|
||||
<div class="form-group">
|
||||
<label class="form-label" for="blockchainNetwork">Цепочка блокчейна для деплоя:</label>
|
||||
<select id="blockchainNetwork" v-model="dleDeploymentSettings.blockchainNetwork" class="form-control">
|
||||
<option value="polygon">Polygon (Matic)</option>
|
||||
<option value="ethereum_mainnet">Ethereum Mainnet</option>
|
||||
<option value="sepolia">Sepolia (Testnet)</option>
|
||||
<option value="goerli">Goerli (Testnet)</option>
|
||||
<!-- TODO: Добавить другие сети по мере необходимости -->
|
||||
</select>
|
||||
</div>
|
||||
|
||||
<!-- 1. Имя DLE -->
|
||||
<div class="form-group">
|
||||
<label class="form-label" for="dleName">Имя DLE (Digital Legal Entity) (и токена):</label>
|
||||
<input type="text" id="dleName" v-model="dleDeploymentSettings.name" class="form-control" placeholder="Например, My DLE">
|
||||
</div>
|
||||
|
||||
<!-- 2. Символ токена -->
|
||||
<div class="form-group">
|
||||
<label class="form-label" for="dleSymbol">Символ токена управления (GT):</label>
|
||||
<input type="text" id="dleSymbol" v-model="dleDeploymentSettings.symbol" class="form-control" placeholder="Например, MDGT (3-5 символов)">
|
||||
</div>
|
||||
|
||||
<!-- 3. Местонахождение -->
|
||||
<h4>Местонахождение</h4>
|
||||
<div class="address-grid">
|
||||
<div class="form-group address-index">
|
||||
@@ -84,6 +76,8 @@
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- 4. Код деятельности -->
|
||||
<h4>Код деятельности</h4>
|
||||
<div v-if="dleDeploymentSettings.selectedIsicCodes && dleDeploymentSettings.selectedIsicCodes.length > 0" class="isic-codes-list mt-3">
|
||||
<h5>Добавленные коды деятельности:</h5>
|
||||
<ul>
|
||||
@@ -139,6 +133,7 @@
|
||||
<button @click="addIsicCode" class="btn btn-success btn-sm" :disabled="!currentSelectedIsicCode">Добавить код деятельности</button>
|
||||
</div>
|
||||
|
||||
<!-- 5. Первоначальное распределение токенов -->
|
||||
<h4>Первоначальное распределение токенов управления</h4>
|
||||
<div v-for="(partner, index) in dleDeploymentSettings.partners" :key="index" class="partner-entry">
|
||||
<div class="form-group">
|
||||
@@ -156,6 +151,7 @@
|
||||
<label class="form-label">Общее количество выпускаемых GT: {{ totalInitialSupply }}</label>
|
||||
</div>
|
||||
|
||||
<!-- 6. Настройки Governor -->
|
||||
<h4>Настройки Governor</h4>
|
||||
<div class="form-group">
|
||||
<label class="form-label" for="proposalThreshold">Порог для создания предложений (кол-во GT):</label>
|
||||
@@ -177,13 +173,148 @@
|
||||
<input type="number" id="votingPeriod" v-model="dleDeploymentSettings.votingPeriodDays" min="1" class="form-control">
|
||||
</div>
|
||||
|
||||
<h4>Настройки Timelock (если используется)</h4>
|
||||
<div class="form-group">
|
||||
<label class="form-label" for="timelockMinDelay">Минимальная задержка Timelock (в днях):</label>
|
||||
<input type="number" id="timelockMinDelay" v-model="dleDeploymentSettings.timelockMinDelayDays" min="0" class="form-control">
|
||||
</div>
|
||||
|
||||
<button class="btn btn-primary btn-lg" @click="deployDLE">Создать и задеплоить DLE (Digital Legal Entity)</button>
|
||||
<!-- 7. RPC Провайдеры -->
|
||||
<h4>RPC Провайдеры</h4>
|
||||
<p>Конфигурации RPC для сетей, которые будут использоваться в приложении.</p>
|
||||
|
||||
<!-- Список добавленных RPC -->
|
||||
<div v-if="securitySettings.rpcConfigs.length > 0" class="rpc-list">
|
||||
<h5>Добавленные RPC конфигурации:</h5>
|
||||
<div v-for="(rpc, index) in securitySettings.rpcConfigs" :key="index" class="rpc-entry">
|
||||
<span><strong>ID Сети:</strong> {{ rpc.networkId }}</span>
|
||||
<span><strong>URL:</strong> {{ rpc.rpcUrl }}</span>
|
||||
<div class="rpc-actions">
|
||||
<button class="btn btn-info btn-sm" @click="testRpcHandler(rpc)" :disabled="testingRpc && testingRpcId === rpc.networkId">
|
||||
<i class="fas" :class="testingRpc && testingRpcId === rpc.networkId ? 'fa-spinner fa-spin' : 'fa-check-circle'"></i>
|
||||
{{ testingRpc && testingRpcId === rpc.networkId ? 'Проверка...' : 'Тест' }}
|
||||
</button>
|
||||
<button class="btn btn-danger btn-sm" @click="removeRpcConfig(index)">
|
||||
<i class="fas fa-trash"></i> Удалить
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
<p v-else>Нет добавленных RPC конфигураций.</p>
|
||||
|
||||
<!-- Форма добавления нового RPC -->
|
||||
<div class="setting-form add-rpc-form">
|
||||
<h5>Добавить новую RPC конфигурацию:</h5>
|
||||
<div class="form-group">
|
||||
<label class="form-label" for="newRpcNetworkId">ID Сети:</label>
|
||||
<select id="newRpcNetworkId" v-model="networkEntry.networkId" class="form-control">
|
||||
<optgroup v-for="(group, groupIndex) in networkGroups" :key="groupIndex" :label="group.label">
|
||||
<option v-for="option in group.options" :key="option.value" :value="option.value">
|
||||
{{ option.label }}
|
||||
</option>
|
||||
</optgroup>
|
||||
</select>
|
||||
<div v-if="networkEntry.networkId === 'custom'" class="mt-2">
|
||||
<label class="form-label" for="customNetworkId">Пользовательский ID:</label>
|
||||
<input type="text" id="customNetworkId" v-model="networkEntry.customNetworkId" class="form-control" placeholder="Введите ID сети">
|
||||
|
||||
<label class="form-label mt-2" for="customChainId">Chain ID:</label>
|
||||
<input type="number" id="customChainId" v-model="networkEntry.customChainId" class="form-control" placeholder="Например, 1 для Ethereum">
|
||||
<small>Chain ID - уникальный идентификатор блокчейн-сети (целое число)</small>
|
||||
</div>
|
||||
<small>ID сети должен совпадать со значением в выпадающем списке сетей при создании DLE</small>
|
||||
</div>
|
||||
<div class="form-group">
|
||||
<label class="form-label" for="newRpcUrl">RPC URL:</label>
|
||||
<input type="text" id="newRpcUrl" v-model="networkEntry.rpcUrl" class="form-control" placeholder="https://...">
|
||||
<!-- Предложение URL на основе выбранной сети -->
|
||||
<small v-if="defaultRpcUrlSuggestion" class="suggestion">
|
||||
Предложение: {{ defaultRpcUrlSuggestion }}
|
||||
<button class="btn-link" @click="useDefaultRpcUrl">Использовать</button>
|
||||
</small>
|
||||
</div>
|
||||
<button class="btn btn-secondary" @click="addRpcConfig">Добавить RPC</button>
|
||||
</div>
|
||||
|
||||
<!-- Кнопка сохранения настроек RPC -->
|
||||
<div class="save-rpc-actions mt-3">
|
||||
<button class="btn btn-primary" @click="saveRpcSettingsWithFeedback" :disabled="isSavingRpc">
|
||||
<i class="fas" :class="isSavingRpc ? 'fa-spinner fa-spin' : 'fa-save'"></i>
|
||||
{{ isSavingRpc ? 'Сохранение...' : 'Сохранить RPC настройки' }}
|
||||
</button>
|
||||
</div>
|
||||
|
||||
<!-- 8. Выбор сети для деплоя -->
|
||||
<h4>Сеть для деплоя</h4>
|
||||
<div class="form-group">
|
||||
<label class="form-label" for="deployNetwork">Выберите сеть блокчейн для деплоя:</label>
|
||||
<select id="deployNetwork" v-model="dleDeploymentSettings.blockchainNetwork" class="form-control">
|
||||
<option v-for="network in networks" :key="network.value" :value="network.value">
|
||||
{{ network.label }}
|
||||
</option>
|
||||
</select>
|
||||
<small class="text-warning" v-if="!dleDeploymentSettings.blockchainNetwork.includes('testnet') &&
|
||||
!['sepolia', 'goerli', 'mumbai'].includes(dleDeploymentSettings.blockchainNetwork)">
|
||||
<i class="fas fa-exclamation-triangle"></i>
|
||||
Внимание! Для тестирования рекомендуется использовать тестовые сети (Sepolia, Goerli, Mumbai).
|
||||
</small>
|
||||
</div>
|
||||
|
||||
<!-- 9. Ключ Деплоера -->
|
||||
<h4>Ключ Деплоера</h4>
|
||||
<div class="form-group">
|
||||
<label class="form-label" for="deployerKey">Приватный ключ для деплоя:</label>
|
||||
<div class="input-group">
|
||||
<input :type="showDeployerKey ? 'text' : 'password'" id="deployerKey" v-model="securitySettings.deployerPrivateKey" class="form-control">
|
||||
<button class="btn btn-outline-secondary" @click="toggleShowDeployerKey">
|
||||
<i :class="showDeployerKey ? 'fas fa-eye-slash' : 'fas fa-eye'"></i>
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- 10. Газовые настройки -->
|
||||
<div class="form-group">
|
||||
<h4>Газовые настройки</h4>
|
||||
<div class="custom-control custom-checkbox">
|
||||
<input type="checkbox" class="custom-control-input" id="customGas" v-model="useCustomGas">
|
||||
<label class="custom-control-label" for="customGas">Использовать пользовательские настройки газа</label>
|
||||
</div>
|
||||
|
||||
<div v-if="useCustomGas" class="gas-settings mt-3">
|
||||
<div class="form-group">
|
||||
<label for="gasLimit">Лимит газа (Gas Limit):</label>
|
||||
<input type="number" id="gasLimit" v-model="gasSettings.gasLimit" class="form-control">
|
||||
</div>
|
||||
<div class="form-group">
|
||||
<label for="maxFeePerGas">Максимальная комиссия (Max Fee, gwei):</label>
|
||||
<input type="number" id="maxFeePerGas" v-model="gasSettings.maxFeePerGas" class="form-control">
|
||||
</div>
|
||||
<div class="form-group">
|
||||
<label for="maxPriorityFee">Приоритетная комиссия (Priority Fee, gwei):</label>
|
||||
<input type="number" id="maxPriorityFee" v-model="gasSettings.maxPriorityFee" class="form-control">
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- 10. Кнопка деплоя DLE -->
|
||||
<div class="deployment-actions mt-4">
|
||||
<button class="btn btn-primary" @click="deployDLE" :disabled="isDeploying">
|
||||
<i class="fas fa-rocket"></i> {{ isDeploying ? 'Создание DLE...' : 'Создать и задеплоить DLE (Digital Legal Entity)' }}
|
||||
</button>
|
||||
|
||||
<!-- Результат деплоя -->
|
||||
<div v-if="deployResult" class="deploy-result mt-3 alert alert-success">
|
||||
<h5>DLE успешно создано!</h5>
|
||||
<p><strong>Адрес токена:</strong> {{ deployResult.data?.tokenAddress }}</p>
|
||||
<p><strong>Адрес таймлока:</strong> {{ deployResult.data?.timelockAddress }}</p>
|
||||
<p><strong>Адрес контракта Governor:</strong> {{ deployResult.data?.governorAddress }}</p>
|
||||
</div>
|
||||
|
||||
<!-- Ошибка деплоя -->
|
||||
<div v-if="deployError" class="deploy-error mt-3 alert alert-danger">
|
||||
<h5>Ошибка при создании DLE</h5>
|
||||
<p>{{ deployError }}</p>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
@@ -193,8 +324,61 @@
|
||||
<script setup>
|
||||
import { reactive, onMounted, computed, ref, watch } from 'vue';
|
||||
import axios from 'axios'; // Предполагаем, что axios доступен
|
||||
import { useAuth } from '@/composables/useAuth'; // Импортируем composable useAuth
|
||||
import dleService from '@/services/dleService';
|
||||
import useBlockchainNetworks from '@/composables/useBlockchainNetworks'; // Импортируем composable для работы с сетями
|
||||
// TODO: Импортировать API
|
||||
|
||||
const { address, isAdmin, auth, user } = useAuth(); // Получаем объект адреса и статус админа
|
||||
|
||||
// Инициализация composable для работы с сетями блокчейн
|
||||
const {
|
||||
networkGroups,
|
||||
networkEntry,
|
||||
defaultRpcUrlSuggestion,
|
||||
useDefaultRpcUrl,
|
||||
validateAndPrepareNetworkConfig,
|
||||
resetNetworkEntry,
|
||||
testRpcConnection,
|
||||
testingRpc,
|
||||
testingRpcId,
|
||||
networks
|
||||
} = useBlockchainNetworks();
|
||||
|
||||
// Добавляем настройки безопасности и подключения
|
||||
const securitySettings = reactive({
|
||||
rpcConfigs: [], // Массив для хранения { networkId: string, rpcUrl: string, chainId: number }
|
||||
deployerPrivateKey: '',
|
||||
});
|
||||
|
||||
// Функция добавления новой RPC конфигурации
|
||||
const addRpcConfig = () => {
|
||||
const result = validateAndPrepareNetworkConfig();
|
||||
|
||||
if (!result.valid) {
|
||||
alert(result.error);
|
||||
return;
|
||||
}
|
||||
|
||||
const { networkId, rpcUrl, chainId } = result.networkConfig;
|
||||
|
||||
// Проверка на дубликат ID
|
||||
if (securitySettings.rpcConfigs.some(rpc => rpc.networkId === networkId)) {
|
||||
alert(`Ошибка: RPC конфигурация для сети с ID '${networkId}' уже существует.`);
|
||||
return;
|
||||
}
|
||||
|
||||
securitySettings.rpcConfigs.push({ networkId, rpcUrl, chainId });
|
||||
|
||||
// Очистка полей ввода
|
||||
resetNetworkEntry();
|
||||
};
|
||||
|
||||
// Функция удаления RPC конфигурации
|
||||
const removeRpcConfig = (index) => {
|
||||
securitySettings.rpcConfigs.splice(index, 1);
|
||||
};
|
||||
|
||||
const settings = reactive({
|
||||
// contractAddress: '', // Удалено
|
||||
// quorumPercent: 51, // Удалено
|
||||
@@ -220,6 +404,16 @@ const dleDeploymentSettings = reactive({
|
||||
selectedIsicCodes: [], // <<< Для хранения массива выбранных кодов ISIC
|
||||
});
|
||||
|
||||
// Добавляем переменную useCustomGas для управления отображением пользовательских настроек газа
|
||||
const useCustomGas = ref(false);
|
||||
|
||||
// Объявляем gasSettings как ref для корректного реактивного доступа в шаблоне
|
||||
const gasSettings = reactive({
|
||||
gasLimit: 3000000,
|
||||
maxFeePerGas: 30,
|
||||
maxPriorityFee: 2
|
||||
});
|
||||
|
||||
// --- Состояние для загрузки и опций ISIC ---
|
||||
const sectionOptions = ref([]);
|
||||
const divisionOptions = ref([]);
|
||||
@@ -360,8 +554,21 @@ watch(selectedClass, () => {
|
||||
// --- Начальная загрузка данных ---
|
||||
onMounted(() => {
|
||||
fetchIsicCodes({ level: 1 }, sectionOptions, isLoadingSections);
|
||||
// TODO: Загрузить настройки блокчейна, если они есть
|
||||
// loadBlockchainSettings(); // Эта функция пока не актуальна, так как settings пустой
|
||||
|
||||
// Автоподстановка адреса авторизированного пользователя в первого партнера, если есть права админа
|
||||
if (address.value && isAdmin.value && dleDeploymentSettings.partners.length > 0) {
|
||||
dleDeploymentSettings.partners[0].address = address.value;
|
||||
}
|
||||
|
||||
// Слушаем изменения адреса авторизированного пользователя
|
||||
watch(address, (newAddress) => {
|
||||
if (newAddress && isAdmin.value && dleDeploymentSettings.partners.length > 0) {
|
||||
dleDeploymentSettings.partners[0].address = newAddress;
|
||||
}
|
||||
});
|
||||
|
||||
// Загрузка настроек RPC с сервера
|
||||
loadRpcSettings();
|
||||
});
|
||||
|
||||
const totalInitialSupply = computed(() => {
|
||||
@@ -389,46 +596,112 @@ const saveSettings = async (section) => {
|
||||
// Если настройки DLE (dleDeploymentSettings) нужно сохранять без деплоя, нужна другая логика.
|
||||
};
|
||||
|
||||
const deployDLE = async () => {
|
||||
console.log('[BlockchainSettingsView] Попытка деплоя DLE (Digital Legal Entity) с настройками:', JSON.parse(JSON.stringify(dleDeploymentSettings)));
|
||||
console.log('[BlockchainSettingsView] Выбранные коды ISIC:', JSON.parse(JSON.stringify(dleDeploymentSettings.selectedIsicCodes)));
|
||||
console.log('[BlockchainSettingsView] Общее начальное количество токенов:', totalInitialSupply.value);
|
||||
const addressString = [
|
||||
const isDeploying = ref(false);
|
||||
const deployResult = ref(null);
|
||||
const deployError = ref(null);
|
||||
|
||||
const formattedDLEParams = computed(() => {
|
||||
// Преобразуем партнеров в формат для API
|
||||
const partners = dleDeploymentSettings.partners.map(p => p.address);
|
||||
const amounts = dleDeploymentSettings.partners.map(p => p.amount.toString());
|
||||
|
||||
// Формируем полный адрес
|
||||
const location = [
|
||||
dleDeploymentSettings.locationIndex,
|
||||
dleDeploymentSettings.locationCountry,
|
||||
dleDeploymentSettings.locationCity,
|
||||
dleDeploymentSettings.locationStreet,
|
||||
dleDeploymentSettings.locationHouse,
|
||||
dleDeploymentSettings.locationOffice
|
||||
].filter(Boolean).join(', '); // Собираем строку адреса для alert/log
|
||||
let finalIsicDisplay = 'Не выбраны';
|
||||
if (dleDeploymentSettings.selectedIsicCodes && dleDeploymentSettings.selectedIsicCodes.length > 0) {
|
||||
finalIsicDisplay = dleDeploymentSettings.selectedIsicCodes.map(c => c.code).join(', ');
|
||||
].filter(Boolean).join(', ');
|
||||
|
||||
// Формируем коды ISIC
|
||||
const isicCodes = dleDeploymentSettings.selectedIsicCodes.map(isic => isic.code);
|
||||
|
||||
return {
|
||||
name: dleDeploymentSettings.name,
|
||||
symbol: dleDeploymentSettings.symbol,
|
||||
location,
|
||||
isicCodes,
|
||||
partners,
|
||||
amounts,
|
||||
network: dleDeploymentSettings.blockchainNetwork, // Добавляем выбранную сеть в параметры
|
||||
minTimelockDelay: dleDeploymentSettings.timelockMinDelayDays,
|
||||
votingDelay: Math.round(dleDeploymentSettings.votingDelayDays * 24 * 60 * 60 / 13), // конвертируем дни в блоки (13 секунд на блок)
|
||||
votingPeriod: Math.round(dleDeploymentSettings.votingPeriodDays * 24 * 60 * 60 / 13), // конвертируем дни в блоки
|
||||
proposalThreshold: dleDeploymentSettings.proposalThreshold,
|
||||
quorumPercentage: dleDeploymentSettings.quorumPercent
|
||||
};
|
||||
});
|
||||
|
||||
const deployDLE = async () => {
|
||||
isDeploying.value = true;
|
||||
deployResult.value = null;
|
||||
deployError.value = null;
|
||||
|
||||
try {
|
||||
// Проверяем валидность формы
|
||||
if (!validateDLEForm()) {
|
||||
isDeploying.value = false;
|
||||
return;
|
||||
}
|
||||
|
||||
// Сначала сохраняем настройки RPC
|
||||
const rpcSaved = await saveRpcSettings();
|
||||
if (!rpcSaved) {
|
||||
// Если не удалось сохранить, спрашиваем пользователя, хочет ли он продолжить
|
||||
if (!confirm('Не удалось сохранить RPC настройки. Продолжить деплой DLE?')) {
|
||||
isDeploying.value = false;
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
// Отправляем запрос на создание DLE
|
||||
const result = await dleService.createDLE(formattedDLEParams.value);
|
||||
|
||||
deployResult.value = result;
|
||||
alert('DLE успешно создано!');
|
||||
} catch (error) {
|
||||
console.error('Ошибка при деплое DLE:', error);
|
||||
deployError.value = error.response?.data?.message || error.message || 'Произошла ошибка при деплое DLE';
|
||||
alert(deployError.value);
|
||||
} finally {
|
||||
isDeploying.value = false;
|
||||
}
|
||||
alert(`Деплой DLE (Digital Legal Entity) инициирован (см. консоль). Имя: ${dleDeploymentSettings.name}, Символ: ${dleDeploymentSettings.symbol}, Сеть: ${dleDeploymentSettings.blockchainNetwork}, Коды деят: ${finalIsicDisplay}, Адрес: ${addressString || 'Не указан'}`);
|
||||
// TODO: Вызвать API бэкенда для деплоя контрактов
|
||||
// Передать dleDeploymentSettings.blockchainNetwork на бэкенд
|
||||
// Передать выбранный dleDeploymentSettings.selectedIsicCodes и детали адреса (dleDeploymentSettings.location...) (возможно, для метаданных контракта или внесения в реестр)
|
||||
// 1. Деплой ERC20Votes токена с параметрами:
|
||||
// - name: dleDeploymentSettings.name
|
||||
// - symbol: dleDeploymentSettings.symbol
|
||||
// - initialSupply: totalInitialSupply.value (или как решит бэкенд по партнерам)
|
||||
// - initialHolders: dleDeploymentSettings.partners (адреса и суммы)
|
||||
// 2. Деплой TimelockController (если используется, получить его адрес)
|
||||
// - minDelay: dleDeploymentSettings.timelockMinDelayDays * 24 * 60 * 60 (конвертация в секунды)
|
||||
// - proposers: [адрес будущего Governor]
|
||||
// - executors: [0x0000...0000] (любой может исполнить) или [адрес будущего Governor]
|
||||
// 3. Деплой Governor контракта:
|
||||
// - name: dleDeploymentSettings.name + " Governor" (или просто уникальное имя для Governor)
|
||||
// - tokenAddress: адрес задеплоенного ERC20Votes
|
||||
// - timelockAddress: адрес задеплоенного TimelockController (если используется)
|
||||
// - quorumPercent: dleDeploymentSettings.quorumPercent
|
||||
// - proposalThreshold: dleDeploymentSettings.proposalThreshold
|
||||
// - votingPeriod: dleDeploymentSettings.votingPeriodDays * 24 * 60 * 60 (конвертация в секунды или блоки на бэкенде)
|
||||
// - votingDelay: dleDeploymentSettings.votingDelayDays * 24 * 60 * 60 (конвертация в секунды или блоки на бэкенде)
|
||||
// 4. (Если Timelock) Передать права администратора Timelock самому Timelock'у (или DLE)
|
||||
// 5. (Опционально) Передать права на другие системные контракты Timelock'у
|
||||
// TODO: Передать dleDeploymentSettings.selectedIsicCodes (массив объектов) на бэкенд
|
||||
};
|
||||
|
||||
const validateDLEForm = () => {
|
||||
// Проверяем обязательные поля
|
||||
if (!dleDeploymentSettings.name) {
|
||||
alert('Необходимо указать имя DLE');
|
||||
return false;
|
||||
}
|
||||
|
||||
if (!dleDeploymentSettings.symbol) {
|
||||
alert('Необходимо указать символ токена');
|
||||
return false;
|
||||
}
|
||||
|
||||
// Проверяем выбрана ли сеть для деплоя
|
||||
if (!dleDeploymentSettings.blockchainNetwork) {
|
||||
alert('Необходимо выбрать сеть для деплоя');
|
||||
return false;
|
||||
}
|
||||
|
||||
// Проверяем адреса партнеров
|
||||
for (const partner of dleDeploymentSettings.partners) {
|
||||
if (!partner.address || !partner.address.startsWith('0x') || partner.address.length !== 42) {
|
||||
alert('Некорректный адрес партнера');
|
||||
return false;
|
||||
}
|
||||
|
||||
if (!partner.amount || partner.amount <= 0) {
|
||||
alert('Сумма токенов должна быть больше 0');
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
return true;
|
||||
};
|
||||
|
||||
// --- Функция для поиска адреса по индексу через Nominatim ---
|
||||
@@ -636,6 +909,91 @@ const removeIsicCode = (index) => {
|
||||
dleDeploymentSettings.selectedIsicCodes.splice(index, 1);
|
||||
};
|
||||
|
||||
const showDeployerKey = ref(false);
|
||||
const toggleShowDeployerKey = () => {
|
||||
showDeployerKey.value = !showDeployerKey.value;
|
||||
};
|
||||
|
||||
// Функция загрузки настроек RPC с сервера
|
||||
const loadRpcSettings = async () => {
|
||||
try {
|
||||
const response = await axios.get('/api/settings/rpc');
|
||||
if (response.data && response.data.success) {
|
||||
securitySettings.rpcConfigs = response.data.data || [];
|
||||
console.log('[BlockchainSettingsView] RPC конфигурации успешно загружены:', securitySettings.rpcConfigs);
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('[BlockchainSettingsView] Ошибка при загрузке RPC конфигураций:', error);
|
||||
// Если нужно, установить дефолтные RPC
|
||||
// setDefaultRpcConfigs();
|
||||
}
|
||||
};
|
||||
|
||||
// Функция сохранения настроек RPC на сервер
|
||||
const saveRpcSettings = async () => {
|
||||
try {
|
||||
const response = await axios.post('/api/settings/rpc', {
|
||||
rpcConfigs: JSON.parse(JSON.stringify(securitySettings.rpcConfigs))
|
||||
});
|
||||
|
||||
if (response.data && response.data.success) {
|
||||
console.log('[BlockchainSettingsView] RPC конфигурации успешно сохранены');
|
||||
return true;
|
||||
} else {
|
||||
console.error('[BlockchainSettingsView] Ошибка при сохранении RPC конфигураций:', response.data);
|
||||
return false;
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('[BlockchainSettingsView] Ошибка при сохранении RPC конфигураций:', error);
|
||||
return false;
|
||||
}
|
||||
};
|
||||
|
||||
const isSavingRpc = ref(false);
|
||||
|
||||
// Функция сохранения настроек RPC с обратной связью
|
||||
const saveRpcSettingsWithFeedback = async () => {
|
||||
isSavingRpc.value = true;
|
||||
try {
|
||||
const success = await saveRpcSettings();
|
||||
if (success) {
|
||||
alert('RPC настройки успешно сохранены.');
|
||||
} else {
|
||||
alert('Ошибка при сохранении RPC настроек.');
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('[BlockchainSettingsView] Ошибка при сохранении RPC настроек:', error);
|
||||
alert(`Ошибка при сохранении: ${error.message || 'Неизвестная ошибка'}`);
|
||||
} finally {
|
||||
isSavingRpc.value = false;
|
||||
}
|
||||
};
|
||||
|
||||
// Определяем группы сетей для деплоя (исключаем локальные и пользовательские)
|
||||
const deployNetworkGroups = computed(() => {
|
||||
// Фильтруем группы, оставляя только основные и тестовые сети
|
||||
return networkGroups.filter(group =>
|
||||
group.label === 'Основные сети' || group.label === 'Тестовые сети'
|
||||
);
|
||||
});
|
||||
|
||||
const testingRpcIndex = ref(-1);
|
||||
|
||||
// Функция-обработчик для тестирования RPC соединения
|
||||
const testRpcHandler = async (rpc) => {
|
||||
try {
|
||||
const result = await testRpcConnection(rpc.networkId, rpc.rpcUrl);
|
||||
if (result.success) {
|
||||
alert(result.message);
|
||||
} else {
|
||||
alert(`Ошибка при подключении к ${rpc.networkId}: ${result.error}`);
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('[BlockchainSettingsView] Ошибка при тестировании RPC:', error);
|
||||
alert(`Ошибка при тестировании RPC: ${error.message || 'Неизвестная ошибка'}`);
|
||||
}
|
||||
};
|
||||
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
@@ -815,4 +1173,117 @@ h3 {
|
||||
padding: 0.2rem 0.4rem;
|
||||
font-size: 0.75rem;
|
||||
}
|
||||
|
||||
.rpc-list {
|
||||
margin-bottom: var(--spacing-md);
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: var(--spacing-sm);
|
||||
}
|
||||
|
||||
.rpc-entry {
|
||||
background-color: var(--color-background);
|
||||
padding: var(--spacing-xs) var(--spacing-sm);
|
||||
border: 1px solid var(--color-grey-light);
|
||||
border-radius: var(--radius-sm);
|
||||
margin-bottom: var(--spacing-xs);
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: var(--spacing-md);
|
||||
}
|
||||
|
||||
.rpc-entry span {
|
||||
flex-grow: 1;
|
||||
}
|
||||
|
||||
.add-rpc-form {
|
||||
margin-top: var(--spacing-sm);
|
||||
padding-top: var(--spacing-sm);
|
||||
border-top: 1px dashed var(--color-grey-light);
|
||||
}
|
||||
|
||||
.add-rpc-form h5 {
|
||||
margin-bottom: var(--spacing-sm);
|
||||
}
|
||||
|
||||
.deployment-actions {
|
||||
margin-top: var(--spacing-md);
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
align-items: flex-start;
|
||||
}
|
||||
|
||||
.deploy-result {
|
||||
margin-top: var(--spacing-md);
|
||||
padding: var(--spacing-md);
|
||||
border-radius: var(--radius-sm);
|
||||
background-color: var(--color-success-light);
|
||||
color: var(--color-success-dark);
|
||||
border: 1px solid var(--color-success-dark);
|
||||
}
|
||||
|
||||
.deploy-error {
|
||||
margin-top: var(--spacing-md);
|
||||
padding: var(--spacing-md);
|
||||
border-radius: var(--radius-sm);
|
||||
background-color: var(--color-danger-light);
|
||||
color: var(--color-danger-dark);
|
||||
border: 1px solid var(--color-danger-dark);
|
||||
}
|
||||
|
||||
.suggestion {
|
||||
background-color: rgba(76, 175, 80, 0.1);
|
||||
border-left: 3px solid var(--color-primary, #4caf50);
|
||||
padding: 6px 10px;
|
||||
margin-top: 8px;
|
||||
border-radius: 0 4px 4px 0;
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: center;
|
||||
}
|
||||
|
||||
.btn-link {
|
||||
background: none;
|
||||
border: none;
|
||||
padding: 0;
|
||||
color: var(--color-primary, #4caf50);
|
||||
text-decoration: underline;
|
||||
cursor: pointer;
|
||||
font-size: 0.875rem;
|
||||
}
|
||||
|
||||
.btn-link:hover {
|
||||
color: var(--color-primary-dark, #388e3c);
|
||||
text-decoration: none;
|
||||
}
|
||||
|
||||
.mt-2 {
|
||||
margin-top: 10px;
|
||||
}
|
||||
|
||||
.text-warning {
|
||||
color: #f57c00; /* оранжевый для предупреждений */
|
||||
margin-top: 5px;
|
||||
display: block;
|
||||
}
|
||||
|
||||
.rpc-actions {
|
||||
display: flex;
|
||||
gap: 8px;
|
||||
align-items: center;
|
||||
}
|
||||
|
||||
.btn-info {
|
||||
background-color: #17a2b8;
|
||||
color: white;
|
||||
}
|
||||
|
||||
.btn-info:hover:not(:disabled) {
|
||||
background-color: #138496;
|
||||
}
|
||||
|
||||
.btn-info:disabled {
|
||||
background-color: #a0d2dc;
|
||||
cursor: not-allowed;
|
||||
}
|
||||
</style>
|
||||
File diff suppressed because it is too large
Load Diff
1826
frontend/yarn.lock
1826
frontend/yarn.lock
File diff suppressed because it is too large
Load Diff
Reference in New Issue
Block a user