update market flow
This commit is contained in:
parent
79eecc4f7d
commit
ea9fe4c868
@ -414,6 +414,19 @@
|
|||||||
<div class="w-full overflow-hidden m-auto mt-5">
|
<div class="w-full overflow-hidden m-auto mt-5">
|
||||||
{#if options !== null}
|
{#if options !== null}
|
||||||
<div class="app w-full relative">
|
<div class="app w-full relative">
|
||||||
|
<div class="flex justify-start space-x-2 absolute right-0 top-0 z-10">
|
||||||
|
{#each ["3M", "6M", "1Y"] as item}
|
||||||
|
<label
|
||||||
|
on:click={() => (timePeriod = item)}
|
||||||
|
class="px-3 py-1 text-sm {timePeriod === item
|
||||||
|
? 'bg-white text-black '
|
||||||
|
: 'text-white bg-table text-opacity-[0.6]'} transition ease-out duration-100 sm:hover:bg-white sm:hover:text-black rounded-md cursor-pointer"
|
||||||
|
>
|
||||||
|
{item}
|
||||||
|
</label>
|
||||||
|
{/each}
|
||||||
|
</div>
|
||||||
|
|
||||||
<Chart {init} {options} class="chart" />
|
<Chart {init} {options} class="chart" />
|
||||||
</div>
|
</div>
|
||||||
{:else}
|
{:else}
|
||||||
|
|||||||
@ -32,12 +32,15 @@
|
|||||||
let isLoading = false;
|
let isLoading = false;
|
||||||
let optionsData = null;
|
let optionsData = null;
|
||||||
//let sectorData = data?.getData?.sectorData || [];
|
//let sectorData = data?.getData?.sectorData || [];
|
||||||
let topSectorTickers = data?.getData?.topSectorTickers || {};
|
let topPosNetPremium = data?.getData?.topPosNetPremium || [];
|
||||||
let marketTideData = data?.getData?.marketTide || {};
|
let topNegNetPremium = data?.getData?.topNegNetPremium || {};
|
||||||
let selectedSector = "SPY";
|
|
||||||
|
|
||||||
let originalTopTickers = [...topSectorTickers[selectedSector]];
|
let marketTideData = data?.getData?.marketTide || {};
|
||||||
let displayTopTickers = topSectorTickers[selectedSector];
|
let originalPosTickers = topPosNetPremium;
|
||||||
|
let displayPosTickers = topPosNetPremium;
|
||||||
|
|
||||||
|
let originalNegTickers = topNegNetPremium;
|
||||||
|
let displayNegTickers = topNegNetPremium;
|
||||||
|
|
||||||
function findLastNonNull(dataArray, key) {
|
function findLastNonNull(dataArray, key) {
|
||||||
for (let i = dataArray.length - 1; i >= 0; i--) {
|
for (let i = dataArray.length - 1; i >= 0; i--) {
|
||||||
@ -89,15 +92,12 @@
|
|||||||
changesPercentage: { order: "none", type: "number" },
|
changesPercentage: { order: "none", type: "number" },
|
||||||
call_volume: { order: "none", type: "number" },
|
call_volume: { order: "none", type: "number" },
|
||||||
put_volume: { order: "none", type: "number" },
|
put_volume: { order: "none", type: "number" },
|
||||||
premium_ratio: { order: "none", type: "number" },
|
net_premium: { order: "none", type: "number" },
|
||||||
avg30_call_volume: { order: "none", type: "string" },
|
net_call_premium: { order: "none", type: "number" },
|
||||||
avg30_put_volume: { order: "none", type: "number" },
|
net_put_premium: { order: "none", type: "number" },
|
||||||
netPremium: { order: "none", type: "number" },
|
|
||||||
netCallPremium: { order: "none", type: "number" },
|
|
||||||
netPutPremium: { order: "none", type: "number" },
|
|
||||||
call_premium: { order: "none", type: "number" },
|
call_premium: { order: "none", type: "number" },
|
||||||
put_premium: { order: "none", type: "number" },
|
put_premium: { order: "none", type: "number" },
|
||||||
ivRank: { order: "none", type: "number" },
|
iv_rank: { order: "none", type: "number" },
|
||||||
};
|
};
|
||||||
|
|
||||||
$: topColumns = [
|
$: topColumns = [
|
||||||
@ -106,13 +106,13 @@
|
|||||||
{ key: "name", label: "Name", align: "left" },
|
{ key: "name", label: "Name", align: "left" },
|
||||||
{ key: "price", label: "Price", align: "right" },
|
{ key: "price", label: "Price", align: "right" },
|
||||||
{ key: "changesPercentage", label: "% Change", align: "right" },
|
{ key: "changesPercentage", label: "% Change", align: "right" },
|
||||||
{ key: "netPremium", label: "Net Prem", align: "right" },
|
{ key: "net_premium", label: "Net Prem", align: "right" },
|
||||||
{ key: "netCallPremium", label: "Net Call Prem", align: "right" },
|
{ key: "net_call_premium", label: "Net Call Prem", align: "right" },
|
||||||
{ key: "netPutPremium", label: "Net Put Prem", align: "right" },
|
{ key: "net_put_premium", label: "Net Put Prem", align: "right" },
|
||||||
{ key: "ivRank", label: "IV Rank", align: "right" },
|
{ key: "iv_rank", label: "IV Rank", align: "right" },
|
||||||
];
|
];
|
||||||
|
|
||||||
const sortTopTickers = (key) => {
|
const sortPosTickers = (key) => {
|
||||||
// Reset all other keys to 'none' except the current key
|
// Reset all other keys to 'none' except the current key
|
||||||
for (const k in sortOrders) {
|
for (const k in sortOrders) {
|
||||||
if (k !== key) {
|
if (k !== key) {
|
||||||
@ -130,8 +130,8 @@
|
|||||||
|
|
||||||
// Reset to original data when 'none' and stop further sorting
|
// Reset to original data when 'none' and stop further sorting
|
||||||
if (sortOrder === "none") {
|
if (sortOrder === "none") {
|
||||||
originalTopTickers = [...topSectorTickers[selectedSector]]; // Reset originalTopTickers to sectorData
|
originalPosTickers = [...topPosNetPremium]; // Reset originalPosTickers to sectorData
|
||||||
displayTopTickers = originalTopTickers;
|
displayPosTickers = originalPosTickers?.slice(0, 50);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -166,7 +166,66 @@
|
|||||||
};
|
};
|
||||||
|
|
||||||
// Sort using the generic comparison function
|
// Sort using the generic comparison function
|
||||||
displayTopTickers = [...originalTopTickers]
|
displayPosTickers = [...originalPosTickers]
|
||||||
|
.sort(compareValues)
|
||||||
|
?.slice(0, 50);
|
||||||
|
};
|
||||||
|
|
||||||
|
const sortNegTickers = (key) => {
|
||||||
|
// Reset all other keys to 'none' except the current key
|
||||||
|
for (const k in sortOrders) {
|
||||||
|
if (k !== key) {
|
||||||
|
sortOrders[k].order = "none";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Cycle through 'none', 'asc', 'desc' for the clicked key
|
||||||
|
const orderCycle = ["none", "asc", "desc"];
|
||||||
|
|
||||||
|
const currentOrderIndex = orderCycle.indexOf(sortOrders[key].order);
|
||||||
|
sortOrders[key].order =
|
||||||
|
orderCycle[(currentOrderIndex + 1) % orderCycle.length];
|
||||||
|
const sortOrder = sortOrders[key].order;
|
||||||
|
|
||||||
|
// Reset to original data when 'none' and stop further sorting
|
||||||
|
if (sortOrder === "none") {
|
||||||
|
originalNegTickers = [...topNegNetPremium];
|
||||||
|
displayNegTickers = originalNegTickers?.slice(0, 50);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Define a generic comparison function
|
||||||
|
const compareValues = (a, b) => {
|
||||||
|
const { type } = sortOrders[key];
|
||||||
|
let valueA, valueB;
|
||||||
|
|
||||||
|
switch (type) {
|
||||||
|
case "date":
|
||||||
|
valueA = new Date(a[key]);
|
||||||
|
valueB = new Date(b[key]);
|
||||||
|
break;
|
||||||
|
case "string":
|
||||||
|
valueA = a[key].toUpperCase();
|
||||||
|
valueB = b[key].toUpperCase();
|
||||||
|
return sortOrder === "asc"
|
||||||
|
? valueA.localeCompare(valueB)
|
||||||
|
: valueB.localeCompare(valueA);
|
||||||
|
case "number":
|
||||||
|
default:
|
||||||
|
valueA = parseFloat(a[key]);
|
||||||
|
valueB = parseFloat(b[key]);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (sortOrder === "asc") {
|
||||||
|
return valueA < valueB ? -1 : valueA > valueB ? 1 : 0;
|
||||||
|
} else {
|
||||||
|
return valueA > valueB ? -1 : valueA < valueB ? 1 : 0;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// Sort using the generic comparison function
|
||||||
|
displayNegTickers = [...originalNegTickers]
|
||||||
.sort(compareValues)
|
.sort(compareValues)
|
||||||
?.slice(0, 50);
|
?.slice(0, 50);
|
||||||
};
|
};
|
||||||
@ -174,7 +233,9 @@
|
|||||||
function getPlotOptions() {
|
function getPlotOptions() {
|
||||||
isLoading = true;
|
isLoading = true;
|
||||||
let dates = marketTideData?.map((item) => item?.time);
|
let dates = marketTideData?.map((item) => item?.time);
|
||||||
const priceList = marketTideData?.map((item) => item?.close);
|
const priceList = marketTideData?.map((item) =>
|
||||||
|
item?.close !== null ? item?.close?.toFixed(2) : item?.close,
|
||||||
|
);
|
||||||
const netCallPremList = marketTideData?.map(
|
const netCallPremList = marketTideData?.map(
|
||||||
(item) => item?.net_call_premium,
|
(item) => item?.net_call_premium,
|
||||||
);
|
);
|
||||||
@ -296,7 +357,7 @@
|
|||||||
let [hours, minutes] = timePart.split(":").map(Number);
|
let [hours, minutes] = timePart.split(":").map(Number);
|
||||||
|
|
||||||
// Only show labels at 30-minute intervals (XX:00 and XX:30)
|
// Only show labels at 30-minute intervals (XX:00 and XX:30)
|
||||||
if (minutes % 60 === 0) {
|
if (minutes % 30 === 0) {
|
||||||
const amPm = hours >= 12 ? "PM" : "AM";
|
const amPm = hours >= 12 ? "PM" : "AM";
|
||||||
hours = hours % 12 || 12;
|
hours = hours % 12 || 12;
|
||||||
return minutes === 0
|
return minutes === 0
|
||||||
@ -305,7 +366,7 @@
|
|||||||
}
|
}
|
||||||
return "";
|
return "";
|
||||||
},
|
},
|
||||||
interval: 29, // Show label every 30 minutes (29 intervals between)
|
interval: "auto", // Show label every 30 minutes (29 intervals between)
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|
||||||
@ -426,18 +487,6 @@
|
|||||||
return options;
|
return options;
|
||||||
}
|
}
|
||||||
optionsData = marketTideData ? getPlotOptions() : null;
|
optionsData = marketTideData ? getPlotOptions() : null;
|
||||||
|
|
||||||
/*
|
|
||||||
$: {
|
|
||||||
if (selectedSector) {
|
|
||||||
originalTopTickers = [...topSectorTickers[selectedSector]];
|
|
||||||
displayTopTickers =
|
|
||||||
data?.user?.tier === "Pro"
|
|
||||||
? displayTopTickers
|
|
||||||
: displayTopTickers?.slice(0, 3);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<SEO
|
<SEO
|
||||||
@ -557,15 +606,15 @@
|
|||||||
>
|
>
|
||||||
<div class="flex flex-row items-center">
|
<div class="flex flex-row items-center">
|
||||||
<label
|
<label
|
||||||
for="topSectorTickers"
|
for="topPosNetPrem"
|
||||||
class="mr-1 cursor-pointer flex flex-row items-center text-white text-xl sm:text-2xl font-bold"
|
class="mr-1 cursor-pointer flex flex-row items-center text-white text-xl sm:text-2xl font-bold"
|
||||||
>
|
>
|
||||||
Top S&P500 Stocks by Net Premium
|
Top Stocks by Positive Net Prem
|
||||||
</label>
|
</label>
|
||||||
<InfoModal
|
<InfoModal
|
||||||
title={"Top S&P500 Stocks by Net Premium"}
|
title={"Top Stocks by Positive Net Prem"}
|
||||||
content={"This list highlights top stocks based on net premium, displaying price changes and options activity. Discover which stocks are driving the sector and explore detailed options data."}
|
content={"This list highlights top stocks based on positive net premium, displaying price changes and options activity. Discover which stocks are driving the sector and explore detailed options data."}
|
||||||
id={"topSectorTickers"}
|
id={"topPosNetPrem"}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@ -580,15 +629,15 @@
|
|||||||
<TableHeader
|
<TableHeader
|
||||||
columns={topColumns}
|
columns={topColumns}
|
||||||
{sortOrders}
|
{sortOrders}
|
||||||
sortData={sortTopTickers}
|
sortData={sortPosTickers}
|
||||||
/>
|
/>
|
||||||
</thead>
|
</thead>
|
||||||
<tbody>
|
<tbody>
|
||||||
{#each displayTopTickers as item, index}
|
{#each displayPosTickers as item, index}
|
||||||
<tr
|
<tr
|
||||||
class="sm:hover:bg-[#245073] border-b border-gray-800 sm:hover:bg-opacity-[0.2] odd:bg-odd {index +
|
class="sm:hover:bg-[#245073] border-b border-gray-800 sm:hover:bg-opacity-[0.2] odd:bg-odd {index +
|
||||||
1 ===
|
1 ===
|
||||||
originalTopTickers?.length &&
|
originalPosTickers?.length &&
|
||||||
data?.user?.tier !== 'Pro'
|
data?.user?.tier !== 'Pro'
|
||||||
? 'opacity-[0.1]'
|
? 'opacity-[0.1]'
|
||||||
: ''}"
|
: ''}"
|
||||||
@ -659,19 +708,18 @@
|
|||||||
</table>
|
</table>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<!--
|
|
||||||
<div class="mb-3 mt-10">
|
<div class="mb-3 mt-10">
|
||||||
<div class="flex flex-row items-center">
|
<div class="flex flex-row items-center">
|
||||||
<label
|
<label
|
||||||
for="sectorFlowInfo"
|
for="topNegNetPrem"
|
||||||
class="mr-1 cursor-pointer flex flex-row items-center text-white text-xl sm:text-2xl font-bold"
|
class="mr-1 cursor-pointer flex flex-row items-center text-white text-xl sm:text-2xl font-bold"
|
||||||
>
|
>
|
||||||
Sector Flow
|
Top Stocks by Negative Net Prem
|
||||||
</label>
|
</label>
|
||||||
<InfoModal
|
<InfoModal
|
||||||
title={"Sector Flow"}
|
title={"Top Stocks by Negative Net Prem"}
|
||||||
content={"Sector Flow offers insights into options activity, helping traders identify trends and make informed decisions across market sectors."}
|
content={"Sector Flow offers insights into options activity, helping traders identify trends and make informed decisions across market sectors."}
|
||||||
id={"sectorFlowInfo"}
|
id={"topNegNetPrem"}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@ -682,37 +730,40 @@
|
|||||||
class="table table-sm table-compact no-scrollbar rounded-none sm:rounded-md text-white w-full bg-table border border-gray-800 m-auto"
|
class="table table-sm table-compact no-scrollbar rounded-none sm:rounded-md text-white w-full bg-table border border-gray-800 m-auto"
|
||||||
>
|
>
|
||||||
<thead>
|
<thead>
|
||||||
<TableHeader {columns} {sortOrders} {sortData} />
|
<TableHeader
|
||||||
|
columns={topColumns}
|
||||||
|
{sortOrders}
|
||||||
|
sortData={sortNegTickers}
|
||||||
|
/>
|
||||||
</thead>
|
</thead>
|
||||||
<tbody>
|
<tbody>
|
||||||
{#each stockList as item, index}
|
{#each displayNegTickers as item, index}
|
||||||
<tr
|
<tr
|
||||||
class="sm:hover:bg-[#245073] border-b border-gray-800 sm:hover:bg-opacity-[0.2] odd:bg-odd {index +
|
class="sm:hover:bg-[#245073] border-b border-gray-800 sm:hover:bg-opacity-[0.2] odd:bg-odd {index +
|
||||||
1 ===
|
1 ===
|
||||||
originalData?.length && data?.user?.tier !== 'Pro'
|
originalNegTickers?.length &&
|
||||||
|
data?.user?.tier !== 'Pro'
|
||||||
? 'opacity-[0.1]'
|
? 'opacity-[0.1]'
|
||||||
: ''}"
|
: ''}"
|
||||||
>
|
>
|
||||||
|
<td
|
||||||
|
class="text-start text-sm sm:text-[1rem] whitespace-nowrap text-white"
|
||||||
|
>
|
||||||
|
{item?.rank}
|
||||||
|
</td>
|
||||||
|
|
||||||
<td
|
<td
|
||||||
class="text-sm sm:text-[1rem] text-start whitespace-nowrap"
|
class="text-sm sm:text-[1rem] text-start whitespace-nowrap"
|
||||||
>
|
>
|
||||||
<HoverStockChart
|
<HoverStockChart symbol={item?.symbol} />
|
||||||
symbol={item?.ticker}
|
|
||||||
assetType="etf"
|
|
||||||
/>
|
|
||||||
</td>
|
</td>
|
||||||
|
|
||||||
<td
|
<td
|
||||||
class="text-start text-sm sm:text-[1rem] whitespace-nowrap text-white"
|
class="text-start text-sm sm:text-[1rem] whitespace-nowrap text-white"
|
||||||
>
|
>
|
||||||
<a
|
{item?.name?.length > 20
|
||||||
href={sectorNavigation?.find(
|
? item?.name?.slice(0, 20) + "..."
|
||||||
(listItem) => listItem?.title === item?.name,
|
: item?.name}
|
||||||
)?.link}
|
|
||||||
class="sm:hover:underline sm:hover:underline-offset-4 text-white"
|
|
||||||
>
|
|
||||||
{item?.name}
|
|
||||||
</a>
|
|
||||||
</td>
|
</td>
|
||||||
|
|
||||||
<td
|
<td
|
||||||
@ -732,129 +783,34 @@
|
|||||||
|
|
||||||
<td class="text-sm sm:text-[1rem] text-end">
|
<td class="text-sm sm:text-[1rem] text-end">
|
||||||
{@html abbreviateNumberWithColor(
|
{@html abbreviateNumberWithColor(
|
||||||
item?.call_volume,
|
item?.net_premium,
|
||||||
false,
|
false,
|
||||||
true,
|
true,
|
||||||
)}
|
)}
|
||||||
</td>
|
</td>
|
||||||
|
|
||||||
<td class="text-sm sm:text-[1rem] text-end">
|
<td class="text-sm sm:text-[1rem] text-end">
|
||||||
{@html abbreviateNumberWithColor(
|
{@html abbreviateNumberWithColor(
|
||||||
item?.avg30_call_volume,
|
item?.net_call_premium,
|
||||||
false,
|
false,
|
||||||
true,
|
true,
|
||||||
)}
|
)}
|
||||||
</td>
|
</td>
|
||||||
|
|
||||||
<td class="text-sm sm:text-[1rem] text-end">
|
<td class="text-sm sm:text-[1rem] text-end">
|
||||||
{@html abbreviateNumberWithColor(
|
{@html abbreviateNumberWithColor(
|
||||||
item?.put_volume,
|
item?.net_put_premium,
|
||||||
false,
|
false,
|
||||||
true,
|
true,
|
||||||
)}
|
)}
|
||||||
</td>
|
</td>
|
||||||
|
|
||||||
<td class="text-sm sm:text-[1rem] text-end">
|
<td class="text-sm sm:text-[1rem] text-end">
|
||||||
{@html abbreviateNumberWithColor(
|
{item?.iv_rank}
|
||||||
item?.avg30_put_volume,
|
|
||||||
false,
|
|
||||||
true,
|
|
||||||
)}
|
|
||||||
</td>
|
|
||||||
|
|
||||||
<td class="text-sm sm:text-[1rem] text-end">
|
|
||||||
{@html abbreviateNumberWithColor(
|
|
||||||
item?.call_premium,
|
|
||||||
false,
|
|
||||||
true,
|
|
||||||
)}
|
|
||||||
</td>
|
|
||||||
|
|
||||||
<td class="text-sm sm:text-[1rem] text-end">
|
|
||||||
{@html abbreviateNumberWithColor(
|
|
||||||
item?.put_premium,
|
|
||||||
false,
|
|
||||||
true,
|
|
||||||
)}
|
|
||||||
</td>
|
|
||||||
|
|
||||||
<td class="text-sm sm:text-[1rem] text-end">
|
|
||||||
<HoverCard.Root>
|
|
||||||
<HoverCard.Trigger
|
|
||||||
class="rounded-sm underline-offset-4 hover:underline focus-visible:outline-2 focus-visible:outline-offset-8 focus-visible:outline-black"
|
|
||||||
>
|
|
||||||
<div class="flex items-center justify-end">
|
|
||||||
<div
|
|
||||||
class="flex w-full max-w-28 h-5 bg-gray-200 rounded-md overflow-hidden"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="bg-red-500 h-full"
|
|
||||||
style="width: calc(({item
|
|
||||||
?.premium_ratio[0]} / ({item
|
|
||||||
?.premium_ratio[0]} + {item
|
|
||||||
?.premium_ratio[1]} + {item
|
|
||||||
?.premium_ratio[2]})) * 100%)"
|
|
||||||
></div>
|
|
||||||
|
|
||||||
<div
|
|
||||||
class="bg-gray-300 h-full"
|
|
||||||
style="width: calc(({item
|
|
||||||
?.premium_ratio[1]} / ({item
|
|
||||||
?.premium_ratio[0]} + {item
|
|
||||||
?.premium_ratio[1]} + {item
|
|
||||||
?.premium_ratio[2]})) * 100%)"
|
|
||||||
></div>
|
|
||||||
|
|
||||||
<div
|
|
||||||
class="bg-green-500 h-full"
|
|
||||||
style="width: calc(({item
|
|
||||||
?.premium_ratio[2]} / ({item
|
|
||||||
?.premium_ratio[0]} + {item
|
|
||||||
?.premium_ratio[1]} + {item
|
|
||||||
?.premium_ratio[2]})) * 100%)"
|
|
||||||
></div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</HoverCard.Trigger>
|
|
||||||
<HoverCard.Content
|
|
||||||
class="w-auto bg-secondary border border-gray-600"
|
|
||||||
>
|
|
||||||
<div class="flex justify-between space-x-4">
|
|
||||||
<div
|
|
||||||
class="space-y-1 flex flex-col items-start text-white"
|
|
||||||
>
|
|
||||||
<div>
|
|
||||||
Bearish: {@html abbreviateNumberWithColor(
|
|
||||||
item?.premium_ratio[0],
|
|
||||||
false,
|
|
||||||
true,
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
Neutral: {@html abbreviateNumberWithColor(
|
|
||||||
item?.premium_ratio[1],
|
|
||||||
false,
|
|
||||||
true,
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
Bullish: {@html abbreviateNumberWithColor(
|
|
||||||
item?.premium_ratio[2],
|
|
||||||
false,
|
|
||||||
true,
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</HoverCard.Content>
|
|
||||||
</HoverCard.Root>
|
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
{/each}
|
{/each}
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
</div>
|
</div>
|
||||||
-->
|
|
||||||
</div>
|
</div>
|
||||||
<UpgradeToPro {data} />
|
<UpgradeToPro {data} />
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user