You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
187 lines
8.4 KiB
HTML
187 lines
8.4 KiB
HTML
4 weeks ago
|
<html>
|
||
|
<head>
|
||
|
<title>Art101 Sales Statistics</title>
|
||
|
<script src="https://cdn.jsdelivr.net/npm/chart.js"></script>
|
||
|
<style>
|
||
|
#container {
|
||
|
display: grid;
|
||
|
grid-template-columns: 5fr 2fr;
|
||
|
grid-gap: 2.5rem;
|
||
|
visibility: hidden;
|
||
|
}
|
||
|
#container > div {
|
||
|
align-self: center;
|
||
|
}
|
||
|
#loading {
|
||
|
font-weight: bold;
|
||
|
}
|
||
|
body {
|
||
|
font-family: "Open Sans";
|
||
|
color: black;
|
||
|
padding: 20px;
|
||
|
}
|
||
|
h1 {
|
||
|
margin-bottom: 0;
|
||
|
}
|
||
|
</style>
|
||
|
<link rel="preconnect" href="https://fonts.googleapis.com">
|
||
|
<link rel="preconnect" href="https://fonts.gstatic.com" crossorigin>
|
||
|
<link href="https://fonts.googleapis.com/css2?family=Open+Sans&display=swap" rel="stylesheet">
|
||
|
</head>
|
||
|
<body>
|
||
|
<h1>Art101 Sales Statistics</h1>
|
||
|
<p><a href="https://github.com/lalanza808/nft-sales-scraper" target="_blank">Source Code</a></p>
|
||
|
<div id="loading">
|
||
|
Loading, please wait...
|
||
|
</div>
|
||
|
<div id="container"></div>
|
||
|
<script>
|
||
|
const container = document.getElementById('container');
|
||
|
|
||
|
fetch('/api/contracts').then(function(response) {
|
||
|
response.json().then(function(json) {
|
||
|
for (contract in json) {
|
||
|
console.log(`Found contract ${contract} (${json[contract]['contract_address']})`)
|
||
|
const contractName = contract;
|
||
|
const contractAddress = json[contract]['contract_address'];
|
||
|
const salesId = `sales-chart-${contractName}`;
|
||
|
const platformId = `platform-chart-${contractName}`;
|
||
|
newSalesChartDiv = document.createElement('div');
|
||
|
newSalesChart = document.createElement('canvas');
|
||
|
newSalesChart.setAttribute('id', salesId);
|
||
|
newSalesChartDiv.appendChild(newSalesChart);
|
||
|
newPlatformChartDiv = document.createElement('div');
|
||
|
newPlatformChart = document.createElement('canvas');
|
||
|
newPlatformChart.setAttribute('id', platformId);
|
||
|
newPlatformChartDiv.appendChild(newPlatformChart);
|
||
|
container.appendChild(newSalesChartDiv);
|
||
|
container.appendChild(newPlatformChartDiv);
|
||
|
let loaded = 0;
|
||
|
fetch(`/api/${contractAddress}/data`).then(function(response) {
|
||
|
response.json().then(function(json) {
|
||
|
loaded++;
|
||
|
if (loaded == 2) {
|
||
|
document.getElementById("container").style.visibility = "visible";
|
||
|
document.getElementById("loading").style.display = "none";
|
||
|
}
|
||
|
const labels = json.map(d => d.block);
|
||
|
const data = {
|
||
|
labels: labels,
|
||
|
datasets: [
|
||
|
{
|
||
|
type: 'line',
|
||
|
label: 'Average price',
|
||
|
backgroundColor: 'rgb(99, 132, 255)',
|
||
|
borderColor: 'rgb(99, 132, 255)',
|
||
|
data: json.map(d => d.average_price),
|
||
|
yAxisID: 'y1',
|
||
|
},
|
||
|
{
|
||
|
type: 'line',
|
||
|
label: 'Floor price',
|
||
|
backgroundColor: '#ccc',
|
||
|
borderColor: '#ccc',
|
||
|
data: json.map(d => d.floor_price),
|
||
|
yAxisID: 'y1',
|
||
|
},
|
||
|
{
|
||
|
type: 'bar',
|
||
|
label: 'Volume',
|
||
|
backgroundColor: 'rgb(255, 99, 132)',
|
||
|
borderColor: 'rgb(255, 99, 132)',
|
||
|
data: json.map(d => d.volume),
|
||
|
yAxisID: 'y',
|
||
|
}]
|
||
|
};
|
||
|
const config = {
|
||
|
type: 'line',
|
||
|
data: data,
|
||
|
options: {
|
||
|
elements: {
|
||
|
point: {
|
||
|
radius: 0
|
||
|
}
|
||
|
},
|
||
|
plugins: {
|
||
|
title: {
|
||
|
display: true,
|
||
|
text: `Sales for ${contractName}`,
|
||
|
font: {
|
||
|
size: '18px'
|
||
|
}
|
||
|
}
|
||
|
},
|
||
|
scales: {
|
||
|
y: {
|
||
|
type: 'linear',
|
||
|
display: true,
|
||
|
position: 'left',
|
||
|
},
|
||
|
y1: {
|
||
|
type: 'linear',
|
||
|
display: true,
|
||
|
position: 'right',
|
||
|
grid: {
|
||
|
drawOnChartArea: false,
|
||
|
},
|
||
|
},
|
||
|
}
|
||
|
}
|
||
|
};
|
||
|
const myChart = new Chart(
|
||
|
document.getElementById(salesId),
|
||
|
config
|
||
|
);
|
||
|
})
|
||
|
});
|
||
|
fetch(`/api/${contractAddress}/platforms`).then(function(response) {
|
||
|
response.json().then(function(json) {
|
||
|
loaded++;
|
||
|
if (loaded == 2) {
|
||
|
document.getElementById("container").style.visibility = "visible";
|
||
|
document.getElementById("loading").style.display = "none";
|
||
|
}
|
||
|
|
||
|
const labels = json.map(d => d.marketplace);
|
||
|
const total = json.map(d => d.volume).reduce((prev,next) => prev+next, 0);
|
||
|
const data = {
|
||
|
labels: labels,
|
||
|
datasets: [
|
||
|
{
|
||
|
label: 'Platform',
|
||
|
data: json.map(d => d.volume),
|
||
|
backgroundColor: ["#7463A8", "#80B7D8", "#AC39A0", "#B0C484", "#B75055", "#F12055"]
|
||
|
}]
|
||
|
};
|
||
|
const config = {
|
||
|
type: 'doughnut',
|
||
|
data: data,
|
||
|
options: {
|
||
|
plugins: {
|
||
|
title: {
|
||
|
display: true,
|
||
|
text: `Volume for ${contractName}: ${total.toFixed(2)}Ξ`,
|
||
|
font: {
|
||
|
size: '18px'
|
||
|
}
|
||
|
},
|
||
|
legend: {
|
||
|
position: 'bottom',
|
||
|
},
|
||
|
}
|
||
|
}
|
||
|
};
|
||
|
const myChart = new Chart(
|
||
|
document.getElementById(platformId),
|
||
|
config
|
||
|
);
|
||
|
})
|
||
|
});
|
||
|
}
|
||
|
})
|
||
|
})
|
||
|
|
||
|
</script>
|
||
|
</body>
|
||
|
</html>
|