frontend/src/routes/stock-screener/[strategyId]/+page.svelte
2024-09-01 20:33:19 +02:00

1504 lines
69 KiB
Svelte

<script lang='ts'>
import { onMount } from 'svelte';
import { goto} from '$app/navigation';
import { screenWidth, strategyId, numberOfUnreadNotification, getCache, setCache} from '$lib/store';
import toast from 'svelte-french-toast';
import { abbreviateNumber, formatRuleValue } from '$lib/utils';
import RuleControl from '$lib/components/RuleControl.svelte';
//const userConfirmation = confirm('Unsaved changes detected. Leaving now will discard your strategy. Continue?');
export let data;
export let form;
$strategyId = data?.getStrategyId;
let ruleOfList = data?.getStrategy?.rules ?? [];
const title = data?.getStrategy?.title;
let stockScreenerData = data?.getStockScreenerData?.filter(item =>
Object?.values(item)?.every(value =>
value !== null && value !== undefined &&
(typeof value !== 'object' || Object?.values(value)?.every(subValue => subValue !== null && subValue !== undefined))
)
);
const getStockScreenerData = async (rules) => {
const ruleNames = rules?.map(rule => rule?.name)?.sort()?.join(',');
const cachedData = getCache(ruleNames, 'getStockScreenerData');
if (cachedData) {
console.log('Using cached data');
return cachedData;
}
console.log('Fetching new data from API');
const postData = { ruleOfList: rules?.map(rule => rule.name) };
const response = await fetch(data?.apiURL + '/stock-screener-data', {
method: 'POST',
headers: {
"Content-Type": "application/json",
"X-API-KEY": data?.apiKey
},
body: JSON.stringify(postData)
});
const output = await response.json();
// Cache the new data
setCache(ruleNames, output, 'getStockScreenerData');
return output;
};
let filteredData = [];
let displayResults = [];
let isSaved = false;
let ruleCondition = {
avgVolume: (ruleOfList?.find(item => item.name === "avgVolume") || { condition: 'over' }).condition,
priceToBookRatio: (ruleOfList?.find(item => item.name === "priceToBookRati") || { condition: 'over' }).condition,
priceToSalesRatio: (ruleOfList?.find(item => item.name === "priceToSalesRatio") || { condition: 'over' }).condition,
rsi: (ruleOfList?.find(item => item.name === "rsi") || { condition: 'over' }).condition,
stochRSI: (ruleOfList?.find(item => item.name === "stochRSI") || { condition: 'over' }).condition,
mfi: (ruleOfList?.find(item => item.name === "mfi") || { condition: 'over' }).condition,
cci: (ruleOfList?.find(item => item.name === "cci") || { condition: 'over' }).condition,
atr: (ruleOfList?.find(item => item.name === "atr") || { condition: 'over' }).condition,
sma50: (ruleOfList?.find(item => item.name === "sma50") || { condition: 'over' }).condition,
sma200: (ruleOfList?.find(item => item.name === "sma200") || { condition: 'over' }).condition,
ema50: (ruleOfList?.find(item => item.name === "ema50") || { condition: 'over' }).condition,
ema200: (ruleOfList?.find(item => item.name === "ema200") || { condition: 'over' }).condition,
change1W: (ruleOfList?.find(item => item.name === "change1W") || { condition: 'over' }).condition,
change1M: (ruleOfList?.find(item => item.name === "change1M") || { condition: 'over' }).condition,
change3M: (ruleOfList?.find(item => item.name === "change3M") || { condition: 'over' }).condition,
change6M: (ruleOfList?.find(item => item.name === "change6M") || { condition: 'over' }).condition,
change1Y: (ruleOfList?.find(item => item.name === "change1Y") || { condition: 'over' }).condition,
change3Y: (ruleOfList?.find(item => item.name === "change3Y") || { condition: 'over' }).condition,
payoutRatio: (ruleOfList?.find(item => item.name === "payoutRatio") || { condition: 'over' }).condition,
annualDividend: (ruleOfList?.find(item => item.name === "annualDividend") || { condition: 'over' }).condition,
dividendYield: (ruleOfList?.find(item => item.name === "dividendYield") || { condition: 'over' }).condition,
dividendGrowth: (ruleOfList?.find(item => item.name === "dividendGrowth") || { condition: 'over' }).condition,
eps: (ruleOfList?.find(item => item.name === "eps") || { condition: 'over' }).condition,
growthEPS: (ruleOfList?.find(item => item.name === "growthEPS") || { condition: 'over' }).condition,
pe: (ruleOfList?.find(item => item.name === "pe") || { condition: 'over' }).condition,
forwardPE: (ruleOfList?.find(item => item.name === "forwardPE") || { condition: 'over' }).condition,
beta: (ruleOfList?.find(item => item.name === "beta") || { condition: 'over' }).condition,
ebitda: (ruleOfList?.find(item => item.name === "ebitda") || { condition: 'over' }).condition,
growthEBITDA: (ruleOfList?.find(item => item.name === "growthEBITDA") || { condition: 'over' }).condition,
revenue: (ruleOfList?.find(item => item.name === "revenue") || { condition: 'over' }).condition,
growthRevenue: (ruleOfList?.find(item => item.name === "growthRevenue") || { condition: 'over' }).condition,
costOfRevenue: (ruleOfList?.find(item => item.name === "costOfRevenue") || { condition: 'over' }).condition,
growthCostOfRevenue: (ruleOfList?.find(item => item.name === "growthCostOfRevenue") || { condition: 'over' }).condition,
costAndExpenses: (ruleOfList?.find(item => item.name === "costAndExpenses") || { condition: 'over' }).condition,
growthCostAndExpenses: (ruleOfList?.find(item => item.name === "growthCostAndExpenses") || { condition: 'over' }).condition,
netIncome: (ruleOfList?.find(item => item.name === "netIncome") || { condition: 'over' }).condition,
growthNetIncome: (ruleOfList?.find(item => item.name === "growthNetIncome") || { condition: 'over' }).condition,
researchAndDevelopmentExpenses: (ruleOfList?.find(item => item.name === "researchAndDevelopmentExpenses") || { condition: 'over' }).condition,
growthResearchAndDevelopmentExpenses: (ruleOfList?.find(item => item.name === "growthResearchAndDevelopmentExpenses") || { condition: 'over' }).condition,
grossProfit: (ruleOfList?.find(item => item.name === "grossProfit") || { condition: 'over' }).condition,
growthGrossProfit: (ruleOfList?.find(item => item.name === "growthGrossProfit") || { condition: 'over' }).condition,
interestIncome: (ruleOfList?.find(item => item.name === "interestIncome") || { condition: 'over' }).condition,
interestExpense: (ruleOfList?.find(item => item.name === "interestExpense") || { condition: 'over' }).condition,
growthInterestExpense: (ruleOfList?.find(item => item.name === "growthInterestExpense") || { condition: 'over' }).condition,
operatingExpenses: (ruleOfList?.find(item => item.name === "operatingExpense") || { condition: 'over' }).condition,
growthOperatingExpenses: (ruleOfList?.find(item => item.name === "growthOperatingExpense") || { condition: 'over' }).condition,
operatingIncome: (ruleOfList?.find(item => item.name === "operatingIncome") || { condition: 'over' }).condition,
growthOperatingIncome: (ruleOfList?.find(item => item.name === "growthOperatingIncome") || { condition: 'over' }).condition,
marketCap: (ruleOfList?.find(item => item.name === "marketCap") || { condition: 'over' }).condition,
var: (ruleOfList?.find(item => item.name === "var") || { condition: 'over' }).condition,
trendAnalysis: (ruleOfList?.find(item => item.name === "trendAnalysis") || { condition: 'over' }).condition,
fundamentalAnalysis: (ruleOfList?.find(item => item.name === "fundamentalAnalysis") || { condition: 'over' }).condition,
currentRatio: (ruleOfList?.find(item => item.name === "currentRatio") || { condition: 'over' }).condition,
quickRatio: (ruleOfList?.find(item => item.name === "quickRatio") || { condition: 'over' }).condition,
debtEquityRatio: (ruleOfList?.find(item => item.name === "debtEquityRatio") || { condition: 'over' }).condition,
debtRatio: (ruleOfList?.find(item => item.name === "debtRatio") || { condition: 'over' }).condition,
returnOnAssets: (ruleOfList?.find(item => item.name === "returnOnAssets") || { condition: 'over' }).condition,
returnOnEquity: (ruleOfList?.find(item => item.name === "returnOnEquity") || { condition: 'over' }).condition,
enterpriseValue: (ruleOfList?.find(item => item.name === "enterpriseValue") || { condition: 'over' }).condition,
freeCashFlowPerShare: (ruleOfList?.find(item => item.name === "freeCashFlowPerShare") || { condition: 'over' }).condition,
cashPerShare: (ruleOfList?.find(item => item.name === "cashPerShare") || { condition: 'over' }).condition,
priceToFreeCashFlowsRatio: (ruleOfList?.find(item => item.name === "priceToFreeCashFlowsRatio") || { condition: 'over' }).condition,
};
let ruleTrend = {
ratingRecommendation: 'Hold',
};
let allRows = [
{ rule: 'avgVolume', label: 'Avg Volume', max: "50", min:"1", step:"0.5", unit: 'Mio', category: 'fund' },
{ rule: 'rsi', label: 'RSI', max: "100", min:"0", step:"2", category: 'ta' },
{ rule: 'stochRSI', label: 'Stoch RSI Fast', max: "100", min:"0", step:"2",category: 'ta' },
{ rule: 'mfi', label: 'MFI', max: "100", min:"0", step:"2", category: 'ta' },
{ rule: 'cci', label: 'CCI', max: "300", min:"-300", step:"10",category: 'ta' },
{ rule: 'atr', label: 'ATR', max: "20", min:"0", step:"0.5",category: 'ta' },
{ rule: 'sma50', label: 'SMA-50', max: "500", min:"0", step:"10", category: 'ta' },
{ rule: 'sma200', label: 'SMA-200', max: "500", min:"0", step:"10", category: 'ta' },
{ rule: 'ema50', label: 'EMA-50', max: "500", min:"0", step:"10", category: 'ta' },
{ rule: 'ema200', label: 'EMA-200', max: "500", min:"0", step:"10",category: 'ta' },
{ rule: 'change1W', label: 'Price Change 1W [%]', max: "200", min:"-100", step:"2", unit: '%',category: 'ta' },
{ rule: 'change1M', label: 'Price Change 1M [%]', max: "200", min:"-100", step:"2", unit: '%',category: 'ta' },
{ rule: 'change3M', label: 'Price Change 3M [%]', max: "200", min:"-100", step:"2", unit: '%',category: 'ta' },
{ rule: 'change6M', label: 'Price Change 6M [%]', max: "200", min:"-100", step:"2", unit: '%',category: 'ta' },
{ rule: 'change1Y', label: 'Price Change 1Y [%]', max: "200", min:"-100", step:"2", unit: '%',category: 'ta' },
{ rule: 'change3Y', label: 'Price Change 3Y [%]', max: "200", min:"-100", step:"2", unit: '%',category: 'ta' },
{ rule: 'marketCap', label: 'Market Cap', max: "800", min:"10", step:"10", unit: 'Bn', category: 'fund'},
{ rule: 'revenue', label: 'Revenue', max: "800", min:"-100", step:"10", unit: 'Bn', category: 'fund' },
{ rule: 'growthRevenue', label: 'Revenue Growth [%]', max: "200", min:"-100", step:"2", unit: '%', category: 'fund' },
{ rule: 'costOfRevenue', label: 'Cost of Revenue', max: "800", min:"-100", step:"10", unit: 'Bn', category: 'fund' },
{ rule: 'growthCostOfRevenue', label: 'Cost of Revenue Growth [%]', max: "200", min:"-100", step:"2", unit: '%', category: 'fund' },
{ rule: 'costAndExpenses', label: 'Cost & Expenses', max: "800", min:"-100", step:"10", unit: 'Bn', category: 'fund' },
{ rule: 'growthCostAndExpenses', label: 'Cost & Expenses Growth [%]', max: "200", min:"-100", step:"2", unit: '%', category: 'fund' },
{ rule: 'netIncome', label: 'Net Income', max: "800", min:"-100", step:"10", unit: 'Bn', category: 'fund'},
{ rule: 'growthNetIncome', label: 'Net Income Growth [%]', max: "200", min:"-100", step:"2", unit: '%', category: 'fund'},
{ rule: 'grossProfit', label: 'Gross Profit', max: "800", min:"-100", step:"10", unit: 'Bn', category: 'fund'},
{ rule: 'growthGrossProfit', label: 'Gross Profit Growth [%]', max: "200", min:"-100", step:"2", unit: '%', category: 'fund'},
{ rule: 'researchAndDevelopmentExpenses', label: 'R&D Expenses', max: "100", min:"0", step:"1", unit: 'Bn', category: 'fund'},
{ rule: 'growthResearchAndDevelopmentExpenses', label: 'R&D Expenses Growth [%]', max: "200", min:"-100", step:"2", unit: '%', category: 'fund'},
{ rule: 'payoutRatio', label: 'Payout Ratio [%]', max: "100", min:"-100", step:"1", unit: '%', category: 'fund' },
{ rule: 'dividendYield', label: 'Dividend Yield [%]', max: "100", min:"0", step:"1", unit: '%',category: 'fund' },
{ rule: 'annualDividend', label: 'Annual Dividend', max: "20", min:"0", step:"0.5", category: 'fund' },
{ rule: 'dividendGrowth', label: 'Dividend Growth [%]', max: "100", min:"-100", step:"2", unit: '%',category: 'fund' },
{ rule: 'eps', label: 'EPS', max: "10", min:"-10", step:"0.1", category: 'fund' },
{ rule: 'growthEPS', label: 'EPS Growth [%]', max: "200", min:"-100", step:"2", unit: '%', category: 'fund' },
{ rule: 'interestIncome', label: 'Interest Income', max: "800", min:"-100", step:"10", unit: 'Bn', category: 'fund' },
{ rule: 'interestExpense', label: 'Interest Expenses', category: 'fund' },
{ rule: 'growthInterestExpense', label: 'Interest Expenses Growth [%]', max: "200", min:"-100", step:"2", unit: '%', category: 'fund' },
{ rule: 'operatingExpenses', label: 'Operating Expenses', max: "800", min:"-100", step:"10", unit: 'Bn', category: 'fund' },
{ rule: 'growthOperatingExpenses', label: 'Operating Expenses Growth [%]', max: "200", min:"-100", step:"2", unit: '%', category: 'fund' },
{ rule: 'operatingIncome', label: 'Operating Income', max: "800", min:"-100", step:"10", unit: 'Bn', category: 'fund' },
{ rule: 'growthOperatingIncome', label: 'Operating Income Growth [%]', max: "200", min:"-100", step:"2", unit: '%',category: 'fund' },
{ rule: 'pe', label: 'P/E', max: "50", min:"0", step:"0.5",category: 'fund'},
{ rule: 'forwardPE', label: 'Forward PE', max: "100", min:"-100", step:"2", category: 'fund'},
{ rule: 'priceToBookRatio', label: 'P/B', max: "50", min:"0", step:"0.5", category: 'fund'},
{ rule: 'priceToSalesRatio', label: 'P/S', max: "50", min:"0", step:"0.5", category: 'fund'},
{ rule: 'beta', label: 'Beta', max: "10", min:"-10", step:"0.1", category: 'fund'},
{ rule: 'ebitda', label: 'EBITDA', max: "800", min:"-100", step:"10", unit: 'Bn', category: 'fund'},
{ rule: 'growthEBITDA', label: 'EBITDA Growth [%]', max: "200", min:"-100", step:"2", unit: '%', category: 'fund'},
{ rule: 'var', label: 'VaR', max: "0", min:"-20", step:"1", unit: '%', category: 'fund' },
{ rule: 'trendAnalysis', label: 'AI Trend Analysis (Bullish)', max: "100", min:"0", step:"0.5", unit: '%', category: 'ai' },
{ rule: 'fundamentalAnalysis', label: 'AI Fundamental Analysis (Bullish)', max: "100", min:"0", step:"0.5", unit: '%', category: 'ai' },
//{ rule: 'ratingRecommendation', label: 'Analyst Rating', max: "2", min:"0", step:"1", category: 'fund'},
{ rule: 'currentRatio', label: 'Current Ratio', max: "50", min:"0", step:"0.5", category: 'fund' },
{ rule: 'quickRatio', label: 'Quick Ratio', max: "50", min:"0", step:"0.5", category: 'fund' },
{ rule: 'debtEquityRatio', label: 'Debt / Equity', max: "50", min:"0", step:"0.5",category: 'fund' },
{ rule: 'debtRatio', label: 'Debt Ratio', max: "50", min:"0", step:"0.5", category: 'fund' },
{ rule: 'returnOnAssets', label: 'Return on Assets', max: "5", min:"-5", step:"0.1", category: 'fund' },
{ rule: 'returnOnEquity', label: 'Return on Equity', max: "5", min:"-5", step:"0.1",category: 'fund' },
{ rule: 'enterpriseValue', label: 'Enterprise Value', max: "800", min:"10", step:"10", unit: 'Bn', category: 'fund' },
{ rule: 'freeCashFlowPerShare', label: 'FCF / Share', max: "20", min:"-20", step:"0.5",category: 'fund' },
{ rule: 'cashPerShare', label: 'Cash / Share', max: "50", min:"-50", step:"1", category: 'fund' },
{ rule: 'priceToFreeCashFlowsRatio', label: 'Price / FCF', max: "100", min:"-100", step:"2", category: 'fund' },
];
// Creating the ruleMappings object from allRows
const ruleMappings = allRows.reduce((acc, row) => {
acc[row.rule] = row.label;
return acc;
}, {});
allRows?.sort((a, b) => a.label.localeCompare(b.label));
let filteredRows;
let searchTerm = '';
/*
let taRows = allRows?.filter(row => row.category === 'ta');
let fundRows = allRows?.filter(row => row.category === 'fund');
taRows?.sort((a, b) => a.label.localeCompare(b.label));
fundRows?.sort((a, b) => a.label.localeCompare(b.label));
*/
function changeRuleCondition(state: string) {
ruleCondition[ruleName] = state;
}
let ruleName = '';
// Define your default values
let valueRevenue = (ruleOfList?.find(item => item.name === "revenue") || { value: 50 }).value;
let valueGrowthRevenue = (ruleOfList?.find(item => item.name === "growthRevenue") || { value: 10 }).value;
let valueCostOfRevenue = (ruleOfList?.find(item => item.name === "costOfRevenue") || { value: 30 }).value;
let valueGrowthCostOfRevenue = (ruleOfList?.find(item => item.name === "growthCostAndExpenses") || { value: 10 }).value;
let valueCostAndExpenses = (ruleOfList?.find(item => item.name === "costAndExpenses") || { value: 50 }).value;
let valueGrowthCostAndExpenses= (ruleOfList?.find(item => item.name === "growthCostAndExpenses") || { value: 10 }).value;
let valueResearchAndDevelopmentExpenses = (ruleOfList?.find(item => item.name === "researchAndDevelopmentExpenses") || { value: 10 }).value;
let valueGrowthResearchAndDevelopmentExpenses = (ruleOfList?.find(item => item.name === "growthResearchAndDevelopmentExpenses") || { value: 10 }).value;
let valueInterestIncome = (ruleOfList?.find(item => item.name === "interestIncome") || { value: 500 }).value;
let valueInterestExpenses = (ruleOfList?.find(item => item.name === "interestExpenses") || { value: 500 }).value;
let valueGrowthInterestExpenses = (ruleOfList?.find(item => item.name === "growthInterestExpenses") || { value: 10 }).value;
let valueEBITDA = (ruleOfList?.find(item => item.name === "ebitda") || { value: 50 }).value;
let valueGrowthEBITDA = (ruleOfList?.find(item => item.name === "growthEBITDA") || { value: 10 }).value;
let valueOperatingExpenses = (ruleOfList?.find(item => item.name === "operatingExpenses") || { value: 50 }).value;
let valueGrowthOperatingExpenses = (ruleOfList?.find(item => item.name === "growthOperatingExpenses") || { value: 10 }).value;
let valueOperatingIncome = (ruleOfList?.find(item => item.name === "operatingIncome") || { value: 50 }).value;
let valueGrowthOperatingIncome = (ruleOfList?.find(item => item.name === "growthOperatingIncome") || { value: 10 }).value;
let valueNetIncome = (ruleOfList?.find(item => item.name === "netIncome") || { value: 30 }).value;
let valueGrowthNetIncome = (ruleOfList?.find(item => item.name === "growthNetIncome") || { value: 10 }).value;
let valueGrossProfit = (ruleOfList?.find(item => item.name === "grossProfit") || { value: 50 }).value;
let valueGrowthGrossProfit = (ruleOfList?.find(item => item.name === "growthGrossProfit") || { value: 10 }).value;
let valueDividendYield = (ruleOfList?.find(item => item.name === "dividendYield") || { value: 20 }).value;
let valueAnnualDividend = (ruleOfList?.find(item => item.name === "annualDividend") || { value: 1 }).value;
let valueDividendGrowth = (ruleOfList?.find(item => item.name === "dividendGrowth") || { value: 5 }).value;
let valuePayoutRatio = (ruleOfList?.find(item => item.name === "payoutRatio") || { value: 20 }).value;
let valueCurrentRatio = (ruleOfList?.find(item => item.name === "currentRatio") || { value: 1 }).value;
let valueQuickRatio = (ruleOfList?.find(item => item.name === "quickRatio") || { value: 1 }).value;
let valueDebtEquityRatio = (ruleOfList?.find(item => item.name === "debtEquityRatio") || { value: 1 }).value;
let valueDebtRatio = (ruleOfList?.find(item => item.name === "debtRatio") || { value: 1 }).value;
let valueReturnOnAssets = (ruleOfList?.find(item => item.name === "returnOnAssets") || { value: 0 }).value;
let valueReturnOnEquity = (ruleOfList?.find(item => item.name === "returnOnEquity") || { value: 0 }).value;
let valueEPS = (ruleOfList?.find(item => item.name === "eps") || { value: 2 }).value;
let valueGrowthEPS = (ruleOfList?.find(item => item.name === "growthEPS") || { value: 10 }).value;
let valuePE = (ruleOfList?.find(item => item.name === "pe") || { value: 10 }).value;
let valueForwardPE = (ruleOfList?.find(item => item.name === "forwardPE") || { value: 10 }).value;
let valuePriceToBookRatio = (ruleOfList?.find(item => item.name === "priceToBookRatio") || { value: 5 }).value;
let valuePriceToSalesRatio = (ruleOfList?.find(item => item.name === "priceToSalesRatio") || { value: 10 }).value;
let valueBeta = (ruleOfList?.find(item => item.name === "beta") || { value: 1 }).value;
let valueMarketCap = (ruleOfList?.find(item => item.name === "marketCap") || { value: 50 }).value;
let valueAnalyst = (ruleOfList?.find(item => item.name === "ratingRecommendation") || { value: 1 }).value;
let valueRSI = (ruleOfList?.find(item => item.name === "rsi") || { value: 40 }).value;
let valueStochRSI = (ruleOfList?.find(item => item.name === "stochRSI") || { value: 40 }).value;
let valueMFI = (ruleOfList?.find(item => item.name === "mfi") || { value: 40 }).value;
let valueCCI = (ruleOfList?.find(item => item.name === "cci") || { value: 0 }).value;
let valueATR = (ruleOfList?.find(item => item.name === "atr") || { value: 2 }).value;
let valueSMA50 = (ruleOfList?.find(item => item.name === "sma50") || { value: 50 }).value;
let valueSMA200 = (ruleOfList?.find(item => item.name === "sma200") || { value: 100 }).value;
let valueEMA50 = (ruleOfList?.find(item => item.name === "ema50") || { value: 50 }).value;
let valueEMA200 = (ruleOfList?.find(item => item.name === "ema200") || { value: 100 }).value;
let valueChange1W = (ruleOfList?.find(item => item.name === "change1W") || { value: 0 }).value;
let valueChange1M = (ruleOfList?.find(item => item.name === "change1M") || { value: 0 }).value;
let valueChange3M = (ruleOfList?.find(item => item.name === "change3M") || { value: 0 }).value;
let valueChange6M = (ruleOfList?.find(item => item.name === "change6M") || { value: 0 }).value;
let valueChange1Y = (ruleOfList?.find(item => item.name === "change1Y") || { value: 0 }).value;
let valueChange3Y = (ruleOfList?.find(item => item.name === "change3Y") || { value: 0 }).value;
let valueAvgVolume = (ruleOfList?.find(item => item.name === "avgVolume") || { value: 10 }).value;
let valueVaR = (ruleOfList?.find(item => item.name === "var") || { value: -10 }).value;
let valueTrendAnalysis = (ruleOfList?.find(item => item.name === "trendAnalysis") || { value: 50 }).value;
let valueFundamentalAnalysis = (ruleOfList?.find(item => item.name === "fundamentalAnalysis") || { value: 50 }).value;
let valueEnterpriseValue = (ruleOfList?.find(item => item.name === "enterpriseValue") || { value: 50 }).value;
let valueFCFShare = (ruleOfList?.find(item => item.name === "freeCashFlowPerShare") || { value: 1 }).value;
let valueCashShare = (ruleOfList?.find(item => item.name === "cashPerShare") || { value: 1 }).value;
let valuePriceFCF = (ruleOfList?.find(item => item.name === "priceToFreeCashFlowsRatio") || { value: 1 }).value;
const ratingRecommendations = [
'Sell',
'Hold',
'Buy',
];
$: {
if (ruleTrend['ratingRecommendation'])
{
ruleTrend['ratingRecommendation'] = ratingRecommendations[valueAnalyst];
}
}
function changeRule(state: string)
{
searchTerm = '';
ruleName = state;
handleAddRule()
//const closePopup = document.getElementById("ruleModal");
//closePopup?.dispatchEvent(new MouseEvent('click'))
}
const valueMappings = {
revenue: valueRevenue,
growthRevenue: valueGrowthRevenue,
costOfRevenue: valueCostOfRevenue,
growthCostOfRevenue: valueGrowthCostOfRevenue,
costAndExpenses: valueCostAndExpenses,
growthCostAndExpenses: valueGrowthCostAndExpenses,
netIncome: valueNetIncome,
growthNetIncome: valueGrowthNetIncome,
grossProfit: valueGrossProfit,
growthGrossProfit: valueGrowthGrossProfit,
researchAndDevelopmentExpenses: valueResearchAndDevelopmentExpenses,
growthResearchAndDevelopmentExpenses: valueGrowthResearchAndDevelopmentExpenses,
eps: valueEPS,
dividendYield: valueDividendYield,
annualDividend: valueAnnualDividend,
dividendGrowth: valueDividendGrowth,
payoutRatio: valuePayoutRatio,
growthEPS: valueGrowthEPS,
interestIncome: valueInterestIncome,
interestExpense: valueInterestExpenses,
growthInterestExpense: valueGrowthInterestExpenses,
operatingExpenses: valueOperatingExpenses,
growthOperatingExpenses: valueGrowthOperatingExpenses,
operatingIncome: valueOperatingIncome,
growthOperatingIncome: valueGrowthOperatingIncome,
pe: valuePE,
forwardPE: valueForwardPE,
priceToBookRatio: valuePriceToBookRatio,
priceToSalesRatio: valuePriceToSalesRatio,
beta: valueBeta,
ebitda: valueEBITDA,
growthEBITDA: valueGrowthEBITDA,
marketCap: valueMarketCap,
rsi: valueRSI,
stochRSI: valueStochRSI,
mfi: valueMFI,
cci: valueCCI,
atr: valueATR,
sma50: valueSMA50,
sma200: valueSMA200,
ema50: valueEMA50,
ema200: valueEMA200,
change1W: valueChange1W,
change1M: valueChange1M,
change3M: valueChange3M,
change6M: valueChange6M,
change1Y: valueChange1Y,
change3Y: valueChange3Y,
avgVolume: valueAvgVolume,
var: valueVaR,
trendAnalysis: valueTrendAnalysis,
fundamentalAnalysis: valueFundamentalAnalysis,
currentRatio: valueCurrentRatio,
quickRatio: valueQuickRatio,
debtEquityRatio: valueDebtEquityRatio,
debtRatio: valueDebtRatio,
returnOnAssets: valueReturnOnAssets,
returnOnEquity: valueReturnOnEquity,
enterpriseValue: valueEnterpriseValue,
freeCashFlowPerShare: valueFCFShare,
cashPerShare: valueCashShare,
priceToFreeCashFlowsRatio: valuePriceFCF,
};
const conditions = {
revenue: ruleCondition.revenue,
growthRevenue: ruleCondition.growthRevenue,
costOfRevenue: ruleCondition.costOfRevenue,
growthCostOfRevenue: ruleCondition.growthCostOfRevenue,
costAndExpenses: ruleCondition.costAndExpenses,
growthCostAndExpenses: ruleCondition.growthCostAndExpenses,
netIncome: ruleCondition.netIncome,
growthNetIncome: ruleCondition.growthNetIncome,
grossProfit: ruleCondition.grossProfit,
growthGrossProfit: ruleCondition.growthGrossProfit,
researchAndDevelopmentExpenses: ruleCondition.researchAndDevelopmentExpenses,
growthResearchAndDevelopmentExpenses: ruleCondition.growthResearchAndDevelopmentExpenses,
eps: ruleCondition.eps,
growthEPS: ruleCondition.growthEPS,
interestIncome: ruleCondition.interestIncome,
interestExpense: ruleCondition.interestExpense,
growthInterestExpense: ruleCondition.growthInterestExpense,
operatingExpenses: ruleCondition.operatingExpenses,
growthOperatingExpenses: ruleCondition.growthOperatingExpenses,
operatingIncome: ruleCondition.operatingIncome,
growthOperatingIncome: ruleCondition.growthOperatingIncome,
pe: ruleCondition.pe,
forwardPE: ruleCondition.forwardPE,
priceToBookRatio: ruleCondition.priceToBookRatio,
priceToSalesRatio: ruleCondition.priceToSalesRatio,
beta: ruleCondition.beta,
ebitda: ruleCondition.ebitda,
growthEBITDA: ruleCondition.growthEBITDA,
marketCap: ruleCondition.marketCap,
rsi: ruleCondition.rsi,
stochRSI: ruleCondition.stochRSI,
mfi: ruleCondition.mfi,
cci: ruleCondition.cci,
atr: ruleCondition.atr,
sma50: ruleCondition.sma50,
sma200: ruleCondition.sma200,
ema50: ruleCondition.ema50,
ema200: ruleCondition.ema200,
change1W: ruleCondition.change1W,
change1M: ruleCondition.change1M,
change3M: ruleCondition.change3M,
change6M: ruleCondition.change6M,
change1Y: ruleCondition.change1Y,
change3Y: ruleCondition.change3Y,
avgVolume: ruleCondition.avgVolume,
var: ruleCondition.var,
trendAnalysis: ruleCondition.trendAnalysis,
fundamentalAnalysis: ruleCondition.fundamentalAnalysis,
dividendGrowth: ruleCondition.dividendGrowth,
dividendYield: ruleCondition.dividendYield,
payoutRatio: ruleCondition.payoutRatio,
annualDividend: ruleCondition.annualDividend,
currentRatio: ruleCondition.currentRatio,
quickRatio: ruleCondition.quickRatio,
debtEquityRatio: ruleCondition.debtEquityRatio,
debtRatio: ruleCondition.debtRatio,
returnOnAssets: ruleCondition.returnOnAssets,
returnOnEquity: ruleCondition.returnOnEquity,
enterpriseValue: ruleCondition.enterpriseValue,
freeCashFlowPerShare: ruleCondition.freeCashFlowPerShare,
cashPerShare: ruleCondition.cashPerShare,
priceToFreeCashFlowsRatio: ruleCondition.priceToFreeCashFlowsRatio,
};
function handleAddRule() {
if (ruleName === '') {
toast.error('Please select a rule', {
style: 'border-radius: 200px; background: #333; color: #fff;'
});
return;
}
let newRule;
switch (ruleName) {
case 'ratingRecommendation':
newRule = { name: ruleName, value: valueAnalyst }; //ruleTrend[ruleName]
handleRule(newRule);
break;
default:
// Handle other cases if needed
newRule = { name: ruleName, condition: conditions[ruleName], value: valueMappings[ruleName] };
handleRule(newRule);
break;
}
async function handleRule(newRule) {
const existingRuleIndex = ruleOfList.findIndex(rule => rule.name === ruleName);
if (existingRuleIndex !== -1) {
const existingRule = ruleOfList[existingRuleIndex];
if (existingRule.value === newRule.value && existingRule.condition === newRule.condition) {
toast.error('Rule already exists!', {
style: 'border-radius: 200px; background: #333; color: #fff;'
});
} else {
ruleOfList[existingRuleIndex] = newRule;
ruleOfList = [...ruleOfList]; // Trigger reactivity
toast.success('Rule updated', {
style: 'border-radius: 200px; background: #333; color: #fff;'
});
await updateStockScreenerData();
}
} else {
ruleOfList = [...ruleOfList, newRule];
toast.success('Rule added', {
style: 'border-radius: 200px; background: #333; color: #fff;'
});
await updateStockScreenerData();
await handleSave(false);
}
}
async function updateStockScreenerData() {
try {
const newData = await getStockScreenerData(ruleOfList);
stockScreenerData = newData?.filter(item =>
Object.values(item).every(value =>
value !== null && value !== undefined &&
(typeof value !== 'object' || Object.values(value).every(subValue => subValue !== null && subValue !== undefined))
)
);
filteredData = filterStockScreenerData();
displayResults = filteredData?.slice(0, 50);
} catch (error) {
console.error('Error fetching new stock screener data:', error);
toast.error('Failed to update stock data. Please try again.', {
style: 'border-radius: 200px; background: #333; color: #fff;'
});
}
}
}
async function handleResetAll() {
ruleOfList = [];
ruleOfList = [...ruleOfList];
ruleName = '';
filteredData = [];
displayResults = [];
await handleSave(false);
}
async function handleDeleteRule(state) {
for (let i = 0; i < ruleOfList.length; i++) {
if (ruleOfList[i].name === state) {
ruleOfList.splice(i, 1); // Remove the element at index i from the ruleOfList
ruleOfList = [...ruleOfList]
break; // Exit the loop after deleting the element
}
}
if(ruleOfList?.length === 0)
{
ruleName = '';
filteredData = [];
displayResults = [];
}
else if (state === ruleName)
{
ruleName = '';
}
await handleSave(false);
}
async function handleScroll() {
const scrollThreshold = document.body.offsetHeight * 0.8; // 80% of the website height
const isBottom = window.innerHeight + window.scrollY >= scrollThreshold;
if (isBottom && displayResults?.length !== filteredData?.length) {
const nextIndex = displayResults?.length;
const filteredNewResults = filteredData?.slice(nextIndex, nextIndex + 30);
displayResults = [...displayResults, ...filteredNewResults];
}
}
/*
const handleKeyDown = (event) => {
if (event.ctrlKey && event.key === 's') {
event.preventDefault(); // prevent the browser's default save action
handleSave();
}
};
*/
let LoginPopup;
onMount(async () => {
if(!data?.user) {
LoginPopup = (await import('$lib/components/LoginPopup.svelte')).default;
}
});
async function handleSave(printToast) {
if(data?.user)
{
if(isSaved === false)
{
const postData = {'strategyId': $strategyId, 'rules': ruleOfList}
const response = await fetch(data?.fastifyURL+'/save-strategy', {
method: 'POST',
headers: {
"Content-Type": "application/json"
},
body: JSON.stringify(postData)
});
const output = (await response.json())?.items
if (printToast === true) {
if (output?.id && output?.id?.length !== 0) {
toast.success('Strategy saved!', {
style: 'border-radius: 200px; background: #333; color: #fff;'});
} else {
toast.error('Something went wrong. Please try again later!', {
style: 'border-radius: 200px; background: #333; color: #fff;'});
}
}
isSaved = true;
}
}
}
async function handleUpdateRule(rule) {
ruleName = rule.name;
}
$: {
if (ruleOfList) {
const ruleToUpdate = ruleOfList.find(rule => rule.name === ruleName);
if (ruleToUpdate) {
const valueMap = {
payoutRatio: valuePayoutRatio,
dividendGrowth: valueDividendGrowth,
dividendYield: valueDividendYield,
annualDividend: valueAnnualDividend,
eps: valueEPS,
growthEPS: valueGrowthEPS,
marketCap: valueMarketCap,
beta: valueBeta,
pe: valuePE,
forwardPE: valueForwardPE,
priceToBookRatio: valuePriceToBookRatio,
priceToSalesRatio: valuePriceToSalesRatio,
interestIncome: valueInterestIncome,
ratingRecommendation: valueAnalyst,
revenue: valueRevenue,
growthRevenue: valueGrowthRevenue,
ebitda: valueEBITDA,
growthEBITDA: valueGrowthEBITDA,
operatingExpenses: valueOperatingExpenses,
growthOperatingExpenses: valueGrowthOperatingExpenses,
costOfRevenue: valueCostOfRevenue,
growthCostOfRevenue: valueGrowthCostOfRevenue,
costAndExpenses: valueCostAndExpenses,
growthCostAndExpenses: valueGrowthCostAndExpenses,
netIncome: valueNetIncome,
growthNetIncome: valueGrowthNetIncome,
grossProfit: valueGrossProfit,
growthGrossProfit: valueGrowthGrossProfit,
researchAndDevelopmentExpenses: valueResearchAndDevelopmentExpenses,
growthResearchAndDevelopmentExpenses: valueGrowthResearchAndDevelopmentExpenses,
interestExpense: valueInterestExpenses,
growthInterestExpense: valueGrowthInterestExpenses,
operatingIncome: valueOperatingIncome,
growthOperatingIncome: valueGrowthOperatingIncome,
rsi: valueRSI,
stochRSI: valueStochRSI,
mfi: valueMFI,
cci: valueCCI,
atr: valueATR,
sma50: valueSMA50,
sma200: valueSMA200,
ema50: valueEMA50,
ema200: valueEMA200,
change1W: valueChange1W,
change1M: valueChange1M,
change3M: valueChange3M,
change6M: valueChange6M,
change1Y: valueChange1Y,
change3Y: valueChange3Y,
avgVolume: valueAvgVolume,
var: valueVaR,
trendAnalysis: valueTrendAnalysis,
fundamentalAnalysis: valueFundamentalAnalysis,
currentRatio: valueCurrentRatio,
quickRatio: valueQuickRatio,
debtEquityRatio: valueDebtEquityRatio,
debtRatio: valueDebtRatio,
returnOnAssets: valueReturnOnAssets,
returnOnEquity: valueReturnOnEquity,
enterpriseValue: valueEnterpriseValue,
freeCashFlowPerShare: valueFCFShare,
cashPerShare: valueCashShare,
priceToFreeCashFlowsRatio: valuePriceFCF,
};
ruleToUpdate.value = valueMap[ruleToUpdate.name] ?? ruleToUpdate.value;
ruleToUpdate.condition = ruleCondition[ruleToUpdate.name];
ruleOfList = [...ruleOfList];
}
filteredData = filterStockScreenerData();
if (ruleOfList?.some(item => item?.name === 'ratingRecommendation')) {
filteredData = filteredData?.filter(item => item?.ratingRecommendation === valueAnalyst);
}
}
}
function filterStockScreenerData() {
return stockScreenerData?.filter(item => {
for (const rule of ruleOfList) {
if ([
'researchAndDevelopmentExpenses',
'operatingIncome',
'operatingExpenses',
'netIncome',
'revenue',
'marketCap',
'enterpriseValue',
'costAndExpenses',
'costOfRevenue',
'ebitda',
'grossProfit'
]?.includes(rule.name))
{
if (rule.condition === "over" && item[rule.name] !== null && item[rule.name] <= rule.value * 10**(9)) {
return false;
}
else if (rule.condition === "under" && item[rule.name] !== null && item[rule.name] > rule.value * 10**(9)) {
return false;
}
}
else if (rule.name === 'interestIncome' || rule.name === 'interestExpense') {
if (rule.condition === "over" && item[rule.name] !== null && item[rule.name] <= rule.value * 10**(6)) {
return false;
}
else if (rule.condition === "under" && item[rule.name] !== null && item[rule.name] > rule.value * 10**(6)) {
return false;
}
}
else if (rule.name === 'avgVolume') {
if (rule.condition === "over" && item[rule.name] <= rule.value * 10**(6)) {
return false;
}
else if (rule.condition === "under" && item[rule.name] > rule.value * 10**(6)) {
return false;
}
}
else if (rule.name === 'trendAnalysis') {
if (rule.condition === "over" && item[rule.name]?.accuracy <= rule.value ) {
return false;
}
else if (rule.condition === "under" && item[rule.name]?.accuracy > rule.value ) {
return false;
}
}
else if (rule.name === 'fundamentalAnalysis') {
if (rule.condition === "over" && item[rule.name]?.accuracy <= rule.value ) {
return false;
}
else if (rule.condition === "under" && item[rule.name]?.accuracy > rule.value ) {
return false;
}
}
else {
if (rule.condition === "over" && item[rule.name] !== null && item[rule.name] <= rule.value) {
return false;
}
else if (rule.condition === "under" && item[rule.name] !== null && item[rule.name] > rule.value) {
return false;
}
}
}
return true;
});
}
let order = 'highToLow';
let sortBy = ''; // Default sorting by change percentage
function changeOrder(state:string) {
if (state === 'highToLow')
{
order = 'lowToHigh';
}
else {
order = 'highToLow';
}
}
const sortByHighestChange = (tickerList) => {
return tickerList?.sort(function(a, b) {
if(order === 'highToLow')
{
return b?.changesPercentage - a?.changesPercentage;
}
else {
return a?.changesPercentage - b?.changesPercentage;
}
});
}
const sortByMarketCap = (tickerList) => {
return tickerList?.sort(function(a, b) {
if(order === 'highToLow')
{
return b?.marketCap - a?.marketCap;
}
else {
return a?.marketCap - b?.marketCap;
}
});
}
$: {
if(order)
{
if(sortBy === 'change')
{
displayResults = sortByHighestChange(filteredData)?.slice(0,50);
}
else if(sortBy === 'marketCap')
{
displayResults = sortByMarketCap(filteredData)?.slice(0,50);
}
}
}
$: {
if(searchTerm)
{
filteredRows = allRows?.filter((row) => row?.label?.toLowerCase()?.includes(searchTerm?.toLowerCase()));
}
}
$: displayResults = filteredData?.slice(0, 50);
$: isSaved = !ruleOfList;
$: charNumber = $screenWidth < 640 ? 20 : 40;
function handleChangeCondition(event) {
ruleCondition[event.detail.rule] = event.detail.condition;
}
function handleChangeValue(event) {
const { rule, value } = event.detail;
if (rule in valueMappings) {
valueMappings[rule] = value;
// If you need to keep the separate variables in sync:
switch (rule) {
case 'payoutRatio':
valuePayoutRatio = value;
break;
case 'dividendGrowth':
valueDividendGrowth = value;
break;
case 'dividendYield':
valueDividendYield = value;
break;
case 'annualDividend':
valueAnnualDividend = value;
break;
case 'eps':
valueEPS = value;
break;
case 'growthEPS':
valueGrowthEPS = value;
break;
case 'marketCap':
valueMarketCap = value;
break;
case 'beta':
valueBeta = value;
break;
case 'pe':
valuePE = value;
break;
case 'forwardPE':
valueForwardPE = value;
break;
case 'priceToBookRatio':
valuePriceToBookRatio = value;
break;
case 'priceToSalesRatio':
valuePriceToSalesRatio = value;
break;
case 'interestIncome':
valueInterestIncome = value;
break;
case 'ratingRecommendation':
valueAnalyst = value;
break;
case 'revenue':
valueRevenue = value;
break;
case 'growthRevenue':
valueGrowthRevenue = value;
break;
case 'ebitda':
valueEBITDA = value;
break;
case 'growthEBITDA':
valueGrowthEBITDA = value;
break;
case 'operatingExpenses':
valueOperatingExpenses = value;
break;
case 'growthOperatingExpenses':
valueGrowthOperatingExpenses = value;
break;
case 'costOfRevenue':
valueCostOfRevenue = value;
break;
case 'growthCostOfRevenue':
valueGrowthCostOfRevenue = value;
break;
case 'costAndExpenses':
valueCostAndExpenses = value;
break;
case 'growthCostAndExpenses':
valueGrowthCostAndExpenses = value;
break;
case 'netIncome':
valueNetIncome = value;
break;
case 'growthNetIncome':
valueGrowthNetIncome = value;
break;
case 'grossProfit':
valueGrossProfit = value;
break;
case 'growthGrossProfit':
valueGrowthGrossProfit = value;
break;
case 'researchAndDevelopmentExpenses':
valueResearchAndDevelopmentExpenses = value;
break;
case 'growthResearchAndDevelopmentExpenses':
valueGrowthResearchAndDevelopmentExpenses = value;
break;
case 'interestExpense':
valueInterestExpenses = value;
break;
case 'growthInterestExpense':
valueGrowthInterestExpenses = value;
break;
case 'operatingIncome':
valueOperatingIncome = value;
break;
case 'growthOperatingIncome':
valueGrowthOperatingIncome = value;
break;
case 'rsi':
valueRSI = value;
break;
case 'stochRSI':
valueStochRSI = value;
break;
case 'mfi':
valueMFI = value;
break;
case 'cci':
valueCCI = value;
break;
case 'atr':
valueATR = value;
break;
case 'sma50':
valueSMA50 = value;
break;
case 'sma200':
valueSMA200 = value;
break;
case 'ema50':
valueEMA50 = value;
break;
case 'ema200':
valueEMA200 = value;
break;
case 'change1W':
valueChange1W = value;
break;
case 'change1M':
valueChange1M = value;
break;
case 'change3M':
valueChange3M = value;
break;
case 'change6M':
valueChange6M = value;
break;
case 'change1Y':
valueChange1Y = value;
break;
case 'change3Y':
valueChange3Y = value;
case 'avgVolume':
valueAvgVolume = value;
break;
case 'var':
valueVaR = value;
break;
case 'trendAnalysis':
valueTrendAnalysis = value;
break;
case 'fundamentalAnalysis':
valueFundamentalAnalysis = value;
break;
case 'currentRatio':
valueCurrentRatio = value;
break;
case 'quickRatio':
valueQuickRatio = value;
break;
case 'debtEquityRatio':
valueDebtEquityRatio = value;
break;
case 'debtRatio':
valueDebtRatio = value;
break;
case 'returnOnAssets':
valueReturnOnAssets = value;
break;
case 'returnOnEquity':
valueReturnOnEquity = value;
break;
case 'freeCashFlowPerShare':
valueFCFShare = value;
break;
case 'cashPerShare':
valueCashShare = value;
break;
case 'priceToFreeCashFlowsRatio':
valuePriceFCF = value;
case 'enterpriseValue':
valueEnterpriseValue = value;
}
} else {
console.warn(`Unhandled rule: ${rule}`);
// Optionally handle unknown rules here
}
}
</script>
<svelte:head>
<meta charset="utf-8" />
<meta name="viewport" content="width=device-width, initial-scale=1.0, maximum-scale=1.0"/>
<title>
{$numberOfUnreadNotification > 0 ? `(${$numberOfUnreadNotification})` : ''} Stock Screener · stocknear
</title>
<meta name="description" content={`Build your Stock Screener to find profitable stocks.`}>
<!-- Other meta tags -->
<meta property="og:title" content={`Stock Screener · stocknear`}/>
<meta property="og:description" content={`Build your Stock Screener to find profitable stocks.`} />
<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={`Stock Screener · stocknear`}/>
<meta name="twitter:description" content={`Build your Stock Screener to find profitable stocks.`} />
<!-- Add more Twitter meta tags as needed -->
</svelte:head>
<svelte:window on:scroll={handleScroll} />
<section class="w-full max-w-3xl sm:max-w-screen-xl overflow-hidden min-h-screen pt-5 pb-40 px-5">
<div class="text-sm sm:text-[1rem] breadcrumbs">
<ul>
<li><a href="/" class="text-gray-300">Home</a></li>
<li><a href="/stock-screener" class="text-gray-300">Stock Screener</a></li>
<li class="text-gray-300">{title ?? 'Demo'}</li>
</ul>
</div>
<!--Start Build Strategy-->
<div class="mt-5 sm:rounded-lg">
<div class="flex flex-row items-center mb-5">
<h1 class="text-white text-3xl font-semibold">
Stock Screener
</h1>
<span class="text-sm font-semibold text-white ml-2 mt-3">
{ruleOfList?.length !== 0 ? filteredData?.length : 0} Matches Found
</span>
</div>
<div class="rounded-lg border border-gray-700 bg-[#262626] p-2">
<div class="items-end border-b border-gray-400">
<div class="mr-1 flex items-center justify-between lg:mr-2 mb-1.5">
<button class="flex cursor-pointer items-center text-lg sm:text-xl font-semibold text-gray-200" title="Hide Filter Area">
<svg class="-mb-0.5 h-6 w-6" viewBox="0 0 20 20" fill="currentColor" style="max-width:40px">
<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>
Filters
</button>
</div>
</div>
<div class="mt-3 flex flex-col gap-y-2.5 sm:flex-row lg:gap-y-2 pb-1">
<button class="inline-flex items-center justify-center space-x-1 whitespace-nowrap rounded-md border border-transparent bg-blue-brand_light py-2 pl-3 pr-4 text-base font-semibold text-white shadow-sm bg-[#000] focus:outline-none focus:ring-2 focus:ring-blue-500 sm:text-smaller">
<svg class="h-5 w-5" viewBox="0 0 20 20" fill="currentColor" style="max-width:40px" aria-hidden="true">
<path fill-rule="evenodd" d="M10 3a1 1 0 011 1v5h5a1 1 0 110 2h-5v5a1 1 0 11-2 0v-5H4a1 1 0 110-2h5V4a1 1 0 011-1z" clip-rule="evenodd"></path>
</svg>
<div>Add Filters</div>
</button>
<div class="relative sm:ml-2">
<div class="absolute inset-y-0 left-0 flex items-center pl-2.5">
<svg class="h-4 w-4 text-gray-400 xs:h-5 xs:w-5" fill="none" stroke-linecap="round" stroke-linejoin="round" stroke-width="3" stroke="currentColor" viewBox="0 0 24 24" style="max-width: 40px" aria-hidden="true">
<path d="M21 21l-6-6m2-5a7 7 0 11-14 0 7 7 0 0114 0z"></path>
</svg>
</div>
<input type="text" placeholder="Search {allRows?.length} filters..." class="controls-input rounded-lg w-full py-2 pl-10 placeholder:text-gray-300 bg-[#313131] sm:w-72">
<div class="absolute inset-y-0 right-0 flex items-center pr-2"></div>
</div>
</div>
<div class="sm:grid sm:grid-cols-2 sm:gap-x-2.5 lg:grid-cols-3 w-full mt-5 border-b border-gray-400">
{#each allRows as row (row.rule)}
{#if ruleOfList.some(rule => rule.name === row.rule)}
<RuleControl
ruleName={row.rule}
title={row.label}
min={row.min}
max={row.max}
step={row.step}
unit={row?.unit}
bind:value={valueMappings[row.rule]}
bind:condition={conditions[row.rule]}
on:changeCondition={handleChangeCondition}
on:changeValue={handleChangeValue}
/>
{/if}
{/each}
</div>
</div>
<!--End Adding Rules-->
<!--Start Rules Preview -->
<!--
<div id="step-3" class="m-auto w-5/6 bg-[#09090B] sm:ml-10 h-auto max-h-[400px] no-scrollbar overflow-hidden overflow-y-scroll p-5 sm:rounded-lg border-b sm:border sm:hover:border-slate-700 border-slate-800 pb-10">
<div class="flex flex-row items-center pb-5 sm:pb-0">
<div class="text-white font-bold text-xl sm:text-2xl flex justify-start items-center">
{ruleOfList.length} Rules Preview
</div>
<label on:click={handleResetAll} class="ml-auto cursor-pointer transition duration-100 bg-purple-600 sm:hover:bg-purple-700 border border-slate-800 py-2 px-3 rounded-lg text-white text-sm">
Reset All
</label>
</div>
{#if ruleOfList.length === 0}
<div class="text-slate-300 font-medium text-sm sm:text-md flex flex-row justify-start items-center mt-4">
<svg class="w-3 h-3 sm:w-4 sm:h-4 inline-block mr-2" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 20 20"><path fill="#ffcc4d" d="M19.59 15.86L12.007 1.924C11.515 1.011 10.779.5 9.989.5c-.79 0-1.515.521-2.016 1.434L.409 15.861c-.49.901-.544 1.825-.138 2.53c.405.707 1.216 1.109 2.219 1.109h15.02c1.003 0 1.814-.402 2.22-1.108c.405-.706.351-1.619-.14-2.531ZM10 4.857c.395 0 .715.326.715.728v6.583c0 .402-.32.728-.715.728a.721.721 0 0 1-.715-.728V5.584c0-.391.32-.728.715-.728Zm0 11.624c-.619 0-1.11-.51-1.11-1.14c0-.63.502-1.141 1.11-1.141c.619 0 1.11.51 1.11 1.14c0 .63-.502 1.141-1.11 1.141Z"/></svg>
At least 1 rule is required
</div>
{:else}
{#each ruleOfList as rule}
<div class="flex flex-row mt-4">
<label on:click={() => handleUpdateRule(rule)} class=" cursor-pointer text-slate-300 sm:font-medium text-sm sm:text-md">
<svg class="flex-shrink-0 w-5 h-5 sm:w-6 sm:h-6 text-green-400 inline-block" fill="currentColor" viewBox="0 0 20 20" xmlns="http://www.w3.org/2000/svg"><path fill-rule="evenodd" d="M16.707 5.293a1 1 0 010 1.414l-8 8a1 1 0 01-1.414 0l-4-4a1 1 0 011.414-1.414L8 12.586l7.293-7.293a1 1 0 011.414 0z" clip-rule="evenodd"></path></svg>
{ruleMappings[rule.name] || rule.name} · {formatRuleValue(rule)}
</label>
<label on:click={() => handleDeleteRule(rule.name)} class="text-sm text-[#FF3131] cursor-pointer ml-auto sm:ml-0">
<svg class="h-6 w-6 ml-2" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path fill="currentColor" d="M7 21q-.825 0-1.413-.588T5 19V6q-.425 0-.713-.288T4 5q0-.425.288-.713T5 4h4q0-.425.288-.713T10 3h4q.425 0 .713.288T15 4h4q.425 0 .713.288T20 5q0 .425-.288.713T19 6v13q0 .825-.588 1.413T17 21H7ZM7 6v13h10V6H7Zm2 10q0 .425.288.713T10 17q.425 0 .713-.288T11 16V9q0-.425-.288-.713T10 8q-.425 0-.713.288T9 9v7Zm4 0q0 .425.288.713T14 17q.425 0 .713-.288T15 16V9q0-.425-.288-.713T14 8q-.425 0-.713.288T13 9v7ZM7 6v13V6Z"/></svg>
</label>
</div>
{/each}
{/if}
</div>
-->
<!--End Rules Preview-->
</div>
<!--End Build Strategy-->
<!--Start Running Mode Preview-->
<div class="bg-[#09090B] sm:border border-slate-800 sm:hover:border-slate-700 mt-5 sm:rounded-2xl ">
<!--Start Number of Matches-->
<div class="text-slate-300 font-bold text-xl flex justify-center sm:justify-start items-center ml-3 sm:ml-5 pt-5 pb-5">
{ruleOfList?.length !== 0 ? filteredData?.length : 0} Matches Found
<label for="modeInfo" class="cursor-pointer">
<!--<svg class="w-6 h-6 inline-block ml-2" viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg" fill="#09090B000"><g id="SVGRepo_bgCarrier" stroke-width="0"></g><g id="SVGRepo_tracerCarrier" stroke-linecap="round" stroke-linejoin="round"></g><g id="SVGRepo_iconCarrier"> <title></title> <g id="Complete"> <g id="info-circle"> <g> <circle cx="12" cy="12" data-name="--Circle" fill="none" id="_--Circle" r="10" stroke="#B46266" stroke-linecap="round" stroke-linejoin="round" stroke-width="2"></circle> <line fill="none" stroke="#B46266" stroke-linecap="round" stroke-linejoin="round" stroke-width="2" x1="12" x2="12" y1="12" y2="16"></line> <line fill="none" stroke="#B46266" stroke-linecap="round" stroke-linejoin="round" stroke-width="2" x1="12" x2="12" y1="8" y2="8"></line> </g> </g> </g> </g></svg>-->
</label>
</div>
<!--End Number of Matches-->
<!--Start Matching Preview-->
{#if displayResults?.length !== 0 && ruleOfList?.length !== 0}
<div class="w-full rounded-lg overflow-x-scroll p-0 sm:p-3">
<table class="table table-sm table-compact w-full bg-[#09090B] border-bg-[#09090B]">
<thead>
<tr class="border-b-[#1A1A27]">
<th class="text-white bg-[#09090B] text-sm sm:text-[1rem] font-semibold border-b-[#09090B]">Symbol</th>
<th class="text-white hidden sm:table-cell bg-[#09090B] text-sm sm:text-[1rem] font-semibold border-b-[#09090B]">Company Name</th>
<th on:click={() => { sortBy = 'marketCap'; changeOrder(order); }} class="whitespace-nowrap cursor-pointer text-white font-semibold text-sm sm:text-[1rem] font-semibold text-center">
Market Cap
<svg class="w-5 h-5 inline-block {order === 'highToLow' && sortBy === 'marketCap' ? 'rotate-180' : ''}" viewBox="0 0 20 20" fill="currentColor" style="max-width:40px"><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>
</th>
<th on:click={() => { sortBy = 'change'; changeOrder(order); }} class="whitespace-nowrap cursor-pointer text-white font-semibold text-sm sm:text-[1rem] font-semibold text-center">
% Change
<svg class="w-5 h-5 inline-block {order === 'highToLow' && sortBy === 'change' ? '' : 'rotate-180'}" viewBox="0 0 20 20" fill="currentColor" style="max-width:40px"><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>
</th>
<th class="text-white bg-[#09090B] text-end text-sm sm:text-[1rem] font-semibold border-b-[#09090B]">Price</th>
</tr>
</thead>
<tbody>
{#each displayResults as item}
<tr on:click={() => {handleSave(false); goto("/stocks/"+item?.symbol)}} class="sm:hover:bg-[#245073] sm:hover:bg-opacity-[0.2] bg-[#09090B] border-b-[#09090B] odd:bg-[#27272A] cursor-pointer">
<td class="border-b-[#09090B] whitespace-nowrap">
<div class="flex flex-col items-start">
<span class="text-blue-400">{item?.symbol}</span>
<span class="text-white text-xs sm:hidden">{item?.name?.length > charNumber ? item?.name?.slice(0,charNumber) + "..." : item?.name}</span>
</div>
</td>
<td class="hidden sm:table-cell whitespace-nowrap text-[1rem] text-white border-b-[#09090B]">
{item?.name?.length > charNumber ? item?.name?.slice(0,charNumber) + "..." : item?.name}
</td>
<td class="text-white text-sm sm:text-[1rem] text-center border-b-[#09090B]">
{#if item?.symbol?.includes('.DE') || item?.symbol?.includes('.F')}
{item?.marketCap < 100 ? '< $100' : abbreviateNumber(item?.marketCap)}
{:else}
{item?.marketCap < 100 ? '< $100' : abbreviateNumber(item?.marketCap,true)}
{/if}
</td>
<td class="text-white text-center text-sm sm:text-[1rem] font-medium border-b-[#09090B]">
{#if item?.changesPercentage >=0}
<span class="text-[#10DB06]">+{item?.changesPercentage >= 1000 ? abbreviateNumber(item?.changesPercentage) : item?.changesPercentage?.toFixed(2)}%</span>
{:else}
<span class="text-[#FF2F1F]">{item?.changesPercentage <= -1000 ? abbreviateNumber(item?.changesPercentage) : item?.changesPercentage?.toFixed(2)}% </span>
{/if}
</td>
<td class="text-white text-sm sm:text-[1rem] text-end border-b-[#09090B]">
{item.price < 0.01 ? '< $0.01' :item.price?.toFixed(2)}
</td>
</tr>
{/each}
</tbody>
</table>
</div>
{/if}
<!--End Matching Preview-->
</div>
</section>
<div class="sm:hidden fixed z-40 bottom-8 sm:bottom-10 right-8 sm:right-16">
<label on:click={() => handleSave(true)} class="inline-flex items-center justify-center w-12 h-12 sm:w-full sm:h-10 font-medium bg-purple-600 ml-1 mr-0 sm:mr-2 rounded-full cursor-pointer">
<svg class="w-6 h-6 text-white inline-block" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24">
<path fill="white" d="M21 7v12q0 .825-.588 1.413T19 21H5q-.825 0-1.413-.588T3 19V5q0-.825.588-1.413T5 3h12l4 4Zm-9 11q1.25 0 2.125-.875T15 15q0-1.25-.875-2.125T12 12q-1.25 0-2.125.875T9 15q0 1.25.875 2.125T12 18Zm-6-8h9V6H6v4Z"/>
</svg>
</label>
</div>
<!--
<div class="tabs w-screen mb-5 ">
<label on:click={() => handleRuleTab('all')} class="tab mr-2 text-white font-medium transition duration-150 ease-out hover:ease-in rounded-md hover:bg-[#333333] {displayTab === 'all' ? 'bg-[#333333]' : ''}">
All
</label>
<label on:click={() => handleRuleTab('ta')} class="tab mr-2 text-white font-medium transition duration-150 ease-out hover:ease-in rounded-md hover:bg-[#333333] {displayTab === 'ta' ? 'bg-[#333333]' : ''}">
Technical Indicators
</label>
<label on:click={() => handleRuleTab('fund')} class="tab mr-2 text-white font-medium transition duration-150 ease-out hover:ease-in rounded-md hover:bg-[#333333] {displayTab === 'fund' ? 'bg-[#333333]' : ''}">
Fundamental Data
</label>
</div>
-->
{#if $screenWidth >= 640}
<!--Start Choose Rule Modal-->
<input type="checkbox" id="ruleModal" class="modal-toggle" />
<dialog id="ruleModal" class="modal modal-bottom sm:modal-middle ">
<label id="ruleModal" for="ruleModal" on:click={() => searchTerm = ''} class="cursor-pointer modal-backdrop bg-[#000] bg-opacity-[0.5]"></label>
<div class="modal-box w-full bg-[#262626] border border-slate-800 h-[800px] overflow-hidden ">
<div class="flex flex-col w-full mt-10 sm:mt-0">
<div class="text-white text-3xl font-semibold mb-5">
Choose Rules
</div>
<!--Start Search bar-->
<form class="w-11/12 h-8 mb-8" on:keydown={(e) => e?.key === 'Enter' ? e.preventDefault() : '' }>
<label for="search" class="mb-2 text-sm font-medium text-gray-200 sr-only">Search</label>
<div class="relative">
<div class="absolute inset-y-0 start-0 flex items-center ps-3 pointer-events-none">
<svg class="w-4 h-4 text-gray-200 dark:text-gray-400" aria-hidden="true" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 20 20">
<path stroke="currentColor" stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="m19 19-4-4m0-7A7 7 0 1 1 1 8a7 7 0 0 1 14 0Z"/>
</svg>
</div>
<input
autocomplete="off"
type="search"
id="search"
class="placeholder-gray-300 block w-full p-2 ps-10 text-sm text-gray-200 border border-gray-300 rounded-lg bg-[#404040] border border-blue-500"
placeholder="Search"
bind:value={searchTerm}
/>
</div>
</form>
<!-- End Search bar-->
<div class="text-white text-sm bg-[#262626] bg-opacity-[0.4] overflow-y-scroll scroller pt-3 rounded-lg max-h-[500px] sm:max-h-[420px] md:max-h-[540px] lg:max-h-[600px]">
<div class="text-white relative">
{#if searchTerm?.length !== 0 && filteredRows?.length === 0}
<span class="text-lg text-white font-medium flex justify-center items-center m-auto">
Nothing Found
</span>
{:else}
<table class="table table-sm table-compact">
<!-- head -->
<tbody>
{#each (searchTerm?.length !== 0 ? filteredRows : allRows) as row, index}
<tr on:click={() => changeRule(row?.rule)} class="hover:bg-[#333333] cursor-pointer">
<td class="border-b border-[#262626]">{index+1}</td>
<td class="text-start border-b border-[#262626]">
{#if ruleOfList.find((rule) => rule?.name === row?.rule)}
<svg class="flex-shrink-0 w-5 h-5 sm:w-6 sm:h-6 text-green-400 inline-block" fill="currentColor" viewBox="0 0 20 20" xmlns="http://www.w3.org/2000/svg"><path fill-rule="evenodd" d="M16.707 5.293a1 1 0 010 1.414l-8 8a1 1 0 01-1.414 0l-4-4a1 1 0 011.414-1.414L8 12.586l7.293-7.293a1 1 0 011.414 0z" clip-rule="evenodd"></path></svg>
{/if}
</td>
<td class="text-start border-b border-[#262626]">{row?.label}</td>
</tr>
{/each}
</tbody>
</table>
{/if}
</div>
</div>
</div>
</dialog>
<!--End Choose Rule Modal-->
{:else}
<div class="drawer drawer-end z-50">
<input id="ruleModal" type="checkbox" class="drawer-toggle" />
<div class="drawer-side overflow-x-hidden">
<div class="menu w-screen min-h-full bg-[#000] text-base-content overflow-hidden">
<div style="top: 0rem;" class="flex flex-row fixed sticky h-14 z-40 bg-[#000] justify-center items-center w-full border-b border-slate-900">
<label on:click={() => searchTerm = ''} for="ruleModal" class="cursor-pointer mr-auto ml-3">
<svg class="w-6 h-6 inline-block " xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path fill="#fff" d="M9.125 21.1L.7 12.7q-.15-.15-.213-.325T.425 12q0-.2.063-.375T.7 11.3l8.425-8.425q.35-.35.875-.35t.9.375q.375.375.375.875t-.375.875L3.55 12l7.35 7.35q.35.35.35.863t-.375.887q-.375.375-.875.375t-.875-.375Z"/></svg>
</label>
</div>
<div class="flex flex-row items-center justify-center w-full m-auto mb-8 mt-5">
<!--Start Search bar-->
<form class="w-11/12 h-8" on:keydown={(e) => e?.key === 'Enter' ? e.preventDefault() : '' }>
<label for="search" class="mb-2 text-sm font-medium text-gray-200 sr-only">Search</label>
<div class="relative">
<div class="absolute inset-y-0 start-0 flex items-center ps-3 pointer-events-none">
<svg class="w-4 h-4 text-gray-200 dark:text-gray-400" aria-hidden="true" xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 20 20">
<path stroke="currentColor" stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="m19 19-4-4m0-7A7 7 0 1 1 1 8a7 7 0 0 1 14 0Z"/>
</svg>
</div>
<input
autocomplete="off"
type="search"
id="search"
class="placeholder-gray-300 block w-full p-2 ps-10 text-sm text-gray-200 border border-gray-300 rounded-lg bg-[#404040] border border-blue-500"
placeholder="Search"
bind:value={searchTerm}
/>
</div>
</form>
<!-- End Search bar-->
</div>
{#if searchTerm?.length !== 0 && filteredRows?.length === 0}
<span class="text-lg text-white font-medium flex justify-center items-center mt-5">
Nothing Found
</span>
{:else}
<table class="table table-sm table-compact overflow-y-scroll text-white mb-10">
<!-- head -->
<tbody>
{#each (searchTerm?.length !== 0 ? filteredRows : allRows) as row, index}
<tr on:click={() => changeRule(row?.rule)} class="hover:bg-[#333333] border-b border-slate-800 cursor-pointer">
<td class="w-6">{index+1}</td>
<td class="w-3">
{#if ruleOfList.find((rule) => rule?.name === row?.rule)}
<svg class="flex-shrink-0 w-5 h-5 sm:w-6 sm:h-6 text-green-400 inline-block" fill="currentColor" viewBox="0 0 20 20" xmlns="http://www.w3.org/2000/svg"><path fill-rule="evenodd" d="M16.707 5.293a1 1 0 010 1.414l-8 8a1 1 0 01-1.414 0l-4-4a1 1 0 011.414-1.414L8 12.586l7.293-7.293a1 1 0 011.414 0z" clip-rule="evenodd"></path></svg>
{/if}
</td>
<td class="text-start">{row?.label}</td>
</tr>
{/each}
</tbody>
</table>
{/if}
</div>
</div>
</div>
{/if}
<!--Start Login Modal-->
{#if LoginPopup}
<LoginPopup form={form}/>
{/if}
<!--End Login Modal-->
<style>
.scroller {
scrollbar-width: thin;
}
</style>