mirror of
https://github.com/basicswap/basicswap.git
synced 2025-01-08 19:59:33 +00:00
Merge pull request #170 from gerlofvanek/offers-4
ui: Offers table smart refresh button (JS), various fixes.
This commit is contained in:
commit
5941f2952e
2 changed files with 184 additions and 61 deletions
|
@ -3,7 +3,7 @@ let lastRefreshTime = null;
|
||||||
let newEntriesCount = 0;
|
let newEntriesCount = 0;
|
||||||
let nextRefreshCountdown = 60;
|
let nextRefreshCountdown = 60;
|
||||||
let currentPage = 1;
|
let currentPage = 1;
|
||||||
const itemsPerPage = 100;
|
const itemsPerPage = 50;
|
||||||
let lastAppliedFilters = {};
|
let lastAppliedFilters = {};
|
||||||
|
|
||||||
const CACHE_KEY = 'latestPricesCache';
|
const CACHE_KEY = 'latestPricesCache';
|
||||||
|
@ -22,6 +22,9 @@ const isSentOffers = window.offersTableConfig.isSentOffers;
|
||||||
let currentSortColumn = 0;
|
let currentSortColumn = 0;
|
||||||
let currentSortDirection = 'desc';
|
let currentSortDirection = 'desc';
|
||||||
|
|
||||||
|
const PRICE_INIT_RETRIES = 3;
|
||||||
|
const PRICE_INIT_RETRY_DELAY = 2000;
|
||||||
|
|
||||||
const offerCache = {
|
const offerCache = {
|
||||||
set: (key, value, customTtl = null) => {
|
set: (key, value, customTtl = null) => {
|
||||||
const item = {
|
const item = {
|
||||||
|
@ -105,7 +108,6 @@ const offerCache = {
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
const coinNameToSymbol = {
|
const coinNameToSymbol = {
|
||||||
'Bitcoin': 'bitcoin',
|
'Bitcoin': 'bitcoin',
|
||||||
'Particl': 'particl',
|
'Particl': 'particl',
|
||||||
|
@ -355,6 +357,51 @@ function makePostRequest(url, headers = {}) {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async function initializePriceData() {
|
||||||
|
console.log('Initializing price data...');
|
||||||
|
let retryCount = 0;
|
||||||
|
let prices = null;
|
||||||
|
|
||||||
|
while (retryCount < PRICE_INIT_RETRIES) {
|
||||||
|
try {
|
||||||
|
prices = await fetchLatestPrices();
|
||||||
|
|
||||||
|
if (prices && Object.keys(prices).length > 0) {
|
||||||
|
console.log('Successfully fetched initial price data:', prices);
|
||||||
|
latestPrices = prices;
|
||||||
|
|
||||||
|
const PRICES_CACHE_KEY = 'prices_coingecko';
|
||||||
|
offerCache.set(PRICES_CACHE_KEY, prices, CACHE_DURATION);
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
console.warn(`Attempt ${retryCount + 1}: Price data incomplete, retrying...`);
|
||||||
|
retryCount++;
|
||||||
|
|
||||||
|
if (retryCount < PRICE_INIT_RETRIES) {
|
||||||
|
await new Promise(resolve => setTimeout(resolve, PRICE_INIT_RETRY_DELAY));
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error(`Error fetching prices (attempt ${retryCount + 1}):`, error);
|
||||||
|
retryCount++;
|
||||||
|
|
||||||
|
if (retryCount < PRICE_INIT_RETRIES) {
|
||||||
|
await new Promise(resolve => setTimeout(resolve, PRICE_INIT_RETRY_DELAY));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const fallbackPrices = getFallbackPrices();
|
||||||
|
if (fallbackPrices && Object.keys(fallbackPrices).length > 0) {
|
||||||
|
console.log('Using fallback prices:', fallbackPrices);
|
||||||
|
latestPrices = fallbackPrices;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
function loadSortPreferences() {
|
function loadSortPreferences() {
|
||||||
const savedColumn = localStorage.getItem('tableSortColumn');
|
const savedColumn = localStorage.getItem('tableSortColumn');
|
||||||
const savedDirection = localStorage.getItem('tableSortDirection');
|
const savedDirection = localStorage.getItem('tableSortDirection');
|
||||||
|
@ -778,7 +825,7 @@ async function checkExpiredAndFetchNew() {
|
||||||
newListings = newListings.filter(offer => !isOfferExpired(offer));
|
newListings = newListings.filter(offer => !isOfferExpired(offer));
|
||||||
originalJsonData = newListings;
|
originalJsonData = newListings;
|
||||||
|
|
||||||
cache.set(OFFERS_CACHE_KEY, newListings, CACHE_DURATION);
|
offerCache.set(OFFERS_CACHE_KEY, newListings, CACHE_DURATION);
|
||||||
|
|
||||||
const currentFilters = new FormData(filterForm);
|
const currentFilters = new FormData(filterForm);
|
||||||
const hasActiveFilters = currentFilters.get('coin_to') !== 'any' ||
|
const hasActiveFilters = currentFilters.get('coin_to') !== 'any' ||
|
||||||
|
@ -1320,13 +1367,22 @@ async function fetchOffers(manualRefresh = false) {
|
||||||
setRefreshButtonLoading(true);
|
setRefreshButtonLoading(true);
|
||||||
const OFFERS_CACHE_KEY = `offers_${isSentOffers ? 'sent' : 'received'}`;
|
const OFFERS_CACHE_KEY = `offers_${isSentOffers ? 'sent' : 'received'}`;
|
||||||
|
|
||||||
|
if (!latestPrices || Object.keys(latestPrices).length === 0) {
|
||||||
|
console.log('No price data available, initializing...');
|
||||||
|
const priceInitSuccess = await initializePriceData();
|
||||||
|
if (!priceInitSuccess) {
|
||||||
|
console.error('Failed to initialize price data');
|
||||||
|
ui.displayErrorMessage('Unable to load cryptocurrency prices. Some features may be limited.');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (!manualRefresh) {
|
if (!manualRefresh) {
|
||||||
const cachedData = offerCache.get(OFFERS_CACHE_KEY);
|
const cachedData = offerCache.get(OFFERS_CACHE_KEY);
|
||||||
if (cachedData) {
|
if (cachedData) {
|
||||||
console.log('Using cached offers data');
|
console.log('Using cached offers data');
|
||||||
jsonData = cachedData.value;
|
jsonData = cachedData.value;
|
||||||
originalJsonData = [...cachedData.value];
|
originalJsonData = [...cachedData.value];
|
||||||
updateOffersTable();
|
await updateOffersTable();
|
||||||
updateJsonView();
|
updateJsonView();
|
||||||
updatePaginationInfo();
|
updatePaginationInfo();
|
||||||
setRefreshButtonLoading(false);
|
setRefreshButtonLoading(false);
|
||||||
|
@ -1338,20 +1394,10 @@ async function fetchOffers(manualRefresh = false) {
|
||||||
const endpoint = isSentOffers ? '/json/sentoffers' : '/json/offers';
|
const endpoint = isSentOffers ? '/json/sentoffers' : '/json/offers';
|
||||||
console.log(`[Debug] Fetching from endpoint: ${endpoint}`);
|
console.log(`[Debug] Fetching from endpoint: ${endpoint}`);
|
||||||
|
|
||||||
const response = await fetch(endpoint, {
|
const response = await fetch(endpoint);
|
||||||
method: 'POST',
|
|
||||||
headers: {
|
|
||||||
'Content-Type': 'application/json',
|
|
||||||
},
|
|
||||||
body: JSON.stringify({
|
|
||||||
with_extra_info: true,
|
|
||||||
limit: 1000
|
|
||||||
})
|
|
||||||
});
|
|
||||||
|
|
||||||
const data = await response.json();
|
const data = await response.json();
|
||||||
let newData = Array.isArray(data) ? data : Object.values(data);
|
|
||||||
|
|
||||||
|
let newData = Array.isArray(data) ? data : Object.values(data);
|
||||||
newData = newData.map(offer => ({
|
newData = newData.map(offer => ({
|
||||||
...offer,
|
...offer,
|
||||||
offer_id: String(offer.offer_id || ''),
|
offer_id: String(offer.offer_id || ''),
|
||||||
|
@ -1379,19 +1425,19 @@ async function fetchOffers(manualRefresh = false) {
|
||||||
originalJsonData = [...mergedData];
|
originalJsonData = [...mergedData];
|
||||||
}
|
}
|
||||||
|
|
||||||
// Cache the new data
|
|
||||||
offerCache.set(OFFERS_CACHE_KEY, jsonData, CACHE_DURATION);
|
offerCache.set(OFFERS_CACHE_KEY, jsonData, CACHE_DURATION);
|
||||||
|
|
||||||
if (newEntriesCountSpan) {
|
if (newEntriesCountSpan) {
|
||||||
newEntriesCountSpan.textContent = jsonData.length;
|
newEntriesCountSpan.textContent = jsonData.length;
|
||||||
}
|
}
|
||||||
|
|
||||||
updateOffersTable();
|
await updateOffersTable();
|
||||||
updateJsonView();
|
updateJsonView();
|
||||||
updatePaginationInfo();
|
updatePaginationInfo();
|
||||||
|
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error('[Debug] Error fetching offers:', error);
|
console.error('[Debug] Error fetching offers:', error);
|
||||||
|
ui.displayErrorMessage('Failed to fetch offers. Please try again later.');
|
||||||
} finally {
|
} finally {
|
||||||
setRefreshButtonLoading(false);
|
setRefreshButtonLoading(false);
|
||||||
}
|
}
|
||||||
|
@ -1532,48 +1578,55 @@ function filterAndSortData() {
|
||||||
return filteredData;
|
return filteredData;
|
||||||
}
|
}
|
||||||
|
|
||||||
async function calculateProfitLoss(fromCoin, toCoin, fromAmount, toAmount, isOwnOffer) {
|
function calculateProfitLoss(fromCoin, toCoin, fromAmount, toAmount, isOwnOffer) {
|
||||||
console.log(`Calculating profit/loss for ${fromAmount} ${fromCoin} to ${toAmount} ${toCoin}, isOwnOffer: ${isOwnOffer}`);
|
return new Promise((resolve) => {
|
||||||
|
console.log(`Calculating profit/loss for ${fromAmount} ${fromCoin} to ${toAmount} ${toCoin}, isOwnOffer: ${isOwnOffer}`);
|
||||||
|
|
||||||
if (!latestPrices) {
|
if (!latestPrices) {
|
||||||
console.error('Latest prices not available. Unable to calculate profit/loss.');
|
console.error('Latest prices not available. Unable to calculate profit/loss.');
|
||||||
return null;
|
resolve(null);
|
||||||
}
|
return;
|
||||||
|
|
||||||
const getPriceKey = (coin) => {
|
|
||||||
const lowerCoin = coin.toLowerCase();
|
|
||||||
if (lowerCoin === 'firo' || lowerCoin === 'zcoin') {
|
|
||||||
return 'zcoin';
|
|
||||||
}
|
}
|
||||||
if (lowerCoin === 'bitcoin cash') {
|
|
||||||
return 'bitcoin-cash';
|
const getPriceKey = (coin) => {
|
||||||
|
const lowerCoin = coin.toLowerCase();
|
||||||
|
if (lowerCoin === 'firo' || lowerCoin === 'zcoin') {
|
||||||
|
return 'zcoin';
|
||||||
|
}
|
||||||
|
if (lowerCoin === 'bitcoin cash') {
|
||||||
|
return 'bitcoin-cash';
|
||||||
|
}
|
||||||
|
if (lowerCoin === 'particl anon' || lowerCoin === 'particl blind') {
|
||||||
|
return 'particl';
|
||||||
|
}
|
||||||
|
return coinNameToSymbol[coin] || lowerCoin;
|
||||||
|
};
|
||||||
|
|
||||||
|
const fromSymbol = getPriceKey(fromCoin);
|
||||||
|
const toSymbol = getPriceKey(toCoin);
|
||||||
|
|
||||||
|
const fromPriceUSD = latestPrices[fromSymbol]?.usd;
|
||||||
|
const toPriceUSD = latestPrices[toSymbol]?.usd;
|
||||||
|
|
||||||
|
if (!fromPriceUSD || !toPriceUSD) {
|
||||||
|
console.warn(`Price data missing for ${fromSymbol} (${fromPriceUSD}) or ${toSymbol} (${toPriceUSD})`);
|
||||||
|
resolve(null);
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
return coinNameToSymbol[coin] || lowerCoin;
|
|
||||||
};
|
|
||||||
|
|
||||||
const fromSymbol = getPriceKey(fromCoin);
|
const fromValueUSD = fromAmount * fromPriceUSD;
|
||||||
const toSymbol = getPriceKey(toCoin);
|
const toValueUSD = toAmount * toPriceUSD;
|
||||||
|
|
||||||
const fromPriceUSD = latestPrices[fromSymbol]?.usd;
|
let percentDiff;
|
||||||
const toPriceUSD = latestPrices[toSymbol]?.usd;
|
if (isOwnOffer) {
|
||||||
|
percentDiff = ((toValueUSD / fromValueUSD) - 1) * 100;
|
||||||
|
} else {
|
||||||
|
percentDiff = ((fromValueUSD / toValueUSD) - 1) * 100;
|
||||||
|
}
|
||||||
|
|
||||||
if (!fromPriceUSD || !toPriceUSD) {
|
console.log(`Percent difference: ${percentDiff.toFixed(2)}%`);
|
||||||
console.error(`Price data missing for ${fromSymbol} or ${toSymbol}`);
|
resolve(percentDiff);
|
||||||
return null;
|
});
|
||||||
}
|
|
||||||
|
|
||||||
const fromValueUSD = fromAmount * fromPriceUSD;
|
|
||||||
const toValueUSD = toAmount * toPriceUSD;
|
|
||||||
|
|
||||||
let percentDiff;
|
|
||||||
if (isOwnOffer) {
|
|
||||||
percentDiff = ((toValueUSD / fromValueUSD) - 1) * 100;
|
|
||||||
} else {
|
|
||||||
percentDiff = ((fromValueUSD / toValueUSD) - 1) * 100;
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log(`Percent difference: ${percentDiff.toFixed(2)}%`);
|
|
||||||
return percentDiff;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async function getMarketRate(fromCoin, toCoin) {
|
async function getMarketRate(fromCoin, toCoin) {
|
||||||
|
@ -1885,13 +1938,83 @@ document.addEventListener('DOMContentLoaded', () => {
|
||||||
updateCoinFilterImages();
|
updateCoinFilterImages();
|
||||||
});
|
});
|
||||||
|
|
||||||
document.getElementById('refreshOffers').addEventListener('click', () => {
|
document.getElementById('refreshOffers').addEventListener('click', async () => {
|
||||||
console.log('🔄 Refresh button clicked');
|
console.log('🔄 Smart refresh initiated');
|
||||||
console.log('Clearing cache before refresh...');
|
setRefreshButtonLoading(true);
|
||||||
offerCache.clear();
|
|
||||||
console.log('Fetching fresh data...');
|
try {
|
||||||
fetchOffers(true);
|
|
||||||
|
const PRICES_CACHE_KEY = 'prices_coingecko';
|
||||||
|
const cachedPrices = offerCache.get(PRICES_CACHE_KEY);
|
||||||
|
|
||||||
|
if (!cachedPrices || !cachedPrices.remainingTime || cachedPrices.remainingTime < 60000) {
|
||||||
|
|
||||||
|
console.log('Fetching fresh price data...');
|
||||||
|
const newPrices = await fetchLatestPrices();
|
||||||
|
if (newPrices) {
|
||||||
|
latestPrices = newPrices;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
console.log('Using cached price data (still valid)');
|
||||||
|
}
|
||||||
|
|
||||||
|
const endpoint = isSentOffers ? '/json/sentoffers' : '/json/offers';
|
||||||
|
const response = await fetch(endpoint);
|
||||||
|
const newData = await response.json();
|
||||||
|
|
||||||
|
const processedNewData = Array.isArray(newData) ? newData : Object.values(newData);
|
||||||
|
const formattedNewData = processedNewData.map(offer => ({
|
||||||
|
...offer,
|
||||||
|
offer_id: String(offer.offer_id || ''),
|
||||||
|
swap_type: String(offer.swap_type || 'N/A'),
|
||||||
|
addr_from: String(offer.addr_from || ''),
|
||||||
|
coin_from: String(offer.coin_from || ''),
|
||||||
|
coin_to: String(offer.coin_to || ''),
|
||||||
|
amount_from: String(offer.amount_from || '0'),
|
||||||
|
amount_to: String(offer.amount_to || '0'),
|
||||||
|
rate: String(offer.rate || '0'),
|
||||||
|
created_at: Number(offer.created_at || 0),
|
||||||
|
expire_at: Number(offer.expire_at || 0),
|
||||||
|
is_own_offer: Boolean(offer.is_own_offer),
|
||||||
|
amount_negotiable: Boolean(offer.amount_negotiable),
|
||||||
|
is_revoked: Boolean(offer.is_revoked),
|
||||||
|
unique_id: `${offer.offer_id}_${offer.created_at}_${offer.coin_from}_${offer.coin_to}`
|
||||||
|
}));
|
||||||
|
|
||||||
|
const existingIds = new Set(jsonData.map(offer => offer.offer_id));
|
||||||
|
const newOffers = formattedNewData.filter(offer => !existingIds.has(offer.offer_id));
|
||||||
|
|
||||||
|
if (newOffers.length > 0) {
|
||||||
|
console.log(`Found ${newOffers.length} new offers`);
|
||||||
|
|
||||||
|
jsonData = mergeSentOffers(jsonData, formattedNewData);
|
||||||
|
originalJsonData = [...jsonData];
|
||||||
|
|
||||||
|
const OFFERS_CACHE_KEY = `offers_${isSentOffers ? 'sent' : 'received'}`;
|
||||||
|
offerCache.set(OFFERS_CACHE_KEY, jsonData, CACHE_DURATION);
|
||||||
|
} else {
|
||||||
|
console.log('No new offers found');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!isSentOffers) {
|
||||||
|
jsonData = jsonData.filter(offer => !isOfferExpired(offer));
|
||||||
|
originalJsonData = [...jsonData];
|
||||||
|
}
|
||||||
|
|
||||||
|
await updateOffersTable();
|
||||||
|
updateJsonView();
|
||||||
|
updatePaginationInfo();
|
||||||
|
|
||||||
|
console.log('Smart refresh completed successfully');
|
||||||
|
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Error during smart refresh:', error);
|
||||||
|
ui.displayErrorMessage('Failed to refresh offers. Please try again later.');
|
||||||
|
} finally {
|
||||||
|
setRefreshButtonLoading(false);
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
toggleButton.addEventListener('click', () => {
|
toggleButton.addEventListener('click', () => {
|
||||||
tableView.classList.toggle('hidden');
|
tableView.classList.toggle('hidden');
|
||||||
jsonView.classList.toggle('hidden');
|
jsonView.classList.toggle('hidden');
|
||||||
|
|
|
@ -33,7 +33,7 @@ from basicswap.basicswap_util import (
|
||||||
|
|
||||||
from basicswap.protocols.xmr_swap_1 import getChainBSplitKey, getChainBRemoteSplitKey
|
from basicswap.protocols.xmr_swap_1 import getChainBSplitKey, getChainBRemoteSplitKey
|
||||||
|
|
||||||
PAGE_LIMIT = 25
|
PAGE_LIMIT = 1000
|
||||||
invalid_coins_from = []
|
invalid_coins_from = []
|
||||||
known_chart_coins = [
|
known_chart_coins = [
|
||||||
"BTC",
|
"BTC",
|
||||||
|
|
Loading…
Reference in a new issue