mirror of
https://github.com/basicswap/basicswap.git
synced 2025-11-05 18:38:09 +01:00
* GUI: Multi-select coin filtering / Various fixes. * Use coin-manager / clean-up. * Fix BCH in filters + fix UX with bid pages modals when amount is empty. * Fix amount not empty. * Abandon Bid under debug_ui
671 lines
29 KiB
JavaScript
671 lines
29 KiB
JavaScript
const PAGE_SIZE = 50;
|
|
|
|
const state = {
|
|
dentities: new Map(),
|
|
currentPage: 1,
|
|
wsConnected: false,
|
|
jsonData: [],
|
|
isLoading: false,
|
|
isRefreshing: false,
|
|
refreshPromise: null
|
|
};
|
|
|
|
const elements = {
|
|
bidsBody: document.getElementById('bids-body'),
|
|
prevPageButton: document.getElementById('prevPage'),
|
|
nextPageButton: document.getElementById('nextPage'),
|
|
currentPageSpan: document.getElementById('currentPage'),
|
|
paginationControls: document.getElementById('pagination-controls'),
|
|
availableBidsCount: document.getElementById('availableBidsCount'),
|
|
refreshBidsButton: document.getElementById('refreshBids'),
|
|
statusDot: document.getElementById('status-dot'),
|
|
statusText: document.getElementById('status-text')
|
|
};
|
|
|
|
const formatTimeAgo = (timestamp) => {
|
|
const now = Math.floor(Date.now() / 1000);
|
|
const diff = now - timestamp;
|
|
|
|
if (diff < 60) return `${diff} seconds ago`;
|
|
if (diff < 3600) return `${Math.floor(diff / 60)} minutes ago`;
|
|
if (diff < 86400) return `${Math.floor(diff / 3600)} hours ago`;
|
|
return `${Math.floor(diff / 86400)} days ago`;
|
|
};
|
|
|
|
const formatTime = (timestamp) => {
|
|
const now = Math.floor(Date.now() / 1000);
|
|
const diff = timestamp - now;
|
|
|
|
if (diff <= 0) return "Expired";
|
|
if (diff < 60) return `${diff} seconds`;
|
|
if (diff < 3600) return `${Math.floor(diff / 60)} minutes`;
|
|
if (diff < 86400) return `${Math.floor(diff / 3600)} hours`;
|
|
return `${Math.floor(diff / 86400)} days`;
|
|
};
|
|
|
|
const formatAddress = (address, displayLength = 15) => {
|
|
if (!address) return '';
|
|
if (address.length <= displayLength) return address;
|
|
return `${address.slice(0, displayLength)}...`;
|
|
};
|
|
|
|
const getTimeStrokeColor = (expireTime) => {
|
|
const now = Math.floor(Date.now() / 1000);
|
|
const timeLeft = expireTime - now;
|
|
|
|
if (timeLeft <= 300) return '#9CA3AF'; // 5 minutes or less
|
|
if (timeLeft <= 1800) return '#3B82F6'; // 30 minutes or less
|
|
return '#10B981'; // More than 30 minutes
|
|
};
|
|
|
|
const createTimeTooltip = (bid) => {
|
|
const postedTime = formatTimeAgo(bid.created_at);
|
|
const expiresIn = formatTime(bid.expire_at);
|
|
return `
|
|
<div class="active-revoked-expired">
|
|
<span class="bold">
|
|
<div class="text-xs"><span class="bold">Posted:</span> ${postedTime}</div>
|
|
<div class="text-xs"><span class="bold">Expires in:</span> ${expiresIn}</div>
|
|
</span>
|
|
</div>
|
|
<div class="mt-5 text-xs">
|
|
<p class="font-bold mb-3">Time Indicator Colors:</p>
|
|
<p class="flex items-center">
|
|
<svg class="w-5 h-5 mr-3" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24">
|
|
<g stroke-linecap="round" stroke-width="2" fill="none" stroke="#10B981" stroke-linejoin="round">
|
|
<circle cx="12" cy="12" r="11"></circle>
|
|
<polyline points="12,6 12,12 18,12" stroke="#10B981"></polyline>
|
|
</g>
|
|
</svg>
|
|
Green: More than 30 minutes left
|
|
</p>
|
|
<p class="flex items-center mt-3">
|
|
<svg class="w-5 h-5 mr-3" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24">
|
|
<g stroke-linecap="round" stroke-width="2" fill="none" stroke="#3B82F6" stroke-linejoin="round">
|
|
<circle cx="12" cy="12" r="11"></circle>
|
|
<polyline points="12,6 12,12 18,12" stroke="#3B82F6"></polyline>
|
|
</g>
|
|
</svg>
|
|
Blue: Between 5 and 30 minutes left
|
|
</p>
|
|
<p class="flex items-center mt-3 mb-3">
|
|
<svg class="w-5 h-5 mr-3" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24">
|
|
<g stroke-linecap="round" stroke-width="2" fill="none" stroke="#9CA3AF" stroke-linejoin="round">
|
|
<circle cx="12" cy="12" r="11"></circle>
|
|
<polyline points="12,6 12,12 18,12" stroke="#9CA3AF"></polyline>
|
|
</g>
|
|
</svg>
|
|
Grey: Less than 5 minutes left or expired
|
|
</p>
|
|
</div>
|
|
`;
|
|
};
|
|
|
|
const safeParseInt = (value) => {
|
|
const parsed = parseInt(value);
|
|
return isNaN(parsed) ? 0 : parsed;
|
|
};
|
|
|
|
const processIdentityStats = (identity) => {
|
|
if (!identity) return null;
|
|
|
|
const stats = {
|
|
sentSuccessful: safeParseInt(identity.num_sent_bids_successful),
|
|
recvSuccessful: safeParseInt(identity.num_recv_bids_successful),
|
|
sentFailed: safeParseInt(identity.num_sent_bids_failed),
|
|
recvFailed: safeParseInt(identity.num_recv_bids_failed),
|
|
sentRejected: safeParseInt(identity.num_sent_bids_rejected),
|
|
recvRejected: safeParseInt(identity.num_recv_bids_rejected)
|
|
};
|
|
|
|
stats.totalSuccessful = stats.sentSuccessful + stats.recvSuccessful;
|
|
stats.totalFailed = stats.sentFailed + stats.recvFailed;
|
|
stats.totalRejected = stats.sentRejected + stats.recvRejected;
|
|
stats.totalBids = stats.totalSuccessful + stats.totalFailed + stats.totalRejected;
|
|
|
|
stats.successRate = stats.totalBids > 0
|
|
? ((stats.totalSuccessful / stats.totalBids) * 100).toFixed(1)
|
|
: '0.0';
|
|
|
|
return stats;
|
|
};
|
|
|
|
const createIdentityTooltip = (identity) => {
|
|
if (!identity) return '';
|
|
|
|
const stats = processIdentityStats(identity);
|
|
if (!stats) return '';
|
|
|
|
const getSuccessRateColor = (rate) => {
|
|
const numRate = parseFloat(rate);
|
|
if (numRate >= 80) return 'text-green-600';
|
|
if (numRate >= 60) return 'text-yellow-600';
|
|
return 'text-red-600';
|
|
};
|
|
|
|
return `
|
|
<div class="identity-info space-y-2">
|
|
${identity.label ? `
|
|
<div class="border-b border-gray-400 pb-2">
|
|
<div class="text-white text-xs tracking-wide font-semibold">Label:</div>
|
|
<div class="text-white">${identity.label}</div>
|
|
</div>
|
|
` : ''}
|
|
|
|
<div class="space-y-1">
|
|
<div class="text-white text-xs tracking-wide font-semibold">Bid From Address:</div>
|
|
<div class="monospace text-xs break-all bg-gray-500 p-2 rounded-md text-white">
|
|
${identity.address || ''}
|
|
</div>
|
|
</div>
|
|
|
|
${identity.note ? `
|
|
<div class="space-y-1">
|
|
<div class="text-white text-xs tracking-wide font-semibold">Note:</div>
|
|
<div class="text-white text-sm italic">${identity.note}</div>
|
|
</div>
|
|
` : ''}
|
|
|
|
<div class="pt-2 mt-2">
|
|
<div class="text-white text-xs tracking-wide font-semibold mb-2">Swap History:</div>
|
|
<div class="grid grid-cols-2 gap-2">
|
|
<div class="text-center p-2 bg-gray-500 rounded-md">
|
|
<div class="text-lg font-bold ${getSuccessRateColor(stats.successRate)}">
|
|
${stats.successRate}%
|
|
</div>
|
|
<div class="text-xs text-white">Success Rate</div>
|
|
</div>
|
|
<div class="text-center p-2 bg-gray-500 rounded-md">
|
|
<div class="text-lg font-bold text-blue-500">${stats.totalBids}</div>
|
|
<div class="text-xs text-white">Total Trades</div>
|
|
</div>
|
|
</div>
|
|
<div class="grid grid-cols-3 gap-2 mt-2 text-center text-xs">
|
|
<div>
|
|
<div class="text-green-600 font-semibold">
|
|
${stats.totalSuccessful}
|
|
</div>
|
|
<div class="text-white">Successful</div>
|
|
</div>
|
|
<div>
|
|
<div class="text-yellow-600 font-semibold">
|
|
${stats.totalRejected}
|
|
</div>
|
|
<div class="text-white">Rejected</div>
|
|
</div>
|
|
<div>
|
|
<div class="text-red-600 font-semibold">
|
|
${stats.totalFailed}
|
|
</div>
|
|
<div class="text-white">Failed</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
`;
|
|
};
|
|
|
|
const updateConnectionStatus = (status) => {
|
|
const { statusDot, statusText } = elements;
|
|
if (!statusDot || !statusText) return;
|
|
|
|
const statusConfig = {
|
|
connected: {
|
|
dotClass: 'w-2.5 h-2.5 rounded-full bg-green-500 mr-2',
|
|
textClass: 'text-sm text-green-500',
|
|
message: 'Connected'
|
|
},
|
|
disconnected: {
|
|
dotClass: 'w-2.5 h-2.5 rounded-full bg-red-500 mr-2',
|
|
textClass: 'text-sm text-red-500',
|
|
message: 'Disconnected - Reconnecting...'
|
|
},
|
|
error: {
|
|
dotClass: 'w-2.5 h-2.5 rounded-full bg-yellow-500 mr-2',
|
|
textClass: 'text-sm text-yellow-500',
|
|
message: 'Connection Error'
|
|
},
|
|
default: {
|
|
dotClass: 'w-2.5 h-2.5 rounded-full bg-gray-500 mr-2',
|
|
textClass: 'text-sm text-gray-500',
|
|
message: 'Connecting...'
|
|
}
|
|
};
|
|
|
|
const config = statusConfig[status] || statusConfig.default;
|
|
statusDot.className = config.dotClass;
|
|
statusText.className = config.textClass;
|
|
statusText.textContent = config.message;
|
|
};
|
|
|
|
const updateLoadingState = (isLoading) => {
|
|
state.isLoading = isLoading;
|
|
if (elements.refreshBidsButton) {
|
|
elements.refreshBidsButton.disabled = isLoading;
|
|
elements.refreshBidsButton.classList.toggle('opacity-75', isLoading);
|
|
elements.refreshBidsButton.classList.toggle('cursor-wait', isLoading);
|
|
|
|
const refreshIcon = elements.refreshBidsButton.querySelector('svg');
|
|
const refreshText = elements.refreshBidsButton.querySelector('#refreshText');
|
|
|
|
if (refreshIcon) {
|
|
// Add CSS transition for smoother animation
|
|
refreshIcon.style.transition = 'transform 0.3s ease';
|
|
refreshIcon.classList.toggle('animate-spin', isLoading);
|
|
}
|
|
|
|
if (refreshText) {
|
|
refreshText.textContent = isLoading ? 'Refreshing...' : 'Refresh';
|
|
}
|
|
}
|
|
};
|
|
|
|
const createBidTableRow = async (bid) => {
|
|
if (!bid || !bid.bid_id) {
|
|
console.error('Invalid bid data:', bid);
|
|
return '';
|
|
}
|
|
|
|
const identity = await IdentityManager.getIdentityData(bid.addr_from);
|
|
const fromAmount = parseFloat(bid.amount_from) || 0;
|
|
const toAmount = parseFloat(bid.amount_to) || 0;
|
|
const rate = toAmount > 0 ? toAmount / fromAmount : 0;
|
|
const inverseRate = fromAmount > 0 ? fromAmount / toAmount : 0;
|
|
|
|
const fromSymbol = window.CoinManager.getSymbol(bid.coin_from) || bid.coin_from;
|
|
const toSymbol = window.CoinManager.getSymbol(bid.coin_to) || bid.coin_to;
|
|
|
|
const timeColor = getTimeStrokeColor(bid.expire_at);
|
|
const uniqueId = `${bid.bid_id}_${bid.created_at}`;
|
|
|
|
return `
|
|
<tr class="relative opacity-100 text-gray-500 dark:text-gray-100 hover:bg-coolGray-200 dark:hover:bg-gray-600" data-bid-id="${bid.bid_id}">
|
|
<td class="relative w-0 p-0 m-0">
|
|
<div class="absolute top-0 bottom-0 left-0 w-1"></div>
|
|
</td>
|
|
|
|
<!-- Time Column -->
|
|
<td class="py-3 pl-1 pr-2 text-xs whitespace-nowrap">
|
|
<div class="flex items-center">
|
|
<div class="relative" data-tooltip-target="tooltip-time-${uniqueId}">
|
|
<svg class="w-5 h-5 rounded-full mr-4 cursor-pointer" xmlns="http://www.w3.org/2000/svg" height="20" width="20" viewBox="0 0 24 24">
|
|
<g stroke-linecap="round" stroke-width="2" fill="none" stroke="${timeColor}" stroke-linejoin="round">
|
|
<circle cx="12" cy="12" r="11"></circle>
|
|
<polyline points="12,6 12,12 18,12"></polyline>
|
|
</g>
|
|
</svg>
|
|
</div>
|
|
<div class="flex flex-col hidden xl:block">
|
|
<div class="text-xs whitespace-nowrap">
|
|
<span class="bold">Posted:</span> ${formatTimeAgo(bid.created_at)}
|
|
</div>
|
|
<div class="text-xs whitespace-nowrap">
|
|
<span class="bold">Expires in:</span> ${formatTime(bid.expire_at)}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</td>
|
|
|
|
<!-- Details Column -->
|
|
<td class="py-8 px-4 text-xs text-left hidden xl:block">
|
|
<div class="flex flex-col gap-2 relative">
|
|
<div class="flex items-center">
|
|
<a href="/identity/${bid.addr_from}" data-tooltip-target="tooltip-identity-${uniqueId}" class="flex items-center">
|
|
<svg class="w-4 h-4 mr-2 text-gray-400 dark:text-white" fill="currentColor" viewBox="0 0 20 20">
|
|
<path fill-rule="evenodd" d="M10 9a3 3 0 100-6 3 3 0 000 6zm-7 9a7 7 0 1114 0H3z" clip-rule="evenodd"></path>
|
|
</svg>
|
|
<span class="monospace ${identity?.label ? 'dark:text-white' : 'dark:text-white'}">
|
|
${identity?.label || formatAddress(bid.addr_from)}
|
|
</span>
|
|
</a>
|
|
</div>
|
|
<div class="monospace text-xs text-gray-500 dark:text-gray-300">
|
|
<span class="font-semibold">Offer ID:</span>
|
|
<a href="/offer/${bid.offer_id}" data-tooltip-target="tooltip-offer-${uniqueId}" class="hover:underline">
|
|
${formatAddress(bid.offer_id)}
|
|
</a>
|
|
</div>
|
|
<div class="monospace text-xs text-gray-500 dark:text-gray-300">
|
|
<span class="font-semibold">Bid ID:</span>
|
|
<a href="/bid/${bid.bid_id}" data-tooltip-target="tooltip-bid-${uniqueId}" class="hover:underline">
|
|
${formatAddress(bid.bid_id)}
|
|
</a>
|
|
</div>
|
|
</div>
|
|
</td>
|
|
|
|
<!-- You Send Column -->
|
|
<td class="py-0">
|
|
<div class="py-3 px-4 text-left">
|
|
<div class="items-center monospace">
|
|
<div class="pr-2">
|
|
<div class="text-sm font-semibold">${fromAmount.toFixed(8)}</div>
|
|
<div class="text-sm text-gray-500 dark:text-gray-400">${bid.coin_from}</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</td>
|
|
|
|
<!-- Swap Column -->
|
|
<td class="py-0">
|
|
<div class="py-3 px-4 text-center">
|
|
<div class="flex items-center justify-center">
|
|
<span class="inline-flex mr-3 align-middle items-center justify-center w-18 h-20 rounded">
|
|
<img class="h-12"
|
|
src="/static/images/coins/${window.CoinManager.getCoinIcon(bid.coin_from)}"
|
|
alt="${bid.coin_from}"
|
|
onerror="this.src='/static/images/coins/default.png'">
|
|
</span>
|
|
<svg class="w-5 h-5" fill="currentColor" viewBox="0 0 20 20">
|
|
<path fill-rule="evenodd" d="M12.293 5.293a1 1 0 011.414 0l4 4a1 1 0 010 1.414l-4 4a1 1 0 01-1.414-1.414L14.586 11H3a1 1 0 110-2h11.586l-2.293-2.293a1 1 0 010-1.414z"></path>
|
|
</svg>
|
|
<span class="inline-flex ml-3 align-middle items-center justify-center w-18 h-20 rounded">
|
|
<img class="h-12"
|
|
src="/static/images/coins/${window.CoinManager.getCoinIcon(bid.coin_to)}"
|
|
alt="${bid.coin_to}"
|
|
onerror="this.src='/static/images/coins/default.png'">
|
|
</span>
|
|
</div>
|
|
</div>
|
|
</td>
|
|
<!-- You Get Column -->
|
|
<td class="py-0">
|
|
<div class="py-3 px-4 text-right">
|
|
<div class="items-center monospace">
|
|
<div class="text-sm font-semibold">${toAmount.toFixed(8)}</div>
|
|
<div class="text-sm text-gray-500 dark:text-gray-400">${bid.coin_to}</div>
|
|
</div>
|
|
</div>
|
|
</td>
|
|
|
|
<!-- Rate Column -->
|
|
<td class="py-3 px-4 text-right semibold monospace item-center text-xs">
|
|
<div class="relative">
|
|
<div class="flex flex-col items-end">
|
|
<span class="bold text-gray-700 dark:text-white">
|
|
${rate.toFixed(8)} ${toSymbol}/${fromSymbol}
|
|
</span>
|
|
<span class="semibold text-gray-400 dark:text-gray-300">
|
|
${inverseRate.toFixed(8)} ${fromSymbol}/${toSymbol}
|
|
</span>
|
|
</div>
|
|
</div>
|
|
</td>
|
|
|
|
<!-- Actions Column -->
|
|
<td class="py-3 px-4 text-center">
|
|
<a href="/bid/${bid.bid_id}/accept"
|
|
class="inline-block w-20 py-1 px-2 font-medium text-center text-sm rounded-md bg-blue-500 text-white border border-blue-500 hover:bg-blue-600 transition duration-200">
|
|
Accept
|
|
</a>
|
|
</td>
|
|
</tr>
|
|
|
|
<!-- Tooltips -->
|
|
<div id="tooltip-time-${uniqueId}" role="tooltip" class="inline-block absolute z-50 py-2 px-3 text-sm font-medium text-white bg-gray-400 rounded-lg shadow-sm opacity-0 transition-opacity duration-300 tooltip dark:bg-gray-600">
|
|
<div class="active-revoked-expired">
|
|
<span class="bold">
|
|
<div class="text-xs"><span class="bold">Posted:</span> ${formatTimeAgo(bid.created_at)}</div>
|
|
<div class="text-xs"><span class="bold">Expires in:</span> ${formatTime(bid.expire_at)}</div>
|
|
</span>
|
|
</div>
|
|
<div class="mt-5 text-xs">
|
|
<p class="font-bold mb-3">Time Indicator Colors:</p>
|
|
<p class="flex items-center">
|
|
<svg class="w-5 h-5 mr-3" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24">
|
|
<g stroke-linecap="round" stroke-width="2" fill="none" stroke="#10B981" stroke-linejoin="round">
|
|
<circle cx="12" cy="12" r="11"></circle>
|
|
<polyline points="12,6 12,12 18,12" stroke="#10B981"></polyline>
|
|
</g>
|
|
</svg>
|
|
Green: More than 30 minutes left
|
|
</p>
|
|
<p class="flex items-center mt-3">
|
|
<svg class="w-5 h-5 mr-3" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24">
|
|
<g stroke-linecap="round" stroke-width="2" fill="none" stroke="#3B82F6" stroke-linejoin="round">
|
|
<circle cx="12" cy="12" r="11"></circle>
|
|
<polyline points="12,6 12,12 18,12" stroke="#3B82F6"></polyline>
|
|
</g>
|
|
</svg>
|
|
Blue: Between 5 and 30 minutes left
|
|
</p>
|
|
<p class="flex items-center mt-3 mb-3">
|
|
<svg class="w-5 h-5 mr-3" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24">
|
|
<g stroke-linecap="round" stroke-width="2" fill="none" stroke="#9CA3AF" stroke-linejoin="round">
|
|
<circle cx="12" cy="12" r="11"></circle>
|
|
<polyline points="12,6 12,12 18,12" stroke="#9CA3AF"></polyline>
|
|
</g>
|
|
</svg>
|
|
Grey: Less than 5 minutes left or expired
|
|
</p>
|
|
</div>
|
|
<div class="tooltip-arrow" data-popper-arrow></div>
|
|
</div>
|
|
|
|
<div id="tooltip-identity-${uniqueId}" role="tooltip" class="fixed z-50 py-3 px-4 text-sm font-medium text-white bg-gray-400 rounded-lg shadow-sm opacity-0 transition-opacity duration-300 tooltip dark:bg-gray-600 max-w-sm pointer-events-none">
|
|
${createIdentityTooltip(identity)}
|
|
<div class="tooltip-arrow" data-popper-arrow></div>
|
|
</div>
|
|
|
|
<div id="tooltip-offer-${uniqueId}" role="tooltip" class="inline-block absolute z-50 py-2 px-3 text-sm font-medium text-white bg-gray-400 rounded-lg shadow-sm opacity-0 transition-opacity duration-300 tooltip dark:bg-gray-600">
|
|
<div class="space-y-1">
|
|
<div class="text-white text-xs tracking-wide font-semibold">Offer ID:</div>
|
|
<div class="monospace text-xs break-all">
|
|
${bid.offer_id}
|
|
</div>
|
|
</div>
|
|
<div class="tooltip-arrow" data-popper-arrow></div>
|
|
</div>
|
|
|
|
<div id="tooltip-bid-${uniqueId}" role="tooltip" class="inline-block absolute z-50 py-2 px-3 text-sm font-medium text-white bg-gray-400 rounded-lg shadow-sm opacity-0 transition-opacity duration-300 tooltip dark:bg-gray-600">
|
|
<div class="space-y-1">
|
|
<div class="text-white text-xs tracking-wide font-semibold">Bid ID:</div>
|
|
<div class="monospace text-xs break-all">
|
|
${bid.bid_id}
|
|
</div>
|
|
</div>
|
|
<div class="tooltip-arrow" data-popper-arrow></div>
|
|
</div>
|
|
`;
|
|
};
|
|
|
|
const getDisplayText = (identity, address) => {
|
|
if (identity?.label) {
|
|
return identity.label;
|
|
}
|
|
return formatAddress(address);
|
|
};
|
|
|
|
const createDetailsColumn = (bid, identity, uniqueId) => `
|
|
<td class="py-8 px-4 text-xs text-left hidden xl:block">
|
|
<div class="flex flex-col gap-2 relative">
|
|
<div class="flex items-center">
|
|
<a href="/identity/${bid.addr_from}" data-tooltip-target="tooltip-identity-${uniqueId}" class="flex items-center">
|
|
<svg class="w-4 h-4 mr-2 text-gray-400 dark:text-white" fill="currentColor" viewBox="0 0 20 20">
|
|
<path fill-rule="evenodd" d="M10 9a3 3 0 100-6 3 3 0 000 6zm-7 9a7 7 0 1114 0H3z" clip-rule="evenodd"></path>
|
|
</svg>
|
|
<span class="monospace ${identity?.label ? 'dark:text-white' : 'dark:text-white'}">
|
|
${getDisplayText(identity, bid.addr_from)}
|
|
</span>
|
|
</a>
|
|
</div>
|
|
<div class="monospace text-xs text-gray-500 dark:text-gray-300">
|
|
<span class="font-semibold">Offer ID:</span>
|
|
<a href="/offer/${bid.offer_id}" data-tooltip-target="tooltip-offer-${uniqueId}" class="hover:underline">
|
|
${formatAddress(bid.offer_id)}
|
|
</a>
|
|
</div>
|
|
<div class="monospace text-xs text-gray-500 dark:text-gray-300">
|
|
<span class="font-semibold">Bid ID:</span>
|
|
<a href="/bid/${bid.bid_id}" data-tooltip-target="tooltip-bid-${uniqueId}" class="hover:underline">
|
|
${formatAddress(bid.bid_id)}
|
|
</a>
|
|
</div>
|
|
</div>
|
|
</td>
|
|
`;
|
|
|
|
async function updateBidsTable(options = {}) {
|
|
const { resetPage = false, refreshData = true } = options;
|
|
|
|
if (state.refreshPromise) {
|
|
await state.refreshPromise;
|
|
return;
|
|
}
|
|
|
|
try {
|
|
updateLoadingState(true);
|
|
|
|
if (refreshData) {
|
|
state.refreshPromise = (async () => {
|
|
try {
|
|
const response = await fetch('/json/bids', {
|
|
method: 'POST',
|
|
headers: { 'Content-Type': 'application/json' },
|
|
body: JSON.stringify({
|
|
sort_by: "created_at",
|
|
sort_dir: "desc",
|
|
with_available_or_active: true
|
|
})
|
|
});
|
|
|
|
if (!response.ok) {
|
|
throw new Error(`HTTP error! status: ${response.status}`);
|
|
}
|
|
|
|
const allBids = await response.json();
|
|
if (!Array.isArray(allBids)) {
|
|
throw new Error('Invalid response format');
|
|
}
|
|
|
|
state.jsonData = allBids.filter(bid => bid.bid_state === "Received");
|
|
state.originalJsonData = [...state.jsonData];
|
|
} finally {
|
|
state.refreshPromise = null;
|
|
}
|
|
})();
|
|
|
|
await state.refreshPromise;
|
|
}
|
|
|
|
if (elements.availableBidsCount) {
|
|
elements.availableBidsCount.textContent = state.jsonData.length;
|
|
}
|
|
|
|
const totalPages = Math.ceil(state.jsonData.length / PAGE_SIZE);
|
|
|
|
if (resetPage && state.jsonData.length > 0) {
|
|
state.currentPage = 1;
|
|
}
|
|
|
|
state.currentPage = Math.min(Math.max(1, state.currentPage), Math.max(1, totalPages));
|
|
|
|
const startIndex = (state.currentPage - 1) * PAGE_SIZE;
|
|
const endIndex = startIndex + PAGE_SIZE;
|
|
const currentPageBids = state.jsonData.slice(startIndex, endIndex);
|
|
|
|
if (elements.bidsBody) {
|
|
if (currentPageBids.length > 0) {
|
|
const rowPromises = currentPageBids.map(bid => createBidTableRow(bid));
|
|
const rows = await Promise.all(rowPromises);
|
|
elements.bidsBody.innerHTML = rows.join('');
|
|
|
|
if (window.TooltipManager) {
|
|
window.TooltipManager.cleanup();
|
|
const tooltipTriggers = document.querySelectorAll('[data-tooltip-target]');
|
|
tooltipTriggers.forEach(trigger => {
|
|
const targetId = trigger.getAttribute('data-tooltip-target');
|
|
const tooltipContent = document.getElementById(targetId);
|
|
if (tooltipContent) {
|
|
window.TooltipManager.create(trigger, tooltipContent.innerHTML, {
|
|
placement: trigger.getAttribute('data-tooltip-placement') || 'top'
|
|
});
|
|
}
|
|
});
|
|
}
|
|
} else {
|
|
elements.bidsBody.innerHTML = `
|
|
<tr>
|
|
<td colspan="8" class="text-center py-4 text-gray-500 dark:text-white">
|
|
No available bids requests found
|
|
</td>
|
|
</tr>`;
|
|
}
|
|
}
|
|
|
|
if (elements.paginationControls) {
|
|
elements.paginationControls.style.display = totalPages > 1 ? 'flex' : 'none';
|
|
}
|
|
|
|
if (elements.currentPageSpan) {
|
|
elements.currentPageSpan.textContent = state.currentPage;
|
|
}
|
|
|
|
if (elements.prevPageButton) {
|
|
elements.prevPageButton.style.display = state.currentPage > 1 ? 'inline-flex' : 'none';
|
|
}
|
|
|
|
if (elements.nextPageButton) {
|
|
elements.nextPageButton.style.display = state.currentPage < totalPages ? 'inline-flex' : 'none';
|
|
}
|
|
|
|
} catch (error) {
|
|
console.error('Error updating bids table:', error);
|
|
if (elements.bidsBody) {
|
|
elements.bidsBody.innerHTML = `
|
|
<tr>
|
|
<td colspan="8" class="text-center py-4 text-red-500">
|
|
Error loading bids. Please try again later.
|
|
</td>
|
|
</tr>`;
|
|
}
|
|
} finally {
|
|
updateLoadingState(false);
|
|
}
|
|
}
|
|
|
|
const setupEventListeners = () => {
|
|
if (elements.refreshBidsButton) {
|
|
elements.refreshBidsButton.addEventListener('click', async () => {
|
|
if (state.isRefreshing) return;
|
|
|
|
updateLoadingState(true);
|
|
|
|
await new Promise(resolve => setTimeout(resolve, 500));
|
|
|
|
try {
|
|
await updateBidsTable({ resetPage: true, refreshData: true });
|
|
} finally {
|
|
updateLoadingState(false);
|
|
}
|
|
});
|
|
}
|
|
|
|
if (elements.prevPageButton) {
|
|
elements.prevPageButton.addEventListener('click', async () => {
|
|
if (state.isLoading) return;
|
|
if (state.currentPage > 1) {
|
|
state.currentPage--;
|
|
await updateBidsTable({ resetPage: false, refreshData: false });
|
|
}
|
|
});
|
|
}
|
|
|
|
if (elements.nextPageButton) {
|
|
elements.nextPageButton.addEventListener('click', async () => {
|
|
if (state.isLoading) return;
|
|
const totalPages = Math.ceil(state.jsonData.length / PAGE_SIZE);
|
|
if (state.currentPage < totalPages) {
|
|
state.currentPage++;
|
|
await updateBidsTable({ resetPage: false, refreshData: false });
|
|
}
|
|
});
|
|
}
|
|
};
|
|
|
|
document.addEventListener('DOMContentLoaded', async () => {
|
|
WebSocketManager.initialize();
|
|
setupEventListeners();
|
|
await updateBidsTable({ resetPage: true, refreshData: true });
|
|
});
|