frontend/src/routes/stocks/[tickerID]/financials/balance-sheet/+page.svelte
2024-12-07 16:23:01 +01:00

909 lines
36 KiB
Svelte
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

<script lang="ts">
import { Chart } from "svelte-echarts";
import {
numberOfUnreadNotification,
coolMode,
timeFrame,
displayCompanyName,
stockTicker,
} from "$lib/store";
import * as DropdownMenu from "$lib/components/shadcn/dropdown-menu/index.js";
import { Button } from "$lib/components/shadcn/button/index.js";
import { abbreviateNumber } from "$lib/utils";
import { goto } from "$app/navigation";
//import * as XLSX from 'xlsx';
import FinancialTable from "$lib/components/FinancialTable.svelte";
import { init, use } from "echarts/core";
import { LineChart, BarChart } from "echarts/charts";
import { GridComponent, TooltipComponent } from "echarts/components";
import { CanvasRenderer } from "echarts/renderers";
use([LineChart, BarChart, GridComponent, TooltipComponent, CanvasRenderer]);
export let data;
let balanceSheet = [];
let fullStatement = [];
let tableList = [];
let filterRule = "annual";
let optionsData;
let displayStatement = "cashAndCashEquivalents";
let activeIdx = 0;
const tabs = [
{
title: "Annual",
},
{
title: "Quarterly",
},
];
const statementConfig = [
{
propertyName: "cashAndCashEquivalents",
growthPropertyName: "growthCashAndCashEquivalents",
label: "Cash & Equivalents",
text: "Cash and equivalents is the amount of money on the company's accounts held as straight cash, or very liquid assets that can be sold for cash at a very short notice.",
},
{
propertyName: "shortTermInvestments",
growthPropertyName: "growthShortTermInvestments",
label: "Short-Term Investments",
text: "Short-term investments are liquid assets like treasury bills, short-term bonds, money-market funds, marketable securities and other investments that can be sold for cash at a short notice.",
},
{
propertyName: "longTermInvestments",
growthPropertyName: "growthLongTermInvestments",
label: "Long-Term Investments",
text: "Long-term investments are investments that the company plans to hold for more than one year. It can include stocks, bonds, real estate and others.",
},
{
propertyName: "otherNonCurrentAssets",
growthPropertyName: "growthOtherNonCurrentAssets",
label: "Other Long-Term Assets",
text: "Other long-term assets include all long-term assets that do not fit into any of the categories mentioned so far.",
},
{
propertyName: "netReceivables",
growthPropertyName: "growthNetReceivables",
label: "Receivables",
text: "Receivables are the money owed to the company for products or services that have been delivered but not yet paid for. If a customer buys something on credit, it is listed under receivables (as a current asset) on the balance sheet.",
},
{
propertyName: "inventory",
growthPropertyName: "growthInventory",
label: "Inventory",
text: "Inventory is the value of product that is available for sale, as well as the value of purchased raw materials for making goods that will be sold. It also includes goods that are currently being produced from raw materials.",
},
{
propertyName: "otherCurrentAssets",
growthPropertyName: "growthOtherCurrentAssets",
label: "Other Current Assets",
text: "Other current assets includes all current assets that do not fit into any of the categories mentioned so far.",
},
{
propertyName: "totalCurrentAssets",
growthPropertyName: "growthTotalCurrentAssets",
label: "Total Current Assets",
text: "Total current assets includes all current assets, including cash and equivalents, short-term investments, receivables, inventory and others. Current assets are things that easily be sold for cash or will be used within one year.",
},
{
propertyName: "propertyPlantEquipmentNet",
growthPropertyName: "growthPropertyPlantEquipmentNet",
label: "Property-Plant & Equipment",
text: "Property, Plant & Equipment are all long-term tangible or physical assets that are needed for business operations. It includes buildings, factories, machinery, furniture and others.",
},
{
propertyName: "goodwillAndIntangibleAssets",
growthPropertyName: "growthGoodwillAndIntangibleAssets",
label: "Goodwill & Intangibles",
text: "Includes goodwill and other assets that are intangible. Intangible assets are assets that provide some benefit for the company, but they are not physical assets that can be measured or counted. Examples include patents, intellectual property and copyrights.",
},
{
propertyName: "totalNonCurrentAssets",
growthPropertyName: "growthTotalNonCurrentAssets",
label: "Total Long-Term Assets",
text: "Total long-term assets includes all long-term assets, including Property-Plant & Equipment, goodwill, intangibles and others. Long-term (non-current) assets are things that can not be sold for cash easily or are considered to last for more than one year.",
},
{
propertyName: "totalAssets",
growthPropertyName: "growthTotalAssets",
label: "Total Assets",
text: "Total assets is the sum of all current and non-current assets on the balance sheet. Assets are everything that the company owns.",
},
{
propertyName: "accountPayables",
growthPropertyName: "growthAccountPayables",
label: "Account Payables",
text: "Accounts payable is the amount that the company owes to vendors and suppliers. The company has purchased products or services on credit, but has not paid for them yet.",
},
{
propertyName: "deferredRevenue",
growthPropertyName: "growthDeferredRevenue",
label: "Deferred Revenue",
text: "Deferred revenue includes payments that have been received in advance for products and services that have not yet been delivered. These revenues are listed as a liability on the company's balance sheet.",
},
{
propertyName: "shortTermDebt",
growthPropertyName: "growthShortTermDebt",
label: "Short-Term Debt",
text: "Current debt is company debt that needs to be paid within one year. It also includes the portion of long-term debt that is due within a year.",
},
{
propertyName: "otherCurrentLiabilities",
growthPropertyName: "growthOtherCurrentLiabilities",
label: "Other Current Liabilities",
text: "Other current liabilities are all current liabilities that do not fit into the categories above.",
},
{
propertyName: "totalCurrentLiabilities",
growthPropertyName: "growthTotalCurrentLiabilities",
label: "Total Current Liabilities",
text: "Total current liabilities are all financial obligations that the company owes and are due within one year. This includes accounts payable, deferred revenue, current debt and others.",
},
{
propertyName: "longTermDebt",
growthPropertyName: "growthLongTermDebt",
label: "Long-Term Debt",
text: "Long-term debt is debt that the company does not need to pay until after one year or more. It includes bank loans and bonds issued by the company.",
},
{
propertyName: "otherNonCurrentLiabilities",
growthPropertyName: "growthOtherNonCurrentLiabilities",
label: "Other Long-Term Liabilities",
text: "Other long-term liabilities are all long-term (non-current) liabilities that are not categorized as long-term debt.",
},
{
propertyName: "totalNonCurrentLiabilities",
growthPropertyName: "growthTotalNonCurrentLiabilities",
label: "Total Long-Term Liabilities",
text: "Total long-term liabilities are all long-term (non-current) financial obligations of the company, including long-term debt and others.",
},
{
propertyName: "totalLiabilities",
growthPropertyName: "growthTotalLiabilities",
label: "Total Liabilities",
text: "Total liabilities are all financial obligations of the company, including both current and long-term (non-current) liabilities. Liabilities are everything that the company owes.",
},
{
propertyName: "totalDebt",
growthPropertyName: "growthTotalDebt",
label: "Total Debt",
text: "Total debt is the total amount of liabilities categorized as debt on the balance sheet. It includes both current and long-term (non-current) debt.",
},
{
propertyName: "commonStock",
growthPropertyName: "growthCommonStock",
label: "Common Stock",
text: "Common stock is the par value of the company's outstanding common stock, multiplied by the par value. This information is not very useful as the par value is usually set as an arbitrary amount of one cent.",
},
{
propertyName: "retainedEarnings",
growthPropertyName: "growthRetainedEarnings",
label: "Retained Earnings",
text: "Retained earnings are net income previously earned that has not been paid out to shareholders as dividends. If retained earnings are negative, they can be listed as Accumulated Deficit instead.",
},
{
propertyName: "accumulatedOtherComprehensiveIncomeLoss",
growthPropertyName: "growthAccumulatedOtherComprehensiveIncomeLoss",
label: "Comprehensive Income",
text: "Comprehensive income includes unrealized gains and losses that do not fall under retained earnings.",
},
{
propertyName: "totalStockholdersEquity",
growthPropertyName: "growthTotalStockholdersEquity",
label: "Shareholders Equity",
text: "Shareholders equity is also called book value or net worth. It can be seen as the amount of money held by investors inside the company. It is calculated by subtracting all liabilities from all assets.",
},
{
propertyName: "totalInvestments",
growthPropertyName: "growthTotalInvestments",
label: "Total Investments",
text: "Total Investments encompass all a company's financial assets, such as stocks, bonds, and real estate, reflecting its financial health and growth potential. Calculated by summing up these asset values, it's a key indicator of a company's financial strength.",
},
];
const fields = statementConfig.map((item) => ({
label: item.label,
key: item.propertyName,
}));
function toggleMode() {
$coolMode = !$coolMode;
}
function changeStatement(event) {
displayStatement = event.target.value;
}
function normalizer(value) {
if (Math?.abs(value) >= 1e18) {
return { unit: "Q", denominator: 1e18 };
} else if (Math?.abs(value) >= 1e12) {
return { unit: "T", denominator: 1e12 };
} else if (Math?.abs(value) >= 1e9) {
return { unit: "B", denominator: 1e9 };
} else if (Math?.abs(value) >= 1e6) {
return { unit: "M", denominator: 1e6 };
} else if (Math?.abs(value) >= 1e5) {
return { unit: "K", denominator: 1e5 };
} else {
return { unit: "", denominator: 1 };
}
}
function plotData() {
let labelName = "-";
let xList = [];
let valueList = [];
tableList = [];
const index = statementConfig?.findIndex(
(item) => item?.propertyName === displayStatement,
);
for (let i = balanceSheet?.length - 1; i >= 0; i--) {
const statement = balanceSheet[i];
const year = statement?.calendarYear?.slice(-2);
const quarter = statement?.period;
// Determine the label based on filterRule
if (filterRule === "annual") {
xList.push("FY" + year);
} else {
xList.push("FY" + year + " " + quarter);
}
// Calculate the value and growth
const value = Number(
statement[statementConfig[index]?.propertyName],
)?.toFixed(2);
valueList.push(value);
// Add the entry to tableList
tableList.push({
date: statement?.date,
value: value,
});
}
//sort tableList by date
tableList?.sort((a, b) => new Date(b?.date) - new Date(a?.date));
labelName = statementConfig[index]?.label;
const { unit, denominator } = normalizer(Math.max(...valueList) ?? 0);
const options = {
animation: false,
grid: {
left: "0%",
right: "0%",
bottom: "2%",
top: "10%",
containLabel: true,
},
xAxis: {
axisLabel: {
color: "#fff",
},
data: xList,
type: "category",
},
yAxis: [
{
type: "value",
splitLine: {
show: false, // Disable x-axis grid lines
},
axisLabel: {
show: false, // Hide y-axis labels
},
},
],
series: [
{
name: labelName,
data: valueList,
type: "bar",
smooth: true,
itemStyle: {
color: "#fff",
},
},
],
tooltip: {
trigger: "axis",
hideDelay: 100,
},
};
return options;
}
fullStatement = data?.getBalanceSheetStatement;
displayStatement = "cashAndCashEquivalents";
const getCurrentYear = () => new Date()?.getFullYear();
const filterStatement = (fullStatement, timeFrame) => {
const currentYear = getCurrentYear();
switch (timeFrame) {
case "5Y":
return fullStatement?.filter(
(item) => currentYear - parseInt(item?.calendarYear) < 5,
);
case "10Y":
return fullStatement?.filter(
(item) => currentYear - parseInt(item?.calendarYear) < 10,
);
default:
return fullStatement;
}
};
const exportFundamentalData = (format = "csv") => {
if (data?.user?.tier === "Pro") {
const data = fullStatement;
if (!data || data.length === 0) {
return;
}
let properties = [
{
key: filterRule === "annual" ? "calendarYear" : "date",
label: filterRule === "annual" ? "Year" : "Quarter",
},
];
for (let i = 0; i < statementConfig?.length; i++) {
properties.push({
key: statementConfig[i]?.propertyName,
label: statementConfig[i]?.label,
});
}
// Helper function to handle special cases
// Create rows for CSV/Excel
let rows = data.map((item) =>
properties?.map((property) => item[property?.key] || 0),
);
// Include headers
const headers = properties.map((prop) => prop.label);
rows.unshift(headers);
// Check the format to export
if (format.toLowerCase() === "csv") {
const csvContent = rows.map((row) => row.join(",")).join("\n");
const blob = new Blob([csvContent], {
type: "data:text/csv;charset=utf-8",
});
const url = URL.createObjectURL(blob);
const a = document.createElement("a");
a.href = url;
a.download =
$stockTicker.toLowerCase() +
`${filterRule === "annual" ? "_annual" : "_quarter"}_cashflow_statement.csv`;
document.body.appendChild(a);
a.click();
document.body.removeChild(a);
URL.revokeObjectURL(url);
} /*else if (format.toLowerCase() === "excel") {
const worksheet = XLSX.utils.aoa_to_sheet(rows);
const workbook = XLSX.utils.book_new();
XLSX.utils.book_append_sheet(workbook, worksheet, "Ratios Statement");
XLSX.writeFile(
workbook,
`${$stockTicker.toLowerCase()}-ratios-statement.xlsx`,
);
}
*/
} else {
goto("/pricing");
}
};
$: {
if ($timeFrame || displayStatement || activeIdx) {
if (activeIdx === 0) {
filterRule = "annual";
fullStatement = data?.getBalanceSheetStatement?.annual;
} else {
filterRule = "quarterly";
fullStatement = data?.getBalanceSheetStatement?.quarter;
}
balanceSheet = filterStatement(fullStatement, $timeFrame);
if ($coolMode === true) {
optionsData = plotData();
}
}
}
</script>
<svelte:head>
<meta charset="utf-8" />
<meta name="viewport" content="width=device-width" />
<title>
{$numberOfUnreadNotification > 0 ? `(${$numberOfUnreadNotification})` : ""}
{$displayCompanyName} ({$stockTicker}) Balance Sheet &#183 stocknear
</title>
<meta
name="description"
content={`Detailed balance sheet for ${$displayCompanyName} (${$stockTicker}), including cash, debt, assets, liabilities, and book value.`}
/>
<!-- Other meta tags -->
<meta
property="og:title"
content={`${$displayCompanyName} (${$stockTicker}) Balance Sheet &#183 stocknear`}
/>
<meta
property="og:description"
content={`Detailed balance sheet for ${$displayCompanyName} (${$stockTicker}), including cash, debt, assets, liabilities, and book value.`}
/>
<meta property="og:type" content="website" />
<!-- Add more Open Graph meta tags as needed -->
<!-- Twitter specific meta tags -->
<meta name="twitter:card" content="summary_large_image" />
<meta
name="twitter:title"
content={`${$displayCompanyName} (${$stockTicker}) Balance Sheet &#183 stocknear`}
/>
<meta
name="twitter:description"
content={`Detailed balance sheet for ${$displayCompanyName} (${$stockTicker}), including cash, debt, assets, liabilities, and book value.`}
/>
<!-- Add more Twitter meta tags as needed -->
</svelte:head>
<section class="bg-[#09090B] overflow-hidden text-white h-full w-full">
<div
class="flex justify-center w-full m-auto h-full overflow-hidden mt-4 sm:mt-0"
>
<div
class="relative flex justify-center items-center overflow-hidden w-full"
>
<main class="w-full">
<div class="sm:p-7 m-auto mt-2 sm:mt-0">
<div class="mb-3">
<h1 class="text-xl sm:text-2xl text-white font-bold">
{#if $coolMode}
{statementConfig?.find(
(item) => item?.propertyName === displayStatement,
)?.label}
{:else}
Balance Sheet {filterRule === "annual"
? "(Annual)"
: "(Quarter)"}
{/if}
</h1>
</div>
<div class="grid grid-cols-1 gap-2">
<div
class="text-white p-3 sm:p-5 rounded-md sm:flex sm:flex-row sm:items-center border border-gray-600 text-sm sm:text-[1rem]"
>
<svg
class="w-6 h-6 flex-shrink-0 inline-block sm:mr-2"
xmlns="http://www.w3.org/2000/svg"
viewBox="0 0 256 256"
><path
fill="#fff"
d="M128 24a104 104 0 1 0 104 104A104.11 104.11 0 0 0 128 24m-4 48a12 12 0 1 1-12 12a12 12 0 0 1 12-12m12 112a16 16 0 0 1-16-16v-40a8 8 0 0 1 0-16a16 16 0 0 1 16 16v40a8 8 0 0 1 0 16"
/></svg
>
{#if $coolMode}
{statementConfig?.find(
(item) => item?.propertyName === displayStatement,
)?.text}
{:else if balanceSheet?.length > 0}
Get detailed breakdowns of the balance-sheet with total debts,
total investments, and much more.
{:else}
No financial data available for {$displayCompanyName}
{/if}
</div>
{#if balanceSheet?.length > 0}
<div
class="inline-flex justify-center w-full rounded-md sm:w-auto sm:ml-auto mt-3 mb-6"
>
<div
class="bg-secondary w-full min-w-24 sm:w-fit relative flex flex-wrap items-center justify-center rounded-md p-1 mt-4"
>
{#each tabs as item, i}
{#if data?.user?.tier !== "Pro" && i > 0}
<button
on:click={() => goto("/pricing")}
class="group relative z-[1] rounded-full w-1/2 min-w-24 md:w-auto px-5 py-1"
>
<span class="relative text-sm block font-semibold">
{item.title}
<svg
class="inline-block ml-0.5 -mt-1 w-3.5 h-3.5"
xmlns="http://www.w3.org/2000/svg"
viewBox="0 0 24 24"
><path
fill="#A3A3A3"
d="M17 9V7c0-2.8-2.2-5-5-5S7 4.2 7 7v2c-1.7 0-3 1.3-3 3v7c0 1.7 1.3 3 3 3h10c1.7 0 3-1.3 3-3v-7c0-1.7-1.3-3-3-3M9 7c0-1.7 1.3-3 3-3s3 1.3 3 3v2H9z"
/></svg
>
</span>
</button>
{:else}
<button
on:click={() => (activeIdx = i)}
class="group relative z-[1] rounded-full w-1/2 min-w-24 md:w-auto px-5 py-1 {activeIdx ===
i
? 'z-0'
: ''} "
>
{#if activeIdx === i}
<div
class="absolute inset-0 rounded-md bg-[#fff]"
></div>
{/if}
<span
class="relative text-sm block font-semibold {activeIdx ===
i
? 'text-black'
: 'text-white'}"
>
{item.title}
</span>
</button>
{/if}
{/each}
</div>
</div>
<div
class="mb-6 sm:mb-3 flex flex-row items-center w-full justify-end sm:justify-center mt-3 sm:mt-0"
>
<label
class="inline-flex mt-2 sm:mt-0 cursor-pointer relative mr-auto"
>
<input
on:click={toggleMode}
type="checkbox"
checked={$coolMode}
value={$coolMode}
class="sr-only peer"
/>
<div
class="w-11 h-6 bg-gray-400 rounded-full peer-checked:after:translate-x-full peer-checked:after:border-white after:content-[''] after:absolute after:top-0.5 after:left-[2px] after:bg-white after:border-gray-300 after:border after:rounded-full after:h-5 after:w-5 after:transition-all peer-checked:bg-[#1563F9]"
></div>
{#if $coolMode}
<span class="ml-2 text-sm font-medium text-white">
Cool Mode
</span>
{:else}
<span class="ml-2 text-sm font-medium text-white">
Boring Mode
</span>
{/if}
</label>
<div
class="flex flex-row items-center w-fit sm:w-[50%] md:w-auto sm:ml-auto"
>
<div class="relative inline-block text-left grow">
<DropdownMenu.Root>
<DropdownMenu.Trigger asChild let:builder>
<Button
builders={[builder]}
class="w-full border-gray-600 border bg-[#09090B] sm:hover:bg-primary ease-out flex flex-row justify-between items-center px-3 py-2 text-white rounded-md truncate"
>
<span class="truncate text-white">{$timeFrame}</span>
<svg
class="-mr-1 ml-1 h-5 w-5 xs:ml-2 inline-block"
viewBox="0 0 20 20"
fill="currentColor"
style="max-width:40px"
aria-hidden="true"
>
<path
fill-rule="evenodd"
d="M5.293 7.293a1 1 0 011.414 0L10 10.586l3.293-3.293a1 1 0 111.414 1.414l-4 4a1 1 0 01-1.414 0l-4-4a1 1 0 010-1.414z"
clip-rule="evenodd"
></path>
</svg>
</Button>
</DropdownMenu.Trigger>
<DropdownMenu.Content
class="w-56 h-fit max-h-72 overflow-y-auto scroller"
>
<DropdownMenu.Label class="text-gray-400">
Select time frame
</DropdownMenu.Label>
<DropdownMenu.Separator />
<DropdownMenu.Group>
<DropdownMenu.Item
on:click={() => ($timeFrame = "5Y")}
class="cursor-pointer hover:bg-primary"
>
5 years
</DropdownMenu.Item>
<DropdownMenu.Item
on:click={() => ($timeFrame = "10Y")}
class="cursor-pointer hover:bg-primary"
>
10 years
</DropdownMenu.Item>
<DropdownMenu.Item
on:click={() => ($timeFrame = "MAX")}
class="cursor-pointer hover:bg-primary"
>
Max
</DropdownMenu.Item>
</DropdownMenu.Group>
</DropdownMenu.Content>
</DropdownMenu.Root>
</div>
<Button
on:click={() => exportFundamentalData("csv")}
class="ml-2 w-full border-gray-600 border bg-[#09090B] sm:hover:bg-primary ease-out flex flex-row justify-between items-center px-3 py-2 text-white rounded-md truncate"
>
<span class="truncate text-white">Download</span>
<svg
class="{data?.user?.tier === 'Pro'
? 'hidden'
: ''} ml-1 -mt-0.5 w-3.5 h-3.5"
xmlns="http://www.w3.org/2000/svg"
viewBox="0 0 24 24"
><path
fill="#A3A3A3"
d="M17 9V7c0-2.8-2.2-5-5-5S7 4.2 7 7v2c-1.7 0-3 1.3-3 3v7c0 1.7 1.3 3 3 3h10c1.7 0 3-1.3 3-3v-7c0-1.7-1.3-3-3-3M9 7c0-1.7 1.3-3 3-3s3 1.3 3 3v2H9z"
/></svg
>
</Button>
</div>
</div>
{#if $coolMode}
<div class="sm:w-full">
<div class="relative">
<select
class="w-40 select select-bordered select-sm p-0 pl-5 overflow-y-auto bg-secondary"
on:change={changeStatement}
>
<option disabled>Choose an Income Variable</option>
<option value="cashAndCashEquivalents" selected
>Cash & Equivalents</option
>
<option value="shortTermInvestments"
>Short-Term Investments</option
>
<option value="longTermInvestments"
>Long-Term Investments</option
>
<option value="otherNonCurrentAssets"
>Other Long-Term Assets</option
>
<option value="netReceivables">Receivables</option>
<option value="inventory">Inventory</option>
<option value="otherCurrentAssets"
>Other Current Assets</option
>
<option value="totalCurrentAssets"
>Total Current Assets</option
>
<option value="propertyPlantEquipmentNet"
>Property, Plant & Equipment</option
>
<option value="goodwillAndIntangibleAssets"
>Goodwill & Intangibles</option
>
<option value="totalNonCurrentAssets"
>Total Long-Term Assets</option
>
<option value="totalAssets">Total Assets</option>
<option value="accountPayables">Account Payables</option>
<option value="deferredRevenue">Deferred Revenue</option>
<option value="shortTermDebt">Short-Term Debt</option>
<option value="otherCurrentLiabilities"
>Other Current Liabilities</option
>
<option value="totalCurrentLiabilities"
>Total Current Liabilities</option
>
<option value="longTermDebt">Long-Term Debt</option>
<option value="otherNonCurrentLiabilities"
>Other Long-Term Liabilities</option
>
<option value="totalNonCurrentLiabilities"
>Total Long-Term Liabilities</option
>
<option value="totalLiabilities">Total Liabilities</option
>
<option value="totalDebt">Total Debt</option>
<option value="commonStock">Common Stock</option>
<option value="retainedEarnings">Retained Earnigns</option
>
<option value="accumulatedOtherComprehensiveIncomeLoss"
>Comprehensive Income</option
>
<option value="totalStockholdersEquity"
>Shareholders' Equity</option
>
<option value="totalInvestments">Total Investments</option
>
</select>
</div>
</div>
<div class="app w-full">
<Chart {init} options={optionsData} class="chart" />
</div>
<h2 class="mt-5 text-2xl text-gray-200 font-semibold">
{statementConfig?.find(
(item) => item?.propertyName === displayStatement,
)?.label} History
</h2>
<div class="w-full overflow-x-scroll">
<table
class="table table-sm table-compact rounded-none sm:rounded-md w-full border-bg-[#09090B] m-auto mt-4"
>
<thead>
<tr class="border border-gray-600">
<th class="text-white font-semibold text-start text-sm"
>{filterRule === "annual"
? "Fiscal Year End"
: "Quarter Ends"}</th
>
<th class="text-white font-semibold text-end text-sm"
>{statementConfig?.find(
(item) => item?.propertyName === displayStatement,
)?.label}</th
>
<th class="text-white font-semibold text-end text-sm"
>Change</th
>
<th class="text-white font-semibold text-end text-sm"
>Growth</th
>
</tr>
</thead>
<tbody>
{#each tableList as item, index}
<!-- row -->
<tr
class="sm:hover:bg-[#245073] sm:hover:bg-opacity-[0.2] odd:bg-secondary border-b-[#09090B] shake-ticker cursor-pointer"
>
<td
class="text-white font-medium text-sm sm:text-[1rem] whitespace-nowrap border-b-[#09090B]"
>
{item?.date}
</td>
<td
class="text-white text-sm sm:text-[1rem] text-right whitespace-nowrap border-b-[#09090B]"
>
{abbreviateNumber(item?.value)}
</td>
<td
class="text-white text-sm sm:text-[1rem] whitespace-nowrap font-medium text-end border-b-[#09090B]"
>
{item?.value - tableList[index + 1]?.value !== 0
? abbreviateNumber(
(
item?.value - tableList[index + 1]?.value
)?.toFixed(2),
)
: "-"}
</td>
<td
class="text-white text-sm sm:text-[1rem] whitespace-nowrap font-medium text-end border-b-[#09090B]"
>
{#if index + 1 - tableList?.length === 0}
-
{:else if item?.value === 0 && tableList[index + 1]?.value < 0}
<span class="text-[#FF2F1F]">-100.00%</span>
{:else if item?.value === 0 && tableList[index + 1]?.value > 0}
<span class="text-[#00FC50]">100.00%</span>
{:else if item?.value - tableList[index + 1]?.value > 0}
<span class="text-[#00FC50]">
{(
((item?.value - tableList[index + 1]?.value) /
Math.abs(item?.value)) *
100
)?.toFixed(2)}%
</span>
{:else if item?.value - tableList[index + 1]?.value < 0}
<span class="text-[#FF2F1F]">
-{(
Math?.abs(
(tableList[index + 1]?.value -
item?.value) /
Math.abs(item?.value),
) * 100
)?.toFixed(2)}%
</span>
{:else}
-
{/if}
</td>
</tr>
{/each}
</tbody>
</table>
</div>
{:else}
<div
class="w-full rounded-none sm:rounded-md m-auto overflow-x-auto"
>
<table class="table table-sm table-compact w-full">
<thead>
<tr class="text-white">
<td
class="text-start bg-[#09090B] text-white text-sm font-semibold pr-10"
>Year</td
>
{#each balanceSheet as cash}
{#if filterRule === "annual"}
<td
class="bg-[#09090B] font-semibold text-sm text-end"
>
{"FY" + cash?.calendarYear?.slice(-2)}
</td>
{:else}
<td
class="bg-[#09090B] font-semibold text-sm text-end"
>
{"FY" +
cash?.calendarYear?.slice(-2) +
" " +
cash?.period}
</td>
{/if}
{/each}
</tr>
</thead>
<tbody>
<!-- row -->
<FinancialTable data={balanceSheet} {fields} />
</tbody>
</table>
</div>
{/if}
{/if}
</div>
</div>
</main>
</div>
</div>
</section>
<style>
.app {
height: 400px;
width: 100%;
}
@media (max-width: 560px) {
.app {
width: 100%;
height: 300px;
}
}
.chart {
width: 100%;
}
</style>